diff --git a/.github/workflows/zip-maplibs.yml b/.github/workflows/zip-maplibs.yml new file mode 100644 index 00000000..d6146002 --- /dev/null +++ b/.github/workflows/zip-maplibs.yml @@ -0,0 +1,39 @@ +name: Create Release with Library Zips + +on: + release: + types: + - created + +jobs: + build: + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v2 + + # If PHP dependencies are not relevant, you can remove the steps related to Composer + - name: Install PHP dependencies + run: composer install --no-dev + + - name: Zip mapbox-gl library + run: cd web/libraries/mapbox-gl && zip -r mapbox-gl.zip . -x "*.git*" + + - name: Zip maplibre-gl library + run: cd web/libraries/maplibre-gl && zip -r maplibre-gl.zip . -x "*.git*" + + - name: Setup Node.js + uses: actions/setup-node@v2 + with: + node-version: '16' # Specify the Node.js version + # Release with the zipped libraries + - name: Create Release + uses: softprops/action-gh-release@v1 + if: startsWith(github.ref, 'refs/tags/') + with: + files: | + mapbox-gl.zip + maplibre-gl.zip + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + diff --git a/.gitignore b/.gitignore index 98963588..fd9bcde2 100755 --- a/.gitignore +++ b/.gitignore @@ -6,7 +6,8 @@ web/modules/contrib web/themes/contrib web/profiles/contrib web/libraries - +!web/libraries/mapbox-gl +!web/libraries/maplibre-gl # Ignore Drupal's file directory web/sites/*/files @@ -32,4 +33,4 @@ web/sites/*/settings.local.php /index.php /robots.txt /update.php -/web.config \ No newline at end of file +/web.config diff --git a/web/libraries/maplibre-gl/LICENSE.txt b/web/libraries/maplibre-gl/LICENSE.txt new file mode 100644 index 00000000..1e8acbb5 --- /dev/null +++ b/web/libraries/maplibre-gl/LICENSE.txt @@ -0,0 +1,116 @@ +Copyright (c) 2023, MapLibre contributors + +All rights reserved. + +Redistribution and use in source and binary forms, with or without modification, +are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. + * Neither the name of MapLibre GL JS nor the names of its contributors + may be used to endorse or promote products derived from this software + without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR +CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, +EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, +PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR +PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING +NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + +------------------------------------------------------------------------------- + +Contains code from mapbox-gl-js v1.13 and earlier + +Version v1.13 of mapbox-gl-js and earlier are licensed under a BSD-3-Clause license + +Copyright (c) 2020, Mapbox +Redistribution and use in source and binary forms, with or without modification, +are permitted provided that the following conditions are met: + +* Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. +* Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. +* Neither the name of Mapbox GL JS nor the names of its contributors + may be used to endorse or promote products derived from this software + without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR +CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, +EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, +PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR +PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING +NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, +EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + +------------------------------------------------------------------------------- + +Contains code from glfx.js + +Copyright (C) 2011 by Evan Wallace + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. + +-------------------------------------------------------------------------------- + +Contains a portion of d3-color https://github.com/d3/d3-color + +Copyright 2010-2016 Mike Bostock +All rights reserved. + +Redistribution and use in source and binary forms, with or without modification, +are permitted provided that the following conditions are met: + +* Redistributions of source code must retain the above copyright notice, this + list of conditions and the following disclaimer. + +* Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. + +* Neither the name of the author nor the names of contributors may be used to + endorse or promote products derived from this software without specific prior + written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR +ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES +(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; +LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON +ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/web/libraries/maplibre-gl/README.md b/web/libraries/maplibre-gl/README.md new file mode 100644 index 00000000..7022fd53 --- /dev/null +++ b/web/libraries/maplibre-gl/README.md @@ -0,0 +1,157 @@ +![MapLibre Logo](https://maplibre.org/img/maplibre-logo-big.svg) + +# MapLibre GL JS + +[![License](https://img.shields.io/badge/License-BSD_3--Clause-blue.svg?style=flat)](LICENSE.txt) [![Version](https://img.shields.io/npm/v/maplibre-gl?style=flat)](https://www.npmjs.com/package/maplibre-gl) [![CI](https://github.com/maplibre/maplibre-gl-js/actions/workflows/test-all.yml/badge.svg)](https://github.com/maplibre/maplibre-gl-js/actions/workflows/test-all.yml) [![PRs](https://img.shields.io/badge/PRs-welcome-brightgreen.svg?style=flat)](https://opensource.org/licenses/BSD-3-Clause) [![codecov](https://codecov.io/gh/maplibre/maplibre-gl-js/branch/main/graph/badge.svg)](https://codecov.io/gh/maplibre/maplibre-gl-js) + +**[MapLibre GL JS](https://maplibre.org/maplibre-gl-js/docs/API/)** is an open-source library for publishing maps on your websites or webview based apps. Fast displaying of maps is possible thanks to GPU-accelerated vector tile rendering. + +It originated as an open-source fork of [mapbox-gl-js](https://github.com/mapbox/mapbox-gl-js), before their switch to a non-OSS license in December 2020. The library's initial versions (1.x) were intended to be a drop-in replacement for the Mapbox’s OSS version (1.x) with additional functionality, but have evolved a lot since then. + +## Getting Started + +Include the JavaScript and CSS files in the `` of your HTML file. + +``` html + + +``` + +Include the following code in the `` of your HTML file. + +``` html +
+ +``` + +Enjoy the map! + +
+ +## Documentation + +Full documentation for this library [is available here](https://maplibre.org/maplibre-gl-js/docs/API/). + +Check out the features through [examples](https://maplibre.org/maplibre-gl-js/docs/examples/). + +| Showcases | | +| ---- | ---- | +|![Display a map](https://maplibre.org/maplibre-gl-js/docs/assets/examples/simple-map.png) |![Third party vector tile source](https://maplibre.org/maplibre-gl-js/docs/assets/examples/3d-terrain.png) | +|![Animate a series of images](https://maplibre.org/maplibre-gl-js/docs/assets/examples/animate-images.png) |![Create a heatmap layer](https://maplibre.org/maplibre-gl-js/docs/assets/examples/heatmap-layer.png) | +|![3D buildings](https://maplibre.org/maplibre-gl-js/docs/assets/examples/3d-buildings.png) |![Visualize population density](https://maplibre.org/maplibre-gl-js/docs/assets/examples/visualize-population-density.png) | + +
+ + +Want an example? Have a look at the official [MapLibre GL JS Documentation](https://maplibre.org/maplibre-gl-js/docs/examples/). + +Use MapLibre GL JS bindings for React (https://visgl.github.io/react-map-gl/docs/get-started/get-started#using-with-a-mapbox-gl-fork) and Angular (https://github.com/maplibre/ngx-maplibre-gl). Find more at [awesome-maplibre](https://github.com/maplibre/awesome-maplibre). + +
+ +## Contribution + +### Getting Involved + +Join the #maplibre slack channel at OSMUS: get an invite at https://slack.openstreetmap.us/ +Read the [CONTRIBUTING.md](CONTRIBUTING.md) guide in order to get familiar with how we do things around here. + +### Community Leadership + +You can find the official status of the backing community and steering committee in the [COMMUNITY.md](COMMUNITY.md) document. + +### Avoid Fragmentation + +If you depend on a free software alternative to `mapbox-gl-js`, please consider joining our effort! Anyone with a stake in a healthy community-led fork is welcome to help us figure out our next steps. We welcome contributors and leaders! MapLibre GL JS already represents the combined efforts of a few early fork efforts, and we all benefit from "one project" rather than "our way". If you know of other forks, please reach out to them and direct them here. + + +> **MapLibre GL JS** is developed following [Semantic Versioning (2.0.0)](https://semver.org/spec/v2.0.0.html). + +### Bounties + +We offer Bounties for some tasks in the MapLibre GL JS repo. Read more about the Bounties in our step-by-step guide: + +https://maplibre.org/roadmap/step-by-step-bounties-guide/ + +And find all currently published Bounties in MapLibre GL JS [here](https://github.com/maplibre/maplibre-gl-js/issues?q=is%3Aissue+is%3Aopen+label%3A%22%F0%9F%92%B0+bounty+L%22%2C%22%F0%9F%92%B0+bounty+S%22%2C%22%F0%9F%92%B0+bounty+M%22%2C%22%F0%9F%92%B0+bounty+XL%22%2C%22%F0%9F%92%B0+bounty+XXL%22+). + +
+ +## Sponsors + +We thank everyone who supported us financially in the past and special thanks to the people and organizations who support us with recurring donations! + +Read more about the MapLibre Sponsorship Program at [https://maplibre.org/sponsors/](https://maplibre.org/sponsors/). + +Platinum: + +Logo AWS + +Gold: + +Logo Meta + +Silver: + +Logo MIERUNE + +Logo komoot + +Logo JawgMaps + +Logo Radar + +Backers and Supporters: + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +## Thank you Mapbox 🙏🏽 + +We'd like to acknowledge the amazing work Mapbox has contributed to open source. The open source community is sad to part ways with them, but we simultaneously feel grateful for everything they already contributed. `mapbox-gl-js` 1.x is an open source achievement that now lives on as `maplibre-gl`. We're proud to develop on the shoulders of giants, thank you Mapbox 🙇🏽‍♀️. + +Please keep in mind: Unauthorized backports are the biggest threat to the MapLibre project. It is unacceptable to backport code from mapbox-gl-js, which is not covered by the former BSD-3 license. If you are unsure about this issue, [please ask](https://github.com/maplibre/maplibre-gl-js/discussions)! + +
+ +## License + +**MapLibre GL JS** is licensed under the [3-Clause BSD license](./LICENSE.txt). diff --git a/web/libraries/maplibre-gl/build/banner.ts b/web/libraries/maplibre-gl/build/banner.ts new file mode 100644 index 00000000..02cccd0c --- /dev/null +++ b/web/libraries/maplibre-gl/build/banner.ts @@ -0,0 +1,3 @@ +import packageJSON from '../package.json' assert {type: 'json'}; + +export default `/* MapLibre GL JS is licensed under the 3-Clause BSD License. Full text of license: https://github.com/maplibre/maplibre-gl-js/blob/v${packageJSON.version}/LICENSE.txt */`; diff --git a/web/libraries/maplibre-gl/build/bump-version-changelog.js b/web/libraries/maplibre-gl/build/bump-version-changelog.js new file mode 100755 index 00000000..8263b585 --- /dev/null +++ b/web/libraries/maplibre-gl/build/bump-version-changelog.js @@ -0,0 +1,26 @@ +#!/usr/bin/env node + +/** + * This script updates the changelog.md file with the version given in the arguments + * It replaces ## main with ## + * Removes _...Add new stuff here..._ + * And adds on top a ## main with add stuff here. + */ + +import * as fs from 'fs'; + +const changelogPath = 'CHANGELOG.md'; +let changelog = fs.readFileSync(changelogPath, 'utf8'); +changelog = changelog.replace('## main', `## ${process.argv[2]}`); +changelog = changelog.replaceAll('- _...Add new stuff here..._\n', ''); +changelog = `## main + +### ✨ Features and improvements +- _...Add new stuff here..._ + +### 🐞 Bug fixes +- _...Add new stuff here..._ + +` + changelog; + +fs.writeFileSync(changelogPath, changelog, 'utf8'); \ No newline at end of file diff --git a/web/libraries/maplibre-gl/build/check-bundle-size.js b/web/libraries/maplibre-gl/build/check-bundle-size.js new file mode 100755 index 00000000..10a8edfd --- /dev/null +++ b/web/libraries/maplibre-gl/build/check-bundle-size.js @@ -0,0 +1,77 @@ +#!/usr/bin/env node + +import fs from "fs"; +import zlib from "zlib"; +import prettyBytes from "pretty-bytes"; +const beforeSourcemap = JSON.parse(fs.readFileSync('./before.json').toString()); +const afterSourcemap = JSON.parse(fs.readFileSync('./after.json').toString()); + +function fileSize(file) { + const {size} = fs.statSync(file); + const gzipped = zlib.gzipSync(fs.readFileSync(file)).length + return { + size, + gzipped + }; +} + +const beforejs = fileSize('./before/maplibre-gl.js'); +const beforecss = fileSize('./before/maplibre-gl.css'); +const afterjs = fileSize('./after/maplibre-gl.js'); +const aftercss = fileSize('./after/maplibre-gl.css'); + +console.log('Bundle size report:\n'); +console.log(`**Size Change:** ${prettyBytes(afterjs.gzipped + aftercss.gzipped - (beforejs.gzipped + beforecss.gzipped), { signed: true })}`); +console.log(`**Total Size Before:** ${prettyBytes(beforejs.gzipped + beforecss.gzipped)}`); +console.log(`**Total Size After:** ${prettyBytes(afterjs.gzipped + aftercss.gzipped)}`); +console.log(` +| Output file | Before | After | Change | +| :--- | :---: | :---: | :---: | +| maplibre-gl.js | ${prettyBytes(beforejs.gzipped)} | ${prettyBytes(afterjs.gzipped)} | ${prettyBytes(afterjs.gzipped - beforejs.gzipped, { signed: true })} | +| maplibre-gl.css | ${prettyBytes(beforecss.gzipped)} | ${prettyBytes(aftercss.gzipped)} | ${prettyBytes(aftercss.gzipped - beforecss.gzipped, { signed: true })} |`); + +const before = {}; +beforeSourcemap.results.forEach(result => { + Object.keys(result.files).forEach(filename => { + const {size} = result.files[filename]; + before[filename] = size; + }); +}); + +const after = {}; +afterSourcemap.results.forEach(result => { + Object.keys(result.files).forEach(filename => { + const {size} = result.files[filename]; + after[filename] = size; + }); +}); + +const diffs = []; +Object.keys(Object.assign({}, before, after)).forEach(filename => { + const beforeSize = before[filename] || 0; + const afterSize = after[filename] || 0; + if (Math.abs(afterSize - beforeSize) > 0) { + diffs.push([ + afterSize - beforeSize, // for sorting + filename.replace(/^[\./]+/, ''), // omit ../ + prettyBytes(beforeSize), + prettyBytes(afterSize), + prettyBytes(afterSize - beforeSize, { signed: true }) + ]); + } +}); + +diffs.sort((a, b) => b[0] - a[0]); + +console.log(` +
ℹ️ View Details`); +if (diffs.length) { + console.log(` +| Source file | Before | After | Change | +| :--- | :---: | :---: | :---: | +${diffs.map(diff => '| ' + diff.slice(1).join(' | ') + ' |').join('\n')} +`); +} else { + console.log('No major changes'); +} +console.log(`
`); diff --git a/web/libraries/maplibre-gl/build/generate-dist-package.js b/web/libraries/maplibre-gl/build/generate-dist-package.js new file mode 100644 index 00000000..7a4fe1b5 --- /dev/null +++ b/web/libraries/maplibre-gl/build/generate-dist-package.js @@ -0,0 +1,26 @@ +'use strict'; +/// js files in this project are esm. +/// This package.json ensures that node imports from outside see them as such +// https://nodejs.org/api/packages.html#type + +import { writeFile, mkdir, copyFile } from "node:fs/promises" + +async function ensureDist() { + const dist = new URL("../dist/", import.meta.url); + await mkdir(dist, { recursive: true }) + return dist +} + +await writeFile( + new URL("package.json", await ensureDist()), + JSON.stringify({ + name: "maplibre-gl", + type: "commonjs", + deprecated: "Please install maplibre-gl from parent directory instead", + }) +) + +await copyFile( + "./LICENSE.txt", + new URL("LICENSE.txt", await ensureDist()) +) \ No newline at end of file diff --git a/web/libraries/maplibre-gl/build/generate-doc-images.ts b/web/libraries/maplibre-gl/build/generate-doc-images.ts new file mode 100644 index 00000000..50ce85bb --- /dev/null +++ b/web/libraries/maplibre-gl/build/generate-doc-images.ts @@ -0,0 +1,73 @@ +import path from 'path'; +import fs from 'fs'; +import puppeteer from 'puppeteer'; +import packageJson from '../package.json' assert { type: 'json' }; + +const exampleName = process.argv[2]; +const examplePath = path.resolve('test', 'examples'); + +const browser = await puppeteer.launch({headless: exampleName === 'all'}); + +const page = await browser.newPage(); +// set viewport and double deviceScaleFactor to get a closer shot of the map +await page.setViewport({ + width: 600, + height: 250, + deviceScaleFactor: 2 +}); + +async function createImage(exampleName) { + // get the example contents + const html = fs.readFileSync(path.resolve(examplePath, `${exampleName}.html`), 'utf-8'); + + await page.setContent(html.replaceAll('../../dist', `https://unpkg.com/maplibre-gl@${packageJson.version}/dist`)); + + // Wait for map to load, then wait two more seconds for images, etc. to load. + await page + .waitForFunction('map.loaded()') + .then(async () => { + // Wait for 5 seconds on 3d model examples, since this takes longer to load. + const waitTime = exampleName.includes('3d-model') ? 5000 : 1500; + await new Promise((resolve) => { + console.log(`waiting for ${waitTime} ms`); + setTimeout(resolve, waitTime); + }); + }) + // map.loaded() does not evaluate to true within 3 seconds, it's probably an animated example. + // In this case we take the screenshot immediately. + .catch(() => { + console.log(`Timed out waiting for map load on ${exampleName}.`); + }); + + await page + .screenshot({ + path: `./docs/assets/examples/${exampleName}.png`, + type: 'png', + clip: { + x: 0, + y: 0, + width: 600, + height: 250 + } + }) + .then(() => console.log(`Created ./docs/assets/examples/${exampleName}.png`)) + .catch((err) => { + console.log(err); + }); +} + +if (exampleName === 'all') { + const allFiles = fs.readdirSync(examplePath).filter(f => f.endsWith('html')); + console.log(`Generating ${allFiles.length} images.`); + for (const file of allFiles) { + await createImage(file); + } +} else if (exampleName) { + await createImage(exampleName); +} else { + throw new Error( + '\n Usage: npm run generate-images \nExample: npm run generate-images 3d-buildings' + ); +} + +await browser.close(); diff --git a/web/libraries/maplibre-gl/build/generate-docs.ts b/web/libraries/maplibre-gl/build/generate-docs.ts new file mode 100644 index 00000000..8b71d8e5 --- /dev/null +++ b/web/libraries/maplibre-gl/build/generate-docs.ts @@ -0,0 +1,100 @@ +import fs from 'fs'; +import path from 'path'; +import typedocConfig from '../typedoc.json' assert {type: 'json'}; +import packageJson from '../package.json' assert {type: 'json'}; + +type HtmlDoc = { + title: string; + description: string; + mdFileName: string; +} + +function generateAPIIntroMarkdown(lines: string[]): string { + let intro = `# Intro + +This file is intended as a reference for the important and public classes of this API. +We recommend looking at the [examples](../examples/index.md) as they will help you the most to start with MapLibre. + +Most of the classes wirtten here have an "Options" object for initialization, it is recommended to check which options exist. + +`; + intro += lines.map(l => l.replace('../', './')).join('\n'); + return intro; +} + +function generateMarkdownForExample(title: string, description: string, file: string, htmlContent: string): string { + return ` +# ${title} + +${description} + + + +\`\`\`html +${htmlContent} +\`\`\` +`; +} + +function generateMarkdownIndexFileOfAllExamples(indexArray: HtmlDoc[]): string { + let indexMarkdown = '# Overview \n\n'; + for (const indexArrayItem of indexArray) { + indexMarkdown += ` +## [${indexArrayItem.title}](./${indexArrayItem.mdFileName}) + +![${indexArrayItem.description}](../assets/examples/${indexArrayItem.mdFileName!.replace('.md', '.png')}) + +${indexArrayItem.description} +`; + } + return indexMarkdown; +} + +if (!fs.existsSync(typedocConfig.out)) { + throw new Error('Please run typedoc generation first!'); +} + +fs.rmSync(path.join(typedocConfig.out, 'README.md')); +fs.rmSync(path.join(typedocConfig.out, 'modules.md')); +// Intro file for the API +const modulesFolder = path.join(typedocConfig.out, 'modules'); +const content = fs.readFileSync(path.join(modulesFolder, `${typedocConfig.internalModule}.md`), 'utf-8'); +let lines = content.split('\n'); +const classesLineIndex = lines.indexOf(lines.find(l => l.endsWith('Classes')) as string); +lines = lines.splice(2, classesLineIndex - 2); +const contentString = generateAPIIntroMarkdown(lines); +fs.writeFileSync(path.join(typedocConfig.out, 'README.md'), contentString); + +// Examples manupilation +const examplesDocsFolder = path.join('docs', 'examples'); +if (fs.existsSync(examplesDocsFolder)) { + fs.rmSync(examplesDocsFolder, {recursive: true, force: true}); +} +fs.mkdirSync(examplesDocsFolder); +const examplesFolder = path.join('test', 'examples'); +const files = fs.readdirSync(examplesFolder).filter(f => f.endsWith('html')); +const maplibreUnpgk = `https://unpkg.com/maplibre-gl@${packageJson.version}/`; +const indexArray = [] as HtmlDoc[]; +for (const file of files) { + const htmlFile = path.join(examplesFolder, file); + let htmlContent = fs.readFileSync(htmlFile, 'utf-8'); + htmlContent = htmlContent.replace(/\.\.\/\.\.\//g, maplibreUnpgk); + htmlContent = htmlContent.replace(/-dev.js/g, '.js'); + const htmlContentLines = htmlContent.split('\n'); + const title = htmlContentLines.find(l => l.includes('', '').replace('', '').trim(); + const description = htmlContentLines.find(l => l.includes('og:description'))?.replace(/.*content=\"(.*)\".*/, '$1'); + fs.writeFileSync(path.join(examplesDocsFolder, file), htmlContent); + const mdFileName = file.replace('.html', '.md'); + indexArray.push({ + title: title!, + description: description!, + mdFileName + }); + const exampleMarkdown = generateMarkdownForExample(title, description, file, htmlContent); + fs.writeFileSync(path.join(examplesDocsFolder, mdFileName), exampleMarkdown); +} + +const indexMarkdown = generateMarkdownIndexFileOfAllExamples(indexArray); +fs.writeFileSync(path.join(examplesDocsFolder, 'index.md'), indexMarkdown); + +console.log('Docs generation completed, to see it in action run\n docker run --rm -it -p 8000:8000 -v ${PWD}:/docs squidfunk/mkdocs-material'); diff --git a/web/libraries/maplibre-gl/build/generate-shaders.ts b/web/libraries/maplibre-gl/build/generate-shaders.ts new file mode 100644 index 00000000..5f3e2409 --- /dev/null +++ b/web/libraries/maplibre-gl/build/generate-shaders.ts @@ -0,0 +1,68 @@ +import fs from 'fs'; +import {globSync} from 'glob'; +import path from 'path'; + +console.log('Generating shaders'); + +/** + * This script is intended to copy the glsl file to the compilation output folder, + * change their extension to .js and export the shaders as strings in javascript. + * It will also minify them a bit if needed and change the extension to .js + * After that it will create a combined typescript definition file and manipulate it a bit + * It will also create a simple package.json file to allow importing this package in webpack + */ + +const vertex = globSync('./src/shaders/*.vertex.glsl'); +for (const file of vertex) { + const code = fs.readFileSync(file, 'utf8'); + const content = glslToTs(code, 'vertex'); + const fileName = path.join('.', 'src', 'shaders', `${file.split(path.sep).splice(-1)}.g.ts`); + fs.writeFileSync(fileName, content); +} + +console.log(`Finished converting ${vertex.length} vertex shaders`); + +const fragment = globSync('./src/shaders/*.fragment.glsl'); +for (const file of fragment) { + const code = fs.readFileSync(file, 'utf8'); + const content = glslToTs(code, 'fragment'); + const fileName = path.join('.', 'src', 'shaders', `${file.split(path.sep).splice(-1)}.g.ts`); + fs.writeFileSync(fileName, content); +} + +console.log(`Finished converting ${fragment.length} fragment shaders`); + +function glslToTs(code: string, type: 'fragment'|'vertex'): string { + code = code + .trim(); // strip whitespace at the start/end + + // WebGL1 Compat -- Start + + if (type === 'fragment') { + code = code + .replace(/\bin\s/g, 'varying ') // For fragment shaders, replace "in " with "varying " + .replace('out highp vec4 fragColor;', ''); + } + + if (type === 'vertex') { + code = code + .replace(/\bin\s/g, 'attribute ') // For vertex shaders, replace "in " with "attribute " + .replace(/\bout\s/g, 'varying '); // For vertex shaders, replace "out " with "varying " + } + + code = code + .replace(/fragColor/g, 'gl_FragColor') + .replace(/texture\(/g, 'texture2D('); + + // WebGL1 Compat -- End + + code = code + .replace(/\s*\/\/[^\n]*\n/g, '\n') // strip double-slash comments + .replace(/\n+/g, '\n') // collapse multi line breaks + .replace(/\n\s+/g, '\n') // strip indentation + .replace(/\s?([+-\/*=,])\s?/g, '$1') // strip whitespace around operators + .replace(/([;\(\),\{\}])\n(?=[^#])/g, '$1'); // strip more line breaks + + return `// This file is generated. Edit build/generate-shaders.ts, then run \`npm run codegen\`. +export default ${JSON.stringify(code).replaceAll('"', '\'')};\n`; +} diff --git a/web/libraries/maplibre-gl/build/generate-struct-arrays.ts b/web/libraries/maplibre-gl/build/generate-struct-arrays.ts new file mode 100644 index 00000000..06562fa7 --- /dev/null +++ b/web/libraries/maplibre-gl/build/generate-struct-arrays.ts @@ -0,0 +1,439 @@ +/* + * Generates the following: + * - data/array_types.js, which consists of: + * - StructArrayLayout_* subclasses, one for each underlying memory layout we need + * - Named exports mapping each conceptual array type (e.g., CircleLayoutArray) to its corresponding StructArrayLayout class + * - Particular, named StructArray subclasses, when fancy struct accessors are needed (e.g. CollisionBoxArray) + */ + +'use strict'; + +import * as fs from 'fs'; +import * as util from '../src/util/util'; +import {createLayout, viewTypes} from '../src/util/struct_array'; +import type {ViewType, StructArrayLayout} from '../src/util/struct_array'; + +import posAttributes from '../src/data/pos_attributes'; +import pos3dAttributes from '../src/data/pos3d_attributes'; +import rasterBoundsAttributes from '../src/data/raster_bounds_attributes'; +import circleAttributes from '../src/data/bucket/circle_attributes'; +import fillAttributes from '../src/data/bucket/fill_attributes'; +import fillExtrusionAttributes from '../src/data/bucket/fill_extrusion_attributes'; +import {lineLayoutAttributes} from '../src/data/bucket/line_attributes'; +import {lineLayoutAttributesExt} from '../src/data/bucket/line_attributes_ext'; +import {patternAttributes} from '../src/data/bucket/pattern_attributes'; +// symbol layer specific arrays +import { + symbolLayoutAttributes, + dynamicLayoutAttributes, + placementOpacityAttributes, + collisionBox, + collisionBoxLayout, + collisionCircleLayout, + collisionVertexAttributes, + quadTriangle, + placement, + symbolInstance, + glyphOffset, + lineVertex, + textAnchorOffset +} from '../src/data/bucket/symbol_attributes'; + +const typeAbbreviations = { + 'Int8': 'b', + 'Uint8': 'ub', + 'Int16': 'i', + 'Uint16': 'ui', + 'Int32': 'l', + 'Uint32': 'ul', + 'Float32': 'f' +}; + +const arraysWithStructAccessors = []; +const arrayTypeEntries = new Set(); +const layoutCache = {}; + +function normalizeMembers(members, usedTypes) { + return members.map((member) => { + if (usedTypes && !usedTypes.has(member.type)) { + usedTypes.add(member.type); + } + + return util.extend(member, { + size: sizeOf(member.type), + view: member.type.toLowerCase() + }); + }); +} + +// - If necessary, write the StructArrayLayout_* class for the given layout +// - If `includeStructAccessors`, write the fancy subclass +// - Add an entry for `name` in the array type registry +function createStructArrayType(name: string, layout: StructArrayLayout, includeStructAccessors: boolean = false) { + const hasAnchorPoint = layout.members.some(m => m.name === 'anchorPointX'); + + // create the underlying StructArrayLayout class exists + const layoutClass = createStructArrayLayoutType(layout); + const arrayClass = `${camelize(name)}Array`; + + if (includeStructAccessors) { + const usedTypes = new Set(['Uint8']); + const members = normalizeMembers(layout.members, usedTypes); + arraysWithStructAccessors.push({ + arrayClass, + members, + size: layout.size, + usedTypes, + hasAnchorPoint, + layoutClass, + includeStructAccessors + }); + } else { + arrayTypeEntries.add(`export class ${arrayClass} extends ${layoutClass} {}`); + } +} + +function createStructArrayLayoutType({members, size, alignment}) { + const usedTypes = new Set(['Uint8']); + members = normalizeMembers(members, usedTypes); + + // combine consecutive 'members' with same underlying type, summing their + // component counts + if (!alignment || alignment === 1) members = members.reduce((memo, member) => { + if (memo.length > 0 && memo[memo.length - 1].type === member.type) { + const last = memo[memo.length - 1]; + return memo.slice(0, -1).concat(util.extend({}, last, { + components: last.components + member.components, + })); + } + return memo.concat(member); + }, []); + + const key = `${members.map(m => `${m.components}${typeAbbreviations[m.type]}`).join('')}${size}`; + const className = `StructArrayLayout${key}`; + // Layout alignment to 4 bytes boundaries can be an issue on some set of graphics cards. Particularly AMD. + if (size % 4 !== 0) { console.warn(`Warning: The layout ${className} is not aligned to 4-bytes boundaries.`); } + if (!layoutCache[key]) { + layoutCache[key] = { + className, + members, + size, + usedTypes + }; + } + + return className; +} + +function sizeOf(type: ViewType): number { + return viewTypes[type].BYTES_PER_ELEMENT; +} + +function camelize (str) { + return str.replace(/(?:^|[-_])(.)/g, (_, x) => { + return /^[0-9]$/.test(x) ? _ : x.toUpperCase(); + }); +} + +createStructArrayType('pos', posAttributes); +createStructArrayType('pos3d', pos3dAttributes); +createStructArrayType('raster_bounds', rasterBoundsAttributes); + +// layout vertex arrays +const layoutAttributes = { + circle: circleAttributes, + fill: fillAttributes, + 'fill-extrusion': fillExtrusionAttributes, + heatmap: circleAttributes, + line: lineLayoutAttributes, + lineExt: lineLayoutAttributesExt, + pattern: patternAttributes +}; +for (const name in layoutAttributes) { + createStructArrayType(`${name.replace(/-/g, '_')}_layout`, layoutAttributes[name]); +} + +createStructArrayType('symbol_layout', symbolLayoutAttributes); +createStructArrayType('symbol_dynamic_layout', dynamicLayoutAttributes); +createStructArrayType('symbol_opacity', placementOpacityAttributes); +createStructArrayType('collision_box', collisionBox, true); +createStructArrayType('collision_box_layout', collisionBoxLayout); +createStructArrayType('collision_circle_layout', collisionCircleLayout); +createStructArrayType('collision_vertex', collisionVertexAttributes); +createStructArrayType('quad_triangle', quadTriangle); +createStructArrayType('placed_symbol', placement, true); +createStructArrayType('symbol_instance', symbolInstance, true); +createStructArrayType('glyph_offset', glyphOffset, true); +createStructArrayType('symbol_line_vertex', lineVertex, true); +createStructArrayType('text_anchor_offset', textAnchorOffset, true); + +// feature index array +createStructArrayType('feature_index', createLayout([ + // the index of the feature in the original vectortile + {type: 'Uint32', name: 'featureIndex'}, + // the source layer the feature appears in + {type: 'Uint16', name: 'sourceLayerIndex'}, + // the bucket the feature appears in + {type: 'Uint16', name: 'bucketIndex'} +]), true); + +// triangle index array +createStructArrayType('triangle_index', createLayout([ + {type: 'Uint16', name: 'vertices', components: 3} +])); + +// line index array +createStructArrayType('line_index', createLayout([ + {type: 'Uint16', name: 'vertices', components: 2} +])); + +// line strip index array +createStructArrayType('line_strip_index', createLayout([ + {type: 'Uint16', name: 'vertices', components: 1} +])); + +// paint vertex arrays + +// used by SourceBinder for float properties +createStructArrayLayoutType(createLayout([{ + name: 'dummy name (unused for StructArrayLayout)', + type: 'Float32', + components: 1 +}], 4)); + +// used by SourceBinder for color properties and CompositeBinder for float properties +createStructArrayLayoutType(createLayout([{ + name: 'dummy name (unused for StructArrayLayout)', + type: 'Float32', + components: 2 +}], 4)); + +// used by CompositeBinder for color properties +createStructArrayLayoutType(createLayout([{ + name: 'dummy name (unused for StructArrayLayout)', + type: 'Float32', + components: 4 +}], 4)); + +const layouts = Object.keys(layoutCache).map(k => layoutCache[k]); + +function emitStructArrayLayout(locals) { + const output = []; + const { + className, + members, + size, + usedTypes + } = locals; + const structArrayLayoutClass = className; + + output.push( + `/** + * @internal + * Implementation of the StructArray layout:`); + + for (const member of members) { + output.push( + ` * [${member.offset}]: ${member.type}[${member.components}]`); + } + + output.push( + ` * + */ +class ${structArrayLayoutClass} extends StructArray {`); + + for (const type of usedTypes) { + output.push( + ` ${type.toLowerCase()}: ${type}Array;`); + } + + output.push(` + _refreshViews() {`); + + for (const type of usedTypes) { + output.push( + ` this.${type.toLowerCase()} = new ${type}Array(this.arrayBuffer);`); + } + + output.push( + ' }'); + + // prep for emplaceBack: collect type sizes and count the number of arguments + // we'll need + const bytesPerElement = size; + const usedTypeSizes = []; + const argNames = []; + const argNamesTyped = []; + + for (const member of members) { + if (usedTypeSizes.indexOf(member.size) < 0) { + usedTypeSizes.push(member.size); + } + for (let c = 0; c < member.components; c++) { + // arguments v0, v1, v2, ... are, in order, the components of + // member 0, then the components of member 1, etc. + const name = `v${argNames.length}`; + argNames.push(name); + argNamesTyped.push(`${name}: number`); + } + } + + output.push( + ` + public emplaceBack(${argNamesTyped.join(', ')}) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, ${argNames.join(', ')}); + } + + public emplace(i: number, ${argNamesTyped.join(', ')}) {`); + + for (const size of usedTypeSizes) { + output.push( + ` const o${size.toFixed(0)} = i * ${(bytesPerElement / size).toFixed(0)};`); + } + + let argIndex = 0; + for (const member of members) { + for (let c = 0; c < member.components; c++) { + // The index for `member` component `c` into the appropriate type array is: + // this.{TYPE}[o{SIZE} + MEMBER_OFFSET + {c}] = v{X} + // where MEMBER_OFFSET = ROUND(member.offset / size) is the per-element + // offset of this member into the array + const index = `o${member.size.toFixed(0)} + ${(member.offset / member.size + c).toFixed(0)}`; + + output.push( + ` this.${member.view}[${index}] = v${argIndex++};`); + } + } + + output.push( + ` return i; + } +} + +${structArrayLayoutClass}.prototype.bytesPerElement = ${size}; +register('${structArrayLayoutClass}', ${structArrayLayoutClass}); +`); + + return output.join('\n'); +} + +function emitStructArray(locals) { + const output = []; + const { + arrayClass, + members, + size, + hasAnchorPoint, + layoutClass, + includeStructAccessors + } = locals; + + const structTypeClass = arrayClass.replace('Array', 'Struct'); + const structArrayClass = arrayClass; + const structArrayLayoutClass = layoutClass; + + // collect components + const components = []; + for (const member of members) { + for (let c = 0; c < member.components; c++) { + let name = member.name; + if (member.components > 1) { + name += c; + } + components.push({name, member, component: c}); + } + } + + // exceptions for which we generate accessors on the array rather than a separate struct for performance + const useComponentGetters = structArrayClass === 'GlyphOffsetArray' || structArrayClass === 'SymbolLineVertexArray'; + + if (includeStructAccessors && !useComponentGetters) { + output.push( + `/** @internal */ +class ${structTypeClass} extends Struct { + _structArray: ${structArrayClass};`); + + for (const {name, member, component} of components) { + const elementOffset = `this._pos${member.size.toFixed(0)}`; + const componentOffset = (member.offset / member.size + component).toFixed(0); + const index = `${elementOffset} + ${componentOffset}`; + const componentAccess = `this._structArray.${member.view}[${index}]`; + + output.push( + ` get ${name}() { return ${componentAccess}; }`); + + // generate setters for properties that are updated during runtime symbol placement; others are read-only + if (name === 'crossTileID' || name === 'placedOrientation' || name === 'hidden') { + output.push( + ` set ${name}(x: number) { ${componentAccess} = x; }`); + } + } + + // Special case used for the CollisionBoxArray type + if (hasAnchorPoint) { + output.push( + ' get anchorPoint() { return new Point(this.anchorPointX, this.anchorPointY); }'); + } + + output.push( + `} + +${structTypeClass}.prototype.size = ${size}; + +export type ${structTypeClass.replace('Struct', '')} = ${structTypeClass}; +`); + } // end 'if (includeStructAccessors)' + + output.push( + `/** @internal */ +export class ${structArrayClass} extends ${structArrayLayoutClass} {`); + + if (useComponentGetters) { + for (const member of members) { + for (let c = 0; c < member.components; c++) { + if (!includeStructAccessors) continue; + let name = `get${member.name}`; + if (member.components > 1) { + name += c; + } + const componentOffset = (member.offset / member.size + c).toFixed(0); + const componentStride = size / member.size; + output.push( + ` ${name}(index: number) { return this.${member.view}[index * ${componentStride} + ${componentOffset}]; }`); + } + } + } else if (includeStructAccessors) { // get(i) + output.push( + ` /** + * Return the ${structTypeClass} at the given location in the array. + * @param index The index of the element. + */ + get(index: number): ${structTypeClass} { + return new ${structTypeClass}(this, index); + }`); + } + output.push( + `} + +register('${structArrayClass}', ${structArrayClass}); +`); + + return output.join('\n'); +} + +fs.writeFileSync('src/data/array_types.g.ts', + `// This file is generated. Edit build/generate-struct-arrays.ts, then run \`npm run codegen\`. + +import {Struct, StructArray} from '../util/struct_array'; +import {register} from '../util/web_worker_transfer'; +import Point from '@mapbox/point-geometry'; + +${layouts.map(emitStructArrayLayout).join('\n')} +${arraysWithStructAccessors.map(emitStructArray).join('\n')} +${[...arrayTypeEntries].join('\n')} +export { + ${layouts.map(layout => layout.className).join(',\n ')} +}; +`); diff --git a/web/libraries/maplibre-gl/build/generate-style-code.ts b/web/libraries/maplibre-gl/build/generate-style-code.ts new file mode 100644 index 00000000..a78fb416 --- /dev/null +++ b/web/libraries/maplibre-gl/build/generate-style-code.ts @@ -0,0 +1,284 @@ +'use strict'; + +import * as fs from 'fs'; + +import {v8} from '@maplibre/maplibre-gl-style-spec'; + +function camelCase(str: string): string { + return str.replace(/-(.)/g, (_, x) => { + return x.toUpperCase(); + }); +} + +function pascalCase(str: string): string { + const almostCamelized = camelCase(str); + return almostCamelized[0].toUpperCase() + almostCamelized.slice(1); +} + +function nativeType(property) { + switch (property.type) { + case 'boolean': + return 'boolean'; + case 'number': + return 'number'; + case 'string': + return 'string'; + case 'enum': + return Object.keys(property.values).map(v => JSON.stringify(v)).join(' | '); + case 'color': + return 'Color'; + case 'padding': + return 'Padding'; + case 'variableAnchorOffsetCollection': + return 'VariableAnchorOffsetCollection'; + case 'sprite': + return 'Sprite'; + case 'formatted': + return 'Formatted'; + case 'resolvedImage': + return 'ResolvedImage'; + case 'array': + if (property.length) { + return `[${new Array(property.length).fill(nativeType({type: property.value})).join(', ')}]`; + } else { + return `Array<${nativeType({type: property.value, values: property.values})}>`; + } + default: throw new Error(`unknown type "${property.type}" for "${property.name}"`); + } +} + +function possiblyEvaluatedType(property) { + const propType = nativeType(property); + + switch (property['property-type']) { + case 'color-ramp': + return 'ColorRampProperty'; + case 'cross-faded': + return `CrossFaded<${propType}>`; + case 'cross-faded-data-driven': + return `PossiblyEvaluatedPropertyValue>`; + case 'data-driven': + return `PossiblyEvaluatedPropertyValue<${propType}>`; + } + + return propType; +} + +function propertyType(property) { + switch (property['property-type']) { + case 'data-driven': + return `DataDrivenProperty<${nativeType(property)}>`; + case 'cross-faded': + return `CrossFadedProperty<${nativeType(property)}>`; + case 'cross-faded-data-driven': + return `CrossFadedDataDrivenProperty<${nativeType(property)}>`; + case 'color-ramp': + return 'ColorRampProperty'; + case 'data-constant': + case 'constant': + return `DataConstantProperty<${nativeType(property)}>`; + default: + throw new Error(`unknown property-type "${property['property-type']}" for ${property.name}`); + } +} + +function runtimeType(property) { + switch (property.type) { + case 'boolean': + return 'BooleanType'; + case 'number': + return 'NumberType'; + case 'string': + case 'enum': + return 'StringType'; + case 'color': + return 'ColorType'; + case 'padding': + return 'PaddingType'; + case 'variableAnchorOffsetCollection': + return 'VariableAnchorOffsetCollectionType'; + case 'sprite': + return 'SpriteType'; + case 'formatted': + return 'FormattedType'; + case 'Image': + return 'ImageType'; + case 'array': + if (property.length) { + return `array(${runtimeType({type: property.value})}, ${property.length})`; + } else { + return `array(${runtimeType({type: property.value})})`; + } + default: throw new Error(`unknown type "${property.type}" for "${property.name}"`); + } +} + +function overrides(property) { + return `{ runtimeType: ${runtimeType(property)}, getOverride: (o) => o.${camelCase(property.name)}, hasOverride: (o) => !!o.${camelCase(property.name)} }`; +} + +function propertyValue(property, type) { + const propertyAsSpec = `styleSpec["${type}_${property.layerType}"]["${property.name}"] as any as StylePropertySpecification`; + + switch (property['property-type']) { + case 'data-driven': + if (property.overridable) { + return `new DataDrivenProperty(${propertyAsSpec}, ${overrides(property)})`; + } else { + return `new DataDrivenProperty(${propertyAsSpec})`; + } + case 'cross-faded': + return `new CrossFadedProperty(${propertyAsSpec})`; + case 'cross-faded-data-driven': + return `new CrossFadedDataDrivenProperty(${propertyAsSpec})`; + case 'color-ramp': + return `new ColorRampProperty(${propertyAsSpec})`; + case 'data-constant': + case 'constant': + return `new DataConstantProperty(${propertyAsSpec})`; + default: + throw new Error(`unknown property-type "${property['property-type']}" for ${property.name}`); + } +} + +const layers = Object.keys(v8.layer.type.values).map((type) => { + const layoutProperties = Object.keys(v8[`layout_${type}`]).reduce((memo, name) => { + if (name !== 'visibility') { + v8[`layout_${type}`][name].name = name; + v8[`layout_${type}`][name].layerType = type; + memo.push(v8[`layout_${type}`][name]); + } + return memo; + }, []); + + const paintProperties = Object.keys(v8[`paint_${type}`]).reduce((memo, name) => { + v8[`paint_${type}`][name].name = name; + v8[`paint_${type}`][name].layerType = type; + memo.push(v8[`paint_${type}`][name]); + return memo; + }, []); + + return {type, layoutProperties, paintProperties}; +}); + +function emitlayerProperties(locals) { + const output = []; + const layerType = pascalCase(locals.type); + const { + layoutProperties, + paintProperties + } = locals; + + output.push( + `// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ + +import {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec'; + +import { + Properties, + DataConstantProperty, + DataDrivenProperty, + CrossFadedDataDrivenProperty, + CrossFadedProperty, + ColorRampProperty, + PossiblyEvaluatedPropertyValue, + CrossFaded +} from '../properties'; + +import type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec'; +import {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec'; +`); + + const overridables = paintProperties.filter(p => p.overridable); + if (overridables.length) { + const overridesArray = `import { + ${overridables.reduce((imports, prop) => { imports.push(runtimeType(prop)); return imports; }, []).join(',\n ')} + } from '@maplibre/maplibre-gl-style-spec'; + `; + console.log(overridesArray); + output.push(overridesArray); + } + + if (layoutProperties.length) { + output.push( + `export type ${layerType}LayoutProps = {`); + + for (const property of layoutProperties) { + output.push( + ` "${property.name}": ${propertyType(property)},`); + } + + output.push( + `}; + +export type ${layerType}LayoutPropsPossiblyEvaluated = {`); + + for (const property of layoutProperties) { + output.push( + ` "${property.name}": ${possiblyEvaluatedType(property)},`); + } + + output.push( + `}; + +let layout: Properties<${layerType}LayoutProps>; +const getLayout = () => layout = layout || new Properties({`); + + for (const property of layoutProperties) { + output.push( + ` "${property.name}": ${propertyValue(property, 'layout')},`); + } + + output.push( + '});'); + } + + if (paintProperties.length) { + output.push( + ` +export type ${layerType}PaintProps = {`); + + for (const property of paintProperties) { + output.push( + ` "${property.name}": ${propertyType(property)},`); + } + + output.push( + `}; + +export type ${layerType}PaintPropsPossiblyEvaluated = {`); + + for (const property of paintProperties) { + output.push( + ` "${property.name}": ${possiblyEvaluatedType(property)},`); + } + + output.push( + '};'); + } else { + output.push( + `export type ${layerType}PaintProps = {};`); + } + + output.push( + ` +let paint: Properties<${layerType}PaintProps>; +const getPaint = () => paint = paint || new Properties({`); + + for (const property of paintProperties) { + output.push( + ` "${property.name}": ${propertyValue(property, 'paint')},`); + } + + output.push( + `}); + +export default ({ get paint() { return getPaint() }${layoutProperties.length ? ', get layout() { return getLayout() }' : ''} });`); + + return output.join('\n'); +} + +for (const layer of layers) { + fs.writeFileSync(`src/style/style_layer/${layer.type.replace('-', '_')}_style_layer_properties.g.ts`, emitlayerProperties(layer)); +} diff --git a/web/libraries/maplibre-gl/build/generate-typings.ts b/web/libraries/maplibre-gl/build/generate-typings.ts new file mode 100644 index 00000000..737520d2 --- /dev/null +++ b/web/libraries/maplibre-gl/build/generate-typings.ts @@ -0,0 +1,18 @@ + +import fs from 'fs'; +import childProcess from 'child_process'; + +if (!fs.existsSync('dist')) { + fs.mkdirSync('dist'); +} + +console.log('Starting bundling types'); +const outputFile = './dist/maplibre-gl.d.ts'; +childProcess.execSync(`dts-bundle-generator --umd-module-name=maplibregl -o ${outputFile} ./src/index.ts`); +let types = fs.readFileSync(outputFile, 'utf8'); +// Classes are not exported but should be since this is exported as UMD - fixing... +types = types.replace(/declare class/g, 'export declare class'); +fs.writeFileSync(outputFile, types); + +console.log('Finished bundling types for MapLibre GL JS'); + diff --git a/web/libraries/maplibre-gl/build/readme.md b/web/libraries/maplibre-gl/build/readme.md new file mode 100644 index 00000000..ba673fa5 --- /dev/null +++ b/web/libraries/maplibre-gl/build/readme.md @@ -0,0 +1,47 @@ +# Build Scripts +This folder holds common build scripts accessed via the various `npm run` commands. +Codegen is executed when calling `npm install` in order to generate all artifacts needed for the build to pass +## Bundeling all the code + +The bundeling process can be split into several steps: + +`npm run build-css` +This command will compile the css code and create the css file. + +`npm run build-prod` and `npm run build-dev` +These commands will use rollup to bundle the code. This is where the magic happens and uses some files in this folder. + +`banner.ts` is used to create a banner at the beginning of the output file + +`rollup_plugins.ts` is used to define common plugins for rollup configurations + +`rollup_plugin_minify_style_spec.ts` is used to specify the plugin used in style spec bundeling + +In the `rollup` folder there are some files that are used as linking files as they link to other files for rollup to pick when bundling. + +Rollup is generating 3 files throughout the process of bundling: + +`index.ts` a file containing all the code that will run in the main thread. + +`shared.ts` a file containing all the code shared between the main and worker code. + +`worker.ts` a file containing all the code the will run in the worker threads. + +These 3 files are then referenced and used by the `bundle_prelude.js` file. It allows loading the web wroker code automatically in web workers without any extra effort from someone who would like to use the library, i.e. it simply works. + +
+ +### `npm run codegen` +The `codegen` command runs the following three scripts, to update the corresponding code files based on the `v8.json` style source, and other data files. Contributors should run this command manually when the underlying style data is modified. The generated code files are then committed to the repo. +#### generate-struct-arrays.ts +Generates `data/array_types.ts`, which consists of: + - `StructArrayLayout_*` subclasses, one for each underlying memory layout + - Named exports mapping each conceptual array type (e.g., `CircleLayoutArray`) to its corresponding `StructArrayLayout` class + - Specific named `StructArray` subclasses, when type-specific struct accessors are needed (e.g., `CollisionBoxArray`) +#### generate-style-code.ts +Generates the various `style/style_layer/[layer type]_style_layer_properties.ts` code files based on the content of `v8.json`. These files provide the type signatures for the paint and layout properties for each type of style layer. +
+ +### Generate Release Nodes + +`release-notes.js` Used to generate release notes when releasing a new version \ No newline at end of file diff --git a/web/libraries/maplibre-gl/build/release-notes.js b/web/libraries/maplibre-gl/build/release-notes.js new file mode 100755 index 00000000..35c3edfb --- /dev/null +++ b/web/libraries/maplibre-gl/build/release-notes.js @@ -0,0 +1,45 @@ +#!/usr/bin/env node + +import * as fs from 'fs'; +import semver from 'semver'; + +const changelogPath = 'CHANGELOG.md'; +const changelog = fs.readFileSync(changelogPath, 'utf8'); + +/* + Parse the raw changelog text and split it into individual releases. + + This regular expression: + - Matches lines starting with "## x.x.x". + - Groups the version number. + - Skips the (optional) release date. + - Groups the changelog content. + - Ends when another "## x.x.x" is found. +*/ +const regex = /^## (\d+\.\d+\.\d+.*?)\n(.+?)(?=\n^## \d+\.\d+\.\d+.*?\n)/gms; + +let releaseNotes = []; +let match; +// eslint-disable-next-line no-cond-assign +while (match = regex.exec(changelog)) { + releaseNotes.push({ + 'version': match[1], + 'changelog': match[2].trim(), + }); +} + +const latest = releaseNotes[0]; +const previous = releaseNotes[1]; + + +// Print the release notes template. + +const templatedReleaseNotes = `https://github.com/maplibre/maplibre-gl-js +[Changes](https://github.com/maplibre/maplibre-gl-js/compare/v${previous.version}...v${latest.version}) since [MapLibre GL JS v${previous.version}](https://github.com/maplibre/releases/tag/v${previous.version}): + +${latest.changelog} + +${semver.prerelease(latest.version) ? 'Pre-release version' : ''}`; + +// eslint-disable-next-line eol-last +process.stdout.write(templatedReleaseNotes.trimEnd()); \ No newline at end of file diff --git a/web/libraries/maplibre-gl/build/rollup/bundle_prelude.js b/web/libraries/maplibre-gl/build/rollup/bundle_prelude.js new file mode 100644 index 00000000..715ecb53 --- /dev/null +++ b/web/libraries/maplibre-gl/build/rollup/bundle_prelude.js @@ -0,0 +1,21 @@ +/* eslint-disable */ + +var shared, worker, maplibregl; +// define gets called three times: one for each chunk. we rely on the order +// they're imported to know which is which +function define(_, chunk) { + if (!shared) { + shared = chunk; + } else if (!worker) { + worker = chunk; + } else { + var workerBundleString = 'var sharedChunk = {}; (' + shared + ')(sharedChunk); (' + worker + ')(sharedChunk);' + + var sharedChunk = {}; + shared(sharedChunk); + maplibregl = chunk(sharedChunk); + if (typeof window !== 'undefined') { + maplibregl.workerUrl = window.URL.createObjectURL(new Blob([workerBundleString], { type: 'text/javascript' })); + } + } +} diff --git a/web/libraries/maplibre-gl/build/rollup/maplibregl.js b/web/libraries/maplibre-gl/build/rollup/maplibregl.js new file mode 100644 index 00000000..b5913a83 --- /dev/null +++ b/web/libraries/maplibre-gl/build/rollup/maplibregl.js @@ -0,0 +1,47 @@ +// +// Our custom intro provides a specialized "define()" function, called by the +// AMD modules below, that sets up the worker blob URL and then executes the +// main module, storing its exported value as 'maplibregl' + +// The three "chunks" imported here are produced by a first Rollup pass, +// which outputs them as AMD modules. + +// Shared dependencies, i.e.: +/* +define(['exports'], function (exports) { + // Code for all common dependencies + // Each module's exports are attached attached to 'exports' (with + // names rewritten to avoid collisions, etc.) +}) +*/ +import '../../staging/maplibregl/shared'; + +// Worker and its unique dependencies, i.e.: +/* +define(['./shared.js'], function (__shared__js) { + // Code for worker script and its unique dependencies. + // Expects the output of 'shared' module to be passed in as an argument, + // since all references to common deps look like, e.g., + // __shared__js.shapeText(). +}); +*/ +// When this wrapper function is passed to our custom define() above, +// it gets stringified, together with the shared wrapper (using +// Function.toString()), and the resulting string of code is made into a +// Blob URL that gets used by the main module to create the web workers. +import '../../staging/maplibregl/worker'; + +// Main module and its unique dependencies +/* +define(['./shared.js'], function (__shared__js) { + // Code for main GL JS module and its unique dependencies. + // Expects the output of 'shared' module to be passed in as an argument, + // since all references to common deps look like, e.g., + // __shared__js.shapeText(). + // + // Returns the actual maplibregl (i.e. src/index.js) +}); +*/ +import '../../staging/maplibregl/index'; + +export default maplibregl; diff --git a/web/libraries/maplibre-gl/build/rollup_plugins.ts b/web/libraries/maplibre-gl/build/rollup_plugins.ts new file mode 100644 index 00000000..6221e023 --- /dev/null +++ b/web/libraries/maplibre-gl/build/rollup_plugins.ts @@ -0,0 +1,58 @@ + +import typescript from '@rollup/plugin-typescript'; +import resolve from '@rollup/plugin-node-resolve'; +import replace from '@rollup/plugin-replace'; +import commonjs from '@rollup/plugin-commonjs'; +import terser from '@rollup/plugin-terser'; +import strip from '@rollup/plugin-strip'; +import {Plugin} from 'rollup'; +import json from '@rollup/plugin-json'; + +// Common set of plugins/transformations shared across different rollup +// builds (main maplibre bundle, style-spec package, benchmarks bundle) + +export const nodeResolve = resolve({ + browser: true, + preferBuiltins: false +}); + +export const plugins = (production: boolean): Plugin[] => [ + json(), + // https://github.com/zaach/jison/issues/351 + replace({ + preventAssignment: true, + include: /\/jsonlint-lines-primitives\/lib\/jsonlint.js/, + delimiters: ['', ''], + values: { + '_token_stack:': '' + } + }), + production && strip({ + sourceMap: true, + functions: ['PerformanceUtils.*', 'Debug.*'] + }), + production && terser({ + compress: { + // eslint-disable-next-line camelcase + pure_getters: true, + passes: 3 + }, + sourceMap: true + }), + nodeResolve, + typescript(), + commonjs({ + // global keyword handling causes Webpack compatibility issues, so we disabled it: + // https://github.com/mapbox/mapbox-gl-js/pull/6956 + ignoreGlobal: true + }) +].filter(Boolean); + +export const watchStagingPlugin: Plugin = { + name: 'watch-external', + buildStart() { + this.addWatchFile('staging/maplibregl/index.js'); + this.addWatchFile('staging/maplibregl/shared.js'); + this.addWatchFile('staging/maplibregl/worker.js'); + } +}; diff --git a/web/libraries/maplibre-gl/dist/LICENSE.txt b/web/libraries/maplibre-gl/dist/LICENSE.txt new file mode 100644 index 00000000..1e8acbb5 --- /dev/null +++ b/web/libraries/maplibre-gl/dist/LICENSE.txt @@ -0,0 +1,116 @@ +Copyright (c) 2023, MapLibre contributors + +All rights reserved. + +Redistribution and use in source and binary forms, with or without modification, +are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. + * Neither the name of MapLibre GL JS nor the names of its contributors + may be used to endorse or promote products derived from this software + without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR +CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, +EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, +PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR +PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING +NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + +------------------------------------------------------------------------------- + +Contains code from mapbox-gl-js v1.13 and earlier + +Version v1.13 of mapbox-gl-js and earlier are licensed under a BSD-3-Clause license + +Copyright (c) 2020, Mapbox +Redistribution and use in source and binary forms, with or without modification, +are permitted provided that the following conditions are met: + +* Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. +* Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. +* Neither the name of Mapbox GL JS nor the names of its contributors + may be used to endorse or promote products derived from this software + without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR +CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, +EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, +PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR +PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING +NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, +EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + +------------------------------------------------------------------------------- + +Contains code from glfx.js + +Copyright (C) 2011 by Evan Wallace + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. + +-------------------------------------------------------------------------------- + +Contains a portion of d3-color https://github.com/d3/d3-color + +Copyright 2010-2016 Mike Bostock +All rights reserved. + +Redistribution and use in source and binary forms, with or without modification, +are permitted provided that the following conditions are met: + +* Redistributions of source code must retain the above copyright notice, this + list of conditions and the following disclaimer. + +* Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. + +* Neither the name of the author nor the names of contributors may be used to + endorse or promote products derived from this software without specific prior + written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR +ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES +(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; +LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON +ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/web/libraries/maplibre-gl/dist/maplibre-gl-csp-worker.js b/web/libraries/maplibre-gl/dist/maplibre-gl-csp-worker.js new file mode 100644 index 00000000..0dc3598f --- /dev/null +++ b/web/libraries/maplibre-gl/dist/maplibre-gl-csp-worker.js @@ -0,0 +1,2 @@ +var maplibregl=function(){"use strict";function t(t,e,r,n){return new(r||(r=Promise))((function(i,s){function a(t){try{l(n.next(t))}catch(t){s(t)}}function o(t){try{l(n.throw(t))}catch(t){s(t)}}function l(t){var e;t.done?i(t.value):(e=t.value,e instanceof r?e:new r((function(t){t(e)}))).then(a,o)}l((n=n.apply(t,e||[])).next())}))}function e(t){return t&&t.__esModule&&Object.prototype.hasOwnProperty.call(t,"default")?t.default:t}"function"==typeof SuppressedError&&SuppressedError;var r=n;function n(t,e){this.x=t,this.y=e}n.prototype={clone:function(){return new n(this.x,this.y)},add:function(t){return this.clone()._add(t)},sub:function(t){return this.clone()._sub(t)},multByPoint:function(t){return this.clone()._multByPoint(t)},divByPoint:function(t){return this.clone()._divByPoint(t)},mult:function(t){return this.clone()._mult(t)},div:function(t){return this.clone()._div(t)},rotate:function(t){return this.clone()._rotate(t)},rotateAround:function(t,e){return this.clone()._rotateAround(t,e)},matMult:function(t){return this.clone()._matMult(t)},unit:function(){return this.clone()._unit()},perp:function(){return this.clone()._perp()},round:function(){return this.clone()._round()},mag:function(){return Math.sqrt(this.x*this.x+this.y*this.y)},equals:function(t){return this.x===t.x&&this.y===t.y},dist:function(t){return Math.sqrt(this.distSqr(t))},distSqr:function(t){var e=t.x-this.x,r=t.y-this.y;return e*e+r*r},angle:function(){return Math.atan2(this.y,this.x)},angleTo:function(t){return Math.atan2(this.y-t.y,this.x-t.x)},angleWith:function(t){return this.angleWithSep(t.x,t.y)},angleWithSep:function(t,e){return Math.atan2(this.x*e-this.y*t,this.x*t+this.y*e)},_matMult:function(t){var e=t[2]*this.x+t[3]*this.y;return this.x=t[0]*this.x+t[1]*this.y,this.y=e,this},_add:function(t){return this.x+=t.x,this.y+=t.y,this},_sub:function(t){return this.x-=t.x,this.y-=t.y,this},_mult:function(t){return this.x*=t,this.y*=t,this},_div:function(t){return this.x/=t,this.y/=t,this},_multByPoint:function(t){return this.x*=t.x,this.y*=t.y,this},_divByPoint:function(t){return this.x/=t.x,this.y/=t.y,this},_unit:function(){return this._div(this.mag()),this},_perp:function(){var t=this.y;return this.y=this.x,this.x=-t,this},_rotate:function(t){var e=Math.cos(t),r=Math.sin(t),n=r*this.x+e*this.y;return this.x=e*this.x-r*this.y,this.y=n,this},_rotateAround:function(t,e){var r=Math.cos(t),n=Math.sin(t),i=e.y+n*(this.x-e.x)+r*(this.y-e.y);return this.x=e.x+r*(this.x-e.x)-n*(this.y-e.y),this.y=i,this},_round:function(){return this.x=Math.round(this.x),this.y=Math.round(this.y),this}},n.convert=function(t){return t instanceof n?t:Array.isArray(t)?new n(t[0],t[1]):t};var i=e(r),s=a;function a(t,e,r,n){this.cx=3*t,this.bx=3*(r-t)-this.cx,this.ax=1-this.cx-this.bx,this.cy=3*e,this.by=3*(n-e)-this.cy,this.ay=1-this.cy-this.by,this.p1x=t,this.p1y=e,this.p2x=r,this.p2y=n}a.prototype={sampleCurveX:function(t){return((this.ax*t+this.bx)*t+this.cx)*t},sampleCurveY:function(t){return((this.ay*t+this.by)*t+this.cy)*t},sampleCurveDerivativeX:function(t){return(3*this.ax*t+2*this.bx)*t+this.cx},solveCurveX:function(t,e){if(void 0===e&&(e=1e-6),t<0)return 0;if(t>1)return 1;for(var r=t,n=0;n<8;n++){var i=this.sampleCurveX(r)-t;if(Math.abs(i)i?a=r:o=r,r=.5*(o-a)+a;return r},solve:function(t,e){return this.sampleCurveY(this.solveCurveX(t,e))}};var o=e(s);let l,u;function c(t,e,r){return Math.min(r,Math.max(e,t))}function h(t,...e){for(const r of e)for(const e in r)t[e]=r[e];return t}function p(t,e,r){const n={};for(const i in t)n[i]=e.call(r||this,t[i],i,t);return n}function f(t){return Array.isArray(t)?t.map(f):"object"==typeof t&&t?p(t,f):t}new o(.25,.1,.25,1);const d={};function y(t){d[t]||("undefined"!=typeof console&&console.warn(t),d[t]=!0)}function m(t,e,r){return(r.y-t.y)*(e.x-t.x)>(e.y-t.y)*(r.x-t.x)}function g(t){let e=0;for(let r,n,i=0,s=t.length,a=s-1;i=u[l+0]&&n>=u[l+1])?(a[h]=!0,s.push(i[h])):a[h]=!1}}}}_forEachCell(t,e,r,n,i,s,a,o){const l=this._convertToCellCoord(t),u=this._convertToCellCoord(e),c=this._convertToCellCoord(r),h=this._convertToCellCoord(n);for(let p=l;p<=c;p++)for(let l=u;l<=h;l++){const u=this.d*l+p;if((!o||o(this._convertFromCellCoord(p),this._convertFromCellCoord(l),this._convertFromCellCoord(p+1),this._convertFromCellCoord(l+1)))&&i.call(this,t,e,r,n,u,s,a,o))return}}_convertFromCellCoord(t){return(t-this.padding)/this.scale}_convertToCellCoord(t){return Math.max(0,Math.min(this.d-1,Math.floor(t*this.scale)+this.padding))}toArrayBuffer(){if(this.arrayBuffer)return this.arrayBuffer;const t=this.cells,e=3+this.cells.length+1+1;let r=0;for(let t=0;t":{},">=":{},"<":{},"<=":{},in:{},"!in":{},all:{},any:{},none:{},has:{},"!has":{},within:{}}},geometry_type:{type:"enum",values:{Point:{},LineString:{},Polygon:{}}},function:{expression:{type:"expression"},stops:{type:"array",value:"function_stop"},base:{type:"number",default:1,minimum:0},property:{type:"string",default:"$zoom"},type:{type:"enum",values:{identity:{},exponential:{},interval:{},categorical:{}},default:"exponential"},colorSpace:{type:"enum",values:{rgb:{},lab:{},hcl:{}},default:"rgb"},default:{type:"*",required:!1}},function_stop:{type:"array",minimum:0,maximum:24,value:["number","color"],length:2},expression:{type:"array",value:"*",minimum:1},light:{anchor:{type:"enum",default:"viewport",values:{map:{},viewport:{}},"property-type":"data-constant",transition:!1,expression:{interpolated:!1,parameters:["zoom"]}},position:{type:"array",default:[1.15,210,30],length:3,value:"number","property-type":"data-constant",transition:!0,expression:{interpolated:!0,parameters:["zoom"]}},color:{type:"color","property-type":"data-constant",default:"#ffffff",expression:{interpolated:!0,parameters:["zoom"]},transition:!0},intensity:{type:"number","property-type":"data-constant",default:.5,minimum:0,maximum:1,expression:{interpolated:!0,parameters:["zoom"]},transition:!0}},terrain:{source:{type:"string",required:!0},exaggeration:{type:"number",minimum:0,default:1}},paint:["paint_fill","paint_line","paint_circle","paint_heatmap","paint_fill-extrusion","paint_symbol","paint_raster","paint_hillshade","paint_background"],paint_fill:{"fill-antialias":{type:"boolean",default:!0,expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"fill-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-color":{type:"color",default:"#000000",transition:!0,requires:[{"!":"fill-pattern"}],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-outline-color":{type:"color",transition:!0,requires:[{"!":"fill-pattern"},{"fill-antialias":!0}],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"fill-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["fill-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"fill-pattern":{type:"resolvedImage",transition:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"cross-faded-data-driven"}},"paint_fill-extrusion":{"fill-extrusion-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"fill-extrusion-color":{type:"color",default:"#000000",transition:!0,requires:[{"!":"fill-extrusion-pattern"}],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-extrusion-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"fill-extrusion-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["fill-extrusion-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"fill-extrusion-pattern":{type:"resolvedImage",transition:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"cross-faded-data-driven"},"fill-extrusion-height":{type:"number",default:0,minimum:0,units:"meters",transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-extrusion-base":{type:"number",default:0,minimum:0,units:"meters",transition:!0,requires:["fill-extrusion-height"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-extrusion-vertical-gradient":{type:"boolean",default:!0,transition:!1,expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"}},paint_line:{"line-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-color":{type:"color",default:"#000000",transition:!0,requires:[{"!":"line-pattern"}],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"line-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["line-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"line-width":{type:"number",default:1,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-gap-width":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-offset":{type:"number",default:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-blur":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-dasharray":{type:"array",value:"number",minimum:0,transition:!0,units:"line widths",requires:[{"!":"line-pattern"}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"cross-faded"},"line-pattern":{type:"resolvedImage",transition:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"cross-faded-data-driven"},"line-gradient":{type:"color",transition:!1,requires:[{"!":"line-dasharray"},{"!":"line-pattern"},{source:"geojson",has:{lineMetrics:!0}}],expression:{interpolated:!0,parameters:["line-progress"]},"property-type":"color-ramp"}},paint_circle:{"circle-radius":{type:"number",default:5,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-color":{type:"color",default:"#000000",transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-blur":{type:"number",default:0,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"circle-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["circle-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"circle-pitch-scale":{type:"enum",values:{map:{},viewport:{}},default:"map",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"circle-pitch-alignment":{type:"enum",values:{map:{},viewport:{}},default:"viewport",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"circle-stroke-width":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-stroke-color":{type:"color",default:"#000000",transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-stroke-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"}},paint_heatmap:{"heatmap-radius":{type:"number",default:30,minimum:1,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"heatmap-weight":{type:"number",default:1,minimum:0,transition:!1,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"heatmap-intensity":{type:"number",default:1,minimum:0,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"heatmap-color":{type:"color",default:["interpolate",["linear"],["heatmap-density"],0,"rgba(0, 0, 255, 0)",.1,"royalblue",.3,"cyan",.5,"lime",.7,"yellow",1,"red"],transition:!1,expression:{interpolated:!0,parameters:["heatmap-density"]},"property-type":"color-ramp"},"heatmap-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"}},paint_symbol:{"icon-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-color":{type:"color",default:"#000000",transition:!0,requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-halo-color":{type:"color",default:"rgba(0, 0, 0, 0)",transition:!0,requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-halo-width":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-halo-blur":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"icon-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["icon-image","icon-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-color":{type:"color",default:"#000000",transition:!0,overridable:!0,requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-halo-color":{type:"color",default:"rgba(0, 0, 0, 0)",transition:!0,requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-halo-width":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-halo-blur":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"text-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["text-field","text-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"}},paint_raster:{"raster-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-hue-rotate":{type:"number",default:0,period:360,transition:!0,units:"degrees",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-brightness-min":{type:"number",default:0,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-brightness-max":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-saturation":{type:"number",default:0,minimum:-1,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-contrast":{type:"number",default:0,minimum:-1,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-resampling":{type:"enum",values:{linear:{},nearest:{}},default:"linear",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"raster-fade-duration":{type:"number",default:300,minimum:0,transition:!1,units:"milliseconds",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"}},paint_hillshade:{"hillshade-illumination-direction":{type:"number",default:335,minimum:0,maximum:359,transition:!1,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-illumination-anchor":{type:"enum",values:{map:{},viewport:{}},default:"viewport",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-exaggeration":{type:"number",default:.5,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-shadow-color":{type:"color",default:"#000000",transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-highlight-color":{type:"color",default:"#FFFFFF",transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-accent-color":{type:"color",default:"#000000",transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"}},paint_background:{"background-color":{type:"color",default:"#000000",transition:!0,requires:[{"!":"background-pattern"}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"background-pattern":{type:"resolvedImage",transition:!0,expression:{interpolated:!1,parameters:["zoom"]},"property-type":"cross-faded"},"background-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"}},transition:{duration:{type:"number",default:300,minimum:0,units:"milliseconds"},delay:{type:"number",default:0,minimum:0,units:"milliseconds"}},"property-type":{"data-driven":{type:"property-type"},"cross-faded":{type:"property-type"},"cross-faded-data-driven":{type:"property-type"},"color-ramp":{type:"property-type"},"data-constant":{type:"property-type"},constant:{type:"property-type"}},promoteId:{"*":{type:"string"}}};const k=["type","source","source-layer","minzoom","maxzoom","filter","layout"];class A{constructor(t,e,r,n){this.message=(t?`${t}: `:"")+r,n&&(this.identifier=n),null!=e&&e.__line__&&(this.line=e.__line__)}}function I(t,...e){for(const r of e)for(const e in r)t[e]=r[e];return t}class z extends Error{constructor(t,e){super(e),this.message=e,this.key=t}}class M{constructor(t,e=[]){this.parent=t,this.bindings={};for(const[t,r]of e)this.bindings[t]=r}concat(t){return new M(this,t)}get(t){if(this.bindings[t])return this.bindings[t];if(this.parent)return this.parent.get(t);throw new Error(`${t} not found in scope.`)}has(t){return!!this.bindings[t]||!!this.parent&&this.parent.has(t)}}const P={kind:"null"},C={kind:"number"},B={kind:"string"},V={kind:"boolean"},E={kind:"color"},F={kind:"object"},T={kind:"value"},D={kind:"collator"},$={kind:"formatted"},L={kind:"padding"},O={kind:"resolvedImage"},U={kind:"variableAnchorOffsetCollection"};function q(t,e){return{kind:"array",itemType:t,N:e}}function j(t){if("array"===t.kind){const e=j(t.itemType);return"number"==typeof t.N?`array<${e}, ${t.N}>`:"value"===t.itemType.kind?"array":`array<${e}>`}return t.kind}const R=[P,C,B,V,E,$,F,q(T),L,O,U];function N(t,e){if("error"===e.kind)return null;if("array"===t.kind){if("array"===e.kind&&(0===e.N&&"value"===e.itemType.kind||!N(t.itemType,e.itemType))&&("number"!=typeof t.N||t.N===e.N))return null}else{if(t.kind===e.kind)return null;if("value"===t.kind)for(const t of R)if(!N(t,e))return null}return`Expected ${j(t)} but found ${j(e)} instead.`}function Z(t,e){return e.some((e=>e.kind===t.kind))}function G(t,e){return e.some((e=>"null"===e?null===t:"array"===e?Array.isArray(t):"object"===e?t&&!Array.isArray(t)&&"object"==typeof t:e===typeof t))}function J(t,e){return"array"===t.kind&&"array"===e.kind?t.itemType.kind===e.itemType.kind&&"number"==typeof t.N:t.kind===e.kind}const K=.96422,X=.82521,Y=4/29,H=6/29,W=3*H*H,Q=H*H*H,tt=Math.PI/180,et=180/Math.PI;function rt(t){return(t%=360)<0&&(t+=360),t}function nt([t,e,r,n]){let i,s;const a=st((.2225045*(t=it(t))+.7168786*(e=it(e))+.0606169*(r=it(r)))/1);t===e&&e===r?i=s=a:(i=st((.4360747*t+.3850649*e+.1430804*r)/K),s=st((.0139322*t+.0971045*e+.7141733*r)/X));const o=116*a-16;return[o<0?0:o,500*(i-a),200*(a-s),n]}function it(t){return t<=.04045?t/12.92:Math.pow((t+.055)/1.055,2.4)}function st(t){return t>Q?Math.pow(t,1/3):t/W+Y}function at([t,e,r,n]){let i=(t+16)/116,s=isNaN(e)?i:i+e/500,a=isNaN(r)?i:i-r/200;return i=1*lt(i),s=K*lt(s),a=X*lt(a),[ot(3.1338561*s-1.6168667*i-.4906146*a),ot(-.9787684*s+1.9161415*i+.033454*a),ot(.0719453*s-.2289914*i+1.4052427*a),n]}function ot(t){return(t=t<=.00304?12.92*t:1.055*Math.pow(t,1/2.4)-.055)<0?0:t>1?1:t}function lt(t){return t>H?t*t*t:W*(t-Y)}function ut(t){return parseInt(t.padEnd(2,t),16)/255}function ct(t,e){return ht(e?t/100:t,0,1)}function ht(t,e,r){return Math.min(Math.max(e,t),r)}function pt(t){return!t.some(Number.isNaN)}const ft={aliceblue:[240,248,255],antiquewhite:[250,235,215],aqua:[0,255,255],aquamarine:[127,255,212],azure:[240,255,255],beige:[245,245,220],bisque:[255,228,196],black:[0,0,0],blanchedalmond:[255,235,205],blue:[0,0,255],blueviolet:[138,43,226],brown:[165,42,42],burlywood:[222,184,135],cadetblue:[95,158,160],chartreuse:[127,255,0],chocolate:[210,105,30],coral:[255,127,80],cornflowerblue:[100,149,237],cornsilk:[255,248,220],crimson:[220,20,60],cyan:[0,255,255],darkblue:[0,0,139],darkcyan:[0,139,139],darkgoldenrod:[184,134,11],darkgray:[169,169,169],darkgreen:[0,100,0],darkgrey:[169,169,169],darkkhaki:[189,183,107],darkmagenta:[139,0,139],darkolivegreen:[85,107,47],darkorange:[255,140,0],darkorchid:[153,50,204],darkred:[139,0,0],darksalmon:[233,150,122],darkseagreen:[143,188,143],darkslateblue:[72,61,139],darkslategray:[47,79,79],darkslategrey:[47,79,79],darkturquoise:[0,206,209],darkviolet:[148,0,211],deeppink:[255,20,147],deepskyblue:[0,191,255],dimgray:[105,105,105],dimgrey:[105,105,105],dodgerblue:[30,144,255],firebrick:[178,34,34],floralwhite:[255,250,240],forestgreen:[34,139,34],fuchsia:[255,0,255],gainsboro:[220,220,220],ghostwhite:[248,248,255],gold:[255,215,0],goldenrod:[218,165,32],gray:[128,128,128],green:[0,128,0],greenyellow:[173,255,47],grey:[128,128,128],honeydew:[240,255,240],hotpink:[255,105,180],indianred:[205,92,92],indigo:[75,0,130],ivory:[255,255,240],khaki:[240,230,140],lavender:[230,230,250],lavenderblush:[255,240,245],lawngreen:[124,252,0],lemonchiffon:[255,250,205],lightblue:[173,216,230],lightcoral:[240,128,128],lightcyan:[224,255,255],lightgoldenrodyellow:[250,250,210],lightgray:[211,211,211],lightgreen:[144,238,144],lightgrey:[211,211,211],lightpink:[255,182,193],lightsalmon:[255,160,122],lightseagreen:[32,178,170],lightskyblue:[135,206,250],lightslategray:[119,136,153],lightslategrey:[119,136,153],lightsteelblue:[176,196,222],lightyellow:[255,255,224],lime:[0,255,0],limegreen:[50,205,50],linen:[250,240,230],magenta:[255,0,255],maroon:[128,0,0],mediumaquamarine:[102,205,170],mediumblue:[0,0,205],mediumorchid:[186,85,211],mediumpurple:[147,112,219],mediumseagreen:[60,179,113],mediumslateblue:[123,104,238],mediumspringgreen:[0,250,154],mediumturquoise:[72,209,204],mediumvioletred:[199,21,133],midnightblue:[25,25,112],mintcream:[245,255,250],mistyrose:[255,228,225],moccasin:[255,228,181],navajowhite:[255,222,173],navy:[0,0,128],oldlace:[253,245,230],olive:[128,128,0],olivedrab:[107,142,35],orange:[255,165,0],orangered:[255,69,0],orchid:[218,112,214],palegoldenrod:[238,232,170],palegreen:[152,251,152],paleturquoise:[175,238,238],palevioletred:[219,112,147],papayawhip:[255,239,213],peachpuff:[255,218,185],peru:[205,133,63],pink:[255,192,203],plum:[221,160,221],powderblue:[176,224,230],purple:[128,0,128],rebeccapurple:[102,51,153],red:[255,0,0],rosybrown:[188,143,143],royalblue:[65,105,225],saddlebrown:[139,69,19],salmon:[250,128,114],sandybrown:[244,164,96],seagreen:[46,139,87],seashell:[255,245,238],sienna:[160,82,45],silver:[192,192,192],skyblue:[135,206,235],slateblue:[106,90,205],slategray:[112,128,144],slategrey:[112,128,144],snow:[255,250,250],springgreen:[0,255,127],steelblue:[70,130,180],tan:[210,180,140],teal:[0,128,128],thistle:[216,191,216],tomato:[255,99,71],turquoise:[64,224,208],violet:[238,130,238],wheat:[245,222,179],white:[255,255,255],whitesmoke:[245,245,245],yellow:[255,255,0],yellowgreen:[154,205,50]};class dt{constructor(t,e,r,n=1,i=!0){this.r=t,this.g=e,this.b=r,this.a=n,i||(this.r*=n,this.g*=n,this.b*=n,n||this.overwriteGetter("rgb",[t,e,r,n]))}static parse(t){if(t instanceof dt)return t;if("string"!=typeof t)return;const e=function(t){if("transparent"===(t=t.toLowerCase().trim()))return[0,0,0,0];const e=ft[t];if(e){const[t,r,n]=e;return[t/255,r/255,n/255,1]}if(t.startsWith("#")&&/^#(?:[0-9a-f]{3,4}|[0-9a-f]{6}|[0-9a-f]{8})$/.test(t)){const e=t.length<6?1:2;let r=1;return[ut(t.slice(r,r+=e)),ut(t.slice(r,r+=e)),ut(t.slice(r,r+=e)),ut(t.slice(r,r+e)||"ff")]}if(t.startsWith("rgb")){const e=t.match(/^rgba?\(\s*([\de.+-]+)(%)?(?:\s+|\s*(,)\s*)([\de.+-]+)(%)?(?:\s+|\s*(,)\s*)([\de.+-]+)(%)?(?:\s*([,\/])\s*([\de.+-]+)(%)?)?\s*\)$/);if(e){const[t,r,n,i,s,a,o,l,u,c,h,p]=e,f=[i||" ",o||" ",c].join("");if(" "===f||" /"===f||",,"===f||",,,"===f){const t=[n,a,u].join(""),e="%%%"===t?100:""===t?255:0;if(e){const t=[ht(+r/e,0,1),ht(+s/e,0,1),ht(+l/e,0,1),h?ct(+h,p):1];if(pt(t))return t}}return}}const r=t.match(/^hsla?\(\s*([\de.+-]+)(?:deg)?(?:\s+|\s*(,)\s*)([\de.+-]+)%(?:\s+|\s*(,)\s*)([\de.+-]+)%(?:\s*([,\/])\s*([\de.+-]+)(%)?)?\s*\)$/);if(r){const[t,e,n,i,s,a,o,l,u]=r,c=[n||" ",s||" ",o].join("");if(" "===c||" /"===c||",,"===c||",,,"===c){const t=[+e,ht(+i,0,100),ht(+a,0,100),l?ct(+l,u):1];if(pt(t))return function([t,e,r,n]){function i(n){const i=(n+t/30)%12,s=e*Math.min(r,1-r);return r-s*Math.max(-1,Math.min(i-3,9-i,1))}return t=rt(t),e/=100,r/=100,[i(0),i(8),i(4),n]}(t)}}}(t);return e?new dt(...e,!1):void 0}get rgb(){const{r:t,g:e,b:r,a:n}=this,i=n||1/0;return this.overwriteGetter("rgb",[t/i,e/i,r/i,n])}get hcl(){return this.overwriteGetter("hcl",function(t){const[e,r,n,i]=nt(t),s=Math.sqrt(r*r+n*n);return[Math.round(1e4*s)?rt(Math.atan2(n,r)*et):NaN,s,e,i]}(this.rgb))}get lab(){return this.overwriteGetter("lab",nt(this.rgb))}overwriteGetter(t,e){return Object.defineProperty(this,t,{value:e}),e}toString(){const[t,e,r,n]=this.rgb;return`rgba(${[t,e,r].map((t=>Math.round(255*t))).join(",")},${n})`}}dt.black=new dt(0,0,0,1),dt.white=new dt(1,1,1,1),dt.transparent=new dt(0,0,0,0),dt.red=new dt(1,0,0,1);class yt{constructor(t,e,r){this.sensitivity=t?e?"variant":"case":e?"accent":"base",this.locale=r,this.collator=new Intl.Collator(this.locale?this.locale:[],{sensitivity:this.sensitivity,usage:"search"})}compare(t,e){return this.collator.compare(t,e)}resolvedLocale(){return new Intl.Collator(this.locale?this.locale:[]).resolvedOptions().locale}}class mt{constructor(t,e,r,n,i){this.text=t,this.image=e,this.scale=r,this.fontStack=n,this.textColor=i}}class gt{constructor(t){this.sections=t}static fromString(t){return new gt([new mt(t,null,null,null,null)])}isEmpty(){return 0===this.sections.length||!this.sections.some((t=>0!==t.text.length||t.image&&0!==t.image.name.length))}static factory(t){return t instanceof gt?t:gt.fromString(t)}toString(){return 0===this.sections.length?"":this.sections.map((t=>t.text)).join("")}}class xt{constructor(t){this.values=t.slice()}static parse(t){if(t instanceof xt)return t;if("number"==typeof t)return new xt([t,t,t,t]);if(Array.isArray(t)&&!(t.length<1||t.length>4)){for(const e of t)if("number"!=typeof e)return;switch(t.length){case 1:t=[t[0],t[0],t[0],t[0]];break;case 2:t=[t[0],t[1],t[0],t[1]];break;case 3:t=[t[0],t[1],t[2],t[1]]}return new xt(t)}}toString(){return JSON.stringify(this.values)}}const vt=new Set(["center","left","right","top","bottom","top-left","top-right","bottom-left","bottom-right"]);class bt{constructor(t){this.values=t.slice()}static parse(t){if(t instanceof bt)return t;if(Array.isArray(t)&&!(t.length<1)&&t.length%2==0){for(let e=0;e=0&&t<=255&&"number"==typeof e&&e>=0&&e<=255&&"number"==typeof r&&r>=0&&r<=255?void 0===n||"number"==typeof n&&n>=0&&n<=1?null:`Invalid rgba value [${[t,e,r,n].join(", ")}]: 'a' must be between 0 and 1.`:`Invalid rgba value [${("number"==typeof n?[t,e,r,n]:[t,e,r]).join(", ")}]: 'r', 'g', and 'b' must be between 0 and 255.`}function St(t){if(null===t||"string"==typeof t||"boolean"==typeof t||"number"==typeof t||t instanceof dt||t instanceof yt||t instanceof gt||t instanceof xt||t instanceof bt||t instanceof wt)return!0;if(Array.isArray(t)){for(const e of t)if(!St(e))return!1;return!0}if("object"==typeof t){for(const e in t)if(!St(t[e]))return!1;return!0}return!1}function kt(t){if(null===t)return P;if("string"==typeof t)return B;if("boolean"==typeof t)return V;if("number"==typeof t)return C;if(t instanceof dt)return E;if(t instanceof yt)return D;if(t instanceof gt)return $;if(t instanceof xt)return L;if(t instanceof bt)return U;if(t instanceof wt)return O;if(Array.isArray(t)){const e=t.length;let r;for(const e of t){const t=kt(e);if(r){if(r===t)continue;r=T;break}r=t}return q(r||T,e)}return F}function At(t){const e=typeof t;return null===t?"":"string"===e||"number"===e||"boolean"===e?String(t):t instanceof dt||t instanceof gt||t instanceof xt||t instanceof bt||t instanceof wt?t.toString():JSON.stringify(t)}class It{constructor(t,e){this.type=t,this.value=e}static parse(t,e){if(2!==t.length)return e.error(`'literal' expression requires exactly one argument, but found ${t.length-1} instead.`);if(!St(t[1]))return e.error("invalid value");const r=t[1];let n=kt(r);const i=e.expectedType;return"array"!==n.kind||0!==n.N||!i||"array"!==i.kind||"number"==typeof i.N&&0!==i.N||(n=i),new It(n,r)}evaluate(){return this.value}eachChild(){}outputDefined(){return!0}}class zt{constructor(t){this.name="ExpressionEvaluationError",this.message=t}toJSON(){return this.message}}const Mt={string:B,number:C,boolean:V,object:F};class Pt{constructor(t,e){this.type=t,this.args=e}static parse(t,e){if(t.length<2)return e.error("Expected at least one argument.");let r,n=1;const i=t[0];if("array"===i){let i,s;if(t.length>2){const r=t[1];if("string"!=typeof r||!(r in Mt)||"object"===r)return e.error('The item type argument of "array" must be one of string, number, boolean',1);i=Mt[r],n++}else i=T;if(t.length>3){if(null!==t[2]&&("number"!=typeof t[2]||t[2]<0||t[2]!==Math.floor(t[2])))return e.error('The length argument to "array" must be a positive integer literal',2);s=t[2],n++}r=q(i,s)}else{if(!Mt[i])throw new Error(`Types doesn't contain name = ${i}`);r=Mt[i]}const s=[];for(;nt.outputDefined()))}}const Ct={"to-boolean":V,"to-color":E,"to-number":C,"to-string":B};class Bt{constructor(t,e){this.type=t,this.args=e}static parse(t,e){if(t.length<2)return e.error("Expected at least one argument.");const r=t[0];if(!Ct[r])throw new Error(`Can't parse ${r} as it is not part of the known types`);if(("to-boolean"===r||"to-string"===r)&&2!==t.length)return e.error("Expected one argument.");const n=Ct[r],i=[];for(let r=1;r4?`Invalid rbga value ${JSON.stringify(e)}: expected an array containing either three or four numeric values.`:_t(e[0],e[1],e[2],e[3]),!r))return new dt(e[0]/255,e[1]/255,e[2]/255,e[3])}throw new zt(r||`Could not parse color from value '${"string"==typeof e?e:JSON.stringify(e)}'`)}case"padding":{let e;for(const r of this.args){e=r.evaluate(t);const n=xt.parse(e);if(n)return n}throw new zt(`Could not parse padding from value '${"string"==typeof e?e:JSON.stringify(e)}'`)}case"variableAnchorOffsetCollection":{let e;for(const r of this.args){e=r.evaluate(t);const n=bt.parse(e);if(n)return n}throw new zt(`Could not parse variableAnchorOffsetCollection from value '${"string"==typeof e?e:JSON.stringify(e)}'`)}case"number":{let e=null;for(const r of this.args){if(e=r.evaluate(t),null===e)return 0;const n=Number(e);if(!isNaN(n))return n}throw new zt(`Could not convert ${JSON.stringify(e)} to number.`)}case"formatted":return gt.fromString(At(this.args[0].evaluate(t)));case"resolvedImage":return wt.fromString(At(this.args[0].evaluate(t)));default:return At(this.args[0].evaluate(t))}}eachChild(t){this.args.forEach(t)}outputDefined(){return this.args.every((t=>t.outputDefined()))}}const Vt=["Unknown","Point","LineString","Polygon"];class Et{constructor(){this.globals=null,this.feature=null,this.featureState=null,this.formattedSection=null,this._parseColorCache={},this.availableImages=null,this.canonical=null}id(){return this.feature&&"id"in this.feature?this.feature.id:null}geometryType(){return this.feature?"number"==typeof this.feature.type?Vt[this.feature.type]:this.feature.type:null}geometry(){return this.feature&&"geometry"in this.feature?this.feature.geometry:null}canonicalID(){return this.canonical}properties(){return this.feature&&this.feature.properties||{}}parseColor(t){let e=this._parseColorCache[t];return e||(e=this._parseColorCache[t]=dt.parse(t)),e}}class Ft{constructor(t,e,r=[],n,i=new M,s=[]){this.registry=t,this.path=r,this.key=r.map((t=>`[${t}]`)).join(""),this.scope=i,this.errors=s,this.expectedType=n,this._isConstant=e}parse(t,e,r,n,i={}){return e?this.concat(e,r,n)._parse(t,i):this._parse(t,i)}_parse(t,e){function r(t,e,r){return"assert"===r?new Pt(e,[t]):"coerce"===r?new Bt(e,[t]):t}if(null!==t&&"string"!=typeof t&&"boolean"!=typeof t&&"number"!=typeof t||(t=["literal",t]),Array.isArray(t)){if(0===t.length)return this.error('Expected an array with at least one element. If you wanted a literal array, use ["literal", []].');const n=t[0];if("string"!=typeof n)return this.error(`Expression name must be a string, but found ${typeof n} instead. If you wanted a literal array, use ["literal", [...]].`,0),null;const i=this.registry[n];if(i){let n=i.parse(t,this);if(!n)return null;if(this.expectedType){const t=this.expectedType,i=n.type;if("string"!==t.kind&&"number"!==t.kind&&"boolean"!==t.kind&&"object"!==t.kind&&"array"!==t.kind||"value"!==i.kind)if("color"!==t.kind&&"formatted"!==t.kind&&"resolvedImage"!==t.kind||"value"!==i.kind&&"string"!==i.kind)if("padding"!==t.kind||"value"!==i.kind&&"number"!==i.kind&&"array"!==i.kind)if("variableAnchorOffsetCollection"!==t.kind||"value"!==i.kind&&"array"!==i.kind){if(this.checkSubtype(t,i))return null}else n=r(n,t,e.typeAnnotation||"coerce");else n=r(n,t,e.typeAnnotation||"coerce");else n=r(n,t,e.typeAnnotation||"coerce");else n=r(n,t,e.typeAnnotation||"assert")}if(!(n instanceof It)&&"resolvedImage"!==n.type.kind&&this._isConstant(n)){const t=new Et;try{n=new It(n.type,n.evaluate(t))}catch(t){return this.error(t.message),null}}return n}return this.error(`Unknown expression "${n}". If you wanted a literal array, use ["literal", [...]].`,0)}return this.error(void 0===t?"'undefined' value invalid. Use null instead.":"object"==typeof t?'Bare objects invalid. Use ["literal", {...}] instead.':`Expected an array, but found ${typeof t} instead.`)}concat(t,e,r){const n="number"==typeof t?this.path.concat(t):this.path,i=r?this.scope.concat(r):this.scope;return new Ft(this.registry,this._isConstant,n,e||null,i,this.errors)}error(t,...e){const r=`${this.key}${e.map((t=>`[${t}]`)).join("")}`;this.errors.push(new z(r,t))}checkSubtype(t,e){const r=N(t,e);return r&&this.error(r),r}}class Tt{constructor(t,e,r){this.type=D,this.locale=r,this.caseSensitive=t,this.diacriticSensitive=e}static parse(t,e){if(2!==t.length)return e.error("Expected one argument.");const r=t[1];if("object"!=typeof r||Array.isArray(r))return e.error("Collator options argument must be an object.");const n=e.parse(void 0!==r["case-sensitive"]&&r["case-sensitive"],1,V);if(!n)return null;const i=e.parse(void 0!==r["diacritic-sensitive"]&&r["diacritic-sensitive"],1,V);if(!i)return null;let s=null;return r.locale&&(s=e.parse(r.locale,1,B),!s)?null:new Tt(n,i,s)}evaluate(t){return new yt(this.caseSensitive.evaluate(t),this.diacriticSensitive.evaluate(t),this.locale?this.locale.evaluate(t):null)}eachChild(t){t(this.caseSensitive),t(this.diacriticSensitive),this.locale&&t(this.locale)}outputDefined(){return!1}}const Dt=8192;function $t(t,e){t[0]=Math.min(t[0],e[0]),t[1]=Math.min(t[1],e[1]),t[2]=Math.max(t[2],e[0]),t[3]=Math.max(t[3],e[1])}function Lt(t,e){return!(t[0]<=e[0]||t[2]>=e[2]||t[1]<=e[1]||t[3]>=e[3])}function Ot(t,e){const r=(180+t[0])/360,n=(180-180/Math.PI*Math.log(Math.tan(Math.PI/4+t[1]*Math.PI/360)))/360,i=Math.pow(2,e.z);return[Math.round(r*i*Dt),Math.round(n*i*Dt)]}function Ut(t,e,r){const n=t[0]-e[0],i=t[1]-e[1],s=t[0]-r[0],a=t[1]-r[1];return n*a-s*i==0&&n*s<=0&&i*a<=0}function qt(t,e){let r=!1;for(let a=0,o=e.length;a(n=t)[1]!=(s=o[e+1])[1]>n[1]&&n[0]<(s[0]-i[0])*(n[1]-i[1])/(s[1]-i[1])+i[0]&&(r=!r)}}var n,i,s;return r}function jt(t,e){for(let r=0;r0&&o<0||a<0&&o>0}function Nt(t,e,r){for(const u of r)for(let r=0;rr[2]){const e=.5*n;let i=t[0]-r[0]>e?-n:r[0]-t[0]>e?n:0;0===i&&(i=t[0]-r[2]>e?-n:r[2]-t[0]>e?n:0),t[0]+=i}$t(e,t)}function Yt(t,e,r,n){const i=Math.pow(2,n.z)*Dt,s=[n.x*Dt,n.y*Dt],a=[];for(const n of t)for(const t of n){const n=[t.x+s[0],t.y+s[1]];Xt(n,e,r,i),a.push(n)}return a}function Ht(t,e,r,n){const i=Math.pow(2,n.z)*Dt,s=[n.x*Dt,n.y*Dt],a=[];for(const r of t){const t=[];for(const n of r){const r=[n.x+s[0],n.y+s[1]];$t(e,r),t.push(r)}a.push(t)}if(e[2]-e[0]<=i/2){(o=e)[0]=o[1]=1/0,o[2]=o[3]=-1/0;for(const t of a)for(const n of t)Xt(n,e,r,i)}var o;return a}class Wt{constructor(t,e){this.type=V,this.geojson=t,this.geometries=e}static parse(t,e){if(2!==t.length)return e.error(`'within' expression requires exactly one argument, but found ${t.length-1} instead.`);if(St(t[1])){const e=t[1];if("FeatureCollection"===e.type)for(let t=0;t!Array.isArray(e)||e.length===t.length-1));let o=null;for(const[n,s]of a){o=new Ft(e.registry,ee,e.path,null,e.scope);const a=[];let l=!1;for(let e=1;e{return e=t,Array.isArray(e)?`(${e.map(j).join(", ")})`:`(${j(e.type)}...)`;var e})).join(" | "),n=[];for(let r=1;r{r=e?r&&ee(t):r&&t instanceof It})),!!r&&re(t)&&ie(t,["zoom","heatmap-density","line-progress","accumulated","is-supported-script"])}function re(t){if(t instanceof te){if("get"===t.name&&1===t.args.length)return!1;if("feature-state"===t.name)return!1;if("has"===t.name&&1===t.args.length)return!1;if("properties"===t.name||"geometry-type"===t.name||"id"===t.name)return!1;if(/^filter-/.test(t.name))return!1}if(t instanceof Wt)return!1;let e=!0;return t.eachChild((t=>{e&&!re(t)&&(e=!1)})),e}function ne(t){if(t instanceof te&&"feature-state"===t.name)return!1;let e=!0;return t.eachChild((t=>{e&&!ne(t)&&(e=!1)})),e}function ie(t,e){if(t instanceof te&&e.indexOf(t.name)>=0)return!1;let r=!0;return t.eachChild((t=>{r&&!ie(t,e)&&(r=!1)})),r}function se(t,e){const r=t.length-1;let n,i,s=0,a=r,o=0;for(;s<=a;)if(o=Math.floor((s+a)/2),n=t[o],i=t[o+1],n<=e){if(o===r||ee))throw new zt("Input is not a number.");a=o-1}return 0}class ae{constructor(t,e,r){this.type=t,this.input=e,this.labels=[],this.outputs=[];for(const[t,e]of r)this.labels.push(t),this.outputs.push(e)}static parse(t,e){if(t.length-1<4)return e.error(`Expected at least 4 arguments, but found only ${t.length-1}.`);if((t.length-1)%2!=0)return e.error("Expected an even number of arguments.");const r=e.parse(t[1],1,C);if(!r)return null;const n=[];let i=null;e.expectedType&&"value"!==e.expectedType.kind&&(i=e.expectedType);for(let r=1;r=s)return e.error('Input/output pairs for "step" expressions must be arranged with input values in strictly ascending order.',o);const u=e.parse(a,l,i);if(!u)return null;i=i||u.type,n.push([s,u])}return new ae(i,r,n)}evaluate(t){const e=this.labels,r=this.outputs;if(1===e.length)return r[0].evaluate(t);const n=this.input.evaluate(t);if(n<=e[0])return r[0].evaluate(t);const i=e.length;return n>=e[i-1]?r[i-1].evaluate(t):r[se(e,n)].evaluate(t)}eachChild(t){t(this.input);for(const e of this.outputs)t(e)}outputDefined(){return this.outputs.every((t=>t.outputDefined()))}}function oe(t,e,r){return t+r*(e-t)}function le(t,e,r){return t.map(((t,n)=>oe(t,e[n],r)))}const ue={number:oe,color:function(t,e,r,n="rgb"){switch(n){case"rgb":{const[n,i,s,a]=le(t.rgb,e.rgb,r);return new dt(n,i,s,a,!1)}case"hcl":{const[n,i,s,a]=t.hcl,[o,l,u,c]=e.hcl;let h,p;if(isNaN(n)||isNaN(o))isNaN(n)?isNaN(o)?h=NaN:(h=o,1!==s&&0!==s||(p=l)):(h=n,1!==u&&0!==u||(p=i));else{let t=o-n;o>n&&t>180?t-=360:o180&&(t+=360),h=n+r*t}const[f,d,y,m]=function([t,e,r,n]){return t=isNaN(t)?0:t*tt,at([r,Math.cos(t)*e,Math.sin(t)*e,n])}([h,null!=p?p:oe(i,l,r),oe(s,u,r),oe(a,c,r)]);return new dt(f,d,y,m,!1)}case"lab":{const[n,i,s,a]=at(le(t.lab,e.lab,r));return new dt(n,i,s,a,!1)}}},array:le,padding:function(t,e,r){return new xt(le(t.values,e.values,r))},variableAnchorOffsetCollection:function(t,e,r){const n=t.values,i=e.values;if(n.length!==i.length)throw new zt(`Cannot interpolate values of different length. from: ${t.toString()}, to: ${e.toString()}`);const s=[];for(let t=0;t"number"!=typeof t||t<0||t>1)))return e.error("Cubic bezier interpolation requires four numeric arguments with values between 0 and 1.",1);n={name:"cubic-bezier",controlPoints:t}}}if(t.length-1<4)return e.error(`Expected at least 4 arguments, but found only ${t.length-1}.`);if((t.length-1)%2!=0)return e.error("Expected an even number of arguments.");if(i=e.parse(i,2,C),!i)return null;const a=[];let o=null;"interpolate-hcl"===r||"interpolate-lab"===r?o=E:e.expectedType&&"value"!==e.expectedType.kind&&(o=e.expectedType);for(let t=0;t=r)return e.error('Input/output pairs for "interpolate" expressions must be arranged with input values in strictly ascending order.',i);const u=e.parse(n,l,o);if(!u)return null;o=o||u.type,a.push([r,u])}return J(o,C)||J(o,E)||J(o,L)||J(o,U)||J(o,q(C))?new ce(o,r,n,i,a):e.error(`Type ${j(o)} is not interpolatable.`)}evaluate(t){const e=this.labels,r=this.outputs;if(1===e.length)return r[0].evaluate(t);const n=this.input.evaluate(t);if(n<=e[0])return r[0].evaluate(t);const i=e.length;if(n>=e[i-1])return r[i-1].evaluate(t);const s=se(e,n),a=ce.interpolationFactor(this.interpolation,n,e[s],e[s+1]),o=r[s].evaluate(t),l=r[s+1].evaluate(t);switch(this.operator){case"interpolate":return ue[this.type.kind](o,l,a);case"interpolate-hcl":return ue.color(o,l,a,"hcl");case"interpolate-lab":return ue.color(o,l,a,"lab")}}eachChild(t){t(this.input);for(const e of this.outputs)t(e)}outputDefined(){return this.outputs.every((t=>t.outputDefined()))}}function he(t,e,r,n){const i=n-r,s=t-r;return 0===i?0:1===e?s/i:(Math.pow(e,s)-1)/(Math.pow(e,i)-1)}class pe{constructor(t,e){this.type=t,this.args=e}static parse(t,e){if(t.length<2)return e.error("Expectected at least one argument.");let r=null;const n=e.expectedType;n&&"value"!==n.kind&&(r=n);const i=[];for(const n of t.slice(1)){const t=e.parse(n,1+i.length,r,void 0,{typeAnnotation:"omit"});if(!t)return null;r=r||t.type,i.push(t)}if(!r)throw new Error("No output type");const s=n&&i.some((t=>N(n,t.type)));return new pe(s?T:r,i)}evaluate(t){let e,r=null,n=0;for(const i of this.args)if(n++,r=i.evaluate(t),r&&r instanceof wt&&!r.available&&(e||(e=r.name),r=null,n===this.args.length&&(r=e)),null!==r)break;return r}eachChild(t){this.args.forEach(t)}outputDefined(){return this.args.every((t=>t.outputDefined()))}}class fe{constructor(t,e){this.type=e.type,this.bindings=[].concat(t),this.result=e}evaluate(t){return this.result.evaluate(t)}eachChild(t){for(const e of this.bindings)t(e[1]);t(this.result)}static parse(t,e){if(t.length<4)return e.error(`Expected at least 3 arguments, but found ${t.length-1} instead.`);const r=[];for(let n=1;n=r.length)throw new zt(`Array index out of bounds: ${e} > ${r.length-1}.`);if(e!==Math.floor(e))throw new zt(`Array index must be an integer, but found ${e} instead.`);return r[e]}eachChild(t){t(this.index),t(this.input)}outputDefined(){return!1}}class ye{constructor(t,e){this.type=V,this.needle=t,this.haystack=e}static parse(t,e){if(3!==t.length)return e.error(`Expected 2 arguments, but found ${t.length-1} instead.`);const r=e.parse(t[1],1,T),n=e.parse(t[2],2,T);return r&&n?Z(r.type,[V,B,C,P,T])?new ye(r,n):e.error(`Expected first argument to be of type boolean, string, number or null, but found ${j(r.type)} instead`):null}evaluate(t){const e=this.needle.evaluate(t),r=this.haystack.evaluate(t);if(!r)return!1;if(!G(e,["boolean","string","number","null"]))throw new zt(`Expected first argument to be of type boolean, string, number or null, but found ${j(kt(e))} instead.`);if(!G(r,["string","array"]))throw new zt(`Expected second argument to be of type array or string, but found ${j(kt(r))} instead.`);return r.indexOf(e)>=0}eachChild(t){t(this.needle),t(this.haystack)}outputDefined(){return!0}}class me{constructor(t,e,r){this.type=C,this.needle=t,this.haystack=e,this.fromIndex=r}static parse(t,e){if(t.length<=2||t.length>=5)return e.error(`Expected 3 or 4 arguments, but found ${t.length-1} instead.`);const r=e.parse(t[1],1,T),n=e.parse(t[2],2,T);if(!r||!n)return null;if(!Z(r.type,[V,B,C,P,T]))return e.error(`Expected first argument to be of type boolean, string, number or null, but found ${j(r.type)} instead`);if(4===t.length){const i=e.parse(t[3],3,C);return i?new me(r,n,i):null}return new me(r,n)}evaluate(t){const e=this.needle.evaluate(t),r=this.haystack.evaluate(t);if(!G(e,["boolean","string","number","null"]))throw new zt(`Expected first argument to be of type boolean, string, number or null, but found ${j(kt(e))} instead.`);if(!G(r,["string","array"]))throw new zt(`Expected second argument to be of type array or string, but found ${j(kt(r))} instead.`);if(this.fromIndex){const n=this.fromIndex.evaluate(t);return r.indexOf(e,n)}return r.indexOf(e)}eachChild(t){t(this.needle),t(this.haystack),this.fromIndex&&t(this.fromIndex)}outputDefined(){return!1}}class ge{constructor(t,e,r,n,i,s){this.inputType=t,this.type=e,this.input=r,this.cases=n,this.outputs=i,this.otherwise=s}static parse(t,e){if(t.length<5)return e.error(`Expected at least 4 arguments, but found only ${t.length-1}.`);if(t.length%2!=1)return e.error("Expected an even number of arguments.");let r,n;e.expectedType&&"value"!==e.expectedType.kind&&(n=e.expectedType);const i={},s=[];for(let a=2;aNumber.MAX_SAFE_INTEGER)return u.error(`Branch labels must be integers no larger than ${Number.MAX_SAFE_INTEGER}.`);if("number"==typeof t&&Math.floor(t)!==t)return u.error("Numeric branch labels must be integer values.");if(r){if(u.checkSubtype(r,kt(t)))return null}else r=kt(t);if(void 0!==i[String(t)])return u.error("Branch labels must be unique.");i[String(t)]=s.length}const c=e.parse(l,a,n);if(!c)return null;n=n||c.type,s.push(c)}const a=e.parse(t[1],1,T);if(!a)return null;const o=e.parse(t[t.length-1],t.length-1,n);return o?"value"!==a.type.kind&&e.concat(1).checkSubtype(r,a.type)?null:new ge(r,n,a,i,s,o):null}evaluate(t){const e=this.input.evaluate(t);return(kt(e)===this.inputType&&this.outputs[this.cases[e]]||this.otherwise).evaluate(t)}eachChild(t){t(this.input),this.outputs.forEach(t),t(this.otherwise)}outputDefined(){return this.outputs.every((t=>t.outputDefined()))&&this.otherwise.outputDefined()}}class xe{constructor(t,e,r){this.type=t,this.branches=e,this.otherwise=r}static parse(t,e){if(t.length<4)return e.error(`Expected at least 3 arguments, but found only ${t.length-1}.`);if(t.length%2!=0)return e.error("Expected an odd number of arguments.");let r;e.expectedType&&"value"!==e.expectedType.kind&&(r=e.expectedType);const n=[];for(let i=1;ie.outputDefined()))&&this.otherwise.outputDefined()}}class ve{constructor(t,e,r,n){this.type=t,this.input=e,this.beginIndex=r,this.endIndex=n}static parse(t,e){if(t.length<=2||t.length>=5)return e.error(`Expected 3 or 4 arguments, but found ${t.length-1} instead.`);const r=e.parse(t[1],1,T),n=e.parse(t[2],2,C);if(!r||!n)return null;if(!Z(r.type,[q(T),B,T]))return e.error(`Expected first argument to be of type array or string, but found ${j(r.type)} instead`);if(4===t.length){const i=e.parse(t[3],3,C);return i?new ve(r.type,r,n,i):null}return new ve(r.type,r,n)}evaluate(t){const e=this.input.evaluate(t),r=this.beginIndex.evaluate(t);if(!G(e,["string","array"]))throw new zt(`Expected first argument to be of type array or string, but found ${j(kt(e))} instead.`);if(this.endIndex){const n=this.endIndex.evaluate(t);return e.slice(r,n)}return e.slice(r)}eachChild(t){t(this.input),t(this.beginIndex),this.endIndex&&t(this.endIndex)}outputDefined(){return!1}}function be(t,e){return"=="===t||"!="===t?"boolean"===e.kind||"string"===e.kind||"number"===e.kind||"null"===e.kind||"value"===e.kind:"string"===e.kind||"number"===e.kind||"value"===e.kind}function we(t,e,r,n){return 0===n.compare(e,r)}function _e(t,e,r){const n="=="!==t&&"!="!==t;return class i{constructor(t,e,r){this.type=V,this.lhs=t,this.rhs=e,this.collator=r,this.hasUntypedArgument="value"===t.type.kind||"value"===e.type.kind}static parse(t,e){if(3!==t.length&&4!==t.length)return e.error("Expected two or three arguments.");const r=t[0];let s=e.parse(t[1],1,T);if(!s)return null;if(!be(r,s.type))return e.concat(1).error(`"${r}" comparisons are not supported for type '${j(s.type)}'.`);let a=e.parse(t[2],2,T);if(!a)return null;if(!be(r,a.type))return e.concat(2).error(`"${r}" comparisons are not supported for type '${j(a.type)}'.`);if(s.type.kind!==a.type.kind&&"value"!==s.type.kind&&"value"!==a.type.kind)return e.error(`Cannot compare types '${j(s.type)}' and '${j(a.type)}'.`);n&&("value"===s.type.kind&&"value"!==a.type.kind?s=new Pt(a.type,[s]):"value"!==s.type.kind&&"value"===a.type.kind&&(a=new Pt(s.type,[a])));let o=null;if(4===t.length){if("string"!==s.type.kind&&"string"!==a.type.kind&&"value"!==s.type.kind&&"value"!==a.type.kind)return e.error("Cannot use collator to compare non-string types.");if(o=e.parse(t[3],3,D),!o)return null}return new i(s,a,o)}evaluate(i){const s=this.lhs.evaluate(i),a=this.rhs.evaluate(i);if(n&&this.hasUntypedArgument){const e=kt(s),r=kt(a);if(e.kind!==r.kind||"string"!==e.kind&&"number"!==e.kind)throw new zt(`Expected arguments for "${t}" to be (string, string) or (number, number), but found (${e.kind}, ${r.kind}) instead.`)}if(this.collator&&!n&&this.hasUntypedArgument){const t=kt(s),r=kt(a);if("string"!==t.kind||"string"!==r.kind)return e(i,s,a)}return this.collator?r(i,s,a,this.collator.evaluate(i)):e(i,s,a)}eachChild(t){t(this.lhs),t(this.rhs),this.collator&&t(this.collator)}outputDefined(){return!0}}}const Se=_e("==",(function(t,e,r){return e===r}),we),ke=_e("!=",(function(t,e,r){return e!==r}),(function(t,e,r,n){return!we(0,e,r,n)})),Ae=_e("<",(function(t,e,r){return e",(function(t,e,r){return e>r}),(function(t,e,r,n){return n.compare(e,r)>0})),ze=_e("<=",(function(t,e,r){return e<=r}),(function(t,e,r,n){return n.compare(e,r)<=0})),Me=_e(">=",(function(t,e,r){return e>=r}),(function(t,e,r,n){return n.compare(e,r)>=0}));class Pe{constructor(t,e,r,n,i){this.type=B,this.number=t,this.locale=e,this.currency=r,this.minFractionDigits=n,this.maxFractionDigits=i}static parse(t,e){if(3!==t.length)return e.error("Expected two arguments.");const r=e.parse(t[1],1,C);if(!r)return null;const n=t[2];if("object"!=typeof n||Array.isArray(n))return e.error("NumberFormat options argument must be an object.");let i=null;if(n.locale&&(i=e.parse(n.locale,1,B),!i))return null;let s=null;if(n.currency&&(s=e.parse(n.currency,1,B),!s))return null;let a=null;if(n["min-fraction-digits"]&&(a=e.parse(n["min-fraction-digits"],1,C),!a))return null;let o=null;return n["max-fraction-digits"]&&(o=e.parse(n["max-fraction-digits"],1,C),!o)?null:new Pe(r,i,s,a,o)}evaluate(t){return new Intl.NumberFormat(this.locale?this.locale.evaluate(t):[],{style:this.currency?"currency":"decimal",currency:this.currency?this.currency.evaluate(t):void 0,minimumFractionDigits:this.minFractionDigits?this.minFractionDigits.evaluate(t):void 0,maximumFractionDigits:this.maxFractionDigits?this.maxFractionDigits.evaluate(t):void 0}).format(this.number.evaluate(t))}eachChild(t){t(this.number),this.locale&&t(this.locale),this.currency&&t(this.currency),this.minFractionDigits&&t(this.minFractionDigits),this.maxFractionDigits&&t(this.maxFractionDigits)}outputDefined(){return!1}}class Ce{constructor(t){this.type=$,this.sections=t}static parse(t,e){if(t.length<2)return e.error("Expected at least one argument.");const r=t[1];if(!Array.isArray(r)&&"object"==typeof r)return e.error("First argument must be an image or text section.");const n=[];let i=!1;for(let r=1;r<=t.length-1;++r){const s=t[r];if(i&&"object"==typeof s&&!Array.isArray(s)){i=!1;let t=null;if(s["font-scale"]&&(t=e.parse(s["font-scale"],1,C),!t))return null;let r=null;if(s["text-font"]&&(r=e.parse(s["text-font"],1,q(B)),!r))return null;let a=null;if(s["text-color"]&&(a=e.parse(s["text-color"],1,E),!a))return null;const o=n[n.length-1];o.scale=t,o.font=r,o.textColor=a}else{const s=e.parse(t[r],1,T);if(!s)return null;const a=s.type.kind;if("string"!==a&&"value"!==a&&"null"!==a&&"resolvedImage"!==a)return e.error("Formatted text type must be 'string', 'value', 'image' or 'null'.");i=!0,n.push({content:s,scale:null,font:null,textColor:null})}}return new Ce(n)}evaluate(t){return new gt(this.sections.map((e=>{const r=e.content.evaluate(t);return kt(r)===O?new mt("",r,null,null,null):new mt(At(r),null,e.scale?e.scale.evaluate(t):null,e.font?e.font.evaluate(t).join(","):null,e.textColor?e.textColor.evaluate(t):null)})))}eachChild(t){for(const e of this.sections)t(e.content),e.scale&&t(e.scale),e.font&&t(e.font),e.textColor&&t(e.textColor)}outputDefined(){return!1}}class Be{constructor(t){this.type=O,this.input=t}static parse(t,e){if(2!==t.length)return e.error("Expected two arguments.");const r=e.parse(t[1],1,B);return r?new Be(r):e.error("No image name provided.")}evaluate(t){const e=this.input.evaluate(t),r=wt.fromString(e);return r&&t.availableImages&&(r.available=t.availableImages.indexOf(e)>-1),r}eachChild(t){t(this.input)}outputDefined(){return!1}}class Ve{constructor(t){this.type=C,this.input=t}static parse(t,e){if(2!==t.length)return e.error(`Expected 1 argument, but found ${t.length-1} instead.`);const r=e.parse(t[1],1);return r?"array"!==r.type.kind&&"string"!==r.type.kind&&"value"!==r.type.kind?e.error(`Expected argument of type string or array, but found ${j(r.type)} instead.`):new Ve(r):null}evaluate(t){const e=this.input.evaluate(t);if("string"==typeof e)return e.length;if(Array.isArray(e))return e.length;throw new zt(`Expected value to be of type string or array, but found ${j(kt(e))} instead.`)}eachChild(t){t(this.input)}outputDefined(){return!1}}const Ee={"==":Se,"!=":ke,">":Ie,"<":Ae,">=":Me,"<=":ze,array:Pt,at:de,boolean:Pt,case:xe,coalesce:pe,collator:Tt,format:Ce,image:Be,in:ye,"index-of":me,interpolate:ce,"interpolate-hcl":ce,"interpolate-lab":ce,length:Ve,let:fe,literal:It,match:ge,number:Pt,"number-format":Pe,object:Pt,slice:ve,step:ae,string:Pt,"to-boolean":Bt,"to-color":Bt,"to-number":Bt,"to-string":Bt,var:Qt,within:Wt};function Fe(t,[e,r,n,i]){e=e.evaluate(t),r=r.evaluate(t),n=n.evaluate(t);const s=i?i.evaluate(t):1,a=_t(e,r,n,s);if(a)throw new zt(a);return new dt(e/255,r/255,n/255,s,!1)}function Te(t,e){return t in e}function De(t,e){const r=e[t];return void 0===r?null:r}function $e(t){return{type:t}}function Le(t){return{result:"success",value:t}}function Oe(t){return{result:"error",value:t}}function Ue(t){return"data-driven"===t["property-type"]||"cross-faded-data-driven"===t["property-type"]}function qe(t){return!!t.expression&&t.expression.parameters.indexOf("zoom")>-1}function je(t){return!!t.expression&&t.expression.interpolated}function Re(t){return t instanceof Number?"number":t instanceof String?"string":t instanceof Boolean?"boolean":Array.isArray(t)?"array":null===t?"null":typeof t}function Ne(t){return"object"==typeof t&&null!==t&&!Array.isArray(t)}function Ze(t){return t}function Ge(t,e){const r="color"===e.type,n=t.stops&&"object"==typeof t.stops[0][0],i=n||!(n||void 0!==t.property),s=t.type||(je(e)?"exponential":"interval");if(r||"padding"===e.type){const n=r?dt.parse:xt.parse;(t=I({},t)).stops&&(t.stops=t.stops.map((t=>[t[0],n(t[1])]))),t.default=n(t.default?t.default:e.default)}if(t.colorSpace&&"rgb"!==(a=t.colorSpace)&&"hcl"!==a&&"lab"!==a)throw new Error(`Unknown color space: "${t.colorSpace}"`);var a;let o,l,u;if("exponential"===s)o=Ye;else if("interval"===s)o=Xe;else if("categorical"===s){o=Ke,l=Object.create(null);for(const e of t.stops)l[e[0]]=e[1];u=typeof t.stops[0][0]}else{if("identity"!==s)throw new Error(`Unknown function type "${s}"`);o=He}if(n){const r={},n=[];for(let e=0;et[0])),evaluate:({zoom:r},n)=>Ye({stops:i,base:t.base},e,r).evaluate(r,n)}}if(i){const r="exponential"===s?{name:"exponential",base:void 0!==t.base?t.base:1}:null;return{kind:"camera",interpolationType:r,interpolationFactor:ce.interpolationFactor.bind(void 0,r),zoomStops:t.stops.map((t=>t[0])),evaluate:({zoom:r})=>o(t,e,r,l,u)}}return{kind:"source",evaluate(r,n){const i=n&&n.properties?n.properties[t.property]:void 0;return void 0===i?Je(t.default,e.default):o(t,e,i,l,u)}}}function Je(t,e,r){return void 0!==t?t:void 0!==e?e:void 0!==r?r:void 0}function Ke(t,e,r,n,i){return Je(typeof r===i?n[r]:void 0,t.default,e.default)}function Xe(t,e,r){if("number"!==Re(r))return Je(t.default,e.default);const n=t.stops.length;if(1===n)return t.stops[0][1];if(r<=t.stops[0][0])return t.stops[0][1];if(r>=t.stops[n-1][0])return t.stops[n-1][1];const i=se(t.stops.map((t=>t[0])),r);return t.stops[i][1]}function Ye(t,e,r){const n=void 0!==t.base?t.base:1;if("number"!==Re(r))return Je(t.default,e.default);const i=t.stops.length;if(1===i)return t.stops[0][1];if(r<=t.stops[0][0])return t.stops[0][1];if(r>=t.stops[i-1][0])return t.stops[i-1][1];const s=se(t.stops.map((t=>t[0])),r),a=function(t,e,r,n){const i=n-r,s=t-r;return 0===i?0:1===e?s/i:(Math.pow(e,s)-1)/(Math.pow(e,i)-1)}(r,n,t.stops[s][0],t.stops[s+1][0]),o=t.stops[s][1],l=t.stops[s+1][1],u=ue[e.type]||Ze;return"function"==typeof o.evaluate?{evaluate(...e){const r=o.evaluate.apply(void 0,e),n=l.evaluate.apply(void 0,e);if(void 0!==r&&void 0!==n)return u(r,n,a,t.colorSpace)}}:u(o,l,a,t.colorSpace)}function He(t,e,r){switch(e.type){case"color":r=dt.parse(r);break;case"formatted":r=gt.fromString(r.toString());break;case"resolvedImage":r=wt.fromString(r.toString());break;case"padding":r=xt.parse(r);break;default:Re(r)===e.type||"enum"===e.type&&e.values[r]||(r=void 0)}return Je(r,t.default,e.default)}te.register(Ee,{error:[{kind:"error"},[B],(t,[e])=>{throw new zt(e.evaluate(t))}],typeof:[B,[T],(t,[e])=>j(kt(e.evaluate(t)))],"to-rgba":[q(C,4),[E],(t,[e])=>{const[r,n,i,s]=e.evaluate(t).rgb;return[255*r,255*n,255*i,s]}],rgb:[E,[C,C,C],Fe],rgba:[E,[C,C,C,C],Fe],has:{type:V,overloads:[[[B],(t,[e])=>Te(e.evaluate(t),t.properties())],[[B,F],(t,[e,r])=>Te(e.evaluate(t),r.evaluate(t))]]},get:{type:T,overloads:[[[B],(t,[e])=>De(e.evaluate(t),t.properties())],[[B,F],(t,[e,r])=>De(e.evaluate(t),r.evaluate(t))]]},"feature-state":[T,[B],(t,[e])=>De(e.evaluate(t),t.featureState||{})],properties:[F,[],t=>t.properties()],"geometry-type":[B,[],t=>t.geometryType()],id:[T,[],t=>t.id()],zoom:[C,[],t=>t.globals.zoom],"heatmap-density":[C,[],t=>t.globals.heatmapDensity||0],"line-progress":[C,[],t=>t.globals.lineProgress||0],accumulated:[T,[],t=>void 0===t.globals.accumulated?null:t.globals.accumulated],"+":[C,$e(C),(t,e)=>{let r=0;for(const n of e)r+=n.evaluate(t);return r}],"*":[C,$e(C),(t,e)=>{let r=1;for(const n of e)r*=n.evaluate(t);return r}],"-":{type:C,overloads:[[[C,C],(t,[e,r])=>e.evaluate(t)-r.evaluate(t)],[[C],(t,[e])=>-e.evaluate(t)]]},"/":[C,[C,C],(t,[e,r])=>e.evaluate(t)/r.evaluate(t)],"%":[C,[C,C],(t,[e,r])=>e.evaluate(t)%r.evaluate(t)],ln2:[C,[],()=>Math.LN2],pi:[C,[],()=>Math.PI],e:[C,[],()=>Math.E],"^":[C,[C,C],(t,[e,r])=>Math.pow(e.evaluate(t),r.evaluate(t))],sqrt:[C,[C],(t,[e])=>Math.sqrt(e.evaluate(t))],log10:[C,[C],(t,[e])=>Math.log(e.evaluate(t))/Math.LN10],ln:[C,[C],(t,[e])=>Math.log(e.evaluate(t))],log2:[C,[C],(t,[e])=>Math.log(e.evaluate(t))/Math.LN2],sin:[C,[C],(t,[e])=>Math.sin(e.evaluate(t))],cos:[C,[C],(t,[e])=>Math.cos(e.evaluate(t))],tan:[C,[C],(t,[e])=>Math.tan(e.evaluate(t))],asin:[C,[C],(t,[e])=>Math.asin(e.evaluate(t))],acos:[C,[C],(t,[e])=>Math.acos(e.evaluate(t))],atan:[C,[C],(t,[e])=>Math.atan(e.evaluate(t))],min:[C,$e(C),(t,e)=>Math.min(...e.map((e=>e.evaluate(t))))],max:[C,$e(C),(t,e)=>Math.max(...e.map((e=>e.evaluate(t))))],abs:[C,[C],(t,[e])=>Math.abs(e.evaluate(t))],round:[C,[C],(t,[e])=>{const r=e.evaluate(t);return r<0?-Math.round(-r):Math.round(r)}],floor:[C,[C],(t,[e])=>Math.floor(e.evaluate(t))],ceil:[C,[C],(t,[e])=>Math.ceil(e.evaluate(t))],"filter-==":[V,[B,T],(t,[e,r])=>t.properties()[e.value]===r.value],"filter-id-==":[V,[T],(t,[e])=>t.id()===e.value],"filter-type-==":[V,[B],(t,[e])=>t.geometryType()===e.value],"filter-<":[V,[B,T],(t,[e,r])=>{const n=t.properties()[e.value],i=r.value;return typeof n==typeof i&&n{const r=t.id(),n=e.value;return typeof r==typeof n&&r":[V,[B,T],(t,[e,r])=>{const n=t.properties()[e.value],i=r.value;return typeof n==typeof i&&n>i}],"filter-id->":[V,[T],(t,[e])=>{const r=t.id(),n=e.value;return typeof r==typeof n&&r>n}],"filter-<=":[V,[B,T],(t,[e,r])=>{const n=t.properties()[e.value],i=r.value;return typeof n==typeof i&&n<=i}],"filter-id-<=":[V,[T],(t,[e])=>{const r=t.id(),n=e.value;return typeof r==typeof n&&r<=n}],"filter->=":[V,[B,T],(t,[e,r])=>{const n=t.properties()[e.value],i=r.value;return typeof n==typeof i&&n>=i}],"filter-id->=":[V,[T],(t,[e])=>{const r=t.id(),n=e.value;return typeof r==typeof n&&r>=n}],"filter-has":[V,[T],(t,[e])=>e.value in t.properties()],"filter-has-id":[V,[],t=>null!==t.id()&&void 0!==t.id()],"filter-type-in":[V,[q(B)],(t,[e])=>e.value.indexOf(t.geometryType())>=0],"filter-id-in":[V,[q(T)],(t,[e])=>e.value.indexOf(t.id())>=0],"filter-in-small":[V,[B,q(T)],(t,[e,r])=>r.value.indexOf(t.properties()[e.value])>=0],"filter-in-large":[V,[B,q(T)],(t,[e,r])=>function(t,e,r,n){for(;r<=n;){const i=r+n>>1;if(e[i]===t)return!0;e[i]>t?n=i-1:r=i+1}return!1}(t.properties()[e.value],r.value,0,r.value.length-1)],all:{type:V,overloads:[[[V,V],(t,[e,r])=>e.evaluate(t)&&r.evaluate(t)],[$e(V),(t,e)=>{for(const r of e)if(!r.evaluate(t))return!1;return!0}]]},any:{type:V,overloads:[[[V,V],(t,[e,r])=>e.evaluate(t)||r.evaluate(t)],[$e(V),(t,e)=>{for(const r of e)if(r.evaluate(t))return!0;return!1}]]},"!":[V,[V],(t,[e])=>!e.evaluate(t)],"is-supported-script":[V,[B],(t,[e])=>{const r=t.globals&&t.globals.isSupportedScript;return!r||r(e.evaluate(t))}],upcase:[B,[B],(t,[e])=>e.evaluate(t).toUpperCase()],downcase:[B,[B],(t,[e])=>e.evaluate(t).toLowerCase()],concat:[B,$e(T),(t,e)=>e.map((e=>At(e.evaluate(t)))).join("")],"resolved-locale":[B,[D],(t,[e])=>e.evaluate(t).resolvedLocale()]});class We{constructor(t,e){var r;this.expression=t,this._warningHistory={},this._evaluator=new Et,this._defaultValue=e?"color"===(r=e).type&&Ne(r.default)?new dt(0,0,0,0):"color"===r.type?dt.parse(r.default)||null:"padding"===r.type?xt.parse(r.default)||null:"variableAnchorOffsetCollection"===r.type?bt.parse(r.default)||null:void 0===r.default?null:r.default:null,this._enumValues=e&&"enum"===e.type?e.values:null}evaluateWithoutErrorHandling(t,e,r,n,i,s){return this._evaluator.globals=t,this._evaluator.feature=e,this._evaluator.featureState=r,this._evaluator.canonical=n,this._evaluator.availableImages=i||null,this._evaluator.formattedSection=s,this.expression.evaluate(this._evaluator)}evaluate(t,e,r,n,i,s){this._evaluator.globals=t,this._evaluator.feature=e||null,this._evaluator.featureState=r||null,this._evaluator.canonical=n,this._evaluator.availableImages=i||null,this._evaluator.formattedSection=s||null;try{const t=this.expression.evaluate(this._evaluator);if(null==t||"number"==typeof t&&t!=t)return this._defaultValue;if(this._enumValues&&!(t in this._enumValues))throw new zt(`Expected value to be one of ${Object.keys(this._enumValues).map((t=>JSON.stringify(t))).join(", ")}, but found ${JSON.stringify(t)} instead.`);return t}catch(t){return this._warningHistory[t.message]||(this._warningHistory[t.message]=!0,"undefined"!=typeof console&&console.warn(t.message)),this._defaultValue}}}function Qe(t){return Array.isArray(t)&&t.length>0&&"string"==typeof t[0]&&t[0]in Ee}function tr(t,e){const r=new Ft(Ee,ee,[],e?function(t){const e={color:E,string:B,number:C,enum:B,boolean:V,formatted:$,padding:L,resolvedImage:O,variableAnchorOffsetCollection:U};return"array"===t.type?q(e[t.value]||T,t.length):e[t.type]}(e):void 0),n=r.parse(t,void 0,void 0,void 0,e&&"string"===e.type?{typeAnnotation:"coerce"}:void 0);return n?Le(new We(n,e)):Oe(r.errors)}class er{constructor(t,e){this.kind=t,this._styleExpression=e,this.isStateDependent="constant"!==t&&!ne(e.expression)}evaluateWithoutErrorHandling(t,e,r,n,i,s){return this._styleExpression.evaluateWithoutErrorHandling(t,e,r,n,i,s)}evaluate(t,e,r,n,i,s){return this._styleExpression.evaluate(t,e,r,n,i,s)}}class rr{constructor(t,e,r,n){this.kind=t,this.zoomStops=r,this._styleExpression=e,this.isStateDependent="camera"!==t&&!ne(e.expression),this.interpolationType=n}evaluateWithoutErrorHandling(t,e,r,n,i,s){return this._styleExpression.evaluateWithoutErrorHandling(t,e,r,n,i,s)}evaluate(t,e,r,n,i,s){return this._styleExpression.evaluate(t,e,r,n,i,s)}interpolationFactor(t,e,r){return this.interpolationType?ce.interpolationFactor(this.interpolationType,t,e,r):0}}function nr(t,e){const r=tr(t,e);if("error"===r.result)return r;const n=r.value.expression,i=re(n);if(!i&&!Ue(e))return Oe([new z("","data expressions not supported")]);const s=ie(n,["zoom"]);if(!s&&!qe(e))return Oe([new z("","zoom expressions not supported")]);const a=sr(n);return a||s?a instanceof z?Oe([a]):a instanceof ce&&!je(e)?Oe([new z("",'"interpolate" expressions cannot be used with this property')]):Le(a?new rr(i?"camera":"composite",r.value,a.labels,a instanceof ce?a.interpolation:void 0):new er(i?"constant":"source",r.value)):Oe([new z("",'"zoom" expression may only be used as input to a top-level "step" or "interpolate" expression.')])}class ir{constructor(t,e){this._parameters=t,this._specification=e,I(this,Ge(this._parameters,this._specification))}static deserialize(t){return new ir(t._parameters,t._specification)}static serialize(t){return{_parameters:t._parameters,_specification:t._specification}}}function sr(t){let e=null;if(t instanceof fe)e=sr(t.result);else if(t instanceof pe){for(const r of t.args)if(e=sr(r),e)break}else(t instanceof ae||t instanceof ce)&&t.input instanceof te&&"zoom"===t.input.name&&(e=t);return e instanceof z||t.eachChild((t=>{const r=sr(t);r instanceof z?e=r:!e&&r?e=new z("",'"zoom" expression may only be used as input to a top-level "step" or "interpolate" expression.'):e&&r&&e!==r&&(e=new z("",'Only one zoom-based "step" or "interpolate" subexpression may be used in an expression.'))})),e}function ar(t){if(!0===t||!1===t)return!0;if(!Array.isArray(t)||0===t.length)return!1;switch(t[0]){case"has":return t.length>=2&&"$id"!==t[1]&&"$type"!==t[1];case"in":return t.length>=3&&("string"!=typeof t[1]||Array.isArray(t[2]));case"!in":case"!has":case"none":return!1;case"==":case"!=":case">":case">=":case"<":case"<=":return 3!==t.length||Array.isArray(t[1])||Array.isArray(t[2]);case"any":case"all":for(const e of t.slice(1))if(!ar(e)&&"boolean"!=typeof e)return!1;return!0;default:return!0}}const or={type:"boolean",default:!1,transition:!1,"property-type":"data-driven",expression:{interpolated:!1,parameters:["zoom","feature"]}};function lr(t){if(null==t)return{filter:()=>!0,needGeometry:!1};ar(t)||(t=hr(t));const e=tr(t,or);if("error"===e.result)throw new Error(e.value.map((t=>`${t.key}: ${t.message}`)).join(", "));return{filter:(t,r,n)=>e.value.evaluate(t,r,{},n),needGeometry:cr(t)}}function ur(t,e){return te?1:0}function cr(t){if(!Array.isArray(t))return!1;if("within"===t[0])return!0;for(let e=1;e"===e||"<="===e||">="===e?pr(t[1],t[2],e):"any"===e?(r=t.slice(1),["any"].concat(r.map(hr))):"all"===e?["all"].concat(t.slice(1).map(hr)):"none"===e?["all"].concat(t.slice(1).map(hr).map(yr)):"in"===e?fr(t[1],t.slice(2)):"!in"===e?yr(fr(t[1],t.slice(2))):"has"===e?dr(t[1]):"!has"===e?yr(dr(t[1])):"within"!==e||t;var r}function pr(t,e,r){switch(t){case"$type":return[`filter-type-${r}`,e];case"$id":return[`filter-id-${r}`,e];default:return[`filter-${r}`,t,e]}}function fr(t,e){if(0===e.length)return!1;switch(t){case"$type":return["filter-type-in",["literal",e]];case"$id":return["filter-id-in",["literal",e]];default:return e.length>200&&!e.some((t=>typeof t!=typeof e[0]))?["filter-in-large",t,["literal",e.sort(ur)]]:["filter-in-small",t,["literal",e]]}}function dr(t){switch(t){case"$type":return!0;case"$id":return["filter-has-id"];default:return["filter-has",t]}}function yr(t){return["!",t]}function mr(t){const e=typeof t;if("number"===e||"boolean"===e||"string"===e||null==t)return JSON.stringify(t);if(Array.isArray(t)){let e="[";for(const r of t)e+=`${mr(r)},`;return`${e}]`}const r=Object.keys(t).sort();let n="{";for(let e=0;en.maximum?[new A(e,r,`${r} is greater than the maximum value ${n.maximum}`)]:[]}function kr(t){const e=t.valueSpec,r=vr(t.value.type);let n,i,s,a={};const o="categorical"!==r&&void 0===t.value.property,l=!o,u="array"===Re(t.value.stops)&&"array"===Re(t.value.stops[0])&&"object"===Re(t.value.stops[0][0]),c=wr({key:t.key,value:t.value,valueSpec:t.styleSpec.function,validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec,objectElementValidators:{stops:function(t){if("identity"===r)return[new A(t.key,t.value,'identity function may not have a "stops" property')];let e=[];const n=t.value;return e=e.concat(_r({key:t.key,value:n,valueSpec:t.valueSpec,validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec,arrayElementValidator:h})),"array"===Re(n)&&0===n.length&&e.push(new A(t.key,n,"array must have at least one stop")),e},default:function(t){return t.validateSpec({key:t.key,value:t.value,valueSpec:e,validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec})}}});return"identity"===r&&o&&c.push(new A(t.key,t.value,'missing required property "property"')),"identity"===r||t.value.stops||c.push(new A(t.key,t.value,'missing required property "stops"')),"exponential"===r&&t.valueSpec.expression&&!je(t.valueSpec)&&c.push(new A(t.key,t.value,"exponential functions not supported")),t.styleSpec.$version>=8&&(l&&!Ue(t.valueSpec)?c.push(new A(t.key,t.value,"property functions not supported")):o&&!qe(t.valueSpec)&&c.push(new A(t.key,t.value,"zoom functions not supported"))),"categorical"!==r&&!u||void 0!==t.value.property||c.push(new A(t.key,t.value,'"property" property is required')),c;function h(t){let r=[];const n=t.value,o=t.key;if("array"!==Re(n))return[new A(o,n,`array expected, ${Re(n)} found`)];if(2!==n.length)return[new A(o,n,`array length 2 expected, length ${n.length} found`)];if(u){if("object"!==Re(n[0]))return[new A(o,n,`object expected, ${Re(n[0])} found`)];if(void 0===n[0].zoom)return[new A(o,n,"object stop key must have zoom")];if(void 0===n[0].value)return[new A(o,n,"object stop key must have value")];if(s&&s>vr(n[0].zoom))return[new A(o,n[0].zoom,"stop zoom values must appear in ascending order")];vr(n[0].zoom)!==s&&(s=vr(n[0].zoom),i=void 0,a={}),r=r.concat(wr({key:`${o}[0]`,value:n[0],valueSpec:{zoom:{}},validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec,objectElementValidators:{zoom:Sr,value:p}}))}else r=r.concat(p({key:`${o}[0]`,value:n[0],valueSpec:{},validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec},n));return Qe(br(n[1]))?r.concat([new A(`${o}[1]`,n[1],"expressions are not allowed in function stops.")]):r.concat(t.validateSpec({key:`${o}[1]`,value:n[1],valueSpec:e,validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec}))}function p(t,s){const o=Re(t.value),l=vr(t.value),u=null!==t.value?t.value:s;if(n){if(o!==n)return[new A(t.key,u,`${o} stop domain type must match previous stop domain type ${n}`)]}else n=o;if("number"!==o&&"string"!==o&&"boolean"!==o)return[new A(t.key,u,"stop domain value must be a number, string, or boolean")];if("number"!==o&&"categorical"!==r){let n=`number expected, ${o} found`;return Ue(e)&&void 0===r&&(n+='\nIf you intended to use a categorical function, specify `"type": "categorical"`.'),[new A(t.key,u,n)]}return"categorical"!==r||"number"!==o||isFinite(l)&&Math.floor(l)===l?"categorical"!==r&&"number"===o&&void 0!==i&&lnew A(`${t.key}${e.key}`,t.value,e.message)));const r=e.value.expression||e.value._styleExpression.expression;if("property"===t.expressionContext&&"text-font"===t.propertyKey&&!r.outputDefined())return[new A(t.key,t.value,`Invalid data expression for "${t.propertyKey}". Output values must be contained as literals within the expression.`)];if("property"===t.expressionContext&&"layout"===t.propertyType&&!ne(r))return[new A(t.key,t.value,'"feature-state" data expressions are not supported with layout properties.')];if("filter"===t.expressionContext&&!ne(r))return[new A(t.key,t.value,'"feature-state" data expressions are not supported with filters.')];if(t.expressionContext&&0===t.expressionContext.indexOf("cluster")){if(!ie(r,["zoom","feature-state"]))return[new A(t.key,t.value,'"zoom" and "feature-state" expressions are not supported with cluster properties.')];if("cluster-initial"===t.expressionContext&&!re(r))return[new A(t.key,t.value,"Feature data expressions are not supported with initial expression part of cluster properties.")]}return[]}function Ir(t){const e=t.key,r=t.value,n=t.valueSpec,i=[];return Array.isArray(n.values)?-1===n.values.indexOf(vr(r))&&i.push(new A(e,r,`expected one of [${n.values.join(", ")}], ${JSON.stringify(r)} found`)):-1===Object.keys(n.values).indexOf(vr(r))&&i.push(new A(e,r,`expected one of [${Object.keys(n.values).join(", ")}], ${JSON.stringify(r)} found`)),i}function zr(t){return ar(br(t.value))?Ar(I({},t,{expressionContext:"filter",valueSpec:{value:"boolean"}})):Mr(t)}function Mr(t){const e=t.value,r=t.key;if("array"!==Re(e))return[new A(r,e,`array expected, ${Re(e)} found`)];const n=t.styleSpec;let i,s=[];if(e.length<1)return[new A(r,e,"filter array must have at least 1 element")];switch(s=s.concat(Ir({key:`${r}[0]`,value:e[0],valueSpec:n.filter_operator,style:t.style,styleSpec:t.styleSpec})),vr(e[0])){case"<":case"<=":case">":case">=":e.length>=2&&"$type"===vr(e[1])&&s.push(new A(r,e,`"$type" cannot be use with operator "${e[0]}"`));case"==":case"!=":3!==e.length&&s.push(new A(r,e,`filter array for operator "${e[0]}" must have 3 elements`));case"in":case"!in":e.length>=2&&(i=Re(e[1]),"string"!==i&&s.push(new A(`${r}[1]`,e[1],`string expected, ${i} found`)));for(let a=2;a{t in r&&e.push(new A(n,r[t],`"${t}" is prohibited for ref layers`))})),i.layers.forEach((e=>{vr(e.id)===o&&(t=e)})),t?t.ref?e.push(new A(n,r.ref,"ref cannot reference another ref layer")):a=vr(t.type):e.push(new A(n,r.ref,`ref layer "${o}" not found`))}else if("background"!==a)if(r.source){const t=i.sources&&i.sources[r.source],s=t&&vr(t.type);t?"vector"===s&&"raster"===a?e.push(new A(n,r.source,`layer "${r.id}" requires a raster source`)):"raster-dem"!==s&&"hillshade"===a?e.push(new A(n,r.source,`layer "${r.id}" requires a raster-dem source`)):"raster"===s&&"raster"!==a?e.push(new A(n,r.source,`layer "${r.id}" requires a vector source`)):"vector"!==s||r["source-layer"]?"raster-dem"===s&&"hillshade"!==a?e.push(new A(n,r.source,"raster-dem source can only be used with layer type 'hillshade'.")):"line"!==a||!r.paint||!r.paint["line-gradient"]||"geojson"===s&&t.lineMetrics||e.push(new A(n,r,`layer "${r.id}" specifies a line-gradient, which requires a GeoJSON source with \`lineMetrics\` enabled.`)):e.push(new A(n,r,`layer "${r.id}" must specify a "source-layer"`)):e.push(new A(n,r.source,`source "${r.source}" not found`))}else e.push(new A(n,r,'missing required property "source"'));return e=e.concat(wr({key:n,value:r,valueSpec:s.layer,style:t.style,styleSpec:t.styleSpec,validateSpec:t.validateSpec,objectElementValidators:{"*":()=>[],type:()=>t.validateSpec({key:`${n}.type`,value:r.type,valueSpec:s.layer.type,style:t.style,styleSpec:t.styleSpec,validateSpec:t.validateSpec,object:r,objectKey:"type"}),filter:zr,layout:t=>wr({layer:r,key:t.key,value:t.value,style:t.style,styleSpec:t.styleSpec,validateSpec:t.validateSpec,objectElementValidators:{"*":t=>Br(I({layerType:a},t))}}),paint:t=>wr({layer:r,key:t.key,value:t.value,style:t.style,styleSpec:t.styleSpec,validateSpec:t.validateSpec,objectElementValidators:{"*":t=>Cr(I({layerType:a},t))}})}})),e}function Er(t){const e=t.value,r=t.key,n=Re(e);return"string"!==n?[new A(r,e,`string expected, ${n} found`)]:[]}const Fr={promoteId:function({key:t,value:e}){if("string"===Re(e))return Er({key:t,value:e});{const r=[];for(const n in e)r.push(...Er({key:`${t}.${n}`,value:e[n]}));return r}}};function Tr(t){const e=t.value,r=t.key,n=t.styleSpec,i=t.style,s=t.validateSpec;if(!e.type)return[new A(r,e,'"type" is required')];const a=vr(e.type);let o;switch(a){case"vector":case"raster":return o=wr({key:r,value:e,valueSpec:n[`source_${a.replace("-","_")}`],style:t.style,styleSpec:n,objectElementValidators:Fr,validateSpec:s}),o;case"raster-dem":return o=function(t){var e;const r=null!==(e=t.sourceName)&&void 0!==e?e:"",n=t.value,i=t.styleSpec,s=i.source_raster_dem,a=t.style;let o=[];const l=Re(n);if(void 0===n)return o;if("object"!==l)return o.push(new A("source_raster_dem",n,`object expected, ${l} found`)),o;const u="custom"===vr(n.encoding),c=["redFactor","greenFactor","blueFactor","baseShift"],h=t.value.encoding?`"${t.value.encoding}"`:"Default";for(const e in n)!u&&c.includes(e)?o.push(new A(e,n[e],`In "${r}": "${e}" is only valid when "encoding" is set to "custom". ${h} encoding found`)):s[e]?o=o.concat(t.validateSpec({key:e,value:n[e],valueSpec:s[e],validateSpec:t.validateSpec,style:a,styleSpec:i})):o.push(new A(e,n[e],`unknown property "${e}"`));return o}({sourceName:r,value:e,style:t.style,styleSpec:n,validateSpec:s}),o;case"geojson":if(o=wr({key:r,value:e,valueSpec:n.source_geojson,style:i,styleSpec:n,validateSpec:s,objectElementValidators:Fr}),e.cluster)for(const t in e.clusterProperties){const[n,i]=e.clusterProperties[t],a="string"==typeof n?[n,["accumulated"],["get",t]]:n;o.push(...Ar({key:`${r}.${t}.map`,value:i,validateSpec:s,expressionContext:"cluster-map"})),o.push(...Ar({key:`${r}.${t}.reduce`,value:a,validateSpec:s,expressionContext:"cluster-reduce"}))}return o;case"video":return wr({key:r,value:e,valueSpec:n.source_video,style:i,validateSpec:s,styleSpec:n});case"image":return wr({key:r,value:e,valueSpec:n.source_image,style:i,validateSpec:s,styleSpec:n});case"canvas":return[new A(r,null,"Please use runtime APIs to add canvas sources, rather than including them in stylesheets.","source.canvas")];default:return Ir({key:`${r}.type`,value:e.type,valueSpec:{values:["vector","raster","raster-dem","geojson","video","image"]},style:i,validateSpec:s,styleSpec:n})}}function Dr(t){const e=t.value,r=t.styleSpec,n=r.light,i=t.style;let s=[];const a=Re(e);if(void 0===e)return s;if("object"!==a)return s=s.concat([new A("light",e,`object expected, ${a} found`)]),s;for(const a in e){const o=a.match(/^(.*)-transition$/);s=s.concat(o&&n[o[1]]&&n[o[1]].transition?t.validateSpec({key:a,value:e[a],valueSpec:r.transition,validateSpec:t.validateSpec,style:i,styleSpec:r}):n[a]?t.validateSpec({key:a,value:e[a],valueSpec:n[a],validateSpec:t.validateSpec,style:i,styleSpec:r}):[new A(a,e[a],`unknown property "${a}"`)])}return s}function $r(t){const e=t.value,r=t.styleSpec,n=r.terrain,i=t.style;let s=[];const a=Re(e);if(void 0===e)return s;if("object"!==a)return s=s.concat([new A("terrain",e,`object expected, ${a} found`)]),s;for(const a in e)s=s.concat(n[a]?t.validateSpec({key:a,value:e[a],valueSpec:n[a],validateSpec:t.validateSpec,style:i,styleSpec:r}):[new A(a,e[a],`unknown property "${a}"`)]);return s}function Lr(t){let e=[];const r=t.value,n=t.key;if(Array.isArray(r)){const i=[],s=[];for(const a in r)r[a].id&&i.includes(r[a].id)&&e.push(new A(n,r,`all the sprites' ids must be unique, but ${r[a].id} is duplicated`)),i.push(r[a].id),r[a].url&&s.includes(r[a].url)&&e.push(new A(n,r,`all the sprites' URLs must be unique, but ${r[a].url} is duplicated`)),s.push(r[a].url),e=e.concat(wr({key:`${n}[${a}]`,value:r[a],valueSpec:{id:{type:"string",required:!0},url:{type:"string",required:!0}},validateSpec:t.validateSpec}));return e}return Er({key:n,value:r})}const Or={"*":()=>[],array:_r,boolean:function(t){const e=t.value,r=t.key,n=Re(e);return"boolean"!==n?[new A(r,e,`boolean expected, ${n} found`)]:[]},number:Sr,color:function(t){const e=t.key,r=t.value,n=Re(r);return"string"!==n?[new A(e,r,`color expected, ${n} found`)]:dt.parse(String(r))?[]:[new A(e,r,`color expected, "${r}" found`)]},constants:xr,enum:Ir,filter:zr,function:kr,layer:Vr,object:wr,source:Tr,light:Dr,terrain:$r,string:Er,formatted:function(t){return 0===Er(t).length?[]:Ar(t)},resolvedImage:function(t){return 0===Er(t).length?[]:Ar(t)},padding:function(t){const e=t.key,r=t.value;if("array"===Re(r)){if(r.length<1||r.length>4)return[new A(e,r,`padding requires 1 to 4 values; ${r.length} values found`)];const n={type:"number"};let i=[];for(let s=0;s[]}})),t.constants&&(r=r.concat(xr({key:"constants",value:t.constants,style:t,styleSpec:e,validateSpec:Ur}))),Nr(r)}function Rr(t){return function(e){return t({...e,validateSpec:Ur})}}function Nr(t){return[].concat(t).sort(((t,e)=>t.line-e.line))}function Zr(t){return function(...e){return Nr(t.apply(this,e))}}jr.source=Zr(Rr(Tr)),jr.sprite=Zr(Rr(Lr)),jr.glyphs=Zr(Rr(qr)),jr.light=Zr(Rr(Dr)),jr.terrain=Zr(Rr($r)),jr.layer=Zr(Rr(Vr)),jr.filter=Zr(Rr(zr)),jr.paintProperty=Zr(Rr(Cr)),jr.layoutProperty=Zr(Rr(Br));const Gr={};class Jr extends Error{constructor(t,e,r,n){super(`AJAXError: ${e} (${t}): ${r}`),this.status=t,this.statusText=e,this.url=r,this.body=n}}const Kr=x()?()=>self.worker&&self.worker.referrer:()=>("blob:"===window.location.protocol?window.parent:window).location.href;function Xr(t,e){const r=new AbortController,n=new Request(t.url,{method:t.method||"GET",body:t.body,credentials:t.credentials,headers:t.headers,cache:t.cache,referrer:Kr(),signal:r.signal});let i=!1,s=!1;"json"===t.type&&n.headers.set("Accept","application/json");return s||fetch(n).then((r=>r.ok?(r=>{("arrayBuffer"===t.type||"image"===t.type?r.arrayBuffer():"json"===t.type?r.json():r.text()).then((t=>{s||(i=!0,e(null,t,r.headers.get("Cache-Control"),r.headers.get("Expires")))})).catch((t=>{s||e(new Error(t.message))}))})(r):r.blob().then((n=>e(new Jr(r.status,r.statusText,t.url,n)))))).catch((t=>{20!==t.code&&e(new Error(t.message))})),{cancel:()=>{s=!0,i||r.abort()}}}const Yr=function(t,e){if(/:\/\//.test(t.url)&&!/^https?:|^file:/.test(t.url)){if(x()&&self.worker&&self.worker.actor)return self.worker.actor.send("getResource",t,e);if(!x())return(Gr[(r=t.url).substring(0,r.indexOf("://"))]||Xr)(t,e)}var r;if(!(t=>/^file:/.test(t)||/^file:/.test(Kr())&&!/^\w+:/.test(t))(t.url)){if(fetch&&Request&&AbortController&&Object.prototype.hasOwnProperty.call(Request.prototype,"signal"))return Xr(t,e);if(x()&&self.worker&&self.worker.actor)return self.worker.actor.send("getResource",t,e,void 0,!0)}return function(t,e){const r=new XMLHttpRequest;r.open(t.method||"GET",t.url,!0),"arrayBuffer"!==t.type&&"image"!==t.type||(r.responseType="arraybuffer");for(const e in t.headers)r.setRequestHeader(e,t.headers[e]);return"json"===t.type&&(r.responseType="text",r.setRequestHeader("Accept","application/json")),r.withCredentials="include"===t.credentials,r.onerror=()=>{e(new Error(r.statusText))},r.onload=()=>{if((r.status>=200&&r.status<300||0===r.status)&&null!==r.response){let n=r.response;if("json"===t.type)try{n=JSON.parse(r.response)}catch(t){return e(t)}e(null,n,r.getResponseHeader("Cache-Control"),r.getResponseHeader("Expires"))}else{const n=new Blob([r.response],{type:r.getResponseHeader("Content-Type")});e(new Jr(r.status,r.statusText,t.url,n))}},r.send(t.body),{cancel:()=>r.abort()}}(t,e)},Hr={};function Wr(t,e,r={}){if(Hr[t])throw new Error(`${t} is already registered.`);Object.defineProperty(e,"_classRegistryKey",{value:t,writeable:!1}),Hr[t]={klass:e,omit:r.omit||[],shallow:r.shallow||[]}}Wr("Object",Object),Wr("TransferableGridIndex",_),Wr("Color",dt),Wr("Error",Error),Wr("AJAXError",Jr),Wr("ResolvedImage",wt),Wr("StylePropertyFunction",ir),Wr("StyleExpression",We,{omit:["_evaluator"]}),Wr("ZoomDependentExpression",rr),Wr("ZoomConstantExpression",er),Wr("CompoundExpression",te,{omit:["_evaluate"]});for(const t in Ee)Ee[t]._classRegistryKey||Wr(`Expression_${t}`,Ee[t]);function Qr(t){return t&&"undefined"!=typeof ArrayBuffer&&(t instanceof ArrayBuffer||t.constructor&&"ArrayBuffer"===t.constructor.name)}function tn(t,e){if(null==t||"boolean"==typeof t||"number"==typeof t||"string"==typeof t||t instanceof Boolean||t instanceof Number||t instanceof String||t instanceof Date||t instanceof RegExp||t instanceof Blob)return t;if(Qr(t))return e&&e.push(t),t;if(v(t))return e&&e.push(t),t;if(ArrayBuffer.isView(t)){const r=t;return e&&e.push(r.buffer),r}if(t instanceof ImageData)return e&&e.push(t.data.buffer),t;if(Array.isArray(t)){const r=[];for(const n of t)r.push(tn(n,e));return r}if("object"==typeof t){const r=t.constructor,n=r._classRegistryKey;if(!n)throw new Error("can't serialize object of unregistered class");if(!Hr[n])throw new Error(`${n} is not registered.`);const i=r.serialize?r.serialize(t,e):{};if(r.serialize){if(e&&i===e[e.length-1])throw new Error("statically serialized object won't survive transfer of $name property")}else{for(const r in t){if(!t.hasOwnProperty(r))continue;if(Hr[n].omit.indexOf(r)>=0)continue;const s=t[r];i[r]=Hr[n].shallow.indexOf(r)>=0?s:tn(s,e)}t instanceof Error&&(i.message=t.message)}if(i.$name)throw new Error("$name property is reserved for worker serialization logic.");return"Object"!==n&&(i.$name=n),i}throw new Error("can't serialize object of type "+typeof t)}function en(t){if(null==t||"boolean"==typeof t||"number"==typeof t||"string"==typeof t||t instanceof Boolean||t instanceof Number||t instanceof String||t instanceof Date||t instanceof RegExp||t instanceof Blob||Qr(t)||v(t)||ArrayBuffer.isView(t)||t instanceof ImageData)return t;if(Array.isArray(t))return t.map(en);if("object"==typeof t){const e=t.$name||"Object";if(!Hr[e])throw new Error(`can't deserialize unregistered class ${e}`);const{klass:r}=Hr[e];if(!r)throw new Error(`can't deserialize unregistered class ${e}`);if(r.deserialize)return r.deserialize(t);const n=Object.create(r.prototype);for(const r of Object.keys(t)){if("$name"===r)continue;const i=t[r];n[r]=Hr[e].shallow.indexOf(r)>=0?i:en(i)}return n}throw new Error("can't deserialize object of type "+typeof t)}class rn{constructor(t){this._callback=t,this._triggered=!1,"undefined"!=typeof MessageChannel&&(this._channel=new MessageChannel,this._channel.port2.onmessage=()=>{this._triggered=!1,this._callback()})}trigger(){this._triggered||(this._triggered=!0,this._channel?this._channel.port1.postMessage(!0):setTimeout((()=>{this._triggered=!1,this._callback()}),0))}remove(){delete this._channel,this._callback=()=>{}}}class nn{constructor(t,e,r){this.receive=t=>{const e=t.data,r=e.id;if(r&&(!e.targetMapId||this.mapId===e.targetMapId))if(""===e.type){delete this.tasks[r];const t=this.cancelCallbacks[r];delete this.cancelCallbacks[r],t&&t()}else x()||e.mustQueue?(this.tasks[r]=e,this.taskQueue.push(r),this.invoker.trigger()):this.processTask(r,e)},this.process=()=>{if(!this.taskQueue.length)return;const t=this.taskQueue.shift(),e=this.tasks[t];delete this.tasks[t],this.taskQueue.length&&this.invoker.trigger(),e&&this.processTask(t,e)},this.target=t,this.parent=e,this.mapId=r,this.callbacks={},this.tasks={},this.taskQueue=[],this.cancelCallbacks={},this.invoker=new rn(this.process),this.target.addEventListener("message",this.receive,!1),this.globalScope=x()?t:window}send(t,e,r,n,i=!1){const s=Math.round(1e18*Math.random()).toString(36).substring(0,10);r&&(this.callbacks[s]=r);const a=[],o={id:s,type:t,hasCallback:!!r,targetMapId:n,mustQueue:i,sourceMapId:this.mapId,data:tn(e,a)};return this.target.postMessage(o,{transfer:a}),{cancel:()=>{r&&delete this.callbacks[s],this.target.postMessage({id:s,type:"",targetMapId:n,sourceMapId:this.mapId})}}}processTask(t,e){if(""===e.type){const r=this.callbacks[t];delete this.callbacks[t],r&&(e.error?r(en(e.error)):r(null,en(e.data)))}else{let r=!1;const n=[],i=e.hasCallback?(e,i)=>{r=!0,delete this.cancelCallbacks[t];const s={id:t,type:"",sourceMapId:this.mapId,error:e?tn(e):null,data:tn(i,n)};this.target.postMessage(s,{transfer:n})}:t=>{r=!0};let s=null;const a=en(e.data);if(this.parent[e.type])s=this.parent[e.type](e.sourceMapId,a,i);else if("getWorkerSource"in this.parent){const t=e.type.split(".");s=this.parent.getWorkerSource(e.sourceMapId,t[0],a.source)[t[1]](a,i)}else i(new Error(`Could not find function ${e.type}`));!r&&s&&s.cancel&&(this.cancelCallbacks[t]=s.cancel)}}remove(){this.invoker.remove(),this.target.removeEventListener("message",this.receive,!1)}}function sn(t,e,r){r[t]&&-1!==r[t].indexOf(e)||(r[t]=r[t]||[],r[t].push(e))}function an(t,e,r){if(r&&r[t]){const n=r[t].indexOf(e);-1!==n&&r[t].splice(n,1)}}class on{constructor(t,e={}){h(this,e),this.type=t}}class ln extends on{constructor(t,e={}){super("error",h({error:t},e))}}class un{on(t,e){return this._listeners=this._listeners||{},sn(t,e,this._listeners),this}off(t,e){return an(t,e,this._listeners),an(t,e,this._oneTimeListeners),this}once(t,e){return e?(this._oneTimeListeners=this._oneTimeListeners||{},sn(t,e,this._oneTimeListeners),this):new Promise((e=>this.once(t,e)))}fire(t,e){"string"==typeof t&&(t=new on(t,e||{}));const r=t.type;if(this.listens(r)){t.target=this;const e=this._listeners&&this._listeners[r]?this._listeners[r].slice():[];for(const r of e)r.call(this,t);const n=this._oneTimeListeners&&this._oneTimeListeners[r]?this._oneTimeListeners[r].slice():[];for(const e of n)an(r,e,this._oneTimeListeners),e.call(this,t);const i=this._eventedParent;i&&(h(t,"function"==typeof this._eventedParentData?this._eventedParentData():this._eventedParentData),i.fire(t))}else t instanceof ln&&console.error(t.error);return this}listens(t){return this._listeners&&this._listeners[t]&&this._listeners[t].length>0||this._oneTimeListeners&&this._oneTimeListeners[t]&&this._oneTimeListeners[t].length>0||this._eventedParent&&this._eventedParent.listens(t)}setEventedParent(t,e){return this._eventedParent=t,this._eventedParentData=e,this}}const cn=jr,hn=cn.paintProperty,pn=cn.layoutProperty;class fn{constructor(){this.first=!0}update(t,e){const r=Math.floor(t);return this.first?(this.first=!1,this.lastIntegerZoom=r,this.lastIntegerZoomTime=0,this.lastZoom=t,this.lastFloorZoom=r,!0):(this.lastFloorZoom>r?(this.lastIntegerZoom=r+1,this.lastIntegerZoomTime=e):this.lastFloorZoomt>=128&&t<=255,Arabic:t=>t>=1536&&t<=1791,"Arabic Supplement":t=>t>=1872&&t<=1919,"Arabic Extended-A":t=>t>=2208&&t<=2303,"Hangul Jamo":t=>t>=4352&&t<=4607,"Unified Canadian Aboriginal Syllabics":t=>t>=5120&&t<=5759,Khmer:t=>t>=6016&&t<=6143,"Unified Canadian Aboriginal Syllabics Extended":t=>t>=6320&&t<=6399,"General Punctuation":t=>t>=8192&&t<=8303,"Letterlike Symbols":t=>t>=8448&&t<=8527,"Number Forms":t=>t>=8528&&t<=8591,"Miscellaneous Technical":t=>t>=8960&&t<=9215,"Control Pictures":t=>t>=9216&&t<=9279,"Optical Character Recognition":t=>t>=9280&&t<=9311,"Enclosed Alphanumerics":t=>t>=9312&&t<=9471,"Geometric Shapes":t=>t>=9632&&t<=9727,"Miscellaneous Symbols":t=>t>=9728&&t<=9983,"Miscellaneous Symbols and Arrows":t=>t>=11008&&t<=11263,"CJK Radicals Supplement":t=>t>=11904&&t<=12031,"Kangxi Radicals":t=>t>=12032&&t<=12255,"Ideographic Description Characters":t=>t>=12272&&t<=12287,"CJK Symbols and Punctuation":t=>t>=12288&&t<=12351,Hiragana:t=>t>=12352&&t<=12447,Katakana:t=>t>=12448&&t<=12543,Bopomofo:t=>t>=12544&&t<=12591,"Hangul Compatibility Jamo":t=>t>=12592&&t<=12687,Kanbun:t=>t>=12688&&t<=12703,"Bopomofo Extended":t=>t>=12704&&t<=12735,"CJK Strokes":t=>t>=12736&&t<=12783,"Katakana Phonetic Extensions":t=>t>=12784&&t<=12799,"Enclosed CJK Letters and Months":t=>t>=12800&&t<=13055,"CJK Compatibility":t=>t>=13056&&t<=13311,"CJK Unified Ideographs Extension A":t=>t>=13312&&t<=19903,"Yijing Hexagram Symbols":t=>t>=19904&&t<=19967,"CJK Unified Ideographs":t=>t>=19968&&t<=40959,"Yi Syllables":t=>t>=40960&&t<=42127,"Yi Radicals":t=>t>=42128&&t<=42191,"Hangul Jamo Extended-A":t=>t>=43360&&t<=43391,"Hangul Syllables":t=>t>=44032&&t<=55215,"Hangul Jamo Extended-B":t=>t>=55216&&t<=55295,"Private Use Area":t=>t>=57344&&t<=63743,"CJK Compatibility Ideographs":t=>t>=63744&&t<=64255,"Arabic Presentation Forms-A":t=>t>=64336&&t<=65023,"Vertical Forms":t=>t>=65040&&t<=65055,"CJK Compatibility Forms":t=>t>=65072&&t<=65103,"Small Form Variants":t=>t>=65104&&t<=65135,"Arabic Presentation Forms-B":t=>t>=65136&&t<=65279,"Halfwidth and Fullwidth Forms":t=>t>=65280&&t<=65519};function yn(t){for(const e of t)if(xn(e.charCodeAt(0)))return!0;return!1}function mn(t){for(const e of t)if(!gn(e.charCodeAt(0)))return!1;return!0}function gn(t){return!(dn.Arabic(t)||dn["Arabic Supplement"](t)||dn["Arabic Extended-A"](t)||dn["Arabic Presentation Forms-A"](t)||dn["Arabic Presentation Forms-B"](t))}function xn(t){return!(746!==t&&747!==t&&(t<4352||!(dn["Bopomofo Extended"](t)||dn.Bopomofo(t)||dn["CJK Compatibility Forms"](t)&&!(t>=65097&&t<=65103)||dn["CJK Compatibility Ideographs"](t)||dn["CJK Compatibility"](t)||dn["CJK Radicals Supplement"](t)||dn["CJK Strokes"](t)||!(!dn["CJK Symbols and Punctuation"](t)||t>=12296&&t<=12305||t>=12308&&t<=12319||12336===t)||dn["CJK Unified Ideographs Extension A"](t)||dn["CJK Unified Ideographs"](t)||dn["Enclosed CJK Letters and Months"](t)||dn["Hangul Compatibility Jamo"](t)||dn["Hangul Jamo Extended-A"](t)||dn["Hangul Jamo Extended-B"](t)||dn["Hangul Jamo"](t)||dn["Hangul Syllables"](t)||dn.Hiragana(t)||dn["Ideographic Description Characters"](t)||dn.Kanbun(t)||dn["Kangxi Radicals"](t)||dn["Katakana Phonetic Extensions"](t)||dn.Katakana(t)&&12540!==t||!(!dn["Halfwidth and Fullwidth Forms"](t)||65288===t||65289===t||65293===t||t>=65306&&t<=65310||65339===t||65341===t||65343===t||t>=65371&&t<=65503||65507===t||t>=65512&&t<=65519)||!(!dn["Small Form Variants"](t)||t>=65112&&t<=65118||t>=65123&&t<=65126)||dn["Unified Canadian Aboriginal Syllabics"](t)||dn["Unified Canadian Aboriginal Syllabics Extended"](t)||dn["Vertical Forms"](t)||dn["Yijing Hexagram Symbols"](t)||dn["Yi Syllables"](t)||dn["Yi Radicals"](t))))}function vn(t){return!(xn(t)||function(t){return!!(dn["Latin-1 Supplement"](t)&&(167===t||169===t||174===t||177===t||188===t||189===t||190===t||215===t||247===t)||dn["General Punctuation"](t)&&(8214===t||8224===t||8225===t||8240===t||8241===t||8251===t||8252===t||8258===t||8263===t||8264===t||8265===t||8273===t)||dn["Letterlike Symbols"](t)||dn["Number Forms"](t)||dn["Miscellaneous Technical"](t)&&(t>=8960&&t<=8967||t>=8972&&t<=8991||t>=8996&&t<=9e3||9003===t||t>=9085&&t<=9114||t>=9150&&t<=9165||9167===t||t>=9169&&t<=9179||t>=9186&&t<=9215)||dn["Control Pictures"](t)&&9251!==t||dn["Optical Character Recognition"](t)||dn["Enclosed Alphanumerics"](t)||dn["Geometric Shapes"](t)||dn["Miscellaneous Symbols"](t)&&!(t>=9754&&t<=9759)||dn["Miscellaneous Symbols and Arrows"](t)&&(t>=11026&&t<=11055||t>=11088&&t<=11097||t>=11192&&t<=11243)||dn["CJK Symbols and Punctuation"](t)||dn.Katakana(t)||dn["Private Use Area"](t)||dn["CJK Compatibility Forms"](t)||dn["Small Form Variants"](t)||dn["Halfwidth and Fullwidth Forms"](t)||8734===t||8756===t||8757===t||t>=9984&&t<=10087||t>=10102&&t<=10131||65532===t||65533===t)}(t))}function bn(t){return t>=1424&&t<=2303||dn["Arabic Presentation Forms-A"](t)||dn["Arabic Presentation Forms-B"](t)}function wn(t,e){return!(!e&&bn(t)||t>=2304&&t<=3583||t>=3840&&t<=4255||dn.Khmer(t))}function _n(t){for(const e of t)if(bn(e.charCodeAt(0)))return!0;return!1}"undefined"!=typeof performance&&performance&&performance.now?performance.now.bind(performance):Date.now.bind(Date);let Sn="unavailable",kn=null;const An={applyArabicShaping:null,processBidirectionalText:null,processStyledBidirectionalText:null,isLoaded:()=>"loaded"===Sn||null!=An.applyArabicShaping,isLoading:()=>"loading"===Sn,setState(t){if(!x())throw new Error("Cannot set the state of the rtl-text-plugin when not in the web-worker context");Sn=t.pluginStatus,kn=t.pluginURL},isParsed(){if(!x())throw new Error("rtl-text-plugin is only parsed on the worker-threads");return null!=An.applyArabicShaping&&null!=An.processBidirectionalText&&null!=An.processStyledBidirectionalText},getPluginURL(){if(!x())throw new Error("rtl-text-plugin url can only be queried from the worker threads");return kn}};class In{constructor(t,e){this.zoom=t,e?(this.now=e.now,this.fadeDuration=e.fadeDuration,this.zoomHistory=e.zoomHistory,this.transition=e.transition):(this.now=0,this.fadeDuration=0,this.zoomHistory=new fn,this.transition={})}isSupportedScript(t){return function(t,e){for(const r of t)if(!wn(r.charCodeAt(0),e))return!1;return!0}(t,An.isLoaded())}crossFadingFactor(){return 0===this.fadeDuration?1:Math.min((this.now-this.zoomHistory.lastIntegerZoomTime)/this.fadeDuration,1)}getCrossfadeParameters(){const t=this.zoom,e=t-Math.floor(t),r=this.crossFadingFactor();return t>this.zoomHistory.lastIntegerZoom?{fromScale:2,toScale:1,t:e+(1-e)*r}:{fromScale:.5,toScale:1,t:1-(1-r)*e}}}class zn{constructor(t,e){this.property=t,this.value=e,this.expression=function(t,e){if(Ne(t))return new ir(t,e);if(Qe(t)){const r=nr(t,e);if("error"===r.result)throw new Error(r.value.map((t=>`${t.key}: ${t.message}`)).join(", "));return r.value}{let r=t;return"color"===e.type&&"string"==typeof t?r=dt.parse(t):"padding"!==e.type||"number"!=typeof t&&!Array.isArray(t)?"variableAnchorOffsetCollection"===e.type&&Array.isArray(t)&&(r=bt.parse(t)):r=xt.parse(t),{kind:"constant",evaluate:()=>r}}}(void 0===e?t.specification.default:e,t.specification)}isDataDriven(){return"source"===this.expression.kind||"composite"===this.expression.kind}possiblyEvaluate(t,e,r){return this.property.possiblyEvaluate(this,t,e,r)}}class Mn{constructor(t){this.property=t,this.value=new zn(t,void 0)}transitioned(t,e){return new Cn(this.property,this.value,e,h({},t.transition,this.transition),t.now)}untransitioned(){return new Cn(this.property,this.value,null,{},0)}}class Pn{constructor(t){this._properties=t,this._values=Object.create(t.defaultTransitionablePropertyValues)}getValue(t){return f(this._values[t].value.value)}setValue(t,e){Object.prototype.hasOwnProperty.call(this._values,t)||(this._values[t]=new Mn(this._values[t].property)),this._values[t].value=new zn(this._values[t].property,null===e?void 0:f(e))}getTransition(t){return f(this._values[t].transition)}setTransition(t,e){Object.prototype.hasOwnProperty.call(this._values,t)||(this._values[t]=new Mn(this._values[t].property)),this._values[t].transition=f(e)||void 0}serialize(){const t={};for(const e of Object.keys(this._values)){const r=this.getValue(e);void 0!==r&&(t[e]=r);const n=this.getTransition(e);void 0!==n&&(t[`${e}-transition`]=n)}return t}transitioned(t,e){const r=new Bn(this._properties);for(const n of Object.keys(this._values))r._values[n]=this._values[n].transitioned(t,e._values[n]);return r}untransitioned(){const t=new Bn(this._properties);for(const e of Object.keys(this._values))t._values[e]=this._values[e].untransitioned();return t}}class Cn{constructor(t,e,r,n,i){this.property=t,this.value=e,this.begin=i+n.delay||0,this.end=this.begin+n.duration||0,t.specification.transition&&(n.delay||n.duration)&&(this.prior=r)}possiblyEvaluate(t,e,r){const n=t.now||0,i=this.value.possiblyEvaluate(t,e,r),s=this.prior;if(s){if(n>this.end)return this.prior=null,i;if(this.value.isDataDriven())return this.prior=null,i;if(n=1)return 1;const e=t*t,r=e*t;return 4*(t<.5?r:3*(t-e)+r-.75)}(a))}}return i}}class Bn{constructor(t){this._properties=t,this._values=Object.create(t.defaultTransitioningPropertyValues)}possiblyEvaluate(t,e,r){const n=new Fn(this._properties);for(const i of Object.keys(this._values))n._values[i]=this._values[i].possiblyEvaluate(t,e,r);return n}hasTransition(){for(const t of Object.keys(this._values))if(this._values[t].prior)return!0;return!1}}class Vn{constructor(t){this._properties=t,this._values=Object.create(t.defaultPropertyValues)}hasValue(t){return void 0!==this._values[t].value}getValue(t){return f(this._values[t].value)}setValue(t,e){this._values[t]=new zn(this._values[t].property,null===e?void 0:f(e))}serialize(){const t={};for(const e of Object.keys(this._values)){const r=this.getValue(e);void 0!==r&&(t[e]=r)}return t}possiblyEvaluate(t,e,r){const n=new Fn(this._properties);for(const i of Object.keys(this._values))n._values[i]=this._values[i].possiblyEvaluate(t,e,r);return n}}class En{constructor(t,e,r){this.property=t,this.value=e,this.parameters=r}isConstant(){return"constant"===this.value.kind}constantOr(t){return"constant"===this.value.kind?this.value.value:t}evaluate(t,e,r,n){return this.property.evaluate(this.value,this.parameters,t,e,r,n)}}class Fn{constructor(t){this._properties=t,this._values=Object.create(t.defaultPossiblyEvaluatedValues)}get(t){return this._values[t]}}class Tn{constructor(t){this.specification=t}possiblyEvaluate(t,e){if(t.isDataDriven())throw new Error("Value should not be data driven");return t.expression.evaluate(e)}interpolate(t,e,r){const n=ue[this.specification.type];return n?n(t,e,r):t}}class Dn{constructor(t,e){this.specification=t,this.overrides=e}possiblyEvaluate(t,e,r,n){return new En(this,"constant"===t.expression.kind||"camera"===t.expression.kind?{kind:"constant",value:t.expression.evaluate(e,null,{},r,n)}:t.expression,e)}interpolate(t,e,r){if("constant"!==t.value.kind||"constant"!==e.value.kind)return t;if(void 0===t.value.value||void 0===e.value.value)return new En(this,{kind:"constant",value:void 0},t.parameters);const n=ue[this.specification.type];if(n){const i=n(t.value.value,e.value.value,r);return new En(this,{kind:"constant",value:i},t.parameters)}return t}evaluate(t,e,r,n,i,s){return"constant"===t.kind?t.value:t.evaluate(e,r,n,i,s)}}class $n extends Dn{possiblyEvaluate(t,e,r,n){if(void 0===t.value)return new En(this,{kind:"constant",value:void 0},e);if("constant"===t.expression.kind){const i=t.expression.evaluate(e,null,{},r,n),s="resolvedImage"===t.property.specification.type&&"string"!=typeof i?i.name:i,a=this._calculate(s,s,s,e);return new En(this,{kind:"constant",value:a},e)}if("camera"===t.expression.kind){const r=this._calculate(t.expression.evaluate({zoom:e.zoom-1}),t.expression.evaluate({zoom:e.zoom}),t.expression.evaluate({zoom:e.zoom+1}),e);return new En(this,{kind:"constant",value:r},e)}return new En(this,t.expression,e)}evaluate(t,e,r,n,i,s){if("source"===t.kind){const a=t.evaluate(e,r,n,i,s);return this._calculate(a,a,a,e)}return"composite"===t.kind?this._calculate(t.evaluate({zoom:Math.floor(e.zoom)-1},r,n),t.evaluate({zoom:Math.floor(e.zoom)},r,n),t.evaluate({zoom:Math.floor(e.zoom)+1},r,n),e):t.value}_calculate(t,e,r,n){return n.zoom>n.zoomHistory.lastIntegerZoom?{from:t,to:e}:{from:r,to:e}}interpolate(t){return t}}class Ln{constructor(t){this.specification=t}possiblyEvaluate(t,e,r,n){if(void 0!==t.value){if("constant"===t.expression.kind){const i=t.expression.evaluate(e,null,{},r,n);return this._calculate(i,i,i,e)}return this._calculate(t.expression.evaluate(new In(Math.floor(e.zoom-1),e)),t.expression.evaluate(new In(Math.floor(e.zoom),e)),t.expression.evaluate(new In(Math.floor(e.zoom+1),e)),e)}}_calculate(t,e,r,n){return n.zoom>n.zoomHistory.lastIntegerZoom?{from:t,to:e}:{from:r,to:e}}interpolate(t){return t}}class On{constructor(t){this.specification=t}possiblyEvaluate(t,e,r,n){return!!t.expression.evaluate(e,null,{},r,n)}interpolate(){return!1}}class Un{constructor(t){this.properties=t,this.defaultPropertyValues={},this.defaultTransitionablePropertyValues={},this.defaultTransitioningPropertyValues={},this.defaultPossiblyEvaluatedValues={},this.overridableProperties=[];for(const e in t){const r=t[e];r.specification.overridable&&this.overridableProperties.push(e);const n=this.defaultPropertyValues[e]=new zn(r,void 0),i=this.defaultTransitionablePropertyValues[e]=new Mn(r);this.defaultTransitioningPropertyValues[e]=i.untransitioned(),this.defaultPossiblyEvaluatedValues[e]=n.possiblyEvaluate({})}}}Wr("DataDrivenProperty",Dn),Wr("DataConstantProperty",Tn),Wr("CrossFadedDataDrivenProperty",$n),Wr("CrossFadedProperty",Ln),Wr("ColorRampProperty",On);const qn="-transition";class jn extends un{constructor(t,e){if(super(),this.id=t.id,this.type=t.type,this._featureFilter={filter:()=>!0,needGeometry:!1},"custom"!==t.type&&(this.metadata=t.metadata,this.minzoom=t.minzoom,this.maxzoom=t.maxzoom,"background"!==t.type&&(this.source=t.source,this.sourceLayer=t["source-layer"],this.filter=t.filter),e.layout&&(this._unevaluatedLayout=new Vn(e.layout)),e.paint)){this._transitionablePaint=new Pn(e.paint);for(const e in t.paint)this.setPaintProperty(e,t.paint[e],{validate:!1});for(const e in t.layout)this.setLayoutProperty(e,t.layout[e],{validate:!1});this._transitioningPaint=this._transitionablePaint.untransitioned(),this.paint=new Fn(e.paint)}}getCrossfadeParameters(){return this._crossfadeParameters}getLayoutProperty(t){return"visibility"===t?this.visibility:this._unevaluatedLayout.getValue(t)}setLayoutProperty(t,e,r={}){null!=e&&this._validate(pn,`layers.${this.id}.layout.${t}`,t,e,r)||("visibility"!==t?this._unevaluatedLayout.setValue(t,e):this.visibility=e)}getPaintProperty(t){return t.endsWith(qn)?this._transitionablePaint.getTransition(t.slice(0,-11)):this._transitionablePaint.getValue(t)}setPaintProperty(t,e,r={}){if(null!=e&&this._validate(hn,`layers.${this.id}.paint.${t}`,t,e,r))return!1;if(t.endsWith(qn))return this._transitionablePaint.setTransition(t.slice(0,-11),e||void 0),!1;{const r=this._transitionablePaint._values[t],n="cross-faded-data-driven"===r.property.specification["property-type"],i=r.value.isDataDriven(),s=r.value;this._transitionablePaint.setValue(t,e),this._handleSpecialPaintPropertyUpdate(t);const a=this._transitionablePaint._values[t].value;return a.isDataDriven()||i||n||this._handleOverridablePaintPropertyUpdate(t,s,a)}}_handleSpecialPaintPropertyUpdate(t){}_handleOverridablePaintPropertyUpdate(t,e,r){return!1}isHidden(t){return!!(this.minzoom&&t=this.maxzoom)||"none"===this.visibility}updateTransitions(t){this._transitioningPaint=this._transitionablePaint.transitioned(t,this._transitioningPaint)}hasTransition(){return this._transitioningPaint.hasTransition()}recalculate(t,e){t.getCrossfadeParameters&&(this._crossfadeParameters=t.getCrossfadeParameters()),this._unevaluatedLayout&&(this.layout=this._unevaluatedLayout.possiblyEvaluate(t,void 0,e)),this.paint=this._transitioningPaint.possiblyEvaluate(t,void 0,e)}serialize(){const t={id:this.id,type:this.type,source:this.source,"source-layer":this.sourceLayer,metadata:this.metadata,minzoom:this.minzoom,maxzoom:this.maxzoom,filter:this.filter,layout:this._unevaluatedLayout&&this._unevaluatedLayout.serialize(),paint:this._transitionablePaint&&this._transitionablePaint.serialize()};return this.visibility&&(t.layout=t.layout||{},t.layout.visibility=this.visibility),function(t,e,r){const n={};for(const r in t)e.call(this,t[r],r,t)&&(n[r]=t[r]);return n}(t,((t,e)=>!(void 0===t||"layout"===e&&!Object.keys(t).length||"paint"===e&&!Object.keys(t).length)))}_validate(t,e,r,n,i={}){return(!i||!1!==i.validate)&&function(t,e){let r=!1;if(e&&e.length)for(const n of e)t.fire(new ln(new Error(n.message))),r=!0;return r}(this,t.call(cn,{key:e,layerType:this.type,objectKey:r,value:n,styleSpec:S,style:{glyphs:!0,sprite:!0}}))}is3D(){return!1}isTileClipped(){return!1}hasOffscreenPass(){return!1}resize(){}isStateDependent(){for(const t in this.paint._values){const e=this.paint.get(t);if(e instanceof En&&Ue(e.property.specification)&&("source"===e.value.kind||"composite"===e.value.kind)&&e.value.isStateDependent)return!0}return!1}}const Rn={Int8:Int8Array,Uint8:Uint8Array,Int16:Int16Array,Uint16:Uint16Array,Int32:Int32Array,Uint32:Uint32Array,Float32:Float32Array};class Nn{constructor(t,e){this._structArray=t,this._pos1=e*this.size,this._pos2=this._pos1/2,this._pos4=this._pos1/4,this._pos8=this._pos1/8}}class Zn{constructor(){this.isTransferred=!1,this.capacity=-1,this.resize(0)}static serialize(t,e){return t._trim(),e&&(t.isTransferred=!0,e.push(t.arrayBuffer)),{length:t.length,arrayBuffer:t.arrayBuffer}}static deserialize(t){const e=Object.create(this.prototype);return e.arrayBuffer=t.arrayBuffer,e.length=t.length,e.capacity=t.arrayBuffer.byteLength/e.bytesPerElement,e._refreshViews(),e}_trim(){this.length!==this.capacity&&(this.capacity=this.length,this.arrayBuffer=this.arrayBuffer.slice(0,this.length*this.bytesPerElement),this._refreshViews())}clear(){this.length=0}resize(t){this.reserve(t),this.length=t}reserve(t){if(t>this.capacity){this.capacity=Math.max(t,Math.floor(5*this.capacity),128),this.arrayBuffer=new ArrayBuffer(this.capacity*this.bytesPerElement);const e=this.uint8;this._refreshViews(),e&&this.uint8.set(e)}}_refreshViews(){throw new Error("_refreshViews() must be implemented by each concrete StructArray layout")}}function Gn(t,e=1){let r=0,n=0;return{members:t.map((t=>{const i=Rn[t.type].BYTES_PER_ELEMENT,s=r=Jn(r,Math.max(e,i)),a=t.components||1;return n=Math.max(n,i),r+=i*a,{name:t.name,type:t.type,components:a,offset:s}})),size:Jn(r,Math.max(n,e)),alignment:e}}function Jn(t,e){return Math.ceil(t/e)*e}class Kn extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e){const r=this.length;return this.resize(r+1),this.emplace(r,t,e)}emplace(t,e,r){const n=2*t;return this.int16[n+0]=e,this.int16[n+1]=r,t}}Kn.prototype.bytesPerElement=4,Wr("StructArrayLayout2i4",Kn);class Xn extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e,r){const n=this.length;return this.resize(n+1),this.emplace(n,t,e,r)}emplace(t,e,r,n){const i=3*t;return this.int16[i+0]=e,this.int16[i+1]=r,this.int16[i+2]=n,t}}Xn.prototype.bytesPerElement=6,Wr("StructArrayLayout3i6",Xn);class Yn extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e,r,n){const i=this.length;return this.resize(i+1),this.emplace(i,t,e,r,n)}emplace(t,e,r,n,i){const s=4*t;return this.int16[s+0]=e,this.int16[s+1]=r,this.int16[s+2]=n,this.int16[s+3]=i,t}}Yn.prototype.bytesPerElement=8,Wr("StructArrayLayout4i8",Yn);class Hn extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e,r,n,i,s){const a=this.length;return this.resize(a+1),this.emplace(a,t,e,r,n,i,s)}emplace(t,e,r,n,i,s,a){const o=6*t;return this.int16[o+0]=e,this.int16[o+1]=r,this.int16[o+2]=n,this.int16[o+3]=i,this.int16[o+4]=s,this.int16[o+5]=a,t}}Hn.prototype.bytesPerElement=12,Wr("StructArrayLayout2i4i12",Hn);class Wn extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e,r,n,i,s){const a=this.length;return this.resize(a+1),this.emplace(a,t,e,r,n,i,s)}emplace(t,e,r,n,i,s,a){const o=4*t,l=8*t;return this.int16[o+0]=e,this.int16[o+1]=r,this.uint8[l+4]=n,this.uint8[l+5]=i,this.uint8[l+6]=s,this.uint8[l+7]=a,t}}Wn.prototype.bytesPerElement=8,Wr("StructArrayLayout2i4ub8",Wn);class Qn extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e){const r=this.length;return this.resize(r+1),this.emplace(r,t,e)}emplace(t,e,r){const n=2*t;return this.float32[n+0]=e,this.float32[n+1]=r,t}}Qn.prototype.bytesPerElement=8,Wr("StructArrayLayout2f8",Qn);class ti extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t,e,r,n,i,s,a,o,l,u){const c=this.length;return this.resize(c+1),this.emplace(c,t,e,r,n,i,s,a,o,l,u)}emplace(t,e,r,n,i,s,a,o,l,u,c){const h=10*t;return this.uint16[h+0]=e,this.uint16[h+1]=r,this.uint16[h+2]=n,this.uint16[h+3]=i,this.uint16[h+4]=s,this.uint16[h+5]=a,this.uint16[h+6]=o,this.uint16[h+7]=l,this.uint16[h+8]=u,this.uint16[h+9]=c,t}}ti.prototype.bytesPerElement=20,Wr("StructArrayLayout10ui20",ti);class ei extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t,e,r,n,i,s,a,o,l,u,c,h){const p=this.length;return this.resize(p+1),this.emplace(p,t,e,r,n,i,s,a,o,l,u,c,h)}emplace(t,e,r,n,i,s,a,o,l,u,c,h,p){const f=12*t;return this.int16[f+0]=e,this.int16[f+1]=r,this.int16[f+2]=n,this.int16[f+3]=i,this.uint16[f+4]=s,this.uint16[f+5]=a,this.uint16[f+6]=o,this.uint16[f+7]=l,this.int16[f+8]=u,this.int16[f+9]=c,this.int16[f+10]=h,this.int16[f+11]=p,t}}ei.prototype.bytesPerElement=24,Wr("StructArrayLayout4i4ui4i24",ei);class ri extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e,r){const n=this.length;return this.resize(n+1),this.emplace(n,t,e,r)}emplace(t,e,r,n){const i=3*t;return this.float32[i+0]=e,this.float32[i+1]=r,this.float32[i+2]=n,t}}ri.prototype.bytesPerElement=12,Wr("StructArrayLayout3f12",ri);class ni extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer)}emplaceBack(t){const e=this.length;return this.resize(e+1),this.emplace(e,t)}emplace(t,e){return this.uint32[1*t+0]=e,t}}ni.prototype.bytesPerElement=4,Wr("StructArrayLayout1ul4",ni);class ii extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t,e,r,n,i,s,a,o,l){const u=this.length;return this.resize(u+1),this.emplace(u,t,e,r,n,i,s,a,o,l)}emplace(t,e,r,n,i,s,a,o,l,u){const c=10*t,h=5*t;return this.int16[c+0]=e,this.int16[c+1]=r,this.int16[c+2]=n,this.int16[c+3]=i,this.int16[c+4]=s,this.int16[c+5]=a,this.uint32[h+3]=o,this.uint16[c+8]=l,this.uint16[c+9]=u,t}}ii.prototype.bytesPerElement=20,Wr("StructArrayLayout6i1ul2ui20",ii);class si extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e,r,n,i,s){const a=this.length;return this.resize(a+1),this.emplace(a,t,e,r,n,i,s)}emplace(t,e,r,n,i,s,a){const o=6*t;return this.int16[o+0]=e,this.int16[o+1]=r,this.int16[o+2]=n,this.int16[o+3]=i,this.int16[o+4]=s,this.int16[o+5]=a,t}}si.prototype.bytesPerElement=12,Wr("StructArrayLayout2i2i2i12",si);class ai extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e,r,n,i){const s=this.length;return this.resize(s+1),this.emplace(s,t,e,r,n,i)}emplace(t,e,r,n,i,s){const a=4*t,o=8*t;return this.float32[a+0]=e,this.float32[a+1]=r,this.float32[a+2]=n,this.int16[o+6]=i,this.int16[o+7]=s,t}}ai.prototype.bytesPerElement=16,Wr("StructArrayLayout2f1f2i16",ai);class oi extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e,r,n){const i=this.length;return this.resize(i+1),this.emplace(i,t,e,r,n)}emplace(t,e,r,n,i){const s=12*t,a=3*t;return this.uint8[s+0]=e,this.uint8[s+1]=r,this.float32[a+1]=n,this.float32[a+2]=i,t}}oi.prototype.bytesPerElement=12,Wr("StructArrayLayout2ub2f12",oi);class li extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t,e,r){const n=this.length;return this.resize(n+1),this.emplace(n,t,e,r)}emplace(t,e,r,n){const i=3*t;return this.uint16[i+0]=e,this.uint16[i+1]=r,this.uint16[i+2]=n,t}}li.prototype.bytesPerElement=6,Wr("StructArrayLayout3ui6",li);class ui extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e,r,n,i,s,a,o,l,u,c,h,p,f,d,y,m){const g=this.length;return this.resize(g+1),this.emplace(g,t,e,r,n,i,s,a,o,l,u,c,h,p,f,d,y,m)}emplace(t,e,r,n,i,s,a,o,l,u,c,h,p,f,d,y,m,g){const x=24*t,v=12*t,b=48*t;return this.int16[x+0]=e,this.int16[x+1]=r,this.uint16[x+2]=n,this.uint16[x+3]=i,this.uint32[v+2]=s,this.uint32[v+3]=a,this.uint32[v+4]=o,this.uint16[x+10]=l,this.uint16[x+11]=u,this.uint16[x+12]=c,this.float32[v+7]=h,this.float32[v+8]=p,this.uint8[b+36]=f,this.uint8[b+37]=d,this.uint8[b+38]=y,this.uint32[v+10]=m,this.int16[x+22]=g,t}}ui.prototype.bytesPerElement=48,Wr("StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48",ui);class ci extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e,r,n,i,s,a,o,l,u,c,h,p,f,d,y,m,g,x,v,b,w,_,S,k,A,I,z){const M=this.length;return this.resize(M+1),this.emplace(M,t,e,r,n,i,s,a,o,l,u,c,h,p,f,d,y,m,g,x,v,b,w,_,S,k,A,I,z)}emplace(t,e,r,n,i,s,a,o,l,u,c,h,p,f,d,y,m,g,x,v,b,w,_,S,k,A,I,z,M){const P=32*t,C=16*t;return this.int16[P+0]=e,this.int16[P+1]=r,this.int16[P+2]=n,this.int16[P+3]=i,this.int16[P+4]=s,this.int16[P+5]=a,this.int16[P+6]=o,this.int16[P+7]=l,this.uint16[P+8]=u,this.uint16[P+9]=c,this.uint16[P+10]=h,this.uint16[P+11]=p,this.uint16[P+12]=f,this.uint16[P+13]=d,this.uint16[P+14]=y,this.uint16[P+15]=m,this.uint16[P+16]=g,this.uint16[P+17]=x,this.uint16[P+18]=v,this.uint16[P+19]=b,this.uint16[P+20]=w,this.uint16[P+21]=_,this.uint16[P+22]=S,this.uint32[C+12]=k,this.float32[C+13]=A,this.float32[C+14]=I,this.uint16[P+30]=z,this.uint16[P+31]=M,t}}ci.prototype.bytesPerElement=64,Wr("StructArrayLayout8i15ui1ul2f2ui64",ci);class hi extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t){const e=this.length;return this.resize(e+1),this.emplace(e,t)}emplace(t,e){return this.float32[1*t+0]=e,t}}hi.prototype.bytesPerElement=4,Wr("StructArrayLayout1f4",hi);class pi extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e,r){const n=this.length;return this.resize(n+1),this.emplace(n,t,e,r)}emplace(t,e,r,n){const i=3*t;return this.uint16[6*t+0]=e,this.float32[i+1]=r,this.float32[i+2]=n,t}}pi.prototype.bytesPerElement=12,Wr("StructArrayLayout1ui2f12",pi);class fi extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t,e,r){const n=this.length;return this.resize(n+1),this.emplace(n,t,e,r)}emplace(t,e,r,n){const i=4*t;return this.uint32[2*t+0]=e,this.uint16[i+2]=r,this.uint16[i+3]=n,t}}fi.prototype.bytesPerElement=8,Wr("StructArrayLayout1ul2ui8",fi);class di extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t,e){const r=this.length;return this.resize(r+1),this.emplace(r,t,e)}emplace(t,e,r){const n=2*t;return this.uint16[n+0]=e,this.uint16[n+1]=r,t}}di.prototype.bytesPerElement=4,Wr("StructArrayLayout2ui4",di);class yi extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t){const e=this.length;return this.resize(e+1),this.emplace(e,t)}emplace(t,e){return this.uint16[1*t+0]=e,t}}yi.prototype.bytesPerElement=2,Wr("StructArrayLayout1ui2",yi);class mi extends Zn{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e,r,n){const i=this.length;return this.resize(i+1),this.emplace(i,t,e,r,n)}emplace(t,e,r,n,i){const s=4*t;return this.float32[s+0]=e,this.float32[s+1]=r,this.float32[s+2]=n,this.float32[s+3]=i,t}}mi.prototype.bytesPerElement=16,Wr("StructArrayLayout4f16",mi);class gi extends Nn{get anchorPointX(){return this._structArray.int16[this._pos2+0]}get anchorPointY(){return this._structArray.int16[this._pos2+1]}get x1(){return this._structArray.int16[this._pos2+2]}get y1(){return this._structArray.int16[this._pos2+3]}get x2(){return this._structArray.int16[this._pos2+4]}get y2(){return this._structArray.int16[this._pos2+5]}get featureIndex(){return this._structArray.uint32[this._pos4+3]}get sourceLayerIndex(){return this._structArray.uint16[this._pos2+8]}get bucketIndex(){return this._structArray.uint16[this._pos2+9]}get anchorPoint(){return new i(this.anchorPointX,this.anchorPointY)}}gi.prototype.size=20;class xi extends ii{get(t){return new gi(this,t)}}Wr("CollisionBoxArray",xi);class vi extends Nn{get anchorX(){return this._structArray.int16[this._pos2+0]}get anchorY(){return this._structArray.int16[this._pos2+1]}get glyphStartIndex(){return this._structArray.uint16[this._pos2+2]}get numGlyphs(){return this._structArray.uint16[this._pos2+3]}get vertexStartIndex(){return this._structArray.uint32[this._pos4+2]}get lineStartIndex(){return this._structArray.uint32[this._pos4+3]}get lineLength(){return this._structArray.uint32[this._pos4+4]}get segment(){return this._structArray.uint16[this._pos2+10]}get lowerSize(){return this._structArray.uint16[this._pos2+11]}get upperSize(){return this._structArray.uint16[this._pos2+12]}get lineOffsetX(){return this._structArray.float32[this._pos4+7]}get lineOffsetY(){return this._structArray.float32[this._pos4+8]}get writingMode(){return this._structArray.uint8[this._pos1+36]}get placedOrientation(){return this._structArray.uint8[this._pos1+37]}set placedOrientation(t){this._structArray.uint8[this._pos1+37]=t}get hidden(){return this._structArray.uint8[this._pos1+38]}set hidden(t){this._structArray.uint8[this._pos1+38]=t}get crossTileID(){return this._structArray.uint32[this._pos4+10]}set crossTileID(t){this._structArray.uint32[this._pos4+10]=t}get associatedIconIndex(){return this._structArray.int16[this._pos2+22]}}vi.prototype.size=48;class bi extends ui{get(t){return new vi(this,t)}}Wr("PlacedSymbolArray",bi);class wi extends Nn{get anchorX(){return this._structArray.int16[this._pos2+0]}get anchorY(){return this._structArray.int16[this._pos2+1]}get rightJustifiedTextSymbolIndex(){return this._structArray.int16[this._pos2+2]}get centerJustifiedTextSymbolIndex(){return this._structArray.int16[this._pos2+3]}get leftJustifiedTextSymbolIndex(){return this._structArray.int16[this._pos2+4]}get verticalPlacedTextSymbolIndex(){return this._structArray.int16[this._pos2+5]}get placedIconSymbolIndex(){return this._structArray.int16[this._pos2+6]}get verticalPlacedIconSymbolIndex(){return this._structArray.int16[this._pos2+7]}get key(){return this._structArray.uint16[this._pos2+8]}get textBoxStartIndex(){return this._structArray.uint16[this._pos2+9]}get textBoxEndIndex(){return this._structArray.uint16[this._pos2+10]}get verticalTextBoxStartIndex(){return this._structArray.uint16[this._pos2+11]}get verticalTextBoxEndIndex(){return this._structArray.uint16[this._pos2+12]}get iconBoxStartIndex(){return this._structArray.uint16[this._pos2+13]}get iconBoxEndIndex(){return this._structArray.uint16[this._pos2+14]}get verticalIconBoxStartIndex(){return this._structArray.uint16[this._pos2+15]}get verticalIconBoxEndIndex(){return this._structArray.uint16[this._pos2+16]}get featureIndex(){return this._structArray.uint16[this._pos2+17]}get numHorizontalGlyphVertices(){return this._structArray.uint16[this._pos2+18]}get numVerticalGlyphVertices(){return this._structArray.uint16[this._pos2+19]}get numIconVertices(){return this._structArray.uint16[this._pos2+20]}get numVerticalIconVertices(){return this._structArray.uint16[this._pos2+21]}get useRuntimeCollisionCircles(){return this._structArray.uint16[this._pos2+22]}get crossTileID(){return this._structArray.uint32[this._pos4+12]}set crossTileID(t){this._structArray.uint32[this._pos4+12]=t}get textBoxScale(){return this._structArray.float32[this._pos4+13]}get collisionCircleDiameter(){return this._structArray.float32[this._pos4+14]}get textAnchorOffsetStartIndex(){return this._structArray.uint16[this._pos2+30]}get textAnchorOffsetEndIndex(){return this._structArray.uint16[this._pos2+31]}}wi.prototype.size=64;class _i extends ci{get(t){return new wi(this,t)}}Wr("SymbolInstanceArray",_i);class Si extends hi{getoffsetX(t){return this.float32[1*t+0]}}Wr("GlyphOffsetArray",Si);class ki extends Xn{getx(t){return this.int16[3*t+0]}gety(t){return this.int16[3*t+1]}gettileUnitDistanceFromAnchor(t){return this.int16[3*t+2]}}Wr("SymbolLineVertexArray",ki);class Ai extends Nn{get textAnchor(){return this._structArray.uint16[this._pos2+0]}get textOffset0(){return this._structArray.float32[this._pos4+1]}get textOffset1(){return this._structArray.float32[this._pos4+2]}}Ai.prototype.size=12;class Ii extends pi{get(t){return new Ai(this,t)}}Wr("TextAnchorOffsetArray",Ii);class zi extends Nn{get featureIndex(){return this._structArray.uint32[this._pos4+0]}get sourceLayerIndex(){return this._structArray.uint16[this._pos2+2]}get bucketIndex(){return this._structArray.uint16[this._pos2+3]}}zi.prototype.size=8;class Mi extends fi{get(t){return new zi(this,t)}}Wr("FeatureIndexArray",Mi);class Pi extends Kn{}class Ci extends Kn{}class Bi extends Kn{}class Vi extends Hn{}class Ei extends Wn{}class Fi extends Qn{}class Ti extends ti{}class Di extends ei{}class $i extends ri{}class Li extends ni{}class Oi extends si{}class Ui extends oi{}class qi extends li{}class ji extends di{}const Ri=Gn([{name:"a_pos",components:2,type:"Int16"}],4),{members:Ni}=Ri;class Zi{constructor(t=[]){this.segments=t}prepareSegment(t,e,r,n){let i=this.segments[this.segments.length-1];return t>Zi.MAX_VERTEX_ARRAY_LENGTH&&y(`Max vertices per segment is ${Zi.MAX_VERTEX_ARRAY_LENGTH}: bucket requested ${t}`),(!i||i.vertexLength+t>Zi.MAX_VERTEX_ARRAY_LENGTH||i.sortKey!==n)&&(i={vertexOffset:e.length,primitiveOffset:r.length,vertexLength:0,primitiveLength:0},void 0!==n&&(i.sortKey=n),this.segments.push(i)),i}get(){return this.segments}destroy(){for(const t of this.segments)for(const e in t.vaos)t.vaos[e].destroy()}static simpleSegment(t,e,r,n){return new Zi([{vertexOffset:t,primitiveOffset:e,vertexLength:r,primitiveLength:n,vaos:{},sortKey:0}])}}function Gi(t,e){return 256*(t=c(Math.floor(t),0,255))+c(Math.floor(e),0,255)}Zi.MAX_VERTEX_ARRAY_LENGTH=Math.pow(2,16)-1,Wr("SegmentVector",Zi);const Ji=Gn([{name:"a_pattern_from",components:4,type:"Uint16"},{name:"a_pattern_to",components:4,type:"Uint16"},{name:"a_pixel_ratio_from",components:1,type:"Uint16"},{name:"a_pixel_ratio_to",components:1,type:"Uint16"}]);var Ki={exports:{}},Xi={exports:{}};Xi.exports=function(t,e){var r,n,i,s,a,o,l,u;for(n=t.length-(r=3&t.length),i=e,a=3432918353,o=461845907,u=0;u>>16)*a&65535)<<16)&4294967295)<<15|l>>>17))*o+(((l>>>16)*o&65535)<<16)&4294967295)<<13|i>>>19))+((5*(i>>>16)&65535)<<16)&4294967295))+((58964+(s>>>16)&65535)<<16);switch(l=0,r){case 3:l^=(255&t.charCodeAt(u+2))<<16;case 2:l^=(255&t.charCodeAt(u+1))<<8;case 1:i^=l=(65535&(l=(l=(65535&(l^=255&t.charCodeAt(u)))*a+(((l>>>16)*a&65535)<<16)&4294967295)<<15|l>>>17))*o+(((l>>>16)*o&65535)<<16)&4294967295}return i^=t.length,i=2246822507*(65535&(i^=i>>>16))+((2246822507*(i>>>16)&65535)<<16)&4294967295,i=3266489909*(65535&(i^=i>>>13))+((3266489909*(i>>>16)&65535)<<16)&4294967295,(i^=i>>>16)>>>0};var Yi=Xi.exports,Hi={exports:{}};Hi.exports=function(t,e){for(var r,n=t.length,i=e^n,s=0;n>=4;)r=1540483477*(65535&(r=255&t.charCodeAt(s)|(255&t.charCodeAt(++s))<<8|(255&t.charCodeAt(++s))<<16|(255&t.charCodeAt(++s))<<24))+((1540483477*(r>>>16)&65535)<<16),i=1540483477*(65535&i)+((1540483477*(i>>>16)&65535)<<16)^(r=1540483477*(65535&(r^=r>>>24))+((1540483477*(r>>>16)&65535)<<16)),n-=4,++s;switch(n){case 3:i^=(255&t.charCodeAt(s+2))<<16;case 2:i^=(255&t.charCodeAt(s+1))<<8;case 1:i=1540483477*(65535&(i^=255&t.charCodeAt(s)))+((1540483477*(i>>>16)&65535)<<16)}return i=1540483477*(65535&(i^=i>>>13))+((1540483477*(i>>>16)&65535)<<16),(i^=i>>>15)>>>0};var Wi=Yi,Qi=Hi.exports;Ki.exports=Wi,Ki.exports.murmur3=Wi,Ki.exports.murmur2=Qi;var ts=e(Ki.exports);class es{constructor(){this.ids=[],this.positions=[],this.indexed=!1}add(t,e,r,n){this.ids.push(rs(t)),this.positions.push(e,r,n)}getPositions(t){if(!this.indexed)throw new Error("Trying to get index, but feature positions are not indexed");const e=rs(t);let r=0,n=this.ids.length-1;for(;r>1;this.ids[t]>=e?n=t:r=t+1}const i=[];for(;this.ids[r]===e;)i.push({index:this.positions[3*r],start:this.positions[3*r+1],end:this.positions[3*r+2]}),r++;return i}static serialize(t,e){const r=new Float64Array(t.ids),n=new Uint32Array(t.positions);return ns(r,n,0,r.length-1),e&&e.push(r.buffer,n.buffer),{ids:r,positions:n}}static deserialize(t){const e=new es;return e.ids=t.ids,e.positions=t.positions,e.indexed=!0,e}}function rs(t){const e=+t;return!isNaN(e)&&e<=Number.MAX_SAFE_INTEGER?e:ts(String(t))}function ns(t,e,r,n){for(;r>1];let s=r-1,a=n+1;for(;;){do{s++}while(t[s]i);if(s>=a)break;is(t,s,a),is(e,3*s,3*a),is(e,3*s+1,3*a+1),is(e,3*s+2,3*a+2)}a-r`u_${t}`)),this.type=r}setUniform(t,e,r){t.set(r.constantOr(this.value))}getBinding(t,e,r){return"color"===this.type?new ls(t,e):new as(t,e)}}class hs{constructor(t,e){this.uniformNames=e.map((t=>`u_${t}`)),this.patternFrom=null,this.patternTo=null,this.pixelRatioFrom=1,this.pixelRatioTo=1}setConstantPatternPositions(t,e){this.pixelRatioFrom=e.pixelRatio,this.pixelRatioTo=t.pixelRatio,this.patternFrom=e.tlbr,this.patternTo=t.tlbr}setUniform(t,e,r,n){const i="u_pattern_to"===n?this.patternTo:"u_pattern_from"===n?this.patternFrom:"u_pixel_ratio_to"===n?this.pixelRatioTo:"u_pixel_ratio_from"===n?this.pixelRatioFrom:null;i&&t.set(i)}getBinding(t,e,r){return"u_pattern"===r.substr(0,9)?new os(t,e):new as(t,e)}}class ps{constructor(t,e,r,n){this.expression=t,this.type=r,this.maxValue=0,this.paintVertexAttributes=e.map((t=>({name:`a_${t}`,type:"Float32",components:"color"===r?2:1,offset:0}))),this.paintVertexArray=new n}populatePaintArray(t,e,r,n,i){const s=this.paintVertexArray.length,a=this.expression.evaluate(new In(0),e,{},n,[],i);this.paintVertexArray.resize(t),this._setPaintValue(s,t,a)}updatePaintArray(t,e,r,n){const i=this.expression.evaluate({zoom:0},r,n);this._setPaintValue(t,e,i)}_setPaintValue(t,e,r){if("color"===this.type){const n=us(r);for(let r=t;r`u_${t}_t`)),this.type=r,this.useIntegerZoom=n,this.zoom=i,this.maxValue=0,this.paintVertexAttributes=e.map((t=>({name:`a_${t}`,type:"Float32",components:"color"===r?4:2,offset:0}))),this.paintVertexArray=new s}populatePaintArray(t,e,r,n,i){const s=this.expression.evaluate(new In(this.zoom),e,{},n,[],i),a=this.expression.evaluate(new In(this.zoom+1),e,{},n,[],i),o=this.paintVertexArray.length;this.paintVertexArray.resize(t),this._setPaintValue(o,t,s,a)}updatePaintArray(t,e,r,n){const i=this.expression.evaluate({zoom:this.zoom},r,n),s=this.expression.evaluate({zoom:this.zoom+1},r,n);this._setPaintValue(t,e,i,s)}_setPaintValue(t,e,r,n){if("color"===this.type){const i=us(r),s=us(n);for(let r=t;r`#define HAS_UNIFORM_${t}`)))}return t}getBinderAttributes(){const t=[];for(const e in this.binders){const r=this.binders[e];if(r instanceof ps||r instanceof fs)for(let e=0;e!0)){this.programConfigurations={};for(const n of t)this.programConfigurations[n.id]=new ys(n,e,r);this.needsUpload=!1,this._featureMap=new es,this._bufferOffset=0}populatePaintArrays(t,e,r,n,i,s){for(const r in this.programConfigurations)this.programConfigurations[r].populatePaintArrays(t,e,n,i,s);void 0!==e.id&&this._featureMap.add(e.id,r,this._bufferOffset,t),this._bufferOffset=t,this.needsUpload=!0}updatePaintArrays(t,e,r,n){for(const i of r)this.needsUpload=this.programConfigurations[i.id].updatePaintArrays(t,this._featureMap,e,i,n)||this.needsUpload}get(t){return this.programConfigurations[t]}upload(t){if(this.needsUpload){for(const e in this.programConfigurations)this.programConfigurations[e].upload(t);this.needsUpload=!1}}destroy(){for(const t in this.programConfigurations)this.programConfigurations[t].destroy()}}function gs(t,e){return{"text-opacity":["opacity"],"icon-opacity":["opacity"],"text-color":["fill_color"],"icon-color":["fill_color"],"text-halo-color":["halo_color"],"icon-halo-color":["halo_color"],"text-halo-blur":["halo_blur"],"icon-halo-blur":["halo_blur"],"text-halo-width":["halo_width"],"icon-halo-width":["halo_width"],"line-gap-width":["gapwidth"],"line-pattern":["pattern_to","pattern_from","pixel_ratio_to","pixel_ratio_from"],"fill-pattern":["pattern_to","pattern_from","pixel_ratio_to","pixel_ratio_from"],"fill-extrusion-pattern":["pattern_to","pattern_from","pixel_ratio_to","pixel_ratio_from"]}[t]||[t.replace(`${e}-`,"").replace(/-/g,"_")]}function xs(t,e,r){const n={color:{source:Qn,composite:mi},number:{source:hi,composite:Qn}},i=function(t){return{"line-pattern":{source:Ti,composite:Ti},"fill-pattern":{source:Ti,composite:Ti},"fill-extrusion-pattern":{source:Ti,composite:Ti}}[t]}(t);return i&&i[r]||n[e][r]}Wr("ConstantBinder",cs),Wr("CrossFadedConstantBinder",hs),Wr("SourceExpressionBinder",ps),Wr("CrossFadedCompositeBinder",ds),Wr("CompositeExpressionBinder",fs),Wr("ProgramConfiguration",ys,{omit:["_buffers"]}),Wr("ProgramConfigurationSet",ms);const vs=8192,bs=Math.pow(2,14)-1,ws=-bs-1;function _s(t){const e=vs/t.extent,r=t.loadGeometry();for(let t=0;tr.x+1||sr.y+1)&&y("Geometry exceeds allowed extent, reduce your vector tile buffer size")}}return r}function Ss(t,e){return{type:t.type,id:t.id,properties:t.properties,geometry:e?_s(t):[]}}function ks(t,e,r,n,i){t.emplaceBack(2*e+(n+1)/2,2*r+(i+1)/2)}class As{constructor(t){this.zoom=t.zoom,this.overscaling=t.overscaling,this.layers=t.layers,this.layerIds=this.layers.map((t=>t.id)),this.index=t.index,this.hasPattern=!1,this.layoutVertexArray=new Ci,this.indexArray=new qi,this.segments=new Zi,this.programConfigurations=new ms(t.layers,t.zoom),this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id))}populate(t,e,r){const n=this.layers[0],i=[];let s=null,a=!1;"circle"===n.type&&(s=n.layout.get("circle-sort-key"),a=!s.isConstant());for(const{feature:e,id:n,index:o,sourceLayerIndex:l}of t){const t=this.layers[0]._featureFilter.needGeometry,u=Ss(e,t);if(!this.layers[0]._featureFilter.filter(new In(this.zoom),u,r))continue;const c=a?s.evaluate(u,{},r):void 0,h={id:n,properties:e.properties,type:e.type,sourceLayerIndex:l,index:o,geometry:t?u.geometry:_s(e),patterns:{},sortKey:c};i.push(h)}a&&i.sort(((t,e)=>t.sortKey-e.sortKey));for(const n of i){const{geometry:i,index:s,sourceLayerIndex:a}=n,o=t[s].feature;this.addFeature(n,i,s,r),e.featureIndex.insert(o,i,s,a,this.index)}}update(t,e,r){this.stateDependentLayers.length&&this.programConfigurations.updatePaintArrays(t,e,this.stateDependentLayers,r)}isEmpty(){return 0===this.layoutVertexArray.length}uploadPending(){return!this.uploaded||this.programConfigurations.needsUpload}upload(t){this.uploaded||(this.layoutVertexBuffer=t.createVertexBuffer(this.layoutVertexArray,Ni),this.indexBuffer=t.createIndexBuffer(this.indexArray)),this.programConfigurations.upload(t),this.uploaded=!0}destroy(){this.layoutVertexBuffer&&(this.layoutVertexBuffer.destroy(),this.indexBuffer.destroy(),this.programConfigurations.destroy(),this.segments.destroy())}addFeature(t,e,r,n){for(const r of e)for(const e of r){const r=e.x,n=e.y;if(r<0||r>=vs||n<0||n>=vs)continue;const i=this.segments.prepareSegment(4,this.layoutVertexArray,this.indexArray,t.sortKey),s=i.vertexLength;ks(this.layoutVertexArray,r,n,-1,-1),ks(this.layoutVertexArray,r,n,1,-1),ks(this.layoutVertexArray,r,n,1,1),ks(this.layoutVertexArray,r,n,-1,1),this.indexArray.emplaceBack(s,s+1,s+2),this.indexArray.emplaceBack(s,s+3,s+2),i.vertexLength+=4,i.primitiveLength+=2}this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length,t,r,{},n)}}function Is(t,e){for(let r=0;r1){if(Cs(t,e))return!0;for(let n=0;n1?r:r.sub(e)._mult(i)._add(e))}function Fs(t,e){let r,n,i,s=!1;for(let a=0;ae.y!=i.y>e.y&&e.x<(i.x-n.x)*(e.y-n.y)/(i.y-n.y)+n.x&&(s=!s)}return s}function Ts(t,e){let r=!1;for(let n=0,i=t.length-1;ne.y!=a.y>e.y&&e.x<(a.x-s.x)*(e.y-s.y)/(a.y-s.y)+s.x&&(r=!r)}return r}function Ds(t,e,r){const n=r[0],i=r[2];if(t.xi.x&&e.x>i.x||t.yi.y&&e.y>i.y)return!1;const s=m(t,e,r[0]);return s!==m(t,e,r[1])||s!==m(t,e,r[2])||s!==m(t,e,r[3])}function $s(t,e,r){const n=e.paint.get(t).value;return"constant"===n.kind?n.value:r.programConfigurations.get(e.id).getMaxValue(t)}function Ls(t){return Math.sqrt(t[0]*t[0]+t[1]*t[1])}function Os(t,e,r,n,s){if(!e[0]&&!e[1])return t;const a=i.convert(e)._mult(s);"viewport"===r&&a._rotate(-n);const o=[];for(let e=0;eKs(t,e)))}(l,o),p=c?u*a:u;for(const t of n)for(const e of t){const t=c?e:Ks(e,o);let r=p;const n=Gs([],[e.x,e.y,0,1],o);if("viewport"===this.paint.get("circle-pitch-scale")&&"map"===this.paint.get("circle-pitch-alignment")?r*=n[3]/s.cameraToCenterDistance:"map"===this.paint.get("circle-pitch-scale")&&"viewport"===this.paint.get("circle-pitch-alignment")&&(r*=s.cameraToCenterDistance/n[3]),zs(h,t,r))return!0}return!1}}function Ks(t,e){const r=Gs([],[t.x,t.y,0,1],e);return new i(r[0]/r[3],r[1]/r[3])}class Xs extends As{}let Ys;Wr("HeatmapBucket",Xs,{omit:["layers"]});var Hs={get paint(){return Ys=Ys||new Un({"heatmap-radius":new Dn(S.paint_heatmap["heatmap-radius"]),"heatmap-weight":new Dn(S.paint_heatmap["heatmap-weight"]),"heatmap-intensity":new Tn(S.paint_heatmap["heatmap-intensity"]),"heatmap-color":new On(S.paint_heatmap["heatmap-color"]),"heatmap-opacity":new Tn(S.paint_heatmap["heatmap-opacity"])})}};function Ws(t,{width:e,height:r},n,i){if(i){if(i instanceof Uint8ClampedArray)i=new Uint8Array(i.buffer);else if(i.length!==e*r*n)throw new RangeError(`mismatched image size. expected: ${i.length} but got: ${e*r*n}`)}else i=new Uint8Array(e*r*n);return t.width=e,t.height=r,t.data=i,t}function Qs(t,{width:e,height:r},n){if(e===t.width&&r===t.height)return;const i=Ws({},{width:e,height:r},n);ta(t,i,{x:0,y:0},{x:0,y:0},{width:Math.min(t.width,e),height:Math.min(t.height,r)},n),t.width=e,t.height=r,t.data=i.data}function ta(t,e,r,n,i,s){if(0===i.width||0===i.height)return e;if(i.width>t.width||i.height>t.height||r.x>t.width-i.width||r.y>t.height-i.height)throw new RangeError("out of range source coordinates for image copy");if(i.width>e.width||i.height>e.height||n.x>e.width-i.width||n.y>e.height-i.height)throw new RangeError("out of range destination coordinates for image copy");const a=t.data,o=e.data;if(a===o)throw new Error("srcData equals dstData, so image is already copied");for(let l=0;l{e[t.evaluationKey]=s;const a=t.expression.evaluate(e);i.data[r+n+0]=Math.floor(255*a.r/a.a),i.data[r+n+1]=Math.floor(255*a.g/a.a),i.data[r+n+2]=Math.floor(255*a.b/a.a),i.data[r+n+3]=Math.floor(255*a.a)};if(t.clips)for(let e=0,i=0;e80*r){n=s=t[0],i=a=t[1];for(var d=r;ds&&(s=o),l>a&&(a=l);u=0!==(u=Math.max(s-n,a-i))?32767/u:0}return fa(p,f,r,n,i,u,0),f}function ha(t,e,r,n,i){var s,a;if(i===Ta(t,e,r,n)>0)for(s=e;s=e;s-=n)a=Va(s,t[s],t[s+1],a);return a&&Ia(a,a.next)&&(Ea(a),a=a.next),a}function pa(t,e){if(!t)return t;e||(e=t);var r,n=t;do{if(r=!1,n.steiner||!Ia(n,n.next)&&0!==Aa(n.prev,n,n.next))n=n.next;else{if(Ea(n),(n=e=n.prev)===n.next)break;r=!0}}while(r||n!==e);return e}function fa(t,e,r,n,i,s,a){if(t){!a&&s&&function(t,e,r,n){var i=t;do{0===i.z&&(i.z=wa(i.x,i.y,e,r,n)),i.prevZ=i.prev,i.nextZ=i.next,i=i.next}while(i!==t);i.prevZ.nextZ=null,i.prevZ=null,function(t){var e,r,n,i,s,a,o,l,u=1;do{for(r=t,t=null,s=null,a=0;r;){for(a++,n=r,o=0,e=0;e0||l>0&&n;)0!==o&&(0===l||!n||r.z<=n.z)?(i=r,r=r.nextZ,o--):(i=n,n=n.nextZ,l--),s?s.nextZ=i:t=i,i.prevZ=s,s=i;r=n}s.nextZ=null,u*=2}while(a>1)}(i)}(t,n,i,s);for(var o,l,u=t;t.prev!==t.next;)if(o=t.prev,l=t.next,s?ya(t,n,i,s):da(t))e.push(o.i/r|0),e.push(t.i/r|0),e.push(l.i/r|0),Ea(t),t=l.next,u=l.next;else if((t=l)===u){a?1===a?fa(t=ma(pa(t),e,r),e,r,n,i,s,2):2===a&&ga(t,e,r,n,i,s):fa(pa(t),e,r,n,i,s,1);break}}}function da(t){var e=t.prev,r=t,n=t.next;if(Aa(e,r,n)>=0)return!1;for(var i=e.x,s=r.x,a=n.x,o=e.y,l=r.y,u=n.y,c=is?i>a?i:a:s>a?s:a,f=o>l?o>u?o:u:l>u?l:u,d=n.next;d!==e;){if(d.x>=c&&d.x<=p&&d.y>=h&&d.y<=f&&Sa(i,o,s,l,a,u,d.x,d.y)&&Aa(d.prev,d,d.next)>=0)return!1;d=d.next}return!0}function ya(t,e,r,n){var i=t.prev,s=t,a=t.next;if(Aa(i,s,a)>=0)return!1;for(var o=i.x,l=s.x,u=a.x,c=i.y,h=s.y,p=a.y,f=ol?o>u?o:u:l>u?l:u,m=c>h?c>p?c:p:h>p?h:p,g=wa(f,d,e,r,n),x=wa(y,m,e,r,n),v=t.prevZ,b=t.nextZ;v&&v.z>=g&&b&&b.z<=x;){if(v.x>=f&&v.x<=y&&v.y>=d&&v.y<=m&&v!==i&&v!==a&&Sa(o,c,l,h,u,p,v.x,v.y)&&Aa(v.prev,v,v.next)>=0)return!1;if(v=v.prevZ,b.x>=f&&b.x<=y&&b.y>=d&&b.y<=m&&b!==i&&b!==a&&Sa(o,c,l,h,u,p,b.x,b.y)&&Aa(b.prev,b,b.next)>=0)return!1;b=b.nextZ}for(;v&&v.z>=g;){if(v.x>=f&&v.x<=y&&v.y>=d&&v.y<=m&&v!==i&&v!==a&&Sa(o,c,l,h,u,p,v.x,v.y)&&Aa(v.prev,v,v.next)>=0)return!1;v=v.prevZ}for(;b&&b.z<=x;){if(b.x>=f&&b.x<=y&&b.y>=d&&b.y<=m&&b!==i&&b!==a&&Sa(o,c,l,h,u,p,b.x,b.y)&&Aa(b.prev,b,b.next)>=0)return!1;b=b.nextZ}return!0}function ma(t,e,r){var n=t;do{var i=n.prev,s=n.next.next;!Ia(i,s)&&za(i,n,n.next,s)&&Ca(i,s)&&Ca(s,i)&&(e.push(i.i/r|0),e.push(n.i/r|0),e.push(s.i/r|0),Ea(n),Ea(n.next),n=t=s),n=n.next}while(n!==t);return pa(n)}function ga(t,e,r,n,i,s){var a=t;do{for(var o=a.next.next;o!==a.prev;){if(a.i!==o.i&&ka(a,o)){var l=Ba(a,o);return a=pa(a,a.next),l=pa(l,l.next),fa(a,e,r,n,i,s,0),void fa(l,e,r,n,i,s,0)}o=o.next}a=a.next}while(a!==t)}function xa(t,e){return t.x-e.x}function va(t,e){var r=function(t,e){var r,n=e,i=t.x,s=t.y,a=-1/0;do{if(s<=n.y&&s>=n.next.y&&n.next.y!==n.y){var o=n.x+(s-n.y)*(n.next.x-n.x)/(n.next.y-n.y);if(o<=i&&o>a&&(a=o,r=n.x=n.x&&n.x>=c&&i!==n.x&&Sa(sr.x||n.x===r.x&&ba(r,n)))&&(r=n,p=l)),n=n.next}while(n!==u);return r}(t,e);if(!r)return e;var n=Ba(r,t);return pa(n,n.next),pa(r,r.next)}function ba(t,e){return Aa(t.prev,t,e.prev)<0&&Aa(e.next,t,t.next)<0}function wa(t,e,r,n,i){return(t=1431655765&((t=858993459&((t=252645135&((t=16711935&((t=(t-r)*i|0)|t<<8))|t<<4))|t<<2))|t<<1))|(e=1431655765&((e=858993459&((e=252645135&((e=16711935&((e=(e-n)*i|0)|e<<8))|e<<4))|e<<2))|e<<1))<<1}function _a(t){var e=t,r=t;do{(e.x=(t-a)*(s-o)&&(t-a)*(n-o)>=(r-a)*(e-o)&&(r-a)*(s-o)>=(i-a)*(n-o)}function ka(t,e){return t.next.i!==e.i&&t.prev.i!==e.i&&!function(t,e){var r=t;do{if(r.i!==t.i&&r.next.i!==t.i&&r.i!==e.i&&r.next.i!==e.i&&za(r,r.next,t,e))return!0;r=r.next}while(r!==t);return!1}(t,e)&&(Ca(t,e)&&Ca(e,t)&&function(t,e){var r=t,n=!1,i=(t.x+e.x)/2,s=(t.y+e.y)/2;do{r.y>s!=r.next.y>s&&r.next.y!==r.y&&i<(r.next.x-r.x)*(s-r.y)/(r.next.y-r.y)+r.x&&(n=!n),r=r.next}while(r!==t);return n}(t,e)&&(Aa(t.prev,t,e.prev)||Aa(t,e.prev,e))||Ia(t,e)&&Aa(t.prev,t,t.next)>0&&Aa(e.prev,e,e.next)>0)}function Aa(t,e,r){return(e.y-t.y)*(r.x-e.x)-(e.x-t.x)*(r.y-e.y)}function Ia(t,e){return t.x===e.x&&t.y===e.y}function za(t,e,r,n){var i=Pa(Aa(t,e,r)),s=Pa(Aa(t,e,n)),a=Pa(Aa(r,n,t)),o=Pa(Aa(r,n,e));return i!==s&&a!==o||!(0!==i||!Ma(t,r,e))||!(0!==s||!Ma(t,n,e))||!(0!==a||!Ma(r,t,n))||!(0!==o||!Ma(r,e,n))}function Ma(t,e,r){return e.x<=Math.max(t.x,r.x)&&e.x>=Math.min(t.x,r.x)&&e.y<=Math.max(t.y,r.y)&&e.y>=Math.min(t.y,r.y)}function Pa(t){return t>0?1:t<0?-1:0}function Ca(t,e){return Aa(t.prev,t,t.next)<0?Aa(t,e,t.next)>=0&&Aa(t,t.prev,e)>=0:Aa(t,e,t.prev)<0||Aa(t,t.next,e)<0}function Ba(t,e){var r=new Fa(t.i,t.x,t.y),n=new Fa(e.i,e.x,e.y),i=t.next,s=e.prev;return t.next=e,e.prev=t,r.next=i,i.prev=r,n.next=r,r.prev=n,s.next=n,n.prev=s,n}function Va(t,e,r,n){var i=new Fa(t,e,r);return n?(i.next=n.next,i.prev=n,n.next.prev=i,n.next=i):(i.prev=i,i.next=i),i}function Ea(t){t.next.prev=t.prev,t.prev.next=t.next,t.prevZ&&(t.prevZ.nextZ=t.nextZ),t.nextZ&&(t.nextZ.prevZ=t.prevZ)}function Fa(t,e,r){this.i=t,this.x=e,this.y=r,this.prev=null,this.next=null,this.z=0,this.prevZ=null,this.nextZ=null,this.steiner=!1}function Ta(t,e,r,n){for(var i=0,s=e,a=r-n;s0&&r.holes.push(n+=t[i-1].length)}return r};var Da=e(ua.exports);function $a(t,e,r,n,i){La(t,e,r||0,n||t.length-1,i||Ua)}function La(t,e,r,n,i){for(;n>r;){if(n-r>600){var s=n-r+1,a=e-r+1,o=Math.log(s),l=.5*Math.exp(2*o/3),u=.5*Math.sqrt(o*l*(s-l)/s)*(a-s/2<0?-1:1);La(t,e,Math.max(r,Math.floor(e-a*l/s+u)),Math.min(n,Math.floor(e+(s-a)*l/s+u)),i)}var c=t[e],h=r,p=n;for(Oa(t,r,e),i(t[n],c)>0&&Oa(t,r,n);h0;)p--}0===i(t[r],c)?Oa(t,r,p):Oa(t,++p,n),p<=e&&(r=p+1),e<=p&&(n=p-1)}}function Oa(t,e,r){var n=t[e];t[e]=t[r],t[r]=n}function Ua(t,e){return te?1:0}function qa(t,e){const r=t.length;if(r<=1)return[t];const n=[];let i,s;for(let e=0;e1)for(let t=0;tt.id)),this.index=t.index,this.hasPattern=!1,this.patternFeatures=[],this.layoutVertexArray=new Bi,this.indexArray=new qi,this.indexArray2=new ji,this.programConfigurations=new ms(t.layers,t.zoom),this.segments=new Zi,this.segments2=new Zi,this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id))}populate(t,e,r){this.hasPattern=Ra("fill",this.layers,e);const n=this.layers[0].layout.get("fill-sort-key"),i=!n.isConstant(),s=[];for(const{feature:a,id:o,index:l,sourceLayerIndex:u}of t){const t=this.layers[0]._featureFilter.needGeometry,c=Ss(a,t);if(!this.layers[0]._featureFilter.filter(new In(this.zoom),c,r))continue;const h=i?n.evaluate(c,{},r,e.availableImages):void 0,p={id:o,properties:a.properties,type:a.type,sourceLayerIndex:u,index:l,geometry:t?c.geometry:_s(a),patterns:{},sortKey:h};s.push(p)}i&&s.sort(((t,e)=>t.sortKey-e.sortKey));for(const n of s){const{geometry:i,index:s,sourceLayerIndex:a}=n;if(this.hasPattern){const t=Na("fill",this.layers,n,this.zoom,e);this.patternFeatures.push(t)}else this.addFeature(n,i,s,r,{});e.featureIndex.insert(t[s].feature,i,s,a,this.index)}}update(t,e,r){this.stateDependentLayers.length&&this.programConfigurations.updatePaintArrays(t,e,this.stateDependentLayers,r)}addFeatures(t,e,r){for(const t of this.patternFeatures)this.addFeature(t,t.geometry,t.index,e,r)}isEmpty(){return 0===this.layoutVertexArray.length}uploadPending(){return!this.uploaded||this.programConfigurations.needsUpload}upload(t){this.uploaded||(this.layoutVertexBuffer=t.createVertexBuffer(this.layoutVertexArray,la),this.indexBuffer=t.createIndexBuffer(this.indexArray),this.indexBuffer2=t.createIndexBuffer(this.indexArray2)),this.programConfigurations.upload(t),this.uploaded=!0}destroy(){this.layoutVertexBuffer&&(this.layoutVertexBuffer.destroy(),this.indexBuffer.destroy(),this.indexBuffer2.destroy(),this.programConfigurations.destroy(),this.segments.destroy(),this.segments2.destroy())}addFeature(t,e,r,n,i){for(const t of qa(e,500)){let e=0;for(const r of t)e+=r.length;const r=this.segments.prepareSegment(e,this.layoutVertexArray,this.indexArray),n=r.vertexLength,i=[],s=[];for(const e of t){if(0===e.length)continue;e!==t[0]&&s.push(i.length/2);const r=this.segments2.prepareSegment(e.length,this.layoutVertexArray,this.indexArray2),n=r.vertexLength;this.layoutVertexArray.emplaceBack(e[0].x,e[0].y),this.indexArray2.emplaceBack(n+e.length-1,n),i.push(e[0].x),i.push(e[0].y);for(let t=1;t>3}if(i--,1===n||2===n)s+=t.readSVarint(),a+=t.readSVarint(),1===n&&(e&&o.push(e),e=[]),e.push(new to(s,a));else{if(7!==n)throw new Error("unknown command "+n);e&&e.push(e[0].clone())}}return e&&o.push(e),o},ro.prototype.bbox=function(){var t=this._pbf;t.pos=this._geometry;for(var e=t.readVarint()+t.pos,r=1,n=0,i=0,s=0,a=1/0,o=-1/0,l=1/0,u=-1/0;t.pos>3}if(n--,1===r||2===r)(i+=t.readSVarint())o&&(o=i),(s+=t.readSVarint())u&&(u=s);else if(7!==r)throw new Error("unknown command "+r)}return[a,l,o,u]},ro.prototype.toGeoJSON=function(t,e,r){var n,i,s=this.extent*Math.pow(2,r),a=this.extent*t,o=this.extent*e,l=this.loadGeometry(),u=ro.types[this.type];function c(t){for(var e=0;e>3;e=1===n?t.readString():2===n?t.readFloat():3===n?t.readDouble():4===n?t.readVarint64():5===n?t.readVarint():6===n?t.readSVarint():7===n?t.readBoolean():null}return e}(r))}oo.prototype.feature=function(t){if(t<0||t>=this._features.length)throw new Error("feature index out of bounds");this._pbf.pos=this._features[t];var e=this._pbf.readVarint()+this._pbf.pos;return new so(this._pbf,e,this.extent,this._keys,this._values)};var uo=ao;function co(t,e,r){if(3===t){var n=new uo(r,r.readVarint()+r.pos);n.length&&(e[n.name]=n)}}Qa.VectorTile=function(t,e){this.layers=t.readFields(co,{},e)},Qa.VectorTileFeature=eo,Qa.VectorTileLayer=ao;const ho=Qa.VectorTileFeature.types,po=Math.pow(2,13);function fo(t,e,r,n,i,s,a,o){t.emplaceBack(e,r,2*Math.floor(n*po)+a,i*po*2,s*po*2,Math.round(o))}class yo{constructor(t){this.zoom=t.zoom,this.overscaling=t.overscaling,this.layers=t.layers,this.layerIds=this.layers.map((t=>t.id)),this.index=t.index,this.hasPattern=!1,this.layoutVertexArray=new Vi,this.centroidVertexArray=new Pi,this.indexArray=new qi,this.programConfigurations=new ms(t.layers,t.zoom),this.segments=new Zi,this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id))}populate(t,e,r){this.features=[],this.hasPattern=Ra("fill-extrusion",this.layers,e);for(const{feature:n,id:i,index:s,sourceLayerIndex:a}of t){const t=this.layers[0]._featureFilter.needGeometry,o=Ss(n,t);if(!this.layers[0]._featureFilter.filter(new In(this.zoom),o,r))continue;const l={id:i,sourceLayerIndex:a,index:s,geometry:t?o.geometry:_s(n),properties:n.properties,type:n.type,patterns:{}};this.hasPattern?this.features.push(Na("fill-extrusion",this.layers,l,this.zoom,e)):this.addFeature(l,l.geometry,s,r,{}),e.featureIndex.insert(n,l.geometry,s,a,this.index,!0)}}addFeatures(t,e,r){for(const t of this.features){const{geometry:n}=t;this.addFeature(t,n,t.index,e,r)}}update(t,e,r){this.stateDependentLayers.length&&this.programConfigurations.updatePaintArrays(t,e,this.stateDependentLayers,r)}isEmpty(){return 0===this.layoutVertexArray.length&&0===this.centroidVertexArray.length}uploadPending(){return!this.uploaded||this.programConfigurations.needsUpload}upload(t){this.uploaded||(this.layoutVertexBuffer=t.createVertexBuffer(this.layoutVertexArray,Wa),this.centroidVertexBuffer=t.createVertexBuffer(this.centroidVertexArray,Ha.members,!0),this.indexBuffer=t.createIndexBuffer(this.indexArray)),this.programConfigurations.upload(t),this.uploaded=!0}destroy(){this.layoutVertexBuffer&&(this.layoutVertexBuffer.destroy(),this.indexBuffer.destroy(),this.programConfigurations.destroy(),this.segments.destroy(),this.centroidVertexBuffer.destroy())}addFeature(t,e,r,n,i){const s={x:0,y:0,vertexCount:0};for(const r of qa(e,500)){let e=0;for(const t of r)e+=t.length;let n=this.segments.prepareSegment(4,this.layoutVertexArray,this.indexArray);for(const t of r){if(0===t.length)continue;if(go(t))continue;let e=0;for(let r=0;r=1){const a=t[r-1];if(!mo(i,a)){n.vertexLength+4>Zi.MAX_VERTEX_ARRAY_LENGTH&&(n=this.segments.prepareSegment(4,this.layoutVertexArray,this.indexArray));const t=i.sub(a)._perp()._unit(),r=a.dist(i);e+r>32768&&(e=0),fo(this.layoutVertexArray,i.x,i.y,t.x,t.y,0,0,e),fo(this.layoutVertexArray,i.x,i.y,t.x,t.y,0,1,e),s.x+=2*i.x,s.y+=2*i.y,s.vertexCount+=2,e+=r,fo(this.layoutVertexArray,a.x,a.y,t.x,t.y,0,0,e),fo(this.layoutVertexArray,a.x,a.y,t.x,t.y,0,1,e),s.x+=2*a.x,s.y+=2*a.y,s.vertexCount+=2;const o=n.vertexLength;this.indexArray.emplaceBack(o,o+2,o+1),this.indexArray.emplaceBack(o+1,o+2,o+3),n.vertexLength+=4,n.primitiveLength+=2}}}}if(n.vertexLength+e>Zi.MAX_VERTEX_ARRAY_LENGTH&&(n=this.segments.prepareSegment(e,this.layoutVertexArray,this.indexArray)),"Polygon"!==ho[t.type])continue;const i=[],a=[],o=n.vertexLength;for(const t of r)if(0!==t.length){t!==r[0]&&a.push(i.length/2);for(let e=0;evs)||t.y===e.y&&(t.y<0||t.y>vs)}function go(t){return t.every((t=>t.x<0))||t.every((t=>t.x>vs))||t.every((t=>t.y<0))||t.every((t=>t.y>vs))}let xo;Wr("FillExtrusionBucket",yo,{omit:["layers","features"]});var vo={get paint(){return xo=xo||new Un({"fill-extrusion-opacity":new Tn(S["paint_fill-extrusion"]["fill-extrusion-opacity"]),"fill-extrusion-color":new Dn(S["paint_fill-extrusion"]["fill-extrusion-color"]),"fill-extrusion-translate":new Tn(S["paint_fill-extrusion"]["fill-extrusion-translate"]),"fill-extrusion-translate-anchor":new Tn(S["paint_fill-extrusion"]["fill-extrusion-translate-anchor"]),"fill-extrusion-pattern":new $n(S["paint_fill-extrusion"]["fill-extrusion-pattern"]),"fill-extrusion-height":new Dn(S["paint_fill-extrusion"]["fill-extrusion-height"]),"fill-extrusion-base":new Dn(S["paint_fill-extrusion"]["fill-extrusion-base"]),"fill-extrusion-vertical-gradient":new Tn(S["paint_fill-extrusion"]["fill-extrusion-vertical-gradient"])})}};class bo extends jn{constructor(t){super(t,vo)}createBucket(t){return new yo(t)}queryRadius(){return Ls(this.paint.get("fill-extrusion-translate"))}is3D(){return!0}queryIntersectsFeature(t,e,r,n,s,a,o,l){const u=Os(t,this.paint.get("fill-extrusion-translate"),this.paint.get("fill-extrusion-translate-anchor"),a.angle,o),c=this.paint.get("fill-extrusion-height").evaluate(e,r),h=this.paint.get("fill-extrusion-base").evaluate(e,r),p=function(t,e,r,n){const s=[];for(const r of t){const t=[r.x,r.y,0,1];Gs(t,t,e),s.push(new i(t[0]/t[3],t[1]/t[3]))}return s}(u,l),f=function(t,e,r,n){const s=[],a=[],o=n[8]*e,l=n[9]*e,u=n[10]*e,c=n[11]*e,h=n[8]*r,p=n[9]*r,f=n[10]*r,d=n[11]*r;for(const e of t){const t=[],r=[];for(const s of e){const e=s.x,a=s.y,y=n[0]*e+n[4]*a+n[12],m=n[1]*e+n[5]*a+n[13],g=n[2]*e+n[6]*a+n[14],x=n[3]*e+n[7]*a+n[15],v=g+u,b=x+c,w=y+h,_=m+p,S=g+f,k=x+d,A=new i((y+o)/b,(m+l)/b);A.z=v/b,t.push(A);const I=new i(w/k,_/k);I.z=S/k,r.push(I)}s.push(t),a.push(r)}return[s,a]}(n,h,c,l);return function(t,e,r){let n=1/0;Ms(r,e)&&(n=_o(r,e[0]));for(let i=0;it.id)),this.index=t.index,this.hasPattern=!1,this.patternFeatures=[],this.lineClipsArray=[],this.gradients={},this.layers.forEach((t=>{this.gradients[t.id]={}})),this.layoutVertexArray=new Ei,this.layoutVertexArray2=new Fi,this.indexArray=new qi,this.programConfigurations=new ms(t.layers,t.zoom),this.segments=new Zi,this.maxLineLength=0,this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id))}populate(t,e,r){this.hasPattern=Ra("line",this.layers,e);const n=this.layers[0].layout.get("line-sort-key"),i=!n.isConstant(),s=[];for(const{feature:e,id:a,index:o,sourceLayerIndex:l}of t){const t=this.layers[0]._featureFilter.needGeometry,u=Ss(e,t);if(!this.layers[0]._featureFilter.filter(new In(this.zoom),u,r))continue;const c=i?n.evaluate(u,{},r):void 0,h={id:a,properties:e.properties,type:e.type,sourceLayerIndex:l,index:o,geometry:t?u.geometry:_s(e),patterns:{},sortKey:c};s.push(h)}i&&s.sort(((t,e)=>t.sortKey-e.sortKey));for(const n of s){const{geometry:i,index:s,sourceLayerIndex:a}=n;if(this.hasPattern){const t=Na("line",this.layers,n,this.zoom,e);this.patternFeatures.push(t)}else this.addFeature(n,i,s,r,{});e.featureIndex.insert(t[s].feature,i,s,a,this.index)}}update(t,e,r){this.stateDependentLayers.length&&this.programConfigurations.updatePaintArrays(t,e,this.stateDependentLayers,r)}addFeatures(t,e,r){for(const t of this.patternFeatures)this.addFeature(t,t.geometry,t.index,e,r)}isEmpty(){return 0===this.layoutVertexArray.length}uploadPending(){return!this.uploaded||this.programConfigurations.needsUpload}upload(t){this.uploaded||(0!==this.layoutVertexArray2.length&&(this.layoutVertexBuffer2=t.createVertexBuffer(this.layoutVertexArray2,Io)),this.layoutVertexBuffer=t.createVertexBuffer(this.layoutVertexArray,ko),this.indexBuffer=t.createIndexBuffer(this.indexArray)),this.programConfigurations.upload(t),this.uploaded=!0}destroy(){this.layoutVertexBuffer&&(this.layoutVertexBuffer.destroy(),this.indexBuffer.destroy(),this.programConfigurations.destroy(),this.segments.destroy())}lineFeatureClips(t){if(t.properties&&Object.prototype.hasOwnProperty.call(t.properties,"mapbox_clip_start")&&Object.prototype.hasOwnProperty.call(t.properties,"mapbox_clip_end"))return{start:+t.properties.mapbox_clip_start,end:+t.properties.mapbox_clip_end}}addFeature(t,e,r,n,i){const s=this.layers[0].layout,a=s.get("line-join").evaluate(t,{}),o=s.get("line-cap"),l=s.get("line-miter-limit"),u=s.get("line-round-limit");this.lineClips=this.lineFeatureClips(t);for(const r of e)this.addLine(r,t,a,o,l,u);this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length,t,r,i,n)}addLine(t,e,r,n,i,s){if(this.distance=0,this.scaledDistance=0,this.totalDistance=0,this.lineClips){this.lineClipsArray.push(this.lineClips);for(let e=0;e=2&&t[o-1].equals(t[o-2]);)o--;let l=0;for(;l0;if(w&&e>l){const t=h.dist(p);if(t>2*u){const e=h.sub(h.sub(p)._mult(u/t)._round());this.updateDistance(p,e),this.addCurrentVertex(e,d,0,0,c),p=e}}const S=p&&f;let k=S?r:a?"butt":n;if(S&&"round"===k&&(vi&&(k="bevel"),"bevel"===k&&(v>2&&(k="flipbevel"),v100)m=y.mult(-1);else{const t=v*d.add(y).mag()/d.sub(y).mag();m._perp()._mult(t*(_?-1:1))}this.addCurrentVertex(h,m,0,0,c),this.addCurrentVertex(h,m.mult(-1),0,0,c)}else if("bevel"===k||"fakeround"===k){const t=-Math.sqrt(v*v-1),e=_?t:0,r=_?0:t;if(p&&this.addCurrentVertex(h,d,e,r,c),"fakeround"===k){const t=Math.round(180*b/Math.PI/20);for(let e=1;e2*u){const e=h.add(f.sub(h)._mult(u/t)._round());this.updateDistance(h,e),this.addCurrentVertex(e,y,0,0,c),h=e}}}}addCurrentVertex(t,e,r,n,i,s=!1){const a=e.y*n-e.x,o=-e.y-e.x*n;this.addHalfVertex(t,e.x+e.y*r,e.y-e.x*r,s,!1,r,i),this.addHalfVertex(t,a,o,s,!0,-n,i),this.distance>Po/2&&0===this.totalDistance&&(this.distance=0,this.updateScaledDistance(),this.addCurrentVertex(t,e,r,n,i,s))}addHalfVertex({x:t,y:e},r,n,i,s,a,o){const l=.5*(this.lineClips?this.scaledDistance*(Po-1):this.scaledDistance);this.layoutVertexArray.emplaceBack((t<<1)+(i?1:0),(e<<1)+(s?1:0),Math.round(63*r)+128,Math.round(63*n)+128,1+(0===a?0:a<0?-1:1)|(63&l)<<2,l>>6),this.lineClips&&this.layoutVertexArray2.emplaceBack((this.scaledDistance-this.lineClips.start)/(this.lineClips.end-this.lineClips.start),this.lineClipsArray.length);const u=o.vertexLength++;this.e1>=0&&this.e2>=0&&(this.indexArray.emplaceBack(this.e1,this.e2,u),o.primitiveLength++),s?this.e2=u:this.e1=u}updateScaledDistance(){this.scaledDistance=this.lineClips?this.lineClips.start+(this.lineClips.end-this.lineClips.start)*this.distance/this.totalDistance:this.distance}updateDistance(t,e){this.distance+=t.dist(e),this.updateScaledDistance()}}let Bo,Vo;Wr("LineBucket",Co,{omit:["layers","patternFeatures"]});var Eo={get paint(){return Vo=Vo||new Un({"line-opacity":new Dn(S.paint_line["line-opacity"]),"line-color":new Dn(S.paint_line["line-color"]),"line-translate":new Tn(S.paint_line["line-translate"]),"line-translate-anchor":new Tn(S.paint_line["line-translate-anchor"]),"line-width":new Dn(S.paint_line["line-width"]),"line-gap-width":new Dn(S.paint_line["line-gap-width"]),"line-offset":new Dn(S.paint_line["line-offset"]),"line-blur":new Dn(S.paint_line["line-blur"]),"line-dasharray":new Ln(S.paint_line["line-dasharray"]),"line-pattern":new $n(S.paint_line["line-pattern"]),"line-gradient":new On(S.paint_line["line-gradient"])})},get layout(){return Bo=Bo||new Un({"line-cap":new Tn(S.layout_line["line-cap"]),"line-join":new Dn(S.layout_line["line-join"]),"line-miter-limit":new Tn(S.layout_line["line-miter-limit"]),"line-round-limit":new Tn(S.layout_line["line-round-limit"]),"line-sort-key":new Dn(S.layout_line["line-sort-key"])})}};class Fo extends Dn{possiblyEvaluate(t,e){return e=new In(Math.floor(e.zoom),{now:e.now,fadeDuration:e.fadeDuration,zoomHistory:e.zoomHistory,transition:e.transition}),super.possiblyEvaluate(t,e)}evaluate(t,e,r,n){return e=h({},e,{zoom:Math.floor(e.zoom)}),super.evaluate(t,e,r,n)}}let To;class Do extends jn{constructor(t){super(t,Eo),this.gradientVersion=0,To||(To=new Fo(Eo.paint.properties["line-width"].specification),To.useIntegerZoom=!0)}_handleSpecialPaintPropertyUpdate(t){if("line-gradient"===t){const t=this.gradientExpression();this.stepInterpolant=!!function(t){return void 0!==t._styleExpression}(t)&&t._styleExpression.expression instanceof ae,this.gradientVersion=(this.gradientVersion+1)%Number.MAX_SAFE_INTEGER}}gradientExpression(){return this._transitionablePaint._values["line-gradient"].value.expression}recalculate(t,e){super.recalculate(t,e),this.paint._values["line-floorwidth"]=To.possiblyEvaluate(this._transitioningPaint._values["line-width"].value,t)}createBucket(t){return new Co(t)}queryRadius(t){const e=t,r=$o($s("line-width",this,e),$s("line-gap-width",this,e)),n=$s("line-offset",this,e);return r/2+Math.abs(n)+Ls(this.paint.get("line-translate"))}queryIntersectsFeature(t,e,r,n,s,a,o){const l=Os(t,this.paint.get("line-translate"),this.paint.get("line-translate-anchor"),a.angle,o),u=o/2*$o(this.paint.get("line-width").evaluate(e,r),this.paint.get("line-gap-width").evaluate(e,r)),c=this.paint.get("line-offset").evaluate(e,r);return c&&(n=function(t,e){const r=[];for(let n=0;n=3)for(let e=0;e0?e+2*t:t}const Lo=Gn([{name:"a_pos_offset",components:4,type:"Int16"},{name:"a_data",components:4,type:"Uint16"},{name:"a_pixeloffset",components:4,type:"Int16"}],4),Oo=Gn([{name:"a_projected_pos",components:3,type:"Float32"}],4);Gn([{name:"a_fade_opacity",components:1,type:"Uint32"}],4);const Uo=Gn([{name:"a_placed",components:2,type:"Uint8"},{name:"a_shift",components:2,type:"Float32"}]);Gn([{type:"Int16",name:"anchorPointX"},{type:"Int16",name:"anchorPointY"},{type:"Int16",name:"x1"},{type:"Int16",name:"y1"},{type:"Int16",name:"x2"},{type:"Int16",name:"y2"},{type:"Uint32",name:"featureIndex"},{type:"Uint16",name:"sourceLayerIndex"},{type:"Uint16",name:"bucketIndex"}]);const qo=Gn([{name:"a_pos",components:2,type:"Int16"},{name:"a_anchor_pos",components:2,type:"Int16"},{name:"a_extrude",components:2,type:"Int16"}],4);function jo(t,e,r){return t.sections.forEach((t=>{t.text=function(t,e,r){const n=e.layout.get("text-transform").evaluate(r,{});return"uppercase"===n?t=t.toLocaleUpperCase():"lowercase"===n&&(t=t.toLocaleLowerCase()),An.applyArabicShaping&&(t=An.applyArabicShaping(t)),t}(t.text,e,r)})),t}Gn([{name:"a_pos",components:2,type:"Float32"},{name:"a_radius",components:1,type:"Float32"},{name:"a_flags",components:2,type:"Int16"}],4),Gn([{name:"triangle",components:3,type:"Uint16"}]),Gn([{type:"Int16",name:"anchorX"},{type:"Int16",name:"anchorY"},{type:"Uint16",name:"glyphStartIndex"},{type:"Uint16",name:"numGlyphs"},{type:"Uint32",name:"vertexStartIndex"},{type:"Uint32",name:"lineStartIndex"},{type:"Uint32",name:"lineLength"},{type:"Uint16",name:"segment"},{type:"Uint16",name:"lowerSize"},{type:"Uint16",name:"upperSize"},{type:"Float32",name:"lineOffsetX"},{type:"Float32",name:"lineOffsetY"},{type:"Uint8",name:"writingMode"},{type:"Uint8",name:"placedOrientation"},{type:"Uint8",name:"hidden"},{type:"Uint32",name:"crossTileID"},{type:"Int16",name:"associatedIconIndex"}]),Gn([{type:"Int16",name:"anchorX"},{type:"Int16",name:"anchorY"},{type:"Int16",name:"rightJustifiedTextSymbolIndex"},{type:"Int16",name:"centerJustifiedTextSymbolIndex"},{type:"Int16",name:"leftJustifiedTextSymbolIndex"},{type:"Int16",name:"verticalPlacedTextSymbolIndex"},{type:"Int16",name:"placedIconSymbolIndex"},{type:"Int16",name:"verticalPlacedIconSymbolIndex"},{type:"Uint16",name:"key"},{type:"Uint16",name:"textBoxStartIndex"},{type:"Uint16",name:"textBoxEndIndex"},{type:"Uint16",name:"verticalTextBoxStartIndex"},{type:"Uint16",name:"verticalTextBoxEndIndex"},{type:"Uint16",name:"iconBoxStartIndex"},{type:"Uint16",name:"iconBoxEndIndex"},{type:"Uint16",name:"verticalIconBoxStartIndex"},{type:"Uint16",name:"verticalIconBoxEndIndex"},{type:"Uint16",name:"featureIndex"},{type:"Uint16",name:"numHorizontalGlyphVertices"},{type:"Uint16",name:"numVerticalGlyphVertices"},{type:"Uint16",name:"numIconVertices"},{type:"Uint16",name:"numVerticalIconVertices"},{type:"Uint16",name:"useRuntimeCollisionCircles"},{type:"Uint32",name:"crossTileID"},{type:"Float32",name:"textBoxScale"},{type:"Float32",name:"collisionCircleDiameter"},{type:"Uint16",name:"textAnchorOffsetStartIndex"},{type:"Uint16",name:"textAnchorOffsetEndIndex"}]),Gn([{type:"Float32",name:"offsetX"}]),Gn([{type:"Int16",name:"x"},{type:"Int16",name:"y"},{type:"Int16",name:"tileUnitDistanceFromAnchor"}]),Gn([{type:"Uint16",name:"textAnchor"},{type:"Float32",components:2,name:"textOffset"}]);const Ro={"!":"︕","#":"#",$:"$","%":"%","&":"&","(":"︵",")":"︶","*":"*","+":"+",",":"︐","-":"︲",".":"・","/":"/",":":"︓",";":"︔","<":"︿","=":"=",">":"﹀","?":"︖","@":"@","[":"﹇","\\":"\","]":"﹈","^":"^",_:"︳","`":"`","{":"︷","|":"―","}":"︸","~":"~","¢":"¢","£":"£","¥":"¥","¦":"¦","¬":"¬","¯":" ̄","–":"︲","—":"︱","‘":"﹃","’":"﹄","“":"﹁","”":"﹂","…":"︙","‧":"・","₩":"₩","、":"︑","。":"︒","〈":"︿","〉":"﹀","《":"︽","》":"︾","「":"﹁","」":"﹂","『":"﹃","』":"﹄","【":"︻","】":"︼","〔":"︹","〕":"︺","〖":"︗","〗":"︘","!":"︕","(":"︵",")":"︶",",":"︐","-":"︲",".":"・",":":"︓",";":"︔","<":"︿",">":"﹀","?":"︖","[":"﹇","]":"﹈","_":"︳","{":"︷","|":"―","}":"︸","⦅":"︵","⦆":"︶","。":"︒","「":"﹁","」":"﹂"};var No=24,Zo=Ko,Go=function(t,e,r,n,i){var s,a,o=8*i-n-1,l=(1<>1,c=-7,h=r?i-1:0,p=r?-1:1,f=t[e+h];for(h+=p,s=f&(1<<-c)-1,f>>=-c,c+=o;c>0;s=256*s+t[e+h],h+=p,c-=8);for(a=s&(1<<-c)-1,s>>=-c,c+=n;c>0;a=256*a+t[e+h],h+=p,c-=8);if(0===s)s=1-u;else{if(s===l)return a?NaN:1/0*(f?-1:1);a+=Math.pow(2,n),s-=u}return(f?-1:1)*a*Math.pow(2,s-n)},Jo=function(t,e,r,n,i,s){var a,o,l,u=8*s-i-1,c=(1<>1,p=23===i?Math.pow(2,-24)-Math.pow(2,-77):0,f=n?0:s-1,d=n?1:-1,y=e<0||0===e&&1/e<0?1:0;for(e=Math.abs(e),isNaN(e)||e===1/0?(o=isNaN(e)?1:0,a=c):(a=Math.floor(Math.log(e)/Math.LN2),e*(l=Math.pow(2,-a))<1&&(a--,l*=2),(e+=a+h>=1?p/l:p*Math.pow(2,1-h))*l>=2&&(a++,l/=2),a+h>=c?(o=0,a=c):a+h>=1?(o=(e*l-1)*Math.pow(2,i),a+=h):(o=e*Math.pow(2,h-1)*Math.pow(2,i),a=0));i>=8;t[r+f]=255&o,f+=d,o/=256,i-=8);for(a=a<0;t[r+f]=255&a,f+=d,a/=256,u-=8);t[r+f-d]|=128*y};function Ko(t){this.buf=ArrayBuffer.isView&&ArrayBuffer.isView(t)?t:new Uint8Array(t||0),this.pos=0,this.type=0,this.length=this.buf.length}Ko.Varint=0,Ko.Fixed64=1,Ko.Bytes=2,Ko.Fixed32=5;var Xo=4294967296,Yo=1/Xo,Ho="undefined"==typeof TextDecoder?null:new TextDecoder("utf8");function Wo(t){return t.type===Ko.Bytes?t.readVarint()+t.pos:t.pos+1}function Qo(t,e,r){return r?4294967296*e+(t>>>0):4294967296*(e>>>0)+(t>>>0)}function tl(t,e,r){var n=e<=16383?1:e<=2097151?2:e<=268435455?3:Math.floor(Math.log(e)/(7*Math.LN2));r.realloc(n);for(var i=r.pos-1;i>=t;i--)r.buf[i+n]=r.buf[i]}function el(t,e){for(var r=0;r>>8,t[r+2]=e>>>16,t[r+3]=e>>>24}function pl(t,e){return(t[e]|t[e+1]<<8|t[e+2]<<16)+(t[e+3]<<24)}Ko.prototype={destroy:function(){this.buf=null},readFields:function(t,e,r){for(r=r||this.length;this.pos>3,s=this.pos;this.type=7&n,t(i,e,this),this.pos===s&&this.skip(n)}return e},readMessage:function(t,e){return this.readFields(t,e,this.readVarint()+this.pos)},readFixed32:function(){var t=cl(this.buf,this.pos);return this.pos+=4,t},readSFixed32:function(){var t=pl(this.buf,this.pos);return this.pos+=4,t},readFixed64:function(){var t=cl(this.buf,this.pos)+cl(this.buf,this.pos+4)*Xo;return this.pos+=8,t},readSFixed64:function(){var t=cl(this.buf,this.pos)+pl(this.buf,this.pos+4)*Xo;return this.pos+=8,t},readFloat:function(){var t=Go(this.buf,this.pos,!0,23,4);return this.pos+=4,t},readDouble:function(){var t=Go(this.buf,this.pos,!0,52,8);return this.pos+=8,t},readVarint:function(t){var e,r,n=this.buf;return e=127&(r=n[this.pos++]),r<128?e:(e|=(127&(r=n[this.pos++]))<<7,r<128?e:(e|=(127&(r=n[this.pos++]))<<14,r<128?e:(e|=(127&(r=n[this.pos++]))<<21,r<128?e:function(t,e,r){var n,i,s=r.buf;if(n=(112&(i=s[r.pos++]))>>4,i<128)return Qo(t,n,e);if(n|=(127&(i=s[r.pos++]))<<3,i<128)return Qo(t,n,e);if(n|=(127&(i=s[r.pos++]))<<10,i<128)return Qo(t,n,e);if(n|=(127&(i=s[r.pos++]))<<17,i<128)return Qo(t,n,e);if(n|=(127&(i=s[r.pos++]))<<24,i<128)return Qo(t,n,e);if(n|=(1&(i=s[r.pos++]))<<31,i<128)return Qo(t,n,e);throw new Error("Expected varint not more than 10 bytes")}(e|=(15&(r=n[this.pos]))<<28,t,this))))},readVarint64:function(){return this.readVarint(!0)},readSVarint:function(){var t=this.readVarint();return t%2==1?(t+1)/-2:t/2},readBoolean:function(){return Boolean(this.readVarint())},readString:function(){var t=this.readVarint()+this.pos,e=this.pos;return this.pos=t,t-e>=12&&Ho?function(t,e,r){return Ho.decode(t.subarray(e,r))}(this.buf,e,t):function(t,e,r){for(var n="",i=e;i239?4:l>223?3:l>191?2:1;if(i+c>r)break;1===c?l<128&&(u=l):2===c?128==(192&(s=t[i+1]))&&(u=(31&l)<<6|63&s)<=127&&(u=null):3===c?(a=t[i+2],128==(192&(s=t[i+1]))&&128==(192&a)&&((u=(15&l)<<12|(63&s)<<6|63&a)<=2047||u>=55296&&u<=57343)&&(u=null)):4===c&&(a=t[i+2],o=t[i+3],128==(192&(s=t[i+1]))&&128==(192&a)&&128==(192&o)&&((u=(15&l)<<18|(63&s)<<12|(63&a)<<6|63&o)<=65535||u>=1114112)&&(u=null)),null===u?(u=65533,c=1):u>65535&&(u-=65536,n+=String.fromCharCode(u>>>10&1023|55296),u=56320|1023&u),n+=String.fromCharCode(u),i+=c}return n}(this.buf,e,t)},readBytes:function(){var t=this.readVarint()+this.pos,e=this.buf.subarray(this.pos,t);return this.pos=t,e},readPackedVarint:function(t,e){if(this.type!==Ko.Bytes)return t.push(this.readVarint(e));var r=Wo(this);for(t=t||[];this.pos127;);else if(e===Ko.Bytes)this.pos=this.readVarint()+this.pos;else if(e===Ko.Fixed32)this.pos+=4;else{if(e!==Ko.Fixed64)throw new Error("Unimplemented type: "+e);this.pos+=8}},writeTag:function(t,e){this.writeVarint(t<<3|e)},realloc:function(t){for(var e=this.length||16;e268435455||t<0?function(t,e){var r,n;if(t>=0?(r=t%4294967296|0,n=t/4294967296|0):(n=~(-t/4294967296),4294967295^(r=~(-t%4294967296))?r=r+1|0:(r=0,n=n+1|0)),t>=0x10000000000000000||t<-0x10000000000000000)throw new Error("Given varint doesn't fit into 10 bytes");e.realloc(10),function(t,e,r){r.buf[r.pos++]=127&t|128,t>>>=7,r.buf[r.pos++]=127&t|128,t>>>=7,r.buf[r.pos++]=127&t|128,t>>>=7,r.buf[r.pos++]=127&t|128,r.buf[r.pos]=127&(t>>>=7)}(r,0,e),function(t,e){var r=(7&t)<<4;e.buf[e.pos++]|=r|((t>>>=3)?128:0),t&&(e.buf[e.pos++]=127&t|((t>>>=7)?128:0),t&&(e.buf[e.pos++]=127&t|((t>>>=7)?128:0),t&&(e.buf[e.pos++]=127&t|((t>>>=7)?128:0),t&&(e.buf[e.pos++]=127&t|((t>>>=7)?128:0),t&&(e.buf[e.pos++]=127&t)))))}(n,e)}(t,this):(this.realloc(4),this.buf[this.pos++]=127&t|(t>127?128:0),t<=127||(this.buf[this.pos++]=127&(t>>>=7)|(t>127?128:0),t<=127||(this.buf[this.pos++]=127&(t>>>=7)|(t>127?128:0),t<=127||(this.buf[this.pos++]=t>>>7&127))))},writeSVarint:function(t){this.writeVarint(t<0?2*-t-1:2*t)},writeBoolean:function(t){this.writeVarint(Boolean(t))},writeString:function(t){t=String(t),this.realloc(4*t.length),this.pos++;var e=this.pos;this.pos=function(t,e,r){for(var n,i,s=0;s55295&&n<57344){if(!i){n>56319||s+1===e.length?(t[r++]=239,t[r++]=191,t[r++]=189):i=n;continue}if(n<56320){t[r++]=239,t[r++]=191,t[r++]=189,i=n;continue}n=i-55296<<10|n-56320|65536,i=null}else i&&(t[r++]=239,t[r++]=191,t[r++]=189,i=null);n<128?t[r++]=n:(n<2048?t[r++]=n>>6|192:(n<65536?t[r++]=n>>12|224:(t[r++]=n>>18|240,t[r++]=n>>12&63|128),t[r++]=n>>6&63|128),t[r++]=63&n|128)}return r}(this.buf,t,this.pos);var r=this.pos-e;r>=128&&tl(e,r,this),this.pos=e-1,this.writeVarint(r),this.pos+=r},writeFloat:function(t){this.realloc(4),Jo(this.buf,t,this.pos,!0,23,4),this.pos+=4},writeDouble:function(t){this.realloc(8),Jo(this.buf,t,this.pos,!0,52,8),this.pos+=8},writeBytes:function(t){var e=t.length;this.writeVarint(e),this.realloc(e);for(var r=0;r=128&&tl(r,n,this),this.pos=r-1,this.writeVarint(n),this.pos+=n},writeMessage:function(t,e,r){this.writeTag(t,Ko.Bytes),this.writeRawMessage(e,r)},writePackedVarint:function(t,e){e.length&&this.writeMessage(t,el,e)},writePackedSVarint:function(t,e){e.length&&this.writeMessage(t,rl,e)},writePackedBoolean:function(t,e){e.length&&this.writeMessage(t,sl,e)},writePackedFloat:function(t,e){e.length&&this.writeMessage(t,nl,e)},writePackedDouble:function(t,e){e.length&&this.writeMessage(t,il,e)},writePackedFixed32:function(t,e){e.length&&this.writeMessage(t,al,e)},writePackedSFixed32:function(t,e){e.length&&this.writeMessage(t,ol,e)},writePackedFixed64:function(t,e){e.length&&this.writeMessage(t,ll,e)},writePackedSFixed64:function(t,e){e.length&&this.writeMessage(t,ul,e)},writeBytesField:function(t,e){this.writeTag(t,Ko.Bytes),this.writeBytes(e)},writeFixed32Field:function(t,e){this.writeTag(t,Ko.Fixed32),this.writeFixed32(e)},writeSFixed32Field:function(t,e){this.writeTag(t,Ko.Fixed32),this.writeSFixed32(e)},writeFixed64Field:function(t,e){this.writeTag(t,Ko.Fixed64),this.writeFixed64(e)},writeSFixed64Field:function(t,e){this.writeTag(t,Ko.Fixed64),this.writeSFixed64(e)},writeVarintField:function(t,e){this.writeTag(t,Ko.Varint),this.writeVarint(e)},writeSVarintField:function(t,e){this.writeTag(t,Ko.Varint),this.writeSVarint(e)},writeStringField:function(t,e){this.writeTag(t,Ko.Bytes),this.writeString(e)},writeFloatField:function(t,e){this.writeTag(t,Ko.Fixed32),this.writeFloat(e)},writeDoubleField:function(t,e){this.writeTag(t,Ko.Fixed64),this.writeDouble(e)},writeBooleanField:function(t,e){this.writeVarintField(t,Boolean(e))}};var fl=e(Zo);const dl=3;function yl(t){let e=0,r=0;for(const n of t)e+=n.w*n.h,r=Math.max(r,n.w);t.sort(((t,e)=>e.h-t.h));const n=[{x:0,y:0,w:Math.max(Math.ceil(Math.sqrt(e/.95)),r),h:1/0}];let i=0,s=0;for(const e of t)for(let t=n.length-1;t>=0;t--){const r=n[t];if(!(e.w>r.w||e.h>r.h)){if(e.x=r.x,e.y=r.y,s=Math.max(s,e.y+e.h),i=Math.max(i,e.x+e.w),e.w===r.w&&e.h===r.h){const e=n.pop();t=0&&r>=t&&kl[this.text.charCodeAt(r)];r--)e--;this.text=this.text.substring(t,e),this.sectionIndex=this.sectionIndex.slice(t,e)}substring(t,e){const r=new _l;return r.text=this.text.substring(t,e),r.sectionIndex=this.sectionIndex.slice(t,e),r.sections=this.sections,r}toString(){return this.text}getMaxScale(){return this.sectionIndex.reduce(((t,e)=>Math.max(t,this.sections[e].scale)),0)}addTextSection(t,e){this.text+=t.text,this.sections.push(wl.forText(t.scale,t.fontStack||e));const r=this.sections.length-1;for(let e=0;e=63743?null:++this.imageSectionID:(this.imageSectionID=57344,this.imageSectionID)}}function Sl(t,e,r,n,i,s,a,o,l,u,c,h,p,f,d,y){const m=_l.fromFeature(t,i);let g;h===vl.vertical&&m.verticalizePunctuation();const{processBidirectionalText:x,processStyledBidirectionalText:v}=An;if(x&&1===m.sections.length){g=[];const t=x(m.toString(),Bl(m,u,s,e,n,f,d));for(const e of t){const t=new _l;t.text=e,t.sections=m.sections;for(let r=0;r0&&s>w&&(w=s)}else{const t=r[d.fontStack],n=t&&t[m];if(n&&n.rect)_=n.rect,v=n.metrics;else{const t=e[d.fontStack],r=t&&t[m];if(!r)continue;v=r.metrics}g=(i-d.scale)*No}A?(t.verticalizable=!0,b.push({glyph:m,imageName:S,x:p,y:f+g,vertical:A,scale:d.scale,fontStack:d.fontStack,sectionIndex:y,metrics:v,rect:_}),p+=k*d.scale+u):(b.push({glyph:m,imageName:S,x:p,y:f+g,vertical:A,scale:d.scale,fontStack:d.fontStack,sectionIndex:y,metrics:v,rect:_}),p+=v.advance*d.scale+u)}0!==b.length&&(d=Math.max(p-u,d),El(b,0,b.length-1,m,w)),p=0;const _=s*i+w;v.lineOffset=Math.max(w,o),f+=_,y=Math.max(_,y),++g}var x;const v=f-bl,{horizontalAlign:b,verticalAlign:w}=Vl(a);(function(t,e,r,n,i,s,a,o,l){const u=(e-r)*i;let c=0;c=s!==a?-o*n-bl:(-n*l+.5)*a;for(const e of t)for(const t of e.positionedGlyphs)t.x+=u,t.y+=c})(t.positionedLines,m,b,w,d,y,s,v,i.length),t.top+=-w*v,t.bottom=t.top+v,t.left+=-b*d,t.right=t.left+d}(w,e,r,n,g,a,o,l,h,u,p,y),!function(t){for(const e of t)if(0!==e.positionedGlyphs.length)return!1;return!0}(b)&&w}const kl={9:!0,10:!0,11:!0,12:!0,13:!0,32:!0},Al={10:!0,32:!0,38:!0,40:!0,41:!0,43:!0,45:!0,47:!0,173:!0,183:!0,8203:!0,8208:!0,8211:!0,8231:!0};function Il(t,e,r,n,i,s){if(e.imageName){const t=n[e.imageName];return t?t.displaySize[0]*e.scale*No/s+i:0}{const n=r[e.fontStack],s=n&&n[t];return s?s.metrics.advance*e.scale+i:0}}function zl(t,e,r,n){const i=Math.pow(t-e,2);return n?t=0;let c=0;for(let r=0;rt.id)),this.index=t.index,this.pixelRatio=t.pixelRatio,this.sourceLayerIndex=t.sourceLayerIndex,this.hasPattern=!1,this.hasRTLText=!1,this.sortKeyRanges=[],this.collisionCircleArray=[],this.placementInvProjMatrix=Zs([]),this.placementViewportMatrix=Zs([]);const e=this.layers[0]._unevaluatedLayout._values;this.textSizeData=Ol(this.zoom,e["text-size"]),this.iconSizeData=Ol(this.zoom,e["icon-size"]);const r=this.layers[0].layout,n=r.get("symbol-sort-key"),i=r.get("symbol-z-order");this.canOverlap="never"!==Ul(r,"text-overlap","text-allow-overlap")||"never"!==Ul(r,"icon-overlap","icon-allow-overlap")||r.get("text-ignore-placement")||r.get("icon-ignore-placement"),this.sortFeaturesByKey="viewport-y"!==i&&!n.isConstant(),this.sortFeaturesByY=("viewport-y"===i||"auto"===i&&!this.sortFeaturesByKey)&&this.canOverlap,"point"===r.get("symbol-placement")&&(this.writingModes=r.get("text-writing-mode").map((t=>vl[t]))),this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id)),this.sourceID=t.sourceID}createArrays(){this.text=new Gl(new ms(this.layers,this.zoom,(t=>/^text/.test(t)))),this.icon=new Gl(new ms(this.layers,this.zoom,(t=>/^icon/.test(t)))),this.glyphOffsetArray=new Si,this.lineVertexArray=new ki,this.symbolInstances=new _i,this.textAnchorOffsets=new Ii}calculateGlyphDependencies(t,e,r,n,i){for(let s=0;s0)&&("constant"!==s.value.kind||s.value.value.length>0),u="constant"!==o.value.kind||!!o.value.value||Object.keys(o.parameters).length>0,c=i.get("symbol-sort-key");if(this.features=[],!l&&!u)return;const h=e.iconDependencies,p=e.glyphDependencies,f=e.availableImages,d=new In(this.zoom);for(const{feature:e,id:a,index:o,sourceLayerIndex:y}of t){const t=n._featureFilter.needGeometry,m=Ss(e,t);if(!n._featureFilter.filter(d,m,r))continue;let g,x;if(t||(m.geometry=_s(e)),l){const t=n.getValueAndResolveTokens("text-field",m,r,f),e=gt.factory(t);Zl(e)&&(this.hasRTLText=!0),(!this.hasRTLText||"unavailable"===Sn||this.hasRTLText&&An.isParsed())&&(g=jo(e,n,m))}if(u){const t=n.getValueAndResolveTokens("icon-image",m,r,f);x=t instanceof wt?t:wt.fromString(t)}if(!g&&!x)continue;const v=this.sortFeaturesByKey?c.evaluate(m,{},r):void 0;if(this.features.push({id:a,text:g,icon:x,index:o,sourceLayerIndex:y,geometry:m.geometry,properties:e.properties,type:ql[e.type],sortKey:v}),x&&(h[x.name]=!0),g){const t=s.evaluate(m,{},r).join(","),e="viewport"!==i.get("text-rotation-alignment")&&"point"!==i.get("symbol-placement");this.allowVerticalPlacement=this.writingModes&&this.writingModes.indexOf(vl.vertical)>=0;for(const r of g.sections)if(r.image)h[r.image.name]=!0;else{const n=yn(g.toString()),i=r.fontStack||t,s=p[i]=p[i]||{};this.calculateGlyphDependencies(r.text,s,e,this.allowVerticalPlacement,n)}}}"line"===i.get("symbol-placement")&&(this.features=function(t){const e={},r={},n=[];let i=0;function s(e){n.push(t[e]),i++}function a(t,e,i){const s=r[t];return delete r[t],r[e]=s,n[s].geometry[0].pop(),n[s].geometry[0]=n[s].geometry[0].concat(i[0]),s}function o(t,r,i){const s=e[r];return delete e[r],e[t]=s,n[s].geometry[0].shift(),n[s].geometry[0]=i[0].concat(n[s].geometry[0]),s}function l(t,e,r){const n=r?e[0][e[0].length-1]:e[0][0];return`${t}:${n.x}:${n.y}`}for(let u=0;ut.geometry))}(this.features)),this.sortFeaturesByKey&&this.features.sort(((t,e)=>t.sortKey-e.sortKey))}update(t,e,r){this.stateDependentLayers.length&&(this.text.programConfigurations.updatePaintArrays(t,e,this.layers,r),this.icon.programConfigurations.updatePaintArrays(t,e,this.layers,r))}isEmpty(){return 0===this.symbolInstances.length&&!this.hasRTLText}uploadPending(){return!this.uploaded||this.text.programConfigurations.needsUpload||this.icon.programConfigurations.needsUpload}upload(t){!this.uploaded&&this.hasDebugData()&&(this.textCollisionBox.upload(t),this.iconCollisionBox.upload(t)),this.text.upload(t,this.sortFeaturesByY,!this.uploaded,this.text.programConfigurations.needsUpload),this.icon.upload(t,this.sortFeaturesByY,!this.uploaded,this.icon.programConfigurations.needsUpload),this.uploaded=!0}destroyDebugData(){this.textCollisionBox.destroy(),this.iconCollisionBox.destroy()}destroy(){this.text.destroy(),this.icon.destroy(),this.hasDebugData()&&this.destroyDebugData()}addToLineVertexArray(t,e){const r=this.lineVertexArray.length;if(void 0!==t.segment){let r=t.dist(e[t.segment+1]),n=t.dist(e[t.segment]);const i={};for(let n=t.segment+1;n=0;r--)i[r]={x:e[r].x,y:e[r].y,tileUnitDistanceFromAnchor:n},r>0&&(n+=e[r-1].dist(e[r]));for(let t=0;t0}hasIconData(){return this.icon.segments.get().length>0}hasDebugData(){return this.textCollisionBox&&this.iconCollisionBox}hasTextCollisionBoxData(){return this.hasDebugData()&&this.textCollisionBox.segments.get().length>0}hasIconCollisionBoxData(){return this.hasDebugData()&&this.iconCollisionBox.segments.get().length>0}addIndicesForPlacedSymbol(t,e){const r=t.placedSymbolArray.get(e),n=r.vertexStartIndex+4*r.numGlyphs;for(let e=r.vertexStartIndex;en[t]-n[e]||i[e]-i[t])),s}addToSortKeyRanges(t,e){const r=this.sortKeyRanges[this.sortKeyRanges.length-1];r&&r.sortKey===e?r.symbolInstanceEnd=t+1:this.sortKeyRanges.push({sortKey:e,symbolInstanceStart:t,symbolInstanceEnd:t+1})}sortFeatures(t){if(this.sortFeaturesByY&&this.sortedAngle!==t&&!(this.text.segments.get().length>1||this.icon.segments.get().length>1)){this.symbolInstanceIndexes=this.getSortedSymbolIndexes(t),this.sortedAngle=t,this.text.indexArray.clear(),this.icon.indexArray.clear(),this.featureSortOrder=[];for(const t of this.symbolInstanceIndexes){const e=this.symbolInstances.get(t);this.featureSortOrder.push(e.featureIndex),[e.rightJustifiedTextSymbolIndex,e.centerJustifiedTextSymbolIndex,e.leftJustifiedTextSymbolIndex].forEach(((t,e,r)=>{t>=0&&r.indexOf(t)===e&&this.addIndicesForPlacedSymbol(this.text,t)})),e.verticalPlacedTextSymbolIndex>=0&&this.addIndicesForPlacedSymbol(this.text,e.verticalPlacedTextSymbolIndex),e.placedIconSymbolIndex>=0&&this.addIndicesForPlacedSymbol(this.icon,e.placedIconSymbolIndex),e.verticalPlacedIconSymbolIndex>=0&&this.addIndicesForPlacedSymbol(this.icon,e.verticalPlacedIconSymbolIndex)}this.text.indexBuffer&&this.text.indexBuffer.updateData(this.text.indexArray),this.icon.indexBuffer&&this.icon.indexBuffer.updateData(this.icon.indexArray)}}}let Xl,Yl;Wr("SymbolBucket",Kl,{omit:["layers","collisionBoxArray","features","compareText"]}),Kl.MAX_GLYPHS=65535,Kl.addDynamicAttributes=Nl;var Hl={get paint(){return Yl=Yl||new Un({"icon-opacity":new Dn(S.paint_symbol["icon-opacity"]),"icon-color":new Dn(S.paint_symbol["icon-color"]),"icon-halo-color":new Dn(S.paint_symbol["icon-halo-color"]),"icon-halo-width":new Dn(S.paint_symbol["icon-halo-width"]),"icon-halo-blur":new Dn(S.paint_symbol["icon-halo-blur"]),"icon-translate":new Tn(S.paint_symbol["icon-translate"]),"icon-translate-anchor":new Tn(S.paint_symbol["icon-translate-anchor"]),"text-opacity":new Dn(S.paint_symbol["text-opacity"]),"text-color":new Dn(S.paint_symbol["text-color"],{runtimeType:E,getOverride:t=>t.textColor,hasOverride:t=>!!t.textColor}),"text-halo-color":new Dn(S.paint_symbol["text-halo-color"]),"text-halo-width":new Dn(S.paint_symbol["text-halo-width"]),"text-halo-blur":new Dn(S.paint_symbol["text-halo-blur"]),"text-translate":new Tn(S.paint_symbol["text-translate"]),"text-translate-anchor":new Tn(S.paint_symbol["text-translate-anchor"])})},get layout(){return Xl=Xl||new Un({"symbol-placement":new Tn(S.layout_symbol["symbol-placement"]),"symbol-spacing":new Tn(S.layout_symbol["symbol-spacing"]),"symbol-avoid-edges":new Tn(S.layout_symbol["symbol-avoid-edges"]),"symbol-sort-key":new Dn(S.layout_symbol["symbol-sort-key"]),"symbol-z-order":new Tn(S.layout_symbol["symbol-z-order"]),"icon-allow-overlap":new Tn(S.layout_symbol["icon-allow-overlap"]),"icon-overlap":new Tn(S.layout_symbol["icon-overlap"]),"icon-ignore-placement":new Tn(S.layout_symbol["icon-ignore-placement"]),"icon-optional":new Tn(S.layout_symbol["icon-optional"]),"icon-rotation-alignment":new Tn(S.layout_symbol["icon-rotation-alignment"]),"icon-size":new Dn(S.layout_symbol["icon-size"]),"icon-text-fit":new Tn(S.layout_symbol["icon-text-fit"]),"icon-text-fit-padding":new Tn(S.layout_symbol["icon-text-fit-padding"]),"icon-image":new Dn(S.layout_symbol["icon-image"]),"icon-rotate":new Dn(S.layout_symbol["icon-rotate"]),"icon-padding":new Dn(S.layout_symbol["icon-padding"]),"icon-keep-upright":new Tn(S.layout_symbol["icon-keep-upright"]),"icon-offset":new Dn(S.layout_symbol["icon-offset"]),"icon-anchor":new Dn(S.layout_symbol["icon-anchor"]),"icon-pitch-alignment":new Tn(S.layout_symbol["icon-pitch-alignment"]),"text-pitch-alignment":new Tn(S.layout_symbol["text-pitch-alignment"]),"text-rotation-alignment":new Tn(S.layout_symbol["text-rotation-alignment"]),"text-field":new Dn(S.layout_symbol["text-field"]),"text-font":new Dn(S.layout_symbol["text-font"]),"text-size":new Dn(S.layout_symbol["text-size"]),"text-max-width":new Dn(S.layout_symbol["text-max-width"]),"text-line-height":new Tn(S.layout_symbol["text-line-height"]),"text-letter-spacing":new Dn(S.layout_symbol["text-letter-spacing"]),"text-justify":new Dn(S.layout_symbol["text-justify"]),"text-radial-offset":new Dn(S.layout_symbol["text-radial-offset"]),"text-variable-anchor":new Tn(S.layout_symbol["text-variable-anchor"]),"text-variable-anchor-offset":new Dn(S.layout_symbol["text-variable-anchor-offset"]),"text-anchor":new Dn(S.layout_symbol["text-anchor"]),"text-max-angle":new Tn(S.layout_symbol["text-max-angle"]),"text-writing-mode":new Tn(S.layout_symbol["text-writing-mode"]),"text-rotate":new Dn(S.layout_symbol["text-rotate"]),"text-padding":new Tn(S.layout_symbol["text-padding"]),"text-keep-upright":new Tn(S.layout_symbol["text-keep-upright"]),"text-transform":new Dn(S.layout_symbol["text-transform"]),"text-offset":new Dn(S.layout_symbol["text-offset"]),"text-allow-overlap":new Tn(S.layout_symbol["text-allow-overlap"]),"text-overlap":new Tn(S.layout_symbol["text-overlap"]),"text-ignore-placement":new Tn(S.layout_symbol["text-ignore-placement"]),"text-optional":new Tn(S.layout_symbol["text-optional"])})}};class Wl{constructor(t){if(void 0===t.property.overrides)throw new Error("overrides must be provided to instantiate FormatSectionOverride class");this.type=t.property.overrides?t.property.overrides.runtimeType:P,this.defaultValue=t}evaluate(t){if(t.formattedSection){const e=this.defaultValue.property.overrides;if(e&&e.hasOverride(t.formattedSection))return e.getOverride(t.formattedSection)}return t.feature&&t.featureState?this.defaultValue.evaluate(t.feature,t.featureState):this.defaultValue.property.specification.default}eachChild(t){this.defaultValue.isConstant()||t(this.defaultValue.value._styleExpression.expression)}outputDefined(){return!1}serialize(){return null}}Wr("FormatSectionOverride",Wl,{omit:["defaultValue"]});class Ql extends jn{constructor(t){super(t,Hl)}recalculate(t,e){if(super.recalculate(t,e),"auto"===this.layout.get("icon-rotation-alignment")&&(this.layout._values["icon-rotation-alignment"]="point"!==this.layout.get("symbol-placement")?"map":"viewport"),"auto"===this.layout.get("text-rotation-alignment")&&(this.layout._values["text-rotation-alignment"]="point"!==this.layout.get("symbol-placement")?"map":"viewport"),"auto"===this.layout.get("text-pitch-alignment")&&(this.layout._values["text-pitch-alignment"]="map"===this.layout.get("text-rotation-alignment")?"map":"viewport"),"auto"===this.layout.get("icon-pitch-alignment")&&(this.layout._values["icon-pitch-alignment"]=this.layout.get("icon-rotation-alignment")),"point"===this.layout.get("symbol-placement")){const t=this.layout.get("text-writing-mode");if(t){const e=[];for(const r of t)e.indexOf(r)<0&&e.push(r);this.layout._values["text-writing-mode"]=e}else this.layout._values["text-writing-mode"]=["horizontal"]}this._setPaintOverrides()}getValueAndResolveTokens(t,e,r,n){const i=this.layout.get(t).evaluate(e,{},r,n),s=this._unevaluatedLayout._values[t];return s.isDataDriven()||Qe(s.value)||!i?i:function(t,e){return e.replace(/{([^{}]+)}/g,((e,r)=>t&&r in t?String(t[r]):""))}(e.properties,i)}createBucket(t){return new Kl(t)}queryRadius(){return 0}queryIntersectsFeature(){throw new Error("Should take a different path in FeatureIndex")}_setPaintOverrides(){for(const t of Hl.paint.overridableProperties){if(!Ql.hasPaintOverride(this.layout,t))continue;const e=this.paint.get(t),r=new Wl(e),n=new We(r,e.property.specification);let i=null;i="constant"===e.value.kind||"source"===e.value.kind?new er("source",n):new rr("composite",n,e.value.zoomStops),this.paint._values[t]=new En(e.property,i,e.parameters)}}_handleOverridablePaintPropertyUpdate(t,e,r){return!(!this.layout||e.isDataDriven()||r.isDataDriven())&&Ql.hasPaintOverride(this.layout,t)}static hasPaintOverride(t,e){const r=t.get("text-field"),n=Hl.paint.properties[e];let i=!1;const s=t=>{for(const e of t)if(n.overrides&&n.overrides.hasOverride(e))return void(i=!0)};if("constant"===r.value.kind&&r.value.value instanceof gt)s(r.value.value.sections);else if("source"===r.value.kind){const t=e=>{i||(e instanceof It&&kt(e.value)===$?s(e.value.sections):e instanceof Ce?s(e.sections):e.eachChild(t))},e=r.value;e._styleExpression&&t(e._styleExpression.expression)}return i}}let tu;var eu={get paint(){return tu=tu||new Un({"background-color":new Tn(S.paint_background["background-color"]),"background-pattern":new Ln(S.paint_background["background-pattern"]),"background-opacity":new Tn(S.paint_background["background-opacity"])})}};class ru extends jn{constructor(t){super(t,eu)}}let nu;var iu={get paint(){return nu=nu||new Un({"raster-opacity":new Tn(S.paint_raster["raster-opacity"]),"raster-hue-rotate":new Tn(S.paint_raster["raster-hue-rotate"]),"raster-brightness-min":new Tn(S.paint_raster["raster-brightness-min"]),"raster-brightness-max":new Tn(S.paint_raster["raster-brightness-max"]),"raster-saturation":new Tn(S.paint_raster["raster-saturation"]),"raster-contrast":new Tn(S.paint_raster["raster-contrast"]),"raster-resampling":new Tn(S.paint_raster["raster-resampling"]),"raster-fade-duration":new Tn(S.paint_raster["raster-fade-duration"])})}};class su extends jn{constructor(t){super(t,iu)}}class au extends jn{constructor(t){super(t,{}),this.onAdd=t=>{this.implementation.onAdd&&this.implementation.onAdd(t,t.painter.context.gl)},this.onRemove=t=>{this.implementation.onRemove&&this.implementation.onRemove(t,t.painter.context.gl)},this.implementation=t}is3D(){return"3d"===this.implementation.renderingMode}hasOffscreenPass(){return void 0!==this.implementation.prerender}recalculate(){}updateTransitions(){}hasTransition(){return!1}serialize(){throw new Error("Custom layers cannot be serialized")}}function ou(t){if("custom"===t.type)return new au(t);switch(t.type){case"background":return new ru(t);case"circle":return new Js(t);case"fill":return new Xa(t);case"fill-extrusion":return new bo(t);case"heatmap":return new na(t);case"hillshade":return new aa(t);case"line":return new Do(t);case"raster":return new su(t);case"symbol":return new Ql(t)}}class lu{constructor(t){this.keyCache={},t&&this.replace(t)}replace(t){this._layerConfigs={},this._layers={},this.update(t,[])}update(t,e){for(const e of t){this._layerConfigs[e.id]=e;const t=this._layers[e.id]=ou(e);t._featureFilter=lr(t.filter),this.keyCache[e.id]&&delete this.keyCache[e.id]}for(const t of e)delete this.keyCache[t],delete this._layerConfigs[t],delete this._layers[t];this.familiesBySource={};const r=function(t,e){const r={};for(let n=0;nthis._layers[t.id])),r=e[0];if("none"===r.visibility)continue;const n=r.source||"";let i=this.familiesBySource[n];i||(i=this.familiesBySource[n]={});const s=r.sourceLayer||"_geojsonTileLayer";let a=i[s];a||(a=i[s]=[]),a.push(e)}}}class uu{constructor(t){this._stringToNumber={},this._numberToString=[];for(let e=0;e=this._numberToString.length)throw new Error(`Out of bounds. Index requested n=${t} can't be >= this._numberToString.length ${this._numberToString.length}`);return this._numberToString[t]}}class cu{constructor(t,e,r,n,i){this.type="Feature",this._vectorTileFeature=t,t._z=e,t._x=r,t._y=n,this.properties=t.properties,this.id=i}get geometry(){return void 0===this._geometry&&(this._geometry=this._vectorTileFeature.toGeoJSON(this._vectorTileFeature._x,this._vectorTileFeature._y,this._vectorTileFeature._z).geometry),this._geometry}set geometry(t){this._geometry=t}toJSON(){const t={geometry:this.geometry};for(const e in this)"_geometry"!==e&&"_vectorTileFeature"!==e&&(t[e]=this[e]);return t}}class hu{constructor(t,e){this.tileID=t,this.x=t.canonical.x,this.y=t.canonical.y,this.z=t.canonical.z,this.grid=new _(vs,16,0),this.grid3D=new _(vs,16,0),this.featureIndexArray=new Mi,this.promoteId=e}insert(t,e,r,n,i,s){const a=this.featureIndexArray.length;this.featureIndexArray.emplaceBack(r,n,i);const o=s?this.grid3D:this.grid;for(let t=0;t=0&&n[3]>=0&&o.insert(a,n[0],n[1],n[2],n[3])}}loadVTLayers(){return this.vtLayers||(this.vtLayers=new Qa.VectorTile(new fl(this.rawTileData)).layers,this.sourceLayerCoder=new uu(this.vtLayers?Object.keys(this.vtLayers).sort():["_geojsonTileLayer"])),this.vtLayers}query(t,e,r,n){this.loadVTLayers();const s=t.params||{},a=vs/t.tileSize/t.scale,o=lr(s.filter),l=t.queryGeometry,u=t.queryPadding*a,c=fu(l),h=this.grid.query(c.minX-u,c.minY-u,c.maxX+u,c.maxY+u),p=fu(t.cameraQueryGeometry),f=this.grid3D.query(p.minX-u,p.minY-u,p.maxX+u,p.maxY+u,((e,r,n,s)=>function(t,e,r,n,s){for(const i of t)if(e<=i.x&&r<=i.y&&n>=i.x&&s>=i.y)return!0;const a=[new i(e,r),new i(e,s),new i(n,s),new i(n,r)];if(t.length>2)for(const e of a)if(Ts(t,e))return!0;for(let e=0;e(p||(p=_s(e)),r.queryIntersectsFeature(l,e,n,p,this.z,t.transform,a,t.pixelPosMatrix))))}return d}loadMatchingFeature(t,e,r,n,i,s,a,o,l,u,c){const p=this.bucketLayerIDs[e];if(s&&!function(t,e){for(let r=0;r=0)return!0;return!1}(s,p))return;const f=this.sourceLayerCoder.decode(r),d=this.vtLayers[f].feature(n);if(i.needGeometry){const t=Ss(d,!0);if(!i.filter(new In(this.tileID.overscaledZ),t,this.tileID.canonical))return}else if(!i.filter(new In(this.tileID.overscaledZ),d))return;const y=this.getId(d,f);for(let e=0;e{const a=e instanceof Fn?e.get(s):null;return a&&a.evaluate?a.evaluate(r,n,i):a}))}function fu(t){let e=1/0,r=1/0,n=-1/0,i=-1/0;for(const s of t)e=Math.min(e,s.x),r=Math.min(r,s.y),n=Math.max(n,s.x),i=Math.max(i,s.y);return{minX:e,minY:r,maxX:n,maxY:i}}function du(t,e){return e-t}Wr("FeatureIndex",hu,{omit:["rawTileData","sourceLayerCoder"]});class yu extends i{constructor(t,e,r,n){super(t,e),this.angle=r,void 0!==n&&(this.segment=n)}clone(){return new yu(this.x,this.y,this.angle,this.segment)}}function mu(t,e,r,n,i){if(void 0===e.segment||0===r)return!0;let s=e,a=e.segment+1,o=0;for(;o>-r/2;){if(a--,a<0)return!1;o-=t[a].dist(s),s=t[a]}o+=t[a].dist(t[a+1]),a++;const l=[];let u=0;for(;on;)u-=l.shift().angleDelta;if(u>i)return!1;a++,o+=e.dist(r)}return!0}function gu(t){let e=0;for(let r=0;ru){const c=(u-l)/s,h=ue.number(n.x,i.x,c),p=ue.number(n.y,i.y,c),f=new yu(h,p,i.angleTo(n),r);return f._round(),!a||mu(t,f,o,a,e)?f:void 0}l+=s}}function wu(t,e,r,n,i,s,a,o,l){const u=xu(n,s,a),c=vu(n,i),h=c*a,p=0===t[0].x||t[0].x===l||0===t[0].y||t[0].y===l;return e-h=0&&g=0&&x=0&&p+u<=c){const r=new yu(g,x,y,e);r._round(),n&&!mu(t,r,s,n,i)||f.push(r)}}h+=d}return o||f.length||a||(f=_u(t,h/2,r,n,i,s,a,!0,l)),f}Wr("Anchor",yu);const Su=ml;function ku(t,e,r,n){const s=[],a=t.image,o=a.pixelRatio,l=a.paddedRect.w-2*Su,u=a.paddedRect.h-2*Su,c=t.right-t.left,h=t.bottom-t.top,p=a.stretchX||[[0,l]],f=a.stretchY||[[0,u]],d=(t,e)=>t+e[1]-e[0],y=p.reduce(d,0),m=f.reduce(d,0),g=l-y,x=u-m;let v=0,b=y,w=0,_=m,S=0,k=g,A=0,I=x;if(a.content&&n){const t=a.content;v=Au(p,0,t[0]),w=Au(f,0,t[1]),b=Au(p,t[0],t[2]),_=Au(f,t[1],t[3]),S=t[0]-v,A=t[1]-w,k=t[2]-t[0]-b,I=t[3]-t[1]-_}const z=(n,s,l,u)=>{const p=zu(n.stretch-v,b,c,t.left),f=Mu(n.fixed-S,k,n.stretch,y),d=zu(s.stretch-w,_,h,t.top),g=Mu(s.fixed-A,I,s.stretch,m),x=zu(l.stretch-v,b,c,t.left),z=Mu(l.fixed-S,k,l.stretch,y),M=zu(u.stretch-w,_,h,t.top),P=Mu(u.fixed-A,I,u.stretch,m),C=new i(p,d),B=new i(x,d),V=new i(x,M),E=new i(p,M),F=new i(f/o,g/o),T=new i(z/o,P/o),D=e*Math.PI/180;if(D){const t=Math.sin(D),e=Math.cos(D),r=[e,-t,t,e];C._matMult(r),B._matMult(r),E._matMult(r),V._matMult(r)}const $=n.stretch+n.fixed,L=s.stretch+s.fixed;return{tl:C,tr:B,bl:E,br:V,tex:{x:a.paddedRect.x+Su+$,y:a.paddedRect.y+Su+L,w:l.stretch+l.fixed-$,h:u.stretch+u.fixed-L},writingMode:void 0,glyphOffset:[0,0],sectionIndex:0,pixelOffsetTL:F,pixelOffsetBR:T,minFontScaleX:k/o/c,minFontScaleY:I/o/h,isSDF:r}};if(n&&(a.stretchX||a.stretchY)){const t=Iu(p,g,y),e=Iu(f,x,m);for(let r=0;r0&&(n=Math.max(10,n),this.circleDiameter=n)}else{let u=a.top*o-l[0],h=a.bottom*o+l[2],p=a.left*o-l[3],f=a.right*o+l[1];const d=a.collisionPadding;if(d&&(p-=d[0]*o,u-=d[1]*o,f+=d[2]*o,h+=d[3]*o),c){const t=new i(p,u),e=new i(f,u),r=new i(p,h),n=new i(f,h),s=c*Math.PI/180;t._rotate(s),e._rotate(s),r._rotate(s),n._rotate(s),p=Math.min(t.x,e.x,r.x,n.x),f=Math.max(t.x,e.x,r.x,n.x),u=Math.min(t.y,e.y,r.y,n.y),h=Math.max(t.y,e.y,r.y,n.y)}t.emplaceBack(e.x,e.y,p,u,f,h,r,n,s)}this.boxEndIndex=t.length}}class Cu{constructor(t=[],e=Bu){if(this.data=t,this.length=this.data.length,this.compare=e,this.length>0)for(let t=(this.length>>1)-1;t>=0;t--)this._down(t)}push(t){this.data.push(t),this.length++,this._up(this.length-1)}pop(){if(0===this.length)return;const t=this.data[0],e=this.data.pop();return this.length--,this.length>0&&(this.data[0]=e,this._down(0)),t}peek(){return this.data[0]}_up(t){const{data:e,compare:r}=this,n=e[t];for(;t>0;){const i=t-1>>1,s=e[i];if(r(n,s)>=0)break;e[t]=s,t=i}e[t]=n}_down(t){const{data:e,compare:r}=this,n=this.length>>1,i=e[t];for(;t=0)break;e[t]=s,t=n}e[t]=i}}function Bu(t,e){return te?1:0}function Vu(t,e=1,r=!1){let n=1/0,s=1/0,a=-1/0,o=-1/0;const l=t[0];for(let t=0;ta)&&(a=e.x),(!t||e.y>o)&&(o=e.y)}const u=Math.min(a-n,o-s);let c=u/2;const h=new Cu([],Eu);if(0===u)return new i(n,s);for(let e=n;ep.d||!p.d)&&(p=n,r&&console.log("found best %d after %d probes",Math.round(1e4*n.d)/1e4,f)),n.max-p.d<=e||(c=n.h/2,h.push(new Fu(n.p.x-c,n.p.y-c,c,t)),h.push(new Fu(n.p.x+c,n.p.y-c,c,t)),h.push(new Fu(n.p.x-c,n.p.y+c,c,t)),h.push(new Fu(n.p.x+c,n.p.y+c,c,t)),f+=4)}return r&&(console.log(`num probes: ${f}`),console.log(`best distance: ${p.d}`)),p.p}function Eu(t,e){return e.max-t.max}function Fu(t,e,r,n){this.p=new i(t,e),this.h=r,this.d=function(t,e){let r=!1,n=1/0;for(let i=0;it.y!=o.y>t.y&&t.x<(o.x-i.x)*(t.y-i.y)/(o.y-i.y)+i.x&&(r=!r),n=Math.min(n,Es(t,i,o))}}return(r?1:-1)*Math.sqrt(n)}(this.p,n),this.max=this.d+this.h*Math.SQRT2}var Tu;!function(t){t[t.center=1]="center",t[t.left=2]="left",t[t.right=3]="right",t[t.top=4]="top",t[t.bottom=5]="bottom",t[t["top-left"]=6]="top-left",t[t["top-right"]=7]="top-right",t[t["bottom-left"]=8]="bottom-left",t[t["bottom-right"]=9]="bottom-right"}(Tu||(Tu={}));const Du=7,$u=Number.POSITIVE_INFINITY;function Lu(t,e){return e[1]!==$u?function(t,e,r){let n=0,i=0;switch(e=Math.abs(e),r=Math.abs(r),t){case"top-right":case"top-left":case"top":i=r-Du;break;case"bottom-right":case"bottom-left":case"bottom":i=-r+Du}switch(t){case"top-right":case"bottom-right":case"right":n=-e;break;case"top-left":case"bottom-left":case"left":n=e}return[n,i]}(t,e[0],e[1]):function(t,e){let r=0,n=0;e<0&&(e=0);const i=e/Math.SQRT2;switch(t){case"top-right":case"top-left":n=i-Du;break;case"bottom-right":case"bottom-left":n=-i+Du;break;case"bottom":n=-e+Du;break;case"top":n=e-Du}switch(t){case"top-right":case"bottom-right":r=-i;break;case"top-left":case"bottom-left":r=i;break;case"left":r=e;break;case"right":r=-e}return[r,n]}(t,e[0])}function Ou(t,e,r){var n;const i=t.layout,s=null===(n=i.get("text-variable-anchor-offset"))||void 0===n?void 0:n.evaluate(e,{},r);if(s){const t=s.values,e=[];for(let r=0;rt*No));n.startsWith("top")?i[1]-=Du:n.startsWith("bottom")&&(i[1]+=Du),e[r+1]=i}return new bt(e)}const a=i.get("text-variable-anchor");if(a){let n;n=void 0!==t._unevaluatedLayout.getValue("text-radial-offset")?[i.get("text-radial-offset").evaluate(e,{},r)*No,$u]:i.get("text-offset").evaluate(e,{},r).map((t=>t*No));const s=[];for(const t of a)s.push(t,Lu(t,n));return new bt(s)}return null}function Uu(t){t.bucket.createArrays(),t.bucket.tilePixelRatio=vs/(512*t.bucket.overscaling),t.bucket.compareText={},t.bucket.iconsNeedLinear=!1;const e=t.bucket.layers[0],r=e.layout,n=e._unevaluatedLayout._values,i={layoutIconSize:n["icon-size"].possiblyEvaluate(new In(t.bucket.zoom+1),t.canonical),layoutTextSize:n["text-size"].possiblyEvaluate(new In(t.bucket.zoom+1),t.canonical),textMaxSize:n["text-size"].possiblyEvaluate(new In(18))};if("composite"===t.bucket.textSizeData.kind){const{minZoom:e,maxZoom:r}=t.bucket.textSizeData;i.compositeTextSizes=[n["text-size"].possiblyEvaluate(new In(e),t.canonical),n["text-size"].possiblyEvaluate(new In(r),t.canonical)]}if("composite"===t.bucket.iconSizeData.kind){const{minZoom:e,maxZoom:r}=t.bucket.iconSizeData;i.compositeIconSizes=[n["icon-size"].possiblyEvaluate(new In(e),t.canonical),n["icon-size"].possiblyEvaluate(new In(r),t.canonical)]}const s=r.get("text-line-height")*No,a="viewport"!==r.get("text-rotation-alignment")&&"point"!==r.get("symbol-placement"),o=r.get("text-keep-upright"),l=r.get("text-size");for(const n of t.bucket.features){const u=r.get("text-font").evaluate(n,{},t.canonical).join(","),c=l.evaluate(n,{},t.canonical),h=i.layoutTextSize.evaluate(n,{},t.canonical),p=i.layoutIconSize.evaluate(n,{},t.canonical),f={horizontal:{},vertical:void 0},d=n.text;let m,g=[0,0];if(d){const i=d.toString(),l=r.get("text-letter-spacing").evaluate(n,{},t.canonical)*No,p=mn(i)?l:0,y=r.get("text-anchor").evaluate(n,{},t.canonical),m=Ou(e,n,t.canonical);if(!m){const e=r.get("text-radial-offset").evaluate(n,{},t.canonical);g=e?Lu(y,[e*No,$u]):r.get("text-offset").evaluate(n,{},t.canonical).map((t=>t*No))}let x=a?"center":r.get("text-justify").evaluate(n,{},t.canonical);const v=r.get("symbol-placement"),b="point"===v?r.get("text-max-width").evaluate(n,{},t.canonical)*No:0,w=()=>{t.bucket.allowVerticalPlacement&&yn(i)&&(f.vertical=Sl(d,t.glyphMap,t.glyphPositions,t.imagePositions,u,b,s,y,"left",p,g,vl.vertical,!0,v,h,c))};if(!a&&m){const e=new Set;if("auto"===x)for(let t=0;t{l.x<0||l.x>=vs||l.y<0||l.y>=vs||function(t,e,r,n,i,s,a,o,l,u,c,h,p,f,d,m,g,x,v,b,w,_,S,k,A){const I=t.addToLineVertexArray(e,r);let z,M,P,C,B=0,V=0,E=0,F=0,T=-1,D=-1;const $={};let L=ts("");if(t.allowVerticalPlacement&&n.vertical){const t=o.layout.get("text-rotate").evaluate(w,{},k)+90;P=new Pu(l,e,u,c,h,n.vertical,p,f,d,t),a&&(C=new Pu(l,e,u,c,h,a,g,x,d,t))}if(i){const r=o.layout.get("icon-rotate").evaluate(w,{}),n="none"!==o.layout.get("icon-text-fit"),s=ku(i,r,S,n),p=a?ku(a,r,S,n):void 0;M=new Pu(l,e,u,c,h,i,g,x,!1,r),B=4*s.length;const f=t.iconSizeData;let d=null;"source"===f.kind?(d=[$l*o.layout.get("icon-size").evaluate(w,{})],d[0]>Ll&&y(`${t.layerIds[0]}: Value for "icon-size" is >= ${Dl}. Reduce your "icon-size".`)):"composite"===f.kind&&(d=[$l*_.compositeIconSizes[0].evaluate(w,{},k),$l*_.compositeIconSizes[1].evaluate(w,{},k)],(d[0]>Ll||d[1]>Ll)&&y(`${t.layerIds[0]}: Value for "icon-size" is >= ${Dl}. Reduce your "icon-size".`)),t.addSymbols(t.icon,s,d,b,v,w,vl.none,e,I.lineStartIndex,I.lineLength,-1,k),T=t.icon.placedSymbolArray.length-1,p&&(V=4*p.length,t.addSymbols(t.icon,p,d,b,v,w,vl.vertical,e,I.lineStartIndex,I.lineLength,-1,k),D=t.icon.placedSymbolArray.length-1)}const O=Object.keys(n.horizontal);for(const r of O){const i=n.horizontal[r];if(!z){L=ts(i.text);const t=o.layout.get("text-rotate").evaluate(w,{},k);z=new Pu(l,e,u,c,h,i,p,f,d,t)}const a=1===i.positionedLines.length;if(E+=Ru(t,e,i,s,o,d,w,m,I,n.vertical?vl.horizontal:vl.horizontalOnly,a?O:[r],$,T,_,k),a)break}n.vertical&&(F+=Ru(t,e,n.vertical,s,o,d,w,m,I,vl.vertical,["vertical"],$,D,_,k));const U=z?z.boxStartIndex:t.collisionBoxArray.length,q=z?z.boxEndIndex:t.collisionBoxArray.length,j=P?P.boxStartIndex:t.collisionBoxArray.length,R=P?P.boxEndIndex:t.collisionBoxArray.length,N=M?M.boxStartIndex:t.collisionBoxArray.length,Z=M?M.boxEndIndex:t.collisionBoxArray.length,G=C?C.boxStartIndex:t.collisionBoxArray.length,J=C?C.boxEndIndex:t.collisionBoxArray.length;let K=-1;const X=(t,e)=>t&&t.circleDiameter?Math.max(t.circleDiameter,e):e;K=X(z,K),K=X(P,K),K=X(M,K),K=X(C,K);const Y=K>-1?1:0;Y&&(K*=A/No),t.glyphOffsetArray.length>=Kl.MAX_GLYPHS&&y("Too many glyphs being rendered in a tile. See https://github.com/mapbox/mapbox-gl-js/issues/2907"),void 0!==w.sortKey&&t.addToSortKeyRanges(t.symbolInstances.length,w.sortKey);const H=Ou(o,w,k),[W,Q]=function(t,e){const r=t.length,n=null==e?void 0:e.values;if((null==n?void 0:n.length)>0)for(let e=0;e=0?$.right:-1,$.center>=0?$.center:-1,$.left>=0?$.left:-1,$.vertical||-1,T,D,L,U,q,j,R,N,Z,G,J,u,E,F,B,V,Y,0,p,K,W,Q)}(t,l,i,r,n,s,C,t.layers[0],t.collisionBoxArray,e.index,e.sourceLayerIndex,t.index,x,[_,_,_,_],A,u,b,S,I,d,e,a,c,h,o)};if("line"===z)for(const s of function(t,e,r,n,s){const a=[];for(let e=0;e=n&&l.x>=n||(e.x>=n?e=new i(n,e.y+(n-e.x)/(l.x-e.x)*(l.y-e.y))._round():l.x>=n&&(l=new i(n,e.y+(n-e.x)/(l.x-e.x)*(l.y-e.y))._round()),e.y>=s&&l.y>=s||(e.y>=s?e=new i(e.x+(s-e.y)/(l.y-e.y)*(l.x-e.x),s)._round():l.y>=s&&(l=new i(e.x+(s-e.y)/(l.y-e.y)*(l.x-e.x),s)._round()),o&&e.equals(o[o.length-1])||(o=[e],a.push(o)),o.push(l)))))}}return a}(e.geometry,0,0,vs,vs)){const e=wu(s,w,k,r.vertical||m,n,24,v,t.overscaling,vs);for(const r of e)m&&Zu(t,m.text,M,r)||B(s,r)}else if("line-center"===z){for(const t of e.geometry)if(t.length>1){const e=bu(t,k,r.vertical||m,n,24,v);e&&B(t,e)}}else if("Polygon"===e.type)for(const t of qa(e.geometry,0)){const e=Vu(t,16);B(t[0],new yu(e.x,e.y,0))}else if("LineString"===e.type)for(const t of e.geometry)B(t,new yu(t[0].x,t[0].y,0));else if("Point"===e.type)for(const t of e.geometry)for(const e of t)B([e],new yu(e.x,e.y,0))}function Ru(t,e,r,n,s,a,o,l,u,c,h,p,f,d,m){const g=function(t,e,r,n,s,a,o,l){const u=n.layout.get("text-rotate").evaluate(a,{})*Math.PI/180,c=[];for(const t of e.positionedLines)for(const n of t.positionedGlyphs){if(!n.rect)continue;const a=n.rect||{};let h=dl+1,p=!0,f=1,d=0;const y=(s||l)&&n.vertical,m=n.metrics.advance*n.scale/2;if(l&&e.verticalizable&&(d=t.lineOffset/2-(n.imageName?-(No-n.metrics.width*n.scale)/2:(n.scale-1)*No)),n.imageName){const t=o[n.imageName];p=t.sdf,f=t.pixelRatio,h=ml/f}const g=s?[n.x+m,n.y]:[0,0];let x=s?[0,0]:[n.x+m+r[0],n.y+r[1]-d],v=[0,0];y&&(v=x,x=[0,0]);const b=n.metrics.isDoubleResolution?2:1,w=(n.metrics.left-h)*n.scale-m+x[0],_=(-n.metrics.top-h)*n.scale+x[1],S=w+a.w/b*n.scale/f,k=_+a.h/b*n.scale/f,A=new i(w,_),I=new i(S,_),z=new i(w,k),M=new i(S,k);if(y){const t=new i(-m,m-bl),e=-Math.PI/2,r=No/2-m,s=new i(5-bl-r,-(n.imageName?r:0)),a=new i(...v);A._rotateAround(e,t)._add(s)._add(a),I._rotateAround(e,t)._add(s)._add(a),z._rotateAround(e,t)._add(s)._add(a),M._rotateAround(e,t)._add(s)._add(a)}if(u){const t=Math.sin(u),e=Math.cos(u),r=[e,-t,t,e];A._matMult(r),I._matMult(r),z._matMult(r),M._matMult(r)}const P=new i(0,0),C=new i(0,0);c.push({tl:A,tr:I,bl:z,br:M,tex:a,writingMode:e.writingMode,glyphOffset:g,sectionIndex:n.sectionIndex,isSDF:p,pixelOffsetTL:P,pixelOffsetBR:C,minFontScaleX:0,minFontScaleY:0})}return c}(0,r,l,s,a,o,n,t.allowVerticalPlacement),x=t.textSizeData;let v=null;"source"===x.kind?(v=[$l*s.layout.get("text-size").evaluate(o,{})],v[0]>Ll&&y(`${t.layerIds[0]}: Value for "text-size" is >= ${Dl}. Reduce your "text-size".`)):"composite"===x.kind&&(v=[$l*d.compositeTextSizes[0].evaluate(o,{},m),$l*d.compositeTextSizes[1].evaluate(o,{},m)],(v[0]>Ll||v[1]>Ll)&&y(`${t.layerIds[0]}: Value for "text-size" is >= ${Dl}. Reduce your "text-size".`)),t.addSymbols(t.text,g,v,l,a,o,c,e,u.lineStartIndex,u.lineLength,f,m);for(const e of h)p[e]=t.text.placedSymbolArray.length-1;return 4*g.length}function Nu(t){for(const e in t)return t[e];return null}function Zu(t,e,r,n){const i=t.compareText;if(e in i){const t=i[e];for(let e=t.length-1;e>=0;e--)if(n.dist(t[e])90||this.lat<-90)throw new Error("Invalid LngLat latitude value: must be between -90 and 90")}wrap(){return new Xu(function(t,e,r){const n=((t-e)%360+360)%360+e;return n===e?180:n}(this.lng,-180),this.lat)}toArray(){return[this.lng,this.lat]}toString(){return`LngLat(${this.lng}, ${this.lat})`}distanceTo(t){const e=Math.PI/180,r=this.lat*e,n=t.lat*e,i=Math.sin(r)*Math.sin(n)+Math.cos(r)*Math.cos(n)*Math.cos((t.lng-this.lng)*e);return Ku*Math.acos(Math.min(i,1))}static convert(t){if(t instanceof Xu)return t;if(Array.isArray(t)&&(2===t.length||3===t.length))return new Xu(Number(t[0]),Number(t[1]));if(!Array.isArray(t)&&"object"==typeof t&&null!==t)return new Xu(Number("lng"in t?t.lng:t.lon),Number(t.lat));throw new Error("`LngLatLike` argument must be specified as a LngLat instance, an object {lng: , lat: }, an object {lon: , lat: }, or an array of [, ]")}}const Yu=2*Math.PI*Ku;function Hu(t){return Yu*Math.cos(t*Math.PI/180)}function Wu(t){return 360/Math.PI*Math.atan(Math.exp((180-360*t)*Math.PI/180))-90}class Qu{constructor(t,e,r=0){this.x=+t,this.y=+e,this.z=+r}static fromLngLat(t,e=0){const r=Xu.convert(t);return new Qu((180+r.lng)/360,(180-180/Math.PI*Math.log(Math.tan(Math.PI/4+r.lat*Math.PI/360)))/360,function(t,e){return t/Hu(e)}(e,r.lat))}toLngLat(){return new Xu(360*this.x-180,Wu(this.y))}toAltitude(){return this.z*Hu(Wu(this.y))}meterInMercatorCoordinateUnits(){return 1/Yu*(t=Wu(this.y),1/Math.cos(t*Math.PI/180));var t}}class tc{constructor(t,e,r){if(t<0||t>25||r<0||r>=Math.pow(2,t)||e<0||e>=Math.pow(2,t))throw new Error(`x=${e}, y=${r}, z=${t} outside of bounds. 0<=x<${Math.pow(2,t)}, 0<=y<${Math.pow(2,t)} 0<=z<=25 `);this.z=t,this.x=e,this.y=r,this.key=nc(0,t,t,e,r)}equals(t){return this.z===t.z&&this.x===t.x&&this.y===t.y}url(t,e,r){const n=(s=this.y,a=this.z,o=Ju(256*(i=this.x),256*(s=Math.pow(2,a)-s-1),a),l=Ju(256*(i+1),256*(s+1),a),o[0]+","+o[1]+","+l[0]+","+l[1]);var i,s,a,o,l;const u=function(t,e,r){let n,i="";for(let s=t;s>0;s--)n=1<1?"@2x":"").replace(/{quadkey}/g,u).replace(/{bbox-epsg-3857}/g,n)}isChildOf(t){const e=this.z-t.z;return e>0&&t.x===this.x>>e&&t.y===this.y>>e}getTilePoint(t){const e=Math.pow(2,this.z);return new i((t.x*e-this.x)*vs,(t.y*e-this.y)*vs)}toString(){return`${this.z}/${this.x}/${this.y}`}}class ec{constructor(t,e){this.wrap=t,this.canonical=e,this.key=nc(t,e.z,e.z,e.x,e.y)}}class rc{constructor(t,e,r,n,i){if(t= z; overscaledZ = ${t}; z = ${r}`);this.overscaledZ=t,this.wrap=e,this.canonical=new tc(r,+n,+i),this.key=nc(e,t,r,n,i)}clone(){return new rc(this.overscaledZ,this.wrap,this.canonical.z,this.canonical.x,this.canonical.y)}equals(t){return this.overscaledZ===t.overscaledZ&&this.wrap===t.wrap&&this.canonical.equals(t.canonical)}scaledTo(t){if(t>this.overscaledZ)throw new Error(`targetZ > this.overscaledZ; targetZ = ${t}; overscaledZ = ${this.overscaledZ}`);const e=this.canonical.z-t;return t>this.canonical.z?new rc(t,this.wrap,this.canonical.z,this.canonical.x,this.canonical.y):new rc(t,this.wrap,t,this.canonical.x>>e,this.canonical.y>>e)}calculateScaledKey(t,e){if(t>this.overscaledZ)throw new Error(`targetZ > this.overscaledZ; targetZ = ${t}; overscaledZ = ${this.overscaledZ}`);const r=this.canonical.z-t;return t>this.canonical.z?nc(this.wrap*+e,t,this.canonical.z,this.canonical.x,this.canonical.y):nc(this.wrap*+e,t,t,this.canonical.x>>r,this.canonical.y>>r)}isChildOf(t){if(t.wrap!==this.wrap)return!1;const e=this.canonical.z-t.canonical.z;return 0===t.overscaledZ||t.overscaledZ>e&&t.canonical.y===this.canonical.y>>e}children(t){if(this.overscaledZ>=t)return[new rc(this.overscaledZ+1,this.wrap,this.canonical.z,this.canonical.x,this.canonical.y)];const e=this.canonical.z+1,r=2*this.canonical.x,n=2*this.canonical.y;return[new rc(e,this.wrap,e,r,n),new rc(e,this.wrap,e,r+1,n),new rc(e,this.wrap,e,r,n+1),new rc(e,this.wrap,e,r+1,n+1)]}isLessThan(t){return this.wrapt.wrap)&&(this.overscaledZt.overscaledZ)&&(this.canonical.xt.canonical.x)&&this.canonical.y=e.maxzoom||"none"!==e.visibility&&(sc(t,this.zoom,r),(o[e.id]=e.createBucket({index:a.bucketLayerIDs.length,layers:t,zoom:this.zoom,pixelRatio:this.pixelRatio,overscaling:this.overscaling,collisionBoxArray:this.collisionBoxArray,sourceLayerIndex:i,sourceID:this.source})).populate(c,l,this.tileID.canonical),a.bucketLayerIDs.push(t.map((t=>t.id))))}}let c,h,f,d;const m=p(l.glyphDependencies,(t=>Object.keys(t).map(Number)));this.inFlightDependencies.forEach((t=>null==t?void 0:t.cancel())),this.inFlightDependencies=[];const g=++this.dependencySentinel;Object.keys(m).length?this.inFlightDependencies.push(n.send("getGlyphs",{uid:this.uid,stacks:m,source:this.source,tileID:this.tileID,type:"glyphs"},((t,e)=>{g===this.dependencySentinel&&(c||(c=t,h=e,b.call(this)))}))):h={};const x=Object.keys(l.iconDependencies);x.length?this.inFlightDependencies.push(n.send("getImages",{icons:x,source:this.source,tileID:this.tileID,type:"icons"},((t,e)=>{g===this.dependencySentinel&&(c||(c=t,f=e,b.call(this)))}))):f={};const v=Object.keys(l.patternDependencies);function b(){if(c)return i(c);if(h&&f&&d){const t=new Gu(h),e=new xl(f,d);for(const n in o){const i=o[n];i instanceof Kl?(sc(i.layers,this.zoom,r),Uu({bucket:i,glyphMap:h,glyphPositions:t.positions,imageMap:f,imagePositions:e.iconPositions,showCollisionBoxes:this.showCollisionBoxes,canonical:this.tileID.canonical})):i.hasPattern&&(i instanceof Co||i instanceof Za||i instanceof yo)&&(sc(i.layers,this.zoom,r),i.addFeatures(l,this.tileID.canonical,e.patternPositions))}this.status="done",i(null,{buckets:Object.values(o).filter((t=>!t.isEmpty())),featureIndex:a,collisionBoxArray:this.collisionBoxArray,glyphAtlasImage:t.image,imageAtlas:e,glyphMap:this.returnDependencies?h:null,iconMap:this.returnDependencies?f:null,glyphPositions:this.returnDependencies?t.positions:null})}}v.length?this.inFlightDependencies.push(n.send("getImages",{icons:v,source:this.source,tileID:this.tileID,type:"patterns"},((t,e)=>{g===this.dependencySentinel&&(c||(c=t,d=e,b.call(this)))}))):d={},b.call(this)}}function sc(t,e,r){const n=new In(e);for(const e of t)e.recalculate(n,r)}var ac;!function(t){t.create="create",t.load="load",t.fullLoad="fullLoad"}(ac||(ac={}));class oc{constructor(t){this._marks={start:[t.url,"start"].join("#"),end:[t.url,"end"].join("#"),measure:t.url.toString()},performance.mark(this._marks.start)}finish(){performance.mark(this._marks.end);let t=performance.getEntriesByName(this._marks.measure);return 0===t.length&&(performance.measure(this._marks.measure,this._marks.start,this._marks.end),t=performance.getEntriesByName(this._marks.measure),performance.clearMarks(this._marks.start),performance.clearMarks(this._marks.end),performance.clearMeasures(this._marks.measure)),t}}function lc(t,e){const r=function(t,e){return Yr(h(t,{type:"arrayBuffer"}),e)}(t.request,((r,n,i,s)=>{if(r)e(r);else if(n)try{const t=new Qa.VectorTile(new fl(n));e(null,{vectorTile:t,rawData:n,cacheControl:i,expires:s})}catch(r){const i=new Uint8Array(n);let s=`Unable to parse the tile at ${t.request.url}, `;s+=31===i[0]&&139===i[1]?"please make sure the data is not gzipped and that you have configured the relevant header in the server":`got error: ${r.messge}`,e(new Error(s))}}));return()=>{r.cancel(),e()}}class uc{constructor(t,e,r,n){this.actor=t,this.layerIndex=e,this.availableImages=r,this.loadVectorData=n||lc,this.fetching={},this.loading={},this.loaded={}}loadTile(t,e){const r=t.uid;this.loading||(this.loading={});const n=!!(t&&t.request&&t.request.collectResourceTiming)&&new oc(t.request),i=this.loading[r]=new ic(t);i.abort=this.loadVectorData(t,((t,s)=>{if(delete this.loading[r],t||!s)return i.status="done",this.loaded[r]=i,e(t);const a=s.rawData,o={};s.expires&&(o.expires=s.expires),s.cacheControl&&(o.cacheControl=s.cacheControl);const l={};if(n){const t=n.finish();t&&(l.resourceTiming=JSON.parse(JSON.stringify(t)))}i.vectorTile=s.vectorTile,i.parse(s.vectorTile,this.layerIndex,this.availableImages,this.actor,((t,n)=>{if(delete this.fetching[r],t||!n)return e(t);e(null,h({rawTileData:a.slice(0)},n,o,l))})),this.loaded=this.loaded||{},this.loaded[r]=i,this.fetching[r]={rawTileData:a,cacheControl:o,resourceTiming:l}}))}reloadTile(t,e){const r=this.loaded,n=t.uid;if(r&&r[n]){const i=r[n];i.showCollisionBoxes=t.showCollisionBoxes,"parsing"===i.status?i.parse(i.vectorTile,this.layerIndex,this.availableImages,this.actor,((t,r)=>{if(t||!r)return e(t,r);let i;if(this.fetching[n]){const{rawTileData:t,cacheControl:e,resourceTiming:s}=this.fetching[n];delete this.fetching[n],i=h({rawTileData:t.slice(0)},r,e,s)}else i=r;e(null,i)})):"done"===i.status&&(i.vectorTile?i.parse(i.vectorTile,this.layerIndex,this.availableImages,this.actor,e):e())}}abortTile(t,e){const r=this.loading,n=t.uid;r&&r[n]&&r[n].abort&&(r[n].abort(),delete r[n]),e()}removeTile(t,e){const r=this.loaded,n=t.uid;r&&r[n]&&delete r[n],e()}}class cc{constructor(t,e,r,n=1,i=1,s=1,a=0){if(this.uid=t,e.height!==e.width)throw new RangeError("DEM tiles must be square");if(r&&!["mapbox","terrarium","custom"].includes(r))return void y(`"${r}" is not a valid encoding type. Valid types include "mapbox", "terrarium" and "custom".`);this.stride=e.height;const o=this.dim=e.height-2;switch(this.data=new Uint32Array(e.data.buffer),r){case"terrarium":this.redFactor=256,this.greenFactor=1,this.blueFactor=1/256,this.baseShift=32768;break;case"custom":this.redFactor=n,this.greenFactor=i,this.blueFactor=s,this.baseShift=a;break;default:this.redFactor=6553.6,this.greenFactor=25.6,this.blueFactor=.1,this.baseShift=1e4}for(let t=0;tthis.max&&(this.max=r),r=this.dim+1||e<-1||e>=this.dim+1)throw new RangeError("out of range source coordinates for DEM data");return(e+1)*this.stride+(t+1)}unpack(t,e,r){return t*this.redFactor+e*this.greenFactor+r*this.blueFactor-this.baseShift}getPixels(){return new ra({width:this.stride,height:this.stride},new Uint8Array(this.data.buffer))}backfillBorder(t,e,r){if(this.dim!==t.dim)throw new Error("dem dimension mismatch");let n=e*this.dim,i=e*this.dim+this.dim,s=r*this.dim,a=r*this.dim+this.dim;switch(e){case-1:n=i-1;break;case 1:i=n+1}switch(r){case-1:s=a-1;break;case 1:a=s+1}const o=-e*this.dim,l=-r*this.dim;for(let e=s;e=Math.abs(o)?r-l+o:o-l+r,r=l}r+n>=0!=!!e&&t.reverse()}var dc=e((function t(e,r){var n,i=e&&e.type;if("FeatureCollection"===i)for(n=0;n>31}function Cc(t,e){for(var r=t.loadGeometry(),n=t.type,i=0,s=0,a=r.length,o=0;o>4;if(1!==n)throw new Error(`Got v${n} data when expected v1.`);const i=Ec[15&r];if(!i)throw new Error("Unrecognized array type.");const[s]=new Uint16Array(t,2,1),[a]=new Uint32Array(t,4,1);return new Fc(a,s,i,t)}constructor(t,e=64,r=Float64Array,n){if(isNaN(t)||t<0)throw new Error(`Unpexpected numItems value: ${t}.`);this.numItems=+t,this.nodeSize=Math.min(Math.max(+e,2),65535),this.ArrayType=r,this.IndexArrayType=t<65536?Uint16Array:Uint32Array;const i=Ec.indexOf(this.ArrayType),s=2*t*this.ArrayType.BYTES_PER_ELEMENT,a=t*this.IndexArrayType.BYTES_PER_ELEMENT,o=(8-a%8)%8;if(i<0)throw new Error(`Unexpected typed array class: ${r}.`);n&&n instanceof ArrayBuffer?(this.data=n,this.ids=new this.IndexArrayType(this.data,8,t),this.coords=new this.ArrayType(this.data,8+a+o,2*t),this._pos=2*t,this._finished=!0):(this.data=new ArrayBuffer(8+s+a+o),this.ids=new this.IndexArrayType(this.data,8,t),this.coords=new this.ArrayType(this.data,8+a+o,2*t),this._pos=0,this._finished=!1,new Uint8Array(this.data,0,2).set([219,16+i]),new Uint16Array(this.data,2,1)[0]=e,new Uint32Array(this.data,4,1)[0]=t)}add(t,e){const r=this._pos>>1;return this.ids[r]=r,this.coords[this._pos++]=t,this.coords[this._pos++]=e,r}finish(){const t=this._pos>>1;if(t!==this.numItems)throw new Error(`Added ${t} items when expected ${this.numItems}.`);return Tc(this.ids,this.coords,this.nodeSize,0,this.numItems-1,0),this._finished=!0,this}range(t,e,r,n){if(!this._finished)throw new Error("Data not yet indexed - call index.finish().");const{ids:i,coords:s,nodeSize:a}=this,o=[0,i.length-1,0],l=[];for(;o.length;){const u=o.pop()||0,c=o.pop()||0,h=o.pop()||0;if(c-h<=a){for(let a=h;a<=c;a++){const o=s[2*a],u=s[2*a+1];o>=t&&o<=r&&u>=e&&u<=n&&l.push(i[a])}continue}const p=h+c>>1,f=s[2*p],d=s[2*p+1];f>=t&&f<=r&&d>=e&&d<=n&&l.push(i[p]),(0===u?t<=f:e<=d)&&(o.push(h),o.push(p-1),o.push(1-u)),(0===u?r>=f:n>=d)&&(o.push(p+1),o.push(c),o.push(1-u))}return l}within(t,e,r){if(!this._finished)throw new Error("Data not yet indexed - call index.finish().");const{ids:n,coords:i,nodeSize:s}=this,a=[0,n.length-1,0],o=[],l=r*r;for(;a.length;){const u=a.pop()||0,c=a.pop()||0,h=a.pop()||0;if(c-h<=s){for(let r=h;r<=c;r++)Oc(i[2*r],i[2*r+1],t,e)<=l&&o.push(n[r]);continue}const p=h+c>>1,f=i[2*p],d=i[2*p+1];Oc(f,d,t,e)<=l&&o.push(n[p]),(0===u?t-r<=f:e-r<=d)&&(a.push(h),a.push(p-1),a.push(1-u)),(0===u?t+r>=f:e+r>=d)&&(a.push(p+1),a.push(c),a.push(1-u))}return o}}function Tc(t,e,r,n,i,s){if(i-n<=r)return;const a=n+i>>1;Dc(t,e,a,n,i,s),Tc(t,e,r,n,a-1,1-s),Tc(t,e,r,a+1,i,1-s)}function Dc(t,e,r,n,i,s){for(;i>n;){if(i-n>600){const a=i-n+1,o=r-n+1,l=Math.log(a),u=.5*Math.exp(2*l/3),c=.5*Math.sqrt(l*u*(a-u)/a)*(o-a/2<0?-1:1);Dc(t,e,r,Math.max(n,Math.floor(r-o*u/a+c)),Math.min(i,Math.floor(r+(a-o)*u/a+c)),s)}const a=e[2*r+s];let o=n,l=i;for($c(t,e,n,r),e[2*i+s]>a&&$c(t,e,n,i);oa;)l--}e[2*n+s]===a?$c(t,e,n,l):(l++,$c(t,e,l,i)),l<=r&&(n=l+1),r<=l&&(i=l-1)}}function $c(t,e,r,n){Lc(t,r,n),Lc(e,2*r,2*n),Lc(e,2*r+1,2*n+1)}function Lc(t,e,r){const n=t[e];t[e]=t[r],t[r]=n}function Oc(t,e,r,n){const i=t-r,s=e-n;return i*i+s*s}const Uc={minZoom:0,maxZoom:16,minPoints:2,radius:40,extent:512,nodeSize:64,log:!1,generateId:!1,reduce:null,map:t=>t},qc=Math.fround||(jc=new Float32Array(1),t=>(jc[0]=+t,jc[0]));var jc;const Rc=3,Nc=5,Zc=6;class Gc{constructor(t){this.options=Object.assign(Object.create(Uc),t),this.trees=new Array(this.options.maxZoom+1),this.stride=this.options.reduce?7:6,this.clusterProps=[]}load(t){const{log:e,minZoom:r,maxZoom:n}=this.options;e&&console.time("total time");const i=`prepare ${t.length} points`;e&&console.time(i),this.points=t;const s=[];for(let e=0;e=r;t--){const r=+Date.now();a=this.trees[t]=this._createTree(this._cluster(a,t)),e&&console.log("z%d: %d clusters in %dms",t,a.numItems,+Date.now()-r)}return e&&console.timeEnd("total time"),this}getClusters(t,e){let r=((t[0]+180)%360+360)%360-180;const n=Math.max(-90,Math.min(90,t[1]));let i=180===t[2]?180:((t[2]+180)%360+360)%360-180;const s=Math.max(-90,Math.min(90,t[3]));if(t[2]-t[0]>=360)r=-180,i=180;else if(r>i){const t=this.getClusters([r,n,180,s],e),a=this.getClusters([-180,n,i,s],e);return t.concat(a)}const a=this.trees[this._limitZoom(e)],o=a.range(Xc(r),Yc(s),Xc(i),Yc(n)),l=a.data,u=[];for(const t of o){const e=this.stride*t;u.push(l[e+Nc]>1?Jc(l,e,this.clusterProps):this.points[l[e+Rc]])}return u}getChildren(t){const e=this._getOriginId(t),r=this._getOriginZoom(t),n="No cluster with the specified id.",i=this.trees[r];if(!i)throw new Error(n);const s=i.data;if(e*this.stride>=s.length)throw new Error(n);const a=this.options.radius/(this.options.extent*Math.pow(2,r-1)),o=i.within(s[e*this.stride],s[e*this.stride+1],a),l=[];for(const e of o){const r=e*this.stride;s[r+4]===t&&l.push(s[r+Nc]>1?Jc(s,r,this.clusterProps):this.points[s[r+Rc]])}if(0===l.length)throw new Error(n);return l}getLeaves(t,e,r){const n=[];return this._appendLeaves(n,t,e=e||10,r=r||0,0),n}getTile(t,e,r){const n=this.trees[this._limitZoom(t)],i=Math.pow(2,t),{extent:s,radius:a}=this.options,o=a/s,l=(r-o)/i,u=(r+1+o)/i,c={features:[]};return this._addTileFeatures(n.range((e-o)/i,l,(e+1+o)/i,u),n.data,e,r,i,c),0===e&&this._addTileFeatures(n.range(1-o/i,l,1,u),n.data,i,r,i,c),e===i-1&&this._addTileFeatures(n.range(0,l,o/i,u),n.data,-1,r,i,c),c.features.length?c:null}getClusterExpansionZoom(t){let e=this._getOriginZoom(t)-1;for(;e<=this.options.maxZoom;){const r=this.getChildren(t);if(e++,1!==r.length)break;t=r[0].properties.cluster_id}return e}_appendLeaves(t,e,r,n,i){const s=this.getChildren(e);for(const e of s){const s=e.properties;if(s&&s.cluster?i+s.point_count<=n?i+=s.point_count:i=this._appendLeaves(t,s.cluster_id,r,n,i):i1;let l,u,c;if(o)l=Kc(e,t,this.clusterProps),u=e[t],c=e[t+1];else{const r=this.points[e[t+Rc]];l=r.properties;const[n,i]=r.geometry.coordinates;u=Xc(n),c=Yc(i)}const h={type:1,geometry:[[Math.round(this.options.extent*(u*i-r)),Math.round(this.options.extent*(c*i-n))]],tags:l};let p;p=o||this.options.generateId?e[t+Rc]:this.points[e[t+Rc]].id,void 0!==p&&(h.id=p),s.features.push(h)}}_limitZoom(t){return Math.max(this.options.minZoom,Math.min(Math.floor(+t),this.options.maxZoom+1))}_cluster(t,e){const{radius:r,extent:n,reduce:i,minPoints:s}=this.options,a=r/(n*Math.pow(2,e)),o=t.data,l=[],u=this.stride;for(let r=0;re&&(f+=o[r+Nc])}if(f>p&&f>=s){let t,s=n*p,a=c*p,d=-1;const y=((r/u|0)<<5)+(e+1)+this.points.length;for(const n of h){const l=n*u;if(o[l+2]<=e)continue;o[l+2]=e;const c=o[l+Nc];s+=o[l]*c,a+=o[l+1]*c,o[l+4]=y,i&&(t||(t=this._map(o,r,!0),d=this.clusterProps.length,this.clusterProps.push(t)),i(t,this._map(o,l)))}o[r+4]=y,l.push(s/f,a/f,1/0,y,-1,f),i&&l.push(d)}else{for(let t=0;t1)for(const t of h){const r=t*u;if(!(o[r+2]<=e)){o[r+2]=e;for(let t=0;t>5}_getOriginZoom(t){return(t-this.points.length)%32}_map(t,e,r){if(t[e+Nc]>1){const n=this.clusterProps[t[e+Zc]];return r?Object.assign({},n):n}const n=this.points[t[e+Rc]].properties,i=this.options.map(n);return r&&i===n?Object.assign({},i):i}}function Jc(t,e,r){return{type:"Feature",id:t[e+Rc],properties:Kc(t,e,r),geometry:{type:"Point",coordinates:[(n=t[e],360*(n-.5)),Hc(t[e+1])]}};var n}function Kc(t,e,r){const n=t[e+Nc],i=n>=1e4?`${Math.round(n/1e3)}k`:n>=1e3?Math.round(n/100)/10+"k":n,s=t[e+Zc],a=-1===s?{}:Object.assign({},r[s]);return Object.assign(a,{cluster:!0,cluster_id:t[e+Rc],point_count:n,point_count_abbreviated:i})}function Xc(t){return t/360+.5}function Yc(t){const e=Math.sin(t*Math.PI/180),r=.5-.25*Math.log((1+e)/(1-e))/Math.PI;return r<0?0:r>1?1:r}function Hc(t){const e=(180-360*t)*Math.PI/180;return 360*Math.atan(Math.exp(e))/Math.PI-90}function Wc(t,e,r,n){for(var i,s=n,a=r-e>>1,o=r-e,l=t[e],u=t[e+1],c=t[r],h=t[r+1],p=e+3;ps)i=p,s=f;else if(f===s){var d=Math.abs(p-a);dn&&(i-e>3&&Wc(t,e,i,n),t[i+2]=s,r-i>3&&Wc(t,i,r,n))}function Qc(t,e,r,n,i,s){var a=i-r,o=s-n;if(0!==a||0!==o){var l=((t-r)*a+(e-n)*o)/(a*a+o*o);l>1?(r=i,n=s):l>0&&(r+=a*l,n+=o*l)}return(a=t-r)*a+(o=e-n)*o}function th(t,e,r,n){var i={id:void 0===t?null:t,type:e,geometry:r,tags:n,minX:1/0,minY:1/0,maxX:-1/0,maxY:-1/0};return function(t){var e=t.geometry,r=t.type;if("Point"===r||"MultiPoint"===r||"LineString"===r)eh(t,e);else if("Polygon"===r||"MultiLineString"===r)for(var n=0;n0&&(a+=n?(i*u-l*s)/2:Math.sqrt(Math.pow(l-i,2)+Math.pow(u-s,2))),i=l,s=u}var c=e.length-3;e[2]=1,Wc(e,0,c,r),e[c+2]=1,e.size=Math.abs(a),e.start=0,e.end=e.size}function sh(t,e,r,n){for(var i=0;i1?1:r}function lh(t,e,r,n,i,s,a,o){if(n/=e,s>=(r/=e)&&a=n)return null;for(var l=[],u=0;u=r&&d=n)){var y=[];if("Point"===p||"MultiPoint"===p)uh(h,y,r,n,i);else if("LineString"===p)ch(h,y,r,n,i,!1,o.lineMetrics);else if("MultiLineString"===p)ph(h,y,r,n,i,!1);else if("Polygon"===p)ph(h,y,r,n,i,!0);else if("MultiPolygon"===p)for(var m=0;m=r&&a<=n&&(e.push(t[s]),e.push(t[s+1]),e.push(t[s+2]))}}function ch(t,e,r,n,i,s,a){for(var o,l,u=hh(t),c=0===i?dh:yh,h=t.start,p=0;pr&&(l=c(u,f,d,m,g,r),a&&(u.start=h+o*l)):x>n?v=r&&(l=c(u,f,d,m,g,r),b=!0),v>n&&x<=n&&(l=c(u,f,d,m,g,n),b=!0),!s&&b&&(a&&(u.end=h+o*l),e.push(u),u=hh(t)),a&&(h+=o)}var w=t.length-3;f=t[w],d=t[w+1],y=t[w+2],(x=0===i?f:d)>=r&&x<=n&&fh(u,f,d,y),w=u.length-3,s&&w>=3&&(u[w]!==u[0]||u[w+1]!==u[1])&&fh(u,u[0],u[1],u[2]),u.length&&e.push(u)}function hh(t){var e=[];return e.size=t.size,e.start=t.start,e.end=t.end,e}function ph(t,e,r,n,i,s){for(var a=0;aa.maxX&&(a.maxX=c),h>a.maxY&&(a.maxY=h)}return a}function wh(t,e,r,n){var i=e.geometry,s=e.type,a=[];if("Point"===s||"MultiPoint"===s)for(var o=0;o0&&e.size<(i?a:n))r.numPoints+=e.length/3;else{for(var o=[],l=0;la)&&(r.numSimplified++,o.push(e[l]),o.push(e[l+1])),r.numPoints++;i&&function(t,e){for(var r=0,n=0,i=t.length,s=i-2;n0===e)for(n=0,i=t.length;n24)throw new Error("maxZoom should be in the 0-24 range");if(e.promoteId&&e.generateId)throw new Error("promoteId and generateId cannot be used together.");var n=function(t,e){var r=[];if("FeatureCollection"===t.type)for(var n=0;n1&&console.time("creation"),p=this.tiles[h]=bh(t,e,r,n,l),this.tileCoords.push({z:e,x:r,y:n}),u)){u>1&&(console.log("tile z%d-%d-%d (features: %d, points: %d, simplified: %d)",e,r,n,p.numFeatures,p.numPoints,p.numSimplified),console.timeEnd("creation"));var f="z"+e;this.stats[f]=(this.stats[f]||0)+1,this.total++}if(p.source=t,i){if(e===l.maxZoom||e===i)continue;var d=1<1&&console.time("clipping");var y,m,g,x,v,b,w=.5*l.buffer/l.extent,_=.5-w,S=.5+w,k=1+w;y=m=g=x=null,v=lh(t,c,r-w,r+S,0,p.minX,p.maxX,l),b=lh(t,c,r+_,r+k,0,p.minX,p.maxX,l),t=null,v&&(y=lh(v,c,n-w,n+S,1,p.minY,p.maxY,l),m=lh(v,c,n+_,n+k,1,p.minY,p.maxY,l),v=null),b&&(g=lh(b,c,n-w,n+S,1,p.minY,p.maxY,l),x=lh(b,c,n+_,n+k,1,p.minY,p.maxY,l),b=null),u>1&&console.timeEnd("clipping"),o.push(y||[],e+1,2*r,2*n),o.push(m||[],e+1,2*r,2*n+1),o.push(g||[],e+1,2*r+1,2*n),o.push(x||[],e+1,2*r+1,2*n+1)}}},Sh.prototype.getTile=function(t,e,r){var n=this.options,i=n.extent,s=n.debug;if(t<0||t>24)return null;var a=1<1&&console.log("drilling down to z%d-%d-%d",t,e,r);for(var l,u=t,c=e,h=r;!l&&u>0;)u--,c=Math.floor(c/2),h=Math.floor(h/2),l=this.tiles[kh(u,c,h)];return l&&l.source?(s>1&&console.log("found parent tile z%d-%d-%d",u,c,h),s>1&&console.time("drilling down"),this.splitTile(l.source,u,c,h,t,e,r),s>1&&console.timeEnd("drilling down"),this.tiles[o]?xh(this.tiles[o],i):null):null};class Mh extends uc{constructor(t,e,r,n){super(t,e,r),this._dataUpdateable=new Map,this.loadGeoJSON=(t,e)=>{const{promoteId:r}=t;if(t.request)return function(t,e){return Yr(h(t,{type:"json"}),e)}(t.request,((t,n,i,s)=>{this._dataUpdateable=Ih(n,r)?zh(n,r):void 0,e(t,n,i,s)}));if("string"==typeof t.data)try{const n=JSON.parse(t.data);this._dataUpdateable=Ih(n,r)?zh(n,r):void 0,e(null,n)}catch(r){e(new Error(`Input data given to '${t.source}' is not a valid GeoJSON object.`))}else t.dataDiff?this._dataUpdateable?(function(t,e,r){var n,i,s,a;if(e.removeAll&&t.clear(),e.remove)for(const r of e.remove)t.delete(r);if(e.add)for(const n of e.add){const e=Ah(n,r);null!=e&&t.set(e,n)}if(e.update)for(const r of e.update){let e=t.get(r.id);if(null==e)continue;const o=!r.removeAllProperties&&((null===(n=r.removeProperties)||void 0===n?void 0:n.length)>0||(null===(i=r.addOrUpdateProperties)||void 0===i?void 0:i.length)>0);if((r.newGeometry||r.removeAllProperties||o)&&(e=Object.assign({},e),t.set(r.id,e),o&&(e.properties=Object.assign({},e.properties))),r.newGeometry&&(e.geometry=r.newGeometry),r.removeAllProperties)e.properties={};else if((null===(s=r.removeProperties)||void 0===s?void 0:s.length)>0)for(const t of r.removeProperties)Object.prototype.hasOwnProperty.call(e.properties,t)&&delete e.properties[t];if((null===(a=r.addOrUpdateProperties)||void 0===a?void 0:a.length)>0)for(const{key:t,value:n}of r.addOrUpdateProperties)e.properties[t]=n}}(this._dataUpdateable,t.dataDiff,r),e(null,{type:"FeatureCollection",features:Array.from(this._dataUpdateable.values())})):e(new Error(`Cannot update existing geojson data in ${t.source}`)):e(new Error(`Input data given to '${t.source}' is not a valid GeoJSON object.`));return{cancel:()=>{}}},this.loadVectorData=this.loadGeoJSONTile,n&&(this.loadGeoJSON=n)}loadGeoJSONTile(t,e){const r=t.tileID.canonical;if(!this._geoJSONIndex)return e(null,null);const n=this._geoJSONIndex.getTile(r.z,r.x,r.y);if(!n)return e(null,null);const s=new class{constructor(t){this.layers={_geojsonTileLayer:this},this.name="_geojsonTileLayer",this.extent=vs,this.length=t.length,this._features=t}feature(t){return new class{constructor(t){this._feature=t,this.extent=vs,this.type=t.type,this.properties=t.tags,"id"in t&&!isNaN(t.id)&&(this.id=parseInt(t.id,10))}loadGeometry(){if(1===this._feature.type){const t=[];for(const e of this._feature.geometry)t.push([new i(e[0],e[1])]);return t}{const t=[];for(const e of this._feature.geometry){const r=[];for(const t of e)r.push(new i(t[0],t[1]));t.push(r)}return t}}toGeoJSON(t,e,r){return yc.call(this,t,e,r)}}(this._features[t])}}(n.features);let a=Vc(s);0===a.byteOffset&&a.byteLength===a.buffer.byteLength||(a=new Uint8Array(a)),e(null,{vectorTile:s,rawData:a.buffer})}loadData(t,e){var r;null===(r=this._pendingRequest)||void 0===r||r.cancel(),this._pendingCallback&&this._pendingCallback(null,{abandoned:!0});const n=!!(t&&t.request&&t.request.collectResourceTiming)&&new oc(t.request);this._pendingCallback=e,this._pendingRequest=this.loadGeoJSON(t,((r,i)=>{if(delete this._pendingCallback,delete this._pendingRequest,r||!i)return e(r);if("object"!=typeof i)return e(new Error(`Input data given to '${t.source}' is not a valid GeoJSON object.`));{dc(i,!0);try{if(t.filter){const e=tr(t.filter,{type:"boolean","property-type":"data-driven",overridable:!1,transition:!1});if("error"===e.result)throw new Error(e.value.map((t=>`${t.key}: ${t.message}`)).join(", "));const r=i.features.filter((t=>e.value.evaluate({zoom:0},t)));i={type:"FeatureCollection",features:r}}this._geoJSONIndex=t.cluster?new Gc(function({superclusterOptions:t,clusterProperties:e}){if(!e||!t)return t;const r={},n={},i={accumulated:null,zoom:0},s={properties:null},a=Object.keys(e);for(const t of a){const[i,s]=e[t],a=tr(s),o=tr("string"==typeof i?[i,["accumulated"],["get",t]]:i);r[t]=a.value,n[t]=o.value}return t.map=t=>{s.properties=t;const e={};for(const t of a)e[t]=r[t].evaluate(i,s);return e},t.reduce=(t,e)=>{s.properties=e;for(const e of a)i.accumulated=t[e],t[e]=n[e].evaluate(i,s)},t}(t)).load(i.features):function(t,e){return new Sh(t,e)}(i,t.geojsonVtOptions)}catch(r){return e(r)}this.loaded={};const s={};if(n){const e=n.finish();e&&(s.resourceTiming={},s.resourceTiming[t.source]=JSON.parse(JSON.stringify(e)))}e(null,s)}}))}reloadTile(t,e){const r=this.loaded;return r&&r[t.uid]?super.reloadTile(t,e):this.loadTile(t,e)}removeSource(t,e){this._pendingCallback&&this._pendingCallback(null,{abandoned:!0}),e()}getClusterExpansionZoom(t,e){try{e(null,this._geoJSONIndex.getClusterExpansionZoom(t.clusterId))}catch(t){e(t)}}getClusterChildren(t,e){try{e(null,this._geoJSONIndex.getChildren(t.clusterId))}catch(t){e(t)}}getClusterLeaves(t,e){try{e(null,this._geoJSONIndex.getLeaves(t.clusterId,t.limit,t.offset))}catch(t){e(t)}}}class Ph{constructor(t){this.self=t,this.actor=new nn(t,this),this.layerIndexes={},this.availableImages={},this.workerSourceTypes={vector:uc,geojson:Mh},this.workerSources={},this.demWorkerSources={},this.self.registerWorkerSource=(t,e)=>{if(this.workerSourceTypes[t])throw new Error(`Worker source with name "${t}" already registered.`);this.workerSourceTypes[t]=e},this.self.registerRTLTextPlugin=t=>{if(An.isParsed())throw new Error("RTL text plugin already registered.");An.applyArabicShaping=t.applyArabicShaping,An.processBidirectionalText=t.processBidirectionalText,An.processStyledBidirectionalText=t.processStyledBidirectionalText}}setReferrer(t,e){this.referrer=e}setImages(t,e,r){this.availableImages[t]=e;for(const r in this.workerSources[t]){const n=this.workerSources[t][r];for(const t in n)n[t].availableImages=e}r()}setLayers(t,e,r){this.getLayerIndex(t).replace(e),r()}updateLayers(t,e,r){this.getLayerIndex(t).update(e.layers,e.removedIds),r()}loadTile(t,e,r){this.getWorkerSource(t,e.type,e.source).loadTile(e,r)}loadDEMTile(t,e,r){this.getDEMWorkerSource(t,e.source).loadTile(e,r)}reloadTile(t,e,r){this.getWorkerSource(t,e.type,e.source).reloadTile(e,r)}abortTile(t,e,r){this.getWorkerSource(t,e.type,e.source).abortTile(e,r)}removeTile(t,e,r){this.getWorkerSource(t,e.type,e.source).removeTile(e,r)}removeDEMTile(t,e){this.getDEMWorkerSource(t,e.source).removeTile(e)}removeSource(t,e,r){if(!this.workerSources[t]||!this.workerSources[t][e.type]||!this.workerSources[t][e.type][e.source])return;const n=this.workerSources[t][e.type][e.source];delete this.workerSources[t][e.type][e.source],void 0!==n.removeSource?n.removeSource(e,r):r()}loadWorkerSource(t,e,r){try{this.self.importScripts(e.url),r()}catch(t){r(t.toString())}}syncRTLPluginState(t,e,r){try{An.setState(e);const t=An.getPluginURL();if(An.isLoaded()&&!An.isParsed()&&null!=t){this.self.importScripts(t);const e=An.isParsed();r(e?void 0:new Error(`RTL Text Plugin failed to import scripts from ${t}`),e)}}catch(t){r(t.toString())}}getAvailableImages(t){let e=this.availableImages[t];return e||(e=[]),e}getLayerIndex(t){let e=this.layerIndexes[t];return e||(e=this.layerIndexes[t]=new lu),e}getWorkerSource(t,e,r){return this.workerSources[t]||(this.workerSources[t]={}),this.workerSources[t][e]||(this.workerSources[t][e]={}),this.workerSources[t][e][r]||(this.workerSources[t][e][r]=new this.workerSourceTypes[e]({send:(e,r,n)=>{this.actor.send(e,r,n,t)}},this.getLayerIndex(t),this.getAvailableImages(t))),this.workerSources[t][e][r]}getDEMWorkerSource(t,e){return this.demWorkerSources[t]||(this.demWorkerSources[t]={}),this.demWorkerSources[t][e]||(this.demWorkerSources[t][e]=new hc),this.demWorkerSources[t][e]}}return x()&&(self.worker=new Ph(self)),Ph}(); +//# sourceMappingURL=maplibre-gl-csp-worker.js.map diff --git a/web/libraries/maplibre-gl/dist/maplibre-gl-csp-worker.js.map b/web/libraries/maplibre-gl/dist/maplibre-gl-csp-worker.js.map new file mode 100644 index 00000000..36580b5c --- /dev/null +++ b/web/libraries/maplibre-gl/dist/maplibre-gl-csp-worker.js.map @@ -0,0 +1 @@ +{"version":3,"file":"maplibre-gl-csp-worker.js","sources":["../node_modules/tslib/tslib.es6.js","../node_modules/@mapbox/point-geometry/index.js","../node_modules/@mapbox/unitbezier/index.js","../src/util/offscreen_canvas_supported.ts","../src/util/offscreen_canvas_distorted.ts","../src/util/util.ts","../src/util/transferable_grid_index.ts","../node_modules/@maplibre/maplibre-gl-style-spec/dist/index.mjs","../src/util/config.ts","../src/util/ajax.ts","../src/util/web_worker_transfer.ts","../src/util/throttled_invoker.ts","../src/util/actor.ts","../src/util/evented.ts","../src/style/validate_style.ts","../src/style/zoom_history.ts","../src/util/is_char_in_unicode_block.ts","../src/util/script_detection.ts","../src/util/browser.ts","../src/source/rtl_text_plugin.ts","../src/style/evaluation_parameters.ts","../src/style/properties.ts","../src/style/style_layer.ts","../src/util/struct_array.ts","../src/data/array_types.g.ts","../src/data/bucket/circle_attributes.ts","../src/data/segment.ts","../src/shaders/encode_attribute.ts","../src/data/bucket/pattern_attributes.ts","../node_modules/murmurhash-js/murmurhash3_gc.js","../node_modules/murmurhash-js/murmurhash2_gc.js","../node_modules/murmurhash-js/index.js","../src/data/feature_position_map.ts","../src/render/uniform_binding.ts","../src/data/program_configuration.ts","../src/data/extent.ts","../src/data/load_geometry.ts","../src/data/evaluation_feature.ts","../src/data/bucket/circle_bucket.ts","../src/util/intersection_tests.ts","../src/style/query_utils.ts","../src/style/style_layer/circle_style_layer_properties.g.ts","../node_modules/gl-matrix/esm/vec4.js","../node_modules/gl-matrix/esm/common.js","../node_modules/gl-matrix/esm/mat4.js","../src/style/style_layer/circle_style_layer.ts","../src/data/bucket/heatmap_bucket.ts","../src/style/style_layer/heatmap_style_layer_properties.g.ts","../src/util/image.ts","../src/style/style_layer/heatmap_style_layer.ts","../src/util/color_ramp.ts","../src/style/style_layer/hillshade_style_layer_properties.g.ts","../src/style/style_layer/hillshade_style_layer.ts","../src/data/bucket/fill_attributes.ts","../node_modules/earcut/src/earcut.js","../node_modules/quickselect/index.js","../src/util/classify_rings.ts","../src/data/bucket/pattern_bucket_features.ts","../src/data/bucket/fill_bucket.ts","../src/style/style_layer/fill_style_layer_properties.g.ts","../src/style/style_layer/fill_style_layer.ts","../src/data/bucket/fill_extrusion_attributes.ts","../node_modules/@mapbox/vector-tile/lib/vectortilefeature.js","../node_modules/@mapbox/vector-tile/lib/vectortilelayer.js","../node_modules/@mapbox/vector-tile/lib/vectortile.js","../node_modules/@mapbox/vector-tile/index.js","../src/data/bucket/fill_extrusion_bucket.ts","../src/style/style_layer/fill_extrusion_style_layer_properties.g.ts","../src/style/style_layer/fill_extrusion_style_layer.ts","../src/data/bucket/line_attributes.ts","../src/data/bucket/line_attributes_ext.ts","../src/data/bucket/line_bucket.ts","../src/style/style_layer/line_style_layer_properties.g.ts","../src/style/style_layer/line_style_layer.ts","../src/data/bucket/symbol_attributes.ts","../src/symbol/transform_text.ts","../src/util/verticalize_punctuation.ts","../src/symbol/one_em.ts","../node_modules/pbf/index.js","../node_modules/ieee754/index.js","../src/style/parse_glyph_pbf.ts","../node_modules/potpack/index.js","../src/render/image_atlas.ts","../src/symbol/shaping.ts","../src/symbol/symbol_size.ts","../src/style/style_layer/overlap_mode.ts","../src/data/bucket/symbol_bucket.ts","../src/symbol/merge_lines.ts","../src/style/style_layer/symbol_style_layer_properties.g.ts","../src/style/format_section_override.ts","../src/style/style_layer/symbol_style_layer.ts","../src/util/resolve_tokens.ts","../src/style/style_layer/background_style_layer_properties.g.ts","../src/style/style_layer/background_style_layer.ts","../src/style/style_layer/raster_style_layer_properties.g.ts","../src/style/style_layer/raster_style_layer.ts","../src/style/style_layer/custom_style_layer.ts","../src/style/create_style_layer.ts","../src/style/style_layer_index.ts","../src/util/dictionary_coder.ts","../src/util/vectortile_to_geojson.ts","../src/data/feature_index.ts","../src/symbol/anchor.ts","../src/symbol/check_max_angle.ts","../src/symbol/get_anchors.ts","../src/symbol/quads.ts","../src/symbol/collision_feature.ts","../node_modules/tinyqueue/index.js","../src/util/find_pole_of_inaccessibility.ts","../src/style/style_layer/variable_text_anchor.ts","../src/symbol/symbol_layout.ts","../src/symbol/clip_line.ts","../src/render/glyph_atlas.ts","../node_modules/@mapbox/whoots-js/index.mjs","../src/geo/lng_lat.ts","../src/geo/mercator_coordinate.ts","../src/source/tile_id.ts","../src/source/worker_tile.ts","../src/util/performance.ts","../src/source/vector_tile_worker_source.ts","../src/data/dem_data.ts","../src/source/raster_dem_tile_worker_source.ts","../node_modules/@mapbox/geojson-rewind/index.js","../src/source/geojson_wrapper.ts","../node_modules/vt-pbf/lib/geojson_wrapper.js","../node_modules/vt-pbf/index.js","../node_modules/kdbush/index.js","../node_modules/supercluster/index.js","../node_modules/geojson-vt/src/simplify.js","../node_modules/geojson-vt/src/feature.js","../node_modules/geojson-vt/src/convert.js","../node_modules/geojson-vt/src/clip.js","../node_modules/geojson-vt/src/wrap.js","../node_modules/geojson-vt/src/transform.js","../node_modules/geojson-vt/src/tile.js","../node_modules/geojson-vt/src/index.js","../src/source/geojson_source_diff.ts","../src/source/geojson_worker_source.ts","../src/source/worker.ts"],"sourcesContent":["/******************************************************************************\r\nCopyright (c) Microsoft Corporation.\r\n\r\nPermission to use, copy, modify, and/or distribute this software for any\r\npurpose with or without fee is hereby granted.\r\n\r\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\r\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\r\nAND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\r\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\r\nLOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\r\nOTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\r\nPERFORMANCE OF THIS SOFTWARE.\r\n***************************************************************************** */\r\n/* global Reflect, Promise, SuppressedError, Symbol */\r\n\r\nvar extendStatics = function(d, b) {\r\n extendStatics = Object.setPrototypeOf ||\r\n ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\r\n function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };\r\n return extendStatics(d, b);\r\n};\r\n\r\nexport function __extends(d, b) {\r\n if (typeof b !== \"function\" && b !== null)\r\n throw new TypeError(\"Class extends value \" + String(b) + \" is not a constructor or null\");\r\n extendStatics(d, b);\r\n function __() { this.constructor = d; }\r\n d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\r\n}\r\n\r\nexport var __assign = function() {\r\n __assign = Object.assign || function __assign(t) {\r\n for (var s, i = 1, n = arguments.length; i < n; i++) {\r\n s = arguments[i];\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];\r\n }\r\n return t;\r\n }\r\n return __assign.apply(this, arguments);\r\n}\r\n\r\nexport function __rest(s, e) {\r\n var t = {};\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\r\n t[p] = s[p];\r\n if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\r\n for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\r\n if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\r\n t[p[i]] = s[p[i]];\r\n }\r\n return t;\r\n}\r\n\r\nexport function __decorate(decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n}\r\n\r\nexport function __param(paramIndex, decorator) {\r\n return function (target, key) { decorator(target, key, paramIndex); }\r\n}\r\n\r\nexport function __esDecorate(ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {\r\n function accept(f) { if (f !== void 0 && typeof f !== \"function\") throw new TypeError(\"Function expected\"); return f; }\r\n var kind = contextIn.kind, key = kind === \"getter\" ? \"get\" : kind === \"setter\" ? \"set\" : \"value\";\r\n var target = !descriptorIn && ctor ? contextIn[\"static\"] ? ctor : ctor.prototype : null;\r\n var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});\r\n var _, done = false;\r\n for (var i = decorators.length - 1; i >= 0; i--) {\r\n var context = {};\r\n for (var p in contextIn) context[p] = p === \"access\" ? {} : contextIn[p];\r\n for (var p in contextIn.access) context.access[p] = contextIn.access[p];\r\n context.addInitializer = function (f) { if (done) throw new TypeError(\"Cannot add initializers after decoration has completed\"); extraInitializers.push(accept(f || null)); };\r\n var result = (0, decorators[i])(kind === \"accessor\" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);\r\n if (kind === \"accessor\") {\r\n if (result === void 0) continue;\r\n if (result === null || typeof result !== \"object\") throw new TypeError(\"Object expected\");\r\n if (_ = accept(result.get)) descriptor.get = _;\r\n if (_ = accept(result.set)) descriptor.set = _;\r\n if (_ = accept(result.init)) initializers.unshift(_);\r\n }\r\n else if (_ = accept(result)) {\r\n if (kind === \"field\") initializers.unshift(_);\r\n else descriptor[key] = _;\r\n }\r\n }\r\n if (target) Object.defineProperty(target, contextIn.name, descriptor);\r\n done = true;\r\n};\r\n\r\nexport function __runInitializers(thisArg, initializers, value) {\r\n var useValue = arguments.length > 2;\r\n for (var i = 0; i < initializers.length; i++) {\r\n value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);\r\n }\r\n return useValue ? value : void 0;\r\n};\r\n\r\nexport function __propKey(x) {\r\n return typeof x === \"symbol\" ? x : \"\".concat(x);\r\n};\r\n\r\nexport function __setFunctionName(f, name, prefix) {\r\n if (typeof name === \"symbol\") name = name.description ? \"[\".concat(name.description, \"]\") : \"\";\r\n return Object.defineProperty(f, \"name\", { configurable: true, value: prefix ? \"\".concat(prefix, \" \", name) : name });\r\n};\r\n\r\nexport function __metadata(metadataKey, metadataValue) {\r\n if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(metadataKey, metadataValue);\r\n}\r\n\r\nexport function __awaiter(thisArg, _arguments, P, generator) {\r\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\r\n return new (P || (P = Promise))(function (resolve, reject) {\r\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\r\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\r\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\r\n step((generator = generator.apply(thisArg, _arguments || [])).next());\r\n });\r\n}\r\n\r\nexport function __generator(thisArg, body) {\r\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\r\n return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\r\n function verb(n) { return function (v) { return step([n, v]); }; }\r\n function step(op) {\r\n if (f) throw new TypeError(\"Generator is already executing.\");\r\n while (g && (g = 0, op[0] && (_ = 0)), _) try {\r\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\r\n if (y = 0, t) op = [op[0] & 2, t.value];\r\n switch (op[0]) {\r\n case 0: case 1: t = op; break;\r\n case 4: _.label++; return { value: op[1], done: false };\r\n case 5: _.label++; y = op[1]; op = [0]; continue;\r\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\r\n default:\r\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\r\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\r\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\r\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\r\n if (t[2]) _.ops.pop();\r\n _.trys.pop(); continue;\r\n }\r\n op = body.call(thisArg, _);\r\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\r\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\r\n }\r\n}\r\n\r\nexport var __createBinding = Object.create ? (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n var desc = Object.getOwnPropertyDescriptor(m, k);\r\n if (!desc || (\"get\" in desc ? !m.__esModule : desc.writable || desc.configurable)) {\r\n desc = { enumerable: true, get: function() { return m[k]; } };\r\n }\r\n Object.defineProperty(o, k2, desc);\r\n}) : (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n o[k2] = m[k];\r\n});\r\n\r\nexport function __exportStar(m, o) {\r\n for (var p in m) if (p !== \"default\" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p);\r\n}\r\n\r\nexport function __values(o) {\r\n var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\r\n if (m) return m.call(o);\r\n if (o && typeof o.length === \"number\") return {\r\n next: function () {\r\n if (o && i >= o.length) o = void 0;\r\n return { value: o && o[i++], done: !o };\r\n }\r\n };\r\n throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\r\n}\r\n\r\nexport function __read(o, n) {\r\n var m = typeof Symbol === \"function\" && o[Symbol.iterator];\r\n if (!m) return o;\r\n var i = m.call(o), r, ar = [], e;\r\n try {\r\n while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\r\n }\r\n catch (error) { e = { error: error }; }\r\n finally {\r\n try {\r\n if (r && !r.done && (m = i[\"return\"])) m.call(i);\r\n }\r\n finally { if (e) throw e.error; }\r\n }\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spread() {\r\n for (var ar = [], i = 0; i < arguments.length; i++)\r\n ar = ar.concat(__read(arguments[i]));\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spreadArrays() {\r\n for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\r\n for (var r = Array(s), k = 0, i = 0; i < il; i++)\r\n for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\r\n r[k] = a[j];\r\n return r;\r\n}\r\n\r\nexport function __spreadArray(to, from, pack) {\r\n if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {\r\n if (ar || !(i in from)) {\r\n if (!ar) ar = Array.prototype.slice.call(from, 0, i);\r\n ar[i] = from[i];\r\n }\r\n }\r\n return to.concat(ar || Array.prototype.slice.call(from));\r\n}\r\n\r\nexport function __await(v) {\r\n return this instanceof __await ? (this.v = v, this) : new __await(v);\r\n}\r\n\r\nexport function __asyncGenerator(thisArg, _arguments, generator) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var g = generator.apply(thisArg, _arguments || []), i, q = [];\r\n return i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i;\r\n function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }\r\n function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\r\n function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\r\n function fulfill(value) { resume(\"next\", value); }\r\n function reject(value) { resume(\"throw\", value); }\r\n function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\r\n}\r\n\r\nexport function __asyncDelegator(o) {\r\n var i, p;\r\n return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\r\n function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: false } : f ? f(v) : v; } : f; }\r\n}\r\n\r\nexport function __asyncValues(o) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var m = o[Symbol.asyncIterator], i;\r\n return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\r\n function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\r\n function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\r\n}\r\n\r\nexport function __makeTemplateObject(cooked, raw) {\r\n if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\r\n return cooked;\r\n};\r\n\r\nvar __setModuleDefault = Object.create ? (function(o, v) {\r\n Object.defineProperty(o, \"default\", { enumerable: true, value: v });\r\n}) : function(o, v) {\r\n o[\"default\"] = v;\r\n};\r\n\r\nexport function __importStar(mod) {\r\n if (mod && mod.__esModule) return mod;\r\n var result = {};\r\n if (mod != null) for (var k in mod) if (k !== \"default\" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);\r\n __setModuleDefault(result, mod);\r\n return result;\r\n}\r\n\r\nexport function __importDefault(mod) {\r\n return (mod && mod.__esModule) ? mod : { default: mod };\r\n}\r\n\r\nexport function __classPrivateFieldGet(receiver, state, kind, f) {\r\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a getter\");\r\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot read private member from an object whose class did not declare it\");\r\n return kind === \"m\" ? f : kind === \"a\" ? f.call(receiver) : f ? f.value : state.get(receiver);\r\n}\r\n\r\nexport function __classPrivateFieldSet(receiver, state, value, kind, f) {\r\n if (kind === \"m\") throw new TypeError(\"Private method is not writable\");\r\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a setter\");\r\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot write private member to an object whose class did not declare it\");\r\n return (kind === \"a\" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;\r\n}\r\n\r\nexport function __classPrivateFieldIn(state, receiver) {\r\n if (receiver === null || (typeof receiver !== \"object\" && typeof receiver !== \"function\")) throw new TypeError(\"Cannot use 'in' operator on non-object\");\r\n return typeof state === \"function\" ? receiver === state : state.has(receiver);\r\n}\r\n\r\nexport function __addDisposableResource(env, value, async) {\r\n if (value !== null && value !== void 0) {\r\n if (typeof value !== \"object\" && typeof value !== \"function\") throw new TypeError(\"Object expected.\");\r\n var dispose;\r\n if (async) {\r\n if (!Symbol.asyncDispose) throw new TypeError(\"Symbol.asyncDispose is not defined.\");\r\n dispose = value[Symbol.asyncDispose];\r\n }\r\n if (dispose === void 0) {\r\n if (!Symbol.dispose) throw new TypeError(\"Symbol.dispose is not defined.\");\r\n dispose = value[Symbol.dispose];\r\n }\r\n if (typeof dispose !== \"function\") throw new TypeError(\"Object not disposable.\");\r\n env.stack.push({ value: value, dispose: dispose, async: async });\r\n }\r\n else if (async) {\r\n env.stack.push({ async: true });\r\n }\r\n return value;\r\n}\r\n\r\nvar _SuppressedError = typeof SuppressedError === \"function\" ? SuppressedError : function (error, suppressed, message) {\r\n var e = new Error(message);\r\n return e.name = \"SuppressedError\", e.error = error, e.suppressed = suppressed, e;\r\n};\r\n\r\nexport function __disposeResources(env) {\r\n function fail(e) {\r\n env.error = env.hasError ? new _SuppressedError(e, env.error, \"An error was suppressed during disposal.\") : e;\r\n env.hasError = true;\r\n }\r\n function next() {\r\n while (env.stack.length) {\r\n var rec = env.stack.pop();\r\n try {\r\n var result = rec.dispose && rec.dispose.call(rec.value);\r\n if (rec.async) return Promise.resolve(result).then(next, function(e) { fail(e); return next(); });\r\n }\r\n catch (e) {\r\n fail(e);\r\n }\r\n }\r\n if (env.hasError) throw env.error;\r\n }\r\n return next();\r\n}\r\n\r\nexport default {\r\n __extends: __extends,\r\n __assign: __assign,\r\n __rest: __rest,\r\n __decorate: __decorate,\r\n __param: __param,\r\n __metadata: __metadata,\r\n __awaiter: __awaiter,\r\n __generator: __generator,\r\n __createBinding: __createBinding,\r\n __exportStar: __exportStar,\r\n __values: __values,\r\n __read: __read,\r\n __spread: __spread,\r\n __spreadArrays: __spreadArrays,\r\n __spreadArray: __spreadArray,\r\n __await: __await,\r\n __asyncGenerator: __asyncGenerator,\r\n __asyncDelegator: __asyncDelegator,\r\n __asyncValues: __asyncValues,\r\n __makeTemplateObject: __makeTemplateObject,\r\n __importStar: __importStar,\r\n __importDefault: __importDefault,\r\n __classPrivateFieldGet: __classPrivateFieldGet,\r\n __classPrivateFieldSet: __classPrivateFieldSet,\r\n __classPrivateFieldIn: __classPrivateFieldIn,\r\n __addDisposableResource: __addDisposableResource,\r\n __disposeResources: __disposeResources,\r\n};\r\n","'use strict';\n\nmodule.exports = Point;\n\n/**\n * A standalone point geometry with useful accessor, comparison, and\n * modification methods.\n *\n * @class Point\n * @param {Number} x the x-coordinate. this could be longitude or screen\n * pixels, or any other sort of unit.\n * @param {Number} y the y-coordinate. this could be latitude or screen\n * pixels, or any other sort of unit.\n * @example\n * var point = new Point(-77, 38);\n */\nfunction Point(x, y) {\n this.x = x;\n this.y = y;\n}\n\nPoint.prototype = {\n\n /**\n * Clone this point, returning a new point that can be modified\n * without affecting the old one.\n * @return {Point} the clone\n */\n clone: function() { return new Point(this.x, this.y); },\n\n /**\n * Add this point's x & y coordinates to another point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n add: function(p) { return this.clone()._add(p); },\n\n /**\n * Subtract this point's x & y coordinates to from point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n sub: function(p) { return this.clone()._sub(p); },\n\n /**\n * Multiply this point's x & y coordinates by point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n multByPoint: function(p) { return this.clone()._multByPoint(p); },\n\n /**\n * Divide this point's x & y coordinates by point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n divByPoint: function(p) { return this.clone()._divByPoint(p); },\n\n /**\n * Multiply this point's x & y coordinates by a factor,\n * yielding a new point.\n * @param {Point} k factor\n * @return {Point} output point\n */\n mult: function(k) { return this.clone()._mult(k); },\n\n /**\n * Divide this point's x & y coordinates by a factor,\n * yielding a new point.\n * @param {Point} k factor\n * @return {Point} output point\n */\n div: function(k) { return this.clone()._div(k); },\n\n /**\n * Rotate this point around the 0, 0 origin by an angle a,\n * given in radians\n * @param {Number} a angle to rotate around, in radians\n * @return {Point} output point\n */\n rotate: function(a) { return this.clone()._rotate(a); },\n\n /**\n * Rotate this point around p point by an angle a,\n * given in radians\n * @param {Number} a angle to rotate around, in radians\n * @param {Point} p Point to rotate around\n * @return {Point} output point\n */\n rotateAround: function(a,p) { return this.clone()._rotateAround(a,p); },\n\n /**\n * Multiply this point by a 4x1 transformation matrix\n * @param {Array} m transformation matrix\n * @return {Point} output point\n */\n matMult: function(m) { return this.clone()._matMult(m); },\n\n /**\n * Calculate this point but as a unit vector from 0, 0, meaning\n * that the distance from the resulting point to the 0, 0\n * coordinate will be equal to 1 and the angle from the resulting\n * point to the 0, 0 coordinate will be the same as before.\n * @return {Point} unit vector point\n */\n unit: function() { return this.clone()._unit(); },\n\n /**\n * Compute a perpendicular point, where the new y coordinate\n * is the old x coordinate and the new x coordinate is the old y\n * coordinate multiplied by -1\n * @return {Point} perpendicular point\n */\n perp: function() { return this.clone()._perp(); },\n\n /**\n * Return a version of this point with the x & y coordinates\n * rounded to integers.\n * @return {Point} rounded point\n */\n round: function() { return this.clone()._round(); },\n\n /**\n * Return the magitude of this point: this is the Euclidean\n * distance from the 0, 0 coordinate to this point's x and y\n * coordinates.\n * @return {Number} magnitude\n */\n mag: function() {\n return Math.sqrt(this.x * this.x + this.y * this.y);\n },\n\n /**\n * Judge whether this point is equal to another point, returning\n * true or false.\n * @param {Point} other the other point\n * @return {boolean} whether the points are equal\n */\n equals: function(other) {\n return this.x === other.x &&\n this.y === other.y;\n },\n\n /**\n * Calculate the distance from this point to another point\n * @param {Point} p the other point\n * @return {Number} distance\n */\n dist: function(p) {\n return Math.sqrt(this.distSqr(p));\n },\n\n /**\n * Calculate the distance from this point to another point,\n * without the square root step. Useful if you're comparing\n * relative distances.\n * @param {Point} p the other point\n * @return {Number} distance\n */\n distSqr: function(p) {\n var dx = p.x - this.x,\n dy = p.y - this.y;\n return dx * dx + dy * dy;\n },\n\n /**\n * Get the angle from the 0, 0 coordinate to this point, in radians\n * coordinates.\n * @return {Number} angle\n */\n angle: function() {\n return Math.atan2(this.y, this.x);\n },\n\n /**\n * Get the angle from this point to another point, in radians\n * @param {Point} b the other point\n * @return {Number} angle\n */\n angleTo: function(b) {\n return Math.atan2(this.y - b.y, this.x - b.x);\n },\n\n /**\n * Get the angle between this point and another point, in radians\n * @param {Point} b the other point\n * @return {Number} angle\n */\n angleWith: function(b) {\n return this.angleWithSep(b.x, b.y);\n },\n\n /*\n * Find the angle of the two vectors, solving the formula for\n * the cross product a x b = |a||b|sin(θ) for θ.\n * @param {Number} x the x-coordinate\n * @param {Number} y the y-coordinate\n * @return {Number} the angle in radians\n */\n angleWithSep: function(x, y) {\n return Math.atan2(\n this.x * y - this.y * x,\n this.x * x + this.y * y);\n },\n\n _matMult: function(m) {\n var x = m[0] * this.x + m[1] * this.y,\n y = m[2] * this.x + m[3] * this.y;\n this.x = x;\n this.y = y;\n return this;\n },\n\n _add: function(p) {\n this.x += p.x;\n this.y += p.y;\n return this;\n },\n\n _sub: function(p) {\n this.x -= p.x;\n this.y -= p.y;\n return this;\n },\n\n _mult: function(k) {\n this.x *= k;\n this.y *= k;\n return this;\n },\n\n _div: function(k) {\n this.x /= k;\n this.y /= k;\n return this;\n },\n\n _multByPoint: function(p) {\n this.x *= p.x;\n this.y *= p.y;\n return this;\n },\n\n _divByPoint: function(p) {\n this.x /= p.x;\n this.y /= p.y;\n return this;\n },\n\n _unit: function() {\n this._div(this.mag());\n return this;\n },\n\n _perp: function() {\n var y = this.y;\n this.y = this.x;\n this.x = -y;\n return this;\n },\n\n _rotate: function(angle) {\n var cos = Math.cos(angle),\n sin = Math.sin(angle),\n x = cos * this.x - sin * this.y,\n y = sin * this.x + cos * this.y;\n this.x = x;\n this.y = y;\n return this;\n },\n\n _rotateAround: function(angle, p) {\n var cos = Math.cos(angle),\n sin = Math.sin(angle),\n x = p.x + cos * (this.x - p.x) - sin * (this.y - p.y),\n y = p.y + sin * (this.x - p.x) + cos * (this.y - p.y);\n this.x = x;\n this.y = y;\n return this;\n },\n\n _round: function() {\n this.x = Math.round(this.x);\n this.y = Math.round(this.y);\n return this;\n }\n};\n\n/**\n * Construct a point from an array if necessary, otherwise if the input\n * is already a Point, or an unknown type, return it unchanged\n * @param {Array|Point|*} a any kind of input value\n * @return {Point} constructed point, or passed-through value.\n * @example\n * // this\n * var point = Point.convert([0, 1]);\n * // is equivalent to\n * var point = new Point(0, 1);\n */\nPoint.convert = function (a) {\n if (a instanceof Point) {\n return a;\n }\n if (Array.isArray(a)) {\n return new Point(a[0], a[1]);\n }\n return a;\n};\n","'use strict';\n\nmodule.exports = UnitBezier;\n\nfunction UnitBezier(p1x, p1y, p2x, p2y) {\n // Calculate the polynomial coefficients, implicit first and last control points are (0,0) and (1,1).\n this.cx = 3.0 * p1x;\n this.bx = 3.0 * (p2x - p1x) - this.cx;\n this.ax = 1.0 - this.cx - this.bx;\n\n this.cy = 3.0 * p1y;\n this.by = 3.0 * (p2y - p1y) - this.cy;\n this.ay = 1.0 - this.cy - this.by;\n\n this.p1x = p1x;\n this.p1y = p1y;\n this.p2x = p2x;\n this.p2y = p2y;\n}\n\nUnitBezier.prototype = {\n sampleCurveX: function (t) {\n // `ax t^3 + bx t^2 + cx t' expanded using Horner's rule.\n return ((this.ax * t + this.bx) * t + this.cx) * t;\n },\n\n sampleCurveY: function (t) {\n return ((this.ay * t + this.by) * t + this.cy) * t;\n },\n\n sampleCurveDerivativeX: function (t) {\n return (3.0 * this.ax * t + 2.0 * this.bx) * t + this.cx;\n },\n\n solveCurveX: function (x, epsilon) {\n if (epsilon === undefined) epsilon = 1e-6;\n\n if (x < 0.0) return 0.0;\n if (x > 1.0) return 1.0;\n\n var t = x;\n\n // First try a few iterations of Newton's method - normally very fast.\n for (var i = 0; i < 8; i++) {\n var x2 = this.sampleCurveX(t) - x;\n if (Math.abs(x2) < epsilon) return t;\n\n var d2 = this.sampleCurveDerivativeX(t);\n if (Math.abs(d2) < 1e-6) break;\n\n t = t - x2 / d2;\n }\n\n // Fall back to the bisection method for reliability.\n var t0 = 0.0;\n var t1 = 1.0;\n t = x;\n\n for (i = 0; i < 20; i++) {\n x2 = this.sampleCurveX(t);\n if (Math.abs(x2 - x) < epsilon) break;\n\n if (x > x2) {\n t0 = t;\n } else {\n t1 = t;\n }\n\n t = (t1 - t0) * 0.5 + t0;\n }\n\n return t;\n },\n\n solve: function (x, epsilon) {\n return this.sampleCurveY(this.solveCurveX(x, epsilon));\n }\n};\n","let supportsOffscreenCanvas: boolean;\n\nexport function offscreenCanvasSupported(): boolean {\n if (supportsOffscreenCanvas == null) {\n supportsOffscreenCanvas = typeof OffscreenCanvas !== 'undefined' &&\n new OffscreenCanvas(1, 1).getContext('2d') &&\n typeof createImageBitmap === 'function';\n }\n\n return supportsOffscreenCanvas;\n}\n","import {offscreenCanvasSupported} from './offscreen_canvas_supported';\n\nlet offscreenCanvasDistorted: boolean;\n\n/**\n * Some browsers don't return the exact pixels from a canvas to prevent user fingerprinting (see #3185).\n * This function writes pixels to an OffscreenCanvas and reads them back using getImageData, returning false\n * if they don't match.\n *\n * @returns true if the browser supports OffscreenCanvas but it distorts getImageData results, false otherwise.\n */\nexport function isOffscreenCanvasDistorted(): boolean {\n if (offscreenCanvasDistorted == null) {\n offscreenCanvasDistorted = false;\n if (offscreenCanvasSupported()) {\n const size = 5;\n const canvas = new OffscreenCanvas(size, size);\n const context = canvas.getContext('2d', {willReadFrequently: true});\n if (context) {\n // fill each pixel with an RGB value that should make the byte at index i equal to i (except alpha channel):\n // [0, 1, 2, 255, 4, 5, 6, 255, 8, 9, 10, 255, ...]\n for (let i = 0; i < size * size; i++) {\n const base = i * 4;\n context.fillStyle = `rgb(${base},${base + 1},${base + 2})`;\n context.fillRect(i % size, Math.floor(i / size), 1, 1);\n }\n const data = context.getImageData(0, 0, size, size).data;\n for (let i = 0; i < size * size * 4; i++) {\n if (i % 4 !== 3 && data[i] !== i) {\n offscreenCanvasDistorted = true;\n break;\n }\n }\n }\n }\n }\n\n return offscreenCanvasDistorted || false;\n}\n","import Point from '@mapbox/point-geometry';\nimport UnitBezier from '@mapbox/unitbezier';\nimport type {Callback} from '../types/callback';\nimport {isOffscreenCanvasDistorted} from './offscreen_canvas_distorted';\nimport type {Size} from './image';\n\n/**\n * Given a value `t` that varies between 0 and 1, return\n * an interpolation function that eases between 0 and 1 in a pleasing\n * cubic in-out fashion.\n */\nexport function easeCubicInOut(t: number): number {\n if (t <= 0) return 0;\n if (t >= 1) return 1;\n const t2 = t * t,\n t3 = t2 * t;\n return 4 * (t < 0.5 ? t3 : 3 * (t - t2) + t3 - 0.75);\n}\n\n/**\n * Given given (x, y), (x1, y1) control points for a bezier curve,\n * return a function that interpolates along that curve.\n *\n * @param p1x - control point 1 x coordinate\n * @param p1y - control point 1 y coordinate\n * @param p2x - control point 2 x coordinate\n * @param p2y - control point 2 y coordinate\n */\nexport function bezier(p1x: number, p1y: number, p2x: number, p2y: number): (t: number) => number {\n const bezier = new UnitBezier(p1x, p1y, p2x, p2y);\n return function(t: number) {\n return bezier.solve(t);\n };\n}\n\n/**\n * A default bezier-curve powered easing function with\n * control points (0.25, 0.1) and (0.25, 1)\n */\nexport const defaultEasing = bezier(0.25, 0.1, 0.25, 1);\n\n/**\n * constrain n to the given range via min + max\n *\n * @param n - value\n * @param min - the minimum value to be returned\n * @param max - the maximum value to be returned\n * @returns the clamped value\n */\nexport function clamp(n: number, min: number, max: number): number {\n return Math.min(max, Math.max(min, n));\n}\n\n/**\n * constrain n to the given range, excluding the minimum, via modular arithmetic\n *\n * @param n - value\n * @param min - the minimum value to be returned, exclusive\n * @param max - the maximum value to be returned, inclusive\n * @returns constrained number\n */\nexport function wrap(n: number, min: number, max: number): number {\n const d = max - min;\n const w = ((n - min) % d + d) % d + min;\n return (w === min) ? max : w;\n}\n\n/**\n * Call an asynchronous function on an array of arguments,\n * calling `callback` with the completed results of all calls.\n *\n * @param array - input to each call of the async function.\n * @param fn - an async function with signature (data, callback)\n * @param callback - a callback run after all async work is done.\n * called with an array, containing the results of each async call.\n */\nexport function asyncAll(\n array: Array,\n fn: (item: Item, fnCallback: Callback) => void,\n callback: Callback>\n) {\n if (!array.length) { return callback(null, []); }\n let remaining = array.length;\n const results = new Array(array.length);\n let error = null;\n array.forEach((item, i) => {\n fn(item, (err, result) => {\n if (err) error = err;\n results[i] = (result as any as Result); // https://github.com/facebook/flow/issues/2123\n if (--remaining === 0) callback(error, results);\n });\n });\n}\n\n/**\n * Compute the difference between the keys in one object and the keys\n * in another object.\n *\n * @returns keys difference\n */\nexport function keysDifference(\n obj: {[key: string]: S},\n other: {[key: string]: T}\n): Array {\n const difference = [];\n for (const i in obj) {\n if (!(i in other)) {\n difference.push(i);\n }\n }\n return difference;\n}\n\n/**\n * Given a destination object and optionally many source objects,\n * copy all properties from the source objects into the destination.\n * The last source object given overrides properties from previous\n * source objects.\n *\n * @param dest - destination object\n * @param sources - sources from which properties are pulled\n */\nexport function extend(dest: any, ...sources: Array): any {\n for (const src of sources) {\n for (const k in src) {\n dest[k] = src[k];\n }\n }\n return dest;\n}\n\n/**\n * Given an object and a number of properties as strings, return version\n * of that object with only those properties.\n *\n * @param src - the object\n * @param properties - an array of property names chosen\n * to appear on the resulting object.\n * @returns object with limited properties.\n * @example\n * ```ts\n * let foo = { name: 'Charlie', age: 10 };\n * let justName = pick(foo, ['name']); // justName = { name: 'Charlie' }\n * ```\n */\nexport function pick(src: any, properties: Array): any {\n const result = {};\n for (let i = 0; i < properties.length; i++) {\n const k = properties[i];\n if (k in src) {\n result[k] = src[k];\n }\n }\n return result;\n}\n\nlet id = 1;\n\n/**\n * Return a unique numeric id, starting at 1 and incrementing with\n * each call.\n *\n * @returns unique numeric id.\n */\nexport function uniqueId(): number {\n return id++;\n}\n\n/**\n * Return whether a given value is a power of two\n */\nexport function isPowerOfTwo(value: number): boolean {\n return (Math.log(value) / Math.LN2) % 1 === 0;\n}\n\n/**\n * Return the next power of two, or the input value if already a power of two\n */\nexport function nextPowerOfTwo(value: number): number {\n if (value <= 1) return 1;\n return Math.pow(2, Math.ceil(Math.log(value) / Math.LN2));\n}\n\n/**\n * Create an object by mapping all the values of an existing object while\n * preserving their keys.\n */\nexport function mapObject(input: any, iterator: Function, context?: any): any {\n const output = {};\n for (const key in input) {\n output[key] = iterator.call(context || this, input[key], key, input);\n }\n return output;\n}\n\n/**\n * Create an object by filtering out values of an existing object.\n */\nexport function filterObject(input: any, iterator: Function, context?: any): any {\n const output = {};\n for (const key in input) {\n if (iterator.call(context || this, input[key], key, input)) {\n output[key] = input[key];\n }\n }\n return output;\n}\n\n/**\n * Deeply compares two object literals.\n * @param a - first object literal to be compared\n * @param b - second object literal to be compared\n * @returns true if the two object literals are deeply equal, false otherwise\n */\nexport function deepEqual(a?: unknown | null, b?: unknown | null): boolean {\n if (Array.isArray(a)) {\n if (!Array.isArray(b) || a.length !== b.length) return false;\n for (let i = 0; i < a.length; i++) {\n if (!deepEqual(a[i], b[i])) return false;\n }\n return true;\n }\n if (typeof a === 'object' && a !== null && b !== null) {\n if (!(typeof b === 'object')) return false;\n const keys = Object.keys(a);\n if (keys.length !== Object.keys(b).length) return false;\n for (const key in a) {\n if (!deepEqual(a[key], b[key])) return false;\n }\n return true;\n }\n return a === b;\n}\n\n/**\n * Deeply clones two objects.\n */\nexport function clone(input: T): T {\n if (Array.isArray(input)) {\n return input.map(clone) as any as T;\n } else if (typeof input === 'object' && input) {\n return mapObject(input, clone) as any as T;\n } else {\n return input;\n }\n}\n\n/**\n * Check if two arrays have at least one common element.\n */\nexport function arraysIntersect(a: Array, b: Array): boolean {\n for (let l = 0; l < a.length; l++) {\n if (b.indexOf(a[l]) >= 0) return true;\n }\n return false;\n}\n\n/**\n * Print a warning message to the console and ensure duplicate warning messages\n * are not printed.\n */\nconst warnOnceHistory: {[key: string]: boolean} = {};\n\nexport function warnOnce(message: string): void {\n if (!warnOnceHistory[message]) {\n // console isn't defined in some WebWorkers, see #2558\n if (typeof console !== 'undefined') console.warn(message);\n warnOnceHistory[message] = true;\n }\n}\n\n/**\n * Indicates if the provided Points are in a counter clockwise (true) or clockwise (false) order\n *\n * @returns true for a counter clockwise set of points\n */\n// http://bryceboe.com/2006/10/23/line-segment-intersection-algorithm/\nexport function isCounterClockwise(a: Point, b: Point, c: Point): boolean {\n return (c.y - a.y) * (b.x - a.x) > (b.y - a.y) * (c.x - a.x);\n}\n\n/**\n * For two lines a and b in 2d space, defined by any two points along the lines,\n * find the intersection point, or return null if the lines are parallel\n *\n * @param a1 - First point on line a\n * @param a2 - Second point on line a\n * @param b1 - First point on line b\n * @param b2 - Second point on line b\n *\n * @returns the intersection point of the two lines or null if they are parallel\n */\nexport function findLineIntersection(a1: Point, a2: Point, b1: Point, b2: Point): Point | null {\n const aDeltaY = a2.y - a1.y;\n const aDeltaX = a2.x - a1.x;\n const bDeltaY = b2.y - b1.y;\n const bDeltaX = b2.x - b1.x;\n\n const denominator = (bDeltaY * aDeltaX) - (bDeltaX * aDeltaY);\n\n if (denominator === 0) {\n // Lines are parallel\n return null;\n }\n\n const originDeltaY = a1.y - b1.y;\n const originDeltaX = a1.x - b1.x;\n const aInterpolation = (bDeltaX * originDeltaY - bDeltaY * originDeltaX) / denominator;\n\n // Find intersection by projecting out from origin of first segment\n return new Point(a1.x + (aInterpolation * aDeltaX), a1.y + (aInterpolation * aDeltaY));\n}\n\n/**\n * Returns the signed area for the polygon ring. Positive areas are exterior rings and\n * have a clockwise winding. Negative areas are interior rings and have a counter clockwise\n * ordering.\n *\n * @param ring - Exterior or interior ring\n */\nexport function calculateSignedArea(ring: Array): number {\n let sum = 0;\n for (let i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) {\n p1 = ring[i];\n p2 = ring[j];\n sum += (p2.x - p1.x) * (p1.y + p2.y);\n }\n return sum;\n}\n\n/**\n * Detects closed polygons, first + last point are equal\n *\n * @param points - array of points\n * @returns `true` if the points are a closed polygon\n */\nexport function isClosedPolygon(points: Array): boolean {\n // If it is 2 points that are the same then it is a point\n // If it is 3 points with start and end the same then it is a line\n if (points.length < 4)\n return false;\n\n const p1 = points[0];\n const p2 = points[points.length - 1];\n\n if (Math.abs(p1.x - p2.x) > 0 ||\n Math.abs(p1.y - p2.y) > 0) {\n return false;\n }\n\n // polygon simplification can produce polygons with zero area and more than 3 points\n return Math.abs(calculateSignedArea(points)) > 0.01;\n}\n\n/**\n * Converts spherical coordinates to cartesian coordinates.\n *\n * @param spherical - Spherical coordinates, in [radial, azimuthal, polar]\n * @returns cartesian coordinates in [x, y, z]\n */\n\nexport function sphericalToCartesian([r, azimuthal, polar]: [number, number, number]): {\n x: number;\n y: number;\n z: number;\n} {\n // We abstract \"north\"/\"up\" (compass-wise) to be 0° when really this is 90° (π/2):\n // correct for that here\n azimuthal += 90;\n\n // Convert azimuthal and polar angles to radians\n azimuthal *= Math.PI / 180;\n polar *= Math.PI / 180;\n\n return {\n x: r * Math.cos(azimuthal) * Math.sin(polar),\n y: r * Math.sin(azimuthal) * Math.sin(polar),\n z: r * Math.cos(polar)\n };\n}\n\n/**\n * Returns true if the when run in the web-worker context.\n *\n * @returns `true` if the when run in the web-worker context.\n */\nexport function isWorker(): boolean {\n // @ts-ignore\n return typeof WorkerGlobalScope !== 'undefined' && typeof self !== 'undefined' && self instanceof WorkerGlobalScope;\n}\n\n/**\n * Parses data from 'Cache-Control' headers.\n *\n * @param cacheControl - Value of 'Cache-Control' header\n * @returns object containing parsed header info.\n */\n\nexport function parseCacheControl(cacheControl: string): any {\n // Taken from [Wreck](https://github.com/hapijs/wreck)\n const re = /(?:^|(?:\\s*\\,\\s*))([^\\x00-\\x20\\(\\)<>@\\,;\\:\\\\\"\\/\\[\\]\\?\\=\\{\\}\\x7F]+)(?:\\=(?:([^\\x00-\\x20\\(\\)<>@\\,;\\:\\\\\"\\/\\[\\]\\?\\=\\{\\}\\x7F]+)|(?:\\\"((?:[^\"\\\\]|\\\\.)*)\\\")))?/g;\n\n const header = {};\n cacheControl.replace(re, ($0, $1, $2, $3) => {\n const value = $2 || $3;\n header[$1] = value ? value.toLowerCase() : true;\n return '';\n });\n\n if (header['max-age']) {\n const maxAge = parseInt(header['max-age'], 10);\n if (isNaN(maxAge)) delete header['max-age'];\n else header['max-age'] = maxAge;\n }\n\n return header;\n}\n\nlet _isSafari = null;\n\n/**\n * Returns true when run in WebKit derived browsers.\n * This is used as a workaround for a memory leak in Safari caused by using Transferable objects to\n * transfer data between WebWorkers and the main thread.\n * https://github.com/mapbox/mapbox-gl-js/issues/8771\n *\n * This should be removed once the underlying Safari issue is fixed.\n *\n * @param scope - Since this function is used both on the main thread and WebWorker context,\n * let the calling scope pass in the global scope object.\n * @returns `true` when run in WebKit derived browsers.\n */\nexport function isSafari(scope: any): boolean {\n if (_isSafari == null) {\n const userAgent = scope.navigator ? scope.navigator.userAgent : null;\n _isSafari = !!scope.safari ||\n !!(userAgent && (/\\b(iPad|iPhone|iPod)\\b/.test(userAgent) || (!!userAgent.match('Safari') && !userAgent.match('Chrome'))));\n }\n return _isSafari;\n}\n\nexport function storageAvailable(type: string): boolean {\n try {\n const storage = window[type];\n storage.setItem('_mapbox_test_', 1);\n storage.removeItem('_mapbox_test_');\n return true;\n } catch (e) {\n return false;\n }\n}\n\n// The following methods are from https://developer.mozilla.org/en-US/docs/Web/API/WindowBase64/Base64_encoding_and_decoding#The_Unicode_Problem\n//Unicode compliant base64 encoder for strings\nexport function b64EncodeUnicode(str: string) {\n return btoa(\n encodeURIComponent(str).replace(/%([0-9A-F]{2})/g,\n (match, p1) => {\n return String.fromCharCode(Number('0x' + p1)); //eslint-disable-line\n }\n )\n );\n}\n\n// Unicode compliant decoder for base64-encoded strings\nexport function b64DecodeUnicode(str: string) {\n return decodeURIComponent(atob(str).split('').map((c) => {\n return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2); //eslint-disable-line\n }).join(''));\n}\n\nexport function isImageBitmap(image: any): image is ImageBitmap {\n return typeof ImageBitmap !== 'undefined' && image instanceof ImageBitmap;\n}\n\n/**\n * Converts an ArrayBuffer to an ImageBitmap.\n *\n * Used mostly for testing purposes only, because mocking libs don't know how to work with ArrayBuffers, but work\n * perfectly fine with ImageBitmaps. Might also be used for environments (other than testing) not supporting\n * ArrayBuffers.\n *\n * @param data - Data to convert\n * @param callback - A callback executed after the conversion is finished. Invoked with error (if any) as the first argument and resulting image bitmap (when no error) as the second\n */\nexport function arrayBufferToImageBitmap(data: ArrayBuffer, callback: (err?: Error | null, image?: ImageBitmap | null) => void) {\n const blob: Blob = new Blob([new Uint8Array(data)], {type: 'image/png'});\n createImageBitmap(blob).then((imgBitmap) => {\n callback(null, imgBitmap);\n }).catch((e) => {\n callback(new Error(`Could not load image because of ${e.message}. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.`));\n });\n}\n\nconst transparentPngUrl = '';\n\n/**\n * Converts an ArrayBuffer to an HTMLImageElement.\n *\n * Used mostly for testing purposes only, because mocking libs don't know how to work with ArrayBuffers, but work\n * perfectly fine with ImageBitmaps. Might also be used for environments (other than testing) not supporting\n * ArrayBuffers.\n *\n * @param data - Data to convert\n * @param callback - A callback executed after the conversion is finished. Invoked with error (if any) as the first argument and resulting image element (when no error) as the second\n */\nexport function arrayBufferToImage(data: ArrayBuffer, callback: (err?: Error | null, image?: HTMLImageElement | null) => void) {\n const img: HTMLImageElement = new Image();\n img.onload = () => {\n callback(null, img);\n URL.revokeObjectURL(img.src);\n // prevent image dataURI memory leak in Safari;\n // but don't free the image immediately because it might be uploaded in the next frame\n // https://github.com/mapbox/mapbox-gl-js/issues/10226\n img.onload = null;\n window.requestAnimationFrame(() => { img.src = transparentPngUrl; });\n };\n img.onerror = () => callback(new Error('Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.'));\n const blob: Blob = new Blob([new Uint8Array(data)], {type: 'image/png'});\n img.src = data.byteLength ? URL.createObjectURL(blob) : transparentPngUrl;\n}\n\n/**\n * Computes the webcodecs VideoFrame API options to select a rectangle out of\n * an image and write it into the destination rectangle.\n *\n * Rect (x/y/width/height) select the overlapping rectangle from the source image\n * and layout (offset/stride) write that overlapping rectangle to the correct place\n * in the destination image.\n *\n * Offset is the byte offset in the dest image that the first pixel appears at\n * and stride is the number of bytes to the start of the next row:\n * ┌───────────┐\n * │ dest │\n * │ ┌───┼───────┐\n * │offset→│▓▓▓│ source│\n * │ │▓▓▓│ │\n * │ └───┼───────┘\n * │stride ⇠╌╌╌│\n * │╌╌╌╌╌╌→ │\n * └───────────┘\n *\n * @param image - source image containing a width and height attribute\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns the layout and rect options to pass into VideoFrame API\n */\nfunction computeVideoFrameParameters(image: Size, x: number, y: number, width: number, height: number): VideoFrameCopyToOptions {\n const destRowOffset = Math.max(-x, 0) * 4;\n const firstSourceRow = Math.max(0, y);\n const firstDestRow = firstSourceRow - y;\n const offset = firstDestRow * width * 4 + destRowOffset;\n const stride = width * 4;\n\n const sourceLeft = Math.max(0, x);\n const sourceTop = Math.max(0, y);\n const sourceRight = Math.min(image.width, x + width);\n const sourceBottom = Math.min(image.height, y + height);\n return {\n rect: {\n x: sourceLeft,\n y: sourceTop,\n width: sourceRight - sourceLeft,\n height: sourceBottom - sourceTop\n },\n layout: [{offset, stride}]\n };\n}\n\n/**\n * Reads pixels from an ImageBitmap/Image/canvas using webcodec VideoFrame API.\n *\n * @param data - image, imagebitmap, or canvas to parse\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns a promise containing the parsed RGBA pixel values of the image, or the error if an error occurred\n */\nexport async function readImageUsingVideoFrame(\n image: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas,\n x: number, y: number, width: number, height: number\n): Promise {\n if (typeof VideoFrame === 'undefined') {\n throw new Error('VideoFrame not supported');\n }\n const frame = new VideoFrame(image, {timestamp: 0});\n try {\n const format = frame?.format;\n if (!format || !(format.startsWith('BGR') || format.startsWith('RGB'))) {\n throw new Error(`Unrecognized format ${format}`);\n }\n const swapBR = format.startsWith('BGR');\n const result = new Uint8ClampedArray(width * height * 4);\n await frame.copyTo(result, computeVideoFrameParameters(image, x, y, width, height));\n if (swapBR) {\n for (let i = 0; i < result.length; i += 4) {\n const tmp = result[i];\n result[i] = result[i + 2];\n result[i + 2] = tmp;\n }\n }\n return result;\n } finally {\n frame.close();\n }\n}\n\nlet offscreenCanvas: OffscreenCanvas;\nlet offscreenCanvasContext: OffscreenCanvasRenderingContext2D;\n\n/**\n * Reads pixels from an ImageBitmap/Image/canvas using OffscreenCanvas\n *\n * @param data - image, imagebitmap, or canvas to parse\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns a promise containing the parsed RGBA pixel values of the image, or the error if an error occurred\n */\nexport function readImageDataUsingOffscreenCanvas(\n imgBitmap: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas,\n x: number, y: number, width: number, height: number\n): Uint8ClampedArray {\n const origWidth = imgBitmap.width;\n const origHeight = imgBitmap.height;\n // Lazily initialize OffscreenCanvas\n if (!offscreenCanvas || !offscreenCanvasContext) {\n // Dem tiles are typically 256x256\n offscreenCanvas = new OffscreenCanvas(origWidth, origHeight);\n offscreenCanvasContext = offscreenCanvas.getContext('2d', {willReadFrequently: true});\n }\n\n offscreenCanvas.width = origWidth;\n offscreenCanvas.height = origHeight;\n\n offscreenCanvasContext.drawImage(imgBitmap, 0, 0, origWidth, origHeight);\n const imgData = offscreenCanvasContext.getImageData(x, y, width, height);\n offscreenCanvasContext.clearRect(0, 0, origWidth, origHeight);\n return imgData.data;\n}\n\n/**\n * Reads RGBA pixels from an preferring OffscreenCanvas, but falling back to VideoFrame if supported and\n * the browser is mangling OffscreenCanvas getImageData results.\n *\n * @param data - image, imagebitmap, or canvas to parse\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns a promise containing the parsed RGBA pixel values of the image\n */\nexport async function getImageData(\n image: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas,\n x: number, y: number, width: number, height: number\n): Promise {\n if (isOffscreenCanvasDistorted()) {\n try {\n return await readImageUsingVideoFrame(image, x, y, width, height);\n } catch (e) {\n // fall back to OffscreenCanvas\n }\n }\n return readImageDataUsingOffscreenCanvas(image, x, y, width, height);\n}\n","/*\nThis file was copied from https://github.com/mapbox/grid-index and was\nmigrated from JavaScript to TypeScript.\n\nCopyright (c) 2016, Mapbox\n\nPermission to use, copy, modify, and/or distribute this software for any purpose\nwith or without fee is hereby granted, provided that the above copyright notice\nand this permission notice appear in all copies.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND\nFITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS\nOF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER\nTORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF\nTHIS SOFTWARE.\n*/\n\nconst NUM_PARAMS = 3;\n\nexport type SerializedGrid = {\n buffer: ArrayBuffer;\n};\n\nexport class TransferableGridIndex {\n cells: number[][];\n arrayBuffer: ArrayBuffer;\n d: number;\n keys: number[];\n bboxes: number[];\n n: number;\n extent: number;\n padding: number;\n scale: any;\n uid: number;\n min: number;\n max: number;\n\n constructor(extent: number | ArrayBuffer, n?: number, padding?: number) {\n const cells = this.cells = [];\n\n if (extent instanceof ArrayBuffer) {\n this.arrayBuffer = extent;\n const array = new Int32Array(this.arrayBuffer);\n extent = array[0];\n n = array[1];\n padding = array[2];\n\n this.d = n + 2 * padding;\n for (let k = 0; k < this.d * this.d; k++) {\n const start = array[NUM_PARAMS + k];\n const end = array[NUM_PARAMS + k + 1];\n cells.push(start === end ? null : array.subarray(start, end));\n }\n const keysOffset = array[NUM_PARAMS + cells.length];\n const bboxesOffset = array[NUM_PARAMS + cells.length + 1];\n this.keys = array.subarray(keysOffset, bboxesOffset) as any as number[];\n this.bboxes = array.subarray(bboxesOffset) as any as number[];\n\n this.insert = this._insertReadonly;\n\n } else {\n this.d = n + 2 * padding;\n for (let i = 0; i < this.d * this.d; i++) {\n cells.push([]);\n }\n this.keys = [];\n this.bboxes = [];\n }\n\n this.n = n;\n this.extent = extent;\n this.padding = padding;\n this.scale = n / extent;\n this.uid = 0;\n\n const p = (padding / n) * extent;\n this.min = -p;\n this.max = extent + p;\n }\n\n insert(key: number, x1: number, y1: number, x2: number, y2: number) {\n this._forEachCell(x1, y1, x2, y2, this._insertCell, this.uid++, undefined, undefined);\n this.keys.push(key);\n this.bboxes.push(x1);\n this.bboxes.push(y1);\n this.bboxes.push(x2);\n this.bboxes.push(y2);\n }\n\n _insertReadonly() {\n throw new Error('Cannot insert into a GridIndex created from an ArrayBuffer.');\n }\n\n _insertCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) {\n this.cells[cellIndex].push(uid);\n }\n\n query(x1: number, y1: number, x2: number, y2: number, intersectionTest?: Function): number[] {\n const min = this.min;\n const max = this.max;\n if (x1 <= min && y1 <= min && max <= x2 && max <= y2 && !intersectionTest) {\n // We use `Array#slice` because `this.keys` may be a `Int32Array` and\n // some browsers (Safari and IE) do not support `TypedArray#slice`\n // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/TypedArray/slice#Browser_compatibility\n return Array.prototype.slice.call(this.keys);\n\n } else {\n const result = [];\n const seenUids = {};\n this._forEachCell(x1, y1, x2, y2, this._queryCell, result, seenUids, intersectionTest);\n return result;\n }\n }\n\n _queryCell(x1: number, y1: number, x2: number, y2:number, cellIndex:number, result, seenUids, intersectionTest: Function) {\n const cell = this.cells[cellIndex];\n if (cell !== null) {\n const keys = this.keys;\n const bboxes = this.bboxes;\n for (let u = 0; u < cell.length; u++) {\n const uid = cell[u];\n if (seenUids[uid] === undefined) {\n const offset = uid * 4;\n if (intersectionTest ?\n intersectionTest(bboxes[offset + 0], bboxes[offset + 1], bboxes[offset + 2], bboxes[offset + 3]) :\n ((x1 <= bboxes[offset + 2]) &&\n (y1 <= bboxes[offset + 3]) &&\n (x2 >= bboxes[offset + 0]) &&\n (y2 >= bboxes[offset + 1]))) {\n seenUids[uid] = true;\n result.push(keys[uid]);\n } else {\n seenUids[uid] = false;\n }\n }\n }\n }\n }\n\n _forEachCell(x1: number, y1: number, x2:number, y2:number, fn: Function, arg1, arg2, intersectionTest) {\n const cx1 = this._convertToCellCoord(x1);\n const cy1 = this._convertToCellCoord(y1);\n const cx2 = this._convertToCellCoord(x2);\n const cy2 = this._convertToCellCoord(y2);\n for (let x = cx1; x <= cx2; x++) {\n for (let y = cy1; y <= cy2; y++) {\n const cellIndex = this.d * y + x;\n if (intersectionTest && !intersectionTest(\n this._convertFromCellCoord(x),\n this._convertFromCellCoord(y),\n this._convertFromCellCoord(x + 1),\n this._convertFromCellCoord(y + 1))) continue;\n if (fn.call(this, x1, y1, x2, y2, cellIndex, arg1, arg2, intersectionTest)) return;\n }\n }\n }\n\n _convertFromCellCoord (x) {\n return (x - this.padding) / this.scale;\n }\n\n _convertToCellCoord(x) {\n return Math.max(0, Math.min(this.d - 1, Math.floor(x * this.scale) + this.padding));\n }\n\n toArrayBuffer(): ArrayBuffer {\n if (this.arrayBuffer) return this.arrayBuffer;\n\n const cells = this.cells;\n\n const metadataLength = NUM_PARAMS + this.cells.length + 1 + 1;\n let totalCellLength = 0;\n for (let i = 0; i < this.cells.length; i++) {\n totalCellLength += this.cells[i].length;\n }\n\n const array = new Int32Array(metadataLength + totalCellLength + this.keys.length + this.bboxes.length);\n array[0] = this.extent;\n array[1] = this.n;\n array[2] = this.padding;\n\n let offset = metadataLength;\n for (let k = 0; k < cells.length; k++) {\n const cell = cells[k];\n array[NUM_PARAMS + k] = offset;\n array.set(cell, offset);\n offset += cell.length;\n }\n\n array[NUM_PARAMS + cells.length] = offset;\n array.set(this.keys, offset);\n offset += this.keys.length;\n\n array[NUM_PARAMS + cells.length + 1] = offset;\n array.set(this.bboxes, offset);\n offset += this.bboxes.length;\n\n return array.buffer;\n }\n\n public static serialize(grid: TransferableGridIndex, transferables?: Array): SerializedGrid {\n const buffer = grid.toArrayBuffer();\n if (transferables) {\n transferables.push(buffer);\n }\n return {buffer};\n }\n\n public static deserialize(serialized: SerializedGrid): TransferableGridIndex {\n return new TransferableGridIndex(serialized.buffer);\n }\n}\n","import UnitBezier from '@mapbox/unitbezier';\n\nvar $version = 8;\nvar $root = {\n\tversion: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: [\n\t\t\t8\n\t\t]\n\t},\n\tname: {\n\t\ttype: \"string\"\n\t},\n\tmetadata: {\n\t\ttype: \"*\"\n\t},\n\tcenter: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\"\n\t},\n\tzoom: {\n\t\ttype: \"number\"\n\t},\n\tbearing: {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\tunits: \"degrees\"\n\t},\n\tpitch: {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tunits: \"degrees\"\n\t},\n\tlight: {\n\t\ttype: \"light\"\n\t},\n\tterrain: {\n\t\ttype: \"terrain\"\n\t},\n\tsources: {\n\t\trequired: true,\n\t\ttype: \"sources\"\n\t},\n\tsprite: {\n\t\ttype: \"sprite\"\n\t},\n\tglyphs: {\n\t\ttype: \"string\"\n\t},\n\ttransition: {\n\t\ttype: \"transition\"\n\t},\n\tlayers: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tvalue: \"layer\"\n\t}\n};\nvar sources = {\n\t\"*\": {\n\t\ttype: \"source\"\n\t}\n};\nvar source = [\n\t\"source_vector\",\n\t\"source_raster\",\n\t\"source_raster_dem\",\n\t\"source_geojson\",\n\t\"source_video\",\n\t\"source_image\"\n];\nvar source_vector = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvector: {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\ttype: \"string\"\n\t},\n\ttiles: {\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tbounds: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t-180,\n\t\t\t-85.051129,\n\t\t\t180,\n\t\t\t85.051129\n\t\t]\n\t},\n\tscheme: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\txyz: {\n\t\t\t},\n\t\t\ttms: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"xyz\"\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 22\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tpromoteId: {\n\t\ttype: \"promoteId\"\n\t},\n\tvolatile: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\t\"*\": {\n\t\ttype: \"*\"\n\t}\n};\nvar source_raster = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\traster: {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\ttype: \"string\"\n\t},\n\ttiles: {\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tbounds: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t-180,\n\t\t\t-85.051129,\n\t\t\t180,\n\t\t\t85.051129\n\t\t]\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 22\n\t},\n\ttileSize: {\n\t\ttype: \"number\",\n\t\t\"default\": 512,\n\t\tunits: \"pixels\"\n\t},\n\tscheme: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\txyz: {\n\t\t\t},\n\t\t\ttms: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"xyz\"\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tvolatile: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\t\"*\": {\n\t\ttype: \"*\"\n\t}\n};\nvar source_raster_dem = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\t\"raster-dem\": {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\ttype: \"string\"\n\t},\n\ttiles: {\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tbounds: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t-180,\n\t\t\t-85.051129,\n\t\t\t180,\n\t\t\t85.051129\n\t\t]\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 22\n\t},\n\ttileSize: {\n\t\ttype: \"number\",\n\t\t\"default\": 512,\n\t\tunits: \"pixels\"\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tencoding: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tterrarium: {\n\t\t\t},\n\t\t\tmapbox: {\n\t\t\t},\n\t\t\tcustom: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"mapbox\"\n\t},\n\tredFactor: {\n\t\ttype: \"number\",\n\t\t\"default\": 1\n\t},\n\tblueFactor: {\n\t\ttype: \"number\",\n\t\t\"default\": 1\n\t},\n\tgreenFactor: {\n\t\ttype: \"number\",\n\t\t\"default\": 1\n\t},\n\tbaseShift: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tvolatile: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\t\"*\": {\n\t\ttype: \"*\"\n\t}\n};\nvar source_geojson = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tgeojson: {\n\t\t\t}\n\t\t}\n\t},\n\tdata: {\n\t\trequired: true,\n\t\ttype: \"*\"\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 18\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tbuffer: {\n\t\ttype: \"number\",\n\t\t\"default\": 128,\n\t\tmaximum: 512,\n\t\tminimum: 0\n\t},\n\tfilter: {\n\t\ttype: \"*\"\n\t},\n\ttolerance: {\n\t\ttype: \"number\",\n\t\t\"default\": 0.375\n\t},\n\tcluster: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\tclusterRadius: {\n\t\ttype: \"number\",\n\t\t\"default\": 50,\n\t\tminimum: 0\n\t},\n\tclusterMaxZoom: {\n\t\ttype: \"number\"\n\t},\n\tclusterMinPoints: {\n\t\ttype: \"number\"\n\t},\n\tclusterProperties: {\n\t\ttype: \"*\"\n\t},\n\tlineMetrics: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\tgenerateId: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\tpromoteId: {\n\t\ttype: \"promoteId\"\n\t}\n};\nvar source_video = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvideo: {\n\t\t\t}\n\t\t}\n\t},\n\turls: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tcoordinates: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tlength: 4,\n\t\tvalue: {\n\t\t\ttype: \"array\",\n\t\t\tlength: 2,\n\t\t\tvalue: \"number\"\n\t\t}\n\t}\n};\nvar source_image = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\timage: {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\trequired: true,\n\t\ttype: \"string\"\n\t},\n\tcoordinates: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tlength: 4,\n\t\tvalue: {\n\t\t\ttype: \"array\",\n\t\t\tlength: 2,\n\t\t\tvalue: \"number\"\n\t\t}\n\t}\n};\nvar layer = {\n\tid: {\n\t\ttype: \"string\",\n\t\trequired: true\n\t},\n\ttype: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tfill: {\n\t\t\t},\n\t\t\tline: {\n\t\t\t},\n\t\t\tsymbol: {\n\t\t\t},\n\t\t\tcircle: {\n\t\t\t},\n\t\t\theatmap: {\n\t\t\t},\n\t\t\t\"fill-extrusion\": {\n\t\t\t},\n\t\t\traster: {\n\t\t\t},\n\t\t\thillshade: {\n\t\t\t},\n\t\t\tbackground: {\n\t\t\t}\n\t\t},\n\t\trequired: true\n\t},\n\tmetadata: {\n\t\ttype: \"*\"\n\t},\n\tsource: {\n\t\ttype: \"string\"\n\t},\n\t\"source-layer\": {\n\t\ttype: \"string\"\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\tminimum: 0,\n\t\tmaximum: 24\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\tminimum: 0,\n\t\tmaximum: 24\n\t},\n\tfilter: {\n\t\ttype: \"filter\"\n\t},\n\tlayout: {\n\t\ttype: \"layout\"\n\t},\n\tpaint: {\n\t\ttype: \"paint\"\n\t}\n};\nvar layout = [\n\t\"layout_fill\",\n\t\"layout_line\",\n\t\"layout_circle\",\n\t\"layout_heatmap\",\n\t\"layout_fill-extrusion\",\n\t\"layout_symbol\",\n\t\"layout_raster\",\n\t\"layout_hillshade\",\n\t\"layout_background\"\n];\nvar layout_background = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_fill = {\n\t\"fill-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_circle = {\n\t\"circle-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_heatmap = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_line = {\n\t\"line-cap\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tbutt: {\n\t\t\t},\n\t\t\tround: {\n\t\t\t},\n\t\t\tsquare: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"butt\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-join\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tbevel: {\n\t\t\t},\n\t\t\tround: {\n\t\t\t},\n\t\t\tmiter: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"miter\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-miter-limit\": {\n\t\ttype: \"number\",\n\t\t\"default\": 2,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"line-join\": \"miter\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-round-limit\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1.05,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"line-join\": \"round\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_symbol = {\n\t\"symbol-placement\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tpoint: {\n\t\t\t},\n\t\t\tline: {\n\t\t\t},\n\t\t\t\"line-center\": {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"point\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"symbol-spacing\": {\n\t\ttype: \"number\",\n\t\t\"default\": 250,\n\t\tminimum: 1,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"symbol-placement\": \"line\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"symbol-avoid-edges\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"symbol-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"symbol-z-order\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tauto: {\n\t\t\t},\n\t\t\t\"viewport-y\": {\n\t\t\t},\n\t\t\tsource: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-allow-overlap\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t{\n\t\t\t\t\"!\": \"icon-overlap\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-overlap\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnever: {\n\t\t\t},\n\t\t\talways: {\n\t\t\t},\n\t\t\tcooperative: {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-ignore-placement\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-optional\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-rotation-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-size\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tunits: \"factor of the original icon size\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-text-fit\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnone: {\n\t\t\t},\n\t\t\twidth: {\n\t\t\t},\n\t\t\theight: {\n\t\t\t},\n\t\t\tboth: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"none\",\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-text-fit-padding\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0,\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"icon-text-fit\": [\n\t\t\t\t\t\"both\",\n\t\t\t\t\t\"width\",\n\t\t\t\t\t\"height\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-image\": {\n\t\ttype: \"resolvedImage\",\n\t\ttokens: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-rotate\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\tunits: \"degrees\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-padding\": {\n\t\ttype: \"padding\",\n\t\t\"default\": [\n\t\t\t2\n\t\t],\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-keep-upright\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t{\n\t\t\t\t\"icon-rotation-alignment\": \"map\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"line\",\n\t\t\t\t\t\"line-center\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-offset\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t},\n\t\t\ttop: {\n\t\t\t},\n\t\t\tbottom: {\n\t\t\t},\n\t\t\t\"top-left\": {\n\t\t\t},\n\t\t\t\"top-right\": {\n\t\t\t},\n\t\t\t\"bottom-left\": {\n\t\t\t},\n\t\t\t\"bottom-right\": {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"center\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-pitch-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-pitch-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-rotation-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\t\"viewport-glyph\": {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-field\": {\n\t\ttype: \"formatted\",\n\t\t\"default\": \"\",\n\t\ttokens: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-font\": {\n\t\ttype: \"array\",\n\t\tvalue: \"string\",\n\t\t\"default\": [\n\t\t\t\"Open Sans Regular\",\n\t\t\t\"Arial Unicode MS Regular\"\n\t\t],\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-size\": {\n\t\ttype: \"number\",\n\t\t\"default\": 16,\n\t\tminimum: 0,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-max-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 10,\n\t\tminimum: 0,\n\t\tunits: \"ems\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-line-height\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1.2,\n\t\tunits: \"ems\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-letter-spacing\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tunits: \"ems\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-justify\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tauto: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"center\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-radial-offset\": {\n\t\ttype: \"number\",\n\t\tunits: \"ems\",\n\t\t\"default\": 0,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\t\"property-type\": \"data-driven\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t}\n\t},\n\t\"text-variable-anchor\": {\n\t\ttype: \"array\",\n\t\tvalue: \"enum\",\n\t\tvalues: {\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t},\n\t\t\ttop: {\n\t\t\t},\n\t\t\tbottom: {\n\t\t\t},\n\t\t\t\"top-left\": {\n\t\t\t},\n\t\t\t\"top-right\": {\n\t\t\t},\n\t\t\t\"bottom-left\": {\n\t\t\t},\n\t\t\t\"bottom-right\": {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"point\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-variable-anchor-offset\": {\n\t\ttype: \"variableAnchorOffsetCollection\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"point\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t},\n\t\t\ttop: {\n\t\t\t},\n\t\t\tbottom: {\n\t\t\t},\n\t\t\t\"top-left\": {\n\t\t\t},\n\t\t\t\"top-right\": {\n\t\t\t},\n\t\t\t\"bottom-left\": {\n\t\t\t},\n\t\t\t\"bottom-right\": {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"center\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"!\": \"text-variable-anchor\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-max-angle\": {\n\t\ttype: \"number\",\n\t\t\"default\": 45,\n\t\tunits: \"degrees\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"line\",\n\t\t\t\t\t\"line-center\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-writing-mode\": {\n\t\ttype: \"array\",\n\t\tvalue: \"enum\",\n\t\tvalues: {\n\t\t\thorizontal: {\n\t\t\t},\n\t\t\tvertical: {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"point\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-rotate\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\tunits: \"degrees\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-padding\": {\n\t\ttype: \"number\",\n\t\t\"default\": 2,\n\t\tminimum: 0,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-keep-upright\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": true,\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"text-rotation-alignment\": \"map\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"line\",\n\t\t\t\t\t\"line-center\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-transform\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnone: {\n\t\t\t},\n\t\t\tuppercase: {\n\t\t\t},\n\t\t\tlowercase: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"none\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-offset\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tunits: \"ems\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"!\": \"text-radial-offset\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-allow-overlap\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"!\": \"text-overlap\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-overlap\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnever: {\n\t\t\t},\n\t\t\talways: {\n\t\t\t},\n\t\t\tcooperative: {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-ignore-placement\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-optional\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_raster = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_hillshade = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar filter = {\n\ttype: \"array\",\n\tvalue: \"*\"\n};\nvar filter_operator = {\n\ttype: \"enum\",\n\tvalues: {\n\t\t\"==\": {\n\t\t},\n\t\t\"!=\": {\n\t\t},\n\t\t\">\": {\n\t\t},\n\t\t\">=\": {\n\t\t},\n\t\t\"<\": {\n\t\t},\n\t\t\"<=\": {\n\t\t},\n\t\t\"in\": {\n\t\t},\n\t\t\"!in\": {\n\t\t},\n\t\tall: {\n\t\t},\n\t\tany: {\n\t\t},\n\t\tnone: {\n\t\t},\n\t\thas: {\n\t\t},\n\t\t\"!has\": {\n\t\t},\n\t\twithin: {\n\t\t}\n\t}\n};\nvar geometry_type = {\n\ttype: \"enum\",\n\tvalues: {\n\t\tPoint: {\n\t\t},\n\t\tLineString: {\n\t\t},\n\t\tPolygon: {\n\t\t}\n\t}\n};\nvar function_stop = {\n\ttype: \"array\",\n\tminimum: 0,\n\tmaximum: 24,\n\tvalue: [\n\t\t\"number\",\n\t\t\"color\"\n\t],\n\tlength: 2\n};\nvar expression$1 = {\n\ttype: \"array\",\n\tvalue: \"*\",\n\tminimum: 1\n};\nvar light = {\n\tanchor: {\n\t\ttype: \"enum\",\n\t\t\"default\": \"viewport\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"property-type\": \"data-constant\",\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t}\n\t},\n\tposition: {\n\t\ttype: \"array\",\n\t\t\"default\": [\n\t\t\t1.15,\n\t\t\t210,\n\t\t\t30\n\t\t],\n\t\tlength: 3,\n\t\tvalue: \"number\",\n\t\t\"property-type\": \"data-constant\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t}\n\t},\n\tcolor: {\n\t\ttype: \"color\",\n\t\t\"property-type\": \"data-constant\",\n\t\t\"default\": \"#ffffff\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\ttransition: true\n\t},\n\tintensity: {\n\t\ttype: \"number\",\n\t\t\"property-type\": \"data-constant\",\n\t\t\"default\": 0.5,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\ttransition: true\n\t}\n};\nvar terrain = {\n\tsource: {\n\t\ttype: \"string\",\n\t\trequired: true\n\t},\n\texaggeration: {\n\t\ttype: \"number\",\n\t\tminimum: 0,\n\t\t\"default\": 1\n\t}\n};\nvar paint = [\n\t\"paint_fill\",\n\t\"paint_line\",\n\t\"paint_circle\",\n\t\"paint_heatmap\",\n\t\"paint_fill-extrusion\",\n\t\"paint_symbol\",\n\t\"paint_raster\",\n\t\"paint_hillshade\",\n\t\"paint_background\"\n];\nvar paint_fill = {\n\t\"fill-antialias\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"fill-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-outline-color\": {\n\t\ttype: \"color\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"fill-pattern\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"fill-antialias\": true\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"fill-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded-data-driven\"\n\t}\n};\nvar paint_line = {\n\t\"line-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"line-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"line-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-gap-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-offset\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-dasharray\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"line widths\",\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"line-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded\"\n\t},\n\t\"line-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded-data-driven\"\n\t},\n\t\"line-gradient\": {\n\t\ttype: \"color\",\n\t\ttransition: false,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"line-dasharray\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"!\": \"line-pattern\"\n\t\t\t},\n\t\t\t{\n\t\t\t\tsource: \"geojson\",\n\t\t\t\thas: {\n\t\t\t\t\tlineMetrics: true\n\t\t\t\t}\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"line-progress\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"color-ramp\"\n\t}\n};\nvar paint_circle = {\n\t\"circle-radius\": {\n\t\ttype: \"number\",\n\t\t\"default\": 5,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"circle-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-pitch-scale\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-pitch-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"viewport\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-stroke-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-stroke-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-stroke-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t}\n};\nvar paint_heatmap = {\n\t\"heatmap-radius\": {\n\t\ttype: \"number\",\n\t\t\"default\": 30,\n\t\tminimum: 1,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"heatmap-weight\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"heatmap-intensity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"heatmap-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": [\n\t\t\t\"interpolate\",\n\t\t\t[\n\t\t\t\t\"linear\"\n\t\t\t],\n\t\t\t[\n\t\t\t\t\"heatmap-density\"\n\t\t\t],\n\t\t\t0,\n\t\t\t\"rgba(0, 0, 255, 0)\",\n\t\t\t0.1,\n\t\t\t\"royalblue\",\n\t\t\t0.3,\n\t\t\t\"cyan\",\n\t\t\t0.5,\n\t\t\t\"lime\",\n\t\t\t0.7,\n\t\t\t\"yellow\",\n\t\t\t1,\n\t\t\t\"red\"\n\t\t],\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"heatmap-density\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"color-ramp\"\n\t},\n\t\"heatmap-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_symbol = {\n\t\"icon-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-halo-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"rgba(0, 0, 0, 0)\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-halo-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-halo-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"icon-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\toverridable: true,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-halo-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"rgba(0, 0, 0, 0)\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-halo-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-halo-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t\"text-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_raster = {\n\t\"raster-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-hue-rotate\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\ttransition: true,\n\t\tunits: \"degrees\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-brightness-min\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-brightness-max\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-saturation\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: -1,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-contrast\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: -1,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-resampling\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tlinear: {\n\t\t\t},\n\t\t\tnearest: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"linear\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-fade-duration\": {\n\t\ttype: \"number\",\n\t\t\"default\": 300,\n\t\tminimum: 0,\n\t\ttransition: false,\n\t\tunits: \"milliseconds\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_hillshade = {\n\t\"hillshade-illumination-direction\": {\n\t\ttype: \"number\",\n\t\t\"default\": 335,\n\t\tminimum: 0,\n\t\tmaximum: 359,\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-illumination-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"viewport\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-exaggeration\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0.5,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-shadow-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-highlight-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#FFFFFF\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-accent-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_background = {\n\t\"background-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"background-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"background-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded\"\n\t},\n\t\"background-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar transition = {\n\tduration: {\n\t\ttype: \"number\",\n\t\t\"default\": 300,\n\t\tminimum: 0,\n\t\tunits: \"milliseconds\"\n\t},\n\tdelay: {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tunits: \"milliseconds\"\n\t}\n};\nvar promoteId = {\n\t\"*\": {\n\t\ttype: \"string\"\n\t}\n};\nvar v8Spec = {\n\t$version: $version,\n\t$root: $root,\n\tsources: sources,\n\tsource: source,\n\tsource_vector: source_vector,\n\tsource_raster: source_raster,\n\tsource_raster_dem: source_raster_dem,\n\tsource_geojson: source_geojson,\n\tsource_video: source_video,\n\tsource_image: source_image,\n\tlayer: layer,\n\tlayout: layout,\n\tlayout_background: layout_background,\n\tlayout_fill: layout_fill,\n\tlayout_circle: layout_circle,\n\tlayout_heatmap: layout_heatmap,\n\t\"layout_fill-extrusion\": {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n},\n\tlayout_line: layout_line,\n\tlayout_symbol: layout_symbol,\n\tlayout_raster: layout_raster,\n\tlayout_hillshade: layout_hillshade,\n\tfilter: filter,\n\tfilter_operator: filter_operator,\n\tgeometry_type: geometry_type,\n\t\"function\": {\n\texpression: {\n\t\ttype: \"expression\"\n\t},\n\tstops: {\n\t\ttype: \"array\",\n\t\tvalue: \"function_stop\"\n\t},\n\tbase: {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0\n\t},\n\tproperty: {\n\t\ttype: \"string\",\n\t\t\"default\": \"$zoom\"\n\t},\n\ttype: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tidentity: {\n\t\t\t},\n\t\t\texponential: {\n\t\t\t},\n\t\t\tinterval: {\n\t\t\t},\n\t\t\tcategorical: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"exponential\"\n\t},\n\tcolorSpace: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\trgb: {\n\t\t\t},\n\t\t\tlab: {\n\t\t\t},\n\t\t\thcl: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"rgb\"\n\t},\n\t\"default\": {\n\t\ttype: \"*\",\n\t\trequired: false\n\t}\n},\n\tfunction_stop: function_stop,\n\texpression: expression$1,\n\tlight: light,\n\tterrain: terrain,\n\tpaint: paint,\n\tpaint_fill: paint_fill,\n\t\"paint_fill-extrusion\": {\n\t\"fill-extrusion-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-extrusion-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"fill-extrusion-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-extrusion-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-extrusion-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"fill-extrusion-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-extrusion-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded-data-driven\"\n\t},\n\t\"fill-extrusion-height\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tunits: \"meters\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-extrusion-base\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tunits: \"meters\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"fill-extrusion-height\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-extrusion-vertical-gradient\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": true,\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n},\n\tpaint_line: paint_line,\n\tpaint_circle: paint_circle,\n\tpaint_heatmap: paint_heatmap,\n\tpaint_symbol: paint_symbol,\n\tpaint_raster: paint_raster,\n\tpaint_hillshade: paint_hillshade,\n\tpaint_background: paint_background,\n\ttransition: transition,\n\t\"property-type\": {\n\t\"data-driven\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"cross-faded\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"cross-faded-data-driven\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"color-ramp\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"data-constant\": {\n\t\ttype: \"property-type\"\n\t},\n\tconstant: {\n\t\ttype: \"property-type\"\n\t}\n},\n\tpromoteId: promoteId\n};\n\nconst refProperties = ['type', 'source', 'source-layer', 'minzoom', 'maxzoom', 'filter', 'layout'];\n\nfunction deref(layer, parent) {\n const result = {};\n for (const k in layer) {\n if (k !== 'ref') {\n result[k] = layer[k];\n }\n }\n refProperties.forEach((k) => {\n if (k in parent) {\n result[k] = parent[k];\n }\n });\n return result;\n}\n/**\n * Given an array of layers, some of which may contain `ref` properties\n * whose value is the `id` of another property, return a new array where\n * such layers have been augmented with the 'type', 'source', etc. properties\n * from the parent layer, and the `ref` property has been removed.\n *\n * The input is not modified. The output may contain references to portions\n * of the input.\n *\n * @private\n * @param {Array} layers\n * @returns {Array}\n */\nfunction derefLayers(layers) {\n layers = layers.slice();\n const map = Object.create(null);\n for (let i = 0; i < layers.length; i++) {\n map[layers[i].id] = layers[i];\n }\n for (let i = 0; i < layers.length; i++) {\n if ('ref' in layers[i]) {\n layers[i] = deref(layers[i], map[layers[i].ref]);\n }\n }\n return layers;\n}\n\n/**\n * Deeply compares two object literals.\n *\n * @private\n */\nfunction deepEqual(a, b) {\n if (Array.isArray(a)) {\n if (!Array.isArray(b) || a.length !== b.length)\n return false;\n for (let i = 0; i < a.length; i++) {\n if (!deepEqual(a[i], b[i]))\n return false;\n }\n return true;\n }\n if (typeof a === 'object' && a !== null && b !== null) {\n if (!(typeof b === 'object'))\n return false;\n const keys = Object.keys(a);\n if (keys.length !== Object.keys(b).length)\n return false;\n for (const key in a) {\n if (!deepEqual(a[key], b[key]))\n return false;\n }\n return true;\n }\n return a === b;\n}\n\nconst operations = {\n /*\n * { command: 'setStyle', args: [stylesheet] }\n */\n setStyle: 'setStyle',\n /*\n * { command: 'addLayer', args: [layer, 'beforeLayerId'] }\n */\n addLayer: 'addLayer',\n /*\n * { command: 'removeLayer', args: ['layerId'] }\n */\n removeLayer: 'removeLayer',\n /*\n * { command: 'setPaintProperty', args: ['layerId', 'prop', value] }\n */\n setPaintProperty: 'setPaintProperty',\n /*\n * { command: 'setLayoutProperty', args: ['layerId', 'prop', value] }\n */\n setLayoutProperty: 'setLayoutProperty',\n /*\n * { command: 'setFilter', args: ['layerId', filter] }\n */\n setFilter: 'setFilter',\n /*\n * { command: 'addSource', args: ['sourceId', source] }\n */\n addSource: 'addSource',\n /*\n * { command: 'removeSource', args: ['sourceId'] }\n */\n removeSource: 'removeSource',\n /*\n * { command: 'setGeoJSONSourceData', args: ['sourceId', data] }\n */\n setGeoJSONSourceData: 'setGeoJSONSourceData',\n /*\n * { command: 'setLayerZoomRange', args: ['layerId', 0, 22] }\n */\n setLayerZoomRange: 'setLayerZoomRange',\n /*\n * { command: 'setLayerProperty', args: ['layerId', 'prop', value] }\n */\n setLayerProperty: 'setLayerProperty',\n /*\n * { command: 'setCenter', args: [[lon, lat]] }\n */\n setCenter: 'setCenter',\n /*\n * { command: 'setZoom', args: [zoom] }\n */\n setZoom: 'setZoom',\n /*\n * { command: 'setBearing', args: [bearing] }\n */\n setBearing: 'setBearing',\n /*\n * { command: 'setPitch', args: [pitch] }\n */\n setPitch: 'setPitch',\n /*\n * { command: 'setSprite', args: ['spriteUrl'] }\n */\n setSprite: 'setSprite',\n /*\n * { command: 'setGlyphs', args: ['glyphsUrl'] }\n */\n setGlyphs: 'setGlyphs',\n /*\n * { command: 'setTransition', args: [transition] }\n */\n setTransition: 'setTransition',\n /*\n * { command: 'setLighting', args: [lightProperties] }\n */\n setLight: 'setLight'\n};\nfunction addSource(sourceId, after, commands) {\n commands.push({ command: operations.addSource, args: [sourceId, after[sourceId]] });\n}\nfunction removeSource(sourceId, commands, sourcesRemoved) {\n commands.push({ command: operations.removeSource, args: [sourceId] });\n sourcesRemoved[sourceId] = true;\n}\nfunction updateSource(sourceId, after, commands, sourcesRemoved) {\n removeSource(sourceId, commands, sourcesRemoved);\n addSource(sourceId, after, commands);\n}\nfunction canUpdateGeoJSON(before, after, sourceId) {\n let prop;\n for (prop in before[sourceId]) {\n if (!Object.prototype.hasOwnProperty.call(before[sourceId], prop))\n continue;\n if (prop !== 'data' && !deepEqual(before[sourceId][prop], after[sourceId][prop])) {\n return false;\n }\n }\n for (prop in after[sourceId]) {\n if (!Object.prototype.hasOwnProperty.call(after[sourceId], prop))\n continue;\n if (prop !== 'data' && !deepEqual(before[sourceId][prop], after[sourceId][prop])) {\n return false;\n }\n }\n return true;\n}\nfunction diffSources(before, after, commands, sourcesRemoved) {\n before = before || {};\n after = after || {};\n let sourceId;\n // look for sources to remove\n for (sourceId in before) {\n if (!Object.prototype.hasOwnProperty.call(before, sourceId))\n continue;\n if (!Object.prototype.hasOwnProperty.call(after, sourceId)) {\n removeSource(sourceId, commands, sourcesRemoved);\n }\n }\n // look for sources to add/update\n for (sourceId in after) {\n if (!Object.prototype.hasOwnProperty.call(after, sourceId))\n continue;\n if (!Object.prototype.hasOwnProperty.call(before, sourceId)) {\n addSource(sourceId, after, commands);\n }\n else if (!deepEqual(before[sourceId], after[sourceId])) {\n if (before[sourceId].type === 'geojson' && after[sourceId].type === 'geojson' && canUpdateGeoJSON(before, after, sourceId)) {\n commands.push({ command: operations.setGeoJSONSourceData, args: [sourceId, after[sourceId].data] });\n }\n else {\n // no update command, must remove then add\n updateSource(sourceId, after, commands, sourcesRemoved);\n }\n }\n }\n}\nfunction diffLayerPropertyChanges(before, after, commands, layerId, klass, command) {\n before = before || {};\n after = after || {};\n let prop;\n for (prop in before) {\n if (!Object.prototype.hasOwnProperty.call(before, prop))\n continue;\n if (!deepEqual(before[prop], after[prop])) {\n commands.push({ command, args: [layerId, prop, after[prop], klass] });\n }\n }\n for (prop in after) {\n if (!Object.prototype.hasOwnProperty.call(after, prop) || Object.prototype.hasOwnProperty.call(before, prop))\n continue;\n if (!deepEqual(before[prop], after[prop])) {\n commands.push({ command, args: [layerId, prop, after[prop], klass] });\n }\n }\n}\nfunction pluckId(layer) {\n return layer.id;\n}\nfunction indexById(group, layer) {\n group[layer.id] = layer;\n return group;\n}\nfunction diffLayers(before, after, commands) {\n before = before || [];\n after = after || [];\n // order of layers by id\n const beforeOrder = before.map(pluckId);\n const afterOrder = after.map(pluckId);\n // index of layer by id\n const beforeIndex = before.reduce(indexById, {});\n const afterIndex = after.reduce(indexById, {});\n // track order of layers as if they have been mutated\n const tracker = beforeOrder.slice();\n // layers that have been added do not need to be diffed\n const clean = Object.create(null);\n let i, d, layerId, beforeLayer, afterLayer, insertBeforeLayerId, prop;\n // remove layers\n for (i = 0, d = 0; i < beforeOrder.length; i++) {\n layerId = beforeOrder[i];\n if (!Object.prototype.hasOwnProperty.call(afterIndex, layerId)) {\n commands.push({ command: operations.removeLayer, args: [layerId] });\n tracker.splice(tracker.indexOf(layerId, d), 1);\n }\n else {\n // limit where in tracker we need to look for a match\n d++;\n }\n }\n // add/reorder layers\n for (i = 0, d = 0; i < afterOrder.length; i++) {\n // work backwards as insert is before an existing layer\n layerId = afterOrder[afterOrder.length - 1 - i];\n if (tracker[tracker.length - 1 - i] === layerId)\n continue;\n if (Object.prototype.hasOwnProperty.call(beforeIndex, layerId)) {\n // remove the layer before we insert at the correct position\n commands.push({ command: operations.removeLayer, args: [layerId] });\n tracker.splice(tracker.lastIndexOf(layerId, tracker.length - d), 1);\n }\n else {\n // limit where in tracker we need to look for a match\n d++;\n }\n // add layer at correct position\n insertBeforeLayerId = tracker[tracker.length - i];\n commands.push({ command: operations.addLayer, args: [afterIndex[layerId], insertBeforeLayerId] });\n tracker.splice(tracker.length - i, 0, layerId);\n clean[layerId] = true;\n }\n // update layers\n for (i = 0; i < afterOrder.length; i++) {\n layerId = afterOrder[i];\n beforeLayer = beforeIndex[layerId];\n afterLayer = afterIndex[layerId];\n // no need to update if previously added (new or moved)\n if (clean[layerId] || deepEqual(beforeLayer, afterLayer))\n continue;\n // If source, source-layer, or type have changes, then remove the layer\n // and add it back 'from scratch'.\n if (!deepEqual(beforeLayer.source, afterLayer.source) || !deepEqual(beforeLayer['source-layer'], afterLayer['source-layer']) || !deepEqual(beforeLayer.type, afterLayer.type)) {\n commands.push({ command: operations.removeLayer, args: [layerId] });\n // we add the layer back at the same position it was already in, so\n // there's no need to update the `tracker`\n insertBeforeLayerId = tracker[tracker.lastIndexOf(layerId) + 1];\n commands.push({ command: operations.addLayer, args: [afterLayer, insertBeforeLayerId] });\n continue;\n }\n // layout, paint, filter, minzoom, maxzoom\n diffLayerPropertyChanges(beforeLayer.layout, afterLayer.layout, commands, layerId, null, operations.setLayoutProperty);\n diffLayerPropertyChanges(beforeLayer.paint, afterLayer.paint, commands, layerId, null, operations.setPaintProperty);\n if (!deepEqual(beforeLayer.filter, afterLayer.filter)) {\n commands.push({ command: operations.setFilter, args: [layerId, afterLayer.filter] });\n }\n if (!deepEqual(beforeLayer.minzoom, afterLayer.minzoom) || !deepEqual(beforeLayer.maxzoom, afterLayer.maxzoom)) {\n commands.push({ command: operations.setLayerZoomRange, args: [layerId, afterLayer.minzoom, afterLayer.maxzoom] });\n }\n // handle all other layer props, including paint.*\n for (prop in beforeLayer) {\n if (!Object.prototype.hasOwnProperty.call(beforeLayer, prop))\n continue;\n if (prop === 'layout' || prop === 'paint' || prop === 'filter' ||\n prop === 'metadata' || prop === 'minzoom' || prop === 'maxzoom')\n continue;\n if (prop.indexOf('paint.') === 0) {\n diffLayerPropertyChanges(beforeLayer[prop], afterLayer[prop], commands, layerId, prop.slice(6), operations.setPaintProperty);\n }\n else if (!deepEqual(beforeLayer[prop], afterLayer[prop])) {\n commands.push({ command: operations.setLayerProperty, args: [layerId, prop, afterLayer[prop]] });\n }\n }\n for (prop in afterLayer) {\n if (!Object.prototype.hasOwnProperty.call(afterLayer, prop) || Object.prototype.hasOwnProperty.call(beforeLayer, prop))\n continue;\n if (prop === 'layout' || prop === 'paint' || prop === 'filter' ||\n prop === 'metadata' || prop === 'minzoom' || prop === 'maxzoom')\n continue;\n if (prop.indexOf('paint.') === 0) {\n diffLayerPropertyChanges(beforeLayer[prop], afterLayer[prop], commands, layerId, prop.slice(6), operations.setPaintProperty);\n }\n else if (!deepEqual(beforeLayer[prop], afterLayer[prop])) {\n commands.push({ command: operations.setLayerProperty, args: [layerId, prop, afterLayer[prop]] });\n }\n }\n }\n}\n/**\n * Diff two stylesheet\n *\n * Creates semanticly aware diffs that can easily be applied at runtime.\n * Operations produced by the diff closely resemble the maplibre-gl-js API. Any\n * error creating the diff will fall back to the 'setStyle' operation.\n *\n * Example diff:\n * [\n * { command: 'setConstant', args: ['@water', '#0000FF'] },\n * { command: 'setPaintProperty', args: ['background', 'background-color', 'black'] }\n * ]\n *\n * @private\n * @param {*} [before] stylesheet to compare from\n * @param {*} after stylesheet to compare to\n * @returns Array list of changes\n */\nfunction diffStyles(before, after) {\n if (!before)\n return [{ command: operations.setStyle, args: [after] }];\n let commands = [];\n try {\n // Handle changes to top-level properties\n if (!deepEqual(before.version, after.version)) {\n return [{ command: operations.setStyle, args: [after] }];\n }\n if (!deepEqual(before.center, after.center)) {\n commands.push({ command: operations.setCenter, args: [after.center] });\n }\n if (!deepEqual(before.zoom, after.zoom)) {\n commands.push({ command: operations.setZoom, args: [after.zoom] });\n }\n if (!deepEqual(before.bearing, after.bearing)) {\n commands.push({ command: operations.setBearing, args: [after.bearing] });\n }\n if (!deepEqual(before.pitch, after.pitch)) {\n commands.push({ command: operations.setPitch, args: [after.pitch] });\n }\n if (!deepEqual(before.sprite, after.sprite)) {\n commands.push({ command: operations.setSprite, args: [after.sprite] });\n }\n if (!deepEqual(before.glyphs, after.glyphs)) {\n commands.push({ command: operations.setGlyphs, args: [after.glyphs] });\n }\n if (!deepEqual(before.transition, after.transition)) {\n commands.push({ command: operations.setTransition, args: [after.transition] });\n }\n if (!deepEqual(before.light, after.light)) {\n commands.push({ command: operations.setLight, args: [after.light] });\n }\n // Handle changes to `sources`\n // If a source is to be removed, we also--before the removeSource\n // command--need to remove all the style layers that depend on it.\n const sourcesRemoved = {};\n // First collect the {add,remove}Source commands\n const removeOrAddSourceCommands = [];\n diffSources(before.sources, after.sources, removeOrAddSourceCommands, sourcesRemoved);\n // Push a removeLayer command for each style layer that depends on a\n // source that's being removed.\n // Also, exclude any such layers them from the input to `diffLayers`\n // below, so that diffLayers produces the appropriate `addLayers`\n // command\n const beforeLayers = [];\n if (before.layers) {\n before.layers.forEach((layer) => {\n if (sourcesRemoved[layer.source]) {\n commands.push({ command: operations.removeLayer, args: [layer.id] });\n }\n else {\n beforeLayers.push(layer);\n }\n });\n }\n commands = commands.concat(removeOrAddSourceCommands);\n // Handle changes to `layers`\n diffLayers(beforeLayers, after.layers, commands);\n }\n catch (e) {\n // fall back to setStyle\n console.warn('Unable to compute style diff:', e);\n commands = [{ command: operations.setStyle, args: [after] }];\n }\n return commands;\n}\n\n// Note: Do not inherit from Error. It breaks when transpiling to ES5.\nclass ValidationError {\n constructor(key, value, message, identifier) {\n this.message = (key ? `${key}: ` : '') + message;\n if (identifier)\n this.identifier = identifier;\n if (value !== null && value !== undefined && value.__line__) {\n this.line = value.__line__;\n }\n }\n}\n\n// Note: Do not inherit from Error. It breaks when transpiling to ES5.\nclass ParsingError {\n constructor(error) {\n this.error = error;\n this.message = error.message;\n const match = error.message.match(/line (\\d+)/);\n this.line = match ? parseInt(match[1], 10) : 0;\n }\n}\n\nfunction extendBy(output, ...inputs) {\n for (const input of inputs) {\n for (const k in input) {\n output[k] = input[k];\n }\n }\n return output;\n}\n\nclass ExpressionParsingError extends Error {\n constructor(key, message) {\n super(message);\n this.message = message;\n this.key = key;\n }\n}\n\n/**\n * Tracks `let` bindings during expression parsing.\n * @private\n */\nclass Scope {\n constructor(parent, bindings = []) {\n this.parent = parent;\n this.bindings = {};\n for (const [name, expression] of bindings) {\n this.bindings[name] = expression;\n }\n }\n concat(bindings) {\n return new Scope(this, bindings);\n }\n get(name) {\n if (this.bindings[name]) {\n return this.bindings[name];\n }\n if (this.parent) {\n return this.parent.get(name);\n }\n throw new Error(`${name} not found in scope.`);\n }\n has(name) {\n if (this.bindings[name])\n return true;\n return this.parent ? this.parent.has(name) : false;\n }\n}\n\nconst NullType = { kind: 'null' };\nconst NumberType = { kind: 'number' };\nconst StringType = { kind: 'string' };\nconst BooleanType = { kind: 'boolean' };\nconst ColorType = { kind: 'color' };\nconst ObjectType = { kind: 'object' };\nconst ValueType = { kind: 'value' };\nconst ErrorType = { kind: 'error' };\nconst CollatorType = { kind: 'collator' };\nconst FormattedType = { kind: 'formatted' };\nconst PaddingType = { kind: 'padding' };\nconst ResolvedImageType = { kind: 'resolvedImage' };\nconst VariableAnchorOffsetCollectionType = { kind: 'variableAnchorOffsetCollection' };\nfunction array$1(itemType, N) {\n return {\n kind: 'array',\n itemType,\n N\n };\n}\nfunction toString$1(type) {\n if (type.kind === 'array') {\n const itemType = toString$1(type.itemType);\n return typeof type.N === 'number' ?\n `array<${itemType}, ${type.N}>` :\n type.itemType.kind === 'value' ? 'array' : `array<${itemType}>`;\n }\n else {\n return type.kind;\n }\n}\nconst valueMemberTypes = [\n NullType,\n NumberType,\n StringType,\n BooleanType,\n ColorType,\n FormattedType,\n ObjectType,\n array$1(ValueType),\n PaddingType,\n ResolvedImageType,\n VariableAnchorOffsetCollectionType\n];\n/**\n * Returns null if `t` is a subtype of `expected`; otherwise returns an\n * error message.\n * @private\n */\nfunction checkSubtype(expected, t) {\n if (t.kind === 'error') {\n // Error is a subtype of every type\n return null;\n }\n else if (expected.kind === 'array') {\n if (t.kind === 'array' &&\n ((t.N === 0 && t.itemType.kind === 'value') || !checkSubtype(expected.itemType, t.itemType)) &&\n (typeof expected.N !== 'number' || expected.N === t.N)) {\n return null;\n }\n }\n else if (expected.kind === t.kind) {\n return null;\n }\n else if (expected.kind === 'value') {\n for (const memberType of valueMemberTypes) {\n if (!checkSubtype(memberType, t)) {\n return null;\n }\n }\n }\n return `Expected ${toString$1(expected)} but found ${toString$1(t)} instead.`;\n}\nfunction isValidType(provided, allowedTypes) {\n return allowedTypes.some(t => t.kind === provided.kind);\n}\nfunction isValidNativeType(provided, allowedTypes) {\n return allowedTypes.some(t => {\n if (t === 'null') {\n return provided === null;\n }\n else if (t === 'array') {\n return Array.isArray(provided);\n }\n else if (t === 'object') {\n return provided && !Array.isArray(provided) && typeof provided === 'object';\n }\n else {\n return t === typeof provided;\n }\n });\n}\n/**\n * Verify whether the specified type is of the same type as the specified sample.\n *\n * @param provided Type to verify\n * @param sample Sample type to reference\n * @returns `true` if both objects are of the same type, `false` otherwise\n * @example basic types\n * if (verifyType(outputType, ValueType)) {\n * // type narrowed to:\n * outputType.kind; // 'value'\n * }\n * @example array types\n * if (verifyType(outputType, array(NumberType))) {\n * // type narrowed to:\n * outputType.kind; // 'array'\n * outputType.itemType; // NumberTypeT\n * outputType.itemType.kind; // 'number'\n * }\n */\nfunction verifyType(provided, sample) {\n if (provided.kind === 'array' && sample.kind === 'array') {\n return provided.itemType.kind === sample.itemType.kind && typeof provided.N === 'number';\n }\n return provided.kind === sample.kind;\n}\n\n// See https://observablehq.com/@mbostock/lab-and-rgb\nconst Xn = 0.96422, Yn = 1, Zn = 0.82521, t0 = 4 / 29, t1 = 6 / 29, t2 = 3 * t1 * t1, t3 = t1 * t1 * t1, deg2rad = Math.PI / 180, rad2deg = 180 / Math.PI;\nfunction constrainAngle(angle) {\n angle = angle % 360;\n if (angle < 0) {\n angle += 360;\n }\n return angle;\n}\nfunction rgbToLab([r, g, b, alpha]) {\n r = rgb2xyz(r);\n g = rgb2xyz(g);\n b = rgb2xyz(b);\n let x, z;\n const y = xyz2lab((0.2225045 * r + 0.7168786 * g + 0.0606169 * b) / Yn);\n if (r === g && g === b) {\n x = z = y;\n }\n else {\n x = xyz2lab((0.4360747 * r + 0.3850649 * g + 0.1430804 * b) / Xn);\n z = xyz2lab((0.0139322 * r + 0.0971045 * g + 0.7141733 * b) / Zn);\n }\n const l = 116 * y - 16;\n return [(l < 0) ? 0 : l, 500 * (x - y), 200 * (y - z), alpha];\n}\nfunction rgb2xyz(x) {\n return (x <= 0.04045) ? x / 12.92 : Math.pow((x + 0.055) / 1.055, 2.4);\n}\nfunction xyz2lab(t) {\n return (t > t3) ? Math.pow(t, 1 / 3) : t / t2 + t0;\n}\nfunction labToRgb([l, a, b, alpha]) {\n let y = (l + 16) / 116, x = isNaN(a) ? y : y + a / 500, z = isNaN(b) ? y : y - b / 200;\n y = Yn * lab2xyz(y);\n x = Xn * lab2xyz(x);\n z = Zn * lab2xyz(z);\n return [\n xyz2rgb(3.1338561 * x - 1.6168667 * y - 0.4906146 * z),\n xyz2rgb(-0.9787684 * x + 1.9161415 * y + 0.0334540 * z),\n xyz2rgb(0.0719453 * x - 0.2289914 * y + 1.4052427 * z),\n alpha,\n ];\n}\nfunction xyz2rgb(x) {\n x = (x <= 0.00304) ? 12.92 * x : 1.055 * Math.pow(x, 1 / 2.4) - 0.055;\n return (x < 0) ? 0 : (x > 1) ? 1 : x; // clip to 0..1 range\n}\nfunction lab2xyz(t) {\n return (t > t1) ? t * t * t : t2 * (t - t0);\n}\nfunction rgbToHcl(rgbColor) {\n const [l, a, b, alpha] = rgbToLab(rgbColor);\n const c = Math.sqrt(a * a + b * b);\n const h = Math.round(c * 10000) ? constrainAngle(Math.atan2(b, a) * rad2deg) : NaN;\n return [h, c, l, alpha];\n}\nfunction hclToRgb([h, c, l, alpha]) {\n h = isNaN(h) ? 0 : h * deg2rad;\n return labToRgb([l, Math.cos(h) * c, Math.sin(h) * c, alpha]);\n}\n// https://drafts.csswg.org/css-color-4/#hsl-to-rgb\nfunction hslToRgb([h, s, l, alpha]) {\n h = constrainAngle(h);\n s /= 100;\n l /= 100;\n function f(n) {\n const k = (n + h / 30) % 12;\n const a = s * Math.min(l, 1 - l);\n return l - a * Math.max(-1, Math.min(k - 3, 9 - k, 1));\n }\n return [f(0), f(8), f(4), alpha];\n}\n\n/**\n * CSS color parser compliant with CSS Color 4 Specification.\n * Supports: named colors, `transparent` keyword, all rgb hex notations,\n * rgb(), rgba(), hsl() and hsla() functions.\n * Does not round the parsed values to integers from the range 0..255.\n *\n * Syntax:\n *\n * = | \n * = | \n *\n * rgb() = rgb( {3} [ / ]? ) | rgb( {3} [ / ]? )\n * rgb() = rgb( #{3} , ? ) | rgb( #{3} , ? )\n *\n * hsl() = hsl( [ / ]? )\n * hsl() = hsl( , , , ? )\n *\n * Caveats:\n * - - with optional `deg` suffix; `grad`, `rad`, `turn` are not supported\n * - `none` keyword is not supported\n * - comments inside rgb()/hsl() are not supported\n * - legacy color syntax rgba() is supported with an identical grammar and behavior to rgb()\n * - legacy color syntax hsla() is supported with an identical grammar and behavior to hsl()\n *\n * @param input CSS color string to parse.\n * @returns Color in sRGB color space, with `red`, `green`, `blue`\n * and `alpha` channels normalized to the range 0..1,\n * or `undefined` if the input is not a valid color string.\n */\nfunction parseCssColor(input) {\n input = input.toLowerCase().trim();\n if (input === 'transparent') {\n return [0, 0, 0, 0];\n }\n // 'white', 'black', 'blue'\n const namedColorsMatch = namedColors[input];\n if (namedColorsMatch) {\n const [r, g, b] = namedColorsMatch;\n return [r / 255, g / 255, b / 255, 1];\n }\n // #f0c, #f0cf, #ff00cc, #ff00ccff\n if (input.startsWith('#')) {\n const hexRegexp = /^#(?:[0-9a-f]{3,4}|[0-9a-f]{6}|[0-9a-f]{8})$/;\n if (hexRegexp.test(input)) {\n const step = input.length < 6 ? 1 : 2;\n let i = 1;\n return [\n parseHex(input.slice(i, i += step)),\n parseHex(input.slice(i, i += step)),\n parseHex(input.slice(i, i += step)),\n parseHex(input.slice(i, i + step) || 'ff'),\n ];\n }\n }\n // rgb(128 0 0), rgb(50% 0% 0%), rgba(255,0,255,0.6), rgb(255 0 255 / 60%), rgb(100% 0% 100% /.6)\n if (input.startsWith('rgb')) {\n const rgbRegExp = /^rgba?\\(\\s*([\\de.+-]+)(%)?(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)(%)?(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)(%)?(?:\\s*([,\\/])\\s*([\\de.+-]+)(%)?)?\\s*\\)$/;\n const rgbMatch = input.match(rgbRegExp);\n if (rgbMatch) {\n const [_, // eslint-disable-line @typescript-eslint/no-unused-vars\n r, // \n rp, // % (optional)\n f1, // , (optional)\n g, // \n gp, // % (optional)\n f2, // , (optional)\n b, // \n bp, // % (optional)\n f3, // ,|/ (optional)\n a, // (optional)\n ap, // % (optional)\n ] = rgbMatch;\n const argFormat = [f1 || ' ', f2 || ' ', f3].join('');\n if (argFormat === ' ' ||\n argFormat === ' /' ||\n argFormat === ',,' ||\n argFormat === ',,,') {\n const valFormat = [rp, gp, bp].join('');\n const maxValue = (valFormat === '%%%') ? 100 :\n (valFormat === '') ? 255 : 0;\n if (maxValue) {\n const rgba = [\n clamp(+r / maxValue, 0, 1),\n clamp(+g / maxValue, 0, 1),\n clamp(+b / maxValue, 0, 1),\n a ? parseAlpha(+a, ap) : 1,\n ];\n if (validateNumbers(rgba)) {\n return rgba;\n }\n // invalid numbers\n }\n // values must be all numbers or all percentages\n }\n return; // comma optional syntax requires no commas at all\n }\n }\n // hsl(120 50% 80%), hsla(120deg,50%,80%,.9), hsl(12e1 50% 80% / 90%)\n const hslRegExp = /^hsla?\\(\\s*([\\de.+-]+)(?:deg)?(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)%(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)%(?:\\s*([,\\/])\\s*([\\de.+-]+)(%)?)?\\s*\\)$/;\n const hslMatch = input.match(hslRegExp);\n if (hslMatch) {\n const [_, // eslint-disable-line @typescript-eslint/no-unused-vars\n h, // \n f1, // , (optional)\n s, // \n f2, // , (optional)\n l, // \n f3, // ,|/ (optional)\n a, // (optional)\n ap, // % (optional)\n ] = hslMatch;\n const argFormat = [f1 || ' ', f2 || ' ', f3].join('');\n if (argFormat === ' ' ||\n argFormat === ' /' ||\n argFormat === ',,' ||\n argFormat === ',,,') {\n const hsla = [\n +h,\n clamp(+s, 0, 100),\n clamp(+l, 0, 100),\n a ? parseAlpha(+a, ap) : 1,\n ];\n if (validateNumbers(hsla)) {\n return hslToRgb(hsla);\n }\n // invalid numbers\n }\n // comma optional syntax requires no commas at all\n }\n}\nfunction parseHex(hex) {\n return parseInt(hex.padEnd(2, hex), 16) / 255;\n}\nfunction parseAlpha(a, asPercentage) {\n return clamp(asPercentage ? (a / 100) : a, 0, 1);\n}\nfunction clamp(n, min, max) {\n return Math.min(Math.max(min, n), max);\n}\n/**\n * The regular expression for numeric values is not super specific, and it may\n * happen that it will accept a value that is not a valid number. In order to\n * detect and eliminate such values this function exists.\n *\n * @param array Array of uncertain numbers.\n * @returns `true` if the specified array contains only valid numbers, `false` otherwise.\n */\nfunction validateNumbers(array) {\n return !array.some(Number.isNaN);\n}\n/**\n * To generate:\n * - visit {@link https://www.w3.org/TR/css-color-4/#named-colors}\n * - run in the console:\n * @example\n * copy(`{\\n${[...document.querySelector('.named-color-table tbody').children].map((tr) => `${tr.cells[2].textContent.trim()}: [${tr.cells[4].textContent.trim().split(/\\s+/).join(', ')}],`).join('\\n')}\\n}`);\n */\nconst namedColors = {\n aliceblue: [240, 248, 255],\n antiquewhite: [250, 235, 215],\n aqua: [0, 255, 255],\n aquamarine: [127, 255, 212],\n azure: [240, 255, 255],\n beige: [245, 245, 220],\n bisque: [255, 228, 196],\n black: [0, 0, 0],\n blanchedalmond: [255, 235, 205],\n blue: [0, 0, 255],\n blueviolet: [138, 43, 226],\n brown: [165, 42, 42],\n burlywood: [222, 184, 135],\n cadetblue: [95, 158, 160],\n chartreuse: [127, 255, 0],\n chocolate: [210, 105, 30],\n coral: [255, 127, 80],\n cornflowerblue: [100, 149, 237],\n cornsilk: [255, 248, 220],\n crimson: [220, 20, 60],\n cyan: [0, 255, 255],\n darkblue: [0, 0, 139],\n darkcyan: [0, 139, 139],\n darkgoldenrod: [184, 134, 11],\n darkgray: [169, 169, 169],\n darkgreen: [0, 100, 0],\n darkgrey: [169, 169, 169],\n darkkhaki: [189, 183, 107],\n darkmagenta: [139, 0, 139],\n darkolivegreen: [85, 107, 47],\n darkorange: [255, 140, 0],\n darkorchid: [153, 50, 204],\n darkred: [139, 0, 0],\n darksalmon: [233, 150, 122],\n darkseagreen: [143, 188, 143],\n darkslateblue: [72, 61, 139],\n darkslategray: [47, 79, 79],\n darkslategrey: [47, 79, 79],\n darkturquoise: [0, 206, 209],\n darkviolet: [148, 0, 211],\n deeppink: [255, 20, 147],\n deepskyblue: [0, 191, 255],\n dimgray: [105, 105, 105],\n dimgrey: [105, 105, 105],\n dodgerblue: [30, 144, 255],\n firebrick: [178, 34, 34],\n floralwhite: [255, 250, 240],\n forestgreen: [34, 139, 34],\n fuchsia: [255, 0, 255],\n gainsboro: [220, 220, 220],\n ghostwhite: [248, 248, 255],\n gold: [255, 215, 0],\n goldenrod: [218, 165, 32],\n gray: [128, 128, 128],\n green: [0, 128, 0],\n greenyellow: [173, 255, 47],\n grey: [128, 128, 128],\n honeydew: [240, 255, 240],\n hotpink: [255, 105, 180],\n indianred: [205, 92, 92],\n indigo: [75, 0, 130],\n ivory: [255, 255, 240],\n khaki: [240, 230, 140],\n lavender: [230, 230, 250],\n lavenderblush: [255, 240, 245],\n lawngreen: [124, 252, 0],\n lemonchiffon: [255, 250, 205],\n lightblue: [173, 216, 230],\n lightcoral: [240, 128, 128],\n lightcyan: [224, 255, 255],\n lightgoldenrodyellow: [250, 250, 210],\n lightgray: [211, 211, 211],\n lightgreen: [144, 238, 144],\n lightgrey: [211, 211, 211],\n lightpink: [255, 182, 193],\n lightsalmon: [255, 160, 122],\n lightseagreen: [32, 178, 170],\n lightskyblue: [135, 206, 250],\n lightslategray: [119, 136, 153],\n lightslategrey: [119, 136, 153],\n lightsteelblue: [176, 196, 222],\n lightyellow: [255, 255, 224],\n lime: [0, 255, 0],\n limegreen: [50, 205, 50],\n linen: [250, 240, 230],\n magenta: [255, 0, 255],\n maroon: [128, 0, 0],\n mediumaquamarine: [102, 205, 170],\n mediumblue: [0, 0, 205],\n mediumorchid: [186, 85, 211],\n mediumpurple: [147, 112, 219],\n mediumseagreen: [60, 179, 113],\n mediumslateblue: [123, 104, 238],\n mediumspringgreen: [0, 250, 154],\n mediumturquoise: [72, 209, 204],\n mediumvioletred: [199, 21, 133],\n midnightblue: [25, 25, 112],\n mintcream: [245, 255, 250],\n mistyrose: [255, 228, 225],\n moccasin: [255, 228, 181],\n navajowhite: [255, 222, 173],\n navy: [0, 0, 128],\n oldlace: [253, 245, 230],\n olive: [128, 128, 0],\n olivedrab: [107, 142, 35],\n orange: [255, 165, 0],\n orangered: [255, 69, 0],\n orchid: [218, 112, 214],\n palegoldenrod: [238, 232, 170],\n palegreen: [152, 251, 152],\n paleturquoise: [175, 238, 238],\n palevioletred: [219, 112, 147],\n papayawhip: [255, 239, 213],\n peachpuff: [255, 218, 185],\n peru: [205, 133, 63],\n pink: [255, 192, 203],\n plum: [221, 160, 221],\n powderblue: [176, 224, 230],\n purple: [128, 0, 128],\n rebeccapurple: [102, 51, 153],\n red: [255, 0, 0],\n rosybrown: [188, 143, 143],\n royalblue: [65, 105, 225],\n saddlebrown: [139, 69, 19],\n salmon: [250, 128, 114],\n sandybrown: [244, 164, 96],\n seagreen: [46, 139, 87],\n seashell: [255, 245, 238],\n sienna: [160, 82, 45],\n silver: [192, 192, 192],\n skyblue: [135, 206, 235],\n slateblue: [106, 90, 205],\n slategray: [112, 128, 144],\n slategrey: [112, 128, 144],\n snow: [255, 250, 250],\n springgreen: [0, 255, 127],\n steelblue: [70, 130, 180],\n tan: [210, 180, 140],\n teal: [0, 128, 128],\n thistle: [216, 191, 216],\n tomato: [255, 99, 71],\n turquoise: [64, 224, 208],\n violet: [238, 130, 238],\n wheat: [245, 222, 179],\n white: [255, 255, 255],\n whitesmoke: [245, 245, 245],\n yellow: [255, 255, 0],\n yellowgreen: [154, 205, 50],\n};\n\n/**\n * Color representation used by WebGL.\n * Defined in sRGB color space and pre-blended with alpha.\n * @private\n */\nclass Color {\n /**\n * @param r Red component premultiplied by `alpha` 0..1\n * @param g Green component premultiplied by `alpha` 0..1\n * @param b Blue component premultiplied by `alpha` 0..1\n * @param [alpha=1] Alpha component 0..1\n * @param [premultiplied=true] Whether the `r`, `g` and `b` values have already\n * been multiplied by alpha. If `true` nothing happens if `false` then they will\n * be multiplied automatically.\n */\n constructor(r, g, b, alpha = 1, premultiplied = true) {\n this.r = r;\n this.g = g;\n this.b = b;\n this.a = alpha;\n if (!premultiplied) {\n this.r *= alpha;\n this.g *= alpha;\n this.b *= alpha;\n if (!alpha) {\n // alpha = 0 erases completely rgb channels. This behavior is not desirable\n // if this particular color is later used in color interpolation.\n // Because of that, a reference to original color is saved.\n this.overwriteGetter('rgb', [r, g, b, alpha]);\n }\n }\n }\n /**\n * Parses CSS color strings and converts colors to sRGB color space if needed.\n * Officially supported color formats:\n * - keyword, e.g. 'aquamarine' or 'steelblue'\n * - hex (with 3, 4, 6 or 8 digits), e.g. '#f0f' or '#e9bebea9'\n * - rgb and rgba, e.g. 'rgb(0,240,120)' or 'rgba(0%,94%,47%,0.1)' or 'rgb(0 240 120 / .3)'\n * - hsl and hsla, e.g. 'hsl(0,0%,83%)' or 'hsla(0,0%,83%,.5)' or 'hsl(0 0% 83% / 20%)'\n *\n * @param input CSS color string to parse.\n * @returns A `Color` instance, or `undefined` if the input is not a valid color string.\n */\n static parse(input) {\n // in zoom-and-property function input could be an instance of Color class\n if (input instanceof Color) {\n return input;\n }\n if (typeof input !== 'string') {\n return;\n }\n const rgba = parseCssColor(input);\n if (rgba) {\n return new Color(...rgba, false);\n }\n }\n /**\n * Used in color interpolation and by 'to-rgba' expression.\n *\n * @returns Gien color, with reversed alpha blending, in sRGB color space.\n */\n get rgb() {\n const { r, g, b, a } = this;\n const f = a || Infinity; // reverse alpha blending factor\n return this.overwriteGetter('rgb', [r / f, g / f, b / f, a]);\n }\n /**\n * Used in color interpolation.\n *\n * @returns Gien color, with reversed alpha blending, in HCL color space.\n */\n get hcl() {\n return this.overwriteGetter('hcl', rgbToHcl(this.rgb));\n }\n /**\n * Used in color interpolation.\n *\n * @returns Gien color, with reversed alpha blending, in LAB color space.\n */\n get lab() {\n return this.overwriteGetter('lab', rgbToLab(this.rgb));\n }\n /**\n * Lazy getter pattern. When getter is called for the first time lazy value\n * is calculated and then overwrites getter function in given object instance.\n *\n * @example:\n * const redColor = Color.parse('red');\n * let x = redColor.hcl; // this will invoke `get hcl()`, which will calculate\n * // the value of red in HCL space and invoke this `overwriteGetter` function\n * // which in turn will set a field with a key 'hcl' in the `redColor` object.\n * // In other words it will override `get hcl()` from its `Color` prototype\n * // with its own property: hcl = [calculated red value in hcl].\n * let y = redColor.hcl; // next call will no longer invoke getter but simply\n * // return the previously calculated value\n * x === y; // true - `x` is exactly the same object as `y`\n *\n * @param getterKey Getter key\n * @param lazyValue Lazily calculated value to be memoized by current instance\n * @private\n */\n overwriteGetter(getterKey, lazyValue) {\n Object.defineProperty(this, getterKey, { value: lazyValue });\n return lazyValue;\n }\n /**\n * Used by 'to-string' expression.\n *\n * @returns Serialized color in format `rgba(r,g,b,a)`\n * where r,g,b are numbers within 0..255 and alpha is number within 1..0\n *\n * @example\n * var purple = new Color.parse('purple');\n * purple.toString; // = \"rgba(128,0,128,1)\"\n * var translucentGreen = new Color.parse('rgba(26, 207, 26, .73)');\n * translucentGreen.toString(); // = \"rgba(26,207,26,0.73)\"\n */\n toString() {\n const [r, g, b, a] = this.rgb;\n return `rgba(${[r, g, b].map(n => Math.round(n * 255)).join(',')},${a})`;\n }\n}\nColor.black = new Color(0, 0, 0, 1);\nColor.white = new Color(1, 1, 1, 1);\nColor.transparent = new Color(0, 0, 0, 0);\nColor.red = new Color(1, 0, 0, 1);\n\n// Flow type declarations for Intl cribbed from\n// https://github.com/facebook/flow/issues/1270\nclass Collator {\n constructor(caseSensitive, diacriticSensitive, locale) {\n if (caseSensitive)\n this.sensitivity = diacriticSensitive ? 'variant' : 'case';\n else\n this.sensitivity = diacriticSensitive ? 'accent' : 'base';\n this.locale = locale;\n this.collator = new Intl.Collator(this.locale ? this.locale : [], { sensitivity: this.sensitivity, usage: 'search' });\n }\n compare(lhs, rhs) {\n return this.collator.compare(lhs, rhs);\n }\n resolvedLocale() {\n // We create a Collator without \"usage: search\" because we don't want\n // the search options encoded in our result (e.g. \"en-u-co-search\")\n return new Intl.Collator(this.locale ? this.locale : [])\n .resolvedOptions().locale;\n }\n}\n\nclass FormattedSection {\n constructor(text, image, scale, fontStack, textColor) {\n this.text = text;\n this.image = image;\n this.scale = scale;\n this.fontStack = fontStack;\n this.textColor = textColor;\n }\n}\nclass Formatted {\n constructor(sections) {\n this.sections = sections;\n }\n static fromString(unformatted) {\n return new Formatted([new FormattedSection(unformatted, null, null, null, null)]);\n }\n isEmpty() {\n if (this.sections.length === 0)\n return true;\n return !this.sections.some(section => section.text.length !== 0 ||\n (section.image && section.image.name.length !== 0));\n }\n static factory(text) {\n if (text instanceof Formatted) {\n return text;\n }\n else {\n return Formatted.fromString(text);\n }\n }\n toString() {\n if (this.sections.length === 0)\n return '';\n return this.sections.map(section => section.text).join('');\n }\n}\n\n/**\n * A set of four numbers representing padding around a box. Create instances from\n * bare arrays or numeric values using the static method `Padding.parse`.\n * @private\n */\nclass Padding {\n constructor(values) {\n this.values = values.slice();\n }\n /**\n * Numeric padding values\n * @param input A padding value\n * @returns A `Padding` instance, or `undefined` if the input is not a valid padding value.\n */\n static parse(input) {\n if (input instanceof Padding) {\n return input;\n }\n // Backwards compatibility: bare number is treated the same as array with single value.\n // Padding applies to all four sides.\n if (typeof input === 'number') {\n return new Padding([input, input, input, input]);\n }\n if (!Array.isArray(input)) {\n return undefined;\n }\n if (input.length < 1 || input.length > 4) {\n return undefined;\n }\n for (const val of input) {\n if (typeof val !== 'number') {\n return undefined;\n }\n }\n // Expand shortcut properties into explicit 4-sided values\n switch (input.length) {\n case 1:\n input = [input[0], input[0], input[0], input[0]];\n break;\n case 2:\n input = [input[0], input[1], input[0], input[1]];\n break;\n case 3:\n input = [input[0], input[1], input[2], input[1]];\n break;\n }\n return new Padding(input);\n }\n toString() {\n return JSON.stringify(this.values);\n }\n}\n\n/** Set of valid anchor positions, as a set for validation */\nconst anchors = new Set(['center', 'left', 'right', 'top', 'bottom', 'top-left', 'top-right', 'bottom-left', 'bottom-right']);\n/**\n * Utility class to assist managing values for text-variable-anchor-offset property. Create instances from\n * bare arrays using the static method `VariableAnchorOffsetCollection.parse`.\n * @private\n */\nclass VariableAnchorOffsetCollection {\n constructor(values) {\n this.values = values.slice();\n }\n static parse(input) {\n if (input instanceof VariableAnchorOffsetCollection) {\n return input;\n }\n if (!Array.isArray(input) ||\n input.length < 1 ||\n input.length % 2 !== 0) {\n return undefined;\n }\n for (let i = 0; i < input.length; i += 2) {\n // Elements in even positions should be anchor positions; Elements in odd positions should be offset values\n const anchorValue = input[i];\n const offsetValue = input[i + 1];\n if (typeof anchorValue !== 'string' || !anchors.has(anchorValue)) {\n return undefined;\n }\n if (!Array.isArray(offsetValue) || offsetValue.length !== 2 || typeof offsetValue[0] !== 'number' || typeof offsetValue[1] !== 'number') {\n return undefined;\n }\n }\n return new VariableAnchorOffsetCollection(input);\n }\n toString() {\n return JSON.stringify(this.values);\n }\n}\n\nclass ResolvedImage {\n constructor(options) {\n this.name = options.name;\n this.available = options.available;\n }\n toString() {\n return this.name;\n }\n static fromString(name) {\n if (!name)\n return null; // treat empty values as no image\n return new ResolvedImage({ name, available: false });\n }\n}\n\nfunction validateRGBA(r, g, b, a) {\n if (!(typeof r === 'number' && r >= 0 && r <= 255 &&\n typeof g === 'number' && g >= 0 && g <= 255 &&\n typeof b === 'number' && b >= 0 && b <= 255)) {\n const value = typeof a === 'number' ? [r, g, b, a] : [r, g, b];\n return `Invalid rgba value [${value.join(', ')}]: 'r', 'g', and 'b' must be between 0 and 255.`;\n }\n if (!(typeof a === 'undefined' || (typeof a === 'number' && a >= 0 && a <= 1))) {\n return `Invalid rgba value [${[r, g, b, a].join(', ')}]: 'a' must be between 0 and 1.`;\n }\n return null;\n}\nfunction isValue(mixed) {\n if (mixed === null ||\n typeof mixed === 'string' ||\n typeof mixed === 'boolean' ||\n typeof mixed === 'number' ||\n mixed instanceof Color ||\n mixed instanceof Collator ||\n mixed instanceof Formatted ||\n mixed instanceof Padding ||\n mixed instanceof VariableAnchorOffsetCollection ||\n mixed instanceof ResolvedImage) {\n return true;\n }\n else if (Array.isArray(mixed)) {\n for (const item of mixed) {\n if (!isValue(item)) {\n return false;\n }\n }\n return true;\n }\n else if (typeof mixed === 'object') {\n for (const key in mixed) {\n if (!isValue(mixed[key])) {\n return false;\n }\n }\n return true;\n }\n else {\n return false;\n }\n}\nfunction typeOf(value) {\n if (value === null) {\n return NullType;\n }\n else if (typeof value === 'string') {\n return StringType;\n }\n else if (typeof value === 'boolean') {\n return BooleanType;\n }\n else if (typeof value === 'number') {\n return NumberType;\n }\n else if (value instanceof Color) {\n return ColorType;\n }\n else if (value instanceof Collator) {\n return CollatorType;\n }\n else if (value instanceof Formatted) {\n return FormattedType;\n }\n else if (value instanceof Padding) {\n return PaddingType;\n }\n else if (value instanceof VariableAnchorOffsetCollection) {\n return VariableAnchorOffsetCollectionType;\n }\n else if (value instanceof ResolvedImage) {\n return ResolvedImageType;\n }\n else if (Array.isArray(value)) {\n const length = value.length;\n let itemType;\n for (const item of value) {\n const t = typeOf(item);\n if (!itemType) {\n itemType = t;\n }\n else if (itemType === t) {\n continue;\n }\n else {\n itemType = ValueType;\n break;\n }\n }\n return array$1(itemType || ValueType, length);\n }\n else {\n return ObjectType;\n }\n}\nfunction toString(value) {\n const type = typeof value;\n if (value === null) {\n return '';\n }\n else if (type === 'string' || type === 'number' || type === 'boolean') {\n return String(value);\n }\n else if (value instanceof Color || value instanceof Formatted || value instanceof Padding || value instanceof VariableAnchorOffsetCollection || value instanceof ResolvedImage) {\n return value.toString();\n }\n else {\n return JSON.stringify(value);\n }\n}\n\nclass Literal {\n constructor(type, value) {\n this.type = type;\n this.value = value;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error(`'literal' expression requires exactly one argument, but found ${args.length - 1} instead.`);\n if (!isValue(args[1]))\n return context.error('invalid value');\n const value = args[1];\n let type = typeOf(value);\n // special case: infer the item type if possible for zero-length arrays\n const expected = context.expectedType;\n if (type.kind === 'array' &&\n type.N === 0 &&\n expected &&\n expected.kind === 'array' &&\n (typeof expected.N !== 'number' || expected.N === 0)) {\n type = expected;\n }\n return new Literal(type, value);\n }\n evaluate() {\n return this.value;\n }\n eachChild() { }\n outputDefined() {\n return true;\n }\n}\n\nclass RuntimeError {\n constructor(message) {\n this.name = 'ExpressionEvaluationError';\n this.message = message;\n }\n toJSON() {\n return this.message;\n }\n}\n\nconst types$1 = {\n string: StringType,\n number: NumberType,\n boolean: BooleanType,\n object: ObjectType\n};\nclass Assertion {\n constructor(type, args) {\n this.type = type;\n this.args = args;\n }\n static parse(args, context) {\n if (args.length < 2)\n return context.error('Expected at least one argument.');\n let i = 1;\n let type;\n const name = args[0];\n if (name === 'array') {\n let itemType;\n if (args.length > 2) {\n const type = args[1];\n if (typeof type !== 'string' || !(type in types$1) || type === 'object')\n return context.error('The item type argument of \"array\" must be one of string, number, boolean', 1);\n itemType = types$1[type];\n i++;\n }\n else {\n itemType = ValueType;\n }\n let N;\n if (args.length > 3) {\n if (args[2] !== null &&\n (typeof args[2] !== 'number' ||\n args[2] < 0 ||\n args[2] !== Math.floor(args[2]))) {\n return context.error('The length argument to \"array\" must be a positive integer literal', 2);\n }\n N = args[2];\n i++;\n }\n type = array$1(itemType, N);\n }\n else {\n if (!types$1[name])\n throw new Error(`Types doesn't contain name = ${name}`);\n type = types$1[name];\n }\n const parsed = [];\n for (; i < args.length; i++) {\n const input = context.parse(args[i], i, ValueType);\n if (!input)\n return null;\n parsed.push(input);\n }\n return new Assertion(type, parsed);\n }\n evaluate(ctx) {\n for (let i = 0; i < this.args.length; i++) {\n const value = this.args[i].evaluate(ctx);\n const error = checkSubtype(this.type, typeOf(value));\n if (!error) {\n return value;\n }\n else if (i === this.args.length - 1) {\n throw new RuntimeError(`Expected value to be of type ${toString$1(this.type)}, but found ${toString$1(typeOf(value))} instead.`);\n }\n }\n throw new Error();\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return this.args.every(arg => arg.outputDefined());\n }\n}\n\nconst types = {\n 'to-boolean': BooleanType,\n 'to-color': ColorType,\n 'to-number': NumberType,\n 'to-string': StringType\n};\n/**\n * Special form for error-coalescing coercion expressions \"to-number\",\n * \"to-color\". Since these coercions can fail at runtime, they accept multiple\n * arguments, only evaluating one at a time until one succeeds.\n *\n * @private\n */\nclass Coercion {\n constructor(type, args) {\n this.type = type;\n this.args = args;\n }\n static parse(args, context) {\n if (args.length < 2)\n return context.error('Expected at least one argument.');\n const name = args[0];\n if (!types[name])\n throw new Error(`Can't parse ${name} as it is not part of the known types`);\n if ((name === 'to-boolean' || name === 'to-string') && args.length !== 2)\n return context.error('Expected one argument.');\n const type = types[name];\n const parsed = [];\n for (let i = 1; i < args.length; i++) {\n const input = context.parse(args[i], i, ValueType);\n if (!input)\n return null;\n parsed.push(input);\n }\n return new Coercion(type, parsed);\n }\n evaluate(ctx) {\n switch (this.type.kind) {\n case 'boolean':\n return Boolean(this.args[0].evaluate(ctx));\n case 'color': {\n let input;\n let error;\n for (const arg of this.args) {\n input = arg.evaluate(ctx);\n error = null;\n if (input instanceof Color) {\n return input;\n }\n else if (typeof input === 'string') {\n const c = ctx.parseColor(input);\n if (c)\n return c;\n }\n else if (Array.isArray(input)) {\n if (input.length < 3 || input.length > 4) {\n error = `Invalid rbga value ${JSON.stringify(input)}: expected an array containing either three or four numeric values.`;\n }\n else {\n error = validateRGBA(input[0], input[1], input[2], input[3]);\n }\n if (!error) {\n return new Color(input[0] / 255, input[1] / 255, input[2] / 255, input[3]);\n }\n }\n }\n throw new RuntimeError(error || `Could not parse color from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`);\n }\n case 'padding': {\n let input;\n for (const arg of this.args) {\n input = arg.evaluate(ctx);\n const pad = Padding.parse(input);\n if (pad) {\n return pad;\n }\n }\n throw new RuntimeError(`Could not parse padding from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`);\n }\n case 'variableAnchorOffsetCollection': {\n let input;\n for (const arg of this.args) {\n input = arg.evaluate(ctx);\n const coll = VariableAnchorOffsetCollection.parse(input);\n if (coll) {\n return coll;\n }\n }\n throw new RuntimeError(`Could not parse variableAnchorOffsetCollection from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`);\n }\n case 'number': {\n let value = null;\n for (const arg of this.args) {\n value = arg.evaluate(ctx);\n if (value === null)\n return 0;\n const num = Number(value);\n if (isNaN(num))\n continue;\n return num;\n }\n throw new RuntimeError(`Could not convert ${JSON.stringify(value)} to number.`);\n }\n case 'formatted':\n // There is no explicit 'to-formatted' but this coercion can be implicitly\n // created by properties that expect the 'formatted' type.\n return Formatted.fromString(toString(this.args[0].evaluate(ctx)));\n case 'resolvedImage':\n return ResolvedImage.fromString(toString(this.args[0].evaluate(ctx)));\n default:\n return toString(this.args[0].evaluate(ctx));\n }\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return this.args.every(arg => arg.outputDefined());\n }\n}\n\nconst geometryTypes = ['Unknown', 'Point', 'LineString', 'Polygon'];\nclass EvaluationContext {\n constructor() {\n this.globals = null;\n this.feature = null;\n this.featureState = null;\n this.formattedSection = null;\n this._parseColorCache = {};\n this.availableImages = null;\n this.canonical = null;\n }\n id() {\n return this.feature && 'id' in this.feature ? this.feature.id : null;\n }\n geometryType() {\n return this.feature ? typeof this.feature.type === 'number' ? geometryTypes[this.feature.type] : this.feature.type : null;\n }\n geometry() {\n return this.feature && 'geometry' in this.feature ? this.feature.geometry : null;\n }\n canonicalID() {\n return this.canonical;\n }\n properties() {\n return this.feature && this.feature.properties || {};\n }\n parseColor(input) {\n let cached = this._parseColorCache[input];\n if (!cached) {\n cached = this._parseColorCache[input] = Color.parse(input);\n }\n return cached;\n }\n}\n\n/**\n * State associated parsing at a given point in an expression tree.\n * @private\n */\nclass ParsingContext {\n constructor(registry, isConstantFunc, path = [], expectedType, scope = new Scope(), errors = []) {\n this.registry = registry;\n this.path = path;\n this.key = path.map(part => `[${part}]`).join('');\n this.scope = scope;\n this.errors = errors;\n this.expectedType = expectedType;\n this._isConstant = isConstantFunc;\n }\n /**\n * @param expr the JSON expression to parse\n * @param index the optional argument index if this expression is an argument of a parent expression that's being parsed\n * @param options\n * @param options.omitTypeAnnotations set true to omit inferred type annotations. Caller beware: with this option set, the parsed expression's type will NOT satisfy `expectedType` if it would normally be wrapped in an inferred annotation.\n * @private\n */\n parse(expr, index, expectedType, bindings, options = {}) {\n if (index) {\n return this.concat(index, expectedType, bindings)._parse(expr, options);\n }\n return this._parse(expr, options);\n }\n _parse(expr, options) {\n if (expr === null || typeof expr === 'string' || typeof expr === 'boolean' || typeof expr === 'number') {\n expr = ['literal', expr];\n }\n function annotate(parsed, type, typeAnnotation) {\n if (typeAnnotation === 'assert') {\n return new Assertion(type, [parsed]);\n }\n else if (typeAnnotation === 'coerce') {\n return new Coercion(type, [parsed]);\n }\n else {\n return parsed;\n }\n }\n if (Array.isArray(expr)) {\n if (expr.length === 0) {\n return this.error('Expected an array with at least one element. If you wanted a literal array, use [\"literal\", []].');\n }\n const op = expr[0];\n if (typeof op !== 'string') {\n this.error(`Expression name must be a string, but found ${typeof op} instead. If you wanted a literal array, use [\"literal\", [...]].`, 0);\n return null;\n }\n const Expr = this.registry[op];\n if (Expr) {\n let parsed = Expr.parse(expr, this);\n if (!parsed)\n return null;\n if (this.expectedType) {\n const expected = this.expectedType;\n const actual = parsed.type;\n // When we expect a number, string, boolean, or array but have a value, wrap it in an assertion.\n // When we expect a color or formatted string, but have a string or value, wrap it in a coercion.\n // Otherwise, we do static type-checking.\n //\n // These behaviors are overridable for:\n // * The \"coalesce\" operator, which needs to omit type annotations.\n // * String-valued properties (e.g. `text-field`), where coercion is more convenient than assertion.\n //\n if ((expected.kind === 'string' || expected.kind === 'number' || expected.kind === 'boolean' || expected.kind === 'object' || expected.kind === 'array') && actual.kind === 'value') {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'assert');\n }\n else if ((expected.kind === 'color' || expected.kind === 'formatted' || expected.kind === 'resolvedImage') && (actual.kind === 'value' || actual.kind === 'string')) {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce');\n }\n else if (expected.kind === 'padding' && (actual.kind === 'value' || actual.kind === 'number' || actual.kind === 'array')) {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce');\n }\n else if (expected.kind === 'variableAnchorOffsetCollection' && (actual.kind === 'value' || actual.kind === 'array')) {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce');\n }\n else if (this.checkSubtype(expected, actual)) {\n return null;\n }\n }\n // If an expression's arguments are all literals, we can evaluate\n // it immediately and replace it with a literal value in the\n // parsed/compiled result. Expressions that expect an image should\n // not be resolved here so we can later get the available images.\n if (!(parsed instanceof Literal) && (parsed.type.kind !== 'resolvedImage') && this._isConstant(parsed)) {\n const ec = new EvaluationContext();\n try {\n parsed = new Literal(parsed.type, parsed.evaluate(ec));\n }\n catch (e) {\n this.error(e.message);\n return null;\n }\n }\n return parsed;\n }\n return this.error(`Unknown expression \"${op}\". If you wanted a literal array, use [\"literal\", [...]].`, 0);\n }\n else if (typeof expr === 'undefined') {\n return this.error('\\'undefined\\' value invalid. Use null instead.');\n }\n else if (typeof expr === 'object') {\n return this.error('Bare objects invalid. Use [\"literal\", {...}] instead.');\n }\n else {\n return this.error(`Expected an array, but found ${typeof expr} instead.`);\n }\n }\n /**\n * Returns a copy of this context suitable for parsing the subexpression at\n * index `index`, optionally appending to 'let' binding map.\n *\n * Note that `errors` property, intended for collecting errors while\n * parsing, is copied by reference rather than cloned.\n * @private\n */\n concat(index, expectedType, bindings) {\n const path = typeof index === 'number' ? this.path.concat(index) : this.path;\n const scope = bindings ? this.scope.concat(bindings) : this.scope;\n return new ParsingContext(this.registry, this._isConstant, path, expectedType || null, scope, this.errors);\n }\n /**\n * Push a parsing (or type checking) error into the `this.errors`\n * @param error The message\n * @param keys Optionally specify the source of the error at a child\n * of the current expression at `this.key`.\n * @private\n */\n error(error, ...keys) {\n const key = `${this.key}${keys.map(k => `[${k}]`).join('')}`;\n this.errors.push(new ExpressionParsingError(key, error));\n }\n /**\n * Returns null if `t` is a subtype of `expected`; otherwise returns an\n * error message and also pushes it to `this.errors`.\n * @param expected The expected type\n * @param t The actual type\n * @returns null if `t` is a subtype of `expected`; otherwise returns an error message\n */\n checkSubtype(expected, t) {\n const error = checkSubtype(expected, t);\n if (error)\n this.error(error);\n return error;\n }\n}\n\nclass CollatorExpression {\n constructor(caseSensitive, diacriticSensitive, locale) {\n this.type = CollatorType;\n this.locale = locale;\n this.caseSensitive = caseSensitive;\n this.diacriticSensitive = diacriticSensitive;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error('Expected one argument.');\n const options = args[1];\n if (typeof options !== 'object' || Array.isArray(options))\n return context.error('Collator options argument must be an object.');\n const caseSensitive = context.parse(options['case-sensitive'] === undefined ? false : options['case-sensitive'], 1, BooleanType);\n if (!caseSensitive)\n return null;\n const diacriticSensitive = context.parse(options['diacritic-sensitive'] === undefined ? false : options['diacritic-sensitive'], 1, BooleanType);\n if (!diacriticSensitive)\n return null;\n let locale = null;\n if (options['locale']) {\n locale = context.parse(options['locale'], 1, StringType);\n if (!locale)\n return null;\n }\n return new CollatorExpression(caseSensitive, diacriticSensitive, locale);\n }\n evaluate(ctx) {\n return new Collator(this.caseSensitive.evaluate(ctx), this.diacriticSensitive.evaluate(ctx), this.locale ? this.locale.evaluate(ctx) : null);\n }\n eachChild(fn) {\n fn(this.caseSensitive);\n fn(this.diacriticSensitive);\n if (this.locale) {\n fn(this.locale);\n }\n }\n outputDefined() {\n // Technically the set of possible outputs is the combinatoric set of Collators produced\n // by all possible outputs of locale/caseSensitive/diacriticSensitive\n // But for the primary use of Collators in comparison operators, we ignore the Collator's\n // possible outputs anyway, so we can get away with leaving this false for now.\n return false;\n }\n}\n\nconst EXTENT = 8192;\nfunction updateBBox(bbox, coord) {\n bbox[0] = Math.min(bbox[0], coord[0]);\n bbox[1] = Math.min(bbox[1], coord[1]);\n bbox[2] = Math.max(bbox[2], coord[0]);\n bbox[3] = Math.max(bbox[3], coord[1]);\n}\nfunction mercatorXfromLng(lng) {\n return (180 + lng) / 360;\n}\nfunction mercatorYfromLat(lat) {\n return (180 - (180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360)))) / 360;\n}\nfunction boxWithinBox(bbox1, bbox2) {\n if (bbox1[0] <= bbox2[0])\n return false;\n if (bbox1[2] >= bbox2[2])\n return false;\n if (bbox1[1] <= bbox2[1])\n return false;\n if (bbox1[3] >= bbox2[3])\n return false;\n return true;\n}\nfunction getTileCoordinates(p, canonical) {\n const x = mercatorXfromLng(p[0]);\n const y = mercatorYfromLat(p[1]);\n const tilesAtZoom = Math.pow(2, canonical.z);\n return [Math.round(x * tilesAtZoom * EXTENT), Math.round(y * tilesAtZoom * EXTENT)];\n}\nfunction onBoundary(p, p1, p2) {\n const x1 = p[0] - p1[0];\n const y1 = p[1] - p1[1];\n const x2 = p[0] - p2[0];\n const y2 = p[1] - p2[1];\n return (x1 * y2 - x2 * y1 === 0) && (x1 * x2 <= 0) && (y1 * y2 <= 0);\n}\nfunction rayIntersect(p, p1, p2) {\n return ((p1[1] > p[1]) !== (p2[1] > p[1])) && (p[0] < (p2[0] - p1[0]) * (p[1] - p1[1]) / (p2[1] - p1[1]) + p1[0]);\n}\n// ray casting algorithm for detecting if point is in polygon\nfunction pointWithinPolygon(point, rings) {\n let inside = false;\n for (let i = 0, len = rings.length; i < len; i++) {\n const ring = rings[i];\n for (let j = 0, len2 = ring.length; j < len2 - 1; j++) {\n if (onBoundary(point, ring[j], ring[j + 1]))\n return false;\n if (rayIntersect(point, ring[j], ring[j + 1]))\n inside = !inside;\n }\n }\n return inside;\n}\nfunction pointWithinPolygons(point, polygons) {\n for (let i = 0; i < polygons.length; i++) {\n if (pointWithinPolygon(point, polygons[i]))\n return true;\n }\n return false;\n}\nfunction perp(v1, v2) {\n return (v1[0] * v2[1] - v1[1] * v2[0]);\n}\n// check if p1 and p2 are in different sides of line segment q1->q2\nfunction twoSided(p1, p2, q1, q2) {\n // q1->p1 (x1, y1), q1->p2 (x2, y2), q1->q2 (x3, y3)\n const x1 = p1[0] - q1[0];\n const y1 = p1[1] - q1[1];\n const x2 = p2[0] - q1[0];\n const y2 = p2[1] - q1[1];\n const x3 = q2[0] - q1[0];\n const y3 = q2[1] - q1[1];\n const det1 = (x1 * y3 - x3 * y1);\n const det2 = (x2 * y3 - x3 * y2);\n if ((det1 > 0 && det2 < 0) || (det1 < 0 && det2 > 0))\n return true;\n return false;\n}\n// a, b are end points for line segment1, c and d are end points for line segment2\nfunction lineIntersectLine(a, b, c, d) {\n // check if two segments are parallel or not\n // precondition is end point a, b is inside polygon, if line a->b is\n // parallel to polygon edge c->d, then a->b won't intersect with c->d\n const vectorP = [b[0] - a[0], b[1] - a[1]];\n const vectorQ = [d[0] - c[0], d[1] - c[1]];\n if (perp(vectorQ, vectorP) === 0)\n return false;\n // If lines are intersecting with each other, the relative location should be:\n // a and b lie in different sides of segment c->d\n // c and d lie in different sides of segment a->b\n if (twoSided(a, b, c, d) && twoSided(c, d, a, b))\n return true;\n return false;\n}\nfunction lineIntersectPolygon(p1, p2, polygon) {\n for (const ring of polygon) {\n // loop through every edge of the ring\n for (let j = 0; j < ring.length - 1; ++j) {\n if (lineIntersectLine(p1, p2, ring[j], ring[j + 1])) {\n return true;\n }\n }\n }\n return false;\n}\nfunction lineStringWithinPolygon(line, polygon) {\n // First, check if geometry points of line segments are all inside polygon\n for (let i = 0; i < line.length; ++i) {\n if (!pointWithinPolygon(line[i], polygon)) {\n return false;\n }\n }\n // Second, check if there is line segment intersecting polygon edge\n for (let i = 0; i < line.length - 1; ++i) {\n if (lineIntersectPolygon(line[i], line[i + 1], polygon)) {\n return false;\n }\n }\n return true;\n}\nfunction lineStringWithinPolygons(line, polygons) {\n for (let i = 0; i < polygons.length; i++) {\n if (lineStringWithinPolygon(line, polygons[i]))\n return true;\n }\n return false;\n}\nfunction getTilePolygon(coordinates, bbox, canonical) {\n const polygon = [];\n for (let i = 0; i < coordinates.length; i++) {\n const ring = [];\n for (let j = 0; j < coordinates[i].length; j++) {\n const coord = getTileCoordinates(coordinates[i][j], canonical);\n updateBBox(bbox, coord);\n ring.push(coord);\n }\n polygon.push(ring);\n }\n return polygon;\n}\nfunction getTilePolygons(coordinates, bbox, canonical) {\n const polygons = [];\n for (let i = 0; i < coordinates.length; i++) {\n const polygon = getTilePolygon(coordinates[i], bbox, canonical);\n polygons.push(polygon);\n }\n return polygons;\n}\nfunction updatePoint(p, bbox, polyBBox, worldSize) {\n if (p[0] < polyBBox[0] || p[0] > polyBBox[2]) {\n const halfWorldSize = worldSize * 0.5;\n let shift = (p[0] - polyBBox[0] > halfWorldSize) ? -worldSize : (polyBBox[0] - p[0] > halfWorldSize) ? worldSize : 0;\n if (shift === 0) {\n shift = (p[0] - polyBBox[2] > halfWorldSize) ? -worldSize : (polyBBox[2] - p[0] > halfWorldSize) ? worldSize : 0;\n }\n p[0] += shift;\n }\n updateBBox(bbox, p);\n}\nfunction resetBBox(bbox) {\n bbox[0] = bbox[1] = Infinity;\n bbox[2] = bbox[3] = -Infinity;\n}\nfunction getTilePoints(geometry, pointBBox, polyBBox, canonical) {\n const worldSize = Math.pow(2, canonical.z) * EXTENT;\n const shifts = [canonical.x * EXTENT, canonical.y * EXTENT];\n const tilePoints = [];\n for (const points of geometry) {\n for (const point of points) {\n const p = [point.x + shifts[0], point.y + shifts[1]];\n updatePoint(p, pointBBox, polyBBox, worldSize);\n tilePoints.push(p);\n }\n }\n return tilePoints;\n}\nfunction getTileLines(geometry, lineBBox, polyBBox, canonical) {\n const worldSize = Math.pow(2, canonical.z) * EXTENT;\n const shifts = [canonical.x * EXTENT, canonical.y * EXTENT];\n const tileLines = [];\n for (const line of geometry) {\n const tileLine = [];\n for (const point of line) {\n const p = [point.x + shifts[0], point.y + shifts[1]];\n updateBBox(lineBBox, p);\n tileLine.push(p);\n }\n tileLines.push(tileLine);\n }\n if (lineBBox[2] - lineBBox[0] <= worldSize / 2) {\n resetBBox(lineBBox);\n for (const line of tileLines) {\n for (const p of line) {\n updatePoint(p, lineBBox, polyBBox, worldSize);\n }\n }\n }\n return tileLines;\n}\nfunction pointsWithinPolygons(ctx, polygonGeometry) {\n const pointBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const polyBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const canonical = ctx.canonicalID();\n if (polygonGeometry.type === 'Polygon') {\n const tilePolygon = getTilePolygon(polygonGeometry.coordinates, polyBBox, canonical);\n const tilePoints = getTilePoints(ctx.geometry(), pointBBox, polyBBox, canonical);\n if (!boxWithinBox(pointBBox, polyBBox))\n return false;\n for (const point of tilePoints) {\n if (!pointWithinPolygon(point, tilePolygon))\n return false;\n }\n }\n if (polygonGeometry.type === 'MultiPolygon') {\n const tilePolygons = getTilePolygons(polygonGeometry.coordinates, polyBBox, canonical);\n const tilePoints = getTilePoints(ctx.geometry(), pointBBox, polyBBox, canonical);\n if (!boxWithinBox(pointBBox, polyBBox))\n return false;\n for (const point of tilePoints) {\n if (!pointWithinPolygons(point, tilePolygons))\n return false;\n }\n }\n return true;\n}\nfunction linesWithinPolygons(ctx, polygonGeometry) {\n const lineBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const polyBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const canonical = ctx.canonicalID();\n if (polygonGeometry.type === 'Polygon') {\n const tilePolygon = getTilePolygon(polygonGeometry.coordinates, polyBBox, canonical);\n const tileLines = getTileLines(ctx.geometry(), lineBBox, polyBBox, canonical);\n if (!boxWithinBox(lineBBox, polyBBox))\n return false;\n for (const line of tileLines) {\n if (!lineStringWithinPolygon(line, tilePolygon))\n return false;\n }\n }\n if (polygonGeometry.type === 'MultiPolygon') {\n const tilePolygons = getTilePolygons(polygonGeometry.coordinates, polyBBox, canonical);\n const tileLines = getTileLines(ctx.geometry(), lineBBox, polyBBox, canonical);\n if (!boxWithinBox(lineBBox, polyBBox))\n return false;\n for (const line of tileLines) {\n if (!lineStringWithinPolygons(line, tilePolygons))\n return false;\n }\n }\n return true;\n}\nclass Within {\n constructor(geojson, geometries) {\n this.type = BooleanType;\n this.geojson = geojson;\n this.geometries = geometries;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error(`'within' expression requires exactly one argument, but found ${args.length - 1} instead.`);\n if (isValue(args[1])) {\n const geojson = args[1];\n if (geojson.type === 'FeatureCollection') {\n for (let i = 0; i < geojson.features.length; ++i) {\n const type = geojson.features[i].geometry.type;\n if (type === 'Polygon' || type === 'MultiPolygon') {\n return new Within(geojson, geojson.features[i].geometry);\n }\n }\n }\n else if (geojson.type === 'Feature') {\n const type = geojson.geometry.type;\n if (type === 'Polygon' || type === 'MultiPolygon') {\n return new Within(geojson, geojson.geometry);\n }\n }\n else if (geojson.type === 'Polygon' || geojson.type === 'MultiPolygon') {\n return new Within(geojson, geojson);\n }\n }\n return context.error('\\'within\\' expression requires valid geojson object that contains polygon geometry type.');\n }\n evaluate(ctx) {\n if (ctx.geometry() != null && ctx.canonicalID() != null) {\n if (ctx.geometryType() === 'Point') {\n return pointsWithinPolygons(ctx, this.geometries);\n }\n else if (ctx.geometryType() === 'LineString') {\n return linesWithinPolygons(ctx, this.geometries);\n }\n }\n return false;\n }\n eachChild() { }\n outputDefined() {\n return true;\n }\n}\n\nclass Var {\n constructor(name, boundExpression) {\n this.type = boundExpression.type;\n this.name = name;\n this.boundExpression = boundExpression;\n }\n static parse(args, context) {\n if (args.length !== 2 || typeof args[1] !== 'string')\n return context.error('\\'var\\' expression requires exactly one string literal argument.');\n const name = args[1];\n if (!context.scope.has(name)) {\n return context.error(`Unknown variable \"${name}\". Make sure \"${name}\" has been bound in an enclosing \"let\" expression before using it.`, 1);\n }\n return new Var(name, context.scope.get(name));\n }\n evaluate(ctx) {\n return this.boundExpression.evaluate(ctx);\n }\n eachChild() { }\n outputDefined() {\n return false;\n }\n}\n\nclass CompoundExpression {\n constructor(name, type, evaluate, args) {\n this.name = name;\n this.type = type;\n this._evaluate = evaluate;\n this.args = args;\n }\n evaluate(ctx) {\n return this._evaluate(ctx, this.args);\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return false;\n }\n static parse(args, context) {\n const op = args[0];\n const definition = CompoundExpression.definitions[op];\n if (!definition) {\n return context.error(`Unknown expression \"${op}\". If you wanted a literal array, use [\"literal\", [...]].`, 0);\n }\n // Now check argument types against each signature\n const type = Array.isArray(definition) ?\n definition[0] : definition.type;\n const availableOverloads = Array.isArray(definition) ?\n [[definition[1], definition[2]]] :\n definition.overloads;\n const overloads = availableOverloads.filter(([signature]) => (!Array.isArray(signature) || // varags\n signature.length === args.length - 1 // correct param count\n ));\n let signatureContext = null;\n for (const [params, evaluate] of overloads) {\n // Use a fresh context for each attempted signature so that, if\n // we eventually succeed, we haven't polluted `context.errors`.\n signatureContext = new ParsingContext(context.registry, isExpressionConstant, context.path, null, context.scope);\n // First parse all the args, potentially coercing to the\n // types expected by this overload.\n const parsedArgs = [];\n let argParseFailed = false;\n for (let i = 1; i < args.length; i++) {\n const arg = args[i];\n const expectedType = Array.isArray(params) ?\n params[i - 1] :\n params.type;\n const parsed = signatureContext.parse(arg, 1 + parsedArgs.length, expectedType);\n if (!parsed) {\n argParseFailed = true;\n break;\n }\n parsedArgs.push(parsed);\n }\n if (argParseFailed) {\n // Couldn't coerce args of this overload to expected type, move\n // on to next one.\n continue;\n }\n if (Array.isArray(params)) {\n if (params.length !== parsedArgs.length) {\n signatureContext.error(`Expected ${params.length} arguments, but found ${parsedArgs.length} instead.`);\n continue;\n }\n }\n for (let i = 0; i < parsedArgs.length; i++) {\n const expected = Array.isArray(params) ? params[i] : params.type;\n const arg = parsedArgs[i];\n signatureContext.concat(i + 1).checkSubtype(expected, arg.type);\n }\n if (signatureContext.errors.length === 0) {\n return new CompoundExpression(op, type, evaluate, parsedArgs);\n }\n }\n if (overloads.length === 1) {\n context.errors.push(...signatureContext.errors);\n }\n else {\n const expected = overloads.length ? overloads : availableOverloads;\n const signatures = expected\n .map(([params]) => stringifySignature(params))\n .join(' | ');\n const actualTypes = [];\n // For error message, re-parse arguments without trying to\n // apply any coercions\n for (let i = 1; i < args.length; i++) {\n const parsed = context.parse(args[i], 1 + actualTypes.length);\n if (!parsed)\n return null;\n actualTypes.push(toString$1(parsed.type));\n }\n context.error(`Expected arguments of type ${signatures}, but found (${actualTypes.join(', ')}) instead.`);\n }\n return null;\n }\n static register(registry, definitions) {\n CompoundExpression.definitions = definitions;\n for (const name in definitions) {\n registry[name] = CompoundExpression;\n }\n }\n}\nfunction stringifySignature(signature) {\n if (Array.isArray(signature)) {\n return `(${signature.map(toString$1).join(', ')})`;\n }\n else {\n return `(${toString$1(signature.type)}...)`;\n }\n}\nfunction isExpressionConstant(expression) {\n if (expression instanceof Var) {\n return isExpressionConstant(expression.boundExpression);\n }\n else if (expression instanceof CompoundExpression && expression.name === 'error') {\n return false;\n }\n else if (expression instanceof CollatorExpression) {\n // Although the results of a Collator expression with fixed arguments\n // generally shouldn't change between executions, we can't serialize them\n // as constant expressions because results change based on environment.\n return false;\n }\n else if (expression instanceof Within) {\n return false;\n }\n const isTypeAnnotation = expression instanceof Coercion ||\n expression instanceof Assertion;\n let childrenConstant = true;\n expression.eachChild(child => {\n // We can _almost_ assume that if `expressions` children are constant,\n // they would already have been evaluated to Literal values when they\n // were parsed. Type annotations are the exception, because they might\n // have been inferred and added after a child was parsed.\n // So we recurse into isConstant() for the children of type annotations,\n // but otherwise simply check whether they are Literals.\n if (isTypeAnnotation) {\n childrenConstant = childrenConstant && isExpressionConstant(child);\n }\n else {\n childrenConstant = childrenConstant && child instanceof Literal;\n }\n });\n if (!childrenConstant) {\n return false;\n }\n return isFeatureConstant(expression) &&\n isGlobalPropertyConstant(expression, ['zoom', 'heatmap-density', 'line-progress', 'accumulated', 'is-supported-script']);\n}\nfunction isFeatureConstant(e) {\n if (e instanceof CompoundExpression) {\n if (e.name === 'get' && e.args.length === 1) {\n return false;\n }\n else if (e.name === 'feature-state') {\n return false;\n }\n else if (e.name === 'has' && e.args.length === 1) {\n return false;\n }\n else if (e.name === 'properties' ||\n e.name === 'geometry-type' ||\n e.name === 'id') {\n return false;\n }\n else if (/^filter-/.test(e.name)) {\n return false;\n }\n }\n if (e instanceof Within) {\n return false;\n }\n let result = true;\n e.eachChild(arg => {\n if (result && !isFeatureConstant(arg)) {\n result = false;\n }\n });\n return result;\n}\nfunction isStateConstant(e) {\n if (e instanceof CompoundExpression) {\n if (e.name === 'feature-state') {\n return false;\n }\n }\n let result = true;\n e.eachChild(arg => {\n if (result && !isStateConstant(arg)) {\n result = false;\n }\n });\n return result;\n}\nfunction isGlobalPropertyConstant(e, properties) {\n if (e instanceof CompoundExpression && properties.indexOf(e.name) >= 0) {\n return false;\n }\n let result = true;\n e.eachChild((arg) => {\n if (result && !isGlobalPropertyConstant(arg, properties)) {\n result = false;\n }\n });\n return result;\n}\n\n/**\n * Returns the index of the last stop <= input, or 0 if it doesn't exist.\n * @private\n */\nfunction findStopLessThanOrEqualTo(stops, input) {\n const lastIndex = stops.length - 1;\n let lowerIndex = 0;\n let upperIndex = lastIndex;\n let currentIndex = 0;\n let currentValue, nextValue;\n while (lowerIndex <= upperIndex) {\n currentIndex = Math.floor((lowerIndex + upperIndex) / 2);\n currentValue = stops[currentIndex];\n nextValue = stops[currentIndex + 1];\n if (currentValue <= input) {\n if (currentIndex === lastIndex || input < nextValue) { // Search complete\n return currentIndex;\n }\n lowerIndex = currentIndex + 1;\n }\n else if (currentValue > input) {\n upperIndex = currentIndex - 1;\n }\n else {\n throw new RuntimeError('Input is not a number.');\n }\n }\n return 0;\n}\n\nclass Step {\n constructor(type, input, stops) {\n this.type = type;\n this.input = input;\n this.labels = [];\n this.outputs = [];\n for (const [label, expression] of stops) {\n this.labels.push(label);\n this.outputs.push(expression);\n }\n }\n static parse(args, context) {\n if (args.length - 1 < 4) {\n return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`);\n }\n if ((args.length - 1) % 2 !== 0) {\n return context.error('Expected an even number of arguments.');\n }\n const input = context.parse(args[1], 1, NumberType);\n if (!input)\n return null;\n const stops = [];\n let outputType = null;\n if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n for (let i = 1; i < args.length; i += 2) {\n const label = i === 1 ? -Infinity : args[i];\n const value = args[i + 1];\n const labelKey = i;\n const valueKey = i + 1;\n if (typeof label !== 'number') {\n return context.error('Input/output pairs for \"step\" expressions must be defined using literal numeric values (not computed expressions) for the input values.', labelKey);\n }\n if (stops.length && stops[stops.length - 1][0] >= label) {\n return context.error('Input/output pairs for \"step\" expressions must be arranged with input values in strictly ascending order.', labelKey);\n }\n const parsed = context.parse(value, valueKey, outputType);\n if (!parsed)\n return null;\n outputType = outputType || parsed.type;\n stops.push([label, parsed]);\n }\n return new Step(outputType, input, stops);\n }\n evaluate(ctx) {\n const labels = this.labels;\n const outputs = this.outputs;\n if (labels.length === 1) {\n return outputs[0].evaluate(ctx);\n }\n const value = this.input.evaluate(ctx);\n if (value <= labels[0]) {\n return outputs[0].evaluate(ctx);\n }\n const stopCount = labels.length;\n if (value >= labels[stopCount - 1]) {\n return outputs[stopCount - 1].evaluate(ctx);\n }\n const index = findStopLessThanOrEqualTo(labels, value);\n return outputs[index].evaluate(ctx);\n }\n eachChild(fn) {\n fn(this.input);\n for (const expression of this.outputs) {\n fn(expression);\n }\n }\n outputDefined() {\n return this.outputs.every(out => out.outputDefined());\n }\n}\n\n/**\n * Checks whether the specified color space is one of the supported interpolation color spaces.\n *\n * @param colorSpace Color space key to verify.\n * @returns `true` if the specified color space is one of the supported\n * interpolation color spaces, `false` otherwise\n */\nfunction isSupportedInterpolationColorSpace(colorSpace) {\n return colorSpace === 'rgb' || colorSpace === 'hcl' || colorSpace === 'lab';\n}\n/**\n * @param interpolationType Interpolation type\n * @returns interpolation fn\n * @deprecated use `interpolate[type]` instead\n */\nconst interpolateFactory = (interpolationType) => {\n switch (interpolationType) {\n case 'number': return number;\n case 'color': return color;\n case 'array': return array;\n case 'padding': return padding;\n case 'variableAnchorOffsetCollection': return variableAnchorOffsetCollection;\n }\n};\nfunction number(from, to, t) {\n return from + t * (to - from);\n}\nfunction color(from, to, t, spaceKey = 'rgb') {\n switch (spaceKey) {\n case 'rgb': {\n const [r, g, b, alpha] = array(from.rgb, to.rgb, t);\n return new Color(r, g, b, alpha, false);\n }\n case 'hcl': {\n const [hue0, chroma0, light0, alphaF] = from.hcl;\n const [hue1, chroma1, light1, alphaT] = to.hcl;\n // https://github.com/gka/chroma.js/blob/cd1b3c0926c7a85cbdc3b1453b3a94006de91a92/src/interpolator/_hsx.js\n let hue, chroma;\n if (!isNaN(hue0) && !isNaN(hue1)) {\n let dh = hue1 - hue0;\n if (hue1 > hue0 && dh > 180) {\n dh -= 360;\n }\n else if (hue1 < hue0 && hue0 - hue1 > 180) {\n dh += 360;\n }\n hue = hue0 + t * dh;\n }\n else if (!isNaN(hue0)) {\n hue = hue0;\n if (light1 === 1 || light1 === 0)\n chroma = chroma0;\n }\n else if (!isNaN(hue1)) {\n hue = hue1;\n if (light0 === 1 || light0 === 0)\n chroma = chroma1;\n }\n else {\n hue = NaN;\n }\n const [r, g, b, alpha] = hclToRgb([\n hue,\n chroma !== null && chroma !== void 0 ? chroma : number(chroma0, chroma1, t),\n number(light0, light1, t),\n number(alphaF, alphaT, t),\n ]);\n return new Color(r, g, b, alpha, false);\n }\n case 'lab': {\n const [r, g, b, alpha] = labToRgb(array(from.lab, to.lab, t));\n return new Color(r, g, b, alpha, false);\n }\n }\n}\nfunction array(from, to, t) {\n return from.map((d, i) => {\n return number(d, to[i], t);\n });\n}\nfunction padding(from, to, t) {\n return new Padding(array(from.values, to.values, t));\n}\nfunction variableAnchorOffsetCollection(from, to, t) {\n const fromValues = from.values;\n const toValues = to.values;\n if (fromValues.length !== toValues.length) {\n throw new RuntimeError(`Cannot interpolate values of different length. from: ${from.toString()}, to: ${to.toString()}`);\n }\n const output = [];\n for (let i = 0; i < fromValues.length; i += 2) {\n // Anchor entries must match\n if (fromValues[i] !== toValues[i]) {\n throw new RuntimeError(`Cannot interpolate values containing mismatched anchors. from[${i}]: ${fromValues[i]}, to[${i}]: ${toValues[i]}`);\n }\n output.push(fromValues[i]);\n // Interpolate the offset values for each anchor\n const [fx, fy] = fromValues[i + 1];\n const [tx, ty] = toValues[i + 1];\n output.push([number(fx, tx, t), number(fy, ty, t)]);\n }\n return new VariableAnchorOffsetCollection(output);\n}\nconst interpolate = {\n number,\n color,\n array,\n padding,\n variableAnchorOffsetCollection\n};\n\nclass Interpolate {\n constructor(type, operator, interpolation, input, stops) {\n this.type = type;\n this.operator = operator;\n this.interpolation = interpolation;\n this.input = input;\n this.labels = [];\n this.outputs = [];\n for (const [label, expression] of stops) {\n this.labels.push(label);\n this.outputs.push(expression);\n }\n }\n static interpolationFactor(interpolation, input, lower, upper) {\n let t = 0;\n if (interpolation.name === 'exponential') {\n t = exponentialInterpolation(input, interpolation.base, lower, upper);\n }\n else if (interpolation.name === 'linear') {\n t = exponentialInterpolation(input, 1, lower, upper);\n }\n else if (interpolation.name === 'cubic-bezier') {\n const c = interpolation.controlPoints;\n const ub = new UnitBezier(c[0], c[1], c[2], c[3]);\n t = ub.solve(exponentialInterpolation(input, 1, lower, upper));\n }\n return t;\n }\n static parse(args, context) {\n let [operator, interpolation, input, ...rest] = args;\n if (!Array.isArray(interpolation) || interpolation.length === 0) {\n return context.error('Expected an interpolation type expression.', 1);\n }\n if (interpolation[0] === 'linear') {\n interpolation = { name: 'linear' };\n }\n else if (interpolation[0] === 'exponential') {\n const base = interpolation[1];\n if (typeof base !== 'number')\n return context.error('Exponential interpolation requires a numeric base.', 1, 1);\n interpolation = {\n name: 'exponential',\n base\n };\n }\n else if (interpolation[0] === 'cubic-bezier') {\n const controlPoints = interpolation.slice(1);\n if (controlPoints.length !== 4 ||\n controlPoints.some(t => typeof t !== 'number' || t < 0 || t > 1)) {\n return context.error('Cubic bezier interpolation requires four numeric arguments with values between 0 and 1.', 1);\n }\n interpolation = {\n name: 'cubic-bezier',\n controlPoints: controlPoints\n };\n }\n else {\n return context.error(`Unknown interpolation type ${String(interpolation[0])}`, 1, 0);\n }\n if (args.length - 1 < 4) {\n return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`);\n }\n if ((args.length - 1) % 2 !== 0) {\n return context.error('Expected an even number of arguments.');\n }\n input = context.parse(input, 2, NumberType);\n if (!input)\n return null;\n const stops = [];\n let outputType = null;\n if (operator === 'interpolate-hcl' || operator === 'interpolate-lab') {\n outputType = ColorType;\n }\n else if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n for (let i = 0; i < rest.length; i += 2) {\n const label = rest[i];\n const value = rest[i + 1];\n const labelKey = i + 3;\n const valueKey = i + 4;\n if (typeof label !== 'number') {\n return context.error('Input/output pairs for \"interpolate\" expressions must be defined using literal numeric values (not computed expressions) for the input values.', labelKey);\n }\n if (stops.length && stops[stops.length - 1][0] >= label) {\n return context.error('Input/output pairs for \"interpolate\" expressions must be arranged with input values in strictly ascending order.', labelKey);\n }\n const parsed = context.parse(value, valueKey, outputType);\n if (!parsed)\n return null;\n outputType = outputType || parsed.type;\n stops.push([label, parsed]);\n }\n if (!verifyType(outputType, NumberType) &&\n !verifyType(outputType, ColorType) &&\n !verifyType(outputType, PaddingType) &&\n !verifyType(outputType, VariableAnchorOffsetCollectionType) &&\n !verifyType(outputType, array$1(NumberType))) {\n return context.error(`Type ${toString$1(outputType)} is not interpolatable.`);\n }\n return new Interpolate(outputType, operator, interpolation, input, stops);\n }\n evaluate(ctx) {\n const labels = this.labels;\n const outputs = this.outputs;\n if (labels.length === 1) {\n return outputs[0].evaluate(ctx);\n }\n const value = this.input.evaluate(ctx);\n if (value <= labels[0]) {\n return outputs[0].evaluate(ctx);\n }\n const stopCount = labels.length;\n if (value >= labels[stopCount - 1]) {\n return outputs[stopCount - 1].evaluate(ctx);\n }\n const index = findStopLessThanOrEqualTo(labels, value);\n const lower = labels[index];\n const upper = labels[index + 1];\n const t = Interpolate.interpolationFactor(this.interpolation, value, lower, upper);\n const outputLower = outputs[index].evaluate(ctx);\n const outputUpper = outputs[index + 1].evaluate(ctx);\n switch (this.operator) {\n case 'interpolate':\n return interpolate[this.type.kind](outputLower, outputUpper, t);\n case 'interpolate-hcl':\n return interpolate.color(outputLower, outputUpper, t, 'hcl');\n case 'interpolate-lab':\n return interpolate.color(outputLower, outputUpper, t, 'lab');\n }\n }\n eachChild(fn) {\n fn(this.input);\n for (const expression of this.outputs) {\n fn(expression);\n }\n }\n outputDefined() {\n return this.outputs.every(out => out.outputDefined());\n }\n}\n/**\n * Returns a ratio that can be used to interpolate between exponential function\n * stops.\n * How it works: Two consecutive stop values define a (scaled and shifted) exponential function `f(x) = a * base^x + b`, where `base` is the user-specified base,\n * and `a` and `b` are constants affording sufficient degrees of freedom to fit\n * the function to the given stops.\n *\n * Here's a bit of algebra that lets us compute `f(x)` directly from the stop\n * values without explicitly solving for `a` and `b`:\n *\n * First stop value: `f(x0) = y0 = a * base^x0 + b`\n * Second stop value: `f(x1) = y1 = a * base^x1 + b`\n * => `y1 - y0 = a(base^x1 - base^x0)`\n * => `a = (y1 - y0)/(base^x1 - base^x0)`\n *\n * Desired value: `f(x) = y = a * base^x + b`\n * => `f(x) = y0 + a * (base^x - base^x0)`\n *\n * From the above, we can replace the `a` in `a * (base^x - base^x0)` and do a\n * little algebra:\n * ```\n * a * (base^x - base^x0) = (y1 - y0)/(base^x1 - base^x0) * (base^x - base^x0)\n * = (y1 - y0) * (base^x - base^x0) / (base^x1 - base^x0)\n * ```\n *\n * If we let `(base^x - base^x0) / (base^x1 base^x0)`, then we have\n * `f(x) = y0 + (y1 - y0) * ratio`. In other words, `ratio` may be treated as\n * an interpolation factor between the two stops' output values.\n *\n * (Note: a slightly different form for `ratio`,\n * `(base^(x-x0) - 1) / (base^(x1-x0) - 1) `, is equivalent, but requires fewer\n * expensive `Math.pow()` operations.)\n *\n * @private\n*/\nfunction exponentialInterpolation(input, base, lowerValue, upperValue) {\n const difference = upperValue - lowerValue;\n const progress = input - lowerValue;\n if (difference === 0) {\n return 0;\n }\n else if (base === 1) {\n return progress / difference;\n }\n else {\n return (Math.pow(base, progress) - 1) / (Math.pow(base, difference) - 1);\n }\n}\n\nclass Coalesce {\n constructor(type, args) {\n this.type = type;\n this.args = args;\n }\n static parse(args, context) {\n if (args.length < 2) {\n return context.error('Expectected at least one argument.');\n }\n let outputType = null;\n const expectedType = context.expectedType;\n if (expectedType && expectedType.kind !== 'value') {\n outputType = expectedType;\n }\n const parsedArgs = [];\n for (const arg of args.slice(1)) {\n const parsed = context.parse(arg, 1 + parsedArgs.length, outputType, undefined, { typeAnnotation: 'omit' });\n if (!parsed)\n return null;\n outputType = outputType || parsed.type;\n parsedArgs.push(parsed);\n }\n if (!outputType)\n throw new Error('No output type');\n // Above, we parse arguments without inferred type annotation so that\n // they don't produce a runtime error for `null` input, which would\n // preempt the desired null-coalescing behavior.\n // Thus, if any of our arguments would have needed an annotation, we\n // need to wrap the enclosing coalesce expression with it instead.\n const needsAnnotation = expectedType &&\n parsedArgs.some(arg => checkSubtype(expectedType, arg.type));\n return needsAnnotation ?\n new Coalesce(ValueType, parsedArgs) :\n new Coalesce(outputType, parsedArgs);\n }\n evaluate(ctx) {\n let result = null;\n let argCount = 0;\n let requestedImageName;\n for (const arg of this.args) {\n argCount++;\n result = arg.evaluate(ctx);\n // we need to keep track of the first requested image in a coalesce statement\n // if coalesce can't find a valid image, we return the first image name so styleimagemissing can fire\n if (result && result instanceof ResolvedImage && !result.available) {\n if (!requestedImageName) {\n requestedImageName = result.name;\n }\n result = null;\n if (argCount === this.args.length) {\n result = requestedImageName;\n }\n }\n if (result !== null)\n break;\n }\n return result;\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return this.args.every(arg => arg.outputDefined());\n }\n}\n\nclass Let {\n constructor(bindings, result) {\n this.type = result.type;\n this.bindings = [].concat(bindings);\n this.result = result;\n }\n evaluate(ctx) {\n return this.result.evaluate(ctx);\n }\n eachChild(fn) {\n for (const binding of this.bindings) {\n fn(binding[1]);\n }\n fn(this.result);\n }\n static parse(args, context) {\n if (args.length < 4)\n return context.error(`Expected at least 3 arguments, but found ${args.length - 1} instead.`);\n const bindings = [];\n for (let i = 1; i < args.length - 1; i += 2) {\n const name = args[i];\n if (typeof name !== 'string') {\n return context.error(`Expected string, but found ${typeof name} instead.`, i);\n }\n if (/[^a-zA-Z0-9_]/.test(name)) {\n return context.error('Variable names must contain only alphanumeric characters or \\'_\\'.', i);\n }\n const value = context.parse(args[i + 1], i + 1);\n if (!value)\n return null;\n bindings.push([name, value]);\n }\n const result = context.parse(args[args.length - 1], args.length - 1, context.expectedType, bindings);\n if (!result)\n return null;\n return new Let(bindings, result);\n }\n outputDefined() {\n return this.result.outputDefined();\n }\n}\n\nclass At {\n constructor(type, index, input) {\n this.type = type;\n this.index = index;\n this.input = input;\n }\n static parse(args, context) {\n if (args.length !== 3)\n return context.error(`Expected 2 arguments, but found ${args.length - 1} instead.`);\n const index = context.parse(args[1], 1, NumberType);\n const input = context.parse(args[2], 2, array$1(context.expectedType || ValueType));\n if (!index || !input)\n return null;\n const t = input.type;\n return new At(t.itemType, index, input);\n }\n evaluate(ctx) {\n const index = this.index.evaluate(ctx);\n const array = this.input.evaluate(ctx);\n if (index < 0) {\n throw new RuntimeError(`Array index out of bounds: ${index} < 0.`);\n }\n if (index >= array.length) {\n throw new RuntimeError(`Array index out of bounds: ${index} > ${array.length - 1}.`);\n }\n if (index !== Math.floor(index)) {\n throw new RuntimeError(`Array index must be an integer, but found ${index} instead.`);\n }\n return array[index];\n }\n eachChild(fn) {\n fn(this.index);\n fn(this.input);\n }\n outputDefined() {\n return false;\n }\n}\n\nclass In {\n constructor(needle, haystack) {\n this.type = BooleanType;\n this.needle = needle;\n this.haystack = haystack;\n }\n static parse(args, context) {\n if (args.length !== 3) {\n return context.error(`Expected 2 arguments, but found ${args.length - 1} instead.`);\n }\n const needle = context.parse(args[1], 1, ValueType);\n const haystack = context.parse(args[2], 2, ValueType);\n if (!needle || !haystack)\n return null;\n if (!isValidType(needle.type, [BooleanType, StringType, NumberType, NullType, ValueType])) {\n return context.error(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(needle.type)} instead`);\n }\n return new In(needle, haystack);\n }\n evaluate(ctx) {\n const needle = this.needle.evaluate(ctx);\n const haystack = this.haystack.evaluate(ctx);\n if (!haystack)\n return false;\n if (!isValidNativeType(needle, ['boolean', 'string', 'number', 'null'])) {\n throw new RuntimeError(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(typeOf(needle))} instead.`);\n }\n if (!isValidNativeType(haystack, ['string', 'array'])) {\n throw new RuntimeError(`Expected second argument to be of type array or string, but found ${toString$1(typeOf(haystack))} instead.`);\n }\n return haystack.indexOf(needle) >= 0;\n }\n eachChild(fn) {\n fn(this.needle);\n fn(this.haystack);\n }\n outputDefined() {\n return true;\n }\n}\n\nclass IndexOf {\n constructor(needle, haystack, fromIndex) {\n this.type = NumberType;\n this.needle = needle;\n this.haystack = haystack;\n this.fromIndex = fromIndex;\n }\n static parse(args, context) {\n if (args.length <= 2 || args.length >= 5) {\n return context.error(`Expected 3 or 4 arguments, but found ${args.length - 1} instead.`);\n }\n const needle = context.parse(args[1], 1, ValueType);\n const haystack = context.parse(args[2], 2, ValueType);\n if (!needle || !haystack)\n return null;\n if (!isValidType(needle.type, [BooleanType, StringType, NumberType, NullType, ValueType])) {\n return context.error(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(needle.type)} instead`);\n }\n if (args.length === 4) {\n const fromIndex = context.parse(args[3], 3, NumberType);\n if (!fromIndex)\n return null;\n return new IndexOf(needle, haystack, fromIndex);\n }\n else {\n return new IndexOf(needle, haystack);\n }\n }\n evaluate(ctx) {\n const needle = this.needle.evaluate(ctx);\n const haystack = this.haystack.evaluate(ctx);\n if (!isValidNativeType(needle, ['boolean', 'string', 'number', 'null'])) {\n throw new RuntimeError(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(typeOf(needle))} instead.`);\n }\n if (!isValidNativeType(haystack, ['string', 'array'])) {\n throw new RuntimeError(`Expected second argument to be of type array or string, but found ${toString$1(typeOf(haystack))} instead.`);\n }\n if (this.fromIndex) {\n const fromIndex = this.fromIndex.evaluate(ctx);\n return haystack.indexOf(needle, fromIndex);\n }\n return haystack.indexOf(needle);\n }\n eachChild(fn) {\n fn(this.needle);\n fn(this.haystack);\n if (this.fromIndex) {\n fn(this.fromIndex);\n }\n }\n outputDefined() {\n return false;\n }\n}\n\nclass Match {\n constructor(inputType, outputType, input, cases, outputs, otherwise) {\n this.inputType = inputType;\n this.type = outputType;\n this.input = input;\n this.cases = cases;\n this.outputs = outputs;\n this.otherwise = otherwise;\n }\n static parse(args, context) {\n if (args.length < 5)\n return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`);\n if (args.length % 2 !== 1)\n return context.error('Expected an even number of arguments.');\n let inputType;\n let outputType;\n if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n const cases = {};\n const outputs = [];\n for (let i = 2; i < args.length - 1; i += 2) {\n let labels = args[i];\n const value = args[i + 1];\n if (!Array.isArray(labels)) {\n labels = [labels];\n }\n const labelContext = context.concat(i);\n if (labels.length === 0) {\n return labelContext.error('Expected at least one branch label.');\n }\n for (const label of labels) {\n if (typeof label !== 'number' && typeof label !== 'string') {\n return labelContext.error('Branch labels must be numbers or strings.');\n }\n else if (typeof label === 'number' && Math.abs(label) > Number.MAX_SAFE_INTEGER) {\n return labelContext.error(`Branch labels must be integers no larger than ${Number.MAX_SAFE_INTEGER}.`);\n }\n else if (typeof label === 'number' && Math.floor(label) !== label) {\n return labelContext.error('Numeric branch labels must be integer values.');\n }\n else if (!inputType) {\n inputType = typeOf(label);\n }\n else if (labelContext.checkSubtype(inputType, typeOf(label))) {\n return null;\n }\n if (typeof cases[String(label)] !== 'undefined') {\n return labelContext.error('Branch labels must be unique.');\n }\n cases[String(label)] = outputs.length;\n }\n const result = context.parse(value, i, outputType);\n if (!result)\n return null;\n outputType = outputType || result.type;\n outputs.push(result);\n }\n const input = context.parse(args[1], 1, ValueType);\n if (!input)\n return null;\n const otherwise = context.parse(args[args.length - 1], args.length - 1, outputType);\n if (!otherwise)\n return null;\n if (input.type.kind !== 'value' && context.concat(1).checkSubtype(inputType, input.type)) {\n return null;\n }\n return new Match(inputType, outputType, input, cases, outputs, otherwise);\n }\n evaluate(ctx) {\n const input = this.input.evaluate(ctx);\n const output = (typeOf(input) === this.inputType && this.outputs[this.cases[input]]) || this.otherwise;\n return output.evaluate(ctx);\n }\n eachChild(fn) {\n fn(this.input);\n this.outputs.forEach(fn);\n fn(this.otherwise);\n }\n outputDefined() {\n return this.outputs.every(out => out.outputDefined()) && this.otherwise.outputDefined();\n }\n}\n\nclass Case {\n constructor(type, branches, otherwise) {\n this.type = type;\n this.branches = branches;\n this.otherwise = otherwise;\n }\n static parse(args, context) {\n if (args.length < 4)\n return context.error(`Expected at least 3 arguments, but found only ${args.length - 1}.`);\n if (args.length % 2 !== 0)\n return context.error('Expected an odd number of arguments.');\n let outputType;\n if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n const branches = [];\n for (let i = 1; i < args.length - 1; i += 2) {\n const test = context.parse(args[i], i, BooleanType);\n if (!test)\n return null;\n const result = context.parse(args[i + 1], i + 1, outputType);\n if (!result)\n return null;\n branches.push([test, result]);\n outputType = outputType || result.type;\n }\n const otherwise = context.parse(args[args.length - 1], args.length - 1, outputType);\n if (!otherwise)\n return null;\n if (!outputType)\n throw new Error('Can\\'t infer output type');\n return new Case(outputType, branches, otherwise);\n }\n evaluate(ctx) {\n for (const [test, expression] of this.branches) {\n if (test.evaluate(ctx)) {\n return expression.evaluate(ctx);\n }\n }\n return this.otherwise.evaluate(ctx);\n }\n eachChild(fn) {\n for (const [test, expression] of this.branches) {\n fn(test);\n fn(expression);\n }\n fn(this.otherwise);\n }\n outputDefined() {\n return this.branches.every(([_, out]) => out.outputDefined()) && this.otherwise.outputDefined();\n }\n}\n\nclass Slice {\n constructor(type, input, beginIndex, endIndex) {\n this.type = type;\n this.input = input;\n this.beginIndex = beginIndex;\n this.endIndex = endIndex;\n }\n static parse(args, context) {\n if (args.length <= 2 || args.length >= 5) {\n return context.error(`Expected 3 or 4 arguments, but found ${args.length - 1} instead.`);\n }\n const input = context.parse(args[1], 1, ValueType);\n const beginIndex = context.parse(args[2], 2, NumberType);\n if (!input || !beginIndex)\n return null;\n if (!isValidType(input.type, [array$1(ValueType), StringType, ValueType])) {\n return context.error(`Expected first argument to be of type array or string, but found ${toString$1(input.type)} instead`);\n }\n if (args.length === 4) {\n const endIndex = context.parse(args[3], 3, NumberType);\n if (!endIndex)\n return null;\n return new Slice(input.type, input, beginIndex, endIndex);\n }\n else {\n return new Slice(input.type, input, beginIndex);\n }\n }\n evaluate(ctx) {\n const input = this.input.evaluate(ctx);\n const beginIndex = this.beginIndex.evaluate(ctx);\n if (!isValidNativeType(input, ['string', 'array'])) {\n throw new RuntimeError(`Expected first argument to be of type array or string, but found ${toString$1(typeOf(input))} instead.`);\n }\n if (this.endIndex) {\n const endIndex = this.endIndex.evaluate(ctx);\n return input.slice(beginIndex, endIndex);\n }\n return input.slice(beginIndex);\n }\n eachChild(fn) {\n fn(this.input);\n fn(this.beginIndex);\n if (this.endIndex) {\n fn(this.endIndex);\n }\n }\n outputDefined() {\n return false;\n }\n}\n\nfunction isComparableType(op, type) {\n if (op === '==' || op === '!=') {\n // equality operator\n return type.kind === 'boolean' ||\n type.kind === 'string' ||\n type.kind === 'number' ||\n type.kind === 'null' ||\n type.kind === 'value';\n }\n else {\n // ordering operator\n return type.kind === 'string' ||\n type.kind === 'number' ||\n type.kind === 'value';\n }\n}\nfunction eq(ctx, a, b) { return a === b; }\nfunction neq(ctx, a, b) { return a !== b; }\nfunction lt(ctx, a, b) { return a < b; }\nfunction gt(ctx, a, b) { return a > b; }\nfunction lteq(ctx, a, b) { return a <= b; }\nfunction gteq(ctx, a, b) { return a >= b; }\nfunction eqCollate(ctx, a, b, c) { return c.compare(a, b) === 0; }\nfunction neqCollate(ctx, a, b, c) { return !eqCollate(ctx, a, b, c); }\nfunction ltCollate(ctx, a, b, c) { return c.compare(a, b) < 0; }\nfunction gtCollate(ctx, a, b, c) { return c.compare(a, b) > 0; }\nfunction lteqCollate(ctx, a, b, c) { return c.compare(a, b) <= 0; }\nfunction gteqCollate(ctx, a, b, c) { return c.compare(a, b) >= 0; }\n/**\n * Special form for comparison operators, implementing the signatures:\n * - (T, T, ?Collator) => boolean\n * - (T, value, ?Collator) => boolean\n * - (value, T, ?Collator) => boolean\n *\n * For inequalities, T must be either value, string, or number. For ==/!=, it\n * can also be boolean or null.\n *\n * Equality semantics are equivalent to Javascript's strict equality (===/!==)\n * -- i.e., when the arguments' types don't match, == evaluates to false, != to\n * true.\n *\n * When types don't match in an ordering comparison, a runtime error is thrown.\n *\n * @private\n */\nfunction makeComparison(op, compareBasic, compareWithCollator) {\n const isOrderComparison = op !== '==' && op !== '!=';\n return class Comparison {\n constructor(lhs, rhs, collator) {\n this.type = BooleanType;\n this.lhs = lhs;\n this.rhs = rhs;\n this.collator = collator;\n this.hasUntypedArgument = lhs.type.kind === 'value' || rhs.type.kind === 'value';\n }\n static parse(args, context) {\n if (args.length !== 3 && args.length !== 4)\n return context.error('Expected two or three arguments.');\n const op = args[0];\n let lhs = context.parse(args[1], 1, ValueType);\n if (!lhs)\n return null;\n if (!isComparableType(op, lhs.type)) {\n return context.concat(1).error(`\"${op}\" comparisons are not supported for type '${toString$1(lhs.type)}'.`);\n }\n let rhs = context.parse(args[2], 2, ValueType);\n if (!rhs)\n return null;\n if (!isComparableType(op, rhs.type)) {\n return context.concat(2).error(`\"${op}\" comparisons are not supported for type '${toString$1(rhs.type)}'.`);\n }\n if (lhs.type.kind !== rhs.type.kind &&\n lhs.type.kind !== 'value' &&\n rhs.type.kind !== 'value') {\n return context.error(`Cannot compare types '${toString$1(lhs.type)}' and '${toString$1(rhs.type)}'.`);\n }\n if (isOrderComparison) {\n // typing rules specific to less/greater than operators\n if (lhs.type.kind === 'value' && rhs.type.kind !== 'value') {\n // (value, T)\n lhs = new Assertion(rhs.type, [lhs]);\n }\n else if (lhs.type.kind !== 'value' && rhs.type.kind === 'value') {\n // (T, value)\n rhs = new Assertion(lhs.type, [rhs]);\n }\n }\n let collator = null;\n if (args.length === 4) {\n if (lhs.type.kind !== 'string' &&\n rhs.type.kind !== 'string' &&\n lhs.type.kind !== 'value' &&\n rhs.type.kind !== 'value') {\n return context.error('Cannot use collator to compare non-string types.');\n }\n collator = context.parse(args[3], 3, CollatorType);\n if (!collator)\n return null;\n }\n return new Comparison(lhs, rhs, collator);\n }\n evaluate(ctx) {\n const lhs = this.lhs.evaluate(ctx);\n const rhs = this.rhs.evaluate(ctx);\n if (isOrderComparison && this.hasUntypedArgument) {\n const lt = typeOf(lhs);\n const rt = typeOf(rhs);\n // check that type is string or number, and equal\n if (lt.kind !== rt.kind || !(lt.kind === 'string' || lt.kind === 'number')) {\n throw new RuntimeError(`Expected arguments for \"${op}\" to be (string, string) or (number, number), but found (${lt.kind}, ${rt.kind}) instead.`);\n }\n }\n if (this.collator && !isOrderComparison && this.hasUntypedArgument) {\n const lt = typeOf(lhs);\n const rt = typeOf(rhs);\n if (lt.kind !== 'string' || rt.kind !== 'string') {\n return compareBasic(ctx, lhs, rhs);\n }\n }\n return this.collator ?\n compareWithCollator(ctx, lhs, rhs, this.collator.evaluate(ctx)) :\n compareBasic(ctx, lhs, rhs);\n }\n eachChild(fn) {\n fn(this.lhs);\n fn(this.rhs);\n if (this.collator) {\n fn(this.collator);\n }\n }\n outputDefined() {\n return true;\n }\n };\n}\nconst Equals = makeComparison('==', eq, eqCollate);\nconst NotEquals = makeComparison('!=', neq, neqCollate);\nconst LessThan = makeComparison('<', lt, ltCollate);\nconst GreaterThan = makeComparison('>', gt, gtCollate);\nconst LessThanOrEqual = makeComparison('<=', lteq, lteqCollate);\nconst GreaterThanOrEqual = makeComparison('>=', gteq, gteqCollate);\n\nclass NumberFormat {\n constructor(number, locale, currency, minFractionDigits, maxFractionDigits) {\n this.type = StringType;\n this.number = number;\n this.locale = locale;\n this.currency = currency;\n this.minFractionDigits = minFractionDigits;\n this.maxFractionDigits = maxFractionDigits;\n }\n static parse(args, context) {\n if (args.length !== 3)\n return context.error('Expected two arguments.');\n const number = context.parse(args[1], 1, NumberType);\n if (!number)\n return null;\n const options = args[2];\n if (typeof options !== 'object' || Array.isArray(options))\n return context.error('NumberFormat options argument must be an object.');\n let locale = null;\n if (options['locale']) {\n locale = context.parse(options['locale'], 1, StringType);\n if (!locale)\n return null;\n }\n let currency = null;\n if (options['currency']) {\n currency = context.parse(options['currency'], 1, StringType);\n if (!currency)\n return null;\n }\n let minFractionDigits = null;\n if (options['min-fraction-digits']) {\n minFractionDigits = context.parse(options['min-fraction-digits'], 1, NumberType);\n if (!minFractionDigits)\n return null;\n }\n let maxFractionDigits = null;\n if (options['max-fraction-digits']) {\n maxFractionDigits = context.parse(options['max-fraction-digits'], 1, NumberType);\n if (!maxFractionDigits)\n return null;\n }\n return new NumberFormat(number, locale, currency, minFractionDigits, maxFractionDigits);\n }\n evaluate(ctx) {\n return new Intl.NumberFormat(this.locale ? this.locale.evaluate(ctx) : [], {\n style: this.currency ? 'currency' : 'decimal',\n currency: this.currency ? this.currency.evaluate(ctx) : undefined,\n minimumFractionDigits: this.minFractionDigits ? this.minFractionDigits.evaluate(ctx) : undefined,\n maximumFractionDigits: this.maxFractionDigits ? this.maxFractionDigits.evaluate(ctx) : undefined,\n }).format(this.number.evaluate(ctx));\n }\n eachChild(fn) {\n fn(this.number);\n if (this.locale) {\n fn(this.locale);\n }\n if (this.currency) {\n fn(this.currency);\n }\n if (this.minFractionDigits) {\n fn(this.minFractionDigits);\n }\n if (this.maxFractionDigits) {\n fn(this.maxFractionDigits);\n }\n }\n outputDefined() {\n return false;\n }\n}\n\nclass FormatExpression {\n constructor(sections) {\n this.type = FormattedType;\n this.sections = sections;\n }\n static parse(args, context) {\n if (args.length < 2) {\n return context.error('Expected at least one argument.');\n }\n const firstArg = args[1];\n if (!Array.isArray(firstArg) && typeof firstArg === 'object') {\n return context.error('First argument must be an image or text section.');\n }\n const sections = [];\n let nextTokenMayBeObject = false;\n for (let i = 1; i <= args.length - 1; ++i) {\n const arg = args[i];\n if (nextTokenMayBeObject && typeof arg === 'object' && !Array.isArray(arg)) {\n nextTokenMayBeObject = false;\n let scale = null;\n if (arg['font-scale']) {\n scale = context.parse(arg['font-scale'], 1, NumberType);\n if (!scale)\n return null;\n }\n let font = null;\n if (arg['text-font']) {\n font = context.parse(arg['text-font'], 1, array$1(StringType));\n if (!font)\n return null;\n }\n let textColor = null;\n if (arg['text-color']) {\n textColor = context.parse(arg['text-color'], 1, ColorType);\n if (!textColor)\n return null;\n }\n const lastExpression = sections[sections.length - 1];\n lastExpression.scale = scale;\n lastExpression.font = font;\n lastExpression.textColor = textColor;\n }\n else {\n const content = context.parse(args[i], 1, ValueType);\n if (!content)\n return null;\n const kind = content.type.kind;\n if (kind !== 'string' && kind !== 'value' && kind !== 'null' && kind !== 'resolvedImage')\n return context.error('Formatted text type must be \\'string\\', \\'value\\', \\'image\\' or \\'null\\'.');\n nextTokenMayBeObject = true;\n sections.push({ content, scale: null, font: null, textColor: null });\n }\n }\n return new FormatExpression(sections);\n }\n evaluate(ctx) {\n const evaluateSection = section => {\n const evaluatedContent = section.content.evaluate(ctx);\n if (typeOf(evaluatedContent) === ResolvedImageType) {\n return new FormattedSection('', evaluatedContent, null, null, null);\n }\n return new FormattedSection(toString(evaluatedContent), null, section.scale ? section.scale.evaluate(ctx) : null, section.font ? section.font.evaluate(ctx).join(',') : null, section.textColor ? section.textColor.evaluate(ctx) : null);\n };\n return new Formatted(this.sections.map(evaluateSection));\n }\n eachChild(fn) {\n for (const section of this.sections) {\n fn(section.content);\n if (section.scale) {\n fn(section.scale);\n }\n if (section.font) {\n fn(section.font);\n }\n if (section.textColor) {\n fn(section.textColor);\n }\n }\n }\n outputDefined() {\n // Technically the combinatoric set of all children\n // Usually, this.text will be undefined anyway\n return false;\n }\n}\n\nclass ImageExpression {\n constructor(input) {\n this.type = ResolvedImageType;\n this.input = input;\n }\n static parse(args, context) {\n if (args.length !== 2) {\n return context.error('Expected two arguments.');\n }\n const name = context.parse(args[1], 1, StringType);\n if (!name)\n return context.error('No image name provided.');\n return new ImageExpression(name);\n }\n evaluate(ctx) {\n const evaluatedImageName = this.input.evaluate(ctx);\n const value = ResolvedImage.fromString(evaluatedImageName);\n if (value && ctx.availableImages)\n value.available = ctx.availableImages.indexOf(evaluatedImageName) > -1;\n return value;\n }\n eachChild(fn) {\n fn(this.input);\n }\n outputDefined() {\n // The output of image is determined by the list of available images in the evaluation context\n return false;\n }\n}\n\nclass Length {\n constructor(input) {\n this.type = NumberType;\n this.input = input;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error(`Expected 1 argument, but found ${args.length - 1} instead.`);\n const input = context.parse(args[1], 1);\n if (!input)\n return null;\n if (input.type.kind !== 'array' && input.type.kind !== 'string' && input.type.kind !== 'value')\n return context.error(`Expected argument of type string or array, but found ${toString$1(input.type)} instead.`);\n return new Length(input);\n }\n evaluate(ctx) {\n const input = this.input.evaluate(ctx);\n if (typeof input === 'string') {\n return input.length;\n }\n else if (Array.isArray(input)) {\n return input.length;\n }\n else {\n throw new RuntimeError(`Expected value to be of type string or array, but found ${toString$1(typeOf(input))} instead.`);\n }\n }\n eachChild(fn) {\n fn(this.input);\n }\n outputDefined() {\n return false;\n }\n}\n\nconst expressions = {\n // special forms\n '==': Equals,\n '!=': NotEquals,\n '>': GreaterThan,\n '<': LessThan,\n '>=': GreaterThanOrEqual,\n '<=': LessThanOrEqual,\n 'array': Assertion,\n 'at': At,\n 'boolean': Assertion,\n 'case': Case,\n 'coalesce': Coalesce,\n 'collator': CollatorExpression,\n 'format': FormatExpression,\n 'image': ImageExpression,\n 'in': In,\n 'index-of': IndexOf,\n 'interpolate': Interpolate,\n 'interpolate-hcl': Interpolate,\n 'interpolate-lab': Interpolate,\n 'length': Length,\n 'let': Let,\n 'literal': Literal,\n 'match': Match,\n 'number': Assertion,\n 'number-format': NumberFormat,\n 'object': Assertion,\n 'slice': Slice,\n 'step': Step,\n 'string': Assertion,\n 'to-boolean': Coercion,\n 'to-color': Coercion,\n 'to-number': Coercion,\n 'to-string': Coercion,\n 'var': Var,\n 'within': Within\n};\nfunction rgba(ctx, [r, g, b, a]) {\n r = r.evaluate(ctx);\n g = g.evaluate(ctx);\n b = b.evaluate(ctx);\n const alpha = a ? a.evaluate(ctx) : 1;\n const error = validateRGBA(r, g, b, alpha);\n if (error)\n throw new RuntimeError(error);\n return new Color(r / 255, g / 255, b / 255, alpha, false);\n}\nfunction has(key, obj) {\n return key in obj;\n}\nfunction get(key, obj) {\n const v = obj[key];\n return typeof v === 'undefined' ? null : v;\n}\nfunction binarySearch(v, a, i, j) {\n while (i <= j) {\n const m = (i + j) >> 1;\n if (a[m] === v)\n return true;\n if (a[m] > v)\n j = m - 1;\n else\n i = m + 1;\n }\n return false;\n}\nfunction varargs(type) {\n return { type };\n}\nCompoundExpression.register(expressions, {\n 'error': [\n ErrorType,\n [StringType],\n (ctx, [v]) => { throw new RuntimeError(v.evaluate(ctx)); }\n ],\n 'typeof': [\n StringType,\n [ValueType],\n (ctx, [v]) => toString$1(typeOf(v.evaluate(ctx)))\n ],\n 'to-rgba': [\n array$1(NumberType, 4),\n [ColorType],\n (ctx, [v]) => {\n const [r, g, b, a] = v.evaluate(ctx).rgb;\n return [r * 255, g * 255, b * 255, a];\n },\n ],\n 'rgb': [\n ColorType,\n [NumberType, NumberType, NumberType],\n rgba\n ],\n 'rgba': [\n ColorType,\n [NumberType, NumberType, NumberType, NumberType],\n rgba\n ],\n 'has': {\n type: BooleanType,\n overloads: [\n [\n [StringType],\n (ctx, [key]) => has(key.evaluate(ctx), ctx.properties())\n ], [\n [StringType, ObjectType],\n (ctx, [key, obj]) => has(key.evaluate(ctx), obj.evaluate(ctx))\n ]\n ]\n },\n 'get': {\n type: ValueType,\n overloads: [\n [\n [StringType],\n (ctx, [key]) => get(key.evaluate(ctx), ctx.properties())\n ], [\n [StringType, ObjectType],\n (ctx, [key, obj]) => get(key.evaluate(ctx), obj.evaluate(ctx))\n ]\n ]\n },\n 'feature-state': [\n ValueType,\n [StringType],\n (ctx, [key]) => get(key.evaluate(ctx), ctx.featureState || {})\n ],\n 'properties': [\n ObjectType,\n [],\n (ctx) => ctx.properties()\n ],\n 'geometry-type': [\n StringType,\n [],\n (ctx) => ctx.geometryType()\n ],\n 'id': [\n ValueType,\n [],\n (ctx) => ctx.id()\n ],\n 'zoom': [\n NumberType,\n [],\n (ctx) => ctx.globals.zoom\n ],\n 'heatmap-density': [\n NumberType,\n [],\n (ctx) => ctx.globals.heatmapDensity || 0\n ],\n 'line-progress': [\n NumberType,\n [],\n (ctx) => ctx.globals.lineProgress || 0\n ],\n 'accumulated': [\n ValueType,\n [],\n (ctx) => ctx.globals.accumulated === undefined ? null : ctx.globals.accumulated\n ],\n '+': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => {\n let result = 0;\n for (const arg of args) {\n result += arg.evaluate(ctx);\n }\n return result;\n }\n ],\n '*': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => {\n let result = 1;\n for (const arg of args) {\n result *= arg.evaluate(ctx);\n }\n return result;\n }\n ],\n '-': {\n type: NumberType,\n overloads: [\n [\n [NumberType, NumberType],\n (ctx, [a, b]) => a.evaluate(ctx) - b.evaluate(ctx)\n ], [\n [NumberType],\n (ctx, [a]) => -a.evaluate(ctx)\n ]\n ]\n },\n '/': [\n NumberType,\n [NumberType, NumberType],\n (ctx, [a, b]) => a.evaluate(ctx) / b.evaluate(ctx)\n ],\n '%': [\n NumberType,\n [NumberType, NumberType],\n (ctx, [a, b]) => a.evaluate(ctx) % b.evaluate(ctx)\n ],\n 'ln2': [\n NumberType,\n [],\n () => Math.LN2\n ],\n 'pi': [\n NumberType,\n [],\n () => Math.PI\n ],\n 'e': [\n NumberType,\n [],\n () => Math.E\n ],\n '^': [\n NumberType,\n [NumberType, NumberType],\n (ctx, [b, e]) => Math.pow(b.evaluate(ctx), e.evaluate(ctx))\n ],\n 'sqrt': [\n NumberType,\n [NumberType],\n (ctx, [x]) => Math.sqrt(x.evaluate(ctx))\n ],\n 'log10': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.log(n.evaluate(ctx)) / Math.LN10\n ],\n 'ln': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.log(n.evaluate(ctx))\n ],\n 'log2': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.log(n.evaluate(ctx)) / Math.LN2\n ],\n 'sin': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.sin(n.evaluate(ctx))\n ],\n 'cos': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.cos(n.evaluate(ctx))\n ],\n 'tan': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.tan(n.evaluate(ctx))\n ],\n 'asin': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.asin(n.evaluate(ctx))\n ],\n 'acos': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.acos(n.evaluate(ctx))\n ],\n 'atan': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.atan(n.evaluate(ctx))\n ],\n 'min': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => Math.min(...args.map(arg => arg.evaluate(ctx)))\n ],\n 'max': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => Math.max(...args.map(arg => arg.evaluate(ctx)))\n ],\n 'abs': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.abs(n.evaluate(ctx))\n ],\n 'round': [\n NumberType,\n [NumberType],\n (ctx, [n]) => {\n const v = n.evaluate(ctx);\n // Javascript's Math.round() rounds towards +Infinity for halfway\n // values, even when they're negative. It's more common to round\n // away from 0 (e.g., this is what python and C++ do)\n return v < 0 ? -Math.round(-v) : Math.round(v);\n }\n ],\n 'floor': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.floor(n.evaluate(ctx))\n ],\n 'ceil': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.ceil(n.evaluate(ctx))\n ],\n 'filter-==': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => ctx.properties()[k.value] === v.value\n ],\n 'filter-id-==': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => ctx.id() === v.value\n ],\n 'filter-type-==': [\n BooleanType,\n [StringType],\n (ctx, [v]) => ctx.geometryType() === v.value\n ],\n 'filter-<': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a < b;\n }\n ],\n 'filter-id-<': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a < b;\n }\n ],\n 'filter->': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a > b;\n }\n ],\n 'filter-id->': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a > b;\n }\n ],\n 'filter-<=': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a <= b;\n }\n ],\n 'filter-id-<=': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a <= b;\n }\n ],\n 'filter->=': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a >= b;\n }\n ],\n 'filter-id->=': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a >= b;\n }\n ],\n 'filter-has': [\n BooleanType,\n [ValueType],\n (ctx, [k]) => k.value in ctx.properties()\n ],\n 'filter-has-id': [\n BooleanType,\n [],\n (ctx) => (ctx.id() !== null && ctx.id() !== undefined)\n ],\n 'filter-type-in': [\n BooleanType,\n [array$1(StringType)],\n (ctx, [v]) => v.value.indexOf(ctx.geometryType()) >= 0\n ],\n 'filter-id-in': [\n BooleanType,\n [array$1(ValueType)],\n (ctx, [v]) => v.value.indexOf(ctx.id()) >= 0\n ],\n 'filter-in-small': [\n BooleanType,\n [StringType, array$1(ValueType)],\n // assumes v is an array literal\n (ctx, [k, v]) => v.value.indexOf(ctx.properties()[k.value]) >= 0\n ],\n 'filter-in-large': [\n BooleanType,\n [StringType, array$1(ValueType)],\n // assumes v is a array literal with values sorted in ascending order and of a single type\n (ctx, [k, v]) => binarySearch(ctx.properties()[k.value], v.value, 0, v.value.length - 1)\n ],\n 'all': {\n type: BooleanType,\n overloads: [\n [\n [BooleanType, BooleanType],\n (ctx, [a, b]) => a.evaluate(ctx) && b.evaluate(ctx)\n ],\n [\n varargs(BooleanType),\n (ctx, args) => {\n for (const arg of args) {\n if (!arg.evaluate(ctx))\n return false;\n }\n return true;\n }\n ]\n ]\n },\n 'any': {\n type: BooleanType,\n overloads: [\n [\n [BooleanType, BooleanType],\n (ctx, [a, b]) => a.evaluate(ctx) || b.evaluate(ctx)\n ],\n [\n varargs(BooleanType),\n (ctx, args) => {\n for (const arg of args) {\n if (arg.evaluate(ctx))\n return true;\n }\n return false;\n }\n ]\n ]\n },\n '!': [\n BooleanType,\n [BooleanType],\n (ctx, [b]) => !b.evaluate(ctx)\n ],\n 'is-supported-script': [\n BooleanType,\n [StringType],\n // At parse time this will always return true, so we need to exclude this expression with isGlobalPropertyConstant\n (ctx, [s]) => {\n const isSupportedScript = ctx.globals && ctx.globals.isSupportedScript;\n if (isSupportedScript) {\n return isSupportedScript(s.evaluate(ctx));\n }\n return true;\n }\n ],\n 'upcase': [\n StringType,\n [StringType],\n (ctx, [s]) => s.evaluate(ctx).toUpperCase()\n ],\n 'downcase': [\n StringType,\n [StringType],\n (ctx, [s]) => s.evaluate(ctx).toLowerCase()\n ],\n 'concat': [\n StringType,\n varargs(ValueType),\n (ctx, args) => args.map(arg => toString(arg.evaluate(ctx))).join('')\n ],\n 'resolved-locale': [\n StringType,\n [CollatorType],\n (ctx, [collator]) => collator.evaluate(ctx).resolvedLocale()\n ]\n});\n\nfunction success(value) {\n return { result: 'success', value };\n}\nfunction error(value) {\n return { result: 'error', value };\n}\n\nfunction supportsPropertyExpression(spec) {\n return spec['property-type'] === 'data-driven' || spec['property-type'] === 'cross-faded-data-driven';\n}\nfunction supportsZoomExpression(spec) {\n return !!spec.expression && spec.expression.parameters.indexOf('zoom') > -1;\n}\nfunction supportsInterpolation(spec) {\n return !!spec.expression && spec.expression.interpolated;\n}\n\nfunction getType(val) {\n if (val instanceof Number) {\n return 'number';\n }\n else if (val instanceof String) {\n return 'string';\n }\n else if (val instanceof Boolean) {\n return 'boolean';\n }\n else if (Array.isArray(val)) {\n return 'array';\n }\n else if (val === null) {\n return 'null';\n }\n else {\n return typeof val;\n }\n}\n\nfunction isFunction(value) {\n return typeof value === 'object' && value !== null && !Array.isArray(value);\n}\nfunction identityFunction(x) {\n return x;\n}\nfunction createFunction(parameters, propertySpec) {\n const isColor = propertySpec.type === 'color';\n const zoomAndFeatureDependent = parameters.stops && typeof parameters.stops[0][0] === 'object';\n const featureDependent = zoomAndFeatureDependent || parameters.property !== undefined;\n const zoomDependent = zoomAndFeatureDependent || !featureDependent;\n const type = parameters.type || (supportsInterpolation(propertySpec) ? 'exponential' : 'interval');\n if (isColor || propertySpec.type === 'padding') {\n const parseFn = isColor ? Color.parse : Padding.parse;\n parameters = extendBy({}, parameters);\n if (parameters.stops) {\n parameters.stops = parameters.stops.map((stop) => {\n return [stop[0], parseFn(stop[1])];\n });\n }\n if (parameters.default) {\n parameters.default = parseFn(parameters.default);\n }\n else {\n parameters.default = parseFn(propertySpec.default);\n }\n }\n if (parameters.colorSpace && !isSupportedInterpolationColorSpace(parameters.colorSpace)) {\n throw new Error(`Unknown color space: \"${parameters.colorSpace}\"`);\n }\n let innerFun;\n let hashedStops;\n let categoricalKeyType;\n if (type === 'exponential') {\n innerFun = evaluateExponentialFunction;\n }\n else if (type === 'interval') {\n innerFun = evaluateIntervalFunction;\n }\n else if (type === 'categorical') {\n innerFun = evaluateCategoricalFunction;\n // For categorical functions, generate an Object as a hashmap of the stops for fast searching\n hashedStops = Object.create(null);\n for (const stop of parameters.stops) {\n hashedStops[stop[0]] = stop[1];\n }\n // Infer key type based on first stop key-- used to encforce strict type checking later\n categoricalKeyType = typeof parameters.stops[0][0];\n }\n else if (type === 'identity') {\n innerFun = evaluateIdentityFunction;\n }\n else {\n throw new Error(`Unknown function type \"${type}\"`);\n }\n if (zoomAndFeatureDependent) {\n const featureFunctions = {};\n const zoomStops = [];\n for (let s = 0; s < parameters.stops.length; s++) {\n const stop = parameters.stops[s];\n const zoom = stop[0].zoom;\n if (featureFunctions[zoom] === undefined) {\n featureFunctions[zoom] = {\n zoom,\n type: parameters.type,\n property: parameters.property,\n default: parameters.default,\n stops: []\n };\n zoomStops.push(zoom);\n }\n featureFunctions[zoom].stops.push([stop[0].value, stop[1]]);\n }\n const featureFunctionStops = [];\n for (const z of zoomStops) {\n featureFunctionStops.push([featureFunctions[z].zoom, createFunction(featureFunctions[z], propertySpec)]);\n }\n const interpolationType = { name: 'linear' };\n return {\n kind: 'composite',\n interpolationType,\n interpolationFactor: Interpolate.interpolationFactor.bind(undefined, interpolationType),\n zoomStops: featureFunctionStops.map(s => s[0]),\n evaluate({ zoom }, properties) {\n return evaluateExponentialFunction({\n stops: featureFunctionStops,\n base: parameters.base\n }, propertySpec, zoom).evaluate(zoom, properties);\n }\n };\n }\n else if (zoomDependent) {\n const interpolationType = type === 'exponential' ?\n { name: 'exponential', base: parameters.base !== undefined ? parameters.base : 1 } : null;\n return {\n kind: 'camera',\n interpolationType,\n interpolationFactor: Interpolate.interpolationFactor.bind(undefined, interpolationType),\n zoomStops: parameters.stops.map(s => s[0]),\n evaluate: ({ zoom }) => innerFun(parameters, propertySpec, zoom, hashedStops, categoricalKeyType)\n };\n }\n else {\n return {\n kind: 'source',\n evaluate(_, feature) {\n const value = feature && feature.properties ? feature.properties[parameters.property] : undefined;\n if (value === undefined) {\n return coalesce$1(parameters.default, propertySpec.default);\n }\n return innerFun(parameters, propertySpec, value, hashedStops, categoricalKeyType);\n }\n };\n }\n}\nfunction coalesce$1(a, b, c) {\n if (a !== undefined)\n return a;\n if (b !== undefined)\n return b;\n if (c !== undefined)\n return c;\n}\nfunction evaluateCategoricalFunction(parameters, propertySpec, input, hashedStops, keyType) {\n const evaluated = typeof input === keyType ? hashedStops[input] : undefined; // Enforce strict typing on input\n return coalesce$1(evaluated, parameters.default, propertySpec.default);\n}\nfunction evaluateIntervalFunction(parameters, propertySpec, input) {\n // Edge cases\n if (getType(input) !== 'number')\n return coalesce$1(parameters.default, propertySpec.default);\n const n = parameters.stops.length;\n if (n === 1)\n return parameters.stops[0][1];\n if (input <= parameters.stops[0][0])\n return parameters.stops[0][1];\n if (input >= parameters.stops[n - 1][0])\n return parameters.stops[n - 1][1];\n const index = findStopLessThanOrEqualTo(parameters.stops.map((stop) => stop[0]), input);\n return parameters.stops[index][1];\n}\nfunction evaluateExponentialFunction(parameters, propertySpec, input) {\n const base = parameters.base !== undefined ? parameters.base : 1;\n // Edge cases\n if (getType(input) !== 'number')\n return coalesce$1(parameters.default, propertySpec.default);\n const n = parameters.stops.length;\n if (n === 1)\n return parameters.stops[0][1];\n if (input <= parameters.stops[0][0])\n return parameters.stops[0][1];\n if (input >= parameters.stops[n - 1][0])\n return parameters.stops[n - 1][1];\n const index = findStopLessThanOrEqualTo(parameters.stops.map((stop) => stop[0]), input);\n const t = interpolationFactor(input, base, parameters.stops[index][0], parameters.stops[index + 1][0]);\n const outputLower = parameters.stops[index][1];\n const outputUpper = parameters.stops[index + 1][1];\n const interp = interpolate[propertySpec.type] || identityFunction;\n if (typeof outputLower.evaluate === 'function') {\n return {\n evaluate(...args) {\n const evaluatedLower = outputLower.evaluate.apply(undefined, args);\n const evaluatedUpper = outputUpper.evaluate.apply(undefined, args);\n // Special case for fill-outline-color, which has no spec default.\n if (evaluatedLower === undefined || evaluatedUpper === undefined) {\n return undefined;\n }\n return interp(evaluatedLower, evaluatedUpper, t, parameters.colorSpace);\n }\n };\n }\n return interp(outputLower, outputUpper, t, parameters.colorSpace);\n}\nfunction evaluateIdentityFunction(parameters, propertySpec, input) {\n switch (propertySpec.type) {\n case 'color':\n input = Color.parse(input);\n break;\n case 'formatted':\n input = Formatted.fromString(input.toString());\n break;\n case 'resolvedImage':\n input = ResolvedImage.fromString(input.toString());\n break;\n case 'padding':\n input = Padding.parse(input);\n break;\n default:\n if (getType(input) !== propertySpec.type && (propertySpec.type !== 'enum' || !propertySpec.values[input])) {\n input = undefined;\n }\n }\n return coalesce$1(input, parameters.default, propertySpec.default);\n}\n/**\n * Returns a ratio that can be used to interpolate between exponential function\n * stops.\n *\n * How it works:\n * Two consecutive stop values define a (scaled and shifted) exponential\n * function `f(x) = a * base^x + b`, where `base` is the user-specified base,\n * and `a` and `b` are constants affording sufficient degrees of freedom to fit\n * the function to the given stops.\n *\n * Here's a bit of algebra that lets us compute `f(x)` directly from the stop\n * values without explicitly solving for `a` and `b`:\n *\n * First stop value: `f(x0) = y0 = a * base^x0 + b`\n * Second stop value: `f(x1) = y1 = a * base^x1 + b`\n * => `y1 - y0 = a(base^x1 - base^x0)`\n * => `a = (y1 - y0)/(base^x1 - base^x0)`\n *\n * Desired value: `f(x) = y = a * base^x + b`\n * => `f(x) = y0 + a * (base^x - base^x0)`\n *\n * From the above, we can replace the `a` in `a * (base^x - base^x0)` and do a\n * little algebra:\n * ```\n * a * (base^x - base^x0) = (y1 - y0)/(base^x1 - base^x0) * (base^x - base^x0)\n * = (y1 - y0) * (base^x - base^x0) / (base^x1 - base^x0)\n * ```\n *\n * If we let `(base^x - base^x0) / (base^x1 base^x0)`, then we have\n * `f(x) = y0 + (y1 - y0) * ratio`. In other words, `ratio` may be treated as\n * an interpolation factor between the two stops' output values.\n *\n * (Note: a slightly different form for `ratio`,\n * `(base^(x-x0) - 1) / (base^(x1-x0) - 1) `, is equivalent, but requires fewer\n * expensive `Math.pow()` operations.)\n *\n * @private\n */\nfunction interpolationFactor(input, base, lowerValue, upperValue) {\n const difference = upperValue - lowerValue;\n const progress = input - lowerValue;\n if (difference === 0) {\n return 0;\n }\n else if (base === 1) {\n return progress / difference;\n }\n else {\n return (Math.pow(base, progress) - 1) / (Math.pow(base, difference) - 1);\n }\n}\n\nclass StyleExpression {\n constructor(expression, propertySpec) {\n this.expression = expression;\n this._warningHistory = {};\n this._evaluator = new EvaluationContext();\n this._defaultValue = propertySpec ? getDefaultValue(propertySpec) : null;\n this._enumValues = propertySpec && propertySpec.type === 'enum' ? propertySpec.values : null;\n }\n evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) {\n this._evaluator.globals = globals;\n this._evaluator.feature = feature;\n this._evaluator.featureState = featureState;\n this._evaluator.canonical = canonical;\n this._evaluator.availableImages = availableImages || null;\n this._evaluator.formattedSection = formattedSection;\n return this.expression.evaluate(this._evaluator);\n }\n evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) {\n this._evaluator.globals = globals;\n this._evaluator.feature = feature || null;\n this._evaluator.featureState = featureState || null;\n this._evaluator.canonical = canonical;\n this._evaluator.availableImages = availableImages || null;\n this._evaluator.formattedSection = formattedSection || null;\n try {\n const val = this.expression.evaluate(this._evaluator);\n // eslint-disable-next-line no-self-compare\n if (val === null || val === undefined || (typeof val === 'number' && val !== val)) {\n return this._defaultValue;\n }\n if (this._enumValues && !(val in this._enumValues)) {\n throw new RuntimeError(`Expected value to be one of ${Object.keys(this._enumValues).map(v => JSON.stringify(v)).join(', ')}, but found ${JSON.stringify(val)} instead.`);\n }\n return val;\n }\n catch (e) {\n if (!this._warningHistory[e.message]) {\n this._warningHistory[e.message] = true;\n if (typeof console !== 'undefined') {\n console.warn(e.message);\n }\n }\n return this._defaultValue;\n }\n }\n}\nfunction isExpression(expression) {\n return Array.isArray(expression) && expression.length > 0 &&\n typeof expression[0] === 'string' && expression[0] in expressions;\n}\n/**\n * Parse and typecheck the given style spec JSON expression. If\n * options.defaultValue is provided, then the resulting StyleExpression's\n * `evaluate()` method will handle errors by logging a warning (once per\n * message) and returning the default value. Otherwise, it will throw\n * evaluation errors.\n *\n * @private\n */\nfunction createExpression(expression, propertySpec) {\n const parser = new ParsingContext(expressions, isExpressionConstant, [], propertySpec ? getExpectedType(propertySpec) : undefined);\n // For string-valued properties, coerce to string at the top level rather than asserting.\n const parsed = parser.parse(expression, undefined, undefined, undefined, propertySpec && propertySpec.type === 'string' ? { typeAnnotation: 'coerce' } : undefined);\n if (!parsed) {\n return error(parser.errors);\n }\n return success(new StyleExpression(parsed, propertySpec));\n}\nclass ZoomConstantExpression {\n constructor(kind, expression) {\n this.kind = kind;\n this._styleExpression = expression;\n this.isStateDependent = kind !== 'constant' && !isStateConstant(expression.expression);\n }\n evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluate(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n}\nclass ZoomDependentExpression {\n constructor(kind, expression, zoomStops, interpolationType) {\n this.kind = kind;\n this.zoomStops = zoomStops;\n this._styleExpression = expression;\n this.isStateDependent = kind !== 'camera' && !isStateConstant(expression.expression);\n this.interpolationType = interpolationType;\n }\n evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluate(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n interpolationFactor(input, lower, upper) {\n if (this.interpolationType) {\n return Interpolate.interpolationFactor(this.interpolationType, input, lower, upper);\n }\n else {\n return 0;\n }\n }\n}\nfunction isZoomExpression(expression) {\n return expression._styleExpression !== undefined;\n}\nfunction createPropertyExpression(expressionInput, propertySpec) {\n const expression = createExpression(expressionInput, propertySpec);\n if (expression.result === 'error') {\n return expression;\n }\n const parsed = expression.value.expression;\n const isFeatureConstantResult = isFeatureConstant(parsed);\n if (!isFeatureConstantResult && !supportsPropertyExpression(propertySpec)) {\n return error([new ExpressionParsingError('', 'data expressions not supported')]);\n }\n const isZoomConstant = isGlobalPropertyConstant(parsed, ['zoom']);\n if (!isZoomConstant && !supportsZoomExpression(propertySpec)) {\n return error([new ExpressionParsingError('', 'zoom expressions not supported')]);\n }\n const zoomCurve = findZoomCurve(parsed);\n if (!zoomCurve && !isZoomConstant) {\n return error([new ExpressionParsingError('', '\"zoom\" expression may only be used as input to a top-level \"step\" or \"interpolate\" expression.')]);\n }\n else if (zoomCurve instanceof ExpressionParsingError) {\n return error([zoomCurve]);\n }\n else if (zoomCurve instanceof Interpolate && !supportsInterpolation(propertySpec)) {\n return error([new ExpressionParsingError('', '\"interpolate\" expressions cannot be used with this property')]);\n }\n if (!zoomCurve) {\n return success(isFeatureConstantResult ?\n new ZoomConstantExpression('constant', expression.value) :\n new ZoomConstantExpression('source', expression.value));\n }\n const interpolationType = zoomCurve instanceof Interpolate ? zoomCurve.interpolation : undefined;\n return success(isFeatureConstantResult ?\n new ZoomDependentExpression('camera', expression.value, zoomCurve.labels, interpolationType) :\n new ZoomDependentExpression('composite', expression.value, zoomCurve.labels, interpolationType));\n}\n// serialization wrapper for old-style stop functions normalized to the\n// expression interface\nclass StylePropertyFunction {\n constructor(parameters, specification) {\n this._parameters = parameters;\n this._specification = specification;\n extendBy(this, createFunction(this._parameters, this._specification));\n }\n static deserialize(serialized) {\n return new StylePropertyFunction(serialized._parameters, serialized._specification);\n }\n static serialize(input) {\n return {\n _parameters: input._parameters,\n _specification: input._specification\n };\n }\n}\nfunction normalizePropertyExpression(value, specification) {\n if (isFunction(value)) {\n return new StylePropertyFunction(value, specification);\n }\n else if (isExpression(value)) {\n const expression = createPropertyExpression(value, specification);\n if (expression.result === 'error') {\n // this should have been caught in validation\n throw new Error(expression.value.map(err => `${err.key}: ${err.message}`).join(', '));\n }\n return expression.value;\n }\n else {\n let constant = value;\n if (specification.type === 'color' && typeof value === 'string') {\n constant = Color.parse(value);\n }\n else if (specification.type === 'padding' && (typeof value === 'number' || Array.isArray(value))) {\n constant = Padding.parse(value);\n }\n else if (specification.type === 'variableAnchorOffsetCollection' && Array.isArray(value)) {\n constant = VariableAnchorOffsetCollection.parse(value);\n }\n return {\n kind: 'constant',\n evaluate: () => constant\n };\n }\n}\n// Zoom-dependent expressions may only use [\"zoom\"] as the input to a top-level \"step\" or \"interpolate\"\n// expression (collectively referred to as a \"curve\"). The curve may be wrapped in one or more \"let\" or\n// \"coalesce\" expressions.\nfunction findZoomCurve(expression) {\n let result = null;\n if (expression instanceof Let) {\n result = findZoomCurve(expression.result);\n }\n else if (expression instanceof Coalesce) {\n for (const arg of expression.args) {\n result = findZoomCurve(arg);\n if (result) {\n break;\n }\n }\n }\n else if ((expression instanceof Step || expression instanceof Interpolate) &&\n expression.input instanceof CompoundExpression &&\n expression.input.name === 'zoom') {\n result = expression;\n }\n if (result instanceof ExpressionParsingError) {\n return result;\n }\n expression.eachChild((child) => {\n const childResult = findZoomCurve(child);\n if (childResult instanceof ExpressionParsingError) {\n result = childResult;\n }\n else if (!result && childResult) {\n result = new ExpressionParsingError('', '\"zoom\" expression may only be used as input to a top-level \"step\" or \"interpolate\" expression.');\n }\n else if (result && childResult && result !== childResult) {\n result = new ExpressionParsingError('', 'Only one zoom-based \"step\" or \"interpolate\" subexpression may be used in an expression.');\n }\n });\n return result;\n}\nfunction getExpectedType(spec) {\n const types = {\n color: ColorType,\n string: StringType,\n number: NumberType,\n enum: StringType,\n boolean: BooleanType,\n formatted: FormattedType,\n padding: PaddingType,\n resolvedImage: ResolvedImageType,\n variableAnchorOffsetCollection: VariableAnchorOffsetCollectionType\n };\n if (spec.type === 'array') {\n return array$1(types[spec.value] || ValueType, spec.length);\n }\n return types[spec.type];\n}\nfunction getDefaultValue(spec) {\n if (spec.type === 'color' && isFunction(spec.default)) {\n // Special case for heatmap-color: it uses the 'default:' to define a\n // default color ramp, but createExpression expects a simple value to fall\n // back to in case of runtime errors\n return new Color(0, 0, 0, 0);\n }\n else if (spec.type === 'color') {\n return Color.parse(spec.default) || null;\n }\n else if (spec.type === 'padding') {\n return Padding.parse(spec.default) || null;\n }\n else if (spec.type === 'variableAnchorOffsetCollection') {\n return VariableAnchorOffsetCollection.parse(spec.default) || null;\n }\n else if (spec.default === undefined) {\n return null;\n }\n else {\n return spec.default;\n }\n}\n\nfunction isExpressionFilter(filter) {\n if (filter === true || filter === false) {\n return true;\n }\n if (!Array.isArray(filter) || filter.length === 0) {\n return false;\n }\n switch (filter[0]) {\n case 'has':\n return filter.length >= 2 && filter[1] !== '$id' && filter[1] !== '$type';\n case 'in':\n return filter.length >= 3 && (typeof filter[1] !== 'string' || Array.isArray(filter[2]));\n case '!in':\n case '!has':\n case 'none':\n return false;\n case '==':\n case '!=':\n case '>':\n case '>=':\n case '<':\n case '<=':\n return filter.length !== 3 || (Array.isArray(filter[1]) || Array.isArray(filter[2]));\n case 'any':\n case 'all':\n for (const f of filter.slice(1)) {\n if (!isExpressionFilter(f) && typeof f !== 'boolean') {\n return false;\n }\n }\n return true;\n default:\n return true;\n }\n}\nconst filterSpec = {\n 'type': 'boolean',\n 'default': false,\n 'transition': false,\n 'property-type': 'data-driven',\n 'expression': {\n 'interpolated': false,\n 'parameters': ['zoom', 'feature']\n }\n};\n/**\n * Given a filter expressed as nested arrays, return a new function\n * that evaluates whether a given feature (with a .properties or .tags property)\n * passes its test.\n *\n * @private\n * @param {Array} filter MapLibre filter\n * @returns {Function} filter-evaluating function\n */\nfunction createFilter(filter) {\n if (filter === null || filter === undefined) {\n return { filter: () => true, needGeometry: false };\n }\n if (!isExpressionFilter(filter)) {\n filter = convertFilter$1(filter);\n }\n const compiled = createExpression(filter, filterSpec);\n if (compiled.result === 'error') {\n throw new Error(compiled.value.map(err => `${err.key}: ${err.message}`).join(', '));\n }\n else {\n const needGeometry = geometryNeeded(filter);\n return { filter: (globalProperties, feature, canonical) => compiled.value.evaluate(globalProperties, feature, {}, canonical),\n needGeometry };\n }\n}\n// Comparison function to sort numbers and strings\nfunction compare(a, b) {\n return a < b ? -1 : a > b ? 1 : 0;\n}\nfunction geometryNeeded(filter) {\n if (!Array.isArray(filter))\n return false;\n if (filter[0] === 'within')\n return true;\n for (let index = 1; index < filter.length; index++) {\n if (geometryNeeded(filter[index]))\n return true;\n }\n return false;\n}\nfunction convertFilter$1(filter) {\n if (!filter)\n return true;\n const op = filter[0];\n if (filter.length <= 1)\n return (op !== 'any');\n const converted = op === '==' ? convertComparisonOp$1(filter[1], filter[2], '==') :\n op === '!=' ? convertNegation(convertComparisonOp$1(filter[1], filter[2], '==')) :\n op === '<' ||\n op === '>' ||\n op === '<=' ||\n op === '>=' ? convertComparisonOp$1(filter[1], filter[2], op) :\n op === 'any' ? convertDisjunctionOp(filter.slice(1)) :\n op === 'all' ? ['all'].concat(filter.slice(1).map(convertFilter$1)) :\n op === 'none' ? ['all'].concat(filter.slice(1).map(convertFilter$1).map(convertNegation)) :\n op === 'in' ? convertInOp$1(filter[1], filter.slice(2)) :\n op === '!in' ? convertNegation(convertInOp$1(filter[1], filter.slice(2))) :\n op === 'has' ? convertHasOp$1(filter[1]) :\n op === '!has' ? convertNegation(convertHasOp$1(filter[1])) :\n op === 'within' ? filter :\n true;\n return converted;\n}\nfunction convertComparisonOp$1(property, value, op) {\n switch (property) {\n case '$type':\n return [`filter-type-${op}`, value];\n case '$id':\n return [`filter-id-${op}`, value];\n default:\n return [`filter-${op}`, property, value];\n }\n}\nfunction convertDisjunctionOp(filters) {\n return ['any'].concat(filters.map(convertFilter$1));\n}\nfunction convertInOp$1(property, values) {\n if (values.length === 0) {\n return false;\n }\n switch (property) {\n case '$type':\n return ['filter-type-in', ['literal', values]];\n case '$id':\n return ['filter-id-in', ['literal', values]];\n default:\n if (values.length > 200 && !values.some(v => typeof v !== typeof values[0])) {\n return ['filter-in-large', property, ['literal', values.sort(compare)]];\n }\n else {\n return ['filter-in-small', property, ['literal', values]];\n }\n }\n}\nfunction convertHasOp$1(property) {\n switch (property) {\n case '$type':\n return true;\n case '$id':\n return ['filter-has-id'];\n default:\n return ['filter-has', property];\n }\n}\nfunction convertNegation(filter) {\n return ['!', filter];\n}\n\n/*\n * Convert the given filter to an expression, storing the expected types for\n * any feature properties referenced in expectedTypes.\n *\n * These expected types are needed in order to construct preflight type checks\n * needed for handling 'any' filters. A preflight type check is necessary in\n * order to mimic legacy filters' semantics around expected type mismatches.\n * For example, consider the legacy filter:\n *\n * [\"any\", [\"all\", [\">\", \"y\", 0], [\">\", \"y\", 0]], [\">\", \"x\", 0]]\n *\n * Naively, we might convert this to the expression:\n *\n * [\"any\", [\"all\", [\">\", [\"get\", \"y\"], 0], [\">\", [\"get\", \"z\"], 0]], [\">\", [\"get\", \"x\"], 0]]\n *\n * But if we tried to evaluate this against, say `{x: 1, y: null, z: 0}`, the\n * [\">\", [\"get\", \"y\"], 0] would cause an evaluation error, leading to the\n * entire filter returning false. Legacy filter semantics, though, ask for\n * [\">\", \"y\", 0] to simply return `false` when `y` is of the wrong type,\n * allowing the subsequent terms of the outer \"any\" expression to be evaluated\n * (resulting, in this case, in a `true` value, because x > 0).\n *\n * We account for this by inserting a preflight type-checking expression before\n * each \"any\" term, allowing us to avoid evaluating the actual converted filter\n * if any type mismatches would cause it to produce an evalaution error:\n *\n * [\"any\",\n * [\"case\",\n * [\"all\", [\"==\", [\"typeof\", [\"get\", \"y\"]], \"number\"], [\"==\", [\"typeof\", [\"get\", \"z\"], \"number]],\n * [\"all\", [\">\", [\"get\", \"y\"], 0], [\">\", [\"get\", \"z\"], 0]],\n * false\n * ],\n * [\"case\",\n * [\"==\", [\"typeof\", [\"get\", \"x\"], \"number\"]],\n * [\">\", [\"get\", \"x\"], 0],\n * false\n * ]\n * ]\n *\n * An alternative, possibly more direct approach would be to use type checks\n * in the conversion of each comparison operator, so that the converted version\n * of each individual ==, >=, etc. would mimic the legacy filter semantics. The\n * downside of this approach is that it can lead to many more type checks than\n * would otherwise be necessary: outside the context of an \"any\" expression,\n * bailing out due to a runtime type error (expression semantics) and returning\n * false (legacy filter semantics) are equivalent: they cause the filter to\n * produce a `false` result.\n */\nfunction convertFilter(filter, expectedTypes = {}) {\n if (isExpressionFilter(filter))\n return filter;\n if (!filter)\n return true;\n const legacyFilter = filter;\n const legacyOp = legacyFilter[0];\n if (filter.length <= 1)\n return (legacyOp !== 'any');\n switch (legacyOp) {\n case '==':\n case '!=':\n case '<':\n case '>':\n case '<=':\n case '>=': {\n const [, property, value] = filter;\n return convertComparisonOp(property, value, legacyOp, expectedTypes);\n }\n case 'any': {\n const [, ...conditions] = legacyFilter;\n const children = conditions.map((f) => {\n const types = {};\n const child = convertFilter(f, types);\n const typechecks = runtimeTypeChecks(types);\n return typechecks === true ? child : ['case', typechecks, child, false];\n });\n return ['any', ...children];\n }\n case 'all': {\n const [, ...conditions] = legacyFilter;\n const children = conditions.map(f => convertFilter(f, expectedTypes));\n return children.length > 1 ? ['all', ...children] : children[0];\n }\n case 'none': {\n const [, ...conditions] = legacyFilter;\n return ['!', convertFilter(['any', ...conditions], {})];\n }\n case 'in': {\n const [, property, ...values] = legacyFilter;\n return convertInOp(property, values);\n }\n case '!in': {\n const [, property, ...values] = legacyFilter;\n return convertInOp(property, values, true);\n }\n case 'has':\n return convertHasOp(legacyFilter[1]);\n case '!has':\n return ['!', convertHasOp(legacyFilter[1])];\n default:\n return true;\n }\n}\n// Given a set of feature properties and an expected type for each one,\n// construct an boolean expression that tests whether each property has the\n// right type.\n// E.g.: for {name: 'string', population: 'number'}, return\n// [ 'all',\n// ['==', ['typeof', ['get', 'name'], 'string']],\n// ['==', ['typeof', ['get', 'population'], 'number]]\n// ]\nfunction runtimeTypeChecks(expectedTypes) {\n const conditions = [];\n for (const property in expectedTypes) {\n const get = property === '$id' ? ['id'] : ['get', property];\n conditions.push(['==', ['typeof', get], expectedTypes[property]]);\n }\n if (conditions.length === 0)\n return true;\n if (conditions.length === 1)\n return conditions[0];\n return ['all', ...conditions];\n}\nfunction convertComparisonOp(property, value, op, expectedTypes) {\n let get;\n if (property === '$type') {\n return [op, ['geometry-type'], value];\n }\n else if (property === '$id') {\n get = ['id'];\n }\n else {\n get = ['get', property];\n }\n if (expectedTypes && value !== null) {\n const type = typeof value;\n expectedTypes[property] = type;\n }\n if (op === '==' && property !== '$id' && value === null) {\n return [\n 'all',\n ['has', property],\n ['==', get, null]\n ];\n }\n else if (op === '!=' && property !== '$id' && value === null) {\n return [\n 'any',\n ['!', ['has', property]],\n ['!=', get, null]\n ];\n }\n return [op, get, value];\n}\nfunction convertInOp(property, values, negate = false) {\n if (values.length === 0)\n return negate;\n let get;\n if (property === '$type') {\n get = ['geometry-type'];\n }\n else if (property === '$id') {\n get = ['id'];\n }\n else {\n get = ['get', property];\n }\n // Determine if the list of values to be searched is homogenously typed.\n // If so (and if the type is string or number), then we can use a\n // [match, input, [...values], true, false] construction rather than a\n // bunch of `==` tests.\n let uniformTypes = true;\n const type = typeof values[0];\n for (const value of values) {\n if (typeof value !== type) {\n uniformTypes = false;\n break;\n }\n }\n if (uniformTypes && (type === 'string' || type === 'number')) {\n // Match expressions must have unique values.\n const uniqueValues = values.sort().filter((v, i) => i === 0 || values[i - 1] !== v);\n return ['match', get, uniqueValues, !negate, negate];\n }\n if (negate) {\n return ['all', ...values.map(v => ['!=', get, v])];\n }\n else {\n return ['any', ...values.map(v => ['==', get, v])];\n }\n}\nfunction convertHasOp(property) {\n if (property === '$type') {\n return true;\n }\n else if (property === '$id') {\n return ['!=', ['id'], null];\n }\n else {\n return ['has', property];\n }\n}\n\nfunction convertLiteral(value) {\n return typeof value === 'object' ? ['literal', value] : value;\n}\nfunction convertFunction(parameters, propertySpec) {\n let stops = parameters.stops;\n if (!stops) {\n // identity function\n return convertIdentityFunction(parameters, propertySpec);\n }\n const zoomAndFeatureDependent = stops && typeof stops[0][0] === 'object';\n const featureDependent = zoomAndFeatureDependent || parameters.property !== undefined;\n const zoomDependent = zoomAndFeatureDependent || !featureDependent;\n stops = stops.map((stop) => {\n if (!featureDependent && propertySpec.tokens && typeof stop[1] === 'string') {\n return [stop[0], convertTokenString(stop[1])];\n }\n return [stop[0], convertLiteral(stop[1])];\n });\n if (zoomAndFeatureDependent) {\n return convertZoomAndPropertyFunction(parameters, propertySpec, stops);\n }\n else if (zoomDependent) {\n return convertZoomFunction(parameters, propertySpec, stops);\n }\n else {\n return convertPropertyFunction(parameters, propertySpec, stops);\n }\n}\nfunction convertIdentityFunction(parameters, propertySpec) {\n const get = ['get', parameters.property];\n if (parameters.default === undefined) {\n // By default, expressions for string-valued properties get coerced. To preserve\n // legacy function semantics, insert an explicit assertion instead.\n return propertySpec.type === 'string' ? ['string', get] : get;\n }\n else if (propertySpec.type === 'enum') {\n return [\n 'match',\n get,\n Object.keys(propertySpec.values),\n get,\n parameters.default\n ];\n }\n else {\n const expression = [propertySpec.type === 'color' ? 'to-color' : propertySpec.type, get, convertLiteral(parameters.default)];\n if (propertySpec.type === 'array') {\n expression.splice(1, 0, propertySpec.value, propertySpec.length || null);\n }\n return expression;\n }\n}\nfunction getInterpolateOperator(parameters) {\n switch (parameters.colorSpace) {\n case 'hcl': return 'interpolate-hcl';\n case 'lab': return 'interpolate-lab';\n default: return 'interpolate';\n }\n}\nfunction convertZoomAndPropertyFunction(parameters, propertySpec, stops) {\n const featureFunctionParameters = {};\n const featureFunctionStops = {};\n const zoomStops = [];\n for (let s = 0; s < stops.length; s++) {\n const stop = stops[s];\n const zoom = stop[0].zoom;\n if (featureFunctionParameters[zoom] === undefined) {\n featureFunctionParameters[zoom] = {\n zoom,\n type: parameters.type,\n property: parameters.property,\n default: parameters.default,\n };\n featureFunctionStops[zoom] = [];\n zoomStops.push(zoom);\n }\n featureFunctionStops[zoom].push([stop[0].value, stop[1]]);\n }\n // the interpolation type for the zoom dimension of a zoom-and-property\n // function is determined directly from the style property specification\n // for which it's being used: linear for interpolatable properties, step\n // otherwise.\n const functionType = getFunctionType({}, propertySpec);\n if (functionType === 'exponential') {\n const expression = [getInterpolateOperator(parameters), ['linear'], ['zoom']];\n for (const z of zoomStops) {\n const output = convertPropertyFunction(featureFunctionParameters[z], propertySpec, featureFunctionStops[z]);\n appendStopPair(expression, z, output, false);\n }\n return expression;\n }\n else {\n const expression = ['step', ['zoom']];\n for (const z of zoomStops) {\n const output = convertPropertyFunction(featureFunctionParameters[z], propertySpec, featureFunctionStops[z]);\n appendStopPair(expression, z, output, true);\n }\n fixupDegenerateStepCurve(expression);\n return expression;\n }\n}\nfunction coalesce(a, b) {\n if (a !== undefined)\n return a;\n if (b !== undefined)\n return b;\n}\nfunction getFallback(parameters, propertySpec) {\n const defaultValue = convertLiteral(coalesce(parameters.default, propertySpec.default));\n /*\n * Some fields with type: resolvedImage have an undefined default.\n * Because undefined is an invalid value for resolvedImage, set fallback to\n * an empty string instead of undefined to ensure output\n * passes validation.\n */\n if (defaultValue === undefined && propertySpec.type === 'resolvedImage') {\n return '';\n }\n return defaultValue;\n}\nfunction convertPropertyFunction(parameters, propertySpec, stops) {\n const type = getFunctionType(parameters, propertySpec);\n const get = ['get', parameters.property];\n if (type === 'categorical' && typeof stops[0][0] === 'boolean') {\n const expression = ['case'];\n for (const stop of stops) {\n expression.push(['==', get, stop[0]], stop[1]);\n }\n expression.push(getFallback(parameters, propertySpec));\n return expression;\n }\n else if (type === 'categorical') {\n const expression = ['match', get];\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], false);\n }\n expression.push(getFallback(parameters, propertySpec));\n return expression;\n }\n else if (type === 'interval') {\n const expression = ['step', ['number', get]];\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], true);\n }\n fixupDegenerateStepCurve(expression);\n return parameters.default === undefined ? expression : [\n 'case',\n ['==', ['typeof', get], 'number'],\n expression,\n convertLiteral(parameters.default)\n ];\n }\n else if (type === 'exponential') {\n const base = parameters.base !== undefined ? parameters.base : 1;\n const expression = [\n getInterpolateOperator(parameters),\n base === 1 ? ['linear'] : ['exponential', base],\n ['number', get]\n ];\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], false);\n }\n return parameters.default === undefined ? expression : [\n 'case',\n ['==', ['typeof', get], 'number'],\n expression,\n convertLiteral(parameters.default)\n ];\n }\n else {\n throw new Error(`Unknown property function type ${type}`);\n }\n}\nfunction convertZoomFunction(parameters, propertySpec, stops, input = ['zoom']) {\n const type = getFunctionType(parameters, propertySpec);\n let expression;\n let isStep = false;\n if (type === 'interval') {\n expression = ['step', input];\n isStep = true;\n }\n else if (type === 'exponential') {\n const base = parameters.base !== undefined ? parameters.base : 1;\n expression = [getInterpolateOperator(parameters), base === 1 ? ['linear'] : ['exponential', base], input];\n }\n else {\n throw new Error(`Unknown zoom function type \"${type}\"`);\n }\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], isStep);\n }\n fixupDegenerateStepCurve(expression);\n return expression;\n}\nfunction fixupDegenerateStepCurve(expression) {\n // degenerate step curve (i.e. a constant function): add a noop stop\n if (expression[0] === 'step' && expression.length === 3) {\n expression.push(0);\n expression.push(expression[3]);\n }\n}\nfunction appendStopPair(curve, input, output, isStep) {\n // Skip duplicate stop values. They were not validated for functions, but they are for expressions.\n // https://github.com/mapbox/mapbox-gl-js/issues/4107\n if (curve.length > 3 && input === curve[curve.length - 2]) {\n return;\n }\n // step curves don't get the first input value, as it is redundant.\n if (!(isStep && curve.length === 2)) {\n curve.push(input);\n }\n curve.push(output);\n}\nfunction getFunctionType(parameters, propertySpec) {\n if (parameters.type) {\n return parameters.type;\n }\n else {\n return propertySpec.expression.interpolated ? 'exponential' : 'interval';\n }\n}\n// \"String with {name} token\" => [\"concat\", \"String with \", [\"get\", \"name\"], \" token\"]\nfunction convertTokenString(s) {\n const result = ['concat'];\n const re = /{([^{}]+)}/g;\n let pos = 0;\n for (let match = re.exec(s); match !== null; match = re.exec(s)) {\n const literal = s.slice(pos, re.lastIndex - match[0].length);\n pos = re.lastIndex;\n if (literal.length > 0)\n result.push(literal);\n result.push(['get', match[1]]);\n }\n if (result.length === 1) {\n return s;\n }\n if (pos < s.length) {\n result.push(s.slice(pos));\n }\n else if (result.length === 2) {\n return ['to-string', result[1]];\n }\n return result;\n}\n\nfunction getPropertyReference(propertyName) {\n for (let i = 0; i < v8Spec.layout.length; i++) {\n for (const key in v8Spec[v8Spec.layout[i]]) {\n if (key === propertyName)\n return v8Spec[v8Spec.layout[i]][key];\n }\n }\n for (let i = 0; i < v8Spec.paint.length; i++) {\n for (const key in v8Spec[v8Spec.paint[i]]) {\n if (key === propertyName)\n return v8Spec[v8Spec.paint[i]][key];\n }\n }\n return null;\n}\nfunction eachSource(style, callback) {\n for (const k in style.sources) {\n callback(style.sources[k]);\n }\n}\nfunction eachLayer(style, callback) {\n for (const layer of style.layers) {\n callback(layer);\n }\n}\nfunction eachProperty(style, options, callback) {\n function inner(layer, propertyType) {\n const properties = layer[propertyType];\n if (!properties)\n return;\n Object.keys(properties).forEach((key) => {\n callback({\n path: [layer.id, propertyType, key],\n key,\n value: properties[key],\n reference: getPropertyReference(key),\n set(x) {\n properties[key] = x;\n }\n });\n });\n }\n eachLayer(style, (layer) => {\n if (options.paint) {\n inner(layer, 'paint');\n }\n if (options.layout) {\n inner(layer, 'layout');\n }\n });\n}\n\nfunction stringify(obj) {\n const type = typeof obj;\n if (type === 'number' || type === 'boolean' || type === 'string' || obj === undefined || obj === null)\n return JSON.stringify(obj);\n if (Array.isArray(obj)) {\n let str = '[';\n for (const val of obj) {\n str += `${stringify(val)},`;\n }\n return `${str}]`;\n }\n const keys = Object.keys(obj).sort();\n let str = '{';\n for (let i = 0; i < keys.length; i++) {\n str += `${JSON.stringify(keys[i])}:${stringify(obj[keys[i]])},`;\n }\n return `${str}}`;\n}\nfunction getKey(layer) {\n let key = '';\n for (const k of refProperties) {\n key += `/${stringify(layer[k])}`;\n }\n return key;\n}\n/**\n * Given an array of layers, return an array of arrays of layers where all\n * layers in each group have identical layout-affecting properties. These\n * are the properties that were formerly used by explicit `ref` mechanism\n * for layers: 'type', 'source', 'source-layer', 'minzoom', 'maxzoom',\n * 'filter', and 'layout'.\n *\n * The input is not modified. The output layers are references to the\n * input layers.\n *\n * @private\n * @param {Array} layers\n * @param {Object} [cachedKeys] - an object to keep already calculated keys.\n * @returns {Array>}\n */\nfunction groupByLayout(layers, cachedKeys) {\n const groups = {};\n for (let i = 0; i < layers.length; i++) {\n const k = (cachedKeys && cachedKeys[layers[i].id]) || getKey(layers[i]);\n // update the cache if there is one\n if (cachedKeys)\n cachedKeys[layers[i].id] = k;\n let group = groups[k];\n if (!group) {\n group = groups[k] = [];\n }\n group.push(layers[i]);\n }\n const result = [];\n for (const k in groups) {\n result.push(groups[k]);\n }\n return result;\n}\n\nfunction emptyStyle() {\n const style = {};\n const version = v8Spec['$version'];\n for (const styleKey in v8Spec['$root']) {\n const spec = v8Spec['$root'][styleKey];\n if (spec.required) {\n let value = null;\n if (styleKey === 'version') {\n value = version;\n }\n else {\n if (spec.type === 'array') {\n value = [];\n }\n else {\n value = {};\n }\n }\n if (value != null) {\n style[styleKey] = value;\n }\n }\n }\n return style;\n}\n\nfunction validateConstants(options) {\n const key = options.key;\n const constants = options.value;\n if (constants) {\n return [new ValidationError(key, constants, 'constants have been deprecated as of v8')];\n }\n else {\n return [];\n }\n}\n\n// Turn jsonlint-lines-primitives objects into primitive objects\nfunction unbundle(value) {\n if (value instanceof Number || value instanceof String || value instanceof Boolean) {\n return value.valueOf();\n }\n else {\n return value;\n }\n}\nfunction deepUnbundle(value) {\n if (Array.isArray(value)) {\n return value.map(deepUnbundle);\n }\n else if (value instanceof Object && !(value instanceof Number || value instanceof String || value instanceof Boolean)) {\n const unbundledValue = {};\n for (const key in value) {\n unbundledValue[key] = deepUnbundle(value[key]);\n }\n return unbundledValue;\n }\n return unbundle(value);\n}\n\nfunction validateObject(options) {\n const key = options.key;\n const object = options.value;\n const elementSpecs = options.valueSpec || {};\n const elementValidators = options.objectElementValidators || {};\n const style = options.style;\n const styleSpec = options.styleSpec;\n const validateSpec = options.validateSpec;\n let errors = [];\n const type = getType(object);\n if (type !== 'object') {\n return [new ValidationError(key, object, `object expected, ${type} found`)];\n }\n for (const objectKey in object) {\n const elementSpecKey = objectKey.split('.')[0]; // treat 'paint.*' as 'paint'\n const elementSpec = elementSpecs[elementSpecKey] || elementSpecs['*'];\n let validateElement;\n if (elementValidators[elementSpecKey]) {\n validateElement = elementValidators[elementSpecKey];\n }\n else if (elementSpecs[elementSpecKey]) {\n validateElement = validateSpec;\n }\n else if (elementValidators['*']) {\n validateElement = elementValidators['*'];\n }\n else if (elementSpecs['*']) {\n validateElement = validateSpec;\n }\n else {\n errors.push(new ValidationError(key, object[objectKey], `unknown property \"${objectKey}\"`));\n continue;\n }\n errors = errors.concat(validateElement({\n key: (key ? `${key}.` : key) + objectKey,\n value: object[objectKey],\n valueSpec: elementSpec,\n style,\n styleSpec,\n object,\n objectKey,\n validateSpec,\n }, object));\n }\n for (const elementSpecKey in elementSpecs) {\n // Don't check `required` when there's a custom validator for that property.\n if (elementValidators[elementSpecKey]) {\n continue;\n }\n if (elementSpecs[elementSpecKey].required && elementSpecs[elementSpecKey]['default'] === undefined && object[elementSpecKey] === undefined) {\n errors.push(new ValidationError(key, object, `missing required property \"${elementSpecKey}\"`));\n }\n }\n return errors;\n}\n\nfunction validateArray(options) {\n const array = options.value;\n const arraySpec = options.valueSpec;\n const validateSpec = options.validateSpec;\n const style = options.style;\n const styleSpec = options.styleSpec;\n const key = options.key;\n const validateArrayElement = options.arrayElementValidator || validateSpec;\n if (getType(array) !== 'array') {\n return [new ValidationError(key, array, `array expected, ${getType(array)} found`)];\n }\n if (arraySpec.length && array.length !== arraySpec.length) {\n return [new ValidationError(key, array, `array length ${arraySpec.length} expected, length ${array.length} found`)];\n }\n if (arraySpec['min-length'] && array.length < arraySpec['min-length']) {\n return [new ValidationError(key, array, `array length at least ${arraySpec['min-length']} expected, length ${array.length} found`)];\n }\n let arrayElementSpec = {\n 'type': arraySpec.value,\n 'values': arraySpec.values\n };\n if (styleSpec.$version < 7) {\n arrayElementSpec['function'] = arraySpec.function;\n }\n if (getType(arraySpec.value) === 'object') {\n arrayElementSpec = arraySpec.value;\n }\n let errors = [];\n for (let i = 0; i < array.length; i++) {\n errors = errors.concat(validateArrayElement({\n array,\n arrayIndex: i,\n value: array[i],\n valueSpec: arrayElementSpec,\n validateSpec: options.validateSpec,\n style,\n styleSpec,\n key: `${key}[${i}]`\n }));\n }\n return errors;\n}\n\nfunction validateNumber(options) {\n const key = options.key;\n const value = options.value;\n const valueSpec = options.valueSpec;\n let type = getType(value);\n // eslint-disable-next-line no-self-compare\n if (type === 'number' && value !== value) {\n type = 'NaN';\n }\n if (type !== 'number') {\n return [new ValidationError(key, value, `number expected, ${type} found`)];\n }\n if ('minimum' in valueSpec && value < valueSpec.minimum) {\n return [new ValidationError(key, value, `${value} is less than the minimum value ${valueSpec.minimum}`)];\n }\n if ('maximum' in valueSpec && value > valueSpec.maximum) {\n return [new ValidationError(key, value, `${value} is greater than the maximum value ${valueSpec.maximum}`)];\n }\n return [];\n}\n\nfunction validateFunction(options) {\n const functionValueSpec = options.valueSpec;\n const functionType = unbundle(options.value.type);\n let stopKeyType;\n let stopDomainValues = {};\n let previousStopDomainValue;\n let previousStopDomainZoom;\n const isZoomFunction = functionType !== 'categorical' && options.value.property === undefined;\n const isPropertyFunction = !isZoomFunction;\n const isZoomAndPropertyFunction = getType(options.value.stops) === 'array' &&\n getType(options.value.stops[0]) === 'array' &&\n getType(options.value.stops[0][0]) === 'object';\n const errors = validateObject({\n key: options.key,\n value: options.value,\n valueSpec: options.styleSpec.function,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec,\n objectElementValidators: {\n stops: validateFunctionStops,\n default: validateFunctionDefault\n }\n });\n if (functionType === 'identity' && isZoomFunction) {\n errors.push(new ValidationError(options.key, options.value, 'missing required property \"property\"'));\n }\n if (functionType !== 'identity' && !options.value.stops) {\n errors.push(new ValidationError(options.key, options.value, 'missing required property \"stops\"'));\n }\n if (functionType === 'exponential' && options.valueSpec.expression && !supportsInterpolation(options.valueSpec)) {\n errors.push(new ValidationError(options.key, options.value, 'exponential functions not supported'));\n }\n if (options.styleSpec.$version >= 8) {\n if (isPropertyFunction && !supportsPropertyExpression(options.valueSpec)) {\n errors.push(new ValidationError(options.key, options.value, 'property functions not supported'));\n }\n else if (isZoomFunction && !supportsZoomExpression(options.valueSpec)) {\n errors.push(new ValidationError(options.key, options.value, 'zoom functions not supported'));\n }\n }\n if ((functionType === 'categorical' || isZoomAndPropertyFunction) && options.value.property === undefined) {\n errors.push(new ValidationError(options.key, options.value, '\"property\" property is required'));\n }\n return errors;\n function validateFunctionStops(options) {\n if (functionType === 'identity') {\n return [new ValidationError(options.key, options.value, 'identity function may not have a \"stops\" property')];\n }\n let errors = [];\n const value = options.value;\n errors = errors.concat(validateArray({\n key: options.key,\n value,\n valueSpec: options.valueSpec,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec,\n arrayElementValidator: validateFunctionStop\n }));\n if (getType(value) === 'array' && value.length === 0) {\n errors.push(new ValidationError(options.key, value, 'array must have at least one stop'));\n }\n return errors;\n }\n function validateFunctionStop(options) {\n let errors = [];\n const value = options.value;\n const key = options.key;\n if (getType(value) !== 'array') {\n return [new ValidationError(key, value, `array expected, ${getType(value)} found`)];\n }\n if (value.length !== 2) {\n return [new ValidationError(key, value, `array length 2 expected, length ${value.length} found`)];\n }\n if (isZoomAndPropertyFunction) {\n if (getType(value[0]) !== 'object') {\n return [new ValidationError(key, value, `object expected, ${getType(value[0])} found`)];\n }\n if (value[0].zoom === undefined) {\n return [new ValidationError(key, value, 'object stop key must have zoom')];\n }\n if (value[0].value === undefined) {\n return [new ValidationError(key, value, 'object stop key must have value')];\n }\n if (previousStopDomainZoom && previousStopDomainZoom > unbundle(value[0].zoom)) {\n return [new ValidationError(key, value[0].zoom, 'stop zoom values must appear in ascending order')];\n }\n if (unbundle(value[0].zoom) !== previousStopDomainZoom) {\n previousStopDomainZoom = unbundle(value[0].zoom);\n previousStopDomainValue = undefined;\n stopDomainValues = {};\n }\n errors = errors.concat(validateObject({\n key: `${key}[0]`,\n value: value[0],\n valueSpec: { zoom: {} },\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec,\n objectElementValidators: { zoom: validateNumber, value: validateStopDomainValue }\n }));\n }\n else {\n errors = errors.concat(validateStopDomainValue({\n key: `${key}[0]`,\n value: value[0],\n valueSpec: {},\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec\n }, value));\n }\n if (isExpression(deepUnbundle(value[1]))) {\n return errors.concat([new ValidationError(`${key}[1]`, value[1], 'expressions are not allowed in function stops.')]);\n }\n return errors.concat(options.validateSpec({\n key: `${key}[1]`,\n value: value[1],\n valueSpec: functionValueSpec,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec\n }));\n }\n function validateStopDomainValue(options, stop) {\n const type = getType(options.value);\n const value = unbundle(options.value);\n const reportValue = options.value !== null ? options.value : stop;\n if (!stopKeyType) {\n stopKeyType = type;\n }\n else if (type !== stopKeyType) {\n return [new ValidationError(options.key, reportValue, `${type} stop domain type must match previous stop domain type ${stopKeyType}`)];\n }\n if (type !== 'number' && type !== 'string' && type !== 'boolean') {\n return [new ValidationError(options.key, reportValue, 'stop domain value must be a number, string, or boolean')];\n }\n if (type !== 'number' && functionType !== 'categorical') {\n let message = `number expected, ${type} found`;\n if (supportsPropertyExpression(functionValueSpec) && functionType === undefined) {\n message += '\\nIf you intended to use a categorical function, specify `\"type\": \"categorical\"`.';\n }\n return [new ValidationError(options.key, reportValue, message)];\n }\n if (functionType === 'categorical' && type === 'number' && (!isFinite(value) || Math.floor(value) !== value)) {\n return [new ValidationError(options.key, reportValue, `integer expected, found ${value}`)];\n }\n if (functionType !== 'categorical' && type === 'number' && previousStopDomainValue !== undefined && value < previousStopDomainValue) {\n return [new ValidationError(options.key, reportValue, 'stop domain values must appear in ascending order')];\n }\n else {\n previousStopDomainValue = value;\n }\n if (functionType === 'categorical' && value in stopDomainValues) {\n return [new ValidationError(options.key, reportValue, 'stop domain values must be unique')];\n }\n else {\n stopDomainValues[value] = true;\n }\n return [];\n }\n function validateFunctionDefault(options) {\n return options.validateSpec({\n key: options.key,\n value: options.value,\n valueSpec: functionValueSpec,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec\n });\n }\n}\n\nfunction validateExpression(options) {\n const expression = (options.expressionContext === 'property' ? createPropertyExpression : createExpression)(deepUnbundle(options.value), options.valueSpec);\n if (expression.result === 'error') {\n return expression.value.map((error) => {\n return new ValidationError(`${options.key}${error.key}`, options.value, error.message);\n });\n }\n const expressionObj = expression.value.expression || expression.value._styleExpression.expression;\n if (options.expressionContext === 'property' && (options.propertyKey === 'text-font') &&\n !expressionObj.outputDefined()) {\n return [new ValidationError(options.key, options.value, `Invalid data expression for \"${options.propertyKey}\". Output values must be contained as literals within the expression.`)];\n }\n if (options.expressionContext === 'property' && options.propertyType === 'layout' &&\n (!isStateConstant(expressionObj))) {\n return [new ValidationError(options.key, options.value, '\"feature-state\" data expressions are not supported with layout properties.')];\n }\n if (options.expressionContext === 'filter' && !isStateConstant(expressionObj)) {\n return [new ValidationError(options.key, options.value, '\"feature-state\" data expressions are not supported with filters.')];\n }\n if (options.expressionContext && options.expressionContext.indexOf('cluster') === 0) {\n if (!isGlobalPropertyConstant(expressionObj, ['zoom', 'feature-state'])) {\n return [new ValidationError(options.key, options.value, '\"zoom\" and \"feature-state\" expressions are not supported with cluster properties.')];\n }\n if (options.expressionContext === 'cluster-initial' && !isFeatureConstant(expressionObj)) {\n return [new ValidationError(options.key, options.value, 'Feature data expressions are not supported with initial expression part of cluster properties.')];\n }\n }\n return [];\n}\n\nfunction validateBoolean(options) {\n const value = options.value;\n const key = options.key;\n const type = getType(value);\n if (type !== 'boolean') {\n return [new ValidationError(key, value, `boolean expected, ${type} found`)];\n }\n return [];\n}\n\nfunction validateColor(options) {\n const key = options.key;\n const value = options.value;\n const type = getType(value);\n if (type !== 'string') {\n return [new ValidationError(key, value, `color expected, ${type} found`)];\n }\n if (!Color.parse(String(value))) { // cast String object to string primitive\n return [new ValidationError(key, value, `color expected, \"${value}\" found`)];\n }\n return [];\n}\n\nfunction validateEnum(options) {\n const key = options.key;\n const value = options.value;\n const valueSpec = options.valueSpec;\n const errors = [];\n if (Array.isArray(valueSpec.values)) { // <=v7\n if (valueSpec.values.indexOf(unbundle(value)) === -1) {\n errors.push(new ValidationError(key, value, `expected one of [${valueSpec.values.join(', ')}], ${JSON.stringify(value)} found`));\n }\n }\n else { // >=v8\n if (Object.keys(valueSpec.values).indexOf(unbundle(value)) === -1) {\n errors.push(new ValidationError(key, value, `expected one of [${Object.keys(valueSpec.values).join(', ')}], ${JSON.stringify(value)} found`));\n }\n }\n return errors;\n}\n\nfunction validateFilter(options) {\n if (isExpressionFilter(deepUnbundle(options.value))) {\n return validateExpression(extendBy({}, options, {\n expressionContext: 'filter',\n valueSpec: { value: 'boolean' }\n }));\n }\n else {\n return validateNonExpressionFilter(options);\n }\n}\nfunction validateNonExpressionFilter(options) {\n const value = options.value;\n const key = options.key;\n if (getType(value) !== 'array') {\n return [new ValidationError(key, value, `array expected, ${getType(value)} found`)];\n }\n const styleSpec = options.styleSpec;\n let type;\n let errors = [];\n if (value.length < 1) {\n return [new ValidationError(key, value, 'filter array must have at least 1 element')];\n }\n errors = errors.concat(validateEnum({\n key: `${key}[0]`,\n value: value[0],\n valueSpec: styleSpec.filter_operator,\n style: options.style,\n styleSpec: options.styleSpec\n }));\n switch (unbundle(value[0])) {\n case '<':\n case '<=':\n case '>':\n case '>=':\n if (value.length >= 2 && unbundle(value[1]) === '$type') {\n errors.push(new ValidationError(key, value, `\"$type\" cannot be use with operator \"${value[0]}\"`));\n }\n /* falls through */\n case '==':\n case '!=':\n if (value.length !== 3) {\n errors.push(new ValidationError(key, value, `filter array for operator \"${value[0]}\" must have 3 elements`));\n }\n /* falls through */\n case 'in':\n case '!in':\n if (value.length >= 2) {\n type = getType(value[1]);\n if (type !== 'string') {\n errors.push(new ValidationError(`${key}[1]`, value[1], `string expected, ${type} found`));\n }\n }\n for (let i = 2; i < value.length; i++) {\n type = getType(value[i]);\n if (unbundle(value[1]) === '$type') {\n errors = errors.concat(validateEnum({\n key: `${key}[${i}]`,\n value: value[i],\n valueSpec: styleSpec.geometry_type,\n style: options.style,\n styleSpec: options.styleSpec\n }));\n }\n else if (type !== 'string' && type !== 'number' && type !== 'boolean') {\n errors.push(new ValidationError(`${key}[${i}]`, value[i], `string, number, or boolean expected, ${type} found`));\n }\n }\n break;\n case 'any':\n case 'all':\n case 'none':\n for (let i = 1; i < value.length; i++) {\n errors = errors.concat(validateNonExpressionFilter({\n key: `${key}[${i}]`,\n value: value[i],\n style: options.style,\n styleSpec: options.styleSpec\n }));\n }\n break;\n case 'has':\n case '!has':\n type = getType(value[1]);\n if (value.length !== 2) {\n errors.push(new ValidationError(key, value, `filter array for \"${value[0]}\" operator must have 2 elements`));\n }\n else if (type !== 'string') {\n errors.push(new ValidationError(`${key}[1]`, value[1], `string expected, ${type} found`));\n }\n break;\n case 'within':\n type = getType(value[1]);\n if (value.length !== 2) {\n errors.push(new ValidationError(key, value, `filter array for \"${value[0]}\" operator must have 2 elements`));\n }\n else if (type !== 'object') {\n errors.push(new ValidationError(`${key}[1]`, value[1], `object expected, ${type} found`));\n }\n break;\n }\n return errors;\n}\n\nfunction validateProperty(options, propertyType) {\n const key = options.key;\n const validateSpec = options.validateSpec;\n const style = options.style;\n const styleSpec = options.styleSpec;\n const value = options.value;\n const propertyKey = options.objectKey;\n const layerSpec = styleSpec[`${propertyType}_${options.layerType}`];\n if (!layerSpec)\n return [];\n const transitionMatch = propertyKey.match(/^(.*)-transition$/);\n if (propertyType === 'paint' && transitionMatch && layerSpec[transitionMatch[1]] && layerSpec[transitionMatch[1]].transition) {\n return validateSpec({\n key,\n value,\n valueSpec: styleSpec.transition,\n style,\n styleSpec\n });\n }\n const valueSpec = options.valueSpec || layerSpec[propertyKey];\n if (!valueSpec) {\n return [new ValidationError(key, value, `unknown property \"${propertyKey}\"`)];\n }\n let tokenMatch;\n if (getType(value) === 'string' && supportsPropertyExpression(valueSpec) && !valueSpec.tokens && (tokenMatch = /^{([^}]+)}$/.exec(value))) {\n return [new ValidationError(key, value, `\"${propertyKey}\" does not support interpolation syntax\\n` +\n `Use an identity property function instead: \\`{ \"type\": \"identity\", \"property\": ${JSON.stringify(tokenMatch[1])} }\\`.`)];\n }\n const errors = [];\n if (options.layerType === 'symbol') {\n if (propertyKey === 'text-field' && style && !style.glyphs) {\n errors.push(new ValidationError(key, value, 'use of \"text-field\" requires a style \"glyphs\" property'));\n }\n if (propertyKey === 'text-font' && isFunction(deepUnbundle(value)) && unbundle(value.type) === 'identity') {\n errors.push(new ValidationError(key, value, '\"text-font\" does not support identity functions'));\n }\n }\n return errors.concat(validateSpec({\n key: options.key,\n value,\n valueSpec,\n style,\n styleSpec,\n expressionContext: 'property',\n propertyType,\n propertyKey\n }));\n}\n\nfunction validatePaintProperty(options) {\n return validateProperty(options, 'paint');\n}\n\nfunction validateLayoutProperty(options) {\n return validateProperty(options, 'layout');\n}\n\nfunction validateLayer(options) {\n let errors = [];\n const layer = options.value;\n const key = options.key;\n const style = options.style;\n const styleSpec = options.styleSpec;\n if (!layer.type && !layer.ref) {\n errors.push(new ValidationError(key, layer, 'either \"type\" or \"ref\" is required'));\n }\n let type = unbundle(layer.type);\n const ref = unbundle(layer.ref);\n if (layer.id) {\n const layerId = unbundle(layer.id);\n for (let i = 0; i < options.arrayIndex; i++) {\n const otherLayer = style.layers[i];\n if (unbundle(otherLayer.id) === layerId) {\n errors.push(new ValidationError(key, layer.id, `duplicate layer id \"${layer.id}\", previously used at line ${otherLayer.id.__line__}`));\n }\n }\n }\n if ('ref' in layer) {\n ['type', 'source', 'source-layer', 'filter', 'layout'].forEach((p) => {\n if (p in layer) {\n errors.push(new ValidationError(key, layer[p], `\"${p}\" is prohibited for ref layers`));\n }\n });\n let parent;\n style.layers.forEach((layer) => {\n if (unbundle(layer.id) === ref)\n parent = layer;\n });\n if (!parent) {\n errors.push(new ValidationError(key, layer.ref, `ref layer \"${ref}\" not found`));\n }\n else if (parent.ref) {\n errors.push(new ValidationError(key, layer.ref, 'ref cannot reference another ref layer'));\n }\n else {\n type = unbundle(parent.type);\n }\n }\n else if (type !== 'background') {\n if (!layer.source) {\n errors.push(new ValidationError(key, layer, 'missing required property \"source\"'));\n }\n else {\n const source = style.sources && style.sources[layer.source];\n const sourceType = source && unbundle(source.type);\n if (!source) {\n errors.push(new ValidationError(key, layer.source, `source \"${layer.source}\" not found`));\n }\n else if (sourceType === 'vector' && type === 'raster') {\n errors.push(new ValidationError(key, layer.source, `layer \"${layer.id}\" requires a raster source`));\n }\n else if (sourceType !== 'raster-dem' && type === 'hillshade') {\n errors.push(new ValidationError(key, layer.source, `layer \"${layer.id}\" requires a raster-dem source`));\n }\n else if (sourceType === 'raster' && type !== 'raster') {\n errors.push(new ValidationError(key, layer.source, `layer \"${layer.id}\" requires a vector source`));\n }\n else if (sourceType === 'vector' && !layer['source-layer']) {\n errors.push(new ValidationError(key, layer, `layer \"${layer.id}\" must specify a \"source-layer\"`));\n }\n else if (sourceType === 'raster-dem' && type !== 'hillshade') {\n errors.push(new ValidationError(key, layer.source, 'raster-dem source can only be used with layer type \\'hillshade\\'.'));\n }\n else if (type === 'line' && layer.paint && layer.paint['line-gradient'] &&\n (sourceType !== 'geojson' || !source.lineMetrics)) {\n errors.push(new ValidationError(key, layer, `layer \"${layer.id}\" specifies a line-gradient, which requires a GeoJSON source with \\`lineMetrics\\` enabled.`));\n }\n }\n }\n errors = errors.concat(validateObject({\n key,\n value: layer,\n valueSpec: styleSpec.layer,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n objectElementValidators: {\n '*'() {\n return [];\n },\n // We don't want to enforce the spec's `\"requires\": true` for backward compatibility with refs;\n // the actual requirement is validated above. See https://github.com/mapbox/mapbox-gl-js/issues/5772.\n type() {\n return options.validateSpec({\n key: `${key}.type`,\n value: layer.type,\n valueSpec: styleSpec.layer.type,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n object: layer,\n objectKey: 'type'\n });\n },\n filter: validateFilter,\n layout(options) {\n return validateObject({\n layer,\n key: options.key,\n value: options.value,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n objectElementValidators: {\n '*'(options) {\n return validateLayoutProperty(extendBy({ layerType: type }, options));\n }\n }\n });\n },\n paint(options) {\n return validateObject({\n layer,\n key: options.key,\n value: options.value,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n objectElementValidators: {\n '*'(options) {\n return validatePaintProperty(extendBy({ layerType: type }, options));\n }\n }\n });\n }\n }\n }));\n return errors;\n}\n\nfunction validateString(options) {\n const value = options.value;\n const key = options.key;\n const type = getType(value);\n if (type !== 'string') {\n return [new ValidationError(key, value, `string expected, ${type} found`)];\n }\n return [];\n}\n\nfunction validateRasterDEMSource(options) {\n var _a;\n const sourceName = (_a = options.sourceName) !== null && _a !== void 0 ? _a : '';\n const rasterDEM = options.value;\n const styleSpec = options.styleSpec;\n const rasterDEMSpec = styleSpec.source_raster_dem;\n const style = options.style;\n let errors = [];\n const rootType = getType(rasterDEM);\n if (rasterDEM === undefined) {\n return errors;\n }\n else if (rootType !== 'object') {\n errors.push(new ValidationError('source_raster_dem', rasterDEM, `object expected, ${rootType} found`));\n return errors;\n }\n const encoding = unbundle(rasterDEM.encoding);\n const isCustomEncoding = encoding === 'custom';\n const customEncodingKeys = ['redFactor', 'greenFactor', 'blueFactor', 'baseShift'];\n const encodingName = options.value.encoding ? `\"${options.value.encoding}\"` : 'Default';\n for (const key in rasterDEM) {\n if (!isCustomEncoding && customEncodingKeys.includes(key)) {\n errors.push(new ValidationError(key, rasterDEM[key], `In \"${sourceName}\": \"${key}\" is only valid when \"encoding\" is set to \"custom\". ${encodingName} encoding found`));\n }\n else if (rasterDEMSpec[key]) {\n errors = errors.concat(options.validateSpec({\n key,\n value: rasterDEM[key],\n valueSpec: rasterDEMSpec[key],\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else {\n errors.push(new ValidationError(key, rasterDEM[key], `unknown property \"${key}\"`));\n }\n }\n return errors;\n}\n\nconst objectElementValidators = {\n promoteId: validatePromoteId\n};\nfunction validateSource(options) {\n const value = options.value;\n const key = options.key;\n const styleSpec = options.styleSpec;\n const style = options.style;\n const validateSpec = options.validateSpec;\n if (!value.type) {\n return [new ValidationError(key, value, '\"type\" is required')];\n }\n const type = unbundle(value.type);\n let errors;\n switch (type) {\n case 'vector':\n case 'raster':\n errors = validateObject({\n key,\n value,\n valueSpec: styleSpec[`source_${type.replace('-', '_')}`],\n style: options.style,\n styleSpec,\n objectElementValidators,\n validateSpec,\n });\n return errors;\n case 'raster-dem':\n errors = validateRasterDEMSource({\n sourceName: key,\n value,\n style: options.style,\n styleSpec,\n validateSpec,\n });\n return errors;\n case 'geojson':\n errors = validateObject({\n key,\n value,\n valueSpec: styleSpec.source_geojson,\n style,\n styleSpec,\n validateSpec,\n objectElementValidators\n });\n if (value.cluster) {\n for (const prop in value.clusterProperties) {\n const [operator, mapExpr] = value.clusterProperties[prop];\n const reduceExpr = typeof operator === 'string' ? [operator, ['accumulated'], ['get', prop]] : operator;\n errors.push(...validateExpression({\n key: `${key}.${prop}.map`,\n value: mapExpr,\n validateSpec,\n expressionContext: 'cluster-map'\n }));\n errors.push(...validateExpression({\n key: `${key}.${prop}.reduce`,\n value: reduceExpr,\n validateSpec,\n expressionContext: 'cluster-reduce'\n }));\n }\n }\n return errors;\n case 'video':\n return validateObject({\n key,\n value,\n valueSpec: styleSpec.source_video,\n style,\n validateSpec,\n styleSpec\n });\n case 'image':\n return validateObject({\n key,\n value,\n valueSpec: styleSpec.source_image,\n style,\n validateSpec,\n styleSpec\n });\n case 'canvas':\n return [new ValidationError(key, null, 'Please use runtime APIs to add canvas sources, rather than including them in stylesheets.', 'source.canvas')];\n default:\n return validateEnum({\n key: `${key}.type`,\n value: value.type,\n valueSpec: { values: ['vector', 'raster', 'raster-dem', 'geojson', 'video', 'image'] },\n style,\n validateSpec,\n styleSpec\n });\n }\n}\nfunction validatePromoteId({ key, value }) {\n if (getType(value) === 'string') {\n return validateString({ key, value });\n }\n else {\n const errors = [];\n for (const prop in value) {\n errors.push(...validateString({ key: `${key}.${prop}`, value: value[prop] }));\n }\n return errors;\n }\n}\n\nfunction validateLight(options) {\n const light = options.value;\n const styleSpec = options.styleSpec;\n const lightSpec = styleSpec.light;\n const style = options.style;\n let errors = [];\n const rootType = getType(light);\n if (light === undefined) {\n return errors;\n }\n else if (rootType !== 'object') {\n errors = errors.concat([new ValidationError('light', light, `object expected, ${rootType} found`)]);\n return errors;\n }\n for (const key in light) {\n const transitionMatch = key.match(/^(.*)-transition$/);\n if (transitionMatch && lightSpec[transitionMatch[1]] && lightSpec[transitionMatch[1]].transition) {\n errors = errors.concat(options.validateSpec({\n key,\n value: light[key],\n valueSpec: styleSpec.transition,\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else if (lightSpec[key]) {\n errors = errors.concat(options.validateSpec({\n key,\n value: light[key],\n valueSpec: lightSpec[key],\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else {\n errors = errors.concat([new ValidationError(key, light[key], `unknown property \"${key}\"`)]);\n }\n }\n return errors;\n}\n\nfunction validateTerrain(options) {\n const terrain = options.value;\n const styleSpec = options.styleSpec;\n const terrainSpec = styleSpec.terrain;\n const style = options.style;\n let errors = [];\n const rootType = getType(terrain);\n if (terrain === undefined) {\n return errors;\n }\n else if (rootType !== 'object') {\n errors = errors.concat([new ValidationError('terrain', terrain, `object expected, ${rootType} found`)]);\n return errors;\n }\n for (const key in terrain) {\n if (terrainSpec[key]) {\n errors = errors.concat(options.validateSpec({\n key,\n value: terrain[key],\n valueSpec: terrainSpec[key],\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else {\n errors = errors.concat([new ValidationError(key, terrain[key], `unknown property \"${key}\"`)]);\n }\n }\n return errors;\n}\n\nfunction validateFormatted(options) {\n if (validateString(options).length === 0) {\n return [];\n }\n return validateExpression(options);\n}\n\nfunction validateImage(options) {\n if (validateString(options).length === 0) {\n return [];\n }\n return validateExpression(options);\n}\n\nfunction validatePadding(options) {\n const key = options.key;\n const value = options.value;\n const type = getType(value);\n if (type === 'array') {\n if (value.length < 1 || value.length > 4) {\n return [new ValidationError(key, value, `padding requires 1 to 4 values; ${value.length} values found`)];\n }\n const arrayElementSpec = {\n type: 'number'\n };\n let errors = [];\n for (let i = 0; i < value.length; i++) {\n errors = errors.concat(options.validateSpec({\n key: `${key}[${i}]`,\n value: value[i],\n validateSpec: options.validateSpec,\n valueSpec: arrayElementSpec\n }));\n }\n return errors;\n }\n else {\n return validateNumber({\n key,\n value,\n valueSpec: {}\n });\n }\n}\n\nfunction validateVariableAnchorOffsetCollection(options) {\n const key = options.key;\n const value = options.value;\n const type = getType(value);\n const styleSpec = options.styleSpec;\n if (type !== 'array' || value.length < 1 || value.length % 2 !== 0) {\n return [new ValidationError(key, value, 'variableAnchorOffsetCollection requires a non-empty array of even length')];\n }\n let errors = [];\n for (let i = 0; i < value.length; i += 2) {\n // Elements in even positions should be values from text-anchor enum\n errors = errors.concat(validateEnum({\n key: `${key}[${i}]`,\n value: value[i],\n valueSpec: styleSpec['layout_symbol']['text-anchor']\n }));\n // Elements in odd positions should be points (2-element numeric arrays)\n errors = errors.concat(validateArray({\n key: `${key}[${i + 1}]`,\n value: value[i + 1],\n valueSpec: {\n length: 2,\n value: 'number'\n },\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec\n }));\n }\n return errors;\n}\n\nfunction validateSprite(options) {\n let errors = [];\n const sprite = options.value;\n const key = options.key;\n if (!Array.isArray(sprite)) {\n return validateString({\n key,\n value: sprite\n });\n }\n else {\n const allSpriteIds = [];\n const allSpriteURLs = [];\n for (const i in sprite) {\n if (sprite[i].id && allSpriteIds.includes(sprite[i].id))\n errors.push(new ValidationError(key, sprite, `all the sprites' ids must be unique, but ${sprite[i].id} is duplicated`));\n allSpriteIds.push(sprite[i].id);\n if (sprite[i].url && allSpriteURLs.includes(sprite[i].url))\n errors.push(new ValidationError(key, sprite, `all the sprites' URLs must be unique, but ${sprite[i].url} is duplicated`));\n allSpriteURLs.push(sprite[i].url);\n const pairSpec = {\n id: {\n type: 'string',\n required: true,\n },\n url: {\n type: 'string',\n required: true,\n }\n };\n errors = errors.concat(validateObject({\n key: `${key}[${i}]`,\n value: sprite[i],\n valueSpec: pairSpec,\n validateSpec: options.validateSpec,\n }));\n }\n return errors;\n }\n}\n\nconst VALIDATORS = {\n '*'() {\n return [];\n },\n 'array': validateArray,\n 'boolean': validateBoolean,\n 'number': validateNumber,\n 'color': validateColor,\n 'constants': validateConstants,\n 'enum': validateEnum,\n 'filter': validateFilter,\n 'function': validateFunction,\n 'layer': validateLayer,\n 'object': validateObject,\n 'source': validateSource,\n 'light': validateLight,\n 'terrain': validateTerrain,\n 'string': validateString,\n 'formatted': validateFormatted,\n 'resolvedImage': validateImage,\n 'padding': validatePadding,\n 'variableAnchorOffsetCollection': validateVariableAnchorOffsetCollection,\n 'sprite': validateSprite,\n};\n// Main recursive validation function. Tracks:\n//\n// - key: string representing location of validation in style tree. Used only\n// for more informative error reporting.\n// - value: current value from style being evaluated. May be anything from a\n// high level object that needs to be descended into deeper or a simple\n// scalar value.\n// - valueSpec: current spec being evaluated. Tracks value.\n// - styleSpec: current full spec being evaluated.\nfunction validate(options) {\n const value = options.value;\n const valueSpec = options.valueSpec;\n const styleSpec = options.styleSpec;\n options.validateSpec = validate;\n if (valueSpec.expression && isFunction(unbundle(value))) {\n return validateFunction(options);\n }\n else if (valueSpec.expression && isExpression(deepUnbundle(value))) {\n return validateExpression(options);\n }\n else if (valueSpec.type && VALIDATORS[valueSpec.type]) {\n return VALIDATORS[valueSpec.type](options);\n }\n else {\n const valid = validateObject(extendBy({}, options, {\n valueSpec: valueSpec.type ? styleSpec[valueSpec.type] : valueSpec\n }));\n return valid;\n }\n}\n\nfunction validateGlyphsUrl(options) {\n const value = options.value;\n const key = options.key;\n const errors = validateString(options);\n if (errors.length)\n return errors;\n if (value.indexOf('{fontstack}') === -1) {\n errors.push(new ValidationError(key, value, '\"glyphs\" url must include a \"{fontstack}\" token'));\n }\n if (value.indexOf('{range}') === -1) {\n errors.push(new ValidationError(key, value, '\"glyphs\" url must include a \"{range}\" token'));\n }\n return errors;\n}\n\n/**\n * Validate a MapLibre style against the style specification. This entrypoint,\n * `maplibre-gl-style-spec/lib/validate_style.min`, is designed to produce as\n * small a browserify bundle as possible by omitting unnecessary functionality\n * and legacy style specifications.\n *\n * @private\n * @param {Object} style The style to be validated.\n * @param {Object} [styleSpec] The style specification to validate against.\n * If omitted, the latest style spec is used.\n * @returns {Array}\n * @example\n * var validate = require('maplibre-gl-style-spec/lib/validate_style.min');\n * var errors = validate(style);\n */\nfunction validateStyleMin(style, styleSpec = v8Spec) {\n let errors = [];\n errors = errors.concat(validate({\n key: '',\n value: style,\n valueSpec: styleSpec.$root,\n styleSpec,\n style,\n validateSpec: validate,\n objectElementValidators: {\n glyphs: validateGlyphsUrl,\n '*'() {\n return [];\n }\n }\n }));\n if (style['constants']) {\n errors = errors.concat(validateConstants({\n key: 'constants',\n value: style['constants'],\n style,\n styleSpec,\n validateSpec: validate,\n }));\n }\n return sortErrors(errors);\n}\nvalidateStyleMin.source = wrapCleanErrors(injectValidateSpec(validateSource));\nvalidateStyleMin.sprite = wrapCleanErrors(injectValidateSpec(validateSprite));\nvalidateStyleMin.glyphs = wrapCleanErrors(injectValidateSpec(validateGlyphsUrl));\nvalidateStyleMin.light = wrapCleanErrors(injectValidateSpec(validateLight));\nvalidateStyleMin.terrain = wrapCleanErrors(injectValidateSpec(validateTerrain));\nvalidateStyleMin.layer = wrapCleanErrors(injectValidateSpec(validateLayer));\nvalidateStyleMin.filter = wrapCleanErrors(injectValidateSpec(validateFilter));\nvalidateStyleMin.paintProperty = wrapCleanErrors(injectValidateSpec(validatePaintProperty));\nvalidateStyleMin.layoutProperty = wrapCleanErrors(injectValidateSpec(validateLayoutProperty));\nfunction injectValidateSpec(validator) {\n return function (options) {\n return validator({\n ...options,\n validateSpec: validate,\n });\n };\n}\nfunction sortErrors(errors) {\n return [].concat(errors).sort((a, b) => {\n return a.line - b.line;\n });\n}\nfunction wrapCleanErrors(inner) {\n return function (...args) {\n return sortErrors(inner.apply(this, args));\n };\n}\n\nconst v8 = v8Spec;\nconst expression = {\n StyleExpression,\n StylePropertyFunction,\n ZoomConstantExpression,\n ZoomDependentExpression,\n createExpression,\n createPropertyExpression,\n isExpression,\n isExpressionFilter,\n isZoomExpression,\n normalizePropertyExpression,\n};\nconst styleFunction = {\n convertFunction,\n createFunction,\n isFunction\n};\nconst visit = { eachLayer, eachProperty, eachSource };\n\nexport { Color, ColorType, CompoundExpression, EvaluationContext, FormatExpression, Formatted, FormattedSection, FormattedType, Interpolate, Literal, NullType, Padding, ParsingError, ResolvedImage, Step, StyleExpression, StylePropertyFunction, ValidationError, VariableAnchorOffsetCollection, ZoomConstantExpression, ZoomDependentExpression, convertFilter, convertFunction, createExpression, createFunction, createPropertyExpression, derefLayers, diffStyles as diff, emptyStyle, expression, expressions, createFilter as featureFilter, styleFunction as function, groupByLayout, interpolateFactory, interpolate as interpolates, isExpression, isFunction, isZoomExpression, v8Spec as latest, normalizePropertyExpression, operations, supportsPropertyExpression, toString$1 as toString, typeOf, v8, validateStyleMin, visit };\n//# sourceMappingURL=index.mjs.map\n","import type {Cancelable} from '../types/cancelable';\nimport type {RequestParameters, ResponseCallback} from './ajax';\n\n/**\n * This is a global config object used to store the configuration\n * It is available in the workers as well.\n * Only serializable data should be stored in it.\n */\ntype Config = {\n MAX_PARALLEL_IMAGE_REQUESTS: number;\n MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME: number;\n MAX_TILE_CACHE_ZOOM_LEVELS: number;\n REGISTERED_PROTOCOLS: {[x: string]: (requestParameters: RequestParameters, callback: ResponseCallback) => Cancelable};\n WORKER_URL: string;\n};\n\nexport const config: Config = {\n MAX_PARALLEL_IMAGE_REQUESTS: 16,\n MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME: 8,\n MAX_TILE_CACHE_ZOOM_LEVELS: 5,\n REGISTERED_PROTOCOLS: {},\n WORKER_URL: ''\n};\n","import {extend, warnOnce, isWorker} from './util';\nimport {config} from './config';\n\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\n\n/**\n * A `RequestParameters` object to be returned from Map.options.transformRequest callbacks.\n * @example\n * ```ts\n * // use transformRequest to modify requests that begin with `http://myHost`\n * transformRequest: function(url, resourceType) {\n * if (resourceType === 'Source' && url.indexOf('http://myHost') > -1) {\n * return {\n * url: url.replace('http', 'https'),\n * headers: { 'my-custom-header': true },\n * credentials: 'include' // Include cookies for cross-origin requests\n * }\n * }\n * }\n * ```\n */\nexport type RequestParameters = {\n /**\n * The URL to be requested.\n */\n url: string;\n /**\n * The headers to be sent with the request.\n */\n headers?: any;\n /**\n * Request method `'GET' | 'POST' | 'PUT'`.\n */\n method?: 'GET' | 'POST' | 'PUT';\n /**\n * Request body.\n */\n body?: string;\n /**\n * Response body type to be returned `'string' | 'json' | 'arrayBuffer'`.\n */\n type?: 'string' | 'json' | 'arrayBuffer' | 'image';\n /**\n * `'same-origin'|'include'` Use 'include' to send cookies with cross-origin requests.\n */\n credentials?: 'same-origin' | 'include';\n /**\n * If `true`, Resource Timing API information will be collected for these transformed requests and returned in a resourceTiming property of relevant data events.\n */\n collectResourceTiming?: boolean;\n /**\n * Parameters supported only by browser fetch API. Property of the Request interface contains the cache mode of the request. It controls how the request will interact with the browser's HTTP cache. (https://developer.mozilla.org/en-US/docs/Web/API/Request/cache)\n */\n cache?: RequestCache;\n};\n\n/**\n * The response callback used in various places\n */\nexport type ResponseCallback = (\n error?: Error | null,\n data?: T | null,\n cacheControl?: string | null,\n expires?: string | null\n) => void;\n\n/**\n * An error thrown when a HTTP request results in an error response.\n */\nexport class AJAXError extends Error {\n /**\n * The response's HTTP status code.\n */\n status: number;\n\n /**\n * The response's HTTP status text.\n */\n statusText: string;\n\n /**\n * The request's URL.\n */\n url: string;\n\n /**\n * The response's body.\n */\n body: Blob;\n\n /**\n * @param status - The response's HTTP status code.\n * @param statusText - The response's HTTP status text.\n * @param url - The request's URL.\n * @param body - The response's body.\n */\n constructor(status: number, statusText: string, url: string, body: Blob) {\n super(`AJAXError: ${statusText} (${status}): ${url}`);\n this.status = status;\n this.statusText = statusText;\n this.url = url;\n this.body = body;\n }\n}\n\n// Ensure that we're sending the correct referrer from blob URL worker bundles.\n// For files loaded from the local file system, `location.origin` will be set\n// to the string(!) \"null\" (Firefox), or \"file://\" (Chrome, Safari, Edge, IE),\n// and we will set an empty referrer. Otherwise, we're using the document's URL.\n/* global self */\nexport const getReferrer = isWorker() ?\n () => (self as any).worker && (self as any).worker.referrer :\n () => (window.location.protocol === 'blob:' ? window.parent : window).location.href;\n\nexport const getProtocolAction = url => config.REGISTERED_PROTOCOLS[url.substring(0, url.indexOf('://'))];\n\n// Determines whether a URL is a file:// URL. This is obviously the case if it begins\n// with file://. Relative URLs are also file:// URLs iff the original document was loaded\n// via a file:// URL.\nconst isFileURL = url => /^file:/.test(url) || (/^file:/.test(getReferrer()) && !/^\\w+:/.test(url));\n\nfunction makeFetchRequest(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n const controller = new AbortController();\n const request = new Request(requestParameters.url, {\n method: requestParameters.method || 'GET',\n body: requestParameters.body,\n credentials: requestParameters.credentials,\n headers: requestParameters.headers,\n cache: requestParameters.cache,\n referrer: getReferrer(),\n signal: controller.signal\n });\n let complete = false;\n let aborted = false;\n\n if (requestParameters.type === 'json') {\n request.headers.set('Accept', 'application/json');\n }\n\n const validateOrFetch = (err, cachedResponse?, responseIsFresh?) => {\n if (aborted) return;\n\n if (err) {\n // Do fetch in case of cache error.\n // HTTP pages in Edge trigger a security error that can be ignored.\n if (err.message !== 'SecurityError') {\n warnOnce(err);\n }\n }\n\n if (cachedResponse && responseIsFresh) {\n return finishRequest(cachedResponse);\n }\n\n if (cachedResponse) {\n // We can't do revalidation with 'If-None-Match' because then the\n // request doesn't have simple cors headers.\n }\n\n fetch(request).then(response => {\n if (response.ok) {\n return finishRequest(response);\n\n } else {\n return response.blob().then(body => callback(new AJAXError(response.status, response.statusText, requestParameters.url, body)));\n }\n }).catch(error => {\n if (error.code === 20) {\n // silence expected AbortError\n return;\n }\n callback(new Error(error.message));\n });\n };\n\n const finishRequest = (response) => {\n (\n (requestParameters.type === 'arrayBuffer' || requestParameters.type === 'image') ? response.arrayBuffer() :\n requestParameters.type === 'json' ? response.json() :\n response.text()\n ).then(result => {\n if (aborted) return;\n complete = true;\n callback(null, result, response.headers.get('Cache-Control'), response.headers.get('Expires'));\n }).catch(err => {\n if (!aborted) callback(new Error(err.message));\n });\n };\n\n validateOrFetch(null, null);\n\n return {cancel: () => {\n aborted = true;\n if (!complete) controller.abort();\n }};\n}\n\nfunction makeXMLHttpRequest(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n const xhr: XMLHttpRequest = new XMLHttpRequest();\n\n xhr.open(requestParameters.method || 'GET', requestParameters.url, true);\n if (requestParameters.type === 'arrayBuffer' || requestParameters.type === 'image') {\n xhr.responseType = 'arraybuffer';\n }\n for (const k in requestParameters.headers) {\n xhr.setRequestHeader(k, requestParameters.headers[k]);\n }\n if (requestParameters.type === 'json') {\n xhr.responseType = 'text';\n xhr.setRequestHeader('Accept', 'application/json');\n }\n xhr.withCredentials = requestParameters.credentials === 'include';\n xhr.onerror = () => {\n callback(new Error(xhr.statusText));\n };\n xhr.onload = () => {\n if (((xhr.status >= 200 && xhr.status < 300) || xhr.status === 0) && xhr.response !== null) {\n let data: unknown = xhr.response;\n if (requestParameters.type === 'json') {\n // We're manually parsing JSON here to get better error messages.\n try {\n data = JSON.parse(xhr.response);\n } catch (err) {\n return callback(err);\n }\n }\n callback(null, data, xhr.getResponseHeader('Cache-Control'), xhr.getResponseHeader('Expires'));\n } else {\n const body = new Blob([xhr.response], {type: xhr.getResponseHeader('Content-Type')});\n callback(new AJAXError(xhr.status, xhr.statusText, requestParameters.url, body));\n }\n };\n xhr.send(requestParameters.body);\n return {cancel: () => xhr.abort()};\n}\n\nexport const makeRequest = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n // We're trying to use the Fetch API if possible. However, in some situations we can't use it:\n // - IE11 doesn't support it at all. In this case, we dispatch the request to the main thread so\n // that we can get an accruate referrer header.\n // - Safari exposes window.AbortController, but it doesn't work actually abort any requests in\n // some versions (see https://bugs.webkit.org/show_bug.cgi?id=174980#c2)\n // - Requests for resources with the file:// URI scheme don't work with the Fetch API either. In\n // this case we unconditionally use XHR on the current thread since referrers don't matter.\n if (/:\\/\\//.test(requestParameters.url) && !(/^https?:|^file:/.test(requestParameters.url))) {\n if (isWorker() && (self as any).worker && (self as any).worker.actor) {\n return (self as any).worker.actor.send('getResource', requestParameters, callback);\n }\n if (!isWorker()) {\n const action = getProtocolAction(requestParameters.url) || makeFetchRequest;\n return action(requestParameters, callback);\n }\n }\n if (!isFileURL(requestParameters.url)) {\n if (fetch && Request && AbortController && Object.prototype.hasOwnProperty.call(Request.prototype, 'signal')) {\n return makeFetchRequest(requestParameters, callback);\n }\n if (isWorker() && (self as any).worker && (self as any).worker.actor) {\n const queueOnMainThread = true;\n return (self as any).worker.actor.send('getResource', requestParameters, callback, undefined, queueOnMainThread);\n }\n }\n return makeXMLHttpRequest(requestParameters, callback);\n};\n\nexport const getJSON = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n return makeRequest(extend(requestParameters, {type: 'json'}), callback);\n};\n\nexport const getArrayBuffer = function(\n requestParameters: RequestParameters,\n callback: ResponseCallback\n): Cancelable {\n return makeRequest(extend(requestParameters, {type: 'arrayBuffer'}), callback);\n};\n\nexport const postData = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n return makeRequest(extend(requestParameters, {method: 'POST'}), callback);\n};\n\nexport function sameOrigin(inComingUrl: string) {\n // URL class should be available everywhere\n // https://developer.mozilla.org/en-US/docs/Web/API/URL\n // In addtion, a relative URL \"/foo\" or \"./foo\" will throw exception in its ctor,\n // try-catch is expansive so just use a heuristic check to avoid it\n // also check data URL\n if (!inComingUrl ||\n inComingUrl.indexOf('://') <= 0 || // relative URL\n inComingUrl.indexOf('data:image/') === 0 || // data image URL\n inComingUrl.indexOf('blob:') === 0) { // blob\n return true;\n }\n const urlObj = new URL(inComingUrl);\n const locationObj = window.location;\n return urlObj.protocol === locationObj.protocol && urlObj.host === locationObj.host;\n}\n/**\n * A type used to store the tile's expiration date and cache control definition\n */\nexport type ExpiryData = {cacheControl?: string | null; expires?: Date | string | null};\nexport const getVideo = function(urls: Array, callback: Callback): Cancelable {\n const video: HTMLVideoElement = window.document.createElement('video');\n video.muted = true;\n video.onloadstart = function() {\n callback(null, video);\n };\n for (let i = 0; i < urls.length; i++) {\n const s: HTMLSourceElement = window.document.createElement('source');\n if (!sameOrigin(urls[i])) {\n video.crossOrigin = 'Anonymous';\n }\n s.src = urls[i];\n video.appendChild(s);\n }\n return {cancel: () => {}};\n};\n","import {TransferableGridIndex} from './transferable_grid_index';\nimport {Color, CompoundExpression, expressions, ResolvedImage, StylePropertyFunction,\n StyleExpression, ZoomDependentExpression, ZoomConstantExpression} from '@maplibre/maplibre-gl-style-spec';\n\nimport {AJAXError} from './ajax';\n\nimport type {Transferable} from '../types/transferable';\nimport {isImageBitmap} from './util';\n\ntype SerializedObject = {\n [_: string]: S;\n};\n\nexport type Serialized = null | void | boolean | number | string | Boolean | Number | String | Date | RegExp | ArrayBuffer | ArrayBufferView | ImageData | ImageBitmap | Blob | Array | SerializedObject;\n\ntype Registry = {\n [_: string]: {\n klass: {\n new (...args: any): any;\n deserialize?: (input: Serialized) => unknown;\n };\n omit: ReadonlyArray;\n shallow: ReadonlyArray;\n };\n};\n\n/**\n * Register options\n */\ntype RegisterOptions = {\n /**\n * List of properties to omit from serialization (e.g., cached/computed properties)\n */\n omit?: ReadonlyArray;\n /**\n * List of properties that should be serialized by a simple shallow copy, rather than by a recursive call to serialize().\n */\n shallow?: ReadonlyArray;\n};\n\nconst registry: Registry = {};\n\n/**\n * Register the given class as serializable.\n *\n * @param options - the registration options\n */\nexport function register(\n name: string,\n klass: {\n new (...args: any): T;\n },\n options: RegisterOptions = {}\n) {\n if (registry[name]) throw new Error(`${name} is already registered.`);\n ((Object.defineProperty as any))(klass, '_classRegistryKey', {\n value: name,\n writeable: false\n });\n registry[name] = {\n klass,\n omit: options.omit as ReadonlyArray || [],\n shallow: options.shallow as ReadonlyArray || []\n };\n}\n\nregister('Object', Object);\nregister('TransferableGridIndex', TransferableGridIndex);\n\nregister('Color', Color);\nregister('Error', Error);\nregister('AJAXError', AJAXError);\nregister('ResolvedImage', ResolvedImage);\n\nregister('StylePropertyFunction', StylePropertyFunction);\nregister('StyleExpression', StyleExpression, {omit: ['_evaluator']});\n\nregister('ZoomDependentExpression', ZoomDependentExpression);\nregister('ZoomConstantExpression', ZoomConstantExpression);\nregister('CompoundExpression', CompoundExpression, {omit: ['_evaluate']});\nfor (const name in expressions) {\n if ((expressions[name] as any)._classRegistryKey) continue;\n register(`Expression_${name}`, expressions[name]);\n}\n\nfunction isArrayBuffer(value: any): value is ArrayBuffer {\n return value && typeof ArrayBuffer !== 'undefined' &&\n (value instanceof ArrayBuffer || (value.constructor && value.constructor.name === 'ArrayBuffer'));\n}\n\n/**\n * Serialize the given object for transfer to or from a web worker.\n *\n * For non-builtin types, recursively serialize each property (possibly\n * omitting certain properties - see register()), and package the result along\n * with the constructor's `name` so that the appropriate constructor can be\n * looked up in `deserialize()`.\n *\n * If a `transferables` array is provided, add any transferable objects (i.e.,\n * any ArrayBuffers or ArrayBuffer views) to the list. (If a copy is needed,\n * this should happen in the client code, before using serialize().)\n */\nexport function serialize(input: unknown, transferables?: Array | null): Serialized {\n if (input === null ||\n input === undefined ||\n typeof input === 'boolean' ||\n typeof input === 'number' ||\n typeof input === 'string' ||\n input instanceof Boolean ||\n input instanceof Number ||\n input instanceof String ||\n input instanceof Date ||\n input instanceof RegExp ||\n input instanceof Blob) {\n return input;\n }\n\n if (isArrayBuffer(input)) {\n if (transferables) {\n transferables.push(input);\n }\n return input;\n }\n\n if (isImageBitmap(input)) {\n if (transferables) {\n transferables.push(input);\n }\n return input;\n }\n\n if (ArrayBuffer.isView(input)) {\n const view = input;\n if (transferables) {\n transferables.push(view.buffer);\n }\n return view;\n }\n\n if (input instanceof ImageData) {\n if (transferables) {\n transferables.push(input.data.buffer);\n }\n return input;\n }\n\n if (Array.isArray(input)) {\n const serialized: Array = [];\n for (const item of input) {\n serialized.push(serialize(item, transferables));\n }\n return serialized;\n }\n\n if (typeof input === 'object') {\n const klass = (input.constructor as any);\n const name = klass._classRegistryKey;\n if (!name) {\n throw new Error('can\\'t serialize object of unregistered class');\n }\n if (!registry[name]) throw new Error(`${name} is not registered.`);\n\n const properties: SerializedObject = klass.serialize ?\n // (Temporary workaround) allow a class to provide static\n // `serialize()` and `deserialize()` methods to bypass the generic\n // approach.\n // This temporary workaround lets us use the generic serialization\n // approach for objects whose members include instances of dynamic\n // StructArray types. Once we refactor StructArray to be static,\n // we can remove this complexity.\n (klass.serialize(input, transferables) as SerializedObject) : {};\n\n if (!klass.serialize) {\n for (const key in input) {\n if (!input.hasOwnProperty(key)) continue; // eslint-disable-line no-prototype-builtins\n if (registry[name].omit.indexOf(key) >= 0) continue;\n const property = input[key];\n properties[key] = registry[name].shallow.indexOf(key) >= 0 ?\n property :\n serialize(property, transferables);\n }\n if (input instanceof Error) {\n properties.message = input.message;\n }\n } else {\n if (transferables && properties === transferables[transferables.length - 1]) {\n throw new Error('statically serialized object won\\'t survive transfer of $name property');\n }\n }\n\n if (properties.$name) {\n throw new Error('$name property is reserved for worker serialization logic.');\n }\n if (name !== 'Object') {\n properties.$name = name;\n }\n\n return properties;\n }\n\n throw new Error(`can't serialize object of type ${typeof input}`);\n}\n\nexport function deserialize(input: Serialized): unknown {\n if (input === null ||\n input === undefined ||\n typeof input === 'boolean' ||\n typeof input === 'number' ||\n typeof input === 'string' ||\n input instanceof Boolean ||\n input instanceof Number ||\n input instanceof String ||\n input instanceof Date ||\n input instanceof RegExp ||\n input instanceof Blob ||\n isArrayBuffer(input) ||\n isImageBitmap(input) ||\n ArrayBuffer.isView(input) ||\n input instanceof ImageData) {\n return input;\n }\n\n if (Array.isArray(input)) {\n return input.map(deserialize);\n }\n\n if (typeof input === 'object') {\n const name = input.$name || 'Object';\n if (!registry[name]) {\n throw new Error(`can't deserialize unregistered class ${name}`);\n }\n const {klass} = registry[name];\n if (!klass) {\n throw new Error(`can't deserialize unregistered class ${name}`);\n }\n\n if (klass.deserialize) {\n return klass.deserialize(input);\n }\n\n const result = Object.create(klass.prototype);\n\n for (const key of Object.keys(input)) {\n if (key === '$name') continue;\n const value = (input as SerializedObject)[key];\n result[key] = registry[name].shallow.indexOf(key) >= 0 ? value : deserialize(value);\n }\n\n return result;\n }\n\n throw new Error(`can't deserialize object of type ${typeof input}`);\n}\n","/**\n * Invokes the wrapped function in a non-blocking way when trigger() is called. Invocation requests\n * are ignored until the function was actually invoked.\n */\nexport class ThrottledInvoker {\n _channel: MessageChannel;\n _triggered: boolean;\n _callback: Function;\n\n constructor(callback: Function) {\n this._callback = callback;\n this._triggered = false;\n if (typeof MessageChannel !== 'undefined') {\n this._channel = new MessageChannel();\n this._channel.port2.onmessage = () => {\n this._triggered = false;\n this._callback();\n };\n }\n }\n\n trigger() {\n if (!this._triggered) {\n this._triggered = true;\n if (this._channel) {\n this._channel.port1.postMessage(true);\n } else {\n setTimeout(() => {\n this._triggered = false;\n this._callback();\n }, 0);\n }\n }\n }\n\n remove() {\n delete this._channel;\n this._callback = () => {};\n }\n}\n","import {isWorker} from './util';\nimport {serialize, deserialize, Serialized} from './web_worker_transfer';\nimport {ThrottledInvoker} from './throttled_invoker';\n\nimport type {Transferable} from '../types/transferable';\nimport type {Cancelable} from '../types/cancelable';\nimport type {WorkerSource} from '../source/worker_source';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {Callback} from '../types/callback';\nimport type {StyleGlyph} from '../style/style_glyph';\n\nexport interface ActorTarget {\n addEventListener: typeof window.addEventListener;\n removeEventListener: typeof window.removeEventListener;\n postMessage: typeof window.postMessage;\n terminate?: () => void;\n}\n\nexport interface WorkerSourceProvider {\n getWorkerSource(mapId: string | number, sourceType: string, sourceName: string): WorkerSource;\n}\n\nexport interface GlyphsProvider {\n getGlyphs(mapId: string, params: {\n stacks: {[_: string]: Array};\n source: string;\n tileID: OverscaledTileID;\n type: string;\n },\n callback: Callback<{[_: string]: {[_: number]: StyleGlyph}}>\n );\n}\n\nexport type MessageType = '' | '' |\n'geojson.getClusterExpansionZoom' | 'geojson.getClusterChildren' | 'geojson.getClusterLeaves' | 'geojson.loadData' |\n'removeSource' | 'loadWorkerSource' | 'loadDEMTile' | 'removeDEMTile' |\n'removeTile' | 'reloadTile' | 'abortTile' | 'loadTile' | 'getTile' |\n'getGlyphs' | 'getImages' | 'setImages' |\n'syncRTLPluginState' | 'setReferrer' | 'setLayers' | 'updateLayers';\n\nexport type MessageData = {\n id: string;\n type: MessageType;\n data?: Serialized;\n targetMapId?: string | number | null;\n mustQueue?: boolean;\n error?: Serialized | null;\n hasCallback?: boolean;\n sourceMapId: string | number | null;\n}\n\nexport type Message = {\n data: MessageData;\n}\n\n/**\n * An implementation of the [Actor design pattern](http://en.wikipedia.org/wiki/Actor_model)\n * that maintains the relationship between asynchronous tasks and the objects\n * that spin them off - in this case, tasks like parsing parts of styles,\n * owned by the styles\n */\nexport class Actor {\n target: ActorTarget;\n parent: WorkerSourceProvider | GlyphsProvider;\n mapId: string | number | null;\n callbacks: { [x: number]: Function};\n name: string;\n tasks: { [x: number]: MessageData };\n taskQueue: Array;\n cancelCallbacks: { [x: number]: () => void };\n invoker: ThrottledInvoker;\n globalScope: ActorTarget;\n\n /**\n * @param target - The target\n * @param parent - The parent\n * @param mapId - A unique identifier for the Map instance using this Actor.\n */\n constructor(target: ActorTarget, parent: WorkerSourceProvider | GlyphsProvider, mapId?: string | number) {\n this.target = target;\n this.parent = parent;\n this.mapId = mapId;\n this.callbacks = {};\n this.tasks = {};\n this.taskQueue = [];\n this.cancelCallbacks = {};\n this.invoker = new ThrottledInvoker(this.process);\n this.target.addEventListener('message', this.receive, false);\n this.globalScope = isWorker() ? target : window;\n }\n\n /**\n * Sends a message from a main-thread map to a Worker or from a Worker back to\n * a main-thread map instance.\n *\n * @param type - The name of the target method to invoke or '[source-type].[source-name].name' for a method on a WorkerSource.\n * @param targetMapId - A particular mapId to which to send this message.\n */\n send(\n type: MessageType,\n data: unknown,\n callback?: Function | null,\n targetMapId?: string | null,\n mustQueue: boolean = false\n ): Cancelable {\n // We're using a string ID instead of numbers because they are being used as object keys\n // anyway, and thus stringified implicitly. We use random IDs because an actor may receive\n // message from multiple other actors which could run in different execution context. A\n // linearly increasing ID could produce collisions.\n const id = Math.round((Math.random() * 1e18)).toString(36).substring(0, 10);\n if (callback) {\n this.callbacks[id] = callback;\n }\n const buffers: Array = [];\n const message: MessageData = {\n id,\n type,\n hasCallback: !!callback,\n targetMapId,\n mustQueue,\n sourceMapId: this.mapId,\n data: serialize(data, buffers)\n };\n\n this.target.postMessage(message, {transfer: buffers});\n return {\n cancel: () => {\n if (callback) {\n // Set the callback to null so that it never fires after the request is aborted.\n delete this.callbacks[id];\n }\n const cancelMessage: MessageData = {\n id,\n type: '',\n targetMapId,\n sourceMapId: this.mapId\n };\n this.target.postMessage(cancelMessage);\n }\n };\n }\n\n receive = (message: Message) => {\n const data = message.data;\n const id = data.id;\n\n if (!id) {\n return;\n }\n\n if (data.targetMapId && this.mapId !== data.targetMapId) {\n return;\n }\n\n if (data.type === '') {\n // Remove the original request from the queue. This is only possible if it\n // hasn't been kicked off yet. The id will remain in the queue, but because\n // there is no associated task, it will be dropped once it's time to execute it.\n delete this.tasks[id];\n const cancel = this.cancelCallbacks[id];\n delete this.cancelCallbacks[id];\n if (cancel) {\n cancel();\n }\n } else {\n if (isWorker() || data.mustQueue) {\n // In workers, store the tasks that we need to process before actually processing them. This\n // is necessary because we want to keep receiving messages, and in particular,\n // messages. Some tasks may take a while in the worker thread, so before\n // executing the next task in our queue, postMessage preempts this and \n // messages can be processed. We're using a MessageChannel object to get throttle the\n // process() flow to one at a time.\n this.tasks[id] = data;\n this.taskQueue.push(id);\n this.invoker.trigger();\n } else {\n // In the main thread, process messages immediately so that other work does not slip in\n // between getting partial data back from workers.\n this.processTask(id, data);\n }\n }\n };\n\n process = () => {\n if (!this.taskQueue.length) {\n return;\n }\n const id = this.taskQueue.shift();\n const task = this.tasks[id];\n delete this.tasks[id];\n // Schedule another process call if we know there's more to process _before_ invoking the\n // current task. This is necessary so that processing continues even if the current task\n // doesn't execute successfully.\n if (this.taskQueue.length) {\n this.invoker.trigger();\n }\n if (!task) {\n // If the task ID doesn't have associated task data anymore, it was canceled.\n return;\n }\n\n this.processTask(id, task);\n };\n\n processTask(id: string, task: MessageData) {\n if (task.type === '') {\n // The done() function in the counterpart has been called, and we are now\n // firing the callback in the originating actor, if there is one.\n const callback = this.callbacks[id];\n delete this.callbacks[id];\n if (callback) {\n // If we get a response, but don't have a callback, the request was canceled.\n if (task.error) {\n callback(deserialize(task.error));\n } else {\n callback(null, deserialize(task.data));\n }\n }\n } else {\n let completed = false;\n const buffers: Array = [];\n const done = task.hasCallback ? (err: Error, data?: any) => {\n completed = true;\n delete this.cancelCallbacks[id];\n const responseMessage: MessageData = {\n id,\n type: '',\n sourceMapId: this.mapId,\n error: err ? serialize(err) : null,\n data: serialize(data, buffers)\n };\n this.target.postMessage(responseMessage, {transfer: buffers});\n } : (_) => {\n completed = true;\n };\n\n let callback: Cancelable = null;\n const params = deserialize(task.data);\n if (this.parent[task.type]) {\n // task.type == 'loadTile', 'removeTile', etc.\n callback = this.parent[task.type](task.sourceMapId, params, done);\n } else if ('getWorkerSource' in this.parent) {\n // task.type == sourcetype.method\n const keys = task.type.split('.');\n const scope = this.parent.getWorkerSource(task.sourceMapId, keys[0], (params as any).source);\n callback = scope[keys[1]](params, done);\n } else {\n // No function was found.\n done(new Error(`Could not find function ${task.type}`));\n }\n\n if (!completed && callback && callback.cancel) {\n // Allows canceling the task as long as it hasn't been completed yet.\n this.cancelCallbacks[id] = callback.cancel;\n }\n }\n }\n\n remove() {\n this.invoker.remove();\n this.target.removeEventListener('message', this.receive, false);\n }\n}\n","import {extend} from './util';\n\n/**\n * A listener method used as a callback to events\n */\nexport type Listener = (a: any) => any;\n\ntype Listeners = {[_: string]: Array};\n\nfunction _addEventListener(type: string, listener: Listener, listenerList: Listeners) {\n const listenerExists = listenerList[type] && listenerList[type].indexOf(listener) !== -1;\n if (!listenerExists) {\n listenerList[type] = listenerList[type] || [];\n listenerList[type].push(listener);\n }\n}\n\nfunction _removeEventListener(type: string, listener: Listener, listenerList: Listeners) {\n if (listenerList && listenerList[type]) {\n const index = listenerList[type].indexOf(listener);\n if (index !== -1) {\n listenerList[type].splice(index, 1);\n }\n }\n}\n\n/**\n * The event class\n */\nexport class Event {\n readonly type: string;\n\n constructor(type: string, data: any = {}) {\n extend(this, data);\n this.type = type;\n }\n}\n\ninterface ErrorLike {\n message: string;\n}\n\n/**\n * An error event\n */\nexport class ErrorEvent extends Event {\n error: ErrorLike;\n\n constructor(error: ErrorLike, data: any = {}) {\n super('error', extend({error}, data));\n }\n}\n\n/**\n * Methods mixed in to other classes for event capabilities.\n *\n * @group Event Related\n */\nexport class Evented {\n _listeners: Listeners;\n _oneTimeListeners: Listeners;\n _eventedParent: Evented;\n _eventedParentData: any | (() => any);\n\n /**\n * Adds a listener to a specified event type.\n *\n * @param type - The event type to add a listen for.\n * @param listener - The function to be called when the event is fired.\n * The listener function is called with the data object passed to `fire`,\n * extended with `target` and `type` properties.\n * @returns `this`\n */\n on(type: string, listener: Listener): this {\n this._listeners = this._listeners || {};\n _addEventListener(type, listener, this._listeners);\n\n return this;\n }\n\n /**\n * Removes a previously registered event listener.\n *\n * @param type - The event type to remove listeners for.\n * @param listener - The listener function to remove.\n * @returns `this`\n */\n off(type: string, listener: Listener) {\n _removeEventListener(type, listener, this._listeners);\n _removeEventListener(type, listener, this._oneTimeListeners);\n\n return this;\n }\n\n /**\n * Adds a listener that will be called only once to a specified event type.\n *\n * The listener will be called first time the event fires after the listener is registered.\n *\n * @param type - The event type to listen for.\n * @param listener - The function to be called when the event is fired the first time.\n * @returns `this` or a promise if a listener is not provided\n */\n once(type: string, listener?: Listener): this | Promise {\n if (!listener) {\n return new Promise((resolve) => this.once(type, resolve));\n }\n this._oneTimeListeners = this._oneTimeListeners || {};\n _addEventListener(type, listener, this._oneTimeListeners);\n\n return this;\n }\n\n fire(event: Event | string, properties?: any) {\n // Compatibility with (type: string, properties: Object) signature from previous versions.\n // See https://github.com/mapbox/mapbox-gl-js/issues/6522,\n // https://github.com/mapbox/mapbox-gl-draw/issues/766\n if (typeof event === 'string') {\n event = new Event(event, properties || {});\n }\n\n const type = event.type;\n\n if (this.listens(type)) {\n (event as any).target = this;\n\n // make sure adding or removing listeners inside other listeners won't cause an infinite loop\n const listeners = this._listeners && this._listeners[type] ? this._listeners[type].slice() : [];\n for (const listener of listeners) {\n listener.call(this, event);\n }\n\n const oneTimeListeners = this._oneTimeListeners && this._oneTimeListeners[type] ? this._oneTimeListeners[type].slice() : [];\n for (const listener of oneTimeListeners) {\n _removeEventListener(type, listener, this._oneTimeListeners);\n listener.call(this, event);\n }\n\n const parent = this._eventedParent;\n if (parent) {\n extend(\n event,\n typeof this._eventedParentData === 'function' ? this._eventedParentData() : this._eventedParentData\n );\n parent.fire(event);\n }\n\n // To ensure that no error events are dropped, print them to the\n // console if they have no listeners.\n } else if (event instanceof ErrorEvent) {\n console.error(event.error);\n }\n\n return this;\n }\n\n /**\n * Returns a true if this instance of Evented or any forwardeed instances of Evented have a listener for the specified type.\n *\n * @param type - The event type\n * @returns `true` if there is at least one registered listener for specified event type, `false` otherwise\n */\n listens(type: string): boolean {\n return (\n (this._listeners && this._listeners[type] && this._listeners[type].length > 0) ||\n (this._oneTimeListeners && this._oneTimeListeners[type] && this._oneTimeListeners[type].length > 0) ||\n (this._eventedParent && this._eventedParent.listens(type))\n );\n }\n\n /**\n * Bubble all events fired by this instance of Evented to this parent instance of Evented.\n * @returns `this`\n */\n setEventedParent(parent?: Evented | null, data?: any | (() => any)) {\n this._eventedParent = parent;\n this._eventedParentData = data;\n\n return this;\n }\n}\n","import {validateStyleMin} from '@maplibre/maplibre-gl-style-spec';\nimport {ErrorEvent} from '../util/evented';\n\nimport type {Evented} from '../util/evented';\n\ntype ValidationError = {\n message: string;\n line: number;\n identifier?: string;\n};\n\nexport type Validator = (a: any) => ReadonlyArray;\n\ntype ValidateStyle = {\n source: Validator;\n sprite: Validator;\n glyphs: Validator;\n layer: Validator;\n light: Validator;\n terrain: Validator;\n filter: Validator;\n paintProperty: Validator;\n layoutProperty: Validator;\n (b: any, a?: any | null): ReadonlyArray;\n};\n\nexport const validateStyle = (validateStyleMin as unknown as ValidateStyle);\n\nexport const validateSource = validateStyle.source;\nexport const validateLight = validateStyle.light;\nexport const validateTerrain = validateStyle.terrain;\nexport const validateFilter = validateStyle.filter;\nexport const validatePaintProperty = validateStyle.paintProperty;\nexport const validateLayoutProperty = validateStyle.layoutProperty;\n\nexport function emitValidationErrors(\n emitter: Evented,\n errors?: ReadonlyArray<{\n message: string;\n identifier?: string;\n }> | null\n): boolean {\n let hasErrors = false;\n if (errors && errors.length) {\n for (const error of errors) {\n emitter.fire(new ErrorEvent(new Error(error.message)));\n hasErrors = true;\n }\n }\n return hasErrors;\n}\n","export class ZoomHistory {\n lastZoom: number;\n lastFloorZoom: number;\n lastIntegerZoom: number;\n lastIntegerZoomTime: number;\n first: boolean;\n\n constructor() {\n this.first = true;\n }\n\n update(z: number, now: number) {\n const floorZ = Math.floor(z);\n\n if (this.first) {\n this.first = false;\n this.lastIntegerZoom = floorZ;\n this.lastIntegerZoomTime = 0;\n this.lastZoom = z;\n this.lastFloorZoom = floorZ;\n return true;\n }\n\n if (this.lastFloorZoom > floorZ) {\n this.lastIntegerZoom = floorZ + 1;\n this.lastIntegerZoomTime = now;\n } else if (this.lastFloorZoom < floorZ) {\n this.lastIntegerZoom = floorZ;\n this.lastIntegerZoomTime = now;\n }\n\n if (z !== this.lastZoom) {\n this.lastZoom = z;\n this.lastFloorZoom = floorZ;\n return true;\n }\n\n return false;\n }\n}\n","// The following table comes from .\n// Keep it synchronized with .\n\ntype UnicodeBlockLookup = {[key: string]: (char: number) => boolean};\n\nexport const unicodeBlockLookup: UnicodeBlockLookup = {\n // 'Basic Latin': (char) => char >= 0x0000 && char <= 0x007F,\n 'Latin-1 Supplement': (char) => char >= 0x0080 && char <= 0x00FF,\n // 'Latin Extended-A': (char) => char >= 0x0100 && char <= 0x017F,\n // 'Latin Extended-B': (char) => char >= 0x0180 && char <= 0x024F,\n // 'IPA Extensions': (char) => char >= 0x0250 && char <= 0x02AF,\n // 'Spacing Modifier Letters': (char) => char >= 0x02B0 && char <= 0x02FF,\n // 'Combining Diacritical Marks': (char) => char >= 0x0300 && char <= 0x036F,\n // 'Greek and Coptic': (char) => char >= 0x0370 && char <= 0x03FF,\n // 'Cyrillic': (char) => char >= 0x0400 && char <= 0x04FF,\n // 'Cyrillic Supplement': (char) => char >= 0x0500 && char <= 0x052F,\n // 'Armenian': (char) => char >= 0x0530 && char <= 0x058F,\n //'Hebrew': (char) => char >= 0x0590 && char <= 0x05FF,\n 'Arabic': (char) => char >= 0x0600 && char <= 0x06FF,\n //'Syriac': (char) => char >= 0x0700 && char <= 0x074F,\n 'Arabic Supplement': (char) => char >= 0x0750 && char <= 0x077F,\n // 'Thaana': (char) => char >= 0x0780 && char <= 0x07BF,\n // 'NKo': (char) => char >= 0x07C0 && char <= 0x07FF,\n // 'Samaritan': (char) => char >= 0x0800 && char <= 0x083F,\n // 'Mandaic': (char) => char >= 0x0840 && char <= 0x085F,\n // 'Syriac Supplement': (char) => char >= 0x0860 && char <= 0x086F,\n 'Arabic Extended-A': (char) => char >= 0x08A0 && char <= 0x08FF,\n // 'Devanagari': (char) => char >= 0x0900 && char <= 0x097F,\n // 'Bengali': (char) => char >= 0x0980 && char <= 0x09FF,\n // 'Gurmukhi': (char) => char >= 0x0A00 && char <= 0x0A7F,\n // 'Gujarati': (char) => char >= 0x0A80 && char <= 0x0AFF,\n // 'Oriya': (char) => char >= 0x0B00 && char <= 0x0B7F,\n // 'Tamil': (char) => char >= 0x0B80 && char <= 0x0BFF,\n // 'Telugu': (char) => char >= 0x0C00 && char <= 0x0C7F,\n // 'Kannada': (char) => char >= 0x0C80 && char <= 0x0CFF,\n // 'Malayalam': (char) => char >= 0x0D00 && char <= 0x0D7F,\n // 'Sinhala': (char) => char >= 0x0D80 && char <= 0x0DFF,\n // 'Thai': (char) => char >= 0x0E00 && char <= 0x0E7F,\n // 'Lao': (char) => char >= 0x0E80 && char <= 0x0EFF,\n // 'Tibetan': (char) => char >= 0x0F00 && char <= 0x0FFF,\n // 'Myanmar': (char) => char >= 0x1000 && char <= 0x109F,\n // 'Georgian': (char) => char >= 0x10A0 && char <= 0x10FF,\n 'Hangul Jamo': (char) => char >= 0x1100 && char <= 0x11FF,\n // 'Ethiopic': (char) => char >= 0x1200 && char <= 0x137F,\n // 'Ethiopic Supplement': (char) => char >= 0x1380 && char <= 0x139F,\n // 'Cherokee': (char) => char >= 0x13A0 && char <= 0x13FF,\n 'Unified Canadian Aboriginal Syllabics': (char) => char >= 0x1400 && char <= 0x167F,\n // 'Ogham': (char) => char >= 0x1680 && char <= 0x169F,\n // 'Runic': (char) => char >= 0x16A0 && char <= 0x16FF,\n // 'Tagalog': (char) => char >= 0x1700 && char <= 0x171F,\n // 'Hanunoo': (char) => char >= 0x1720 && char <= 0x173F,\n // 'Buhid': (char) => char >= 0x1740 && char <= 0x175F,\n // 'Tagbanwa': (char) => char >= 0x1760 && char <= 0x177F,\n 'Khmer': (char) => char >= 0x1780 && char <= 0x17FF,\n // 'Mongolian': (char) => char >= 0x1800 && char <= 0x18AF,\n 'Unified Canadian Aboriginal Syllabics Extended': (char) => char >= 0x18B0 && char <= 0x18FF,\n // 'Limbu': (char) => char >= 0x1900 && char <= 0x194F,\n // 'Tai Le': (char) => char >= 0x1950 && char <= 0x197F,\n // 'New Tai Lue': (char) => char >= 0x1980 && char <= 0x19DF,\n // 'Khmer Symbols': (char) => char >= 0x19E0 && char <= 0x19FF,\n // 'Buginese': (char) => char >= 0x1A00 && char <= 0x1A1F,\n // 'Tai Tham': (char) => char >= 0x1A20 && char <= 0x1AAF,\n // 'Combining Diacritical Marks Extended': (char) => char >= 0x1AB0 && char <= 0x1AFF,\n // 'Balinese': (char) => char >= 0x1B00 && char <= 0x1B7F,\n // 'Sundanese': (char) => char >= 0x1B80 && char <= 0x1BBF,\n // 'Batak': (char) => char >= 0x1BC0 && char <= 0x1BFF,\n // 'Lepcha': (char) => char >= 0x1C00 && char <= 0x1C4F,\n // 'Ol Chiki': (char) => char >= 0x1C50 && char <= 0x1C7F,\n // 'Cyrillic Extended-C': (char) => char >= 0x1C80 && char <= 0x1C8F,\n // 'Georgian Extended': (char) => char >= 0x1C90 && char <= 0x1CBF,\n // 'Sundanese Supplement': (char) => char >= 0x1CC0 && char <= 0x1CCF,\n // 'Vedic Extensions': (char) => char >= 0x1CD0 && char <= 0x1CFF,\n // 'Phonetic Extensions': (char) => char >= 0x1D00 && char <= 0x1D7F,\n // 'Phonetic Extensions Supplement': (char) => char >= 0x1D80 && char <= 0x1DBF,\n // 'Combining Diacritical Marks Supplement': (char) => char >= 0x1DC0 && char <= 0x1DFF,\n // 'Latin Extended Additional': (char) => char >= 0x1E00 && char <= 0x1EFF,\n // 'Greek Extended': (char) => char >= 0x1F00 && char <= 0x1FFF,\n 'General Punctuation': (char) => char >= 0x2000 && char <= 0x206F,\n // 'Superscripts and Subscripts': (char) => char >= 0x2070 && char <= 0x209F,\n // 'Currency Symbols': (char) => char >= 0x20A0 && char <= 0x20CF,\n // 'Combining Diacritical Marks for Symbols': (char) => char >= 0x20D0 && char <= 0x20FF,\n 'Letterlike Symbols': (char) => char >= 0x2100 && char <= 0x214F,\n 'Number Forms': (char) => char >= 0x2150 && char <= 0x218F,\n // 'Arrows': (char) => char >= 0x2190 && char <= 0x21FF,\n // 'Mathematical Operators': (char) => char >= 0x2200 && char <= 0x22FF,\n 'Miscellaneous Technical': (char) => char >= 0x2300 && char <= 0x23FF,\n 'Control Pictures': (char) => char >= 0x2400 && char <= 0x243F,\n 'Optical Character Recognition': (char) => char >= 0x2440 && char <= 0x245F,\n 'Enclosed Alphanumerics': (char) => char >= 0x2460 && char <= 0x24FF,\n // 'Box Drawing': (char) => char >= 0x2500 && char <= 0x257F,\n // 'Block Elements': (char) => char >= 0x2580 && char <= 0x259F,\n 'Geometric Shapes': (char) => char >= 0x25A0 && char <= 0x25FF,\n 'Miscellaneous Symbols': (char) => char >= 0x2600 && char <= 0x26FF,\n // 'Dingbats': (char) => char >= 0x2700 && char <= 0x27BF,\n // 'Miscellaneous Mathematical Symbols-A': (char) => char >= 0x27C0 && char <= 0x27EF,\n // 'Supplemental Arrows-A': (char) => char >= 0x27F0 && char <= 0x27FF,\n // 'Braille Patterns': (char) => char >= 0x2800 && char <= 0x28FF,\n // 'Supplemental Arrows-B': (char) => char >= 0x2900 && char <= 0x297F,\n // 'Miscellaneous Mathematical Symbols-B': (char) => char >= 0x2980 && char <= 0x29FF,\n // 'Supplemental Mathematical Operators': (char) => char >= 0x2A00 && char <= 0x2AFF,\n 'Miscellaneous Symbols and Arrows': (char) => char >= 0x2B00 && char <= 0x2BFF,\n // 'Glagolitic': (char) => char >= 0x2C00 && char <= 0x2C5F,\n // 'Latin Extended-C': (char) => char >= 0x2C60 && char <= 0x2C7F,\n // 'Coptic': (char) => char >= 0x2C80 && char <= 0x2CFF,\n // 'Georgian Supplement': (char) => char >= 0x2D00 && char <= 0x2D2F,\n // 'Tifinagh': (char) => char >= 0x2D30 && char <= 0x2D7F,\n // 'Ethiopic Extended': (char) => char >= 0x2D80 && char <= 0x2DDF,\n // 'Cyrillic Extended-A': (char) => char >= 0x2DE0 && char <= 0x2DFF,\n // 'Supplemental Punctuation': (char) => char >= 0x2E00 && char <= 0x2E7F,\n 'CJK Radicals Supplement': (char) => char >= 0x2E80 && char <= 0x2EFF,\n 'Kangxi Radicals': (char) => char >= 0x2F00 && char <= 0x2FDF,\n 'Ideographic Description Characters': (char) => char >= 0x2FF0 && char <= 0x2FFF,\n 'CJK Symbols and Punctuation': (char) => char >= 0x3000 && char <= 0x303F,\n 'Hiragana': (char) => char >= 0x3040 && char <= 0x309F,\n 'Katakana': (char) => char >= 0x30A0 && char <= 0x30FF,\n 'Bopomofo': (char) => char >= 0x3100 && char <= 0x312F,\n 'Hangul Compatibility Jamo': (char) => char >= 0x3130 && char <= 0x318F,\n 'Kanbun': (char) => char >= 0x3190 && char <= 0x319F,\n 'Bopomofo Extended': (char) => char >= 0x31A0 && char <= 0x31BF,\n 'CJK Strokes': (char) => char >= 0x31C0 && char <= 0x31EF,\n 'Katakana Phonetic Extensions': (char) => char >= 0x31F0 && char <= 0x31FF,\n 'Enclosed CJK Letters and Months': (char) => char >= 0x3200 && char <= 0x32FF,\n 'CJK Compatibility': (char) => char >= 0x3300 && char <= 0x33FF,\n 'CJK Unified Ideographs Extension A': (char) => char >= 0x3400 && char <= 0x4DBF,\n 'Yijing Hexagram Symbols': (char) => char >= 0x4DC0 && char <= 0x4DFF,\n 'CJK Unified Ideographs': (char) => char >= 0x4E00 && char <= 0x9FFF,\n 'Yi Syllables': (char) => char >= 0xA000 && char <= 0xA48F,\n 'Yi Radicals': (char) => char >= 0xA490 && char <= 0xA4CF,\n // 'Lisu': (char) => char >= 0xA4D0 && char <= 0xA4FF,\n // 'Vai': (char) => char >= 0xA500 && char <= 0xA63F,\n // 'Cyrillic Extended-B': (char) => char >= 0xA640 && char <= 0xA69F,\n // 'Bamum': (char) => char >= 0xA6A0 && char <= 0xA6FF,\n // 'Modifier Tone Letters': (char) => char >= 0xA700 && char <= 0xA71F,\n // 'Latin Extended-D': (char) => char >= 0xA720 && char <= 0xA7FF,\n // 'Syloti Nagri': (char) => char >= 0xA800 && char <= 0xA82F,\n // 'Common Indic Number Forms': (char) => char >= 0xA830 && char <= 0xA83F,\n // 'Phags-pa': (char) => char >= 0xA840 && char <= 0xA87F,\n // 'Saurashtra': (char) => char >= 0xA880 && char <= 0xA8DF,\n // 'Devanagari Extended': (char) => char >= 0xA8E0 && char <= 0xA8FF,\n // 'Kayah Li': (char) => char >= 0xA900 && char <= 0xA92F,\n // 'Rejang': (char) => char >= 0xA930 && char <= 0xA95F,\n 'Hangul Jamo Extended-A': (char) => char >= 0xA960 && char <= 0xA97F,\n // 'Javanese': (char) => char >= 0xA980 && char <= 0xA9DF,\n // 'Myanmar Extended-B': (char) => char >= 0xA9E0 && char <= 0xA9FF,\n // 'Cham': (char) => char >= 0xAA00 && char <= 0xAA5F,\n // 'Myanmar Extended-A': (char) => char >= 0xAA60 && char <= 0xAA7F,\n // 'Tai Viet': (char) => char >= 0xAA80 && char <= 0xAADF,\n // 'Meetei Mayek Extensions': (char) => char >= 0xAAE0 && char <= 0xAAFF,\n // 'Ethiopic Extended-A': (char) => char >= 0xAB00 && char <= 0xAB2F,\n // 'Latin Extended-E': (char) => char >= 0xAB30 && char <= 0xAB6F,\n // 'Cherokee Supplement': (char) => char >= 0xAB70 && char <= 0xABBF,\n // 'Meetei Mayek': (char) => char >= 0xABC0 && char <= 0xABFF,\n 'Hangul Syllables': (char) => char >= 0xAC00 && char <= 0xD7AF,\n 'Hangul Jamo Extended-B': (char) => char >= 0xD7B0 && char <= 0xD7FF,\n // 'High Surrogates': (char) => char >= 0xD800 && char <= 0xDB7F,\n // 'High Private Use Surrogates': (char) => char >= 0xDB80 && char <= 0xDBFF,\n // 'Low Surrogates': (char) => char >= 0xDC00 && char <= 0xDFFF,\n 'Private Use Area': (char) => char >= 0xE000 && char <= 0xF8FF,\n 'CJK Compatibility Ideographs': (char) => char >= 0xF900 && char <= 0xFAFF,\n // 'Alphabetic Presentation Forms': (char) => char >= 0xFB00 && char <= 0xFB4F,\n 'Arabic Presentation Forms-A': (char) => char >= 0xFB50 && char <= 0xFDFF,\n // 'Variation Selectors': (char) => char >= 0xFE00 && char <= 0xFE0F,\n 'Vertical Forms': (char) => char >= 0xFE10 && char <= 0xFE1F,\n // 'Combining Half Marks': (char) => char >= 0xFE20 && char <= 0xFE2F,\n 'CJK Compatibility Forms': (char) => char >= 0xFE30 && char <= 0xFE4F,\n 'Small Form Variants': (char) => char >= 0xFE50 && char <= 0xFE6F,\n 'Arabic Presentation Forms-B': (char) => char >= 0xFE70 && char <= 0xFEFF,\n 'Halfwidth and Fullwidth Forms': (char) => char >= 0xFF00 && char <= 0xFFEF\n // 'Specials': (char) => char >= 0xFFF0 && char <= 0xFFFF,\n // 'Linear B Syllabary': (char) => char >= 0x10000 && char <= 0x1007F,\n // 'Linear B Ideograms': (char) => char >= 0x10080 && char <= 0x100FF,\n // 'Aegean Numbers': (char) => char >= 0x10100 && char <= 0x1013F,\n // 'Ancient Greek Numbers': (char) => char >= 0x10140 && char <= 0x1018F,\n // 'Ancient Symbols': (char) => char >= 0x10190 && char <= 0x101CF,\n // 'Phaistos Disc': (char) => char >= 0x101D0 && char <= 0x101FF,\n // 'Lycian': (char) => char >= 0x10280 && char <= 0x1029F,\n // 'Carian': (char) => char >= 0x102A0 && char <= 0x102DF,\n // 'Coptic Epact Numbers': (char) => char >= 0x102E0 && char <= 0x102FF,\n // 'Old Italic': (char) => char >= 0x10300 && char <= 0x1032F,\n // 'Gothic': (char) => char >= 0x10330 && char <= 0x1034F,\n // 'Old Permic': (char) => char >= 0x10350 && char <= 0x1037F,\n // 'Ugaritic': (char) => char >= 0x10380 && char <= 0x1039F,\n // 'Old Persian': (char) => char >= 0x103A0 && char <= 0x103DF,\n // 'Deseret': (char) => char >= 0x10400 && char <= 0x1044F,\n // 'Shavian': (char) => char >= 0x10450 && char <= 0x1047F,\n // 'Osmanya': (char) => char >= 0x10480 && char <= 0x104AF,\n // 'Osage': (char) => char >= 0x104B0 && char <= 0x104FF,\n // 'Elbasan': (char) => char >= 0x10500 && char <= 0x1052F,\n // 'Caucasian Albanian': (char) => char >= 0x10530 && char <= 0x1056F,\n // 'Linear A': (char) => char >= 0x10600 && char <= 0x1077F,\n // 'Cypriot Syllabary': (char) => char >= 0x10800 && char <= 0x1083F,\n // 'Imperial Aramaic': (char) => char >= 0x10840 && char <= 0x1085F,\n // 'Palmyrene': (char) => char >= 0x10860 && char <= 0x1087F,\n // 'Nabataean': (char) => char >= 0x10880 && char <= 0x108AF,\n // 'Hatran': (char) => char >= 0x108E0 && char <= 0x108FF,\n // 'Phoenician': (char) => char >= 0x10900 && char <= 0x1091F,\n // 'Lydian': (char) => char >= 0x10920 && char <= 0x1093F,\n // 'Meroitic Hieroglyphs': (char) => char >= 0x10980 && char <= 0x1099F,\n // 'Meroitic Cursive': (char) => char >= 0x109A0 && char <= 0x109FF,\n // 'Kharoshthi': (char) => char >= 0x10A00 && char <= 0x10A5F,\n // 'Old South Arabian': (char) => char >= 0x10A60 && char <= 0x10A7F,\n // 'Old North Arabian': (char) => char >= 0x10A80 && char <= 0x10A9F,\n // 'Manichaean': (char) => char >= 0x10AC0 && char <= 0x10AFF,\n // 'Avestan': (char) => char >= 0x10B00 && char <= 0x10B3F,\n // 'Inscriptional Parthian': (char) => char >= 0x10B40 && char <= 0x10B5F,\n // 'Inscriptional Pahlavi': (char) => char >= 0x10B60 && char <= 0x10B7F,\n // 'Psalter Pahlavi': (char) => char >= 0x10B80 && char <= 0x10BAF,\n // 'Old Turkic': (char) => char >= 0x10C00 && char <= 0x10C4F,\n // 'Old Hungarian': (char) => char >= 0x10C80 && char <= 0x10CFF,\n // 'Hanifi Rohingya': (char) => char >= 0x10D00 && char <= 0x10D3F,\n // 'Rumi Numeral Symbols': (char) => char >= 0x10E60 && char <= 0x10E7F,\n // 'Old Sogdian': (char) => char >= 0x10F00 && char <= 0x10F2F,\n // 'Sogdian': (char) => char >= 0x10F30 && char <= 0x10F6F,\n // 'Elymaic': (char) => char >= 0x10FE0 && char <= 0x10FFF,\n // 'Brahmi': (char) => char >= 0x11000 && char <= 0x1107F,\n // 'Kaithi': (char) => char >= 0x11080 && char <= 0x110CF,\n // 'Sora Sompeng': (char) => char >= 0x110D0 && char <= 0x110FF,\n // 'Chakma': (char) => char >= 0x11100 && char <= 0x1114F,\n // 'Mahajani': (char) => char >= 0x11150 && char <= 0x1117F,\n // 'Sharada': (char) => char >= 0x11180 && char <= 0x111DF,\n // 'Sinhala Archaic Numbers': (char) => char >= 0x111E0 && char <= 0x111FF,\n // 'Khojki': (char) => char >= 0x11200 && char <= 0x1124F,\n // 'Multani': (char) => char >= 0x11280 && char <= 0x112AF,\n // 'Khudawadi': (char) => char >= 0x112B0 && char <= 0x112FF,\n // 'Grantha': (char) => char >= 0x11300 && char <= 0x1137F,\n // 'Newa': (char) => char >= 0x11400 && char <= 0x1147F,\n // 'Tirhuta': (char) => char >= 0x11480 && char <= 0x114DF,\n // 'Siddham': (char) => char >= 0x11580 && char <= 0x115FF,\n // 'Modi': (char) => char >= 0x11600 && char <= 0x1165F,\n // 'Mongolian Supplement': (char) => char >= 0x11660 && char <= 0x1167F,\n // 'Takri': (char) => char >= 0x11680 && char <= 0x116CF,\n // 'Ahom': (char) => char >= 0x11700 && char <= 0x1173F,\n // 'Dogra': (char) => char >= 0x11800 && char <= 0x1184F,\n // 'Warang Citi': (char) => char >= 0x118A0 && char <= 0x118FF,\n // 'Nandinagari': (char) => char >= 0x119A0 && char <= 0x119FF,\n // 'Zanabazar Square': (char) => char >= 0x11A00 && char <= 0x11A4F,\n // 'Soyombo': (char) => char >= 0x11A50 && char <= 0x11AAF,\n // 'Pau Cin Hau': (char) => char >= 0x11AC0 && char <= 0x11AFF,\n // 'Bhaiksuki': (char) => char >= 0x11C00 && char <= 0x11C6F,\n // 'Marchen': (char) => char >= 0x11C70 && char <= 0x11CBF,\n // 'Masaram Gondi': (char) => char >= 0x11D00 && char <= 0x11D5F,\n // 'Gunjala Gondi': (char) => char >= 0x11D60 && char <= 0x11DAF,\n // 'Makasar': (char) => char >= 0x11EE0 && char <= 0x11EFF,\n // 'Tamil Supplement': (char) => char >= 0x11FC0 && char <= 0x11FFF,\n // 'Cuneiform': (char) => char >= 0x12000 && char <= 0x123FF,\n // 'Cuneiform Numbers and Punctuation': (char) => char >= 0x12400 && char <= 0x1247F,\n // 'Early Dynastic Cuneiform': (char) => char >= 0x12480 && char <= 0x1254F,\n // 'Egyptian Hieroglyphs': (char) => char >= 0x13000 && char <= 0x1342F,\n // 'Egyptian Hieroglyph Format Controls': (char) => char >= 0x13430 && char <= 0x1343F,\n // 'Anatolian Hieroglyphs': (char) => char >= 0x14400 && char <= 0x1467F,\n // 'Bamum Supplement': (char) => char >= 0x16800 && char <= 0x16A3F,\n // 'Mro': (char) => char >= 0x16A40 && char <= 0x16A6F,\n // 'Bassa Vah': (char) => char >= 0x16AD0 && char <= 0x16AFF,\n // 'Pahawh Hmong': (char) => char >= 0x16B00 && char <= 0x16B8F,\n // 'Medefaidrin': (char) => char >= 0x16E40 && char <= 0x16E9F,\n // 'Miao': (char) => char >= 0x16F00 && char <= 0x16F9F,\n // 'Ideographic Symbols and Punctuation': (char) => char >= 0x16FE0 && char <= 0x16FFF,\n // 'Tangut': (char) => char >= 0x17000 && char <= 0x187FF,\n // 'Tangut Components': (char) => char >= 0x18800 && char <= 0x18AFF,\n // 'Kana Supplement': (char) => char >= 0x1B000 && char <= 0x1B0FF,\n // 'Kana Extended-A': (char) => char >= 0x1B100 && char <= 0x1B12F,\n // 'Small Kana Extension': (char) => char >= 0x1B130 && char <= 0x1B16F,\n // 'Nushu': (char) => char >= 0x1B170 && char <= 0x1B2FF,\n // 'Duployan': (char) => char >= 0x1BC00 && char <= 0x1BC9F,\n // 'Shorthand Format Controls': (char) => char >= 0x1BCA0 && char <= 0x1BCAF,\n // 'Byzantine Musical Symbols': (char) => char >= 0x1D000 && char <= 0x1D0FF,\n // 'Musical Symbols': (char) => char >= 0x1D100 && char <= 0x1D1FF,\n // 'Ancient Greek Musical Notation': (char) => char >= 0x1D200 && char <= 0x1D24F,\n // 'Mayan Numerals': (char) => char >= 0x1D2E0 && char <= 0x1D2FF,\n // 'Tai Xuan Jing Symbols': (char) => char >= 0x1D300 && char <= 0x1D35F,\n // 'Counting Rod Numerals': (char) => char >= 0x1D360 && char <= 0x1D37F,\n // 'Mathematical Alphanumeric Symbols': (char) => char >= 0x1D400 && char <= 0x1D7FF,\n // 'Sutton SignWriting': (char) => char >= 0x1D800 && char <= 0x1DAAF,\n // 'Glagolitic Supplement': (char) => char >= 0x1E000 && char <= 0x1E02F,\n // 'Nyiakeng Puachue Hmong': (char) => char >= 0x1E100 && char <= 0x1E14F,\n // 'Wancho': (char) => char >= 0x1E2C0 && char <= 0x1E2FF,\n // 'Mende Kikakui': (char) => char >= 0x1E800 && char <= 0x1E8DF,\n // 'Adlam': (char) => char >= 0x1E900 && char <= 0x1E95F,\n // 'Indic Siyaq Numbers': (char) => char >= 0x1EC70 && char <= 0x1ECBF,\n // 'Ottoman Siyaq Numbers': (char) => char >= 0x1ED00 && char <= 0x1ED4F,\n // 'Arabic Mathematical Alphabetic Symbols': (char) => char >= 0x1EE00 && char <= 0x1EEFF,\n // 'Mahjong Tiles': (char) => char >= 0x1F000 && char <= 0x1F02F,\n // 'Domino Tiles': (char) => char >= 0x1F030 && char <= 0x1F09F,\n // 'Playing Cards': (char) => char >= 0x1F0A0 && char <= 0x1F0FF,\n // 'Enclosed Alphanumeric Supplement': (char) => char >= 0x1F100 && char <= 0x1F1FF,\n // 'Enclosed Ideographic Supplement': (char) => char >= 0x1F200 && char <= 0x1F2FF,\n // 'Miscellaneous Symbols and Pictographs': (char) => char >= 0x1F300 && char <= 0x1F5FF,\n // 'Emoticons': (char) => char >= 0x1F600 && char <= 0x1F64F,\n // 'Ornamental Dingbats': (char) => char >= 0x1F650 && char <= 0x1F67F,\n // 'Transport and Map Symbols': (char) => char >= 0x1F680 && char <= 0x1F6FF,\n // 'Alchemical Symbols': (char) => char >= 0x1F700 && char <= 0x1F77F,\n // 'Geometric Shapes Extended': (char) => char >= 0x1F780 && char <= 0x1F7FF,\n // 'Supplemental Arrows-C': (char) => char >= 0x1F800 && char <= 0x1F8FF,\n // 'Supplemental Symbols and Pictographs': (char) => char >= 0x1F900 && char <= 0x1F9FF,\n // 'Chess Symbols': (char) => char >= 0x1FA00 && char <= 0x1FA6F,\n // 'Symbols and Pictographs Extended-A': (char) => char >= 0x1FA70 && char <= 0x1FAFF,\n // 'CJK Unified Ideographs Extension B': (char) => char >= 0x20000 && char <= 0x2A6DF,\n // 'CJK Unified Ideographs Extension C': (char) => char >= 0x2A700 && char <= 0x2B73F,\n // 'CJK Unified Ideographs Extension D': (char) => char >= 0x2B740 && char <= 0x2B81F,\n // 'CJK Unified Ideographs Extension E': (char) => char >= 0x2B820 && char <= 0x2CEAF,\n // 'CJK Unified Ideographs Extension F': (char) => char >= 0x2CEB0 && char <= 0x2EBEF,\n // 'CJK Compatibility Ideographs Supplement': (char) => char >= 0x2F800 && char <= 0x2FA1F,\n // 'Tags': (char) => char >= 0xE0000 && char <= 0xE007F,\n // 'Variation Selectors Supplement': (char) => char >= 0xE0100 && char <= 0xE01EF,\n // 'Supplementary Private Use Area-A': (char) => char >= 0xF0000 && char <= 0xFFFFF,\n // 'Supplementary Private Use Area-B': (char) => char >= 0x100000 && char <= 0x10FFFF,\n};\n","/* eslint-disable new-cap */\n\nimport {unicodeBlockLookup as isChar} from './is_char_in_unicode_block';\n\nexport function allowsIdeographicBreaking(chars: string) {\n for (const char of chars) {\n if (!charAllowsIdeographicBreaking(char.charCodeAt(0))) return false;\n }\n return true;\n}\n\nexport function allowsVerticalWritingMode(chars: string) {\n for (const char of chars) {\n if (charHasUprightVerticalOrientation(char.charCodeAt(0))) return true;\n }\n return false;\n}\n\nexport function allowsLetterSpacing(chars: string) {\n for (const char of chars) {\n if (!charAllowsLetterSpacing(char.charCodeAt(0))) return false;\n }\n return true;\n}\n\nexport function charAllowsLetterSpacing(char: number) {\n if (isChar['Arabic'](char)) return false;\n if (isChar['Arabic Supplement'](char)) return false;\n if (isChar['Arabic Extended-A'](char)) return false;\n if (isChar['Arabic Presentation Forms-A'](char)) return false;\n if (isChar['Arabic Presentation Forms-B'](char)) return false;\n\n return true;\n}\n\nexport function charAllowsIdeographicBreaking(char: number) {\n // Return early for characters outside all ideographic ranges.\n if (char < 0x2E80) return false;\n\n if (isChar['Bopomofo Extended'](char)) return true;\n if (isChar['Bopomofo'](char)) return true;\n if (isChar['CJK Compatibility Forms'](char)) return true;\n if (isChar['CJK Compatibility Ideographs'](char)) return true;\n if (isChar['CJK Compatibility'](char)) return true;\n if (isChar['CJK Radicals Supplement'](char)) return true;\n if (isChar['CJK Strokes'](char)) return true;\n if (isChar['CJK Symbols and Punctuation'](char)) return true;\n if (isChar['CJK Unified Ideographs Extension A'](char)) return true;\n if (isChar['CJK Unified Ideographs'](char)) return true;\n if (isChar['Enclosed CJK Letters and Months'](char)) return true;\n if (isChar['Halfwidth and Fullwidth Forms'](char)) return true;\n if (isChar['Hiragana'](char)) return true;\n if (isChar['Ideographic Description Characters'](char)) return true;\n if (isChar['Kangxi Radicals'](char)) return true;\n if (isChar['Katakana Phonetic Extensions'](char)) return true;\n if (isChar['Katakana'](char)) return true;\n if (isChar['Vertical Forms'](char)) return true;\n if (isChar['Yi Radicals'](char)) return true;\n if (isChar['Yi Syllables'](char)) return true;\n\n return false;\n}\n\n// The following logic comes from\n// .\n// Keep it synchronized with\n// .\n// The data file denotes with “U” or “Tu” any codepoint that may be drawn\n// upright in vertical text but does not distinguish between upright and\n// “neutral” characters.\n\n// Blocks in the Unicode supplementary planes are excluded from this module due\n// to .\n\n/**\n * Returns true if the given Unicode codepoint identifies a character with\n * upright orientation.\n *\n * A character has upright orientation if it is drawn upright (unrotated)\n * whether the line is oriented horizontally or vertically, even if both\n * adjacent characters can be rotated. For example, a Chinese character is\n * always drawn upright. An uprightly oriented character causes an adjacent\n * “neutral” character to be drawn upright as well.\n */\nexport function charHasUprightVerticalOrientation(char: number) {\n if (char === 0x02EA /* modifier letter yin departing tone mark */ ||\n char === 0x02EB /* modifier letter yang departing tone mark */) {\n return true;\n }\n\n // Return early for characters outside all ranges whose characters remain\n // upright in vertical writing mode.\n if (char < 0x1100) return false;\n\n if (isChar['Bopomofo Extended'](char)) return true;\n if (isChar['Bopomofo'](char)) return true;\n if (isChar['CJK Compatibility Forms'](char)) {\n if (!((char >= 0xFE49 /* dashed overline */ && char <= 0xFE4F) /* wavy low line */)) {\n return true;\n }\n }\n if (isChar['CJK Compatibility Ideographs'](char)) return true;\n if (isChar['CJK Compatibility'](char)) return true;\n if (isChar['CJK Radicals Supplement'](char)) return true;\n if (isChar['CJK Strokes'](char)) return true;\n if (isChar['CJK Symbols and Punctuation'](char)) {\n if (!((char >= 0x3008 /* left angle bracket */ && char <= 0x3011) /* right black lenticular bracket */) &&\n !((char >= 0x3014 /* left tortoise shell bracket */ && char <= 0x301F) /* low double prime quotation mark */) &&\n char !== 0x3030 /* wavy dash */) {\n return true;\n }\n }\n if (isChar['CJK Unified Ideographs Extension A'](char)) return true;\n if (isChar['CJK Unified Ideographs'](char)) return true;\n if (isChar['Enclosed CJK Letters and Months'](char)) return true;\n if (isChar['Hangul Compatibility Jamo'](char)) return true;\n if (isChar['Hangul Jamo Extended-A'](char)) return true;\n if (isChar['Hangul Jamo Extended-B'](char)) return true;\n if (isChar['Hangul Jamo'](char)) return true;\n if (isChar['Hangul Syllables'](char)) return true;\n if (isChar['Hiragana'](char)) return true;\n if (isChar['Ideographic Description Characters'](char)) return true;\n if (isChar['Kanbun'](char)) return true;\n if (isChar['Kangxi Radicals'](char)) return true;\n if (isChar['Katakana Phonetic Extensions'](char)) return true;\n if (isChar['Katakana'](char)) {\n if (char !== 0x30FC /* katakana-hiragana prolonged sound mark */) {\n return true;\n }\n }\n if (isChar['Halfwidth and Fullwidth Forms'](char)) {\n if (char !== 0xFF08 /* fullwidth left parenthesis */ &&\n char !== 0xFF09 /* fullwidth right parenthesis */ &&\n char !== 0xFF0D /* fullwidth hyphen-minus */ &&\n !((char >= 0xFF1A /* fullwidth colon */ && char <= 0xFF1E) /* fullwidth greater-than sign */) &&\n char !== 0xFF3B /* fullwidth left square bracket */ &&\n char !== 0xFF3D /* fullwidth right square bracket */ &&\n char !== 0xFF3F /* fullwidth low line */ &&\n !(char >= 0xFF5B /* fullwidth left curly bracket */ && char <= 0xFFDF) &&\n char !== 0xFFE3 /* fullwidth macron */ &&\n !(char >= 0xFFE8 /* halfwidth forms light vertical */ && char <= 0xFFEF)) {\n return true;\n }\n }\n if (isChar['Small Form Variants'](char)) {\n if (!((char >= 0xFE58 /* small em dash */ && char <= 0xFE5E) /* small right tortoise shell bracket */) &&\n !((char >= 0xFE63 /* small hyphen-minus */ && char <= 0xFE66) /* small equals sign */)) {\n return true;\n }\n }\n if (isChar['Unified Canadian Aboriginal Syllabics'](char)) return true;\n if (isChar['Unified Canadian Aboriginal Syllabics Extended'](char)) return true;\n if (isChar['Vertical Forms'](char)) return true;\n if (isChar['Yijing Hexagram Symbols'](char)) return true;\n if (isChar['Yi Syllables'](char)) return true;\n if (isChar['Yi Radicals'](char)) return true;\n\n return false;\n}\n\n/**\n * Returns true if the given Unicode codepoint identifies a character with\n * neutral orientation.\n *\n * A character has neutral orientation if it may be drawn rotated or unrotated\n * when the line is oriented vertically, depending on the orientation of the\n * adjacent characters. For example, along a verticlly oriented line, the vulgar\n * fraction ½ is drawn upright among Chinese characters but rotated among Latin\n * letters. A neutrally oriented character does not influence whether an\n * adjacent character is drawn upright or rotated.\n */\nexport function charHasNeutralVerticalOrientation(char: number) {\n if (isChar['Latin-1 Supplement'](char)) {\n if (char === 0x00A7 /* section sign */ ||\n char === 0x00A9 /* copyright sign */ ||\n char === 0x00AE /* registered sign */ ||\n char === 0x00B1 /* plus-minus sign */ ||\n char === 0x00BC /* vulgar fraction one quarter */ ||\n char === 0x00BD /* vulgar fraction one half */ ||\n char === 0x00BE /* vulgar fraction three quarters */ ||\n char === 0x00D7 /* multiplication sign */ ||\n char === 0x00F7 /* division sign */) {\n return true;\n }\n }\n if (isChar['General Punctuation'](char)) {\n if (char === 0x2016 /* double vertical line */ ||\n char === 0x2020 /* dagger */ ||\n char === 0x2021 /* double dagger */ ||\n char === 0x2030 /* per mille sign */ ||\n char === 0x2031 /* per ten thousand sign */ ||\n char === 0x203B /* reference mark */ ||\n char === 0x203C /* double exclamation mark */ ||\n char === 0x2042 /* asterism */ ||\n char === 0x2047 /* double question mark */ ||\n char === 0x2048 /* question exclamation mark */ ||\n char === 0x2049 /* exclamation question mark */ ||\n char === 0x2051 /* two asterisks aligned vertically */) {\n return true;\n }\n }\n if (isChar['Letterlike Symbols'](char)) return true;\n if (isChar['Number Forms'](char)) return true;\n if (isChar['Miscellaneous Technical'](char)) {\n if ((char >= 0x2300 /* diameter sign */ && char <= 0x2307 /* wavy line */) ||\n (char >= 0x230C /* bottom right crop */ && char <= 0x231F /* bottom right corner */) ||\n (char >= 0x2324 /* up arrowhead between two horizontal bars */ && char <= 0x2328 /* keyboard */) ||\n char === 0x232B /* erase to the left */ ||\n (char >= 0x237D /* shouldered open box */ && char <= 0x239A /* clear screen symbol */) ||\n (char >= 0x23BE /* dentistry symbol light vertical and top right */ && char <= 0x23CD /* square foot */) ||\n char === 0x23CF /* eject symbol */ ||\n (char >= 0x23D1 /* metrical breve */ && char <= 0x23DB /* fuse */) ||\n (char >= 0x23E2 /* white trapezium */ && char <= 0x23FF)) {\n return true;\n }\n }\n if (isChar['Control Pictures'](char) && char !== 0x2423 /* open box */) return true;\n if (isChar['Optical Character Recognition'](char)) return true;\n if (isChar['Enclosed Alphanumerics'](char)) return true;\n if (isChar['Geometric Shapes'](char)) return true;\n if (isChar['Miscellaneous Symbols'](char)) {\n if (!((char >= 0x261A /* black left pointing index */ && char <= 0x261F) /* white down pointing index */)) {\n return true;\n }\n }\n if (isChar['Miscellaneous Symbols and Arrows'](char)) {\n if ((char >= 0x2B12 /* square with top half black */ && char <= 0x2B2F /* white vertical ellipse */) ||\n (char >= 0x2B50 /* white medium star */ && char <= 0x2B59 /* heavy circled saltire */) ||\n (char >= 0x2BB8 /* upwards white arrow from bar with horizontal bar */ && char <= 0x2BEB)) {\n return true;\n }\n }\n if (isChar['CJK Symbols and Punctuation'](char)) return true;\n if (isChar['Katakana'](char)) return true;\n if (isChar['Private Use Area'](char)) return true;\n if (isChar['CJK Compatibility Forms'](char)) return true;\n if (isChar['Small Form Variants'](char)) return true;\n if (isChar['Halfwidth and Fullwidth Forms'](char)) return true;\n\n if (char === 0x221E /* infinity */ ||\n char === 0x2234 /* therefore */ ||\n char === 0x2235 /* because */ ||\n (char >= 0x2700 /* black safety scissors */ && char <= 0x2767 /* rotated floral heart bullet */) ||\n (char >= 0x2776 /* dingbat negative circled digit one */ && char <= 0x2793 /* dingbat negative circled sans-serif number ten */) ||\n char === 0xFFFC /* object replacement character */ ||\n char === 0xFFFD /* replacement character */) {\n return true;\n }\n\n return false;\n}\n\n/**\n * Returns true if the given Unicode codepoint identifies a character with\n * rotated orientation.\n *\n * A character has rotated orientation if it is drawn rotated when the line is\n * oriented vertically, even if both adjacent characters are upright. For\n * example, a Latin letter is drawn rotated along a vertical line. A rotated\n * character causes an adjacent “neutral” character to be drawn rotated as well.\n */\nexport function charHasRotatedVerticalOrientation(char: number) {\n return !(charHasUprightVerticalOrientation(char) ||\n charHasNeutralVerticalOrientation(char));\n}\n\nexport function charInComplexShapingScript(char: number) {\n return isChar['Arabic'](char) ||\n isChar['Arabic Supplement'](char) ||\n isChar['Arabic Extended-A'](char) ||\n isChar['Arabic Presentation Forms-A'](char) ||\n isChar['Arabic Presentation Forms-B'](char);\n}\n\nexport function charInRTLScript(char: number) {\n // Main blocks for Hebrew, Arabic, Thaana and other RTL scripts\n return (char >= 0x0590 && char <= 0x08FF) ||\n isChar['Arabic Presentation Forms-A'](char) ||\n isChar['Arabic Presentation Forms-B'](char);\n}\n\nexport function charInSupportedScript(char: number, canRenderRTL: boolean) {\n // This is a rough heuristic: whether we \"can render\" a script\n // actually depends on the properties of the font being used\n // and whether differences from the ideal rendering are considered\n // semantically significant.\n\n // Even in Latin script, we \"can't render\" combinations such as the fi\n // ligature, but we don't consider that semantically significant.\n if (!canRenderRTL && charInRTLScript(char)) {\n return false;\n }\n if ((char >= 0x0900 && char <= 0x0DFF) ||\n // Main blocks for Indic scripts and Sinhala\n (char >= 0x0F00 && char <= 0x109F) ||\n // Main blocks for Tibetan and Myanmar\n isChar['Khmer'](char)) {\n // These blocks cover common scripts that require\n // complex text shaping, based on unicode script metadata:\n // http://www.unicode.org/repos/cldr/trunk/common/properties/scriptMetadata.txt\n // where \"Web Rank <= 32\" \"Shaping Required = YES\"\n return false;\n }\n return true;\n}\n\nexport function stringContainsRTLText(chars: string): boolean {\n for (const char of chars) {\n if (charInRTLScript(char.charCodeAt(0))) {\n return true;\n }\n }\n return false;\n}\n\nexport function isStringInSupportedScript(chars: string, canRenderRTL: boolean) {\n for (const char of chars) {\n if (!charInSupportedScript(char.charCodeAt(0), canRenderRTL)) {\n return false;\n }\n }\n return true;\n}\n","import type {Cancelable} from '../types/cancelable';\n\nconst now = typeof performance !== 'undefined' && performance && performance.now ?\n performance.now.bind(performance) :\n Date.now.bind(Date);\n\nlet linkEl;\n\nlet reducedMotionQuery: MediaQueryList;\n\n/** */\nexport const browser = {\n /**\n * Provides a function that outputs milliseconds: either performance.now()\n * or a fallback to Date.now()\n */\n now,\n\n frame(fn: (paintStartTimestamp: number) => void): Cancelable {\n const frame = requestAnimationFrame(fn);\n return {cancel: () => cancelAnimationFrame(frame)};\n },\n\n getImageData(img: HTMLImageElement | ImageBitmap, padding: number = 0): ImageData {\n const context = this.getImageCanvasContext(img);\n return context.getImageData(-padding, -padding, img.width as number + 2 * padding, img.height as number + 2 * padding);\n },\n\n getImageCanvasContext(img: HTMLImageElement | ImageBitmap): CanvasRenderingContext2D {\n const canvas = window.document.createElement('canvas') as HTMLCanvasElement;\n const context = canvas.getContext('2d', {willReadFrequently: true});\n if (!context) {\n throw new Error('failed to create canvas 2d context');\n }\n canvas.width = img.width as number;\n canvas.height = img.height as number;\n context.drawImage(img, 0, 0, img.width as number, img.height as number);\n return context;\n },\n\n resolveURL(path: string) {\n if (!linkEl) linkEl = document.createElement('a');\n linkEl.href = path;\n return linkEl.href;\n },\n\n hardwareConcurrency: typeof navigator !== 'undefined' && navigator.hardwareConcurrency || 4,\n\n get prefersReducedMotion(): boolean {\n // In case your test crashes when checking matchMedia, call setMatchMedia from 'src/util/test/util'\n if (!matchMedia) return false;\n //Lazily initialize media query\n if (reducedMotionQuery == null) {\n reducedMotionQuery = matchMedia('(prefers-reduced-motion: reduce)');\n }\n return reducedMotionQuery.matches;\n },\n};\n","import {getArrayBuffer} from '../util/ajax';\nimport {browser} from '../util/browser';\nimport {Event, Evented} from '../util/evented';\nimport {isWorker} from '../util/util';\n\nconst status = {\n unavailable: 'unavailable', // Not loaded\n deferred: 'deferred', // The plugin URL has been specified, but loading has been deferred\n loading: 'loading', // request in-flight\n loaded: 'loaded',\n error: 'error'\n};\n\nexport type PluginState = {\n pluginStatus: typeof status[keyof typeof status];\n pluginURL: string;\n};\n\n/**\n * An error callback\n */\ntype ErrorCallback = (error?: Error | null) => void;\ntype PluginStateSyncCallback = (state: PluginState) => void;\nlet _completionCallback = null;\n\n//Variables defining the current state of the plugin\nlet pluginStatus = status.unavailable;\nlet pluginURL = null;\n\nexport const triggerPluginCompletionEvent = function(error: Error | string) {\n // NetworkError's are not correctly reflected by the plugin status which prevents reloading plugin\n if (error && typeof error === 'string' && error.indexOf('NetworkError') > -1) {\n pluginStatus = status.error;\n }\n\n if (_completionCallback) {\n _completionCallback(error);\n }\n};\n\nfunction sendPluginStateToWorker() {\n evented.fire(new Event('pluginStateChange', {pluginStatus, pluginURL}));\n}\n\nexport const evented = new Evented();\n\nexport const getRTLTextPluginStatus = function () {\n return pluginStatus;\n};\n\nexport const registerForPluginStateChange = function(callback: PluginStateSyncCallback) {\n // Do an initial sync of the state\n callback({pluginStatus, pluginURL});\n // Listen for all future state changes\n evented.on('pluginStateChange', callback);\n return callback;\n};\n\nexport const clearRTLTextPlugin = function() {\n pluginStatus = status.unavailable;\n pluginURL = null;\n _completionCallback = null;\n};\n\nexport const setRTLTextPlugin = function(url: string, callback: ErrorCallback, deferred: boolean = false) {\n if (pluginStatus === status.deferred || pluginStatus === status.loading || pluginStatus === status.loaded) {\n throw new Error('setRTLTextPlugin cannot be called multiple times.');\n }\n pluginURL = browser.resolveURL(url);\n pluginStatus = status.deferred;\n _completionCallback = callback;\n sendPluginStateToWorker();\n\n //Start downloading the plugin immediately if not intending to lazy-load\n if (!deferred) {\n downloadRTLTextPlugin();\n }\n};\n\nexport const downloadRTLTextPlugin = function() {\n if (pluginStatus !== status.deferred || !pluginURL) {\n throw new Error('rtl-text-plugin cannot be downloaded unless a pluginURL is specified');\n }\n pluginStatus = status.loading;\n sendPluginStateToWorker();\n if (pluginURL) {\n getArrayBuffer({url: pluginURL}, (error) => {\n if (error) {\n triggerPluginCompletionEvent(error);\n } else {\n pluginStatus = status.loaded;\n sendPluginStateToWorker();\n }\n });\n }\n};\n\nexport const plugin: {\n applyArabicShaping: Function;\n processBidirectionalText: ((b: string, a: Array) => Array);\n processStyledBidirectionalText: ((c: string, b: Array, a: Array) => Array<[string, Array]>);\n isLoaded: () => boolean;\n isLoading: () => boolean;\n setState: (state: PluginState) => void;\n isParsed: () => boolean;\n getPluginURL: () => string;\n} = {\n applyArabicShaping: null,\n processBidirectionalText: null,\n processStyledBidirectionalText: null,\n isLoaded() {\n return pluginStatus === status.loaded || // Main Thread: loaded if the completion callback returned successfully\n plugin.applyArabicShaping != null; // Web-worker: loaded if the plugin functions have been compiled\n },\n isLoading() { // Main Thread Only: query the loading status, this function does not return the correct value in the worker context.\n return pluginStatus === status.loading;\n },\n setState(state: PluginState) { // Worker thread only: this tells the worker threads that the plugin is available on the Main thread\n if (!isWorker()) throw new Error('Cannot set the state of the rtl-text-plugin when not in the web-worker context');\n\n pluginStatus = state.pluginStatus;\n pluginURL = state.pluginURL;\n },\n isParsed(): boolean {\n if (!isWorker()) throw new Error('rtl-text-plugin is only parsed on the worker-threads');\n\n return plugin.applyArabicShaping != null &&\n plugin.processBidirectionalText != null &&\n plugin.processStyledBidirectionalText != null;\n },\n getPluginURL(): string {\n if (!isWorker()) throw new Error('rtl-text-plugin url can only be queried from the worker threads');\n return pluginURL;\n }\n};\n\nexport const lazyLoadRTLTextPlugin = function() {\n if (!plugin.isLoading() &&\n !plugin.isLoaded() &&\n getRTLTextPluginStatus() === 'deferred'\n ) {\n downloadRTLTextPlugin();\n }\n};\n","import {ZoomHistory} from './zoom_history';\nimport {isStringInSupportedScript} from '../util/script_detection';\nimport {plugin as rtlTextPlugin} from '../source/rtl_text_plugin';\n\nimport type {TransitionSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type CrossfadeParameters = {\n fromScale: number;\n toScale: number;\n t: number;\n};\n\n/**\n * @internal\n * A parameter that can be evaluated to a value\n */\nexport class EvaluationParameters {\n zoom: number;\n now: number;\n fadeDuration: number;\n zoomHistory: ZoomHistory;\n transition: TransitionSpecification;\n\n // \"options\" may also be another EvaluationParameters to copy, see CrossFadedProperty.possiblyEvaluate\n constructor(zoom: number, options?: any) {\n this.zoom = zoom;\n\n if (options) {\n this.now = options.now;\n this.fadeDuration = options.fadeDuration;\n this.zoomHistory = options.zoomHistory;\n this.transition = options.transition;\n } else {\n this.now = 0;\n this.fadeDuration = 0;\n this.zoomHistory = new ZoomHistory();\n this.transition = {};\n }\n }\n\n isSupportedScript(str: string): boolean {\n return isStringInSupportedScript(str, rtlTextPlugin.isLoaded());\n }\n\n crossFadingFactor() {\n if (this.fadeDuration === 0) {\n return 1;\n } else {\n return Math.min((this.now - this.zoomHistory.lastIntegerZoomTime) / this.fadeDuration, 1);\n }\n }\n\n getCrossfadeParameters(): CrossfadeParameters {\n const z = this.zoom;\n const fraction = z - Math.floor(z);\n const t = this.crossFadingFactor();\n\n return z > this.zoomHistory.lastIntegerZoom ?\n {fromScale: 2, toScale: 1, t: fraction + (1 - fraction) * t} :\n {fromScale: 0.5, toScale: 1, t: 1 - (1 - t) * fraction};\n }\n}\n","import {clone, extend, easeCubicInOut} from '../util/util';\nimport {interpolates, Color, StylePropertySpecification, normalizePropertyExpression,\n Feature,\n FeatureState,\n StylePropertyExpression,\n SourceExpression,\n CompositeExpression, TransitionSpecification,\n PropertyValueSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport {register} from '../util/web_worker_transfer';\nimport {EvaluationParameters} from './evaluation_parameters';\n\nimport {CanonicalTileID} from '../source/tile_id';\n\ntype TimePoint = number;\n\n/**\n * A from-to type\n */\nexport type CrossFaded = {\n to: T;\n from: T;\n};\n\n/**\n * @internal\n * Implementations of the `Property` interface:\n *\n * * Hold metadata about a property that's independent of any specific value: stuff like the type of the value,\n * the default value, etc. This comes from the style specification JSON.\n * * Define behavior that needs to be polymorphic across different properties: \"possibly evaluating\"\n * an input value (see below), and interpolating between two possibly-evaluted values.\n *\n * The type `T` is the fully-evaluated value type (e.g. `number`, `string`, `Color`).\n * The type `R` is the intermediate \"possibly evaluated\" value type. See below.\n *\n * There are two main implementations of the interface -- one for properties that allow data-driven values,\n * and one for properties that don't. There are a few \"special case\" implementations as well: one for properties\n * which cross-fade between two values rather than interpolating, one for `heatmap-color` and `line-gradient`,\n * and one for `light-position`.\n */\nexport interface Property {\n specification: StylePropertySpecification;\n possiblyEvaluate(\n value: PropertyValue,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): R;\n interpolate(a: R, b: R, t: number): R;\n}\n\n/**\n * @internal\n * `PropertyValue` represents the value part of a property key-value unit. It's used to represent both\n * paint and layout property values, and regardless of whether or not their property supports data-driven\n * expressions.\n *\n * `PropertyValue` stores the raw input value as seen in a style or a runtime styling API call, i.e. one of the\n * following:\n *\n * * A constant value of the type appropriate for the property\n * * A function which produces a value of that type (but functions are quasi-deprecated in favor of expressions)\n * * An expression which produces a value of that type\n * * \"undefined\"/\"not present\", in which case the property is assumed to take on its default value.\n *\n * In addition to storing the original input value, `PropertyValue` also stores a normalized representation,\n * effectively treating functions as if they are expressions, and constant or default values as if they are\n * (constant) expressions.\n */\nexport class PropertyValue {\n property: Property;\n value: PropertyValueSpecification | void;\n expression: StylePropertyExpression;\n\n constructor(property: Property, value: PropertyValueSpecification | void) {\n this.property = property;\n this.value = value;\n this.expression = normalizePropertyExpression(value === undefined ? property.specification.default : value, property.specification);\n }\n\n isDataDriven(): boolean {\n return this.expression.kind === 'source' || this.expression.kind === 'composite';\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): R {\n return this.property.possiblyEvaluate(this, parameters, canonical, availableImages);\n }\n}\n\nexport type TransitionParameters = {\n now: TimePoint;\n transition: TransitionSpecification;\n};\n\n/**\n * @internal\n * Paint properties are _transitionable_: they can change in a fluid manner, interpolating or cross-fading between\n * old and new value. The duration of the transition, and the delay before it begins, is configurable.\n *\n * `TransitionablePropertyValue` is a compositional class that stores both the property value and that transition\n * configuration.\n *\n * A `TransitionablePropertyValue` can calculate the next step in the evaluation chain for paint property values:\n * `TransitioningPropertyValue`.\n */\nclass TransitionablePropertyValue {\n property: Property;\n value: PropertyValue;\n transition: TransitionSpecification | void;\n\n constructor(property: Property) {\n this.property = property;\n this.value = new PropertyValue(property, undefined);\n }\n\n transitioned(parameters: TransitionParameters, prior: TransitioningPropertyValue): TransitioningPropertyValue {\n return new TransitioningPropertyValue(this.property, this.value, prior,\n extend({}, parameters.transition, this.transition), parameters.now);\n }\n\n untransitioned(): TransitioningPropertyValue {\n return new TransitioningPropertyValue(this.property, this.value, null, {}, 0);\n }\n}\n\n/**\n * @internal\n * `Transitionable` stores a map of all (property name, `TransitionablePropertyValue`) pairs for paint properties of a\n * given layer type. It can calculate the `TransitioningPropertyValue`s for all of them at once, producing a\n * `Transitioning` instance for the same set of properties.\n */\nexport class Transitionable {\n _properties: Properties;\n _values: {[K in keyof Props]: TransitionablePropertyValue};\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = (Object.create(properties.defaultTransitionablePropertyValues) as any);\n }\n\n getValue(name: S): PropertyValueSpecification | void {\n return clone(this._values[name].value.value);\n }\n\n setValue(name: S, value: PropertyValueSpecification | void) {\n if (!Object.prototype.hasOwnProperty.call(this._values, name)) {\n this._values[name] = new TransitionablePropertyValue(this._values[name].property);\n }\n // Note that we do not _remove_ an own property in the case where a value is being reset\n // to the default: the transition might still be non-default.\n this._values[name].value = new PropertyValue(this._values[name].property, value === null ? undefined : clone(value));\n }\n\n getTransition(name: S): TransitionSpecification | void {\n return clone(this._values[name].transition);\n }\n\n setTransition(name: S, value: TransitionSpecification | void) {\n if (!Object.prototype.hasOwnProperty.call(this._values, name)) {\n this._values[name] = new TransitionablePropertyValue(this._values[name].property);\n }\n this._values[name].transition = clone(value) || undefined;\n }\n\n serialize() {\n const result: any = {};\n for (const property of Object.keys(this._values)) {\n const value = this.getValue(property as keyof Props);\n if (value !== undefined) {\n result[property] = value;\n }\n\n const transition = this.getTransition(property as keyof Props);\n if (transition !== undefined) {\n result[`${property}-transition`] = transition;\n }\n }\n return result;\n }\n\n transitioned(parameters: TransitionParameters, prior: Transitioning): Transitioning {\n const result = new Transitioning(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].transitioned(parameters, prior._values[property]);\n }\n return result;\n }\n\n untransitioned(): Transitioning {\n const result = new Transitioning(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].untransitioned();\n }\n return result;\n }\n}\n\n/**\n * @internal\n * `TransitioningPropertyValue` implements the first of two intermediate steps in the evaluation chain of a paint\n * property value. In this step, transitions between old and new values are handled: as long as the transition is in\n * progress, `TransitioningPropertyValue` maintains a reference to the prior value, and interpolates between it and\n * the new value based on the current time and the configured transition duration and delay. The product is the next\n * step in the evaluation chain: the \"possibly evaluated\" result type `R`. See below for more on this concept.\n */\nclass TransitioningPropertyValue {\n property: Property;\n value: PropertyValue;\n prior: TransitioningPropertyValue;\n begin: TimePoint;\n end: TimePoint;\n\n constructor(property: Property,\n value: PropertyValue,\n prior: TransitioningPropertyValue,\n transition: TransitionSpecification,\n now: TimePoint) {\n this.property = property;\n this.value = value;\n this.begin = now + transition.delay || 0;\n this.end = this.begin + transition.duration || 0;\n if (property.specification.transition && (transition.delay || transition.duration)) {\n this.prior = prior;\n }\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical: CanonicalTileID,\n availableImages: Array\n ): R {\n const now = parameters.now || 0;\n const finalValue = this.value.possiblyEvaluate(parameters, canonical, availableImages);\n const prior = this.prior;\n if (!prior) {\n // No prior value.\n return finalValue;\n } else if (now > this.end) {\n // Transition from prior value is now complete.\n this.prior = null;\n return finalValue;\n } else if (this.value.isDataDriven()) {\n // Transitions to data-driven properties are not supported.\n // We snap immediately to the data-driven value so that, when we perform layout,\n // we see the data-driven function and can use it to populate vertex buffers.\n this.prior = null;\n return finalValue;\n } else if (now < this.begin) {\n // Transition hasn't started yet.\n return prior.possiblyEvaluate(parameters, canonical, availableImages);\n } else {\n // Interpolate between recursively-calculated prior value and final.\n const t = (now - this.begin) / (this.end - this.begin);\n return this.property.interpolate(prior.possiblyEvaluate(parameters, canonical, availableImages), finalValue, easeCubicInOut(t));\n }\n }\n}\n\n/**\n * @internal\n * `Transitioning` stores a map of all (property name, `TransitioningPropertyValue`) pairs for paint properties of a\n * given layer type. It can calculate the possibly-evaluated values for all of them at once, producing a\n * `PossiblyEvaluated` instance for the same set of properties.\n */\nexport class Transitioning {\n _properties: Properties;\n _values: {[K in keyof Props]: PossiblyEvaluatedPropertyValue};\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = (Object.create(properties.defaultTransitioningPropertyValues) as any);\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluated {\n const result = new PossiblyEvaluated(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].possiblyEvaluate(parameters, canonical, availableImages);\n }\n return result;\n }\n\n hasTransition() {\n for (const property of Object.keys(this._values)) {\n if (this._values[property].prior) {\n return true;\n }\n }\n return false;\n }\n}\n\n// ------- Layout -------\n\n/**\n * Because layout properties are not transitionable, they have a simpler representation and evaluation chain than\n * paint properties: `PropertyValue`s are possibly evaluated, producing possibly evaluated values, which are then\n * fully evaluated.\n *\n * `Layout` stores a map of all (property name, `PropertyValue`) pairs for layout properties of a\n * given layer type. It can calculate the possibly-evaluated values for all of them at once, producing a\n * `PossiblyEvaluated` instance for the same set of properties.\n */\nexport class Layout {\n _properties: Properties;\n _values: {[K in keyof Props]: PropertyValue>};\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = (Object.create(properties.defaultPropertyValues) as any);\n }\n\n hasValue(name: S) {\n return this._values[name].value !== undefined;\n }\n\n getValue(name: S) {\n return clone(this._values[name].value);\n }\n\n setValue(name: S, value: any) {\n this._values[name] = new PropertyValue(this._values[name].property, value === null ? undefined : clone(value)) as any;\n }\n\n serialize() {\n const result: any = {};\n for (const property of Object.keys(this._values)) {\n const value = this.getValue(property as keyof Props);\n if (value !== undefined) {\n result[property] = value;\n }\n }\n return result;\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluated {\n const result = new PossiblyEvaluated(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].possiblyEvaluate(parameters, canonical, availableImages);\n }\n return result;\n }\n}\n\n// ------- PossiblyEvaluated -------\n\n/**\n * \"Possibly evaluated value\" is an intermediate stage in the evaluation chain for both paint and layout property\n * values. The purpose of this stage is to optimize away unnecessary recalculations for data-driven properties. Code\n * which uses data-driven property values must assume that the value is dependent on feature data, and request that it\n * be evaluated for each feature. But when that property value is in fact a constant or camera function, the calculation\n * will not actually depend on the feature, and we can benefit from returning the prior result of having done the\n * evaluation once, ahead of time, in an intermediate step whose inputs are just the value and \"global\" parameters\n * such as current zoom level.\n *\n * `PossiblyEvaluatedValue` represents the three possible outcomes of this step: if the input value was a constant or\n * camera expression, then the \"possibly evaluated\" result is a constant value. Otherwise, the input value was either\n * a source or composite expression, and we must defer final evaluation until supplied a feature. We separate\n * the source and composite cases because they are handled differently when generating GL attributes, buffers, and\n * uniforms.\n *\n * Note that `PossiblyEvaluatedValue` (and `PossiblyEvaluatedPropertyValue`, below) are _not_ used for properties that\n * do not allow data-driven values. For such properties, we know that the \"possibly evaluated\" result is always a constant\n * scalar value. See below.\n */\ntype PossiblyEvaluatedValue = {\n kind: 'constant';\n value: T;\n} | SourceExpression | CompositeExpression;\n\n/**\n * @internal\n * `PossiblyEvaluatedPropertyValue` is used for data-driven paint and layout property values. It holds a\n * `PossiblyEvaluatedValue` and the `GlobalProperties` that were used to generate it. You're not allowed to supply\n * a different set of `GlobalProperties` when performing the final evaluation because they would be ignored in the\n * case where the input value was a constant or camera function.\n */\nexport class PossiblyEvaluatedPropertyValue {\n property: DataDrivenProperty;\n value: PossiblyEvaluatedValue;\n parameters: EvaluationParameters;\n\n constructor(property: DataDrivenProperty, value: PossiblyEvaluatedValue, parameters: EvaluationParameters) {\n this.property = property;\n this.value = value;\n this.parameters = parameters;\n }\n\n isConstant(): boolean {\n return this.value.kind === 'constant';\n }\n\n constantOr(value: T): T {\n if (this.value.kind === 'constant') {\n return this.value.value;\n } else {\n return value;\n }\n }\n\n evaluate(\n feature: Feature,\n featureState: FeatureState,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): T {\n return this.property.evaluate(this.value, this.parameters, feature, featureState, canonical, availableImages);\n }\n}\n\n/**\n * @internal\n * `PossiblyEvaluated` stores a map of all (property name, `R`) pairs for paint or layout properties of a\n * given layer type.\n */\nexport class PossiblyEvaluated {\n _properties: Properties;\n _values: PossibleEvaluatedProps;\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = Object.create(properties.defaultPossiblyEvaluatedValues);\n }\n\n get(name: S): PossibleEvaluatedProps[S] {\n return this._values[name];\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for properties that do not permit data-driven (source or composite) expressions.\n * This restriction allows us to declare statically that the result of possibly evaluating this kind of property\n * is in fact always the scalar type `T`, and can be used without further evaluating the value on a per-feature basis.\n */\nexport class DataConstantProperty implements Property {\n specification: StylePropertySpecification;\n\n constructor(specification: StylePropertySpecification) {\n this.specification = specification;\n }\n\n possiblyEvaluate(value: PropertyValue, parameters: EvaluationParameters): T {\n if (value.isDataDriven()) throw new Error('Value should not be data driven');\n return value.expression.evaluate(parameters);\n }\n\n interpolate(a: T, b: T, t: number): T {\n const interpolationType = this.specification.type as keyof typeof interpolates;\n const interpolationFn = interpolates[interpolationType] as ((from: T, to: T, t: number) => T) | undefined;\n if (interpolationFn) {\n return interpolationFn(a, b, t);\n } else {\n return a;\n }\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for properties that permit data-driven (source or composite) expressions.\n * The result of possibly evaluating this kind of property is `PossiblyEvaluatedPropertyValue`; obtaining\n * a scalar value `T` requires further evaluation on a per-feature basis.\n */\nexport class DataDrivenProperty implements Property> {\n specification: StylePropertySpecification;\n overrides: any;\n\n constructor(specification: StylePropertySpecification, overrides?: any) {\n this.specification = specification;\n this.overrides = overrides;\n }\n\n possiblyEvaluate(\n value: PropertyValue>,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluatedPropertyValue {\n if (value.expression.kind === 'constant' || value.expression.kind === 'camera') {\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: value.expression.evaluate(parameters, null, {}, canonical, availableImages)}, parameters);\n } else {\n return new PossiblyEvaluatedPropertyValue(this, value.expression, parameters);\n }\n }\n\n interpolate(\n a: PossiblyEvaluatedPropertyValue,\n b: PossiblyEvaluatedPropertyValue,\n t: number\n ): PossiblyEvaluatedPropertyValue {\n // If either possibly-evaluated value is non-constant, give up: we aren't able to interpolate data-driven values.\n if (a.value.kind !== 'constant' || b.value.kind !== 'constant') {\n return a;\n }\n\n // Special case hack solely for fill-outline-color. The undefined value is subsequently handled in\n // FillStyleLayer#recalculate, which sets fill-outline-color to the fill-color value if the former\n // is a PossiblyEvaluatedPropertyValue containing a constant undefined value. In addition to the\n // return value here, the other source of a PossiblyEvaluatedPropertyValue containing a constant\n // undefined value is the \"default value\" for fill-outline-color held in\n // `Properties#defaultPossiblyEvaluatedValues`, which serves as the prototype of\n // `PossiblyEvaluated#_values`.\n if (a.value.value === undefined || b.value.value === undefined) {\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: undefined}, a.parameters);\n }\n\n const interpolationType = this.specification.type as keyof typeof interpolates;\n const interpolationFn = interpolates[interpolationType] as ((from: T, to: T, t: number) => T) | undefined;\n if (interpolationFn) {\n const interpolatedValue = interpolationFn(a.value.value, b.value.value, t);\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: interpolatedValue}, a.parameters);\n } else {\n return a;\n }\n }\n\n evaluate(\n value: PossiblyEvaluatedValue,\n parameters: EvaluationParameters,\n feature: Feature,\n featureState: FeatureState,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): T {\n if (value.kind === 'constant') {\n return value.value;\n } else {\n return value.evaluate(parameters, feature, featureState, canonical, availableImages);\n }\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for data driven `line-pattern` which are transitioned by cross-fading\n * rather than interpolation.\n */\n\nexport class CrossFadedDataDrivenProperty extends DataDrivenProperty> {\n\n possiblyEvaluate(\n value: PropertyValue, PossiblyEvaluatedPropertyValue>>,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluatedPropertyValue> {\n if (value.value === undefined) {\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: undefined}, parameters);\n } else if (value.expression.kind === 'constant') {\n const evaluatedValue = value.expression.evaluate(parameters, null, {}, canonical, availableImages);\n const isImageExpression = value.property.specification.type as any === 'resolvedImage';\n const constantValue = isImageExpression && typeof evaluatedValue !== 'string' ? evaluatedValue.name : evaluatedValue;\n const constant = this._calculate(constantValue, constantValue, constantValue, parameters);\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: constant}, parameters);\n } else if (value.expression.kind === 'camera') {\n const cameraVal = this._calculate(\n value.expression.evaluate({zoom: parameters.zoom - 1.0}),\n value.expression.evaluate({zoom: parameters.zoom}),\n value.expression.evaluate({zoom: parameters.zoom + 1.0}),\n parameters);\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: cameraVal}, parameters);\n } else {\n // source or composite expression\n return new PossiblyEvaluatedPropertyValue(this, value.expression, parameters);\n }\n }\n\n evaluate(\n value: PossiblyEvaluatedValue>,\n globals: EvaluationParameters,\n feature: Feature,\n featureState: FeatureState,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): CrossFaded {\n if (value.kind === 'source') {\n const constant = value.evaluate(globals, feature, featureState, canonical, availableImages);\n return this._calculate(constant, constant, constant, globals);\n } else if (value.kind === 'composite') {\n return this._calculate(\n value.evaluate({zoom: Math.floor(globals.zoom) - 1.0}, feature, featureState),\n value.evaluate({zoom: Math.floor(globals.zoom)}, feature, featureState),\n value.evaluate({zoom: Math.floor(globals.zoom) + 1.0}, feature, featureState),\n globals);\n } else {\n return value.value;\n }\n }\n\n _calculate(min: T, mid: T, max: T, parameters: EvaluationParameters): CrossFaded {\n const z = parameters.zoom;\n return z > parameters.zoomHistory.lastIntegerZoom ? {from: min, to: mid} : {from: max, to: mid};\n }\n\n interpolate(a: PossiblyEvaluatedPropertyValue>): PossiblyEvaluatedPropertyValue> {\n return a;\n }\n}\n/**\n * @internal\n * An implementation of `Property` for `*-pattern` and `line-dasharray`, which are transitioned by cross-fading\n * rather than interpolation.\n */\nexport class CrossFadedProperty implements Property> {\n specification: StylePropertySpecification;\n\n constructor(specification: StylePropertySpecification) {\n this.specification = specification;\n }\n\n possiblyEvaluate(\n value: PropertyValue>,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): CrossFaded {\n if (value.value === undefined) {\n return undefined;\n } else if (value.expression.kind === 'constant') {\n const constant = value.expression.evaluate(parameters, null, {}, canonical, availableImages);\n return this._calculate(constant, constant, constant, parameters);\n } else {\n return this._calculate(\n value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom - 1.0), parameters)),\n value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom), parameters)),\n value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom + 1.0), parameters)),\n parameters);\n }\n }\n\n _calculate(min: T, mid: T, max: T, parameters: EvaluationParameters): CrossFaded {\n const z = parameters.zoom;\n return z > parameters.zoomHistory.lastIntegerZoom ? {from: min, to: mid} : {from: max, to: mid};\n }\n\n interpolate(a?: CrossFaded | null): CrossFaded {\n return a;\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for `heatmap-color` and `line-gradient`. Interpolation is a no-op, and\n * evaluation returns a boolean value in order to indicate its presence, but the real\n * evaluation happens in StyleLayer classes.\n */\n\nexport class ColorRampProperty implements Property {\n specification: StylePropertySpecification;\n\n constructor(specification: StylePropertySpecification) {\n this.specification = specification;\n }\n\n possiblyEvaluate(\n value: PropertyValue,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): boolean {\n return !!value.expression.evaluate(parameters, null, {}, canonical, availableImages);\n }\n\n interpolate(): boolean { return false; }\n}\n\n/**\n * @internal\n * `Properties` holds objects containing default values for the layout or paint property set of a given\n * layer type. These objects are immutable, and they are used as the prototypes for the `_values` members of\n * `Transitionable`, `Transitioning`, `Layout`, and `PossiblyEvaluated`. This allows these classes to avoid\n * doing work in the common case where a property has no explicit value set and should be considered to take\n * on the default value: using `for (const property of Object.keys(this._values))`, they can iterate over\n * only the _own_ properties of `_values`, skipping repeated calculation of transitions and possible/final\n * evaluations for defaults, the result of which will always be the same.\n */\nexport class Properties {\n properties: Props;\n defaultPropertyValues: {[K in keyof Props]: PropertyValue};\n defaultTransitionablePropertyValues: {[K in keyof Props]: TransitionablePropertyValue};\n defaultTransitioningPropertyValues: {[K in keyof Props]: TransitioningPropertyValue};\n defaultPossiblyEvaluatedValues: {[K in keyof Props]: PossiblyEvaluatedPropertyValue};\n overridableProperties: Array;\n\n constructor(properties: Props) {\n this.properties = properties;\n this.defaultPropertyValues = ({} as any);\n this.defaultTransitionablePropertyValues = ({} as any);\n this.defaultTransitioningPropertyValues = ({} as any);\n this.defaultPossiblyEvaluatedValues = ({} as any);\n this.overridableProperties = ([] as any);\n\n for (const property in properties) {\n const prop = properties[property] as any;\n if (prop.specification.overridable) {\n this.overridableProperties.push(property);\n }\n const defaultPropertyValue = this.defaultPropertyValues[property] =\n new PropertyValue(prop, undefined);\n const defaultTransitionablePropertyValue = this.defaultTransitionablePropertyValues[property] =\n new TransitionablePropertyValue(prop);\n this.defaultTransitioningPropertyValues[property] =\n defaultTransitionablePropertyValue.untransitioned();\n this.defaultPossiblyEvaluatedValues[property] =\n defaultPropertyValue.possiblyEvaluate({} as any);\n }\n }\n}\n\nregister('DataDrivenProperty', DataDrivenProperty);\nregister('DataConstantProperty', DataConstantProperty);\nregister('CrossFadedDataDrivenProperty', CrossFadedDataDrivenProperty);\nregister('CrossFadedProperty', CrossFadedProperty);\nregister('ColorRampProperty', ColorRampProperty);\n","import {filterObject} from '../util/util';\n\nimport {latest as styleSpec, supportsPropertyExpression} from '@maplibre/maplibre-gl-style-spec';\nimport {\n validateStyle,\n validateLayoutProperty,\n validatePaintProperty,\n emitValidationErrors\n} from './validate_style';\nimport {Evented} from '../util/evented';\nimport {Layout, Transitionable, Transitioning, Properties, PossiblyEvaluated, PossiblyEvaluatedPropertyValue} from './properties';\n\nimport type {Bucket} from '../data/bucket';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureFilter, FeatureState,\n LayerSpecification,\n FilterSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {TransitionParameters, PropertyValue} from './properties';\nimport {EvaluationParameters} from './evaluation_parameters';\nimport type {CrossfadeParameters} from './evaluation_parameters';\n\nimport type {Transform} from '../geo/transform';\nimport type {CustomLayerInterface} from './style_layer/custom_style_layer';\nimport type {Map} from '../ui/map';\nimport type {StyleSetterOptions} from './style';\nimport {mat4} from 'gl-matrix';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nconst TRANSITION_SUFFIX = '-transition';\n\n/**\n * A base class for style layers\n */\nexport abstract class StyleLayer extends Evented {\n id: string;\n metadata: unknown;\n type: LayerSpecification['type'] | CustomLayerInterface['type'];\n source: string;\n sourceLayer: string;\n minzoom: number;\n maxzoom: number;\n filter: FilterSpecification | void;\n visibility: 'visible' | 'none' | void;\n _crossfadeParameters: CrossfadeParameters;\n\n _unevaluatedLayout: Layout;\n readonly layout: unknown;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n readonly paint: unknown;\n\n _featureFilter: FeatureFilter;\n\n readonly onAdd: ((map: Map) => void);\n readonly onRemove: ((map: Map) => void);\n\n queryRadius?(bucket: Bucket): number;\n queryIntersectsFeature?(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number,\n pixelPosMatrix: mat4\n ): boolean | number;\n\n constructor(layer: LayerSpecification | CustomLayerInterface, properties: Readonly<{\n layout?: Properties;\n paint?: Properties;\n }>) {\n super();\n\n this.id = layer.id;\n this.type = layer.type;\n this._featureFilter = {filter: () => true, needGeometry: false};\n\n if (layer.type === 'custom') return;\n\n layer = (layer as any as LayerSpecification);\n\n this.metadata = layer.metadata;\n this.minzoom = layer.minzoom;\n this.maxzoom = layer.maxzoom;\n\n if (layer.type !== 'background') {\n this.source = layer.source;\n this.sourceLayer = layer['source-layer'];\n this.filter = layer.filter;\n }\n\n if (properties.layout) {\n this._unevaluatedLayout = new Layout(properties.layout);\n }\n\n if (properties.paint) {\n this._transitionablePaint = new Transitionable(properties.paint);\n\n for (const property in layer.paint) {\n this.setPaintProperty(property, layer.paint[property], {validate: false});\n }\n for (const property in layer.layout) {\n this.setLayoutProperty(property, layer.layout[property], {validate: false});\n }\n\n this._transitioningPaint = this._transitionablePaint.untransitioned();\n //$FlowFixMe\n this.paint = new PossiblyEvaluated(properties.paint);\n }\n }\n\n getCrossfadeParameters() {\n return this._crossfadeParameters;\n }\n\n getLayoutProperty(name: string) {\n if (name === 'visibility') {\n return this.visibility;\n }\n\n return this._unevaluatedLayout.getValue(name);\n }\n\n setLayoutProperty(name: string, value: any, options: StyleSetterOptions = {}) {\n if (value !== null && value !== undefined) {\n const key = `layers.${this.id}.layout.${name}`;\n if (this._validate(validateLayoutProperty, key, name, value, options)) {\n return;\n }\n }\n\n if (name === 'visibility') {\n this.visibility = value;\n return;\n }\n\n this._unevaluatedLayout.setValue(name, value);\n }\n\n getPaintProperty(name: string) {\n if (name.endsWith(TRANSITION_SUFFIX)) {\n return this._transitionablePaint.getTransition(name.slice(0, -TRANSITION_SUFFIX.length));\n } else {\n return this._transitionablePaint.getValue(name);\n }\n }\n\n setPaintProperty(name: string, value: unknown, options: StyleSetterOptions = {}) {\n if (value !== null && value !== undefined) {\n const key = `layers.${this.id}.paint.${name}`;\n if (this._validate(validatePaintProperty, key, name, value, options)) {\n return false;\n }\n }\n\n if (name.endsWith(TRANSITION_SUFFIX)) {\n this._transitionablePaint.setTransition(name.slice(0, -TRANSITION_SUFFIX.length), (value as any) || undefined);\n return false;\n } else {\n const transitionable = this._transitionablePaint._values[name];\n const isCrossFadedProperty = transitionable.property.specification['property-type'] === 'cross-faded-data-driven';\n const wasDataDriven = transitionable.value.isDataDriven();\n const oldValue = transitionable.value;\n\n this._transitionablePaint.setValue(name, value);\n this._handleSpecialPaintPropertyUpdate(name);\n\n const newValue = this._transitionablePaint._values[name].value;\n const isDataDriven = newValue.isDataDriven();\n\n // if a cross-faded value is changed, we need to make sure the new icons get added to each tile's iconAtlas\n // so a call to _updateLayer is necessary, and we return true from this function so it gets called in\n // Style#setPaintProperty\n return isDataDriven || wasDataDriven || isCrossFadedProperty || this._handleOverridablePaintPropertyUpdate(name, oldValue, newValue);\n }\n }\n\n _handleSpecialPaintPropertyUpdate(_: string) {\n // No-op; can be overridden by derived classes.\n }\n\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _handleOverridablePaintPropertyUpdate(name: string, oldValue: PropertyValue, newValue: PropertyValue): boolean {\n // No-op; can be overridden by derived classes.\n return false;\n }\n\n isHidden(zoom: number) {\n if (this.minzoom && zoom < this.minzoom) return true;\n if (this.maxzoom && zoom >= this.maxzoom) return true;\n return this.visibility === 'none';\n }\n\n updateTransitions(parameters: TransitionParameters) {\n this._transitioningPaint = this._transitionablePaint.transitioned(parameters, this._transitioningPaint);\n }\n\n hasTransition() {\n return this._transitioningPaint.hasTransition();\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n if (parameters.getCrossfadeParameters) {\n this._crossfadeParameters = parameters.getCrossfadeParameters();\n }\n\n if (this._unevaluatedLayout) {\n (this as any).layout = this._unevaluatedLayout.possiblyEvaluate(parameters, undefined, availableImages);\n }\n\n (this as any).paint = this._transitioningPaint.possiblyEvaluate(parameters, undefined, availableImages);\n }\n\n serialize(): LayerSpecification {\n const output: LayerSpecification = {\n 'id': this.id,\n 'type': this.type as LayerSpecification['type'],\n 'source': this.source,\n 'source-layer': this.sourceLayer,\n 'metadata': this.metadata,\n 'minzoom': this.minzoom,\n 'maxzoom': this.maxzoom,\n 'filter': this.filter as FilterSpecification,\n 'layout': this._unevaluatedLayout && this._unevaluatedLayout.serialize(),\n 'paint': this._transitionablePaint && this._transitionablePaint.serialize()\n };\n\n if (this.visibility) {\n output.layout = output.layout || {};\n output.layout.visibility = this.visibility;\n }\n\n return filterObject(output, (value, key) => {\n return value !== undefined &&\n !(key === 'layout' && !Object.keys(value).length) &&\n !(key === 'paint' && !Object.keys(value).length);\n });\n }\n\n _validate(validate: Function, key: string, name: string, value: unknown, options: StyleSetterOptions = {}) {\n if (options && options.validate === false) {\n return false;\n }\n return emitValidationErrors(this, validate.call(validateStyle, {\n key,\n layerType: this.type,\n objectKey: name,\n value,\n styleSpec,\n // Workaround for https://github.com/mapbox/mapbox-gl-js/issues/2407\n style: {glyphs: true, sprite: true}\n }));\n }\n\n is3D() {\n return false;\n }\n\n isTileClipped() {\n return false;\n }\n\n hasOffscreenPass() {\n return false;\n }\n\n resize() {\n // noop\n }\n\n isStateDependent() {\n for (const property in (this as any).paint._values) {\n const value = (this as any).paint.get(property);\n if (!(value instanceof PossiblyEvaluatedPropertyValue) || !supportsPropertyExpression(value.property.specification)) {\n continue;\n }\n\n if ((value.value.kind === 'source' || value.value.kind === 'composite') &&\n value.value.isStateDependent) {\n return true;\n }\n }\n return false;\n }\n}\n","// Note: all \"sizes\" are measured in bytes\n\nimport type {Transferable} from '../types/transferable';\n\n/**\n * @internal\n * A view type size\n */\nconst viewTypes = {\n 'Int8': Int8Array,\n 'Uint8': Uint8Array,\n 'Int16': Int16Array,\n 'Uint16': Uint16Array,\n 'Int32': Int32Array,\n 'Uint32': Uint32Array,\n 'Float32': Float32Array\n};\n\n/**\n * @internal\n * A view type size\n */\nexport type ViewType = keyof typeof viewTypes;\n\n/** @internal */\nclass Struct {\n _pos1: number;\n _pos2: number;\n _pos4: number;\n _pos8: number;\n readonly _structArray: StructArray;\n\n // The following properties are defined on the prototype of sub classes.\n size: number;\n\n /**\n * @param structArray - The StructArray the struct is stored in\n * @param index - The index of the struct in the StructArray.\n */\n constructor(structArray: StructArray, index: number) {\n (this as any)._structArray = structArray;\n this._pos1 = index * this.size;\n this._pos2 = this._pos1 / 2;\n this._pos4 = this._pos1 / 4;\n this._pos8 = this._pos1 / 8;\n }\n}\n\nconst DEFAULT_CAPACITY = 128;\nconst RESIZE_MULTIPLIER = 5;\n\n/**\n * @internal\n * A struct array memeber\n */\nexport type StructArrayMember = {\n name: string;\n type: ViewType;\n components: number;\n offset: number;\n};\n\nexport type StructArrayLayout = {\n members: Array;\n size: number;\n alignment: number;\n};\n\n/**\n * An array that can be desialized\n */\nexport type SerializedStructArray = {\n length: number;\n arrayBuffer: ArrayBuffer;\n};\n\n/**\n * @internal\n * `StructArray` provides an abstraction over `ArrayBuffer` and `TypedArray`\n * making it behave like an array of typed structs.\n *\n * Conceptually, a StructArray is comprised of elements, i.e., instances of its\n * associated struct type. Each particular struct type, together with an\n * alignment size, determines the memory layout of a StructArray whose elements\n * are of that type. Thus, for each such layout that we need, we have\n * a corresponding StructArrayLayout class, inheriting from StructArray and\n * implementing `emplaceBack()` and `_refreshViews()`.\n *\n * In some cases, where we need to access particular elements of a StructArray,\n * we implement a more specific subclass that inherits from one of the\n * StructArrayLayouts and adds a `get(i): T` accessor that returns a structured\n * object whose properties are proxies into the underlying memory space for the\n * i-th element. This affords the convience of working with (seemingly) plain\n * Javascript objects without the overhead of serializing/deserializing them\n * into ArrayBuffers for efficient web worker transfer.\n */\nabstract class StructArray {\n capacity: number;\n length: number;\n isTransferred: boolean;\n arrayBuffer: ArrayBuffer;\n uint8: Uint8Array;\n\n // The following properties are defined on the prototype.\n members: Array;\n bytesPerElement: number;\n abstract emplaceBack(...v: number[]);\n abstract emplace(i: number, ...v: number[]);\n\n constructor() {\n this.isTransferred = false;\n this.capacity = -1;\n this.resize(0);\n }\n\n /**\n * Serialize a StructArray instance. Serializes both the raw data and the\n * metadata needed to reconstruct the StructArray base class during\n * deserialization.\n */\n static serialize(array: StructArray, transferables?: Array): SerializedStructArray {\n\n array._trim();\n\n if (transferables) {\n array.isTransferred = true;\n transferables.push(array.arrayBuffer);\n }\n\n return {\n length: array.length,\n arrayBuffer: array.arrayBuffer,\n };\n }\n\n static deserialize(input: SerializedStructArray) {\n const structArray = Object.create(this.prototype);\n structArray.arrayBuffer = input.arrayBuffer;\n structArray.length = input.length;\n structArray.capacity = input.arrayBuffer.byteLength / structArray.bytesPerElement;\n structArray._refreshViews();\n return structArray;\n }\n\n /**\n * Resize the array to discard unused capacity.\n */\n _trim() {\n if (this.length !== this.capacity) {\n this.capacity = this.length;\n this.arrayBuffer = this.arrayBuffer.slice(0, this.length * this.bytesPerElement);\n this._refreshViews();\n }\n }\n\n /**\n * Resets the length of the array to 0 without de-allocating capcacity.\n */\n clear() {\n this.length = 0;\n }\n\n /**\n * Resize the array.\n * If `n` is greater than the current length then additional elements with undefined values are added.\n * If `n` is less than the current length then the array will be reduced to the first `n` elements.\n * @param n - The new size of the array.\n */\n resize(n: number) {\n this.reserve(n);\n this.length = n;\n }\n\n /**\n * Indicate a planned increase in size, so that any necessary allocation may\n * be done once, ahead of time.\n * @param n - The expected size of the array.\n */\n reserve(n: number) {\n if (n > this.capacity) {\n this.capacity = Math.max(n, Math.floor(this.capacity * RESIZE_MULTIPLIER), DEFAULT_CAPACITY);\n this.arrayBuffer = new ArrayBuffer(this.capacity * this.bytesPerElement);\n\n const oldUint8Array = this.uint8;\n this._refreshViews();\n if (oldUint8Array) this.uint8.set(oldUint8Array);\n }\n }\n\n /**\n * Create TypedArray views for the current ArrayBuffer.\n */\n _refreshViews() {\n throw new Error('_refreshViews() must be implemented by each concrete StructArray layout');\n }\n}\n\n/**\n * Given a list of member fields, create a full StructArrayLayout, in\n * particular calculating the correct byte offset for each field. This data\n * is used at build time to generate StructArrayLayout_*#emplaceBack() and\n * other accessors, and at runtime for binding vertex buffer attributes.\n */\nfunction createLayout(\n members: Array<{\n name: string;\n type: ViewType;\n readonly components?: number;\n }>,\n alignment: number = 1\n): StructArrayLayout {\n\n let offset = 0;\n let maxSize = 0;\n const layoutMembers = members.map((member) => {\n const typeSize = sizeOf(member.type);\n const memberOffset = offset = align(offset, Math.max(alignment, typeSize));\n const components = member.components || 1;\n\n maxSize = Math.max(maxSize, typeSize);\n offset += typeSize * components;\n\n return {\n name: member.name,\n type: member.type,\n components,\n offset: memberOffset,\n };\n });\n\n const size = align(offset, Math.max(maxSize, alignment));\n\n return {\n members: layoutMembers,\n size,\n alignment\n };\n}\n\nfunction sizeOf(type: ViewType): number {\n return viewTypes[type].BYTES_PER_ELEMENT;\n}\n\nfunction align(offset: number, size: number): number {\n return Math.ceil(offset / size) * size;\n}\n\nexport {StructArray, Struct, viewTypes, createLayout};\n","// This file is generated. Edit build/generate-struct-arrays.ts, then run `npm run codegen`.\n\nimport {Struct, StructArray} from '../util/struct_array';\nimport {register} from '../util/web_worker_transfer';\nimport Point from '@mapbox/point-geometry';\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n *\n */\nclass StructArrayLayout2i4 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1);\n }\n\n public emplace(i: number, v0: number, v1: number) {\n const o2 = i * 2;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n return i;\n }\n}\n\nStructArrayLayout2i4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout2i4', StructArrayLayout2i4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[3]\n *\n */\nclass StructArrayLayout3i6 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o2 = i * 3;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout3i6.prototype.bytesPerElement = 6;\nregister('StructArrayLayout3i6', StructArrayLayout3i6);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[4]\n *\n */\nclass StructArrayLayout4i8 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number) {\n const o2 = i * 4;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n return i;\n }\n}\n\nStructArrayLayout4i8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout4i8', StructArrayLayout4i8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Int16[4]\n *\n */\nclass StructArrayLayout2i4i12 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const o2 = i * 6;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n return i;\n }\n}\n\nStructArrayLayout2i4i12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout2i4i12', StructArrayLayout2i4i12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Uint8[4]\n *\n */\nclass StructArrayLayout2i4ub8 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const o2 = i * 4;\n const o1 = i * 8;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.uint8[o1 + 4] = v2;\n this.uint8[o1 + 5] = v3;\n this.uint8[o1 + 6] = v4;\n this.uint8[o1 + 7] = v5;\n return i;\n }\n}\n\nStructArrayLayout2i4ub8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout2i4ub8', StructArrayLayout2i4ub8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[2]\n *\n */\nclass StructArrayLayout2f8 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1);\n }\n\n public emplace(i: number, v0: number, v1: number) {\n const o4 = i * 2;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n return i;\n }\n}\n\nStructArrayLayout2f8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout2f8', StructArrayLayout2f8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[10]\n *\n */\nclass StructArrayLayout10ui20 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number) {\n const o2 = i * 10;\n this.uint16[o2 + 0] = v0;\n this.uint16[o2 + 1] = v1;\n this.uint16[o2 + 2] = v2;\n this.uint16[o2 + 3] = v3;\n this.uint16[o2 + 4] = v4;\n this.uint16[o2 + 5] = v5;\n this.uint16[o2 + 6] = v6;\n this.uint16[o2 + 7] = v7;\n this.uint16[o2 + 8] = v8;\n this.uint16[o2 + 9] = v9;\n return i;\n }\n}\n\nStructArrayLayout10ui20.prototype.bytesPerElement = 20;\nregister('StructArrayLayout10ui20', StructArrayLayout10ui20);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[4]\n * [8]: Uint16[4]\n * [16]: Int16[4]\n *\n */\nclass StructArrayLayout4i4ui4i24 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number) {\n const o2 = i * 12;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.uint16[o2 + 4] = v4;\n this.uint16[o2 + 5] = v5;\n this.uint16[o2 + 6] = v6;\n this.uint16[o2 + 7] = v7;\n this.int16[o2 + 8] = v8;\n this.int16[o2 + 9] = v9;\n this.int16[o2 + 10] = v10;\n this.int16[o2 + 11] = v11;\n return i;\n }\n}\n\nStructArrayLayout4i4ui4i24.prototype.bytesPerElement = 24;\nregister('StructArrayLayout4i4ui4i24', StructArrayLayout4i4ui4i24);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[3]\n *\n */\nclass StructArrayLayout3f12 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o4 = i * 3;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout3f12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout3f12', StructArrayLayout3f12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint32[1]\n *\n */\nclass StructArrayLayout1ul4 extends StructArray {\n uint8: Uint8Array;\n uint32: Uint32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0);\n }\n\n public emplace(i: number, v0: number) {\n const o4 = i * 1;\n this.uint32[o4 + 0] = v0;\n return i;\n }\n}\n\nStructArrayLayout1ul4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout1ul4', StructArrayLayout1ul4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[6]\n * [12]: Uint32[1]\n * [16]: Uint16[2]\n *\n */\nclass StructArrayLayout6i1ul2ui20 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint32: Uint32Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number) {\n const o2 = i * 10;\n const o4 = i * 5;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n this.uint32[o4 + 3] = v6;\n this.uint16[o2 + 8] = v7;\n this.uint16[o2 + 9] = v8;\n return i;\n }\n}\n\nStructArrayLayout6i1ul2ui20.prototype.bytesPerElement = 20;\nregister('StructArrayLayout6i1ul2ui20', StructArrayLayout6i1ul2ui20);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Int16[2]\n * [8]: Int16[2]\n *\n */\nclass StructArrayLayout2i2i2i12 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const o2 = i * 6;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n return i;\n }\n}\n\nStructArrayLayout2i2i2i12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout2i2i2i12', StructArrayLayout2i2i2i12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[2]\n * [8]: Float32[1]\n * [12]: Int16[2]\n *\n */\nclass StructArrayLayout2f1f2i16 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number) {\n const o4 = i * 4;\n const o2 = i * 8;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n this.int16[o2 + 6] = v3;\n this.int16[o2 + 7] = v4;\n return i;\n }\n}\n\nStructArrayLayout2f1f2i16.prototype.bytesPerElement = 16;\nregister('StructArrayLayout2f1f2i16', StructArrayLayout2f1f2i16);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint8[2]\n * [4]: Float32[2]\n *\n */\nclass StructArrayLayout2ub2f12 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number) {\n const o1 = i * 12;\n const o4 = i * 3;\n this.uint8[o1 + 0] = v0;\n this.uint8[o1 + 1] = v1;\n this.float32[o4 + 1] = v2;\n this.float32[o4 + 2] = v3;\n return i;\n }\n}\n\nStructArrayLayout2ub2f12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout2ub2f12', StructArrayLayout2ub2f12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[3]\n *\n */\nclass StructArrayLayout3ui6 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o2 = i * 3;\n this.uint16[o2 + 0] = v0;\n this.uint16[o2 + 1] = v1;\n this.uint16[o2 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout3ui6.prototype.bytesPerElement = 6;\nregister('StructArrayLayout3ui6', StructArrayLayout3ui6);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Uint16[2]\n * [8]: Uint32[3]\n * [20]: Uint16[3]\n * [28]: Float32[2]\n * [36]: Uint8[3]\n * [40]: Uint32[1]\n * [44]: Int16[1]\n *\n */\nclass StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint16: Uint16Array;\n uint32: Uint32Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number) {\n const o2 = i * 24;\n const o4 = i * 12;\n const o1 = i * 48;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.uint16[o2 + 2] = v2;\n this.uint16[o2 + 3] = v3;\n this.uint32[o4 + 2] = v4;\n this.uint32[o4 + 3] = v5;\n this.uint32[o4 + 4] = v6;\n this.uint16[o2 + 10] = v7;\n this.uint16[o2 + 11] = v8;\n this.uint16[o2 + 12] = v9;\n this.float32[o4 + 7] = v10;\n this.float32[o4 + 8] = v11;\n this.uint8[o1 + 36] = v12;\n this.uint8[o1 + 37] = v13;\n this.uint8[o1 + 38] = v14;\n this.uint32[o4 + 10] = v15;\n this.int16[o2 + 22] = v16;\n return i;\n }\n}\n\nStructArrayLayout2i2ui3ul3ui2f3ub1ul1i48.prototype.bytesPerElement = 48;\nregister('StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48', StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[8]\n * [16]: Uint16[15]\n * [48]: Uint32[1]\n * [52]: Float32[2]\n * [60]: Uint16[2]\n *\n */\nclass StructArrayLayout8i15ui1ul2f2ui64 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint16: Uint16Array;\n uint32: Uint32Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number, v17: number, v18: number, v19: number, v20: number, v21: number, v22: number, v23: number, v24: number, v25: number, v26: number, v27: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number, v17: number, v18: number, v19: number, v20: number, v21: number, v22: number, v23: number, v24: number, v25: number, v26: number, v27: number) {\n const o2 = i * 32;\n const o4 = i * 16;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n this.int16[o2 + 6] = v6;\n this.int16[o2 + 7] = v7;\n this.uint16[o2 + 8] = v8;\n this.uint16[o2 + 9] = v9;\n this.uint16[o2 + 10] = v10;\n this.uint16[o2 + 11] = v11;\n this.uint16[o2 + 12] = v12;\n this.uint16[o2 + 13] = v13;\n this.uint16[o2 + 14] = v14;\n this.uint16[o2 + 15] = v15;\n this.uint16[o2 + 16] = v16;\n this.uint16[o2 + 17] = v17;\n this.uint16[o2 + 18] = v18;\n this.uint16[o2 + 19] = v19;\n this.uint16[o2 + 20] = v20;\n this.uint16[o2 + 21] = v21;\n this.uint16[o2 + 22] = v22;\n this.uint32[o4 + 12] = v23;\n this.float32[o4 + 13] = v24;\n this.float32[o4 + 14] = v25;\n this.uint16[o2 + 30] = v26;\n this.uint16[o2 + 31] = v27;\n return i;\n }\n}\n\nStructArrayLayout8i15ui1ul2f2ui64.prototype.bytesPerElement = 64;\nregister('StructArrayLayout8i15ui1ul2f2ui64', StructArrayLayout8i15ui1ul2f2ui64);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[1]\n *\n */\nclass StructArrayLayout1f4 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0);\n }\n\n public emplace(i: number, v0: number) {\n const o4 = i * 1;\n this.float32[o4 + 0] = v0;\n return i;\n }\n}\n\nStructArrayLayout1f4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout1f4', StructArrayLayout1f4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[1]\n * [4]: Float32[2]\n *\n */\nclass StructArrayLayout1ui2f12 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o2 = i * 6;\n const o4 = i * 3;\n this.uint16[o2 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout1ui2f12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout1ui2f12', StructArrayLayout1ui2f12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint32[1]\n * [4]: Uint16[2]\n *\n */\nclass StructArrayLayout1ul2ui8 extends StructArray {\n uint8: Uint8Array;\n uint32: Uint32Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o4 = i * 2;\n const o2 = i * 4;\n this.uint32[o4 + 0] = v0;\n this.uint16[o2 + 2] = v1;\n this.uint16[o2 + 3] = v2;\n return i;\n }\n}\n\nStructArrayLayout1ul2ui8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout1ul2ui8', StructArrayLayout1ul2ui8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[2]\n *\n */\nclass StructArrayLayout2ui4 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1);\n }\n\n public emplace(i: number, v0: number, v1: number) {\n const o2 = i * 2;\n this.uint16[o2 + 0] = v0;\n this.uint16[o2 + 1] = v1;\n return i;\n }\n}\n\nStructArrayLayout2ui4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout2ui4', StructArrayLayout2ui4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[1]\n *\n */\nclass StructArrayLayout1ui2 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0);\n }\n\n public emplace(i: number, v0: number) {\n const o2 = i * 1;\n this.uint16[o2 + 0] = v0;\n return i;\n }\n}\n\nStructArrayLayout1ui2.prototype.bytesPerElement = 2;\nregister('StructArrayLayout1ui2', StructArrayLayout1ui2);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[4]\n *\n */\nclass StructArrayLayout4f16 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number) {\n const o4 = i * 4;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n this.float32[o4 + 3] = v3;\n return i;\n }\n}\n\nStructArrayLayout4f16.prototype.bytesPerElement = 16;\nregister('StructArrayLayout4f16', StructArrayLayout4f16);\n\n/** @internal */\nclass CollisionBoxStruct extends Struct {\n _structArray: CollisionBoxArray;\n get anchorPointX() { return this._structArray.int16[this._pos2 + 0]; }\n get anchorPointY() { return this._structArray.int16[this._pos2 + 1]; }\n get x1() { return this._structArray.int16[this._pos2 + 2]; }\n get y1() { return this._structArray.int16[this._pos2 + 3]; }\n get x2() { return this._structArray.int16[this._pos2 + 4]; }\n get y2() { return this._structArray.int16[this._pos2 + 5]; }\n get featureIndex() { return this._structArray.uint32[this._pos4 + 3]; }\n get sourceLayerIndex() { return this._structArray.uint16[this._pos2 + 8]; }\n get bucketIndex() { return this._structArray.uint16[this._pos2 + 9]; }\n get anchorPoint() { return new Point(this.anchorPointX, this.anchorPointY); }\n}\n\nCollisionBoxStruct.prototype.size = 20;\n\nexport type CollisionBox = CollisionBoxStruct;\n\n/** @internal */\nexport class CollisionBoxArray extends StructArrayLayout6i1ul2ui20 {\n /**\n * Return the CollisionBoxStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): CollisionBoxStruct {\n return new CollisionBoxStruct(this, index);\n }\n}\n\nregister('CollisionBoxArray', CollisionBoxArray);\n\n/** @internal */\nclass PlacedSymbolStruct extends Struct {\n _structArray: PlacedSymbolArray;\n get anchorX() { return this._structArray.int16[this._pos2 + 0]; }\n get anchorY() { return this._structArray.int16[this._pos2 + 1]; }\n get glyphStartIndex() { return this._structArray.uint16[this._pos2 + 2]; }\n get numGlyphs() { return this._structArray.uint16[this._pos2 + 3]; }\n get vertexStartIndex() { return this._structArray.uint32[this._pos4 + 2]; }\n get lineStartIndex() { return this._structArray.uint32[this._pos4 + 3]; }\n get lineLength() { return this._structArray.uint32[this._pos4 + 4]; }\n get segment() { return this._structArray.uint16[this._pos2 + 10]; }\n get lowerSize() { return this._structArray.uint16[this._pos2 + 11]; }\n get upperSize() { return this._structArray.uint16[this._pos2 + 12]; }\n get lineOffsetX() { return this._structArray.float32[this._pos4 + 7]; }\n get lineOffsetY() { return this._structArray.float32[this._pos4 + 8]; }\n get writingMode() { return this._structArray.uint8[this._pos1 + 36]; }\n get placedOrientation() { return this._structArray.uint8[this._pos1 + 37]; }\n set placedOrientation(x: number) { this._structArray.uint8[this._pos1 + 37] = x; }\n get hidden() { return this._structArray.uint8[this._pos1 + 38]; }\n set hidden(x: number) { this._structArray.uint8[this._pos1 + 38] = x; }\n get crossTileID() { return this._structArray.uint32[this._pos4 + 10]; }\n set crossTileID(x: number) { this._structArray.uint32[this._pos4 + 10] = x; }\n get associatedIconIndex() { return this._structArray.int16[this._pos2 + 22]; }\n}\n\nPlacedSymbolStruct.prototype.size = 48;\n\nexport type PlacedSymbol = PlacedSymbolStruct;\n\n/** @internal */\nexport class PlacedSymbolArray extends StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 {\n /**\n * Return the PlacedSymbolStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): PlacedSymbolStruct {\n return new PlacedSymbolStruct(this, index);\n }\n}\n\nregister('PlacedSymbolArray', PlacedSymbolArray);\n\n/** @internal */\nclass SymbolInstanceStruct extends Struct {\n _structArray: SymbolInstanceArray;\n get anchorX() { return this._structArray.int16[this._pos2 + 0]; }\n get anchorY() { return this._structArray.int16[this._pos2 + 1]; }\n get rightJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 2]; }\n get centerJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 3]; }\n get leftJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 4]; }\n get verticalPlacedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 5]; }\n get placedIconSymbolIndex() { return this._structArray.int16[this._pos2 + 6]; }\n get verticalPlacedIconSymbolIndex() { return this._structArray.int16[this._pos2 + 7]; }\n get key() { return this._structArray.uint16[this._pos2 + 8]; }\n get textBoxStartIndex() { return this._structArray.uint16[this._pos2 + 9]; }\n get textBoxEndIndex() { return this._structArray.uint16[this._pos2 + 10]; }\n get verticalTextBoxStartIndex() { return this._structArray.uint16[this._pos2 + 11]; }\n get verticalTextBoxEndIndex() { return this._structArray.uint16[this._pos2 + 12]; }\n get iconBoxStartIndex() { return this._structArray.uint16[this._pos2 + 13]; }\n get iconBoxEndIndex() { return this._structArray.uint16[this._pos2 + 14]; }\n get verticalIconBoxStartIndex() { return this._structArray.uint16[this._pos2 + 15]; }\n get verticalIconBoxEndIndex() { return this._structArray.uint16[this._pos2 + 16]; }\n get featureIndex() { return this._structArray.uint16[this._pos2 + 17]; }\n get numHorizontalGlyphVertices() { return this._structArray.uint16[this._pos2 + 18]; }\n get numVerticalGlyphVertices() { return this._structArray.uint16[this._pos2 + 19]; }\n get numIconVertices() { return this._structArray.uint16[this._pos2 + 20]; }\n get numVerticalIconVertices() { return this._structArray.uint16[this._pos2 + 21]; }\n get useRuntimeCollisionCircles() { return this._structArray.uint16[this._pos2 + 22]; }\n get crossTileID() { return this._structArray.uint32[this._pos4 + 12]; }\n set crossTileID(x: number) { this._structArray.uint32[this._pos4 + 12] = x; }\n get textBoxScale() { return this._structArray.float32[this._pos4 + 13]; }\n get collisionCircleDiameter() { return this._structArray.float32[this._pos4 + 14]; }\n get textAnchorOffsetStartIndex() { return this._structArray.uint16[this._pos2 + 30]; }\n get textAnchorOffsetEndIndex() { return this._structArray.uint16[this._pos2 + 31]; }\n}\n\nSymbolInstanceStruct.prototype.size = 64;\n\nexport type SymbolInstance = SymbolInstanceStruct;\n\n/** @internal */\nexport class SymbolInstanceArray extends StructArrayLayout8i15ui1ul2f2ui64 {\n /**\n * Return the SymbolInstanceStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): SymbolInstanceStruct {\n return new SymbolInstanceStruct(this, index);\n }\n}\n\nregister('SymbolInstanceArray', SymbolInstanceArray);\n\n/** @internal */\nexport class GlyphOffsetArray extends StructArrayLayout1f4 {\n getoffsetX(index: number) { return this.float32[index * 1 + 0]; }\n}\n\nregister('GlyphOffsetArray', GlyphOffsetArray);\n\n/** @internal */\nexport class SymbolLineVertexArray extends StructArrayLayout3i6 {\n getx(index: number) { return this.int16[index * 3 + 0]; }\n gety(index: number) { return this.int16[index * 3 + 1]; }\n gettileUnitDistanceFromAnchor(index: number) { return this.int16[index * 3 + 2]; }\n}\n\nregister('SymbolLineVertexArray', SymbolLineVertexArray);\n\n/** @internal */\nclass TextAnchorOffsetStruct extends Struct {\n _structArray: TextAnchorOffsetArray;\n get textAnchor() { return this._structArray.uint16[this._pos2 + 0]; }\n get textOffset0() { return this._structArray.float32[this._pos4 + 1]; }\n get textOffset1() { return this._structArray.float32[this._pos4 + 2]; }\n}\n\nTextAnchorOffsetStruct.prototype.size = 12;\n\nexport type TextAnchorOffset = TextAnchorOffsetStruct;\n\n/** @internal */\nexport class TextAnchorOffsetArray extends StructArrayLayout1ui2f12 {\n /**\n * Return the TextAnchorOffsetStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): TextAnchorOffsetStruct {\n return new TextAnchorOffsetStruct(this, index);\n }\n}\n\nregister('TextAnchorOffsetArray', TextAnchorOffsetArray);\n\n/** @internal */\nclass FeatureIndexStruct extends Struct {\n _structArray: FeatureIndexArray;\n get featureIndex() { return this._structArray.uint32[this._pos4 + 0]; }\n get sourceLayerIndex() { return this._structArray.uint16[this._pos2 + 2]; }\n get bucketIndex() { return this._structArray.uint16[this._pos2 + 3]; }\n}\n\nFeatureIndexStruct.prototype.size = 8;\n\nexport type FeatureIndex = FeatureIndexStruct;\n\n/** @internal */\nexport class FeatureIndexArray extends StructArrayLayout1ul2ui8 {\n /**\n * Return the FeatureIndexStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): FeatureIndexStruct {\n return new FeatureIndexStruct(this, index);\n }\n}\n\nregister('FeatureIndexArray', FeatureIndexArray);\n\nexport class PosArray extends StructArrayLayout2i4 {}\nexport class Pos3dArray extends StructArrayLayout3i6 {}\nexport class RasterBoundsArray extends StructArrayLayout4i8 {}\nexport class CircleLayoutArray extends StructArrayLayout2i4 {}\nexport class FillLayoutArray extends StructArrayLayout2i4 {}\nexport class FillExtrusionLayoutArray extends StructArrayLayout2i4i12 {}\nexport class HeatmapLayoutArray extends StructArrayLayout2i4 {}\nexport class LineLayoutArray extends StructArrayLayout2i4ub8 {}\nexport class LineExtLayoutArray extends StructArrayLayout2f8 {}\nexport class PatternLayoutArray extends StructArrayLayout10ui20 {}\nexport class SymbolLayoutArray extends StructArrayLayout4i4ui4i24 {}\nexport class SymbolDynamicLayoutArray extends StructArrayLayout3f12 {}\nexport class SymbolOpacityArray extends StructArrayLayout1ul4 {}\nexport class CollisionBoxLayoutArray extends StructArrayLayout2i2i2i12 {}\nexport class CollisionCircleLayoutArray extends StructArrayLayout2f1f2i16 {}\nexport class CollisionVertexArray extends StructArrayLayout2ub2f12 {}\nexport class QuadTriangleArray extends StructArrayLayout3ui6 {}\nexport class TriangleIndexArray extends StructArrayLayout3ui6 {}\nexport class LineIndexArray extends StructArrayLayout2ui4 {}\nexport class LineStripIndexArray extends StructArrayLayout1ui2 {}\nexport {\n StructArrayLayout2i4,\n StructArrayLayout3i6,\n StructArrayLayout4i8,\n StructArrayLayout2i4i12,\n StructArrayLayout2i4ub8,\n StructArrayLayout2f8,\n StructArrayLayout10ui20,\n StructArrayLayout4i4ui4i24,\n StructArrayLayout3f12,\n StructArrayLayout1ul4,\n StructArrayLayout6i1ul2ui20,\n StructArrayLayout2i2i2i12,\n StructArrayLayout2f1f2i16,\n StructArrayLayout2ub2f12,\n StructArrayLayout3ui6,\n StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48,\n StructArrayLayout8i15ui1ul2f2ui64,\n StructArrayLayout1f4,\n StructArrayLayout1ui2f12,\n StructArrayLayout1ul2ui8,\n StructArrayLayout2ui4,\n StructArrayLayout1ui2,\n StructArrayLayout4f16\n};\n","import {createLayout} from '../../util/struct_array';\n\nconst layout = createLayout([\n {name: 'a_pos', components: 2, type: 'Int16'}\n], 4);\n\nexport default layout;\nexport const {members, size, alignment} = layout;\n","import {warnOnce} from '../util/util';\n\nimport {register} from '../util/web_worker_transfer';\n\nimport type {VertexArrayObject} from '../render/vertex_array_object';\nimport type {StructArray} from '../util/struct_array';\n\n/**\n * @internal\n * A single segment of a vector\n */\nexport type Segment = {\n sortKey?: number;\n vertexOffset: number;\n primitiveOffset: number;\n vertexLength: number;\n primitiveLength: number;\n vaos: {[_: string]: VertexArrayObject};\n};\n\n/**\n * @internal\n * Used for calculations on vector segments\n */\nexport class SegmentVector {\n static MAX_VERTEX_ARRAY_LENGTH: number;\n segments: Array;\n\n constructor(segments: Array = []) {\n this.segments = segments;\n }\n\n prepareSegment(\n numVertices: number,\n layoutVertexArray: StructArray,\n indexArray: StructArray,\n sortKey?: number\n ): Segment {\n let segment: Segment = this.segments[this.segments.length - 1];\n if (numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) warnOnce(`Max vertices per segment is ${SegmentVector.MAX_VERTEX_ARRAY_LENGTH}: bucket requested ${numVertices}`);\n if (!segment || segment.vertexLength + numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH || segment.sortKey !== sortKey) {\n segment = ({\n vertexOffset: layoutVertexArray.length,\n primitiveOffset: indexArray.length,\n vertexLength: 0,\n primitiveLength: 0\n } as any);\n if (sortKey !== undefined) segment.sortKey = sortKey;\n this.segments.push(segment);\n }\n return segment;\n }\n\n get() {\n return this.segments;\n }\n\n destroy() {\n for (const segment of this.segments) {\n for (const k in segment.vaos) {\n segment.vaos[k].destroy();\n }\n }\n }\n\n static simpleSegment(\n vertexOffset: number,\n primitiveOffset: number,\n vertexLength: number,\n primitiveLength: number\n ): SegmentVector {\n return new SegmentVector([{\n vertexOffset,\n primitiveOffset,\n vertexLength,\n primitiveLength,\n vaos: {},\n sortKey: 0\n }]);\n }\n}\n\n/**\n * The maximum size of a vertex array. This limit is imposed by WebGL's 16 bit\n * addressing of vertex buffers.\n */\nSegmentVector.MAX_VERTEX_ARRAY_LENGTH = Math.pow(2, 16) - 1;\n\nregister('SegmentVector', SegmentVector);\n","import {clamp} from '../util/util';\n\n/**\n * Packs two numbers, interpreted as 8-bit unsigned integers, into a single\n * float. Unpack them in the shader using the `unpack_float()` function,\n * defined in _prelude.vertex.glsl\n */\nexport function packUint8ToFloat(a: number, b: number) {\n // coerce a and b to 8-bit ints\n a = clamp(Math.floor(a), 0, 255);\n b = clamp(Math.floor(b), 0, 255);\n return 256 * a + b;\n}\n","import {createLayout} from '../../util/struct_array';\n\nexport const patternAttributes = createLayout([\n // [tl.x, tl.y, br.x, br.y]\n {name: 'a_pattern_from', components: 4, type: 'Uint16'},\n {name: 'a_pattern_to', components: 4, type: 'Uint16'},\n {name: 'a_pixel_ratio_from', components: 1, type: 'Uint16'},\n {name: 'a_pixel_ratio_to', components: 1, type: 'Uint16'},\n]);\n","/**\n * JS Implementation of MurmurHash3 (r136) (as of May 20, 2011)\n * \n * @author Gary Court\n * @see http://github.com/garycourt/murmurhash-js\n * @author Austin Appleby\n * @see http://sites.google.com/site/murmurhash/\n * \n * @param {string} key ASCII only\n * @param {number} seed Positive integer only\n * @return {number} 32-bit positive integer hash \n */\n\nfunction murmurhash3_32_gc(key, seed) {\n\tvar remainder, bytes, h1, h1b, c1, c1b, c2, c2b, k1, i;\n\t\n\tremainder = key.length & 3; // key.length % 4\n\tbytes = key.length - remainder;\n\th1 = seed;\n\tc1 = 0xcc9e2d51;\n\tc2 = 0x1b873593;\n\ti = 0;\n\t\n\twhile (i < bytes) {\n\t \tk1 = \n\t \t ((key.charCodeAt(i) & 0xff)) |\n\t \t ((key.charCodeAt(++i) & 0xff) << 8) |\n\t \t ((key.charCodeAt(++i) & 0xff) << 16) |\n\t \t ((key.charCodeAt(++i) & 0xff) << 24);\n\t\t++i;\n\t\t\n\t\tk1 = ((((k1 & 0xffff) * c1) + ((((k1 >>> 16) * c1) & 0xffff) << 16))) & 0xffffffff;\n\t\tk1 = (k1 << 15) | (k1 >>> 17);\n\t\tk1 = ((((k1 & 0xffff) * c2) + ((((k1 >>> 16) * c2) & 0xffff) << 16))) & 0xffffffff;\n\n\t\th1 ^= k1;\n h1 = (h1 << 13) | (h1 >>> 19);\n\t\th1b = ((((h1 & 0xffff) * 5) + ((((h1 >>> 16) * 5) & 0xffff) << 16))) & 0xffffffff;\n\t\th1 = (((h1b & 0xffff) + 0x6b64) + ((((h1b >>> 16) + 0xe654) & 0xffff) << 16));\n\t}\n\t\n\tk1 = 0;\n\t\n\tswitch (remainder) {\n\t\tcase 3: k1 ^= (key.charCodeAt(i + 2) & 0xff) << 16;\n\t\tcase 2: k1 ^= (key.charCodeAt(i + 1) & 0xff) << 8;\n\t\tcase 1: k1 ^= (key.charCodeAt(i) & 0xff);\n\t\t\n\t\tk1 = (((k1 & 0xffff) * c1) + ((((k1 >>> 16) * c1) & 0xffff) << 16)) & 0xffffffff;\n\t\tk1 = (k1 << 15) | (k1 >>> 17);\n\t\tk1 = (((k1 & 0xffff) * c2) + ((((k1 >>> 16) * c2) & 0xffff) << 16)) & 0xffffffff;\n\t\th1 ^= k1;\n\t}\n\t\n\th1 ^= key.length;\n\n\th1 ^= h1 >>> 16;\n\th1 = (((h1 & 0xffff) * 0x85ebca6b) + ((((h1 >>> 16) * 0x85ebca6b) & 0xffff) << 16)) & 0xffffffff;\n\th1 ^= h1 >>> 13;\n\th1 = ((((h1 & 0xffff) * 0xc2b2ae35) + ((((h1 >>> 16) * 0xc2b2ae35) & 0xffff) << 16))) & 0xffffffff;\n\th1 ^= h1 >>> 16;\n\n\treturn h1 >>> 0;\n}\n\nif(typeof module !== \"undefined\") {\n module.exports = murmurhash3_32_gc\n}","/**\n * JS Implementation of MurmurHash2\n * \n * @author Gary Court\n * @see http://github.com/garycourt/murmurhash-js\n * @author Austin Appleby\n * @see http://sites.google.com/site/murmurhash/\n * \n * @param {string} str ASCII only\n * @param {number} seed Positive integer only\n * @return {number} 32-bit positive integer hash\n */\n\nfunction murmurhash2_32_gc(str, seed) {\n var\n l = str.length,\n h = seed ^ l,\n i = 0,\n k;\n \n while (l >= 4) {\n \tk = \n \t ((str.charCodeAt(i) & 0xff)) |\n \t ((str.charCodeAt(++i) & 0xff) << 8) |\n \t ((str.charCodeAt(++i) & 0xff) << 16) |\n \t ((str.charCodeAt(++i) & 0xff) << 24);\n \n k = (((k & 0xffff) * 0x5bd1e995) + ((((k >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n k ^= k >>> 24;\n k = (((k & 0xffff) * 0x5bd1e995) + ((((k >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n\n\th = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16)) ^ k;\n\n l -= 4;\n ++i;\n }\n \n switch (l) {\n case 3: h ^= (str.charCodeAt(i + 2) & 0xff) << 16;\n case 2: h ^= (str.charCodeAt(i + 1) & 0xff) << 8;\n case 1: h ^= (str.charCodeAt(i) & 0xff);\n h = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n }\n\n h ^= h >>> 13;\n h = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n h ^= h >>> 15;\n\n return h >>> 0;\n}\n\nif(typeof module !== undefined) {\n module.exports = murmurhash2_32_gc\n}\n","var murmur3 = require(\"./murmurhash3_gc.js\")\nvar murmur2 = require(\"./murmurhash2_gc.js\")\n\nmodule.exports = murmur3\nmodule.exports.murmur3 = murmur3\nmodule.exports.murmur2 = murmur2\n","import murmur3 from 'murmurhash-js';\nimport {register} from '../util/web_worker_transfer';\n\ntype SerializedFeaturePositionMap = {\n ids: Float64Array;\n positions: Uint32Array;\n};\n\ntype FeaturePosition = {\n index: number;\n start: number;\n end: number;\n};\n\n// A transferable data structure that maps feature ids to their indices and buffer offsets\nexport class FeaturePositionMap {\n ids: Array;\n positions: Array;\n indexed: boolean;\n\n constructor() {\n this.ids = [];\n this.positions = [];\n this.indexed = false;\n }\n\n add(id: unknown, index: number, start: number, end: number) {\n this.ids.push(getNumericId(id));\n this.positions.push(index, start, end);\n }\n\n getPositions(id: unknown): Array {\n if (!this.indexed) throw new Error('Trying to get index, but feature positions are not indexed');\n\n const intId = getNumericId(id);\n\n // binary search for the first occurrence of id in this.ids;\n // relies on ids/positions being sorted by id, which happens in serialization\n let i = 0;\n let j = this.ids.length - 1;\n while (i < j) {\n const m = (i + j) >> 1;\n if (this.ids[m] >= intId) {\n j = m;\n } else {\n i = m + 1;\n }\n }\n const positions = [];\n while (this.ids[i] === intId) {\n const index = this.positions[3 * i];\n const start = this.positions[3 * i + 1];\n const end = this.positions[3 * i + 2];\n positions.push({index, start, end});\n i++;\n }\n return positions;\n }\n\n static serialize(map: FeaturePositionMap, transferables: Array): SerializedFeaturePositionMap {\n const ids = new Float64Array(map.ids);\n const positions = new Uint32Array(map.positions);\n\n sort(ids, positions, 0, ids.length - 1);\n\n if (transferables) {\n transferables.push(ids.buffer, positions.buffer);\n }\n\n return {ids, positions};\n }\n\n static deserialize(obj: SerializedFeaturePositionMap): FeaturePositionMap {\n const map = new FeaturePositionMap();\n // after transferring, we only use these arrays statically (no pushes),\n // so TypedArray vs Array distinction that flow points out doesn't matter\n map.ids = (obj.ids as any);\n map.positions = (obj.positions as any);\n map.indexed = true;\n return map;\n }\n}\n\nfunction getNumericId(value: unknown) {\n const numValue = +value;\n if (!isNaN(numValue) && numValue <= Number.MAX_SAFE_INTEGER) {\n return numValue;\n }\n return murmur3(String(value));\n}\n\n// custom quicksort that sorts ids, indices and offsets together (by ids)\n// uses Hoare partitioning & manual tail call optimization to avoid worst case scenarios\nfunction sort(ids, positions, left, right) {\n while (left < right) {\n const pivot = ids[(left + right) >> 1];\n let i = left - 1;\n let j = right + 1;\n\n while (true) {\n do i++; while (ids[i] < pivot);\n do j--; while (ids[j] > pivot);\n if (i >= j) break;\n swap(ids, i, j);\n swap(positions, 3 * i, 3 * j);\n swap(positions, 3 * i + 1, 3 * j + 1);\n swap(positions, 3 * i + 2, 3 * j + 2);\n }\n\n if (j - left < right - j) {\n sort(ids, positions, left, j);\n left = j + 1;\n } else {\n sort(ids, positions, j + 1, right);\n right = j;\n }\n }\n}\n\nfunction swap(arr, i, j) {\n const tmp = arr[i];\n arr[i] = arr[j];\n arr[j] = tmp;\n}\n\nregister('FeaturePositionMap', FeaturePositionMap);\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {Context} from '../gl/context';\nimport {mat4, vec2, vec3, vec4} from 'gl-matrix';\n\ntype $ObjMap any> = {\n [K in keyof T]: F extends (v: T[K]) => infer R ? R : never;\n};\n\nexport type UniformValues = $ObjMap(u: Uniform) => V>;\nexport type UniformLocations = {[_: string]: WebGLUniformLocation};\n\n/**\n * @internal\n * A base uniform abstract class\n */\nabstract class Uniform {\n gl: WebGLRenderingContext|WebGL2RenderingContext;\n location: WebGLUniformLocation;\n current: T;\n\n constructor(context: Context, location: WebGLUniformLocation) {\n this.gl = context.gl;\n this.location = location;\n }\n\n abstract set(v: T): void;\n}\n\nclass Uniform1i extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = 0;\n }\n\n set(v: number): void {\n if (this.current !== v) {\n this.current = v;\n this.gl.uniform1i(this.location, v);\n }\n }\n}\n\nclass Uniform1f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = 0;\n }\n\n set(v: number): void {\n if (this.current !== v) {\n this.current = v;\n this.gl.uniform1f(this.location, v);\n }\n }\n}\n\nclass Uniform2f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = [0, 0];\n }\n\n set(v: vec2): void {\n if (v[0] !== this.current[0] || v[1] !== this.current[1]) {\n this.current = v;\n this.gl.uniform2f(this.location, v[0], v[1]);\n }\n }\n}\n\nclass Uniform3f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = [0, 0, 0];\n }\n\n set(v: vec3): void {\n if (v[0] !== this.current[0] || v[1] !== this.current[1] || v[2] !== this.current[2]) {\n this.current = v;\n this.gl.uniform3f(this.location, v[0], v[1], v[2]);\n }\n }\n}\n\nclass Uniform4f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = [0, 0, 0, 0];\n }\n\n set(v: vec4): void {\n if (v[0] !== this.current[0] || v[1] !== this.current[1] ||\n v[2] !== this.current[2] || v[3] !== this.current[3]) {\n this.current = v;\n this.gl.uniform4f(this.location, v[0], v[1], v[2], v[3]);\n }\n }\n}\n\nclass UniformColor extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = Color.transparent;\n }\n\n set(v: Color): void {\n if (v.r !== this.current.r || v.g !== this.current.g ||\n v.b !== this.current.b || v.a !== this.current.a) {\n this.current = v;\n this.gl.uniform4f(this.location, v.r, v.g, v.b, v.a);\n }\n }\n}\n\nconst emptyMat4 = new Float32Array(16) as mat4;\nclass UniformMatrix4f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = emptyMat4;\n }\n\n set(v: mat4): void {\n // The vast majority of matrix comparisons that will trip this set\n // happen at i=12 or i=0, so we check those first to avoid lots of\n // unnecessary iteration:\n if (v[12] !== this.current[12] || v[0] !== this.current[0]) {\n this.current = v;\n this.gl.uniformMatrix4fv(this.location, false, v);\n return;\n }\n for (let i = 1; i < 16; i++) {\n if (v[i] !== this.current[i]) {\n this.current = v;\n this.gl.uniformMatrix4fv(this.location, false, v);\n break;\n }\n }\n }\n}\n\nexport {\n Uniform,\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f,\n Uniform4f,\n UniformColor,\n UniformMatrix4f\n};\n\n/**\n * @internal\n * A uniform bindings\n */\nexport type UniformBindings = {[_: string]: Uniform};\n","import {packUint8ToFloat} from '../shaders/encode_attribute';\nimport {Color, supportsPropertyExpression} from '@maplibre/maplibre-gl-style-spec';\nimport {register} from '../util/web_worker_transfer';\nimport {PossiblyEvaluatedPropertyValue} from '../style/properties';\nimport {StructArrayLayout1f4, StructArrayLayout2f8, StructArrayLayout4f16, PatternLayoutArray} from './array_types.g';\nimport {clamp} from '../util/util';\nimport {patternAttributes} from './bucket/pattern_attributes';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {FeaturePositionMap} from './feature_position_map';\nimport {Uniform, Uniform1f, UniformColor, Uniform4f} from '../render/uniform_binding';\n\nimport type {UniformLocations} from '../render/uniform_binding';\n\nimport type {CanonicalTileID} from '../source/tile_id';\nimport type {Context} from '../gl/context';\nimport type {TypedStyleLayer} from '../style/style_layer/typed_style_layer';\nimport type {CrossfadeParameters} from '../style/evaluation_parameters';\nimport type {StructArray, StructArrayMember} from '../util/struct_array';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {ImagePosition} from '../render/image_atlas';\nimport type {\n Feature,\n FeatureState,\n GlobalProperties,\n SourceExpression,\n CompositeExpression,\n FormattedSection\n} from '@maplibre/maplibre-gl-style-spec';\nimport type {FeatureStates} from '../source/source_state';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nexport type BinderUniform = {\n name: string;\n property: string;\n binding: Uniform;\n};\n\nfunction packColor(color: Color): [number, number] {\n return [\n packUint8ToFloat(255 * color.r, 255 * color.g),\n packUint8ToFloat(255 * color.b, 255 * color.a)\n ];\n}\n\n/**\n * `Binder` is the interface definition for the strategies for constructing,\n * uploading, and binding paint property data as GLSL attributes. Most style-\n * spec properties have a 1:1 relationship to shader attribute/uniforms, but\n * some require multiple values per feature to be passed to the GPU, and in\n * those cases we bind multiple attributes/uniforms.\n *\n * It has three implementations, one for each of the three strategies we use:\n *\n * * For _constant_ properties -- those whose value is a constant, or the constant\n * result of evaluating a camera expression at a particular camera position -- we\n * don't need a vertex attribute buffer, and instead use a uniform.\n * * For data expressions, we use a vertex buffer with a single attribute value,\n * the evaluated result of the source function for the given feature.\n * * For composite expressions, we use a vertex buffer with two attributes: min and\n * max values covering the range of zooms at which we expect the tile to be\n * displayed. These values are calculated by evaluating the composite expression for\n * the given feature at strategically chosen zoom levels. In addition to this\n * attribute data, we also use a uniform value which the shader uses to interpolate\n * between the min and max value at the final displayed zoom level. The use of a\n * uniform allows us to cheaply update the value on every frame.\n *\n * Note that the shader source varies depending on whether we're using a uniform or\n * attribute. We dynamically compile shaders at runtime to accommodate this.\n */\ninterface AttributeBinder {\n populatePaintArray(\n length: number,\n feature: Feature,\n imagePositions: {[_: string]: ImagePosition},\n canonical?: CanonicalTileID,\n formattedSection?: FormattedSection\n ): void;\n updatePaintArray(\n start: number,\n length: number,\n feature: Feature,\n featureState: FeatureState,\n imagePositions: {[_: string]: ImagePosition}\n ): void;\n upload(a: Context): void;\n destroy(): void;\n}\n\ninterface UniformBinder {\n uniformNames: Array;\n setUniform(\n uniform: Uniform,\n globals: GlobalProperties,\n currentValue: PossiblyEvaluatedPropertyValue,\n uniformName: string\n ): void;\n getBinding(context: Context, location: WebGLUniformLocation, name: string): Partial>;\n}\n\nclass ConstantBinder implements UniformBinder {\n value: unknown;\n type: string;\n uniformNames: Array;\n\n constructor(value: unknown, names: Array, type: string) {\n this.value = value;\n this.uniformNames = names.map(name => `u_${name}`);\n this.type = type;\n }\n\n setUniform(\n uniform: Uniform,\n globals: GlobalProperties,\n currentValue: PossiblyEvaluatedPropertyValue\n ): void {\n uniform.set(currentValue.constantOr(this.value));\n }\n\n getBinding(context: Context, location: WebGLUniformLocation, _: string): Partial> {\n return (this.type === 'color') ?\n new UniformColor(context, location) :\n new Uniform1f(context, location);\n }\n}\n\nclass CrossFadedConstantBinder implements UniformBinder {\n uniformNames: Array;\n patternFrom: Array;\n patternTo: Array;\n pixelRatioFrom: number;\n pixelRatioTo: number;\n\n constructor(value: unknown, names: Array) {\n this.uniformNames = names.map(name => `u_${name}`);\n this.patternFrom = null;\n this.patternTo = null;\n this.pixelRatioFrom = 1.0;\n this.pixelRatioTo = 1.0;\n }\n\n setConstantPatternPositions(posTo: ImagePosition, posFrom: ImagePosition) {\n this.pixelRatioFrom = posFrom.pixelRatio;\n this.pixelRatioTo = posTo.pixelRatio;\n this.patternFrom = posFrom.tlbr;\n this.patternTo = posTo.tlbr;\n }\n\n setUniform(uniform: Uniform, globals: GlobalProperties, currentValue: PossiblyEvaluatedPropertyValue, uniformName: string) {\n const pos =\n uniformName === 'u_pattern_to' ? this.patternTo :\n uniformName === 'u_pattern_from' ? this.patternFrom :\n uniformName === 'u_pixel_ratio_to' ? this.pixelRatioTo :\n uniformName === 'u_pixel_ratio_from' ? this.pixelRatioFrom : null;\n if (pos) uniform.set(pos);\n }\n\n getBinding(context: Context, location: WebGLUniformLocation, name: string): Partial> {\n return name.substr(0, 9) === 'u_pattern' ?\n new Uniform4f(context, location) :\n new Uniform1f(context, location);\n }\n}\n\nclass SourceExpressionBinder implements AttributeBinder {\n expression: SourceExpression;\n type: string;\n maxValue: number;\n\n paintVertexArray: StructArray;\n paintVertexAttributes: Array;\n paintVertexBuffer: VertexBuffer;\n\n constructor(expression: SourceExpression, names: Array, type: string, PaintVertexArray: {\n new (...args: any): StructArray;\n }) {\n this.expression = expression;\n this.type = type;\n this.maxValue = 0;\n this.paintVertexAttributes = names.map((name) => ({\n name: `a_${name}`,\n type: 'Float32',\n components: type === 'color' ? 2 : 1,\n offset: 0\n }));\n this.paintVertexArray = new PaintVertexArray();\n }\n\n populatePaintArray(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) {\n const start = this.paintVertexArray.length;\n const value = this.expression.evaluate(new EvaluationParameters(0), feature, {}, canonical, [], formattedSection);\n this.paintVertexArray.resize(newLength);\n this._setPaintValue(start, newLength, value);\n }\n\n updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState) {\n const value = this.expression.evaluate({zoom: 0}, feature, featureState);\n this._setPaintValue(start, end, value);\n }\n\n _setPaintValue(start, end, value) {\n if (this.type === 'color') {\n const color = packColor(value);\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, color[0], color[1]);\n }\n } else {\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, value);\n }\n this.maxValue = Math.max(this.maxValue, Math.abs(value));\n }\n }\n\n upload(context: Context) {\n if (this.paintVertexArray && this.paintVertexArray.arrayBuffer) {\n if (this.paintVertexBuffer && this.paintVertexBuffer.buffer) {\n this.paintVertexBuffer.updateData(this.paintVertexArray);\n } else {\n this.paintVertexBuffer = context.createVertexBuffer(this.paintVertexArray, this.paintVertexAttributes, this.expression.isStateDependent);\n }\n }\n }\n\n destroy() {\n if (this.paintVertexBuffer) {\n this.paintVertexBuffer.destroy();\n }\n }\n}\n\nclass CompositeExpressionBinder implements AttributeBinder, UniformBinder {\n expression: CompositeExpression;\n uniformNames: Array;\n type: string;\n useIntegerZoom: boolean;\n zoom: number;\n maxValue: number;\n\n paintVertexArray: StructArray;\n paintVertexAttributes: Array;\n paintVertexBuffer: VertexBuffer;\n\n constructor(expression: CompositeExpression, names: Array, type: string, useIntegerZoom: boolean, zoom: number, PaintVertexArray: {\n new (...args: any): StructArray;\n }) {\n this.expression = expression;\n this.uniformNames = names.map(name => `u_${name}_t`);\n this.type = type;\n this.useIntegerZoom = useIntegerZoom;\n this.zoom = zoom;\n this.maxValue = 0;\n this.paintVertexAttributes = names.map((name) => ({\n name: `a_${name}`,\n type: 'Float32',\n components: type === 'color' ? 4 : 2,\n offset: 0\n }));\n this.paintVertexArray = new PaintVertexArray();\n }\n\n populatePaintArray(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) {\n const min = this.expression.evaluate(new EvaluationParameters(this.zoom), feature, {}, canonical, [], formattedSection);\n const max = this.expression.evaluate(new EvaluationParameters(this.zoom + 1), feature, {}, canonical, [], formattedSection);\n const start = this.paintVertexArray.length;\n this.paintVertexArray.resize(newLength);\n this._setPaintValue(start, newLength, min, max);\n }\n\n updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState) {\n const min = this.expression.evaluate({zoom: this.zoom}, feature, featureState);\n const max = this.expression.evaluate({zoom: this.zoom + 1}, feature, featureState);\n this._setPaintValue(start, end, min, max);\n }\n\n _setPaintValue(start, end, min, max) {\n if (this.type === 'color') {\n const minColor = packColor(min);\n const maxColor = packColor(max);\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, minColor[0], minColor[1], maxColor[0], maxColor[1]);\n }\n } else {\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, min, max);\n }\n this.maxValue = Math.max(this.maxValue, Math.abs(min), Math.abs(max));\n }\n }\n\n upload(context: Context) {\n if (this.paintVertexArray && this.paintVertexArray.arrayBuffer) {\n if (this.paintVertexBuffer && this.paintVertexBuffer.buffer) {\n this.paintVertexBuffer.updateData(this.paintVertexArray);\n } else {\n this.paintVertexBuffer = context.createVertexBuffer(this.paintVertexArray, this.paintVertexAttributes, this.expression.isStateDependent);\n }\n }\n }\n\n destroy() {\n if (this.paintVertexBuffer) {\n this.paintVertexBuffer.destroy();\n }\n }\n\n setUniform(uniform: Uniform, globals: GlobalProperties): void {\n const currentZoom = this.useIntegerZoom ? Math.floor(globals.zoom) : globals.zoom;\n const factor = clamp(this.expression.interpolationFactor(currentZoom, this.zoom, this.zoom + 1), 0, 1);\n uniform.set(factor);\n }\n\n getBinding(context: Context, location: WebGLUniformLocation, _: string): Uniform1f {\n return new Uniform1f(context, location);\n }\n}\n\nclass CrossFadedCompositeBinder implements AttributeBinder {\n expression: CompositeExpression;\n type: string;\n useIntegerZoom: boolean;\n zoom: number;\n layerId: string;\n\n zoomInPaintVertexArray: StructArray;\n zoomOutPaintVertexArray: StructArray;\n zoomInPaintVertexBuffer: VertexBuffer;\n zoomOutPaintVertexBuffer: VertexBuffer;\n paintVertexAttributes: Array;\n\n constructor(expression: CompositeExpression, type: string, useIntegerZoom: boolean, zoom: number, PaintVertexArray: {\n new (...args: any): StructArray;\n }, layerId: string) {\n this.expression = expression;\n this.type = type;\n this.useIntegerZoom = useIntegerZoom;\n this.zoom = zoom;\n this.layerId = layerId;\n\n this.zoomInPaintVertexArray = new PaintVertexArray();\n this.zoomOutPaintVertexArray = new PaintVertexArray();\n }\n\n populatePaintArray(length: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}) {\n const start = this.zoomInPaintVertexArray.length;\n this.zoomInPaintVertexArray.resize(length);\n this.zoomOutPaintVertexArray.resize(length);\n this._setPaintValues(start, length, feature.patterns && feature.patterns[this.layerId], imagePositions);\n }\n\n updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState, imagePositions: {[_: string]: ImagePosition}) {\n this._setPaintValues(start, end, feature.patterns && feature.patterns[this.layerId], imagePositions);\n }\n\n _setPaintValues(start, end, patterns, positions) {\n if (!positions || !patterns) return;\n\n const {min, mid, max} = patterns;\n const imageMin = positions[min];\n const imageMid = positions[mid];\n const imageMax = positions[max];\n if (!imageMin || !imageMid || !imageMax) return;\n\n // We populate two paint arrays because, for cross-faded properties, we don't know which direction\n // we're cross-fading to at layout time. In order to keep vertex attributes to a minimum and not pass\n // unnecessary vertex data to the shaders, we determine which to upload at draw time.\n for (let i = start; i < end; i++) {\n this.zoomInPaintVertexArray.emplace(i,\n imageMid.tl[0], imageMid.tl[1], imageMid.br[0], imageMid.br[1],\n imageMin.tl[0], imageMin.tl[1], imageMin.br[0], imageMin.br[1],\n imageMid.pixelRatio,\n imageMin.pixelRatio,\n );\n this.zoomOutPaintVertexArray.emplace(i,\n imageMid.tl[0], imageMid.tl[1], imageMid.br[0], imageMid.br[1],\n imageMax.tl[0], imageMax.tl[1], imageMax.br[0], imageMax.br[1],\n imageMid.pixelRatio,\n imageMax.pixelRatio,\n );\n }\n }\n\n upload(context: Context) {\n if (this.zoomInPaintVertexArray && this.zoomInPaintVertexArray.arrayBuffer && this.zoomOutPaintVertexArray && this.zoomOutPaintVertexArray.arrayBuffer) {\n this.zoomInPaintVertexBuffer = context.createVertexBuffer(this.zoomInPaintVertexArray, patternAttributes.members, this.expression.isStateDependent);\n this.zoomOutPaintVertexBuffer = context.createVertexBuffer(this.zoomOutPaintVertexArray, patternAttributes.members, this.expression.isStateDependent);\n }\n }\n\n destroy() {\n if (this.zoomOutPaintVertexBuffer) this.zoomOutPaintVertexBuffer.destroy();\n if (this.zoomInPaintVertexBuffer) this.zoomInPaintVertexBuffer.destroy();\n }\n}\n\n/**\n * @internal\n * ProgramConfiguration contains the logic for binding style layer properties and tile\n * layer feature data into GL program uniforms and vertex attributes.\n *\n * Non-data-driven property values are bound to shader uniforms. Data-driven property\n * values are bound to vertex attributes. In order to support a uniform GLSL syntax over\n * both, [Mapbox GL Shaders](https://github.com/mapbox/mapbox-gl-shaders) defines a `#pragma`\n * abstraction, which ProgramConfiguration is responsible for implementing. At runtime,\n * it examines the attributes of a particular layer, combines this with fixed knowledge\n * about how layers of the particular type are implemented, and determines which uniforms\n * and vertex attributes will be required. It can then substitute the appropriate text\n * into the shader source code, create and link a program, and bind the uniforms and\n * vertex attributes in preparation for drawing.\n *\n * When a vector tile is parsed, this same configuration information is used to\n * populate the attribute buffers needed for data-driven styling using the zoom\n * level and feature property data.\n */\nexport class ProgramConfiguration {\n binders: {[_: string]: AttributeBinder | UniformBinder};\n cacheKey: string;\n\n _buffers: Array;\n\n constructor(layer: TypedStyleLayer, zoom: number, filterProperties: (_: string) => boolean) {\n this.binders = {};\n this._buffers = [];\n\n const keys = [];\n\n for (const property in layer.paint._values) {\n if (!filterProperties(property)) continue;\n const value = (layer.paint as any).get(property);\n if (!(value instanceof PossiblyEvaluatedPropertyValue) || !supportsPropertyExpression(value.property.specification)) {\n continue;\n }\n const names = paintAttributeNames(property, layer.type);\n const expression = value.value;\n const type = value.property.specification.type;\n const useIntegerZoom = (value.property as any).useIntegerZoom;\n const propType = value.property.specification['property-type'];\n const isCrossFaded = propType === 'cross-faded' || propType === 'cross-faded-data-driven';\n\n if (expression.kind === 'constant') {\n this.binders[property] = isCrossFaded ?\n new CrossFadedConstantBinder(expression.value, names) :\n new ConstantBinder(expression.value, names, type);\n keys.push(`/u_${property}`);\n\n } else if (expression.kind === 'source' || isCrossFaded) {\n const StructArrayLayout = layoutType(property, type, 'source');\n this.binders[property] = isCrossFaded ?\n new CrossFadedCompositeBinder(expression as CompositeExpression, type, useIntegerZoom, zoom, StructArrayLayout, layer.id) :\n new SourceExpressionBinder(expression as SourceExpression, names, type, StructArrayLayout);\n keys.push(`/a_${property}`);\n\n } else {\n const StructArrayLayout = layoutType(property, type, 'composite');\n this.binders[property] = new CompositeExpressionBinder(expression, names, type, useIntegerZoom, zoom, StructArrayLayout);\n keys.push(`/z_${property}`);\n }\n }\n\n this.cacheKey = keys.sort().join('');\n }\n\n getMaxValue(property: string): number {\n const binder = this.binders[property];\n return binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder ? binder.maxValue : 0;\n }\n\n populatePaintArrays(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder)\n (binder as AttributeBinder).populatePaintArray(newLength, feature, imagePositions, canonical, formattedSection);\n }\n }\n setConstantPatternPositions(posTo: ImagePosition, posFrom: ImagePosition) {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof CrossFadedConstantBinder)\n binder.setConstantPatternPositions(posTo, posFrom);\n }\n }\n\n updatePaintArrays(\n featureStates: FeatureStates,\n featureMap: FeaturePositionMap,\n vtLayer: VectorTileLayer,\n layer: TypedStyleLayer,\n imagePositions: {[_: string]: ImagePosition}\n ): boolean {\n let dirty: boolean = false;\n for (const id in featureStates) {\n const positions = featureMap.getPositions(id);\n\n for (const pos of positions) {\n const feature = vtLayer.feature(pos.index);\n\n for (const property in this.binders) {\n const binder = this.binders[property];\n if ((binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder ||\n binder instanceof CrossFadedCompositeBinder) && (binder as any).expression.isStateDependent === true) {\n //AHM: Remove after https://github.com/mapbox/mapbox-gl-js/issues/6255\n const value = (layer.paint as any).get(property);\n (binder as any).expression = value.value;\n (binder as AttributeBinder).updatePaintArray(pos.start, pos.end, feature, featureStates[id], imagePositions);\n dirty = true;\n }\n }\n }\n }\n return dirty;\n }\n\n defines(): Array {\n const result = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder) {\n result.push(...binder.uniformNames.map(name => `#define HAS_UNIFORM_${name}`));\n }\n }\n return result;\n }\n\n getBinderAttributes(): Array {\n const result = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder) {\n for (let i = 0; i < binder.paintVertexAttributes.length; i++) {\n result.push(binder.paintVertexAttributes[i].name);\n }\n } else if (binder instanceof CrossFadedCompositeBinder) {\n for (let i = 0; i < patternAttributes.members.length; i++) {\n result.push(patternAttributes.members[i].name);\n }\n }\n }\n return result;\n }\n\n getBinderUniforms(): Array {\n const uniforms = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder || binder instanceof CompositeExpressionBinder) {\n for (const uniformName of binder.uniformNames) {\n uniforms.push(uniformName);\n }\n }\n }\n return uniforms;\n }\n\n getPaintVertexBuffers(): Array {\n return this._buffers;\n }\n\n getUniforms(context: Context, locations: UniformLocations): Array {\n const uniforms = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder || binder instanceof CompositeExpressionBinder) {\n for (const name of binder.uniformNames) {\n if (locations[name]) {\n const binding = binder.getBinding(context, locations[name], name);\n uniforms.push({name, property, binding});\n }\n }\n }\n }\n return uniforms;\n }\n\n setUniforms(\n context: Context,\n binderUniforms: Array,\n properties: any,\n globals: GlobalProperties\n ) {\n // Uniform state bindings are owned by the Program, but we set them\n // from within the ProgramConfiguraton's binder members.\n for (const {name, property, binding} of binderUniforms) {\n (this.binders[property] as any).setUniform(binding, globals, properties.get(property), name);\n }\n }\n\n updatePaintBuffers(crossfade?: CrossfadeParameters) {\n this._buffers = [];\n\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (crossfade && binder instanceof CrossFadedCompositeBinder) {\n const patternVertexBuffer = crossfade.fromScale === 2 ? binder.zoomInPaintVertexBuffer : binder.zoomOutPaintVertexBuffer;\n if (patternVertexBuffer) this._buffers.push(patternVertexBuffer);\n\n } else if ((binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder) && binder.paintVertexBuffer) {\n this._buffers.push(binder.paintVertexBuffer);\n }\n }\n }\n\n upload(context: Context) {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder)\n binder.upload(context);\n }\n this.updatePaintBuffers();\n }\n\n destroy() {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder)\n binder.destroy();\n }\n }\n}\n\nexport class ProgramConfigurationSet {\n programConfigurations: {[_: string]: ProgramConfiguration};\n needsUpload: boolean;\n _featureMap: FeaturePositionMap;\n _bufferOffset: number;\n\n constructor(layers: ReadonlyArray, zoom: number, filterProperties: (_: string) => boolean = () => true) {\n this.programConfigurations = {};\n for (const layer of layers) {\n this.programConfigurations[layer.id] = new ProgramConfiguration(layer, zoom, filterProperties);\n }\n this.needsUpload = false;\n this._featureMap = new FeaturePositionMap();\n this._bufferOffset = 0;\n }\n\n populatePaintArrays(length: number, feature: Feature, index: number, imagePositions: {[_: string]: ImagePosition}, canonical: CanonicalTileID, formattedSection?: FormattedSection) {\n for (const key in this.programConfigurations) {\n this.programConfigurations[key].populatePaintArrays(length, feature, imagePositions, canonical, formattedSection);\n }\n\n if (feature.id !== undefined) {\n this._featureMap.add(feature.id, index, this._bufferOffset, length);\n }\n this._bufferOffset = length;\n\n this.needsUpload = true;\n }\n\n updatePaintArrays(featureStates: FeatureStates, vtLayer: VectorTileLayer, layers: ReadonlyArray, imagePositions: {[_: string]: ImagePosition}) {\n for (const layer of layers) {\n this.needsUpload = this.programConfigurations[layer.id].updatePaintArrays(featureStates, this._featureMap, vtLayer, layer, imagePositions) || this.needsUpload;\n }\n }\n\n get(layerId: string) {\n return this.programConfigurations[layerId];\n }\n\n upload(context: Context) {\n if (!this.needsUpload) return;\n for (const layerId in this.programConfigurations) {\n this.programConfigurations[layerId].upload(context);\n }\n this.needsUpload = false;\n }\n\n destroy() {\n for (const layerId in this.programConfigurations) {\n this.programConfigurations[layerId].destroy();\n }\n }\n}\n\nfunction paintAttributeNames(property, type) {\n const attributeNameExceptions = {\n 'text-opacity': ['opacity'],\n 'icon-opacity': ['opacity'],\n 'text-color': ['fill_color'],\n 'icon-color': ['fill_color'],\n 'text-halo-color': ['halo_color'],\n 'icon-halo-color': ['halo_color'],\n 'text-halo-blur': ['halo_blur'],\n 'icon-halo-blur': ['halo_blur'],\n 'text-halo-width': ['halo_width'],\n 'icon-halo-width': ['halo_width'],\n 'line-gap-width': ['gapwidth'],\n 'line-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'],\n 'fill-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'],\n 'fill-extrusion-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'],\n };\n\n return attributeNameExceptions[property] || [property.replace(`${type}-`, '').replace(/-/g, '_')];\n}\n\nfunction getLayoutException(property) {\n const propertyExceptions = {\n 'line-pattern': {\n 'source': PatternLayoutArray,\n 'composite': PatternLayoutArray\n },\n 'fill-pattern': {\n 'source': PatternLayoutArray,\n 'composite': PatternLayoutArray\n },\n 'fill-extrusion-pattern': {\n 'source': PatternLayoutArray,\n 'composite': PatternLayoutArray\n }\n };\n\n return propertyExceptions[property];\n}\n\nfunction layoutType(property, type, binderType) {\n const defaultLayouts = {\n 'color': {\n 'source': StructArrayLayout2f8,\n 'composite': StructArrayLayout4f16\n },\n 'number': {\n 'source': StructArrayLayout1f4,\n 'composite': StructArrayLayout2f8\n }\n };\n\n const layoutException = getLayoutException(property);\n return layoutException && layoutException[binderType] || defaultLayouts[type][binderType];\n}\n\nregister('ConstantBinder', ConstantBinder);\nregister('CrossFadedConstantBinder', CrossFadedConstantBinder);\nregister('SourceExpressionBinder', SourceExpressionBinder);\nregister('CrossFadedCompositeBinder', CrossFadedCompositeBinder);\nregister('CompositeExpressionBinder', CompositeExpressionBinder);\nregister('ProgramConfiguration', ProgramConfiguration, {omit: ['_buffers']});\nregister('ProgramConfigurationSet', ProgramConfigurationSet);\n","/**\n * The maximum value of a coordinate in the internal tile coordinate system. Coordinates of\n * all source features normalized to this extent upon load.\n *\n * The value is a consequence of the following:\n *\n * * Vertex buffer store positions as signed 16 bit integers.\n * * One bit is lost for signedness to support tile buffers.\n * * One bit is lost because the line vertex buffer used to pack 1 bit of other data into the int.\n * * One bit is lost to support features extending past the extent on the right edge of the tile.\n * * This leaves us with 2^13 = 8192\n */\nexport const EXTENT = 8192;\n","import {warnOnce, clamp} from '../util/util';\n\nimport {EXTENT} from './extent';\n\nimport type Point from '@mapbox/point-geometry';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\n// These bounds define the minimum and maximum supported coordinate values.\n// While visible coordinates are within [0, EXTENT], tiles may theoretically\n// contain coordinates within [-Infinity, Infinity]. Our range is limited by the\n// number of bits used to represent the coordinate.\nconst BITS = 15;\nconst MAX = Math.pow(2, BITS - 1) - 1;\nconst MIN = -MAX - 1;\n\n/**\n * Loads a geometry from a VectorTileFeature and scales it to the common extent\n * used internally.\n * @param feature - the vector tile feature to load\n */\nexport function loadGeometry(feature: VectorTileFeature): Array> {\n const scale = EXTENT / feature.extent;\n const geometry = feature.loadGeometry();\n for (let r = 0; r < geometry.length; r++) {\n const ring = geometry[r];\n for (let p = 0; p < ring.length; p++) {\n const point = ring[p];\n // round here because mapbox-gl-native uses integers to represent\n // points and we need to do the same to avoid renering differences.\n const x = Math.round(point.x * scale);\n const y = Math.round(point.y * scale);\n\n point.x = clamp(x, MIN, MAX);\n point.y = clamp(y, MIN, MAX);\n\n if (x < point.x || x > point.x + 1 || y < point.y || y > point.y + 1) {\n // warn when exceeding allowed extent except for the 1-px-off case\n // https://github.com/mapbox/mapbox-gl-js/issues/8992\n warnOnce('Geometry exceeds allowed extent, reduce your vector tile buffer size');\n }\n }\n }\n return geometry;\n}\n","import {loadGeometry} from './load_geometry';\nimport type Point from '@mapbox/point-geometry';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\nimport type {Feature} from '@maplibre/maplibre-gl-style-spec';\n\ntype EvaluationFeature = Feature & { geometry: Array> };\n/**\n * Construct a new feature based on a VectorTileFeature for expression evaluation, the geometry of which\n * will be loaded based on necessity.\n * @param feature - the feature to evaluate\n * @param needGeometry - if set to true this will load the geometry\n */\nexport function toEvaluationFeature(feature: VectorTileFeature, needGeometry: boolean): EvaluationFeature {\n return {type: feature.type,\n id: feature.id,\n properties: feature.properties,\n geometry: needGeometry ? loadGeometry(feature) : []};\n}\n","import {CircleLayoutArray} from '../array_types.g';\n\nimport {members as layoutAttributes} from './circle_attributes';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray} from '../index_array_type';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EXTENT} from '../extent';\nimport {register} from '../../util/web_worker_transfer';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {CircleStyleLayer} from '../../style/style_layer/circle_style_layer';\nimport type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nfunction addCircleVertex(layoutVertexArray, x, y, extrudeX, extrudeY) {\n layoutVertexArray.emplaceBack(\n (x * 2) + ((extrudeX + 1) / 2),\n (y * 2) + ((extrudeY + 1) / 2));\n}\n\n/**\n * @internal\n * Circles are represented by two triangles.\n *\n * Each corner has a pos that is the center of the circle and an extrusion\n * vector that is where it points.\n */\nexport class CircleBucket implements Bucket {\n index: number;\n zoom: number;\n overscaling: number;\n layerIds: Array;\n layers: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n\n layoutVertexArray: CircleLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n uploaded: boolean;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n\n this.layoutVertexArray = new CircleLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.segments = new SegmentVector();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n const styleLayer = this.layers[0];\n const bucketFeatures: BucketFeature[] = [];\n let circleSortKey = null;\n let sortFeaturesByKey = false;\n\n // Heatmap layers are handled in this bucket and have no evaluated properties, so we check our access\n if (styleLayer.type === 'circle') {\n circleSortKey = (styleLayer as CircleStyleLayer).layout.get('circle-sort-key');\n sortFeaturesByKey = !circleSortKey.isConstant();\n }\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const sortKey = sortFeaturesByKey ?\n circleSortKey.evaluate(evaluationFeature, {}, canonical) :\n undefined;\n\n const bucketFeature: BucketFeature = {\n id,\n properties: feature.properties,\n type: feature.type,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n patterns: {},\n sortKey\n };\n\n bucketFeatures.push(bucketFeature);\n\n }\n\n if (sortFeaturesByKey) {\n bucketFeatures.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const bucketFeature of bucketFeatures) {\n const {geometry, index, sourceLayerIndex} = bucketFeature;\n const feature = features[index].feature;\n\n this.addFeature(bucketFeature, geometry, index, canonical);\n options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0;\n }\n\n uploadPending() {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID) {\n for (const ring of geometry) {\n for (const point of ring) {\n const x = point.x;\n const y = point.y;\n\n // Do not include points that are outside the tile boundaries.\n if (x < 0 || x >= EXTENT || y < 0 || y >= EXTENT) continue;\n\n // this geometry will be of the Point type, and we'll derive\n // two triangles from it.\n //\n // ┌─────────┐\n // │ 3 2 │\n // │ │\n // │ 0 1 │\n // └─────────┘\n\n const segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray, feature.sortKey);\n const index = segment.vertexLength;\n\n addCircleVertex(this.layoutVertexArray, x, y, -1, -1);\n addCircleVertex(this.layoutVertexArray, x, y, 1, -1);\n addCircleVertex(this.layoutVertexArray, x, y, 1, 1);\n addCircleVertex(this.layoutVertexArray, x, y, -1, 1);\n\n this.indexArray.emplaceBack(index, index + 1, index + 2);\n this.indexArray.emplaceBack(index, index + 3, index + 2);\n\n segment.vertexLength += 4;\n segment.primitiveLength += 2;\n }\n }\n\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, {}, canonical);\n }\n}\n\nregister('CircleBucket', CircleBucket, {omit: ['layers']});\n","import {isCounterClockwise} from './util';\n\nimport Point from '@mapbox/point-geometry';\n\nexport {polygonIntersectsBufferedPoint, polygonIntersectsMultiPolygon, polygonIntersectsBufferedMultiLine, polygonIntersectsPolygon, distToSegmentSquared, polygonIntersectsBox};\n\ntype Line = Array;\ntype MultiLine = Array;\ntype Ring = Array;\ntype Polygon = Array;\ntype MultiPolygon = Array;\n\nfunction polygonIntersectsPolygon(polygonA: Polygon, polygonB: Polygon) {\n for (let i = 0; i < polygonA.length; i++) {\n if (polygonContainsPoint(polygonB, polygonA[i])) return true;\n }\n\n for (let i = 0; i < polygonB.length; i++) {\n if (polygonContainsPoint(polygonA, polygonB[i])) return true;\n }\n\n if (lineIntersectsLine(polygonA, polygonB)) return true;\n\n return false;\n}\n\nfunction polygonIntersectsBufferedPoint(polygon: Polygon, point: Point, radius: number) {\n if (polygonContainsPoint(polygon, point)) return true;\n if (pointIntersectsBufferedLine(point, polygon, radius)) return true;\n return false;\n}\n\nfunction polygonIntersectsMultiPolygon(polygon: Polygon, multiPolygon: MultiPolygon) {\n\n if (polygon.length === 1) {\n return multiPolygonContainsPoint(multiPolygon, polygon[0]);\n }\n\n for (let m = 0; m < multiPolygon.length; m++) {\n const ring = multiPolygon[m];\n for (let n = 0; n < ring.length; n++) {\n if (polygonContainsPoint(polygon, ring[n])) return true;\n }\n }\n\n for (let i = 0; i < polygon.length; i++) {\n if (multiPolygonContainsPoint(multiPolygon, polygon[i])) return true;\n }\n\n for (let k = 0; k < multiPolygon.length; k++) {\n if (lineIntersectsLine(polygon, multiPolygon[k])) return true;\n }\n\n return false;\n}\n\nfunction polygonIntersectsBufferedMultiLine(polygon: Polygon, multiLine: MultiLine, radius: number) {\n for (let i = 0; i < multiLine.length; i++) {\n const line = multiLine[i];\n\n if (polygon.length >= 3) {\n for (let k = 0; k < line.length; k++) {\n if (polygonContainsPoint(polygon, line[k])) return true;\n }\n }\n\n if (lineIntersectsBufferedLine(polygon, line, radius)) return true;\n }\n return false;\n}\n\nfunction lineIntersectsBufferedLine(lineA: Line, lineB: Line, radius: number) {\n\n if (lineA.length > 1) {\n if (lineIntersectsLine(lineA, lineB)) return true;\n\n // Check whether any point in either line is within radius of the other line\n for (let j = 0; j < lineB.length; j++) {\n if (pointIntersectsBufferedLine(lineB[j], lineA, radius)) return true;\n }\n }\n\n for (let k = 0; k < lineA.length; k++) {\n if (pointIntersectsBufferedLine(lineA[k], lineB, radius)) return true;\n }\n\n return false;\n}\n\nfunction lineIntersectsLine(lineA: Line, lineB: Line) {\n if (lineA.length === 0 || lineB.length === 0) return false;\n for (let i = 0; i < lineA.length - 1; i++) {\n const a0 = lineA[i];\n const a1 = lineA[i + 1];\n for (let j = 0; j < lineB.length - 1; j++) {\n const b0 = lineB[j];\n const b1 = lineB[j + 1];\n if (lineSegmentIntersectsLineSegment(a0, a1, b0, b1)) return true;\n }\n }\n return false;\n}\n\nfunction lineSegmentIntersectsLineSegment(a0: Point, a1: Point, b0: Point, b1: Point) {\n return isCounterClockwise(a0, b0, b1) !== isCounterClockwise(a1, b0, b1) &&\n isCounterClockwise(a0, a1, b0) !== isCounterClockwise(a0, a1, b1);\n}\n\nfunction pointIntersectsBufferedLine(p: Point, line: Line, radius: number) {\n const radiusSquared = radius * radius;\n\n if (line.length === 1) return p.distSqr(line[0]) < radiusSquared;\n\n for (let i = 1; i < line.length; i++) {\n // Find line segments that have a distance <= radius^2 to p\n // In that case, we treat the line as \"containing point p\".\n const v = line[i - 1], w = line[i];\n if (distToSegmentSquared(p, v, w) < radiusSquared) return true;\n }\n return false;\n}\n\n// Code from http://stackoverflow.com/a/1501725/331379.\nfunction distToSegmentSquared(p: Point, v: Point, w: Point) {\n const l2 = v.distSqr(w);\n if (l2 === 0) return p.distSqr(v);\n const t = ((p.x - v.x) * (w.x - v.x) + (p.y - v.y) * (w.y - v.y)) / l2;\n if (t < 0) return p.distSqr(v);\n if (t > 1) return p.distSqr(w);\n return p.distSqr(w.sub(v)._mult(t)._add(v));\n}\n\n// point in polygon ray casting algorithm\nfunction multiPolygonContainsPoint(rings: Array, p: Point) {\n let c = false,\n ring, p1, p2;\n\n for (let k = 0; k < rings.length; k++) {\n ring = rings[k];\n for (let i = 0, j = ring.length - 1; i < ring.length; j = i++) {\n p1 = ring[i];\n p2 = ring[j];\n if (((p1.y > p.y) !== (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) {\n c = !c;\n }\n }\n }\n return c;\n}\n\nfunction polygonContainsPoint(ring: Ring, p: Point) {\n let c = false;\n for (let i = 0, j = ring.length - 1; i < ring.length; j = i++) {\n const p1 = ring[i];\n const p2 = ring[j];\n if (((p1.y > p.y) !== (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) {\n c = !c;\n }\n }\n return c;\n}\n\nfunction polygonIntersectsBox(ring: Ring, boxX1: number, boxY1: number, boxX2: number, boxY2: number) {\n for (const p of ring) {\n if (boxX1 <= p.x &&\n boxY1 <= p.y &&\n boxX2 >= p.x &&\n boxY2 >= p.y) return true;\n }\n\n const corners = [\n new Point(boxX1, boxY1),\n new Point(boxX1, boxY2),\n new Point(boxX2, boxY2),\n new Point(boxX2, boxY1)];\n\n if (ring.length > 2) {\n for (const corner of corners) {\n if (polygonContainsPoint(ring, corner)) return true;\n }\n }\n\n for (let i = 0; i < ring.length - 1; i++) {\n const p1 = ring[i];\n const p2 = ring[i + 1];\n if (edgeIntersectsBox(p1, p2, corners)) return true;\n }\n\n return false;\n}\n\nfunction edgeIntersectsBox(e1: Point, e2: Point, corners: Array) {\n const tl = corners[0];\n const br = corners[2];\n // the edge and box do not intersect in either the x or y dimensions\n if (((e1.x < tl.x) && (e2.x < tl.x)) ||\n ((e1.x > br.x) && (e2.x > br.x)) ||\n ((e1.y < tl.y) && (e2.y < tl.y)) ||\n ((e1.y > br.y) && (e2.y > br.y))) return false;\n\n // check if all corners of the box are on the same side of the edge\n const dir = isCounterClockwise(e1, e2, corners[0]);\n return dir !== isCounterClockwise(e1, e2, corners[1]) ||\n dir !== isCounterClockwise(e1, e2, corners[2]) ||\n dir !== isCounterClockwise(e1, e2, corners[3]);\n}\n","import Point from '@mapbox/point-geometry';\n\nimport type {PossiblyEvaluatedPropertyValue} from './properties';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {CircleBucket} from '../data/bucket/circle_bucket';\nimport type {LineBucket} from '../data/bucket/line_bucket';\n\nexport function getMaximumPaintValue(\n property: string,\n layer: StyleLayer,\n bucket: CircleBucket | LineBucket\n): number {\n const value = ((layer.paint as any).get(property) as PossiblyEvaluatedPropertyValue).value;\n if (value.kind === 'constant') {\n return value.value;\n } else {\n return bucket.programConfigurations.get(layer.id).getMaxValue(property);\n }\n}\n\nexport function translateDistance(translate: [number, number]) {\n return Math.sqrt(translate[0] * translate[0] + translate[1] * translate[1]);\n}\n\nexport function translate(queryGeometry: Array,\n translate: [number, number],\n translateAnchor: 'viewport' | 'map',\n bearing: number,\n pixelsToTileUnits: number) {\n if (!translate[0] && !translate[1]) {\n return queryGeometry;\n }\n const pt = Point.convert(translate)._mult(pixelsToTileUnits);\n\n if (translateAnchor === 'viewport') {\n pt._rotate(-bearing);\n }\n\n const translated = [];\n for (let i = 0; i < queryGeometry.length; i++) {\n const point = queryGeometry[i];\n translated.push(point.sub(pt));\n }\n return translated;\n}\n\nexport function offsetLine(rings: Array>, offset: number) {\n const newRings: Array> = [];\n for (let ringIndex = 0; ringIndex < rings.length; ringIndex++) {\n const ring = rings[ringIndex];\n const newRing: Array = [];\n for (let index = 0; index < ring.length; index++) {\n const a = ring[index - 1];\n const b = ring[index];\n const c = ring[index + 1];\n const aToB = index === 0 ? new Point(0, 0) : b.sub(a)._unit()._perp();\n const bToC = index === ring.length - 1 ? new Point(0, 0) : c.sub(b)._unit()._perp();\n const extrude = aToB._add(bToC)._unit();\n\n const cosHalfAngle = extrude.x * bToC.x + extrude.y * bToC.y;\n if (cosHalfAngle !== 0) {\n extrude._mult(1 / cosHalfAngle);\n }\n\n newRing.push(extrude._mult(offset)._add(b));\n }\n newRings.push(newRing);\n }\n return newRings;\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type CircleLayoutProps = {\n \"circle-sort-key\": DataDrivenProperty,\n};\n\nexport type CircleLayoutPropsPossiblyEvaluated = {\n \"circle-sort-key\": PossiblyEvaluatedPropertyValue,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"circle-sort-key\": new DataDrivenProperty(styleSpec[\"layout_circle\"][\"circle-sort-key\"] as any as StylePropertySpecification),\n});\n\nexport type CirclePaintProps = {\n \"circle-radius\": DataDrivenProperty,\n \"circle-color\": DataDrivenProperty,\n \"circle-blur\": DataDrivenProperty,\n \"circle-opacity\": DataDrivenProperty,\n \"circle-translate\": DataConstantProperty<[number, number]>,\n \"circle-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"circle-pitch-scale\": DataConstantProperty<\"map\" | \"viewport\">,\n \"circle-pitch-alignment\": DataConstantProperty<\"map\" | \"viewport\">,\n \"circle-stroke-width\": DataDrivenProperty,\n \"circle-stroke-color\": DataDrivenProperty,\n \"circle-stroke-opacity\": DataDrivenProperty,\n};\n\nexport type CirclePaintPropsPossiblyEvaluated = {\n \"circle-radius\": PossiblyEvaluatedPropertyValue,\n \"circle-color\": PossiblyEvaluatedPropertyValue,\n \"circle-blur\": PossiblyEvaluatedPropertyValue,\n \"circle-opacity\": PossiblyEvaluatedPropertyValue,\n \"circle-translate\": [number, number],\n \"circle-translate-anchor\": \"map\" | \"viewport\",\n \"circle-pitch-scale\": \"map\" | \"viewport\",\n \"circle-pitch-alignment\": \"map\" | \"viewport\",\n \"circle-stroke-width\": PossiblyEvaluatedPropertyValue,\n \"circle-stroke-color\": PossiblyEvaluatedPropertyValue,\n \"circle-stroke-opacity\": PossiblyEvaluatedPropertyValue,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"circle-radius\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-radius\"] as any as StylePropertySpecification),\n \"circle-color\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-color\"] as any as StylePropertySpecification),\n \"circle-blur\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-blur\"] as any as StylePropertySpecification),\n \"circle-opacity\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-opacity\"] as any as StylePropertySpecification),\n \"circle-translate\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-translate\"] as any as StylePropertySpecification),\n \"circle-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-translate-anchor\"] as any as StylePropertySpecification),\n \"circle-pitch-scale\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-pitch-scale\"] as any as StylePropertySpecification),\n \"circle-pitch-alignment\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-pitch-alignment\"] as any as StylePropertySpecification),\n \"circle-stroke-width\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-stroke-width\"] as any as StylePropertySpecification),\n \"circle-stroke-color\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-stroke-color\"] as any as StylePropertySpecification),\n \"circle-stroke-opacity\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-stroke-opacity\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import * as glMatrix from \"./common.js\";\n/**\n * 4 Dimensional Vector\n * @module vec4\n */\n\n/**\n * Creates a new, empty vec4\n *\n * @returns {vec4} a new 4D vector\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(4);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[0] = 0;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n }\n\n return out;\n}\n/**\n * Creates a new vec4 initialized with values from an existing vector\n *\n * @param {ReadonlyVec4} a vector to clone\n * @returns {vec4} a new 4D vector\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Creates a new vec4 initialized with the given values\n *\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @param {Number} w W component\n * @returns {vec4} a new 4D vector\n */\n\nexport function fromValues(x, y, z, w) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = x;\n out[1] = y;\n out[2] = z;\n out[3] = w;\n return out;\n}\n/**\n * Copy the values from one vec4 to another\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the source vector\n * @returns {vec4} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Set the components of a vec4 to the given values\n *\n * @param {vec4} out the receiving vector\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @param {Number} w W component\n * @returns {vec4} out\n */\n\nexport function set(out, x, y, z, w) {\n out[0] = x;\n out[1] = y;\n out[2] = z;\n out[3] = w;\n return out;\n}\n/**\n * Adds two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n return out;\n}\n/**\n * Subtracts vector b from vector a\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n return out;\n}\n/**\n * Multiplies two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function multiply(out, a, b) {\n out[0] = a[0] * b[0];\n out[1] = a[1] * b[1];\n out[2] = a[2] * b[2];\n out[3] = a[3] * b[3];\n return out;\n}\n/**\n * Divides two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function divide(out, a, b) {\n out[0] = a[0] / b[0];\n out[1] = a[1] / b[1];\n out[2] = a[2] / b[2];\n out[3] = a[3] / b[3];\n return out;\n}\n/**\n * Math.ceil the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to ceil\n * @returns {vec4} out\n */\n\nexport function ceil(out, a) {\n out[0] = Math.ceil(a[0]);\n out[1] = Math.ceil(a[1]);\n out[2] = Math.ceil(a[2]);\n out[3] = Math.ceil(a[3]);\n return out;\n}\n/**\n * Math.floor the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to floor\n * @returns {vec4} out\n */\n\nexport function floor(out, a) {\n out[0] = Math.floor(a[0]);\n out[1] = Math.floor(a[1]);\n out[2] = Math.floor(a[2]);\n out[3] = Math.floor(a[3]);\n return out;\n}\n/**\n * Returns the minimum of two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function min(out, a, b) {\n out[0] = Math.min(a[0], b[0]);\n out[1] = Math.min(a[1], b[1]);\n out[2] = Math.min(a[2], b[2]);\n out[3] = Math.min(a[3], b[3]);\n return out;\n}\n/**\n * Returns the maximum of two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function max(out, a, b) {\n out[0] = Math.max(a[0], b[0]);\n out[1] = Math.max(a[1], b[1]);\n out[2] = Math.max(a[2], b[2]);\n out[3] = Math.max(a[3], b[3]);\n return out;\n}\n/**\n * Math.round the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to round\n * @returns {vec4} out\n */\n\nexport function round(out, a) {\n out[0] = Math.round(a[0]);\n out[1] = Math.round(a[1]);\n out[2] = Math.round(a[2]);\n out[3] = Math.round(a[3]);\n return out;\n}\n/**\n * Scales a vec4 by a scalar number\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the vector to scale\n * @param {Number} b amount to scale the vector by\n * @returns {vec4} out\n */\n\nexport function scale(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n return out;\n}\n/**\n * Adds two vec4's after scaling the second operand by a scalar value\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @param {Number} scale the amount to scale b by before adding\n * @returns {vec4} out\n */\n\nexport function scaleAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n return out;\n}\n/**\n * Calculates the euclidian distance between two vec4's\n *\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {Number} distance between a and b\n */\n\nexport function distance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n var w = b[3] - a[3];\n return Math.hypot(x, y, z, w);\n}\n/**\n * Calculates the squared euclidian distance between two vec4's\n *\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {Number} squared distance between a and b\n */\n\nexport function squaredDistance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n var w = b[3] - a[3];\n return x * x + y * y + z * z + w * w;\n}\n/**\n * Calculates the length of a vec4\n *\n * @param {ReadonlyVec4} a vector to calculate length of\n * @returns {Number} length of a\n */\n\nexport function length(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var w = a[3];\n return Math.hypot(x, y, z, w);\n}\n/**\n * Calculates the squared length of a vec4\n *\n * @param {ReadonlyVec4} a vector to calculate squared length of\n * @returns {Number} squared length of a\n */\n\nexport function squaredLength(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var w = a[3];\n return x * x + y * y + z * z + w * w;\n}\n/**\n * Negates the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to negate\n * @returns {vec4} out\n */\n\nexport function negate(out, a) {\n out[0] = -a[0];\n out[1] = -a[1];\n out[2] = -a[2];\n out[3] = -a[3];\n return out;\n}\n/**\n * Returns the inverse of the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to invert\n * @returns {vec4} out\n */\n\nexport function inverse(out, a) {\n out[0] = 1.0 / a[0];\n out[1] = 1.0 / a[1];\n out[2] = 1.0 / a[2];\n out[3] = 1.0 / a[3];\n return out;\n}\n/**\n * Normalize a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to normalize\n * @returns {vec4} out\n */\n\nexport function normalize(out, a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var w = a[3];\n var len = x * x + y * y + z * z + w * w;\n\n if (len > 0) {\n len = 1 / Math.sqrt(len);\n }\n\n out[0] = x * len;\n out[1] = y * len;\n out[2] = z * len;\n out[3] = w * len;\n return out;\n}\n/**\n * Calculates the dot product of two vec4's\n *\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {Number} dot product of a and b\n */\n\nexport function dot(a, b) {\n return a[0] * b[0] + a[1] * b[1] + a[2] * b[2] + a[3] * b[3];\n}\n/**\n * Returns the cross-product of three vectors in a 4-dimensional space\n *\n * @param {ReadonlyVec4} result the receiving vector\n * @param {ReadonlyVec4} U the first vector\n * @param {ReadonlyVec4} V the second vector\n * @param {ReadonlyVec4} W the third vector\n * @returns {vec4} result\n */\n\nexport function cross(out, u, v, w) {\n var A = v[0] * w[1] - v[1] * w[0],\n B = v[0] * w[2] - v[2] * w[0],\n C = v[0] * w[3] - v[3] * w[0],\n D = v[1] * w[2] - v[2] * w[1],\n E = v[1] * w[3] - v[3] * w[1],\n F = v[2] * w[3] - v[3] * w[2];\n var G = u[0];\n var H = u[1];\n var I = u[2];\n var J = u[3];\n out[0] = H * F - I * E + J * D;\n out[1] = -(G * F) + I * C - J * B;\n out[2] = G * E - H * C + J * A;\n out[3] = -(G * D) + H * B - I * A;\n return out;\n}\n/**\n * Performs a linear interpolation between two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec4} out\n */\n\nexport function lerp(out, a, b, t) {\n var ax = a[0];\n var ay = a[1];\n var az = a[2];\n var aw = a[3];\n out[0] = ax + t * (b[0] - ax);\n out[1] = ay + t * (b[1] - ay);\n out[2] = az + t * (b[2] - az);\n out[3] = aw + t * (b[3] - aw);\n return out;\n}\n/**\n * Generates a random vector with the given scale\n *\n * @param {vec4} out the receiving vector\n * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned\n * @returns {vec4} out\n */\n\nexport function random(out, scale) {\n scale = scale || 1.0; // Marsaglia, George. Choosing a Point from the Surface of a\n // Sphere. Ann. Math. Statist. 43 (1972), no. 2, 645--646.\n // http://projecteuclid.org/euclid.aoms/1177692644;\n\n var v1, v2, v3, v4;\n var s1, s2;\n\n do {\n v1 = glMatrix.RANDOM() * 2 - 1;\n v2 = glMatrix.RANDOM() * 2 - 1;\n s1 = v1 * v1 + v2 * v2;\n } while (s1 >= 1);\n\n do {\n v3 = glMatrix.RANDOM() * 2 - 1;\n v4 = glMatrix.RANDOM() * 2 - 1;\n s2 = v3 * v3 + v4 * v4;\n } while (s2 >= 1);\n\n var d = Math.sqrt((1 - s1) / s2);\n out[0] = scale * v1;\n out[1] = scale * v2;\n out[2] = scale * v3 * d;\n out[3] = scale * v4 * d;\n return out;\n}\n/**\n * Transforms the vec4 with a mat4.\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the vector to transform\n * @param {ReadonlyMat4} m matrix to transform with\n * @returns {vec4} out\n */\n\nexport function transformMat4(out, a, m) {\n var x = a[0],\n y = a[1],\n z = a[2],\n w = a[3];\n out[0] = m[0] * x + m[4] * y + m[8] * z + m[12] * w;\n out[1] = m[1] * x + m[5] * y + m[9] * z + m[13] * w;\n out[2] = m[2] * x + m[6] * y + m[10] * z + m[14] * w;\n out[3] = m[3] * x + m[7] * y + m[11] * z + m[15] * w;\n return out;\n}\n/**\n * Transforms the vec4 with a quat\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the vector to transform\n * @param {ReadonlyQuat} q quaternion to transform with\n * @returns {vec4} out\n */\n\nexport function transformQuat(out, a, q) {\n var x = a[0],\n y = a[1],\n z = a[2];\n var qx = q[0],\n qy = q[1],\n qz = q[2],\n qw = q[3]; // calculate quat * vec\n\n var ix = qw * x + qy * z - qz * y;\n var iy = qw * y + qz * x - qx * z;\n var iz = qw * z + qx * y - qy * x;\n var iw = -qx * x - qy * y - qz * z; // calculate result * inverse quat\n\n out[0] = ix * qw + iw * -qx + iy * -qz - iz * -qy;\n out[1] = iy * qw + iw * -qy + iz * -qx - ix * -qz;\n out[2] = iz * qw + iw * -qz + ix * -qy - iy * -qx;\n out[3] = a[3];\n return out;\n}\n/**\n * Set the components of a vec4 to zero\n *\n * @param {vec4} out the receiving vector\n * @returns {vec4} out\n */\n\nexport function zero(out) {\n out[0] = 0.0;\n out[1] = 0.0;\n out[2] = 0.0;\n out[3] = 0.0;\n return out;\n}\n/**\n * Returns a string representation of a vector\n *\n * @param {ReadonlyVec4} a vector to represent as a string\n * @returns {String} string representation of the vector\n */\n\nexport function str(a) {\n return \"vec4(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \")\";\n}\n/**\n * Returns whether or not the vectors have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyVec4} a The first vector.\n * @param {ReadonlyVec4} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3];\n}\n/**\n * Returns whether or not the vectors have approximately the same elements in the same position.\n *\n * @param {ReadonlyVec4} a The first vector.\n * @param {ReadonlyVec4} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3));\n}\n/**\n * Alias for {@link vec4.subtract}\n * @function\n */\n\nexport var sub = subtract;\n/**\n * Alias for {@link vec4.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link vec4.divide}\n * @function\n */\n\nexport var div = divide;\n/**\n * Alias for {@link vec4.distance}\n * @function\n */\n\nexport var dist = distance;\n/**\n * Alias for {@link vec4.squaredDistance}\n * @function\n */\n\nexport var sqrDist = squaredDistance;\n/**\n * Alias for {@link vec4.length}\n * @function\n */\n\nexport var len = length;\n/**\n * Alias for {@link vec4.squaredLength}\n * @function\n */\n\nexport var sqrLen = squaredLength;\n/**\n * Perform some operation over an array of vec4s.\n *\n * @param {Array} a the array of vectors to iterate over\n * @param {Number} stride Number of elements between the start of each vec4. If 0 assumes tightly packed\n * @param {Number} offset Number of elements to skip at the beginning of the array\n * @param {Number} count Number of vec4s to iterate over. If 0 iterates over entire array\n * @param {Function} fn Function to call for each vector in the array\n * @param {Object} [arg] additional argument to pass to fn\n * @returns {Array} a\n * @function\n */\n\nexport var forEach = function () {\n var vec = create();\n return function (a, stride, offset, count, fn, arg) {\n var i, l;\n\n if (!stride) {\n stride = 4;\n }\n\n if (!offset) {\n offset = 0;\n }\n\n if (count) {\n l = Math.min(count * stride + offset, a.length);\n } else {\n l = a.length;\n }\n\n for (i = offset; i < l; i += stride) {\n vec[0] = a[i];\n vec[1] = a[i + 1];\n vec[2] = a[i + 2];\n vec[3] = a[i + 3];\n fn(vec, vec, arg);\n a[i] = vec[0];\n a[i + 1] = vec[1];\n a[i + 2] = vec[2];\n a[i + 3] = vec[3];\n }\n\n return a;\n };\n}();","/**\n * Common utilities\n * @module glMatrix\n */\n// Configuration Constants\nexport var EPSILON = 0.000001;\nexport var ARRAY_TYPE = typeof Float32Array !== 'undefined' ? Float32Array : Array;\nexport var RANDOM = Math.random;\n/**\n * Sets the type of array used when creating new vectors and matrices\n *\n * @param {Float32ArrayConstructor | ArrayConstructor} type Array type, such as Float32Array or Array\n */\n\nexport function setMatrixArrayType(type) {\n ARRAY_TYPE = type;\n}\nvar degree = Math.PI / 180;\n/**\n * Convert Degree To Radian\n *\n * @param {Number} a Angle in Degrees\n */\n\nexport function toRadian(a) {\n return a * degree;\n}\n/**\n * Tests whether or not the arguments have approximately the same value, within an absolute\n * or relative tolerance of glMatrix.EPSILON (an absolute tolerance is used for values less\n * than or equal to 1.0, and a relative tolerance is used for larger values)\n *\n * @param {Number} a The first number to test.\n * @param {Number} b The second number to test.\n * @returns {Boolean} True if the numbers are approximately equal, false otherwise.\n */\n\nexport function equals(a, b) {\n return Math.abs(a - b) <= EPSILON * Math.max(1.0, Math.abs(a), Math.abs(b));\n}\nif (!Math.hypot) Math.hypot = function () {\n var y = 0,\n i = arguments.length;\n\n while (i--) {\n y += arguments[i] * arguments[i];\n }\n\n return Math.sqrt(y);\n};","import * as glMatrix from \"./common.js\";\n/**\n * 4x4 Matrix
Format: column-major, when typed out it looks like row-major
The matrices are being post multiplied.\n * @module mat4\n */\n\n/**\n * Creates a new identity mat4\n *\n * @returns {mat4} a new 4x4 matrix\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(16);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n }\n\n out[0] = 1;\n out[5] = 1;\n out[10] = 1;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a new mat4 initialized with values from an existing matrix\n *\n * @param {ReadonlyMat4} a matrix to clone\n * @returns {mat4} a new 4x4 matrix\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(16);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n out[9] = a[9];\n out[10] = a[10];\n out[11] = a[11];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n return out;\n}\n/**\n * Copy the values from one mat4 to another\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n out[9] = a[9];\n out[10] = a[10];\n out[11] = a[11];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n return out;\n}\n/**\n * Create a new mat4 with the given values\n *\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m03 Component in column 0, row 3 position (index 3)\n * @param {Number} m10 Component in column 1, row 0 position (index 4)\n * @param {Number} m11 Component in column 1, row 1 position (index 5)\n * @param {Number} m12 Component in column 1, row 2 position (index 6)\n * @param {Number} m13 Component in column 1, row 3 position (index 7)\n * @param {Number} m20 Component in column 2, row 0 position (index 8)\n * @param {Number} m21 Component in column 2, row 1 position (index 9)\n * @param {Number} m22 Component in column 2, row 2 position (index 10)\n * @param {Number} m23 Component in column 2, row 3 position (index 11)\n * @param {Number} m30 Component in column 3, row 0 position (index 12)\n * @param {Number} m31 Component in column 3, row 1 position (index 13)\n * @param {Number} m32 Component in column 3, row 2 position (index 14)\n * @param {Number} m33 Component in column 3, row 3 position (index 15)\n * @returns {mat4} A new mat4\n */\n\nexport function fromValues(m00, m01, m02, m03, m10, m11, m12, m13, m20, m21, m22, m23, m30, m31, m32, m33) {\n var out = new glMatrix.ARRAY_TYPE(16);\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m03;\n out[4] = m10;\n out[5] = m11;\n out[6] = m12;\n out[7] = m13;\n out[8] = m20;\n out[9] = m21;\n out[10] = m22;\n out[11] = m23;\n out[12] = m30;\n out[13] = m31;\n out[14] = m32;\n out[15] = m33;\n return out;\n}\n/**\n * Set the components of a mat4 to the given values\n *\n * @param {mat4} out the receiving matrix\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m03 Component in column 0, row 3 position (index 3)\n * @param {Number} m10 Component in column 1, row 0 position (index 4)\n * @param {Number} m11 Component in column 1, row 1 position (index 5)\n * @param {Number} m12 Component in column 1, row 2 position (index 6)\n * @param {Number} m13 Component in column 1, row 3 position (index 7)\n * @param {Number} m20 Component in column 2, row 0 position (index 8)\n * @param {Number} m21 Component in column 2, row 1 position (index 9)\n * @param {Number} m22 Component in column 2, row 2 position (index 10)\n * @param {Number} m23 Component in column 2, row 3 position (index 11)\n * @param {Number} m30 Component in column 3, row 0 position (index 12)\n * @param {Number} m31 Component in column 3, row 1 position (index 13)\n * @param {Number} m32 Component in column 3, row 2 position (index 14)\n * @param {Number} m33 Component in column 3, row 3 position (index 15)\n * @returns {mat4} out\n */\n\nexport function set(out, m00, m01, m02, m03, m10, m11, m12, m13, m20, m21, m22, m23, m30, m31, m32, m33) {\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m03;\n out[4] = m10;\n out[5] = m11;\n out[6] = m12;\n out[7] = m13;\n out[8] = m20;\n out[9] = m21;\n out[10] = m22;\n out[11] = m23;\n out[12] = m30;\n out[13] = m31;\n out[14] = m32;\n out[15] = m33;\n return out;\n}\n/**\n * Set a mat4 to the identity matrix\n *\n * @param {mat4} out the receiving matrix\n * @returns {mat4} out\n */\n\nexport function identity(out) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = 1;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 1;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Transpose the values of a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function transpose(out, a) {\n // If we are transposing ourselves we can skip a few steps but have to cache some values\n if (out === a) {\n var a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a12 = a[6],\n a13 = a[7];\n var a23 = a[11];\n out[1] = a[4];\n out[2] = a[8];\n out[3] = a[12];\n out[4] = a01;\n out[6] = a[9];\n out[7] = a[13];\n out[8] = a02;\n out[9] = a12;\n out[11] = a[14];\n out[12] = a03;\n out[13] = a13;\n out[14] = a23;\n } else {\n out[0] = a[0];\n out[1] = a[4];\n out[2] = a[8];\n out[3] = a[12];\n out[4] = a[1];\n out[5] = a[5];\n out[6] = a[9];\n out[7] = a[13];\n out[8] = a[2];\n out[9] = a[6];\n out[10] = a[10];\n out[11] = a[14];\n out[12] = a[3];\n out[13] = a[7];\n out[14] = a[11];\n out[15] = a[15];\n }\n\n return out;\n}\n/**\n * Inverts a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function invert(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n var b00 = a00 * a11 - a01 * a10;\n var b01 = a00 * a12 - a02 * a10;\n var b02 = a00 * a13 - a03 * a10;\n var b03 = a01 * a12 - a02 * a11;\n var b04 = a01 * a13 - a03 * a11;\n var b05 = a02 * a13 - a03 * a12;\n var b06 = a20 * a31 - a21 * a30;\n var b07 = a20 * a32 - a22 * a30;\n var b08 = a20 * a33 - a23 * a30;\n var b09 = a21 * a32 - a22 * a31;\n var b10 = a21 * a33 - a23 * a31;\n var b11 = a22 * a33 - a23 * a32; // Calculate the determinant\n\n var det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det;\n out[1] = (a02 * b10 - a01 * b11 - a03 * b09) * det;\n out[2] = (a31 * b05 - a32 * b04 + a33 * b03) * det;\n out[3] = (a22 * b04 - a21 * b05 - a23 * b03) * det;\n out[4] = (a12 * b08 - a10 * b11 - a13 * b07) * det;\n out[5] = (a00 * b11 - a02 * b08 + a03 * b07) * det;\n out[6] = (a32 * b02 - a30 * b05 - a33 * b01) * det;\n out[7] = (a20 * b05 - a22 * b02 + a23 * b01) * det;\n out[8] = (a10 * b10 - a11 * b08 + a13 * b06) * det;\n out[9] = (a01 * b08 - a00 * b10 - a03 * b06) * det;\n out[10] = (a30 * b04 - a31 * b02 + a33 * b00) * det;\n out[11] = (a21 * b02 - a20 * b04 - a23 * b00) * det;\n out[12] = (a11 * b07 - a10 * b09 - a12 * b06) * det;\n out[13] = (a00 * b09 - a01 * b07 + a02 * b06) * det;\n out[14] = (a31 * b01 - a30 * b03 - a32 * b00) * det;\n out[15] = (a20 * b03 - a21 * b01 + a22 * b00) * det;\n return out;\n}\n/**\n * Calculates the adjugate of a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function adjoint(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n out[0] = a11 * (a22 * a33 - a23 * a32) - a21 * (a12 * a33 - a13 * a32) + a31 * (a12 * a23 - a13 * a22);\n out[1] = -(a01 * (a22 * a33 - a23 * a32) - a21 * (a02 * a33 - a03 * a32) + a31 * (a02 * a23 - a03 * a22));\n out[2] = a01 * (a12 * a33 - a13 * a32) - a11 * (a02 * a33 - a03 * a32) + a31 * (a02 * a13 - a03 * a12);\n out[3] = -(a01 * (a12 * a23 - a13 * a22) - a11 * (a02 * a23 - a03 * a22) + a21 * (a02 * a13 - a03 * a12));\n out[4] = -(a10 * (a22 * a33 - a23 * a32) - a20 * (a12 * a33 - a13 * a32) + a30 * (a12 * a23 - a13 * a22));\n out[5] = a00 * (a22 * a33 - a23 * a32) - a20 * (a02 * a33 - a03 * a32) + a30 * (a02 * a23 - a03 * a22);\n out[6] = -(a00 * (a12 * a33 - a13 * a32) - a10 * (a02 * a33 - a03 * a32) + a30 * (a02 * a13 - a03 * a12));\n out[7] = a00 * (a12 * a23 - a13 * a22) - a10 * (a02 * a23 - a03 * a22) + a20 * (a02 * a13 - a03 * a12);\n out[8] = a10 * (a21 * a33 - a23 * a31) - a20 * (a11 * a33 - a13 * a31) + a30 * (a11 * a23 - a13 * a21);\n out[9] = -(a00 * (a21 * a33 - a23 * a31) - a20 * (a01 * a33 - a03 * a31) + a30 * (a01 * a23 - a03 * a21));\n out[10] = a00 * (a11 * a33 - a13 * a31) - a10 * (a01 * a33 - a03 * a31) + a30 * (a01 * a13 - a03 * a11);\n out[11] = -(a00 * (a11 * a23 - a13 * a21) - a10 * (a01 * a23 - a03 * a21) + a20 * (a01 * a13 - a03 * a11));\n out[12] = -(a10 * (a21 * a32 - a22 * a31) - a20 * (a11 * a32 - a12 * a31) + a30 * (a11 * a22 - a12 * a21));\n out[13] = a00 * (a21 * a32 - a22 * a31) - a20 * (a01 * a32 - a02 * a31) + a30 * (a01 * a22 - a02 * a21);\n out[14] = -(a00 * (a11 * a32 - a12 * a31) - a10 * (a01 * a32 - a02 * a31) + a30 * (a01 * a12 - a02 * a11));\n out[15] = a00 * (a11 * a22 - a12 * a21) - a10 * (a01 * a22 - a02 * a21) + a20 * (a01 * a12 - a02 * a11);\n return out;\n}\n/**\n * Calculates the determinant of a mat4\n *\n * @param {ReadonlyMat4} a the source matrix\n * @returns {Number} determinant of a\n */\n\nexport function determinant(a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n var b00 = a00 * a11 - a01 * a10;\n var b01 = a00 * a12 - a02 * a10;\n var b02 = a00 * a13 - a03 * a10;\n var b03 = a01 * a12 - a02 * a11;\n var b04 = a01 * a13 - a03 * a11;\n var b05 = a02 * a13 - a03 * a12;\n var b06 = a20 * a31 - a21 * a30;\n var b07 = a20 * a32 - a22 * a30;\n var b08 = a20 * a33 - a23 * a30;\n var b09 = a21 * a32 - a22 * a31;\n var b10 = a21 * a33 - a23 * a31;\n var b11 = a22 * a33 - a23 * a32; // Calculate the determinant\n\n return b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n}\n/**\n * Multiplies two mat4s\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @returns {mat4} out\n */\n\nexport function multiply(out, a, b) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15]; // Cache only the current line of the second matrix\n\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n out[0] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[1] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[2] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[3] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n b0 = b[4];\n b1 = b[5];\n b2 = b[6];\n b3 = b[7];\n out[4] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[5] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[6] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[7] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n b0 = b[8];\n b1 = b[9];\n b2 = b[10];\n b3 = b[11];\n out[8] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[9] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[10] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[11] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n b0 = b[12];\n b1 = b[13];\n b2 = b[14];\n b3 = b[15];\n out[12] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[13] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[14] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[15] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n return out;\n}\n/**\n * Translate a mat4 by the given vector\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to translate\n * @param {ReadonlyVec3} v vector to translate by\n * @returns {mat4} out\n */\n\nexport function translate(out, a, v) {\n var x = v[0],\n y = v[1],\n z = v[2];\n var a00, a01, a02, a03;\n var a10, a11, a12, a13;\n var a20, a21, a22, a23;\n\n if (a === out) {\n out[12] = a[0] * x + a[4] * y + a[8] * z + a[12];\n out[13] = a[1] * x + a[5] * y + a[9] * z + a[13];\n out[14] = a[2] * x + a[6] * y + a[10] * z + a[14];\n out[15] = a[3] * x + a[7] * y + a[11] * z + a[15];\n } else {\n a00 = a[0];\n a01 = a[1];\n a02 = a[2];\n a03 = a[3];\n a10 = a[4];\n a11 = a[5];\n a12 = a[6];\n a13 = a[7];\n a20 = a[8];\n a21 = a[9];\n a22 = a[10];\n a23 = a[11];\n out[0] = a00;\n out[1] = a01;\n out[2] = a02;\n out[3] = a03;\n out[4] = a10;\n out[5] = a11;\n out[6] = a12;\n out[7] = a13;\n out[8] = a20;\n out[9] = a21;\n out[10] = a22;\n out[11] = a23;\n out[12] = a00 * x + a10 * y + a20 * z + a[12];\n out[13] = a01 * x + a11 * y + a21 * z + a[13];\n out[14] = a02 * x + a12 * y + a22 * z + a[14];\n out[15] = a03 * x + a13 * y + a23 * z + a[15];\n }\n\n return out;\n}\n/**\n * Scales the mat4 by the dimensions in the given vec3 not using vectorization\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to scale\n * @param {ReadonlyVec3} v the vec3 to scale the matrix by\n * @returns {mat4} out\n **/\n\nexport function scale(out, a, v) {\n var x = v[0],\n y = v[1],\n z = v[2];\n out[0] = a[0] * x;\n out[1] = a[1] * x;\n out[2] = a[2] * x;\n out[3] = a[3] * x;\n out[4] = a[4] * y;\n out[5] = a[5] * y;\n out[6] = a[6] * y;\n out[7] = a[7] * y;\n out[8] = a[8] * z;\n out[9] = a[9] * z;\n out[10] = a[10] * z;\n out[11] = a[11] * z;\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n return out;\n}\n/**\n * Rotates a mat4 by the given angle around the given axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @param {ReadonlyVec3} axis the axis to rotate around\n * @returns {mat4} out\n */\n\nexport function rotate(out, a, rad, axis) {\n var x = axis[0],\n y = axis[1],\n z = axis[2];\n var len = Math.hypot(x, y, z);\n var s, c, t;\n var a00, a01, a02, a03;\n var a10, a11, a12, a13;\n var a20, a21, a22, a23;\n var b00, b01, b02;\n var b10, b11, b12;\n var b20, b21, b22;\n\n if (len < glMatrix.EPSILON) {\n return null;\n }\n\n len = 1 / len;\n x *= len;\n y *= len;\n z *= len;\n s = Math.sin(rad);\n c = Math.cos(rad);\n t = 1 - c;\n a00 = a[0];\n a01 = a[1];\n a02 = a[2];\n a03 = a[3];\n a10 = a[4];\n a11 = a[5];\n a12 = a[6];\n a13 = a[7];\n a20 = a[8];\n a21 = a[9];\n a22 = a[10];\n a23 = a[11]; // Construct the elements of the rotation matrix\n\n b00 = x * x * t + c;\n b01 = y * x * t + z * s;\n b02 = z * x * t - y * s;\n b10 = x * y * t - z * s;\n b11 = y * y * t + c;\n b12 = z * y * t + x * s;\n b20 = x * z * t + y * s;\n b21 = y * z * t - x * s;\n b22 = z * z * t + c; // Perform rotation-specific matrix multiplication\n\n out[0] = a00 * b00 + a10 * b01 + a20 * b02;\n out[1] = a01 * b00 + a11 * b01 + a21 * b02;\n out[2] = a02 * b00 + a12 * b01 + a22 * b02;\n out[3] = a03 * b00 + a13 * b01 + a23 * b02;\n out[4] = a00 * b10 + a10 * b11 + a20 * b12;\n out[5] = a01 * b10 + a11 * b11 + a21 * b12;\n out[6] = a02 * b10 + a12 * b11 + a22 * b12;\n out[7] = a03 * b10 + a13 * b11 + a23 * b12;\n out[8] = a00 * b20 + a10 * b21 + a20 * b22;\n out[9] = a01 * b20 + a11 * b21 + a21 * b22;\n out[10] = a02 * b20 + a12 * b21 + a22 * b22;\n out[11] = a03 * b20 + a13 * b21 + a23 * b22;\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged last row\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n }\n\n return out;\n}\n/**\n * Rotates a matrix by the given angle around the X axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function rotateX(out, a, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n var a10 = a[4];\n var a11 = a[5];\n var a12 = a[6];\n var a13 = a[7];\n var a20 = a[8];\n var a21 = a[9];\n var a22 = a[10];\n var a23 = a[11];\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged rows\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n } // Perform axis-specific matrix multiplication\n\n\n out[4] = a10 * c + a20 * s;\n out[5] = a11 * c + a21 * s;\n out[6] = a12 * c + a22 * s;\n out[7] = a13 * c + a23 * s;\n out[8] = a20 * c - a10 * s;\n out[9] = a21 * c - a11 * s;\n out[10] = a22 * c - a12 * s;\n out[11] = a23 * c - a13 * s;\n return out;\n}\n/**\n * Rotates a matrix by the given angle around the Y axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function rotateY(out, a, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n var a00 = a[0];\n var a01 = a[1];\n var a02 = a[2];\n var a03 = a[3];\n var a20 = a[8];\n var a21 = a[9];\n var a22 = a[10];\n var a23 = a[11];\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged rows\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n } // Perform axis-specific matrix multiplication\n\n\n out[0] = a00 * c - a20 * s;\n out[1] = a01 * c - a21 * s;\n out[2] = a02 * c - a22 * s;\n out[3] = a03 * c - a23 * s;\n out[8] = a00 * s + a20 * c;\n out[9] = a01 * s + a21 * c;\n out[10] = a02 * s + a22 * c;\n out[11] = a03 * s + a23 * c;\n return out;\n}\n/**\n * Rotates a matrix by the given angle around the Z axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function rotateZ(out, a, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n var a00 = a[0];\n var a01 = a[1];\n var a02 = a[2];\n var a03 = a[3];\n var a10 = a[4];\n var a11 = a[5];\n var a12 = a[6];\n var a13 = a[7];\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged last row\n out[8] = a[8];\n out[9] = a[9];\n out[10] = a[10];\n out[11] = a[11];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n } // Perform axis-specific matrix multiplication\n\n\n out[0] = a00 * c + a10 * s;\n out[1] = a01 * c + a11 * s;\n out[2] = a02 * c + a12 * s;\n out[3] = a03 * c + a13 * s;\n out[4] = a10 * c - a00 * s;\n out[5] = a11 * c - a01 * s;\n out[6] = a12 * c - a02 * s;\n out[7] = a13 * c - a03 * s;\n return out;\n}\n/**\n * Creates a matrix from a vector translation\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, dest, vec);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {ReadonlyVec3} v Translation vector\n * @returns {mat4} out\n */\n\nexport function fromTranslation(out, v) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = 1;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 1;\n out[11] = 0;\n out[12] = v[0];\n out[13] = v[1];\n out[14] = v[2];\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a vector scaling\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.scale(dest, dest, vec);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {ReadonlyVec3} v Scaling vector\n * @returns {mat4} out\n */\n\nexport function fromScaling(out, v) {\n out[0] = v[0];\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = v[1];\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = v[2];\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a given angle around a given axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotate(dest, dest, rad, axis);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @param {ReadonlyVec3} axis the axis to rotate around\n * @returns {mat4} out\n */\n\nexport function fromRotation(out, rad, axis) {\n var x = axis[0],\n y = axis[1],\n z = axis[2];\n var len = Math.hypot(x, y, z);\n var s, c, t;\n\n if (len < glMatrix.EPSILON) {\n return null;\n }\n\n len = 1 / len;\n x *= len;\n y *= len;\n z *= len;\n s = Math.sin(rad);\n c = Math.cos(rad);\n t = 1 - c; // Perform rotation-specific matrix multiplication\n\n out[0] = x * x * t + c;\n out[1] = y * x * t + z * s;\n out[2] = z * x * t - y * s;\n out[3] = 0;\n out[4] = x * y * t - z * s;\n out[5] = y * y * t + c;\n out[6] = z * y * t + x * s;\n out[7] = 0;\n out[8] = x * z * t + y * s;\n out[9] = y * z * t - x * s;\n out[10] = z * z * t + c;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from the given angle around the X axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotateX(dest, dest, rad);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function fromXRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad); // Perform axis-specific matrix multiplication\n\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = c;\n out[6] = s;\n out[7] = 0;\n out[8] = 0;\n out[9] = -s;\n out[10] = c;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from the given angle around the Y axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotateY(dest, dest, rad);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function fromYRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad); // Perform axis-specific matrix multiplication\n\n out[0] = c;\n out[1] = 0;\n out[2] = -s;\n out[3] = 0;\n out[4] = 0;\n out[5] = 1;\n out[6] = 0;\n out[7] = 0;\n out[8] = s;\n out[9] = 0;\n out[10] = c;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from the given angle around the Z axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotateZ(dest, dest, rad);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function fromZRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad); // Perform axis-specific matrix multiplication\n\n out[0] = c;\n out[1] = s;\n out[2] = 0;\n out[3] = 0;\n out[4] = -s;\n out[5] = c;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 1;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a quaternion rotation and vector translation\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, vec);\n * let quatMat = mat4.create();\n * quat4.toMat4(quat, quatMat);\n * mat4.multiply(dest, quatMat);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {ReadonlyVec3} v Translation vector\n * @returns {mat4} out\n */\n\nexport function fromRotationTranslation(out, q, v) {\n // Quaternion math\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var xy = x * y2;\n var xz = x * z2;\n var yy = y * y2;\n var yz = y * z2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n out[0] = 1 - (yy + zz);\n out[1] = xy + wz;\n out[2] = xz - wy;\n out[3] = 0;\n out[4] = xy - wz;\n out[5] = 1 - (xx + zz);\n out[6] = yz + wx;\n out[7] = 0;\n out[8] = xz + wy;\n out[9] = yz - wx;\n out[10] = 1 - (xx + yy);\n out[11] = 0;\n out[12] = v[0];\n out[13] = v[1];\n out[14] = v[2];\n out[15] = 1;\n return out;\n}\n/**\n * Creates a new mat4 from a dual quat.\n *\n * @param {mat4} out Matrix\n * @param {ReadonlyQuat2} a Dual Quaternion\n * @returns {mat4} mat4 receiving operation result\n */\n\nexport function fromQuat2(out, a) {\n var translation = new glMatrix.ARRAY_TYPE(3);\n var bx = -a[0],\n by = -a[1],\n bz = -a[2],\n bw = a[3],\n ax = a[4],\n ay = a[5],\n az = a[6],\n aw = a[7];\n var magnitude = bx * bx + by * by + bz * bz + bw * bw; //Only scale if it makes sense\n\n if (magnitude > 0) {\n translation[0] = (ax * bw + aw * bx + ay * bz - az * by) * 2 / magnitude;\n translation[1] = (ay * bw + aw * by + az * bx - ax * bz) * 2 / magnitude;\n translation[2] = (az * bw + aw * bz + ax * by - ay * bx) * 2 / magnitude;\n } else {\n translation[0] = (ax * bw + aw * bx + ay * bz - az * by) * 2;\n translation[1] = (ay * bw + aw * by + az * bx - ax * bz) * 2;\n translation[2] = (az * bw + aw * bz + ax * by - ay * bx) * 2;\n }\n\n fromRotationTranslation(out, a, translation);\n return out;\n}\n/**\n * Returns the translation vector component of a transformation\n * matrix. If a matrix is built with fromRotationTranslation,\n * the returned vector will be the same as the translation vector\n * originally supplied.\n * @param {vec3} out Vector to receive translation component\n * @param {ReadonlyMat4} mat Matrix to be decomposed (input)\n * @return {vec3} out\n */\n\nexport function getTranslation(out, mat) {\n out[0] = mat[12];\n out[1] = mat[13];\n out[2] = mat[14];\n return out;\n}\n/**\n * Returns the scaling factor component of a transformation\n * matrix. If a matrix is built with fromRotationTranslationScale\n * with a normalized Quaternion paramter, the returned vector will be\n * the same as the scaling vector\n * originally supplied.\n * @param {vec3} out Vector to receive scaling factor component\n * @param {ReadonlyMat4} mat Matrix to be decomposed (input)\n * @return {vec3} out\n */\n\nexport function getScaling(out, mat) {\n var m11 = mat[0];\n var m12 = mat[1];\n var m13 = mat[2];\n var m21 = mat[4];\n var m22 = mat[5];\n var m23 = mat[6];\n var m31 = mat[8];\n var m32 = mat[9];\n var m33 = mat[10];\n out[0] = Math.hypot(m11, m12, m13);\n out[1] = Math.hypot(m21, m22, m23);\n out[2] = Math.hypot(m31, m32, m33);\n return out;\n}\n/**\n * Returns a quaternion representing the rotational component\n * of a transformation matrix. If a matrix is built with\n * fromRotationTranslation, the returned quaternion will be the\n * same as the quaternion originally supplied.\n * @param {quat} out Quaternion to receive the rotation component\n * @param {ReadonlyMat4} mat Matrix to be decomposed (input)\n * @return {quat} out\n */\n\nexport function getRotation(out, mat) {\n var scaling = new glMatrix.ARRAY_TYPE(3);\n getScaling(scaling, mat);\n var is1 = 1 / scaling[0];\n var is2 = 1 / scaling[1];\n var is3 = 1 / scaling[2];\n var sm11 = mat[0] * is1;\n var sm12 = mat[1] * is2;\n var sm13 = mat[2] * is3;\n var sm21 = mat[4] * is1;\n var sm22 = mat[5] * is2;\n var sm23 = mat[6] * is3;\n var sm31 = mat[8] * is1;\n var sm32 = mat[9] * is2;\n var sm33 = mat[10] * is3;\n var trace = sm11 + sm22 + sm33;\n var S = 0;\n\n if (trace > 0) {\n S = Math.sqrt(trace + 1.0) * 2;\n out[3] = 0.25 * S;\n out[0] = (sm23 - sm32) / S;\n out[1] = (sm31 - sm13) / S;\n out[2] = (sm12 - sm21) / S;\n } else if (sm11 > sm22 && sm11 > sm33) {\n S = Math.sqrt(1.0 + sm11 - sm22 - sm33) * 2;\n out[3] = (sm23 - sm32) / S;\n out[0] = 0.25 * S;\n out[1] = (sm12 + sm21) / S;\n out[2] = (sm31 + sm13) / S;\n } else if (sm22 > sm33) {\n S = Math.sqrt(1.0 + sm22 - sm11 - sm33) * 2;\n out[3] = (sm31 - sm13) / S;\n out[0] = (sm12 + sm21) / S;\n out[1] = 0.25 * S;\n out[2] = (sm23 + sm32) / S;\n } else {\n S = Math.sqrt(1.0 + sm33 - sm11 - sm22) * 2;\n out[3] = (sm12 - sm21) / S;\n out[0] = (sm31 + sm13) / S;\n out[1] = (sm23 + sm32) / S;\n out[2] = 0.25 * S;\n }\n\n return out;\n}\n/**\n * Creates a matrix from a quaternion rotation, vector translation and vector scale\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, vec);\n * let quatMat = mat4.create();\n * quat4.toMat4(quat, quatMat);\n * mat4.multiply(dest, quatMat);\n * mat4.scale(dest, scale)\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {ReadonlyVec3} v Translation vector\n * @param {ReadonlyVec3} s Scaling vector\n * @returns {mat4} out\n */\n\nexport function fromRotationTranslationScale(out, q, v, s) {\n // Quaternion math\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var xy = x * y2;\n var xz = x * z2;\n var yy = y * y2;\n var yz = y * z2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n var sx = s[0];\n var sy = s[1];\n var sz = s[2];\n out[0] = (1 - (yy + zz)) * sx;\n out[1] = (xy + wz) * sx;\n out[2] = (xz - wy) * sx;\n out[3] = 0;\n out[4] = (xy - wz) * sy;\n out[5] = (1 - (xx + zz)) * sy;\n out[6] = (yz + wx) * sy;\n out[7] = 0;\n out[8] = (xz + wy) * sz;\n out[9] = (yz - wx) * sz;\n out[10] = (1 - (xx + yy)) * sz;\n out[11] = 0;\n out[12] = v[0];\n out[13] = v[1];\n out[14] = v[2];\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a quaternion rotation, vector translation and vector scale, rotating and scaling around the given origin\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, vec);\n * mat4.translate(dest, origin);\n * let quatMat = mat4.create();\n * quat4.toMat4(quat, quatMat);\n * mat4.multiply(dest, quatMat);\n * mat4.scale(dest, scale)\n * mat4.translate(dest, negativeOrigin);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {ReadonlyVec3} v Translation vector\n * @param {ReadonlyVec3} s Scaling vector\n * @param {ReadonlyVec3} o The origin vector around which to scale and rotate\n * @returns {mat4} out\n */\n\nexport function fromRotationTranslationScaleOrigin(out, q, v, s, o) {\n // Quaternion math\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var xy = x * y2;\n var xz = x * z2;\n var yy = y * y2;\n var yz = y * z2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n var sx = s[0];\n var sy = s[1];\n var sz = s[2];\n var ox = o[0];\n var oy = o[1];\n var oz = o[2];\n var out0 = (1 - (yy + zz)) * sx;\n var out1 = (xy + wz) * sx;\n var out2 = (xz - wy) * sx;\n var out4 = (xy - wz) * sy;\n var out5 = (1 - (xx + zz)) * sy;\n var out6 = (yz + wx) * sy;\n var out8 = (xz + wy) * sz;\n var out9 = (yz - wx) * sz;\n var out10 = (1 - (xx + yy)) * sz;\n out[0] = out0;\n out[1] = out1;\n out[2] = out2;\n out[3] = 0;\n out[4] = out4;\n out[5] = out5;\n out[6] = out6;\n out[7] = 0;\n out[8] = out8;\n out[9] = out9;\n out[10] = out10;\n out[11] = 0;\n out[12] = v[0] + ox - (out0 * ox + out4 * oy + out8 * oz);\n out[13] = v[1] + oy - (out1 * ox + out5 * oy + out9 * oz);\n out[14] = v[2] + oz - (out2 * ox + out6 * oy + out10 * oz);\n out[15] = 1;\n return out;\n}\n/**\n * Calculates a 4x4 matrix from the given quaternion\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {ReadonlyQuat} q Quaternion to create matrix from\n *\n * @returns {mat4} out\n */\n\nexport function fromQuat(out, q) {\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var yx = y * x2;\n var yy = y * y2;\n var zx = z * x2;\n var zy = z * y2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n out[0] = 1 - yy - zz;\n out[1] = yx + wz;\n out[2] = zx - wy;\n out[3] = 0;\n out[4] = yx - wz;\n out[5] = 1 - xx - zz;\n out[6] = zy + wx;\n out[7] = 0;\n out[8] = zx + wy;\n out[9] = zy - wx;\n out[10] = 1 - xx - yy;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Generates a frustum matrix with the given bounds\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {Number} left Left bound of the frustum\n * @param {Number} right Right bound of the frustum\n * @param {Number} bottom Bottom bound of the frustum\n * @param {Number} top Top bound of the frustum\n * @param {Number} near Near bound of the frustum\n * @param {Number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function frustum(out, left, right, bottom, top, near, far) {\n var rl = 1 / (right - left);\n var tb = 1 / (top - bottom);\n var nf = 1 / (near - far);\n out[0] = near * 2 * rl;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = near * 2 * tb;\n out[6] = 0;\n out[7] = 0;\n out[8] = (right + left) * rl;\n out[9] = (top + bottom) * tb;\n out[10] = (far + near) * nf;\n out[11] = -1;\n out[12] = 0;\n out[13] = 0;\n out[14] = far * near * 2 * nf;\n out[15] = 0;\n return out;\n}\n/**\n * Generates a perspective projection matrix with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [-1, 1],\n * which matches WebGL/OpenGL's clip volume.\n * Passing null/undefined/no value for far will generate infinite projection matrix.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} fovy Vertical field of view in radians\n * @param {number} aspect Aspect ratio. typically viewport width/height\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum, can be null or Infinity\n * @returns {mat4} out\n */\n\nexport function perspectiveNO(out, fovy, aspect, near, far) {\n var f = 1.0 / Math.tan(fovy / 2),\n nf;\n out[0] = f / aspect;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = f;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[11] = -1;\n out[12] = 0;\n out[13] = 0;\n out[15] = 0;\n\n if (far != null && far !== Infinity) {\n nf = 1 / (near - far);\n out[10] = (far + near) * nf;\n out[14] = 2 * far * near * nf;\n } else {\n out[10] = -1;\n out[14] = -2 * near;\n }\n\n return out;\n}\n/**\n * Alias for {@link mat4.perspectiveNO}\n * @function\n */\n\nexport var perspective = perspectiveNO;\n/**\n * Generates a perspective projection matrix suitable for WebGPU with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [0, 1],\n * which matches WebGPU/Vulkan/DirectX/Metal's clip volume.\n * Passing null/undefined/no value for far will generate infinite projection matrix.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} fovy Vertical field of view in radians\n * @param {number} aspect Aspect ratio. typically viewport width/height\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum, can be null or Infinity\n * @returns {mat4} out\n */\n\nexport function perspectiveZO(out, fovy, aspect, near, far) {\n var f = 1.0 / Math.tan(fovy / 2),\n nf;\n out[0] = f / aspect;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = f;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[11] = -1;\n out[12] = 0;\n out[13] = 0;\n out[15] = 0;\n\n if (far != null && far !== Infinity) {\n nf = 1 / (near - far);\n out[10] = far * nf;\n out[14] = far * near * nf;\n } else {\n out[10] = -1;\n out[14] = -near;\n }\n\n return out;\n}\n/**\n * Generates a perspective projection matrix with the given field of view.\n * This is primarily useful for generating projection matrices to be used\n * with the still experiemental WebVR API.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {Object} fov Object containing the following values: upDegrees, downDegrees, leftDegrees, rightDegrees\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function perspectiveFromFieldOfView(out, fov, near, far) {\n var upTan = Math.tan(fov.upDegrees * Math.PI / 180.0);\n var downTan = Math.tan(fov.downDegrees * Math.PI / 180.0);\n var leftTan = Math.tan(fov.leftDegrees * Math.PI / 180.0);\n var rightTan = Math.tan(fov.rightDegrees * Math.PI / 180.0);\n var xScale = 2.0 / (leftTan + rightTan);\n var yScale = 2.0 / (upTan + downTan);\n out[0] = xScale;\n out[1] = 0.0;\n out[2] = 0.0;\n out[3] = 0.0;\n out[4] = 0.0;\n out[5] = yScale;\n out[6] = 0.0;\n out[7] = 0.0;\n out[8] = -((leftTan - rightTan) * xScale * 0.5);\n out[9] = (upTan - downTan) * yScale * 0.5;\n out[10] = far / (near - far);\n out[11] = -1.0;\n out[12] = 0.0;\n out[13] = 0.0;\n out[14] = far * near / (near - far);\n out[15] = 0.0;\n return out;\n}\n/**\n * Generates a orthogonal projection matrix with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [-1, 1],\n * which matches WebGL/OpenGL's clip volume.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} left Left bound of the frustum\n * @param {number} right Right bound of the frustum\n * @param {number} bottom Bottom bound of the frustum\n * @param {number} top Top bound of the frustum\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function orthoNO(out, left, right, bottom, top, near, far) {\n var lr = 1 / (left - right);\n var bt = 1 / (bottom - top);\n var nf = 1 / (near - far);\n out[0] = -2 * lr;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = -2 * bt;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 2 * nf;\n out[11] = 0;\n out[12] = (left + right) * lr;\n out[13] = (top + bottom) * bt;\n out[14] = (far + near) * nf;\n out[15] = 1;\n return out;\n}\n/**\n * Alias for {@link mat4.orthoNO}\n * @function\n */\n\nexport var ortho = orthoNO;\n/**\n * Generates a orthogonal projection matrix with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [0, 1],\n * which matches WebGPU/Vulkan/DirectX/Metal's clip volume.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} left Left bound of the frustum\n * @param {number} right Right bound of the frustum\n * @param {number} bottom Bottom bound of the frustum\n * @param {number} top Top bound of the frustum\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function orthoZO(out, left, right, bottom, top, near, far) {\n var lr = 1 / (left - right);\n var bt = 1 / (bottom - top);\n var nf = 1 / (near - far);\n out[0] = -2 * lr;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = -2 * bt;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = nf;\n out[11] = 0;\n out[12] = (left + right) * lr;\n out[13] = (top + bottom) * bt;\n out[14] = near * nf;\n out[15] = 1;\n return out;\n}\n/**\n * Generates a look-at matrix with the given eye position, focal point, and up axis.\n * If you want a matrix that actually makes an object look at another object, you should use targetTo instead.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {ReadonlyVec3} eye Position of the viewer\n * @param {ReadonlyVec3} center Point the viewer is looking at\n * @param {ReadonlyVec3} up vec3 pointing up\n * @returns {mat4} out\n */\n\nexport function lookAt(out, eye, center, up) {\n var x0, x1, x2, y0, y1, y2, z0, z1, z2, len;\n var eyex = eye[0];\n var eyey = eye[1];\n var eyez = eye[2];\n var upx = up[0];\n var upy = up[1];\n var upz = up[2];\n var centerx = center[0];\n var centery = center[1];\n var centerz = center[2];\n\n if (Math.abs(eyex - centerx) < glMatrix.EPSILON && Math.abs(eyey - centery) < glMatrix.EPSILON && Math.abs(eyez - centerz) < glMatrix.EPSILON) {\n return identity(out);\n }\n\n z0 = eyex - centerx;\n z1 = eyey - centery;\n z2 = eyez - centerz;\n len = 1 / Math.hypot(z0, z1, z2);\n z0 *= len;\n z1 *= len;\n z2 *= len;\n x0 = upy * z2 - upz * z1;\n x1 = upz * z0 - upx * z2;\n x2 = upx * z1 - upy * z0;\n len = Math.hypot(x0, x1, x2);\n\n if (!len) {\n x0 = 0;\n x1 = 0;\n x2 = 0;\n } else {\n len = 1 / len;\n x0 *= len;\n x1 *= len;\n x2 *= len;\n }\n\n y0 = z1 * x2 - z2 * x1;\n y1 = z2 * x0 - z0 * x2;\n y2 = z0 * x1 - z1 * x0;\n len = Math.hypot(y0, y1, y2);\n\n if (!len) {\n y0 = 0;\n y1 = 0;\n y2 = 0;\n } else {\n len = 1 / len;\n y0 *= len;\n y1 *= len;\n y2 *= len;\n }\n\n out[0] = x0;\n out[1] = y0;\n out[2] = z0;\n out[3] = 0;\n out[4] = x1;\n out[5] = y1;\n out[6] = z1;\n out[7] = 0;\n out[8] = x2;\n out[9] = y2;\n out[10] = z2;\n out[11] = 0;\n out[12] = -(x0 * eyex + x1 * eyey + x2 * eyez);\n out[13] = -(y0 * eyex + y1 * eyey + y2 * eyez);\n out[14] = -(z0 * eyex + z1 * eyey + z2 * eyez);\n out[15] = 1;\n return out;\n}\n/**\n * Generates a matrix that makes something look at something else.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {ReadonlyVec3} eye Position of the viewer\n * @param {ReadonlyVec3} center Point the viewer is looking at\n * @param {ReadonlyVec3} up vec3 pointing up\n * @returns {mat4} out\n */\n\nexport function targetTo(out, eye, target, up) {\n var eyex = eye[0],\n eyey = eye[1],\n eyez = eye[2],\n upx = up[0],\n upy = up[1],\n upz = up[2];\n var z0 = eyex - target[0],\n z1 = eyey - target[1],\n z2 = eyez - target[2];\n var len = z0 * z0 + z1 * z1 + z2 * z2;\n\n if (len > 0) {\n len = 1 / Math.sqrt(len);\n z0 *= len;\n z1 *= len;\n z2 *= len;\n }\n\n var x0 = upy * z2 - upz * z1,\n x1 = upz * z0 - upx * z2,\n x2 = upx * z1 - upy * z0;\n len = x0 * x0 + x1 * x1 + x2 * x2;\n\n if (len > 0) {\n len = 1 / Math.sqrt(len);\n x0 *= len;\n x1 *= len;\n x2 *= len;\n }\n\n out[0] = x0;\n out[1] = x1;\n out[2] = x2;\n out[3] = 0;\n out[4] = z1 * x2 - z2 * x1;\n out[5] = z2 * x0 - z0 * x2;\n out[6] = z0 * x1 - z1 * x0;\n out[7] = 0;\n out[8] = z0;\n out[9] = z1;\n out[10] = z2;\n out[11] = 0;\n out[12] = eyex;\n out[13] = eyey;\n out[14] = eyez;\n out[15] = 1;\n return out;\n}\n/**\n * Returns a string representation of a mat4\n *\n * @param {ReadonlyMat4} a matrix to represent as a string\n * @returns {String} string representation of the matrix\n */\n\nexport function str(a) {\n return \"mat4(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \", \" + a[4] + \", \" + a[5] + \", \" + a[6] + \", \" + a[7] + \", \" + a[8] + \", \" + a[9] + \", \" + a[10] + \", \" + a[11] + \", \" + a[12] + \", \" + a[13] + \", \" + a[14] + \", \" + a[15] + \")\";\n}\n/**\n * Returns Frobenius norm of a mat4\n *\n * @param {ReadonlyMat4} a the matrix to calculate Frobenius norm of\n * @returns {Number} Frobenius norm\n */\n\nexport function frob(a) {\n return Math.hypot(a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8], a[9], a[10], a[11], a[12], a[13], a[14], a[15]);\n}\n/**\n * Adds two mat4's\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @returns {mat4} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n out[4] = a[4] + b[4];\n out[5] = a[5] + b[5];\n out[6] = a[6] + b[6];\n out[7] = a[7] + b[7];\n out[8] = a[8] + b[8];\n out[9] = a[9] + b[9];\n out[10] = a[10] + b[10];\n out[11] = a[11] + b[11];\n out[12] = a[12] + b[12];\n out[13] = a[13] + b[13];\n out[14] = a[14] + b[14];\n out[15] = a[15] + b[15];\n return out;\n}\n/**\n * Subtracts matrix b from matrix a\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @returns {mat4} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n out[4] = a[4] - b[4];\n out[5] = a[5] - b[5];\n out[6] = a[6] - b[6];\n out[7] = a[7] - b[7];\n out[8] = a[8] - b[8];\n out[9] = a[9] - b[9];\n out[10] = a[10] - b[10];\n out[11] = a[11] - b[11];\n out[12] = a[12] - b[12];\n out[13] = a[13] - b[13];\n out[14] = a[14] - b[14];\n out[15] = a[15] - b[15];\n return out;\n}\n/**\n * Multiply each element of the matrix by a scalar.\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to scale\n * @param {Number} b amount to scale the matrix's elements by\n * @returns {mat4} out\n */\n\nexport function multiplyScalar(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n out[4] = a[4] * b;\n out[5] = a[5] * b;\n out[6] = a[6] * b;\n out[7] = a[7] * b;\n out[8] = a[8] * b;\n out[9] = a[9] * b;\n out[10] = a[10] * b;\n out[11] = a[11] * b;\n out[12] = a[12] * b;\n out[13] = a[13] * b;\n out[14] = a[14] * b;\n out[15] = a[15] * b;\n return out;\n}\n/**\n * Adds two mat4's after multiplying each element of the second operand by a scalar value.\n *\n * @param {mat4} out the receiving vector\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @param {Number} scale the amount to scale b's elements by before adding\n * @returns {mat4} out\n */\n\nexport function multiplyScalarAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n out[4] = a[4] + b[4] * scale;\n out[5] = a[5] + b[5] * scale;\n out[6] = a[6] + b[6] * scale;\n out[7] = a[7] + b[7] * scale;\n out[8] = a[8] + b[8] * scale;\n out[9] = a[9] + b[9] * scale;\n out[10] = a[10] + b[10] * scale;\n out[11] = a[11] + b[11] * scale;\n out[12] = a[12] + b[12] * scale;\n out[13] = a[13] + b[13] * scale;\n out[14] = a[14] + b[14] * scale;\n out[15] = a[15] + b[15] * scale;\n return out;\n}\n/**\n * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyMat4} a The first matrix.\n * @param {ReadonlyMat4} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5] && a[6] === b[6] && a[7] === b[7] && a[8] === b[8] && a[9] === b[9] && a[10] === b[10] && a[11] === b[11] && a[12] === b[12] && a[13] === b[13] && a[14] === b[14] && a[15] === b[15];\n}\n/**\n * Returns whether or not the matrices have approximately the same elements in the same position.\n *\n * @param {ReadonlyMat4} a The first matrix.\n * @param {ReadonlyMat4} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var a4 = a[4],\n a5 = a[5],\n a6 = a[6],\n a7 = a[7];\n var a8 = a[8],\n a9 = a[9],\n a10 = a[10],\n a11 = a[11];\n var a12 = a[12],\n a13 = a[13],\n a14 = a[14],\n a15 = a[15];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n var b4 = b[4],\n b5 = b[5],\n b6 = b[6],\n b7 = b[7];\n var b8 = b[8],\n b9 = b[9],\n b10 = b[10],\n b11 = b[11];\n var b12 = b[12],\n b13 = b[13],\n b14 = b[14],\n b15 = b[15];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5)) && Math.abs(a6 - b6) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a6), Math.abs(b6)) && Math.abs(a7 - b7) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a7), Math.abs(b7)) && Math.abs(a8 - b8) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a8), Math.abs(b8)) && Math.abs(a9 - b9) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a9), Math.abs(b9)) && Math.abs(a10 - b10) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a10), Math.abs(b10)) && Math.abs(a11 - b11) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a11), Math.abs(b11)) && Math.abs(a12 - b12) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a12), Math.abs(b12)) && Math.abs(a13 - b13) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a13), Math.abs(b13)) && Math.abs(a14 - b14) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a14), Math.abs(b14)) && Math.abs(a15 - b15) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a15), Math.abs(b15));\n}\n/**\n * Alias for {@link mat4.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link mat4.subtract}\n * @function\n */\n\nexport var sub = subtract;","import {StyleLayer} from '../style_layer';\n\nimport {CircleBucket} from '../../data/bucket/circle_bucket';\nimport {polygonIntersectsBufferedPoint} from '../../util/intersection_tests';\nimport {getMaximumPaintValue, translateDistance, translate} from '../query_utils';\nimport properties, {CircleLayoutPropsPossiblyEvaluated, CirclePaintPropsPossiblyEvaluated} from './circle_style_layer_properties.g';\nimport {Transitionable, Transitioning, Layout, PossiblyEvaluated} from '../properties';\nimport {mat4, vec4} from 'gl-matrix';\nimport Point from '@mapbox/point-geometry';\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Transform} from '../../geo/transform';\nimport type {Bucket, BucketParameters} from '../../data/bucket';\nimport type {CircleLayoutProps, CirclePaintProps} from './circle_style_layer_properties.g';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\n/**\n * A style layer that defines a circle\n */\nexport class CircleStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n createBucket(parameters: BucketParameters) {\n return new CircleBucket(parameters);\n }\n\n queryRadius(bucket: Bucket): number {\n const circleBucket: CircleBucket = (bucket as any);\n return getMaximumPaintValue('circle-radius', this, circleBucket) +\n getMaximumPaintValue('circle-stroke-width', this, circleBucket) +\n translateDistance(this.paint.get('circle-translate'));\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number,\n pixelPosMatrix: mat4\n ): boolean {\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('circle-translate'),\n this.paint.get('circle-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n const radius = this.paint.get('circle-radius').evaluate(feature, featureState);\n const stroke = this.paint.get('circle-stroke-width').evaluate(feature, featureState);\n const size = radius + stroke;\n\n // For pitch-alignment: map, compare feature geometry to query geometry in the plane of the tile\n // // Otherwise, compare geometry in the plane of the viewport\n // // A circle with fixed scaling relative to the viewport gets larger in tile space as it moves into the distance\n // // A circle with fixed scaling relative to the map gets smaller in viewport space as it moves into the distance\n const alignWithMap = this.paint.get('circle-pitch-alignment') === 'map';\n const transformedPolygon = alignWithMap ? translatedPolygon : projectQueryGeometry(translatedPolygon, pixelPosMatrix);\n const transformedSize = alignWithMap ? size * pixelsToTileUnits : size;\n\n for (const ring of geometry) {\n for (const point of ring) {\n\n const transformedPoint = alignWithMap ? point : projectPoint(point, pixelPosMatrix);\n\n let adjustedSize = transformedSize;\n const projectedCenter = vec4.transformMat4([] as any, [point.x, point.y, 0, 1], pixelPosMatrix);\n if (this.paint.get('circle-pitch-scale') === 'viewport' && this.paint.get('circle-pitch-alignment') === 'map') {\n adjustedSize *= projectedCenter[3] / transform.cameraToCenterDistance;\n } else if (this.paint.get('circle-pitch-scale') === 'map' && this.paint.get('circle-pitch-alignment') === 'viewport') {\n adjustedSize *= transform.cameraToCenterDistance / projectedCenter[3];\n }\n\n if (polygonIntersectsBufferedPoint(transformedPolygon, transformedPoint, adjustedSize)) return true;\n }\n }\n\n return false;\n }\n}\n\nfunction projectPoint(p: Point, pixelPosMatrix: mat4) {\n const point = vec4.transformMat4([] as any, [p.x, p.y, 0, 1], pixelPosMatrix);\n return new Point(point[0] / point[3], point[1] / point[3]);\n}\n\nfunction projectQueryGeometry(queryGeometry: Array, pixelPosMatrix: mat4) {\n return queryGeometry.map((p) => {\n return projectPoint(p, pixelPosMatrix);\n });\n}\n","import {CircleBucket} from './circle_bucket';\nimport {register} from '../../util/web_worker_transfer';\n\nimport type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer';\n\nexport class HeatmapBucket extends CircleBucket {\n // Needed for flow to accept omit: ['layers'] below, due to\n // https://github.com/facebook/flow/issues/4262\n layers: Array;\n}\n\nregister('HeatmapBucket', HeatmapBucket, {omit: ['layers']});\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type HeatmapPaintProps = {\n \"heatmap-radius\": DataDrivenProperty,\n \"heatmap-weight\": DataDrivenProperty,\n \"heatmap-intensity\": DataConstantProperty,\n \"heatmap-color\": ColorRampProperty,\n \"heatmap-opacity\": DataConstantProperty,\n};\n\nexport type HeatmapPaintPropsPossiblyEvaluated = {\n \"heatmap-radius\": PossiblyEvaluatedPropertyValue,\n \"heatmap-weight\": PossiblyEvaluatedPropertyValue,\n \"heatmap-intensity\": number,\n \"heatmap-color\": ColorRampProperty,\n \"heatmap-opacity\": number,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"heatmap-radius\": new DataDrivenProperty(styleSpec[\"paint_heatmap\"][\"heatmap-radius\"] as any as StylePropertySpecification),\n \"heatmap-weight\": new DataDrivenProperty(styleSpec[\"paint_heatmap\"][\"heatmap-weight\"] as any as StylePropertySpecification),\n \"heatmap-intensity\": new DataConstantProperty(styleSpec[\"paint_heatmap\"][\"heatmap-intensity\"] as any as StylePropertySpecification),\n \"heatmap-color\": new ColorRampProperty(styleSpec[\"paint_heatmap\"][\"heatmap-color\"] as any as StylePropertySpecification),\n \"heatmap-opacity\": new DataConstantProperty(styleSpec[\"paint_heatmap\"][\"heatmap-opacity\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {register} from './web_worker_transfer';\n\nexport type Size = {\n width: number;\n height: number;\n};\n\ntype Point2D = {\n x: number;\n y: number;\n};\n\nfunction createImage(image: any, {\n width,\n height\n}: Size, channels: number, data?: Uint8Array | Uint8ClampedArray) {\n if (!data) {\n data = new Uint8Array(width * height * channels);\n } else if (data instanceof Uint8ClampedArray) {\n data = new Uint8Array(data.buffer);\n } else if (data.length !== width * height * channels) {\n throw new RangeError(`mismatched image size. expected: ${data.length} but got: ${width * height * channels}`);\n }\n image.width = width;\n image.height = height;\n image.data = data;\n return image;\n}\n\nfunction resizeImage(image: any, {\n width,\n height\n}: Size, channels: number) {\n if (width === image.width && height === image.height) {\n return;\n }\n\n const newImage = createImage({}, {width, height}, channels);\n\n copyImage(image, newImage, {x: 0, y: 0}, {x: 0, y: 0}, {\n width: Math.min(image.width, width),\n height: Math.min(image.height, height)\n }, channels);\n\n image.width = width;\n image.height = height;\n image.data = newImage.data;\n}\n\nfunction copyImage(srcImg: any, dstImg: any, srcPt: Point2D, dstPt: Point2D, size: Size, channels: number) {\n if (size.width === 0 || size.height === 0) {\n return dstImg;\n }\n\n if (size.width > srcImg.width ||\n size.height > srcImg.height ||\n srcPt.x > srcImg.width - size.width ||\n srcPt.y > srcImg.height - size.height) {\n throw new RangeError('out of range source coordinates for image copy');\n }\n\n if (size.width > dstImg.width ||\n size.height > dstImg.height ||\n dstPt.x > dstImg.width - size.width ||\n dstPt.y > dstImg.height - size.height) {\n throw new RangeError('out of range destination coordinates for image copy');\n }\n\n const srcData = srcImg.data;\n const dstData = dstImg.data;\n\n if (srcData === dstData) throw new Error('srcData equals dstData, so image is already copied');\n\n for (let y = 0; y < size.height; y++) {\n const srcOffset = ((srcPt.y + y) * srcImg.width + srcPt.x) * channels;\n const dstOffset = ((dstPt.y + y) * dstImg.width + dstPt.x) * channels;\n for (let i = 0; i < size.width * channels; i++) {\n dstData[dstOffset + i] = srcData[srcOffset + i];\n }\n }\n return dstImg;\n}\n\n/**\n * @internal\n * An image with alpha color value\n */\nexport class AlphaImage {\n width: number;\n height: number;\n data: Uint8Array;\n\n constructor(size: Size, data?: Uint8Array | Uint8ClampedArray) {\n createImage(this, size, 1, data);\n }\n\n resize(size: Size) {\n resizeImage(this, size, 1);\n }\n\n clone() {\n return new AlphaImage({width: this.width, height: this.height}, new Uint8Array(this.data));\n }\n\n static copy(srcImg: AlphaImage, dstImg: AlphaImage, srcPt: Point2D, dstPt: Point2D, size: Size) {\n copyImage(srcImg, dstImg, srcPt, dstPt, size, 1);\n }\n}\n\n/**\n * An object to store image data not premultiplied, because ImageData is not premultiplied.\n * UNPACK_PREMULTIPLY_ALPHA_WEBGL must be used when uploading to a texture.\n */\nexport class RGBAImage {\n width: number;\n height: number;\n\n /**\n * data must be a Uint8Array instead of Uint8ClampedArray because texImage2D does not support Uint8ClampedArray in all browsers.\n */\n data: Uint8Array;\n\n constructor(size: Size, data?: Uint8Array | Uint8ClampedArray) {\n createImage(this, size, 4, data);\n }\n\n resize(size: Size) {\n resizeImage(this, size, 4);\n }\n\n replace(data: Uint8Array | Uint8ClampedArray, copy?: boolean) {\n if (copy) {\n this.data.set(data);\n } else if (data instanceof Uint8ClampedArray) {\n this.data = new Uint8Array(data.buffer);\n } else {\n this.data = data;\n }\n }\n\n clone() {\n return new RGBAImage({width: this.width, height: this.height}, new Uint8Array(this.data));\n }\n\n static copy(srcImg: RGBAImage | ImageData, dstImg: RGBAImage, srcPt: Point2D, dstPt: Point2D, size: Size) {\n copyImage(srcImg, dstImg, srcPt, dstPt, size, 4);\n }\n}\n\nregister('AlphaImage', AlphaImage);\nregister('RGBAImage', RGBAImage);\n","import {StyleLayer} from '../style_layer';\n\nimport {HeatmapBucket} from '../../data/bucket/heatmap_bucket';\nimport {RGBAImage} from '../../util/image';\nimport properties, {HeatmapPaintPropsPossiblyEvaluated} from './heatmap_style_layer_properties.g';\nimport {renderColorRamp} from '../../util/color_ramp';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {Texture} from '../../render/texture';\nimport type {Framebuffer} from '../../gl/framebuffer';\nimport type {HeatmapPaintProps} from './heatmap_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A style layer that defines a heatmap\n */\nexport class HeatmapStyleLayer extends StyleLayer {\n\n heatmapFbo: Framebuffer;\n colorRamp: RGBAImage;\n colorRampTexture: Texture;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n createBucket(options: any) {\n return new HeatmapBucket(options);\n }\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n\n // make sure color ramp texture is generated for default heatmap color too\n this._updateColorRamp();\n }\n\n _handleSpecialPaintPropertyUpdate(name: string) {\n if (name === 'heatmap-color') {\n this._updateColorRamp();\n }\n }\n\n _updateColorRamp() {\n const expression = this._transitionablePaint._values['heatmap-color'].value.expression;\n this.colorRamp = renderColorRamp({\n expression,\n evaluationKey: 'heatmapDensity',\n image: this.colorRamp\n });\n this.colorRampTexture = null;\n }\n\n resize() {\n if (this.heatmapFbo) {\n this.heatmapFbo.destroy();\n this.heatmapFbo = null;\n }\n }\n\n queryRadius(): number {\n return 0;\n }\n\n queryIntersectsFeature(): boolean {\n return false;\n }\n\n hasOffscreenPass() {\n return this.paint.get('heatmap-opacity') !== 0 && this.visibility !== 'none';\n }\n}\n","import {RGBAImage} from './image';\nimport {isPowerOfTwo} from './util';\n\nimport type {StylePropertyExpression} from '@maplibre/maplibre-gl-style-spec';\n\nexport type ColorRampParams = {\n expression: StylePropertyExpression;\n evaluationKey: string;\n resolution?: number;\n image?: RGBAImage;\n clips?: Array;\n};\n\n/**\n * Given an expression that should evaluate to a color ramp,\n * return a RGBA image representing that ramp expression.\n */\nexport function renderColorRamp(params: ColorRampParams): RGBAImage {\n const evaluationGlobals = {};\n const width = params.resolution || 256;\n const height = params.clips ? params.clips.length : 1;\n const image = params.image || new RGBAImage({width, height});\n\n if (!isPowerOfTwo(width)) throw new Error(`width is not a power of 2 - ${width}`);\n\n const renderPixel = (stride, index, progress) => {\n evaluationGlobals[params.evaluationKey] = progress;\n const pxColor = params.expression.evaluate(evaluationGlobals as any);\n // the colors are being unpremultiplied because Color uses\n // premultiplied values, and the Texture class expects unpremultiplied ones\n image.data[stride + index + 0] = Math.floor(pxColor.r * 255 / pxColor.a);\n image.data[stride + index + 1] = Math.floor(pxColor.g * 255 / pxColor.a);\n image.data[stride + index + 2] = Math.floor(pxColor.b * 255 / pxColor.a);\n image.data[stride + index + 3] = Math.floor(pxColor.a * 255);\n };\n\n if (!params.clips) {\n for (let i = 0, j = 0; i < width; i++, j += 4) {\n const progress = i / (width - 1);\n\n renderPixel(0, j, progress);\n }\n } else {\n for (let clip = 0, stride = 0; clip < height; ++clip, stride += width * 4) {\n for (let i = 0, j = 0; i < width; i++, j += 4) {\n // Remap progress between clips\n const progress = i / (width - 1);\n const {start, end} = params.clips[clip];\n const evaluationProgress = start * (1 - progress) + end * progress;\n renderPixel(stride, j, evaluationProgress);\n }\n }\n }\n\n return image;\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type HillshadePaintProps = {\n \"hillshade-illumination-direction\": DataConstantProperty,\n \"hillshade-illumination-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"hillshade-exaggeration\": DataConstantProperty,\n \"hillshade-shadow-color\": DataConstantProperty,\n \"hillshade-highlight-color\": DataConstantProperty,\n \"hillshade-accent-color\": DataConstantProperty,\n};\n\nexport type HillshadePaintPropsPossiblyEvaluated = {\n \"hillshade-illumination-direction\": number,\n \"hillshade-illumination-anchor\": \"map\" | \"viewport\",\n \"hillshade-exaggeration\": number,\n \"hillshade-shadow-color\": Color,\n \"hillshade-highlight-color\": Color,\n \"hillshade-accent-color\": Color,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"hillshade-illumination-direction\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-illumination-direction\"] as any as StylePropertySpecification),\n \"hillshade-illumination-anchor\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-illumination-anchor\"] as any as StylePropertySpecification),\n \"hillshade-exaggeration\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-exaggeration\"] as any as StylePropertySpecification),\n \"hillshade-shadow-color\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-shadow-color\"] as any as StylePropertySpecification),\n \"hillshade-highlight-color\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-highlight-color\"] as any as StylePropertySpecification),\n \"hillshade-accent-color\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-accent-color\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport properties, {HillshadePaintPropsPossiblyEvaluated} from './hillshade_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {HillshadePaintProps} from './hillshade_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport class HillshadeStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n hasOffscreenPass() {\n return this.paint.get('hillshade-exaggeration') !== 0 && this.visibility !== 'none';\n }\n}\n","import {createLayout} from '../../util/struct_array';\n\nconst layout = createLayout([\n {name: 'a_pos', components: 2, type: 'Int16'}\n], 4);\n\nexport default layout;\nexport const {members, size, alignment} = layout;\n","'use strict';\n\nmodule.exports = earcut;\nmodule.exports.default = earcut;\n\nfunction earcut(data, holeIndices, dim) {\n\n dim = dim || 2;\n\n var hasHoles = holeIndices && holeIndices.length,\n outerLen = hasHoles ? holeIndices[0] * dim : data.length,\n outerNode = linkedList(data, 0, outerLen, dim, true),\n triangles = [];\n\n if (!outerNode || outerNode.next === outerNode.prev) return triangles;\n\n var minX, minY, maxX, maxY, x, y, invSize;\n\n if (hasHoles) outerNode = eliminateHoles(data, holeIndices, outerNode, dim);\n\n // if the shape is not too simple, we'll use z-order curve hash later; calculate polygon bbox\n if (data.length > 80 * dim) {\n minX = maxX = data[0];\n minY = maxY = data[1];\n\n for (var i = dim; i < outerLen; i += dim) {\n x = data[i];\n y = data[i + 1];\n if (x < minX) minX = x;\n if (y < minY) minY = y;\n if (x > maxX) maxX = x;\n if (y > maxY) maxY = y;\n }\n\n // minX, minY and invSize are later used to transform coords into integers for z-order calculation\n invSize = Math.max(maxX - minX, maxY - minY);\n invSize = invSize !== 0 ? 32767 / invSize : 0;\n }\n\n earcutLinked(outerNode, triangles, dim, minX, minY, invSize, 0);\n\n return triangles;\n}\n\n// create a circular doubly linked list from polygon points in the specified winding order\nfunction linkedList(data, start, end, dim, clockwise) {\n var i, last;\n\n if (clockwise === (signedArea(data, start, end, dim) > 0)) {\n for (i = start; i < end; i += dim) last = insertNode(i, data[i], data[i + 1], last);\n } else {\n for (i = end - dim; i >= start; i -= dim) last = insertNode(i, data[i], data[i + 1], last);\n }\n\n if (last && equals(last, last.next)) {\n removeNode(last);\n last = last.next;\n }\n\n return last;\n}\n\n// eliminate colinear or duplicate points\nfunction filterPoints(start, end) {\n if (!start) return start;\n if (!end) end = start;\n\n var p = start,\n again;\n do {\n again = false;\n\n if (!p.steiner && (equals(p, p.next) || area(p.prev, p, p.next) === 0)) {\n removeNode(p);\n p = end = p.prev;\n if (p === p.next) break;\n again = true;\n\n } else {\n p = p.next;\n }\n } while (again || p !== end);\n\n return end;\n}\n\n// main ear slicing loop which triangulates a polygon (given as a linked list)\nfunction earcutLinked(ear, triangles, dim, minX, minY, invSize, pass) {\n if (!ear) return;\n\n // interlink polygon nodes in z-order\n if (!pass && invSize) indexCurve(ear, minX, minY, invSize);\n\n var stop = ear,\n prev, next;\n\n // iterate through ears, slicing them one by one\n while (ear.prev !== ear.next) {\n prev = ear.prev;\n next = ear.next;\n\n if (invSize ? isEarHashed(ear, minX, minY, invSize) : isEar(ear)) {\n // cut off the triangle\n triangles.push(prev.i / dim | 0);\n triangles.push(ear.i / dim | 0);\n triangles.push(next.i / dim | 0);\n\n removeNode(ear);\n\n // skipping the next vertex leads to less sliver triangles\n ear = next.next;\n stop = next.next;\n\n continue;\n }\n\n ear = next;\n\n // if we looped through the whole remaining polygon and can't find any more ears\n if (ear === stop) {\n // try filtering points and slicing again\n if (!pass) {\n earcutLinked(filterPoints(ear), triangles, dim, minX, minY, invSize, 1);\n\n // if this didn't work, try curing all small self-intersections locally\n } else if (pass === 1) {\n ear = cureLocalIntersections(filterPoints(ear), triangles, dim);\n earcutLinked(ear, triangles, dim, minX, minY, invSize, 2);\n\n // as a last resort, try splitting the remaining polygon into two\n } else if (pass === 2) {\n splitEarcut(ear, triangles, dim, minX, minY, invSize);\n }\n\n break;\n }\n }\n}\n\n// check whether a polygon node forms a valid ear with adjacent nodes\nfunction isEar(ear) {\n var a = ear.prev,\n b = ear,\n c = ear.next;\n\n if (area(a, b, c) >= 0) return false; // reflex, can't be an ear\n\n // now make sure we don't have other points inside the potential ear\n var ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y;\n\n // triangle bbox; min & max are calculated like this for speed\n var x0 = ax < bx ? (ax < cx ? ax : cx) : (bx < cx ? bx : cx),\n y0 = ay < by ? (ay < cy ? ay : cy) : (by < cy ? by : cy),\n x1 = ax > bx ? (ax > cx ? ax : cx) : (bx > cx ? bx : cx),\n y1 = ay > by ? (ay > cy ? ay : cy) : (by > cy ? by : cy);\n\n var p = c.next;\n while (p !== a) {\n if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) &&\n area(p.prev, p, p.next) >= 0) return false;\n p = p.next;\n }\n\n return true;\n}\n\nfunction isEarHashed(ear, minX, minY, invSize) {\n var a = ear.prev,\n b = ear,\n c = ear.next;\n\n if (area(a, b, c) >= 0) return false; // reflex, can't be an ear\n\n var ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y;\n\n // triangle bbox; min & max are calculated like this for speed\n var x0 = ax < bx ? (ax < cx ? ax : cx) : (bx < cx ? bx : cx),\n y0 = ay < by ? (ay < cy ? ay : cy) : (by < cy ? by : cy),\n x1 = ax > bx ? (ax > cx ? ax : cx) : (bx > cx ? bx : cx),\n y1 = ay > by ? (ay > cy ? ay : cy) : (by > cy ? by : cy);\n\n // z-order range for the current triangle bbox;\n var minZ = zOrder(x0, y0, minX, minY, invSize),\n maxZ = zOrder(x1, y1, minX, minY, invSize);\n\n var p = ear.prevZ,\n n = ear.nextZ;\n\n // look for points inside the triangle in both directions\n while (p && p.z >= minZ && n && n.z <= maxZ) {\n if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 && p !== a && p !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) && area(p.prev, p, p.next) >= 0) return false;\n p = p.prevZ;\n\n if (n.x >= x0 && n.x <= x1 && n.y >= y0 && n.y <= y1 && n !== a && n !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) && area(n.prev, n, n.next) >= 0) return false;\n n = n.nextZ;\n }\n\n // look for remaining points in decreasing z-order\n while (p && p.z >= minZ) {\n if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 && p !== a && p !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) && area(p.prev, p, p.next) >= 0) return false;\n p = p.prevZ;\n }\n\n // look for remaining points in increasing z-order\n while (n && n.z <= maxZ) {\n if (n.x >= x0 && n.x <= x1 && n.y >= y0 && n.y <= y1 && n !== a && n !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) && area(n.prev, n, n.next) >= 0) return false;\n n = n.nextZ;\n }\n\n return true;\n}\n\n// go through all polygon nodes and cure small local self-intersections\nfunction cureLocalIntersections(start, triangles, dim) {\n var p = start;\n do {\n var a = p.prev,\n b = p.next.next;\n\n if (!equals(a, b) && intersects(a, p, p.next, b) && locallyInside(a, b) && locallyInside(b, a)) {\n\n triangles.push(a.i / dim | 0);\n triangles.push(p.i / dim | 0);\n triangles.push(b.i / dim | 0);\n\n // remove two nodes involved\n removeNode(p);\n removeNode(p.next);\n\n p = start = b;\n }\n p = p.next;\n } while (p !== start);\n\n return filterPoints(p);\n}\n\n// try splitting polygon into two and triangulate them independently\nfunction splitEarcut(start, triangles, dim, minX, minY, invSize) {\n // look for a valid diagonal that divides the polygon into two\n var a = start;\n do {\n var b = a.next.next;\n while (b !== a.prev) {\n if (a.i !== b.i && isValidDiagonal(a, b)) {\n // split the polygon in two by the diagonal\n var c = splitPolygon(a, b);\n\n // filter colinear points around the cuts\n a = filterPoints(a, a.next);\n c = filterPoints(c, c.next);\n\n // run earcut on each half\n earcutLinked(a, triangles, dim, minX, minY, invSize, 0);\n earcutLinked(c, triangles, dim, minX, minY, invSize, 0);\n return;\n }\n b = b.next;\n }\n a = a.next;\n } while (a !== start);\n}\n\n// link every hole into the outer loop, producing a single-ring polygon without holes\nfunction eliminateHoles(data, holeIndices, outerNode, dim) {\n var queue = [],\n i, len, start, end, list;\n\n for (i = 0, len = holeIndices.length; i < len; i++) {\n start = holeIndices[i] * dim;\n end = i < len - 1 ? holeIndices[i + 1] * dim : data.length;\n list = linkedList(data, start, end, dim, false);\n if (list === list.next) list.steiner = true;\n queue.push(getLeftmost(list));\n }\n\n queue.sort(compareX);\n\n // process holes from left to right\n for (i = 0; i < queue.length; i++) {\n outerNode = eliminateHole(queue[i], outerNode);\n }\n\n return outerNode;\n}\n\nfunction compareX(a, b) {\n return a.x - b.x;\n}\n\n// find a bridge between vertices that connects hole with an outer ring and and link it\nfunction eliminateHole(hole, outerNode) {\n var bridge = findHoleBridge(hole, outerNode);\n if (!bridge) {\n return outerNode;\n }\n\n var bridgeReverse = splitPolygon(bridge, hole);\n\n // filter collinear points around the cuts\n filterPoints(bridgeReverse, bridgeReverse.next);\n return filterPoints(bridge, bridge.next);\n}\n\n// David Eberly's algorithm for finding a bridge between hole and outer polygon\nfunction findHoleBridge(hole, outerNode) {\n var p = outerNode,\n hx = hole.x,\n hy = hole.y,\n qx = -Infinity,\n m;\n\n // find a segment intersected by a ray from the hole's leftmost point to the left;\n // segment's endpoint with lesser x will be potential connection point\n do {\n if (hy <= p.y && hy >= p.next.y && p.next.y !== p.y) {\n var x = p.x + (hy - p.y) * (p.next.x - p.x) / (p.next.y - p.y);\n if (x <= hx && x > qx) {\n qx = x;\n m = p.x < p.next.x ? p : p.next;\n if (x === hx) return m; // hole touches outer segment; pick leftmost endpoint\n }\n }\n p = p.next;\n } while (p !== outerNode);\n\n if (!m) return null;\n\n // look for points inside the triangle of hole point, segment intersection and endpoint;\n // if there are no points found, we have a valid connection;\n // otherwise choose the point of the minimum angle with the ray as connection point\n\n var stop = m,\n mx = m.x,\n my = m.y,\n tanMin = Infinity,\n tan;\n\n p = m;\n\n do {\n if (hx >= p.x && p.x >= mx && hx !== p.x &&\n pointInTriangle(hy < my ? hx : qx, hy, mx, my, hy < my ? qx : hx, hy, p.x, p.y)) {\n\n tan = Math.abs(hy - p.y) / (hx - p.x); // tangential\n\n if (locallyInside(p, hole) &&\n (tan < tanMin || (tan === tanMin && (p.x > m.x || (p.x === m.x && sectorContainsSector(m, p)))))) {\n m = p;\n tanMin = tan;\n }\n }\n\n p = p.next;\n } while (p !== stop);\n\n return m;\n}\n\n// whether sector in vertex m contains sector in vertex p in the same coordinates\nfunction sectorContainsSector(m, p) {\n return area(m.prev, m, p.prev) < 0 && area(p.next, m, m.next) < 0;\n}\n\n// interlink polygon nodes in z-order\nfunction indexCurve(start, minX, minY, invSize) {\n var p = start;\n do {\n if (p.z === 0) p.z = zOrder(p.x, p.y, minX, minY, invSize);\n p.prevZ = p.prev;\n p.nextZ = p.next;\n p = p.next;\n } while (p !== start);\n\n p.prevZ.nextZ = null;\n p.prevZ = null;\n\n sortLinked(p);\n}\n\n// Simon Tatham's linked list merge sort algorithm\n// http://www.chiark.greenend.org.uk/~sgtatham/algorithms/listsort.html\nfunction sortLinked(list) {\n var i, p, q, e, tail, numMerges, pSize, qSize,\n inSize = 1;\n\n do {\n p = list;\n list = null;\n tail = null;\n numMerges = 0;\n\n while (p) {\n numMerges++;\n q = p;\n pSize = 0;\n for (i = 0; i < inSize; i++) {\n pSize++;\n q = q.nextZ;\n if (!q) break;\n }\n qSize = inSize;\n\n while (pSize > 0 || (qSize > 0 && q)) {\n\n if (pSize !== 0 && (qSize === 0 || !q || p.z <= q.z)) {\n e = p;\n p = p.nextZ;\n pSize--;\n } else {\n e = q;\n q = q.nextZ;\n qSize--;\n }\n\n if (tail) tail.nextZ = e;\n else list = e;\n\n e.prevZ = tail;\n tail = e;\n }\n\n p = q;\n }\n\n tail.nextZ = null;\n inSize *= 2;\n\n } while (numMerges > 1);\n\n return list;\n}\n\n// z-order of a point given coords and inverse of the longer side of data bbox\nfunction zOrder(x, y, minX, minY, invSize) {\n // coords are transformed into non-negative 15-bit integer range\n x = (x - minX) * invSize | 0;\n y = (y - minY) * invSize | 0;\n\n x = (x | (x << 8)) & 0x00FF00FF;\n x = (x | (x << 4)) & 0x0F0F0F0F;\n x = (x | (x << 2)) & 0x33333333;\n x = (x | (x << 1)) & 0x55555555;\n\n y = (y | (y << 8)) & 0x00FF00FF;\n y = (y | (y << 4)) & 0x0F0F0F0F;\n y = (y | (y << 2)) & 0x33333333;\n y = (y | (y << 1)) & 0x55555555;\n\n return x | (y << 1);\n}\n\n// find the leftmost node of a polygon ring\nfunction getLeftmost(start) {\n var p = start,\n leftmost = start;\n do {\n if (p.x < leftmost.x || (p.x === leftmost.x && p.y < leftmost.y)) leftmost = p;\n p = p.next;\n } while (p !== start);\n\n return leftmost;\n}\n\n// check if a point lies within a convex triangle\nfunction pointInTriangle(ax, ay, bx, by, cx, cy, px, py) {\n return (cx - px) * (ay - py) >= (ax - px) * (cy - py) &&\n (ax - px) * (by - py) >= (bx - px) * (ay - py) &&\n (bx - px) * (cy - py) >= (cx - px) * (by - py);\n}\n\n// check if a diagonal between two polygon nodes is valid (lies in polygon interior)\nfunction isValidDiagonal(a, b) {\n return a.next.i !== b.i && a.prev.i !== b.i && !intersectsPolygon(a, b) && // dones't intersect other edges\n (locallyInside(a, b) && locallyInside(b, a) && middleInside(a, b) && // locally visible\n (area(a.prev, a, b.prev) || area(a, b.prev, b)) || // does not create opposite-facing sectors\n equals(a, b) && area(a.prev, a, a.next) > 0 && area(b.prev, b, b.next) > 0); // special zero-length case\n}\n\n// signed area of a triangle\nfunction area(p, q, r) {\n return (q.y - p.y) * (r.x - q.x) - (q.x - p.x) * (r.y - q.y);\n}\n\n// check if two points are equal\nfunction equals(p1, p2) {\n return p1.x === p2.x && p1.y === p2.y;\n}\n\n// check if two segments intersect\nfunction intersects(p1, q1, p2, q2) {\n var o1 = sign(area(p1, q1, p2));\n var o2 = sign(area(p1, q1, q2));\n var o3 = sign(area(p2, q2, p1));\n var o4 = sign(area(p2, q2, q1));\n\n if (o1 !== o2 && o3 !== o4) return true; // general case\n\n if (o1 === 0 && onSegment(p1, p2, q1)) return true; // p1, q1 and p2 are collinear and p2 lies on p1q1\n if (o2 === 0 && onSegment(p1, q2, q1)) return true; // p1, q1 and q2 are collinear and q2 lies on p1q1\n if (o3 === 0 && onSegment(p2, p1, q2)) return true; // p2, q2 and p1 are collinear and p1 lies on p2q2\n if (o4 === 0 && onSegment(p2, q1, q2)) return true; // p2, q2 and q1 are collinear and q1 lies on p2q2\n\n return false;\n}\n\n// for collinear points p, q, r, check if point q lies on segment pr\nfunction onSegment(p, q, r) {\n return q.x <= Math.max(p.x, r.x) && q.x >= Math.min(p.x, r.x) && q.y <= Math.max(p.y, r.y) && q.y >= Math.min(p.y, r.y);\n}\n\nfunction sign(num) {\n return num > 0 ? 1 : num < 0 ? -1 : 0;\n}\n\n// check if a polygon diagonal intersects any polygon segments\nfunction intersectsPolygon(a, b) {\n var p = a;\n do {\n if (p.i !== a.i && p.next.i !== a.i && p.i !== b.i && p.next.i !== b.i &&\n intersects(p, p.next, a, b)) return true;\n p = p.next;\n } while (p !== a);\n\n return false;\n}\n\n// check if a polygon diagonal is locally inside the polygon\nfunction locallyInside(a, b) {\n return area(a.prev, a, a.next) < 0 ?\n area(a, b, a.next) >= 0 && area(a, a.prev, b) >= 0 :\n area(a, b, a.prev) < 0 || area(a, a.next, b) < 0;\n}\n\n// check if the middle point of a polygon diagonal is inside the polygon\nfunction middleInside(a, b) {\n var p = a,\n inside = false,\n px = (a.x + b.x) / 2,\n py = (a.y + b.y) / 2;\n do {\n if (((p.y > py) !== (p.next.y > py)) && p.next.y !== p.y &&\n (px < (p.next.x - p.x) * (py - p.y) / (p.next.y - p.y) + p.x))\n inside = !inside;\n p = p.next;\n } while (p !== a);\n\n return inside;\n}\n\n// link two polygon vertices with a bridge; if the vertices belong to the same ring, it splits polygon into two;\n// if one belongs to the outer ring and another to a hole, it merges it into a single ring\nfunction splitPolygon(a, b) {\n var a2 = new Node(a.i, a.x, a.y),\n b2 = new Node(b.i, b.x, b.y),\n an = a.next,\n bp = b.prev;\n\n a.next = b;\n b.prev = a;\n\n a2.next = an;\n an.prev = a2;\n\n b2.next = a2;\n a2.prev = b2;\n\n bp.next = b2;\n b2.prev = bp;\n\n return b2;\n}\n\n// create a node and optionally link it with previous one (in a circular doubly linked list)\nfunction insertNode(i, x, y, last) {\n var p = new Node(i, x, y);\n\n if (!last) {\n p.prev = p;\n p.next = p;\n\n } else {\n p.next = last.next;\n p.prev = last;\n last.next.prev = p;\n last.next = p;\n }\n return p;\n}\n\nfunction removeNode(p) {\n p.next.prev = p.prev;\n p.prev.next = p.next;\n\n if (p.prevZ) p.prevZ.nextZ = p.nextZ;\n if (p.nextZ) p.nextZ.prevZ = p.prevZ;\n}\n\nfunction Node(i, x, y) {\n // vertex index in coordinates array\n this.i = i;\n\n // vertex coordinates\n this.x = x;\n this.y = y;\n\n // previous and next vertex nodes in a polygon ring\n this.prev = null;\n this.next = null;\n\n // z-order curve value\n this.z = 0;\n\n // previous and next nodes in z-order\n this.prevZ = null;\n this.nextZ = null;\n\n // indicates whether this is a steiner point\n this.steiner = false;\n}\n\n// return a percentage difference between the polygon area and its triangulation area;\n// used to verify correctness of triangulation\nearcut.deviation = function (data, holeIndices, dim, triangles) {\n var hasHoles = holeIndices && holeIndices.length;\n var outerLen = hasHoles ? holeIndices[0] * dim : data.length;\n\n var polygonArea = Math.abs(signedArea(data, 0, outerLen, dim));\n if (hasHoles) {\n for (var i = 0, len = holeIndices.length; i < len; i++) {\n var start = holeIndices[i] * dim;\n var end = i < len - 1 ? holeIndices[i + 1] * dim : data.length;\n polygonArea -= Math.abs(signedArea(data, start, end, dim));\n }\n }\n\n var trianglesArea = 0;\n for (i = 0; i < triangles.length; i += 3) {\n var a = triangles[i] * dim;\n var b = triangles[i + 1] * dim;\n var c = triangles[i + 2] * dim;\n trianglesArea += Math.abs(\n (data[a] - data[c]) * (data[b + 1] - data[a + 1]) -\n (data[a] - data[b]) * (data[c + 1] - data[a + 1]));\n }\n\n return polygonArea === 0 && trianglesArea === 0 ? 0 :\n Math.abs((trianglesArea - polygonArea) / polygonArea);\n};\n\nfunction signedArea(data, start, end, dim) {\n var sum = 0;\n for (var i = start, j = end - dim; i < end; i += dim) {\n sum += (data[j] - data[i]) * (data[i + 1] + data[j + 1]);\n j = i;\n }\n return sum;\n}\n\n// turn a polygon in a multi-dimensional array form (e.g. as in GeoJSON) into a form Earcut accepts\nearcut.flatten = function (data) {\n var dim = data[0][0].length,\n result = {vertices: [], holes: [], dimensions: dim},\n holeIndex = 0;\n\n for (var i = 0; i < data.length; i++) {\n for (var j = 0; j < data[i].length; j++) {\n for (var d = 0; d < dim; d++) result.vertices.push(data[i][j][d]);\n }\n if (i > 0) {\n holeIndex += data[i - 1].length;\n result.holes.push(holeIndex);\n }\n }\n return result;\n};\n","\nexport default function quickselect(arr, k, left, right, compare) {\n quickselectStep(arr, k, left || 0, right || (arr.length - 1), compare || defaultCompare);\n}\n\nfunction quickselectStep(arr, k, left, right, compare) {\n\n while (right > left) {\n if (right - left > 600) {\n var n = right - left + 1;\n var m = k - left + 1;\n var z = Math.log(n);\n var s = 0.5 * Math.exp(2 * z / 3);\n var sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);\n var newLeft = Math.max(left, Math.floor(k - m * s / n + sd));\n var newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));\n quickselectStep(arr, k, newLeft, newRight, compare);\n }\n\n var t = arr[k];\n var i = left;\n var j = right;\n\n swap(arr, left, k);\n if (compare(arr[right], t) > 0) swap(arr, left, right);\n\n while (i < j) {\n swap(arr, i, j);\n i++;\n j--;\n while (compare(arr[i], t) < 0) i++;\n while (compare(arr[j], t) > 0) j--;\n }\n\n if (compare(arr[left], t) === 0) swap(arr, left, j);\n else {\n j++;\n swap(arr, j, right);\n }\n\n if (j <= k) left = j + 1;\n if (k <= j) right = j - 1;\n }\n}\n\nfunction swap(arr, i, j) {\n var tmp = arr[i];\n arr[i] = arr[j];\n arr[j] = tmp;\n}\n\nfunction defaultCompare(a, b) {\n return a < b ? -1 : a > b ? 1 : 0;\n}\n","import quickselect from 'quickselect';\n\nimport {calculateSignedArea} from './util';\n\nimport type Point from '@mapbox/point-geometry';\n\n// classifies an array of rings into polygons with outer rings and holes\nexport function classifyRings(rings: Array>, maxRings: number) {\n const len = rings.length;\n\n if (len <= 1) return [rings];\n\n const polygons = [];\n let polygon,\n ccw;\n\n for (let i = 0; i < len; i++) {\n const area = calculateSignedArea(rings[i]);\n if (area === 0) continue;\n\n (rings[i] as any).area = Math.abs(area);\n\n if (ccw === undefined) ccw = area < 0;\n\n if (ccw === area < 0) {\n if (polygon) polygons.push(polygon);\n polygon = [rings[i]];\n\n } else {\n (polygon as any).push(rings[i]);\n }\n }\n if (polygon) polygons.push(polygon);\n\n // Earcut performance degrades with the # of rings in a polygon. For this\n // reason, we limit strip out all but the `maxRings` largest rings.\n if (maxRings > 1) {\n for (let j = 0; j < polygons.length; j++) {\n if (polygons[j].length <= maxRings) continue;\n quickselect(polygons[j], maxRings, 1, polygons[j].length - 1, compareAreas);\n polygons[j] = polygons[j].slice(0, maxRings);\n }\n }\n\n return polygons;\n}\n\nfunction compareAreas(a, b) {\n return b.area - a.area;\n}\n","import type {FillStyleLayer} from '../../style/style_layer/fill_style_layer';\nimport type {FillExtrusionStyleLayer} from '../../style/style_layer/fill_extrusion_style_layer';\nimport type {LineStyleLayer} from '../../style/style_layer/line_style_layer';\n\nimport type {\n BucketFeature,\n PopulateParameters\n} from '../bucket';\nimport {PossiblyEvaluated} from '../../style/properties';\n\ntype PatternStyleLayers = Array | Array | Array;\n\nexport function hasPattern(type: string, layers: PatternStyleLayers, options: PopulateParameters) {\n const patterns = options.patternDependencies;\n let hasPattern = false;\n\n for (const layer of layers) {\n const patternProperty = (layer.paint as PossiblyEvaluated).get(`${type}-pattern`);\n if (!patternProperty.isConstant()) {\n hasPattern = true;\n }\n\n const constantPattern = patternProperty.constantOr(null);\n if (constantPattern) {\n hasPattern = true;\n patterns[constantPattern.to] = true;\n patterns[constantPattern.from] = true;\n }\n }\n\n return hasPattern;\n}\n\nexport function addPatternDependencies(type: string, layers: PatternStyleLayers, patternFeature: BucketFeature, zoom: number, options: PopulateParameters) {\n const patterns = options.patternDependencies;\n for (const layer of layers) {\n const patternProperty = (layer.paint as PossiblyEvaluated).get(`${type}-pattern`);\n\n const patternPropertyValue = patternProperty.value;\n if (patternPropertyValue.kind !== 'constant') {\n let min = patternPropertyValue.evaluate({zoom: zoom - 1}, patternFeature, {}, options.availableImages);\n let mid = patternPropertyValue.evaluate({zoom}, patternFeature, {}, options.availableImages);\n let max = patternPropertyValue.evaluate({zoom: zoom + 1}, patternFeature, {}, options.availableImages);\n min = min && min.name ? min.name : min;\n mid = mid && mid.name ? mid.name : mid;\n max = max && max.name ? max.name : max;\n // add to patternDependencies\n patterns[min] = true;\n patterns[mid] = true;\n patterns[max] = true;\n\n // save for layout\n patternFeature.patterns[layer.id] = {min, mid, max};\n }\n }\n return patternFeature;\n}\n","import {FillLayoutArray} from '../array_types.g';\n\nimport {members as layoutAttributes} from './fill_attributes';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {LineIndexArray, TriangleIndexArray} from '../index_array_type';\nimport earcut from 'earcut';\nimport {classifyRings} from '../../util/classify_rings';\nconst EARCUT_MAX_RINGS = 500;\nimport {register} from '../../util/web_worker_transfer';\nimport {hasPattern, addPatternDependencies} from './pattern_bucket_features';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {FillStyleLayer} from '../../style/style_layer/fill_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nexport class FillBucket implements Bucket {\n index: number;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n patternFeatures: Array;\n\n layoutVertexArray: FillLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n indexArray2: LineIndexArray;\n indexBuffer2: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n segments2: SegmentVector;\n uploaded: boolean;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n this.patternFeatures = [];\n\n this.layoutVertexArray = new FillLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.indexArray2 = new LineIndexArray();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.segments = new SegmentVector();\n this.segments2 = new SegmentVector();\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n this.hasPattern = hasPattern('fill', this.layers, options);\n const fillSortKey = this.layers[0].layout.get('fill-sort-key');\n const sortFeaturesByKey = !fillSortKey.isConstant();\n const bucketFeatures: BucketFeature[] = [];\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const sortKey = sortFeaturesByKey ?\n fillSortKey.evaluate(evaluationFeature, {}, canonical, options.availableImages) :\n undefined;\n\n const bucketFeature: BucketFeature = {\n id,\n properties: feature.properties,\n type: feature.type,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n patterns: {},\n sortKey\n };\n\n bucketFeatures.push(bucketFeature);\n }\n\n if (sortFeaturesByKey) {\n bucketFeatures.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const bucketFeature of bucketFeatures) {\n const {geometry, index, sourceLayerIndex} = bucketFeature;\n\n if (this.hasPattern) {\n const patternFeature = addPatternDependencies('fill', this.layers, bucketFeature, this.zoom, options);\n // pattern features are added only once the pattern is loaded into the image atlas\n // so are stored during populate until later updated with positions by tile worker in addFeatures\n this.patternFeatures.push(patternFeature);\n } else {\n this.addFeature(bucketFeature, geometry, index, canonical, {});\n }\n\n const feature = features[index].feature;\n options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {\n [_: string]: ImagePosition;\n }) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {\n [_: string]: ImagePosition;\n }) {\n for (const feature of this.patternFeatures) {\n this.addFeature(feature, feature.geometry, feature.index, canonical, imagePositions);\n }\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0;\n }\n\n uploadPending(): boolean {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n upload(context: Context) {\n if (!this.uploaded) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n this.indexBuffer2 = context.createIndexBuffer(this.indexArray2);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.indexBuffer2.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n this.segments2.destroy();\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {\n [_: string]: ImagePosition;\n }) {\n for (const polygon of classifyRings(geometry, EARCUT_MAX_RINGS)) {\n let numVertices = 0;\n for (const ring of polygon) {\n numVertices += ring.length;\n }\n\n const triangleSegment = this.segments.prepareSegment(numVertices, this.layoutVertexArray, this.indexArray);\n const triangleIndex = triangleSegment.vertexLength;\n\n const flattened = [];\n const holeIndices = [];\n\n for (const ring of polygon) {\n if (ring.length === 0) {\n continue;\n }\n\n if (ring !== polygon[0]) {\n holeIndices.push(flattened.length / 2);\n }\n\n const lineSegment = this.segments2.prepareSegment(ring.length, this.layoutVertexArray, this.indexArray2);\n const lineIndex = lineSegment.vertexLength;\n\n this.layoutVertexArray.emplaceBack(ring[0].x, ring[0].y);\n this.indexArray2.emplaceBack(lineIndex + ring.length - 1, lineIndex);\n flattened.push(ring[0].x);\n flattened.push(ring[0].y);\n\n for (let i = 1; i < ring.length; i++) {\n this.layoutVertexArray.emplaceBack(ring[i].x, ring[i].y);\n this.indexArray2.emplaceBack(lineIndex + i - 1, lineIndex + i);\n flattened.push(ring[i].x);\n flattened.push(ring[i].y);\n }\n\n lineSegment.vertexLength += ring.length;\n lineSegment.primitiveLength += ring.length;\n }\n\n const indices = earcut(flattened, holeIndices);\n\n for (let i = 0; i < indices.length; i += 3) {\n this.indexArray.emplaceBack(\n triangleIndex + indices[i],\n triangleIndex + indices[i + 1],\n triangleIndex + indices[i + 2]);\n }\n\n triangleSegment.vertexLength += numVertices;\n triangleSegment.primitiveLength += indices.length / 3;\n }\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical);\n }\n}\n\nregister('FillBucket', FillBucket, {omit: ['layers', 'patternFeatures']});\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type FillLayoutProps = {\n \"fill-sort-key\": DataDrivenProperty,\n};\n\nexport type FillLayoutPropsPossiblyEvaluated = {\n \"fill-sort-key\": PossiblyEvaluatedPropertyValue,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"fill-sort-key\": new DataDrivenProperty(styleSpec[\"layout_fill\"][\"fill-sort-key\"] as any as StylePropertySpecification),\n});\n\nexport type FillPaintProps = {\n \"fill-antialias\": DataConstantProperty,\n \"fill-opacity\": DataDrivenProperty,\n \"fill-color\": DataDrivenProperty,\n \"fill-outline-color\": DataDrivenProperty,\n \"fill-translate\": DataConstantProperty<[number, number]>,\n \"fill-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"fill-pattern\": CrossFadedDataDrivenProperty,\n};\n\nexport type FillPaintPropsPossiblyEvaluated = {\n \"fill-antialias\": boolean,\n \"fill-opacity\": PossiblyEvaluatedPropertyValue,\n \"fill-color\": PossiblyEvaluatedPropertyValue,\n \"fill-outline-color\": PossiblyEvaluatedPropertyValue,\n \"fill-translate\": [number, number],\n \"fill-translate-anchor\": \"map\" | \"viewport\",\n \"fill-pattern\": PossiblyEvaluatedPropertyValue>,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"fill-antialias\": new DataConstantProperty(styleSpec[\"paint_fill\"][\"fill-antialias\"] as any as StylePropertySpecification),\n \"fill-opacity\": new DataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-opacity\"] as any as StylePropertySpecification),\n \"fill-color\": new DataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-color\"] as any as StylePropertySpecification),\n \"fill-outline-color\": new DataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-outline-color\"] as any as StylePropertySpecification),\n \"fill-translate\": new DataConstantProperty(styleSpec[\"paint_fill\"][\"fill-translate\"] as any as StylePropertySpecification),\n \"fill-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_fill\"][\"fill-translate-anchor\"] as any as StylePropertySpecification),\n \"fill-pattern\": new CrossFadedDataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-pattern\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import {StyleLayer} from '../style_layer';\n\nimport {FillBucket} from '../../data/bucket/fill_bucket';\nimport {polygonIntersectsMultiPolygon} from '../../util/intersection_tests';\nimport {translateDistance, translate} from '../query_utils';\nimport properties, {FillLayoutPropsPossiblyEvaluated, FillPaintPropsPossiblyEvaluated} from './fill_style_layer_properties.g';\nimport {Transitionable, Transitioning, Layout, PossiblyEvaluated} from '../properties';\n\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {BucketParameters} from '../../data/bucket';\nimport type Point from '@mapbox/point-geometry';\nimport type {FillLayoutProps, FillPaintProps} from './fill_style_layer_properties.g';\nimport type {EvaluationParameters} from '../evaluation_parameters';\nimport type {Transform} from '../../geo/transform';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nexport class FillStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n super.recalculate(parameters, availableImages);\n\n const outlineColor = this.paint._values['fill-outline-color'];\n if (outlineColor.value.kind === 'constant' && outlineColor.value.value === undefined) {\n this.paint._values['fill-outline-color'] = this.paint._values['fill-color'];\n }\n }\n\n createBucket(parameters: BucketParameters) {\n return new FillBucket(parameters);\n }\n\n queryRadius(): number {\n return translateDistance(this.paint.get('fill-translate'));\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number\n ): boolean {\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('fill-translate'),\n this.paint.get('fill-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n return polygonIntersectsMultiPolygon(translatedPolygon, geometry);\n }\n\n isTileClipped() {\n return true;\n }\n}\n","import {createLayout} from '../../util/struct_array';\n\nconst layout = createLayout([\n {name: 'a_pos', components: 2, type: 'Int16'},\n {name: 'a_normal_ed', components: 4, type: 'Int16'},\n], 4);\n\nexport const centroidAttributes = createLayout([\n {name: 'a_centroid', components: 2, type: 'Int16'}\n], 4);\n\nexport default layout;\nexport const {members, size, alignment} = layout;\n","'use strict';\n\nvar Point = require('@mapbox/point-geometry');\n\nmodule.exports = VectorTileFeature;\n\nfunction VectorTileFeature(pbf, end, extent, keys, values) {\n // Public\n this.properties = {};\n this.extent = extent;\n this.type = 0;\n\n // Private\n this._pbf = pbf;\n this._geometry = -1;\n this._keys = keys;\n this._values = values;\n\n pbf.readFields(readFeature, this, end);\n}\n\nfunction readFeature(tag, feature, pbf) {\n if (tag == 1) feature.id = pbf.readVarint();\n else if (tag == 2) readTag(pbf, feature);\n else if (tag == 3) feature.type = pbf.readVarint();\n else if (tag == 4) feature._geometry = pbf.pos;\n}\n\nfunction readTag(pbf, feature) {\n var end = pbf.readVarint() + pbf.pos;\n\n while (pbf.pos < end) {\n var key = feature._keys[pbf.readVarint()],\n value = feature._values[pbf.readVarint()];\n feature.properties[key] = value;\n }\n}\n\nVectorTileFeature.types = ['Unknown', 'Point', 'LineString', 'Polygon'];\n\nVectorTileFeature.prototype.loadGeometry = function() {\n var pbf = this._pbf;\n pbf.pos = this._geometry;\n\n var end = pbf.readVarint() + pbf.pos,\n cmd = 1,\n length = 0,\n x = 0,\n y = 0,\n lines = [],\n line;\n\n while (pbf.pos < end) {\n if (length <= 0) {\n var cmdLen = pbf.readVarint();\n cmd = cmdLen & 0x7;\n length = cmdLen >> 3;\n }\n\n length--;\n\n if (cmd === 1 || cmd === 2) {\n x += pbf.readSVarint();\n y += pbf.readSVarint();\n\n if (cmd === 1) { // moveTo\n if (line) lines.push(line);\n line = [];\n }\n\n line.push(new Point(x, y));\n\n } else if (cmd === 7) {\n\n // Workaround for https://github.com/mapbox/mapnik-vector-tile/issues/90\n if (line) {\n line.push(line[0].clone()); // closePolygon\n }\n\n } else {\n throw new Error('unknown command ' + cmd);\n }\n }\n\n if (line) lines.push(line);\n\n return lines;\n};\n\nVectorTileFeature.prototype.bbox = function() {\n var pbf = this._pbf;\n pbf.pos = this._geometry;\n\n var end = pbf.readVarint() + pbf.pos,\n cmd = 1,\n length = 0,\n x = 0,\n y = 0,\n x1 = Infinity,\n x2 = -Infinity,\n y1 = Infinity,\n y2 = -Infinity;\n\n while (pbf.pos < end) {\n if (length <= 0) {\n var cmdLen = pbf.readVarint();\n cmd = cmdLen & 0x7;\n length = cmdLen >> 3;\n }\n\n length--;\n\n if (cmd === 1 || cmd === 2) {\n x += pbf.readSVarint();\n y += pbf.readSVarint();\n if (x < x1) x1 = x;\n if (x > x2) x2 = x;\n if (y < y1) y1 = y;\n if (y > y2) y2 = y;\n\n } else if (cmd !== 7) {\n throw new Error('unknown command ' + cmd);\n }\n }\n\n return [x1, y1, x2, y2];\n};\n\nVectorTileFeature.prototype.toGeoJSON = function(x, y, z) {\n var size = this.extent * Math.pow(2, z),\n x0 = this.extent * x,\n y0 = this.extent * y,\n coords = this.loadGeometry(),\n type = VectorTileFeature.types[this.type],\n i, j;\n\n function project(line) {\n for (var j = 0; j < line.length; j++) {\n var p = line[j], y2 = 180 - (p.y + y0) * 360 / size;\n line[j] = [\n (p.x + x0) * 360 / size - 180,\n 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90\n ];\n }\n }\n\n switch (this.type) {\n case 1:\n var points = [];\n for (i = 0; i < coords.length; i++) {\n points[i] = coords[i][0];\n }\n coords = points;\n project(coords);\n break;\n\n case 2:\n for (i = 0; i < coords.length; i++) {\n project(coords[i]);\n }\n break;\n\n case 3:\n coords = classifyRings(coords);\n for (i = 0; i < coords.length; i++) {\n for (j = 0; j < coords[i].length; j++) {\n project(coords[i][j]);\n }\n }\n break;\n }\n\n if (coords.length === 1) {\n coords = coords[0];\n } else {\n type = 'Multi' + type;\n }\n\n var result = {\n type: \"Feature\",\n geometry: {\n type: type,\n coordinates: coords\n },\n properties: this.properties\n };\n\n if ('id' in this) {\n result.id = this.id;\n }\n\n return result;\n};\n\n// classifies an array of rings into polygons with outer rings and holes\n\nfunction classifyRings(rings) {\n var len = rings.length;\n\n if (len <= 1) return [rings];\n\n var polygons = [],\n polygon,\n ccw;\n\n for (var i = 0; i < len; i++) {\n var area = signedArea(rings[i]);\n if (area === 0) continue;\n\n if (ccw === undefined) ccw = area < 0;\n\n if (ccw === area < 0) {\n if (polygon) polygons.push(polygon);\n polygon = [rings[i]];\n\n } else {\n polygon.push(rings[i]);\n }\n }\n if (polygon) polygons.push(polygon);\n\n return polygons;\n}\n\nfunction signedArea(ring) {\n var sum = 0;\n for (var i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) {\n p1 = ring[i];\n p2 = ring[j];\n sum += (p2.x - p1.x) * (p1.y + p2.y);\n }\n return sum;\n}\n","'use strict';\n\nvar VectorTileFeature = require('./vectortilefeature.js');\n\nmodule.exports = VectorTileLayer;\n\nfunction VectorTileLayer(pbf, end) {\n // Public\n this.version = 1;\n this.name = null;\n this.extent = 4096;\n this.length = 0;\n\n // Private\n this._pbf = pbf;\n this._keys = [];\n this._values = [];\n this._features = [];\n\n pbf.readFields(readLayer, this, end);\n\n this.length = this._features.length;\n}\n\nfunction readLayer(tag, layer, pbf) {\n if (tag === 15) layer.version = pbf.readVarint();\n else if (tag === 1) layer.name = pbf.readString();\n else if (tag === 5) layer.extent = pbf.readVarint();\n else if (tag === 2) layer._features.push(pbf.pos);\n else if (tag === 3) layer._keys.push(pbf.readString());\n else if (tag === 4) layer._values.push(readValueMessage(pbf));\n}\n\nfunction readValueMessage(pbf) {\n var value = null,\n end = pbf.readVarint() + pbf.pos;\n\n while (pbf.pos < end) {\n var tag = pbf.readVarint() >> 3;\n\n value = tag === 1 ? pbf.readString() :\n tag === 2 ? pbf.readFloat() :\n tag === 3 ? pbf.readDouble() :\n tag === 4 ? pbf.readVarint64() :\n tag === 5 ? pbf.readVarint() :\n tag === 6 ? pbf.readSVarint() :\n tag === 7 ? pbf.readBoolean() : null;\n }\n\n return value;\n}\n\n// return feature `i` from this layer as a `VectorTileFeature`\nVectorTileLayer.prototype.feature = function(i) {\n if (i < 0 || i >= this._features.length) throw new Error('feature index out of bounds');\n\n this._pbf.pos = this._features[i];\n\n var end = this._pbf.readVarint() + this._pbf.pos;\n return new VectorTileFeature(this._pbf, end, this.extent, this._keys, this._values);\n};\n","'use strict';\n\nvar VectorTileLayer = require('./vectortilelayer');\n\nmodule.exports = VectorTile;\n\nfunction VectorTile(pbf, end) {\n this.layers = pbf.readFields(readTile, {}, end);\n}\n\nfunction readTile(tag, layers, pbf) {\n if (tag === 3) {\n var layer = new VectorTileLayer(pbf, pbf.readVarint() + pbf.pos);\n if (layer.length) layers[layer.name] = layer;\n }\n}\n\n","module.exports.VectorTile = require('./lib/vectortile.js');\nmodule.exports.VectorTileFeature = require('./lib/vectortilefeature.js');\nmodule.exports.VectorTileLayer = require('./lib/vectortilelayer.js');\n","import {FillExtrusionLayoutArray, PosArray} from '../array_types.g';\n\nimport {members as layoutAttributes, centroidAttributes} from './fill_extrusion_attributes';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray} from '../index_array_type';\nimport {EXTENT} from '../extent';\nimport earcut from 'earcut';\nimport mvt from '@mapbox/vector-tile';\nconst vectorTileFeatureTypes = mvt.VectorTileFeature.types;\nimport {classifyRings} from '../../util/classify_rings';\nconst EARCUT_MAX_RINGS = 500;\nimport {register} from '../../util/web_worker_transfer';\nimport {hasPattern, addPatternDependencies} from './pattern_bucket_features';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\n\nimport type {FillExtrusionStyleLayer} from '../../style/style_layer/fill_extrusion_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nconst FACTOR = Math.pow(2, 13);\n\nfunction addVertex(vertexArray, x, y, nx, ny, nz, t, e) {\n vertexArray.emplaceBack(\n // a_pos\n x,\n y,\n // a_normal_ed: 3-component normal and 1-component edgedistance\n Math.floor(nx * FACTOR) * 2 + t,\n ny * FACTOR * 2,\n nz * FACTOR * 2,\n // edgedistance (used for wrapping patterns around extrusion sides)\n Math.round(e)\n );\n}\n\nexport class FillExtrusionBucket implements Bucket {\n index: number;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n\n layoutVertexArray: FillExtrusionLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n centroidVertexArray: PosArray;\n centroidVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n uploaded: boolean;\n features: Array;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n\n this.layoutVertexArray = new FillExtrusionLayoutArray();\n this.centroidVertexArray = new PosArray();\n this.indexArray = new TriangleIndexArray();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.segments = new SegmentVector();\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n this.features = [];\n this.hasPattern = hasPattern('fill-extrusion', this.layers, options);\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const bucketFeature: BucketFeature = {\n id,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n properties: feature.properties,\n type: feature.type,\n patterns: {}\n };\n\n if (this.hasPattern) {\n this.features.push(addPatternDependencies('fill-extrusion', this.layers, bucketFeature, this.zoom, options));\n } else {\n this.addFeature(bucketFeature, bucketFeature.geometry, index, canonical, {});\n }\n\n options.featureIndex.insert(feature, bucketFeature.geometry, index, sourceLayerIndex, this.index, true);\n }\n }\n\n addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n for (const feature of this.features) {\n const {geometry} = feature;\n this.addFeature(feature, geometry, feature.index, canonical, imagePositions);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0 && this.centroidVertexArray.length === 0;\n }\n\n uploadPending() {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.centroidVertexBuffer = context.createVertexBuffer(this.centroidVertexArray, centroidAttributes.members, true);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n this.centroidVertexBuffer.destroy();\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n const centroid = {x: 0, y: 0, vertexCount: 0};\n for (const polygon of classifyRings(geometry, EARCUT_MAX_RINGS)) {\n let numVertices = 0;\n for (const ring of polygon) {\n numVertices += ring.length;\n }\n let segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray);\n\n for (const ring of polygon) {\n if (ring.length === 0) {\n continue;\n }\n\n if (isEntirelyOutside(ring)) {\n continue;\n }\n\n let edgeDistance = 0;\n\n for (let p = 0; p < ring.length; p++) {\n const p1 = ring[p];\n\n if (p >= 1) {\n const p2 = ring[p - 1];\n\n if (!isBoundaryEdge(p1, p2)) {\n if (segment.vertexLength + 4 > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) {\n segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray);\n }\n\n const perp = p1.sub(p2)._perp()._unit();\n const dist = p2.dist(p1);\n if (edgeDistance + dist > 32768) edgeDistance = 0;\n\n addVertex(this.layoutVertexArray, p1.x, p1.y, perp.x, perp.y, 0, 0, edgeDistance);\n addVertex(this.layoutVertexArray, p1.x, p1.y, perp.x, perp.y, 0, 1, edgeDistance);\n centroid.x += 2 * p1.x;\n centroid.y += 2 * p1.y;\n centroid.vertexCount += 2;\n\n edgeDistance += dist;\n\n addVertex(this.layoutVertexArray, p2.x, p2.y, perp.x, perp.y, 0, 0, edgeDistance);\n addVertex(this.layoutVertexArray, p2.x, p2.y, perp.x, perp.y, 0, 1, edgeDistance);\n centroid.x += 2 * p2.x;\n centroid.y += 2 * p2.y;\n centroid.vertexCount += 2;\n\n const bottomRight = segment.vertexLength;\n\n // ┌──────┐\n // │ 0 1 │ Counter-clockwise winding order.\n // │ │ Triangle 1: 0 => 2 => 1\n // │ 2 3 │ Triangle 2: 1 => 2 => 3\n // └──────┘\n this.indexArray.emplaceBack(bottomRight, bottomRight + 2, bottomRight + 1);\n this.indexArray.emplaceBack(bottomRight + 1, bottomRight + 2, bottomRight + 3);\n\n segment.vertexLength += 4;\n segment.primitiveLength += 2;\n }\n }\n }\n\n }\n\n if (segment.vertexLength + numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) {\n segment = this.segments.prepareSegment(numVertices, this.layoutVertexArray, this.indexArray);\n }\n\n //Only triangulate and draw the area of the feature if it is a polygon\n //Other feature types (e.g. LineString) do not have area, so triangulation is pointless / undefined\n if (vectorTileFeatureTypes[feature.type] !== 'Polygon')\n continue;\n\n const flattened = [];\n const holeIndices = [];\n const triangleIndex = segment.vertexLength;\n\n for (const ring of polygon) {\n if (ring.length === 0) {\n continue;\n }\n\n if (ring !== polygon[0]) {\n holeIndices.push(flattened.length / 2);\n }\n\n for (let i = 0; i < ring.length; i++) {\n const p = ring[i];\n\n addVertex(this.layoutVertexArray, p.x, p.y, 0, 0, 1, 1, 0);\n centroid.x += p.x;\n centroid.y += p.y;\n centroid.vertexCount += 1;\n\n flattened.push(p.x);\n flattened.push(p.y);\n }\n\n }\n\n const indices = earcut(flattened, holeIndices);\n\n for (let j = 0; j < indices.length; j += 3) {\n // Counter-clockwise winding order.\n this.indexArray.emplaceBack(\n triangleIndex + indices[j],\n triangleIndex + indices[j + 2],\n triangleIndex + indices[j + 1]);\n }\n\n segment.primitiveLength += indices.length / 3;\n segment.vertexLength += numVertices;\n }\n\n // remember polygon centroid to calculate elevation in GPU\n for (let i = 0; i < centroid.vertexCount; i++) {\n this.centroidVertexArray.emplaceBack(\n Math.floor(centroid.x / centroid.vertexCount),\n Math.floor(centroid.y / centroid.vertexCount)\n );\n }\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical);\n }\n}\n\nregister('FillExtrusionBucket', FillExtrusionBucket, {omit: ['layers', 'features']});\n\nfunction isBoundaryEdge(p1, p2) {\n return (p1.x === p2.x && (p1.x < 0 || p1.x > EXTENT)) ||\n (p1.y === p2.y && (p1.y < 0 || p1.y > EXTENT));\n}\n\nfunction isEntirelyOutside(ring) {\n return ring.every(p => p.x < 0) ||\n ring.every(p => p.x > EXTENT) ||\n ring.every(p => p.y < 0) ||\n ring.every(p => p.y > EXTENT);\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type FillExtrusionPaintProps = {\n \"fill-extrusion-opacity\": DataConstantProperty,\n \"fill-extrusion-color\": DataDrivenProperty,\n \"fill-extrusion-translate\": DataConstantProperty<[number, number]>,\n \"fill-extrusion-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"fill-extrusion-pattern\": CrossFadedDataDrivenProperty,\n \"fill-extrusion-height\": DataDrivenProperty,\n \"fill-extrusion-base\": DataDrivenProperty,\n \"fill-extrusion-vertical-gradient\": DataConstantProperty,\n};\n\nexport type FillExtrusionPaintPropsPossiblyEvaluated = {\n \"fill-extrusion-opacity\": number,\n \"fill-extrusion-color\": PossiblyEvaluatedPropertyValue,\n \"fill-extrusion-translate\": [number, number],\n \"fill-extrusion-translate-anchor\": \"map\" | \"viewport\",\n \"fill-extrusion-pattern\": PossiblyEvaluatedPropertyValue>,\n \"fill-extrusion-height\": PossiblyEvaluatedPropertyValue,\n \"fill-extrusion-base\": PossiblyEvaluatedPropertyValue,\n \"fill-extrusion-vertical-gradient\": boolean,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"fill-extrusion-opacity\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-opacity\"] as any as StylePropertySpecification),\n \"fill-extrusion-color\": new DataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-color\"] as any as StylePropertySpecification),\n \"fill-extrusion-translate\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-translate\"] as any as StylePropertySpecification),\n \"fill-extrusion-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-translate-anchor\"] as any as StylePropertySpecification),\n \"fill-extrusion-pattern\": new CrossFadedDataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-pattern\"] as any as StylePropertySpecification),\n \"fill-extrusion-height\": new DataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-height\"] as any as StylePropertySpecification),\n \"fill-extrusion-base\": new DataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-base\"] as any as StylePropertySpecification),\n \"fill-extrusion-vertical-gradient\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-vertical-gradient\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport {FillExtrusionBucket} from '../../data/bucket/fill_extrusion_bucket';\nimport {polygonIntersectsPolygon, polygonIntersectsMultiPolygon} from '../../util/intersection_tests';\nimport {translateDistance, translate} from '../query_utils';\nimport properties, {FillExtrusionPaintPropsPossiblyEvaluated} from './fill_extrusion_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\nimport {mat4, vec4} from 'gl-matrix';\nimport Point from '@mapbox/point-geometry';\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {BucketParameters} from '../../data/bucket';\nimport type {FillExtrusionPaintProps} from './fill_extrusion_style_layer_properties.g';\nimport type {Transform} from '../../geo/transform';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nexport class Point3D extends Point {\n z: number;\n}\n\nexport class FillExtrusionStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n createBucket(parameters: BucketParameters) {\n return new FillExtrusionBucket(parameters);\n }\n\n queryRadius(): number {\n return translateDistance(this.paint.get('fill-extrusion-translate'));\n }\n\n is3D(): boolean {\n return true;\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number,\n pixelPosMatrix: mat4\n ): boolean | number {\n\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('fill-extrusion-translate'),\n this.paint.get('fill-extrusion-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n\n const height = this.paint.get('fill-extrusion-height').evaluate(feature, featureState);\n const base = this.paint.get('fill-extrusion-base').evaluate(feature, featureState);\n\n const projectedQueryGeometry = projectQueryGeometry(translatedPolygon, pixelPosMatrix, transform, 0);\n\n const projected = projectExtrusion(geometry, base, height, pixelPosMatrix);\n const projectedBase = projected[0];\n const projectedTop = projected[1];\n return checkIntersection(projectedBase, projectedTop, projectedQueryGeometry);\n }\n}\n\nfunction dot(a, b) {\n return a.x * b.x + a.y * b.y;\n}\n\nexport function getIntersectionDistance(projectedQueryGeometry: Array, projectedFace: Array) {\n\n if (projectedQueryGeometry.length === 1) {\n // For point queries calculate the z at which the point intersects the face\n // using barycentric coordinates.\n\n // Find the barycentric coordinates of the projected point within the first\n // triangle of the face, using only the xy plane. It doesn't matter if the\n // point is outside the first triangle because all the triangles in the face\n // are in the same plane.\n //\n // Check whether points are coincident and use other points if they are.\n let i = 0;\n const a = projectedFace[i++];\n let b;\n while (!b || a.equals(b)) {\n b = projectedFace[i++];\n if (!b) return Infinity;\n }\n\n // Loop until point `c` is not colinear with points `a` and `b`.\n for (; i < projectedFace.length; i++) {\n const c = projectedFace[i];\n\n const p = projectedQueryGeometry[0];\n\n const ab = b.sub(a);\n const ac = c.sub(a);\n const ap = p.sub(a);\n\n const dotABAB = dot(ab, ab);\n const dotABAC = dot(ab, ac);\n const dotACAC = dot(ac, ac);\n const dotAPAB = dot(ap, ab);\n const dotAPAC = dot(ap, ac);\n const denom = dotABAB * dotACAC - dotABAC * dotABAC;\n\n const v = (dotACAC * dotAPAB - dotABAC * dotAPAC) / denom;\n const w = (dotABAB * dotAPAC - dotABAC * dotAPAB) / denom;\n const u = 1 - v - w;\n\n // Use the barycentric weighting along with the original triangle z coordinates to get the point of intersection.\n const distance = a.z * u + b.z * v + c.z * w;\n\n if (isFinite(distance)) return distance;\n }\n\n return Infinity;\n\n } else {\n // The counts as closest is less clear when the query is a box. This\n // returns the distance to the nearest point on the face, whether it is\n // within the query or not. It could be more correct to return the\n // distance to the closest point within the query box but this would be\n // more complicated and expensive to calculate with little benefit.\n let closestDistance = Infinity;\n for (const p of projectedFace) {\n closestDistance = Math.min(closestDistance, p.z);\n }\n return closestDistance;\n }\n}\n\nfunction checkIntersection(projectedBase: Array>, projectedTop: Array>, projectedQueryGeometry: Array) {\n let closestDistance = Infinity;\n\n if (polygonIntersectsMultiPolygon(projectedQueryGeometry, projectedTop)) {\n closestDistance = getIntersectionDistance(projectedQueryGeometry, projectedTop[0]);\n }\n\n for (let r = 0; r < projectedTop.length; r++) {\n const ringTop = projectedTop[r];\n const ringBase = projectedBase[r];\n for (let p = 0; p < ringTop.length - 1; p++) {\n const topA = ringTop[p];\n const topB = ringTop[p + 1];\n const baseA = ringBase[p];\n const baseB = ringBase[p + 1];\n const face = [topA, topB, baseB, baseA, topA];\n if (polygonIntersectsPolygon(projectedQueryGeometry, face)) {\n closestDistance = Math.min(closestDistance, getIntersectionDistance(projectedQueryGeometry, face));\n }\n }\n }\n\n return closestDistance === Infinity ? false : closestDistance;\n}\n\n/*\n * Project the geometry using matrix `m`. This is essentially doing\n * `vec4.transformMat4([], [p.x, p.y, z, 1], m)` but the multiplication\n * is inlined so that parts of the projection that are the same across\n * different points can only be done once. This produced a measurable\n * performance improvement.\n */\nfunction projectExtrusion(geometry: Array>, zBase: number, zTop: number, m: mat4): [Array>, Array>] {\n const projectedBase = [] as Array>;\n const projectedTop = [] as Array>;\n const baseXZ = m[8] * zBase;\n const baseYZ = m[9] * zBase;\n const baseZZ = m[10] * zBase;\n const baseWZ = m[11] * zBase;\n const topXZ = m[8] * zTop;\n const topYZ = m[9] * zTop;\n const topZZ = m[10] * zTop;\n const topWZ = m[11] * zTop;\n\n for (const r of geometry) {\n const ringBase = [] as Array;\n const ringTop = [] as Array;\n for (const p of r) {\n const x = p.x;\n const y = p.y;\n\n const sX = m[0] * x + m[4] * y + m[12];\n const sY = m[1] * x + m[5] * y + m[13];\n const sZ = m[2] * x + m[6] * y + m[14];\n const sW = m[3] * x + m[7] * y + m[15];\n\n const baseX = sX + baseXZ;\n const baseY = sY + baseYZ;\n const baseZ = sZ + baseZZ;\n const baseW = sW + baseWZ;\n\n const topX = sX + topXZ;\n const topY = sY + topYZ;\n const topZ = sZ + topZZ;\n const topW = sW + topWZ;\n\n const b = new Point(baseX / baseW, baseY / baseW) as Point3D;\n b.z = baseZ / baseW;\n ringBase.push(b);\n\n const t = new Point(topX / topW, topY / topW) as Point3D;\n t.z = topZ / topW;\n ringTop.push(t);\n }\n projectedBase.push(ringBase);\n projectedTop.push(ringTop);\n }\n return [projectedBase, projectedTop];\n}\n\nfunction projectQueryGeometry(queryGeometry: Array, pixelPosMatrix: mat4, transform: Transform, z: number) {\n const projectedQueryGeometry = [];\n for (const p of queryGeometry) {\n const v = [p.x, p.y, z, 1] as vec4;\n vec4.transformMat4(v, v, pixelPosMatrix);\n projectedQueryGeometry.push(new Point(v[0] / v[3], v[1] / v[3]));\n }\n return projectedQueryGeometry;\n}\n","import {createLayout} from '../../util/struct_array';\n\nexport const lineLayoutAttributes = createLayout([\n {name: 'a_pos_normal', components: 2, type: 'Int16'},\n {name: 'a_data', components: 4, type: 'Uint8'}\n], 4);\n\nexport const {members, size, alignment} = lineLayoutAttributes;\n","import {createLayout} from '../../util/struct_array';\n\nexport const lineLayoutAttributesExt = createLayout([\n {name: 'a_uv_x', components: 1, type: 'Float32'},\n {name: 'a_split_index', components: 1, type: 'Float32'},\n]);\n\nexport const {members, size, alignment} = lineLayoutAttributesExt;\n","import {LineLayoutArray, LineExtLayoutArray} from '../array_types.g';\n\nimport {members as layoutAttributes} from './line_attributes';\nimport {members as layoutAttributesExt} from './line_attributes_ext';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray} from '../index_array_type';\nimport {EXTENT} from '../extent';\nimport mvt from '@mapbox/vector-tile';\nconst vectorTileFeatureTypes = mvt.VectorTileFeature.types;\nimport {register} from '../../util/web_worker_transfer';\nimport {hasPattern, addPatternDependencies} from './pattern_bucket_features';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {LineStyleLayer} from '../../style/style_layer/line_style_layer';\nimport type Point from '@mapbox/point-geometry';\nimport type {Segment} from '../segment';\nimport {RGBAImage} from '../../util/image';\nimport type {Context} from '../../gl/context';\nimport type {Texture} from '../../render/texture';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\n// NOTE ON EXTRUDE SCALE:\n// scale the extrusion vector so that the normal length is this value.\n// contains the \"texture\" normals (-1..1). this is distinct from the extrude\n// normals for line joins, because the x-value remains 0 for the texture\n// normal array, while the extrude normal actually moves the vertex to create\n// the acute/bevelled line join.\nconst EXTRUDE_SCALE = 63;\n\n/*\n * Sharp corners cause dashed lines to tilt because the distance along the line\n * is the same at both the inner and outer corners. To improve the appearance of\n * dashed lines we add extra points near sharp corners so that a smaller part\n * of the line is tilted.\n *\n * COS_HALF_SHARP_CORNER controls how sharp a corner has to be for us to add an\n * extra vertex. The default is 75 degrees.\n *\n * The newly created vertices are placed SHARP_CORNER_OFFSET pixels from the corner.\n */\nconst COS_HALF_SHARP_CORNER = Math.cos(75 / 2 * (Math.PI / 180));\nconst SHARP_CORNER_OFFSET = 15;\n\n// Angle per triangle for approximating round line joins.\nconst DEG_PER_TRIANGLE = 20;\n\n// The number of bits that is used to store the line distance in the buffer.\nconst LINE_DISTANCE_BUFFER_BITS = 15;\n\n// We don't have enough bits for the line distance as we'd like to have, so\n// use this value to scale the line distance (in tile units) down to a smaller\n// value. This lets us store longer distances while sacrificing precision.\nconst LINE_DISTANCE_SCALE = 1 / 2;\n\n// The maximum line distance, in tile units, that fits in the buffer.\nconst MAX_LINE_DISTANCE = Math.pow(2, LINE_DISTANCE_BUFFER_BITS - 1) / LINE_DISTANCE_SCALE;\n\ntype LineClips = {\n start: number;\n end: number;\n};\n\ntype GradientTexture = {\n texture?: Texture;\n gradient?: RGBAImage;\n version?: number;\n};\n\n/**\n * @internal\n * Line bucket class\n */\nexport class LineBucket implements Bucket {\n distance: number;\n totalDistance: number;\n maxLineLength: number;\n scaledDistance: number;\n lineClips?: LineClips;\n\n e1: number;\n e2: number;\n\n index: number;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n gradients: {[x: string]: GradientTexture};\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n patternFeatures: Array;\n lineClipsArray: Array;\n\n layoutVertexArray: LineLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n layoutVertexArray2: LineExtLayoutArray;\n layoutVertexBuffer2: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n uploaded: boolean;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n this.patternFeatures = [];\n this.lineClipsArray = [];\n this.gradients = {};\n this.layers.forEach(layer => {\n this.gradients[layer.id] = {};\n });\n\n this.layoutVertexArray = new LineLayoutArray();\n this.layoutVertexArray2 = new LineExtLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.segments = new SegmentVector();\n this.maxLineLength = 0;\n\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n this.hasPattern = hasPattern('line', this.layers, options);\n const lineSortKey = this.layers[0].layout.get('line-sort-key');\n const sortFeaturesByKey = !lineSortKey.isConstant();\n const bucketFeatures: BucketFeature[] = [];\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const sortKey = sortFeaturesByKey ?\n lineSortKey.evaluate(evaluationFeature, {}, canonical) :\n undefined;\n\n const bucketFeature: BucketFeature = {\n id,\n properties: feature.properties,\n type: feature.type,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n patterns: {},\n sortKey\n };\n\n bucketFeatures.push(bucketFeature);\n }\n\n if (sortFeaturesByKey) {\n bucketFeatures.sort((a, b) => {\n return (a.sortKey) - (b.sortKey);\n });\n }\n\n for (const bucketFeature of bucketFeatures) {\n const {geometry, index, sourceLayerIndex} = bucketFeature;\n\n if (this.hasPattern) {\n const patternBucketFeature = addPatternDependencies('line', this.layers, bucketFeature, this.zoom, options);\n // pattern features are added only once the pattern is loaded into the image atlas\n // so are stored during populate until later updated with positions by tile worker in addFeatures\n this.patternFeatures.push(patternBucketFeature);\n } else {\n this.addFeature(bucketFeature, geometry, index, canonical, {});\n }\n\n const feature = features[index].feature;\n options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n for (const feature of this.patternFeatures) {\n this.addFeature(feature, feature.geometry, feature.index, canonical, imagePositions);\n }\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0;\n }\n\n uploadPending() {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded) {\n if (this.layoutVertexArray2.length !== 0) {\n this.layoutVertexBuffer2 = context.createVertexBuffer(this.layoutVertexArray2, layoutAttributesExt);\n }\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n }\n\n lineFeatureClips(feature: BucketFeature): LineClips | undefined {\n if (!!feature.properties && Object.prototype.hasOwnProperty.call(feature.properties, 'mapbox_clip_start') && Object.prototype.hasOwnProperty.call(feature.properties, 'mapbox_clip_end')) {\n const start = +feature.properties['mapbox_clip_start'];\n const end = +feature.properties['mapbox_clip_end'];\n return {start, end};\n }\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n const layout = this.layers[0].layout;\n const join = layout.get('line-join').evaluate(feature, {});\n const cap = layout.get('line-cap');\n const miterLimit = layout.get('line-miter-limit');\n const roundLimit = layout.get('line-round-limit');\n this.lineClips = this.lineFeatureClips(feature);\n\n for (const line of geometry) {\n this.addLine(line, feature, join, cap, miterLimit, roundLimit);\n }\n\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical);\n }\n\n addLine(vertices: Array, feature: BucketFeature, join: string, cap: string, miterLimit: number, roundLimit: number) {\n this.distance = 0;\n this.scaledDistance = 0;\n this.totalDistance = 0;\n\n if (this.lineClips) {\n this.lineClipsArray.push(this.lineClips);\n // Calculate the total distance, in tile units, of this tiled line feature\n for (let i = 0; i < vertices.length - 1; i++) {\n this.totalDistance += vertices[i].dist(vertices[i + 1]);\n }\n this.updateScaledDistance();\n this.maxLineLength = Math.max(this.maxLineLength, this.totalDistance);\n }\n\n const isPolygon = vectorTileFeatureTypes[feature.type] === 'Polygon';\n\n // If the line has duplicate vertices at the ends, adjust start/length to remove them.\n let len = vertices.length;\n while (len >= 2 && vertices[len - 1].equals(vertices[len - 2])) {\n len--;\n }\n let first = 0;\n while (first < len - 1 && vertices[first].equals(vertices[first + 1])) {\n first++;\n }\n\n // Ignore invalid geometry.\n if (len < (isPolygon ? 3 : 2)) return;\n\n if (join === 'bevel') miterLimit = 1.05;\n\n const sharpCornerOffset = this.overscaling <= 16 ?\n SHARP_CORNER_OFFSET * EXTENT / (512 * this.overscaling) :\n 0;\n\n // we could be more precise, but it would only save a negligible amount of space\n const segment = this.segments.prepareSegment(len * 10, this.layoutVertexArray, this.indexArray);\n\n let currentVertex: Point;\n let prevVertex: Point;\n let nextVertex: Point;\n let prevNormal: Point;\n let nextNormal: Point;\n\n // the last two vertices added\n this.e1 = this.e2 = -1;\n\n if (isPolygon) {\n currentVertex = vertices[len - 2];\n nextNormal = vertices[first].sub(currentVertex)._unit()._perp();\n }\n\n for (let i = first; i < len; i++) {\n\n nextVertex = i === len - 1 ?\n (isPolygon ? vertices[first + 1] : undefined) : // if it's a polygon, treat the last vertex like the first\n vertices[i + 1]; // just the next vertex\n\n // if two consecutive vertices exist, skip the current one\n if (nextVertex && vertices[i].equals(nextVertex)) continue;\n\n if (nextNormal) prevNormal = nextNormal;\n if (currentVertex) prevVertex = currentVertex;\n\n currentVertex = vertices[i];\n\n // Calculate the normal towards the next vertex in this line. In case\n // there is no next vertex, pretend that the line is continuing straight,\n // meaning that we are just using the previous normal.\n nextNormal = nextVertex ? nextVertex.sub(currentVertex)._unit()._perp() : prevNormal;\n\n // If we still don't have a previous normal, this is the beginning of a\n // non-closed line, so we're doing a straight \"join\".\n prevNormal = prevNormal || nextNormal;\n\n // Determine the normal of the join extrusion. It is the angle bisector\n // of the segments between the previous line and the next line.\n // In the case of 180° angles, the prev and next normals cancel each other out:\n // prevNormal + nextNormal = (0, 0), its magnitude is 0, so the unit vector would be\n // undefined. In that case, we're keeping the joinNormal at (0, 0), so that the cosHalfAngle\n // below will also become 0 and miterLength will become Infinity.\n let joinNormal = prevNormal.add(nextNormal);\n if (joinNormal.x !== 0 || joinNormal.y !== 0) {\n joinNormal._unit();\n }\n /* joinNormal prevNormal\n * ↖ ↑\n * .________. prevVertex\n * |\n * nextNormal ← | currentVertex\n * |\n * nextVertex !\n *\n */\n\n // calculate cosines of the angle (and its half) using dot product\n const cosAngle = prevNormal.x * nextNormal.x + prevNormal.y * nextNormal.y;\n const cosHalfAngle = joinNormal.x * nextNormal.x + joinNormal.y * nextNormal.y;\n\n // Calculate the length of the miter (the ratio of the miter to the width)\n // as the inverse of cosine of the angle between next and join normals\n const miterLength = cosHalfAngle !== 0 ? 1 / cosHalfAngle : Infinity;\n\n // approximate angle from cosine\n const approxAngle = 2 * Math.sqrt(2 - 2 * cosHalfAngle);\n\n const isSharpCorner = cosHalfAngle < COS_HALF_SHARP_CORNER && prevVertex && nextVertex;\n const lineTurnsLeft = prevNormal.x * nextNormal.y - prevNormal.y * nextNormal.x > 0;\n\n if (isSharpCorner && i > first) {\n const prevSegmentLength = currentVertex.dist(prevVertex);\n if (prevSegmentLength > 2 * sharpCornerOffset) {\n const newPrevVertex = currentVertex.sub(currentVertex.sub(prevVertex)._mult(sharpCornerOffset / prevSegmentLength)._round());\n this.updateDistance(prevVertex, newPrevVertex);\n this.addCurrentVertex(newPrevVertex, prevNormal, 0, 0, segment);\n prevVertex = newPrevVertex;\n }\n }\n\n // The join if a middle vertex, otherwise the cap.\n const middleVertex = prevVertex && nextVertex;\n let currentJoin = middleVertex ? join : isPolygon ? 'butt' : cap;\n\n if (middleVertex && currentJoin === 'round') {\n if (miterLength < roundLimit) {\n currentJoin = 'miter';\n } else if (miterLength <= 2) {\n currentJoin = 'fakeround';\n }\n }\n\n if (currentJoin === 'miter' && miterLength > miterLimit) {\n currentJoin = 'bevel';\n }\n\n if (currentJoin === 'bevel') {\n // The maximum extrude length is 128 / 63 = 2 times the width of the line\n // so if miterLength >= 2 we need to draw a different type of bevel here.\n if (miterLength > 2) currentJoin = 'flipbevel';\n\n // If the miterLength is really small and the line bevel wouldn't be visible,\n // just draw a miter join to save a triangle.\n if (miterLength < miterLimit) currentJoin = 'miter';\n }\n\n // Calculate how far along the line the currentVertex is\n if (prevVertex) this.updateDistance(prevVertex, currentVertex);\n\n if (currentJoin === 'miter') {\n\n joinNormal._mult(miterLength);\n this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment);\n\n } else if (currentJoin === 'flipbevel') {\n // miter is too big, flip the direction to make a beveled join\n\n if (miterLength > 100) {\n // Almost parallel lines\n joinNormal = nextNormal.mult(-1);\n\n } else {\n const bevelLength = miterLength * prevNormal.add(nextNormal).mag() / prevNormal.sub(nextNormal).mag();\n joinNormal._perp()._mult(bevelLength * (lineTurnsLeft ? -1 : 1));\n }\n this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment);\n this.addCurrentVertex(currentVertex, joinNormal.mult(-1), 0, 0, segment);\n\n } else if (currentJoin === 'bevel' || currentJoin === 'fakeround') {\n const offset = -Math.sqrt(miterLength * miterLength - 1);\n const offsetA = lineTurnsLeft ? offset : 0;\n const offsetB = lineTurnsLeft ? 0 : offset;\n\n // Close previous segment with a bevel\n if (prevVertex) {\n this.addCurrentVertex(currentVertex, prevNormal, offsetA, offsetB, segment);\n }\n\n if (currentJoin === 'fakeround') {\n // The join angle is sharp enough that a round join would be visible.\n // Bevel joins fill the gap between segments with a single pie slice triangle.\n // Create a round join by adding multiple pie slices. The join isn't actually round, but\n // it looks like it is at the sizes we render lines at.\n\n // pick the number of triangles for approximating round join by based on the angle between normals\n const n = Math.round((approxAngle * 180 / Math.PI) / DEG_PER_TRIANGLE);\n\n for (let m = 1; m < n; m++) {\n let t = m / n;\n if (t !== 0.5) {\n // approximate spherical interpolation https://observablehq.com/@mourner/approximating-geometric-slerp\n const t2 = t - 0.5;\n const A = 1.0904 + cosAngle * (-3.2452 + cosAngle * (3.55645 - cosAngle * 1.43519));\n const B = 0.848013 + cosAngle * (-1.06021 + cosAngle * 0.215638);\n t = t + t * t2 * (t - 1) * (A * t2 * t2 + B);\n }\n const extrude = nextNormal.sub(prevNormal)._mult(t)._add(prevNormal)._unit()._mult(lineTurnsLeft ? -1 : 1);\n this.addHalfVertex(currentVertex, extrude.x, extrude.y, false, lineTurnsLeft, 0, segment);\n }\n }\n\n if (nextVertex) {\n // Start next segment\n this.addCurrentVertex(currentVertex, nextNormal, -offsetA, -offsetB, segment);\n }\n\n } else if (currentJoin === 'butt') {\n this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment); // butt cap\n\n } else if (currentJoin === 'square') {\n const offset = prevVertex ? 1 : -1; // closing or starting square cap\n this.addCurrentVertex(currentVertex, joinNormal, offset, offset, segment);\n\n } else if (currentJoin === 'round') {\n\n if (prevVertex) {\n // Close previous segment with butt\n this.addCurrentVertex(currentVertex, prevNormal, 0, 0, segment);\n\n // Add round cap or linejoin at end of segment\n this.addCurrentVertex(currentVertex, prevNormal, 1, 1, segment, true);\n }\n if (nextVertex) {\n // Add round cap before first segment\n this.addCurrentVertex(currentVertex, nextNormal, -1, -1, segment, true);\n\n // Start next segment with a butt\n this.addCurrentVertex(currentVertex, nextNormal, 0, 0, segment);\n }\n }\n\n if (isSharpCorner && i < len - 1) {\n const nextSegmentLength = currentVertex.dist(nextVertex);\n if (nextSegmentLength > 2 * sharpCornerOffset) {\n const newCurrentVertex = currentVertex.add(nextVertex.sub(currentVertex)._mult(sharpCornerOffset / nextSegmentLength)._round());\n this.updateDistance(currentVertex, newCurrentVertex);\n this.addCurrentVertex(newCurrentVertex, nextNormal, 0, 0, segment);\n currentVertex = newCurrentVertex;\n }\n }\n }\n }\n\n /**\n * Add two vertices to the buffers.\n *\n * @param p - the line vertex to add buffer vertices for\n * @param normal - vertex normal\n * @param endLeft - extrude to shift the left vertex along the line\n * @param endRight - extrude to shift the left vertex along the line\n * @param segment - the segment object to add the vertex to\n * @param round - whether this is a round cap\n */\n addCurrentVertex(p: Point, normal: Point, endLeft: number, endRight: number, segment: Segment, round: boolean = false) {\n // left and right extrude vectors, perpendicularly shifted by endLeft/endRight\n const leftX = normal.x + normal.y * endLeft;\n const leftY = normal.y - normal.x * endLeft;\n const rightX = -normal.x + normal.y * endRight;\n const rightY = -normal.y - normal.x * endRight;\n\n this.addHalfVertex(p, leftX, leftY, round, false, endLeft, segment);\n this.addHalfVertex(p, rightX, rightY, round, true, -endRight, segment);\n\n // There is a maximum \"distance along the line\" that we can store in the buffers.\n // When we get close to the distance, reset it to zero and add the vertex again with\n // a distance of zero. The max distance is determined by the number of bits we allocate\n // to `linesofar`.\n if (this.distance > MAX_LINE_DISTANCE / 2 && this.totalDistance === 0) {\n this.distance = 0;\n this.updateScaledDistance();\n this.addCurrentVertex(p, normal, endLeft, endRight, segment, round);\n }\n }\n\n addHalfVertex({x, y}: Point, extrudeX: number, extrudeY: number, round: boolean, up: boolean, dir: number, segment: Segment) {\n const totalDistance = this.lineClips ? this.scaledDistance * (MAX_LINE_DISTANCE - 1) : this.scaledDistance;\n // scale down so that we can store longer distances while sacrificing precision.\n const linesofarScaled = totalDistance * LINE_DISTANCE_SCALE;\n\n this.layoutVertexArray.emplaceBack(\n // a_pos_normal\n // Encode round/up the least significant bits\n (x << 1) + (round ? 1 : 0),\n (y << 1) + (up ? 1 : 0),\n // a_data\n // add 128 to store a byte in an unsigned byte\n Math.round(EXTRUDE_SCALE * extrudeX) + 128,\n Math.round(EXTRUDE_SCALE * extrudeY) + 128,\n // Encode the -1/0/1 direction value into the first two bits of .z of a_data.\n // Combine it with the lower 6 bits of `linesofarScaled` (shifted by 2 bits to make\n // room for the direction value). The upper 8 bits of `linesofarScaled` are placed in\n // the `w` component.\n ((dir === 0 ? 0 : (dir < 0 ? -1 : 1)) + 1) | ((linesofarScaled & 0x3F) << 2),\n linesofarScaled >> 6);\n\n // Constructs a second vertex buffer with higher precision line progress\n if (this.lineClips) {\n const progressRealigned = this.scaledDistance - this.lineClips.start;\n const endClipRealigned = this.lineClips.end - this.lineClips.start;\n const uvX = progressRealigned / endClipRealigned;\n this.layoutVertexArray2.emplaceBack(uvX, this.lineClipsArray.length);\n }\n\n const e = segment.vertexLength++;\n if (this.e1 >= 0 && this.e2 >= 0) {\n this.indexArray.emplaceBack(this.e1, this.e2, e);\n segment.primitiveLength++;\n }\n if (up) {\n this.e2 = e;\n } else {\n this.e1 = e;\n }\n }\n\n updateScaledDistance() {\n // Knowing the ratio of the full linestring covered by this tiled feature, as well\n // as the total distance (in tile units) of this tiled feature, and the distance\n // (in tile units) of the current vertex, we can determine the relative distance\n // of this vertex along the full linestring feature and scale it to [0, 2^15)\n this.scaledDistance = this.lineClips ?\n this.lineClips.start + (this.lineClips.end - this.lineClips.start) * this.distance / this.totalDistance :\n this.distance;\n }\n\n updateDistance(prev: Point, next: Point) {\n this.distance += prev.dist(next);\n this.updateScaledDistance();\n }\n}\n\nregister('LineBucket', LineBucket, {omit: ['layers', 'patternFeatures']});\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type LineLayoutProps = {\n \"line-cap\": DataConstantProperty<\"butt\" | \"round\" | \"square\">,\n \"line-join\": DataDrivenProperty<\"bevel\" | \"round\" | \"miter\">,\n \"line-miter-limit\": DataConstantProperty,\n \"line-round-limit\": DataConstantProperty,\n \"line-sort-key\": DataDrivenProperty,\n};\n\nexport type LineLayoutPropsPossiblyEvaluated = {\n \"line-cap\": \"butt\" | \"round\" | \"square\",\n \"line-join\": PossiblyEvaluatedPropertyValue<\"bevel\" | \"round\" | \"miter\">,\n \"line-miter-limit\": number,\n \"line-round-limit\": number,\n \"line-sort-key\": PossiblyEvaluatedPropertyValue,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"line-cap\": new DataConstantProperty(styleSpec[\"layout_line\"][\"line-cap\"] as any as StylePropertySpecification),\n \"line-join\": new DataDrivenProperty(styleSpec[\"layout_line\"][\"line-join\"] as any as StylePropertySpecification),\n \"line-miter-limit\": new DataConstantProperty(styleSpec[\"layout_line\"][\"line-miter-limit\"] as any as StylePropertySpecification),\n \"line-round-limit\": new DataConstantProperty(styleSpec[\"layout_line\"][\"line-round-limit\"] as any as StylePropertySpecification),\n \"line-sort-key\": new DataDrivenProperty(styleSpec[\"layout_line\"][\"line-sort-key\"] as any as StylePropertySpecification),\n});\n\nexport type LinePaintProps = {\n \"line-opacity\": DataDrivenProperty,\n \"line-color\": DataDrivenProperty,\n \"line-translate\": DataConstantProperty<[number, number]>,\n \"line-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"line-width\": DataDrivenProperty,\n \"line-gap-width\": DataDrivenProperty,\n \"line-offset\": DataDrivenProperty,\n \"line-blur\": DataDrivenProperty,\n \"line-dasharray\": CrossFadedProperty>,\n \"line-pattern\": CrossFadedDataDrivenProperty,\n \"line-gradient\": ColorRampProperty,\n};\n\nexport type LinePaintPropsPossiblyEvaluated = {\n \"line-opacity\": PossiblyEvaluatedPropertyValue,\n \"line-color\": PossiblyEvaluatedPropertyValue,\n \"line-translate\": [number, number],\n \"line-translate-anchor\": \"map\" | \"viewport\",\n \"line-width\": PossiblyEvaluatedPropertyValue,\n \"line-gap-width\": PossiblyEvaluatedPropertyValue,\n \"line-offset\": PossiblyEvaluatedPropertyValue,\n \"line-blur\": PossiblyEvaluatedPropertyValue,\n \"line-dasharray\": CrossFaded>,\n \"line-pattern\": PossiblyEvaluatedPropertyValue>,\n \"line-gradient\": ColorRampProperty,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"line-opacity\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-opacity\"] as any as StylePropertySpecification),\n \"line-color\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-color\"] as any as StylePropertySpecification),\n \"line-translate\": new DataConstantProperty(styleSpec[\"paint_line\"][\"line-translate\"] as any as StylePropertySpecification),\n \"line-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_line\"][\"line-translate-anchor\"] as any as StylePropertySpecification),\n \"line-width\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-width\"] as any as StylePropertySpecification),\n \"line-gap-width\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-gap-width\"] as any as StylePropertySpecification),\n \"line-offset\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-offset\"] as any as StylePropertySpecification),\n \"line-blur\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-blur\"] as any as StylePropertySpecification),\n \"line-dasharray\": new CrossFadedProperty(styleSpec[\"paint_line\"][\"line-dasharray\"] as any as StylePropertySpecification),\n \"line-pattern\": new CrossFadedDataDrivenProperty(styleSpec[\"paint_line\"][\"line-pattern\"] as any as StylePropertySpecification),\n \"line-gradient\": new ColorRampProperty(styleSpec[\"paint_line\"][\"line-gradient\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import Point from '@mapbox/point-geometry';\n\nimport {StyleLayer} from '../style_layer';\nimport {LineBucket} from '../../data/bucket/line_bucket';\nimport {polygonIntersectsBufferedMultiLine} from '../../util/intersection_tests';\nimport {getMaximumPaintValue, translateDistance, translate, offsetLine} from '../query_utils';\nimport properties, {LineLayoutPropsPossiblyEvaluated, LinePaintPropsPossiblyEvaluated} from './line_style_layer_properties.g';\nimport {extend} from '../../util/util';\nimport {EvaluationParameters} from '../evaluation_parameters';\nimport {Transitionable, Transitioning, Layout, PossiblyEvaluated, DataDrivenProperty} from '../properties';\n\nimport {isZoomExpression, Step} from '@maplibre/maplibre-gl-style-spec';\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Bucket, BucketParameters} from '../../data/bucket';\nimport type {LineLayoutProps, LinePaintProps} from './line_style_layer_properties.g';\nimport type {Transform} from '../../geo/transform';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nexport class LineFloorwidthProperty extends DataDrivenProperty {\n useIntegerZoom: true;\n\n possiblyEvaluate(value, parameters) {\n parameters = new EvaluationParameters(Math.floor(parameters.zoom), {\n now: parameters.now,\n fadeDuration: parameters.fadeDuration,\n zoomHistory: parameters.zoomHistory,\n transition: parameters.transition\n });\n return super.possiblyEvaluate(value, parameters);\n }\n\n evaluate(value, globals, feature, featureState) {\n globals = extend({}, globals, {zoom: Math.floor(globals.zoom)});\n return super.evaluate(value, globals, feature, featureState);\n }\n}\n\nlet lineFloorwidthProperty: LineFloorwidthProperty;\n\nexport class LineStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n gradientVersion: number;\n stepInterpolant: boolean;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n this.gradientVersion = 0;\n if (!lineFloorwidthProperty) {\n lineFloorwidthProperty =\n new LineFloorwidthProperty(properties.paint.properties['line-width'].specification);\n lineFloorwidthProperty.useIntegerZoom = true;\n }\n }\n\n _handleSpecialPaintPropertyUpdate(name: string) {\n if (name === 'line-gradient') {\n const expression = this.gradientExpression();\n if (isZoomExpression(expression)) {\n this.stepInterpolant = expression._styleExpression.expression instanceof Step;\n } else {\n this.stepInterpolant = false;\n }\n this.gradientVersion = (this.gradientVersion + 1) % Number.MAX_SAFE_INTEGER;\n }\n }\n\n gradientExpression() {\n return this._transitionablePaint._values['line-gradient'].value.expression;\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n super.recalculate(parameters, availableImages);\n (this.paint._values as any)['line-floorwidth'] =\n lineFloorwidthProperty.possiblyEvaluate(this._transitioningPaint._values['line-width'].value, parameters);\n }\n\n createBucket(parameters: BucketParameters) {\n return new LineBucket(parameters);\n }\n\n queryRadius(bucket: Bucket): number {\n const lineBucket: LineBucket = (bucket as any);\n const width = getLineWidth(\n getMaximumPaintValue('line-width', this, lineBucket),\n getMaximumPaintValue('line-gap-width', this, lineBucket));\n const offset = getMaximumPaintValue('line-offset', this, lineBucket);\n return width / 2 + Math.abs(offset) + translateDistance(this.paint.get('line-translate'));\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number\n ): boolean {\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('line-translate'),\n this.paint.get('line-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n const halfWidth = pixelsToTileUnits / 2 * getLineWidth(\n this.paint.get('line-width').evaluate(feature, featureState),\n this.paint.get('line-gap-width').evaluate(feature, featureState));\n const lineOffset = this.paint.get('line-offset').evaluate(feature, featureState);\n if (lineOffset) {\n geometry = offsetLine(geometry, lineOffset * pixelsToTileUnits);\n }\n\n return polygonIntersectsBufferedMultiLine(translatedPolygon, geometry, halfWidth);\n }\n\n isTileClipped() {\n return true;\n }\n}\n\nfunction getLineWidth(lineWidth, lineGapWidth) {\n if (lineGapWidth > 0) {\n return lineGapWidth + 2 * lineWidth;\n } else {\n return lineWidth;\n }\n}\n","import {createLayout} from '../../util/struct_array';\n\nexport const symbolLayoutAttributes = createLayout([\n {name: 'a_pos_offset', components: 4, type: 'Int16'},\n {name: 'a_data', components: 4, type: 'Uint16'},\n {name: 'a_pixeloffset', components: 4, type: 'Int16'}\n], 4);\n\nexport const dynamicLayoutAttributes = createLayout([\n {name: 'a_projected_pos', components: 3, type: 'Float32'}\n], 4);\n\nexport const placementOpacityAttributes = createLayout([\n {name: 'a_fade_opacity', components: 1, type: 'Uint32'}\n], 4);\n\nexport const collisionVertexAttributes = createLayout([\n {name: 'a_placed', components: 2, type: 'Uint8'},\n {name: 'a_shift', components: 2, type: 'Float32'}\n]);\n\nexport const collisionBox = createLayout([\n // the box is centered around the anchor point\n {type: 'Int16', name: 'anchorPointX'},\n {type: 'Int16', name: 'anchorPointY'},\n\n // distances to the edges from the anchor\n {type: 'Int16', name: 'x1'},\n {type: 'Int16', name: 'y1'},\n {type: 'Int16', name: 'x2'},\n {type: 'Int16', name: 'y2'},\n\n // the index of the feature in the original vectortile\n {type: 'Uint32', name: 'featureIndex'},\n // the source layer the feature appears in\n {type: 'Uint16', name: 'sourceLayerIndex'},\n // the bucket the feature appears in\n {type: 'Uint16', name: 'bucketIndex'},\n]);\n\nexport const collisionBoxLayout = createLayout([ // used to render collision boxes for debugging purposes\n {name: 'a_pos', components: 2, type: 'Int16'},\n {name: 'a_anchor_pos', components: 2, type: 'Int16'},\n {name: 'a_extrude', components: 2, type: 'Int16'}\n], 4);\n\nexport const collisionCircleLayout = createLayout([ // used to render collision circles for debugging purposes\n {name: 'a_pos', components: 2, type: 'Float32'},\n {name: 'a_radius', components: 1, type: 'Float32'},\n {name: 'a_flags', components: 2, type: 'Int16'}\n], 4);\n\nexport const quadTriangle = createLayout([\n {name: 'triangle', components: 3, type: 'Uint16'},\n]);\n\nexport const placement = createLayout([\n {type: 'Int16', name: 'anchorX'},\n {type: 'Int16', name: 'anchorY'},\n {type: 'Uint16', name: 'glyphStartIndex'},\n {type: 'Uint16', name: 'numGlyphs'},\n {type: 'Uint32', name: 'vertexStartIndex'},\n {type: 'Uint32', name: 'lineStartIndex'},\n {type: 'Uint32', name: 'lineLength'},\n {type: 'Uint16', name: 'segment'},\n {type: 'Uint16', name: 'lowerSize'},\n {type: 'Uint16', name: 'upperSize'},\n {type: 'Float32', name: 'lineOffsetX'},\n {type: 'Float32', name: 'lineOffsetY'},\n {type: 'Uint8', name: 'writingMode'},\n {type: 'Uint8', name: 'placedOrientation'},\n {type: 'Uint8', name: 'hidden'},\n {type: 'Uint32', name: 'crossTileID'},\n {type: 'Int16', name: 'associatedIconIndex'}\n]);\n\nexport const symbolInstance = createLayout([\n {type: 'Int16', name: 'anchorX'},\n {type: 'Int16', name: 'anchorY'},\n {type: 'Int16', name: 'rightJustifiedTextSymbolIndex'},\n {type: 'Int16', name: 'centerJustifiedTextSymbolIndex'},\n {type: 'Int16', name: 'leftJustifiedTextSymbolIndex'},\n {type: 'Int16', name: 'verticalPlacedTextSymbolIndex'},\n {type: 'Int16', name: 'placedIconSymbolIndex'},\n {type: 'Int16', name: 'verticalPlacedIconSymbolIndex'},\n {type: 'Uint16', name: 'key'},\n {type: 'Uint16', name: 'textBoxStartIndex'},\n {type: 'Uint16', name: 'textBoxEndIndex'},\n {type: 'Uint16', name: 'verticalTextBoxStartIndex'},\n {type: 'Uint16', name: 'verticalTextBoxEndIndex'},\n {type: 'Uint16', name: 'iconBoxStartIndex'},\n {type: 'Uint16', name: 'iconBoxEndIndex'},\n {type: 'Uint16', name: 'verticalIconBoxStartIndex'},\n {type: 'Uint16', name: 'verticalIconBoxEndIndex'},\n {type: 'Uint16', name: 'featureIndex'},\n {type: 'Uint16', name: 'numHorizontalGlyphVertices'},\n {type: 'Uint16', name: 'numVerticalGlyphVertices'},\n {type: 'Uint16', name: 'numIconVertices'},\n {type: 'Uint16', name: 'numVerticalIconVertices'},\n {type: 'Uint16', name: 'useRuntimeCollisionCircles'},\n {type: 'Uint32', name: 'crossTileID'},\n {type: 'Float32', name: 'textBoxScale'},\n {type: 'Float32', name: 'collisionCircleDiameter'},\n {type: 'Uint16', name: 'textAnchorOffsetStartIndex'},\n {type: 'Uint16', name: 'textAnchorOffsetEndIndex'}\n]);\n\nexport const glyphOffset = createLayout([\n {type: 'Float32', name: 'offsetX'}\n]);\n\nexport const lineVertex = createLayout([\n {type: 'Int16', name: 'x'},\n {type: 'Int16', name: 'y'},\n {type: 'Int16', name: 'tileUnitDistanceFromAnchor'}\n]);\n\nexport const textAnchorOffset = createLayout([\n {type: 'Uint16', name: 'textAnchor'},\n {type: 'Float32', components: 2, name: 'textOffset'}\n]);\n","import {plugin as rtlTextPlugin} from '../source/rtl_text_plugin';\n\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\nimport type {Feature} from '@maplibre/maplibre-gl-style-spec';\nimport {Formatted} from '@maplibre/maplibre-gl-style-spec';\n\nfunction transformTextInternal(text: string, layer: SymbolStyleLayer, feature: Feature) {\n const transform = layer.layout.get('text-transform').evaluate(feature, {});\n if (transform === 'uppercase') {\n text = text.toLocaleUpperCase();\n } else if (transform === 'lowercase') {\n text = text.toLocaleLowerCase();\n }\n\n if (rtlTextPlugin.applyArabicShaping) {\n text = rtlTextPlugin.applyArabicShaping(text);\n }\n\n return text;\n}\n\nexport function transformText(text: Formatted, layer: SymbolStyleLayer, feature: Feature): Formatted {\n text.sections.forEach(section => {\n section.text = transformTextInternal(section.text, layer, feature);\n });\n return text;\n}\n","import {charHasRotatedVerticalOrientation} from './script_detection';\n\nexport const verticalizedCharacterMap = {\n '!': '︕',\n '#': '#',\n '$': '$',\n '%': '%',\n '&': '&',\n '(': '︵',\n ')': '︶',\n '*': '*',\n '+': '+',\n ',': '︐',\n '-': '︲',\n '.': '・',\n '/': '/',\n ':': '︓',\n ';': '︔',\n '<': '︿',\n '=': '=',\n '>': '﹀',\n '?': '︖',\n '@': '@',\n '[': '﹇',\n '\\\\': '\',\n ']': '﹈',\n '^': '^',\n '_': '︳',\n '`': '`',\n '{': '︷',\n '|': '―',\n '}': '︸',\n '~': '~',\n '¢': '¢',\n '£': '£',\n '¥': '¥',\n '¦': '¦',\n '¬': '¬',\n '¯': ' ̄',\n '–': '︲',\n '—': '︱',\n '‘': '﹃',\n '’': '﹄',\n '“': '﹁',\n '”': '﹂',\n '…': '︙',\n '‧': '・',\n '₩': '₩',\n '、': '︑',\n '。': '︒',\n '〈': '︿',\n '〉': '﹀',\n '《': '︽',\n '》': '︾',\n '「': '﹁',\n '」': '﹂',\n '『': '﹃',\n '』': '﹄',\n '【': '︻',\n '】': '︼',\n '〔': '︹',\n '〕': '︺',\n '〖': '︗',\n '〗': '︘',\n '!': '︕',\n '(': '︵',\n ')': '︶',\n ',': '︐',\n '-': '︲',\n '.': '・',\n ':': '︓',\n ';': '︔',\n '<': '︿',\n '>': '﹀',\n '?': '︖',\n '[': '﹇',\n ']': '﹈',\n '_': '︳',\n '{': '︷',\n '|': '―',\n '}': '︸',\n '⦅': '︵',\n '⦆': '︶',\n '。': '︒',\n '「': '﹁',\n '」': '﹂'\n};\n\nexport function verticalizePunctuation(input: string) {\n let output = '';\n\n for (let i = 0; i < input.length; i++) {\n const nextCharCode = input.charCodeAt(i + 1) || null;\n const prevCharCode = input.charCodeAt(i - 1) || null;\n\n const canReplacePunctuation = (\n (!nextCharCode || !charHasRotatedVerticalOrientation(nextCharCode) || verticalizedCharacterMap[input[i + 1]]) &&\n (!prevCharCode || !charHasRotatedVerticalOrientation(prevCharCode) || verticalizedCharacterMap[input[i - 1]])\n );\n\n if (canReplacePunctuation && verticalizedCharacterMap[input[i]]) {\n output += verticalizedCharacterMap[input[i]];\n } else {\n output += input[i];\n }\n }\n\n return output;\n}\n\n","// ONE_EM constant used to go between \"em\" units used in style spec and \"points\" used internally for layout\n\nexport default 24;\n","'use strict';\n\nmodule.exports = Pbf;\n\nvar ieee754 = require('ieee754');\n\nfunction Pbf(buf) {\n this.buf = ArrayBuffer.isView && ArrayBuffer.isView(buf) ? buf : new Uint8Array(buf || 0);\n this.pos = 0;\n this.type = 0;\n this.length = this.buf.length;\n}\n\nPbf.Varint = 0; // varint: int32, int64, uint32, uint64, sint32, sint64, bool, enum\nPbf.Fixed64 = 1; // 64-bit: double, fixed64, sfixed64\nPbf.Bytes = 2; // length-delimited: string, bytes, embedded messages, packed repeated fields\nPbf.Fixed32 = 5; // 32-bit: float, fixed32, sfixed32\n\nvar SHIFT_LEFT_32 = (1 << 16) * (1 << 16),\n SHIFT_RIGHT_32 = 1 / SHIFT_LEFT_32;\n\n// Threshold chosen based on both benchmarking and knowledge about browser string\n// data structures (which currently switch structure types at 12 bytes or more)\nvar TEXT_DECODER_MIN_LENGTH = 12;\nvar utf8TextDecoder = typeof TextDecoder === 'undefined' ? null : new TextDecoder('utf8');\n\nPbf.prototype = {\n\n destroy: function() {\n this.buf = null;\n },\n\n // === READING =================================================================\n\n readFields: function(readField, result, end) {\n end = end || this.length;\n\n while (this.pos < end) {\n var val = this.readVarint(),\n tag = val >> 3,\n startPos = this.pos;\n\n this.type = val & 0x7;\n readField(tag, result, this);\n\n if (this.pos === startPos) this.skip(val);\n }\n return result;\n },\n\n readMessage: function(readField, result) {\n return this.readFields(readField, result, this.readVarint() + this.pos);\n },\n\n readFixed32: function() {\n var val = readUInt32(this.buf, this.pos);\n this.pos += 4;\n return val;\n },\n\n readSFixed32: function() {\n var val = readInt32(this.buf, this.pos);\n this.pos += 4;\n return val;\n },\n\n // 64-bit int handling is based on github.com/dpw/node-buffer-more-ints (MIT-licensed)\n\n readFixed64: function() {\n var val = readUInt32(this.buf, this.pos) + readUInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;\n this.pos += 8;\n return val;\n },\n\n readSFixed64: function() {\n var val = readUInt32(this.buf, this.pos) + readInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;\n this.pos += 8;\n return val;\n },\n\n readFloat: function() {\n var val = ieee754.read(this.buf, this.pos, true, 23, 4);\n this.pos += 4;\n return val;\n },\n\n readDouble: function() {\n var val = ieee754.read(this.buf, this.pos, true, 52, 8);\n this.pos += 8;\n return val;\n },\n\n readVarint: function(isSigned) {\n var buf = this.buf,\n val, b;\n\n b = buf[this.pos++]; val = b & 0x7f; if (b < 0x80) return val;\n b = buf[this.pos++]; val |= (b & 0x7f) << 7; if (b < 0x80) return val;\n b = buf[this.pos++]; val |= (b & 0x7f) << 14; if (b < 0x80) return val;\n b = buf[this.pos++]; val |= (b & 0x7f) << 21; if (b < 0x80) return val;\n b = buf[this.pos]; val |= (b & 0x0f) << 28;\n\n return readVarintRemainder(val, isSigned, this);\n },\n\n readVarint64: function() { // for compatibility with v2.0.1\n return this.readVarint(true);\n },\n\n readSVarint: function() {\n var num = this.readVarint();\n return num % 2 === 1 ? (num + 1) / -2 : num / 2; // zigzag encoding\n },\n\n readBoolean: function() {\n return Boolean(this.readVarint());\n },\n\n readString: function() {\n var end = this.readVarint() + this.pos;\n var pos = this.pos;\n this.pos = end;\n\n if (end - pos >= TEXT_DECODER_MIN_LENGTH && utf8TextDecoder) {\n // longer strings are fast with the built-in browser TextDecoder API\n return readUtf8TextDecoder(this.buf, pos, end);\n }\n // short strings are fast with our custom implementation\n return readUtf8(this.buf, pos, end);\n },\n\n readBytes: function() {\n var end = this.readVarint() + this.pos,\n buffer = this.buf.subarray(this.pos, end);\n this.pos = end;\n return buffer;\n },\n\n // verbose for performance reasons; doesn't affect gzipped size\n\n readPackedVarint: function(arr, isSigned) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readVarint(isSigned));\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readVarint(isSigned));\n return arr;\n },\n readPackedSVarint: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readSVarint());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readSVarint());\n return arr;\n },\n readPackedBoolean: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readBoolean());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readBoolean());\n return arr;\n },\n readPackedFloat: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readFloat());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readFloat());\n return arr;\n },\n readPackedDouble: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readDouble());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readDouble());\n return arr;\n },\n readPackedFixed32: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readFixed32());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readFixed32());\n return arr;\n },\n readPackedSFixed32: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed32());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readSFixed32());\n return arr;\n },\n readPackedFixed64: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readFixed64());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readFixed64());\n return arr;\n },\n readPackedSFixed64: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed64());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readSFixed64());\n return arr;\n },\n\n skip: function(val) {\n var type = val & 0x7;\n if (type === Pbf.Varint) while (this.buf[this.pos++] > 0x7f) {}\n else if (type === Pbf.Bytes) this.pos = this.readVarint() + this.pos;\n else if (type === Pbf.Fixed32) this.pos += 4;\n else if (type === Pbf.Fixed64) this.pos += 8;\n else throw new Error('Unimplemented type: ' + type);\n },\n\n // === WRITING =================================================================\n\n writeTag: function(tag, type) {\n this.writeVarint((tag << 3) | type);\n },\n\n realloc: function(min) {\n var length = this.length || 16;\n\n while (length < this.pos + min) length *= 2;\n\n if (length !== this.length) {\n var buf = new Uint8Array(length);\n buf.set(this.buf);\n this.buf = buf;\n this.length = length;\n }\n },\n\n finish: function() {\n this.length = this.pos;\n this.pos = 0;\n return this.buf.subarray(0, this.length);\n },\n\n writeFixed32: function(val) {\n this.realloc(4);\n writeInt32(this.buf, val, this.pos);\n this.pos += 4;\n },\n\n writeSFixed32: function(val) {\n this.realloc(4);\n writeInt32(this.buf, val, this.pos);\n this.pos += 4;\n },\n\n writeFixed64: function(val) {\n this.realloc(8);\n writeInt32(this.buf, val & -1, this.pos);\n writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4);\n this.pos += 8;\n },\n\n writeSFixed64: function(val) {\n this.realloc(8);\n writeInt32(this.buf, val & -1, this.pos);\n writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4);\n this.pos += 8;\n },\n\n writeVarint: function(val) {\n val = +val || 0;\n\n if (val > 0xfffffff || val < 0) {\n writeBigVarint(val, this);\n return;\n }\n\n this.realloc(4);\n\n this.buf[this.pos++] = val & 0x7f | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return;\n this.buf[this.pos++] = ((val >>>= 7) & 0x7f) | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return;\n this.buf[this.pos++] = ((val >>>= 7) & 0x7f) | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return;\n this.buf[this.pos++] = (val >>> 7) & 0x7f;\n },\n\n writeSVarint: function(val) {\n this.writeVarint(val < 0 ? -val * 2 - 1 : val * 2);\n },\n\n writeBoolean: function(val) {\n this.writeVarint(Boolean(val));\n },\n\n writeString: function(str) {\n str = String(str);\n this.realloc(str.length * 4);\n\n this.pos++; // reserve 1 byte for short string length\n\n var startPos = this.pos;\n // write the string directly to the buffer and see how much was written\n this.pos = writeUtf8(this.buf, str, this.pos);\n var len = this.pos - startPos;\n\n if (len >= 0x80) makeRoomForExtraLength(startPos, len, this);\n\n // finally, write the message length in the reserved place and restore the position\n this.pos = startPos - 1;\n this.writeVarint(len);\n this.pos += len;\n },\n\n writeFloat: function(val) {\n this.realloc(4);\n ieee754.write(this.buf, val, this.pos, true, 23, 4);\n this.pos += 4;\n },\n\n writeDouble: function(val) {\n this.realloc(8);\n ieee754.write(this.buf, val, this.pos, true, 52, 8);\n this.pos += 8;\n },\n\n writeBytes: function(buffer) {\n var len = buffer.length;\n this.writeVarint(len);\n this.realloc(len);\n for (var i = 0; i < len; i++) this.buf[this.pos++] = buffer[i];\n },\n\n writeRawMessage: function(fn, obj) {\n this.pos++; // reserve 1 byte for short message length\n\n // write the message directly to the buffer and see how much was written\n var startPos = this.pos;\n fn(obj, this);\n var len = this.pos - startPos;\n\n if (len >= 0x80) makeRoomForExtraLength(startPos, len, this);\n\n // finally, write the message length in the reserved place and restore the position\n this.pos = startPos - 1;\n this.writeVarint(len);\n this.pos += len;\n },\n\n writeMessage: function(tag, fn, obj) {\n this.writeTag(tag, Pbf.Bytes);\n this.writeRawMessage(fn, obj);\n },\n\n writePackedVarint: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedVarint, arr); },\n writePackedSVarint: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSVarint, arr); },\n writePackedBoolean: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedBoolean, arr); },\n writePackedFloat: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFloat, arr); },\n writePackedDouble: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedDouble, arr); },\n writePackedFixed32: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFixed32, arr); },\n writePackedSFixed32: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSFixed32, arr); },\n writePackedFixed64: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFixed64, arr); },\n writePackedSFixed64: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSFixed64, arr); },\n\n writeBytesField: function(tag, buffer) {\n this.writeTag(tag, Pbf.Bytes);\n this.writeBytes(buffer);\n },\n writeFixed32Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed32);\n this.writeFixed32(val);\n },\n writeSFixed32Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed32);\n this.writeSFixed32(val);\n },\n writeFixed64Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed64);\n this.writeFixed64(val);\n },\n writeSFixed64Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed64);\n this.writeSFixed64(val);\n },\n writeVarintField: function(tag, val) {\n this.writeTag(tag, Pbf.Varint);\n this.writeVarint(val);\n },\n writeSVarintField: function(tag, val) {\n this.writeTag(tag, Pbf.Varint);\n this.writeSVarint(val);\n },\n writeStringField: function(tag, str) {\n this.writeTag(tag, Pbf.Bytes);\n this.writeString(str);\n },\n writeFloatField: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed32);\n this.writeFloat(val);\n },\n writeDoubleField: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed64);\n this.writeDouble(val);\n },\n writeBooleanField: function(tag, val) {\n this.writeVarintField(tag, Boolean(val));\n }\n};\n\nfunction readVarintRemainder(l, s, p) {\n var buf = p.buf,\n h, b;\n\n b = buf[p.pos++]; h = (b & 0x70) >> 4; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 3; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 10; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 17; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 24; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x01) << 31; if (b < 0x80) return toNum(l, h, s);\n\n throw new Error('Expected varint not more than 10 bytes');\n}\n\nfunction readPackedEnd(pbf) {\n return pbf.type === Pbf.Bytes ?\n pbf.readVarint() + pbf.pos : pbf.pos + 1;\n}\n\nfunction toNum(low, high, isSigned) {\n if (isSigned) {\n return high * 0x100000000 + (low >>> 0);\n }\n\n return ((high >>> 0) * 0x100000000) + (low >>> 0);\n}\n\nfunction writeBigVarint(val, pbf) {\n var low, high;\n\n if (val >= 0) {\n low = (val % 0x100000000) | 0;\n high = (val / 0x100000000) | 0;\n } else {\n low = ~(-val % 0x100000000);\n high = ~(-val / 0x100000000);\n\n if (low ^ 0xffffffff) {\n low = (low + 1) | 0;\n } else {\n low = 0;\n high = (high + 1) | 0;\n }\n }\n\n if (val >= 0x10000000000000000 || val < -0x10000000000000000) {\n throw new Error('Given varint doesn\\'t fit into 10 bytes');\n }\n\n pbf.realloc(10);\n\n writeBigVarintLow(low, high, pbf);\n writeBigVarintHigh(high, pbf);\n}\n\nfunction writeBigVarintLow(low, high, pbf) {\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos] = low & 0x7f;\n}\n\nfunction writeBigVarintHigh(high, pbf) {\n var lsb = (high & 0x07) << 4;\n\n pbf.buf[pbf.pos++] |= lsb | ((high >>>= 3) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f;\n}\n\nfunction makeRoomForExtraLength(startPos, len, pbf) {\n var extraLen =\n len <= 0x3fff ? 1 :\n len <= 0x1fffff ? 2 :\n len <= 0xfffffff ? 3 : Math.floor(Math.log(len) / (Math.LN2 * 7));\n\n // if 1 byte isn't enough for encoding message length, shift the data to the right\n pbf.realloc(extraLen);\n for (var i = pbf.pos - 1; i >= startPos; i--) pbf.buf[i + extraLen] = pbf.buf[i];\n}\n\nfunction writePackedVarint(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeVarint(arr[i]); }\nfunction writePackedSVarint(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSVarint(arr[i]); }\nfunction writePackedFloat(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFloat(arr[i]); }\nfunction writePackedDouble(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeDouble(arr[i]); }\nfunction writePackedBoolean(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeBoolean(arr[i]); }\nfunction writePackedFixed32(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFixed32(arr[i]); }\nfunction writePackedSFixed32(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSFixed32(arr[i]); }\nfunction writePackedFixed64(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFixed64(arr[i]); }\nfunction writePackedSFixed64(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSFixed64(arr[i]); }\n\n// Buffer code below from https://github.com/feross/buffer, MIT-licensed\n\nfunction readUInt32(buf, pos) {\n return ((buf[pos]) |\n (buf[pos + 1] << 8) |\n (buf[pos + 2] << 16)) +\n (buf[pos + 3] * 0x1000000);\n}\n\nfunction writeInt32(buf, val, pos) {\n buf[pos] = val;\n buf[pos + 1] = (val >>> 8);\n buf[pos + 2] = (val >>> 16);\n buf[pos + 3] = (val >>> 24);\n}\n\nfunction readInt32(buf, pos) {\n return ((buf[pos]) |\n (buf[pos + 1] << 8) |\n (buf[pos + 2] << 16)) +\n (buf[pos + 3] << 24);\n}\n\nfunction readUtf8(buf, pos, end) {\n var str = '';\n var i = pos;\n\n while (i < end) {\n var b0 = buf[i];\n var c = null; // codepoint\n var bytesPerSequence =\n b0 > 0xEF ? 4 :\n b0 > 0xDF ? 3 :\n b0 > 0xBF ? 2 : 1;\n\n if (i + bytesPerSequence > end) break;\n\n var b1, b2, b3;\n\n if (bytesPerSequence === 1) {\n if (b0 < 0x80) {\n c = b0;\n }\n } else if (bytesPerSequence === 2) {\n b1 = buf[i + 1];\n if ((b1 & 0xC0) === 0x80) {\n c = (b0 & 0x1F) << 0x6 | (b1 & 0x3F);\n if (c <= 0x7F) {\n c = null;\n }\n }\n } else if (bytesPerSequence === 3) {\n b1 = buf[i + 1];\n b2 = buf[i + 2];\n if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80) {\n c = (b0 & 0xF) << 0xC | (b1 & 0x3F) << 0x6 | (b2 & 0x3F);\n if (c <= 0x7FF || (c >= 0xD800 && c <= 0xDFFF)) {\n c = null;\n }\n }\n } else if (bytesPerSequence === 4) {\n b1 = buf[i + 1];\n b2 = buf[i + 2];\n b3 = buf[i + 3];\n if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80 && (b3 & 0xC0) === 0x80) {\n c = (b0 & 0xF) << 0x12 | (b1 & 0x3F) << 0xC | (b2 & 0x3F) << 0x6 | (b3 & 0x3F);\n if (c <= 0xFFFF || c >= 0x110000) {\n c = null;\n }\n }\n }\n\n if (c === null) {\n c = 0xFFFD;\n bytesPerSequence = 1;\n\n } else if (c > 0xFFFF) {\n c -= 0x10000;\n str += String.fromCharCode(c >>> 10 & 0x3FF | 0xD800);\n c = 0xDC00 | c & 0x3FF;\n }\n\n str += String.fromCharCode(c);\n i += bytesPerSequence;\n }\n\n return str;\n}\n\nfunction readUtf8TextDecoder(buf, pos, end) {\n return utf8TextDecoder.decode(buf.subarray(pos, end));\n}\n\nfunction writeUtf8(buf, str, pos) {\n for (var i = 0, c, lead; i < str.length; i++) {\n c = str.charCodeAt(i); // code point\n\n if (c > 0xD7FF && c < 0xE000) {\n if (lead) {\n if (c < 0xDC00) {\n buf[pos++] = 0xEF;\n buf[pos++] = 0xBF;\n buf[pos++] = 0xBD;\n lead = c;\n continue;\n } else {\n c = lead - 0xD800 << 10 | c - 0xDC00 | 0x10000;\n lead = null;\n }\n } else {\n if (c > 0xDBFF || (i + 1 === str.length)) {\n buf[pos++] = 0xEF;\n buf[pos++] = 0xBF;\n buf[pos++] = 0xBD;\n } else {\n lead = c;\n }\n continue;\n }\n } else if (lead) {\n buf[pos++] = 0xEF;\n buf[pos++] = 0xBF;\n buf[pos++] = 0xBD;\n lead = null;\n }\n\n if (c < 0x80) {\n buf[pos++] = c;\n } else {\n if (c < 0x800) {\n buf[pos++] = c >> 0x6 | 0xC0;\n } else {\n if (c < 0x10000) {\n buf[pos++] = c >> 0xC | 0xE0;\n } else {\n buf[pos++] = c >> 0x12 | 0xF0;\n buf[pos++] = c >> 0xC & 0x3F | 0x80;\n }\n buf[pos++] = c >> 0x6 & 0x3F | 0x80;\n }\n buf[pos++] = c & 0x3F | 0x80;\n }\n }\n return pos;\n}\n","/*! ieee754. BSD-3-Clause License. Feross Aboukhadijeh */\nexports.read = function (buffer, offset, isLE, mLen, nBytes) {\n var e, m\n var eLen = (nBytes * 8) - mLen - 1\n var eMax = (1 << eLen) - 1\n var eBias = eMax >> 1\n var nBits = -7\n var i = isLE ? (nBytes - 1) : 0\n var d = isLE ? -1 : 1\n var s = buffer[offset + i]\n\n i += d\n\n e = s & ((1 << (-nBits)) - 1)\n s >>= (-nBits)\n nBits += eLen\n for (; nBits > 0; e = (e * 256) + buffer[offset + i], i += d, nBits -= 8) {}\n\n m = e & ((1 << (-nBits)) - 1)\n e >>= (-nBits)\n nBits += mLen\n for (; nBits > 0; m = (m * 256) + buffer[offset + i], i += d, nBits -= 8) {}\n\n if (e === 0) {\n e = 1 - eBias\n } else if (e === eMax) {\n return m ? NaN : ((s ? -1 : 1) * Infinity)\n } else {\n m = m + Math.pow(2, mLen)\n e = e - eBias\n }\n return (s ? -1 : 1) * m * Math.pow(2, e - mLen)\n}\n\nexports.write = function (buffer, value, offset, isLE, mLen, nBytes) {\n var e, m, c\n var eLen = (nBytes * 8) - mLen - 1\n var eMax = (1 << eLen) - 1\n var eBias = eMax >> 1\n var rt = (mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0)\n var i = isLE ? 0 : (nBytes - 1)\n var d = isLE ? 1 : -1\n var s = value < 0 || (value === 0 && 1 / value < 0) ? 1 : 0\n\n value = Math.abs(value)\n\n if (isNaN(value) || value === Infinity) {\n m = isNaN(value) ? 1 : 0\n e = eMax\n } else {\n e = Math.floor(Math.log(value) / Math.LN2)\n if (value * (c = Math.pow(2, -e)) < 1) {\n e--\n c *= 2\n }\n if (e + eBias >= 1) {\n value += rt / c\n } else {\n value += rt * Math.pow(2, 1 - eBias)\n }\n if (value * c >= 2) {\n e++\n c /= 2\n }\n\n if (e + eBias >= eMax) {\n m = 0\n e = eMax\n } else if (e + eBias >= 1) {\n m = ((value * c) - 1) * Math.pow(2, mLen)\n e = e + eBias\n } else {\n m = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen)\n e = 0\n }\n }\n\n for (; mLen >= 8; buffer[offset + i] = m & 0xff, i += d, m /= 256, mLen -= 8) {}\n\n e = (e << mLen) | m\n eLen += mLen\n for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8) {}\n\n buffer[offset + i - d] |= s * 128\n}\n","import {AlphaImage} from '../util/image';\n\nimport Protobuf from 'pbf';\nconst border = 3;\n\nimport type {StyleGlyph} from './style_glyph';\n\nfunction readFontstacks(tag: number, glyphs: Array, pbf: Protobuf) {\n if (tag === 1) {\n pbf.readMessage(readFontstack, glyphs);\n }\n}\n\nfunction readFontstack(tag: number, glyphs: Array, pbf: Protobuf) {\n if (tag === 3) {\n const {id, bitmap, width, height, left, top, advance} = pbf.readMessage(readGlyph, {});\n glyphs.push({\n id,\n bitmap: new AlphaImage({\n width: width + 2 * border,\n height: height + 2 * border\n }, bitmap),\n metrics: {width, height, left, top, advance}\n });\n }\n}\n\nfunction readGlyph(tag: number, glyph: any, pbf: Protobuf) {\n if (tag === 1) glyph.id = pbf.readVarint();\n else if (tag === 2) glyph.bitmap = pbf.readBytes();\n else if (tag === 3) glyph.width = pbf.readVarint();\n else if (tag === 4) glyph.height = pbf.readVarint();\n else if (tag === 5) glyph.left = pbf.readSVarint();\n else if (tag === 6) glyph.top = pbf.readSVarint();\n else if (tag === 7) glyph.advance = pbf.readVarint();\n}\n\nexport function parseGlyphPbf(data: ArrayBuffer | Uint8Array): Array {\n return new Protobuf(data).readFields(readFontstacks, []);\n}\n\nexport const GLYPH_PBF_BORDER = border;\n","\nexport default function potpack(boxes) {\n\n // calculate total box area and maximum box width\n let area = 0;\n let maxWidth = 0;\n\n for (const box of boxes) {\n area += box.w * box.h;\n maxWidth = Math.max(maxWidth, box.w);\n }\n\n // sort the boxes for insertion by height, descending\n boxes.sort((a, b) => b.h - a.h);\n\n // aim for a squarish resulting container,\n // slightly adjusted for sub-100% space utilization\n const startWidth = Math.max(Math.ceil(Math.sqrt(area / 0.95)), maxWidth);\n\n // start with a single empty space, unbounded at the bottom\n const spaces = [{x: 0, y: 0, w: startWidth, h: Infinity}];\n\n let width = 0;\n let height = 0;\n\n for (const box of boxes) {\n // look through spaces backwards so that we check smaller spaces first\n for (let i = spaces.length - 1; i >= 0; i--) {\n const space = spaces[i];\n\n // look for empty spaces that can accommodate the current box\n if (box.w > space.w || box.h > space.h) continue;\n\n // found the space; add the box to its top-left corner\n // |-------|-------|\n // | box | |\n // |_______| |\n // | space |\n // |_______________|\n box.x = space.x;\n box.y = space.y;\n\n height = Math.max(height, box.y + box.h);\n width = Math.max(width, box.x + box.w);\n\n if (box.w === space.w && box.h === space.h) {\n // space matches the box exactly; remove it\n const last = spaces.pop();\n if (i < spaces.length) spaces[i] = last;\n\n } else if (box.h === space.h) {\n // space matches the box height; update it accordingly\n // |-------|---------------|\n // | box | updated space |\n // |_______|_______________|\n space.x += box.w;\n space.w -= box.w;\n\n } else if (box.w === space.w) {\n // space matches the box width; update it accordingly\n // |---------------|\n // | box |\n // |_______________|\n // | updated space |\n // |_______________|\n space.y += box.h;\n space.h -= box.h;\n\n } else {\n // otherwise the box splits the space into two spaces\n // |-------|-----------|\n // | box | new space |\n // |_______|___________|\n // | updated space |\n // |___________________|\n spaces.push({\n x: space.x + box.w,\n y: space.y,\n w: space.w - box.w,\n h: box.h\n });\n space.y += box.h;\n space.h -= box.h;\n }\n break;\n }\n }\n\n return {\n w: width, // container width\n h: height, // container height\n fill: (area / (width * height)) || 0 // space utilization\n };\n}\n","/* eslint-disable key-spacing */\nimport {RGBAImage} from '../util/image';\nimport {register} from '../util/web_worker_transfer';\nimport potpack from 'potpack';\n\nimport type {StyleImage} from '../style/style_image';\nimport type {ImageManager} from './image_manager';\nimport type {Texture} from './texture';\nimport type {Rect} from './glyph_atlas';\n\nconst IMAGE_PADDING: number = 1;\nexport {IMAGE_PADDING};\n\nexport class ImagePosition {\n paddedRect: Rect;\n pixelRatio: number;\n version: number;\n stretchY: Array<[number, number]>;\n stretchX: Array<[number, number]>;\n content: [number, number, number, number];\n\n constructor(paddedRect: Rect, {\n pixelRatio,\n version,\n stretchX,\n stretchY,\n content\n }: StyleImage) {\n this.paddedRect = paddedRect;\n this.pixelRatio = pixelRatio;\n this.stretchX = stretchX;\n this.stretchY = stretchY;\n this.content = content;\n this.version = version;\n }\n\n get tl(): [number, number] {\n return [\n this.paddedRect.x + IMAGE_PADDING,\n this.paddedRect.y + IMAGE_PADDING\n ];\n }\n\n get br(): [number, number] {\n return [\n this.paddedRect.x + this.paddedRect.w - IMAGE_PADDING,\n this.paddedRect.y + this.paddedRect.h - IMAGE_PADDING\n ];\n }\n\n get tlbr(): Array {\n return this.tl.concat(this.br);\n }\n\n get displaySize(): [number, number] {\n return [\n (this.paddedRect.w - IMAGE_PADDING * 2) / this.pixelRatio,\n (this.paddedRect.h - IMAGE_PADDING * 2) / this.pixelRatio\n ];\n }\n}\n\n/**\n * @internal\n * A class holding all the images\n */\nexport class ImageAtlas {\n image: RGBAImage;\n iconPositions: {[_: string]: ImagePosition};\n patternPositions: {[_: string]: ImagePosition};\n haveRenderCallbacks: Array;\n uploaded: boolean;\n\n constructor(icons: {[_: string]: StyleImage}, patterns: {[_: string]: StyleImage}) {\n const iconPositions = {}, patternPositions = {};\n this.haveRenderCallbacks = [];\n\n const bins = [];\n\n this.addImages(icons, iconPositions, bins);\n this.addImages(patterns, patternPositions, bins);\n\n const {w, h} = potpack(bins);\n const image = new RGBAImage({width: w || 1, height: h || 1});\n\n for (const id in icons) {\n const src = icons[id];\n const bin = iconPositions[id].paddedRect;\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x: bin.x + IMAGE_PADDING, y: bin.y + IMAGE_PADDING}, src.data);\n }\n\n for (const id in patterns) {\n const src = patterns[id];\n const bin = patternPositions[id].paddedRect;\n const x = bin.x + IMAGE_PADDING,\n y = bin.y + IMAGE_PADDING,\n w = src.data.width,\n h = src.data.height;\n\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x, y}, src.data);\n // Add 1 pixel wrapped padding on each side of the image.\n RGBAImage.copy(src.data, image, {x: 0, y: h - 1}, {x, y: y - 1}, {width: w, height: 1}); // T\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x, y: y + h}, {width: w, height: 1}); // B\n RGBAImage.copy(src.data, image, {x: w - 1, y: 0}, {x: x - 1, y}, {width: 1, height: h}); // L\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x: x + w, y}, {width: 1, height: h}); // R\n }\n\n this.image = image;\n this.iconPositions = iconPositions;\n this.patternPositions = patternPositions;\n }\n\n addImages(images: {[_: string]: StyleImage}, positions: {[_: string]: ImagePosition}, bins: Array) {\n for (const id in images) {\n const src = images[id];\n const bin = {\n x: 0,\n y: 0,\n w: src.data.width + 2 * IMAGE_PADDING,\n h: src.data.height + 2 * IMAGE_PADDING,\n };\n bins.push(bin);\n positions[id] = new ImagePosition(bin, src);\n\n if (src.hasRenderCallback) {\n this.haveRenderCallbacks.push(id);\n }\n }\n }\n\n patchUpdatedImages(imageManager: ImageManager, texture: Texture) {\n imageManager.dispatchRenderCallbacks(this.haveRenderCallbacks);\n for (const name in imageManager.updatedImages) {\n this.patchUpdatedImage(this.iconPositions[name], imageManager.getImage(name), texture);\n this.patchUpdatedImage(this.patternPositions[name], imageManager.getImage(name), texture);\n }\n }\n\n patchUpdatedImage(position: ImagePosition, image: StyleImage, texture: Texture) {\n if (!position || !image) return;\n\n if (position.version === image.version) return;\n\n position.version = image.version;\n const [x, y] = position.tl;\n texture.update(image.data, undefined, {x, y});\n }\n\n}\n\nregister('ImagePosition', ImagePosition);\nregister('ImageAtlas', ImageAtlas);\n","import {\n charHasUprightVerticalOrientation,\n charAllowsIdeographicBreaking,\n charInComplexShapingScript\n} from '../util/script_detection';\nimport {verticalizePunctuation} from '../util/verticalize_punctuation';\nimport {plugin as rtlTextPlugin} from '../source/rtl_text_plugin';\nimport ONE_EM from './one_em';\nimport {warnOnce} from '../util/util';\n\nimport type {StyleGlyph, GlyphMetrics} from '../style/style_glyph';\nimport {GLYPH_PBF_BORDER} from '../style/parse_glyph_pbf';\nimport type {ImagePosition} from '../render/image_atlas';\nimport {IMAGE_PADDING} from '../render/image_atlas';\nimport type {Rect, GlyphPosition} from '../render/glyph_atlas';\nimport {Formatted, FormattedSection} from '@maplibre/maplibre-gl-style-spec';\n\nenum WritingMode {\n none = 0,\n horizontal = 1,\n vertical = 2,\n horizontalOnly = 3\n}\n\nconst SHAPING_DEFAULT_OFFSET = -17;\nexport {shapeText, shapeIcon, fitIconToText, getAnchorAlignment, WritingMode, SHAPING_DEFAULT_OFFSET};\n\n// The position of a glyph relative to the text's anchor point.\nexport type PositionedGlyph = {\n glyph: number;\n imageName: string | null;\n x: number;\n y: number;\n vertical: boolean;\n scale: number;\n fontStack: string;\n sectionIndex: number;\n metrics: GlyphMetrics;\n rect: Rect | null;\n};\n\nexport type PositionedLine = {\n positionedGlyphs: Array;\n lineOffset: number;\n};\n\n// A collection of positioned glyphs and some metadata\nexport type Shaping = {\n positionedLines: Array;\n top: number;\n bottom: number;\n left: number;\n right: number;\n writingMode: WritingMode.horizontal | WritingMode.vertical;\n text: string;\n iconsInText: boolean;\n verticalizable: boolean;\n};\n\nfunction isEmpty(positionedLines: Array) {\n for (const line of positionedLines) {\n if (line.positionedGlyphs.length !== 0) {\n return false;\n }\n }\n return true;\n}\n\nexport type SymbolAnchor = 'center' | 'left' | 'right' | 'top' | 'bottom' | 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right';\nexport type TextJustify = 'left' | 'center' | 'right';\n\n// Max number of images in label is 6401 U+E000–U+F8FF that covers\n// Basic Multilingual Plane Unicode Private Use Area (PUA).\nconst PUAbegin = 0xE000;\nconst PUAend = 0xF8FF;\n\nclass SectionOptions {\n // Text options\n scale: number;\n fontStack: string;\n // Image options\n imageName: string | null;\n\n constructor() {\n this.scale = 1.0;\n this.fontStack = '';\n this.imageName = null;\n }\n\n static forText(scale: number | null, fontStack: string) {\n const textOptions = new SectionOptions();\n textOptions.scale = scale || 1;\n textOptions.fontStack = fontStack;\n return textOptions;\n }\n\n static forImage(imageName: string) {\n const imageOptions = new SectionOptions();\n imageOptions.imageName = imageName;\n return imageOptions;\n }\n\n}\n\nclass TaggedString {\n text: string;\n sectionIndex: Array; // maps each character in 'text' to its corresponding entry in 'sections'\n sections: Array;\n imageSectionID: number | null;\n\n constructor() {\n this.text = '';\n this.sectionIndex = [];\n this.sections = [];\n this.imageSectionID = null;\n }\n\n static fromFeature(text: Formatted, defaultFontStack: string) {\n const result = new TaggedString();\n for (let i = 0; i < text.sections.length; i++) {\n const section = text.sections[i];\n if (!section.image) {\n result.addTextSection(section, defaultFontStack);\n } else {\n result.addImageSection(section);\n }\n }\n return result;\n }\n\n length(): number {\n return this.text.length;\n }\n\n getSection(index: number): SectionOptions {\n return this.sections[this.sectionIndex[index]];\n }\n\n getSectionIndex(index: number): number {\n return this.sectionIndex[index];\n }\n\n getCharCode(index: number): number {\n return this.text.charCodeAt(index);\n }\n\n verticalizePunctuation() {\n this.text = verticalizePunctuation(this.text);\n }\n\n trim() {\n let beginningWhitespace = 0;\n for (let i = 0;\n i < this.text.length && whitespace[this.text.charCodeAt(i)];\n i++) {\n beginningWhitespace++;\n }\n let trailingWhitespace = this.text.length;\n for (let i = this.text.length - 1;\n i >= 0 && i >= beginningWhitespace && whitespace[this.text.charCodeAt(i)];\n i--) {\n trailingWhitespace--;\n }\n this.text = this.text.substring(beginningWhitespace, trailingWhitespace);\n this.sectionIndex = this.sectionIndex.slice(beginningWhitespace, trailingWhitespace);\n }\n\n substring(start: number, end: number): TaggedString {\n const substring = new TaggedString();\n substring.text = this.text.substring(start, end);\n substring.sectionIndex = this.sectionIndex.slice(start, end);\n substring.sections = this.sections;\n return substring;\n }\n\n toString(): string {\n return this.text;\n }\n\n getMaxScale() {\n return this.sectionIndex.reduce((max, index) => Math.max(max, this.sections[index].scale), 0);\n }\n\n addTextSection(section: FormattedSection, defaultFontStack: string) {\n this.text += section.text;\n this.sections.push(SectionOptions.forText(section.scale, section.fontStack || defaultFontStack));\n const index = this.sections.length - 1;\n for (let i = 0; i < section.text.length; ++i) {\n this.sectionIndex.push(index);\n }\n }\n\n addImageSection(section: FormattedSection) {\n const imageName = section.image ? section.image.name : '';\n if (imageName.length === 0) {\n warnOnce('Can\\'t add FormattedSection with an empty image.');\n return;\n }\n\n const nextImageSectionCharCode = this.getNextImageSectionCharCode();\n if (!nextImageSectionCharCode) {\n warnOnce(`Reached maximum number of images ${PUAend - PUAbegin + 2}`);\n return;\n }\n\n this.text += String.fromCharCode(nextImageSectionCharCode);\n this.sections.push(SectionOptions.forImage(imageName));\n this.sectionIndex.push(this.sections.length - 1);\n }\n\n getNextImageSectionCharCode(): number | null {\n if (!this.imageSectionID) {\n this.imageSectionID = PUAbegin;\n return this.imageSectionID;\n }\n\n if (this.imageSectionID >= PUAend) return null;\n return ++this.imageSectionID;\n }\n}\n\nfunction breakLines(input: TaggedString, lineBreakPoints: Array): Array {\n const lines = [];\n const text = input.text;\n let start = 0;\n for (const lineBreak of lineBreakPoints) {\n lines.push(input.substring(start, lineBreak));\n start = lineBreak;\n }\n\n if (start < text.length) {\n lines.push(input.substring(start, text.length));\n }\n return lines;\n}\n\nfunction shapeText(\n text: Formatted,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n glyphPositions: {\n [_: string]: {\n [_: number]: GlyphPosition;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n defaultFontStack: string,\n maxWidth: number,\n lineHeight: number,\n textAnchor: SymbolAnchor,\n textJustify: TextJustify,\n spacing: number,\n translate: [number, number],\n writingMode: WritingMode.horizontal | WritingMode.vertical,\n allowVerticalPlacement: boolean,\n symbolPlacement: string,\n layoutTextSize: number,\n layoutTextSizeThisZoom: number\n): Shaping | false {\n const logicalInput = TaggedString.fromFeature(text, defaultFontStack);\n\n if (writingMode === WritingMode.vertical) {\n logicalInput.verticalizePunctuation();\n }\n\n let lines: Array;\n\n const {processBidirectionalText, processStyledBidirectionalText} = rtlTextPlugin;\n if (processBidirectionalText && logicalInput.sections.length === 1) {\n // Bidi doesn't have to be style-aware\n lines = [];\n const untaggedLines =\n processBidirectionalText(logicalInput.toString(),\n determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize));\n for (const line of untaggedLines) {\n const taggedLine = new TaggedString();\n taggedLine.text = line;\n taggedLine.sections = logicalInput.sections;\n for (let i = 0; i < line.length; i++) {\n taggedLine.sectionIndex.push(0);\n }\n lines.push(taggedLine);\n }\n } else if (processStyledBidirectionalText) {\n // Need version of mapbox-gl-rtl-text with style support for combining RTL text\n // with formatting\n lines = [];\n const processedLines =\n processStyledBidirectionalText(logicalInput.text,\n logicalInput.sectionIndex,\n determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize));\n for (const line of processedLines) {\n const taggedLine = new TaggedString();\n taggedLine.text = line[0];\n taggedLine.sectionIndex = line[1];\n taggedLine.sections = logicalInput.sections;\n lines.push(taggedLine);\n }\n } else {\n lines = breakLines(logicalInput, determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize));\n }\n\n const positionedLines = [];\n const shaping = {\n positionedLines,\n text: logicalInput.toString(),\n top: translate[1],\n bottom: translate[1],\n left: translate[0],\n right: translate[0],\n writingMode,\n iconsInText: false,\n verticalizable: false\n };\n\n shapeLines(shaping, glyphMap, glyphPositions, imagePositions, lines, lineHeight, textAnchor, textJustify, writingMode, spacing, allowVerticalPlacement, layoutTextSizeThisZoom);\n if (isEmpty(positionedLines)) return false;\n\n return shaping;\n}\n\n// using computed properties due to https://github.com/facebook/flow/issues/380\n/* eslint no-useless-computed-key: 0 */\n\nconst whitespace: {\n [_: number]: boolean;\n} = {\n [0x09]: true, // tab\n [0x0a]: true, // newline\n [0x0b]: true, // vertical tab\n [0x0c]: true, // form feed\n [0x0d]: true, // carriage return\n [0x20]: true, // space\n};\n\nconst breakable: {\n [_: number]: boolean;\n} = {\n [0x0a]: true, // newline\n [0x20]: true, // space\n [0x26]: true, // ampersand\n [0x28]: true, // left parenthesis\n [0x29]: true, // right parenthesis\n [0x2b]: true, // plus sign\n [0x2d]: true, // hyphen-minus\n [0x2f]: true, // solidus\n [0xad]: true, // soft hyphen\n [0xb7]: true, // middle dot\n [0x200b]: true, // zero-width space\n [0x2010]: true, // hyphen\n [0x2013]: true, // en dash\n [0x2027]: true // interpunct\n // Many other characters may be reasonable breakpoints\n // Consider \"neutral orientation\" characters at scriptDetection.charHasNeutralVerticalOrientation\n // See https://github.com/mapbox/mapbox-gl-js/issues/3658\n};\n\nfunction getGlyphAdvance(\n codePoint: number,\n section: SectionOptions,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n spacing: number,\n layoutTextSize: number\n): number {\n if (!section.imageName) {\n const positions = glyphMap[section.fontStack];\n const glyph = positions && positions[codePoint];\n if (!glyph) return 0;\n return glyph.metrics.advance * section.scale + spacing;\n } else {\n const imagePosition = imagePositions[section.imageName];\n if (!imagePosition) return 0;\n return imagePosition.displaySize[0] * section.scale * ONE_EM / layoutTextSize + spacing;\n }\n}\n\nfunction determineAverageLineWidth(logicalInput: TaggedString,\n spacing: number,\n maxWidth: number,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n layoutTextSize: number) {\n let totalWidth = 0;\n\n for (let index = 0; index < logicalInput.length(); index++) {\n const section = logicalInput.getSection(index);\n totalWidth += getGlyphAdvance(logicalInput.getCharCode(index), section, glyphMap, imagePositions, spacing, layoutTextSize);\n }\n\n const lineCount = Math.max(1, Math.ceil(totalWidth / maxWidth));\n return totalWidth / lineCount;\n}\n\nfunction calculateBadness(lineWidth: number,\n targetWidth: number,\n penalty: number,\n isLastBreak: boolean) {\n const raggedness = Math.pow(lineWidth - targetWidth, 2);\n if (isLastBreak) {\n // Favor finals lines shorter than average over longer than average\n if (lineWidth < targetWidth) {\n return raggedness / 2;\n } else {\n return raggedness * 2;\n }\n }\n\n return raggedness + Math.abs(penalty) * penalty;\n}\n\nfunction calculatePenalty(codePoint: number, nextCodePoint: number, penalizableIdeographicBreak: boolean) {\n let penalty = 0;\n // Force break on newline\n if (codePoint === 0x0a) {\n penalty -= 10000;\n }\n // Penalize breaks between characters that allow ideographic breaking because\n // they are less preferable than breaks at spaces (or zero width spaces).\n if (penalizableIdeographicBreak) {\n penalty += 150;\n }\n\n // Penalize open parenthesis at end of line\n if (codePoint === 0x28 || codePoint === 0xff08) {\n penalty += 50;\n }\n\n // Penalize close parenthesis at beginning of line\n if (nextCodePoint === 0x29 || nextCodePoint === 0xff09) {\n penalty += 50;\n }\n return penalty;\n}\n\ntype Break = {\n index: number;\n x: number;\n priorBreak: Break;\n badness: number;\n};\n\nfunction evaluateBreak(\n breakIndex: number,\n breakX: number,\n targetWidth: number,\n potentialBreaks: Array,\n penalty: number,\n isLastBreak: boolean\n): Break {\n // We could skip evaluating breaks where the line length (breakX - priorBreak.x) > maxWidth\n // ...but in fact we allow lines longer than maxWidth (if there's no break points)\n // ...and when targetWidth and maxWidth are close, strictly enforcing maxWidth can give\n // more lopsided results.\n\n let bestPriorBreak: Break = null;\n let bestBreakBadness = calculateBadness(breakX, targetWidth, penalty, isLastBreak);\n\n for (const potentialBreak of potentialBreaks) {\n const lineWidth = breakX - potentialBreak.x;\n const breakBadness =\n calculateBadness(lineWidth, targetWidth, penalty, isLastBreak) + potentialBreak.badness;\n if (breakBadness <= bestBreakBadness) {\n bestPriorBreak = potentialBreak;\n bestBreakBadness = breakBadness;\n }\n }\n\n return {\n index: breakIndex,\n x: breakX,\n priorBreak: bestPriorBreak,\n badness: bestBreakBadness\n };\n}\n\nfunction leastBadBreaks(lastLineBreak?: Break | null): Array {\n if (!lastLineBreak) {\n return [];\n }\n return leastBadBreaks(lastLineBreak.priorBreak).concat(lastLineBreak.index);\n}\n\nfunction determineLineBreaks(\n logicalInput: TaggedString,\n spacing: number,\n maxWidth: number,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n symbolPlacement: string,\n layoutTextSize: number\n): Array {\n if (symbolPlacement !== 'point')\n return [];\n\n if (!logicalInput)\n return [];\n\n const potentialLineBreaks = [];\n const targetWidth = determineAverageLineWidth(logicalInput, spacing, maxWidth, glyphMap, imagePositions, layoutTextSize);\n\n const hasServerSuggestedBreakpoints = logicalInput.text.indexOf('\\u200b') >= 0;\n\n let currentX = 0;\n\n for (let i = 0; i < logicalInput.length(); i++) {\n const section = logicalInput.getSection(i);\n const codePoint = logicalInput.getCharCode(i);\n if (!whitespace[codePoint]) currentX += getGlyphAdvance(codePoint, section, glyphMap, imagePositions, spacing, layoutTextSize);\n\n // Ideographic characters, spaces, and word-breaking punctuation that often appear without\n // surrounding spaces.\n if ((i < logicalInput.length() - 1)) {\n const ideographicBreak = charAllowsIdeographicBreaking(codePoint);\n if (breakable[codePoint] || ideographicBreak || section.imageName) {\n\n potentialLineBreaks.push(\n evaluateBreak(\n i + 1,\n currentX,\n targetWidth,\n potentialLineBreaks,\n calculatePenalty(codePoint, logicalInput.getCharCode(i + 1), ideographicBreak && hasServerSuggestedBreakpoints),\n false));\n }\n }\n }\n\n return leastBadBreaks(\n evaluateBreak(\n logicalInput.length(),\n currentX,\n targetWidth,\n potentialLineBreaks,\n 0,\n true));\n}\n\nfunction getAnchorAlignment(anchor: SymbolAnchor) {\n let horizontalAlign = 0.5, verticalAlign = 0.5;\n\n switch (anchor) {\n case 'right':\n case 'top-right':\n case 'bottom-right':\n horizontalAlign = 1;\n break;\n case 'left':\n case 'top-left':\n case 'bottom-left':\n horizontalAlign = 0;\n break;\n }\n\n switch (anchor) {\n case 'bottom':\n case 'bottom-right':\n case 'bottom-left':\n verticalAlign = 1;\n break;\n case 'top':\n case 'top-right':\n case 'top-left':\n verticalAlign = 0;\n break;\n }\n\n return {horizontalAlign, verticalAlign};\n}\n\nfunction shapeLines(shaping: Shaping,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n glyphPositions: {\n [_: string]: {\n [_: number]: GlyphPosition;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n lines: Array,\n lineHeight: number,\n textAnchor: SymbolAnchor,\n textJustify: TextJustify,\n writingMode: WritingMode.horizontal | WritingMode.vertical,\n spacing: number,\n allowVerticalPlacement: boolean,\n layoutTextSizeThisZoom: number) {\n\n let x = 0;\n let y = SHAPING_DEFAULT_OFFSET;\n\n let maxLineLength = 0;\n let maxLineHeight = 0;\n\n const justify =\n textJustify === 'right' ? 1 :\n textJustify === 'left' ? 0 : 0.5;\n\n let lineIndex = 0;\n for (const line of lines) {\n line.trim();\n\n const lineMaxScale = line.getMaxScale();\n const maxLineOffset = (lineMaxScale - 1) * ONE_EM;\n const positionedLine = {positionedGlyphs: [], lineOffset: 0};\n shaping.positionedLines[lineIndex] = positionedLine;\n const positionedGlyphs = positionedLine.positionedGlyphs;\n let lineOffset = 0.0;\n\n if (!line.length()) {\n y += lineHeight; // Still need a line feed after empty line\n ++lineIndex;\n continue;\n }\n\n for (let i = 0; i < line.length(); i++) {\n const section = line.getSection(i);\n const sectionIndex = line.getSectionIndex(i);\n const codePoint = line.getCharCode(i);\n let baselineOffset = 0.0;\n let metrics = null;\n let rect = null;\n let imageName = null;\n let verticalAdvance = ONE_EM;\n const vertical = !(writingMode === WritingMode.horizontal ||\n // Don't verticalize glyphs that have no upright orientation if vertical placement is disabled.\n (!allowVerticalPlacement && !charHasUprightVerticalOrientation(codePoint)) ||\n // If vertical placement is enabled, don't verticalize glyphs that\n // are from complex text layout script, or whitespaces.\n (allowVerticalPlacement && (whitespace[codePoint] || charInComplexShapingScript(codePoint))));\n\n if (!section.imageName) {\n const positions = glyphPositions[section.fontStack];\n const glyphPosition = positions && positions[codePoint];\n if (glyphPosition && glyphPosition.rect) {\n rect = glyphPosition.rect;\n metrics = glyphPosition.metrics;\n } else {\n const glyphs = glyphMap[section.fontStack];\n const glyph = glyphs && glyphs[codePoint];\n if (!glyph) continue;\n metrics = glyph.metrics;\n }\n\n // We don't know the baseline, but since we're laying out\n // at 24 points, we can calculate how much it will move when\n // we scale up or down.\n baselineOffset = (lineMaxScale - section.scale) * ONE_EM;\n } else {\n const imagePosition = imagePositions[section.imageName];\n if (!imagePosition) continue;\n imageName = section.imageName;\n shaping.iconsInText = shaping.iconsInText || true;\n rect = imagePosition.paddedRect;\n const size = imagePosition.displaySize;\n // If needed, allow to set scale factor for an image using\n // alias \"image-scale\" that could be alias for \"font-scale\"\n // when FormattedSection is an image section.\n section.scale = section.scale * ONE_EM / layoutTextSizeThisZoom;\n\n metrics = {width: size[0],\n height: size[1],\n left: IMAGE_PADDING,\n top: -GLYPH_PBF_BORDER,\n advance: vertical ? size[1] : size[0]};\n\n // Difference between one EM and an image size.\n // Aligns bottom of an image to a baseline level.\n const imageOffset = ONE_EM - size[1] * section.scale;\n baselineOffset = maxLineOffset + imageOffset;\n verticalAdvance = metrics.advance;\n\n // Difference between height of an image and one EM at max line scale.\n // Pushes current line down if an image size is over 1 EM at max line scale.\n const offset = vertical ? size[0] * section.scale - ONE_EM * lineMaxScale :\n size[1] * section.scale - ONE_EM * lineMaxScale;\n if (offset > 0 && offset > lineOffset) {\n lineOffset = offset;\n }\n }\n\n if (!vertical) {\n positionedGlyphs.push({glyph: codePoint, imageName, x, y: y + baselineOffset, vertical, scale: section.scale, fontStack: section.fontStack, sectionIndex, metrics, rect});\n x += metrics.advance * section.scale + spacing;\n } else {\n shaping.verticalizable = true;\n positionedGlyphs.push({glyph: codePoint, imageName, x, y: y + baselineOffset, vertical, scale: section.scale, fontStack: section.fontStack, sectionIndex, metrics, rect});\n x += verticalAdvance * section.scale + spacing;\n }\n }\n\n // Only justify if we placed at least one glyph\n if (positionedGlyphs.length !== 0) {\n const lineLength = x - spacing;\n maxLineLength = Math.max(lineLength, maxLineLength);\n justifyLine(positionedGlyphs, 0, positionedGlyphs.length - 1, justify, lineOffset);\n }\n\n x = 0;\n const currentLineHeight = lineHeight * lineMaxScale + lineOffset;\n positionedLine.lineOffset = Math.max(lineOffset, maxLineOffset);\n y += currentLineHeight;\n maxLineHeight = Math.max(currentLineHeight, maxLineHeight);\n ++lineIndex;\n }\n\n // Calculate the bounding box and justify / align text block.\n const height = y - SHAPING_DEFAULT_OFFSET;\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(textAnchor);\n align(shaping.positionedLines, justify, horizontalAlign, verticalAlign, maxLineLength, maxLineHeight, lineHeight, height, lines.length);\n\n shaping.top += -verticalAlign * height;\n shaping.bottom = shaping.top + height;\n shaping.left += -horizontalAlign * maxLineLength;\n shaping.right = shaping.left + maxLineLength;\n}\n\n// justify right = 1, left = 0, center = 0.5\nfunction justifyLine(positionedGlyphs: Array,\n start: number,\n end: number,\n justify: 1 | 0 | 0.5,\n lineOffset: number) {\n if (!justify && !lineOffset)\n return;\n\n const lastPositionedGlyph = positionedGlyphs[end];\n const lastAdvance = lastPositionedGlyph.metrics.advance * lastPositionedGlyph.scale;\n const lineIndent = (positionedGlyphs[end].x + lastAdvance) * justify;\n\n for (let j = start; j <= end; j++) {\n positionedGlyphs[j].x -= lineIndent;\n positionedGlyphs[j].y += lineOffset;\n }\n}\n\nfunction align(positionedLines: Array,\n justify: number,\n horizontalAlign: number,\n verticalAlign: number,\n maxLineLength: number,\n maxLineHeight: number,\n lineHeight: number,\n blockHeight: number,\n lineCount: number) {\n const shiftX = (justify - horizontalAlign) * maxLineLength;\n let shiftY = 0;\n\n if (maxLineHeight !== lineHeight) {\n shiftY = -blockHeight * verticalAlign - SHAPING_DEFAULT_OFFSET;\n } else {\n shiftY = (-verticalAlign * lineCount + 0.5) * lineHeight;\n }\n\n for (const line of positionedLines) {\n for (const positionedGlyph of line.positionedGlyphs) {\n positionedGlyph.x += shiftX;\n positionedGlyph.y += shiftY;\n }\n }\n}\n\nexport type PositionedIcon = {\n image: ImagePosition;\n top: number;\n bottom: number;\n left: number;\n right: number;\n collisionPadding?: [number, number, number, number];\n};\n\nfunction shapeIcon(\n image: ImagePosition,\n iconOffset: [number, number],\n iconAnchor: SymbolAnchor\n): PositionedIcon {\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(iconAnchor);\n const dx = iconOffset[0];\n const dy = iconOffset[1];\n const x1 = dx - image.displaySize[0] * horizontalAlign;\n const x2 = x1 + image.displaySize[0];\n const y1 = dy - image.displaySize[1] * verticalAlign;\n const y2 = y1 + image.displaySize[1];\n return {image, top: y1, bottom: y2, left: x1, right: x2};\n}\n\nfunction fitIconToText(\n shapedIcon: PositionedIcon,\n shapedText: Shaping,\n textFit: string,\n padding: [number, number, number, number],\n iconOffset: [number, number],\n fontScale: number\n): PositionedIcon {\n\n const image = shapedIcon.image;\n\n let collisionPadding;\n if (image.content) {\n const content = image.content;\n const pixelRatio = image.pixelRatio || 1;\n collisionPadding = [\n content[0] / pixelRatio,\n content[1] / pixelRatio,\n image.displaySize[0] - content[2] / pixelRatio,\n image.displaySize[1] - content[3] / pixelRatio\n ];\n }\n\n // We don't respect the icon-anchor, because icon-text-fit is set. Instead,\n // the icon will be centered on the text, then stretched in the given\n // dimensions.\n\n const textLeft = shapedText.left * fontScale;\n const textRight = shapedText.right * fontScale;\n\n let top, right, bottom, left;\n if (textFit === 'width' || textFit === 'both') {\n // Stretched horizontally to the text width\n left = iconOffset[0] + textLeft - padding[3];\n right = iconOffset[0] + textRight + padding[1];\n } else {\n // Centered on the text\n left = iconOffset[0] + (textLeft + textRight - image.displaySize[0]) / 2;\n right = left + image.displaySize[0];\n }\n\n const textTop = shapedText.top * fontScale;\n const textBottom = shapedText.bottom * fontScale;\n if (textFit === 'height' || textFit === 'both') {\n // Stretched vertically to the text height\n top = iconOffset[1] + textTop - padding[0];\n bottom = iconOffset[1] + textBottom + padding[2];\n } else {\n // Centered on the text\n top = iconOffset[1] + (textTop + textBottom - image.displaySize[1]) / 2;\n bottom = top + image.displaySize[1];\n }\n\n return {image, top, right, bottom, left, collisionPadding};\n}\n","import {Interpolate, interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {clamp} from '../util/util';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\n\nimport type {PropertyValue, PossiblyEvaluatedPropertyValue} from '../style/properties';\nimport type {InterpolationType} from '@maplibre/maplibre-gl-style-spec';\n\nconst MAX_GLYPH_ICON_SIZE = 255;\nconst SIZE_PACK_FACTOR = 128;\nconst MAX_PACKED_SIZE = MAX_GLYPH_ICON_SIZE * SIZE_PACK_FACTOR;\n\nexport {getSizeData, evaluateSizeForFeature, evaluateSizeForZoom, SIZE_PACK_FACTOR, MAX_GLYPH_ICON_SIZE, MAX_PACKED_SIZE};\n\nexport type SizeData = {\n kind: 'constant';\n layoutSize: number;\n} | {\n kind: 'source';\n} | {\n kind: 'camera';\n minZoom: number;\n maxZoom: number;\n minSize: number;\n maxSize: number;\n interpolationType: InterpolationType;\n} | {\n kind: 'composite';\n minZoom: number;\n maxZoom: number;\n interpolationType: InterpolationType;\n};\n\nexport type EvaluatedZoomSize = {uSizeT: number; uSize: number};\n\n// For {text,icon}-size, get the bucket-level data that will be needed by\n// the painter to set symbol-size-related uniforms\nfunction getSizeData(\n tileZoom: number,\n value: PropertyValue>\n): SizeData {\n const {expression} = value;\n\n if (expression.kind === 'constant') {\n const layoutSize = expression.evaluate(new EvaluationParameters(tileZoom + 1));\n return {kind: 'constant', layoutSize};\n\n } else if (expression.kind === 'source') {\n return {kind: 'source'};\n\n } else {\n const {zoomStops, interpolationType} = expression;\n\n // calculate covering zoom stops for zoom-dependent values\n let lower = 0;\n while (lower < zoomStops.length && zoomStops[lower] <= tileZoom) lower++;\n lower = Math.max(0, lower - 1);\n let upper = lower;\n while (upper < zoomStops.length && zoomStops[upper] < tileZoom + 1) upper++;\n upper = Math.min(zoomStops.length - 1, upper);\n\n const minZoom = zoomStops[lower];\n const maxZoom = zoomStops[upper];\n\n // We'd like to be able to use CameraExpression or CompositeExpression in these\n // return types rather than ExpressionSpecification, but the former are not\n // transferrable across Web Worker boundaries.\n if (expression.kind === 'composite') {\n return {kind: 'composite', minZoom, maxZoom, interpolationType};\n }\n\n // for camera functions, also save off the function values\n // evaluated at the covering zoom levels\n const minSize = expression.evaluate(new EvaluationParameters(minZoom));\n const maxSize = expression.evaluate(new EvaluationParameters(maxZoom));\n\n return {kind: 'camera', minZoom, maxZoom, minSize, maxSize, interpolationType};\n }\n}\n\nfunction evaluateSizeForFeature(sizeData: SizeData,\n {\n uSize,\n uSizeT\n }: {\n uSize: number;\n uSizeT: number;\n },\n {\n lowerSize,\n upperSize\n }: {\n lowerSize: number;\n upperSize: number;\n }): number {\n if (sizeData.kind === 'source') {\n return lowerSize / SIZE_PACK_FACTOR;\n } else if (sizeData.kind === 'composite') {\n return interpolates.number(lowerSize / SIZE_PACK_FACTOR, upperSize / SIZE_PACK_FACTOR, uSizeT);\n }\n return uSize;\n}\n\nfunction evaluateSizeForZoom(sizeData: SizeData, zoom: number): EvaluatedZoomSize {\n let uSizeT = 0;\n let uSize = 0;\n\n if (sizeData.kind === 'constant') {\n uSize = sizeData.layoutSize;\n\n } else if (sizeData.kind !== 'source') {\n const {interpolationType, minZoom, maxZoom} = sizeData;\n\n // Even though we could get the exact value of the camera function\n // at z = tr.zoom, we intentionally do not: instead, we interpolate\n // between the camera function values at a pair of zoom stops covering\n // [tileZoom, tileZoom + 1] in order to be consistent with this\n // restriction on composite functions\n const t = !interpolationType ? 0 : clamp(\n Interpolate.interpolationFactor(interpolationType, zoom, minZoom, maxZoom), 0, 1);\n\n if (sizeData.kind === 'camera') {\n uSize = interpolates.number(sizeData.minSize, sizeData.maxSize, t);\n } else {\n uSizeT = t;\n }\n }\n\n return {uSizeT, uSize};\n}\n","import {SymbolLayoutPropsPossiblyEvaluated} from './symbol_style_layer_properties.g';\nimport type {SymbolLayoutProps} from './symbol_style_layer_properties.g';\nimport {PossiblyEvaluated} from '../properties';\n\n/**\n * The overlap mode for properties like `icon-overlap`and `text-overlap`\n */\nexport type OverlapMode = 'never' | 'always' | 'cooperative';\n\nexport function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'icon-overlap', allowOverlapProp: 'icon-allow-overlap'): OverlapMode;\nexport function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'text-overlap', allowOverlapProp: 'text-allow-overlap'): OverlapMode;\nexport function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'icon-overlap' | 'text-overlap', allowOverlapProp: 'icon-allow-overlap' | 'text-allow-overlap'): OverlapMode {\n let result: OverlapMode = 'never';\n const overlap = layout.get(overlapProp);\n\n if (overlap) {\n // if -overlap is set, use it\n result = overlap;\n } else if (layout.get(allowOverlapProp)) {\n // fall back to -allow-overlap, with false='never', true='always'\n result = 'always';\n }\n\n return result;\n}\n","import {\n symbolLayoutAttributes,\n collisionVertexAttributes,\n collisionBoxLayout,\n dynamicLayoutAttributes,\n} from './symbol_attributes';\n\nimport {SymbolLayoutArray,\n SymbolDynamicLayoutArray,\n SymbolOpacityArray,\n CollisionBoxLayoutArray,\n CollisionVertexArray,\n PlacedSymbolArray,\n SymbolInstanceArray,\n GlyphOffsetArray,\n SymbolLineVertexArray,\n TextAnchorOffsetArray\n} from '../array_types.g';\n\nimport Point from '@mapbox/point-geometry';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray, LineIndexArray} from '../index_array_type';\nimport {transformText} from '../../symbol/transform_text';\nimport {mergeLines} from '../../symbol/merge_lines';\nimport {allowsVerticalWritingMode, stringContainsRTLText} from '../../util/script_detection';\nimport {WritingMode} from '../../symbol/shaping';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport mvt from '@mapbox/vector-tile';\nconst vectorTileFeatureTypes = mvt.VectorTileFeature.types;\nimport {verticalizedCharacterMap} from '../../util/verticalize_punctuation';\nimport {Anchor} from '../../symbol/anchor';\nimport {getSizeData, MAX_PACKED_SIZE} from '../../symbol/symbol_size';\n\nimport {register} from '../../util/web_worker_transfer';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\nimport {Formatted, ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport {plugin as globalRTLTextPlugin, getRTLTextPluginStatus} from '../../source/rtl_text_plugin';\nimport {mat4} from 'gl-matrix';\nimport {getOverlapMode} from '../../style/style_layer/overlap_mode';\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {CollisionBoxArray, CollisionBox, SymbolInstance} from '../array_types.g';\nimport type {StructArray, StructArrayMember, ViewType} from '../../util/struct_array';\nimport type {SymbolStyleLayer} from '../../style/style_layer/symbol_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type {SymbolQuad} from '../../symbol/quads';\nimport type {SizeData} from '../../symbol/symbol_size';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nexport type SingleCollisionBox = {\n x1: number;\n y1: number;\n x2: number;\n y2: number;\n anchorPointX: number;\n anchorPointY: number;\n};\n\nexport type CollisionArrays = {\n textBox?: SingleCollisionBox;\n verticalTextBox?: SingleCollisionBox;\n iconBox?: SingleCollisionBox;\n verticalIconBox?: SingleCollisionBox;\n textFeatureIndex?: number;\n verticalTextFeatureIndex?: number;\n iconFeatureIndex?: number;\n verticalIconFeatureIndex?: number;\n};\n\nexport type SymbolFeature = {\n sortKey: number | void;\n text: Formatted | void;\n icon: ResolvedImage;\n index: number;\n sourceLayerIndex: number;\n geometry: Array>;\n properties: any;\n type: 'Unknown' | 'Point' | 'LineString' | 'Polygon';\n id?: any;\n};\n\nexport type SortKeyRange = {\n sortKey: number;\n symbolInstanceStart: number;\n symbolInstanceEnd: number;\n};\n\n// Opacity arrays are frequently updated but don't contain a lot of information, so we pack them\n// tight. Each Uint32 is actually four duplicate Uint8s for the four corners of a glyph\n// 7 bits are for the current opacity, and the lowest bit is the target opacity\n\n// actually defined in symbol_attributes.js\n// const placementOpacityAttributes = [\n// { name: 'a_fade_opacity', components: 1, type: 'Uint32' }\n// ];\nconst shaderOpacityAttributes = [\n {name: 'a_fade_opacity', components: 1, type: 'Uint8' as ViewType, offset: 0}\n];\n\nfunction addVertex(\n array: StructArray,\n anchorX: number,\n anchorY: number,\n ox: number,\n oy: number,\n tx: number,\n ty: number,\n sizeVertex: number,\n isSDF: boolean,\n pixelOffsetX: number,\n pixelOffsetY: number,\n minFontScaleX: number,\n minFontScaleY: number\n) {\n const aSizeX = sizeVertex ? Math.min(MAX_PACKED_SIZE, Math.round(sizeVertex[0])) : 0;\n const aSizeY = sizeVertex ? Math.min(MAX_PACKED_SIZE, Math.round(sizeVertex[1])) : 0;\n array.emplaceBack(\n // a_pos_offset\n anchorX,\n anchorY,\n Math.round(ox * 32),\n Math.round(oy * 32),\n\n // a_data\n tx, // x coordinate of symbol on glyph atlas texture\n ty, // y coordinate of symbol on glyph atlas texture\n (aSizeX << 1) + (isSDF ? 1 : 0),\n aSizeY,\n pixelOffsetX * 16,\n pixelOffsetY * 16,\n minFontScaleX * 256,\n minFontScaleY * 256\n );\n}\n\nfunction addDynamicAttributes(dynamicLayoutVertexArray: StructArray, p: Point, angle: number) {\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n}\n\nfunction containsRTLText(formattedText: Formatted): boolean {\n for (const section of formattedText.sections) {\n if (stringContainsRTLText(section.text)) {\n return true;\n }\n }\n return false;\n}\n\nexport class SymbolBuffers {\n layoutVertexArray: SymbolLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n\n dynamicLayoutVertexArray: SymbolDynamicLayoutArray;\n dynamicLayoutVertexBuffer: VertexBuffer;\n\n opacityVertexArray: SymbolOpacityArray;\n opacityVertexBuffer: VertexBuffer;\n hasVisibleVertices: boolean;\n\n collisionVertexArray: CollisionVertexArray;\n collisionVertexBuffer: VertexBuffer;\n\n placedSymbolArray: PlacedSymbolArray;\n\n constructor(programConfigurations: ProgramConfigurationSet) {\n this.layoutVertexArray = new SymbolLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.programConfigurations = programConfigurations;\n this.segments = new SegmentVector();\n this.dynamicLayoutVertexArray = new SymbolDynamicLayoutArray();\n this.opacityVertexArray = new SymbolOpacityArray();\n this.hasVisibleVertices = false;\n this.placedSymbolArray = new PlacedSymbolArray();\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0 &&\n this.indexArray.length === 0 &&\n this.dynamicLayoutVertexArray.length === 0 &&\n this.opacityVertexArray.length === 0;\n }\n\n upload(context: Context, dynamicIndexBuffer: boolean, upload?: boolean, update?: boolean) {\n if (this.isEmpty()) {\n return;\n }\n\n if (upload) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, symbolLayoutAttributes.members);\n this.indexBuffer = context.createIndexBuffer(this.indexArray, dynamicIndexBuffer);\n this.dynamicLayoutVertexBuffer = context.createVertexBuffer(this.dynamicLayoutVertexArray, dynamicLayoutAttributes.members, true);\n this.opacityVertexBuffer = context.createVertexBuffer(this.opacityVertexArray, shaderOpacityAttributes, true);\n // This is a performance hack so that we can write to opacityVertexArray with uint32s\n // even though the shaders read uint8s\n this.opacityVertexBuffer.itemSize = 1;\n }\n if (upload || update) {\n this.programConfigurations.upload(context);\n }\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n this.dynamicLayoutVertexBuffer.destroy();\n this.opacityVertexBuffer.destroy();\n }\n}\n\nregister('SymbolBuffers', SymbolBuffers);\n\nclass CollisionBuffers {\n layoutVertexArray: StructArray;\n layoutAttributes: Array;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray | LineIndexArray;\n indexBuffer: IndexBuffer;\n\n segments: SegmentVector;\n\n collisionVertexArray: CollisionVertexArray;\n collisionVertexBuffer: VertexBuffer;\n\n constructor(LayoutArray: {\n new (...args: any): StructArray;\n },\n layoutAttributes: Array,\n IndexArray: {\n new (...args: any): TriangleIndexArray | LineIndexArray;\n }) {\n this.layoutVertexArray = new LayoutArray();\n this.layoutAttributes = layoutAttributes;\n this.indexArray = new IndexArray();\n this.segments = new SegmentVector();\n this.collisionVertexArray = new CollisionVertexArray();\n }\n\n upload(context: Context) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, this.layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n this.collisionVertexBuffer = context.createVertexBuffer(this.collisionVertexArray, collisionVertexAttributes.members, true);\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.segments.destroy();\n this.collisionVertexBuffer.destroy();\n }\n}\n\nregister('CollisionBuffers', CollisionBuffers);\n\n/**\n * @internal\n * Unlike other buckets, which simply implement #addFeature with type-specific\n * logic for (essentially) triangulating feature geometries, SymbolBucket\n * requires specialized behavior:\n *\n * 1. WorkerTile#parse(), the logical owner of the bucket creation process,\n * calls SymbolBucket#populate(), which resolves text and icon tokens on\n * each feature, adds each glyphs and symbols needed to the passed-in\n * collections options.glyphDependencies and options.iconDependencies, and\n * stores the feature data for use in subsequent step (this.features).\n *\n * 2. WorkerTile asynchronously requests from the main thread all of the glyphs\n * and icons needed (by this bucket and any others). When glyphs and icons\n * have been received, the WorkerTile creates a CollisionIndex and invokes:\n *\n * 3. performSymbolLayout(bucket, stacks, icons) perform texts shaping and\n * layout on a Symbol Bucket. This step populates:\n * `this.symbolInstances`: metadata on generated symbols\n * `this.collisionBoxArray`: collision data for use by foreground\n * `this.text`: SymbolBuffers for text symbols\n * `this.icons`: SymbolBuffers for icons\n * `this.iconCollisionBox`: Debug SymbolBuffers for icon collision boxes\n * `this.textCollisionBox`: Debug SymbolBuffers for text collision boxes\n * The results are sent to the foreground for rendering\n *\n * 4. performSymbolPlacement(bucket, collisionIndex) is run on the foreground,\n * and uses the CollisionIndex along with current camera settings to determine\n * which symbols can actually show on the map. Collided symbols are hidden\n * using a dynamic \"OpacityVertexArray\".\n */\nexport class SymbolBucket implements Bucket {\n static MAX_GLYPHS: number;\n static addDynamicAttributes: typeof addDynamicAttributes;\n\n collisionBoxArray: CollisionBoxArray;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n\n index: number;\n sdfIcons: boolean;\n iconsInText: boolean;\n iconsNeedLinear: boolean;\n bucketInstanceId: number;\n justReloaded: boolean;\n hasPattern: boolean;\n\n textSizeData: SizeData;\n iconSizeData: SizeData;\n\n glyphOffsetArray: GlyphOffsetArray;\n lineVertexArray: SymbolLineVertexArray;\n features: Array;\n symbolInstances: SymbolInstanceArray;\n textAnchorOffsets: TextAnchorOffsetArray;\n collisionArrays: Array;\n sortKeyRanges: Array;\n pixelRatio: number;\n tilePixelRatio: number;\n compareText: {[_: string]: Array};\n fadeStartTime: number;\n sortFeaturesByKey: boolean;\n sortFeaturesByY: boolean;\n canOverlap: boolean;\n sortedAngle: number;\n featureSortOrder: Array;\n\n collisionCircleArray: Array;\n placementInvProjMatrix: mat4;\n placementViewportMatrix: mat4;\n\n text: SymbolBuffers;\n icon: SymbolBuffers;\n textCollisionBox: CollisionBuffers;\n iconCollisionBox: CollisionBuffers;\n uploaded: boolean;\n sourceLayerIndex: number;\n sourceID: string;\n symbolInstanceIndexes: Array;\n writingModes: WritingMode[];\n allowVerticalPlacement: boolean;\n hasRTLText: boolean;\n\n constructor(options: BucketParameters) {\n this.collisionBoxArray = options.collisionBoxArray;\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.pixelRatio = options.pixelRatio;\n this.sourceLayerIndex = options.sourceLayerIndex;\n this.hasPattern = false;\n this.hasRTLText = false;\n this.sortKeyRanges = [];\n\n this.collisionCircleArray = [];\n this.placementInvProjMatrix = mat4.identity([] as any);\n this.placementViewportMatrix = mat4.identity([] as any);\n\n const layer = this.layers[0];\n const unevaluatedLayoutValues = layer._unevaluatedLayout._values;\n\n this.textSizeData = getSizeData(this.zoom, unevaluatedLayoutValues['text-size']);\n this.iconSizeData = getSizeData(this.zoom, unevaluatedLayoutValues['icon-size']);\n\n const layout = this.layers[0].layout;\n const sortKey = layout.get('symbol-sort-key');\n const zOrder = layout.get('symbol-z-order');\n this.canOverlap =\n getOverlapMode(layout, 'text-overlap', 'text-allow-overlap') !== 'never' ||\n getOverlapMode(layout, 'icon-overlap', 'icon-allow-overlap') !== 'never' ||\n layout.get('text-ignore-placement') ||\n layout.get('icon-ignore-placement');\n this.sortFeaturesByKey = zOrder !== 'viewport-y' && !sortKey.isConstant();\n const zOrderByViewportY = zOrder === 'viewport-y' || (zOrder === 'auto' && !this.sortFeaturesByKey);\n this.sortFeaturesByY = zOrderByViewportY && this.canOverlap;\n\n if (layout.get('symbol-placement') === 'point') {\n this.writingModes = layout.get('text-writing-mode').map(wm => WritingMode[wm]);\n }\n\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n\n this.sourceID = options.sourceID;\n }\n\n createArrays() {\n this.text = new SymbolBuffers(new ProgramConfigurationSet(this.layers, this.zoom, property => /^text/.test(property)));\n this.icon = new SymbolBuffers(new ProgramConfigurationSet(this.layers, this.zoom, property => /^icon/.test(property)));\n\n this.glyphOffsetArray = new GlyphOffsetArray();\n this.lineVertexArray = new SymbolLineVertexArray();\n this.symbolInstances = new SymbolInstanceArray();\n this.textAnchorOffsets = new TextAnchorOffsetArray();\n }\n\n calculateGlyphDependencies(text: string, stack: {[_: number]: boolean}, textAlongLine: boolean, allowVerticalPlacement: boolean, doesAllowVerticalWritingMode: boolean) {\n for (let i = 0; i < text.length; i++) {\n stack[text.charCodeAt(i)] = true;\n if ((textAlongLine || allowVerticalPlacement) && doesAllowVerticalWritingMode) {\n const verticalChar = verticalizedCharacterMap[text.charAt(i)];\n if (verticalChar) {\n stack[verticalChar.charCodeAt(0)] = true;\n }\n }\n }\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n const layer = this.layers[0];\n const layout = layer.layout;\n\n const textFont = layout.get('text-font');\n const textField = layout.get('text-field');\n const iconImage = layout.get('icon-image');\n const hasText =\n (textField.value.kind !== 'constant' ||\n (textField.value.value instanceof Formatted && !textField.value.value.isEmpty()) ||\n textField.value.value.toString().length > 0) &&\n (textFont.value.kind !== 'constant' || textFont.value.value.length > 0);\n // we should always resolve the icon-image value if the property was defined in the style\n // this allows us to fire the styleimagemissing event if image evaluation returns null\n // the only way to distinguish between null returned from a coalesce statement with no valid images\n // and null returned because icon-image wasn't defined is to check whether or not iconImage.parameters is an empty object\n const hasIcon = iconImage.value.kind !== 'constant' || !!iconImage.value.value || Object.keys(iconImage.parameters).length > 0;\n const symbolSortKey = layout.get('symbol-sort-key');\n\n this.features = [];\n\n if (!hasText && !hasIcon) {\n return;\n }\n\n const icons = options.iconDependencies;\n const stacks = options.glyphDependencies;\n const availableImages = options.availableImages;\n const globalProperties = new EvaluationParameters(this.zoom);\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n\n const needGeometry = layer._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n if (!layer._featureFilter.filter(globalProperties, evaluationFeature, canonical)) {\n continue;\n }\n\n if (!needGeometry) evaluationFeature.geometry = loadGeometry(feature);\n\n let text: Formatted | void;\n if (hasText) {\n // Expression evaluation will automatically coerce to Formatted\n // but plain string token evaluation skips that pathway so do the\n // conversion here.\n const resolvedTokens = layer.getValueAndResolveTokens('text-field', evaluationFeature, canonical, availableImages);\n const formattedText = Formatted.factory(resolvedTokens);\n if (containsRTLText(formattedText)) {\n this.hasRTLText = true;\n }\n if (\n !this.hasRTLText || // non-rtl text so can proceed safely\n getRTLTextPluginStatus() === 'unavailable' || // We don't intend to lazy-load the rtl text plugin, so proceed with incorrect shaping\n this.hasRTLText && globalRTLTextPlugin.isParsed() // Use the rtlText plugin to shape text\n ) {\n text = transformText(formattedText, layer, evaluationFeature);\n }\n }\n\n let icon: ResolvedImage;\n if (hasIcon) {\n // Expression evaluation will automatically coerce to Image\n // but plain string token evaluation skips that pathway so do the\n // conversion here.\n const resolvedTokens = layer.getValueAndResolveTokens('icon-image', evaluationFeature, canonical, availableImages);\n if (resolvedTokens instanceof ResolvedImage) {\n icon = resolvedTokens;\n } else {\n icon = ResolvedImage.fromString(resolvedTokens);\n }\n }\n\n if (!text && !icon) {\n continue;\n }\n const sortKey = this.sortFeaturesByKey ?\n symbolSortKey.evaluate(evaluationFeature, {}, canonical) :\n undefined;\n\n const symbolFeature: SymbolFeature = {\n id,\n text,\n icon,\n index,\n sourceLayerIndex,\n geometry: evaluationFeature.geometry,\n properties: feature.properties,\n type: vectorTileFeatureTypes[feature.type],\n sortKey\n };\n this.features.push(symbolFeature);\n\n if (icon) {\n icons[icon.name] = true;\n }\n\n if (text) {\n const fontStack = textFont.evaluate(evaluationFeature, {}, canonical).join(',');\n const textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point';\n this.allowVerticalPlacement = this.writingModes && this.writingModes.indexOf(WritingMode.vertical) >= 0;\n for (const section of text.sections) {\n if (!section.image) {\n const doesAllowVerticalWritingMode = allowsVerticalWritingMode(text.toString());\n const sectionFont = section.fontStack || fontStack;\n const sectionStack = stacks[sectionFont] = stacks[sectionFont] || {};\n this.calculateGlyphDependencies(section.text, sectionStack, textAlongLine, this.allowVerticalPlacement, doesAllowVerticalWritingMode);\n } else {\n // Add section image to the list of dependencies.\n icons[section.image.name] = true;\n }\n }\n }\n }\n\n if (layout.get('symbol-placement') === 'line') {\n // Merge adjacent lines with the same text to improve labelling.\n // It's better to place labels on one long line than on many short segments.\n this.features = mergeLines(this.features);\n }\n\n if (this.sortFeaturesByKey) {\n this.features.sort((a, b) => {\n // a.sortKey is always a number when sortFeaturesByKey is true\n return (a.sortKey as number) - (b.sortKey as number);\n });\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.text.programConfigurations.updatePaintArrays(states, vtLayer, this.layers, imagePositions);\n this.icon.programConfigurations.updatePaintArrays(states, vtLayer, this.layers, imagePositions);\n }\n\n isEmpty() {\n // When the bucket encounters only rtl-text but the plugin isnt loaded, no symbol instances will be created.\n // In order for the bucket to be serialized, and not discarded as an empty bucket both checks are necessary.\n return this.symbolInstances.length === 0 && !this.hasRTLText;\n }\n\n uploadPending() {\n return !this.uploaded || this.text.programConfigurations.needsUpload || this.icon.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded && this.hasDebugData()) {\n this.textCollisionBox.upload(context);\n this.iconCollisionBox.upload(context);\n }\n this.text.upload(context, this.sortFeaturesByY, !this.uploaded, this.text.programConfigurations.needsUpload);\n this.icon.upload(context, this.sortFeaturesByY, !this.uploaded, this.icon.programConfigurations.needsUpload);\n this.uploaded = true;\n }\n\n destroyDebugData() {\n this.textCollisionBox.destroy();\n this.iconCollisionBox.destroy();\n }\n\n destroy() {\n this.text.destroy();\n this.icon.destroy();\n\n if (this.hasDebugData()) {\n this.destroyDebugData();\n }\n }\n\n addToLineVertexArray(anchor: Anchor, line: any) {\n const lineStartIndex = this.lineVertexArray.length;\n if (anchor.segment !== undefined) {\n let sumForwardLength = anchor.dist(line[anchor.segment + 1]);\n let sumBackwardLength = anchor.dist(line[anchor.segment]);\n const vertices = {};\n for (let i = anchor.segment + 1; i < line.length; i++) {\n vertices[i] = {x: line[i].x, y: line[i].y, tileUnitDistanceFromAnchor: sumForwardLength};\n if (i < line.length - 1) {\n sumForwardLength += line[i + 1].dist(line[i]);\n }\n }\n for (let i = anchor.segment || 0; i >= 0; i--) {\n vertices[i] = {x: line[i].x, y: line[i].y, tileUnitDistanceFromAnchor: sumBackwardLength};\n if (i > 0) {\n sumBackwardLength += line[i - 1].dist(line[i]);\n }\n }\n for (let i = 0; i < line.length; i++) {\n const vertex = vertices[i];\n this.lineVertexArray.emplaceBack(vertex.x, vertex.y, vertex.tileUnitDistanceFromAnchor);\n }\n }\n return {\n lineStartIndex,\n lineLength: this.lineVertexArray.length - lineStartIndex\n };\n }\n\n addSymbols(arrays: SymbolBuffers,\n quads: Array,\n sizeVertex: any,\n lineOffset: [number, number],\n alongLine: boolean,\n feature: SymbolFeature,\n writingMode: WritingMode,\n labelAnchor: Anchor,\n lineStartIndex: number,\n lineLength: number,\n associatedIconIndex: number,\n canonical: CanonicalTileID) {\n const indexArray = arrays.indexArray;\n const layoutVertexArray = arrays.layoutVertexArray;\n\n const segment = arrays.segments.prepareSegment(4 * quads.length, layoutVertexArray, indexArray, this.canOverlap ? feature.sortKey as number : undefined);\n const glyphOffsetArrayStart = this.glyphOffsetArray.length;\n const vertexStartIndex = segment.vertexLength;\n\n const angle = (this.allowVerticalPlacement && writingMode === WritingMode.vertical) ? Math.PI / 2 : 0;\n\n const sections = feature.text && feature.text.sections;\n\n for (let i = 0; i < quads.length; i++) {\n const {tl, tr, bl, br, tex, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY, glyphOffset, isSDF, sectionIndex} = quads[i];\n const index = segment.vertexLength;\n\n const y = glyphOffset[1];\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, tl.x, y + tl.y, tex.x, tex.y, sizeVertex, isSDF, pixelOffsetTL.x, pixelOffsetTL.y, minFontScaleX, minFontScaleY);\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, tr.x, y + tr.y, tex.x + tex.w, tex.y, sizeVertex, isSDF, pixelOffsetBR.x, pixelOffsetTL.y, minFontScaleX, minFontScaleY);\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, bl.x, y + bl.y, tex.x, tex.y + tex.h, sizeVertex, isSDF, pixelOffsetTL.x, pixelOffsetBR.y, minFontScaleX, minFontScaleY);\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, br.x, y + br.y, tex.x + tex.w, tex.y + tex.h, sizeVertex, isSDF, pixelOffsetBR.x, pixelOffsetBR.y, minFontScaleX, minFontScaleY);\n\n addDynamicAttributes(arrays.dynamicLayoutVertexArray, labelAnchor, angle);\n\n indexArray.emplaceBack(index, index + 1, index + 2);\n indexArray.emplaceBack(index + 1, index + 2, index + 3);\n\n segment.vertexLength += 4;\n segment.primitiveLength += 2;\n\n this.glyphOffsetArray.emplaceBack(glyphOffset[0]);\n\n if (i === quads.length - 1 || sectionIndex !== quads[i + 1].sectionIndex) {\n arrays.programConfigurations.populatePaintArrays(layoutVertexArray.length, feature, feature.index, {}, canonical, sections && sections[sectionIndex]);\n }\n }\n\n arrays.placedSymbolArray.emplaceBack(\n labelAnchor.x, labelAnchor.y,\n glyphOffsetArrayStart,\n this.glyphOffsetArray.length - glyphOffsetArrayStart,\n vertexStartIndex,\n lineStartIndex,\n lineLength,\n labelAnchor.segment,\n sizeVertex ? sizeVertex[0] : 0,\n sizeVertex ? sizeVertex[1] : 0,\n lineOffset[0], lineOffset[1],\n writingMode,\n // placedOrientation is null initially; will be updated to horizontal(1)/vertical(2) if placed\n 0,\n false as unknown as number,\n // The crossTileID is only filled/used on the foreground for dynamic text anchors\n 0,\n associatedIconIndex\n );\n }\n\n _addCollisionDebugVertex(layoutVertexArray: StructArray, collisionVertexArray: StructArray, point: Point, anchorX: number, anchorY: number, extrude: Point) {\n collisionVertexArray.emplaceBack(0, 0);\n return layoutVertexArray.emplaceBack(\n // pos\n point.x,\n point.y,\n // a_anchor_pos\n anchorX,\n anchorY,\n // extrude\n Math.round(extrude.x),\n Math.round(extrude.y));\n }\n\n addCollisionDebugVertices(x1: number, y1: number, x2: number, y2: number, arrays: CollisionBuffers, boxAnchorPoint: Point, symbolInstance: SymbolInstance) {\n const segment = arrays.segments.prepareSegment(4, arrays.layoutVertexArray, arrays.indexArray);\n const index = segment.vertexLength;\n\n const layoutVertexArray = arrays.layoutVertexArray;\n const collisionVertexArray = arrays.collisionVertexArray;\n\n const anchorX = symbolInstance.anchorX;\n const anchorY = symbolInstance.anchorY;\n\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x1, y1));\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x2, y1));\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x2, y2));\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x1, y2));\n\n segment.vertexLength += 4;\n\n const indexArray = arrays.indexArray as LineIndexArray;\n indexArray.emplaceBack(index, index + 1);\n indexArray.emplaceBack(index + 1, index + 2);\n indexArray.emplaceBack(index + 2, index + 3);\n indexArray.emplaceBack(index + 3, index);\n\n segment.primitiveLength += 4;\n }\n\n addDebugCollisionBoxes(startIndex: number, endIndex: number, symbolInstance: SymbolInstance, isText: boolean) {\n for (let b = startIndex; b < endIndex; b++) {\n const box: CollisionBox = this.collisionBoxArray.get(b);\n const x1 = box.x1;\n const y1 = box.y1;\n const x2 = box.x2;\n const y2 = box.y2;\n\n this.addCollisionDebugVertices(x1, y1, x2, y2,\n isText ? this.textCollisionBox : this.iconCollisionBox,\n box.anchorPoint, symbolInstance);\n }\n }\n\n generateCollisionDebugBuffers() {\n if (this.hasDebugData()) {\n this.destroyDebugData();\n }\n\n this.textCollisionBox = new CollisionBuffers(CollisionBoxLayoutArray, collisionBoxLayout.members, LineIndexArray);\n this.iconCollisionBox = new CollisionBuffers(CollisionBoxLayoutArray, collisionBoxLayout.members, LineIndexArray);\n\n for (let i = 0; i < this.symbolInstances.length; i++) {\n const symbolInstance = this.symbolInstances.get(i);\n this.addDebugCollisionBoxes(symbolInstance.textBoxStartIndex, symbolInstance.textBoxEndIndex, symbolInstance, true);\n this.addDebugCollisionBoxes(symbolInstance.verticalTextBoxStartIndex, symbolInstance.verticalTextBoxEndIndex, symbolInstance, true);\n this.addDebugCollisionBoxes(symbolInstance.iconBoxStartIndex, symbolInstance.iconBoxEndIndex, symbolInstance, false);\n this.addDebugCollisionBoxes(symbolInstance.verticalIconBoxStartIndex, symbolInstance.verticalIconBoxEndIndex, symbolInstance, false);\n }\n }\n\n // These flat arrays are meant to be quicker to iterate over than the source\n // CollisionBoxArray\n _deserializeCollisionBoxesForSymbol(\n collisionBoxArray: CollisionBoxArray,\n textStartIndex: number,\n textEndIndex: number,\n verticalTextStartIndex: number,\n verticalTextEndIndex: number,\n iconStartIndex: number,\n iconEndIndex: number,\n verticalIconStartIndex: number,\n verticalIconEndIndex: number\n ): CollisionArrays {\n\n const collisionArrays = {} as CollisionArrays;\n for (let k = textStartIndex; k < textEndIndex; k++) {\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.textBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.textFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n for (let k = verticalTextStartIndex; k < verticalTextEndIndex; k++) {\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.verticalTextBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.verticalTextFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n for (let k = iconStartIndex; k < iconEndIndex; k++) {\n // An icon can only have one box now, so this indexing is a bit vestigial...\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.iconBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.iconFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n for (let k = verticalIconStartIndex; k < verticalIconEndIndex; k++) {\n // An icon can only have one box now, so this indexing is a bit vestigial...\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.verticalIconBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.verticalIconFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n return collisionArrays;\n }\n\n deserializeCollisionBoxes(collisionBoxArray: CollisionBoxArray) {\n this.collisionArrays = [];\n for (let i = 0; i < this.symbolInstances.length; i++) {\n const symbolInstance = this.symbolInstances.get(i);\n this.collisionArrays.push(this._deserializeCollisionBoxesForSymbol(\n collisionBoxArray,\n symbolInstance.textBoxStartIndex,\n symbolInstance.textBoxEndIndex,\n symbolInstance.verticalTextBoxStartIndex,\n symbolInstance.verticalTextBoxEndIndex,\n symbolInstance.iconBoxStartIndex,\n symbolInstance.iconBoxEndIndex,\n symbolInstance.verticalIconBoxStartIndex,\n symbolInstance.verticalIconBoxEndIndex\n ));\n }\n }\n\n hasTextData() {\n return this.text.segments.get().length > 0;\n }\n\n hasIconData() {\n return this.icon.segments.get().length > 0;\n }\n\n hasDebugData() {\n return this.textCollisionBox && this.iconCollisionBox;\n }\n\n hasTextCollisionBoxData() {\n return this.hasDebugData() && this.textCollisionBox.segments.get().length > 0;\n }\n\n hasIconCollisionBoxData() {\n return this.hasDebugData() && this.iconCollisionBox.segments.get().length > 0;\n }\n\n addIndicesForPlacedSymbol(iconOrText: SymbolBuffers, placedSymbolIndex: number) {\n const placedSymbol = iconOrText.placedSymbolArray.get(placedSymbolIndex);\n\n const endIndex = placedSymbol.vertexStartIndex + placedSymbol.numGlyphs * 4;\n for (let vertexIndex = placedSymbol.vertexStartIndex; vertexIndex < endIndex; vertexIndex += 4) {\n iconOrText.indexArray.emplaceBack(vertexIndex, vertexIndex + 1, vertexIndex + 2);\n iconOrText.indexArray.emplaceBack(vertexIndex + 1, vertexIndex + 2, vertexIndex + 3);\n }\n }\n\n getSortedSymbolIndexes(angle: number) {\n if (this.sortedAngle === angle && this.symbolInstanceIndexes !== undefined) {\n return this.symbolInstanceIndexes;\n }\n const sin = Math.sin(angle);\n const cos = Math.cos(angle);\n const rotatedYs = [];\n const featureIndexes = [];\n const result = [];\n\n for (let i = 0; i < this.symbolInstances.length; ++i) {\n result.push(i);\n const symbolInstance = this.symbolInstances.get(i);\n rotatedYs.push(Math.round(sin * symbolInstance.anchorX + cos * symbolInstance.anchorY) | 0);\n featureIndexes.push(symbolInstance.featureIndex);\n }\n\n result.sort((aIndex, bIndex) => {\n return (rotatedYs[aIndex] - rotatedYs[bIndex]) ||\n (featureIndexes[bIndex] - featureIndexes[aIndex]);\n });\n\n return result;\n }\n\n addToSortKeyRanges(symbolInstanceIndex: number, sortKey: number) {\n const last = this.sortKeyRanges[this.sortKeyRanges.length - 1];\n if (last && last.sortKey === sortKey) {\n last.symbolInstanceEnd = symbolInstanceIndex + 1;\n } else {\n this.sortKeyRanges.push({\n sortKey,\n symbolInstanceStart: symbolInstanceIndex,\n symbolInstanceEnd: symbolInstanceIndex + 1\n });\n }\n }\n\n sortFeatures(angle: number) {\n if (!this.sortFeaturesByY) return;\n if (this.sortedAngle === angle) return;\n\n // The current approach to sorting doesn't sort across segments so don't try.\n // Sorting within segments separately seemed not to be worth the complexity.\n if (this.text.segments.get().length > 1 || this.icon.segments.get().length > 1) return;\n\n // If the symbols are allowed to overlap sort them by their vertical screen position.\n // The index array buffer is rewritten to reference the (unchanged) vertices in the\n // sorted order.\n\n // To avoid sorting the actual symbolInstance array we sort an array of indexes.\n this.symbolInstanceIndexes = this.getSortedSymbolIndexes(angle);\n this.sortedAngle = angle;\n\n this.text.indexArray.clear();\n this.icon.indexArray.clear();\n\n this.featureSortOrder = [];\n\n for (const i of this.symbolInstanceIndexes) {\n const symbolInstance = this.symbolInstances.get(i);\n this.featureSortOrder.push(symbolInstance.featureIndex);\n\n [\n symbolInstance.rightJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.leftJustifiedTextSymbolIndex\n ].forEach((index, i, array) => {\n // Only add a given index the first time it shows up,\n // to avoid duplicate opacity entries when multiple justifications\n // share the same glyphs.\n if (index >= 0 && array.indexOf(index) === i) {\n this.addIndicesForPlacedSymbol(this.text, index);\n }\n });\n\n if (symbolInstance.verticalPlacedTextSymbolIndex >= 0) {\n this.addIndicesForPlacedSymbol(this.text, symbolInstance.verticalPlacedTextSymbolIndex);\n }\n\n if (symbolInstance.placedIconSymbolIndex >= 0) {\n this.addIndicesForPlacedSymbol(this.icon, symbolInstance.placedIconSymbolIndex);\n }\n\n if (symbolInstance.verticalPlacedIconSymbolIndex >= 0) {\n this.addIndicesForPlacedSymbol(this.icon, symbolInstance.verticalPlacedIconSymbolIndex);\n }\n }\n\n if (this.text.indexBuffer) this.text.indexBuffer.updateData(this.text.indexArray);\n if (this.icon.indexBuffer) this.icon.indexBuffer.updateData(this.icon.indexArray);\n }\n}\n\nregister('SymbolBucket', SymbolBucket, {\n omit: ['layers', 'collisionBoxArray', 'features', 'compareText']\n});\n\n// this constant is based on the size of StructArray indexes used in a symbol\n// bucket--namely, glyphOffsetArrayStart\n// eg the max valid UInt16 is 65,535\n// See https://github.com/mapbox/mapbox-gl-js/issues/2907 for motivation\n// lineStartIndex and textBoxStartIndex could potentially be concerns\n// but we expect there to be many fewer boxes/lines than glyphs\nSymbolBucket.MAX_GLYPHS = 65535;\n\nSymbolBucket.addDynamicAttributes = addDynamicAttributes;\n\nexport {addDynamicAttributes};\n","import type {SymbolFeature} from '../data/bucket/symbol_bucket';\n\nexport function mergeLines(features: Array): Array {\n const leftIndex: {[_: string]: number} = {};\n const rightIndex: {[_: string]: number} = {};\n const mergedFeatures = [];\n let mergedIndex = 0;\n\n function add(k) {\n mergedFeatures.push(features[k]);\n mergedIndex++;\n }\n\n function mergeFromRight(leftKey: string, rightKey: string, geom) {\n const i = rightIndex[leftKey];\n delete rightIndex[leftKey];\n rightIndex[rightKey] = i;\n\n mergedFeatures[i].geometry[0].pop();\n mergedFeatures[i].geometry[0] = mergedFeatures[i].geometry[0].concat(geom[0]);\n return i;\n }\n\n function mergeFromLeft(leftKey: string, rightKey: string, geom) {\n const i = leftIndex[rightKey];\n delete leftIndex[rightKey];\n leftIndex[leftKey] = i;\n\n mergedFeatures[i].geometry[0].shift();\n mergedFeatures[i].geometry[0] = geom[0].concat(mergedFeatures[i].geometry[0]);\n return i;\n }\n\n function getKey(text, geom, onRight?) {\n const point = onRight ? geom[0][geom[0].length - 1] : geom[0][0];\n return `${text}:${point.x}:${point.y}`;\n }\n\n for (let k = 0; k < features.length; k++) {\n const feature = features[k];\n const geom = feature.geometry;\n const text = feature.text ? feature.text.toString() : null;\n\n if (!text) {\n add(k);\n continue;\n }\n\n const leftKey = getKey(text, geom),\n rightKey = getKey(text, geom, true);\n\n if ((leftKey in rightIndex) && (rightKey in leftIndex) && (rightIndex[leftKey] !== leftIndex[rightKey])) {\n // found lines with the same text adjacent to both ends of the current line, merge all three\n const j = mergeFromLeft(leftKey, rightKey, geom);\n const i = mergeFromRight(leftKey, rightKey, mergedFeatures[j].geometry);\n\n delete leftIndex[leftKey];\n delete rightIndex[rightKey];\n\n rightIndex[getKey(text, mergedFeatures[i].geometry, true)] = i;\n mergedFeatures[j].geometry = null;\n\n } else if (leftKey in rightIndex) {\n // found mergeable line adjacent to the start of the current line, merge\n mergeFromRight(leftKey, rightKey, geom);\n\n } else if (rightKey in leftIndex) {\n // found mergeable line adjacent to the end of the current line, merge\n mergeFromLeft(leftKey, rightKey, geom);\n\n } else {\n // no adjacent lines, add as a new item\n add(k);\n leftIndex[leftKey] = mergedIndex - 1;\n rightIndex[rightKey] = mergedIndex - 1;\n }\n }\n\n return mergedFeatures.filter((f) => f.geometry);\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n ColorType\n } from '@maplibre/maplibre-gl-style-spec';\n \nexport type SymbolLayoutProps = {\n \"symbol-placement\": DataConstantProperty<\"point\" | \"line\" | \"line-center\">,\n \"symbol-spacing\": DataConstantProperty,\n \"symbol-avoid-edges\": DataConstantProperty,\n \"symbol-sort-key\": DataDrivenProperty,\n \"symbol-z-order\": DataConstantProperty<\"auto\" | \"viewport-y\" | \"source\">,\n \"icon-allow-overlap\": DataConstantProperty,\n \"icon-overlap\": DataConstantProperty<\"never\" | \"always\" | \"cooperative\">,\n \"icon-ignore-placement\": DataConstantProperty,\n \"icon-optional\": DataConstantProperty,\n \"icon-rotation-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"auto\">,\n \"icon-size\": DataDrivenProperty,\n \"icon-text-fit\": DataConstantProperty<\"none\" | \"width\" | \"height\" | \"both\">,\n \"icon-text-fit-padding\": DataConstantProperty<[number, number, number, number]>,\n \"icon-image\": DataDrivenProperty,\n \"icon-rotate\": DataDrivenProperty,\n \"icon-padding\": DataDrivenProperty,\n \"icon-keep-upright\": DataConstantProperty,\n \"icon-offset\": DataDrivenProperty<[number, number]>,\n \"icon-anchor\": DataDrivenProperty<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"icon-pitch-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"auto\">,\n \"text-pitch-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"auto\">,\n \"text-rotation-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"viewport-glyph\" | \"auto\">,\n \"text-field\": DataDrivenProperty,\n \"text-font\": DataDrivenProperty>,\n \"text-size\": DataDrivenProperty,\n \"text-max-width\": DataDrivenProperty,\n \"text-line-height\": DataConstantProperty,\n \"text-letter-spacing\": DataDrivenProperty,\n \"text-justify\": DataDrivenProperty<\"auto\" | \"left\" | \"center\" | \"right\">,\n \"text-radial-offset\": DataDrivenProperty,\n \"text-variable-anchor\": DataConstantProperty>,\n \"text-variable-anchor-offset\": DataDrivenProperty,\n \"text-anchor\": DataDrivenProperty<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"text-max-angle\": DataConstantProperty,\n \"text-writing-mode\": DataConstantProperty>,\n \"text-rotate\": DataDrivenProperty,\n \"text-padding\": DataConstantProperty,\n \"text-keep-upright\": DataConstantProperty,\n \"text-transform\": DataDrivenProperty<\"none\" | \"uppercase\" | \"lowercase\">,\n \"text-offset\": DataDrivenProperty<[number, number]>,\n \"text-allow-overlap\": DataConstantProperty,\n \"text-overlap\": DataConstantProperty<\"never\" | \"always\" | \"cooperative\">,\n \"text-ignore-placement\": DataConstantProperty,\n \"text-optional\": DataConstantProperty,\n};\n\nexport type SymbolLayoutPropsPossiblyEvaluated = {\n \"symbol-placement\": \"point\" | \"line\" | \"line-center\",\n \"symbol-spacing\": number,\n \"symbol-avoid-edges\": boolean,\n \"symbol-sort-key\": PossiblyEvaluatedPropertyValue,\n \"symbol-z-order\": \"auto\" | \"viewport-y\" | \"source\",\n \"icon-allow-overlap\": boolean,\n \"icon-overlap\": \"never\" | \"always\" | \"cooperative\",\n \"icon-ignore-placement\": boolean,\n \"icon-optional\": boolean,\n \"icon-rotation-alignment\": \"map\" | \"viewport\" | \"auto\",\n \"icon-size\": PossiblyEvaluatedPropertyValue,\n \"icon-text-fit\": \"none\" | \"width\" | \"height\" | \"both\",\n \"icon-text-fit-padding\": [number, number, number, number],\n \"icon-image\": PossiblyEvaluatedPropertyValue,\n \"icon-rotate\": PossiblyEvaluatedPropertyValue,\n \"icon-padding\": PossiblyEvaluatedPropertyValue,\n \"icon-keep-upright\": boolean,\n \"icon-offset\": PossiblyEvaluatedPropertyValue<[number, number]>,\n \"icon-anchor\": PossiblyEvaluatedPropertyValue<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"icon-pitch-alignment\": \"map\" | \"viewport\" | \"auto\",\n \"text-pitch-alignment\": \"map\" | \"viewport\" | \"auto\",\n \"text-rotation-alignment\": \"map\" | \"viewport\" | \"viewport-glyph\" | \"auto\",\n \"text-field\": PossiblyEvaluatedPropertyValue,\n \"text-font\": PossiblyEvaluatedPropertyValue>,\n \"text-size\": PossiblyEvaluatedPropertyValue,\n \"text-max-width\": PossiblyEvaluatedPropertyValue,\n \"text-line-height\": number,\n \"text-letter-spacing\": PossiblyEvaluatedPropertyValue,\n \"text-justify\": PossiblyEvaluatedPropertyValue<\"auto\" | \"left\" | \"center\" | \"right\">,\n \"text-radial-offset\": PossiblyEvaluatedPropertyValue,\n \"text-variable-anchor\": Array<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"text-variable-anchor-offset\": PossiblyEvaluatedPropertyValue,\n \"text-anchor\": PossiblyEvaluatedPropertyValue<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"text-max-angle\": number,\n \"text-writing-mode\": Array<\"horizontal\" | \"vertical\">,\n \"text-rotate\": PossiblyEvaluatedPropertyValue,\n \"text-padding\": number,\n \"text-keep-upright\": boolean,\n \"text-transform\": PossiblyEvaluatedPropertyValue<\"none\" | \"uppercase\" | \"lowercase\">,\n \"text-offset\": PossiblyEvaluatedPropertyValue<[number, number]>,\n \"text-allow-overlap\": boolean,\n \"text-overlap\": \"never\" | \"always\" | \"cooperative\",\n \"text-ignore-placement\": boolean,\n \"text-optional\": boolean,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"symbol-placement\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-placement\"] as any as StylePropertySpecification),\n \"symbol-spacing\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-spacing\"] as any as StylePropertySpecification),\n \"symbol-avoid-edges\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-avoid-edges\"] as any as StylePropertySpecification),\n \"symbol-sort-key\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"symbol-sort-key\"] as any as StylePropertySpecification),\n \"symbol-z-order\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-z-order\"] as any as StylePropertySpecification),\n \"icon-allow-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-allow-overlap\"] as any as StylePropertySpecification),\n \"icon-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-overlap\"] as any as StylePropertySpecification),\n \"icon-ignore-placement\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-ignore-placement\"] as any as StylePropertySpecification),\n \"icon-optional\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-optional\"] as any as StylePropertySpecification),\n \"icon-rotation-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-rotation-alignment\"] as any as StylePropertySpecification),\n \"icon-size\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-size\"] as any as StylePropertySpecification),\n \"icon-text-fit\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-text-fit\"] as any as StylePropertySpecification),\n \"icon-text-fit-padding\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-text-fit-padding\"] as any as StylePropertySpecification),\n \"icon-image\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-image\"] as any as StylePropertySpecification),\n \"icon-rotate\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-rotate\"] as any as StylePropertySpecification),\n \"icon-padding\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-padding\"] as any as StylePropertySpecification),\n \"icon-keep-upright\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-keep-upright\"] as any as StylePropertySpecification),\n \"icon-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-offset\"] as any as StylePropertySpecification),\n \"icon-anchor\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-anchor\"] as any as StylePropertySpecification),\n \"icon-pitch-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-pitch-alignment\"] as any as StylePropertySpecification),\n \"text-pitch-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-pitch-alignment\"] as any as StylePropertySpecification),\n \"text-rotation-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-rotation-alignment\"] as any as StylePropertySpecification),\n \"text-field\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-field\"] as any as StylePropertySpecification),\n \"text-font\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-font\"] as any as StylePropertySpecification),\n \"text-size\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-size\"] as any as StylePropertySpecification),\n \"text-max-width\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-max-width\"] as any as StylePropertySpecification),\n \"text-line-height\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-line-height\"] as any as StylePropertySpecification),\n \"text-letter-spacing\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-letter-spacing\"] as any as StylePropertySpecification),\n \"text-justify\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-justify\"] as any as StylePropertySpecification),\n \"text-radial-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-radial-offset\"] as any as StylePropertySpecification),\n \"text-variable-anchor\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-variable-anchor\"] as any as StylePropertySpecification),\n \"text-variable-anchor-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-variable-anchor-offset\"] as any as StylePropertySpecification),\n \"text-anchor\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-anchor\"] as any as StylePropertySpecification),\n \"text-max-angle\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-max-angle\"] as any as StylePropertySpecification),\n \"text-writing-mode\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-writing-mode\"] as any as StylePropertySpecification),\n \"text-rotate\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-rotate\"] as any as StylePropertySpecification),\n \"text-padding\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-padding\"] as any as StylePropertySpecification),\n \"text-keep-upright\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-keep-upright\"] as any as StylePropertySpecification),\n \"text-transform\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-transform\"] as any as StylePropertySpecification),\n \"text-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-offset\"] as any as StylePropertySpecification),\n \"text-allow-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-allow-overlap\"] as any as StylePropertySpecification),\n \"text-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-overlap\"] as any as StylePropertySpecification),\n \"text-ignore-placement\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-ignore-placement\"] as any as StylePropertySpecification),\n \"text-optional\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-optional\"] as any as StylePropertySpecification),\n});\n\nexport type SymbolPaintProps = {\n \"icon-opacity\": DataDrivenProperty,\n \"icon-color\": DataDrivenProperty,\n \"icon-halo-color\": DataDrivenProperty,\n \"icon-halo-width\": DataDrivenProperty,\n \"icon-halo-blur\": DataDrivenProperty,\n \"icon-translate\": DataConstantProperty<[number, number]>,\n \"icon-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"text-opacity\": DataDrivenProperty,\n \"text-color\": DataDrivenProperty,\n \"text-halo-color\": DataDrivenProperty,\n \"text-halo-width\": DataDrivenProperty,\n \"text-halo-blur\": DataDrivenProperty,\n \"text-translate\": DataConstantProperty<[number, number]>,\n \"text-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n};\n\nexport type SymbolPaintPropsPossiblyEvaluated = {\n \"icon-opacity\": PossiblyEvaluatedPropertyValue,\n \"icon-color\": PossiblyEvaluatedPropertyValue,\n \"icon-halo-color\": PossiblyEvaluatedPropertyValue,\n \"icon-halo-width\": PossiblyEvaluatedPropertyValue,\n \"icon-halo-blur\": PossiblyEvaluatedPropertyValue,\n \"icon-translate\": [number, number],\n \"icon-translate-anchor\": \"map\" | \"viewport\",\n \"text-opacity\": PossiblyEvaluatedPropertyValue,\n \"text-color\": PossiblyEvaluatedPropertyValue,\n \"text-halo-color\": PossiblyEvaluatedPropertyValue,\n \"text-halo-width\": PossiblyEvaluatedPropertyValue,\n \"text-halo-blur\": PossiblyEvaluatedPropertyValue,\n \"text-translate\": [number, number],\n \"text-translate-anchor\": \"map\" | \"viewport\",\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"icon-opacity\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-opacity\"] as any as StylePropertySpecification),\n \"icon-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-color\"] as any as StylePropertySpecification),\n \"icon-halo-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-halo-color\"] as any as StylePropertySpecification),\n \"icon-halo-width\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-halo-width\"] as any as StylePropertySpecification),\n \"icon-halo-blur\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-halo-blur\"] as any as StylePropertySpecification),\n \"icon-translate\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"icon-translate\"] as any as StylePropertySpecification),\n \"icon-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"icon-translate-anchor\"] as any as StylePropertySpecification),\n \"text-opacity\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-opacity\"] as any as StylePropertySpecification),\n \"text-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-color\"] as any as StylePropertySpecification, { runtimeType: ColorType, getOverride: (o) => o.textColor, hasOverride: (o) => !!o.textColor }),\n \"text-halo-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-halo-color\"] as any as StylePropertySpecification),\n \"text-halo-width\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-halo-width\"] as any as StylePropertySpecification),\n \"text-halo-blur\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-halo-blur\"] as any as StylePropertySpecification),\n \"text-translate\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"text-translate\"] as any as StylePropertySpecification),\n \"text-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"text-translate-anchor\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import type {Expression, EvaluationContext, Type, ZoomConstantExpression} from '@maplibre/maplibre-gl-style-spec';\nimport {NullType} from '@maplibre/maplibre-gl-style-spec';\nimport {PossiblyEvaluatedPropertyValue} from './properties';\nimport {register} from '../util/web_worker_transfer';\n\n// This is an internal expression class. It is only used in GL JS and\n// has GL JS dependencies which can break the standalone style-spec module\nexport class FormatSectionOverride implements Expression {\n type: Type;\n defaultValue: PossiblyEvaluatedPropertyValue;\n\n constructor(defaultValue: PossiblyEvaluatedPropertyValue) {\n if (defaultValue.property.overrides === undefined) throw new Error('overrides must be provided to instantiate FormatSectionOverride class');\n this.type = defaultValue.property.overrides ? defaultValue.property.overrides.runtimeType : NullType;\n this.defaultValue = defaultValue;\n }\n\n evaluate(ctx: EvaluationContext) {\n if (ctx.formattedSection) {\n const overrides = this.defaultValue.property.overrides;\n if (overrides && overrides.hasOverride(ctx.formattedSection)) {\n return overrides.getOverride(ctx.formattedSection);\n }\n }\n\n if (ctx.feature && ctx.featureState) {\n return this.defaultValue.evaluate(ctx.feature, ctx.featureState);\n }\n\n return this.defaultValue.property.specification.default;\n }\n\n eachChild(fn: (_: Expression) => void) {\n if (!this.defaultValue.isConstant()) {\n const expr: ZoomConstantExpression<'source'> = (this.defaultValue.value as any);\n fn(expr._styleExpression.expression);\n }\n }\n\n // Cannot be statically evaluated, as the output depends on the evaluation context.\n outputDefined() {\n return false;\n }\n\n serialize() {\n return null;\n }\n}\n\nregister('FormatSectionOverride', FormatSectionOverride, {omit: ['defaultValue']});\n","import {StyleLayer} from '../style_layer';\n\nimport {SymbolBucket, SymbolFeature} from '../../data/bucket/symbol_bucket';\nimport {resolveTokens} from '../../util/resolve_tokens';\nimport properties, {SymbolLayoutPropsPossiblyEvaluated, SymbolPaintPropsPossiblyEvaluated} from './symbol_style_layer_properties.g';\n\nimport {\n Transitionable,\n Transitioning,\n Layout,\n PossiblyEvaluated,\n PossiblyEvaluatedPropertyValue,\n PropertyValue\n} from '../properties';\n\nimport {\n isExpression,\n StyleExpression,\n ZoomConstantExpression,\n ZoomDependentExpression,\n FormattedType,\n typeOf,\n Formatted,\n FormatExpression,\n Literal} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {BucketParameters} from '../../data/bucket';\nimport type {SymbolLayoutProps, SymbolPaintProps} from './symbol_style_layer_properties.g';\nimport type {EvaluationParameters} from '../evaluation_parameters';\nimport type {Expression, Feature, SourceExpression, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport {FormatSectionOverride} from '../format_section_override';\n\nexport class SymbolStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n super.recalculate(parameters, availableImages);\n\n if (this.layout.get('icon-rotation-alignment') === 'auto') {\n if (this.layout.get('symbol-placement') !== 'point') {\n this.layout._values['icon-rotation-alignment'] = 'map';\n } else {\n this.layout._values['icon-rotation-alignment'] = 'viewport';\n }\n }\n\n if (this.layout.get('text-rotation-alignment') === 'auto') {\n if (this.layout.get('symbol-placement') !== 'point') {\n this.layout._values['text-rotation-alignment'] = 'map';\n } else {\n this.layout._values['text-rotation-alignment'] = 'viewport';\n }\n }\n\n // If unspecified, `*-pitch-alignment` inherits `*-rotation-alignment`\n if (this.layout.get('text-pitch-alignment') === 'auto') {\n this.layout._values['text-pitch-alignment'] = this.layout.get('text-rotation-alignment') === 'map' ? 'map' : 'viewport';\n }\n if (this.layout.get('icon-pitch-alignment') === 'auto') {\n this.layout._values['icon-pitch-alignment'] = this.layout.get('icon-rotation-alignment');\n }\n\n if (this.layout.get('symbol-placement') === 'point') {\n const writingModes = this.layout.get('text-writing-mode');\n if (writingModes) {\n // remove duplicates, preserving order\n const deduped = [];\n for (const m of writingModes) {\n if (deduped.indexOf(m) < 0) deduped.push(m);\n }\n this.layout._values['text-writing-mode'] = deduped;\n } else {\n this.layout._values['text-writing-mode'] = ['horizontal'];\n }\n }\n\n this._setPaintOverrides();\n }\n\n getValueAndResolveTokens(name: any, feature: Feature, canonical: CanonicalTileID, availableImages: Array) {\n const value = this.layout.get(name).evaluate(feature, {}, canonical, availableImages);\n const unevaluated = this._unevaluatedLayout._values[name];\n if (!unevaluated.isDataDriven() && !isExpression(unevaluated.value) && value) {\n return resolveTokens(feature.properties, value);\n }\n\n return value;\n }\n\n createBucket(parameters: BucketParameters) {\n return new SymbolBucket(parameters);\n }\n\n queryRadius(): number {\n return 0;\n }\n\n queryIntersectsFeature(): boolean {\n throw new Error('Should take a different path in FeatureIndex');\n }\n\n _setPaintOverrides() {\n for (const overridable of properties.paint.overridableProperties) {\n if (!SymbolStyleLayer.hasPaintOverride(this.layout, overridable)) {\n continue;\n }\n const overriden = this.paint.get(overridable as keyof SymbolPaintPropsPossiblyEvaluated) as PossiblyEvaluatedPropertyValue;\n const override = new FormatSectionOverride(overriden);\n const styleExpression = new StyleExpression(override, overriden.property.specification);\n let expression = null;\n if (overriden.value.kind === 'constant' || overriden.value.kind === 'source') {\n expression = new ZoomConstantExpression('source', styleExpression) as SourceExpression;\n } else {\n expression = new ZoomDependentExpression('composite',\n styleExpression,\n overriden.value.zoomStops);\n }\n this.paint._values[overridable] = new PossiblyEvaluatedPropertyValue(overriden.property,\n expression,\n overriden.parameters);\n }\n }\n\n _handleOverridablePaintPropertyUpdate(name: string, oldValue: PropertyValue, newValue: PropertyValue): boolean {\n if (!this.layout || oldValue.isDataDriven() || newValue.isDataDriven()) {\n return false;\n }\n return SymbolStyleLayer.hasPaintOverride(this.layout, name);\n }\n\n static hasPaintOverride(layout: PossiblyEvaluated, propertyName: string): boolean {\n const textField = layout.get('text-field');\n const property = properties.paint.properties[propertyName];\n let hasOverrides = false;\n\n const checkSections = (sections) => {\n for (const section of sections) {\n if (property.overrides && property.overrides.hasOverride(section)) {\n hasOverrides = true;\n return;\n }\n }\n };\n\n if (textField.value.kind === 'constant' && textField.value.value instanceof Formatted) {\n checkSections(textField.value.value.sections);\n } else if (textField.value.kind === 'source') {\n\n const checkExpression = (expression: Expression) => {\n if (hasOverrides) return;\n\n if (expression instanceof Literal && typeOf(expression.value) === FormattedType) {\n const formatted: Formatted = (expression.value as any);\n checkSections(formatted.sections);\n } else if (expression instanceof FormatExpression) {\n checkSections(expression.sections);\n } else {\n expression.eachChild(checkExpression);\n }\n };\n\n const expr: ZoomConstantExpression<'source'> = (textField.value as any);\n if (expr._styleExpression) {\n checkExpression(expr._styleExpression.expression);\n }\n }\n\n return hasOverrides;\n }\n}\n\nexport type SymbolPadding = [number, number, number, number];\n\nexport function getIconPadding(layout: PossiblyEvaluated, feature: SymbolFeature, canonical: CanonicalTileID, pixelRatio = 1): SymbolPadding {\n // Support text-padding in addition to icon-padding? Unclear how to apply asymmetric text-padding to the radius for collision circles.\n const result = layout.get('icon-padding').evaluate(feature, {}, canonical);\n const values = result && result.values;\n\n return [\n values[0] * pixelRatio,\n values[1] * pixelRatio,\n values[2] * pixelRatio,\n values[3] * pixelRatio,\n ];\n}\n","/**\n * Replace tokens in a string template with values in an object\n *\n * @param properties - a key/value relationship between tokens and replacements\n * @param text - the template string\n * @returns the template with tokens replaced\n */\nexport function resolveTokens(\n properties: {\n readonly [x: string]: unknown;\n } | null,\n text: string\n): string {\n return text.replace(/{([^{}]+)}/g, (match, key: string) => {\n return properties && key in properties ? String(properties[key]) : '';\n });\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type BackgroundPaintProps = {\n \"background-color\": DataConstantProperty,\n \"background-pattern\": CrossFadedProperty,\n \"background-opacity\": DataConstantProperty,\n};\n\nexport type BackgroundPaintPropsPossiblyEvaluated = {\n \"background-color\": Color,\n \"background-pattern\": CrossFaded,\n \"background-opacity\": number,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"background-color\": new DataConstantProperty(styleSpec[\"paint_background\"][\"background-color\"] as any as StylePropertySpecification),\n \"background-pattern\": new CrossFadedProperty(styleSpec[\"paint_background\"][\"background-pattern\"] as any as StylePropertySpecification),\n \"background-opacity\": new DataConstantProperty(styleSpec[\"paint_background\"][\"background-opacity\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport properties, {BackgroundPaintPropsPossiblyEvaluated} from './background_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {BackgroundPaintProps} from './background_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport class BackgroundStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type RasterPaintProps = {\n \"raster-opacity\": DataConstantProperty,\n \"raster-hue-rotate\": DataConstantProperty,\n \"raster-brightness-min\": DataConstantProperty,\n \"raster-brightness-max\": DataConstantProperty,\n \"raster-saturation\": DataConstantProperty,\n \"raster-contrast\": DataConstantProperty,\n \"raster-resampling\": DataConstantProperty<\"linear\" | \"nearest\">,\n \"raster-fade-duration\": DataConstantProperty,\n};\n\nexport type RasterPaintPropsPossiblyEvaluated = {\n \"raster-opacity\": number,\n \"raster-hue-rotate\": number,\n \"raster-brightness-min\": number,\n \"raster-brightness-max\": number,\n \"raster-saturation\": number,\n \"raster-contrast\": number,\n \"raster-resampling\": \"linear\" | \"nearest\",\n \"raster-fade-duration\": number,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"raster-opacity\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-opacity\"] as any as StylePropertySpecification),\n \"raster-hue-rotate\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-hue-rotate\"] as any as StylePropertySpecification),\n \"raster-brightness-min\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-brightness-min\"] as any as StylePropertySpecification),\n \"raster-brightness-max\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-brightness-max\"] as any as StylePropertySpecification),\n \"raster-saturation\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-saturation\"] as any as StylePropertySpecification),\n \"raster-contrast\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-contrast\"] as any as StylePropertySpecification),\n \"raster-resampling\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-resampling\"] as any as StylePropertySpecification),\n \"raster-fade-duration\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-fade-duration\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport properties, {RasterPaintPropsPossiblyEvaluated} from './raster_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {RasterPaintProps} from './raster_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport class RasterStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n}\n","import {StyleLayer} from '../style_layer';\nimport type {Map} from '../../ui/map';\nimport {mat4} from 'gl-matrix';\nimport {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * @param gl - The map's gl context.\n * @param matrix - The map's camera matrix. It projects spherical mercator\n * coordinates to gl coordinates. The spherical mercator coordinate `[0, 0]` represents the\n * top left corner of the mercator world and `[1, 1]` represents the bottom right corner. When\n * the `renderingMode` is `\"3d\"`, the z coordinate is conformal. A box with identical x, y, and z\n * lengths in mercator units would be rendered as a cube. {@link MercatorCoordinate.fromLngLat}\n * can be used to project a `LngLat` to a mercator coordinate.\n */\ntype CustomRenderMethod = (gl: WebGLRenderingContext|WebGL2RenderingContext, matrix: mat4) => void;\n\n/**\n * Interface for custom style layers. This is a specification for\n * implementers to model: it is not an exported method or class.\n *\n * Custom layers allow a user to render directly into the map's GL context using the map's camera.\n * These layers can be added between any regular layers using {@link Map#addLayer}.\n *\n * Custom layers must have a unique `id` and must have the `type` of `\"custom\"`.\n * They must implement `render` and may implement `prerender`, `onAdd` and `onRemove`.\n * They can trigger rendering using {@link Map#triggerRepaint}\n * and they should appropriately handle {@link MapContextEvent} with `webglcontextlost` and `webglcontextrestored`.\n *\n * The `renderingMode` property controls whether the layer is treated as a `\"2d\"` or `\"3d\"` map layer. Use:\n * - `\"renderingMode\": \"3d\"` to use the depth buffer and share it with other layers\n * - `\"renderingMode\": \"2d\"` to add a layer with no depth. If you need to use the depth buffer for a `\"2d\"` layer you must use an offscreen\n * framebuffer and {@link CustomLayerInterface#prerender}\n *\n * @example\n * Custom layer implemented as ES6 class\n * ```ts\n * class NullIslandLayer {\n * constructor() {\n * this.id = 'null-island';\n * this.type = 'custom';\n * this.renderingMode = '2d';\n * }\n *\n * onAdd(map, gl) {\n * const vertexSource = `\n * uniform mat4 u_matrix;\n * void main() {\n * gl_Position = u_matrix * vec4(0.5, 0.5, 0.0, 1.0);\n * gl_PointSize = 20.0;\n * }`;\n *\n * const fragmentSource = `\n * void main() {\n * fragColor = vec4(1.0, 0.0, 0.0, 1.0);\n * }`;\n *\n * const vertexShader = gl.createShader(gl.VERTEX_SHADER);\n * gl.shaderSource(vertexShader, vertexSource);\n * gl.compileShader(vertexShader);\n * const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);\n * gl.shaderSource(fragmentShader, fragmentSource);\n * gl.compileShader(fragmentShader);\n *\n * this.program = gl.createProgram();\n * gl.attachShader(this.program, vertexShader);\n * gl.attachShader(this.program, fragmentShader);\n * gl.linkProgram(this.program);\n * }\n *\n * render(gl, matrix) {\n * gl.useProgram(this.program);\n * gl.uniformMatrix4fv(gl.getUniformLocation(this.program, \"u_matrix\"), false, matrix);\n * gl.drawArrays(gl.POINTS, 0, 1);\n * }\n * }\n *\n * map.on('load', function() {\n * map.addLayer(new NullIslandLayer());\n * });\n * ```\n */\nexport interface CustomLayerInterface {\n /**\n * A unique layer id.\n */\n id: string;\n /**\n * The layer's type. Must be `\"custom\"`.\n */\n type: 'custom';\n /**\n * Either `\"2d\"` or `\"3d\"`. Defaults to `\"2d\"`.\n */\n renderingMode?: '2d' | '3d';\n /**\n * Called during a render frame allowing the layer to draw into the GL context.\n *\n * The layer can assume blending and depth state is set to allow the layer to properly\n * blend and clip other layers. The layer cannot make any other assumptions about the\n * current GL state.\n *\n * If the layer needs to render to a texture, it should implement the `prerender` method\n * to do this and only use the `render` method for drawing directly into the main framebuffer.\n *\n * The blend function is set to `gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA)`. This expects\n * colors to be provided in premultiplied alpha form where the `r`, `g` and `b` values are already\n * multiplied by the `a` value. If you are unable to provide colors in premultiplied form you\n * may want to change the blend function to\n * `gl.blendFuncSeparate(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA, gl.ONE, gl.ONE_MINUS_SRC_ALPHA)`.\n */\n render: CustomRenderMethod;\n /**\n * Optional method called during a render frame to allow a layer to prepare resources or render into a texture.\n *\n * The layer cannot make any assumptions about the current GL state and must bind a framebuffer before rendering.\n */\n prerender?: CustomRenderMethod;\n /**\n * Optional method called when the layer has been added to the Map with {@link Map#addLayer}. This\n * gives the layer a chance to initialize gl resources and register event listeners.\n *\n * @param map - The Map this custom layer was just added to.\n * @param gl - The gl context for the map.\n */\n onAdd?(map: Map, gl: WebGLRenderingContext | WebGL2RenderingContext): void;\n /**\n * Optional method called when the layer has been removed from the Map with {@link Map#removeLayer}. This\n * gives the layer a chance to clean up gl resources and event listeners.\n *\n * @param map - The Map this custom layer was just added to.\n * @param gl - The gl context for the map.\n */\n onRemove?(map: Map, gl: WebGLRenderingContext | WebGL2RenderingContext): void;\n}\n\nexport function validateCustomStyleLayer(layerObject: CustomLayerInterface) {\n const errors = [];\n const id = layerObject.id;\n\n if (id === undefined) {\n errors.push({\n message: `layers.${id}: missing required property \"id\"`\n });\n }\n\n if (layerObject.render === undefined) {\n errors.push({\n message: `layers.${id}: missing required method \"render\"`\n });\n }\n\n if (layerObject.renderingMode &&\n layerObject.renderingMode !== '2d' &&\n layerObject.renderingMode !== '3d') {\n errors.push({\n message: `layers.${id}: property \"renderingMode\" must be either \"2d\" or \"3d\"`\n });\n }\n\n return errors;\n}\n\nexport class CustomStyleLayer extends StyleLayer {\n\n implementation: CustomLayerInterface;\n\n constructor(implementation: CustomLayerInterface) {\n super(implementation, {});\n this.implementation = implementation;\n }\n\n is3D() {\n return this.implementation.renderingMode === '3d';\n }\n\n hasOffscreenPass() {\n return this.implementation.prerender !== undefined;\n }\n\n recalculate() {}\n updateTransitions() {}\n hasTransition() { return false; }\n\n serialize(): LayerSpecification {\n throw new Error('Custom layers cannot be serialized');\n }\n\n onAdd = (map: Map) => {\n if (this.implementation.onAdd) {\n this.implementation.onAdd(map, map.painter.context.gl);\n }\n };\n\n onRemove = (map: Map) => {\n if (this.implementation.onRemove) {\n this.implementation.onRemove(map, map.painter.context.gl);\n }\n };\n}\n","import {CircleStyleLayer} from './style_layer/circle_style_layer';\nimport {HeatmapStyleLayer} from './style_layer/heatmap_style_layer';\nimport {HillshadeStyleLayer} from './style_layer/hillshade_style_layer';\nimport {FillStyleLayer} from './style_layer/fill_style_layer';\nimport {FillExtrusionStyleLayer} from './style_layer/fill_extrusion_style_layer';\nimport {LineStyleLayer} from './style_layer/line_style_layer';\nimport {SymbolStyleLayer} from './style_layer/symbol_style_layer';\nimport {BackgroundStyleLayer} from './style_layer/background_style_layer';\nimport {RasterStyleLayer} from './style_layer/raster_style_layer';\nimport {CustomStyleLayer, type CustomLayerInterface} from './style_layer/custom_style_layer';\n\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport function createStyleLayer(layer: LayerSpecification | CustomLayerInterface) {\n if (layer.type === 'custom') {\n return new CustomStyleLayer(layer);\n }\n switch (layer.type) {\n case 'background':\n return new BackgroundStyleLayer(layer);\n case 'circle':\n return new CircleStyleLayer(layer);\n case 'fill':\n return new FillStyleLayer(layer);\n case 'fill-extrusion':\n return new FillExtrusionStyleLayer(layer);\n case 'heatmap':\n return new HeatmapStyleLayer(layer);\n case 'hillshade':\n return new HillshadeStyleLayer(layer);\n case 'line':\n return new LineStyleLayer(layer);\n case 'raster':\n return new RasterStyleLayer(layer);\n case 'symbol':\n return new SymbolStyleLayer(layer);\n }\n}\n\n","import {StyleLayer} from './style_layer';\nimport {createStyleLayer} from './create_style_layer';\n\nimport {featureFilter, groupByLayout} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {TypedStyleLayer} from './style_layer/typed_style_layer';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type LayerConfigs = {[_: string]: LayerSpecification};\nexport type Family = Array;\n\nexport class StyleLayerIndex {\n familiesBySource: {\n [source: string]: {\n [sourceLayer: string]: Array>;\n };\n };\n keyCache: {[source: string]: string};\n\n _layerConfigs: LayerConfigs;\n _layers: {[_: string]: StyleLayer};\n\n constructor(layerConfigs?: Array | null) {\n this.keyCache = {};\n if (layerConfigs) {\n this.replace(layerConfigs);\n }\n }\n\n replace(layerConfigs: Array) {\n this._layerConfigs = {};\n this._layers = {};\n this.update(layerConfigs, []);\n }\n\n update(layerConfigs: Array, removedIds: Array) {\n for (const layerConfig of layerConfigs) {\n this._layerConfigs[layerConfig.id] = layerConfig;\n\n const layer = this._layers[layerConfig.id] = createStyleLayer(layerConfig);\n layer._featureFilter = featureFilter(layer.filter);\n if (this.keyCache[layerConfig.id])\n delete this.keyCache[layerConfig.id];\n }\n for (const id of removedIds) {\n delete this.keyCache[id];\n delete this._layerConfigs[id];\n delete this._layers[id];\n }\n\n this.familiesBySource = {};\n\n const groups = groupByLayout(Object.values(this._layerConfigs), this.keyCache);\n\n for (const layerConfigs of groups) {\n const layers = layerConfigs.map((layerConfig) => this._layers[layerConfig.id]);\n\n const layer = layers[0];\n if (layer.visibility === 'none') {\n continue;\n }\n\n const sourceId = layer.source || '';\n let sourceGroup = this.familiesBySource[sourceId];\n if (!sourceGroup) {\n sourceGroup = this.familiesBySource[sourceId] = {};\n }\n\n const sourceLayerId = layer.sourceLayer || '_geojsonTileLayer';\n let sourceLayerFamilies = sourceGroup[sourceLayerId];\n if (!sourceLayerFamilies) {\n sourceLayerFamilies = sourceGroup[sourceLayerId] = [];\n }\n\n sourceLayerFamilies.push(layers);\n }\n }\n}\n","export class DictionaryCoder {\n _stringToNumber: {[_: string]: number};\n _numberToString: Array;\n\n constructor(strings: Array) {\n this._stringToNumber = {};\n this._numberToString = [];\n for (let i = 0; i < strings.length; i++) {\n const string = strings[i];\n this._stringToNumber[string] = i;\n this._numberToString[i] = string;\n }\n }\n\n encode(string: string) {\n return this._stringToNumber[string];\n }\n\n decode(n: number) {\n if (n >= this._numberToString.length) throw new Error(`Out of bounds. Index requested n=${n} can't be >= this._numberToString.length ${this._numberToString.length}`);\n return this._numberToString[n];\n }\n}\n","import type {VectorTileFeature} from '@mapbox/vector-tile';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A helper for type to omit a property from a type\n */\ntype DistributiveKeys = T extends T ? keyof T : never;\n/**\n * A helper for type to omit a property from a type\n */\ntype DistributiveOmit> = T extends unknown\n ? Omit\n : never;\n\n/**\n * An extended geojson feature used by the events to return data to the listener\n */\nexport type MapGeoJSONFeature = GeoJSONFeature & {\n layer: DistributiveOmit & {source: string};\n source: string;\n sourceLayer?: string;\n state: { [key: string]: any };\n}\n\n/**\n * A geojson feature\n */\nexport class GeoJSONFeature {\n type: 'Feature';\n _geometry: GeoJSON.Geometry;\n properties: { [name: string]: any };\n id: number | string | undefined;\n\n _vectorTileFeature: VectorTileFeature;\n\n constructor(vectorTileFeature: VectorTileFeature, z: number, x: number, y: number, id: string | number | undefined) {\n this.type = 'Feature';\n\n this._vectorTileFeature = vectorTileFeature;\n (vectorTileFeature as any)._z = z;\n (vectorTileFeature as any)._x = x;\n (vectorTileFeature as any)._y = y;\n\n this.properties = vectorTileFeature.properties;\n this.id = id;\n }\n\n get geometry(): GeoJSON.Geometry {\n if (this._geometry === undefined) {\n this._geometry = this._vectorTileFeature.toGeoJSON(\n (this._vectorTileFeature as any)._x,\n (this._vectorTileFeature as any)._y,\n (this._vectorTileFeature as any)._z).geometry;\n }\n return this._geometry;\n }\n\n set geometry(g: GeoJSON.Geometry) {\n this._geometry = g;\n }\n\n toJSON() {\n const json: any = {\n geometry: this.geometry\n };\n for (const i in this) {\n if (i === '_geometry' || i === '_vectorTileFeature') continue;\n json[i] = (this)[i];\n }\n return json;\n }\n}\n","import Point from '@mapbox/point-geometry';\nimport {loadGeometry} from './load_geometry';\nimport {toEvaluationFeature} from './evaluation_feature';\nimport {EXTENT} from './extent';\nimport {featureFilter} from '@maplibre/maplibre-gl-style-spec';\nimport {TransferableGridIndex} from '../util/transferable_grid_index';\nimport {DictionaryCoder} from '../util/dictionary_coder';\nimport vt from '@mapbox/vector-tile';\nimport Protobuf from 'pbf';\nimport {GeoJSONFeature} from '../util/vectortile_to_geojson';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\nimport {arraysIntersect, mapObject, extend} from '../util/util';\nimport {OverscaledTileID} from '../source/tile_id';\nimport {register} from '../util/web_worker_transfer';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {SourceFeatureState} from '../source/source_state';\nimport {polygonIntersectsBox} from '../util/intersection_tests';\nimport {PossiblyEvaluated} from '../style/properties';\nimport {FeatureIndexArray} from './array_types.g';\nimport {mat4} from 'gl-matrix';\n\nimport type {StyleLayer} from '../style/style_layer';\nimport type {FeatureFilter, FeatureState, FilterSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Transform} from '../geo/transform';\nimport type {VectorTileFeature, VectorTileLayer} from '@mapbox/vector-tile';\n\ntype QueryParameters = {\n scale: number;\n pixelPosMatrix: mat4;\n transform: Transform;\n tileSize: number;\n queryGeometry: Array;\n cameraQueryGeometry: Array;\n queryPadding: number;\n params: {\n filter: FilterSpecification;\n layers: Array;\n availableImages: Array;\n };\n};\n\n/**\n * @internal\n * An in memory index class to allow fast interaction with features\n */\nexport class FeatureIndex {\n tileID: OverscaledTileID;\n x: number;\n y: number;\n z: number;\n grid: TransferableGridIndex;\n grid3D: TransferableGridIndex;\n featureIndexArray: FeatureIndexArray;\n promoteId?: PromoteIdSpecification;\n\n rawTileData: ArrayBuffer;\n bucketLayerIDs: Array>;\n\n vtLayers: {[_: string]: VectorTileLayer};\n sourceLayerCoder: DictionaryCoder;\n\n constructor(tileID: OverscaledTileID, promoteId?: PromoteIdSpecification | null) {\n this.tileID = tileID;\n this.x = tileID.canonical.x;\n this.y = tileID.canonical.y;\n this.z = tileID.canonical.z;\n this.grid = new TransferableGridIndex(EXTENT, 16, 0);\n this.grid3D = new TransferableGridIndex(EXTENT, 16, 0);\n this.featureIndexArray = new FeatureIndexArray();\n this.promoteId = promoteId;\n }\n\n insert(feature: VectorTileFeature, geometry: Array>, featureIndex: number, sourceLayerIndex: number, bucketIndex: number, is3D?: boolean) {\n const key = this.featureIndexArray.length;\n this.featureIndexArray.emplaceBack(featureIndex, sourceLayerIndex, bucketIndex);\n\n const grid = is3D ? this.grid3D : this.grid;\n\n for (let r = 0; r < geometry.length; r++) {\n const ring = geometry[r];\n\n const bbox = [Infinity, Infinity, -Infinity, -Infinity];\n for (let i = 0; i < ring.length; i++) {\n const p = ring[i];\n bbox[0] = Math.min(bbox[0], p.x);\n bbox[1] = Math.min(bbox[1], p.y);\n bbox[2] = Math.max(bbox[2], p.x);\n bbox[3] = Math.max(bbox[3], p.y);\n }\n\n if (bbox[0] < EXTENT &&\n bbox[1] < EXTENT &&\n bbox[2] >= 0 &&\n bbox[3] >= 0) {\n grid.insert(key, bbox[0], bbox[1], bbox[2], bbox[3]);\n }\n }\n }\n\n loadVTLayers(): {[_: string]: VectorTileLayer} {\n if (!this.vtLayers) {\n this.vtLayers = new vt.VectorTile(new Protobuf(this.rawTileData)).layers;\n this.sourceLayerCoder = new DictionaryCoder(this.vtLayers ? Object.keys(this.vtLayers).sort() : ['_geojsonTileLayer']);\n }\n return this.vtLayers;\n }\n\n // Finds non-symbol features in this tile at a particular position.\n query(\n args: QueryParameters,\n styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n sourceFeatureState: SourceFeatureState\n ): {[_: string]: Array<{featureIndex: number; feature: GeoJSONFeature}>} {\n this.loadVTLayers();\n\n const params = args.params || {} as { filter: any; layers: string[]; availableImages: string[] },\n pixelsToTileUnits = EXTENT / args.tileSize / args.scale,\n filter = featureFilter(params.filter);\n\n const queryGeometry = args.queryGeometry;\n const queryPadding = args.queryPadding * pixelsToTileUnits;\n\n const bounds = getBounds(queryGeometry);\n const matching = this.grid.query(bounds.minX - queryPadding, bounds.minY - queryPadding, bounds.maxX + queryPadding, bounds.maxY + queryPadding);\n\n const cameraBounds = getBounds(args.cameraQueryGeometry);\n const matching3D = this.grid3D.query(\n cameraBounds.minX - queryPadding, cameraBounds.minY - queryPadding, cameraBounds.maxX + queryPadding, cameraBounds.maxY + queryPadding,\n (bx1, by1, bx2, by2) => {\n return polygonIntersectsBox(args.cameraQueryGeometry, bx1 - queryPadding, by1 - queryPadding, bx2 + queryPadding, by2 + queryPadding);\n });\n\n for (const key of matching3D) {\n matching.push(key);\n }\n\n matching.sort(topDownFeatureComparator);\n\n const result = {};\n let previousIndex;\n for (let k = 0; k < matching.length; k++) {\n const index = matching[k];\n\n // don't check the same feature more than once\n if (index === previousIndex) continue;\n previousIndex = index;\n\n const match = this.featureIndexArray.get(index);\n let featureGeometry = null;\n this.loadMatchingFeature(\n result,\n match.bucketIndex,\n match.sourceLayerIndex,\n match.featureIndex,\n filter,\n params.layers,\n params.availableImages,\n styleLayers,\n serializedLayers,\n sourceFeatureState,\n (feature: VectorTileFeature, styleLayer: StyleLayer, featureState: FeatureState) => {\n if (!featureGeometry) {\n featureGeometry = loadGeometry(feature);\n }\n\n return styleLayer.queryIntersectsFeature(queryGeometry, feature, featureState, featureGeometry, this.z, args.transform, pixelsToTileUnits, args.pixelPosMatrix);\n }\n );\n }\n\n return result;\n }\n\n loadMatchingFeature(\n result: {\n [_: string]: Array<{\n featureIndex: number;\n feature: GeoJSONFeature;\n intersectionZ?: boolean | number;\n }>;\n },\n bucketIndex: number,\n sourceLayerIndex: number,\n featureIndex: number,\n filter: FeatureFilter,\n filterLayerIDs: Array,\n availableImages: Array,\n styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n sourceFeatureState?: SourceFeatureState,\n intersectionTest?: (\n feature: VectorTileFeature,\n styleLayer: StyleLayer,\n featureState: any,\n id: string | number | void\n ) => boolean | number) {\n\n const layerIDs = this.bucketLayerIDs[bucketIndex];\n if (filterLayerIDs && !arraysIntersect(filterLayerIDs, layerIDs))\n return;\n\n const sourceLayerName = this.sourceLayerCoder.decode(sourceLayerIndex);\n const sourceLayer = this.vtLayers[sourceLayerName];\n const feature = sourceLayer.feature(featureIndex);\n\n if (filter.needGeometry) {\n const evaluationFeature = toEvaluationFeature(feature, true);\n if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), evaluationFeature, this.tileID.canonical)) {\n return;\n }\n } else if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), feature)) {\n return;\n }\n\n const id = this.getId(feature, sourceLayerName);\n\n for (let l = 0; l < layerIDs.length; l++) {\n const layerID = layerIDs[l];\n\n if (filterLayerIDs && filterLayerIDs.indexOf(layerID) < 0) {\n continue;\n }\n\n const styleLayer = styleLayers[layerID];\n\n if (!styleLayer) continue;\n\n let featureState = {};\n if (id && sourceFeatureState) {\n // `feature-state` expression evaluation requires feature state to be available\n featureState = sourceFeatureState.getState(styleLayer.sourceLayer || '_geojsonTileLayer', id);\n }\n\n const serializedLayer = extend({}, serializedLayers[layerID]);\n\n serializedLayer.paint = evaluateProperties(serializedLayer.paint, styleLayer.paint, feature, featureState, availableImages);\n serializedLayer.layout = evaluateProperties(serializedLayer.layout, styleLayer.layout, feature, featureState, availableImages);\n\n const intersectionZ = !intersectionTest || intersectionTest(feature, styleLayer, featureState);\n if (!intersectionZ) {\n // Only applied for non-symbol features\n continue;\n }\n\n const geojsonFeature = new GeoJSONFeature(feature, this.z, this.x, this.y, id) as MapGeoJSONFeature;\n geojsonFeature.layer = serializedLayer;\n let layerResult = result[layerID];\n if (layerResult === undefined) {\n layerResult = result[layerID] = [];\n }\n layerResult.push({featureIndex, feature: geojsonFeature, intersectionZ});\n }\n }\n\n // Given a set of symbol indexes that have already been looked up,\n // return a matching set of GeoJSONFeatures\n lookupSymbolFeatures(symbolFeatureIndexes: Array,\n serializedLayers: {[_: string]: StyleLayer},\n bucketIndex: number,\n sourceLayerIndex: number,\n filterSpec: FilterSpecification,\n filterLayerIDs: Array,\n availableImages: Array,\n styleLayers: {[_: string]: StyleLayer}) {\n const result = {};\n this.loadVTLayers();\n\n const filter = featureFilter(filterSpec);\n\n for (const symbolFeatureIndex of symbolFeatureIndexes) {\n this.loadMatchingFeature(\n result,\n bucketIndex,\n sourceLayerIndex,\n symbolFeatureIndex,\n filter,\n filterLayerIDs,\n availableImages,\n styleLayers,\n serializedLayers\n );\n\n }\n return result;\n }\n\n hasLayer(id: string) {\n for (const layerIDs of this.bucketLayerIDs) {\n for (const layerID of layerIDs) {\n if (id === layerID) return true;\n }\n }\n\n return false;\n }\n\n getId(feature: VectorTileFeature, sourceLayerId: string): string | number {\n let id: string | number = feature.id;\n if (this.promoteId) {\n const propName = typeof this.promoteId === 'string' ? this.promoteId : this.promoteId[sourceLayerId];\n id = feature.properties[propName] as string | number;\n if (typeof id === 'boolean') id = Number(id);\n }\n return id;\n }\n}\n\nregister(\n 'FeatureIndex',\n FeatureIndex,\n {omit: ['rawTileData', 'sourceLayerCoder']}\n);\n\nfunction evaluateProperties(serializedProperties, styleLayerProperties, feature, featureState, availableImages) {\n return mapObject(serializedProperties, (property, key) => {\n const prop = styleLayerProperties instanceof PossiblyEvaluated ? styleLayerProperties.get(key) : null;\n return prop && prop.evaluate ? prop.evaluate(feature, featureState, availableImages) : prop;\n });\n}\n\nfunction getBounds(geometry: Array) {\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n for (const p of geometry) {\n minX = Math.min(minX, p.x);\n minY = Math.min(minY, p.y);\n maxX = Math.max(maxX, p.x);\n maxY = Math.max(maxY, p.y);\n }\n return {minX, minY, maxX, maxY};\n}\n\nfunction topDownFeatureComparator(a, b) {\n return b - a;\n}\n","import Point from '@mapbox/point-geometry';\n\nimport {register} from '../util/web_worker_transfer';\n\nexport class Anchor extends Point {\n angle: any;\n segment?: number;\n\n constructor(x: number, y: number, angle: number, segment?: number) {\n super(x, y);\n this.angle = angle;\n if (segment !== undefined) {\n this.segment = segment;\n }\n }\n\n clone() {\n return new Anchor(this.x, this.y, this.angle, this.segment);\n }\n}\n\nregister('Anchor', Anchor);\n","import type Point from '@mapbox/point-geometry';\nimport type {Anchor} from './anchor';\n\n/**\n * Labels placed around really sharp angles aren't readable. Check if any\n * part of the potential label has a combined angle that is too big.\n *\n * @param line - The line to check\n * @param anchor - The point on the line around which the label is anchored.\n * @param labelLength - The length of the label in geometry units.\n * @param windowSize - The check fails if the combined angles within a part of the line that is `windowSize` long is too big.\n * @param maxAngle - The maximum combined angle that any window along the label is allowed to have.\n *\n * @returns whether the label should be placed\n */\nexport function checkMaxAngle(line: Array, anchor: Anchor, labelLength: number, windowSize: number, maxAngle: number) {\n\n // horizontal labels and labels with length 0 always pass\n if (anchor.segment === undefined || labelLength === 0) return true;\n\n let p = anchor;\n let index = anchor.segment + 1;\n let anchorDistance = 0;\n\n // move backwards along the line to the first segment the label appears on\n while (anchorDistance > -labelLength / 2) {\n index--;\n\n // there isn't enough room for the label after the beginning of the line\n if (index < 0) return false;\n\n anchorDistance -= line[index].dist(p);\n p = line[index];\n }\n\n anchorDistance += line[index].dist(line[index + 1]);\n index++;\n\n // store recent corners and their total angle difference\n const recentCorners = [];\n let recentAngleDelta = 0;\n\n // move forwards by the length of the label and check angles along the way\n while (anchorDistance < labelLength / 2) {\n const prev = line[index - 1];\n const current = line[index];\n const next = line[index + 1];\n\n // there isn't enough room for the label before the end of the line\n if (!next) return false;\n\n let angleDelta = prev.angleTo(current) - current.angleTo(next);\n // restrict angle to -pi..pi range\n angleDelta = Math.abs(((angleDelta + 3 * Math.PI) % (Math.PI * 2)) - Math.PI);\n\n recentCorners.push({\n distance: anchorDistance,\n angleDelta\n });\n recentAngleDelta += angleDelta;\n\n // remove corners that are far enough away from the list of recent anchors\n while (anchorDistance - recentCorners[0].distance > windowSize) {\n recentAngleDelta -= recentCorners.shift().angleDelta;\n }\n\n // the sum of angles within the window area exceeds the maximum allowed value. check fails.\n if (recentAngleDelta > maxAngle) return false;\n\n index++;\n anchorDistance += current.dist(next);\n }\n\n // no part of the line had an angle greater than the maximum allowed. check passes.\n return true;\n}\n","import {interpolates} from '@maplibre/maplibre-gl-style-spec';\n\nimport {Anchor} from '../symbol/anchor';\nimport {checkMaxAngle} from './check_max_angle';\n\nimport type Point from '@mapbox/point-geometry';\nimport type {Shaping, PositionedIcon} from './shaping';\n\nexport {getAnchors, getCenterAnchor};\n\nfunction getLineLength(line: Array): number {\n let lineLength = 0;\n for (let k = 0; k < line.length - 1; k++) {\n lineLength += line[k].dist(line[k + 1]);\n }\n return lineLength;\n}\n\nfunction getAngleWindowSize(\n shapedText: Shaping,\n glyphSize: number,\n boxScale: number\n): number {\n return shapedText ?\n 3 / 5 * glyphSize * boxScale :\n 0;\n}\n\nfunction getShapedLabelLength(shapedText?: Shaping | null, shapedIcon?: PositionedIcon | null): number {\n return Math.max(\n shapedText ? shapedText.right - shapedText.left : 0,\n shapedIcon ? shapedIcon.right - shapedIcon.left : 0);\n}\n\nfunction getCenterAnchor(line: Array,\n maxAngle: number,\n shapedText: Shaping,\n shapedIcon: PositionedIcon,\n glyphSize: number,\n boxScale: number) {\n const angleWindowSize = getAngleWindowSize(shapedText, glyphSize, boxScale);\n const labelLength = getShapedLabelLength(shapedText, shapedIcon) * boxScale;\n\n let prevDistance = 0;\n const centerDistance = getLineLength(line) / 2;\n\n for (let i = 0; i < line.length - 1; i++) {\n\n const a = line[i],\n b = line[i + 1];\n\n const segmentDistance = a.dist(b);\n\n if (prevDistance + segmentDistance > centerDistance) {\n // The center is on this segment\n const t = (centerDistance - prevDistance) / segmentDistance,\n x = interpolates.number(a.x, b.x, t),\n y = interpolates.number(a.y, b.y, t);\n\n const anchor = new Anchor(x, y, b.angleTo(a), i);\n anchor._round();\n if (!angleWindowSize || checkMaxAngle(line, anchor, labelLength, angleWindowSize, maxAngle)) {\n return anchor;\n } else {\n return;\n }\n }\n\n prevDistance += segmentDistance;\n }\n}\n\nfunction getAnchors(line: Array,\n spacing: number,\n maxAngle: number,\n shapedText: Shaping,\n shapedIcon: PositionedIcon,\n glyphSize: number,\n boxScale: number,\n overscaling: number,\n tileExtent: number) {\n\n // Resample a line to get anchor points for labels and check that each\n // potential label passes text-max-angle check and has enough froom to fit\n // on the line.\n\n const angleWindowSize = getAngleWindowSize(shapedText, glyphSize, boxScale);\n const shapedLabelLength = getShapedLabelLength(shapedText, shapedIcon);\n const labelLength = shapedLabelLength * boxScale;\n\n // Is the line continued from outside the tile boundary?\n const isLineContinued = line[0].x === 0 || line[0].x === tileExtent || line[0].y === 0 || line[0].y === tileExtent;\n\n // Is the label long, relative to the spacing?\n // If so, adjust the spacing so there is always a minimum space of `spacing / 4` between label edges.\n if (spacing - labelLength < spacing / 4) {\n spacing = labelLength + spacing / 4;\n }\n\n // Offset the first anchor by:\n // Either half the label length plus a fixed extra offset if the line is not continued\n // Or half the spacing if the line is continued.\n\n // For non-continued lines, add a bit of fixed extra offset to avoid collisions at T intersections.\n const fixedExtraOffset = glyphSize * 2;\n\n const offset = !isLineContinued ?\n ((shapedLabelLength / 2 + fixedExtraOffset) * boxScale * overscaling) % spacing :\n (spacing / 2 * overscaling) % spacing;\n\n return resample(line, offset, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, false, tileExtent);\n}\n\nfunction resample(line, offset, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, placeAtMiddle, tileExtent) {\n\n const halfLabelLength = labelLength / 2;\n const lineLength = getLineLength(line);\n\n let distance = 0,\n markedDistance = offset - spacing;\n\n let anchors = [];\n\n for (let i = 0; i < line.length - 1; i++) {\n\n const a = line[i],\n b = line[i + 1];\n\n const segmentDist = a.dist(b),\n angle = b.angleTo(a);\n\n while (markedDistance + spacing < distance + segmentDist) {\n markedDistance += spacing;\n\n const t = (markedDistance - distance) / segmentDist,\n x = interpolates.number(a.x, b.x, t),\n y = interpolates.number(a.y, b.y, t);\n\n // Check that the point is within the tile boundaries and that\n // the label would fit before the beginning and end of the line\n // if placed at this point.\n if (x >= 0 && x < tileExtent && y >= 0 && y < tileExtent &&\n markedDistance - halfLabelLength >= 0 &&\n markedDistance + halfLabelLength <= lineLength) {\n const anchor = new Anchor(x, y, angle, i);\n anchor._round();\n\n if (!angleWindowSize || checkMaxAngle(line, anchor, labelLength, angleWindowSize, maxAngle)) {\n anchors.push(anchor);\n }\n }\n }\n\n distance += segmentDist;\n }\n\n if (!placeAtMiddle && !anchors.length && !isLineContinued) {\n // The first attempt at finding anchors at which labels can be placed failed.\n // Try again, but this time just try placing one anchor at the middle of the line.\n // This has the most effect for short lines in overscaled tiles, since the\n // initial offset used in overscaled tiles is calculated to align labels with positions in\n // parent tiles instead of placing the label as close to the beginning as possible.\n anchors = resample(line, distance / 2, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, true, tileExtent);\n }\n\n return anchors;\n}\n","import Point from '@mapbox/point-geometry';\n\nimport {GLYPH_PBF_BORDER} from '../style/parse_glyph_pbf';\n\nimport type {Anchor} from './anchor';\nimport type {PositionedIcon, Shaping} from './shaping';\nimport {SHAPING_DEFAULT_OFFSET} from './shaping';\nimport {IMAGE_PADDING} from '../render/image_atlas';\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\nimport type {Feature} from '@maplibre/maplibre-gl-style-spec';\nimport type {StyleImage} from '../style/style_image';\nimport ONE_EM from './one_em';\nimport {Rect} from '../render/glyph_atlas';\n\n/**\n * A textured quad for rendering a single icon or glyph.\n *\n * The zoom range the glyph can be shown is defined by minScale and maxScale.\n *\n * @param tl - The offset of the top left corner from the anchor.\n * @param tr - The offset of the top right corner from the anchor.\n * @param bl - The offset of the bottom left corner from the anchor.\n * @param br - The offset of the bottom right corner from the anchor.\n * @param tex - The texture coordinates.\n */\nexport type SymbolQuad = {\n tl: Point;\n tr: Point;\n bl: Point;\n br: Point;\n tex: {\n x: number;\n y: number;\n w: number;\n h: number;\n };\n pixelOffsetTL: Point;\n pixelOffsetBR: Point;\n writingMode: any | void;\n glyphOffset: [number, number];\n sectionIndex: number;\n isSDF: boolean;\n minFontScaleX: number;\n minFontScaleY: number;\n};\n\n// If you have a 10px icon that isn't perfectly aligned to the pixel grid it will cover 11 actual\n// pixels. The quad needs to be padded to account for this, otherwise they'll look slightly clipped\n// on one edge in some cases.\nconst border = IMAGE_PADDING;\n\n/**\n * Create the quads used for rendering an icon.\n */\nexport function getIconQuads(\n shapedIcon: PositionedIcon,\n iconRotate: number,\n isSDFIcon: boolean,\n hasIconTextFit: boolean\n): Array {\n const quads = [];\n\n const image = shapedIcon.image;\n const pixelRatio = image.pixelRatio;\n const imageWidth = image.paddedRect.w - 2 * border;\n const imageHeight = image.paddedRect.h - 2 * border;\n\n const iconWidth = shapedIcon.right - shapedIcon.left;\n const iconHeight = shapedIcon.bottom - shapedIcon.top;\n\n const stretchX = image.stretchX || [[0, imageWidth]];\n const stretchY = image.stretchY || [[0, imageHeight]];\n\n const reduceRanges = (sum, range) => sum + range[1] - range[0];\n const stretchWidth = stretchX.reduce(reduceRanges, 0);\n const stretchHeight = stretchY.reduce(reduceRanges, 0);\n const fixedWidth = imageWidth - stretchWidth;\n const fixedHeight = imageHeight - stretchHeight;\n\n let stretchOffsetX = 0;\n let stretchContentWidth = stretchWidth;\n let stretchOffsetY = 0;\n let stretchContentHeight = stretchHeight;\n let fixedOffsetX = 0;\n let fixedContentWidth = fixedWidth;\n let fixedOffsetY = 0;\n let fixedContentHeight = fixedHeight;\n\n if (image.content && hasIconTextFit) {\n const content = image.content;\n stretchOffsetX = sumWithinRange(stretchX, 0, content[0]);\n stretchOffsetY = sumWithinRange(stretchY, 0, content[1]);\n stretchContentWidth = sumWithinRange(stretchX, content[0], content[2]);\n stretchContentHeight = sumWithinRange(stretchY, content[1], content[3]);\n fixedOffsetX = content[0] - stretchOffsetX;\n fixedOffsetY = content[1] - stretchOffsetY;\n fixedContentWidth = content[2] - content[0] - stretchContentWidth;\n fixedContentHeight = content[3] - content[1] - stretchContentHeight;\n }\n\n const makeBox = (left, top, right, bottom) => {\n\n const leftEm = getEmOffset(left.stretch - stretchOffsetX, stretchContentWidth, iconWidth, shapedIcon.left);\n const leftPx = getPxOffset(left.fixed - fixedOffsetX, fixedContentWidth, left.stretch, stretchWidth);\n\n const topEm = getEmOffset(top.stretch - stretchOffsetY, stretchContentHeight, iconHeight, shapedIcon.top);\n const topPx = getPxOffset(top.fixed - fixedOffsetY, fixedContentHeight, top.stretch, stretchHeight);\n\n const rightEm = getEmOffset(right.stretch - stretchOffsetX, stretchContentWidth, iconWidth, shapedIcon.left);\n const rightPx = getPxOffset(right.fixed - fixedOffsetX, fixedContentWidth, right.stretch, stretchWidth);\n\n const bottomEm = getEmOffset(bottom.stretch - stretchOffsetY, stretchContentHeight, iconHeight, shapedIcon.top);\n const bottomPx = getPxOffset(bottom.fixed - fixedOffsetY, fixedContentHeight, bottom.stretch, stretchHeight);\n\n const tl = new Point(leftEm, topEm);\n const tr = new Point(rightEm, topEm);\n const br = new Point(rightEm, bottomEm);\n const bl = new Point(leftEm, bottomEm);\n const pixelOffsetTL = new Point(leftPx / pixelRatio, topPx / pixelRatio);\n const pixelOffsetBR = new Point(rightPx / pixelRatio, bottomPx / pixelRatio);\n\n const angle = iconRotate * Math.PI / 180;\n\n if (angle) {\n const sin = Math.sin(angle),\n cos = Math.cos(angle),\n matrix = [cos, -sin, sin, cos];\n\n tl._matMult(matrix);\n tr._matMult(matrix);\n bl._matMult(matrix);\n br._matMult(matrix);\n }\n\n const x1 = left.stretch + left.fixed;\n const x2 = right.stretch + right.fixed;\n const y1 = top.stretch + top.fixed;\n const y2 = bottom.stretch + bottom.fixed;\n\n const subRect = {\n x: image.paddedRect.x + border + x1,\n y: image.paddedRect.y + border + y1,\n w: x2 - x1,\n h: y2 - y1\n };\n\n const minFontScaleX = fixedContentWidth / pixelRatio / iconWidth;\n const minFontScaleY = fixedContentHeight / pixelRatio / iconHeight;\n\n // Icon quad is padded, so texture coordinates also need to be padded.\n return {tl, tr, bl, br, tex: subRect, writingMode: undefined, glyphOffset: [0, 0], sectionIndex: 0, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY, isSDF: isSDFIcon};\n };\n\n if (!hasIconTextFit || (!image.stretchX && !image.stretchY)) {\n quads.push(makeBox(\n {fixed: 0, stretch: -1},\n {fixed: 0, stretch: -1},\n {fixed: 0, stretch: imageWidth + 1},\n {fixed: 0, stretch: imageHeight + 1}));\n } else {\n const xCuts = stretchZonesToCuts(stretchX, fixedWidth, stretchWidth);\n const yCuts = stretchZonesToCuts(stretchY, fixedHeight, stretchHeight);\n\n for (let xi = 0; xi < xCuts.length - 1; xi++) {\n const x1 = xCuts[xi];\n const x2 = xCuts[xi + 1];\n for (let yi = 0; yi < yCuts.length - 1; yi++) {\n const y1 = yCuts[yi];\n const y2 = yCuts[yi + 1];\n quads.push(makeBox(x1, y1, x2, y2));\n }\n }\n }\n\n return quads;\n}\n\nfunction sumWithinRange(ranges, min, max) {\n let sum = 0;\n for (const range of ranges) {\n sum += Math.max(min, Math.min(max, range[1])) - Math.max(min, Math.min(max, range[0]));\n }\n return sum;\n}\n\nfunction stretchZonesToCuts(stretchZones, fixedSize, stretchSize) {\n const cuts = [{fixed: -border, stretch: 0}];\n\n for (const [c1, c2] of stretchZones) {\n const last = cuts[cuts.length - 1];\n cuts.push({\n fixed: c1 - last.stretch,\n stretch: last.stretch\n });\n cuts.push({\n fixed: c1 - last.stretch,\n stretch: last.stretch + (c2 - c1)\n });\n }\n cuts.push({\n fixed: fixedSize + border,\n stretch: stretchSize\n });\n return cuts;\n}\n\nfunction getEmOffset(stretchOffset, stretchSize, iconSize, iconOffset) {\n return stretchOffset / stretchSize * iconSize + iconOffset;\n}\n\nfunction getPxOffset(fixedOffset, fixedSize, stretchOffset, stretchSize) {\n return fixedOffset - fixedSize * stretchOffset / stretchSize;\n}\n\n/**\n * Create the quads used for rendering a text label.\n */\nexport function getGlyphQuads(\n anchor: Anchor,\n shaping: Shaping,\n textOffset: [number, number],\n layer: SymbolStyleLayer,\n alongLine: boolean,\n feature: Feature,\n imageMap: {[_: string]: StyleImage},\n allowVerticalPlacement: boolean\n): Array {\n\n const textRotate = layer.layout.get('text-rotate').evaluate(feature, {}) * Math.PI / 180;\n const quads = [];\n\n for (const line of shaping.positionedLines) {\n for (const positionedGlyph of line.positionedGlyphs) {\n if (!positionedGlyph.rect) continue;\n const textureRect: Rect = positionedGlyph.rect || {} as Rect;\n\n // The rects have an additional buffer that is not included in their size.\n const glyphPadding = 1.0;\n let rectBuffer = GLYPH_PBF_BORDER + glyphPadding;\n let isSDF = true;\n let pixelRatio = 1.0;\n let lineOffset = 0.0;\n\n const rotateVerticalGlyph = (alongLine || allowVerticalPlacement) && positionedGlyph.vertical;\n const halfAdvance = positionedGlyph.metrics.advance * positionedGlyph.scale / 2;\n\n // Align images and scaled glyphs in the middle of a vertical line.\n if (allowVerticalPlacement && shaping.verticalizable) {\n const scaledGlyphOffset = (positionedGlyph.scale - 1) * ONE_EM;\n const imageOffset = (ONE_EM - positionedGlyph.metrics.width * positionedGlyph.scale) / 2;\n lineOffset = line.lineOffset / 2 - (positionedGlyph.imageName ? -imageOffset : scaledGlyphOffset);\n }\n\n if (positionedGlyph.imageName) {\n const image = imageMap[positionedGlyph.imageName];\n isSDF = image.sdf;\n pixelRatio = image.pixelRatio;\n rectBuffer = IMAGE_PADDING / pixelRatio;\n }\n\n const glyphOffset = alongLine ?\n [positionedGlyph.x + halfAdvance, positionedGlyph.y] :\n [0, 0];\n\n let builtInOffset: [number, number] = alongLine ?\n [0, 0] :\n [positionedGlyph.x + halfAdvance + textOffset[0], positionedGlyph.y + textOffset[1] - lineOffset];\n\n let verticalizedLabelOffset = [0, 0] as [number, number];\n if (rotateVerticalGlyph) {\n // Vertical POI labels that are rotated 90deg CW and whose glyphs must preserve upright orientation\n // need to be rotated 90deg CCW. After a quad is rotated, it is translated to the original built-in offset.\n verticalizedLabelOffset = builtInOffset;\n builtInOffset = [0, 0];\n }\n\n const textureScale = positionedGlyph.metrics.isDoubleResolution ? 2 : 1;\n\n const x1 = (positionedGlyph.metrics.left - rectBuffer) * positionedGlyph.scale - halfAdvance + builtInOffset[0];\n const y1 = (-positionedGlyph.metrics.top - rectBuffer) * positionedGlyph.scale + builtInOffset[1];\n const x2 = x1 + textureRect.w / textureScale * positionedGlyph.scale / pixelRatio;\n const y2 = y1 + textureRect.h / textureScale * positionedGlyph.scale / pixelRatio;\n\n const tl = new Point(x1, y1);\n const tr = new Point(x2, y1);\n const bl = new Point(x1, y2);\n const br = new Point(x2, y2);\n\n if (rotateVerticalGlyph) {\n // Vertical-supporting glyphs are laid out in 24x24 point boxes (1 square em)\n // In horizontal orientation, the y values for glyphs are below the midline\n // and we use a \"yOffset\" of -17 to pull them up to the middle.\n // By rotating counter-clockwise around the point at the center of the left\n // edge of a 24x24 layout box centered below the midline, we align the center\n // of the glyphs with the horizontal midline, so the yOffset is no longer\n // necessary, but we also pull the glyph to the left along the x axis.\n // The y coordinate includes baseline yOffset, thus needs to be accounted\n // for when glyph is rotated and translated.\n const center = new Point(-halfAdvance, halfAdvance - SHAPING_DEFAULT_OFFSET);\n const verticalRotation = -Math.PI / 2;\n\n // xHalfWidthOffsetCorrection is a difference between full-width and half-width\n // advance, should be 0 for full-width glyphs and will pull up half-width glyphs.\n const xHalfWidthOffsetCorrection = ONE_EM / 2 - halfAdvance;\n const yImageOffsetCorrection = positionedGlyph.imageName ? xHalfWidthOffsetCorrection : 0.0;\n const halfWidthOffsetCorrection = new Point(5 - SHAPING_DEFAULT_OFFSET - xHalfWidthOffsetCorrection, -yImageOffsetCorrection);\n const verticalOffsetCorrection = new Point(...verticalizedLabelOffset);\n tl._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection);\n tr._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection);\n bl._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection);\n br._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection);\n }\n\n if (textRotate) {\n const sin = Math.sin(textRotate),\n cos = Math.cos(textRotate),\n matrix = [cos, -sin, sin, cos];\n\n tl._matMult(matrix);\n tr._matMult(matrix);\n bl._matMult(matrix);\n br._matMult(matrix);\n }\n\n const pixelOffsetTL = new Point(0, 0);\n const pixelOffsetBR = new Point(0, 0);\n const minFontScaleX = 0;\n const minFontScaleY = 0;\n quads.push({tl, tr, bl, br, tex: textureRect, writingMode: shaping.writingMode, glyphOffset, sectionIndex: positionedGlyph.sectionIndex, isSDF, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY});\n }\n }\n\n return quads;\n}\n","import type {CollisionBoxArray} from '../data/array_types.g';\nimport Point from '@mapbox/point-geometry';\nimport type {Anchor} from './anchor';\nimport {SymbolPadding} from '../style/style_layer/symbol_style_layer';\n\n/**\n * A CollisionFeature represents the area of the tile covered by a single label.\n * It is used with CollisionIndex to check if the label overlaps with any\n * previous labels. A CollisionFeature is mostly just a set of CollisionBox\n * objects.\n */\nexport class CollisionFeature {\n boxStartIndex: number;\n boxEndIndex: number;\n circleDiameter: number;\n\n /**\n * Create a CollisionFeature, adding its collision box data to the given collisionBoxArray in the process.\n * For line aligned labels a collision circle diameter is computed instead.\n *\n * @param anchor - The point along the line around which the label is anchored.\n * @param shaped - The text or icon shaping results.\n * @param boxScale - A magic number used to convert from glyph metrics units to geometry units.\n * @param padding - The amount of padding to add around the label edges.\n * @param alignLine - Whether the label is aligned with the line or the viewport.\n */\n constructor(collisionBoxArray: CollisionBoxArray,\n anchor: Anchor,\n featureIndex: number,\n sourceLayerIndex: number,\n bucketIndex: number,\n shaped: any,\n boxScale: number,\n padding: SymbolPadding,\n alignLine: boolean,\n rotate: number) {\n\n this.boxStartIndex = collisionBoxArray.length;\n\n if (alignLine) {\n // Compute height of the shape in glyph metrics and apply collision padding.\n // Note that the pixel based 'text-padding' is applied at runtime\n let top = shaped.top;\n let bottom = shaped.bottom;\n const collisionPadding = shaped.collisionPadding;\n\n if (collisionPadding) {\n top -= collisionPadding[1];\n bottom += collisionPadding[3];\n }\n\n let height = bottom - top;\n\n if (height > 0) {\n // set minimum box height to avoid very many small labels\n height = Math.max(10, height);\n this.circleDiameter = height;\n }\n } else {\n // margin is in CSS order: [top, right, bottom, left]\n let y1 = shaped.top * boxScale - padding[0];\n let y2 = shaped.bottom * boxScale + padding[2];\n let x1 = shaped.left * boxScale - padding[3];\n let x2 = shaped.right * boxScale + padding[1];\n\n const collisionPadding = shaped.collisionPadding;\n if (collisionPadding) {\n x1 -= collisionPadding[0] * boxScale;\n y1 -= collisionPadding[1] * boxScale;\n x2 += collisionPadding[2] * boxScale;\n y2 += collisionPadding[3] * boxScale;\n }\n\n if (rotate) {\n // Account for *-rotate in point collision boxes\n // See https://github.com/mapbox/mapbox-gl-js/issues/6075\n // Doesn't account for icon-text-fit\n\n const tl = new Point(x1, y1);\n const tr = new Point(x2, y1);\n const bl = new Point(x1, y2);\n const br = new Point(x2, y2);\n\n const rotateRadians = rotate * Math.PI / 180;\n\n tl._rotate(rotateRadians);\n tr._rotate(rotateRadians);\n bl._rotate(rotateRadians);\n br._rotate(rotateRadians);\n\n // Collision features require an \"on-axis\" geometry,\n // so take the envelope of the rotated geometry\n // (may be quite large for wide labels rotated 45 degrees)\n x1 = Math.min(tl.x, tr.x, bl.x, br.x);\n x2 = Math.max(tl.x, tr.x, bl.x, br.x);\n y1 = Math.min(tl.y, tr.y, bl.y, br.y);\n y2 = Math.max(tl.y, tr.y, bl.y, br.y);\n }\n collisionBoxArray.emplaceBack(anchor.x, anchor.y, x1, y1, x2, y2, featureIndex, sourceLayerIndex, bucketIndex);\n }\n\n this.boxEndIndex = collisionBoxArray.length;\n }\n}\n","\nexport default class TinyQueue {\n constructor(data = [], compare = defaultCompare) {\n this.data = data;\n this.length = this.data.length;\n this.compare = compare;\n\n if (this.length > 0) {\n for (let i = (this.length >> 1) - 1; i >= 0; i--) this._down(i);\n }\n }\n\n push(item) {\n this.data.push(item);\n this.length++;\n this._up(this.length - 1);\n }\n\n pop() {\n if (this.length === 0) return undefined;\n\n const top = this.data[0];\n const bottom = this.data.pop();\n this.length--;\n\n if (this.length > 0) {\n this.data[0] = bottom;\n this._down(0);\n }\n\n return top;\n }\n\n peek() {\n return this.data[0];\n }\n\n _up(pos) {\n const {data, compare} = this;\n const item = data[pos];\n\n while (pos > 0) {\n const parent = (pos - 1) >> 1;\n const current = data[parent];\n if (compare(item, current) >= 0) break;\n data[pos] = current;\n pos = parent;\n }\n\n data[pos] = item;\n }\n\n _down(pos) {\n const {data, compare} = this;\n const halfLength = this.length >> 1;\n const item = data[pos];\n\n while (pos < halfLength) {\n let left = (pos << 1) + 1;\n let best = data[left];\n const right = left + 1;\n\n if (right < this.length && compare(data[right], best) < 0) {\n left = right;\n best = data[right];\n }\n if (compare(best, item) >= 0) break;\n\n data[pos] = best;\n pos = left;\n }\n\n data[pos] = item;\n }\n}\n\nfunction defaultCompare(a, b) {\n return a < b ? -1 : a > b ? 1 : 0;\n}\n","import Queue from 'tinyqueue';\n\nimport Point from '@mapbox/point-geometry';\nimport {distToSegmentSquared} from './intersection_tests';\n\n/**\n * Finds an approximation of a polygon's Pole Of Inaccessibiliy https://en.wikipedia.org/wiki/Pole_of_inaccessibility\n * This is a copy of http://github.com/mapbox/polylabel adapted to use Points\n *\n * @param polygonRings - first item in array is the outer ring followed optionally by the list of holes, should be an element of the result of util/classify_rings\n * @param precision - Specified in input coordinate units. If 0 returns after first run, if `> 0` repeatedly narrows the search space until the radius of the area searched for the best pole is less than precision\n * @param debug - Print some statistics to the console during execution\n * @returns Pole of Inaccessibiliy.\n */\nexport function findPoleOfInaccessibility(\n polygonRings: Array>,\n precision: number = 1,\n debug: boolean = false\n): Point {\n // find the bounding box of the outer ring\n let minX = Infinity, minY = Infinity, maxX = -Infinity, maxY = -Infinity;\n const outerRing = polygonRings[0];\n for (let i = 0; i < outerRing.length; i++) {\n const p = outerRing[i];\n if (!i || p.x < minX) minX = p.x;\n if (!i || p.y < minY) minY = p.y;\n if (!i || p.x > maxX) maxX = p.x;\n if (!i || p.y > maxY) maxY = p.y;\n }\n\n const width = maxX - minX;\n const height = maxY - minY;\n const cellSize = Math.min(width, height);\n let h = cellSize / 2;\n\n // a priority queue of cells in order of their \"potential\" (max distance to polygon)\n const cellQueue = new Queue([], compareMax);\n\n if (cellSize === 0) return new Point(minX, minY);\n\n // cover polygon with initial cells\n for (let x = minX; x < maxX; x += cellSize) {\n for (let y = minY; y < maxY; y += cellSize) {\n cellQueue.push(new Cell(x + h, y + h, h, polygonRings));\n }\n }\n\n // take centroid as the first best guess\n let bestCell = getCentroidCell(polygonRings);\n let numProbes = cellQueue.length;\n\n while (cellQueue.length) {\n // pick the most promising cell from the queue\n const cell = cellQueue.pop();\n\n // update the best cell if we found a better one\n if (cell.d > bestCell.d || !bestCell.d) {\n bestCell = cell;\n if (debug) console.log('found best %d after %d probes', Math.round(1e4 * cell.d) / 1e4, numProbes);\n }\n\n // do not drill down further if there's no chance of a better solution\n if (cell.max - bestCell.d <= precision) continue;\n\n // split the cell into four cells\n h = cell.h / 2;\n cellQueue.push(new Cell(cell.p.x - h, cell.p.y - h, h, polygonRings));\n cellQueue.push(new Cell(cell.p.x + h, cell.p.y - h, h, polygonRings));\n cellQueue.push(new Cell(cell.p.x - h, cell.p.y + h, h, polygonRings));\n cellQueue.push(new Cell(cell.p.x + h, cell.p.y + h, h, polygonRings));\n numProbes += 4;\n }\n\n if (debug) {\n console.log(`num probes: ${numProbes}`);\n console.log(`best distance: ${bestCell.d}`);\n }\n\n return bestCell.p;\n}\n\nfunction compareMax(a, b) {\n return b.max - a.max;\n}\n\nfunction Cell(x, y, h, polygon) {\n this.p = new Point(x, y);\n this.h = h; // half the cell size\n this.d = pointToPolygonDist(this.p, polygon); // distance from cell center to polygon\n this.max = this.d + this.h * Math.SQRT2; // max distance to polygon within a cell\n}\n\n// signed distance from point to polygon outline (negative if point is outside)\nfunction pointToPolygonDist(p, polygon) {\n let inside = false;\n let minDistSq = Infinity;\n\n for (let k = 0; k < polygon.length; k++) {\n const ring = polygon[k];\n\n for (let i = 0, len = ring.length, j = len - 1; i < len; j = i++) {\n const a = ring[i];\n const b = ring[j];\n\n if ((a.y > p.y !== b.y > p.y) &&\n (p.x < (b.x - a.x) * (p.y - a.y) / (b.y - a.y) + a.x)) inside = !inside;\n\n minDistSq = Math.min(minDistSq, distToSegmentSquared(p, a, b));\n }\n }\n\n return (inside ? 1 : -1) * Math.sqrt(minDistSq);\n}\n\n// get polygon centroid\nfunction getCentroidCell(polygon) {\n let area = 0;\n let x = 0;\n let y = 0;\n const points = polygon[0];\n for (let i = 0, len = points.length, j = len - 1; i < len; j = i++) {\n const a = points[i];\n const b = points[j];\n const f = a.x * b.y - b.x * a.y;\n x += (a.x + b.x) * f;\n y += (a.y + b.y) * f;\n area += f * 3;\n }\n return new Cell(x / area, y / area, 0, polygon);\n}\n","import {VariableAnchorOffsetCollection, VariableAnchorOffsetCollectionSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport {SymbolFeature} from '../../data/bucket/symbol_bucket';\nimport {CanonicalTileID} from '../../source/tile_id';\nimport ONE_EM from '../../symbol/one_em';\nimport {SymbolStyleLayer} from './symbol_style_layer';\n\nexport enum TextAnchorEnum {\n 'center' = 1,\n 'left' = 2,\n 'right' = 3,\n 'top' = 4,\n 'bottom' = 5,\n 'top-left' = 6,\n 'top-right' = 7,\n 'bottom-left' = 8,\n 'bottom-right' = 9\n}\n\nexport type TextAnchor = keyof typeof TextAnchorEnum;\n\n// The radial offset is to the edge of the text box\n// In the horizontal direction, the edge of the text box is where glyphs start\n// But in the vertical direction, the glyphs appear to \"start\" at the baseline\n// We don't actually load baseline data, but we assume an offset of ONE_EM - 17\n// (see \"yOffset\" in shaping.js)\nconst baselineOffset = 7;\nexport const INVALID_TEXT_OFFSET = Number.POSITIVE_INFINITY;\n\nexport function evaluateVariableOffset(anchor: TextAnchor, offset: [number, number]): [number, number] {\n\n function fromRadialOffset(anchor: TextAnchor, radialOffset: number): [number, number] {\n let x = 0, y = 0;\n if (radialOffset < 0) radialOffset = 0; // Ignore negative offset.\n // solve for r where r^2 + r^2 = radialOffset^2\n const hypotenuse = radialOffset / Math.SQRT2;\n switch (anchor) {\n case 'top-right':\n case 'top-left':\n y = hypotenuse - baselineOffset;\n break;\n case 'bottom-right':\n case 'bottom-left':\n y = -hypotenuse + baselineOffset;\n break;\n case 'bottom':\n y = -radialOffset + baselineOffset;\n break;\n case 'top':\n y = radialOffset - baselineOffset;\n break;\n }\n\n switch (anchor) {\n case 'top-right':\n case 'bottom-right':\n x = -hypotenuse;\n break;\n case 'top-left':\n case 'bottom-left':\n x = hypotenuse;\n break;\n case 'left':\n x = radialOffset;\n break;\n case 'right':\n x = -radialOffset;\n break;\n }\n\n return [x, y];\n }\n\n function fromTextOffset(anchor: TextAnchor, offsetX: number, offsetY: number): [number, number] {\n let x = 0, y = 0;\n // Use absolute offset values.\n offsetX = Math.abs(offsetX);\n offsetY = Math.abs(offsetY);\n\n switch (anchor) {\n case 'top-right':\n case 'top-left':\n case 'top':\n y = offsetY - baselineOffset;\n break;\n case 'bottom-right':\n case 'bottom-left':\n case 'bottom':\n y = -offsetY + baselineOffset;\n break;\n }\n\n switch (anchor) {\n case 'top-right':\n case 'bottom-right':\n case 'right':\n x = -offsetX;\n break;\n case 'top-left':\n case 'bottom-left':\n case 'left':\n x = offsetX;\n break;\n }\n\n return [x, y];\n }\n\n return (offset[1] !== INVALID_TEXT_OFFSET) ? fromTextOffset(anchor, offset[0], offset[1]) : fromRadialOffset(anchor, offset[0]);\n}\n\n// Helper to support both text-variable-anchor and text-variable-anchor-offset. Offset values converted from EMs to PXs\nexport function getTextVariableAnchorOffset(layer: SymbolStyleLayer, feature: SymbolFeature, canonical: CanonicalTileID): VariableAnchorOffsetCollection | null {\n const layout = layer.layout;\n // If style specifies text-variable-anchor-offset, just return it\n const variableAnchorOffset = layout.get('text-variable-anchor-offset')?.evaluate(feature, {}, canonical);\n\n if (variableAnchorOffset) {\n const sourceValues = variableAnchorOffset.values;\n const destValues: VariableAnchorOffsetCollectionSpecification = [];\n\n // Convert offsets from EM to PX, and apply baseline shift\n for (let i = 0; i < sourceValues.length; i += 2) {\n const anchor = destValues[i] = sourceValues[i] as TextAnchor;\n const offset = (sourceValues[i + 1] as [number, number]).map(t => t * ONE_EM) as [number, number];\n\n if (anchor.startsWith('top')) {\n offset[1] -= baselineOffset;\n } else if (anchor.startsWith('bottom')) {\n offset[1] += baselineOffset;\n }\n\n destValues[i + 1] = offset;\n }\n\n return new VariableAnchorOffsetCollection(destValues);\n }\n\n // If style specifies text-variable-anchor, convert to the new format\n const variableAnchor = layout.get('text-variable-anchor');\n\n if (variableAnchor) {\n let textOffset: [number, number];\n const unevaluatedLayout = layer._unevaluatedLayout;\n\n // The style spec says don't use `text-offset` and `text-radial-offset` together\n // but doesn't actually specify what happens if you use both. We go with the radial offset.\n if (unevaluatedLayout.getValue('text-radial-offset') !== undefined) {\n textOffset = [layout.get('text-radial-offset').evaluate(feature, {}, canonical) * ONE_EM, INVALID_TEXT_OFFSET];\n } else {\n textOffset = layout.get('text-offset').evaluate(feature, {}, canonical).map(t => t * ONE_EM) as [number, number];\n }\n\n const anchorOffsets: VariableAnchorOffsetCollectionSpecification = [];\n\n for (const anchor of variableAnchor) {\n anchorOffsets.push(anchor, evaluateVariableOffset(anchor, textOffset));\n }\n\n return new VariableAnchorOffsetCollection(anchorOffsets);\n }\n\n return null;\n}\n","import {Anchor} from './anchor';\n\nimport {getAnchors, getCenterAnchor} from './get_anchors';\nimport {clipLine} from './clip_line';\nimport {shapeText, shapeIcon, WritingMode, fitIconToText} from './shaping';\nimport {getGlyphQuads, getIconQuads} from './quads';\nimport {CollisionFeature} from './collision_feature';\nimport {warnOnce} from '../util/util';\nimport {\n allowsVerticalWritingMode,\n allowsLetterSpacing\n} from '../util/script_detection';\nimport {findPoleOfInaccessibility} from '../util/find_pole_of_inaccessibility';\nimport {classifyRings} from '../util/classify_rings';\nimport {EXTENT} from '../data/extent';\nimport {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {SIZE_PACK_FACTOR, MAX_PACKED_SIZE, MAX_GLYPH_ICON_SIZE} from './symbol_size';\nimport ONE_EM from './one_em';\nimport type {CanonicalTileID} from '../source/tile_id';\nimport type {Shaping, PositionedIcon, TextJustify} from './shaping';\nimport type {CollisionBoxArray, TextAnchorOffsetArray} from '../data/array_types.g';\nimport type {SymbolFeature} from '../data/bucket/symbol_bucket';\nimport type {StyleImage} from '../style/style_image';\nimport type {StyleGlyph} from '../style/style_glyph';\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\nimport type {ImagePosition} from '../render/image_atlas';\nimport type {GlyphPosition} from '../render/glyph_atlas';\nimport type {PossiblyEvaluatedPropertyValue} from '../style/properties';\n\nimport Point from '@mapbox/point-geometry';\nimport murmur3 from 'murmurhash-js';\nimport {getIconPadding, SymbolPadding} from '../style/style_layer/symbol_style_layer';\nimport {VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {getTextVariableAnchorOffset, evaluateVariableOffset, INVALID_TEXT_OFFSET, TextAnchor, TextAnchorEnum} from '../style/style_layer/variable_text_anchor';\n\n// The symbol layout process needs `text-size` evaluated at up to five different zoom levels, and\n// `icon-size` at up to three:\n//\n// 1. `text-size` at the zoom level of the bucket. Used to calculate a per-feature size for source `text-size`\n// expressions, and to calculate the box dimensions for icon-text-fit.\n// 2. `icon-size` at the zoom level of the bucket. Used to calculate a per-feature size for source `icon-size`\n// expressions.\n// 3. `text-size` and `icon-size` at the zoom level of the bucket, plus one. Used to calculate collision boxes.\n// 4. `text-size` at zoom level 18. Used for something line-symbol-placement-related.\n// 5. For composite `*-size` expressions: two zoom levels of curve stops that \"cover\" the zoom level of the\n// bucket. These go into a vertex buffer and are used by the shader to interpolate the size at render time.\n//\n// (1) and (2) are stored in `bucket.layers[0].layout`. The remainder are below.\n//\ntype Sizes = {\n layoutTextSize: PossiblyEvaluatedPropertyValue; // (3),\n layoutIconSize: PossiblyEvaluatedPropertyValue; // (3),\n textMaxSize: PossiblyEvaluatedPropertyValue; // (4),\n compositeTextSizes: [PossiblyEvaluatedPropertyValue, PossiblyEvaluatedPropertyValue]; // (5),\n compositeIconSizes: [PossiblyEvaluatedPropertyValue, PossiblyEvaluatedPropertyValue]; // (5)\n};\n\ntype ShapedTextOrientations = {\n vertical: Shaping | false;\n horizontal: Record;\n};\n\nexport function performSymbolLayout(args: {\n bucket: SymbolBucket;\n glyphMap: {\n [_: string]: {\n [x: number]: StyleGlyph;\n };\n };\n glyphPositions: {\n [_: string]: {\n [x: number]: GlyphPosition;\n };\n };\n imageMap: {[_: string]: StyleImage};\n imagePositions: {[_: string]: ImagePosition};\n showCollisionBoxes: boolean;\n canonical: CanonicalTileID;\n}) {\n args.bucket.createArrays();\n\n const tileSize = 512 * args.bucket.overscaling;\n args.bucket.tilePixelRatio = EXTENT / tileSize;\n args.bucket.compareText = {};\n args.bucket.iconsNeedLinear = false;\n\n const layer = args.bucket.layers[0];\n const layout = layer.layout;\n const unevaluatedLayoutValues = layer._unevaluatedLayout._values;\n\n const sizes: Sizes = {\n // Filled in below, if *SizeData.kind is 'composite'\n // compositeIconSizes: undefined,\n // compositeTextSizes: undefined,\n layoutIconSize: unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(args.bucket.zoom + 1), args.canonical),\n layoutTextSize: unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(args.bucket.zoom + 1), args.canonical),\n textMaxSize: unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(18))\n } as Sizes;\n\n if (args.bucket.textSizeData.kind === 'composite') {\n const {minZoom, maxZoom} = args.bucket.textSizeData;\n sizes.compositeTextSizes = [\n unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(minZoom), args.canonical),\n unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), args.canonical)\n ];\n }\n\n if (args.bucket.iconSizeData.kind === 'composite') {\n const {minZoom, maxZoom} = args.bucket.iconSizeData;\n sizes.compositeIconSizes = [\n unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(minZoom), args.canonical),\n unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), args.canonical)\n ];\n }\n\n const lineHeight = layout.get('text-line-height') * ONE_EM;\n const textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point';\n const keepUpright = layout.get('text-keep-upright');\n const textSize = layout.get('text-size');\n\n for (const feature of args.bucket.features) {\n const fontstack = layout.get('text-font').evaluate(feature, {}, args.canonical).join(',');\n const layoutTextSizeThisZoom = textSize.evaluate(feature, {}, args.canonical);\n const layoutTextSize = sizes.layoutTextSize.evaluate(feature, {}, args.canonical);\n const layoutIconSize = sizes.layoutIconSize.evaluate(feature, {}, args.canonical);\n\n const shapedTextOrientations: ShapedTextOrientations = {\n horizontal: {} as Record,\n vertical: undefined\n };\n const text = feature.text;\n let textOffset: [number, number] = [0, 0];\n if (text) {\n const unformattedText = text.toString();\n const spacing = layout.get('text-letter-spacing').evaluate(feature, {}, args.canonical) * ONE_EM;\n const spacingIfAllowed = allowsLetterSpacing(unformattedText) ? spacing : 0;\n\n const textAnchor = layout.get('text-anchor').evaluate(feature, {}, args.canonical);\n const variableAnchorOffset = getTextVariableAnchorOffset(layer, feature, args.canonical);\n\n if (!variableAnchorOffset) {\n const radialOffset = layout.get('text-radial-offset').evaluate(feature, {}, args.canonical);\n // Layers with variable anchors use the `text-radial-offset` property and the [x, y] offset vector\n // is calculated at placement time instead of layout time\n if (radialOffset) {\n // The style spec says don't use `text-offset` and `text-radial-offset` together\n // but doesn't actually specify what happens if you use both. We go with the radial offset.\n textOffset = evaluateVariableOffset(textAnchor, [radialOffset * ONE_EM, INVALID_TEXT_OFFSET]);\n } else {\n textOffset = (layout.get('text-offset').evaluate(feature, {}, args.canonical).map(t => t * ONE_EM) as [number, number]);\n }\n }\n\n let textJustify = textAlongLine ?\n 'center' :\n layout.get('text-justify').evaluate(feature, {}, args.canonical);\n\n const symbolPlacement = layout.get('symbol-placement');\n const maxWidth = symbolPlacement === 'point' ?\n layout.get('text-max-width').evaluate(feature, {}, args.canonical) * ONE_EM :\n 0;\n\n const addVerticalShapingForPointLabelIfNeeded = () => {\n if (args.bucket.allowVerticalPlacement && allowsVerticalWritingMode(unformattedText)) {\n // Vertical POI label placement is meant to be used for scripts that support vertical\n // writing mode, thus, default left justification is used. If Latin\n // scripts would need to be supported, this should take into account other justifications.\n shapedTextOrientations.vertical = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor,\n 'left', spacingIfAllowed, textOffset, WritingMode.vertical, true, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n }\n };\n\n // If this layer uses text-variable-anchor, generate shapings for all justification possibilities.\n if (!textAlongLine && variableAnchorOffset) {\n const justifications = new Set();\n\n if (textJustify === 'auto') {\n for (let i = 0; i < variableAnchorOffset.values.length; i += 2) {\n justifications.add(getAnchorJustification(variableAnchorOffset.values[i] as TextAnchor));\n }\n } else {\n justifications.add(textJustify);\n }\n\n let singleLine = false;\n for (const justification of justifications) {\n if (shapedTextOrientations.horizontal[justification]) continue;\n if (singleLine) {\n // If the shaping for the first justification was only a single line, we\n // can re-use it for the other justifications\n shapedTextOrientations.horizontal[justification] = shapedTextOrientations.horizontal[0];\n } else {\n // If using text-variable-anchor for the layer, we use a center anchor for all shapings and apply\n // the offsets for the anchor in the placement step.\n const shaping = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, 'center',\n justification, spacingIfAllowed, textOffset, WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n if (shaping) {\n shapedTextOrientations.horizontal[justification] = shaping;\n singleLine = shaping.positionedLines.length === 1;\n }\n }\n }\n\n addVerticalShapingForPointLabelIfNeeded();\n } else {\n if (textJustify === 'auto') {\n textJustify = getAnchorJustification(textAnchor);\n }\n\n // Horizontal point or line label.\n const shaping = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify, spacingIfAllowed,\n textOffset, WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n if (shaping) shapedTextOrientations.horizontal[textJustify] = shaping;\n\n // Vertical point label (if allowVerticalPlacement is enabled).\n addVerticalShapingForPointLabelIfNeeded();\n\n // Verticalized line label.\n if (allowsVerticalWritingMode(unformattedText) && textAlongLine && keepUpright) {\n shapedTextOrientations.vertical = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify,\n spacingIfAllowed, textOffset, WritingMode.vertical, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n }\n }\n }\n\n let shapedIcon;\n let isSDFIcon = false;\n if (feature.icon && feature.icon.name) {\n const image = args.imageMap[feature.icon.name];\n if (image) {\n shapedIcon = shapeIcon(\n args.imagePositions[feature.icon.name],\n layout.get('icon-offset').evaluate(feature, {}, args.canonical),\n layout.get('icon-anchor').evaluate(feature, {}, args.canonical));\n // null/undefined SDF property treated same as default (false)\n isSDFIcon = !!image.sdf;\n if (args.bucket.sdfIcons === undefined) {\n args.bucket.sdfIcons = isSDFIcon;\n } else if (args.bucket.sdfIcons !== isSDFIcon) {\n warnOnce('Style sheet warning: Cannot mix SDF and non-SDF icons in one buffer');\n }\n if (image.pixelRatio !== args.bucket.pixelRatio) {\n args.bucket.iconsNeedLinear = true;\n } else if (layout.get('icon-rotate').constantOr(1) !== 0) {\n args.bucket.iconsNeedLinear = true;\n }\n }\n }\n\n const shapedText = getDefaultHorizontalShaping(shapedTextOrientations.horizontal) || shapedTextOrientations.vertical;\n args.bucket.iconsInText = shapedText ? shapedText.iconsInText : false;\n if (shapedText || shapedIcon) {\n addFeature(args.bucket, feature, shapedTextOrientations, shapedIcon, args.imageMap, sizes, layoutTextSize, layoutIconSize, textOffset, isSDFIcon, args.canonical);\n }\n }\n\n if (args.showCollisionBoxes) {\n args.bucket.generateCollisionDebugBuffers();\n }\n}\n\n// Choose the justification that matches the direction of the TextAnchor\nexport function getAnchorJustification(anchor: TextAnchor): TextJustify {\n switch (anchor) {\n case 'right':\n case 'top-right':\n case 'bottom-right':\n return 'right';\n case 'left':\n case 'top-left':\n case 'bottom-left':\n return 'left';\n }\n return 'center';\n}\n\n/**\n * Given a feature and its shaped text and icon data, add a 'symbol\n * instance' for each _possible_ placement of the symbol feature.\n * (At render timePlaceSymbols#place() selects which of these instances to\n * show or hide based on collisions with symbols in other layers.)\n */\nfunction addFeature(bucket: SymbolBucket,\n feature: SymbolFeature,\n shapedTextOrientations: ShapedTextOrientations,\n shapedIcon: PositionedIcon,\n imageMap: {[_: string]: StyleImage},\n sizes: Sizes,\n layoutTextSize: number,\n layoutIconSize: number,\n textOffset: [number, number],\n isSDFIcon: boolean, canonical: CanonicalTileID) {\n // To reduce the number of labels that jump around when zooming we need\n // to use a text-size value that is the same for all zoom levels.\n // bucket calculates text-size at a high zoom level so that all tiles can\n // use the same value when calculating anchor positions.\n let textMaxSize = sizes.textMaxSize.evaluate(feature, {});\n if (textMaxSize === undefined) {\n textMaxSize = layoutTextSize;\n }\n const layout = bucket.layers[0].layout;\n const iconOffset = layout.get('icon-offset').evaluate(feature, {}, canonical);\n const defaultHorizontalShaping = getDefaultHorizontalShaping(shapedTextOrientations.horizontal);\n const glyphSize = 24,\n fontScale = layoutTextSize / glyphSize,\n textBoxScale = bucket.tilePixelRatio * fontScale,\n textMaxBoxScale = bucket.tilePixelRatio * textMaxSize / glyphSize,\n iconBoxScale = bucket.tilePixelRatio * layoutIconSize,\n symbolMinDistance = bucket.tilePixelRatio * layout.get('symbol-spacing'),\n textPadding = layout.get('text-padding') * bucket.tilePixelRatio,\n iconPadding = getIconPadding(layout, feature, canonical, bucket.tilePixelRatio),\n textMaxAngle = layout.get('text-max-angle') / 180 * Math.PI,\n textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point',\n iconAlongLine = layout.get('icon-rotation-alignment') === 'map' && layout.get('symbol-placement') !== 'point',\n symbolPlacement = layout.get('symbol-placement'),\n textRepeatDistance = symbolMinDistance / 2;\n\n const iconTextFit = layout.get('icon-text-fit');\n let verticallyShapedIcon;\n // Adjust shaped icon size when icon-text-fit is used.\n if (shapedIcon && iconTextFit !== 'none') {\n if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) {\n verticallyShapedIcon = fitIconToText(shapedIcon, shapedTextOrientations.vertical, iconTextFit,\n layout.get('icon-text-fit-padding'), iconOffset, fontScale);\n }\n if (defaultHorizontalShaping) {\n shapedIcon = fitIconToText(shapedIcon, defaultHorizontalShaping, iconTextFit,\n layout.get('icon-text-fit-padding'), iconOffset, fontScale);\n }\n }\n\n const addSymbolAtAnchor = (line, anchor) => {\n if (anchor.x < 0 || anchor.x >= EXTENT || anchor.y < 0 || anchor.y >= EXTENT) {\n // Symbol layers are drawn across tile boundaries, We filter out symbols\n // outside our tile boundaries (which may be included in vector tile buffers)\n // to prevent double-drawing symbols.\n return;\n }\n\n addSymbol(bucket, anchor, line, shapedTextOrientations, shapedIcon, imageMap, verticallyShapedIcon, bucket.layers[0],\n bucket.collisionBoxArray, feature.index, feature.sourceLayerIndex, bucket.index,\n textBoxScale, [textPadding, textPadding, textPadding, textPadding], textAlongLine, textOffset,\n iconBoxScale, iconPadding, iconAlongLine, iconOffset,\n feature, sizes, isSDFIcon, canonical, layoutTextSize);\n };\n\n if (symbolPlacement === 'line') {\n for (const line of clipLine(feature.geometry, 0, 0, EXTENT, EXTENT)) {\n const anchors = getAnchors(\n line,\n symbolMinDistance,\n textMaxAngle,\n shapedTextOrientations.vertical || defaultHorizontalShaping,\n shapedIcon,\n glyphSize,\n textMaxBoxScale,\n bucket.overscaling,\n EXTENT\n );\n for (const anchor of anchors) {\n const shapedText = defaultHorizontalShaping;\n if (!shapedText || !anchorIsTooClose(bucket, shapedText.text, textRepeatDistance, anchor)) {\n addSymbolAtAnchor(line, anchor);\n }\n }\n }\n } else if (symbolPlacement === 'line-center') {\n // No clipping, multiple lines per feature are allowed\n // \"lines\" with only one point are ignored as in clipLines\n for (const line of feature.geometry) {\n if (line.length > 1) {\n const anchor = getCenterAnchor(\n line,\n textMaxAngle,\n shapedTextOrientations.vertical || defaultHorizontalShaping,\n shapedIcon,\n glyphSize,\n textMaxBoxScale);\n if (anchor) {\n addSymbolAtAnchor(line, anchor);\n }\n }\n }\n } else if (feature.type === 'Polygon') {\n for (const polygon of classifyRings(feature.geometry, 0)) {\n // 16 here represents 2 pixels\n const poi = findPoleOfInaccessibility(polygon, 16);\n addSymbolAtAnchor(polygon[0], new Anchor(poi.x, poi.y, 0));\n }\n } else if (feature.type === 'LineString') {\n // https://github.com/mapbox/mapbox-gl-js/issues/3808\n for (const line of feature.geometry) {\n addSymbolAtAnchor(line, new Anchor(line[0].x, line[0].y, 0));\n }\n } else if (feature.type === 'Point') {\n for (const points of feature.geometry) {\n for (const point of points) {\n addSymbolAtAnchor([point], new Anchor(point.x, point.y, 0));\n }\n }\n }\n}\n\nfunction addTextVariableAnchorOffsets(textAnchorOffsets: TextAnchorOffsetArray, variableAnchorOffset: VariableAnchorOffsetCollection): [number, number] {\n const startIndex = textAnchorOffsets.length;\n const values = variableAnchorOffset?.values;\n\n if (values?.length > 0) {\n for (let i = 0; i < values.length; i += 2) {\n const anchor = TextAnchorEnum[values[i] as TextAnchor];\n const offset = values[i + 1] as [number, number];\n\n textAnchorOffsets.emplaceBack(anchor, offset[0], offset[1]);\n }\n }\n\n return [startIndex, textAnchorOffsets.length];\n}\n\nfunction addTextVertices(bucket: SymbolBucket,\n anchor: Point,\n shapedText: Shaping,\n imageMap: {[_: string]: StyleImage},\n layer: SymbolStyleLayer,\n textAlongLine: boolean,\n feature: SymbolFeature,\n textOffset: [number, number],\n lineArray: {\n lineStartIndex: number;\n lineLength: number;\n },\n writingMode: WritingMode,\n placementTypes: Array<'vertical' | 'center' | 'left' | 'right'>,\n placedTextSymbolIndices: {[_: string]: number},\n placedIconIndex: number,\n sizes: Sizes,\n canonical: CanonicalTileID) {\n const glyphQuads = getGlyphQuads(anchor, shapedText, textOffset,\n layer, textAlongLine, feature, imageMap, bucket.allowVerticalPlacement);\n\n const sizeData = bucket.textSizeData;\n let textSizeData = null;\n\n if (sizeData.kind === 'source') {\n textSizeData = [\n SIZE_PACK_FACTOR * layer.layout.get('text-size').evaluate(feature, {})\n ];\n if (textSizeData[0] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"text-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"text-size\".`);\n }\n } else if (sizeData.kind === 'composite') {\n textSizeData = [\n SIZE_PACK_FACTOR * sizes.compositeTextSizes[0].evaluate(feature, {}, canonical),\n SIZE_PACK_FACTOR * sizes.compositeTextSizes[1].evaluate(feature, {}, canonical)\n ];\n if (textSizeData[0] > MAX_PACKED_SIZE || textSizeData[1] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"text-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"text-size\".`);\n }\n }\n\n bucket.addSymbols(\n bucket.text,\n glyphQuads,\n textSizeData,\n textOffset,\n textAlongLine,\n feature,\n writingMode,\n anchor,\n lineArray.lineStartIndex,\n lineArray.lineLength,\n placedIconIndex,\n canonical);\n\n // The placedSymbolArray is used at render time in drawTileSymbols\n // These indices allow access to the array at collision detection time\n for (const placementType of placementTypes) {\n placedTextSymbolIndices[placementType] = bucket.text.placedSymbolArray.length - 1;\n }\n\n return glyphQuads.length * 4;\n}\n\nfunction getDefaultHorizontalShaping(\n horizontalShaping: Record\n): Shaping | null {\n // We don't care which shaping we get because this is used for collision purposes\n // and all the justifications have the same collision box\n for (const justification in horizontalShaping) {\n return horizontalShaping[justification];\n }\n return null;\n}\n\n/**\n * Add a single label & icon placement.\n */\nfunction addSymbol(bucket: SymbolBucket,\n anchor: Anchor,\n line: Array,\n shapedTextOrientations: ShapedTextOrientations,\n shapedIcon: PositionedIcon | void,\n imageMap: {[_: string]: StyleImage},\n verticallyShapedIcon: PositionedIcon | void,\n layer: SymbolStyleLayer,\n collisionBoxArray: CollisionBoxArray,\n featureIndex: number,\n sourceLayerIndex: number,\n bucketIndex: number,\n textBoxScale: number,\n textPadding: SymbolPadding,\n textAlongLine: boolean,\n textOffset: [number, number],\n iconBoxScale: number,\n iconPadding: SymbolPadding,\n iconAlongLine: boolean,\n iconOffset: [number, number],\n feature: SymbolFeature,\n sizes: Sizes,\n isSDFIcon: boolean,\n canonical: CanonicalTileID,\n layoutTextSize: number) {\n const lineArray = bucket.addToLineVertexArray(anchor, line);\n\n let textCollisionFeature, iconCollisionFeature, verticalTextCollisionFeature, verticalIconCollisionFeature;\n\n let numIconVertices = 0;\n let numVerticalIconVertices = 0;\n let numHorizontalGlyphVertices = 0;\n let numVerticalGlyphVertices = 0;\n let placedIconSymbolIndex = -1;\n let verticalPlacedIconSymbolIndex = -1;\n const placedTextSymbolIndices: {[k: string]: number} = {};\n let key = murmur3('');\n\n if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) {\n const textRotation = layer.layout.get('text-rotate').evaluate(feature, {}, canonical);\n const verticalTextRotation = textRotation + 90.0;\n const verticalShaping = shapedTextOrientations.vertical;\n verticalTextCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticalShaping, textBoxScale, textPadding, textAlongLine, verticalTextRotation);\n\n if (verticallyShapedIcon) {\n verticalIconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticallyShapedIcon, iconBoxScale, iconPadding, textAlongLine, verticalTextRotation);\n }\n }\n\n //Place icon first, so text can have a reference to its index in the placed symbol array.\n //Text symbols can lazily shift at render-time because of variable anchor placement.\n //If the style specifies an `icon-text-fit` then the icon would have to shift along with it.\n // For more info check `updateVariableAnchors` in `draw_symbol.js` .\n if (shapedIcon) {\n const iconRotate = layer.layout.get('icon-rotate').evaluate(feature, {});\n const hasIconTextFit = layer.layout.get('icon-text-fit') !== 'none';\n const iconQuads = getIconQuads(shapedIcon, iconRotate, isSDFIcon, hasIconTextFit);\n const verticalIconQuads = verticallyShapedIcon ? getIconQuads(verticallyShapedIcon, iconRotate, isSDFIcon, hasIconTextFit) : undefined;\n iconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shapedIcon, iconBoxScale, iconPadding, /*align boxes to line*/false, iconRotate);\n\n numIconVertices = iconQuads.length * 4;\n\n const sizeData = bucket.iconSizeData;\n let iconSizeData = null;\n\n if (sizeData.kind === 'source') {\n iconSizeData = [\n SIZE_PACK_FACTOR * layer.layout.get('icon-size').evaluate(feature, {})\n ];\n if (iconSizeData[0] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"icon-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"icon-size\".`);\n }\n } else if (sizeData.kind === 'composite') {\n iconSizeData = [\n SIZE_PACK_FACTOR * sizes.compositeIconSizes[0].evaluate(feature, {}, canonical),\n SIZE_PACK_FACTOR * sizes.compositeIconSizes[1].evaluate(feature, {}, canonical)\n ];\n if (iconSizeData[0] > MAX_PACKED_SIZE || iconSizeData[1] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"icon-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"icon-size\".`);\n }\n }\n\n bucket.addSymbols(\n bucket.icon,\n iconQuads,\n iconSizeData,\n iconOffset,\n iconAlongLine,\n feature,\n WritingMode.none,\n anchor,\n lineArray.lineStartIndex,\n lineArray.lineLength,\n // The icon itself does not have an associated symbol since the text isnt placed yet\n -1, canonical);\n\n placedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1;\n\n if (verticalIconQuads) {\n numVerticalIconVertices = verticalIconQuads.length * 4;\n\n bucket.addSymbols(\n bucket.icon,\n verticalIconQuads,\n iconSizeData,\n iconOffset,\n iconAlongLine,\n feature,\n WritingMode.vertical,\n anchor,\n lineArray.lineStartIndex,\n lineArray.lineLength,\n // The icon itself does not have an associated symbol since the text isnt placed yet\n -1, canonical);\n\n verticalPlacedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1;\n }\n }\n\n const justifications = Object.keys(shapedTextOrientations.horizontal) as TextJustify[];\n for (const justification of justifications) {\n const shaping = shapedTextOrientations.horizontal[justification];\n\n if (!textCollisionFeature) {\n key = murmur3(shaping.text);\n const textRotate = layer.layout.get('text-rotate').evaluate(feature, {}, canonical);\n // As a collision approximation, we can use either the vertical or any of the horizontal versions of the feature\n // We're counting on all versions having similar dimensions\n textCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shaping, textBoxScale, textPadding, textAlongLine, textRotate);\n }\n\n const singleLine = shaping.positionedLines.length === 1;\n numHorizontalGlyphVertices += addTextVertices(\n bucket, anchor, shaping, imageMap, layer, textAlongLine, feature, textOffset, lineArray,\n shapedTextOrientations.vertical ? WritingMode.horizontal : WritingMode.horizontalOnly,\n singleLine ? justifications : [justification],\n placedTextSymbolIndices, placedIconSymbolIndex, sizes, canonical);\n\n if (singleLine) {\n break;\n }\n }\n\n if (shapedTextOrientations.vertical) {\n numVerticalGlyphVertices += addTextVertices(\n bucket, anchor, shapedTextOrientations.vertical, imageMap, layer, textAlongLine, feature,\n textOffset, lineArray, WritingMode.vertical, ['vertical'], placedTextSymbolIndices, verticalPlacedIconSymbolIndex, sizes, canonical);\n }\n\n const textBoxStartIndex = textCollisionFeature ? textCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const textBoxEndIndex = textCollisionFeature ? textCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n const verticalTextBoxStartIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const verticalTextBoxEndIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n const iconBoxStartIndex = iconCollisionFeature ? iconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const iconBoxEndIndex = iconCollisionFeature ? iconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n const verticalIconBoxStartIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const verticalIconBoxEndIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n // Check if runtime collision circles should be used for any of the collision features.\n // It is enough to choose the tallest feature shape as circles are always placed on a line.\n // All measurements are in glyph metrics and later converted into pixels using proper font size \"layoutTextSize\"\n let collisionCircleDiameter = -1;\n\n const getCollisionCircleHeight = (feature: CollisionFeature, prevHeight: number): number => {\n if (feature && feature.circleDiameter)\n return Math.max(feature.circleDiameter, prevHeight);\n return prevHeight;\n };\n\n collisionCircleDiameter = getCollisionCircleHeight(textCollisionFeature, collisionCircleDiameter);\n collisionCircleDiameter = getCollisionCircleHeight(verticalTextCollisionFeature, collisionCircleDiameter);\n collisionCircleDiameter = getCollisionCircleHeight(iconCollisionFeature, collisionCircleDiameter);\n collisionCircleDiameter = getCollisionCircleHeight(verticalIconCollisionFeature, collisionCircleDiameter);\n const useRuntimeCollisionCircles = (collisionCircleDiameter > -1) ? 1 : 0;\n\n // Convert circle collision height into pixels\n if (useRuntimeCollisionCircles)\n collisionCircleDiameter *= layoutTextSize / ONE_EM;\n\n if (bucket.glyphOffsetArray.length >= SymbolBucket.MAX_GLYPHS) warnOnce(\n 'Too many glyphs being rendered in a tile. See https://github.com/mapbox/mapbox-gl-js/issues/2907'\n );\n\n if (feature.sortKey !== undefined) {\n bucket.addToSortKeyRanges(bucket.symbolInstances.length, feature.sortKey as number);\n }\n\n const variableAnchorOffset = getTextVariableAnchorOffset(layer, feature, canonical);\n const [textAnchorOffsetStartIndex, textAnchorOffsetEndIndex] = addTextVariableAnchorOffsets(bucket.textAnchorOffsets, variableAnchorOffset);\n\n bucket.symbolInstances.emplaceBack(\n anchor.x,\n anchor.y,\n placedTextSymbolIndices.right >= 0 ? placedTextSymbolIndices.right : -1,\n placedTextSymbolIndices.center >= 0 ? placedTextSymbolIndices.center : -1,\n placedTextSymbolIndices.left >= 0 ? placedTextSymbolIndices.left : -1,\n placedTextSymbolIndices.vertical || -1,\n placedIconSymbolIndex,\n verticalPlacedIconSymbolIndex,\n key,\n textBoxStartIndex,\n textBoxEndIndex,\n verticalTextBoxStartIndex,\n verticalTextBoxEndIndex,\n iconBoxStartIndex,\n iconBoxEndIndex,\n verticalIconBoxStartIndex,\n verticalIconBoxEndIndex,\n featureIndex,\n numHorizontalGlyphVertices,\n numVerticalGlyphVertices,\n numIconVertices,\n numVerticalIconVertices,\n useRuntimeCollisionCircles,\n 0,\n textBoxScale,\n collisionCircleDiameter,\n textAnchorOffsetStartIndex,\n textAnchorOffsetEndIndex);\n}\n\nfunction anchorIsTooClose(bucket: SymbolBucket, text: string, repeatDistance: number, anchor: Point) {\n const compareText = bucket.compareText;\n if (!(text in compareText)) {\n compareText[text] = [];\n } else {\n const otherAnchors = compareText[text];\n for (let k = otherAnchors.length - 1; k >= 0; k--) {\n if (anchor.dist(otherAnchors[k]) < repeatDistance) {\n // If it's within repeatDistance of one anchor, stop looking\n return true;\n }\n }\n }\n // If anchor is not within repeatDistance of any other anchor, add to array\n compareText[text].push(anchor);\n return false;\n}\n","import Point from '@mapbox/point-geometry';\n\n/**\n * Returns the part of a multiline that intersects with the provided rectangular box.\n *\n * @param lines - the lines to check\n * @param x1 - the left edge of the box\n * @param y1 - the top edge of the box\n * @param x2 - the right edge of the box\n * @param y2 - the bottom edge of the box\n * @returns lines\n */\nexport function clipLine(lines: Array>, x1: number, y1: number, x2: number, y2: number): Array> {\n const clippedLines = [];\n\n for (let l = 0; l < lines.length; l++) {\n const line = lines[l];\n let clippedLine;\n\n for (let i = 0; i < line.length - 1; i++) {\n let p0 = line[i];\n let p1 = line[i + 1];\n\n if (p0.x < x1 && p1.x < x1) {\n continue;\n } else if (p0.x < x1) {\n p0 = new Point(x1, p0.y + (p1.y - p0.y) * ((x1 - p0.x) / (p1.x - p0.x)))._round();\n } else if (p1.x < x1) {\n p1 = new Point(x1, p0.y + (p1.y - p0.y) * ((x1 - p0.x) / (p1.x - p0.x)))._round();\n }\n\n if (p0.y < y1 && p1.y < y1) {\n continue;\n } else if (p0.y < y1) {\n p0 = new Point(p0.x + (p1.x - p0.x) * ((y1 - p0.y) / (p1.y - p0.y)), y1)._round();\n } else if (p1.y < y1) {\n p1 = new Point(p0.x + (p1.x - p0.x) * ((y1 - p0.y) / (p1.y - p0.y)), y1)._round();\n }\n\n if (p0.x >= x2 && p1.x >= x2) {\n continue;\n } else if (p0.x >= x2) {\n p0 = new Point(x2, p0.y + (p1.y - p0.y) * ((x2 - p0.x) / (p1.x - p0.x)))._round();\n } else if (p1.x >= x2) {\n p1 = new Point(x2, p0.y + (p1.y - p0.y) * ((x2 - p0.x) / (p1.x - p0.x)))._round();\n }\n\n if (p0.y >= y2 && p1.y >= y2) {\n continue;\n } else if (p0.y >= y2) {\n p0 = new Point(p0.x + (p1.x - p0.x) * ((y2 - p0.y) / (p1.y - p0.y)), y2)._round();\n } else if (p1.y >= y2) {\n p1 = new Point(p0.x + (p1.x - p0.x) * ((y2 - p0.y) / (p1.y - p0.y)), y2)._round();\n }\n\n if (!clippedLine || !p0.equals(clippedLine[clippedLine.length - 1])) {\n clippedLine = [p0];\n clippedLines.push(clippedLine);\n }\n\n clippedLine.push(p1);\n }\n }\n\n return clippedLines;\n}\n","import {AlphaImage} from '../util/image';\nimport {register} from '../util/web_worker_transfer';\nimport potpack from 'potpack';\n\nimport type {GlyphMetrics, StyleGlyph} from '../style/style_glyph';\n\nconst padding = 1;\n\n/**\n * A rectangle type with postion, width and height.\n */\nexport type Rect = {\n x: number;\n y: number;\n w: number;\n h: number;\n};\n\n/**\n * The glyph's position\n */\nexport type GlyphPosition = {\n rect: Rect;\n metrics: GlyphMetrics;\n};\n\n/**\n * The glyphs' positions\n */\nexport type GlyphPositions = {\n [_: string]: {\n [_: number]: GlyphPosition;\n };\n};\n\nexport class GlyphAtlas {\n image: AlphaImage;\n positions: GlyphPositions;\n\n constructor(stacks: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n }) {\n const positions = {};\n const bins = [];\n\n for (const stack in stacks) {\n const glyphs = stacks[stack];\n const stackPositions = positions[stack] = {};\n\n for (const id in glyphs) {\n const src = glyphs[+id];\n if (!src || src.bitmap.width === 0 || src.bitmap.height === 0) continue;\n\n const bin = {\n x: 0,\n y: 0,\n w: src.bitmap.width + 2 * padding,\n h: src.bitmap.height + 2 * padding\n };\n bins.push(bin);\n stackPositions[id] = {rect: bin, metrics: src.metrics};\n }\n }\n\n const {w, h} = potpack(bins);\n const image = new AlphaImage({width: w || 1, height: h || 1});\n\n for (const stack in stacks) {\n const glyphs = stacks[stack];\n\n for (const id in glyphs) {\n const src = glyphs[+id];\n if (!src || src.bitmap.width === 0 || src.bitmap.height === 0) continue;\n const bin = positions[stack][id].rect;\n AlphaImage.copy(src.bitmap, image, {x: 0, y: 0}, {x: bin.x + padding, y: bin.y + padding}, src.bitmap);\n }\n }\n\n this.image = image;\n this.positions = positions;\n }\n}\n\nregister('GlyphAtlas', GlyphAtlas);\n","export { getURL, getTileBBox, getMercCoords };\n\n\n/**\n * getURL\n *\n * @param {String} baseUrl Base url of the WMS server\n * @param {String} layer Layer name\n * @param {Number} x Tile coordinate x\n * @param {Number} y Tile coordinate y\n * @param {Number} z Tile zoom\n * @param {Object} [options]\n * @param {String} [options.format='image/png']\n * @param {String} [options.service='WMS']\n * @param {String} [options.version='1.1.1']\n * @param {String} [options.request='GetMap']\n * @param {String} [options.srs='EPSG:3857']\n * @param {Number} [options.width='256']\n * @param {Number} [options.height='256']\n * @returns {String} url\n * @example\n * var baseUrl = 'http://geodata.state.nj.us/imagerywms/Natural2015';\n * var layer = 'Natural2015';\n * var url = whoots.getURL(baseUrl, layer, 154308, 197167, 19);\n */\nfunction getURL(baseUrl, layer, x, y, z, options) {\n options = options || {};\n\n var url = baseUrl + '?' + [\n 'bbox=' + getTileBBox(x, y, z),\n 'format=' + (options.format || 'image/png'),\n 'service=' + (options.service || 'WMS'),\n 'version=' + (options.version || '1.1.1'),\n 'request=' + (options.request || 'GetMap'),\n 'srs=' + (options.srs || 'EPSG:3857'),\n 'width=' + (options.width || 256),\n 'height=' + (options.height || 256),\n 'layers=' + layer\n ].join('&');\n\n return url;\n}\n\n\n/**\n * getTileBBox\n *\n * @param {Number} x Tile coordinate x\n * @param {Number} y Tile coordinate y\n * @param {Number} z Tile zoom\n * @returns {String} String of the bounding box\n */\nfunction getTileBBox(x, y, z) {\n // for Google/OSM tile scheme we need to alter the y\n y = (Math.pow(2, z) - y - 1);\n\n var min = getMercCoords(x * 256, y * 256, z),\n max = getMercCoords((x + 1) * 256, (y + 1) * 256, z);\n\n return min[0] + ',' + min[1] + ',' + max[0] + ',' + max[1];\n}\n\n\n/**\n * getMercCoords\n *\n * @param {Number} x Pixel coordinate x\n * @param {Number} y Pixel coordinate y\n * @param {Number} z Tile zoom\n * @returns {Array} [x, y]\n */\nfunction getMercCoords(x, y, z) {\n var resolution = (2 * Math.PI * 6378137 / 256) / Math.pow(2, z),\n merc_x = (x * resolution - 2 * Math.PI * 6378137 / 2.0),\n merc_y = (y * resolution - 2 * Math.PI * 6378137 / 2.0);\n\n return [merc_x, merc_y];\n}\n","import {wrap} from '../util/util';\n\n/*\n* Approximate radius of the earth in meters.\n* Uses the WGS-84 approximation. The radius at the equator is ~6378137 and at the poles is ~6356752. https://en.wikipedia.org/wiki/World_Geodetic_System#WGS84\n* 6371008.8 is one published \"average radius\" see https://en.wikipedia.org/wiki/Earth_radius#Mean_radius, or ftp://athena.fsv.cvut.cz/ZFG/grs80-Moritz.pdf p.4\n*/\nexport const earthRadius = 6371008.8;\n\n/**\n * A {@link LngLat} object, an array of two numbers representing longitude and latitude,\n * or an object with `lng` and `lat` or `lon` and `lat` properties.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let v1 = new maplibregl.LngLat(-122.420679, 37.772537);\n * let v2 = [-122.420679, 37.772537];\n * let v3 = {lon: -122.420679, lat: 37.772537};\n * ```\n */\nexport type LngLatLike = LngLat | {\n lng: number;\n lat: number;\n} | {\n lon: number;\n lat: number;\n} | [number, number];\n\n/**\n * A `LngLat` object represents a given longitude and latitude coordinate, measured in degrees.\n * These coordinates are based on the [WGS84 (EPSG:4326) standard](https://en.wikipedia.org/wiki/World_Geodetic_System#WGS84).\n *\n * MapLibre GL JS uses longitude, latitude coordinate order (as opposed to latitude, longitude) to match the\n * [GeoJSON specification](https://tools.ietf.org/html/rfc7946).\n *\n * Note that any MapLibre GL JS method that accepts a `LngLat` object as an argument or option\n * can also accept an `Array` of two numbers and will perform an implicit conversion.\n * This flexible type is documented as {@link LngLatLike}.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(-123.9749, 40.7736);\n * ll.lng; // = -123.9749\n * ```\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Create a timeline animation](https://maplibre.org/maplibre-gl-js/docs/examples/timeline-animation/)\n */\nexport class LngLat {\n lng: number;\n lat: number;\n\n /**\n * @param lng - Longitude, measured in degrees.\n * @param lat - Latitude, measured in degrees.\n */\n constructor(lng: number, lat: number) {\n if (isNaN(lng) || isNaN(lat)) {\n throw new Error(`Invalid LngLat object: (${lng}, ${lat})`);\n }\n this.lng = +lng;\n this.lat = +lat;\n if (this.lat > 90 || this.lat < -90) {\n throw new Error('Invalid LngLat latitude value: must be between -90 and 90');\n }\n }\n\n /**\n * Returns a new `LngLat` object whose longitude is wrapped to the range (-180, 180).\n *\n * @returns The wrapped `LngLat` object.\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(286.0251, 40.7736);\n * let wrapped = ll.wrap();\n * wrapped.lng; // = -73.9749\n * ```\n */\n wrap() {\n return new LngLat(wrap(this.lng, -180, 180), this.lat);\n }\n\n /**\n * Returns the coordinates represented as an array of two numbers.\n *\n * @returns The coordinates represented as an array of longitude and latitude.\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(-73.9749, 40.7736);\n * ll.toArray(); // = [-73.9749, 40.7736]\n * ```\n */\n toArray(): [number, number] {\n return [this.lng, this.lat];\n }\n\n /**\n * Returns the coordinates represent as a string.\n *\n * @returns The coordinates represented as a string of the format `'LngLat(lng, lat)'`.\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(-73.9749, 40.7736);\n * ll.toString(); // = \"LngLat(-73.9749, 40.7736)\"\n * ```\n */\n toString(): string {\n return `LngLat(${this.lng}, ${this.lat})`;\n }\n\n /**\n * Returns the approximate distance between a pair of coordinates in meters\n * Uses the Haversine Formula (from R.W. Sinnott, \"Virtues of the Haversine\", Sky and Telescope, vol. 68, no. 2, 1984, p. 159)\n *\n * @param lngLat - coordinates to compute the distance to\n * @returns Distance in meters between the two coordinates.\n * @example\n * ```ts\n * let new_york = new maplibregl.LngLat(-74.0060, 40.7128);\n * let los_angeles = new maplibregl.LngLat(-118.2437, 34.0522);\n * new_york.distanceTo(los_angeles); // = 3935751.690893987, \"true distance\" using a non-spherical approximation is ~3966km\n * ```\n */\n distanceTo(lngLat: LngLat): number {\n const rad = Math.PI / 180;\n const lat1 = this.lat * rad;\n const lat2 = lngLat.lat * rad;\n const a = Math.sin(lat1) * Math.sin(lat2) + Math.cos(lat1) * Math.cos(lat2) * Math.cos((lngLat.lng - this.lng) * rad);\n\n const maxMeters = earthRadius * Math.acos(Math.min(a, 1));\n return maxMeters;\n }\n\n /**\n * Converts an array of two numbers or an object with `lng` and `lat` or `lon` and `lat` properties\n * to a `LngLat` object.\n *\n * If a `LngLat` object is passed in, the function returns it unchanged.\n *\n * @param input - An array of two numbers or object to convert, or a `LngLat` object to return.\n * @returns A new `LngLat` object, if a conversion occurred, or the original `LngLat` object.\n * @example\n * ```ts\n * let arr = [-73.9749, 40.7736];\n * let ll = maplibregl.LngLat.convert(arr);\n * ll; // = LngLat {lng: -73.9749, lat: 40.7736}\n * ```\n */\n static convert(input: LngLatLike): LngLat {\n if (input instanceof LngLat) {\n return input;\n }\n if (Array.isArray(input) && (input.length === 2 || input.length === 3)) {\n return new LngLat(Number(input[0]), Number(input[1]));\n }\n if (!Array.isArray(input) && typeof input === 'object' && input !== null) {\n return new LngLat(\n // flow can't refine this to have one of lng or lat, so we have to cast to any\n Number('lng' in input ? (input as any).lng : (input as any).lon),\n Number(input.lat)\n );\n }\n throw new Error('`LngLatLike` argument must be specified as a LngLat instance, an object {lng: , lat: }, an object {lon: , lat: }, or an array of [, ]');\n }\n}\n","import {LngLat, earthRadius} from '../geo/lng_lat';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport {IMercatorCoordinate} from '@maplibre/maplibre-gl-style-spec';\n\n/*\n * The average circumference of the world in meters.\n */\nconst earthCircumfrence = 2 * Math.PI * earthRadius; // meters\n\n/*\n * The circumference at a line of latitude in meters.\n */\nfunction circumferenceAtLatitude(latitude: number) {\n return earthCircumfrence * Math.cos(latitude * Math.PI / 180);\n}\n\nexport function mercatorXfromLng(lng: number) {\n return (180 + lng) / 360;\n}\n\nexport function mercatorYfromLat(lat: number) {\n return (180 - (180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360)))) / 360;\n}\n\nexport function mercatorZfromAltitude(altitude: number, lat: number) {\n return altitude / circumferenceAtLatitude(lat);\n}\n\nexport function lngFromMercatorX(x: number) {\n return x * 360 - 180;\n}\n\nexport function latFromMercatorY(y: number) {\n const y2 = 180 - y * 360;\n return 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90;\n}\n\nexport function altitudeFromMercatorZ(z: number, y: number) {\n return z * circumferenceAtLatitude(latFromMercatorY(y));\n}\n\n/**\n * Determine the Mercator scale factor for a given latitude, see\n * https://en.wikipedia.org/wiki/Mercator_projection#Scale_factor\n *\n * At the equator the scale factor will be 1, which increases at higher latitudes.\n *\n * @param lat - Latitude\n * @returns scale factor\n */\nexport function mercatorScale(lat: number) {\n return 1 / Math.cos(lat * Math.PI / 180);\n}\n\n/**\n * A `MercatorCoordinate` object represents a projected three dimensional position.\n *\n * `MercatorCoordinate` uses the web mercator projection ([EPSG:3857](https://epsg.io/3857)) with slightly different units:\n * - the size of 1 unit is the width of the projected world instead of the \"mercator meter\"\n * - the origin of the coordinate space is at the north-west corner instead of the middle\n *\n * For example, `MercatorCoordinate(0, 0, 0)` is the north-west corner of the mercator world and\n * `MercatorCoordinate(1, 1, 0)` is the south-east corner. If you are familiar with\n * [vector tiles](https://github.com/mapbox/vector-tile-spec) it may be helpful to think\n * of the coordinate space as the `0/0/0` tile with an extent of `1`.\n *\n * The `z` dimension of `MercatorCoordinate` is conformal. A cube in the mercator coordinate space would be rendered as a cube.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let nullIsland = new maplibregl.MercatorCoordinate(0.5, 0.5, 0);\n * ```\n * @see [Add a custom style layer](https://maplibre.org/maplibre-gl-js/docs/examples/custom-style-layer/)\n */\nexport class MercatorCoordinate implements IMercatorCoordinate {\n x: number;\n y: number;\n z: number;\n\n /**\n * @param x - The x component of the position.\n * @param y - The y component of the position.\n * @param z - The z component of the position.\n */\n constructor(x: number, y: number, z: number = 0) {\n this.x = +x;\n this.y = +y;\n this.z = +z;\n }\n\n /**\n * Project a `LngLat` to a `MercatorCoordinate`.\n *\n * @param lngLatLike - The location to project.\n * @param altitude - The altitude in meters of the position.\n * @returns The projected mercator coordinate.\n * @example\n * ```ts\n * let coord = maplibregl.MercatorCoordinate.fromLngLat({ lng: 0, lat: 0}, 0);\n * coord; // MercatorCoordinate(0.5, 0.5, 0)\n * ```\n */\n static fromLngLat(lngLatLike: LngLatLike, altitude: number = 0): MercatorCoordinate {\n const lngLat = LngLat.convert(lngLatLike);\n\n return new MercatorCoordinate(\n mercatorXfromLng(lngLat.lng),\n mercatorYfromLat(lngLat.lat),\n mercatorZfromAltitude(altitude, lngLat.lat));\n }\n\n /**\n * Returns the `LngLat` for the coordinate.\n *\n * @returns The `LngLat` object.\n * @example\n * ```ts\n * let coord = new maplibregl.MercatorCoordinate(0.5, 0.5, 0);\n * let lngLat = coord.toLngLat(); // LngLat(0, 0)\n * ```\n */\n toLngLat() {\n return new LngLat(\n lngFromMercatorX(this.x),\n latFromMercatorY(this.y));\n }\n\n /**\n * Returns the altitude in meters of the coordinate.\n *\n * @returns The altitude in meters.\n * @example\n * ```ts\n * let coord = new maplibregl.MercatorCoordinate(0, 0, 0.02);\n * coord.toAltitude(); // 6914.281956295339\n * ```\n */\n toAltitude(): number {\n return altitudeFromMercatorZ(this.z, this.y);\n }\n\n /**\n * Returns the distance of 1 meter in `MercatorCoordinate` units at this latitude.\n *\n * For coordinates in real world units using meters, this naturally provides the scale\n * to transform into `MercatorCoordinate`s.\n *\n * @returns Distance of 1 meter in `MercatorCoordinate` units.\n */\n meterInMercatorCoordinateUnits(): number {\n // 1 meter / circumference at equator in meters * Mercator projection scale factor at this latitude\n return 1 / earthCircumfrence * mercatorScale(latFromMercatorY(this.y));\n }\n}\n","import {getTileBBox} from '@mapbox/whoots-js';\nimport {EXTENT} from '../data/extent';\nimport Point from '@mapbox/point-geometry';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\nimport {register} from '../util/web_worker_transfer';\nimport {mat4} from 'gl-matrix';\nimport {ICanonicalTileID, IMercatorCoordinate} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A canonical way to define a tile ID\n */\nexport class CanonicalTileID implements ICanonicalTileID {\n z: number;\n x: number;\n y: number;\n key: string;\n\n constructor(z: number, x: number, y: number) {\n\n if (z < 0 || z > 25 || y < 0 || y >= Math.pow(2, z) || x < 0 || x >= Math.pow(2, z)) {\n throw new Error(`x=${x}, y=${y}, z=${z} outside of bounds. 0<=x<${Math.pow(2, z)}, 0<=y<${Math.pow(2, z)} 0<=z<=25 `);\n }\n\n this.z = z;\n this.x = x;\n this.y = y;\n this.key = calculateKey(0, z, z, x, y);\n }\n\n equals(id: ICanonicalTileID) {\n return this.z === id.z && this.x === id.x && this.y === id.y;\n }\n\n // given a list of urls, choose a url template and return a tile URL\n url(urls: Array, pixelRatio: number, scheme?: string | null) {\n const bbox = getTileBBox(this.x, this.y, this.z);\n const quadkey = getQuadkey(this.z, this.x, this.y);\n\n return urls[(this.x + this.y) % urls.length]\n .replace(/{prefix}/g, (this.x % 16).toString(16) + (this.y % 16).toString(16))\n .replace(/{z}/g, String(this.z))\n .replace(/{x}/g, String(this.x))\n .replace(/{y}/g, String(scheme === 'tms' ? (Math.pow(2, this.z) - this.y - 1) : this.y))\n .replace(/{ratio}/g, pixelRatio > 1 ? '@2x' : '')\n .replace(/{quadkey}/g, quadkey)\n .replace(/{bbox-epsg-3857}/g, bbox);\n }\n\n isChildOf(parent: ICanonicalTileID) {\n const dz = this.z - parent.z;\n return dz > 0 && parent.x === (this.x >> dz) && parent.y === (this.y >> dz);\n }\n\n getTilePoint(coord: IMercatorCoordinate) {\n const tilesAtZoom = Math.pow(2, this.z);\n return new Point(\n (coord.x * tilesAtZoom - this.x) * EXTENT,\n (coord.y * tilesAtZoom - this.y) * EXTENT);\n }\n\n toString() {\n return `${this.z}/${this.x}/${this.y}`;\n }\n}\n\n/**\n * @internal\n * An unwrapped tile identifier\n */\nexport class UnwrappedTileID {\n wrap: number;\n canonical: CanonicalTileID;\n key: string;\n\n constructor(wrap: number, canonical: CanonicalTileID) {\n this.wrap = wrap;\n this.canonical = canonical;\n this.key = calculateKey(wrap, canonical.z, canonical.z, canonical.x, canonical.y);\n }\n}\n\n/**\n * An overscaled tile identifier\n */\nexport class OverscaledTileID {\n overscaledZ: number;\n wrap: number;\n canonical: CanonicalTileID;\n key: string;\n posMatrix: mat4;\n\n constructor(overscaledZ: number, wrap: number, z: number, x: number, y: number) {\n if (overscaledZ < z) throw new Error(`overscaledZ should be >= z; overscaledZ = ${overscaledZ}; z = ${z}`);\n this.overscaledZ = overscaledZ;\n this.wrap = wrap;\n this.canonical = new CanonicalTileID(z, +x, +y);\n this.key = calculateKey(wrap, overscaledZ, z, x, y);\n }\n\n clone() {\n return new OverscaledTileID(this.overscaledZ, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y);\n }\n\n equals(id: OverscaledTileID) {\n return this.overscaledZ === id.overscaledZ && this.wrap === id.wrap && this.canonical.equals(id.canonical);\n }\n\n scaledTo(targetZ: number) {\n if (targetZ > this.overscaledZ) throw new Error(`targetZ > this.overscaledZ; targetZ = ${targetZ}; overscaledZ = ${this.overscaledZ}`);\n const zDifference = this.canonical.z - targetZ;\n if (targetZ > this.canonical.z) {\n return new OverscaledTileID(targetZ, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y);\n } else {\n return new OverscaledTileID(targetZ, this.wrap, targetZ, this.canonical.x >> zDifference, this.canonical.y >> zDifference);\n }\n }\n\n /*\n * calculateScaledKey is an optimization:\n * when withWrap == true, implements the same as this.scaledTo(z).key,\n * when withWrap == false, implements the same as this.scaledTo(z).wrapped().key.\n */\n calculateScaledKey(targetZ: number, withWrap: boolean): string {\n if (targetZ > this.overscaledZ) throw new Error(`targetZ > this.overscaledZ; targetZ = ${targetZ}; overscaledZ = ${this.overscaledZ}`);\n const zDifference = this.canonical.z - targetZ;\n if (targetZ > this.canonical.z) {\n return calculateKey(this.wrap * +withWrap, targetZ, this.canonical.z, this.canonical.x, this.canonical.y);\n } else {\n return calculateKey(this.wrap * +withWrap, targetZ, targetZ, this.canonical.x >> zDifference, this.canonical.y >> zDifference);\n }\n }\n\n isChildOf(parent: OverscaledTileID) {\n if (parent.wrap !== this.wrap) {\n // We can't be a child if we're in a different world copy\n return false;\n }\n const zDifference = this.canonical.z - parent.canonical.z;\n // We're first testing for z == 0, to avoid a 32 bit shift, which is undefined.\n return parent.overscaledZ === 0 || (\n parent.overscaledZ < this.overscaledZ &&\n parent.canonical.x === (this.canonical.x >> zDifference) &&\n parent.canonical.y === (this.canonical.y >> zDifference));\n }\n\n children(sourceMaxZoom: number) {\n if (this.overscaledZ >= sourceMaxZoom) {\n // return a single tile coord representing a an overscaled tile\n return [new OverscaledTileID(this.overscaledZ + 1, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y)];\n }\n\n const z = this.canonical.z + 1;\n const x = this.canonical.x * 2;\n const y = this.canonical.y * 2;\n return [\n new OverscaledTileID(z, this.wrap, z, x, y),\n new OverscaledTileID(z, this.wrap, z, x + 1, y),\n new OverscaledTileID(z, this.wrap, z, x, y + 1),\n new OverscaledTileID(z, this.wrap, z, x + 1, y + 1)\n ];\n }\n\n isLessThan(rhs: OverscaledTileID) {\n if (this.wrap < rhs.wrap) return true;\n if (this.wrap > rhs.wrap) return false;\n\n if (this.overscaledZ < rhs.overscaledZ) return true;\n if (this.overscaledZ > rhs.overscaledZ) return false;\n\n if (this.canonical.x < rhs.canonical.x) return true;\n if (this.canonical.x > rhs.canonical.x) return false;\n\n if (this.canonical.y < rhs.canonical.y) return true;\n return false;\n }\n\n wrapped() {\n return new OverscaledTileID(this.overscaledZ, 0, this.canonical.z, this.canonical.x, this.canonical.y);\n }\n\n unwrapTo(wrap: number) {\n return new OverscaledTileID(this.overscaledZ, wrap, this.canonical.z, this.canonical.x, this.canonical.y);\n }\n\n overscaleFactor() {\n return Math.pow(2, this.overscaledZ - this.canonical.z);\n }\n\n toUnwrapped() {\n return new UnwrappedTileID(this.wrap, this.canonical);\n }\n\n toString() {\n return `${this.overscaledZ}/${this.canonical.x}/${this.canonical.y}`;\n }\n\n getTilePoint(coord: MercatorCoordinate) {\n return this.canonical.getTilePoint(new MercatorCoordinate(coord.x - this.wrap, coord.y));\n }\n}\n\nfunction calculateKey(wrap: number, overscaledZ: number, z: number, x: number, y: number): string {\n wrap *= 2;\n if (wrap < 0) wrap = wrap * -1 - 1;\n const dim = 1 << z;\n return (dim * dim * wrap + dim * y + x).toString(36) + z.toString(36) + overscaledZ.toString(36);\n}\n\nfunction getQuadkey(z, x, y) {\n let quadkey = '', mask;\n for (let i = z; i > 0; i--) {\n mask = 1 << (i - 1);\n quadkey += ((x & mask ? 1 : 0) + (y & mask ? 2 : 0));\n }\n return quadkey;\n}\n\nregister('CanonicalTileID', CanonicalTileID);\nregister('OverscaledTileID', OverscaledTileID, {omit: ['posMatrix']});\n","import {FeatureIndex} from '../data/feature_index';\nimport {performSymbolLayout} from '../symbol/symbol_layout';\nimport {CollisionBoxArray} from '../data/array_types.g';\nimport {DictionaryCoder} from '../util/dictionary_coder';\nimport {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {LineBucket} from '../data/bucket/line_bucket';\nimport {FillBucket} from '../data/bucket/fill_bucket';\nimport {FillExtrusionBucket} from '../data/bucket/fill_extrusion_bucket';\nimport {warnOnce, mapObject} from '../util/util';\nimport {ImageAtlas} from '../render/image_atlas';\nimport {GlyphAtlas} from '../render/glyph_atlas';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {OverscaledTileID} from './tile_id';\n\nimport type {Bucket} from '../data/bucket';\nimport type {Actor} from '../util/actor';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {StyleLayerIndex} from '../style/style_layer_index';\nimport type {StyleImage} from '../style/style_image';\nimport type {StyleGlyph} from '../style/style_glyph';\nimport type {\n WorkerTileParameters,\n WorkerTileCallback,\n} from '../source/worker_source';\nimport type {PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {VectorTile} from '@mapbox/vector-tile';\nimport {Cancelable} from '../types/cancelable';\n\nexport class WorkerTile {\n tileID: OverscaledTileID;\n uid: string;\n zoom: number;\n pixelRatio: number;\n tileSize: number;\n source: string;\n promoteId: PromoteIdSpecification;\n overscaling: number;\n showCollisionBoxes: boolean;\n collectResourceTiming: boolean;\n returnDependencies: boolean;\n\n status: 'parsing' | 'done';\n data: VectorTile;\n collisionBoxArray: CollisionBoxArray;\n\n abort: (() => void);\n vectorTile: VectorTile;\n inFlightDependencies: Cancelable[];\n dependencySentinel: number;\n\n constructor(params: WorkerTileParameters) {\n this.tileID = new OverscaledTileID(params.tileID.overscaledZ, params.tileID.wrap, params.tileID.canonical.z, params.tileID.canonical.x, params.tileID.canonical.y);\n this.uid = params.uid;\n this.zoom = params.zoom;\n this.pixelRatio = params.pixelRatio;\n this.tileSize = params.tileSize;\n this.source = params.source;\n this.overscaling = this.tileID.overscaleFactor();\n this.showCollisionBoxes = params.showCollisionBoxes;\n this.collectResourceTiming = !!params.collectResourceTiming;\n this.returnDependencies = !!params.returnDependencies;\n this.promoteId = params.promoteId;\n this.inFlightDependencies = [];\n this.dependencySentinel = -1;\n }\n\n parse(data: VectorTile, layerIndex: StyleLayerIndex, availableImages: Array, actor: Actor, callback: WorkerTileCallback) {\n this.status = 'parsing';\n this.data = data;\n\n this.collisionBoxArray = new CollisionBoxArray();\n const sourceLayerCoder = new DictionaryCoder(Object.keys(data.layers).sort());\n\n const featureIndex = new FeatureIndex(this.tileID, this.promoteId);\n featureIndex.bucketLayerIDs = [];\n\n const buckets: {[_: string]: Bucket} = {};\n\n const options = {\n featureIndex,\n iconDependencies: {},\n patternDependencies: {},\n glyphDependencies: {},\n availableImages\n };\n\n const layerFamilies = layerIndex.familiesBySource[this.source];\n for (const sourceLayerId in layerFamilies) {\n const sourceLayer = data.layers[sourceLayerId];\n if (!sourceLayer) {\n continue;\n }\n\n if (sourceLayer.version === 1) {\n warnOnce(`Vector tile source \"${this.source}\" layer \"${sourceLayerId}\" ` +\n 'does not use vector tile spec v2 and therefore may have some rendering errors.');\n }\n\n const sourceLayerIndex = sourceLayerCoder.encode(sourceLayerId);\n const features = [];\n for (let index = 0; index < sourceLayer.length; index++) {\n const feature = sourceLayer.feature(index);\n const id = featureIndex.getId(feature, sourceLayerId);\n features.push({feature, id, index, sourceLayerIndex});\n }\n\n for (const family of layerFamilies[sourceLayerId]) {\n const layer = family[0];\n\n if (layer.source !== this.source) {\n warnOnce(`layer.source = ${layer.source} does not equal this.source = ${this.source}`);\n }\n if (layer.minzoom && this.zoom < Math.floor(layer.minzoom)) continue;\n if (layer.maxzoom && this.zoom >= layer.maxzoom) continue;\n if (layer.visibility === 'none') continue;\n\n recalculateLayers(family, this.zoom, availableImages);\n\n const bucket = buckets[layer.id] = layer.createBucket({\n index: featureIndex.bucketLayerIDs.length,\n layers: family,\n zoom: this.zoom,\n pixelRatio: this.pixelRatio,\n overscaling: this.overscaling,\n collisionBoxArray: this.collisionBoxArray,\n sourceLayerIndex,\n sourceID: this.source\n });\n\n bucket.populate(features, options, this.tileID.canonical);\n featureIndex.bucketLayerIDs.push(family.map((l) => l.id));\n }\n }\n\n let error: Error;\n let glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n };\n let iconMap: {[_: string]: StyleImage};\n let patternMap: {[_: string]: StyleImage};\n\n const stacks = mapObject(options.glyphDependencies, (glyphs) => Object.keys(glyphs).map(Number));\n\n this.inFlightDependencies.forEach((request) => request?.cancel());\n this.inFlightDependencies = [];\n\n // cancelling seems to be not sufficient, we seems to still manage to get a callback hit, so use a sentinel to drop stale results\n const dependencySentinel = ++this.dependencySentinel;\n if (Object.keys(stacks).length) {\n this.inFlightDependencies.push(actor.send('getGlyphs', {uid: this.uid, stacks, source: this.source, tileID: this.tileID, type: 'glyphs'}, (err, result) => {\n if (dependencySentinel !== this.dependencySentinel) {\n return;\n }\n if (!error) {\n error = err;\n glyphMap = result;\n maybePrepare.call(this);\n }\n }));\n } else {\n glyphMap = {};\n }\n\n const icons = Object.keys(options.iconDependencies);\n if (icons.length) {\n this.inFlightDependencies.push(actor.send('getImages', {icons, source: this.source, tileID: this.tileID, type: 'icons'}, (err, result) => {\n if (dependencySentinel !== this.dependencySentinel) {\n return;\n }\n if (!error) {\n error = err;\n iconMap = result;\n maybePrepare.call(this);\n }\n }));\n } else {\n iconMap = {};\n }\n\n const patterns = Object.keys(options.patternDependencies);\n if (patterns.length) {\n this.inFlightDependencies.push(actor.send('getImages', {icons: patterns, source: this.source, tileID: this.tileID, type: 'patterns'}, (err, result) => {\n if (dependencySentinel !== this.dependencySentinel) {\n return;\n }\n if (!error) {\n error = err;\n patternMap = result;\n maybePrepare.call(this);\n }\n }));\n } else {\n patternMap = {};\n }\n\n maybePrepare.call(this);\n\n function maybePrepare() {\n if (error) {\n return callback(error);\n } else if (glyphMap && iconMap && patternMap) {\n const glyphAtlas = new GlyphAtlas(glyphMap);\n const imageAtlas = new ImageAtlas(iconMap, patternMap);\n\n for (const key in buckets) {\n const bucket = buckets[key];\n if (bucket instanceof SymbolBucket) {\n recalculateLayers(bucket.layers, this.zoom, availableImages);\n performSymbolLayout({\n bucket,\n glyphMap,\n glyphPositions: glyphAtlas.positions,\n imageMap: iconMap,\n imagePositions: imageAtlas.iconPositions,\n showCollisionBoxes: this.showCollisionBoxes,\n canonical: this.tileID.canonical\n });\n } else if (bucket.hasPattern &&\n (bucket instanceof LineBucket ||\n bucket instanceof FillBucket ||\n bucket instanceof FillExtrusionBucket)) {\n recalculateLayers(bucket.layers, this.zoom, availableImages);\n bucket.addFeatures(options, this.tileID.canonical, imageAtlas.patternPositions);\n }\n }\n\n this.status = 'done';\n callback(null, {\n buckets: Object.values(buckets).filter(b => !b.isEmpty()),\n featureIndex,\n collisionBoxArray: this.collisionBoxArray,\n glyphAtlasImage: glyphAtlas.image,\n imageAtlas,\n // Only used for benchmarking:\n glyphMap: this.returnDependencies ? glyphMap : null,\n iconMap: this.returnDependencies ? iconMap : null,\n glyphPositions: this.returnDependencies ? glyphAtlas.positions : null\n });\n }\n }\n }\n}\n\nfunction recalculateLayers(layers: ReadonlyArray, zoom: number, availableImages: Array) {\n // Layers are shared and may have been used by a WorkerTile with a different zoom.\n const parameters = new EvaluationParameters(zoom);\n for (const layer of layers) {\n layer.recalculate(parameters, availableImages);\n }\n}\n","import type {RequestParameters} from '../util/ajax';\n\nexport type PerformanceMetrics = {\n loadTime: number;\n fullLoadTime: number;\n fps: number;\n percentDroppedFrames: number;\n totalFrames: number;\n};\n\nexport enum PerformanceMarkers {\n create = 'create',\n load = 'load',\n fullLoad = 'fullLoad'\n}\n\nlet lastFrameTime = null;\nlet frameTimes = [];\n\nconst minFramerateTarget = 60;\nconst frameTimeTarget = 1000 / minFramerateTarget;\n\nconst loadTimeKey = 'loadTime';\nconst fullLoadTimeKey = 'fullLoadTime';\n\nexport const PerformanceUtils = {\n mark(marker: PerformanceMarkers) {\n performance.mark(marker);\n },\n frame(timestamp: number) {\n const currTimestamp = timestamp;\n if (lastFrameTime != null) {\n const frameTime = currTimestamp - lastFrameTime;\n frameTimes.push(frameTime);\n }\n lastFrameTime = currTimestamp;\n },\n clearMetrics() {\n lastFrameTime = null;\n frameTimes = [];\n performance.clearMeasures(loadTimeKey);\n performance.clearMeasures(fullLoadTimeKey);\n\n for (const marker in PerformanceMarkers) {\n performance.clearMarks(PerformanceMarkers[marker]);\n }\n },\n\n getPerformanceMetrics(): PerformanceMetrics {\n performance.measure(loadTimeKey, PerformanceMarkers.create, PerformanceMarkers.load);\n performance.measure(fullLoadTimeKey, PerformanceMarkers.create, PerformanceMarkers.fullLoad);\n const loadTime = performance.getEntriesByName(loadTimeKey)[0].duration;\n const fullLoadTime = performance.getEntriesByName(fullLoadTimeKey)[0].duration;\n const totalFrames = frameTimes.length;\n\n const avgFrameTime = frameTimes.reduce((prev, curr) => prev + curr, 0) / totalFrames / 1000;\n const fps = 1 / avgFrameTime;\n\n // count frames that missed our framerate target\n const droppedFrames = frameTimes\n .filter((frameTime) => frameTime > frameTimeTarget)\n .reduce((acc, curr) => {\n return acc + (curr - frameTimeTarget) / frameTimeTarget;\n }, 0);\n const percentDroppedFrames = (droppedFrames / (totalFrames + droppedFrames)) * 100;\n\n return {\n loadTime,\n fullLoadTime,\n fps,\n percentDroppedFrames,\n totalFrames\n };\n }\n};\n\n/**\n * @internal\n * Safe wrapper for the performance resource timing API in web workers with graceful degradation\n */\nexport class RequestPerformance {\n _marks: {\n start: string;\n end: string;\n measure: string;\n };\n\n constructor (request: RequestParameters) {\n this._marks = {\n start: [request.url, 'start'].join('#'),\n end: [request.url, 'end'].join('#'),\n measure: request.url.toString()\n };\n\n performance.mark(this._marks.start);\n }\n\n finish() {\n performance.mark(this._marks.end);\n let resourceTimingData = performance.getEntriesByName(this._marks.measure);\n\n // fallback if web worker implementation of perf.getEntriesByName returns empty\n if (resourceTimingData.length === 0) {\n performance.measure(this._marks.measure, this._marks.start, this._marks.end);\n resourceTimingData = performance.getEntriesByName(this._marks.measure);\n\n // cleanup\n performance.clearMarks(this._marks.start);\n performance.clearMarks(this._marks.end);\n performance.clearMeasures(this._marks.measure);\n }\n\n return resourceTimingData;\n }\n}\n\nexport default performance;\n","import {ExpiryData, getArrayBuffer} from '../util/ajax';\n\nimport vt from '@mapbox/vector-tile';\nimport Protobuf from 'pbf';\nimport {WorkerTile} from './worker_tile';\nimport {extend} from '../util/util';\nimport {RequestPerformance} from '../util/performance';\n\nimport type {\n WorkerSource,\n WorkerTileParameters,\n WorkerTileCallback,\n TileParameters\n} from '../source/worker_source';\n\nimport type {Actor} from '../util/actor';\nimport type {StyleLayerIndex} from '../style/style_layer_index';\nimport type {Callback} from '../types/callback';\nimport type {VectorTile} from '@mapbox/vector-tile';\n\nexport type LoadVectorTileResult = {\n vectorTile: VectorTile;\n rawData: ArrayBuffer;\n resourceTiming?: Array;\n} & ExpiryData;\n\ntype FetchingState = {\n rawTileData: ArrayBuffer;\n cacheControl: ExpiryData;\n resourceTiming: any;\n}\n\n/**\n * The callback when finished loading vector data\n */\nexport type LoadVectorDataCallback = Callback;\n\nexport type AbortVectorData = () => void;\nexport type LoadVectorData = (params: WorkerTileParameters, callback: LoadVectorDataCallback) => AbortVectorData | void;\n\n/**\n * Loads a vector tile\n */\nfunction loadVectorTile(params: WorkerTileParameters, callback: LoadVectorDataCallback) {\n const request = getArrayBuffer(params.request, (err?: Error | null, data?: ArrayBuffer | null, cacheControl?: string | null, expires?: string | null) => {\n if (err) {\n callback(err);\n } else if (data) {\n try {\n const vectorTile = new vt.VectorTile(new Protobuf(data));\n callback(null, {\n vectorTile,\n rawData: data,\n cacheControl,\n expires\n });\n } catch (ex) {\n const bytes = new Uint8Array(data);\n const isGzipped = bytes[0] === 0x1f && bytes[1] === 0x8b;\n let errorMessage = `Unable to parse the tile at ${params.request.url}, `;\n if (isGzipped) {\n errorMessage += 'please make sure the data is not gzipped and that you have configured the relevant header in the server';\n } else {\n errorMessage += `got error: ${ex.messge}`;\n }\n callback(new Error(errorMessage));\n }\n }\n });\n return () => {\n request.cancel();\n callback();\n };\n}\n\n/**\n * The {@link WorkerSource} implementation that supports {@link VectorTileSource}.\n * This class is designed to be easily reused to support custom source types\n * for data formats that can be parsed/converted into an in-memory VectorTile\n * representation. To do so, create it with\n * `new VectorTileWorkerSource(actor, styleLayers, customLoadVectorDataFunction)`.\n */\nexport class VectorTileWorkerSource implements WorkerSource {\n actor: Actor;\n layerIndex: StyleLayerIndex;\n availableImages: Array;\n loadVectorData: LoadVectorData;\n fetching: {[_: string]: FetchingState };\n loading: {[_: string]: WorkerTile};\n loaded: {[_: string]: WorkerTile};\n\n /**\n * @param loadVectorData - Optional method for custom loading of a VectorTile\n * object based on parameters passed from the main-thread Source. See\n * {@link VectorTileWorkerSource#loadTile}. The default implementation simply\n * loads the pbf at `params.url`.\n */\n constructor(actor: Actor, layerIndex: StyleLayerIndex, availableImages: Array, loadVectorData?: LoadVectorData | null) {\n this.actor = actor;\n this.layerIndex = layerIndex;\n this.availableImages = availableImages;\n this.loadVectorData = loadVectorData || loadVectorTile;\n this.fetching = {};\n this.loading = {};\n this.loaded = {};\n }\n\n /**\n * Implements {@link WorkerSource#loadTile}. Delegates to\n * {@link VectorTileWorkerSource#loadVectorData} (which by default expects\n * a `params.url` property) for fetching and producing a VectorTile object.\n */\n loadTile(params: WorkerTileParameters, callback: WorkerTileCallback) {\n const uid = params.uid;\n\n if (!this.loading)\n this.loading = {};\n\n const perf = (params && params.request && params.request.collectResourceTiming) ?\n new RequestPerformance(params.request) : false;\n\n const workerTile = this.loading[uid] = new WorkerTile(params);\n workerTile.abort = this.loadVectorData(params, (err, response) => {\n delete this.loading[uid];\n\n if (err || !response) {\n workerTile.status = 'done';\n this.loaded[uid] = workerTile;\n return callback(err);\n }\n\n const rawTileData = response.rawData;\n const cacheControl = {} as ExpiryData;\n if (response.expires) cacheControl.expires = response.expires;\n if (response.cacheControl) cacheControl.cacheControl = response.cacheControl;\n\n const resourceTiming = {} as {resourceTiming: any};\n if (perf) {\n const resourceTimingData = perf.finish();\n // it's necessary to eval the result of getEntriesByName() here via parse/stringify\n // late evaluation in the main thread causes TypeError: illegal invocation\n if (resourceTimingData)\n resourceTiming.resourceTiming = JSON.parse(JSON.stringify(resourceTimingData));\n }\n\n workerTile.vectorTile = response.vectorTile;\n workerTile.parse(response.vectorTile, this.layerIndex, this.availableImages, this.actor, (err, result) => {\n delete this.fetching[uid];\n if (err || !result) return callback(err);\n\n // Transferring a copy of rawTileData because the worker needs to retain its copy.\n callback(null, extend({rawTileData: rawTileData.slice(0)}, result, cacheControl, resourceTiming));\n });\n\n this.loaded = this.loaded || {};\n this.loaded[uid] = workerTile;\n // keep the original fetching state so that reload tile can pick it up if the original parse is cancelled by reloads' parse\n this.fetching[uid] = {rawTileData, cacheControl, resourceTiming};\n }) as AbortVectorData;\n }\n\n /**\n * Implements {@link WorkerSource#reloadTile}.\n */\n reloadTile(params: WorkerTileParameters, callback: WorkerTileCallback) {\n const loaded = this.loaded;\n const uid = params.uid;\n if (loaded && loaded[uid]) {\n const workerTile = loaded[uid];\n workerTile.showCollisionBoxes = params.showCollisionBoxes;\n if (workerTile.status === 'parsing') {\n workerTile.parse(workerTile.vectorTile, this.layerIndex, this.availableImages, this.actor, (err, result) => {\n if (err || !result) return callback(err, result);\n\n // if we have cancelled the original parse, make sure to pass the rawTileData from the original fetch\n let parseResult;\n if (this.fetching[uid]) {\n const {rawTileData, cacheControl, resourceTiming} = this.fetching[uid];\n delete this.fetching[uid];\n parseResult = extend({rawTileData: rawTileData.slice(0)}, result, cacheControl, resourceTiming);\n } else {\n parseResult = result;\n }\n\n callback(null, parseResult);\n });\n } else if (workerTile.status === 'done') {\n // if there was no vector tile data on the initial load, don't try and re-parse tile\n if (workerTile.vectorTile) {\n workerTile.parse(workerTile.vectorTile, this.layerIndex, this.availableImages, this.actor, callback);\n } else {\n callback();\n }\n }\n }\n }\n\n /**\n * Implements {@link WorkerSource#abortTile}.\n *\n * @param params - The tile parameters\n * @param callback - The callback\n */\n abortTile(params: TileParameters, callback: WorkerTileCallback) {\n const loading = this.loading,\n uid = params.uid;\n if (loading && loading[uid] && loading[uid].abort) {\n loading[uid].abort();\n delete loading[uid];\n }\n callback();\n }\n\n /**\n * Implements {@link WorkerSource#removeTile}.\n *\n * @param params - The tile parameters\n * @param callback - The callback\n */\n removeTile(params: TileParameters, callback: WorkerTileCallback) {\n const loaded = this.loaded,\n uid = params.uid;\n if (loaded && loaded[uid]) {\n delete loaded[uid];\n }\n callback();\n }\n}\n","import {RGBAImage} from '../util/image';\n\nimport {warnOnce} from '../util/util';\nimport {register} from '../util/web_worker_transfer';\n\n// DEMData is a data structure for decoding, backfilling, and storing elevation data for processing in the hillshade shaders\n// data can be populated either from a pngraw image tile or from serliazed data sent back from a worker. When data is initially\n// loaded from a image tile, we decode the pixel values using the appropriate decoding formula, but we store the\n// elevation data as an Int32 value. we add 65536 (2^16) to eliminate negative values and enable the use of\n// integer overflow when creating the texture used in the hillshadePrepare step.\n\n// DEMData also handles the backfilling of data from a tile's neighboring tiles. This is necessary because we use a pixel's 8\n// surrounding pixel values to compute the slope at that pixel, and we cannot accurately calculate the slope at pixels on a\n// tile's edge without backfilling from neighboring tiles.\n\nexport type DEMEncoding = 'mapbox' | 'terrarium' | 'custom'\n\nexport class DEMData {\n uid: string;\n data: Uint32Array;\n stride: number;\n dim: number;\n min: number;\n max: number;\n redFactor: number;\n greenFactor: number;\n blueFactor: number;\n baseShift: number;\n\n // RGBAImage data has uniform 1px padding on all sides: square tile edge size defines stride\n // and dim is calculated as stride - 2.\n constructor(uid: string, data: RGBAImage, encoding: DEMEncoding, redFactor = 1.0, greenFactor = 1.0, blueFactor = 1.0, baseShift = 0.0) {\n this.uid = uid;\n if (data.height !== data.width) throw new RangeError('DEM tiles must be square');\n if (encoding && !['mapbox', 'terrarium', 'custom'].includes(encoding)) {\n warnOnce(`\"${encoding}\" is not a valid encoding type. Valid types include \"mapbox\", \"terrarium\" and \"custom\".`);\n return;\n }\n this.stride = data.height;\n const dim = this.dim = data.height - 2;\n this.data = new Uint32Array(data.data.buffer);\n switch (encoding) {\n case 'terrarium':\n // unpacking formula for mapzen terrarium:\n // https://aws.amazon.com/public-datasets/terrain/\n this.redFactor = 256.0;\n this.greenFactor = 1.0;\n this.blueFactor = 1.0 / 256.0;\n this.baseShift = 32768.0;\n break;\n case 'custom':\n this.redFactor = redFactor;\n this.greenFactor = greenFactor;\n this.blueFactor = blueFactor;\n this.baseShift = baseShift;\n break;\n case 'mapbox':\n default:\n // unpacking formula for mapbox.terrain-rgb:\n // https://www.mapbox.com/help/access-elevation-data/#mapbox-terrain-rgb\n this.redFactor = 6553.6;\n this.greenFactor = 25.6;\n this.blueFactor = 0.1;\n this.baseShift = 10000.0;\n break;\n }\n\n // in order to avoid flashing seams between tiles, here we are initially populating a 1px border of pixels around the image\n // with the data of the nearest pixel from the image. this data is eventually replaced when the tile's neighboring\n // tiles are loaded and the accurate data can be backfilled using DEMData#backfillBorder\n for (let x = 0; x < dim; x++) {\n // left vertical border\n this.data[this._idx(-1, x)] = this.data[this._idx(0, x)];\n // right vertical border\n this.data[this._idx(dim, x)] = this.data[this._idx(dim - 1, x)];\n // left horizontal border\n this.data[this._idx(x, -1)] = this.data[this._idx(x, 0)];\n // right horizontal border\n this.data[this._idx(x, dim)] = this.data[this._idx(x, dim - 1)];\n }\n // corners\n this.data[this._idx(-1, -1)] = this.data[this._idx(0, 0)];\n this.data[this._idx(dim, -1)] = this.data[this._idx(dim - 1, 0)];\n this.data[this._idx(-1, dim)] = this.data[this._idx(0, dim - 1)];\n this.data[this._idx(dim, dim)] = this.data[this._idx(dim - 1, dim - 1)];\n\n // calculate min/max values\n this.min = Number.MAX_SAFE_INTEGER;\n this.max = Number.MIN_SAFE_INTEGER;\n for (let x = 0; x < dim; x++) {\n for (let y = 0; y < dim; y++) {\n const ele = this.get(x, y);\n if (ele > this.max) this.max = ele;\n if (ele < this.min) this.min = ele;\n }\n }\n }\n\n get(x: number, y: number) {\n const pixels = new Uint8Array(this.data.buffer);\n const index = this._idx(x, y) * 4;\n return this.unpack(pixels[index], pixels[index + 1], pixels[index + 2]);\n }\n\n getUnpackVector() {\n return [this.redFactor, this.greenFactor, this.blueFactor, this.baseShift];\n }\n\n _idx(x: number, y: number) {\n if (x < -1 || x >= this.dim + 1 || y < -1 || y >= this.dim + 1) throw new RangeError('out of range source coordinates for DEM data');\n return (y + 1) * this.stride + (x + 1);\n }\n\n unpack(r: number, g: number, b: number) {\n return (r * this.redFactor + g * this.greenFactor + b * this.blueFactor - this.baseShift);\n }\n\n getPixels() {\n return new RGBAImage({width: this.stride, height: this.stride}, new Uint8Array(this.data.buffer));\n }\n\n backfillBorder(borderTile: DEMData, dx: number, dy: number) {\n if (this.dim !== borderTile.dim) throw new Error('dem dimension mismatch');\n\n let xMin = dx * this.dim,\n xMax = dx * this.dim + this.dim,\n yMin = dy * this.dim,\n yMax = dy * this.dim + this.dim;\n\n switch (dx) {\n case -1:\n xMin = xMax - 1;\n break;\n case 1:\n xMax = xMin + 1;\n break;\n }\n\n switch (dy) {\n case -1:\n yMin = yMax - 1;\n break;\n case 1:\n yMax = yMin + 1;\n break;\n }\n\n const ox = -dx * this.dim;\n const oy = -dy * this.dim;\n for (let y = yMin; y < yMax; y++) {\n for (let x = xMin; x < xMax; x++) {\n this.data[this._idx(x, y)] = borderTile.data[this._idx(x + ox, y + oy)];\n }\n }\n }\n}\n\nregister('DEMData', DEMData);\n","import {DEMData} from '../data/dem_data';\nimport {RGBAImage} from '../util/image';\nimport type {Actor} from '../util/actor';\nimport type {\n WorkerDEMTileParameters,\n WorkerDEMTileCallback,\n TileParameters\n} from './worker_source';\nimport {getImageData, isImageBitmap} from '../util/util';\n\nexport class RasterDEMTileWorkerSource {\n actor: Actor;\n loaded: {[_: string]: DEMData};\n\n constructor() {\n this.loaded = {};\n }\n\n async loadTile(params: WorkerDEMTileParameters, callback: WorkerDEMTileCallback) {\n const {uid, encoding, rawImageData, redFactor, greenFactor, blueFactor, baseShift} = params;\n const width = rawImageData.width + 2;\n const height = rawImageData.height + 2;\n const imagePixels: RGBAImage = isImageBitmap(rawImageData) ?\n new RGBAImage({width, height}, await getImageData(rawImageData, -1, -1, width, height)) :\n rawImageData;\n const dem = new DEMData(uid, imagePixels, encoding, redFactor, greenFactor, blueFactor, baseShift);\n this.loaded = this.loaded || {};\n this.loaded[uid] = dem;\n callback(null, dem);\n }\n\n removeTile(params: TileParameters) {\n const loaded = this.loaded,\n uid = params.uid;\n if (loaded && loaded[uid]) {\n delete loaded[uid];\n }\n }\n}\n","\nmodule.exports = rewind;\n\nfunction rewind(gj, outer) {\n var type = gj && gj.type, i;\n\n if (type === 'FeatureCollection') {\n for (i = 0; i < gj.features.length; i++) rewind(gj.features[i], outer);\n\n } else if (type === 'GeometryCollection') {\n for (i = 0; i < gj.geometries.length; i++) rewind(gj.geometries[i], outer);\n\n } else if (type === 'Feature') {\n rewind(gj.geometry, outer);\n\n } else if (type === 'Polygon') {\n rewindRings(gj.coordinates, outer);\n\n } else if (type === 'MultiPolygon') {\n for (i = 0; i < gj.coordinates.length; i++) rewindRings(gj.coordinates[i], outer);\n }\n\n return gj;\n}\n\nfunction rewindRings(rings, outer) {\n if (rings.length === 0) return;\n\n rewindRing(rings[0], outer);\n for (var i = 1; i < rings.length; i++) {\n rewindRing(rings[i], !outer);\n }\n}\n\nfunction rewindRing(ring, dir) {\n var area = 0, err = 0;\n for (var i = 0, len = ring.length, j = len - 1; i < len; j = i++) {\n var k = (ring[i][0] - ring[j][0]) * (ring[j][1] + ring[i][1]);\n var m = area + k;\n err += Math.abs(area) >= Math.abs(k) ? area - m + k : k - m + area;\n area = m;\n }\n if (area + err >= 0 !== !!dir) ring.reverse();\n}\n","import Point from '@mapbox/point-geometry';\n\nimport mvt from '@mapbox/vector-tile';\nimport type {VectorTileFeature, VectorTileLayer, VectorTile} from '@mapbox/vector-tile';\nconst toGeoJSON = mvt.VectorTileFeature.prototype.toGeoJSON;\nimport {EXTENT} from '../data/extent';\nimport type {TileFeature, AnyProps} from 'supercluster';\nimport type {Feature as GeoJSONVTFeature} from 'geojson-vt';\n\nexport type Feature = TileFeature | GeoJSONVTFeature;\n\nclass FeatureWrapper implements VectorTileFeature {\n _feature: Feature;\n\n extent: number;\n type: Feature['type'];\n id: number;\n properties: {[_: string]: string | number | boolean};\n\n constructor(feature: Feature) {\n this._feature = feature;\n\n this.extent = EXTENT;\n this.type = feature.type;\n this.properties = feature.tags;\n\n // If the feature has a top-level `id` property, copy it over, but only\n // if it can be coerced to an integer, because this wrapper is used for\n // serializing geojson feature data into vector tile PBF data, and the\n // vector tile spec only supports integer values for feature ids --\n // allowing non-integer values here results in a non-compliant PBF\n // that causes an exception when it is parsed with vector-tile-js\n if ('id' in feature && !isNaN(feature.id as any)) {\n this.id = parseInt(feature.id, 10);\n }\n }\n\n loadGeometry() {\n if (this._feature.type === 1) {\n const geometry = [];\n for (const point of this._feature.geometry) {\n geometry.push([new Point(point[0], point[1])]);\n }\n return geometry;\n } else {\n const geometry = [];\n for (const ring of this._feature.geometry) {\n const newRing = [];\n for (const point of ring) {\n newRing.push(new Point(point[0], point[1]));\n }\n geometry.push(newRing);\n }\n return geometry;\n }\n }\n\n toGeoJSON(x: number, y: number, z: number) {\n return toGeoJSON.call(this, x, y, z);\n }\n}\n\nexport class GeoJSONWrapper implements VectorTile, VectorTileLayer {\n layers: {[_: string]: VectorTileLayer};\n name: string;\n extent: number;\n length: number;\n _features: Array;\n\n constructor(features: Array) {\n this.layers = {'_geojsonTileLayer': this};\n this.name = '_geojsonTileLayer';\n this.extent = EXTENT;\n this.length = features.length;\n this._features = features;\n }\n\n feature(i: number): VectorTileFeature {\n return new FeatureWrapper(this._features[i]);\n }\n}\n","'use strict'\n\nvar Point = require('@mapbox/point-geometry')\nvar VectorTileFeature = require('@mapbox/vector-tile').VectorTileFeature\n\nmodule.exports = GeoJSONWrapper\n\n// conform to vectortile api\nfunction GeoJSONWrapper (features, options) {\n this.options = options || {}\n this.features = features\n this.length = features.length\n}\n\nGeoJSONWrapper.prototype.feature = function (i) {\n return new FeatureWrapper(this.features[i], this.options.extent)\n}\n\nfunction FeatureWrapper (feature, extent) {\n this.id = typeof feature.id === 'number' ? feature.id : undefined\n this.type = feature.type\n this.rawGeometry = feature.type === 1 ? [feature.geometry] : feature.geometry\n this.properties = feature.tags\n this.extent = extent || 4096\n}\n\nFeatureWrapper.prototype.loadGeometry = function () {\n var rings = this.rawGeometry\n this.geometry = []\n\n for (var i = 0; i < rings.length; i++) {\n var ring = rings[i]\n var newRing = []\n for (var j = 0; j < ring.length; j++) {\n newRing.push(new Point(ring[j][0], ring[j][1]))\n }\n this.geometry.push(newRing)\n }\n return this.geometry\n}\n\nFeatureWrapper.prototype.bbox = function () {\n if (!this.geometry) this.loadGeometry()\n\n var rings = this.geometry\n var x1 = Infinity\n var x2 = -Infinity\n var y1 = Infinity\n var y2 = -Infinity\n\n for (var i = 0; i < rings.length; i++) {\n var ring = rings[i]\n\n for (var j = 0; j < ring.length; j++) {\n var coord = ring[j]\n\n x1 = Math.min(x1, coord.x)\n x2 = Math.max(x2, coord.x)\n y1 = Math.min(y1, coord.y)\n y2 = Math.max(y2, coord.y)\n }\n }\n\n return [x1, y1, x2, y2]\n}\n\nFeatureWrapper.prototype.toGeoJSON = VectorTileFeature.prototype.toGeoJSON\n","var Pbf = require('pbf')\nvar GeoJSONWrapper = require('./lib/geojson_wrapper')\n\nmodule.exports = fromVectorTileJs\nmodule.exports.fromVectorTileJs = fromVectorTileJs\nmodule.exports.fromGeojsonVt = fromGeojsonVt\nmodule.exports.GeoJSONWrapper = GeoJSONWrapper\n\n/**\n * Serialize a vector-tile-js-created tile to pbf\n *\n * @param {Object} tile\n * @return {Buffer} uncompressed, pbf-serialized tile data\n */\nfunction fromVectorTileJs (tile) {\n var out = new Pbf()\n writeTile(tile, out)\n return out.finish()\n}\n\n/**\n * Serialized a geojson-vt-created tile to pbf.\n *\n * @param {Object} layers - An object mapping layer names to geojson-vt-created vector tile objects\n * @param {Object} [options] - An object specifying the vector-tile specification version and extent that were used to create `layers`.\n * @param {Number} [options.version=1] - Version of vector-tile spec used\n * @param {Number} [options.extent=4096] - Extent of the vector tile\n * @return {Buffer} uncompressed, pbf-serialized tile data\n */\nfunction fromGeojsonVt (layers, options) {\n options = options || {}\n var l = {}\n for (var k in layers) {\n l[k] = new GeoJSONWrapper(layers[k].features, options)\n l[k].name = k\n l[k].version = options.version\n l[k].extent = options.extent\n }\n return fromVectorTileJs({ layers: l })\n}\n\nfunction writeTile (tile, pbf) {\n for (var key in tile.layers) {\n pbf.writeMessage(3, writeLayer, tile.layers[key])\n }\n}\n\nfunction writeLayer (layer, pbf) {\n pbf.writeVarintField(15, layer.version || 1)\n pbf.writeStringField(1, layer.name || '')\n pbf.writeVarintField(5, layer.extent || 4096)\n\n var i\n var context = {\n keys: [],\n values: [],\n keycache: {},\n valuecache: {}\n }\n\n for (i = 0; i < layer.length; i++) {\n context.feature = layer.feature(i)\n pbf.writeMessage(2, writeFeature, context)\n }\n\n var keys = context.keys\n for (i = 0; i < keys.length; i++) {\n pbf.writeStringField(3, keys[i])\n }\n\n var values = context.values\n for (i = 0; i < values.length; i++) {\n pbf.writeMessage(4, writeValue, values[i])\n }\n}\n\nfunction writeFeature (context, pbf) {\n var feature = context.feature\n\n if (feature.id !== undefined) {\n pbf.writeVarintField(1, feature.id)\n }\n\n pbf.writeMessage(2, writeProperties, context)\n pbf.writeVarintField(3, feature.type)\n pbf.writeMessage(4, writeGeometry, feature)\n}\n\nfunction writeProperties (context, pbf) {\n var feature = context.feature\n var keys = context.keys\n var values = context.values\n var keycache = context.keycache\n var valuecache = context.valuecache\n\n for (var key in feature.properties) {\n var value = feature.properties[key]\n\n var keyIndex = keycache[key]\n if (value === null) continue // don't encode null value properties\n\n if (typeof keyIndex === 'undefined') {\n keys.push(key)\n keyIndex = keys.length - 1\n keycache[key] = keyIndex\n }\n pbf.writeVarint(keyIndex)\n\n var type = typeof value\n if (type !== 'string' && type !== 'boolean' && type !== 'number') {\n value = JSON.stringify(value)\n }\n var valueKey = type + ':' + value\n var valueIndex = valuecache[valueKey]\n if (typeof valueIndex === 'undefined') {\n values.push(value)\n valueIndex = values.length - 1\n valuecache[valueKey] = valueIndex\n }\n pbf.writeVarint(valueIndex)\n }\n}\n\nfunction command (cmd, length) {\n return (length << 3) + (cmd & 0x7)\n}\n\nfunction zigzag (num) {\n return (num << 1) ^ (num >> 31)\n}\n\nfunction writeGeometry (feature, pbf) {\n var geometry = feature.loadGeometry()\n var type = feature.type\n var x = 0\n var y = 0\n var rings = geometry.length\n for (var r = 0; r < rings; r++) {\n var ring = geometry[r]\n var count = 1\n if (type === 1) {\n count = ring.length\n }\n pbf.writeVarint(command(1, count)) // moveto\n // do not write polygon closing path as lineto\n var lineCount = type === 3 ? ring.length - 1 : ring.length\n for (var i = 0; i < lineCount; i++) {\n if (i === 1 && type !== 1) {\n pbf.writeVarint(command(2, lineCount - 1)) // lineto\n }\n var dx = ring[i].x - x\n var dy = ring[i].y - y\n pbf.writeVarint(zigzag(dx))\n pbf.writeVarint(zigzag(dy))\n x += dx\n y += dy\n }\n if (type === 3) {\n pbf.writeVarint(command(7, 1)) // closepath\n }\n }\n}\n\nfunction writeValue (value, pbf) {\n var type = typeof value\n if (type === 'string') {\n pbf.writeStringField(1, value)\n } else if (type === 'boolean') {\n pbf.writeBooleanField(7, value)\n } else if (type === 'number') {\n if (value % 1 !== 0) {\n pbf.writeDoubleField(3, value)\n } else if (value < 0) {\n pbf.writeSVarintField(6, value)\n } else {\n pbf.writeVarintField(5, value)\n }\n }\n}\n","\nconst ARRAY_TYPES = [\n Int8Array, Uint8Array, Uint8ClampedArray, Int16Array, Uint16Array,\n Int32Array, Uint32Array, Float32Array, Float64Array\n];\n\n/** @typedef {Int8ArrayConstructor | Uint8ArrayConstructor | Uint8ClampedArrayConstructor | Int16ArrayConstructor | Uint16ArrayConstructor | Int32ArrayConstructor | Uint32ArrayConstructor | Float32ArrayConstructor | Float64ArrayConstructor} TypedArrayConstructor */\n\nconst VERSION = 1; // serialized format version\nconst HEADER_SIZE = 8;\n\nexport default class KDBush {\n\n /**\n * Creates an index from raw `ArrayBuffer` data.\n * @param {ArrayBuffer} data\n */\n static from(data) {\n if (!(data instanceof ArrayBuffer)) {\n throw new Error('Data must be an instance of ArrayBuffer.');\n }\n const [magic, versionAndType] = new Uint8Array(data, 0, 2);\n if (magic !== 0xdb) {\n throw new Error('Data does not appear to be in a KDBush format.');\n }\n const version = versionAndType >> 4;\n if (version !== VERSION) {\n throw new Error(`Got v${version} data when expected v${VERSION}.`);\n }\n const ArrayType = ARRAY_TYPES[versionAndType & 0x0f];\n if (!ArrayType) {\n throw new Error('Unrecognized array type.');\n }\n const [nodeSize] = new Uint16Array(data, 2, 1);\n const [numItems] = new Uint32Array(data, 4, 1);\n\n return new KDBush(numItems, nodeSize, ArrayType, data);\n }\n\n /**\n * Creates an index that will hold a given number of items.\n * @param {number} numItems\n * @param {number} [nodeSize=64] Size of the KD-tree node (64 by default).\n * @param {TypedArrayConstructor} [ArrayType=Float64Array] The array type used for coordinates storage (`Float64Array` by default).\n * @param {ArrayBuffer} [data] (For internal use only)\n */\n constructor(numItems, nodeSize = 64, ArrayType = Float64Array, data) {\n if (isNaN(numItems) || numItems < 0) throw new Error(`Unpexpected numItems value: ${numItems}.`);\n\n this.numItems = +numItems;\n this.nodeSize = Math.min(Math.max(+nodeSize, 2), 65535);\n this.ArrayType = ArrayType;\n this.IndexArrayType = numItems < 65536 ? Uint16Array : Uint32Array;\n\n const arrayTypeIndex = ARRAY_TYPES.indexOf(this.ArrayType);\n const coordsByteSize = numItems * 2 * this.ArrayType.BYTES_PER_ELEMENT;\n const idsByteSize = numItems * this.IndexArrayType.BYTES_PER_ELEMENT;\n const padCoords = (8 - idsByteSize % 8) % 8;\n\n if (arrayTypeIndex < 0) {\n throw new Error(`Unexpected typed array class: ${ArrayType}.`);\n }\n\n if (data && (data instanceof ArrayBuffer)) { // reconstruct an index from a buffer\n this.data = data;\n this.ids = new this.IndexArrayType(this.data, HEADER_SIZE, numItems);\n this.coords = new this.ArrayType(this.data, HEADER_SIZE + idsByteSize + padCoords, numItems * 2);\n this._pos = numItems * 2;\n this._finished = true;\n } else { // initialize a new index\n this.data = new ArrayBuffer(HEADER_SIZE + coordsByteSize + idsByteSize + padCoords);\n this.ids = new this.IndexArrayType(this.data, HEADER_SIZE, numItems);\n this.coords = new this.ArrayType(this.data, HEADER_SIZE + idsByteSize + padCoords, numItems * 2);\n this._pos = 0;\n this._finished = false;\n\n // set header\n new Uint8Array(this.data, 0, 2).set([0xdb, (VERSION << 4) + arrayTypeIndex]);\n new Uint16Array(this.data, 2, 1)[0] = nodeSize;\n new Uint32Array(this.data, 4, 1)[0] = numItems;\n }\n }\n\n /**\n * Add a point to the index.\n * @param {number} x\n * @param {number} y\n * @returns {number} An incremental index associated with the added item (starting from `0`).\n */\n add(x, y) {\n const index = this._pos >> 1;\n this.ids[index] = index;\n this.coords[this._pos++] = x;\n this.coords[this._pos++] = y;\n return index;\n }\n\n /**\n * Perform indexing of the added points.\n */\n finish() {\n const numAdded = this._pos >> 1;\n if (numAdded !== this.numItems) {\n throw new Error(`Added ${numAdded} items when expected ${this.numItems}.`);\n }\n // kd-sort both arrays for efficient search\n sort(this.ids, this.coords, this.nodeSize, 0, this.numItems - 1, 0);\n\n this._finished = true;\n return this;\n }\n\n /**\n * Search the index for items within a given bounding box.\n * @param {number} minX\n * @param {number} minY\n * @param {number} maxX\n * @param {number} maxY\n * @returns {number[]} An array of indices correponding to the found items.\n */\n range(minX, minY, maxX, maxY) {\n if (!this._finished) throw new Error('Data not yet indexed - call index.finish().');\n\n const {ids, coords, nodeSize} = this;\n const stack = [0, ids.length - 1, 0];\n const result = [];\n\n // recursively search for items in range in the kd-sorted arrays\n while (stack.length) {\n const axis = stack.pop() || 0;\n const right = stack.pop() || 0;\n const left = stack.pop() || 0;\n\n // if we reached \"tree node\", search linearly\n if (right - left <= nodeSize) {\n for (let i = left; i <= right; i++) {\n const x = coords[2 * i];\n const y = coords[2 * i + 1];\n if (x >= minX && x <= maxX && y >= minY && y <= maxY) result.push(ids[i]);\n }\n continue;\n }\n\n // otherwise find the middle index\n const m = (left + right) >> 1;\n\n // include the middle item if it's in range\n const x = coords[2 * m];\n const y = coords[2 * m + 1];\n if (x >= minX && x <= maxX && y >= minY && y <= maxY) result.push(ids[m]);\n\n // queue search in halves that intersect the query\n if (axis === 0 ? minX <= x : minY <= y) {\n stack.push(left);\n stack.push(m - 1);\n stack.push(1 - axis);\n }\n if (axis === 0 ? maxX >= x : maxY >= y) {\n stack.push(m + 1);\n stack.push(right);\n stack.push(1 - axis);\n }\n }\n\n return result;\n }\n\n /**\n * Search the index for items within a given radius.\n * @param {number} qx\n * @param {number} qy\n * @param {number} r Query radius.\n * @returns {number[]} An array of indices correponding to the found items.\n */\n within(qx, qy, r) {\n if (!this._finished) throw new Error('Data not yet indexed - call index.finish().');\n\n const {ids, coords, nodeSize} = this;\n const stack = [0, ids.length - 1, 0];\n const result = [];\n const r2 = r * r;\n\n // recursively search for items within radius in the kd-sorted arrays\n while (stack.length) {\n const axis = stack.pop() || 0;\n const right = stack.pop() || 0;\n const left = stack.pop() || 0;\n\n // if we reached \"tree node\", search linearly\n if (right - left <= nodeSize) {\n for (let i = left; i <= right; i++) {\n if (sqDist(coords[2 * i], coords[2 * i + 1], qx, qy) <= r2) result.push(ids[i]);\n }\n continue;\n }\n\n // otherwise find the middle index\n const m = (left + right) >> 1;\n\n // include the middle item if it's in range\n const x = coords[2 * m];\n const y = coords[2 * m + 1];\n if (sqDist(x, y, qx, qy) <= r2) result.push(ids[m]);\n\n // queue search in halves that intersect the query\n if (axis === 0 ? qx - r <= x : qy - r <= y) {\n stack.push(left);\n stack.push(m - 1);\n stack.push(1 - axis);\n }\n if (axis === 0 ? qx + r >= x : qy + r >= y) {\n stack.push(m + 1);\n stack.push(right);\n stack.push(1 - axis);\n }\n }\n\n return result;\n }\n}\n\n/**\n * @param {Uint16Array | Uint32Array} ids\n * @param {InstanceType} coords\n * @param {number} nodeSize\n * @param {number} left\n * @param {number} right\n * @param {number} axis\n */\nfunction sort(ids, coords, nodeSize, left, right, axis) {\n if (right - left <= nodeSize) return;\n\n const m = (left + right) >> 1; // middle index\n\n // sort ids and coords around the middle index so that the halves lie\n // either left/right or top/bottom correspondingly (taking turns)\n select(ids, coords, m, left, right, axis);\n\n // recursively kd-sort first half and second half on the opposite axis\n sort(ids, coords, nodeSize, left, m - 1, 1 - axis);\n sort(ids, coords, nodeSize, m + 1, right, 1 - axis);\n}\n\n/**\n * Custom Floyd-Rivest selection algorithm: sort ids and coords so that\n * [left..k-1] items are smaller than k-th item (on either x or y axis)\n * @param {Uint16Array | Uint32Array} ids\n * @param {InstanceType} coords\n * @param {number} k\n * @param {number} left\n * @param {number} right\n * @param {number} axis\n */\nfunction select(ids, coords, k, left, right, axis) {\n\n while (right > left) {\n if (right - left > 600) {\n const n = right - left + 1;\n const m = k - left + 1;\n const z = Math.log(n);\n const s = 0.5 * Math.exp(2 * z / 3);\n const sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);\n const newLeft = Math.max(left, Math.floor(k - m * s / n + sd));\n const newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));\n select(ids, coords, k, newLeft, newRight, axis);\n }\n\n const t = coords[2 * k + axis];\n let i = left;\n let j = right;\n\n swapItem(ids, coords, left, k);\n if (coords[2 * right + axis] > t) swapItem(ids, coords, left, right);\n\n while (i < j) {\n swapItem(ids, coords, i, j);\n i++;\n j--;\n while (coords[2 * i + axis] < t) i++;\n while (coords[2 * j + axis] > t) j--;\n }\n\n if (coords[2 * left + axis] === t) swapItem(ids, coords, left, j);\n else {\n j++;\n swapItem(ids, coords, j, right);\n }\n\n if (j <= k) left = j + 1;\n if (k <= j) right = j - 1;\n }\n}\n\n/**\n * @param {Uint16Array | Uint32Array} ids\n * @param {InstanceType} coords\n * @param {number} i\n * @param {number} j\n */\nfunction swapItem(ids, coords, i, j) {\n swap(ids, i, j);\n swap(coords, 2 * i, 2 * j);\n swap(coords, 2 * i + 1, 2 * j + 1);\n}\n\n/**\n * @param {InstanceType} arr\n * @param {number} i\n * @param {number} j\n */\nfunction swap(arr, i, j) {\n const tmp = arr[i];\n arr[i] = arr[j];\n arr[j] = tmp;\n}\n\n/**\n * @param {number} ax\n * @param {number} ay\n * @param {number} bx\n * @param {number} by\n */\nfunction sqDist(ax, ay, bx, by) {\n const dx = ax - bx;\n const dy = ay - by;\n return dx * dx + dy * dy;\n}\n","\nimport KDBush from 'kdbush';\n\nconst defaultOptions = {\n minZoom: 0, // min zoom to generate clusters on\n maxZoom: 16, // max zoom level to cluster the points on\n minPoints: 2, // minimum points to form a cluster\n radius: 40, // cluster radius in pixels\n extent: 512, // tile extent (radius is calculated relative to it)\n nodeSize: 64, // size of the KD-tree leaf node, affects performance\n log: false, // whether to log timing info\n\n // whether to generate numeric ids for input features (in vector tiles)\n generateId: false,\n\n // a reduce function for calculating custom cluster properties\n reduce: null, // (accumulated, props) => { accumulated.sum += props.sum; }\n\n // properties to use for individual points when running the reducer\n map: props => props // props => ({sum: props.my_value})\n};\n\nconst fround = Math.fround || (tmp => ((x) => { tmp[0] = +x; return tmp[0]; }))(new Float32Array(1));\n\nconst OFFSET_ZOOM = 2;\nconst OFFSET_ID = 3;\nconst OFFSET_PARENT = 4;\nconst OFFSET_NUM = 5;\nconst OFFSET_PROP = 6;\n\nexport default class Supercluster {\n constructor(options) {\n this.options = Object.assign(Object.create(defaultOptions), options);\n this.trees = new Array(this.options.maxZoom + 1);\n this.stride = this.options.reduce ? 7 : 6;\n this.clusterProps = [];\n }\n\n load(points) {\n const {log, minZoom, maxZoom} = this.options;\n\n if (log) console.time('total time');\n\n const timerId = `prepare ${ points.length } points`;\n if (log) console.time(timerId);\n\n this.points = points;\n\n // generate a cluster object for each point and index input points into a KD-tree\n const data = [];\n\n for (let i = 0; i < points.length; i++) {\n const p = points[i];\n if (!p.geometry) continue;\n\n const [lng, lat] = p.geometry.coordinates;\n const x = fround(lngX(lng));\n const y = fround(latY(lat));\n // store internal point/cluster data in flat numeric arrays for performance\n data.push(\n x, y, // projected point coordinates\n Infinity, // the last zoom the point was processed at\n i, // index of the source feature in the original input array\n -1, // parent cluster id\n 1 // number of points in a cluster\n );\n if (this.options.reduce) data.push(0); // noop\n }\n let tree = this.trees[maxZoom + 1] = this._createTree(data);\n\n if (log) console.timeEnd(timerId);\n\n // cluster points on max zoom, then cluster the results on previous zoom, etc.;\n // results in a cluster hierarchy across zoom levels\n for (let z = maxZoom; z >= minZoom; z--) {\n const now = +Date.now();\n\n // create a new set of clusters for the zoom and index them with a KD-tree\n tree = this.trees[z] = this._createTree(this._cluster(tree, z));\n\n if (log) console.log('z%d: %d clusters in %dms', z, tree.numItems, +Date.now() - now);\n }\n\n if (log) console.timeEnd('total time');\n\n return this;\n }\n\n getClusters(bbox, zoom) {\n let minLng = ((bbox[0] + 180) % 360 + 360) % 360 - 180;\n const minLat = Math.max(-90, Math.min(90, bbox[1]));\n let maxLng = bbox[2] === 180 ? 180 : ((bbox[2] + 180) % 360 + 360) % 360 - 180;\n const maxLat = Math.max(-90, Math.min(90, bbox[3]));\n\n if (bbox[2] - bbox[0] >= 360) {\n minLng = -180;\n maxLng = 180;\n } else if (minLng > maxLng) {\n const easternHem = this.getClusters([minLng, minLat, 180, maxLat], zoom);\n const westernHem = this.getClusters([-180, minLat, maxLng, maxLat], zoom);\n return easternHem.concat(westernHem);\n }\n\n const tree = this.trees[this._limitZoom(zoom)];\n const ids = tree.range(lngX(minLng), latY(maxLat), lngX(maxLng), latY(minLat));\n const data = tree.data;\n const clusters = [];\n for (const id of ids) {\n const k = this.stride * id;\n clusters.push(data[k + OFFSET_NUM] > 1 ? getClusterJSON(data, k, this.clusterProps) : this.points[data[k + OFFSET_ID]]);\n }\n return clusters;\n }\n\n getChildren(clusterId) {\n const originId = this._getOriginId(clusterId);\n const originZoom = this._getOriginZoom(clusterId);\n const errorMsg = 'No cluster with the specified id.';\n\n const tree = this.trees[originZoom];\n if (!tree) throw new Error(errorMsg);\n\n const data = tree.data;\n if (originId * this.stride >= data.length) throw new Error(errorMsg);\n\n const r = this.options.radius / (this.options.extent * Math.pow(2, originZoom - 1));\n const x = data[originId * this.stride];\n const y = data[originId * this.stride + 1];\n const ids = tree.within(x, y, r);\n const children = [];\n for (const id of ids) {\n const k = id * this.stride;\n if (data[k + OFFSET_PARENT] === clusterId) {\n children.push(data[k + OFFSET_NUM] > 1 ? getClusterJSON(data, k, this.clusterProps) : this.points[data[k + OFFSET_ID]]);\n }\n }\n\n if (children.length === 0) throw new Error(errorMsg);\n\n return children;\n }\n\n getLeaves(clusterId, limit, offset) {\n limit = limit || 10;\n offset = offset || 0;\n\n const leaves = [];\n this._appendLeaves(leaves, clusterId, limit, offset, 0);\n\n return leaves;\n }\n\n getTile(z, x, y) {\n const tree = this.trees[this._limitZoom(z)];\n const z2 = Math.pow(2, z);\n const {extent, radius} = this.options;\n const p = radius / extent;\n const top = (y - p) / z2;\n const bottom = (y + 1 + p) / z2;\n\n const tile = {\n features: []\n };\n\n this._addTileFeatures(\n tree.range((x - p) / z2, top, (x + 1 + p) / z2, bottom),\n tree.data, x, y, z2, tile);\n\n if (x === 0) {\n this._addTileFeatures(\n tree.range(1 - p / z2, top, 1, bottom),\n tree.data, z2, y, z2, tile);\n }\n if (x === z2 - 1) {\n this._addTileFeatures(\n tree.range(0, top, p / z2, bottom),\n tree.data, -1, y, z2, tile);\n }\n\n return tile.features.length ? tile : null;\n }\n\n getClusterExpansionZoom(clusterId) {\n let expansionZoom = this._getOriginZoom(clusterId) - 1;\n while (expansionZoom <= this.options.maxZoom) {\n const children = this.getChildren(clusterId);\n expansionZoom++;\n if (children.length !== 1) break;\n clusterId = children[0].properties.cluster_id;\n }\n return expansionZoom;\n }\n\n _appendLeaves(result, clusterId, limit, offset, skipped) {\n const children = this.getChildren(clusterId);\n\n for (const child of children) {\n const props = child.properties;\n\n if (props && props.cluster) {\n if (skipped + props.point_count <= offset) {\n // skip the whole cluster\n skipped += props.point_count;\n } else {\n // enter the cluster\n skipped = this._appendLeaves(result, props.cluster_id, limit, offset, skipped);\n // exit the cluster\n }\n } else if (skipped < offset) {\n // skip a single point\n skipped++;\n } else {\n // add a single point\n result.push(child);\n }\n if (result.length === limit) break;\n }\n\n return skipped;\n }\n\n _createTree(data) {\n const tree = new KDBush(data.length / this.stride | 0, this.options.nodeSize, Float32Array);\n for (let i = 0; i < data.length; i += this.stride) tree.add(data[i], data[i + 1]);\n tree.finish();\n tree.data = data;\n return tree;\n }\n\n _addTileFeatures(ids, data, x, y, z2, tile) {\n for (const i of ids) {\n const k = i * this.stride;\n const isCluster = data[k + OFFSET_NUM] > 1;\n\n let tags, px, py;\n if (isCluster) {\n tags = getClusterProperties(data, k, this.clusterProps);\n px = data[k];\n py = data[k + 1];\n } else {\n const p = this.points[data[k + OFFSET_ID]];\n tags = p.properties;\n const [lng, lat] = p.geometry.coordinates;\n px = lngX(lng);\n py = latY(lat);\n }\n\n const f = {\n type: 1,\n geometry: [[\n Math.round(this.options.extent * (px * z2 - x)),\n Math.round(this.options.extent * (py * z2 - y))\n ]],\n tags\n };\n\n // assign id\n let id;\n if (isCluster || this.options.generateId) {\n // optionally generate id for points\n id = data[k + OFFSET_ID];\n } else {\n // keep id if already assigned\n id = this.points[data[k + OFFSET_ID]].id;\n }\n\n if (id !== undefined) f.id = id;\n\n tile.features.push(f);\n }\n }\n\n _limitZoom(z) {\n return Math.max(this.options.minZoom, Math.min(Math.floor(+z), this.options.maxZoom + 1));\n }\n\n _cluster(tree, zoom) {\n const {radius, extent, reduce, minPoints} = this.options;\n const r = radius / (extent * Math.pow(2, zoom));\n const data = tree.data;\n const nextData = [];\n const stride = this.stride;\n\n // loop through each point\n for (let i = 0; i < data.length; i += stride) {\n // if we've already visited the point at this zoom level, skip it\n if (data[i + OFFSET_ZOOM] <= zoom) continue;\n data[i + OFFSET_ZOOM] = zoom;\n\n // find all nearby points\n const x = data[i];\n const y = data[i + 1];\n const neighborIds = tree.within(data[i], data[i + 1], r);\n\n const numPointsOrigin = data[i + OFFSET_NUM];\n let numPoints = numPointsOrigin;\n\n // count the number of points in a potential cluster\n for (const neighborId of neighborIds) {\n const k = neighborId * stride;\n // filter out neighbors that are already processed\n if (data[k + OFFSET_ZOOM] > zoom) numPoints += data[k + OFFSET_NUM];\n }\n\n // if there were neighbors to merge, and there are enough points to form a cluster\n if (numPoints > numPointsOrigin && numPoints >= minPoints) {\n let wx = x * numPointsOrigin;\n let wy = y * numPointsOrigin;\n\n let clusterProperties;\n let clusterPropIndex = -1;\n\n // encode both zoom and point index on which the cluster originated -- offset by total length of features\n const id = ((i / stride | 0) << 5) + (zoom + 1) + this.points.length;\n\n for (const neighborId of neighborIds) {\n const k = neighborId * stride;\n\n if (data[k + OFFSET_ZOOM] <= zoom) continue;\n data[k + OFFSET_ZOOM] = zoom; // save the zoom (so it doesn't get processed twice)\n\n const numPoints2 = data[k + OFFSET_NUM];\n wx += data[k] * numPoints2; // accumulate coordinates for calculating weighted center\n wy += data[k + 1] * numPoints2;\n\n data[k + OFFSET_PARENT] = id;\n\n if (reduce) {\n if (!clusterProperties) {\n clusterProperties = this._map(data, i, true);\n clusterPropIndex = this.clusterProps.length;\n this.clusterProps.push(clusterProperties);\n }\n reduce(clusterProperties, this._map(data, k));\n }\n }\n\n data[i + OFFSET_PARENT] = id;\n nextData.push(wx / numPoints, wy / numPoints, Infinity, id, -1, numPoints);\n if (reduce) nextData.push(clusterPropIndex);\n\n } else { // left points as unclustered\n for (let j = 0; j < stride; j++) nextData.push(data[i + j]);\n\n if (numPoints > 1) {\n for (const neighborId of neighborIds) {\n const k = neighborId * stride;\n if (data[k + OFFSET_ZOOM] <= zoom) continue;\n data[k + OFFSET_ZOOM] = zoom;\n for (let j = 0; j < stride; j++) nextData.push(data[k + j]);\n }\n }\n }\n }\n\n return nextData;\n }\n\n // get index of the point from which the cluster originated\n _getOriginId(clusterId) {\n return (clusterId - this.points.length) >> 5;\n }\n\n // get zoom of the point from which the cluster originated\n _getOriginZoom(clusterId) {\n return (clusterId - this.points.length) % 32;\n }\n\n _map(data, i, clone) {\n if (data[i + OFFSET_NUM] > 1) {\n const props = this.clusterProps[data[i + OFFSET_PROP]];\n return clone ? Object.assign({}, props) : props;\n }\n const original = this.points[data[i + OFFSET_ID]].properties;\n const result = this.options.map(original);\n return clone && result === original ? Object.assign({}, result) : result;\n }\n}\n\nfunction getClusterJSON(data, i, clusterProps) {\n return {\n type: 'Feature',\n id: data[i + OFFSET_ID],\n properties: getClusterProperties(data, i, clusterProps),\n geometry: {\n type: 'Point',\n coordinates: [xLng(data[i]), yLat(data[i + 1])]\n }\n };\n}\n\nfunction getClusterProperties(data, i, clusterProps) {\n const count = data[i + OFFSET_NUM];\n const abbrev =\n count >= 10000 ? `${Math.round(count / 1000) }k` :\n count >= 1000 ? `${Math.round(count / 100) / 10 }k` : count;\n const propIndex = data[i + OFFSET_PROP];\n const properties = propIndex === -1 ? {} : Object.assign({}, clusterProps[propIndex]);\n return Object.assign(properties, {\n cluster: true,\n cluster_id: data[i + OFFSET_ID],\n point_count: count,\n point_count_abbreviated: abbrev\n });\n}\n\n// longitude/latitude to spherical mercator in [0..1] range\nfunction lngX(lng) {\n return lng / 360 + 0.5;\n}\nfunction latY(lat) {\n const sin = Math.sin(lat * Math.PI / 180);\n const y = (0.5 - 0.25 * Math.log((1 + sin) / (1 - sin)) / Math.PI);\n return y < 0 ? 0 : y > 1 ? 1 : y;\n}\n\n// spherical mercator to longitude/latitude\nfunction xLng(x) {\n return (x - 0.5) * 360;\n}\nfunction yLat(y) {\n const y2 = (180 - y * 360) * Math.PI / 180;\n return 360 * Math.atan(Math.exp(y2)) / Math.PI - 90;\n}\n","\n// calculate simplification data using optimized Douglas-Peucker algorithm\n\nexport default function simplify(coords, first, last, sqTolerance) {\n var maxSqDist = sqTolerance;\n var mid = (last - first) >> 1;\n var minPosToMid = last - first;\n var index;\n\n var ax = coords[first];\n var ay = coords[first + 1];\n var bx = coords[last];\n var by = coords[last + 1];\n\n for (var i = first + 3; i < last; i += 3) {\n var d = getSqSegDist(coords[i], coords[i + 1], ax, ay, bx, by);\n\n if (d > maxSqDist) {\n index = i;\n maxSqDist = d;\n\n } else if (d === maxSqDist) {\n // a workaround to ensure we choose a pivot close to the middle of the list,\n // reducing recursion depth, for certain degenerate inputs\n // https://github.com/mapbox/geojson-vt/issues/104\n var posToMid = Math.abs(i - mid);\n if (posToMid < minPosToMid) {\n index = i;\n minPosToMid = posToMid;\n }\n }\n }\n\n if (maxSqDist > sqTolerance) {\n if (index - first > 3) simplify(coords, first, index, sqTolerance);\n coords[index + 2] = maxSqDist;\n if (last - index > 3) simplify(coords, index, last, sqTolerance);\n }\n}\n\n// square distance from a point to a segment\nfunction getSqSegDist(px, py, x, y, bx, by) {\n\n var dx = bx - x;\n var dy = by - y;\n\n if (dx !== 0 || dy !== 0) {\n\n var t = ((px - x) * dx + (py - y) * dy) / (dx * dx + dy * dy);\n\n if (t > 1) {\n x = bx;\n y = by;\n\n } else if (t > 0) {\n x += dx * t;\n y += dy * t;\n }\n }\n\n dx = px - x;\n dy = py - y;\n\n return dx * dx + dy * dy;\n}\n","\nexport default function createFeature(id, type, geom, tags) {\n var feature = {\n id: typeof id === 'undefined' ? null : id,\n type: type,\n geometry: geom,\n tags: tags,\n minX: Infinity,\n minY: Infinity,\n maxX: -Infinity,\n maxY: -Infinity\n };\n calcBBox(feature);\n return feature;\n}\n\nfunction calcBBox(feature) {\n var geom = feature.geometry;\n var type = feature.type;\n\n if (type === 'Point' || type === 'MultiPoint' || type === 'LineString') {\n calcLineBBox(feature, geom);\n\n } else if (type === 'Polygon' || type === 'MultiLineString') {\n for (var i = 0; i < geom.length; i++) {\n calcLineBBox(feature, geom[i]);\n }\n\n } else if (type === 'MultiPolygon') {\n for (i = 0; i < geom.length; i++) {\n for (var j = 0; j < geom[i].length; j++) {\n calcLineBBox(feature, geom[i][j]);\n }\n }\n }\n}\n\nfunction calcLineBBox(feature, geom) {\n for (var i = 0; i < geom.length; i += 3) {\n feature.minX = Math.min(feature.minX, geom[i]);\n feature.minY = Math.min(feature.minY, geom[i + 1]);\n feature.maxX = Math.max(feature.maxX, geom[i]);\n feature.maxY = Math.max(feature.maxY, geom[i + 1]);\n }\n}\n","\nimport simplify from './simplify';\nimport createFeature from './feature';\n\n// converts GeoJSON feature into an intermediate projected JSON vector format with simplification data\n\nexport default function convert(data, options) {\n var features = [];\n if (data.type === 'FeatureCollection') {\n for (var i = 0; i < data.features.length; i++) {\n convertFeature(features, data.features[i], options, i);\n }\n\n } else if (data.type === 'Feature') {\n convertFeature(features, data, options);\n\n } else {\n // single geometry or a geometry collection\n convertFeature(features, {geometry: data}, options);\n }\n\n return features;\n}\n\nfunction convertFeature(features, geojson, options, index) {\n if (!geojson.geometry) return;\n\n var coords = geojson.geometry.coordinates;\n var type = geojson.geometry.type;\n var tolerance = Math.pow(options.tolerance / ((1 << options.maxZoom) * options.extent), 2);\n var geometry = [];\n var id = geojson.id;\n if (options.promoteId) {\n id = geojson.properties[options.promoteId];\n } else if (options.generateId) {\n id = index || 0;\n }\n if (type === 'Point') {\n convertPoint(coords, geometry);\n\n } else if (type === 'MultiPoint') {\n for (var i = 0; i < coords.length; i++) {\n convertPoint(coords[i], geometry);\n }\n\n } else if (type === 'LineString') {\n convertLine(coords, geometry, tolerance, false);\n\n } else if (type === 'MultiLineString') {\n if (options.lineMetrics) {\n // explode into linestrings to be able to track metrics\n for (i = 0; i < coords.length; i++) {\n geometry = [];\n convertLine(coords[i], geometry, tolerance, false);\n features.push(createFeature(id, 'LineString', geometry, geojson.properties));\n }\n return;\n } else {\n convertLines(coords, geometry, tolerance, false);\n }\n\n } else if (type === 'Polygon') {\n convertLines(coords, geometry, tolerance, true);\n\n } else if (type === 'MultiPolygon') {\n for (i = 0; i < coords.length; i++) {\n var polygon = [];\n convertLines(coords[i], polygon, tolerance, true);\n geometry.push(polygon);\n }\n } else if (type === 'GeometryCollection') {\n for (i = 0; i < geojson.geometry.geometries.length; i++) {\n convertFeature(features, {\n id: id,\n geometry: geojson.geometry.geometries[i],\n properties: geojson.properties\n }, options, index);\n }\n return;\n } else {\n throw new Error('Input data is not a valid GeoJSON object.');\n }\n\n features.push(createFeature(id, type, geometry, geojson.properties));\n}\n\nfunction convertPoint(coords, out) {\n out.push(projectX(coords[0]));\n out.push(projectY(coords[1]));\n out.push(0);\n}\n\nfunction convertLine(ring, out, tolerance, isPolygon) {\n var x0, y0;\n var size = 0;\n\n for (var j = 0; j < ring.length; j++) {\n var x = projectX(ring[j][0]);\n var y = projectY(ring[j][1]);\n\n out.push(x);\n out.push(y);\n out.push(0);\n\n if (j > 0) {\n if (isPolygon) {\n size += (x0 * y - x * y0) / 2; // area\n } else {\n size += Math.sqrt(Math.pow(x - x0, 2) + Math.pow(y - y0, 2)); // length\n }\n }\n x0 = x;\n y0 = y;\n }\n\n var last = out.length - 3;\n out[2] = 1;\n simplify(out, 0, last, tolerance);\n out[last + 2] = 1;\n\n out.size = Math.abs(size);\n out.start = 0;\n out.end = out.size;\n}\n\nfunction convertLines(rings, out, tolerance, isPolygon) {\n for (var i = 0; i < rings.length; i++) {\n var geom = [];\n convertLine(rings[i], geom, tolerance, isPolygon);\n out.push(geom);\n }\n}\n\nfunction projectX(x) {\n return x / 360 + 0.5;\n}\n\nfunction projectY(y) {\n var sin = Math.sin(y * Math.PI / 180);\n var y2 = 0.5 - 0.25 * Math.log((1 + sin) / (1 - sin)) / Math.PI;\n return y2 < 0 ? 0 : y2 > 1 ? 1 : y2;\n}\n","\nimport createFeature from './feature';\n\n/* clip features between two axis-parallel lines:\n * | |\n * ___|___ | /\n * / | \\____|____/\n * | |\n */\n\nexport default function clip(features, scale, k1, k2, axis, minAll, maxAll, options) {\n\n k1 /= scale;\n k2 /= scale;\n\n if (minAll >= k1 && maxAll < k2) return features; // trivial accept\n else if (maxAll < k1 || minAll >= k2) return null; // trivial reject\n\n var clipped = [];\n\n for (var i = 0; i < features.length; i++) {\n\n var feature = features[i];\n var geometry = feature.geometry;\n var type = feature.type;\n\n var min = axis === 0 ? feature.minX : feature.minY;\n var max = axis === 0 ? feature.maxX : feature.maxY;\n\n if (min >= k1 && max < k2) { // trivial accept\n clipped.push(feature);\n continue;\n } else if (max < k1 || min >= k2) { // trivial reject\n continue;\n }\n\n var newGeometry = [];\n\n if (type === 'Point' || type === 'MultiPoint') {\n clipPoints(geometry, newGeometry, k1, k2, axis);\n\n } else if (type === 'LineString') {\n clipLine(geometry, newGeometry, k1, k2, axis, false, options.lineMetrics);\n\n } else if (type === 'MultiLineString') {\n clipLines(geometry, newGeometry, k1, k2, axis, false);\n\n } else if (type === 'Polygon') {\n clipLines(geometry, newGeometry, k1, k2, axis, true);\n\n } else if (type === 'MultiPolygon') {\n for (var j = 0; j < geometry.length; j++) {\n var polygon = [];\n clipLines(geometry[j], polygon, k1, k2, axis, true);\n if (polygon.length) {\n newGeometry.push(polygon);\n }\n }\n }\n\n if (newGeometry.length) {\n if (options.lineMetrics && type === 'LineString') {\n for (j = 0; j < newGeometry.length; j++) {\n clipped.push(createFeature(feature.id, type, newGeometry[j], feature.tags));\n }\n continue;\n }\n\n if (type === 'LineString' || type === 'MultiLineString') {\n if (newGeometry.length === 1) {\n type = 'LineString';\n newGeometry = newGeometry[0];\n } else {\n type = 'MultiLineString';\n }\n }\n if (type === 'Point' || type === 'MultiPoint') {\n type = newGeometry.length === 3 ? 'Point' : 'MultiPoint';\n }\n\n clipped.push(createFeature(feature.id, type, newGeometry, feature.tags));\n }\n }\n\n return clipped.length ? clipped : null;\n}\n\nfunction clipPoints(geom, newGeom, k1, k2, axis) {\n for (var i = 0; i < geom.length; i += 3) {\n var a = geom[i + axis];\n\n if (a >= k1 && a <= k2) {\n newGeom.push(geom[i]);\n newGeom.push(geom[i + 1]);\n newGeom.push(geom[i + 2]);\n }\n }\n}\n\nfunction clipLine(geom, newGeom, k1, k2, axis, isPolygon, trackMetrics) {\n\n var slice = newSlice(geom);\n var intersect = axis === 0 ? intersectX : intersectY;\n var len = geom.start;\n var segLen, t;\n\n for (var i = 0; i < geom.length - 3; i += 3) {\n var ax = geom[i];\n var ay = geom[i + 1];\n var az = geom[i + 2];\n var bx = geom[i + 3];\n var by = geom[i + 4];\n var a = axis === 0 ? ax : ay;\n var b = axis === 0 ? bx : by;\n var exited = false;\n\n if (trackMetrics) segLen = Math.sqrt(Math.pow(ax - bx, 2) + Math.pow(ay - by, 2));\n\n if (a < k1) {\n // ---|--> | (line enters the clip region from the left)\n if (b > k1) {\n t = intersect(slice, ax, ay, bx, by, k1);\n if (trackMetrics) slice.start = len + segLen * t;\n }\n } else if (a > k2) {\n // | <--|--- (line enters the clip region from the right)\n if (b < k2) {\n t = intersect(slice, ax, ay, bx, by, k2);\n if (trackMetrics) slice.start = len + segLen * t;\n }\n } else {\n addPoint(slice, ax, ay, az);\n }\n if (b < k1 && a >= k1) {\n // <--|--- | or <--|-----|--- (line exits the clip region on the left)\n t = intersect(slice, ax, ay, bx, by, k1);\n exited = true;\n }\n if (b > k2 && a <= k2) {\n // | ---|--> or ---|-----|--> (line exits the clip region on the right)\n t = intersect(slice, ax, ay, bx, by, k2);\n exited = true;\n }\n\n if (!isPolygon && exited) {\n if (trackMetrics) slice.end = len + segLen * t;\n newGeom.push(slice);\n slice = newSlice(geom);\n }\n\n if (trackMetrics) len += segLen;\n }\n\n // add the last point\n var last = geom.length - 3;\n ax = geom[last];\n ay = geom[last + 1];\n az = geom[last + 2];\n a = axis === 0 ? ax : ay;\n if (a >= k1 && a <= k2) addPoint(slice, ax, ay, az);\n\n // close the polygon if its endpoints are not the same after clipping\n last = slice.length - 3;\n if (isPolygon && last >= 3 && (slice[last] !== slice[0] || slice[last + 1] !== slice[1])) {\n addPoint(slice, slice[0], slice[1], slice[2]);\n }\n\n // add the final slice\n if (slice.length) {\n newGeom.push(slice);\n }\n}\n\nfunction newSlice(line) {\n var slice = [];\n slice.size = line.size;\n slice.start = line.start;\n slice.end = line.end;\n return slice;\n}\n\nfunction clipLines(geom, newGeom, k1, k2, axis, isPolygon) {\n for (var i = 0; i < geom.length; i++) {\n clipLine(geom[i], newGeom, k1, k2, axis, isPolygon, false);\n }\n}\n\nfunction addPoint(out, x, y, z) {\n out.push(x);\n out.push(y);\n out.push(z);\n}\n\nfunction intersectX(out, ax, ay, bx, by, x) {\n var t = (x - ax) / (bx - ax);\n out.push(x);\n out.push(ay + (by - ay) * t);\n out.push(1);\n return t;\n}\n\nfunction intersectY(out, ax, ay, bx, by, y) {\n var t = (y - ay) / (by - ay);\n out.push(ax + (bx - ax) * t);\n out.push(y);\n out.push(1);\n return t;\n}\n","\nimport clip from './clip';\nimport createFeature from './feature';\n\nexport default function wrap(features, options) {\n var buffer = options.buffer / options.extent;\n var merged = features;\n var left = clip(features, 1, -1 - buffer, buffer, 0, -1, 2, options); // left world copy\n var right = clip(features, 1, 1 - buffer, 2 + buffer, 0, -1, 2, options); // right world copy\n\n if (left || right) {\n merged = clip(features, 1, -buffer, 1 + buffer, 0, -1, 2, options) || []; // center world copy\n\n if (left) merged = shiftFeatureCoords(left, 1).concat(merged); // merge left into center\n if (right) merged = merged.concat(shiftFeatureCoords(right, -1)); // merge right into center\n }\n\n return merged;\n}\n\nfunction shiftFeatureCoords(features, offset) {\n var newFeatures = [];\n\n for (var i = 0; i < features.length; i++) {\n var feature = features[i],\n type = feature.type;\n\n var newGeometry;\n\n if (type === 'Point' || type === 'MultiPoint' || type === 'LineString') {\n newGeometry = shiftCoords(feature.geometry, offset);\n\n } else if (type === 'MultiLineString' || type === 'Polygon') {\n newGeometry = [];\n for (var j = 0; j < feature.geometry.length; j++) {\n newGeometry.push(shiftCoords(feature.geometry[j], offset));\n }\n } else if (type === 'MultiPolygon') {\n newGeometry = [];\n for (j = 0; j < feature.geometry.length; j++) {\n var newPolygon = [];\n for (var k = 0; k < feature.geometry[j].length; k++) {\n newPolygon.push(shiftCoords(feature.geometry[j][k], offset));\n }\n newGeometry.push(newPolygon);\n }\n }\n\n newFeatures.push(createFeature(feature.id, type, newGeometry, feature.tags));\n }\n\n return newFeatures;\n}\n\nfunction shiftCoords(points, offset) {\n var newPoints = [];\n newPoints.size = points.size;\n\n if (points.start !== undefined) {\n newPoints.start = points.start;\n newPoints.end = points.end;\n }\n\n for (var i = 0; i < points.length; i += 3) {\n newPoints.push(points[i] + offset, points[i + 1], points[i + 2]);\n }\n return newPoints;\n}\n","\n// Transforms the coordinates of each feature in the given tile from\n// mercator-projected space into (extent x extent) tile space.\nexport default function transformTile(tile, extent) {\n if (tile.transformed) return tile;\n\n var z2 = 1 << tile.z,\n tx = tile.x,\n ty = tile.y,\n i, j, k;\n\n for (i = 0; i < tile.features.length; i++) {\n var feature = tile.features[i],\n geom = feature.geometry,\n type = feature.type;\n\n feature.geometry = [];\n\n if (type === 1) {\n for (j = 0; j < geom.length; j += 2) {\n feature.geometry.push(transformPoint(geom[j], geom[j + 1], extent, z2, tx, ty));\n }\n } else {\n for (j = 0; j < geom.length; j++) {\n var ring = [];\n for (k = 0; k < geom[j].length; k += 2) {\n ring.push(transformPoint(geom[j][k], geom[j][k + 1], extent, z2, tx, ty));\n }\n feature.geometry.push(ring);\n }\n }\n }\n\n tile.transformed = true;\n\n return tile;\n}\n\nfunction transformPoint(x, y, extent, z2, tx, ty) {\n return [\n Math.round(extent * (x * z2 - tx)),\n Math.round(extent * (y * z2 - ty))];\n}\n","\nexport default function createTile(features, z, tx, ty, options) {\n var tolerance = z === options.maxZoom ? 0 : options.tolerance / ((1 << z) * options.extent);\n var tile = {\n features: [],\n numPoints: 0,\n numSimplified: 0,\n numFeatures: 0,\n source: null,\n x: tx,\n y: ty,\n z: z,\n transformed: false,\n minX: 2,\n minY: 1,\n maxX: -1,\n maxY: 0\n };\n for (var i = 0; i < features.length; i++) {\n tile.numFeatures++;\n addFeature(tile, features[i], tolerance, options);\n\n var minX = features[i].minX;\n var minY = features[i].minY;\n var maxX = features[i].maxX;\n var maxY = features[i].maxY;\n\n if (minX < tile.minX) tile.minX = minX;\n if (minY < tile.minY) tile.minY = minY;\n if (maxX > tile.maxX) tile.maxX = maxX;\n if (maxY > tile.maxY) tile.maxY = maxY;\n }\n return tile;\n}\n\nfunction addFeature(tile, feature, tolerance, options) {\n\n var geom = feature.geometry,\n type = feature.type,\n simplified = [];\n\n if (type === 'Point' || type === 'MultiPoint') {\n for (var i = 0; i < geom.length; i += 3) {\n simplified.push(geom[i]);\n simplified.push(geom[i + 1]);\n tile.numPoints++;\n tile.numSimplified++;\n }\n\n } else if (type === 'LineString') {\n addLine(simplified, geom, tile, tolerance, false, false);\n\n } else if (type === 'MultiLineString' || type === 'Polygon') {\n for (i = 0; i < geom.length; i++) {\n addLine(simplified, geom[i], tile, tolerance, type === 'Polygon', i === 0);\n }\n\n } else if (type === 'MultiPolygon') {\n\n for (var k = 0; k < geom.length; k++) {\n var polygon = geom[k];\n for (i = 0; i < polygon.length; i++) {\n addLine(simplified, polygon[i], tile, tolerance, true, i === 0);\n }\n }\n }\n\n if (simplified.length) {\n var tags = feature.tags || null;\n if (type === 'LineString' && options.lineMetrics) {\n tags = {};\n for (var key in feature.tags) tags[key] = feature.tags[key];\n tags['mapbox_clip_start'] = geom.start / geom.size;\n tags['mapbox_clip_end'] = geom.end / geom.size;\n }\n var tileFeature = {\n geometry: simplified,\n type: type === 'Polygon' || type === 'MultiPolygon' ? 3 :\n type === 'LineString' || type === 'MultiLineString' ? 2 : 1,\n tags: tags\n };\n if (feature.id !== null) {\n tileFeature.id = feature.id;\n }\n tile.features.push(tileFeature);\n }\n}\n\nfunction addLine(result, geom, tile, tolerance, isPolygon, isOuter) {\n var sqTolerance = tolerance * tolerance;\n\n if (tolerance > 0 && (geom.size < (isPolygon ? sqTolerance : tolerance))) {\n tile.numPoints += geom.length / 3;\n return;\n }\n\n var ring = [];\n\n for (var i = 0; i < geom.length; i += 3) {\n if (tolerance === 0 || geom[i + 2] > sqTolerance) {\n tile.numSimplified++;\n ring.push(geom[i]);\n ring.push(geom[i + 1]);\n }\n tile.numPoints++;\n }\n\n if (isPolygon) rewind(ring, isOuter);\n\n result.push(ring);\n}\n\nfunction rewind(ring, clockwise) {\n var area = 0;\n for (var i = 0, len = ring.length, j = len - 2; i < len; j = i, i += 2) {\n area += (ring[i] - ring[j]) * (ring[i + 1] + ring[j + 1]);\n }\n if (area > 0 === clockwise) {\n for (i = 0, len = ring.length; i < len / 2; i += 2) {\n var x = ring[i];\n var y = ring[i + 1];\n ring[i] = ring[len - 2 - i];\n ring[i + 1] = ring[len - 1 - i];\n ring[len - 2 - i] = x;\n ring[len - 1 - i] = y;\n }\n }\n}\n","\nimport convert from './convert'; // GeoJSON conversion and preprocessing\nimport clip from './clip'; // stripe clipping algorithm\nimport wrap from './wrap'; // date line processing\nimport transform from './transform'; // coordinate transformation\nimport createTile from './tile'; // final simplified tile generation\n\nexport default function geojsonvt(data, options) {\n return new GeoJSONVT(data, options);\n}\n\nfunction GeoJSONVT(data, options) {\n options = this.options = extend(Object.create(this.options), options);\n\n var debug = options.debug;\n\n if (debug) console.time('preprocess data');\n\n if (options.maxZoom < 0 || options.maxZoom > 24) throw new Error('maxZoom should be in the 0-24 range');\n if (options.promoteId && options.generateId) throw new Error('promoteId and generateId cannot be used together.');\n\n var features = convert(data, options);\n\n this.tiles = {};\n this.tileCoords = [];\n\n if (debug) {\n console.timeEnd('preprocess data');\n console.log('index: maxZoom: %d, maxPoints: %d', options.indexMaxZoom, options.indexMaxPoints);\n console.time('generate tiles');\n this.stats = {};\n this.total = 0;\n }\n\n features = wrap(features, options);\n\n // start slicing from the top tile down\n if (features.length) this.splitTile(features, 0, 0, 0);\n\n if (debug) {\n if (features.length) console.log('features: %d, points: %d', this.tiles[0].numFeatures, this.tiles[0].numPoints);\n console.timeEnd('generate tiles');\n console.log('tiles generated:', this.total, JSON.stringify(this.stats));\n }\n}\n\nGeoJSONVT.prototype.options = {\n maxZoom: 14, // max zoom to preserve detail on\n indexMaxZoom: 5, // max zoom in the tile index\n indexMaxPoints: 100000, // max number of points per tile in the tile index\n tolerance: 3, // simplification tolerance (higher means simpler)\n extent: 4096, // tile extent\n buffer: 64, // tile buffer on each side\n lineMetrics: false, // whether to calculate line metrics\n promoteId: null, // name of a feature property to be promoted to feature.id\n generateId: false, // whether to generate feature ids. Cannot be used with promoteId\n debug: 0 // logging level (0, 1 or 2)\n};\n\nGeoJSONVT.prototype.splitTile = function (features, z, x, y, cz, cx, cy) {\n\n var stack = [features, z, x, y],\n options = this.options,\n debug = options.debug;\n\n // avoid recursion by using a processing queue\n while (stack.length) {\n y = stack.pop();\n x = stack.pop();\n z = stack.pop();\n features = stack.pop();\n\n var z2 = 1 << z,\n id = toID(z, x, y),\n tile = this.tiles[id];\n\n if (!tile) {\n if (debug > 1) console.time('creation');\n\n tile = this.tiles[id] = createTile(features, z, x, y, options);\n this.tileCoords.push({z: z, x: x, y: y});\n\n if (debug) {\n if (debug > 1) {\n console.log('tile z%d-%d-%d (features: %d, points: %d, simplified: %d)',\n z, x, y, tile.numFeatures, tile.numPoints, tile.numSimplified);\n console.timeEnd('creation');\n }\n var key = 'z' + z;\n this.stats[key] = (this.stats[key] || 0) + 1;\n this.total++;\n }\n }\n\n // save reference to original geometry in tile so that we can drill down later if we stop now\n tile.source = features;\n\n // if it's the first-pass tiling\n if (!cz) {\n // stop tiling if we reached max zoom, or if the tile is too simple\n if (z === options.indexMaxZoom || tile.numPoints <= options.indexMaxPoints) continue;\n\n // if a drilldown to a specific tile\n } else {\n // stop tiling if we reached base zoom or our target tile zoom\n if (z === options.maxZoom || z === cz) continue;\n\n // stop tiling if it's not an ancestor of the target tile\n var m = 1 << (cz - z);\n if (x !== Math.floor(cx / m) || y !== Math.floor(cy / m)) continue;\n }\n\n // if we slice further down, no need to keep source geometry\n tile.source = null;\n\n if (features.length === 0) continue;\n\n if (debug > 1) console.time('clipping');\n\n // values we'll use for clipping\n var k1 = 0.5 * options.buffer / options.extent,\n k2 = 0.5 - k1,\n k3 = 0.5 + k1,\n k4 = 1 + k1,\n tl, bl, tr, br, left, right;\n\n tl = bl = tr = br = null;\n\n left = clip(features, z2, x - k1, x + k3, 0, tile.minX, tile.maxX, options);\n right = clip(features, z2, x + k2, x + k4, 0, tile.minX, tile.maxX, options);\n features = null;\n\n if (left) {\n tl = clip(left, z2, y - k1, y + k3, 1, tile.minY, tile.maxY, options);\n bl = clip(left, z2, y + k2, y + k4, 1, tile.minY, tile.maxY, options);\n left = null;\n }\n\n if (right) {\n tr = clip(right, z2, y - k1, y + k3, 1, tile.minY, tile.maxY, options);\n br = clip(right, z2, y + k2, y + k4, 1, tile.minY, tile.maxY, options);\n right = null;\n }\n\n if (debug > 1) console.timeEnd('clipping');\n\n stack.push(tl || [], z + 1, x * 2, y * 2);\n stack.push(bl || [], z + 1, x * 2, y * 2 + 1);\n stack.push(tr || [], z + 1, x * 2 + 1, y * 2);\n stack.push(br || [], z + 1, x * 2 + 1, y * 2 + 1);\n }\n};\n\nGeoJSONVT.prototype.getTile = function (z, x, y) {\n var options = this.options,\n extent = options.extent,\n debug = options.debug;\n\n if (z < 0 || z > 24) return null;\n\n var z2 = 1 << z;\n x = ((x % z2) + z2) % z2; // wrap tile x coordinate\n\n var id = toID(z, x, y);\n if (this.tiles[id]) return transform(this.tiles[id], extent);\n\n if (debug > 1) console.log('drilling down to z%d-%d-%d', z, x, y);\n\n var z0 = z,\n x0 = x,\n y0 = y,\n parent;\n\n while (!parent && z0 > 0) {\n z0--;\n x0 = Math.floor(x0 / 2);\n y0 = Math.floor(y0 / 2);\n parent = this.tiles[toID(z0, x0, y0)];\n }\n\n if (!parent || !parent.source) return null;\n\n // if we found a parent tile containing the original geometry, we can drill down from it\n if (debug > 1) console.log('found parent tile z%d-%d-%d', z0, x0, y0);\n\n if (debug > 1) console.time('drilling down');\n this.splitTile(parent.source, z0, x0, y0, z, x, y);\n if (debug > 1) console.timeEnd('drilling down');\n\n return this.tiles[id] ? transform(this.tiles[id], extent) : null;\n};\n\nfunction toID(z, x, y) {\n return (((1 << z) * y + x) * 32) + z;\n}\n\nfunction extend(dest, src) {\n for (var i in src) dest[i] = src[i];\n return dest;\n}\n","/**\n * A way to indentify a feature, either by string or by number\n */\nexport type GeoJSONFeatureId = number | string;\n\n/**\n * The geojson source diff object\n */\nexport type GeoJSONSourceDiff = {\n /**\n * When set to `true` it will remove all features\n */\n removeAll?: boolean;\n /**\n * An array of features IDs to remove\n */\n remove?: Array;\n /**\n * An array of features to add\n */\n add?: Array;\n /**\n * An array of update objects\n */\n update?: Array;\n}\n\n/**\n * A geojson feature diff object\n */\nexport type GeoJSONFeatureDiff = {\n /**\n * The feature ID\n */\n id: GeoJSONFeatureId;\n /**\n * If it's a new geometry, place it here\n */\n newGeometry?: GeoJSON.Geometry;\n /**\n * Setting to `true` will remove all preperties\n */\n removeAllProperties?: boolean;\n /**\n * The properties keys to remove\n */\n removeProperties?: Array;\n /**\n * The properties to add or update along side their values\n */\n addOrUpdateProperties?: Array<{key: string; value: any}>;\n}\n\nexport type UpdateableGeoJSON = GeoJSON.Feature | GeoJSON.FeatureCollection | undefined;\n\nfunction getFeatureId(feature: GeoJSON.Feature, promoteId?: string): GeoJSONFeatureId | undefined {\n return promoteId ? feature.properties[promoteId] : feature.id;\n}\n\nexport function isUpdateableGeoJSON(data: GeoJSON.GeoJSON | undefined, promoteId?: string): data is UpdateableGeoJSON {\n // null can be updated\n if (data == null) {\n return true;\n }\n\n // a single feature with an id can be updated, need to explicitly check against null because 0 is a valid feature id that is falsy\n if (data.type === 'Feature') {\n return getFeatureId(data, promoteId) != null;\n }\n\n // a feature collection can be updated if every feature has an id, and the ids are all unique\n // this prevents us from silently dropping features if ids get reused\n if (data.type === 'FeatureCollection') {\n const seenIds = new Set();\n for (const feature of data.features) {\n const id = getFeatureId(feature, promoteId);\n if (id == null) {\n return false;\n }\n\n if (seenIds.has(id)) {\n return false;\n }\n\n seenIds.add(id);\n }\n\n return true;\n }\n\n return false;\n}\n\nexport function toUpdateable(data: UpdateableGeoJSON, promoteId?: string) {\n const result = new Map();\n if (data == null) {\n // empty result\n } else if (data.type === 'Feature') {\n result.set(getFeatureId(data, promoteId)!, data);\n } else {\n for (const feature of data.features) {\n result.set(getFeatureId(feature, promoteId)!, feature);\n }\n }\n\n return result;\n}\n\n// mutates updateable\nexport function applySourceDiff(updateable: Map, diff: GeoJSONSourceDiff, promoteId?: string): void {\n if (diff.removeAll) {\n updateable.clear();\n }\n\n if (diff.remove) {\n for (const id of diff.remove) {\n updateable.delete(id);\n }\n }\n\n if (diff.add) {\n for (const feature of diff.add) {\n const id = getFeatureId(feature, promoteId);\n\n if (id != null) {\n updateable.set(id, feature);\n }\n }\n }\n\n if (diff.update) {\n for (const update of diff.update) {\n let feature = updateable.get(update.id);\n\n if (feature == null) {\n continue;\n }\n\n // be careful to clone the feature and/or properties objects to avoid mutating our input\n const cloneFeature = update.newGeometry || update.removeAllProperties;\n // note: removeAllProperties gives us a new properties object, so we can skip the clone step\n const cloneProperties = !update.removeAllProperties && (update.removeProperties?.length > 0 || update.addOrUpdateProperties?.length > 0);\n if (cloneFeature || cloneProperties) {\n feature = {...feature};\n updateable.set(update.id, feature);\n if (cloneProperties) {\n feature.properties = {...feature.properties};\n }\n }\n\n if (update.newGeometry) {\n feature.geometry = update.newGeometry;\n }\n\n if (update.removeAllProperties) {\n feature.properties = {};\n } else if (update.removeProperties?.length > 0) {\n for (const prop of update.removeProperties) {\n if (Object.prototype.hasOwnProperty.call(feature.properties, prop)) {\n delete feature.properties[prop];\n }\n }\n }\n\n if (update.addOrUpdateProperties?.length > 0) {\n for (const {key, value} of update.addOrUpdateProperties) {\n feature.properties[key] = value;\n }\n }\n }\n }\n}\n","import {getJSON} from '../util/ajax';\n\nimport {RequestPerformance} from '../util/performance';\nimport rewind from '@mapbox/geojson-rewind';\nimport {GeoJSONWrapper} from './geojson_wrapper';\nimport vtpbf from 'vt-pbf';\nimport Supercluster, {type Options as SuperclusterOptions, type ClusterProperties} from 'supercluster';\nimport geojsonvt, {type Options as GeoJSONVTOptions} from 'geojson-vt';\nimport {VectorTileWorkerSource} from './vector_tile_worker_source';\nimport {createExpression} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {\n WorkerTileParameters,\n WorkerTileCallback,\n} from '../source/worker_source';\n\nimport type {Actor} from '../util/actor';\nimport type {StyleLayerIndex} from '../style/style_layer_index';\n\nimport type {LoadVectorDataCallback} from './vector_tile_worker_source';\nimport type {RequestParameters, ResponseCallback} from '../util/ajax';\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\nimport {isUpdateableGeoJSON, type GeoJSONSourceDiff, applySourceDiff, toUpdateable, GeoJSONFeatureId} from './geojson_source_diff';\n\nexport type LoadGeoJSONParameters = {\n request?: RequestParameters;\n /**\n * Literal GeoJSON data. Must be provided if `request.url` is not.\n */\n data?: string;\n dataDiff?: GeoJSONSourceDiff;\n source: string;\n cluster: boolean;\n superclusterOptions?: SuperclusterOptions;\n geojsonVtOptions?: GeoJSONVTOptions;\n clusterProperties?: ClusterProperties;\n filter?: Array;\n promoteId?: string;\n};\n\nexport type LoadGeoJSON = (params: LoadGeoJSONParameters, callback: ResponseCallback) => Cancelable;\n\ntype GeoJSONIndex = ReturnType | Supercluster;\n\n/**\n * The {@link WorkerSource} implementation that supports {@link GeoJSONSource}.\n * This class is designed to be easily reused to support custom source types\n * for data formats that can be parsed/converted into an in-memory GeoJSON\n * representation. To do so, create it with\n * `new GeoJSONWorkerSource(actor, layerIndex, customLoadGeoJSONFunction)`.\n * For a full example, see [mapbox-gl-topojson](https://github.com/developmentseed/mapbox-gl-topojson).\n */\nexport class GeoJSONWorkerSource extends VectorTileWorkerSource {\n _pendingCallback: Callback<{\n resourceTiming?: {[_: string]: Array};\n abandoned?: boolean;\n }>;\n _pendingRequest: Cancelable;\n _geoJSONIndex: GeoJSONIndex;\n _dataUpdateable = new Map();\n\n /**\n * @param loadGeoJSON - Optional method for custom loading/parsing of\n * GeoJSON based on parameters passed from the main-thread Source.\n * See {@link GeoJSONWorkerSource#loadGeoJSON}.\n */\n constructor(actor: Actor, layerIndex: StyleLayerIndex, availableImages: Array, loadGeoJSON?: LoadGeoJSON | null) {\n super(actor, layerIndex, availableImages);\n this.loadVectorData = this.loadGeoJSONTile;\n if (loadGeoJSON) {\n this.loadGeoJSON = loadGeoJSON;\n }\n }\n\n loadGeoJSONTile(params: WorkerTileParameters, callback: LoadVectorDataCallback): (() => void) | void {\n const canonical = params.tileID.canonical;\n\n if (!this._geoJSONIndex) {\n return callback(null, null); // we couldn't load the file\n }\n\n const geoJSONTile = this._geoJSONIndex.getTile(canonical.z, canonical.x, canonical.y);\n if (!geoJSONTile) {\n return callback(null, null); // nothing in the given tile\n }\n\n const geojsonWrapper = new GeoJSONWrapper(geoJSONTile.features);\n // Encode the geojson-vt tile into binary vector tile form. This\n // is a convenience that allows `FeatureIndex` to operate the same way\n // across `VectorTileSource` and `GeoJSONSource` data.\n let pbf = vtpbf(geojsonWrapper);\n if (pbf.byteOffset !== 0 || pbf.byteLength !== pbf.buffer.byteLength) {\n // Compatibility with node Buffer (https://github.com/mapbox/pbf/issues/35)\n pbf = new Uint8Array(pbf);\n }\n\n callback(null, {\n vectorTile: geojsonWrapper,\n rawData: pbf.buffer\n });\n }\n\n /**\n * Fetches (if appropriate), parses, and index geojson data into tiles. This\n * preparatory method must be called before {@link GeoJSONWorkerSource#loadTile}\n * can correctly serve up tiles.\n *\n * Defers to {@link GeoJSONWorkerSource#loadGeoJSON} for the fetching/parsing,\n * expecting `callback(error, data)` to be called with either an error or a\n * parsed GeoJSON object.\n *\n * When a `loadData` request comes in while a previous one is being processed,\n * the previous one is aborted.\n *\n * @param params - the parameters\n * @param callback - the callback for completion or error\n */\n loadData(params: LoadGeoJSONParameters, callback: Callback<{\n resourceTiming?: {[_: string]: Array};\n abandoned?: boolean;\n }>) {\n this._pendingRequest?.cancel();\n if (this._pendingCallback) {\n // Tell the foreground the previous call has been abandoned\n this._pendingCallback(null, {abandoned: true});\n }\n\n const perf = (params && params.request && params.request.collectResourceTiming) ?\n new RequestPerformance(params.request) : false;\n\n this._pendingCallback = callback;\n this._pendingRequest = this.loadGeoJSON(params, (err?: Error | null, data?: any | null) => {\n delete this._pendingCallback;\n delete this._pendingRequest;\n\n if (err || !data) {\n return callback(err);\n } else if (typeof data !== 'object') {\n return callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`));\n } else {\n rewind(data, true);\n\n try {\n if (params.filter) {\n const compiled = createExpression(params.filter, {type: 'boolean', 'property-type': 'data-driven', overridable: false, transition: false} as any);\n if (compiled.result === 'error')\n throw new Error(compiled.value.map(err => `${err.key}: ${err.message}`).join(', '));\n\n const features = data.features.filter(feature => compiled.value.evaluate({zoom: 0}, feature));\n data = {type: 'FeatureCollection', features};\n }\n\n this._geoJSONIndex = params.cluster ?\n new Supercluster(getSuperclusterOptions(params)).load(data.features) :\n geojsonvt(data, params.geojsonVtOptions);\n } catch (err) {\n return callback(err);\n }\n\n this.loaded = {};\n\n const result = {} as { resourceTiming: any };\n if (perf) {\n const resourceTimingData = perf.finish();\n // it's necessary to eval the result of getEntriesByName() here via parse/stringify\n // late evaluation in the main thread causes TypeError: illegal invocation\n if (resourceTimingData) {\n result.resourceTiming = {};\n result.resourceTiming[params.source] = JSON.parse(JSON.stringify(resourceTimingData));\n }\n }\n callback(null, result);\n }\n });\n }\n\n /**\n * Implements {@link WorkerSource#reloadTile}.\n *\n * If the tile is loaded, uses the implementation in VectorTileWorkerSource.\n * Otherwise, such as after a setData() call, we load the tile fresh.\n *\n * @param params - the parameters\n * @param callback - the callback for completion or error\n */\n reloadTile(params: WorkerTileParameters, callback: WorkerTileCallback) {\n const loaded = this.loaded,\n uid = params.uid;\n\n if (loaded && loaded[uid]) {\n return super.reloadTile(params, callback);\n } else {\n return this.loadTile(params, callback);\n }\n }\n\n /**\n * Fetch and parse GeoJSON according to the given params. Calls `callback`\n * with `(err, data)`, where `data` is a parsed GeoJSON object.\n *\n * GeoJSON is loaded and parsed from `params.url` if it exists, or else\n * expected as a literal (string or object) `params.data`.\n *\n * @param params - the parameters\n * @param callback - the callback for completion or error\n * @returns A Cancelable object.\n */\n loadGeoJSON = (params: LoadGeoJSONParameters, callback: ResponseCallback): Cancelable => {\n const {promoteId} = params;\n // Because of same origin issues, urls must either include an explicit\n // origin or absolute path.\n // ie: /foo/bar.json or http://example.com/bar.json\n // but not ../foo/bar.json\n if (params.request) {\n return getJSON(params.request, (\n error?: Error,\n data?: any,\n cacheControl?: string,\n expires?: string\n ) => {\n this._dataUpdateable = isUpdateableGeoJSON(data, promoteId) ? toUpdateable(data, promoteId) : undefined;\n callback(error, data, cacheControl, expires);\n });\n } else if (typeof params.data === 'string') {\n try {\n const parsed = JSON.parse(params.data);\n this._dataUpdateable = isUpdateableGeoJSON(parsed, promoteId) ? toUpdateable(parsed, promoteId) : undefined;\n callback(null, parsed);\n } catch (e) {\n callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`));\n }\n } else if (params.dataDiff) {\n if (this._dataUpdateable) {\n applySourceDiff(this._dataUpdateable, params.dataDiff, promoteId);\n callback(null, {type: 'FeatureCollection', features: Array.from(this._dataUpdateable.values())});\n } else {\n callback(new Error(`Cannot update existing geojson data in ${params.source}`));\n }\n } else {\n callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`));\n }\n\n return {cancel: () => {}};\n };\n\n removeSource(params: {\n source: string;\n }, callback: WorkerTileCallback) {\n if (this._pendingCallback) {\n // Don't leak callbacks\n this._pendingCallback(null, {abandoned: true});\n }\n callback();\n }\n\n getClusterExpansionZoom(params: {\n clusterId: number;\n }, callback: Callback) {\n try {\n callback(null, (this._geoJSONIndex as Supercluster).getClusterExpansionZoom(params.clusterId));\n } catch (e) {\n callback(e);\n }\n }\n\n getClusterChildren(params: {\n clusterId: number;\n }, callback: Callback>) {\n try {\n callback(null, (this._geoJSONIndex as Supercluster).getChildren(params.clusterId));\n } catch (e) {\n callback(e);\n }\n }\n\n getClusterLeaves(params: {\n clusterId: number;\n limit: number;\n offset: number;\n }, callback: Callback>) {\n try {\n callback(null, (this._geoJSONIndex as Supercluster).getLeaves(params.clusterId, params.limit, params.offset));\n } catch (e) {\n callback(e);\n }\n }\n}\n\nfunction getSuperclusterOptions({superclusterOptions, clusterProperties}: LoadGeoJSONParameters) {\n if (!clusterProperties || !superclusterOptions) return superclusterOptions;\n\n const mapExpressions = {};\n const reduceExpressions = {};\n const globals = {accumulated: null, zoom: 0};\n const feature = {properties: null};\n const propertyNames = Object.keys(clusterProperties);\n\n for (const key of propertyNames) {\n const [operator, mapExpression] = clusterProperties[key];\n\n const mapExpressionParsed = createExpression(mapExpression);\n const reduceExpressionParsed = createExpression(\n typeof operator === 'string' ? [operator, ['accumulated'], ['get', key]] : operator);\n\n mapExpressions[key] = mapExpressionParsed.value;\n reduceExpressions[key] = reduceExpressionParsed.value;\n }\n\n superclusterOptions.map = (pointProperties) => {\n feature.properties = pointProperties;\n const properties = {};\n for (const key of propertyNames) {\n properties[key] = mapExpressions[key].evaluate(globals, feature);\n }\n return properties;\n };\n superclusterOptions.reduce = (accumulated, clusterProperties) => {\n feature.properties = clusterProperties;\n for (const key of propertyNames) {\n globals.accumulated = accumulated[key];\n accumulated[key] = reduceExpressions[key].evaluate(globals, feature);\n }\n };\n\n return superclusterOptions;\n}\n","import {Actor, ActorTarget} from '../util/actor';\nimport {StyleLayerIndex} from '../style/style_layer_index';\nimport {VectorTileWorkerSource} from './vector_tile_worker_source';\nimport {RasterDEMTileWorkerSource} from './raster_dem_tile_worker_source';\nimport {GeoJSONWorkerSource} from './geojson_worker_source';\nimport {plugin as globalRTLTextPlugin} from './rtl_text_plugin';\nimport {isWorker} from '../util/util';\n\nimport type {\n WorkerSource,\n WorkerTileParameters,\n WorkerDEMTileParameters,\n WorkerTileCallback,\n WorkerDEMTileCallback,\n TileParameters\n} from '../source/worker_source';\n\nimport type {WorkerGlobalScopeInterface} from '../util/web_worker';\nimport type {Callback} from '../types/callback';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {PluginState} from './rtl_text_plugin';\n\n/**\n * The Worker class responsidble for background thread related execution\n */\nexport default class Worker {\n self: WorkerGlobalScopeInterface & ActorTarget;\n actor: Actor;\n layerIndexes: {[_: string]: StyleLayerIndex};\n availableImages: {[_: string]: Array};\n workerSourceTypes: {\n [_: string]: {\n new (...args: any): WorkerSource;\n };\n };\n workerSources: {\n [_: string]: {\n [_: string]: {\n [_: string]: WorkerSource;\n };\n };\n };\n demWorkerSources: {\n [_: string]: {\n [_: string]: RasterDEMTileWorkerSource;\n };\n };\n referrer: string;\n\n constructor(self: WorkerGlobalScopeInterface & ActorTarget) {\n this.self = self;\n this.actor = new Actor(self, this);\n\n this.layerIndexes = {};\n this.availableImages = {};\n\n this.workerSourceTypes = {\n vector: VectorTileWorkerSource,\n geojson: GeoJSONWorkerSource\n };\n\n // [mapId][sourceType][sourceName] => worker source instance\n this.workerSources = {};\n this.demWorkerSources = {};\n\n this.self.registerWorkerSource = (name: string, WorkerSource: {\n new (...args: any): WorkerSource;\n }) => {\n if (this.workerSourceTypes[name]) {\n throw new Error(`Worker source with name \"${name}\" already registered.`);\n }\n this.workerSourceTypes[name] = WorkerSource;\n };\n\n // This is invoked by the RTL text plugin when the download via the `importScripts` call has finished, and the code has been parsed.\n this.self.registerRTLTextPlugin = (rtlTextPlugin: {\n applyArabicShaping: Function;\n processBidirectionalText: ((b: string, a: Array) => Array);\n processStyledBidirectionalText?: ((c: string, b: Array, a: Array) => Array<[string, Array]>);\n }) => {\n if (globalRTLTextPlugin.isParsed()) {\n throw new Error('RTL text plugin already registered.');\n }\n globalRTLTextPlugin['applyArabicShaping'] = rtlTextPlugin.applyArabicShaping;\n globalRTLTextPlugin['processBidirectionalText'] = rtlTextPlugin.processBidirectionalText;\n globalRTLTextPlugin['processStyledBidirectionalText'] = rtlTextPlugin.processStyledBidirectionalText;\n };\n }\n\n setReferrer(mapID: string, referrer: string) {\n this.referrer = referrer;\n }\n\n setImages(mapId: string, images: Array, callback: WorkerTileCallback) {\n this.availableImages[mapId] = images;\n for (const workerSource in this.workerSources[mapId]) {\n const ws = this.workerSources[mapId][workerSource];\n for (const source in ws) {\n ws[source].availableImages = images;\n }\n }\n callback();\n }\n\n setLayers(mapId: string, layers: Array, callback: WorkerTileCallback) {\n this.getLayerIndex(mapId).replace(layers);\n callback();\n }\n\n updateLayers(mapId: string, params: {\n layers: Array;\n removedIds: Array;\n }, callback: WorkerTileCallback) {\n this.getLayerIndex(mapId).update(params.layers, params.removedIds);\n callback();\n }\n\n loadTile(mapId: string, params: WorkerTileParameters & {\n type: string;\n }, callback: WorkerTileCallback) {\n this.getWorkerSource(mapId, params.type, params.source).loadTile(params, callback);\n }\n\n loadDEMTile(mapId: string, params: WorkerDEMTileParameters, callback: WorkerDEMTileCallback) {\n this.getDEMWorkerSource(mapId, params.source).loadTile(params, callback);\n }\n\n reloadTile(mapId: string, params: WorkerTileParameters & {\n type: string;\n }, callback: WorkerTileCallback) {\n this.getWorkerSource(mapId, params.type, params.source).reloadTile(params, callback);\n }\n\n abortTile(mapId: string, params: TileParameters & {\n type: string;\n }, callback: WorkerTileCallback) {\n this.getWorkerSource(mapId, params.type, params.source).abortTile(params, callback);\n }\n\n removeTile(mapId: string, params: TileParameters & {\n type: string;\n }, callback: WorkerTileCallback) {\n this.getWorkerSource(mapId, params.type, params.source).removeTile(params, callback);\n }\n\n removeDEMTile(mapId: string, params: TileParameters) {\n this.getDEMWorkerSource(mapId, params.source).removeTile(params);\n }\n\n removeSource(mapId: string, params: {\n source: string;\n } & {\n type: string;\n }, callback: WorkerTileCallback) {\n\n if (!this.workerSources[mapId] ||\n !this.workerSources[mapId][params.type] ||\n !this.workerSources[mapId][params.type][params.source]) {\n return;\n }\n\n const worker = this.workerSources[mapId][params.type][params.source];\n delete this.workerSources[mapId][params.type][params.source];\n\n if (worker.removeSource !== undefined) {\n worker.removeSource(params, callback);\n } else {\n callback();\n }\n }\n\n /**\n * Load a {@link WorkerSource} script at params.url. The script is run\n * (using importScripts) with `registerWorkerSource` in scope, which is a\n * function taking `(name, workerSourceObject)`.\n */\n loadWorkerSource(map: string, params: {\n url: string;\n }, callback: Callback) {\n try {\n this.self.importScripts(params.url);\n callback();\n } catch (e) {\n callback(e.toString());\n }\n }\n\n syncRTLPluginState(map: string, state: PluginState, callback: Callback) {\n try {\n globalRTLTextPlugin.setState(state);\n const pluginURL = globalRTLTextPlugin.getPluginURL();\n if (\n globalRTLTextPlugin.isLoaded() &&\n !globalRTLTextPlugin.isParsed() &&\n pluginURL != null // Not possible when `isLoaded` is true, but keeps flow happy\n ) {\n this.self.importScripts(pluginURL);\n const complete = globalRTLTextPlugin.isParsed();\n const error = complete ? undefined : new Error(`RTL Text Plugin failed to import scripts from ${pluginURL}`);\n callback(error, complete);\n }\n } catch (e) {\n callback(e.toString());\n }\n }\n\n getAvailableImages(mapId: string) {\n let availableImages = this.availableImages[mapId];\n\n if (!availableImages) {\n availableImages = [];\n }\n\n return availableImages;\n }\n\n getLayerIndex(mapId: string) {\n let layerIndexes = this.layerIndexes[mapId];\n if (!layerIndexes) {\n layerIndexes = this.layerIndexes[mapId] = new StyleLayerIndex();\n }\n return layerIndexes;\n }\n\n getWorkerSource(mapId: string, sourceType: string, sourceName: string): WorkerSource {\n if (!this.workerSources[mapId])\n this.workerSources[mapId] = {};\n if (!this.workerSources[mapId][sourceType])\n this.workerSources[mapId][sourceType] = {};\n\n if (!this.workerSources[mapId][sourceType][sourceName]) {\n // use a wrapped actor so that we can attach a target mapId param\n // to any messages invoked by the WorkerSource\n const actor = {\n send: (type, data, callback) => {\n this.actor.send(type, data, callback, mapId);\n }\n };\n this.workerSources[mapId][sourceType][sourceName] = new (this.workerSourceTypes[sourceType] as any)((actor as any), this.getLayerIndex(mapId), this.getAvailableImages(mapId));\n }\n\n return this.workerSources[mapId][sourceType][sourceName];\n }\n\n getDEMWorkerSource(mapId: string, source: string) {\n if (!this.demWorkerSources[mapId])\n this.demWorkerSources[mapId] = {};\n\n if (!this.demWorkerSources[mapId][source]) {\n this.demWorkerSources[mapId][source] = new RasterDEMTileWorkerSource();\n }\n\n return this.demWorkerSources[mapId][source];\n }\n}\n\nif (isWorker()) {\n (self as any).worker = new Worker(self as any);\n}\n"],"names":["__awaiter","thisArg","_arguments","P","generator","Promise","resolve","reject","fulfilled","value","step","next","e","rejected","result","done","then","apply","SuppressedError","pointGeometry","Point","x","y","this","prototype","clone","add","p","_add","sub","_sub","multByPoint","_multByPoint","divByPoint","_divByPoint","mult","k","_mult","div","_div","rotate","a","_rotate","rotateAround","_rotateAround","matMult","m","_matMult","unit","_unit","perp","_perp","round","_round","mag","Math","sqrt","equals","other","dist","distSqr","dx","dy","angle","atan2","angleTo","b","angleWith","angleWithSep","cos","sin","convert","Array","isArray","unitbezier","UnitBezier","p1x","p1y","p2x","p2y","cx","bx","ax","cy","by","ay","sampleCurveX","t","sampleCurveY","sampleCurveDerivativeX","solveCurveX","epsilon","undefined","i","x2","abs","d2","t0","t1","solve","supportsOffscreenCanvas","offscreenCanvasDistorted","clamp","n","min","max","extend","dest","sources","src","mapObject","input","iterator","context","output","key","call","map","warnOnceHistory","warnOnce","message","console","warn","isCounterClockwise","c","calculateSignedArea","ring","sum","p1","p2","len","length","j","isWorker","WorkerGlobalScope","self","isImageBitmap","image","ImageBitmap","offscreenCanvas","offscreenCanvasContext","TransferableGridIndex","constructor","extent","padding","cells","ArrayBuffer","arrayBuffer","array","Int32Array","d","start","end","push","subarray","bboxesOffset","keys","bboxes","insert","_insertReadonly","scale","uid","x1","y1","y2","_forEachCell","_insertCell","Error","cellIndex","query","intersectionTest","slice","_queryCell","seenUids","cell","u","offset","fn","arg1","arg2","cx1","_convertToCellCoord","cy1","cx2","cy2","_convertFromCellCoord","floor","toArrayBuffer","metadataLength","totalCellLength","set","buffer","static","grid","transferables","serialized","v8Spec","$version","$root","version","required","type","values","name","metadata","center","zoom","bearing","default","period","units","pitch","light","terrain","sprite","glyphs","transition","layers","source","source_vector","vector","url","tiles","bounds","scheme","xyz","tms","minzoom","maxzoom","attribution","promoteId","volatile","source_raster","raster","tileSize","source_raster_dem","encoding","terrarium","mapbox","custom","redFactor","blueFactor","greenFactor","baseShift","source_geojson","geojson","data","maximum","minimum","filter","tolerance","cluster","clusterRadius","clusterMaxZoom","clusterMinPoints","clusterProperties","lineMetrics","generateId","source_video","video","urls","coordinates","source_image","layer","id","fill","line","symbol","circle","heatmap","hillshade","background","layout","paint","layout_background","visibility","visible","none","layout_fill","expression","interpolated","parameters","layout_circle","layout_heatmap","layout_line","butt","square","bevel","miter","requires","layout_symbol","point","auto","never","always","cooperative","viewport","width","height","both","tokens","left","right","top","bottom","horizontal","vertical","uppercase","lowercase","layout_raster","layout_hillshade","filter_operator","in","all","any","has","within","geometry_type","LineString","Polygon","function","stops","base","property","identity","exponential","interval","categorical","colorSpace","rgb","lab","hcl","function_stop","anchor","position","color","intensity","exaggeration","paint_fill","paint_line","paint_circle","paint_heatmap","paint_symbol","overridable","paint_raster","linear","nearest","paint_hillshade","paint_background","duration","delay","constant","refProperties","ValidationError","identifier","__line__","extendBy","inputs","ExpressionParsingError","super","Scope","parent","bindings","concat","get","NullType","kind","NumberType","StringType","BooleanType","ColorType","ObjectType","ValueType","CollatorType","FormattedType","PaddingType","ResolvedImageType","VariableAnchorOffsetCollectionType","array$1","itemType","N","toString$1","valueMemberTypes","checkSubtype","expected","memberType","isValidType","provided","allowedTypes","some","isValidNativeType","verifyType","sample","Xn","Zn","t2","t3","deg2rad","PI","rad2deg","constrainAngle","rgbToLab","r","g","alpha","z","xyz2lab","rgb2xyz","l","pow","labToRgb","isNaN","lab2xyz","xyz2rgb","parseHex","hex","parseInt","padEnd","parseAlpha","asPercentage","validateNumbers","Number","namedColors","aliceblue","antiquewhite","aqua","aquamarine","azure","beige","bisque","black","blanchedalmond","blue","blueviolet","brown","burlywood","cadetblue","chartreuse","chocolate","coral","cornflowerblue","cornsilk","crimson","cyan","darkblue","darkcyan","darkgoldenrod","darkgray","darkgreen","darkgrey","darkkhaki","darkmagenta","darkolivegreen","darkorange","darkorchid","darkred","darksalmon","darkseagreen","darkslateblue","darkslategray","darkslategrey","darkturquoise","darkviolet","deeppink","deepskyblue","dimgray","dimgrey","dodgerblue","firebrick","floralwhite","forestgreen","fuchsia","gainsboro","ghostwhite","gold","goldenrod","gray","green","greenyellow","grey","honeydew","hotpink","indianred","indigo","ivory","khaki","lavender","lavenderblush","lawngreen","lemonchiffon","lightblue","lightcoral","lightcyan","lightgoldenrodyellow","lightgray","lightgreen","lightgrey","lightpink","lightsalmon","lightseagreen","lightskyblue","lightslategray","lightslategrey","lightsteelblue","lightyellow","lime","limegreen","linen","magenta","maroon","mediumaquamarine","mediumblue","mediumorchid","mediumpurple","mediumseagreen","mediumslateblue","mediumspringgreen","mediumturquoise","mediumvioletred","midnightblue","mintcream","mistyrose","moccasin","navajowhite","navy","oldlace","olive","olivedrab","orange","orangered","orchid","palegoldenrod","palegreen","paleturquoise","palevioletred","papayawhip","peachpuff","peru","pink","plum","powderblue","purple","rebeccapurple","red","rosybrown","royalblue","saddlebrown","salmon","sandybrown","seagreen","seashell","sienna","silver","skyblue","slateblue","slategray","slategrey","snow","springgreen","steelblue","tan","teal","thistle","tomato","turquoise","violet","wheat","white","whitesmoke","yellow","yellowgreen","Color","premultiplied","overwriteGetter","rgba","toLowerCase","trim","namedColorsMatch","startsWith","test","rgbMatch","match","_","rp","f1","gp","f2","bp","f3","ap","argFormat","join","valFormat","maxValue","hslMatch","h","s","hsla","f","hslToRgb","parseCssColor","Infinity","rgbColor","NaN","rgbToHcl","getterKey","lazyValue","Object","defineProperty","toString","transparent","Collator","caseSensitive","diacriticSensitive","locale","sensitivity","collator","Intl","usage","compare","lhs","rhs","resolvedLocale","resolvedOptions","FormattedSection","text","fontStack","textColor","Formatted","sections","unformatted","isEmpty","section","fromString","Padding","val","JSON","stringify","anchors","Set","VariableAnchorOffsetCollection","anchorValue","offsetValue","ResolvedImage","options","available","validateRGBA","isValue","mixed","item","typeOf","String","Literal","args","error","expectedType","evaluate","eachChild","outputDefined","RuntimeError","toJSON","types$1","string","number","boolean","object","Assertion","parsed","parse","ctx","forEach","every","arg","types","Coercion","Boolean","parseColor","pad","coll","num","geometryTypes","EvaluationContext","globals","feature","featureState","formattedSection","_parseColorCache","availableImages","canonical","geometryType","geometry","canonicalID","properties","cached","ParsingContext","registry","isConstantFunc","path","scope","errors","part","_isConstant","expr","index","_parse","annotate","typeAnnotation","op","Expr","actual","ec","CollatorExpression","EXTENT","updateBBox","bbox","coord","boxWithinBox","bbox1","bbox2","getTileCoordinates","log","tilesAtZoom","onBoundary","pointWithinPolygon","rings","inside","len2","pointWithinPolygons","polygons","twoSided","q1","q2","x3","y3","det1","det2","lineIntersectPolygon","polygon","v1","v2","lineStringWithinPolygon","lineStringWithinPolygons","getTilePolygon","getTilePolygons","updatePoint","polyBBox","worldSize","halfWorldSize","shift","getTilePoints","pointBBox","shifts","tilePoints","points","getTileLines","lineBBox","tileLines","tileLine","Within","geometries","features","polygonGeometry","tilePolygon","tilePolygons","pointsWithinPolygons","linesWithinPolygons","Var","boundExpression","CompoundExpression","_evaluate","definition","definitions","availableOverloads","overloads","signature","signatureContext","params","isExpressionConstant","parsedArgs","argParseFailed","signatures","stringifySignature","actualTypes","isTypeAnnotation","childrenConstant","child","isFeatureConstant","isGlobalPropertyConstant","isStateConstant","indexOf","findStopLessThanOrEqualTo","lastIndex","currentValue","nextValue","lowerIndex","upperIndex","currentIndex","Step","labels","outputs","label","outputType","labelKey","valueKey","stopCount","out","from","to","interpolate","spaceKey","hue0","chroma0","light0","alphaF","hue1","chroma1","light1","alphaT","hue","chroma","dh","hclToRgb","variableAnchorOffsetCollection","fromValues","toValues","fx","fy","tx","ty","Interpolate","operator","interpolation","lower","upper","exponentialInterpolation","controlPoints","rest","interpolationFactor","outputLower","outputUpper","lowerValue","upperValue","difference","progress","Coalesce","needsAnnotation","requestedImageName","argCount","Let","binding","At","In","needle","haystack","IndexOf","fromIndex","Match","inputType","cases","otherwise","labelContext","MAX_SAFE_INTEGER","Case","branches","Slice","beginIndex","endIndex","isComparableType","eqCollate","makeComparison","compareBasic","compareWithCollator","isOrderComparison","Comparison","hasUntypedArgument","lt","rt","Equals","NotEquals","LessThan","GreaterThan","LessThanOrEqual","GreaterThanOrEqual","NumberFormat","currency","minFractionDigits","maxFractionDigits","style","minimumFractionDigits","maximumFractionDigits","format","FormatExpression","firstArg","nextTokenMayBeObject","font","lastExpression","content","evaluatedContent","ImageExpression","evaluatedImageName","Length","expressions","at","case","coalesce","let","literal","var","obj","v","varargs","success","supportsPropertyExpression","spec","supportsZoomExpression","supportsInterpolation","getType","isFunction","identityFunction","createFunction","propertySpec","isColor","zoomAndFeatureDependent","zoomDependent","parseFn","stop","innerFun","hashedStops","categoricalKeyType","evaluateExponentialFunction","evaluateIntervalFunction","evaluateCategoricalFunction","create","evaluateIdentityFunction","featureFunctions","zoomStops","featureFunctionStops","interpolationType","bind","coalesce$1","keyType","interp","evaluatedLower","evaluatedUpper","register","typeof","heatmapDensity","lineProgress","accumulated","ln2","LN2","pi","E","log10","LN10","ln","log2","asin","acos","atan","ceil","binarySearch","isSupportedScript","upcase","toUpperCase","downcase","StyleExpression","_warningHistory","_evaluator","_defaultValue","_enumValues","evaluateWithoutErrorHandling","isExpression","createExpression","parser","enum","formatted","resolvedImage","getExpectedType","ZoomConstantExpression","_styleExpression","isStateDependent","ZoomDependentExpression","createPropertyExpression","expressionInput","isFeatureConstantResult","isZoomConstant","zoomCurve","findZoomCurve","StylePropertyFunction","specification","_parameters","_specification","childResult","isExpressionFilter","filterSpec","createFilter","needGeometry","convertFilter$1","compiled","err","globalProperties","geometryNeeded","convertComparisonOp$1","convertNegation","filters","convertInOp$1","convertHasOp$1","sort","str","getKey","validateConstants","constants","unbundle","valueOf","deepUnbundle","unbundledValue","validateObject","elementSpecs","valueSpec","elementValidators","objectElementValidators","styleSpec","validateSpec","objectKey","elementSpecKey","split","elementSpec","validateElement","validateArray","arraySpec","validateArrayElement","arrayElementValidator","arrayElementSpec","arrayIndex","validateNumber","validateFunction","functionValueSpec","functionType","stopKeyType","previousStopDomainValue","previousStopDomainZoom","stopDomainValues","isZoomFunction","isPropertyFunction","isZoomAndPropertyFunction","validateFunctionStop","validateStopDomainValue","reportValue","isFinite","validateExpression","expressionContext","expressionObj","propertyKey","propertyType","validateEnum","validateFilter","validateNonExpressionFilter","validateProperty","layerSpec","layerType","transitionMatch","tokenMatch","exec","validatePaintProperty","validateLayoutProperty","validateLayer","ref","layerId","otherLayer","sourceType","validateString","prop","validateSource","replace","_a","sourceName","rasterDEM","rasterDEMSpec","rootType","isCustomEncoding","customEncodingKeys","encodingName","includes","validateRasterDEMSource","mapExpr","reduceExpr","validateLight","lightSpec","validateTerrain","terrainSpec","validateSprite","allSpriteIds","allSpriteURLs","VALIDATORS","validate","validateGlyphsUrl","validateStyleMin","sortErrors","injectValidateSpec","validator","wrapCleanErrors","inner","paintProperty","layoutProperty","config","AJAXError","status","statusText","body","getReferrer","worker","referrer","window","location","protocol","href","makeFetchRequest","requestParameters","callback","controller","AbortController","request","Request","method","credentials","headers","cache","signal","complete","aborted","fetch","response","ok","json","catch","finishRequest","blob","code","cancel","abort","makeRequest","actor","send","substring","isFileURL","hasOwnProperty","xhr","XMLHttpRequest","open","responseType","setRequestHeader","withCredentials","onerror","onload","getResponseHeader","Blob","makeXMLHttpRequest","klass","writeable","omit","shallow","_classRegistryKey","isArrayBuffer","serialize","Date","RegExp","isView","view","ImageData","$name","deserialize","ThrottledInvoker","_callback","_triggered","MessageChannel","_channel","port2","onmessage","trigger","port1","postMessage","setTimeout","remove","Actor","target","mapId","receive","targetMapId","tasks","cancelCallbacks","mustQueue","taskQueue","invoker","processTask","process","task","callbacks","addEventListener","globalScope","random","buffers","hasCallback","sourceMapId","transfer","completed","responseMessage","getWorkerSource","removeEventListener","_addEventListener","listener","listenerList","_removeEventListener","splice","Event","ErrorEvent","Evented","on","_listeners","off","_oneTimeListeners","once","fire","event","listens","listeners","oneTimeListeners","_eventedParent","_eventedParentData","setEventedParent","validateStyle","ZoomHistory","first","update","now","floorZ","lastIntegerZoom","lastIntegerZoomTime","lastZoom","lastFloorZoom","unicodeBlockLookup","char","Arabic","Khmer","Hiragana","Katakana","Bopomofo","Kanbun","allowsVerticalWritingMode","chars","charHasUprightVerticalOrientation","charCodeAt","allowsLetterSpacing","charAllowsLetterSpacing","isChar","charHasRotatedVerticalOrientation","charHasNeutralVerticalOrientation","charInRTLScript","charInSupportedScript","canRenderRTL","stringContainsRTLText","performance","pluginStatus","pluginURL","plugin","applyArabicShaping","processBidirectionalText","processStyledBidirectionalText","isLoaded","isLoading","setState","state","isParsed","getPluginURL","EvaluationParameters","fadeDuration","zoomHistory","isStringInSupportedScript","rtlTextPlugin","crossFadingFactor","getCrossfadeParameters","fraction","fromScale","toScale","PropertyValue","normalizePropertyExpression","isDataDriven","possiblyEvaluate","TransitionablePropertyValue","transitioned","prior","TransitioningPropertyValue","untransitioned","Transitionable","_properties","_values","defaultTransitionablePropertyValues","getValue","setValue","getTransition","setTransition","Transitioning","begin","finalValue","easeCubicInOut","defaultTransitioningPropertyValues","PossiblyEvaluated","hasTransition","Layout","defaultPropertyValues","hasValue","PossiblyEvaluatedPropertyValue","isConstant","constantOr","defaultPossiblyEvaluatedValues","DataConstantProperty","interpolationFn","interpolates","DataDrivenProperty","overrides","interpolatedValue","CrossFadedDataDrivenProperty","evaluatedValue","constantValue","_calculate","cameraVal","mid","CrossFadedProperty","ColorRampProperty","Properties","overridableProperties","defaultPropertyValue","defaultTransitionablePropertyValue","TRANSITION_SUFFIX","StyleLayer","_featureFilter","sourceLayer","_unevaluatedLayout","_transitionablePaint","setPaintProperty","setLayoutProperty","_transitioningPaint","_crossfadeParameters","getLayoutProperty","_validate","getPaintProperty","endsWith","transitionable","isCrossFadedProperty","wasDataDriven","oldValue","_handleSpecialPaintPropertyUpdate","newValue","_handleOverridablePaintPropertyUpdate","isHidden","updateTransitions","recalculate","filterObject","emitter","hasErrors","emitValidationErrors","is3D","isTileClipped","hasOffscreenPass","resize","viewTypes","Int8","Int8Array","Uint8","Uint8Array","Int16","Int16Array","Uint16","Uint16Array","Int32","Uint32","Uint32Array","Float32","Float32Array","Struct","structArray","_structArray","_pos1","size","_pos2","_pos4","_pos8","StructArray","isTransferred","capacity","_trim","byteLength","bytesPerElement","_refreshViews","clear","reserve","oldUint8Array","uint8","createLayout","members","alignment","maxSize","member","typeSize","BYTES_PER_ELEMENT","memberOffset","align","components","StructArrayLayout2i4","int16","emplaceBack","v0","emplace","o2","StructArrayLayout3i6","StructArrayLayout4i8","v3","StructArrayLayout2i4i12","v4","v5","StructArrayLayout2i4ub8","o1","StructArrayLayout2f8","float32","o4","StructArrayLayout10ui20","uint16","v6","v7","v8","v9","StructArrayLayout4i4ui4i24","v10","v11","StructArrayLayout3f12","StructArrayLayout1ul4","uint32","StructArrayLayout6i1ul2ui20","StructArrayLayout2i2i2i12","StructArrayLayout2f1f2i16","StructArrayLayout2ub2f12","StructArrayLayout3ui6","StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48","v12","v13","v14","v15","v16","StructArrayLayout8i15ui1ul2f2ui64","v17","v18","v19","v20","v21","v22","v23","v24","v25","v26","v27","StructArrayLayout1f4","StructArrayLayout1ui2f12","StructArrayLayout1ul2ui8","StructArrayLayout2ui4","StructArrayLayout1ui2","StructArrayLayout4f16","CollisionBoxStruct","anchorPointX","anchorPointY","featureIndex","sourceLayerIndex","bucketIndex","anchorPoint","CollisionBoxArray","PlacedSymbolStruct","anchorX","anchorY","glyphStartIndex","numGlyphs","vertexStartIndex","lineStartIndex","lineLength","segment","lowerSize","upperSize","lineOffsetX","lineOffsetY","writingMode","placedOrientation","hidden","crossTileID","associatedIconIndex","PlacedSymbolArray","SymbolInstanceStruct","rightJustifiedTextSymbolIndex","centerJustifiedTextSymbolIndex","leftJustifiedTextSymbolIndex","verticalPlacedTextSymbolIndex","placedIconSymbolIndex","verticalPlacedIconSymbolIndex","textBoxStartIndex","textBoxEndIndex","verticalTextBoxStartIndex","verticalTextBoxEndIndex","iconBoxStartIndex","iconBoxEndIndex","verticalIconBoxStartIndex","verticalIconBoxEndIndex","numHorizontalGlyphVertices","numVerticalGlyphVertices","numIconVertices","numVerticalIconVertices","useRuntimeCollisionCircles","textBoxScale","collisionCircleDiameter","textAnchorOffsetStartIndex","textAnchorOffsetEndIndex","SymbolInstanceArray","GlyphOffsetArray","getoffsetX","SymbolLineVertexArray","getx","gety","gettileUnitDistanceFromAnchor","TextAnchorOffsetStruct","textAnchor","textOffset0","textOffset1","TextAnchorOffsetArray","FeatureIndexStruct","FeatureIndexArray","PosArray","CircleLayoutArray","FillLayoutArray","FillExtrusionLayoutArray","LineLayoutArray","LineExtLayoutArray","PatternLayoutArray","SymbolLayoutArray","SymbolDynamicLayoutArray","SymbolOpacityArray","CollisionBoxLayoutArray","CollisionVertexArray","TriangleIndexArray","LineIndexArray","SegmentVector","segments","prepareSegment","numVertices","layoutVertexArray","indexArray","sortKey","MAX_VERTEX_ARRAY_LENGTH","vertexLength","vertexOffset","primitiveOffset","primitiveLength","destroy","vaos","packUint8ToFloat","patternAttributes","seed","remainder","bytes","h1","h1b","c1","c2","k1","murmur3","require$$0","murmur2","murmurhashJsModule","exports","FeaturePositionMap","ids","positions","indexed","getNumericId","getPositions","intId","Float64Array","numValue","pivot","swap","arr","tmp","Uniform","gl","Uniform1f","current","uniform1f","Uniform4f","uniform4f","UniformColor","packColor","ConstantBinder","names","uniformNames","setUniform","uniform","getBinding","CrossFadedConstantBinder","patternFrom","patternTo","pixelRatioFrom","pixelRatioTo","setConstantPatternPositions","posTo","posFrom","pixelRatio","tlbr","uniformName","pos","substr","SourceExpressionBinder","PaintVertexArray","paintVertexAttributes","paintVertexArray","populatePaintArray","newLength","imagePositions","_setPaintValue","updatePaintArray","upload","paintVertexBuffer","updateData","createVertexBuffer","CompositeExpressionBinder","useIntegerZoom","minColor","maxColor","currentZoom","factor","CrossFadedCompositeBinder","zoomInPaintVertexArray","zoomOutPaintVertexArray","_setPaintValues","patterns","imageMin","imageMid","imageMax","tl","br","zoomInPaintVertexBuffer","zoomOutPaintVertexBuffer","ProgramConfiguration","filterProperties","binders","_buffers","paintAttributeNames","propType","isCrossFaded","StructArrayLayout","layoutType","cacheKey","getMaxValue","binder","populatePaintArrays","updatePaintArrays","featureStates","featureMap","vtLayer","dirty","defines","getBinderAttributes","getBinderUniforms","uniforms","getPaintVertexBuffers","getUniforms","locations","setUniforms","binderUniforms","updatePaintBuffers","crossfade","patternVertexBuffer","ProgramConfigurationSet","programConfigurations","needsUpload","_featureMap","_bufferOffset","binderType","defaultLayouts","composite","layoutException","getLayoutException","MAX","BITS","MIN","loadGeometry","toEvaluationFeature","addCircleVertex","extrudeX","extrudeY","CircleBucket","overscaling","layerIds","hasPattern","stateDependentLayerIds","populate","styleLayer","bucketFeatures","circleSortKey","sortFeaturesByKey","evaluationFeature","bucketFeature","addFeature","states","stateDependentLayers","uploadPending","uploaded","layoutVertexBuffer","layoutAttributes","indexBuffer","createIndexBuffer","polygonIntersectsPolygon","polygonA","polygonB","polygonContainsPoint","lineIntersectsLine","polygonIntersectsBufferedPoint","radius","pointIntersectsBufferedLine","polygonIntersectsMultiPolygon","multiPolygon","multiPolygonContainsPoint","lineIntersectsBufferedLine","lineA","lineB","a0","a1","lineSegmentIntersectsLineSegment","b0","b1","radiusSquared","distToSegmentSquared","w","l2","edgeIntersectsBox","e1","e2","corners","dir","getMaximumPaintValue","bucket","translateDistance","translate","queryGeometry","translateAnchor","pixelsToTileUnits","pt","translated","properties$8","ARRAY_TYPE","transformMat4","hypot","arguments","glMatrix.ARRAY_TYPE","CircleStyleLayer","createBucket","queryRadius","circleBucket","queryIntersectsFeature","transform","pixelPosMatrix","translatedPolygon","alignWithMap","transformedPolygon","projectPoint","projectQueryGeometry","transformedSize","transformedPoint","adjustedSize","projectedCenter","vec4.transformMat4","cameraToCenterDistance","HeatmapBucket","properties$7","createImage","channels","Uint8ClampedArray","RangeError","resizeImage","newImage","copyImage","srcImg","dstImg","srcPt","dstPt","srcData","dstData","srcOffset","dstOffset","AlphaImage","RGBAImage","copy","HeatmapStyleLayer","_updateColorRamp","colorRamp","evaluationGlobals","resolution","clips","renderPixel","stride","evaluationKey","pxColor","clip","renderColorRamp","colorRampTexture","heatmapFbo","properties$6","HillshadeStyleLayer","earcut","holeIndices","dim","minX","minY","maxX","maxY","invSize","hasHoles","outerLen","outerNode","linkedList","triangles","prev","list","queue","steiner","getLeftmost","compareX","eliminateHole","eliminateHoles","earcutLinked","clockwise","last","signedArea","insertNode","removeNode","filterPoints","again","area","ear","pass","zOrder","prevZ","nextZ","q","tail","numMerges","pSize","qSize","inSize","sortLinked","indexCurve","isEarHashed","isEar","cureLocalIntersections","splitEarcut","x0","y0","pointInTriangle","minZ","maxZ","intersects","locallyInside","isValidDiagonal","splitPolygon","hole","bridge","hx","hy","qx","mx","my","tanMin","sectorContainsSector","findHoleBridge","bridgeReverse","leftmost","px","py","intersectsPolygon","middleInside","sign","o3","onSegment","a2","Node","b2","an","earcutModule","deviation","polygonArea","trianglesArea","flatten","vertices","holes","dimensions","holeIndex","quickselect","quickselectStep","defaultCompare","exp","sd","classifyRings","maxRings","ccw","compareAreas","patternDependencies","patternProperty","constantPattern","addPatternDependencies","patternFeature","patternPropertyValue","FillBucket","patternFeatures","indexArray2","segments2","fillSortKey","addFeatures","indexBuffer2","triangleSegment","triangleIndex","flattened","lineSegment","lineIndex","indices","properties$5","FillStyleLayer","outlineColor","centroidAttributes","vectortilefeature","VectorTileFeature","pbf","_pbf","_geometry","_keys","readFields","readFeature","tag","readVarint","readTag","cmd","lines","cmdLen","readSVarint","toGeoJSON","coords","project","vectortilelayer","VectorTileLayer","_features","readLayer","readString","readFloat","readDouble","readVarint64","readBoolean","readValueMessage","readTile","vectorTile","VectorTile","require$$1","require$$2","vectorTileFeatureTypes","mvt","FACTOR","addVertex","vertexArray","nx","ny","nz","FillExtrusionBucket","centroidVertexArray","centroidVertexBuffer","centroid","vertexCount","isEntirelyOutside","edgeDistance","isBoundaryEdge","bottomRight","properties$4","FillExtrusionStyleLayer","projectedQueryGeometry","projected","zBase","zTop","projectedBase","projectedTop","baseXZ","baseYZ","baseZZ","baseWZ","topXZ","topYZ","topZZ","topWZ","ringBase","ringTop","sX","sY","sZ","sW","baseZ","baseW","topX","topY","topZ","topW","projectExtrusion","closestDistance","getIntersectionDistance","topA","face","checkIntersection","dot","projectedFace","ab","ac","dotABAB","dotABAC","dotACAC","dotAPAB","dotAPAC","denom","distance","lineLayoutAttributes","lineLayoutAttributesExt","COS_HALF_SHARP_CORNER","MAX_LINE_DISTANCE","LINE_DISTANCE_BUFFER_BITS","LineBucket","lineClipsArray","gradients","layoutVertexArray2","maxLineLength","lineSortKey","patternBucketFeature","layoutVertexBuffer2","layoutAttributesExt","lineFeatureClips","cap","miterLimit","roundLimit","lineClips","addLine","scaledDistance","totalDistance","updateScaledDistance","isPolygon","sharpCornerOffset","currentVertex","prevVertex","nextVertex","prevNormal","nextNormal","joinNormal","cosAngle","cosHalfAngle","miterLength","approxAngle","isSharpCorner","lineTurnsLeft","prevSegmentLength","newPrevVertex","updateDistance","addCurrentVertex","middleVertex","currentJoin","bevelLength","offsetA","offsetB","extrude","addHalfVertex","nextSegmentLength","newCurrentVertex","normal","endLeft","endRight","rightX","rightY","up","linesofarScaled","properties$3","LineFloorwidthProperty","lineFloorwidthProperty","LineStyleLayer","gradientVersion","gradientExpression","stepInterpolant","isZoomExpression","lineBucket","getLineWidth","halfWidth","lineOffset","newRings","ringIndex","newRing","aToB","bToC","offsetLine","multiLine","polygonIntersectsBufferedMultiLine","lineWidth","lineGapWidth","symbolLayoutAttributes","dynamicLayoutAttributes","collisionVertexAttributes","collisionBoxLayout","transformText","toLocaleUpperCase","toLocaleLowerCase","transformTextInternal","verticalizedCharacterMap","$","ONE_EM","Pbf","ieee754","isLE","mLen","nBytes","eLen","eMax","eBias","nBits","buf","Varint","Fixed64","Bytes","Fixed32","SHIFT_LEFT_32","SHIFT_RIGHT_32","utf8TextDecoder","TextDecoder","readPackedEnd","toNum","low","high","isSigned","makeRoomForExtraLength","startPos","extraLen","realloc","writePackedVarint","writeVarint","writePackedSVarint","writeSVarint","writePackedFloat","writeFloat","writePackedDouble","writeDouble","writePackedBoolean","writeBoolean","writePackedFixed32","writeFixed32","writePackedSFixed32","writeSFixed32","writePackedFixed64","writeFixed64","writePackedSFixed64","writeSFixed64","readUInt32","writeInt32","readInt32","readField","skip","readMessage","readFixed32","readSFixed32","readFixed64","readSFixed64","readVarintRemainder","decode","readUtf8TextDecoder","b3","bytesPerSequence","fromCharCode","readUtf8","readBytes","readPackedVarint","readPackedSVarint","readPackedBoolean","readPackedFloat","readPackedDouble","readPackedFixed32","readPackedSFixed32","readPackedFixed64","readPackedSFixed64","writeTag","finish","writeBigVarintLow","lsb","writeBigVarintHigh","writeBigVarint","writeString","lead","writeUtf8","writeBytes","writeRawMessage","writeMessage","writeBytesField","writeFixed32Field","writeSFixed32Field","writeFixed64Field","writeSFixed64Field","writeVarintField","writeSVarintField","writeStringField","writeFloatField","writeDoubleField","writeBooleanField","GLYPH_PBF_BORDER","potpack","boxes","maxWidth","box","spaces","space","pop","IMAGE_PADDING","ImagePosition","paddedRect","stretchX","stretchY","displaySize","ImageAtlas","icons","iconPositions","patternPositions","haveRenderCallbacks","bins","addImages","bin","images","hasRenderCallback","patchUpdatedImages","imageManager","texture","dispatchRenderCallbacks","updatedImages","patchUpdatedImage","getImage","WritingMode","SHAPING_DEFAULT_OFFSET","SectionOptions","imageName","textOptions","imageOptions","TaggedString","sectionIndex","imageSectionID","defaultFontStack","addImageSection","addTextSection","getSection","getSectionIndex","getCharCode","verticalizePunctuation","nextCharCode","prevCharCode","beginningWhitespace","whitespace","trailingWhitespace","getMaxScale","reduce","forText","nextImageSectionCharCode","getNextImageSectionCharCode","forImage","shapeText","glyphMap","glyphPositions","lineHeight","textJustify","spacing","allowVerticalPlacement","symbolPlacement","layoutTextSize","layoutTextSizeThisZoom","logicalInput","fromFeature","untaggedLines","determineLineBreaks","taggedLine","processedLines","lineBreakPoints","lineBreak","breakLines","positionedLines","shaping","iconsInText","verticalizable","maxLineHeight","justify","lineMaxScale","maxLineOffset","positionedLine","positionedGlyphs","codePoint","baselineOffset","metrics","rect","verticalAdvance","imagePosition","advance","glyphPosition","glyph","justifyLine","currentLineHeight","horizontalAlign","verticalAlign","getAnchorAlignment","blockHeight","lineCount","shiftX","shiftY","positionedGlyph","shapeLines","breakable","getGlyphAdvance","calculateBadness","targetWidth","penalty","isLastBreak","raggedness","calculatePenalty","nextCodePoint","penalizableIdeographicBreak","evaluateBreak","breakIndex","breakX","potentialBreaks","bestPriorBreak","bestBreakBadness","potentialBreak","breakBadness","badness","priorBreak","leastBadBreaks","lastLineBreak","potentialLineBreaks","totalWidth","determineAverageLineWidth","hasServerSuggestedBreakpoints","currentX","ideographicBreak","lastPositionedGlyph","lineIndent","shapeIcon","iconOffset","iconAnchor","fitIconToText","shapedIcon","shapedText","textFit","fontScale","collisionPadding","textLeft","textRight","textTop","textBottom","MAX_GLYPH_ICON_SIZE","SIZE_PACK_FACTOR","MAX_PACKED_SIZE","getSizeData","tileZoom","layoutSize","minZoom","maxZoom","minSize","getOverlapMode","overlapProp","allowOverlapProp","overlap","shaderOpacityAttributes","ox","oy","sizeVertex","isSDF","pixelOffsetX","pixelOffsetY","minFontScaleX","minFontScaleY","aSizeX","aSizeY","addDynamicAttributes","dynamicLayoutVertexArray","containsRTLText","formattedText","SymbolBuffers","opacityVertexArray","hasVisibleVertices","placedSymbolArray","dynamicIndexBuffer","dynamicLayoutVertexBuffer","opacityVertexBuffer","itemSize","CollisionBuffers","LayoutArray","IndexArray","collisionVertexArray","collisionVertexBuffer","SymbolBucket","collisionBoxArray","hasRTLText","sortKeyRanges","collisionCircleArray","placementInvProjMatrix","mat4.identity","placementViewportMatrix","unevaluatedLayoutValues","textSizeData","iconSizeData","canOverlap","sortFeaturesByY","writingModes","wm","sourceID","createArrays","icon","glyphOffsetArray","lineVertexArray","symbolInstances","textAnchorOffsets","calculateGlyphDependencies","stack","textAlongLine","doesAllowVerticalWritingMode","verticalChar","charAt","textFont","textField","iconImage","hasText","hasIcon","symbolSortKey","iconDependencies","stacks","glyphDependencies","resolvedTokens","getValueAndResolveTokens","factory","globalRTLTextPlugin","sectionFont","sectionStack","leftIndex","rightIndex","mergedFeatures","mergedIndex","mergeFromRight","leftKey","rightKey","geom","mergeFromLeft","onRight","mergeLines","hasDebugData","textCollisionBox","iconCollisionBox","destroyDebugData","addToLineVertexArray","sumForwardLength","sumBackwardLength","tileUnitDistanceFromAnchor","vertex","addSymbols","arrays","quads","alongLine","labelAnchor","glyphOffsetArrayStart","tr","bl","tex","pixelOffsetTL","pixelOffsetBR","glyphOffset","_addCollisionDebugVertex","addCollisionDebugVertices","boxAnchorPoint","symbolInstance","addDebugCollisionBoxes","startIndex","isText","generateCollisionDebugBuffers","_deserializeCollisionBoxesForSymbol","textStartIndex","textEndIndex","verticalTextStartIndex","verticalTextEndIndex","iconStartIndex","iconEndIndex","verticalIconStartIndex","verticalIconEndIndex","collisionArrays","textBox","textFeatureIndex","verticalTextBox","verticalTextFeatureIndex","iconBox","iconFeatureIndex","verticalIconBox","verticalIconFeatureIndex","deserializeCollisionBoxes","hasTextData","hasIconData","hasTextCollisionBoxData","hasIconCollisionBoxData","addIndicesForPlacedSymbol","iconOrText","placedSymbolIndex","placedSymbol","vertexIndex","getSortedSymbolIndexes","sortedAngle","symbolInstanceIndexes","rotatedYs","featureIndexes","aIndex","bIndex","addToSortKeyRanges","symbolInstanceIndex","symbolInstanceEnd","symbolInstanceStart","sortFeatures","featureSortOrder","MAX_GLYPHS","properties$2","runtimeType","getOverride","o","hasOverride","FormatSectionOverride","defaultValue","SymbolStyleLayer","deduped","_setPaintOverrides","unevaluated","resolveTokens","hasPaintOverride","overriden","override","styleExpression","propertyName","hasOverrides","checkSections","checkExpression","properties$1","BackgroundStyleLayer","RasterStyleLayer","CustomStyleLayer","implementation","onAdd","painter","onRemove","renderingMode","prerender","createStyleLayer","StyleLayerIndex","layerConfigs","keyCache","_layerConfigs","_layers","removedIds","layerConfig","featureFilter","familiesBySource","groups","cachedKeys","group","groupByLayout","sourceId","sourceGroup","sourceLayerId","sourceLayerFamilies","DictionaryCoder","strings","_stringToNumber","_numberToString","encode","GeoJSONFeature","vectorTileFeature","_vectorTileFeature","_z","_x","_y","FeatureIndex","tileID","grid3D","featureIndexArray","loadVTLayers","vtLayers","vt","Protobuf","rawTileData","sourceLayerCoder","styleLayers","serializedLayers","sourceFeatureState","queryPadding","getBounds","matching","cameraBounds","cameraQueryGeometry","matching3D","bx1","by1","bx2","by2","boxX1","boxY1","boxX2","boxY2","corner","polygonIntersectsBox","topDownFeatureComparator","previousIndex","featureGeometry","loadMatchingFeature","filterLayerIDs","layerIDs","bucketLayerIDs","arraysIntersect","sourceLayerName","overscaledZ","getId","layerID","getState","serializedLayer","evaluateProperties","intersectionZ","geojsonFeature","layerResult","lookupSymbolFeatures","symbolFeatureIndexes","symbolFeatureIndex","hasLayer","serializedProperties","styleLayerProperties","Anchor","checkMaxAngle","labelLength","windowSize","maxAngle","anchorDistance","recentCorners","recentAngleDelta","angleDelta","getLineLength","getAngleWindowSize","glyphSize","boxScale","getShapedLabelLength","getCenterAnchor","angleWindowSize","prevDistance","centerDistance","segmentDistance","getAnchors","tileExtent","shapedLabelLength","isLineContinued","resample","placeAtMiddle","halfLabelLength","markedDistance","segmentDist","border","getIconQuads","iconRotate","isSDFIcon","hasIconTextFit","imageWidth","imageHeight","iconWidth","iconHeight","reduceRanges","range","stretchWidth","stretchHeight","fixedWidth","fixedHeight","stretchOffsetX","stretchContentWidth","stretchOffsetY","stretchContentHeight","fixedOffsetX","fixedContentWidth","fixedOffsetY","fixedContentHeight","sumWithinRange","makeBox","leftEm","getEmOffset","stretch","leftPx","getPxOffset","fixed","topEm","topPx","rightEm","rightPx","bottomEm","bottomPx","matrix","xCuts","stretchZonesToCuts","yCuts","xi","yi","ranges","stretchZones","fixedSize","stretchSize","cuts","stretchOffset","iconSize","fixedOffset","CollisionFeature","shaped","alignLine","boxStartIndex","circleDiameter","rotateRadians","boxEndIndex","TinyQueue","_down","_up","peek","halfLength","best","findPoleOfInaccessibility","polygonRings","precision","debug","outerRing","cellSize","cellQueue","Queue","compareMax","Cell","bestCell","getCentroidCell","numProbes","minDistSq","pointToPolygonDist","SQRT2","TextAnchorEnum","INVALID_TEXT_OFFSET","POSITIVE_INFINITY","evaluateVariableOffset","offsetX","offsetY","fromTextOffset","radialOffset","hypotenuse","fromRadialOffset","getTextVariableAnchorOffset","variableAnchorOffset","sourceValues","destValues","variableAnchor","textOffset","anchorOffsets","performSymbolLayout","tilePixelRatio","compareText","iconsNeedLinear","sizes","layoutIconSize","textMaxSize","compositeTextSizes","compositeIconSizes","keepUpright","textSize","fontstack","shapedTextOrientations","unformattedText","spacingIfAllowed","addVerticalShapingForPointLabelIfNeeded","justifications","getAnchorJustification","singleLine","justification","imageMap","sdf","sdfIcons","getDefaultHorizontalShaping","showCollisionBoxes","defaultHorizontalShaping","textMaxBoxScale","iconBoxScale","symbolMinDistance","textPadding","iconPadding","getIconPadding","textMaxAngle","iconAlongLine","textRepeatDistance","iconTextFit","verticallyShapedIcon","addSymbolAtAnchor","lineArray","textCollisionFeature","iconCollisionFeature","verticalTextCollisionFeature","verticalIconCollisionFeature","placedTextSymbolIndices","verticalTextRotation","iconQuads","verticalIconQuads","sizeData","textRotate","addTextVertices","horizontalOnly","getCollisionCircleHeight","prevHeight","addTextVariableAnchorOffsets","addSymbol","clippedLines","clippedLine","p0","clipLine","anchorIsTooClose","poi","placementTypes","placedIconIndex","glyphQuads","textureRect","rectBuffer","rotateVerticalGlyph","halfAdvance","builtInOffset","verticalizedLabelOffset","textureScale","isDoubleResolution","verticalRotation","xHalfWidthOffsetCorrection","halfWidthOffsetCorrection","verticalOffsetCorrection","getGlyphQuads","placementType","horizontalShaping","repeatDistance","otherAnchors","GlyphAtlas","stackPositions","bitmap","getMercCoords","earthRadius","LngLat","lng","lat","wrap","toArray","distanceTo","lngLat","rad","lat1","lat2","lon","earthCircumfrence","circumferenceAtLatitude","latitude","latFromMercatorY","MercatorCoordinate","lngLatLike","altitude","mercatorZfromAltitude","toLngLat","toAltitude","meterInMercatorCoordinateUnits","CanonicalTileID","calculateKey","quadkey","mask","getQuadkey","isChildOf","dz","getTilePoint","UnwrappedTileID","OverscaledTileID","scaledTo","targetZ","zDifference","calculateScaledKey","withWrap","children","sourceMaxZoom","isLessThan","wrapped","unwrapTo","overscaleFactor","toUnwrapped","WorkerTile","collectResourceTiming","returnDependencies","inFlightDependencies","dependencySentinel","layerIndex","buckets","layerFamilies","family","recalculateLayers","iconMap","patternMap","maybePrepare","glyphAtlas","imageAtlas","glyphAtlasImage","PerformanceMarkers","RequestPerformance","_marks","measure","mark","resourceTimingData","getEntriesByName","clearMarks","clearMeasures","loadVectorTile","getArrayBuffer","cacheControl","expires","rawData","ex","errorMessage","messge","VectorTileWorkerSource","loadVectorData","fetching","loading","loaded","loadTile","perf","workerTile","resourceTiming","reloadTile","parseResult","abortTile","removeTile","DEMData","_idx","MIN_SAFE_INTEGER","ele","pixels","unpack","getUnpackVector","getPixels","backfillBorder","borderTile","xMin","xMax","yMin","yMax","RasterDEMTileWorkerSource","rawImageData","imagePixels","OffscreenCanvas","getContext","createImageBitmap","willReadFrequently","fillStyle","fillRect","getImageData","isOffscreenCanvasDistorted","VideoFrame","frame","timestamp","swapBR","copyTo","destRowOffset","sourceLeft","sourceTop","computeVideoFrameParameters","close","readImageUsingVideoFrame","imgBitmap","origWidth","origHeight","drawImage","imgData","clearRect","readImageDataUsingOffscreenCanvas","dem","rewindRings","outer","rewindRing","reverse","rewind","gj","geojson_wrapper","GeoJSONWrapper","FeatureWrapper","rawGeometry","tags","fromVectorTileJs","tile","writeLayer","writeTile","keycache","valuecache","writeFeature","writeValue","writeProperties","writeGeometry","keyIndex","valueIndex","command","zigzag","count","vtPbfModule","fromGeojsonVt","ARRAY_TYPES","KDBush","magic","versionAndType","ArrayType","nodeSize","numItems","IndexArrayType","arrayTypeIndex","coordsByteSize","idsByteSize","padCoords","_pos","_finished","numAdded","axis","qy","r2","sqDist","select","swapItem","defaultOptions","minPoints","props","fround","OFFSET_ID","OFFSET_NUM","OFFSET_PROP","Supercluster","assign","trees","clusterProps","load","time","timerId","lngX","latY","tree","_createTree","timeEnd","_cluster","getClusters","minLng","minLat","maxLng","maxLat","easternHem","westernHem","_limitZoom","clusters","getClusterJSON","getChildren","clusterId","originId","_getOriginId","originZoom","_getOriginZoom","errorMsg","getLeaves","limit","leaves","_appendLeaves","getTile","z2","_addTileFeatures","getClusterExpansionZoom","expansionZoom","cluster_id","skipped","point_count","isCluster","getClusterProperties","nextData","neighborIds","numPointsOrigin","numPoints","neighborId","wx","wy","clusterPropIndex","numPoints2","_map","original","yLat","abbrev","propIndex","point_count_abbreviated","simplify","sqTolerance","maxSqDist","minPosToMid","getSqSegDist","posToMid","createFeature","calcLineBBox","calcBBox","convertFeature","convertPoint","convertLine","convertLines","projectX","projectY","k2","minAll","maxAll","clipped","newGeometry","clipPoints","clipLines","newGeom","trackMetrics","segLen","newSlice","intersect","intersectX","intersectY","az","exited","addPoint","shiftFeatureCoords","newFeatures","shiftCoords","newPolygon","newPoints","transformTile","transformed","transformPoint","createTile","numSimplified","numFeatures","simplified","tileFeature","isOuter","GeoJSONVT","tileCoords","indexMaxZoom","indexMaxPoints","stats","total","merged","splitTile","toID","getFeatureId","isUpdateableGeoJSON","seenIds","toUpdateable","Map","cz","k3","k4","z0","GeoJSONWorkerSource","loadGeoJSON","_dataUpdateable","getJSON","dataDiff","updateable","diff","removeAll","delete","cloneProperties","removeAllProperties","removeProperties","_b","addOrUpdateProperties","_c","_d","applySourceDiff","loadGeoJSONTile","_geoJSONIndex","geoJSONTile","geojsonWrapper","_geojsonTileLayer","_feature","vtpbf","byteOffset","loadData","_pendingRequest","_pendingCallback","abandoned","superclusterOptions","mapExpressions","reduceExpressions","propertyNames","mapExpression","mapExpressionParsed","reduceExpressionParsed","pointProperties","getSuperclusterOptions","geojsonvt","geojsonVtOptions","removeSource","getClusterChildren","getClusterLeaves","Worker","layerIndexes","workerSourceTypes","workerSources","demWorkerSources","registerWorkerSource","WorkerSource","registerRTLTextPlugin","setReferrer","mapID","setImages","workerSource","ws","setLayers","getLayerIndex","updateLayers","loadDEMTile","getDEMWorkerSource","removeDEMTile","loadWorkerSource","importScripts","syncRTLPluginState","getAvailableImages"],"mappings":"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","x_google_ignoreList":[0,1,2,7,29,30,31,42,43,44,54,55,62,63,64,65,78,79,81,107,113,122,124,125,126,127,128,129,130,131,132,133,134,135]} \ No newline at end of file diff --git a/web/libraries/maplibre-gl/dist/maplibre-gl-csp.js b/web/libraries/maplibre-gl/dist/maplibre-gl-csp.js new file mode 100644 index 00000000..cddc9f31 --- /dev/null +++ b/web/libraries/maplibre-gl/dist/maplibre-gl-csp.js @@ -0,0 +1,2 @@ +!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?module.exports=e():"function"==typeof define&&define.amd?define(e):(t="undefined"!=typeof globalThis?globalThis:t||self).maplibregl=e()}(this,(function(){"use strict";var t="3.6.2";function e(t,e,i,r){return new(i||(i=Promise))((function(s,n){function a(t){try{l(r.next(t))}catch(t){n(t)}}function o(t){try{l(r.throw(t))}catch(t){n(t)}}function l(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(a,o)}l((r=r.apply(t,e||[])).next())}))}function i(t){return t&&t.__esModule&&Object.prototype.hasOwnProperty.call(t,"default")?t.default:t}"function"==typeof SuppressedError&&SuppressedError;var r=s;function s(t,e){this.x=t,this.y=e}s.prototype={clone:function(){return new s(this.x,this.y)},add:function(t){return this.clone()._add(t)},sub:function(t){return this.clone()._sub(t)},multByPoint:function(t){return this.clone()._multByPoint(t)},divByPoint:function(t){return this.clone()._divByPoint(t)},mult:function(t){return this.clone()._mult(t)},div:function(t){return this.clone()._div(t)},rotate:function(t){return this.clone()._rotate(t)},rotateAround:function(t,e){return this.clone()._rotateAround(t,e)},matMult:function(t){return this.clone()._matMult(t)},unit:function(){return this.clone()._unit()},perp:function(){return this.clone()._perp()},round:function(){return this.clone()._round()},mag:function(){return Math.sqrt(this.x*this.x+this.y*this.y)},equals:function(t){return this.x===t.x&&this.y===t.y},dist:function(t){return Math.sqrt(this.distSqr(t))},distSqr:function(t){var e=t.x-this.x,i=t.y-this.y;return e*e+i*i},angle:function(){return Math.atan2(this.y,this.x)},angleTo:function(t){return Math.atan2(this.y-t.y,this.x-t.x)},angleWith:function(t){return this.angleWithSep(t.x,t.y)},angleWithSep:function(t,e){return Math.atan2(this.x*e-this.y*t,this.x*t+this.y*e)},_matMult:function(t){var e=t[2]*this.x+t[3]*this.y;return this.x=t[0]*this.x+t[1]*this.y,this.y=e,this},_add:function(t){return this.x+=t.x,this.y+=t.y,this},_sub:function(t){return this.x-=t.x,this.y-=t.y,this},_mult:function(t){return this.x*=t,this.y*=t,this},_div:function(t){return this.x/=t,this.y/=t,this},_multByPoint:function(t){return this.x*=t.x,this.y*=t.y,this},_divByPoint:function(t){return this.x/=t.x,this.y/=t.y,this},_unit:function(){return this._div(this.mag()),this},_perp:function(){var t=this.y;return this.y=this.x,this.x=-t,this},_rotate:function(t){var e=Math.cos(t),i=Math.sin(t),r=i*this.x+e*this.y;return this.x=e*this.x-i*this.y,this.y=r,this},_rotateAround:function(t,e){var i=Math.cos(t),r=Math.sin(t),s=e.y+r*(this.x-e.x)+i*(this.y-e.y);return this.x=e.x+i*(this.x-e.x)-r*(this.y-e.y),this.y=s,this},_round:function(){return this.x=Math.round(this.x),this.y=Math.round(this.y),this}},s.convert=function(t){return t instanceof s?t:Array.isArray(t)?new s(t[0],t[1]):t};var n=i(r),a=o;function o(t,e,i,r){this.cx=3*t,this.bx=3*(i-t)-this.cx,this.ax=1-this.cx-this.bx,this.cy=3*e,this.by=3*(r-e)-this.cy,this.ay=1-this.cy-this.by,this.p1x=t,this.p1y=e,this.p2x=i,this.p2y=r}o.prototype={sampleCurveX:function(t){return((this.ax*t+this.bx)*t+this.cx)*t},sampleCurveY:function(t){return((this.ay*t+this.by)*t+this.cy)*t},sampleCurveDerivativeX:function(t){return(3*this.ax*t+2*this.bx)*t+this.cx},solveCurveX:function(t,e){if(void 0===e&&(e=1e-6),t<0)return 0;if(t>1)return 1;for(var i=t,r=0;r<8;r++){var s=this.sampleCurveX(i)-t;if(Math.abs(s)s?a=i:o=i,i=.5*(o-a)+a;return i},solve:function(t,e){return this.sampleCurveY(this.solveCurveX(t,e))}};var l=i(a);let c,h;function u(){return null==c&&(c="undefined"!=typeof OffscreenCanvas&&new OffscreenCanvas(1,1).getContext("2d")&&"function"==typeof createImageBitmap),c}function p(t,e,i,r){const s=new l(t,e,i,r);return function(t){return s.solve(t)}}const d=p(.25,.1,.25,1);function m(t,e,i){return Math.min(i,Math.max(e,t))}function f(t,e,i){const r=i-e,s=((t-e)%r+r)%r+e;return s===e?i:s}function _(t,e,i){if(!t.length)return i(null,[]);let r=t.length;const s=new Array(t.length);let n=null;t.forEach(((t,a)=>{e(t,((t,e)=>{t&&(n=t),s[a]=e,0==--r&&i(n,s)}))}))}function g(t,...e){for(const i of e)for(const e in i)t[e]=i[e];return t}function y(t,e){const i={};for(let r=0;r(e.y-t.y)*(i.x-t.x)}function C(t){let e=0;for(let i,r,s=0,n=t.length,a=n-1;scancelAnimationFrame(e)}},getImageData(t,e=0){return this.getImageCanvasContext(t).getImageData(-e,-e,t.width+2*e,t.height+2*e)},getImageCanvasContext(t){const e=window.document.createElement("canvas"),i=e.getContext("2d",{willReadFrequently:!0});if(!i)throw new Error("failed to create canvas 2d context");return e.width=t.width,e.height=t.height,i.drawImage(t,0,0,t.width,t.height),i},resolveURL:t=>(L||(L=document.createElement("a")),L.href=t,L.href),hardwareConcurrency:"undefined"!=typeof navigator&&navigator.hardwareConcurrency||4,get prefersReducedMotion(){return!!matchMedia&&(null==R&&(R=matchMedia("(prefers-reduced-motion: reduce)")),R.matches)}};class F{static testProp(t){if(!F.docStyle)return t[0];for(let e=0;e{window.removeEventListener("click",F.suppressClickInternal,!0)}),0)}static mousePos(t,e){const i=t.getBoundingClientRect();return new n(e.clientX-i.left-t.clientLeft,e.clientY-i.top-t.clientTop)}static touchPos(t,e){const i=t.getBoundingClientRect(),r=[];for(let s=0;sself.worker&&self.worker.referrer:()=>("blob:"===window.location.protocol?window.parent:window).location.href,N=t=>O.REGISTERED_PROTOCOLS[t.substring(0,t.indexOf("://"))];function $(t,e){const i=new AbortController,r=new Request(t.url,{method:t.method||"GET",body:t.body,credentials:t.credentials,headers:t.headers,cache:t.cache,referrer:U(),signal:i.signal});let s=!1,n=!1;"json"===t.type&&r.headers.set("Accept","application/json");return n||fetch(r).then((i=>i.ok?(i=>{("arrayBuffer"===t.type||"image"===t.type?i.arrayBuffer():"json"===t.type?i.json():i.text()).then((t=>{n||(s=!0,e(null,t,i.headers.get("Cache-Control"),i.headers.get("Expires")))})).catch((t=>{n||e(new Error(t.message))}))})(i):i.blob().then((r=>e(new V(i.status,i.statusText,t.url,r)))))).catch((t=>{20!==t.code&&e(new Error(t.message))})),{cancel:()=>{n=!0,s||i.abort()}}}const q=function(t,e){if(/:\/\//.test(t.url)&&!/^https?:|^file:/.test(t.url)){if(z()&&self.worker&&self.worker.actor)return self.worker.actor.send("getResource",t,e);if(!z())return(N(t.url)||$)(t,e)}if(!(/^file:/.test(i=t.url)||/^file:/.test(U())&&!/^\w+:/.test(i))){if(fetch&&Request&&AbortController&&Object.prototype.hasOwnProperty.call(Request.prototype,"signal"))return $(t,e);if(z()&&self.worker&&self.worker.actor)return self.worker.actor.send("getResource",t,e,void 0,!0)}var i;return function(t,e){const i=new XMLHttpRequest;i.open(t.method||"GET",t.url,!0),"arrayBuffer"!==t.type&&"image"!==t.type||(i.responseType="arraybuffer");for(const e in t.headers)i.setRequestHeader(e,t.headers[e]);return"json"===t.type&&(i.responseType="text",i.setRequestHeader("Accept","application/json")),i.withCredentials="include"===t.credentials,i.onerror=()=>{e(new Error(i.statusText))},i.onload=()=>{if((i.status>=200&&i.status<300||0===i.status)&&null!==i.response){let r=i.response;if("json"===t.type)try{r=JSON.parse(i.response)}catch(t){return e(t)}e(null,r,i.getResponseHeader("Cache-Control"),i.getResponseHeader("Expires"))}else{const r=new Blob([i.response],{type:i.getResponseHeader("Content-Type")});e(new V(i.status,i.statusText,t.url,r))}},i.send(t.body),{cancel:()=>i.abort()}}(t,e)},Z=function(t,e){return q(g(t,{type:"json"}),e)},j=function(t,e){return q(g(t,{type:"arrayBuffer"}),e)};function G(t){if(!t||t.indexOf("://")<=0||0===t.indexOf("data:image/")||0===t.indexOf("blob:"))return!0;const e=new URL(t),i=window.location;return e.protocol===i.protocol&&e.host===i.host}const H={supported:!1,testSupport:function(t){!K&&X&&(Y?J(t):W=t)}};let W,X,K=!1,Y=!1;function J(t){const e=t.createTexture();t.bindTexture(t.TEXTURE_2D,e);try{if(t.texImage2D(t.TEXTURE_2D,0,t.RGBA,t.RGBA,t.UNSIGNED_BYTE,X),t.isContextLost())return;H.supported=!0}catch(t){}t.deleteTexture(e),K=!0}var Q,tt;"undefined"!=typeof document&&(X=document.createElement("img"),X.onload=function(){W&&J(W),W=null,Y=!0},X.onerror=function(){K=!0,W=null},X.src=""),function(t){let e,i,r,s;t.resetRequestQueue=()=>{e=[],i=0,r=0,s={}},t.addThrottleControl=t=>{const e=r++;return s[e]=t,e},t.removeThrottleControl=t=>{delete s[t],o()},t.getImage=(t,r,s=!0)=>{H.supported&&(t.headers||(t.headers={}),t.headers.accept="image/webp,*/*");const n={requestParameters:t,supportImageRefresh:s,callback:r,cancelled:!1,completed:!1,cancel:()=>{n.completed||n.cancelled||(n.cancelled=!0,n.innerRequest&&(n.innerRequest.cancel(),i--),o())}};return e.push(n),o(),n};const n=t=>{const{requestParameters:e,supportImageRefresh:i,callback:r}=t;return g(e,{type:"image"}),(!1!==i||z()||N(e.url)||e.headers&&!Object.keys(e.headers).reduce(((t,e)=>t&&"accept"===e),!0)?q:l)(e,((e,i,s,n)=>{a(t,r,e,i,s,n)}))},a=(t,e,r,s,n,a)=>{r?e(r):s instanceof HTMLImageElement||k(s)?e(null,s):s&&((t,e)=>{"function"==typeof createImageBitmap?function(t,e){const i=new Blob([new Uint8Array(t)],{type:"image/png"});createImageBitmap(i).then((t=>{e(null,t)})).catch((t=>{e(new Error(`Could not load image because of ${t.message}. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.`))}))}(t,e):function(t,e){const i=new Image;i.onload=()=>{e(null,i),URL.revokeObjectURL(i.src),i.onload=null,window.requestAnimationFrame((()=>{i.src=D}))},i.onerror=()=>e(new Error("Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported."));const r=new Blob([new Uint8Array(t)],{type:"image/png"});i.src=t.byteLength?URL.createObjectURL(r):D}(t,e)})(s,((t,i)=>{null!=t?e(t):null!=i&&e(null,i,{cacheControl:n,expires:a})})),t.cancelled||(t.completed=!0,i--,o())},o=()=>{const t=(()=>{const t=Object.keys(s);let e=!1;if(t.length>0)for(const i of t)if(e=s[i](),e)break;return e})()?O.MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME:O.MAX_PARALLEL_IMAGE_REQUESTS;for(let r=i;r0;r++){const t=e.shift();if(t.cancelled){r--;continue}const s=n(t);i++,t.innerRequest=s}},l=(t,e)=>{const i=new Image,r=t.url;let s=!1;const n=t.credentials;return n&&"include"===n?i.crossOrigin="use-credentials":(n&&"same-origin"===n||!G(r))&&(i.crossOrigin="anonymous"),i.fetchPriority="high",i.onload=()=>{e(null,i),i.onerror=i.onload=null},i.onerror=()=>{s||e(new Error("Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.")),i.onerror=i.onload=null},i.src=r,{cancel:()=>{s=!0,i.src=""}}}}(Q||(Q={})),Q.resetRequestQueue(),function(t){t.Glyphs="Glyphs",t.Image="Image",t.Source="Source",t.SpriteImage="SpriteImage",t.SpriteJSON="SpriteJSON",t.Style="Style",t.Tile="Tile",t.Unknown="Unknown"}(tt||(tt={}));class et{constructor(t){this._transformRequestFn=t}transformRequest(t,e){return this._transformRequestFn&&this._transformRequestFn(t,e)||{url:t}}normalizeSpriteURL(t,e,i){const r=function(t){const e=t.match(it);if(!e)throw new Error(`Unable to parse URL "${t}"`);return{protocol:e[1],authority:e[2],path:e[3]||"/",params:e[4]?e[4].split("&"):[]}}(t);return r.path+=`${e}${i}`,function(t){const e=t.params.length?`?${t.params.join("&")}`:"";return`${t.protocol}://${t.authority}${t.path}${e}`}(r)}setTransformRequest(t){this._transformRequestFn=t}}const it=/^(\w+):\/\/([^/?]*)(\/[^?]+)?\??(.+)?/;function rt(t,e,i){i[t]&&-1!==i[t].indexOf(e)||(i[t]=i[t]||[],i[t].push(e))}function st(t,e,i){if(i&&i[t]){const r=i[t].indexOf(e);-1!==r&&i[t].splice(r,1)}}class nt{constructor(t,e={}){g(this,e),this.type=t}}class at extends nt{constructor(t,e={}){super("error",g({error:t},e))}}class ot{on(t,e){return this._listeners=this._listeners||{},rt(t,e,this._listeners),this}off(t,e){return st(t,e,this._listeners),st(t,e,this._oneTimeListeners),this}once(t,e){return e?(this._oneTimeListeners=this._oneTimeListeners||{},rt(t,e,this._oneTimeListeners),this):new Promise((e=>this.once(t,e)))}fire(t,e){"string"==typeof t&&(t=new nt(t,e||{}));const i=t.type;if(this.listens(i)){t.target=this;const e=this._listeners&&this._listeners[i]?this._listeners[i].slice():[];for(const i of e)i.call(this,t);const r=this._oneTimeListeners&&this._oneTimeListeners[i]?this._oneTimeListeners[i].slice():[];for(const e of r)st(i,e,this._oneTimeListeners),e.call(this,t);const s=this._eventedParent;s&&(g(t,"function"==typeof this._eventedParentData?this._eventedParentData():this._eventedParentData),s.fire(t))}else t instanceof at&&console.error(t.error);return this}listens(t){return this._listeners&&this._listeners[t]&&this._listeners[t].length>0||this._oneTimeListeners&&this._oneTimeListeners[t]&&this._oneTimeListeners[t].length>0||this._eventedParent&&this._eventedParent.listens(t)}setEventedParent(t,e){return this._eventedParent=t,this._eventedParentData=e,this}}var lt={$version:8,$root:{version:{required:!0,type:"enum",values:[8]},name:{type:"string"},metadata:{type:"*"},center:{type:"array",value:"number"},zoom:{type:"number"},bearing:{type:"number",default:0,period:360,units:"degrees"},pitch:{type:"number",default:0,units:"degrees"},light:{type:"light"},terrain:{type:"terrain"},sources:{required:!0,type:"sources"},sprite:{type:"sprite"},glyphs:{type:"string"},transition:{type:"transition"},layers:{required:!0,type:"array",value:"layer"}},sources:{"*":{type:"source"}},source:["source_vector","source_raster","source_raster_dem","source_geojson","source_video","source_image"],source_vector:{type:{required:!0,type:"enum",values:{vector:{}}},url:{type:"string"},tiles:{type:"array",value:"string"},bounds:{type:"array",value:"number",length:4,default:[-180,-85.051129,180,85.051129]},scheme:{type:"enum",values:{xyz:{},tms:{}},default:"xyz"},minzoom:{type:"number",default:0},maxzoom:{type:"number",default:22},attribution:{type:"string"},promoteId:{type:"promoteId"},volatile:{type:"boolean",default:!1},"*":{type:"*"}},source_raster:{type:{required:!0,type:"enum",values:{raster:{}}},url:{type:"string"},tiles:{type:"array",value:"string"},bounds:{type:"array",value:"number",length:4,default:[-180,-85.051129,180,85.051129]},minzoom:{type:"number",default:0},maxzoom:{type:"number",default:22},tileSize:{type:"number",default:512,units:"pixels"},scheme:{type:"enum",values:{xyz:{},tms:{}},default:"xyz"},attribution:{type:"string"},volatile:{type:"boolean",default:!1},"*":{type:"*"}},source_raster_dem:{type:{required:!0,type:"enum",values:{"raster-dem":{}}},url:{type:"string"},tiles:{type:"array",value:"string"},bounds:{type:"array",value:"number",length:4,default:[-180,-85.051129,180,85.051129]},minzoom:{type:"number",default:0},maxzoom:{type:"number",default:22},tileSize:{type:"number",default:512,units:"pixels"},attribution:{type:"string"},encoding:{type:"enum",values:{terrarium:{},mapbox:{},custom:{}},default:"mapbox"},redFactor:{type:"number",default:1},blueFactor:{type:"number",default:1},greenFactor:{type:"number",default:1},baseShift:{type:"number",default:0},volatile:{type:"boolean",default:!1},"*":{type:"*"}},source_geojson:{type:{required:!0,type:"enum",values:{geojson:{}}},data:{required:!0,type:"*"},maxzoom:{type:"number",default:18},attribution:{type:"string"},buffer:{type:"number",default:128,maximum:512,minimum:0},filter:{type:"*"},tolerance:{type:"number",default:.375},cluster:{type:"boolean",default:!1},clusterRadius:{type:"number",default:50,minimum:0},clusterMaxZoom:{type:"number"},clusterMinPoints:{type:"number"},clusterProperties:{type:"*"},lineMetrics:{type:"boolean",default:!1},generateId:{type:"boolean",default:!1},promoteId:{type:"promoteId"}},source_video:{type:{required:!0,type:"enum",values:{video:{}}},urls:{required:!0,type:"array",value:"string"},coordinates:{required:!0,type:"array",length:4,value:{type:"array",length:2,value:"number"}}},source_image:{type:{required:!0,type:"enum",values:{image:{}}},url:{required:!0,type:"string"},coordinates:{required:!0,type:"array",length:4,value:{type:"array",length:2,value:"number"}}},layer:{id:{type:"string",required:!0},type:{type:"enum",values:{fill:{},line:{},symbol:{},circle:{},heatmap:{},"fill-extrusion":{},raster:{},hillshade:{},background:{}},required:!0},metadata:{type:"*"},source:{type:"string"},"source-layer":{type:"string"},minzoom:{type:"number",minimum:0,maximum:24},maxzoom:{type:"number",minimum:0,maximum:24},filter:{type:"filter"},layout:{type:"layout"},paint:{type:"paint"}},layout:["layout_fill","layout_line","layout_circle","layout_heatmap","layout_fill-extrusion","layout_symbol","layout_raster","layout_hillshade","layout_background"],layout_background:{visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_fill:{"fill-sort-key":{type:"number",expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_circle:{"circle-sort-key":{type:"number",expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_heatmap:{visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},"layout_fill-extrusion":{visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_line:{"line-cap":{type:"enum",values:{butt:{},round:{},square:{}},default:"butt",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"line-join":{type:"enum",values:{bevel:{},round:{},miter:{}},default:"miter",expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"line-miter-limit":{type:"number",default:2,requires:[{"line-join":"miter"}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"line-round-limit":{type:"number",default:1.05,requires:[{"line-join":"round"}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"line-sort-key":{type:"number",expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_symbol:{"symbol-placement":{type:"enum",values:{point:{},line:{},"line-center":{}},default:"point",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"symbol-spacing":{type:"number",default:250,minimum:1,units:"pixels",requires:[{"symbol-placement":"line"}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"symbol-avoid-edges":{type:"boolean",default:!1,expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"symbol-sort-key":{type:"number",expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"symbol-z-order":{type:"enum",values:{auto:{},"viewport-y":{},source:{}},default:"auto",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-allow-overlap":{type:"boolean",default:!1,requires:["icon-image",{"!":"icon-overlap"}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-overlap":{type:"enum",values:{never:{},always:{},cooperative:{}},requires:["icon-image"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-ignore-placement":{type:"boolean",default:!1,requires:["icon-image"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-optional":{type:"boolean",default:!1,requires:["icon-image","text-field"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-rotation-alignment":{type:"enum",values:{map:{},viewport:{},auto:{}},default:"auto",requires:["icon-image"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-size":{type:"number",default:1,minimum:0,units:"factor of the original icon size",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"icon-text-fit":{type:"enum",values:{none:{},width:{},height:{},both:{}},default:"none",requires:["icon-image","text-field"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-text-fit-padding":{type:"array",value:"number",length:4,default:[0,0,0,0],units:"pixels",requires:["icon-image","text-field",{"icon-text-fit":["both","width","height"]}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"icon-image":{type:"resolvedImage",tokens:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"icon-rotate":{type:"number",default:0,period:360,units:"degrees",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"icon-padding":{type:"padding",default:[2],units:"pixels",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"icon-keep-upright":{type:"boolean",default:!1,requires:["icon-image",{"icon-rotation-alignment":"map"},{"symbol-placement":["line","line-center"]}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-offset":{type:"array",value:"number",length:2,default:[0,0],requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"icon-anchor":{type:"enum",values:{center:{},left:{},right:{},top:{},bottom:{},"top-left":{},"top-right":{},"bottom-left":{},"bottom-right":{}},default:"center",requires:["icon-image"],expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"icon-pitch-alignment":{type:"enum",values:{map:{},viewport:{},auto:{}},default:"auto",requires:["icon-image"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-pitch-alignment":{type:"enum",values:{map:{},viewport:{},auto:{}},default:"auto",requires:["text-field"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-rotation-alignment":{type:"enum",values:{map:{},viewport:{},"viewport-glyph":{},auto:{}},default:"auto",requires:["text-field"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-field":{type:"formatted",default:"",tokens:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-font":{type:"array",value:"string",default:["Open Sans Regular","Arial Unicode MS Regular"],requires:["text-field"],expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-size":{type:"number",default:16,minimum:0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-max-width":{type:"number",default:10,minimum:0,units:"ems",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-line-height":{type:"number",default:1.2,units:"ems",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"text-letter-spacing":{type:"number",default:0,units:"ems",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-justify":{type:"enum",values:{auto:{},left:{},center:{},right:{}},default:"center",requires:["text-field"],expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-radial-offset":{type:"number",units:"ems",default:0,requires:["text-field"],"property-type":"data-driven",expression:{interpolated:!0,parameters:["zoom","feature"]}},"text-variable-anchor":{type:"array",value:"enum",values:{center:{},left:{},right:{},top:{},bottom:{},"top-left":{},"top-right":{},"bottom-left":{},"bottom-right":{}},requires:["text-field",{"symbol-placement":["point"]}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-variable-anchor-offset":{type:"variableAnchorOffsetCollection",requires:["text-field",{"symbol-placement":["point"]}],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-anchor":{type:"enum",values:{center:{},left:{},right:{},top:{},bottom:{},"top-left":{},"top-right":{},"bottom-left":{},"bottom-right":{}},default:"center",requires:["text-field",{"!":"text-variable-anchor"}],expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-max-angle":{type:"number",default:45,units:"degrees",requires:["text-field",{"symbol-placement":["line","line-center"]}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"text-writing-mode":{type:"array",value:"enum",values:{horizontal:{},vertical:{}},requires:["text-field",{"symbol-placement":["point"]}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-rotate":{type:"number",default:0,period:360,units:"degrees",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-padding":{type:"number",default:2,minimum:0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"text-keep-upright":{type:"boolean",default:!0,requires:["text-field",{"text-rotation-alignment":"map"},{"symbol-placement":["line","line-center"]}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-transform":{type:"enum",values:{none:{},uppercase:{},lowercase:{}},default:"none",requires:["text-field"],expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-offset":{type:"array",value:"number",units:"ems",length:2,default:[0,0],requires:["text-field",{"!":"text-radial-offset"}],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-allow-overlap":{type:"boolean",default:!1,requires:["text-field",{"!":"text-overlap"}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-overlap":{type:"enum",values:{never:{},always:{},cooperative:{}},requires:["text-field"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-ignore-placement":{type:"boolean",default:!1,requires:["text-field"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-optional":{type:"boolean",default:!1,requires:["text-field","icon-image"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_raster:{visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_hillshade:{visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},filter:{type:"array",value:"*"},filter_operator:{type:"enum",values:{"==":{},"!=":{},">":{},">=":{},"<":{},"<=":{},in:{},"!in":{},all:{},any:{},none:{},has:{},"!has":{},within:{}}},geometry_type:{type:"enum",values:{Point:{},LineString:{},Polygon:{}}},function:{expression:{type:"expression"},stops:{type:"array",value:"function_stop"},base:{type:"number",default:1,minimum:0},property:{type:"string",default:"$zoom"},type:{type:"enum",values:{identity:{},exponential:{},interval:{},categorical:{}},default:"exponential"},colorSpace:{type:"enum",values:{rgb:{},lab:{},hcl:{}},default:"rgb"},default:{type:"*",required:!1}},function_stop:{type:"array",minimum:0,maximum:24,value:["number","color"],length:2},expression:{type:"array",value:"*",minimum:1},light:{anchor:{type:"enum",default:"viewport",values:{map:{},viewport:{}},"property-type":"data-constant",transition:!1,expression:{interpolated:!1,parameters:["zoom"]}},position:{type:"array",default:[1.15,210,30],length:3,value:"number","property-type":"data-constant",transition:!0,expression:{interpolated:!0,parameters:["zoom"]}},color:{type:"color","property-type":"data-constant",default:"#ffffff",expression:{interpolated:!0,parameters:["zoom"]},transition:!0},intensity:{type:"number","property-type":"data-constant",default:.5,minimum:0,maximum:1,expression:{interpolated:!0,parameters:["zoom"]},transition:!0}},terrain:{source:{type:"string",required:!0},exaggeration:{type:"number",minimum:0,default:1}},paint:["paint_fill","paint_line","paint_circle","paint_heatmap","paint_fill-extrusion","paint_symbol","paint_raster","paint_hillshade","paint_background"],paint_fill:{"fill-antialias":{type:"boolean",default:!0,expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"fill-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-color":{type:"color",default:"#000000",transition:!0,requires:[{"!":"fill-pattern"}],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-outline-color":{type:"color",transition:!0,requires:[{"!":"fill-pattern"},{"fill-antialias":!0}],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"fill-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["fill-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"fill-pattern":{type:"resolvedImage",transition:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"cross-faded-data-driven"}},"paint_fill-extrusion":{"fill-extrusion-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"fill-extrusion-color":{type:"color",default:"#000000",transition:!0,requires:[{"!":"fill-extrusion-pattern"}],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-extrusion-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"fill-extrusion-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["fill-extrusion-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"fill-extrusion-pattern":{type:"resolvedImage",transition:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"cross-faded-data-driven"},"fill-extrusion-height":{type:"number",default:0,minimum:0,units:"meters",transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-extrusion-base":{type:"number",default:0,minimum:0,units:"meters",transition:!0,requires:["fill-extrusion-height"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-extrusion-vertical-gradient":{type:"boolean",default:!0,transition:!1,expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"}},paint_line:{"line-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-color":{type:"color",default:"#000000",transition:!0,requires:[{"!":"line-pattern"}],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"line-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["line-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"line-width":{type:"number",default:1,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-gap-width":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-offset":{type:"number",default:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-blur":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-dasharray":{type:"array",value:"number",minimum:0,transition:!0,units:"line widths",requires:[{"!":"line-pattern"}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"cross-faded"},"line-pattern":{type:"resolvedImage",transition:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"cross-faded-data-driven"},"line-gradient":{type:"color",transition:!1,requires:[{"!":"line-dasharray"},{"!":"line-pattern"},{source:"geojson",has:{lineMetrics:!0}}],expression:{interpolated:!0,parameters:["line-progress"]},"property-type":"color-ramp"}},paint_circle:{"circle-radius":{type:"number",default:5,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-color":{type:"color",default:"#000000",transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-blur":{type:"number",default:0,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"circle-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["circle-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"circle-pitch-scale":{type:"enum",values:{map:{},viewport:{}},default:"map",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"circle-pitch-alignment":{type:"enum",values:{map:{},viewport:{}},default:"viewport",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"circle-stroke-width":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-stroke-color":{type:"color",default:"#000000",transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-stroke-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"}},paint_heatmap:{"heatmap-radius":{type:"number",default:30,minimum:1,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"heatmap-weight":{type:"number",default:1,minimum:0,transition:!1,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"heatmap-intensity":{type:"number",default:1,minimum:0,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"heatmap-color":{type:"color",default:["interpolate",["linear"],["heatmap-density"],0,"rgba(0, 0, 255, 0)",.1,"royalblue",.3,"cyan",.5,"lime",.7,"yellow",1,"red"],transition:!1,expression:{interpolated:!0,parameters:["heatmap-density"]},"property-type":"color-ramp"},"heatmap-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"}},paint_symbol:{"icon-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-color":{type:"color",default:"#000000",transition:!0,requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-halo-color":{type:"color",default:"rgba(0, 0, 0, 0)",transition:!0,requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-halo-width":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-halo-blur":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"icon-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["icon-image","icon-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-color":{type:"color",default:"#000000",transition:!0,overridable:!0,requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-halo-color":{type:"color",default:"rgba(0, 0, 0, 0)",transition:!0,requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-halo-width":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-halo-blur":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"text-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["text-field","text-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"}},paint_raster:{"raster-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-hue-rotate":{type:"number",default:0,period:360,transition:!0,units:"degrees",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-brightness-min":{type:"number",default:0,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-brightness-max":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-saturation":{type:"number",default:0,minimum:-1,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-contrast":{type:"number",default:0,minimum:-1,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-resampling":{type:"enum",values:{linear:{},nearest:{}},default:"linear",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"raster-fade-duration":{type:"number",default:300,minimum:0,transition:!1,units:"milliseconds",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"}},paint_hillshade:{"hillshade-illumination-direction":{type:"number",default:335,minimum:0,maximum:359,transition:!1,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-illumination-anchor":{type:"enum",values:{map:{},viewport:{}},default:"viewport",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-exaggeration":{type:"number",default:.5,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-shadow-color":{type:"color",default:"#000000",transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-highlight-color":{type:"color",default:"#FFFFFF",transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-accent-color":{type:"color",default:"#000000",transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"}},paint_background:{"background-color":{type:"color",default:"#000000",transition:!0,requires:[{"!":"background-pattern"}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"background-pattern":{type:"resolvedImage",transition:!0,expression:{interpolated:!1,parameters:["zoom"]},"property-type":"cross-faded"},"background-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"}},transition:{duration:{type:"number",default:300,minimum:0,units:"milliseconds"},delay:{type:"number",default:0,minimum:0,units:"milliseconds"}},"property-type":{"data-driven":{type:"property-type"},"cross-faded":{type:"property-type"},"cross-faded-data-driven":{type:"property-type"},"color-ramp":{type:"property-type"},"data-constant":{type:"property-type"},constant:{type:"property-type"}},promoteId:{"*":{type:"string"}}};const ct=["type","source","source-layer","minzoom","maxzoom","filter","layout"];function ht(t,e){const i={};for(const e in t)"ref"!==e&&(i[e]=t[e]);return ct.forEach((t=>{t in e&&(i[t]=e[t])})),i}function ut(t){t=t.slice();const e=Object.create(null);for(let i=0;i`:"value"===t.itemType.kind?"array":`array<${e}>`}return t.kind}const Vt=[It,At,Et,Ct,zt,Dt,Mt,Ft(Pt),Lt,Rt,Bt];function Ut(t,e){if("error"===e.kind)return null;if("array"===t.kind){if("array"===e.kind&&(0===e.N&&"value"===e.itemType.kind||!Ut(t.itemType,e.itemType))&&("number"!=typeof t.N||t.N===e.N))return null}else{if(t.kind===e.kind)return null;if("value"===t.kind)for(const t of Vt)if(!Ut(t,e))return null}return`Expected ${Ot(t)} but found ${Ot(e)} instead.`}function Nt(t,e){return e.some((e=>e.kind===t.kind))}function $t(t,e){return e.some((e=>"null"===e?null===t:"array"===e?Array.isArray(t):"object"===e?t&&!Array.isArray(t)&&"object"==typeof t:e===typeof t))}function qt(t,e){return"array"===t.kind&&"array"===e.kind?t.itemType.kind===e.itemType.kind&&"number"==typeof t.N:t.kind===e.kind}const Zt=.96422,jt=.82521,Gt=4/29,Ht=6/29,Wt=3*Ht*Ht,Xt=Ht*Ht*Ht,Kt=Math.PI/180,Yt=180/Math.PI;function Jt(t){return(t%=360)<0&&(t+=360),t}function Qt([t,e,i,r]){let s,n;const a=ee((.2225045*(t=te(t))+.7168786*(e=te(e))+.0606169*(i=te(i)))/1);t===e&&e===i?s=n=a:(s=ee((.4360747*t+.3850649*e+.1430804*i)/Zt),n=ee((.0139322*t+.0971045*e+.7141733*i)/jt));const o=116*a-16;return[o<0?0:o,500*(s-a),200*(a-n),r]}function te(t){return t<=.04045?t/12.92:Math.pow((t+.055)/1.055,2.4)}function ee(t){return t>Xt?Math.pow(t,1/3):t/Wt+Gt}function ie([t,e,i,r]){let s=(t+16)/116,n=isNaN(e)?s:s+e/500,a=isNaN(i)?s:s-i/200;return s=1*se(s),n=Zt*se(n),a=jt*se(a),[re(3.1338561*n-1.6168667*s-.4906146*a),re(-.9787684*n+1.9161415*s+.033454*a),re(.0719453*n-.2289914*s+1.4052427*a),r]}function re(t){return(t=t<=.00304?12.92*t:1.055*Math.pow(t,1/2.4)-.055)<0?0:t>1?1:t}function se(t){return t>Ht?t*t*t:Wt*(t-Gt)}function ne(t){return parseInt(t.padEnd(2,t),16)/255}function ae(t,e){return oe(e?t/100:t,0,1)}function oe(t,e,i){return Math.min(Math.max(e,t),i)}function le(t){return!t.some(Number.isNaN)}const ce={aliceblue:[240,248,255],antiquewhite:[250,235,215],aqua:[0,255,255],aquamarine:[127,255,212],azure:[240,255,255],beige:[245,245,220],bisque:[255,228,196],black:[0,0,0],blanchedalmond:[255,235,205],blue:[0,0,255],blueviolet:[138,43,226],brown:[165,42,42],burlywood:[222,184,135],cadetblue:[95,158,160],chartreuse:[127,255,0],chocolate:[210,105,30],coral:[255,127,80],cornflowerblue:[100,149,237],cornsilk:[255,248,220],crimson:[220,20,60],cyan:[0,255,255],darkblue:[0,0,139],darkcyan:[0,139,139],darkgoldenrod:[184,134,11],darkgray:[169,169,169],darkgreen:[0,100,0],darkgrey:[169,169,169],darkkhaki:[189,183,107],darkmagenta:[139,0,139],darkolivegreen:[85,107,47],darkorange:[255,140,0],darkorchid:[153,50,204],darkred:[139,0,0],darksalmon:[233,150,122],darkseagreen:[143,188,143],darkslateblue:[72,61,139],darkslategray:[47,79,79],darkslategrey:[47,79,79],darkturquoise:[0,206,209],darkviolet:[148,0,211],deeppink:[255,20,147],deepskyblue:[0,191,255],dimgray:[105,105,105],dimgrey:[105,105,105],dodgerblue:[30,144,255],firebrick:[178,34,34],floralwhite:[255,250,240],forestgreen:[34,139,34],fuchsia:[255,0,255],gainsboro:[220,220,220],ghostwhite:[248,248,255],gold:[255,215,0],goldenrod:[218,165,32],gray:[128,128,128],green:[0,128,0],greenyellow:[173,255,47],grey:[128,128,128],honeydew:[240,255,240],hotpink:[255,105,180],indianred:[205,92,92],indigo:[75,0,130],ivory:[255,255,240],khaki:[240,230,140],lavender:[230,230,250],lavenderblush:[255,240,245],lawngreen:[124,252,0],lemonchiffon:[255,250,205],lightblue:[173,216,230],lightcoral:[240,128,128],lightcyan:[224,255,255],lightgoldenrodyellow:[250,250,210],lightgray:[211,211,211],lightgreen:[144,238,144],lightgrey:[211,211,211],lightpink:[255,182,193],lightsalmon:[255,160,122],lightseagreen:[32,178,170],lightskyblue:[135,206,250],lightslategray:[119,136,153],lightslategrey:[119,136,153],lightsteelblue:[176,196,222],lightyellow:[255,255,224],lime:[0,255,0],limegreen:[50,205,50],linen:[250,240,230],magenta:[255,0,255],maroon:[128,0,0],mediumaquamarine:[102,205,170],mediumblue:[0,0,205],mediumorchid:[186,85,211],mediumpurple:[147,112,219],mediumseagreen:[60,179,113],mediumslateblue:[123,104,238],mediumspringgreen:[0,250,154],mediumturquoise:[72,209,204],mediumvioletred:[199,21,133],midnightblue:[25,25,112],mintcream:[245,255,250],mistyrose:[255,228,225],moccasin:[255,228,181],navajowhite:[255,222,173],navy:[0,0,128],oldlace:[253,245,230],olive:[128,128,0],olivedrab:[107,142,35],orange:[255,165,0],orangered:[255,69,0],orchid:[218,112,214],palegoldenrod:[238,232,170],palegreen:[152,251,152],paleturquoise:[175,238,238],palevioletred:[219,112,147],papayawhip:[255,239,213],peachpuff:[255,218,185],peru:[205,133,63],pink:[255,192,203],plum:[221,160,221],powderblue:[176,224,230],purple:[128,0,128],rebeccapurple:[102,51,153],red:[255,0,0],rosybrown:[188,143,143],royalblue:[65,105,225],saddlebrown:[139,69,19],salmon:[250,128,114],sandybrown:[244,164,96],seagreen:[46,139,87],seashell:[255,245,238],sienna:[160,82,45],silver:[192,192,192],skyblue:[135,206,235],slateblue:[106,90,205],slategray:[112,128,144],slategrey:[112,128,144],snow:[255,250,250],springgreen:[0,255,127],steelblue:[70,130,180],tan:[210,180,140],teal:[0,128,128],thistle:[216,191,216],tomato:[255,99,71],turquoise:[64,224,208],violet:[238,130,238],wheat:[245,222,179],white:[255,255,255],whitesmoke:[245,245,245],yellow:[255,255,0],yellowgreen:[154,205,50]};class he{constructor(t,e,i,r=1,s=!0){this.r=t,this.g=e,this.b=i,this.a=r,s||(this.r*=r,this.g*=r,this.b*=r,r||this.overwriteGetter("rgb",[t,e,i,r]))}static parse(t){if(t instanceof he)return t;if("string"!=typeof t)return;const e=function(t){if("transparent"===(t=t.toLowerCase().trim()))return[0,0,0,0];const e=ce[t];if(e){const[t,i,r]=e;return[t/255,i/255,r/255,1]}if(t.startsWith("#")&&/^#(?:[0-9a-f]{3,4}|[0-9a-f]{6}|[0-9a-f]{8})$/.test(t)){const e=t.length<6?1:2;let i=1;return[ne(t.slice(i,i+=e)),ne(t.slice(i,i+=e)),ne(t.slice(i,i+=e)),ne(t.slice(i,i+e)||"ff")]}if(t.startsWith("rgb")){const e=t.match(/^rgba?\(\s*([\de.+-]+)(%)?(?:\s+|\s*(,)\s*)([\de.+-]+)(%)?(?:\s+|\s*(,)\s*)([\de.+-]+)(%)?(?:\s*([,\/])\s*([\de.+-]+)(%)?)?\s*\)$/);if(e){const[t,i,r,s,n,a,o,l,c,h,u,p]=e,d=[s||" ",o||" ",h].join("");if(" "===d||" /"===d||",,"===d||",,,"===d){const t=[r,a,c].join(""),e="%%%"===t?100:""===t?255:0;if(e){const t=[oe(+i/e,0,1),oe(+n/e,0,1),oe(+l/e,0,1),u?ae(+u,p):1];if(le(t))return t}}return}}const i=t.match(/^hsla?\(\s*([\de.+-]+)(?:deg)?(?:\s+|\s*(,)\s*)([\de.+-]+)%(?:\s+|\s*(,)\s*)([\de.+-]+)%(?:\s*([,\/])\s*([\de.+-]+)(%)?)?\s*\)$/);if(i){const[t,e,r,s,n,a,o,l,c]=i,h=[r||" ",n||" ",o].join("");if(" "===h||" /"===h||",,"===h||",,,"===h){const t=[+e,oe(+s,0,100),oe(+a,0,100),l?ae(+l,c):1];if(le(t))return function([t,e,i,r]){function s(r){const s=(r+t/30)%12,n=e*Math.min(i,1-i);return i-n*Math.max(-1,Math.min(s-3,9-s,1))}return t=Jt(t),e/=100,i/=100,[s(0),s(8),s(4),r]}(t)}}}(t);return e?new he(...e,!1):void 0}get rgb(){const{r:t,g:e,b:i,a:r}=this,s=r||1/0;return this.overwriteGetter("rgb",[t/s,e/s,i/s,r])}get hcl(){return this.overwriteGetter("hcl",function(t){const[e,i,r,s]=Qt(t),n=Math.sqrt(i*i+r*r);return[Math.round(1e4*n)?Jt(Math.atan2(r,i)*Yt):NaN,n,e,s]}(this.rgb))}get lab(){return this.overwriteGetter("lab",Qt(this.rgb))}overwriteGetter(t,e){return Object.defineProperty(this,t,{value:e}),e}toString(){const[t,e,i,r]=this.rgb;return`rgba(${[t,e,i].map((t=>Math.round(255*t))).join(",")},${r})`}}he.black=new he(0,0,0,1),he.white=new he(1,1,1,1),he.transparent=new he(0,0,0,0),he.red=new he(1,0,0,1);class ue{constructor(t,e,i){this.sensitivity=t?e?"variant":"case":e?"accent":"base",this.locale=i,this.collator=new Intl.Collator(this.locale?this.locale:[],{sensitivity:this.sensitivity,usage:"search"})}compare(t,e){return this.collator.compare(t,e)}resolvedLocale(){return new Intl.Collator(this.locale?this.locale:[]).resolvedOptions().locale}}class pe{constructor(t,e,i,r,s){this.text=t,this.image=e,this.scale=i,this.fontStack=r,this.textColor=s}}class de{constructor(t){this.sections=t}static fromString(t){return new de([new pe(t,null,null,null,null)])}isEmpty(){return 0===this.sections.length||!this.sections.some((t=>0!==t.text.length||t.image&&0!==t.image.name.length))}static factory(t){return t instanceof de?t:de.fromString(t)}toString(){return 0===this.sections.length?"":this.sections.map((t=>t.text)).join("")}}class me{constructor(t){this.values=t.slice()}static parse(t){if(t instanceof me)return t;if("number"==typeof t)return new me([t,t,t,t]);if(Array.isArray(t)&&!(t.length<1||t.length>4)){for(const e of t)if("number"!=typeof e)return;switch(t.length){case 1:t=[t[0],t[0],t[0],t[0]];break;case 2:t=[t[0],t[1],t[0],t[1]];break;case 3:t=[t[0],t[1],t[2],t[1]]}return new me(t)}}toString(){return JSON.stringify(this.values)}}const fe=new Set(["center","left","right","top","bottom","top-left","top-right","bottom-left","bottom-right"]);class _e{constructor(t){this.values=t.slice()}static parse(t){if(t instanceof _e)return t;if(Array.isArray(t)&&!(t.length<1)&&t.length%2==0){for(let e=0;e=0&&t<=255&&"number"==typeof e&&e>=0&&e<=255&&"number"==typeof i&&i>=0&&i<=255?void 0===r||"number"==typeof r&&r>=0&&r<=1?null:`Invalid rgba value [${[t,e,i,r].join(", ")}]: 'a' must be between 0 and 1.`:`Invalid rgba value [${("number"==typeof r?[t,e,i,r]:[t,e,i]).join(", ")}]: 'r', 'g', and 'b' must be between 0 and 255.`}function xe(t){if(null===t||"string"==typeof t||"boolean"==typeof t||"number"==typeof t||t instanceof he||t instanceof ue||t instanceof de||t instanceof me||t instanceof _e||t instanceof ge)return!0;if(Array.isArray(t)){for(const e of t)if(!xe(e))return!1;return!0}if("object"==typeof t){for(const e in t)if(!xe(t[e]))return!1;return!0}return!1}function ve(t){if(null===t)return It;if("string"==typeof t)return Et;if("boolean"==typeof t)return Ct;if("number"==typeof t)return At;if(t instanceof he)return zt;if(t instanceof ue)return kt;if(t instanceof de)return Dt;if(t instanceof me)return Lt;if(t instanceof _e)return Bt;if(t instanceof ge)return Rt;if(Array.isArray(t)){const e=t.length;let i;for(const e of t){const t=ve(e);if(i){if(i===t)continue;i=Pt;break}i=t}return Ft(i||Pt,e)}return Mt}function be(t){const e=typeof t;return null===t?"":"string"===e||"number"===e||"boolean"===e?String(t):t instanceof he||t instanceof de||t instanceof me||t instanceof _e||t instanceof ge?t.toString():JSON.stringify(t)}class we{constructor(t,e){this.type=t,this.value=e}static parse(t,e){if(2!==t.length)return e.error(`'literal' expression requires exactly one argument, but found ${t.length-1} instead.`);if(!xe(t[1]))return e.error("invalid value");const i=t[1];let r=ve(i);const s=e.expectedType;return"array"!==r.kind||0!==r.N||!s||"array"!==s.kind||"number"==typeof s.N&&0!==s.N||(r=s),new we(r,i)}evaluate(){return this.value}eachChild(){}outputDefined(){return!0}}class Te{constructor(t){this.name="ExpressionEvaluationError",this.message=t}toJSON(){return this.message}}const Se={string:Et,number:At,boolean:Ct,object:Mt};class Ie{constructor(t,e){this.type=t,this.args=e}static parse(t,e){if(t.length<2)return e.error("Expected at least one argument.");let i,r=1;const s=t[0];if("array"===s){let s,n;if(t.length>2){const i=t[1];if("string"!=typeof i||!(i in Se)||"object"===i)return e.error('The item type argument of "array" must be one of string, number, boolean',1);s=Se[i],r++}else s=Pt;if(t.length>3){if(null!==t[2]&&("number"!=typeof t[2]||t[2]<0||t[2]!==Math.floor(t[2])))return e.error('The length argument to "array" must be a positive integer literal',2);n=t[2],r++}i=Ft(s,n)}else{if(!Se[s])throw new Error(`Types doesn't contain name = ${s}`);i=Se[s]}const n=[];for(;rt.outputDefined()))}}const Ae={"to-boolean":Ct,"to-color":zt,"to-number":At,"to-string":Et};class Ee{constructor(t,e){this.type=t,this.args=e}static parse(t,e){if(t.length<2)return e.error("Expected at least one argument.");const i=t[0];if(!Ae[i])throw new Error(`Can't parse ${i} as it is not part of the known types`);if(("to-boolean"===i||"to-string"===i)&&2!==t.length)return e.error("Expected one argument.");const r=Ae[i],s=[];for(let i=1;i4?`Invalid rbga value ${JSON.stringify(e)}: expected an array containing either three or four numeric values.`:ye(e[0],e[1],e[2],e[3]),!i))return new he(e[0]/255,e[1]/255,e[2]/255,e[3])}throw new Te(i||`Could not parse color from value '${"string"==typeof e?e:JSON.stringify(e)}'`)}case"padding":{let e;for(const i of this.args){e=i.evaluate(t);const r=me.parse(e);if(r)return r}throw new Te(`Could not parse padding from value '${"string"==typeof e?e:JSON.stringify(e)}'`)}case"variableAnchorOffsetCollection":{let e;for(const i of this.args){e=i.evaluate(t);const r=_e.parse(e);if(r)return r}throw new Te(`Could not parse variableAnchorOffsetCollection from value '${"string"==typeof e?e:JSON.stringify(e)}'`)}case"number":{let e=null;for(const i of this.args){if(e=i.evaluate(t),null===e)return 0;const r=Number(e);if(!isNaN(r))return r}throw new Te(`Could not convert ${JSON.stringify(e)} to number.`)}case"formatted":return de.fromString(be(this.args[0].evaluate(t)));case"resolvedImage":return ge.fromString(be(this.args[0].evaluate(t)));default:return be(this.args[0].evaluate(t))}}eachChild(t){this.args.forEach(t)}outputDefined(){return this.args.every((t=>t.outputDefined()))}}const Ce=["Unknown","Point","LineString","Polygon"];class ze{constructor(){this.globals=null,this.feature=null,this.featureState=null,this.formattedSection=null,this._parseColorCache={},this.availableImages=null,this.canonical=null}id(){return this.feature&&"id"in this.feature?this.feature.id:null}geometryType(){return this.feature?"number"==typeof this.feature.type?Ce[this.feature.type]:this.feature.type:null}geometry(){return this.feature&&"geometry"in this.feature?this.feature.geometry:null}canonicalID(){return this.canonical}properties(){return this.feature&&this.feature.properties||{}}parseColor(t){let e=this._parseColorCache[t];return e||(e=this._parseColorCache[t]=he.parse(t)),e}}class Me{constructor(t,e,i=[],r,s=new St,n=[]){this.registry=t,this.path=i,this.key=i.map((t=>`[${t}]`)).join(""),this.scope=s,this.errors=n,this.expectedType=r,this._isConstant=e}parse(t,e,i,r,s={}){return e?this.concat(e,i,r)._parse(t,s):this._parse(t,s)}_parse(t,e){function i(t,e,i){return"assert"===i?new Ie(e,[t]):"coerce"===i?new Ee(e,[t]):t}if(null!==t&&"string"!=typeof t&&"boolean"!=typeof t&&"number"!=typeof t||(t=["literal",t]),Array.isArray(t)){if(0===t.length)return this.error('Expected an array with at least one element. If you wanted a literal array, use ["literal", []].');const r=t[0];if("string"!=typeof r)return this.error(`Expression name must be a string, but found ${typeof r} instead. If you wanted a literal array, use ["literal", [...]].`,0),null;const s=this.registry[r];if(s){let r=s.parse(t,this);if(!r)return null;if(this.expectedType){const t=this.expectedType,s=r.type;if("string"!==t.kind&&"number"!==t.kind&&"boolean"!==t.kind&&"object"!==t.kind&&"array"!==t.kind||"value"!==s.kind)if("color"!==t.kind&&"formatted"!==t.kind&&"resolvedImage"!==t.kind||"value"!==s.kind&&"string"!==s.kind)if("padding"!==t.kind||"value"!==s.kind&&"number"!==s.kind&&"array"!==s.kind)if("variableAnchorOffsetCollection"!==t.kind||"value"!==s.kind&&"array"!==s.kind){if(this.checkSubtype(t,s))return null}else r=i(r,t,e.typeAnnotation||"coerce");else r=i(r,t,e.typeAnnotation||"coerce");else r=i(r,t,e.typeAnnotation||"coerce");else r=i(r,t,e.typeAnnotation||"assert")}if(!(r instanceof we)&&"resolvedImage"!==r.type.kind&&this._isConstant(r)){const t=new ze;try{r=new we(r.type,r.evaluate(t))}catch(t){return this.error(t.message),null}}return r}return this.error(`Unknown expression "${r}". If you wanted a literal array, use ["literal", [...]].`,0)}return this.error(void 0===t?"'undefined' value invalid. Use null instead.":"object"==typeof t?'Bare objects invalid. Use ["literal", {...}] instead.':`Expected an array, but found ${typeof t} instead.`)}concat(t,e,i){const r="number"==typeof t?this.path.concat(t):this.path,s=i?this.scope.concat(i):this.scope;return new Me(this.registry,this._isConstant,r,e||null,s,this.errors)}error(t,...e){const i=`${this.key}${e.map((t=>`[${t}]`)).join("")}`;this.errors.push(new Tt(i,t))}checkSubtype(t,e){const i=Ut(t,e);return i&&this.error(i),i}}class Pe{constructor(t,e,i){this.type=kt,this.locale=i,this.caseSensitive=t,this.diacriticSensitive=e}static parse(t,e){if(2!==t.length)return e.error("Expected one argument.");const i=t[1];if("object"!=typeof i||Array.isArray(i))return e.error("Collator options argument must be an object.");const r=e.parse(void 0!==i["case-sensitive"]&&i["case-sensitive"],1,Ct);if(!r)return null;const s=e.parse(void 0!==i["diacritic-sensitive"]&&i["diacritic-sensitive"],1,Ct);if(!s)return null;let n=null;return i.locale&&(n=e.parse(i.locale,1,Et),!n)?null:new Pe(r,s,n)}evaluate(t){return new ue(this.caseSensitive.evaluate(t),this.diacriticSensitive.evaluate(t),this.locale?this.locale.evaluate(t):null)}eachChild(t){t(this.caseSensitive),t(this.diacriticSensitive),this.locale&&t(this.locale)}outputDefined(){return!1}}const ke=8192;function De(t,e){t[0]=Math.min(t[0],e[0]),t[1]=Math.min(t[1],e[1]),t[2]=Math.max(t[2],e[0]),t[3]=Math.max(t[3],e[1])}function Le(t,e){return!(t[0]<=e[0]||t[2]>=e[2]||t[1]<=e[1]||t[3]>=e[3])}function Re(t,e){const i=(180+t[0])/360,r=(180-180/Math.PI*Math.log(Math.tan(Math.PI/4+t[1]*Math.PI/360)))/360,s=Math.pow(2,e.z);return[Math.round(i*s*ke),Math.round(r*s*ke)]}function Be(t,e,i){const r=t[0]-e[0],s=t[1]-e[1],n=t[0]-i[0],a=t[1]-i[1];return r*a-n*s==0&&r*n<=0&&s*a<=0}function Fe(t,e){let i=!1;for(let a=0,o=e.length;a(r=t)[1]!=(n=o[e+1])[1]>r[1]&&r[0]<(n[0]-s[0])*(r[1]-s[1])/(n[1]-s[1])+s[0]&&(i=!i)}}var r,s,n;return i}function Oe(t,e){for(let i=0;i0&&o<0||a<0&&o>0}function Ue(t,e,i){for(const c of i)for(let i=0;ii[2]){const e=.5*r;let s=t[0]-i[0]>e?-r:i[0]-t[0]>e?r:0;0===s&&(s=t[0]-i[2]>e?-r:i[2]-t[0]>e?r:0),t[0]+=s}De(e,t)}function Ge(t,e,i,r){const s=Math.pow(2,r.z)*ke,n=[r.x*ke,r.y*ke],a=[];for(const r of t)for(const t of r){const r=[t.x+n[0],t.y+n[1]];je(r,e,i,s),a.push(r)}return a}function He(t,e,i,r){const s=Math.pow(2,r.z)*ke,n=[r.x*ke,r.y*ke],a=[];for(const i of t){const t=[];for(const r of i){const i=[r.x+n[0],r.y+n[1]];De(e,i),t.push(i)}a.push(t)}if(e[2]-e[0]<=s/2){(o=e)[0]=o[1]=1/0,o[2]=o[3]=-1/0;for(const t of a)for(const r of t)je(r,e,i,s)}var o;return a}class We{constructor(t,e){this.type=Ct,this.geojson=t,this.geometries=e}static parse(t,e){if(2!==t.length)return e.error(`'within' expression requires exactly one argument, but found ${t.length-1} instead.`);if(xe(t[1])){const e=t[1];if("FeatureCollection"===e.type)for(let t=0;t!Array.isArray(e)||e.length===t.length-1));let o=null;for(const[r,n]of a){o=new Me(e.registry,Ye,e.path,null,e.scope);const a=[];let l=!1;for(let e=1;e{return e=t,Array.isArray(e)?`(${e.map(Ot).join(", ")})`:`(${Ot(e.type)}...)`;var e})).join(" | "),r=[];for(let i=1;i{i=e?i&&Ye(t):i&&t instanceof we})),!!i&&Je(t)&&ti(t,["zoom","heatmap-density","line-progress","accumulated","is-supported-script"])}function Je(t){if(t instanceof Ke){if("get"===t.name&&1===t.args.length)return!1;if("feature-state"===t.name)return!1;if("has"===t.name&&1===t.args.length)return!1;if("properties"===t.name||"geometry-type"===t.name||"id"===t.name)return!1;if(/^filter-/.test(t.name))return!1}if(t instanceof We)return!1;let e=!0;return t.eachChild((t=>{e&&!Je(t)&&(e=!1)})),e}function Qe(t){if(t instanceof Ke&&"feature-state"===t.name)return!1;let e=!0;return t.eachChild((t=>{e&&!Qe(t)&&(e=!1)})),e}function ti(t,e){if(t instanceof Ke&&e.indexOf(t.name)>=0)return!1;let i=!0;return t.eachChild((t=>{i&&!ti(t,e)&&(i=!1)})),i}function ei(t,e){const i=t.length-1;let r,s,n=0,a=i,o=0;for(;n<=a;)if(o=Math.floor((n+a)/2),r=t[o],s=t[o+1],r<=e){if(o===i||ee))throw new Te("Input is not a number.");a=o-1}return 0}class ii{constructor(t,e,i){this.type=t,this.input=e,this.labels=[],this.outputs=[];for(const[t,e]of i)this.labels.push(t),this.outputs.push(e)}static parse(t,e){if(t.length-1<4)return e.error(`Expected at least 4 arguments, but found only ${t.length-1}.`);if((t.length-1)%2!=0)return e.error("Expected an even number of arguments.");const i=e.parse(t[1],1,At);if(!i)return null;const r=[];let s=null;e.expectedType&&"value"!==e.expectedType.kind&&(s=e.expectedType);for(let i=1;i=n)return e.error('Input/output pairs for "step" expressions must be arranged with input values in strictly ascending order.',o);const c=e.parse(a,l,s);if(!c)return null;s=s||c.type,r.push([n,c])}return new ii(s,i,r)}evaluate(t){const e=this.labels,i=this.outputs;if(1===e.length)return i[0].evaluate(t);const r=this.input.evaluate(t);if(r<=e[0])return i[0].evaluate(t);const s=e.length;return r>=e[s-1]?i[s-1].evaluate(t):i[ei(e,r)].evaluate(t)}eachChild(t){t(this.input);for(const e of this.outputs)t(e)}outputDefined(){return this.outputs.every((t=>t.outputDefined()))}}function ri(t,e,i){return t+i*(e-t)}function si(t,e,i){return t.map(((t,r)=>ri(t,e[r],i)))}const ni={number:ri,color:function(t,e,i,r="rgb"){switch(r){case"rgb":{const[r,s,n,a]=si(t.rgb,e.rgb,i);return new he(r,s,n,a,!1)}case"hcl":{const[r,s,n,a]=t.hcl,[o,l,c,h]=e.hcl;let u,p;if(isNaN(r)||isNaN(o))isNaN(r)?isNaN(o)?u=NaN:(u=o,1!==n&&0!==n||(p=l)):(u=r,1!==c&&0!==c||(p=s));else{let t=o-r;o>r&&t>180?t-=360:o180&&(t+=360),u=r+i*t}const[d,m,f,_]=function([t,e,i,r]){return t=isNaN(t)?0:t*Kt,ie([i,Math.cos(t)*e,Math.sin(t)*e,r])}([u,null!=p?p:ri(s,l,i),ri(n,c,i),ri(a,h,i)]);return new he(d,m,f,_,!1)}case"lab":{const[r,s,n,a]=ie(si(t.lab,e.lab,i));return new he(r,s,n,a,!1)}}},array:si,padding:function(t,e,i){return new me(si(t.values,e.values,i))},variableAnchorOffsetCollection:function(t,e,i){const r=t.values,s=e.values;if(r.length!==s.length)throw new Te(`Cannot interpolate values of different length. from: ${t.toString()}, to: ${e.toString()}`);const n=[];for(let t=0;t"number"!=typeof t||t<0||t>1)))return e.error("Cubic bezier interpolation requires four numeric arguments with values between 0 and 1.",1);r={name:"cubic-bezier",controlPoints:t}}}if(t.length-1<4)return e.error(`Expected at least 4 arguments, but found only ${t.length-1}.`);if((t.length-1)%2!=0)return e.error("Expected an even number of arguments.");if(s=e.parse(s,2,At),!s)return null;const a=[];let o=null;"interpolate-hcl"===i||"interpolate-lab"===i?o=zt:e.expectedType&&"value"!==e.expectedType.kind&&(o=e.expectedType);for(let t=0;t=i)return e.error('Input/output pairs for "interpolate" expressions must be arranged with input values in strictly ascending order.',s);const c=e.parse(r,l,o);if(!c)return null;o=o||c.type,a.push([i,c])}return qt(o,At)||qt(o,zt)||qt(o,Lt)||qt(o,Bt)||qt(o,Ft(At))?new ai(o,i,r,s,a):e.error(`Type ${Ot(o)} is not interpolatable.`)}evaluate(t){const e=this.labels,i=this.outputs;if(1===e.length)return i[0].evaluate(t);const r=this.input.evaluate(t);if(r<=e[0])return i[0].evaluate(t);const s=e.length;if(r>=e[s-1])return i[s-1].evaluate(t);const n=ei(e,r),a=ai.interpolationFactor(this.interpolation,r,e[n],e[n+1]),o=i[n].evaluate(t),l=i[n+1].evaluate(t);switch(this.operator){case"interpolate":return ni[this.type.kind](o,l,a);case"interpolate-hcl":return ni.color(o,l,a,"hcl");case"interpolate-lab":return ni.color(o,l,a,"lab")}}eachChild(t){t(this.input);for(const e of this.outputs)t(e)}outputDefined(){return this.outputs.every((t=>t.outputDefined()))}}function oi(t,e,i,r){const s=r-i,n=t-i;return 0===s?0:1===e?n/s:(Math.pow(e,n)-1)/(Math.pow(e,s)-1)}class li{constructor(t,e){this.type=t,this.args=e}static parse(t,e){if(t.length<2)return e.error("Expectected at least one argument.");let i=null;const r=e.expectedType;r&&"value"!==r.kind&&(i=r);const s=[];for(const r of t.slice(1)){const t=e.parse(r,1+s.length,i,void 0,{typeAnnotation:"omit"});if(!t)return null;i=i||t.type,s.push(t)}if(!i)throw new Error("No output type");const n=r&&s.some((t=>Ut(r,t.type)));return new li(n?Pt:i,s)}evaluate(t){let e,i=null,r=0;for(const s of this.args)if(r++,i=s.evaluate(t),i&&i instanceof ge&&!i.available&&(e||(e=i.name),i=null,r===this.args.length&&(i=e)),null!==i)break;return i}eachChild(t){this.args.forEach(t)}outputDefined(){return this.args.every((t=>t.outputDefined()))}}class ci{constructor(t,e){this.type=e.type,this.bindings=[].concat(t),this.result=e}evaluate(t){return this.result.evaluate(t)}eachChild(t){for(const e of this.bindings)t(e[1]);t(this.result)}static parse(t,e){if(t.length<4)return e.error(`Expected at least 3 arguments, but found ${t.length-1} instead.`);const i=[];for(let r=1;r=i.length)throw new Te(`Array index out of bounds: ${e} > ${i.length-1}.`);if(e!==Math.floor(e))throw new Te(`Array index must be an integer, but found ${e} instead.`);return i[e]}eachChild(t){t(this.index),t(this.input)}outputDefined(){return!1}}class ui{constructor(t,e){this.type=Ct,this.needle=t,this.haystack=e}static parse(t,e){if(3!==t.length)return e.error(`Expected 2 arguments, but found ${t.length-1} instead.`);const i=e.parse(t[1],1,Pt),r=e.parse(t[2],2,Pt);return i&&r?Nt(i.type,[Ct,Et,At,It,Pt])?new ui(i,r):e.error(`Expected first argument to be of type boolean, string, number or null, but found ${Ot(i.type)} instead`):null}evaluate(t){const e=this.needle.evaluate(t),i=this.haystack.evaluate(t);if(!i)return!1;if(!$t(e,["boolean","string","number","null"]))throw new Te(`Expected first argument to be of type boolean, string, number or null, but found ${Ot(ve(e))} instead.`);if(!$t(i,["string","array"]))throw new Te(`Expected second argument to be of type array or string, but found ${Ot(ve(i))} instead.`);return i.indexOf(e)>=0}eachChild(t){t(this.needle),t(this.haystack)}outputDefined(){return!0}}class pi{constructor(t,e,i){this.type=At,this.needle=t,this.haystack=e,this.fromIndex=i}static parse(t,e){if(t.length<=2||t.length>=5)return e.error(`Expected 3 or 4 arguments, but found ${t.length-1} instead.`);const i=e.parse(t[1],1,Pt),r=e.parse(t[2],2,Pt);if(!i||!r)return null;if(!Nt(i.type,[Ct,Et,At,It,Pt]))return e.error(`Expected first argument to be of type boolean, string, number or null, but found ${Ot(i.type)} instead`);if(4===t.length){const s=e.parse(t[3],3,At);return s?new pi(i,r,s):null}return new pi(i,r)}evaluate(t){const e=this.needle.evaluate(t),i=this.haystack.evaluate(t);if(!$t(e,["boolean","string","number","null"]))throw new Te(`Expected first argument to be of type boolean, string, number or null, but found ${Ot(ve(e))} instead.`);if(!$t(i,["string","array"]))throw new Te(`Expected second argument to be of type array or string, but found ${Ot(ve(i))} instead.`);if(this.fromIndex){const r=this.fromIndex.evaluate(t);return i.indexOf(e,r)}return i.indexOf(e)}eachChild(t){t(this.needle),t(this.haystack),this.fromIndex&&t(this.fromIndex)}outputDefined(){return!1}}class di{constructor(t,e,i,r,s,n){this.inputType=t,this.type=e,this.input=i,this.cases=r,this.outputs=s,this.otherwise=n}static parse(t,e){if(t.length<5)return e.error(`Expected at least 4 arguments, but found only ${t.length-1}.`);if(t.length%2!=1)return e.error("Expected an even number of arguments.");let i,r;e.expectedType&&"value"!==e.expectedType.kind&&(r=e.expectedType);const s={},n=[];for(let a=2;aNumber.MAX_SAFE_INTEGER)return c.error(`Branch labels must be integers no larger than ${Number.MAX_SAFE_INTEGER}.`);if("number"==typeof t&&Math.floor(t)!==t)return c.error("Numeric branch labels must be integer values.");if(i){if(c.checkSubtype(i,ve(t)))return null}else i=ve(t);if(void 0!==s[String(t)])return c.error("Branch labels must be unique.");s[String(t)]=n.length}const h=e.parse(l,a,r);if(!h)return null;r=r||h.type,n.push(h)}const a=e.parse(t[1],1,Pt);if(!a)return null;const o=e.parse(t[t.length-1],t.length-1,r);return o?"value"!==a.type.kind&&e.concat(1).checkSubtype(i,a.type)?null:new di(i,r,a,s,n,o):null}evaluate(t){const e=this.input.evaluate(t);return(ve(e)===this.inputType&&this.outputs[this.cases[e]]||this.otherwise).evaluate(t)}eachChild(t){t(this.input),this.outputs.forEach(t),t(this.otherwise)}outputDefined(){return this.outputs.every((t=>t.outputDefined()))&&this.otherwise.outputDefined()}}class mi{constructor(t,e,i){this.type=t,this.branches=e,this.otherwise=i}static parse(t,e){if(t.length<4)return e.error(`Expected at least 3 arguments, but found only ${t.length-1}.`);if(t.length%2!=0)return e.error("Expected an odd number of arguments.");let i;e.expectedType&&"value"!==e.expectedType.kind&&(i=e.expectedType);const r=[];for(let s=1;se.outputDefined()))&&this.otherwise.outputDefined()}}class fi{constructor(t,e,i,r){this.type=t,this.input=e,this.beginIndex=i,this.endIndex=r}static parse(t,e){if(t.length<=2||t.length>=5)return e.error(`Expected 3 or 4 arguments, but found ${t.length-1} instead.`);const i=e.parse(t[1],1,Pt),r=e.parse(t[2],2,At);if(!i||!r)return null;if(!Nt(i.type,[Ft(Pt),Et,Pt]))return e.error(`Expected first argument to be of type array or string, but found ${Ot(i.type)} instead`);if(4===t.length){const s=e.parse(t[3],3,At);return s?new fi(i.type,i,r,s):null}return new fi(i.type,i,r)}evaluate(t){const e=this.input.evaluate(t),i=this.beginIndex.evaluate(t);if(!$t(e,["string","array"]))throw new Te(`Expected first argument to be of type array or string, but found ${Ot(ve(e))} instead.`);if(this.endIndex){const r=this.endIndex.evaluate(t);return e.slice(i,r)}return e.slice(i)}eachChild(t){t(this.input),t(this.beginIndex),this.endIndex&&t(this.endIndex)}outputDefined(){return!1}}function _i(t,e){return"=="===t||"!="===t?"boolean"===e.kind||"string"===e.kind||"number"===e.kind||"null"===e.kind||"value"===e.kind:"string"===e.kind||"number"===e.kind||"value"===e.kind}function gi(t,e,i,r){return 0===r.compare(e,i)}function yi(t,e,i){const r="=="!==t&&"!="!==t;return class s{constructor(t,e,i){this.type=Ct,this.lhs=t,this.rhs=e,this.collator=i,this.hasUntypedArgument="value"===t.type.kind||"value"===e.type.kind}static parse(t,e){if(3!==t.length&&4!==t.length)return e.error("Expected two or three arguments.");const i=t[0];let n=e.parse(t[1],1,Pt);if(!n)return null;if(!_i(i,n.type))return e.concat(1).error(`"${i}" comparisons are not supported for type '${Ot(n.type)}'.`);let a=e.parse(t[2],2,Pt);if(!a)return null;if(!_i(i,a.type))return e.concat(2).error(`"${i}" comparisons are not supported for type '${Ot(a.type)}'.`);if(n.type.kind!==a.type.kind&&"value"!==n.type.kind&&"value"!==a.type.kind)return e.error(`Cannot compare types '${Ot(n.type)}' and '${Ot(a.type)}'.`);r&&("value"===n.type.kind&&"value"!==a.type.kind?n=new Ie(a.type,[n]):"value"!==n.type.kind&&"value"===a.type.kind&&(a=new Ie(n.type,[a])));let o=null;if(4===t.length){if("string"!==n.type.kind&&"string"!==a.type.kind&&"value"!==n.type.kind&&"value"!==a.type.kind)return e.error("Cannot use collator to compare non-string types.");if(o=e.parse(t[3],3,kt),!o)return null}return new s(n,a,o)}evaluate(s){const n=this.lhs.evaluate(s),a=this.rhs.evaluate(s);if(r&&this.hasUntypedArgument){const e=ve(n),i=ve(a);if(e.kind!==i.kind||"string"!==e.kind&&"number"!==e.kind)throw new Te(`Expected arguments for "${t}" to be (string, string) or (number, number), but found (${e.kind}, ${i.kind}) instead.`)}if(this.collator&&!r&&this.hasUntypedArgument){const t=ve(n),i=ve(a);if("string"!==t.kind||"string"!==i.kind)return e(s,n,a)}return this.collator?i(s,n,a,this.collator.evaluate(s)):e(s,n,a)}eachChild(t){t(this.lhs),t(this.rhs),this.collator&&t(this.collator)}outputDefined(){return!0}}}const xi=yi("==",(function(t,e,i){return e===i}),gi),vi=yi("!=",(function(t,e,i){return e!==i}),(function(t,e,i,r){return!gi(0,e,i,r)})),bi=yi("<",(function(t,e,i){return e",(function(t,e,i){return e>i}),(function(t,e,i,r){return r.compare(e,i)>0})),Ti=yi("<=",(function(t,e,i){return e<=i}),(function(t,e,i,r){return r.compare(e,i)<=0})),Si=yi(">=",(function(t,e,i){return e>=i}),(function(t,e,i,r){return r.compare(e,i)>=0}));class Ii{constructor(t,e,i,r,s){this.type=Et,this.number=t,this.locale=e,this.currency=i,this.minFractionDigits=r,this.maxFractionDigits=s}static parse(t,e){if(3!==t.length)return e.error("Expected two arguments.");const i=e.parse(t[1],1,At);if(!i)return null;const r=t[2];if("object"!=typeof r||Array.isArray(r))return e.error("NumberFormat options argument must be an object.");let s=null;if(r.locale&&(s=e.parse(r.locale,1,Et),!s))return null;let n=null;if(r.currency&&(n=e.parse(r.currency,1,Et),!n))return null;let a=null;if(r["min-fraction-digits"]&&(a=e.parse(r["min-fraction-digits"],1,At),!a))return null;let o=null;return r["max-fraction-digits"]&&(o=e.parse(r["max-fraction-digits"],1,At),!o)?null:new Ii(i,s,n,a,o)}evaluate(t){return new Intl.NumberFormat(this.locale?this.locale.evaluate(t):[],{style:this.currency?"currency":"decimal",currency:this.currency?this.currency.evaluate(t):void 0,minimumFractionDigits:this.minFractionDigits?this.minFractionDigits.evaluate(t):void 0,maximumFractionDigits:this.maxFractionDigits?this.maxFractionDigits.evaluate(t):void 0}).format(this.number.evaluate(t))}eachChild(t){t(this.number),this.locale&&t(this.locale),this.currency&&t(this.currency),this.minFractionDigits&&t(this.minFractionDigits),this.maxFractionDigits&&t(this.maxFractionDigits)}outputDefined(){return!1}}class Ai{constructor(t){this.type=Dt,this.sections=t}static parse(t,e){if(t.length<2)return e.error("Expected at least one argument.");const i=t[1];if(!Array.isArray(i)&&"object"==typeof i)return e.error("First argument must be an image or text section.");const r=[];let s=!1;for(let i=1;i<=t.length-1;++i){const n=t[i];if(s&&"object"==typeof n&&!Array.isArray(n)){s=!1;let t=null;if(n["font-scale"]&&(t=e.parse(n["font-scale"],1,At),!t))return null;let i=null;if(n["text-font"]&&(i=e.parse(n["text-font"],1,Ft(Et)),!i))return null;let a=null;if(n["text-color"]&&(a=e.parse(n["text-color"],1,zt),!a))return null;const o=r[r.length-1];o.scale=t,o.font=i,o.textColor=a}else{const n=e.parse(t[i],1,Pt);if(!n)return null;const a=n.type.kind;if("string"!==a&&"value"!==a&&"null"!==a&&"resolvedImage"!==a)return e.error("Formatted text type must be 'string', 'value', 'image' or 'null'.");s=!0,r.push({content:n,scale:null,font:null,textColor:null})}}return new Ai(r)}evaluate(t){return new de(this.sections.map((e=>{const i=e.content.evaluate(t);return ve(i)===Rt?new pe("",i,null,null,null):new pe(be(i),null,e.scale?e.scale.evaluate(t):null,e.font?e.font.evaluate(t).join(","):null,e.textColor?e.textColor.evaluate(t):null)})))}eachChild(t){for(const e of this.sections)t(e.content),e.scale&&t(e.scale),e.font&&t(e.font),e.textColor&&t(e.textColor)}outputDefined(){return!1}}class Ei{constructor(t){this.type=Rt,this.input=t}static parse(t,e){if(2!==t.length)return e.error("Expected two arguments.");const i=e.parse(t[1],1,Et);return i?new Ei(i):e.error("No image name provided.")}evaluate(t){const e=this.input.evaluate(t),i=ge.fromString(e);return i&&t.availableImages&&(i.available=t.availableImages.indexOf(e)>-1),i}eachChild(t){t(this.input)}outputDefined(){return!1}}class Ci{constructor(t){this.type=At,this.input=t}static parse(t,e){if(2!==t.length)return e.error(`Expected 1 argument, but found ${t.length-1} instead.`);const i=e.parse(t[1],1);return i?"array"!==i.type.kind&&"string"!==i.type.kind&&"value"!==i.type.kind?e.error(`Expected argument of type string or array, but found ${Ot(i.type)} instead.`):new Ci(i):null}evaluate(t){const e=this.input.evaluate(t);if("string"==typeof e)return e.length;if(Array.isArray(e))return e.length;throw new Te(`Expected value to be of type string or array, but found ${Ot(ve(e))} instead.`)}eachChild(t){t(this.input)}outputDefined(){return!1}}const zi={"==":xi,"!=":vi,">":wi,"<":bi,">=":Si,"<=":Ti,array:Ie,at:hi,boolean:Ie,case:mi,coalesce:li,collator:Pe,format:Ai,image:Ei,in:ui,"index-of":pi,interpolate:ai,"interpolate-hcl":ai,"interpolate-lab":ai,length:Ci,let:ci,literal:we,match:di,number:Ie,"number-format":Ii,object:Ie,slice:fi,step:ii,string:Ie,"to-boolean":Ee,"to-color":Ee,"to-number":Ee,"to-string":Ee,var:Xe,within:We};function Mi(t,[e,i,r,s]){e=e.evaluate(t),i=i.evaluate(t),r=r.evaluate(t);const n=s?s.evaluate(t):1,a=ye(e,i,r,n);if(a)throw new Te(a);return new he(e/255,i/255,r/255,n,!1)}function Pi(t,e){return t in e}function ki(t,e){const i=e[t];return void 0===i?null:i}function Di(t){return{type:t}}function Li(t){return{result:"success",value:t}}function Ri(t){return{result:"error",value:t}}function Bi(t){return"data-driven"===t["property-type"]||"cross-faded-data-driven"===t["property-type"]}function Fi(t){return!!t.expression&&t.expression.parameters.indexOf("zoom")>-1}function Oi(t){return!!t.expression&&t.expression.interpolated}function Vi(t){return t instanceof Number?"number":t instanceof String?"string":t instanceof Boolean?"boolean":Array.isArray(t)?"array":null===t?"null":typeof t}function Ui(t){return"object"==typeof t&&null!==t&&!Array.isArray(t)}function Ni(t){return t}function $i(t,e){const i="color"===e.type,r=t.stops&&"object"==typeof t.stops[0][0],s=r||!(r||void 0!==t.property),n=t.type||(Oi(e)?"exponential":"interval");if(i||"padding"===e.type){const r=i?he.parse:me.parse;(t=wt({},t)).stops&&(t.stops=t.stops.map((t=>[t[0],r(t[1])]))),t.default=r(t.default?t.default:e.default)}if(t.colorSpace&&"rgb"!==(a=t.colorSpace)&&"hcl"!==a&&"lab"!==a)throw new Error(`Unknown color space: "${t.colorSpace}"`);var a;let o,l,c;if("exponential"===n)o=Gi;else if("interval"===n)o=ji;else if("categorical"===n){o=Zi,l=Object.create(null);for(const e of t.stops)l[e[0]]=e[1];c=typeof t.stops[0][0]}else{if("identity"!==n)throw new Error(`Unknown function type "${n}"`);o=Hi}if(r){const i={},r=[];for(let e=0;et[0])),evaluate:({zoom:i},r)=>Gi({stops:s,base:t.base},e,i).evaluate(i,r)}}if(s){const i="exponential"===n?{name:"exponential",base:void 0!==t.base?t.base:1}:null;return{kind:"camera",interpolationType:i,interpolationFactor:ai.interpolationFactor.bind(void 0,i),zoomStops:t.stops.map((t=>t[0])),evaluate:({zoom:i})=>o(t,e,i,l,c)}}return{kind:"source",evaluate(i,r){const s=r&&r.properties?r.properties[t.property]:void 0;return void 0===s?qi(t.default,e.default):o(t,e,s,l,c)}}}function qi(t,e,i){return void 0!==t?t:void 0!==e?e:void 0!==i?i:void 0}function Zi(t,e,i,r,s){return qi(typeof i===s?r[i]:void 0,t.default,e.default)}function ji(t,e,i){if("number"!==Vi(i))return qi(t.default,e.default);const r=t.stops.length;if(1===r)return t.stops[0][1];if(i<=t.stops[0][0])return t.stops[0][1];if(i>=t.stops[r-1][0])return t.stops[r-1][1];const s=ei(t.stops.map((t=>t[0])),i);return t.stops[s][1]}function Gi(t,e,i){const r=void 0!==t.base?t.base:1;if("number"!==Vi(i))return qi(t.default,e.default);const s=t.stops.length;if(1===s)return t.stops[0][1];if(i<=t.stops[0][0])return t.stops[0][1];if(i>=t.stops[s-1][0])return t.stops[s-1][1];const n=ei(t.stops.map((t=>t[0])),i),a=function(t,e,i,r){const s=r-i,n=t-i;return 0===s?0:1===e?n/s:(Math.pow(e,n)-1)/(Math.pow(e,s)-1)}(i,r,t.stops[n][0],t.stops[n+1][0]),o=t.stops[n][1],l=t.stops[n+1][1],c=ni[e.type]||Ni;return"function"==typeof o.evaluate?{evaluate(...e){const i=o.evaluate.apply(void 0,e),r=l.evaluate.apply(void 0,e);if(void 0!==i&&void 0!==r)return c(i,r,a,t.colorSpace)}}:c(o,l,a,t.colorSpace)}function Hi(t,e,i){switch(e.type){case"color":i=he.parse(i);break;case"formatted":i=de.fromString(i.toString());break;case"resolvedImage":i=ge.fromString(i.toString());break;case"padding":i=me.parse(i);break;default:Vi(i)===e.type||"enum"===e.type&&e.values[i]||(i=void 0)}return qi(i,t.default,e.default)}Ke.register(zi,{error:[{kind:"error"},[Et],(t,[e])=>{throw new Te(e.evaluate(t))}],typeof:[Et,[Pt],(t,[e])=>Ot(ve(e.evaluate(t)))],"to-rgba":[Ft(At,4),[zt],(t,[e])=>{const[i,r,s,n]=e.evaluate(t).rgb;return[255*i,255*r,255*s,n]}],rgb:[zt,[At,At,At],Mi],rgba:[zt,[At,At,At,At],Mi],has:{type:Ct,overloads:[[[Et],(t,[e])=>Pi(e.evaluate(t),t.properties())],[[Et,Mt],(t,[e,i])=>Pi(e.evaluate(t),i.evaluate(t))]]},get:{type:Pt,overloads:[[[Et],(t,[e])=>ki(e.evaluate(t),t.properties())],[[Et,Mt],(t,[e,i])=>ki(e.evaluate(t),i.evaluate(t))]]},"feature-state":[Pt,[Et],(t,[e])=>ki(e.evaluate(t),t.featureState||{})],properties:[Mt,[],t=>t.properties()],"geometry-type":[Et,[],t=>t.geometryType()],id:[Pt,[],t=>t.id()],zoom:[At,[],t=>t.globals.zoom],"heatmap-density":[At,[],t=>t.globals.heatmapDensity||0],"line-progress":[At,[],t=>t.globals.lineProgress||0],accumulated:[Pt,[],t=>void 0===t.globals.accumulated?null:t.globals.accumulated],"+":[At,Di(At),(t,e)=>{let i=0;for(const r of e)i+=r.evaluate(t);return i}],"*":[At,Di(At),(t,e)=>{let i=1;for(const r of e)i*=r.evaluate(t);return i}],"-":{type:At,overloads:[[[At,At],(t,[e,i])=>e.evaluate(t)-i.evaluate(t)],[[At],(t,[e])=>-e.evaluate(t)]]},"/":[At,[At,At],(t,[e,i])=>e.evaluate(t)/i.evaluate(t)],"%":[At,[At,At],(t,[e,i])=>e.evaluate(t)%i.evaluate(t)],ln2:[At,[],()=>Math.LN2],pi:[At,[],()=>Math.PI],e:[At,[],()=>Math.E],"^":[At,[At,At],(t,[e,i])=>Math.pow(e.evaluate(t),i.evaluate(t))],sqrt:[At,[At],(t,[e])=>Math.sqrt(e.evaluate(t))],log10:[At,[At],(t,[e])=>Math.log(e.evaluate(t))/Math.LN10],ln:[At,[At],(t,[e])=>Math.log(e.evaluate(t))],log2:[At,[At],(t,[e])=>Math.log(e.evaluate(t))/Math.LN2],sin:[At,[At],(t,[e])=>Math.sin(e.evaluate(t))],cos:[At,[At],(t,[e])=>Math.cos(e.evaluate(t))],tan:[At,[At],(t,[e])=>Math.tan(e.evaluate(t))],asin:[At,[At],(t,[e])=>Math.asin(e.evaluate(t))],acos:[At,[At],(t,[e])=>Math.acos(e.evaluate(t))],atan:[At,[At],(t,[e])=>Math.atan(e.evaluate(t))],min:[At,Di(At),(t,e)=>Math.min(...e.map((e=>e.evaluate(t))))],max:[At,Di(At),(t,e)=>Math.max(...e.map((e=>e.evaluate(t))))],abs:[At,[At],(t,[e])=>Math.abs(e.evaluate(t))],round:[At,[At],(t,[e])=>{const i=e.evaluate(t);return i<0?-Math.round(-i):Math.round(i)}],floor:[At,[At],(t,[e])=>Math.floor(e.evaluate(t))],ceil:[At,[At],(t,[e])=>Math.ceil(e.evaluate(t))],"filter-==":[Ct,[Et,Pt],(t,[e,i])=>t.properties()[e.value]===i.value],"filter-id-==":[Ct,[Pt],(t,[e])=>t.id()===e.value],"filter-type-==":[Ct,[Et],(t,[e])=>t.geometryType()===e.value],"filter-<":[Ct,[Et,Pt],(t,[e,i])=>{const r=t.properties()[e.value],s=i.value;return typeof r==typeof s&&r{const i=t.id(),r=e.value;return typeof i==typeof r&&i":[Ct,[Et,Pt],(t,[e,i])=>{const r=t.properties()[e.value],s=i.value;return typeof r==typeof s&&r>s}],"filter-id->":[Ct,[Pt],(t,[e])=>{const i=t.id(),r=e.value;return typeof i==typeof r&&i>r}],"filter-<=":[Ct,[Et,Pt],(t,[e,i])=>{const r=t.properties()[e.value],s=i.value;return typeof r==typeof s&&r<=s}],"filter-id-<=":[Ct,[Pt],(t,[e])=>{const i=t.id(),r=e.value;return typeof i==typeof r&&i<=r}],"filter->=":[Ct,[Et,Pt],(t,[e,i])=>{const r=t.properties()[e.value],s=i.value;return typeof r==typeof s&&r>=s}],"filter-id->=":[Ct,[Pt],(t,[e])=>{const i=t.id(),r=e.value;return typeof i==typeof r&&i>=r}],"filter-has":[Ct,[Pt],(t,[e])=>e.value in t.properties()],"filter-has-id":[Ct,[],t=>null!==t.id()&&void 0!==t.id()],"filter-type-in":[Ct,[Ft(Et)],(t,[e])=>e.value.indexOf(t.geometryType())>=0],"filter-id-in":[Ct,[Ft(Pt)],(t,[e])=>e.value.indexOf(t.id())>=0],"filter-in-small":[Ct,[Et,Ft(Pt)],(t,[e,i])=>i.value.indexOf(t.properties()[e.value])>=0],"filter-in-large":[Ct,[Et,Ft(Pt)],(t,[e,i])=>function(t,e,i,r){for(;i<=r;){const s=i+r>>1;if(e[s]===t)return!0;e[s]>t?r=s-1:i=s+1}return!1}(t.properties()[e.value],i.value,0,i.value.length-1)],all:{type:Ct,overloads:[[[Ct,Ct],(t,[e,i])=>e.evaluate(t)&&i.evaluate(t)],[Di(Ct),(t,e)=>{for(const i of e)if(!i.evaluate(t))return!1;return!0}]]},any:{type:Ct,overloads:[[[Ct,Ct],(t,[e,i])=>e.evaluate(t)||i.evaluate(t)],[Di(Ct),(t,e)=>{for(const i of e)if(i.evaluate(t))return!0;return!1}]]},"!":[Ct,[Ct],(t,[e])=>!e.evaluate(t)],"is-supported-script":[Ct,[Et],(t,[e])=>{const i=t.globals&&t.globals.isSupportedScript;return!i||i(e.evaluate(t))}],upcase:[Et,[Et],(t,[e])=>e.evaluate(t).toUpperCase()],downcase:[Et,[Et],(t,[e])=>e.evaluate(t).toLowerCase()],concat:[Et,Di(Pt),(t,e)=>e.map((e=>be(e.evaluate(t)))).join("")],"resolved-locale":[Et,[kt],(t,[e])=>e.evaluate(t).resolvedLocale()]});class Wi{constructor(t,e){var i;this.expression=t,this._warningHistory={},this._evaluator=new ze,this._defaultValue=e?"color"===(i=e).type&&Ui(i.default)?new he(0,0,0,0):"color"===i.type?he.parse(i.default)||null:"padding"===i.type?me.parse(i.default)||null:"variableAnchorOffsetCollection"===i.type?_e.parse(i.default)||null:void 0===i.default?null:i.default:null,this._enumValues=e&&"enum"===e.type?e.values:null}evaluateWithoutErrorHandling(t,e,i,r,s,n){return this._evaluator.globals=t,this._evaluator.feature=e,this._evaluator.featureState=i,this._evaluator.canonical=r,this._evaluator.availableImages=s||null,this._evaluator.formattedSection=n,this.expression.evaluate(this._evaluator)}evaluate(t,e,i,r,s,n){this._evaluator.globals=t,this._evaluator.feature=e||null,this._evaluator.featureState=i||null,this._evaluator.canonical=r,this._evaluator.availableImages=s||null,this._evaluator.formattedSection=n||null;try{const t=this.expression.evaluate(this._evaluator);if(null==t||"number"==typeof t&&t!=t)return this._defaultValue;if(this._enumValues&&!(t in this._enumValues))throw new Te(`Expected value to be one of ${Object.keys(this._enumValues).map((t=>JSON.stringify(t))).join(", ")}, but found ${JSON.stringify(t)} instead.`);return t}catch(t){return this._warningHistory[t.message]||(this._warningHistory[t.message]=!0,"undefined"!=typeof console&&console.warn(t.message)),this._defaultValue}}}function Xi(t){return Array.isArray(t)&&t.length>0&&"string"==typeof t[0]&&t[0]in zi}function Ki(t,e){const i=new Me(zi,Ye,[],e?function(t){const e={color:zt,string:Et,number:At,enum:Et,boolean:Ct,formatted:Dt,padding:Lt,resolvedImage:Rt,variableAnchorOffsetCollection:Bt};return"array"===t.type?Ft(e[t.value]||Pt,t.length):e[t.type]}(e):void 0),r=i.parse(t,void 0,void 0,void 0,e&&"string"===e.type?{typeAnnotation:"coerce"}:void 0);return r?Li(new Wi(r,e)):Ri(i.errors)}class Yi{constructor(t,e){this.kind=t,this._styleExpression=e,this.isStateDependent="constant"!==t&&!Qe(e.expression)}evaluateWithoutErrorHandling(t,e,i,r,s,n){return this._styleExpression.evaluateWithoutErrorHandling(t,e,i,r,s,n)}evaluate(t,e,i,r,s,n){return this._styleExpression.evaluate(t,e,i,r,s,n)}}class Ji{constructor(t,e,i,r){this.kind=t,this.zoomStops=i,this._styleExpression=e,this.isStateDependent="camera"!==t&&!Qe(e.expression),this.interpolationType=r}evaluateWithoutErrorHandling(t,e,i,r,s,n){return this._styleExpression.evaluateWithoutErrorHandling(t,e,i,r,s,n)}evaluate(t,e,i,r,s,n){return this._styleExpression.evaluate(t,e,i,r,s,n)}interpolationFactor(t,e,i){return this.interpolationType?ai.interpolationFactor(this.interpolationType,t,e,i):0}}function Qi(t,e){const i=Ki(t,e);if("error"===i.result)return i;const r=i.value.expression,s=Je(r);if(!s&&!Bi(e))return Ri([new Tt("","data expressions not supported")]);const n=ti(r,["zoom"]);if(!n&&!Fi(e))return Ri([new Tt("","zoom expressions not supported")]);const a=er(r);return a||n?a instanceof Tt?Ri([a]):a instanceof ai&&!Oi(e)?Ri([new Tt("",'"interpolate" expressions cannot be used with this property')]):Li(a?new Ji(s?"camera":"composite",i.value,a.labels,a instanceof ai?a.interpolation:void 0):new Yi(s?"constant":"source",i.value)):Ri([new Tt("",'"zoom" expression may only be used as input to a top-level "step" or "interpolate" expression.')])}class tr{constructor(t,e){this._parameters=t,this._specification=e,wt(this,$i(this._parameters,this._specification))}static deserialize(t){return new tr(t._parameters,t._specification)}static serialize(t){return{_parameters:t._parameters,_specification:t._specification}}}function er(t){let e=null;if(t instanceof ci)e=er(t.result);else if(t instanceof li){for(const i of t.args)if(e=er(i),e)break}else(t instanceof ii||t instanceof ai)&&t.input instanceof Ke&&"zoom"===t.input.name&&(e=t);return e instanceof Tt||t.eachChild((t=>{const i=er(t);i instanceof Tt?e=i:!e&&i?e=new Tt("",'"zoom" expression may only be used as input to a top-level "step" or "interpolate" expression.'):e&&i&&e!==i&&(e=new Tt("",'Only one zoom-based "step" or "interpolate" subexpression may be used in an expression.'))})),e}function ir(t){if(!0===t||!1===t)return!0;if(!Array.isArray(t)||0===t.length)return!1;switch(t[0]){case"has":return t.length>=2&&"$id"!==t[1]&&"$type"!==t[1];case"in":return t.length>=3&&("string"!=typeof t[1]||Array.isArray(t[2]));case"!in":case"!has":case"none":return!1;case"==":case"!=":case">":case">=":case"<":case"<=":return 3!==t.length||Array.isArray(t[1])||Array.isArray(t[2]);case"any":case"all":for(const e of t.slice(1))if(!ir(e)&&"boolean"!=typeof e)return!1;return!0;default:return!0}}const rr={type:"boolean",default:!1,transition:!1,"property-type":"data-driven",expression:{interpolated:!1,parameters:["zoom","feature"]}};function sr(t){if(null==t)return{filter:()=>!0,needGeometry:!1};ir(t)||(t=or(t));const e=Ki(t,rr);if("error"===e.result)throw new Error(e.value.map((t=>`${t.key}: ${t.message}`)).join(", "));return{filter:(t,i,r)=>e.value.evaluate(t,i,{},r),needGeometry:ar(t)}}function nr(t,e){return te?1:0}function ar(t){if(!Array.isArray(t))return!1;if("within"===t[0])return!0;for(let e=1;e"===e||"<="===e||">="===e?lr(t[1],t[2],e):"any"===e?(i=t.slice(1),["any"].concat(i.map(or))):"all"===e?["all"].concat(t.slice(1).map(or)):"none"===e?["all"].concat(t.slice(1).map(or).map(ur)):"in"===e?cr(t[1],t.slice(2)):"!in"===e?ur(cr(t[1],t.slice(2))):"has"===e?hr(t[1]):"!has"===e?ur(hr(t[1])):"within"!==e||t;var i}function lr(t,e,i){switch(t){case"$type":return[`filter-type-${i}`,e];case"$id":return[`filter-id-${i}`,e];default:return[`filter-${i}`,t,e]}}function cr(t,e){if(0===e.length)return!1;switch(t){case"$type":return["filter-type-in",["literal",e]];case"$id":return["filter-id-in",["literal",e]];default:return e.length>200&&!e.some((t=>typeof t!=typeof e[0]))?["filter-in-large",t,["literal",e.sort(nr)]]:["filter-in-small",t,["literal",e]]}}function hr(t){switch(t){case"$type":return!0;case"$id":return["filter-has-id"];default:return["filter-has",t]}}function ur(t){return["!",t]}function pr(t){const e=t.value;return e?[new bt(t.key,e,"constants have been deprecated as of v8")]:[]}function dr(t){return t instanceof Number||t instanceof String||t instanceof Boolean?t.valueOf():t}function mr(t){if(Array.isArray(t))return t.map(mr);if(t instanceof Object&&!(t instanceof Number||t instanceof String||t instanceof Boolean)){const e={};for(const i in t)e[i]=mr(t[i]);return e}return dr(t)}function fr(t){const e=t.key,i=t.value,r=t.valueSpec||{},s=t.objectElementValidators||{},n=t.style,a=t.styleSpec,o=t.validateSpec;let l=[];const c=Vi(i);if("object"!==c)return[new bt(e,i,`object expected, ${c} found`)];for(const t in i){const c=t.split(".")[0],h=r[c]||r["*"];let u;if(s[c])u=s[c];else if(r[c])u=o;else if(s["*"])u=s["*"];else{if(!r["*"]){l.push(new bt(e,i[t],`unknown property "${t}"`));continue}u=o}l=l.concat(u({key:(e?`${e}.`:e)+t,value:i[t],valueSpec:h,style:n,styleSpec:a,object:i,objectKey:t,validateSpec:o},i))}for(const t in r)s[t]||r[t].required&&void 0===r[t].default&&void 0===i[t]&&l.push(new bt(e,i,`missing required property "${t}"`));return l}function _r(t){const e=t.value,i=t.valueSpec,r=t.style,s=t.styleSpec,n=t.key,a=t.arrayElementValidator||t.validateSpec;if("array"!==Vi(e))return[new bt(n,e,`array expected, ${Vi(e)} found`)];if(i.length&&e.length!==i.length)return[new bt(n,e,`array length ${i.length} expected, length ${e.length} found`)];if(i["min-length"]&&e.lengthr.maximum?[new bt(e,i,`${i} is greater than the maximum value ${r.maximum}`)]:[]}function yr(t){const e=t.valueSpec,i=dr(t.value.type);let r,s,n,a={};const o="categorical"!==i&&void 0===t.value.property,l=!o,c="array"===Vi(t.value.stops)&&"array"===Vi(t.value.stops[0])&&"object"===Vi(t.value.stops[0][0]),h=fr({key:t.key,value:t.value,valueSpec:t.styleSpec.function,validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec,objectElementValidators:{stops:function(t){if("identity"===i)return[new bt(t.key,t.value,'identity function may not have a "stops" property')];let e=[];const r=t.value;return e=e.concat(_r({key:t.key,value:r,valueSpec:t.valueSpec,validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec,arrayElementValidator:u})),"array"===Vi(r)&&0===r.length&&e.push(new bt(t.key,r,"array must have at least one stop")),e},default:function(t){return t.validateSpec({key:t.key,value:t.value,valueSpec:e,validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec})}}});return"identity"===i&&o&&h.push(new bt(t.key,t.value,'missing required property "property"')),"identity"===i||t.value.stops||h.push(new bt(t.key,t.value,'missing required property "stops"')),"exponential"===i&&t.valueSpec.expression&&!Oi(t.valueSpec)&&h.push(new bt(t.key,t.value,"exponential functions not supported")),t.styleSpec.$version>=8&&(l&&!Bi(t.valueSpec)?h.push(new bt(t.key,t.value,"property functions not supported")):o&&!Fi(t.valueSpec)&&h.push(new bt(t.key,t.value,"zoom functions not supported"))),"categorical"!==i&&!c||void 0!==t.value.property||h.push(new bt(t.key,t.value,'"property" property is required')),h;function u(t){let i=[];const r=t.value,o=t.key;if("array"!==Vi(r))return[new bt(o,r,`array expected, ${Vi(r)} found`)];if(2!==r.length)return[new bt(o,r,`array length 2 expected, length ${r.length} found`)];if(c){if("object"!==Vi(r[0]))return[new bt(o,r,`object expected, ${Vi(r[0])} found`)];if(void 0===r[0].zoom)return[new bt(o,r,"object stop key must have zoom")];if(void 0===r[0].value)return[new bt(o,r,"object stop key must have value")];if(n&&n>dr(r[0].zoom))return[new bt(o,r[0].zoom,"stop zoom values must appear in ascending order")];dr(r[0].zoom)!==n&&(n=dr(r[0].zoom),s=void 0,a={}),i=i.concat(fr({key:`${o}[0]`,value:r[0],valueSpec:{zoom:{}},validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec,objectElementValidators:{zoom:gr,value:p}}))}else i=i.concat(p({key:`${o}[0]`,value:r[0],valueSpec:{},validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec},r));return Xi(mr(r[1]))?i.concat([new bt(`${o}[1]`,r[1],"expressions are not allowed in function stops.")]):i.concat(t.validateSpec({key:`${o}[1]`,value:r[1],valueSpec:e,validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec}))}function p(t,n){const o=Vi(t.value),l=dr(t.value),c=null!==t.value?t.value:n;if(r){if(o!==r)return[new bt(t.key,c,`${o} stop domain type must match previous stop domain type ${r}`)]}else r=o;if("number"!==o&&"string"!==o&&"boolean"!==o)return[new bt(t.key,c,"stop domain value must be a number, string, or boolean")];if("number"!==o&&"categorical"!==i){let r=`number expected, ${o} found`;return Bi(e)&&void 0===i&&(r+='\nIf you intended to use a categorical function, specify `"type": "categorical"`.'),[new bt(t.key,c,r)]}return"categorical"!==i||"number"!==o||isFinite(l)&&Math.floor(l)===l?"categorical"!==i&&"number"===o&&void 0!==s&&lnew bt(`${t.key}${e.key}`,t.value,e.message)));const i=e.value.expression||e.value._styleExpression.expression;if("property"===t.expressionContext&&"text-font"===t.propertyKey&&!i.outputDefined())return[new bt(t.key,t.value,`Invalid data expression for "${t.propertyKey}". Output values must be contained as literals within the expression.`)];if("property"===t.expressionContext&&"layout"===t.propertyType&&!Qe(i))return[new bt(t.key,t.value,'"feature-state" data expressions are not supported with layout properties.')];if("filter"===t.expressionContext&&!Qe(i))return[new bt(t.key,t.value,'"feature-state" data expressions are not supported with filters.')];if(t.expressionContext&&0===t.expressionContext.indexOf("cluster")){if(!ti(i,["zoom","feature-state"]))return[new bt(t.key,t.value,'"zoom" and "feature-state" expressions are not supported with cluster properties.')];if("cluster-initial"===t.expressionContext&&!Je(i))return[new bt(t.key,t.value,"Feature data expressions are not supported with initial expression part of cluster properties.")]}return[]}function vr(t){const e=t.key,i=t.value,r=t.valueSpec,s=[];return Array.isArray(r.values)?-1===r.values.indexOf(dr(i))&&s.push(new bt(e,i,`expected one of [${r.values.join(", ")}], ${JSON.stringify(i)} found`)):-1===Object.keys(r.values).indexOf(dr(i))&&s.push(new bt(e,i,`expected one of [${Object.keys(r.values).join(", ")}], ${JSON.stringify(i)} found`)),s}function br(t){return ir(mr(t.value))?xr(wt({},t,{expressionContext:"filter",valueSpec:{value:"boolean"}})):wr(t)}function wr(t){const e=t.value,i=t.key;if("array"!==Vi(e))return[new bt(i,e,`array expected, ${Vi(e)} found`)];const r=t.styleSpec;let s,n=[];if(e.length<1)return[new bt(i,e,"filter array must have at least 1 element")];switch(n=n.concat(vr({key:`${i}[0]`,value:e[0],valueSpec:r.filter_operator,style:t.style,styleSpec:t.styleSpec})),dr(e[0])){case"<":case"<=":case">":case">=":e.length>=2&&"$type"===dr(e[1])&&n.push(new bt(i,e,`"$type" cannot be use with operator "${e[0]}"`));case"==":case"!=":3!==e.length&&n.push(new bt(i,e,`filter array for operator "${e[0]}" must have 3 elements`));case"in":case"!in":e.length>=2&&(s=Vi(e[1]),"string"!==s&&n.push(new bt(`${i}[1]`,e[1],`string expected, ${s} found`)));for(let a=2;a{t in i&&e.push(new bt(r,i[t],`"${t}" is prohibited for ref layers`))})),s.layers.forEach((e=>{dr(e.id)===o&&(t=e)})),t?t.ref?e.push(new bt(r,i.ref,"ref cannot reference another ref layer")):a=dr(t.type):e.push(new bt(r,i.ref,`ref layer "${o}" not found`))}else if("background"!==a)if(i.source){const t=s.sources&&s.sources[i.source],n=t&&dr(t.type);t?"vector"===n&&"raster"===a?e.push(new bt(r,i.source,`layer "${i.id}" requires a raster source`)):"raster-dem"!==n&&"hillshade"===a?e.push(new bt(r,i.source,`layer "${i.id}" requires a raster-dem source`)):"raster"===n&&"raster"!==a?e.push(new bt(r,i.source,`layer "${i.id}" requires a vector source`)):"vector"!==n||i["source-layer"]?"raster-dem"===n&&"hillshade"!==a?e.push(new bt(r,i.source,"raster-dem source can only be used with layer type 'hillshade'.")):"line"!==a||!i.paint||!i.paint["line-gradient"]||"geojson"===n&&t.lineMetrics||e.push(new bt(r,i,`layer "${i.id}" specifies a line-gradient, which requires a GeoJSON source with \`lineMetrics\` enabled.`)):e.push(new bt(r,i,`layer "${i.id}" must specify a "source-layer"`)):e.push(new bt(r,i.source,`source "${i.source}" not found`))}else e.push(new bt(r,i,'missing required property "source"'));return e=e.concat(fr({key:r,value:i,valueSpec:n.layer,style:t.style,styleSpec:t.styleSpec,validateSpec:t.validateSpec,objectElementValidators:{"*":()=>[],type:()=>t.validateSpec({key:`${r}.type`,value:i.type,valueSpec:n.layer.type,style:t.style,styleSpec:t.styleSpec,validateSpec:t.validateSpec,object:i,objectKey:"type"}),filter:br,layout:t=>fr({layer:i,key:t.key,value:t.value,style:t.style,styleSpec:t.styleSpec,validateSpec:t.validateSpec,objectElementValidators:{"*":t=>Ir(wt({layerType:a},t))}}),paint:t=>fr({layer:i,key:t.key,value:t.value,style:t.style,styleSpec:t.styleSpec,validateSpec:t.validateSpec,objectElementValidators:{"*":t=>Sr(wt({layerType:a},t))}})}})),e}function Er(t){const e=t.value,i=t.key,r=Vi(e);return"string"!==r?[new bt(i,e,`string expected, ${r} found`)]:[]}const Cr={promoteId:function({key:t,value:e}){if("string"===Vi(e))return Er({key:t,value:e});{const i=[];for(const r in e)i.push(...Er({key:`${t}.${r}`,value:e[r]}));return i}}};function zr(t){const e=t.value,i=t.key,r=t.styleSpec,s=t.style,n=t.validateSpec;if(!e.type)return[new bt(i,e,'"type" is required')];const a=dr(e.type);let o;switch(a){case"vector":case"raster":return o=fr({key:i,value:e,valueSpec:r[`source_${a.replace("-","_")}`],style:t.style,styleSpec:r,objectElementValidators:Cr,validateSpec:n}),o;case"raster-dem":return o=function(t){var e;const i=null!==(e=t.sourceName)&&void 0!==e?e:"",r=t.value,s=t.styleSpec,n=s.source_raster_dem,a=t.style;let o=[];const l=Vi(r);if(void 0===r)return o;if("object"!==l)return o.push(new bt("source_raster_dem",r,`object expected, ${l} found`)),o;const c="custom"===dr(r.encoding),h=["redFactor","greenFactor","blueFactor","baseShift"],u=t.value.encoding?`"${t.value.encoding}"`:"Default";for(const e in r)!c&&h.includes(e)?o.push(new bt(e,r[e],`In "${i}": "${e}" is only valid when "encoding" is set to "custom". ${u} encoding found`)):n[e]?o=o.concat(t.validateSpec({key:e,value:r[e],valueSpec:n[e],validateSpec:t.validateSpec,style:a,styleSpec:s})):o.push(new bt(e,r[e],`unknown property "${e}"`));return o}({sourceName:i,value:e,style:t.style,styleSpec:r,validateSpec:n}),o;case"geojson":if(o=fr({key:i,value:e,valueSpec:r.source_geojson,style:s,styleSpec:r,validateSpec:n,objectElementValidators:Cr}),e.cluster)for(const t in e.clusterProperties){const[r,s]=e.clusterProperties[t],a="string"==typeof r?[r,["accumulated"],["get",t]]:r;o.push(...xr({key:`${i}.${t}.map`,value:s,validateSpec:n,expressionContext:"cluster-map"})),o.push(...xr({key:`${i}.${t}.reduce`,value:a,validateSpec:n,expressionContext:"cluster-reduce"}))}return o;case"video":return fr({key:i,value:e,valueSpec:r.source_video,style:s,validateSpec:n,styleSpec:r});case"image":return fr({key:i,value:e,valueSpec:r.source_image,style:s,validateSpec:n,styleSpec:r});case"canvas":return[new bt(i,null,"Please use runtime APIs to add canvas sources, rather than including them in stylesheets.","source.canvas")];default:return vr({key:`${i}.type`,value:e.type,valueSpec:{values:["vector","raster","raster-dem","geojson","video","image"]},style:s,validateSpec:n,styleSpec:r})}}function Mr(t){const e=t.value,i=t.styleSpec,r=i.light,s=t.style;let n=[];const a=Vi(e);if(void 0===e)return n;if("object"!==a)return n=n.concat([new bt("light",e,`object expected, ${a} found`)]),n;for(const a in e){const o=a.match(/^(.*)-transition$/);n=n.concat(o&&r[o[1]]&&r[o[1]].transition?t.validateSpec({key:a,value:e[a],valueSpec:i.transition,validateSpec:t.validateSpec,style:s,styleSpec:i}):r[a]?t.validateSpec({key:a,value:e[a],valueSpec:r[a],validateSpec:t.validateSpec,style:s,styleSpec:i}):[new bt(a,e[a],`unknown property "${a}"`)])}return n}function Pr(t){const e=t.value,i=t.styleSpec,r=i.terrain,s=t.style;let n=[];const a=Vi(e);if(void 0===e)return n;if("object"!==a)return n=n.concat([new bt("terrain",e,`object expected, ${a} found`)]),n;for(const a in e)n=n.concat(r[a]?t.validateSpec({key:a,value:e[a],valueSpec:r[a],validateSpec:t.validateSpec,style:s,styleSpec:i}):[new bt(a,e[a],`unknown property "${a}"`)]);return n}function kr(t){let e=[];const i=t.value,r=t.key;if(Array.isArray(i)){const s=[],n=[];for(const a in i)i[a].id&&s.includes(i[a].id)&&e.push(new bt(r,i,`all the sprites' ids must be unique, but ${i[a].id} is duplicated`)),s.push(i[a].id),i[a].url&&n.includes(i[a].url)&&e.push(new bt(r,i,`all the sprites' URLs must be unique, but ${i[a].url} is duplicated`)),n.push(i[a].url),e=e.concat(fr({key:`${r}[${a}]`,value:i[a],valueSpec:{id:{type:"string",required:!0},url:{type:"string",required:!0}},validateSpec:t.validateSpec}));return e}return Er({key:r,value:i})}const Dr={"*":()=>[],array:_r,boolean:function(t){const e=t.value,i=t.key,r=Vi(e);return"boolean"!==r?[new bt(i,e,`boolean expected, ${r} found`)]:[]},number:gr,color:function(t){const e=t.key,i=t.value,r=Vi(i);return"string"!==r?[new bt(e,i,`color expected, ${r} found`)]:he.parse(String(i))?[]:[new bt(e,i,`color expected, "${i}" found`)]},constants:pr,enum:vr,filter:br,function:yr,layer:Ar,object:fr,source:zr,light:Mr,terrain:Pr,string:Er,formatted:function(t){return 0===Er(t).length?[]:xr(t)},resolvedImage:function(t){return 0===Er(t).length?[]:xr(t)},padding:function(t){const e=t.key,i=t.value;if("array"===Vi(i)){if(i.length<1||i.length>4)return[new bt(e,i,`padding requires 1 to 4 values; ${i.length} values found`)];const r={type:"number"};let s=[];for(let n=0;n[]}})),t.constants&&(i=i.concat(pr({key:"constants",value:t.constants,style:t,styleSpec:e,validateSpec:Lr}))),Or(i)}function Fr(t){return function(e){return t({...e,validateSpec:Lr})}}function Or(t){return[].concat(t).sort(((t,e)=>t.line-e.line))}function Vr(t){return function(...e){return Or(t.apply(this,e))}}Br.source=Vr(Fr(zr)),Br.sprite=Vr(Fr(kr)),Br.glyphs=Vr(Fr(Rr)),Br.light=Vr(Fr(Mr)),Br.terrain=Vr(Fr(Pr)),Br.layer=Vr(Fr(Ar)),Br.filter=Vr(Fr(br)),Br.paintProperty=Vr(Fr(Sr)),Br.layoutProperty=Vr(Fr(Ir));const Ur=Br,Nr=Ur.light,$r=Ur.paintProperty,qr=Ur.layoutProperty;function Zr(t,e){let i=!1;if(e&&e.length)for(const r of e)t.fire(new at(new Error(r.message))),i=!0;return i}class jr{constructor(t,e,i){const r=this.cells=[];if(t instanceof ArrayBuffer){this.arrayBuffer=t;const s=new Int32Array(this.arrayBuffer);t=s[0],this.d=(e=s[1])+2*(i=s[2]);for(let t=0;t=c[l+0]&&r>=c[l+1])?(a[u]=!0,n.push(s[u])):a[u]=!1}}}}_forEachCell(t,e,i,r,s,n,a,o){const l=this._convertToCellCoord(t),c=this._convertToCellCoord(e),h=this._convertToCellCoord(i),u=this._convertToCellCoord(r);for(let p=l;p<=h;p++)for(let l=c;l<=u;l++){const c=this.d*l+p;if((!o||o(this._convertFromCellCoord(p),this._convertFromCellCoord(l),this._convertFromCellCoord(p+1),this._convertFromCellCoord(l+1)))&&s.call(this,t,e,i,r,c,n,a,o))return}}_convertFromCellCoord(t){return(t-this.padding)/this.scale}_convertToCellCoord(t){return Math.max(0,Math.min(this.d-1,Math.floor(t*this.scale)+this.padding))}toArrayBuffer(){if(this.arrayBuffer)return this.arrayBuffer;const t=this.cells,e=3+this.cells.length+1+1;let i=0;for(let t=0;t=0)continue;const n=t[i];s[i]=Gr[r].shallow.indexOf(i)>=0?n:Xr(n,e)}t instanceof Error&&(s.message=t.message)}if(s.$name)throw new Error("$name property is reserved for worker serialization logic.");return"Object"!==r&&(s.$name=r),s}throw new Error("can't serialize object of type "+typeof t)}function Kr(t){if(null==t||"boolean"==typeof t||"number"==typeof t||"string"==typeof t||t instanceof Boolean||t instanceof Number||t instanceof String||t instanceof Date||t instanceof RegExp||t instanceof Blob||Wr(t)||k(t)||ArrayBuffer.isView(t)||t instanceof ImageData)return t;if(Array.isArray(t))return t.map(Kr);if("object"==typeof t){const e=t.$name||"Object";if(!Gr[e])throw new Error(`can't deserialize unregistered class ${e}`);const{klass:i}=Gr[e];if(!i)throw new Error(`can't deserialize unregistered class ${e}`);if(i.deserialize)return i.deserialize(t);const r=Object.create(i.prototype);for(const i of Object.keys(t)){if("$name"===i)continue;const s=t[i];r[i]=Gr[e].shallow.indexOf(i)>=0?s:Kr(s)}return r}throw new Error("can't deserialize object of type "+typeof t)}class Yr{constructor(){this.first=!0}update(t,e){const i=Math.floor(t);return this.first?(this.first=!1,this.lastIntegerZoom=i,this.lastIntegerZoomTime=0,this.lastZoom=t,this.lastFloorZoom=i,!0):(this.lastFloorZoom>i?(this.lastIntegerZoom=i+1,this.lastIntegerZoomTime=e):this.lastFloorZoomt>=128&&t<=255,Arabic:t=>t>=1536&&t<=1791,"Arabic Supplement":t=>t>=1872&&t<=1919,"Arabic Extended-A":t=>t>=2208&&t<=2303,"Hangul Jamo":t=>t>=4352&&t<=4607,"Unified Canadian Aboriginal Syllabics":t=>t>=5120&&t<=5759,Khmer:t=>t>=6016&&t<=6143,"Unified Canadian Aboriginal Syllabics Extended":t=>t>=6320&&t<=6399,"General Punctuation":t=>t>=8192&&t<=8303,"Letterlike Symbols":t=>t>=8448&&t<=8527,"Number Forms":t=>t>=8528&&t<=8591,"Miscellaneous Technical":t=>t>=8960&&t<=9215,"Control Pictures":t=>t>=9216&&t<=9279,"Optical Character Recognition":t=>t>=9280&&t<=9311,"Enclosed Alphanumerics":t=>t>=9312&&t<=9471,"Geometric Shapes":t=>t>=9632&&t<=9727,"Miscellaneous Symbols":t=>t>=9728&&t<=9983,"Miscellaneous Symbols and Arrows":t=>t>=11008&&t<=11263,"CJK Radicals Supplement":t=>t>=11904&&t<=12031,"Kangxi Radicals":t=>t>=12032&&t<=12255,"Ideographic Description Characters":t=>t>=12272&&t<=12287,"CJK Symbols and Punctuation":t=>t>=12288&&t<=12351,Hiragana:t=>t>=12352&&t<=12447,Katakana:t=>t>=12448&&t<=12543,Bopomofo:t=>t>=12544&&t<=12591,"Hangul Compatibility Jamo":t=>t>=12592&&t<=12687,Kanbun:t=>t>=12688&&t<=12703,"Bopomofo Extended":t=>t>=12704&&t<=12735,"CJK Strokes":t=>t>=12736&&t<=12783,"Katakana Phonetic Extensions":t=>t>=12784&&t<=12799,"Enclosed CJK Letters and Months":t=>t>=12800&&t<=13055,"CJK Compatibility":t=>t>=13056&&t<=13311,"CJK Unified Ideographs Extension A":t=>t>=13312&&t<=19903,"Yijing Hexagram Symbols":t=>t>=19904&&t<=19967,"CJK Unified Ideographs":t=>t>=19968&&t<=40959,"Yi Syllables":t=>t>=40960&&t<=42127,"Yi Radicals":t=>t>=42128&&t<=42191,"Hangul Jamo Extended-A":t=>t>=43360&&t<=43391,"Hangul Syllables":t=>t>=44032&&t<=55215,"Hangul Jamo Extended-B":t=>t>=55216&&t<=55295,"Private Use Area":t=>t>=57344&&t<=63743,"CJK Compatibility Ideographs":t=>t>=63744&&t<=64255,"Arabic Presentation Forms-A":t=>t>=64336&&t<=65023,"Vertical Forms":t=>t>=65040&&t<=65055,"CJK Compatibility Forms":t=>t>=65072&&t<=65103,"Small Form Variants":t=>t>=65104&&t<=65135,"Arabic Presentation Forms-B":t=>t>=65136&&t<=65279,"Halfwidth and Fullwidth Forms":t=>t>=65280&&t<=65519};function Qr(t){for(const e of t)if(ts(e.charCodeAt(0)))return!0;return!1}function ts(t){return!(746!==t&&747!==t&&(t<4352||!(Jr["Bopomofo Extended"](t)||Jr.Bopomofo(t)||Jr["CJK Compatibility Forms"](t)&&!(t>=65097&&t<=65103)||Jr["CJK Compatibility Ideographs"](t)||Jr["CJK Compatibility"](t)||Jr["CJK Radicals Supplement"](t)||Jr["CJK Strokes"](t)||!(!Jr["CJK Symbols and Punctuation"](t)||t>=12296&&t<=12305||t>=12308&&t<=12319||12336===t)||Jr["CJK Unified Ideographs Extension A"](t)||Jr["CJK Unified Ideographs"](t)||Jr["Enclosed CJK Letters and Months"](t)||Jr["Hangul Compatibility Jamo"](t)||Jr["Hangul Jamo Extended-A"](t)||Jr["Hangul Jamo Extended-B"](t)||Jr["Hangul Jamo"](t)||Jr["Hangul Syllables"](t)||Jr.Hiragana(t)||Jr["Ideographic Description Characters"](t)||Jr.Kanbun(t)||Jr["Kangxi Radicals"](t)||Jr["Katakana Phonetic Extensions"](t)||Jr.Katakana(t)&&12540!==t||!(!Jr["Halfwidth and Fullwidth Forms"](t)||65288===t||65289===t||65293===t||t>=65306&&t<=65310||65339===t||65341===t||65343===t||t>=65371&&t<=65503||65507===t||t>=65512&&t<=65519)||!(!Jr["Small Form Variants"](t)||t>=65112&&t<=65118||t>=65123&&t<=65126)||Jr["Unified Canadian Aboriginal Syllabics"](t)||Jr["Unified Canadian Aboriginal Syllabics Extended"](t)||Jr["Vertical Forms"](t)||Jr["Yijing Hexagram Symbols"](t)||Jr["Yi Syllables"](t)||Jr["Yi Radicals"](t))))}function es(t){return t>=1424&&t<=2303||Jr["Arabic Presentation Forms-A"](t)||Jr["Arabic Presentation Forms-B"](t)}function is(t,e){return!(!e&&es(t)||t>=2304&&t<=3583||t>=3840&&t<=4255||Jr.Khmer(t))}function rs(t){for(const e of t)if(es(e.charCodeAt(0)))return!0;return!1}const ss="deferred",ns="loading",as="loaded";let os=null,ls="unavailable",cs=null;const hs=function(t){t&&"string"==typeof t&&t.indexOf("NetworkError")>-1&&(ls="error"),os&&os(t)};function us(){ps.fire(new nt("pluginStateChange",{pluginStatus:ls,pluginURL:cs}))}const ps=new ot,ds=function(){return ls},ms=function(){if(ls!==ss||!cs)throw new Error("rtl-text-plugin cannot be downloaded unless a pluginURL is specified");ls=ns,us(),cs&&j({url:cs},(t=>{t?hs(t):(ls=as,us())}))},fs={applyArabicShaping:null,processBidirectionalText:null,processStyledBidirectionalText:null,isLoaded:()=>ls===as||null!=fs.applyArabicShaping,isLoading:()=>ls===ns,setState(t){if(!z())throw new Error("Cannot set the state of the rtl-text-plugin when not in the web-worker context");ls=t.pluginStatus,cs=t.pluginURL},isParsed(){if(!z())throw new Error("rtl-text-plugin is only parsed on the worker-threads");return null!=fs.applyArabicShaping},getPluginURL(){if(!z())throw new Error("rtl-text-plugin url can only be queried from the worker threads");return cs}};class _s{constructor(t,e){this.zoom=t,e?(this.now=e.now,this.fadeDuration=e.fadeDuration,this.zoomHistory=e.zoomHistory,this.transition=e.transition):(this.now=0,this.fadeDuration=0,this.zoomHistory=new Yr,this.transition={})}isSupportedScript(t){return function(t,e){for(const i of t)if(!is(i.charCodeAt(0),e))return!1;return!0}(t,fs.isLoaded())}crossFadingFactor(){return 0===this.fadeDuration?1:Math.min((this.now-this.zoomHistory.lastIntegerZoomTime)/this.fadeDuration,1)}getCrossfadeParameters(){const t=this.zoom,e=t-Math.floor(t),i=this.crossFadingFactor();return t>this.zoomHistory.lastIntegerZoom?{fromScale:2,toScale:1,t:e+(1-e)*i}:{fromScale:.5,toScale:1,t:1-(1-i)*e}}}class gs{constructor(t,e){this.property=t,this.value=e,this.expression=function(t,e){if(Ui(t))return new tr(t,e);if(Xi(t)){const i=Qi(t,e);if("error"===i.result)throw new Error(i.value.map((t=>`${t.key}: ${t.message}`)).join(", "));return i.value}{let i=t;return"color"===e.type&&"string"==typeof t?i=he.parse(t):"padding"!==e.type||"number"!=typeof t&&!Array.isArray(t)?"variableAnchorOffsetCollection"===e.type&&Array.isArray(t)&&(i=_e.parse(t)):i=me.parse(t),{kind:"constant",evaluate:()=>i}}}(void 0===e?t.specification.default:e,t.specification)}isDataDriven(){return"source"===this.expression.kind||"composite"===this.expression.kind}possiblyEvaluate(t,e,i){return this.property.possiblyEvaluate(this,t,e,i)}}class ys{constructor(t){this.property=t,this.value=new gs(t,void 0)}transitioned(t,e){return new vs(this.property,this.value,e,g({},t.transition,this.transition),t.now)}untransitioned(){return new vs(this.property,this.value,null,{},0)}}class xs{constructor(t){this._properties=t,this._values=Object.create(t.defaultTransitionablePropertyValues)}getValue(t){return S(this._values[t].value.value)}setValue(t,e){Object.prototype.hasOwnProperty.call(this._values,t)||(this._values[t]=new ys(this._values[t].property)),this._values[t].value=new gs(this._values[t].property,null===e?void 0:S(e))}getTransition(t){return S(this._values[t].transition)}setTransition(t,e){Object.prototype.hasOwnProperty.call(this._values,t)||(this._values[t]=new ys(this._values[t].property)),this._values[t].transition=S(e)||void 0}serialize(){const t={};for(const e of Object.keys(this._values)){const i=this.getValue(e);void 0!==i&&(t[e]=i);const r=this.getTransition(e);void 0!==r&&(t[`${e}-transition`]=r)}return t}transitioned(t,e){const i=new bs(this._properties);for(const r of Object.keys(this._values))i._values[r]=this._values[r].transitioned(t,e._values[r]);return i}untransitioned(){const t=new bs(this._properties);for(const e of Object.keys(this._values))t._values[e]=this._values[e].untransitioned();return t}}class vs{constructor(t,e,i,r,s){this.property=t,this.value=e,this.begin=s+r.delay||0,this.end=this.begin+r.duration||0,t.specification.transition&&(r.delay||r.duration)&&(this.prior=i)}possiblyEvaluate(t,e,i){const r=t.now||0,s=this.value.possiblyEvaluate(t,e,i),n=this.prior;if(n){if(r>this.end)return this.prior=null,s;if(this.value.isDataDriven())return this.prior=null,s;if(r=1)return 1;const e=t*t,i=e*t;return 4*(t<.5?i:3*(t-e)+i-.75)}(a))}}return s}}class bs{constructor(t){this._properties=t,this._values=Object.create(t.defaultTransitioningPropertyValues)}possiblyEvaluate(t,e,i){const r=new Ss(this._properties);for(const s of Object.keys(this._values))r._values[s]=this._values[s].possiblyEvaluate(t,e,i);return r}hasTransition(){for(const t of Object.keys(this._values))if(this._values[t].prior)return!0;return!1}}class ws{constructor(t){this._properties=t,this._values=Object.create(t.defaultPropertyValues)}hasValue(t){return void 0!==this._values[t].value}getValue(t){return S(this._values[t].value)}setValue(t,e){this._values[t]=new gs(this._values[t].property,null===e?void 0:S(e))}serialize(){const t={};for(const e of Object.keys(this._values)){const i=this.getValue(e);void 0!==i&&(t[e]=i)}return t}possiblyEvaluate(t,e,i){const r=new Ss(this._properties);for(const s of Object.keys(this._values))r._values[s]=this._values[s].possiblyEvaluate(t,e,i);return r}}class Ts{constructor(t,e,i){this.property=t,this.value=e,this.parameters=i}isConstant(){return"constant"===this.value.kind}constantOr(t){return"constant"===this.value.kind?this.value.value:t}evaluate(t,e,i,r){return this.property.evaluate(this.value,this.parameters,t,e,i,r)}}class Ss{constructor(t){this._properties=t,this._values=Object.create(t.defaultPossiblyEvaluatedValues)}get(t){return this._values[t]}}class Is{constructor(t){this.specification=t}possiblyEvaluate(t,e){if(t.isDataDriven())throw new Error("Value should not be data driven");return t.expression.evaluate(e)}interpolate(t,e,i){const r=ni[this.specification.type];return r?r(t,e,i):t}}class As{constructor(t,e){this.specification=t,this.overrides=e}possiblyEvaluate(t,e,i,r){return new Ts(this,"constant"===t.expression.kind||"camera"===t.expression.kind?{kind:"constant",value:t.expression.evaluate(e,null,{},i,r)}:t.expression,e)}interpolate(t,e,i){if("constant"!==t.value.kind||"constant"!==e.value.kind)return t;if(void 0===t.value.value||void 0===e.value.value)return new Ts(this,{kind:"constant",value:void 0},t.parameters);const r=ni[this.specification.type];if(r){const s=r(t.value.value,e.value.value,i);return new Ts(this,{kind:"constant",value:s},t.parameters)}return t}evaluate(t,e,i,r,s,n){return"constant"===t.kind?t.value:t.evaluate(e,i,r,s,n)}}class Es extends As{possiblyEvaluate(t,e,i,r){if(void 0===t.value)return new Ts(this,{kind:"constant",value:void 0},e);if("constant"===t.expression.kind){const s=t.expression.evaluate(e,null,{},i,r),n="resolvedImage"===t.property.specification.type&&"string"!=typeof s?s.name:s,a=this._calculate(n,n,n,e);return new Ts(this,{kind:"constant",value:a},e)}if("camera"===t.expression.kind){const i=this._calculate(t.expression.evaluate({zoom:e.zoom-1}),t.expression.evaluate({zoom:e.zoom}),t.expression.evaluate({zoom:e.zoom+1}),e);return new Ts(this,{kind:"constant",value:i},e)}return new Ts(this,t.expression,e)}evaluate(t,e,i,r,s,n){if("source"===t.kind){const a=t.evaluate(e,i,r,s,n);return this._calculate(a,a,a,e)}return"composite"===t.kind?this._calculate(t.evaluate({zoom:Math.floor(e.zoom)-1},i,r),t.evaluate({zoom:Math.floor(e.zoom)},i,r),t.evaluate({zoom:Math.floor(e.zoom)+1},i,r),e):t.value}_calculate(t,e,i,r){return r.zoom>r.zoomHistory.lastIntegerZoom?{from:t,to:e}:{from:i,to:e}}interpolate(t){return t}}class Cs{constructor(t){this.specification=t}possiblyEvaluate(t,e,i,r){if(void 0!==t.value){if("constant"===t.expression.kind){const s=t.expression.evaluate(e,null,{},i,r);return this._calculate(s,s,s,e)}return this._calculate(t.expression.evaluate(new _s(Math.floor(e.zoom-1),e)),t.expression.evaluate(new _s(Math.floor(e.zoom),e)),t.expression.evaluate(new _s(Math.floor(e.zoom+1),e)),e)}}_calculate(t,e,i,r){return r.zoom>r.zoomHistory.lastIntegerZoom?{from:t,to:e}:{from:i,to:e}}interpolate(t){return t}}class zs{constructor(t){this.specification=t}possiblyEvaluate(t,e,i,r){return!!t.expression.evaluate(e,null,{},i,r)}interpolate(){return!1}}class Ms{constructor(t){this.properties=t,this.defaultPropertyValues={},this.defaultTransitionablePropertyValues={},this.defaultTransitioningPropertyValues={},this.defaultPossiblyEvaluatedValues={},this.overridableProperties=[];for(const e in t){const i=t[e];i.specification.overridable&&this.overridableProperties.push(e);const r=this.defaultPropertyValues[e]=new gs(i,void 0),s=this.defaultTransitionablePropertyValues[e]=new ys(i);this.defaultTransitioningPropertyValues[e]=s.untransitioned(),this.defaultPossiblyEvaluatedValues[e]=r.possiblyEvaluate({})}}}Hr("DataDrivenProperty",As),Hr("DataConstantProperty",Is),Hr("CrossFadedDataDrivenProperty",Es),Hr("CrossFadedProperty",Cs),Hr("ColorRampProperty",zs);const Ps="-transition";class ks extends ot{constructor(t,e){if(super(),this.id=t.id,this.type=t.type,this._featureFilter={filter:()=>!0,needGeometry:!1},"custom"!==t.type&&(this.metadata=t.metadata,this.minzoom=t.minzoom,this.maxzoom=t.maxzoom,"background"!==t.type&&(this.source=t.source,this.sourceLayer=t["source-layer"],this.filter=t.filter),e.layout&&(this._unevaluatedLayout=new ws(e.layout)),e.paint)){this._transitionablePaint=new xs(e.paint);for(const e in t.paint)this.setPaintProperty(e,t.paint[e],{validate:!1});for(const e in t.layout)this.setLayoutProperty(e,t.layout[e],{validate:!1});this._transitioningPaint=this._transitionablePaint.untransitioned(),this.paint=new Ss(e.paint)}}getCrossfadeParameters(){return this._crossfadeParameters}getLayoutProperty(t){return"visibility"===t?this.visibility:this._unevaluatedLayout.getValue(t)}setLayoutProperty(t,e,i={}){null!=e&&this._validate(qr,`layers.${this.id}.layout.${t}`,t,e,i)||("visibility"!==t?this._unevaluatedLayout.setValue(t,e):this.visibility=e)}getPaintProperty(t){return t.endsWith(Ps)?this._transitionablePaint.getTransition(t.slice(0,-11)):this._transitionablePaint.getValue(t)}setPaintProperty(t,e,i={}){if(null!=e&&this._validate($r,`layers.${this.id}.paint.${t}`,t,e,i))return!1;if(t.endsWith(Ps))return this._transitionablePaint.setTransition(t.slice(0,-11),e||void 0),!1;{const i=this._transitionablePaint._values[t],r="cross-faded-data-driven"===i.property.specification["property-type"],s=i.value.isDataDriven(),n=i.value;this._transitionablePaint.setValue(t,e),this._handleSpecialPaintPropertyUpdate(t);const a=this._transitionablePaint._values[t].value;return a.isDataDriven()||s||r||this._handleOverridablePaintPropertyUpdate(t,n,a)}}_handleSpecialPaintPropertyUpdate(t){}_handleOverridablePaintPropertyUpdate(t,e,i){return!1}isHidden(t){return!!(this.minzoom&&t=this.maxzoom)||"none"===this.visibility}updateTransitions(t){this._transitioningPaint=this._transitionablePaint.transitioned(t,this._transitioningPaint)}hasTransition(){return this._transitioningPaint.hasTransition()}recalculate(t,e){t.getCrossfadeParameters&&(this._crossfadeParameters=t.getCrossfadeParameters()),this._unevaluatedLayout&&(this.layout=this._unevaluatedLayout.possiblyEvaluate(t,void 0,e)),this.paint=this._transitioningPaint.possiblyEvaluate(t,void 0,e)}serialize(){const t={id:this.id,type:this.type,source:this.source,"source-layer":this.sourceLayer,metadata:this.metadata,minzoom:this.minzoom,maxzoom:this.maxzoom,filter:this.filter,layout:this._unevaluatedLayout&&this._unevaluatedLayout.serialize(),paint:this._transitionablePaint&&this._transitionablePaint.serialize()};return this.visibility&&(t.layout=t.layout||{},t.layout.visibility=this.visibility),w(t,((t,e)=>!(void 0===t||"layout"===e&&!Object.keys(t).length||"paint"===e&&!Object.keys(t).length)))}_validate(t,e,i,r,s={}){return(!s||!1!==s.validate)&&Zr(this,t.call(Ur,{key:e,layerType:this.type,objectKey:i,value:r,styleSpec:lt,style:{glyphs:!0,sprite:!0}}))}is3D(){return!1}isTileClipped(){return!1}hasOffscreenPass(){return!1}resize(){}isStateDependent(){for(const t in this.paint._values){const e=this.paint.get(t);if(e instanceof Ts&&Bi(e.property.specification)&&("source"===e.value.kind||"composite"===e.value.kind)&&e.value.isStateDependent)return!0}return!1}}const Ds={Int8:Int8Array,Uint8:Uint8Array,Int16:Int16Array,Uint16:Uint16Array,Int32:Int32Array,Uint32:Uint32Array,Float32:Float32Array};class Ls{constructor(t,e){this._structArray=t,this._pos1=e*this.size,this._pos2=this._pos1/2,this._pos4=this._pos1/4,this._pos8=this._pos1/8}}class Rs{constructor(){this.isTransferred=!1,this.capacity=-1,this.resize(0)}static serialize(t,e){return t._trim(),e&&(t.isTransferred=!0,e.push(t.arrayBuffer)),{length:t.length,arrayBuffer:t.arrayBuffer}}static deserialize(t){const e=Object.create(this.prototype);return e.arrayBuffer=t.arrayBuffer,e.length=t.length,e.capacity=t.arrayBuffer.byteLength/e.bytesPerElement,e._refreshViews(),e}_trim(){this.length!==this.capacity&&(this.capacity=this.length,this.arrayBuffer=this.arrayBuffer.slice(0,this.length*this.bytesPerElement),this._refreshViews())}clear(){this.length=0}resize(t){this.reserve(t),this.length=t}reserve(t){if(t>this.capacity){this.capacity=Math.max(t,Math.floor(5*this.capacity),128),this.arrayBuffer=new ArrayBuffer(this.capacity*this.bytesPerElement);const e=this.uint8;this._refreshViews(),e&&this.uint8.set(e)}}_refreshViews(){throw new Error("_refreshViews() must be implemented by each concrete StructArray layout")}}function Bs(t,e=1){let i=0,r=0;return{members:t.map((t=>{const s=Ds[t.type].BYTES_PER_ELEMENT,n=i=Fs(i,Math.max(e,s)),a=t.components||1;return r=Math.max(r,s),i+=s*a,{name:t.name,type:t.type,components:a,offset:n}})),size:Fs(i,Math.max(r,e)),alignment:e}}function Fs(t,e){return Math.ceil(t/e)*e}class Os extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e){const i=this.length;return this.resize(i+1),this.emplace(i,t,e)}emplace(t,e,i){const r=2*t;return this.int16[r+0]=e,this.int16[r+1]=i,t}}Os.prototype.bytesPerElement=4,Hr("StructArrayLayout2i4",Os);class Vs extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e,i){const r=this.length;return this.resize(r+1),this.emplace(r,t,e,i)}emplace(t,e,i,r){const s=3*t;return this.int16[s+0]=e,this.int16[s+1]=i,this.int16[s+2]=r,t}}Vs.prototype.bytesPerElement=6,Hr("StructArrayLayout3i6",Vs);class Us extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e,i,r){const s=this.length;return this.resize(s+1),this.emplace(s,t,e,i,r)}emplace(t,e,i,r,s){const n=4*t;return this.int16[n+0]=e,this.int16[n+1]=i,this.int16[n+2]=r,this.int16[n+3]=s,t}}Us.prototype.bytesPerElement=8,Hr("StructArrayLayout4i8",Us);class Ns extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e,i,r,s,n){const a=this.length;return this.resize(a+1),this.emplace(a,t,e,i,r,s,n)}emplace(t,e,i,r,s,n,a){const o=6*t;return this.int16[o+0]=e,this.int16[o+1]=i,this.int16[o+2]=r,this.int16[o+3]=s,this.int16[o+4]=n,this.int16[o+5]=a,t}}Ns.prototype.bytesPerElement=12,Hr("StructArrayLayout2i4i12",Ns);class $s extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e,i,r,s,n){const a=this.length;return this.resize(a+1),this.emplace(a,t,e,i,r,s,n)}emplace(t,e,i,r,s,n,a){const o=4*t,l=8*t;return this.int16[o+0]=e,this.int16[o+1]=i,this.uint8[l+4]=r,this.uint8[l+5]=s,this.uint8[l+6]=n,this.uint8[l+7]=a,t}}$s.prototype.bytesPerElement=8,Hr("StructArrayLayout2i4ub8",$s);class qs extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e){const i=this.length;return this.resize(i+1),this.emplace(i,t,e)}emplace(t,e,i){const r=2*t;return this.float32[r+0]=e,this.float32[r+1]=i,t}}qs.prototype.bytesPerElement=8,Hr("StructArrayLayout2f8",qs);class Zs extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t,e,i,r,s,n,a,o,l,c){const h=this.length;return this.resize(h+1),this.emplace(h,t,e,i,r,s,n,a,o,l,c)}emplace(t,e,i,r,s,n,a,o,l,c,h){const u=10*t;return this.uint16[u+0]=e,this.uint16[u+1]=i,this.uint16[u+2]=r,this.uint16[u+3]=s,this.uint16[u+4]=n,this.uint16[u+5]=a,this.uint16[u+6]=o,this.uint16[u+7]=l,this.uint16[u+8]=c,this.uint16[u+9]=h,t}}Zs.prototype.bytesPerElement=20,Hr("StructArrayLayout10ui20",Zs);class js extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t,e,i,r,s,n,a,o,l,c,h,u){const p=this.length;return this.resize(p+1),this.emplace(p,t,e,i,r,s,n,a,o,l,c,h,u)}emplace(t,e,i,r,s,n,a,o,l,c,h,u,p){const d=12*t;return this.int16[d+0]=e,this.int16[d+1]=i,this.int16[d+2]=r,this.int16[d+3]=s,this.uint16[d+4]=n,this.uint16[d+5]=a,this.uint16[d+6]=o,this.uint16[d+7]=l,this.int16[d+8]=c,this.int16[d+9]=h,this.int16[d+10]=u,this.int16[d+11]=p,t}}js.prototype.bytesPerElement=24,Hr("StructArrayLayout4i4ui4i24",js);class Gs extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e,i){const r=this.length;return this.resize(r+1),this.emplace(r,t,e,i)}emplace(t,e,i,r){const s=3*t;return this.float32[s+0]=e,this.float32[s+1]=i,this.float32[s+2]=r,t}}Gs.prototype.bytesPerElement=12,Hr("StructArrayLayout3f12",Gs);class Hs extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer)}emplaceBack(t){const e=this.length;return this.resize(e+1),this.emplace(e,t)}emplace(t,e){return this.uint32[1*t+0]=e,t}}Hs.prototype.bytesPerElement=4,Hr("StructArrayLayout1ul4",Hs);class Ws extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t,e,i,r,s,n,a,o,l){const c=this.length;return this.resize(c+1),this.emplace(c,t,e,i,r,s,n,a,o,l)}emplace(t,e,i,r,s,n,a,o,l,c){const h=10*t,u=5*t;return this.int16[h+0]=e,this.int16[h+1]=i,this.int16[h+2]=r,this.int16[h+3]=s,this.int16[h+4]=n,this.int16[h+5]=a,this.uint32[u+3]=o,this.uint16[h+8]=l,this.uint16[h+9]=c,t}}Ws.prototype.bytesPerElement=20,Hr("StructArrayLayout6i1ul2ui20",Ws);class Xs extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e,i,r,s,n){const a=this.length;return this.resize(a+1),this.emplace(a,t,e,i,r,s,n)}emplace(t,e,i,r,s,n,a){const o=6*t;return this.int16[o+0]=e,this.int16[o+1]=i,this.int16[o+2]=r,this.int16[o+3]=s,this.int16[o+4]=n,this.int16[o+5]=a,t}}Xs.prototype.bytesPerElement=12,Hr("StructArrayLayout2i2i2i12",Xs);class Ks extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer)}emplaceBack(t,e,i,r,s){const n=this.length;return this.resize(n+1),this.emplace(n,t,e,i,r,s)}emplace(t,e,i,r,s,n){const a=4*t,o=8*t;return this.float32[a+0]=e,this.float32[a+1]=i,this.float32[a+2]=r,this.int16[o+6]=s,this.int16[o+7]=n,t}}Ks.prototype.bytesPerElement=16,Hr("StructArrayLayout2f1f2i16",Ks);class Ys extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e,i,r){const s=this.length;return this.resize(s+1),this.emplace(s,t,e,i,r)}emplace(t,e,i,r,s){const n=12*t,a=3*t;return this.uint8[n+0]=e,this.uint8[n+1]=i,this.float32[a+1]=r,this.float32[a+2]=s,t}}Ys.prototype.bytesPerElement=12,Hr("StructArrayLayout2ub2f12",Ys);class Js extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t,e,i){const r=this.length;return this.resize(r+1),this.emplace(r,t,e,i)}emplace(t,e,i,r){const s=3*t;return this.uint16[s+0]=e,this.uint16[s+1]=i,this.uint16[s+2]=r,t}}Js.prototype.bytesPerElement=6,Hr("StructArrayLayout3ui6",Js);class Qs extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e,i,r,s,n,a,o,l,c,h,u,p,d,m,f,_){const g=this.length;return this.resize(g+1),this.emplace(g,t,e,i,r,s,n,a,o,l,c,h,u,p,d,m,f,_)}emplace(t,e,i,r,s,n,a,o,l,c,h,u,p,d,m,f,_,g){const y=24*t,x=12*t,v=48*t;return this.int16[y+0]=e,this.int16[y+1]=i,this.uint16[y+2]=r,this.uint16[y+3]=s,this.uint32[x+2]=n,this.uint32[x+3]=a,this.uint32[x+4]=o,this.uint16[y+10]=l,this.uint16[y+11]=c,this.uint16[y+12]=h,this.float32[x+7]=u,this.float32[x+8]=p,this.uint8[v+36]=d,this.uint8[v+37]=m,this.uint8[v+38]=f,this.uint32[x+10]=_,this.int16[y+22]=g,t}}Qs.prototype.bytesPerElement=48,Hr("StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48",Qs);class tn extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e,i,r,s,n,a,o,l,c,h,u,p,d,m,f,_,g,y,x,v,b,w,T,S,I,A,E){const C=this.length;return this.resize(C+1),this.emplace(C,t,e,i,r,s,n,a,o,l,c,h,u,p,d,m,f,_,g,y,x,v,b,w,T,S,I,A,E)}emplace(t,e,i,r,s,n,a,o,l,c,h,u,p,d,m,f,_,g,y,x,v,b,w,T,S,I,A,E,C){const z=32*t,M=16*t;return this.int16[z+0]=e,this.int16[z+1]=i,this.int16[z+2]=r,this.int16[z+3]=s,this.int16[z+4]=n,this.int16[z+5]=a,this.int16[z+6]=o,this.int16[z+7]=l,this.uint16[z+8]=c,this.uint16[z+9]=h,this.uint16[z+10]=u,this.uint16[z+11]=p,this.uint16[z+12]=d,this.uint16[z+13]=m,this.uint16[z+14]=f,this.uint16[z+15]=_,this.uint16[z+16]=g,this.uint16[z+17]=y,this.uint16[z+18]=x,this.uint16[z+19]=v,this.uint16[z+20]=b,this.uint16[z+21]=w,this.uint16[z+22]=T,this.uint32[M+12]=S,this.float32[M+13]=I,this.float32[M+14]=A,this.uint16[z+30]=E,this.uint16[z+31]=C,t}}tn.prototype.bytesPerElement=64,Hr("StructArrayLayout8i15ui1ul2f2ui64",tn);class en extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t){const e=this.length;return this.resize(e+1),this.emplace(e,t)}emplace(t,e){return this.float32[1*t+0]=e,t}}en.prototype.bytesPerElement=4,Hr("StructArrayLayout1f4",en);class rn extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e,i){const r=this.length;return this.resize(r+1),this.emplace(r,t,e,i)}emplace(t,e,i,r){const s=3*t;return this.uint16[6*t+0]=e,this.float32[s+1]=i,this.float32[s+2]=r,t}}rn.prototype.bytesPerElement=12,Hr("StructArrayLayout1ui2f12",rn);class sn extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t,e,i){const r=this.length;return this.resize(r+1),this.emplace(r,t,e,i)}emplace(t,e,i,r){const s=4*t;return this.uint32[2*t+0]=e,this.uint16[s+2]=i,this.uint16[s+3]=r,t}}sn.prototype.bytesPerElement=8,Hr("StructArrayLayout1ul2ui8",sn);class nn extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t,e){const i=this.length;return this.resize(i+1),this.emplace(i,t,e)}emplace(t,e,i){const r=2*t;return this.uint16[r+0]=e,this.uint16[r+1]=i,t}}nn.prototype.bytesPerElement=4,Hr("StructArrayLayout2ui4",nn);class an extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer)}emplaceBack(t){const e=this.length;return this.resize(e+1),this.emplace(e,t)}emplace(t,e){return this.uint16[1*t+0]=e,t}}an.prototype.bytesPerElement=2,Hr("StructArrayLayout1ui2",an);class on extends Rs{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer)}emplaceBack(t,e,i,r){const s=this.length;return this.resize(s+1),this.emplace(s,t,e,i,r)}emplace(t,e,i,r,s){const n=4*t;return this.float32[n+0]=e,this.float32[n+1]=i,this.float32[n+2]=r,this.float32[n+3]=s,t}}on.prototype.bytesPerElement=16,Hr("StructArrayLayout4f16",on);class ln extends Ls{get anchorPointX(){return this._structArray.int16[this._pos2+0]}get anchorPointY(){return this._structArray.int16[this._pos2+1]}get x1(){return this._structArray.int16[this._pos2+2]}get y1(){return this._structArray.int16[this._pos2+3]}get x2(){return this._structArray.int16[this._pos2+4]}get y2(){return this._structArray.int16[this._pos2+5]}get featureIndex(){return this._structArray.uint32[this._pos4+3]}get sourceLayerIndex(){return this._structArray.uint16[this._pos2+8]}get bucketIndex(){return this._structArray.uint16[this._pos2+9]}get anchorPoint(){return new n(this.anchorPointX,this.anchorPointY)}}ln.prototype.size=20;class cn extends Ws{get(t){return new ln(this,t)}}Hr("CollisionBoxArray",cn);class hn extends Ls{get anchorX(){return this._structArray.int16[this._pos2+0]}get anchorY(){return this._structArray.int16[this._pos2+1]}get glyphStartIndex(){return this._structArray.uint16[this._pos2+2]}get numGlyphs(){return this._structArray.uint16[this._pos2+3]}get vertexStartIndex(){return this._structArray.uint32[this._pos4+2]}get lineStartIndex(){return this._structArray.uint32[this._pos4+3]}get lineLength(){return this._structArray.uint32[this._pos4+4]}get segment(){return this._structArray.uint16[this._pos2+10]}get lowerSize(){return this._structArray.uint16[this._pos2+11]}get upperSize(){return this._structArray.uint16[this._pos2+12]}get lineOffsetX(){return this._structArray.float32[this._pos4+7]}get lineOffsetY(){return this._structArray.float32[this._pos4+8]}get writingMode(){return this._structArray.uint8[this._pos1+36]}get placedOrientation(){return this._structArray.uint8[this._pos1+37]}set placedOrientation(t){this._structArray.uint8[this._pos1+37]=t}get hidden(){return this._structArray.uint8[this._pos1+38]}set hidden(t){this._structArray.uint8[this._pos1+38]=t}get crossTileID(){return this._structArray.uint32[this._pos4+10]}set crossTileID(t){this._structArray.uint32[this._pos4+10]=t}get associatedIconIndex(){return this._structArray.int16[this._pos2+22]}}hn.prototype.size=48;class un extends Qs{get(t){return new hn(this,t)}}Hr("PlacedSymbolArray",un);class pn extends Ls{get anchorX(){return this._structArray.int16[this._pos2+0]}get anchorY(){return this._structArray.int16[this._pos2+1]}get rightJustifiedTextSymbolIndex(){return this._structArray.int16[this._pos2+2]}get centerJustifiedTextSymbolIndex(){return this._structArray.int16[this._pos2+3]}get leftJustifiedTextSymbolIndex(){return this._structArray.int16[this._pos2+4]}get verticalPlacedTextSymbolIndex(){return this._structArray.int16[this._pos2+5]}get placedIconSymbolIndex(){return this._structArray.int16[this._pos2+6]}get verticalPlacedIconSymbolIndex(){return this._structArray.int16[this._pos2+7]}get key(){return this._structArray.uint16[this._pos2+8]}get textBoxStartIndex(){return this._structArray.uint16[this._pos2+9]}get textBoxEndIndex(){return this._structArray.uint16[this._pos2+10]}get verticalTextBoxStartIndex(){return this._structArray.uint16[this._pos2+11]}get verticalTextBoxEndIndex(){return this._structArray.uint16[this._pos2+12]}get iconBoxStartIndex(){return this._structArray.uint16[this._pos2+13]}get iconBoxEndIndex(){return this._structArray.uint16[this._pos2+14]}get verticalIconBoxStartIndex(){return this._structArray.uint16[this._pos2+15]}get verticalIconBoxEndIndex(){return this._structArray.uint16[this._pos2+16]}get featureIndex(){return this._structArray.uint16[this._pos2+17]}get numHorizontalGlyphVertices(){return this._structArray.uint16[this._pos2+18]}get numVerticalGlyphVertices(){return this._structArray.uint16[this._pos2+19]}get numIconVertices(){return this._structArray.uint16[this._pos2+20]}get numVerticalIconVertices(){return this._structArray.uint16[this._pos2+21]}get useRuntimeCollisionCircles(){return this._structArray.uint16[this._pos2+22]}get crossTileID(){return this._structArray.uint32[this._pos4+12]}set crossTileID(t){this._structArray.uint32[this._pos4+12]=t}get textBoxScale(){return this._structArray.float32[this._pos4+13]}get collisionCircleDiameter(){return this._structArray.float32[this._pos4+14]}get textAnchorOffsetStartIndex(){return this._structArray.uint16[this._pos2+30]}get textAnchorOffsetEndIndex(){return this._structArray.uint16[this._pos2+31]}}pn.prototype.size=64;class dn extends tn{get(t){return new pn(this,t)}}Hr("SymbolInstanceArray",dn);class mn extends en{getoffsetX(t){return this.float32[1*t+0]}}Hr("GlyphOffsetArray",mn);class fn extends Vs{getx(t){return this.int16[3*t+0]}gety(t){return this.int16[3*t+1]}gettileUnitDistanceFromAnchor(t){return this.int16[3*t+2]}}Hr("SymbolLineVertexArray",fn);class _n extends Ls{get textAnchor(){return this._structArray.uint16[this._pos2+0]}get textOffset0(){return this._structArray.float32[this._pos4+1]}get textOffset1(){return this._structArray.float32[this._pos4+2]}}_n.prototype.size=12;class gn extends rn{get(t){return new _n(this,t)}}Hr("TextAnchorOffsetArray",gn);class yn extends Ls{get featureIndex(){return this._structArray.uint32[this._pos4+0]}get sourceLayerIndex(){return this._structArray.uint16[this._pos2+2]}get bucketIndex(){return this._structArray.uint16[this._pos2+3]}}yn.prototype.size=8;class xn extends sn{get(t){return new yn(this,t)}}Hr("FeatureIndexArray",xn);class vn extends Os{}class bn extends Vs{}class wn extends Us{}class Tn extends Os{}class Sn extends Os{}class In extends Ns{}class An extends $s{}class En extends qs{}class Cn extends Zs{}class zn extends js{}class Mn extends Gs{}class Pn extends Hs{}class kn extends Xs{}class Dn extends Ks{}class Ln extends Ys{}class Rn extends Js{}class Bn extends Js{}class Fn extends nn{}class On extends an{}const Vn=Bs([{name:"a_pos",components:2,type:"Int16"}],4),{members:Un}=Vn;class Nn{constructor(t=[]){this.segments=t}prepareSegment(t,e,i,r){let s=this.segments[this.segments.length-1];return t>Nn.MAX_VERTEX_ARRAY_LENGTH&&A(`Max vertices per segment is ${Nn.MAX_VERTEX_ARRAY_LENGTH}: bucket requested ${t}`),(!s||s.vertexLength+t>Nn.MAX_VERTEX_ARRAY_LENGTH||s.sortKey!==r)&&(s={vertexOffset:e.length,primitiveOffset:i.length,vertexLength:0,primitiveLength:0},void 0!==r&&(s.sortKey=r),this.segments.push(s)),s}get(){return this.segments}destroy(){for(const t of this.segments)for(const e in t.vaos)t.vaos[e].destroy()}static simpleSegment(t,e,i,r){return new Nn([{vertexOffset:t,primitiveOffset:e,vertexLength:i,primitiveLength:r,vaos:{},sortKey:0}])}}function $n(t,e){return 256*(t=m(Math.floor(t),0,255))+m(Math.floor(e),0,255)}Nn.MAX_VERTEX_ARRAY_LENGTH=Math.pow(2,16)-1,Hr("SegmentVector",Nn);const qn=Bs([{name:"a_pattern_from",components:4,type:"Uint16"},{name:"a_pattern_to",components:4,type:"Uint16"},{name:"a_pixel_ratio_from",components:1,type:"Uint16"},{name:"a_pixel_ratio_to",components:1,type:"Uint16"}]);var Zn={exports:{}},jn={exports:{}};!function(t){t.exports=function(t,e){var i,r,s,n,a,o,l,c;for(r=t.length-(i=3&t.length),s=e,a=3432918353,o=461845907,c=0;c>>16)*a&65535)<<16)&4294967295)<<15|l>>>17))*o+(((l>>>16)*o&65535)<<16)&4294967295)<<13|s>>>19))+((5*(s>>>16)&65535)<<16)&4294967295))+((58964+(n>>>16)&65535)<<16);switch(l=0,i){case 3:l^=(255&t.charCodeAt(c+2))<<16;case 2:l^=(255&t.charCodeAt(c+1))<<8;case 1:s^=l=(65535&(l=(l=(65535&(l^=255&t.charCodeAt(c)))*a+(((l>>>16)*a&65535)<<16)&4294967295)<<15|l>>>17))*o+(((l>>>16)*o&65535)<<16)&4294967295}return s^=t.length,s=2246822507*(65535&(s^=s>>>16))+((2246822507*(s>>>16)&65535)<<16)&4294967295,s=3266489909*(65535&(s^=s>>>13))+((3266489909*(s>>>16)&65535)<<16)&4294967295,(s^=s>>>16)>>>0}}(jn);var Gn=jn.exports,Hn={exports:{}};!function(t){t.exports=function(t,e){for(var i,r=t.length,s=e^r,n=0;r>=4;)i=1540483477*(65535&(i=255&t.charCodeAt(n)|(255&t.charCodeAt(++n))<<8|(255&t.charCodeAt(++n))<<16|(255&t.charCodeAt(++n))<<24))+((1540483477*(i>>>16)&65535)<<16),s=1540483477*(65535&s)+((1540483477*(s>>>16)&65535)<<16)^(i=1540483477*(65535&(i^=i>>>24))+((1540483477*(i>>>16)&65535)<<16)),r-=4,++n;switch(r){case 3:s^=(255&t.charCodeAt(n+2))<<16;case 2:s^=(255&t.charCodeAt(n+1))<<8;case 1:s=1540483477*(65535&(s^=255&t.charCodeAt(n)))+((1540483477*(s>>>16)&65535)<<16)}return s=1540483477*(65535&(s^=s>>>13))+((1540483477*(s>>>16)&65535)<<16),(s^=s>>>15)>>>0}}(Hn);var Wn=Gn,Xn=Hn.exports;Zn.exports=Wn,Zn.exports.murmur3=Wn,Zn.exports.murmur2=Xn;var Kn=i(Zn.exports);class Yn{constructor(){this.ids=[],this.positions=[],this.indexed=!1}add(t,e,i,r){this.ids.push(Jn(t)),this.positions.push(e,i,r)}getPositions(t){if(!this.indexed)throw new Error("Trying to get index, but feature positions are not indexed");const e=Jn(t);let i=0,r=this.ids.length-1;for(;i>1;this.ids[t]>=e?r=t:i=t+1}const s=[];for(;this.ids[i]===e;)s.push({index:this.positions[3*i],start:this.positions[3*i+1],end:this.positions[3*i+2]}),i++;return s}static serialize(t,e){const i=new Float64Array(t.ids),r=new Uint32Array(t.positions);return Qn(i,r,0,i.length-1),e&&e.push(i.buffer,r.buffer),{ids:i,positions:r}}static deserialize(t){const e=new Yn;return e.ids=t.ids,e.positions=t.positions,e.indexed=!0,e}}function Jn(t){const e=+t;return!isNaN(e)&&e<=Number.MAX_SAFE_INTEGER?e:Kn(String(t))}function Qn(t,e,i,r){for(;i>1];let n=i-1,a=r+1;for(;;){do{n++}while(t[n]s);if(n>=a)break;ta(t,n,a),ta(e,3*n,3*a),ta(e,3*n+1,3*a+1),ta(e,3*n+2,3*a+2)}a-i`u_${t}`)),this.type=i}setUniform(t,e,i){t.set(i.constantOr(this.value))}getBinding(t,e,i){return"color"===this.type?new oa(t,e):new ra(t,e)}}class pa{constructor(t,e){this.uniformNames=e.map((t=>`u_${t}`)),this.patternFrom=null,this.patternTo=null,this.pixelRatioFrom=1,this.pixelRatioTo=1}setConstantPatternPositions(t,e){this.pixelRatioFrom=e.pixelRatio,this.pixelRatioTo=t.pixelRatio,this.patternFrom=e.tlbr,this.patternTo=t.tlbr}setUniform(t,e,i,r){const s="u_pattern_to"===r?this.patternTo:"u_pattern_from"===r?this.patternFrom:"u_pixel_ratio_to"===r?this.pixelRatioTo:"u_pixel_ratio_from"===r?this.pixelRatioFrom:null;s&&t.set(s)}getBinding(t,e,i){return"u_pattern"===i.substr(0,9)?new aa(t,e):new ra(t,e)}}class da{constructor(t,e,i,r){this.expression=t,this.type=i,this.maxValue=0,this.paintVertexAttributes=e.map((t=>({name:`a_${t}`,type:"Float32",components:"color"===i?2:1,offset:0}))),this.paintVertexArray=new r}populatePaintArray(t,e,i,r,s){const n=this.paintVertexArray.length,a=this.expression.evaluate(new _s(0),e,{},r,[],s);this.paintVertexArray.resize(t),this._setPaintValue(n,t,a)}updatePaintArray(t,e,i,r){const s=this.expression.evaluate({zoom:0},i,r);this._setPaintValue(t,e,s)}_setPaintValue(t,e,i){if("color"===this.type){const r=ha(i);for(let i=t;i`u_${t}_t`)),this.type=i,this.useIntegerZoom=r,this.zoom=s,this.maxValue=0,this.paintVertexAttributes=e.map((t=>({name:`a_${t}`,type:"Float32",components:"color"===i?4:2,offset:0}))),this.paintVertexArray=new n}populatePaintArray(t,e,i,r,s){const n=this.expression.evaluate(new _s(this.zoom),e,{},r,[],s),a=this.expression.evaluate(new _s(this.zoom+1),e,{},r,[],s),o=this.paintVertexArray.length;this.paintVertexArray.resize(t),this._setPaintValue(o,t,n,a)}updatePaintArray(t,e,i,r){const s=this.expression.evaluate({zoom:this.zoom},i,r),n=this.expression.evaluate({zoom:this.zoom+1},i,r);this._setPaintValue(t,e,s,n)}_setPaintValue(t,e,i,r){if("color"===this.type){const s=ha(i),n=ha(r);for(let i=t;i`#define HAS_UNIFORM_${t}`)))}return t}getBinderAttributes(){const t=[];for(const e in this.binders){const i=this.binders[e];if(i instanceof da||i instanceof ma)for(let e=0;e!0)){this.programConfigurations={};for(const r of t)this.programConfigurations[r.id]=new _a(r,e,i);this.needsUpload=!1,this._featureMap=new Yn,this._bufferOffset=0}populatePaintArrays(t,e,i,r,s,n){for(const i in this.programConfigurations)this.programConfigurations[i].populatePaintArrays(t,e,r,s,n);void 0!==e.id&&this._featureMap.add(e.id,i,this._bufferOffset,t),this._bufferOffset=t,this.needsUpload=!0}updatePaintArrays(t,e,i,r){for(const s of i)this.needsUpload=this.programConfigurations[s.id].updatePaintArrays(t,this._featureMap,e,s,r)||this.needsUpload}get(t){return this.programConfigurations[t]}upload(t){if(this.needsUpload){for(const e in this.programConfigurations)this.programConfigurations[e].upload(t);this.needsUpload=!1}}destroy(){for(const t in this.programConfigurations)this.programConfigurations[t].destroy()}}function ya(t,e){return{"text-opacity":["opacity"],"icon-opacity":["opacity"],"text-color":["fill_color"],"icon-color":["fill_color"],"text-halo-color":["halo_color"],"icon-halo-color":["halo_color"],"text-halo-blur":["halo_blur"],"icon-halo-blur":["halo_blur"],"text-halo-width":["halo_width"],"icon-halo-width":["halo_width"],"line-gap-width":["gapwidth"],"line-pattern":["pattern_to","pattern_from","pixel_ratio_to","pixel_ratio_from"],"fill-pattern":["pattern_to","pattern_from","pixel_ratio_to","pixel_ratio_from"],"fill-extrusion-pattern":["pattern_to","pattern_from","pixel_ratio_to","pixel_ratio_from"]}[t]||[t.replace(`${e}-`,"").replace(/-/g,"_")]}function xa(t,e,i){const r={color:{source:qs,composite:on},number:{source:en,composite:qs}},s=function(t){return{"line-pattern":{source:Cn,composite:Cn},"fill-pattern":{source:Cn,composite:Cn},"fill-extrusion-pattern":{source:Cn,composite:Cn}}[t]}(t);return s&&s[i]||r[e][i]}Hr("ConstantBinder",ua),Hr("CrossFadedConstantBinder",pa),Hr("SourceExpressionBinder",da),Hr("CrossFadedCompositeBinder",fa),Hr("CompositeExpressionBinder",ma),Hr("ProgramConfiguration",_a,{omit:["_buffers"]}),Hr("ProgramConfigurationSet",ga);const va=8192,ba=Math.pow(2,14)-1,wa=-ba-1;function Ta(t){const e=va/t.extent,i=t.loadGeometry();for(let t=0;ti.x+1||ni.y+1)&&A("Geometry exceeds allowed extent, reduce your vector tile buffer size")}}return i}function Sa(t,e){return{type:t.type,id:t.id,properties:t.properties,geometry:e?Ta(t):[]}}function Ia(t,e,i,r,s){t.emplaceBack(2*e+(r+1)/2,2*i+(s+1)/2)}class Aa{constructor(t){this.zoom=t.zoom,this.overscaling=t.overscaling,this.layers=t.layers,this.layerIds=this.layers.map((t=>t.id)),this.index=t.index,this.hasPattern=!1,this.layoutVertexArray=new Tn,this.indexArray=new Bn,this.segments=new Nn,this.programConfigurations=new ga(t.layers,t.zoom),this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id))}populate(t,e,i){const r=this.layers[0],s=[];let n=null,a=!1;"circle"===r.type&&(n=r.layout.get("circle-sort-key"),a=!n.isConstant());for(const{feature:e,id:r,index:o,sourceLayerIndex:l}of t){const t=this.layers[0]._featureFilter.needGeometry,c=Sa(e,t);if(!this.layers[0]._featureFilter.filter(new _s(this.zoom),c,i))continue;const h=a?n.evaluate(c,{},i):void 0,u={id:r,properties:e.properties,type:e.type,sourceLayerIndex:l,index:o,geometry:t?c.geometry:Ta(e),patterns:{},sortKey:h};s.push(u)}a&&s.sort(((t,e)=>t.sortKey-e.sortKey));for(const r of s){const{geometry:s,index:n,sourceLayerIndex:a}=r,o=t[n].feature;this.addFeature(r,s,n,i),e.featureIndex.insert(o,s,n,a,this.index)}}update(t,e,i){this.stateDependentLayers.length&&this.programConfigurations.updatePaintArrays(t,e,this.stateDependentLayers,i)}isEmpty(){return 0===this.layoutVertexArray.length}uploadPending(){return!this.uploaded||this.programConfigurations.needsUpload}upload(t){this.uploaded||(this.layoutVertexBuffer=t.createVertexBuffer(this.layoutVertexArray,Un),this.indexBuffer=t.createIndexBuffer(this.indexArray)),this.programConfigurations.upload(t),this.uploaded=!0}destroy(){this.layoutVertexBuffer&&(this.layoutVertexBuffer.destroy(),this.indexBuffer.destroy(),this.programConfigurations.destroy(),this.segments.destroy())}addFeature(t,e,i,r){for(const i of e)for(const e of i){const i=e.x,r=e.y;if(i<0||i>=va||r<0||r>=va)continue;const s=this.segments.prepareSegment(4,this.layoutVertexArray,this.indexArray,t.sortKey),n=s.vertexLength;Ia(this.layoutVertexArray,i,r,-1,-1),Ia(this.layoutVertexArray,i,r,1,-1),Ia(this.layoutVertexArray,i,r,1,1),Ia(this.layoutVertexArray,i,r,-1,1),this.indexArray.emplaceBack(n,n+1,n+2),this.indexArray.emplaceBack(n,n+3,n+2),s.vertexLength+=4,s.primitiveLength+=2}this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length,t,i,{},r)}}function Ea(t,e){for(let i=0;i1){if(Pa(t,e))return!0;for(let r=0;r1?i:i.sub(e)._mult(s)._add(e))}function Ra(t,e){let i,r,s,n=!1;for(let a=0;ae.y!=s.y>e.y&&e.x<(s.x-r.x)*(e.y-r.y)/(s.y-r.y)+r.x&&(n=!n)}return n}function Ba(t,e){let i=!1;for(let r=0,s=t.length-1;re.y!=a.y>e.y&&e.x<(a.x-n.x)*(e.y-n.y)/(a.y-n.y)+n.x&&(i=!i)}return i}function Fa(t,e,i){const r=i[0],s=i[2];if(t.xs.x&&e.x>s.x||t.ys.y&&e.y>s.y)return!1;const n=E(t,e,i[0]);return n!==E(t,e,i[1])||n!==E(t,e,i[2])||n!==E(t,e,i[3])}function Oa(t,e,i){const r=e.paint.get(t).value;return"constant"===r.kind?r.value:i.programConfigurations.get(e.id).getMaxValue(t)}function Va(t){return Math.sqrt(t[0]*t[0]+t[1]*t[1])}function Ua(t,e,i,r,s){if(!e[0]&&!e[1])return t;const a=n.convert(e)._mult(s);"viewport"===i&&a._rotate(-r);const o=[];for(let e=0;eoo(t,e)))}(l,o),p=h?c*a:c;for(const t of r)for(const e of t){const t=h?e:oo(e,o);let i=p;const r=so([],[e.x,e.y,0,1],o);if("viewport"===this.paint.get("circle-pitch-scale")&&"map"===this.paint.get("circle-pitch-alignment")?i*=r[3]/n.cameraToCenterDistance:"map"===this.paint.get("circle-pitch-scale")&&"viewport"===this.paint.get("circle-pitch-alignment")&&(i*=n.cameraToCenterDistance/r[3]),Ca(u,t,i))return!0}return!1}}function oo(t,e){const i=so([],[t.x,t.y,0,1],e);return new n(i[0]/i[3],i[1]/i[3])}class lo extends Aa{}let co;Hr("HeatmapBucket",lo,{omit:["layers"]});var ho={get paint(){return co=co||new Ms({"heatmap-radius":new As(lt.paint_heatmap["heatmap-radius"]),"heatmap-weight":new As(lt.paint_heatmap["heatmap-weight"]),"heatmap-intensity":new Is(lt.paint_heatmap["heatmap-intensity"]),"heatmap-color":new zs(lt.paint_heatmap["heatmap-color"]),"heatmap-opacity":new Is(lt.paint_heatmap["heatmap-opacity"])})}};function uo(t,{width:e,height:i},r,s){if(s){if(s instanceof Uint8ClampedArray)s=new Uint8Array(s.buffer);else if(s.length!==e*i*r)throw new RangeError(`mismatched image size. expected: ${s.length} but got: ${e*i*r}`)}else s=new Uint8Array(e*i*r);return t.width=e,t.height=i,t.data=s,t}function po(t,{width:e,height:i},r){if(e===t.width&&i===t.height)return;const s=uo({},{width:e,height:i},r);mo(t,s,{x:0,y:0},{x:0,y:0},{width:Math.min(t.width,e),height:Math.min(t.height,i)},r),t.width=e,t.height=i,t.data=s.data}function mo(t,e,i,r,s,n){if(0===s.width||0===s.height)return e;if(s.width>t.width||s.height>t.height||i.x>t.width-s.width||i.y>t.height-s.height)throw new RangeError("out of range source coordinates for image copy");if(s.width>e.width||s.height>e.height||r.x>e.width-s.width||r.y>e.height-s.height)throw new RangeError("out of range destination coordinates for image copy");const a=t.data,o=e.data;if(a===o)throw new Error("srcData equals dstData, so image is already copied");for(let l=0;l{e[t.evaluationKey]=n;const a=t.expression.evaluate(e);s.data[i+r+0]=Math.floor(255*a.r/a.a),s.data[i+r+1]=Math.floor(255*a.g/a.a),s.data[i+r+2]=Math.floor(255*a.b/a.a),s.data[i+r+3]=Math.floor(255*a.a)};if(t.clips)for(let e=0,s=0;e80*i){r=n=t[0],s=a=t[1];for(var m=i;mn&&(n=o),l>a&&(a=l);c=0!==(c=Math.max(n-r,a-s))?32767/c:0}return Co(p,d,i,r,s,c,0),d}function Ao(t,e,i,r,s){var n,a;if(s===Ko(t,e,i,r)>0)for(n=e;n=e;n-=r)a=Ho(n,t[n],t[n+1],a);return a&&No(a,a.next)&&(Wo(a),a=a.next),a}function Eo(t,e){if(!t)return t;e||(e=t);var i,r=t;do{if(i=!1,r.steiner||!No(r,r.next)&&0!==Uo(r.prev,r,r.next))r=r.next;else{if(Wo(r),(r=e=r.prev)===r.next)break;i=!0}}while(i||r!==e);return e}function Co(t,e,i,r,s,n,a){if(t){!a&&n&&function(t,e,i,r){var s=t;do{0===s.z&&(s.z=Bo(s.x,s.y,e,i,r)),s.prevZ=s.prev,s.nextZ=s.next,s=s.next}while(s!==t);s.prevZ.nextZ=null,s.prevZ=null,function(t){var e,i,r,s,n,a,o,l,c=1;do{for(i=t,t=null,n=null,a=0;i;){for(a++,r=i,o=0,e=0;e0||l>0&&r;)0!==o&&(0===l||!r||i.z<=r.z)?(s=i,i=i.nextZ,o--):(s=r,r=r.nextZ,l--),n?n.nextZ=s:t=s,s.prevZ=n,n=s;i=r}n.nextZ=null,c*=2}while(a>1)}(s)}(t,r,s,n);for(var o,l,c=t;t.prev!==t.next;)if(o=t.prev,l=t.next,n?Mo(t,r,s,n):zo(t))e.push(o.i/i|0),e.push(t.i/i|0),e.push(l.i/i|0),Wo(t),t=l.next,c=l.next;else if((t=l)===c){a?1===a?Co(t=Po(Eo(t),e,i),e,i,r,s,n,2):2===a&&ko(t,e,i,r,s,n):Co(Eo(t),e,i,r,s,n,1);break}}}function zo(t){var e=t.prev,i=t,r=t.next;if(Uo(e,i,r)>=0)return!1;for(var s=e.x,n=i.x,a=r.x,o=e.y,l=i.y,c=r.y,h=sn?s>a?s:a:n>a?n:a,d=o>l?o>c?o:c:l>c?l:c,m=r.next;m!==e;){if(m.x>=h&&m.x<=p&&m.y>=u&&m.y<=d&&Oo(s,o,n,l,a,c,m.x,m.y)&&Uo(m.prev,m,m.next)>=0)return!1;m=m.next}return!0}function Mo(t,e,i,r){var s=t.prev,n=t,a=t.next;if(Uo(s,n,a)>=0)return!1;for(var o=s.x,l=n.x,c=a.x,h=s.y,u=n.y,p=a.y,d=ol?o>c?o:c:l>c?l:c,_=h>u?h>p?h:p:u>p?u:p,g=Bo(d,m,e,i,r),y=Bo(f,_,e,i,r),x=t.prevZ,v=t.nextZ;x&&x.z>=g&&v&&v.z<=y;){if(x.x>=d&&x.x<=f&&x.y>=m&&x.y<=_&&x!==s&&x!==a&&Oo(o,h,l,u,c,p,x.x,x.y)&&Uo(x.prev,x,x.next)>=0)return!1;if(x=x.prevZ,v.x>=d&&v.x<=f&&v.y>=m&&v.y<=_&&v!==s&&v!==a&&Oo(o,h,l,u,c,p,v.x,v.y)&&Uo(v.prev,v,v.next)>=0)return!1;v=v.nextZ}for(;x&&x.z>=g;){if(x.x>=d&&x.x<=f&&x.y>=m&&x.y<=_&&x!==s&&x!==a&&Oo(o,h,l,u,c,p,x.x,x.y)&&Uo(x.prev,x,x.next)>=0)return!1;x=x.prevZ}for(;v&&v.z<=y;){if(v.x>=d&&v.x<=f&&v.y>=m&&v.y<=_&&v!==s&&v!==a&&Oo(o,h,l,u,c,p,v.x,v.y)&&Uo(v.prev,v,v.next)>=0)return!1;v=v.nextZ}return!0}function Po(t,e,i){var r=t;do{var s=r.prev,n=r.next.next;!No(s,n)&&$o(s,r,r.next,n)&&jo(s,n)&&jo(n,s)&&(e.push(s.i/i|0),e.push(r.i/i|0),e.push(n.i/i|0),Wo(r),Wo(r.next),r=t=n),r=r.next}while(r!==t);return Eo(r)}function ko(t,e,i,r,s,n){var a=t;do{for(var o=a.next.next;o!==a.prev;){if(a.i!==o.i&&Vo(a,o)){var l=Go(a,o);return a=Eo(a,a.next),l=Eo(l,l.next),Co(a,e,i,r,s,n,0),void Co(l,e,i,r,s,n,0)}o=o.next}a=a.next}while(a!==t)}function Do(t,e){return t.x-e.x}function Lo(t,e){var i=function(t,e){var i,r=e,s=t.x,n=t.y,a=-1/0;do{if(n<=r.y&&n>=r.next.y&&r.next.y!==r.y){var o=r.x+(n-r.y)*(r.next.x-r.x)/(r.next.y-r.y);if(o<=s&&o>a&&(a=o,i=r.x=r.x&&r.x>=h&&s!==r.x&&Oo(ni.x||r.x===i.x&&Ro(i,r)))&&(i=r,p=l)),r=r.next}while(r!==c);return i}(t,e);if(!i)return e;var r=Go(i,t);return Eo(r,r.next),Eo(i,i.next)}function Ro(t,e){return Uo(t.prev,t,e.prev)<0&&Uo(e.next,t,t.next)<0}function Bo(t,e,i,r,s){return(t=1431655765&((t=858993459&((t=252645135&((t=16711935&((t=(t-i)*s|0)|t<<8))|t<<4))|t<<2))|t<<1))|(e=1431655765&((e=858993459&((e=252645135&((e=16711935&((e=(e-r)*s|0)|e<<8))|e<<4))|e<<2))|e<<1))<<1}function Fo(t){var e=t,i=t;do{(e.x=(t-a)*(n-o)&&(t-a)*(r-o)>=(i-a)*(e-o)&&(i-a)*(n-o)>=(s-a)*(r-o)}function Vo(t,e){return t.next.i!==e.i&&t.prev.i!==e.i&&!function(t,e){var i=t;do{if(i.i!==t.i&&i.next.i!==t.i&&i.i!==e.i&&i.next.i!==e.i&&$o(i,i.next,t,e))return!0;i=i.next}while(i!==t);return!1}(t,e)&&(jo(t,e)&&jo(e,t)&&function(t,e){var i=t,r=!1,s=(t.x+e.x)/2,n=(t.y+e.y)/2;do{i.y>n!=i.next.y>n&&i.next.y!==i.y&&s<(i.next.x-i.x)*(n-i.y)/(i.next.y-i.y)+i.x&&(r=!r),i=i.next}while(i!==t);return r}(t,e)&&(Uo(t.prev,t,e.prev)||Uo(t,e.prev,e))||No(t,e)&&Uo(t.prev,t,t.next)>0&&Uo(e.prev,e,e.next)>0)}function Uo(t,e,i){return(e.y-t.y)*(i.x-e.x)-(e.x-t.x)*(i.y-e.y)}function No(t,e){return t.x===e.x&&t.y===e.y}function $o(t,e,i,r){var s=Zo(Uo(t,e,i)),n=Zo(Uo(t,e,r)),a=Zo(Uo(i,r,t)),o=Zo(Uo(i,r,e));return s!==n&&a!==o||!(0!==s||!qo(t,i,e))||!(0!==n||!qo(t,r,e))||!(0!==a||!qo(i,t,r))||!(0!==o||!qo(i,e,r))}function qo(t,e,i){return e.x<=Math.max(t.x,i.x)&&e.x>=Math.min(t.x,i.x)&&e.y<=Math.max(t.y,i.y)&&e.y>=Math.min(t.y,i.y)}function Zo(t){return t>0?1:t<0?-1:0}function jo(t,e){return Uo(t.prev,t,t.next)<0?Uo(t,e,t.next)>=0&&Uo(t,t.prev,e)>=0:Uo(t,e,t.prev)<0||Uo(t,t.next,e)<0}function Go(t,e){var i=new Xo(t.i,t.x,t.y),r=new Xo(e.i,e.x,e.y),s=t.next,n=e.prev;return t.next=e,e.prev=t,i.next=s,s.prev=i,r.next=i,i.prev=r,n.next=r,r.prev=n,r}function Ho(t,e,i,r){var s=new Xo(t,e,i);return r?(s.next=r.next,s.prev=r,r.next.prev=s,r.next=s):(s.prev=s,s.next=s),s}function Wo(t){t.next.prev=t.prev,t.prev.next=t.next,t.prevZ&&(t.prevZ.nextZ=t.nextZ),t.nextZ&&(t.nextZ.prevZ=t.prevZ)}function Xo(t,e,i){this.i=t,this.x=e,this.y=i,this.prev=null,this.next=null,this.z=0,this.prevZ=null,this.nextZ=null,this.steiner=!1}function Ko(t,e,i,r){for(var s=0,n=e,a=i-r;n0&&i.holes.push(r+=t[s-1].length)}return i};var Yo=i(So.exports);function Jo(t,e,i,r,s){Qo(t,e,i||0,r||t.length-1,s||el)}function Qo(t,e,i,r,s){for(;r>i;){if(r-i>600){var n=r-i+1,a=e-i+1,o=Math.log(n),l=.5*Math.exp(2*o/3),c=.5*Math.sqrt(o*l*(n-l)/n)*(a-n/2<0?-1:1);Qo(t,e,Math.max(i,Math.floor(e-a*l/n+c)),Math.min(r,Math.floor(e+(n-a)*l/n+c)),s)}var h=t[e],u=i,p=r;for(tl(t,i,e),s(t[r],h)>0&&tl(t,i,r);u0;)p--}0===s(t[i],h)?tl(t,i,p):tl(t,++p,r),p<=e&&(i=p+1),e<=p&&(r=p-1)}}function tl(t,e,i){var r=t[e];t[e]=t[i],t[i]=r}function el(t,e){return te?1:0}function il(t,e){const i=t.length;if(i<=1)return[t];const r=[];let s,n;for(let e=0;e1)for(let t=0;tt.id)),this.index=t.index,this.hasPattern=!1,this.patternFeatures=[],this.layoutVertexArray=new Sn,this.indexArray=new Bn,this.indexArray2=new Fn,this.programConfigurations=new ga(t.layers,t.zoom),this.segments=new Nn,this.segments2=new Nn,this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id))}populate(t,e,i){this.hasPattern=sl("fill",this.layers,e);const r=this.layers[0].layout.get("fill-sort-key"),s=!r.isConstant(),n=[];for(const{feature:a,id:o,index:l,sourceLayerIndex:c}of t){const t=this.layers[0]._featureFilter.needGeometry,h=Sa(a,t);if(!this.layers[0]._featureFilter.filter(new _s(this.zoom),h,i))continue;const u=s?r.evaluate(h,{},i,e.availableImages):void 0,p={id:o,properties:a.properties,type:a.type,sourceLayerIndex:c,index:l,geometry:t?h.geometry:Ta(a),patterns:{},sortKey:u};n.push(p)}s&&n.sort(((t,e)=>t.sortKey-e.sortKey));for(const r of n){const{geometry:s,index:n,sourceLayerIndex:a}=r;if(this.hasPattern){const t=nl("fill",this.layers,r,this.zoom,e);this.patternFeatures.push(t)}else this.addFeature(r,s,n,i,{});e.featureIndex.insert(t[n].feature,s,n,a,this.index)}}update(t,e,i){this.stateDependentLayers.length&&this.programConfigurations.updatePaintArrays(t,e,this.stateDependentLayers,i)}addFeatures(t,e,i){for(const t of this.patternFeatures)this.addFeature(t,t.geometry,t.index,e,i)}isEmpty(){return 0===this.layoutVertexArray.length}uploadPending(){return!this.uploaded||this.programConfigurations.needsUpload}upload(t){this.uploaded||(this.layoutVertexBuffer=t.createVertexBuffer(this.layoutVertexArray,To),this.indexBuffer=t.createIndexBuffer(this.indexArray),this.indexBuffer2=t.createIndexBuffer(this.indexArray2)),this.programConfigurations.upload(t),this.uploaded=!0}destroy(){this.layoutVertexBuffer&&(this.layoutVertexBuffer.destroy(),this.indexBuffer.destroy(),this.indexBuffer2.destroy(),this.programConfigurations.destroy(),this.segments.destroy(),this.segments2.destroy())}addFeature(t,e,i,r,s){for(const t of il(e,500)){let e=0;for(const i of t)e+=i.length;const i=this.segments.prepareSegment(e,this.layoutVertexArray,this.indexArray),r=i.vertexLength,s=[],n=[];for(const e of t){if(0===e.length)continue;e!==t[0]&&n.push(s.length/2);const i=this.segments2.prepareSegment(e.length,this.layoutVertexArray,this.indexArray2),r=i.vertexLength;this.layoutVertexArray.emplaceBack(e[0].x,e[0].y),this.indexArray2.emplaceBack(r+e.length-1,r),s.push(e[0].x),s.push(e[0].y);for(let t=1;t>3}if(s--,1===r||2===r)n+=t.readSVarint(),a+=t.readSVarint(),1===r&&(e&&o.push(e),e=[]),e.push(new fl(n,a));else{if(7!==r)throw new Error("unknown command "+r);e&&e.push(e[0].clone())}}return e&&o.push(e),o},gl.prototype.bbox=function(){var t=this._pbf;t.pos=this._geometry;for(var e=t.readVarint()+t.pos,i=1,r=0,s=0,n=0,a=1/0,o=-1/0,l=1/0,c=-1/0;t.pos>3}if(r--,1===i||2===i)(s+=t.readSVarint())o&&(o=s),(n+=t.readSVarint())c&&(c=n);else if(7!==i)throw new Error("unknown command "+i)}return[a,l,o,c]},gl.prototype.toGeoJSON=function(t,e,i){var r,s,n=this.extent*Math.pow(2,i),a=this.extent*t,o=this.extent*e,l=this.loadGeometry(),c=gl.types[this.type];function h(t){for(var e=0;e>3;e=1===r?t.readString():2===r?t.readFloat():3===r?t.readDouble():4===r?t.readVarint64():5===r?t.readVarint():6===r?t.readSVarint():7===r?t.readBoolean():null}return e}(i))}wl.prototype.feature=function(t){if(t<0||t>=this._features.length)throw new Error("feature index out of bounds");this._pbf.pos=this._features[t];var e=this._pbf.readVarint()+this._pbf.pos;return new vl(this._pbf,e,this.extent,this._keys,this._values)};var Sl=bl;function Il(t,e,i){if(3===t){var r=new Sl(i,i.readVarint()+i.pos);r.length&&(e[r.name]=r)}}ml.VectorTile=function(t,e){this.layers=t.readFields(Il,{},e)},ml.VectorTileFeature=_l,ml.VectorTileLayer=bl;const Al=ml.VectorTileFeature.types,El=Math.pow(2,13);function Cl(t,e,i,r,s,n,a,o){t.emplaceBack(e,i,2*Math.floor(r*El)+a,s*El*2,n*El*2,Math.round(o))}class zl{constructor(t){this.zoom=t.zoom,this.overscaling=t.overscaling,this.layers=t.layers,this.layerIds=this.layers.map((t=>t.id)),this.index=t.index,this.hasPattern=!1,this.layoutVertexArray=new In,this.centroidVertexArray=new vn,this.indexArray=new Bn,this.programConfigurations=new ga(t.layers,t.zoom),this.segments=new Nn,this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id))}populate(t,e,i){this.features=[],this.hasPattern=sl("fill-extrusion",this.layers,e);for(const{feature:r,id:s,index:n,sourceLayerIndex:a}of t){const t=this.layers[0]._featureFilter.needGeometry,o=Sa(r,t);if(!this.layers[0]._featureFilter.filter(new _s(this.zoom),o,i))continue;const l={id:s,sourceLayerIndex:a,index:n,geometry:t?o.geometry:Ta(r),properties:r.properties,type:r.type,patterns:{}};this.hasPattern?this.features.push(nl("fill-extrusion",this.layers,l,this.zoom,e)):this.addFeature(l,l.geometry,n,i,{}),e.featureIndex.insert(r,l.geometry,n,a,this.index,!0)}}addFeatures(t,e,i){for(const t of this.features){const{geometry:r}=t;this.addFeature(t,r,t.index,e,i)}}update(t,e,i){this.stateDependentLayers.length&&this.programConfigurations.updatePaintArrays(t,e,this.stateDependentLayers,i)}isEmpty(){return 0===this.layoutVertexArray.length&&0===this.centroidVertexArray.length}uploadPending(){return!this.uploaded||this.programConfigurations.needsUpload}upload(t){this.uploaded||(this.layoutVertexBuffer=t.createVertexBuffer(this.layoutVertexArray,dl),this.centroidVertexBuffer=t.createVertexBuffer(this.centroidVertexArray,pl.members,!0),this.indexBuffer=t.createIndexBuffer(this.indexArray)),this.programConfigurations.upload(t),this.uploaded=!0}destroy(){this.layoutVertexBuffer&&(this.layoutVertexBuffer.destroy(),this.indexBuffer.destroy(),this.programConfigurations.destroy(),this.segments.destroy(),this.centroidVertexBuffer.destroy())}addFeature(t,e,i,r,s){const n={x:0,y:0,vertexCount:0};for(const i of il(e,500)){let e=0;for(const t of i)e+=t.length;let r=this.segments.prepareSegment(4,this.layoutVertexArray,this.indexArray);for(const t of i){if(0===t.length)continue;if(Pl(t))continue;let e=0;for(let i=0;i=1){const a=t[i-1];if(!Ml(s,a)){r.vertexLength+4>Nn.MAX_VERTEX_ARRAY_LENGTH&&(r=this.segments.prepareSegment(4,this.layoutVertexArray,this.indexArray));const t=s.sub(a)._perp()._unit(),i=a.dist(s);e+i>32768&&(e=0),Cl(this.layoutVertexArray,s.x,s.y,t.x,t.y,0,0,e),Cl(this.layoutVertexArray,s.x,s.y,t.x,t.y,0,1,e),n.x+=2*s.x,n.y+=2*s.y,n.vertexCount+=2,e+=i,Cl(this.layoutVertexArray,a.x,a.y,t.x,t.y,0,0,e),Cl(this.layoutVertexArray,a.x,a.y,t.x,t.y,0,1,e),n.x+=2*a.x,n.y+=2*a.y,n.vertexCount+=2;const o=r.vertexLength;this.indexArray.emplaceBack(o,o+2,o+1),this.indexArray.emplaceBack(o+1,o+2,o+3),r.vertexLength+=4,r.primitiveLength+=2}}}}if(r.vertexLength+e>Nn.MAX_VERTEX_ARRAY_LENGTH&&(r=this.segments.prepareSegment(e,this.layoutVertexArray,this.indexArray)),"Polygon"!==Al[t.type])continue;const s=[],a=[],o=r.vertexLength;for(const t of i)if(0!==t.length){t!==i[0]&&a.push(s.length/2);for(let e=0;eva)||t.y===e.y&&(t.y<0||t.y>va)}function Pl(t){return t.every((t=>t.x<0))||t.every((t=>t.x>va))||t.every((t=>t.y<0))||t.every((t=>t.y>va))}let kl;Hr("FillExtrusionBucket",zl,{omit:["layers","features"]});var Dl={get paint(){return kl=kl||new Ms({"fill-extrusion-opacity":new Is(lt["paint_fill-extrusion"]["fill-extrusion-opacity"]),"fill-extrusion-color":new As(lt["paint_fill-extrusion"]["fill-extrusion-color"]),"fill-extrusion-translate":new Is(lt["paint_fill-extrusion"]["fill-extrusion-translate"]),"fill-extrusion-translate-anchor":new Is(lt["paint_fill-extrusion"]["fill-extrusion-translate-anchor"]),"fill-extrusion-pattern":new Es(lt["paint_fill-extrusion"]["fill-extrusion-pattern"]),"fill-extrusion-height":new As(lt["paint_fill-extrusion"]["fill-extrusion-height"]),"fill-extrusion-base":new As(lt["paint_fill-extrusion"]["fill-extrusion-base"]),"fill-extrusion-vertical-gradient":new Is(lt["paint_fill-extrusion"]["fill-extrusion-vertical-gradient"])})}};class Ll extends ks{constructor(t){super(t,Dl)}createBucket(t){return new zl(t)}queryRadius(){return Va(this.paint.get("fill-extrusion-translate"))}is3D(){return!0}queryIntersectsFeature(t,e,i,r,s,a,o,l){const c=Ua(t,this.paint.get("fill-extrusion-translate"),this.paint.get("fill-extrusion-translate-anchor"),a.angle,o),h=this.paint.get("fill-extrusion-height").evaluate(e,i),u=this.paint.get("fill-extrusion-base").evaluate(e,i),p=function(t,e,i,r){const s=[];for(const i of t){const t=[i.x,i.y,0,1];so(t,t,e),s.push(new n(t[0]/t[3],t[1]/t[3]))}return s}(c,l),d=function(t,e,i,r){const s=[],a=[],o=r[8]*e,l=r[9]*e,c=r[10]*e,h=r[11]*e,u=r[8]*i,p=r[9]*i,d=r[10]*i,m=r[11]*i;for(const e of t){const t=[],i=[];for(const s of e){const e=s.x,a=s.y,f=r[0]*e+r[4]*a+r[12],_=r[1]*e+r[5]*a+r[13],g=r[2]*e+r[6]*a+r[14],y=r[3]*e+r[7]*a+r[15],x=g+c,v=y+h,b=f+u,w=_+p,T=g+d,S=y+m,I=new n((f+o)/v,(_+l)/v);I.z=x/v,t.push(I);const A=new n(b/S,w/S);A.z=T/S,i.push(A)}s.push(t),a.push(i)}return[s,a]}(r,u,h,l);return function(t,e,i){let r=1/0;za(i,e)&&(r=Bl(i,e[0]));for(let s=0;st.id)),this.index=t.index,this.hasPattern=!1,this.patternFeatures=[],this.lineClipsArray=[],this.gradients={},this.layers.forEach((t=>{this.gradients[t.id]={}})),this.layoutVertexArray=new An,this.layoutVertexArray2=new En,this.indexArray=new Bn,this.programConfigurations=new ga(t.layers,t.zoom),this.segments=new Nn,this.maxLineLength=0,this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id))}populate(t,e,i){this.hasPattern=sl("line",this.layers,e);const r=this.layers[0].layout.get("line-sort-key"),s=!r.isConstant(),n=[];for(const{feature:e,id:a,index:o,sourceLayerIndex:l}of t){const t=this.layers[0]._featureFilter.needGeometry,c=Sa(e,t);if(!this.layers[0]._featureFilter.filter(new _s(this.zoom),c,i))continue;const h=s?r.evaluate(c,{},i):void 0,u={id:a,properties:e.properties,type:e.type,sourceLayerIndex:l,index:o,geometry:t?c.geometry:Ta(e),patterns:{},sortKey:h};n.push(u)}s&&n.sort(((t,e)=>t.sortKey-e.sortKey));for(const r of n){const{geometry:s,index:n,sourceLayerIndex:a}=r;if(this.hasPattern){const t=nl("line",this.layers,r,this.zoom,e);this.patternFeatures.push(t)}else this.addFeature(r,s,n,i,{});e.featureIndex.insert(t[n].feature,s,n,a,this.index)}}update(t,e,i){this.stateDependentLayers.length&&this.programConfigurations.updatePaintArrays(t,e,this.stateDependentLayers,i)}addFeatures(t,e,i){for(const t of this.patternFeatures)this.addFeature(t,t.geometry,t.index,e,i)}isEmpty(){return 0===this.layoutVertexArray.length}uploadPending(){return!this.uploaded||this.programConfigurations.needsUpload}upload(t){this.uploaded||(0!==this.layoutVertexArray2.length&&(this.layoutVertexBuffer2=t.createVertexBuffer(this.layoutVertexArray2,Ul)),this.layoutVertexBuffer=t.createVertexBuffer(this.layoutVertexArray,Ol),this.indexBuffer=t.createIndexBuffer(this.indexArray)),this.programConfigurations.upload(t),this.uploaded=!0}destroy(){this.layoutVertexBuffer&&(this.layoutVertexBuffer.destroy(),this.indexBuffer.destroy(),this.programConfigurations.destroy(),this.segments.destroy())}lineFeatureClips(t){if(t.properties&&Object.prototype.hasOwnProperty.call(t.properties,"mapbox_clip_start")&&Object.prototype.hasOwnProperty.call(t.properties,"mapbox_clip_end"))return{start:+t.properties.mapbox_clip_start,end:+t.properties.mapbox_clip_end}}addFeature(t,e,i,r,s){const n=this.layers[0].layout,a=n.get("line-join").evaluate(t,{}),o=n.get("line-cap"),l=n.get("line-miter-limit"),c=n.get("line-round-limit");this.lineClips=this.lineFeatureClips(t);for(const i of e)this.addLine(i,t,a,o,l,c);this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length,t,i,s,r)}addLine(t,e,i,r,s,n){if(this.distance=0,this.scaledDistance=0,this.totalDistance=0,this.lineClips){this.lineClipsArray.push(this.lineClips);for(let e=0;e=2&&t[o-1].equals(t[o-2]);)o--;let l=0;for(;l0;if(b&&e>l){const t=u.dist(p);if(t>2*c){const e=u.sub(u.sub(p)._mult(c/t)._round());this.updateDistance(p,e),this.addCurrentVertex(e,m,0,0,h),p=e}}const T=p&&d;let S=T?i:a?"butt":r;if(T&&"round"===S&&(xs&&(S="bevel"),"bevel"===S&&(x>2&&(S="flipbevel"),x100)_=f.mult(-1);else{const t=x*m.add(f).mag()/m.sub(f).mag();_._perp()._mult(t*(w?-1:1))}this.addCurrentVertex(u,_,0,0,h),this.addCurrentVertex(u,_.mult(-1),0,0,h)}else if("bevel"===S||"fakeround"===S){const t=-Math.sqrt(x*x-1),e=w?t:0,i=w?0:t;if(p&&this.addCurrentVertex(u,m,e,i,h),"fakeround"===S){const t=Math.round(180*v/Math.PI/20);for(let e=1;e2*c){const e=u.add(d.sub(u)._mult(c/t)._round());this.updateDistance(u,e),this.addCurrentVertex(e,f,0,0,h),u=e}}}}addCurrentVertex(t,e,i,r,s,n=!1){const a=e.y*r-e.x,o=-e.y-e.x*r;this.addHalfVertex(t,e.x+e.y*i,e.y-e.x*i,n,!1,i,s),this.addHalfVertex(t,a,o,n,!0,-r,s),this.distance>ql/2&&0===this.totalDistance&&(this.distance=0,this.updateScaledDistance(),this.addCurrentVertex(t,e,i,r,s,n))}addHalfVertex({x:t,y:e},i,r,s,n,a,o){const l=.5*(this.lineClips?this.scaledDistance*(ql-1):this.scaledDistance);this.layoutVertexArray.emplaceBack((t<<1)+(s?1:0),(e<<1)+(n?1:0),Math.round(63*i)+128,Math.round(63*r)+128,1+(0===a?0:a<0?-1:1)|(63&l)<<2,l>>6),this.lineClips&&this.layoutVertexArray2.emplaceBack((this.scaledDistance-this.lineClips.start)/(this.lineClips.end-this.lineClips.start),this.lineClipsArray.length);const c=o.vertexLength++;this.e1>=0&&this.e2>=0&&(this.indexArray.emplaceBack(this.e1,this.e2,c),o.primitiveLength++),n?this.e2=c:this.e1=c}updateScaledDistance(){this.scaledDistance=this.lineClips?this.lineClips.start+(this.lineClips.end-this.lineClips.start)*this.distance/this.totalDistance:this.distance}updateDistance(t,e){this.distance+=t.dist(e),this.updateScaledDistance()}}let jl,Gl;Hr("LineBucket",Zl,{omit:["layers","patternFeatures"]});var Hl={get paint(){return Gl=Gl||new Ms({"line-opacity":new As(lt.paint_line["line-opacity"]),"line-color":new As(lt.paint_line["line-color"]),"line-translate":new Is(lt.paint_line["line-translate"]),"line-translate-anchor":new Is(lt.paint_line["line-translate-anchor"]),"line-width":new As(lt.paint_line["line-width"]),"line-gap-width":new As(lt.paint_line["line-gap-width"]),"line-offset":new As(lt.paint_line["line-offset"]),"line-blur":new As(lt.paint_line["line-blur"]),"line-dasharray":new Cs(lt.paint_line["line-dasharray"]),"line-pattern":new Es(lt.paint_line["line-pattern"]),"line-gradient":new zs(lt.paint_line["line-gradient"])})},get layout(){return jl=jl||new Ms({"line-cap":new Is(lt.layout_line["line-cap"]),"line-join":new As(lt.layout_line["line-join"]),"line-miter-limit":new Is(lt.layout_line["line-miter-limit"]),"line-round-limit":new Is(lt.layout_line["line-round-limit"]),"line-sort-key":new As(lt.layout_line["line-sort-key"])})}};class Wl extends As{possiblyEvaluate(t,e){return e=new _s(Math.floor(e.zoom),{now:e.now,fadeDuration:e.fadeDuration,zoomHistory:e.zoomHistory,transition:e.transition}),super.possiblyEvaluate(t,e)}evaluate(t,e,i,r){return e=g({},e,{zoom:Math.floor(e.zoom)}),super.evaluate(t,e,i,r)}}let Xl;class Kl extends ks{constructor(t){super(t,Hl),this.gradientVersion=0,Xl||(Xl=new Wl(Hl.paint.properties["line-width"].specification),Xl.useIntegerZoom=!0)}_handleSpecialPaintPropertyUpdate(t){if("line-gradient"===t){const t=this.gradientExpression();this.stepInterpolant=!!function(t){return void 0!==t._styleExpression}(t)&&t._styleExpression.expression instanceof ii,this.gradientVersion=(this.gradientVersion+1)%Number.MAX_SAFE_INTEGER}}gradientExpression(){return this._transitionablePaint._values["line-gradient"].value.expression}recalculate(t,e){super.recalculate(t,e),this.paint._values["line-floorwidth"]=Xl.possiblyEvaluate(this._transitioningPaint._values["line-width"].value,t)}createBucket(t){return new Zl(t)}queryRadius(t){const e=t,i=Yl(Oa("line-width",this,e),Oa("line-gap-width",this,e)),r=Oa("line-offset",this,e);return i/2+Math.abs(r)+Va(this.paint.get("line-translate"))}queryIntersectsFeature(t,e,i,r,s,a,o){const l=Ua(t,this.paint.get("line-translate"),this.paint.get("line-translate-anchor"),a.angle,o),c=o/2*Yl(this.paint.get("line-width").evaluate(e,i),this.paint.get("line-gap-width").evaluate(e,i)),h=this.paint.get("line-offset").evaluate(e,i);return h&&(r=function(t,e){const i=[];for(let r=0;r=3)for(let e=0;e0?e+2*t:t}const Jl=Bs([{name:"a_pos_offset",components:4,type:"Int16"},{name:"a_data",components:4,type:"Uint16"},{name:"a_pixeloffset",components:4,type:"Int16"}],4),Ql=Bs([{name:"a_projected_pos",components:3,type:"Float32"}],4);Bs([{name:"a_fade_opacity",components:1,type:"Uint32"}],4);const tc=Bs([{name:"a_placed",components:2,type:"Uint8"},{name:"a_shift",components:2,type:"Float32"}]);Bs([{type:"Int16",name:"anchorPointX"},{type:"Int16",name:"anchorPointY"},{type:"Int16",name:"x1"},{type:"Int16",name:"y1"},{type:"Int16",name:"x2"},{type:"Int16",name:"y2"},{type:"Uint32",name:"featureIndex"},{type:"Uint16",name:"sourceLayerIndex"},{type:"Uint16",name:"bucketIndex"}]);const ec=Bs([{name:"a_pos",components:2,type:"Int16"},{name:"a_anchor_pos",components:2,type:"Int16"},{name:"a_extrude",components:2,type:"Int16"}],4),ic=Bs([{name:"a_pos",components:2,type:"Float32"},{name:"a_radius",components:1,type:"Float32"},{name:"a_flags",components:2,type:"Int16"}],4);function rc(t,e,i){return t.sections.forEach((t=>{t.text=function(t,e,i){const r=e.layout.get("text-transform").evaluate(i,{});return"uppercase"===r?t=t.toLocaleUpperCase():"lowercase"===r&&(t=t.toLocaleLowerCase()),t}(t.text,e,i)})),t}Bs([{name:"triangle",components:3,type:"Uint16"}]),Bs([{type:"Int16",name:"anchorX"},{type:"Int16",name:"anchorY"},{type:"Uint16",name:"glyphStartIndex"},{type:"Uint16",name:"numGlyphs"},{type:"Uint32",name:"vertexStartIndex"},{type:"Uint32",name:"lineStartIndex"},{type:"Uint32",name:"lineLength"},{type:"Uint16",name:"segment"},{type:"Uint16",name:"lowerSize"},{type:"Uint16",name:"upperSize"},{type:"Float32",name:"lineOffsetX"},{type:"Float32",name:"lineOffsetY"},{type:"Uint8",name:"writingMode"},{type:"Uint8",name:"placedOrientation"},{type:"Uint8",name:"hidden"},{type:"Uint32",name:"crossTileID"},{type:"Int16",name:"associatedIconIndex"}]),Bs([{type:"Int16",name:"anchorX"},{type:"Int16",name:"anchorY"},{type:"Int16",name:"rightJustifiedTextSymbolIndex"},{type:"Int16",name:"centerJustifiedTextSymbolIndex"},{type:"Int16",name:"leftJustifiedTextSymbolIndex"},{type:"Int16",name:"verticalPlacedTextSymbolIndex"},{type:"Int16",name:"placedIconSymbolIndex"},{type:"Int16",name:"verticalPlacedIconSymbolIndex"},{type:"Uint16",name:"key"},{type:"Uint16",name:"textBoxStartIndex"},{type:"Uint16",name:"textBoxEndIndex"},{type:"Uint16",name:"verticalTextBoxStartIndex"},{type:"Uint16",name:"verticalTextBoxEndIndex"},{type:"Uint16",name:"iconBoxStartIndex"},{type:"Uint16",name:"iconBoxEndIndex"},{type:"Uint16",name:"verticalIconBoxStartIndex"},{type:"Uint16",name:"verticalIconBoxEndIndex"},{type:"Uint16",name:"featureIndex"},{type:"Uint16",name:"numHorizontalGlyphVertices"},{type:"Uint16",name:"numVerticalGlyphVertices"},{type:"Uint16",name:"numIconVertices"},{type:"Uint16",name:"numVerticalIconVertices"},{type:"Uint16",name:"useRuntimeCollisionCircles"},{type:"Uint32",name:"crossTileID"},{type:"Float32",name:"textBoxScale"},{type:"Float32",name:"collisionCircleDiameter"},{type:"Uint16",name:"textAnchorOffsetStartIndex"},{type:"Uint16",name:"textAnchorOffsetEndIndex"}]),Bs([{type:"Float32",name:"offsetX"}]),Bs([{type:"Int16",name:"x"},{type:"Int16",name:"y"},{type:"Int16",name:"tileUnitDistanceFromAnchor"}]),Bs([{type:"Uint16",name:"textAnchor"},{type:"Float32",components:2,name:"textOffset"}]);const sc={"!":"︕","#":"#",$:"$","%":"%","&":"&","(":"︵",")":"︶","*":"*","+":"+",",":"︐","-":"︲",".":"・","/":"/",":":"︓",";":"︔","<":"︿","=":"=",">":"﹀","?":"︖","@":"@","[":"﹇","\\":"\","]":"﹈","^":"^",_:"︳","`":"`","{":"︷","|":"―","}":"︸","~":"~","¢":"¢","£":"£","¥":"¥","¦":"¦","¬":"¬","¯":" ̄","–":"︲","—":"︱","‘":"﹃","’":"﹄","“":"﹁","”":"﹂","…":"︙","‧":"・","₩":"₩","、":"︑","。":"︒","〈":"︿","〉":"﹀","《":"︽","》":"︾","「":"﹁","」":"﹂","『":"﹃","』":"﹄","【":"︻","】":"︼","〔":"︹","〕":"︺","〖":"︗","〗":"︘","!":"︕","(":"︵",")":"︶",",":"︐","-":"︲",".":"・",":":"︓",";":"︔","<":"︿",">":"﹀","?":"︖","[":"﹇","]":"﹈","_":"︳","{":"︷","|":"―","}":"︸","⦅":"︵","⦆":"︶","。":"︒","「":"﹁","」":"﹂"};var nc=lc,ac=function(t,e,i,r,s){var n,a,o=8*s-r-1,l=(1<>1,h=-7,u=i?s-1:0,p=i?-1:1,d=t[e+u];for(u+=p,n=d&(1<<-h)-1,d>>=-h,h+=o;h>0;n=256*n+t[e+u],u+=p,h-=8);for(a=n&(1<<-h)-1,n>>=-h,h+=r;h>0;a=256*a+t[e+u],u+=p,h-=8);if(0===n)n=1-c;else{if(n===l)return a?NaN:1/0*(d?-1:1);a+=Math.pow(2,r),n-=c}return(d?-1:1)*a*Math.pow(2,n-r)},oc=function(t,e,i,r,s,n){var a,o,l,c=8*n-s-1,h=(1<>1,p=23===s?Math.pow(2,-24)-Math.pow(2,-77):0,d=r?0:n-1,m=r?1:-1,f=e<0||0===e&&1/e<0?1:0;for(e=Math.abs(e),isNaN(e)||e===1/0?(o=isNaN(e)?1:0,a=h):(a=Math.floor(Math.log(e)/Math.LN2),e*(l=Math.pow(2,-a))<1&&(a--,l*=2),(e+=a+u>=1?p/l:p*Math.pow(2,1-u))*l>=2&&(a++,l/=2),a+u>=h?(o=0,a=h):a+u>=1?(o=(e*l-1)*Math.pow(2,s),a+=u):(o=e*Math.pow(2,u-1)*Math.pow(2,s),a=0));s>=8;t[i+d]=255&o,d+=m,o/=256,s-=8);for(a=a<0;t[i+d]=255&a,d+=m,a/=256,c-=8);t[i+d-m]|=128*f};function lc(t){this.buf=ArrayBuffer.isView&&ArrayBuffer.isView(t)?t:new Uint8Array(t||0),this.pos=0,this.type=0,this.length=this.buf.length}lc.Varint=0,lc.Fixed64=1,lc.Bytes=2,lc.Fixed32=5;var cc=4294967296,hc=1/cc,uc="undefined"==typeof TextDecoder?null:new TextDecoder("utf8");function pc(t){return t.type===lc.Bytes?t.readVarint()+t.pos:t.pos+1}function dc(t,e,i){return i?4294967296*e+(t>>>0):4294967296*(e>>>0)+(t>>>0)}function mc(t,e,i){var r=e<=16383?1:e<=2097151?2:e<=268435455?3:Math.floor(Math.log(e)/(7*Math.LN2));i.realloc(r);for(var s=i.pos-1;s>=t;s--)i.buf[s+r]=i.buf[s]}function fc(t,e){for(var i=0;i>>8,t[i+2]=e>>>16,t[i+3]=e>>>24}function Ac(t,e){return(t[e]|t[e+1]<<8|t[e+2]<<16)+(t[e+3]<<24)}lc.prototype={destroy:function(){this.buf=null},readFields:function(t,e,i){for(i=i||this.length;this.pos>3,n=this.pos;this.type=7&r,t(s,e,this),this.pos===n&&this.skip(r)}return e},readMessage:function(t,e){return this.readFields(t,e,this.readVarint()+this.pos)},readFixed32:function(){var t=Sc(this.buf,this.pos);return this.pos+=4,t},readSFixed32:function(){var t=Ac(this.buf,this.pos);return this.pos+=4,t},readFixed64:function(){var t=Sc(this.buf,this.pos)+Sc(this.buf,this.pos+4)*cc;return this.pos+=8,t},readSFixed64:function(){var t=Sc(this.buf,this.pos)+Ac(this.buf,this.pos+4)*cc;return this.pos+=8,t},readFloat:function(){var t=ac(this.buf,this.pos,!0,23,4);return this.pos+=4,t},readDouble:function(){var t=ac(this.buf,this.pos,!0,52,8);return this.pos+=8,t},readVarint:function(t){var e,i,r=this.buf;return e=127&(i=r[this.pos++]),i<128?e:(e|=(127&(i=r[this.pos++]))<<7,i<128?e:(e|=(127&(i=r[this.pos++]))<<14,i<128?e:(e|=(127&(i=r[this.pos++]))<<21,i<128?e:function(t,e,i){var r,s,n=i.buf;if(r=(112&(s=n[i.pos++]))>>4,s<128)return dc(t,r,e);if(r|=(127&(s=n[i.pos++]))<<3,s<128)return dc(t,r,e);if(r|=(127&(s=n[i.pos++]))<<10,s<128)return dc(t,r,e);if(r|=(127&(s=n[i.pos++]))<<17,s<128)return dc(t,r,e);if(r|=(127&(s=n[i.pos++]))<<24,s<128)return dc(t,r,e);if(r|=(1&(s=n[i.pos++]))<<31,s<128)return dc(t,r,e);throw new Error("Expected varint not more than 10 bytes")}(e|=(15&(i=r[this.pos]))<<28,t,this))))},readVarint64:function(){return this.readVarint(!0)},readSVarint:function(){var t=this.readVarint();return t%2==1?(t+1)/-2:t/2},readBoolean:function(){return Boolean(this.readVarint())},readString:function(){var t=this.readVarint()+this.pos,e=this.pos;return this.pos=t,t-e>=12&&uc?function(t,e,i){return uc.decode(t.subarray(e,i))}(this.buf,e,t):function(t,e,i){for(var r="",s=e;s239?4:l>223?3:l>191?2:1;if(s+h>i)break;1===h?l<128&&(c=l):2===h?128==(192&(n=t[s+1]))&&(c=(31&l)<<6|63&n)<=127&&(c=null):3===h?(a=t[s+2],128==(192&(n=t[s+1]))&&128==(192&a)&&((c=(15&l)<<12|(63&n)<<6|63&a)<=2047||c>=55296&&c<=57343)&&(c=null)):4===h&&(a=t[s+2],o=t[s+3],128==(192&(n=t[s+1]))&&128==(192&a)&&128==(192&o)&&((c=(15&l)<<18|(63&n)<<12|(63&a)<<6|63&o)<=65535||c>=1114112)&&(c=null)),null===c?(c=65533,h=1):c>65535&&(c-=65536,r+=String.fromCharCode(c>>>10&1023|55296),c=56320|1023&c),r+=String.fromCharCode(c),s+=h}return r}(this.buf,e,t)},readBytes:function(){var t=this.readVarint()+this.pos,e=this.buf.subarray(this.pos,t);return this.pos=t,e},readPackedVarint:function(t,e){if(this.type!==lc.Bytes)return t.push(this.readVarint(e));var i=pc(this);for(t=t||[];this.pos127;);else if(e===lc.Bytes)this.pos=this.readVarint()+this.pos;else if(e===lc.Fixed32)this.pos+=4;else{if(e!==lc.Fixed64)throw new Error("Unimplemented type: "+e);this.pos+=8}},writeTag:function(t,e){this.writeVarint(t<<3|e)},realloc:function(t){for(var e=this.length||16;e268435455||t<0?function(t,e){var i,r;if(t>=0?(i=t%4294967296|0,r=t/4294967296|0):(r=~(-t/4294967296),4294967295^(i=~(-t%4294967296))?i=i+1|0:(i=0,r=r+1|0)),t>=0x10000000000000000||t<-0x10000000000000000)throw new Error("Given varint doesn't fit into 10 bytes");e.realloc(10),function(t,e,i){i.buf[i.pos++]=127&t|128,t>>>=7,i.buf[i.pos++]=127&t|128,t>>>=7,i.buf[i.pos++]=127&t|128,t>>>=7,i.buf[i.pos++]=127&t|128,i.buf[i.pos]=127&(t>>>=7)}(i,0,e),function(t,e){var i=(7&t)<<4;e.buf[e.pos++]|=i|((t>>>=3)?128:0),t&&(e.buf[e.pos++]=127&t|((t>>>=7)?128:0),t&&(e.buf[e.pos++]=127&t|((t>>>=7)?128:0),t&&(e.buf[e.pos++]=127&t|((t>>>=7)?128:0),t&&(e.buf[e.pos++]=127&t|((t>>>=7)?128:0),t&&(e.buf[e.pos++]=127&t)))))}(r,e)}(t,this):(this.realloc(4),this.buf[this.pos++]=127&t|(t>127?128:0),t<=127||(this.buf[this.pos++]=127&(t>>>=7)|(t>127?128:0),t<=127||(this.buf[this.pos++]=127&(t>>>=7)|(t>127?128:0),t<=127||(this.buf[this.pos++]=t>>>7&127))))},writeSVarint:function(t){this.writeVarint(t<0?2*-t-1:2*t)},writeBoolean:function(t){this.writeVarint(Boolean(t))},writeString:function(t){t=String(t),this.realloc(4*t.length),this.pos++;var e=this.pos;this.pos=function(t,e,i){for(var r,s,n=0;n55295&&r<57344){if(!s){r>56319||n+1===e.length?(t[i++]=239,t[i++]=191,t[i++]=189):s=r;continue}if(r<56320){t[i++]=239,t[i++]=191,t[i++]=189,s=r;continue}r=s-55296<<10|r-56320|65536,s=null}else s&&(t[i++]=239,t[i++]=191,t[i++]=189,s=null);r<128?t[i++]=r:(r<2048?t[i++]=r>>6|192:(r<65536?t[i++]=r>>12|224:(t[i++]=r>>18|240,t[i++]=r>>12&63|128),t[i++]=r>>6&63|128),t[i++]=63&r|128)}return i}(this.buf,t,this.pos);var i=this.pos-e;i>=128&&mc(e,i,this),this.pos=e-1,this.writeVarint(i),this.pos+=i},writeFloat:function(t){this.realloc(4),oc(this.buf,t,this.pos,!0,23,4),this.pos+=4},writeDouble:function(t){this.realloc(8),oc(this.buf,t,this.pos,!0,52,8),this.pos+=8},writeBytes:function(t){var e=t.length;this.writeVarint(e),this.realloc(e);for(var i=0;i=128&&mc(i,r,this),this.pos=i-1,this.writeVarint(r),this.pos+=r},writeMessage:function(t,e,i){this.writeTag(t,lc.Bytes),this.writeRawMessage(e,i)},writePackedVarint:function(t,e){e.length&&this.writeMessage(t,fc,e)},writePackedSVarint:function(t,e){e.length&&this.writeMessage(t,_c,e)},writePackedBoolean:function(t,e){e.length&&this.writeMessage(t,xc,e)},writePackedFloat:function(t,e){e.length&&this.writeMessage(t,gc,e)},writePackedDouble:function(t,e){e.length&&this.writeMessage(t,yc,e)},writePackedFixed32:function(t,e){e.length&&this.writeMessage(t,vc,e)},writePackedSFixed32:function(t,e){e.length&&this.writeMessage(t,bc,e)},writePackedFixed64:function(t,e){e.length&&this.writeMessage(t,wc,e)},writePackedSFixed64:function(t,e){e.length&&this.writeMessage(t,Tc,e)},writeBytesField:function(t,e){this.writeTag(t,lc.Bytes),this.writeBytes(e)},writeFixed32Field:function(t,e){this.writeTag(t,lc.Fixed32),this.writeFixed32(e)},writeSFixed32Field:function(t,e){this.writeTag(t,lc.Fixed32),this.writeSFixed32(e)},writeFixed64Field:function(t,e){this.writeTag(t,lc.Fixed64),this.writeFixed64(e)},writeSFixed64Field:function(t,e){this.writeTag(t,lc.Fixed64),this.writeSFixed64(e)},writeVarintField:function(t,e){this.writeTag(t,lc.Varint),this.writeVarint(e)},writeSVarintField:function(t,e){this.writeTag(t,lc.Varint),this.writeSVarint(e)},writeStringField:function(t,e){this.writeTag(t,lc.Bytes),this.writeString(e)},writeFloatField:function(t,e){this.writeTag(t,lc.Fixed32),this.writeFloat(e)},writeDoubleField:function(t,e){this.writeTag(t,lc.Fixed64),this.writeDouble(e)},writeBooleanField:function(t,e){this.writeVarintField(t,Boolean(e))}};var Ec=i(nc);const Cc=3;function zc(t,e,i){1===t&&i.readMessage(Mc,e)}function Mc(t,e,i){if(3===t){const{id:t,bitmap:r,width:s,height:n,left:a,top:o,advance:l}=i.readMessage(Pc,{});e.push({id:t,bitmap:new fo({width:s+2*Cc,height:n+2*Cc},r),metrics:{width:s,height:n,left:a,top:o,advance:l}})}}function Pc(t,e,i){1===t?e.id=i.readVarint():2===t?e.bitmap=i.readBytes():3===t?e.width=i.readVarint():4===t?e.height=i.readVarint():5===t?e.left=i.readSVarint():6===t?e.top=i.readSVarint():7===t&&(e.advance=i.readVarint())}function kc(t){let e=0,i=0;for(const r of t)e+=r.w*r.h,i=Math.max(i,r.w);t.sort(((t,e)=>e.h-t.h));const r=[{x:0,y:0,w:Math.max(Math.ceil(Math.sqrt(e/.95)),i),h:1/0}];let s=0,n=0;for(const e of t)for(let t=r.length-1;t>=0;t--){const i=r[t];if(!(e.w>i.w||e.h>i.h)){if(e.x=i.x,e.y=i.y,n=Math.max(n,e.y+e.h),s=Math.max(s,e.x+e.w),e.w===i.w&&e.h===i.h){const e=r.pop();tt.id)),this.index=t.index,this.pixelRatio=t.pixelRatio,this.sourceLayerIndex=t.sourceLayerIndex,this.hasPattern=!1,this.hasRTLText=!1,this.sortKeyRanges=[],this.collisionCircleArray=[],this.placementInvProjMatrix=Ha([]),this.placementViewportMatrix=Ha([]);const e=this.layers[0]._unevaluatedLayout._values;this.textSizeData=Oc(this.zoom,e["text-size"]),this.iconSizeData=Oc(this.zoom,e["icon-size"]);const i=this.layers[0].layout,r=i.get("symbol-sort-key"),s=i.get("symbol-z-order");this.canOverlap="never"!==Nc(i,"text-overlap","text-allow-overlap")||"never"!==Nc(i,"icon-overlap","icon-allow-overlap")||i.get("text-ignore-placement")||i.get("icon-ignore-placement"),this.sortFeaturesByKey="viewport-y"!==s&&!r.isConstant(),this.sortFeaturesByY=("viewport-y"===s||"auto"===s&&!this.sortFeaturesByKey)&&this.canOverlap,"point"===i.get("symbol-placement")&&(this.writingModes=i.get("text-writing-mode").map((t=>Lc[t]))),this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id)),this.sourceID=t.sourceID}createArrays(){this.text=new Hc(new ga(this.layers,this.zoom,(t=>/^text/.test(t)))),this.icon=new Hc(new ga(this.layers,this.zoom,(t=>/^icon/.test(t)))),this.glyphOffsetArray=new mn,this.lineVertexArray=new fn,this.symbolInstances=new dn,this.textAnchorOffsets=new gn}calculateGlyphDependencies(t,e,i,r,s){for(let n=0;n0)&&("constant"!==n.value.kind||n.value.value.length>0),c="constant"!==o.value.kind||!!o.value.value||Object.keys(o.parameters).length>0,h=s.get("symbol-sort-key");if(this.features=[],!l&&!c)return;const u=e.iconDependencies,p=e.glyphDependencies,d=e.availableImages,m=new _s(this.zoom);for(const{feature:e,id:a,index:o,sourceLayerIndex:f}of t){const t=r._featureFilter.needGeometry,_=Sa(e,t);if(!r._featureFilter.filter(m,_,i))continue;let g,y;if(t||(_.geometry=Ta(e)),l){const t=r.getValueAndResolveTokens("text-field",_,i,d),e=de.factory(t);Gc(e)&&(this.hasRTLText=!0),(!this.hasRTLText||"unavailable"===ds()||this.hasRTLText&&fs.isParsed())&&(g=rc(e,r,_))}if(c){const t=r.getValueAndResolveTokens("icon-image",_,i,d);y=t instanceof ge?t:ge.fromString(t)}if(!g&&!y)continue;const x=this.sortFeaturesByKey?h.evaluate(_,{},i):void 0;if(this.features.push({id:a,text:g,icon:y,index:o,sourceLayerIndex:f,geometry:_.geometry,properties:e.properties,type:$c[e.type],sortKey:x}),y&&(u[y.name]=!0),g){const t=n.evaluate(_,{},i).join(","),e="viewport"!==s.get("text-rotation-alignment")&&"point"!==s.get("symbol-placement");this.allowVerticalPlacement=this.writingModes&&this.writingModes.indexOf(Lc.vertical)>=0;for(const i of g.sections)if(i.image)u[i.image.name]=!0;else{const r=Qr(g.toString()),s=i.fontStack||t,n=p[s]=p[s]||{};this.calculateGlyphDependencies(i.text,n,e,this.allowVerticalPlacement,r)}}}"line"===s.get("symbol-placement")&&(this.features=function(t){const e={},i={},r=[];let s=0;function n(e){r.push(t[e]),s++}function a(t,e,s){const n=i[t];return delete i[t],i[e]=n,r[n].geometry[0].pop(),r[n].geometry[0]=r[n].geometry[0].concat(s[0]),n}function o(t,i,s){const n=e[i];return delete e[i],e[t]=n,r[n].geometry[0].shift(),r[n].geometry[0]=s[0].concat(r[n].geometry[0]),n}function l(t,e,i){const r=i?e[0][e[0].length-1]:e[0][0];return`${t}:${r.x}:${r.y}`}for(let c=0;ct.geometry))}(this.features)),this.sortFeaturesByKey&&this.features.sort(((t,e)=>t.sortKey-e.sortKey))}update(t,e,i){this.stateDependentLayers.length&&(this.text.programConfigurations.updatePaintArrays(t,e,this.layers,i),this.icon.programConfigurations.updatePaintArrays(t,e,this.layers,i))}isEmpty(){return 0===this.symbolInstances.length&&!this.hasRTLText}uploadPending(){return!this.uploaded||this.text.programConfigurations.needsUpload||this.icon.programConfigurations.needsUpload}upload(t){!this.uploaded&&this.hasDebugData()&&(this.textCollisionBox.upload(t),this.iconCollisionBox.upload(t)),this.text.upload(t,this.sortFeaturesByY,!this.uploaded,this.text.programConfigurations.needsUpload),this.icon.upload(t,this.sortFeaturesByY,!this.uploaded,this.icon.programConfigurations.needsUpload),this.uploaded=!0}destroyDebugData(){this.textCollisionBox.destroy(),this.iconCollisionBox.destroy()}destroy(){this.text.destroy(),this.icon.destroy(),this.hasDebugData()&&this.destroyDebugData()}addToLineVertexArray(t,e){const i=this.lineVertexArray.length;if(void 0!==t.segment){let i=t.dist(e[t.segment+1]),r=t.dist(e[t.segment]);const s={};for(let r=t.segment+1;r=0;i--)s[i]={x:e[i].x,y:e[i].y,tileUnitDistanceFromAnchor:r},i>0&&(r+=e[i-1].dist(e[i]));for(let t=0;t0}hasIconData(){return this.icon.segments.get().length>0}hasDebugData(){return this.textCollisionBox&&this.iconCollisionBox}hasTextCollisionBoxData(){return this.hasDebugData()&&this.textCollisionBox.segments.get().length>0}hasIconCollisionBoxData(){return this.hasDebugData()&&this.iconCollisionBox.segments.get().length>0}addIndicesForPlacedSymbol(t,e){const i=t.placedSymbolArray.get(e),r=i.vertexStartIndex+4*i.numGlyphs;for(let e=i.vertexStartIndex;er[t]-r[e]||s[e]-s[t])),n}addToSortKeyRanges(t,e){const i=this.sortKeyRanges[this.sortKeyRanges.length-1];i&&i.sortKey===e?i.symbolInstanceEnd=t+1:this.sortKeyRanges.push({sortKey:e,symbolInstanceStart:t,symbolInstanceEnd:t+1})}sortFeatures(t){if(this.sortFeaturesByY&&this.sortedAngle!==t&&!(this.text.segments.get().length>1||this.icon.segments.get().length>1)){this.symbolInstanceIndexes=this.getSortedSymbolIndexes(t),this.sortedAngle=t,this.text.indexArray.clear(),this.icon.indexArray.clear(),this.featureSortOrder=[];for(const t of this.symbolInstanceIndexes){const e=this.symbolInstances.get(t);this.featureSortOrder.push(e.featureIndex),[e.rightJustifiedTextSymbolIndex,e.centerJustifiedTextSymbolIndex,e.leftJustifiedTextSymbolIndex].forEach(((t,e,i)=>{t>=0&&i.indexOf(t)===e&&this.addIndicesForPlacedSymbol(this.text,t)})),e.verticalPlacedTextSymbolIndex>=0&&this.addIndicesForPlacedSymbol(this.text,e.verticalPlacedTextSymbolIndex),e.placedIconSymbolIndex>=0&&this.addIndicesForPlacedSymbol(this.icon,e.placedIconSymbolIndex),e.verticalPlacedIconSymbolIndex>=0&&this.addIndicesForPlacedSymbol(this.icon,e.verticalPlacedIconSymbolIndex)}this.text.indexBuffer&&this.text.indexBuffer.updateData(this.text.indexArray),this.icon.indexBuffer&&this.icon.indexBuffer.updateData(this.icon.indexArray)}}}let Kc,Yc;Hr("SymbolBucket",Xc,{omit:["layers","collisionBoxArray","features","compareText"]}),Xc.MAX_GLYPHS=65535,Xc.addDynamicAttributes=jc;var Jc={get paint(){return Yc=Yc||new Ms({"icon-opacity":new As(lt.paint_symbol["icon-opacity"]),"icon-color":new As(lt.paint_symbol["icon-color"]),"icon-halo-color":new As(lt.paint_symbol["icon-halo-color"]),"icon-halo-width":new As(lt.paint_symbol["icon-halo-width"]),"icon-halo-blur":new As(lt.paint_symbol["icon-halo-blur"]),"icon-translate":new Is(lt.paint_symbol["icon-translate"]),"icon-translate-anchor":new Is(lt.paint_symbol["icon-translate-anchor"]),"text-opacity":new As(lt.paint_symbol["text-opacity"]),"text-color":new As(lt.paint_symbol["text-color"],{runtimeType:zt,getOverride:t=>t.textColor,hasOverride:t=>!!t.textColor}),"text-halo-color":new As(lt.paint_symbol["text-halo-color"]),"text-halo-width":new As(lt.paint_symbol["text-halo-width"]),"text-halo-blur":new As(lt.paint_symbol["text-halo-blur"]),"text-translate":new Is(lt.paint_symbol["text-translate"]),"text-translate-anchor":new Is(lt.paint_symbol["text-translate-anchor"])})},get layout(){return Kc=Kc||new Ms({"symbol-placement":new Is(lt.layout_symbol["symbol-placement"]),"symbol-spacing":new Is(lt.layout_symbol["symbol-spacing"]),"symbol-avoid-edges":new Is(lt.layout_symbol["symbol-avoid-edges"]),"symbol-sort-key":new As(lt.layout_symbol["symbol-sort-key"]),"symbol-z-order":new Is(lt.layout_symbol["symbol-z-order"]),"icon-allow-overlap":new Is(lt.layout_symbol["icon-allow-overlap"]),"icon-overlap":new Is(lt.layout_symbol["icon-overlap"]),"icon-ignore-placement":new Is(lt.layout_symbol["icon-ignore-placement"]),"icon-optional":new Is(lt.layout_symbol["icon-optional"]),"icon-rotation-alignment":new Is(lt.layout_symbol["icon-rotation-alignment"]),"icon-size":new As(lt.layout_symbol["icon-size"]),"icon-text-fit":new Is(lt.layout_symbol["icon-text-fit"]),"icon-text-fit-padding":new Is(lt.layout_symbol["icon-text-fit-padding"]),"icon-image":new As(lt.layout_symbol["icon-image"]),"icon-rotate":new As(lt.layout_symbol["icon-rotate"]),"icon-padding":new As(lt.layout_symbol["icon-padding"]),"icon-keep-upright":new Is(lt.layout_symbol["icon-keep-upright"]),"icon-offset":new As(lt.layout_symbol["icon-offset"]),"icon-anchor":new As(lt.layout_symbol["icon-anchor"]),"icon-pitch-alignment":new Is(lt.layout_symbol["icon-pitch-alignment"]),"text-pitch-alignment":new Is(lt.layout_symbol["text-pitch-alignment"]),"text-rotation-alignment":new Is(lt.layout_symbol["text-rotation-alignment"]),"text-field":new As(lt.layout_symbol["text-field"]),"text-font":new As(lt.layout_symbol["text-font"]),"text-size":new As(lt.layout_symbol["text-size"]),"text-max-width":new As(lt.layout_symbol["text-max-width"]),"text-line-height":new Is(lt.layout_symbol["text-line-height"]),"text-letter-spacing":new As(lt.layout_symbol["text-letter-spacing"]),"text-justify":new As(lt.layout_symbol["text-justify"]),"text-radial-offset":new As(lt.layout_symbol["text-radial-offset"]),"text-variable-anchor":new Is(lt.layout_symbol["text-variable-anchor"]),"text-variable-anchor-offset":new As(lt.layout_symbol["text-variable-anchor-offset"]),"text-anchor":new As(lt.layout_symbol["text-anchor"]),"text-max-angle":new Is(lt.layout_symbol["text-max-angle"]),"text-writing-mode":new Is(lt.layout_symbol["text-writing-mode"]),"text-rotate":new As(lt.layout_symbol["text-rotate"]),"text-padding":new Is(lt.layout_symbol["text-padding"]),"text-keep-upright":new Is(lt.layout_symbol["text-keep-upright"]),"text-transform":new As(lt.layout_symbol["text-transform"]),"text-offset":new As(lt.layout_symbol["text-offset"]),"text-allow-overlap":new Is(lt.layout_symbol["text-allow-overlap"]),"text-overlap":new Is(lt.layout_symbol["text-overlap"]),"text-ignore-placement":new Is(lt.layout_symbol["text-ignore-placement"]),"text-optional":new Is(lt.layout_symbol["text-optional"])})}};class Qc{constructor(t){if(void 0===t.property.overrides)throw new Error("overrides must be provided to instantiate FormatSectionOverride class");this.type=t.property.overrides?t.property.overrides.runtimeType:It,this.defaultValue=t}evaluate(t){if(t.formattedSection){const e=this.defaultValue.property.overrides;if(e&&e.hasOverride(t.formattedSection))return e.getOverride(t.formattedSection)}return t.feature&&t.featureState?this.defaultValue.evaluate(t.feature,t.featureState):this.defaultValue.property.specification.default}eachChild(t){this.defaultValue.isConstant()||t(this.defaultValue.value._styleExpression.expression)}outputDefined(){return!1}serialize(){return null}}Hr("FormatSectionOverride",Qc,{omit:["defaultValue"]});class th extends ks{constructor(t){super(t,Jc)}recalculate(t,e){if(super.recalculate(t,e),"auto"===this.layout.get("icon-rotation-alignment")&&(this.layout._values["icon-rotation-alignment"]="point"!==this.layout.get("symbol-placement")?"map":"viewport"),"auto"===this.layout.get("text-rotation-alignment")&&(this.layout._values["text-rotation-alignment"]="point"!==this.layout.get("symbol-placement")?"map":"viewport"),"auto"===this.layout.get("text-pitch-alignment")&&(this.layout._values["text-pitch-alignment"]="map"===this.layout.get("text-rotation-alignment")?"map":"viewport"),"auto"===this.layout.get("icon-pitch-alignment")&&(this.layout._values["icon-pitch-alignment"]=this.layout.get("icon-rotation-alignment")),"point"===this.layout.get("symbol-placement")){const t=this.layout.get("text-writing-mode");if(t){const e=[];for(const i of t)e.indexOf(i)<0&&e.push(i);this.layout._values["text-writing-mode"]=e}else this.layout._values["text-writing-mode"]=["horizontal"]}this._setPaintOverrides()}getValueAndResolveTokens(t,e,i,r){const s=this.layout.get(t).evaluate(e,{},i,r),n=this._unevaluatedLayout._values[t];return n.isDataDriven()||Xi(n.value)||!s?s:function(t,e){return e.replace(/{([^{}]+)}/g,((e,i)=>t&&i in t?String(t[i]):""))}(e.properties,s)}createBucket(t){return new Xc(t)}queryRadius(){return 0}queryIntersectsFeature(){throw new Error("Should take a different path in FeatureIndex")}_setPaintOverrides(){for(const t of Jc.paint.overridableProperties){if(!th.hasPaintOverride(this.layout,t))continue;const e=this.paint.get(t),i=new Qc(e),r=new Wi(i,e.property.specification);let s=null;s="constant"===e.value.kind||"source"===e.value.kind?new Yi("source",r):new Ji("composite",r,e.value.zoomStops),this.paint._values[t]=new Ts(e.property,s,e.parameters)}}_handleOverridablePaintPropertyUpdate(t,e,i){return!(!this.layout||e.isDataDriven()||i.isDataDriven())&&th.hasPaintOverride(this.layout,t)}static hasPaintOverride(t,e){const i=t.get("text-field"),r=Jc.paint.properties[e];let s=!1;const n=t=>{for(const e of t)if(r.overrides&&r.overrides.hasOverride(e))return void(s=!0)};if("constant"===i.value.kind&&i.value.value instanceof de)n(i.value.value.sections);else if("source"===i.value.kind){const t=e=>{s||(e instanceof we&&ve(e.value)===Dt?n(e.value.sections):e instanceof Ai?n(e.sections):e.eachChild(t))},e=i.value;e._styleExpression&&t(e._styleExpression.expression)}return s}}let eh;var ih={get paint(){return eh=eh||new Ms({"background-color":new Is(lt.paint_background["background-color"]),"background-pattern":new Cs(lt.paint_background["background-pattern"]),"background-opacity":new Is(lt.paint_background["background-opacity"])})}};class rh extends ks{constructor(t){super(t,ih)}}let sh;var nh={get paint(){return sh=sh||new Ms({"raster-opacity":new Is(lt.paint_raster["raster-opacity"]),"raster-hue-rotate":new Is(lt.paint_raster["raster-hue-rotate"]),"raster-brightness-min":new Is(lt.paint_raster["raster-brightness-min"]),"raster-brightness-max":new Is(lt.paint_raster["raster-brightness-max"]),"raster-saturation":new Is(lt.paint_raster["raster-saturation"]),"raster-contrast":new Is(lt.paint_raster["raster-contrast"]),"raster-resampling":new Is(lt.paint_raster["raster-resampling"]),"raster-fade-duration":new Is(lt.paint_raster["raster-fade-duration"])})}};class ah extends ks{constructor(t){super(t,nh)}}class oh extends ks{constructor(t){super(t,{}),this.onAdd=t=>{this.implementation.onAdd&&this.implementation.onAdd(t,t.painter.context.gl)},this.onRemove=t=>{this.implementation.onRemove&&this.implementation.onRemove(t,t.painter.context.gl)},this.implementation=t}is3D(){return"3d"===this.implementation.renderingMode}hasOffscreenPass(){return void 0!==this.implementation.prerender}recalculate(){}updateTransitions(){}hasTransition(){return!1}serialize(){throw new Error("Custom layers cannot be serialized")}}function lh(t){if("custom"===t.type)return new oh(t);switch(t.type){case"background":return new rh(t);case"circle":return new ao(t);case"fill":return new hl(t);case"fill-extrusion":return new Ll(t);case"heatmap":return new yo(t);case"hillshade":return new bo(t);case"line":return new Kl(t);case"raster":return new ah(t);case"symbol":return new th(t)}}function ch(t){const e=[];if("string"==typeof t)e.push({id:"default",url:t});else if(t&&t.length>0){const i=[];for(const{id:r,url:s}of t){const t=`${r}${s}`;-1===i.indexOf(t)&&(i.push(t),e.push({id:r,url:s}))}}return e}function hh(t,e,i,r,s){if(r)return void t(r);if(s!==Object.values(e).length||s!==Object.values(i).length)return;const n={};for(const t in e){n[t]={};const r=B.getImageCanvasContext(i[t]),s=e[t];for(const e in s){const{width:i,height:a,x:o,y:l,sdf:c,pixelRatio:h,stretchX:u,stretchY:p,content:d}=s[e];n[t][e]={data:null,pixelRatio:h,sdf:c,stretchX:u,stretchY:p,content:d,spriteData:{width:i,height:a,x:o,y:l,context:r}}}}t(null,n)}class uh{constructor(t,e,i,r){this.context=t,this.format=i,this.texture=t.gl.createTexture(),this.update(e,r)}update(t,e,i){const{width:r,height:s}=t,n=!(this.size&&this.size[0]===r&&this.size[1]===s||i),{context:a}=this,{gl:o}=a;if(this.useMipmap=Boolean(e&&e.useMipmap),o.bindTexture(o.TEXTURE_2D,this.texture),a.pixelStoreUnpackFlipY.set(!1),a.pixelStoreUnpack.set(1),a.pixelStoreUnpackPremultiplyAlpha.set(this.format===o.RGBA&&(!e||!1!==e.premultiply)),n)this.size=[r,s],t instanceof HTMLImageElement||t instanceof HTMLCanvasElement||t instanceof HTMLVideoElement||t instanceof ImageData||k(t)?o.texImage2D(o.TEXTURE_2D,0,this.format,this.format,o.UNSIGNED_BYTE,t):o.texImage2D(o.TEXTURE_2D,0,this.format,r,s,0,this.format,o.UNSIGNED_BYTE,t.data);else{const{x:e,y:n}=i||{x:0,y:0};t instanceof HTMLImageElement||t instanceof HTMLCanvasElement||t instanceof HTMLVideoElement||t instanceof ImageData||k(t)?o.texSubImage2D(o.TEXTURE_2D,0,e,n,o.RGBA,o.UNSIGNED_BYTE,t):o.texSubImage2D(o.TEXTURE_2D,0,e,n,r,s,o.RGBA,o.UNSIGNED_BYTE,t.data)}this.useMipmap&&this.isSizePowerOfTwo()&&o.generateMipmap(o.TEXTURE_2D)}bind(t,e,i){const{context:r}=this,{gl:s}=r;s.bindTexture(s.TEXTURE_2D,this.texture),i!==s.LINEAR_MIPMAP_NEAREST||this.isSizePowerOfTwo()||(i=s.LINEAR),t!==this.filter&&(s.texParameteri(s.TEXTURE_2D,s.TEXTURE_MAG_FILTER,t),s.texParameteri(s.TEXTURE_2D,s.TEXTURE_MIN_FILTER,i||t),this.filter=t),e!==this.wrap&&(s.texParameteri(s.TEXTURE_2D,s.TEXTURE_WRAP_S,e),s.texParameteri(s.TEXTURE_2D,s.TEXTURE_WRAP_T,e),this.wrap=e)}isSizePowerOfTwo(){return this.size[0]===this.size[1]&&Math.log(this.size[0])/Math.LN2%1==0}destroy(){const{gl:t}=this.context;t.deleteTexture(this.texture),this.texture=null}}function ph(t){const{userImage:e}=t;return!!(e&&e.render&&e.render())&&(t.data.replace(new Uint8Array(e.data.buffer)),!0)}class dh extends ot{constructor(){super(),this.images={},this.updatedImages={},this.callbackDispatchedThisFrame={},this.loaded=!1,this.requestors=[],this.patterns={},this.atlasImage=new _o({width:1,height:1}),this.dirty=!0}isLoaded(){return this.loaded}setLoaded(t){if(this.loaded!==t&&(this.loaded=t,t)){for(const{ids:t,callback:e}of this.requestors)this._notify(t,e);this.requestors=[]}}getImage(t){const e=this.images[t];if(e&&!e.data&&e.spriteData){const t=e.spriteData;e.data=new _o({width:t.width,height:t.height},t.context.getImageData(t.x,t.y,t.width,t.height).data),e.spriteData=null}return e}addImage(t,e){if(this.images[t])throw new Error(`Image id ${t} already exist, use updateImage instead`);this._validate(t,e)&&(this.images[t]=e)}_validate(t,e){let i=!0;const r=e.data||e.spriteData;return this._validateStretch(e.stretchX,r&&r.width)||(this.fire(new at(new Error(`Image "${t}" has invalid "stretchX" value`))),i=!1),this._validateStretch(e.stretchY,r&&r.height)||(this.fire(new at(new Error(`Image "${t}" has invalid "stretchY" value`))),i=!1),this._validateContent(e.content,e)||(this.fire(new at(new Error(`Image "${t}" has invalid "content" value`))),i=!1),i}_validateStretch(t,e){if(!t)return!0;let i=0;for(const r of t){if(r[0]-1);l++,n[l]=o,a[l]=c,a[l+1]=mh}for(let o=0,l=0;o{let r=this.entries[t];r||(r=this.entries[t]={glyphs:{},requests:{},ranges:{}});let s=r.glyphs[e];if(void 0!==s)return void i(null,{stack:t,id:e,glyph:s});if(s=this._tinySDF(r,t,e),s)return r.glyphs[e]=s,void i(null,{stack:t,id:e,glyph:s});const n=Math.floor(e/256);if(256*n>65535)return void i(new Error("glyphs > 65535 not supported"));if(r.ranges[n])return void i(null,{stack:t,id:e,glyph:s});if(!this.url)return void i(new Error("glyphsUrl is not set"));let a=r.requests[n];a||(a=r.requests[n]=[],gh.loadGlyphRange(t,n,this.url,this.requestManager,((t,e)=>{if(e){for(const t in e)this._doesCharSupportLocalGlyph(+t)||(r.glyphs[+t]=e[+t]);r.ranges[n]=!0}for(const i of a)i(t,e);delete r.requests[n]}))),a.push(((r,s)=>{r?i(r):s&&i(null,{stack:t,id:e,glyph:s[e]||null})}))}),((t,i)=>{if(t)e(t);else if(i){const t={};for(const{stack:e,id:r,glyph:s}of i)(t[e]||(t[e]={}))[r]=s&&{id:s.id,bitmap:s.bitmap.clone(),metrics:s.metrics};e(null,t)}}))}_doesCharSupportLocalGlyph(t){return!!this.localIdeographFontFamily&&(Jr["CJK Unified Ideographs"](t)||Jr["Hangul Syllables"](t)||Jr.Hiragana(t)||Jr.Katakana(t))}_tinySDF(t,e,i){const r=this.localIdeographFontFamily;if(!r)return;if(!this._doesCharSupportLocalGlyph(i))return;let s=t.tinySDF;if(!s){let i="400";/bold/i.test(e)?i="900":/medium/i.test(e)?i="500":/light/i.test(e)&&(i="200"),s=t.tinySDF=new gh.TinySDF({fontSize:48,buffer:6,radius:16,cutoff:.25,fontFamily:r,fontWeight:i})}const n=s.draw(String.fromCharCode(i));return{id:i,bitmap:new fo({width:n.width||60,height:n.height||60},n.data),metrics:{width:n.glyphWidth/2||24,height:n.glyphHeight/2||24,left:n.glyphLeft/2+.5||0,top:n.glyphTop/2-27.5||-8,advance:n.glyphAdvance/2||24,isDoubleResolution:!0}}}}gh.loadGlyphRange=function(t,e,i,r,s){const n=256*e,a=n+255,o=r.transformRequest(i.replace("{fontstack}",t).replace("{range}",`${n}-${a}`),tt.Glyphs);j(o,((t,e)=>{if(t)s(t);else if(e){const t={};for(const i of function(t){return new Ec(t).readFields(zc,[])}(e))t[i.id]=i;s(null,t)}}))},gh.TinySDF=class{constructor({fontSize:t=24,buffer:e=3,radius:i=8,cutoff:r=.25,fontFamily:s="sans-serif",fontWeight:n="normal",fontStyle:a="normal"}={}){this.buffer=e,this.cutoff=r,this.radius=i;const o=this.size=t+4*e,l=this._createCanvas(o),c=this.ctx=l.getContext("2d",{willReadFrequently:!0});c.font=`${a} ${n} ${t}px ${s}`,c.textBaseline="alphabetic",c.textAlign="left",c.fillStyle="black",this.gridOuter=new Float64Array(o*o),this.gridInner=new Float64Array(o*o),this.f=new Float64Array(o),this.z=new Float64Array(o+1),this.v=new Uint16Array(o)}_createCanvas(t){const e=document.createElement("canvas");return e.width=e.height=t,e}draw(t){const{width:e,actualBoundingBoxAscent:i,actualBoundingBoxDescent:r,actualBoundingBoxLeft:s,actualBoundingBoxRight:n}=this.ctx.measureText(t),a=Math.ceil(i),o=Math.max(0,Math.min(this.size-this.buffer,Math.ceil(n-s))),l=Math.min(this.size-this.buffer,a+Math.ceil(r)),c=o+2*this.buffer,h=l+2*this.buffer,u=Math.max(c*h,0),p=new Uint8ClampedArray(u),d={data:p,width:c,height:h,glyphWidth:o,glyphHeight:l,glyphTop:a,glyphLeft:0,glyphAdvance:e};if(0===o||0===l)return d;const{ctx:m,buffer:f,gridInner:_,gridOuter:g}=this;m.clearRect(f,f,o,l),m.fillText(t,f,f+a);const y=m.getImageData(f,f,o,l);g.fill(mh,0,u),_.fill(0,0,u);for(let t=0;t0?t*t:0,_[r]=t<0?t*t:0}}fh(g,0,0,c,h,c,this.f,this.v,this.z),fh(_,f,f,o,l,c,this.f,this.v,this.z);for(let t=0;t1&&(a=t[++n]);const l=Math.abs(o-a.left),c=Math.abs(o-a.right),h=Math.min(l,c);let u;const p=e/i*(r+1);if(a.isDash){const t=r-Math.abs(p);u=Math.sqrt(h*h+t*t)}else u=r-Math.sqrt(h*h+p*p);this.data[s+o]=Math.max(0,Math.min(255,u+128))}}}addRegularDash(t){for(let e=t.length-1;e>=0;--e){const i=t[e],r=t[e+1];i.zeroLength?t.splice(e,1):r&&r.isDash===i.isDash&&(r.left=i.left,t.splice(e,1))}const e=t[0],i=t[t.length-1];e.isDash===i.isDash&&(e.left=i.left-this.width,i.right=e.right+this.width);const r=this.width*this.nextRow;let s=0,n=t[s];for(let e=0;e1&&(n=t[++s]);const i=Math.abs(e-n.left),a=Math.abs(e-n.right),o=Math.min(i,a);this.data[r+e]=Math.max(0,Math.min(255,(n.isDash?o:-o)+128))}}addDash(t,e){const i=e?7:0,r=2*i+1;if(this.nextRow+r>this.height)return A("LineAtlas out of space"),null;let s=0;for(let e=0;e{this._triggered=!1,this._callback()})}trigger(){this._triggered||(this._triggered=!0,this._channel?this._channel.port1.postMessage(!0):setTimeout((()=>{this._triggered=!1,this._callback()}),0))}remove(){delete this._channel,this._callback=()=>{}}}class Th{constructor(t,e,i){this.receive=t=>{const e=t.data,i=e.id;if(i&&(!e.targetMapId||this.mapId===e.targetMapId))if(""===e.type){delete this.tasks[i];const t=this.cancelCallbacks[i];delete this.cancelCallbacks[i],t&&t()}else z()||e.mustQueue?(this.tasks[i]=e,this.taskQueue.push(i),this.invoker.trigger()):this.processTask(i,e)},this.process=()=>{if(!this.taskQueue.length)return;const t=this.taskQueue.shift(),e=this.tasks[t];delete this.tasks[t],this.taskQueue.length&&this.invoker.trigger(),e&&this.processTask(t,e)},this.target=t,this.parent=e,this.mapId=i,this.callbacks={},this.tasks={},this.taskQueue=[],this.cancelCallbacks={},this.invoker=new wh(this.process),this.target.addEventListener("message",this.receive,!1),this.globalScope=z()?t:window}send(t,e,i,r,s=!1){const n=Math.round(1e18*Math.random()).toString(36).substring(0,10);i&&(this.callbacks[n]=i);const a=[],o={id:n,type:t,hasCallback:!!i,targetMapId:r,mustQueue:s,sourceMapId:this.mapId,data:Xr(e,a)};return this.target.postMessage(o,{transfer:a}),{cancel:()=>{i&&delete this.callbacks[n],this.target.postMessage({id:n,type:"",targetMapId:r,sourceMapId:this.mapId})}}}processTask(t,e){if(""===e.type){const i=this.callbacks[t];delete this.callbacks[t],i&&(e.error?i(Kr(e.error)):i(null,Kr(e.data)))}else{let i=!1;const r=[],s=e.hasCallback?(e,s)=>{i=!0,delete this.cancelCallbacks[t];const n={id:t,type:"",sourceMapId:this.mapId,error:e?Xr(e):null,data:Xr(s,r)};this.target.postMessage(n,{transfer:r})}:t=>{i=!0};let n=null;const a=Kr(e.data);if(this.parent[e.type])n=this.parent[e.type](e.sourceMapId,a,s);else if("getWorkerSource"in this.parent){const t=e.type.split(".");n=this.parent.getWorkerSource(e.sourceMapId,t[0],a.source)[t[1]](a,s)}else s(new Error(`Could not find function ${e.type}`));!i&&n&&n.cancel&&(this.cancelCallbacks[t]=n.cancel)}}remove(){this.invoker.remove(),this.target.removeEventListener("message",this.receive,!1)}}class Sh{constructor(t,e,i){this.workerPool=t,this.actors=[],this.currentActor=0,this.id=i;const r=this.workerPool.acquire(i);for(let t=0;t{i.send(t,e,r)}),i=i||function(){})}getActor(){return this.currentActor=(this.currentActor+1)%this.actors.length,this.actors[this.currentActor]}remove(t=!0){this.actors.forEach((t=>{t.remove()})),this.actors=[],t&&this.workerPool.release(this.id)}}function Ih(t,e,i){const r=function(e,r){if(e)return i(e);if(r){const e=y(g(r,t),["tiles","minzoom","maxzoom","attribution","bounds","scheme","tileSize","encoding"]);r.vector_layers&&(e.vectorLayers=r.vector_layers,e.vectorLayerIds=e.vectorLayers.map((t=>t.id))),i(null,e)}};return t.url?Z(e.transformRequest(t.url,tt.Source),r):B.frame((()=>r(null,t)))}const Ah=6371008.8;class Eh{constructor(t,e){if(isNaN(t)||isNaN(e))throw new Error(`Invalid LngLat object: (${t}, ${e})`);if(this.lng=+t,this.lat=+e,this.lat>90||this.lat<-90)throw new Error("Invalid LngLat latitude value: must be between -90 and 90")}wrap(){return new Eh(f(this.lng,-180,180),this.lat)}toArray(){return[this.lng,this.lat]}toString(){return`LngLat(${this.lng}, ${this.lat})`}distanceTo(t){const e=Math.PI/180,i=this.lat*e,r=t.lat*e,s=Math.sin(i)*Math.sin(r)+Math.cos(i)*Math.cos(r)*Math.cos((t.lng-this.lng)*e);return Ah*Math.acos(Math.min(s,1))}static convert(t){if(t instanceof Eh)return t;if(Array.isArray(t)&&(2===t.length||3===t.length))return new Eh(Number(t[0]),Number(t[1]));if(!Array.isArray(t)&&"object"==typeof t&&null!==t)return new Eh(Number("lng"in t?t.lng:t.lon),Number(t.lat));throw new Error("`LngLatLike` argument must be specified as a LngLat instance, an object {lng: , lat: }, an object {lon: , lat: }, or an array of [, ]")}}class Ch{constructor(t,e){t&&(e?this.setSouthWest(t).setNorthEast(e):Array.isArray(t)&&(4===t.length?this.setSouthWest([t[0],t[1]]).setNorthEast([t[2],t[3]]):this.setSouthWest(t[0]).setNorthEast(t[1])))}setNorthEast(t){return this._ne=t instanceof Eh?new Eh(t.lng,t.lat):Eh.convert(t),this}setSouthWest(t){return this._sw=t instanceof Eh?new Eh(t.lng,t.lat):Eh.convert(t),this}extend(t){const e=this._sw,i=this._ne;let r,s;if(t instanceof Eh)r=t,s=t;else{if(!(t instanceof Ch))return Array.isArray(t)?4===t.length||t.every(Array.isArray)?this.extend(Ch.convert(t)):this.extend(Eh.convert(t)):t&&("lng"in t||"lon"in t)&&"lat"in t?this.extend(Eh.convert(t)):this;if(r=t._sw,s=t._ne,!r||!s)return this}return e||i?(e.lng=Math.min(r.lng,e.lng),e.lat=Math.min(r.lat,e.lat),i.lng=Math.max(s.lng,i.lng),i.lat=Math.max(s.lat,i.lat)):(this._sw=new Eh(r.lng,r.lat),this._ne=new Eh(s.lng,s.lat)),this}getCenter(){return new Eh((this._sw.lng+this._ne.lng)/2,(this._sw.lat+this._ne.lat)/2)}getSouthWest(){return this._sw}getNorthEast(){return this._ne}getNorthWest(){return new Eh(this.getWest(),this.getNorth())}getSouthEast(){return new Eh(this.getEast(),this.getSouth())}getWest(){return this._sw.lng}getSouth(){return this._sw.lat}getEast(){return this._ne.lng}getNorth(){return this._ne.lat}toArray(){return[this._sw.toArray(),this._ne.toArray()]}toString(){return`LngLatBounds(${this._sw.toString()}, ${this._ne.toString()})`}isEmpty(){return!(this._sw&&this._ne)}contains(t){const{lng:e,lat:i}=Eh.convert(t);let r=this._sw.lng<=e&&e<=this._ne.lng;return this._sw.lng>this._ne.lng&&(r=this._sw.lng>=e&&e>=this._ne.lng),this._sw.lat<=i&&i<=this._ne.lat&&r}static convert(t){return t instanceof Ch?t:t?new Ch(t):t}static fromLngLat(t,e=0){const i=360*e/40075017,r=i/Math.cos(Math.PI/180*t.lat);return new Ch(new Eh(t.lng-r,t.lat-i),new Eh(t.lng+r,t.lat+i))}}const zh=2*Math.PI*Ah;function Mh(t){return zh*Math.cos(t*Math.PI/180)}function Ph(t){return(180+t)/360}function kh(t){return(180-180/Math.PI*Math.log(Math.tan(Math.PI/4+t*Math.PI/360)))/360}function Dh(t,e){return t/Mh(e)}function Lh(t){return 360*t-180}function Rh(t){return 360/Math.PI*Math.atan(Math.exp((180-360*t)*Math.PI/180))-90}class Bh{constructor(t,e,i=0){this.x=+t,this.y=+e,this.z=+i}static fromLngLat(t,e=0){const i=Eh.convert(t);return new Bh(Ph(i.lng),kh(i.lat),Dh(e,i.lat))}toLngLat(){return new Eh(Lh(this.x),Rh(this.y))}toAltitude(){return this.z*Mh(Rh(this.y))}meterInMercatorCoordinateUnits(){return 1/zh*(t=Rh(this.y),1/Math.cos(t*Math.PI/180));var t}}class Fh{constructor(t,e,i){this.bounds=Ch.convert(this.validateBounds(t)),this.minzoom=e||0,this.maxzoom=i||24}validateBounds(t){return Array.isArray(t)&&4===t.length?[Math.max(-180,t[0]),Math.max(-90,t[1]),Math.min(180,t[2]),Math.min(90,t[3])]:[-180,-90,180,90]}contains(t){const e=Math.pow(2,t.z),i=Math.floor(Ph(this.bounds.getWest())*e),r=Math.floor(kh(this.bounds.getNorth())*e),s=Math.ceil(Ph(this.bounds.getEast())*e),n=Math.ceil(kh(this.bounds.getSouth())*e);return t.x>=i&&t.x=r&&t.y{this._loaded=!1,this.fire(new nt("dataloading",{dataType:"source"})),this._tileJSONRequest=Ih(this._options,this.map._requestManager,((t,e)=>{this._tileJSONRequest=null,this._loaded=!0,this.map.style.sourceCaches[this.id].clearTiles(),t?this.fire(new at(t)):e&&(g(this,e),e.bounds&&(this.tileBounds=new Fh(e.bounds,this.minzoom,this.maxzoom)),this.fire(new nt("data",{dataType:"source",sourceDataType:"metadata"})),this.fire(new nt("data",{dataType:"source",sourceDataType:"content"})))}))},this.serialize=()=>g({},this._options),this.id=t,this.dispatcher=i,this.type="vector",this.minzoom=0,this.maxzoom=22,this.scheme="xyz",this.tileSize=512,this.reparseOverscaled=!0,this.isTileClipped=!0,this._loaded=!1,g(this,y(e,["url","scheme","tileSize","promoteId"])),this._options=g({type:"vector"},e),this._collectResourceTiming=e.collectResourceTiming,512!==this.tileSize)throw new Error("vector tile sources must have a tileSize of 512");this.setEventedParent(r)}loaded(){return this._loaded}hasTile(t){return!this.tileBounds||this.tileBounds.contains(t.canonical)}onAdd(t){this.map=t,this.load()}setSourceProperty(t){this._tileJSONRequest&&this._tileJSONRequest.cancel(),t(),this.load()}setTiles(t){return this.setSourceProperty((()=>{this._options.tiles=t})),this}setUrl(t){return this.setSourceProperty((()=>{this.url=t,this._options.url=t})),this}onRemove(){this._tileJSONRequest&&(this._tileJSONRequest.cancel(),this._tileJSONRequest=null)}loadTile(t,e){const i=t.tileID.canonical.url(this.tiles,this.map.getPixelRatio(),this.scheme),r={request:this.map._requestManager.transformRequest(i,tt.Tile),uid:t.uid,tileID:t.tileID,zoom:t.tileID.overscaledZ,tileSize:this.tileSize*t.tileID.overscaleFactor(),type:this.type,source:this.id,pixelRatio:this.map.getPixelRatio(),showCollisionBoxes:this.map.showCollisionBoxes,promoteId:this.promoteId};function s(i,r){return delete t.request,t.aborted?e(null):i&&404!==i.status?e(i):(r&&r.resourceTiming&&(t.resourceTiming=r.resourceTiming),this.map._refreshExpiredTiles&&r&&t.setExpiryData(r),t.loadVectorData(r,this.map.painter),e(null),void(t.reloadCallback&&(this.loadTile(t,t.reloadCallback),t.reloadCallback=null)))}r.request.collectResourceTiming=this._collectResourceTiming,t.actor&&"expired"!==t.state?"loading"===t.state?t.reloadCallback=e:t.request=t.actor.send("reloadTile",r,s.bind(this)):(t.actor=this.dispatcher.getActor(),t.request=t.actor.send("loadTile",r,s.bind(this)))}abortTile(t){t.request&&(t.request.cancel(),delete t.request),t.actor&&t.actor.send("abortTile",{uid:t.uid,type:this.type,source:this.id},void 0)}unloadTile(t){t.unloadVectorData(),t.actor&&t.actor.send("removeTile",{uid:t.uid,type:this.type,source:this.id},void 0)}hasTransition(){return!1}}class Vh extends ot{constructor(t,e,i,r){super(),this.id=t,this.dispatcher=i,this.setEventedParent(r),this.type="raster",this.minzoom=0,this.maxzoom=22,this.roundZoom=!0,this.scheme="xyz",this.tileSize=512,this._loaded=!1,this._options=g({type:"raster"},e),g(this,y(e,["url","scheme","tileSize"]))}load(){this._loaded=!1,this.fire(new nt("dataloading",{dataType:"source"})),this._tileJSONRequest=Ih(this._options,this.map._requestManager,((t,e)=>{this._tileJSONRequest=null,this._loaded=!0,t?this.fire(new at(t)):e&&(g(this,e),e.bounds&&(this.tileBounds=new Fh(e.bounds,this.minzoom,this.maxzoom)),this.fire(new nt("data",{dataType:"source",sourceDataType:"metadata"})),this.fire(new nt("data",{dataType:"source",sourceDataType:"content"})))}))}loaded(){return this._loaded}onAdd(t){this.map=t,this.load()}onRemove(){this._tileJSONRequest&&(this._tileJSONRequest.cancel(),this._tileJSONRequest=null)}setSourceProperty(t){this._tileJSONRequest&&this._tileJSONRequest.cancel(),t(),this.load()}setTiles(t){return this.setSourceProperty((()=>{this._options.tiles=t})),this}serialize(){return g({},this._options)}hasTile(t){return!this.tileBounds||this.tileBounds.contains(t.canonical)}loadTile(t,e){const i=t.tileID.canonical.url(this.tiles,this.map.getPixelRatio(),this.scheme);t.request=Q.getImage(this.map._requestManager.transformRequest(i,tt.Tile),((i,r,s)=>{if(delete t.request,t.aborted)t.state="unloaded",e(null);else if(i)t.state="errored",e(i);else if(r){this.map._refreshExpiredTiles&&s&&t.setExpiryData(s);const i=this.map.painter.context,n=i.gl;t.texture=this.map.painter.getTileTexture(r.width),t.texture?t.texture.update(r,{useMipmap:!0}):(t.texture=new uh(i,r,n.RGBA,{useMipmap:!0}),t.texture.bind(n.LINEAR,n.CLAMP_TO_EDGE,n.LINEAR_MIPMAP_NEAREST),i.extTextureFilterAnisotropic&&n.texParameterf(n.TEXTURE_2D,i.extTextureFilterAnisotropic.TEXTURE_MAX_ANISOTROPY_EXT,i.extTextureFilterAnisotropicMax)),t.state="loaded",e(null)}}),this.map._refreshExpiredTiles)}abortTile(t,e){t.request&&(t.request.cancel(),delete t.request),e()}unloadTile(t,e){t.texture&&this.map.painter.saveTileTexture(t.texture),e()}hasTransition(){return!1}}function Uh(t,e,i){var r=2*Math.PI*6378137/256/Math.pow(2,i);return[t*r-2*Math.PI*6378137/2,e*r-2*Math.PI*6378137/2]}class Nh{constructor(t,e,i){if(t<0||t>25||i<0||i>=Math.pow(2,t)||e<0||e>=Math.pow(2,t))throw new Error(`x=${e}, y=${i}, z=${t} outside of bounds. 0<=x<${Math.pow(2,t)}, 0<=y<${Math.pow(2,t)} 0<=z<=25 `);this.z=t,this.x=e,this.y=i,this.key=Zh(0,t,t,e,i)}equals(t){return this.z===t.z&&this.x===t.x&&this.y===t.y}url(t,e,i){const r=(n=this.y,a=this.z,o=Uh(256*(s=this.x),256*(n=Math.pow(2,a)-n-1),a),l=Uh(256*(s+1),256*(n+1),a),o[0]+","+o[1]+","+l[0]+","+l[1]);var s,n,a,o,l;const c=function(t,e,i){let r,s="";for(let n=t;n>0;n--)r=1<1?"@2x":"").replace(/{quadkey}/g,c).replace(/{bbox-epsg-3857}/g,r)}isChildOf(t){const e=this.z-t.z;return e>0&&t.x===this.x>>e&&t.y===this.y>>e}getTilePoint(t){const e=Math.pow(2,this.z);return new n((t.x*e-this.x)*va,(t.y*e-this.y)*va)}toString(){return`${this.z}/${this.x}/${this.y}`}}class $h{constructor(t,e){this.wrap=t,this.canonical=e,this.key=Zh(t,e.z,e.z,e.x,e.y)}}class qh{constructor(t,e,i,r,s){if(t= z; overscaledZ = ${t}; z = ${i}`);this.overscaledZ=t,this.wrap=e,this.canonical=new Nh(i,+r,+s),this.key=Zh(e,t,i,r,s)}clone(){return new qh(this.overscaledZ,this.wrap,this.canonical.z,this.canonical.x,this.canonical.y)}equals(t){return this.overscaledZ===t.overscaledZ&&this.wrap===t.wrap&&this.canonical.equals(t.canonical)}scaledTo(t){if(t>this.overscaledZ)throw new Error(`targetZ > this.overscaledZ; targetZ = ${t}; overscaledZ = ${this.overscaledZ}`);const e=this.canonical.z-t;return t>this.canonical.z?new qh(t,this.wrap,this.canonical.z,this.canonical.x,this.canonical.y):new qh(t,this.wrap,t,this.canonical.x>>e,this.canonical.y>>e)}calculateScaledKey(t,e){if(t>this.overscaledZ)throw new Error(`targetZ > this.overscaledZ; targetZ = ${t}; overscaledZ = ${this.overscaledZ}`);const i=this.canonical.z-t;return t>this.canonical.z?Zh(this.wrap*+e,t,this.canonical.z,this.canonical.x,this.canonical.y):Zh(this.wrap*+e,t,t,this.canonical.x>>i,this.canonical.y>>i)}isChildOf(t){if(t.wrap!==this.wrap)return!1;const e=this.canonical.z-t.canonical.z;return 0===t.overscaledZ||t.overscaledZ>e&&t.canonical.y===this.canonical.y>>e}children(t){if(this.overscaledZ>=t)return[new qh(this.overscaledZ+1,this.wrap,this.canonical.z,this.canonical.x,this.canonical.y)];const e=this.canonical.z+1,i=2*this.canonical.x,r=2*this.canonical.y;return[new qh(e,this.wrap,e,i,r),new qh(e,this.wrap,e,i+1,r),new qh(e,this.wrap,e,i,r+1),new qh(e,this.wrap,e,i+1,r+1)]}isLessThan(t){return this.wrapt.wrap)&&(this.overscaledZt.overscaledZ)&&(this.canonical.xt.canonical.x)&&this.canonical.ythis.max&&(this.max=i),i=this.dim+1||e<-1||e>=this.dim+1)throw new RangeError("out of range source coordinates for DEM data");return(e+1)*this.stride+(t+1)}unpack(t,e,i){return t*this.redFactor+e*this.greenFactor+i*this.blueFactor-this.baseShift}getPixels(){return new _o({width:this.stride,height:this.stride},new Uint8Array(this.data.buffer))}backfillBorder(t,e,i){if(this.dim!==t.dim)throw new Error("dem dimension mismatch");let r=e*this.dim,s=e*this.dim+this.dim,n=i*this.dim,a=i*this.dim+this.dim;switch(e){case-1:r=s-1;break;case 1:s=r+1}switch(i){case-1:n=a-1;break;case 1:a=n+1}const o=-e*this.dim,l=-i*this.dim;for(let e=n;ee(this,void 0,void 0,(function*(){if(delete t.request,t.aborted)t.state="unloaded",i(null);else if(r)t.state="errored",i(r);else if(s){this.map._refreshExpiredTiles&&t.setExpiryData(a);const i=k(s)&&u()?s:yield function(t){return e(this,void 0,void 0,(function*(){if("undefined"!=typeof VideoFrame&&function(){if(null==h&&(h=!1,u())){const t=5,e=new OffscreenCanvas(t,t).getContext("2d",{willReadFrequently:!0});if(e){for(let i=0;i0&&(o[new qh(t.overscaledZ,s,e.z,r,e.y-1).key]={backfilled:!1},o[new qh(t.overscaledZ,t.wrap,e.z,e.x,e.y-1).key]={backfilled:!1},o[new qh(t.overscaledZ,a,e.z,n,e.y-1).key]={backfilled:!1}),e.y+1{this._updateWorkerData()},this.serialize=()=>g({},this._options,{type:this.type,data:this._data}),this.id=t,this.type="geojson",this.minzoom=0,this.maxzoom=18,this.tileSize=512,this.isTileClipped=!0,this.reparseOverscaled=!0,this._removed=!1,this._pendingLoads=0,this.actor=i.getActor(),this.setEventedParent(r),this._data=e.data,this._options=g({},e),this._collectResourceTiming=e.collectResourceTiming,void 0!==e.maxzoom&&(this.maxzoom=e.maxzoom),e.type&&(this.type=e.type),e.attribution&&(this.attribution=e.attribution),this.promoteId=e.promoteId;const s=va/this.tileSize;this.workerOptions=g({source:this.id,cluster:e.cluster||!1,geojsonVtOptions:{buffer:(void 0!==e.buffer?e.buffer:128)*s,tolerance:(void 0!==e.tolerance?e.tolerance:.375)*s,extent:va,maxZoom:this.maxzoom,lineMetrics:e.lineMetrics||!1,generateId:e.generateId||!1},superclusterOptions:{maxZoom:void 0!==e.clusterMaxZoom?e.clusterMaxZoom:this.maxzoom-1,minPoints:Math.max(2,e.clusterMinPoints||2),extent:va,radius:(e.clusterRadius||50)*s,log:!1,generateId:e.generateId||!1},clusterProperties:e.clusterProperties,filter:e.filter},e.workerOptions),"string"==typeof this.promoteId&&(this.workerOptions.promoteId=this.promoteId)}onAdd(t){this.map=t,this.load()}setData(t){return this._data=t,this._updateWorkerData(),this}updateData(t){return this._updateWorkerData(t),this}setClusterOptions(t){return this.workerOptions.cluster=t.cluster,t&&(void 0!==t.clusterRadius&&(this.workerOptions.superclusterOptions.radius=t.clusterRadius),void 0!==t.clusterMaxZoom&&(this.workerOptions.superclusterOptions.maxZoom=t.clusterMaxZoom)),this._updateWorkerData(),this}getClusterExpansionZoom(t,e){return this.actor.send("geojson.getClusterExpansionZoom",{clusterId:t,source:this.id},e),this}getClusterChildren(t,e){return this.actor.send("geojson.getClusterChildren",{clusterId:t,source:this.id},e),this}getClusterLeaves(t,e,i,r){return this.actor.send("geojson.getClusterLeaves",{source:this.id,clusterId:t,limit:e,offset:i},r),this}_updateWorkerData(t){const e=g({},this.workerOptions);t?e.dataDiff=t:"string"==typeof this._data?(e.request=this.map._requestManager.transformRequest(B.resolveURL(this._data),tt.Source),e.request.collectResourceTiming=this._collectResourceTiming):e.data=JSON.stringify(this._data),this._pendingLoads++,this.fire(new nt("dataloading",{dataType:"source"})),this.actor.send(`${this.type}.loadData`,e,((t,e)=>{if(this._pendingLoads--,this._removed||e&&e.abandoned)return void this.fire(new nt("dataabort",{dataType:"source"}));let i=null;if(e&&e.resourceTiming&&e.resourceTiming[this.id]&&(i=e.resourceTiming[this.id].slice(0)),t)return void this.fire(new at(t));const r={dataType:"source"};this._collectResourceTiming&&i&&i.length>0&&g(r,{resourceTiming:i}),this.fire(new nt("data",Object.assign(Object.assign({},r),{sourceDataType:"metadata"}))),this.fire(new nt("data",Object.assign(Object.assign({},r),{sourceDataType:"content"})))}))}loaded(){return 0===this._pendingLoads}loadTile(t,e){const i=t.actor?"reloadTile":"loadTile";t.actor=this.actor;const r={type:this.type,uid:t.uid,tileID:t.tileID,zoom:t.tileID.overscaledZ,maxZoom:this.maxzoom,tileSize:this.tileSize,source:this.id,pixelRatio:this.map.getPixelRatio(),showCollisionBoxes:this.map.showCollisionBoxes,promoteId:this.promoteId};t.request=this.actor.send(i,r,((r,s)=>(delete t.request,t.unloadVectorData(),t.aborted?e(null):r?e(r):(t.loadVectorData(s,this.map.painter,"reloadTile"===i),e(null)))))}abortTile(t){t.request&&(t.request.cancel(),delete t.request),t.aborted=!0}unloadTile(t){t.unloadVectorData(),this.actor.send("removeTile",{uid:t.uid,type:this.type,source:this.id})}onRemove(){this._removed=!0,this.actor.send("removeSource",{type:this.type,source:this.id})}hasTransition(){return!1}}var Hh=Bs([{name:"a_pos",type:"Int16",components:2},{name:"a_texture_pos",type:"Int16",components:2}]);class Wh extends ot{constructor(t,e,i,r){super(),this.load=(t,e)=>{this._loaded=!1,this.fire(new nt("dataloading",{dataType:"source"})),this.url=this.options.url,this._request=Q.getImage(this.map._requestManager.transformRequest(this.url,tt.Image),((i,r)=>{this._request=null,this._loaded=!0,i?this.fire(new at(i)):r&&(this.image=r,t&&(this.coordinates=t),e&&e(),this._finishLoading())}))},this.prepare=()=>{if(0===Object.keys(this.tiles).length||!this.image)return;const t=this.map.painter.context,e=t.gl;this.boundsBuffer||(this.boundsBuffer=t.createVertexBuffer(this._boundsArray,Hh.members)),this.boundsSegments||(this.boundsSegments=Nn.simpleSegment(0,0,4,2)),this.texture||(this.texture=new uh(t,this.image,e.RGBA),this.texture.bind(e.LINEAR,e.CLAMP_TO_EDGE));let i=!1;for(const t in this.tiles){const e=this.tiles[t];"loaded"!==e.state&&(e.state="loaded",e.texture=this.texture,i=!0)}i&&this.fire(new nt("data",{dataType:"source",sourceDataType:"idle",sourceId:this.id}))},this.serialize=()=>({type:"image",url:this.options.url,coordinates:this.coordinates}),this.id=t,this.dispatcher=i,this.coordinates=e.coordinates,this.type="image",this.minzoom=0,this.maxzoom=22,this.tileSize=512,this.tiles={},this._loaded=!1,this.setEventedParent(r),this.options=e}loaded(){return this._loaded}updateImage(t){return t.url?(this._request&&(this._request.cancel(),this._request=null),this.options.url=t.url,this.load(t.coordinates,(()=>{this.texture=null})),this):this}_finishLoading(){this.map&&(this.setCoordinates(this.coordinates),this.fire(new nt("data",{dataType:"source",sourceDataType:"metadata"})))}onAdd(t){this.map=t,this.load()}onRemove(){this._request&&(this._request.cancel(),this._request=null)}setCoordinates(t){this.coordinates=t;const e=t.map(Bh.fromLngLat);this.tileID=function(t){let e=1/0,i=1/0,r=-1/0,s=-1/0;for(const n of t)e=Math.min(e,n.x),i=Math.min(i,n.y),r=Math.max(r,n.x),s=Math.max(s,n.y);const n=Math.max(r-e,s-i),a=Math.max(0,Math.floor(-Math.log(n)/Math.LN2)),o=Math.pow(2,a);return new Nh(a,Math.floor((e+r)/2*o),Math.floor((i+s)/2*o))}(e),this.minzoom=this.maxzoom=this.tileID.z;const i=e.map((t=>this.tileID.getTilePoint(t)._round()));return this._boundsArray=new wn,this._boundsArray.emplaceBack(i[0].x,i[0].y,0,0),this._boundsArray.emplaceBack(i[1].x,i[1].y,va,0),this._boundsArray.emplaceBack(i[3].x,i[3].y,0,va),this._boundsArray.emplaceBack(i[2].x,i[2].y,va,va),this.boundsBuffer&&(this.boundsBuffer.destroy(),delete this.boundsBuffer),this.fire(new nt("data",{dataType:"source",sourceDataType:"content"})),this}loadTile(t,e){this.tileID&&this.tileID.equals(t.tileID.canonical)?(this.tiles[String(t.tileID.wrap)]=t,t.buckets={},e(null)):(t.state="errored",e(null))}hasTransition(){return!1}}class Xh extends Wh{constructor(t,e,i,r){super(t,e,i,r),this.load=()=>{this._loaded=!1;const t=this.options;this.urls=[];for(const e of t.urls)this.urls.push(this.map._requestManager.transformRequest(e,tt.Source).url);!function(t,e){const i=window.document.createElement("video");i.muted=!0,i.onloadstart=function(){e(null,i)};for(let e=0;e{this._loaded=!0,t?this.fire(new at(t)):e&&(this.video=e,this.video.loop=!0,this.video.addEventListener("playing",(()=>{this.map.triggerRepaint()})),this.map&&this.video.play(),this._finishLoading())}))},this.prepare=()=>{if(0===Object.keys(this.tiles).length||this.video.readyState<2)return;const t=this.map.painter.context,e=t.gl;this.boundsBuffer||(this.boundsBuffer=t.createVertexBuffer(this._boundsArray,Hh.members)),this.boundsSegments||(this.boundsSegments=Nn.simpleSegment(0,0,4,2)),this.texture?this.video.paused||(this.texture.bind(e.LINEAR,e.CLAMP_TO_EDGE),e.texSubImage2D(e.TEXTURE_2D,0,0,0,e.RGBA,e.UNSIGNED_BYTE,this.video)):(this.texture=new uh(t,this.video,e.RGBA),this.texture.bind(e.LINEAR,e.CLAMP_TO_EDGE));let i=!1;for(const t in this.tiles){const e=this.tiles[t];"loaded"!==e.state&&(e.state="loaded",e.texture=this.texture,i=!0)}i&&this.fire(new nt("data",{dataType:"source",sourceDataType:"idle",sourceId:this.id}))},this.serialize=()=>({type:"video",urls:this.urls,coordinates:this.coordinates}),this.roundZoom=!0,this.type="video",this.options=e}pause(){this.video&&this.video.pause()}play(){this.video&&this.video.play()}seek(t){if(this.video){const e=this.video.seekable;te.end(0)?this.fire(new at(new bt(`sources.${this.id}`,null,`Playback for this video can be set only between the ${e.start(0)} and ${e.end(0)}-second mark.`))):this.video.currentTime=t}}getVideo(){return this.video}onAdd(t){this.map||(this.map=t,this.load(),this.video&&(this.video.play(),this.setCoordinates(this.coordinates)))}hasTransition(){return this.video&&!this.video.paused}}class Kh extends Wh{constructor(t,e,i,r){super(t,e,i,r),this.load=()=>{this._loaded=!0,this.canvas||(this.canvas=this.options.canvas instanceof HTMLCanvasElement?this.options.canvas:document.getElementById(this.options.canvas)),this.width=this.canvas.width,this.height=this.canvas.height,this._hasInvalidDimensions()?this.fire(new at(new Error("Canvas dimensions cannot be less than or equal to zero."))):(this.play=function(){this._playing=!0,this.map.triggerRepaint()},this.pause=function(){this._playing&&(this.prepare(),this._playing=!1)},this._finishLoading())},this.prepare=()=>{let t=!1;if(this.canvas.width!==this.width&&(this.width=this.canvas.width,t=!0),this.canvas.height!==this.height&&(this.height=this.canvas.height,t=!0),this._hasInvalidDimensions())return;if(0===Object.keys(this.tiles).length)return;const e=this.map.painter.context,i=e.gl;this.boundsBuffer||(this.boundsBuffer=e.createVertexBuffer(this._boundsArray,Hh.members)),this.boundsSegments||(this.boundsSegments=Nn.simpleSegment(0,0,4,2)),this.texture?(t||this._playing)&&this.texture.update(this.canvas,{premultiply:!0}):this.texture=new uh(e,this.canvas,i.RGBA,{premultiply:!0});let r=!1;for(const t in this.tiles){const e=this.tiles[t];"loaded"!==e.state&&(e.state="loaded",e.texture=this.texture,r=!0)}r&&this.fire(new nt("data",{dataType:"source",sourceDataType:"idle",sourceId:this.id}))},this.serialize=()=>({type:"canvas",coordinates:this.coordinates}),e.coordinates?Array.isArray(e.coordinates)&&4===e.coordinates.length&&!e.coordinates.some((t=>!Array.isArray(t)||2!==t.length||t.some((t=>"number"!=typeof t))))||this.fire(new at(new bt(`sources.${t}`,null,'"coordinates" property must be an array of 4 longitude/latitude array pairs'))):this.fire(new at(new bt(`sources.${t}`,null,'missing required property "coordinates"'))),e.animate&&"boolean"!=typeof e.animate&&this.fire(new at(new bt(`sources.${t}`,null,'optional "animate" property must be a boolean value'))),e.canvas?"string"==typeof e.canvas||e.canvas instanceof HTMLCanvasElement||this.fire(new at(new bt(`sources.${t}`,null,'"canvas" must be either a string representing the ID of the canvas element from which to read, or an HTMLCanvasElement instance'))):this.fire(new at(new bt(`sources.${t}`,null,'missing required property "canvas"'))),this.options=e,this.animate=void 0===e.animate||e.animate}getCanvas(){return this.canvas}onAdd(t){this.map=t,this.load(),this.canvas&&this.animate&&this.play()}onRemove(){this.pause()}hasTransition(){return this._playing}_hasInvalidDimensions(){for(const t of[this.canvas.width,this.canvas.height])if(isNaN(t)||t<=0)return!0;return!1}}const Yh={},Jh=t=>{switch(t){case"geojson":return Gh;case"image":return Wh;case"raster":return Vh;case"raster-dem":return jh;case"vector":return Oh;case"video":return Xh;case"canvas":return Kh}return Yh[t]};function Qh(t,e){const i=Ga();return Ka(i,i,[1,1,0]),Ya(i,i,[.5*t.width,.5*t.height,1]),Xa(i,i,t.calculatePosMatrix(e.toUnwrapped()))}function tu(t,e,i,r,s,n){const a=function(t,e,i){if(t)for(const r of t){const t=e[r];if(t&&t.source===i&&"fill-extrusion"===t.type)return!0}else for(const t in e){const r=e[t];if(r.source===i&&"fill-extrusion"===r.type)return!0}return!1}(s&&s.layers,e,t.id),o=n.maxPitchScaleFactor(),l=t.tilesIn(r,o,a);l.sort(eu);const c=[];for(const r of l)c.push({wrappedTileID:r.tileID.wrapped().key,queryResults:r.tile.queryRenderedFeatures(e,i,t._state,r.queryGeometry,r.cameraQueryGeometry,r.scale,s,n,o,Qh(t.transform,r.tileID))});const h=function(t){const e={},i={};for(const r of t){const t=r.queryResults,s=r.wrappedTileID,n=i[s]=i[s]||{};for(const i in t){const r=t[i],s=n[i]=n[i]||{},a=e[i]=e[i]||[];for(const t of r)s[t.featureIndex]||(s[t.featureIndex]=!0,a.push(t))}}return e}(c);for(const e in h)h[e].forEach((e=>{const i=e.feature,r=t.getFeatureState(i.layer["source-layer"],i.id);i.source=i.layer.source,i.layer["source-layer"]&&(i.sourceLayer=i.layer["source-layer"]),i.state=r}));return h}function eu(t,e){const i=t.tileID,r=e.tileID;return i.overscaledZ-r.overscaledZ||i.canonical.y-r.canonical.y||i.wrap-r.wrap||i.canonical.x-r.canonical.x}class iu{constructor(t){this._stringToNumber={},this._numberToString=[];for(let e=0;e=this._numberToString.length)throw new Error(`Out of bounds. Index requested n=${t} can't be >= this._numberToString.length ${this._numberToString.length}`);return this._numberToString[t]}}class ru{constructor(t,e,i,r,s){this.type="Feature",this._vectorTileFeature=t,t._z=e,t._x=i,t._y=r,this.properties=t.properties,this.id=s}get geometry(){return void 0===this._geometry&&(this._geometry=this._vectorTileFeature.toGeoJSON(this._vectorTileFeature._x,this._vectorTileFeature._y,this._vectorTileFeature._z).geometry),this._geometry}set geometry(t){this._geometry=t}toJSON(){const t={geometry:this.geometry};for(const e in this)"_geometry"!==e&&"_vectorTileFeature"!==e&&(t[e]=this[e]);return t}}function su(t,e,i,r,s){return b(t,((t,n)=>{const a=e instanceof Ss?e.get(n):null;return a&&a.evaluate?a.evaluate(i,r,s):a}))}function nu(t){let e=1/0,i=1/0,r=-1/0,s=-1/0;for(const n of t)e=Math.min(e,n.x),i=Math.min(i,n.y),r=Math.max(r,n.x),s=Math.max(s,n.y);return{minX:e,minY:i,maxX:r,maxY:s}}function au(t,e){return e-t}Hr("FeatureIndex",class{constructor(t,e){this.tileID=t,this.x=t.canonical.x,this.y=t.canonical.y,this.z=t.canonical.z,this.grid=new jr(va,16,0),this.grid3D=new jr(va,16,0),this.featureIndexArray=new xn,this.promoteId=e}insert(t,e,i,r,s,n){const a=this.featureIndexArray.length;this.featureIndexArray.emplaceBack(i,r,s);const o=n?this.grid3D:this.grid;for(let t=0;t=0&&r[3]>=0&&o.insert(a,r[0],r[1],r[2],r[3])}}loadVTLayers(){return this.vtLayers||(this.vtLayers=new ml.VectorTile(new Ec(this.rawTileData)).layers,this.sourceLayerCoder=new iu(this.vtLayers?Object.keys(this.vtLayers).sort():["_geojsonTileLayer"])),this.vtLayers}query(t,e,i,r){this.loadVTLayers();const s=t.params||{},a=va/t.tileSize/t.scale,o=sr(s.filter),l=t.queryGeometry,c=t.queryPadding*a,h=nu(l),u=this.grid.query(h.minX-c,h.minY-c,h.maxX+c,h.maxY+c),p=nu(t.cameraQueryGeometry),d=this.grid3D.query(p.minX-c,p.minY-c,p.maxX+c,p.maxY+c,((e,i,r,s)=>function(t,e,i,r,s){for(const n of t)if(e<=n.x&&i<=n.y&&r>=n.x&&s>=n.y)return!0;const a=[new n(e,i),new n(e,s),new n(r,s),new n(r,i)];if(t.length>2)for(const e of a)if(Ba(t,e))return!0;for(let e=0;e(p||(p=Ta(e)),i.queryIntersectsFeature(l,e,r,p,this.z,t.transform,a,t.pixelPosMatrix))))}return m}loadMatchingFeature(t,e,i,r,s,n,a,o,l,c,h){const u=this.bucketLayerIDs[e];if(n&&!function(t,e){for(let i=0;i=0)return!0;return!1}(n,u))return;const p=this.sourceLayerCoder.decode(i),d=this.vtLayers[p].feature(r);if(s.needGeometry){const t=Sa(d,!0);if(!s.filter(new _s(this.tileID.overscaledZ),t,this.tileID.canonical))return}else if(!s.filter(new _s(this.tileID.overscaledZ),d))return;const m=this.getId(d,p);for(let e=0;ee.getLayer(t))).filter(Boolean);if(0!==t.length){r.layers=t,r.stateDependentLayerIds&&(r.stateDependentLayers=r.stateDependentLayerIds.map((e=>t.filter((t=>t.id===e))[0])));for(const e of t)i[e.id]=r}}return i}(t.buckets,e.style),this.hasSymbolBuckets=!1;for(const t in this.buckets){const e=this.buckets[t];if(e instanceof Xc){if(this.hasSymbolBuckets=!0,!i)break;e.justReloaded=!0}}if(this.hasRTLText=!1,this.hasSymbolBuckets)for(const t in this.buckets){const e=this.buckets[t];if(e instanceof Xc&&e.hasRTLText){this.hasRTLText=!0,fs.isLoading()||fs.isLoaded()||"deferred"!==ds()||ms();break}}this.queryPadding=0;for(const t in this.buckets){const i=this.buckets[t];this.queryPadding=Math.max(this.queryPadding,e.style.getLayer(t).queryRadius(i))}t.imageAtlas&&(this.imageAtlas=t.imageAtlas),t.glyphAtlasImage&&(this.glyphAtlasImage=t.glyphAtlasImage)}else this.collisionBoxArray=new cn}unloadVectorData(){for(const t in this.buckets)this.buckets[t].destroy();this.buckets={},this.imageAtlasTexture&&this.imageAtlasTexture.destroy(),this.imageAtlas&&(this.imageAtlas=null),this.glyphAtlasTexture&&this.glyphAtlasTexture.destroy(),this.latestFeatureIndex=null,this.state="unloaded"}getBucket(t){return this.buckets[t.id]}upload(t){for(const e in this.buckets){const i=this.buckets[e];i.uploadPending()&&i.upload(t)}const e=t.gl;this.imageAtlas&&!this.imageAtlas.uploaded&&(this.imageAtlasTexture=new uh(t,this.imageAtlas.image,e.RGBA),this.imageAtlas.uploaded=!0),this.glyphAtlasImage&&(this.glyphAtlasTexture=new uh(t,this.glyphAtlasImage,e.ALPHA),this.glyphAtlasImage=null)}prepare(t){this.imageAtlas&&this.imageAtlas.patchUpdatedImages(t,this.imageAtlasTexture)}queryRenderedFeatures(t,e,i,r,s,n,a,o,l,c){return this.latestFeatureIndex&&this.latestFeatureIndex.rawTileData?this.latestFeatureIndex.query({queryGeometry:r,cameraQueryGeometry:s,scale:n,tileSize:this.tileSize,pixelPosMatrix:c,transform:o,params:a,queryPadding:this.queryPadding*l},t,e,i):{}}querySourceFeatures(t,e){const i=this.latestFeatureIndex;if(!i||!i.rawTileData)return;const r=i.loadVTLayers(),s=e&&e.sourceLayer?e.sourceLayer:"",n=r._geojsonTileLayer||r[s];if(!n)return;const a=sr(e&&e.filter),{z:o,x:l,y:c}=this.tileID.canonical,h={z:o,x:l,y:c};for(let e=0;e@\,;\:\\"\/\[\]\?\=\{\}\x7F]+)(?:\=(?:([^\x00-\x20\(\)<>@\,;\:\\"\/\[\]\?\=\{\}\x7F]+)|(?:\"((?:[^"\\]|\\.)*)\")))?/g,((t,i,r,s)=>{const n=r||s;return e[i]=!n||n.toLowerCase(),""})),e["max-age"]){const t=parseInt(e["max-age"],10);isNaN(t)?delete e["max-age"]:e["max-age"]=t}return e}(t.cacheControl);e["max-age"]&&(this.expirationTime=Date.now()+1e3*e["max-age"])}else t.expires&&(this.expirationTime=new Date(t.expires).getTime());if(this.expirationTime){const t=Date.now();let i=!1;if(this.expirationTime>t)i=!1;else if(e)if(this.expirationTime{this.remove(t,s)}),i)),this.data[r].push(s),this.order.push(r),this.order.length>this.max){const t=this._getAndRemoveByKey(this.order[0]);t&&this.onRemove(t)}return this}has(t){return t.wrapped().key in this.data}getAndRemove(t){return this.has(t)?this._getAndRemoveByKey(t.wrapped().key):null}_getAndRemoveByKey(t){const e=this.data[t].shift();return e.timeout&&clearTimeout(e.timeout),0===this.data[t].length&&delete this.data[t],this.order.splice(this.order.indexOf(t),1),e.value}getByKey(t){const e=this.data[t];return e?e[0].value:null}get(t){return this.has(t)?this.data[t.wrapped().key][0].value:null}remove(t,e){if(!this.has(t))return this;const i=t.wrapped().key,r=void 0===e?0:this.data[i].indexOf(e),s=this.data[i][r];return this.data[i].splice(r,1),s.timeout&&clearTimeout(s.timeout),0===this.data[i].length&&delete this.data[i],this.onRemove(s.value),this.order.splice(this.order.indexOf(i),1),this}setMaxSize(t){for(this.max=t;this.order.length>this.max;){const t=this._getAndRemoveByKey(this.order[0]);t&&this.onRemove(t)}return this}filter(t){const e=[];for(const i in this.data)for(const r of this.data[i])t(r.value)||e.push(r);for(const t of e)this.remove(t.value.tileID,t)}}class cu{constructor(){this.state={},this.stateChanges={},this.deletedStates={}}updateState(t,e,i){const r=String(e);if(this.stateChanges[t]=this.stateChanges[t]||{},this.stateChanges[t][r]=this.stateChanges[t][r]||{},g(this.stateChanges[t][r],i),null===this.deletedStates[t]){this.deletedStates[t]={};for(const e in this.state[t])e!==r&&(this.deletedStates[t][e]=null)}else if(this.deletedStates[t]&&null===this.deletedStates[t][r]){this.deletedStates[t][r]={};for(const e in this.state[t][r])i[e]||(this.deletedStates[t][r][e]=null)}else for(const e in i)this.deletedStates[t]&&this.deletedStates[t][r]&&null===this.deletedStates[t][r][e]&&delete this.deletedStates[t][r][e]}removeFeatureState(t,e,i){if(null===this.deletedStates[t])return;const r=String(e);if(this.deletedStates[t]=this.deletedStates[t]||{},i&&void 0!==e)null!==this.deletedStates[t][r]&&(this.deletedStates[t][r]=this.deletedStates[t][r]||{},this.deletedStates[t][r][i]=null);else if(void 0!==e)if(this.stateChanges[t]&&this.stateChanges[t][r])for(i in this.deletedStates[t][r]={},this.stateChanges[t][r])this.deletedStates[t][r][i]=null;else this.deletedStates[t][r]=null;else this.deletedStates[t]=null}getState(t,e){const i=String(e),r=g({},(this.state[t]||{})[i],(this.stateChanges[t]||{})[i]);if(null===this.deletedStates[t])return{};if(this.deletedStates[t]){const i=this.deletedStates[t][e];if(null===i)return{};for(const t in i)delete r[t]}return r}initializeTileState(t,e){t.setFeatureState(this.state,e)}coalesceChanges(t,e){const i={};for(const t in this.stateChanges){this.state[t]=this.state[t]||{};const e={};for(const i in this.stateChanges[t])this.state[t][i]||(this.state[t][i]={}),g(this.state[t][i],this.stateChanges[t][i]),e[i]=this.state[t][i];i[t]=e}for(const t in this.deletedStates){this.state[t]=this.state[t]||{};const e={};if(null===this.deletedStates[t])for(const i in this.state[t])e[i]={},this.state[t][i]={};else for(const i in this.deletedStates[t]){if(null===this.deletedStates[t][i])this.state[t][i]={};else for(const e of Object.keys(this.deletedStates[t][i]))delete this.state[t][i][e];e[i]=this.state[t][i]}i[t]=i[t]||{},g(i[t],e)}if(this.stateChanges={},this.deletedStates={},0!==Object.keys(i).length)for(const r in t)t[r].setFeatureState(i,e)}}class hu extends ot{constructor(t,e,i){super(),this.id=t,this.dispatcher=i,this.on("data",(t=>{"source"===t.dataType&&"metadata"===t.sourceDataType&&(this._sourceLoaded=!0),this._sourceLoaded&&!this._paused&&"source"===t.dataType&&"content"===t.sourceDataType&&(this.reload(),this.transform&&this.update(this.transform,this.terrain),this._didEmitContent=!0)})),this.on("dataloading",(()=>{this._sourceErrored=!1})),this.on("error",(()=>{this._sourceErrored=this._source.loaded()})),this._source=((t,e,i,r)=>{const s=new(Jh(e.type))(t,e,i,r);if(s.id!==t)throw new Error(`Expected Source id to be ${t} instead of ${s.id}`);return s})(t,e,i,this),this._tiles={},this._cache=new lu(0,this._unloadTile.bind(this)),this._timers={},this._cacheTimers={},this._maxTileCacheSize=null,this._maxTileCacheZoomLevels=null,this._loadedParentTiles={},this._coveredTiles={},this._state=new cu,this._didEmitContent=!1,this._updated=!1}onAdd(t){this.map=t,this._maxTileCacheSize=t?t._maxTileCacheSize:null,this._maxTileCacheZoomLevels=t?t._maxTileCacheZoomLevels:null,this._source&&this._source.onAdd&&this._source.onAdd(t)}onRemove(t){this.clearTiles(),this._source&&this._source.onRemove&&this._source.onRemove(t)}loaded(){if(this._sourceErrored)return!0;if(!this._sourceLoaded)return!1;if(!this._source.loaded())return!1;if(!(void 0===this.used&&void 0===this.usedForTerrain||this.used||this.usedForTerrain))return!0;if(!this._updated)return!1;for(const t in this._tiles){const e=this._tiles[t];if("loaded"!==e.state&&"errored"!==e.state)return!1}return!0}getSource(){return this._source}pause(){this._paused=!0}resume(){if(!this._paused)return;const t=this._shouldReloadOnResume;this._paused=!1,this._shouldReloadOnResume=!1,t&&this.reload(),this.transform&&this.update(this.transform,this.terrain)}_loadTile(t,e){return this._source.loadTile(t,e)}_unloadTile(t){if(this._source.unloadTile)return this._source.unloadTile(t,(()=>{}))}_abortTile(t){this._source.abortTile&&this._source.abortTile(t,(()=>{})),this._source.fire(new nt("dataabort",{tile:t,coord:t.tileID,dataType:"source"}))}serialize(){return this._source.serialize()}prepare(t){this._source.prepare&&this._source.prepare(),this._state.coalesceChanges(this._tiles,this.map?this.map.painter:null);for(const e in this._tiles){const i=this._tiles[e];i.upload(t),i.prepare(this.map.style.imageManager)}}getIds(){return Object.values(this._tiles).map((t=>t.tileID)).sort(uu).map((t=>t.key))}getRenderableIds(t){const e=[];for(const i in this._tiles)this._isIdRenderable(i,t)&&e.push(this._tiles[i]);return t?e.sort(((t,e)=>{const i=t.tileID,r=e.tileID,s=new n(i.canonical.x,i.canonical.y)._rotate(this.transform.angle),a=new n(r.canonical.x,r.canonical.y)._rotate(this.transform.angle);return i.overscaledZ-r.overscaledZ||a.y-s.y||a.x-s.x})).map((t=>t.tileID.key)):e.map((t=>t.tileID)).sort(uu).map((t=>t.key))}hasRenderableParent(t){const e=this.findLoadedParent(t,0);return!!e&&this._isIdRenderable(e.tileID.key)}_isIdRenderable(t,e){return this._tiles[t]&&this._tiles[t].hasData()&&!this._coveredTiles[t]&&(e||!this._tiles[t].holdingForFade())}reload(){if(this._paused)this._shouldReloadOnResume=!0;else{this._cache.reset();for(const t in this._tiles)"errored"!==this._tiles[t].state&&this._reloadTile(t,"reloading")}}_reloadTile(t,e){const i=this._tiles[t];i&&("loading"!==i.state&&(i.state=e),this._loadTile(i,this._tileLoaded.bind(this,i,t,e)))}_tileLoaded(t,e,i,r){if(r)return t.state="errored",void(404!==r.status?this._source.fire(new at(r,{tile:t})):this.update(this.transform,this.terrain));t.timeAdded=B.now(),"expired"===i&&(t.refreshedUponExpiration=!0),this._setTileReloadTimer(e,t),"raster-dem"===this.getSource().type&&t.dem&&this._backfillDEM(t),this._state.initializeTileState(t,this.map?this.map.painter:null),t.aborted||this._source.fire(new nt("data",{dataType:"source",tile:t,coord:t.tileID}))}_backfillDEM(t){const e=this.getRenderableIds();for(let r=0;r1||(Math.abs(i)>1&&(1===Math.abs(i+s)?i+=s:1===Math.abs(i-s)&&(i-=s)),e.dem&&t.dem&&(t.dem.backfillBorder(e.dem,i,r),t.neighboringTiles&&t.neighboringTiles[n]&&(t.neighboringTiles[n].backfilled=!0)))}}getTile(t){return this.getTileByID(t.key)}getTileByID(t){return this._tiles[t]}_retainLoadedChildren(t,e,i,r){for(const s in this._tiles){let n=this._tiles[s];if(r[s]||!n.hasData()||n.tileID.overscaledZ<=e||n.tileID.overscaledZ>i)continue;let a=n.tileID;for(;n&&n.tileID.overscaledZ>e+1;){const t=n.tileID.scaledTo(n.tileID.overscaledZ-1);n=this._tiles[t.key],n&&n.hasData()&&(a=t)}let o=a;for(;o.overscaledZ>e;)if(o=o.scaledTo(o.overscaledZ-1),t[o.key]){r[a.key]=a;break}}}findLoadedParent(t,e){if(t.key in this._loadedParentTiles){const i=this._loadedParentTiles[t.key];return i&&i.tileID.overscaledZ>=e?i:null}for(let i=t.overscaledZ-1;i>=e;i--){const e=t.scaledTo(i),r=this._getLoadedTile(e);if(r)return r}}_getLoadedTile(t){const e=this._tiles[t.key];return e&&e.hasData()?e:this._cache.getByKey(t.wrapped().key)}updateCacheSize(t){const e=Math.ceil(t.width/this._source.tileSize)+1,i=Math.ceil(t.height/this._source.tileSize)+1,r=Math.floor(e*i*(null===this._maxTileCacheZoomLevels?O.MAX_TILE_CACHE_ZOOM_LEVELS:this._maxTileCacheZoomLevels)),s="number"==typeof this._maxTileCacheSize?Math.min(this._maxTileCacheSize,r):r;this._cache.setMaxSize(s)}handleWrapJump(t){const e=Math.round((t-(void 0===this._prevLng?t:this._prevLng))/360);if(this._prevLng=t,e){const t={};for(const i in this._tiles){const r=this._tiles[i];r.tileID=r.tileID.unwrapTo(r.tileID.wrap+e),t[r.tileID.key]=r}this._tiles=t;for(const t in this._timers)clearTimeout(this._timers[t]),delete this._timers[t];for(const t in this._tiles)this._setTileReloadTimer(t,this._tiles[t])}}update(t,e){if(this.transform=t,this.terrain=e,!this._sourceLoaded||this._paused)return;let i;this.updateCacheSize(t),this.handleWrapJump(this.transform.center.lng),this._coveredTiles={},this.used||this.usedForTerrain?this._source.tileID?i=t.getVisibleUnwrappedCoordinates(this._source.tileID).map((t=>new qh(t.canonical.z,t.wrap,t.canonical.z,t.canonical.x,t.canonical.y))):(i=t.coveringTiles({tileSize:this.usedForTerrain?this.tileSize:this._source.tileSize,minzoom:this._source.minzoom,maxzoom:this._source.maxzoom,roundZoom:!this.usedForTerrain&&this._source.roundZoom,reparseOverscaled:this._source.reparseOverscaled,terrain:e}),this._source.hasTile&&(i=i.filter((t=>this._source.hasTile(t))))):i=[];const r=t.coveringZoomLevel(this._source),s=Math.max(r-hu.maxOverzooming,this._source.minzoom),n=Math.max(r+hu.maxUnderzooming,this._source.minzoom);if(this.usedForTerrain){const t={};for(const e of i)if(e.canonical.z>this._source.minzoom){const i=e.scaledTo(e.canonical.z-1);t[i.key]=i;const r=e.scaledTo(Math.max(this._source.minzoom,Math.min(e.canonical.z,5)));t[r.key]=r}i=i.concat(Object.values(t))}const a=0===i.length&&!this._updated&&this._didEmitContent;this._updated=!0,a&&this.fire(new nt("data",{sourceDataType:"idle",dataType:"source",sourceId:this.id}));const o=this._updateRetainedTiles(i,r);if(pu(this._source.type)){const t={},a={},l=Object.keys(o),c=B.now();for(const e of l){const i=o[e],r=this._tiles[e];if(!r||0!==r.fadeEndTime&&r.fadeEndTime<=c)continue;const n=this.findLoadedParent(i,s);n&&(this._addTile(n.tileID),t[n.tileID.key]=n.tileID),a[e]=i}this._retainLoadedChildren(a,r,n,o);for(const e in t)o[e]||(this._coveredTiles[e]=!0,o[e]=t[e]);if(e){const t={},e={};for(const r of i)this._tiles[r.key].hasData()?t[r.key]=r:e[r.key]=r;for(const i in e){const r=e[i].children(this._source.maxzoom);this._tiles[r[0].key]&&this._tiles[r[1].key]&&this._tiles[r[2].key]&&this._tiles[r[3].key]&&(t[r[0].key]=o[r[0].key]=r[0],t[r[1].key]=o[r[1].key]=r[1],t[r[2].key]=o[r[2].key]=r[2],t[r[3].key]=o[r[3].key]=r[3],delete e[i])}for(const i in e){const r=this.findLoadedParent(e[i],this._source.minzoom);if(r){t[r.tileID.key]=o[r.tileID.key]=r.tileID;for(const e in t)t[e].isChildOf(r.tileID)&&delete t[e]}}for(const e in this._tiles)t[e]||(this._coveredTiles[e]=!0)}}for(const t in o)this._tiles[t].clearFadeHold();const l=function(t,e){const i=[];for(const r in t)r in e||i.push(r);return i}(this._tiles,o);for(const t of l){const e=this._tiles[t];e.hasSymbolBuckets&&!e.holdingForFade()?e.setHoldDuration(this.map._fadeDuration):e.hasSymbolBuckets&&!e.symbolFadeFinished()||this._removeTile(t)}this._updateLoadedParentTileCache()}releaseSymbolFadeTiles(){for(const t in this._tiles)this._tiles[t].holdingForFade()&&this._removeTile(t)}_updateRetainedTiles(t,e){const i={},r={},s=Math.max(e-hu.maxOverzooming,this._source.minzoom),n=Math.max(e+hu.maxUnderzooming,this._source.minzoom),a={};for(const r of t){const t=this._addTile(r);i[r.key]=r,t.hasData()||ethis._source.maxzoom){const t=n.children(this._source.maxzoom)[0],e=this.getTile(t);if(e&&e.hasData()){i[t.key]=t;continue}}else{const t=n.children(this._source.maxzoom);if(i[t[0].key]&&i[t[1].key]&&i[t[2].key]&&i[t[3].key])continue}let a=t.wasRequested();for(let e=n.overscaledZ-1;e>=s;--e){const s=n.scaledTo(e);if(r[s.key])break;if(r[s.key]=!0,t=this.getTile(s),!t&&a&&(t=this._addTile(s)),t){const e=t.hasData();if((a||e)&&(i[s.key]=s),a=t.wasRequested(),e)break}}}return i}_updateLoadedParentTileCache(){this._loadedParentTiles={};for(const t in this._tiles){const e=[];let i,r=this._tiles[t].tileID;for(;r.overscaledZ>0;){if(r.key in this._loadedParentTiles){i=this._loadedParentTiles[r.key];break}e.push(r.key);const t=r.scaledTo(r.overscaledZ-1);if(i=this._getLoadedTile(t),i)break;r=t}for(const t of e)this._loadedParentTiles[t]=i}}_addTile(t){let e=this._tiles[t.key];if(e)return e;e=this._cache.getAndRemove(t),e&&(this._setTileReloadTimer(t.key,e),e.tileID=t,this._state.initializeTileState(e,this.map?this.map.painter:null),this._cacheTimers[t.key]&&(clearTimeout(this._cacheTimers[t.key]),delete this._cacheTimers[t.key],this._setTileReloadTimer(t.key,e)));const i=e;return e||(e=new ou(t,this._source.tileSize*t.overscaleFactor()),this._loadTile(e,this._tileLoaded.bind(this,e,t.key,e.state))),e.uses++,this._tiles[t.key]=e,i||this._source.fire(new nt("dataloading",{tile:e,coord:e.tileID,dataType:"source"})),e}_setTileReloadTimer(t,e){t in this._timers&&(clearTimeout(this._timers[t]),delete this._timers[t]);const i=e.getExpiryTimeout();i&&(this._timers[t]=setTimeout((()=>{this._reloadTile(t,"expired"),delete this._timers[t]}),i))}_removeTile(t){const e=this._tiles[t];e&&(e.uses--,delete this._tiles[t],this._timers[t]&&(clearTimeout(this._timers[t]),delete this._timers[t]),e.uses>0||(e.hasData()&&"reloading"!==e.state?this._cache.add(e.tileID,e,e.getExpiryTimeout()):(e.aborted=!0,this._abortTile(e),this._unloadTile(e))))}clearTiles(){this._shouldReloadOnResume=!1,this._paused=!1;for(const t in this._tiles)this._removeTile(t);this._cache.reset()}tilesIn(t,e,i){const r=[],s=this.transform;if(!s)return r;const n=i?s.getCameraQueryGeometry(t):t,a=t.map((t=>s.pointCoordinate(t,this.terrain))),o=n.map((t=>s.pointCoordinate(t,this.terrain))),l=this.getIds();let c=1/0,h=1/0,u=-1/0,p=-1/0;for(const t of o)c=Math.min(c,t.x),h=Math.min(h,t.y),u=Math.max(u,t.x),p=Math.max(p,t.y);for(let t=0;t=0&&f[1].y+m>=0){const t=a.map((t=>n.getTilePoint(t))),e=o.map((t=>n.getTilePoint(t)));r.push({tile:i,tileID:n,queryGeometry:t,cameraQueryGeometry:e,scale:d})}}return r}getVisibleCoordinates(t){const e=this.getRenderableIds(t).map((t=>this._tiles[t].tileID));for(const t of e)t.posMatrix=this.transform.calculatePosMatrix(t.toUnwrapped());return e}hasTransition(){if(this._source.hasTransition())return!0;if(pu(this._source.type)){const t=B.now();for(const e in this._tiles)if(this._tiles[e].fadeEndTime>=t)return!0}return!1}setFeatureState(t,e,i){this._state.updateState(t=t||"_geojsonTileLayer",e,i)}removeFeatureState(t,e,i){this._state.removeFeatureState(t=t||"_geojsonTileLayer",e,i)}getFeatureState(t,e){return this._state.getState(t=t||"_geojsonTileLayer",e)}setDependencies(t,e,i){const r=this._tiles[t];r&&r.setDependencies(e,i)}reloadTilesForDependencies(t,e){for(const i in this._tiles)this._tiles[i].hasDependency(t,e)&&this._reloadTile(i,"reloading");this._cache.filter((i=>!i.hasDependency(t,e)))}}function uu(t,e){const i=Math.abs(2*t.wrap)-+(t.wrap<0),r=Math.abs(2*e.wrap)-+(e.wrap<0);return t.overscaledZ-e.overscaledZ||r-i||e.canonical.y-t.canonical.y||e.canonical.x-t.canonical.x}function pu(t){return"raster"===t||"image"===t||"video"===t}hu.maxOverzooming=10,hu.maxUnderzooming=3;const du="mapboxgl_preloaded_worker_pool";class mu{constructor(){this.active={}}acquire(t){if(!this.workers)for(this.workers=[];this.workers.length{t.terminate()})),this.workers=null)}isPreloaded(){return!!this.active[du]}numActive(){return Object.keys(this.active).length}}const fu=Math.floor(B.hardwareConcurrency/2);let _u;function gu(){return _u||(_u=new mu),_u}mu.workerCount=P(globalThis)?Math.max(Math.min(fu,3),1):1;class yu{constructor(t,e){this.reset(t,e)}reset(t,e){this.points=t||[],this._distances=[0];for(let t=1;t0?(r-n)/a:0;return this.points[s].mult(1-o).add(this.points[e].mult(o))}}function xu(t,e){let i=!0;return"always"===t||"never"!==t&&"never"!==e||(i=!1),i}class vu{constructor(t,e,i){const r=this.boxCells=[],s=this.circleCells=[];this.xCellCount=Math.ceil(t/i),this.yCellCount=Math.ceil(e/i);for(let t=0;tthis.width||r<0||e>this.height)return[];const o=[];if(t<=0&&e<=0&&this.width<=i&&this.height<=r){if(s)return[{key:null,x1:t,y1:e,x2:i,y2:r}];for(let t=0;t0}hitTestCircle(t,e,i,r,s){const n=t-i,a=t+i,o=e-i,l=e+i;if(a<0||n>this.width||l<0||o>this.height)return!1;const c=[];return this._forEachCell(n,o,a,l,this._queryCellCircle,c,{hitTest:!0,overlapMode:r,circle:{x:t,y:e,radius:i},seenUids:{box:{},circle:{}}},s),c.length>0}_queryCell(t,e,i,r,s,n,a,o){const{seenUids:l,hitTest:c,overlapMode:h}=a,u=this.boxCells[s];if(null!==u){const s=this.bboxes;for(const a of u)if(!l.box[a]){l.box[a]=!0;const u=4*a,p=this.boxKeys[a];if(t<=s[u+2]&&e<=s[u+3]&&i>=s[u+0]&&r>=s[u+1]&&(!o||o(p))&&(!c||!xu(h,p.overlapMode))&&(n.push({key:p,x1:s[u],y1:s[u+1],x2:s[u+2],y2:s[u+3]}),c))return!0}}const p=this.circleCells[s];if(null!==p){const s=this.circles;for(const a of p)if(!l.circle[a]){l.circle[a]=!0;const u=3*a,p=this.circleKeys[a];if(this._circleAndRectCollide(s[u],s[u+1],s[u+2],t,e,i,r)&&(!o||o(p))&&(!c||!xu(h,p.overlapMode))){const t=s[u],e=s[u+1],i=s[u+2];if(n.push({key:p,x1:t-i,y1:e-i,x2:t+i,y2:e+i}),c)return!0}}}return!1}_queryCellCircle(t,e,i,r,s,n,a,o){const{circle:l,seenUids:c,overlapMode:h}=a,u=this.boxCells[s];if(null!==u){const t=this.bboxes;for(const e of u)if(!c.box[e]){c.box[e]=!0;const i=4*e,r=this.boxKeys[e];if(this._circleAndRectCollide(l.x,l.y,l.radius,t[i+0],t[i+1],t[i+2],t[i+3])&&(!o||o(r))&&!xu(h,r.overlapMode))return n.push(!0),!0}}const p=this.circleCells[s];if(null!==p){const t=this.circles;for(const e of p)if(!c.circle[e]){c.circle[e]=!0;const i=3*e,r=this.circleKeys[e];if(this._circlesCollide(t[i],t[i+1],t[i+2],l.x,l.y,l.radius)&&(!o||o(r))&&!xu(h,r.overlapMode))return n.push(!0),!0}}}_forEachCell(t,e,i,r,s,n,a,o){const l=this._convertToXCellCoord(t),c=this._convertToYCellCoord(e),h=this._convertToXCellCoord(i),u=this._convertToYCellCoord(r);for(let p=l;p<=h;p++)for(let l=c;l<=u;l++)if(s.call(this,t,e,i,r,this.xCellCount*l+p,n,a,o))return}_convertToXCellCoord(t){return Math.max(0,Math.min(this.xCellCount-1,Math.floor(t*this.xScale)))}_convertToYCellCoord(t){return Math.max(0,Math.min(this.yCellCount-1,Math.floor(t*this.yScale)))}_circlesCollide(t,e,i,r,s,n){const a=r-t,o=s-e,l=i+n;return l*l>a*a+o*o}_circleAndRectCollide(t,e,i,r,s,n,a){const o=(n-r)/2,l=Math.abs(t-(r+o));if(l>o+i)return!1;const c=(a-s)/2,h=Math.abs(e-(s+c));if(h>c+i)return!1;if(l<=o||h<=c)return!0;const u=l-o,p=h-c;return u*u+p*p<=i*i}}function bu(t,e,i,r,s){const n=Ga();return e?(Ya(n,n,[1/s,1/s,1]),i||Ja(n,n,r.angle)):Xa(n,r.labelPlaneMatrix,t),n}function wu(t,e,i,r,s){if(e){const e=function(t){var e=new ja(16);return e[0]=t[0],e[1]=t[1],e[2]=t[2],e[3]=t[3],e[4]=t[4],e[5]=t[5],e[6]=t[6],e[7]=t[7],e[8]=t[8],e[9]=t[9],e[10]=t[10],e[11]=t[11],e[12]=t[12],e[13]=t[13],e[14]=t[14],e[15]=t[15],e}(t);return Ya(e,e,[s,s,1]),i||Ja(e,e,-r.angle),e}return r.glCoordMatrix}function Tu(t,e,i){let r;i?(r=[t.x,t.y,i(t.x,t.y),1],so(r,r,e)):(r=[t.x,t.y,0,1],Fu(r,r,e));const s=r[3];return{point:new n(r[0]/s,r[1]/s),signedDistanceFromCamera:s}}function Su(t,e){return.5+t/e*.5}function Iu(t,e){const i=t[0]/t[3],r=t[1]/t[3];return i>=-e[0]&&i<=e[0]&&r>=-e[1]&&r<=e[1]}function Au(t,e,i,r,s,a,o,l,c,h){const u=r?t.textSizeData:t.iconSizeData,p=Uc(u,i.transform.zoom),d=[256/i.width*2+1,256/i.height*2+1],m=r?t.text.dynamicLayoutVertexArray:t.icon.dynamicLayoutVertexArray;m.clear();const f=t.lineVertexArray,_=r?t.text.placedSymbolArray:t.icon.placedSymbolArray,g=i.transform.width/i.transform.height;let y=!1;for(let r=0;r<_.length;r++){const x=_.get(r);if(x.hidden||x.writingMode===Lc.vertical&&!y){Bu(x.numGlyphs,m);continue}let v;if(y=!1,h?(v=[x.anchorX,x.anchorY,h(x.anchorX,x.anchorY),1],so(v,v,e)):(v=[x.anchorX,x.anchorY,0,1],Fu(v,v,e)),!Iu(v,d)){Bu(x.numGlyphs,m);continue}const b=Su(i.transform.cameraToCenterDistance,v[3]),w=Vc(u,p,x),T=o?w/b:w*b,S=new n(x.anchorX,x.anchorY),I=Tu(S,s,h).point,A={projections:{},offsets:{}},E=zu(x,T,!1,l,e,s,a,t.glyphOffsetArray,f,m,I,S,A,g,c,h);y=E.useVertical,(E.notEnoughRoom||y||E.needsFlipping&&zu(x,T,!0,l,e,s,a,t.glyphOffsetArray,f,m,I,S,A,g,c,h).notEnoughRoom)&&Bu(x.numGlyphs,m)}r?t.text.dynamicLayoutVertexBuffer.updateData(m):t.icon.dynamicLayoutVertexBuffer.updateData(m)}function Eu(t,e,i,r,s,n,a,o,l,c,h,u,p){const d=o.glyphStartIndex+o.numGlyphs,m=o.lineStartIndex,f=o.lineStartIndex+o.lineLength,_=e.getoffsetX(o.glyphStartIndex),g=e.getoffsetX(d-1),y=Lu(t*_,i,r,s,n,a,o.segment,m,f,l,c,h,u,p);if(!y)return null;const x=Lu(t*g,i,r,s,n,a,o.segment,m,f,l,c,h,u,p);return x?{first:y,last:x}:null}function Cu(t,e,i,r){return t===Lc.horizontal&&Math.abs(i.y-e.y)>Math.abs(i.x-e.x)*r?{useVertical:!0}:(t===Lc.vertical?e.yi.x)?{needsFlipping:!0}:null}function zu(t,e,i,r,s,a,o,l,c,h,u,p,d,m,f,_){const g=e/24,y=t.lineOffsetX*g,x=t.lineOffsetY*g;let v;if(t.numGlyphs>1){const e=t.glyphStartIndex+t.numGlyphs,s=t.lineStartIndex,n=t.lineStartIndex+t.lineLength,h=Eu(g,l,y,x,i,u,p,t,c,a,d,f,_);if(!h)return{notEnoughRoom:!0};const b=Tu(h.first.point,o,_).point,w=Tu(h.last.point,o,_).point;if(r&&!i){const e=Cu(t.writingMode,b,w,m);if(e)return e}v=[h.first];for(let r=t.glyphStartIndex+1;r0?a.point:Mu(p,r,e,1,s,_),l=Cu(t.writingMode,e,o,m);if(l)return l}const e=Lu(g*l.getoffsetX(t.glyphStartIndex),y,x,i,u,p,t.segment,t.lineStartIndex,t.lineStartIndex+t.lineLength,c,a,d,f,_);if(!e)return{notEnoughRoom:!0};v=[e]}for(const t of v)jc(h,t.point,t.angle);return{}}function Mu(t,e,i,r,s,n){const a=Tu(t.add(t.sub(e)._unit()),s,n).point,o=i.sub(a);return i.add(o._mult(r/o.mag()))}function Pu(t,e){const{projectionCache:i,lineVertexArray:r,labelPlaneMatrix:s,tileAnchorPoint:a,distanceFromAnchor:o,getElevation:l,previousVertex:c,direction:h,absOffsetX:u}=e;if(i.projections[t])return i.projections[t];const p=new n(r.getx(t),r.gety(t)),d=Tu(p,s,l);if(d.signedDistanceFromCamera>0)return i.projections[t]=d.point,d.point;const m=t-h;return Mu(0===o?a:new n(r.getx(m),r.gety(m)),p,c,u-o+1,s,l)}function ku(t,e,i){return t._unit()._perp()._mult(e*i)}function Du(t,e,i,r,s,a,o,l){const{projectionCache:c,direction:h}=l;if(c.offsets[t])return c.offsets[t];const u=i.add(e);if(t+h=s)return c.offsets[t]=u,u;const p=Pu(t+h,l),d=ku(p.sub(i),o,h),m=i.add(d),f=p.add(d);return c.offsets[t]=function(t,e,i,r){const s=e.y-t.y,a=e.x-t.x,o=r.y-i.y,l=r.x-i.x,c=o*a-l*s;if(0===c)return null;const h=(l*(t.y-i.y)-o*(t.x-i.x))/c;return new n(t.x+h*a,t.y+h*s)}(a,u,m,f)||u,c.offsets[t]}function Lu(t,e,i,r,s,n,a,o,l,c,h,u,p,d){const m=r?t-e:t+e;let f=m>0?1:-1,_=0;r&&(f*=-1,_=Math.PI),f<0&&(_+=Math.PI);let g,y,x=f>0?o+a:o+a+1,v=s,b=s,w=0,T=0;const S=Math.abs(m),I=[];let A;for(;w+T<=S;){if(x+=f,x=l)return null;w+=T,b=v,y=g;const t={projectionCache:u,lineVertexArray:c,labelPlaneMatrix:h,tileAnchorPoint:n,distanceFromAnchor:w,getElevation:d,previousVertex:b,direction:f,absOffsetX:S};if(v=Pu(x,t),0===i)I.push(b),A=v.sub(b);else{let e;const r=v.sub(b);e=0===r.mag()?ku(Pu(x+f,t).sub(v),i,f):ku(r,i,f),y||(y=b.add(e)),g=Du(x,e,v,o,l,y,i,t),I.push(y),A=g.sub(y)}T=A.mag()}const E=A._mult((S-w)/T)._add(y||b),C=_+Math.atan2(v.y-b.y,v.x-b.x);return I.push(E),{point:E,angle:p?C:0,path:I}}const Ru=new Float32Array([-1/0,-1/0,0,-1/0,-1/0,0,-1/0,-1/0,0,-1/0,-1/0,0]);function Bu(t,e){for(let i=0;i=1;t--)h.push(a.path[t]);for(let t=1;tTu(t,l,m)));h=t.some((t=>t.signedDistanceFromCamera<=0))?[]:t.map((t=>t.point))}let g=[];if(h.length>0){const t=h[0].clone(),e=h[0].clone();for(let i=1;i=i.x&&e.x<=r.x&&t.y>=i.y&&e.y<=r.y?[h]:e.xr.x||e.yr.y?[]:function(t,e,i,r,s){const a=[];for(let o=0;o=r&&h.x>=r||(o.x>=r?o=new n(r,o.y+(r-o.x)/(h.x-o.x)*(h.y-o.y))._round():h.x>=r&&(h=new n(r,o.y+(r-o.x)/(h.x-o.x)*(h.y-o.y))._round()),o.y>=s&&h.y>=s||(o.y>=s?o=new n(o.x+(s-o.y)/(h.y-o.y)*(h.x-o.x),s)._round():h.y>=s&&(h=new n(o.x+(s-o.y)/(h.y-o.y)*(h.x-o.x),s)._round()),c&&o.equals(c[c.length-1])||(c=[o],a.push(c)),c.push(h)))))}}return a}([h],i.x,i.y,r.x,r.y)}for(const i of g){s.reset(i,.25*e);let r=0;r=s.length<=.5*e?1:Math.ceil(s.paddedLength/_)+1;for(let i=0;i=this.screenRightBoundary||rthis.screenBottomBoundary}isInsideGrid(t,e,i,r){return i>=0&&t=0&&et.collisionGroupID===e}}return this.collisionGroups[t]}}function Xu(t,e,i,r,s){const{horizontalAlign:a,verticalAlign:o}=Rc(t);return new n(-(a-.5)*e+r[0]*s,-(o-.5)*i+r[1]*s)}function Ku(t,e,i,r,s,a){const{x1:o,x2:l,y1:c,y2:h,anchorPointX:u,anchorPointY:p}=t,d=new n(e,i);return r&&d._rotate(s?a:-a),{x1:o+d.x,y1:c+d.y,x2:l+d.x,y2:h+d.y,anchorPointX:u,anchorPointY:p}}class Yu{constructor(t,e,i,r,s){this.transform=t.clone(),this.terrain=e,this.collisionIndex=new Vu(this.transform),this.placements={},this.opacities={},this.variableOffsets={},this.stale=!1,this.commitTime=0,this.fadeDuration=i,this.retainedQueryData={},this.collisionGroups=new Wu(r),this.collisionCircleArrays={},this.prevPlacement=s,s&&(s.prevPlacement=void 0),this.placedOrientations={}}getBucketParts(t,e,i,r){const s=i.getBucket(e),n=i.latestFeatureIndex;if(!s||!n||e.id!==s.layerIds[0])return;const a=i.collisionBoxArray,o=s.layers[0].layout,l=Math.pow(2,this.transform.zoom-i.tileID.overscaledZ),c=i.tileSize/va,h=this.transform.calculatePosMatrix(i.tileID.toUnwrapped()),u="map"===o.get("text-pitch-alignment"),p="map"===o.get("text-rotation-alignment"),d=$u(i,1,this.transform.zoom),m=bu(h,u,p,this.transform,d);let f=null;if(u){const t=wu(h,u,p,this.transform,d);f=Xa([],this.transform.labelPlaneMatrix,t)}this.retainedQueryData[s.bucketInstanceId]=new Hu(s.bucketInstanceId,n,s.sourceLayerIndex,s.index,i.tileID);const _={bucket:s,layout:o,posMatrix:h,textLabelPlaneMatrix:m,labelToScreenMatrix:f,scale:l,textPixelRatio:c,holdingForFade:i.holdingForFade(),collisionBoxArray:a,partiallyEvaluatedTextSize:Uc(s.textSizeData,this.transform.zoom),collisionGroup:this.collisionGroups.get(s.sourceID)};if(r)for(const e of s.sortKeyRanges){const{sortKey:i,symbolInstanceStart:r,symbolInstanceEnd:s}=e;t.push({sortKey:i,symbolInstanceStart:r,symbolInstanceEnd:s,parameters:_})}else t.push({symbolInstanceStart:0,symbolInstanceEnd:s.symbolInstances.length,parameters:_})}attemptAnchorPlacement(t,e,i,r,s,n,a,o,l,c,h,u,p,d,m,f){const _=Nu[t.textAnchor],g=[t.textOffset0,t.textOffset1],y=Xu(_,i,r,g,s),x=this.collisionIndex.placeCollisionBox(Ku(e,y.x,y.y,n,a,this.transform.angle),h,o,l,c.predicate,f);if((!m||0!==this.collisionIndex.placeCollisionBox(Ku(m,y.x,y.y,n,a,this.transform.angle),h,o,l,c.predicate,f).box.length)&&x.box.length>0){let t;if(this.prevPlacement&&this.prevPlacement.variableOffsets[u.crossTileID]&&this.prevPlacement.placements[u.crossTileID]&&this.prevPlacement.placements[u.crossTileID].text&&(t=this.prevPlacement.variableOffsets[u.crossTileID].anchor),0===u.crossTileID)throw new Error("symbolInstance.crossTileID can't be 0");return this.variableOffsets[u.crossTileID]={textOffset:g,width:i,height:r,anchor:_,textBoxScale:s,prevAnchor:t},this.markUsedJustification(p,_,u,d),p.allowVerticalPlacement&&(this.markUsedOrientation(p,d,u),this.placedOrientations[u.crossTileID]=d),{shift:y,placedGlyphBoxes:x}}}placeLayerBucketPart(t,e,i){const{bucket:r,layout:s,posMatrix:n,textLabelPlaneMatrix:a,labelToScreenMatrix:o,textPixelRatio:l,holdingForFade:c,collisionBoxArray:h,partiallyEvaluatedTextSize:u,collisionGroup:p}=t.parameters,d=s.get("text-optional"),m=s.get("icon-optional"),f=Nc(s,"text-overlap","text-allow-overlap"),_="always"===f,g=Nc(s,"icon-overlap","icon-allow-overlap"),y="always"===g,x="map"===s.get("text-rotation-alignment"),v="map"===s.get("text-pitch-alignment"),b="none"!==s.get("icon-text-fit"),w="viewport-y"===s.get("symbol-z-order"),T=_&&(y||!r.hasIconData()||m),S=y&&(_||!r.hasTextData()||d);!r.collisionArrays&&h&&r.deserializeCollisionBoxes(h);const I=this.retainedQueryData[r.bucketInstanceId].tileID,E=this.terrain?(t,e)=>this.terrain.getElevation(I,t,e):null,C=(t,h)=>{var y,w;if(e[t.crossTileID])return;if(c)return void(this.placements[t.crossTileID]=new ju(!1,!1,!1));let I=!1,C=!1,z=!0,M=null,P={box:null,offscreen:null},k={box:null,offscreen:null},D=null,L=null,R=null,B=0,F=0,O=0;h.textFeatureIndex?B=h.textFeatureIndex:t.useRuntimeCollisionCircles&&(B=t.featureIndex),h.verticalTextFeatureIndex&&(F=h.verticalTextFeatureIndex);const V=h.textBox;if(V){const e=e=>{let i=Lc.horizontal;if(r.allowVerticalPlacement&&!e&&this.prevPlacement){const e=this.prevPlacement.placedOrientations[t.crossTileID];e&&(this.placedOrientations[t.crossTileID]=e,i=e,this.markUsedOrientation(r,i,t))}return i},i=(e,i)=>{if(r.allowVerticalPlacement&&t.numVerticalGlyphVertices>0&&h.verticalTextBox){for(const t of r.writingModes)if(t===Lc.vertical?(P=i(),k=P):P=e(),P&&P.box&&P.box.length)break}else P=e()},s=t.textAnchorOffsetStartIndex,a=t.textAnchorOffsetEndIndex;if(a===s){const s=(e,i)=>{const s=this.collisionIndex.placeCollisionBox(e,f,l,n,p.predicate,E);return s&&s.box&&s.box.length&&(this.markUsedOrientation(r,i,t),this.placedOrientations[t.crossTileID]=i),s};i((()=>s(V,Lc.horizontal)),(()=>{const e=h.verticalTextBox;return r.allowVerticalPlacement&&t.numVerticalGlyphVertices>0&&e?s(e,Lc.vertical):{box:null,offscreen:null}})),e(P&&P.box&&P.box.length)}else{let o=Nu[null===(w=null===(y=this.prevPlacement)||void 0===y?void 0:y.variableOffsets[t.crossTileID])||void 0===w?void 0:w.anchor];const c=(e,i,c)=>{const h=e.x2-e.x1,u=e.y2-e.y1,d=t.textBoxScale,m=b&&"never"===g?i:null;let _={box:[],offscreen:!1},y="never"===f?1:2,w="never";o&&y++;for(let i=0;ic(V,h.iconBox,Lc.horizontal)),(()=>{const e=h.verticalTextBox;return r.allowVerticalPlacement&&!(P&&P.box&&P.box.length)&&t.numVerticalGlyphVertices>0&&e?c(e,h.verticalIconBox,Lc.vertical):{box:null,offscreen:null}})),P&&(I=P.box,z=P.offscreen);const u=e(P&&P.box);if(!I&&this.prevPlacement){const e=this.prevPlacement.variableOffsets[t.crossTileID];e&&(this.variableOffsets[t.crossTileID]=e,this.markUsedJustification(r,e.anchor,t,u))}}}if(D=P,I=D&&D.box&&D.box.length>0,z=D&&D.offscreen,t.useRuntimeCollisionCircles){const e=r.text.placedSymbolArray.get(t.centerJustifiedTextSymbolIndex),l=Vc(r.textSizeData,u,e),c=s.get("text-padding");L=this.collisionIndex.placeCollisionCircles(f,e,r.lineVertexArray,r.glyphOffsetArray,l,n,a,o,i,v,p.predicate,t.collisionCircleDiameter,c,E),L.circles.length&&L.collisionDetected&&!i&&A("Collisions detected, but collision boxes are not shown"),I=_||L.circles.length>0&&!L.collisionDetected,z=z&&L.offscreen}if(h.iconFeatureIndex&&(O=h.iconFeatureIndex),h.iconBox){const t=t=>{const e=b&&M?Ku(t,M.x,M.y,x,v,this.transform.angle):t;return this.collisionIndex.placeCollisionBox(e,g,l,n,p.predicate,E)};k&&k.box&&k.box.length&&h.verticalIconBox?(R=t(h.verticalIconBox),C=R.box.length>0):(R=t(h.iconBox),C=R.box.length>0),z=z&&R.offscreen}const U=d||0===t.numHorizontalGlyphVertices&&0===t.numVerticalGlyphVertices,N=m||0===t.numIconVertices;if(U||N?N?U||(C=C&&I):I=C&&I:C=I=C&&I,I&&D&&D.box&&this.collisionIndex.insertCollisionBox(D.box,f,s.get("text-ignore-placement"),r.bucketInstanceId,k&&k.box&&F?F:B,p.ID),C&&R&&this.collisionIndex.insertCollisionBox(R.box,g,s.get("icon-ignore-placement"),r.bucketInstanceId,O,p.ID),L&&(I&&this.collisionIndex.insertCollisionCircles(L.circles,f,s.get("text-ignore-placement"),r.bucketInstanceId,B,p.ID),i)){const t=r.bucketInstanceId;let e=this.collisionCircleArrays[t];void 0===e&&(e=this.collisionCircleArrays[t]=new Gu);for(let t=0;t=0;--t){const i=e[t];C(r.symbolInstances.get(i),r.collisionArrays[i])}}else for(let e=t.symbolInstanceStart;e=0&&(t.text.placedSymbolArray.get(e).crossTileID=s>=0&&e!==s?0:i.crossTileID)}markUsedOrientation(t,e,i){const r=e===Lc.horizontal||e===Lc.horizontalOnly?e:0,s=e===Lc.vertical?e:0,n=[i.leftJustifiedTextSymbolIndex,i.centerJustifiedTextSymbolIndex,i.rightJustifiedTextSymbolIndex];for(const e of n)t.text.placedSymbolArray.get(e).placedOrientation=r;i.verticalPlacedTextSymbolIndex&&(t.text.placedSymbolArray.get(i.verticalPlacedTextSymbolIndex).placedOrientation=s)}commit(t){this.commitTime=t,this.zoomAtLastRecencyCheck=this.transform.zoom;const e=this.prevPlacement;let i=!1;this.prevZoomAdjustment=e?e.zoomAdjustment(this.transform.zoom):0;const r=e?e.symbolFadeChange(t):1,s=e?e.opacities:{},n=e?e.variableOffsets:{},a=e?e.placedOrientations:{};for(const t in this.placements){const e=this.placements[t],n=s[t];n?(this.opacities[t]=new Zu(n,r,e.text,e.icon),i=i||e.text!==n.text.placed||e.icon!==n.icon.placed):(this.opacities[t]=new Zu(null,r,e.text,e.icon,e.skipFade),i=i||e.text||e.icon)}for(const t in s){const e=s[t];if(!this.opacities[t]){const s=new Zu(e,r,!1,!1);s.isHidden()||(this.opacities[t]=s,i=i||e.text.placed||e.icon.placed)}}for(const t in n)this.variableOffsets[t]||!this.opacities[t]||this.opacities[t].isHidden()||(this.variableOffsets[t]=n[t]);for(const t in a)this.placedOrientations[t]||!this.opacities[t]||this.opacities[t].isHidden()||(this.placedOrientations[t]=a[t]);if(e&&void 0===e.lastPlacementChangeTime)throw new Error("Last placement time for previous placement is not defined");i?this.lastPlacementChangeTime=t:"number"!=typeof this.lastPlacementChangeTime&&(this.lastPlacementChangeTime=e?e.lastPlacementChangeTime:t)}updateLayerOpacities(t,e){const i={};for(const r of e){const e=r.getBucket(t);e&&r.latestFeatureIndex&&t.id===e.layerIds[0]&&this.updateBucketOpacities(e,i,r.collisionBoxArray)}}updateBucketOpacities(t,e,i){t.hasTextData()&&(t.text.opacityVertexArray.clear(),t.text.hasVisibleVertices=!1),t.hasIconData()&&(t.icon.opacityVertexArray.clear(),t.icon.hasVisibleVertices=!1),t.hasIconCollisionBoxData()&&t.iconCollisionBox.collisionVertexArray.clear(),t.hasTextCollisionBoxData()&&t.textCollisionBox.collisionVertexArray.clear();const r=t.layers[0],s=r.layout,a=new Zu(null,0,!1,!1,!0),o=s.get("text-allow-overlap"),l=s.get("icon-allow-overlap"),c=r._unevaluatedLayout.hasValue("text-variable-anchor")||r._unevaluatedLayout.hasValue("text-variable-anchor-offset"),h="map"===s.get("text-rotation-alignment"),u="map"===s.get("text-pitch-alignment"),p="none"!==s.get("icon-text-fit"),d=new Zu(null,0,o&&(l||!t.hasIconData()||s.get("icon-optional")),l&&(o||!t.hasTextData()||s.get("text-optional")),!0);!t.collisionArrays&&i&&(t.hasIconCollisionBoxData()||t.hasTextCollisionBoxData())&&t.deserializeCollisionBoxes(i);const m=(t,e,i)=>{for(let r=0;r0,g=this.placedOrientations[r.crossTileID],y=g===Lc.vertical,x=g===Lc.horizontal||g===Lc.horizontalOnly;if(s>0||o>0){const e=ap(f.text);m(t.text,s,y?op:e),m(t.text,o,x?op:e);const i=f.text.isHidden();[r.rightJustifiedTextSymbolIndex,r.centerJustifiedTextSymbolIndex,r.leftJustifiedTextSymbolIndex].forEach((e=>{e>=0&&(t.text.placedSymbolArray.get(e).hidden=i||y?1:0)})),r.verticalPlacedTextSymbolIndex>=0&&(t.text.placedSymbolArray.get(r.verticalPlacedTextSymbolIndex).hidden=i||x?1:0);const n=this.variableOffsets[r.crossTileID];n&&this.markUsedJustification(t,n.anchor,r,g);const a=this.placedOrientations[r.crossTileID];a&&(this.markUsedJustification(t,"left",r,a),this.markUsedOrientation(t,a,r))}if(_){const e=ap(f.icon),i=!(p&&r.verticalPlacedIconSymbolIndex&&y);r.placedIconSymbolIndex>=0&&(m(t.icon,r.numIconVertices,i?e:op),t.icon.placedSymbolArray.get(r.placedIconSymbolIndex).hidden=f.icon.isHidden()),r.verticalPlacedIconSymbolIndex>=0&&(m(t.icon,r.numVerticalIconVertices,i?op:e),t.icon.placedSymbolArray.get(r.verticalPlacedIconSymbolIndex).hidden=f.icon.isHidden())}if(t.hasIconCollisionBoxData()||t.hasTextCollisionBoxData()){const e=t.collisionArrays[i];if(e){let i=new n(0,0);if(e.textBox||e.verticalTextBox){let r=!0;if(c){const t=this.variableOffsets[l];t?(i=Xu(t.anchor,t.width,t.height,t.textOffset,t.textBoxScale),h&&i._rotate(u?this.transform.angle:-this.transform.angle)):r=!1}e.textBox&&Ju(t.textCollisionBox.collisionVertexArray,f.text.placed,!r||y,i.x,i.y),e.verticalTextBox&&Ju(t.textCollisionBox.collisionVertexArray,f.text.placed,!r||x,i.x,i.y)}const r=Boolean(!x&&e.verticalIconBox);e.iconBox&&Ju(t.iconCollisionBox.collisionVertexArray,f.icon.placed,r,p?i.x:0,p?i.y:0),e.verticalIconBox&&Ju(t.iconCollisionBox.collisionVertexArray,f.icon.placed,!r,p?i.x:0,p?i.y:0)}}}if(t.sortFeatures(this.transform.angle),this.retainedQueryData[t.bucketInstanceId]&&(this.retainedQueryData[t.bucketInstanceId].featureSortOrder=t.featureSortOrder),t.hasTextData()&&t.text.opacityVertexBuffer&&t.text.opacityVertexBuffer.updateData(t.text.opacityVertexArray),t.hasIconData()&&t.icon.opacityVertexBuffer&&t.icon.opacityVertexBuffer.updateData(t.icon.opacityVertexArray),t.hasIconCollisionBoxData()&&t.iconCollisionBox.collisionVertexBuffer&&t.iconCollisionBox.collisionVertexBuffer.updateData(t.iconCollisionBox.collisionVertexArray),t.hasTextCollisionBoxData()&&t.textCollisionBox.collisionVertexBuffer&&t.textCollisionBox.collisionVertexBuffer.updateData(t.textCollisionBox.collisionVertexArray),t.text.opacityVertexArray.length!==t.text.layoutVertexArray.length/4)throw new Error(`bucket.text.opacityVertexArray.length (= ${t.text.opacityVertexArray.length}) !== bucket.text.layoutVertexArray.length (= ${t.text.layoutVertexArray.length}) / 4`);if(t.icon.opacityVertexArray.length!==t.icon.layoutVertexArray.length/4)throw new Error(`bucket.icon.opacityVertexArray.length (= ${t.icon.opacityVertexArray.length}) !== bucket.icon.layoutVertexArray.length (= ${t.icon.layoutVertexArray.length}) / 4`);if(t.bucketInstanceId in this.collisionCircleArrays){const e=this.collisionCircleArrays[t.bucketInstanceId];t.placementInvProjMatrix=e.invProjMatrix,t.placementViewportMatrix=e.viewportMatrix,t.collisionCircleArray=e.circles,delete this.collisionCircleArrays[t.bucketInstanceId]}}symbolFadeChange(t){return 0===this.fadeDuration?1:(t-this.commitTime)/this.fadeDuration+this.prevZoomAdjustment}zoomAdjustment(t){return Math.max(0,(this.transform.zoom-t)/1.5)}hasTransitions(t){return this.stale||t-this.lastPlacementChangeTimet}setStale(){this.stale=!0}}function Ju(t,e,i,r,s){t.emplaceBack(e?1:0,i?1:0,r||0,s||0),t.emplaceBack(e?1:0,i?1:0,r||0,s||0),t.emplaceBack(e?1:0,i?1:0,r||0,s||0),t.emplaceBack(e?1:0,i?1:0,r||0,s||0)}const Qu=Math.pow(2,25),tp=Math.pow(2,24),ep=Math.pow(2,17),ip=Math.pow(2,16),rp=Math.pow(2,9),sp=Math.pow(2,8),np=Math.pow(2,1);function ap(t){if(0===t.opacity&&!t.placed)return 0;if(1===t.opacity&&t.placed)return 4294967295;const e=t.placed?1:0,i=Math.floor(127*t.opacity);return i*Qu+e*tp+i*ep+e*ip+i*rp+e*sp+i*np+e}const op=0;class lp{constructor(t){this._sortAcrossTiles="viewport-y"!==t.layout.get("symbol-z-order")&&!t.layout.get("symbol-sort-key").isConstant(),this._currentTileIndex=0,this._currentPartIndex=0,this._seenCrossTileIDs={},this._bucketParts=[]}continuePlacement(t,e,i,r,s){const n=this._bucketParts;for(;this._currentTileIndext.sortKey-e.sortKey)));this._currentPartIndex!this._forceFullPlacement&&B.now()-r>2;for(;this._currentPlacementIndex>=0;){const r=e[t[this._currentPlacementIndex]],n=this.placement.collisionIndex.transform.zoom;if("symbol"===r.type&&(!r.minzoom||r.minzoom<=n)&&(!r.maxzoom||r.maxzoom>n)){if(this._inProgressLayer||(this._inProgressLayer=new lp(r)),this._inProgressLayer.continuePlacement(i[r.source],this.placement,this._showCollisionBoxes,r,s))return;delete this._inProgressLayer}this._currentPlacementIndex--}this._done=!0}commit(t){return this.placement.commit(t),this.placement}}const hp=[Int8Array,Uint8Array,Uint8ClampedArray,Int16Array,Uint16Array,Int32Array,Uint32Array,Float32Array,Float64Array];class up{static from(t){if(!(t instanceof ArrayBuffer))throw new Error("Data must be an instance of ArrayBuffer.");const[e,i]=new Uint8Array(t,0,2);if(219!==e)throw new Error("Data does not appear to be in a KDBush format.");const r=i>>4;if(1!==r)throw new Error(`Got v${r} data when expected v1.`);const s=hp[15&i];if(!s)throw new Error("Unrecognized array type.");const[n]=new Uint16Array(t,2,1),[a]=new Uint32Array(t,4,1);return new up(a,n,s,t)}constructor(t,e=64,i=Float64Array,r){if(isNaN(t)||t<0)throw new Error(`Unpexpected numItems value: ${t}.`);this.numItems=+t,this.nodeSize=Math.min(Math.max(+e,2),65535),this.ArrayType=i,this.IndexArrayType=t<65536?Uint16Array:Uint32Array;const s=hp.indexOf(this.ArrayType),n=2*t*this.ArrayType.BYTES_PER_ELEMENT,a=t*this.IndexArrayType.BYTES_PER_ELEMENT,o=(8-a%8)%8;if(s<0)throw new Error(`Unexpected typed array class: ${i}.`);r&&r instanceof ArrayBuffer?(this.data=r,this.ids=new this.IndexArrayType(this.data,8,t),this.coords=new this.ArrayType(this.data,8+a+o,2*t),this._pos=2*t,this._finished=!0):(this.data=new ArrayBuffer(8+n+a+o),this.ids=new this.IndexArrayType(this.data,8,t),this.coords=new this.ArrayType(this.data,8+a+o,2*t),this._pos=0,this._finished=!1,new Uint8Array(this.data,0,2).set([219,16+s]),new Uint16Array(this.data,2,1)[0]=e,new Uint32Array(this.data,4,1)[0]=t)}add(t,e){const i=this._pos>>1;return this.ids[i]=i,this.coords[this._pos++]=t,this.coords[this._pos++]=e,i}finish(){const t=this._pos>>1;if(t!==this.numItems)throw new Error(`Added ${t} items when expected ${this.numItems}.`);return pp(this.ids,this.coords,this.nodeSize,0,this.numItems-1,0),this._finished=!0,this}range(t,e,i,r){if(!this._finished)throw new Error("Data not yet indexed - call index.finish().");const{ids:s,coords:n,nodeSize:a}=this,o=[0,s.length-1,0],l=[];for(;o.length;){const c=o.pop()||0,h=o.pop()||0,u=o.pop()||0;if(h-u<=a){for(let a=u;a<=h;a++){const o=n[2*a],c=n[2*a+1];o>=t&&o<=i&&c>=e&&c<=r&&l.push(s[a])}continue}const p=u+h>>1,d=n[2*p],m=n[2*p+1];d>=t&&d<=i&&m>=e&&m<=r&&l.push(s[p]),(0===c?t<=d:e<=m)&&(o.push(u),o.push(p-1),o.push(1-c)),(0===c?i>=d:r>=m)&&(o.push(p+1),o.push(h),o.push(1-c))}return l}within(t,e,i){if(!this._finished)throw new Error("Data not yet indexed - call index.finish().");const{ids:r,coords:s,nodeSize:n}=this,a=[0,r.length-1,0],o=[],l=i*i;for(;a.length;){const c=a.pop()||0,h=a.pop()||0,u=a.pop()||0;if(h-u<=n){for(let i=u;i<=h;i++)_p(s[2*i],s[2*i+1],t,e)<=l&&o.push(r[i]);continue}const p=u+h>>1,d=s[2*p],m=s[2*p+1];_p(d,m,t,e)<=l&&o.push(r[p]),(0===c?t-i<=d:e-i<=m)&&(a.push(u),a.push(p-1),a.push(1-c)),(0===c?t+i>=d:e+i>=m)&&(a.push(p+1),a.push(h),a.push(1-c))}return o}}function pp(t,e,i,r,s,n){if(s-r<=i)return;const a=r+s>>1;dp(t,e,a,r,s,n),pp(t,e,i,r,a-1,1-n),pp(t,e,i,a+1,s,1-n)}function dp(t,e,i,r,s,n){for(;s>r;){if(s-r>600){const a=s-r+1,o=i-r+1,l=Math.log(a),c=.5*Math.exp(2*l/3),h=.5*Math.sqrt(l*c*(a-c)/a)*(o-a/2<0?-1:1);dp(t,e,i,Math.max(r,Math.floor(i-o*c/a+h)),Math.min(s,Math.floor(i+(a-o)*c/a+h)),n)}const a=e[2*i+n];let o=r,l=s;for(mp(t,e,r,i),e[2*s+n]>a&&mp(t,e,r,s);oa;)l--}e[2*r+n]===a?mp(t,e,r,l):(l++,mp(t,e,l,s)),l<=i&&(r=l+1),i<=l&&(s=l-1)}}function mp(t,e,i,r){fp(t,i,r),fp(e,2*i,2*r),fp(e,2*i+1,2*r+1)}function fp(t,e,i){const r=t[e];t[e]=t[i],t[i]=r}function _p(t,e,i,r){const s=t-i,n=e-r;return s*s+n*n}const gp=512/va/2;class yp{constructor(t,e,i){this.tileID=t,this.bucketInstanceId=i,this._symbolsByKey={};const r=new Map;for(let t=0;t({x:Math.floor(t.anchorX*gp),y:Math.floor(t.anchorY*gp)}))),crossTileIDs:e.map((t=>t.crossTileID))};if(i.positions.length>128){const t=new up(i.positions.length,16,Uint16Array);for(const{x:e,y:r}of i.positions)t.add(e,r);t.finish(),delete i.positions,i.index=t}this._symbolsByKey[t]=i}}getScaledCoordinates(t,e){const{x:i,y:r,z:s}=this.tileID.canonical,{x:n,y:a,z:o}=e.canonical,l=gp/Math.pow(2,o-s),c=(a*va+t.anchorY)*l,h=r*va*gp;return{x:Math.floor((n*va+t.anchorX)*l-i*va*gp),y:Math.floor(c-h)}}findMatches(t,e,i){const r=this.tileID.canonical.zt))}}class xp{constructor(){this.maxCrossTileID=0}generate(){return++this.maxCrossTileID}}class vp{constructor(){this.indexes={},this.usedCrossTileIDs={},this.lng=0}handleWrapJump(t){const e=Math.round((t-this.lng)/360);if(0!==e)for(const t in this.indexes){const i=this.indexes[t],r={};for(const t in i){const s=i[t];s.tileID=s.tileID.unwrapTo(s.tileID.wrap+e),r[s.tileID.key]=s}this.indexes[t]=r}this.lng=t}addBucket(t,e,i){if(this.indexes[t.overscaledZ]&&this.indexes[t.overscaledZ][t.key]){if(this.indexes[t.overscaledZ][t.key].bucketInstanceId===e.bucketInstanceId)return!1;this.removeBucketCrossTileIDs(t.overscaledZ,this.indexes[t.overscaledZ][t.key])}for(let t=0;tt.overscaledZ)for(const i in s){const n=s[i];n.tileID.isChildOf(t)&&n.findMatches(e.symbolInstances,t,r)}else{const n=s[t.scaledTo(Number(i)).key];n&&n.findMatches(e.symbolInstances,t,r)}}for(let t=0;t{e[t]=!0}));for(const t in this.layerIndexes)e[t]||delete this.layerIndexes[t]}}const wp=(t,e)=>Zr(t,e&&e.filter((t=>"source.canvas"!==t.identifier))),Tp=y(dt,["addLayer","removeLayer","setPaintProperty","setLayoutProperty","setFilter","addSource","removeSource","setLayerZoomRange","setLight","setTransition","setGeoJSONSourceData","setGlyphs","setSprite"]),Sp=y(dt,["setCenter","setZoom","setBearing","setPitch"]),Ip=function(){const t={},e=lt.$version;for(const i in lt.$root){const r=lt.$root[i];if(r.required){let s=null;s="version"===i?e:"array"===r.type?[]:{},null!=s&&(t[i]=s)}}return t}();class Ap extends ot{constructor(t,e={}){super(),this.map=t,this.dispatcher=new Sh(gu(),this,t._getMapId()),this.imageManager=new dh,this.imageManager.setEventedParent(this),this.glyphManager=new gh(t._requestManager,e.localIdeographFontFamily),this.lineAtlas=new bh(256,512),this.crossTileSymbolIndex=new bp,this._spritesImagesIds={},this._layers={},this._order=[],this.sourceCaches={},this.zoomHistory=new Yr,this._loaded=!1,this._availableImages=[],this._resetUpdates(),this.dispatcher.broadcast("setReferrer",U());const i=this;this._rtlTextPluginCallback=Ap.registerForPluginStateChange((t=>{i.dispatcher.broadcast("syncRTLPluginState",{pluginStatus:t.pluginStatus,pluginURL:t.pluginURL},((t,e)=>{if(hs(t),e&&e.every((t=>t)))for(const t in i.sourceCaches){const e=i.sourceCaches[t].getSource().type;"vector"!==e&&"geojson"!==e||i.sourceCaches[t].reload()}}))})),this.on("data",(t=>{if("source"!==t.dataType||"metadata"!==t.sourceDataType)return;const e=this.sourceCaches[t.sourceId];if(!e)return;const i=e.getSource();if(i&&i.vectorLayerIds)for(const t in this._layers){const e=this._layers[t];e.source===i.id&&this._validateLayer(e)}}))}loadURL(t,e={},i){this.fire(new nt("dataloading",{dataType:"style"})),e.validate="boolean"!=typeof e.validate||e.validate;const r=this.map._requestManager.transformRequest(t,tt.Style);this._request=Z(r,((t,r)=>{this._request=null,t?this.fire(new at(t)):r&&this._load(r,e,i)}))}loadJSON(t,e={},i){this.fire(new nt("dataloading",{dataType:"style"})),this._request=B.frame((()=>{this._request=null,e.validate=!1!==e.validate,this._load(t,e,i)}))}loadEmpty(){this.fire(new nt("dataloading",{dataType:"style"})),this._load(Ip,{validate:!1})}_load(t,e,i){var r;const s=e.transformStyle?e.transformStyle(i,t):t;if(!e.validate||!wp(this,Ur(s))){this._loaded=!0,this.stylesheet=s;for(const t in s.sources)this.addSource(t,s.sources[t],{validate:!1});s.sprite?this._loadSprite(s.sprite):this.imageManager.setLoaded(!0),this.glyphManager.setURL(s.glyphs),this._createLayers(),this.light=new vh(this.stylesheet.light),this.map.setTerrain(null!==(r=this.stylesheet.terrain)&&void 0!==r?r:null),this.fire(new nt("data",{dataType:"style"})),this.fire(new nt("style.load"))}}_createLayers(){const t=ut(this.stylesheet.layers);this.dispatcher.broadcast("setLayers",t),this._order=t.map((t=>t.id)),this._layers={},this._serializedLayers=null;for(const e of t){const t=lh(e);t.setEventedParent(this,{layer:{id:e.id}}),this._layers[e.id]=t}}_loadSprite(t,e=!1,i=void 0){this.imageManager.setLoaded(!1),this._spriteRequest=function(t,e,i,r){const s=ch(t),n=s.length,a=i>1?"@2x":"",o={},l={},c={};for(const{id:t,url:i}of s){const s=e.transformRequest(e.normalizeSpriteURL(i,a,".json"),tt.SpriteJSON),h=`${t}_${s.url}`;o[h]=Z(s,((e,i)=>{delete o[h],l[t]=i,hh(r,l,c,e,n)}));const u=e.transformRequest(e.normalizeSpriteURL(i,a,".png"),tt.SpriteImage),p=`${t}_${u.url}`;o[p]=Q.getImage(u,((e,i)=>{delete o[p],c[t]=i,hh(r,l,c,e,n)}))}return{cancel(){for(const t of Object.values(o))t.cancel()}}}(t,this.map._requestManager,this.map.getPixelRatio(),((t,r)=>{if(this._spriteRequest=null,t)this.fire(new at(t));else if(r)for(const t in r){this._spritesImagesIds[t]=[];const i=this._spritesImagesIds[t]?this._spritesImagesIds[t].filter((t=>!(t in r))):[];for(const t of i)this.imageManager.removeImage(t),this._changedImages[t]=!0;for(const i in r[t]){const s="default"===t?i:`${t}:${i}`;this._spritesImagesIds[t].push(s),s in this.imageManager.images?this.imageManager.updateImage(s,r[t][i],!1):this.imageManager.addImage(s,r[t][i]),e&&(this._changedImages[s]=!0)}}this.imageManager.setLoaded(!0),this._availableImages=this.imageManager.listImages(),e&&(this._changed=!0),this.dispatcher.broadcast("setImages",this._availableImages),this.fire(new nt("data",{dataType:"style"})),i&&i(t)}))}_unloadSprite(){for(const t of Object.values(this._spritesImagesIds).flat())this.imageManager.removeImage(t),this._changedImages[t]=!0;this._spritesImagesIds={},this._availableImages=this.imageManager.listImages(),this._changed=!0,this.dispatcher.broadcast("setImages",this._availableImages),this.fire(new nt("data",{dataType:"style"}))}_validateLayer(t){const e=this.sourceCaches[t.source];if(!e)return;const i=t.sourceLayer;if(!i)return;const r=e.getSource();("geojson"===r.type||r.vectorLayerIds&&-1===r.vectorLayerIds.indexOf(i))&&this.fire(new at(new Error(`Source layer "${i}" does not exist on source "${r.id}" as specified by style layer "${t.id}".`)))}loaded(){if(!this._loaded)return!1;if(Object.keys(this._updatedSources).length)return!1;for(const t in this.sourceCaches)if(!this.sourceCaches[t].loaded())return!1;return!!this.imageManager.isLoaded()}_serializeByIds(t){const e=this._serializedAllLayers();if(!t||0===t.length)return Object.values(e);const i=[];for(const r of t)e[r]&&i.push(e[r]);return i}_serializedAllLayers(){let t=this._serializedLayers;if(t)return t;t=this._serializedLayers={};const e=Object.keys(this._layers);for(const i of e){const e=this._layers[i];"custom"!==e.type&&(t[i]=e.serialize())}return t}hasTransitions(){if(this.light&&this.light.hasTransition())return!0;for(const t in this.sourceCaches)if(this.sourceCaches[t].hasTransition())return!0;for(const t in this._layers)if(this._layers[t].hasTransition())return!0;return!1}_checkLoaded(){if(!this._loaded)throw new Error("Style is not done loading.")}update(t){if(!this._loaded)return;const e=this._changed;if(this._changed){const e=Object.keys(this._updatedLayers),i=Object.keys(this._removedLayers);(e.length||i.length)&&this._updateWorkerLayers(e,i);for(const t in this._updatedSources){const e=this._updatedSources[t];if("reload"===e)this._reloadSource(t);else{if("clear"!==e)throw new Error(`Invalid action ${e}`);this._clearSource(t)}}this._updateTilesForChangedImages(),this._updateTilesForChangedGlyphs();for(const e in this._updatedPaintProps)this._layers[e].updateTransitions(t);this.light.updateTransitions(t),this._resetUpdates()}const i={};for(const t in this.sourceCaches){const e=this.sourceCaches[t];i[t]=e.used,e.used=!1}for(const e of this._order){const i=this._layers[e];i.recalculate(t,this._availableImages),!i.isHidden(t.zoom)&&i.source&&(this.sourceCaches[i.source].used=!0)}for(const t in i){const e=this.sourceCaches[t];i[t]!==e.used&&e.fire(new nt("data",{sourceDataType:"visibility",dataType:"source",sourceId:t}))}this.light.recalculate(t),this.z=t.zoom,e&&this.fire(new nt("data",{dataType:"style"}))}_updateTilesForChangedImages(){const t=Object.keys(this._changedImages);if(t.length){for(const e in this.sourceCaches)this.sourceCaches[e].reloadTilesForDependencies(["icons","patterns"],t);this._changedImages={}}}_updateTilesForChangedGlyphs(){if(this._glyphsDidChange){for(const t in this.sourceCaches)this.sourceCaches[t].reloadTilesForDependencies(["glyphs"],[""]);this._glyphsDidChange=!1}}_updateWorkerLayers(t,e){this.dispatcher.broadcast("updateLayers",{layers:this._serializeByIds(t),removedIds:e})}_resetUpdates(){this._changed=!1,this._updatedLayers={},this._removedLayers={},this._updatedSources={},this._updatedPaintProps={},this._changedImages={},this._glyphsDidChange=!1}setState(t,e={}){this._checkLoaded();const i=this.serialize();if(t=e.transformStyle?e.transformStyle(i,t):t,wp(this,Ur(t)))return!1;(t=S(t)).layers=ut(t.layers);const r=function(t,e){if(!t)return[{command:dt.setStyle,args:[e]}];let i=[];try{if(!pt(t.version,e.version))return[{command:dt.setStyle,args:[e]}];pt(t.center,e.center)||i.push({command:dt.setCenter,args:[e.center]}),pt(t.zoom,e.zoom)||i.push({command:dt.setZoom,args:[e.zoom]}),pt(t.bearing,e.bearing)||i.push({command:dt.setBearing,args:[e.bearing]}),pt(t.pitch,e.pitch)||i.push({command:dt.setPitch,args:[e.pitch]}),pt(t.sprite,e.sprite)||i.push({command:dt.setSprite,args:[e.sprite]}),pt(t.glyphs,e.glyphs)||i.push({command:dt.setGlyphs,args:[e.glyphs]}),pt(t.transition,e.transition)||i.push({command:dt.setTransition,args:[e.transition]}),pt(t.light,e.light)||i.push({command:dt.setLight,args:[e.light]});const r={},s=[];!function(t,e,i,r){let s;for(s in e=e||{},t=t||{})Object.prototype.hasOwnProperty.call(t,s)&&(Object.prototype.hasOwnProperty.call(e,s)||ft(s,i,r));for(s in e)Object.prototype.hasOwnProperty.call(e,s)&&(Object.prototype.hasOwnProperty.call(t,s)?pt(t[s],e[s])||("geojson"===t[s].type&&"geojson"===e[s].type&>(t,e,s)?i.push({command:dt.setGeoJSONSourceData,args:[s,e[s].data]}):_t(s,e,i,r)):mt(s,e,i))}(t.sources,e.sources,s,r);const n=[];t.layers&&t.layers.forEach((t=>{r[t.source]?i.push({command:dt.removeLayer,args:[t.id]}):n.push(t)})),i=i.concat(s),function(t,e,i){e=e||[];const r=(t=t||[]).map(xt),s=e.map(xt),n=t.reduce(vt,{}),a=e.reduce(vt,{}),o=r.slice(),l=Object.create(null);let c,h,u,p,d,m,f;for(c=0,h=0;c!(t.command in Sp)));if(0===r.length)return!1;const s=r.filter((t=>!(t.command in Tp)));if(s.length>0)throw new Error(`Unimplemented: ${s.map((t=>t.command)).join(", ")}.`);for(const t of r)"setTransition"!==t.command&&this[t.command].apply(this,t.args);return this.stylesheet=t,this._serializedLayers=null,!0}addImage(t,e){if(this.getImage(t))return this.fire(new at(new Error(`An image named "${t}" already exists.`)));this.imageManager.addImage(t,e),this._afterImageUpdated(t)}updateImage(t,e){this.imageManager.updateImage(t,e)}getImage(t){return this.imageManager.getImage(t)}removeImage(t){if(!this.getImage(t))return this.fire(new at(new Error(`An image named "${t}" does not exist.`)));this.imageManager.removeImage(t),this._afterImageUpdated(t)}_afterImageUpdated(t){this._availableImages=this.imageManager.listImages(),this._changedImages[t]=!0,this._changed=!0,this.dispatcher.broadcast("setImages",this._availableImages),this.fire(new nt("data",{dataType:"style"}))}listImages(){return this._checkLoaded(),this.imageManager.listImages()}addSource(t,e,i={}){if(this._checkLoaded(),void 0!==this.sourceCaches[t])throw new Error(`Source "${t}" already exists.`);if(!e.type)throw new Error(`The type property must be defined, but only the following properties were given: ${Object.keys(e).join(", ")}.`);if(["vector","raster","geojson","video","image"].indexOf(e.type)>=0&&this._validate(Ur.source,`sources.${t}`,e,null,i))return;this.map&&this.map._collectResourceTiming&&(e.collectResourceTiming=!0);const r=this.sourceCaches[t]=new hu(t,e,this.dispatcher);r.style=this,r.setEventedParent(this,(()=>({isSourceLoaded:r.loaded(),source:r.serialize(),sourceId:t}))),r.onAdd(this.map),this._changed=!0}removeSource(t){if(this._checkLoaded(),void 0===this.sourceCaches[t])throw new Error("There is no source with this ID");for(const e in this._layers)if(this._layers[e].source===t)return this.fire(new at(new Error(`Source "${t}" cannot be removed while layer "${e}" is using it.`)));const e=this.sourceCaches[t];delete this.sourceCaches[t],delete this._updatedSources[t],e.fire(new nt("data",{sourceDataType:"metadata",dataType:"source",sourceId:t})),e.setEventedParent(null),e.onRemove(this.map),this._changed=!0}setGeoJSONSourceData(t,e){if(this._checkLoaded(),void 0===this.sourceCaches[t])throw new Error(`There is no source with this ID=${t}`);const i=this.sourceCaches[t].getSource();if("geojson"!==i.type)throw new Error(`geojsonSource.type is ${i.type}, which is !== 'geojson`);i.setData(e),this._changed=!0}getSource(t){return this.sourceCaches[t]&&this.sourceCaches[t].getSource()}addLayer(t,e,i={}){this._checkLoaded();const r=t.id;if(this.getLayer(r))return void this.fire(new at(new Error(`Layer "${r}" already exists on this map.`)));let s;if("custom"===t.type){if(wp(this,function(t){const e=[],i=t.id;return void 0===i&&e.push({message:`layers.${i}: missing required property "id"`}),void 0===t.render&&e.push({message:`layers.${i}: missing required method "render"`}),t.renderingMode&&"2d"!==t.renderingMode&&"3d"!==t.renderingMode&&e.push({message:`layers.${i}: property "renderingMode" must be either "2d" or "3d"`}),e}(t)))return;s=lh(t)}else{if("source"in t&&"object"==typeof t.source&&(this.addSource(r,t.source),t=g(t=S(t),{source:r})),this._validate(Ur.layer,`layers.${r}`,t,{arrayIndex:-1},i))return;s=lh(t),this._validateLayer(s),s.setEventedParent(this,{layer:{id:r}})}const n=e?this._order.indexOf(e):this._order.length;if(e&&-1===n)this.fire(new at(new Error(`Cannot add layer "${r}" before non-existing layer "${e}".`)));else{if(this._order.splice(n,0,r),this._layerOrderChanged=!0,this._layers[r]=s,this._removedLayers[r]&&s.source&&"custom"!==s.type){const t=this._removedLayers[r];delete this._removedLayers[r],t.type!==s.type?this._updatedSources[s.source]="clear":(this._updatedSources[s.source]="reload",this.sourceCaches[s.source].pause())}this._updateLayer(s),s.onAdd&&s.onAdd(this.map)}}moveLayer(t,e){if(this._checkLoaded(),this._changed=!0,!this._layers[t])return void this.fire(new at(new Error(`The layer '${t}' does not exist in the map's style and cannot be moved.`)));if(t===e)return;const i=this._order.indexOf(t);this._order.splice(i,1);const r=e?this._order.indexOf(e):this._order.length;e&&-1===r?this.fire(new at(new Error(`Cannot move layer "${t}" before non-existing layer "${e}".`))):(this._order.splice(r,0,t),this._layerOrderChanged=!0)}removeLayer(t){this._checkLoaded();const e=this._layers[t];if(!e)return void this.fire(new at(new Error(`Cannot remove non-existing layer "${t}".`)));e.setEventedParent(null);const i=this._order.indexOf(t);this._order.splice(i,1),this._layerOrderChanged=!0,this._changed=!0,this._removedLayers[t]=e,delete this._layers[t],this._serializedLayers&&delete this._serializedLayers[t],delete this._updatedLayers[t],delete this._updatedPaintProps[t],e.onRemove&&e.onRemove(this.map)}getLayer(t){return this._layers[t]}getLayersOrder(){return[...this._order]}hasLayer(t){return t in this._layers}setLayerZoomRange(t,e,i){this._checkLoaded();const r=this.getLayer(t);r?r.minzoom===e&&r.maxzoom===i||(null!=e&&(r.minzoom=e),null!=i&&(r.maxzoom=i),this._updateLayer(r)):this.fire(new at(new Error(`Cannot set the zoom range of non-existing layer "${t}".`)))}setFilter(t,e,i={}){this._checkLoaded();const r=this.getLayer(t);if(r){if(!T(r.filter,e))return null==e?(r.filter=void 0,void this._updateLayer(r)):void(this._validate(Ur.filter,`layers.${r.id}.filter`,e,null,i)||(r.filter=S(e),this._updateLayer(r)))}else this.fire(new at(new Error(`Cannot filter non-existing layer "${t}".`)))}getFilter(t){return S(this.getLayer(t).filter)}setLayoutProperty(t,e,i,r={}){this._checkLoaded();const s=this.getLayer(t);s?T(s.getLayoutProperty(e),i)||(s.setLayoutProperty(e,i,r),this._updateLayer(s)):this.fire(new at(new Error(`Cannot style non-existing layer "${t}".`)))}getLayoutProperty(t,e){const i=this.getLayer(t);if(i)return i.getLayoutProperty(e);this.fire(new at(new Error(`Cannot get style of non-existing layer "${t}".`)))}setPaintProperty(t,e,i,r={}){this._checkLoaded();const s=this.getLayer(t);s?T(s.getPaintProperty(e),i)||(s.setPaintProperty(e,i,r)&&this._updateLayer(s),this._changed=!0,this._updatedPaintProps[t]=!0):this.fire(new at(new Error(`Cannot style non-existing layer "${t}".`)))}getPaintProperty(t,e){return this.getLayer(t).getPaintProperty(e)}setFeatureState(t,e){this._checkLoaded();const i=t.source,r=t.sourceLayer,s=this.sourceCaches[i];if(void 0===s)return void this.fire(new at(new Error(`The source '${i}' does not exist in the map's style.`)));const n=s.getSource().type;"geojson"===n&&r?this.fire(new at(new Error("GeoJSON sources cannot have a sourceLayer parameter."))):"vector"!==n||r?(void 0===t.id&&this.fire(new at(new Error("The feature id parameter must be provided."))),s.setFeatureState(r,t.id,e)):this.fire(new at(new Error("The sourceLayer parameter must be provided for vector source types.")))}removeFeatureState(t,e){this._checkLoaded();const i=t.source,r=this.sourceCaches[i];if(void 0===r)return void this.fire(new at(new Error(`The source '${i}' does not exist in the map's style.`)));const s=r.getSource().type,n="vector"===s?t.sourceLayer:void 0;"vector"!==s||n?e&&"string"!=typeof t.id&&"number"!=typeof t.id?this.fire(new at(new Error("A feature id is required to remove its specific state property."))):r.removeFeatureState(n,t.id,e):this.fire(new at(new Error("The sourceLayer parameter must be provided for vector source types.")))}getFeatureState(t){this._checkLoaded();const e=t.source,i=t.sourceLayer,r=this.sourceCaches[e];if(void 0!==r)return"vector"!==r.getSource().type||i?(void 0===t.id&&this.fire(new at(new Error("The feature id parameter must be provided."))),r.getFeatureState(i,t.id)):void this.fire(new at(new Error("The sourceLayer parameter must be provided for vector source types.")));this.fire(new at(new Error(`The source '${e}' does not exist in the map's style.`)))}getTransition(){return g({duration:300,delay:0},this.stylesheet&&this.stylesheet.transition)}serialize(){if(!this._loaded)return;const t=b(this.sourceCaches,(t=>t.serialize())),e=this._serializeByIds(this._order),i=this.map.getTerrain()||void 0,r=this.stylesheet;return w({version:r.version,name:r.name,metadata:r.metadata,light:r.light,center:r.center,zoom:r.zoom,bearing:r.bearing,pitch:r.pitch,sprite:r.sprite,glyphs:r.glyphs,transition:r.transition,sources:t,layers:e,terrain:i},(t=>void 0!==t))}_updateLayer(t){this._updatedLayers[t.id]=!0,t.source&&!this._updatedSources[t.source]&&"raster"!==this.sourceCaches[t.source].getSource().type&&(this._updatedSources[t.source]="reload",this.sourceCaches[t.source].pause()),this._serializedLayers=null,this._changed=!0}_flattenAndSortRenderedFeatures(t){const e=t=>"fill-extrusion"===this._layers[t].type,i={},r=[];for(let s=this._order.length-1;s>=0;s--){const n=this._order[s];if(e(n)){i[n]=s;for(const e of t){const t=e[n];if(t)for(const e of t)r.push(e)}}}r.sort(((t,e)=>e.intersectionZ-t.intersectionZ));const s=[];for(let n=this._order.length-1;n>=0;n--){const a=this._order[n];if(e(a))for(let t=r.length-1;t>=0;t--){const e=r[t].feature;if(i[e.layer.id]{const r=i.featureSortOrder;if(r){const i=r.indexOf(t.featureIndex);return r.indexOf(e.featureIndex)-i}return e.featureIndex-t.featureIndex}));for(const t of s)e.push(t)}}for(const e in o)o[e].forEach((r=>{const s=r.feature,n=i[t[e].source].getFeatureState(s.layer["source-layer"],s.id);s.source=s.layer.source,s.layer["source-layer"]&&(s.sourceLayer=s.layer["source-layer"]),s.state=n}));return o}(this._layers,n,this.sourceCaches,t,e,this.placement.collisionIndex,this.placement.retainedQueryData)),this._flattenAndSortRenderedFeatures(s)}querySourceFeatures(t,e){e&&e.filter&&this._validate(Ur.filter,"querySourceFeatures.filter",e.filter,null,e);const i=this.sourceCaches[t];return i?function(t,e){const i=t.getRenderableIds().map((e=>t.getTileByID(e))),r=[],s={};for(let t=0;t{Yh[t]=e})(t,e),e.workerSourceURL?void this.dispatcher.broadcast("loadWorkerSource",{name:t,url:e.workerSourceURL},i):i(null,null))}getLight(){return this.light.getLight()}setLight(t,e={}){this._checkLoaded();const i=this.light.getLight();let r=!1;for(const e in t)if(!T(t[e],i[e])){r=!0;break}if(!r)return;const s={now:B.now(),transition:g({duration:300,delay:0},this.stylesheet.transition)};this.light.setLight(t,e),this.light.updateTransitions(s)}_validate(t,e,i,r,s={}){return(!s||!1!==s.validate)&&wp(this,t.call(Ur,g({key:e,style:this.serialize(),value:i,styleSpec:lt},r)))}_remove(t=!0){this._request&&(this._request.cancel(),this._request=null),this._spriteRequest&&(this._spriteRequest.cancel(),this._spriteRequest=null),ps.off("pluginStateChange",this._rtlTextPluginCallback);for(const t in this._layers)this._layers[t].setEventedParent(null);for(const t in this.sourceCaches){const e=this.sourceCaches[t];e.setEventedParent(null),e.onRemove(this.map)}this.imageManager.setEventedParent(null),this.setEventedParent(null),this.dispatcher.remove(t)}_clearSource(t){this.sourceCaches[t].clearTiles()}_reloadSource(t){this.sourceCaches[t].resume(),this.sourceCaches[t].reload()}_updateSources(t){for(const e in this.sourceCaches)this.sourceCaches[e].update(t,this.map.terrain)}_generateCollisionBoxes(){for(const t in this.sourceCaches)this._reloadSource(t)}_updatePlacement(t,e,i,r,s=!1){let n=!1,a=!1;const o={};for(const e of this._order){const i=this._layers[e];if("symbol"!==i.type)continue;if(!o[i.source]){const t=this.sourceCaches[i.source];o[i.source]=t.getRenderableIds(!0).map((e=>t.getTileByID(e))).sort(((t,e)=>e.tileID.overscaledZ-t.tileID.overscaledZ||(t.tileID.isLessThan(e.tileID)?-1:1)))}const r=this.crossTileSymbolIndex.addLayer(i,o[i.source],t.center.lng);n=n||r}if(this.crossTileSymbolIndex.pruneUnusedLayers(this._order),((s=s||this._layerOrderChanged||0===i)||!this.pauseablePlacement||this.pauseablePlacement.isDone()&&!this.placement.stillRecent(B.now(),t.zoom))&&(this.pauseablePlacement=new cp(t,this.map.terrain,this._order,s,e,i,r,this.placement),this._layerOrderChanged=!1),this.pauseablePlacement.isDone()?this.placement.setStale():(this.pauseablePlacement.continuePlacement(this._order,this._layers,o),this.pauseablePlacement.isDone()&&(this.placement=this.pauseablePlacement.commit(B.now()),a=!0),n&&this.pauseablePlacement.placement.setStale()),a||n)for(const t of this._order){const e=this._layers[t];"symbol"===e.type&&this.placement.updateLayerOpacities(e,o[e.source])}return!this.pauseablePlacement.isDone()||this.placement.hasTransitions(B.now())}_releaseSymbolFadeTiles(){for(const t in this.sourceCaches)this.sourceCaches[t].releaseSymbolFadeTiles()}getImages(t,e,i){this.imageManager.getImages(e.icons,i),this._updateTilesForChangedImages();const r=this.sourceCaches[e.source];r&&r.setDependencies(e.tileID.key,e.type,e.icons)}getGlyphs(t,e,i){this.glyphManager.getGlyphs(e.stacks,i);const r=this.sourceCaches[e.source];r&&r.setDependencies(e.tileID.key,e.type,[""])}getResource(t,e,i){return q(e,i)}getGlyphsUrl(){return this.stylesheet.glyphs||null}setGlyphs(t,e={}){this._checkLoaded(),t&&this._validate(Ur.glyphs,"glyphs",t,null,e)||(this._glyphsDidChange=!0,this.stylesheet.glyphs=t,this.glyphManager.entries={},this.glyphManager.setURL(t))}addSprite(t,e,i={},r){this._checkLoaded();const s=[{id:t,url:e}],n=[...ch(this.stylesheet.sprite),...s];this._validate(Ur.sprite,"sprite",n,null,i)||(this.stylesheet.sprite=n,this._loadSprite(s,!0,r))}removeSprite(t){this._checkLoaded();const e=ch(this.stylesheet.sprite);if(e.find((e=>e.id===t))){if(this._spritesImagesIds[t])for(const e of this._spritesImagesIds[t])this.imageManager.removeImage(e),this._changedImages[e]=!0;e.splice(e.findIndex((e=>e.id===t)),1),this.stylesheet.sprite=e.length>0?e:void 0,delete this._spritesImagesIds[t],this._availableImages=this.imageManager.listImages(),this._changed=!0,this.dispatcher.broadcast("setImages",this._availableImages),this.fire(new nt("data",{dataType:"style"}))}else this.fire(new at(new Error(`Sprite "${t}" doesn't exists on this map.`)))}getSprite(){return ch(this.stylesheet.sprite)}setSprite(t,e={},i){this._checkLoaded(),t&&this._validate(Ur.sprite,"sprite",t,null,e)||(this.stylesheet.sprite=t,t?this._loadSprite(t,!0,i):(this._unloadSprite(),i&&i(null)))}}Ap.registerForPluginStateChange=function(t){return t({pluginStatus:ls,pluginURL:cs}),ps.on("pluginStateChange",t),t};var Ep=Bs([{name:"a_pos",type:"Int16",components:2}]),Cp="attribute vec3 a_pos3d;uniform mat4 u_matrix;uniform float u_ele_delta;varying vec2 v_texture_pos;varying float v_depth;void main() {float extent=8192.0;float ele_delta=a_pos3d.z==1.0 ? u_ele_delta : 0.0;v_texture_pos=a_pos3d.xy/extent;gl_Position=u_matrix*vec4(a_pos3d.xy,get_elevation(a_pos3d.xy)-ele_delta,1.0);v_depth=gl_Position.z/gl_Position.w;}";const zp={prelude:Mp("#ifdef GL_ES\nprecision mediump float;\n#else\n#if !defined(lowp)\n#define lowp\n#endif\n#if !defined(mediump)\n#define mediump\n#endif\n#if !defined(highp)\n#define highp\n#endif\n#endif\n","#ifdef GL_ES\nprecision highp float;\n#else\n#if !defined(lowp)\n#define lowp\n#endif\n#if !defined(mediump)\n#define mediump\n#endif\n#if !defined(highp)\n#define highp\n#endif\n#endif\nvec2 unpack_float(const float packedValue) {int packedIntValue=int(packedValue);int v0=packedIntValue/256;return vec2(v0,packedIntValue-v0*256);}vec2 unpack_opacity(const float packedOpacity) {int intOpacity=int(packedOpacity)/2;return vec2(float(intOpacity)/127.0,mod(packedOpacity,2.0));}vec4 decode_color(const vec2 encodedColor) {return vec4(unpack_float(encodedColor[0])/255.0,unpack_float(encodedColor[1])/255.0\n);}float unpack_mix_vec2(const vec2 packedValue,const float t) {return mix(packedValue[0],packedValue[1],t);}vec4 unpack_mix_color(const vec4 packedColors,const float t) {vec4 minColor=decode_color(vec2(packedColors[0],packedColors[1]));vec4 maxColor=decode_color(vec2(packedColors[2],packedColors[3]));return mix(minColor,maxColor,t);}vec2 get_pattern_pos(const vec2 pixel_coord_upper,const vec2 pixel_coord_lower,const vec2 pattern_size,const float tile_units_to_pixels,const vec2 pos) {vec2 offset=mod(mod(mod(pixel_coord_upper,pattern_size)*256.0,pattern_size)*256.0+pixel_coord_lower,pattern_size);return (tile_units_to_pixels*pos+offset)/pattern_size;}\n#ifdef TERRAIN3D\nuniform sampler2D u_terrain;uniform float u_terrain_dim;uniform mat4 u_terrain_matrix;uniform vec4 u_terrain_unpack;uniform float u_terrain_exaggeration;uniform highp sampler2D u_depth;\n#endif\nconst highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitShifts=vec4(1.)/bitSh;highp float unpack(highp vec4 color) {return dot(color,bitShifts);}highp float depthOpacity(vec3 frag) {\n#ifdef TERRAIN3D\nhighp float d=unpack(texture2D(u_depth,frag.xy*0.5+0.5))+0.0001-frag.z;return 1.0-max(0.0,min(1.0,-d*500.0));\n#else\nreturn 1.0;\n#endif\n}float calculate_visibility(vec4 pos) {\n#ifdef TERRAIN3D\nvec3 frag=pos.xyz/pos.w;highp float d=depthOpacity(frag);if (d > 0.95) return 1.0;return (d+depthOpacity(frag+vec3(0.0,0.01,0.0)))/2.0;\n#else\nreturn 1.0;\n#endif\n}float ele(vec2 pos) {\n#ifdef TERRAIN3D\nvec4 rgb=(texture2D(u_terrain,pos)*255.0)*u_terrain_unpack;return rgb.r+rgb.g+rgb.b-u_terrain_unpack.a;\n#else\nreturn 0.0;\n#endif\n}float get_elevation(vec2 pos) {\n#ifdef TERRAIN3D\nvec2 coord=(u_terrain_matrix*vec4(pos,0.0,1.0)).xy*u_terrain_dim+1.0;vec2 f=fract(coord);vec2 c=(floor(coord)+0.5)/(u_terrain_dim+2.0);float d=1.0/(u_terrain_dim+2.0);float tl=ele(c);float tr=ele(c+vec2(d,0.0));float bl=ele(c+vec2(0.0,d));float br=ele(c+vec2(d,d));float elevation=mix(mix(tl,tr,f.x),mix(bl,br,f.x),f.y);return elevation*u_terrain_exaggeration;\n#else\nreturn 0.0;\n#endif\n}"),background:Mp("uniform vec4 u_color;uniform float u_opacity;void main() {gl_FragColor=u_color*u_opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}"),backgroundPattern:Mp("uniform vec2 u_pattern_tl_a;uniform vec2 u_pattern_br_a;uniform vec2 u_pattern_tl_b;uniform vec2 u_pattern_br_b;uniform vec2 u_texsize;uniform float u_mix;uniform float u_opacity;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(u_pattern_tl_a/u_texsize,u_pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(u_pattern_tl_b/u_texsize,u_pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_mix)*u_opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_pattern_size_a;uniform vec2 u_pattern_size_b;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_scale_a;uniform float u_scale_b;uniform float u_tile_units_to_pixels;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_a*u_pattern_size_a,u_tile_units_to_pixels,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_b*u_pattern_size_b,u_tile_units_to_pixels,a_pos);}"),circle:Mp("varying vec3 v_data;varying float v_visibility;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define mediump float radius\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define highp vec4 stroke_color\n#pragma mapbox: define mediump float stroke_width\n#pragma mapbox: define lowp float stroke_opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize mediump float radius\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize highp vec4 stroke_color\n#pragma mapbox: initialize mediump float stroke_width\n#pragma mapbox: initialize lowp float stroke_opacity\nvec2 extrude=v_data.xy;float extrude_length=length(extrude);lowp float antialiasblur=v_data.z;float antialiased_blur=-max(blur,antialiasblur);float opacity_t=smoothstep(0.0,antialiased_blur,extrude_length-1.0);float color_t=stroke_width < 0.01 ? 0.0 : smoothstep(antialiased_blur,0.0,extrude_length-radius/(radius+stroke_width));gl_FragColor=v_visibility*opacity_t*mix(color*opacity,stroke_color*stroke_opacity,color_t);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform bool u_scale_with_map;uniform bool u_pitch_with_map;uniform vec2 u_extrude_scale;uniform lowp float u_device_pixel_ratio;uniform highp float u_camera_to_center_distance;attribute vec2 a_pos;varying vec3 v_data;varying float v_visibility;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define mediump float radius\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define highp vec4 stroke_color\n#pragma mapbox: define mediump float stroke_width\n#pragma mapbox: define lowp float stroke_opacity\nvoid main(void) {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize mediump float radius\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize highp vec4 stroke_color\n#pragma mapbox: initialize mediump float stroke_width\n#pragma mapbox: initialize lowp float stroke_opacity\nvec2 extrude=vec2(mod(a_pos,2.0)*2.0-1.0);vec2 circle_center=floor(a_pos*0.5);float ele=get_elevation(circle_center);v_visibility=calculate_visibility(u_matrix*vec4(circle_center,ele,1.0));if (u_pitch_with_map) {vec2 corner_position=circle_center;if (u_scale_with_map) {corner_position+=extrude*(radius+stroke_width)*u_extrude_scale;} else {vec4 projected_center=u_matrix*vec4(circle_center,0,1);corner_position+=extrude*(radius+stroke_width)*u_extrude_scale*(projected_center.w/u_camera_to_center_distance);}gl_Position=u_matrix*vec4(corner_position,ele,1);} else {gl_Position=u_matrix*vec4(circle_center,ele,1);if (u_scale_with_map) {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*u_camera_to_center_distance;} else {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*gl_Position.w;}}lowp float antialiasblur=1.0/u_device_pixel_ratio/(radius+stroke_width);v_data=vec3(extrude.x,extrude.y,antialiasblur);}"),clippingMask:Mp("void main() {gl_FragColor=vec4(1.0);}","attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}"),heatmap:Mp("uniform highp float u_intensity;varying vec2 v_extrude;\n#pragma mapbox: define highp float weight\n#define GAUSS_COEF 0.3989422804014327\nvoid main() {\n#pragma mapbox: initialize highp float weight\nfloat d=-0.5*3.0*3.0*dot(v_extrude,v_extrude);float val=weight*u_intensity*GAUSS_COEF*exp(d);gl_FragColor=vec4(val,1.0,1.0,1.0);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform float u_extrude_scale;uniform float u_opacity;uniform float u_intensity;attribute vec2 a_pos;varying vec2 v_extrude;\n#pragma mapbox: define highp float weight\n#pragma mapbox: define mediump float radius\nconst highp float ZERO=1.0/255.0/16.0;\n#define GAUSS_COEF 0.3989422804014327\nvoid main(void) {\n#pragma mapbox: initialize highp float weight\n#pragma mapbox: initialize mediump float radius\nvec2 unscaled_extrude=vec2(mod(a_pos,2.0)*2.0-1.0);float S=sqrt(-2.0*log(ZERO/weight/u_intensity/GAUSS_COEF))/3.0;v_extrude=S*unscaled_extrude;vec2 extrude=v_extrude*radius*u_extrude_scale;vec4 pos=vec4(floor(a_pos*0.5)+extrude,0,1);gl_Position=u_matrix*pos;}"),heatmapTexture:Mp("uniform sampler2D u_image;uniform sampler2D u_color_ramp;uniform float u_opacity;varying vec2 v_pos;void main() {float t=texture2D(u_image,v_pos).r;vec4 color=texture2D(u_color_ramp,vec2(t,0.5));gl_FragColor=color*u_opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(0.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_world;attribute vec2 a_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos*u_world,0,1);v_pos.x=a_pos.x;v_pos.y=1.0-a_pos.y;}"),collisionBox:Mp("varying float v_placed;varying float v_notUsed;void main() {float alpha=0.5;gl_FragColor=vec4(1.0,0.0,0.0,1.0)*alpha;if (v_placed > 0.5) {gl_FragColor=vec4(0.0,0.0,1.0,0.5)*alpha;}if (v_notUsed > 0.5) {gl_FragColor*=.1;}}","attribute vec2 a_pos;attribute vec2 a_anchor_pos;attribute vec2 a_extrude;attribute vec2 a_placed;attribute vec2 a_shift;uniform mat4 u_matrix;uniform vec2 u_extrude_scale;uniform float u_camera_to_center_distance;varying float v_placed;varying float v_notUsed;void main() {vec4 projectedPoint=u_matrix*vec4(a_anchor_pos,0,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);gl_Position=u_matrix*vec4(a_pos,get_elevation(a_pos),1.0);gl_Position.xy+=(a_extrude+a_shift)*u_extrude_scale*gl_Position.w*collision_perspective_ratio;v_placed=a_placed.x;v_notUsed=a_placed.y;}"),collisionCircle:Mp("varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;void main() {float alpha=0.5*min(v_perspective_ratio,1.0);float stroke_radius=0.9*max(v_perspective_ratio,1.0);float distance_to_center=length(v_extrude);float distance_to_edge=abs(distance_to_center-v_radius);float opacity_t=smoothstep(-stroke_radius,0.0,-distance_to_edge);vec4 color=mix(vec4(0.0,0.0,1.0,0.5),vec4(1.0,0.0,0.0,1.0),v_collision);gl_FragColor=color*alpha*opacity_t;}","attribute vec2 a_pos;attribute float a_radius;attribute vec2 a_flags;uniform mat4 u_matrix;uniform mat4 u_inv_matrix;uniform vec2 u_viewport_size;uniform float u_camera_to_center_distance;varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;vec3 toTilePosition(vec2 screenPos) {vec4 rayStart=u_inv_matrix*vec4(screenPos,-1.0,1.0);vec4 rayEnd =u_inv_matrix*vec4(screenPos, 1.0,1.0);rayStart.xyz/=rayStart.w;rayEnd.xyz /=rayEnd.w;highp float t=(0.0-rayStart.z)/(rayEnd.z-rayStart.z);return mix(rayStart.xyz,rayEnd.xyz,t);}void main() {vec2 quadCenterPos=a_pos;float radius=a_radius;float collision=a_flags.x;float vertexIdx=a_flags.y;vec2 quadVertexOffset=vec2(mix(-1.0,1.0,float(vertexIdx >=2.0)),mix(-1.0,1.0,float(vertexIdx >=1.0 && vertexIdx <=2.0)));vec2 quadVertexExtent=quadVertexOffset*radius;vec3 tilePos=toTilePosition(quadCenterPos);vec4 clipPos=u_matrix*vec4(tilePos,1.0);highp float camera_to_anchor_distance=clipPos.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);float padding_factor=1.2;v_radius=radius;v_extrude=quadVertexExtent*padding_factor;v_perspective_ratio=collision_perspective_ratio;v_collision=collision;gl_Position=vec4(clipPos.xyz/clipPos.w,1.0)+vec4(quadVertexExtent*padding_factor/u_viewport_size*2.0,0.0,0.0);}"),debug:Mp("uniform highp vec4 u_color;uniform sampler2D u_overlay;varying vec2 v_uv;void main() {vec4 overlay_color=texture2D(u_overlay,v_uv);gl_FragColor=mix(u_color,overlay_color,overlay_color.a);}","attribute vec2 a_pos;varying vec2 v_uv;uniform mat4 u_matrix;uniform float u_overlay_scale;void main() {v_uv=a_pos/8192.0;gl_Position=u_matrix*vec4(a_pos*u_overlay_scale,get_elevation(a_pos),1);}"),fill:Mp("#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float opacity\ngl_FragColor=color*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","attribute vec2 a_pos;uniform mat4 u_matrix;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float opacity\ngl_Position=u_matrix*vec4(a_pos,0,1);}"),fillOutline:Mp("varying vec2 v_pos;\n#pragma mapbox: define highp vec4 outline_color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 outline_color\n#pragma mapbox: initialize lowp float opacity\nfloat dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=outline_color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","attribute vec2 a_pos;uniform mat4 u_matrix;uniform vec2 u_world;varying vec2 v_pos;\n#pragma mapbox: define highp vec4 outline_color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 outline_color\n#pragma mapbox: initialize lowp float opacity\ngl_Position=u_matrix*vec4(a_pos,0,1);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}"),fillOutlinePattern:Mp("uniform vec2 u_texsize;uniform sampler2D u_image;uniform float u_fade;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);float dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=mix(color1,color2,u_fade)*alpha*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_world;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;gl_Position=u_matrix*vec4(a_pos,0,1);vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,a_pos);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}"),fillPattern:Mp("#ifdef GL_ES\nprecision highp float;\n#endif\nuniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_fade)*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileZoomRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileZoomRatio,a_pos);}"),fillExtrusion:Mp("varying vec4 v_color;void main() {gl_FragColor=v_color;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;uniform float u_vertical_gradient;uniform lowp float u_opacity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\n#ifdef TERRAIN3D\nattribute vec2 a_centroid;\n#endif\nvarying vec4 v_color;\n#pragma mapbox: define highp float base\n#pragma mapbox: define highp float height\n#pragma mapbox: define highp vec4 color\nvoid main() {\n#pragma mapbox: initialize highp float base\n#pragma mapbox: initialize highp float height\n#pragma mapbox: initialize highp vec4 color\nvec3 normal=a_normal_ed.xyz;\n#ifdef TERRAIN3D\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\n#else\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\n#endif\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);gl_Position=u_matrix*vec4(a_pos,t > 0.0 ? height : base,1);float colorvalue=color.r*0.2126+color.g*0.7152+color.b*0.0722;v_color=vec4(0.0,0.0,0.0,1.0);vec4 ambientlight=vec4(0.03,0.03,0.03,1.0);color+=ambientlight;float directional=clamp(dot(normal/16384.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((1.0-colorvalue+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_color.r+=clamp(color.r*directional*u_lightcolor.r,mix(0.0,0.3,1.0-u_lightcolor.r),1.0);v_color.g+=clamp(color.g*directional*u_lightcolor.g,mix(0.0,0.3,1.0-u_lightcolor.g),1.0);v_color.b+=clamp(color.b*directional*u_lightcolor.b,mix(0.0,0.3,1.0-u_lightcolor.b),1.0);v_color*=u_opacity;}"),fillExtrusionPattern:Mp("uniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\n#pragma mapbox: define lowp float base\n#pragma mapbox: define lowp float height\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float base\n#pragma mapbox: initialize lowp float height\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);vec4 mixedColor=mix(color1,color2,u_fade);gl_FragColor=mixedColor*v_lighting;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_height_factor;uniform vec3 u_scale;uniform float u_vertical_gradient;uniform lowp float u_opacity;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\n#ifdef TERRAIN3D\nattribute vec2 a_centroid;\n#endif\nvarying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\n#pragma mapbox: define lowp float base\n#pragma mapbox: define lowp float height\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float base\n#pragma mapbox: initialize lowp float height\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec3 normal=a_normal_ed.xyz;float edgedistance=a_normal_ed.w;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;\n#ifdef TERRAIN3D\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\n#else\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\n#endif\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);float z=t > 0.0 ? height : base;gl_Position=u_matrix*vec4(a_pos,z,1);vec2 pos=normal.x==1.0 && normal.y==0.0 && normal.z==16384.0\n? a_pos\n: vec2(edgedistance,z*u_height_factor);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,pos);v_lighting=vec4(0.0,0.0,0.0,1.0);float directional=clamp(dot(normal/16383.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((0.5+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_lighting.rgb+=clamp(directional*u_lightcolor,mix(vec3(0.0),vec3(0.3),1.0-u_lightcolor),vec3(1.0));v_lighting*=u_opacity;}"),hillshadePrepare:Mp("#ifdef GL_ES\nprecision highp float;\n#endif\nuniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_dimension;uniform float u_zoom;uniform vec4 u_unpack;float getElevation(vec2 coord,float bias) {vec4 data=texture2D(u_image,coord)*255.0;data.a=-1.0;return dot(data,u_unpack)/4.0;}void main() {vec2 epsilon=1.0/u_dimension;float a=getElevation(v_pos+vec2(-epsilon.x,-epsilon.y),0.0);float b=getElevation(v_pos+vec2(0,-epsilon.y),0.0);float c=getElevation(v_pos+vec2(epsilon.x,-epsilon.y),0.0);float d=getElevation(v_pos+vec2(-epsilon.x,0),0.0);float e=getElevation(v_pos,0.0);float f=getElevation(v_pos+vec2(epsilon.x,0),0.0);float g=getElevation(v_pos+vec2(-epsilon.x,epsilon.y),0.0);float h=getElevation(v_pos+vec2(0,epsilon.y),0.0);float i=getElevation(v_pos+vec2(epsilon.x,epsilon.y),0.0);float exaggerationFactor=u_zoom < 2.0 ? 0.4 : u_zoom < 4.5 ? 0.35 : 0.3;float exaggeration=u_zoom < 15.0 ? (u_zoom-15.0)*exaggerationFactor : 0.0;vec2 deriv=vec2((c+f+f+i)-(a+d+d+g),(g+h+h+i)-(a+b+b+c))/pow(2.0,exaggeration+(19.2562-u_zoom));gl_FragColor=clamp(vec4(deriv.x/2.0+0.5,deriv.y/2.0+0.5,1.0,1.0),0.0,1.0);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_dimension;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);highp vec2 epsilon=1.0/u_dimension;float scale=(u_dimension.x-2.0)/u_dimension.x;v_pos=(a_texture_pos/8192.0)*scale+epsilon;}"),hillshade:Mp("uniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_latrange;uniform vec2 u_light;uniform vec4 u_shadow;uniform vec4 u_highlight;uniform vec4 u_accent;\n#define PI 3.141592653589793\nvoid main() {vec4 pixel=texture2D(u_image,v_pos);vec2 deriv=((pixel.rg*2.0)-1.0);float scaleFactor=cos(radians((u_latrange[0]-u_latrange[1])*(1.0-v_pos.y)+u_latrange[1]));float slope=atan(1.25*length(deriv)/scaleFactor);float aspect=deriv.x !=0.0 ? atan(deriv.y,-deriv.x) : PI/2.0*(deriv.y > 0.0 ? 1.0 :-1.0);float intensity=u_light.x;float azimuth=u_light.y+PI;float base=1.875-intensity*1.75;float maxValue=0.5*PI;float scaledSlope=intensity !=0.5 ? ((pow(base,slope)-1.0)/(pow(base,maxValue)-1.0))*maxValue : slope;float accent=cos(scaledSlope);vec4 accent_color=(1.0-accent)*u_accent*clamp(intensity*2.0,0.0,1.0);float shade=abs(mod((aspect+azimuth)/PI+0.5,2.0)-1.0);vec4 shade_color=mix(u_shadow,u_highlight,shade)*sin(scaledSlope)*clamp(intensity*2.0,0.0,1.0);gl_FragColor=accent_color*(1.0-shade_color.a)+shade_color;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos=a_texture_pos/8192.0;}"),line:Mp("uniform lowp float u_device_pixel_ratio;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);gl_FragColor=color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","\n#define scale 0.015873016\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform vec2 u_units_to_pixels;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp float v_linesofar;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float width\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float width\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;v_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*2.0;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_width2=vec2(outset,inset);}"),lineGradient:Mp("uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;varying highp vec2 v_uv;\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);vec4 color=texture2D(u_image,v_uv);gl_FragColor=color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","\n#define scale 0.015873016\nattribute vec2 a_pos_normal;attribute vec4 a_data;attribute float a_uv_x;attribute float a_split_index;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_units_to_pixels;uniform float u_image_height;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp vec2 v_uv;\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float width\nvoid main() {\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float width\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;highp float texel_height=1.0/u_image_height;highp float half_texel_height=0.5*texel_height;v_uv=vec2(a_uv_x,a_split_index*texel_height-half_texel_height);vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_width2=vec2(outset,inset);}"),linePattern:Mp("#ifdef GL_ES\nprecision highp float;\n#endif\nuniform lowp float u_device_pixel_ratio;uniform vec2 u_texsize;uniform float u_fade;uniform mediump vec3 u_scale;uniform sampler2D u_image;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;vec2 pattern_size_a=vec2(display_size_a.x*fromScale/tileZoomRatio,display_size_a.y);vec2 pattern_size_b=vec2(display_size_b.x*toScale/tileZoomRatio,display_size_b.y);float aspect_a=display_size_a.y/v_width;float aspect_b=display_size_b.y/v_width;float dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float x_a=mod(v_linesofar/pattern_size_a.x*aspect_a,1.0);float x_b=mod(v_linesofar/pattern_size_b.x*aspect_b,1.0);float y=0.5*v_normal.y+0.5;vec2 texel_size=1.0/u_texsize;vec2 pos_a=mix(pattern_tl_a*texel_size-texel_size,pattern_br_a*texel_size+texel_size,vec2(x_a,y));vec2 pos_b=mix(pattern_tl_b*texel_size-texel_size,pattern_br_b*texel_size+texel_size,vec2(x_b,y));vec4 color=mix(texture2D(u_image,pos_a),texture2D(u_image,pos_b),u_fade);gl_FragColor=color*alpha*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","\n#define scale 0.015873016\n#define LINE_DISTANCE_SCALE 2.0\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform vec2 u_units_to_pixels;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define mediump float width\n#pragma mapbox: define lowp float floorwidth\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize mediump float width\n#pragma mapbox: initialize lowp float floorwidth\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_linesofar=a_linesofar;v_width2=vec2(outset,inset);v_width=floorwidth;}"),lineSDF:Mp("uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;uniform float u_sdfgamma;uniform float u_mix;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float width\n#pragma mapbox: define lowp float floorwidth\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float width\n#pragma mapbox: initialize lowp float floorwidth\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float sdfdist_a=texture2D(u_image,v_tex_a).a;float sdfdist_b=texture2D(u_image,v_tex_b).a;float sdfdist=mix(sdfdist_a,sdfdist_b,u_mix);alpha*=smoothstep(0.5-u_sdfgamma/floorwidth,0.5+u_sdfgamma/floorwidth,sdfdist);gl_FragColor=color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","\n#define scale 0.015873016\n#define LINE_DISTANCE_SCALE 2.0\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_patternscale_a;uniform float u_tex_y_a;uniform vec2 u_patternscale_b;uniform float u_tex_y_b;uniform vec2 u_units_to_pixels;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float width\n#pragma mapbox: define lowp float floorwidth\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float width\n#pragma mapbox: initialize lowp float floorwidth\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_tex_a=vec2(a_linesofar*u_patternscale_a.x/floorwidth,normal.y*u_patternscale_a.y+u_tex_y_a);v_tex_b=vec2(a_linesofar*u_patternscale_b.x/floorwidth,normal.y*u_patternscale_b.y+u_tex_y_b);v_width2=vec2(outset,inset);}"),raster:Mp("uniform float u_fade_t;uniform float u_opacity;uniform sampler2D u_image0;uniform sampler2D u_image1;varying vec2 v_pos0;varying vec2 v_pos1;uniform float u_brightness_low;uniform float u_brightness_high;uniform float u_saturation_factor;uniform float u_contrast_factor;uniform vec3 u_spin_weights;void main() {vec4 color0=texture2D(u_image0,v_pos0);vec4 color1=texture2D(u_image1,v_pos1);if (color0.a > 0.0) {color0.rgb=color0.rgb/color0.a;}if (color1.a > 0.0) {color1.rgb=color1.rgb/color1.a;}vec4 color=mix(color0,color1,u_fade_t);color.a*=u_opacity;vec3 rgb=color.rgb;rgb=vec3(dot(rgb,u_spin_weights.xyz),dot(rgb,u_spin_weights.zxy),dot(rgb,u_spin_weights.yzx));float average=(color.r+color.g+color.b)/3.0;rgb+=(average-rgb)*u_saturation_factor;rgb=(rgb-0.5)*u_contrast_factor+0.5;vec3 u_high_vec=vec3(u_brightness_low,u_brightness_low,u_brightness_low);vec3 u_low_vec=vec3(u_brightness_high,u_brightness_high,u_brightness_high);gl_FragColor=vec4(mix(u_high_vec,u_low_vec,rgb)*color.a,color.a);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_tl_parent;uniform float u_scale_parent;uniform float u_buffer_scale;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos0;varying vec2 v_pos1;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos0=(((a_texture_pos/8192.0)-0.5)/u_buffer_scale )+0.5;v_pos1=(v_pos0*u_scale_parent)+u_tl_parent;}"),symbolIcon:Mp("uniform sampler2D u_texture;varying vec2 v_tex;varying float v_fade_opacity;\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\nlowp float alpha=opacity*v_fade_opacity;gl_FragColor=texture2D(u_texture,v_tex)*alpha;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform highp float u_camera_to_center_distance;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform float u_fade_change;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform vec2 u_texsize;varying vec2 v_tex;varying float v_fade_opacity;\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;vec2 a_minFontScale=a_pixeloffset.zw/256.0;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\ncamera_to_anchor_distance/u_camera_to_center_distance :\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*max(a_minFontScale,fontScale)+a_pxoffset/16.0),z,1.0);v_tex=a_tex/u_texsize;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float visibility=calculate_visibility(projectedPoint);v_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));}"),symbolSDF:Mp("#define SDF_PX 8.0\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;uniform bool u_is_text;varying vec2 v_data0;varying vec3 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nfloat EDGE_GAMMA=0.105/u_device_pixel_ratio;vec2 tex=v_data0.xy;float gamma_scale=v_data1.x;float size=v_data1.y;float fade_opacity=v_data1[2];float fontScale=u_is_text ? size/24.0 : size;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float inner_edge=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);inner_edge=inner_edge+gamma*gamma_scale;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(inner_edge-gamma_scaled,inner_edge+gamma_scaled,dist);if (u_is_halo) {lowp float halo_edge=(6.0-halo_width/fontScale)/SDF_PX;alpha=min(smoothstep(halo_edge-gamma_scaled,halo_edge+gamma_scaled,dist),1.0-alpha);}gl_FragColor=color*(alpha*opacity*fade_opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;varying vec2 v_data0;varying vec3 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\ncamera_to_anchor_distance/u_camera_to_center_distance :\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale+a_pxoffset),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0=a_tex/u_texsize;v_data1=vec3(gamma_scale,size,interpolated_fade_opacity);}"),symbolTextAndIcon:Mp("#define SDF_PX 8.0\n#define SDF 1.0\n#define ICON 0.0\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform sampler2D u_texture_icon;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;varying vec4 v_data0;varying vec4 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nfloat fade_opacity=v_data1[2];if (v_data1.w==ICON) {vec2 tex_icon=v_data0.zw;lowp float alpha=opacity*fade_opacity;gl_FragColor=texture2D(u_texture_icon,tex_icon)*alpha;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\nreturn;}vec2 tex=v_data0.xy;float EDGE_GAMMA=0.105/u_device_pixel_ratio;float gamma_scale=v_data1.x;float size=v_data1.y;float fontScale=size/24.0;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float buff=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);buff=(6.0-halo_width/fontScale)/SDF_PX;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(buff-gamma_scaled,buff+gamma_scaled,dist);gl_FragColor=color*(alpha*opacity*fade_opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;uniform vec2 u_texsize_icon;varying vec4 v_data0;varying vec4 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);float is_sdf=a_size[0]-2.0*a_size_min;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\ncamera_to_anchor_distance/u_camera_to_center_distance :\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=size/24.0;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0.xy=a_tex/u_texsize;v_data0.zw=a_tex/u_texsize_icon;v_data1=vec4(gamma_scale,size,interpolated_fade_opacity,is_sdf);}"),terrain:Mp("uniform sampler2D u_texture;varying vec2 v_texture_pos;void main() {gl_FragColor=texture2D(u_texture,v_texture_pos);}",Cp),terrainDepth:Mp("varying float v_depth;const highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitMsk=vec4(0.,vec3(1./256.0));highp vec4 pack(highp float value) {highp vec4 comp=fract(value*bitSh);comp-=comp.xxyz*bitMsk;return comp;}void main() {gl_FragColor=pack(v_depth);}",Cp),terrainCoords:Mp("precision mediump float;uniform sampler2D u_texture;uniform float u_terrain_coords_id;varying vec2 v_texture_pos;void main() {vec4 rgba=texture2D(u_texture,v_texture_pos);gl_FragColor=vec4(rgba.r,rgba.g,rgba.b,u_terrain_coords_id);}",Cp)};function Mp(t,e){const i=/#pragma mapbox: ([\w]+) ([\w]+) ([\w]+) ([\w]+)/g,r=e.match(/attribute ([\w]+) ([\w]+)/g),s=t.match(/uniform ([\w]+) ([\w]+)([\s]*)([\w]*)/g),n=e.match(/uniform ([\w]+) ([\w]+)([\s]*)([\w]*)/g),a=n?n.concat(s):s,o={};return{fragmentSource:t=t.replace(i,((t,e,i,r,s)=>(o[s]=!0,"define"===e?`\n#ifndef HAS_UNIFORM_u_${s}\nvarying ${i} ${r} ${s};\n#else\nuniform ${i} ${r} u_${s};\n#endif\n`:`\n#ifdef HAS_UNIFORM_u_${s}\n ${i} ${r} ${s} = u_${s};\n#endif\n`))),vertexSource:e=e.replace(i,((t,e,i,r,s)=>{const n="float"===r?"vec2":"vec4",a=s.match(/color/)?"color":n;return o[s]?"define"===e?`\n#ifndef HAS_UNIFORM_u_${s}\nuniform lowp float u_${s}_t;\nattribute ${i} ${n} a_${s};\nvarying ${i} ${r} ${s};\n#else\nuniform ${i} ${r} u_${s};\n#endif\n`:"vec4"===a?`\n#ifndef HAS_UNIFORM_u_${s}\n ${s} = a_${s};\n#else\n ${i} ${r} ${s} = u_${s};\n#endif\n`:`\n#ifndef HAS_UNIFORM_u_${s}\n ${s} = unpack_mix_${a}(a_${s}, u_${s}_t);\n#else\n ${i} ${r} ${s} = u_${s};\n#endif\n`:"define"===e?`\n#ifndef HAS_UNIFORM_u_${s}\nuniform lowp float u_${s}_t;\nattribute ${i} ${n} a_${s};\n#else\nuniform ${i} ${r} u_${s};\n#endif\n`:"vec4"===a?`\n#ifndef HAS_UNIFORM_u_${s}\n ${i} ${r} ${s} = a_${s};\n#else\n ${i} ${r} ${s} = u_${s};\n#endif\n`:`\n#ifndef HAS_UNIFORM_u_${s}\n ${i} ${r} ${s} = unpack_mix_${a}(a_${s}, u_${s}_t);\n#else\n ${i} ${r} ${s} = u_${s};\n#endif\n`})),staticAttributes:r,staticUniforms:a}}class Pp{constructor(){this.boundProgram=null,this.boundLayoutVertexBuffer=null,this.boundPaintVertexBuffers=[],this.boundIndexBuffer=null,this.boundVertexOffset=null,this.boundDynamicVertexBuffer=null,this.vao=null}bind(t,e,i,r,s,n,a,o,l){this.context=t;let c=this.boundPaintVertexBuffers.length!==r.length;for(let t=0;!c&&t({u_depth:new ia(t,e.u_depth),u_terrain:new ia(t,e.u_terrain),u_terrain_dim:new ra(t,e.u_terrain_dim),u_terrain_matrix:new ca(t,e.u_terrain_matrix),u_terrain_unpack:new aa(t,e.u_terrain_unpack),u_terrain_exaggeration:new ra(t,e.u_terrain_exaggeration)}))(t,v),this.binderUniforms=i?i.getUniforms(t,v):[]}draw(t,e,i,r,s,n,a,o,l,c,h,u,p,d,m,f,_,g){const y=t.gl;if(this.failedToCreate)return;if(t.program.set(this.program),t.setDepthMode(i),t.setStencilMode(r),t.setColorMode(s),t.setCullFace(n),o){t.activeTexture.set(y.TEXTURE2),y.bindTexture(y.TEXTURE_2D,o.depthTexture),t.activeTexture.set(y.TEXTURE3),y.bindTexture(y.TEXTURE_2D,o.texture);for(const t in this.terrainUniforms)this.terrainUniforms[t].set(o[t])}for(const t in this.fixedUniforms)this.fixedUniforms[t].set(a[t]);m&&m.setUniforms(t,this.binderUniforms,p,{zoom:d});let x=0;switch(e){case y.LINES:x=2;break;case y.TRIANGLES:x=3;break;case y.LINE_STRIP:x=1}for(const i of u.get()){const r=i.vaos||(i.vaos={});(r[l]||(r[l]=new Pp)).bind(t,this,c,m?m.getPaintVertexBuffers():[],h,i.vertexOffset,f,_,g),y.drawElements(e,i.primitiveLength*x,y.UNSIGNED_SHORT,i.primitiveOffset*x*2)}}}function Lp(t,e,i){const r=1/$u(i,1,e.transform.tileZoom),s=Math.pow(2,i.tileID.overscaledZ),n=i.tileSize*Math.pow(2,e.transform.tileZoom)/s,a=n*(i.tileID.canonical.x+i.tileID.wrap*s),o=n*i.tileID.canonical.y;return{u_image:0,u_texsize:i.imageAtlasTexture.size,u_scale:[r,t.fromScale,t.toScale],u_fade:t.t,u_pixel_coord_upper:[a>>16,o>>16],u_pixel_coord_lower:[65535&a,65535&o]}}const Rp=(t,e,i,r)=>{const s=e.style.light,n=s.properties.get("position"),a=[n.x,n.y,n.z],o=function(){var t=new ja(9);return ja!=Float32Array&&(t[1]=0,t[2]=0,t[3]=0,t[5]=0,t[6]=0,t[7]=0),t[0]=1,t[4]=1,t[8]=1,t}();"viewport"===s.properties.get("anchor")&&function(t,e){var i=Math.sin(e),r=Math.cos(e);t[0]=r,t[1]=i,t[2]=0,t[3]=-i,t[4]=r,t[5]=0,t[6]=0,t[7]=0,t[8]=1}(o,-e.transform.angle),function(t,e,i){var r=e[0],s=e[1],n=e[2];t[0]=r*i[0]+s*i[3]+n*i[6],t[1]=r*i[1]+s*i[4]+n*i[7],t[2]=r*i[2]+s*i[5]+n*i[8]}(a,a,o);const l=s.properties.get("color");return{u_matrix:t,u_lightpos:a,u_lightintensity:s.properties.get("intensity"),u_lightcolor:[l.r,l.g,l.b],u_vertical_gradient:+i,u_opacity:r}},Bp=(t,e,i,r,s,n,a)=>g(Rp(t,e,i,r),Lp(n,e,a),{u_height_factor:-Math.pow(2,s.overscaledZ)/a.tileSize/8}),Fp=t=>({u_matrix:t}),Op=(t,e,i,r)=>g(Fp(t),Lp(i,e,r)),Vp=(t,e)=>({u_matrix:t,u_world:e}),Up=(t,e,i,r,s)=>g(Op(t,e,i,r),{u_world:s}),Np=(t,e,i,r)=>{const s=t.transform;let n,a;if("map"===r.paint.get("circle-pitch-alignment")){const t=$u(i,1,s.zoom);n=!0,a=[t,t]}else n=!1,a=s.pixelsToGLUnits;return{u_camera_to_center_distance:s.cameraToCenterDistance,u_scale_with_map:+("map"===r.paint.get("circle-pitch-scale")),u_matrix:t.translatePosMatrix(e.posMatrix,i,r.paint.get("circle-translate"),r.paint.get("circle-translate-anchor")),u_pitch_with_map:+n,u_device_pixel_ratio:t.pixelRatio,u_extrude_scale:a}},$p=(t,e,i)=>{const r=$u(i,1,e.zoom),s=Math.pow(2,e.zoom-i.tileID.overscaledZ),n=i.tileID.overscaleFactor();return{u_matrix:t,u_camera_to_center_distance:e.cameraToCenterDistance,u_pixels_to_tile_units:r,u_extrude_scale:[e.pixelsToGLUnits[0]/(r*s),e.pixelsToGLUnits[1]/(r*s)],u_overscale_factor:n}},qp=(t,e,i=1)=>({u_matrix:t,u_color:e,u_overlay:0,u_overlay_scale:i}),Zp=t=>({u_matrix:t}),jp=(t,e,i,r)=>({u_matrix:t,u_extrude_scale:$u(e,1,i),u_intensity:r});function Gp(t,e){const i=Math.pow(2,e.canonical.z),r=e.canonical.y;return[new Bh(0,r/i).toLngLat().lat,new Bh(0,(r+1)/i).toLngLat().lat]}const Hp=(t,e,i,r)=>{const s=t.transform;return{u_matrix:Jp(t,e,i,r),u_ratio:1/$u(e,1,s.zoom),u_device_pixel_ratio:t.pixelRatio,u_units_to_pixels:[1/s.pixelsToGLUnits[0],1/s.pixelsToGLUnits[1]]}},Wp=(t,e,i,r,s)=>g(Hp(t,e,i,s),{u_image:0,u_image_height:r}),Xp=(t,e,i,r,s)=>{const n=t.transform,a=Yp(e,n);return{u_matrix:Jp(t,e,i,s),u_texsize:e.imageAtlasTexture.size,u_ratio:1/$u(e,1,n.zoom),u_device_pixel_ratio:t.pixelRatio,u_image:0,u_scale:[a,r.fromScale,r.toScale],u_fade:r.t,u_units_to_pixels:[1/n.pixelsToGLUnits[0],1/n.pixelsToGLUnits[1]]}},Kp=(t,e,i,r,s,n)=>{const a=t.lineAtlas,o=Yp(e,t.transform),l="round"===i.layout.get("line-cap"),c=a.getDash(r.from,l),h=a.getDash(r.to,l),u=c.width*s.fromScale,p=h.width*s.toScale;return g(Hp(t,e,i,n),{u_patternscale_a:[o/u,-c.height/2],u_patternscale_b:[o/p,-h.height/2],u_sdfgamma:a.width/(256*Math.min(u,p)*t.pixelRatio)/2,u_image:0,u_tex_y_a:c.y,u_tex_y_b:h.y,u_mix:s.t})};function Yp(t,e){return 1/$u(t,1,e.tileZoom)}function Jp(t,e,i,r){return t.translatePosMatrix(r?r.posMatrix:e.tileID.posMatrix,e,i.paint.get("line-translate"),i.paint.get("line-translate-anchor"))}const Qp=(t,e,i,r,s)=>{return{u_matrix:t,u_tl_parent:e,u_scale_parent:i,u_buffer_scale:1,u_fade_t:r.mix,u_opacity:r.opacity*s.paint.get("raster-opacity"),u_image0:0,u_image1:1,u_brightness_low:s.paint.get("raster-brightness-min"),u_brightness_high:s.paint.get("raster-brightness-max"),u_saturation_factor:(a=s.paint.get("raster-saturation"),a>0?1-1/(1.001-a):-a),u_contrast_factor:(n=s.paint.get("raster-contrast"),n>0?1/(1-n):1+n),u_spin_weights:td(s.paint.get("raster-hue-rotate"))};var n,a};function td(t){t*=Math.PI/180;const e=Math.sin(t),i=Math.cos(t);return[(2*i+1)/3,(-Math.sqrt(3)*e-i+1)/3,(Math.sqrt(3)*e-i+1)/3]}const ed=(t,e,i,r,s,n,a,o,l,c)=>{const h=s.transform;return{u_is_size_zoom_constant:+("constant"===t||"source"===t),u_is_size_feature_constant:+("constant"===t||"camera"===t),u_size_t:e?e.uSizeT:0,u_size:e?e.uSize:0,u_camera_to_center_distance:h.cameraToCenterDistance,u_pitch:h.pitch/360*2*Math.PI,u_rotate_symbol:+i,u_aspect_ratio:h.width/h.height,u_fade_change:s.options.fadeDuration?s.symbolFadeChange:1,u_matrix:n,u_label_plane_matrix:a,u_coord_matrix:o,u_is_text:+l,u_pitch_with_map:+r,u_texsize:c,u_texture:0}},id=(t,e,i,r,s,n,a,o,l,c,h)=>{const u=s.transform;return g(ed(t,e,i,r,s,n,a,o,l,c),{u_gamma_scale:r?Math.cos(u._pitch)*u.cameraToCenterDistance:1,u_device_pixel_ratio:s.pixelRatio,u_is_halo:+h})},rd=(t,e,i,r,s,n,a,o,l,c)=>g(id(t,e,i,r,s,n,a,o,!0,l,!0),{u_texsize_icon:c,u_texture_icon:1}),sd=(t,e,i)=>({u_matrix:t,u_opacity:e,u_color:i}),nd=(t,e,i,r,s,n)=>g(function(t,e,i,r){const s=i.imageManager.getPattern(t.from.toString()),n=i.imageManager.getPattern(t.to.toString()),{width:a,height:o}=i.imageManager.getPixelSize(),l=Math.pow(2,r.tileID.overscaledZ),c=r.tileSize*Math.pow(2,i.transform.tileZoom)/l,h=c*(r.tileID.canonical.x+r.tileID.wrap*l),u=c*r.tileID.canonical.y;return{u_image:0,u_pattern_tl_a:s.tl,u_pattern_br_a:s.br,u_pattern_tl_b:n.tl,u_pattern_br_b:n.br,u_texsize:[a,o],u_mix:e.t,u_pattern_size_a:s.displaySize,u_pattern_size_b:n.displaySize,u_scale_a:e.fromScale,u_scale_b:e.toScale,u_tile_units_to_pixels:1/$u(r,1,i.transform.tileZoom),u_pixel_coord_upper:[h>>16,u>>16],u_pixel_coord_lower:[65535&h,65535&u]}}(r,n,i,s),{u_matrix:t,u_opacity:e}),ad={fillExtrusion:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_lightpos:new na(t,e.u_lightpos),u_lightintensity:new ra(t,e.u_lightintensity),u_lightcolor:new na(t,e.u_lightcolor),u_vertical_gradient:new ra(t,e.u_vertical_gradient),u_opacity:new ra(t,e.u_opacity)}),fillExtrusionPattern:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_lightpos:new na(t,e.u_lightpos),u_lightintensity:new ra(t,e.u_lightintensity),u_lightcolor:new na(t,e.u_lightcolor),u_vertical_gradient:new ra(t,e.u_vertical_gradient),u_height_factor:new ra(t,e.u_height_factor),u_image:new ia(t,e.u_image),u_texsize:new sa(t,e.u_texsize),u_pixel_coord_upper:new sa(t,e.u_pixel_coord_upper),u_pixel_coord_lower:new sa(t,e.u_pixel_coord_lower),u_scale:new na(t,e.u_scale),u_fade:new ra(t,e.u_fade),u_opacity:new ra(t,e.u_opacity)}),fill:(t,e)=>({u_matrix:new ca(t,e.u_matrix)}),fillPattern:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_image:new ia(t,e.u_image),u_texsize:new sa(t,e.u_texsize),u_pixel_coord_upper:new sa(t,e.u_pixel_coord_upper),u_pixel_coord_lower:new sa(t,e.u_pixel_coord_lower),u_scale:new na(t,e.u_scale),u_fade:new ra(t,e.u_fade)}),fillOutline:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_world:new sa(t,e.u_world)}),fillOutlinePattern:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_world:new sa(t,e.u_world),u_image:new ia(t,e.u_image),u_texsize:new sa(t,e.u_texsize),u_pixel_coord_upper:new sa(t,e.u_pixel_coord_upper),u_pixel_coord_lower:new sa(t,e.u_pixel_coord_lower),u_scale:new na(t,e.u_scale),u_fade:new ra(t,e.u_fade)}),circle:(t,e)=>({u_camera_to_center_distance:new ra(t,e.u_camera_to_center_distance),u_scale_with_map:new ia(t,e.u_scale_with_map),u_pitch_with_map:new ia(t,e.u_pitch_with_map),u_extrude_scale:new sa(t,e.u_extrude_scale),u_device_pixel_ratio:new ra(t,e.u_device_pixel_ratio),u_matrix:new ca(t,e.u_matrix)}),collisionBox:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_camera_to_center_distance:new ra(t,e.u_camera_to_center_distance),u_pixels_to_tile_units:new ra(t,e.u_pixels_to_tile_units),u_extrude_scale:new sa(t,e.u_extrude_scale),u_overscale_factor:new ra(t,e.u_overscale_factor)}),collisionCircle:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_inv_matrix:new ca(t,e.u_inv_matrix),u_camera_to_center_distance:new ra(t,e.u_camera_to_center_distance),u_viewport_size:new sa(t,e.u_viewport_size)}),debug:(t,e)=>({u_color:new oa(t,e.u_color),u_matrix:new ca(t,e.u_matrix),u_overlay:new ia(t,e.u_overlay),u_overlay_scale:new ra(t,e.u_overlay_scale)}),clippingMask:(t,e)=>({u_matrix:new ca(t,e.u_matrix)}),heatmap:(t,e)=>({u_extrude_scale:new ra(t,e.u_extrude_scale),u_intensity:new ra(t,e.u_intensity),u_matrix:new ca(t,e.u_matrix)}),heatmapTexture:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_world:new sa(t,e.u_world),u_image:new ia(t,e.u_image),u_color_ramp:new ia(t,e.u_color_ramp),u_opacity:new ra(t,e.u_opacity)}),hillshade:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_image:new ia(t,e.u_image),u_latrange:new sa(t,e.u_latrange),u_light:new sa(t,e.u_light),u_shadow:new oa(t,e.u_shadow),u_highlight:new oa(t,e.u_highlight),u_accent:new oa(t,e.u_accent)}),hillshadePrepare:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_image:new ia(t,e.u_image),u_dimension:new sa(t,e.u_dimension),u_zoom:new ra(t,e.u_zoom),u_unpack:new aa(t,e.u_unpack)}),line:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_ratio:new ra(t,e.u_ratio),u_device_pixel_ratio:new ra(t,e.u_device_pixel_ratio),u_units_to_pixels:new sa(t,e.u_units_to_pixels)}),lineGradient:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_ratio:new ra(t,e.u_ratio),u_device_pixel_ratio:new ra(t,e.u_device_pixel_ratio),u_units_to_pixels:new sa(t,e.u_units_to_pixels),u_image:new ia(t,e.u_image),u_image_height:new ra(t,e.u_image_height)}),linePattern:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_texsize:new sa(t,e.u_texsize),u_ratio:new ra(t,e.u_ratio),u_device_pixel_ratio:new ra(t,e.u_device_pixel_ratio),u_image:new ia(t,e.u_image),u_units_to_pixels:new sa(t,e.u_units_to_pixels),u_scale:new na(t,e.u_scale),u_fade:new ra(t,e.u_fade)}),lineSDF:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_ratio:new ra(t,e.u_ratio),u_device_pixel_ratio:new ra(t,e.u_device_pixel_ratio),u_units_to_pixels:new sa(t,e.u_units_to_pixels),u_patternscale_a:new sa(t,e.u_patternscale_a),u_patternscale_b:new sa(t,e.u_patternscale_b),u_sdfgamma:new ra(t,e.u_sdfgamma),u_image:new ia(t,e.u_image),u_tex_y_a:new ra(t,e.u_tex_y_a),u_tex_y_b:new ra(t,e.u_tex_y_b),u_mix:new ra(t,e.u_mix)}),raster:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_tl_parent:new sa(t,e.u_tl_parent),u_scale_parent:new ra(t,e.u_scale_parent),u_buffer_scale:new ra(t,e.u_buffer_scale),u_fade_t:new ra(t,e.u_fade_t),u_opacity:new ra(t,e.u_opacity),u_image0:new ia(t,e.u_image0),u_image1:new ia(t,e.u_image1),u_brightness_low:new ra(t,e.u_brightness_low),u_brightness_high:new ra(t,e.u_brightness_high),u_saturation_factor:new ra(t,e.u_saturation_factor),u_contrast_factor:new ra(t,e.u_contrast_factor),u_spin_weights:new na(t,e.u_spin_weights)}),symbolIcon:(t,e)=>({u_is_size_zoom_constant:new ia(t,e.u_is_size_zoom_constant),u_is_size_feature_constant:new ia(t,e.u_is_size_feature_constant),u_size_t:new ra(t,e.u_size_t),u_size:new ra(t,e.u_size),u_camera_to_center_distance:new ra(t,e.u_camera_to_center_distance),u_pitch:new ra(t,e.u_pitch),u_rotate_symbol:new ia(t,e.u_rotate_symbol),u_aspect_ratio:new ra(t,e.u_aspect_ratio),u_fade_change:new ra(t,e.u_fade_change),u_matrix:new ca(t,e.u_matrix),u_label_plane_matrix:new ca(t,e.u_label_plane_matrix),u_coord_matrix:new ca(t,e.u_coord_matrix),u_is_text:new ia(t,e.u_is_text),u_pitch_with_map:new ia(t,e.u_pitch_with_map),u_texsize:new sa(t,e.u_texsize),u_texture:new ia(t,e.u_texture)}),symbolSDF:(t,e)=>({u_is_size_zoom_constant:new ia(t,e.u_is_size_zoom_constant),u_is_size_feature_constant:new ia(t,e.u_is_size_feature_constant),u_size_t:new ra(t,e.u_size_t),u_size:new ra(t,e.u_size),u_camera_to_center_distance:new ra(t,e.u_camera_to_center_distance),u_pitch:new ra(t,e.u_pitch),u_rotate_symbol:new ia(t,e.u_rotate_symbol),u_aspect_ratio:new ra(t,e.u_aspect_ratio),u_fade_change:new ra(t,e.u_fade_change),u_matrix:new ca(t,e.u_matrix),u_label_plane_matrix:new ca(t,e.u_label_plane_matrix),u_coord_matrix:new ca(t,e.u_coord_matrix),u_is_text:new ia(t,e.u_is_text),u_pitch_with_map:new ia(t,e.u_pitch_with_map),u_texsize:new sa(t,e.u_texsize),u_texture:new ia(t,e.u_texture),u_gamma_scale:new ra(t,e.u_gamma_scale),u_device_pixel_ratio:new ra(t,e.u_device_pixel_ratio),u_is_halo:new ia(t,e.u_is_halo)}),symbolTextAndIcon:(t,e)=>({u_is_size_zoom_constant:new ia(t,e.u_is_size_zoom_constant),u_is_size_feature_constant:new ia(t,e.u_is_size_feature_constant),u_size_t:new ra(t,e.u_size_t),u_size:new ra(t,e.u_size),u_camera_to_center_distance:new ra(t,e.u_camera_to_center_distance),u_pitch:new ra(t,e.u_pitch),u_rotate_symbol:new ia(t,e.u_rotate_symbol),u_aspect_ratio:new ra(t,e.u_aspect_ratio),u_fade_change:new ra(t,e.u_fade_change),u_matrix:new ca(t,e.u_matrix),u_label_plane_matrix:new ca(t,e.u_label_plane_matrix),u_coord_matrix:new ca(t,e.u_coord_matrix),u_is_text:new ia(t,e.u_is_text),u_pitch_with_map:new ia(t,e.u_pitch_with_map),u_texsize:new sa(t,e.u_texsize),u_texsize_icon:new sa(t,e.u_texsize_icon),u_texture:new ia(t,e.u_texture),u_texture_icon:new ia(t,e.u_texture_icon),u_gamma_scale:new ra(t,e.u_gamma_scale),u_device_pixel_ratio:new ra(t,e.u_device_pixel_ratio),u_is_halo:new ia(t,e.u_is_halo)}),background:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_opacity:new ra(t,e.u_opacity),u_color:new oa(t,e.u_color)}),backgroundPattern:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_opacity:new ra(t,e.u_opacity),u_image:new ia(t,e.u_image),u_pattern_tl_a:new sa(t,e.u_pattern_tl_a),u_pattern_br_a:new sa(t,e.u_pattern_br_a),u_pattern_tl_b:new sa(t,e.u_pattern_tl_b),u_pattern_br_b:new sa(t,e.u_pattern_br_b),u_texsize:new sa(t,e.u_texsize),u_mix:new ra(t,e.u_mix),u_pattern_size_a:new sa(t,e.u_pattern_size_a),u_pattern_size_b:new sa(t,e.u_pattern_size_b),u_scale_a:new ra(t,e.u_scale_a),u_scale_b:new ra(t,e.u_scale_b),u_pixel_coord_upper:new sa(t,e.u_pixel_coord_upper),u_pixel_coord_lower:new sa(t,e.u_pixel_coord_lower),u_tile_units_to_pixels:new ra(t,e.u_tile_units_to_pixels)}),terrain:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_texture:new ia(t,e.u_texture),u_ele_delta:new ra(t,e.u_ele_delta)}),terrainDepth:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_ele_delta:new ra(t,e.u_ele_delta)}),terrainCoords:(t,e)=>({u_matrix:new ca(t,e.u_matrix),u_texture:new ia(t,e.u_texture),u_terrain_coords_id:new ra(t,e.u_terrain_coords_id),u_ele_delta:new ra(t,e.u_ele_delta)})};class od{constructor(t,e,i){this.context=t;const r=t.gl;this.buffer=r.createBuffer(),this.dynamicDraw=Boolean(i),this.context.unbindVAO(),t.bindElementBuffer.set(this.buffer),r.bufferData(r.ELEMENT_ARRAY_BUFFER,e.arrayBuffer,this.dynamicDraw?r.DYNAMIC_DRAW:r.STATIC_DRAW),this.dynamicDraw||delete e.arrayBuffer}bind(){this.context.bindElementBuffer.set(this.buffer)}updateData(t){const e=this.context.gl;if(!this.dynamicDraw)throw new Error("Attempted to update data while not in dynamic mode.");this.context.unbindVAO(),this.bind(),e.bufferSubData(e.ELEMENT_ARRAY_BUFFER,0,t.arrayBuffer)}destroy(){this.buffer&&(this.context.gl.deleteBuffer(this.buffer),delete this.buffer)}}const ld={Int8:"BYTE",Uint8:"UNSIGNED_BYTE",Int16:"SHORT",Uint16:"UNSIGNED_SHORT",Int32:"INT",Uint32:"UNSIGNED_INT",Float32:"FLOAT"};class cd{constructor(t,e,i,r){this.length=e.length,this.attributes=i,this.itemSize=e.bytesPerElement,this.dynamicDraw=r,this.context=t;const s=t.gl;this.buffer=s.createBuffer(),t.bindVertexBuffer.set(this.buffer),s.bufferData(s.ARRAY_BUFFER,e.arrayBuffer,this.dynamicDraw?s.DYNAMIC_DRAW:s.STATIC_DRAW),this.dynamicDraw||delete e.arrayBuffer}bind(){this.context.bindVertexBuffer.set(this.buffer)}updateData(t){if(t.length!==this.length)throw new Error(`Length of new data is ${t.length}, which doesn't match current length of ${this.length}`);const e=this.context.gl;this.bind(),e.bufferSubData(e.ARRAY_BUFFER,0,t.arrayBuffer)}enableAttributes(t,e){for(let i=0;i0){const e=Ga(),i=g;to(e,_.placementInvProjMatrix,t.transform.glCoordMatrix),to(e,e,_.placementViewportMatrix),h.push({circleArray:x,circleOffset:p,transform:i,invTransform:e,coord:m}),u+=x.length/4,p=u}y&&c.draw(o,l.LINES,Yd.disabled,Qd.disabled,t.colorModeForRenderPass(),tm.disabled,$p(g,t.transform,f),t.style.map.terrain&&t.style.map.terrain.getTerrainData(m),i.id,y.layoutVertexBuffer,y.indexBuffer,y.segments,null,t.transform.zoom,null,null,y.collisionVertexBuffer)}if(!a||!h.length)return;const d=t.useProgram("collisionCircle"),m=new Dn;m.resize(4*u),m._trim();let f=0;for(const t of h)for(let e=0;e=0&&(f[_.associatedIconIndex]={shiftedAnchor:I,angle:A})}else Bu(_.numGlyphs,d)}if(h){m.clear();const e=t.icon.placedSymbolArray;for(let t=0;tt.style.map.terrain.getElevation(o,e,i):null,r="map"===i.layout.get("text-rotation-alignment");Au(l,o.posMatrix,t,s,R,B,_,c,r,e)}const V=t.translatePosMatrix(o.posMatrix,r,n,a),U=g||s&&w||O?rm:R,N=t.translatePosMatrix(B,r,n,a,!0),$=p&&0!==i.paint.get(s?"text-halo-width":"icon-halo-width").constantOr(1);let q;q=p?l.iconsInText?rd(b.kind,A,y,_,t,V,U,N,C,k):id(b.kind,A,y,_,t,V,U,N,s,C,!0):ed(b.kind,A,y,_,t,V,U,N,s,C);const Z={program:I,buffers:h,uniformValues:q,atlasTexture:z,atlasTextureIcon:D,atlasInterpolation:M,atlasInterpolationIcon:P,isSDF:p,hasHalo:$};if(x&&l.canOverlap){v=!0;const t=h.segments.get();for(const e of t)T.push({segments:new Nn([e]),sortKey:e.sortKey,state:Z,terrainData:E})}else T.push({segments:h.segments,sortKey:0,state:Z,terrainData:E})}v&&T.sort(((t,e)=>t.sortKey-e.sortKey));for(const e of T){const r=e.state;if(p.activeTexture.set(d.TEXTURE0),r.atlasTexture.bind(r.atlasInterpolation,d.CLAMP_TO_EDGE),r.atlasTextureIcon&&(p.activeTexture.set(d.TEXTURE1),r.atlasTextureIcon&&r.atlasTextureIcon.bind(r.atlasInterpolationIcon,d.CLAMP_TO_EDGE)),r.isSDF){const s=r.uniformValues;r.hasHalo&&(s.u_is_halo=1,lm(r.buffers,e.segments,i,t,r.program,b,h,u,s,e.terrainData)),s.u_is_halo=0}lm(r.buffers,e.segments,i,t,r.program,b,h,u,r.uniformValues,e.terrainData)}}function lm(t,e,i,r,s,n,a,o,l,c){const h=r.context;s.draw(h,h.gl.TRIANGLES,n,a,o,tm.disabled,l,c,i.id,t.layoutVertexBuffer,t.indexBuffer,e,i.paint,r.transform.zoom,t.programConfigurations.get(i.id),t.dynamicLayoutVertexBuffer,t.opacityVertexBuffer)}function cm(t,e,i,r,s){if(!i||!r||!r.imageAtlas)return;const n=r.imageAtlas.patternPositions;let a=n[i.to.toString()],o=n[i.from.toString()];if(!a&&o&&(a=o),!o&&a&&(o=a),!a||!o){const t=s.getPaintProperty(e);a=n[t],o=n[t]}a&&o&&t.setConstantPatternPositions(a,o)}function hm(t,e,i,r,s,n,a){const o=t.context.gl,l="fill-pattern",c=i.paint.get(l),h=c&&c.constantOr(1),u=i.getCrossfadeParameters();let p,d,m,f,_;a?(d=h&&!i.getPaintProperty("fill-outline-color")?"fillOutlinePattern":"fillOutline",p=o.LINES):(d=h?"fillPattern":"fill",p=o.TRIANGLES);const g=c.constantOr(null);for(const c of r){const r=e.getTile(c);if(h&&!r.patternsLoaded())continue;const y=r.getBucket(i);if(!y)continue;const x=y.programConfigurations.get(i.id),v=t.useProgram(d,x),b=t.style.map.terrain&&t.style.map.terrain.getTerrainData(c);h&&(t.context.activeTexture.set(o.TEXTURE0),r.imageAtlasTexture.bind(o.LINEAR,o.CLAMP_TO_EDGE),x.updatePaintBuffers(u)),cm(x,l,g,r,i);const w=b?c:null,T=t.translatePosMatrix(w?w.posMatrix:c.posMatrix,r,i.paint.get("fill-translate"),i.paint.get("fill-translate-anchor"));if(a){f=y.indexBuffer2,_=y.segments2;const e=[o.drawingBufferWidth,o.drawingBufferHeight];m="fillOutlinePattern"===d&&h?Up(T,t,u,r,e):Vp(T,e)}else f=y.indexBuffer,_=y.segments,m=h?Op(T,t,u,r):Fp(T);v.draw(t.context,p,s,t.stencilModeForClipping(c),n,tm.disabled,m,b,i.id,y.layoutVertexBuffer,f,_,i.paint,t.transform.zoom,x)}}function um(t,e,i,r,s,n,a){const o=t.context,l=o.gl,c="fill-extrusion-pattern",h=i.paint.get(c),u=h.constantOr(1),p=i.getCrossfadeParameters(),d=i.paint.get("fill-extrusion-opacity"),m=h.constantOr(null);for(const h of r){const r=e.getTile(h),f=r.getBucket(i);if(!f)continue;const _=t.style.map.terrain&&t.style.map.terrain.getTerrainData(h),g=f.programConfigurations.get(i.id),y=t.useProgram(u?"fillExtrusionPattern":"fillExtrusion",g);u&&(t.context.activeTexture.set(l.TEXTURE0),r.imageAtlasTexture.bind(l.LINEAR,l.CLAMP_TO_EDGE),g.updatePaintBuffers(p)),cm(g,c,m,r,i);const x=t.translatePosMatrix(h.posMatrix,r,i.paint.get("fill-extrusion-translate"),i.paint.get("fill-extrusion-translate-anchor")),v=i.paint.get("fill-extrusion-vertical-gradient"),b=u?Bp(x,t,v,d,h,p,r):Rp(x,t,v,d);y.draw(o,o.gl.TRIANGLES,s,n,a,tm.backCCW,b,_,i.id,f.layoutVertexBuffer,f.indexBuffer,f.segments,i.paint,t.transform.zoom,g,t.style.map.terrain&&f.centroidVertexBuffer)}}function pm(t,e,i,r,s,n,a){const o=t.context,l=o.gl,c=i.fbo;if(!c)return;const h=t.useProgram("hillshade"),u=t.style.map.terrain&&t.style.map.terrain.getTerrainData(e);o.activeTexture.set(l.TEXTURE0),l.bindTexture(l.TEXTURE_2D,c.colorAttachment.get()),h.draw(o,l.TRIANGLES,s,n,a,tm.disabled,((t,e,i,r)=>{const s=i.paint.get("hillshade-shadow-color"),n=i.paint.get("hillshade-highlight-color"),a=i.paint.get("hillshade-accent-color");let o=i.paint.get("hillshade-illumination-direction")*(Math.PI/180);"viewport"===i.paint.get("hillshade-illumination-anchor")&&(o-=t.transform.angle);const l=!t.options.moving;return{u_matrix:r?r.posMatrix:t.transform.calculatePosMatrix(e.tileID.toUnwrapped(),l),u_image:0,u_latrange:Gp(0,e.tileID),u_light:[i.paint.get("hillshade-exaggeration"),o],u_shadow:s,u_highlight:n,u_accent:a}})(t,i,r,u?e:null),u,r.id,t.rasterBoundsBuffer,t.quadTriangleIndexBuffer,t.rasterBoundsSegments)}function dm(t,e,i,r,s,n){const a=t.context,o=a.gl,l=e.dem;if(l&&l.data){const c=l.dim,h=l.stride,u=l.getPixels();if(a.activeTexture.set(o.TEXTURE1),a.pixelStoreUnpackPremultiplyAlpha.set(!1),e.demTexture=e.demTexture||t.getTileTexture(h),e.demTexture){const t=e.demTexture;t.update(u,{premultiply:!1}),t.bind(o.NEAREST,o.CLAMP_TO_EDGE)}else e.demTexture=new uh(a,u,o.RGBA,{premultiply:!1}),e.demTexture.bind(o.NEAREST,o.CLAMP_TO_EDGE);a.activeTexture.set(o.TEXTURE0);let p=e.fbo;if(!p){const t=new uh(a,{width:c,height:c,data:null},o.RGBA);t.bind(o.LINEAR,o.CLAMP_TO_EDGE),p=e.fbo=a.createFramebuffer(c,c,!0,!1),p.colorAttachment.set(t.texture)}a.bindFramebuffer.set(p.framebuffer),a.viewport.set([0,0,c,c]),t.useProgram("hillshadePrepare").draw(a,o.TRIANGLES,r,s,n,tm.disabled,((t,e)=>{const i=e.stride,r=Ga();return Qa(r,0,va,-va,0,0,1),Ka(r,r,[0,-va,0]),{u_matrix:r,u_image:1,u_dimension:[i,i],u_zoom:t.overscaledZ,u_unpack:e.getUnpackVector()}})(e.tileID,l),null,i.id,t.rasterBoundsBuffer,t.quadTriangleIndexBuffer,t.rasterBoundsSegments),e.needsHillshadePrepare=!1}}function mm(t,e,i,r,s,n){const a=r.paint.get("raster-fade-duration");if(!n&&a>0){const r=B.now(),n=(r-t.timeAdded)/a,o=e?(r-e.timeAdded)/a:-1,l=i.getSource(),c=s.coveringZoomLevel({tileSize:l.tileSize,roundZoom:l.roundZoom}),h=!e||Math.abs(e.tileID.overscaledZ-c)>Math.abs(t.tileID.overscaledZ-c),u=h&&t.refreshedUponExpiration?1:m(h?n:1-o,0,1);return t.refreshedUponExpiration&&n>=1&&(t.refreshedUponExpiration=!1),e?{opacity:1,mix:1-u}:{opacity:u,mix:0}}return{opacity:1,mix:0}}const fm=new he(1,0,0,1),_m=new he(0,1,0,1),gm=new he(0,0,1,1),ym=new he(1,0,1,1),xm=new he(0,1,1,1);function vm(t,e,i,r){wm(t,0,e+i/2,t.transform.width,i,r)}function bm(t,e,i,r){wm(t,e-i/2,0,i,t.transform.height,r)}function wm(t,e,i,r,s,n){const a=t.context,o=a.gl;o.enable(o.SCISSOR_TEST),o.scissor(e*t.pixelRatio,i*t.pixelRatio,r*t.pixelRatio,s*t.pixelRatio),a.clear({color:n}),o.disable(o.SCISSOR_TEST)}function Tm(t,e,i){const r=t.context,s=r.gl,n=i.posMatrix,a=t.useProgram("debug"),o=Yd.disabled,l=Qd.disabled,c=t.colorModeForRenderPass(),h="$debug",u=t.style.map.terrain&&t.style.map.terrain.getTerrainData(i);r.activeTexture.set(s.TEXTURE0);const p=e.getTileByID(i.key).latestRawTileData,d=Math.floor((p&&p.byteLength||0)/1024),m=e.getTile(i).tileSize,f=512/Math.min(m,512)*(i.overscaledZ/t.transform.zoom)*.5;let _=i.canonical.toString();i.overscaledZ!==i.canonical.z&&(_+=` => ${i.overscaledZ}`),function(t,e){t.initDebugOverlayCanvas();const i=t.debugOverlayCanvas,r=t.context.gl,s=t.debugOverlayCanvas.getContext("2d");s.clearRect(0,0,i.width,i.height),s.shadowColor="white",s.shadowBlur=2,s.lineWidth=1.5,s.strokeStyle="white",s.textBaseline="top",s.font="bold 36px Open Sans, sans-serif",s.fillText(e,5,5),s.strokeText(e,5,5),t.debugOverlayTexture.update(i),t.debugOverlayTexture.bind(r.LINEAR,r.CLAMP_TO_EDGE)}(t,`${_} ${d}kB`),a.draw(r,s.TRIANGLES,o,l,Xd.alphaBlended,tm.disabled,qp(n,he.transparent,f),null,h,t.debugBuffer,t.quadTriangleIndexBuffer,t.debugSegments),a.draw(r,s.LINE_STRIP,o,l,c,tm.disabled,qp(n,he.red),u,h,t.debugBuffer,t.tileBorderIndexBuffer,t.debugSegments)}function Sm(t,e,i){const r=t.context,s=r.gl,n=t.colorModeForRenderPass(),a=new Yd(s.LEQUAL,Yd.ReadWrite,t.depthRangeFor3D),o=t.useProgram("terrain"),l=e.getTerrainMesh();r.bindFramebuffer.set(null),r.viewport.set([0,0,t.width,t.height]);for(const c of i){const i=t.renderToTexture.getTexture(c),h=e.getTerrainData(c.tileID);r.activeTexture.set(s.TEXTURE0),s.bindTexture(s.TEXTURE_2D,i.texture);const u={u_matrix:t.transform.calculatePosMatrix(c.tileID.toUnwrapped()),u_texture:0,u_ele_delta:e.getMeshFrameDelta(t.transform.zoom)};o.draw(r,s.TRIANGLES,a,Qd.disabled,n,tm.backCCW,u,h,"terrain",l.vertexBuffer,l.indexBuffer,l.segments)}}class Im{constructor(t,e){this.context=new Kd(t),this.transform=e,this._tileTextures={},this.terrainFacilitator={dirty:!0,matrix:Ga(),renderTime:0},this.setup(),this.numSublayers=hu.maxUnderzooming+hu.maxOverzooming+1,this.depthEpsilon=1/Math.pow(2,16),this.crossTileSymbolIndex=new bp}resize(t,e,i){if(this.width=Math.floor(t*i),this.height=Math.floor(e*i),this.pixelRatio=i,this.context.viewport.set([0,0,this.width,this.height]),this.style)for(const t of this.style._order)this.style._layers[t].resize()}setup(){const t=this.context,e=new vn;e.emplaceBack(0,0),e.emplaceBack(va,0),e.emplaceBack(0,va),e.emplaceBack(va,va),this.tileExtentBuffer=t.createVertexBuffer(e,Ep.members),this.tileExtentSegments=Nn.simpleSegment(0,0,4,2);const i=new vn;i.emplaceBack(0,0),i.emplaceBack(va,0),i.emplaceBack(0,va),i.emplaceBack(va,va),this.debugBuffer=t.createVertexBuffer(i,Ep.members),this.debugSegments=Nn.simpleSegment(0,0,4,5);const r=new wn;r.emplaceBack(0,0,0,0),r.emplaceBack(va,0,va,0),r.emplaceBack(0,va,0,va),r.emplaceBack(va,va,va,va),this.rasterBoundsBuffer=t.createVertexBuffer(r,Hh.members),this.rasterBoundsSegments=Nn.simpleSegment(0,0,4,2);const s=new vn;s.emplaceBack(0,0),s.emplaceBack(1,0),s.emplaceBack(0,1),s.emplaceBack(1,1),this.viewportBuffer=t.createVertexBuffer(s,Ep.members),this.viewportSegments=Nn.simpleSegment(0,0,4,2);const n=new On;n.emplaceBack(0),n.emplaceBack(1),n.emplaceBack(3),n.emplaceBack(2),n.emplaceBack(0),this.tileBorderIndexBuffer=t.createIndexBuffer(n);const a=new Bn;a.emplaceBack(0,1,2),a.emplaceBack(2,1,3),this.quadTriangleIndexBuffer=t.createIndexBuffer(a);const o=this.context.gl;this.stencilClearMode=new Qd({func:o.ALWAYS,mask:0},0,255,o.ZERO,o.ZERO,o.ZERO)}clearStencil(){const t=this.context,e=t.gl;this.nextStencilID=1,this.currentStencilSource=void 0;const i=Ga();Qa(i,0,this.width,this.height,0,0,1),Ya(i,i,[e.drawingBufferWidth,e.drawingBufferHeight,0]),this.useProgram("clippingMask").draw(t,e.TRIANGLES,Yd.disabled,this.stencilClearMode,Xd.disabled,tm.disabled,Zp(i),null,"$clipping",this.viewportBuffer,this.quadTriangleIndexBuffer,this.viewportSegments)}_renderTileClippingMasks(t,e){if(this.currentStencilSource===t.source||!t.isTileClipped()||!e||!e.length)return;this.currentStencilSource=t.source;const i=this.context,r=i.gl;this.nextStencilID+e.length>256&&this.clearStencil(),i.setColorMode(Xd.disabled),i.setDepthMode(Yd.disabled);const s=this.useProgram("clippingMask");this._tileClippingMaskIDs={};for(const t of e){const e=this._tileClippingMaskIDs[t.key]=this.nextStencilID++,n=this.style.map.terrain&&this.style.map.terrain.getTerrainData(t);s.draw(i,r.TRIANGLES,Yd.disabled,new Qd({func:r.ALWAYS,mask:0},e,255,r.KEEP,r.KEEP,r.REPLACE),Xd.disabled,tm.disabled,Zp(t.posMatrix),n,"$clipping",this.tileExtentBuffer,this.quadTriangleIndexBuffer,this.tileExtentSegments)}}stencilModeFor3D(){this.currentStencilSource=void 0,this.nextStencilID+1>256&&this.clearStencil();const t=this.nextStencilID++,e=this.context.gl;return new Qd({func:e.NOTEQUAL,mask:255},t,255,e.KEEP,e.KEEP,e.REPLACE)}stencilModeForClipping(t){const e=this.context.gl;return new Qd({func:e.EQUAL,mask:255},this._tileClippingMaskIDs[t.key],0,e.KEEP,e.KEEP,e.REPLACE)}stencilConfigForOverlap(t){const e=this.context.gl,i=t.sort(((t,e)=>e.overscaledZ-t.overscaledZ)),r=i[i.length-1].overscaledZ,s=i[0].overscaledZ-r+1;if(s>1){this.currentStencilSource=void 0,this.nextStencilID+s>256&&this.clearStencil();const t={};for(let i=0;i=0;this.currentLayer--){const t=this.style._layers[i[this.currentLayer]],e=r[t.source],n=s[t.source];this._renderTileClippingMasks(t,n),this.renderLayer(this,e,t,n)}for(this.renderPass="translucent",this.currentLayer=0;this.currentLayeri.source&&!i.isHidden(e)?[t.sourceCaches[i.source]]:[])),s=r.filter((t=>"vector"===t.getSource().type)),n=r.filter((t=>"vector"!==t.getSource().type)),a=t=>{(!i||i.getSource().maxzooma(t))),i||n.forEach((t=>a(t))),i}(this.style,this.transform.zoom);t&&function(t,e,i){for(let r=0;re.style.map.terrain.getElevation(s,t,i):null)}}}(r,t,i,e,i.layout.get("text-rotation-alignment"),i.layout.get("text-pitch-alignment"),s),0!==i.paint.get("icon-opacity").constantOr(1)&&om(t,e,i,r,!1,i.paint.get("icon-translate"),i.paint.get("icon-translate-anchor"),i.layout.get("icon-rotation-alignment"),i.layout.get("icon-pitch-alignment"),i.layout.get("icon-keep-upright"),n,a),0!==i.paint.get("text-opacity").constantOr(1)&&om(t,e,i,r,!0,i.paint.get("text-translate"),i.paint.get("text-translate-anchor"),i.layout.get("text-rotation-alignment"),i.layout.get("text-pitch-alignment"),i.layout.get("text-keep-upright"),n,a),e.map.showCollisionBoxes&&(im(t,e,i,r,i.paint.get("text-translate"),i.paint.get("text-translate-anchor"),!0),im(t,e,i,r,i.paint.get("icon-translate"),i.paint.get("icon-translate-anchor"),!1))}(t,e,i,r,this.style.placement.variableOffsets);break;case"circle":!function(t,e,i,r){if("translucent"!==t.renderPass)return;const s=i.paint.get("circle-opacity"),n=i.paint.get("circle-stroke-width"),a=i.paint.get("circle-stroke-opacity"),o=!i.layout.get("circle-sort-key").isConstant();if(0===s.constantOr(1)&&(0===n.constantOr(1)||0===a.constantOr(1)))return;const l=t.context,c=l.gl,h=t.depthModeForSublayer(0,Yd.ReadOnly),u=Qd.disabled,p=t.colorModeForRenderPass(),d=[];for(let s=0;st.sortKey-e.sortKey));for(const e of d){const{programConfiguration:r,program:s,layoutVertexBuffer:n,indexBuffer:a,uniformValues:o,terrainData:d}=e.state;s.draw(l,c.TRIANGLES,h,u,p,tm.disabled,o,d,i.id,n,a,e.segments,i.paint,t.transform.zoom,r)}}(t,e,i,r);break;case"heatmap":!function(t,e,i,r){if(0!==i.paint.get("heatmap-opacity"))if("offscreen"===t.renderPass){const s=t.context,n=s.gl,a=Qd.disabled,o=new Xd([n.ONE,n.ONE],he.transparent,[!0,!0,!0,!0]);!function(t,e,i){const r=t.gl;t.activeTexture.set(r.TEXTURE1),t.viewport.set([0,0,e.width/4,e.height/4]);let s=i.heatmapFbo;if(s)r.bindTexture(r.TEXTURE_2D,s.colorAttachment.get()),t.bindFramebuffer.set(s.framebuffer);else{const n=r.createTexture();r.bindTexture(r.TEXTURE_2D,n),r.texParameteri(r.TEXTURE_2D,r.TEXTURE_WRAP_S,r.CLAMP_TO_EDGE),r.texParameteri(r.TEXTURE_2D,r.TEXTURE_WRAP_T,r.CLAMP_TO_EDGE),r.texParameteri(r.TEXTURE_2D,r.TEXTURE_MIN_FILTER,r.LINEAR),r.texParameteri(r.TEXTURE_2D,r.TEXTURE_MAG_FILTER,r.LINEAR),s=i.heatmapFbo=t.createFramebuffer(e.width/4,e.height/4,!1,!1),function(t,e,i,r){var s,n;const a=t.gl,o=null!==(s=t.HALF_FLOAT)&&void 0!==s?s:a.UNSIGNED_BYTE,l=null!==(n=t.RGBA16F)&&void 0!==n?n:a.RGBA;a.texImage2D(a.TEXTURE_2D,0,l,e.width/4,e.height/4,0,a.RGBA,o,null),r.colorAttachment.set(i)}(t,e,n,s)}}(s,t,i),s.clear({color:he.transparent});for(let l=0;l{const s=Ga();Qa(s,0,t.width,t.height,0,0,1);const n=t.context.gl;return{u_matrix:s,u_world:[n.drawingBufferWidth,n.drawingBufferHeight],u_image:0,u_color_ramp:1,u_opacity:e.paint.get("heatmap-opacity")}})(t,e),null,e.id,t.viewportBuffer,t.quadTriangleIndexBuffer,t.viewportSegments,e.paint,t.transform.zoom)}(t,i))}(t,e,i,r);break;case"line":!function(t,e,i,r){if("translucent"!==t.renderPass)return;const s=i.paint.get("line-opacity"),n=i.paint.get("line-width");if(0===s.constantOr(1)||0===n.constantOr(1))return;const a=t.depthModeForSublayer(0,Yd.ReadOnly),o=t.colorModeForRenderPass(),l=i.paint.get("line-dasharray"),c=i.paint.get("line-pattern"),h=c.constantOr(1),u=i.paint.get("line-gradient"),p=i.getCrossfadeParameters(),d=h?"linePattern":l?"lineSDF":u?"lineGradient":"line",f=t.context,_=f.gl;let g=!0;for(const s of r){const r=e.getTile(s);if(h&&!r.patternsLoaded())continue;const n=r.getBucket(i);if(!n)continue;const x=n.programConfigurations.get(i.id),v=t.context.program.get(),b=t.useProgram(d,x),w=g||b.program!==v,T=t.style.map.terrain&&t.style.map.terrain.getTerrainData(s),S=c.constantOr(null);if(S&&r.imageAtlas){const t=r.imageAtlas,e=t.patternPositions[S.to.toString()],i=t.patternPositions[S.from.toString()];e&&i&&x.setConstantPatternPositions(e,i)}const I=T?s:null,A=h?Xp(t,r,i,p,I):l?Kp(t,r,i,l,p,I):u?Wp(t,r,i,n.lineClipsArray.length,I):Hp(t,r,i,I);if(h)f.activeTexture.set(_.TEXTURE0),r.imageAtlasTexture.bind(_.LINEAR,_.CLAMP_TO_EDGE),x.updatePaintBuffers(p);else if(l&&(w||t.lineAtlas.dirty))f.activeTexture.set(_.TEXTURE0),t.lineAtlas.bind(f);else if(u){const r=n.gradients[i.id];let a=r.texture;if(i.gradientVersion!==r.version){let o=256;if(i.stepInterpolant){const i=e.getSource().maxzoom,r=s.canonical.z===i?Math.ceil(1<0?e.pop():null}isPatternMissing(t){if(!t)return!1;if(!t.from||!t.to)return!0;const e=this.imageManager.getPattern(t.from.toString()),i=this.imageManager.getPattern(t.to.toString());return!e||!i}useProgram(t,e){this.cache=this.cache||{};const i=t+(e?e.cacheKey:"")+(this._showOverdrawInspector?"/overdraw":"")+(this.style.map.terrain?"/terrain":"");return this.cache[i]||(this.cache[i]=new Dp(this.context,zp[t],e,ad[t],this._showOverdrawInspector,this.style.map.terrain)),this.cache[i]}setCustomLayerDefaults(){this.context.unbindVAO(),this.context.cullFace.setDefault(),this.context.activeTexture.setDefault(),this.context.pixelStoreUnpack.setDefault(),this.context.pixelStoreUnpackPremultiplyAlpha.setDefault(),this.context.pixelStoreUnpackFlipY.setDefault()}setBaseState(){const t=this.context.gl;this.context.cullFace.set(!1),this.context.viewport.set([0,0,this.width,this.height]),this.context.blendEquation.set(t.FUNC_ADD)}initDebugOverlayCanvas(){null==this.debugOverlayCanvas&&(this.debugOverlayCanvas=document.createElement("canvas"),this.debugOverlayCanvas.width=512,this.debugOverlayCanvas.height=512,this.debugOverlayTexture=new uh(this.context,this.debugOverlayCanvas,this.context.gl.RGBA))}destroy(){this.debugOverlayTexture&&this.debugOverlayTexture.destroy()}overLimit(){const{drawingBufferWidth:t,drawingBufferHeight:e}=this.context.gl;return this.width!==t||this.height!==e}}class Am{constructor(t,e){this.points=t,this.planes=e}static fromInvProjectionMatrix(t,e,i){const r=Math.pow(2,i),s=[[-1,1,-1,1],[1,1,-1,1],[1,-1,-1,1],[-1,-1,-1,1],[-1,1,1,1],[1,1,1,1],[1,-1,1,1],[-1,-1,1,1]].map((i=>{const s=1/(i=so([],i,t))[3]/e*r;return function(t,e,i){return t[0]=e[0]*i[0],t[1]=e[1]*i[1],t[2]=e[2]*i[2],t[3]=e[3]*i[3],t}(i,i,[s,s,1/i[3],s])})),n=[[0,1,2],[6,5,4],[0,3,7],[2,1,5],[3,2,6],[0,4,5]].map((t=>{const e=function(t,e){var i=e[0],r=e[1],s=e[2],n=i*i+r*r+s*s;return n>0&&(n=1/Math.sqrt(n)),t[0]=e[0]*n,t[1]=e[1]*n,t[2]=e[2]*n,t}([],function(t,e,i){var r=e[0],s=e[1],n=e[2],a=i[0],o=i[1],l=i[2];return t[0]=s*l-n*o,t[1]=n*a-r*l,t[2]=r*o-s*a,t}([],ro([],s[t[0]],s[t[1]]),ro([],s[t[2]],s[t[1]]))),i=-((r=e)[0]*(n=s[t[1]])[0]+r[1]*n[1]+r[2]*n[2]);var r,n;return e.concat(i)}));return new Am(s,n)}}class Em{constructor(t,e){this.min=t,this.max=e,this.center=function(t,e,i){return t[0]=.5*e[0],t[1]=.5*e[1],t[2]=.5*e[2],t}([],function(t,e,i){return t[0]=e[0]+i[0],t[1]=e[1]+i[1],t[2]=e[2]+i[2],t}([],this.min,this.max))}quadrant(t){const e=[t%2==0,t<2],i=eo(this.min),r=eo(this.max);for(let t=0;t=0&&o++;if(0===o)return 0;o!==e.length&&(i=!1)}var r,s;if(i)return 2;for(let e=0;e<3;e++){let i=Number.MAX_VALUE,r=-Number.MAX_VALUE;for(let s=0;sthis.max[e]-this.min[e])return 0}return 1}}class Cm{constructor(t=0,e=0,i=0,r=0){if(isNaN(t)||t<0||isNaN(e)||e<0||isNaN(i)||i<0||isNaN(r)||r<0)throw new Error("Invalid value for edge-insets, top, bottom, left and right must all be numbers");this.top=t,this.bottom=e,this.left=i,this.right=r}interpolate(t,e,i){return null!=e.top&&null!=t.top&&(this.top=ni.number(t.top,e.top,i)),null!=e.bottom&&null!=t.bottom&&(this.bottom=ni.number(t.bottom,e.bottom,i)),null!=e.left&&null!=t.left&&(this.left=ni.number(t.left,e.left,i)),null!=e.right&&null!=t.right&&(this.right=ni.number(t.right,e.right,i)),this}getCenter(t,e){const i=m((this.left+t-this.right)/2,0,t),r=m((this.top+e-this.bottom)/2,0,e);return new n(i,r)}equals(t){return this.top===t.top&&this.bottom===t.bottom&&this.left===t.left&&this.right===t.right}clone(){return new Cm(this.top,this.bottom,this.left,this.right)}toJSON(){return{top:this.top,bottom:this.bottom,left:this.left,right:this.right}}}class zm{constructor(t,e,i,r,s){this.tileSize=512,this.maxValidLatitude=85.051129,this._renderWorldCopies=void 0===s||!!s,this._minZoom=t||0,this._maxZoom=e||22,this._minPitch=null==i?0:i,this._maxPitch=null==r?60:r,this.setMaxBounds(),this.width=0,this.height=0,this._center=new Eh(0,0),this._elevation=0,this.zoom=0,this.angle=0,this._fov=.6435011087932844,this._pitch=0,this._unmodified=!0,this._edgeInsets=new Cm,this._posMatrixCache={},this._alignedPosMatrixCache={},this._minEleveationForCurrentTile=0}clone(){const t=new zm(this._minZoom,this._maxZoom,this._minPitch,this.maxPitch,this._renderWorldCopies);return t.apply(this),t}apply(t){this.tileSize=t.tileSize,this.latRange=t.latRange,this.width=t.width,this.height=t.height,this._center=t._center,this._elevation=t._elevation,this._minEleveationForCurrentTile=t._minEleveationForCurrentTile,this.zoom=t.zoom,this.angle=t.angle,this._fov=t._fov,this._pitch=t._pitch,this._unmodified=t._unmodified,this._edgeInsets=t._edgeInsets.clone(),this._calcMatrices()}get minZoom(){return this._minZoom}set minZoom(t){this._minZoom!==t&&(this._minZoom=t,this.zoom=Math.max(this.zoom,t))}get maxZoom(){return this._maxZoom}set maxZoom(t){this._maxZoom!==t&&(this._maxZoom=t,this.zoom=Math.min(this.zoom,t))}get minPitch(){return this._minPitch}set minPitch(t){this._minPitch!==t&&(this._minPitch=t,this.pitch=Math.max(this.pitch,t))}get maxPitch(){return this._maxPitch}set maxPitch(t){this._maxPitch!==t&&(this._maxPitch=t,this.pitch=Math.min(this.pitch,t))}get renderWorldCopies(){return this._renderWorldCopies}set renderWorldCopies(t){void 0===t?t=!0:null===t&&(t=!1),this._renderWorldCopies=t}get worldSize(){return this.tileSize*this.scale}get centerOffset(){return this.centerPoint._sub(this.size._div(2))}get size(){return new n(this.width,this.height)}get bearing(){return-this.angle/Math.PI*180}set bearing(t){const e=-f(t,-180,180)*Math.PI/180;this.angle!==e&&(this._unmodified=!1,this.angle=e,this._calcMatrices(),this.rotationMatrix=function(){var t=new ja(4);return ja!=Float32Array&&(t[1]=0,t[2]=0),t[0]=1,t[3]=1,t}(),function(t,e,i){var r=e[0],s=e[1],n=e[2],a=e[3],o=Math.sin(i),l=Math.cos(i);t[0]=r*l+n*o,t[1]=s*l+a*o,t[2]=r*-o+n*l,t[3]=s*-o+a*l}(this.rotationMatrix,this.rotationMatrix,this.angle))}get pitch(){return this._pitch/Math.PI*180}set pitch(t){const e=m(t,this.minPitch,this.maxPitch)/180*Math.PI;this._pitch!==e&&(this._unmodified=!1,this._pitch=e,this._calcMatrices())}get fov(){return this._fov/Math.PI*180}set fov(t){t=Math.max(.01,Math.min(60,t)),this._fov!==t&&(this._unmodified=!1,this._fov=t/180*Math.PI,this._calcMatrices())}get zoom(){return this._zoom}set zoom(t){const e=Math.min(Math.max(t,this.minZoom),this.maxZoom);this._zoom!==e&&(this._unmodified=!1,this._zoom=e,this.tileZoom=Math.max(0,Math.floor(e)),this.scale=this.zoomScale(e),this._constrain(),this._calcMatrices())}get center(){return this._center}set center(t){t.lat===this._center.lat&&t.lng===this._center.lng||(this._unmodified=!1,this._center=t,this._constrain(),this._calcMatrices())}get elevation(){return this._elevation}set elevation(t){t!==this._elevation&&(this._elevation=t,this._constrain(),this._calcMatrices())}get padding(){return this._edgeInsets.toJSON()}set padding(t){this._edgeInsets.equals(t)||(this._unmodified=!1,this._edgeInsets.interpolate(this._edgeInsets,t,1),this._calcMatrices())}get centerPoint(){return this._edgeInsets.getCenter(this.width,this.height)}isPaddingEqual(t){return this._edgeInsets.equals(t)}interpolatePadding(t,e,i){this._unmodified=!1,this._edgeInsets.interpolate(t,e,i),this._constrain(),this._calcMatrices()}coveringZoomLevel(t){const e=(t.roundZoom?Math.round:Math.floor)(this.zoom+this.scaleZoom(this.tileSize/t.tileSize));return Math.max(0,e)}getVisibleUnwrappedCoordinates(t){const e=[new $h(0,t)];if(this._renderWorldCopies){const i=this.pointCoordinate(new n(0,0)),r=this.pointCoordinate(new n(this.width,0)),s=this.pointCoordinate(new n(this.width,this.height)),a=this.pointCoordinate(new n(0,this.height)),o=Math.floor(Math.min(i.x,r.x,s.x,a.x)),l=Math.floor(Math.max(i.x,r.x,s.x,a.x)),c=1;for(let i=o-c;i<=l+c;i++)0!==i&&e.push(new $h(i,t))}return e}coveringTiles(t){var e,i;let r=this.coveringZoomLevel(t);const s=r;if(void 0!==t.minzoom&&rt.maxzoom&&(r=t.maxzoom);const n=this.pointCoordinate(this.getCameraPoint()),a=Bh.fromLngLat(this.center),o=Math.pow(2,r),l=[o*n.x,o*n.y,0],c=[o*a.x,o*a.y,0],h=Am.fromInvProjectionMatrix(this.invProjMatrix,this.worldSize,r);let u=t.minzoom||0;!t.terrain&&this.pitch<=60&&this._edgeInsets.top<.1&&(u=r);const p=t.terrain?2/Math.min(this.tileSize,t.tileSize)*this.tileSize:3,d=t=>({aabb:new Em([t*o,0,0],[(t+1)*o,o,0]),zoom:0,x:0,y:0,wrap:t,fullyVisible:!1}),m=[],f=[],_=r,g=t.reparseOverscaled?s:r;if(this._renderWorldCopies)for(let t=1;t<=3;t++)m.push(d(-t)),m.push(d(t));for(m.push(d(0));m.length>0;){const r=m.pop(),s=r.x,n=r.y;let a=r.fullyVisible;if(!a){const t=r.aabb.intersects(h);if(0===t)continue;a=2===t}const o=t.terrain?l:c,d=r.aabb.distanceX(o),y=r.aabb.distanceY(o),x=Math.max(Math.abs(d),Math.abs(y));if(r.zoom===_||x>p+(1<<_-r.zoom)-2&&r.zoom>=u){const t=_-r.zoom,e=l[0]-.5-(s<>1),h=r.zoom+1;let u=r.aabb.quadrant(o);if(t.terrain){const s=new qh(h,r.wrap,h,l,c),n=t.terrain.getMinMaxElevation(s),a=null!==(e=n.minElevation)&&void 0!==e?e:this.elevation,o=null!==(i=n.maxElevation)&&void 0!==i?i:this.elevation;u=new Em([u.min[0],u.min[1],a],[u.max[0],u.max[1],o])}m.push({aabb:u,zoom:h,x:l,y:c,wrap:r.wrap,fullyVisible:a})}}return f.sort(((t,e)=>t.distanceSq-e.distanceSq)).map((t=>t.tileID))}resize(t,e){this.width=t,this.height=e,this.pixelsToGLUnits=[2/t,-2/e],this._constrain(),this._calcMatrices()}get unmodified(){return this._unmodified}zoomScale(t){return Math.pow(2,t)}scaleZoom(t){return Math.log(t)/Math.LN2}project(t){const e=m(t.lat,-this.maxValidLatitude,this.maxValidLatitude);return new n(Ph(t.lng)*this.worldSize,kh(e)*this.worldSize)}unproject(t){return new Bh(t.x/this.worldSize,t.y/this.worldSize).toLngLat()}get point(){return this.project(this.center)}getCameraPosition(){return{lngLat:this.pointLocation(this.getCameraPoint()),altitude:Math.cos(this._pitch)*this.cameraToCenterDistance/this._pixelPerMeter+this.elevation}}recalculateZoom(t){const e=this.pointLocation(this.centerPoint,t),i=t.getElevationForLngLatZoom(e,this.tileZoom);if(!(this.elevation-i))return;const r=this.getCameraPosition(),s=Bh.fromLngLat(r.lngLat,r.altitude),n=Bh.fromLngLat(e,i),a=s.x-n.x,o=s.y-n.y,l=s.z-n.z,c=Math.sqrt(a*a+o*o+l*l),h=this.scaleZoom(this.cameraToCenterDistance/c/this.tileSize);this._elevation=i,this._center=e,this.zoom=h}setLocationAtPoint(t,e){const i=this.pointCoordinate(e),r=this.pointCoordinate(this.centerPoint),s=this.locationCoordinate(t),n=new Bh(s.x-(i.x-r.x),s.y-(i.y-r.y));this.center=this.coordinateLocation(n),this._renderWorldCopies&&(this.center=this.center.wrap())}locationPoint(t,e){return e?this.coordinatePoint(this.locationCoordinate(t),e.getElevationForLngLatZoom(t,this.tileZoom),this.pixelMatrix3D):this.coordinatePoint(this.locationCoordinate(t))}pointLocation(t,e){return this.coordinateLocation(this.pointCoordinate(t,e))}locationCoordinate(t){return Bh.fromLngLat(t)}coordinateLocation(t){return t&&t.toLngLat()}pointCoordinate(t,e){if(e){const i=e.pointCoordinate(t);if(null!=i)return i}const i=[t.x,t.y,0,1],r=[t.x,t.y,1,1];so(i,i,this.pixelMatrixInverse),so(r,r,this.pixelMatrixInverse);const s=i[3],n=r[3],a=i[1]/s,o=r[1]/n,l=i[2]/s,c=r[2]/n,h=l===c?0:(0-l)/(c-l);return new Bh(ni.number(i[0]/s,r[0]/n,h)/this.worldSize,ni.number(a,o,h)/this.worldSize)}coordinatePoint(t,e=0,i=this.pixelMatrix){const r=[t.x*this.worldSize,t.y*this.worldSize,e,1];return so(r,r,i),new n(r[0]/r[3],r[1]/r[3])}getBounds(){const t=Math.max(0,this.height/2-this.getHorizon());return(new Ch).extend(this.pointLocation(new n(0,t))).extend(this.pointLocation(new n(this.width,t))).extend(this.pointLocation(new n(this.width,this.height))).extend(this.pointLocation(new n(0,this.height)))}getMaxBounds(){return this.latRange&&2===this.latRange.length&&this.lngRange&&2===this.lngRange.length?new Ch([this.lngRange[0],this.latRange[0]],[this.lngRange[1],this.latRange[1]]):null}getHorizon(){return Math.tan(Math.PI/2-this._pitch)*this.cameraToCenterDistance*.85}setMaxBounds(t){t?(this.lngRange=[t.getWest(),t.getEast()],this.latRange=[t.getSouth(),t.getNorth()],this._constrain()):(this.lngRange=null,this.latRange=[-this.maxValidLatitude,this.maxValidLatitude])}calculatePosMatrix(t,e=!1){const i=t.key,r=e?this._alignedPosMatrixCache:this._posMatrixCache;if(r[i])return r[i];const s=t.canonical,n=this.worldSize/this.zoomScale(s.z),a=s.x+Math.pow(2,s.z)*t.wrap,o=Ha(new Float64Array(16));return Ka(o,o,[a*n,s.y*n,0]),Ya(o,o,[n/va,n/va,1]),Xa(o,e?this.alignedProjMatrix:this.projMatrix,o),r[i]=new Float32Array(o),r[i]}customLayerMatrix(){return this.mercatorMatrix.slice()}_constrain(){if(!this.center||!this.width||!this.height||this._constraining)return;this._constraining=!0;let t,e,i,r,s=-90,a=90,o=-180,l=180;const c=this.size,h=this._unmodified;if(this.latRange){const e=this.latRange;s=kh(e[1])*this.worldSize,a=kh(e[0])*this.worldSize,t=a-sa&&(r=a-e)}if(this.lngRange){const t=(o+l)/2,e=f(u.x,t-this.worldSize/2,t+this.worldSize/2),r=c.x/2;e-rl&&(i=l-r)}void 0===i&&void 0===r||(this.center=this.unproject(new n(void 0!==i?i:u.x,void 0!==r?r:u.y)).wrap()),this._unmodified=h,this._constraining=!1}_calcMatrices(){if(!this.height)return;const t=this.centerOffset,e=this.point.x,i=this.point.y;this.cameraToCenterDistance=.5/Math.tan(this._fov/2)*this.height,this._pixelPerMeter=Dh(1,this.center.lat)*this.worldSize;let r=Ha(new Float64Array(16));Ya(r,r,[this.width/2,-this.height/2,1]),Ka(r,r,[1,-1,0]),this.labelPlaneMatrix=r,r=Ha(new Float64Array(16)),Ya(r,r,[1,-1,1]),Ka(r,r,[-1,-1,0]),Ya(r,r,[2/this.width,2/this.height,1]),this.glCoordMatrix=r;const s=this.cameraToCenterDistance+this._elevation*this._pixelPerMeter/Math.cos(this._pitch),n=Math.min(this.elevation,this._minEleveationForCurrentTile),a=s-n*this._pixelPerMeter/Math.cos(this._pitch),o=n<0?a:s,l=Math.PI/2+this._pitch,c=this._fov*(.5+t.y/this.height),h=Math.sin(c)*o/Math.sin(m(Math.PI-l-c,.01,Math.PI-.01)),u=this.getHorizon(),p=2*Math.atan(u/this.cameraToCenterDistance)*(.5+t.y/(2*u)),d=Math.sin(p)*o/Math.sin(m(Math.PI-l-p,.01,Math.PI-.01)),f=Math.min(h,d),_=1.01*(Math.cos(Math.PI/2-this._pitch)*f+o),g=this.height/50;r=new Float64Array(16),function(t,e,i,r,s){var n,a=1/Math.tan(e/2);t[0]=a/i,t[1]=0,t[2]=0,t[3]=0,t[4]=0,t[5]=a,t[6]=0,t[7]=0,t[8]=0,t[9]=0,t[11]=-1,t[12]=0,t[13]=0,t[15]=0,null!=s&&s!==1/0?(t[10]=(s+r)*(n=1/(r-s)),t[14]=2*s*r*n):(t[10]=-1,t[14]=-2*r)}(r,this._fov,this.width/this.height,g,_),r[8]=2*-t.x/this.width,r[9]=2*t.y/this.height,Ya(r,r,[1,-1,1]),Ka(r,r,[0,0,-this.cameraToCenterDistance]),function(t,e,i){var r=Math.sin(i),s=Math.cos(i),n=e[4],a=e[5],o=e[6],l=e[7],c=e[8],h=e[9],u=e[10],p=e[11];e!==t&&(t[0]=e[0],t[1]=e[1],t[2]=e[2],t[3]=e[3],t[12]=e[12],t[13]=e[13],t[14]=e[14],t[15]=e[15]),t[4]=n*s+c*r,t[5]=a*s+h*r,t[6]=o*s+u*r,t[7]=l*s+p*r,t[8]=c*s-n*r,t[9]=h*s-a*r,t[10]=u*s-o*r,t[11]=p*s-l*r}(r,r,this._pitch),Ja(r,r,this.angle),Ka(r,r,[-e,-i,0]),this.mercatorMatrix=Ya([],r,[this.worldSize,this.worldSize,this.worldSize]),Ya(r,r,[1,1,this._pixelPerMeter]),this.pixelMatrix=Xa(new Float64Array(16),this.labelPlaneMatrix,r),Ka(r,r,[0,0,-this.elevation]),this.projMatrix=r,this.invProjMatrix=Wa([],r),this.pixelMatrix3D=Xa(new Float64Array(16),this.labelPlaneMatrix,r);const y=this.width%2/2,x=this.height%2/2,v=Math.cos(this.angle),b=Math.sin(this.angle),w=e-Math.round(e)+v*y+b*x,T=i-Math.round(i)+v*x+b*y,S=new Float64Array(r);if(Ka(S,S,[w>.5?w-1:w,T>.5?T-1:T,0]),this.alignedProjMatrix=S,r=Wa(new Float64Array(16),this.pixelMatrix),!r)throw new Error("failed to invert matrix");this.pixelMatrixInverse=r,this._posMatrixCache={},this._alignedPosMatrixCache={}}maxPitchScaleFactor(){if(!this.pixelMatrixInverse)return 1;const t=this.pointCoordinate(new n(0,0)),e=[t.x*this.worldSize,t.y*this.worldSize,0,1];return so(e,e,this.pixelMatrix)[3]/this.cameraToCenterDistance}getCameraPoint(){const t=Math.tan(this._pitch)*(this.cameraToCenterDistance||1);return this.centerPoint.add(new n(0,t))}getCameraQueryGeometry(t){const e=this.getCameraPoint();if(1===t.length)return[t[0],e];{let i=e.x,r=e.y,s=e.x,a=e.y;for(const e of t)i=Math.min(i,e.x),r=Math.min(r,e.y),s=Math.max(s,e.x),a=Math.max(a,e.y);return[new n(i,r),new n(s,r),new n(s,a),new n(i,a),new n(i,r)]}}}function Mm(t,e){let i,r=!1,s=null,n=null;const a=()=>{s=null,r&&(t.apply(n,i),s=setTimeout(a,e),r=!1)};return(...t)=>(r=!0,n=this,i=t,s||a(),s)}class Pm{constructor(t){this._getCurrentHash=()=>{const t=window.location.hash.replace("#","");if(this._hashName){let e;return t.split("&").map((t=>t.split("="))).forEach((t=>{t[0]===this._hashName&&(e=t)})),(e&&e[1]||"").split("/")}return t.split("/")},this._onHashChange=()=>{const t=this._getCurrentHash();if(t.length>=3&&!t.some((t=>isNaN(t)))){const e=this._map.dragRotate.isEnabled()&&this._map.touchZoomRotate.isEnabled()?+(t[3]||0):this._map.getBearing();return this._map.jumpTo({center:[+t[2],+t[1]],zoom:+t[0],bearing:e,pitch:+(t[4]||0)}),!0}return!1},this._updateHashUnthrottled=()=>{const t=window.location.href.replace(/(#.+)?$/,this.getHashString());try{window.history.replaceState(window.history.state,null,t)}catch(t){}},this._updateHash=Mm(this._updateHashUnthrottled,300),this._hashName=t&&encodeURIComponent(t)}addTo(t){return this._map=t,addEventListener("hashchange",this._onHashChange,!1),this._map.on("moveend",this._updateHash),this}remove(){return removeEventListener("hashchange",this._onHashChange,!1),this._map.off("moveend",this._updateHash),clearTimeout(this._updateHash()),delete this._map,this}getHashString(t){const e=this._map.getCenter(),i=Math.round(100*this._map.getZoom())/100,r=Math.ceil((i*Math.LN2+Math.log(512/360/.5))/Math.LN10),s=Math.pow(10,r),n=Math.round(e.lng*s)/s,a=Math.round(e.lat*s)/s,o=this._map.getBearing(),l=this._map.getPitch();let c="";if(c+=t?`/${n}/${a}/${i}`:`${i}/${a}/${n}`,(o||l)&&(c+="/"+Math.round(10*o)/10),l&&(c+=`/${Math.round(l)}`),this._hashName){const t=this._hashName;let e=!1;const i=window.location.hash.slice(1).split("&").map((i=>{const r=i.split("=")[0];return r===t?(e=!0,`${r}=${c}`):i})).filter((t=>t));return e||i.push(`${t}=${c}`),`#${i.join("&")}`}return`#${c}`}}const km={linearity:.3,easing:p(0,0,.3,1)},Dm=g({deceleration:2500,maxSpeed:1400},km),Lm=g({deceleration:20,maxSpeed:1400},km),Rm=g({deceleration:1e3,maxSpeed:360},km),Bm=g({deceleration:1e3,maxSpeed:90},km);class Fm{constructor(t){this._map=t,this.clear()}clear(){this._inertiaBuffer=[]}record(t){this._drainInertiaBuffer(),this._inertiaBuffer.push({time:B.now(),settings:t})}_drainInertiaBuffer(){const t=this._inertiaBuffer,e=B.now();for(;t.length>0&&e-t[0].time>160;)t.shift()}_onMoveEnd(t){if(this._drainInertiaBuffer(),this._inertiaBuffer.length<2)return;const e={zoom:0,bearing:0,pitch:0,pan:new n(0,0),pinchAround:void 0,around:void 0};for(const{settings:t}of this._inertiaBuffer)e.zoom+=t.zoomDelta||0,e.bearing+=t.bearingDelta||0,e.pitch+=t.pitchDelta||0,t.panDelta&&e.pan._add(t.panDelta),t.around&&(e.around=t.around),t.pinchAround&&(e.pinchAround=t.pinchAround);const i=this._inertiaBuffer[this._inertiaBuffer.length-1].time-this._inertiaBuffer[0].time,r={};if(e.pan.mag()){const s=Vm(e.pan.mag(),i,g({},Dm,t||{}));r.offset=e.pan.mult(s.amount/e.pan.mag()),r.center=this._map.transform.center,Om(r,s)}if(e.zoom){const t=Vm(e.zoom,i,Lm);r.zoom=this._map.transform.zoom+t.amount,Om(r,t)}if(e.bearing){const t=Vm(e.bearing,i,Rm);r.bearing=this._map.transform.bearing+m(t.amount,-179,179),Om(r,t)}if(e.pitch){const t=Vm(e.pitch,i,Bm);r.pitch=this._map.transform.pitch+t.amount,Om(r,t)}if(r.zoom||r.bearing){const t=void 0===e.pinchAround?e.around:e.pinchAround;r.around=t?this._map.unproject(t):this._map.getCenter()}return this.clear(),g(r,{noMoveStart:!0})}}function Om(t,e){(!t.duration||t.duratione.unproject(t))),o=s.reduce(((t,e,i,r)=>t.add(e.div(r.length))),new n(0,0));super(t,{points:s,point:o,lngLats:a,lngLat:e.unproject(o),originalEvent:i}),this._defaultPrevented=!1}}class $m extends nt{preventDefault(){this._defaultPrevented=!0}get defaultPrevented(){return this._defaultPrevented}constructor(t,e,i){super(t,{originalEvent:i}),this._defaultPrevented=!1}}class qm{constructor(t,e){this._map=t,this._clickTolerance=e.clickTolerance}reset(){delete this._mousedownPos}wheel(t){return this._firePreventable(new $m(t.type,this._map,t))}mousedown(t,e){return this._mousedownPos=e,this._firePreventable(new Um(t.type,this._map,t))}mouseup(t){this._map.fire(new Um(t.type,this._map,t))}click(t,e){this._mousedownPos&&this._mousedownPos.dist(e)>=this._clickTolerance||this._map.fire(new Um(t.type,this._map,t))}dblclick(t){return this._firePreventable(new Um(t.type,this._map,t))}mouseover(t){this._map.fire(new Um(t.type,this._map,t))}mouseout(t){this._map.fire(new Um(t.type,this._map,t))}touchstart(t){return this._firePreventable(new Nm(t.type,this._map,t))}touchmove(t){this._map.fire(new Nm(t.type,this._map,t))}touchend(t){this._map.fire(new Nm(t.type,this._map,t))}touchcancel(t){this._map.fire(new Nm(t.type,this._map,t))}_firePreventable(t){if(this._map.fire(t),t.defaultPrevented)return{}}isEnabled(){return!0}isActive(){return!1}enable(){}disable(){}}class Zm{constructor(t){this._map=t}reset(){this._delayContextMenu=!1,this._ignoreContextMenu=!0,delete this._contextMenuEvent}mousemove(t){this._map.fire(new Um(t.type,this._map,t))}mousedown(){this._delayContextMenu=!0,this._ignoreContextMenu=!1}mouseup(){this._delayContextMenu=!1,this._contextMenuEvent&&(this._map.fire(new Um("contextmenu",this._map,this._contextMenuEvent)),delete this._contextMenuEvent)}contextmenu(t){this._delayContextMenu?this._contextMenuEvent=t:this._ignoreContextMenu||this._map.fire(new Um(t.type,this._map,t)),this._map.listens("contextmenu")&&t.preventDefault()}isEnabled(){return!0}isActive(){return!1}enable(){}disable(){}}class jm{constructor(t){this._map=t}get transform(){return this._map._requestedCameraState||this._map.transform}get center(){return{lng:this.transform.center.lng,lat:this.transform.center.lat}}get zoom(){return this.transform.zoom}get pitch(){return this.transform.pitch}get bearing(){return this.transform.bearing}unproject(t){return this.transform.pointLocation(n.convert(t),this._map.terrain)}}class Gm{constructor(t,e){this._map=t,this._tr=new jm(t),this._el=t.getCanvasContainer(),this._container=t.getContainer(),this._clickTolerance=e.clickTolerance||1}isEnabled(){return!!this._enabled}isActive(){return!!this._active}enable(){this.isEnabled()||(this._enabled=!0)}disable(){this.isEnabled()&&(this._enabled=!1)}mousedown(t,e){this.isEnabled()&&t.shiftKey&&0===t.button&&(F.disableDrag(),this._startPos=this._lastPos=e,this._active=!0)}mousemoveWindow(t,e){if(!this._active)return;const i=e;if(this._lastPos.equals(i)||!this._box&&i.dist(this._startPos)t.fitScreenCoordinates(i,r,this._tr.bearing,{linear:!0})};this._fireEvent("boxzoomcancel",t)}keydown(t){this._active&&27===t.keyCode&&(this.reset(),this._fireEvent("boxzoomcancel",t))}reset(){this._active=!1,this._container.classList.remove("maplibregl-crosshair"),this._box&&(F.remove(this._box),this._box=null),F.enableDrag(),delete this._startPos,delete this._lastPos}_fireEvent(t,e){return this._map.fire(new nt(t,{originalEvent:e}))}}function Hm(t,e){if(t.length!==e.length)throw new Error(`The number of touches and points are not equal - touches ${t.length}, points ${e.length}`);const i={};for(let r=0;rthis.numTouches)&&(this.aborted=!0),this.aborted||(void 0===this.startTime&&(this.startTime=t.timeStamp),i.length===this.numTouches&&(this.centroid=function(t){const e=new n(0,0);for(const i of t)e._add(i);return e.div(t.length)}(e),this.touches=Hm(i,e)))}touchmove(t,e,i){if(this.aborted||!this.centroid)return;const r=Hm(i,e);for(const t in this.touches){const e=r[t];(!e||e.dist(this.touches[t])>30)&&(this.aborted=!0)}}touchend(t,e,i){if((!this.centroid||t.timeStamp-this.startTime>500)&&(this.aborted=!0),0===i.length){const t=!this.aborted&&this.centroid;if(this.reset(),t)return t}}}class Xm{constructor(t){this.singleTap=new Wm(t),this.numTaps=t.numTaps,this.reset()}reset(){this.lastTime=1/0,delete this.lastTap,this.count=0,this.singleTap.reset()}touchstart(t,e,i){this.singleTap.touchstart(t,e,i)}touchmove(t,e,i){this.singleTap.touchmove(t,e,i)}touchend(t,e,i){const r=this.singleTap.touchend(t,e,i);if(r){const e=t.timeStamp-this.lastTime<500,i=!this.lastTap||this.lastTap.dist(r)<30;if(e&&i||this.reset(),this.count++,this.lastTime=t.timeStamp,this.lastTap=r,this.count===this.numTaps)return this.reset(),r}}}class Km{constructor(t){this._tr=new jm(t),this._zoomIn=new Xm({numTouches:1,numTaps:2}),this._zoomOut=new Xm({numTouches:2,numTaps:1}),this.reset()}reset(){this._active=!1,this._zoomIn.reset(),this._zoomOut.reset()}touchstart(t,e,i){this._zoomIn.touchstart(t,e,i),this._zoomOut.touchstart(t,e,i)}touchmove(t,e,i){this._zoomIn.touchmove(t,e,i),this._zoomOut.touchmove(t,e,i)}touchend(t,e,i){const r=this._zoomIn.touchend(t,e,i),s=this._zoomOut.touchend(t,e,i),n=this._tr;return r?(this._active=!0,t.preventDefault(),setTimeout((()=>this.reset()),0),{cameraAnimation:e=>e.easeTo({duration:300,zoom:n.zoom+1,around:n.unproject(r)},{originalEvent:t})}):s?(this._active=!0,t.preventDefault(),setTimeout((()=>this.reset()),0),{cameraAnimation:e=>e.easeTo({duration:300,zoom:n.zoom-1,around:n.unproject(s)},{originalEvent:t})}):void 0}touchcancel(){this.reset()}enable(){this._enabled=!0}disable(){this._enabled=!1,this.reset()}isEnabled(){return this._enabled}isActive(){return this._active}}class Ym{constructor(t){this._enabled=!!t.enable,this._moveStateManager=t.moveStateManager,this._clickTolerance=t.clickTolerance||1,this._moveFunction=t.move,this._activateOnStart=!!t.activateOnStart,t.assignEvents(this),this.reset()}reset(t){this._active=!1,this._moved=!1,delete this._lastPoint,this._moveStateManager.endMove(t)}_move(...t){const e=this._moveFunction(...t);if(e.bearingDelta||e.pitchDelta||e.around||e.panDelta)return this._active=!0,e}dragStart(t,e){this.isEnabled()&&!this._lastPoint&&this._moveStateManager.isValidStartEvent(t)&&(this._moveStateManager.startMove(t),this._lastPoint=e.length?e[0]:e,this._activateOnStart&&this._lastPoint&&(this._active=!0))}dragMove(t,e){if(!this.isEnabled())return;const i=this._lastPoint;if(!i)return;if(t.preventDefault(),!this._moveStateManager.isValidMoveEvent(t))return void this.reset(t);const r=e.length?e[0]:e;return!this._moved&&r.dist(i){t.mousedown=t.dragStart,t.mousemoveWindow=t.dragMove,t.mouseup=t.dragEnd,t.contextmenu=function(t){t.preventDefault()}},rf=({enable:t,clickTolerance:e,bearingDegreesPerPixelMoved:i=.8})=>{const r=new Qm({checkCorrectEvent:t=>0===F.mouseButton(t)&&t.ctrlKey||2===F.mouseButton(t)});return new Ym({clickTolerance:e,move:(t,e)=>({bearingDelta:(e.x-t.x)*i}),moveStateManager:r,enable:t,assignEvents:ef})},sf=({enable:t,clickTolerance:e,pitchDegreesPerPixelMoved:i=-.5})=>{const r=new Qm({checkCorrectEvent:t=>0===F.mouseButton(t)&&t.ctrlKey||2===F.mouseButton(t)});return new Ym({clickTolerance:e,move:(t,e)=>({pitchDelta:(e.y-t.y)*i}),moveStateManager:r,enable:t,assignEvents:ef})};class nf{constructor(t,e){this._minTouches=t.cooperativeGestures?2:1,this._clickTolerance=t.clickTolerance||1,this._map=e,this.reset()}reset(){this._active=!1,this._touches={},this._sum=new n(0,0),setTimeout((()=>{this._cancelCooperativeMessage=!1}),200)}touchstart(t,e,i){return this._calculateTransform(t,e,i)}touchmove(t,e,i){if(this._map._cooperativeGestures&&(2===this._minTouches&&i.length<2&&!this._cancelCooperativeMessage?this._map._onCooperativeGesture(t,!1,i.length):this._cancelCooperativeMessage||(this._cancelCooperativeMessage=!0)),this._active&&!(i.length0&&(this._active=!0);const r=Hm(i,e),s=new n(0,0),a=new n(0,0);let o=0;for(const t in r){const e=r[t],i=this._touches[t];i&&(s._add(e),a._add(e.sub(i)),o++,r[t]=e)}if(this._touches=r,oMath.abs(t.x)}class df extends af{constructor(t){super(),this._map=t}reset(){super.reset(),this._valid=void 0,delete this._firstMove,delete this._lastPoints}touchstart(t,e,i){super.touchstart(t,e,i),this._currentTouchCount=i.length}_start(t){this._lastPoints=t,pf(t[0].sub(t[1]))&&(this._valid=!1)}_move(t,e,i){if(this._map._cooperativeGestures&&this._currentTouchCount<3)return;const r=t[0].sub(this._lastPoints[0]),s=t[1].sub(this._lastPoints[1]);return this._valid=this.gestureBeginsVertically(r,s,i.timeStamp),this._valid?(this._lastPoints=t,this._active=!0,{pitchDelta:(r.y+s.y)/2*-.5}):void 0}gestureBeginsVertically(t,e,i){if(void 0!==this._valid)return this._valid;const r=t.mag()>=2,s=e.mag()>=2;if(!r&&!s)return;if(!r||!s)return void 0===this._firstMove&&(this._firstMove=i),i-this._firstMove<100&&void 0;const n=t.y>0==e.y>0;return pf(t)&&pf(e)&&n}}const mf={panStep:100,bearingStep:15,pitchStep:10};class ff{constructor(t){this._tr=new jm(t);const e=mf;this._panStep=e.panStep,this._bearingStep=e.bearingStep,this._pitchStep=e.pitchStep,this._rotationDisabled=!1}reset(){this._active=!1}keydown(t){if(t.altKey||t.ctrlKey||t.metaKey)return;let e=0,i=0,r=0,s=0,n=0;switch(t.keyCode){case 61:case 107:case 171:case 187:e=1;break;case 189:case 109:case 173:e=-1;break;case 37:t.shiftKey?i=-1:(t.preventDefault(),s=-1);break;case 39:t.shiftKey?i=1:(t.preventDefault(),s=1);break;case 38:t.shiftKey?r=1:(t.preventDefault(),n=-1);break;case 40:t.shiftKey?r=-1:(t.preventDefault(),n=1);break;default:return}return this._rotationDisabled&&(i=0,r=0),{cameraAnimation:a=>{const o=this._tr;a.easeTo({duration:300,easeId:"keyboardHandler",easing:_f,zoom:e?Math.round(o.zoom)+e*(t.shiftKey?2:1):o.zoom,bearing:o.bearing+i*this._bearingStep,pitch:o.pitch+r*this._pitchStep,offset:[-s*this._panStep,-n*this._panStep],center:o.center},{originalEvent:t})}}}enable(){this._enabled=!0}disable(){this._enabled=!1,this.reset()}isEnabled(){return this._enabled}isActive(){return this._active}disableRotation(){this._rotationDisabled=!0}enableRotation(){this._rotationDisabled=!1}}function _f(t){return t*(2-t)}const gf=4.000244140625;class yf{constructor(t,e){this._onTimeout=t=>{this._type="wheel",this._delta-=this._lastValue,this._active||this._start(t)},this._map=t,this._tr=new jm(t),this._el=t.getCanvasContainer(),this._triggerRenderFrame=e,this._delta=0,this._defaultZoomRate=.01,this._wheelZoomRate=.0022222222222222222}setZoomRate(t){this._defaultZoomRate=t}setWheelZoomRate(t){this._wheelZoomRate=t}isEnabled(){return!!this._enabled}isActive(){return!!this._active||void 0!==this._finishTimeout}isZooming(){return!!this._zooming}enable(t){this.isEnabled()||(this._enabled=!0,this._aroundCenter=!!t&&"center"===t.around)}disable(){this.isEnabled()&&(this._enabled=!1)}wheel(t){if(!this.isEnabled())return;if(this._map._cooperativeGestures){if(!t[this._map._metaKey])return;t.preventDefault()}let e=t.deltaMode===WheelEvent.DOM_DELTA_LINE?40*t.deltaY:t.deltaY;const i=B.now(),r=i-(this._lastWheelEventTime||0);this._lastWheelEventTime=i,0!==e&&e%gf==0?this._type="wheel":0!==e&&Math.abs(e)<4?this._type="trackpad":r>400?(this._type=null,this._lastValue=e,this._timeout=setTimeout(this._onTimeout,40,t)):this._type||(this._type=Math.abs(r*e)<200?"trackpad":"wheel",this._timeout&&(clearTimeout(this._timeout),this._timeout=null,e+=this._lastValue)),t.shiftKey&&e&&(e/=4),this._type&&(this._lastWheelEvent=t,this._delta-=e,this._active||this._start(t)),t.preventDefault()}_start(t){if(!this._delta)return;this._frameId&&(this._frameId=null),this._active=!0,this.isZooming()||(this._zooming=!0),this._finishTimeout&&(clearTimeout(this._finishTimeout),delete this._finishTimeout);const e=F.mousePos(this._el,t),i=this._tr;this._around=Eh.convert(this._aroundCenter?i.center:i.unproject(e)),this._aroundPoint=i.transform.locationPoint(this._around),this._frameId||(this._frameId=!0,this._triggerRenderFrame())}renderFrame(){if(!this._frameId)return;if(this._frameId=null,!this.isActive())return;const t=this._tr.transform;if(0!==this._delta){const e="wheel"===this._type&&Math.abs(this._delta)>gf?this._wheelZoomRate:this._defaultZoomRate;let i=2/(1+Math.exp(-Math.abs(this._delta*e)));this._delta<0&&0!==i&&(i=1/i);const r="number"==typeof this._targetZoom?t.zoomScale(this._targetZoom):t.scale;this._targetZoom=Math.min(t.maxZoom,Math.max(t.minZoom,t.scaleZoom(r*i))),"wheel"===this._type&&(this._startZoom=t.zoom,this._easing=this._smoothOutEasing(200)),this._delta=0}const e="number"==typeof this._targetZoom?this._targetZoom:t.zoom,i=this._startZoom,r=this._easing;let s,n=!1;if("wheel"===this._type&&i&&r){const t=Math.min((B.now()-this._lastWheelEventTime)/200,1),a=r(t);s=ni.number(i,e,a),t<1?this._frameId||(this._frameId=!0):n=!0}else s=e,n=!0;return this._active=!0,n&&(this._active=!1,this._finishTimeout=setTimeout((()=>{this._zooming=!1,this._triggerRenderFrame(),delete this._targetZoom,delete this._finishTimeout}),200)),{noInertia:!0,needsRenderFrame:!n,zoomDelta:s-t.zoom,around:this._aroundPoint,originalEvent:this._lastWheelEvent}}_smoothOutEasing(t){let e=d;if(this._prevEase){const t=this._prevEase,i=(B.now()-t.start)/t.duration,r=t.easing(i+.01)-t.easing(i),s=.27/Math.sqrt(r*r+1e-4)*.01;e=p(s,Math.sqrt(.0729-s*s),.25,1)}return this._prevEase={start:B.now(),duration:t,easing:e},e}reset(){this._active=!1,this._zooming=!1,delete this._targetZoom,this._finishTimeout&&(clearTimeout(this._finishTimeout),delete this._finishTimeout)}}class xf{constructor(t,e){this._clickZoom=t,this._tapZoom=e}enable(){this._clickZoom.enable(),this._tapZoom.enable()}disable(){this._clickZoom.disable(),this._tapZoom.disable()}isEnabled(){return this._clickZoom.isEnabled()&&this._tapZoom.isEnabled()}isActive(){return this._clickZoom.isActive()||this._tapZoom.isActive()}}class vf{constructor(t){this._tr=new jm(t),this.reset()}reset(){this._active=!1}dblclick(t,e){return t.preventDefault(),{cameraAnimation:i=>{i.easeTo({duration:300,zoom:this._tr.zoom+(t.shiftKey?-1:1),around:this._tr.unproject(e)},{originalEvent:t})}}}enable(){this._enabled=!0}disable(){this._enabled=!1,this.reset()}isEnabled(){return this._enabled}isActive(){return this._active}}class bf{constructor(){this._tap=new Xm({numTouches:1,numTaps:1}),this.reset()}reset(){this._active=!1,delete this._swipePoint,delete this._swipeTouch,delete this._tapTime,delete this._tapPoint,this._tap.reset()}touchstart(t,e,i){if(!this._swipePoint)if(this._tapTime){const r=e[0],s=t.timeStamp-this._tapTime<500,n=this._tapPoint.dist(r)<30;s&&n?i.length>0&&(this._swipePoint=r,this._swipeTouch=i[0].identifier):this.reset()}else this._tap.touchstart(t,e,i)}touchmove(t,e,i){if(this._tapTime){if(this._swipePoint){if(i[0].identifier!==this._swipeTouch)return;const r=e[0],s=r.y-this._swipePoint.y;return this._swipePoint=r,t.preventDefault(),this._active=!0,{zoomDelta:s/128}}}else this._tap.touchmove(t,e,i)}touchend(t,e,i){if(this._tapTime)this._swipePoint&&0===i.length&&this.reset();else{const r=this._tap.touchend(t,e,i);r&&(this._tapTime=t.timeStamp,this._tapPoint=r)}}touchcancel(){this.reset()}enable(){this._enabled=!0}disable(){this._enabled=!1,this.reset()}isEnabled(){return this._enabled}isActive(){return this._active}}class wf{constructor(t,e,i){this._el=t,this._mousePan=e,this._touchPan=i}enable(t){this._inertiaOptions=t||{},this._mousePan.enable(),this._touchPan.enable(),this._el.classList.add("maplibregl-touch-drag-pan")}disable(){this._mousePan.disable(),this._touchPan.disable(),this._el.classList.remove("maplibregl-touch-drag-pan")}isEnabled(){return this._mousePan.isEnabled()&&this._touchPan.isEnabled()}isActive(){return this._mousePan.isActive()||this._touchPan.isActive()}}class Tf{constructor(t,e,i){this._pitchWithRotate=t.pitchWithRotate,this._mouseRotate=e,this._mousePitch=i}enable(){this._mouseRotate.enable(),this._pitchWithRotate&&this._mousePitch.enable()}disable(){this._mouseRotate.disable(),this._mousePitch.disable()}isEnabled(){return this._mouseRotate.isEnabled()&&(!this._pitchWithRotate||this._mousePitch.isEnabled())}isActive(){return this._mouseRotate.isActive()||this._mousePitch.isActive()}}class Sf{constructor(t,e,i,r){this._el=t,this._touchZoom=e,this._touchRotate=i,this._tapDragZoom=r,this._rotationDisabled=!1,this._enabled=!0}enable(t){this._touchZoom.enable(t),this._rotationDisabled||this._touchRotate.enable(t),this._tapDragZoom.enable(),this._el.classList.add("maplibregl-touch-zoom-rotate")}disable(){this._touchZoom.disable(),this._touchRotate.disable(),this._tapDragZoom.disable(),this._el.classList.remove("maplibregl-touch-zoom-rotate")}isEnabled(){return this._touchZoom.isEnabled()&&(this._rotationDisabled||this._touchRotate.isEnabled())&&this._tapDragZoom.isEnabled()}isActive(){return this._touchZoom.isActive()||this._touchRotate.isActive()||this._tapDragZoom.isActive()}disableRotation(){this._rotationDisabled=!0,this._touchRotate.disable()}enableRotation(){this._rotationDisabled=!1,this._touchZoom.isEnabled()&&this._touchRotate.enable()}}const If=t=>t.zoom||t.drag||t.pitch||t.rotate;class Af extends nt{}function Ef(t){return t.panDelta&&t.panDelta.mag()||t.zoomDelta||t.bearingDelta||t.pitchDelta}class Cf{constructor(t,e){this.handleWindowEvent=t=>{this.handleEvent(t,`${t.type}Window`)},this.handleEvent=(t,e)=>{if("blur"===t.type)return void this.stop(!0);this._updatingCamera=!0;const i="renderFrame"===t.type?void 0:t,r={needsRenderFrame:!1},s={},n={},a=t.touches,o=a?this._getMapTouches(a):void 0,l=o?F.touchPos(this._el,o):F.mousePos(this._el,t);for(const{handlerName:a,handler:c,allowed:h}of this._handlers){if(!c.isEnabled())continue;let u;this._blockedByActive(n,h,a)?c.reset():c[e||t.type]&&(u=c[e||t.type](t,l,o),this.mergeHandlerResult(r,s,u,a,i),u&&u.needsRenderFrame&&this._triggerRenderFrame()),(u||c.isActive())&&(n[a]=c)}const c={};for(const t in this._previousActiveHandlers)n[t]||(c[t]=i);this._previousActiveHandlers=n,(Object.keys(c).length||Ef(r))&&(this._changes.push([r,s,c]),this._triggerRenderFrame()),(Object.keys(n).length||Ef(r))&&this._map._stop(!0),this._updatingCamera=!1;const{cameraAnimation:h}=r;h&&(this._inertia.clear(),this._fireEvents({},{},!0),this._changes=[],h(this._map))},this._map=t,this._el=this._map.getCanvasContainer(),this._handlers=[],this._handlersById={},this._changes=[],this._inertia=new Fm(t),this._bearingSnap=e.bearingSnap,this._previousActiveHandlers={},this._eventsInProgress={},this._addDefaultHandlers(e);const i=this._el;this._listeners=[[i,"touchstart",{passive:!0}],[i,"touchmove",{passive:!1}],[i,"touchend",void 0],[i,"touchcancel",void 0],[i,"mousedown",void 0],[i,"mousemove",void 0],[i,"mouseup",void 0],[document,"mousemove",{capture:!0}],[document,"mouseup",void 0],[i,"mouseover",void 0],[i,"mouseout",void 0],[i,"dblclick",void 0],[i,"click",void 0],[i,"keydown",{capture:!1}],[i,"keyup",void 0],[i,"wheel",{passive:!1}],[i,"contextmenu",void 0],[window,"blur",void 0]];for(const[t,e,i]of this._listeners)F.addEventListener(t,e,t===document?this.handleWindowEvent:this.handleEvent,i)}destroy(){for(const[t,e,i]of this._listeners)F.removeEventListener(t,e,t===document?this.handleWindowEvent:this.handleEvent,i)}_addDefaultHandlers(t){const e=this._map,i=e.getCanvasContainer();this._add("mapEvent",new qm(e,t));const r=e.boxZoom=new Gm(e,t);this._add("boxZoom",r),t.interactive&&t.boxZoom&&r.enable();const s=new Km(e),n=new vf(e);e.doubleClickZoom=new xf(n,s),this._add("tapZoom",s),this._add("clickZoom",n),t.interactive&&t.doubleClickZoom&&e.doubleClickZoom.enable();const a=new bf;this._add("tapDragZoom",a);const o=e.touchPitch=new df(e);this._add("touchPitch",o),t.interactive&&t.touchPitch&&e.touchPitch.enable(t.touchPitch);const l=rf(t),c=sf(t);e.dragRotate=new Tf(t,l,c),this._add("mouseRotate",l,["mousePitch"]),this._add("mousePitch",c,["mouseRotate"]),t.interactive&&t.dragRotate&&e.dragRotate.enable();const h=(({enable:t,clickTolerance:e})=>{const i=new Qm({checkCorrectEvent:t=>0===F.mouseButton(t)&&!t.ctrlKey});return new Ym({clickTolerance:e,move:(t,e)=>({around:e,panDelta:e.sub(t)}),activateOnStart:!0,moveStateManager:i,enable:t,assignEvents:ef})})(t),u=new nf(t,e);e.dragPan=new wf(i,h,u),this._add("mousePan",h),this._add("touchPan",u,["touchZoom","touchRotate"]),t.interactive&&t.dragPan&&e.dragPan.enable(t.dragPan);const p=new uf,d=new cf;e.touchZoomRotate=new Sf(i,d,p,a),this._add("touchRotate",p,["touchPan","touchZoom"]),this._add("touchZoom",d,["touchPan","touchRotate"]),t.interactive&&t.touchZoomRotate&&e.touchZoomRotate.enable(t.touchZoomRotate);const m=e.scrollZoom=new yf(e,(()=>this._triggerRenderFrame()));this._add("scrollZoom",m,["mousePan"]),t.interactive&&t.scrollZoom&&e.scrollZoom.enable(t.scrollZoom);const f=e.keyboard=new ff(e);this._add("keyboard",f),t.interactive&&t.keyboard&&e.keyboard.enable(),this._add("blockableMapEvent",new Zm(e))}_add(t,e,i){this._handlers.push({handlerName:t,handler:e,allowed:i}),this._handlersById[t]=e}stop(t){if(!this._updatingCamera){for(const{handler:t}of this._handlers)t.reset();this._inertia.clear(),this._fireEvents({},{},t),this._changes=[]}}isActive(){for(const{handler:t}of this._handlers)if(t.isActive())return!0;return!1}isZooming(){return!!this._eventsInProgress.zoom||this._map.scrollZoom.isZooming()}isRotating(){return!!this._eventsInProgress.rotate}isMoving(){return Boolean(If(this._eventsInProgress))||this.isZooming()}_blockedByActive(t,e,i){for(const r in t)if(r!==i&&(!e||e.indexOf(r)<0))return!0;return!1}_getMapTouches(t){const e=[];for(const i of t)this._el.contains(i.target)&&e.push(i);return e}mergeHandlerResult(t,e,i,r,s){if(!i)return;g(t,i);const n={handlerName:r,originalEvent:i.originalEvent||s};void 0!==i.zoomDelta&&(e.zoom=n),void 0!==i.panDelta&&(e.drag=n),void 0!==i.pitchDelta&&(e.pitch=n),void 0!==i.bearingDelta&&(e.rotate=n)}_applyChanges(){const t={},e={},i={};for(const[r,s,a]of this._changes)r.panDelta&&(t.panDelta=(t.panDelta||new n(0,0))._add(r.panDelta)),r.zoomDelta&&(t.zoomDelta=(t.zoomDelta||0)+r.zoomDelta),r.bearingDelta&&(t.bearingDelta=(t.bearingDelta||0)+r.bearingDelta),r.pitchDelta&&(t.pitchDelta=(t.pitchDelta||0)+r.pitchDelta),void 0!==r.around&&(t.around=r.around),void 0!==r.pinchAround&&(t.pinchAround=r.pinchAround),r.noInertia&&(t.noInertia=r.noInertia),g(e,s),g(i,a);this._updateMapTransform(t,e,i),this._changes=[]}_updateMapTransform(t,e,i){const r=this._map,s=r._getTransformForUpdate(),n=r.terrain;if(!(Ef(t)||n&&this._terrainMovement))return this._fireEvents(e,i,!0);let{panDelta:a,zoomDelta:o,bearingDelta:l,pitchDelta:c,around:h,pinchAround:u}=t;void 0!==u&&(h=u),r._stop(!0),h=h||r.transform.centerPoint;const p=s.pointLocation(a?h.sub(a):h);l&&(s.bearing+=l),c&&(s.pitch+=c),o&&(s.zoom+=o),n?this._terrainMovement||!e.drag&&!e.zoom?e.drag&&this._terrainMovement?s.center=s.pointLocation(s.centerPoint.sub(a)):s.setLocationAtPoint(p,h):(this._terrainMovement=!0,this._map._elevationFreeze=!0,s.setLocationAtPoint(p,h),this._map.once("moveend",(()=>{this._map._elevationFreeze=!1,this._terrainMovement=!1,s.recalculateZoom(r.terrain)}))):s.setLocationAtPoint(p,h),r._applyUpdatedTransform(s),this._map._update(),t.noInertia||this._inertia.record(t),this._fireEvents(e,i,!0)}_fireEvents(t,e,i){const r=If(this._eventsInProgress),s=If(t),n={};for(const e in t){const{originalEvent:i}=t[e];this._eventsInProgress[e]||(n[`${e}start`]=i),this._eventsInProgress[e]=t[e]}!r&&s&&this._fireEvent("movestart",s.originalEvent);for(const t in n)this._fireEvent(t,n[t]);s&&this._fireEvent("move",s.originalEvent);for(const e in t){const{originalEvent:i}=t[e];this._fireEvent(e,i)}const a={};let o;for(const t in this._eventsInProgress){const{handlerName:i,originalEvent:r}=this._eventsInProgress[t];this._handlersById[i].isActive()||(delete this._eventsInProgress[t],o=e[i]||r,a[`${t}end`]=o)}for(const t in a)this._fireEvent(t,a[t]);const l=If(this._eventsInProgress);if(i&&(r||s)&&!l){this._updatingCamera=!0;const t=this._inertia._onMoveEnd(this._map.dragPan._inertiaOptions),e=t=>0!==t&&-this._bearingSnap{delete this._frameId,this.handleEvent(new Af("renderFrame",{timeStamp:t})),this._applyChanges()}))}_triggerRenderFrame(){void 0===this._frameId&&(this._frameId=this._requestFrame())}}class zf extends ot{constructor(t,e){super(),this._renderFrameCallback=()=>{const t=Math.min((B.now()-this._easeStart)/this._easeOptions.duration,1);this._onEaseFrame(this._easeOptions.easing(t)),t<1&&this._easeFrameId?this._easeFrameId=this._requestRenderFrame(this._renderFrameCallback):this.stop()},this._moving=!1,this._zooming=!1,this.transform=t,this._bearingSnap=e.bearingSnap,this.on("moveend",(()=>{delete this._requestedCameraState}))}getCenter(){return new Eh(this.transform.center.lng,this.transform.center.lat)}setCenter(t,e){return this.jumpTo({center:t},e)}panBy(t,e,i){return t=n.convert(t).mult(-1),this.panTo(this.transform.center,g({offset:t},e),i)}panTo(t,e,i){return this.easeTo(g({center:t},e),i)}getZoom(){return this.transform.zoom}setZoom(t,e){return this.jumpTo({zoom:t},e),this}zoomTo(t,e,i){return this.easeTo(g({zoom:t},e),i)}zoomIn(t,e){return this.zoomTo(this.getZoom()+1,t,e),this}zoomOut(t,e){return this.zoomTo(this.getZoom()-1,t,e),this}getBearing(){return this.transform.bearing}setBearing(t,e){return this.jumpTo({bearing:t},e),this}getPadding(){return this.transform.padding}setPadding(t,e){return this.jumpTo({padding:t},e),this}rotateTo(t,e,i){return this.easeTo(g({bearing:t},e),i)}resetNorth(t,e){return this.rotateTo(0,g({duration:1e3},t),e),this}resetNorthPitch(t,e){return this.easeTo(g({bearing:0,pitch:0,duration:1e3},t),e),this}snapToNorth(t,e){return Math.abs(this.getBearing()){if(this._zooming&&(i.zoom=ni.number(r,l,n)),this._rotating&&(i.bearing=ni.number(s,c,n)),this._pitching&&(i.pitch=ni.number(a,h,n)),this._padding&&(i.interpolatePadding(o,u,n),m=i.centerPoint.add(p)),this.terrain&&!t.freezeElevation&&this._updateElevation(n),b)i.setLocationAtPoint(b,w);else{const t=i.zoomScale(i.zoom-r),e=l>r?Math.min(2,v):Math.max(.5,v),s=Math.pow(e,1-n),a=i.unproject(y.add(x.mult(n*s)).mult(t));i.setLocationAtPoint(i.renderWorldCopies?a.wrap():a,m)}this._applyUpdatedTransform(i),this._fireMoveEvents(e)}),(t=>{this.terrain&&this._finalizeElevation(),this._afterEase(e,t)}),t),this}_prepareEase(t,e,i={}){this._moving=!0,e||i.moving||this.fire(new nt("movestart",t)),this._zooming&&!i.zooming&&this.fire(new nt("zoomstart",t)),this._rotating&&!i.rotating&&this.fire(new nt("rotatestart",t)),this._pitching&&!i.pitching&&this.fire(new nt("pitchstart",t))}_prepareElevation(t){this._elevationCenter=t,this._elevationStart=this.transform.elevation,this._elevationTarget=this.terrain.getElevationForLngLatZoom(t,this.transform.tileZoom),this._elevationFreeze=!0}_updateElevation(t){this.transform._minEleveationForCurrentTile=this.terrain.getMinTileElevationForLngLatZoom(this._elevationCenter,this.transform.tileZoom);const e=this.terrain.getElevationForLngLatZoom(this._elevationCenter,this.transform.tileZoom);if(t<1&&e!==this._elevationTarget){const i=this._elevationTarget-this._elevationStart;this._elevationStart+=t*(i-(e-(i*t+this._elevationStart))/(1-t)),this._elevationTarget=e}this.transform.elevation=ni.number(this._elevationStart,this._elevationTarget,t)}_finalizeElevation(){this._elevationFreeze=!1,this.transform.recalculateZoom(this.terrain)}_getTransformForUpdate(){return this.transformCameraUpdate?(this._requestedCameraState||(this._requestedCameraState=this.transform.clone()),this._requestedCameraState):this.transform}_applyUpdatedTransform(t){if(!this.transformCameraUpdate)return;const e=t.clone(),{center:i,zoom:r,pitch:s,bearing:n,elevation:a}=this.transformCameraUpdate(e);i&&(e.center=i),void 0!==r&&(e.zoom=r),void 0!==s&&(e.pitch=s),void 0!==n&&(e.bearing=n),void 0!==a&&(e.elevation=a),this.transform.apply(e)}_fireMoveEvents(t){this.fire(new nt("move",t)),this._zooming&&this.fire(new nt("zoom",t)),this._rotating&&this.fire(new nt("rotate",t)),this._pitching&&this.fire(new nt("pitch",t))}_afterEase(t,e){if(this._easeId&&e&&this._easeId===e)return;delete this._easeId;const i=this._zooming,r=this._rotating,s=this._pitching;this._moving=!1,this._zooming=!1,this._rotating=!1,this._pitching=!1,this._padding=!1,i&&this.fire(new nt("zoomend",t)),r&&this.fire(new nt("rotateend",t)),s&&this.fire(new nt("pitchend",t)),this.fire(new nt("moveend",t))}flyTo(t,e){if(!t.essential&&B.prefersReducedMotion){const i=y(t,["center","zoom","bearing","pitch","around"]);return this.jumpTo(i,e)}this.stop(),t=g({offset:[0,0],speed:1.2,curve:1.42,easing:d},t);const i=this._getTransformForUpdate(),r=this.getZoom(),s=this.getBearing(),a=this.getPitch(),o=this.getPadding(),l="zoom"in t?m(+t.zoom,i.minZoom,i.maxZoom):r,c="bearing"in t?this._normalizeBearing(t.bearing,s):s,h="pitch"in t?+t.pitch:a,u="padding"in t?t.padding:i.padding,p=i.zoomScale(l-r),f=n.convert(t.offset);let _=i.centerPoint.add(f);const x=i.pointLocation(_),v=Eh.convert(t.center||x);this._normalizeCenter(v);const b=i.project(x),w=i.project(v).sub(b);let T=t.curve;const S=Math.max(i.width,i.height),I=S/p,A=w.mag();if("minZoom"in t){const e=m(Math.min(t.minZoom,r,l),i.minZoom,i.maxZoom),s=S/i.zoomScale(e-r);T=Math.sqrt(s/A*2)}const E=T*T;function C(t){const e=(I*I-S*S+(t?-1:1)*E*E*A*A)/(2*(t?I:S)*E*A);return Math.log(Math.sqrt(e*e+1)-e)}function z(t){return(Math.exp(t)-Math.exp(-t))/2}function M(t){return(Math.exp(t)+Math.exp(-t))/2}const P=C(!1);let k=function(t){return M(P)/M(P+T*t)},D=function(t){return S*((M(P)*(z(e=P+T*t)/M(e))-z(P))/E)/A;var e},L=(C(!0)-P)/T;if(Math.abs(A)<1e-6||!isFinite(L)){if(Math.abs(S-I)<1e-6)return this.easeTo(t,e);const i=It.maxDuration&&(t.duration=0),this._zooming=!0,this._rotating=s!==c,this._pitching=h!==a,this._padding=!i.isPaddingEqual(u),this._prepareEase(e,!1),this.terrain&&this._prepareElevation(v),this._ease((n=>{const p=n*L,d=1/k(p);i.zoom=1===n?l:r+i.scaleZoom(d),this._rotating&&(i.bearing=ni.number(s,c,n)),this._pitching&&(i.pitch=ni.number(a,h,n)),this._padding&&(i.interpolatePadding(o,u,n),_=i.centerPoint.add(f)),this.terrain&&!t.freezeElevation&&this._updateElevation(n);const m=1===n?v:i.unproject(b.add(w.mult(D(p))).mult(d));i.setLocationAtPoint(i.renderWorldCopies?m.wrap():m,_),this._applyUpdatedTransform(i),this._fireMoveEvents(e)}),(()=>{this.terrain&&this._finalizeElevation(),this._afterEase(e)}),t),this}isEasing(){return!!this._easeFrameId}stop(){return this._stop()}_stop(t,e){if(this._easeFrameId&&(this._cancelRenderFrame(this._easeFrameId),delete this._easeFrameId,delete this._onEaseFrame),this._onEaseEnd){const t=this._onEaseEnd;delete this._onEaseEnd,t.call(this,e)}if(!t){const t=this.handlers;t&&t.stop(!1)}return this}_ease(t,e,i){!1===i.animate||0===i.duration?(t(1),e()):(this._easeStart=B.now(),this._easeOptions=i,this._onEaseFrame=t,this._onEaseEnd=e,this._easeFrameId=this._requestRenderFrame(this._renderFrameCallback))}_normalizeBearing(t,e){t=f(t,-180,180);const i=Math.abs(t-e);return Math.abs(t-360-e)180?-360:i<-180?360:0}queryTerrainElevation(t){return this.terrain?this.terrain.getElevationForLngLatZoom(Eh.convert(t),this.transform.tileZoom)-this.transform.elevation:null}}class Mf{constructor(t={}){this._toggleAttribution=()=>{this._container.classList.contains("maplibregl-compact")&&(this._container.classList.contains("maplibregl-compact-show")?(this._container.setAttribute("open",""),this._container.classList.remove("maplibregl-compact-show")):(this._container.classList.add("maplibregl-compact-show"),this._container.removeAttribute("open")))},this._updateData=t=>{!t||"metadata"!==t.sourceDataType&&"visibility"!==t.sourceDataType&&"style"!==t.dataType&&"terrain"!==t.type||this._updateAttributions()},this._updateCompact=()=>{this._map.getCanvasContainer().offsetWidth<=640||this._compact?!1===this._compact?this._container.setAttribute("open",""):this._container.classList.contains("maplibregl-compact")||this._container.classList.contains("maplibregl-attrib-empty")||(this._container.setAttribute("open",""),this._container.classList.add("maplibregl-compact","maplibregl-compact-show")):(this._container.setAttribute("open",""),this._container.classList.contains("maplibregl-compact")&&this._container.classList.remove("maplibregl-compact","maplibregl-compact-show"))},this._updateCompactMinimize=()=>{this._container.classList.contains("maplibregl-compact")&&this._container.classList.contains("maplibregl-compact-show")&&this._container.classList.remove("maplibregl-compact-show")},this.options=t}getDefaultPosition(){return"bottom-right"}onAdd(t){return this._map=t,this._compact=this.options&&this.options.compact,this._container=F.create("details","maplibregl-ctrl maplibregl-ctrl-attrib"),this._compactButton=F.create("summary","maplibregl-ctrl-attrib-button",this._container),this._compactButton.addEventListener("click",this._toggleAttribution),this._setElementTitle(this._compactButton,"ToggleAttribution"),this._innerContainer=F.create("div","maplibregl-ctrl-attrib-inner",this._container),this._updateAttributions(),this._updateCompact(),this._map.on("styledata",this._updateData),this._map.on("sourcedata",this._updateData),this._map.on("terrain",this._updateData),this._map.on("resize",this._updateCompact),this._map.on("drag",this._updateCompactMinimize),this._container}onRemove(){F.remove(this._container),this._map.off("styledata",this._updateData),this._map.off("sourcedata",this._updateData),this._map.off("terrain",this._updateData),this._map.off("resize",this._updateCompact),this._map.off("drag",this._updateCompactMinimize),this._map=void 0,this._compact=void 0,this._attribHTML=void 0}_setElementTitle(t,e){const i=this._map._getUIString(`AttributionControl.${e}`);t.title=i,t.setAttribute("aria-label",i)}_updateAttributions(){if(!this._map.style)return;let t=[];if(this.options.customAttribution&&(Array.isArray(this.options.customAttribution)?t=t.concat(this.options.customAttribution.map((t=>"string"!=typeof t?"":t))):"string"==typeof this.options.customAttribution&&t.push(this.options.customAttribution)),this._map.style.stylesheet){const t=this._map.style.stylesheet;this.styleOwner=t.owner,this.styleId=t.id}const e=this._map.style.sourceCaches;for(const i in e){const r=e[i];if(r.used||r.usedForTerrain){const e=r.getSource();e.attribution&&t.indexOf(e.attribution)<0&&t.push(e.attribution)}}t=t.filter((t=>String(t).trim())),t.sort(((t,e)=>t.length-e.length)),t=t.filter(((e,i)=>{for(let r=i+1;r=0)return!1;return!0}));const i=t.join(" | ");i!==this._attribHTML&&(this._attribHTML=i,t.length?(this._innerContainer.innerHTML=i,this._container.classList.remove("maplibregl-attrib-empty")):this._container.classList.add("maplibregl-attrib-empty"),this._updateCompact(),this._editLink=null)}}class Pf{constructor(t={}){this._updateCompact=()=>{const t=this._container.children;if(t.length){const e=t[0];this._map.getCanvasContainer().offsetWidth<=640||this._compact?!1!==this._compact&&e.classList.add("maplibregl-compact"):e.classList.remove("maplibregl-compact")}},this.options=t}getDefaultPosition(){return"bottom-left"}onAdd(t){this._map=t,this._compact=this.options&&this.options.compact,this._container=F.create("div","maplibregl-ctrl");const e=F.create("a","maplibregl-ctrl-logo");return e.target="_blank",e.rel="noopener nofollow",e.href="https://maplibre.org/",e.setAttribute("aria-label",this._map._getUIString("LogoControl.Title")),e.setAttribute("rel","noopener nofollow"),this._container.appendChild(e),this._container.style.display="block",this._map.on("resize",this._updateCompact),this._updateCompact(),this._container}onRemove(){F.remove(this._container),this._map.off("resize",this._updateCompact),this._map=void 0,this._compact=void 0}}class kf{constructor(){this._queue=[],this._id=0,this._cleared=!1,this._currentlyRunning=!1}add(t){const e=++this._id;return this._queue.push({callback:t,id:e,cancelled:!1}),e}remove(t){const e=this._currentlyRunning,i=e?this._queue.concat(e):this._queue;for(const e of i)if(e.id===t)return void(e.cancelled=!0)}run(t=0){if(this._currentlyRunning)throw new Error("Attempting to run(), but is already running.");const e=this._currentlyRunning=this._queue;this._queue=[];for(const i of e)if(!i.cancelled&&(i.callback(t),this._cleared))break;this._cleared=!1,this._currentlyRunning=!1}clear(){this._currentlyRunning&&(this._cleared=!0),this._queue=[]}}var Df;!function(t){t.create="create",t.load="load",t.fullLoad="fullLoad"}(Df||(Df={}));let Lf=null,Rf=[];const Bf=1e3/60,Ff="loadTime",Of="fullLoadTime",Vf={mark(t){performance.mark(t)},frame(t){const e=t;null!=Lf&&Rf.push(e-Lf),Lf=e},clearMetrics(){Lf=null,Rf=[],performance.clearMeasures(Ff),performance.clearMeasures(Of);for(const t in Df)performance.clearMarks(Df[t])},getPerformanceMetrics(){performance.measure(Ff,Df.create,Df.load),performance.measure(Of,Df.create,Df.fullLoad);const t=performance.getEntriesByName(Ff)[0].duration,e=performance.getEntriesByName(Of)[0].duration,i=Rf.length,r=1/(Rf.reduce(((t,e)=>t+e),0)/i/1e3),s=Rf.filter((t=>t>Bf)).reduce(((t,e)=>t+(e-Bf)/Bf),0);return{loadTime:t,fullLoadTime:e,fps:r,percentDroppedFrames:s/(i+s)*100,totalFrames:i}}},Uf={"AttributionControl.ToggleAttribution":"Toggle attribution","AttributionControl.MapFeedback":"Map feedback","FullscreenControl.Enter":"Enter fullscreen","FullscreenControl.Exit":"Exit fullscreen","GeolocateControl.FindMyLocation":"Find my location","GeolocateControl.LocationNotAvailable":"Location not available","LogoControl.Title":"Mapbox logo","NavigationControl.ResetBearing":"Reset bearing to north","NavigationControl.ZoomIn":"Zoom in","NavigationControl.ZoomOut":"Zoom out","ScaleControl.Feet":"ft","ScaleControl.Meters":"m","ScaleControl.Kilometers":"km","ScaleControl.Miles":"mi","ScaleControl.NauticalMiles":"nm","TerrainControl.enableTerrain":"Enable terrain","TerrainControl.disableTerrain":"Disable terrain"};var Nf=Bs([{name:"a_pos3d",type:"Int16",components:3}]);class $f extends ot{constructor(t){super(),this.sourceCache=t,this._tiles={},this._renderableTilesKeys=[],this._sourceTileCache={},this.minzoom=0,this.maxzoom=22,this.tileSize=512,this.deltaZoom=1,t.usedForTerrain=!0,t.tileSize=this.tileSize*2**this.deltaZoom}destruct(){this.sourceCache.usedForTerrain=!1,this.sourceCache.tileSize=null}update(t,e){this.sourceCache.update(t,e),this._renderableTilesKeys=[];const i={};for(const r of t.coveringTiles({tileSize:this.tileSize,minzoom:this.minzoom,maxzoom:this.maxzoom,reparseOverscaled:!1,terrain:e}))i[r.key]=!0,this._renderableTilesKeys.push(r.key),this._tiles[r.key]||(r.posMatrix=new Float64Array(16),Qa(r.posMatrix,0,va,0,va,0,1),this._tiles[r.key]=new ou(r,this.tileSize));for(const t in this._tiles)i[t]||delete this._tiles[t]}freeRtt(t){for(const e in this._tiles){const i=this._tiles[e];(!t||i.tileID.equals(t)||i.tileID.isChildOf(t)||t.isChildOf(i.tileID))&&(i.rtt=[])}}getRenderableTiles(){return this._renderableTilesKeys.map((t=>this.getTileByID(t)))}getTileByID(t){return this._tiles[t]}getTerrainCoords(t){const e={};for(const i of this._renderableTilesKeys){const r=this._tiles[i].tileID;if(r.canonical.equals(t.canonical)){const r=t.clone();r.posMatrix=new Float64Array(16),Qa(r.posMatrix,0,va,0,va,0,1),e[i]=r}else if(r.canonical.isChildOf(t.canonical)){const s=t.clone();s.posMatrix=new Float64Array(16);const n=r.canonical.z-t.canonical.z,a=r.canonical.x-(r.canonical.x>>n<>n<>n;Qa(s.posMatrix,0,l,0,l,0,1),Ka(s.posMatrix,s.posMatrix,[-a*l,-o*l,0]),e[i]=s}else if(t.canonical.isChildOf(r.canonical)){const s=t.clone();s.posMatrix=new Float64Array(16);const n=t.canonical.z-r.canonical.z,a=t.canonical.x-(t.canonical.x>>n<>n<>n;Qa(s.posMatrix,0,va,0,va,0,1),Ka(s.posMatrix,s.posMatrix,[a*l,o*l,0]),Ya(s.posMatrix,s.posMatrix,[1/2**n,1/2**n,0]),e[i]=s}}return e}getSourceTile(t,e){const i=this.sourceCache._source;let r=t.overscaledZ-this.deltaZoom;if(r>i.maxzoom&&(r=i.maxzoom),r=i.minzoom&&(!s||!s.dem);)s=this.sourceCache.getTileByID(t.scaledTo(r--).key);return s}tilesAfterTime(t=Date.now()){return Object.values(this._tiles).filter((e=>e.timeAdded>=t))}}class qf{constructor(t,e,i){this.painter=t,this.sourceCache=new $f(e),this.options=i,this.exaggeration="number"==typeof i.exaggeration?i.exaggeration:1,this.qualityFactor=2,this.meshSize=128,this._demMatrixCache={},this.coordsIndex=[],this._coordsTextureSize=1024}getDEMElevation(t,e,i,r=va){var s;if(!(e>=0&&e=0&&it.canonical.z&&(t.canonical.z>=i?r=t.canonical.z-i:A("cannot calculate elevation if elevation maxzoom > source.maxzoom"));const s=t.canonical.x-(t.canonical.x>>r<>r<>8<<4|t>>8,e[i+3]=0;const i=new _o({width:this._coordsTextureSize,height:this._coordsTextureSize},new Uint8Array(e.buffer)),r=new uh(t,i,t.gl.RGBA,{premultiply:!1});return r.bind(t.gl.NEAREST,t.gl.CLAMP_TO_EDGE),this._coordsTexture=r,r}pointCoordinate(t){const e=new Uint8Array(4),i=this.painter.context,r=i.gl;i.bindFramebuffer.set(this.getFramebuffer("coords").framebuffer),r.readPixels(t.x,this.painter.height/devicePixelRatio-t.y-1,1,1,r.RGBA,r.UNSIGNED_BYTE,e),i.bindFramebuffer.set(null);const s=e[0]+(e[2]>>4<<8),n=e[1]+((15&e[2])<<8),a=this.coordsIndex[255-e[3]],o=a&&this.sourceCache.getTileByID(a);if(!o)return null;const l=this._coordsTextureSize,c=(1<0&&Math.sign(s)<0||!i&&Math.sign(r)<0&&Math.sign(s)>0?(r=360*Math.sign(s)+r,Ph(r)):e}}class Zf{constructor(t,e,i){this._context=t,this._size=e,this._tileSize=i,this._objects=[],this._recentlyUsed=[],this._stamp=0}destruct(){for(const t of this._objects)t.texture.destroy(),t.fbo.destroy()}_createObject(t){const e=this._context.createFramebuffer(this._tileSize,this._tileSize,!0,!0),i=new uh(this._context,{width:this._tileSize,height:this._tileSize,data:null},this._context.gl.RGBA);return i.bind(this._context.gl.LINEAR,this._context.gl.CLAMP_TO_EDGE),e.depthAttachment.set(this._context.createRenderbuffer(this._context.gl.DEPTH_STENCIL,this._tileSize,this._tileSize)),e.colorAttachment.set(i.texture),{id:t,fbo:e,texture:i,stamp:-1,inUse:!1}}getObjectForId(t){return this._objects[t]}useObject(t){t.inUse=!0,this._recentlyUsed=this._recentlyUsed.filter((e=>t.id!==e)),this._recentlyUsed.push(t.id)}stampObject(t){t.stamp=++this._stamp}getOrCreateFreeObject(){for(const t of this._recentlyUsed)if(!this._objects[t].inUse)return this._objects[t];if(this._objects.length>=this._size)throw new Error("No free RenderPool available, call freeAllObjects() required!");const t=this._createObject(this._objects.length);return this._objects.push(t),t}freeObject(t){t.inUse=!1}freeAllObjects(){for(const t of this._objects)this.freeObject(t)}isFull(){return!(this._objects.length!t.inUse))}}const jf={background:!0,fill:!0,line:!0,raster:!0,hillshade:!0};class Gf{constructor(t,e){this.painter=t,this.terrain=e,this.pool=new Zf(t.context,30,e.sourceCache.tileSize*e.qualityFactor)}destruct(){this.pool.destruct()}getTexture(t){return this.pool.getObjectForId(t.rtt[this._stacks.length-1].id).texture}prepareForRender(t,e){this._stacks=[],this._prevType=null,this._rttTiles=[],this._renderableTiles=this.terrain.sourceCache.getRenderableTiles(),this._renderableLayerIds=t._order.filter((i=>!t._layers[i].isHidden(e))),this._coordsDescendingInv={};for(const e in t.sourceCaches){this._coordsDescendingInv[e]={};const i=t.sourceCaches[e].getVisibleCoordinates();for(const t of i){const i=this.terrain.sourceCache.getTerrainCoords(t);for(const t in i)this._coordsDescendingInv[e][t]||(this._coordsDescendingInv[e][t]=[]),this._coordsDescendingInv[e][t].push(i[t])}}this._coordsDescendingInvStr={};for(const e of t._order){const i=t._layers[e],r=i.source;if(jf[i.type]&&!this._coordsDescendingInvStr[r]){this._coordsDescendingInvStr[r]={};for(const t in this._coordsDescendingInv[r])this._coordsDescendingInvStr[r][t]=this._coordsDescendingInv[r][t].map((t=>t.key)).sort().join()}}for(const t of this._renderableTiles)for(const e in this._coordsDescendingInvStr){const i=this._coordsDescendingInvStr[e][t.tileID.key];i&&i!==t.rttCoords[e]&&(t.rtt=[])}}renderLayer(t){if(t.isHidden(this.painter.transform.zoom))return!1;const e=t.type,i=this.painter,r=this._renderableLayerIds[this._renderableLayerIds.length-1]===t.id;if(jf[e]&&(this._prevType&&jf[this._prevType]||this._stacks.push([]),this._prevType=e,this._stacks[this._stacks.length-1].push(t.id),!r))return!0;if(jf[this._prevType]||jf[e]&&r){this._prevType=e;const t=this._stacks.length-1,r=this._stacks[t]||[];for(const e of this._renderableTiles){if(this.pool.isFull()&&(Sm(this.painter,this.terrain,this._rttTiles),this._rttTiles=[],this.pool.freeAllObjects()),this._rttTiles.push(e),e.rtt[t]){const i=this.pool.getObjectForId(e.rtt[t].id);if(i.stamp===e.rtt[t].stamp){this.pool.useObject(i);continue}}const s=this.pool.getOrCreateFreeObject();this.pool.useObject(s),this.pool.stampObject(s),e.rtt[t]={id:s.id,stamp:s.stamp},i.context.bindFramebuffer.set(s.fbo.framebuffer),i.context.clear({color:he.transparent,stencil:0}),i.currentStencilSource=void 0;for(let t=0;t{t.touchstart=t.dragStart,t.touchmoveWindow=t.dragMove,t.touchend=t.dragEnd},Kf={showCompass:!0,showZoom:!0,visualizePitch:!1};class Yf{constructor(t,e,i=!1){this.mousedown=t=>{this.startMouse(g({},t,{ctrlKey:!0,preventDefault:()=>t.preventDefault()}),F.mousePos(this.element,t)),F.addEventListener(window,"mousemove",this.mousemove),F.addEventListener(window,"mouseup",this.mouseup)},this.mousemove=t=>{this.moveMouse(t,F.mousePos(this.element,t))},this.mouseup=t=>{this.mouseRotate.dragEnd(t),this.mousePitch&&this.mousePitch.dragEnd(t),this.offTemp()},this.touchstart=t=>{1!==t.targetTouches.length?this.reset():(this._startPos=this._lastPos=F.touchPos(this.element,t.targetTouches)[0],this.startTouch(t,this._startPos),F.addEventListener(window,"touchmove",this.touchmove,{passive:!1}),F.addEventListener(window,"touchend",this.touchend))},this.touchmove=t=>{1!==t.targetTouches.length?this.reset():(this._lastPos=F.touchPos(this.element,t.targetTouches)[0],this.moveTouch(t,this._lastPos))},this.touchend=t=>{0===t.targetTouches.length&&this._startPos&&this._lastPos&&this._startPos.dist(this._lastPos){this.mouseRotate.reset(),this.mousePitch&&this.mousePitch.reset(),this.touchRotate.reset(),this.touchPitch&&this.touchPitch.reset(),delete this._startPos,delete this._lastPos,this.offTemp()},this._clickTolerance=10;const r=t.dragRotate._mouseRotate.getClickTolerance(),s=t.dragRotate._mousePitch.getClickTolerance();this.element=e,this.mouseRotate=rf({clickTolerance:r,enable:!0}),this.touchRotate=(({enable:t,clickTolerance:e,bearingDegreesPerPixelMoved:i=.8})=>{const r=new tf;return new Ym({clickTolerance:e,move:(t,e)=>({bearingDelta:(e.x-t.x)*i}),moveStateManager:r,enable:t,assignEvents:Xf})})({clickTolerance:r,enable:!0}),this.map=t,i&&(this.mousePitch=sf({clickTolerance:s,enable:!0}),this.touchPitch=(({enable:t,clickTolerance:e,pitchDegreesPerPixelMoved:i=-.5})=>{const r=new tf;return new Ym({clickTolerance:e,move:(t,e)=>({pitchDelta:(e.y-t.y)*i}),moveStateManager:r,enable:t,assignEvents:Xf})})({clickTolerance:s,enable:!0})),F.addEventListener(e,"mousedown",this.mousedown),F.addEventListener(e,"touchstart",this.touchstart,{passive:!1}),F.addEventListener(e,"touchcancel",this.reset)}startMouse(t,e){this.mouseRotate.dragStart(t,e),this.mousePitch&&this.mousePitch.dragStart(t,e),F.disableDrag()}startTouch(t,e){this.touchRotate.dragStart(t,e),this.touchPitch&&this.touchPitch.dragStart(t,e),F.disableDrag()}moveMouse(t,e){const i=this.map,{bearingDelta:r}=this.mouseRotate.dragMove(t,e)||{};if(r&&i.setBearing(i.getBearing()+r),this.mousePitch){const{pitchDelta:r}=this.mousePitch.dragMove(t,e)||{};r&&i.setPitch(i.getPitch()+r)}}moveTouch(t,e){const i=this.map,{bearingDelta:r}=this.touchRotate.dragMove(t,e)||{};if(r&&i.setBearing(i.getBearing()+r),this.touchPitch){const{pitchDelta:r}=this.touchPitch.dragMove(t,e)||{};r&&i.setPitch(i.getPitch()+r)}}off(){const t=this.element;F.removeEventListener(t,"mousedown",this.mousedown),F.removeEventListener(t,"touchstart",this.touchstart,{passive:!1}),F.removeEventListener(window,"touchmove",this.touchmove,{passive:!1}),F.removeEventListener(window,"touchend",this.touchend),F.removeEventListener(t,"touchcancel",this.reset),this.offTemp()}offTemp(){F.enableDrag(),F.removeEventListener(window,"mousemove",this.mousemove),F.removeEventListener(window,"mouseup",this.mouseup),F.removeEventListener(window,"touchmove",this.touchmove,{passive:!1}),F.removeEventListener(window,"touchend",this.touchend)}}let Jf;function Qf(t,e,i){if(t=new Eh(t.lng,t.lat),e){const r=new Eh(t.lng-360,t.lat),s=new Eh(t.lng+360,t.lat),n=i.locationPoint(t).distSqr(e);i.locationPoint(r).distSqr(e)180;){const e=i.locationPoint(t);if(e.x>=0&&e.y>=0&&e.x<=i.width&&e.y<=i.height)break;t.lng>i.center.lng?t.lng-=360:t.lng+=360}return t}const t_={center:"translate(-50%,-50%)",top:"translate(-50%,0)","top-left":"translate(0,0)","top-right":"translate(-100%,0)",bottom:"translate(-50%,-100%)","bottom-left":"translate(0,-100%)","bottom-right":"translate(-100%,-100%)",left:"translate(0,-50%)",right:"translate(-100%,-50%)"};function e_(t,e,i){const r=t.classList;for(const t in t_)r.remove(`maplibregl-${i}-anchor-${t}`);r.add(`maplibregl-${i}-anchor-${e}`)}class i_ extends ot{constructor(t){if(super(),this._onKeyPress=t=>{const e=t.code,i=t.charCode||t.keyCode;"Space"!==e&&"Enter"!==e&&32!==i&&13!==i||this.togglePopup()},this._onMapClick=t=>{const e=t.originalEvent.target,i=this._element;this._popup&&(e===i||i.contains(e))&&this.togglePopup()},this._update=t=>{if(!this._map)return;const e=this._map.loaded()&&!this._map.isMoving();("terrain"===(null==t?void 0:t.type)||"render"===(null==t?void 0:t.type)&&!e)&&this._map.once("render",this._update),this._map.transform.renderWorldCopies&&(this._lngLat=Qf(this._lngLat,this._pos,this._map.transform)),this._pos=this._map.project(this._lngLat)._add(this._offset);let i="";"viewport"===this._rotationAlignment||"auto"===this._rotationAlignment?i=`rotateZ(${this._rotation}deg)`:"map"===this._rotationAlignment&&(i=`rotateZ(${this._rotation-this._map.getBearing()}deg)`);let r="";"viewport"===this._pitchAlignment||"auto"===this._pitchAlignment?r="rotateX(0deg)":"map"===this._pitchAlignment&&(r=`rotateX(${this._map.getPitch()}deg)`),t&&"moveend"!==t.type||(this._pos=this._pos.round()),F.setTransform(this._element,`${t_[this._anchor]} translate(${this._pos.x}px, ${this._pos.y}px) ${r} ${i}`),this._map.terrain&&!this._opacityTimeout&&(this._opacityTimeout=setTimeout((()=>{const t=this._map.unproject(this._pos),e=40075016.686*Math.abs(Math.cos(this._lngLat.lat*Math.PI/180))/Math.pow(2,this._map.transform.tileZoom+8);this._element.style.opacity=t.distanceTo(this._lngLat)>20*e?"0.2":"1.0",this._opacityTimeout=null}),100))},this._onMove=t=>{if(!this._isDragging){const e=this._clickTolerance||this._map._clickTolerance;this._isDragging=t.point.dist(this._pointerdownPos)>=e}this._isDragging&&(this._pos=t.point.sub(this._positionDelta),this._lngLat=this._map.unproject(this._pos),this.setLngLat(this._lngLat),this._element.style.pointerEvents="none","pending"===this._state&&(this._state="active",this.fire(new nt("dragstart"))),this.fire(new nt("drag")))},this._onUp=()=>{this._element.style.pointerEvents="auto",this._positionDelta=null,this._pointerdownPos=null,this._isDragging=!1,this._map.off("mousemove",this._onMove),this._map.off("touchmove",this._onMove),"active"===this._state&&this.fire(new nt("dragend")),this._state="inactive"},this._addDragHandler=t=>{this._element.contains(t.originalEvent.target)&&(t.preventDefault(),this._positionDelta=t.point.sub(this._pos).add(this._offset),this._pointerdownPos=t.point,this._state="pending",this._map.on("mousemove",this._onMove),this._map.on("touchmove",this._onMove),this._map.once("mouseup",this._onUp),this._map.once("touchend",this._onUp))},this._anchor=t&&t.anchor||"center",this._color=t&&t.color||"#3FB1CE",this._scale=t&&t.scale||1,this._draggable=t&&t.draggable||!1,this._clickTolerance=t&&t.clickTolerance||0,this._isDragging=!1,this._state="inactive",this._rotation=t&&t.rotation||0,this._rotationAlignment=t&&t.rotationAlignment||"auto",this._pitchAlignment=t&&t.pitchAlignment&&"auto"!==t.pitchAlignment?t.pitchAlignment:this._rotationAlignment,t&&t.element)this._element=t.element,this._offset=n.convert(t&&t.offset||[0,0]);else{this._defaultMarker=!0,this._element=F.create("div"),this._element.setAttribute("aria-label","Map marker");const e=F.createNS("http://www.w3.org/2000/svg","svg"),i=41,r=27;e.setAttributeNS(null,"display","block"),e.setAttributeNS(null,"height",`${i}px`),e.setAttributeNS(null,"width",`${r}px`),e.setAttributeNS(null,"viewBox",`0 0 ${r} ${i}`);const s=F.createNS("http://www.w3.org/2000/svg","g");s.setAttributeNS(null,"stroke","none"),s.setAttributeNS(null,"stroke-width","1"),s.setAttributeNS(null,"fill","none"),s.setAttributeNS(null,"fill-rule","evenodd");const a=F.createNS("http://www.w3.org/2000/svg","g");a.setAttributeNS(null,"fill-rule","nonzero");const o=F.createNS("http://www.w3.org/2000/svg","g");o.setAttributeNS(null,"transform","translate(3.0, 29.0)"),o.setAttributeNS(null,"fill","#000000");const l=[{rx:"10.5",ry:"5.25002273"},{rx:"10.5",ry:"5.25002273"},{rx:"9.5",ry:"4.77275007"},{rx:"8.5",ry:"4.29549936"},{rx:"7.5",ry:"3.81822308"},{rx:"6.5",ry:"3.34094679"},{rx:"5.5",ry:"2.86367051"},{rx:"4.5",ry:"2.38636864"}];for(const t of l){const e=F.createNS("http://www.w3.org/2000/svg","ellipse");e.setAttributeNS(null,"opacity","0.04"),e.setAttributeNS(null,"cx","10.5"),e.setAttributeNS(null,"cy","5.80029008"),e.setAttributeNS(null,"rx",t.rx),e.setAttributeNS(null,"ry",t.ry),o.appendChild(e)}const c=F.createNS("http://www.w3.org/2000/svg","g");c.setAttributeNS(null,"fill",this._color);const h=F.createNS("http://www.w3.org/2000/svg","path");h.setAttributeNS(null,"d","M27,13.5 C27,19.074644 20.250001,27.000002 14.75,34.500002 C14.016665,35.500004 12.983335,35.500004 12.25,34.500002 C6.7499993,27.000002 0,19.222562 0,13.5 C0,6.0441559 6.0441559,0 13.5,0 C20.955844,0 27,6.0441559 27,13.5 Z"),c.appendChild(h);const u=F.createNS("http://www.w3.org/2000/svg","g");u.setAttributeNS(null,"opacity","0.25"),u.setAttributeNS(null,"fill","#000000");const p=F.createNS("http://www.w3.org/2000/svg","path");p.setAttributeNS(null,"d","M13.5,0 C6.0441559,0 0,6.0441559 0,13.5 C0,19.222562 6.7499993,27 12.25,34.5 C13,35.522727 14.016664,35.500004 14.75,34.5 C20.250001,27 27,19.074644 27,13.5 C27,6.0441559 20.955844,0 13.5,0 Z M13.5,1 C20.415404,1 26,6.584596 26,13.5 C26,15.898657 24.495584,19.181431 22.220703,22.738281 C19.945823,26.295132 16.705119,30.142167 13.943359,33.908203 C13.743445,34.180814 13.612715,34.322738 13.5,34.441406 C13.387285,34.322738 13.256555,34.180814 13.056641,33.908203 C10.284481,30.127985 7.4148684,26.314159 5.015625,22.773438 C2.6163816,19.232715 1,15.953538 1,13.5 C1,6.584596 6.584596,1 13.5,1 Z"),u.appendChild(p);const d=F.createNS("http://www.w3.org/2000/svg","g");d.setAttributeNS(null,"transform","translate(6.0, 7.0)"),d.setAttributeNS(null,"fill","#FFFFFF");const m=F.createNS("http://www.w3.org/2000/svg","g");m.setAttributeNS(null,"transform","translate(8.0, 8.0)");const f=F.createNS("http://www.w3.org/2000/svg","circle");f.setAttributeNS(null,"fill","#000000"),f.setAttributeNS(null,"opacity","0.25"),f.setAttributeNS(null,"cx","5.5"),f.setAttributeNS(null,"cy","5.5"),f.setAttributeNS(null,"r","5.4999962");const _=F.createNS("http://www.w3.org/2000/svg","circle");_.setAttributeNS(null,"fill","#FFFFFF"),_.setAttributeNS(null,"cx","5.5"),_.setAttributeNS(null,"cy","5.5"),_.setAttributeNS(null,"r","5.4999962"),m.appendChild(f),m.appendChild(_),a.appendChild(o),a.appendChild(c),a.appendChild(u),a.appendChild(d),a.appendChild(m),e.appendChild(a),e.setAttributeNS(null,"height",i*this._scale+"px"),e.setAttributeNS(null,"width",r*this._scale+"px"),this._element.appendChild(e),this._offset=n.convert(t&&t.offset||[0,-14])}if(this._element.classList.add("maplibregl-marker"),this._element.addEventListener("dragstart",(t=>{t.preventDefault()})),this._element.addEventListener("mousedown",(t=>{t.preventDefault()})),e_(this._element,this._anchor,"marker"),t&&t.className)for(const e of t.className.split(" "))this._element.classList.add(e);this._popup=null}addTo(t){return this.remove(),this._map=t,t.getCanvasContainer().appendChild(this._element),t.on("move",this._update),t.on("moveend",this._update),t.on("terrain",this._update),this.setDraggable(this._draggable),this._update(),this._map.on("click",this._onMapClick),this}remove(){return this._opacityTimeout&&(clearTimeout(this._opacityTimeout),delete this._opacityTimeout),this._map&&(this._map.off("click",this._onMapClick),this._map.off("move",this._update),this._map.off("moveend",this._update),this._map.off("mousedown",this._addDragHandler),this._map.off("touchstart",this._addDragHandler),this._map.off("mouseup",this._onUp),this._map.off("touchend",this._onUp),this._map.off("mousemove",this._onMove),this._map.off("touchmove",this._onMove),delete this._map),F.remove(this._element),this._popup&&this._popup.remove(),this}getLngLat(){return this._lngLat}setLngLat(t){return this._lngLat=Eh.convert(t),this._pos=null,this._popup&&this._popup.setLngLat(this._lngLat),this._update(),this}getElement(){return this._element}setPopup(t){if(this._popup&&(this._popup.remove(),this._popup=null,this._element.removeEventListener("keypress",this._onKeyPress),this._originalTabIndex||this._element.removeAttribute("tabindex")),t){if(!("offset"in t.options)){const e=38.1,i=13.5,r=Math.abs(i)/Math.SQRT2;t.options.offset=this._defaultMarker?{top:[0,0],"top-left":[0,0],"top-right":[0,0],bottom:[0,-e],"bottom-left":[r,-1*(e-i+r)],"bottom-right":[-r,-1*(e-i+r)],left:[i,-1*(e-i)],right:[-i,-1*(e-i)]}:this._offset}this._popup=t,this._lngLat&&this._popup.setLngLat(this._lngLat),this._originalTabIndex=this._element.getAttribute("tabindex"),this._originalTabIndex||this._element.setAttribute("tabindex","0"),this._element.addEventListener("keypress",this._onKeyPress)}return this}getPopup(){return this._popup}togglePopup(){const t=this._popup;return t?(t.isOpen()?t.remove():t.addTo(this._map),this):this}getOffset(){return this._offset}setOffset(t){return this._offset=n.convert(t),this._update(),this}addClassName(t){this._element.classList.add(t)}removeClassName(t){this._element.classList.remove(t)}toggleClassName(t){return this._element.classList.toggle(t)}setDraggable(t){return this._draggable=!!t,this._map&&(t?(this._map.on("mousedown",this._addDragHandler),this._map.on("touchstart",this._addDragHandler)):(this._map.off("mousedown",this._addDragHandler),this._map.off("touchstart",this._addDragHandler))),this}isDraggable(){return this._draggable}setRotation(t){return this._rotation=t||0,this._update(),this}getRotation(){return this._rotation}setRotationAlignment(t){return this._rotationAlignment=t||"auto",this._update(),this}getRotationAlignment(){return this._rotationAlignment}setPitchAlignment(t){return this._pitchAlignment=t&&"auto"!==t?t:this._rotationAlignment,this._update(),this}getPitchAlignment(){return this._pitchAlignment}}const r_={positionOptions:{enableHighAccuracy:!1,maximumAge:0,timeout:6e3},fitBoundsOptions:{maxZoom:15},trackUserLocation:!1,showAccuracyCircle:!0,showUserLocation:!0};let s_=0,n_=!1;const a_={maxWidth:100,unit:"metric"};function o_(t,e,i){const r=i&&i.maxWidth||100,s=t._container.clientHeight/2,n=t.unproject([0,s]),a=t.unproject([r,s]),o=n.distanceTo(a);if(i&&"imperial"===i.unit){const i=3.2808*o;i>5280?l_(e,r,i/5280,t._getUIString("ScaleControl.Miles")):l_(e,r,i,t._getUIString("ScaleControl.Feet"))}else i&&"nautical"===i.unit?l_(e,r,o/1852,t._getUIString("ScaleControl.NauticalMiles")):o>=1e3?l_(e,r,o/1e3,t._getUIString("ScaleControl.Kilometers")):l_(e,r,o,t._getUIString("ScaleControl.Meters"))}function l_(t,e,i,r){const s=function(t){const e=Math.pow(10,`${Math.floor(t)}`.length-1);let i=t/e;return i=i>=10?10:i>=5?5:i>=3?3:i>=2?2:i>=1?1:function(t){const e=Math.pow(10,Math.ceil(-Math.log(t)/Math.LN10));return Math.round(t*e)/e}(i),e*i}(i);t.style.width=e*(s/i)+"px",t.innerHTML=`${s} ${r}`}const c_={closeButton:!0,closeOnClick:!0,focusAfterOpen:!0,className:"",maxWidth:"240px"},h_=["a[href]","[tabindex]:not([tabindex='-1'])","[contenteditable]:not([contenteditable='false'])","button:not([disabled])","input:not([disabled])","select:not([disabled])","textarea:not([disabled])"].join(", ");function u_(t){if(t){if("number"==typeof t){const e=Math.round(Math.abs(t)/Math.SQRT2);return{center:new n(0,0),top:new n(0,t),"top-left":new n(e,e),"top-right":new n(-e,e),bottom:new n(0,-t),"bottom-left":new n(e,-e),"bottom-right":new n(-e,-e),left:new n(t,0),right:new n(-t,0)}}if(t instanceof n||Array.isArray(t)){const e=n.convert(t);return{center:e,top:e,"top-left":e,"top-right":e,bottom:e,"bottom-left":e,"bottom-right":e,left:e,right:e}}return{center:n.convert(t.center||[0,0]),top:n.convert(t.top||[0,0]),"top-left":n.convert(t["top-left"]||[0,0]),"top-right":n.convert(t["top-right"]||[0,0]),bottom:n.convert(t.bottom||[0,0]),"bottom-left":n.convert(t["bottom-left"]||[0,0]),"bottom-right":n.convert(t["bottom-right"]||[0,0]),left:n.convert(t.left||[0,0]),right:n.convert(t.right||[0,0])}}return u_(new n(0,0))}const p_={extend:(t,...e)=>g(t,...e),run(t){t()},logToElement(t,e=!1,i="log"){const r=window.document.getElementById(i);r&&(e&&(r.innerHTML=""),r.innerHTML+=`
${t}`)}},d_=t;class m_{static get version(){return d_}static get workerCount(){return mu.workerCount}static set workerCount(t){mu.workerCount=t}static get maxParallelImageRequests(){return O.MAX_PARALLEL_IMAGE_REQUESTS}static set maxParallelImageRequests(t){O.MAX_PARALLEL_IMAGE_REQUESTS=t}static get workerUrl(){return O.WORKER_URL}static set workerUrl(t){O.WORKER_URL=t}static addProtocol(t,e){O.REGISTERED_PROTOCOLS[t]=e}static removeProtocol(t){delete O.REGISTERED_PROTOCOLS[t]}}return m_.Map=class extends zf{constructor(t){if(Vf.mark(Df.create),null!=(t=g({},Wf,t)).minZoom&&null!=t.maxZoom&&t.minZoom>t.maxZoom)throw new Error("maxZoom must be greater than or equal to minZoom");if(null!=t.minPitch&&null!=t.maxPitch&&t.minPitch>t.maxPitch)throw new Error("maxPitch must be greater than or equal to minPitch");if(null!=t.minPitch&&t.minPitch<0)throw new Error("minPitch must be greater than or equal to 0");if(null!=t.maxPitch&&t.maxPitch>85)throw new Error("maxPitch must be less than or equal to 85");if(super(new zm(t.minZoom,t.maxZoom,t.minPitch,t.maxPitch,t.renderWorldCopies),{bearingSnap:t.bearingSnap}),this._cooperativeGesturesOnWheel=t=>{this._onCooperativeGesture(t,t[this._metaKey],1)},this._contextLost=t=>{t.preventDefault(),this._frame&&(this._frame.cancel(),this._frame=null),this.fire(new nt("webglcontextlost",{originalEvent:t}))},this._contextRestored=t=>{this._setupPainter(),this.resize(),this._update(),this.fire(new nt("webglcontextrestored",{originalEvent:t}))},this._onMapScroll=t=>{if(t.target===this._container)return this._container.scrollTop=0,this._container.scrollLeft=0,!1},this._onWindowOnline=()=>{this._update()},this._interactive=t.interactive,this._cooperativeGestures=t.cooperativeGestures,this._metaKey=0===navigator.platform.indexOf("Mac")?"metaKey":"ctrlKey",this._maxTileCacheSize=t.maxTileCacheSize,this._maxTileCacheZoomLevels=t.maxTileCacheZoomLevels,this._failIfMajorPerformanceCaveat=t.failIfMajorPerformanceCaveat,this._preserveDrawingBuffer=t.preserveDrawingBuffer,this._antialias=t.antialias,this._trackResize=t.trackResize,this._bearingSnap=t.bearingSnap,this._refreshExpiredTiles=t.refreshExpiredTiles,this._fadeDuration=t.fadeDuration,this._crossSourceCollisions=t.crossSourceCollisions,this._crossFadingFactor=1,this._collectResourceTiming=t.collectResourceTiming,this._renderTaskQueue=new kf,this._controls=[],this._mapId=v(),this._locale=g({},Uf,t.locale),this._clickTolerance=t.clickTolerance,this._overridePixelRatio=t.pixelRatio,this._maxCanvasSize=t.maxCanvasSize,this.transformCameraUpdate=t.transformCameraUpdate,this._imageQueueHandle=Q.addThrottleControl((()=>this.isMoving())),this._requestManager=new et(t.transformRequest),"string"==typeof t.container){if(this._container=document.getElementById(t.container),!this._container)throw new Error(`Container '${t.container}' not found.`)}else{if(!(t.container instanceof HTMLElement))throw new Error("Invalid type: 'container' must be a String or HTMLElement.");this._container=t.container}if(t.maxBounds&&this.setMaxBounds(t.maxBounds),this._setupContainer(),this._setupPainter(),this.on("move",(()=>this._update(!1))),this.on("moveend",(()=>this._update(!1))),this.on("zoom",(()=>this._update(!0))),this.on("terrain",(()=>{this.painter.terrainFacilitator.dirty=!0,this._update(!0)})),this.once("idle",(()=>{this._idleTriggered=!0})),"undefined"!=typeof window){addEventListener("online",this._onWindowOnline,!1);let t=!1;const e=Mm((t=>{this._trackResize&&!this._removed&&this.resize(t)._update()}),50);this._resizeObserver=new ResizeObserver((i=>{t?e(i):t=!0})),this._resizeObserver.observe(this._container)}this.handlers=new Cf(this,t),this._cooperativeGestures&&this._setupCooperativeGestures(),this._hash=t.hash&&new Pm("string"==typeof t.hash&&t.hash||void 0).addTo(this),this._hash&&this._hash._onHashChange()||(this.jumpTo({center:t.center,zoom:t.zoom,bearing:t.bearing,pitch:t.pitch}),t.bounds&&(this.resize(),this.fitBounds(t.bounds,g({},t.fitBoundsOptions,{duration:0})))),this.resize(),this._localIdeographFontFamily=t.localIdeographFontFamily,this._validateStyle=t.validateStyle,t.style&&this.setStyle(t.style,{localIdeographFontFamily:t.localIdeographFontFamily}),t.attributionControl&&this.addControl(new Mf({customAttribution:t.customAttribution})),t.maplibreLogo&&this.addControl(new Pf,t.logoPosition),this.on("style.load",(()=>{this.transform.unmodified&&this.jumpTo(this.style.stylesheet)})),this.on("data",(t=>{this._update("style"===t.dataType),this.fire(new nt(`${t.dataType}data`,t))})),this.on("dataloading",(t=>{this.fire(new nt(`${t.dataType}dataloading`,t))})),this.on("dataabort",(t=>{this.fire(new nt("sourcedataabort",t))}))}_getMapId(){return this._mapId}addControl(t,e){if(void 0===e&&(e=t.getDefaultPosition?t.getDefaultPosition():"top-right"),!t||!t.onAdd)return this.fire(new at(new Error("Invalid argument to map.addControl(). Argument must be a control with onAdd and onRemove methods.")));const i=t.onAdd(this);this._controls.push(t);const r=this._controlPositions[e];return-1!==e.indexOf("bottom")?r.insertBefore(i,r.firstChild):r.appendChild(i),this}removeControl(t){if(!t||!t.onRemove)return this.fire(new at(new Error("Invalid argument to map.removeControl(). Argument must be a control with onAdd and onRemove methods.")));const e=this._controls.indexOf(t);return e>-1&&this._controls.splice(e,1),t.onRemove(this),this}hasControl(t){return this._controls.indexOf(t)>-1}calculateCameraOptionsFromTo(t,e,i,r){return null==r&&this.terrain&&(r=this.terrain.getElevationForLngLatZoom(i,this.transform.tileZoom)),super.calculateCameraOptionsFromTo(t,e,i,r)}resize(t){var e;const i=this._containerDimensions(),r=i[0],s=i[1],n=this._getClampedPixelRatio(r,s);if(this._resizeCanvas(r,s,n),this.painter.resize(r,s,n),this.painter.overLimit()){const t=this.painter.context.gl;this._maxCanvasSize=[t.drawingBufferWidth,t.drawingBufferHeight];const e=this._getClampedPixelRatio(r,s);this._resizeCanvas(r,s,e),this.painter.resize(r,s,e)}this.transform.resize(r,s),null===(e=this._requestedCameraState)||void 0===e||e.resize(r,s);const a=!this._moving;return a&&(this.stop(),this.fire(new nt("movestart",t)).fire(new nt("move",t))),this.fire(new nt("resize",t)),a&&this.fire(new nt("moveend",t)),this}_getClampedPixelRatio(t,e){const{0:i,1:r}=this._maxCanvasSize,s=this.getPixelRatio(),n=t*s,a=e*s;return Math.min(n>i?i/n:1,a>r?r/a:1)*s}getPixelRatio(){var t;return null!==(t=this._overridePixelRatio)&&void 0!==t?t:devicePixelRatio}setPixelRatio(t){this._overridePixelRatio=t,this.resize()}getBounds(){return this.transform.getBounds()}getMaxBounds(){return this.transform.getMaxBounds()}setMaxBounds(t){return this.transform.setMaxBounds(Ch.convert(t)),this._update()}setMinZoom(t){if((t=null==t?-2:t)>=-2&&t<=this.transform.maxZoom)return this.transform.minZoom=t,this._update(),this.getZoom()=this.transform.minZoom)return this.transform.maxZoom=t,this._update(),this.getZoom()>t&&this.setZoom(t),this;throw new Error("maxZoom must be greater than the current minZoom")}getMaxZoom(){return this.transform.maxZoom}setMinPitch(t){if((t=null==t?0:t)<0)throw new Error("minPitch must be greater than or equal to 0");if(t>=0&&t<=this.transform.maxPitch)return this.transform.minPitch=t,this._update(),this.getPitch()85)throw new Error("maxPitch must be less than or equal to 85");if(t>=this.transform.minPitch)return this.transform.maxPitch=t,this._update(),this.getPitch()>t&&this.setPitch(t),this;throw new Error("maxPitch must be greater than the current minPitch")}getMaxPitch(){return this.transform.maxPitch}getRenderWorldCopies(){return this.transform.renderWorldCopies}setRenderWorldCopies(t){return this.transform.renderWorldCopies=t,this._update()}getCooperativeGestures(){return this._cooperativeGestures}setCooperativeGestures(t){return this._cooperativeGestures=t,this._cooperativeGestures?this._setupCooperativeGestures():this._destroyCooperativeGestures(),this}project(t){return this.transform.locationPoint(Eh.convert(t),this.style&&this.terrain)}unproject(t){return this.transform.pointLocation(n.convert(t),this.terrain)}isMoving(){var t;return this._moving||(null===(t=this.handlers)||void 0===t?void 0:t.isMoving())}isZooming(){var t;return this._zooming||(null===(t=this.handlers)||void 0===t?void 0:t.isZooming())}isRotating(){var t;return this._rotating||(null===(t=this.handlers)||void 0===t?void 0:t.isRotating())}_createDelegatedListener(t,e,i){if("mouseenter"===t||"mouseover"===t){let r=!1;const s=s=>{const n=this.getLayer(e)?this.queryRenderedFeatures(s.point,{layers:[e]}):[];n.length?r||(r=!0,i.call(this,new Um(t,this,s.originalEvent,{features:n}))):r=!1};return{layer:e,listener:i,delegates:{mousemove:s,mouseout:()=>{r=!1}}}}if("mouseleave"===t||"mouseout"===t){let r=!1;const s=s=>{(this.getLayer(e)?this.queryRenderedFeatures(s.point,{layers:[e]}):[]).length?r=!0:r&&(r=!1,i.call(this,new Um(t,this,s.originalEvent)))},n=e=>{r&&(r=!1,i.call(this,new Um(t,this,e.originalEvent)))};return{layer:e,listener:i,delegates:{mousemove:s,mouseout:n}}}{const r=t=>{const r=this.getLayer(e)?this.queryRenderedFeatures(t.point,{layers:[e]}):[];r.length&&(t.features=r,i.call(this,t),delete t.features)};return{layer:e,listener:i,delegates:{[t]:r}}}}on(t,e,i){if(void 0===i)return super.on(t,e);const r=this._createDelegatedListener(t,e,i);this._delegatedListeners=this._delegatedListeners||{},this._delegatedListeners[t]=this._delegatedListeners[t]||[],this._delegatedListeners[t].push(r);for(const t in r.delegates)this.on(t,r.delegates[t]);return this}once(t,e,i){if(void 0===i)return super.once(t,e);const r=this._createDelegatedListener(t,e,i);for(const t in r.delegates)this.once(t,r.delegates[t]);return this}off(t,e,i){return void 0===i?super.off(t,e):(this._delegatedListeners&&this._delegatedListeners[t]&&(r=>{const s=this._delegatedListeners[t];for(let t=0;tthis._updateStyle(t,e)));const i=this.style&&e.transformStyle?this.style.serialize():void 0;return this.style&&(this.style.setEventedParent(null),this.style._remove(!t)),t?(this.style=new Ap(this,e||{}),this.style.setEventedParent(this,{style:this.style}),"string"==typeof t?this.style.loadURL(t,e,i):this.style.loadJSON(t,e,i),this):(delete this.style,this)}_lazyInitEmptyStyle(){this.style||(this.style=new Ap(this,{}),this.style.setEventedParent(this,{style:this.style}),this.style.loadEmpty())}_diffStyle(t,e){if("string"==typeof t){const i=this._requestManager.transformRequest(t,tt.Style);Z(i,((t,i)=>{t?this.fire(new at(t)):i&&this._updateDiff(i,e)}))}else"object"==typeof t&&this._updateDiff(t,e)}_updateDiff(t,e){try{this.style.setState(t,e)&&this._update(!0)}catch(i){A(`Unable to perform style diff: ${i.message||i.error||i}. Rebuilding the style from scratch.`),this._updateStyle(t,e)}}getStyle(){if(this.style)return this.style.serialize()}isStyleLoaded(){return this.style?this.style.loaded():A("There is no style added to the map.")}addSource(t,e){return this._lazyInitEmptyStyle(),this.style.addSource(t,e),this._update(!0)}isSourceLoaded(t){const e=this.style&&this.style.sourceCaches[t];if(void 0!==e)return e.loaded();this.fire(new at(new Error(`There is no source with ID '${t}'`)))}setTerrain(t){if(this.style._checkLoaded(),this._terrainDataCallback&&this.style.off("data",this._terrainDataCallback),t){const e=this.style.sourceCaches[t.source];if(!e)throw new Error(`cannot load terrain, because there exists no source with ID: ${t.source}`);for(const e in this.style._layers){const i=this.style._layers[e];"hillshade"===i.type&&i.source===t.source&&A("You are using the same source for a hillshade layer and for 3D terrain. Please consider using two separate sources to improve rendering quality.")}this.terrain=new qf(this.painter,e,t),this.painter.renderToTexture=new Gf(this.painter,this.terrain),this.transform._minEleveationForCurrentTile=this.terrain.getMinTileElevationForLngLatZoom(this.transform.center,this.transform.tileZoom),this.transform.elevation=this.terrain.getElevationForLngLatZoom(this.transform.center,this.transform.tileZoom),this._terrainDataCallback=e=>{"style"===e.dataType?this.terrain.sourceCache.freeRtt():"source"===e.dataType&&e.tile&&(e.sourceId!==t.source||this._elevationFreeze||(this.transform._minEleveationForCurrentTile=this.terrain.getMinTileElevationForLngLatZoom(this.transform.center,this.transform.tileZoom),this.transform.elevation=this.terrain.getElevationForLngLatZoom(this.transform.center,this.transform.tileZoom)),this.terrain.sourceCache.freeRtt(e.tile.tileID))},this.style.on("data",this._terrainDataCallback)}else this.terrain&&this.terrain.sourceCache.destruct(),this.terrain=null,this.painter.renderToTexture&&this.painter.renderToTexture.destruct(),this.painter.renderToTexture=null,this.transform._minEleveationForCurrentTile=0,this.transform.elevation=0;return this.fire(new nt("terrain",{terrain:t})),this}getTerrain(){var t,e;return null!==(e=null===(t=this.terrain)||void 0===t?void 0:t.options)&&void 0!==e?e:null}areTilesLoaded(){const t=this.style&&this.style.sourceCaches;for(const e in t){const i=t[e]._tiles;for(const t in i){const e=i[t];if("loaded"!==e.state&&"errored"!==e.state)return!1}}return!0}addSourceType(t,e,i){return this._lazyInitEmptyStyle(),this.style.addSourceType(t,e,i)}removeSource(t){return this.style.removeSource(t),this._update(!0)}getSource(t){return this.style.getSource(t)}addImage(t,e,i={}){const{pixelRatio:r=1,sdf:s=!1,stretchX:n,stretchY:a,content:o}=i;if(this._lazyInitEmptyStyle(),!(e instanceof HTMLImageElement||k(e))){if(void 0===e.width||void 0===e.height)return this.fire(new at(new Error("Invalid arguments to map.addImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, or object with `width`, `height`, and `data` properties with the same format as `ImageData`")));{const{width:i,height:l,data:c}=e,h=e;return this.style.addImage(t,{data:new _o({width:i,height:l},new Uint8Array(c)),pixelRatio:r,stretchX:n,stretchY:a,content:o,sdf:s,version:0,userImage:h}),h.onAdd&&h.onAdd(this,t),this}}{const{width:i,height:l,data:c}=B.getImageData(e);this.style.addImage(t,{data:new _o({width:i,height:l},c),pixelRatio:r,stretchX:n,stretchY:a,content:o,sdf:s,version:0})}}updateImage(t,e){const i=this.style.getImage(t);if(!i)return this.fire(new at(new Error("The map has no image with that id. If you are adding a new image use `map.addImage(...)` instead.")));const r=e instanceof HTMLImageElement||k(e)?B.getImageData(e):e,{width:s,height:n,data:a}=r;if(void 0===s||void 0===n)return this.fire(new at(new Error("Invalid arguments to map.updateImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, or object with `width`, `height`, and `data` properties with the same format as `ImageData`")));if(s!==i.data.width||n!==i.data.height)return this.fire(new at(new Error("The width and height of the updated image must be that same as the previous version of the image")));const o=!(e instanceof HTMLImageElement||k(e));return i.data.replace(a,o),this.style.updateImage(t,i),this}getImage(t){return this.style.getImage(t)}hasImage(t){return t?!!this.style.getImage(t):(this.fire(new at(new Error("Missing required image id"))),!1)}removeImage(t){this.style.removeImage(t)}loadImage(t,e){Q.getImage(this._requestManager.transformRequest(t,tt.Image),e)}listImages(){return this.style.listImages()}addLayer(t,e){return this._lazyInitEmptyStyle(),this.style.addLayer(t,e),this._update(!0)}moveLayer(t,e){return this.style.moveLayer(t,e),this._update(!0)}removeLayer(t){return this.style.removeLayer(t),this._update(!0)}getLayer(t){return this.style.getLayer(t)}getLayersOrder(){return this.style.getLayersOrder()}setLayerZoomRange(t,e,i){return this.style.setLayerZoomRange(t,e,i),this._update(!0)}setFilter(t,e,i={}){return this.style.setFilter(t,e,i),this._update(!0)}getFilter(t){return this.style.getFilter(t)}setPaintProperty(t,e,i,r={}){return this.style.setPaintProperty(t,e,i,r),this._update(!0)}getPaintProperty(t,e){return this.style.getPaintProperty(t,e)}setLayoutProperty(t,e,i,r={}){return this.style.setLayoutProperty(t,e,i,r),this._update(!0)}getLayoutProperty(t,e){return this.style.getLayoutProperty(t,e)}setGlyphs(t,e={}){return this._lazyInitEmptyStyle(),this.style.setGlyphs(t,e),this._update(!0)}getGlyphs(){return this.style.getGlyphsUrl()}addSprite(t,e,i={}){return this._lazyInitEmptyStyle(),this.style.addSprite(t,e,i,(t=>{t||this._update(!0)})),this}removeSprite(t){return this._lazyInitEmptyStyle(),this.style.removeSprite(t),this._update(!0)}getSprite(){return this.style.getSprite()}setSprite(t,e={}){return this._lazyInitEmptyStyle(),this.style.setSprite(t,e,(t=>{t||this._update(!0)})),this}setLight(t,e={}){return this._lazyInitEmptyStyle(),this.style.setLight(t,e),this._update(!0)}getLight(){return this.style.getLight()}setFeatureState(t,e){return this.style.setFeatureState(t,e),this._update()}removeFeatureState(t,e){return this.style.removeFeatureState(t,e),this._update()}getFeatureState(t){return this.style.getFeatureState(t)}getContainer(){return this._container}getCanvasContainer(){return this._canvasContainer}getCanvas(){return this._canvas}_containerDimensions(){let t=0,e=0;return this._container&&(t=this._container.clientWidth||400,e=this._container.clientHeight||300),[t,e]}_setupContainer(){const t=this._container;t.classList.add("maplibregl-map");const e=this._canvasContainer=F.create("div","maplibregl-canvas-container",t);this._interactive&&e.classList.add("maplibregl-interactive"),this._canvas=F.create("canvas","maplibregl-canvas",e),this._canvas.addEventListener("webglcontextlost",this._contextLost,!1),this._canvas.addEventListener("webglcontextrestored",this._contextRestored,!1),this._canvas.setAttribute("tabindex","0"),this._canvas.setAttribute("aria-label","Map"),this._canvas.setAttribute("role","region");const i=this._containerDimensions(),r=this._getClampedPixelRatio(i[0],i[1]);this._resizeCanvas(i[0],i[1],r);const s=this._controlContainer=F.create("div","maplibregl-control-container",t),n=this._controlPositions={};["top-left","top-right","bottom-left","bottom-right"].forEach((t=>{n[t]=F.create("div",`maplibregl-ctrl-${t} `,s)})),this._container.addEventListener("scroll",this._onMapScroll,!1)}_setupCooperativeGestures(){this._cooperativeGesturesScreen=F.create("div","maplibregl-cooperative-gesture-screen",this._container);let t="boolean"!=typeof this._cooperativeGestures&&this._cooperativeGestures.windowsHelpText?this._cooperativeGestures.windowsHelpText:"Use Ctrl + scroll to zoom the map";0===navigator.platform.indexOf("Mac")&&(t="boolean"!=typeof this._cooperativeGestures&&this._cooperativeGestures.macHelpText?this._cooperativeGestures.macHelpText:"Use ⌘ + scroll to zoom the map"),this._cooperativeGesturesScreen.innerHTML=`\n
${t}
\n
${"boolean"!=typeof this._cooperativeGestures&&this._cooperativeGestures.mobileHelpText?this._cooperativeGestures.mobileHelpText:"Use two fingers to move the map"}
\n `,this._cooperativeGesturesScreen.setAttribute("aria-hidden","true"),this._canvasContainer.addEventListener("wheel",this._cooperativeGesturesOnWheel,!1),this._canvasContainer.classList.add("maplibregl-cooperative-gestures")}_destroyCooperativeGestures(){F.remove(this._cooperativeGesturesScreen),this._canvasContainer.removeEventListener("wheel",this._cooperativeGesturesOnWheel,!1),this._canvasContainer.classList.remove("maplibregl-cooperative-gestures")}_resizeCanvas(t,e,i){this._canvas.width=Math.floor(i*t),this._canvas.height=Math.floor(i*e),this._canvas.style.width=`${t}px`,this._canvas.style.height=`${e}px`}_setupPainter(){const t={alpha:!0,stencil:!0,depth:!0,failIfMajorPerformanceCaveat:this._failIfMajorPerformanceCaveat,preserveDrawingBuffer:this._preserveDrawingBuffer,antialias:this._antialias||!1};let e=null;this._canvas.addEventListener("webglcontextcreationerror",(i=>{e={requestedAttributes:t},i&&(e.statusMessage=i.statusMessage,e.type=i.type)}),{once:!0});const i=this._canvas.getContext("webgl2",t)||this._canvas.getContext("webgl",t);if(!i){const t="Failed to initialize WebGL";throw e?(e.message=t,new Error(JSON.stringify(e))):new Error(t)}this.painter=new Im(i,this.transform),H.testSupport(i)}_onCooperativeGesture(t,e,i){return!e&&i<2&&(this._cooperativeGesturesScreen.classList.add("maplibregl-show"),setTimeout((()=>{this._cooperativeGesturesScreen.classList.remove("maplibregl-show")}),100)),!1}loaded(){return!this._styleDirty&&!this._sourcesDirty&&!!this.style&&this.style.loaded()}_update(t){return this.style&&this.style._loaded?(this._styleDirty=this._styleDirty||t,this._sourcesDirty=!0,this.triggerRepaint(),this):this}_requestRenderFrame(t){return this._update(),this._renderTaskQueue.add(t)}_cancelRenderFrame(t){this._renderTaskQueue.remove(t)}_render(t){const e=this._idleTriggered?this._fadeDuration:0;if(this.painter.context.setDirty(),this.painter.setBaseState(),this._renderTaskQueue.run(t),this._removed)return;let i=!1;if(this.style&&this._styleDirty){this._styleDirty=!1;const t=this.transform.zoom,r=B.now();this.style.zoomHistory.update(t,r);const s=new _s(t,{now:r,fadeDuration:e,zoomHistory:this.style.zoomHistory,transition:this.style.getTransition()}),n=s.crossFadingFactor();1===n&&n===this._crossFadingFactor||(i=!0,this._crossFadingFactor=n),this.style.update(s)}this.style&&this._sourcesDirty&&(this._sourcesDirty=!1,this.style._updateSources(this.transform)),this.terrain?(this.terrain.sourceCache.update(this.transform,this.terrain),this.transform._minEleveationForCurrentTile=this.terrain.getMinTileElevationForLngLatZoom(this.transform.center,this.transform.tileZoom),this._elevationFreeze||(this.transform.elevation=this.terrain.getElevationForLngLatZoom(this.transform.center,this.transform.tileZoom))):(this.transform._minEleveationForCurrentTile=0,this.transform.elevation=0),this._placementDirty=this.style&&this.style._updatePlacement(this.painter.transform,this.showCollisionBoxes,e,this._crossSourceCollisions),this.painter.render(this.style,{showTileBoundaries:this.showTileBoundaries,showOverdrawInspector:this._showOverdrawInspector,rotating:this.isRotating(),zooming:this.isZooming(),moving:this.isMoving(),fadeDuration:e,showPadding:this.showPadding}),this.fire(new nt("render")),this.loaded()&&!this._loaded&&(this._loaded=!0,Vf.mark(Df.load),this.fire(new nt("load"))),this.style&&(this.style.hasTransitions()||i)&&(this._styleDirty=!0),this.style&&!this._placementDirty&&this.style._releaseSymbolFadeTiles();const r=this._sourcesDirty||this._styleDirty||this._placementDirty;return r||this._repaint?this.triggerRepaint():!this.isMoving()&&this.loaded()&&this.fire(new nt("idle")),!this._loaded||this._fullyLoaded||r||(this._fullyLoaded=!0,Vf.mark(Df.fullLoad)),this}redraw(){return this.style&&(this._frame&&(this._frame.cancel(),this._frame=null),this._render(0)),this}remove(){var t;this._hash&&this._hash.remove();for(const t of this._controls)t.onRemove(this);this._controls=[],this._frame&&(this._frame.cancel(),this._frame=null),this._renderTaskQueue.clear(),this.painter.destroy(),this.handlers.destroy(),delete this.handlers,this.setStyle(null),"undefined"!=typeof window&&removeEventListener("online",this._onWindowOnline,!1),Q.removeThrottleControl(this._imageQueueHandle),null===(t=this._resizeObserver)||void 0===t||t.disconnect();const e=this.painter.context.gl.getExtension("WEBGL_lose_context");e&&e.loseContext(),this._canvas.removeEventListener("webglcontextrestored",this._contextRestored,!1),this._canvas.removeEventListener("webglcontextlost",this._contextLost,!1),F.remove(this._canvasContainer),F.remove(this._controlContainer),this._cooperativeGestures&&this._destroyCooperativeGestures(),this._container.classList.remove("maplibregl-map"),Vf.clearMetrics(),this._removed=!0,this.fire(new nt("remove"))}triggerRepaint(){this.style&&!this._frame&&(this._frame=B.frame((t=>{Vf.frame(t),this._frame=null,this._render(t)})))}get showTileBoundaries(){return!!this._showTileBoundaries}set showTileBoundaries(t){this._showTileBoundaries!==t&&(this._showTileBoundaries=t,this._update())}get showPadding(){return!!this._showPadding}set showPadding(t){this._showPadding!==t&&(this._showPadding=t,this._update())}get showCollisionBoxes(){return!!this._showCollisionBoxes}set showCollisionBoxes(t){this._showCollisionBoxes!==t&&(this._showCollisionBoxes=t,t?this.style._generateCollisionBoxes():this._update())}get showOverdrawInspector(){return!!this._showOverdrawInspector}set showOverdrawInspector(t){this._showOverdrawInspector!==t&&(this._showOverdrawInspector=t,this._update())}get repaint(){return!!this._repaint}set repaint(t){this._repaint!==t&&(this._repaint=t,this.triggerRepaint())}get vertices(){return!!this._vertices}set vertices(t){this._vertices=t,this._update()}get version(){return Hf}getCameraTargetElevation(){return this.transform.elevation}},m_.NavigationControl=class{constructor(t){this._updateZoomButtons=()=>{const t=this._map.getZoom(),e=t===this._map.getMaxZoom(),i=t===this._map.getMinZoom();this._zoomInButton.disabled=e,this._zoomOutButton.disabled=i,this._zoomInButton.setAttribute("aria-disabled",e.toString()),this._zoomOutButton.setAttribute("aria-disabled",i.toString())},this._rotateCompassArrow=()=>{const t=this.options.visualizePitch?`scale(${1/Math.pow(Math.cos(this._map.transform.pitch*(Math.PI/180)),.5)}) rotateX(${this._map.transform.pitch}deg) rotateZ(${this._map.transform.angle*(180/Math.PI)}deg)`:`rotate(${this._map.transform.angle*(180/Math.PI)}deg)`;this._compassIcon.style.transform=t},this._setButtonTitle=(t,e)=>{const i=this._map._getUIString(`NavigationControl.${e}`);t.title=i,t.setAttribute("aria-label",i)},this.options=g({},Kf,t),this._container=F.create("div","maplibregl-ctrl maplibregl-ctrl-group"),this._container.addEventListener("contextmenu",(t=>t.preventDefault())),this.options.showZoom&&(this._zoomInButton=this._createButton("maplibregl-ctrl-zoom-in",(t=>this._map.zoomIn({},{originalEvent:t}))),F.create("span","maplibregl-ctrl-icon",this._zoomInButton).setAttribute("aria-hidden","true"),this._zoomOutButton=this._createButton("maplibregl-ctrl-zoom-out",(t=>this._map.zoomOut({},{originalEvent:t}))),F.create("span","maplibregl-ctrl-icon",this._zoomOutButton).setAttribute("aria-hidden","true")),this.options.showCompass&&(this._compass=this._createButton("maplibregl-ctrl-compass",(t=>{this.options.visualizePitch?this._map.resetNorthPitch({},{originalEvent:t}):this._map.resetNorth({},{originalEvent:t})})),this._compassIcon=F.create("span","maplibregl-ctrl-icon",this._compass),this._compassIcon.setAttribute("aria-hidden","true"))}onAdd(t){return this._map=t,this.options.showZoom&&(this._setButtonTitle(this._zoomInButton,"ZoomIn"),this._setButtonTitle(this._zoomOutButton,"ZoomOut"),this._map.on("zoom",this._updateZoomButtons),this._updateZoomButtons()),this.options.showCompass&&(this._setButtonTitle(this._compass,"ResetBearing"),this.options.visualizePitch&&this._map.on("pitch",this._rotateCompassArrow),this._map.on("rotate",this._rotateCompassArrow),this._rotateCompassArrow(),this._handler=new Yf(this._map,this._compass,this.options.visualizePitch)),this._container}onRemove(){F.remove(this._container),this.options.showZoom&&this._map.off("zoom",this._updateZoomButtons),this.options.showCompass&&(this.options.visualizePitch&&this._map.off("pitch",this._rotateCompassArrow),this._map.off("rotate",this._rotateCompassArrow),this._handler.off(),delete this._handler),delete this._map}_createButton(t,e){const i=F.create("button",t,this._container);return i.type="button",i.addEventListener("click",e),i}},m_.GeolocateControl=class extends ot{constructor(t){super(),this._onSuccess=t=>{if(this._map){if(this._isOutOfMapMaxBounds(t))return this._setErrorState(),this.fire(new nt("outofmaxbounds",t)),this._updateMarker(),void this._finish();if(this.options.trackUserLocation)switch(this._lastKnownPosition=t,this._watchState){case"WAITING_ACTIVE":case"ACTIVE_LOCK":case"ACTIVE_ERROR":this._watchState="ACTIVE_LOCK",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active-error"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-active");break;case"BACKGROUND":case"BACKGROUND_ERROR":this._watchState="BACKGROUND",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background-error"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-background");break;default:throw new Error(`Unexpected watchState ${this._watchState}`)}this.options.showUserLocation&&"OFF"!==this._watchState&&this._updateMarker(t),this.options.trackUserLocation&&"ACTIVE_LOCK"!==this._watchState||this._updateCamera(t),this.options.showUserLocation&&this._dotElement.classList.remove("maplibregl-user-location-dot-stale"),this.fire(new nt("geolocate",t)),this._finish()}},this._updateCamera=t=>{const e=new Eh(t.coords.longitude,t.coords.latitude),i=t.coords.accuracy,r=g({bearing:this._map.getBearing()},this.options.fitBoundsOptions),s=Ch.fromLngLat(e,i);this._map.fitBounds(s,r,{geolocateSource:!0})},this._updateMarker=t=>{if(t){const e=new Eh(t.coords.longitude,t.coords.latitude);this._accuracyCircleMarker.setLngLat(e).addTo(this._map),this._userLocationDotMarker.setLngLat(e).addTo(this._map),this._accuracy=t.coords.accuracy,this.options.showUserLocation&&this.options.showAccuracyCircle&&this._updateCircleRadius()}else this._userLocationDotMarker.remove(),this._accuracyCircleMarker.remove()},this._onZoom=()=>{this.options.showUserLocation&&this.options.showAccuracyCircle&&this._updateCircleRadius()},this._onError=t=>{if(this._map){if(this.options.trackUserLocation)if(1===t.code){this._watchState="OFF",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active-error"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background-error"),this._geolocateButton.disabled=!0;const t=this._map._getUIString("GeolocateControl.LocationNotAvailable");this._geolocateButton.title=t,this._geolocateButton.setAttribute("aria-label",t),void 0!==this._geolocationWatchID&&this._clearWatch()}else{if(3===t.code&&n_)return;this._setErrorState()}"OFF"!==this._watchState&&this.options.showUserLocation&&this._dotElement.classList.add("maplibregl-user-location-dot-stale"),this.fire(new nt("error",t)),this._finish()}},this._finish=()=>{this._timeoutId&&clearTimeout(this._timeoutId),this._timeoutId=void 0},this._setupUI=t=>{if(this._map){if(this._container.addEventListener("contextmenu",(t=>t.preventDefault())),this._geolocateButton=F.create("button","maplibregl-ctrl-geolocate",this._container),F.create("span","maplibregl-ctrl-icon",this._geolocateButton).setAttribute("aria-hidden","true"),this._geolocateButton.type="button",!1===t){A("Geolocation support is not available so the GeolocateControl will be disabled.");const t=this._map._getUIString("GeolocateControl.LocationNotAvailable");this._geolocateButton.disabled=!0,this._geolocateButton.title=t,this._geolocateButton.setAttribute("aria-label",t)}else{const t=this._map._getUIString("GeolocateControl.FindMyLocation");this._geolocateButton.title=t,this._geolocateButton.setAttribute("aria-label",t)}this.options.trackUserLocation&&(this._geolocateButton.setAttribute("aria-pressed","false"),this._watchState="OFF"),this.options.showUserLocation&&(this._dotElement=F.create("div","maplibregl-user-location-dot"),this._userLocationDotMarker=new i_({element:this._dotElement}),this._circleElement=F.create("div","maplibregl-user-location-accuracy-circle"),this._accuracyCircleMarker=new i_({element:this._circleElement,pitchAlignment:"map"}),this.options.trackUserLocation&&(this._watchState="OFF"),this._map.on("zoom",this._onZoom)),this._geolocateButton.addEventListener("click",this.trigger.bind(this)),this._setup=!0,this.options.trackUserLocation&&this._map.on("movestart",(t=>{t.geolocateSource||"ACTIVE_LOCK"!==this._watchState||t.originalEvent&&"resize"===t.originalEvent.type||(this._watchState="BACKGROUND",this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-background"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active"),this.fire(new nt("trackuserlocationend")))}))}},this.options=g({},r_,t)}onAdd(t){return this._map=t,this._container=F.create("div","maplibregl-ctrl maplibregl-ctrl-group"),function(t,e=!1){void 0===Jf||e?void 0!==window.navigator.permissions?window.navigator.permissions.query({name:"geolocation"}).then((e=>{Jf="denied"!==e.state,t(Jf)})).catch((()=>{Jf=!!window.navigator.geolocation,t(Jf)})):(Jf=!!window.navigator.geolocation,t(Jf)):t(Jf)}(this._setupUI),this._container}onRemove(){void 0!==this._geolocationWatchID&&(window.navigator.geolocation.clearWatch(this._geolocationWatchID),this._geolocationWatchID=void 0),this.options.showUserLocation&&this._userLocationDotMarker&&this._userLocationDotMarker.remove(),this.options.showAccuracyCircle&&this._accuracyCircleMarker&&this._accuracyCircleMarker.remove(),F.remove(this._container),this._map.off("zoom",this._onZoom),this._map=void 0,s_=0,n_=!1}_isOutOfMapMaxBounds(t){const e=this._map.getMaxBounds(),i=t.coords;return e&&(i.longitudee.getEast()||i.latitudee.getNorth())}_setErrorState(){switch(this._watchState){case"WAITING_ACTIVE":this._watchState="ACTIVE_ERROR",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-active-error");break;case"ACTIVE_LOCK":this._watchState="ACTIVE_ERROR",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-active-error"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-waiting");break;case"BACKGROUND":this._watchState="BACKGROUND_ERROR",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-background-error"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-waiting");break;case"ACTIVE_ERROR":break;default:throw new Error(`Unexpected watchState ${this._watchState}`)}}_updateCircleRadius(){const t=this._map.getBounds(),e=t.getSouthEast(),i=t.getNorthEast(),r=e.distanceTo(i),s=Math.ceil(this._accuracy/(r/this._map._container.clientHeight)*2);this._circleElement.style.width=`${s}px`,this._circleElement.style.height=`${s}px`}trigger(){if(!this._setup)return A("Geolocate control triggered before added to a map"),!1;if(this.options.trackUserLocation){switch(this._watchState){case"OFF":this._watchState="WAITING_ACTIVE",this.fire(new nt("trackuserlocationstart"));break;case"WAITING_ACTIVE":case"ACTIVE_LOCK":case"ACTIVE_ERROR":case"BACKGROUND_ERROR":s_--,n_=!1,this._watchState="OFF",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active-error"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background-error"),this.fire(new nt("trackuserlocationend"));break;case"BACKGROUND":this._watchState="ACTIVE_LOCK",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background"),this._lastKnownPosition&&this._updateCamera(this._lastKnownPosition),this.fire(new nt("trackuserlocationstart"));break;default:throw new Error(`Unexpected watchState ${this._watchState}`)}switch(this._watchState){case"WAITING_ACTIVE":this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-active");break;case"ACTIVE_LOCK":this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-active");break;case"OFF":break;default:throw new Error(`Unexpected watchState ${this._watchState}`)}if("OFF"===this._watchState&&void 0!==this._geolocationWatchID)this._clearWatch();else if(void 0===this._geolocationWatchID){let t;this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.setAttribute("aria-pressed","true"),s_++,s_>1?(t={maximumAge:6e5,timeout:0},n_=!0):(t=this.options.positionOptions,n_=!1),this._geolocationWatchID=window.navigator.geolocation.watchPosition(this._onSuccess,this._onError,t)}}else window.navigator.geolocation.getCurrentPosition(this._onSuccess,this._onError,this.options.positionOptions),this._timeoutId=setTimeout(this._finish,1e4);return!0}_clearWatch(){window.navigator.geolocation.clearWatch(this._geolocationWatchID),this._geolocationWatchID=void 0,this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.setAttribute("aria-pressed","false"),this.options.showUserLocation&&this._updateMarker(null)}},m_.AttributionControl=Mf,m_.LogoControl=Pf,m_.ScaleControl=class{constructor(t){this._onMove=()=>{o_(this._map,this._container,this.options)},this.setUnit=t=>{this.options.unit=t,o_(this._map,this._container,this.options)},this.options=g({},a_,t)}getDefaultPosition(){return"bottom-left"}onAdd(t){return this._map=t,this._container=F.create("div","maplibregl-ctrl maplibregl-ctrl-scale",t.getContainer()),this._map.on("move",this._onMove),this._onMove(),this._container}onRemove(){F.remove(this._container),this._map.off("move",this._onMove),this._map=void 0}},m_.FullscreenControl=class extends ot{constructor(t={}){super(),this._onFullscreenChange=()=>{(window.document.fullscreenElement||window.document.mozFullScreenElement||window.document.webkitFullscreenElement||window.document.msFullscreenElement)===this._container!==this._fullscreen&&this._handleFullscreenChange()},this._onClickFullscreen=()=>{this._isFullscreen()?this._exitFullscreen():this._requestFullscreen()},this._fullscreen=!1,t&&t.container&&(t.container instanceof HTMLElement?this._container=t.container:A("Full screen control 'container' must be a DOM element.")),"onfullscreenchange"in document?this._fullscreenchange="fullscreenchange":"onmozfullscreenchange"in document?this._fullscreenchange="mozfullscreenchange":"onwebkitfullscreenchange"in document?this._fullscreenchange="webkitfullscreenchange":"onmsfullscreenchange"in document&&(this._fullscreenchange="MSFullscreenChange")}onAdd(t){return this._map=t,this._container||(this._container=this._map.getContainer()),this._controlContainer=F.create("div","maplibregl-ctrl maplibregl-ctrl-group"),this._setupUI(),this._controlContainer}onRemove(){F.remove(this._controlContainer),this._map=null,window.document.removeEventListener(this._fullscreenchange,this._onFullscreenChange)}_setupUI(){const t=this._fullscreenButton=F.create("button","maplibregl-ctrl-fullscreen",this._controlContainer);F.create("span","maplibregl-ctrl-icon",t).setAttribute("aria-hidden","true"),t.type="button",this._updateTitle(),this._fullscreenButton.addEventListener("click",this._onClickFullscreen),window.document.addEventListener(this._fullscreenchange,this._onFullscreenChange)}_updateTitle(){const t=this._getTitle();this._fullscreenButton.setAttribute("aria-label",t),this._fullscreenButton.title=t}_getTitle(){return this._map._getUIString(this._isFullscreen()?"FullscreenControl.Exit":"FullscreenControl.Enter")}_isFullscreen(){return this._fullscreen}_handleFullscreenChange(){this._fullscreen=!this._fullscreen,this._fullscreenButton.classList.toggle("maplibregl-ctrl-shrink"),this._fullscreenButton.classList.toggle("maplibregl-ctrl-fullscreen"),this._updateTitle(),this._fullscreen?(this.fire(new nt("fullscreenstart")),this._map._cooperativeGestures&&(this._prevCooperativeGestures=this._map._cooperativeGestures,this._map.setCooperativeGestures())):(this.fire(new nt("fullscreenend")),this._prevCooperativeGestures&&(this._map.setCooperativeGestures(this._prevCooperativeGestures),delete this._prevCooperativeGestures))}_exitFullscreen(){window.document.exitFullscreen?window.document.exitFullscreen():window.document.mozCancelFullScreen?window.document.mozCancelFullScreen():window.document.msExitFullscreen?window.document.msExitFullscreen():window.document.webkitCancelFullScreen?window.document.webkitCancelFullScreen():this._togglePseudoFullScreen()}_requestFullscreen(){this._container.requestFullscreen?this._container.requestFullscreen():this._container.mozRequestFullScreen?this._container.mozRequestFullScreen():this._container.msRequestFullscreen?this._container.msRequestFullscreen():this._container.webkitRequestFullscreen?this._container.webkitRequestFullscreen():this._togglePseudoFullScreen()}_togglePseudoFullScreen(){this._container.classList.toggle("maplibregl-pseudo-fullscreen"),this._handleFullscreenChange(),this._map.resize()}},m_.TerrainControl=class{constructor(t){this._toggleTerrain=()=>{this._map.getTerrain()?this._map.setTerrain(null):this._map.setTerrain(this.options),this._updateTerrainIcon()},this._updateTerrainIcon=()=>{this._terrainButton.classList.remove("maplibregl-ctrl-terrain"),this._terrainButton.classList.remove("maplibregl-ctrl-terrain-enabled"),this._map.terrain?(this._terrainButton.classList.add("maplibregl-ctrl-terrain-enabled"),this._terrainButton.title=this._map._getUIString("TerrainControl.disableTerrain")):(this._terrainButton.classList.add("maplibregl-ctrl-terrain"),this._terrainButton.title=this._map._getUIString("TerrainControl.enableTerrain"))},this.options=t}onAdd(t){return this._map=t,this._container=F.create("div","maplibregl-ctrl maplibregl-ctrl-group"),this._terrainButton=F.create("button","maplibregl-ctrl-terrain",this._container),F.create("span","maplibregl-ctrl-icon",this._terrainButton).setAttribute("aria-hidden","true"),this._terrainButton.type="button",this._terrainButton.addEventListener("click",this._toggleTerrain),this._updateTerrainIcon(),this._map.on("terrain",this._updateTerrainIcon),this._container}onRemove(){F.remove(this._container),this._map.off("terrain",this._updateTerrainIcon),this._map=void 0}},m_.Popup=class extends ot{constructor(t){super(),this.remove=()=>(this._content&&F.remove(this._content),this._container&&(F.remove(this._container),delete this._container),this._map&&(this._map.off("move",this._update),this._map.off("move",this._onClose),this._map.off("click",this._onClose),this._map.off("remove",this.remove),this._map.off("mousemove",this._onMouseMove),this._map.off("mouseup",this._onMouseUp),this._map.off("drag",this._onDrag),delete this._map),this.fire(new nt("close")),this),this._onMouseUp=t=>{this._update(t.point)},this._onMouseMove=t=>{this._update(t.point)},this._onDrag=t=>{this._update(t.point)},this._update=t=>{if(!this._map||!this._lngLat&&!this._trackPointer||!this._content)return;if(!this._container){if(this._container=F.create("div","maplibregl-popup",this._map.getContainer()),this._tip=F.create("div","maplibregl-popup-tip",this._container),this._container.appendChild(this._content),this.options.className)for(const t of this.options.className.split(" "))this._container.classList.add(t);this._trackPointer&&this._container.classList.add("maplibregl-popup-track-pointer")}if(this.options.maxWidth&&this._container.style.maxWidth!==this.options.maxWidth&&(this._container.style.maxWidth=this.options.maxWidth),this._map.transform.renderWorldCopies&&!this._trackPointer&&(this._lngLat=Qf(this._lngLat,this._pos,this._map.transform)),this._trackPointer&&!t)return;const e=this._pos=this._trackPointer&&t?t:this._map.project(this._lngLat);let i=this.options.anchor;const r=u_(this.options.offset);if(!i){const t=this._container.offsetWidth,s=this._container.offsetHeight;let n;n=e.y+r.bottom.ythis._map.transform.height-s?["bottom"]:[],e.xthis._map.transform.width-t/2&&n.push("right"),i=0===n.length?"bottom":n.join("-")}const s=e.add(r[i]).round();F.setTransform(this._container,`${t_[i]} translate(${s.x}px,${s.y}px)`),e_(this._container,i,"popup")},this._onClose=()=>{this.remove()},this.options=g(Object.create(c_),t)}addTo(t){return this._map&&this.remove(),this._map=t,this.options.closeOnClick&&this._map.on("click",this._onClose),this.options.closeOnMove&&this._map.on("move",this._onClose),this._map.on("remove",this.remove),this._update(),this._focusFirstElement(),this._trackPointer?(this._map.on("mousemove",this._onMouseMove),this._map.on("mouseup",this._onMouseUp),this._container&&this._container.classList.add("maplibregl-popup-track-pointer"),this._map._canvasContainer.classList.add("maplibregl-track-pointer")):this._map.on("move",this._update),this.fire(new nt("open")),this}isOpen(){return!!this._map}getLngLat(){return this._lngLat}setLngLat(t){return this._lngLat=Eh.convert(t),this._pos=null,this._trackPointer=!1,this._update(),this._map&&(this._map.on("move",this._update),this._map.off("mousemove",this._onMouseMove),this._container&&this._container.classList.remove("maplibregl-popup-track-pointer"),this._map._canvasContainer.classList.remove("maplibregl-track-pointer")),this}trackPointer(){return this._trackPointer=!0,this._pos=null,this._update(),this._map&&(this._map.off("move",this._update),this._map.on("mousemove",this._onMouseMove),this._map.on("drag",this._onDrag),this._container&&this._container.classList.add("maplibregl-popup-track-pointer"),this._map._canvasContainer.classList.add("maplibregl-track-pointer")),this}getElement(){return this._container}setText(t){return this.setDOMContent(document.createTextNode(t))}setHTML(t){const e=document.createDocumentFragment(),i=document.createElement("body");let r;for(i.innerHTML=t;r=i.firstChild,r;)e.appendChild(r);return this.setDOMContent(e)}getMaxWidth(){var t;return null===(t=this._container)||void 0===t?void 0:t.style.maxWidth}setMaxWidth(t){return this.options.maxWidth=t,this._update(),this}setDOMContent(t){if(this._content)for(;this._content.hasChildNodes();)this._content.firstChild&&this._content.removeChild(this._content.firstChild);else this._content=F.create("div","maplibregl-popup-content",this._container);return this._content.appendChild(t),this._createCloseButton(),this._update(),this._focusFirstElement(),this}addClassName(t){this._container&&this._container.classList.add(t)}removeClassName(t){this._container&&this._container.classList.remove(t)}setOffset(t){return this.options.offset=t,this._update(),this}toggleClassName(t){if(this._container)return this._container.classList.toggle(t)}_createCloseButton(){this.options.closeButton&&(this._closeButton=F.create("button","maplibregl-popup-close-button",this._content),this._closeButton.type="button",this._closeButton.setAttribute("aria-label","Close popup"),this._closeButton.innerHTML="×",this._closeButton.addEventListener("click",this._onClose))}_focusFirstElement(){if(!this.options.focusAfterOpen||!this._container)return;const t=this._container.querySelector(h_);t&&t.focus()}},m_.Marker=i_,m_.Style=Ap,m_.LngLat=Eh,m_.LngLatBounds=Ch,m_.Point=n,m_.MercatorCoordinate=Bh,m_.Evented=ot,m_.AJAXError=V,m_.config=O,m_.CanvasSource=Kh,m_.GeoJSONSource=Gh,m_.ImageSource=Wh,m_.RasterDEMTileSource=jh,m_.RasterTileSource=Vh,m_.VectorTileSource=Oh,m_.VideoSource=Xh,m_.setRTLTextPlugin=function(t,e,i=!1){if(ls===ss||ls===ns||ls===as)throw new Error("setRTLTextPlugin cannot be called multiple times.");cs=B.resolveURL(t),ls=ss,os=e,us(),i||ms()},m_.getRTLTextPluginStatus=ds,m_.prewarm=function(){gu().acquire(du)},m_.clearPrewarmedResources=function(){const t=_u;t&&(t.isPreloaded()&&1===t.numActive()?(t.release(du),_u=null):console.warn("Could not clear WebWorkers since there are active Map instances that still reference it. The pre-warmed WebWorker pool can only be cleared when all map instances have been removed with map.remove()"))},p_.extend(m_,{isSafari:P,getPerformanceMetrics:Vf.getPerformanceMetrics}),m_})); +//# sourceMappingURL=maplibre-gl-csp.js.map diff --git a/web/libraries/maplibre-gl/dist/maplibre-gl-csp.js.map b/web/libraries/maplibre-gl/dist/maplibre-gl-csp.js.map new file mode 100644 index 00000000..49cb3001 --- /dev/null +++ b/web/libraries/maplibre-gl/dist/maplibre-gl-csp.js.map @@ -0,0 +1 @@ +{"version":3,"file":"maplibre-gl-csp.js","sources":["../node_modules/tslib/tslib.es6.js","../node_modules/@mapbox/point-geometry/index.js","../node_modules/@mapbox/unitbezier/index.js","../src/util/offscreen_canvas_supported.ts","../src/util/offscreen_canvas_distorted.ts","../src/util/util.ts","../src/util/browser.ts","../src/util/dom.ts","../src/util/config.ts","../src/util/ajax.ts","../src/util/webp_supported.ts","../src/util/image_request.ts","../src/util/request_manager.ts","../src/util/evented.ts","../node_modules/@maplibre/maplibre-gl-style-spec/dist/index.mjs","../src/style/validate_style.ts","../src/util/transferable_grid_index.ts","../src/util/web_worker_transfer.ts","../src/style/zoom_history.ts","../src/util/is_char_in_unicode_block.ts","../src/util/script_detection.ts","../src/source/rtl_text_plugin.ts","../src/style/evaluation_parameters.ts","../src/style/properties.ts","../src/style/style_layer.ts","../src/util/struct_array.ts","../src/data/array_types.g.ts","../src/data/bucket/circle_attributes.ts","../src/data/segment.ts","../src/shaders/encode_attribute.ts","../src/data/bucket/pattern_attributes.ts","../node_modules/murmurhash-js/murmurhash3_gc.js","../node_modules/murmurhash-js/murmurhash2_gc.js","../node_modules/murmurhash-js/index.js","../src/data/feature_position_map.ts","../src/render/uniform_binding.ts","../src/data/program_configuration.ts","../src/data/extent.ts","../src/data/load_geometry.ts","../src/data/evaluation_feature.ts","../src/data/bucket/circle_bucket.ts","../src/util/intersection_tests.ts","../src/style/query_utils.ts","../src/style/style_layer/circle_style_layer_properties.g.ts","../node_modules/gl-matrix/esm/common.js","../node_modules/gl-matrix/esm/mat4.js","../node_modules/gl-matrix/esm/vec3.js","../node_modules/gl-matrix/esm/vec4.js","../node_modules/gl-matrix/esm/vec2.js","../src/style/style_layer/circle_style_layer.ts","../src/data/bucket/heatmap_bucket.ts","../src/style/style_layer/heatmap_style_layer_properties.g.ts","../src/util/image.ts","../src/util/color_ramp.ts","../src/style/style_layer/heatmap_style_layer.ts","../src/style/style_layer/hillshade_style_layer_properties.g.ts","../src/style/style_layer/hillshade_style_layer.ts","../src/data/bucket/fill_attributes.ts","../node_modules/earcut/src/earcut.js","../node_modules/quickselect/index.js","../src/util/classify_rings.ts","../src/data/bucket/pattern_bucket_features.ts","../src/data/bucket/fill_bucket.ts","../src/style/style_layer/fill_style_layer_properties.g.ts","../src/style/style_layer/fill_style_layer.ts","../src/data/bucket/fill_extrusion_attributes.ts","../node_modules/@mapbox/vector-tile/lib/vectortilefeature.js","../node_modules/@mapbox/vector-tile/lib/vectortilelayer.js","../node_modules/@mapbox/vector-tile/lib/vectortile.js","../node_modules/@mapbox/vector-tile/index.js","../src/data/bucket/fill_extrusion_bucket.ts","../src/style/style_layer/fill_extrusion_style_layer_properties.g.ts","../src/style/style_layer/fill_extrusion_style_layer.ts","../src/data/bucket/line_attributes.ts","../src/data/bucket/line_attributes_ext.ts","../src/data/bucket/line_bucket.ts","../src/style/style_layer/line_style_layer_properties.g.ts","../src/style/style_layer/line_style_layer.ts","../src/data/bucket/symbol_attributes.ts","../src/symbol/transform_text.ts","../src/util/verticalize_punctuation.ts","../src/symbol/one_em.ts","../node_modules/pbf/index.js","../node_modules/ieee754/index.js","../src/style/parse_glyph_pbf.ts","../node_modules/potpack/index.js","../src/render/image_atlas.ts","../src/symbol/shaping.ts","../src/symbol/symbol_size.ts","../src/style/style_layer/overlap_mode.ts","../src/data/bucket/symbol_bucket.ts","../src/symbol/merge_lines.ts","../src/style/style_layer/symbol_style_layer_properties.g.ts","../src/style/format_section_override.ts","../src/style/style_layer/symbol_style_layer.ts","../src/util/resolve_tokens.ts","../src/style/style_layer/background_style_layer_properties.g.ts","../src/style/style_layer/background_style_layer.ts","../src/style/style_layer/raster_style_layer_properties.g.ts","../src/style/style_layer/raster_style_layer.ts","../src/style/style_layer/custom_style_layer.ts","../src/style/create_style_layer.ts","../src/util/style.ts","../src/style/load_sprite.ts","../src/render/texture.ts","../src/style/style_image.ts","../src/render/image_manager.ts","../node_modules/@mapbox/tiny-sdf/index.js","../src/render/glyph_manager.ts","../src/style/load_glyph_range.ts","../src/style/light.ts","../src/render/line_atlas.ts","../src/util/throttled_invoker.ts","../src/util/actor.ts","../src/util/dispatcher.ts","../src/source/load_tilejson.ts","../src/geo/lng_lat.ts","../src/geo/lng_lat_bounds.ts","../src/geo/mercator_coordinate.ts","../src/source/tile_bounds.ts","../src/source/vector_tile_source.ts","../src/source/raster_tile_source.ts","../node_modules/@mapbox/whoots-js/index.mjs","../src/source/tile_id.ts","../src/data/dem_data.ts","../src/source/raster_dem_tile_source.ts","../src/source/geojson_source.ts","../src/data/raster_bounds_attributes.ts","../src/source/image_source.ts","../src/source/video_source.ts","../src/source/canvas_source.ts","../src/source/source.ts","../src/source/query_features.ts","../src/util/dictionary_coder.ts","../src/util/vectortile_to_geojson.ts","../src/data/feature_index.ts","../src/source/tile.ts","../src/data/bucket.ts","../src/source/tile_cache.ts","../src/source/source_state.ts","../src/source/source_cache.ts","../src/util/worker_pool.ts","../src/util/web_worker.ts","../src/util/global_worker_pool.ts","../src/symbol/path_interpolator.ts","../src/symbol/grid_index.ts","../src/symbol/projection.ts","../src/symbol/collision_index.ts","../src/symbol/clip_line.ts","../src/symbol/anchor.ts","../src/style/style_layer/variable_text_anchor.ts","../src/source/pixels_to_tile_units.ts","../src/symbol/placement.ts","../src/symbol/symbol_layout.ts","../src/style/pauseable_placement.ts","../node_modules/kdbush/index.js","../src/symbol/cross_tile_symbol_index.ts","../src/style/style.ts","../src/data/pos_attributes.ts","../src/shaders/terrain.vertex.glsl.g.ts","../src/shaders/shaders.ts","../src/shaders/_prelude.fragment.glsl.g.ts","../src/shaders/_prelude.vertex.glsl.g.ts","../src/shaders/background.fragment.glsl.g.ts","../src/shaders/background.vertex.glsl.g.ts","../src/shaders/background_pattern.fragment.glsl.g.ts","../src/shaders/background_pattern.vertex.glsl.g.ts","../src/shaders/circle.fragment.glsl.g.ts","../src/shaders/circle.vertex.glsl.g.ts","../src/shaders/clipping_mask.fragment.glsl.g.ts","../src/shaders/clipping_mask.vertex.glsl.g.ts","../src/shaders/heatmap.fragment.glsl.g.ts","../src/shaders/heatmap.vertex.glsl.g.ts","../src/shaders/heatmap_texture.fragment.glsl.g.ts","../src/shaders/heatmap_texture.vertex.glsl.g.ts","../src/shaders/collision_box.fragment.glsl.g.ts","../src/shaders/collision_box.vertex.glsl.g.ts","../src/shaders/collision_circle.fragment.glsl.g.ts","../src/shaders/collision_circle.vertex.glsl.g.ts","../src/shaders/debug.fragment.glsl.g.ts","../src/shaders/debug.vertex.glsl.g.ts","../src/shaders/fill.fragment.glsl.g.ts","../src/shaders/fill.vertex.glsl.g.ts","../src/shaders/fill_outline.fragment.glsl.g.ts","../src/shaders/fill_outline.vertex.glsl.g.ts","../src/shaders/fill_outline_pattern.fragment.glsl.g.ts","../src/shaders/fill_outline_pattern.vertex.glsl.g.ts","../src/shaders/fill_pattern.fragment.glsl.g.ts","../src/shaders/fill_pattern.vertex.glsl.g.ts","../src/shaders/fill_extrusion.fragment.glsl.g.ts","../src/shaders/fill_extrusion.vertex.glsl.g.ts","../src/shaders/fill_extrusion_pattern.fragment.glsl.g.ts","../src/shaders/fill_extrusion_pattern.vertex.glsl.g.ts","../src/shaders/hillshade_prepare.fragment.glsl.g.ts","../src/shaders/hillshade_prepare.vertex.glsl.g.ts","../src/shaders/hillshade.fragment.glsl.g.ts","../src/shaders/hillshade.vertex.glsl.g.ts","../src/shaders/line.fragment.glsl.g.ts","../src/shaders/line.vertex.glsl.g.ts","../src/shaders/line_gradient.fragment.glsl.g.ts","../src/shaders/line_gradient.vertex.glsl.g.ts","../src/shaders/line_pattern.fragment.glsl.g.ts","../src/shaders/line_pattern.vertex.glsl.g.ts","../src/shaders/line_sdf.fragment.glsl.g.ts","../src/shaders/line_sdf.vertex.glsl.g.ts","../src/shaders/raster.fragment.glsl.g.ts","../src/shaders/raster.vertex.glsl.g.ts","../src/shaders/symbol_icon.fragment.glsl.g.ts","../src/shaders/symbol_icon.vertex.glsl.g.ts","../src/shaders/symbol_sdf.fragment.glsl.g.ts","../src/shaders/symbol_sdf.vertex.glsl.g.ts","../src/shaders/symbol_text_and_icon.fragment.glsl.g.ts","../src/shaders/symbol_text_and_icon.vertex.glsl.g.ts","../src/shaders/terrain.fragment.glsl.g.ts","../src/shaders/terrain_depth.fragment.glsl.g.ts","../src/shaders/terrain_coords.fragment.glsl.g.ts","../src/render/vertex_array_object.ts","../src/render/program.ts","../src/render/program/terrain_program.ts","../src/render/program/pattern.ts","../src/render/program/fill_extrusion_program.ts","../node_modules/gl-matrix/esm/mat3.js","../src/render/program/fill_program.ts","../src/render/program/circle_program.ts","../src/render/program/collision_program.ts","../src/render/program/debug_program.ts","../src/render/program/clipping_mask_program.ts","../src/render/program/heatmap_program.ts","../src/render/program/hillshade_program.ts","../src/render/program/line_program.ts","../src/render/program/raster_program.ts","../src/render/program/symbol_program.ts","../src/render/program/background_program.ts","../src/render/program/program_uniforms.ts","../src/gl/index_buffer.ts","../src/gl/vertex_buffer.ts","../src/gl/webgl2.ts","../src/gl/value.ts","../src/gl/framebuffer.ts","../src/gl/color_mode.ts","../src/gl/context.ts","../src/gl/depth_mode.ts","../src/gl/stencil_mode.ts","../src/gl/cull_face_mode.ts","../src/render/draw_collision_debug.ts","../src/render/draw_symbol.ts","../src/render/update_pattern_positions_in_program.ts","../src/render/draw_fill.ts","../src/render/draw_fill_extrusion.ts","../src/render/draw_hillshade.ts","../src/render/draw_raster.ts","../src/render/draw_debug.ts","../src/render/draw_terrain.ts","../src/render/painter.ts","../src/render/draw_circle.ts","../src/render/draw_heatmap.ts","../src/render/draw_line.ts","../src/render/draw_background.ts","../src/render/draw_custom.ts","../src/util/primitives.ts","../src/geo/edge_insets.ts","../src/geo/transform.ts","../node_modules/gl-matrix/esm/mat2.js","../src/util/throttle.ts","../src/ui/hash.ts","../src/ui/handler_inertia.ts","../src/ui/events.ts","../src/ui/handler/map_event.ts","../src/ui/handler/transform-provider.ts","../src/ui/handler/box_zoom.ts","../src/ui/handler/handler_util.ts","../src/ui/handler/tap_recognizer.ts","../src/ui/handler/tap_zoom.ts","../src/ui/handler/drag_handler.ts","../src/ui/handler/drag_move_state_manager.ts","../src/ui/handler/mouse.ts","../src/ui/handler/touch_pan.ts","../src/ui/handler/two_fingers_touch.ts","../src/ui/handler/keyboard.ts","../src/ui/handler/scroll_zoom.ts","../src/ui/handler/shim/dblclick_zoom.ts","../src/ui/handler/click_zoom.ts","../src/ui/handler/tap_drag_zoom.ts","../src/ui/handler/shim/drag_pan.ts","../src/ui/handler/shim/drag_rotate.ts","../src/ui/handler/shim/two_fingers_touch.ts","../src/ui/handler_manager.ts","../src/ui/camera.ts","../src/ui/control/attribution_control.ts","../src/ui/control/logo_control.ts","../src/util/task_queue.ts","../src/util/performance.ts","../src/ui/default_locale.ts","../src/data/pos3d_attributes.ts","../src/source/terrain_source_cache.ts","../src/render/terrain.ts","../src/gl/render_pool.ts","../src/render/render_to_texture.ts","../src/ui/map.ts","../src/ui/handler/one_finger_touch_drag.ts","../src/ui/control/navigation_control.ts","../src/util/geolocation_support.ts","../src/util/smart_wrap.ts","../src/ui/anchor.ts","../src/ui/marker.ts","../src/ui/control/geolocate_control.ts","../src/ui/control/scale_control.ts","../src/ui/popup.ts","../src/util/debug.ts","../src/index.ts","../src/ui/control/fullscreen_control.ts","../src/ui/control/terrain_control.ts"],"sourcesContent":["/******************************************************************************\r\nCopyright (c) Microsoft Corporation.\r\n\r\nPermission to use, copy, modify, and/or distribute this software for any\r\npurpose with or without fee is hereby granted.\r\n\r\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\r\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\r\nAND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\r\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\r\nLOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\r\nOTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\r\nPERFORMANCE OF THIS SOFTWARE.\r\n***************************************************************************** */\r\n/* global Reflect, Promise, SuppressedError, Symbol */\r\n\r\nvar extendStatics = function(d, b) {\r\n extendStatics = Object.setPrototypeOf ||\r\n ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\r\n function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };\r\n return extendStatics(d, b);\r\n};\r\n\r\nexport function __extends(d, b) {\r\n if (typeof b !== \"function\" && b !== null)\r\n throw new TypeError(\"Class extends value \" + String(b) + \" is not a constructor or null\");\r\n extendStatics(d, b);\r\n function __() { this.constructor = d; }\r\n d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\r\n}\r\n\r\nexport var __assign = function() {\r\n __assign = Object.assign || function __assign(t) {\r\n for (var s, i = 1, n = arguments.length; i < n; i++) {\r\n s = arguments[i];\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];\r\n }\r\n return t;\r\n }\r\n return __assign.apply(this, arguments);\r\n}\r\n\r\nexport function __rest(s, e) {\r\n var t = {};\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\r\n t[p] = s[p];\r\n if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\r\n for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\r\n if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\r\n t[p[i]] = s[p[i]];\r\n }\r\n return t;\r\n}\r\n\r\nexport function __decorate(decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n}\r\n\r\nexport function __param(paramIndex, decorator) {\r\n return function (target, key) { decorator(target, key, paramIndex); }\r\n}\r\n\r\nexport function __esDecorate(ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {\r\n function accept(f) { if (f !== void 0 && typeof f !== \"function\") throw new TypeError(\"Function expected\"); return f; }\r\n var kind = contextIn.kind, key = kind === \"getter\" ? \"get\" : kind === \"setter\" ? \"set\" : \"value\";\r\n var target = !descriptorIn && ctor ? contextIn[\"static\"] ? ctor : ctor.prototype : null;\r\n var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});\r\n var _, done = false;\r\n for (var i = decorators.length - 1; i >= 0; i--) {\r\n var context = {};\r\n for (var p in contextIn) context[p] = p === \"access\" ? {} : contextIn[p];\r\n for (var p in contextIn.access) context.access[p] = contextIn.access[p];\r\n context.addInitializer = function (f) { if (done) throw new TypeError(\"Cannot add initializers after decoration has completed\"); extraInitializers.push(accept(f || null)); };\r\n var result = (0, decorators[i])(kind === \"accessor\" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);\r\n if (kind === \"accessor\") {\r\n if (result === void 0) continue;\r\n if (result === null || typeof result !== \"object\") throw new TypeError(\"Object expected\");\r\n if (_ = accept(result.get)) descriptor.get = _;\r\n if (_ = accept(result.set)) descriptor.set = _;\r\n if (_ = accept(result.init)) initializers.unshift(_);\r\n }\r\n else if (_ = accept(result)) {\r\n if (kind === \"field\") initializers.unshift(_);\r\n else descriptor[key] = _;\r\n }\r\n }\r\n if (target) Object.defineProperty(target, contextIn.name, descriptor);\r\n done = true;\r\n};\r\n\r\nexport function __runInitializers(thisArg, initializers, value) {\r\n var useValue = arguments.length > 2;\r\n for (var i = 0; i < initializers.length; i++) {\r\n value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);\r\n }\r\n return useValue ? value : void 0;\r\n};\r\n\r\nexport function __propKey(x) {\r\n return typeof x === \"symbol\" ? x : \"\".concat(x);\r\n};\r\n\r\nexport function __setFunctionName(f, name, prefix) {\r\n if (typeof name === \"symbol\") name = name.description ? \"[\".concat(name.description, \"]\") : \"\";\r\n return Object.defineProperty(f, \"name\", { configurable: true, value: prefix ? \"\".concat(prefix, \" \", name) : name });\r\n};\r\n\r\nexport function __metadata(metadataKey, metadataValue) {\r\n if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(metadataKey, metadataValue);\r\n}\r\n\r\nexport function __awaiter(thisArg, _arguments, P, generator) {\r\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\r\n return new (P || (P = Promise))(function (resolve, reject) {\r\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\r\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\r\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\r\n step((generator = generator.apply(thisArg, _arguments || [])).next());\r\n });\r\n}\r\n\r\nexport function __generator(thisArg, body) {\r\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\r\n return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\r\n function verb(n) { return function (v) { return step([n, v]); }; }\r\n function step(op) {\r\n if (f) throw new TypeError(\"Generator is already executing.\");\r\n while (g && (g = 0, op[0] && (_ = 0)), _) try {\r\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\r\n if (y = 0, t) op = [op[0] & 2, t.value];\r\n switch (op[0]) {\r\n case 0: case 1: t = op; break;\r\n case 4: _.label++; return { value: op[1], done: false };\r\n case 5: _.label++; y = op[1]; op = [0]; continue;\r\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\r\n default:\r\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\r\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\r\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\r\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\r\n if (t[2]) _.ops.pop();\r\n _.trys.pop(); continue;\r\n }\r\n op = body.call(thisArg, _);\r\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\r\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\r\n }\r\n}\r\n\r\nexport var __createBinding = Object.create ? (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n var desc = Object.getOwnPropertyDescriptor(m, k);\r\n if (!desc || (\"get\" in desc ? !m.__esModule : desc.writable || desc.configurable)) {\r\n desc = { enumerable: true, get: function() { return m[k]; } };\r\n }\r\n Object.defineProperty(o, k2, desc);\r\n}) : (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n o[k2] = m[k];\r\n});\r\n\r\nexport function __exportStar(m, o) {\r\n for (var p in m) if (p !== \"default\" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p);\r\n}\r\n\r\nexport function __values(o) {\r\n var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\r\n if (m) return m.call(o);\r\n if (o && typeof o.length === \"number\") return {\r\n next: function () {\r\n if (o && i >= o.length) o = void 0;\r\n return { value: o && o[i++], done: !o };\r\n }\r\n };\r\n throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\r\n}\r\n\r\nexport function __read(o, n) {\r\n var m = typeof Symbol === \"function\" && o[Symbol.iterator];\r\n if (!m) return o;\r\n var i = m.call(o), r, ar = [], e;\r\n try {\r\n while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\r\n }\r\n catch (error) { e = { error: error }; }\r\n finally {\r\n try {\r\n if (r && !r.done && (m = i[\"return\"])) m.call(i);\r\n }\r\n finally { if (e) throw e.error; }\r\n }\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spread() {\r\n for (var ar = [], i = 0; i < arguments.length; i++)\r\n ar = ar.concat(__read(arguments[i]));\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spreadArrays() {\r\n for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\r\n for (var r = Array(s), k = 0, i = 0; i < il; i++)\r\n for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\r\n r[k] = a[j];\r\n return r;\r\n}\r\n\r\nexport function __spreadArray(to, from, pack) {\r\n if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {\r\n if (ar || !(i in from)) {\r\n if (!ar) ar = Array.prototype.slice.call(from, 0, i);\r\n ar[i] = from[i];\r\n }\r\n }\r\n return to.concat(ar || Array.prototype.slice.call(from));\r\n}\r\n\r\nexport function __await(v) {\r\n return this instanceof __await ? (this.v = v, this) : new __await(v);\r\n}\r\n\r\nexport function __asyncGenerator(thisArg, _arguments, generator) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var g = generator.apply(thisArg, _arguments || []), i, q = [];\r\n return i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i;\r\n function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }\r\n function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\r\n function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\r\n function fulfill(value) { resume(\"next\", value); }\r\n function reject(value) { resume(\"throw\", value); }\r\n function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\r\n}\r\n\r\nexport function __asyncDelegator(o) {\r\n var i, p;\r\n return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\r\n function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: false } : f ? f(v) : v; } : f; }\r\n}\r\n\r\nexport function __asyncValues(o) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var m = o[Symbol.asyncIterator], i;\r\n return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\r\n function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\r\n function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\r\n}\r\n\r\nexport function __makeTemplateObject(cooked, raw) {\r\n if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\r\n return cooked;\r\n};\r\n\r\nvar __setModuleDefault = Object.create ? (function(o, v) {\r\n Object.defineProperty(o, \"default\", { enumerable: true, value: v });\r\n}) : function(o, v) {\r\n o[\"default\"] = v;\r\n};\r\n\r\nexport function __importStar(mod) {\r\n if (mod && mod.__esModule) return mod;\r\n var result = {};\r\n if (mod != null) for (var k in mod) if (k !== \"default\" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);\r\n __setModuleDefault(result, mod);\r\n return result;\r\n}\r\n\r\nexport function __importDefault(mod) {\r\n return (mod && mod.__esModule) ? mod : { default: mod };\r\n}\r\n\r\nexport function __classPrivateFieldGet(receiver, state, kind, f) {\r\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a getter\");\r\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot read private member from an object whose class did not declare it\");\r\n return kind === \"m\" ? f : kind === \"a\" ? f.call(receiver) : f ? f.value : state.get(receiver);\r\n}\r\n\r\nexport function __classPrivateFieldSet(receiver, state, value, kind, f) {\r\n if (kind === \"m\") throw new TypeError(\"Private method is not writable\");\r\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a setter\");\r\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot write private member to an object whose class did not declare it\");\r\n return (kind === \"a\" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;\r\n}\r\n\r\nexport function __classPrivateFieldIn(state, receiver) {\r\n if (receiver === null || (typeof receiver !== \"object\" && typeof receiver !== \"function\")) throw new TypeError(\"Cannot use 'in' operator on non-object\");\r\n return typeof state === \"function\" ? receiver === state : state.has(receiver);\r\n}\r\n\r\nexport function __addDisposableResource(env, value, async) {\r\n if (value !== null && value !== void 0) {\r\n if (typeof value !== \"object\" && typeof value !== \"function\") throw new TypeError(\"Object expected.\");\r\n var dispose;\r\n if (async) {\r\n if (!Symbol.asyncDispose) throw new TypeError(\"Symbol.asyncDispose is not defined.\");\r\n dispose = value[Symbol.asyncDispose];\r\n }\r\n if (dispose === void 0) {\r\n if (!Symbol.dispose) throw new TypeError(\"Symbol.dispose is not defined.\");\r\n dispose = value[Symbol.dispose];\r\n }\r\n if (typeof dispose !== \"function\") throw new TypeError(\"Object not disposable.\");\r\n env.stack.push({ value: value, dispose: dispose, async: async });\r\n }\r\n else if (async) {\r\n env.stack.push({ async: true });\r\n }\r\n return value;\r\n}\r\n\r\nvar _SuppressedError = typeof SuppressedError === \"function\" ? SuppressedError : function (error, suppressed, message) {\r\n var e = new Error(message);\r\n return e.name = \"SuppressedError\", e.error = error, e.suppressed = suppressed, e;\r\n};\r\n\r\nexport function __disposeResources(env) {\r\n function fail(e) {\r\n env.error = env.hasError ? new _SuppressedError(e, env.error, \"An error was suppressed during disposal.\") : e;\r\n env.hasError = true;\r\n }\r\n function next() {\r\n while (env.stack.length) {\r\n var rec = env.stack.pop();\r\n try {\r\n var result = rec.dispose && rec.dispose.call(rec.value);\r\n if (rec.async) return Promise.resolve(result).then(next, function(e) { fail(e); return next(); });\r\n }\r\n catch (e) {\r\n fail(e);\r\n }\r\n }\r\n if (env.hasError) throw env.error;\r\n }\r\n return next();\r\n}\r\n\r\nexport default {\r\n __extends: __extends,\r\n __assign: __assign,\r\n __rest: __rest,\r\n __decorate: __decorate,\r\n __param: __param,\r\n __metadata: __metadata,\r\n __awaiter: __awaiter,\r\n __generator: __generator,\r\n __createBinding: __createBinding,\r\n __exportStar: __exportStar,\r\n __values: __values,\r\n __read: __read,\r\n __spread: __spread,\r\n __spreadArrays: __spreadArrays,\r\n __spreadArray: __spreadArray,\r\n __await: __await,\r\n __asyncGenerator: __asyncGenerator,\r\n __asyncDelegator: __asyncDelegator,\r\n __asyncValues: __asyncValues,\r\n __makeTemplateObject: __makeTemplateObject,\r\n __importStar: __importStar,\r\n __importDefault: __importDefault,\r\n __classPrivateFieldGet: __classPrivateFieldGet,\r\n __classPrivateFieldSet: __classPrivateFieldSet,\r\n __classPrivateFieldIn: __classPrivateFieldIn,\r\n __addDisposableResource: __addDisposableResource,\r\n __disposeResources: __disposeResources,\r\n};\r\n","'use strict';\n\nmodule.exports = Point;\n\n/**\n * A standalone point geometry with useful accessor, comparison, and\n * modification methods.\n *\n * @class Point\n * @param {Number} x the x-coordinate. this could be longitude or screen\n * pixels, or any other sort of unit.\n * @param {Number} y the y-coordinate. this could be latitude or screen\n * pixels, or any other sort of unit.\n * @example\n * var point = new Point(-77, 38);\n */\nfunction Point(x, y) {\n this.x = x;\n this.y = y;\n}\n\nPoint.prototype = {\n\n /**\n * Clone this point, returning a new point that can be modified\n * without affecting the old one.\n * @return {Point} the clone\n */\n clone: function() { return new Point(this.x, this.y); },\n\n /**\n * Add this point's x & y coordinates to another point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n add: function(p) { return this.clone()._add(p); },\n\n /**\n * Subtract this point's x & y coordinates to from point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n sub: function(p) { return this.clone()._sub(p); },\n\n /**\n * Multiply this point's x & y coordinates by point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n multByPoint: function(p) { return this.clone()._multByPoint(p); },\n\n /**\n * Divide this point's x & y coordinates by point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n divByPoint: function(p) { return this.clone()._divByPoint(p); },\n\n /**\n * Multiply this point's x & y coordinates by a factor,\n * yielding a new point.\n * @param {Point} k factor\n * @return {Point} output point\n */\n mult: function(k) { return this.clone()._mult(k); },\n\n /**\n * Divide this point's x & y coordinates by a factor,\n * yielding a new point.\n * @param {Point} k factor\n * @return {Point} output point\n */\n div: function(k) { return this.clone()._div(k); },\n\n /**\n * Rotate this point around the 0, 0 origin by an angle a,\n * given in radians\n * @param {Number} a angle to rotate around, in radians\n * @return {Point} output point\n */\n rotate: function(a) { return this.clone()._rotate(a); },\n\n /**\n * Rotate this point around p point by an angle a,\n * given in radians\n * @param {Number} a angle to rotate around, in radians\n * @param {Point} p Point to rotate around\n * @return {Point} output point\n */\n rotateAround: function(a,p) { return this.clone()._rotateAround(a,p); },\n\n /**\n * Multiply this point by a 4x1 transformation matrix\n * @param {Array} m transformation matrix\n * @return {Point} output point\n */\n matMult: function(m) { return this.clone()._matMult(m); },\n\n /**\n * Calculate this point but as a unit vector from 0, 0, meaning\n * that the distance from the resulting point to the 0, 0\n * coordinate will be equal to 1 and the angle from the resulting\n * point to the 0, 0 coordinate will be the same as before.\n * @return {Point} unit vector point\n */\n unit: function() { return this.clone()._unit(); },\n\n /**\n * Compute a perpendicular point, where the new y coordinate\n * is the old x coordinate and the new x coordinate is the old y\n * coordinate multiplied by -1\n * @return {Point} perpendicular point\n */\n perp: function() { return this.clone()._perp(); },\n\n /**\n * Return a version of this point with the x & y coordinates\n * rounded to integers.\n * @return {Point} rounded point\n */\n round: function() { return this.clone()._round(); },\n\n /**\n * Return the magitude of this point: this is the Euclidean\n * distance from the 0, 0 coordinate to this point's x and y\n * coordinates.\n * @return {Number} magnitude\n */\n mag: function() {\n return Math.sqrt(this.x * this.x + this.y * this.y);\n },\n\n /**\n * Judge whether this point is equal to another point, returning\n * true or false.\n * @param {Point} other the other point\n * @return {boolean} whether the points are equal\n */\n equals: function(other) {\n return this.x === other.x &&\n this.y === other.y;\n },\n\n /**\n * Calculate the distance from this point to another point\n * @param {Point} p the other point\n * @return {Number} distance\n */\n dist: function(p) {\n return Math.sqrt(this.distSqr(p));\n },\n\n /**\n * Calculate the distance from this point to another point,\n * without the square root step. Useful if you're comparing\n * relative distances.\n * @param {Point} p the other point\n * @return {Number} distance\n */\n distSqr: function(p) {\n var dx = p.x - this.x,\n dy = p.y - this.y;\n return dx * dx + dy * dy;\n },\n\n /**\n * Get the angle from the 0, 0 coordinate to this point, in radians\n * coordinates.\n * @return {Number} angle\n */\n angle: function() {\n return Math.atan2(this.y, this.x);\n },\n\n /**\n * Get the angle from this point to another point, in radians\n * @param {Point} b the other point\n * @return {Number} angle\n */\n angleTo: function(b) {\n return Math.atan2(this.y - b.y, this.x - b.x);\n },\n\n /**\n * Get the angle between this point and another point, in radians\n * @param {Point} b the other point\n * @return {Number} angle\n */\n angleWith: function(b) {\n return this.angleWithSep(b.x, b.y);\n },\n\n /*\n * Find the angle of the two vectors, solving the formula for\n * the cross product a x b = |a||b|sin(θ) for θ.\n * @param {Number} x the x-coordinate\n * @param {Number} y the y-coordinate\n * @return {Number} the angle in radians\n */\n angleWithSep: function(x, y) {\n return Math.atan2(\n this.x * y - this.y * x,\n this.x * x + this.y * y);\n },\n\n _matMult: function(m) {\n var x = m[0] * this.x + m[1] * this.y,\n y = m[2] * this.x + m[3] * this.y;\n this.x = x;\n this.y = y;\n return this;\n },\n\n _add: function(p) {\n this.x += p.x;\n this.y += p.y;\n return this;\n },\n\n _sub: function(p) {\n this.x -= p.x;\n this.y -= p.y;\n return this;\n },\n\n _mult: function(k) {\n this.x *= k;\n this.y *= k;\n return this;\n },\n\n _div: function(k) {\n this.x /= k;\n this.y /= k;\n return this;\n },\n\n _multByPoint: function(p) {\n this.x *= p.x;\n this.y *= p.y;\n return this;\n },\n\n _divByPoint: function(p) {\n this.x /= p.x;\n this.y /= p.y;\n return this;\n },\n\n _unit: function() {\n this._div(this.mag());\n return this;\n },\n\n _perp: function() {\n var y = this.y;\n this.y = this.x;\n this.x = -y;\n return this;\n },\n\n _rotate: function(angle) {\n var cos = Math.cos(angle),\n sin = Math.sin(angle),\n x = cos * this.x - sin * this.y,\n y = sin * this.x + cos * this.y;\n this.x = x;\n this.y = y;\n return this;\n },\n\n _rotateAround: function(angle, p) {\n var cos = Math.cos(angle),\n sin = Math.sin(angle),\n x = p.x + cos * (this.x - p.x) - sin * (this.y - p.y),\n y = p.y + sin * (this.x - p.x) + cos * (this.y - p.y);\n this.x = x;\n this.y = y;\n return this;\n },\n\n _round: function() {\n this.x = Math.round(this.x);\n this.y = Math.round(this.y);\n return this;\n }\n};\n\n/**\n * Construct a point from an array if necessary, otherwise if the input\n * is already a Point, or an unknown type, return it unchanged\n * @param {Array|Point|*} a any kind of input value\n * @return {Point} constructed point, or passed-through value.\n * @example\n * // this\n * var point = Point.convert([0, 1]);\n * // is equivalent to\n * var point = new Point(0, 1);\n */\nPoint.convert = function (a) {\n if (a instanceof Point) {\n return a;\n }\n if (Array.isArray(a)) {\n return new Point(a[0], a[1]);\n }\n return a;\n};\n","'use strict';\n\nmodule.exports = UnitBezier;\n\nfunction UnitBezier(p1x, p1y, p2x, p2y) {\n // Calculate the polynomial coefficients, implicit first and last control points are (0,0) and (1,1).\n this.cx = 3.0 * p1x;\n this.bx = 3.0 * (p2x - p1x) - this.cx;\n this.ax = 1.0 - this.cx - this.bx;\n\n this.cy = 3.0 * p1y;\n this.by = 3.0 * (p2y - p1y) - this.cy;\n this.ay = 1.0 - this.cy - this.by;\n\n this.p1x = p1x;\n this.p1y = p1y;\n this.p2x = p2x;\n this.p2y = p2y;\n}\n\nUnitBezier.prototype = {\n sampleCurveX: function (t) {\n // `ax t^3 + bx t^2 + cx t' expanded using Horner's rule.\n return ((this.ax * t + this.bx) * t + this.cx) * t;\n },\n\n sampleCurveY: function (t) {\n return ((this.ay * t + this.by) * t + this.cy) * t;\n },\n\n sampleCurveDerivativeX: function (t) {\n return (3.0 * this.ax * t + 2.0 * this.bx) * t + this.cx;\n },\n\n solveCurveX: function (x, epsilon) {\n if (epsilon === undefined) epsilon = 1e-6;\n\n if (x < 0.0) return 0.0;\n if (x > 1.0) return 1.0;\n\n var t = x;\n\n // First try a few iterations of Newton's method - normally very fast.\n for (var i = 0; i < 8; i++) {\n var x2 = this.sampleCurveX(t) - x;\n if (Math.abs(x2) < epsilon) return t;\n\n var d2 = this.sampleCurveDerivativeX(t);\n if (Math.abs(d2) < 1e-6) break;\n\n t = t - x2 / d2;\n }\n\n // Fall back to the bisection method for reliability.\n var t0 = 0.0;\n var t1 = 1.0;\n t = x;\n\n for (i = 0; i < 20; i++) {\n x2 = this.sampleCurveX(t);\n if (Math.abs(x2 - x) < epsilon) break;\n\n if (x > x2) {\n t0 = t;\n } else {\n t1 = t;\n }\n\n t = (t1 - t0) * 0.5 + t0;\n }\n\n return t;\n },\n\n solve: function (x, epsilon) {\n return this.sampleCurveY(this.solveCurveX(x, epsilon));\n }\n};\n","let supportsOffscreenCanvas: boolean;\n\nexport function offscreenCanvasSupported(): boolean {\n if (supportsOffscreenCanvas == null) {\n supportsOffscreenCanvas = typeof OffscreenCanvas !== 'undefined' &&\n new OffscreenCanvas(1, 1).getContext('2d') &&\n typeof createImageBitmap === 'function';\n }\n\n return supportsOffscreenCanvas;\n}\n","import {offscreenCanvasSupported} from './offscreen_canvas_supported';\n\nlet offscreenCanvasDistorted: boolean;\n\n/**\n * Some browsers don't return the exact pixels from a canvas to prevent user fingerprinting (see #3185).\n * This function writes pixels to an OffscreenCanvas and reads them back using getImageData, returning false\n * if they don't match.\n *\n * @returns true if the browser supports OffscreenCanvas but it distorts getImageData results, false otherwise.\n */\nexport function isOffscreenCanvasDistorted(): boolean {\n if (offscreenCanvasDistorted == null) {\n offscreenCanvasDistorted = false;\n if (offscreenCanvasSupported()) {\n const size = 5;\n const canvas = new OffscreenCanvas(size, size);\n const context = canvas.getContext('2d', {willReadFrequently: true});\n if (context) {\n // fill each pixel with an RGB value that should make the byte at index i equal to i (except alpha channel):\n // [0, 1, 2, 255, 4, 5, 6, 255, 8, 9, 10, 255, ...]\n for (let i = 0; i < size * size; i++) {\n const base = i * 4;\n context.fillStyle = `rgb(${base},${base + 1},${base + 2})`;\n context.fillRect(i % size, Math.floor(i / size), 1, 1);\n }\n const data = context.getImageData(0, 0, size, size).data;\n for (let i = 0; i < size * size * 4; i++) {\n if (i % 4 !== 3 && data[i] !== i) {\n offscreenCanvasDistorted = true;\n break;\n }\n }\n }\n }\n }\n\n return offscreenCanvasDistorted || false;\n}\n","import Point from '@mapbox/point-geometry';\nimport UnitBezier from '@mapbox/unitbezier';\nimport type {Callback} from '../types/callback';\nimport {isOffscreenCanvasDistorted} from './offscreen_canvas_distorted';\nimport type {Size} from './image';\n\n/**\n * Given a value `t` that varies between 0 and 1, return\n * an interpolation function that eases between 0 and 1 in a pleasing\n * cubic in-out fashion.\n */\nexport function easeCubicInOut(t: number): number {\n if (t <= 0) return 0;\n if (t >= 1) return 1;\n const t2 = t * t,\n t3 = t2 * t;\n return 4 * (t < 0.5 ? t3 : 3 * (t - t2) + t3 - 0.75);\n}\n\n/**\n * Given given (x, y), (x1, y1) control points for a bezier curve,\n * return a function that interpolates along that curve.\n *\n * @param p1x - control point 1 x coordinate\n * @param p1y - control point 1 y coordinate\n * @param p2x - control point 2 x coordinate\n * @param p2y - control point 2 y coordinate\n */\nexport function bezier(p1x: number, p1y: number, p2x: number, p2y: number): (t: number) => number {\n const bezier = new UnitBezier(p1x, p1y, p2x, p2y);\n return function(t: number) {\n return bezier.solve(t);\n };\n}\n\n/**\n * A default bezier-curve powered easing function with\n * control points (0.25, 0.1) and (0.25, 1)\n */\nexport const defaultEasing = bezier(0.25, 0.1, 0.25, 1);\n\n/**\n * constrain n to the given range via min + max\n *\n * @param n - value\n * @param min - the minimum value to be returned\n * @param max - the maximum value to be returned\n * @returns the clamped value\n */\nexport function clamp(n: number, min: number, max: number): number {\n return Math.min(max, Math.max(min, n));\n}\n\n/**\n * constrain n to the given range, excluding the minimum, via modular arithmetic\n *\n * @param n - value\n * @param min - the minimum value to be returned, exclusive\n * @param max - the maximum value to be returned, inclusive\n * @returns constrained number\n */\nexport function wrap(n: number, min: number, max: number): number {\n const d = max - min;\n const w = ((n - min) % d + d) % d + min;\n return (w === min) ? max : w;\n}\n\n/**\n * Call an asynchronous function on an array of arguments,\n * calling `callback` with the completed results of all calls.\n *\n * @param array - input to each call of the async function.\n * @param fn - an async function with signature (data, callback)\n * @param callback - a callback run after all async work is done.\n * called with an array, containing the results of each async call.\n */\nexport function asyncAll(\n array: Array,\n fn: (item: Item, fnCallback: Callback) => void,\n callback: Callback>\n) {\n if (!array.length) { return callback(null, []); }\n let remaining = array.length;\n const results = new Array(array.length);\n let error = null;\n array.forEach((item, i) => {\n fn(item, (err, result) => {\n if (err) error = err;\n results[i] = (result as any as Result); // https://github.com/facebook/flow/issues/2123\n if (--remaining === 0) callback(error, results);\n });\n });\n}\n\n/**\n * Compute the difference between the keys in one object and the keys\n * in another object.\n *\n * @returns keys difference\n */\nexport function keysDifference(\n obj: {[key: string]: S},\n other: {[key: string]: T}\n): Array {\n const difference = [];\n for (const i in obj) {\n if (!(i in other)) {\n difference.push(i);\n }\n }\n return difference;\n}\n\n/**\n * Given a destination object and optionally many source objects,\n * copy all properties from the source objects into the destination.\n * The last source object given overrides properties from previous\n * source objects.\n *\n * @param dest - destination object\n * @param sources - sources from which properties are pulled\n */\nexport function extend(dest: any, ...sources: Array): any {\n for (const src of sources) {\n for (const k in src) {\n dest[k] = src[k];\n }\n }\n return dest;\n}\n\n/**\n * Given an object and a number of properties as strings, return version\n * of that object with only those properties.\n *\n * @param src - the object\n * @param properties - an array of property names chosen\n * to appear on the resulting object.\n * @returns object with limited properties.\n * @example\n * ```ts\n * let foo = { name: 'Charlie', age: 10 };\n * let justName = pick(foo, ['name']); // justName = { name: 'Charlie' }\n * ```\n */\nexport function pick(src: any, properties: Array): any {\n const result = {};\n for (let i = 0; i < properties.length; i++) {\n const k = properties[i];\n if (k in src) {\n result[k] = src[k];\n }\n }\n return result;\n}\n\nlet id = 1;\n\n/**\n * Return a unique numeric id, starting at 1 and incrementing with\n * each call.\n *\n * @returns unique numeric id.\n */\nexport function uniqueId(): number {\n return id++;\n}\n\n/**\n * Return whether a given value is a power of two\n */\nexport function isPowerOfTwo(value: number): boolean {\n return (Math.log(value) / Math.LN2) % 1 === 0;\n}\n\n/**\n * Return the next power of two, or the input value if already a power of two\n */\nexport function nextPowerOfTwo(value: number): number {\n if (value <= 1) return 1;\n return Math.pow(2, Math.ceil(Math.log(value) / Math.LN2));\n}\n\n/**\n * Create an object by mapping all the values of an existing object while\n * preserving their keys.\n */\nexport function mapObject(input: any, iterator: Function, context?: any): any {\n const output = {};\n for (const key in input) {\n output[key] = iterator.call(context || this, input[key], key, input);\n }\n return output;\n}\n\n/**\n * Create an object by filtering out values of an existing object.\n */\nexport function filterObject(input: any, iterator: Function, context?: any): any {\n const output = {};\n for (const key in input) {\n if (iterator.call(context || this, input[key], key, input)) {\n output[key] = input[key];\n }\n }\n return output;\n}\n\n/**\n * Deeply compares two object literals.\n * @param a - first object literal to be compared\n * @param b - second object literal to be compared\n * @returns true if the two object literals are deeply equal, false otherwise\n */\nexport function deepEqual(a?: unknown | null, b?: unknown | null): boolean {\n if (Array.isArray(a)) {\n if (!Array.isArray(b) || a.length !== b.length) return false;\n for (let i = 0; i < a.length; i++) {\n if (!deepEqual(a[i], b[i])) return false;\n }\n return true;\n }\n if (typeof a === 'object' && a !== null && b !== null) {\n if (!(typeof b === 'object')) return false;\n const keys = Object.keys(a);\n if (keys.length !== Object.keys(b).length) return false;\n for (const key in a) {\n if (!deepEqual(a[key], b[key])) return false;\n }\n return true;\n }\n return a === b;\n}\n\n/**\n * Deeply clones two objects.\n */\nexport function clone(input: T): T {\n if (Array.isArray(input)) {\n return input.map(clone) as any as T;\n } else if (typeof input === 'object' && input) {\n return mapObject(input, clone) as any as T;\n } else {\n return input;\n }\n}\n\n/**\n * Check if two arrays have at least one common element.\n */\nexport function arraysIntersect(a: Array, b: Array): boolean {\n for (let l = 0; l < a.length; l++) {\n if (b.indexOf(a[l]) >= 0) return true;\n }\n return false;\n}\n\n/**\n * Print a warning message to the console and ensure duplicate warning messages\n * are not printed.\n */\nconst warnOnceHistory: {[key: string]: boolean} = {};\n\nexport function warnOnce(message: string): void {\n if (!warnOnceHistory[message]) {\n // console isn't defined in some WebWorkers, see #2558\n if (typeof console !== 'undefined') console.warn(message);\n warnOnceHistory[message] = true;\n }\n}\n\n/**\n * Indicates if the provided Points are in a counter clockwise (true) or clockwise (false) order\n *\n * @returns true for a counter clockwise set of points\n */\n// http://bryceboe.com/2006/10/23/line-segment-intersection-algorithm/\nexport function isCounterClockwise(a: Point, b: Point, c: Point): boolean {\n return (c.y - a.y) * (b.x - a.x) > (b.y - a.y) * (c.x - a.x);\n}\n\n/**\n * For two lines a and b in 2d space, defined by any two points along the lines,\n * find the intersection point, or return null if the lines are parallel\n *\n * @param a1 - First point on line a\n * @param a2 - Second point on line a\n * @param b1 - First point on line b\n * @param b2 - Second point on line b\n *\n * @returns the intersection point of the two lines or null if they are parallel\n */\nexport function findLineIntersection(a1: Point, a2: Point, b1: Point, b2: Point): Point | null {\n const aDeltaY = a2.y - a1.y;\n const aDeltaX = a2.x - a1.x;\n const bDeltaY = b2.y - b1.y;\n const bDeltaX = b2.x - b1.x;\n\n const denominator = (bDeltaY * aDeltaX) - (bDeltaX * aDeltaY);\n\n if (denominator === 0) {\n // Lines are parallel\n return null;\n }\n\n const originDeltaY = a1.y - b1.y;\n const originDeltaX = a1.x - b1.x;\n const aInterpolation = (bDeltaX * originDeltaY - bDeltaY * originDeltaX) / denominator;\n\n // Find intersection by projecting out from origin of first segment\n return new Point(a1.x + (aInterpolation * aDeltaX), a1.y + (aInterpolation * aDeltaY));\n}\n\n/**\n * Returns the signed area for the polygon ring. Positive areas are exterior rings and\n * have a clockwise winding. Negative areas are interior rings and have a counter clockwise\n * ordering.\n *\n * @param ring - Exterior or interior ring\n */\nexport function calculateSignedArea(ring: Array): number {\n let sum = 0;\n for (let i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) {\n p1 = ring[i];\n p2 = ring[j];\n sum += (p2.x - p1.x) * (p1.y + p2.y);\n }\n return sum;\n}\n\n/**\n * Detects closed polygons, first + last point are equal\n *\n * @param points - array of points\n * @returns `true` if the points are a closed polygon\n */\nexport function isClosedPolygon(points: Array): boolean {\n // If it is 2 points that are the same then it is a point\n // If it is 3 points with start and end the same then it is a line\n if (points.length < 4)\n return false;\n\n const p1 = points[0];\n const p2 = points[points.length - 1];\n\n if (Math.abs(p1.x - p2.x) > 0 ||\n Math.abs(p1.y - p2.y) > 0) {\n return false;\n }\n\n // polygon simplification can produce polygons with zero area and more than 3 points\n return Math.abs(calculateSignedArea(points)) > 0.01;\n}\n\n/**\n * Converts spherical coordinates to cartesian coordinates.\n *\n * @param spherical - Spherical coordinates, in [radial, azimuthal, polar]\n * @returns cartesian coordinates in [x, y, z]\n */\n\nexport function sphericalToCartesian([r, azimuthal, polar]: [number, number, number]): {\n x: number;\n y: number;\n z: number;\n} {\n // We abstract \"north\"/\"up\" (compass-wise) to be 0° when really this is 90° (π/2):\n // correct for that here\n azimuthal += 90;\n\n // Convert azimuthal and polar angles to radians\n azimuthal *= Math.PI / 180;\n polar *= Math.PI / 180;\n\n return {\n x: r * Math.cos(azimuthal) * Math.sin(polar),\n y: r * Math.sin(azimuthal) * Math.sin(polar),\n z: r * Math.cos(polar)\n };\n}\n\n/**\n * Returns true if the when run in the web-worker context.\n *\n * @returns `true` if the when run in the web-worker context.\n */\nexport function isWorker(): boolean {\n // @ts-ignore\n return typeof WorkerGlobalScope !== 'undefined' && typeof self !== 'undefined' && self instanceof WorkerGlobalScope;\n}\n\n/**\n * Parses data from 'Cache-Control' headers.\n *\n * @param cacheControl - Value of 'Cache-Control' header\n * @returns object containing parsed header info.\n */\n\nexport function parseCacheControl(cacheControl: string): any {\n // Taken from [Wreck](https://github.com/hapijs/wreck)\n const re = /(?:^|(?:\\s*\\,\\s*))([^\\x00-\\x20\\(\\)<>@\\,;\\:\\\\\"\\/\\[\\]\\?\\=\\{\\}\\x7F]+)(?:\\=(?:([^\\x00-\\x20\\(\\)<>@\\,;\\:\\\\\"\\/\\[\\]\\?\\=\\{\\}\\x7F]+)|(?:\\\"((?:[^\"\\\\]|\\\\.)*)\\\")))?/g;\n\n const header = {};\n cacheControl.replace(re, ($0, $1, $2, $3) => {\n const value = $2 || $3;\n header[$1] = value ? value.toLowerCase() : true;\n return '';\n });\n\n if (header['max-age']) {\n const maxAge = parseInt(header['max-age'], 10);\n if (isNaN(maxAge)) delete header['max-age'];\n else header['max-age'] = maxAge;\n }\n\n return header;\n}\n\nlet _isSafari = null;\n\n/**\n * Returns true when run in WebKit derived browsers.\n * This is used as a workaround for a memory leak in Safari caused by using Transferable objects to\n * transfer data between WebWorkers and the main thread.\n * https://github.com/mapbox/mapbox-gl-js/issues/8771\n *\n * This should be removed once the underlying Safari issue is fixed.\n *\n * @param scope - Since this function is used both on the main thread and WebWorker context,\n * let the calling scope pass in the global scope object.\n * @returns `true` when run in WebKit derived browsers.\n */\nexport function isSafari(scope: any): boolean {\n if (_isSafari == null) {\n const userAgent = scope.navigator ? scope.navigator.userAgent : null;\n _isSafari = !!scope.safari ||\n !!(userAgent && (/\\b(iPad|iPhone|iPod)\\b/.test(userAgent) || (!!userAgent.match('Safari') && !userAgent.match('Chrome'))));\n }\n return _isSafari;\n}\n\nexport function storageAvailable(type: string): boolean {\n try {\n const storage = window[type];\n storage.setItem('_mapbox_test_', 1);\n storage.removeItem('_mapbox_test_');\n return true;\n } catch (e) {\n return false;\n }\n}\n\n// The following methods are from https://developer.mozilla.org/en-US/docs/Web/API/WindowBase64/Base64_encoding_and_decoding#The_Unicode_Problem\n//Unicode compliant base64 encoder for strings\nexport function b64EncodeUnicode(str: string) {\n return btoa(\n encodeURIComponent(str).replace(/%([0-9A-F]{2})/g,\n (match, p1) => {\n return String.fromCharCode(Number('0x' + p1)); //eslint-disable-line\n }\n )\n );\n}\n\n// Unicode compliant decoder for base64-encoded strings\nexport function b64DecodeUnicode(str: string) {\n return decodeURIComponent(atob(str).split('').map((c) => {\n return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2); //eslint-disable-line\n }).join(''));\n}\n\nexport function isImageBitmap(image: any): image is ImageBitmap {\n return typeof ImageBitmap !== 'undefined' && image instanceof ImageBitmap;\n}\n\n/**\n * Converts an ArrayBuffer to an ImageBitmap.\n *\n * Used mostly for testing purposes only, because mocking libs don't know how to work with ArrayBuffers, but work\n * perfectly fine with ImageBitmaps. Might also be used for environments (other than testing) not supporting\n * ArrayBuffers.\n *\n * @param data - Data to convert\n * @param callback - A callback executed after the conversion is finished. Invoked with error (if any) as the first argument and resulting image bitmap (when no error) as the second\n */\nexport function arrayBufferToImageBitmap(data: ArrayBuffer, callback: (err?: Error | null, image?: ImageBitmap | null) => void) {\n const blob: Blob = new Blob([new Uint8Array(data)], {type: 'image/png'});\n createImageBitmap(blob).then((imgBitmap) => {\n callback(null, imgBitmap);\n }).catch((e) => {\n callback(new Error(`Could not load image because of ${e.message}. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.`));\n });\n}\n\nconst transparentPngUrl = '';\n\n/**\n * Converts an ArrayBuffer to an HTMLImageElement.\n *\n * Used mostly for testing purposes only, because mocking libs don't know how to work with ArrayBuffers, but work\n * perfectly fine with ImageBitmaps. Might also be used for environments (other than testing) not supporting\n * ArrayBuffers.\n *\n * @param data - Data to convert\n * @param callback - A callback executed after the conversion is finished. Invoked with error (if any) as the first argument and resulting image element (when no error) as the second\n */\nexport function arrayBufferToImage(data: ArrayBuffer, callback: (err?: Error | null, image?: HTMLImageElement | null) => void) {\n const img: HTMLImageElement = new Image();\n img.onload = () => {\n callback(null, img);\n URL.revokeObjectURL(img.src);\n // prevent image dataURI memory leak in Safari;\n // but don't free the image immediately because it might be uploaded in the next frame\n // https://github.com/mapbox/mapbox-gl-js/issues/10226\n img.onload = null;\n window.requestAnimationFrame(() => { img.src = transparentPngUrl; });\n };\n img.onerror = () => callback(new Error('Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.'));\n const blob: Blob = new Blob([new Uint8Array(data)], {type: 'image/png'});\n img.src = data.byteLength ? URL.createObjectURL(blob) : transparentPngUrl;\n}\n\n/**\n * Computes the webcodecs VideoFrame API options to select a rectangle out of\n * an image and write it into the destination rectangle.\n *\n * Rect (x/y/width/height) select the overlapping rectangle from the source image\n * and layout (offset/stride) write that overlapping rectangle to the correct place\n * in the destination image.\n *\n * Offset is the byte offset in the dest image that the first pixel appears at\n * and stride is the number of bytes to the start of the next row:\n * ┌───────────┐\n * │ dest │\n * │ ┌───┼───────┐\n * │offset→│▓▓▓│ source│\n * │ │▓▓▓│ │\n * │ └───┼───────┘\n * │stride ⇠╌╌╌│\n * │╌╌╌╌╌╌→ │\n * └───────────┘\n *\n * @param image - source image containing a width and height attribute\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns the layout and rect options to pass into VideoFrame API\n */\nfunction computeVideoFrameParameters(image: Size, x: number, y: number, width: number, height: number): VideoFrameCopyToOptions {\n const destRowOffset = Math.max(-x, 0) * 4;\n const firstSourceRow = Math.max(0, y);\n const firstDestRow = firstSourceRow - y;\n const offset = firstDestRow * width * 4 + destRowOffset;\n const stride = width * 4;\n\n const sourceLeft = Math.max(0, x);\n const sourceTop = Math.max(0, y);\n const sourceRight = Math.min(image.width, x + width);\n const sourceBottom = Math.min(image.height, y + height);\n return {\n rect: {\n x: sourceLeft,\n y: sourceTop,\n width: sourceRight - sourceLeft,\n height: sourceBottom - sourceTop\n },\n layout: [{offset, stride}]\n };\n}\n\n/**\n * Reads pixels from an ImageBitmap/Image/canvas using webcodec VideoFrame API.\n *\n * @param data - image, imagebitmap, or canvas to parse\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns a promise containing the parsed RGBA pixel values of the image, or the error if an error occurred\n */\nexport async function readImageUsingVideoFrame(\n image: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas,\n x: number, y: number, width: number, height: number\n): Promise {\n if (typeof VideoFrame === 'undefined') {\n throw new Error('VideoFrame not supported');\n }\n const frame = new VideoFrame(image, {timestamp: 0});\n try {\n const format = frame?.format;\n if (!format || !(format.startsWith('BGR') || format.startsWith('RGB'))) {\n throw new Error(`Unrecognized format ${format}`);\n }\n const swapBR = format.startsWith('BGR');\n const result = new Uint8ClampedArray(width * height * 4);\n await frame.copyTo(result, computeVideoFrameParameters(image, x, y, width, height));\n if (swapBR) {\n for (let i = 0; i < result.length; i += 4) {\n const tmp = result[i];\n result[i] = result[i + 2];\n result[i + 2] = tmp;\n }\n }\n return result;\n } finally {\n frame.close();\n }\n}\n\nlet offscreenCanvas: OffscreenCanvas;\nlet offscreenCanvasContext: OffscreenCanvasRenderingContext2D;\n\n/**\n * Reads pixels from an ImageBitmap/Image/canvas using OffscreenCanvas\n *\n * @param data - image, imagebitmap, or canvas to parse\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns a promise containing the parsed RGBA pixel values of the image, or the error if an error occurred\n */\nexport function readImageDataUsingOffscreenCanvas(\n imgBitmap: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas,\n x: number, y: number, width: number, height: number\n): Uint8ClampedArray {\n const origWidth = imgBitmap.width;\n const origHeight = imgBitmap.height;\n // Lazily initialize OffscreenCanvas\n if (!offscreenCanvas || !offscreenCanvasContext) {\n // Dem tiles are typically 256x256\n offscreenCanvas = new OffscreenCanvas(origWidth, origHeight);\n offscreenCanvasContext = offscreenCanvas.getContext('2d', {willReadFrequently: true});\n }\n\n offscreenCanvas.width = origWidth;\n offscreenCanvas.height = origHeight;\n\n offscreenCanvasContext.drawImage(imgBitmap, 0, 0, origWidth, origHeight);\n const imgData = offscreenCanvasContext.getImageData(x, y, width, height);\n offscreenCanvasContext.clearRect(0, 0, origWidth, origHeight);\n return imgData.data;\n}\n\n/**\n * Reads RGBA pixels from an preferring OffscreenCanvas, but falling back to VideoFrame if supported and\n * the browser is mangling OffscreenCanvas getImageData results.\n *\n * @param data - image, imagebitmap, or canvas to parse\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns a promise containing the parsed RGBA pixel values of the image\n */\nexport async function getImageData(\n image: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas,\n x: number, y: number, width: number, height: number\n): Promise {\n if (isOffscreenCanvasDistorted()) {\n try {\n return await readImageUsingVideoFrame(image, x, y, width, height);\n } catch (e) {\n // fall back to OffscreenCanvas\n }\n }\n return readImageDataUsingOffscreenCanvas(image, x, y, width, height);\n}\n","import type {Cancelable} from '../types/cancelable';\n\nconst now = typeof performance !== 'undefined' && performance && performance.now ?\n performance.now.bind(performance) :\n Date.now.bind(Date);\n\nlet linkEl;\n\nlet reducedMotionQuery: MediaQueryList;\n\n/** */\nexport const browser = {\n /**\n * Provides a function that outputs milliseconds: either performance.now()\n * or a fallback to Date.now()\n */\n now,\n\n frame(fn: (paintStartTimestamp: number) => void): Cancelable {\n const frame = requestAnimationFrame(fn);\n return {cancel: () => cancelAnimationFrame(frame)};\n },\n\n getImageData(img: HTMLImageElement | ImageBitmap, padding: number = 0): ImageData {\n const context = this.getImageCanvasContext(img);\n return context.getImageData(-padding, -padding, img.width as number + 2 * padding, img.height as number + 2 * padding);\n },\n\n getImageCanvasContext(img: HTMLImageElement | ImageBitmap): CanvasRenderingContext2D {\n const canvas = window.document.createElement('canvas') as HTMLCanvasElement;\n const context = canvas.getContext('2d', {willReadFrequently: true});\n if (!context) {\n throw new Error('failed to create canvas 2d context');\n }\n canvas.width = img.width as number;\n canvas.height = img.height as number;\n context.drawImage(img, 0, 0, img.width as number, img.height as number);\n return context;\n },\n\n resolveURL(path: string) {\n if (!linkEl) linkEl = document.createElement('a');\n linkEl.href = path;\n return linkEl.href;\n },\n\n hardwareConcurrency: typeof navigator !== 'undefined' && navigator.hardwareConcurrency || 4,\n\n get prefersReducedMotion(): boolean {\n // In case your test crashes when checking matchMedia, call setMatchMedia from 'src/util/test/util'\n if (!matchMedia) return false;\n //Lazily initialize media query\n if (reducedMotionQuery == null) {\n reducedMotionQuery = matchMedia('(prefers-reduced-motion: reduce)');\n }\n return reducedMotionQuery.matches;\n },\n};\n","import Point from '@mapbox/point-geometry';\n\nexport class DOM {\n private static readonly docStyle = typeof window !== 'undefined' && window.document && window.document.documentElement.style;\n\n private static userSelect: string;\n\n private static selectProp = DOM.testProp(['userSelect', 'MozUserSelect', 'WebkitUserSelect', 'msUserSelect']);\n\n private static transformProp = DOM.testProp(['transform', 'WebkitTransform']);\n\n private static testProp(props: string[]): string {\n if (!DOM.docStyle) return props[0];\n for (let i = 0; i < props.length; i++) {\n if (props[i] in DOM.docStyle) {\n return props[i];\n }\n }\n return props[0];\n }\n\n public static create(tagName: K, className?: string, container?: HTMLElement): HTMLElementTagNameMap[K] {\n const el = window.document.createElement(tagName);\n if (className !== undefined) el.className = className;\n if (container) container.appendChild(el);\n return el;\n }\n\n public static createNS(namespaceURI: string, tagName: string) {\n const el = window.document.createElementNS(namespaceURI, tagName);\n return el;\n }\n\n public static disableDrag() {\n if (DOM.docStyle && DOM.selectProp) {\n DOM.userSelect = DOM.docStyle[DOM.selectProp];\n DOM.docStyle[DOM.selectProp] = 'none';\n }\n }\n\n public static enableDrag() {\n if (DOM.docStyle && DOM.selectProp) {\n DOM.docStyle[DOM.selectProp] = DOM.userSelect;\n }\n }\n\n public static setTransform(el: HTMLElement, value: string) {\n el.style[DOM.transformProp] = value;\n }\n\n public static addEventListener(target: HTMLElement | Window | Document, type: string, callback: EventListenerOrEventListenerObject, options: {\n passive?: boolean;\n capture?: boolean;\n } = {}) {\n if ('passive' in options) {\n target.addEventListener(type, callback, options);\n } else {\n target.addEventListener(type, callback, options.capture);\n }\n }\n\n public static removeEventListener(target: HTMLElement | Window | Document, type: string, callback: EventListenerOrEventListenerObject, options: {\n passive?: boolean;\n capture?: boolean;\n } = {}) {\n if ('passive' in options) {\n target.removeEventListener(type, callback, options);\n } else {\n target.removeEventListener(type, callback, options.capture);\n }\n }\n\n // Suppress the next click, but only if it's immediate.\n private static suppressClickInternal(e) {\n e.preventDefault();\n e.stopPropagation();\n window.removeEventListener('click', DOM.suppressClickInternal, true);\n }\n\n public static suppressClick() {\n window.addEventListener('click', DOM.suppressClickInternal, true);\n window.setTimeout(() => {\n window.removeEventListener('click', DOM.suppressClickInternal, true);\n }, 0);\n }\n\n public static mousePos(el: HTMLElement, e: MouseEvent | Touch) {\n const rect = el.getBoundingClientRect();\n return new Point(\n e.clientX - rect.left - el.clientLeft,\n e.clientY - rect.top - el.clientTop\n );\n }\n\n public static touchPos(el: HTMLElement, touches: TouchList) {\n const rect = el.getBoundingClientRect();\n const points: Point[] = [];\n for (let i = 0; i < touches.length; i++) {\n points.push(new Point(\n touches[i].clientX - rect.left - el.clientLeft,\n touches[i].clientY - rect.top - el.clientTop\n ));\n }\n return points;\n }\n\n public static mouseButton(e: MouseEvent) {\n return e.button;\n }\n\n public static remove(node: HTMLElement) {\n if (node.parentNode) {\n node.parentNode.removeChild(node);\n }\n }\n}\n","import type {Cancelable} from '../types/cancelable';\nimport type {RequestParameters, ResponseCallback} from './ajax';\n\n/**\n * This is a global config object used to store the configuration\n * It is available in the workers as well.\n * Only serializable data should be stored in it.\n */\ntype Config = {\n MAX_PARALLEL_IMAGE_REQUESTS: number;\n MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME: number;\n MAX_TILE_CACHE_ZOOM_LEVELS: number;\n REGISTERED_PROTOCOLS: {[x: string]: (requestParameters: RequestParameters, callback: ResponseCallback) => Cancelable};\n WORKER_URL: string;\n};\n\nexport const config: Config = {\n MAX_PARALLEL_IMAGE_REQUESTS: 16,\n MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME: 8,\n MAX_TILE_CACHE_ZOOM_LEVELS: 5,\n REGISTERED_PROTOCOLS: {},\n WORKER_URL: ''\n};\n","import {extend, warnOnce, isWorker} from './util';\nimport {config} from './config';\n\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\n\n/**\n * A `RequestParameters` object to be returned from Map.options.transformRequest callbacks.\n * @example\n * ```ts\n * // use transformRequest to modify requests that begin with `http://myHost`\n * transformRequest: function(url, resourceType) {\n * if (resourceType === 'Source' && url.indexOf('http://myHost') > -1) {\n * return {\n * url: url.replace('http', 'https'),\n * headers: { 'my-custom-header': true },\n * credentials: 'include' // Include cookies for cross-origin requests\n * }\n * }\n * }\n * ```\n */\nexport type RequestParameters = {\n /**\n * The URL to be requested.\n */\n url: string;\n /**\n * The headers to be sent with the request.\n */\n headers?: any;\n /**\n * Request method `'GET' | 'POST' | 'PUT'`.\n */\n method?: 'GET' | 'POST' | 'PUT';\n /**\n * Request body.\n */\n body?: string;\n /**\n * Response body type to be returned `'string' | 'json' | 'arrayBuffer'`.\n */\n type?: 'string' | 'json' | 'arrayBuffer' | 'image';\n /**\n * `'same-origin'|'include'` Use 'include' to send cookies with cross-origin requests.\n */\n credentials?: 'same-origin' | 'include';\n /**\n * If `true`, Resource Timing API information will be collected for these transformed requests and returned in a resourceTiming property of relevant data events.\n */\n collectResourceTiming?: boolean;\n /**\n * Parameters supported only by browser fetch API. Property of the Request interface contains the cache mode of the request. It controls how the request will interact with the browser's HTTP cache. (https://developer.mozilla.org/en-US/docs/Web/API/Request/cache)\n */\n cache?: RequestCache;\n};\n\n/**\n * The response callback used in various places\n */\nexport type ResponseCallback = (\n error?: Error | null,\n data?: T | null,\n cacheControl?: string | null,\n expires?: string | null\n) => void;\n\n/**\n * An error thrown when a HTTP request results in an error response.\n */\nexport class AJAXError extends Error {\n /**\n * The response's HTTP status code.\n */\n status: number;\n\n /**\n * The response's HTTP status text.\n */\n statusText: string;\n\n /**\n * The request's URL.\n */\n url: string;\n\n /**\n * The response's body.\n */\n body: Blob;\n\n /**\n * @param status - The response's HTTP status code.\n * @param statusText - The response's HTTP status text.\n * @param url - The request's URL.\n * @param body - The response's body.\n */\n constructor(status: number, statusText: string, url: string, body: Blob) {\n super(`AJAXError: ${statusText} (${status}): ${url}`);\n this.status = status;\n this.statusText = statusText;\n this.url = url;\n this.body = body;\n }\n}\n\n// Ensure that we're sending the correct referrer from blob URL worker bundles.\n// For files loaded from the local file system, `location.origin` will be set\n// to the string(!) \"null\" (Firefox), or \"file://\" (Chrome, Safari, Edge, IE),\n// and we will set an empty referrer. Otherwise, we're using the document's URL.\n/* global self */\nexport const getReferrer = isWorker() ?\n () => (self as any).worker && (self as any).worker.referrer :\n () => (window.location.protocol === 'blob:' ? window.parent : window).location.href;\n\nexport const getProtocolAction = url => config.REGISTERED_PROTOCOLS[url.substring(0, url.indexOf('://'))];\n\n// Determines whether a URL is a file:// URL. This is obviously the case if it begins\n// with file://. Relative URLs are also file:// URLs iff the original document was loaded\n// via a file:// URL.\nconst isFileURL = url => /^file:/.test(url) || (/^file:/.test(getReferrer()) && !/^\\w+:/.test(url));\n\nfunction makeFetchRequest(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n const controller = new AbortController();\n const request = new Request(requestParameters.url, {\n method: requestParameters.method || 'GET',\n body: requestParameters.body,\n credentials: requestParameters.credentials,\n headers: requestParameters.headers,\n cache: requestParameters.cache,\n referrer: getReferrer(),\n signal: controller.signal\n });\n let complete = false;\n let aborted = false;\n\n if (requestParameters.type === 'json') {\n request.headers.set('Accept', 'application/json');\n }\n\n const validateOrFetch = (err, cachedResponse?, responseIsFresh?) => {\n if (aborted) return;\n\n if (err) {\n // Do fetch in case of cache error.\n // HTTP pages in Edge trigger a security error that can be ignored.\n if (err.message !== 'SecurityError') {\n warnOnce(err);\n }\n }\n\n if (cachedResponse && responseIsFresh) {\n return finishRequest(cachedResponse);\n }\n\n if (cachedResponse) {\n // We can't do revalidation with 'If-None-Match' because then the\n // request doesn't have simple cors headers.\n }\n\n fetch(request).then(response => {\n if (response.ok) {\n return finishRequest(response);\n\n } else {\n return response.blob().then(body => callback(new AJAXError(response.status, response.statusText, requestParameters.url, body)));\n }\n }).catch(error => {\n if (error.code === 20) {\n // silence expected AbortError\n return;\n }\n callback(new Error(error.message));\n });\n };\n\n const finishRequest = (response) => {\n (\n (requestParameters.type === 'arrayBuffer' || requestParameters.type === 'image') ? response.arrayBuffer() :\n requestParameters.type === 'json' ? response.json() :\n response.text()\n ).then(result => {\n if (aborted) return;\n complete = true;\n callback(null, result, response.headers.get('Cache-Control'), response.headers.get('Expires'));\n }).catch(err => {\n if (!aborted) callback(new Error(err.message));\n });\n };\n\n validateOrFetch(null, null);\n\n return {cancel: () => {\n aborted = true;\n if (!complete) controller.abort();\n }};\n}\n\nfunction makeXMLHttpRequest(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n const xhr: XMLHttpRequest = new XMLHttpRequest();\n\n xhr.open(requestParameters.method || 'GET', requestParameters.url, true);\n if (requestParameters.type === 'arrayBuffer' || requestParameters.type === 'image') {\n xhr.responseType = 'arraybuffer';\n }\n for (const k in requestParameters.headers) {\n xhr.setRequestHeader(k, requestParameters.headers[k]);\n }\n if (requestParameters.type === 'json') {\n xhr.responseType = 'text';\n xhr.setRequestHeader('Accept', 'application/json');\n }\n xhr.withCredentials = requestParameters.credentials === 'include';\n xhr.onerror = () => {\n callback(new Error(xhr.statusText));\n };\n xhr.onload = () => {\n if (((xhr.status >= 200 && xhr.status < 300) || xhr.status === 0) && xhr.response !== null) {\n let data: unknown = xhr.response;\n if (requestParameters.type === 'json') {\n // We're manually parsing JSON here to get better error messages.\n try {\n data = JSON.parse(xhr.response);\n } catch (err) {\n return callback(err);\n }\n }\n callback(null, data, xhr.getResponseHeader('Cache-Control'), xhr.getResponseHeader('Expires'));\n } else {\n const body = new Blob([xhr.response], {type: xhr.getResponseHeader('Content-Type')});\n callback(new AJAXError(xhr.status, xhr.statusText, requestParameters.url, body));\n }\n };\n xhr.send(requestParameters.body);\n return {cancel: () => xhr.abort()};\n}\n\nexport const makeRequest = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n // We're trying to use the Fetch API if possible. However, in some situations we can't use it:\n // - IE11 doesn't support it at all. In this case, we dispatch the request to the main thread so\n // that we can get an accruate referrer header.\n // - Safari exposes window.AbortController, but it doesn't work actually abort any requests in\n // some versions (see https://bugs.webkit.org/show_bug.cgi?id=174980#c2)\n // - Requests for resources with the file:// URI scheme don't work with the Fetch API either. In\n // this case we unconditionally use XHR on the current thread since referrers don't matter.\n if (/:\\/\\//.test(requestParameters.url) && !(/^https?:|^file:/.test(requestParameters.url))) {\n if (isWorker() && (self as any).worker && (self as any).worker.actor) {\n return (self as any).worker.actor.send('getResource', requestParameters, callback);\n }\n if (!isWorker()) {\n const action = getProtocolAction(requestParameters.url) || makeFetchRequest;\n return action(requestParameters, callback);\n }\n }\n if (!isFileURL(requestParameters.url)) {\n if (fetch && Request && AbortController && Object.prototype.hasOwnProperty.call(Request.prototype, 'signal')) {\n return makeFetchRequest(requestParameters, callback);\n }\n if (isWorker() && (self as any).worker && (self as any).worker.actor) {\n const queueOnMainThread = true;\n return (self as any).worker.actor.send('getResource', requestParameters, callback, undefined, queueOnMainThread);\n }\n }\n return makeXMLHttpRequest(requestParameters, callback);\n};\n\nexport const getJSON = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n return makeRequest(extend(requestParameters, {type: 'json'}), callback);\n};\n\nexport const getArrayBuffer = function(\n requestParameters: RequestParameters,\n callback: ResponseCallback\n): Cancelable {\n return makeRequest(extend(requestParameters, {type: 'arrayBuffer'}), callback);\n};\n\nexport const postData = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n return makeRequest(extend(requestParameters, {method: 'POST'}), callback);\n};\n\nexport function sameOrigin(inComingUrl: string) {\n // URL class should be available everywhere\n // https://developer.mozilla.org/en-US/docs/Web/API/URL\n // In addtion, a relative URL \"/foo\" or \"./foo\" will throw exception in its ctor,\n // try-catch is expansive so just use a heuristic check to avoid it\n // also check data URL\n if (!inComingUrl ||\n inComingUrl.indexOf('://') <= 0 || // relative URL\n inComingUrl.indexOf('data:image/') === 0 || // data image URL\n inComingUrl.indexOf('blob:') === 0) { // blob\n return true;\n }\n const urlObj = new URL(inComingUrl);\n const locationObj = window.location;\n return urlObj.protocol === locationObj.protocol && urlObj.host === locationObj.host;\n}\n/**\n * A type used to store the tile's expiration date and cache control definition\n */\nexport type ExpiryData = {cacheControl?: string | null; expires?: Date | string | null};\nexport const getVideo = function(urls: Array, callback: Callback): Cancelable {\n const video: HTMLVideoElement = window.document.createElement('video');\n video.muted = true;\n video.onloadstart = function() {\n callback(null, video);\n };\n for (let i = 0; i < urls.length; i++) {\n const s: HTMLSourceElement = window.document.createElement('source');\n if (!sameOrigin(urls[i])) {\n video.crossOrigin = 'Anonymous';\n }\n s.src = urls[i];\n video.appendChild(s);\n }\n return {cancel: () => {}};\n};\n","export const webpSupported = {\n supported: false,\n testSupport\n};\n\nlet glForTesting: WebGLRenderingContext|WebGL2RenderingContext;\nlet webpCheckComplete = false;\nlet webpImgTest;\nlet webpImgTestOnloadComplete = false;\n\nif (typeof document !== 'undefined') {\n webpImgTest = document.createElement('img');\n webpImgTest.onload = function() {\n if (glForTesting) testWebpTextureUpload(glForTesting);\n glForTesting = null;\n webpImgTestOnloadComplete = true;\n };\n webpImgTest.onerror = function() {\n webpCheckComplete = true;\n glForTesting = null;\n };\n webpImgTest.src = '';\n}\n\nfunction testSupport(gl: WebGLRenderingContext | WebGL2RenderingContext) {\n if (webpCheckComplete || !webpImgTest) return;\n\n // HTMLImageElement.complete is set when an image is done loading it's source\n // regardless of whether the load was successful or not.\n // It's possible for an error to set HTMLImageElement.complete to true which would trigger\n // testWebpTextureUpload and mistakenly set exported.supported to true in browsers which don't support webp\n // To avoid this, we set a flag in the image's onload handler and only call testWebpTextureUpload\n // after a successful image load event.\n if (webpImgTestOnloadComplete) {\n testWebpTextureUpload(gl);\n } else {\n glForTesting = gl;\n\n }\n}\n\nfunction testWebpTextureUpload(gl: WebGLRenderingContext|WebGL2RenderingContext) {\n // Edge 18 supports WebP but not uploading a WebP image to a gl texture\n // Test support for this before allowing WebP images.\n // https://github.com/mapbox/mapbox-gl-js/issues/7671\n const texture = gl.createTexture();\n gl.bindTexture(gl.TEXTURE_2D, texture);\n\n try {\n gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, webpImgTest);\n\n // The error does not get triggered in Edge if the context is lost\n if (gl.isContextLost()) return;\n\n webpSupported.supported = true;\n } catch (e) {\n // Catch \"Unspecified Error.\" in Edge 18.\n }\n\n gl.deleteTexture(texture);\n\n webpCheckComplete = true;\n}\n","import type {Cancelable} from '../types/cancelable';\nimport {RequestParameters, ExpiryData, makeRequest, sameOrigin, getProtocolAction} from './ajax';\nimport type {Callback} from '../types/callback';\n\nimport {arrayBufferToImageBitmap, arrayBufferToImage, extend, isWorker, isImageBitmap} from './util';\nimport {webpSupported} from './webp_supported';\nimport {config} from './config';\n\n/**\n * The callback that is being called after an image was fetched\n */\nexport type GetImageCallback = (error?: Error | null, image?: HTMLImageElement | ImageBitmap | null, expiry?: ExpiryData | null) => void;\n\ntype ImageQueueThrottleControlCallback = () => boolean;\n\nexport type ImageRequestQueueItem = Cancelable & {\n requestParameters: RequestParameters;\n supportImageRefresh: boolean;\n callback: GetImageCallback;\n cancelled: boolean;\n completed: boolean;\n innerRequest?: Cancelable;\n}\n\ntype ImageQueueThrottleCallbackDictionary = {\n [Key: number]: ImageQueueThrottleControlCallback;\n}\n\ntype HTMLImageElementWithPriority = HTMLImageElement &\n{\n // fetchPriority is experimental property supported on Chromium browsers from Version 102\n // By default images are downloaded with priority low, whereas fetch request downloads with priority high\n // https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/fetchPriority\n fetchPriority?: 'auto' | 'high' | 'low';\n};\n\n/**\n * By default, the image queue is self driven, meaning as soon as one requested item is processed,\n * it will move on to next one as quickly as it can while limiting\n * the number of concurrent requests to MAX_PARALLEL_IMAGE_REQUESTS. The default behavior\n * ensures that static views of the map can be rendered with minimal delay.\n *\n * However, the default behavior can prevent dynamic views of the map from rendering\n * smoothly in that many requests can finish in one render frame, putting too much pressure on GPU.\n *\n * When the view of the map is moving dynamically, smoother frame rates can be achieved\n * by throttling the number of items processed by the queue per frame. This can be\n * accomplished by using {@link addThrottleControl} to allow the caller to\n * use a lambda function to determine when the queue should be throttled (e.g. when isMoving())\n * and manually calling {@link processQueue} in the render loop.\n */\nexport namespace ImageRequest {\n let imageRequestQueue : ImageRequestQueueItem[];\n let currentParallelImageRequests:number;\n\n let throttleControlCallbackHandleCounter: number;\n let throttleControlCallbacks: ImageQueueThrottleCallbackDictionary;\n\n /**\n * Reset the image request queue, removing all pending requests.\n */\n export const resetRequestQueue = (): void => {\n imageRequestQueue = [];\n currentParallelImageRequests = 0;\n throttleControlCallbackHandleCounter = 0;\n throttleControlCallbacks = {};\n };\n\n /**\n * Install a callback to control when image queue throttling is desired.\n * (e.g. when the map view is moving)\n * @param callback - The callback function to install\n * @returns handle that identifies the installed callback.\n */\n export const addThrottleControl = (callback: ImageQueueThrottleControlCallback): number => {\n const handle = throttleControlCallbackHandleCounter++;\n throttleControlCallbacks[handle] = callback;\n return handle;\n };\n\n /**\n * Remove a previously installed callback by passing in the handle returned\n * by {@link addThrottleControl}.\n * @param callbackHandle - The handle for the callback to remove.\n */\n export const removeThrottleControl = (callbackHandle: number): void => {\n delete throttleControlCallbacks[callbackHandle];\n // Try updating the queue\n processQueue();\n };\n\n /**\n * Check to see if any of the installed callbacks are requesting the queue\n * to be throttled.\n * @returns `true` if any callback is causing the queue to be throttled.\n */\n const isThrottled = (): boolean => {\n const allControlKeys = Object.keys(throttleControlCallbacks);\n let throttleingRequested = false;\n if (allControlKeys.length > 0) {\n for (const key of allControlKeys) {\n throttleingRequested = throttleControlCallbacks[key]();\n if (throttleingRequested) {\n break;\n }\n }\n }\n return throttleingRequested;\n };\n\n /**\n * Request to load an image.\n * @param requestParameters - Request parameters.\n * @param callback - Callback to issue when the request completes.\n * @param supportImageRefresh - `true`, if the image request need to support refresh based on cache headers.\n * @returns Cancelable request.\n */\n export const getImage = (\n requestParameters: RequestParameters,\n callback: GetImageCallback,\n supportImageRefresh: boolean = true\n ): ImageRequestQueueItem => {\n if (webpSupported.supported) {\n if (!requestParameters.headers) {\n requestParameters.headers = {};\n }\n requestParameters.headers.accept = 'image/webp,*/*';\n }\n\n const request:ImageRequestQueueItem = {\n requestParameters,\n supportImageRefresh,\n callback,\n cancelled: false,\n completed: false,\n cancel: () => {\n if (!request.completed && !request.cancelled) {\n request.cancelled = true;\n\n // Only reduce currentParallelImageRequests, if the image request was issued.\n if (request.innerRequest) {\n request.innerRequest.cancel();\n currentParallelImageRequests--;\n }\n\n // in the case of cancelling, it WILL move on\n processQueue();\n }\n }\n };\n\n imageRequestQueue.push(request);\n processQueue();\n return request;\n };\n\n const arrayBufferToCanvasImageSource = (data: ArrayBuffer, callback: Callback) => {\n const imageBitmapSupported = typeof createImageBitmap === 'function';\n if (imageBitmapSupported) {\n arrayBufferToImageBitmap(data, callback);\n } else {\n arrayBufferToImage(data, callback);\n }\n };\n\n const doImageRequest = (itemInQueue: ImageRequestQueueItem): Cancelable => {\n const {requestParameters, supportImageRefresh, callback} = itemInQueue;\n extend(requestParameters, {type: 'image'});\n\n // - If refreshExpiredTiles is false, then we can use HTMLImageElement to download raster images.\n // - Fetch/XHR (via MakeRequest API) will be used to download images for following scenarios:\n // 1. Style image sprite will had a issue with HTMLImageElement as described\n // here: https://github.com/mapbox/mapbox-gl-js/issues/1470\n // 2. If refreshExpiredTiles is true (default), then in order to read the image cache header,\n // fetch/XHR request will be required\n // - For any special case handling like use of AddProtocol, worker initiated request or additional headers\n // let makeRequest handle it.\n // - HtmlImageElement request automatically adds accept header for all the browser supported images\n const canUseHTMLImageElement = supportImageRefresh === false &&\n !isWorker() &&\n !getProtocolAction(requestParameters.url) &&\n (!requestParameters.headers ||\n Object.keys(requestParameters.headers).reduce((acc, item) => acc && item === 'accept', true));\n\n const action = canUseHTMLImageElement ? getImageUsingHtmlImage : makeRequest;\n return action(\n requestParameters,\n (err?: Error | null,\n data?: HTMLImageElement | ImageBitmap | ArrayBuffer | null,\n cacheControl?: string | null,\n expires?: string | null) => {\n onImageResponse(itemInQueue, callback, err, data, cacheControl, expires);\n });\n };\n\n const onImageResponse = (\n itemInQueue: ImageRequestQueueItem,\n callback:GetImageCallback,\n err?: Error | null,\n data?: HTMLImageElement | ImageBitmap | ArrayBuffer | null,\n cacheControl?: string | null,\n expires?: string | null): void => {\n if (err) {\n callback(err);\n } else if (data instanceof HTMLImageElement || isImageBitmap(data)) {\n // User using addProtocol can directly return HTMLImageElement/ImageBitmap type\n // If HtmlImageElement is used to get image then response type will be HTMLImageElement\n callback(null, data);\n } else if (data) {\n const decoratedCallback = (imgErr?: Error | null, imgResult?: CanvasImageSource | null) => {\n if (imgErr != null) {\n callback(imgErr);\n } else if (imgResult != null) {\n callback(null, imgResult as (HTMLImageElement | ImageBitmap), {cacheControl, expires});\n }\n };\n arrayBufferToCanvasImageSource(data, decoratedCallback);\n }\n if (!itemInQueue.cancelled) {\n itemInQueue.completed = true;\n currentParallelImageRequests--;\n\n processQueue();\n }\n };\n\n /**\n * Process some number of items in the image request queue.\n */\n const processQueue = (): void => {\n\n const maxImageRequests = isThrottled() ?\n config.MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME :\n config.MAX_PARALLEL_IMAGE_REQUESTS;\n\n // limit concurrent image loads to help with raster sources performance on big screens\n for (let numImageRequests = currentParallelImageRequests;\n numImageRequests < maxImageRequests && imageRequestQueue.length > 0;\n numImageRequests++) {\n\n const topItemInQueue: ImageRequestQueueItem = imageRequestQueue.shift();\n if (topItemInQueue.cancelled) {\n numImageRequests--;\n continue;\n }\n\n const innerRequest = doImageRequest(topItemInQueue);\n\n currentParallelImageRequests++;\n\n topItemInQueue.innerRequest = innerRequest;\n }\n };\n\n const getImageUsingHtmlImage = (requestParameters: RequestParameters, callback: GetImageCallback): Cancelable => {\n const image = new Image() as HTMLImageElementWithPriority;\n const url = requestParameters.url;\n let requestCancelled = false;\n const credentials = requestParameters.credentials;\n if (credentials && credentials === 'include') {\n image.crossOrigin = 'use-credentials';\n } else if ((credentials && credentials === 'same-origin') || !sameOrigin(url)) {\n image.crossOrigin = 'anonymous';\n }\n\n image.fetchPriority = 'high';\n image.onload = () => {\n callback(null, image);\n image.onerror = image.onload = null;\n };\n image.onerror = () => {\n if (!requestCancelled) {\n callback(new Error('Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.'));\n }\n image.onerror = image.onload = null;\n };\n image.src = url;\n return {\n cancel: () => {\n requestCancelled = true;\n // Set src to '' to actually cancel the request\n image.src = '';\n }\n };\n };\n}\n\nImageRequest.resetRequestQueue();\n","import type {RequestParameters} from './ajax';\n\n/**\n * A type of MapLibre resource.\n */\nexport const enum ResourceType {\n Glyphs = 'Glyphs',\n Image = 'Image',\n Source = 'Source',\n SpriteImage = 'SpriteImage',\n SpriteJSON = 'SpriteJSON',\n Style = 'Style',\n Tile = 'Tile',\n Unknown = 'Unknown',\n}\n\n/**\n * This function is used to tranform a request.\n * It is used just before executing the relevant request.\n */\nexport type RequestTransformFunction = (url: string, resourceType?: ResourceType) => RequestParameters | undefined;\n\ntype UrlObject = {\n protocol: string;\n authority: string;\n path: string;\n params: Array;\n};\n\nexport class RequestManager {\n _transformRequestFn: RequestTransformFunction;\n\n constructor(transformRequestFn?: RequestTransformFunction) {\n this._transformRequestFn = transformRequestFn;\n }\n\n transformRequest(url: string, type: ResourceType) {\n if (this._transformRequestFn) {\n return this._transformRequestFn(url, type) || {url};\n }\n\n return {url};\n }\n\n normalizeSpriteURL(url: string, format: string, extension: string): string {\n const urlObject = parseUrl(url);\n urlObject.path += `${format}${extension}`;\n return formatUrl(urlObject);\n }\n\n setTransformRequest(transformRequest: RequestTransformFunction) {\n this._transformRequestFn = transformRequest;\n }\n}\n\nconst urlRe = /^(\\w+):\\/\\/([^/?]*)(\\/[^?]+)?\\??(.+)?/;\n\nfunction parseUrl(url: string): UrlObject {\n const parts = url.match(urlRe);\n if (!parts) {\n throw new Error(`Unable to parse URL \"${url}\"`);\n }\n return {\n protocol: parts[1],\n authority: parts[2],\n path: parts[3] || '/',\n params: parts[4] ? parts[4].split('&') : []\n };\n}\n\nfunction formatUrl(obj: UrlObject): string {\n const params = obj.params.length ? `?${obj.params.join('&')}` : '';\n return `${obj.protocol}://${obj.authority}${obj.path}${params}`;\n}\n","import {extend} from './util';\n\n/**\n * A listener method used as a callback to events\n */\nexport type Listener = (a: any) => any;\n\ntype Listeners = {[_: string]: Array};\n\nfunction _addEventListener(type: string, listener: Listener, listenerList: Listeners) {\n const listenerExists = listenerList[type] && listenerList[type].indexOf(listener) !== -1;\n if (!listenerExists) {\n listenerList[type] = listenerList[type] || [];\n listenerList[type].push(listener);\n }\n}\n\nfunction _removeEventListener(type: string, listener: Listener, listenerList: Listeners) {\n if (listenerList && listenerList[type]) {\n const index = listenerList[type].indexOf(listener);\n if (index !== -1) {\n listenerList[type].splice(index, 1);\n }\n }\n}\n\n/**\n * The event class\n */\nexport class Event {\n readonly type: string;\n\n constructor(type: string, data: any = {}) {\n extend(this, data);\n this.type = type;\n }\n}\n\ninterface ErrorLike {\n message: string;\n}\n\n/**\n * An error event\n */\nexport class ErrorEvent extends Event {\n error: ErrorLike;\n\n constructor(error: ErrorLike, data: any = {}) {\n super('error', extend({error}, data));\n }\n}\n\n/**\n * Methods mixed in to other classes for event capabilities.\n *\n * @group Event Related\n */\nexport class Evented {\n _listeners: Listeners;\n _oneTimeListeners: Listeners;\n _eventedParent: Evented;\n _eventedParentData: any | (() => any);\n\n /**\n * Adds a listener to a specified event type.\n *\n * @param type - The event type to add a listen for.\n * @param listener - The function to be called when the event is fired.\n * The listener function is called with the data object passed to `fire`,\n * extended with `target` and `type` properties.\n * @returns `this`\n */\n on(type: string, listener: Listener): this {\n this._listeners = this._listeners || {};\n _addEventListener(type, listener, this._listeners);\n\n return this;\n }\n\n /**\n * Removes a previously registered event listener.\n *\n * @param type - The event type to remove listeners for.\n * @param listener - The listener function to remove.\n * @returns `this`\n */\n off(type: string, listener: Listener) {\n _removeEventListener(type, listener, this._listeners);\n _removeEventListener(type, listener, this._oneTimeListeners);\n\n return this;\n }\n\n /**\n * Adds a listener that will be called only once to a specified event type.\n *\n * The listener will be called first time the event fires after the listener is registered.\n *\n * @param type - The event type to listen for.\n * @param listener - The function to be called when the event is fired the first time.\n * @returns `this` or a promise if a listener is not provided\n */\n once(type: string, listener?: Listener): this | Promise {\n if (!listener) {\n return new Promise((resolve) => this.once(type, resolve));\n }\n this._oneTimeListeners = this._oneTimeListeners || {};\n _addEventListener(type, listener, this._oneTimeListeners);\n\n return this;\n }\n\n fire(event: Event | string, properties?: any) {\n // Compatibility with (type: string, properties: Object) signature from previous versions.\n // See https://github.com/mapbox/mapbox-gl-js/issues/6522,\n // https://github.com/mapbox/mapbox-gl-draw/issues/766\n if (typeof event === 'string') {\n event = new Event(event, properties || {});\n }\n\n const type = event.type;\n\n if (this.listens(type)) {\n (event as any).target = this;\n\n // make sure adding or removing listeners inside other listeners won't cause an infinite loop\n const listeners = this._listeners && this._listeners[type] ? this._listeners[type].slice() : [];\n for (const listener of listeners) {\n listener.call(this, event);\n }\n\n const oneTimeListeners = this._oneTimeListeners && this._oneTimeListeners[type] ? this._oneTimeListeners[type].slice() : [];\n for (const listener of oneTimeListeners) {\n _removeEventListener(type, listener, this._oneTimeListeners);\n listener.call(this, event);\n }\n\n const parent = this._eventedParent;\n if (parent) {\n extend(\n event,\n typeof this._eventedParentData === 'function' ? this._eventedParentData() : this._eventedParentData\n );\n parent.fire(event);\n }\n\n // To ensure that no error events are dropped, print them to the\n // console if they have no listeners.\n } else if (event instanceof ErrorEvent) {\n console.error(event.error);\n }\n\n return this;\n }\n\n /**\n * Returns a true if this instance of Evented or any forwardeed instances of Evented have a listener for the specified type.\n *\n * @param type - The event type\n * @returns `true` if there is at least one registered listener for specified event type, `false` otherwise\n */\n listens(type: string): boolean {\n return (\n (this._listeners && this._listeners[type] && this._listeners[type].length > 0) ||\n (this._oneTimeListeners && this._oneTimeListeners[type] && this._oneTimeListeners[type].length > 0) ||\n (this._eventedParent && this._eventedParent.listens(type))\n );\n }\n\n /**\n * Bubble all events fired by this instance of Evented to this parent instance of Evented.\n * @returns `this`\n */\n setEventedParent(parent?: Evented | null, data?: any | (() => any)) {\n this._eventedParent = parent;\n this._eventedParentData = data;\n\n return this;\n }\n}\n","import UnitBezier from '@mapbox/unitbezier';\n\nvar $version = 8;\nvar $root = {\n\tversion: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: [\n\t\t\t8\n\t\t]\n\t},\n\tname: {\n\t\ttype: \"string\"\n\t},\n\tmetadata: {\n\t\ttype: \"*\"\n\t},\n\tcenter: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\"\n\t},\n\tzoom: {\n\t\ttype: \"number\"\n\t},\n\tbearing: {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\tunits: \"degrees\"\n\t},\n\tpitch: {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tunits: \"degrees\"\n\t},\n\tlight: {\n\t\ttype: \"light\"\n\t},\n\tterrain: {\n\t\ttype: \"terrain\"\n\t},\n\tsources: {\n\t\trequired: true,\n\t\ttype: \"sources\"\n\t},\n\tsprite: {\n\t\ttype: \"sprite\"\n\t},\n\tglyphs: {\n\t\ttype: \"string\"\n\t},\n\ttransition: {\n\t\ttype: \"transition\"\n\t},\n\tlayers: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tvalue: \"layer\"\n\t}\n};\nvar sources = {\n\t\"*\": {\n\t\ttype: \"source\"\n\t}\n};\nvar source = [\n\t\"source_vector\",\n\t\"source_raster\",\n\t\"source_raster_dem\",\n\t\"source_geojson\",\n\t\"source_video\",\n\t\"source_image\"\n];\nvar source_vector = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvector: {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\ttype: \"string\"\n\t},\n\ttiles: {\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tbounds: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t-180,\n\t\t\t-85.051129,\n\t\t\t180,\n\t\t\t85.051129\n\t\t]\n\t},\n\tscheme: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\txyz: {\n\t\t\t},\n\t\t\ttms: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"xyz\"\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 22\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tpromoteId: {\n\t\ttype: \"promoteId\"\n\t},\n\tvolatile: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\t\"*\": {\n\t\ttype: \"*\"\n\t}\n};\nvar source_raster = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\traster: {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\ttype: \"string\"\n\t},\n\ttiles: {\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tbounds: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t-180,\n\t\t\t-85.051129,\n\t\t\t180,\n\t\t\t85.051129\n\t\t]\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 22\n\t},\n\ttileSize: {\n\t\ttype: \"number\",\n\t\t\"default\": 512,\n\t\tunits: \"pixels\"\n\t},\n\tscheme: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\txyz: {\n\t\t\t},\n\t\t\ttms: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"xyz\"\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tvolatile: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\t\"*\": {\n\t\ttype: \"*\"\n\t}\n};\nvar source_raster_dem = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\t\"raster-dem\": {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\ttype: \"string\"\n\t},\n\ttiles: {\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tbounds: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t-180,\n\t\t\t-85.051129,\n\t\t\t180,\n\t\t\t85.051129\n\t\t]\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 22\n\t},\n\ttileSize: {\n\t\ttype: \"number\",\n\t\t\"default\": 512,\n\t\tunits: \"pixels\"\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tencoding: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tterrarium: {\n\t\t\t},\n\t\t\tmapbox: {\n\t\t\t},\n\t\t\tcustom: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"mapbox\"\n\t},\n\tredFactor: {\n\t\ttype: \"number\",\n\t\t\"default\": 1\n\t},\n\tblueFactor: {\n\t\ttype: \"number\",\n\t\t\"default\": 1\n\t},\n\tgreenFactor: {\n\t\ttype: \"number\",\n\t\t\"default\": 1\n\t},\n\tbaseShift: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tvolatile: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\t\"*\": {\n\t\ttype: \"*\"\n\t}\n};\nvar source_geojson = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tgeojson: {\n\t\t\t}\n\t\t}\n\t},\n\tdata: {\n\t\trequired: true,\n\t\ttype: \"*\"\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 18\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tbuffer: {\n\t\ttype: \"number\",\n\t\t\"default\": 128,\n\t\tmaximum: 512,\n\t\tminimum: 0\n\t},\n\tfilter: {\n\t\ttype: \"*\"\n\t},\n\ttolerance: {\n\t\ttype: \"number\",\n\t\t\"default\": 0.375\n\t},\n\tcluster: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\tclusterRadius: {\n\t\ttype: \"number\",\n\t\t\"default\": 50,\n\t\tminimum: 0\n\t},\n\tclusterMaxZoom: {\n\t\ttype: \"number\"\n\t},\n\tclusterMinPoints: {\n\t\ttype: \"number\"\n\t},\n\tclusterProperties: {\n\t\ttype: \"*\"\n\t},\n\tlineMetrics: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\tgenerateId: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\tpromoteId: {\n\t\ttype: \"promoteId\"\n\t}\n};\nvar source_video = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvideo: {\n\t\t\t}\n\t\t}\n\t},\n\turls: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tcoordinates: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tlength: 4,\n\t\tvalue: {\n\t\t\ttype: \"array\",\n\t\t\tlength: 2,\n\t\t\tvalue: \"number\"\n\t\t}\n\t}\n};\nvar source_image = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\timage: {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\trequired: true,\n\t\ttype: \"string\"\n\t},\n\tcoordinates: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tlength: 4,\n\t\tvalue: {\n\t\t\ttype: \"array\",\n\t\t\tlength: 2,\n\t\t\tvalue: \"number\"\n\t\t}\n\t}\n};\nvar layer = {\n\tid: {\n\t\ttype: \"string\",\n\t\trequired: true\n\t},\n\ttype: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tfill: {\n\t\t\t},\n\t\t\tline: {\n\t\t\t},\n\t\t\tsymbol: {\n\t\t\t},\n\t\t\tcircle: {\n\t\t\t},\n\t\t\theatmap: {\n\t\t\t},\n\t\t\t\"fill-extrusion\": {\n\t\t\t},\n\t\t\traster: {\n\t\t\t},\n\t\t\thillshade: {\n\t\t\t},\n\t\t\tbackground: {\n\t\t\t}\n\t\t},\n\t\trequired: true\n\t},\n\tmetadata: {\n\t\ttype: \"*\"\n\t},\n\tsource: {\n\t\ttype: \"string\"\n\t},\n\t\"source-layer\": {\n\t\ttype: \"string\"\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\tminimum: 0,\n\t\tmaximum: 24\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\tminimum: 0,\n\t\tmaximum: 24\n\t},\n\tfilter: {\n\t\ttype: \"filter\"\n\t},\n\tlayout: {\n\t\ttype: \"layout\"\n\t},\n\tpaint: {\n\t\ttype: \"paint\"\n\t}\n};\nvar layout = [\n\t\"layout_fill\",\n\t\"layout_line\",\n\t\"layout_circle\",\n\t\"layout_heatmap\",\n\t\"layout_fill-extrusion\",\n\t\"layout_symbol\",\n\t\"layout_raster\",\n\t\"layout_hillshade\",\n\t\"layout_background\"\n];\nvar layout_background = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_fill = {\n\t\"fill-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_circle = {\n\t\"circle-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_heatmap = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_line = {\n\t\"line-cap\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tbutt: {\n\t\t\t},\n\t\t\tround: {\n\t\t\t},\n\t\t\tsquare: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"butt\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-join\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tbevel: {\n\t\t\t},\n\t\t\tround: {\n\t\t\t},\n\t\t\tmiter: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"miter\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-miter-limit\": {\n\t\ttype: \"number\",\n\t\t\"default\": 2,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"line-join\": \"miter\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-round-limit\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1.05,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"line-join\": \"round\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_symbol = {\n\t\"symbol-placement\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tpoint: {\n\t\t\t},\n\t\t\tline: {\n\t\t\t},\n\t\t\t\"line-center\": {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"point\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"symbol-spacing\": {\n\t\ttype: \"number\",\n\t\t\"default\": 250,\n\t\tminimum: 1,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"symbol-placement\": \"line\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"symbol-avoid-edges\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"symbol-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"symbol-z-order\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tauto: {\n\t\t\t},\n\t\t\t\"viewport-y\": {\n\t\t\t},\n\t\t\tsource: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-allow-overlap\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t{\n\t\t\t\t\"!\": \"icon-overlap\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-overlap\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnever: {\n\t\t\t},\n\t\t\talways: {\n\t\t\t},\n\t\t\tcooperative: {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-ignore-placement\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-optional\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-rotation-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-size\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tunits: \"factor of the original icon size\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-text-fit\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnone: {\n\t\t\t},\n\t\t\twidth: {\n\t\t\t},\n\t\t\theight: {\n\t\t\t},\n\t\t\tboth: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"none\",\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-text-fit-padding\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0,\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"icon-text-fit\": [\n\t\t\t\t\t\"both\",\n\t\t\t\t\t\"width\",\n\t\t\t\t\t\"height\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-image\": {\n\t\ttype: \"resolvedImage\",\n\t\ttokens: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-rotate\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\tunits: \"degrees\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-padding\": {\n\t\ttype: \"padding\",\n\t\t\"default\": [\n\t\t\t2\n\t\t],\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-keep-upright\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t{\n\t\t\t\t\"icon-rotation-alignment\": \"map\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"line\",\n\t\t\t\t\t\"line-center\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-offset\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t},\n\t\t\ttop: {\n\t\t\t},\n\t\t\tbottom: {\n\t\t\t},\n\t\t\t\"top-left\": {\n\t\t\t},\n\t\t\t\"top-right\": {\n\t\t\t},\n\t\t\t\"bottom-left\": {\n\t\t\t},\n\t\t\t\"bottom-right\": {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"center\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-pitch-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-pitch-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-rotation-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\t\"viewport-glyph\": {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-field\": {\n\t\ttype: \"formatted\",\n\t\t\"default\": \"\",\n\t\ttokens: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-font\": {\n\t\ttype: \"array\",\n\t\tvalue: \"string\",\n\t\t\"default\": [\n\t\t\t\"Open Sans Regular\",\n\t\t\t\"Arial Unicode MS Regular\"\n\t\t],\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-size\": {\n\t\ttype: \"number\",\n\t\t\"default\": 16,\n\t\tminimum: 0,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-max-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 10,\n\t\tminimum: 0,\n\t\tunits: \"ems\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-line-height\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1.2,\n\t\tunits: \"ems\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-letter-spacing\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tunits: \"ems\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-justify\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tauto: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"center\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-radial-offset\": {\n\t\ttype: \"number\",\n\t\tunits: \"ems\",\n\t\t\"default\": 0,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\t\"property-type\": \"data-driven\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t}\n\t},\n\t\"text-variable-anchor\": {\n\t\ttype: \"array\",\n\t\tvalue: \"enum\",\n\t\tvalues: {\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t},\n\t\t\ttop: {\n\t\t\t},\n\t\t\tbottom: {\n\t\t\t},\n\t\t\t\"top-left\": {\n\t\t\t},\n\t\t\t\"top-right\": {\n\t\t\t},\n\t\t\t\"bottom-left\": {\n\t\t\t},\n\t\t\t\"bottom-right\": {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"point\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-variable-anchor-offset\": {\n\t\ttype: \"variableAnchorOffsetCollection\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"point\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t},\n\t\t\ttop: {\n\t\t\t},\n\t\t\tbottom: {\n\t\t\t},\n\t\t\t\"top-left\": {\n\t\t\t},\n\t\t\t\"top-right\": {\n\t\t\t},\n\t\t\t\"bottom-left\": {\n\t\t\t},\n\t\t\t\"bottom-right\": {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"center\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"!\": \"text-variable-anchor\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-max-angle\": {\n\t\ttype: \"number\",\n\t\t\"default\": 45,\n\t\tunits: \"degrees\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"line\",\n\t\t\t\t\t\"line-center\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-writing-mode\": {\n\t\ttype: \"array\",\n\t\tvalue: \"enum\",\n\t\tvalues: {\n\t\t\thorizontal: {\n\t\t\t},\n\t\t\tvertical: {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"point\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-rotate\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\tunits: \"degrees\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-padding\": {\n\t\ttype: \"number\",\n\t\t\"default\": 2,\n\t\tminimum: 0,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-keep-upright\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": true,\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"text-rotation-alignment\": \"map\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"line\",\n\t\t\t\t\t\"line-center\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-transform\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnone: {\n\t\t\t},\n\t\t\tuppercase: {\n\t\t\t},\n\t\t\tlowercase: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"none\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-offset\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tunits: \"ems\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"!\": \"text-radial-offset\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-allow-overlap\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"!\": \"text-overlap\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-overlap\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnever: {\n\t\t\t},\n\t\t\talways: {\n\t\t\t},\n\t\t\tcooperative: {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-ignore-placement\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-optional\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_raster = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_hillshade = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar filter = {\n\ttype: \"array\",\n\tvalue: \"*\"\n};\nvar filter_operator = {\n\ttype: \"enum\",\n\tvalues: {\n\t\t\"==\": {\n\t\t},\n\t\t\"!=\": {\n\t\t},\n\t\t\">\": {\n\t\t},\n\t\t\">=\": {\n\t\t},\n\t\t\"<\": {\n\t\t},\n\t\t\"<=\": {\n\t\t},\n\t\t\"in\": {\n\t\t},\n\t\t\"!in\": {\n\t\t},\n\t\tall: {\n\t\t},\n\t\tany: {\n\t\t},\n\t\tnone: {\n\t\t},\n\t\thas: {\n\t\t},\n\t\t\"!has\": {\n\t\t},\n\t\twithin: {\n\t\t}\n\t}\n};\nvar geometry_type = {\n\ttype: \"enum\",\n\tvalues: {\n\t\tPoint: {\n\t\t},\n\t\tLineString: {\n\t\t},\n\t\tPolygon: {\n\t\t}\n\t}\n};\nvar function_stop = {\n\ttype: \"array\",\n\tminimum: 0,\n\tmaximum: 24,\n\tvalue: [\n\t\t\"number\",\n\t\t\"color\"\n\t],\n\tlength: 2\n};\nvar expression$1 = {\n\ttype: \"array\",\n\tvalue: \"*\",\n\tminimum: 1\n};\nvar light = {\n\tanchor: {\n\t\ttype: \"enum\",\n\t\t\"default\": \"viewport\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"property-type\": \"data-constant\",\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t}\n\t},\n\tposition: {\n\t\ttype: \"array\",\n\t\t\"default\": [\n\t\t\t1.15,\n\t\t\t210,\n\t\t\t30\n\t\t],\n\t\tlength: 3,\n\t\tvalue: \"number\",\n\t\t\"property-type\": \"data-constant\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t}\n\t},\n\tcolor: {\n\t\ttype: \"color\",\n\t\t\"property-type\": \"data-constant\",\n\t\t\"default\": \"#ffffff\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\ttransition: true\n\t},\n\tintensity: {\n\t\ttype: \"number\",\n\t\t\"property-type\": \"data-constant\",\n\t\t\"default\": 0.5,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\ttransition: true\n\t}\n};\nvar terrain = {\n\tsource: {\n\t\ttype: \"string\",\n\t\trequired: true\n\t},\n\texaggeration: {\n\t\ttype: \"number\",\n\t\tminimum: 0,\n\t\t\"default\": 1\n\t}\n};\nvar paint = [\n\t\"paint_fill\",\n\t\"paint_line\",\n\t\"paint_circle\",\n\t\"paint_heatmap\",\n\t\"paint_fill-extrusion\",\n\t\"paint_symbol\",\n\t\"paint_raster\",\n\t\"paint_hillshade\",\n\t\"paint_background\"\n];\nvar paint_fill = {\n\t\"fill-antialias\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"fill-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-outline-color\": {\n\t\ttype: \"color\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"fill-pattern\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"fill-antialias\": true\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"fill-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded-data-driven\"\n\t}\n};\nvar paint_line = {\n\t\"line-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"line-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"line-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-gap-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-offset\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-dasharray\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"line widths\",\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"line-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded\"\n\t},\n\t\"line-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded-data-driven\"\n\t},\n\t\"line-gradient\": {\n\t\ttype: \"color\",\n\t\ttransition: false,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"line-dasharray\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"!\": \"line-pattern\"\n\t\t\t},\n\t\t\t{\n\t\t\t\tsource: \"geojson\",\n\t\t\t\thas: {\n\t\t\t\t\tlineMetrics: true\n\t\t\t\t}\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"line-progress\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"color-ramp\"\n\t}\n};\nvar paint_circle = {\n\t\"circle-radius\": {\n\t\ttype: \"number\",\n\t\t\"default\": 5,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"circle-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-pitch-scale\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-pitch-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"viewport\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-stroke-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-stroke-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-stroke-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t}\n};\nvar paint_heatmap = {\n\t\"heatmap-radius\": {\n\t\ttype: \"number\",\n\t\t\"default\": 30,\n\t\tminimum: 1,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"heatmap-weight\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"heatmap-intensity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"heatmap-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": [\n\t\t\t\"interpolate\",\n\t\t\t[\n\t\t\t\t\"linear\"\n\t\t\t],\n\t\t\t[\n\t\t\t\t\"heatmap-density\"\n\t\t\t],\n\t\t\t0,\n\t\t\t\"rgba(0, 0, 255, 0)\",\n\t\t\t0.1,\n\t\t\t\"royalblue\",\n\t\t\t0.3,\n\t\t\t\"cyan\",\n\t\t\t0.5,\n\t\t\t\"lime\",\n\t\t\t0.7,\n\t\t\t\"yellow\",\n\t\t\t1,\n\t\t\t\"red\"\n\t\t],\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"heatmap-density\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"color-ramp\"\n\t},\n\t\"heatmap-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_symbol = {\n\t\"icon-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-halo-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"rgba(0, 0, 0, 0)\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-halo-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-halo-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"icon-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\toverridable: true,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-halo-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"rgba(0, 0, 0, 0)\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-halo-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-halo-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t\"text-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_raster = {\n\t\"raster-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-hue-rotate\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\ttransition: true,\n\t\tunits: \"degrees\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-brightness-min\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-brightness-max\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-saturation\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: -1,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-contrast\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: -1,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-resampling\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tlinear: {\n\t\t\t},\n\t\t\tnearest: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"linear\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-fade-duration\": {\n\t\ttype: \"number\",\n\t\t\"default\": 300,\n\t\tminimum: 0,\n\t\ttransition: false,\n\t\tunits: \"milliseconds\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_hillshade = {\n\t\"hillshade-illumination-direction\": {\n\t\ttype: \"number\",\n\t\t\"default\": 335,\n\t\tminimum: 0,\n\t\tmaximum: 359,\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-illumination-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"viewport\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-exaggeration\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0.5,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-shadow-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-highlight-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#FFFFFF\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-accent-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_background = {\n\t\"background-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"background-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"background-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded\"\n\t},\n\t\"background-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar transition = {\n\tduration: {\n\t\ttype: \"number\",\n\t\t\"default\": 300,\n\t\tminimum: 0,\n\t\tunits: \"milliseconds\"\n\t},\n\tdelay: {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tunits: \"milliseconds\"\n\t}\n};\nvar promoteId = {\n\t\"*\": {\n\t\ttype: \"string\"\n\t}\n};\nvar v8Spec = {\n\t$version: $version,\n\t$root: $root,\n\tsources: sources,\n\tsource: source,\n\tsource_vector: source_vector,\n\tsource_raster: source_raster,\n\tsource_raster_dem: source_raster_dem,\n\tsource_geojson: source_geojson,\n\tsource_video: source_video,\n\tsource_image: source_image,\n\tlayer: layer,\n\tlayout: layout,\n\tlayout_background: layout_background,\n\tlayout_fill: layout_fill,\n\tlayout_circle: layout_circle,\n\tlayout_heatmap: layout_heatmap,\n\t\"layout_fill-extrusion\": {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n},\n\tlayout_line: layout_line,\n\tlayout_symbol: layout_symbol,\n\tlayout_raster: layout_raster,\n\tlayout_hillshade: layout_hillshade,\n\tfilter: filter,\n\tfilter_operator: filter_operator,\n\tgeometry_type: geometry_type,\n\t\"function\": {\n\texpression: {\n\t\ttype: \"expression\"\n\t},\n\tstops: {\n\t\ttype: \"array\",\n\t\tvalue: \"function_stop\"\n\t},\n\tbase: {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0\n\t},\n\tproperty: {\n\t\ttype: \"string\",\n\t\t\"default\": \"$zoom\"\n\t},\n\ttype: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tidentity: {\n\t\t\t},\n\t\t\texponential: {\n\t\t\t},\n\t\t\tinterval: {\n\t\t\t},\n\t\t\tcategorical: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"exponential\"\n\t},\n\tcolorSpace: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\trgb: {\n\t\t\t},\n\t\t\tlab: {\n\t\t\t},\n\t\t\thcl: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"rgb\"\n\t},\n\t\"default\": {\n\t\ttype: \"*\",\n\t\trequired: false\n\t}\n},\n\tfunction_stop: function_stop,\n\texpression: expression$1,\n\tlight: light,\n\tterrain: terrain,\n\tpaint: paint,\n\tpaint_fill: paint_fill,\n\t\"paint_fill-extrusion\": {\n\t\"fill-extrusion-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-extrusion-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"fill-extrusion-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-extrusion-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-extrusion-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"fill-extrusion-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-extrusion-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded-data-driven\"\n\t},\n\t\"fill-extrusion-height\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tunits: \"meters\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-extrusion-base\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tunits: \"meters\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"fill-extrusion-height\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-extrusion-vertical-gradient\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": true,\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n},\n\tpaint_line: paint_line,\n\tpaint_circle: paint_circle,\n\tpaint_heatmap: paint_heatmap,\n\tpaint_symbol: paint_symbol,\n\tpaint_raster: paint_raster,\n\tpaint_hillshade: paint_hillshade,\n\tpaint_background: paint_background,\n\ttransition: transition,\n\t\"property-type\": {\n\t\"data-driven\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"cross-faded\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"cross-faded-data-driven\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"color-ramp\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"data-constant\": {\n\t\ttype: \"property-type\"\n\t},\n\tconstant: {\n\t\ttype: \"property-type\"\n\t}\n},\n\tpromoteId: promoteId\n};\n\nconst refProperties = ['type', 'source', 'source-layer', 'minzoom', 'maxzoom', 'filter', 'layout'];\n\nfunction deref(layer, parent) {\n const result = {};\n for (const k in layer) {\n if (k !== 'ref') {\n result[k] = layer[k];\n }\n }\n refProperties.forEach((k) => {\n if (k in parent) {\n result[k] = parent[k];\n }\n });\n return result;\n}\n/**\n * Given an array of layers, some of which may contain `ref` properties\n * whose value is the `id` of another property, return a new array where\n * such layers have been augmented with the 'type', 'source', etc. properties\n * from the parent layer, and the `ref` property has been removed.\n *\n * The input is not modified. The output may contain references to portions\n * of the input.\n *\n * @private\n * @param {Array} layers\n * @returns {Array}\n */\nfunction derefLayers(layers) {\n layers = layers.slice();\n const map = Object.create(null);\n for (let i = 0; i < layers.length; i++) {\n map[layers[i].id] = layers[i];\n }\n for (let i = 0; i < layers.length; i++) {\n if ('ref' in layers[i]) {\n layers[i] = deref(layers[i], map[layers[i].ref]);\n }\n }\n return layers;\n}\n\n/**\n * Deeply compares two object literals.\n *\n * @private\n */\nfunction deepEqual(a, b) {\n if (Array.isArray(a)) {\n if (!Array.isArray(b) || a.length !== b.length)\n return false;\n for (let i = 0; i < a.length; i++) {\n if (!deepEqual(a[i], b[i]))\n return false;\n }\n return true;\n }\n if (typeof a === 'object' && a !== null && b !== null) {\n if (!(typeof b === 'object'))\n return false;\n const keys = Object.keys(a);\n if (keys.length !== Object.keys(b).length)\n return false;\n for (const key in a) {\n if (!deepEqual(a[key], b[key]))\n return false;\n }\n return true;\n }\n return a === b;\n}\n\nconst operations = {\n /*\n * { command: 'setStyle', args: [stylesheet] }\n */\n setStyle: 'setStyle',\n /*\n * { command: 'addLayer', args: [layer, 'beforeLayerId'] }\n */\n addLayer: 'addLayer',\n /*\n * { command: 'removeLayer', args: ['layerId'] }\n */\n removeLayer: 'removeLayer',\n /*\n * { command: 'setPaintProperty', args: ['layerId', 'prop', value] }\n */\n setPaintProperty: 'setPaintProperty',\n /*\n * { command: 'setLayoutProperty', args: ['layerId', 'prop', value] }\n */\n setLayoutProperty: 'setLayoutProperty',\n /*\n * { command: 'setFilter', args: ['layerId', filter] }\n */\n setFilter: 'setFilter',\n /*\n * { command: 'addSource', args: ['sourceId', source] }\n */\n addSource: 'addSource',\n /*\n * { command: 'removeSource', args: ['sourceId'] }\n */\n removeSource: 'removeSource',\n /*\n * { command: 'setGeoJSONSourceData', args: ['sourceId', data] }\n */\n setGeoJSONSourceData: 'setGeoJSONSourceData',\n /*\n * { command: 'setLayerZoomRange', args: ['layerId', 0, 22] }\n */\n setLayerZoomRange: 'setLayerZoomRange',\n /*\n * { command: 'setLayerProperty', args: ['layerId', 'prop', value] }\n */\n setLayerProperty: 'setLayerProperty',\n /*\n * { command: 'setCenter', args: [[lon, lat]] }\n */\n setCenter: 'setCenter',\n /*\n * { command: 'setZoom', args: [zoom] }\n */\n setZoom: 'setZoom',\n /*\n * { command: 'setBearing', args: [bearing] }\n */\n setBearing: 'setBearing',\n /*\n * { command: 'setPitch', args: [pitch] }\n */\n setPitch: 'setPitch',\n /*\n * { command: 'setSprite', args: ['spriteUrl'] }\n */\n setSprite: 'setSprite',\n /*\n * { command: 'setGlyphs', args: ['glyphsUrl'] }\n */\n setGlyphs: 'setGlyphs',\n /*\n * { command: 'setTransition', args: [transition] }\n */\n setTransition: 'setTransition',\n /*\n * { command: 'setLighting', args: [lightProperties] }\n */\n setLight: 'setLight'\n};\nfunction addSource(sourceId, after, commands) {\n commands.push({ command: operations.addSource, args: [sourceId, after[sourceId]] });\n}\nfunction removeSource(sourceId, commands, sourcesRemoved) {\n commands.push({ command: operations.removeSource, args: [sourceId] });\n sourcesRemoved[sourceId] = true;\n}\nfunction updateSource(sourceId, after, commands, sourcesRemoved) {\n removeSource(sourceId, commands, sourcesRemoved);\n addSource(sourceId, after, commands);\n}\nfunction canUpdateGeoJSON(before, after, sourceId) {\n let prop;\n for (prop in before[sourceId]) {\n if (!Object.prototype.hasOwnProperty.call(before[sourceId], prop))\n continue;\n if (prop !== 'data' && !deepEqual(before[sourceId][prop], after[sourceId][prop])) {\n return false;\n }\n }\n for (prop in after[sourceId]) {\n if (!Object.prototype.hasOwnProperty.call(after[sourceId], prop))\n continue;\n if (prop !== 'data' && !deepEqual(before[sourceId][prop], after[sourceId][prop])) {\n return false;\n }\n }\n return true;\n}\nfunction diffSources(before, after, commands, sourcesRemoved) {\n before = before || {};\n after = after || {};\n let sourceId;\n // look for sources to remove\n for (sourceId in before) {\n if (!Object.prototype.hasOwnProperty.call(before, sourceId))\n continue;\n if (!Object.prototype.hasOwnProperty.call(after, sourceId)) {\n removeSource(sourceId, commands, sourcesRemoved);\n }\n }\n // look for sources to add/update\n for (sourceId in after) {\n if (!Object.prototype.hasOwnProperty.call(after, sourceId))\n continue;\n if (!Object.prototype.hasOwnProperty.call(before, sourceId)) {\n addSource(sourceId, after, commands);\n }\n else if (!deepEqual(before[sourceId], after[sourceId])) {\n if (before[sourceId].type === 'geojson' && after[sourceId].type === 'geojson' && canUpdateGeoJSON(before, after, sourceId)) {\n commands.push({ command: operations.setGeoJSONSourceData, args: [sourceId, after[sourceId].data] });\n }\n else {\n // no update command, must remove then add\n updateSource(sourceId, after, commands, sourcesRemoved);\n }\n }\n }\n}\nfunction diffLayerPropertyChanges(before, after, commands, layerId, klass, command) {\n before = before || {};\n after = after || {};\n let prop;\n for (prop in before) {\n if (!Object.prototype.hasOwnProperty.call(before, prop))\n continue;\n if (!deepEqual(before[prop], after[prop])) {\n commands.push({ command, args: [layerId, prop, after[prop], klass] });\n }\n }\n for (prop in after) {\n if (!Object.prototype.hasOwnProperty.call(after, prop) || Object.prototype.hasOwnProperty.call(before, prop))\n continue;\n if (!deepEqual(before[prop], after[prop])) {\n commands.push({ command, args: [layerId, prop, after[prop], klass] });\n }\n }\n}\nfunction pluckId(layer) {\n return layer.id;\n}\nfunction indexById(group, layer) {\n group[layer.id] = layer;\n return group;\n}\nfunction diffLayers(before, after, commands) {\n before = before || [];\n after = after || [];\n // order of layers by id\n const beforeOrder = before.map(pluckId);\n const afterOrder = after.map(pluckId);\n // index of layer by id\n const beforeIndex = before.reduce(indexById, {});\n const afterIndex = after.reduce(indexById, {});\n // track order of layers as if they have been mutated\n const tracker = beforeOrder.slice();\n // layers that have been added do not need to be diffed\n const clean = Object.create(null);\n let i, d, layerId, beforeLayer, afterLayer, insertBeforeLayerId, prop;\n // remove layers\n for (i = 0, d = 0; i < beforeOrder.length; i++) {\n layerId = beforeOrder[i];\n if (!Object.prototype.hasOwnProperty.call(afterIndex, layerId)) {\n commands.push({ command: operations.removeLayer, args: [layerId] });\n tracker.splice(tracker.indexOf(layerId, d), 1);\n }\n else {\n // limit where in tracker we need to look for a match\n d++;\n }\n }\n // add/reorder layers\n for (i = 0, d = 0; i < afterOrder.length; i++) {\n // work backwards as insert is before an existing layer\n layerId = afterOrder[afterOrder.length - 1 - i];\n if (tracker[tracker.length - 1 - i] === layerId)\n continue;\n if (Object.prototype.hasOwnProperty.call(beforeIndex, layerId)) {\n // remove the layer before we insert at the correct position\n commands.push({ command: operations.removeLayer, args: [layerId] });\n tracker.splice(tracker.lastIndexOf(layerId, tracker.length - d), 1);\n }\n else {\n // limit where in tracker we need to look for a match\n d++;\n }\n // add layer at correct position\n insertBeforeLayerId = tracker[tracker.length - i];\n commands.push({ command: operations.addLayer, args: [afterIndex[layerId], insertBeforeLayerId] });\n tracker.splice(tracker.length - i, 0, layerId);\n clean[layerId] = true;\n }\n // update layers\n for (i = 0; i < afterOrder.length; i++) {\n layerId = afterOrder[i];\n beforeLayer = beforeIndex[layerId];\n afterLayer = afterIndex[layerId];\n // no need to update if previously added (new or moved)\n if (clean[layerId] || deepEqual(beforeLayer, afterLayer))\n continue;\n // If source, source-layer, or type have changes, then remove the layer\n // and add it back 'from scratch'.\n if (!deepEqual(beforeLayer.source, afterLayer.source) || !deepEqual(beforeLayer['source-layer'], afterLayer['source-layer']) || !deepEqual(beforeLayer.type, afterLayer.type)) {\n commands.push({ command: operations.removeLayer, args: [layerId] });\n // we add the layer back at the same position it was already in, so\n // there's no need to update the `tracker`\n insertBeforeLayerId = tracker[tracker.lastIndexOf(layerId) + 1];\n commands.push({ command: operations.addLayer, args: [afterLayer, insertBeforeLayerId] });\n continue;\n }\n // layout, paint, filter, minzoom, maxzoom\n diffLayerPropertyChanges(beforeLayer.layout, afterLayer.layout, commands, layerId, null, operations.setLayoutProperty);\n diffLayerPropertyChanges(beforeLayer.paint, afterLayer.paint, commands, layerId, null, operations.setPaintProperty);\n if (!deepEqual(beforeLayer.filter, afterLayer.filter)) {\n commands.push({ command: operations.setFilter, args: [layerId, afterLayer.filter] });\n }\n if (!deepEqual(beforeLayer.minzoom, afterLayer.minzoom) || !deepEqual(beforeLayer.maxzoom, afterLayer.maxzoom)) {\n commands.push({ command: operations.setLayerZoomRange, args: [layerId, afterLayer.minzoom, afterLayer.maxzoom] });\n }\n // handle all other layer props, including paint.*\n for (prop in beforeLayer) {\n if (!Object.prototype.hasOwnProperty.call(beforeLayer, prop))\n continue;\n if (prop === 'layout' || prop === 'paint' || prop === 'filter' ||\n prop === 'metadata' || prop === 'minzoom' || prop === 'maxzoom')\n continue;\n if (prop.indexOf('paint.') === 0) {\n diffLayerPropertyChanges(beforeLayer[prop], afterLayer[prop], commands, layerId, prop.slice(6), operations.setPaintProperty);\n }\n else if (!deepEqual(beforeLayer[prop], afterLayer[prop])) {\n commands.push({ command: operations.setLayerProperty, args: [layerId, prop, afterLayer[prop]] });\n }\n }\n for (prop in afterLayer) {\n if (!Object.prototype.hasOwnProperty.call(afterLayer, prop) || Object.prototype.hasOwnProperty.call(beforeLayer, prop))\n continue;\n if (prop === 'layout' || prop === 'paint' || prop === 'filter' ||\n prop === 'metadata' || prop === 'minzoom' || prop === 'maxzoom')\n continue;\n if (prop.indexOf('paint.') === 0) {\n diffLayerPropertyChanges(beforeLayer[prop], afterLayer[prop], commands, layerId, prop.slice(6), operations.setPaintProperty);\n }\n else if (!deepEqual(beforeLayer[prop], afterLayer[prop])) {\n commands.push({ command: operations.setLayerProperty, args: [layerId, prop, afterLayer[prop]] });\n }\n }\n }\n}\n/**\n * Diff two stylesheet\n *\n * Creates semanticly aware diffs that can easily be applied at runtime.\n * Operations produced by the diff closely resemble the maplibre-gl-js API. Any\n * error creating the diff will fall back to the 'setStyle' operation.\n *\n * Example diff:\n * [\n * { command: 'setConstant', args: ['@water', '#0000FF'] },\n * { command: 'setPaintProperty', args: ['background', 'background-color', 'black'] }\n * ]\n *\n * @private\n * @param {*} [before] stylesheet to compare from\n * @param {*} after stylesheet to compare to\n * @returns Array list of changes\n */\nfunction diffStyles(before, after) {\n if (!before)\n return [{ command: operations.setStyle, args: [after] }];\n let commands = [];\n try {\n // Handle changes to top-level properties\n if (!deepEqual(before.version, after.version)) {\n return [{ command: operations.setStyle, args: [after] }];\n }\n if (!deepEqual(before.center, after.center)) {\n commands.push({ command: operations.setCenter, args: [after.center] });\n }\n if (!deepEqual(before.zoom, after.zoom)) {\n commands.push({ command: operations.setZoom, args: [after.zoom] });\n }\n if (!deepEqual(before.bearing, after.bearing)) {\n commands.push({ command: operations.setBearing, args: [after.bearing] });\n }\n if (!deepEqual(before.pitch, after.pitch)) {\n commands.push({ command: operations.setPitch, args: [after.pitch] });\n }\n if (!deepEqual(before.sprite, after.sprite)) {\n commands.push({ command: operations.setSprite, args: [after.sprite] });\n }\n if (!deepEqual(before.glyphs, after.glyphs)) {\n commands.push({ command: operations.setGlyphs, args: [after.glyphs] });\n }\n if (!deepEqual(before.transition, after.transition)) {\n commands.push({ command: operations.setTransition, args: [after.transition] });\n }\n if (!deepEqual(before.light, after.light)) {\n commands.push({ command: operations.setLight, args: [after.light] });\n }\n // Handle changes to `sources`\n // If a source is to be removed, we also--before the removeSource\n // command--need to remove all the style layers that depend on it.\n const sourcesRemoved = {};\n // First collect the {add,remove}Source commands\n const removeOrAddSourceCommands = [];\n diffSources(before.sources, after.sources, removeOrAddSourceCommands, sourcesRemoved);\n // Push a removeLayer command for each style layer that depends on a\n // source that's being removed.\n // Also, exclude any such layers them from the input to `diffLayers`\n // below, so that diffLayers produces the appropriate `addLayers`\n // command\n const beforeLayers = [];\n if (before.layers) {\n before.layers.forEach((layer) => {\n if (sourcesRemoved[layer.source]) {\n commands.push({ command: operations.removeLayer, args: [layer.id] });\n }\n else {\n beforeLayers.push(layer);\n }\n });\n }\n commands = commands.concat(removeOrAddSourceCommands);\n // Handle changes to `layers`\n diffLayers(beforeLayers, after.layers, commands);\n }\n catch (e) {\n // fall back to setStyle\n console.warn('Unable to compute style diff:', e);\n commands = [{ command: operations.setStyle, args: [after] }];\n }\n return commands;\n}\n\n// Note: Do not inherit from Error. It breaks when transpiling to ES5.\nclass ValidationError {\n constructor(key, value, message, identifier) {\n this.message = (key ? `${key}: ` : '') + message;\n if (identifier)\n this.identifier = identifier;\n if (value !== null && value !== undefined && value.__line__) {\n this.line = value.__line__;\n }\n }\n}\n\n// Note: Do not inherit from Error. It breaks when transpiling to ES5.\nclass ParsingError {\n constructor(error) {\n this.error = error;\n this.message = error.message;\n const match = error.message.match(/line (\\d+)/);\n this.line = match ? parseInt(match[1], 10) : 0;\n }\n}\n\nfunction extendBy(output, ...inputs) {\n for (const input of inputs) {\n for (const k in input) {\n output[k] = input[k];\n }\n }\n return output;\n}\n\nclass ExpressionParsingError extends Error {\n constructor(key, message) {\n super(message);\n this.message = message;\n this.key = key;\n }\n}\n\n/**\n * Tracks `let` bindings during expression parsing.\n * @private\n */\nclass Scope {\n constructor(parent, bindings = []) {\n this.parent = parent;\n this.bindings = {};\n for (const [name, expression] of bindings) {\n this.bindings[name] = expression;\n }\n }\n concat(bindings) {\n return new Scope(this, bindings);\n }\n get(name) {\n if (this.bindings[name]) {\n return this.bindings[name];\n }\n if (this.parent) {\n return this.parent.get(name);\n }\n throw new Error(`${name} not found in scope.`);\n }\n has(name) {\n if (this.bindings[name])\n return true;\n return this.parent ? this.parent.has(name) : false;\n }\n}\n\nconst NullType = { kind: 'null' };\nconst NumberType = { kind: 'number' };\nconst StringType = { kind: 'string' };\nconst BooleanType = { kind: 'boolean' };\nconst ColorType = { kind: 'color' };\nconst ObjectType = { kind: 'object' };\nconst ValueType = { kind: 'value' };\nconst ErrorType = { kind: 'error' };\nconst CollatorType = { kind: 'collator' };\nconst FormattedType = { kind: 'formatted' };\nconst PaddingType = { kind: 'padding' };\nconst ResolvedImageType = { kind: 'resolvedImage' };\nconst VariableAnchorOffsetCollectionType = { kind: 'variableAnchorOffsetCollection' };\nfunction array$1(itemType, N) {\n return {\n kind: 'array',\n itemType,\n N\n };\n}\nfunction toString$1(type) {\n if (type.kind === 'array') {\n const itemType = toString$1(type.itemType);\n return typeof type.N === 'number' ?\n `array<${itemType}, ${type.N}>` :\n type.itemType.kind === 'value' ? 'array' : `array<${itemType}>`;\n }\n else {\n return type.kind;\n }\n}\nconst valueMemberTypes = [\n NullType,\n NumberType,\n StringType,\n BooleanType,\n ColorType,\n FormattedType,\n ObjectType,\n array$1(ValueType),\n PaddingType,\n ResolvedImageType,\n VariableAnchorOffsetCollectionType\n];\n/**\n * Returns null if `t` is a subtype of `expected`; otherwise returns an\n * error message.\n * @private\n */\nfunction checkSubtype(expected, t) {\n if (t.kind === 'error') {\n // Error is a subtype of every type\n return null;\n }\n else if (expected.kind === 'array') {\n if (t.kind === 'array' &&\n ((t.N === 0 && t.itemType.kind === 'value') || !checkSubtype(expected.itemType, t.itemType)) &&\n (typeof expected.N !== 'number' || expected.N === t.N)) {\n return null;\n }\n }\n else if (expected.kind === t.kind) {\n return null;\n }\n else if (expected.kind === 'value') {\n for (const memberType of valueMemberTypes) {\n if (!checkSubtype(memberType, t)) {\n return null;\n }\n }\n }\n return `Expected ${toString$1(expected)} but found ${toString$1(t)} instead.`;\n}\nfunction isValidType(provided, allowedTypes) {\n return allowedTypes.some(t => t.kind === provided.kind);\n}\nfunction isValidNativeType(provided, allowedTypes) {\n return allowedTypes.some(t => {\n if (t === 'null') {\n return provided === null;\n }\n else if (t === 'array') {\n return Array.isArray(provided);\n }\n else if (t === 'object') {\n return provided && !Array.isArray(provided) && typeof provided === 'object';\n }\n else {\n return t === typeof provided;\n }\n });\n}\n/**\n * Verify whether the specified type is of the same type as the specified sample.\n *\n * @param provided Type to verify\n * @param sample Sample type to reference\n * @returns `true` if both objects are of the same type, `false` otherwise\n * @example basic types\n * if (verifyType(outputType, ValueType)) {\n * // type narrowed to:\n * outputType.kind; // 'value'\n * }\n * @example array types\n * if (verifyType(outputType, array(NumberType))) {\n * // type narrowed to:\n * outputType.kind; // 'array'\n * outputType.itemType; // NumberTypeT\n * outputType.itemType.kind; // 'number'\n * }\n */\nfunction verifyType(provided, sample) {\n if (provided.kind === 'array' && sample.kind === 'array') {\n return provided.itemType.kind === sample.itemType.kind && typeof provided.N === 'number';\n }\n return provided.kind === sample.kind;\n}\n\n// See https://observablehq.com/@mbostock/lab-and-rgb\nconst Xn = 0.96422, Yn = 1, Zn = 0.82521, t0 = 4 / 29, t1 = 6 / 29, t2 = 3 * t1 * t1, t3 = t1 * t1 * t1, deg2rad = Math.PI / 180, rad2deg = 180 / Math.PI;\nfunction constrainAngle(angle) {\n angle = angle % 360;\n if (angle < 0) {\n angle += 360;\n }\n return angle;\n}\nfunction rgbToLab([r, g, b, alpha]) {\n r = rgb2xyz(r);\n g = rgb2xyz(g);\n b = rgb2xyz(b);\n let x, z;\n const y = xyz2lab((0.2225045 * r + 0.7168786 * g + 0.0606169 * b) / Yn);\n if (r === g && g === b) {\n x = z = y;\n }\n else {\n x = xyz2lab((0.4360747 * r + 0.3850649 * g + 0.1430804 * b) / Xn);\n z = xyz2lab((0.0139322 * r + 0.0971045 * g + 0.7141733 * b) / Zn);\n }\n const l = 116 * y - 16;\n return [(l < 0) ? 0 : l, 500 * (x - y), 200 * (y - z), alpha];\n}\nfunction rgb2xyz(x) {\n return (x <= 0.04045) ? x / 12.92 : Math.pow((x + 0.055) / 1.055, 2.4);\n}\nfunction xyz2lab(t) {\n return (t > t3) ? Math.pow(t, 1 / 3) : t / t2 + t0;\n}\nfunction labToRgb([l, a, b, alpha]) {\n let y = (l + 16) / 116, x = isNaN(a) ? y : y + a / 500, z = isNaN(b) ? y : y - b / 200;\n y = Yn * lab2xyz(y);\n x = Xn * lab2xyz(x);\n z = Zn * lab2xyz(z);\n return [\n xyz2rgb(3.1338561 * x - 1.6168667 * y - 0.4906146 * z),\n xyz2rgb(-0.9787684 * x + 1.9161415 * y + 0.0334540 * z),\n xyz2rgb(0.0719453 * x - 0.2289914 * y + 1.4052427 * z),\n alpha,\n ];\n}\nfunction xyz2rgb(x) {\n x = (x <= 0.00304) ? 12.92 * x : 1.055 * Math.pow(x, 1 / 2.4) - 0.055;\n return (x < 0) ? 0 : (x > 1) ? 1 : x; // clip to 0..1 range\n}\nfunction lab2xyz(t) {\n return (t > t1) ? t * t * t : t2 * (t - t0);\n}\nfunction rgbToHcl(rgbColor) {\n const [l, a, b, alpha] = rgbToLab(rgbColor);\n const c = Math.sqrt(a * a + b * b);\n const h = Math.round(c * 10000) ? constrainAngle(Math.atan2(b, a) * rad2deg) : NaN;\n return [h, c, l, alpha];\n}\nfunction hclToRgb([h, c, l, alpha]) {\n h = isNaN(h) ? 0 : h * deg2rad;\n return labToRgb([l, Math.cos(h) * c, Math.sin(h) * c, alpha]);\n}\n// https://drafts.csswg.org/css-color-4/#hsl-to-rgb\nfunction hslToRgb([h, s, l, alpha]) {\n h = constrainAngle(h);\n s /= 100;\n l /= 100;\n function f(n) {\n const k = (n + h / 30) % 12;\n const a = s * Math.min(l, 1 - l);\n return l - a * Math.max(-1, Math.min(k - 3, 9 - k, 1));\n }\n return [f(0), f(8), f(4), alpha];\n}\n\n/**\n * CSS color parser compliant with CSS Color 4 Specification.\n * Supports: named colors, `transparent` keyword, all rgb hex notations,\n * rgb(), rgba(), hsl() and hsla() functions.\n * Does not round the parsed values to integers from the range 0..255.\n *\n * Syntax:\n *\n * = | \n * = | \n *\n * rgb() = rgb( {3} [ / ]? ) | rgb( {3} [ / ]? )\n * rgb() = rgb( #{3} , ? ) | rgb( #{3} , ? )\n *\n * hsl() = hsl( [ / ]? )\n * hsl() = hsl( , , , ? )\n *\n * Caveats:\n * - - with optional `deg` suffix; `grad`, `rad`, `turn` are not supported\n * - `none` keyword is not supported\n * - comments inside rgb()/hsl() are not supported\n * - legacy color syntax rgba() is supported with an identical grammar and behavior to rgb()\n * - legacy color syntax hsla() is supported with an identical grammar and behavior to hsl()\n *\n * @param input CSS color string to parse.\n * @returns Color in sRGB color space, with `red`, `green`, `blue`\n * and `alpha` channels normalized to the range 0..1,\n * or `undefined` if the input is not a valid color string.\n */\nfunction parseCssColor(input) {\n input = input.toLowerCase().trim();\n if (input === 'transparent') {\n return [0, 0, 0, 0];\n }\n // 'white', 'black', 'blue'\n const namedColorsMatch = namedColors[input];\n if (namedColorsMatch) {\n const [r, g, b] = namedColorsMatch;\n return [r / 255, g / 255, b / 255, 1];\n }\n // #f0c, #f0cf, #ff00cc, #ff00ccff\n if (input.startsWith('#')) {\n const hexRegexp = /^#(?:[0-9a-f]{3,4}|[0-9a-f]{6}|[0-9a-f]{8})$/;\n if (hexRegexp.test(input)) {\n const step = input.length < 6 ? 1 : 2;\n let i = 1;\n return [\n parseHex(input.slice(i, i += step)),\n parseHex(input.slice(i, i += step)),\n parseHex(input.slice(i, i += step)),\n parseHex(input.slice(i, i + step) || 'ff'),\n ];\n }\n }\n // rgb(128 0 0), rgb(50% 0% 0%), rgba(255,0,255,0.6), rgb(255 0 255 / 60%), rgb(100% 0% 100% /.6)\n if (input.startsWith('rgb')) {\n const rgbRegExp = /^rgba?\\(\\s*([\\de.+-]+)(%)?(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)(%)?(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)(%)?(?:\\s*([,\\/])\\s*([\\de.+-]+)(%)?)?\\s*\\)$/;\n const rgbMatch = input.match(rgbRegExp);\n if (rgbMatch) {\n const [_, // eslint-disable-line @typescript-eslint/no-unused-vars\n r, // \n rp, // % (optional)\n f1, // , (optional)\n g, // \n gp, // % (optional)\n f2, // , (optional)\n b, // \n bp, // % (optional)\n f3, // ,|/ (optional)\n a, // (optional)\n ap, // % (optional)\n ] = rgbMatch;\n const argFormat = [f1 || ' ', f2 || ' ', f3].join('');\n if (argFormat === ' ' ||\n argFormat === ' /' ||\n argFormat === ',,' ||\n argFormat === ',,,') {\n const valFormat = [rp, gp, bp].join('');\n const maxValue = (valFormat === '%%%') ? 100 :\n (valFormat === '') ? 255 : 0;\n if (maxValue) {\n const rgba = [\n clamp(+r / maxValue, 0, 1),\n clamp(+g / maxValue, 0, 1),\n clamp(+b / maxValue, 0, 1),\n a ? parseAlpha(+a, ap) : 1,\n ];\n if (validateNumbers(rgba)) {\n return rgba;\n }\n // invalid numbers\n }\n // values must be all numbers or all percentages\n }\n return; // comma optional syntax requires no commas at all\n }\n }\n // hsl(120 50% 80%), hsla(120deg,50%,80%,.9), hsl(12e1 50% 80% / 90%)\n const hslRegExp = /^hsla?\\(\\s*([\\de.+-]+)(?:deg)?(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)%(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)%(?:\\s*([,\\/])\\s*([\\de.+-]+)(%)?)?\\s*\\)$/;\n const hslMatch = input.match(hslRegExp);\n if (hslMatch) {\n const [_, // eslint-disable-line @typescript-eslint/no-unused-vars\n h, // \n f1, // , (optional)\n s, // \n f2, // , (optional)\n l, // \n f3, // ,|/ (optional)\n a, // (optional)\n ap, // % (optional)\n ] = hslMatch;\n const argFormat = [f1 || ' ', f2 || ' ', f3].join('');\n if (argFormat === ' ' ||\n argFormat === ' /' ||\n argFormat === ',,' ||\n argFormat === ',,,') {\n const hsla = [\n +h,\n clamp(+s, 0, 100),\n clamp(+l, 0, 100),\n a ? parseAlpha(+a, ap) : 1,\n ];\n if (validateNumbers(hsla)) {\n return hslToRgb(hsla);\n }\n // invalid numbers\n }\n // comma optional syntax requires no commas at all\n }\n}\nfunction parseHex(hex) {\n return parseInt(hex.padEnd(2, hex), 16) / 255;\n}\nfunction parseAlpha(a, asPercentage) {\n return clamp(asPercentage ? (a / 100) : a, 0, 1);\n}\nfunction clamp(n, min, max) {\n return Math.min(Math.max(min, n), max);\n}\n/**\n * The regular expression for numeric values is not super specific, and it may\n * happen that it will accept a value that is not a valid number. In order to\n * detect and eliminate such values this function exists.\n *\n * @param array Array of uncertain numbers.\n * @returns `true` if the specified array contains only valid numbers, `false` otherwise.\n */\nfunction validateNumbers(array) {\n return !array.some(Number.isNaN);\n}\n/**\n * To generate:\n * - visit {@link https://www.w3.org/TR/css-color-4/#named-colors}\n * - run in the console:\n * @example\n * copy(`{\\n${[...document.querySelector('.named-color-table tbody').children].map((tr) => `${tr.cells[2].textContent.trim()}: [${tr.cells[4].textContent.trim().split(/\\s+/).join(', ')}],`).join('\\n')}\\n}`);\n */\nconst namedColors = {\n aliceblue: [240, 248, 255],\n antiquewhite: [250, 235, 215],\n aqua: [0, 255, 255],\n aquamarine: [127, 255, 212],\n azure: [240, 255, 255],\n beige: [245, 245, 220],\n bisque: [255, 228, 196],\n black: [0, 0, 0],\n blanchedalmond: [255, 235, 205],\n blue: [0, 0, 255],\n blueviolet: [138, 43, 226],\n brown: [165, 42, 42],\n burlywood: [222, 184, 135],\n cadetblue: [95, 158, 160],\n chartreuse: [127, 255, 0],\n chocolate: [210, 105, 30],\n coral: [255, 127, 80],\n cornflowerblue: [100, 149, 237],\n cornsilk: [255, 248, 220],\n crimson: [220, 20, 60],\n cyan: [0, 255, 255],\n darkblue: [0, 0, 139],\n darkcyan: [0, 139, 139],\n darkgoldenrod: [184, 134, 11],\n darkgray: [169, 169, 169],\n darkgreen: [0, 100, 0],\n darkgrey: [169, 169, 169],\n darkkhaki: [189, 183, 107],\n darkmagenta: [139, 0, 139],\n darkolivegreen: [85, 107, 47],\n darkorange: [255, 140, 0],\n darkorchid: [153, 50, 204],\n darkred: [139, 0, 0],\n darksalmon: [233, 150, 122],\n darkseagreen: [143, 188, 143],\n darkslateblue: [72, 61, 139],\n darkslategray: [47, 79, 79],\n darkslategrey: [47, 79, 79],\n darkturquoise: [0, 206, 209],\n darkviolet: [148, 0, 211],\n deeppink: [255, 20, 147],\n deepskyblue: [0, 191, 255],\n dimgray: [105, 105, 105],\n dimgrey: [105, 105, 105],\n dodgerblue: [30, 144, 255],\n firebrick: [178, 34, 34],\n floralwhite: [255, 250, 240],\n forestgreen: [34, 139, 34],\n fuchsia: [255, 0, 255],\n gainsboro: [220, 220, 220],\n ghostwhite: [248, 248, 255],\n gold: [255, 215, 0],\n goldenrod: [218, 165, 32],\n gray: [128, 128, 128],\n green: [0, 128, 0],\n greenyellow: [173, 255, 47],\n grey: [128, 128, 128],\n honeydew: [240, 255, 240],\n hotpink: [255, 105, 180],\n indianred: [205, 92, 92],\n indigo: [75, 0, 130],\n ivory: [255, 255, 240],\n khaki: [240, 230, 140],\n lavender: [230, 230, 250],\n lavenderblush: [255, 240, 245],\n lawngreen: [124, 252, 0],\n lemonchiffon: [255, 250, 205],\n lightblue: [173, 216, 230],\n lightcoral: [240, 128, 128],\n lightcyan: [224, 255, 255],\n lightgoldenrodyellow: [250, 250, 210],\n lightgray: [211, 211, 211],\n lightgreen: [144, 238, 144],\n lightgrey: [211, 211, 211],\n lightpink: [255, 182, 193],\n lightsalmon: [255, 160, 122],\n lightseagreen: [32, 178, 170],\n lightskyblue: [135, 206, 250],\n lightslategray: [119, 136, 153],\n lightslategrey: [119, 136, 153],\n lightsteelblue: [176, 196, 222],\n lightyellow: [255, 255, 224],\n lime: [0, 255, 0],\n limegreen: [50, 205, 50],\n linen: [250, 240, 230],\n magenta: [255, 0, 255],\n maroon: [128, 0, 0],\n mediumaquamarine: [102, 205, 170],\n mediumblue: [0, 0, 205],\n mediumorchid: [186, 85, 211],\n mediumpurple: [147, 112, 219],\n mediumseagreen: [60, 179, 113],\n mediumslateblue: [123, 104, 238],\n mediumspringgreen: [0, 250, 154],\n mediumturquoise: [72, 209, 204],\n mediumvioletred: [199, 21, 133],\n midnightblue: [25, 25, 112],\n mintcream: [245, 255, 250],\n mistyrose: [255, 228, 225],\n moccasin: [255, 228, 181],\n navajowhite: [255, 222, 173],\n navy: [0, 0, 128],\n oldlace: [253, 245, 230],\n olive: [128, 128, 0],\n olivedrab: [107, 142, 35],\n orange: [255, 165, 0],\n orangered: [255, 69, 0],\n orchid: [218, 112, 214],\n palegoldenrod: [238, 232, 170],\n palegreen: [152, 251, 152],\n paleturquoise: [175, 238, 238],\n palevioletred: [219, 112, 147],\n papayawhip: [255, 239, 213],\n peachpuff: [255, 218, 185],\n peru: [205, 133, 63],\n pink: [255, 192, 203],\n plum: [221, 160, 221],\n powderblue: [176, 224, 230],\n purple: [128, 0, 128],\n rebeccapurple: [102, 51, 153],\n red: [255, 0, 0],\n rosybrown: [188, 143, 143],\n royalblue: [65, 105, 225],\n saddlebrown: [139, 69, 19],\n salmon: [250, 128, 114],\n sandybrown: [244, 164, 96],\n seagreen: [46, 139, 87],\n seashell: [255, 245, 238],\n sienna: [160, 82, 45],\n silver: [192, 192, 192],\n skyblue: [135, 206, 235],\n slateblue: [106, 90, 205],\n slategray: [112, 128, 144],\n slategrey: [112, 128, 144],\n snow: [255, 250, 250],\n springgreen: [0, 255, 127],\n steelblue: [70, 130, 180],\n tan: [210, 180, 140],\n teal: [0, 128, 128],\n thistle: [216, 191, 216],\n tomato: [255, 99, 71],\n turquoise: [64, 224, 208],\n violet: [238, 130, 238],\n wheat: [245, 222, 179],\n white: [255, 255, 255],\n whitesmoke: [245, 245, 245],\n yellow: [255, 255, 0],\n yellowgreen: [154, 205, 50],\n};\n\n/**\n * Color representation used by WebGL.\n * Defined in sRGB color space and pre-blended with alpha.\n * @private\n */\nclass Color {\n /**\n * @param r Red component premultiplied by `alpha` 0..1\n * @param g Green component premultiplied by `alpha` 0..1\n * @param b Blue component premultiplied by `alpha` 0..1\n * @param [alpha=1] Alpha component 0..1\n * @param [premultiplied=true] Whether the `r`, `g` and `b` values have already\n * been multiplied by alpha. If `true` nothing happens if `false` then they will\n * be multiplied automatically.\n */\n constructor(r, g, b, alpha = 1, premultiplied = true) {\n this.r = r;\n this.g = g;\n this.b = b;\n this.a = alpha;\n if (!premultiplied) {\n this.r *= alpha;\n this.g *= alpha;\n this.b *= alpha;\n if (!alpha) {\n // alpha = 0 erases completely rgb channels. This behavior is not desirable\n // if this particular color is later used in color interpolation.\n // Because of that, a reference to original color is saved.\n this.overwriteGetter('rgb', [r, g, b, alpha]);\n }\n }\n }\n /**\n * Parses CSS color strings and converts colors to sRGB color space if needed.\n * Officially supported color formats:\n * - keyword, e.g. 'aquamarine' or 'steelblue'\n * - hex (with 3, 4, 6 or 8 digits), e.g. '#f0f' or '#e9bebea9'\n * - rgb and rgba, e.g. 'rgb(0,240,120)' or 'rgba(0%,94%,47%,0.1)' or 'rgb(0 240 120 / .3)'\n * - hsl and hsla, e.g. 'hsl(0,0%,83%)' or 'hsla(0,0%,83%,.5)' or 'hsl(0 0% 83% / 20%)'\n *\n * @param input CSS color string to parse.\n * @returns A `Color` instance, or `undefined` if the input is not a valid color string.\n */\n static parse(input) {\n // in zoom-and-property function input could be an instance of Color class\n if (input instanceof Color) {\n return input;\n }\n if (typeof input !== 'string') {\n return;\n }\n const rgba = parseCssColor(input);\n if (rgba) {\n return new Color(...rgba, false);\n }\n }\n /**\n * Used in color interpolation and by 'to-rgba' expression.\n *\n * @returns Gien color, with reversed alpha blending, in sRGB color space.\n */\n get rgb() {\n const { r, g, b, a } = this;\n const f = a || Infinity; // reverse alpha blending factor\n return this.overwriteGetter('rgb', [r / f, g / f, b / f, a]);\n }\n /**\n * Used in color interpolation.\n *\n * @returns Gien color, with reversed alpha blending, in HCL color space.\n */\n get hcl() {\n return this.overwriteGetter('hcl', rgbToHcl(this.rgb));\n }\n /**\n * Used in color interpolation.\n *\n * @returns Gien color, with reversed alpha blending, in LAB color space.\n */\n get lab() {\n return this.overwriteGetter('lab', rgbToLab(this.rgb));\n }\n /**\n * Lazy getter pattern. When getter is called for the first time lazy value\n * is calculated and then overwrites getter function in given object instance.\n *\n * @example:\n * const redColor = Color.parse('red');\n * let x = redColor.hcl; // this will invoke `get hcl()`, which will calculate\n * // the value of red in HCL space and invoke this `overwriteGetter` function\n * // which in turn will set a field with a key 'hcl' in the `redColor` object.\n * // In other words it will override `get hcl()` from its `Color` prototype\n * // with its own property: hcl = [calculated red value in hcl].\n * let y = redColor.hcl; // next call will no longer invoke getter but simply\n * // return the previously calculated value\n * x === y; // true - `x` is exactly the same object as `y`\n *\n * @param getterKey Getter key\n * @param lazyValue Lazily calculated value to be memoized by current instance\n * @private\n */\n overwriteGetter(getterKey, lazyValue) {\n Object.defineProperty(this, getterKey, { value: lazyValue });\n return lazyValue;\n }\n /**\n * Used by 'to-string' expression.\n *\n * @returns Serialized color in format `rgba(r,g,b,a)`\n * where r,g,b are numbers within 0..255 and alpha is number within 1..0\n *\n * @example\n * var purple = new Color.parse('purple');\n * purple.toString; // = \"rgba(128,0,128,1)\"\n * var translucentGreen = new Color.parse('rgba(26, 207, 26, .73)');\n * translucentGreen.toString(); // = \"rgba(26,207,26,0.73)\"\n */\n toString() {\n const [r, g, b, a] = this.rgb;\n return `rgba(${[r, g, b].map(n => Math.round(n * 255)).join(',')},${a})`;\n }\n}\nColor.black = new Color(0, 0, 0, 1);\nColor.white = new Color(1, 1, 1, 1);\nColor.transparent = new Color(0, 0, 0, 0);\nColor.red = new Color(1, 0, 0, 1);\n\n// Flow type declarations for Intl cribbed from\n// https://github.com/facebook/flow/issues/1270\nclass Collator {\n constructor(caseSensitive, diacriticSensitive, locale) {\n if (caseSensitive)\n this.sensitivity = diacriticSensitive ? 'variant' : 'case';\n else\n this.sensitivity = diacriticSensitive ? 'accent' : 'base';\n this.locale = locale;\n this.collator = new Intl.Collator(this.locale ? this.locale : [], { sensitivity: this.sensitivity, usage: 'search' });\n }\n compare(lhs, rhs) {\n return this.collator.compare(lhs, rhs);\n }\n resolvedLocale() {\n // We create a Collator without \"usage: search\" because we don't want\n // the search options encoded in our result (e.g. \"en-u-co-search\")\n return new Intl.Collator(this.locale ? this.locale : [])\n .resolvedOptions().locale;\n }\n}\n\nclass FormattedSection {\n constructor(text, image, scale, fontStack, textColor) {\n this.text = text;\n this.image = image;\n this.scale = scale;\n this.fontStack = fontStack;\n this.textColor = textColor;\n }\n}\nclass Formatted {\n constructor(sections) {\n this.sections = sections;\n }\n static fromString(unformatted) {\n return new Formatted([new FormattedSection(unformatted, null, null, null, null)]);\n }\n isEmpty() {\n if (this.sections.length === 0)\n return true;\n return !this.sections.some(section => section.text.length !== 0 ||\n (section.image && section.image.name.length !== 0));\n }\n static factory(text) {\n if (text instanceof Formatted) {\n return text;\n }\n else {\n return Formatted.fromString(text);\n }\n }\n toString() {\n if (this.sections.length === 0)\n return '';\n return this.sections.map(section => section.text).join('');\n }\n}\n\n/**\n * A set of four numbers representing padding around a box. Create instances from\n * bare arrays or numeric values using the static method `Padding.parse`.\n * @private\n */\nclass Padding {\n constructor(values) {\n this.values = values.slice();\n }\n /**\n * Numeric padding values\n * @param input A padding value\n * @returns A `Padding` instance, or `undefined` if the input is not a valid padding value.\n */\n static parse(input) {\n if (input instanceof Padding) {\n return input;\n }\n // Backwards compatibility: bare number is treated the same as array with single value.\n // Padding applies to all four sides.\n if (typeof input === 'number') {\n return new Padding([input, input, input, input]);\n }\n if (!Array.isArray(input)) {\n return undefined;\n }\n if (input.length < 1 || input.length > 4) {\n return undefined;\n }\n for (const val of input) {\n if (typeof val !== 'number') {\n return undefined;\n }\n }\n // Expand shortcut properties into explicit 4-sided values\n switch (input.length) {\n case 1:\n input = [input[0], input[0], input[0], input[0]];\n break;\n case 2:\n input = [input[0], input[1], input[0], input[1]];\n break;\n case 3:\n input = [input[0], input[1], input[2], input[1]];\n break;\n }\n return new Padding(input);\n }\n toString() {\n return JSON.stringify(this.values);\n }\n}\n\n/** Set of valid anchor positions, as a set for validation */\nconst anchors = new Set(['center', 'left', 'right', 'top', 'bottom', 'top-left', 'top-right', 'bottom-left', 'bottom-right']);\n/**\n * Utility class to assist managing values for text-variable-anchor-offset property. Create instances from\n * bare arrays using the static method `VariableAnchorOffsetCollection.parse`.\n * @private\n */\nclass VariableAnchorOffsetCollection {\n constructor(values) {\n this.values = values.slice();\n }\n static parse(input) {\n if (input instanceof VariableAnchorOffsetCollection) {\n return input;\n }\n if (!Array.isArray(input) ||\n input.length < 1 ||\n input.length % 2 !== 0) {\n return undefined;\n }\n for (let i = 0; i < input.length; i += 2) {\n // Elements in even positions should be anchor positions; Elements in odd positions should be offset values\n const anchorValue = input[i];\n const offsetValue = input[i + 1];\n if (typeof anchorValue !== 'string' || !anchors.has(anchorValue)) {\n return undefined;\n }\n if (!Array.isArray(offsetValue) || offsetValue.length !== 2 || typeof offsetValue[0] !== 'number' || typeof offsetValue[1] !== 'number') {\n return undefined;\n }\n }\n return new VariableAnchorOffsetCollection(input);\n }\n toString() {\n return JSON.stringify(this.values);\n }\n}\n\nclass ResolvedImage {\n constructor(options) {\n this.name = options.name;\n this.available = options.available;\n }\n toString() {\n return this.name;\n }\n static fromString(name) {\n if (!name)\n return null; // treat empty values as no image\n return new ResolvedImage({ name, available: false });\n }\n}\n\nfunction validateRGBA(r, g, b, a) {\n if (!(typeof r === 'number' && r >= 0 && r <= 255 &&\n typeof g === 'number' && g >= 0 && g <= 255 &&\n typeof b === 'number' && b >= 0 && b <= 255)) {\n const value = typeof a === 'number' ? [r, g, b, a] : [r, g, b];\n return `Invalid rgba value [${value.join(', ')}]: 'r', 'g', and 'b' must be between 0 and 255.`;\n }\n if (!(typeof a === 'undefined' || (typeof a === 'number' && a >= 0 && a <= 1))) {\n return `Invalid rgba value [${[r, g, b, a].join(', ')}]: 'a' must be between 0 and 1.`;\n }\n return null;\n}\nfunction isValue(mixed) {\n if (mixed === null ||\n typeof mixed === 'string' ||\n typeof mixed === 'boolean' ||\n typeof mixed === 'number' ||\n mixed instanceof Color ||\n mixed instanceof Collator ||\n mixed instanceof Formatted ||\n mixed instanceof Padding ||\n mixed instanceof VariableAnchorOffsetCollection ||\n mixed instanceof ResolvedImage) {\n return true;\n }\n else if (Array.isArray(mixed)) {\n for (const item of mixed) {\n if (!isValue(item)) {\n return false;\n }\n }\n return true;\n }\n else if (typeof mixed === 'object') {\n for (const key in mixed) {\n if (!isValue(mixed[key])) {\n return false;\n }\n }\n return true;\n }\n else {\n return false;\n }\n}\nfunction typeOf(value) {\n if (value === null) {\n return NullType;\n }\n else if (typeof value === 'string') {\n return StringType;\n }\n else if (typeof value === 'boolean') {\n return BooleanType;\n }\n else if (typeof value === 'number') {\n return NumberType;\n }\n else if (value instanceof Color) {\n return ColorType;\n }\n else if (value instanceof Collator) {\n return CollatorType;\n }\n else if (value instanceof Formatted) {\n return FormattedType;\n }\n else if (value instanceof Padding) {\n return PaddingType;\n }\n else if (value instanceof VariableAnchorOffsetCollection) {\n return VariableAnchorOffsetCollectionType;\n }\n else if (value instanceof ResolvedImage) {\n return ResolvedImageType;\n }\n else if (Array.isArray(value)) {\n const length = value.length;\n let itemType;\n for (const item of value) {\n const t = typeOf(item);\n if (!itemType) {\n itemType = t;\n }\n else if (itemType === t) {\n continue;\n }\n else {\n itemType = ValueType;\n break;\n }\n }\n return array$1(itemType || ValueType, length);\n }\n else {\n return ObjectType;\n }\n}\nfunction toString(value) {\n const type = typeof value;\n if (value === null) {\n return '';\n }\n else if (type === 'string' || type === 'number' || type === 'boolean') {\n return String(value);\n }\n else if (value instanceof Color || value instanceof Formatted || value instanceof Padding || value instanceof VariableAnchorOffsetCollection || value instanceof ResolvedImage) {\n return value.toString();\n }\n else {\n return JSON.stringify(value);\n }\n}\n\nclass Literal {\n constructor(type, value) {\n this.type = type;\n this.value = value;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error(`'literal' expression requires exactly one argument, but found ${args.length - 1} instead.`);\n if (!isValue(args[1]))\n return context.error('invalid value');\n const value = args[1];\n let type = typeOf(value);\n // special case: infer the item type if possible for zero-length arrays\n const expected = context.expectedType;\n if (type.kind === 'array' &&\n type.N === 0 &&\n expected &&\n expected.kind === 'array' &&\n (typeof expected.N !== 'number' || expected.N === 0)) {\n type = expected;\n }\n return new Literal(type, value);\n }\n evaluate() {\n return this.value;\n }\n eachChild() { }\n outputDefined() {\n return true;\n }\n}\n\nclass RuntimeError {\n constructor(message) {\n this.name = 'ExpressionEvaluationError';\n this.message = message;\n }\n toJSON() {\n return this.message;\n }\n}\n\nconst types$1 = {\n string: StringType,\n number: NumberType,\n boolean: BooleanType,\n object: ObjectType\n};\nclass Assertion {\n constructor(type, args) {\n this.type = type;\n this.args = args;\n }\n static parse(args, context) {\n if (args.length < 2)\n return context.error('Expected at least one argument.');\n let i = 1;\n let type;\n const name = args[0];\n if (name === 'array') {\n let itemType;\n if (args.length > 2) {\n const type = args[1];\n if (typeof type !== 'string' || !(type in types$1) || type === 'object')\n return context.error('The item type argument of \"array\" must be one of string, number, boolean', 1);\n itemType = types$1[type];\n i++;\n }\n else {\n itemType = ValueType;\n }\n let N;\n if (args.length > 3) {\n if (args[2] !== null &&\n (typeof args[2] !== 'number' ||\n args[2] < 0 ||\n args[2] !== Math.floor(args[2]))) {\n return context.error('The length argument to \"array\" must be a positive integer literal', 2);\n }\n N = args[2];\n i++;\n }\n type = array$1(itemType, N);\n }\n else {\n if (!types$1[name])\n throw new Error(`Types doesn't contain name = ${name}`);\n type = types$1[name];\n }\n const parsed = [];\n for (; i < args.length; i++) {\n const input = context.parse(args[i], i, ValueType);\n if (!input)\n return null;\n parsed.push(input);\n }\n return new Assertion(type, parsed);\n }\n evaluate(ctx) {\n for (let i = 0; i < this.args.length; i++) {\n const value = this.args[i].evaluate(ctx);\n const error = checkSubtype(this.type, typeOf(value));\n if (!error) {\n return value;\n }\n else if (i === this.args.length - 1) {\n throw new RuntimeError(`Expected value to be of type ${toString$1(this.type)}, but found ${toString$1(typeOf(value))} instead.`);\n }\n }\n throw new Error();\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return this.args.every(arg => arg.outputDefined());\n }\n}\n\nconst types = {\n 'to-boolean': BooleanType,\n 'to-color': ColorType,\n 'to-number': NumberType,\n 'to-string': StringType\n};\n/**\n * Special form for error-coalescing coercion expressions \"to-number\",\n * \"to-color\". Since these coercions can fail at runtime, they accept multiple\n * arguments, only evaluating one at a time until one succeeds.\n *\n * @private\n */\nclass Coercion {\n constructor(type, args) {\n this.type = type;\n this.args = args;\n }\n static parse(args, context) {\n if (args.length < 2)\n return context.error('Expected at least one argument.');\n const name = args[0];\n if (!types[name])\n throw new Error(`Can't parse ${name} as it is not part of the known types`);\n if ((name === 'to-boolean' || name === 'to-string') && args.length !== 2)\n return context.error('Expected one argument.');\n const type = types[name];\n const parsed = [];\n for (let i = 1; i < args.length; i++) {\n const input = context.parse(args[i], i, ValueType);\n if (!input)\n return null;\n parsed.push(input);\n }\n return new Coercion(type, parsed);\n }\n evaluate(ctx) {\n switch (this.type.kind) {\n case 'boolean':\n return Boolean(this.args[0].evaluate(ctx));\n case 'color': {\n let input;\n let error;\n for (const arg of this.args) {\n input = arg.evaluate(ctx);\n error = null;\n if (input instanceof Color) {\n return input;\n }\n else if (typeof input === 'string') {\n const c = ctx.parseColor(input);\n if (c)\n return c;\n }\n else if (Array.isArray(input)) {\n if (input.length < 3 || input.length > 4) {\n error = `Invalid rbga value ${JSON.stringify(input)}: expected an array containing either three or four numeric values.`;\n }\n else {\n error = validateRGBA(input[0], input[1], input[2], input[3]);\n }\n if (!error) {\n return new Color(input[0] / 255, input[1] / 255, input[2] / 255, input[3]);\n }\n }\n }\n throw new RuntimeError(error || `Could not parse color from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`);\n }\n case 'padding': {\n let input;\n for (const arg of this.args) {\n input = arg.evaluate(ctx);\n const pad = Padding.parse(input);\n if (pad) {\n return pad;\n }\n }\n throw new RuntimeError(`Could not parse padding from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`);\n }\n case 'variableAnchorOffsetCollection': {\n let input;\n for (const arg of this.args) {\n input = arg.evaluate(ctx);\n const coll = VariableAnchorOffsetCollection.parse(input);\n if (coll) {\n return coll;\n }\n }\n throw new RuntimeError(`Could not parse variableAnchorOffsetCollection from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`);\n }\n case 'number': {\n let value = null;\n for (const arg of this.args) {\n value = arg.evaluate(ctx);\n if (value === null)\n return 0;\n const num = Number(value);\n if (isNaN(num))\n continue;\n return num;\n }\n throw new RuntimeError(`Could not convert ${JSON.stringify(value)} to number.`);\n }\n case 'formatted':\n // There is no explicit 'to-formatted' but this coercion can be implicitly\n // created by properties that expect the 'formatted' type.\n return Formatted.fromString(toString(this.args[0].evaluate(ctx)));\n case 'resolvedImage':\n return ResolvedImage.fromString(toString(this.args[0].evaluate(ctx)));\n default:\n return toString(this.args[0].evaluate(ctx));\n }\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return this.args.every(arg => arg.outputDefined());\n }\n}\n\nconst geometryTypes = ['Unknown', 'Point', 'LineString', 'Polygon'];\nclass EvaluationContext {\n constructor() {\n this.globals = null;\n this.feature = null;\n this.featureState = null;\n this.formattedSection = null;\n this._parseColorCache = {};\n this.availableImages = null;\n this.canonical = null;\n }\n id() {\n return this.feature && 'id' in this.feature ? this.feature.id : null;\n }\n geometryType() {\n return this.feature ? typeof this.feature.type === 'number' ? geometryTypes[this.feature.type] : this.feature.type : null;\n }\n geometry() {\n return this.feature && 'geometry' in this.feature ? this.feature.geometry : null;\n }\n canonicalID() {\n return this.canonical;\n }\n properties() {\n return this.feature && this.feature.properties || {};\n }\n parseColor(input) {\n let cached = this._parseColorCache[input];\n if (!cached) {\n cached = this._parseColorCache[input] = Color.parse(input);\n }\n return cached;\n }\n}\n\n/**\n * State associated parsing at a given point in an expression tree.\n * @private\n */\nclass ParsingContext {\n constructor(registry, isConstantFunc, path = [], expectedType, scope = new Scope(), errors = []) {\n this.registry = registry;\n this.path = path;\n this.key = path.map(part => `[${part}]`).join('');\n this.scope = scope;\n this.errors = errors;\n this.expectedType = expectedType;\n this._isConstant = isConstantFunc;\n }\n /**\n * @param expr the JSON expression to parse\n * @param index the optional argument index if this expression is an argument of a parent expression that's being parsed\n * @param options\n * @param options.omitTypeAnnotations set true to omit inferred type annotations. Caller beware: with this option set, the parsed expression's type will NOT satisfy `expectedType` if it would normally be wrapped in an inferred annotation.\n * @private\n */\n parse(expr, index, expectedType, bindings, options = {}) {\n if (index) {\n return this.concat(index, expectedType, bindings)._parse(expr, options);\n }\n return this._parse(expr, options);\n }\n _parse(expr, options) {\n if (expr === null || typeof expr === 'string' || typeof expr === 'boolean' || typeof expr === 'number') {\n expr = ['literal', expr];\n }\n function annotate(parsed, type, typeAnnotation) {\n if (typeAnnotation === 'assert') {\n return new Assertion(type, [parsed]);\n }\n else if (typeAnnotation === 'coerce') {\n return new Coercion(type, [parsed]);\n }\n else {\n return parsed;\n }\n }\n if (Array.isArray(expr)) {\n if (expr.length === 0) {\n return this.error('Expected an array with at least one element. If you wanted a literal array, use [\"literal\", []].');\n }\n const op = expr[0];\n if (typeof op !== 'string') {\n this.error(`Expression name must be a string, but found ${typeof op} instead. If you wanted a literal array, use [\"literal\", [...]].`, 0);\n return null;\n }\n const Expr = this.registry[op];\n if (Expr) {\n let parsed = Expr.parse(expr, this);\n if (!parsed)\n return null;\n if (this.expectedType) {\n const expected = this.expectedType;\n const actual = parsed.type;\n // When we expect a number, string, boolean, or array but have a value, wrap it in an assertion.\n // When we expect a color or formatted string, but have a string or value, wrap it in a coercion.\n // Otherwise, we do static type-checking.\n //\n // These behaviors are overridable for:\n // * The \"coalesce\" operator, which needs to omit type annotations.\n // * String-valued properties (e.g. `text-field`), where coercion is more convenient than assertion.\n //\n if ((expected.kind === 'string' || expected.kind === 'number' || expected.kind === 'boolean' || expected.kind === 'object' || expected.kind === 'array') && actual.kind === 'value') {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'assert');\n }\n else if ((expected.kind === 'color' || expected.kind === 'formatted' || expected.kind === 'resolvedImage') && (actual.kind === 'value' || actual.kind === 'string')) {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce');\n }\n else if (expected.kind === 'padding' && (actual.kind === 'value' || actual.kind === 'number' || actual.kind === 'array')) {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce');\n }\n else if (expected.kind === 'variableAnchorOffsetCollection' && (actual.kind === 'value' || actual.kind === 'array')) {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce');\n }\n else if (this.checkSubtype(expected, actual)) {\n return null;\n }\n }\n // If an expression's arguments are all literals, we can evaluate\n // it immediately and replace it with a literal value in the\n // parsed/compiled result. Expressions that expect an image should\n // not be resolved here so we can later get the available images.\n if (!(parsed instanceof Literal) && (parsed.type.kind !== 'resolvedImage') && this._isConstant(parsed)) {\n const ec = new EvaluationContext();\n try {\n parsed = new Literal(parsed.type, parsed.evaluate(ec));\n }\n catch (e) {\n this.error(e.message);\n return null;\n }\n }\n return parsed;\n }\n return this.error(`Unknown expression \"${op}\". If you wanted a literal array, use [\"literal\", [...]].`, 0);\n }\n else if (typeof expr === 'undefined') {\n return this.error('\\'undefined\\' value invalid. Use null instead.');\n }\n else if (typeof expr === 'object') {\n return this.error('Bare objects invalid. Use [\"literal\", {...}] instead.');\n }\n else {\n return this.error(`Expected an array, but found ${typeof expr} instead.`);\n }\n }\n /**\n * Returns a copy of this context suitable for parsing the subexpression at\n * index `index`, optionally appending to 'let' binding map.\n *\n * Note that `errors` property, intended for collecting errors while\n * parsing, is copied by reference rather than cloned.\n * @private\n */\n concat(index, expectedType, bindings) {\n const path = typeof index === 'number' ? this.path.concat(index) : this.path;\n const scope = bindings ? this.scope.concat(bindings) : this.scope;\n return new ParsingContext(this.registry, this._isConstant, path, expectedType || null, scope, this.errors);\n }\n /**\n * Push a parsing (or type checking) error into the `this.errors`\n * @param error The message\n * @param keys Optionally specify the source of the error at a child\n * of the current expression at `this.key`.\n * @private\n */\n error(error, ...keys) {\n const key = `${this.key}${keys.map(k => `[${k}]`).join('')}`;\n this.errors.push(new ExpressionParsingError(key, error));\n }\n /**\n * Returns null if `t` is a subtype of `expected`; otherwise returns an\n * error message and also pushes it to `this.errors`.\n * @param expected The expected type\n * @param t The actual type\n * @returns null if `t` is a subtype of `expected`; otherwise returns an error message\n */\n checkSubtype(expected, t) {\n const error = checkSubtype(expected, t);\n if (error)\n this.error(error);\n return error;\n }\n}\n\nclass CollatorExpression {\n constructor(caseSensitive, diacriticSensitive, locale) {\n this.type = CollatorType;\n this.locale = locale;\n this.caseSensitive = caseSensitive;\n this.diacriticSensitive = diacriticSensitive;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error('Expected one argument.');\n const options = args[1];\n if (typeof options !== 'object' || Array.isArray(options))\n return context.error('Collator options argument must be an object.');\n const caseSensitive = context.parse(options['case-sensitive'] === undefined ? false : options['case-sensitive'], 1, BooleanType);\n if (!caseSensitive)\n return null;\n const diacriticSensitive = context.parse(options['diacritic-sensitive'] === undefined ? false : options['diacritic-sensitive'], 1, BooleanType);\n if (!diacriticSensitive)\n return null;\n let locale = null;\n if (options['locale']) {\n locale = context.parse(options['locale'], 1, StringType);\n if (!locale)\n return null;\n }\n return new CollatorExpression(caseSensitive, diacriticSensitive, locale);\n }\n evaluate(ctx) {\n return new Collator(this.caseSensitive.evaluate(ctx), this.diacriticSensitive.evaluate(ctx), this.locale ? this.locale.evaluate(ctx) : null);\n }\n eachChild(fn) {\n fn(this.caseSensitive);\n fn(this.diacriticSensitive);\n if (this.locale) {\n fn(this.locale);\n }\n }\n outputDefined() {\n // Technically the set of possible outputs is the combinatoric set of Collators produced\n // by all possible outputs of locale/caseSensitive/diacriticSensitive\n // But for the primary use of Collators in comparison operators, we ignore the Collator's\n // possible outputs anyway, so we can get away with leaving this false for now.\n return false;\n }\n}\n\nconst EXTENT = 8192;\nfunction updateBBox(bbox, coord) {\n bbox[0] = Math.min(bbox[0], coord[0]);\n bbox[1] = Math.min(bbox[1], coord[1]);\n bbox[2] = Math.max(bbox[2], coord[0]);\n bbox[3] = Math.max(bbox[3], coord[1]);\n}\nfunction mercatorXfromLng(lng) {\n return (180 + lng) / 360;\n}\nfunction mercatorYfromLat(lat) {\n return (180 - (180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360)))) / 360;\n}\nfunction boxWithinBox(bbox1, bbox2) {\n if (bbox1[0] <= bbox2[0])\n return false;\n if (bbox1[2] >= bbox2[2])\n return false;\n if (bbox1[1] <= bbox2[1])\n return false;\n if (bbox1[3] >= bbox2[3])\n return false;\n return true;\n}\nfunction getTileCoordinates(p, canonical) {\n const x = mercatorXfromLng(p[0]);\n const y = mercatorYfromLat(p[1]);\n const tilesAtZoom = Math.pow(2, canonical.z);\n return [Math.round(x * tilesAtZoom * EXTENT), Math.round(y * tilesAtZoom * EXTENT)];\n}\nfunction onBoundary(p, p1, p2) {\n const x1 = p[0] - p1[0];\n const y1 = p[1] - p1[1];\n const x2 = p[0] - p2[0];\n const y2 = p[1] - p2[1];\n return (x1 * y2 - x2 * y1 === 0) && (x1 * x2 <= 0) && (y1 * y2 <= 0);\n}\nfunction rayIntersect(p, p1, p2) {\n return ((p1[1] > p[1]) !== (p2[1] > p[1])) && (p[0] < (p2[0] - p1[0]) * (p[1] - p1[1]) / (p2[1] - p1[1]) + p1[0]);\n}\n// ray casting algorithm for detecting if point is in polygon\nfunction pointWithinPolygon(point, rings) {\n let inside = false;\n for (let i = 0, len = rings.length; i < len; i++) {\n const ring = rings[i];\n for (let j = 0, len2 = ring.length; j < len2 - 1; j++) {\n if (onBoundary(point, ring[j], ring[j + 1]))\n return false;\n if (rayIntersect(point, ring[j], ring[j + 1]))\n inside = !inside;\n }\n }\n return inside;\n}\nfunction pointWithinPolygons(point, polygons) {\n for (let i = 0; i < polygons.length; i++) {\n if (pointWithinPolygon(point, polygons[i]))\n return true;\n }\n return false;\n}\nfunction perp(v1, v2) {\n return (v1[0] * v2[1] - v1[1] * v2[0]);\n}\n// check if p1 and p2 are in different sides of line segment q1->q2\nfunction twoSided(p1, p2, q1, q2) {\n // q1->p1 (x1, y1), q1->p2 (x2, y2), q1->q2 (x3, y3)\n const x1 = p1[0] - q1[0];\n const y1 = p1[1] - q1[1];\n const x2 = p2[0] - q1[0];\n const y2 = p2[1] - q1[1];\n const x3 = q2[0] - q1[0];\n const y3 = q2[1] - q1[1];\n const det1 = (x1 * y3 - x3 * y1);\n const det2 = (x2 * y3 - x3 * y2);\n if ((det1 > 0 && det2 < 0) || (det1 < 0 && det2 > 0))\n return true;\n return false;\n}\n// a, b are end points for line segment1, c and d are end points for line segment2\nfunction lineIntersectLine(a, b, c, d) {\n // check if two segments are parallel or not\n // precondition is end point a, b is inside polygon, if line a->b is\n // parallel to polygon edge c->d, then a->b won't intersect with c->d\n const vectorP = [b[0] - a[0], b[1] - a[1]];\n const vectorQ = [d[0] - c[0], d[1] - c[1]];\n if (perp(vectorQ, vectorP) === 0)\n return false;\n // If lines are intersecting with each other, the relative location should be:\n // a and b lie in different sides of segment c->d\n // c and d lie in different sides of segment a->b\n if (twoSided(a, b, c, d) && twoSided(c, d, a, b))\n return true;\n return false;\n}\nfunction lineIntersectPolygon(p1, p2, polygon) {\n for (const ring of polygon) {\n // loop through every edge of the ring\n for (let j = 0; j < ring.length - 1; ++j) {\n if (lineIntersectLine(p1, p2, ring[j], ring[j + 1])) {\n return true;\n }\n }\n }\n return false;\n}\nfunction lineStringWithinPolygon(line, polygon) {\n // First, check if geometry points of line segments are all inside polygon\n for (let i = 0; i < line.length; ++i) {\n if (!pointWithinPolygon(line[i], polygon)) {\n return false;\n }\n }\n // Second, check if there is line segment intersecting polygon edge\n for (let i = 0; i < line.length - 1; ++i) {\n if (lineIntersectPolygon(line[i], line[i + 1], polygon)) {\n return false;\n }\n }\n return true;\n}\nfunction lineStringWithinPolygons(line, polygons) {\n for (let i = 0; i < polygons.length; i++) {\n if (lineStringWithinPolygon(line, polygons[i]))\n return true;\n }\n return false;\n}\nfunction getTilePolygon(coordinates, bbox, canonical) {\n const polygon = [];\n for (let i = 0; i < coordinates.length; i++) {\n const ring = [];\n for (let j = 0; j < coordinates[i].length; j++) {\n const coord = getTileCoordinates(coordinates[i][j], canonical);\n updateBBox(bbox, coord);\n ring.push(coord);\n }\n polygon.push(ring);\n }\n return polygon;\n}\nfunction getTilePolygons(coordinates, bbox, canonical) {\n const polygons = [];\n for (let i = 0; i < coordinates.length; i++) {\n const polygon = getTilePolygon(coordinates[i], bbox, canonical);\n polygons.push(polygon);\n }\n return polygons;\n}\nfunction updatePoint(p, bbox, polyBBox, worldSize) {\n if (p[0] < polyBBox[0] || p[0] > polyBBox[2]) {\n const halfWorldSize = worldSize * 0.5;\n let shift = (p[0] - polyBBox[0] > halfWorldSize) ? -worldSize : (polyBBox[0] - p[0] > halfWorldSize) ? worldSize : 0;\n if (shift === 0) {\n shift = (p[0] - polyBBox[2] > halfWorldSize) ? -worldSize : (polyBBox[2] - p[0] > halfWorldSize) ? worldSize : 0;\n }\n p[0] += shift;\n }\n updateBBox(bbox, p);\n}\nfunction resetBBox(bbox) {\n bbox[0] = bbox[1] = Infinity;\n bbox[2] = bbox[3] = -Infinity;\n}\nfunction getTilePoints(geometry, pointBBox, polyBBox, canonical) {\n const worldSize = Math.pow(2, canonical.z) * EXTENT;\n const shifts = [canonical.x * EXTENT, canonical.y * EXTENT];\n const tilePoints = [];\n for (const points of geometry) {\n for (const point of points) {\n const p = [point.x + shifts[0], point.y + shifts[1]];\n updatePoint(p, pointBBox, polyBBox, worldSize);\n tilePoints.push(p);\n }\n }\n return tilePoints;\n}\nfunction getTileLines(geometry, lineBBox, polyBBox, canonical) {\n const worldSize = Math.pow(2, canonical.z) * EXTENT;\n const shifts = [canonical.x * EXTENT, canonical.y * EXTENT];\n const tileLines = [];\n for (const line of geometry) {\n const tileLine = [];\n for (const point of line) {\n const p = [point.x + shifts[0], point.y + shifts[1]];\n updateBBox(lineBBox, p);\n tileLine.push(p);\n }\n tileLines.push(tileLine);\n }\n if (lineBBox[2] - lineBBox[0] <= worldSize / 2) {\n resetBBox(lineBBox);\n for (const line of tileLines) {\n for (const p of line) {\n updatePoint(p, lineBBox, polyBBox, worldSize);\n }\n }\n }\n return tileLines;\n}\nfunction pointsWithinPolygons(ctx, polygonGeometry) {\n const pointBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const polyBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const canonical = ctx.canonicalID();\n if (polygonGeometry.type === 'Polygon') {\n const tilePolygon = getTilePolygon(polygonGeometry.coordinates, polyBBox, canonical);\n const tilePoints = getTilePoints(ctx.geometry(), pointBBox, polyBBox, canonical);\n if (!boxWithinBox(pointBBox, polyBBox))\n return false;\n for (const point of tilePoints) {\n if (!pointWithinPolygon(point, tilePolygon))\n return false;\n }\n }\n if (polygonGeometry.type === 'MultiPolygon') {\n const tilePolygons = getTilePolygons(polygonGeometry.coordinates, polyBBox, canonical);\n const tilePoints = getTilePoints(ctx.geometry(), pointBBox, polyBBox, canonical);\n if (!boxWithinBox(pointBBox, polyBBox))\n return false;\n for (const point of tilePoints) {\n if (!pointWithinPolygons(point, tilePolygons))\n return false;\n }\n }\n return true;\n}\nfunction linesWithinPolygons(ctx, polygonGeometry) {\n const lineBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const polyBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const canonical = ctx.canonicalID();\n if (polygonGeometry.type === 'Polygon') {\n const tilePolygon = getTilePolygon(polygonGeometry.coordinates, polyBBox, canonical);\n const tileLines = getTileLines(ctx.geometry(), lineBBox, polyBBox, canonical);\n if (!boxWithinBox(lineBBox, polyBBox))\n return false;\n for (const line of tileLines) {\n if (!lineStringWithinPolygon(line, tilePolygon))\n return false;\n }\n }\n if (polygonGeometry.type === 'MultiPolygon') {\n const tilePolygons = getTilePolygons(polygonGeometry.coordinates, polyBBox, canonical);\n const tileLines = getTileLines(ctx.geometry(), lineBBox, polyBBox, canonical);\n if (!boxWithinBox(lineBBox, polyBBox))\n return false;\n for (const line of tileLines) {\n if (!lineStringWithinPolygons(line, tilePolygons))\n return false;\n }\n }\n return true;\n}\nclass Within {\n constructor(geojson, geometries) {\n this.type = BooleanType;\n this.geojson = geojson;\n this.geometries = geometries;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error(`'within' expression requires exactly one argument, but found ${args.length - 1} instead.`);\n if (isValue(args[1])) {\n const geojson = args[1];\n if (geojson.type === 'FeatureCollection') {\n for (let i = 0; i < geojson.features.length; ++i) {\n const type = geojson.features[i].geometry.type;\n if (type === 'Polygon' || type === 'MultiPolygon') {\n return new Within(geojson, geojson.features[i].geometry);\n }\n }\n }\n else if (geojson.type === 'Feature') {\n const type = geojson.geometry.type;\n if (type === 'Polygon' || type === 'MultiPolygon') {\n return new Within(geojson, geojson.geometry);\n }\n }\n else if (geojson.type === 'Polygon' || geojson.type === 'MultiPolygon') {\n return new Within(geojson, geojson);\n }\n }\n return context.error('\\'within\\' expression requires valid geojson object that contains polygon geometry type.');\n }\n evaluate(ctx) {\n if (ctx.geometry() != null && ctx.canonicalID() != null) {\n if (ctx.geometryType() === 'Point') {\n return pointsWithinPolygons(ctx, this.geometries);\n }\n else if (ctx.geometryType() === 'LineString') {\n return linesWithinPolygons(ctx, this.geometries);\n }\n }\n return false;\n }\n eachChild() { }\n outputDefined() {\n return true;\n }\n}\n\nclass Var {\n constructor(name, boundExpression) {\n this.type = boundExpression.type;\n this.name = name;\n this.boundExpression = boundExpression;\n }\n static parse(args, context) {\n if (args.length !== 2 || typeof args[1] !== 'string')\n return context.error('\\'var\\' expression requires exactly one string literal argument.');\n const name = args[1];\n if (!context.scope.has(name)) {\n return context.error(`Unknown variable \"${name}\". Make sure \"${name}\" has been bound in an enclosing \"let\" expression before using it.`, 1);\n }\n return new Var(name, context.scope.get(name));\n }\n evaluate(ctx) {\n return this.boundExpression.evaluate(ctx);\n }\n eachChild() { }\n outputDefined() {\n return false;\n }\n}\n\nclass CompoundExpression {\n constructor(name, type, evaluate, args) {\n this.name = name;\n this.type = type;\n this._evaluate = evaluate;\n this.args = args;\n }\n evaluate(ctx) {\n return this._evaluate(ctx, this.args);\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return false;\n }\n static parse(args, context) {\n const op = args[0];\n const definition = CompoundExpression.definitions[op];\n if (!definition) {\n return context.error(`Unknown expression \"${op}\". If you wanted a literal array, use [\"literal\", [...]].`, 0);\n }\n // Now check argument types against each signature\n const type = Array.isArray(definition) ?\n definition[0] : definition.type;\n const availableOverloads = Array.isArray(definition) ?\n [[definition[1], definition[2]]] :\n definition.overloads;\n const overloads = availableOverloads.filter(([signature]) => (!Array.isArray(signature) || // varags\n signature.length === args.length - 1 // correct param count\n ));\n let signatureContext = null;\n for (const [params, evaluate] of overloads) {\n // Use a fresh context for each attempted signature so that, if\n // we eventually succeed, we haven't polluted `context.errors`.\n signatureContext = new ParsingContext(context.registry, isExpressionConstant, context.path, null, context.scope);\n // First parse all the args, potentially coercing to the\n // types expected by this overload.\n const parsedArgs = [];\n let argParseFailed = false;\n for (let i = 1; i < args.length; i++) {\n const arg = args[i];\n const expectedType = Array.isArray(params) ?\n params[i - 1] :\n params.type;\n const parsed = signatureContext.parse(arg, 1 + parsedArgs.length, expectedType);\n if (!parsed) {\n argParseFailed = true;\n break;\n }\n parsedArgs.push(parsed);\n }\n if (argParseFailed) {\n // Couldn't coerce args of this overload to expected type, move\n // on to next one.\n continue;\n }\n if (Array.isArray(params)) {\n if (params.length !== parsedArgs.length) {\n signatureContext.error(`Expected ${params.length} arguments, but found ${parsedArgs.length} instead.`);\n continue;\n }\n }\n for (let i = 0; i < parsedArgs.length; i++) {\n const expected = Array.isArray(params) ? params[i] : params.type;\n const arg = parsedArgs[i];\n signatureContext.concat(i + 1).checkSubtype(expected, arg.type);\n }\n if (signatureContext.errors.length === 0) {\n return new CompoundExpression(op, type, evaluate, parsedArgs);\n }\n }\n if (overloads.length === 1) {\n context.errors.push(...signatureContext.errors);\n }\n else {\n const expected = overloads.length ? overloads : availableOverloads;\n const signatures = expected\n .map(([params]) => stringifySignature(params))\n .join(' | ');\n const actualTypes = [];\n // For error message, re-parse arguments without trying to\n // apply any coercions\n for (let i = 1; i < args.length; i++) {\n const parsed = context.parse(args[i], 1 + actualTypes.length);\n if (!parsed)\n return null;\n actualTypes.push(toString$1(parsed.type));\n }\n context.error(`Expected arguments of type ${signatures}, but found (${actualTypes.join(', ')}) instead.`);\n }\n return null;\n }\n static register(registry, definitions) {\n CompoundExpression.definitions = definitions;\n for (const name in definitions) {\n registry[name] = CompoundExpression;\n }\n }\n}\nfunction stringifySignature(signature) {\n if (Array.isArray(signature)) {\n return `(${signature.map(toString$1).join(', ')})`;\n }\n else {\n return `(${toString$1(signature.type)}...)`;\n }\n}\nfunction isExpressionConstant(expression) {\n if (expression instanceof Var) {\n return isExpressionConstant(expression.boundExpression);\n }\n else if (expression instanceof CompoundExpression && expression.name === 'error') {\n return false;\n }\n else if (expression instanceof CollatorExpression) {\n // Although the results of a Collator expression with fixed arguments\n // generally shouldn't change between executions, we can't serialize them\n // as constant expressions because results change based on environment.\n return false;\n }\n else if (expression instanceof Within) {\n return false;\n }\n const isTypeAnnotation = expression instanceof Coercion ||\n expression instanceof Assertion;\n let childrenConstant = true;\n expression.eachChild(child => {\n // We can _almost_ assume that if `expressions` children are constant,\n // they would already have been evaluated to Literal values when they\n // were parsed. Type annotations are the exception, because they might\n // have been inferred and added after a child was parsed.\n // So we recurse into isConstant() for the children of type annotations,\n // but otherwise simply check whether they are Literals.\n if (isTypeAnnotation) {\n childrenConstant = childrenConstant && isExpressionConstant(child);\n }\n else {\n childrenConstant = childrenConstant && child instanceof Literal;\n }\n });\n if (!childrenConstant) {\n return false;\n }\n return isFeatureConstant(expression) &&\n isGlobalPropertyConstant(expression, ['zoom', 'heatmap-density', 'line-progress', 'accumulated', 'is-supported-script']);\n}\nfunction isFeatureConstant(e) {\n if (e instanceof CompoundExpression) {\n if (e.name === 'get' && e.args.length === 1) {\n return false;\n }\n else if (e.name === 'feature-state') {\n return false;\n }\n else if (e.name === 'has' && e.args.length === 1) {\n return false;\n }\n else if (e.name === 'properties' ||\n e.name === 'geometry-type' ||\n e.name === 'id') {\n return false;\n }\n else if (/^filter-/.test(e.name)) {\n return false;\n }\n }\n if (e instanceof Within) {\n return false;\n }\n let result = true;\n e.eachChild(arg => {\n if (result && !isFeatureConstant(arg)) {\n result = false;\n }\n });\n return result;\n}\nfunction isStateConstant(e) {\n if (e instanceof CompoundExpression) {\n if (e.name === 'feature-state') {\n return false;\n }\n }\n let result = true;\n e.eachChild(arg => {\n if (result && !isStateConstant(arg)) {\n result = false;\n }\n });\n return result;\n}\nfunction isGlobalPropertyConstant(e, properties) {\n if (e instanceof CompoundExpression && properties.indexOf(e.name) >= 0) {\n return false;\n }\n let result = true;\n e.eachChild((arg) => {\n if (result && !isGlobalPropertyConstant(arg, properties)) {\n result = false;\n }\n });\n return result;\n}\n\n/**\n * Returns the index of the last stop <= input, or 0 if it doesn't exist.\n * @private\n */\nfunction findStopLessThanOrEqualTo(stops, input) {\n const lastIndex = stops.length - 1;\n let lowerIndex = 0;\n let upperIndex = lastIndex;\n let currentIndex = 0;\n let currentValue, nextValue;\n while (lowerIndex <= upperIndex) {\n currentIndex = Math.floor((lowerIndex + upperIndex) / 2);\n currentValue = stops[currentIndex];\n nextValue = stops[currentIndex + 1];\n if (currentValue <= input) {\n if (currentIndex === lastIndex || input < nextValue) { // Search complete\n return currentIndex;\n }\n lowerIndex = currentIndex + 1;\n }\n else if (currentValue > input) {\n upperIndex = currentIndex - 1;\n }\n else {\n throw new RuntimeError('Input is not a number.');\n }\n }\n return 0;\n}\n\nclass Step {\n constructor(type, input, stops) {\n this.type = type;\n this.input = input;\n this.labels = [];\n this.outputs = [];\n for (const [label, expression] of stops) {\n this.labels.push(label);\n this.outputs.push(expression);\n }\n }\n static parse(args, context) {\n if (args.length - 1 < 4) {\n return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`);\n }\n if ((args.length - 1) % 2 !== 0) {\n return context.error('Expected an even number of arguments.');\n }\n const input = context.parse(args[1], 1, NumberType);\n if (!input)\n return null;\n const stops = [];\n let outputType = null;\n if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n for (let i = 1; i < args.length; i += 2) {\n const label = i === 1 ? -Infinity : args[i];\n const value = args[i + 1];\n const labelKey = i;\n const valueKey = i + 1;\n if (typeof label !== 'number') {\n return context.error('Input/output pairs for \"step\" expressions must be defined using literal numeric values (not computed expressions) for the input values.', labelKey);\n }\n if (stops.length && stops[stops.length - 1][0] >= label) {\n return context.error('Input/output pairs for \"step\" expressions must be arranged with input values in strictly ascending order.', labelKey);\n }\n const parsed = context.parse(value, valueKey, outputType);\n if (!parsed)\n return null;\n outputType = outputType || parsed.type;\n stops.push([label, parsed]);\n }\n return new Step(outputType, input, stops);\n }\n evaluate(ctx) {\n const labels = this.labels;\n const outputs = this.outputs;\n if (labels.length === 1) {\n return outputs[0].evaluate(ctx);\n }\n const value = this.input.evaluate(ctx);\n if (value <= labels[0]) {\n return outputs[0].evaluate(ctx);\n }\n const stopCount = labels.length;\n if (value >= labels[stopCount - 1]) {\n return outputs[stopCount - 1].evaluate(ctx);\n }\n const index = findStopLessThanOrEqualTo(labels, value);\n return outputs[index].evaluate(ctx);\n }\n eachChild(fn) {\n fn(this.input);\n for (const expression of this.outputs) {\n fn(expression);\n }\n }\n outputDefined() {\n return this.outputs.every(out => out.outputDefined());\n }\n}\n\n/**\n * Checks whether the specified color space is one of the supported interpolation color spaces.\n *\n * @param colorSpace Color space key to verify.\n * @returns `true` if the specified color space is one of the supported\n * interpolation color spaces, `false` otherwise\n */\nfunction isSupportedInterpolationColorSpace(colorSpace) {\n return colorSpace === 'rgb' || colorSpace === 'hcl' || colorSpace === 'lab';\n}\n/**\n * @param interpolationType Interpolation type\n * @returns interpolation fn\n * @deprecated use `interpolate[type]` instead\n */\nconst interpolateFactory = (interpolationType) => {\n switch (interpolationType) {\n case 'number': return number;\n case 'color': return color;\n case 'array': return array;\n case 'padding': return padding;\n case 'variableAnchorOffsetCollection': return variableAnchorOffsetCollection;\n }\n};\nfunction number(from, to, t) {\n return from + t * (to - from);\n}\nfunction color(from, to, t, spaceKey = 'rgb') {\n switch (spaceKey) {\n case 'rgb': {\n const [r, g, b, alpha] = array(from.rgb, to.rgb, t);\n return new Color(r, g, b, alpha, false);\n }\n case 'hcl': {\n const [hue0, chroma0, light0, alphaF] = from.hcl;\n const [hue1, chroma1, light1, alphaT] = to.hcl;\n // https://github.com/gka/chroma.js/blob/cd1b3c0926c7a85cbdc3b1453b3a94006de91a92/src/interpolator/_hsx.js\n let hue, chroma;\n if (!isNaN(hue0) && !isNaN(hue1)) {\n let dh = hue1 - hue0;\n if (hue1 > hue0 && dh > 180) {\n dh -= 360;\n }\n else if (hue1 < hue0 && hue0 - hue1 > 180) {\n dh += 360;\n }\n hue = hue0 + t * dh;\n }\n else if (!isNaN(hue0)) {\n hue = hue0;\n if (light1 === 1 || light1 === 0)\n chroma = chroma0;\n }\n else if (!isNaN(hue1)) {\n hue = hue1;\n if (light0 === 1 || light0 === 0)\n chroma = chroma1;\n }\n else {\n hue = NaN;\n }\n const [r, g, b, alpha] = hclToRgb([\n hue,\n chroma !== null && chroma !== void 0 ? chroma : number(chroma0, chroma1, t),\n number(light0, light1, t),\n number(alphaF, alphaT, t),\n ]);\n return new Color(r, g, b, alpha, false);\n }\n case 'lab': {\n const [r, g, b, alpha] = labToRgb(array(from.lab, to.lab, t));\n return new Color(r, g, b, alpha, false);\n }\n }\n}\nfunction array(from, to, t) {\n return from.map((d, i) => {\n return number(d, to[i], t);\n });\n}\nfunction padding(from, to, t) {\n return new Padding(array(from.values, to.values, t));\n}\nfunction variableAnchorOffsetCollection(from, to, t) {\n const fromValues = from.values;\n const toValues = to.values;\n if (fromValues.length !== toValues.length) {\n throw new RuntimeError(`Cannot interpolate values of different length. from: ${from.toString()}, to: ${to.toString()}`);\n }\n const output = [];\n for (let i = 0; i < fromValues.length; i += 2) {\n // Anchor entries must match\n if (fromValues[i] !== toValues[i]) {\n throw new RuntimeError(`Cannot interpolate values containing mismatched anchors. from[${i}]: ${fromValues[i]}, to[${i}]: ${toValues[i]}`);\n }\n output.push(fromValues[i]);\n // Interpolate the offset values for each anchor\n const [fx, fy] = fromValues[i + 1];\n const [tx, ty] = toValues[i + 1];\n output.push([number(fx, tx, t), number(fy, ty, t)]);\n }\n return new VariableAnchorOffsetCollection(output);\n}\nconst interpolate = {\n number,\n color,\n array,\n padding,\n variableAnchorOffsetCollection\n};\n\nclass Interpolate {\n constructor(type, operator, interpolation, input, stops) {\n this.type = type;\n this.operator = operator;\n this.interpolation = interpolation;\n this.input = input;\n this.labels = [];\n this.outputs = [];\n for (const [label, expression] of stops) {\n this.labels.push(label);\n this.outputs.push(expression);\n }\n }\n static interpolationFactor(interpolation, input, lower, upper) {\n let t = 0;\n if (interpolation.name === 'exponential') {\n t = exponentialInterpolation(input, interpolation.base, lower, upper);\n }\n else if (interpolation.name === 'linear') {\n t = exponentialInterpolation(input, 1, lower, upper);\n }\n else if (interpolation.name === 'cubic-bezier') {\n const c = interpolation.controlPoints;\n const ub = new UnitBezier(c[0], c[1], c[2], c[3]);\n t = ub.solve(exponentialInterpolation(input, 1, lower, upper));\n }\n return t;\n }\n static parse(args, context) {\n let [operator, interpolation, input, ...rest] = args;\n if (!Array.isArray(interpolation) || interpolation.length === 0) {\n return context.error('Expected an interpolation type expression.', 1);\n }\n if (interpolation[0] === 'linear') {\n interpolation = { name: 'linear' };\n }\n else if (interpolation[0] === 'exponential') {\n const base = interpolation[1];\n if (typeof base !== 'number')\n return context.error('Exponential interpolation requires a numeric base.', 1, 1);\n interpolation = {\n name: 'exponential',\n base\n };\n }\n else if (interpolation[0] === 'cubic-bezier') {\n const controlPoints = interpolation.slice(1);\n if (controlPoints.length !== 4 ||\n controlPoints.some(t => typeof t !== 'number' || t < 0 || t > 1)) {\n return context.error('Cubic bezier interpolation requires four numeric arguments with values between 0 and 1.', 1);\n }\n interpolation = {\n name: 'cubic-bezier',\n controlPoints: controlPoints\n };\n }\n else {\n return context.error(`Unknown interpolation type ${String(interpolation[0])}`, 1, 0);\n }\n if (args.length - 1 < 4) {\n return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`);\n }\n if ((args.length - 1) % 2 !== 0) {\n return context.error('Expected an even number of arguments.');\n }\n input = context.parse(input, 2, NumberType);\n if (!input)\n return null;\n const stops = [];\n let outputType = null;\n if (operator === 'interpolate-hcl' || operator === 'interpolate-lab') {\n outputType = ColorType;\n }\n else if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n for (let i = 0; i < rest.length; i += 2) {\n const label = rest[i];\n const value = rest[i + 1];\n const labelKey = i + 3;\n const valueKey = i + 4;\n if (typeof label !== 'number') {\n return context.error('Input/output pairs for \"interpolate\" expressions must be defined using literal numeric values (not computed expressions) for the input values.', labelKey);\n }\n if (stops.length && stops[stops.length - 1][0] >= label) {\n return context.error('Input/output pairs for \"interpolate\" expressions must be arranged with input values in strictly ascending order.', labelKey);\n }\n const parsed = context.parse(value, valueKey, outputType);\n if (!parsed)\n return null;\n outputType = outputType || parsed.type;\n stops.push([label, parsed]);\n }\n if (!verifyType(outputType, NumberType) &&\n !verifyType(outputType, ColorType) &&\n !verifyType(outputType, PaddingType) &&\n !verifyType(outputType, VariableAnchorOffsetCollectionType) &&\n !verifyType(outputType, array$1(NumberType))) {\n return context.error(`Type ${toString$1(outputType)} is not interpolatable.`);\n }\n return new Interpolate(outputType, operator, interpolation, input, stops);\n }\n evaluate(ctx) {\n const labels = this.labels;\n const outputs = this.outputs;\n if (labels.length === 1) {\n return outputs[0].evaluate(ctx);\n }\n const value = this.input.evaluate(ctx);\n if (value <= labels[0]) {\n return outputs[0].evaluate(ctx);\n }\n const stopCount = labels.length;\n if (value >= labels[stopCount - 1]) {\n return outputs[stopCount - 1].evaluate(ctx);\n }\n const index = findStopLessThanOrEqualTo(labels, value);\n const lower = labels[index];\n const upper = labels[index + 1];\n const t = Interpolate.interpolationFactor(this.interpolation, value, lower, upper);\n const outputLower = outputs[index].evaluate(ctx);\n const outputUpper = outputs[index + 1].evaluate(ctx);\n switch (this.operator) {\n case 'interpolate':\n return interpolate[this.type.kind](outputLower, outputUpper, t);\n case 'interpolate-hcl':\n return interpolate.color(outputLower, outputUpper, t, 'hcl');\n case 'interpolate-lab':\n return interpolate.color(outputLower, outputUpper, t, 'lab');\n }\n }\n eachChild(fn) {\n fn(this.input);\n for (const expression of this.outputs) {\n fn(expression);\n }\n }\n outputDefined() {\n return this.outputs.every(out => out.outputDefined());\n }\n}\n/**\n * Returns a ratio that can be used to interpolate between exponential function\n * stops.\n * How it works: Two consecutive stop values define a (scaled and shifted) exponential function `f(x) = a * base^x + b`, where `base` is the user-specified base,\n * and `a` and `b` are constants affording sufficient degrees of freedom to fit\n * the function to the given stops.\n *\n * Here's a bit of algebra that lets us compute `f(x)` directly from the stop\n * values without explicitly solving for `a` and `b`:\n *\n * First stop value: `f(x0) = y0 = a * base^x0 + b`\n * Second stop value: `f(x1) = y1 = a * base^x1 + b`\n * => `y1 - y0 = a(base^x1 - base^x0)`\n * => `a = (y1 - y0)/(base^x1 - base^x0)`\n *\n * Desired value: `f(x) = y = a * base^x + b`\n * => `f(x) = y0 + a * (base^x - base^x0)`\n *\n * From the above, we can replace the `a` in `a * (base^x - base^x0)` and do a\n * little algebra:\n * ```\n * a * (base^x - base^x0) = (y1 - y0)/(base^x1 - base^x0) * (base^x - base^x0)\n * = (y1 - y0) * (base^x - base^x0) / (base^x1 - base^x0)\n * ```\n *\n * If we let `(base^x - base^x0) / (base^x1 base^x0)`, then we have\n * `f(x) = y0 + (y1 - y0) * ratio`. In other words, `ratio` may be treated as\n * an interpolation factor between the two stops' output values.\n *\n * (Note: a slightly different form for `ratio`,\n * `(base^(x-x0) - 1) / (base^(x1-x0) - 1) `, is equivalent, but requires fewer\n * expensive `Math.pow()` operations.)\n *\n * @private\n*/\nfunction exponentialInterpolation(input, base, lowerValue, upperValue) {\n const difference = upperValue - lowerValue;\n const progress = input - lowerValue;\n if (difference === 0) {\n return 0;\n }\n else if (base === 1) {\n return progress / difference;\n }\n else {\n return (Math.pow(base, progress) - 1) / (Math.pow(base, difference) - 1);\n }\n}\n\nclass Coalesce {\n constructor(type, args) {\n this.type = type;\n this.args = args;\n }\n static parse(args, context) {\n if (args.length < 2) {\n return context.error('Expectected at least one argument.');\n }\n let outputType = null;\n const expectedType = context.expectedType;\n if (expectedType && expectedType.kind !== 'value') {\n outputType = expectedType;\n }\n const parsedArgs = [];\n for (const arg of args.slice(1)) {\n const parsed = context.parse(arg, 1 + parsedArgs.length, outputType, undefined, { typeAnnotation: 'omit' });\n if (!parsed)\n return null;\n outputType = outputType || parsed.type;\n parsedArgs.push(parsed);\n }\n if (!outputType)\n throw new Error('No output type');\n // Above, we parse arguments without inferred type annotation so that\n // they don't produce a runtime error for `null` input, which would\n // preempt the desired null-coalescing behavior.\n // Thus, if any of our arguments would have needed an annotation, we\n // need to wrap the enclosing coalesce expression with it instead.\n const needsAnnotation = expectedType &&\n parsedArgs.some(arg => checkSubtype(expectedType, arg.type));\n return needsAnnotation ?\n new Coalesce(ValueType, parsedArgs) :\n new Coalesce(outputType, parsedArgs);\n }\n evaluate(ctx) {\n let result = null;\n let argCount = 0;\n let requestedImageName;\n for (const arg of this.args) {\n argCount++;\n result = arg.evaluate(ctx);\n // we need to keep track of the first requested image in a coalesce statement\n // if coalesce can't find a valid image, we return the first image name so styleimagemissing can fire\n if (result && result instanceof ResolvedImage && !result.available) {\n if (!requestedImageName) {\n requestedImageName = result.name;\n }\n result = null;\n if (argCount === this.args.length) {\n result = requestedImageName;\n }\n }\n if (result !== null)\n break;\n }\n return result;\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return this.args.every(arg => arg.outputDefined());\n }\n}\n\nclass Let {\n constructor(bindings, result) {\n this.type = result.type;\n this.bindings = [].concat(bindings);\n this.result = result;\n }\n evaluate(ctx) {\n return this.result.evaluate(ctx);\n }\n eachChild(fn) {\n for (const binding of this.bindings) {\n fn(binding[1]);\n }\n fn(this.result);\n }\n static parse(args, context) {\n if (args.length < 4)\n return context.error(`Expected at least 3 arguments, but found ${args.length - 1} instead.`);\n const bindings = [];\n for (let i = 1; i < args.length - 1; i += 2) {\n const name = args[i];\n if (typeof name !== 'string') {\n return context.error(`Expected string, but found ${typeof name} instead.`, i);\n }\n if (/[^a-zA-Z0-9_]/.test(name)) {\n return context.error('Variable names must contain only alphanumeric characters or \\'_\\'.', i);\n }\n const value = context.parse(args[i + 1], i + 1);\n if (!value)\n return null;\n bindings.push([name, value]);\n }\n const result = context.parse(args[args.length - 1], args.length - 1, context.expectedType, bindings);\n if (!result)\n return null;\n return new Let(bindings, result);\n }\n outputDefined() {\n return this.result.outputDefined();\n }\n}\n\nclass At {\n constructor(type, index, input) {\n this.type = type;\n this.index = index;\n this.input = input;\n }\n static parse(args, context) {\n if (args.length !== 3)\n return context.error(`Expected 2 arguments, but found ${args.length - 1} instead.`);\n const index = context.parse(args[1], 1, NumberType);\n const input = context.parse(args[2], 2, array$1(context.expectedType || ValueType));\n if (!index || !input)\n return null;\n const t = input.type;\n return new At(t.itemType, index, input);\n }\n evaluate(ctx) {\n const index = this.index.evaluate(ctx);\n const array = this.input.evaluate(ctx);\n if (index < 0) {\n throw new RuntimeError(`Array index out of bounds: ${index} < 0.`);\n }\n if (index >= array.length) {\n throw new RuntimeError(`Array index out of bounds: ${index} > ${array.length - 1}.`);\n }\n if (index !== Math.floor(index)) {\n throw new RuntimeError(`Array index must be an integer, but found ${index} instead.`);\n }\n return array[index];\n }\n eachChild(fn) {\n fn(this.index);\n fn(this.input);\n }\n outputDefined() {\n return false;\n }\n}\n\nclass In {\n constructor(needle, haystack) {\n this.type = BooleanType;\n this.needle = needle;\n this.haystack = haystack;\n }\n static parse(args, context) {\n if (args.length !== 3) {\n return context.error(`Expected 2 arguments, but found ${args.length - 1} instead.`);\n }\n const needle = context.parse(args[1], 1, ValueType);\n const haystack = context.parse(args[2], 2, ValueType);\n if (!needle || !haystack)\n return null;\n if (!isValidType(needle.type, [BooleanType, StringType, NumberType, NullType, ValueType])) {\n return context.error(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(needle.type)} instead`);\n }\n return new In(needle, haystack);\n }\n evaluate(ctx) {\n const needle = this.needle.evaluate(ctx);\n const haystack = this.haystack.evaluate(ctx);\n if (!haystack)\n return false;\n if (!isValidNativeType(needle, ['boolean', 'string', 'number', 'null'])) {\n throw new RuntimeError(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(typeOf(needle))} instead.`);\n }\n if (!isValidNativeType(haystack, ['string', 'array'])) {\n throw new RuntimeError(`Expected second argument to be of type array or string, but found ${toString$1(typeOf(haystack))} instead.`);\n }\n return haystack.indexOf(needle) >= 0;\n }\n eachChild(fn) {\n fn(this.needle);\n fn(this.haystack);\n }\n outputDefined() {\n return true;\n }\n}\n\nclass IndexOf {\n constructor(needle, haystack, fromIndex) {\n this.type = NumberType;\n this.needle = needle;\n this.haystack = haystack;\n this.fromIndex = fromIndex;\n }\n static parse(args, context) {\n if (args.length <= 2 || args.length >= 5) {\n return context.error(`Expected 3 or 4 arguments, but found ${args.length - 1} instead.`);\n }\n const needle = context.parse(args[1], 1, ValueType);\n const haystack = context.parse(args[2], 2, ValueType);\n if (!needle || !haystack)\n return null;\n if (!isValidType(needle.type, [BooleanType, StringType, NumberType, NullType, ValueType])) {\n return context.error(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(needle.type)} instead`);\n }\n if (args.length === 4) {\n const fromIndex = context.parse(args[3], 3, NumberType);\n if (!fromIndex)\n return null;\n return new IndexOf(needle, haystack, fromIndex);\n }\n else {\n return new IndexOf(needle, haystack);\n }\n }\n evaluate(ctx) {\n const needle = this.needle.evaluate(ctx);\n const haystack = this.haystack.evaluate(ctx);\n if (!isValidNativeType(needle, ['boolean', 'string', 'number', 'null'])) {\n throw new RuntimeError(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(typeOf(needle))} instead.`);\n }\n if (!isValidNativeType(haystack, ['string', 'array'])) {\n throw new RuntimeError(`Expected second argument to be of type array or string, but found ${toString$1(typeOf(haystack))} instead.`);\n }\n if (this.fromIndex) {\n const fromIndex = this.fromIndex.evaluate(ctx);\n return haystack.indexOf(needle, fromIndex);\n }\n return haystack.indexOf(needle);\n }\n eachChild(fn) {\n fn(this.needle);\n fn(this.haystack);\n if (this.fromIndex) {\n fn(this.fromIndex);\n }\n }\n outputDefined() {\n return false;\n }\n}\n\nclass Match {\n constructor(inputType, outputType, input, cases, outputs, otherwise) {\n this.inputType = inputType;\n this.type = outputType;\n this.input = input;\n this.cases = cases;\n this.outputs = outputs;\n this.otherwise = otherwise;\n }\n static parse(args, context) {\n if (args.length < 5)\n return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`);\n if (args.length % 2 !== 1)\n return context.error('Expected an even number of arguments.');\n let inputType;\n let outputType;\n if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n const cases = {};\n const outputs = [];\n for (let i = 2; i < args.length - 1; i += 2) {\n let labels = args[i];\n const value = args[i + 1];\n if (!Array.isArray(labels)) {\n labels = [labels];\n }\n const labelContext = context.concat(i);\n if (labels.length === 0) {\n return labelContext.error('Expected at least one branch label.');\n }\n for (const label of labels) {\n if (typeof label !== 'number' && typeof label !== 'string') {\n return labelContext.error('Branch labels must be numbers or strings.');\n }\n else if (typeof label === 'number' && Math.abs(label) > Number.MAX_SAFE_INTEGER) {\n return labelContext.error(`Branch labels must be integers no larger than ${Number.MAX_SAFE_INTEGER}.`);\n }\n else if (typeof label === 'number' && Math.floor(label) !== label) {\n return labelContext.error('Numeric branch labels must be integer values.');\n }\n else if (!inputType) {\n inputType = typeOf(label);\n }\n else if (labelContext.checkSubtype(inputType, typeOf(label))) {\n return null;\n }\n if (typeof cases[String(label)] !== 'undefined') {\n return labelContext.error('Branch labels must be unique.');\n }\n cases[String(label)] = outputs.length;\n }\n const result = context.parse(value, i, outputType);\n if (!result)\n return null;\n outputType = outputType || result.type;\n outputs.push(result);\n }\n const input = context.parse(args[1], 1, ValueType);\n if (!input)\n return null;\n const otherwise = context.parse(args[args.length - 1], args.length - 1, outputType);\n if (!otherwise)\n return null;\n if (input.type.kind !== 'value' && context.concat(1).checkSubtype(inputType, input.type)) {\n return null;\n }\n return new Match(inputType, outputType, input, cases, outputs, otherwise);\n }\n evaluate(ctx) {\n const input = this.input.evaluate(ctx);\n const output = (typeOf(input) === this.inputType && this.outputs[this.cases[input]]) || this.otherwise;\n return output.evaluate(ctx);\n }\n eachChild(fn) {\n fn(this.input);\n this.outputs.forEach(fn);\n fn(this.otherwise);\n }\n outputDefined() {\n return this.outputs.every(out => out.outputDefined()) && this.otherwise.outputDefined();\n }\n}\n\nclass Case {\n constructor(type, branches, otherwise) {\n this.type = type;\n this.branches = branches;\n this.otherwise = otherwise;\n }\n static parse(args, context) {\n if (args.length < 4)\n return context.error(`Expected at least 3 arguments, but found only ${args.length - 1}.`);\n if (args.length % 2 !== 0)\n return context.error('Expected an odd number of arguments.');\n let outputType;\n if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n const branches = [];\n for (let i = 1; i < args.length - 1; i += 2) {\n const test = context.parse(args[i], i, BooleanType);\n if (!test)\n return null;\n const result = context.parse(args[i + 1], i + 1, outputType);\n if (!result)\n return null;\n branches.push([test, result]);\n outputType = outputType || result.type;\n }\n const otherwise = context.parse(args[args.length - 1], args.length - 1, outputType);\n if (!otherwise)\n return null;\n if (!outputType)\n throw new Error('Can\\'t infer output type');\n return new Case(outputType, branches, otherwise);\n }\n evaluate(ctx) {\n for (const [test, expression] of this.branches) {\n if (test.evaluate(ctx)) {\n return expression.evaluate(ctx);\n }\n }\n return this.otherwise.evaluate(ctx);\n }\n eachChild(fn) {\n for (const [test, expression] of this.branches) {\n fn(test);\n fn(expression);\n }\n fn(this.otherwise);\n }\n outputDefined() {\n return this.branches.every(([_, out]) => out.outputDefined()) && this.otherwise.outputDefined();\n }\n}\n\nclass Slice {\n constructor(type, input, beginIndex, endIndex) {\n this.type = type;\n this.input = input;\n this.beginIndex = beginIndex;\n this.endIndex = endIndex;\n }\n static parse(args, context) {\n if (args.length <= 2 || args.length >= 5) {\n return context.error(`Expected 3 or 4 arguments, but found ${args.length - 1} instead.`);\n }\n const input = context.parse(args[1], 1, ValueType);\n const beginIndex = context.parse(args[2], 2, NumberType);\n if (!input || !beginIndex)\n return null;\n if (!isValidType(input.type, [array$1(ValueType), StringType, ValueType])) {\n return context.error(`Expected first argument to be of type array or string, but found ${toString$1(input.type)} instead`);\n }\n if (args.length === 4) {\n const endIndex = context.parse(args[3], 3, NumberType);\n if (!endIndex)\n return null;\n return new Slice(input.type, input, beginIndex, endIndex);\n }\n else {\n return new Slice(input.type, input, beginIndex);\n }\n }\n evaluate(ctx) {\n const input = this.input.evaluate(ctx);\n const beginIndex = this.beginIndex.evaluate(ctx);\n if (!isValidNativeType(input, ['string', 'array'])) {\n throw new RuntimeError(`Expected first argument to be of type array or string, but found ${toString$1(typeOf(input))} instead.`);\n }\n if (this.endIndex) {\n const endIndex = this.endIndex.evaluate(ctx);\n return input.slice(beginIndex, endIndex);\n }\n return input.slice(beginIndex);\n }\n eachChild(fn) {\n fn(this.input);\n fn(this.beginIndex);\n if (this.endIndex) {\n fn(this.endIndex);\n }\n }\n outputDefined() {\n return false;\n }\n}\n\nfunction isComparableType(op, type) {\n if (op === '==' || op === '!=') {\n // equality operator\n return type.kind === 'boolean' ||\n type.kind === 'string' ||\n type.kind === 'number' ||\n type.kind === 'null' ||\n type.kind === 'value';\n }\n else {\n // ordering operator\n return type.kind === 'string' ||\n type.kind === 'number' ||\n type.kind === 'value';\n }\n}\nfunction eq(ctx, a, b) { return a === b; }\nfunction neq(ctx, a, b) { return a !== b; }\nfunction lt(ctx, a, b) { return a < b; }\nfunction gt(ctx, a, b) { return a > b; }\nfunction lteq(ctx, a, b) { return a <= b; }\nfunction gteq(ctx, a, b) { return a >= b; }\nfunction eqCollate(ctx, a, b, c) { return c.compare(a, b) === 0; }\nfunction neqCollate(ctx, a, b, c) { return !eqCollate(ctx, a, b, c); }\nfunction ltCollate(ctx, a, b, c) { return c.compare(a, b) < 0; }\nfunction gtCollate(ctx, a, b, c) { return c.compare(a, b) > 0; }\nfunction lteqCollate(ctx, a, b, c) { return c.compare(a, b) <= 0; }\nfunction gteqCollate(ctx, a, b, c) { return c.compare(a, b) >= 0; }\n/**\n * Special form for comparison operators, implementing the signatures:\n * - (T, T, ?Collator) => boolean\n * - (T, value, ?Collator) => boolean\n * - (value, T, ?Collator) => boolean\n *\n * For inequalities, T must be either value, string, or number. For ==/!=, it\n * can also be boolean or null.\n *\n * Equality semantics are equivalent to Javascript's strict equality (===/!==)\n * -- i.e., when the arguments' types don't match, == evaluates to false, != to\n * true.\n *\n * When types don't match in an ordering comparison, a runtime error is thrown.\n *\n * @private\n */\nfunction makeComparison(op, compareBasic, compareWithCollator) {\n const isOrderComparison = op !== '==' && op !== '!=';\n return class Comparison {\n constructor(lhs, rhs, collator) {\n this.type = BooleanType;\n this.lhs = lhs;\n this.rhs = rhs;\n this.collator = collator;\n this.hasUntypedArgument = lhs.type.kind === 'value' || rhs.type.kind === 'value';\n }\n static parse(args, context) {\n if (args.length !== 3 && args.length !== 4)\n return context.error('Expected two or three arguments.');\n const op = args[0];\n let lhs = context.parse(args[1], 1, ValueType);\n if (!lhs)\n return null;\n if (!isComparableType(op, lhs.type)) {\n return context.concat(1).error(`\"${op}\" comparisons are not supported for type '${toString$1(lhs.type)}'.`);\n }\n let rhs = context.parse(args[2], 2, ValueType);\n if (!rhs)\n return null;\n if (!isComparableType(op, rhs.type)) {\n return context.concat(2).error(`\"${op}\" comparisons are not supported for type '${toString$1(rhs.type)}'.`);\n }\n if (lhs.type.kind !== rhs.type.kind &&\n lhs.type.kind !== 'value' &&\n rhs.type.kind !== 'value') {\n return context.error(`Cannot compare types '${toString$1(lhs.type)}' and '${toString$1(rhs.type)}'.`);\n }\n if (isOrderComparison) {\n // typing rules specific to less/greater than operators\n if (lhs.type.kind === 'value' && rhs.type.kind !== 'value') {\n // (value, T)\n lhs = new Assertion(rhs.type, [lhs]);\n }\n else if (lhs.type.kind !== 'value' && rhs.type.kind === 'value') {\n // (T, value)\n rhs = new Assertion(lhs.type, [rhs]);\n }\n }\n let collator = null;\n if (args.length === 4) {\n if (lhs.type.kind !== 'string' &&\n rhs.type.kind !== 'string' &&\n lhs.type.kind !== 'value' &&\n rhs.type.kind !== 'value') {\n return context.error('Cannot use collator to compare non-string types.');\n }\n collator = context.parse(args[3], 3, CollatorType);\n if (!collator)\n return null;\n }\n return new Comparison(lhs, rhs, collator);\n }\n evaluate(ctx) {\n const lhs = this.lhs.evaluate(ctx);\n const rhs = this.rhs.evaluate(ctx);\n if (isOrderComparison && this.hasUntypedArgument) {\n const lt = typeOf(lhs);\n const rt = typeOf(rhs);\n // check that type is string or number, and equal\n if (lt.kind !== rt.kind || !(lt.kind === 'string' || lt.kind === 'number')) {\n throw new RuntimeError(`Expected arguments for \"${op}\" to be (string, string) or (number, number), but found (${lt.kind}, ${rt.kind}) instead.`);\n }\n }\n if (this.collator && !isOrderComparison && this.hasUntypedArgument) {\n const lt = typeOf(lhs);\n const rt = typeOf(rhs);\n if (lt.kind !== 'string' || rt.kind !== 'string') {\n return compareBasic(ctx, lhs, rhs);\n }\n }\n return this.collator ?\n compareWithCollator(ctx, lhs, rhs, this.collator.evaluate(ctx)) :\n compareBasic(ctx, lhs, rhs);\n }\n eachChild(fn) {\n fn(this.lhs);\n fn(this.rhs);\n if (this.collator) {\n fn(this.collator);\n }\n }\n outputDefined() {\n return true;\n }\n };\n}\nconst Equals = makeComparison('==', eq, eqCollate);\nconst NotEquals = makeComparison('!=', neq, neqCollate);\nconst LessThan = makeComparison('<', lt, ltCollate);\nconst GreaterThan = makeComparison('>', gt, gtCollate);\nconst LessThanOrEqual = makeComparison('<=', lteq, lteqCollate);\nconst GreaterThanOrEqual = makeComparison('>=', gteq, gteqCollate);\n\nclass NumberFormat {\n constructor(number, locale, currency, minFractionDigits, maxFractionDigits) {\n this.type = StringType;\n this.number = number;\n this.locale = locale;\n this.currency = currency;\n this.minFractionDigits = minFractionDigits;\n this.maxFractionDigits = maxFractionDigits;\n }\n static parse(args, context) {\n if (args.length !== 3)\n return context.error('Expected two arguments.');\n const number = context.parse(args[1], 1, NumberType);\n if (!number)\n return null;\n const options = args[2];\n if (typeof options !== 'object' || Array.isArray(options))\n return context.error('NumberFormat options argument must be an object.');\n let locale = null;\n if (options['locale']) {\n locale = context.parse(options['locale'], 1, StringType);\n if (!locale)\n return null;\n }\n let currency = null;\n if (options['currency']) {\n currency = context.parse(options['currency'], 1, StringType);\n if (!currency)\n return null;\n }\n let minFractionDigits = null;\n if (options['min-fraction-digits']) {\n minFractionDigits = context.parse(options['min-fraction-digits'], 1, NumberType);\n if (!minFractionDigits)\n return null;\n }\n let maxFractionDigits = null;\n if (options['max-fraction-digits']) {\n maxFractionDigits = context.parse(options['max-fraction-digits'], 1, NumberType);\n if (!maxFractionDigits)\n return null;\n }\n return new NumberFormat(number, locale, currency, minFractionDigits, maxFractionDigits);\n }\n evaluate(ctx) {\n return new Intl.NumberFormat(this.locale ? this.locale.evaluate(ctx) : [], {\n style: this.currency ? 'currency' : 'decimal',\n currency: this.currency ? this.currency.evaluate(ctx) : undefined,\n minimumFractionDigits: this.minFractionDigits ? this.minFractionDigits.evaluate(ctx) : undefined,\n maximumFractionDigits: this.maxFractionDigits ? this.maxFractionDigits.evaluate(ctx) : undefined,\n }).format(this.number.evaluate(ctx));\n }\n eachChild(fn) {\n fn(this.number);\n if (this.locale) {\n fn(this.locale);\n }\n if (this.currency) {\n fn(this.currency);\n }\n if (this.minFractionDigits) {\n fn(this.minFractionDigits);\n }\n if (this.maxFractionDigits) {\n fn(this.maxFractionDigits);\n }\n }\n outputDefined() {\n return false;\n }\n}\n\nclass FormatExpression {\n constructor(sections) {\n this.type = FormattedType;\n this.sections = sections;\n }\n static parse(args, context) {\n if (args.length < 2) {\n return context.error('Expected at least one argument.');\n }\n const firstArg = args[1];\n if (!Array.isArray(firstArg) && typeof firstArg === 'object') {\n return context.error('First argument must be an image or text section.');\n }\n const sections = [];\n let nextTokenMayBeObject = false;\n for (let i = 1; i <= args.length - 1; ++i) {\n const arg = args[i];\n if (nextTokenMayBeObject && typeof arg === 'object' && !Array.isArray(arg)) {\n nextTokenMayBeObject = false;\n let scale = null;\n if (arg['font-scale']) {\n scale = context.parse(arg['font-scale'], 1, NumberType);\n if (!scale)\n return null;\n }\n let font = null;\n if (arg['text-font']) {\n font = context.parse(arg['text-font'], 1, array$1(StringType));\n if (!font)\n return null;\n }\n let textColor = null;\n if (arg['text-color']) {\n textColor = context.parse(arg['text-color'], 1, ColorType);\n if (!textColor)\n return null;\n }\n const lastExpression = sections[sections.length - 1];\n lastExpression.scale = scale;\n lastExpression.font = font;\n lastExpression.textColor = textColor;\n }\n else {\n const content = context.parse(args[i], 1, ValueType);\n if (!content)\n return null;\n const kind = content.type.kind;\n if (kind !== 'string' && kind !== 'value' && kind !== 'null' && kind !== 'resolvedImage')\n return context.error('Formatted text type must be \\'string\\', \\'value\\', \\'image\\' or \\'null\\'.');\n nextTokenMayBeObject = true;\n sections.push({ content, scale: null, font: null, textColor: null });\n }\n }\n return new FormatExpression(sections);\n }\n evaluate(ctx) {\n const evaluateSection = section => {\n const evaluatedContent = section.content.evaluate(ctx);\n if (typeOf(evaluatedContent) === ResolvedImageType) {\n return new FormattedSection('', evaluatedContent, null, null, null);\n }\n return new FormattedSection(toString(evaluatedContent), null, section.scale ? section.scale.evaluate(ctx) : null, section.font ? section.font.evaluate(ctx).join(',') : null, section.textColor ? section.textColor.evaluate(ctx) : null);\n };\n return new Formatted(this.sections.map(evaluateSection));\n }\n eachChild(fn) {\n for (const section of this.sections) {\n fn(section.content);\n if (section.scale) {\n fn(section.scale);\n }\n if (section.font) {\n fn(section.font);\n }\n if (section.textColor) {\n fn(section.textColor);\n }\n }\n }\n outputDefined() {\n // Technically the combinatoric set of all children\n // Usually, this.text will be undefined anyway\n return false;\n }\n}\n\nclass ImageExpression {\n constructor(input) {\n this.type = ResolvedImageType;\n this.input = input;\n }\n static parse(args, context) {\n if (args.length !== 2) {\n return context.error('Expected two arguments.');\n }\n const name = context.parse(args[1], 1, StringType);\n if (!name)\n return context.error('No image name provided.');\n return new ImageExpression(name);\n }\n evaluate(ctx) {\n const evaluatedImageName = this.input.evaluate(ctx);\n const value = ResolvedImage.fromString(evaluatedImageName);\n if (value && ctx.availableImages)\n value.available = ctx.availableImages.indexOf(evaluatedImageName) > -1;\n return value;\n }\n eachChild(fn) {\n fn(this.input);\n }\n outputDefined() {\n // The output of image is determined by the list of available images in the evaluation context\n return false;\n }\n}\n\nclass Length {\n constructor(input) {\n this.type = NumberType;\n this.input = input;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error(`Expected 1 argument, but found ${args.length - 1} instead.`);\n const input = context.parse(args[1], 1);\n if (!input)\n return null;\n if (input.type.kind !== 'array' && input.type.kind !== 'string' && input.type.kind !== 'value')\n return context.error(`Expected argument of type string or array, but found ${toString$1(input.type)} instead.`);\n return new Length(input);\n }\n evaluate(ctx) {\n const input = this.input.evaluate(ctx);\n if (typeof input === 'string') {\n return input.length;\n }\n else if (Array.isArray(input)) {\n return input.length;\n }\n else {\n throw new RuntimeError(`Expected value to be of type string or array, but found ${toString$1(typeOf(input))} instead.`);\n }\n }\n eachChild(fn) {\n fn(this.input);\n }\n outputDefined() {\n return false;\n }\n}\n\nconst expressions = {\n // special forms\n '==': Equals,\n '!=': NotEquals,\n '>': GreaterThan,\n '<': LessThan,\n '>=': GreaterThanOrEqual,\n '<=': LessThanOrEqual,\n 'array': Assertion,\n 'at': At,\n 'boolean': Assertion,\n 'case': Case,\n 'coalesce': Coalesce,\n 'collator': CollatorExpression,\n 'format': FormatExpression,\n 'image': ImageExpression,\n 'in': In,\n 'index-of': IndexOf,\n 'interpolate': Interpolate,\n 'interpolate-hcl': Interpolate,\n 'interpolate-lab': Interpolate,\n 'length': Length,\n 'let': Let,\n 'literal': Literal,\n 'match': Match,\n 'number': Assertion,\n 'number-format': NumberFormat,\n 'object': Assertion,\n 'slice': Slice,\n 'step': Step,\n 'string': Assertion,\n 'to-boolean': Coercion,\n 'to-color': Coercion,\n 'to-number': Coercion,\n 'to-string': Coercion,\n 'var': Var,\n 'within': Within\n};\nfunction rgba(ctx, [r, g, b, a]) {\n r = r.evaluate(ctx);\n g = g.evaluate(ctx);\n b = b.evaluate(ctx);\n const alpha = a ? a.evaluate(ctx) : 1;\n const error = validateRGBA(r, g, b, alpha);\n if (error)\n throw new RuntimeError(error);\n return new Color(r / 255, g / 255, b / 255, alpha, false);\n}\nfunction has(key, obj) {\n return key in obj;\n}\nfunction get(key, obj) {\n const v = obj[key];\n return typeof v === 'undefined' ? null : v;\n}\nfunction binarySearch(v, a, i, j) {\n while (i <= j) {\n const m = (i + j) >> 1;\n if (a[m] === v)\n return true;\n if (a[m] > v)\n j = m - 1;\n else\n i = m + 1;\n }\n return false;\n}\nfunction varargs(type) {\n return { type };\n}\nCompoundExpression.register(expressions, {\n 'error': [\n ErrorType,\n [StringType],\n (ctx, [v]) => { throw new RuntimeError(v.evaluate(ctx)); }\n ],\n 'typeof': [\n StringType,\n [ValueType],\n (ctx, [v]) => toString$1(typeOf(v.evaluate(ctx)))\n ],\n 'to-rgba': [\n array$1(NumberType, 4),\n [ColorType],\n (ctx, [v]) => {\n const [r, g, b, a] = v.evaluate(ctx).rgb;\n return [r * 255, g * 255, b * 255, a];\n },\n ],\n 'rgb': [\n ColorType,\n [NumberType, NumberType, NumberType],\n rgba\n ],\n 'rgba': [\n ColorType,\n [NumberType, NumberType, NumberType, NumberType],\n rgba\n ],\n 'has': {\n type: BooleanType,\n overloads: [\n [\n [StringType],\n (ctx, [key]) => has(key.evaluate(ctx), ctx.properties())\n ], [\n [StringType, ObjectType],\n (ctx, [key, obj]) => has(key.evaluate(ctx), obj.evaluate(ctx))\n ]\n ]\n },\n 'get': {\n type: ValueType,\n overloads: [\n [\n [StringType],\n (ctx, [key]) => get(key.evaluate(ctx), ctx.properties())\n ], [\n [StringType, ObjectType],\n (ctx, [key, obj]) => get(key.evaluate(ctx), obj.evaluate(ctx))\n ]\n ]\n },\n 'feature-state': [\n ValueType,\n [StringType],\n (ctx, [key]) => get(key.evaluate(ctx), ctx.featureState || {})\n ],\n 'properties': [\n ObjectType,\n [],\n (ctx) => ctx.properties()\n ],\n 'geometry-type': [\n StringType,\n [],\n (ctx) => ctx.geometryType()\n ],\n 'id': [\n ValueType,\n [],\n (ctx) => ctx.id()\n ],\n 'zoom': [\n NumberType,\n [],\n (ctx) => ctx.globals.zoom\n ],\n 'heatmap-density': [\n NumberType,\n [],\n (ctx) => ctx.globals.heatmapDensity || 0\n ],\n 'line-progress': [\n NumberType,\n [],\n (ctx) => ctx.globals.lineProgress || 0\n ],\n 'accumulated': [\n ValueType,\n [],\n (ctx) => ctx.globals.accumulated === undefined ? null : ctx.globals.accumulated\n ],\n '+': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => {\n let result = 0;\n for (const arg of args) {\n result += arg.evaluate(ctx);\n }\n return result;\n }\n ],\n '*': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => {\n let result = 1;\n for (const arg of args) {\n result *= arg.evaluate(ctx);\n }\n return result;\n }\n ],\n '-': {\n type: NumberType,\n overloads: [\n [\n [NumberType, NumberType],\n (ctx, [a, b]) => a.evaluate(ctx) - b.evaluate(ctx)\n ], [\n [NumberType],\n (ctx, [a]) => -a.evaluate(ctx)\n ]\n ]\n },\n '/': [\n NumberType,\n [NumberType, NumberType],\n (ctx, [a, b]) => a.evaluate(ctx) / b.evaluate(ctx)\n ],\n '%': [\n NumberType,\n [NumberType, NumberType],\n (ctx, [a, b]) => a.evaluate(ctx) % b.evaluate(ctx)\n ],\n 'ln2': [\n NumberType,\n [],\n () => Math.LN2\n ],\n 'pi': [\n NumberType,\n [],\n () => Math.PI\n ],\n 'e': [\n NumberType,\n [],\n () => Math.E\n ],\n '^': [\n NumberType,\n [NumberType, NumberType],\n (ctx, [b, e]) => Math.pow(b.evaluate(ctx), e.evaluate(ctx))\n ],\n 'sqrt': [\n NumberType,\n [NumberType],\n (ctx, [x]) => Math.sqrt(x.evaluate(ctx))\n ],\n 'log10': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.log(n.evaluate(ctx)) / Math.LN10\n ],\n 'ln': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.log(n.evaluate(ctx))\n ],\n 'log2': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.log(n.evaluate(ctx)) / Math.LN2\n ],\n 'sin': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.sin(n.evaluate(ctx))\n ],\n 'cos': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.cos(n.evaluate(ctx))\n ],\n 'tan': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.tan(n.evaluate(ctx))\n ],\n 'asin': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.asin(n.evaluate(ctx))\n ],\n 'acos': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.acos(n.evaluate(ctx))\n ],\n 'atan': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.atan(n.evaluate(ctx))\n ],\n 'min': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => Math.min(...args.map(arg => arg.evaluate(ctx)))\n ],\n 'max': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => Math.max(...args.map(arg => arg.evaluate(ctx)))\n ],\n 'abs': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.abs(n.evaluate(ctx))\n ],\n 'round': [\n NumberType,\n [NumberType],\n (ctx, [n]) => {\n const v = n.evaluate(ctx);\n // Javascript's Math.round() rounds towards +Infinity for halfway\n // values, even when they're negative. It's more common to round\n // away from 0 (e.g., this is what python and C++ do)\n return v < 0 ? -Math.round(-v) : Math.round(v);\n }\n ],\n 'floor': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.floor(n.evaluate(ctx))\n ],\n 'ceil': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.ceil(n.evaluate(ctx))\n ],\n 'filter-==': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => ctx.properties()[k.value] === v.value\n ],\n 'filter-id-==': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => ctx.id() === v.value\n ],\n 'filter-type-==': [\n BooleanType,\n [StringType],\n (ctx, [v]) => ctx.geometryType() === v.value\n ],\n 'filter-<': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a < b;\n }\n ],\n 'filter-id-<': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a < b;\n }\n ],\n 'filter->': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a > b;\n }\n ],\n 'filter-id->': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a > b;\n }\n ],\n 'filter-<=': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a <= b;\n }\n ],\n 'filter-id-<=': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a <= b;\n }\n ],\n 'filter->=': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a >= b;\n }\n ],\n 'filter-id->=': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a >= b;\n }\n ],\n 'filter-has': [\n BooleanType,\n [ValueType],\n (ctx, [k]) => k.value in ctx.properties()\n ],\n 'filter-has-id': [\n BooleanType,\n [],\n (ctx) => (ctx.id() !== null && ctx.id() !== undefined)\n ],\n 'filter-type-in': [\n BooleanType,\n [array$1(StringType)],\n (ctx, [v]) => v.value.indexOf(ctx.geometryType()) >= 0\n ],\n 'filter-id-in': [\n BooleanType,\n [array$1(ValueType)],\n (ctx, [v]) => v.value.indexOf(ctx.id()) >= 0\n ],\n 'filter-in-small': [\n BooleanType,\n [StringType, array$1(ValueType)],\n // assumes v is an array literal\n (ctx, [k, v]) => v.value.indexOf(ctx.properties()[k.value]) >= 0\n ],\n 'filter-in-large': [\n BooleanType,\n [StringType, array$1(ValueType)],\n // assumes v is a array literal with values sorted in ascending order and of a single type\n (ctx, [k, v]) => binarySearch(ctx.properties()[k.value], v.value, 0, v.value.length - 1)\n ],\n 'all': {\n type: BooleanType,\n overloads: [\n [\n [BooleanType, BooleanType],\n (ctx, [a, b]) => a.evaluate(ctx) && b.evaluate(ctx)\n ],\n [\n varargs(BooleanType),\n (ctx, args) => {\n for (const arg of args) {\n if (!arg.evaluate(ctx))\n return false;\n }\n return true;\n }\n ]\n ]\n },\n 'any': {\n type: BooleanType,\n overloads: [\n [\n [BooleanType, BooleanType],\n (ctx, [a, b]) => a.evaluate(ctx) || b.evaluate(ctx)\n ],\n [\n varargs(BooleanType),\n (ctx, args) => {\n for (const arg of args) {\n if (arg.evaluate(ctx))\n return true;\n }\n return false;\n }\n ]\n ]\n },\n '!': [\n BooleanType,\n [BooleanType],\n (ctx, [b]) => !b.evaluate(ctx)\n ],\n 'is-supported-script': [\n BooleanType,\n [StringType],\n // At parse time this will always return true, so we need to exclude this expression with isGlobalPropertyConstant\n (ctx, [s]) => {\n const isSupportedScript = ctx.globals && ctx.globals.isSupportedScript;\n if (isSupportedScript) {\n return isSupportedScript(s.evaluate(ctx));\n }\n return true;\n }\n ],\n 'upcase': [\n StringType,\n [StringType],\n (ctx, [s]) => s.evaluate(ctx).toUpperCase()\n ],\n 'downcase': [\n StringType,\n [StringType],\n (ctx, [s]) => s.evaluate(ctx).toLowerCase()\n ],\n 'concat': [\n StringType,\n varargs(ValueType),\n (ctx, args) => args.map(arg => toString(arg.evaluate(ctx))).join('')\n ],\n 'resolved-locale': [\n StringType,\n [CollatorType],\n (ctx, [collator]) => collator.evaluate(ctx).resolvedLocale()\n ]\n});\n\nfunction success(value) {\n return { result: 'success', value };\n}\nfunction error(value) {\n return { result: 'error', value };\n}\n\nfunction supportsPropertyExpression(spec) {\n return spec['property-type'] === 'data-driven' || spec['property-type'] === 'cross-faded-data-driven';\n}\nfunction supportsZoomExpression(spec) {\n return !!spec.expression && spec.expression.parameters.indexOf('zoom') > -1;\n}\nfunction supportsInterpolation(spec) {\n return !!spec.expression && spec.expression.interpolated;\n}\n\nfunction getType(val) {\n if (val instanceof Number) {\n return 'number';\n }\n else if (val instanceof String) {\n return 'string';\n }\n else if (val instanceof Boolean) {\n return 'boolean';\n }\n else if (Array.isArray(val)) {\n return 'array';\n }\n else if (val === null) {\n return 'null';\n }\n else {\n return typeof val;\n }\n}\n\nfunction isFunction(value) {\n return typeof value === 'object' && value !== null && !Array.isArray(value);\n}\nfunction identityFunction(x) {\n return x;\n}\nfunction createFunction(parameters, propertySpec) {\n const isColor = propertySpec.type === 'color';\n const zoomAndFeatureDependent = parameters.stops && typeof parameters.stops[0][0] === 'object';\n const featureDependent = zoomAndFeatureDependent || parameters.property !== undefined;\n const zoomDependent = zoomAndFeatureDependent || !featureDependent;\n const type = parameters.type || (supportsInterpolation(propertySpec) ? 'exponential' : 'interval');\n if (isColor || propertySpec.type === 'padding') {\n const parseFn = isColor ? Color.parse : Padding.parse;\n parameters = extendBy({}, parameters);\n if (parameters.stops) {\n parameters.stops = parameters.stops.map((stop) => {\n return [stop[0], parseFn(stop[1])];\n });\n }\n if (parameters.default) {\n parameters.default = parseFn(parameters.default);\n }\n else {\n parameters.default = parseFn(propertySpec.default);\n }\n }\n if (parameters.colorSpace && !isSupportedInterpolationColorSpace(parameters.colorSpace)) {\n throw new Error(`Unknown color space: \"${parameters.colorSpace}\"`);\n }\n let innerFun;\n let hashedStops;\n let categoricalKeyType;\n if (type === 'exponential') {\n innerFun = evaluateExponentialFunction;\n }\n else if (type === 'interval') {\n innerFun = evaluateIntervalFunction;\n }\n else if (type === 'categorical') {\n innerFun = evaluateCategoricalFunction;\n // For categorical functions, generate an Object as a hashmap of the stops for fast searching\n hashedStops = Object.create(null);\n for (const stop of parameters.stops) {\n hashedStops[stop[0]] = stop[1];\n }\n // Infer key type based on first stop key-- used to encforce strict type checking later\n categoricalKeyType = typeof parameters.stops[0][0];\n }\n else if (type === 'identity') {\n innerFun = evaluateIdentityFunction;\n }\n else {\n throw new Error(`Unknown function type \"${type}\"`);\n }\n if (zoomAndFeatureDependent) {\n const featureFunctions = {};\n const zoomStops = [];\n for (let s = 0; s < parameters.stops.length; s++) {\n const stop = parameters.stops[s];\n const zoom = stop[0].zoom;\n if (featureFunctions[zoom] === undefined) {\n featureFunctions[zoom] = {\n zoom,\n type: parameters.type,\n property: parameters.property,\n default: parameters.default,\n stops: []\n };\n zoomStops.push(zoom);\n }\n featureFunctions[zoom].stops.push([stop[0].value, stop[1]]);\n }\n const featureFunctionStops = [];\n for (const z of zoomStops) {\n featureFunctionStops.push([featureFunctions[z].zoom, createFunction(featureFunctions[z], propertySpec)]);\n }\n const interpolationType = { name: 'linear' };\n return {\n kind: 'composite',\n interpolationType,\n interpolationFactor: Interpolate.interpolationFactor.bind(undefined, interpolationType),\n zoomStops: featureFunctionStops.map(s => s[0]),\n evaluate({ zoom }, properties) {\n return evaluateExponentialFunction({\n stops: featureFunctionStops,\n base: parameters.base\n }, propertySpec, zoom).evaluate(zoom, properties);\n }\n };\n }\n else if (zoomDependent) {\n const interpolationType = type === 'exponential' ?\n { name: 'exponential', base: parameters.base !== undefined ? parameters.base : 1 } : null;\n return {\n kind: 'camera',\n interpolationType,\n interpolationFactor: Interpolate.interpolationFactor.bind(undefined, interpolationType),\n zoomStops: parameters.stops.map(s => s[0]),\n evaluate: ({ zoom }) => innerFun(parameters, propertySpec, zoom, hashedStops, categoricalKeyType)\n };\n }\n else {\n return {\n kind: 'source',\n evaluate(_, feature) {\n const value = feature && feature.properties ? feature.properties[parameters.property] : undefined;\n if (value === undefined) {\n return coalesce$1(parameters.default, propertySpec.default);\n }\n return innerFun(parameters, propertySpec, value, hashedStops, categoricalKeyType);\n }\n };\n }\n}\nfunction coalesce$1(a, b, c) {\n if (a !== undefined)\n return a;\n if (b !== undefined)\n return b;\n if (c !== undefined)\n return c;\n}\nfunction evaluateCategoricalFunction(parameters, propertySpec, input, hashedStops, keyType) {\n const evaluated = typeof input === keyType ? hashedStops[input] : undefined; // Enforce strict typing on input\n return coalesce$1(evaluated, parameters.default, propertySpec.default);\n}\nfunction evaluateIntervalFunction(parameters, propertySpec, input) {\n // Edge cases\n if (getType(input) !== 'number')\n return coalesce$1(parameters.default, propertySpec.default);\n const n = parameters.stops.length;\n if (n === 1)\n return parameters.stops[0][1];\n if (input <= parameters.stops[0][0])\n return parameters.stops[0][1];\n if (input >= parameters.stops[n - 1][0])\n return parameters.stops[n - 1][1];\n const index = findStopLessThanOrEqualTo(parameters.stops.map((stop) => stop[0]), input);\n return parameters.stops[index][1];\n}\nfunction evaluateExponentialFunction(parameters, propertySpec, input) {\n const base = parameters.base !== undefined ? parameters.base : 1;\n // Edge cases\n if (getType(input) !== 'number')\n return coalesce$1(parameters.default, propertySpec.default);\n const n = parameters.stops.length;\n if (n === 1)\n return parameters.stops[0][1];\n if (input <= parameters.stops[0][0])\n return parameters.stops[0][1];\n if (input >= parameters.stops[n - 1][0])\n return parameters.stops[n - 1][1];\n const index = findStopLessThanOrEqualTo(parameters.stops.map((stop) => stop[0]), input);\n const t = interpolationFactor(input, base, parameters.stops[index][0], parameters.stops[index + 1][0]);\n const outputLower = parameters.stops[index][1];\n const outputUpper = parameters.stops[index + 1][1];\n const interp = interpolate[propertySpec.type] || identityFunction;\n if (typeof outputLower.evaluate === 'function') {\n return {\n evaluate(...args) {\n const evaluatedLower = outputLower.evaluate.apply(undefined, args);\n const evaluatedUpper = outputUpper.evaluate.apply(undefined, args);\n // Special case for fill-outline-color, which has no spec default.\n if (evaluatedLower === undefined || evaluatedUpper === undefined) {\n return undefined;\n }\n return interp(evaluatedLower, evaluatedUpper, t, parameters.colorSpace);\n }\n };\n }\n return interp(outputLower, outputUpper, t, parameters.colorSpace);\n}\nfunction evaluateIdentityFunction(parameters, propertySpec, input) {\n switch (propertySpec.type) {\n case 'color':\n input = Color.parse(input);\n break;\n case 'formatted':\n input = Formatted.fromString(input.toString());\n break;\n case 'resolvedImage':\n input = ResolvedImage.fromString(input.toString());\n break;\n case 'padding':\n input = Padding.parse(input);\n break;\n default:\n if (getType(input) !== propertySpec.type && (propertySpec.type !== 'enum' || !propertySpec.values[input])) {\n input = undefined;\n }\n }\n return coalesce$1(input, parameters.default, propertySpec.default);\n}\n/**\n * Returns a ratio that can be used to interpolate between exponential function\n * stops.\n *\n * How it works:\n * Two consecutive stop values define a (scaled and shifted) exponential\n * function `f(x) = a * base^x + b`, where `base` is the user-specified base,\n * and `a` and `b` are constants affording sufficient degrees of freedom to fit\n * the function to the given stops.\n *\n * Here's a bit of algebra that lets us compute `f(x)` directly from the stop\n * values without explicitly solving for `a` and `b`:\n *\n * First stop value: `f(x0) = y0 = a * base^x0 + b`\n * Second stop value: `f(x1) = y1 = a * base^x1 + b`\n * => `y1 - y0 = a(base^x1 - base^x0)`\n * => `a = (y1 - y0)/(base^x1 - base^x0)`\n *\n * Desired value: `f(x) = y = a * base^x + b`\n * => `f(x) = y0 + a * (base^x - base^x0)`\n *\n * From the above, we can replace the `a` in `a * (base^x - base^x0)` and do a\n * little algebra:\n * ```\n * a * (base^x - base^x0) = (y1 - y0)/(base^x1 - base^x0) * (base^x - base^x0)\n * = (y1 - y0) * (base^x - base^x0) / (base^x1 - base^x0)\n * ```\n *\n * If we let `(base^x - base^x0) / (base^x1 base^x0)`, then we have\n * `f(x) = y0 + (y1 - y0) * ratio`. In other words, `ratio` may be treated as\n * an interpolation factor between the two stops' output values.\n *\n * (Note: a slightly different form for `ratio`,\n * `(base^(x-x0) - 1) / (base^(x1-x0) - 1) `, is equivalent, but requires fewer\n * expensive `Math.pow()` operations.)\n *\n * @private\n */\nfunction interpolationFactor(input, base, lowerValue, upperValue) {\n const difference = upperValue - lowerValue;\n const progress = input - lowerValue;\n if (difference === 0) {\n return 0;\n }\n else if (base === 1) {\n return progress / difference;\n }\n else {\n return (Math.pow(base, progress) - 1) / (Math.pow(base, difference) - 1);\n }\n}\n\nclass StyleExpression {\n constructor(expression, propertySpec) {\n this.expression = expression;\n this._warningHistory = {};\n this._evaluator = new EvaluationContext();\n this._defaultValue = propertySpec ? getDefaultValue(propertySpec) : null;\n this._enumValues = propertySpec && propertySpec.type === 'enum' ? propertySpec.values : null;\n }\n evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) {\n this._evaluator.globals = globals;\n this._evaluator.feature = feature;\n this._evaluator.featureState = featureState;\n this._evaluator.canonical = canonical;\n this._evaluator.availableImages = availableImages || null;\n this._evaluator.formattedSection = formattedSection;\n return this.expression.evaluate(this._evaluator);\n }\n evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) {\n this._evaluator.globals = globals;\n this._evaluator.feature = feature || null;\n this._evaluator.featureState = featureState || null;\n this._evaluator.canonical = canonical;\n this._evaluator.availableImages = availableImages || null;\n this._evaluator.formattedSection = formattedSection || null;\n try {\n const val = this.expression.evaluate(this._evaluator);\n // eslint-disable-next-line no-self-compare\n if (val === null || val === undefined || (typeof val === 'number' && val !== val)) {\n return this._defaultValue;\n }\n if (this._enumValues && !(val in this._enumValues)) {\n throw new RuntimeError(`Expected value to be one of ${Object.keys(this._enumValues).map(v => JSON.stringify(v)).join(', ')}, but found ${JSON.stringify(val)} instead.`);\n }\n return val;\n }\n catch (e) {\n if (!this._warningHistory[e.message]) {\n this._warningHistory[e.message] = true;\n if (typeof console !== 'undefined') {\n console.warn(e.message);\n }\n }\n return this._defaultValue;\n }\n }\n}\nfunction isExpression(expression) {\n return Array.isArray(expression) && expression.length > 0 &&\n typeof expression[0] === 'string' && expression[0] in expressions;\n}\n/**\n * Parse and typecheck the given style spec JSON expression. If\n * options.defaultValue is provided, then the resulting StyleExpression's\n * `evaluate()` method will handle errors by logging a warning (once per\n * message) and returning the default value. Otherwise, it will throw\n * evaluation errors.\n *\n * @private\n */\nfunction createExpression(expression, propertySpec) {\n const parser = new ParsingContext(expressions, isExpressionConstant, [], propertySpec ? getExpectedType(propertySpec) : undefined);\n // For string-valued properties, coerce to string at the top level rather than asserting.\n const parsed = parser.parse(expression, undefined, undefined, undefined, propertySpec && propertySpec.type === 'string' ? { typeAnnotation: 'coerce' } : undefined);\n if (!parsed) {\n return error(parser.errors);\n }\n return success(new StyleExpression(parsed, propertySpec));\n}\nclass ZoomConstantExpression {\n constructor(kind, expression) {\n this.kind = kind;\n this._styleExpression = expression;\n this.isStateDependent = kind !== 'constant' && !isStateConstant(expression.expression);\n }\n evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluate(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n}\nclass ZoomDependentExpression {\n constructor(kind, expression, zoomStops, interpolationType) {\n this.kind = kind;\n this.zoomStops = zoomStops;\n this._styleExpression = expression;\n this.isStateDependent = kind !== 'camera' && !isStateConstant(expression.expression);\n this.interpolationType = interpolationType;\n }\n evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluate(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n interpolationFactor(input, lower, upper) {\n if (this.interpolationType) {\n return Interpolate.interpolationFactor(this.interpolationType, input, lower, upper);\n }\n else {\n return 0;\n }\n }\n}\nfunction isZoomExpression(expression) {\n return expression._styleExpression !== undefined;\n}\nfunction createPropertyExpression(expressionInput, propertySpec) {\n const expression = createExpression(expressionInput, propertySpec);\n if (expression.result === 'error') {\n return expression;\n }\n const parsed = expression.value.expression;\n const isFeatureConstantResult = isFeatureConstant(parsed);\n if (!isFeatureConstantResult && !supportsPropertyExpression(propertySpec)) {\n return error([new ExpressionParsingError('', 'data expressions not supported')]);\n }\n const isZoomConstant = isGlobalPropertyConstant(parsed, ['zoom']);\n if (!isZoomConstant && !supportsZoomExpression(propertySpec)) {\n return error([new ExpressionParsingError('', 'zoom expressions not supported')]);\n }\n const zoomCurve = findZoomCurve(parsed);\n if (!zoomCurve && !isZoomConstant) {\n return error([new ExpressionParsingError('', '\"zoom\" expression may only be used as input to a top-level \"step\" or \"interpolate\" expression.')]);\n }\n else if (zoomCurve instanceof ExpressionParsingError) {\n return error([zoomCurve]);\n }\n else if (zoomCurve instanceof Interpolate && !supportsInterpolation(propertySpec)) {\n return error([new ExpressionParsingError('', '\"interpolate\" expressions cannot be used with this property')]);\n }\n if (!zoomCurve) {\n return success(isFeatureConstantResult ?\n new ZoomConstantExpression('constant', expression.value) :\n new ZoomConstantExpression('source', expression.value));\n }\n const interpolationType = zoomCurve instanceof Interpolate ? zoomCurve.interpolation : undefined;\n return success(isFeatureConstantResult ?\n new ZoomDependentExpression('camera', expression.value, zoomCurve.labels, interpolationType) :\n new ZoomDependentExpression('composite', expression.value, zoomCurve.labels, interpolationType));\n}\n// serialization wrapper for old-style stop functions normalized to the\n// expression interface\nclass StylePropertyFunction {\n constructor(parameters, specification) {\n this._parameters = parameters;\n this._specification = specification;\n extendBy(this, createFunction(this._parameters, this._specification));\n }\n static deserialize(serialized) {\n return new StylePropertyFunction(serialized._parameters, serialized._specification);\n }\n static serialize(input) {\n return {\n _parameters: input._parameters,\n _specification: input._specification\n };\n }\n}\nfunction normalizePropertyExpression(value, specification) {\n if (isFunction(value)) {\n return new StylePropertyFunction(value, specification);\n }\n else if (isExpression(value)) {\n const expression = createPropertyExpression(value, specification);\n if (expression.result === 'error') {\n // this should have been caught in validation\n throw new Error(expression.value.map(err => `${err.key}: ${err.message}`).join(', '));\n }\n return expression.value;\n }\n else {\n let constant = value;\n if (specification.type === 'color' && typeof value === 'string') {\n constant = Color.parse(value);\n }\n else if (specification.type === 'padding' && (typeof value === 'number' || Array.isArray(value))) {\n constant = Padding.parse(value);\n }\n else if (specification.type === 'variableAnchorOffsetCollection' && Array.isArray(value)) {\n constant = VariableAnchorOffsetCollection.parse(value);\n }\n return {\n kind: 'constant',\n evaluate: () => constant\n };\n }\n}\n// Zoom-dependent expressions may only use [\"zoom\"] as the input to a top-level \"step\" or \"interpolate\"\n// expression (collectively referred to as a \"curve\"). The curve may be wrapped in one or more \"let\" or\n// \"coalesce\" expressions.\nfunction findZoomCurve(expression) {\n let result = null;\n if (expression instanceof Let) {\n result = findZoomCurve(expression.result);\n }\n else if (expression instanceof Coalesce) {\n for (const arg of expression.args) {\n result = findZoomCurve(arg);\n if (result) {\n break;\n }\n }\n }\n else if ((expression instanceof Step || expression instanceof Interpolate) &&\n expression.input instanceof CompoundExpression &&\n expression.input.name === 'zoom') {\n result = expression;\n }\n if (result instanceof ExpressionParsingError) {\n return result;\n }\n expression.eachChild((child) => {\n const childResult = findZoomCurve(child);\n if (childResult instanceof ExpressionParsingError) {\n result = childResult;\n }\n else if (!result && childResult) {\n result = new ExpressionParsingError('', '\"zoom\" expression may only be used as input to a top-level \"step\" or \"interpolate\" expression.');\n }\n else if (result && childResult && result !== childResult) {\n result = new ExpressionParsingError('', 'Only one zoom-based \"step\" or \"interpolate\" subexpression may be used in an expression.');\n }\n });\n return result;\n}\nfunction getExpectedType(spec) {\n const types = {\n color: ColorType,\n string: StringType,\n number: NumberType,\n enum: StringType,\n boolean: BooleanType,\n formatted: FormattedType,\n padding: PaddingType,\n resolvedImage: ResolvedImageType,\n variableAnchorOffsetCollection: VariableAnchorOffsetCollectionType\n };\n if (spec.type === 'array') {\n return array$1(types[spec.value] || ValueType, spec.length);\n }\n return types[spec.type];\n}\nfunction getDefaultValue(spec) {\n if (spec.type === 'color' && isFunction(spec.default)) {\n // Special case for heatmap-color: it uses the 'default:' to define a\n // default color ramp, but createExpression expects a simple value to fall\n // back to in case of runtime errors\n return new Color(0, 0, 0, 0);\n }\n else if (spec.type === 'color') {\n return Color.parse(spec.default) || null;\n }\n else if (spec.type === 'padding') {\n return Padding.parse(spec.default) || null;\n }\n else if (spec.type === 'variableAnchorOffsetCollection') {\n return VariableAnchorOffsetCollection.parse(spec.default) || null;\n }\n else if (spec.default === undefined) {\n return null;\n }\n else {\n return spec.default;\n }\n}\n\nfunction isExpressionFilter(filter) {\n if (filter === true || filter === false) {\n return true;\n }\n if (!Array.isArray(filter) || filter.length === 0) {\n return false;\n }\n switch (filter[0]) {\n case 'has':\n return filter.length >= 2 && filter[1] !== '$id' && filter[1] !== '$type';\n case 'in':\n return filter.length >= 3 && (typeof filter[1] !== 'string' || Array.isArray(filter[2]));\n case '!in':\n case '!has':\n case 'none':\n return false;\n case '==':\n case '!=':\n case '>':\n case '>=':\n case '<':\n case '<=':\n return filter.length !== 3 || (Array.isArray(filter[1]) || Array.isArray(filter[2]));\n case 'any':\n case 'all':\n for (const f of filter.slice(1)) {\n if (!isExpressionFilter(f) && typeof f !== 'boolean') {\n return false;\n }\n }\n return true;\n default:\n return true;\n }\n}\nconst filterSpec = {\n 'type': 'boolean',\n 'default': false,\n 'transition': false,\n 'property-type': 'data-driven',\n 'expression': {\n 'interpolated': false,\n 'parameters': ['zoom', 'feature']\n }\n};\n/**\n * Given a filter expressed as nested arrays, return a new function\n * that evaluates whether a given feature (with a .properties or .tags property)\n * passes its test.\n *\n * @private\n * @param {Array} filter MapLibre filter\n * @returns {Function} filter-evaluating function\n */\nfunction createFilter(filter) {\n if (filter === null || filter === undefined) {\n return { filter: () => true, needGeometry: false };\n }\n if (!isExpressionFilter(filter)) {\n filter = convertFilter$1(filter);\n }\n const compiled = createExpression(filter, filterSpec);\n if (compiled.result === 'error') {\n throw new Error(compiled.value.map(err => `${err.key}: ${err.message}`).join(', '));\n }\n else {\n const needGeometry = geometryNeeded(filter);\n return { filter: (globalProperties, feature, canonical) => compiled.value.evaluate(globalProperties, feature, {}, canonical),\n needGeometry };\n }\n}\n// Comparison function to sort numbers and strings\nfunction compare(a, b) {\n return a < b ? -1 : a > b ? 1 : 0;\n}\nfunction geometryNeeded(filter) {\n if (!Array.isArray(filter))\n return false;\n if (filter[0] === 'within')\n return true;\n for (let index = 1; index < filter.length; index++) {\n if (geometryNeeded(filter[index]))\n return true;\n }\n return false;\n}\nfunction convertFilter$1(filter) {\n if (!filter)\n return true;\n const op = filter[0];\n if (filter.length <= 1)\n return (op !== 'any');\n const converted = op === '==' ? convertComparisonOp$1(filter[1], filter[2], '==') :\n op === '!=' ? convertNegation(convertComparisonOp$1(filter[1], filter[2], '==')) :\n op === '<' ||\n op === '>' ||\n op === '<=' ||\n op === '>=' ? convertComparisonOp$1(filter[1], filter[2], op) :\n op === 'any' ? convertDisjunctionOp(filter.slice(1)) :\n op === 'all' ? ['all'].concat(filter.slice(1).map(convertFilter$1)) :\n op === 'none' ? ['all'].concat(filter.slice(1).map(convertFilter$1).map(convertNegation)) :\n op === 'in' ? convertInOp$1(filter[1], filter.slice(2)) :\n op === '!in' ? convertNegation(convertInOp$1(filter[1], filter.slice(2))) :\n op === 'has' ? convertHasOp$1(filter[1]) :\n op === '!has' ? convertNegation(convertHasOp$1(filter[1])) :\n op === 'within' ? filter :\n true;\n return converted;\n}\nfunction convertComparisonOp$1(property, value, op) {\n switch (property) {\n case '$type':\n return [`filter-type-${op}`, value];\n case '$id':\n return [`filter-id-${op}`, value];\n default:\n return [`filter-${op}`, property, value];\n }\n}\nfunction convertDisjunctionOp(filters) {\n return ['any'].concat(filters.map(convertFilter$1));\n}\nfunction convertInOp$1(property, values) {\n if (values.length === 0) {\n return false;\n }\n switch (property) {\n case '$type':\n return ['filter-type-in', ['literal', values]];\n case '$id':\n return ['filter-id-in', ['literal', values]];\n default:\n if (values.length > 200 && !values.some(v => typeof v !== typeof values[0])) {\n return ['filter-in-large', property, ['literal', values.sort(compare)]];\n }\n else {\n return ['filter-in-small', property, ['literal', values]];\n }\n }\n}\nfunction convertHasOp$1(property) {\n switch (property) {\n case '$type':\n return true;\n case '$id':\n return ['filter-has-id'];\n default:\n return ['filter-has', property];\n }\n}\nfunction convertNegation(filter) {\n return ['!', filter];\n}\n\n/*\n * Convert the given filter to an expression, storing the expected types for\n * any feature properties referenced in expectedTypes.\n *\n * These expected types are needed in order to construct preflight type checks\n * needed for handling 'any' filters. A preflight type check is necessary in\n * order to mimic legacy filters' semantics around expected type mismatches.\n * For example, consider the legacy filter:\n *\n * [\"any\", [\"all\", [\">\", \"y\", 0], [\">\", \"y\", 0]], [\">\", \"x\", 0]]\n *\n * Naively, we might convert this to the expression:\n *\n * [\"any\", [\"all\", [\">\", [\"get\", \"y\"], 0], [\">\", [\"get\", \"z\"], 0]], [\">\", [\"get\", \"x\"], 0]]\n *\n * But if we tried to evaluate this against, say `{x: 1, y: null, z: 0}`, the\n * [\">\", [\"get\", \"y\"], 0] would cause an evaluation error, leading to the\n * entire filter returning false. Legacy filter semantics, though, ask for\n * [\">\", \"y\", 0] to simply return `false` when `y` is of the wrong type,\n * allowing the subsequent terms of the outer \"any\" expression to be evaluated\n * (resulting, in this case, in a `true` value, because x > 0).\n *\n * We account for this by inserting a preflight type-checking expression before\n * each \"any\" term, allowing us to avoid evaluating the actual converted filter\n * if any type mismatches would cause it to produce an evalaution error:\n *\n * [\"any\",\n * [\"case\",\n * [\"all\", [\"==\", [\"typeof\", [\"get\", \"y\"]], \"number\"], [\"==\", [\"typeof\", [\"get\", \"z\"], \"number]],\n * [\"all\", [\">\", [\"get\", \"y\"], 0], [\">\", [\"get\", \"z\"], 0]],\n * false\n * ],\n * [\"case\",\n * [\"==\", [\"typeof\", [\"get\", \"x\"], \"number\"]],\n * [\">\", [\"get\", \"x\"], 0],\n * false\n * ]\n * ]\n *\n * An alternative, possibly more direct approach would be to use type checks\n * in the conversion of each comparison operator, so that the converted version\n * of each individual ==, >=, etc. would mimic the legacy filter semantics. The\n * downside of this approach is that it can lead to many more type checks than\n * would otherwise be necessary: outside the context of an \"any\" expression,\n * bailing out due to a runtime type error (expression semantics) and returning\n * false (legacy filter semantics) are equivalent: they cause the filter to\n * produce a `false` result.\n */\nfunction convertFilter(filter, expectedTypes = {}) {\n if (isExpressionFilter(filter))\n return filter;\n if (!filter)\n return true;\n const legacyFilter = filter;\n const legacyOp = legacyFilter[0];\n if (filter.length <= 1)\n return (legacyOp !== 'any');\n switch (legacyOp) {\n case '==':\n case '!=':\n case '<':\n case '>':\n case '<=':\n case '>=': {\n const [, property, value] = filter;\n return convertComparisonOp(property, value, legacyOp, expectedTypes);\n }\n case 'any': {\n const [, ...conditions] = legacyFilter;\n const children = conditions.map((f) => {\n const types = {};\n const child = convertFilter(f, types);\n const typechecks = runtimeTypeChecks(types);\n return typechecks === true ? child : ['case', typechecks, child, false];\n });\n return ['any', ...children];\n }\n case 'all': {\n const [, ...conditions] = legacyFilter;\n const children = conditions.map(f => convertFilter(f, expectedTypes));\n return children.length > 1 ? ['all', ...children] : children[0];\n }\n case 'none': {\n const [, ...conditions] = legacyFilter;\n return ['!', convertFilter(['any', ...conditions], {})];\n }\n case 'in': {\n const [, property, ...values] = legacyFilter;\n return convertInOp(property, values);\n }\n case '!in': {\n const [, property, ...values] = legacyFilter;\n return convertInOp(property, values, true);\n }\n case 'has':\n return convertHasOp(legacyFilter[1]);\n case '!has':\n return ['!', convertHasOp(legacyFilter[1])];\n default:\n return true;\n }\n}\n// Given a set of feature properties and an expected type for each one,\n// construct an boolean expression that tests whether each property has the\n// right type.\n// E.g.: for {name: 'string', population: 'number'}, return\n// [ 'all',\n// ['==', ['typeof', ['get', 'name'], 'string']],\n// ['==', ['typeof', ['get', 'population'], 'number]]\n// ]\nfunction runtimeTypeChecks(expectedTypes) {\n const conditions = [];\n for (const property in expectedTypes) {\n const get = property === '$id' ? ['id'] : ['get', property];\n conditions.push(['==', ['typeof', get], expectedTypes[property]]);\n }\n if (conditions.length === 0)\n return true;\n if (conditions.length === 1)\n return conditions[0];\n return ['all', ...conditions];\n}\nfunction convertComparisonOp(property, value, op, expectedTypes) {\n let get;\n if (property === '$type') {\n return [op, ['geometry-type'], value];\n }\n else if (property === '$id') {\n get = ['id'];\n }\n else {\n get = ['get', property];\n }\n if (expectedTypes && value !== null) {\n const type = typeof value;\n expectedTypes[property] = type;\n }\n if (op === '==' && property !== '$id' && value === null) {\n return [\n 'all',\n ['has', property],\n ['==', get, null]\n ];\n }\n else if (op === '!=' && property !== '$id' && value === null) {\n return [\n 'any',\n ['!', ['has', property]],\n ['!=', get, null]\n ];\n }\n return [op, get, value];\n}\nfunction convertInOp(property, values, negate = false) {\n if (values.length === 0)\n return negate;\n let get;\n if (property === '$type') {\n get = ['geometry-type'];\n }\n else if (property === '$id') {\n get = ['id'];\n }\n else {\n get = ['get', property];\n }\n // Determine if the list of values to be searched is homogenously typed.\n // If so (and if the type is string or number), then we can use a\n // [match, input, [...values], true, false] construction rather than a\n // bunch of `==` tests.\n let uniformTypes = true;\n const type = typeof values[0];\n for (const value of values) {\n if (typeof value !== type) {\n uniformTypes = false;\n break;\n }\n }\n if (uniformTypes && (type === 'string' || type === 'number')) {\n // Match expressions must have unique values.\n const uniqueValues = values.sort().filter((v, i) => i === 0 || values[i - 1] !== v);\n return ['match', get, uniqueValues, !negate, negate];\n }\n if (negate) {\n return ['all', ...values.map(v => ['!=', get, v])];\n }\n else {\n return ['any', ...values.map(v => ['==', get, v])];\n }\n}\nfunction convertHasOp(property) {\n if (property === '$type') {\n return true;\n }\n else if (property === '$id') {\n return ['!=', ['id'], null];\n }\n else {\n return ['has', property];\n }\n}\n\nfunction convertLiteral(value) {\n return typeof value === 'object' ? ['literal', value] : value;\n}\nfunction convertFunction(parameters, propertySpec) {\n let stops = parameters.stops;\n if (!stops) {\n // identity function\n return convertIdentityFunction(parameters, propertySpec);\n }\n const zoomAndFeatureDependent = stops && typeof stops[0][0] === 'object';\n const featureDependent = zoomAndFeatureDependent || parameters.property !== undefined;\n const zoomDependent = zoomAndFeatureDependent || !featureDependent;\n stops = stops.map((stop) => {\n if (!featureDependent && propertySpec.tokens && typeof stop[1] === 'string') {\n return [stop[0], convertTokenString(stop[1])];\n }\n return [stop[0], convertLiteral(stop[1])];\n });\n if (zoomAndFeatureDependent) {\n return convertZoomAndPropertyFunction(parameters, propertySpec, stops);\n }\n else if (zoomDependent) {\n return convertZoomFunction(parameters, propertySpec, stops);\n }\n else {\n return convertPropertyFunction(parameters, propertySpec, stops);\n }\n}\nfunction convertIdentityFunction(parameters, propertySpec) {\n const get = ['get', parameters.property];\n if (parameters.default === undefined) {\n // By default, expressions for string-valued properties get coerced. To preserve\n // legacy function semantics, insert an explicit assertion instead.\n return propertySpec.type === 'string' ? ['string', get] : get;\n }\n else if (propertySpec.type === 'enum') {\n return [\n 'match',\n get,\n Object.keys(propertySpec.values),\n get,\n parameters.default\n ];\n }\n else {\n const expression = [propertySpec.type === 'color' ? 'to-color' : propertySpec.type, get, convertLiteral(parameters.default)];\n if (propertySpec.type === 'array') {\n expression.splice(1, 0, propertySpec.value, propertySpec.length || null);\n }\n return expression;\n }\n}\nfunction getInterpolateOperator(parameters) {\n switch (parameters.colorSpace) {\n case 'hcl': return 'interpolate-hcl';\n case 'lab': return 'interpolate-lab';\n default: return 'interpolate';\n }\n}\nfunction convertZoomAndPropertyFunction(parameters, propertySpec, stops) {\n const featureFunctionParameters = {};\n const featureFunctionStops = {};\n const zoomStops = [];\n for (let s = 0; s < stops.length; s++) {\n const stop = stops[s];\n const zoom = stop[0].zoom;\n if (featureFunctionParameters[zoom] === undefined) {\n featureFunctionParameters[zoom] = {\n zoom,\n type: parameters.type,\n property: parameters.property,\n default: parameters.default,\n };\n featureFunctionStops[zoom] = [];\n zoomStops.push(zoom);\n }\n featureFunctionStops[zoom].push([stop[0].value, stop[1]]);\n }\n // the interpolation type for the zoom dimension of a zoom-and-property\n // function is determined directly from the style property specification\n // for which it's being used: linear for interpolatable properties, step\n // otherwise.\n const functionType = getFunctionType({}, propertySpec);\n if (functionType === 'exponential') {\n const expression = [getInterpolateOperator(parameters), ['linear'], ['zoom']];\n for (const z of zoomStops) {\n const output = convertPropertyFunction(featureFunctionParameters[z], propertySpec, featureFunctionStops[z]);\n appendStopPair(expression, z, output, false);\n }\n return expression;\n }\n else {\n const expression = ['step', ['zoom']];\n for (const z of zoomStops) {\n const output = convertPropertyFunction(featureFunctionParameters[z], propertySpec, featureFunctionStops[z]);\n appendStopPair(expression, z, output, true);\n }\n fixupDegenerateStepCurve(expression);\n return expression;\n }\n}\nfunction coalesce(a, b) {\n if (a !== undefined)\n return a;\n if (b !== undefined)\n return b;\n}\nfunction getFallback(parameters, propertySpec) {\n const defaultValue = convertLiteral(coalesce(parameters.default, propertySpec.default));\n /*\n * Some fields with type: resolvedImage have an undefined default.\n * Because undefined is an invalid value for resolvedImage, set fallback to\n * an empty string instead of undefined to ensure output\n * passes validation.\n */\n if (defaultValue === undefined && propertySpec.type === 'resolvedImage') {\n return '';\n }\n return defaultValue;\n}\nfunction convertPropertyFunction(parameters, propertySpec, stops) {\n const type = getFunctionType(parameters, propertySpec);\n const get = ['get', parameters.property];\n if (type === 'categorical' && typeof stops[0][0] === 'boolean') {\n const expression = ['case'];\n for (const stop of stops) {\n expression.push(['==', get, stop[0]], stop[1]);\n }\n expression.push(getFallback(parameters, propertySpec));\n return expression;\n }\n else if (type === 'categorical') {\n const expression = ['match', get];\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], false);\n }\n expression.push(getFallback(parameters, propertySpec));\n return expression;\n }\n else if (type === 'interval') {\n const expression = ['step', ['number', get]];\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], true);\n }\n fixupDegenerateStepCurve(expression);\n return parameters.default === undefined ? expression : [\n 'case',\n ['==', ['typeof', get], 'number'],\n expression,\n convertLiteral(parameters.default)\n ];\n }\n else if (type === 'exponential') {\n const base = parameters.base !== undefined ? parameters.base : 1;\n const expression = [\n getInterpolateOperator(parameters),\n base === 1 ? ['linear'] : ['exponential', base],\n ['number', get]\n ];\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], false);\n }\n return parameters.default === undefined ? expression : [\n 'case',\n ['==', ['typeof', get], 'number'],\n expression,\n convertLiteral(parameters.default)\n ];\n }\n else {\n throw new Error(`Unknown property function type ${type}`);\n }\n}\nfunction convertZoomFunction(parameters, propertySpec, stops, input = ['zoom']) {\n const type = getFunctionType(parameters, propertySpec);\n let expression;\n let isStep = false;\n if (type === 'interval') {\n expression = ['step', input];\n isStep = true;\n }\n else if (type === 'exponential') {\n const base = parameters.base !== undefined ? parameters.base : 1;\n expression = [getInterpolateOperator(parameters), base === 1 ? ['linear'] : ['exponential', base], input];\n }\n else {\n throw new Error(`Unknown zoom function type \"${type}\"`);\n }\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], isStep);\n }\n fixupDegenerateStepCurve(expression);\n return expression;\n}\nfunction fixupDegenerateStepCurve(expression) {\n // degenerate step curve (i.e. a constant function): add a noop stop\n if (expression[0] === 'step' && expression.length === 3) {\n expression.push(0);\n expression.push(expression[3]);\n }\n}\nfunction appendStopPair(curve, input, output, isStep) {\n // Skip duplicate stop values. They were not validated for functions, but they are for expressions.\n // https://github.com/mapbox/mapbox-gl-js/issues/4107\n if (curve.length > 3 && input === curve[curve.length - 2]) {\n return;\n }\n // step curves don't get the first input value, as it is redundant.\n if (!(isStep && curve.length === 2)) {\n curve.push(input);\n }\n curve.push(output);\n}\nfunction getFunctionType(parameters, propertySpec) {\n if (parameters.type) {\n return parameters.type;\n }\n else {\n return propertySpec.expression.interpolated ? 'exponential' : 'interval';\n }\n}\n// \"String with {name} token\" => [\"concat\", \"String with \", [\"get\", \"name\"], \" token\"]\nfunction convertTokenString(s) {\n const result = ['concat'];\n const re = /{([^{}]+)}/g;\n let pos = 0;\n for (let match = re.exec(s); match !== null; match = re.exec(s)) {\n const literal = s.slice(pos, re.lastIndex - match[0].length);\n pos = re.lastIndex;\n if (literal.length > 0)\n result.push(literal);\n result.push(['get', match[1]]);\n }\n if (result.length === 1) {\n return s;\n }\n if (pos < s.length) {\n result.push(s.slice(pos));\n }\n else if (result.length === 2) {\n return ['to-string', result[1]];\n }\n return result;\n}\n\nfunction getPropertyReference(propertyName) {\n for (let i = 0; i < v8Spec.layout.length; i++) {\n for (const key in v8Spec[v8Spec.layout[i]]) {\n if (key === propertyName)\n return v8Spec[v8Spec.layout[i]][key];\n }\n }\n for (let i = 0; i < v8Spec.paint.length; i++) {\n for (const key in v8Spec[v8Spec.paint[i]]) {\n if (key === propertyName)\n return v8Spec[v8Spec.paint[i]][key];\n }\n }\n return null;\n}\nfunction eachSource(style, callback) {\n for (const k in style.sources) {\n callback(style.sources[k]);\n }\n}\nfunction eachLayer(style, callback) {\n for (const layer of style.layers) {\n callback(layer);\n }\n}\nfunction eachProperty(style, options, callback) {\n function inner(layer, propertyType) {\n const properties = layer[propertyType];\n if (!properties)\n return;\n Object.keys(properties).forEach((key) => {\n callback({\n path: [layer.id, propertyType, key],\n key,\n value: properties[key],\n reference: getPropertyReference(key),\n set(x) {\n properties[key] = x;\n }\n });\n });\n }\n eachLayer(style, (layer) => {\n if (options.paint) {\n inner(layer, 'paint');\n }\n if (options.layout) {\n inner(layer, 'layout');\n }\n });\n}\n\nfunction stringify(obj) {\n const type = typeof obj;\n if (type === 'number' || type === 'boolean' || type === 'string' || obj === undefined || obj === null)\n return JSON.stringify(obj);\n if (Array.isArray(obj)) {\n let str = '[';\n for (const val of obj) {\n str += `${stringify(val)},`;\n }\n return `${str}]`;\n }\n const keys = Object.keys(obj).sort();\n let str = '{';\n for (let i = 0; i < keys.length; i++) {\n str += `${JSON.stringify(keys[i])}:${stringify(obj[keys[i]])},`;\n }\n return `${str}}`;\n}\nfunction getKey(layer) {\n let key = '';\n for (const k of refProperties) {\n key += `/${stringify(layer[k])}`;\n }\n return key;\n}\n/**\n * Given an array of layers, return an array of arrays of layers where all\n * layers in each group have identical layout-affecting properties. These\n * are the properties that were formerly used by explicit `ref` mechanism\n * for layers: 'type', 'source', 'source-layer', 'minzoom', 'maxzoom',\n * 'filter', and 'layout'.\n *\n * The input is not modified. The output layers are references to the\n * input layers.\n *\n * @private\n * @param {Array} layers\n * @param {Object} [cachedKeys] - an object to keep already calculated keys.\n * @returns {Array>}\n */\nfunction groupByLayout(layers, cachedKeys) {\n const groups = {};\n for (let i = 0; i < layers.length; i++) {\n const k = (cachedKeys && cachedKeys[layers[i].id]) || getKey(layers[i]);\n // update the cache if there is one\n if (cachedKeys)\n cachedKeys[layers[i].id] = k;\n let group = groups[k];\n if (!group) {\n group = groups[k] = [];\n }\n group.push(layers[i]);\n }\n const result = [];\n for (const k in groups) {\n result.push(groups[k]);\n }\n return result;\n}\n\nfunction emptyStyle() {\n const style = {};\n const version = v8Spec['$version'];\n for (const styleKey in v8Spec['$root']) {\n const spec = v8Spec['$root'][styleKey];\n if (spec.required) {\n let value = null;\n if (styleKey === 'version') {\n value = version;\n }\n else {\n if (spec.type === 'array') {\n value = [];\n }\n else {\n value = {};\n }\n }\n if (value != null) {\n style[styleKey] = value;\n }\n }\n }\n return style;\n}\n\nfunction validateConstants(options) {\n const key = options.key;\n const constants = options.value;\n if (constants) {\n return [new ValidationError(key, constants, 'constants have been deprecated as of v8')];\n }\n else {\n return [];\n }\n}\n\n// Turn jsonlint-lines-primitives objects into primitive objects\nfunction unbundle(value) {\n if (value instanceof Number || value instanceof String || value instanceof Boolean) {\n return value.valueOf();\n }\n else {\n return value;\n }\n}\nfunction deepUnbundle(value) {\n if (Array.isArray(value)) {\n return value.map(deepUnbundle);\n }\n else if (value instanceof Object && !(value instanceof Number || value instanceof String || value instanceof Boolean)) {\n const unbundledValue = {};\n for (const key in value) {\n unbundledValue[key] = deepUnbundle(value[key]);\n }\n return unbundledValue;\n }\n return unbundle(value);\n}\n\nfunction validateObject(options) {\n const key = options.key;\n const object = options.value;\n const elementSpecs = options.valueSpec || {};\n const elementValidators = options.objectElementValidators || {};\n const style = options.style;\n const styleSpec = options.styleSpec;\n const validateSpec = options.validateSpec;\n let errors = [];\n const type = getType(object);\n if (type !== 'object') {\n return [new ValidationError(key, object, `object expected, ${type} found`)];\n }\n for (const objectKey in object) {\n const elementSpecKey = objectKey.split('.')[0]; // treat 'paint.*' as 'paint'\n const elementSpec = elementSpecs[elementSpecKey] || elementSpecs['*'];\n let validateElement;\n if (elementValidators[elementSpecKey]) {\n validateElement = elementValidators[elementSpecKey];\n }\n else if (elementSpecs[elementSpecKey]) {\n validateElement = validateSpec;\n }\n else if (elementValidators['*']) {\n validateElement = elementValidators['*'];\n }\n else if (elementSpecs['*']) {\n validateElement = validateSpec;\n }\n else {\n errors.push(new ValidationError(key, object[objectKey], `unknown property \"${objectKey}\"`));\n continue;\n }\n errors = errors.concat(validateElement({\n key: (key ? `${key}.` : key) + objectKey,\n value: object[objectKey],\n valueSpec: elementSpec,\n style,\n styleSpec,\n object,\n objectKey,\n validateSpec,\n }, object));\n }\n for (const elementSpecKey in elementSpecs) {\n // Don't check `required` when there's a custom validator for that property.\n if (elementValidators[elementSpecKey]) {\n continue;\n }\n if (elementSpecs[elementSpecKey].required && elementSpecs[elementSpecKey]['default'] === undefined && object[elementSpecKey] === undefined) {\n errors.push(new ValidationError(key, object, `missing required property \"${elementSpecKey}\"`));\n }\n }\n return errors;\n}\n\nfunction validateArray(options) {\n const array = options.value;\n const arraySpec = options.valueSpec;\n const validateSpec = options.validateSpec;\n const style = options.style;\n const styleSpec = options.styleSpec;\n const key = options.key;\n const validateArrayElement = options.arrayElementValidator || validateSpec;\n if (getType(array) !== 'array') {\n return [new ValidationError(key, array, `array expected, ${getType(array)} found`)];\n }\n if (arraySpec.length && array.length !== arraySpec.length) {\n return [new ValidationError(key, array, `array length ${arraySpec.length} expected, length ${array.length} found`)];\n }\n if (arraySpec['min-length'] && array.length < arraySpec['min-length']) {\n return [new ValidationError(key, array, `array length at least ${arraySpec['min-length']} expected, length ${array.length} found`)];\n }\n let arrayElementSpec = {\n 'type': arraySpec.value,\n 'values': arraySpec.values\n };\n if (styleSpec.$version < 7) {\n arrayElementSpec['function'] = arraySpec.function;\n }\n if (getType(arraySpec.value) === 'object') {\n arrayElementSpec = arraySpec.value;\n }\n let errors = [];\n for (let i = 0; i < array.length; i++) {\n errors = errors.concat(validateArrayElement({\n array,\n arrayIndex: i,\n value: array[i],\n valueSpec: arrayElementSpec,\n validateSpec: options.validateSpec,\n style,\n styleSpec,\n key: `${key}[${i}]`\n }));\n }\n return errors;\n}\n\nfunction validateNumber(options) {\n const key = options.key;\n const value = options.value;\n const valueSpec = options.valueSpec;\n let type = getType(value);\n // eslint-disable-next-line no-self-compare\n if (type === 'number' && value !== value) {\n type = 'NaN';\n }\n if (type !== 'number') {\n return [new ValidationError(key, value, `number expected, ${type} found`)];\n }\n if ('minimum' in valueSpec && value < valueSpec.minimum) {\n return [new ValidationError(key, value, `${value} is less than the minimum value ${valueSpec.minimum}`)];\n }\n if ('maximum' in valueSpec && value > valueSpec.maximum) {\n return [new ValidationError(key, value, `${value} is greater than the maximum value ${valueSpec.maximum}`)];\n }\n return [];\n}\n\nfunction validateFunction(options) {\n const functionValueSpec = options.valueSpec;\n const functionType = unbundle(options.value.type);\n let stopKeyType;\n let stopDomainValues = {};\n let previousStopDomainValue;\n let previousStopDomainZoom;\n const isZoomFunction = functionType !== 'categorical' && options.value.property === undefined;\n const isPropertyFunction = !isZoomFunction;\n const isZoomAndPropertyFunction = getType(options.value.stops) === 'array' &&\n getType(options.value.stops[0]) === 'array' &&\n getType(options.value.stops[0][0]) === 'object';\n const errors = validateObject({\n key: options.key,\n value: options.value,\n valueSpec: options.styleSpec.function,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec,\n objectElementValidators: {\n stops: validateFunctionStops,\n default: validateFunctionDefault\n }\n });\n if (functionType === 'identity' && isZoomFunction) {\n errors.push(new ValidationError(options.key, options.value, 'missing required property \"property\"'));\n }\n if (functionType !== 'identity' && !options.value.stops) {\n errors.push(new ValidationError(options.key, options.value, 'missing required property \"stops\"'));\n }\n if (functionType === 'exponential' && options.valueSpec.expression && !supportsInterpolation(options.valueSpec)) {\n errors.push(new ValidationError(options.key, options.value, 'exponential functions not supported'));\n }\n if (options.styleSpec.$version >= 8) {\n if (isPropertyFunction && !supportsPropertyExpression(options.valueSpec)) {\n errors.push(new ValidationError(options.key, options.value, 'property functions not supported'));\n }\n else if (isZoomFunction && !supportsZoomExpression(options.valueSpec)) {\n errors.push(new ValidationError(options.key, options.value, 'zoom functions not supported'));\n }\n }\n if ((functionType === 'categorical' || isZoomAndPropertyFunction) && options.value.property === undefined) {\n errors.push(new ValidationError(options.key, options.value, '\"property\" property is required'));\n }\n return errors;\n function validateFunctionStops(options) {\n if (functionType === 'identity') {\n return [new ValidationError(options.key, options.value, 'identity function may not have a \"stops\" property')];\n }\n let errors = [];\n const value = options.value;\n errors = errors.concat(validateArray({\n key: options.key,\n value,\n valueSpec: options.valueSpec,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec,\n arrayElementValidator: validateFunctionStop\n }));\n if (getType(value) === 'array' && value.length === 0) {\n errors.push(new ValidationError(options.key, value, 'array must have at least one stop'));\n }\n return errors;\n }\n function validateFunctionStop(options) {\n let errors = [];\n const value = options.value;\n const key = options.key;\n if (getType(value) !== 'array') {\n return [new ValidationError(key, value, `array expected, ${getType(value)} found`)];\n }\n if (value.length !== 2) {\n return [new ValidationError(key, value, `array length 2 expected, length ${value.length} found`)];\n }\n if (isZoomAndPropertyFunction) {\n if (getType(value[0]) !== 'object') {\n return [new ValidationError(key, value, `object expected, ${getType(value[0])} found`)];\n }\n if (value[0].zoom === undefined) {\n return [new ValidationError(key, value, 'object stop key must have zoom')];\n }\n if (value[0].value === undefined) {\n return [new ValidationError(key, value, 'object stop key must have value')];\n }\n if (previousStopDomainZoom && previousStopDomainZoom > unbundle(value[0].zoom)) {\n return [new ValidationError(key, value[0].zoom, 'stop zoom values must appear in ascending order')];\n }\n if (unbundle(value[0].zoom) !== previousStopDomainZoom) {\n previousStopDomainZoom = unbundle(value[0].zoom);\n previousStopDomainValue = undefined;\n stopDomainValues = {};\n }\n errors = errors.concat(validateObject({\n key: `${key}[0]`,\n value: value[0],\n valueSpec: { zoom: {} },\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec,\n objectElementValidators: { zoom: validateNumber, value: validateStopDomainValue }\n }));\n }\n else {\n errors = errors.concat(validateStopDomainValue({\n key: `${key}[0]`,\n value: value[0],\n valueSpec: {},\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec\n }, value));\n }\n if (isExpression(deepUnbundle(value[1]))) {\n return errors.concat([new ValidationError(`${key}[1]`, value[1], 'expressions are not allowed in function stops.')]);\n }\n return errors.concat(options.validateSpec({\n key: `${key}[1]`,\n value: value[1],\n valueSpec: functionValueSpec,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec\n }));\n }\n function validateStopDomainValue(options, stop) {\n const type = getType(options.value);\n const value = unbundle(options.value);\n const reportValue = options.value !== null ? options.value : stop;\n if (!stopKeyType) {\n stopKeyType = type;\n }\n else if (type !== stopKeyType) {\n return [new ValidationError(options.key, reportValue, `${type} stop domain type must match previous stop domain type ${stopKeyType}`)];\n }\n if (type !== 'number' && type !== 'string' && type !== 'boolean') {\n return [new ValidationError(options.key, reportValue, 'stop domain value must be a number, string, or boolean')];\n }\n if (type !== 'number' && functionType !== 'categorical') {\n let message = `number expected, ${type} found`;\n if (supportsPropertyExpression(functionValueSpec) && functionType === undefined) {\n message += '\\nIf you intended to use a categorical function, specify `\"type\": \"categorical\"`.';\n }\n return [new ValidationError(options.key, reportValue, message)];\n }\n if (functionType === 'categorical' && type === 'number' && (!isFinite(value) || Math.floor(value) !== value)) {\n return [new ValidationError(options.key, reportValue, `integer expected, found ${value}`)];\n }\n if (functionType !== 'categorical' && type === 'number' && previousStopDomainValue !== undefined && value < previousStopDomainValue) {\n return [new ValidationError(options.key, reportValue, 'stop domain values must appear in ascending order')];\n }\n else {\n previousStopDomainValue = value;\n }\n if (functionType === 'categorical' && value in stopDomainValues) {\n return [new ValidationError(options.key, reportValue, 'stop domain values must be unique')];\n }\n else {\n stopDomainValues[value] = true;\n }\n return [];\n }\n function validateFunctionDefault(options) {\n return options.validateSpec({\n key: options.key,\n value: options.value,\n valueSpec: functionValueSpec,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec\n });\n }\n}\n\nfunction validateExpression(options) {\n const expression = (options.expressionContext === 'property' ? createPropertyExpression : createExpression)(deepUnbundle(options.value), options.valueSpec);\n if (expression.result === 'error') {\n return expression.value.map((error) => {\n return new ValidationError(`${options.key}${error.key}`, options.value, error.message);\n });\n }\n const expressionObj = expression.value.expression || expression.value._styleExpression.expression;\n if (options.expressionContext === 'property' && (options.propertyKey === 'text-font') &&\n !expressionObj.outputDefined()) {\n return [new ValidationError(options.key, options.value, `Invalid data expression for \"${options.propertyKey}\". Output values must be contained as literals within the expression.`)];\n }\n if (options.expressionContext === 'property' && options.propertyType === 'layout' &&\n (!isStateConstant(expressionObj))) {\n return [new ValidationError(options.key, options.value, '\"feature-state\" data expressions are not supported with layout properties.')];\n }\n if (options.expressionContext === 'filter' && !isStateConstant(expressionObj)) {\n return [new ValidationError(options.key, options.value, '\"feature-state\" data expressions are not supported with filters.')];\n }\n if (options.expressionContext && options.expressionContext.indexOf('cluster') === 0) {\n if (!isGlobalPropertyConstant(expressionObj, ['zoom', 'feature-state'])) {\n return [new ValidationError(options.key, options.value, '\"zoom\" and \"feature-state\" expressions are not supported with cluster properties.')];\n }\n if (options.expressionContext === 'cluster-initial' && !isFeatureConstant(expressionObj)) {\n return [new ValidationError(options.key, options.value, 'Feature data expressions are not supported with initial expression part of cluster properties.')];\n }\n }\n return [];\n}\n\nfunction validateBoolean(options) {\n const value = options.value;\n const key = options.key;\n const type = getType(value);\n if (type !== 'boolean') {\n return [new ValidationError(key, value, `boolean expected, ${type} found`)];\n }\n return [];\n}\n\nfunction validateColor(options) {\n const key = options.key;\n const value = options.value;\n const type = getType(value);\n if (type !== 'string') {\n return [new ValidationError(key, value, `color expected, ${type} found`)];\n }\n if (!Color.parse(String(value))) { // cast String object to string primitive\n return [new ValidationError(key, value, `color expected, \"${value}\" found`)];\n }\n return [];\n}\n\nfunction validateEnum(options) {\n const key = options.key;\n const value = options.value;\n const valueSpec = options.valueSpec;\n const errors = [];\n if (Array.isArray(valueSpec.values)) { // <=v7\n if (valueSpec.values.indexOf(unbundle(value)) === -1) {\n errors.push(new ValidationError(key, value, `expected one of [${valueSpec.values.join(', ')}], ${JSON.stringify(value)} found`));\n }\n }\n else { // >=v8\n if (Object.keys(valueSpec.values).indexOf(unbundle(value)) === -1) {\n errors.push(new ValidationError(key, value, `expected one of [${Object.keys(valueSpec.values).join(', ')}], ${JSON.stringify(value)} found`));\n }\n }\n return errors;\n}\n\nfunction validateFilter(options) {\n if (isExpressionFilter(deepUnbundle(options.value))) {\n return validateExpression(extendBy({}, options, {\n expressionContext: 'filter',\n valueSpec: { value: 'boolean' }\n }));\n }\n else {\n return validateNonExpressionFilter(options);\n }\n}\nfunction validateNonExpressionFilter(options) {\n const value = options.value;\n const key = options.key;\n if (getType(value) !== 'array') {\n return [new ValidationError(key, value, `array expected, ${getType(value)} found`)];\n }\n const styleSpec = options.styleSpec;\n let type;\n let errors = [];\n if (value.length < 1) {\n return [new ValidationError(key, value, 'filter array must have at least 1 element')];\n }\n errors = errors.concat(validateEnum({\n key: `${key}[0]`,\n value: value[0],\n valueSpec: styleSpec.filter_operator,\n style: options.style,\n styleSpec: options.styleSpec\n }));\n switch (unbundle(value[0])) {\n case '<':\n case '<=':\n case '>':\n case '>=':\n if (value.length >= 2 && unbundle(value[1]) === '$type') {\n errors.push(new ValidationError(key, value, `\"$type\" cannot be use with operator \"${value[0]}\"`));\n }\n /* falls through */\n case '==':\n case '!=':\n if (value.length !== 3) {\n errors.push(new ValidationError(key, value, `filter array for operator \"${value[0]}\" must have 3 elements`));\n }\n /* falls through */\n case 'in':\n case '!in':\n if (value.length >= 2) {\n type = getType(value[1]);\n if (type !== 'string') {\n errors.push(new ValidationError(`${key}[1]`, value[1], `string expected, ${type} found`));\n }\n }\n for (let i = 2; i < value.length; i++) {\n type = getType(value[i]);\n if (unbundle(value[1]) === '$type') {\n errors = errors.concat(validateEnum({\n key: `${key}[${i}]`,\n value: value[i],\n valueSpec: styleSpec.geometry_type,\n style: options.style,\n styleSpec: options.styleSpec\n }));\n }\n else if (type !== 'string' && type !== 'number' && type !== 'boolean') {\n errors.push(new ValidationError(`${key}[${i}]`, value[i], `string, number, or boolean expected, ${type} found`));\n }\n }\n break;\n case 'any':\n case 'all':\n case 'none':\n for (let i = 1; i < value.length; i++) {\n errors = errors.concat(validateNonExpressionFilter({\n key: `${key}[${i}]`,\n value: value[i],\n style: options.style,\n styleSpec: options.styleSpec\n }));\n }\n break;\n case 'has':\n case '!has':\n type = getType(value[1]);\n if (value.length !== 2) {\n errors.push(new ValidationError(key, value, `filter array for \"${value[0]}\" operator must have 2 elements`));\n }\n else if (type !== 'string') {\n errors.push(new ValidationError(`${key}[1]`, value[1], `string expected, ${type} found`));\n }\n break;\n case 'within':\n type = getType(value[1]);\n if (value.length !== 2) {\n errors.push(new ValidationError(key, value, `filter array for \"${value[0]}\" operator must have 2 elements`));\n }\n else if (type !== 'object') {\n errors.push(new ValidationError(`${key}[1]`, value[1], `object expected, ${type} found`));\n }\n break;\n }\n return errors;\n}\n\nfunction validateProperty(options, propertyType) {\n const key = options.key;\n const validateSpec = options.validateSpec;\n const style = options.style;\n const styleSpec = options.styleSpec;\n const value = options.value;\n const propertyKey = options.objectKey;\n const layerSpec = styleSpec[`${propertyType}_${options.layerType}`];\n if (!layerSpec)\n return [];\n const transitionMatch = propertyKey.match(/^(.*)-transition$/);\n if (propertyType === 'paint' && transitionMatch && layerSpec[transitionMatch[1]] && layerSpec[transitionMatch[1]].transition) {\n return validateSpec({\n key,\n value,\n valueSpec: styleSpec.transition,\n style,\n styleSpec\n });\n }\n const valueSpec = options.valueSpec || layerSpec[propertyKey];\n if (!valueSpec) {\n return [new ValidationError(key, value, `unknown property \"${propertyKey}\"`)];\n }\n let tokenMatch;\n if (getType(value) === 'string' && supportsPropertyExpression(valueSpec) && !valueSpec.tokens && (tokenMatch = /^{([^}]+)}$/.exec(value))) {\n return [new ValidationError(key, value, `\"${propertyKey}\" does not support interpolation syntax\\n` +\n `Use an identity property function instead: \\`{ \"type\": \"identity\", \"property\": ${JSON.stringify(tokenMatch[1])} }\\`.`)];\n }\n const errors = [];\n if (options.layerType === 'symbol') {\n if (propertyKey === 'text-field' && style && !style.glyphs) {\n errors.push(new ValidationError(key, value, 'use of \"text-field\" requires a style \"glyphs\" property'));\n }\n if (propertyKey === 'text-font' && isFunction(deepUnbundle(value)) && unbundle(value.type) === 'identity') {\n errors.push(new ValidationError(key, value, '\"text-font\" does not support identity functions'));\n }\n }\n return errors.concat(validateSpec({\n key: options.key,\n value,\n valueSpec,\n style,\n styleSpec,\n expressionContext: 'property',\n propertyType,\n propertyKey\n }));\n}\n\nfunction validatePaintProperty(options) {\n return validateProperty(options, 'paint');\n}\n\nfunction validateLayoutProperty(options) {\n return validateProperty(options, 'layout');\n}\n\nfunction validateLayer(options) {\n let errors = [];\n const layer = options.value;\n const key = options.key;\n const style = options.style;\n const styleSpec = options.styleSpec;\n if (!layer.type && !layer.ref) {\n errors.push(new ValidationError(key, layer, 'either \"type\" or \"ref\" is required'));\n }\n let type = unbundle(layer.type);\n const ref = unbundle(layer.ref);\n if (layer.id) {\n const layerId = unbundle(layer.id);\n for (let i = 0; i < options.arrayIndex; i++) {\n const otherLayer = style.layers[i];\n if (unbundle(otherLayer.id) === layerId) {\n errors.push(new ValidationError(key, layer.id, `duplicate layer id \"${layer.id}\", previously used at line ${otherLayer.id.__line__}`));\n }\n }\n }\n if ('ref' in layer) {\n ['type', 'source', 'source-layer', 'filter', 'layout'].forEach((p) => {\n if (p in layer) {\n errors.push(new ValidationError(key, layer[p], `\"${p}\" is prohibited for ref layers`));\n }\n });\n let parent;\n style.layers.forEach((layer) => {\n if (unbundle(layer.id) === ref)\n parent = layer;\n });\n if (!parent) {\n errors.push(new ValidationError(key, layer.ref, `ref layer \"${ref}\" not found`));\n }\n else if (parent.ref) {\n errors.push(new ValidationError(key, layer.ref, 'ref cannot reference another ref layer'));\n }\n else {\n type = unbundle(parent.type);\n }\n }\n else if (type !== 'background') {\n if (!layer.source) {\n errors.push(new ValidationError(key, layer, 'missing required property \"source\"'));\n }\n else {\n const source = style.sources && style.sources[layer.source];\n const sourceType = source && unbundle(source.type);\n if (!source) {\n errors.push(new ValidationError(key, layer.source, `source \"${layer.source}\" not found`));\n }\n else if (sourceType === 'vector' && type === 'raster') {\n errors.push(new ValidationError(key, layer.source, `layer \"${layer.id}\" requires a raster source`));\n }\n else if (sourceType !== 'raster-dem' && type === 'hillshade') {\n errors.push(new ValidationError(key, layer.source, `layer \"${layer.id}\" requires a raster-dem source`));\n }\n else if (sourceType === 'raster' && type !== 'raster') {\n errors.push(new ValidationError(key, layer.source, `layer \"${layer.id}\" requires a vector source`));\n }\n else if (sourceType === 'vector' && !layer['source-layer']) {\n errors.push(new ValidationError(key, layer, `layer \"${layer.id}\" must specify a \"source-layer\"`));\n }\n else if (sourceType === 'raster-dem' && type !== 'hillshade') {\n errors.push(new ValidationError(key, layer.source, 'raster-dem source can only be used with layer type \\'hillshade\\'.'));\n }\n else if (type === 'line' && layer.paint && layer.paint['line-gradient'] &&\n (sourceType !== 'geojson' || !source.lineMetrics)) {\n errors.push(new ValidationError(key, layer, `layer \"${layer.id}\" specifies a line-gradient, which requires a GeoJSON source with \\`lineMetrics\\` enabled.`));\n }\n }\n }\n errors = errors.concat(validateObject({\n key,\n value: layer,\n valueSpec: styleSpec.layer,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n objectElementValidators: {\n '*'() {\n return [];\n },\n // We don't want to enforce the spec's `\"requires\": true` for backward compatibility with refs;\n // the actual requirement is validated above. See https://github.com/mapbox/mapbox-gl-js/issues/5772.\n type() {\n return options.validateSpec({\n key: `${key}.type`,\n value: layer.type,\n valueSpec: styleSpec.layer.type,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n object: layer,\n objectKey: 'type'\n });\n },\n filter: validateFilter,\n layout(options) {\n return validateObject({\n layer,\n key: options.key,\n value: options.value,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n objectElementValidators: {\n '*'(options) {\n return validateLayoutProperty(extendBy({ layerType: type }, options));\n }\n }\n });\n },\n paint(options) {\n return validateObject({\n layer,\n key: options.key,\n value: options.value,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n objectElementValidators: {\n '*'(options) {\n return validatePaintProperty(extendBy({ layerType: type }, options));\n }\n }\n });\n }\n }\n }));\n return errors;\n}\n\nfunction validateString(options) {\n const value = options.value;\n const key = options.key;\n const type = getType(value);\n if (type !== 'string') {\n return [new ValidationError(key, value, `string expected, ${type} found`)];\n }\n return [];\n}\n\nfunction validateRasterDEMSource(options) {\n var _a;\n const sourceName = (_a = options.sourceName) !== null && _a !== void 0 ? _a : '';\n const rasterDEM = options.value;\n const styleSpec = options.styleSpec;\n const rasterDEMSpec = styleSpec.source_raster_dem;\n const style = options.style;\n let errors = [];\n const rootType = getType(rasterDEM);\n if (rasterDEM === undefined) {\n return errors;\n }\n else if (rootType !== 'object') {\n errors.push(new ValidationError('source_raster_dem', rasterDEM, `object expected, ${rootType} found`));\n return errors;\n }\n const encoding = unbundle(rasterDEM.encoding);\n const isCustomEncoding = encoding === 'custom';\n const customEncodingKeys = ['redFactor', 'greenFactor', 'blueFactor', 'baseShift'];\n const encodingName = options.value.encoding ? `\"${options.value.encoding}\"` : 'Default';\n for (const key in rasterDEM) {\n if (!isCustomEncoding && customEncodingKeys.includes(key)) {\n errors.push(new ValidationError(key, rasterDEM[key], `In \"${sourceName}\": \"${key}\" is only valid when \"encoding\" is set to \"custom\". ${encodingName} encoding found`));\n }\n else if (rasterDEMSpec[key]) {\n errors = errors.concat(options.validateSpec({\n key,\n value: rasterDEM[key],\n valueSpec: rasterDEMSpec[key],\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else {\n errors.push(new ValidationError(key, rasterDEM[key], `unknown property \"${key}\"`));\n }\n }\n return errors;\n}\n\nconst objectElementValidators = {\n promoteId: validatePromoteId\n};\nfunction validateSource(options) {\n const value = options.value;\n const key = options.key;\n const styleSpec = options.styleSpec;\n const style = options.style;\n const validateSpec = options.validateSpec;\n if (!value.type) {\n return [new ValidationError(key, value, '\"type\" is required')];\n }\n const type = unbundle(value.type);\n let errors;\n switch (type) {\n case 'vector':\n case 'raster':\n errors = validateObject({\n key,\n value,\n valueSpec: styleSpec[`source_${type.replace('-', '_')}`],\n style: options.style,\n styleSpec,\n objectElementValidators,\n validateSpec,\n });\n return errors;\n case 'raster-dem':\n errors = validateRasterDEMSource({\n sourceName: key,\n value,\n style: options.style,\n styleSpec,\n validateSpec,\n });\n return errors;\n case 'geojson':\n errors = validateObject({\n key,\n value,\n valueSpec: styleSpec.source_geojson,\n style,\n styleSpec,\n validateSpec,\n objectElementValidators\n });\n if (value.cluster) {\n for (const prop in value.clusterProperties) {\n const [operator, mapExpr] = value.clusterProperties[prop];\n const reduceExpr = typeof operator === 'string' ? [operator, ['accumulated'], ['get', prop]] : operator;\n errors.push(...validateExpression({\n key: `${key}.${prop}.map`,\n value: mapExpr,\n validateSpec,\n expressionContext: 'cluster-map'\n }));\n errors.push(...validateExpression({\n key: `${key}.${prop}.reduce`,\n value: reduceExpr,\n validateSpec,\n expressionContext: 'cluster-reduce'\n }));\n }\n }\n return errors;\n case 'video':\n return validateObject({\n key,\n value,\n valueSpec: styleSpec.source_video,\n style,\n validateSpec,\n styleSpec\n });\n case 'image':\n return validateObject({\n key,\n value,\n valueSpec: styleSpec.source_image,\n style,\n validateSpec,\n styleSpec\n });\n case 'canvas':\n return [new ValidationError(key, null, 'Please use runtime APIs to add canvas sources, rather than including them in stylesheets.', 'source.canvas')];\n default:\n return validateEnum({\n key: `${key}.type`,\n value: value.type,\n valueSpec: { values: ['vector', 'raster', 'raster-dem', 'geojson', 'video', 'image'] },\n style,\n validateSpec,\n styleSpec\n });\n }\n}\nfunction validatePromoteId({ key, value }) {\n if (getType(value) === 'string') {\n return validateString({ key, value });\n }\n else {\n const errors = [];\n for (const prop in value) {\n errors.push(...validateString({ key: `${key}.${prop}`, value: value[prop] }));\n }\n return errors;\n }\n}\n\nfunction validateLight(options) {\n const light = options.value;\n const styleSpec = options.styleSpec;\n const lightSpec = styleSpec.light;\n const style = options.style;\n let errors = [];\n const rootType = getType(light);\n if (light === undefined) {\n return errors;\n }\n else if (rootType !== 'object') {\n errors = errors.concat([new ValidationError('light', light, `object expected, ${rootType} found`)]);\n return errors;\n }\n for (const key in light) {\n const transitionMatch = key.match(/^(.*)-transition$/);\n if (transitionMatch && lightSpec[transitionMatch[1]] && lightSpec[transitionMatch[1]].transition) {\n errors = errors.concat(options.validateSpec({\n key,\n value: light[key],\n valueSpec: styleSpec.transition,\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else if (lightSpec[key]) {\n errors = errors.concat(options.validateSpec({\n key,\n value: light[key],\n valueSpec: lightSpec[key],\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else {\n errors = errors.concat([new ValidationError(key, light[key], `unknown property \"${key}\"`)]);\n }\n }\n return errors;\n}\n\nfunction validateTerrain(options) {\n const terrain = options.value;\n const styleSpec = options.styleSpec;\n const terrainSpec = styleSpec.terrain;\n const style = options.style;\n let errors = [];\n const rootType = getType(terrain);\n if (terrain === undefined) {\n return errors;\n }\n else if (rootType !== 'object') {\n errors = errors.concat([new ValidationError('terrain', terrain, `object expected, ${rootType} found`)]);\n return errors;\n }\n for (const key in terrain) {\n if (terrainSpec[key]) {\n errors = errors.concat(options.validateSpec({\n key,\n value: terrain[key],\n valueSpec: terrainSpec[key],\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else {\n errors = errors.concat([new ValidationError(key, terrain[key], `unknown property \"${key}\"`)]);\n }\n }\n return errors;\n}\n\nfunction validateFormatted(options) {\n if (validateString(options).length === 0) {\n return [];\n }\n return validateExpression(options);\n}\n\nfunction validateImage(options) {\n if (validateString(options).length === 0) {\n return [];\n }\n return validateExpression(options);\n}\n\nfunction validatePadding(options) {\n const key = options.key;\n const value = options.value;\n const type = getType(value);\n if (type === 'array') {\n if (value.length < 1 || value.length > 4) {\n return [new ValidationError(key, value, `padding requires 1 to 4 values; ${value.length} values found`)];\n }\n const arrayElementSpec = {\n type: 'number'\n };\n let errors = [];\n for (let i = 0; i < value.length; i++) {\n errors = errors.concat(options.validateSpec({\n key: `${key}[${i}]`,\n value: value[i],\n validateSpec: options.validateSpec,\n valueSpec: arrayElementSpec\n }));\n }\n return errors;\n }\n else {\n return validateNumber({\n key,\n value,\n valueSpec: {}\n });\n }\n}\n\nfunction validateVariableAnchorOffsetCollection(options) {\n const key = options.key;\n const value = options.value;\n const type = getType(value);\n const styleSpec = options.styleSpec;\n if (type !== 'array' || value.length < 1 || value.length % 2 !== 0) {\n return [new ValidationError(key, value, 'variableAnchorOffsetCollection requires a non-empty array of even length')];\n }\n let errors = [];\n for (let i = 0; i < value.length; i += 2) {\n // Elements in even positions should be values from text-anchor enum\n errors = errors.concat(validateEnum({\n key: `${key}[${i}]`,\n value: value[i],\n valueSpec: styleSpec['layout_symbol']['text-anchor']\n }));\n // Elements in odd positions should be points (2-element numeric arrays)\n errors = errors.concat(validateArray({\n key: `${key}[${i + 1}]`,\n value: value[i + 1],\n valueSpec: {\n length: 2,\n value: 'number'\n },\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec\n }));\n }\n return errors;\n}\n\nfunction validateSprite(options) {\n let errors = [];\n const sprite = options.value;\n const key = options.key;\n if (!Array.isArray(sprite)) {\n return validateString({\n key,\n value: sprite\n });\n }\n else {\n const allSpriteIds = [];\n const allSpriteURLs = [];\n for (const i in sprite) {\n if (sprite[i].id && allSpriteIds.includes(sprite[i].id))\n errors.push(new ValidationError(key, sprite, `all the sprites' ids must be unique, but ${sprite[i].id} is duplicated`));\n allSpriteIds.push(sprite[i].id);\n if (sprite[i].url && allSpriteURLs.includes(sprite[i].url))\n errors.push(new ValidationError(key, sprite, `all the sprites' URLs must be unique, but ${sprite[i].url} is duplicated`));\n allSpriteURLs.push(sprite[i].url);\n const pairSpec = {\n id: {\n type: 'string',\n required: true,\n },\n url: {\n type: 'string',\n required: true,\n }\n };\n errors = errors.concat(validateObject({\n key: `${key}[${i}]`,\n value: sprite[i],\n valueSpec: pairSpec,\n validateSpec: options.validateSpec,\n }));\n }\n return errors;\n }\n}\n\nconst VALIDATORS = {\n '*'() {\n return [];\n },\n 'array': validateArray,\n 'boolean': validateBoolean,\n 'number': validateNumber,\n 'color': validateColor,\n 'constants': validateConstants,\n 'enum': validateEnum,\n 'filter': validateFilter,\n 'function': validateFunction,\n 'layer': validateLayer,\n 'object': validateObject,\n 'source': validateSource,\n 'light': validateLight,\n 'terrain': validateTerrain,\n 'string': validateString,\n 'formatted': validateFormatted,\n 'resolvedImage': validateImage,\n 'padding': validatePadding,\n 'variableAnchorOffsetCollection': validateVariableAnchorOffsetCollection,\n 'sprite': validateSprite,\n};\n// Main recursive validation function. Tracks:\n//\n// - key: string representing location of validation in style tree. Used only\n// for more informative error reporting.\n// - value: current value from style being evaluated. May be anything from a\n// high level object that needs to be descended into deeper or a simple\n// scalar value.\n// - valueSpec: current spec being evaluated. Tracks value.\n// - styleSpec: current full spec being evaluated.\nfunction validate(options) {\n const value = options.value;\n const valueSpec = options.valueSpec;\n const styleSpec = options.styleSpec;\n options.validateSpec = validate;\n if (valueSpec.expression && isFunction(unbundle(value))) {\n return validateFunction(options);\n }\n else if (valueSpec.expression && isExpression(deepUnbundle(value))) {\n return validateExpression(options);\n }\n else if (valueSpec.type && VALIDATORS[valueSpec.type]) {\n return VALIDATORS[valueSpec.type](options);\n }\n else {\n const valid = validateObject(extendBy({}, options, {\n valueSpec: valueSpec.type ? styleSpec[valueSpec.type] : valueSpec\n }));\n return valid;\n }\n}\n\nfunction validateGlyphsUrl(options) {\n const value = options.value;\n const key = options.key;\n const errors = validateString(options);\n if (errors.length)\n return errors;\n if (value.indexOf('{fontstack}') === -1) {\n errors.push(new ValidationError(key, value, '\"glyphs\" url must include a \"{fontstack}\" token'));\n }\n if (value.indexOf('{range}') === -1) {\n errors.push(new ValidationError(key, value, '\"glyphs\" url must include a \"{range}\" token'));\n }\n return errors;\n}\n\n/**\n * Validate a MapLibre style against the style specification. This entrypoint,\n * `maplibre-gl-style-spec/lib/validate_style.min`, is designed to produce as\n * small a browserify bundle as possible by omitting unnecessary functionality\n * and legacy style specifications.\n *\n * @private\n * @param {Object} style The style to be validated.\n * @param {Object} [styleSpec] The style specification to validate against.\n * If omitted, the latest style spec is used.\n * @returns {Array}\n * @example\n * var validate = require('maplibre-gl-style-spec/lib/validate_style.min');\n * var errors = validate(style);\n */\nfunction validateStyleMin(style, styleSpec = v8Spec) {\n let errors = [];\n errors = errors.concat(validate({\n key: '',\n value: style,\n valueSpec: styleSpec.$root,\n styleSpec,\n style,\n validateSpec: validate,\n objectElementValidators: {\n glyphs: validateGlyphsUrl,\n '*'() {\n return [];\n }\n }\n }));\n if (style['constants']) {\n errors = errors.concat(validateConstants({\n key: 'constants',\n value: style['constants'],\n style,\n styleSpec,\n validateSpec: validate,\n }));\n }\n return sortErrors(errors);\n}\nvalidateStyleMin.source = wrapCleanErrors(injectValidateSpec(validateSource));\nvalidateStyleMin.sprite = wrapCleanErrors(injectValidateSpec(validateSprite));\nvalidateStyleMin.glyphs = wrapCleanErrors(injectValidateSpec(validateGlyphsUrl));\nvalidateStyleMin.light = wrapCleanErrors(injectValidateSpec(validateLight));\nvalidateStyleMin.terrain = wrapCleanErrors(injectValidateSpec(validateTerrain));\nvalidateStyleMin.layer = wrapCleanErrors(injectValidateSpec(validateLayer));\nvalidateStyleMin.filter = wrapCleanErrors(injectValidateSpec(validateFilter));\nvalidateStyleMin.paintProperty = wrapCleanErrors(injectValidateSpec(validatePaintProperty));\nvalidateStyleMin.layoutProperty = wrapCleanErrors(injectValidateSpec(validateLayoutProperty));\nfunction injectValidateSpec(validator) {\n return function (options) {\n return validator({\n ...options,\n validateSpec: validate,\n });\n };\n}\nfunction sortErrors(errors) {\n return [].concat(errors).sort((a, b) => {\n return a.line - b.line;\n });\n}\nfunction wrapCleanErrors(inner) {\n return function (...args) {\n return sortErrors(inner.apply(this, args));\n };\n}\n\nconst v8 = v8Spec;\nconst expression = {\n StyleExpression,\n StylePropertyFunction,\n ZoomConstantExpression,\n ZoomDependentExpression,\n createExpression,\n createPropertyExpression,\n isExpression,\n isExpressionFilter,\n isZoomExpression,\n normalizePropertyExpression,\n};\nconst styleFunction = {\n convertFunction,\n createFunction,\n isFunction\n};\nconst visit = { eachLayer, eachProperty, eachSource };\n\nexport { Color, ColorType, CompoundExpression, EvaluationContext, FormatExpression, Formatted, FormattedSection, FormattedType, Interpolate, Literal, NullType, Padding, ParsingError, ResolvedImage, Step, StyleExpression, StylePropertyFunction, ValidationError, VariableAnchorOffsetCollection, ZoomConstantExpression, ZoomDependentExpression, convertFilter, convertFunction, createExpression, createFunction, createPropertyExpression, derefLayers, diffStyles as diff, emptyStyle, expression, expressions, createFilter as featureFilter, styleFunction as function, groupByLayout, interpolateFactory, interpolate as interpolates, isExpression, isFunction, isZoomExpression, v8Spec as latest, normalizePropertyExpression, operations, supportsPropertyExpression, toString$1 as toString, typeOf, v8, validateStyleMin, visit };\n//# sourceMappingURL=index.mjs.map\n","import {validateStyleMin} from '@maplibre/maplibre-gl-style-spec';\nimport {ErrorEvent} from '../util/evented';\n\nimport type {Evented} from '../util/evented';\n\ntype ValidationError = {\n message: string;\n line: number;\n identifier?: string;\n};\n\nexport type Validator = (a: any) => ReadonlyArray;\n\ntype ValidateStyle = {\n source: Validator;\n sprite: Validator;\n glyphs: Validator;\n layer: Validator;\n light: Validator;\n terrain: Validator;\n filter: Validator;\n paintProperty: Validator;\n layoutProperty: Validator;\n (b: any, a?: any | null): ReadonlyArray;\n};\n\nexport const validateStyle = (validateStyleMin as unknown as ValidateStyle);\n\nexport const validateSource = validateStyle.source;\nexport const validateLight = validateStyle.light;\nexport const validateTerrain = validateStyle.terrain;\nexport const validateFilter = validateStyle.filter;\nexport const validatePaintProperty = validateStyle.paintProperty;\nexport const validateLayoutProperty = validateStyle.layoutProperty;\n\nexport function emitValidationErrors(\n emitter: Evented,\n errors?: ReadonlyArray<{\n message: string;\n identifier?: string;\n }> | null\n): boolean {\n let hasErrors = false;\n if (errors && errors.length) {\n for (const error of errors) {\n emitter.fire(new ErrorEvent(new Error(error.message)));\n hasErrors = true;\n }\n }\n return hasErrors;\n}\n","/*\nThis file was copied from https://github.com/mapbox/grid-index and was\nmigrated from JavaScript to TypeScript.\n\nCopyright (c) 2016, Mapbox\n\nPermission to use, copy, modify, and/or distribute this software for any purpose\nwith or without fee is hereby granted, provided that the above copyright notice\nand this permission notice appear in all copies.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND\nFITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS\nOF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER\nTORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF\nTHIS SOFTWARE.\n*/\n\nconst NUM_PARAMS = 3;\n\nexport type SerializedGrid = {\n buffer: ArrayBuffer;\n};\n\nexport class TransferableGridIndex {\n cells: number[][];\n arrayBuffer: ArrayBuffer;\n d: number;\n keys: number[];\n bboxes: number[];\n n: number;\n extent: number;\n padding: number;\n scale: any;\n uid: number;\n min: number;\n max: number;\n\n constructor(extent: number | ArrayBuffer, n?: number, padding?: number) {\n const cells = this.cells = [];\n\n if (extent instanceof ArrayBuffer) {\n this.arrayBuffer = extent;\n const array = new Int32Array(this.arrayBuffer);\n extent = array[0];\n n = array[1];\n padding = array[2];\n\n this.d = n + 2 * padding;\n for (let k = 0; k < this.d * this.d; k++) {\n const start = array[NUM_PARAMS + k];\n const end = array[NUM_PARAMS + k + 1];\n cells.push(start === end ? null : array.subarray(start, end));\n }\n const keysOffset = array[NUM_PARAMS + cells.length];\n const bboxesOffset = array[NUM_PARAMS + cells.length + 1];\n this.keys = array.subarray(keysOffset, bboxesOffset) as any as number[];\n this.bboxes = array.subarray(bboxesOffset) as any as number[];\n\n this.insert = this._insertReadonly;\n\n } else {\n this.d = n + 2 * padding;\n for (let i = 0; i < this.d * this.d; i++) {\n cells.push([]);\n }\n this.keys = [];\n this.bboxes = [];\n }\n\n this.n = n;\n this.extent = extent;\n this.padding = padding;\n this.scale = n / extent;\n this.uid = 0;\n\n const p = (padding / n) * extent;\n this.min = -p;\n this.max = extent + p;\n }\n\n insert(key: number, x1: number, y1: number, x2: number, y2: number) {\n this._forEachCell(x1, y1, x2, y2, this._insertCell, this.uid++, undefined, undefined);\n this.keys.push(key);\n this.bboxes.push(x1);\n this.bboxes.push(y1);\n this.bboxes.push(x2);\n this.bboxes.push(y2);\n }\n\n _insertReadonly() {\n throw new Error('Cannot insert into a GridIndex created from an ArrayBuffer.');\n }\n\n _insertCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) {\n this.cells[cellIndex].push(uid);\n }\n\n query(x1: number, y1: number, x2: number, y2: number, intersectionTest?: Function): number[] {\n const min = this.min;\n const max = this.max;\n if (x1 <= min && y1 <= min && max <= x2 && max <= y2 && !intersectionTest) {\n // We use `Array#slice` because `this.keys` may be a `Int32Array` and\n // some browsers (Safari and IE) do not support `TypedArray#slice`\n // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/TypedArray/slice#Browser_compatibility\n return Array.prototype.slice.call(this.keys);\n\n } else {\n const result = [];\n const seenUids = {};\n this._forEachCell(x1, y1, x2, y2, this._queryCell, result, seenUids, intersectionTest);\n return result;\n }\n }\n\n _queryCell(x1: number, y1: number, x2: number, y2:number, cellIndex:number, result, seenUids, intersectionTest: Function) {\n const cell = this.cells[cellIndex];\n if (cell !== null) {\n const keys = this.keys;\n const bboxes = this.bboxes;\n for (let u = 0; u < cell.length; u++) {\n const uid = cell[u];\n if (seenUids[uid] === undefined) {\n const offset = uid * 4;\n if (intersectionTest ?\n intersectionTest(bboxes[offset + 0], bboxes[offset + 1], bboxes[offset + 2], bboxes[offset + 3]) :\n ((x1 <= bboxes[offset + 2]) &&\n (y1 <= bboxes[offset + 3]) &&\n (x2 >= bboxes[offset + 0]) &&\n (y2 >= bboxes[offset + 1]))) {\n seenUids[uid] = true;\n result.push(keys[uid]);\n } else {\n seenUids[uid] = false;\n }\n }\n }\n }\n }\n\n _forEachCell(x1: number, y1: number, x2:number, y2:number, fn: Function, arg1, arg2, intersectionTest) {\n const cx1 = this._convertToCellCoord(x1);\n const cy1 = this._convertToCellCoord(y1);\n const cx2 = this._convertToCellCoord(x2);\n const cy2 = this._convertToCellCoord(y2);\n for (let x = cx1; x <= cx2; x++) {\n for (let y = cy1; y <= cy2; y++) {\n const cellIndex = this.d * y + x;\n if (intersectionTest && !intersectionTest(\n this._convertFromCellCoord(x),\n this._convertFromCellCoord(y),\n this._convertFromCellCoord(x + 1),\n this._convertFromCellCoord(y + 1))) continue;\n if (fn.call(this, x1, y1, x2, y2, cellIndex, arg1, arg2, intersectionTest)) return;\n }\n }\n }\n\n _convertFromCellCoord (x) {\n return (x - this.padding) / this.scale;\n }\n\n _convertToCellCoord(x) {\n return Math.max(0, Math.min(this.d - 1, Math.floor(x * this.scale) + this.padding));\n }\n\n toArrayBuffer(): ArrayBuffer {\n if (this.arrayBuffer) return this.arrayBuffer;\n\n const cells = this.cells;\n\n const metadataLength = NUM_PARAMS + this.cells.length + 1 + 1;\n let totalCellLength = 0;\n for (let i = 0; i < this.cells.length; i++) {\n totalCellLength += this.cells[i].length;\n }\n\n const array = new Int32Array(metadataLength + totalCellLength + this.keys.length + this.bboxes.length);\n array[0] = this.extent;\n array[1] = this.n;\n array[2] = this.padding;\n\n let offset = metadataLength;\n for (let k = 0; k < cells.length; k++) {\n const cell = cells[k];\n array[NUM_PARAMS + k] = offset;\n array.set(cell, offset);\n offset += cell.length;\n }\n\n array[NUM_PARAMS + cells.length] = offset;\n array.set(this.keys, offset);\n offset += this.keys.length;\n\n array[NUM_PARAMS + cells.length + 1] = offset;\n array.set(this.bboxes, offset);\n offset += this.bboxes.length;\n\n return array.buffer;\n }\n\n public static serialize(grid: TransferableGridIndex, transferables?: Array): SerializedGrid {\n const buffer = grid.toArrayBuffer();\n if (transferables) {\n transferables.push(buffer);\n }\n return {buffer};\n }\n\n public static deserialize(serialized: SerializedGrid): TransferableGridIndex {\n return new TransferableGridIndex(serialized.buffer);\n }\n}\n","import {TransferableGridIndex} from './transferable_grid_index';\nimport {Color, CompoundExpression, expressions, ResolvedImage, StylePropertyFunction,\n StyleExpression, ZoomDependentExpression, ZoomConstantExpression} from '@maplibre/maplibre-gl-style-spec';\n\nimport {AJAXError} from './ajax';\n\nimport type {Transferable} from '../types/transferable';\nimport {isImageBitmap} from './util';\n\ntype SerializedObject = {\n [_: string]: S;\n};\n\nexport type Serialized = null | void | boolean | number | string | Boolean | Number | String | Date | RegExp | ArrayBuffer | ArrayBufferView | ImageData | ImageBitmap | Blob | Array | SerializedObject;\n\ntype Registry = {\n [_: string]: {\n klass: {\n new (...args: any): any;\n deserialize?: (input: Serialized) => unknown;\n };\n omit: ReadonlyArray;\n shallow: ReadonlyArray;\n };\n};\n\n/**\n * Register options\n */\ntype RegisterOptions = {\n /**\n * List of properties to omit from serialization (e.g., cached/computed properties)\n */\n omit?: ReadonlyArray;\n /**\n * List of properties that should be serialized by a simple shallow copy, rather than by a recursive call to serialize().\n */\n shallow?: ReadonlyArray;\n};\n\nconst registry: Registry = {};\n\n/**\n * Register the given class as serializable.\n *\n * @param options - the registration options\n */\nexport function register(\n name: string,\n klass: {\n new (...args: any): T;\n },\n options: RegisterOptions = {}\n) {\n if (registry[name]) throw new Error(`${name} is already registered.`);\n ((Object.defineProperty as any))(klass, '_classRegistryKey', {\n value: name,\n writeable: false\n });\n registry[name] = {\n klass,\n omit: options.omit as ReadonlyArray || [],\n shallow: options.shallow as ReadonlyArray || []\n };\n}\n\nregister('Object', Object);\nregister('TransferableGridIndex', TransferableGridIndex);\n\nregister('Color', Color);\nregister('Error', Error);\nregister('AJAXError', AJAXError);\nregister('ResolvedImage', ResolvedImage);\n\nregister('StylePropertyFunction', StylePropertyFunction);\nregister('StyleExpression', StyleExpression, {omit: ['_evaluator']});\n\nregister('ZoomDependentExpression', ZoomDependentExpression);\nregister('ZoomConstantExpression', ZoomConstantExpression);\nregister('CompoundExpression', CompoundExpression, {omit: ['_evaluate']});\nfor (const name in expressions) {\n if ((expressions[name] as any)._classRegistryKey) continue;\n register(`Expression_${name}`, expressions[name]);\n}\n\nfunction isArrayBuffer(value: any): value is ArrayBuffer {\n return value && typeof ArrayBuffer !== 'undefined' &&\n (value instanceof ArrayBuffer || (value.constructor && value.constructor.name === 'ArrayBuffer'));\n}\n\n/**\n * Serialize the given object for transfer to or from a web worker.\n *\n * For non-builtin types, recursively serialize each property (possibly\n * omitting certain properties - see register()), and package the result along\n * with the constructor's `name` so that the appropriate constructor can be\n * looked up in `deserialize()`.\n *\n * If a `transferables` array is provided, add any transferable objects (i.e.,\n * any ArrayBuffers or ArrayBuffer views) to the list. (If a copy is needed,\n * this should happen in the client code, before using serialize().)\n */\nexport function serialize(input: unknown, transferables?: Array | null): Serialized {\n if (input === null ||\n input === undefined ||\n typeof input === 'boolean' ||\n typeof input === 'number' ||\n typeof input === 'string' ||\n input instanceof Boolean ||\n input instanceof Number ||\n input instanceof String ||\n input instanceof Date ||\n input instanceof RegExp ||\n input instanceof Blob) {\n return input;\n }\n\n if (isArrayBuffer(input)) {\n if (transferables) {\n transferables.push(input);\n }\n return input;\n }\n\n if (isImageBitmap(input)) {\n if (transferables) {\n transferables.push(input);\n }\n return input;\n }\n\n if (ArrayBuffer.isView(input)) {\n const view = input;\n if (transferables) {\n transferables.push(view.buffer);\n }\n return view;\n }\n\n if (input instanceof ImageData) {\n if (transferables) {\n transferables.push(input.data.buffer);\n }\n return input;\n }\n\n if (Array.isArray(input)) {\n const serialized: Array = [];\n for (const item of input) {\n serialized.push(serialize(item, transferables));\n }\n return serialized;\n }\n\n if (typeof input === 'object') {\n const klass = (input.constructor as any);\n const name = klass._classRegistryKey;\n if (!name) {\n throw new Error('can\\'t serialize object of unregistered class');\n }\n if (!registry[name]) throw new Error(`${name} is not registered.`);\n\n const properties: SerializedObject = klass.serialize ?\n // (Temporary workaround) allow a class to provide static\n // `serialize()` and `deserialize()` methods to bypass the generic\n // approach.\n // This temporary workaround lets us use the generic serialization\n // approach for objects whose members include instances of dynamic\n // StructArray types. Once we refactor StructArray to be static,\n // we can remove this complexity.\n (klass.serialize(input, transferables) as SerializedObject) : {};\n\n if (!klass.serialize) {\n for (const key in input) {\n if (!input.hasOwnProperty(key)) continue; // eslint-disable-line no-prototype-builtins\n if (registry[name].omit.indexOf(key) >= 0) continue;\n const property = input[key];\n properties[key] = registry[name].shallow.indexOf(key) >= 0 ?\n property :\n serialize(property, transferables);\n }\n if (input instanceof Error) {\n properties.message = input.message;\n }\n } else {\n if (transferables && properties === transferables[transferables.length - 1]) {\n throw new Error('statically serialized object won\\'t survive transfer of $name property');\n }\n }\n\n if (properties.$name) {\n throw new Error('$name property is reserved for worker serialization logic.');\n }\n if (name !== 'Object') {\n properties.$name = name;\n }\n\n return properties;\n }\n\n throw new Error(`can't serialize object of type ${typeof input}`);\n}\n\nexport function deserialize(input: Serialized): unknown {\n if (input === null ||\n input === undefined ||\n typeof input === 'boolean' ||\n typeof input === 'number' ||\n typeof input === 'string' ||\n input instanceof Boolean ||\n input instanceof Number ||\n input instanceof String ||\n input instanceof Date ||\n input instanceof RegExp ||\n input instanceof Blob ||\n isArrayBuffer(input) ||\n isImageBitmap(input) ||\n ArrayBuffer.isView(input) ||\n input instanceof ImageData) {\n return input;\n }\n\n if (Array.isArray(input)) {\n return input.map(deserialize);\n }\n\n if (typeof input === 'object') {\n const name = input.$name || 'Object';\n if (!registry[name]) {\n throw new Error(`can't deserialize unregistered class ${name}`);\n }\n const {klass} = registry[name];\n if (!klass) {\n throw new Error(`can't deserialize unregistered class ${name}`);\n }\n\n if (klass.deserialize) {\n return klass.deserialize(input);\n }\n\n const result = Object.create(klass.prototype);\n\n for (const key of Object.keys(input)) {\n if (key === '$name') continue;\n const value = (input as SerializedObject)[key];\n result[key] = registry[name].shallow.indexOf(key) >= 0 ? value : deserialize(value);\n }\n\n return result;\n }\n\n throw new Error(`can't deserialize object of type ${typeof input}`);\n}\n","export class ZoomHistory {\n lastZoom: number;\n lastFloorZoom: number;\n lastIntegerZoom: number;\n lastIntegerZoomTime: number;\n first: boolean;\n\n constructor() {\n this.first = true;\n }\n\n update(z: number, now: number) {\n const floorZ = Math.floor(z);\n\n if (this.first) {\n this.first = false;\n this.lastIntegerZoom = floorZ;\n this.lastIntegerZoomTime = 0;\n this.lastZoom = z;\n this.lastFloorZoom = floorZ;\n return true;\n }\n\n if (this.lastFloorZoom > floorZ) {\n this.lastIntegerZoom = floorZ + 1;\n this.lastIntegerZoomTime = now;\n } else if (this.lastFloorZoom < floorZ) {\n this.lastIntegerZoom = floorZ;\n this.lastIntegerZoomTime = now;\n }\n\n if (z !== this.lastZoom) {\n this.lastZoom = z;\n this.lastFloorZoom = floorZ;\n return true;\n }\n\n return false;\n }\n}\n","// The following table comes from .\n// Keep it synchronized with .\n\ntype UnicodeBlockLookup = {[key: string]: (char: number) => boolean};\n\nexport const unicodeBlockLookup: UnicodeBlockLookup = {\n // 'Basic Latin': (char) => char >= 0x0000 && char <= 0x007F,\n 'Latin-1 Supplement': (char) => char >= 0x0080 && char <= 0x00FF,\n // 'Latin Extended-A': (char) => char >= 0x0100 && char <= 0x017F,\n // 'Latin Extended-B': (char) => char >= 0x0180 && char <= 0x024F,\n // 'IPA Extensions': (char) => char >= 0x0250 && char <= 0x02AF,\n // 'Spacing Modifier Letters': (char) => char >= 0x02B0 && char <= 0x02FF,\n // 'Combining Diacritical Marks': (char) => char >= 0x0300 && char <= 0x036F,\n // 'Greek and Coptic': (char) => char >= 0x0370 && char <= 0x03FF,\n // 'Cyrillic': (char) => char >= 0x0400 && char <= 0x04FF,\n // 'Cyrillic Supplement': (char) => char >= 0x0500 && char <= 0x052F,\n // 'Armenian': (char) => char >= 0x0530 && char <= 0x058F,\n //'Hebrew': (char) => char >= 0x0590 && char <= 0x05FF,\n 'Arabic': (char) => char >= 0x0600 && char <= 0x06FF,\n //'Syriac': (char) => char >= 0x0700 && char <= 0x074F,\n 'Arabic Supplement': (char) => char >= 0x0750 && char <= 0x077F,\n // 'Thaana': (char) => char >= 0x0780 && char <= 0x07BF,\n // 'NKo': (char) => char >= 0x07C0 && char <= 0x07FF,\n // 'Samaritan': (char) => char >= 0x0800 && char <= 0x083F,\n // 'Mandaic': (char) => char >= 0x0840 && char <= 0x085F,\n // 'Syriac Supplement': (char) => char >= 0x0860 && char <= 0x086F,\n 'Arabic Extended-A': (char) => char >= 0x08A0 && char <= 0x08FF,\n // 'Devanagari': (char) => char >= 0x0900 && char <= 0x097F,\n // 'Bengali': (char) => char >= 0x0980 && char <= 0x09FF,\n // 'Gurmukhi': (char) => char >= 0x0A00 && char <= 0x0A7F,\n // 'Gujarati': (char) => char >= 0x0A80 && char <= 0x0AFF,\n // 'Oriya': (char) => char >= 0x0B00 && char <= 0x0B7F,\n // 'Tamil': (char) => char >= 0x0B80 && char <= 0x0BFF,\n // 'Telugu': (char) => char >= 0x0C00 && char <= 0x0C7F,\n // 'Kannada': (char) => char >= 0x0C80 && char <= 0x0CFF,\n // 'Malayalam': (char) => char >= 0x0D00 && char <= 0x0D7F,\n // 'Sinhala': (char) => char >= 0x0D80 && char <= 0x0DFF,\n // 'Thai': (char) => char >= 0x0E00 && char <= 0x0E7F,\n // 'Lao': (char) => char >= 0x0E80 && char <= 0x0EFF,\n // 'Tibetan': (char) => char >= 0x0F00 && char <= 0x0FFF,\n // 'Myanmar': (char) => char >= 0x1000 && char <= 0x109F,\n // 'Georgian': (char) => char >= 0x10A0 && char <= 0x10FF,\n 'Hangul Jamo': (char) => char >= 0x1100 && char <= 0x11FF,\n // 'Ethiopic': (char) => char >= 0x1200 && char <= 0x137F,\n // 'Ethiopic Supplement': (char) => char >= 0x1380 && char <= 0x139F,\n // 'Cherokee': (char) => char >= 0x13A0 && char <= 0x13FF,\n 'Unified Canadian Aboriginal Syllabics': (char) => char >= 0x1400 && char <= 0x167F,\n // 'Ogham': (char) => char >= 0x1680 && char <= 0x169F,\n // 'Runic': (char) => char >= 0x16A0 && char <= 0x16FF,\n // 'Tagalog': (char) => char >= 0x1700 && char <= 0x171F,\n // 'Hanunoo': (char) => char >= 0x1720 && char <= 0x173F,\n // 'Buhid': (char) => char >= 0x1740 && char <= 0x175F,\n // 'Tagbanwa': (char) => char >= 0x1760 && char <= 0x177F,\n 'Khmer': (char) => char >= 0x1780 && char <= 0x17FF,\n // 'Mongolian': (char) => char >= 0x1800 && char <= 0x18AF,\n 'Unified Canadian Aboriginal Syllabics Extended': (char) => char >= 0x18B0 && char <= 0x18FF,\n // 'Limbu': (char) => char >= 0x1900 && char <= 0x194F,\n // 'Tai Le': (char) => char >= 0x1950 && char <= 0x197F,\n // 'New Tai Lue': (char) => char >= 0x1980 && char <= 0x19DF,\n // 'Khmer Symbols': (char) => char >= 0x19E0 && char <= 0x19FF,\n // 'Buginese': (char) => char >= 0x1A00 && char <= 0x1A1F,\n // 'Tai Tham': (char) => char >= 0x1A20 && char <= 0x1AAF,\n // 'Combining Diacritical Marks Extended': (char) => char >= 0x1AB0 && char <= 0x1AFF,\n // 'Balinese': (char) => char >= 0x1B00 && char <= 0x1B7F,\n // 'Sundanese': (char) => char >= 0x1B80 && char <= 0x1BBF,\n // 'Batak': (char) => char >= 0x1BC0 && char <= 0x1BFF,\n // 'Lepcha': (char) => char >= 0x1C00 && char <= 0x1C4F,\n // 'Ol Chiki': (char) => char >= 0x1C50 && char <= 0x1C7F,\n // 'Cyrillic Extended-C': (char) => char >= 0x1C80 && char <= 0x1C8F,\n // 'Georgian Extended': (char) => char >= 0x1C90 && char <= 0x1CBF,\n // 'Sundanese Supplement': (char) => char >= 0x1CC0 && char <= 0x1CCF,\n // 'Vedic Extensions': (char) => char >= 0x1CD0 && char <= 0x1CFF,\n // 'Phonetic Extensions': (char) => char >= 0x1D00 && char <= 0x1D7F,\n // 'Phonetic Extensions Supplement': (char) => char >= 0x1D80 && char <= 0x1DBF,\n // 'Combining Diacritical Marks Supplement': (char) => char >= 0x1DC0 && char <= 0x1DFF,\n // 'Latin Extended Additional': (char) => char >= 0x1E00 && char <= 0x1EFF,\n // 'Greek Extended': (char) => char >= 0x1F00 && char <= 0x1FFF,\n 'General Punctuation': (char) => char >= 0x2000 && char <= 0x206F,\n // 'Superscripts and Subscripts': (char) => char >= 0x2070 && char <= 0x209F,\n // 'Currency Symbols': (char) => char >= 0x20A0 && char <= 0x20CF,\n // 'Combining Diacritical Marks for Symbols': (char) => char >= 0x20D0 && char <= 0x20FF,\n 'Letterlike Symbols': (char) => char >= 0x2100 && char <= 0x214F,\n 'Number Forms': (char) => char >= 0x2150 && char <= 0x218F,\n // 'Arrows': (char) => char >= 0x2190 && char <= 0x21FF,\n // 'Mathematical Operators': (char) => char >= 0x2200 && char <= 0x22FF,\n 'Miscellaneous Technical': (char) => char >= 0x2300 && char <= 0x23FF,\n 'Control Pictures': (char) => char >= 0x2400 && char <= 0x243F,\n 'Optical Character Recognition': (char) => char >= 0x2440 && char <= 0x245F,\n 'Enclosed Alphanumerics': (char) => char >= 0x2460 && char <= 0x24FF,\n // 'Box Drawing': (char) => char >= 0x2500 && char <= 0x257F,\n // 'Block Elements': (char) => char >= 0x2580 && char <= 0x259F,\n 'Geometric Shapes': (char) => char >= 0x25A0 && char <= 0x25FF,\n 'Miscellaneous Symbols': (char) => char >= 0x2600 && char <= 0x26FF,\n // 'Dingbats': (char) => char >= 0x2700 && char <= 0x27BF,\n // 'Miscellaneous Mathematical Symbols-A': (char) => char >= 0x27C0 && char <= 0x27EF,\n // 'Supplemental Arrows-A': (char) => char >= 0x27F0 && char <= 0x27FF,\n // 'Braille Patterns': (char) => char >= 0x2800 && char <= 0x28FF,\n // 'Supplemental Arrows-B': (char) => char >= 0x2900 && char <= 0x297F,\n // 'Miscellaneous Mathematical Symbols-B': (char) => char >= 0x2980 && char <= 0x29FF,\n // 'Supplemental Mathematical Operators': (char) => char >= 0x2A00 && char <= 0x2AFF,\n 'Miscellaneous Symbols and Arrows': (char) => char >= 0x2B00 && char <= 0x2BFF,\n // 'Glagolitic': (char) => char >= 0x2C00 && char <= 0x2C5F,\n // 'Latin Extended-C': (char) => char >= 0x2C60 && char <= 0x2C7F,\n // 'Coptic': (char) => char >= 0x2C80 && char <= 0x2CFF,\n // 'Georgian Supplement': (char) => char >= 0x2D00 && char <= 0x2D2F,\n // 'Tifinagh': (char) => char >= 0x2D30 && char <= 0x2D7F,\n // 'Ethiopic Extended': (char) => char >= 0x2D80 && char <= 0x2DDF,\n // 'Cyrillic Extended-A': (char) => char >= 0x2DE0 && char <= 0x2DFF,\n // 'Supplemental Punctuation': (char) => char >= 0x2E00 && char <= 0x2E7F,\n 'CJK Radicals Supplement': (char) => char >= 0x2E80 && char <= 0x2EFF,\n 'Kangxi Radicals': (char) => char >= 0x2F00 && char <= 0x2FDF,\n 'Ideographic Description Characters': (char) => char >= 0x2FF0 && char <= 0x2FFF,\n 'CJK Symbols and Punctuation': (char) => char >= 0x3000 && char <= 0x303F,\n 'Hiragana': (char) => char >= 0x3040 && char <= 0x309F,\n 'Katakana': (char) => char >= 0x30A0 && char <= 0x30FF,\n 'Bopomofo': (char) => char >= 0x3100 && char <= 0x312F,\n 'Hangul Compatibility Jamo': (char) => char >= 0x3130 && char <= 0x318F,\n 'Kanbun': (char) => char >= 0x3190 && char <= 0x319F,\n 'Bopomofo Extended': (char) => char >= 0x31A0 && char <= 0x31BF,\n 'CJK Strokes': (char) => char >= 0x31C0 && char <= 0x31EF,\n 'Katakana Phonetic Extensions': (char) => char >= 0x31F0 && char <= 0x31FF,\n 'Enclosed CJK Letters and Months': (char) => char >= 0x3200 && char <= 0x32FF,\n 'CJK Compatibility': (char) => char >= 0x3300 && char <= 0x33FF,\n 'CJK Unified Ideographs Extension A': (char) => char >= 0x3400 && char <= 0x4DBF,\n 'Yijing Hexagram Symbols': (char) => char >= 0x4DC0 && char <= 0x4DFF,\n 'CJK Unified Ideographs': (char) => char >= 0x4E00 && char <= 0x9FFF,\n 'Yi Syllables': (char) => char >= 0xA000 && char <= 0xA48F,\n 'Yi Radicals': (char) => char >= 0xA490 && char <= 0xA4CF,\n // 'Lisu': (char) => char >= 0xA4D0 && char <= 0xA4FF,\n // 'Vai': (char) => char >= 0xA500 && char <= 0xA63F,\n // 'Cyrillic Extended-B': (char) => char >= 0xA640 && char <= 0xA69F,\n // 'Bamum': (char) => char >= 0xA6A0 && char <= 0xA6FF,\n // 'Modifier Tone Letters': (char) => char >= 0xA700 && char <= 0xA71F,\n // 'Latin Extended-D': (char) => char >= 0xA720 && char <= 0xA7FF,\n // 'Syloti Nagri': (char) => char >= 0xA800 && char <= 0xA82F,\n // 'Common Indic Number Forms': (char) => char >= 0xA830 && char <= 0xA83F,\n // 'Phags-pa': (char) => char >= 0xA840 && char <= 0xA87F,\n // 'Saurashtra': (char) => char >= 0xA880 && char <= 0xA8DF,\n // 'Devanagari Extended': (char) => char >= 0xA8E0 && char <= 0xA8FF,\n // 'Kayah Li': (char) => char >= 0xA900 && char <= 0xA92F,\n // 'Rejang': (char) => char >= 0xA930 && char <= 0xA95F,\n 'Hangul Jamo Extended-A': (char) => char >= 0xA960 && char <= 0xA97F,\n // 'Javanese': (char) => char >= 0xA980 && char <= 0xA9DF,\n // 'Myanmar Extended-B': (char) => char >= 0xA9E0 && char <= 0xA9FF,\n // 'Cham': (char) => char >= 0xAA00 && char <= 0xAA5F,\n // 'Myanmar Extended-A': (char) => char >= 0xAA60 && char <= 0xAA7F,\n // 'Tai Viet': (char) => char >= 0xAA80 && char <= 0xAADF,\n // 'Meetei Mayek Extensions': (char) => char >= 0xAAE0 && char <= 0xAAFF,\n // 'Ethiopic Extended-A': (char) => char >= 0xAB00 && char <= 0xAB2F,\n // 'Latin Extended-E': (char) => char >= 0xAB30 && char <= 0xAB6F,\n // 'Cherokee Supplement': (char) => char >= 0xAB70 && char <= 0xABBF,\n // 'Meetei Mayek': (char) => char >= 0xABC0 && char <= 0xABFF,\n 'Hangul Syllables': (char) => char >= 0xAC00 && char <= 0xD7AF,\n 'Hangul Jamo Extended-B': (char) => char >= 0xD7B0 && char <= 0xD7FF,\n // 'High Surrogates': (char) => char >= 0xD800 && char <= 0xDB7F,\n // 'High Private Use Surrogates': (char) => char >= 0xDB80 && char <= 0xDBFF,\n // 'Low Surrogates': (char) => char >= 0xDC00 && char <= 0xDFFF,\n 'Private Use Area': (char) => char >= 0xE000 && char <= 0xF8FF,\n 'CJK Compatibility Ideographs': (char) => char >= 0xF900 && char <= 0xFAFF,\n // 'Alphabetic Presentation Forms': (char) => char >= 0xFB00 && char <= 0xFB4F,\n 'Arabic Presentation Forms-A': (char) => char >= 0xFB50 && char <= 0xFDFF,\n // 'Variation Selectors': (char) => char >= 0xFE00 && char <= 0xFE0F,\n 'Vertical Forms': (char) => char >= 0xFE10 && char <= 0xFE1F,\n // 'Combining Half Marks': (char) => char >= 0xFE20 && char <= 0xFE2F,\n 'CJK Compatibility Forms': (char) => char >= 0xFE30 && char <= 0xFE4F,\n 'Small Form Variants': (char) => char >= 0xFE50 && char <= 0xFE6F,\n 'Arabic Presentation Forms-B': (char) => char >= 0xFE70 && char <= 0xFEFF,\n 'Halfwidth and Fullwidth Forms': (char) => char >= 0xFF00 && char <= 0xFFEF\n // 'Specials': (char) => char >= 0xFFF0 && char <= 0xFFFF,\n // 'Linear B Syllabary': (char) => char >= 0x10000 && char <= 0x1007F,\n // 'Linear B Ideograms': (char) => char >= 0x10080 && char <= 0x100FF,\n // 'Aegean Numbers': (char) => char >= 0x10100 && char <= 0x1013F,\n // 'Ancient Greek Numbers': (char) => char >= 0x10140 && char <= 0x1018F,\n // 'Ancient Symbols': (char) => char >= 0x10190 && char <= 0x101CF,\n // 'Phaistos Disc': (char) => char >= 0x101D0 && char <= 0x101FF,\n // 'Lycian': (char) => char >= 0x10280 && char <= 0x1029F,\n // 'Carian': (char) => char >= 0x102A0 && char <= 0x102DF,\n // 'Coptic Epact Numbers': (char) => char >= 0x102E0 && char <= 0x102FF,\n // 'Old Italic': (char) => char >= 0x10300 && char <= 0x1032F,\n // 'Gothic': (char) => char >= 0x10330 && char <= 0x1034F,\n // 'Old Permic': (char) => char >= 0x10350 && char <= 0x1037F,\n // 'Ugaritic': (char) => char >= 0x10380 && char <= 0x1039F,\n // 'Old Persian': (char) => char >= 0x103A0 && char <= 0x103DF,\n // 'Deseret': (char) => char >= 0x10400 && char <= 0x1044F,\n // 'Shavian': (char) => char >= 0x10450 && char <= 0x1047F,\n // 'Osmanya': (char) => char >= 0x10480 && char <= 0x104AF,\n // 'Osage': (char) => char >= 0x104B0 && char <= 0x104FF,\n // 'Elbasan': (char) => char >= 0x10500 && char <= 0x1052F,\n // 'Caucasian Albanian': (char) => char >= 0x10530 && char <= 0x1056F,\n // 'Linear A': (char) => char >= 0x10600 && char <= 0x1077F,\n // 'Cypriot Syllabary': (char) => char >= 0x10800 && char <= 0x1083F,\n // 'Imperial Aramaic': (char) => char >= 0x10840 && char <= 0x1085F,\n // 'Palmyrene': (char) => char >= 0x10860 && char <= 0x1087F,\n // 'Nabataean': (char) => char >= 0x10880 && char <= 0x108AF,\n // 'Hatran': (char) => char >= 0x108E0 && char <= 0x108FF,\n // 'Phoenician': (char) => char >= 0x10900 && char <= 0x1091F,\n // 'Lydian': (char) => char >= 0x10920 && char <= 0x1093F,\n // 'Meroitic Hieroglyphs': (char) => char >= 0x10980 && char <= 0x1099F,\n // 'Meroitic Cursive': (char) => char >= 0x109A0 && char <= 0x109FF,\n // 'Kharoshthi': (char) => char >= 0x10A00 && char <= 0x10A5F,\n // 'Old South Arabian': (char) => char >= 0x10A60 && char <= 0x10A7F,\n // 'Old North Arabian': (char) => char >= 0x10A80 && char <= 0x10A9F,\n // 'Manichaean': (char) => char >= 0x10AC0 && char <= 0x10AFF,\n // 'Avestan': (char) => char >= 0x10B00 && char <= 0x10B3F,\n // 'Inscriptional Parthian': (char) => char >= 0x10B40 && char <= 0x10B5F,\n // 'Inscriptional Pahlavi': (char) => char >= 0x10B60 && char <= 0x10B7F,\n // 'Psalter Pahlavi': (char) => char >= 0x10B80 && char <= 0x10BAF,\n // 'Old Turkic': (char) => char >= 0x10C00 && char <= 0x10C4F,\n // 'Old Hungarian': (char) => char >= 0x10C80 && char <= 0x10CFF,\n // 'Hanifi Rohingya': (char) => char >= 0x10D00 && char <= 0x10D3F,\n // 'Rumi Numeral Symbols': (char) => char >= 0x10E60 && char <= 0x10E7F,\n // 'Old Sogdian': (char) => char >= 0x10F00 && char <= 0x10F2F,\n // 'Sogdian': (char) => char >= 0x10F30 && char <= 0x10F6F,\n // 'Elymaic': (char) => char >= 0x10FE0 && char <= 0x10FFF,\n // 'Brahmi': (char) => char >= 0x11000 && char <= 0x1107F,\n // 'Kaithi': (char) => char >= 0x11080 && char <= 0x110CF,\n // 'Sora Sompeng': (char) => char >= 0x110D0 && char <= 0x110FF,\n // 'Chakma': (char) => char >= 0x11100 && char <= 0x1114F,\n // 'Mahajani': (char) => char >= 0x11150 && char <= 0x1117F,\n // 'Sharada': (char) => char >= 0x11180 && char <= 0x111DF,\n // 'Sinhala Archaic Numbers': (char) => char >= 0x111E0 && char <= 0x111FF,\n // 'Khojki': (char) => char >= 0x11200 && char <= 0x1124F,\n // 'Multani': (char) => char >= 0x11280 && char <= 0x112AF,\n // 'Khudawadi': (char) => char >= 0x112B0 && char <= 0x112FF,\n // 'Grantha': (char) => char >= 0x11300 && char <= 0x1137F,\n // 'Newa': (char) => char >= 0x11400 && char <= 0x1147F,\n // 'Tirhuta': (char) => char >= 0x11480 && char <= 0x114DF,\n // 'Siddham': (char) => char >= 0x11580 && char <= 0x115FF,\n // 'Modi': (char) => char >= 0x11600 && char <= 0x1165F,\n // 'Mongolian Supplement': (char) => char >= 0x11660 && char <= 0x1167F,\n // 'Takri': (char) => char >= 0x11680 && char <= 0x116CF,\n // 'Ahom': (char) => char >= 0x11700 && char <= 0x1173F,\n // 'Dogra': (char) => char >= 0x11800 && char <= 0x1184F,\n // 'Warang Citi': (char) => char >= 0x118A0 && char <= 0x118FF,\n // 'Nandinagari': (char) => char >= 0x119A0 && char <= 0x119FF,\n // 'Zanabazar Square': (char) => char >= 0x11A00 && char <= 0x11A4F,\n // 'Soyombo': (char) => char >= 0x11A50 && char <= 0x11AAF,\n // 'Pau Cin Hau': (char) => char >= 0x11AC0 && char <= 0x11AFF,\n // 'Bhaiksuki': (char) => char >= 0x11C00 && char <= 0x11C6F,\n // 'Marchen': (char) => char >= 0x11C70 && char <= 0x11CBF,\n // 'Masaram Gondi': (char) => char >= 0x11D00 && char <= 0x11D5F,\n // 'Gunjala Gondi': (char) => char >= 0x11D60 && char <= 0x11DAF,\n // 'Makasar': (char) => char >= 0x11EE0 && char <= 0x11EFF,\n // 'Tamil Supplement': (char) => char >= 0x11FC0 && char <= 0x11FFF,\n // 'Cuneiform': (char) => char >= 0x12000 && char <= 0x123FF,\n // 'Cuneiform Numbers and Punctuation': (char) => char >= 0x12400 && char <= 0x1247F,\n // 'Early Dynastic Cuneiform': (char) => char >= 0x12480 && char <= 0x1254F,\n // 'Egyptian Hieroglyphs': (char) => char >= 0x13000 && char <= 0x1342F,\n // 'Egyptian Hieroglyph Format Controls': (char) => char >= 0x13430 && char <= 0x1343F,\n // 'Anatolian Hieroglyphs': (char) => char >= 0x14400 && char <= 0x1467F,\n // 'Bamum Supplement': (char) => char >= 0x16800 && char <= 0x16A3F,\n // 'Mro': (char) => char >= 0x16A40 && char <= 0x16A6F,\n // 'Bassa Vah': (char) => char >= 0x16AD0 && char <= 0x16AFF,\n // 'Pahawh Hmong': (char) => char >= 0x16B00 && char <= 0x16B8F,\n // 'Medefaidrin': (char) => char >= 0x16E40 && char <= 0x16E9F,\n // 'Miao': (char) => char >= 0x16F00 && char <= 0x16F9F,\n // 'Ideographic Symbols and Punctuation': (char) => char >= 0x16FE0 && char <= 0x16FFF,\n // 'Tangut': (char) => char >= 0x17000 && char <= 0x187FF,\n // 'Tangut Components': (char) => char >= 0x18800 && char <= 0x18AFF,\n // 'Kana Supplement': (char) => char >= 0x1B000 && char <= 0x1B0FF,\n // 'Kana Extended-A': (char) => char >= 0x1B100 && char <= 0x1B12F,\n // 'Small Kana Extension': (char) => char >= 0x1B130 && char <= 0x1B16F,\n // 'Nushu': (char) => char >= 0x1B170 && char <= 0x1B2FF,\n // 'Duployan': (char) => char >= 0x1BC00 && char <= 0x1BC9F,\n // 'Shorthand Format Controls': (char) => char >= 0x1BCA0 && char <= 0x1BCAF,\n // 'Byzantine Musical Symbols': (char) => char >= 0x1D000 && char <= 0x1D0FF,\n // 'Musical Symbols': (char) => char >= 0x1D100 && char <= 0x1D1FF,\n // 'Ancient Greek Musical Notation': (char) => char >= 0x1D200 && char <= 0x1D24F,\n // 'Mayan Numerals': (char) => char >= 0x1D2E0 && char <= 0x1D2FF,\n // 'Tai Xuan Jing Symbols': (char) => char >= 0x1D300 && char <= 0x1D35F,\n // 'Counting Rod Numerals': (char) => char >= 0x1D360 && char <= 0x1D37F,\n // 'Mathematical Alphanumeric Symbols': (char) => char >= 0x1D400 && char <= 0x1D7FF,\n // 'Sutton SignWriting': (char) => char >= 0x1D800 && char <= 0x1DAAF,\n // 'Glagolitic Supplement': (char) => char >= 0x1E000 && char <= 0x1E02F,\n // 'Nyiakeng Puachue Hmong': (char) => char >= 0x1E100 && char <= 0x1E14F,\n // 'Wancho': (char) => char >= 0x1E2C0 && char <= 0x1E2FF,\n // 'Mende Kikakui': (char) => char >= 0x1E800 && char <= 0x1E8DF,\n // 'Adlam': (char) => char >= 0x1E900 && char <= 0x1E95F,\n // 'Indic Siyaq Numbers': (char) => char >= 0x1EC70 && char <= 0x1ECBF,\n // 'Ottoman Siyaq Numbers': (char) => char >= 0x1ED00 && char <= 0x1ED4F,\n // 'Arabic Mathematical Alphabetic Symbols': (char) => char >= 0x1EE00 && char <= 0x1EEFF,\n // 'Mahjong Tiles': (char) => char >= 0x1F000 && char <= 0x1F02F,\n // 'Domino Tiles': (char) => char >= 0x1F030 && char <= 0x1F09F,\n // 'Playing Cards': (char) => char >= 0x1F0A0 && char <= 0x1F0FF,\n // 'Enclosed Alphanumeric Supplement': (char) => char >= 0x1F100 && char <= 0x1F1FF,\n // 'Enclosed Ideographic Supplement': (char) => char >= 0x1F200 && char <= 0x1F2FF,\n // 'Miscellaneous Symbols and Pictographs': (char) => char >= 0x1F300 && char <= 0x1F5FF,\n // 'Emoticons': (char) => char >= 0x1F600 && char <= 0x1F64F,\n // 'Ornamental Dingbats': (char) => char >= 0x1F650 && char <= 0x1F67F,\n // 'Transport and Map Symbols': (char) => char >= 0x1F680 && char <= 0x1F6FF,\n // 'Alchemical Symbols': (char) => char >= 0x1F700 && char <= 0x1F77F,\n // 'Geometric Shapes Extended': (char) => char >= 0x1F780 && char <= 0x1F7FF,\n // 'Supplemental Arrows-C': (char) => char >= 0x1F800 && char <= 0x1F8FF,\n // 'Supplemental Symbols and Pictographs': (char) => char >= 0x1F900 && char <= 0x1F9FF,\n // 'Chess Symbols': (char) => char >= 0x1FA00 && char <= 0x1FA6F,\n // 'Symbols and Pictographs Extended-A': (char) => char >= 0x1FA70 && char <= 0x1FAFF,\n // 'CJK Unified Ideographs Extension B': (char) => char >= 0x20000 && char <= 0x2A6DF,\n // 'CJK Unified Ideographs Extension C': (char) => char >= 0x2A700 && char <= 0x2B73F,\n // 'CJK Unified Ideographs Extension D': (char) => char >= 0x2B740 && char <= 0x2B81F,\n // 'CJK Unified Ideographs Extension E': (char) => char >= 0x2B820 && char <= 0x2CEAF,\n // 'CJK Unified Ideographs Extension F': (char) => char >= 0x2CEB0 && char <= 0x2EBEF,\n // 'CJK Compatibility Ideographs Supplement': (char) => char >= 0x2F800 && char <= 0x2FA1F,\n // 'Tags': (char) => char >= 0xE0000 && char <= 0xE007F,\n // 'Variation Selectors Supplement': (char) => char >= 0xE0100 && char <= 0xE01EF,\n // 'Supplementary Private Use Area-A': (char) => char >= 0xF0000 && char <= 0xFFFFF,\n // 'Supplementary Private Use Area-B': (char) => char >= 0x100000 && char <= 0x10FFFF,\n};\n","/* eslint-disable new-cap */\n\nimport {unicodeBlockLookup as isChar} from './is_char_in_unicode_block';\n\nexport function allowsIdeographicBreaking(chars: string) {\n for (const char of chars) {\n if (!charAllowsIdeographicBreaking(char.charCodeAt(0))) return false;\n }\n return true;\n}\n\nexport function allowsVerticalWritingMode(chars: string) {\n for (const char of chars) {\n if (charHasUprightVerticalOrientation(char.charCodeAt(0))) return true;\n }\n return false;\n}\n\nexport function allowsLetterSpacing(chars: string) {\n for (const char of chars) {\n if (!charAllowsLetterSpacing(char.charCodeAt(0))) return false;\n }\n return true;\n}\n\nexport function charAllowsLetterSpacing(char: number) {\n if (isChar['Arabic'](char)) return false;\n if (isChar['Arabic Supplement'](char)) return false;\n if (isChar['Arabic Extended-A'](char)) return false;\n if (isChar['Arabic Presentation Forms-A'](char)) return false;\n if (isChar['Arabic Presentation Forms-B'](char)) return false;\n\n return true;\n}\n\nexport function charAllowsIdeographicBreaking(char: number) {\n // Return early for characters outside all ideographic ranges.\n if (char < 0x2E80) return false;\n\n if (isChar['Bopomofo Extended'](char)) return true;\n if (isChar['Bopomofo'](char)) return true;\n if (isChar['CJK Compatibility Forms'](char)) return true;\n if (isChar['CJK Compatibility Ideographs'](char)) return true;\n if (isChar['CJK Compatibility'](char)) return true;\n if (isChar['CJK Radicals Supplement'](char)) return true;\n if (isChar['CJK Strokes'](char)) return true;\n if (isChar['CJK Symbols and Punctuation'](char)) return true;\n if (isChar['CJK Unified Ideographs Extension A'](char)) return true;\n if (isChar['CJK Unified Ideographs'](char)) return true;\n if (isChar['Enclosed CJK Letters and Months'](char)) return true;\n if (isChar['Halfwidth and Fullwidth Forms'](char)) return true;\n if (isChar['Hiragana'](char)) return true;\n if (isChar['Ideographic Description Characters'](char)) return true;\n if (isChar['Kangxi Radicals'](char)) return true;\n if (isChar['Katakana Phonetic Extensions'](char)) return true;\n if (isChar['Katakana'](char)) return true;\n if (isChar['Vertical Forms'](char)) return true;\n if (isChar['Yi Radicals'](char)) return true;\n if (isChar['Yi Syllables'](char)) return true;\n\n return false;\n}\n\n// The following logic comes from\n// .\n// Keep it synchronized with\n// .\n// The data file denotes with “U” or “Tu” any codepoint that may be drawn\n// upright in vertical text but does not distinguish between upright and\n// “neutral” characters.\n\n// Blocks in the Unicode supplementary planes are excluded from this module due\n// to .\n\n/**\n * Returns true if the given Unicode codepoint identifies a character with\n * upright orientation.\n *\n * A character has upright orientation if it is drawn upright (unrotated)\n * whether the line is oriented horizontally or vertically, even if both\n * adjacent characters can be rotated. For example, a Chinese character is\n * always drawn upright. An uprightly oriented character causes an adjacent\n * “neutral” character to be drawn upright as well.\n */\nexport function charHasUprightVerticalOrientation(char: number) {\n if (char === 0x02EA /* modifier letter yin departing tone mark */ ||\n char === 0x02EB /* modifier letter yang departing tone mark */) {\n return true;\n }\n\n // Return early for characters outside all ranges whose characters remain\n // upright in vertical writing mode.\n if (char < 0x1100) return false;\n\n if (isChar['Bopomofo Extended'](char)) return true;\n if (isChar['Bopomofo'](char)) return true;\n if (isChar['CJK Compatibility Forms'](char)) {\n if (!((char >= 0xFE49 /* dashed overline */ && char <= 0xFE4F) /* wavy low line */)) {\n return true;\n }\n }\n if (isChar['CJK Compatibility Ideographs'](char)) return true;\n if (isChar['CJK Compatibility'](char)) return true;\n if (isChar['CJK Radicals Supplement'](char)) return true;\n if (isChar['CJK Strokes'](char)) return true;\n if (isChar['CJK Symbols and Punctuation'](char)) {\n if (!((char >= 0x3008 /* left angle bracket */ && char <= 0x3011) /* right black lenticular bracket */) &&\n !((char >= 0x3014 /* left tortoise shell bracket */ && char <= 0x301F) /* low double prime quotation mark */) &&\n char !== 0x3030 /* wavy dash */) {\n return true;\n }\n }\n if (isChar['CJK Unified Ideographs Extension A'](char)) return true;\n if (isChar['CJK Unified Ideographs'](char)) return true;\n if (isChar['Enclosed CJK Letters and Months'](char)) return true;\n if (isChar['Hangul Compatibility Jamo'](char)) return true;\n if (isChar['Hangul Jamo Extended-A'](char)) return true;\n if (isChar['Hangul Jamo Extended-B'](char)) return true;\n if (isChar['Hangul Jamo'](char)) return true;\n if (isChar['Hangul Syllables'](char)) return true;\n if (isChar['Hiragana'](char)) return true;\n if (isChar['Ideographic Description Characters'](char)) return true;\n if (isChar['Kanbun'](char)) return true;\n if (isChar['Kangxi Radicals'](char)) return true;\n if (isChar['Katakana Phonetic Extensions'](char)) return true;\n if (isChar['Katakana'](char)) {\n if (char !== 0x30FC /* katakana-hiragana prolonged sound mark */) {\n return true;\n }\n }\n if (isChar['Halfwidth and Fullwidth Forms'](char)) {\n if (char !== 0xFF08 /* fullwidth left parenthesis */ &&\n char !== 0xFF09 /* fullwidth right parenthesis */ &&\n char !== 0xFF0D /* fullwidth hyphen-minus */ &&\n !((char >= 0xFF1A /* fullwidth colon */ && char <= 0xFF1E) /* fullwidth greater-than sign */) &&\n char !== 0xFF3B /* fullwidth left square bracket */ &&\n char !== 0xFF3D /* fullwidth right square bracket */ &&\n char !== 0xFF3F /* fullwidth low line */ &&\n !(char >= 0xFF5B /* fullwidth left curly bracket */ && char <= 0xFFDF) &&\n char !== 0xFFE3 /* fullwidth macron */ &&\n !(char >= 0xFFE8 /* halfwidth forms light vertical */ && char <= 0xFFEF)) {\n return true;\n }\n }\n if (isChar['Small Form Variants'](char)) {\n if (!((char >= 0xFE58 /* small em dash */ && char <= 0xFE5E) /* small right tortoise shell bracket */) &&\n !((char >= 0xFE63 /* small hyphen-minus */ && char <= 0xFE66) /* small equals sign */)) {\n return true;\n }\n }\n if (isChar['Unified Canadian Aboriginal Syllabics'](char)) return true;\n if (isChar['Unified Canadian Aboriginal Syllabics Extended'](char)) return true;\n if (isChar['Vertical Forms'](char)) return true;\n if (isChar['Yijing Hexagram Symbols'](char)) return true;\n if (isChar['Yi Syllables'](char)) return true;\n if (isChar['Yi Radicals'](char)) return true;\n\n return false;\n}\n\n/**\n * Returns true if the given Unicode codepoint identifies a character with\n * neutral orientation.\n *\n * A character has neutral orientation if it may be drawn rotated or unrotated\n * when the line is oriented vertically, depending on the orientation of the\n * adjacent characters. For example, along a verticlly oriented line, the vulgar\n * fraction ½ is drawn upright among Chinese characters but rotated among Latin\n * letters. A neutrally oriented character does not influence whether an\n * adjacent character is drawn upright or rotated.\n */\nexport function charHasNeutralVerticalOrientation(char: number) {\n if (isChar['Latin-1 Supplement'](char)) {\n if (char === 0x00A7 /* section sign */ ||\n char === 0x00A9 /* copyright sign */ ||\n char === 0x00AE /* registered sign */ ||\n char === 0x00B1 /* plus-minus sign */ ||\n char === 0x00BC /* vulgar fraction one quarter */ ||\n char === 0x00BD /* vulgar fraction one half */ ||\n char === 0x00BE /* vulgar fraction three quarters */ ||\n char === 0x00D7 /* multiplication sign */ ||\n char === 0x00F7 /* division sign */) {\n return true;\n }\n }\n if (isChar['General Punctuation'](char)) {\n if (char === 0x2016 /* double vertical line */ ||\n char === 0x2020 /* dagger */ ||\n char === 0x2021 /* double dagger */ ||\n char === 0x2030 /* per mille sign */ ||\n char === 0x2031 /* per ten thousand sign */ ||\n char === 0x203B /* reference mark */ ||\n char === 0x203C /* double exclamation mark */ ||\n char === 0x2042 /* asterism */ ||\n char === 0x2047 /* double question mark */ ||\n char === 0x2048 /* question exclamation mark */ ||\n char === 0x2049 /* exclamation question mark */ ||\n char === 0x2051 /* two asterisks aligned vertically */) {\n return true;\n }\n }\n if (isChar['Letterlike Symbols'](char)) return true;\n if (isChar['Number Forms'](char)) return true;\n if (isChar['Miscellaneous Technical'](char)) {\n if ((char >= 0x2300 /* diameter sign */ && char <= 0x2307 /* wavy line */) ||\n (char >= 0x230C /* bottom right crop */ && char <= 0x231F /* bottom right corner */) ||\n (char >= 0x2324 /* up arrowhead between two horizontal bars */ && char <= 0x2328 /* keyboard */) ||\n char === 0x232B /* erase to the left */ ||\n (char >= 0x237D /* shouldered open box */ && char <= 0x239A /* clear screen symbol */) ||\n (char >= 0x23BE /* dentistry symbol light vertical and top right */ && char <= 0x23CD /* square foot */) ||\n char === 0x23CF /* eject symbol */ ||\n (char >= 0x23D1 /* metrical breve */ && char <= 0x23DB /* fuse */) ||\n (char >= 0x23E2 /* white trapezium */ && char <= 0x23FF)) {\n return true;\n }\n }\n if (isChar['Control Pictures'](char) && char !== 0x2423 /* open box */) return true;\n if (isChar['Optical Character Recognition'](char)) return true;\n if (isChar['Enclosed Alphanumerics'](char)) return true;\n if (isChar['Geometric Shapes'](char)) return true;\n if (isChar['Miscellaneous Symbols'](char)) {\n if (!((char >= 0x261A /* black left pointing index */ && char <= 0x261F) /* white down pointing index */)) {\n return true;\n }\n }\n if (isChar['Miscellaneous Symbols and Arrows'](char)) {\n if ((char >= 0x2B12 /* square with top half black */ && char <= 0x2B2F /* white vertical ellipse */) ||\n (char >= 0x2B50 /* white medium star */ && char <= 0x2B59 /* heavy circled saltire */) ||\n (char >= 0x2BB8 /* upwards white arrow from bar with horizontal bar */ && char <= 0x2BEB)) {\n return true;\n }\n }\n if (isChar['CJK Symbols and Punctuation'](char)) return true;\n if (isChar['Katakana'](char)) return true;\n if (isChar['Private Use Area'](char)) return true;\n if (isChar['CJK Compatibility Forms'](char)) return true;\n if (isChar['Small Form Variants'](char)) return true;\n if (isChar['Halfwidth and Fullwidth Forms'](char)) return true;\n\n if (char === 0x221E /* infinity */ ||\n char === 0x2234 /* therefore */ ||\n char === 0x2235 /* because */ ||\n (char >= 0x2700 /* black safety scissors */ && char <= 0x2767 /* rotated floral heart bullet */) ||\n (char >= 0x2776 /* dingbat negative circled digit one */ && char <= 0x2793 /* dingbat negative circled sans-serif number ten */) ||\n char === 0xFFFC /* object replacement character */ ||\n char === 0xFFFD /* replacement character */) {\n return true;\n }\n\n return false;\n}\n\n/**\n * Returns true if the given Unicode codepoint identifies a character with\n * rotated orientation.\n *\n * A character has rotated orientation if it is drawn rotated when the line is\n * oriented vertically, even if both adjacent characters are upright. For\n * example, a Latin letter is drawn rotated along a vertical line. A rotated\n * character causes an adjacent “neutral” character to be drawn rotated as well.\n */\nexport function charHasRotatedVerticalOrientation(char: number) {\n return !(charHasUprightVerticalOrientation(char) ||\n charHasNeutralVerticalOrientation(char));\n}\n\nexport function charInComplexShapingScript(char: number) {\n return isChar['Arabic'](char) ||\n isChar['Arabic Supplement'](char) ||\n isChar['Arabic Extended-A'](char) ||\n isChar['Arabic Presentation Forms-A'](char) ||\n isChar['Arabic Presentation Forms-B'](char);\n}\n\nexport function charInRTLScript(char: number) {\n // Main blocks for Hebrew, Arabic, Thaana and other RTL scripts\n return (char >= 0x0590 && char <= 0x08FF) ||\n isChar['Arabic Presentation Forms-A'](char) ||\n isChar['Arabic Presentation Forms-B'](char);\n}\n\nexport function charInSupportedScript(char: number, canRenderRTL: boolean) {\n // This is a rough heuristic: whether we \"can render\" a script\n // actually depends on the properties of the font being used\n // and whether differences from the ideal rendering are considered\n // semantically significant.\n\n // Even in Latin script, we \"can't render\" combinations such as the fi\n // ligature, but we don't consider that semantically significant.\n if (!canRenderRTL && charInRTLScript(char)) {\n return false;\n }\n if ((char >= 0x0900 && char <= 0x0DFF) ||\n // Main blocks for Indic scripts and Sinhala\n (char >= 0x0F00 && char <= 0x109F) ||\n // Main blocks for Tibetan and Myanmar\n isChar['Khmer'](char)) {\n // These blocks cover common scripts that require\n // complex text shaping, based on unicode script metadata:\n // http://www.unicode.org/repos/cldr/trunk/common/properties/scriptMetadata.txt\n // where \"Web Rank <= 32\" \"Shaping Required = YES\"\n return false;\n }\n return true;\n}\n\nexport function stringContainsRTLText(chars: string): boolean {\n for (const char of chars) {\n if (charInRTLScript(char.charCodeAt(0))) {\n return true;\n }\n }\n return false;\n}\n\nexport function isStringInSupportedScript(chars: string, canRenderRTL: boolean) {\n for (const char of chars) {\n if (!charInSupportedScript(char.charCodeAt(0), canRenderRTL)) {\n return false;\n }\n }\n return true;\n}\n","import {getArrayBuffer} from '../util/ajax';\nimport {browser} from '../util/browser';\nimport {Event, Evented} from '../util/evented';\nimport {isWorker} from '../util/util';\n\nconst status = {\n unavailable: 'unavailable', // Not loaded\n deferred: 'deferred', // The plugin URL has been specified, but loading has been deferred\n loading: 'loading', // request in-flight\n loaded: 'loaded',\n error: 'error'\n};\n\nexport type PluginState = {\n pluginStatus: typeof status[keyof typeof status];\n pluginURL: string;\n};\n\n/**\n * An error callback\n */\ntype ErrorCallback = (error?: Error | null) => void;\ntype PluginStateSyncCallback = (state: PluginState) => void;\nlet _completionCallback = null;\n\n//Variables defining the current state of the plugin\nlet pluginStatus = status.unavailable;\nlet pluginURL = null;\n\nexport const triggerPluginCompletionEvent = function(error: Error | string) {\n // NetworkError's are not correctly reflected by the plugin status which prevents reloading plugin\n if (error && typeof error === 'string' && error.indexOf('NetworkError') > -1) {\n pluginStatus = status.error;\n }\n\n if (_completionCallback) {\n _completionCallback(error);\n }\n};\n\nfunction sendPluginStateToWorker() {\n evented.fire(new Event('pluginStateChange', {pluginStatus, pluginURL}));\n}\n\nexport const evented = new Evented();\n\nexport const getRTLTextPluginStatus = function () {\n return pluginStatus;\n};\n\nexport const registerForPluginStateChange = function(callback: PluginStateSyncCallback) {\n // Do an initial sync of the state\n callback({pluginStatus, pluginURL});\n // Listen for all future state changes\n evented.on('pluginStateChange', callback);\n return callback;\n};\n\nexport const clearRTLTextPlugin = function() {\n pluginStatus = status.unavailable;\n pluginURL = null;\n _completionCallback = null;\n};\n\nexport const setRTLTextPlugin = function(url: string, callback: ErrorCallback, deferred: boolean = false) {\n if (pluginStatus === status.deferred || pluginStatus === status.loading || pluginStatus === status.loaded) {\n throw new Error('setRTLTextPlugin cannot be called multiple times.');\n }\n pluginURL = browser.resolveURL(url);\n pluginStatus = status.deferred;\n _completionCallback = callback;\n sendPluginStateToWorker();\n\n //Start downloading the plugin immediately if not intending to lazy-load\n if (!deferred) {\n downloadRTLTextPlugin();\n }\n};\n\nexport const downloadRTLTextPlugin = function() {\n if (pluginStatus !== status.deferred || !pluginURL) {\n throw new Error('rtl-text-plugin cannot be downloaded unless a pluginURL is specified');\n }\n pluginStatus = status.loading;\n sendPluginStateToWorker();\n if (pluginURL) {\n getArrayBuffer({url: pluginURL}, (error) => {\n if (error) {\n triggerPluginCompletionEvent(error);\n } else {\n pluginStatus = status.loaded;\n sendPluginStateToWorker();\n }\n });\n }\n};\n\nexport const plugin: {\n applyArabicShaping: Function;\n processBidirectionalText: ((b: string, a: Array) => Array);\n processStyledBidirectionalText: ((c: string, b: Array, a: Array) => Array<[string, Array]>);\n isLoaded: () => boolean;\n isLoading: () => boolean;\n setState: (state: PluginState) => void;\n isParsed: () => boolean;\n getPluginURL: () => string;\n} = {\n applyArabicShaping: null,\n processBidirectionalText: null,\n processStyledBidirectionalText: null,\n isLoaded() {\n return pluginStatus === status.loaded || // Main Thread: loaded if the completion callback returned successfully\n plugin.applyArabicShaping != null; // Web-worker: loaded if the plugin functions have been compiled\n },\n isLoading() { // Main Thread Only: query the loading status, this function does not return the correct value in the worker context.\n return pluginStatus === status.loading;\n },\n setState(state: PluginState) { // Worker thread only: this tells the worker threads that the plugin is available on the Main thread\n if (!isWorker()) throw new Error('Cannot set the state of the rtl-text-plugin when not in the web-worker context');\n\n pluginStatus = state.pluginStatus;\n pluginURL = state.pluginURL;\n },\n isParsed(): boolean {\n if (!isWorker()) throw new Error('rtl-text-plugin is only parsed on the worker-threads');\n\n return plugin.applyArabicShaping != null &&\n plugin.processBidirectionalText != null &&\n plugin.processStyledBidirectionalText != null;\n },\n getPluginURL(): string {\n if (!isWorker()) throw new Error('rtl-text-plugin url can only be queried from the worker threads');\n return pluginURL;\n }\n};\n\nexport const lazyLoadRTLTextPlugin = function() {\n if (!plugin.isLoading() &&\n !plugin.isLoaded() &&\n getRTLTextPluginStatus() === 'deferred'\n ) {\n downloadRTLTextPlugin();\n }\n};\n","import {ZoomHistory} from './zoom_history';\nimport {isStringInSupportedScript} from '../util/script_detection';\nimport {plugin as rtlTextPlugin} from '../source/rtl_text_plugin';\n\nimport type {TransitionSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type CrossfadeParameters = {\n fromScale: number;\n toScale: number;\n t: number;\n};\n\n/**\n * @internal\n * A parameter that can be evaluated to a value\n */\nexport class EvaluationParameters {\n zoom: number;\n now: number;\n fadeDuration: number;\n zoomHistory: ZoomHistory;\n transition: TransitionSpecification;\n\n // \"options\" may also be another EvaluationParameters to copy, see CrossFadedProperty.possiblyEvaluate\n constructor(zoom: number, options?: any) {\n this.zoom = zoom;\n\n if (options) {\n this.now = options.now;\n this.fadeDuration = options.fadeDuration;\n this.zoomHistory = options.zoomHistory;\n this.transition = options.transition;\n } else {\n this.now = 0;\n this.fadeDuration = 0;\n this.zoomHistory = new ZoomHistory();\n this.transition = {};\n }\n }\n\n isSupportedScript(str: string): boolean {\n return isStringInSupportedScript(str, rtlTextPlugin.isLoaded());\n }\n\n crossFadingFactor() {\n if (this.fadeDuration === 0) {\n return 1;\n } else {\n return Math.min((this.now - this.zoomHistory.lastIntegerZoomTime) / this.fadeDuration, 1);\n }\n }\n\n getCrossfadeParameters(): CrossfadeParameters {\n const z = this.zoom;\n const fraction = z - Math.floor(z);\n const t = this.crossFadingFactor();\n\n return z > this.zoomHistory.lastIntegerZoom ?\n {fromScale: 2, toScale: 1, t: fraction + (1 - fraction) * t} :\n {fromScale: 0.5, toScale: 1, t: 1 - (1 - t) * fraction};\n }\n}\n","import {clone, extend, easeCubicInOut} from '../util/util';\nimport {interpolates, Color, StylePropertySpecification, normalizePropertyExpression,\n Feature,\n FeatureState,\n StylePropertyExpression,\n SourceExpression,\n CompositeExpression, TransitionSpecification,\n PropertyValueSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport {register} from '../util/web_worker_transfer';\nimport {EvaluationParameters} from './evaluation_parameters';\n\nimport {CanonicalTileID} from '../source/tile_id';\n\ntype TimePoint = number;\n\n/**\n * A from-to type\n */\nexport type CrossFaded = {\n to: T;\n from: T;\n};\n\n/**\n * @internal\n * Implementations of the `Property` interface:\n *\n * * Hold metadata about a property that's independent of any specific value: stuff like the type of the value,\n * the default value, etc. This comes from the style specification JSON.\n * * Define behavior that needs to be polymorphic across different properties: \"possibly evaluating\"\n * an input value (see below), and interpolating between two possibly-evaluted values.\n *\n * The type `T` is the fully-evaluated value type (e.g. `number`, `string`, `Color`).\n * The type `R` is the intermediate \"possibly evaluated\" value type. See below.\n *\n * There are two main implementations of the interface -- one for properties that allow data-driven values,\n * and one for properties that don't. There are a few \"special case\" implementations as well: one for properties\n * which cross-fade between two values rather than interpolating, one for `heatmap-color` and `line-gradient`,\n * and one for `light-position`.\n */\nexport interface Property {\n specification: StylePropertySpecification;\n possiblyEvaluate(\n value: PropertyValue,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): R;\n interpolate(a: R, b: R, t: number): R;\n}\n\n/**\n * @internal\n * `PropertyValue` represents the value part of a property key-value unit. It's used to represent both\n * paint and layout property values, and regardless of whether or not their property supports data-driven\n * expressions.\n *\n * `PropertyValue` stores the raw input value as seen in a style or a runtime styling API call, i.e. one of the\n * following:\n *\n * * A constant value of the type appropriate for the property\n * * A function which produces a value of that type (but functions are quasi-deprecated in favor of expressions)\n * * An expression which produces a value of that type\n * * \"undefined\"/\"not present\", in which case the property is assumed to take on its default value.\n *\n * In addition to storing the original input value, `PropertyValue` also stores a normalized representation,\n * effectively treating functions as if they are expressions, and constant or default values as if they are\n * (constant) expressions.\n */\nexport class PropertyValue {\n property: Property;\n value: PropertyValueSpecification | void;\n expression: StylePropertyExpression;\n\n constructor(property: Property, value: PropertyValueSpecification | void) {\n this.property = property;\n this.value = value;\n this.expression = normalizePropertyExpression(value === undefined ? property.specification.default : value, property.specification);\n }\n\n isDataDriven(): boolean {\n return this.expression.kind === 'source' || this.expression.kind === 'composite';\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): R {\n return this.property.possiblyEvaluate(this, parameters, canonical, availableImages);\n }\n}\n\nexport type TransitionParameters = {\n now: TimePoint;\n transition: TransitionSpecification;\n};\n\n/**\n * @internal\n * Paint properties are _transitionable_: they can change in a fluid manner, interpolating or cross-fading between\n * old and new value. The duration of the transition, and the delay before it begins, is configurable.\n *\n * `TransitionablePropertyValue` is a compositional class that stores both the property value and that transition\n * configuration.\n *\n * A `TransitionablePropertyValue` can calculate the next step in the evaluation chain for paint property values:\n * `TransitioningPropertyValue`.\n */\nclass TransitionablePropertyValue {\n property: Property;\n value: PropertyValue;\n transition: TransitionSpecification | void;\n\n constructor(property: Property) {\n this.property = property;\n this.value = new PropertyValue(property, undefined);\n }\n\n transitioned(parameters: TransitionParameters, prior: TransitioningPropertyValue): TransitioningPropertyValue {\n return new TransitioningPropertyValue(this.property, this.value, prior,\n extend({}, parameters.transition, this.transition), parameters.now);\n }\n\n untransitioned(): TransitioningPropertyValue {\n return new TransitioningPropertyValue(this.property, this.value, null, {}, 0);\n }\n}\n\n/**\n * @internal\n * `Transitionable` stores a map of all (property name, `TransitionablePropertyValue`) pairs for paint properties of a\n * given layer type. It can calculate the `TransitioningPropertyValue`s for all of them at once, producing a\n * `Transitioning` instance for the same set of properties.\n */\nexport class Transitionable {\n _properties: Properties;\n _values: {[K in keyof Props]: TransitionablePropertyValue};\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = (Object.create(properties.defaultTransitionablePropertyValues) as any);\n }\n\n getValue(name: S): PropertyValueSpecification | void {\n return clone(this._values[name].value.value);\n }\n\n setValue(name: S, value: PropertyValueSpecification | void) {\n if (!Object.prototype.hasOwnProperty.call(this._values, name)) {\n this._values[name] = new TransitionablePropertyValue(this._values[name].property);\n }\n // Note that we do not _remove_ an own property in the case where a value is being reset\n // to the default: the transition might still be non-default.\n this._values[name].value = new PropertyValue(this._values[name].property, value === null ? undefined : clone(value));\n }\n\n getTransition(name: S): TransitionSpecification | void {\n return clone(this._values[name].transition);\n }\n\n setTransition(name: S, value: TransitionSpecification | void) {\n if (!Object.prototype.hasOwnProperty.call(this._values, name)) {\n this._values[name] = new TransitionablePropertyValue(this._values[name].property);\n }\n this._values[name].transition = clone(value) || undefined;\n }\n\n serialize() {\n const result: any = {};\n for (const property of Object.keys(this._values)) {\n const value = this.getValue(property as keyof Props);\n if (value !== undefined) {\n result[property] = value;\n }\n\n const transition = this.getTransition(property as keyof Props);\n if (transition !== undefined) {\n result[`${property}-transition`] = transition;\n }\n }\n return result;\n }\n\n transitioned(parameters: TransitionParameters, prior: Transitioning): Transitioning {\n const result = new Transitioning(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].transitioned(parameters, prior._values[property]);\n }\n return result;\n }\n\n untransitioned(): Transitioning {\n const result = new Transitioning(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].untransitioned();\n }\n return result;\n }\n}\n\n/**\n * @internal\n * `TransitioningPropertyValue` implements the first of two intermediate steps in the evaluation chain of a paint\n * property value. In this step, transitions between old and new values are handled: as long as the transition is in\n * progress, `TransitioningPropertyValue` maintains a reference to the prior value, and interpolates between it and\n * the new value based on the current time and the configured transition duration and delay. The product is the next\n * step in the evaluation chain: the \"possibly evaluated\" result type `R`. See below for more on this concept.\n */\nclass TransitioningPropertyValue {\n property: Property;\n value: PropertyValue;\n prior: TransitioningPropertyValue;\n begin: TimePoint;\n end: TimePoint;\n\n constructor(property: Property,\n value: PropertyValue,\n prior: TransitioningPropertyValue,\n transition: TransitionSpecification,\n now: TimePoint) {\n this.property = property;\n this.value = value;\n this.begin = now + transition.delay || 0;\n this.end = this.begin + transition.duration || 0;\n if (property.specification.transition && (transition.delay || transition.duration)) {\n this.prior = prior;\n }\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical: CanonicalTileID,\n availableImages: Array\n ): R {\n const now = parameters.now || 0;\n const finalValue = this.value.possiblyEvaluate(parameters, canonical, availableImages);\n const prior = this.prior;\n if (!prior) {\n // No prior value.\n return finalValue;\n } else if (now > this.end) {\n // Transition from prior value is now complete.\n this.prior = null;\n return finalValue;\n } else if (this.value.isDataDriven()) {\n // Transitions to data-driven properties are not supported.\n // We snap immediately to the data-driven value so that, when we perform layout,\n // we see the data-driven function and can use it to populate vertex buffers.\n this.prior = null;\n return finalValue;\n } else if (now < this.begin) {\n // Transition hasn't started yet.\n return prior.possiblyEvaluate(parameters, canonical, availableImages);\n } else {\n // Interpolate between recursively-calculated prior value and final.\n const t = (now - this.begin) / (this.end - this.begin);\n return this.property.interpolate(prior.possiblyEvaluate(parameters, canonical, availableImages), finalValue, easeCubicInOut(t));\n }\n }\n}\n\n/**\n * @internal\n * `Transitioning` stores a map of all (property name, `TransitioningPropertyValue`) pairs for paint properties of a\n * given layer type. It can calculate the possibly-evaluated values for all of them at once, producing a\n * `PossiblyEvaluated` instance for the same set of properties.\n */\nexport class Transitioning {\n _properties: Properties;\n _values: {[K in keyof Props]: PossiblyEvaluatedPropertyValue};\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = (Object.create(properties.defaultTransitioningPropertyValues) as any);\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluated {\n const result = new PossiblyEvaluated(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].possiblyEvaluate(parameters, canonical, availableImages);\n }\n return result;\n }\n\n hasTransition() {\n for (const property of Object.keys(this._values)) {\n if (this._values[property].prior) {\n return true;\n }\n }\n return false;\n }\n}\n\n// ------- Layout -------\n\n/**\n * Because layout properties are not transitionable, they have a simpler representation and evaluation chain than\n * paint properties: `PropertyValue`s are possibly evaluated, producing possibly evaluated values, which are then\n * fully evaluated.\n *\n * `Layout` stores a map of all (property name, `PropertyValue`) pairs for layout properties of a\n * given layer type. It can calculate the possibly-evaluated values for all of them at once, producing a\n * `PossiblyEvaluated` instance for the same set of properties.\n */\nexport class Layout {\n _properties: Properties;\n _values: {[K in keyof Props]: PropertyValue>};\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = (Object.create(properties.defaultPropertyValues) as any);\n }\n\n hasValue(name: S) {\n return this._values[name].value !== undefined;\n }\n\n getValue(name: S) {\n return clone(this._values[name].value);\n }\n\n setValue(name: S, value: any) {\n this._values[name] = new PropertyValue(this._values[name].property, value === null ? undefined : clone(value)) as any;\n }\n\n serialize() {\n const result: any = {};\n for (const property of Object.keys(this._values)) {\n const value = this.getValue(property as keyof Props);\n if (value !== undefined) {\n result[property] = value;\n }\n }\n return result;\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluated {\n const result = new PossiblyEvaluated(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].possiblyEvaluate(parameters, canonical, availableImages);\n }\n return result;\n }\n}\n\n// ------- PossiblyEvaluated -------\n\n/**\n * \"Possibly evaluated value\" is an intermediate stage in the evaluation chain for both paint and layout property\n * values. The purpose of this stage is to optimize away unnecessary recalculations for data-driven properties. Code\n * which uses data-driven property values must assume that the value is dependent on feature data, and request that it\n * be evaluated for each feature. But when that property value is in fact a constant or camera function, the calculation\n * will not actually depend on the feature, and we can benefit from returning the prior result of having done the\n * evaluation once, ahead of time, in an intermediate step whose inputs are just the value and \"global\" parameters\n * such as current zoom level.\n *\n * `PossiblyEvaluatedValue` represents the three possible outcomes of this step: if the input value was a constant or\n * camera expression, then the \"possibly evaluated\" result is a constant value. Otherwise, the input value was either\n * a source or composite expression, and we must defer final evaluation until supplied a feature. We separate\n * the source and composite cases because they are handled differently when generating GL attributes, buffers, and\n * uniforms.\n *\n * Note that `PossiblyEvaluatedValue` (and `PossiblyEvaluatedPropertyValue`, below) are _not_ used for properties that\n * do not allow data-driven values. For such properties, we know that the \"possibly evaluated\" result is always a constant\n * scalar value. See below.\n */\ntype PossiblyEvaluatedValue = {\n kind: 'constant';\n value: T;\n} | SourceExpression | CompositeExpression;\n\n/**\n * @internal\n * `PossiblyEvaluatedPropertyValue` is used for data-driven paint and layout property values. It holds a\n * `PossiblyEvaluatedValue` and the `GlobalProperties` that were used to generate it. You're not allowed to supply\n * a different set of `GlobalProperties` when performing the final evaluation because they would be ignored in the\n * case where the input value was a constant or camera function.\n */\nexport class PossiblyEvaluatedPropertyValue {\n property: DataDrivenProperty;\n value: PossiblyEvaluatedValue;\n parameters: EvaluationParameters;\n\n constructor(property: DataDrivenProperty, value: PossiblyEvaluatedValue, parameters: EvaluationParameters) {\n this.property = property;\n this.value = value;\n this.parameters = parameters;\n }\n\n isConstant(): boolean {\n return this.value.kind === 'constant';\n }\n\n constantOr(value: T): T {\n if (this.value.kind === 'constant') {\n return this.value.value;\n } else {\n return value;\n }\n }\n\n evaluate(\n feature: Feature,\n featureState: FeatureState,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): T {\n return this.property.evaluate(this.value, this.parameters, feature, featureState, canonical, availableImages);\n }\n}\n\n/**\n * @internal\n * `PossiblyEvaluated` stores a map of all (property name, `R`) pairs for paint or layout properties of a\n * given layer type.\n */\nexport class PossiblyEvaluated {\n _properties: Properties;\n _values: PossibleEvaluatedProps;\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = Object.create(properties.defaultPossiblyEvaluatedValues);\n }\n\n get(name: S): PossibleEvaluatedProps[S] {\n return this._values[name];\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for properties that do not permit data-driven (source or composite) expressions.\n * This restriction allows us to declare statically that the result of possibly evaluating this kind of property\n * is in fact always the scalar type `T`, and can be used without further evaluating the value on a per-feature basis.\n */\nexport class DataConstantProperty implements Property {\n specification: StylePropertySpecification;\n\n constructor(specification: StylePropertySpecification) {\n this.specification = specification;\n }\n\n possiblyEvaluate(value: PropertyValue, parameters: EvaluationParameters): T {\n if (value.isDataDriven()) throw new Error('Value should not be data driven');\n return value.expression.evaluate(parameters);\n }\n\n interpolate(a: T, b: T, t: number): T {\n const interpolationType = this.specification.type as keyof typeof interpolates;\n const interpolationFn = interpolates[interpolationType] as ((from: T, to: T, t: number) => T) | undefined;\n if (interpolationFn) {\n return interpolationFn(a, b, t);\n } else {\n return a;\n }\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for properties that permit data-driven (source or composite) expressions.\n * The result of possibly evaluating this kind of property is `PossiblyEvaluatedPropertyValue`; obtaining\n * a scalar value `T` requires further evaluation on a per-feature basis.\n */\nexport class DataDrivenProperty implements Property> {\n specification: StylePropertySpecification;\n overrides: any;\n\n constructor(specification: StylePropertySpecification, overrides?: any) {\n this.specification = specification;\n this.overrides = overrides;\n }\n\n possiblyEvaluate(\n value: PropertyValue>,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluatedPropertyValue {\n if (value.expression.kind === 'constant' || value.expression.kind === 'camera') {\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: value.expression.evaluate(parameters, null, {}, canonical, availableImages)}, parameters);\n } else {\n return new PossiblyEvaluatedPropertyValue(this, value.expression, parameters);\n }\n }\n\n interpolate(\n a: PossiblyEvaluatedPropertyValue,\n b: PossiblyEvaluatedPropertyValue,\n t: number\n ): PossiblyEvaluatedPropertyValue {\n // If either possibly-evaluated value is non-constant, give up: we aren't able to interpolate data-driven values.\n if (a.value.kind !== 'constant' || b.value.kind !== 'constant') {\n return a;\n }\n\n // Special case hack solely for fill-outline-color. The undefined value is subsequently handled in\n // FillStyleLayer#recalculate, which sets fill-outline-color to the fill-color value if the former\n // is a PossiblyEvaluatedPropertyValue containing a constant undefined value. In addition to the\n // return value here, the other source of a PossiblyEvaluatedPropertyValue containing a constant\n // undefined value is the \"default value\" for fill-outline-color held in\n // `Properties#defaultPossiblyEvaluatedValues`, which serves as the prototype of\n // `PossiblyEvaluated#_values`.\n if (a.value.value === undefined || b.value.value === undefined) {\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: undefined}, a.parameters);\n }\n\n const interpolationType = this.specification.type as keyof typeof interpolates;\n const interpolationFn = interpolates[interpolationType] as ((from: T, to: T, t: number) => T) | undefined;\n if (interpolationFn) {\n const interpolatedValue = interpolationFn(a.value.value, b.value.value, t);\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: interpolatedValue}, a.parameters);\n } else {\n return a;\n }\n }\n\n evaluate(\n value: PossiblyEvaluatedValue,\n parameters: EvaluationParameters,\n feature: Feature,\n featureState: FeatureState,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): T {\n if (value.kind === 'constant') {\n return value.value;\n } else {\n return value.evaluate(parameters, feature, featureState, canonical, availableImages);\n }\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for data driven `line-pattern` which are transitioned by cross-fading\n * rather than interpolation.\n */\n\nexport class CrossFadedDataDrivenProperty extends DataDrivenProperty> {\n\n possiblyEvaluate(\n value: PropertyValue, PossiblyEvaluatedPropertyValue>>,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluatedPropertyValue> {\n if (value.value === undefined) {\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: undefined}, parameters);\n } else if (value.expression.kind === 'constant') {\n const evaluatedValue = value.expression.evaluate(parameters, null, {}, canonical, availableImages);\n const isImageExpression = value.property.specification.type as any === 'resolvedImage';\n const constantValue = isImageExpression && typeof evaluatedValue !== 'string' ? evaluatedValue.name : evaluatedValue;\n const constant = this._calculate(constantValue, constantValue, constantValue, parameters);\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: constant}, parameters);\n } else if (value.expression.kind === 'camera') {\n const cameraVal = this._calculate(\n value.expression.evaluate({zoom: parameters.zoom - 1.0}),\n value.expression.evaluate({zoom: parameters.zoom}),\n value.expression.evaluate({zoom: parameters.zoom + 1.0}),\n parameters);\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: cameraVal}, parameters);\n } else {\n // source or composite expression\n return new PossiblyEvaluatedPropertyValue(this, value.expression, parameters);\n }\n }\n\n evaluate(\n value: PossiblyEvaluatedValue>,\n globals: EvaluationParameters,\n feature: Feature,\n featureState: FeatureState,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): CrossFaded {\n if (value.kind === 'source') {\n const constant = value.evaluate(globals, feature, featureState, canonical, availableImages);\n return this._calculate(constant, constant, constant, globals);\n } else if (value.kind === 'composite') {\n return this._calculate(\n value.evaluate({zoom: Math.floor(globals.zoom) - 1.0}, feature, featureState),\n value.evaluate({zoom: Math.floor(globals.zoom)}, feature, featureState),\n value.evaluate({zoom: Math.floor(globals.zoom) + 1.0}, feature, featureState),\n globals);\n } else {\n return value.value;\n }\n }\n\n _calculate(min: T, mid: T, max: T, parameters: EvaluationParameters): CrossFaded {\n const z = parameters.zoom;\n return z > parameters.zoomHistory.lastIntegerZoom ? {from: min, to: mid} : {from: max, to: mid};\n }\n\n interpolate(a: PossiblyEvaluatedPropertyValue>): PossiblyEvaluatedPropertyValue> {\n return a;\n }\n}\n/**\n * @internal\n * An implementation of `Property` for `*-pattern` and `line-dasharray`, which are transitioned by cross-fading\n * rather than interpolation.\n */\nexport class CrossFadedProperty implements Property> {\n specification: StylePropertySpecification;\n\n constructor(specification: StylePropertySpecification) {\n this.specification = specification;\n }\n\n possiblyEvaluate(\n value: PropertyValue>,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): CrossFaded {\n if (value.value === undefined) {\n return undefined;\n } else if (value.expression.kind === 'constant') {\n const constant = value.expression.evaluate(parameters, null, {}, canonical, availableImages);\n return this._calculate(constant, constant, constant, parameters);\n } else {\n return this._calculate(\n value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom - 1.0), parameters)),\n value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom), parameters)),\n value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom + 1.0), parameters)),\n parameters);\n }\n }\n\n _calculate(min: T, mid: T, max: T, parameters: EvaluationParameters): CrossFaded {\n const z = parameters.zoom;\n return z > parameters.zoomHistory.lastIntegerZoom ? {from: min, to: mid} : {from: max, to: mid};\n }\n\n interpolate(a?: CrossFaded | null): CrossFaded {\n return a;\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for `heatmap-color` and `line-gradient`. Interpolation is a no-op, and\n * evaluation returns a boolean value in order to indicate its presence, but the real\n * evaluation happens in StyleLayer classes.\n */\n\nexport class ColorRampProperty implements Property {\n specification: StylePropertySpecification;\n\n constructor(specification: StylePropertySpecification) {\n this.specification = specification;\n }\n\n possiblyEvaluate(\n value: PropertyValue,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): boolean {\n return !!value.expression.evaluate(parameters, null, {}, canonical, availableImages);\n }\n\n interpolate(): boolean { return false; }\n}\n\n/**\n * @internal\n * `Properties` holds objects containing default values for the layout or paint property set of a given\n * layer type. These objects are immutable, and they are used as the prototypes for the `_values` members of\n * `Transitionable`, `Transitioning`, `Layout`, and `PossiblyEvaluated`. This allows these classes to avoid\n * doing work in the common case where a property has no explicit value set and should be considered to take\n * on the default value: using `for (const property of Object.keys(this._values))`, they can iterate over\n * only the _own_ properties of `_values`, skipping repeated calculation of transitions and possible/final\n * evaluations for defaults, the result of which will always be the same.\n */\nexport class Properties {\n properties: Props;\n defaultPropertyValues: {[K in keyof Props]: PropertyValue};\n defaultTransitionablePropertyValues: {[K in keyof Props]: TransitionablePropertyValue};\n defaultTransitioningPropertyValues: {[K in keyof Props]: TransitioningPropertyValue};\n defaultPossiblyEvaluatedValues: {[K in keyof Props]: PossiblyEvaluatedPropertyValue};\n overridableProperties: Array;\n\n constructor(properties: Props) {\n this.properties = properties;\n this.defaultPropertyValues = ({} as any);\n this.defaultTransitionablePropertyValues = ({} as any);\n this.defaultTransitioningPropertyValues = ({} as any);\n this.defaultPossiblyEvaluatedValues = ({} as any);\n this.overridableProperties = ([] as any);\n\n for (const property in properties) {\n const prop = properties[property] as any;\n if (prop.specification.overridable) {\n this.overridableProperties.push(property);\n }\n const defaultPropertyValue = this.defaultPropertyValues[property] =\n new PropertyValue(prop, undefined);\n const defaultTransitionablePropertyValue = this.defaultTransitionablePropertyValues[property] =\n new TransitionablePropertyValue(prop);\n this.defaultTransitioningPropertyValues[property] =\n defaultTransitionablePropertyValue.untransitioned();\n this.defaultPossiblyEvaluatedValues[property] =\n defaultPropertyValue.possiblyEvaluate({} as any);\n }\n }\n}\n\nregister('DataDrivenProperty', DataDrivenProperty);\nregister('DataConstantProperty', DataConstantProperty);\nregister('CrossFadedDataDrivenProperty', CrossFadedDataDrivenProperty);\nregister('CrossFadedProperty', CrossFadedProperty);\nregister('ColorRampProperty', ColorRampProperty);\n","import {filterObject} from '../util/util';\n\nimport {latest as styleSpec, supportsPropertyExpression} from '@maplibre/maplibre-gl-style-spec';\nimport {\n validateStyle,\n validateLayoutProperty,\n validatePaintProperty,\n emitValidationErrors\n} from './validate_style';\nimport {Evented} from '../util/evented';\nimport {Layout, Transitionable, Transitioning, Properties, PossiblyEvaluated, PossiblyEvaluatedPropertyValue} from './properties';\n\nimport type {Bucket} from '../data/bucket';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureFilter, FeatureState,\n LayerSpecification,\n FilterSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {TransitionParameters, PropertyValue} from './properties';\nimport {EvaluationParameters} from './evaluation_parameters';\nimport type {CrossfadeParameters} from './evaluation_parameters';\n\nimport type {Transform} from '../geo/transform';\nimport type {CustomLayerInterface} from './style_layer/custom_style_layer';\nimport type {Map} from '../ui/map';\nimport type {StyleSetterOptions} from './style';\nimport {mat4} from 'gl-matrix';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nconst TRANSITION_SUFFIX = '-transition';\n\n/**\n * A base class for style layers\n */\nexport abstract class StyleLayer extends Evented {\n id: string;\n metadata: unknown;\n type: LayerSpecification['type'] | CustomLayerInterface['type'];\n source: string;\n sourceLayer: string;\n minzoom: number;\n maxzoom: number;\n filter: FilterSpecification | void;\n visibility: 'visible' | 'none' | void;\n _crossfadeParameters: CrossfadeParameters;\n\n _unevaluatedLayout: Layout;\n readonly layout: unknown;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n readonly paint: unknown;\n\n _featureFilter: FeatureFilter;\n\n readonly onAdd: ((map: Map) => void);\n readonly onRemove: ((map: Map) => void);\n\n queryRadius?(bucket: Bucket): number;\n queryIntersectsFeature?(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number,\n pixelPosMatrix: mat4\n ): boolean | number;\n\n constructor(layer: LayerSpecification | CustomLayerInterface, properties: Readonly<{\n layout?: Properties;\n paint?: Properties;\n }>) {\n super();\n\n this.id = layer.id;\n this.type = layer.type;\n this._featureFilter = {filter: () => true, needGeometry: false};\n\n if (layer.type === 'custom') return;\n\n layer = (layer as any as LayerSpecification);\n\n this.metadata = layer.metadata;\n this.minzoom = layer.minzoom;\n this.maxzoom = layer.maxzoom;\n\n if (layer.type !== 'background') {\n this.source = layer.source;\n this.sourceLayer = layer['source-layer'];\n this.filter = layer.filter;\n }\n\n if (properties.layout) {\n this._unevaluatedLayout = new Layout(properties.layout);\n }\n\n if (properties.paint) {\n this._transitionablePaint = new Transitionable(properties.paint);\n\n for (const property in layer.paint) {\n this.setPaintProperty(property, layer.paint[property], {validate: false});\n }\n for (const property in layer.layout) {\n this.setLayoutProperty(property, layer.layout[property], {validate: false});\n }\n\n this._transitioningPaint = this._transitionablePaint.untransitioned();\n //$FlowFixMe\n this.paint = new PossiblyEvaluated(properties.paint);\n }\n }\n\n getCrossfadeParameters() {\n return this._crossfadeParameters;\n }\n\n getLayoutProperty(name: string) {\n if (name === 'visibility') {\n return this.visibility;\n }\n\n return this._unevaluatedLayout.getValue(name);\n }\n\n setLayoutProperty(name: string, value: any, options: StyleSetterOptions = {}) {\n if (value !== null && value !== undefined) {\n const key = `layers.${this.id}.layout.${name}`;\n if (this._validate(validateLayoutProperty, key, name, value, options)) {\n return;\n }\n }\n\n if (name === 'visibility') {\n this.visibility = value;\n return;\n }\n\n this._unevaluatedLayout.setValue(name, value);\n }\n\n getPaintProperty(name: string) {\n if (name.endsWith(TRANSITION_SUFFIX)) {\n return this._transitionablePaint.getTransition(name.slice(0, -TRANSITION_SUFFIX.length));\n } else {\n return this._transitionablePaint.getValue(name);\n }\n }\n\n setPaintProperty(name: string, value: unknown, options: StyleSetterOptions = {}) {\n if (value !== null && value !== undefined) {\n const key = `layers.${this.id}.paint.${name}`;\n if (this._validate(validatePaintProperty, key, name, value, options)) {\n return false;\n }\n }\n\n if (name.endsWith(TRANSITION_SUFFIX)) {\n this._transitionablePaint.setTransition(name.slice(0, -TRANSITION_SUFFIX.length), (value as any) || undefined);\n return false;\n } else {\n const transitionable = this._transitionablePaint._values[name];\n const isCrossFadedProperty = transitionable.property.specification['property-type'] === 'cross-faded-data-driven';\n const wasDataDriven = transitionable.value.isDataDriven();\n const oldValue = transitionable.value;\n\n this._transitionablePaint.setValue(name, value);\n this._handleSpecialPaintPropertyUpdate(name);\n\n const newValue = this._transitionablePaint._values[name].value;\n const isDataDriven = newValue.isDataDriven();\n\n // if a cross-faded value is changed, we need to make sure the new icons get added to each tile's iconAtlas\n // so a call to _updateLayer is necessary, and we return true from this function so it gets called in\n // Style#setPaintProperty\n return isDataDriven || wasDataDriven || isCrossFadedProperty || this._handleOverridablePaintPropertyUpdate(name, oldValue, newValue);\n }\n }\n\n _handleSpecialPaintPropertyUpdate(_: string) {\n // No-op; can be overridden by derived classes.\n }\n\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _handleOverridablePaintPropertyUpdate(name: string, oldValue: PropertyValue, newValue: PropertyValue): boolean {\n // No-op; can be overridden by derived classes.\n return false;\n }\n\n isHidden(zoom: number) {\n if (this.minzoom && zoom < this.minzoom) return true;\n if (this.maxzoom && zoom >= this.maxzoom) return true;\n return this.visibility === 'none';\n }\n\n updateTransitions(parameters: TransitionParameters) {\n this._transitioningPaint = this._transitionablePaint.transitioned(parameters, this._transitioningPaint);\n }\n\n hasTransition() {\n return this._transitioningPaint.hasTransition();\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n if (parameters.getCrossfadeParameters) {\n this._crossfadeParameters = parameters.getCrossfadeParameters();\n }\n\n if (this._unevaluatedLayout) {\n (this as any).layout = this._unevaluatedLayout.possiblyEvaluate(parameters, undefined, availableImages);\n }\n\n (this as any).paint = this._transitioningPaint.possiblyEvaluate(parameters, undefined, availableImages);\n }\n\n serialize(): LayerSpecification {\n const output: LayerSpecification = {\n 'id': this.id,\n 'type': this.type as LayerSpecification['type'],\n 'source': this.source,\n 'source-layer': this.sourceLayer,\n 'metadata': this.metadata,\n 'minzoom': this.minzoom,\n 'maxzoom': this.maxzoom,\n 'filter': this.filter as FilterSpecification,\n 'layout': this._unevaluatedLayout && this._unevaluatedLayout.serialize(),\n 'paint': this._transitionablePaint && this._transitionablePaint.serialize()\n };\n\n if (this.visibility) {\n output.layout = output.layout || {};\n output.layout.visibility = this.visibility;\n }\n\n return filterObject(output, (value, key) => {\n return value !== undefined &&\n !(key === 'layout' && !Object.keys(value).length) &&\n !(key === 'paint' && !Object.keys(value).length);\n });\n }\n\n _validate(validate: Function, key: string, name: string, value: unknown, options: StyleSetterOptions = {}) {\n if (options && options.validate === false) {\n return false;\n }\n return emitValidationErrors(this, validate.call(validateStyle, {\n key,\n layerType: this.type,\n objectKey: name,\n value,\n styleSpec,\n // Workaround for https://github.com/mapbox/mapbox-gl-js/issues/2407\n style: {glyphs: true, sprite: true}\n }));\n }\n\n is3D() {\n return false;\n }\n\n isTileClipped() {\n return false;\n }\n\n hasOffscreenPass() {\n return false;\n }\n\n resize() {\n // noop\n }\n\n isStateDependent() {\n for (const property in (this as any).paint._values) {\n const value = (this as any).paint.get(property);\n if (!(value instanceof PossiblyEvaluatedPropertyValue) || !supportsPropertyExpression(value.property.specification)) {\n continue;\n }\n\n if ((value.value.kind === 'source' || value.value.kind === 'composite') &&\n value.value.isStateDependent) {\n return true;\n }\n }\n return false;\n }\n}\n","// Note: all \"sizes\" are measured in bytes\n\nimport type {Transferable} from '../types/transferable';\n\n/**\n * @internal\n * A view type size\n */\nconst viewTypes = {\n 'Int8': Int8Array,\n 'Uint8': Uint8Array,\n 'Int16': Int16Array,\n 'Uint16': Uint16Array,\n 'Int32': Int32Array,\n 'Uint32': Uint32Array,\n 'Float32': Float32Array\n};\n\n/**\n * @internal\n * A view type size\n */\nexport type ViewType = keyof typeof viewTypes;\n\n/** @internal */\nclass Struct {\n _pos1: number;\n _pos2: number;\n _pos4: number;\n _pos8: number;\n readonly _structArray: StructArray;\n\n // The following properties are defined on the prototype of sub classes.\n size: number;\n\n /**\n * @param structArray - The StructArray the struct is stored in\n * @param index - The index of the struct in the StructArray.\n */\n constructor(structArray: StructArray, index: number) {\n (this as any)._structArray = structArray;\n this._pos1 = index * this.size;\n this._pos2 = this._pos1 / 2;\n this._pos4 = this._pos1 / 4;\n this._pos8 = this._pos1 / 8;\n }\n}\n\nconst DEFAULT_CAPACITY = 128;\nconst RESIZE_MULTIPLIER = 5;\n\n/**\n * @internal\n * A struct array memeber\n */\nexport type StructArrayMember = {\n name: string;\n type: ViewType;\n components: number;\n offset: number;\n};\n\nexport type StructArrayLayout = {\n members: Array;\n size: number;\n alignment: number;\n};\n\n/**\n * An array that can be desialized\n */\nexport type SerializedStructArray = {\n length: number;\n arrayBuffer: ArrayBuffer;\n};\n\n/**\n * @internal\n * `StructArray` provides an abstraction over `ArrayBuffer` and `TypedArray`\n * making it behave like an array of typed structs.\n *\n * Conceptually, a StructArray is comprised of elements, i.e., instances of its\n * associated struct type. Each particular struct type, together with an\n * alignment size, determines the memory layout of a StructArray whose elements\n * are of that type. Thus, for each such layout that we need, we have\n * a corresponding StructArrayLayout class, inheriting from StructArray and\n * implementing `emplaceBack()` and `_refreshViews()`.\n *\n * In some cases, where we need to access particular elements of a StructArray,\n * we implement a more specific subclass that inherits from one of the\n * StructArrayLayouts and adds a `get(i): T` accessor that returns a structured\n * object whose properties are proxies into the underlying memory space for the\n * i-th element. This affords the convience of working with (seemingly) plain\n * Javascript objects without the overhead of serializing/deserializing them\n * into ArrayBuffers for efficient web worker transfer.\n */\nabstract class StructArray {\n capacity: number;\n length: number;\n isTransferred: boolean;\n arrayBuffer: ArrayBuffer;\n uint8: Uint8Array;\n\n // The following properties are defined on the prototype.\n members: Array;\n bytesPerElement: number;\n abstract emplaceBack(...v: number[]);\n abstract emplace(i: number, ...v: number[]);\n\n constructor() {\n this.isTransferred = false;\n this.capacity = -1;\n this.resize(0);\n }\n\n /**\n * Serialize a StructArray instance. Serializes both the raw data and the\n * metadata needed to reconstruct the StructArray base class during\n * deserialization.\n */\n static serialize(array: StructArray, transferables?: Array): SerializedStructArray {\n\n array._trim();\n\n if (transferables) {\n array.isTransferred = true;\n transferables.push(array.arrayBuffer);\n }\n\n return {\n length: array.length,\n arrayBuffer: array.arrayBuffer,\n };\n }\n\n static deserialize(input: SerializedStructArray) {\n const structArray = Object.create(this.prototype);\n structArray.arrayBuffer = input.arrayBuffer;\n structArray.length = input.length;\n structArray.capacity = input.arrayBuffer.byteLength / structArray.bytesPerElement;\n structArray._refreshViews();\n return structArray;\n }\n\n /**\n * Resize the array to discard unused capacity.\n */\n _trim() {\n if (this.length !== this.capacity) {\n this.capacity = this.length;\n this.arrayBuffer = this.arrayBuffer.slice(0, this.length * this.bytesPerElement);\n this._refreshViews();\n }\n }\n\n /**\n * Resets the length of the array to 0 without de-allocating capcacity.\n */\n clear() {\n this.length = 0;\n }\n\n /**\n * Resize the array.\n * If `n` is greater than the current length then additional elements with undefined values are added.\n * If `n` is less than the current length then the array will be reduced to the first `n` elements.\n * @param n - The new size of the array.\n */\n resize(n: number) {\n this.reserve(n);\n this.length = n;\n }\n\n /**\n * Indicate a planned increase in size, so that any necessary allocation may\n * be done once, ahead of time.\n * @param n - The expected size of the array.\n */\n reserve(n: number) {\n if (n > this.capacity) {\n this.capacity = Math.max(n, Math.floor(this.capacity * RESIZE_MULTIPLIER), DEFAULT_CAPACITY);\n this.arrayBuffer = new ArrayBuffer(this.capacity * this.bytesPerElement);\n\n const oldUint8Array = this.uint8;\n this._refreshViews();\n if (oldUint8Array) this.uint8.set(oldUint8Array);\n }\n }\n\n /**\n * Create TypedArray views for the current ArrayBuffer.\n */\n _refreshViews() {\n throw new Error('_refreshViews() must be implemented by each concrete StructArray layout');\n }\n}\n\n/**\n * Given a list of member fields, create a full StructArrayLayout, in\n * particular calculating the correct byte offset for each field. This data\n * is used at build time to generate StructArrayLayout_*#emplaceBack() and\n * other accessors, and at runtime for binding vertex buffer attributes.\n */\nfunction createLayout(\n members: Array<{\n name: string;\n type: ViewType;\n readonly components?: number;\n }>,\n alignment: number = 1\n): StructArrayLayout {\n\n let offset = 0;\n let maxSize = 0;\n const layoutMembers = members.map((member) => {\n const typeSize = sizeOf(member.type);\n const memberOffset = offset = align(offset, Math.max(alignment, typeSize));\n const components = member.components || 1;\n\n maxSize = Math.max(maxSize, typeSize);\n offset += typeSize * components;\n\n return {\n name: member.name,\n type: member.type,\n components,\n offset: memberOffset,\n };\n });\n\n const size = align(offset, Math.max(maxSize, alignment));\n\n return {\n members: layoutMembers,\n size,\n alignment\n };\n}\n\nfunction sizeOf(type: ViewType): number {\n return viewTypes[type].BYTES_PER_ELEMENT;\n}\n\nfunction align(offset: number, size: number): number {\n return Math.ceil(offset / size) * size;\n}\n\nexport {StructArray, Struct, viewTypes, createLayout};\n","// This file is generated. Edit build/generate-struct-arrays.ts, then run `npm run codegen`.\n\nimport {Struct, StructArray} from '../util/struct_array';\nimport {register} from '../util/web_worker_transfer';\nimport Point from '@mapbox/point-geometry';\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n *\n */\nclass StructArrayLayout2i4 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1);\n }\n\n public emplace(i: number, v0: number, v1: number) {\n const o2 = i * 2;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n return i;\n }\n}\n\nStructArrayLayout2i4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout2i4', StructArrayLayout2i4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[3]\n *\n */\nclass StructArrayLayout3i6 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o2 = i * 3;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout3i6.prototype.bytesPerElement = 6;\nregister('StructArrayLayout3i6', StructArrayLayout3i6);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[4]\n *\n */\nclass StructArrayLayout4i8 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number) {\n const o2 = i * 4;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n return i;\n }\n}\n\nStructArrayLayout4i8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout4i8', StructArrayLayout4i8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Int16[4]\n *\n */\nclass StructArrayLayout2i4i12 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const o2 = i * 6;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n return i;\n }\n}\n\nStructArrayLayout2i4i12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout2i4i12', StructArrayLayout2i4i12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Uint8[4]\n *\n */\nclass StructArrayLayout2i4ub8 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const o2 = i * 4;\n const o1 = i * 8;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.uint8[o1 + 4] = v2;\n this.uint8[o1 + 5] = v3;\n this.uint8[o1 + 6] = v4;\n this.uint8[o1 + 7] = v5;\n return i;\n }\n}\n\nStructArrayLayout2i4ub8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout2i4ub8', StructArrayLayout2i4ub8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[2]\n *\n */\nclass StructArrayLayout2f8 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1);\n }\n\n public emplace(i: number, v0: number, v1: number) {\n const o4 = i * 2;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n return i;\n }\n}\n\nStructArrayLayout2f8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout2f8', StructArrayLayout2f8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[10]\n *\n */\nclass StructArrayLayout10ui20 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number) {\n const o2 = i * 10;\n this.uint16[o2 + 0] = v0;\n this.uint16[o2 + 1] = v1;\n this.uint16[o2 + 2] = v2;\n this.uint16[o2 + 3] = v3;\n this.uint16[o2 + 4] = v4;\n this.uint16[o2 + 5] = v5;\n this.uint16[o2 + 6] = v6;\n this.uint16[o2 + 7] = v7;\n this.uint16[o2 + 8] = v8;\n this.uint16[o2 + 9] = v9;\n return i;\n }\n}\n\nStructArrayLayout10ui20.prototype.bytesPerElement = 20;\nregister('StructArrayLayout10ui20', StructArrayLayout10ui20);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[4]\n * [8]: Uint16[4]\n * [16]: Int16[4]\n *\n */\nclass StructArrayLayout4i4ui4i24 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number) {\n const o2 = i * 12;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.uint16[o2 + 4] = v4;\n this.uint16[o2 + 5] = v5;\n this.uint16[o2 + 6] = v6;\n this.uint16[o2 + 7] = v7;\n this.int16[o2 + 8] = v8;\n this.int16[o2 + 9] = v9;\n this.int16[o2 + 10] = v10;\n this.int16[o2 + 11] = v11;\n return i;\n }\n}\n\nStructArrayLayout4i4ui4i24.prototype.bytesPerElement = 24;\nregister('StructArrayLayout4i4ui4i24', StructArrayLayout4i4ui4i24);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[3]\n *\n */\nclass StructArrayLayout3f12 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o4 = i * 3;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout3f12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout3f12', StructArrayLayout3f12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint32[1]\n *\n */\nclass StructArrayLayout1ul4 extends StructArray {\n uint8: Uint8Array;\n uint32: Uint32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0);\n }\n\n public emplace(i: number, v0: number) {\n const o4 = i * 1;\n this.uint32[o4 + 0] = v0;\n return i;\n }\n}\n\nStructArrayLayout1ul4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout1ul4', StructArrayLayout1ul4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[6]\n * [12]: Uint32[1]\n * [16]: Uint16[2]\n *\n */\nclass StructArrayLayout6i1ul2ui20 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint32: Uint32Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number) {\n const o2 = i * 10;\n const o4 = i * 5;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n this.uint32[o4 + 3] = v6;\n this.uint16[o2 + 8] = v7;\n this.uint16[o2 + 9] = v8;\n return i;\n }\n}\n\nStructArrayLayout6i1ul2ui20.prototype.bytesPerElement = 20;\nregister('StructArrayLayout6i1ul2ui20', StructArrayLayout6i1ul2ui20);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Int16[2]\n * [8]: Int16[2]\n *\n */\nclass StructArrayLayout2i2i2i12 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const o2 = i * 6;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n return i;\n }\n}\n\nStructArrayLayout2i2i2i12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout2i2i2i12', StructArrayLayout2i2i2i12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[2]\n * [8]: Float32[1]\n * [12]: Int16[2]\n *\n */\nclass StructArrayLayout2f1f2i16 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number) {\n const o4 = i * 4;\n const o2 = i * 8;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n this.int16[o2 + 6] = v3;\n this.int16[o2 + 7] = v4;\n return i;\n }\n}\n\nStructArrayLayout2f1f2i16.prototype.bytesPerElement = 16;\nregister('StructArrayLayout2f1f2i16', StructArrayLayout2f1f2i16);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint8[2]\n * [4]: Float32[2]\n *\n */\nclass StructArrayLayout2ub2f12 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number) {\n const o1 = i * 12;\n const o4 = i * 3;\n this.uint8[o1 + 0] = v0;\n this.uint8[o1 + 1] = v1;\n this.float32[o4 + 1] = v2;\n this.float32[o4 + 2] = v3;\n return i;\n }\n}\n\nStructArrayLayout2ub2f12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout2ub2f12', StructArrayLayout2ub2f12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[3]\n *\n */\nclass StructArrayLayout3ui6 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o2 = i * 3;\n this.uint16[o2 + 0] = v0;\n this.uint16[o2 + 1] = v1;\n this.uint16[o2 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout3ui6.prototype.bytesPerElement = 6;\nregister('StructArrayLayout3ui6', StructArrayLayout3ui6);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Uint16[2]\n * [8]: Uint32[3]\n * [20]: Uint16[3]\n * [28]: Float32[2]\n * [36]: Uint8[3]\n * [40]: Uint32[1]\n * [44]: Int16[1]\n *\n */\nclass StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint16: Uint16Array;\n uint32: Uint32Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number) {\n const o2 = i * 24;\n const o4 = i * 12;\n const o1 = i * 48;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.uint16[o2 + 2] = v2;\n this.uint16[o2 + 3] = v3;\n this.uint32[o4 + 2] = v4;\n this.uint32[o4 + 3] = v5;\n this.uint32[o4 + 4] = v6;\n this.uint16[o2 + 10] = v7;\n this.uint16[o2 + 11] = v8;\n this.uint16[o2 + 12] = v9;\n this.float32[o4 + 7] = v10;\n this.float32[o4 + 8] = v11;\n this.uint8[o1 + 36] = v12;\n this.uint8[o1 + 37] = v13;\n this.uint8[o1 + 38] = v14;\n this.uint32[o4 + 10] = v15;\n this.int16[o2 + 22] = v16;\n return i;\n }\n}\n\nStructArrayLayout2i2ui3ul3ui2f3ub1ul1i48.prototype.bytesPerElement = 48;\nregister('StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48', StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[8]\n * [16]: Uint16[15]\n * [48]: Uint32[1]\n * [52]: Float32[2]\n * [60]: Uint16[2]\n *\n */\nclass StructArrayLayout8i15ui1ul2f2ui64 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint16: Uint16Array;\n uint32: Uint32Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number, v17: number, v18: number, v19: number, v20: number, v21: number, v22: number, v23: number, v24: number, v25: number, v26: number, v27: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number, v17: number, v18: number, v19: number, v20: number, v21: number, v22: number, v23: number, v24: number, v25: number, v26: number, v27: number) {\n const o2 = i * 32;\n const o4 = i * 16;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n this.int16[o2 + 6] = v6;\n this.int16[o2 + 7] = v7;\n this.uint16[o2 + 8] = v8;\n this.uint16[o2 + 9] = v9;\n this.uint16[o2 + 10] = v10;\n this.uint16[o2 + 11] = v11;\n this.uint16[o2 + 12] = v12;\n this.uint16[o2 + 13] = v13;\n this.uint16[o2 + 14] = v14;\n this.uint16[o2 + 15] = v15;\n this.uint16[o2 + 16] = v16;\n this.uint16[o2 + 17] = v17;\n this.uint16[o2 + 18] = v18;\n this.uint16[o2 + 19] = v19;\n this.uint16[o2 + 20] = v20;\n this.uint16[o2 + 21] = v21;\n this.uint16[o2 + 22] = v22;\n this.uint32[o4 + 12] = v23;\n this.float32[o4 + 13] = v24;\n this.float32[o4 + 14] = v25;\n this.uint16[o2 + 30] = v26;\n this.uint16[o2 + 31] = v27;\n return i;\n }\n}\n\nStructArrayLayout8i15ui1ul2f2ui64.prototype.bytesPerElement = 64;\nregister('StructArrayLayout8i15ui1ul2f2ui64', StructArrayLayout8i15ui1ul2f2ui64);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[1]\n *\n */\nclass StructArrayLayout1f4 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0);\n }\n\n public emplace(i: number, v0: number) {\n const o4 = i * 1;\n this.float32[o4 + 0] = v0;\n return i;\n }\n}\n\nStructArrayLayout1f4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout1f4', StructArrayLayout1f4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[1]\n * [4]: Float32[2]\n *\n */\nclass StructArrayLayout1ui2f12 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o2 = i * 6;\n const o4 = i * 3;\n this.uint16[o2 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout1ui2f12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout1ui2f12', StructArrayLayout1ui2f12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint32[1]\n * [4]: Uint16[2]\n *\n */\nclass StructArrayLayout1ul2ui8 extends StructArray {\n uint8: Uint8Array;\n uint32: Uint32Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o4 = i * 2;\n const o2 = i * 4;\n this.uint32[o4 + 0] = v0;\n this.uint16[o2 + 2] = v1;\n this.uint16[o2 + 3] = v2;\n return i;\n }\n}\n\nStructArrayLayout1ul2ui8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout1ul2ui8', StructArrayLayout1ul2ui8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[2]\n *\n */\nclass StructArrayLayout2ui4 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1);\n }\n\n public emplace(i: number, v0: number, v1: number) {\n const o2 = i * 2;\n this.uint16[o2 + 0] = v0;\n this.uint16[o2 + 1] = v1;\n return i;\n }\n}\n\nStructArrayLayout2ui4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout2ui4', StructArrayLayout2ui4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[1]\n *\n */\nclass StructArrayLayout1ui2 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0);\n }\n\n public emplace(i: number, v0: number) {\n const o2 = i * 1;\n this.uint16[o2 + 0] = v0;\n return i;\n }\n}\n\nStructArrayLayout1ui2.prototype.bytesPerElement = 2;\nregister('StructArrayLayout1ui2', StructArrayLayout1ui2);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[4]\n *\n */\nclass StructArrayLayout4f16 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number) {\n const o4 = i * 4;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n this.float32[o4 + 3] = v3;\n return i;\n }\n}\n\nStructArrayLayout4f16.prototype.bytesPerElement = 16;\nregister('StructArrayLayout4f16', StructArrayLayout4f16);\n\n/** @internal */\nclass CollisionBoxStruct extends Struct {\n _structArray: CollisionBoxArray;\n get anchorPointX() { return this._structArray.int16[this._pos2 + 0]; }\n get anchorPointY() { return this._structArray.int16[this._pos2 + 1]; }\n get x1() { return this._structArray.int16[this._pos2 + 2]; }\n get y1() { return this._structArray.int16[this._pos2 + 3]; }\n get x2() { return this._structArray.int16[this._pos2 + 4]; }\n get y2() { return this._structArray.int16[this._pos2 + 5]; }\n get featureIndex() { return this._structArray.uint32[this._pos4 + 3]; }\n get sourceLayerIndex() { return this._structArray.uint16[this._pos2 + 8]; }\n get bucketIndex() { return this._structArray.uint16[this._pos2 + 9]; }\n get anchorPoint() { return new Point(this.anchorPointX, this.anchorPointY); }\n}\n\nCollisionBoxStruct.prototype.size = 20;\n\nexport type CollisionBox = CollisionBoxStruct;\n\n/** @internal */\nexport class CollisionBoxArray extends StructArrayLayout6i1ul2ui20 {\n /**\n * Return the CollisionBoxStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): CollisionBoxStruct {\n return new CollisionBoxStruct(this, index);\n }\n}\n\nregister('CollisionBoxArray', CollisionBoxArray);\n\n/** @internal */\nclass PlacedSymbolStruct extends Struct {\n _structArray: PlacedSymbolArray;\n get anchorX() { return this._structArray.int16[this._pos2 + 0]; }\n get anchorY() { return this._structArray.int16[this._pos2 + 1]; }\n get glyphStartIndex() { return this._structArray.uint16[this._pos2 + 2]; }\n get numGlyphs() { return this._structArray.uint16[this._pos2 + 3]; }\n get vertexStartIndex() { return this._structArray.uint32[this._pos4 + 2]; }\n get lineStartIndex() { return this._structArray.uint32[this._pos4 + 3]; }\n get lineLength() { return this._structArray.uint32[this._pos4 + 4]; }\n get segment() { return this._structArray.uint16[this._pos2 + 10]; }\n get lowerSize() { return this._structArray.uint16[this._pos2 + 11]; }\n get upperSize() { return this._structArray.uint16[this._pos2 + 12]; }\n get lineOffsetX() { return this._structArray.float32[this._pos4 + 7]; }\n get lineOffsetY() { return this._structArray.float32[this._pos4 + 8]; }\n get writingMode() { return this._structArray.uint8[this._pos1 + 36]; }\n get placedOrientation() { return this._structArray.uint8[this._pos1 + 37]; }\n set placedOrientation(x: number) { this._structArray.uint8[this._pos1 + 37] = x; }\n get hidden() { return this._structArray.uint8[this._pos1 + 38]; }\n set hidden(x: number) { this._structArray.uint8[this._pos1 + 38] = x; }\n get crossTileID() { return this._structArray.uint32[this._pos4 + 10]; }\n set crossTileID(x: number) { this._structArray.uint32[this._pos4 + 10] = x; }\n get associatedIconIndex() { return this._structArray.int16[this._pos2 + 22]; }\n}\n\nPlacedSymbolStruct.prototype.size = 48;\n\nexport type PlacedSymbol = PlacedSymbolStruct;\n\n/** @internal */\nexport class PlacedSymbolArray extends StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 {\n /**\n * Return the PlacedSymbolStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): PlacedSymbolStruct {\n return new PlacedSymbolStruct(this, index);\n }\n}\n\nregister('PlacedSymbolArray', PlacedSymbolArray);\n\n/** @internal */\nclass SymbolInstanceStruct extends Struct {\n _structArray: SymbolInstanceArray;\n get anchorX() { return this._structArray.int16[this._pos2 + 0]; }\n get anchorY() { return this._structArray.int16[this._pos2 + 1]; }\n get rightJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 2]; }\n get centerJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 3]; }\n get leftJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 4]; }\n get verticalPlacedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 5]; }\n get placedIconSymbolIndex() { return this._structArray.int16[this._pos2 + 6]; }\n get verticalPlacedIconSymbolIndex() { return this._structArray.int16[this._pos2 + 7]; }\n get key() { return this._structArray.uint16[this._pos2 + 8]; }\n get textBoxStartIndex() { return this._structArray.uint16[this._pos2 + 9]; }\n get textBoxEndIndex() { return this._structArray.uint16[this._pos2 + 10]; }\n get verticalTextBoxStartIndex() { return this._structArray.uint16[this._pos2 + 11]; }\n get verticalTextBoxEndIndex() { return this._structArray.uint16[this._pos2 + 12]; }\n get iconBoxStartIndex() { return this._structArray.uint16[this._pos2 + 13]; }\n get iconBoxEndIndex() { return this._structArray.uint16[this._pos2 + 14]; }\n get verticalIconBoxStartIndex() { return this._structArray.uint16[this._pos2 + 15]; }\n get verticalIconBoxEndIndex() { return this._structArray.uint16[this._pos2 + 16]; }\n get featureIndex() { return this._structArray.uint16[this._pos2 + 17]; }\n get numHorizontalGlyphVertices() { return this._structArray.uint16[this._pos2 + 18]; }\n get numVerticalGlyphVertices() { return this._structArray.uint16[this._pos2 + 19]; }\n get numIconVertices() { return this._structArray.uint16[this._pos2 + 20]; }\n get numVerticalIconVertices() { return this._structArray.uint16[this._pos2 + 21]; }\n get useRuntimeCollisionCircles() { return this._structArray.uint16[this._pos2 + 22]; }\n get crossTileID() { return this._structArray.uint32[this._pos4 + 12]; }\n set crossTileID(x: number) { this._structArray.uint32[this._pos4 + 12] = x; }\n get textBoxScale() { return this._structArray.float32[this._pos4 + 13]; }\n get collisionCircleDiameter() { return this._structArray.float32[this._pos4 + 14]; }\n get textAnchorOffsetStartIndex() { return this._structArray.uint16[this._pos2 + 30]; }\n get textAnchorOffsetEndIndex() { return this._structArray.uint16[this._pos2 + 31]; }\n}\n\nSymbolInstanceStruct.prototype.size = 64;\n\nexport type SymbolInstance = SymbolInstanceStruct;\n\n/** @internal */\nexport class SymbolInstanceArray extends StructArrayLayout8i15ui1ul2f2ui64 {\n /**\n * Return the SymbolInstanceStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): SymbolInstanceStruct {\n return new SymbolInstanceStruct(this, index);\n }\n}\n\nregister('SymbolInstanceArray', SymbolInstanceArray);\n\n/** @internal */\nexport class GlyphOffsetArray extends StructArrayLayout1f4 {\n getoffsetX(index: number) { return this.float32[index * 1 + 0]; }\n}\n\nregister('GlyphOffsetArray', GlyphOffsetArray);\n\n/** @internal */\nexport class SymbolLineVertexArray extends StructArrayLayout3i6 {\n getx(index: number) { return this.int16[index * 3 + 0]; }\n gety(index: number) { return this.int16[index * 3 + 1]; }\n gettileUnitDistanceFromAnchor(index: number) { return this.int16[index * 3 + 2]; }\n}\n\nregister('SymbolLineVertexArray', SymbolLineVertexArray);\n\n/** @internal */\nclass TextAnchorOffsetStruct extends Struct {\n _structArray: TextAnchorOffsetArray;\n get textAnchor() { return this._structArray.uint16[this._pos2 + 0]; }\n get textOffset0() { return this._structArray.float32[this._pos4 + 1]; }\n get textOffset1() { return this._structArray.float32[this._pos4 + 2]; }\n}\n\nTextAnchorOffsetStruct.prototype.size = 12;\n\nexport type TextAnchorOffset = TextAnchorOffsetStruct;\n\n/** @internal */\nexport class TextAnchorOffsetArray extends StructArrayLayout1ui2f12 {\n /**\n * Return the TextAnchorOffsetStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): TextAnchorOffsetStruct {\n return new TextAnchorOffsetStruct(this, index);\n }\n}\n\nregister('TextAnchorOffsetArray', TextAnchorOffsetArray);\n\n/** @internal */\nclass FeatureIndexStruct extends Struct {\n _structArray: FeatureIndexArray;\n get featureIndex() { return this._structArray.uint32[this._pos4 + 0]; }\n get sourceLayerIndex() { return this._structArray.uint16[this._pos2 + 2]; }\n get bucketIndex() { return this._structArray.uint16[this._pos2 + 3]; }\n}\n\nFeatureIndexStruct.prototype.size = 8;\n\nexport type FeatureIndex = FeatureIndexStruct;\n\n/** @internal */\nexport class FeatureIndexArray extends StructArrayLayout1ul2ui8 {\n /**\n * Return the FeatureIndexStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): FeatureIndexStruct {\n return new FeatureIndexStruct(this, index);\n }\n}\n\nregister('FeatureIndexArray', FeatureIndexArray);\n\nexport class PosArray extends StructArrayLayout2i4 {}\nexport class Pos3dArray extends StructArrayLayout3i6 {}\nexport class RasterBoundsArray extends StructArrayLayout4i8 {}\nexport class CircleLayoutArray extends StructArrayLayout2i4 {}\nexport class FillLayoutArray extends StructArrayLayout2i4 {}\nexport class FillExtrusionLayoutArray extends StructArrayLayout2i4i12 {}\nexport class HeatmapLayoutArray extends StructArrayLayout2i4 {}\nexport class LineLayoutArray extends StructArrayLayout2i4ub8 {}\nexport class LineExtLayoutArray extends StructArrayLayout2f8 {}\nexport class PatternLayoutArray extends StructArrayLayout10ui20 {}\nexport class SymbolLayoutArray extends StructArrayLayout4i4ui4i24 {}\nexport class SymbolDynamicLayoutArray extends StructArrayLayout3f12 {}\nexport class SymbolOpacityArray extends StructArrayLayout1ul4 {}\nexport class CollisionBoxLayoutArray extends StructArrayLayout2i2i2i12 {}\nexport class CollisionCircleLayoutArray extends StructArrayLayout2f1f2i16 {}\nexport class CollisionVertexArray extends StructArrayLayout2ub2f12 {}\nexport class QuadTriangleArray extends StructArrayLayout3ui6 {}\nexport class TriangleIndexArray extends StructArrayLayout3ui6 {}\nexport class LineIndexArray extends StructArrayLayout2ui4 {}\nexport class LineStripIndexArray extends StructArrayLayout1ui2 {}\nexport {\n StructArrayLayout2i4,\n StructArrayLayout3i6,\n StructArrayLayout4i8,\n StructArrayLayout2i4i12,\n StructArrayLayout2i4ub8,\n StructArrayLayout2f8,\n StructArrayLayout10ui20,\n StructArrayLayout4i4ui4i24,\n StructArrayLayout3f12,\n StructArrayLayout1ul4,\n StructArrayLayout6i1ul2ui20,\n StructArrayLayout2i2i2i12,\n StructArrayLayout2f1f2i16,\n StructArrayLayout2ub2f12,\n StructArrayLayout3ui6,\n StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48,\n StructArrayLayout8i15ui1ul2f2ui64,\n StructArrayLayout1f4,\n StructArrayLayout1ui2f12,\n StructArrayLayout1ul2ui8,\n StructArrayLayout2ui4,\n StructArrayLayout1ui2,\n StructArrayLayout4f16\n};\n","import {createLayout} from '../../util/struct_array';\n\nconst layout = createLayout([\n {name: 'a_pos', components: 2, type: 'Int16'}\n], 4);\n\nexport default layout;\nexport const {members, size, alignment} = layout;\n","import {warnOnce} from '../util/util';\n\nimport {register} from '../util/web_worker_transfer';\n\nimport type {VertexArrayObject} from '../render/vertex_array_object';\nimport type {StructArray} from '../util/struct_array';\n\n/**\n * @internal\n * A single segment of a vector\n */\nexport type Segment = {\n sortKey?: number;\n vertexOffset: number;\n primitiveOffset: number;\n vertexLength: number;\n primitiveLength: number;\n vaos: {[_: string]: VertexArrayObject};\n};\n\n/**\n * @internal\n * Used for calculations on vector segments\n */\nexport class SegmentVector {\n static MAX_VERTEX_ARRAY_LENGTH: number;\n segments: Array;\n\n constructor(segments: Array = []) {\n this.segments = segments;\n }\n\n prepareSegment(\n numVertices: number,\n layoutVertexArray: StructArray,\n indexArray: StructArray,\n sortKey?: number\n ): Segment {\n let segment: Segment = this.segments[this.segments.length - 1];\n if (numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) warnOnce(`Max vertices per segment is ${SegmentVector.MAX_VERTEX_ARRAY_LENGTH}: bucket requested ${numVertices}`);\n if (!segment || segment.vertexLength + numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH || segment.sortKey !== sortKey) {\n segment = ({\n vertexOffset: layoutVertexArray.length,\n primitiveOffset: indexArray.length,\n vertexLength: 0,\n primitiveLength: 0\n } as any);\n if (sortKey !== undefined) segment.sortKey = sortKey;\n this.segments.push(segment);\n }\n return segment;\n }\n\n get() {\n return this.segments;\n }\n\n destroy() {\n for (const segment of this.segments) {\n for (const k in segment.vaos) {\n segment.vaos[k].destroy();\n }\n }\n }\n\n static simpleSegment(\n vertexOffset: number,\n primitiveOffset: number,\n vertexLength: number,\n primitiveLength: number\n ): SegmentVector {\n return new SegmentVector([{\n vertexOffset,\n primitiveOffset,\n vertexLength,\n primitiveLength,\n vaos: {},\n sortKey: 0\n }]);\n }\n}\n\n/**\n * The maximum size of a vertex array. This limit is imposed by WebGL's 16 bit\n * addressing of vertex buffers.\n */\nSegmentVector.MAX_VERTEX_ARRAY_LENGTH = Math.pow(2, 16) - 1;\n\nregister('SegmentVector', SegmentVector);\n","import {clamp} from '../util/util';\n\n/**\n * Packs two numbers, interpreted as 8-bit unsigned integers, into a single\n * float. Unpack them in the shader using the `unpack_float()` function,\n * defined in _prelude.vertex.glsl\n */\nexport function packUint8ToFloat(a: number, b: number) {\n // coerce a and b to 8-bit ints\n a = clamp(Math.floor(a), 0, 255);\n b = clamp(Math.floor(b), 0, 255);\n return 256 * a + b;\n}\n","import {createLayout} from '../../util/struct_array';\n\nexport const patternAttributes = createLayout([\n // [tl.x, tl.y, br.x, br.y]\n {name: 'a_pattern_from', components: 4, type: 'Uint16'},\n {name: 'a_pattern_to', components: 4, type: 'Uint16'},\n {name: 'a_pixel_ratio_from', components: 1, type: 'Uint16'},\n {name: 'a_pixel_ratio_to', components: 1, type: 'Uint16'},\n]);\n","/**\n * JS Implementation of MurmurHash3 (r136) (as of May 20, 2011)\n * \n * @author Gary Court\n * @see http://github.com/garycourt/murmurhash-js\n * @author Austin Appleby\n * @see http://sites.google.com/site/murmurhash/\n * \n * @param {string} key ASCII only\n * @param {number} seed Positive integer only\n * @return {number} 32-bit positive integer hash \n */\n\nfunction murmurhash3_32_gc(key, seed) {\n\tvar remainder, bytes, h1, h1b, c1, c1b, c2, c2b, k1, i;\n\t\n\tremainder = key.length & 3; // key.length % 4\n\tbytes = key.length - remainder;\n\th1 = seed;\n\tc1 = 0xcc9e2d51;\n\tc2 = 0x1b873593;\n\ti = 0;\n\t\n\twhile (i < bytes) {\n\t \tk1 = \n\t \t ((key.charCodeAt(i) & 0xff)) |\n\t \t ((key.charCodeAt(++i) & 0xff) << 8) |\n\t \t ((key.charCodeAt(++i) & 0xff) << 16) |\n\t \t ((key.charCodeAt(++i) & 0xff) << 24);\n\t\t++i;\n\t\t\n\t\tk1 = ((((k1 & 0xffff) * c1) + ((((k1 >>> 16) * c1) & 0xffff) << 16))) & 0xffffffff;\n\t\tk1 = (k1 << 15) | (k1 >>> 17);\n\t\tk1 = ((((k1 & 0xffff) * c2) + ((((k1 >>> 16) * c2) & 0xffff) << 16))) & 0xffffffff;\n\n\t\th1 ^= k1;\n h1 = (h1 << 13) | (h1 >>> 19);\n\t\th1b = ((((h1 & 0xffff) * 5) + ((((h1 >>> 16) * 5) & 0xffff) << 16))) & 0xffffffff;\n\t\th1 = (((h1b & 0xffff) + 0x6b64) + ((((h1b >>> 16) + 0xe654) & 0xffff) << 16));\n\t}\n\t\n\tk1 = 0;\n\t\n\tswitch (remainder) {\n\t\tcase 3: k1 ^= (key.charCodeAt(i + 2) & 0xff) << 16;\n\t\tcase 2: k1 ^= (key.charCodeAt(i + 1) & 0xff) << 8;\n\t\tcase 1: k1 ^= (key.charCodeAt(i) & 0xff);\n\t\t\n\t\tk1 = (((k1 & 0xffff) * c1) + ((((k1 >>> 16) * c1) & 0xffff) << 16)) & 0xffffffff;\n\t\tk1 = (k1 << 15) | (k1 >>> 17);\n\t\tk1 = (((k1 & 0xffff) * c2) + ((((k1 >>> 16) * c2) & 0xffff) << 16)) & 0xffffffff;\n\t\th1 ^= k1;\n\t}\n\t\n\th1 ^= key.length;\n\n\th1 ^= h1 >>> 16;\n\th1 = (((h1 & 0xffff) * 0x85ebca6b) + ((((h1 >>> 16) * 0x85ebca6b) & 0xffff) << 16)) & 0xffffffff;\n\th1 ^= h1 >>> 13;\n\th1 = ((((h1 & 0xffff) * 0xc2b2ae35) + ((((h1 >>> 16) * 0xc2b2ae35) & 0xffff) << 16))) & 0xffffffff;\n\th1 ^= h1 >>> 16;\n\n\treturn h1 >>> 0;\n}\n\nif(typeof module !== \"undefined\") {\n module.exports = murmurhash3_32_gc\n}","/**\n * JS Implementation of MurmurHash2\n * \n * @author Gary Court\n * @see http://github.com/garycourt/murmurhash-js\n * @author Austin Appleby\n * @see http://sites.google.com/site/murmurhash/\n * \n * @param {string} str ASCII only\n * @param {number} seed Positive integer only\n * @return {number} 32-bit positive integer hash\n */\n\nfunction murmurhash2_32_gc(str, seed) {\n var\n l = str.length,\n h = seed ^ l,\n i = 0,\n k;\n \n while (l >= 4) {\n \tk = \n \t ((str.charCodeAt(i) & 0xff)) |\n \t ((str.charCodeAt(++i) & 0xff) << 8) |\n \t ((str.charCodeAt(++i) & 0xff) << 16) |\n \t ((str.charCodeAt(++i) & 0xff) << 24);\n \n k = (((k & 0xffff) * 0x5bd1e995) + ((((k >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n k ^= k >>> 24;\n k = (((k & 0xffff) * 0x5bd1e995) + ((((k >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n\n\th = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16)) ^ k;\n\n l -= 4;\n ++i;\n }\n \n switch (l) {\n case 3: h ^= (str.charCodeAt(i + 2) & 0xff) << 16;\n case 2: h ^= (str.charCodeAt(i + 1) & 0xff) << 8;\n case 1: h ^= (str.charCodeAt(i) & 0xff);\n h = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n }\n\n h ^= h >>> 13;\n h = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n h ^= h >>> 15;\n\n return h >>> 0;\n}\n\nif(typeof module !== undefined) {\n module.exports = murmurhash2_32_gc\n}\n","var murmur3 = require(\"./murmurhash3_gc.js\")\nvar murmur2 = require(\"./murmurhash2_gc.js\")\n\nmodule.exports = murmur3\nmodule.exports.murmur3 = murmur3\nmodule.exports.murmur2 = murmur2\n","import murmur3 from 'murmurhash-js';\nimport {register} from '../util/web_worker_transfer';\n\ntype SerializedFeaturePositionMap = {\n ids: Float64Array;\n positions: Uint32Array;\n};\n\ntype FeaturePosition = {\n index: number;\n start: number;\n end: number;\n};\n\n// A transferable data structure that maps feature ids to their indices and buffer offsets\nexport class FeaturePositionMap {\n ids: Array;\n positions: Array;\n indexed: boolean;\n\n constructor() {\n this.ids = [];\n this.positions = [];\n this.indexed = false;\n }\n\n add(id: unknown, index: number, start: number, end: number) {\n this.ids.push(getNumericId(id));\n this.positions.push(index, start, end);\n }\n\n getPositions(id: unknown): Array {\n if (!this.indexed) throw new Error('Trying to get index, but feature positions are not indexed');\n\n const intId = getNumericId(id);\n\n // binary search for the first occurrence of id in this.ids;\n // relies on ids/positions being sorted by id, which happens in serialization\n let i = 0;\n let j = this.ids.length - 1;\n while (i < j) {\n const m = (i + j) >> 1;\n if (this.ids[m] >= intId) {\n j = m;\n } else {\n i = m + 1;\n }\n }\n const positions = [];\n while (this.ids[i] === intId) {\n const index = this.positions[3 * i];\n const start = this.positions[3 * i + 1];\n const end = this.positions[3 * i + 2];\n positions.push({index, start, end});\n i++;\n }\n return positions;\n }\n\n static serialize(map: FeaturePositionMap, transferables: Array): SerializedFeaturePositionMap {\n const ids = new Float64Array(map.ids);\n const positions = new Uint32Array(map.positions);\n\n sort(ids, positions, 0, ids.length - 1);\n\n if (transferables) {\n transferables.push(ids.buffer, positions.buffer);\n }\n\n return {ids, positions};\n }\n\n static deserialize(obj: SerializedFeaturePositionMap): FeaturePositionMap {\n const map = new FeaturePositionMap();\n // after transferring, we only use these arrays statically (no pushes),\n // so TypedArray vs Array distinction that flow points out doesn't matter\n map.ids = (obj.ids as any);\n map.positions = (obj.positions as any);\n map.indexed = true;\n return map;\n }\n}\n\nfunction getNumericId(value: unknown) {\n const numValue = +value;\n if (!isNaN(numValue) && numValue <= Number.MAX_SAFE_INTEGER) {\n return numValue;\n }\n return murmur3(String(value));\n}\n\n// custom quicksort that sorts ids, indices and offsets together (by ids)\n// uses Hoare partitioning & manual tail call optimization to avoid worst case scenarios\nfunction sort(ids, positions, left, right) {\n while (left < right) {\n const pivot = ids[(left + right) >> 1];\n let i = left - 1;\n let j = right + 1;\n\n while (true) {\n do i++; while (ids[i] < pivot);\n do j--; while (ids[j] > pivot);\n if (i >= j) break;\n swap(ids, i, j);\n swap(positions, 3 * i, 3 * j);\n swap(positions, 3 * i + 1, 3 * j + 1);\n swap(positions, 3 * i + 2, 3 * j + 2);\n }\n\n if (j - left < right - j) {\n sort(ids, positions, left, j);\n left = j + 1;\n } else {\n sort(ids, positions, j + 1, right);\n right = j;\n }\n }\n}\n\nfunction swap(arr, i, j) {\n const tmp = arr[i];\n arr[i] = arr[j];\n arr[j] = tmp;\n}\n\nregister('FeaturePositionMap', FeaturePositionMap);\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {Context} from '../gl/context';\nimport {mat4, vec2, vec3, vec4} from 'gl-matrix';\n\ntype $ObjMap any> = {\n [K in keyof T]: F extends (v: T[K]) => infer R ? R : never;\n};\n\nexport type UniformValues = $ObjMap(u: Uniform) => V>;\nexport type UniformLocations = {[_: string]: WebGLUniformLocation};\n\n/**\n * @internal\n * A base uniform abstract class\n */\nabstract class Uniform {\n gl: WebGLRenderingContext|WebGL2RenderingContext;\n location: WebGLUniformLocation;\n current: T;\n\n constructor(context: Context, location: WebGLUniformLocation) {\n this.gl = context.gl;\n this.location = location;\n }\n\n abstract set(v: T): void;\n}\n\nclass Uniform1i extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = 0;\n }\n\n set(v: number): void {\n if (this.current !== v) {\n this.current = v;\n this.gl.uniform1i(this.location, v);\n }\n }\n}\n\nclass Uniform1f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = 0;\n }\n\n set(v: number): void {\n if (this.current !== v) {\n this.current = v;\n this.gl.uniform1f(this.location, v);\n }\n }\n}\n\nclass Uniform2f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = [0, 0];\n }\n\n set(v: vec2): void {\n if (v[0] !== this.current[0] || v[1] !== this.current[1]) {\n this.current = v;\n this.gl.uniform2f(this.location, v[0], v[1]);\n }\n }\n}\n\nclass Uniform3f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = [0, 0, 0];\n }\n\n set(v: vec3): void {\n if (v[0] !== this.current[0] || v[1] !== this.current[1] || v[2] !== this.current[2]) {\n this.current = v;\n this.gl.uniform3f(this.location, v[0], v[1], v[2]);\n }\n }\n}\n\nclass Uniform4f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = [0, 0, 0, 0];\n }\n\n set(v: vec4): void {\n if (v[0] !== this.current[0] || v[1] !== this.current[1] ||\n v[2] !== this.current[2] || v[3] !== this.current[3]) {\n this.current = v;\n this.gl.uniform4f(this.location, v[0], v[1], v[2], v[3]);\n }\n }\n}\n\nclass UniformColor extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = Color.transparent;\n }\n\n set(v: Color): void {\n if (v.r !== this.current.r || v.g !== this.current.g ||\n v.b !== this.current.b || v.a !== this.current.a) {\n this.current = v;\n this.gl.uniform4f(this.location, v.r, v.g, v.b, v.a);\n }\n }\n}\n\nconst emptyMat4 = new Float32Array(16) as mat4;\nclass UniformMatrix4f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = emptyMat4;\n }\n\n set(v: mat4): void {\n // The vast majority of matrix comparisons that will trip this set\n // happen at i=12 or i=0, so we check those first to avoid lots of\n // unnecessary iteration:\n if (v[12] !== this.current[12] || v[0] !== this.current[0]) {\n this.current = v;\n this.gl.uniformMatrix4fv(this.location, false, v);\n return;\n }\n for (let i = 1; i < 16; i++) {\n if (v[i] !== this.current[i]) {\n this.current = v;\n this.gl.uniformMatrix4fv(this.location, false, v);\n break;\n }\n }\n }\n}\n\nexport {\n Uniform,\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f,\n Uniform4f,\n UniformColor,\n UniformMatrix4f\n};\n\n/**\n * @internal\n * A uniform bindings\n */\nexport type UniformBindings = {[_: string]: Uniform};\n","import {packUint8ToFloat} from '../shaders/encode_attribute';\nimport {Color, supportsPropertyExpression} from '@maplibre/maplibre-gl-style-spec';\nimport {register} from '../util/web_worker_transfer';\nimport {PossiblyEvaluatedPropertyValue} from '../style/properties';\nimport {StructArrayLayout1f4, StructArrayLayout2f8, StructArrayLayout4f16, PatternLayoutArray} from './array_types.g';\nimport {clamp} from '../util/util';\nimport {patternAttributes} from './bucket/pattern_attributes';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {FeaturePositionMap} from './feature_position_map';\nimport {Uniform, Uniform1f, UniformColor, Uniform4f} from '../render/uniform_binding';\n\nimport type {UniformLocations} from '../render/uniform_binding';\n\nimport type {CanonicalTileID} from '../source/tile_id';\nimport type {Context} from '../gl/context';\nimport type {TypedStyleLayer} from '../style/style_layer/typed_style_layer';\nimport type {CrossfadeParameters} from '../style/evaluation_parameters';\nimport type {StructArray, StructArrayMember} from '../util/struct_array';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {ImagePosition} from '../render/image_atlas';\nimport type {\n Feature,\n FeatureState,\n GlobalProperties,\n SourceExpression,\n CompositeExpression,\n FormattedSection\n} from '@maplibre/maplibre-gl-style-spec';\nimport type {FeatureStates} from '../source/source_state';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nexport type BinderUniform = {\n name: string;\n property: string;\n binding: Uniform;\n};\n\nfunction packColor(color: Color): [number, number] {\n return [\n packUint8ToFloat(255 * color.r, 255 * color.g),\n packUint8ToFloat(255 * color.b, 255 * color.a)\n ];\n}\n\n/**\n * `Binder` is the interface definition for the strategies for constructing,\n * uploading, and binding paint property data as GLSL attributes. Most style-\n * spec properties have a 1:1 relationship to shader attribute/uniforms, but\n * some require multiple values per feature to be passed to the GPU, and in\n * those cases we bind multiple attributes/uniforms.\n *\n * It has three implementations, one for each of the three strategies we use:\n *\n * * For _constant_ properties -- those whose value is a constant, or the constant\n * result of evaluating a camera expression at a particular camera position -- we\n * don't need a vertex attribute buffer, and instead use a uniform.\n * * For data expressions, we use a vertex buffer with a single attribute value,\n * the evaluated result of the source function for the given feature.\n * * For composite expressions, we use a vertex buffer with two attributes: min and\n * max values covering the range of zooms at which we expect the tile to be\n * displayed. These values are calculated by evaluating the composite expression for\n * the given feature at strategically chosen zoom levels. In addition to this\n * attribute data, we also use a uniform value which the shader uses to interpolate\n * between the min and max value at the final displayed zoom level. The use of a\n * uniform allows us to cheaply update the value on every frame.\n *\n * Note that the shader source varies depending on whether we're using a uniform or\n * attribute. We dynamically compile shaders at runtime to accommodate this.\n */\ninterface AttributeBinder {\n populatePaintArray(\n length: number,\n feature: Feature,\n imagePositions: {[_: string]: ImagePosition},\n canonical?: CanonicalTileID,\n formattedSection?: FormattedSection\n ): void;\n updatePaintArray(\n start: number,\n length: number,\n feature: Feature,\n featureState: FeatureState,\n imagePositions: {[_: string]: ImagePosition}\n ): void;\n upload(a: Context): void;\n destroy(): void;\n}\n\ninterface UniformBinder {\n uniformNames: Array;\n setUniform(\n uniform: Uniform,\n globals: GlobalProperties,\n currentValue: PossiblyEvaluatedPropertyValue,\n uniformName: string\n ): void;\n getBinding(context: Context, location: WebGLUniformLocation, name: string): Partial>;\n}\n\nclass ConstantBinder implements UniformBinder {\n value: unknown;\n type: string;\n uniformNames: Array;\n\n constructor(value: unknown, names: Array, type: string) {\n this.value = value;\n this.uniformNames = names.map(name => `u_${name}`);\n this.type = type;\n }\n\n setUniform(\n uniform: Uniform,\n globals: GlobalProperties,\n currentValue: PossiblyEvaluatedPropertyValue\n ): void {\n uniform.set(currentValue.constantOr(this.value));\n }\n\n getBinding(context: Context, location: WebGLUniformLocation, _: string): Partial> {\n return (this.type === 'color') ?\n new UniformColor(context, location) :\n new Uniform1f(context, location);\n }\n}\n\nclass CrossFadedConstantBinder implements UniformBinder {\n uniformNames: Array;\n patternFrom: Array;\n patternTo: Array;\n pixelRatioFrom: number;\n pixelRatioTo: number;\n\n constructor(value: unknown, names: Array) {\n this.uniformNames = names.map(name => `u_${name}`);\n this.patternFrom = null;\n this.patternTo = null;\n this.pixelRatioFrom = 1.0;\n this.pixelRatioTo = 1.0;\n }\n\n setConstantPatternPositions(posTo: ImagePosition, posFrom: ImagePosition) {\n this.pixelRatioFrom = posFrom.pixelRatio;\n this.pixelRatioTo = posTo.pixelRatio;\n this.patternFrom = posFrom.tlbr;\n this.patternTo = posTo.tlbr;\n }\n\n setUniform(uniform: Uniform, globals: GlobalProperties, currentValue: PossiblyEvaluatedPropertyValue, uniformName: string) {\n const pos =\n uniformName === 'u_pattern_to' ? this.patternTo :\n uniformName === 'u_pattern_from' ? this.patternFrom :\n uniformName === 'u_pixel_ratio_to' ? this.pixelRatioTo :\n uniformName === 'u_pixel_ratio_from' ? this.pixelRatioFrom : null;\n if (pos) uniform.set(pos);\n }\n\n getBinding(context: Context, location: WebGLUniformLocation, name: string): Partial> {\n return name.substr(0, 9) === 'u_pattern' ?\n new Uniform4f(context, location) :\n new Uniform1f(context, location);\n }\n}\n\nclass SourceExpressionBinder implements AttributeBinder {\n expression: SourceExpression;\n type: string;\n maxValue: number;\n\n paintVertexArray: StructArray;\n paintVertexAttributes: Array;\n paintVertexBuffer: VertexBuffer;\n\n constructor(expression: SourceExpression, names: Array, type: string, PaintVertexArray: {\n new (...args: any): StructArray;\n }) {\n this.expression = expression;\n this.type = type;\n this.maxValue = 0;\n this.paintVertexAttributes = names.map((name) => ({\n name: `a_${name}`,\n type: 'Float32',\n components: type === 'color' ? 2 : 1,\n offset: 0\n }));\n this.paintVertexArray = new PaintVertexArray();\n }\n\n populatePaintArray(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) {\n const start = this.paintVertexArray.length;\n const value = this.expression.evaluate(new EvaluationParameters(0), feature, {}, canonical, [], formattedSection);\n this.paintVertexArray.resize(newLength);\n this._setPaintValue(start, newLength, value);\n }\n\n updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState) {\n const value = this.expression.evaluate({zoom: 0}, feature, featureState);\n this._setPaintValue(start, end, value);\n }\n\n _setPaintValue(start, end, value) {\n if (this.type === 'color') {\n const color = packColor(value);\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, color[0], color[1]);\n }\n } else {\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, value);\n }\n this.maxValue = Math.max(this.maxValue, Math.abs(value));\n }\n }\n\n upload(context: Context) {\n if (this.paintVertexArray && this.paintVertexArray.arrayBuffer) {\n if (this.paintVertexBuffer && this.paintVertexBuffer.buffer) {\n this.paintVertexBuffer.updateData(this.paintVertexArray);\n } else {\n this.paintVertexBuffer = context.createVertexBuffer(this.paintVertexArray, this.paintVertexAttributes, this.expression.isStateDependent);\n }\n }\n }\n\n destroy() {\n if (this.paintVertexBuffer) {\n this.paintVertexBuffer.destroy();\n }\n }\n}\n\nclass CompositeExpressionBinder implements AttributeBinder, UniformBinder {\n expression: CompositeExpression;\n uniformNames: Array;\n type: string;\n useIntegerZoom: boolean;\n zoom: number;\n maxValue: number;\n\n paintVertexArray: StructArray;\n paintVertexAttributes: Array;\n paintVertexBuffer: VertexBuffer;\n\n constructor(expression: CompositeExpression, names: Array, type: string, useIntegerZoom: boolean, zoom: number, PaintVertexArray: {\n new (...args: any): StructArray;\n }) {\n this.expression = expression;\n this.uniformNames = names.map(name => `u_${name}_t`);\n this.type = type;\n this.useIntegerZoom = useIntegerZoom;\n this.zoom = zoom;\n this.maxValue = 0;\n this.paintVertexAttributes = names.map((name) => ({\n name: `a_${name}`,\n type: 'Float32',\n components: type === 'color' ? 4 : 2,\n offset: 0\n }));\n this.paintVertexArray = new PaintVertexArray();\n }\n\n populatePaintArray(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) {\n const min = this.expression.evaluate(new EvaluationParameters(this.zoom), feature, {}, canonical, [], formattedSection);\n const max = this.expression.evaluate(new EvaluationParameters(this.zoom + 1), feature, {}, canonical, [], formattedSection);\n const start = this.paintVertexArray.length;\n this.paintVertexArray.resize(newLength);\n this._setPaintValue(start, newLength, min, max);\n }\n\n updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState) {\n const min = this.expression.evaluate({zoom: this.zoom}, feature, featureState);\n const max = this.expression.evaluate({zoom: this.zoom + 1}, feature, featureState);\n this._setPaintValue(start, end, min, max);\n }\n\n _setPaintValue(start, end, min, max) {\n if (this.type === 'color') {\n const minColor = packColor(min);\n const maxColor = packColor(max);\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, minColor[0], minColor[1], maxColor[0], maxColor[1]);\n }\n } else {\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, min, max);\n }\n this.maxValue = Math.max(this.maxValue, Math.abs(min), Math.abs(max));\n }\n }\n\n upload(context: Context) {\n if (this.paintVertexArray && this.paintVertexArray.arrayBuffer) {\n if (this.paintVertexBuffer && this.paintVertexBuffer.buffer) {\n this.paintVertexBuffer.updateData(this.paintVertexArray);\n } else {\n this.paintVertexBuffer = context.createVertexBuffer(this.paintVertexArray, this.paintVertexAttributes, this.expression.isStateDependent);\n }\n }\n }\n\n destroy() {\n if (this.paintVertexBuffer) {\n this.paintVertexBuffer.destroy();\n }\n }\n\n setUniform(uniform: Uniform, globals: GlobalProperties): void {\n const currentZoom = this.useIntegerZoom ? Math.floor(globals.zoom) : globals.zoom;\n const factor = clamp(this.expression.interpolationFactor(currentZoom, this.zoom, this.zoom + 1), 0, 1);\n uniform.set(factor);\n }\n\n getBinding(context: Context, location: WebGLUniformLocation, _: string): Uniform1f {\n return new Uniform1f(context, location);\n }\n}\n\nclass CrossFadedCompositeBinder implements AttributeBinder {\n expression: CompositeExpression;\n type: string;\n useIntegerZoom: boolean;\n zoom: number;\n layerId: string;\n\n zoomInPaintVertexArray: StructArray;\n zoomOutPaintVertexArray: StructArray;\n zoomInPaintVertexBuffer: VertexBuffer;\n zoomOutPaintVertexBuffer: VertexBuffer;\n paintVertexAttributes: Array;\n\n constructor(expression: CompositeExpression, type: string, useIntegerZoom: boolean, zoom: number, PaintVertexArray: {\n new (...args: any): StructArray;\n }, layerId: string) {\n this.expression = expression;\n this.type = type;\n this.useIntegerZoom = useIntegerZoom;\n this.zoom = zoom;\n this.layerId = layerId;\n\n this.zoomInPaintVertexArray = new PaintVertexArray();\n this.zoomOutPaintVertexArray = new PaintVertexArray();\n }\n\n populatePaintArray(length: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}) {\n const start = this.zoomInPaintVertexArray.length;\n this.zoomInPaintVertexArray.resize(length);\n this.zoomOutPaintVertexArray.resize(length);\n this._setPaintValues(start, length, feature.patterns && feature.patterns[this.layerId], imagePositions);\n }\n\n updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState, imagePositions: {[_: string]: ImagePosition}) {\n this._setPaintValues(start, end, feature.patterns && feature.patterns[this.layerId], imagePositions);\n }\n\n _setPaintValues(start, end, patterns, positions) {\n if (!positions || !patterns) return;\n\n const {min, mid, max} = patterns;\n const imageMin = positions[min];\n const imageMid = positions[mid];\n const imageMax = positions[max];\n if (!imageMin || !imageMid || !imageMax) return;\n\n // We populate two paint arrays because, for cross-faded properties, we don't know which direction\n // we're cross-fading to at layout time. In order to keep vertex attributes to a minimum and not pass\n // unnecessary vertex data to the shaders, we determine which to upload at draw time.\n for (let i = start; i < end; i++) {\n this.zoomInPaintVertexArray.emplace(i,\n imageMid.tl[0], imageMid.tl[1], imageMid.br[0], imageMid.br[1],\n imageMin.tl[0], imageMin.tl[1], imageMin.br[0], imageMin.br[1],\n imageMid.pixelRatio,\n imageMin.pixelRatio,\n );\n this.zoomOutPaintVertexArray.emplace(i,\n imageMid.tl[0], imageMid.tl[1], imageMid.br[0], imageMid.br[1],\n imageMax.tl[0], imageMax.tl[1], imageMax.br[0], imageMax.br[1],\n imageMid.pixelRatio,\n imageMax.pixelRatio,\n );\n }\n }\n\n upload(context: Context) {\n if (this.zoomInPaintVertexArray && this.zoomInPaintVertexArray.arrayBuffer && this.zoomOutPaintVertexArray && this.zoomOutPaintVertexArray.arrayBuffer) {\n this.zoomInPaintVertexBuffer = context.createVertexBuffer(this.zoomInPaintVertexArray, patternAttributes.members, this.expression.isStateDependent);\n this.zoomOutPaintVertexBuffer = context.createVertexBuffer(this.zoomOutPaintVertexArray, patternAttributes.members, this.expression.isStateDependent);\n }\n }\n\n destroy() {\n if (this.zoomOutPaintVertexBuffer) this.zoomOutPaintVertexBuffer.destroy();\n if (this.zoomInPaintVertexBuffer) this.zoomInPaintVertexBuffer.destroy();\n }\n}\n\n/**\n * @internal\n * ProgramConfiguration contains the logic for binding style layer properties and tile\n * layer feature data into GL program uniforms and vertex attributes.\n *\n * Non-data-driven property values are bound to shader uniforms. Data-driven property\n * values are bound to vertex attributes. In order to support a uniform GLSL syntax over\n * both, [Mapbox GL Shaders](https://github.com/mapbox/mapbox-gl-shaders) defines a `#pragma`\n * abstraction, which ProgramConfiguration is responsible for implementing. At runtime,\n * it examines the attributes of a particular layer, combines this with fixed knowledge\n * about how layers of the particular type are implemented, and determines which uniforms\n * and vertex attributes will be required. It can then substitute the appropriate text\n * into the shader source code, create and link a program, and bind the uniforms and\n * vertex attributes in preparation for drawing.\n *\n * When a vector tile is parsed, this same configuration information is used to\n * populate the attribute buffers needed for data-driven styling using the zoom\n * level and feature property data.\n */\nexport class ProgramConfiguration {\n binders: {[_: string]: AttributeBinder | UniformBinder};\n cacheKey: string;\n\n _buffers: Array;\n\n constructor(layer: TypedStyleLayer, zoom: number, filterProperties: (_: string) => boolean) {\n this.binders = {};\n this._buffers = [];\n\n const keys = [];\n\n for (const property in layer.paint._values) {\n if (!filterProperties(property)) continue;\n const value = (layer.paint as any).get(property);\n if (!(value instanceof PossiblyEvaluatedPropertyValue) || !supportsPropertyExpression(value.property.specification)) {\n continue;\n }\n const names = paintAttributeNames(property, layer.type);\n const expression = value.value;\n const type = value.property.specification.type;\n const useIntegerZoom = (value.property as any).useIntegerZoom;\n const propType = value.property.specification['property-type'];\n const isCrossFaded = propType === 'cross-faded' || propType === 'cross-faded-data-driven';\n\n if (expression.kind === 'constant') {\n this.binders[property] = isCrossFaded ?\n new CrossFadedConstantBinder(expression.value, names) :\n new ConstantBinder(expression.value, names, type);\n keys.push(`/u_${property}`);\n\n } else if (expression.kind === 'source' || isCrossFaded) {\n const StructArrayLayout = layoutType(property, type, 'source');\n this.binders[property] = isCrossFaded ?\n new CrossFadedCompositeBinder(expression as CompositeExpression, type, useIntegerZoom, zoom, StructArrayLayout, layer.id) :\n new SourceExpressionBinder(expression as SourceExpression, names, type, StructArrayLayout);\n keys.push(`/a_${property}`);\n\n } else {\n const StructArrayLayout = layoutType(property, type, 'composite');\n this.binders[property] = new CompositeExpressionBinder(expression, names, type, useIntegerZoom, zoom, StructArrayLayout);\n keys.push(`/z_${property}`);\n }\n }\n\n this.cacheKey = keys.sort().join('');\n }\n\n getMaxValue(property: string): number {\n const binder = this.binders[property];\n return binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder ? binder.maxValue : 0;\n }\n\n populatePaintArrays(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder)\n (binder as AttributeBinder).populatePaintArray(newLength, feature, imagePositions, canonical, formattedSection);\n }\n }\n setConstantPatternPositions(posTo: ImagePosition, posFrom: ImagePosition) {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof CrossFadedConstantBinder)\n binder.setConstantPatternPositions(posTo, posFrom);\n }\n }\n\n updatePaintArrays(\n featureStates: FeatureStates,\n featureMap: FeaturePositionMap,\n vtLayer: VectorTileLayer,\n layer: TypedStyleLayer,\n imagePositions: {[_: string]: ImagePosition}\n ): boolean {\n let dirty: boolean = false;\n for (const id in featureStates) {\n const positions = featureMap.getPositions(id);\n\n for (const pos of positions) {\n const feature = vtLayer.feature(pos.index);\n\n for (const property in this.binders) {\n const binder = this.binders[property];\n if ((binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder ||\n binder instanceof CrossFadedCompositeBinder) && (binder as any).expression.isStateDependent === true) {\n //AHM: Remove after https://github.com/mapbox/mapbox-gl-js/issues/6255\n const value = (layer.paint as any).get(property);\n (binder as any).expression = value.value;\n (binder as AttributeBinder).updatePaintArray(pos.start, pos.end, feature, featureStates[id], imagePositions);\n dirty = true;\n }\n }\n }\n }\n return dirty;\n }\n\n defines(): Array {\n const result = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder) {\n result.push(...binder.uniformNames.map(name => `#define HAS_UNIFORM_${name}`));\n }\n }\n return result;\n }\n\n getBinderAttributes(): Array {\n const result = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder) {\n for (let i = 0; i < binder.paintVertexAttributes.length; i++) {\n result.push(binder.paintVertexAttributes[i].name);\n }\n } else if (binder instanceof CrossFadedCompositeBinder) {\n for (let i = 0; i < patternAttributes.members.length; i++) {\n result.push(patternAttributes.members[i].name);\n }\n }\n }\n return result;\n }\n\n getBinderUniforms(): Array {\n const uniforms = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder || binder instanceof CompositeExpressionBinder) {\n for (const uniformName of binder.uniformNames) {\n uniforms.push(uniformName);\n }\n }\n }\n return uniforms;\n }\n\n getPaintVertexBuffers(): Array {\n return this._buffers;\n }\n\n getUniforms(context: Context, locations: UniformLocations): Array {\n const uniforms = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder || binder instanceof CompositeExpressionBinder) {\n for (const name of binder.uniformNames) {\n if (locations[name]) {\n const binding = binder.getBinding(context, locations[name], name);\n uniforms.push({name, property, binding});\n }\n }\n }\n }\n return uniforms;\n }\n\n setUniforms(\n context: Context,\n binderUniforms: Array,\n properties: any,\n globals: GlobalProperties\n ) {\n // Uniform state bindings are owned by the Program, but we set them\n // from within the ProgramConfiguraton's binder members.\n for (const {name, property, binding} of binderUniforms) {\n (this.binders[property] as any).setUniform(binding, globals, properties.get(property), name);\n }\n }\n\n updatePaintBuffers(crossfade?: CrossfadeParameters) {\n this._buffers = [];\n\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (crossfade && binder instanceof CrossFadedCompositeBinder) {\n const patternVertexBuffer = crossfade.fromScale === 2 ? binder.zoomInPaintVertexBuffer : binder.zoomOutPaintVertexBuffer;\n if (patternVertexBuffer) this._buffers.push(patternVertexBuffer);\n\n } else if ((binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder) && binder.paintVertexBuffer) {\n this._buffers.push(binder.paintVertexBuffer);\n }\n }\n }\n\n upload(context: Context) {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder)\n binder.upload(context);\n }\n this.updatePaintBuffers();\n }\n\n destroy() {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder)\n binder.destroy();\n }\n }\n}\n\nexport class ProgramConfigurationSet {\n programConfigurations: {[_: string]: ProgramConfiguration};\n needsUpload: boolean;\n _featureMap: FeaturePositionMap;\n _bufferOffset: number;\n\n constructor(layers: ReadonlyArray, zoom: number, filterProperties: (_: string) => boolean = () => true) {\n this.programConfigurations = {};\n for (const layer of layers) {\n this.programConfigurations[layer.id] = new ProgramConfiguration(layer, zoom, filterProperties);\n }\n this.needsUpload = false;\n this._featureMap = new FeaturePositionMap();\n this._bufferOffset = 0;\n }\n\n populatePaintArrays(length: number, feature: Feature, index: number, imagePositions: {[_: string]: ImagePosition}, canonical: CanonicalTileID, formattedSection?: FormattedSection) {\n for (const key in this.programConfigurations) {\n this.programConfigurations[key].populatePaintArrays(length, feature, imagePositions, canonical, formattedSection);\n }\n\n if (feature.id !== undefined) {\n this._featureMap.add(feature.id, index, this._bufferOffset, length);\n }\n this._bufferOffset = length;\n\n this.needsUpload = true;\n }\n\n updatePaintArrays(featureStates: FeatureStates, vtLayer: VectorTileLayer, layers: ReadonlyArray, imagePositions: {[_: string]: ImagePosition}) {\n for (const layer of layers) {\n this.needsUpload = this.programConfigurations[layer.id].updatePaintArrays(featureStates, this._featureMap, vtLayer, layer, imagePositions) || this.needsUpload;\n }\n }\n\n get(layerId: string) {\n return this.programConfigurations[layerId];\n }\n\n upload(context: Context) {\n if (!this.needsUpload) return;\n for (const layerId in this.programConfigurations) {\n this.programConfigurations[layerId].upload(context);\n }\n this.needsUpload = false;\n }\n\n destroy() {\n for (const layerId in this.programConfigurations) {\n this.programConfigurations[layerId].destroy();\n }\n }\n}\n\nfunction paintAttributeNames(property, type) {\n const attributeNameExceptions = {\n 'text-opacity': ['opacity'],\n 'icon-opacity': ['opacity'],\n 'text-color': ['fill_color'],\n 'icon-color': ['fill_color'],\n 'text-halo-color': ['halo_color'],\n 'icon-halo-color': ['halo_color'],\n 'text-halo-blur': ['halo_blur'],\n 'icon-halo-blur': ['halo_blur'],\n 'text-halo-width': ['halo_width'],\n 'icon-halo-width': ['halo_width'],\n 'line-gap-width': ['gapwidth'],\n 'line-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'],\n 'fill-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'],\n 'fill-extrusion-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'],\n };\n\n return attributeNameExceptions[property] || [property.replace(`${type}-`, '').replace(/-/g, '_')];\n}\n\nfunction getLayoutException(property) {\n const propertyExceptions = {\n 'line-pattern': {\n 'source': PatternLayoutArray,\n 'composite': PatternLayoutArray\n },\n 'fill-pattern': {\n 'source': PatternLayoutArray,\n 'composite': PatternLayoutArray\n },\n 'fill-extrusion-pattern': {\n 'source': PatternLayoutArray,\n 'composite': PatternLayoutArray\n }\n };\n\n return propertyExceptions[property];\n}\n\nfunction layoutType(property, type, binderType) {\n const defaultLayouts = {\n 'color': {\n 'source': StructArrayLayout2f8,\n 'composite': StructArrayLayout4f16\n },\n 'number': {\n 'source': StructArrayLayout1f4,\n 'composite': StructArrayLayout2f8\n }\n };\n\n const layoutException = getLayoutException(property);\n return layoutException && layoutException[binderType] || defaultLayouts[type][binderType];\n}\n\nregister('ConstantBinder', ConstantBinder);\nregister('CrossFadedConstantBinder', CrossFadedConstantBinder);\nregister('SourceExpressionBinder', SourceExpressionBinder);\nregister('CrossFadedCompositeBinder', CrossFadedCompositeBinder);\nregister('CompositeExpressionBinder', CompositeExpressionBinder);\nregister('ProgramConfiguration', ProgramConfiguration, {omit: ['_buffers']});\nregister('ProgramConfigurationSet', ProgramConfigurationSet);\n","/**\n * The maximum value of a coordinate in the internal tile coordinate system. Coordinates of\n * all source features normalized to this extent upon load.\n *\n * The value is a consequence of the following:\n *\n * * Vertex buffer store positions as signed 16 bit integers.\n * * One bit is lost for signedness to support tile buffers.\n * * One bit is lost because the line vertex buffer used to pack 1 bit of other data into the int.\n * * One bit is lost to support features extending past the extent on the right edge of the tile.\n * * This leaves us with 2^13 = 8192\n */\nexport const EXTENT = 8192;\n","import {warnOnce, clamp} from '../util/util';\n\nimport {EXTENT} from './extent';\n\nimport type Point from '@mapbox/point-geometry';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\n// These bounds define the minimum and maximum supported coordinate values.\n// While visible coordinates are within [0, EXTENT], tiles may theoretically\n// contain coordinates within [-Infinity, Infinity]. Our range is limited by the\n// number of bits used to represent the coordinate.\nconst BITS = 15;\nconst MAX = Math.pow(2, BITS - 1) - 1;\nconst MIN = -MAX - 1;\n\n/**\n * Loads a geometry from a VectorTileFeature and scales it to the common extent\n * used internally.\n * @param feature - the vector tile feature to load\n */\nexport function loadGeometry(feature: VectorTileFeature): Array> {\n const scale = EXTENT / feature.extent;\n const geometry = feature.loadGeometry();\n for (let r = 0; r < geometry.length; r++) {\n const ring = geometry[r];\n for (let p = 0; p < ring.length; p++) {\n const point = ring[p];\n // round here because mapbox-gl-native uses integers to represent\n // points and we need to do the same to avoid renering differences.\n const x = Math.round(point.x * scale);\n const y = Math.round(point.y * scale);\n\n point.x = clamp(x, MIN, MAX);\n point.y = clamp(y, MIN, MAX);\n\n if (x < point.x || x > point.x + 1 || y < point.y || y > point.y + 1) {\n // warn when exceeding allowed extent except for the 1-px-off case\n // https://github.com/mapbox/mapbox-gl-js/issues/8992\n warnOnce('Geometry exceeds allowed extent, reduce your vector tile buffer size');\n }\n }\n }\n return geometry;\n}\n","import {loadGeometry} from './load_geometry';\nimport type Point from '@mapbox/point-geometry';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\nimport type {Feature} from '@maplibre/maplibre-gl-style-spec';\n\ntype EvaluationFeature = Feature & { geometry: Array> };\n/**\n * Construct a new feature based on a VectorTileFeature for expression evaluation, the geometry of which\n * will be loaded based on necessity.\n * @param feature - the feature to evaluate\n * @param needGeometry - if set to true this will load the geometry\n */\nexport function toEvaluationFeature(feature: VectorTileFeature, needGeometry: boolean): EvaluationFeature {\n return {type: feature.type,\n id: feature.id,\n properties: feature.properties,\n geometry: needGeometry ? loadGeometry(feature) : []};\n}\n","import {CircleLayoutArray} from '../array_types.g';\n\nimport {members as layoutAttributes} from './circle_attributes';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray} from '../index_array_type';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EXTENT} from '../extent';\nimport {register} from '../../util/web_worker_transfer';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {CircleStyleLayer} from '../../style/style_layer/circle_style_layer';\nimport type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nfunction addCircleVertex(layoutVertexArray, x, y, extrudeX, extrudeY) {\n layoutVertexArray.emplaceBack(\n (x * 2) + ((extrudeX + 1) / 2),\n (y * 2) + ((extrudeY + 1) / 2));\n}\n\n/**\n * @internal\n * Circles are represented by two triangles.\n *\n * Each corner has a pos that is the center of the circle and an extrusion\n * vector that is where it points.\n */\nexport class CircleBucket implements Bucket {\n index: number;\n zoom: number;\n overscaling: number;\n layerIds: Array;\n layers: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n\n layoutVertexArray: CircleLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n uploaded: boolean;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n\n this.layoutVertexArray = new CircleLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.segments = new SegmentVector();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n const styleLayer = this.layers[0];\n const bucketFeatures: BucketFeature[] = [];\n let circleSortKey = null;\n let sortFeaturesByKey = false;\n\n // Heatmap layers are handled in this bucket and have no evaluated properties, so we check our access\n if (styleLayer.type === 'circle') {\n circleSortKey = (styleLayer as CircleStyleLayer).layout.get('circle-sort-key');\n sortFeaturesByKey = !circleSortKey.isConstant();\n }\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const sortKey = sortFeaturesByKey ?\n circleSortKey.evaluate(evaluationFeature, {}, canonical) :\n undefined;\n\n const bucketFeature: BucketFeature = {\n id,\n properties: feature.properties,\n type: feature.type,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n patterns: {},\n sortKey\n };\n\n bucketFeatures.push(bucketFeature);\n\n }\n\n if (sortFeaturesByKey) {\n bucketFeatures.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const bucketFeature of bucketFeatures) {\n const {geometry, index, sourceLayerIndex} = bucketFeature;\n const feature = features[index].feature;\n\n this.addFeature(bucketFeature, geometry, index, canonical);\n options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0;\n }\n\n uploadPending() {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID) {\n for (const ring of geometry) {\n for (const point of ring) {\n const x = point.x;\n const y = point.y;\n\n // Do not include points that are outside the tile boundaries.\n if (x < 0 || x >= EXTENT || y < 0 || y >= EXTENT) continue;\n\n // this geometry will be of the Point type, and we'll derive\n // two triangles from it.\n //\n // ┌─────────┐\n // │ 3 2 │\n // │ │\n // │ 0 1 │\n // └─────────┘\n\n const segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray, feature.sortKey);\n const index = segment.vertexLength;\n\n addCircleVertex(this.layoutVertexArray, x, y, -1, -1);\n addCircleVertex(this.layoutVertexArray, x, y, 1, -1);\n addCircleVertex(this.layoutVertexArray, x, y, 1, 1);\n addCircleVertex(this.layoutVertexArray, x, y, -1, 1);\n\n this.indexArray.emplaceBack(index, index + 1, index + 2);\n this.indexArray.emplaceBack(index, index + 3, index + 2);\n\n segment.vertexLength += 4;\n segment.primitiveLength += 2;\n }\n }\n\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, {}, canonical);\n }\n}\n\nregister('CircleBucket', CircleBucket, {omit: ['layers']});\n","import {isCounterClockwise} from './util';\n\nimport Point from '@mapbox/point-geometry';\n\nexport {polygonIntersectsBufferedPoint, polygonIntersectsMultiPolygon, polygonIntersectsBufferedMultiLine, polygonIntersectsPolygon, distToSegmentSquared, polygonIntersectsBox};\n\ntype Line = Array;\ntype MultiLine = Array;\ntype Ring = Array;\ntype Polygon = Array;\ntype MultiPolygon = Array;\n\nfunction polygonIntersectsPolygon(polygonA: Polygon, polygonB: Polygon) {\n for (let i = 0; i < polygonA.length; i++) {\n if (polygonContainsPoint(polygonB, polygonA[i])) return true;\n }\n\n for (let i = 0; i < polygonB.length; i++) {\n if (polygonContainsPoint(polygonA, polygonB[i])) return true;\n }\n\n if (lineIntersectsLine(polygonA, polygonB)) return true;\n\n return false;\n}\n\nfunction polygonIntersectsBufferedPoint(polygon: Polygon, point: Point, radius: number) {\n if (polygonContainsPoint(polygon, point)) return true;\n if (pointIntersectsBufferedLine(point, polygon, radius)) return true;\n return false;\n}\n\nfunction polygonIntersectsMultiPolygon(polygon: Polygon, multiPolygon: MultiPolygon) {\n\n if (polygon.length === 1) {\n return multiPolygonContainsPoint(multiPolygon, polygon[0]);\n }\n\n for (let m = 0; m < multiPolygon.length; m++) {\n const ring = multiPolygon[m];\n for (let n = 0; n < ring.length; n++) {\n if (polygonContainsPoint(polygon, ring[n])) return true;\n }\n }\n\n for (let i = 0; i < polygon.length; i++) {\n if (multiPolygonContainsPoint(multiPolygon, polygon[i])) return true;\n }\n\n for (let k = 0; k < multiPolygon.length; k++) {\n if (lineIntersectsLine(polygon, multiPolygon[k])) return true;\n }\n\n return false;\n}\n\nfunction polygonIntersectsBufferedMultiLine(polygon: Polygon, multiLine: MultiLine, radius: number) {\n for (let i = 0; i < multiLine.length; i++) {\n const line = multiLine[i];\n\n if (polygon.length >= 3) {\n for (let k = 0; k < line.length; k++) {\n if (polygonContainsPoint(polygon, line[k])) return true;\n }\n }\n\n if (lineIntersectsBufferedLine(polygon, line, radius)) return true;\n }\n return false;\n}\n\nfunction lineIntersectsBufferedLine(lineA: Line, lineB: Line, radius: number) {\n\n if (lineA.length > 1) {\n if (lineIntersectsLine(lineA, lineB)) return true;\n\n // Check whether any point in either line is within radius of the other line\n for (let j = 0; j < lineB.length; j++) {\n if (pointIntersectsBufferedLine(lineB[j], lineA, radius)) return true;\n }\n }\n\n for (let k = 0; k < lineA.length; k++) {\n if (pointIntersectsBufferedLine(lineA[k], lineB, radius)) return true;\n }\n\n return false;\n}\n\nfunction lineIntersectsLine(lineA: Line, lineB: Line) {\n if (lineA.length === 0 || lineB.length === 0) return false;\n for (let i = 0; i < lineA.length - 1; i++) {\n const a0 = lineA[i];\n const a1 = lineA[i + 1];\n for (let j = 0; j < lineB.length - 1; j++) {\n const b0 = lineB[j];\n const b1 = lineB[j + 1];\n if (lineSegmentIntersectsLineSegment(a0, a1, b0, b1)) return true;\n }\n }\n return false;\n}\n\nfunction lineSegmentIntersectsLineSegment(a0: Point, a1: Point, b0: Point, b1: Point) {\n return isCounterClockwise(a0, b0, b1) !== isCounterClockwise(a1, b0, b1) &&\n isCounterClockwise(a0, a1, b0) !== isCounterClockwise(a0, a1, b1);\n}\n\nfunction pointIntersectsBufferedLine(p: Point, line: Line, radius: number) {\n const radiusSquared = radius * radius;\n\n if (line.length === 1) return p.distSqr(line[0]) < radiusSquared;\n\n for (let i = 1; i < line.length; i++) {\n // Find line segments that have a distance <= radius^2 to p\n // In that case, we treat the line as \"containing point p\".\n const v = line[i - 1], w = line[i];\n if (distToSegmentSquared(p, v, w) < radiusSquared) return true;\n }\n return false;\n}\n\n// Code from http://stackoverflow.com/a/1501725/331379.\nfunction distToSegmentSquared(p: Point, v: Point, w: Point) {\n const l2 = v.distSqr(w);\n if (l2 === 0) return p.distSqr(v);\n const t = ((p.x - v.x) * (w.x - v.x) + (p.y - v.y) * (w.y - v.y)) / l2;\n if (t < 0) return p.distSqr(v);\n if (t > 1) return p.distSqr(w);\n return p.distSqr(w.sub(v)._mult(t)._add(v));\n}\n\n// point in polygon ray casting algorithm\nfunction multiPolygonContainsPoint(rings: Array, p: Point) {\n let c = false,\n ring, p1, p2;\n\n for (let k = 0; k < rings.length; k++) {\n ring = rings[k];\n for (let i = 0, j = ring.length - 1; i < ring.length; j = i++) {\n p1 = ring[i];\n p2 = ring[j];\n if (((p1.y > p.y) !== (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) {\n c = !c;\n }\n }\n }\n return c;\n}\n\nfunction polygonContainsPoint(ring: Ring, p: Point) {\n let c = false;\n for (let i = 0, j = ring.length - 1; i < ring.length; j = i++) {\n const p1 = ring[i];\n const p2 = ring[j];\n if (((p1.y > p.y) !== (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) {\n c = !c;\n }\n }\n return c;\n}\n\nfunction polygonIntersectsBox(ring: Ring, boxX1: number, boxY1: number, boxX2: number, boxY2: number) {\n for (const p of ring) {\n if (boxX1 <= p.x &&\n boxY1 <= p.y &&\n boxX2 >= p.x &&\n boxY2 >= p.y) return true;\n }\n\n const corners = [\n new Point(boxX1, boxY1),\n new Point(boxX1, boxY2),\n new Point(boxX2, boxY2),\n new Point(boxX2, boxY1)];\n\n if (ring.length > 2) {\n for (const corner of corners) {\n if (polygonContainsPoint(ring, corner)) return true;\n }\n }\n\n for (let i = 0; i < ring.length - 1; i++) {\n const p1 = ring[i];\n const p2 = ring[i + 1];\n if (edgeIntersectsBox(p1, p2, corners)) return true;\n }\n\n return false;\n}\n\nfunction edgeIntersectsBox(e1: Point, e2: Point, corners: Array) {\n const tl = corners[0];\n const br = corners[2];\n // the edge and box do not intersect in either the x or y dimensions\n if (((e1.x < tl.x) && (e2.x < tl.x)) ||\n ((e1.x > br.x) && (e2.x > br.x)) ||\n ((e1.y < tl.y) && (e2.y < tl.y)) ||\n ((e1.y > br.y) && (e2.y > br.y))) return false;\n\n // check if all corners of the box are on the same side of the edge\n const dir = isCounterClockwise(e1, e2, corners[0]);\n return dir !== isCounterClockwise(e1, e2, corners[1]) ||\n dir !== isCounterClockwise(e1, e2, corners[2]) ||\n dir !== isCounterClockwise(e1, e2, corners[3]);\n}\n","import Point from '@mapbox/point-geometry';\n\nimport type {PossiblyEvaluatedPropertyValue} from './properties';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {CircleBucket} from '../data/bucket/circle_bucket';\nimport type {LineBucket} from '../data/bucket/line_bucket';\n\nexport function getMaximumPaintValue(\n property: string,\n layer: StyleLayer,\n bucket: CircleBucket | LineBucket\n): number {\n const value = ((layer.paint as any).get(property) as PossiblyEvaluatedPropertyValue).value;\n if (value.kind === 'constant') {\n return value.value;\n } else {\n return bucket.programConfigurations.get(layer.id).getMaxValue(property);\n }\n}\n\nexport function translateDistance(translate: [number, number]) {\n return Math.sqrt(translate[0] * translate[0] + translate[1] * translate[1]);\n}\n\nexport function translate(queryGeometry: Array,\n translate: [number, number],\n translateAnchor: 'viewport' | 'map',\n bearing: number,\n pixelsToTileUnits: number) {\n if (!translate[0] && !translate[1]) {\n return queryGeometry;\n }\n const pt = Point.convert(translate)._mult(pixelsToTileUnits);\n\n if (translateAnchor === 'viewport') {\n pt._rotate(-bearing);\n }\n\n const translated = [];\n for (let i = 0; i < queryGeometry.length; i++) {\n const point = queryGeometry[i];\n translated.push(point.sub(pt));\n }\n return translated;\n}\n\nexport function offsetLine(rings: Array>, offset: number) {\n const newRings: Array> = [];\n for (let ringIndex = 0; ringIndex < rings.length; ringIndex++) {\n const ring = rings[ringIndex];\n const newRing: Array = [];\n for (let index = 0; index < ring.length; index++) {\n const a = ring[index - 1];\n const b = ring[index];\n const c = ring[index + 1];\n const aToB = index === 0 ? new Point(0, 0) : b.sub(a)._unit()._perp();\n const bToC = index === ring.length - 1 ? new Point(0, 0) : c.sub(b)._unit()._perp();\n const extrude = aToB._add(bToC)._unit();\n\n const cosHalfAngle = extrude.x * bToC.x + extrude.y * bToC.y;\n if (cosHalfAngle !== 0) {\n extrude._mult(1 / cosHalfAngle);\n }\n\n newRing.push(extrude._mult(offset)._add(b));\n }\n newRings.push(newRing);\n }\n return newRings;\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type CircleLayoutProps = {\n \"circle-sort-key\": DataDrivenProperty,\n};\n\nexport type CircleLayoutPropsPossiblyEvaluated = {\n \"circle-sort-key\": PossiblyEvaluatedPropertyValue,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"circle-sort-key\": new DataDrivenProperty(styleSpec[\"layout_circle\"][\"circle-sort-key\"] as any as StylePropertySpecification),\n});\n\nexport type CirclePaintProps = {\n \"circle-radius\": DataDrivenProperty,\n \"circle-color\": DataDrivenProperty,\n \"circle-blur\": DataDrivenProperty,\n \"circle-opacity\": DataDrivenProperty,\n \"circle-translate\": DataConstantProperty<[number, number]>,\n \"circle-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"circle-pitch-scale\": DataConstantProperty<\"map\" | \"viewport\">,\n \"circle-pitch-alignment\": DataConstantProperty<\"map\" | \"viewport\">,\n \"circle-stroke-width\": DataDrivenProperty,\n \"circle-stroke-color\": DataDrivenProperty,\n \"circle-stroke-opacity\": DataDrivenProperty,\n};\n\nexport type CirclePaintPropsPossiblyEvaluated = {\n \"circle-radius\": PossiblyEvaluatedPropertyValue,\n \"circle-color\": PossiblyEvaluatedPropertyValue,\n \"circle-blur\": PossiblyEvaluatedPropertyValue,\n \"circle-opacity\": PossiblyEvaluatedPropertyValue,\n \"circle-translate\": [number, number],\n \"circle-translate-anchor\": \"map\" | \"viewport\",\n \"circle-pitch-scale\": \"map\" | \"viewport\",\n \"circle-pitch-alignment\": \"map\" | \"viewport\",\n \"circle-stroke-width\": PossiblyEvaluatedPropertyValue,\n \"circle-stroke-color\": PossiblyEvaluatedPropertyValue,\n \"circle-stroke-opacity\": PossiblyEvaluatedPropertyValue,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"circle-radius\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-radius\"] as any as StylePropertySpecification),\n \"circle-color\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-color\"] as any as StylePropertySpecification),\n \"circle-blur\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-blur\"] as any as StylePropertySpecification),\n \"circle-opacity\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-opacity\"] as any as StylePropertySpecification),\n \"circle-translate\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-translate\"] as any as StylePropertySpecification),\n \"circle-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-translate-anchor\"] as any as StylePropertySpecification),\n \"circle-pitch-scale\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-pitch-scale\"] as any as StylePropertySpecification),\n \"circle-pitch-alignment\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-pitch-alignment\"] as any as StylePropertySpecification),\n \"circle-stroke-width\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-stroke-width\"] as any as StylePropertySpecification),\n \"circle-stroke-color\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-stroke-color\"] as any as StylePropertySpecification),\n \"circle-stroke-opacity\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-stroke-opacity\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","/**\n * Common utilities\n * @module glMatrix\n */\n// Configuration Constants\nexport var EPSILON = 0.000001;\nexport var ARRAY_TYPE = typeof Float32Array !== 'undefined' ? Float32Array : Array;\nexport var RANDOM = Math.random;\n/**\n * Sets the type of array used when creating new vectors and matrices\n *\n * @param {Float32ArrayConstructor | ArrayConstructor} type Array type, such as Float32Array or Array\n */\n\nexport function setMatrixArrayType(type) {\n ARRAY_TYPE = type;\n}\nvar degree = Math.PI / 180;\n/**\n * Convert Degree To Radian\n *\n * @param {Number} a Angle in Degrees\n */\n\nexport function toRadian(a) {\n return a * degree;\n}\n/**\n * Tests whether or not the arguments have approximately the same value, within an absolute\n * or relative tolerance of glMatrix.EPSILON (an absolute tolerance is used for values less\n * than or equal to 1.0, and a relative tolerance is used for larger values)\n *\n * @param {Number} a The first number to test.\n * @param {Number} b The second number to test.\n * @returns {Boolean} True if the numbers are approximately equal, false otherwise.\n */\n\nexport function equals(a, b) {\n return Math.abs(a - b) <= EPSILON * Math.max(1.0, Math.abs(a), Math.abs(b));\n}\nif (!Math.hypot) Math.hypot = function () {\n var y = 0,\n i = arguments.length;\n\n while (i--) {\n y += arguments[i] * arguments[i];\n }\n\n return Math.sqrt(y);\n};","import * as glMatrix from \"./common.js\";\n/**\n * 4x4 Matrix
Format: column-major, when typed out it looks like row-major
The matrices are being post multiplied.\n * @module mat4\n */\n\n/**\n * Creates a new identity mat4\n *\n * @returns {mat4} a new 4x4 matrix\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(16);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n }\n\n out[0] = 1;\n out[5] = 1;\n out[10] = 1;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a new mat4 initialized with values from an existing matrix\n *\n * @param {ReadonlyMat4} a matrix to clone\n * @returns {mat4} a new 4x4 matrix\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(16);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n out[9] = a[9];\n out[10] = a[10];\n out[11] = a[11];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n return out;\n}\n/**\n * Copy the values from one mat4 to another\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n out[9] = a[9];\n out[10] = a[10];\n out[11] = a[11];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n return out;\n}\n/**\n * Create a new mat4 with the given values\n *\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m03 Component in column 0, row 3 position (index 3)\n * @param {Number} m10 Component in column 1, row 0 position (index 4)\n * @param {Number} m11 Component in column 1, row 1 position (index 5)\n * @param {Number} m12 Component in column 1, row 2 position (index 6)\n * @param {Number} m13 Component in column 1, row 3 position (index 7)\n * @param {Number} m20 Component in column 2, row 0 position (index 8)\n * @param {Number} m21 Component in column 2, row 1 position (index 9)\n * @param {Number} m22 Component in column 2, row 2 position (index 10)\n * @param {Number} m23 Component in column 2, row 3 position (index 11)\n * @param {Number} m30 Component in column 3, row 0 position (index 12)\n * @param {Number} m31 Component in column 3, row 1 position (index 13)\n * @param {Number} m32 Component in column 3, row 2 position (index 14)\n * @param {Number} m33 Component in column 3, row 3 position (index 15)\n * @returns {mat4} A new mat4\n */\n\nexport function fromValues(m00, m01, m02, m03, m10, m11, m12, m13, m20, m21, m22, m23, m30, m31, m32, m33) {\n var out = new glMatrix.ARRAY_TYPE(16);\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m03;\n out[4] = m10;\n out[5] = m11;\n out[6] = m12;\n out[7] = m13;\n out[8] = m20;\n out[9] = m21;\n out[10] = m22;\n out[11] = m23;\n out[12] = m30;\n out[13] = m31;\n out[14] = m32;\n out[15] = m33;\n return out;\n}\n/**\n * Set the components of a mat4 to the given values\n *\n * @param {mat4} out the receiving matrix\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m03 Component in column 0, row 3 position (index 3)\n * @param {Number} m10 Component in column 1, row 0 position (index 4)\n * @param {Number} m11 Component in column 1, row 1 position (index 5)\n * @param {Number} m12 Component in column 1, row 2 position (index 6)\n * @param {Number} m13 Component in column 1, row 3 position (index 7)\n * @param {Number} m20 Component in column 2, row 0 position (index 8)\n * @param {Number} m21 Component in column 2, row 1 position (index 9)\n * @param {Number} m22 Component in column 2, row 2 position (index 10)\n * @param {Number} m23 Component in column 2, row 3 position (index 11)\n * @param {Number} m30 Component in column 3, row 0 position (index 12)\n * @param {Number} m31 Component in column 3, row 1 position (index 13)\n * @param {Number} m32 Component in column 3, row 2 position (index 14)\n * @param {Number} m33 Component in column 3, row 3 position (index 15)\n * @returns {mat4} out\n */\n\nexport function set(out, m00, m01, m02, m03, m10, m11, m12, m13, m20, m21, m22, m23, m30, m31, m32, m33) {\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m03;\n out[4] = m10;\n out[5] = m11;\n out[6] = m12;\n out[7] = m13;\n out[8] = m20;\n out[9] = m21;\n out[10] = m22;\n out[11] = m23;\n out[12] = m30;\n out[13] = m31;\n out[14] = m32;\n out[15] = m33;\n return out;\n}\n/**\n * Set a mat4 to the identity matrix\n *\n * @param {mat4} out the receiving matrix\n * @returns {mat4} out\n */\n\nexport function identity(out) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = 1;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 1;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Transpose the values of a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function transpose(out, a) {\n // If we are transposing ourselves we can skip a few steps but have to cache some values\n if (out === a) {\n var a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a12 = a[6],\n a13 = a[7];\n var a23 = a[11];\n out[1] = a[4];\n out[2] = a[8];\n out[3] = a[12];\n out[4] = a01;\n out[6] = a[9];\n out[7] = a[13];\n out[8] = a02;\n out[9] = a12;\n out[11] = a[14];\n out[12] = a03;\n out[13] = a13;\n out[14] = a23;\n } else {\n out[0] = a[0];\n out[1] = a[4];\n out[2] = a[8];\n out[3] = a[12];\n out[4] = a[1];\n out[5] = a[5];\n out[6] = a[9];\n out[7] = a[13];\n out[8] = a[2];\n out[9] = a[6];\n out[10] = a[10];\n out[11] = a[14];\n out[12] = a[3];\n out[13] = a[7];\n out[14] = a[11];\n out[15] = a[15];\n }\n\n return out;\n}\n/**\n * Inverts a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function invert(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n var b00 = a00 * a11 - a01 * a10;\n var b01 = a00 * a12 - a02 * a10;\n var b02 = a00 * a13 - a03 * a10;\n var b03 = a01 * a12 - a02 * a11;\n var b04 = a01 * a13 - a03 * a11;\n var b05 = a02 * a13 - a03 * a12;\n var b06 = a20 * a31 - a21 * a30;\n var b07 = a20 * a32 - a22 * a30;\n var b08 = a20 * a33 - a23 * a30;\n var b09 = a21 * a32 - a22 * a31;\n var b10 = a21 * a33 - a23 * a31;\n var b11 = a22 * a33 - a23 * a32; // Calculate the determinant\n\n var det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det;\n out[1] = (a02 * b10 - a01 * b11 - a03 * b09) * det;\n out[2] = (a31 * b05 - a32 * b04 + a33 * b03) * det;\n out[3] = (a22 * b04 - a21 * b05 - a23 * b03) * det;\n out[4] = (a12 * b08 - a10 * b11 - a13 * b07) * det;\n out[5] = (a00 * b11 - a02 * b08 + a03 * b07) * det;\n out[6] = (a32 * b02 - a30 * b05 - a33 * b01) * det;\n out[7] = (a20 * b05 - a22 * b02 + a23 * b01) * det;\n out[8] = (a10 * b10 - a11 * b08 + a13 * b06) * det;\n out[9] = (a01 * b08 - a00 * b10 - a03 * b06) * det;\n out[10] = (a30 * b04 - a31 * b02 + a33 * b00) * det;\n out[11] = (a21 * b02 - a20 * b04 - a23 * b00) * det;\n out[12] = (a11 * b07 - a10 * b09 - a12 * b06) * det;\n out[13] = (a00 * b09 - a01 * b07 + a02 * b06) * det;\n out[14] = (a31 * b01 - a30 * b03 - a32 * b00) * det;\n out[15] = (a20 * b03 - a21 * b01 + a22 * b00) * det;\n return out;\n}\n/**\n * Calculates the adjugate of a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function adjoint(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n out[0] = a11 * (a22 * a33 - a23 * a32) - a21 * (a12 * a33 - a13 * a32) + a31 * (a12 * a23 - a13 * a22);\n out[1] = -(a01 * (a22 * a33 - a23 * a32) - a21 * (a02 * a33 - a03 * a32) + a31 * (a02 * a23 - a03 * a22));\n out[2] = a01 * (a12 * a33 - a13 * a32) - a11 * (a02 * a33 - a03 * a32) + a31 * (a02 * a13 - a03 * a12);\n out[3] = -(a01 * (a12 * a23 - a13 * a22) - a11 * (a02 * a23 - a03 * a22) + a21 * (a02 * a13 - a03 * a12));\n out[4] = -(a10 * (a22 * a33 - a23 * a32) - a20 * (a12 * a33 - a13 * a32) + a30 * (a12 * a23 - a13 * a22));\n out[5] = a00 * (a22 * a33 - a23 * a32) - a20 * (a02 * a33 - a03 * a32) + a30 * (a02 * a23 - a03 * a22);\n out[6] = -(a00 * (a12 * a33 - a13 * a32) - a10 * (a02 * a33 - a03 * a32) + a30 * (a02 * a13 - a03 * a12));\n out[7] = a00 * (a12 * a23 - a13 * a22) - a10 * (a02 * a23 - a03 * a22) + a20 * (a02 * a13 - a03 * a12);\n out[8] = a10 * (a21 * a33 - a23 * a31) - a20 * (a11 * a33 - a13 * a31) + a30 * (a11 * a23 - a13 * a21);\n out[9] = -(a00 * (a21 * a33 - a23 * a31) - a20 * (a01 * a33 - a03 * a31) + a30 * (a01 * a23 - a03 * a21));\n out[10] = a00 * (a11 * a33 - a13 * a31) - a10 * (a01 * a33 - a03 * a31) + a30 * (a01 * a13 - a03 * a11);\n out[11] = -(a00 * (a11 * a23 - a13 * a21) - a10 * (a01 * a23 - a03 * a21) + a20 * (a01 * a13 - a03 * a11));\n out[12] = -(a10 * (a21 * a32 - a22 * a31) - a20 * (a11 * a32 - a12 * a31) + a30 * (a11 * a22 - a12 * a21));\n out[13] = a00 * (a21 * a32 - a22 * a31) - a20 * (a01 * a32 - a02 * a31) + a30 * (a01 * a22 - a02 * a21);\n out[14] = -(a00 * (a11 * a32 - a12 * a31) - a10 * (a01 * a32 - a02 * a31) + a30 * (a01 * a12 - a02 * a11));\n out[15] = a00 * (a11 * a22 - a12 * a21) - a10 * (a01 * a22 - a02 * a21) + a20 * (a01 * a12 - a02 * a11);\n return out;\n}\n/**\n * Calculates the determinant of a mat4\n *\n * @param {ReadonlyMat4} a the source matrix\n * @returns {Number} determinant of a\n */\n\nexport function determinant(a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n var b00 = a00 * a11 - a01 * a10;\n var b01 = a00 * a12 - a02 * a10;\n var b02 = a00 * a13 - a03 * a10;\n var b03 = a01 * a12 - a02 * a11;\n var b04 = a01 * a13 - a03 * a11;\n var b05 = a02 * a13 - a03 * a12;\n var b06 = a20 * a31 - a21 * a30;\n var b07 = a20 * a32 - a22 * a30;\n var b08 = a20 * a33 - a23 * a30;\n var b09 = a21 * a32 - a22 * a31;\n var b10 = a21 * a33 - a23 * a31;\n var b11 = a22 * a33 - a23 * a32; // Calculate the determinant\n\n return b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n}\n/**\n * Multiplies two mat4s\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @returns {mat4} out\n */\n\nexport function multiply(out, a, b) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15]; // Cache only the current line of the second matrix\n\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n out[0] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[1] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[2] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[3] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n b0 = b[4];\n b1 = b[5];\n b2 = b[6];\n b3 = b[7];\n out[4] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[5] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[6] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[7] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n b0 = b[8];\n b1 = b[9];\n b2 = b[10];\n b3 = b[11];\n out[8] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[9] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[10] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[11] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n b0 = b[12];\n b1 = b[13];\n b2 = b[14];\n b3 = b[15];\n out[12] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[13] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[14] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[15] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n return out;\n}\n/**\n * Translate a mat4 by the given vector\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to translate\n * @param {ReadonlyVec3} v vector to translate by\n * @returns {mat4} out\n */\n\nexport function translate(out, a, v) {\n var x = v[0],\n y = v[1],\n z = v[2];\n var a00, a01, a02, a03;\n var a10, a11, a12, a13;\n var a20, a21, a22, a23;\n\n if (a === out) {\n out[12] = a[0] * x + a[4] * y + a[8] * z + a[12];\n out[13] = a[1] * x + a[5] * y + a[9] * z + a[13];\n out[14] = a[2] * x + a[6] * y + a[10] * z + a[14];\n out[15] = a[3] * x + a[7] * y + a[11] * z + a[15];\n } else {\n a00 = a[0];\n a01 = a[1];\n a02 = a[2];\n a03 = a[3];\n a10 = a[4];\n a11 = a[5];\n a12 = a[6];\n a13 = a[7];\n a20 = a[8];\n a21 = a[9];\n a22 = a[10];\n a23 = a[11];\n out[0] = a00;\n out[1] = a01;\n out[2] = a02;\n out[3] = a03;\n out[4] = a10;\n out[5] = a11;\n out[6] = a12;\n out[7] = a13;\n out[8] = a20;\n out[9] = a21;\n out[10] = a22;\n out[11] = a23;\n out[12] = a00 * x + a10 * y + a20 * z + a[12];\n out[13] = a01 * x + a11 * y + a21 * z + a[13];\n out[14] = a02 * x + a12 * y + a22 * z + a[14];\n out[15] = a03 * x + a13 * y + a23 * z + a[15];\n }\n\n return out;\n}\n/**\n * Scales the mat4 by the dimensions in the given vec3 not using vectorization\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to scale\n * @param {ReadonlyVec3} v the vec3 to scale the matrix by\n * @returns {mat4} out\n **/\n\nexport function scale(out, a, v) {\n var x = v[0],\n y = v[1],\n z = v[2];\n out[0] = a[0] * x;\n out[1] = a[1] * x;\n out[2] = a[2] * x;\n out[3] = a[3] * x;\n out[4] = a[4] * y;\n out[5] = a[5] * y;\n out[6] = a[6] * y;\n out[7] = a[7] * y;\n out[8] = a[8] * z;\n out[9] = a[9] * z;\n out[10] = a[10] * z;\n out[11] = a[11] * z;\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n return out;\n}\n/**\n * Rotates a mat4 by the given angle around the given axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @param {ReadonlyVec3} axis the axis to rotate around\n * @returns {mat4} out\n */\n\nexport function rotate(out, a, rad, axis) {\n var x = axis[0],\n y = axis[1],\n z = axis[2];\n var len = Math.hypot(x, y, z);\n var s, c, t;\n var a00, a01, a02, a03;\n var a10, a11, a12, a13;\n var a20, a21, a22, a23;\n var b00, b01, b02;\n var b10, b11, b12;\n var b20, b21, b22;\n\n if (len < glMatrix.EPSILON) {\n return null;\n }\n\n len = 1 / len;\n x *= len;\n y *= len;\n z *= len;\n s = Math.sin(rad);\n c = Math.cos(rad);\n t = 1 - c;\n a00 = a[0];\n a01 = a[1];\n a02 = a[2];\n a03 = a[3];\n a10 = a[4];\n a11 = a[5];\n a12 = a[6];\n a13 = a[7];\n a20 = a[8];\n a21 = a[9];\n a22 = a[10];\n a23 = a[11]; // Construct the elements of the rotation matrix\n\n b00 = x * x * t + c;\n b01 = y * x * t + z * s;\n b02 = z * x * t - y * s;\n b10 = x * y * t - z * s;\n b11 = y * y * t + c;\n b12 = z * y * t + x * s;\n b20 = x * z * t + y * s;\n b21 = y * z * t - x * s;\n b22 = z * z * t + c; // Perform rotation-specific matrix multiplication\n\n out[0] = a00 * b00 + a10 * b01 + a20 * b02;\n out[1] = a01 * b00 + a11 * b01 + a21 * b02;\n out[2] = a02 * b00 + a12 * b01 + a22 * b02;\n out[3] = a03 * b00 + a13 * b01 + a23 * b02;\n out[4] = a00 * b10 + a10 * b11 + a20 * b12;\n out[5] = a01 * b10 + a11 * b11 + a21 * b12;\n out[6] = a02 * b10 + a12 * b11 + a22 * b12;\n out[7] = a03 * b10 + a13 * b11 + a23 * b12;\n out[8] = a00 * b20 + a10 * b21 + a20 * b22;\n out[9] = a01 * b20 + a11 * b21 + a21 * b22;\n out[10] = a02 * b20 + a12 * b21 + a22 * b22;\n out[11] = a03 * b20 + a13 * b21 + a23 * b22;\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged last row\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n }\n\n return out;\n}\n/**\n * Rotates a matrix by the given angle around the X axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function rotateX(out, a, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n var a10 = a[4];\n var a11 = a[5];\n var a12 = a[6];\n var a13 = a[7];\n var a20 = a[8];\n var a21 = a[9];\n var a22 = a[10];\n var a23 = a[11];\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged rows\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n } // Perform axis-specific matrix multiplication\n\n\n out[4] = a10 * c + a20 * s;\n out[5] = a11 * c + a21 * s;\n out[6] = a12 * c + a22 * s;\n out[7] = a13 * c + a23 * s;\n out[8] = a20 * c - a10 * s;\n out[9] = a21 * c - a11 * s;\n out[10] = a22 * c - a12 * s;\n out[11] = a23 * c - a13 * s;\n return out;\n}\n/**\n * Rotates a matrix by the given angle around the Y axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function rotateY(out, a, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n var a00 = a[0];\n var a01 = a[1];\n var a02 = a[2];\n var a03 = a[3];\n var a20 = a[8];\n var a21 = a[9];\n var a22 = a[10];\n var a23 = a[11];\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged rows\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n } // Perform axis-specific matrix multiplication\n\n\n out[0] = a00 * c - a20 * s;\n out[1] = a01 * c - a21 * s;\n out[2] = a02 * c - a22 * s;\n out[3] = a03 * c - a23 * s;\n out[8] = a00 * s + a20 * c;\n out[9] = a01 * s + a21 * c;\n out[10] = a02 * s + a22 * c;\n out[11] = a03 * s + a23 * c;\n return out;\n}\n/**\n * Rotates a matrix by the given angle around the Z axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function rotateZ(out, a, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n var a00 = a[0];\n var a01 = a[1];\n var a02 = a[2];\n var a03 = a[3];\n var a10 = a[4];\n var a11 = a[5];\n var a12 = a[6];\n var a13 = a[7];\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged last row\n out[8] = a[8];\n out[9] = a[9];\n out[10] = a[10];\n out[11] = a[11];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n } // Perform axis-specific matrix multiplication\n\n\n out[0] = a00 * c + a10 * s;\n out[1] = a01 * c + a11 * s;\n out[2] = a02 * c + a12 * s;\n out[3] = a03 * c + a13 * s;\n out[4] = a10 * c - a00 * s;\n out[5] = a11 * c - a01 * s;\n out[6] = a12 * c - a02 * s;\n out[7] = a13 * c - a03 * s;\n return out;\n}\n/**\n * Creates a matrix from a vector translation\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, dest, vec);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {ReadonlyVec3} v Translation vector\n * @returns {mat4} out\n */\n\nexport function fromTranslation(out, v) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = 1;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 1;\n out[11] = 0;\n out[12] = v[0];\n out[13] = v[1];\n out[14] = v[2];\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a vector scaling\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.scale(dest, dest, vec);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {ReadonlyVec3} v Scaling vector\n * @returns {mat4} out\n */\n\nexport function fromScaling(out, v) {\n out[0] = v[0];\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = v[1];\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = v[2];\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a given angle around a given axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotate(dest, dest, rad, axis);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @param {ReadonlyVec3} axis the axis to rotate around\n * @returns {mat4} out\n */\n\nexport function fromRotation(out, rad, axis) {\n var x = axis[0],\n y = axis[1],\n z = axis[2];\n var len = Math.hypot(x, y, z);\n var s, c, t;\n\n if (len < glMatrix.EPSILON) {\n return null;\n }\n\n len = 1 / len;\n x *= len;\n y *= len;\n z *= len;\n s = Math.sin(rad);\n c = Math.cos(rad);\n t = 1 - c; // Perform rotation-specific matrix multiplication\n\n out[0] = x * x * t + c;\n out[1] = y * x * t + z * s;\n out[2] = z * x * t - y * s;\n out[3] = 0;\n out[4] = x * y * t - z * s;\n out[5] = y * y * t + c;\n out[6] = z * y * t + x * s;\n out[7] = 0;\n out[8] = x * z * t + y * s;\n out[9] = y * z * t - x * s;\n out[10] = z * z * t + c;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from the given angle around the X axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotateX(dest, dest, rad);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function fromXRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad); // Perform axis-specific matrix multiplication\n\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = c;\n out[6] = s;\n out[7] = 0;\n out[8] = 0;\n out[9] = -s;\n out[10] = c;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from the given angle around the Y axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotateY(dest, dest, rad);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function fromYRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad); // Perform axis-specific matrix multiplication\n\n out[0] = c;\n out[1] = 0;\n out[2] = -s;\n out[3] = 0;\n out[4] = 0;\n out[5] = 1;\n out[6] = 0;\n out[7] = 0;\n out[8] = s;\n out[9] = 0;\n out[10] = c;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from the given angle around the Z axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotateZ(dest, dest, rad);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function fromZRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad); // Perform axis-specific matrix multiplication\n\n out[0] = c;\n out[1] = s;\n out[2] = 0;\n out[3] = 0;\n out[4] = -s;\n out[5] = c;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 1;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a quaternion rotation and vector translation\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, vec);\n * let quatMat = mat4.create();\n * quat4.toMat4(quat, quatMat);\n * mat4.multiply(dest, quatMat);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {ReadonlyVec3} v Translation vector\n * @returns {mat4} out\n */\n\nexport function fromRotationTranslation(out, q, v) {\n // Quaternion math\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var xy = x * y2;\n var xz = x * z2;\n var yy = y * y2;\n var yz = y * z2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n out[0] = 1 - (yy + zz);\n out[1] = xy + wz;\n out[2] = xz - wy;\n out[3] = 0;\n out[4] = xy - wz;\n out[5] = 1 - (xx + zz);\n out[6] = yz + wx;\n out[7] = 0;\n out[8] = xz + wy;\n out[9] = yz - wx;\n out[10] = 1 - (xx + yy);\n out[11] = 0;\n out[12] = v[0];\n out[13] = v[1];\n out[14] = v[2];\n out[15] = 1;\n return out;\n}\n/**\n * Creates a new mat4 from a dual quat.\n *\n * @param {mat4} out Matrix\n * @param {ReadonlyQuat2} a Dual Quaternion\n * @returns {mat4} mat4 receiving operation result\n */\n\nexport function fromQuat2(out, a) {\n var translation = new glMatrix.ARRAY_TYPE(3);\n var bx = -a[0],\n by = -a[1],\n bz = -a[2],\n bw = a[3],\n ax = a[4],\n ay = a[5],\n az = a[6],\n aw = a[7];\n var magnitude = bx * bx + by * by + bz * bz + bw * bw; //Only scale if it makes sense\n\n if (magnitude > 0) {\n translation[0] = (ax * bw + aw * bx + ay * bz - az * by) * 2 / magnitude;\n translation[1] = (ay * bw + aw * by + az * bx - ax * bz) * 2 / magnitude;\n translation[2] = (az * bw + aw * bz + ax * by - ay * bx) * 2 / magnitude;\n } else {\n translation[0] = (ax * bw + aw * bx + ay * bz - az * by) * 2;\n translation[1] = (ay * bw + aw * by + az * bx - ax * bz) * 2;\n translation[2] = (az * bw + aw * bz + ax * by - ay * bx) * 2;\n }\n\n fromRotationTranslation(out, a, translation);\n return out;\n}\n/**\n * Returns the translation vector component of a transformation\n * matrix. If a matrix is built with fromRotationTranslation,\n * the returned vector will be the same as the translation vector\n * originally supplied.\n * @param {vec3} out Vector to receive translation component\n * @param {ReadonlyMat4} mat Matrix to be decomposed (input)\n * @return {vec3} out\n */\n\nexport function getTranslation(out, mat) {\n out[0] = mat[12];\n out[1] = mat[13];\n out[2] = mat[14];\n return out;\n}\n/**\n * Returns the scaling factor component of a transformation\n * matrix. If a matrix is built with fromRotationTranslationScale\n * with a normalized Quaternion paramter, the returned vector will be\n * the same as the scaling vector\n * originally supplied.\n * @param {vec3} out Vector to receive scaling factor component\n * @param {ReadonlyMat4} mat Matrix to be decomposed (input)\n * @return {vec3} out\n */\n\nexport function getScaling(out, mat) {\n var m11 = mat[0];\n var m12 = mat[1];\n var m13 = mat[2];\n var m21 = mat[4];\n var m22 = mat[5];\n var m23 = mat[6];\n var m31 = mat[8];\n var m32 = mat[9];\n var m33 = mat[10];\n out[0] = Math.hypot(m11, m12, m13);\n out[1] = Math.hypot(m21, m22, m23);\n out[2] = Math.hypot(m31, m32, m33);\n return out;\n}\n/**\n * Returns a quaternion representing the rotational component\n * of a transformation matrix. If a matrix is built with\n * fromRotationTranslation, the returned quaternion will be the\n * same as the quaternion originally supplied.\n * @param {quat} out Quaternion to receive the rotation component\n * @param {ReadonlyMat4} mat Matrix to be decomposed (input)\n * @return {quat} out\n */\n\nexport function getRotation(out, mat) {\n var scaling = new glMatrix.ARRAY_TYPE(3);\n getScaling(scaling, mat);\n var is1 = 1 / scaling[0];\n var is2 = 1 / scaling[1];\n var is3 = 1 / scaling[2];\n var sm11 = mat[0] * is1;\n var sm12 = mat[1] * is2;\n var sm13 = mat[2] * is3;\n var sm21 = mat[4] * is1;\n var sm22 = mat[5] * is2;\n var sm23 = mat[6] * is3;\n var sm31 = mat[8] * is1;\n var sm32 = mat[9] * is2;\n var sm33 = mat[10] * is3;\n var trace = sm11 + sm22 + sm33;\n var S = 0;\n\n if (trace > 0) {\n S = Math.sqrt(trace + 1.0) * 2;\n out[3] = 0.25 * S;\n out[0] = (sm23 - sm32) / S;\n out[1] = (sm31 - sm13) / S;\n out[2] = (sm12 - sm21) / S;\n } else if (sm11 > sm22 && sm11 > sm33) {\n S = Math.sqrt(1.0 + sm11 - sm22 - sm33) * 2;\n out[3] = (sm23 - sm32) / S;\n out[0] = 0.25 * S;\n out[1] = (sm12 + sm21) / S;\n out[2] = (sm31 + sm13) / S;\n } else if (sm22 > sm33) {\n S = Math.sqrt(1.0 + sm22 - sm11 - sm33) * 2;\n out[3] = (sm31 - sm13) / S;\n out[0] = (sm12 + sm21) / S;\n out[1] = 0.25 * S;\n out[2] = (sm23 + sm32) / S;\n } else {\n S = Math.sqrt(1.0 + sm33 - sm11 - sm22) * 2;\n out[3] = (sm12 - sm21) / S;\n out[0] = (sm31 + sm13) / S;\n out[1] = (sm23 + sm32) / S;\n out[2] = 0.25 * S;\n }\n\n return out;\n}\n/**\n * Creates a matrix from a quaternion rotation, vector translation and vector scale\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, vec);\n * let quatMat = mat4.create();\n * quat4.toMat4(quat, quatMat);\n * mat4.multiply(dest, quatMat);\n * mat4.scale(dest, scale)\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {ReadonlyVec3} v Translation vector\n * @param {ReadonlyVec3} s Scaling vector\n * @returns {mat4} out\n */\n\nexport function fromRotationTranslationScale(out, q, v, s) {\n // Quaternion math\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var xy = x * y2;\n var xz = x * z2;\n var yy = y * y2;\n var yz = y * z2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n var sx = s[0];\n var sy = s[1];\n var sz = s[2];\n out[0] = (1 - (yy + zz)) * sx;\n out[1] = (xy + wz) * sx;\n out[2] = (xz - wy) * sx;\n out[3] = 0;\n out[4] = (xy - wz) * sy;\n out[5] = (1 - (xx + zz)) * sy;\n out[6] = (yz + wx) * sy;\n out[7] = 0;\n out[8] = (xz + wy) * sz;\n out[9] = (yz - wx) * sz;\n out[10] = (1 - (xx + yy)) * sz;\n out[11] = 0;\n out[12] = v[0];\n out[13] = v[1];\n out[14] = v[2];\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a quaternion rotation, vector translation and vector scale, rotating and scaling around the given origin\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, vec);\n * mat4.translate(dest, origin);\n * let quatMat = mat4.create();\n * quat4.toMat4(quat, quatMat);\n * mat4.multiply(dest, quatMat);\n * mat4.scale(dest, scale)\n * mat4.translate(dest, negativeOrigin);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {ReadonlyVec3} v Translation vector\n * @param {ReadonlyVec3} s Scaling vector\n * @param {ReadonlyVec3} o The origin vector around which to scale and rotate\n * @returns {mat4} out\n */\n\nexport function fromRotationTranslationScaleOrigin(out, q, v, s, o) {\n // Quaternion math\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var xy = x * y2;\n var xz = x * z2;\n var yy = y * y2;\n var yz = y * z2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n var sx = s[0];\n var sy = s[1];\n var sz = s[2];\n var ox = o[0];\n var oy = o[1];\n var oz = o[2];\n var out0 = (1 - (yy + zz)) * sx;\n var out1 = (xy + wz) * sx;\n var out2 = (xz - wy) * sx;\n var out4 = (xy - wz) * sy;\n var out5 = (1 - (xx + zz)) * sy;\n var out6 = (yz + wx) * sy;\n var out8 = (xz + wy) * sz;\n var out9 = (yz - wx) * sz;\n var out10 = (1 - (xx + yy)) * sz;\n out[0] = out0;\n out[1] = out1;\n out[2] = out2;\n out[3] = 0;\n out[4] = out4;\n out[5] = out5;\n out[6] = out6;\n out[7] = 0;\n out[8] = out8;\n out[9] = out9;\n out[10] = out10;\n out[11] = 0;\n out[12] = v[0] + ox - (out0 * ox + out4 * oy + out8 * oz);\n out[13] = v[1] + oy - (out1 * ox + out5 * oy + out9 * oz);\n out[14] = v[2] + oz - (out2 * ox + out6 * oy + out10 * oz);\n out[15] = 1;\n return out;\n}\n/**\n * Calculates a 4x4 matrix from the given quaternion\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {ReadonlyQuat} q Quaternion to create matrix from\n *\n * @returns {mat4} out\n */\n\nexport function fromQuat(out, q) {\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var yx = y * x2;\n var yy = y * y2;\n var zx = z * x2;\n var zy = z * y2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n out[0] = 1 - yy - zz;\n out[1] = yx + wz;\n out[2] = zx - wy;\n out[3] = 0;\n out[4] = yx - wz;\n out[5] = 1 - xx - zz;\n out[6] = zy + wx;\n out[7] = 0;\n out[8] = zx + wy;\n out[9] = zy - wx;\n out[10] = 1 - xx - yy;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Generates a frustum matrix with the given bounds\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {Number} left Left bound of the frustum\n * @param {Number} right Right bound of the frustum\n * @param {Number} bottom Bottom bound of the frustum\n * @param {Number} top Top bound of the frustum\n * @param {Number} near Near bound of the frustum\n * @param {Number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function frustum(out, left, right, bottom, top, near, far) {\n var rl = 1 / (right - left);\n var tb = 1 / (top - bottom);\n var nf = 1 / (near - far);\n out[0] = near * 2 * rl;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = near * 2 * tb;\n out[6] = 0;\n out[7] = 0;\n out[8] = (right + left) * rl;\n out[9] = (top + bottom) * tb;\n out[10] = (far + near) * nf;\n out[11] = -1;\n out[12] = 0;\n out[13] = 0;\n out[14] = far * near * 2 * nf;\n out[15] = 0;\n return out;\n}\n/**\n * Generates a perspective projection matrix with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [-1, 1],\n * which matches WebGL/OpenGL's clip volume.\n * Passing null/undefined/no value for far will generate infinite projection matrix.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} fovy Vertical field of view in radians\n * @param {number} aspect Aspect ratio. typically viewport width/height\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum, can be null or Infinity\n * @returns {mat4} out\n */\n\nexport function perspectiveNO(out, fovy, aspect, near, far) {\n var f = 1.0 / Math.tan(fovy / 2),\n nf;\n out[0] = f / aspect;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = f;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[11] = -1;\n out[12] = 0;\n out[13] = 0;\n out[15] = 0;\n\n if (far != null && far !== Infinity) {\n nf = 1 / (near - far);\n out[10] = (far + near) * nf;\n out[14] = 2 * far * near * nf;\n } else {\n out[10] = -1;\n out[14] = -2 * near;\n }\n\n return out;\n}\n/**\n * Alias for {@link mat4.perspectiveNO}\n * @function\n */\n\nexport var perspective = perspectiveNO;\n/**\n * Generates a perspective projection matrix suitable for WebGPU with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [0, 1],\n * which matches WebGPU/Vulkan/DirectX/Metal's clip volume.\n * Passing null/undefined/no value for far will generate infinite projection matrix.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} fovy Vertical field of view in radians\n * @param {number} aspect Aspect ratio. typically viewport width/height\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum, can be null or Infinity\n * @returns {mat4} out\n */\n\nexport function perspectiveZO(out, fovy, aspect, near, far) {\n var f = 1.0 / Math.tan(fovy / 2),\n nf;\n out[0] = f / aspect;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = f;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[11] = -1;\n out[12] = 0;\n out[13] = 0;\n out[15] = 0;\n\n if (far != null && far !== Infinity) {\n nf = 1 / (near - far);\n out[10] = far * nf;\n out[14] = far * near * nf;\n } else {\n out[10] = -1;\n out[14] = -near;\n }\n\n return out;\n}\n/**\n * Generates a perspective projection matrix with the given field of view.\n * This is primarily useful for generating projection matrices to be used\n * with the still experiemental WebVR API.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {Object} fov Object containing the following values: upDegrees, downDegrees, leftDegrees, rightDegrees\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function perspectiveFromFieldOfView(out, fov, near, far) {\n var upTan = Math.tan(fov.upDegrees * Math.PI / 180.0);\n var downTan = Math.tan(fov.downDegrees * Math.PI / 180.0);\n var leftTan = Math.tan(fov.leftDegrees * Math.PI / 180.0);\n var rightTan = Math.tan(fov.rightDegrees * Math.PI / 180.0);\n var xScale = 2.0 / (leftTan + rightTan);\n var yScale = 2.0 / (upTan + downTan);\n out[0] = xScale;\n out[1] = 0.0;\n out[2] = 0.0;\n out[3] = 0.0;\n out[4] = 0.0;\n out[5] = yScale;\n out[6] = 0.0;\n out[7] = 0.0;\n out[8] = -((leftTan - rightTan) * xScale * 0.5);\n out[9] = (upTan - downTan) * yScale * 0.5;\n out[10] = far / (near - far);\n out[11] = -1.0;\n out[12] = 0.0;\n out[13] = 0.0;\n out[14] = far * near / (near - far);\n out[15] = 0.0;\n return out;\n}\n/**\n * Generates a orthogonal projection matrix with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [-1, 1],\n * which matches WebGL/OpenGL's clip volume.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} left Left bound of the frustum\n * @param {number} right Right bound of the frustum\n * @param {number} bottom Bottom bound of the frustum\n * @param {number} top Top bound of the frustum\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function orthoNO(out, left, right, bottom, top, near, far) {\n var lr = 1 / (left - right);\n var bt = 1 / (bottom - top);\n var nf = 1 / (near - far);\n out[0] = -2 * lr;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = -2 * bt;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 2 * nf;\n out[11] = 0;\n out[12] = (left + right) * lr;\n out[13] = (top + bottom) * bt;\n out[14] = (far + near) * nf;\n out[15] = 1;\n return out;\n}\n/**\n * Alias for {@link mat4.orthoNO}\n * @function\n */\n\nexport var ortho = orthoNO;\n/**\n * Generates a orthogonal projection matrix with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [0, 1],\n * which matches WebGPU/Vulkan/DirectX/Metal's clip volume.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} left Left bound of the frustum\n * @param {number} right Right bound of the frustum\n * @param {number} bottom Bottom bound of the frustum\n * @param {number} top Top bound of the frustum\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function orthoZO(out, left, right, bottom, top, near, far) {\n var lr = 1 / (left - right);\n var bt = 1 / (bottom - top);\n var nf = 1 / (near - far);\n out[0] = -2 * lr;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = -2 * bt;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = nf;\n out[11] = 0;\n out[12] = (left + right) * lr;\n out[13] = (top + bottom) * bt;\n out[14] = near * nf;\n out[15] = 1;\n return out;\n}\n/**\n * Generates a look-at matrix with the given eye position, focal point, and up axis.\n * If you want a matrix that actually makes an object look at another object, you should use targetTo instead.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {ReadonlyVec3} eye Position of the viewer\n * @param {ReadonlyVec3} center Point the viewer is looking at\n * @param {ReadonlyVec3} up vec3 pointing up\n * @returns {mat4} out\n */\n\nexport function lookAt(out, eye, center, up) {\n var x0, x1, x2, y0, y1, y2, z0, z1, z2, len;\n var eyex = eye[0];\n var eyey = eye[1];\n var eyez = eye[2];\n var upx = up[0];\n var upy = up[1];\n var upz = up[2];\n var centerx = center[0];\n var centery = center[1];\n var centerz = center[2];\n\n if (Math.abs(eyex - centerx) < glMatrix.EPSILON && Math.abs(eyey - centery) < glMatrix.EPSILON && Math.abs(eyez - centerz) < glMatrix.EPSILON) {\n return identity(out);\n }\n\n z0 = eyex - centerx;\n z1 = eyey - centery;\n z2 = eyez - centerz;\n len = 1 / Math.hypot(z0, z1, z2);\n z0 *= len;\n z1 *= len;\n z2 *= len;\n x0 = upy * z2 - upz * z1;\n x1 = upz * z0 - upx * z2;\n x2 = upx * z1 - upy * z0;\n len = Math.hypot(x0, x1, x2);\n\n if (!len) {\n x0 = 0;\n x1 = 0;\n x2 = 0;\n } else {\n len = 1 / len;\n x0 *= len;\n x1 *= len;\n x2 *= len;\n }\n\n y0 = z1 * x2 - z2 * x1;\n y1 = z2 * x0 - z0 * x2;\n y2 = z0 * x1 - z1 * x0;\n len = Math.hypot(y0, y1, y2);\n\n if (!len) {\n y0 = 0;\n y1 = 0;\n y2 = 0;\n } else {\n len = 1 / len;\n y0 *= len;\n y1 *= len;\n y2 *= len;\n }\n\n out[0] = x0;\n out[1] = y0;\n out[2] = z0;\n out[3] = 0;\n out[4] = x1;\n out[5] = y1;\n out[6] = z1;\n out[7] = 0;\n out[8] = x2;\n out[9] = y2;\n out[10] = z2;\n out[11] = 0;\n out[12] = -(x0 * eyex + x1 * eyey + x2 * eyez);\n out[13] = -(y0 * eyex + y1 * eyey + y2 * eyez);\n out[14] = -(z0 * eyex + z1 * eyey + z2 * eyez);\n out[15] = 1;\n return out;\n}\n/**\n * Generates a matrix that makes something look at something else.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {ReadonlyVec3} eye Position of the viewer\n * @param {ReadonlyVec3} center Point the viewer is looking at\n * @param {ReadonlyVec3} up vec3 pointing up\n * @returns {mat4} out\n */\n\nexport function targetTo(out, eye, target, up) {\n var eyex = eye[0],\n eyey = eye[1],\n eyez = eye[2],\n upx = up[0],\n upy = up[1],\n upz = up[2];\n var z0 = eyex - target[0],\n z1 = eyey - target[1],\n z2 = eyez - target[2];\n var len = z0 * z0 + z1 * z1 + z2 * z2;\n\n if (len > 0) {\n len = 1 / Math.sqrt(len);\n z0 *= len;\n z1 *= len;\n z2 *= len;\n }\n\n var x0 = upy * z2 - upz * z1,\n x1 = upz * z0 - upx * z2,\n x2 = upx * z1 - upy * z0;\n len = x0 * x0 + x1 * x1 + x2 * x2;\n\n if (len > 0) {\n len = 1 / Math.sqrt(len);\n x0 *= len;\n x1 *= len;\n x2 *= len;\n }\n\n out[0] = x0;\n out[1] = x1;\n out[2] = x2;\n out[3] = 0;\n out[4] = z1 * x2 - z2 * x1;\n out[5] = z2 * x0 - z0 * x2;\n out[6] = z0 * x1 - z1 * x0;\n out[7] = 0;\n out[8] = z0;\n out[9] = z1;\n out[10] = z2;\n out[11] = 0;\n out[12] = eyex;\n out[13] = eyey;\n out[14] = eyez;\n out[15] = 1;\n return out;\n}\n/**\n * Returns a string representation of a mat4\n *\n * @param {ReadonlyMat4} a matrix to represent as a string\n * @returns {String} string representation of the matrix\n */\n\nexport function str(a) {\n return \"mat4(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \", \" + a[4] + \", \" + a[5] + \", \" + a[6] + \", \" + a[7] + \", \" + a[8] + \", \" + a[9] + \", \" + a[10] + \", \" + a[11] + \", \" + a[12] + \", \" + a[13] + \", \" + a[14] + \", \" + a[15] + \")\";\n}\n/**\n * Returns Frobenius norm of a mat4\n *\n * @param {ReadonlyMat4} a the matrix to calculate Frobenius norm of\n * @returns {Number} Frobenius norm\n */\n\nexport function frob(a) {\n return Math.hypot(a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8], a[9], a[10], a[11], a[12], a[13], a[14], a[15]);\n}\n/**\n * Adds two mat4's\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @returns {mat4} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n out[4] = a[4] + b[4];\n out[5] = a[5] + b[5];\n out[6] = a[6] + b[6];\n out[7] = a[7] + b[7];\n out[8] = a[8] + b[8];\n out[9] = a[9] + b[9];\n out[10] = a[10] + b[10];\n out[11] = a[11] + b[11];\n out[12] = a[12] + b[12];\n out[13] = a[13] + b[13];\n out[14] = a[14] + b[14];\n out[15] = a[15] + b[15];\n return out;\n}\n/**\n * Subtracts matrix b from matrix a\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @returns {mat4} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n out[4] = a[4] - b[4];\n out[5] = a[5] - b[5];\n out[6] = a[6] - b[6];\n out[7] = a[7] - b[7];\n out[8] = a[8] - b[8];\n out[9] = a[9] - b[9];\n out[10] = a[10] - b[10];\n out[11] = a[11] - b[11];\n out[12] = a[12] - b[12];\n out[13] = a[13] - b[13];\n out[14] = a[14] - b[14];\n out[15] = a[15] - b[15];\n return out;\n}\n/**\n * Multiply each element of the matrix by a scalar.\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to scale\n * @param {Number} b amount to scale the matrix's elements by\n * @returns {mat4} out\n */\n\nexport function multiplyScalar(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n out[4] = a[4] * b;\n out[5] = a[5] * b;\n out[6] = a[6] * b;\n out[7] = a[7] * b;\n out[8] = a[8] * b;\n out[9] = a[9] * b;\n out[10] = a[10] * b;\n out[11] = a[11] * b;\n out[12] = a[12] * b;\n out[13] = a[13] * b;\n out[14] = a[14] * b;\n out[15] = a[15] * b;\n return out;\n}\n/**\n * Adds two mat4's after multiplying each element of the second operand by a scalar value.\n *\n * @param {mat4} out the receiving vector\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @param {Number} scale the amount to scale b's elements by before adding\n * @returns {mat4} out\n */\n\nexport function multiplyScalarAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n out[4] = a[4] + b[4] * scale;\n out[5] = a[5] + b[5] * scale;\n out[6] = a[6] + b[6] * scale;\n out[7] = a[7] + b[7] * scale;\n out[8] = a[8] + b[8] * scale;\n out[9] = a[9] + b[9] * scale;\n out[10] = a[10] + b[10] * scale;\n out[11] = a[11] + b[11] * scale;\n out[12] = a[12] + b[12] * scale;\n out[13] = a[13] + b[13] * scale;\n out[14] = a[14] + b[14] * scale;\n out[15] = a[15] + b[15] * scale;\n return out;\n}\n/**\n * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyMat4} a The first matrix.\n * @param {ReadonlyMat4} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5] && a[6] === b[6] && a[7] === b[7] && a[8] === b[8] && a[9] === b[9] && a[10] === b[10] && a[11] === b[11] && a[12] === b[12] && a[13] === b[13] && a[14] === b[14] && a[15] === b[15];\n}\n/**\n * Returns whether or not the matrices have approximately the same elements in the same position.\n *\n * @param {ReadonlyMat4} a The first matrix.\n * @param {ReadonlyMat4} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var a4 = a[4],\n a5 = a[5],\n a6 = a[6],\n a7 = a[7];\n var a8 = a[8],\n a9 = a[9],\n a10 = a[10],\n a11 = a[11];\n var a12 = a[12],\n a13 = a[13],\n a14 = a[14],\n a15 = a[15];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n var b4 = b[4],\n b5 = b[5],\n b6 = b[6],\n b7 = b[7];\n var b8 = b[8],\n b9 = b[9],\n b10 = b[10],\n b11 = b[11];\n var b12 = b[12],\n b13 = b[13],\n b14 = b[14],\n b15 = b[15];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5)) && Math.abs(a6 - b6) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a6), Math.abs(b6)) && Math.abs(a7 - b7) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a7), Math.abs(b7)) && Math.abs(a8 - b8) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a8), Math.abs(b8)) && Math.abs(a9 - b9) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a9), Math.abs(b9)) && Math.abs(a10 - b10) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a10), Math.abs(b10)) && Math.abs(a11 - b11) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a11), Math.abs(b11)) && Math.abs(a12 - b12) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a12), Math.abs(b12)) && Math.abs(a13 - b13) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a13), Math.abs(b13)) && Math.abs(a14 - b14) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a14), Math.abs(b14)) && Math.abs(a15 - b15) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a15), Math.abs(b15));\n}\n/**\n * Alias for {@link mat4.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link mat4.subtract}\n * @function\n */\n\nexport var sub = subtract;","import * as glMatrix from \"./common.js\";\n/**\n * 3 Dimensional Vector\n * @module vec3\n */\n\n/**\n * Creates a new, empty vec3\n *\n * @returns {vec3} a new 3D vector\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(3);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[0] = 0;\n out[1] = 0;\n out[2] = 0;\n }\n\n return out;\n}\n/**\n * Creates a new vec3 initialized with values from an existing vector\n *\n * @param {ReadonlyVec3} a vector to clone\n * @returns {vec3} a new 3D vector\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(3);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n return out;\n}\n/**\n * Calculates the length of a vec3\n *\n * @param {ReadonlyVec3} a vector to calculate length of\n * @returns {Number} length of a\n */\n\nexport function length(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n return Math.hypot(x, y, z);\n}\n/**\n * Creates a new vec3 initialized with the given values\n *\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @returns {vec3} a new 3D vector\n */\n\nexport function fromValues(x, y, z) {\n var out = new glMatrix.ARRAY_TYPE(3);\n out[0] = x;\n out[1] = y;\n out[2] = z;\n return out;\n}\n/**\n * Copy the values from one vec3 to another\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the source vector\n * @returns {vec3} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n return out;\n}\n/**\n * Set the components of a vec3 to the given values\n *\n * @param {vec3} out the receiving vector\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @returns {vec3} out\n */\n\nexport function set(out, x, y, z) {\n out[0] = x;\n out[1] = y;\n out[2] = z;\n return out;\n}\n/**\n * Adds two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n return out;\n}\n/**\n * Subtracts vector b from vector a\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n return out;\n}\n/**\n * Multiplies two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function multiply(out, a, b) {\n out[0] = a[0] * b[0];\n out[1] = a[1] * b[1];\n out[2] = a[2] * b[2];\n return out;\n}\n/**\n * Divides two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function divide(out, a, b) {\n out[0] = a[0] / b[0];\n out[1] = a[1] / b[1];\n out[2] = a[2] / b[2];\n return out;\n}\n/**\n * Math.ceil the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to ceil\n * @returns {vec3} out\n */\n\nexport function ceil(out, a) {\n out[0] = Math.ceil(a[0]);\n out[1] = Math.ceil(a[1]);\n out[2] = Math.ceil(a[2]);\n return out;\n}\n/**\n * Math.floor the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to floor\n * @returns {vec3} out\n */\n\nexport function floor(out, a) {\n out[0] = Math.floor(a[0]);\n out[1] = Math.floor(a[1]);\n out[2] = Math.floor(a[2]);\n return out;\n}\n/**\n * Returns the minimum of two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function min(out, a, b) {\n out[0] = Math.min(a[0], b[0]);\n out[1] = Math.min(a[1], b[1]);\n out[2] = Math.min(a[2], b[2]);\n return out;\n}\n/**\n * Returns the maximum of two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function max(out, a, b) {\n out[0] = Math.max(a[0], b[0]);\n out[1] = Math.max(a[1], b[1]);\n out[2] = Math.max(a[2], b[2]);\n return out;\n}\n/**\n * Math.round the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to round\n * @returns {vec3} out\n */\n\nexport function round(out, a) {\n out[0] = Math.round(a[0]);\n out[1] = Math.round(a[1]);\n out[2] = Math.round(a[2]);\n return out;\n}\n/**\n * Scales a vec3 by a scalar number\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the vector to scale\n * @param {Number} b amount to scale the vector by\n * @returns {vec3} out\n */\n\nexport function scale(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n return out;\n}\n/**\n * Adds two vec3's after scaling the second operand by a scalar value\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @param {Number} scale the amount to scale b by before adding\n * @returns {vec3} out\n */\n\nexport function scaleAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n return out;\n}\n/**\n * Calculates the euclidian distance between two vec3's\n *\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {Number} distance between a and b\n */\n\nexport function distance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n return Math.hypot(x, y, z);\n}\n/**\n * Calculates the squared euclidian distance between two vec3's\n *\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {Number} squared distance between a and b\n */\n\nexport function squaredDistance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n return x * x + y * y + z * z;\n}\n/**\n * Calculates the squared length of a vec3\n *\n * @param {ReadonlyVec3} a vector to calculate squared length of\n * @returns {Number} squared length of a\n */\n\nexport function squaredLength(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n return x * x + y * y + z * z;\n}\n/**\n * Negates the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to negate\n * @returns {vec3} out\n */\n\nexport function negate(out, a) {\n out[0] = -a[0];\n out[1] = -a[1];\n out[2] = -a[2];\n return out;\n}\n/**\n * Returns the inverse of the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to invert\n * @returns {vec3} out\n */\n\nexport function inverse(out, a) {\n out[0] = 1.0 / a[0];\n out[1] = 1.0 / a[1];\n out[2] = 1.0 / a[2];\n return out;\n}\n/**\n * Normalize a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to normalize\n * @returns {vec3} out\n */\n\nexport function normalize(out, a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var len = x * x + y * y + z * z;\n\n if (len > 0) {\n //TODO: evaluate use of glm_invsqrt here?\n len = 1 / Math.sqrt(len);\n }\n\n out[0] = a[0] * len;\n out[1] = a[1] * len;\n out[2] = a[2] * len;\n return out;\n}\n/**\n * Calculates the dot product of two vec3's\n *\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {Number} dot product of a and b\n */\n\nexport function dot(a, b) {\n return a[0] * b[0] + a[1] * b[1] + a[2] * b[2];\n}\n/**\n * Computes the cross product of two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function cross(out, a, b) {\n var ax = a[0],\n ay = a[1],\n az = a[2];\n var bx = b[0],\n by = b[1],\n bz = b[2];\n out[0] = ay * bz - az * by;\n out[1] = az * bx - ax * bz;\n out[2] = ax * by - ay * bx;\n return out;\n}\n/**\n * Performs a linear interpolation between two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec3} out\n */\n\nexport function lerp(out, a, b, t) {\n var ax = a[0];\n var ay = a[1];\n var az = a[2];\n out[0] = ax + t * (b[0] - ax);\n out[1] = ay + t * (b[1] - ay);\n out[2] = az + t * (b[2] - az);\n return out;\n}\n/**\n * Performs a hermite interpolation with two control points\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @param {ReadonlyVec3} c the third operand\n * @param {ReadonlyVec3} d the fourth operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec3} out\n */\n\nexport function hermite(out, a, b, c, d, t) {\n var factorTimes2 = t * t;\n var factor1 = factorTimes2 * (2 * t - 3) + 1;\n var factor2 = factorTimes2 * (t - 2) + t;\n var factor3 = factorTimes2 * (t - 1);\n var factor4 = factorTimes2 * (3 - 2 * t);\n out[0] = a[0] * factor1 + b[0] * factor2 + c[0] * factor3 + d[0] * factor4;\n out[1] = a[1] * factor1 + b[1] * factor2 + c[1] * factor3 + d[1] * factor4;\n out[2] = a[2] * factor1 + b[2] * factor2 + c[2] * factor3 + d[2] * factor4;\n return out;\n}\n/**\n * Performs a bezier interpolation with two control points\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @param {ReadonlyVec3} c the third operand\n * @param {ReadonlyVec3} d the fourth operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec3} out\n */\n\nexport function bezier(out, a, b, c, d, t) {\n var inverseFactor = 1 - t;\n var inverseFactorTimesTwo = inverseFactor * inverseFactor;\n var factorTimes2 = t * t;\n var factor1 = inverseFactorTimesTwo * inverseFactor;\n var factor2 = 3 * t * inverseFactorTimesTwo;\n var factor3 = 3 * factorTimes2 * inverseFactor;\n var factor4 = factorTimes2 * t;\n out[0] = a[0] * factor1 + b[0] * factor2 + c[0] * factor3 + d[0] * factor4;\n out[1] = a[1] * factor1 + b[1] * factor2 + c[1] * factor3 + d[1] * factor4;\n out[2] = a[2] * factor1 + b[2] * factor2 + c[2] * factor3 + d[2] * factor4;\n return out;\n}\n/**\n * Generates a random vector with the given scale\n *\n * @param {vec3} out the receiving vector\n * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned\n * @returns {vec3} out\n */\n\nexport function random(out, scale) {\n scale = scale || 1.0;\n var r = glMatrix.RANDOM() * 2.0 * Math.PI;\n var z = glMatrix.RANDOM() * 2.0 - 1.0;\n var zScale = Math.sqrt(1.0 - z * z) * scale;\n out[0] = Math.cos(r) * zScale;\n out[1] = Math.sin(r) * zScale;\n out[2] = z * scale;\n return out;\n}\n/**\n * Transforms the vec3 with a mat4.\n * 4th vector component is implicitly '1'\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the vector to transform\n * @param {ReadonlyMat4} m matrix to transform with\n * @returns {vec3} out\n */\n\nexport function transformMat4(out, a, m) {\n var x = a[0],\n y = a[1],\n z = a[2];\n var w = m[3] * x + m[7] * y + m[11] * z + m[15];\n w = w || 1.0;\n out[0] = (m[0] * x + m[4] * y + m[8] * z + m[12]) / w;\n out[1] = (m[1] * x + m[5] * y + m[9] * z + m[13]) / w;\n out[2] = (m[2] * x + m[6] * y + m[10] * z + m[14]) / w;\n return out;\n}\n/**\n * Transforms the vec3 with a mat3.\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the vector to transform\n * @param {ReadonlyMat3} m the 3x3 matrix to transform with\n * @returns {vec3} out\n */\n\nexport function transformMat3(out, a, m) {\n var x = a[0],\n y = a[1],\n z = a[2];\n out[0] = x * m[0] + y * m[3] + z * m[6];\n out[1] = x * m[1] + y * m[4] + z * m[7];\n out[2] = x * m[2] + y * m[5] + z * m[8];\n return out;\n}\n/**\n * Transforms the vec3 with a quat\n * Can also be used for dual quaternions. (Multiply it with the real part)\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the vector to transform\n * @param {ReadonlyQuat} q quaternion to transform with\n * @returns {vec3} out\n */\n\nexport function transformQuat(out, a, q) {\n // benchmarks: https://jsperf.com/quaternion-transform-vec3-implementations-fixed\n var qx = q[0],\n qy = q[1],\n qz = q[2],\n qw = q[3];\n var x = a[0],\n y = a[1],\n z = a[2]; // var qvec = [qx, qy, qz];\n // var uv = vec3.cross([], qvec, a);\n\n var uvx = qy * z - qz * y,\n uvy = qz * x - qx * z,\n uvz = qx * y - qy * x; // var uuv = vec3.cross([], qvec, uv);\n\n var uuvx = qy * uvz - qz * uvy,\n uuvy = qz * uvx - qx * uvz,\n uuvz = qx * uvy - qy * uvx; // vec3.scale(uv, uv, 2 * w);\n\n var w2 = qw * 2;\n uvx *= w2;\n uvy *= w2;\n uvz *= w2; // vec3.scale(uuv, uuv, 2);\n\n uuvx *= 2;\n uuvy *= 2;\n uuvz *= 2; // return vec3.add(out, a, vec3.add(out, uv, uuv));\n\n out[0] = x + uvx + uuvx;\n out[1] = y + uvy + uuvy;\n out[2] = z + uvz + uuvz;\n return out;\n}\n/**\n * Rotate a 3D vector around the x-axis\n * @param {vec3} out The receiving vec3\n * @param {ReadonlyVec3} a The vec3 point to rotate\n * @param {ReadonlyVec3} b The origin of the rotation\n * @param {Number} rad The angle of rotation in radians\n * @returns {vec3} out\n */\n\nexport function rotateX(out, a, b, rad) {\n var p = [],\n r = []; //Translate point to the origin\n\n p[0] = a[0] - b[0];\n p[1] = a[1] - b[1];\n p[2] = a[2] - b[2]; //perform rotation\n\n r[0] = p[0];\n r[1] = p[1] * Math.cos(rad) - p[2] * Math.sin(rad);\n r[2] = p[1] * Math.sin(rad) + p[2] * Math.cos(rad); //translate to correct position\n\n out[0] = r[0] + b[0];\n out[1] = r[1] + b[1];\n out[2] = r[2] + b[2];\n return out;\n}\n/**\n * Rotate a 3D vector around the y-axis\n * @param {vec3} out The receiving vec3\n * @param {ReadonlyVec3} a The vec3 point to rotate\n * @param {ReadonlyVec3} b The origin of the rotation\n * @param {Number} rad The angle of rotation in radians\n * @returns {vec3} out\n */\n\nexport function rotateY(out, a, b, rad) {\n var p = [],\n r = []; //Translate point to the origin\n\n p[0] = a[0] - b[0];\n p[1] = a[1] - b[1];\n p[2] = a[2] - b[2]; //perform rotation\n\n r[0] = p[2] * Math.sin(rad) + p[0] * Math.cos(rad);\n r[1] = p[1];\n r[2] = p[2] * Math.cos(rad) - p[0] * Math.sin(rad); //translate to correct position\n\n out[0] = r[0] + b[0];\n out[1] = r[1] + b[1];\n out[2] = r[2] + b[2];\n return out;\n}\n/**\n * Rotate a 3D vector around the z-axis\n * @param {vec3} out The receiving vec3\n * @param {ReadonlyVec3} a The vec3 point to rotate\n * @param {ReadonlyVec3} b The origin of the rotation\n * @param {Number} rad The angle of rotation in radians\n * @returns {vec3} out\n */\n\nexport function rotateZ(out, a, b, rad) {\n var p = [],\n r = []; //Translate point to the origin\n\n p[0] = a[0] - b[0];\n p[1] = a[1] - b[1];\n p[2] = a[2] - b[2]; //perform rotation\n\n r[0] = p[0] * Math.cos(rad) - p[1] * Math.sin(rad);\n r[1] = p[0] * Math.sin(rad) + p[1] * Math.cos(rad);\n r[2] = p[2]; //translate to correct position\n\n out[0] = r[0] + b[0];\n out[1] = r[1] + b[1];\n out[2] = r[2] + b[2];\n return out;\n}\n/**\n * Get the angle between two 3D vectors\n * @param {ReadonlyVec3} a The first operand\n * @param {ReadonlyVec3} b The second operand\n * @returns {Number} The angle in radians\n */\n\nexport function angle(a, b) {\n var ax = a[0],\n ay = a[1],\n az = a[2],\n bx = b[0],\n by = b[1],\n bz = b[2],\n mag1 = Math.sqrt(ax * ax + ay * ay + az * az),\n mag2 = Math.sqrt(bx * bx + by * by + bz * bz),\n mag = mag1 * mag2,\n cosine = mag && dot(a, b) / mag;\n return Math.acos(Math.min(Math.max(cosine, -1), 1));\n}\n/**\n * Set the components of a vec3 to zero\n *\n * @param {vec3} out the receiving vector\n * @returns {vec3} out\n */\n\nexport function zero(out) {\n out[0] = 0.0;\n out[1] = 0.0;\n out[2] = 0.0;\n return out;\n}\n/**\n * Returns a string representation of a vector\n *\n * @param {ReadonlyVec3} a vector to represent as a string\n * @returns {String} string representation of the vector\n */\n\nexport function str(a) {\n return \"vec3(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \")\";\n}\n/**\n * Returns whether or not the vectors have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyVec3} a The first vector.\n * @param {ReadonlyVec3} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2];\n}\n/**\n * Returns whether or not the vectors have approximately the same elements in the same position.\n *\n * @param {ReadonlyVec3} a The first vector.\n * @param {ReadonlyVec3} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2));\n}\n/**\n * Alias for {@link vec3.subtract}\n * @function\n */\n\nexport var sub = subtract;\n/**\n * Alias for {@link vec3.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link vec3.divide}\n * @function\n */\n\nexport var div = divide;\n/**\n * Alias for {@link vec3.distance}\n * @function\n */\n\nexport var dist = distance;\n/**\n * Alias for {@link vec3.squaredDistance}\n * @function\n */\n\nexport var sqrDist = squaredDistance;\n/**\n * Alias for {@link vec3.length}\n * @function\n */\n\nexport var len = length;\n/**\n * Alias for {@link vec3.squaredLength}\n * @function\n */\n\nexport var sqrLen = squaredLength;\n/**\n * Perform some operation over an array of vec3s.\n *\n * @param {Array} a the array of vectors to iterate over\n * @param {Number} stride Number of elements between the start of each vec3. If 0 assumes tightly packed\n * @param {Number} offset Number of elements to skip at the beginning of the array\n * @param {Number} count Number of vec3s to iterate over. If 0 iterates over entire array\n * @param {Function} fn Function to call for each vector in the array\n * @param {Object} [arg] additional argument to pass to fn\n * @returns {Array} a\n * @function\n */\n\nexport var forEach = function () {\n var vec = create();\n return function (a, stride, offset, count, fn, arg) {\n var i, l;\n\n if (!stride) {\n stride = 3;\n }\n\n if (!offset) {\n offset = 0;\n }\n\n if (count) {\n l = Math.min(count * stride + offset, a.length);\n } else {\n l = a.length;\n }\n\n for (i = offset; i < l; i += stride) {\n vec[0] = a[i];\n vec[1] = a[i + 1];\n vec[2] = a[i + 2];\n fn(vec, vec, arg);\n a[i] = vec[0];\n a[i + 1] = vec[1];\n a[i + 2] = vec[2];\n }\n\n return a;\n };\n}();","import * as glMatrix from \"./common.js\";\n/**\n * 4 Dimensional Vector\n * @module vec4\n */\n\n/**\n * Creates a new, empty vec4\n *\n * @returns {vec4} a new 4D vector\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(4);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[0] = 0;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n }\n\n return out;\n}\n/**\n * Creates a new vec4 initialized with values from an existing vector\n *\n * @param {ReadonlyVec4} a vector to clone\n * @returns {vec4} a new 4D vector\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Creates a new vec4 initialized with the given values\n *\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @param {Number} w W component\n * @returns {vec4} a new 4D vector\n */\n\nexport function fromValues(x, y, z, w) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = x;\n out[1] = y;\n out[2] = z;\n out[3] = w;\n return out;\n}\n/**\n * Copy the values from one vec4 to another\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the source vector\n * @returns {vec4} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Set the components of a vec4 to the given values\n *\n * @param {vec4} out the receiving vector\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @param {Number} w W component\n * @returns {vec4} out\n */\n\nexport function set(out, x, y, z, w) {\n out[0] = x;\n out[1] = y;\n out[2] = z;\n out[3] = w;\n return out;\n}\n/**\n * Adds two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n return out;\n}\n/**\n * Subtracts vector b from vector a\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n return out;\n}\n/**\n * Multiplies two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function multiply(out, a, b) {\n out[0] = a[0] * b[0];\n out[1] = a[1] * b[1];\n out[2] = a[2] * b[2];\n out[3] = a[3] * b[3];\n return out;\n}\n/**\n * Divides two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function divide(out, a, b) {\n out[0] = a[0] / b[0];\n out[1] = a[1] / b[1];\n out[2] = a[2] / b[2];\n out[3] = a[3] / b[3];\n return out;\n}\n/**\n * Math.ceil the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to ceil\n * @returns {vec4} out\n */\n\nexport function ceil(out, a) {\n out[0] = Math.ceil(a[0]);\n out[1] = Math.ceil(a[1]);\n out[2] = Math.ceil(a[2]);\n out[3] = Math.ceil(a[3]);\n return out;\n}\n/**\n * Math.floor the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to floor\n * @returns {vec4} out\n */\n\nexport function floor(out, a) {\n out[0] = Math.floor(a[0]);\n out[1] = Math.floor(a[1]);\n out[2] = Math.floor(a[2]);\n out[3] = Math.floor(a[3]);\n return out;\n}\n/**\n * Returns the minimum of two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function min(out, a, b) {\n out[0] = Math.min(a[0], b[0]);\n out[1] = Math.min(a[1], b[1]);\n out[2] = Math.min(a[2], b[2]);\n out[3] = Math.min(a[3], b[3]);\n return out;\n}\n/**\n * Returns the maximum of two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function max(out, a, b) {\n out[0] = Math.max(a[0], b[0]);\n out[1] = Math.max(a[1], b[1]);\n out[2] = Math.max(a[2], b[2]);\n out[3] = Math.max(a[3], b[3]);\n return out;\n}\n/**\n * Math.round the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to round\n * @returns {vec4} out\n */\n\nexport function round(out, a) {\n out[0] = Math.round(a[0]);\n out[1] = Math.round(a[1]);\n out[2] = Math.round(a[2]);\n out[3] = Math.round(a[3]);\n return out;\n}\n/**\n * Scales a vec4 by a scalar number\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the vector to scale\n * @param {Number} b amount to scale the vector by\n * @returns {vec4} out\n */\n\nexport function scale(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n return out;\n}\n/**\n * Adds two vec4's after scaling the second operand by a scalar value\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @param {Number} scale the amount to scale b by before adding\n * @returns {vec4} out\n */\n\nexport function scaleAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n return out;\n}\n/**\n * Calculates the euclidian distance between two vec4's\n *\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {Number} distance between a and b\n */\n\nexport function distance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n var w = b[3] - a[3];\n return Math.hypot(x, y, z, w);\n}\n/**\n * Calculates the squared euclidian distance between two vec4's\n *\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {Number} squared distance between a and b\n */\n\nexport function squaredDistance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n var w = b[3] - a[3];\n return x * x + y * y + z * z + w * w;\n}\n/**\n * Calculates the length of a vec4\n *\n * @param {ReadonlyVec4} a vector to calculate length of\n * @returns {Number} length of a\n */\n\nexport function length(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var w = a[3];\n return Math.hypot(x, y, z, w);\n}\n/**\n * Calculates the squared length of a vec4\n *\n * @param {ReadonlyVec4} a vector to calculate squared length of\n * @returns {Number} squared length of a\n */\n\nexport function squaredLength(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var w = a[3];\n return x * x + y * y + z * z + w * w;\n}\n/**\n * Negates the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to negate\n * @returns {vec4} out\n */\n\nexport function negate(out, a) {\n out[0] = -a[0];\n out[1] = -a[1];\n out[2] = -a[2];\n out[3] = -a[3];\n return out;\n}\n/**\n * Returns the inverse of the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to invert\n * @returns {vec4} out\n */\n\nexport function inverse(out, a) {\n out[0] = 1.0 / a[0];\n out[1] = 1.0 / a[1];\n out[2] = 1.0 / a[2];\n out[3] = 1.0 / a[3];\n return out;\n}\n/**\n * Normalize a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to normalize\n * @returns {vec4} out\n */\n\nexport function normalize(out, a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var w = a[3];\n var len = x * x + y * y + z * z + w * w;\n\n if (len > 0) {\n len = 1 / Math.sqrt(len);\n }\n\n out[0] = x * len;\n out[1] = y * len;\n out[2] = z * len;\n out[3] = w * len;\n return out;\n}\n/**\n * Calculates the dot product of two vec4's\n *\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {Number} dot product of a and b\n */\n\nexport function dot(a, b) {\n return a[0] * b[0] + a[1] * b[1] + a[2] * b[2] + a[3] * b[3];\n}\n/**\n * Returns the cross-product of three vectors in a 4-dimensional space\n *\n * @param {ReadonlyVec4} result the receiving vector\n * @param {ReadonlyVec4} U the first vector\n * @param {ReadonlyVec4} V the second vector\n * @param {ReadonlyVec4} W the third vector\n * @returns {vec4} result\n */\n\nexport function cross(out, u, v, w) {\n var A = v[0] * w[1] - v[1] * w[0],\n B = v[0] * w[2] - v[2] * w[0],\n C = v[0] * w[3] - v[3] * w[0],\n D = v[1] * w[2] - v[2] * w[1],\n E = v[1] * w[3] - v[3] * w[1],\n F = v[2] * w[3] - v[3] * w[2];\n var G = u[0];\n var H = u[1];\n var I = u[2];\n var J = u[3];\n out[0] = H * F - I * E + J * D;\n out[1] = -(G * F) + I * C - J * B;\n out[2] = G * E - H * C + J * A;\n out[3] = -(G * D) + H * B - I * A;\n return out;\n}\n/**\n * Performs a linear interpolation between two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec4} out\n */\n\nexport function lerp(out, a, b, t) {\n var ax = a[0];\n var ay = a[1];\n var az = a[2];\n var aw = a[3];\n out[0] = ax + t * (b[0] - ax);\n out[1] = ay + t * (b[1] - ay);\n out[2] = az + t * (b[2] - az);\n out[3] = aw + t * (b[3] - aw);\n return out;\n}\n/**\n * Generates a random vector with the given scale\n *\n * @param {vec4} out the receiving vector\n * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned\n * @returns {vec4} out\n */\n\nexport function random(out, scale) {\n scale = scale || 1.0; // Marsaglia, George. Choosing a Point from the Surface of a\n // Sphere. Ann. Math. Statist. 43 (1972), no. 2, 645--646.\n // http://projecteuclid.org/euclid.aoms/1177692644;\n\n var v1, v2, v3, v4;\n var s1, s2;\n\n do {\n v1 = glMatrix.RANDOM() * 2 - 1;\n v2 = glMatrix.RANDOM() * 2 - 1;\n s1 = v1 * v1 + v2 * v2;\n } while (s1 >= 1);\n\n do {\n v3 = glMatrix.RANDOM() * 2 - 1;\n v4 = glMatrix.RANDOM() * 2 - 1;\n s2 = v3 * v3 + v4 * v4;\n } while (s2 >= 1);\n\n var d = Math.sqrt((1 - s1) / s2);\n out[0] = scale * v1;\n out[1] = scale * v2;\n out[2] = scale * v3 * d;\n out[3] = scale * v4 * d;\n return out;\n}\n/**\n * Transforms the vec4 with a mat4.\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the vector to transform\n * @param {ReadonlyMat4} m matrix to transform with\n * @returns {vec4} out\n */\n\nexport function transformMat4(out, a, m) {\n var x = a[0],\n y = a[1],\n z = a[2],\n w = a[3];\n out[0] = m[0] * x + m[4] * y + m[8] * z + m[12] * w;\n out[1] = m[1] * x + m[5] * y + m[9] * z + m[13] * w;\n out[2] = m[2] * x + m[6] * y + m[10] * z + m[14] * w;\n out[3] = m[3] * x + m[7] * y + m[11] * z + m[15] * w;\n return out;\n}\n/**\n * Transforms the vec4 with a quat\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the vector to transform\n * @param {ReadonlyQuat} q quaternion to transform with\n * @returns {vec4} out\n */\n\nexport function transformQuat(out, a, q) {\n var x = a[0],\n y = a[1],\n z = a[2];\n var qx = q[0],\n qy = q[1],\n qz = q[2],\n qw = q[3]; // calculate quat * vec\n\n var ix = qw * x + qy * z - qz * y;\n var iy = qw * y + qz * x - qx * z;\n var iz = qw * z + qx * y - qy * x;\n var iw = -qx * x - qy * y - qz * z; // calculate result * inverse quat\n\n out[0] = ix * qw + iw * -qx + iy * -qz - iz * -qy;\n out[1] = iy * qw + iw * -qy + iz * -qx - ix * -qz;\n out[2] = iz * qw + iw * -qz + ix * -qy - iy * -qx;\n out[3] = a[3];\n return out;\n}\n/**\n * Set the components of a vec4 to zero\n *\n * @param {vec4} out the receiving vector\n * @returns {vec4} out\n */\n\nexport function zero(out) {\n out[0] = 0.0;\n out[1] = 0.0;\n out[2] = 0.0;\n out[3] = 0.0;\n return out;\n}\n/**\n * Returns a string representation of a vector\n *\n * @param {ReadonlyVec4} a vector to represent as a string\n * @returns {String} string representation of the vector\n */\n\nexport function str(a) {\n return \"vec4(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \")\";\n}\n/**\n * Returns whether or not the vectors have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyVec4} a The first vector.\n * @param {ReadonlyVec4} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3];\n}\n/**\n * Returns whether or not the vectors have approximately the same elements in the same position.\n *\n * @param {ReadonlyVec4} a The first vector.\n * @param {ReadonlyVec4} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3));\n}\n/**\n * Alias for {@link vec4.subtract}\n * @function\n */\n\nexport var sub = subtract;\n/**\n * Alias for {@link vec4.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link vec4.divide}\n * @function\n */\n\nexport var div = divide;\n/**\n * Alias for {@link vec4.distance}\n * @function\n */\n\nexport var dist = distance;\n/**\n * Alias for {@link vec4.squaredDistance}\n * @function\n */\n\nexport var sqrDist = squaredDistance;\n/**\n * Alias for {@link vec4.length}\n * @function\n */\n\nexport var len = length;\n/**\n * Alias for {@link vec4.squaredLength}\n * @function\n */\n\nexport var sqrLen = squaredLength;\n/**\n * Perform some operation over an array of vec4s.\n *\n * @param {Array} a the array of vectors to iterate over\n * @param {Number} stride Number of elements between the start of each vec4. If 0 assumes tightly packed\n * @param {Number} offset Number of elements to skip at the beginning of the array\n * @param {Number} count Number of vec4s to iterate over. If 0 iterates over entire array\n * @param {Function} fn Function to call for each vector in the array\n * @param {Object} [arg] additional argument to pass to fn\n * @returns {Array} a\n * @function\n */\n\nexport var forEach = function () {\n var vec = create();\n return function (a, stride, offset, count, fn, arg) {\n var i, l;\n\n if (!stride) {\n stride = 4;\n }\n\n if (!offset) {\n offset = 0;\n }\n\n if (count) {\n l = Math.min(count * stride + offset, a.length);\n } else {\n l = a.length;\n }\n\n for (i = offset; i < l; i += stride) {\n vec[0] = a[i];\n vec[1] = a[i + 1];\n vec[2] = a[i + 2];\n vec[3] = a[i + 3];\n fn(vec, vec, arg);\n a[i] = vec[0];\n a[i + 1] = vec[1];\n a[i + 2] = vec[2];\n a[i + 3] = vec[3];\n }\n\n return a;\n };\n}();","import * as glMatrix from \"./common.js\";\n/**\n * 2 Dimensional Vector\n * @module vec2\n */\n\n/**\n * Creates a new, empty vec2\n *\n * @returns {vec2} a new 2D vector\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(2);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[0] = 0;\n out[1] = 0;\n }\n\n return out;\n}\n/**\n * Creates a new vec2 initialized with values from an existing vector\n *\n * @param {ReadonlyVec2} a vector to clone\n * @returns {vec2} a new 2D vector\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(2);\n out[0] = a[0];\n out[1] = a[1];\n return out;\n}\n/**\n * Creates a new vec2 initialized with the given values\n *\n * @param {Number} x X component\n * @param {Number} y Y component\n * @returns {vec2} a new 2D vector\n */\n\nexport function fromValues(x, y) {\n var out = new glMatrix.ARRAY_TYPE(2);\n out[0] = x;\n out[1] = y;\n return out;\n}\n/**\n * Copy the values from one vec2 to another\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the source vector\n * @returns {vec2} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n return out;\n}\n/**\n * Set the components of a vec2 to the given values\n *\n * @param {vec2} out the receiving vector\n * @param {Number} x X component\n * @param {Number} y Y component\n * @returns {vec2} out\n */\n\nexport function set(out, x, y) {\n out[0] = x;\n out[1] = y;\n return out;\n}\n/**\n * Adds two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n return out;\n}\n/**\n * Subtracts vector b from vector a\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n return out;\n}\n/**\n * Multiplies two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function multiply(out, a, b) {\n out[0] = a[0] * b[0];\n out[1] = a[1] * b[1];\n return out;\n}\n/**\n * Divides two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function divide(out, a, b) {\n out[0] = a[0] / b[0];\n out[1] = a[1] / b[1];\n return out;\n}\n/**\n * Math.ceil the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to ceil\n * @returns {vec2} out\n */\n\nexport function ceil(out, a) {\n out[0] = Math.ceil(a[0]);\n out[1] = Math.ceil(a[1]);\n return out;\n}\n/**\n * Math.floor the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to floor\n * @returns {vec2} out\n */\n\nexport function floor(out, a) {\n out[0] = Math.floor(a[0]);\n out[1] = Math.floor(a[1]);\n return out;\n}\n/**\n * Returns the minimum of two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function min(out, a, b) {\n out[0] = Math.min(a[0], b[0]);\n out[1] = Math.min(a[1], b[1]);\n return out;\n}\n/**\n * Returns the maximum of two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function max(out, a, b) {\n out[0] = Math.max(a[0], b[0]);\n out[1] = Math.max(a[1], b[1]);\n return out;\n}\n/**\n * Math.round the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to round\n * @returns {vec2} out\n */\n\nexport function round(out, a) {\n out[0] = Math.round(a[0]);\n out[1] = Math.round(a[1]);\n return out;\n}\n/**\n * Scales a vec2 by a scalar number\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to scale\n * @param {Number} b amount to scale the vector by\n * @returns {vec2} out\n */\n\nexport function scale(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n return out;\n}\n/**\n * Adds two vec2's after scaling the second operand by a scalar value\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @param {Number} scale the amount to scale b by before adding\n * @returns {vec2} out\n */\n\nexport function scaleAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n return out;\n}\n/**\n * Calculates the euclidian distance between two vec2's\n *\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {Number} distance between a and b\n */\n\nexport function distance(a, b) {\n var x = b[0] - a[0],\n y = b[1] - a[1];\n return Math.hypot(x, y);\n}\n/**\n * Calculates the squared euclidian distance between two vec2's\n *\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {Number} squared distance between a and b\n */\n\nexport function squaredDistance(a, b) {\n var x = b[0] - a[0],\n y = b[1] - a[1];\n return x * x + y * y;\n}\n/**\n * Calculates the length of a vec2\n *\n * @param {ReadonlyVec2} a vector to calculate length of\n * @returns {Number} length of a\n */\n\nexport function length(a) {\n var x = a[0],\n y = a[1];\n return Math.hypot(x, y);\n}\n/**\n * Calculates the squared length of a vec2\n *\n * @param {ReadonlyVec2} a vector to calculate squared length of\n * @returns {Number} squared length of a\n */\n\nexport function squaredLength(a) {\n var x = a[0],\n y = a[1];\n return x * x + y * y;\n}\n/**\n * Negates the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to negate\n * @returns {vec2} out\n */\n\nexport function negate(out, a) {\n out[0] = -a[0];\n out[1] = -a[1];\n return out;\n}\n/**\n * Returns the inverse of the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to invert\n * @returns {vec2} out\n */\n\nexport function inverse(out, a) {\n out[0] = 1.0 / a[0];\n out[1] = 1.0 / a[1];\n return out;\n}\n/**\n * Normalize a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to normalize\n * @returns {vec2} out\n */\n\nexport function normalize(out, a) {\n var x = a[0],\n y = a[1];\n var len = x * x + y * y;\n\n if (len > 0) {\n //TODO: evaluate use of glm_invsqrt here?\n len = 1 / Math.sqrt(len);\n }\n\n out[0] = a[0] * len;\n out[1] = a[1] * len;\n return out;\n}\n/**\n * Calculates the dot product of two vec2's\n *\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {Number} dot product of a and b\n */\n\nexport function dot(a, b) {\n return a[0] * b[0] + a[1] * b[1];\n}\n/**\n * Computes the cross product of two vec2's\n * Note that the cross product must by definition produce a 3D vector\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec3} out\n */\n\nexport function cross(out, a, b) {\n var z = a[0] * b[1] - a[1] * b[0];\n out[0] = out[1] = 0;\n out[2] = z;\n return out;\n}\n/**\n * Performs a linear interpolation between two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec2} out\n */\n\nexport function lerp(out, a, b, t) {\n var ax = a[0],\n ay = a[1];\n out[0] = ax + t * (b[0] - ax);\n out[1] = ay + t * (b[1] - ay);\n return out;\n}\n/**\n * Generates a random vector with the given scale\n *\n * @param {vec2} out the receiving vector\n * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned\n * @returns {vec2} out\n */\n\nexport function random(out, scale) {\n scale = scale || 1.0;\n var r = glMatrix.RANDOM() * 2.0 * Math.PI;\n out[0] = Math.cos(r) * scale;\n out[1] = Math.sin(r) * scale;\n return out;\n}\n/**\n * Transforms the vec2 with a mat2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to transform\n * @param {ReadonlyMat2} m matrix to transform with\n * @returns {vec2} out\n */\n\nexport function transformMat2(out, a, m) {\n var x = a[0],\n y = a[1];\n out[0] = m[0] * x + m[2] * y;\n out[1] = m[1] * x + m[3] * y;\n return out;\n}\n/**\n * Transforms the vec2 with a mat2d\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to transform\n * @param {ReadonlyMat2d} m matrix to transform with\n * @returns {vec2} out\n */\n\nexport function transformMat2d(out, a, m) {\n var x = a[0],\n y = a[1];\n out[0] = m[0] * x + m[2] * y + m[4];\n out[1] = m[1] * x + m[3] * y + m[5];\n return out;\n}\n/**\n * Transforms the vec2 with a mat3\n * 3rd vector component is implicitly '1'\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to transform\n * @param {ReadonlyMat3} m matrix to transform with\n * @returns {vec2} out\n */\n\nexport function transformMat3(out, a, m) {\n var x = a[0],\n y = a[1];\n out[0] = m[0] * x + m[3] * y + m[6];\n out[1] = m[1] * x + m[4] * y + m[7];\n return out;\n}\n/**\n * Transforms the vec2 with a mat4\n * 3rd vector component is implicitly '0'\n * 4th vector component is implicitly '1'\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to transform\n * @param {ReadonlyMat4} m matrix to transform with\n * @returns {vec2} out\n */\n\nexport function transformMat4(out, a, m) {\n var x = a[0];\n var y = a[1];\n out[0] = m[0] * x + m[4] * y + m[12];\n out[1] = m[1] * x + m[5] * y + m[13];\n return out;\n}\n/**\n * Rotate a 2D vector\n * @param {vec2} out The receiving vec2\n * @param {ReadonlyVec2} a The vec2 point to rotate\n * @param {ReadonlyVec2} b The origin of the rotation\n * @param {Number} rad The angle of rotation in radians\n * @returns {vec2} out\n */\n\nexport function rotate(out, a, b, rad) {\n //Translate point to the origin\n var p0 = a[0] - b[0],\n p1 = a[1] - b[1],\n sinC = Math.sin(rad),\n cosC = Math.cos(rad); //perform rotation and translate to correct position\n\n out[0] = p0 * cosC - p1 * sinC + b[0];\n out[1] = p0 * sinC + p1 * cosC + b[1];\n return out;\n}\n/**\n * Get the angle between two 2D vectors\n * @param {ReadonlyVec2} a The first operand\n * @param {ReadonlyVec2} b The second operand\n * @returns {Number} The angle in radians\n */\n\nexport function angle(a, b) {\n var x1 = a[0],\n y1 = a[1],\n x2 = b[0],\n y2 = b[1],\n // mag is the product of the magnitudes of a and b\n mag = Math.sqrt(x1 * x1 + y1 * y1) * Math.sqrt(x2 * x2 + y2 * y2),\n // mag &&.. short circuits if mag == 0\n cosine = mag && (x1 * x2 + y1 * y2) / mag; // Math.min(Math.max(cosine, -1), 1) clamps the cosine between -1 and 1\n\n return Math.acos(Math.min(Math.max(cosine, -1), 1));\n}\n/**\n * Set the components of a vec2 to zero\n *\n * @param {vec2} out the receiving vector\n * @returns {vec2} out\n */\n\nexport function zero(out) {\n out[0] = 0.0;\n out[1] = 0.0;\n return out;\n}\n/**\n * Returns a string representation of a vector\n *\n * @param {ReadonlyVec2} a vector to represent as a string\n * @returns {String} string representation of the vector\n */\n\nexport function str(a) {\n return \"vec2(\" + a[0] + \", \" + a[1] + \")\";\n}\n/**\n * Returns whether or not the vectors exactly have the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyVec2} a The first vector.\n * @param {ReadonlyVec2} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1];\n}\n/**\n * Returns whether or not the vectors have approximately the same elements in the same position.\n *\n * @param {ReadonlyVec2} a The first vector.\n * @param {ReadonlyVec2} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1];\n var b0 = b[0],\n b1 = b[1];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1));\n}\n/**\n * Alias for {@link vec2.length}\n * @function\n */\n\nexport var len = length;\n/**\n * Alias for {@link vec2.subtract}\n * @function\n */\n\nexport var sub = subtract;\n/**\n * Alias for {@link vec2.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link vec2.divide}\n * @function\n */\n\nexport var div = divide;\n/**\n * Alias for {@link vec2.distance}\n * @function\n */\n\nexport var dist = distance;\n/**\n * Alias for {@link vec2.squaredDistance}\n * @function\n */\n\nexport var sqrDist = squaredDistance;\n/**\n * Alias for {@link vec2.squaredLength}\n * @function\n */\n\nexport var sqrLen = squaredLength;\n/**\n * Perform some operation over an array of vec2s.\n *\n * @param {Array} a the array of vectors to iterate over\n * @param {Number} stride Number of elements between the start of each vec2. If 0 assumes tightly packed\n * @param {Number} offset Number of elements to skip at the beginning of the array\n * @param {Number} count Number of vec2s to iterate over. If 0 iterates over entire array\n * @param {Function} fn Function to call for each vector in the array\n * @param {Object} [arg] additional argument to pass to fn\n * @returns {Array} a\n * @function\n */\n\nexport var forEach = function () {\n var vec = create();\n return function (a, stride, offset, count, fn, arg) {\n var i, l;\n\n if (!stride) {\n stride = 2;\n }\n\n if (!offset) {\n offset = 0;\n }\n\n if (count) {\n l = Math.min(count * stride + offset, a.length);\n } else {\n l = a.length;\n }\n\n for (i = offset; i < l; i += stride) {\n vec[0] = a[i];\n vec[1] = a[i + 1];\n fn(vec, vec, arg);\n a[i] = vec[0];\n a[i + 1] = vec[1];\n }\n\n return a;\n };\n}();","import {StyleLayer} from '../style_layer';\n\nimport {CircleBucket} from '../../data/bucket/circle_bucket';\nimport {polygonIntersectsBufferedPoint} from '../../util/intersection_tests';\nimport {getMaximumPaintValue, translateDistance, translate} from '../query_utils';\nimport properties, {CircleLayoutPropsPossiblyEvaluated, CirclePaintPropsPossiblyEvaluated} from './circle_style_layer_properties.g';\nimport {Transitionable, Transitioning, Layout, PossiblyEvaluated} from '../properties';\nimport {mat4, vec4} from 'gl-matrix';\nimport Point from '@mapbox/point-geometry';\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Transform} from '../../geo/transform';\nimport type {Bucket, BucketParameters} from '../../data/bucket';\nimport type {CircleLayoutProps, CirclePaintProps} from './circle_style_layer_properties.g';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\n/**\n * A style layer that defines a circle\n */\nexport class CircleStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n createBucket(parameters: BucketParameters) {\n return new CircleBucket(parameters);\n }\n\n queryRadius(bucket: Bucket): number {\n const circleBucket: CircleBucket = (bucket as any);\n return getMaximumPaintValue('circle-radius', this, circleBucket) +\n getMaximumPaintValue('circle-stroke-width', this, circleBucket) +\n translateDistance(this.paint.get('circle-translate'));\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number,\n pixelPosMatrix: mat4\n ): boolean {\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('circle-translate'),\n this.paint.get('circle-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n const radius = this.paint.get('circle-radius').evaluate(feature, featureState);\n const stroke = this.paint.get('circle-stroke-width').evaluate(feature, featureState);\n const size = radius + stroke;\n\n // For pitch-alignment: map, compare feature geometry to query geometry in the plane of the tile\n // // Otherwise, compare geometry in the plane of the viewport\n // // A circle with fixed scaling relative to the viewport gets larger in tile space as it moves into the distance\n // // A circle with fixed scaling relative to the map gets smaller in viewport space as it moves into the distance\n const alignWithMap = this.paint.get('circle-pitch-alignment') === 'map';\n const transformedPolygon = alignWithMap ? translatedPolygon : projectQueryGeometry(translatedPolygon, pixelPosMatrix);\n const transformedSize = alignWithMap ? size * pixelsToTileUnits : size;\n\n for (const ring of geometry) {\n for (const point of ring) {\n\n const transformedPoint = alignWithMap ? point : projectPoint(point, pixelPosMatrix);\n\n let adjustedSize = transformedSize;\n const projectedCenter = vec4.transformMat4([] as any, [point.x, point.y, 0, 1], pixelPosMatrix);\n if (this.paint.get('circle-pitch-scale') === 'viewport' && this.paint.get('circle-pitch-alignment') === 'map') {\n adjustedSize *= projectedCenter[3] / transform.cameraToCenterDistance;\n } else if (this.paint.get('circle-pitch-scale') === 'map' && this.paint.get('circle-pitch-alignment') === 'viewport') {\n adjustedSize *= transform.cameraToCenterDistance / projectedCenter[3];\n }\n\n if (polygonIntersectsBufferedPoint(transformedPolygon, transformedPoint, adjustedSize)) return true;\n }\n }\n\n return false;\n }\n}\n\nfunction projectPoint(p: Point, pixelPosMatrix: mat4) {\n const point = vec4.transformMat4([] as any, [p.x, p.y, 0, 1], pixelPosMatrix);\n return new Point(point[0] / point[3], point[1] / point[3]);\n}\n\nfunction projectQueryGeometry(queryGeometry: Array, pixelPosMatrix: mat4) {\n return queryGeometry.map((p) => {\n return projectPoint(p, pixelPosMatrix);\n });\n}\n","import {CircleBucket} from './circle_bucket';\nimport {register} from '../../util/web_worker_transfer';\n\nimport type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer';\n\nexport class HeatmapBucket extends CircleBucket {\n // Needed for flow to accept omit: ['layers'] below, due to\n // https://github.com/facebook/flow/issues/4262\n layers: Array;\n}\n\nregister('HeatmapBucket', HeatmapBucket, {omit: ['layers']});\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type HeatmapPaintProps = {\n \"heatmap-radius\": DataDrivenProperty,\n \"heatmap-weight\": DataDrivenProperty,\n \"heatmap-intensity\": DataConstantProperty,\n \"heatmap-color\": ColorRampProperty,\n \"heatmap-opacity\": DataConstantProperty,\n};\n\nexport type HeatmapPaintPropsPossiblyEvaluated = {\n \"heatmap-radius\": PossiblyEvaluatedPropertyValue,\n \"heatmap-weight\": PossiblyEvaluatedPropertyValue,\n \"heatmap-intensity\": number,\n \"heatmap-color\": ColorRampProperty,\n \"heatmap-opacity\": number,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"heatmap-radius\": new DataDrivenProperty(styleSpec[\"paint_heatmap\"][\"heatmap-radius\"] as any as StylePropertySpecification),\n \"heatmap-weight\": new DataDrivenProperty(styleSpec[\"paint_heatmap\"][\"heatmap-weight\"] as any as StylePropertySpecification),\n \"heatmap-intensity\": new DataConstantProperty(styleSpec[\"paint_heatmap\"][\"heatmap-intensity\"] as any as StylePropertySpecification),\n \"heatmap-color\": new ColorRampProperty(styleSpec[\"paint_heatmap\"][\"heatmap-color\"] as any as StylePropertySpecification),\n \"heatmap-opacity\": new DataConstantProperty(styleSpec[\"paint_heatmap\"][\"heatmap-opacity\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {register} from './web_worker_transfer';\n\nexport type Size = {\n width: number;\n height: number;\n};\n\ntype Point2D = {\n x: number;\n y: number;\n};\n\nfunction createImage(image: any, {\n width,\n height\n}: Size, channels: number, data?: Uint8Array | Uint8ClampedArray) {\n if (!data) {\n data = new Uint8Array(width * height * channels);\n } else if (data instanceof Uint8ClampedArray) {\n data = new Uint8Array(data.buffer);\n } else if (data.length !== width * height * channels) {\n throw new RangeError(`mismatched image size. expected: ${data.length} but got: ${width * height * channels}`);\n }\n image.width = width;\n image.height = height;\n image.data = data;\n return image;\n}\n\nfunction resizeImage(image: any, {\n width,\n height\n}: Size, channels: number) {\n if (width === image.width && height === image.height) {\n return;\n }\n\n const newImage = createImage({}, {width, height}, channels);\n\n copyImage(image, newImage, {x: 0, y: 0}, {x: 0, y: 0}, {\n width: Math.min(image.width, width),\n height: Math.min(image.height, height)\n }, channels);\n\n image.width = width;\n image.height = height;\n image.data = newImage.data;\n}\n\nfunction copyImage(srcImg: any, dstImg: any, srcPt: Point2D, dstPt: Point2D, size: Size, channels: number) {\n if (size.width === 0 || size.height === 0) {\n return dstImg;\n }\n\n if (size.width > srcImg.width ||\n size.height > srcImg.height ||\n srcPt.x > srcImg.width - size.width ||\n srcPt.y > srcImg.height - size.height) {\n throw new RangeError('out of range source coordinates for image copy');\n }\n\n if (size.width > dstImg.width ||\n size.height > dstImg.height ||\n dstPt.x > dstImg.width - size.width ||\n dstPt.y > dstImg.height - size.height) {\n throw new RangeError('out of range destination coordinates for image copy');\n }\n\n const srcData = srcImg.data;\n const dstData = dstImg.data;\n\n if (srcData === dstData) throw new Error('srcData equals dstData, so image is already copied');\n\n for (let y = 0; y < size.height; y++) {\n const srcOffset = ((srcPt.y + y) * srcImg.width + srcPt.x) * channels;\n const dstOffset = ((dstPt.y + y) * dstImg.width + dstPt.x) * channels;\n for (let i = 0; i < size.width * channels; i++) {\n dstData[dstOffset + i] = srcData[srcOffset + i];\n }\n }\n return dstImg;\n}\n\n/**\n * @internal\n * An image with alpha color value\n */\nexport class AlphaImage {\n width: number;\n height: number;\n data: Uint8Array;\n\n constructor(size: Size, data?: Uint8Array | Uint8ClampedArray) {\n createImage(this, size, 1, data);\n }\n\n resize(size: Size) {\n resizeImage(this, size, 1);\n }\n\n clone() {\n return new AlphaImage({width: this.width, height: this.height}, new Uint8Array(this.data));\n }\n\n static copy(srcImg: AlphaImage, dstImg: AlphaImage, srcPt: Point2D, dstPt: Point2D, size: Size) {\n copyImage(srcImg, dstImg, srcPt, dstPt, size, 1);\n }\n}\n\n/**\n * An object to store image data not premultiplied, because ImageData is not premultiplied.\n * UNPACK_PREMULTIPLY_ALPHA_WEBGL must be used when uploading to a texture.\n */\nexport class RGBAImage {\n width: number;\n height: number;\n\n /**\n * data must be a Uint8Array instead of Uint8ClampedArray because texImage2D does not support Uint8ClampedArray in all browsers.\n */\n data: Uint8Array;\n\n constructor(size: Size, data?: Uint8Array | Uint8ClampedArray) {\n createImage(this, size, 4, data);\n }\n\n resize(size: Size) {\n resizeImage(this, size, 4);\n }\n\n replace(data: Uint8Array | Uint8ClampedArray, copy?: boolean) {\n if (copy) {\n this.data.set(data);\n } else if (data instanceof Uint8ClampedArray) {\n this.data = new Uint8Array(data.buffer);\n } else {\n this.data = data;\n }\n }\n\n clone() {\n return new RGBAImage({width: this.width, height: this.height}, new Uint8Array(this.data));\n }\n\n static copy(srcImg: RGBAImage | ImageData, dstImg: RGBAImage, srcPt: Point2D, dstPt: Point2D, size: Size) {\n copyImage(srcImg, dstImg, srcPt, dstPt, size, 4);\n }\n}\n\nregister('AlphaImage', AlphaImage);\nregister('RGBAImage', RGBAImage);\n","import {RGBAImage} from './image';\nimport {isPowerOfTwo} from './util';\n\nimport type {StylePropertyExpression} from '@maplibre/maplibre-gl-style-spec';\n\nexport type ColorRampParams = {\n expression: StylePropertyExpression;\n evaluationKey: string;\n resolution?: number;\n image?: RGBAImage;\n clips?: Array;\n};\n\n/**\n * Given an expression that should evaluate to a color ramp,\n * return a RGBA image representing that ramp expression.\n */\nexport function renderColorRamp(params: ColorRampParams): RGBAImage {\n const evaluationGlobals = {};\n const width = params.resolution || 256;\n const height = params.clips ? params.clips.length : 1;\n const image = params.image || new RGBAImage({width, height});\n\n if (!isPowerOfTwo(width)) throw new Error(`width is not a power of 2 - ${width}`);\n\n const renderPixel = (stride, index, progress) => {\n evaluationGlobals[params.evaluationKey] = progress;\n const pxColor = params.expression.evaluate(evaluationGlobals as any);\n // the colors are being unpremultiplied because Color uses\n // premultiplied values, and the Texture class expects unpremultiplied ones\n image.data[stride + index + 0] = Math.floor(pxColor.r * 255 / pxColor.a);\n image.data[stride + index + 1] = Math.floor(pxColor.g * 255 / pxColor.a);\n image.data[stride + index + 2] = Math.floor(pxColor.b * 255 / pxColor.a);\n image.data[stride + index + 3] = Math.floor(pxColor.a * 255);\n };\n\n if (!params.clips) {\n for (let i = 0, j = 0; i < width; i++, j += 4) {\n const progress = i / (width - 1);\n\n renderPixel(0, j, progress);\n }\n } else {\n for (let clip = 0, stride = 0; clip < height; ++clip, stride += width * 4) {\n for (let i = 0, j = 0; i < width; i++, j += 4) {\n // Remap progress between clips\n const progress = i / (width - 1);\n const {start, end} = params.clips[clip];\n const evaluationProgress = start * (1 - progress) + end * progress;\n renderPixel(stride, j, evaluationProgress);\n }\n }\n }\n\n return image;\n}\n","import {StyleLayer} from '../style_layer';\n\nimport {HeatmapBucket} from '../../data/bucket/heatmap_bucket';\nimport {RGBAImage} from '../../util/image';\nimport properties, {HeatmapPaintPropsPossiblyEvaluated} from './heatmap_style_layer_properties.g';\nimport {renderColorRamp} from '../../util/color_ramp';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {Texture} from '../../render/texture';\nimport type {Framebuffer} from '../../gl/framebuffer';\nimport type {HeatmapPaintProps} from './heatmap_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A style layer that defines a heatmap\n */\nexport class HeatmapStyleLayer extends StyleLayer {\n\n heatmapFbo: Framebuffer;\n colorRamp: RGBAImage;\n colorRampTexture: Texture;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n createBucket(options: any) {\n return new HeatmapBucket(options);\n }\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n\n // make sure color ramp texture is generated for default heatmap color too\n this._updateColorRamp();\n }\n\n _handleSpecialPaintPropertyUpdate(name: string) {\n if (name === 'heatmap-color') {\n this._updateColorRamp();\n }\n }\n\n _updateColorRamp() {\n const expression = this._transitionablePaint._values['heatmap-color'].value.expression;\n this.colorRamp = renderColorRamp({\n expression,\n evaluationKey: 'heatmapDensity',\n image: this.colorRamp\n });\n this.colorRampTexture = null;\n }\n\n resize() {\n if (this.heatmapFbo) {\n this.heatmapFbo.destroy();\n this.heatmapFbo = null;\n }\n }\n\n queryRadius(): number {\n return 0;\n }\n\n queryIntersectsFeature(): boolean {\n return false;\n }\n\n hasOffscreenPass() {\n return this.paint.get('heatmap-opacity') !== 0 && this.visibility !== 'none';\n }\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type HillshadePaintProps = {\n \"hillshade-illumination-direction\": DataConstantProperty,\n \"hillshade-illumination-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"hillshade-exaggeration\": DataConstantProperty,\n \"hillshade-shadow-color\": DataConstantProperty,\n \"hillshade-highlight-color\": DataConstantProperty,\n \"hillshade-accent-color\": DataConstantProperty,\n};\n\nexport type HillshadePaintPropsPossiblyEvaluated = {\n \"hillshade-illumination-direction\": number,\n \"hillshade-illumination-anchor\": \"map\" | \"viewport\",\n \"hillshade-exaggeration\": number,\n \"hillshade-shadow-color\": Color,\n \"hillshade-highlight-color\": Color,\n \"hillshade-accent-color\": Color,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"hillshade-illumination-direction\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-illumination-direction\"] as any as StylePropertySpecification),\n \"hillshade-illumination-anchor\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-illumination-anchor\"] as any as StylePropertySpecification),\n \"hillshade-exaggeration\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-exaggeration\"] as any as StylePropertySpecification),\n \"hillshade-shadow-color\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-shadow-color\"] as any as StylePropertySpecification),\n \"hillshade-highlight-color\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-highlight-color\"] as any as StylePropertySpecification),\n \"hillshade-accent-color\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-accent-color\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport properties, {HillshadePaintPropsPossiblyEvaluated} from './hillshade_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {HillshadePaintProps} from './hillshade_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport class HillshadeStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n hasOffscreenPass() {\n return this.paint.get('hillshade-exaggeration') !== 0 && this.visibility !== 'none';\n }\n}\n","import {createLayout} from '../../util/struct_array';\n\nconst layout = createLayout([\n {name: 'a_pos', components: 2, type: 'Int16'}\n], 4);\n\nexport default layout;\nexport const {members, size, alignment} = layout;\n","'use strict';\n\nmodule.exports = earcut;\nmodule.exports.default = earcut;\n\nfunction earcut(data, holeIndices, dim) {\n\n dim = dim || 2;\n\n var hasHoles = holeIndices && holeIndices.length,\n outerLen = hasHoles ? holeIndices[0] * dim : data.length,\n outerNode = linkedList(data, 0, outerLen, dim, true),\n triangles = [];\n\n if (!outerNode || outerNode.next === outerNode.prev) return triangles;\n\n var minX, minY, maxX, maxY, x, y, invSize;\n\n if (hasHoles) outerNode = eliminateHoles(data, holeIndices, outerNode, dim);\n\n // if the shape is not too simple, we'll use z-order curve hash later; calculate polygon bbox\n if (data.length > 80 * dim) {\n minX = maxX = data[0];\n minY = maxY = data[1];\n\n for (var i = dim; i < outerLen; i += dim) {\n x = data[i];\n y = data[i + 1];\n if (x < minX) minX = x;\n if (y < minY) minY = y;\n if (x > maxX) maxX = x;\n if (y > maxY) maxY = y;\n }\n\n // minX, minY and invSize are later used to transform coords into integers for z-order calculation\n invSize = Math.max(maxX - minX, maxY - minY);\n invSize = invSize !== 0 ? 32767 / invSize : 0;\n }\n\n earcutLinked(outerNode, triangles, dim, minX, minY, invSize, 0);\n\n return triangles;\n}\n\n// create a circular doubly linked list from polygon points in the specified winding order\nfunction linkedList(data, start, end, dim, clockwise) {\n var i, last;\n\n if (clockwise === (signedArea(data, start, end, dim) > 0)) {\n for (i = start; i < end; i += dim) last = insertNode(i, data[i], data[i + 1], last);\n } else {\n for (i = end - dim; i >= start; i -= dim) last = insertNode(i, data[i], data[i + 1], last);\n }\n\n if (last && equals(last, last.next)) {\n removeNode(last);\n last = last.next;\n }\n\n return last;\n}\n\n// eliminate colinear or duplicate points\nfunction filterPoints(start, end) {\n if (!start) return start;\n if (!end) end = start;\n\n var p = start,\n again;\n do {\n again = false;\n\n if (!p.steiner && (equals(p, p.next) || area(p.prev, p, p.next) === 0)) {\n removeNode(p);\n p = end = p.prev;\n if (p === p.next) break;\n again = true;\n\n } else {\n p = p.next;\n }\n } while (again || p !== end);\n\n return end;\n}\n\n// main ear slicing loop which triangulates a polygon (given as a linked list)\nfunction earcutLinked(ear, triangles, dim, minX, minY, invSize, pass) {\n if (!ear) return;\n\n // interlink polygon nodes in z-order\n if (!pass && invSize) indexCurve(ear, minX, minY, invSize);\n\n var stop = ear,\n prev, next;\n\n // iterate through ears, slicing them one by one\n while (ear.prev !== ear.next) {\n prev = ear.prev;\n next = ear.next;\n\n if (invSize ? isEarHashed(ear, minX, minY, invSize) : isEar(ear)) {\n // cut off the triangle\n triangles.push(prev.i / dim | 0);\n triangles.push(ear.i / dim | 0);\n triangles.push(next.i / dim | 0);\n\n removeNode(ear);\n\n // skipping the next vertex leads to less sliver triangles\n ear = next.next;\n stop = next.next;\n\n continue;\n }\n\n ear = next;\n\n // if we looped through the whole remaining polygon and can't find any more ears\n if (ear === stop) {\n // try filtering points and slicing again\n if (!pass) {\n earcutLinked(filterPoints(ear), triangles, dim, minX, minY, invSize, 1);\n\n // if this didn't work, try curing all small self-intersections locally\n } else if (pass === 1) {\n ear = cureLocalIntersections(filterPoints(ear), triangles, dim);\n earcutLinked(ear, triangles, dim, minX, minY, invSize, 2);\n\n // as a last resort, try splitting the remaining polygon into two\n } else if (pass === 2) {\n splitEarcut(ear, triangles, dim, minX, minY, invSize);\n }\n\n break;\n }\n }\n}\n\n// check whether a polygon node forms a valid ear with adjacent nodes\nfunction isEar(ear) {\n var a = ear.prev,\n b = ear,\n c = ear.next;\n\n if (area(a, b, c) >= 0) return false; // reflex, can't be an ear\n\n // now make sure we don't have other points inside the potential ear\n var ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y;\n\n // triangle bbox; min & max are calculated like this for speed\n var x0 = ax < bx ? (ax < cx ? ax : cx) : (bx < cx ? bx : cx),\n y0 = ay < by ? (ay < cy ? ay : cy) : (by < cy ? by : cy),\n x1 = ax > bx ? (ax > cx ? ax : cx) : (bx > cx ? bx : cx),\n y1 = ay > by ? (ay > cy ? ay : cy) : (by > cy ? by : cy);\n\n var p = c.next;\n while (p !== a) {\n if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) &&\n area(p.prev, p, p.next) >= 0) return false;\n p = p.next;\n }\n\n return true;\n}\n\nfunction isEarHashed(ear, minX, minY, invSize) {\n var a = ear.prev,\n b = ear,\n c = ear.next;\n\n if (area(a, b, c) >= 0) return false; // reflex, can't be an ear\n\n var ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y;\n\n // triangle bbox; min & max are calculated like this for speed\n var x0 = ax < bx ? (ax < cx ? ax : cx) : (bx < cx ? bx : cx),\n y0 = ay < by ? (ay < cy ? ay : cy) : (by < cy ? by : cy),\n x1 = ax > bx ? (ax > cx ? ax : cx) : (bx > cx ? bx : cx),\n y1 = ay > by ? (ay > cy ? ay : cy) : (by > cy ? by : cy);\n\n // z-order range for the current triangle bbox;\n var minZ = zOrder(x0, y0, minX, minY, invSize),\n maxZ = zOrder(x1, y1, minX, minY, invSize);\n\n var p = ear.prevZ,\n n = ear.nextZ;\n\n // look for points inside the triangle in both directions\n while (p && p.z >= minZ && n && n.z <= maxZ) {\n if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 && p !== a && p !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) && area(p.prev, p, p.next) >= 0) return false;\n p = p.prevZ;\n\n if (n.x >= x0 && n.x <= x1 && n.y >= y0 && n.y <= y1 && n !== a && n !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) && area(n.prev, n, n.next) >= 0) return false;\n n = n.nextZ;\n }\n\n // look for remaining points in decreasing z-order\n while (p && p.z >= minZ) {\n if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 && p !== a && p !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) && area(p.prev, p, p.next) >= 0) return false;\n p = p.prevZ;\n }\n\n // look for remaining points in increasing z-order\n while (n && n.z <= maxZ) {\n if (n.x >= x0 && n.x <= x1 && n.y >= y0 && n.y <= y1 && n !== a && n !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) && area(n.prev, n, n.next) >= 0) return false;\n n = n.nextZ;\n }\n\n return true;\n}\n\n// go through all polygon nodes and cure small local self-intersections\nfunction cureLocalIntersections(start, triangles, dim) {\n var p = start;\n do {\n var a = p.prev,\n b = p.next.next;\n\n if (!equals(a, b) && intersects(a, p, p.next, b) && locallyInside(a, b) && locallyInside(b, a)) {\n\n triangles.push(a.i / dim | 0);\n triangles.push(p.i / dim | 0);\n triangles.push(b.i / dim | 0);\n\n // remove two nodes involved\n removeNode(p);\n removeNode(p.next);\n\n p = start = b;\n }\n p = p.next;\n } while (p !== start);\n\n return filterPoints(p);\n}\n\n// try splitting polygon into two and triangulate them independently\nfunction splitEarcut(start, triangles, dim, minX, minY, invSize) {\n // look for a valid diagonal that divides the polygon into two\n var a = start;\n do {\n var b = a.next.next;\n while (b !== a.prev) {\n if (a.i !== b.i && isValidDiagonal(a, b)) {\n // split the polygon in two by the diagonal\n var c = splitPolygon(a, b);\n\n // filter colinear points around the cuts\n a = filterPoints(a, a.next);\n c = filterPoints(c, c.next);\n\n // run earcut on each half\n earcutLinked(a, triangles, dim, minX, minY, invSize, 0);\n earcutLinked(c, triangles, dim, minX, minY, invSize, 0);\n return;\n }\n b = b.next;\n }\n a = a.next;\n } while (a !== start);\n}\n\n// link every hole into the outer loop, producing a single-ring polygon without holes\nfunction eliminateHoles(data, holeIndices, outerNode, dim) {\n var queue = [],\n i, len, start, end, list;\n\n for (i = 0, len = holeIndices.length; i < len; i++) {\n start = holeIndices[i] * dim;\n end = i < len - 1 ? holeIndices[i + 1] * dim : data.length;\n list = linkedList(data, start, end, dim, false);\n if (list === list.next) list.steiner = true;\n queue.push(getLeftmost(list));\n }\n\n queue.sort(compareX);\n\n // process holes from left to right\n for (i = 0; i < queue.length; i++) {\n outerNode = eliminateHole(queue[i], outerNode);\n }\n\n return outerNode;\n}\n\nfunction compareX(a, b) {\n return a.x - b.x;\n}\n\n// find a bridge between vertices that connects hole with an outer ring and and link it\nfunction eliminateHole(hole, outerNode) {\n var bridge = findHoleBridge(hole, outerNode);\n if (!bridge) {\n return outerNode;\n }\n\n var bridgeReverse = splitPolygon(bridge, hole);\n\n // filter collinear points around the cuts\n filterPoints(bridgeReverse, bridgeReverse.next);\n return filterPoints(bridge, bridge.next);\n}\n\n// David Eberly's algorithm for finding a bridge between hole and outer polygon\nfunction findHoleBridge(hole, outerNode) {\n var p = outerNode,\n hx = hole.x,\n hy = hole.y,\n qx = -Infinity,\n m;\n\n // find a segment intersected by a ray from the hole's leftmost point to the left;\n // segment's endpoint with lesser x will be potential connection point\n do {\n if (hy <= p.y && hy >= p.next.y && p.next.y !== p.y) {\n var x = p.x + (hy - p.y) * (p.next.x - p.x) / (p.next.y - p.y);\n if (x <= hx && x > qx) {\n qx = x;\n m = p.x < p.next.x ? p : p.next;\n if (x === hx) return m; // hole touches outer segment; pick leftmost endpoint\n }\n }\n p = p.next;\n } while (p !== outerNode);\n\n if (!m) return null;\n\n // look for points inside the triangle of hole point, segment intersection and endpoint;\n // if there are no points found, we have a valid connection;\n // otherwise choose the point of the minimum angle with the ray as connection point\n\n var stop = m,\n mx = m.x,\n my = m.y,\n tanMin = Infinity,\n tan;\n\n p = m;\n\n do {\n if (hx >= p.x && p.x >= mx && hx !== p.x &&\n pointInTriangle(hy < my ? hx : qx, hy, mx, my, hy < my ? qx : hx, hy, p.x, p.y)) {\n\n tan = Math.abs(hy - p.y) / (hx - p.x); // tangential\n\n if (locallyInside(p, hole) &&\n (tan < tanMin || (tan === tanMin && (p.x > m.x || (p.x === m.x && sectorContainsSector(m, p)))))) {\n m = p;\n tanMin = tan;\n }\n }\n\n p = p.next;\n } while (p !== stop);\n\n return m;\n}\n\n// whether sector in vertex m contains sector in vertex p in the same coordinates\nfunction sectorContainsSector(m, p) {\n return area(m.prev, m, p.prev) < 0 && area(p.next, m, m.next) < 0;\n}\n\n// interlink polygon nodes in z-order\nfunction indexCurve(start, minX, minY, invSize) {\n var p = start;\n do {\n if (p.z === 0) p.z = zOrder(p.x, p.y, minX, minY, invSize);\n p.prevZ = p.prev;\n p.nextZ = p.next;\n p = p.next;\n } while (p !== start);\n\n p.prevZ.nextZ = null;\n p.prevZ = null;\n\n sortLinked(p);\n}\n\n// Simon Tatham's linked list merge sort algorithm\n// http://www.chiark.greenend.org.uk/~sgtatham/algorithms/listsort.html\nfunction sortLinked(list) {\n var i, p, q, e, tail, numMerges, pSize, qSize,\n inSize = 1;\n\n do {\n p = list;\n list = null;\n tail = null;\n numMerges = 0;\n\n while (p) {\n numMerges++;\n q = p;\n pSize = 0;\n for (i = 0; i < inSize; i++) {\n pSize++;\n q = q.nextZ;\n if (!q) break;\n }\n qSize = inSize;\n\n while (pSize > 0 || (qSize > 0 && q)) {\n\n if (pSize !== 0 && (qSize === 0 || !q || p.z <= q.z)) {\n e = p;\n p = p.nextZ;\n pSize--;\n } else {\n e = q;\n q = q.nextZ;\n qSize--;\n }\n\n if (tail) tail.nextZ = e;\n else list = e;\n\n e.prevZ = tail;\n tail = e;\n }\n\n p = q;\n }\n\n tail.nextZ = null;\n inSize *= 2;\n\n } while (numMerges > 1);\n\n return list;\n}\n\n// z-order of a point given coords and inverse of the longer side of data bbox\nfunction zOrder(x, y, minX, minY, invSize) {\n // coords are transformed into non-negative 15-bit integer range\n x = (x - minX) * invSize | 0;\n y = (y - minY) * invSize | 0;\n\n x = (x | (x << 8)) & 0x00FF00FF;\n x = (x | (x << 4)) & 0x0F0F0F0F;\n x = (x | (x << 2)) & 0x33333333;\n x = (x | (x << 1)) & 0x55555555;\n\n y = (y | (y << 8)) & 0x00FF00FF;\n y = (y | (y << 4)) & 0x0F0F0F0F;\n y = (y | (y << 2)) & 0x33333333;\n y = (y | (y << 1)) & 0x55555555;\n\n return x | (y << 1);\n}\n\n// find the leftmost node of a polygon ring\nfunction getLeftmost(start) {\n var p = start,\n leftmost = start;\n do {\n if (p.x < leftmost.x || (p.x === leftmost.x && p.y < leftmost.y)) leftmost = p;\n p = p.next;\n } while (p !== start);\n\n return leftmost;\n}\n\n// check if a point lies within a convex triangle\nfunction pointInTriangle(ax, ay, bx, by, cx, cy, px, py) {\n return (cx - px) * (ay - py) >= (ax - px) * (cy - py) &&\n (ax - px) * (by - py) >= (bx - px) * (ay - py) &&\n (bx - px) * (cy - py) >= (cx - px) * (by - py);\n}\n\n// check if a diagonal between two polygon nodes is valid (lies in polygon interior)\nfunction isValidDiagonal(a, b) {\n return a.next.i !== b.i && a.prev.i !== b.i && !intersectsPolygon(a, b) && // dones't intersect other edges\n (locallyInside(a, b) && locallyInside(b, a) && middleInside(a, b) && // locally visible\n (area(a.prev, a, b.prev) || area(a, b.prev, b)) || // does not create opposite-facing sectors\n equals(a, b) && area(a.prev, a, a.next) > 0 && area(b.prev, b, b.next) > 0); // special zero-length case\n}\n\n// signed area of a triangle\nfunction area(p, q, r) {\n return (q.y - p.y) * (r.x - q.x) - (q.x - p.x) * (r.y - q.y);\n}\n\n// check if two points are equal\nfunction equals(p1, p2) {\n return p1.x === p2.x && p1.y === p2.y;\n}\n\n// check if two segments intersect\nfunction intersects(p1, q1, p2, q2) {\n var o1 = sign(area(p1, q1, p2));\n var o2 = sign(area(p1, q1, q2));\n var o3 = sign(area(p2, q2, p1));\n var o4 = sign(area(p2, q2, q1));\n\n if (o1 !== o2 && o3 !== o4) return true; // general case\n\n if (o1 === 0 && onSegment(p1, p2, q1)) return true; // p1, q1 and p2 are collinear and p2 lies on p1q1\n if (o2 === 0 && onSegment(p1, q2, q1)) return true; // p1, q1 and q2 are collinear and q2 lies on p1q1\n if (o3 === 0 && onSegment(p2, p1, q2)) return true; // p2, q2 and p1 are collinear and p1 lies on p2q2\n if (o4 === 0 && onSegment(p2, q1, q2)) return true; // p2, q2 and q1 are collinear and q1 lies on p2q2\n\n return false;\n}\n\n// for collinear points p, q, r, check if point q lies on segment pr\nfunction onSegment(p, q, r) {\n return q.x <= Math.max(p.x, r.x) && q.x >= Math.min(p.x, r.x) && q.y <= Math.max(p.y, r.y) && q.y >= Math.min(p.y, r.y);\n}\n\nfunction sign(num) {\n return num > 0 ? 1 : num < 0 ? -1 : 0;\n}\n\n// check if a polygon diagonal intersects any polygon segments\nfunction intersectsPolygon(a, b) {\n var p = a;\n do {\n if (p.i !== a.i && p.next.i !== a.i && p.i !== b.i && p.next.i !== b.i &&\n intersects(p, p.next, a, b)) return true;\n p = p.next;\n } while (p !== a);\n\n return false;\n}\n\n// check if a polygon diagonal is locally inside the polygon\nfunction locallyInside(a, b) {\n return area(a.prev, a, a.next) < 0 ?\n area(a, b, a.next) >= 0 && area(a, a.prev, b) >= 0 :\n area(a, b, a.prev) < 0 || area(a, a.next, b) < 0;\n}\n\n// check if the middle point of a polygon diagonal is inside the polygon\nfunction middleInside(a, b) {\n var p = a,\n inside = false,\n px = (a.x + b.x) / 2,\n py = (a.y + b.y) / 2;\n do {\n if (((p.y > py) !== (p.next.y > py)) && p.next.y !== p.y &&\n (px < (p.next.x - p.x) * (py - p.y) / (p.next.y - p.y) + p.x))\n inside = !inside;\n p = p.next;\n } while (p !== a);\n\n return inside;\n}\n\n// link two polygon vertices with a bridge; if the vertices belong to the same ring, it splits polygon into two;\n// if one belongs to the outer ring and another to a hole, it merges it into a single ring\nfunction splitPolygon(a, b) {\n var a2 = new Node(a.i, a.x, a.y),\n b2 = new Node(b.i, b.x, b.y),\n an = a.next,\n bp = b.prev;\n\n a.next = b;\n b.prev = a;\n\n a2.next = an;\n an.prev = a2;\n\n b2.next = a2;\n a2.prev = b2;\n\n bp.next = b2;\n b2.prev = bp;\n\n return b2;\n}\n\n// create a node and optionally link it with previous one (in a circular doubly linked list)\nfunction insertNode(i, x, y, last) {\n var p = new Node(i, x, y);\n\n if (!last) {\n p.prev = p;\n p.next = p;\n\n } else {\n p.next = last.next;\n p.prev = last;\n last.next.prev = p;\n last.next = p;\n }\n return p;\n}\n\nfunction removeNode(p) {\n p.next.prev = p.prev;\n p.prev.next = p.next;\n\n if (p.prevZ) p.prevZ.nextZ = p.nextZ;\n if (p.nextZ) p.nextZ.prevZ = p.prevZ;\n}\n\nfunction Node(i, x, y) {\n // vertex index in coordinates array\n this.i = i;\n\n // vertex coordinates\n this.x = x;\n this.y = y;\n\n // previous and next vertex nodes in a polygon ring\n this.prev = null;\n this.next = null;\n\n // z-order curve value\n this.z = 0;\n\n // previous and next nodes in z-order\n this.prevZ = null;\n this.nextZ = null;\n\n // indicates whether this is a steiner point\n this.steiner = false;\n}\n\n// return a percentage difference between the polygon area and its triangulation area;\n// used to verify correctness of triangulation\nearcut.deviation = function (data, holeIndices, dim, triangles) {\n var hasHoles = holeIndices && holeIndices.length;\n var outerLen = hasHoles ? holeIndices[0] * dim : data.length;\n\n var polygonArea = Math.abs(signedArea(data, 0, outerLen, dim));\n if (hasHoles) {\n for (var i = 0, len = holeIndices.length; i < len; i++) {\n var start = holeIndices[i] * dim;\n var end = i < len - 1 ? holeIndices[i + 1] * dim : data.length;\n polygonArea -= Math.abs(signedArea(data, start, end, dim));\n }\n }\n\n var trianglesArea = 0;\n for (i = 0; i < triangles.length; i += 3) {\n var a = triangles[i] * dim;\n var b = triangles[i + 1] * dim;\n var c = triangles[i + 2] * dim;\n trianglesArea += Math.abs(\n (data[a] - data[c]) * (data[b + 1] - data[a + 1]) -\n (data[a] - data[b]) * (data[c + 1] - data[a + 1]));\n }\n\n return polygonArea === 0 && trianglesArea === 0 ? 0 :\n Math.abs((trianglesArea - polygonArea) / polygonArea);\n};\n\nfunction signedArea(data, start, end, dim) {\n var sum = 0;\n for (var i = start, j = end - dim; i < end; i += dim) {\n sum += (data[j] - data[i]) * (data[i + 1] + data[j + 1]);\n j = i;\n }\n return sum;\n}\n\n// turn a polygon in a multi-dimensional array form (e.g. as in GeoJSON) into a form Earcut accepts\nearcut.flatten = function (data) {\n var dim = data[0][0].length,\n result = {vertices: [], holes: [], dimensions: dim},\n holeIndex = 0;\n\n for (var i = 0; i < data.length; i++) {\n for (var j = 0; j < data[i].length; j++) {\n for (var d = 0; d < dim; d++) result.vertices.push(data[i][j][d]);\n }\n if (i > 0) {\n holeIndex += data[i - 1].length;\n result.holes.push(holeIndex);\n }\n }\n return result;\n};\n","\nexport default function quickselect(arr, k, left, right, compare) {\n quickselectStep(arr, k, left || 0, right || (arr.length - 1), compare || defaultCompare);\n}\n\nfunction quickselectStep(arr, k, left, right, compare) {\n\n while (right > left) {\n if (right - left > 600) {\n var n = right - left + 1;\n var m = k - left + 1;\n var z = Math.log(n);\n var s = 0.5 * Math.exp(2 * z / 3);\n var sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);\n var newLeft = Math.max(left, Math.floor(k - m * s / n + sd));\n var newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));\n quickselectStep(arr, k, newLeft, newRight, compare);\n }\n\n var t = arr[k];\n var i = left;\n var j = right;\n\n swap(arr, left, k);\n if (compare(arr[right], t) > 0) swap(arr, left, right);\n\n while (i < j) {\n swap(arr, i, j);\n i++;\n j--;\n while (compare(arr[i], t) < 0) i++;\n while (compare(arr[j], t) > 0) j--;\n }\n\n if (compare(arr[left], t) === 0) swap(arr, left, j);\n else {\n j++;\n swap(arr, j, right);\n }\n\n if (j <= k) left = j + 1;\n if (k <= j) right = j - 1;\n }\n}\n\nfunction swap(arr, i, j) {\n var tmp = arr[i];\n arr[i] = arr[j];\n arr[j] = tmp;\n}\n\nfunction defaultCompare(a, b) {\n return a < b ? -1 : a > b ? 1 : 0;\n}\n","import quickselect from 'quickselect';\n\nimport {calculateSignedArea} from './util';\n\nimport type Point from '@mapbox/point-geometry';\n\n// classifies an array of rings into polygons with outer rings and holes\nexport function classifyRings(rings: Array>, maxRings: number) {\n const len = rings.length;\n\n if (len <= 1) return [rings];\n\n const polygons = [];\n let polygon,\n ccw;\n\n for (let i = 0; i < len; i++) {\n const area = calculateSignedArea(rings[i]);\n if (area === 0) continue;\n\n (rings[i] as any).area = Math.abs(area);\n\n if (ccw === undefined) ccw = area < 0;\n\n if (ccw === area < 0) {\n if (polygon) polygons.push(polygon);\n polygon = [rings[i]];\n\n } else {\n (polygon as any).push(rings[i]);\n }\n }\n if (polygon) polygons.push(polygon);\n\n // Earcut performance degrades with the # of rings in a polygon. For this\n // reason, we limit strip out all but the `maxRings` largest rings.\n if (maxRings > 1) {\n for (let j = 0; j < polygons.length; j++) {\n if (polygons[j].length <= maxRings) continue;\n quickselect(polygons[j], maxRings, 1, polygons[j].length - 1, compareAreas);\n polygons[j] = polygons[j].slice(0, maxRings);\n }\n }\n\n return polygons;\n}\n\nfunction compareAreas(a, b) {\n return b.area - a.area;\n}\n","import type {FillStyleLayer} from '../../style/style_layer/fill_style_layer';\nimport type {FillExtrusionStyleLayer} from '../../style/style_layer/fill_extrusion_style_layer';\nimport type {LineStyleLayer} from '../../style/style_layer/line_style_layer';\n\nimport type {\n BucketFeature,\n PopulateParameters\n} from '../bucket';\nimport {PossiblyEvaluated} from '../../style/properties';\n\ntype PatternStyleLayers = Array | Array | Array;\n\nexport function hasPattern(type: string, layers: PatternStyleLayers, options: PopulateParameters) {\n const patterns = options.patternDependencies;\n let hasPattern = false;\n\n for (const layer of layers) {\n const patternProperty = (layer.paint as PossiblyEvaluated).get(`${type}-pattern`);\n if (!patternProperty.isConstant()) {\n hasPattern = true;\n }\n\n const constantPattern = patternProperty.constantOr(null);\n if (constantPattern) {\n hasPattern = true;\n patterns[constantPattern.to] = true;\n patterns[constantPattern.from] = true;\n }\n }\n\n return hasPattern;\n}\n\nexport function addPatternDependencies(type: string, layers: PatternStyleLayers, patternFeature: BucketFeature, zoom: number, options: PopulateParameters) {\n const patterns = options.patternDependencies;\n for (const layer of layers) {\n const patternProperty = (layer.paint as PossiblyEvaluated).get(`${type}-pattern`);\n\n const patternPropertyValue = patternProperty.value;\n if (patternPropertyValue.kind !== 'constant') {\n let min = patternPropertyValue.evaluate({zoom: zoom - 1}, patternFeature, {}, options.availableImages);\n let mid = patternPropertyValue.evaluate({zoom}, patternFeature, {}, options.availableImages);\n let max = patternPropertyValue.evaluate({zoom: zoom + 1}, patternFeature, {}, options.availableImages);\n min = min && min.name ? min.name : min;\n mid = mid && mid.name ? mid.name : mid;\n max = max && max.name ? max.name : max;\n // add to patternDependencies\n patterns[min] = true;\n patterns[mid] = true;\n patterns[max] = true;\n\n // save for layout\n patternFeature.patterns[layer.id] = {min, mid, max};\n }\n }\n return patternFeature;\n}\n","import {FillLayoutArray} from '../array_types.g';\n\nimport {members as layoutAttributes} from './fill_attributes';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {LineIndexArray, TriangleIndexArray} from '../index_array_type';\nimport earcut from 'earcut';\nimport {classifyRings} from '../../util/classify_rings';\nconst EARCUT_MAX_RINGS = 500;\nimport {register} from '../../util/web_worker_transfer';\nimport {hasPattern, addPatternDependencies} from './pattern_bucket_features';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {FillStyleLayer} from '../../style/style_layer/fill_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nexport class FillBucket implements Bucket {\n index: number;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n patternFeatures: Array;\n\n layoutVertexArray: FillLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n indexArray2: LineIndexArray;\n indexBuffer2: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n segments2: SegmentVector;\n uploaded: boolean;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n this.patternFeatures = [];\n\n this.layoutVertexArray = new FillLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.indexArray2 = new LineIndexArray();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.segments = new SegmentVector();\n this.segments2 = new SegmentVector();\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n this.hasPattern = hasPattern('fill', this.layers, options);\n const fillSortKey = this.layers[0].layout.get('fill-sort-key');\n const sortFeaturesByKey = !fillSortKey.isConstant();\n const bucketFeatures: BucketFeature[] = [];\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const sortKey = sortFeaturesByKey ?\n fillSortKey.evaluate(evaluationFeature, {}, canonical, options.availableImages) :\n undefined;\n\n const bucketFeature: BucketFeature = {\n id,\n properties: feature.properties,\n type: feature.type,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n patterns: {},\n sortKey\n };\n\n bucketFeatures.push(bucketFeature);\n }\n\n if (sortFeaturesByKey) {\n bucketFeatures.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const bucketFeature of bucketFeatures) {\n const {geometry, index, sourceLayerIndex} = bucketFeature;\n\n if (this.hasPattern) {\n const patternFeature = addPatternDependencies('fill', this.layers, bucketFeature, this.zoom, options);\n // pattern features are added only once the pattern is loaded into the image atlas\n // so are stored during populate until later updated with positions by tile worker in addFeatures\n this.patternFeatures.push(patternFeature);\n } else {\n this.addFeature(bucketFeature, geometry, index, canonical, {});\n }\n\n const feature = features[index].feature;\n options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {\n [_: string]: ImagePosition;\n }) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {\n [_: string]: ImagePosition;\n }) {\n for (const feature of this.patternFeatures) {\n this.addFeature(feature, feature.geometry, feature.index, canonical, imagePositions);\n }\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0;\n }\n\n uploadPending(): boolean {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n upload(context: Context) {\n if (!this.uploaded) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n this.indexBuffer2 = context.createIndexBuffer(this.indexArray2);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.indexBuffer2.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n this.segments2.destroy();\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {\n [_: string]: ImagePosition;\n }) {\n for (const polygon of classifyRings(geometry, EARCUT_MAX_RINGS)) {\n let numVertices = 0;\n for (const ring of polygon) {\n numVertices += ring.length;\n }\n\n const triangleSegment = this.segments.prepareSegment(numVertices, this.layoutVertexArray, this.indexArray);\n const triangleIndex = triangleSegment.vertexLength;\n\n const flattened = [];\n const holeIndices = [];\n\n for (const ring of polygon) {\n if (ring.length === 0) {\n continue;\n }\n\n if (ring !== polygon[0]) {\n holeIndices.push(flattened.length / 2);\n }\n\n const lineSegment = this.segments2.prepareSegment(ring.length, this.layoutVertexArray, this.indexArray2);\n const lineIndex = lineSegment.vertexLength;\n\n this.layoutVertexArray.emplaceBack(ring[0].x, ring[0].y);\n this.indexArray2.emplaceBack(lineIndex + ring.length - 1, lineIndex);\n flattened.push(ring[0].x);\n flattened.push(ring[0].y);\n\n for (let i = 1; i < ring.length; i++) {\n this.layoutVertexArray.emplaceBack(ring[i].x, ring[i].y);\n this.indexArray2.emplaceBack(lineIndex + i - 1, lineIndex + i);\n flattened.push(ring[i].x);\n flattened.push(ring[i].y);\n }\n\n lineSegment.vertexLength += ring.length;\n lineSegment.primitiveLength += ring.length;\n }\n\n const indices = earcut(flattened, holeIndices);\n\n for (let i = 0; i < indices.length; i += 3) {\n this.indexArray.emplaceBack(\n triangleIndex + indices[i],\n triangleIndex + indices[i + 1],\n triangleIndex + indices[i + 2]);\n }\n\n triangleSegment.vertexLength += numVertices;\n triangleSegment.primitiveLength += indices.length / 3;\n }\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical);\n }\n}\n\nregister('FillBucket', FillBucket, {omit: ['layers', 'patternFeatures']});\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type FillLayoutProps = {\n \"fill-sort-key\": DataDrivenProperty,\n};\n\nexport type FillLayoutPropsPossiblyEvaluated = {\n \"fill-sort-key\": PossiblyEvaluatedPropertyValue,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"fill-sort-key\": new DataDrivenProperty(styleSpec[\"layout_fill\"][\"fill-sort-key\"] as any as StylePropertySpecification),\n});\n\nexport type FillPaintProps = {\n \"fill-antialias\": DataConstantProperty,\n \"fill-opacity\": DataDrivenProperty,\n \"fill-color\": DataDrivenProperty,\n \"fill-outline-color\": DataDrivenProperty,\n \"fill-translate\": DataConstantProperty<[number, number]>,\n \"fill-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"fill-pattern\": CrossFadedDataDrivenProperty,\n};\n\nexport type FillPaintPropsPossiblyEvaluated = {\n \"fill-antialias\": boolean,\n \"fill-opacity\": PossiblyEvaluatedPropertyValue,\n \"fill-color\": PossiblyEvaluatedPropertyValue,\n \"fill-outline-color\": PossiblyEvaluatedPropertyValue,\n \"fill-translate\": [number, number],\n \"fill-translate-anchor\": \"map\" | \"viewport\",\n \"fill-pattern\": PossiblyEvaluatedPropertyValue>,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"fill-antialias\": new DataConstantProperty(styleSpec[\"paint_fill\"][\"fill-antialias\"] as any as StylePropertySpecification),\n \"fill-opacity\": new DataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-opacity\"] as any as StylePropertySpecification),\n \"fill-color\": new DataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-color\"] as any as StylePropertySpecification),\n \"fill-outline-color\": new DataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-outline-color\"] as any as StylePropertySpecification),\n \"fill-translate\": new DataConstantProperty(styleSpec[\"paint_fill\"][\"fill-translate\"] as any as StylePropertySpecification),\n \"fill-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_fill\"][\"fill-translate-anchor\"] as any as StylePropertySpecification),\n \"fill-pattern\": new CrossFadedDataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-pattern\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import {StyleLayer} from '../style_layer';\n\nimport {FillBucket} from '../../data/bucket/fill_bucket';\nimport {polygonIntersectsMultiPolygon} from '../../util/intersection_tests';\nimport {translateDistance, translate} from '../query_utils';\nimport properties, {FillLayoutPropsPossiblyEvaluated, FillPaintPropsPossiblyEvaluated} from './fill_style_layer_properties.g';\nimport {Transitionable, Transitioning, Layout, PossiblyEvaluated} from '../properties';\n\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {BucketParameters} from '../../data/bucket';\nimport type Point from '@mapbox/point-geometry';\nimport type {FillLayoutProps, FillPaintProps} from './fill_style_layer_properties.g';\nimport type {EvaluationParameters} from '../evaluation_parameters';\nimport type {Transform} from '../../geo/transform';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nexport class FillStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n super.recalculate(parameters, availableImages);\n\n const outlineColor = this.paint._values['fill-outline-color'];\n if (outlineColor.value.kind === 'constant' && outlineColor.value.value === undefined) {\n this.paint._values['fill-outline-color'] = this.paint._values['fill-color'];\n }\n }\n\n createBucket(parameters: BucketParameters) {\n return new FillBucket(parameters);\n }\n\n queryRadius(): number {\n return translateDistance(this.paint.get('fill-translate'));\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number\n ): boolean {\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('fill-translate'),\n this.paint.get('fill-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n return polygonIntersectsMultiPolygon(translatedPolygon, geometry);\n }\n\n isTileClipped() {\n return true;\n }\n}\n","import {createLayout} from '../../util/struct_array';\n\nconst layout = createLayout([\n {name: 'a_pos', components: 2, type: 'Int16'},\n {name: 'a_normal_ed', components: 4, type: 'Int16'},\n], 4);\n\nexport const centroidAttributes = createLayout([\n {name: 'a_centroid', components: 2, type: 'Int16'}\n], 4);\n\nexport default layout;\nexport const {members, size, alignment} = layout;\n","'use strict';\n\nvar Point = require('@mapbox/point-geometry');\n\nmodule.exports = VectorTileFeature;\n\nfunction VectorTileFeature(pbf, end, extent, keys, values) {\n // Public\n this.properties = {};\n this.extent = extent;\n this.type = 0;\n\n // Private\n this._pbf = pbf;\n this._geometry = -1;\n this._keys = keys;\n this._values = values;\n\n pbf.readFields(readFeature, this, end);\n}\n\nfunction readFeature(tag, feature, pbf) {\n if (tag == 1) feature.id = pbf.readVarint();\n else if (tag == 2) readTag(pbf, feature);\n else if (tag == 3) feature.type = pbf.readVarint();\n else if (tag == 4) feature._geometry = pbf.pos;\n}\n\nfunction readTag(pbf, feature) {\n var end = pbf.readVarint() + pbf.pos;\n\n while (pbf.pos < end) {\n var key = feature._keys[pbf.readVarint()],\n value = feature._values[pbf.readVarint()];\n feature.properties[key] = value;\n }\n}\n\nVectorTileFeature.types = ['Unknown', 'Point', 'LineString', 'Polygon'];\n\nVectorTileFeature.prototype.loadGeometry = function() {\n var pbf = this._pbf;\n pbf.pos = this._geometry;\n\n var end = pbf.readVarint() + pbf.pos,\n cmd = 1,\n length = 0,\n x = 0,\n y = 0,\n lines = [],\n line;\n\n while (pbf.pos < end) {\n if (length <= 0) {\n var cmdLen = pbf.readVarint();\n cmd = cmdLen & 0x7;\n length = cmdLen >> 3;\n }\n\n length--;\n\n if (cmd === 1 || cmd === 2) {\n x += pbf.readSVarint();\n y += pbf.readSVarint();\n\n if (cmd === 1) { // moveTo\n if (line) lines.push(line);\n line = [];\n }\n\n line.push(new Point(x, y));\n\n } else if (cmd === 7) {\n\n // Workaround for https://github.com/mapbox/mapnik-vector-tile/issues/90\n if (line) {\n line.push(line[0].clone()); // closePolygon\n }\n\n } else {\n throw new Error('unknown command ' + cmd);\n }\n }\n\n if (line) lines.push(line);\n\n return lines;\n};\n\nVectorTileFeature.prototype.bbox = function() {\n var pbf = this._pbf;\n pbf.pos = this._geometry;\n\n var end = pbf.readVarint() + pbf.pos,\n cmd = 1,\n length = 0,\n x = 0,\n y = 0,\n x1 = Infinity,\n x2 = -Infinity,\n y1 = Infinity,\n y2 = -Infinity;\n\n while (pbf.pos < end) {\n if (length <= 0) {\n var cmdLen = pbf.readVarint();\n cmd = cmdLen & 0x7;\n length = cmdLen >> 3;\n }\n\n length--;\n\n if (cmd === 1 || cmd === 2) {\n x += pbf.readSVarint();\n y += pbf.readSVarint();\n if (x < x1) x1 = x;\n if (x > x2) x2 = x;\n if (y < y1) y1 = y;\n if (y > y2) y2 = y;\n\n } else if (cmd !== 7) {\n throw new Error('unknown command ' + cmd);\n }\n }\n\n return [x1, y1, x2, y2];\n};\n\nVectorTileFeature.prototype.toGeoJSON = function(x, y, z) {\n var size = this.extent * Math.pow(2, z),\n x0 = this.extent * x,\n y0 = this.extent * y,\n coords = this.loadGeometry(),\n type = VectorTileFeature.types[this.type],\n i, j;\n\n function project(line) {\n for (var j = 0; j < line.length; j++) {\n var p = line[j], y2 = 180 - (p.y + y0) * 360 / size;\n line[j] = [\n (p.x + x0) * 360 / size - 180,\n 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90\n ];\n }\n }\n\n switch (this.type) {\n case 1:\n var points = [];\n for (i = 0; i < coords.length; i++) {\n points[i] = coords[i][0];\n }\n coords = points;\n project(coords);\n break;\n\n case 2:\n for (i = 0; i < coords.length; i++) {\n project(coords[i]);\n }\n break;\n\n case 3:\n coords = classifyRings(coords);\n for (i = 0; i < coords.length; i++) {\n for (j = 0; j < coords[i].length; j++) {\n project(coords[i][j]);\n }\n }\n break;\n }\n\n if (coords.length === 1) {\n coords = coords[0];\n } else {\n type = 'Multi' + type;\n }\n\n var result = {\n type: \"Feature\",\n geometry: {\n type: type,\n coordinates: coords\n },\n properties: this.properties\n };\n\n if ('id' in this) {\n result.id = this.id;\n }\n\n return result;\n};\n\n// classifies an array of rings into polygons with outer rings and holes\n\nfunction classifyRings(rings) {\n var len = rings.length;\n\n if (len <= 1) return [rings];\n\n var polygons = [],\n polygon,\n ccw;\n\n for (var i = 0; i < len; i++) {\n var area = signedArea(rings[i]);\n if (area === 0) continue;\n\n if (ccw === undefined) ccw = area < 0;\n\n if (ccw === area < 0) {\n if (polygon) polygons.push(polygon);\n polygon = [rings[i]];\n\n } else {\n polygon.push(rings[i]);\n }\n }\n if (polygon) polygons.push(polygon);\n\n return polygons;\n}\n\nfunction signedArea(ring) {\n var sum = 0;\n for (var i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) {\n p1 = ring[i];\n p2 = ring[j];\n sum += (p2.x - p1.x) * (p1.y + p2.y);\n }\n return sum;\n}\n","'use strict';\n\nvar VectorTileFeature = require('./vectortilefeature.js');\n\nmodule.exports = VectorTileLayer;\n\nfunction VectorTileLayer(pbf, end) {\n // Public\n this.version = 1;\n this.name = null;\n this.extent = 4096;\n this.length = 0;\n\n // Private\n this._pbf = pbf;\n this._keys = [];\n this._values = [];\n this._features = [];\n\n pbf.readFields(readLayer, this, end);\n\n this.length = this._features.length;\n}\n\nfunction readLayer(tag, layer, pbf) {\n if (tag === 15) layer.version = pbf.readVarint();\n else if (tag === 1) layer.name = pbf.readString();\n else if (tag === 5) layer.extent = pbf.readVarint();\n else if (tag === 2) layer._features.push(pbf.pos);\n else if (tag === 3) layer._keys.push(pbf.readString());\n else if (tag === 4) layer._values.push(readValueMessage(pbf));\n}\n\nfunction readValueMessage(pbf) {\n var value = null,\n end = pbf.readVarint() + pbf.pos;\n\n while (pbf.pos < end) {\n var tag = pbf.readVarint() >> 3;\n\n value = tag === 1 ? pbf.readString() :\n tag === 2 ? pbf.readFloat() :\n tag === 3 ? pbf.readDouble() :\n tag === 4 ? pbf.readVarint64() :\n tag === 5 ? pbf.readVarint() :\n tag === 6 ? pbf.readSVarint() :\n tag === 7 ? pbf.readBoolean() : null;\n }\n\n return value;\n}\n\n// return feature `i` from this layer as a `VectorTileFeature`\nVectorTileLayer.prototype.feature = function(i) {\n if (i < 0 || i >= this._features.length) throw new Error('feature index out of bounds');\n\n this._pbf.pos = this._features[i];\n\n var end = this._pbf.readVarint() + this._pbf.pos;\n return new VectorTileFeature(this._pbf, end, this.extent, this._keys, this._values);\n};\n","'use strict';\n\nvar VectorTileLayer = require('./vectortilelayer');\n\nmodule.exports = VectorTile;\n\nfunction VectorTile(pbf, end) {\n this.layers = pbf.readFields(readTile, {}, end);\n}\n\nfunction readTile(tag, layers, pbf) {\n if (tag === 3) {\n var layer = new VectorTileLayer(pbf, pbf.readVarint() + pbf.pos);\n if (layer.length) layers[layer.name] = layer;\n }\n}\n\n","module.exports.VectorTile = require('./lib/vectortile.js');\nmodule.exports.VectorTileFeature = require('./lib/vectortilefeature.js');\nmodule.exports.VectorTileLayer = require('./lib/vectortilelayer.js');\n","import {FillExtrusionLayoutArray, PosArray} from '../array_types.g';\n\nimport {members as layoutAttributes, centroidAttributes} from './fill_extrusion_attributes';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray} from '../index_array_type';\nimport {EXTENT} from '../extent';\nimport earcut from 'earcut';\nimport mvt from '@mapbox/vector-tile';\nconst vectorTileFeatureTypes = mvt.VectorTileFeature.types;\nimport {classifyRings} from '../../util/classify_rings';\nconst EARCUT_MAX_RINGS = 500;\nimport {register} from '../../util/web_worker_transfer';\nimport {hasPattern, addPatternDependencies} from './pattern_bucket_features';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\n\nimport type {FillExtrusionStyleLayer} from '../../style/style_layer/fill_extrusion_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nconst FACTOR = Math.pow(2, 13);\n\nfunction addVertex(vertexArray, x, y, nx, ny, nz, t, e) {\n vertexArray.emplaceBack(\n // a_pos\n x,\n y,\n // a_normal_ed: 3-component normal and 1-component edgedistance\n Math.floor(nx * FACTOR) * 2 + t,\n ny * FACTOR * 2,\n nz * FACTOR * 2,\n // edgedistance (used for wrapping patterns around extrusion sides)\n Math.round(e)\n );\n}\n\nexport class FillExtrusionBucket implements Bucket {\n index: number;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n\n layoutVertexArray: FillExtrusionLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n centroidVertexArray: PosArray;\n centroidVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n uploaded: boolean;\n features: Array;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n\n this.layoutVertexArray = new FillExtrusionLayoutArray();\n this.centroidVertexArray = new PosArray();\n this.indexArray = new TriangleIndexArray();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.segments = new SegmentVector();\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n this.features = [];\n this.hasPattern = hasPattern('fill-extrusion', this.layers, options);\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const bucketFeature: BucketFeature = {\n id,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n properties: feature.properties,\n type: feature.type,\n patterns: {}\n };\n\n if (this.hasPattern) {\n this.features.push(addPatternDependencies('fill-extrusion', this.layers, bucketFeature, this.zoom, options));\n } else {\n this.addFeature(bucketFeature, bucketFeature.geometry, index, canonical, {});\n }\n\n options.featureIndex.insert(feature, bucketFeature.geometry, index, sourceLayerIndex, this.index, true);\n }\n }\n\n addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n for (const feature of this.features) {\n const {geometry} = feature;\n this.addFeature(feature, geometry, feature.index, canonical, imagePositions);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0 && this.centroidVertexArray.length === 0;\n }\n\n uploadPending() {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.centroidVertexBuffer = context.createVertexBuffer(this.centroidVertexArray, centroidAttributes.members, true);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n this.centroidVertexBuffer.destroy();\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n const centroid = {x: 0, y: 0, vertexCount: 0};\n for (const polygon of classifyRings(geometry, EARCUT_MAX_RINGS)) {\n let numVertices = 0;\n for (const ring of polygon) {\n numVertices += ring.length;\n }\n let segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray);\n\n for (const ring of polygon) {\n if (ring.length === 0) {\n continue;\n }\n\n if (isEntirelyOutside(ring)) {\n continue;\n }\n\n let edgeDistance = 0;\n\n for (let p = 0; p < ring.length; p++) {\n const p1 = ring[p];\n\n if (p >= 1) {\n const p2 = ring[p - 1];\n\n if (!isBoundaryEdge(p1, p2)) {\n if (segment.vertexLength + 4 > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) {\n segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray);\n }\n\n const perp = p1.sub(p2)._perp()._unit();\n const dist = p2.dist(p1);\n if (edgeDistance + dist > 32768) edgeDistance = 0;\n\n addVertex(this.layoutVertexArray, p1.x, p1.y, perp.x, perp.y, 0, 0, edgeDistance);\n addVertex(this.layoutVertexArray, p1.x, p1.y, perp.x, perp.y, 0, 1, edgeDistance);\n centroid.x += 2 * p1.x;\n centroid.y += 2 * p1.y;\n centroid.vertexCount += 2;\n\n edgeDistance += dist;\n\n addVertex(this.layoutVertexArray, p2.x, p2.y, perp.x, perp.y, 0, 0, edgeDistance);\n addVertex(this.layoutVertexArray, p2.x, p2.y, perp.x, perp.y, 0, 1, edgeDistance);\n centroid.x += 2 * p2.x;\n centroid.y += 2 * p2.y;\n centroid.vertexCount += 2;\n\n const bottomRight = segment.vertexLength;\n\n // ┌──────┐\n // │ 0 1 │ Counter-clockwise winding order.\n // │ │ Triangle 1: 0 => 2 => 1\n // │ 2 3 │ Triangle 2: 1 => 2 => 3\n // └──────┘\n this.indexArray.emplaceBack(bottomRight, bottomRight + 2, bottomRight + 1);\n this.indexArray.emplaceBack(bottomRight + 1, bottomRight + 2, bottomRight + 3);\n\n segment.vertexLength += 4;\n segment.primitiveLength += 2;\n }\n }\n }\n\n }\n\n if (segment.vertexLength + numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) {\n segment = this.segments.prepareSegment(numVertices, this.layoutVertexArray, this.indexArray);\n }\n\n //Only triangulate and draw the area of the feature if it is a polygon\n //Other feature types (e.g. LineString) do not have area, so triangulation is pointless / undefined\n if (vectorTileFeatureTypes[feature.type] !== 'Polygon')\n continue;\n\n const flattened = [];\n const holeIndices = [];\n const triangleIndex = segment.vertexLength;\n\n for (const ring of polygon) {\n if (ring.length === 0) {\n continue;\n }\n\n if (ring !== polygon[0]) {\n holeIndices.push(flattened.length / 2);\n }\n\n for (let i = 0; i < ring.length; i++) {\n const p = ring[i];\n\n addVertex(this.layoutVertexArray, p.x, p.y, 0, 0, 1, 1, 0);\n centroid.x += p.x;\n centroid.y += p.y;\n centroid.vertexCount += 1;\n\n flattened.push(p.x);\n flattened.push(p.y);\n }\n\n }\n\n const indices = earcut(flattened, holeIndices);\n\n for (let j = 0; j < indices.length; j += 3) {\n // Counter-clockwise winding order.\n this.indexArray.emplaceBack(\n triangleIndex + indices[j],\n triangleIndex + indices[j + 2],\n triangleIndex + indices[j + 1]);\n }\n\n segment.primitiveLength += indices.length / 3;\n segment.vertexLength += numVertices;\n }\n\n // remember polygon centroid to calculate elevation in GPU\n for (let i = 0; i < centroid.vertexCount; i++) {\n this.centroidVertexArray.emplaceBack(\n Math.floor(centroid.x / centroid.vertexCount),\n Math.floor(centroid.y / centroid.vertexCount)\n );\n }\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical);\n }\n}\n\nregister('FillExtrusionBucket', FillExtrusionBucket, {omit: ['layers', 'features']});\n\nfunction isBoundaryEdge(p1, p2) {\n return (p1.x === p2.x && (p1.x < 0 || p1.x > EXTENT)) ||\n (p1.y === p2.y && (p1.y < 0 || p1.y > EXTENT));\n}\n\nfunction isEntirelyOutside(ring) {\n return ring.every(p => p.x < 0) ||\n ring.every(p => p.x > EXTENT) ||\n ring.every(p => p.y < 0) ||\n ring.every(p => p.y > EXTENT);\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type FillExtrusionPaintProps = {\n \"fill-extrusion-opacity\": DataConstantProperty,\n \"fill-extrusion-color\": DataDrivenProperty,\n \"fill-extrusion-translate\": DataConstantProperty<[number, number]>,\n \"fill-extrusion-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"fill-extrusion-pattern\": CrossFadedDataDrivenProperty,\n \"fill-extrusion-height\": DataDrivenProperty,\n \"fill-extrusion-base\": DataDrivenProperty,\n \"fill-extrusion-vertical-gradient\": DataConstantProperty,\n};\n\nexport type FillExtrusionPaintPropsPossiblyEvaluated = {\n \"fill-extrusion-opacity\": number,\n \"fill-extrusion-color\": PossiblyEvaluatedPropertyValue,\n \"fill-extrusion-translate\": [number, number],\n \"fill-extrusion-translate-anchor\": \"map\" | \"viewport\",\n \"fill-extrusion-pattern\": PossiblyEvaluatedPropertyValue>,\n \"fill-extrusion-height\": PossiblyEvaluatedPropertyValue,\n \"fill-extrusion-base\": PossiblyEvaluatedPropertyValue,\n \"fill-extrusion-vertical-gradient\": boolean,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"fill-extrusion-opacity\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-opacity\"] as any as StylePropertySpecification),\n \"fill-extrusion-color\": new DataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-color\"] as any as StylePropertySpecification),\n \"fill-extrusion-translate\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-translate\"] as any as StylePropertySpecification),\n \"fill-extrusion-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-translate-anchor\"] as any as StylePropertySpecification),\n \"fill-extrusion-pattern\": new CrossFadedDataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-pattern\"] as any as StylePropertySpecification),\n \"fill-extrusion-height\": new DataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-height\"] as any as StylePropertySpecification),\n \"fill-extrusion-base\": new DataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-base\"] as any as StylePropertySpecification),\n \"fill-extrusion-vertical-gradient\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-vertical-gradient\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport {FillExtrusionBucket} from '../../data/bucket/fill_extrusion_bucket';\nimport {polygonIntersectsPolygon, polygonIntersectsMultiPolygon} from '../../util/intersection_tests';\nimport {translateDistance, translate} from '../query_utils';\nimport properties, {FillExtrusionPaintPropsPossiblyEvaluated} from './fill_extrusion_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\nimport {mat4, vec4} from 'gl-matrix';\nimport Point from '@mapbox/point-geometry';\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {BucketParameters} from '../../data/bucket';\nimport type {FillExtrusionPaintProps} from './fill_extrusion_style_layer_properties.g';\nimport type {Transform} from '../../geo/transform';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nexport class Point3D extends Point {\n z: number;\n}\n\nexport class FillExtrusionStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n createBucket(parameters: BucketParameters) {\n return new FillExtrusionBucket(parameters);\n }\n\n queryRadius(): number {\n return translateDistance(this.paint.get('fill-extrusion-translate'));\n }\n\n is3D(): boolean {\n return true;\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number,\n pixelPosMatrix: mat4\n ): boolean | number {\n\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('fill-extrusion-translate'),\n this.paint.get('fill-extrusion-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n\n const height = this.paint.get('fill-extrusion-height').evaluate(feature, featureState);\n const base = this.paint.get('fill-extrusion-base').evaluate(feature, featureState);\n\n const projectedQueryGeometry = projectQueryGeometry(translatedPolygon, pixelPosMatrix, transform, 0);\n\n const projected = projectExtrusion(geometry, base, height, pixelPosMatrix);\n const projectedBase = projected[0];\n const projectedTop = projected[1];\n return checkIntersection(projectedBase, projectedTop, projectedQueryGeometry);\n }\n}\n\nfunction dot(a, b) {\n return a.x * b.x + a.y * b.y;\n}\n\nexport function getIntersectionDistance(projectedQueryGeometry: Array, projectedFace: Array) {\n\n if (projectedQueryGeometry.length === 1) {\n // For point queries calculate the z at which the point intersects the face\n // using barycentric coordinates.\n\n // Find the barycentric coordinates of the projected point within the first\n // triangle of the face, using only the xy plane. It doesn't matter if the\n // point is outside the first triangle because all the triangles in the face\n // are in the same plane.\n //\n // Check whether points are coincident and use other points if they are.\n let i = 0;\n const a = projectedFace[i++];\n let b;\n while (!b || a.equals(b)) {\n b = projectedFace[i++];\n if (!b) return Infinity;\n }\n\n // Loop until point `c` is not colinear with points `a` and `b`.\n for (; i < projectedFace.length; i++) {\n const c = projectedFace[i];\n\n const p = projectedQueryGeometry[0];\n\n const ab = b.sub(a);\n const ac = c.sub(a);\n const ap = p.sub(a);\n\n const dotABAB = dot(ab, ab);\n const dotABAC = dot(ab, ac);\n const dotACAC = dot(ac, ac);\n const dotAPAB = dot(ap, ab);\n const dotAPAC = dot(ap, ac);\n const denom = dotABAB * dotACAC - dotABAC * dotABAC;\n\n const v = (dotACAC * dotAPAB - dotABAC * dotAPAC) / denom;\n const w = (dotABAB * dotAPAC - dotABAC * dotAPAB) / denom;\n const u = 1 - v - w;\n\n // Use the barycentric weighting along with the original triangle z coordinates to get the point of intersection.\n const distance = a.z * u + b.z * v + c.z * w;\n\n if (isFinite(distance)) return distance;\n }\n\n return Infinity;\n\n } else {\n // The counts as closest is less clear when the query is a box. This\n // returns the distance to the nearest point on the face, whether it is\n // within the query or not. It could be more correct to return the\n // distance to the closest point within the query box but this would be\n // more complicated and expensive to calculate with little benefit.\n let closestDistance = Infinity;\n for (const p of projectedFace) {\n closestDistance = Math.min(closestDistance, p.z);\n }\n return closestDistance;\n }\n}\n\nfunction checkIntersection(projectedBase: Array>, projectedTop: Array>, projectedQueryGeometry: Array) {\n let closestDistance = Infinity;\n\n if (polygonIntersectsMultiPolygon(projectedQueryGeometry, projectedTop)) {\n closestDistance = getIntersectionDistance(projectedQueryGeometry, projectedTop[0]);\n }\n\n for (let r = 0; r < projectedTop.length; r++) {\n const ringTop = projectedTop[r];\n const ringBase = projectedBase[r];\n for (let p = 0; p < ringTop.length - 1; p++) {\n const topA = ringTop[p];\n const topB = ringTop[p + 1];\n const baseA = ringBase[p];\n const baseB = ringBase[p + 1];\n const face = [topA, topB, baseB, baseA, topA];\n if (polygonIntersectsPolygon(projectedQueryGeometry, face)) {\n closestDistance = Math.min(closestDistance, getIntersectionDistance(projectedQueryGeometry, face));\n }\n }\n }\n\n return closestDistance === Infinity ? false : closestDistance;\n}\n\n/*\n * Project the geometry using matrix `m`. This is essentially doing\n * `vec4.transformMat4([], [p.x, p.y, z, 1], m)` but the multiplication\n * is inlined so that parts of the projection that are the same across\n * different points can only be done once. This produced a measurable\n * performance improvement.\n */\nfunction projectExtrusion(geometry: Array>, zBase: number, zTop: number, m: mat4): [Array>, Array>] {\n const projectedBase = [] as Array>;\n const projectedTop = [] as Array>;\n const baseXZ = m[8] * zBase;\n const baseYZ = m[9] * zBase;\n const baseZZ = m[10] * zBase;\n const baseWZ = m[11] * zBase;\n const topXZ = m[8] * zTop;\n const topYZ = m[9] * zTop;\n const topZZ = m[10] * zTop;\n const topWZ = m[11] * zTop;\n\n for (const r of geometry) {\n const ringBase = [] as Array;\n const ringTop = [] as Array;\n for (const p of r) {\n const x = p.x;\n const y = p.y;\n\n const sX = m[0] * x + m[4] * y + m[12];\n const sY = m[1] * x + m[5] * y + m[13];\n const sZ = m[2] * x + m[6] * y + m[14];\n const sW = m[3] * x + m[7] * y + m[15];\n\n const baseX = sX + baseXZ;\n const baseY = sY + baseYZ;\n const baseZ = sZ + baseZZ;\n const baseW = sW + baseWZ;\n\n const topX = sX + topXZ;\n const topY = sY + topYZ;\n const topZ = sZ + topZZ;\n const topW = sW + topWZ;\n\n const b = new Point(baseX / baseW, baseY / baseW) as Point3D;\n b.z = baseZ / baseW;\n ringBase.push(b);\n\n const t = new Point(topX / topW, topY / topW) as Point3D;\n t.z = topZ / topW;\n ringTop.push(t);\n }\n projectedBase.push(ringBase);\n projectedTop.push(ringTop);\n }\n return [projectedBase, projectedTop];\n}\n\nfunction projectQueryGeometry(queryGeometry: Array, pixelPosMatrix: mat4, transform: Transform, z: number) {\n const projectedQueryGeometry = [];\n for (const p of queryGeometry) {\n const v = [p.x, p.y, z, 1] as vec4;\n vec4.transformMat4(v, v, pixelPosMatrix);\n projectedQueryGeometry.push(new Point(v[0] / v[3], v[1] / v[3]));\n }\n return projectedQueryGeometry;\n}\n","import {createLayout} from '../../util/struct_array';\n\nexport const lineLayoutAttributes = createLayout([\n {name: 'a_pos_normal', components: 2, type: 'Int16'},\n {name: 'a_data', components: 4, type: 'Uint8'}\n], 4);\n\nexport const {members, size, alignment} = lineLayoutAttributes;\n","import {createLayout} from '../../util/struct_array';\n\nexport const lineLayoutAttributesExt = createLayout([\n {name: 'a_uv_x', components: 1, type: 'Float32'},\n {name: 'a_split_index', components: 1, type: 'Float32'},\n]);\n\nexport const {members, size, alignment} = lineLayoutAttributesExt;\n","import {LineLayoutArray, LineExtLayoutArray} from '../array_types.g';\n\nimport {members as layoutAttributes} from './line_attributes';\nimport {members as layoutAttributesExt} from './line_attributes_ext';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray} from '../index_array_type';\nimport {EXTENT} from '../extent';\nimport mvt from '@mapbox/vector-tile';\nconst vectorTileFeatureTypes = mvt.VectorTileFeature.types;\nimport {register} from '../../util/web_worker_transfer';\nimport {hasPattern, addPatternDependencies} from './pattern_bucket_features';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {LineStyleLayer} from '../../style/style_layer/line_style_layer';\nimport type Point from '@mapbox/point-geometry';\nimport type {Segment} from '../segment';\nimport {RGBAImage} from '../../util/image';\nimport type {Context} from '../../gl/context';\nimport type {Texture} from '../../render/texture';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\n// NOTE ON EXTRUDE SCALE:\n// scale the extrusion vector so that the normal length is this value.\n// contains the \"texture\" normals (-1..1). this is distinct from the extrude\n// normals for line joins, because the x-value remains 0 for the texture\n// normal array, while the extrude normal actually moves the vertex to create\n// the acute/bevelled line join.\nconst EXTRUDE_SCALE = 63;\n\n/*\n * Sharp corners cause dashed lines to tilt because the distance along the line\n * is the same at both the inner and outer corners. To improve the appearance of\n * dashed lines we add extra points near sharp corners so that a smaller part\n * of the line is tilted.\n *\n * COS_HALF_SHARP_CORNER controls how sharp a corner has to be for us to add an\n * extra vertex. The default is 75 degrees.\n *\n * The newly created vertices are placed SHARP_CORNER_OFFSET pixels from the corner.\n */\nconst COS_HALF_SHARP_CORNER = Math.cos(75 / 2 * (Math.PI / 180));\nconst SHARP_CORNER_OFFSET = 15;\n\n// Angle per triangle for approximating round line joins.\nconst DEG_PER_TRIANGLE = 20;\n\n// The number of bits that is used to store the line distance in the buffer.\nconst LINE_DISTANCE_BUFFER_BITS = 15;\n\n// We don't have enough bits for the line distance as we'd like to have, so\n// use this value to scale the line distance (in tile units) down to a smaller\n// value. This lets us store longer distances while sacrificing precision.\nconst LINE_DISTANCE_SCALE = 1 / 2;\n\n// The maximum line distance, in tile units, that fits in the buffer.\nconst MAX_LINE_DISTANCE = Math.pow(2, LINE_DISTANCE_BUFFER_BITS - 1) / LINE_DISTANCE_SCALE;\n\ntype LineClips = {\n start: number;\n end: number;\n};\n\ntype GradientTexture = {\n texture?: Texture;\n gradient?: RGBAImage;\n version?: number;\n};\n\n/**\n * @internal\n * Line bucket class\n */\nexport class LineBucket implements Bucket {\n distance: number;\n totalDistance: number;\n maxLineLength: number;\n scaledDistance: number;\n lineClips?: LineClips;\n\n e1: number;\n e2: number;\n\n index: number;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n gradients: {[x: string]: GradientTexture};\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n patternFeatures: Array;\n lineClipsArray: Array;\n\n layoutVertexArray: LineLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n layoutVertexArray2: LineExtLayoutArray;\n layoutVertexBuffer2: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n uploaded: boolean;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n this.patternFeatures = [];\n this.lineClipsArray = [];\n this.gradients = {};\n this.layers.forEach(layer => {\n this.gradients[layer.id] = {};\n });\n\n this.layoutVertexArray = new LineLayoutArray();\n this.layoutVertexArray2 = new LineExtLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.segments = new SegmentVector();\n this.maxLineLength = 0;\n\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n this.hasPattern = hasPattern('line', this.layers, options);\n const lineSortKey = this.layers[0].layout.get('line-sort-key');\n const sortFeaturesByKey = !lineSortKey.isConstant();\n const bucketFeatures: BucketFeature[] = [];\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const sortKey = sortFeaturesByKey ?\n lineSortKey.evaluate(evaluationFeature, {}, canonical) :\n undefined;\n\n const bucketFeature: BucketFeature = {\n id,\n properties: feature.properties,\n type: feature.type,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n patterns: {},\n sortKey\n };\n\n bucketFeatures.push(bucketFeature);\n }\n\n if (sortFeaturesByKey) {\n bucketFeatures.sort((a, b) => {\n return (a.sortKey) - (b.sortKey);\n });\n }\n\n for (const bucketFeature of bucketFeatures) {\n const {geometry, index, sourceLayerIndex} = bucketFeature;\n\n if (this.hasPattern) {\n const patternBucketFeature = addPatternDependencies('line', this.layers, bucketFeature, this.zoom, options);\n // pattern features are added only once the pattern is loaded into the image atlas\n // so are stored during populate until later updated with positions by tile worker in addFeatures\n this.patternFeatures.push(patternBucketFeature);\n } else {\n this.addFeature(bucketFeature, geometry, index, canonical, {});\n }\n\n const feature = features[index].feature;\n options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n for (const feature of this.patternFeatures) {\n this.addFeature(feature, feature.geometry, feature.index, canonical, imagePositions);\n }\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0;\n }\n\n uploadPending() {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded) {\n if (this.layoutVertexArray2.length !== 0) {\n this.layoutVertexBuffer2 = context.createVertexBuffer(this.layoutVertexArray2, layoutAttributesExt);\n }\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n }\n\n lineFeatureClips(feature: BucketFeature): LineClips | undefined {\n if (!!feature.properties && Object.prototype.hasOwnProperty.call(feature.properties, 'mapbox_clip_start') && Object.prototype.hasOwnProperty.call(feature.properties, 'mapbox_clip_end')) {\n const start = +feature.properties['mapbox_clip_start'];\n const end = +feature.properties['mapbox_clip_end'];\n return {start, end};\n }\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n const layout = this.layers[0].layout;\n const join = layout.get('line-join').evaluate(feature, {});\n const cap = layout.get('line-cap');\n const miterLimit = layout.get('line-miter-limit');\n const roundLimit = layout.get('line-round-limit');\n this.lineClips = this.lineFeatureClips(feature);\n\n for (const line of geometry) {\n this.addLine(line, feature, join, cap, miterLimit, roundLimit);\n }\n\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical);\n }\n\n addLine(vertices: Array, feature: BucketFeature, join: string, cap: string, miterLimit: number, roundLimit: number) {\n this.distance = 0;\n this.scaledDistance = 0;\n this.totalDistance = 0;\n\n if (this.lineClips) {\n this.lineClipsArray.push(this.lineClips);\n // Calculate the total distance, in tile units, of this tiled line feature\n for (let i = 0; i < vertices.length - 1; i++) {\n this.totalDistance += vertices[i].dist(vertices[i + 1]);\n }\n this.updateScaledDistance();\n this.maxLineLength = Math.max(this.maxLineLength, this.totalDistance);\n }\n\n const isPolygon = vectorTileFeatureTypes[feature.type] === 'Polygon';\n\n // If the line has duplicate vertices at the ends, adjust start/length to remove them.\n let len = vertices.length;\n while (len >= 2 && vertices[len - 1].equals(vertices[len - 2])) {\n len--;\n }\n let first = 0;\n while (first < len - 1 && vertices[first].equals(vertices[first + 1])) {\n first++;\n }\n\n // Ignore invalid geometry.\n if (len < (isPolygon ? 3 : 2)) return;\n\n if (join === 'bevel') miterLimit = 1.05;\n\n const sharpCornerOffset = this.overscaling <= 16 ?\n SHARP_CORNER_OFFSET * EXTENT / (512 * this.overscaling) :\n 0;\n\n // we could be more precise, but it would only save a negligible amount of space\n const segment = this.segments.prepareSegment(len * 10, this.layoutVertexArray, this.indexArray);\n\n let currentVertex: Point;\n let prevVertex: Point;\n let nextVertex: Point;\n let prevNormal: Point;\n let nextNormal: Point;\n\n // the last two vertices added\n this.e1 = this.e2 = -1;\n\n if (isPolygon) {\n currentVertex = vertices[len - 2];\n nextNormal = vertices[first].sub(currentVertex)._unit()._perp();\n }\n\n for (let i = first; i < len; i++) {\n\n nextVertex = i === len - 1 ?\n (isPolygon ? vertices[first + 1] : undefined) : // if it's a polygon, treat the last vertex like the first\n vertices[i + 1]; // just the next vertex\n\n // if two consecutive vertices exist, skip the current one\n if (nextVertex && vertices[i].equals(nextVertex)) continue;\n\n if (nextNormal) prevNormal = nextNormal;\n if (currentVertex) prevVertex = currentVertex;\n\n currentVertex = vertices[i];\n\n // Calculate the normal towards the next vertex in this line. In case\n // there is no next vertex, pretend that the line is continuing straight,\n // meaning that we are just using the previous normal.\n nextNormal = nextVertex ? nextVertex.sub(currentVertex)._unit()._perp() : prevNormal;\n\n // If we still don't have a previous normal, this is the beginning of a\n // non-closed line, so we're doing a straight \"join\".\n prevNormal = prevNormal || nextNormal;\n\n // Determine the normal of the join extrusion. It is the angle bisector\n // of the segments between the previous line and the next line.\n // In the case of 180° angles, the prev and next normals cancel each other out:\n // prevNormal + nextNormal = (0, 0), its magnitude is 0, so the unit vector would be\n // undefined. In that case, we're keeping the joinNormal at (0, 0), so that the cosHalfAngle\n // below will also become 0 and miterLength will become Infinity.\n let joinNormal = prevNormal.add(nextNormal);\n if (joinNormal.x !== 0 || joinNormal.y !== 0) {\n joinNormal._unit();\n }\n /* joinNormal prevNormal\n * ↖ ↑\n * .________. prevVertex\n * |\n * nextNormal ← | currentVertex\n * |\n * nextVertex !\n *\n */\n\n // calculate cosines of the angle (and its half) using dot product\n const cosAngle = prevNormal.x * nextNormal.x + prevNormal.y * nextNormal.y;\n const cosHalfAngle = joinNormal.x * nextNormal.x + joinNormal.y * nextNormal.y;\n\n // Calculate the length of the miter (the ratio of the miter to the width)\n // as the inverse of cosine of the angle between next and join normals\n const miterLength = cosHalfAngle !== 0 ? 1 / cosHalfAngle : Infinity;\n\n // approximate angle from cosine\n const approxAngle = 2 * Math.sqrt(2 - 2 * cosHalfAngle);\n\n const isSharpCorner = cosHalfAngle < COS_HALF_SHARP_CORNER && prevVertex && nextVertex;\n const lineTurnsLeft = prevNormal.x * nextNormal.y - prevNormal.y * nextNormal.x > 0;\n\n if (isSharpCorner && i > first) {\n const prevSegmentLength = currentVertex.dist(prevVertex);\n if (prevSegmentLength > 2 * sharpCornerOffset) {\n const newPrevVertex = currentVertex.sub(currentVertex.sub(prevVertex)._mult(sharpCornerOffset / prevSegmentLength)._round());\n this.updateDistance(prevVertex, newPrevVertex);\n this.addCurrentVertex(newPrevVertex, prevNormal, 0, 0, segment);\n prevVertex = newPrevVertex;\n }\n }\n\n // The join if a middle vertex, otherwise the cap.\n const middleVertex = prevVertex && nextVertex;\n let currentJoin = middleVertex ? join : isPolygon ? 'butt' : cap;\n\n if (middleVertex && currentJoin === 'round') {\n if (miterLength < roundLimit) {\n currentJoin = 'miter';\n } else if (miterLength <= 2) {\n currentJoin = 'fakeround';\n }\n }\n\n if (currentJoin === 'miter' && miterLength > miterLimit) {\n currentJoin = 'bevel';\n }\n\n if (currentJoin === 'bevel') {\n // The maximum extrude length is 128 / 63 = 2 times the width of the line\n // so if miterLength >= 2 we need to draw a different type of bevel here.\n if (miterLength > 2) currentJoin = 'flipbevel';\n\n // If the miterLength is really small and the line bevel wouldn't be visible,\n // just draw a miter join to save a triangle.\n if (miterLength < miterLimit) currentJoin = 'miter';\n }\n\n // Calculate how far along the line the currentVertex is\n if (prevVertex) this.updateDistance(prevVertex, currentVertex);\n\n if (currentJoin === 'miter') {\n\n joinNormal._mult(miterLength);\n this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment);\n\n } else if (currentJoin === 'flipbevel') {\n // miter is too big, flip the direction to make a beveled join\n\n if (miterLength > 100) {\n // Almost parallel lines\n joinNormal = nextNormal.mult(-1);\n\n } else {\n const bevelLength = miterLength * prevNormal.add(nextNormal).mag() / prevNormal.sub(nextNormal).mag();\n joinNormal._perp()._mult(bevelLength * (lineTurnsLeft ? -1 : 1));\n }\n this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment);\n this.addCurrentVertex(currentVertex, joinNormal.mult(-1), 0, 0, segment);\n\n } else if (currentJoin === 'bevel' || currentJoin === 'fakeround') {\n const offset = -Math.sqrt(miterLength * miterLength - 1);\n const offsetA = lineTurnsLeft ? offset : 0;\n const offsetB = lineTurnsLeft ? 0 : offset;\n\n // Close previous segment with a bevel\n if (prevVertex) {\n this.addCurrentVertex(currentVertex, prevNormal, offsetA, offsetB, segment);\n }\n\n if (currentJoin === 'fakeround') {\n // The join angle is sharp enough that a round join would be visible.\n // Bevel joins fill the gap between segments with a single pie slice triangle.\n // Create a round join by adding multiple pie slices. The join isn't actually round, but\n // it looks like it is at the sizes we render lines at.\n\n // pick the number of triangles for approximating round join by based on the angle between normals\n const n = Math.round((approxAngle * 180 / Math.PI) / DEG_PER_TRIANGLE);\n\n for (let m = 1; m < n; m++) {\n let t = m / n;\n if (t !== 0.5) {\n // approximate spherical interpolation https://observablehq.com/@mourner/approximating-geometric-slerp\n const t2 = t - 0.5;\n const A = 1.0904 + cosAngle * (-3.2452 + cosAngle * (3.55645 - cosAngle * 1.43519));\n const B = 0.848013 + cosAngle * (-1.06021 + cosAngle * 0.215638);\n t = t + t * t2 * (t - 1) * (A * t2 * t2 + B);\n }\n const extrude = nextNormal.sub(prevNormal)._mult(t)._add(prevNormal)._unit()._mult(lineTurnsLeft ? -1 : 1);\n this.addHalfVertex(currentVertex, extrude.x, extrude.y, false, lineTurnsLeft, 0, segment);\n }\n }\n\n if (nextVertex) {\n // Start next segment\n this.addCurrentVertex(currentVertex, nextNormal, -offsetA, -offsetB, segment);\n }\n\n } else if (currentJoin === 'butt') {\n this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment); // butt cap\n\n } else if (currentJoin === 'square') {\n const offset = prevVertex ? 1 : -1; // closing or starting square cap\n this.addCurrentVertex(currentVertex, joinNormal, offset, offset, segment);\n\n } else if (currentJoin === 'round') {\n\n if (prevVertex) {\n // Close previous segment with butt\n this.addCurrentVertex(currentVertex, prevNormal, 0, 0, segment);\n\n // Add round cap or linejoin at end of segment\n this.addCurrentVertex(currentVertex, prevNormal, 1, 1, segment, true);\n }\n if (nextVertex) {\n // Add round cap before first segment\n this.addCurrentVertex(currentVertex, nextNormal, -1, -1, segment, true);\n\n // Start next segment with a butt\n this.addCurrentVertex(currentVertex, nextNormal, 0, 0, segment);\n }\n }\n\n if (isSharpCorner && i < len - 1) {\n const nextSegmentLength = currentVertex.dist(nextVertex);\n if (nextSegmentLength > 2 * sharpCornerOffset) {\n const newCurrentVertex = currentVertex.add(nextVertex.sub(currentVertex)._mult(sharpCornerOffset / nextSegmentLength)._round());\n this.updateDistance(currentVertex, newCurrentVertex);\n this.addCurrentVertex(newCurrentVertex, nextNormal, 0, 0, segment);\n currentVertex = newCurrentVertex;\n }\n }\n }\n }\n\n /**\n * Add two vertices to the buffers.\n *\n * @param p - the line vertex to add buffer vertices for\n * @param normal - vertex normal\n * @param endLeft - extrude to shift the left vertex along the line\n * @param endRight - extrude to shift the left vertex along the line\n * @param segment - the segment object to add the vertex to\n * @param round - whether this is a round cap\n */\n addCurrentVertex(p: Point, normal: Point, endLeft: number, endRight: number, segment: Segment, round: boolean = false) {\n // left and right extrude vectors, perpendicularly shifted by endLeft/endRight\n const leftX = normal.x + normal.y * endLeft;\n const leftY = normal.y - normal.x * endLeft;\n const rightX = -normal.x + normal.y * endRight;\n const rightY = -normal.y - normal.x * endRight;\n\n this.addHalfVertex(p, leftX, leftY, round, false, endLeft, segment);\n this.addHalfVertex(p, rightX, rightY, round, true, -endRight, segment);\n\n // There is a maximum \"distance along the line\" that we can store in the buffers.\n // When we get close to the distance, reset it to zero and add the vertex again with\n // a distance of zero. The max distance is determined by the number of bits we allocate\n // to `linesofar`.\n if (this.distance > MAX_LINE_DISTANCE / 2 && this.totalDistance === 0) {\n this.distance = 0;\n this.updateScaledDistance();\n this.addCurrentVertex(p, normal, endLeft, endRight, segment, round);\n }\n }\n\n addHalfVertex({x, y}: Point, extrudeX: number, extrudeY: number, round: boolean, up: boolean, dir: number, segment: Segment) {\n const totalDistance = this.lineClips ? this.scaledDistance * (MAX_LINE_DISTANCE - 1) : this.scaledDistance;\n // scale down so that we can store longer distances while sacrificing precision.\n const linesofarScaled = totalDistance * LINE_DISTANCE_SCALE;\n\n this.layoutVertexArray.emplaceBack(\n // a_pos_normal\n // Encode round/up the least significant bits\n (x << 1) + (round ? 1 : 0),\n (y << 1) + (up ? 1 : 0),\n // a_data\n // add 128 to store a byte in an unsigned byte\n Math.round(EXTRUDE_SCALE * extrudeX) + 128,\n Math.round(EXTRUDE_SCALE * extrudeY) + 128,\n // Encode the -1/0/1 direction value into the first two bits of .z of a_data.\n // Combine it with the lower 6 bits of `linesofarScaled` (shifted by 2 bits to make\n // room for the direction value). The upper 8 bits of `linesofarScaled` are placed in\n // the `w` component.\n ((dir === 0 ? 0 : (dir < 0 ? -1 : 1)) + 1) | ((linesofarScaled & 0x3F) << 2),\n linesofarScaled >> 6);\n\n // Constructs a second vertex buffer with higher precision line progress\n if (this.lineClips) {\n const progressRealigned = this.scaledDistance - this.lineClips.start;\n const endClipRealigned = this.lineClips.end - this.lineClips.start;\n const uvX = progressRealigned / endClipRealigned;\n this.layoutVertexArray2.emplaceBack(uvX, this.lineClipsArray.length);\n }\n\n const e = segment.vertexLength++;\n if (this.e1 >= 0 && this.e2 >= 0) {\n this.indexArray.emplaceBack(this.e1, this.e2, e);\n segment.primitiveLength++;\n }\n if (up) {\n this.e2 = e;\n } else {\n this.e1 = e;\n }\n }\n\n updateScaledDistance() {\n // Knowing the ratio of the full linestring covered by this tiled feature, as well\n // as the total distance (in tile units) of this tiled feature, and the distance\n // (in tile units) of the current vertex, we can determine the relative distance\n // of this vertex along the full linestring feature and scale it to [0, 2^15)\n this.scaledDistance = this.lineClips ?\n this.lineClips.start + (this.lineClips.end - this.lineClips.start) * this.distance / this.totalDistance :\n this.distance;\n }\n\n updateDistance(prev: Point, next: Point) {\n this.distance += prev.dist(next);\n this.updateScaledDistance();\n }\n}\n\nregister('LineBucket', LineBucket, {omit: ['layers', 'patternFeatures']});\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type LineLayoutProps = {\n \"line-cap\": DataConstantProperty<\"butt\" | \"round\" | \"square\">,\n \"line-join\": DataDrivenProperty<\"bevel\" | \"round\" | \"miter\">,\n \"line-miter-limit\": DataConstantProperty,\n \"line-round-limit\": DataConstantProperty,\n \"line-sort-key\": DataDrivenProperty,\n};\n\nexport type LineLayoutPropsPossiblyEvaluated = {\n \"line-cap\": \"butt\" | \"round\" | \"square\",\n \"line-join\": PossiblyEvaluatedPropertyValue<\"bevel\" | \"round\" | \"miter\">,\n \"line-miter-limit\": number,\n \"line-round-limit\": number,\n \"line-sort-key\": PossiblyEvaluatedPropertyValue,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"line-cap\": new DataConstantProperty(styleSpec[\"layout_line\"][\"line-cap\"] as any as StylePropertySpecification),\n \"line-join\": new DataDrivenProperty(styleSpec[\"layout_line\"][\"line-join\"] as any as StylePropertySpecification),\n \"line-miter-limit\": new DataConstantProperty(styleSpec[\"layout_line\"][\"line-miter-limit\"] as any as StylePropertySpecification),\n \"line-round-limit\": new DataConstantProperty(styleSpec[\"layout_line\"][\"line-round-limit\"] as any as StylePropertySpecification),\n \"line-sort-key\": new DataDrivenProperty(styleSpec[\"layout_line\"][\"line-sort-key\"] as any as StylePropertySpecification),\n});\n\nexport type LinePaintProps = {\n \"line-opacity\": DataDrivenProperty,\n \"line-color\": DataDrivenProperty,\n \"line-translate\": DataConstantProperty<[number, number]>,\n \"line-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"line-width\": DataDrivenProperty,\n \"line-gap-width\": DataDrivenProperty,\n \"line-offset\": DataDrivenProperty,\n \"line-blur\": DataDrivenProperty,\n \"line-dasharray\": CrossFadedProperty>,\n \"line-pattern\": CrossFadedDataDrivenProperty,\n \"line-gradient\": ColorRampProperty,\n};\n\nexport type LinePaintPropsPossiblyEvaluated = {\n \"line-opacity\": PossiblyEvaluatedPropertyValue,\n \"line-color\": PossiblyEvaluatedPropertyValue,\n \"line-translate\": [number, number],\n \"line-translate-anchor\": \"map\" | \"viewport\",\n \"line-width\": PossiblyEvaluatedPropertyValue,\n \"line-gap-width\": PossiblyEvaluatedPropertyValue,\n \"line-offset\": PossiblyEvaluatedPropertyValue,\n \"line-blur\": PossiblyEvaluatedPropertyValue,\n \"line-dasharray\": CrossFaded>,\n \"line-pattern\": PossiblyEvaluatedPropertyValue>,\n \"line-gradient\": ColorRampProperty,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"line-opacity\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-opacity\"] as any as StylePropertySpecification),\n \"line-color\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-color\"] as any as StylePropertySpecification),\n \"line-translate\": new DataConstantProperty(styleSpec[\"paint_line\"][\"line-translate\"] as any as StylePropertySpecification),\n \"line-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_line\"][\"line-translate-anchor\"] as any as StylePropertySpecification),\n \"line-width\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-width\"] as any as StylePropertySpecification),\n \"line-gap-width\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-gap-width\"] as any as StylePropertySpecification),\n \"line-offset\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-offset\"] as any as StylePropertySpecification),\n \"line-blur\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-blur\"] as any as StylePropertySpecification),\n \"line-dasharray\": new CrossFadedProperty(styleSpec[\"paint_line\"][\"line-dasharray\"] as any as StylePropertySpecification),\n \"line-pattern\": new CrossFadedDataDrivenProperty(styleSpec[\"paint_line\"][\"line-pattern\"] as any as StylePropertySpecification),\n \"line-gradient\": new ColorRampProperty(styleSpec[\"paint_line\"][\"line-gradient\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import Point from '@mapbox/point-geometry';\n\nimport {StyleLayer} from '../style_layer';\nimport {LineBucket} from '../../data/bucket/line_bucket';\nimport {polygonIntersectsBufferedMultiLine} from '../../util/intersection_tests';\nimport {getMaximumPaintValue, translateDistance, translate, offsetLine} from '../query_utils';\nimport properties, {LineLayoutPropsPossiblyEvaluated, LinePaintPropsPossiblyEvaluated} from './line_style_layer_properties.g';\nimport {extend} from '../../util/util';\nimport {EvaluationParameters} from '../evaluation_parameters';\nimport {Transitionable, Transitioning, Layout, PossiblyEvaluated, DataDrivenProperty} from '../properties';\n\nimport {isZoomExpression, Step} from '@maplibre/maplibre-gl-style-spec';\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Bucket, BucketParameters} from '../../data/bucket';\nimport type {LineLayoutProps, LinePaintProps} from './line_style_layer_properties.g';\nimport type {Transform} from '../../geo/transform';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nexport class LineFloorwidthProperty extends DataDrivenProperty {\n useIntegerZoom: true;\n\n possiblyEvaluate(value, parameters) {\n parameters = new EvaluationParameters(Math.floor(parameters.zoom), {\n now: parameters.now,\n fadeDuration: parameters.fadeDuration,\n zoomHistory: parameters.zoomHistory,\n transition: parameters.transition\n });\n return super.possiblyEvaluate(value, parameters);\n }\n\n evaluate(value, globals, feature, featureState) {\n globals = extend({}, globals, {zoom: Math.floor(globals.zoom)});\n return super.evaluate(value, globals, feature, featureState);\n }\n}\n\nlet lineFloorwidthProperty: LineFloorwidthProperty;\n\nexport class LineStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n gradientVersion: number;\n stepInterpolant: boolean;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n this.gradientVersion = 0;\n if (!lineFloorwidthProperty) {\n lineFloorwidthProperty =\n new LineFloorwidthProperty(properties.paint.properties['line-width'].specification);\n lineFloorwidthProperty.useIntegerZoom = true;\n }\n }\n\n _handleSpecialPaintPropertyUpdate(name: string) {\n if (name === 'line-gradient') {\n const expression = this.gradientExpression();\n if (isZoomExpression(expression)) {\n this.stepInterpolant = expression._styleExpression.expression instanceof Step;\n } else {\n this.stepInterpolant = false;\n }\n this.gradientVersion = (this.gradientVersion + 1) % Number.MAX_SAFE_INTEGER;\n }\n }\n\n gradientExpression() {\n return this._transitionablePaint._values['line-gradient'].value.expression;\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n super.recalculate(parameters, availableImages);\n (this.paint._values as any)['line-floorwidth'] =\n lineFloorwidthProperty.possiblyEvaluate(this._transitioningPaint._values['line-width'].value, parameters);\n }\n\n createBucket(parameters: BucketParameters) {\n return new LineBucket(parameters);\n }\n\n queryRadius(bucket: Bucket): number {\n const lineBucket: LineBucket = (bucket as any);\n const width = getLineWidth(\n getMaximumPaintValue('line-width', this, lineBucket),\n getMaximumPaintValue('line-gap-width', this, lineBucket));\n const offset = getMaximumPaintValue('line-offset', this, lineBucket);\n return width / 2 + Math.abs(offset) + translateDistance(this.paint.get('line-translate'));\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number\n ): boolean {\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('line-translate'),\n this.paint.get('line-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n const halfWidth = pixelsToTileUnits / 2 * getLineWidth(\n this.paint.get('line-width').evaluate(feature, featureState),\n this.paint.get('line-gap-width').evaluate(feature, featureState));\n const lineOffset = this.paint.get('line-offset').evaluate(feature, featureState);\n if (lineOffset) {\n geometry = offsetLine(geometry, lineOffset * pixelsToTileUnits);\n }\n\n return polygonIntersectsBufferedMultiLine(translatedPolygon, geometry, halfWidth);\n }\n\n isTileClipped() {\n return true;\n }\n}\n\nfunction getLineWidth(lineWidth, lineGapWidth) {\n if (lineGapWidth > 0) {\n return lineGapWidth + 2 * lineWidth;\n } else {\n return lineWidth;\n }\n}\n","import {createLayout} from '../../util/struct_array';\n\nexport const symbolLayoutAttributes = createLayout([\n {name: 'a_pos_offset', components: 4, type: 'Int16'},\n {name: 'a_data', components: 4, type: 'Uint16'},\n {name: 'a_pixeloffset', components: 4, type: 'Int16'}\n], 4);\n\nexport const dynamicLayoutAttributes = createLayout([\n {name: 'a_projected_pos', components: 3, type: 'Float32'}\n], 4);\n\nexport const placementOpacityAttributes = createLayout([\n {name: 'a_fade_opacity', components: 1, type: 'Uint32'}\n], 4);\n\nexport const collisionVertexAttributes = createLayout([\n {name: 'a_placed', components: 2, type: 'Uint8'},\n {name: 'a_shift', components: 2, type: 'Float32'}\n]);\n\nexport const collisionBox = createLayout([\n // the box is centered around the anchor point\n {type: 'Int16', name: 'anchorPointX'},\n {type: 'Int16', name: 'anchorPointY'},\n\n // distances to the edges from the anchor\n {type: 'Int16', name: 'x1'},\n {type: 'Int16', name: 'y1'},\n {type: 'Int16', name: 'x2'},\n {type: 'Int16', name: 'y2'},\n\n // the index of the feature in the original vectortile\n {type: 'Uint32', name: 'featureIndex'},\n // the source layer the feature appears in\n {type: 'Uint16', name: 'sourceLayerIndex'},\n // the bucket the feature appears in\n {type: 'Uint16', name: 'bucketIndex'},\n]);\n\nexport const collisionBoxLayout = createLayout([ // used to render collision boxes for debugging purposes\n {name: 'a_pos', components: 2, type: 'Int16'},\n {name: 'a_anchor_pos', components: 2, type: 'Int16'},\n {name: 'a_extrude', components: 2, type: 'Int16'}\n], 4);\n\nexport const collisionCircleLayout = createLayout([ // used to render collision circles for debugging purposes\n {name: 'a_pos', components: 2, type: 'Float32'},\n {name: 'a_radius', components: 1, type: 'Float32'},\n {name: 'a_flags', components: 2, type: 'Int16'}\n], 4);\n\nexport const quadTriangle = createLayout([\n {name: 'triangle', components: 3, type: 'Uint16'},\n]);\n\nexport const placement = createLayout([\n {type: 'Int16', name: 'anchorX'},\n {type: 'Int16', name: 'anchorY'},\n {type: 'Uint16', name: 'glyphStartIndex'},\n {type: 'Uint16', name: 'numGlyphs'},\n {type: 'Uint32', name: 'vertexStartIndex'},\n {type: 'Uint32', name: 'lineStartIndex'},\n {type: 'Uint32', name: 'lineLength'},\n {type: 'Uint16', name: 'segment'},\n {type: 'Uint16', name: 'lowerSize'},\n {type: 'Uint16', name: 'upperSize'},\n {type: 'Float32', name: 'lineOffsetX'},\n {type: 'Float32', name: 'lineOffsetY'},\n {type: 'Uint8', name: 'writingMode'},\n {type: 'Uint8', name: 'placedOrientation'},\n {type: 'Uint8', name: 'hidden'},\n {type: 'Uint32', name: 'crossTileID'},\n {type: 'Int16', name: 'associatedIconIndex'}\n]);\n\nexport const symbolInstance = createLayout([\n {type: 'Int16', name: 'anchorX'},\n {type: 'Int16', name: 'anchorY'},\n {type: 'Int16', name: 'rightJustifiedTextSymbolIndex'},\n {type: 'Int16', name: 'centerJustifiedTextSymbolIndex'},\n {type: 'Int16', name: 'leftJustifiedTextSymbolIndex'},\n {type: 'Int16', name: 'verticalPlacedTextSymbolIndex'},\n {type: 'Int16', name: 'placedIconSymbolIndex'},\n {type: 'Int16', name: 'verticalPlacedIconSymbolIndex'},\n {type: 'Uint16', name: 'key'},\n {type: 'Uint16', name: 'textBoxStartIndex'},\n {type: 'Uint16', name: 'textBoxEndIndex'},\n {type: 'Uint16', name: 'verticalTextBoxStartIndex'},\n {type: 'Uint16', name: 'verticalTextBoxEndIndex'},\n {type: 'Uint16', name: 'iconBoxStartIndex'},\n {type: 'Uint16', name: 'iconBoxEndIndex'},\n {type: 'Uint16', name: 'verticalIconBoxStartIndex'},\n {type: 'Uint16', name: 'verticalIconBoxEndIndex'},\n {type: 'Uint16', name: 'featureIndex'},\n {type: 'Uint16', name: 'numHorizontalGlyphVertices'},\n {type: 'Uint16', name: 'numVerticalGlyphVertices'},\n {type: 'Uint16', name: 'numIconVertices'},\n {type: 'Uint16', name: 'numVerticalIconVertices'},\n {type: 'Uint16', name: 'useRuntimeCollisionCircles'},\n {type: 'Uint32', name: 'crossTileID'},\n {type: 'Float32', name: 'textBoxScale'},\n {type: 'Float32', name: 'collisionCircleDiameter'},\n {type: 'Uint16', name: 'textAnchorOffsetStartIndex'},\n {type: 'Uint16', name: 'textAnchorOffsetEndIndex'}\n]);\n\nexport const glyphOffset = createLayout([\n {type: 'Float32', name: 'offsetX'}\n]);\n\nexport const lineVertex = createLayout([\n {type: 'Int16', name: 'x'},\n {type: 'Int16', name: 'y'},\n {type: 'Int16', name: 'tileUnitDistanceFromAnchor'}\n]);\n\nexport const textAnchorOffset = createLayout([\n {type: 'Uint16', name: 'textAnchor'},\n {type: 'Float32', components: 2, name: 'textOffset'}\n]);\n","import {plugin as rtlTextPlugin} from '../source/rtl_text_plugin';\n\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\nimport type {Feature} from '@maplibre/maplibre-gl-style-spec';\nimport {Formatted} from '@maplibre/maplibre-gl-style-spec';\n\nfunction transformTextInternal(text: string, layer: SymbolStyleLayer, feature: Feature) {\n const transform = layer.layout.get('text-transform').evaluate(feature, {});\n if (transform === 'uppercase') {\n text = text.toLocaleUpperCase();\n } else if (transform === 'lowercase') {\n text = text.toLocaleLowerCase();\n }\n\n if (rtlTextPlugin.applyArabicShaping) {\n text = rtlTextPlugin.applyArabicShaping(text);\n }\n\n return text;\n}\n\nexport function transformText(text: Formatted, layer: SymbolStyleLayer, feature: Feature): Formatted {\n text.sections.forEach(section => {\n section.text = transformTextInternal(section.text, layer, feature);\n });\n return text;\n}\n","import {charHasRotatedVerticalOrientation} from './script_detection';\n\nexport const verticalizedCharacterMap = {\n '!': '︕',\n '#': '#',\n '$': '$',\n '%': '%',\n '&': '&',\n '(': '︵',\n ')': '︶',\n '*': '*',\n '+': '+',\n ',': '︐',\n '-': '︲',\n '.': '・',\n '/': '/',\n ':': '︓',\n ';': '︔',\n '<': '︿',\n '=': '=',\n '>': '﹀',\n '?': '︖',\n '@': '@',\n '[': '﹇',\n '\\\\': '\',\n ']': '﹈',\n '^': '^',\n '_': '︳',\n '`': '`',\n '{': '︷',\n '|': '―',\n '}': '︸',\n '~': '~',\n '¢': '¢',\n '£': '£',\n '¥': '¥',\n '¦': '¦',\n '¬': '¬',\n '¯': ' ̄',\n '–': '︲',\n '—': '︱',\n '‘': '﹃',\n '’': '﹄',\n '“': '﹁',\n '”': '﹂',\n '…': '︙',\n '‧': '・',\n '₩': '₩',\n '、': '︑',\n '。': '︒',\n '〈': '︿',\n '〉': '﹀',\n '《': '︽',\n '》': '︾',\n '「': '﹁',\n '」': '﹂',\n '『': '﹃',\n '』': '﹄',\n '【': '︻',\n '】': '︼',\n '〔': '︹',\n '〕': '︺',\n '〖': '︗',\n '〗': '︘',\n '!': '︕',\n '(': '︵',\n ')': '︶',\n ',': '︐',\n '-': '︲',\n '.': '・',\n ':': '︓',\n ';': '︔',\n '<': '︿',\n '>': '﹀',\n '?': '︖',\n '[': '﹇',\n ']': '﹈',\n '_': '︳',\n '{': '︷',\n '|': '―',\n '}': '︸',\n '⦅': '︵',\n '⦆': '︶',\n '。': '︒',\n '「': '﹁',\n '」': '﹂'\n};\n\nexport function verticalizePunctuation(input: string) {\n let output = '';\n\n for (let i = 0; i < input.length; i++) {\n const nextCharCode = input.charCodeAt(i + 1) || null;\n const prevCharCode = input.charCodeAt(i - 1) || null;\n\n const canReplacePunctuation = (\n (!nextCharCode || !charHasRotatedVerticalOrientation(nextCharCode) || verticalizedCharacterMap[input[i + 1]]) &&\n (!prevCharCode || !charHasRotatedVerticalOrientation(prevCharCode) || verticalizedCharacterMap[input[i - 1]])\n );\n\n if (canReplacePunctuation && verticalizedCharacterMap[input[i]]) {\n output += verticalizedCharacterMap[input[i]];\n } else {\n output += input[i];\n }\n }\n\n return output;\n}\n\n","// ONE_EM constant used to go between \"em\" units used in style spec and \"points\" used internally for layout\n\nexport default 24;\n","'use strict';\n\nmodule.exports = Pbf;\n\nvar ieee754 = require('ieee754');\n\nfunction Pbf(buf) {\n this.buf = ArrayBuffer.isView && ArrayBuffer.isView(buf) ? buf : new Uint8Array(buf || 0);\n this.pos = 0;\n this.type = 0;\n this.length = this.buf.length;\n}\n\nPbf.Varint = 0; // varint: int32, int64, uint32, uint64, sint32, sint64, bool, enum\nPbf.Fixed64 = 1; // 64-bit: double, fixed64, sfixed64\nPbf.Bytes = 2; // length-delimited: string, bytes, embedded messages, packed repeated fields\nPbf.Fixed32 = 5; // 32-bit: float, fixed32, sfixed32\n\nvar SHIFT_LEFT_32 = (1 << 16) * (1 << 16),\n SHIFT_RIGHT_32 = 1 / SHIFT_LEFT_32;\n\n// Threshold chosen based on both benchmarking and knowledge about browser string\n// data structures (which currently switch structure types at 12 bytes or more)\nvar TEXT_DECODER_MIN_LENGTH = 12;\nvar utf8TextDecoder = typeof TextDecoder === 'undefined' ? null : new TextDecoder('utf8');\n\nPbf.prototype = {\n\n destroy: function() {\n this.buf = null;\n },\n\n // === READING =================================================================\n\n readFields: function(readField, result, end) {\n end = end || this.length;\n\n while (this.pos < end) {\n var val = this.readVarint(),\n tag = val >> 3,\n startPos = this.pos;\n\n this.type = val & 0x7;\n readField(tag, result, this);\n\n if (this.pos === startPos) this.skip(val);\n }\n return result;\n },\n\n readMessage: function(readField, result) {\n return this.readFields(readField, result, this.readVarint() + this.pos);\n },\n\n readFixed32: function() {\n var val = readUInt32(this.buf, this.pos);\n this.pos += 4;\n return val;\n },\n\n readSFixed32: function() {\n var val = readInt32(this.buf, this.pos);\n this.pos += 4;\n return val;\n },\n\n // 64-bit int handling is based on github.com/dpw/node-buffer-more-ints (MIT-licensed)\n\n readFixed64: function() {\n var val = readUInt32(this.buf, this.pos) + readUInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;\n this.pos += 8;\n return val;\n },\n\n readSFixed64: function() {\n var val = readUInt32(this.buf, this.pos) + readInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;\n this.pos += 8;\n return val;\n },\n\n readFloat: function() {\n var val = ieee754.read(this.buf, this.pos, true, 23, 4);\n this.pos += 4;\n return val;\n },\n\n readDouble: function() {\n var val = ieee754.read(this.buf, this.pos, true, 52, 8);\n this.pos += 8;\n return val;\n },\n\n readVarint: function(isSigned) {\n var buf = this.buf,\n val, b;\n\n b = buf[this.pos++]; val = b & 0x7f; if (b < 0x80) return val;\n b = buf[this.pos++]; val |= (b & 0x7f) << 7; if (b < 0x80) return val;\n b = buf[this.pos++]; val |= (b & 0x7f) << 14; if (b < 0x80) return val;\n b = buf[this.pos++]; val |= (b & 0x7f) << 21; if (b < 0x80) return val;\n b = buf[this.pos]; val |= (b & 0x0f) << 28;\n\n return readVarintRemainder(val, isSigned, this);\n },\n\n readVarint64: function() { // for compatibility with v2.0.1\n return this.readVarint(true);\n },\n\n readSVarint: function() {\n var num = this.readVarint();\n return num % 2 === 1 ? (num + 1) / -2 : num / 2; // zigzag encoding\n },\n\n readBoolean: function() {\n return Boolean(this.readVarint());\n },\n\n readString: function() {\n var end = this.readVarint() + this.pos;\n var pos = this.pos;\n this.pos = end;\n\n if (end - pos >= TEXT_DECODER_MIN_LENGTH && utf8TextDecoder) {\n // longer strings are fast with the built-in browser TextDecoder API\n return readUtf8TextDecoder(this.buf, pos, end);\n }\n // short strings are fast with our custom implementation\n return readUtf8(this.buf, pos, end);\n },\n\n readBytes: function() {\n var end = this.readVarint() + this.pos,\n buffer = this.buf.subarray(this.pos, end);\n this.pos = end;\n return buffer;\n },\n\n // verbose for performance reasons; doesn't affect gzipped size\n\n readPackedVarint: function(arr, isSigned) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readVarint(isSigned));\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readVarint(isSigned));\n return arr;\n },\n readPackedSVarint: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readSVarint());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readSVarint());\n return arr;\n },\n readPackedBoolean: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readBoolean());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readBoolean());\n return arr;\n },\n readPackedFloat: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readFloat());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readFloat());\n return arr;\n },\n readPackedDouble: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readDouble());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readDouble());\n return arr;\n },\n readPackedFixed32: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readFixed32());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readFixed32());\n return arr;\n },\n readPackedSFixed32: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed32());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readSFixed32());\n return arr;\n },\n readPackedFixed64: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readFixed64());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readFixed64());\n return arr;\n },\n readPackedSFixed64: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed64());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readSFixed64());\n return arr;\n },\n\n skip: function(val) {\n var type = val & 0x7;\n if (type === Pbf.Varint) while (this.buf[this.pos++] > 0x7f) {}\n else if (type === Pbf.Bytes) this.pos = this.readVarint() + this.pos;\n else if (type === Pbf.Fixed32) this.pos += 4;\n else if (type === Pbf.Fixed64) this.pos += 8;\n else throw new Error('Unimplemented type: ' + type);\n },\n\n // === WRITING =================================================================\n\n writeTag: function(tag, type) {\n this.writeVarint((tag << 3) | type);\n },\n\n realloc: function(min) {\n var length = this.length || 16;\n\n while (length < this.pos + min) length *= 2;\n\n if (length !== this.length) {\n var buf = new Uint8Array(length);\n buf.set(this.buf);\n this.buf = buf;\n this.length = length;\n }\n },\n\n finish: function() {\n this.length = this.pos;\n this.pos = 0;\n return this.buf.subarray(0, this.length);\n },\n\n writeFixed32: function(val) {\n this.realloc(4);\n writeInt32(this.buf, val, this.pos);\n this.pos += 4;\n },\n\n writeSFixed32: function(val) {\n this.realloc(4);\n writeInt32(this.buf, val, this.pos);\n this.pos += 4;\n },\n\n writeFixed64: function(val) {\n this.realloc(8);\n writeInt32(this.buf, val & -1, this.pos);\n writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4);\n this.pos += 8;\n },\n\n writeSFixed64: function(val) {\n this.realloc(8);\n writeInt32(this.buf, val & -1, this.pos);\n writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4);\n this.pos += 8;\n },\n\n writeVarint: function(val) {\n val = +val || 0;\n\n if (val > 0xfffffff || val < 0) {\n writeBigVarint(val, this);\n return;\n }\n\n this.realloc(4);\n\n this.buf[this.pos++] = val & 0x7f | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return;\n this.buf[this.pos++] = ((val >>>= 7) & 0x7f) | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return;\n this.buf[this.pos++] = ((val >>>= 7) & 0x7f) | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return;\n this.buf[this.pos++] = (val >>> 7) & 0x7f;\n },\n\n writeSVarint: function(val) {\n this.writeVarint(val < 0 ? -val * 2 - 1 : val * 2);\n },\n\n writeBoolean: function(val) {\n this.writeVarint(Boolean(val));\n },\n\n writeString: function(str) {\n str = String(str);\n this.realloc(str.length * 4);\n\n this.pos++; // reserve 1 byte for short string length\n\n var startPos = this.pos;\n // write the string directly to the buffer and see how much was written\n this.pos = writeUtf8(this.buf, str, this.pos);\n var len = this.pos - startPos;\n\n if (len >= 0x80) makeRoomForExtraLength(startPos, len, this);\n\n // finally, write the message length in the reserved place and restore the position\n this.pos = startPos - 1;\n this.writeVarint(len);\n this.pos += len;\n },\n\n writeFloat: function(val) {\n this.realloc(4);\n ieee754.write(this.buf, val, this.pos, true, 23, 4);\n this.pos += 4;\n },\n\n writeDouble: function(val) {\n this.realloc(8);\n ieee754.write(this.buf, val, this.pos, true, 52, 8);\n this.pos += 8;\n },\n\n writeBytes: function(buffer) {\n var len = buffer.length;\n this.writeVarint(len);\n this.realloc(len);\n for (var i = 0; i < len; i++) this.buf[this.pos++] = buffer[i];\n },\n\n writeRawMessage: function(fn, obj) {\n this.pos++; // reserve 1 byte for short message length\n\n // write the message directly to the buffer and see how much was written\n var startPos = this.pos;\n fn(obj, this);\n var len = this.pos - startPos;\n\n if (len >= 0x80) makeRoomForExtraLength(startPos, len, this);\n\n // finally, write the message length in the reserved place and restore the position\n this.pos = startPos - 1;\n this.writeVarint(len);\n this.pos += len;\n },\n\n writeMessage: function(tag, fn, obj) {\n this.writeTag(tag, Pbf.Bytes);\n this.writeRawMessage(fn, obj);\n },\n\n writePackedVarint: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedVarint, arr); },\n writePackedSVarint: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSVarint, arr); },\n writePackedBoolean: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedBoolean, arr); },\n writePackedFloat: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFloat, arr); },\n writePackedDouble: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedDouble, arr); },\n writePackedFixed32: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFixed32, arr); },\n writePackedSFixed32: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSFixed32, arr); },\n writePackedFixed64: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFixed64, arr); },\n writePackedSFixed64: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSFixed64, arr); },\n\n writeBytesField: function(tag, buffer) {\n this.writeTag(tag, Pbf.Bytes);\n this.writeBytes(buffer);\n },\n writeFixed32Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed32);\n this.writeFixed32(val);\n },\n writeSFixed32Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed32);\n this.writeSFixed32(val);\n },\n writeFixed64Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed64);\n this.writeFixed64(val);\n },\n writeSFixed64Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed64);\n this.writeSFixed64(val);\n },\n writeVarintField: function(tag, val) {\n this.writeTag(tag, Pbf.Varint);\n this.writeVarint(val);\n },\n writeSVarintField: function(tag, val) {\n this.writeTag(tag, Pbf.Varint);\n this.writeSVarint(val);\n },\n writeStringField: function(tag, str) {\n this.writeTag(tag, Pbf.Bytes);\n this.writeString(str);\n },\n writeFloatField: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed32);\n this.writeFloat(val);\n },\n writeDoubleField: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed64);\n this.writeDouble(val);\n },\n writeBooleanField: function(tag, val) {\n this.writeVarintField(tag, Boolean(val));\n }\n};\n\nfunction readVarintRemainder(l, s, p) {\n var buf = p.buf,\n h, b;\n\n b = buf[p.pos++]; h = (b & 0x70) >> 4; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 3; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 10; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 17; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 24; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x01) << 31; if (b < 0x80) return toNum(l, h, s);\n\n throw new Error('Expected varint not more than 10 bytes');\n}\n\nfunction readPackedEnd(pbf) {\n return pbf.type === Pbf.Bytes ?\n pbf.readVarint() + pbf.pos : pbf.pos + 1;\n}\n\nfunction toNum(low, high, isSigned) {\n if (isSigned) {\n return high * 0x100000000 + (low >>> 0);\n }\n\n return ((high >>> 0) * 0x100000000) + (low >>> 0);\n}\n\nfunction writeBigVarint(val, pbf) {\n var low, high;\n\n if (val >= 0) {\n low = (val % 0x100000000) | 0;\n high = (val / 0x100000000) | 0;\n } else {\n low = ~(-val % 0x100000000);\n high = ~(-val / 0x100000000);\n\n if (low ^ 0xffffffff) {\n low = (low + 1) | 0;\n } else {\n low = 0;\n high = (high + 1) | 0;\n }\n }\n\n if (val >= 0x10000000000000000 || val < -0x10000000000000000) {\n throw new Error('Given varint doesn\\'t fit into 10 bytes');\n }\n\n pbf.realloc(10);\n\n writeBigVarintLow(low, high, pbf);\n writeBigVarintHigh(high, pbf);\n}\n\nfunction writeBigVarintLow(low, high, pbf) {\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos] = low & 0x7f;\n}\n\nfunction writeBigVarintHigh(high, pbf) {\n var lsb = (high & 0x07) << 4;\n\n pbf.buf[pbf.pos++] |= lsb | ((high >>>= 3) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f;\n}\n\nfunction makeRoomForExtraLength(startPos, len, pbf) {\n var extraLen =\n len <= 0x3fff ? 1 :\n len <= 0x1fffff ? 2 :\n len <= 0xfffffff ? 3 : Math.floor(Math.log(len) / (Math.LN2 * 7));\n\n // if 1 byte isn't enough for encoding message length, shift the data to the right\n pbf.realloc(extraLen);\n for (var i = pbf.pos - 1; i >= startPos; i--) pbf.buf[i + extraLen] = pbf.buf[i];\n}\n\nfunction writePackedVarint(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeVarint(arr[i]); }\nfunction writePackedSVarint(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSVarint(arr[i]); }\nfunction writePackedFloat(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFloat(arr[i]); }\nfunction writePackedDouble(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeDouble(arr[i]); }\nfunction writePackedBoolean(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeBoolean(arr[i]); }\nfunction writePackedFixed32(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFixed32(arr[i]); }\nfunction writePackedSFixed32(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSFixed32(arr[i]); }\nfunction writePackedFixed64(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFixed64(arr[i]); }\nfunction writePackedSFixed64(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSFixed64(arr[i]); }\n\n// Buffer code below from https://github.com/feross/buffer, MIT-licensed\n\nfunction readUInt32(buf, pos) {\n return ((buf[pos]) |\n (buf[pos + 1] << 8) |\n (buf[pos + 2] << 16)) +\n (buf[pos + 3] * 0x1000000);\n}\n\nfunction writeInt32(buf, val, pos) {\n buf[pos] = val;\n buf[pos + 1] = (val >>> 8);\n buf[pos + 2] = (val >>> 16);\n buf[pos + 3] = (val >>> 24);\n}\n\nfunction readInt32(buf, pos) {\n return ((buf[pos]) |\n (buf[pos + 1] << 8) |\n (buf[pos + 2] << 16)) +\n (buf[pos + 3] << 24);\n}\n\nfunction readUtf8(buf, pos, end) {\n var str = '';\n var i = pos;\n\n while (i < end) {\n var b0 = buf[i];\n var c = null; // codepoint\n var bytesPerSequence =\n b0 > 0xEF ? 4 :\n b0 > 0xDF ? 3 :\n b0 > 0xBF ? 2 : 1;\n\n if (i + bytesPerSequence > end) break;\n\n var b1, b2, b3;\n\n if (bytesPerSequence === 1) {\n if (b0 < 0x80) {\n c = b0;\n }\n } else if (bytesPerSequence === 2) {\n b1 = buf[i + 1];\n if ((b1 & 0xC0) === 0x80) {\n c = (b0 & 0x1F) << 0x6 | (b1 & 0x3F);\n if (c <= 0x7F) {\n c = null;\n }\n }\n } else if (bytesPerSequence === 3) {\n b1 = buf[i + 1];\n b2 = buf[i + 2];\n if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80) {\n c = (b0 & 0xF) << 0xC | (b1 & 0x3F) << 0x6 | (b2 & 0x3F);\n if (c <= 0x7FF || (c >= 0xD800 && c <= 0xDFFF)) {\n c = null;\n }\n }\n } else if (bytesPerSequence === 4) {\n b1 = buf[i + 1];\n b2 = buf[i + 2];\n b3 = buf[i + 3];\n if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80 && (b3 & 0xC0) === 0x80) {\n c = (b0 & 0xF) << 0x12 | (b1 & 0x3F) << 0xC | (b2 & 0x3F) << 0x6 | (b3 & 0x3F);\n if (c <= 0xFFFF || c >= 0x110000) {\n c = null;\n }\n }\n }\n\n if (c === null) {\n c = 0xFFFD;\n bytesPerSequence = 1;\n\n } else if (c > 0xFFFF) {\n c -= 0x10000;\n str += String.fromCharCode(c >>> 10 & 0x3FF | 0xD800);\n c = 0xDC00 | c & 0x3FF;\n }\n\n str += String.fromCharCode(c);\n i += bytesPerSequence;\n }\n\n return str;\n}\n\nfunction readUtf8TextDecoder(buf, pos, end) {\n return utf8TextDecoder.decode(buf.subarray(pos, end));\n}\n\nfunction writeUtf8(buf, str, pos) {\n for (var i = 0, c, lead; i < str.length; i++) {\n c = str.charCodeAt(i); // code point\n\n if (c > 0xD7FF && c < 0xE000) {\n if (lead) {\n if (c < 0xDC00) {\n buf[pos++] = 0xEF;\n buf[pos++] = 0xBF;\n buf[pos++] = 0xBD;\n lead = c;\n continue;\n } else {\n c = lead - 0xD800 << 10 | c - 0xDC00 | 0x10000;\n lead = null;\n }\n } else {\n if (c > 0xDBFF || (i + 1 === str.length)) {\n buf[pos++] = 0xEF;\n buf[pos++] = 0xBF;\n buf[pos++] = 0xBD;\n } else {\n lead = c;\n }\n continue;\n }\n } else if (lead) {\n buf[pos++] = 0xEF;\n buf[pos++] = 0xBF;\n buf[pos++] = 0xBD;\n lead = null;\n }\n\n if (c < 0x80) {\n buf[pos++] = c;\n } else {\n if (c < 0x800) {\n buf[pos++] = c >> 0x6 | 0xC0;\n } else {\n if (c < 0x10000) {\n buf[pos++] = c >> 0xC | 0xE0;\n } else {\n buf[pos++] = c >> 0x12 | 0xF0;\n buf[pos++] = c >> 0xC & 0x3F | 0x80;\n }\n buf[pos++] = c >> 0x6 & 0x3F | 0x80;\n }\n buf[pos++] = c & 0x3F | 0x80;\n }\n }\n return pos;\n}\n","/*! ieee754. BSD-3-Clause License. Feross Aboukhadijeh */\nexports.read = function (buffer, offset, isLE, mLen, nBytes) {\n var e, m\n var eLen = (nBytes * 8) - mLen - 1\n var eMax = (1 << eLen) - 1\n var eBias = eMax >> 1\n var nBits = -7\n var i = isLE ? (nBytes - 1) : 0\n var d = isLE ? -1 : 1\n var s = buffer[offset + i]\n\n i += d\n\n e = s & ((1 << (-nBits)) - 1)\n s >>= (-nBits)\n nBits += eLen\n for (; nBits > 0; e = (e * 256) + buffer[offset + i], i += d, nBits -= 8) {}\n\n m = e & ((1 << (-nBits)) - 1)\n e >>= (-nBits)\n nBits += mLen\n for (; nBits > 0; m = (m * 256) + buffer[offset + i], i += d, nBits -= 8) {}\n\n if (e === 0) {\n e = 1 - eBias\n } else if (e === eMax) {\n return m ? NaN : ((s ? -1 : 1) * Infinity)\n } else {\n m = m + Math.pow(2, mLen)\n e = e - eBias\n }\n return (s ? -1 : 1) * m * Math.pow(2, e - mLen)\n}\n\nexports.write = function (buffer, value, offset, isLE, mLen, nBytes) {\n var e, m, c\n var eLen = (nBytes * 8) - mLen - 1\n var eMax = (1 << eLen) - 1\n var eBias = eMax >> 1\n var rt = (mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0)\n var i = isLE ? 0 : (nBytes - 1)\n var d = isLE ? 1 : -1\n var s = value < 0 || (value === 0 && 1 / value < 0) ? 1 : 0\n\n value = Math.abs(value)\n\n if (isNaN(value) || value === Infinity) {\n m = isNaN(value) ? 1 : 0\n e = eMax\n } else {\n e = Math.floor(Math.log(value) / Math.LN2)\n if (value * (c = Math.pow(2, -e)) < 1) {\n e--\n c *= 2\n }\n if (e + eBias >= 1) {\n value += rt / c\n } else {\n value += rt * Math.pow(2, 1 - eBias)\n }\n if (value * c >= 2) {\n e++\n c /= 2\n }\n\n if (e + eBias >= eMax) {\n m = 0\n e = eMax\n } else if (e + eBias >= 1) {\n m = ((value * c) - 1) * Math.pow(2, mLen)\n e = e + eBias\n } else {\n m = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen)\n e = 0\n }\n }\n\n for (; mLen >= 8; buffer[offset + i] = m & 0xff, i += d, m /= 256, mLen -= 8) {}\n\n e = (e << mLen) | m\n eLen += mLen\n for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8) {}\n\n buffer[offset + i - d] |= s * 128\n}\n","import {AlphaImage} from '../util/image';\n\nimport Protobuf from 'pbf';\nconst border = 3;\n\nimport type {StyleGlyph} from './style_glyph';\n\nfunction readFontstacks(tag: number, glyphs: Array, pbf: Protobuf) {\n if (tag === 1) {\n pbf.readMessage(readFontstack, glyphs);\n }\n}\n\nfunction readFontstack(tag: number, glyphs: Array, pbf: Protobuf) {\n if (tag === 3) {\n const {id, bitmap, width, height, left, top, advance} = pbf.readMessage(readGlyph, {});\n glyphs.push({\n id,\n bitmap: new AlphaImage({\n width: width + 2 * border,\n height: height + 2 * border\n }, bitmap),\n metrics: {width, height, left, top, advance}\n });\n }\n}\n\nfunction readGlyph(tag: number, glyph: any, pbf: Protobuf) {\n if (tag === 1) glyph.id = pbf.readVarint();\n else if (tag === 2) glyph.bitmap = pbf.readBytes();\n else if (tag === 3) glyph.width = pbf.readVarint();\n else if (tag === 4) glyph.height = pbf.readVarint();\n else if (tag === 5) glyph.left = pbf.readSVarint();\n else if (tag === 6) glyph.top = pbf.readSVarint();\n else if (tag === 7) glyph.advance = pbf.readVarint();\n}\n\nexport function parseGlyphPbf(data: ArrayBuffer | Uint8Array): Array {\n return new Protobuf(data).readFields(readFontstacks, []);\n}\n\nexport const GLYPH_PBF_BORDER = border;\n","\nexport default function potpack(boxes) {\n\n // calculate total box area and maximum box width\n let area = 0;\n let maxWidth = 0;\n\n for (const box of boxes) {\n area += box.w * box.h;\n maxWidth = Math.max(maxWidth, box.w);\n }\n\n // sort the boxes for insertion by height, descending\n boxes.sort((a, b) => b.h - a.h);\n\n // aim for a squarish resulting container,\n // slightly adjusted for sub-100% space utilization\n const startWidth = Math.max(Math.ceil(Math.sqrt(area / 0.95)), maxWidth);\n\n // start with a single empty space, unbounded at the bottom\n const spaces = [{x: 0, y: 0, w: startWidth, h: Infinity}];\n\n let width = 0;\n let height = 0;\n\n for (const box of boxes) {\n // look through spaces backwards so that we check smaller spaces first\n for (let i = spaces.length - 1; i >= 0; i--) {\n const space = spaces[i];\n\n // look for empty spaces that can accommodate the current box\n if (box.w > space.w || box.h > space.h) continue;\n\n // found the space; add the box to its top-left corner\n // |-------|-------|\n // | box | |\n // |_______| |\n // | space |\n // |_______________|\n box.x = space.x;\n box.y = space.y;\n\n height = Math.max(height, box.y + box.h);\n width = Math.max(width, box.x + box.w);\n\n if (box.w === space.w && box.h === space.h) {\n // space matches the box exactly; remove it\n const last = spaces.pop();\n if (i < spaces.length) spaces[i] = last;\n\n } else if (box.h === space.h) {\n // space matches the box height; update it accordingly\n // |-------|---------------|\n // | box | updated space |\n // |_______|_______________|\n space.x += box.w;\n space.w -= box.w;\n\n } else if (box.w === space.w) {\n // space matches the box width; update it accordingly\n // |---------------|\n // | box |\n // |_______________|\n // | updated space |\n // |_______________|\n space.y += box.h;\n space.h -= box.h;\n\n } else {\n // otherwise the box splits the space into two spaces\n // |-------|-----------|\n // | box | new space |\n // |_______|___________|\n // | updated space |\n // |___________________|\n spaces.push({\n x: space.x + box.w,\n y: space.y,\n w: space.w - box.w,\n h: box.h\n });\n space.y += box.h;\n space.h -= box.h;\n }\n break;\n }\n }\n\n return {\n w: width, // container width\n h: height, // container height\n fill: (area / (width * height)) || 0 // space utilization\n };\n}\n","/* eslint-disable key-spacing */\nimport {RGBAImage} from '../util/image';\nimport {register} from '../util/web_worker_transfer';\nimport potpack from 'potpack';\n\nimport type {StyleImage} from '../style/style_image';\nimport type {ImageManager} from './image_manager';\nimport type {Texture} from './texture';\nimport type {Rect} from './glyph_atlas';\n\nconst IMAGE_PADDING: number = 1;\nexport {IMAGE_PADDING};\n\nexport class ImagePosition {\n paddedRect: Rect;\n pixelRatio: number;\n version: number;\n stretchY: Array<[number, number]>;\n stretchX: Array<[number, number]>;\n content: [number, number, number, number];\n\n constructor(paddedRect: Rect, {\n pixelRatio,\n version,\n stretchX,\n stretchY,\n content\n }: StyleImage) {\n this.paddedRect = paddedRect;\n this.pixelRatio = pixelRatio;\n this.stretchX = stretchX;\n this.stretchY = stretchY;\n this.content = content;\n this.version = version;\n }\n\n get tl(): [number, number] {\n return [\n this.paddedRect.x + IMAGE_PADDING,\n this.paddedRect.y + IMAGE_PADDING\n ];\n }\n\n get br(): [number, number] {\n return [\n this.paddedRect.x + this.paddedRect.w - IMAGE_PADDING,\n this.paddedRect.y + this.paddedRect.h - IMAGE_PADDING\n ];\n }\n\n get tlbr(): Array {\n return this.tl.concat(this.br);\n }\n\n get displaySize(): [number, number] {\n return [\n (this.paddedRect.w - IMAGE_PADDING * 2) / this.pixelRatio,\n (this.paddedRect.h - IMAGE_PADDING * 2) / this.pixelRatio\n ];\n }\n}\n\n/**\n * @internal\n * A class holding all the images\n */\nexport class ImageAtlas {\n image: RGBAImage;\n iconPositions: {[_: string]: ImagePosition};\n patternPositions: {[_: string]: ImagePosition};\n haveRenderCallbacks: Array;\n uploaded: boolean;\n\n constructor(icons: {[_: string]: StyleImage}, patterns: {[_: string]: StyleImage}) {\n const iconPositions = {}, patternPositions = {};\n this.haveRenderCallbacks = [];\n\n const bins = [];\n\n this.addImages(icons, iconPositions, bins);\n this.addImages(patterns, patternPositions, bins);\n\n const {w, h} = potpack(bins);\n const image = new RGBAImage({width: w || 1, height: h || 1});\n\n for (const id in icons) {\n const src = icons[id];\n const bin = iconPositions[id].paddedRect;\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x: bin.x + IMAGE_PADDING, y: bin.y + IMAGE_PADDING}, src.data);\n }\n\n for (const id in patterns) {\n const src = patterns[id];\n const bin = patternPositions[id].paddedRect;\n const x = bin.x + IMAGE_PADDING,\n y = bin.y + IMAGE_PADDING,\n w = src.data.width,\n h = src.data.height;\n\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x, y}, src.data);\n // Add 1 pixel wrapped padding on each side of the image.\n RGBAImage.copy(src.data, image, {x: 0, y: h - 1}, {x, y: y - 1}, {width: w, height: 1}); // T\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x, y: y + h}, {width: w, height: 1}); // B\n RGBAImage.copy(src.data, image, {x: w - 1, y: 0}, {x: x - 1, y}, {width: 1, height: h}); // L\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x: x + w, y}, {width: 1, height: h}); // R\n }\n\n this.image = image;\n this.iconPositions = iconPositions;\n this.patternPositions = patternPositions;\n }\n\n addImages(images: {[_: string]: StyleImage}, positions: {[_: string]: ImagePosition}, bins: Array) {\n for (const id in images) {\n const src = images[id];\n const bin = {\n x: 0,\n y: 0,\n w: src.data.width + 2 * IMAGE_PADDING,\n h: src.data.height + 2 * IMAGE_PADDING,\n };\n bins.push(bin);\n positions[id] = new ImagePosition(bin, src);\n\n if (src.hasRenderCallback) {\n this.haveRenderCallbacks.push(id);\n }\n }\n }\n\n patchUpdatedImages(imageManager: ImageManager, texture: Texture) {\n imageManager.dispatchRenderCallbacks(this.haveRenderCallbacks);\n for (const name in imageManager.updatedImages) {\n this.patchUpdatedImage(this.iconPositions[name], imageManager.getImage(name), texture);\n this.patchUpdatedImage(this.patternPositions[name], imageManager.getImage(name), texture);\n }\n }\n\n patchUpdatedImage(position: ImagePosition, image: StyleImage, texture: Texture) {\n if (!position || !image) return;\n\n if (position.version === image.version) return;\n\n position.version = image.version;\n const [x, y] = position.tl;\n texture.update(image.data, undefined, {x, y});\n }\n\n}\n\nregister('ImagePosition', ImagePosition);\nregister('ImageAtlas', ImageAtlas);\n","import {\n charHasUprightVerticalOrientation,\n charAllowsIdeographicBreaking,\n charInComplexShapingScript\n} from '../util/script_detection';\nimport {verticalizePunctuation} from '../util/verticalize_punctuation';\nimport {plugin as rtlTextPlugin} from '../source/rtl_text_plugin';\nimport ONE_EM from './one_em';\nimport {warnOnce} from '../util/util';\n\nimport type {StyleGlyph, GlyphMetrics} from '../style/style_glyph';\nimport {GLYPH_PBF_BORDER} from '../style/parse_glyph_pbf';\nimport type {ImagePosition} from '../render/image_atlas';\nimport {IMAGE_PADDING} from '../render/image_atlas';\nimport type {Rect, GlyphPosition} from '../render/glyph_atlas';\nimport {Formatted, FormattedSection} from '@maplibre/maplibre-gl-style-spec';\n\nenum WritingMode {\n none = 0,\n horizontal = 1,\n vertical = 2,\n horizontalOnly = 3\n}\n\nconst SHAPING_DEFAULT_OFFSET = -17;\nexport {shapeText, shapeIcon, fitIconToText, getAnchorAlignment, WritingMode, SHAPING_DEFAULT_OFFSET};\n\n// The position of a glyph relative to the text's anchor point.\nexport type PositionedGlyph = {\n glyph: number;\n imageName: string | null;\n x: number;\n y: number;\n vertical: boolean;\n scale: number;\n fontStack: string;\n sectionIndex: number;\n metrics: GlyphMetrics;\n rect: Rect | null;\n};\n\nexport type PositionedLine = {\n positionedGlyphs: Array;\n lineOffset: number;\n};\n\n// A collection of positioned glyphs and some metadata\nexport type Shaping = {\n positionedLines: Array;\n top: number;\n bottom: number;\n left: number;\n right: number;\n writingMode: WritingMode.horizontal | WritingMode.vertical;\n text: string;\n iconsInText: boolean;\n verticalizable: boolean;\n};\n\nfunction isEmpty(positionedLines: Array) {\n for (const line of positionedLines) {\n if (line.positionedGlyphs.length !== 0) {\n return false;\n }\n }\n return true;\n}\n\nexport type SymbolAnchor = 'center' | 'left' | 'right' | 'top' | 'bottom' | 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right';\nexport type TextJustify = 'left' | 'center' | 'right';\n\n// Max number of images in label is 6401 U+E000–U+F8FF that covers\n// Basic Multilingual Plane Unicode Private Use Area (PUA).\nconst PUAbegin = 0xE000;\nconst PUAend = 0xF8FF;\n\nclass SectionOptions {\n // Text options\n scale: number;\n fontStack: string;\n // Image options\n imageName: string | null;\n\n constructor() {\n this.scale = 1.0;\n this.fontStack = '';\n this.imageName = null;\n }\n\n static forText(scale: number | null, fontStack: string) {\n const textOptions = new SectionOptions();\n textOptions.scale = scale || 1;\n textOptions.fontStack = fontStack;\n return textOptions;\n }\n\n static forImage(imageName: string) {\n const imageOptions = new SectionOptions();\n imageOptions.imageName = imageName;\n return imageOptions;\n }\n\n}\n\nclass TaggedString {\n text: string;\n sectionIndex: Array; // maps each character in 'text' to its corresponding entry in 'sections'\n sections: Array;\n imageSectionID: number | null;\n\n constructor() {\n this.text = '';\n this.sectionIndex = [];\n this.sections = [];\n this.imageSectionID = null;\n }\n\n static fromFeature(text: Formatted, defaultFontStack: string) {\n const result = new TaggedString();\n for (let i = 0; i < text.sections.length; i++) {\n const section = text.sections[i];\n if (!section.image) {\n result.addTextSection(section, defaultFontStack);\n } else {\n result.addImageSection(section);\n }\n }\n return result;\n }\n\n length(): number {\n return this.text.length;\n }\n\n getSection(index: number): SectionOptions {\n return this.sections[this.sectionIndex[index]];\n }\n\n getSectionIndex(index: number): number {\n return this.sectionIndex[index];\n }\n\n getCharCode(index: number): number {\n return this.text.charCodeAt(index);\n }\n\n verticalizePunctuation() {\n this.text = verticalizePunctuation(this.text);\n }\n\n trim() {\n let beginningWhitespace = 0;\n for (let i = 0;\n i < this.text.length && whitespace[this.text.charCodeAt(i)];\n i++) {\n beginningWhitespace++;\n }\n let trailingWhitespace = this.text.length;\n for (let i = this.text.length - 1;\n i >= 0 && i >= beginningWhitespace && whitespace[this.text.charCodeAt(i)];\n i--) {\n trailingWhitespace--;\n }\n this.text = this.text.substring(beginningWhitespace, trailingWhitespace);\n this.sectionIndex = this.sectionIndex.slice(beginningWhitespace, trailingWhitespace);\n }\n\n substring(start: number, end: number): TaggedString {\n const substring = new TaggedString();\n substring.text = this.text.substring(start, end);\n substring.sectionIndex = this.sectionIndex.slice(start, end);\n substring.sections = this.sections;\n return substring;\n }\n\n toString(): string {\n return this.text;\n }\n\n getMaxScale() {\n return this.sectionIndex.reduce((max, index) => Math.max(max, this.sections[index].scale), 0);\n }\n\n addTextSection(section: FormattedSection, defaultFontStack: string) {\n this.text += section.text;\n this.sections.push(SectionOptions.forText(section.scale, section.fontStack || defaultFontStack));\n const index = this.sections.length - 1;\n for (let i = 0; i < section.text.length; ++i) {\n this.sectionIndex.push(index);\n }\n }\n\n addImageSection(section: FormattedSection) {\n const imageName = section.image ? section.image.name : '';\n if (imageName.length === 0) {\n warnOnce('Can\\'t add FormattedSection with an empty image.');\n return;\n }\n\n const nextImageSectionCharCode = this.getNextImageSectionCharCode();\n if (!nextImageSectionCharCode) {\n warnOnce(`Reached maximum number of images ${PUAend - PUAbegin + 2}`);\n return;\n }\n\n this.text += String.fromCharCode(nextImageSectionCharCode);\n this.sections.push(SectionOptions.forImage(imageName));\n this.sectionIndex.push(this.sections.length - 1);\n }\n\n getNextImageSectionCharCode(): number | null {\n if (!this.imageSectionID) {\n this.imageSectionID = PUAbegin;\n return this.imageSectionID;\n }\n\n if (this.imageSectionID >= PUAend) return null;\n return ++this.imageSectionID;\n }\n}\n\nfunction breakLines(input: TaggedString, lineBreakPoints: Array): Array {\n const lines = [];\n const text = input.text;\n let start = 0;\n for (const lineBreak of lineBreakPoints) {\n lines.push(input.substring(start, lineBreak));\n start = lineBreak;\n }\n\n if (start < text.length) {\n lines.push(input.substring(start, text.length));\n }\n return lines;\n}\n\nfunction shapeText(\n text: Formatted,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n glyphPositions: {\n [_: string]: {\n [_: number]: GlyphPosition;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n defaultFontStack: string,\n maxWidth: number,\n lineHeight: number,\n textAnchor: SymbolAnchor,\n textJustify: TextJustify,\n spacing: number,\n translate: [number, number],\n writingMode: WritingMode.horizontal | WritingMode.vertical,\n allowVerticalPlacement: boolean,\n symbolPlacement: string,\n layoutTextSize: number,\n layoutTextSizeThisZoom: number\n): Shaping | false {\n const logicalInput = TaggedString.fromFeature(text, defaultFontStack);\n\n if (writingMode === WritingMode.vertical) {\n logicalInput.verticalizePunctuation();\n }\n\n let lines: Array;\n\n const {processBidirectionalText, processStyledBidirectionalText} = rtlTextPlugin;\n if (processBidirectionalText && logicalInput.sections.length === 1) {\n // Bidi doesn't have to be style-aware\n lines = [];\n const untaggedLines =\n processBidirectionalText(logicalInput.toString(),\n determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize));\n for (const line of untaggedLines) {\n const taggedLine = new TaggedString();\n taggedLine.text = line;\n taggedLine.sections = logicalInput.sections;\n for (let i = 0; i < line.length; i++) {\n taggedLine.sectionIndex.push(0);\n }\n lines.push(taggedLine);\n }\n } else if (processStyledBidirectionalText) {\n // Need version of mapbox-gl-rtl-text with style support for combining RTL text\n // with formatting\n lines = [];\n const processedLines =\n processStyledBidirectionalText(logicalInput.text,\n logicalInput.sectionIndex,\n determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize));\n for (const line of processedLines) {\n const taggedLine = new TaggedString();\n taggedLine.text = line[0];\n taggedLine.sectionIndex = line[1];\n taggedLine.sections = logicalInput.sections;\n lines.push(taggedLine);\n }\n } else {\n lines = breakLines(logicalInput, determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize));\n }\n\n const positionedLines = [];\n const shaping = {\n positionedLines,\n text: logicalInput.toString(),\n top: translate[1],\n bottom: translate[1],\n left: translate[0],\n right: translate[0],\n writingMode,\n iconsInText: false,\n verticalizable: false\n };\n\n shapeLines(shaping, glyphMap, glyphPositions, imagePositions, lines, lineHeight, textAnchor, textJustify, writingMode, spacing, allowVerticalPlacement, layoutTextSizeThisZoom);\n if (isEmpty(positionedLines)) return false;\n\n return shaping;\n}\n\n// using computed properties due to https://github.com/facebook/flow/issues/380\n/* eslint no-useless-computed-key: 0 */\n\nconst whitespace: {\n [_: number]: boolean;\n} = {\n [0x09]: true, // tab\n [0x0a]: true, // newline\n [0x0b]: true, // vertical tab\n [0x0c]: true, // form feed\n [0x0d]: true, // carriage return\n [0x20]: true, // space\n};\n\nconst breakable: {\n [_: number]: boolean;\n} = {\n [0x0a]: true, // newline\n [0x20]: true, // space\n [0x26]: true, // ampersand\n [0x28]: true, // left parenthesis\n [0x29]: true, // right parenthesis\n [0x2b]: true, // plus sign\n [0x2d]: true, // hyphen-minus\n [0x2f]: true, // solidus\n [0xad]: true, // soft hyphen\n [0xb7]: true, // middle dot\n [0x200b]: true, // zero-width space\n [0x2010]: true, // hyphen\n [0x2013]: true, // en dash\n [0x2027]: true // interpunct\n // Many other characters may be reasonable breakpoints\n // Consider \"neutral orientation\" characters at scriptDetection.charHasNeutralVerticalOrientation\n // See https://github.com/mapbox/mapbox-gl-js/issues/3658\n};\n\nfunction getGlyphAdvance(\n codePoint: number,\n section: SectionOptions,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n spacing: number,\n layoutTextSize: number\n): number {\n if (!section.imageName) {\n const positions = glyphMap[section.fontStack];\n const glyph = positions && positions[codePoint];\n if (!glyph) return 0;\n return glyph.metrics.advance * section.scale + spacing;\n } else {\n const imagePosition = imagePositions[section.imageName];\n if (!imagePosition) return 0;\n return imagePosition.displaySize[0] * section.scale * ONE_EM / layoutTextSize + spacing;\n }\n}\n\nfunction determineAverageLineWidth(logicalInput: TaggedString,\n spacing: number,\n maxWidth: number,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n layoutTextSize: number) {\n let totalWidth = 0;\n\n for (let index = 0; index < logicalInput.length(); index++) {\n const section = logicalInput.getSection(index);\n totalWidth += getGlyphAdvance(logicalInput.getCharCode(index), section, glyphMap, imagePositions, spacing, layoutTextSize);\n }\n\n const lineCount = Math.max(1, Math.ceil(totalWidth / maxWidth));\n return totalWidth / lineCount;\n}\n\nfunction calculateBadness(lineWidth: number,\n targetWidth: number,\n penalty: number,\n isLastBreak: boolean) {\n const raggedness = Math.pow(lineWidth - targetWidth, 2);\n if (isLastBreak) {\n // Favor finals lines shorter than average over longer than average\n if (lineWidth < targetWidth) {\n return raggedness / 2;\n } else {\n return raggedness * 2;\n }\n }\n\n return raggedness + Math.abs(penalty) * penalty;\n}\n\nfunction calculatePenalty(codePoint: number, nextCodePoint: number, penalizableIdeographicBreak: boolean) {\n let penalty = 0;\n // Force break on newline\n if (codePoint === 0x0a) {\n penalty -= 10000;\n }\n // Penalize breaks between characters that allow ideographic breaking because\n // they are less preferable than breaks at spaces (or zero width spaces).\n if (penalizableIdeographicBreak) {\n penalty += 150;\n }\n\n // Penalize open parenthesis at end of line\n if (codePoint === 0x28 || codePoint === 0xff08) {\n penalty += 50;\n }\n\n // Penalize close parenthesis at beginning of line\n if (nextCodePoint === 0x29 || nextCodePoint === 0xff09) {\n penalty += 50;\n }\n return penalty;\n}\n\ntype Break = {\n index: number;\n x: number;\n priorBreak: Break;\n badness: number;\n};\n\nfunction evaluateBreak(\n breakIndex: number,\n breakX: number,\n targetWidth: number,\n potentialBreaks: Array,\n penalty: number,\n isLastBreak: boolean\n): Break {\n // We could skip evaluating breaks where the line length (breakX - priorBreak.x) > maxWidth\n // ...but in fact we allow lines longer than maxWidth (if there's no break points)\n // ...and when targetWidth and maxWidth are close, strictly enforcing maxWidth can give\n // more lopsided results.\n\n let bestPriorBreak: Break = null;\n let bestBreakBadness = calculateBadness(breakX, targetWidth, penalty, isLastBreak);\n\n for (const potentialBreak of potentialBreaks) {\n const lineWidth = breakX - potentialBreak.x;\n const breakBadness =\n calculateBadness(lineWidth, targetWidth, penalty, isLastBreak) + potentialBreak.badness;\n if (breakBadness <= bestBreakBadness) {\n bestPriorBreak = potentialBreak;\n bestBreakBadness = breakBadness;\n }\n }\n\n return {\n index: breakIndex,\n x: breakX,\n priorBreak: bestPriorBreak,\n badness: bestBreakBadness\n };\n}\n\nfunction leastBadBreaks(lastLineBreak?: Break | null): Array {\n if (!lastLineBreak) {\n return [];\n }\n return leastBadBreaks(lastLineBreak.priorBreak).concat(lastLineBreak.index);\n}\n\nfunction determineLineBreaks(\n logicalInput: TaggedString,\n spacing: number,\n maxWidth: number,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n symbolPlacement: string,\n layoutTextSize: number\n): Array {\n if (symbolPlacement !== 'point')\n return [];\n\n if (!logicalInput)\n return [];\n\n const potentialLineBreaks = [];\n const targetWidth = determineAverageLineWidth(logicalInput, spacing, maxWidth, glyphMap, imagePositions, layoutTextSize);\n\n const hasServerSuggestedBreakpoints = logicalInput.text.indexOf('\\u200b') >= 0;\n\n let currentX = 0;\n\n for (let i = 0; i < logicalInput.length(); i++) {\n const section = logicalInput.getSection(i);\n const codePoint = logicalInput.getCharCode(i);\n if (!whitespace[codePoint]) currentX += getGlyphAdvance(codePoint, section, glyphMap, imagePositions, spacing, layoutTextSize);\n\n // Ideographic characters, spaces, and word-breaking punctuation that often appear without\n // surrounding spaces.\n if ((i < logicalInput.length() - 1)) {\n const ideographicBreak = charAllowsIdeographicBreaking(codePoint);\n if (breakable[codePoint] || ideographicBreak || section.imageName) {\n\n potentialLineBreaks.push(\n evaluateBreak(\n i + 1,\n currentX,\n targetWidth,\n potentialLineBreaks,\n calculatePenalty(codePoint, logicalInput.getCharCode(i + 1), ideographicBreak && hasServerSuggestedBreakpoints),\n false));\n }\n }\n }\n\n return leastBadBreaks(\n evaluateBreak(\n logicalInput.length(),\n currentX,\n targetWidth,\n potentialLineBreaks,\n 0,\n true));\n}\n\nfunction getAnchorAlignment(anchor: SymbolAnchor) {\n let horizontalAlign = 0.5, verticalAlign = 0.5;\n\n switch (anchor) {\n case 'right':\n case 'top-right':\n case 'bottom-right':\n horizontalAlign = 1;\n break;\n case 'left':\n case 'top-left':\n case 'bottom-left':\n horizontalAlign = 0;\n break;\n }\n\n switch (anchor) {\n case 'bottom':\n case 'bottom-right':\n case 'bottom-left':\n verticalAlign = 1;\n break;\n case 'top':\n case 'top-right':\n case 'top-left':\n verticalAlign = 0;\n break;\n }\n\n return {horizontalAlign, verticalAlign};\n}\n\nfunction shapeLines(shaping: Shaping,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n glyphPositions: {\n [_: string]: {\n [_: number]: GlyphPosition;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n lines: Array,\n lineHeight: number,\n textAnchor: SymbolAnchor,\n textJustify: TextJustify,\n writingMode: WritingMode.horizontal | WritingMode.vertical,\n spacing: number,\n allowVerticalPlacement: boolean,\n layoutTextSizeThisZoom: number) {\n\n let x = 0;\n let y = SHAPING_DEFAULT_OFFSET;\n\n let maxLineLength = 0;\n let maxLineHeight = 0;\n\n const justify =\n textJustify === 'right' ? 1 :\n textJustify === 'left' ? 0 : 0.5;\n\n let lineIndex = 0;\n for (const line of lines) {\n line.trim();\n\n const lineMaxScale = line.getMaxScale();\n const maxLineOffset = (lineMaxScale - 1) * ONE_EM;\n const positionedLine = {positionedGlyphs: [], lineOffset: 0};\n shaping.positionedLines[lineIndex] = positionedLine;\n const positionedGlyphs = positionedLine.positionedGlyphs;\n let lineOffset = 0.0;\n\n if (!line.length()) {\n y += lineHeight; // Still need a line feed after empty line\n ++lineIndex;\n continue;\n }\n\n for (let i = 0; i < line.length(); i++) {\n const section = line.getSection(i);\n const sectionIndex = line.getSectionIndex(i);\n const codePoint = line.getCharCode(i);\n let baselineOffset = 0.0;\n let metrics = null;\n let rect = null;\n let imageName = null;\n let verticalAdvance = ONE_EM;\n const vertical = !(writingMode === WritingMode.horizontal ||\n // Don't verticalize glyphs that have no upright orientation if vertical placement is disabled.\n (!allowVerticalPlacement && !charHasUprightVerticalOrientation(codePoint)) ||\n // If vertical placement is enabled, don't verticalize glyphs that\n // are from complex text layout script, or whitespaces.\n (allowVerticalPlacement && (whitespace[codePoint] || charInComplexShapingScript(codePoint))));\n\n if (!section.imageName) {\n const positions = glyphPositions[section.fontStack];\n const glyphPosition = positions && positions[codePoint];\n if (glyphPosition && glyphPosition.rect) {\n rect = glyphPosition.rect;\n metrics = glyphPosition.metrics;\n } else {\n const glyphs = glyphMap[section.fontStack];\n const glyph = glyphs && glyphs[codePoint];\n if (!glyph) continue;\n metrics = glyph.metrics;\n }\n\n // We don't know the baseline, but since we're laying out\n // at 24 points, we can calculate how much it will move when\n // we scale up or down.\n baselineOffset = (lineMaxScale - section.scale) * ONE_EM;\n } else {\n const imagePosition = imagePositions[section.imageName];\n if (!imagePosition) continue;\n imageName = section.imageName;\n shaping.iconsInText = shaping.iconsInText || true;\n rect = imagePosition.paddedRect;\n const size = imagePosition.displaySize;\n // If needed, allow to set scale factor for an image using\n // alias \"image-scale\" that could be alias for \"font-scale\"\n // when FormattedSection is an image section.\n section.scale = section.scale * ONE_EM / layoutTextSizeThisZoom;\n\n metrics = {width: size[0],\n height: size[1],\n left: IMAGE_PADDING,\n top: -GLYPH_PBF_BORDER,\n advance: vertical ? size[1] : size[0]};\n\n // Difference between one EM and an image size.\n // Aligns bottom of an image to a baseline level.\n const imageOffset = ONE_EM - size[1] * section.scale;\n baselineOffset = maxLineOffset + imageOffset;\n verticalAdvance = metrics.advance;\n\n // Difference between height of an image and one EM at max line scale.\n // Pushes current line down if an image size is over 1 EM at max line scale.\n const offset = vertical ? size[0] * section.scale - ONE_EM * lineMaxScale :\n size[1] * section.scale - ONE_EM * lineMaxScale;\n if (offset > 0 && offset > lineOffset) {\n lineOffset = offset;\n }\n }\n\n if (!vertical) {\n positionedGlyphs.push({glyph: codePoint, imageName, x, y: y + baselineOffset, vertical, scale: section.scale, fontStack: section.fontStack, sectionIndex, metrics, rect});\n x += metrics.advance * section.scale + spacing;\n } else {\n shaping.verticalizable = true;\n positionedGlyphs.push({glyph: codePoint, imageName, x, y: y + baselineOffset, vertical, scale: section.scale, fontStack: section.fontStack, sectionIndex, metrics, rect});\n x += verticalAdvance * section.scale + spacing;\n }\n }\n\n // Only justify if we placed at least one glyph\n if (positionedGlyphs.length !== 0) {\n const lineLength = x - spacing;\n maxLineLength = Math.max(lineLength, maxLineLength);\n justifyLine(positionedGlyphs, 0, positionedGlyphs.length - 1, justify, lineOffset);\n }\n\n x = 0;\n const currentLineHeight = lineHeight * lineMaxScale + lineOffset;\n positionedLine.lineOffset = Math.max(lineOffset, maxLineOffset);\n y += currentLineHeight;\n maxLineHeight = Math.max(currentLineHeight, maxLineHeight);\n ++lineIndex;\n }\n\n // Calculate the bounding box and justify / align text block.\n const height = y - SHAPING_DEFAULT_OFFSET;\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(textAnchor);\n align(shaping.positionedLines, justify, horizontalAlign, verticalAlign, maxLineLength, maxLineHeight, lineHeight, height, lines.length);\n\n shaping.top += -verticalAlign * height;\n shaping.bottom = shaping.top + height;\n shaping.left += -horizontalAlign * maxLineLength;\n shaping.right = shaping.left + maxLineLength;\n}\n\n// justify right = 1, left = 0, center = 0.5\nfunction justifyLine(positionedGlyphs: Array,\n start: number,\n end: number,\n justify: 1 | 0 | 0.5,\n lineOffset: number) {\n if (!justify && !lineOffset)\n return;\n\n const lastPositionedGlyph = positionedGlyphs[end];\n const lastAdvance = lastPositionedGlyph.metrics.advance * lastPositionedGlyph.scale;\n const lineIndent = (positionedGlyphs[end].x + lastAdvance) * justify;\n\n for (let j = start; j <= end; j++) {\n positionedGlyphs[j].x -= lineIndent;\n positionedGlyphs[j].y += lineOffset;\n }\n}\n\nfunction align(positionedLines: Array,\n justify: number,\n horizontalAlign: number,\n verticalAlign: number,\n maxLineLength: number,\n maxLineHeight: number,\n lineHeight: number,\n blockHeight: number,\n lineCount: number) {\n const shiftX = (justify - horizontalAlign) * maxLineLength;\n let shiftY = 0;\n\n if (maxLineHeight !== lineHeight) {\n shiftY = -blockHeight * verticalAlign - SHAPING_DEFAULT_OFFSET;\n } else {\n shiftY = (-verticalAlign * lineCount + 0.5) * lineHeight;\n }\n\n for (const line of positionedLines) {\n for (const positionedGlyph of line.positionedGlyphs) {\n positionedGlyph.x += shiftX;\n positionedGlyph.y += shiftY;\n }\n }\n}\n\nexport type PositionedIcon = {\n image: ImagePosition;\n top: number;\n bottom: number;\n left: number;\n right: number;\n collisionPadding?: [number, number, number, number];\n};\n\nfunction shapeIcon(\n image: ImagePosition,\n iconOffset: [number, number],\n iconAnchor: SymbolAnchor\n): PositionedIcon {\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(iconAnchor);\n const dx = iconOffset[0];\n const dy = iconOffset[1];\n const x1 = dx - image.displaySize[0] * horizontalAlign;\n const x2 = x1 + image.displaySize[0];\n const y1 = dy - image.displaySize[1] * verticalAlign;\n const y2 = y1 + image.displaySize[1];\n return {image, top: y1, bottom: y2, left: x1, right: x2};\n}\n\nfunction fitIconToText(\n shapedIcon: PositionedIcon,\n shapedText: Shaping,\n textFit: string,\n padding: [number, number, number, number],\n iconOffset: [number, number],\n fontScale: number\n): PositionedIcon {\n\n const image = shapedIcon.image;\n\n let collisionPadding;\n if (image.content) {\n const content = image.content;\n const pixelRatio = image.pixelRatio || 1;\n collisionPadding = [\n content[0] / pixelRatio,\n content[1] / pixelRatio,\n image.displaySize[0] - content[2] / pixelRatio,\n image.displaySize[1] - content[3] / pixelRatio\n ];\n }\n\n // We don't respect the icon-anchor, because icon-text-fit is set. Instead,\n // the icon will be centered on the text, then stretched in the given\n // dimensions.\n\n const textLeft = shapedText.left * fontScale;\n const textRight = shapedText.right * fontScale;\n\n let top, right, bottom, left;\n if (textFit === 'width' || textFit === 'both') {\n // Stretched horizontally to the text width\n left = iconOffset[0] + textLeft - padding[3];\n right = iconOffset[0] + textRight + padding[1];\n } else {\n // Centered on the text\n left = iconOffset[0] + (textLeft + textRight - image.displaySize[0]) / 2;\n right = left + image.displaySize[0];\n }\n\n const textTop = shapedText.top * fontScale;\n const textBottom = shapedText.bottom * fontScale;\n if (textFit === 'height' || textFit === 'both') {\n // Stretched vertically to the text height\n top = iconOffset[1] + textTop - padding[0];\n bottom = iconOffset[1] + textBottom + padding[2];\n } else {\n // Centered on the text\n top = iconOffset[1] + (textTop + textBottom - image.displaySize[1]) / 2;\n bottom = top + image.displaySize[1];\n }\n\n return {image, top, right, bottom, left, collisionPadding};\n}\n","import {Interpolate, interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {clamp} from '../util/util';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\n\nimport type {PropertyValue, PossiblyEvaluatedPropertyValue} from '../style/properties';\nimport type {InterpolationType} from '@maplibre/maplibre-gl-style-spec';\n\nconst MAX_GLYPH_ICON_SIZE = 255;\nconst SIZE_PACK_FACTOR = 128;\nconst MAX_PACKED_SIZE = MAX_GLYPH_ICON_SIZE * SIZE_PACK_FACTOR;\n\nexport {getSizeData, evaluateSizeForFeature, evaluateSizeForZoom, SIZE_PACK_FACTOR, MAX_GLYPH_ICON_SIZE, MAX_PACKED_SIZE};\n\nexport type SizeData = {\n kind: 'constant';\n layoutSize: number;\n} | {\n kind: 'source';\n} | {\n kind: 'camera';\n minZoom: number;\n maxZoom: number;\n minSize: number;\n maxSize: number;\n interpolationType: InterpolationType;\n} | {\n kind: 'composite';\n minZoom: number;\n maxZoom: number;\n interpolationType: InterpolationType;\n};\n\nexport type EvaluatedZoomSize = {uSizeT: number; uSize: number};\n\n// For {text,icon}-size, get the bucket-level data that will be needed by\n// the painter to set symbol-size-related uniforms\nfunction getSizeData(\n tileZoom: number,\n value: PropertyValue>\n): SizeData {\n const {expression} = value;\n\n if (expression.kind === 'constant') {\n const layoutSize = expression.evaluate(new EvaluationParameters(tileZoom + 1));\n return {kind: 'constant', layoutSize};\n\n } else if (expression.kind === 'source') {\n return {kind: 'source'};\n\n } else {\n const {zoomStops, interpolationType} = expression;\n\n // calculate covering zoom stops for zoom-dependent values\n let lower = 0;\n while (lower < zoomStops.length && zoomStops[lower] <= tileZoom) lower++;\n lower = Math.max(0, lower - 1);\n let upper = lower;\n while (upper < zoomStops.length && zoomStops[upper] < tileZoom + 1) upper++;\n upper = Math.min(zoomStops.length - 1, upper);\n\n const minZoom = zoomStops[lower];\n const maxZoom = zoomStops[upper];\n\n // We'd like to be able to use CameraExpression or CompositeExpression in these\n // return types rather than ExpressionSpecification, but the former are not\n // transferrable across Web Worker boundaries.\n if (expression.kind === 'composite') {\n return {kind: 'composite', minZoom, maxZoom, interpolationType};\n }\n\n // for camera functions, also save off the function values\n // evaluated at the covering zoom levels\n const minSize = expression.evaluate(new EvaluationParameters(minZoom));\n const maxSize = expression.evaluate(new EvaluationParameters(maxZoom));\n\n return {kind: 'camera', minZoom, maxZoom, minSize, maxSize, interpolationType};\n }\n}\n\nfunction evaluateSizeForFeature(sizeData: SizeData,\n {\n uSize,\n uSizeT\n }: {\n uSize: number;\n uSizeT: number;\n },\n {\n lowerSize,\n upperSize\n }: {\n lowerSize: number;\n upperSize: number;\n }): number {\n if (sizeData.kind === 'source') {\n return lowerSize / SIZE_PACK_FACTOR;\n } else if (sizeData.kind === 'composite') {\n return interpolates.number(lowerSize / SIZE_PACK_FACTOR, upperSize / SIZE_PACK_FACTOR, uSizeT);\n }\n return uSize;\n}\n\nfunction evaluateSizeForZoom(sizeData: SizeData, zoom: number): EvaluatedZoomSize {\n let uSizeT = 0;\n let uSize = 0;\n\n if (sizeData.kind === 'constant') {\n uSize = sizeData.layoutSize;\n\n } else if (sizeData.kind !== 'source') {\n const {interpolationType, minZoom, maxZoom} = sizeData;\n\n // Even though we could get the exact value of the camera function\n // at z = tr.zoom, we intentionally do not: instead, we interpolate\n // between the camera function values at a pair of zoom stops covering\n // [tileZoom, tileZoom + 1] in order to be consistent with this\n // restriction on composite functions\n const t = !interpolationType ? 0 : clamp(\n Interpolate.interpolationFactor(interpolationType, zoom, minZoom, maxZoom), 0, 1);\n\n if (sizeData.kind === 'camera') {\n uSize = interpolates.number(sizeData.minSize, sizeData.maxSize, t);\n } else {\n uSizeT = t;\n }\n }\n\n return {uSizeT, uSize};\n}\n","import {SymbolLayoutPropsPossiblyEvaluated} from './symbol_style_layer_properties.g';\nimport type {SymbolLayoutProps} from './symbol_style_layer_properties.g';\nimport {PossiblyEvaluated} from '../properties';\n\n/**\n * The overlap mode for properties like `icon-overlap`and `text-overlap`\n */\nexport type OverlapMode = 'never' | 'always' | 'cooperative';\n\nexport function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'icon-overlap', allowOverlapProp: 'icon-allow-overlap'): OverlapMode;\nexport function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'text-overlap', allowOverlapProp: 'text-allow-overlap'): OverlapMode;\nexport function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'icon-overlap' | 'text-overlap', allowOverlapProp: 'icon-allow-overlap' | 'text-allow-overlap'): OverlapMode {\n let result: OverlapMode = 'never';\n const overlap = layout.get(overlapProp);\n\n if (overlap) {\n // if -overlap is set, use it\n result = overlap;\n } else if (layout.get(allowOverlapProp)) {\n // fall back to -allow-overlap, with false='never', true='always'\n result = 'always';\n }\n\n return result;\n}\n","import {\n symbolLayoutAttributes,\n collisionVertexAttributes,\n collisionBoxLayout,\n dynamicLayoutAttributes,\n} from './symbol_attributes';\n\nimport {SymbolLayoutArray,\n SymbolDynamicLayoutArray,\n SymbolOpacityArray,\n CollisionBoxLayoutArray,\n CollisionVertexArray,\n PlacedSymbolArray,\n SymbolInstanceArray,\n GlyphOffsetArray,\n SymbolLineVertexArray,\n TextAnchorOffsetArray\n} from '../array_types.g';\n\nimport Point from '@mapbox/point-geometry';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray, LineIndexArray} from '../index_array_type';\nimport {transformText} from '../../symbol/transform_text';\nimport {mergeLines} from '../../symbol/merge_lines';\nimport {allowsVerticalWritingMode, stringContainsRTLText} from '../../util/script_detection';\nimport {WritingMode} from '../../symbol/shaping';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport mvt from '@mapbox/vector-tile';\nconst vectorTileFeatureTypes = mvt.VectorTileFeature.types;\nimport {verticalizedCharacterMap} from '../../util/verticalize_punctuation';\nimport {Anchor} from '../../symbol/anchor';\nimport {getSizeData, MAX_PACKED_SIZE} from '../../symbol/symbol_size';\n\nimport {register} from '../../util/web_worker_transfer';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\nimport {Formatted, ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport {plugin as globalRTLTextPlugin, getRTLTextPluginStatus} from '../../source/rtl_text_plugin';\nimport {mat4} from 'gl-matrix';\nimport {getOverlapMode} from '../../style/style_layer/overlap_mode';\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {CollisionBoxArray, CollisionBox, SymbolInstance} from '../array_types.g';\nimport type {StructArray, StructArrayMember, ViewType} from '../../util/struct_array';\nimport type {SymbolStyleLayer} from '../../style/style_layer/symbol_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type {SymbolQuad} from '../../symbol/quads';\nimport type {SizeData} from '../../symbol/symbol_size';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nexport type SingleCollisionBox = {\n x1: number;\n y1: number;\n x2: number;\n y2: number;\n anchorPointX: number;\n anchorPointY: number;\n};\n\nexport type CollisionArrays = {\n textBox?: SingleCollisionBox;\n verticalTextBox?: SingleCollisionBox;\n iconBox?: SingleCollisionBox;\n verticalIconBox?: SingleCollisionBox;\n textFeatureIndex?: number;\n verticalTextFeatureIndex?: number;\n iconFeatureIndex?: number;\n verticalIconFeatureIndex?: number;\n};\n\nexport type SymbolFeature = {\n sortKey: number | void;\n text: Formatted | void;\n icon: ResolvedImage;\n index: number;\n sourceLayerIndex: number;\n geometry: Array>;\n properties: any;\n type: 'Unknown' | 'Point' | 'LineString' | 'Polygon';\n id?: any;\n};\n\nexport type SortKeyRange = {\n sortKey: number;\n symbolInstanceStart: number;\n symbolInstanceEnd: number;\n};\n\n// Opacity arrays are frequently updated but don't contain a lot of information, so we pack them\n// tight. Each Uint32 is actually four duplicate Uint8s for the four corners of a glyph\n// 7 bits are for the current opacity, and the lowest bit is the target opacity\n\n// actually defined in symbol_attributes.js\n// const placementOpacityAttributes = [\n// { name: 'a_fade_opacity', components: 1, type: 'Uint32' }\n// ];\nconst shaderOpacityAttributes = [\n {name: 'a_fade_opacity', components: 1, type: 'Uint8' as ViewType, offset: 0}\n];\n\nfunction addVertex(\n array: StructArray,\n anchorX: number,\n anchorY: number,\n ox: number,\n oy: number,\n tx: number,\n ty: number,\n sizeVertex: number,\n isSDF: boolean,\n pixelOffsetX: number,\n pixelOffsetY: number,\n minFontScaleX: number,\n minFontScaleY: number\n) {\n const aSizeX = sizeVertex ? Math.min(MAX_PACKED_SIZE, Math.round(sizeVertex[0])) : 0;\n const aSizeY = sizeVertex ? Math.min(MAX_PACKED_SIZE, Math.round(sizeVertex[1])) : 0;\n array.emplaceBack(\n // a_pos_offset\n anchorX,\n anchorY,\n Math.round(ox * 32),\n Math.round(oy * 32),\n\n // a_data\n tx, // x coordinate of symbol on glyph atlas texture\n ty, // y coordinate of symbol on glyph atlas texture\n (aSizeX << 1) + (isSDF ? 1 : 0),\n aSizeY,\n pixelOffsetX * 16,\n pixelOffsetY * 16,\n minFontScaleX * 256,\n minFontScaleY * 256\n );\n}\n\nfunction addDynamicAttributes(dynamicLayoutVertexArray: StructArray, p: Point, angle: number) {\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n}\n\nfunction containsRTLText(formattedText: Formatted): boolean {\n for (const section of formattedText.sections) {\n if (stringContainsRTLText(section.text)) {\n return true;\n }\n }\n return false;\n}\n\nexport class SymbolBuffers {\n layoutVertexArray: SymbolLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n\n dynamicLayoutVertexArray: SymbolDynamicLayoutArray;\n dynamicLayoutVertexBuffer: VertexBuffer;\n\n opacityVertexArray: SymbolOpacityArray;\n opacityVertexBuffer: VertexBuffer;\n hasVisibleVertices: boolean;\n\n collisionVertexArray: CollisionVertexArray;\n collisionVertexBuffer: VertexBuffer;\n\n placedSymbolArray: PlacedSymbolArray;\n\n constructor(programConfigurations: ProgramConfigurationSet) {\n this.layoutVertexArray = new SymbolLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.programConfigurations = programConfigurations;\n this.segments = new SegmentVector();\n this.dynamicLayoutVertexArray = new SymbolDynamicLayoutArray();\n this.opacityVertexArray = new SymbolOpacityArray();\n this.hasVisibleVertices = false;\n this.placedSymbolArray = new PlacedSymbolArray();\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0 &&\n this.indexArray.length === 0 &&\n this.dynamicLayoutVertexArray.length === 0 &&\n this.opacityVertexArray.length === 0;\n }\n\n upload(context: Context, dynamicIndexBuffer: boolean, upload?: boolean, update?: boolean) {\n if (this.isEmpty()) {\n return;\n }\n\n if (upload) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, symbolLayoutAttributes.members);\n this.indexBuffer = context.createIndexBuffer(this.indexArray, dynamicIndexBuffer);\n this.dynamicLayoutVertexBuffer = context.createVertexBuffer(this.dynamicLayoutVertexArray, dynamicLayoutAttributes.members, true);\n this.opacityVertexBuffer = context.createVertexBuffer(this.opacityVertexArray, shaderOpacityAttributes, true);\n // This is a performance hack so that we can write to opacityVertexArray with uint32s\n // even though the shaders read uint8s\n this.opacityVertexBuffer.itemSize = 1;\n }\n if (upload || update) {\n this.programConfigurations.upload(context);\n }\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n this.dynamicLayoutVertexBuffer.destroy();\n this.opacityVertexBuffer.destroy();\n }\n}\n\nregister('SymbolBuffers', SymbolBuffers);\n\nclass CollisionBuffers {\n layoutVertexArray: StructArray;\n layoutAttributes: Array;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray | LineIndexArray;\n indexBuffer: IndexBuffer;\n\n segments: SegmentVector;\n\n collisionVertexArray: CollisionVertexArray;\n collisionVertexBuffer: VertexBuffer;\n\n constructor(LayoutArray: {\n new (...args: any): StructArray;\n },\n layoutAttributes: Array,\n IndexArray: {\n new (...args: any): TriangleIndexArray | LineIndexArray;\n }) {\n this.layoutVertexArray = new LayoutArray();\n this.layoutAttributes = layoutAttributes;\n this.indexArray = new IndexArray();\n this.segments = new SegmentVector();\n this.collisionVertexArray = new CollisionVertexArray();\n }\n\n upload(context: Context) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, this.layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n this.collisionVertexBuffer = context.createVertexBuffer(this.collisionVertexArray, collisionVertexAttributes.members, true);\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.segments.destroy();\n this.collisionVertexBuffer.destroy();\n }\n}\n\nregister('CollisionBuffers', CollisionBuffers);\n\n/**\n * @internal\n * Unlike other buckets, which simply implement #addFeature with type-specific\n * logic for (essentially) triangulating feature geometries, SymbolBucket\n * requires specialized behavior:\n *\n * 1. WorkerTile#parse(), the logical owner of the bucket creation process,\n * calls SymbolBucket#populate(), which resolves text and icon tokens on\n * each feature, adds each glyphs and symbols needed to the passed-in\n * collections options.glyphDependencies and options.iconDependencies, and\n * stores the feature data for use in subsequent step (this.features).\n *\n * 2. WorkerTile asynchronously requests from the main thread all of the glyphs\n * and icons needed (by this bucket and any others). When glyphs and icons\n * have been received, the WorkerTile creates a CollisionIndex and invokes:\n *\n * 3. performSymbolLayout(bucket, stacks, icons) perform texts shaping and\n * layout on a Symbol Bucket. This step populates:\n * `this.symbolInstances`: metadata on generated symbols\n * `this.collisionBoxArray`: collision data for use by foreground\n * `this.text`: SymbolBuffers for text symbols\n * `this.icons`: SymbolBuffers for icons\n * `this.iconCollisionBox`: Debug SymbolBuffers for icon collision boxes\n * `this.textCollisionBox`: Debug SymbolBuffers for text collision boxes\n * The results are sent to the foreground for rendering\n *\n * 4. performSymbolPlacement(bucket, collisionIndex) is run on the foreground,\n * and uses the CollisionIndex along with current camera settings to determine\n * which symbols can actually show on the map. Collided symbols are hidden\n * using a dynamic \"OpacityVertexArray\".\n */\nexport class SymbolBucket implements Bucket {\n static MAX_GLYPHS: number;\n static addDynamicAttributes: typeof addDynamicAttributes;\n\n collisionBoxArray: CollisionBoxArray;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n\n index: number;\n sdfIcons: boolean;\n iconsInText: boolean;\n iconsNeedLinear: boolean;\n bucketInstanceId: number;\n justReloaded: boolean;\n hasPattern: boolean;\n\n textSizeData: SizeData;\n iconSizeData: SizeData;\n\n glyphOffsetArray: GlyphOffsetArray;\n lineVertexArray: SymbolLineVertexArray;\n features: Array;\n symbolInstances: SymbolInstanceArray;\n textAnchorOffsets: TextAnchorOffsetArray;\n collisionArrays: Array;\n sortKeyRanges: Array;\n pixelRatio: number;\n tilePixelRatio: number;\n compareText: {[_: string]: Array};\n fadeStartTime: number;\n sortFeaturesByKey: boolean;\n sortFeaturesByY: boolean;\n canOverlap: boolean;\n sortedAngle: number;\n featureSortOrder: Array;\n\n collisionCircleArray: Array;\n placementInvProjMatrix: mat4;\n placementViewportMatrix: mat4;\n\n text: SymbolBuffers;\n icon: SymbolBuffers;\n textCollisionBox: CollisionBuffers;\n iconCollisionBox: CollisionBuffers;\n uploaded: boolean;\n sourceLayerIndex: number;\n sourceID: string;\n symbolInstanceIndexes: Array;\n writingModes: WritingMode[];\n allowVerticalPlacement: boolean;\n hasRTLText: boolean;\n\n constructor(options: BucketParameters) {\n this.collisionBoxArray = options.collisionBoxArray;\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.pixelRatio = options.pixelRatio;\n this.sourceLayerIndex = options.sourceLayerIndex;\n this.hasPattern = false;\n this.hasRTLText = false;\n this.sortKeyRanges = [];\n\n this.collisionCircleArray = [];\n this.placementInvProjMatrix = mat4.identity([] as any);\n this.placementViewportMatrix = mat4.identity([] as any);\n\n const layer = this.layers[0];\n const unevaluatedLayoutValues = layer._unevaluatedLayout._values;\n\n this.textSizeData = getSizeData(this.zoom, unevaluatedLayoutValues['text-size']);\n this.iconSizeData = getSizeData(this.zoom, unevaluatedLayoutValues['icon-size']);\n\n const layout = this.layers[0].layout;\n const sortKey = layout.get('symbol-sort-key');\n const zOrder = layout.get('symbol-z-order');\n this.canOverlap =\n getOverlapMode(layout, 'text-overlap', 'text-allow-overlap') !== 'never' ||\n getOverlapMode(layout, 'icon-overlap', 'icon-allow-overlap') !== 'never' ||\n layout.get('text-ignore-placement') ||\n layout.get('icon-ignore-placement');\n this.sortFeaturesByKey = zOrder !== 'viewport-y' && !sortKey.isConstant();\n const zOrderByViewportY = zOrder === 'viewport-y' || (zOrder === 'auto' && !this.sortFeaturesByKey);\n this.sortFeaturesByY = zOrderByViewportY && this.canOverlap;\n\n if (layout.get('symbol-placement') === 'point') {\n this.writingModes = layout.get('text-writing-mode').map(wm => WritingMode[wm]);\n }\n\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n\n this.sourceID = options.sourceID;\n }\n\n createArrays() {\n this.text = new SymbolBuffers(new ProgramConfigurationSet(this.layers, this.zoom, property => /^text/.test(property)));\n this.icon = new SymbolBuffers(new ProgramConfigurationSet(this.layers, this.zoom, property => /^icon/.test(property)));\n\n this.glyphOffsetArray = new GlyphOffsetArray();\n this.lineVertexArray = new SymbolLineVertexArray();\n this.symbolInstances = new SymbolInstanceArray();\n this.textAnchorOffsets = new TextAnchorOffsetArray();\n }\n\n calculateGlyphDependencies(text: string, stack: {[_: number]: boolean}, textAlongLine: boolean, allowVerticalPlacement: boolean, doesAllowVerticalWritingMode: boolean) {\n for (let i = 0; i < text.length; i++) {\n stack[text.charCodeAt(i)] = true;\n if ((textAlongLine || allowVerticalPlacement) && doesAllowVerticalWritingMode) {\n const verticalChar = verticalizedCharacterMap[text.charAt(i)];\n if (verticalChar) {\n stack[verticalChar.charCodeAt(0)] = true;\n }\n }\n }\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n const layer = this.layers[0];\n const layout = layer.layout;\n\n const textFont = layout.get('text-font');\n const textField = layout.get('text-field');\n const iconImage = layout.get('icon-image');\n const hasText =\n (textField.value.kind !== 'constant' ||\n (textField.value.value instanceof Formatted && !textField.value.value.isEmpty()) ||\n textField.value.value.toString().length > 0) &&\n (textFont.value.kind !== 'constant' || textFont.value.value.length > 0);\n // we should always resolve the icon-image value if the property was defined in the style\n // this allows us to fire the styleimagemissing event if image evaluation returns null\n // the only way to distinguish between null returned from a coalesce statement with no valid images\n // and null returned because icon-image wasn't defined is to check whether or not iconImage.parameters is an empty object\n const hasIcon = iconImage.value.kind !== 'constant' || !!iconImage.value.value || Object.keys(iconImage.parameters).length > 0;\n const symbolSortKey = layout.get('symbol-sort-key');\n\n this.features = [];\n\n if (!hasText && !hasIcon) {\n return;\n }\n\n const icons = options.iconDependencies;\n const stacks = options.glyphDependencies;\n const availableImages = options.availableImages;\n const globalProperties = new EvaluationParameters(this.zoom);\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n\n const needGeometry = layer._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n if (!layer._featureFilter.filter(globalProperties, evaluationFeature, canonical)) {\n continue;\n }\n\n if (!needGeometry) evaluationFeature.geometry = loadGeometry(feature);\n\n let text: Formatted | void;\n if (hasText) {\n // Expression evaluation will automatically coerce to Formatted\n // but plain string token evaluation skips that pathway so do the\n // conversion here.\n const resolvedTokens = layer.getValueAndResolveTokens('text-field', evaluationFeature, canonical, availableImages);\n const formattedText = Formatted.factory(resolvedTokens);\n if (containsRTLText(formattedText)) {\n this.hasRTLText = true;\n }\n if (\n !this.hasRTLText || // non-rtl text so can proceed safely\n getRTLTextPluginStatus() === 'unavailable' || // We don't intend to lazy-load the rtl text plugin, so proceed with incorrect shaping\n this.hasRTLText && globalRTLTextPlugin.isParsed() // Use the rtlText plugin to shape text\n ) {\n text = transformText(formattedText, layer, evaluationFeature);\n }\n }\n\n let icon: ResolvedImage;\n if (hasIcon) {\n // Expression evaluation will automatically coerce to Image\n // but plain string token evaluation skips that pathway so do the\n // conversion here.\n const resolvedTokens = layer.getValueAndResolveTokens('icon-image', evaluationFeature, canonical, availableImages);\n if (resolvedTokens instanceof ResolvedImage) {\n icon = resolvedTokens;\n } else {\n icon = ResolvedImage.fromString(resolvedTokens);\n }\n }\n\n if (!text && !icon) {\n continue;\n }\n const sortKey = this.sortFeaturesByKey ?\n symbolSortKey.evaluate(evaluationFeature, {}, canonical) :\n undefined;\n\n const symbolFeature: SymbolFeature = {\n id,\n text,\n icon,\n index,\n sourceLayerIndex,\n geometry: evaluationFeature.geometry,\n properties: feature.properties,\n type: vectorTileFeatureTypes[feature.type],\n sortKey\n };\n this.features.push(symbolFeature);\n\n if (icon) {\n icons[icon.name] = true;\n }\n\n if (text) {\n const fontStack = textFont.evaluate(evaluationFeature, {}, canonical).join(',');\n const textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point';\n this.allowVerticalPlacement = this.writingModes && this.writingModes.indexOf(WritingMode.vertical) >= 0;\n for (const section of text.sections) {\n if (!section.image) {\n const doesAllowVerticalWritingMode = allowsVerticalWritingMode(text.toString());\n const sectionFont = section.fontStack || fontStack;\n const sectionStack = stacks[sectionFont] = stacks[sectionFont] || {};\n this.calculateGlyphDependencies(section.text, sectionStack, textAlongLine, this.allowVerticalPlacement, doesAllowVerticalWritingMode);\n } else {\n // Add section image to the list of dependencies.\n icons[section.image.name] = true;\n }\n }\n }\n }\n\n if (layout.get('symbol-placement') === 'line') {\n // Merge adjacent lines with the same text to improve labelling.\n // It's better to place labels on one long line than on many short segments.\n this.features = mergeLines(this.features);\n }\n\n if (this.sortFeaturesByKey) {\n this.features.sort((a, b) => {\n // a.sortKey is always a number when sortFeaturesByKey is true\n return (a.sortKey as number) - (b.sortKey as number);\n });\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.text.programConfigurations.updatePaintArrays(states, vtLayer, this.layers, imagePositions);\n this.icon.programConfigurations.updatePaintArrays(states, vtLayer, this.layers, imagePositions);\n }\n\n isEmpty() {\n // When the bucket encounters only rtl-text but the plugin isnt loaded, no symbol instances will be created.\n // In order for the bucket to be serialized, and not discarded as an empty bucket both checks are necessary.\n return this.symbolInstances.length === 0 && !this.hasRTLText;\n }\n\n uploadPending() {\n return !this.uploaded || this.text.programConfigurations.needsUpload || this.icon.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded && this.hasDebugData()) {\n this.textCollisionBox.upload(context);\n this.iconCollisionBox.upload(context);\n }\n this.text.upload(context, this.sortFeaturesByY, !this.uploaded, this.text.programConfigurations.needsUpload);\n this.icon.upload(context, this.sortFeaturesByY, !this.uploaded, this.icon.programConfigurations.needsUpload);\n this.uploaded = true;\n }\n\n destroyDebugData() {\n this.textCollisionBox.destroy();\n this.iconCollisionBox.destroy();\n }\n\n destroy() {\n this.text.destroy();\n this.icon.destroy();\n\n if (this.hasDebugData()) {\n this.destroyDebugData();\n }\n }\n\n addToLineVertexArray(anchor: Anchor, line: any) {\n const lineStartIndex = this.lineVertexArray.length;\n if (anchor.segment !== undefined) {\n let sumForwardLength = anchor.dist(line[anchor.segment + 1]);\n let sumBackwardLength = anchor.dist(line[anchor.segment]);\n const vertices = {};\n for (let i = anchor.segment + 1; i < line.length; i++) {\n vertices[i] = {x: line[i].x, y: line[i].y, tileUnitDistanceFromAnchor: sumForwardLength};\n if (i < line.length - 1) {\n sumForwardLength += line[i + 1].dist(line[i]);\n }\n }\n for (let i = anchor.segment || 0; i >= 0; i--) {\n vertices[i] = {x: line[i].x, y: line[i].y, tileUnitDistanceFromAnchor: sumBackwardLength};\n if (i > 0) {\n sumBackwardLength += line[i - 1].dist(line[i]);\n }\n }\n for (let i = 0; i < line.length; i++) {\n const vertex = vertices[i];\n this.lineVertexArray.emplaceBack(vertex.x, vertex.y, vertex.tileUnitDistanceFromAnchor);\n }\n }\n return {\n lineStartIndex,\n lineLength: this.lineVertexArray.length - lineStartIndex\n };\n }\n\n addSymbols(arrays: SymbolBuffers,\n quads: Array,\n sizeVertex: any,\n lineOffset: [number, number],\n alongLine: boolean,\n feature: SymbolFeature,\n writingMode: WritingMode,\n labelAnchor: Anchor,\n lineStartIndex: number,\n lineLength: number,\n associatedIconIndex: number,\n canonical: CanonicalTileID) {\n const indexArray = arrays.indexArray;\n const layoutVertexArray = arrays.layoutVertexArray;\n\n const segment = arrays.segments.prepareSegment(4 * quads.length, layoutVertexArray, indexArray, this.canOverlap ? feature.sortKey as number : undefined);\n const glyphOffsetArrayStart = this.glyphOffsetArray.length;\n const vertexStartIndex = segment.vertexLength;\n\n const angle = (this.allowVerticalPlacement && writingMode === WritingMode.vertical) ? Math.PI / 2 : 0;\n\n const sections = feature.text && feature.text.sections;\n\n for (let i = 0; i < quads.length; i++) {\n const {tl, tr, bl, br, tex, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY, glyphOffset, isSDF, sectionIndex} = quads[i];\n const index = segment.vertexLength;\n\n const y = glyphOffset[1];\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, tl.x, y + tl.y, tex.x, tex.y, sizeVertex, isSDF, pixelOffsetTL.x, pixelOffsetTL.y, minFontScaleX, minFontScaleY);\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, tr.x, y + tr.y, tex.x + tex.w, tex.y, sizeVertex, isSDF, pixelOffsetBR.x, pixelOffsetTL.y, minFontScaleX, minFontScaleY);\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, bl.x, y + bl.y, tex.x, tex.y + tex.h, sizeVertex, isSDF, pixelOffsetTL.x, pixelOffsetBR.y, minFontScaleX, minFontScaleY);\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, br.x, y + br.y, tex.x + tex.w, tex.y + tex.h, sizeVertex, isSDF, pixelOffsetBR.x, pixelOffsetBR.y, minFontScaleX, minFontScaleY);\n\n addDynamicAttributes(arrays.dynamicLayoutVertexArray, labelAnchor, angle);\n\n indexArray.emplaceBack(index, index + 1, index + 2);\n indexArray.emplaceBack(index + 1, index + 2, index + 3);\n\n segment.vertexLength += 4;\n segment.primitiveLength += 2;\n\n this.glyphOffsetArray.emplaceBack(glyphOffset[0]);\n\n if (i === quads.length - 1 || sectionIndex !== quads[i + 1].sectionIndex) {\n arrays.programConfigurations.populatePaintArrays(layoutVertexArray.length, feature, feature.index, {}, canonical, sections && sections[sectionIndex]);\n }\n }\n\n arrays.placedSymbolArray.emplaceBack(\n labelAnchor.x, labelAnchor.y,\n glyphOffsetArrayStart,\n this.glyphOffsetArray.length - glyphOffsetArrayStart,\n vertexStartIndex,\n lineStartIndex,\n lineLength,\n labelAnchor.segment,\n sizeVertex ? sizeVertex[0] : 0,\n sizeVertex ? sizeVertex[1] : 0,\n lineOffset[0], lineOffset[1],\n writingMode,\n // placedOrientation is null initially; will be updated to horizontal(1)/vertical(2) if placed\n 0,\n false as unknown as number,\n // The crossTileID is only filled/used on the foreground for dynamic text anchors\n 0,\n associatedIconIndex\n );\n }\n\n _addCollisionDebugVertex(layoutVertexArray: StructArray, collisionVertexArray: StructArray, point: Point, anchorX: number, anchorY: number, extrude: Point) {\n collisionVertexArray.emplaceBack(0, 0);\n return layoutVertexArray.emplaceBack(\n // pos\n point.x,\n point.y,\n // a_anchor_pos\n anchorX,\n anchorY,\n // extrude\n Math.round(extrude.x),\n Math.round(extrude.y));\n }\n\n addCollisionDebugVertices(x1: number, y1: number, x2: number, y2: number, arrays: CollisionBuffers, boxAnchorPoint: Point, symbolInstance: SymbolInstance) {\n const segment = arrays.segments.prepareSegment(4, arrays.layoutVertexArray, arrays.indexArray);\n const index = segment.vertexLength;\n\n const layoutVertexArray = arrays.layoutVertexArray;\n const collisionVertexArray = arrays.collisionVertexArray;\n\n const anchorX = symbolInstance.anchorX;\n const anchorY = symbolInstance.anchorY;\n\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x1, y1));\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x2, y1));\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x2, y2));\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x1, y2));\n\n segment.vertexLength += 4;\n\n const indexArray = arrays.indexArray as LineIndexArray;\n indexArray.emplaceBack(index, index + 1);\n indexArray.emplaceBack(index + 1, index + 2);\n indexArray.emplaceBack(index + 2, index + 3);\n indexArray.emplaceBack(index + 3, index);\n\n segment.primitiveLength += 4;\n }\n\n addDebugCollisionBoxes(startIndex: number, endIndex: number, symbolInstance: SymbolInstance, isText: boolean) {\n for (let b = startIndex; b < endIndex; b++) {\n const box: CollisionBox = this.collisionBoxArray.get(b);\n const x1 = box.x1;\n const y1 = box.y1;\n const x2 = box.x2;\n const y2 = box.y2;\n\n this.addCollisionDebugVertices(x1, y1, x2, y2,\n isText ? this.textCollisionBox : this.iconCollisionBox,\n box.anchorPoint, symbolInstance);\n }\n }\n\n generateCollisionDebugBuffers() {\n if (this.hasDebugData()) {\n this.destroyDebugData();\n }\n\n this.textCollisionBox = new CollisionBuffers(CollisionBoxLayoutArray, collisionBoxLayout.members, LineIndexArray);\n this.iconCollisionBox = new CollisionBuffers(CollisionBoxLayoutArray, collisionBoxLayout.members, LineIndexArray);\n\n for (let i = 0; i < this.symbolInstances.length; i++) {\n const symbolInstance = this.symbolInstances.get(i);\n this.addDebugCollisionBoxes(symbolInstance.textBoxStartIndex, symbolInstance.textBoxEndIndex, symbolInstance, true);\n this.addDebugCollisionBoxes(symbolInstance.verticalTextBoxStartIndex, symbolInstance.verticalTextBoxEndIndex, symbolInstance, true);\n this.addDebugCollisionBoxes(symbolInstance.iconBoxStartIndex, symbolInstance.iconBoxEndIndex, symbolInstance, false);\n this.addDebugCollisionBoxes(symbolInstance.verticalIconBoxStartIndex, symbolInstance.verticalIconBoxEndIndex, symbolInstance, false);\n }\n }\n\n // These flat arrays are meant to be quicker to iterate over than the source\n // CollisionBoxArray\n _deserializeCollisionBoxesForSymbol(\n collisionBoxArray: CollisionBoxArray,\n textStartIndex: number,\n textEndIndex: number,\n verticalTextStartIndex: number,\n verticalTextEndIndex: number,\n iconStartIndex: number,\n iconEndIndex: number,\n verticalIconStartIndex: number,\n verticalIconEndIndex: number\n ): CollisionArrays {\n\n const collisionArrays = {} as CollisionArrays;\n for (let k = textStartIndex; k < textEndIndex; k++) {\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.textBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.textFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n for (let k = verticalTextStartIndex; k < verticalTextEndIndex; k++) {\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.verticalTextBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.verticalTextFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n for (let k = iconStartIndex; k < iconEndIndex; k++) {\n // An icon can only have one box now, so this indexing is a bit vestigial...\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.iconBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.iconFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n for (let k = verticalIconStartIndex; k < verticalIconEndIndex; k++) {\n // An icon can only have one box now, so this indexing is a bit vestigial...\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.verticalIconBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.verticalIconFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n return collisionArrays;\n }\n\n deserializeCollisionBoxes(collisionBoxArray: CollisionBoxArray) {\n this.collisionArrays = [];\n for (let i = 0; i < this.symbolInstances.length; i++) {\n const symbolInstance = this.symbolInstances.get(i);\n this.collisionArrays.push(this._deserializeCollisionBoxesForSymbol(\n collisionBoxArray,\n symbolInstance.textBoxStartIndex,\n symbolInstance.textBoxEndIndex,\n symbolInstance.verticalTextBoxStartIndex,\n symbolInstance.verticalTextBoxEndIndex,\n symbolInstance.iconBoxStartIndex,\n symbolInstance.iconBoxEndIndex,\n symbolInstance.verticalIconBoxStartIndex,\n symbolInstance.verticalIconBoxEndIndex\n ));\n }\n }\n\n hasTextData() {\n return this.text.segments.get().length > 0;\n }\n\n hasIconData() {\n return this.icon.segments.get().length > 0;\n }\n\n hasDebugData() {\n return this.textCollisionBox && this.iconCollisionBox;\n }\n\n hasTextCollisionBoxData() {\n return this.hasDebugData() && this.textCollisionBox.segments.get().length > 0;\n }\n\n hasIconCollisionBoxData() {\n return this.hasDebugData() && this.iconCollisionBox.segments.get().length > 0;\n }\n\n addIndicesForPlacedSymbol(iconOrText: SymbolBuffers, placedSymbolIndex: number) {\n const placedSymbol = iconOrText.placedSymbolArray.get(placedSymbolIndex);\n\n const endIndex = placedSymbol.vertexStartIndex + placedSymbol.numGlyphs * 4;\n for (let vertexIndex = placedSymbol.vertexStartIndex; vertexIndex < endIndex; vertexIndex += 4) {\n iconOrText.indexArray.emplaceBack(vertexIndex, vertexIndex + 1, vertexIndex + 2);\n iconOrText.indexArray.emplaceBack(vertexIndex + 1, vertexIndex + 2, vertexIndex + 3);\n }\n }\n\n getSortedSymbolIndexes(angle: number) {\n if (this.sortedAngle === angle && this.symbolInstanceIndexes !== undefined) {\n return this.symbolInstanceIndexes;\n }\n const sin = Math.sin(angle);\n const cos = Math.cos(angle);\n const rotatedYs = [];\n const featureIndexes = [];\n const result = [];\n\n for (let i = 0; i < this.symbolInstances.length; ++i) {\n result.push(i);\n const symbolInstance = this.symbolInstances.get(i);\n rotatedYs.push(Math.round(sin * symbolInstance.anchorX + cos * symbolInstance.anchorY) | 0);\n featureIndexes.push(symbolInstance.featureIndex);\n }\n\n result.sort((aIndex, bIndex) => {\n return (rotatedYs[aIndex] - rotatedYs[bIndex]) ||\n (featureIndexes[bIndex] - featureIndexes[aIndex]);\n });\n\n return result;\n }\n\n addToSortKeyRanges(symbolInstanceIndex: number, sortKey: number) {\n const last = this.sortKeyRanges[this.sortKeyRanges.length - 1];\n if (last && last.sortKey === sortKey) {\n last.symbolInstanceEnd = symbolInstanceIndex + 1;\n } else {\n this.sortKeyRanges.push({\n sortKey,\n symbolInstanceStart: symbolInstanceIndex,\n symbolInstanceEnd: symbolInstanceIndex + 1\n });\n }\n }\n\n sortFeatures(angle: number) {\n if (!this.sortFeaturesByY) return;\n if (this.sortedAngle === angle) return;\n\n // The current approach to sorting doesn't sort across segments so don't try.\n // Sorting within segments separately seemed not to be worth the complexity.\n if (this.text.segments.get().length > 1 || this.icon.segments.get().length > 1) return;\n\n // If the symbols are allowed to overlap sort them by their vertical screen position.\n // The index array buffer is rewritten to reference the (unchanged) vertices in the\n // sorted order.\n\n // To avoid sorting the actual symbolInstance array we sort an array of indexes.\n this.symbolInstanceIndexes = this.getSortedSymbolIndexes(angle);\n this.sortedAngle = angle;\n\n this.text.indexArray.clear();\n this.icon.indexArray.clear();\n\n this.featureSortOrder = [];\n\n for (const i of this.symbolInstanceIndexes) {\n const symbolInstance = this.symbolInstances.get(i);\n this.featureSortOrder.push(symbolInstance.featureIndex);\n\n [\n symbolInstance.rightJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.leftJustifiedTextSymbolIndex\n ].forEach((index, i, array) => {\n // Only add a given index the first time it shows up,\n // to avoid duplicate opacity entries when multiple justifications\n // share the same glyphs.\n if (index >= 0 && array.indexOf(index) === i) {\n this.addIndicesForPlacedSymbol(this.text, index);\n }\n });\n\n if (symbolInstance.verticalPlacedTextSymbolIndex >= 0) {\n this.addIndicesForPlacedSymbol(this.text, symbolInstance.verticalPlacedTextSymbolIndex);\n }\n\n if (symbolInstance.placedIconSymbolIndex >= 0) {\n this.addIndicesForPlacedSymbol(this.icon, symbolInstance.placedIconSymbolIndex);\n }\n\n if (symbolInstance.verticalPlacedIconSymbolIndex >= 0) {\n this.addIndicesForPlacedSymbol(this.icon, symbolInstance.verticalPlacedIconSymbolIndex);\n }\n }\n\n if (this.text.indexBuffer) this.text.indexBuffer.updateData(this.text.indexArray);\n if (this.icon.indexBuffer) this.icon.indexBuffer.updateData(this.icon.indexArray);\n }\n}\n\nregister('SymbolBucket', SymbolBucket, {\n omit: ['layers', 'collisionBoxArray', 'features', 'compareText']\n});\n\n// this constant is based on the size of StructArray indexes used in a symbol\n// bucket--namely, glyphOffsetArrayStart\n// eg the max valid UInt16 is 65,535\n// See https://github.com/mapbox/mapbox-gl-js/issues/2907 for motivation\n// lineStartIndex and textBoxStartIndex could potentially be concerns\n// but we expect there to be many fewer boxes/lines than glyphs\nSymbolBucket.MAX_GLYPHS = 65535;\n\nSymbolBucket.addDynamicAttributes = addDynamicAttributes;\n\nexport {addDynamicAttributes};\n","import type {SymbolFeature} from '../data/bucket/symbol_bucket';\n\nexport function mergeLines(features: Array): Array {\n const leftIndex: {[_: string]: number} = {};\n const rightIndex: {[_: string]: number} = {};\n const mergedFeatures = [];\n let mergedIndex = 0;\n\n function add(k) {\n mergedFeatures.push(features[k]);\n mergedIndex++;\n }\n\n function mergeFromRight(leftKey: string, rightKey: string, geom) {\n const i = rightIndex[leftKey];\n delete rightIndex[leftKey];\n rightIndex[rightKey] = i;\n\n mergedFeatures[i].geometry[0].pop();\n mergedFeatures[i].geometry[0] = mergedFeatures[i].geometry[0].concat(geom[0]);\n return i;\n }\n\n function mergeFromLeft(leftKey: string, rightKey: string, geom) {\n const i = leftIndex[rightKey];\n delete leftIndex[rightKey];\n leftIndex[leftKey] = i;\n\n mergedFeatures[i].geometry[0].shift();\n mergedFeatures[i].geometry[0] = geom[0].concat(mergedFeatures[i].geometry[0]);\n return i;\n }\n\n function getKey(text, geom, onRight?) {\n const point = onRight ? geom[0][geom[0].length - 1] : geom[0][0];\n return `${text}:${point.x}:${point.y}`;\n }\n\n for (let k = 0; k < features.length; k++) {\n const feature = features[k];\n const geom = feature.geometry;\n const text = feature.text ? feature.text.toString() : null;\n\n if (!text) {\n add(k);\n continue;\n }\n\n const leftKey = getKey(text, geom),\n rightKey = getKey(text, geom, true);\n\n if ((leftKey in rightIndex) && (rightKey in leftIndex) && (rightIndex[leftKey] !== leftIndex[rightKey])) {\n // found lines with the same text adjacent to both ends of the current line, merge all three\n const j = mergeFromLeft(leftKey, rightKey, geom);\n const i = mergeFromRight(leftKey, rightKey, mergedFeatures[j].geometry);\n\n delete leftIndex[leftKey];\n delete rightIndex[rightKey];\n\n rightIndex[getKey(text, mergedFeatures[i].geometry, true)] = i;\n mergedFeatures[j].geometry = null;\n\n } else if (leftKey in rightIndex) {\n // found mergeable line adjacent to the start of the current line, merge\n mergeFromRight(leftKey, rightKey, geom);\n\n } else if (rightKey in leftIndex) {\n // found mergeable line adjacent to the end of the current line, merge\n mergeFromLeft(leftKey, rightKey, geom);\n\n } else {\n // no adjacent lines, add as a new item\n add(k);\n leftIndex[leftKey] = mergedIndex - 1;\n rightIndex[rightKey] = mergedIndex - 1;\n }\n }\n\n return mergedFeatures.filter((f) => f.geometry);\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n ColorType\n } from '@maplibre/maplibre-gl-style-spec';\n \nexport type SymbolLayoutProps = {\n \"symbol-placement\": DataConstantProperty<\"point\" | \"line\" | \"line-center\">,\n \"symbol-spacing\": DataConstantProperty,\n \"symbol-avoid-edges\": DataConstantProperty,\n \"symbol-sort-key\": DataDrivenProperty,\n \"symbol-z-order\": DataConstantProperty<\"auto\" | \"viewport-y\" | \"source\">,\n \"icon-allow-overlap\": DataConstantProperty,\n \"icon-overlap\": DataConstantProperty<\"never\" | \"always\" | \"cooperative\">,\n \"icon-ignore-placement\": DataConstantProperty,\n \"icon-optional\": DataConstantProperty,\n \"icon-rotation-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"auto\">,\n \"icon-size\": DataDrivenProperty,\n \"icon-text-fit\": DataConstantProperty<\"none\" | \"width\" | \"height\" | \"both\">,\n \"icon-text-fit-padding\": DataConstantProperty<[number, number, number, number]>,\n \"icon-image\": DataDrivenProperty,\n \"icon-rotate\": DataDrivenProperty,\n \"icon-padding\": DataDrivenProperty,\n \"icon-keep-upright\": DataConstantProperty,\n \"icon-offset\": DataDrivenProperty<[number, number]>,\n \"icon-anchor\": DataDrivenProperty<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"icon-pitch-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"auto\">,\n \"text-pitch-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"auto\">,\n \"text-rotation-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"viewport-glyph\" | \"auto\">,\n \"text-field\": DataDrivenProperty,\n \"text-font\": DataDrivenProperty>,\n \"text-size\": DataDrivenProperty,\n \"text-max-width\": DataDrivenProperty,\n \"text-line-height\": DataConstantProperty,\n \"text-letter-spacing\": DataDrivenProperty,\n \"text-justify\": DataDrivenProperty<\"auto\" | \"left\" | \"center\" | \"right\">,\n \"text-radial-offset\": DataDrivenProperty,\n \"text-variable-anchor\": DataConstantProperty>,\n \"text-variable-anchor-offset\": DataDrivenProperty,\n \"text-anchor\": DataDrivenProperty<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"text-max-angle\": DataConstantProperty,\n \"text-writing-mode\": DataConstantProperty>,\n \"text-rotate\": DataDrivenProperty,\n \"text-padding\": DataConstantProperty,\n \"text-keep-upright\": DataConstantProperty,\n \"text-transform\": DataDrivenProperty<\"none\" | \"uppercase\" | \"lowercase\">,\n \"text-offset\": DataDrivenProperty<[number, number]>,\n \"text-allow-overlap\": DataConstantProperty,\n \"text-overlap\": DataConstantProperty<\"never\" | \"always\" | \"cooperative\">,\n \"text-ignore-placement\": DataConstantProperty,\n \"text-optional\": DataConstantProperty,\n};\n\nexport type SymbolLayoutPropsPossiblyEvaluated = {\n \"symbol-placement\": \"point\" | \"line\" | \"line-center\",\n \"symbol-spacing\": number,\n \"symbol-avoid-edges\": boolean,\n \"symbol-sort-key\": PossiblyEvaluatedPropertyValue,\n \"symbol-z-order\": \"auto\" | \"viewport-y\" | \"source\",\n \"icon-allow-overlap\": boolean,\n \"icon-overlap\": \"never\" | \"always\" | \"cooperative\",\n \"icon-ignore-placement\": boolean,\n \"icon-optional\": boolean,\n \"icon-rotation-alignment\": \"map\" | \"viewport\" | \"auto\",\n \"icon-size\": PossiblyEvaluatedPropertyValue,\n \"icon-text-fit\": \"none\" | \"width\" | \"height\" | \"both\",\n \"icon-text-fit-padding\": [number, number, number, number],\n \"icon-image\": PossiblyEvaluatedPropertyValue,\n \"icon-rotate\": PossiblyEvaluatedPropertyValue,\n \"icon-padding\": PossiblyEvaluatedPropertyValue,\n \"icon-keep-upright\": boolean,\n \"icon-offset\": PossiblyEvaluatedPropertyValue<[number, number]>,\n \"icon-anchor\": PossiblyEvaluatedPropertyValue<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"icon-pitch-alignment\": \"map\" | \"viewport\" | \"auto\",\n \"text-pitch-alignment\": \"map\" | \"viewport\" | \"auto\",\n \"text-rotation-alignment\": \"map\" | \"viewport\" | \"viewport-glyph\" | \"auto\",\n \"text-field\": PossiblyEvaluatedPropertyValue,\n \"text-font\": PossiblyEvaluatedPropertyValue>,\n \"text-size\": PossiblyEvaluatedPropertyValue,\n \"text-max-width\": PossiblyEvaluatedPropertyValue,\n \"text-line-height\": number,\n \"text-letter-spacing\": PossiblyEvaluatedPropertyValue,\n \"text-justify\": PossiblyEvaluatedPropertyValue<\"auto\" | \"left\" | \"center\" | \"right\">,\n \"text-radial-offset\": PossiblyEvaluatedPropertyValue,\n \"text-variable-anchor\": Array<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"text-variable-anchor-offset\": PossiblyEvaluatedPropertyValue,\n \"text-anchor\": PossiblyEvaluatedPropertyValue<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"text-max-angle\": number,\n \"text-writing-mode\": Array<\"horizontal\" | \"vertical\">,\n \"text-rotate\": PossiblyEvaluatedPropertyValue,\n \"text-padding\": number,\n \"text-keep-upright\": boolean,\n \"text-transform\": PossiblyEvaluatedPropertyValue<\"none\" | \"uppercase\" | \"lowercase\">,\n \"text-offset\": PossiblyEvaluatedPropertyValue<[number, number]>,\n \"text-allow-overlap\": boolean,\n \"text-overlap\": \"never\" | \"always\" | \"cooperative\",\n \"text-ignore-placement\": boolean,\n \"text-optional\": boolean,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"symbol-placement\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-placement\"] as any as StylePropertySpecification),\n \"symbol-spacing\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-spacing\"] as any as StylePropertySpecification),\n \"symbol-avoid-edges\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-avoid-edges\"] as any as StylePropertySpecification),\n \"symbol-sort-key\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"symbol-sort-key\"] as any as StylePropertySpecification),\n \"symbol-z-order\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-z-order\"] as any as StylePropertySpecification),\n \"icon-allow-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-allow-overlap\"] as any as StylePropertySpecification),\n \"icon-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-overlap\"] as any as StylePropertySpecification),\n \"icon-ignore-placement\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-ignore-placement\"] as any as StylePropertySpecification),\n \"icon-optional\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-optional\"] as any as StylePropertySpecification),\n \"icon-rotation-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-rotation-alignment\"] as any as StylePropertySpecification),\n \"icon-size\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-size\"] as any as StylePropertySpecification),\n \"icon-text-fit\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-text-fit\"] as any as StylePropertySpecification),\n \"icon-text-fit-padding\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-text-fit-padding\"] as any as StylePropertySpecification),\n \"icon-image\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-image\"] as any as StylePropertySpecification),\n \"icon-rotate\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-rotate\"] as any as StylePropertySpecification),\n \"icon-padding\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-padding\"] as any as StylePropertySpecification),\n \"icon-keep-upright\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-keep-upright\"] as any as StylePropertySpecification),\n \"icon-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-offset\"] as any as StylePropertySpecification),\n \"icon-anchor\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-anchor\"] as any as StylePropertySpecification),\n \"icon-pitch-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-pitch-alignment\"] as any as StylePropertySpecification),\n \"text-pitch-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-pitch-alignment\"] as any as StylePropertySpecification),\n \"text-rotation-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-rotation-alignment\"] as any as StylePropertySpecification),\n \"text-field\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-field\"] as any as StylePropertySpecification),\n \"text-font\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-font\"] as any as StylePropertySpecification),\n \"text-size\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-size\"] as any as StylePropertySpecification),\n \"text-max-width\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-max-width\"] as any as StylePropertySpecification),\n \"text-line-height\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-line-height\"] as any as StylePropertySpecification),\n \"text-letter-spacing\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-letter-spacing\"] as any as StylePropertySpecification),\n \"text-justify\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-justify\"] as any as StylePropertySpecification),\n \"text-radial-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-radial-offset\"] as any as StylePropertySpecification),\n \"text-variable-anchor\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-variable-anchor\"] as any as StylePropertySpecification),\n \"text-variable-anchor-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-variable-anchor-offset\"] as any as StylePropertySpecification),\n \"text-anchor\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-anchor\"] as any as StylePropertySpecification),\n \"text-max-angle\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-max-angle\"] as any as StylePropertySpecification),\n \"text-writing-mode\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-writing-mode\"] as any as StylePropertySpecification),\n \"text-rotate\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-rotate\"] as any as StylePropertySpecification),\n \"text-padding\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-padding\"] as any as StylePropertySpecification),\n \"text-keep-upright\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-keep-upright\"] as any as StylePropertySpecification),\n \"text-transform\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-transform\"] as any as StylePropertySpecification),\n \"text-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-offset\"] as any as StylePropertySpecification),\n \"text-allow-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-allow-overlap\"] as any as StylePropertySpecification),\n \"text-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-overlap\"] as any as StylePropertySpecification),\n \"text-ignore-placement\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-ignore-placement\"] as any as StylePropertySpecification),\n \"text-optional\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-optional\"] as any as StylePropertySpecification),\n});\n\nexport type SymbolPaintProps = {\n \"icon-opacity\": DataDrivenProperty,\n \"icon-color\": DataDrivenProperty,\n \"icon-halo-color\": DataDrivenProperty,\n \"icon-halo-width\": DataDrivenProperty,\n \"icon-halo-blur\": DataDrivenProperty,\n \"icon-translate\": DataConstantProperty<[number, number]>,\n \"icon-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"text-opacity\": DataDrivenProperty,\n \"text-color\": DataDrivenProperty,\n \"text-halo-color\": DataDrivenProperty,\n \"text-halo-width\": DataDrivenProperty,\n \"text-halo-blur\": DataDrivenProperty,\n \"text-translate\": DataConstantProperty<[number, number]>,\n \"text-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n};\n\nexport type SymbolPaintPropsPossiblyEvaluated = {\n \"icon-opacity\": PossiblyEvaluatedPropertyValue,\n \"icon-color\": PossiblyEvaluatedPropertyValue,\n \"icon-halo-color\": PossiblyEvaluatedPropertyValue,\n \"icon-halo-width\": PossiblyEvaluatedPropertyValue,\n \"icon-halo-blur\": PossiblyEvaluatedPropertyValue,\n \"icon-translate\": [number, number],\n \"icon-translate-anchor\": \"map\" | \"viewport\",\n \"text-opacity\": PossiblyEvaluatedPropertyValue,\n \"text-color\": PossiblyEvaluatedPropertyValue,\n \"text-halo-color\": PossiblyEvaluatedPropertyValue,\n \"text-halo-width\": PossiblyEvaluatedPropertyValue,\n \"text-halo-blur\": PossiblyEvaluatedPropertyValue,\n \"text-translate\": [number, number],\n \"text-translate-anchor\": \"map\" | \"viewport\",\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"icon-opacity\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-opacity\"] as any as StylePropertySpecification),\n \"icon-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-color\"] as any as StylePropertySpecification),\n \"icon-halo-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-halo-color\"] as any as StylePropertySpecification),\n \"icon-halo-width\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-halo-width\"] as any as StylePropertySpecification),\n \"icon-halo-blur\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-halo-blur\"] as any as StylePropertySpecification),\n \"icon-translate\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"icon-translate\"] as any as StylePropertySpecification),\n \"icon-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"icon-translate-anchor\"] as any as StylePropertySpecification),\n \"text-opacity\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-opacity\"] as any as StylePropertySpecification),\n \"text-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-color\"] as any as StylePropertySpecification, { runtimeType: ColorType, getOverride: (o) => o.textColor, hasOverride: (o) => !!o.textColor }),\n \"text-halo-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-halo-color\"] as any as StylePropertySpecification),\n \"text-halo-width\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-halo-width\"] as any as StylePropertySpecification),\n \"text-halo-blur\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-halo-blur\"] as any as StylePropertySpecification),\n \"text-translate\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"text-translate\"] as any as StylePropertySpecification),\n \"text-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"text-translate-anchor\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import type {Expression, EvaluationContext, Type, ZoomConstantExpression} from '@maplibre/maplibre-gl-style-spec';\nimport {NullType} from '@maplibre/maplibre-gl-style-spec';\nimport {PossiblyEvaluatedPropertyValue} from './properties';\nimport {register} from '../util/web_worker_transfer';\n\n// This is an internal expression class. It is only used in GL JS and\n// has GL JS dependencies which can break the standalone style-spec module\nexport class FormatSectionOverride implements Expression {\n type: Type;\n defaultValue: PossiblyEvaluatedPropertyValue;\n\n constructor(defaultValue: PossiblyEvaluatedPropertyValue) {\n if (defaultValue.property.overrides === undefined) throw new Error('overrides must be provided to instantiate FormatSectionOverride class');\n this.type = defaultValue.property.overrides ? defaultValue.property.overrides.runtimeType : NullType;\n this.defaultValue = defaultValue;\n }\n\n evaluate(ctx: EvaluationContext) {\n if (ctx.formattedSection) {\n const overrides = this.defaultValue.property.overrides;\n if (overrides && overrides.hasOverride(ctx.formattedSection)) {\n return overrides.getOverride(ctx.formattedSection);\n }\n }\n\n if (ctx.feature && ctx.featureState) {\n return this.defaultValue.evaluate(ctx.feature, ctx.featureState);\n }\n\n return this.defaultValue.property.specification.default;\n }\n\n eachChild(fn: (_: Expression) => void) {\n if (!this.defaultValue.isConstant()) {\n const expr: ZoomConstantExpression<'source'> = (this.defaultValue.value as any);\n fn(expr._styleExpression.expression);\n }\n }\n\n // Cannot be statically evaluated, as the output depends on the evaluation context.\n outputDefined() {\n return false;\n }\n\n serialize() {\n return null;\n }\n}\n\nregister('FormatSectionOverride', FormatSectionOverride, {omit: ['defaultValue']});\n","import {StyleLayer} from '../style_layer';\n\nimport {SymbolBucket, SymbolFeature} from '../../data/bucket/symbol_bucket';\nimport {resolveTokens} from '../../util/resolve_tokens';\nimport properties, {SymbolLayoutPropsPossiblyEvaluated, SymbolPaintPropsPossiblyEvaluated} from './symbol_style_layer_properties.g';\n\nimport {\n Transitionable,\n Transitioning,\n Layout,\n PossiblyEvaluated,\n PossiblyEvaluatedPropertyValue,\n PropertyValue\n} from '../properties';\n\nimport {\n isExpression,\n StyleExpression,\n ZoomConstantExpression,\n ZoomDependentExpression,\n FormattedType,\n typeOf,\n Formatted,\n FormatExpression,\n Literal} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {BucketParameters} from '../../data/bucket';\nimport type {SymbolLayoutProps, SymbolPaintProps} from './symbol_style_layer_properties.g';\nimport type {EvaluationParameters} from '../evaluation_parameters';\nimport type {Expression, Feature, SourceExpression, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport {FormatSectionOverride} from '../format_section_override';\n\nexport class SymbolStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n super.recalculate(parameters, availableImages);\n\n if (this.layout.get('icon-rotation-alignment') === 'auto') {\n if (this.layout.get('symbol-placement') !== 'point') {\n this.layout._values['icon-rotation-alignment'] = 'map';\n } else {\n this.layout._values['icon-rotation-alignment'] = 'viewport';\n }\n }\n\n if (this.layout.get('text-rotation-alignment') === 'auto') {\n if (this.layout.get('symbol-placement') !== 'point') {\n this.layout._values['text-rotation-alignment'] = 'map';\n } else {\n this.layout._values['text-rotation-alignment'] = 'viewport';\n }\n }\n\n // If unspecified, `*-pitch-alignment` inherits `*-rotation-alignment`\n if (this.layout.get('text-pitch-alignment') === 'auto') {\n this.layout._values['text-pitch-alignment'] = this.layout.get('text-rotation-alignment') === 'map' ? 'map' : 'viewport';\n }\n if (this.layout.get('icon-pitch-alignment') === 'auto') {\n this.layout._values['icon-pitch-alignment'] = this.layout.get('icon-rotation-alignment');\n }\n\n if (this.layout.get('symbol-placement') === 'point') {\n const writingModes = this.layout.get('text-writing-mode');\n if (writingModes) {\n // remove duplicates, preserving order\n const deduped = [];\n for (const m of writingModes) {\n if (deduped.indexOf(m) < 0) deduped.push(m);\n }\n this.layout._values['text-writing-mode'] = deduped;\n } else {\n this.layout._values['text-writing-mode'] = ['horizontal'];\n }\n }\n\n this._setPaintOverrides();\n }\n\n getValueAndResolveTokens(name: any, feature: Feature, canonical: CanonicalTileID, availableImages: Array) {\n const value = this.layout.get(name).evaluate(feature, {}, canonical, availableImages);\n const unevaluated = this._unevaluatedLayout._values[name];\n if (!unevaluated.isDataDriven() && !isExpression(unevaluated.value) && value) {\n return resolveTokens(feature.properties, value);\n }\n\n return value;\n }\n\n createBucket(parameters: BucketParameters) {\n return new SymbolBucket(parameters);\n }\n\n queryRadius(): number {\n return 0;\n }\n\n queryIntersectsFeature(): boolean {\n throw new Error('Should take a different path in FeatureIndex');\n }\n\n _setPaintOverrides() {\n for (const overridable of properties.paint.overridableProperties) {\n if (!SymbolStyleLayer.hasPaintOverride(this.layout, overridable)) {\n continue;\n }\n const overriden = this.paint.get(overridable as keyof SymbolPaintPropsPossiblyEvaluated) as PossiblyEvaluatedPropertyValue;\n const override = new FormatSectionOverride(overriden);\n const styleExpression = new StyleExpression(override, overriden.property.specification);\n let expression = null;\n if (overriden.value.kind === 'constant' || overriden.value.kind === 'source') {\n expression = new ZoomConstantExpression('source', styleExpression) as SourceExpression;\n } else {\n expression = new ZoomDependentExpression('composite',\n styleExpression,\n overriden.value.zoomStops);\n }\n this.paint._values[overridable] = new PossiblyEvaluatedPropertyValue(overriden.property,\n expression,\n overriden.parameters);\n }\n }\n\n _handleOverridablePaintPropertyUpdate(name: string, oldValue: PropertyValue, newValue: PropertyValue): boolean {\n if (!this.layout || oldValue.isDataDriven() || newValue.isDataDriven()) {\n return false;\n }\n return SymbolStyleLayer.hasPaintOverride(this.layout, name);\n }\n\n static hasPaintOverride(layout: PossiblyEvaluated, propertyName: string): boolean {\n const textField = layout.get('text-field');\n const property = properties.paint.properties[propertyName];\n let hasOverrides = false;\n\n const checkSections = (sections) => {\n for (const section of sections) {\n if (property.overrides && property.overrides.hasOverride(section)) {\n hasOverrides = true;\n return;\n }\n }\n };\n\n if (textField.value.kind === 'constant' && textField.value.value instanceof Formatted) {\n checkSections(textField.value.value.sections);\n } else if (textField.value.kind === 'source') {\n\n const checkExpression = (expression: Expression) => {\n if (hasOverrides) return;\n\n if (expression instanceof Literal && typeOf(expression.value) === FormattedType) {\n const formatted: Formatted = (expression.value as any);\n checkSections(formatted.sections);\n } else if (expression instanceof FormatExpression) {\n checkSections(expression.sections);\n } else {\n expression.eachChild(checkExpression);\n }\n };\n\n const expr: ZoomConstantExpression<'source'> = (textField.value as any);\n if (expr._styleExpression) {\n checkExpression(expr._styleExpression.expression);\n }\n }\n\n return hasOverrides;\n }\n}\n\nexport type SymbolPadding = [number, number, number, number];\n\nexport function getIconPadding(layout: PossiblyEvaluated, feature: SymbolFeature, canonical: CanonicalTileID, pixelRatio = 1): SymbolPadding {\n // Support text-padding in addition to icon-padding? Unclear how to apply asymmetric text-padding to the radius for collision circles.\n const result = layout.get('icon-padding').evaluate(feature, {}, canonical);\n const values = result && result.values;\n\n return [\n values[0] * pixelRatio,\n values[1] * pixelRatio,\n values[2] * pixelRatio,\n values[3] * pixelRatio,\n ];\n}\n","/**\n * Replace tokens in a string template with values in an object\n *\n * @param properties - a key/value relationship between tokens and replacements\n * @param text - the template string\n * @returns the template with tokens replaced\n */\nexport function resolveTokens(\n properties: {\n readonly [x: string]: unknown;\n } | null,\n text: string\n): string {\n return text.replace(/{([^{}]+)}/g, (match, key: string) => {\n return properties && key in properties ? String(properties[key]) : '';\n });\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type BackgroundPaintProps = {\n \"background-color\": DataConstantProperty,\n \"background-pattern\": CrossFadedProperty,\n \"background-opacity\": DataConstantProperty,\n};\n\nexport type BackgroundPaintPropsPossiblyEvaluated = {\n \"background-color\": Color,\n \"background-pattern\": CrossFaded,\n \"background-opacity\": number,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"background-color\": new DataConstantProperty(styleSpec[\"paint_background\"][\"background-color\"] as any as StylePropertySpecification),\n \"background-pattern\": new CrossFadedProperty(styleSpec[\"paint_background\"][\"background-pattern\"] as any as StylePropertySpecification),\n \"background-opacity\": new DataConstantProperty(styleSpec[\"paint_background\"][\"background-opacity\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport properties, {BackgroundPaintPropsPossiblyEvaluated} from './background_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {BackgroundPaintProps} from './background_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport class BackgroundStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type RasterPaintProps = {\n \"raster-opacity\": DataConstantProperty,\n \"raster-hue-rotate\": DataConstantProperty,\n \"raster-brightness-min\": DataConstantProperty,\n \"raster-brightness-max\": DataConstantProperty,\n \"raster-saturation\": DataConstantProperty,\n \"raster-contrast\": DataConstantProperty,\n \"raster-resampling\": DataConstantProperty<\"linear\" | \"nearest\">,\n \"raster-fade-duration\": DataConstantProperty,\n};\n\nexport type RasterPaintPropsPossiblyEvaluated = {\n \"raster-opacity\": number,\n \"raster-hue-rotate\": number,\n \"raster-brightness-min\": number,\n \"raster-brightness-max\": number,\n \"raster-saturation\": number,\n \"raster-contrast\": number,\n \"raster-resampling\": \"linear\" | \"nearest\",\n \"raster-fade-duration\": number,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"raster-opacity\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-opacity\"] as any as StylePropertySpecification),\n \"raster-hue-rotate\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-hue-rotate\"] as any as StylePropertySpecification),\n \"raster-brightness-min\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-brightness-min\"] as any as StylePropertySpecification),\n \"raster-brightness-max\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-brightness-max\"] as any as StylePropertySpecification),\n \"raster-saturation\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-saturation\"] as any as StylePropertySpecification),\n \"raster-contrast\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-contrast\"] as any as StylePropertySpecification),\n \"raster-resampling\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-resampling\"] as any as StylePropertySpecification),\n \"raster-fade-duration\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-fade-duration\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport properties, {RasterPaintPropsPossiblyEvaluated} from './raster_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {RasterPaintProps} from './raster_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport class RasterStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n}\n","import {StyleLayer} from '../style_layer';\nimport type {Map} from '../../ui/map';\nimport {mat4} from 'gl-matrix';\nimport {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * @param gl - The map's gl context.\n * @param matrix - The map's camera matrix. It projects spherical mercator\n * coordinates to gl coordinates. The spherical mercator coordinate `[0, 0]` represents the\n * top left corner of the mercator world and `[1, 1]` represents the bottom right corner. When\n * the `renderingMode` is `\"3d\"`, the z coordinate is conformal. A box with identical x, y, and z\n * lengths in mercator units would be rendered as a cube. {@link MercatorCoordinate.fromLngLat}\n * can be used to project a `LngLat` to a mercator coordinate.\n */\ntype CustomRenderMethod = (gl: WebGLRenderingContext|WebGL2RenderingContext, matrix: mat4) => void;\n\n/**\n * Interface for custom style layers. This is a specification for\n * implementers to model: it is not an exported method or class.\n *\n * Custom layers allow a user to render directly into the map's GL context using the map's camera.\n * These layers can be added between any regular layers using {@link Map#addLayer}.\n *\n * Custom layers must have a unique `id` and must have the `type` of `\"custom\"`.\n * They must implement `render` and may implement `prerender`, `onAdd` and `onRemove`.\n * They can trigger rendering using {@link Map#triggerRepaint}\n * and they should appropriately handle {@link MapContextEvent} with `webglcontextlost` and `webglcontextrestored`.\n *\n * The `renderingMode` property controls whether the layer is treated as a `\"2d\"` or `\"3d\"` map layer. Use:\n * - `\"renderingMode\": \"3d\"` to use the depth buffer and share it with other layers\n * - `\"renderingMode\": \"2d\"` to add a layer with no depth. If you need to use the depth buffer for a `\"2d\"` layer you must use an offscreen\n * framebuffer and {@link CustomLayerInterface#prerender}\n *\n * @example\n * Custom layer implemented as ES6 class\n * ```ts\n * class NullIslandLayer {\n * constructor() {\n * this.id = 'null-island';\n * this.type = 'custom';\n * this.renderingMode = '2d';\n * }\n *\n * onAdd(map, gl) {\n * const vertexSource = `\n * uniform mat4 u_matrix;\n * void main() {\n * gl_Position = u_matrix * vec4(0.5, 0.5, 0.0, 1.0);\n * gl_PointSize = 20.0;\n * }`;\n *\n * const fragmentSource = `\n * void main() {\n * fragColor = vec4(1.0, 0.0, 0.0, 1.0);\n * }`;\n *\n * const vertexShader = gl.createShader(gl.VERTEX_SHADER);\n * gl.shaderSource(vertexShader, vertexSource);\n * gl.compileShader(vertexShader);\n * const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);\n * gl.shaderSource(fragmentShader, fragmentSource);\n * gl.compileShader(fragmentShader);\n *\n * this.program = gl.createProgram();\n * gl.attachShader(this.program, vertexShader);\n * gl.attachShader(this.program, fragmentShader);\n * gl.linkProgram(this.program);\n * }\n *\n * render(gl, matrix) {\n * gl.useProgram(this.program);\n * gl.uniformMatrix4fv(gl.getUniformLocation(this.program, \"u_matrix\"), false, matrix);\n * gl.drawArrays(gl.POINTS, 0, 1);\n * }\n * }\n *\n * map.on('load', function() {\n * map.addLayer(new NullIslandLayer());\n * });\n * ```\n */\nexport interface CustomLayerInterface {\n /**\n * A unique layer id.\n */\n id: string;\n /**\n * The layer's type. Must be `\"custom\"`.\n */\n type: 'custom';\n /**\n * Either `\"2d\"` or `\"3d\"`. Defaults to `\"2d\"`.\n */\n renderingMode?: '2d' | '3d';\n /**\n * Called during a render frame allowing the layer to draw into the GL context.\n *\n * The layer can assume blending and depth state is set to allow the layer to properly\n * blend and clip other layers. The layer cannot make any other assumptions about the\n * current GL state.\n *\n * If the layer needs to render to a texture, it should implement the `prerender` method\n * to do this and only use the `render` method for drawing directly into the main framebuffer.\n *\n * The blend function is set to `gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA)`. This expects\n * colors to be provided in premultiplied alpha form where the `r`, `g` and `b` values are already\n * multiplied by the `a` value. If you are unable to provide colors in premultiplied form you\n * may want to change the blend function to\n * `gl.blendFuncSeparate(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA, gl.ONE, gl.ONE_MINUS_SRC_ALPHA)`.\n */\n render: CustomRenderMethod;\n /**\n * Optional method called during a render frame to allow a layer to prepare resources or render into a texture.\n *\n * The layer cannot make any assumptions about the current GL state and must bind a framebuffer before rendering.\n */\n prerender?: CustomRenderMethod;\n /**\n * Optional method called when the layer has been added to the Map with {@link Map#addLayer}. This\n * gives the layer a chance to initialize gl resources and register event listeners.\n *\n * @param map - The Map this custom layer was just added to.\n * @param gl - The gl context for the map.\n */\n onAdd?(map: Map, gl: WebGLRenderingContext | WebGL2RenderingContext): void;\n /**\n * Optional method called when the layer has been removed from the Map with {@link Map#removeLayer}. This\n * gives the layer a chance to clean up gl resources and event listeners.\n *\n * @param map - The Map this custom layer was just added to.\n * @param gl - The gl context for the map.\n */\n onRemove?(map: Map, gl: WebGLRenderingContext | WebGL2RenderingContext): void;\n}\n\nexport function validateCustomStyleLayer(layerObject: CustomLayerInterface) {\n const errors = [];\n const id = layerObject.id;\n\n if (id === undefined) {\n errors.push({\n message: `layers.${id}: missing required property \"id\"`\n });\n }\n\n if (layerObject.render === undefined) {\n errors.push({\n message: `layers.${id}: missing required method \"render\"`\n });\n }\n\n if (layerObject.renderingMode &&\n layerObject.renderingMode !== '2d' &&\n layerObject.renderingMode !== '3d') {\n errors.push({\n message: `layers.${id}: property \"renderingMode\" must be either \"2d\" or \"3d\"`\n });\n }\n\n return errors;\n}\n\nexport class CustomStyleLayer extends StyleLayer {\n\n implementation: CustomLayerInterface;\n\n constructor(implementation: CustomLayerInterface) {\n super(implementation, {});\n this.implementation = implementation;\n }\n\n is3D() {\n return this.implementation.renderingMode === '3d';\n }\n\n hasOffscreenPass() {\n return this.implementation.prerender !== undefined;\n }\n\n recalculate() {}\n updateTransitions() {}\n hasTransition() { return false; }\n\n serialize(): LayerSpecification {\n throw new Error('Custom layers cannot be serialized');\n }\n\n onAdd = (map: Map) => {\n if (this.implementation.onAdd) {\n this.implementation.onAdd(map, map.painter.context.gl);\n }\n };\n\n onRemove = (map: Map) => {\n if (this.implementation.onRemove) {\n this.implementation.onRemove(map, map.painter.context.gl);\n }\n };\n}\n","import {CircleStyleLayer} from './style_layer/circle_style_layer';\nimport {HeatmapStyleLayer} from './style_layer/heatmap_style_layer';\nimport {HillshadeStyleLayer} from './style_layer/hillshade_style_layer';\nimport {FillStyleLayer} from './style_layer/fill_style_layer';\nimport {FillExtrusionStyleLayer} from './style_layer/fill_extrusion_style_layer';\nimport {LineStyleLayer} from './style_layer/line_style_layer';\nimport {SymbolStyleLayer} from './style_layer/symbol_style_layer';\nimport {BackgroundStyleLayer} from './style_layer/background_style_layer';\nimport {RasterStyleLayer} from './style_layer/raster_style_layer';\nimport {CustomStyleLayer, type CustomLayerInterface} from './style_layer/custom_style_layer';\n\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport function createStyleLayer(layer: LayerSpecification | CustomLayerInterface) {\n if (layer.type === 'custom') {\n return new CustomStyleLayer(layer);\n }\n switch (layer.type) {\n case 'background':\n return new BackgroundStyleLayer(layer);\n case 'circle':\n return new CircleStyleLayer(layer);\n case 'fill':\n return new FillStyleLayer(layer);\n case 'fill-extrusion':\n return new FillExtrusionStyleLayer(layer);\n case 'heatmap':\n return new HeatmapStyleLayer(layer);\n case 'hillshade':\n return new HillshadeStyleLayer(layer);\n case 'line':\n return new LineStyleLayer(layer);\n case 'raster':\n return new RasterStyleLayer(layer);\n case 'symbol':\n return new SymbolStyleLayer(layer);\n }\n}\n\n","import {SpriteSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * Takes a SpriteSpecification value and returns it in its array form. If `undefined` is passed as an input value, an\n * empty array is returned.\n * duplicated entries with identical id/url will be removed in returned array\n * @param sprite - optional sprite to coerce\n * @returns an empty array in case `undefined` is passed; id-url pairs otherwise\n */\nexport function coerceSpriteToArray(sprite?: SpriteSpecification): {id: string; url: string}[] {\n const resultArray: {id: string; url: string}[] = [];\n\n if (typeof sprite === 'string') {\n resultArray.push({id: 'default', url: sprite});\n } else if (sprite && sprite.length > 0) {\n const dedupArray: string[] = [];\n for (const {id, url} of sprite) {\n const key = `${id}${url}`;\n if (dedupArray.indexOf(key) === -1) {\n dedupArray.push(key);\n resultArray.push({id, url});\n }\n }\n }\n\n return resultArray;\n\n}\n","import {getJSON} from '../util/ajax';\nimport {ImageRequest} from '../util/image_request';\nimport {ResourceType} from '../util/request_manager';\n\nimport {browser} from '../util/browser';\nimport {coerceSpriteToArray} from '../util/style';\n\nimport type {SpriteSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {StyleImage} from './style_image';\nimport type {RequestManager} from '../util/request_manager';\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\n\nexport function loadSprite(\n originalSprite: SpriteSpecification,\n requestManager: RequestManager,\n pixelRatio: number,\n callback: Callback<{[spriteName: string]: {[id: string]: StyleImage}}>\n): Cancelable {\n const spriteArray = coerceSpriteToArray(originalSprite);\n const spriteArrayLength = spriteArray.length;\n const format = pixelRatio > 1 ? '@2x' : '';\n\n const combinedRequestsMap: {[requestKey: string]: Cancelable} = {};\n const jsonsMap: {[id: string]: any} = {};\n const imagesMap: {[id: string]: (HTMLImageElement | ImageBitmap)} = {};\n\n for (const {id, url} of spriteArray) {\n const jsonRequestParameters = requestManager.transformRequest(requestManager.normalizeSpriteURL(url, format, '.json'), ResourceType.SpriteJSON);\n const jsonRequestKey = `${id}_${jsonRequestParameters.url}`; // use id_url as requestMap key to make sure it is unique\n combinedRequestsMap[jsonRequestKey] = getJSON(jsonRequestParameters, (err?: Error | null, data?: any | null) => {\n delete combinedRequestsMap[jsonRequestKey];\n jsonsMap[id] = data;\n doOnceCompleted(callback, jsonsMap, imagesMap, err, spriteArrayLength);\n });\n\n const imageRequestParameters = requestManager.transformRequest(requestManager.normalizeSpriteURL(url, format, '.png'), ResourceType.SpriteImage);\n const imageRequestKey = `${id}_${imageRequestParameters.url}`; // use id_url as requestMap key to make sure it is unique\n combinedRequestsMap[imageRequestKey] = ImageRequest.getImage(imageRequestParameters, (err, img) => {\n delete combinedRequestsMap[imageRequestKey];\n imagesMap[id] = img;\n doOnceCompleted(callback, jsonsMap, imagesMap, err, spriteArrayLength);\n });\n }\n\n return {\n cancel() {\n for (const requst of Object.values(combinedRequestsMap)) {\n requst.cancel();\n }\n }\n };\n}\n\n/**\n * @param callbackFunc - the callback function (both erro and success)\n * @param jsonsMap - JSON data map\n * @param imagesMap - image data map\n * @param err - error object\n * @param expectedResultCounter - number of expected JSON or Image results when everything is finished, respectively.\n */\nfunction doOnceCompleted(\n callbackFunc:Callback<{[spriteName: string]: {[id: string]: StyleImage}}>,\n jsonsMap:{[id: string]: any},\n imagesMap:{[id: string]: (HTMLImageElement | ImageBitmap)},\n err: Error,\n expectedResultCounter: number): void {\n\n if (err) {\n callbackFunc(err);\n return;\n }\n\n if (expectedResultCounter !== Object.values(jsonsMap).length || expectedResultCounter !== Object.values(imagesMap).length) {\n // not done yet, nothing to do\n return;\n }\n\n const result = {} as {[spriteName: string]: {[id: string]: StyleImage}};\n for (const spriteName in jsonsMap) {\n result[spriteName] = {};\n\n const context = browser.getImageCanvasContext(imagesMap[spriteName]);\n const json = jsonsMap[spriteName];\n\n for (const id in json) {\n const {width, height, x, y, sdf, pixelRatio, stretchX, stretchY, content} = json[id];\n const spriteData = {width, height, x, y, context};\n result[spriteName][id] = {data: null, pixelRatio, sdf, stretchX, stretchY, content, spriteData};\n }\n }\n\n callbackFunc(null, result);\n}\n","import type {Context} from '../gl/context';\nimport type {RGBAImage, AlphaImage} from '../util/image';\nimport {isImageBitmap} from '../util/util';\n\nexport type TextureFormat = WebGLRenderingContextBase['RGBA'] | WebGLRenderingContextBase['ALPHA'];\nexport type TextureFilter = WebGLRenderingContextBase['LINEAR'] | WebGLRenderingContextBase['LINEAR_MIPMAP_NEAREST'] | WebGLRenderingContextBase['NEAREST'];\nexport type TextureWrap = WebGLRenderingContextBase['REPEAT'] | WebGLRenderingContextBase['CLAMP_TO_EDGE'] | WebGLRenderingContextBase['MIRRORED_REPEAT'];\n\ntype EmptyImage = {\n width: number;\n height: number;\n data: null;\n};\n\ntype DataTextureImage = RGBAImage | AlphaImage | EmptyImage;\nexport type TextureImage = TexImageSource | DataTextureImage;\n\n/**\n * @internal\n * A `Texture` GL related object\n */\nexport class Texture {\n context: Context;\n size: [number, number];\n texture: WebGLTexture;\n format: TextureFormat;\n filter: TextureFilter;\n wrap: TextureWrap;\n useMipmap: boolean;\n\n constructor(context: Context, image: TextureImage, format: TextureFormat, options?: {\n premultiply?: boolean;\n useMipmap?: boolean;\n } | null) {\n this.context = context;\n this.format = format;\n this.texture = context.gl.createTexture();\n this.update(image, options);\n }\n\n update(image: TextureImage, options?: {\n premultiply?: boolean;\n useMipmap?: boolean;\n } | null, position?: {\n x: number;\n y: number;\n }) {\n const {width, height} = image as {width: number; height: number};\n const resize = (!this.size || this.size[0] !== width || this.size[1] !== height) && !position;\n const {context} = this;\n const {gl} = context;\n\n this.useMipmap = Boolean(options && options.useMipmap);\n gl.bindTexture(gl.TEXTURE_2D, this.texture);\n\n context.pixelStoreUnpackFlipY.set(false);\n context.pixelStoreUnpack.set(1);\n context.pixelStoreUnpackPremultiplyAlpha.set(this.format === gl.RGBA && (!options || options.premultiply !== false));\n\n if (resize) {\n this.size = [width, height];\n\n if (image instanceof HTMLImageElement || image instanceof HTMLCanvasElement || image instanceof HTMLVideoElement || image instanceof ImageData || isImageBitmap(image)) {\n gl.texImage2D(gl.TEXTURE_2D, 0, this.format, this.format, gl.UNSIGNED_BYTE, image);\n } else {\n gl.texImage2D(gl.TEXTURE_2D, 0, this.format, width, height, 0, this.format, gl.UNSIGNED_BYTE, (image as DataTextureImage).data);\n }\n\n } else {\n const {x, y} = position || {x: 0, y: 0};\n if (image instanceof HTMLImageElement || image instanceof HTMLCanvasElement || image instanceof HTMLVideoElement || image instanceof ImageData || isImageBitmap(image)) {\n gl.texSubImage2D(gl.TEXTURE_2D, 0, x, y, gl.RGBA, gl.UNSIGNED_BYTE, image);\n } else {\n gl.texSubImage2D(gl.TEXTURE_2D, 0, x, y, width, height, gl.RGBA, gl.UNSIGNED_BYTE, (image as DataTextureImage).data);\n }\n }\n\n if (this.useMipmap && this.isSizePowerOfTwo()) {\n gl.generateMipmap(gl.TEXTURE_2D);\n }\n }\n\n bind(filter: TextureFilter, wrap: TextureWrap, minFilter?: TextureFilter | null) {\n const {context} = this;\n const {gl} = context;\n gl.bindTexture(gl.TEXTURE_2D, this.texture);\n\n if (minFilter === gl.LINEAR_MIPMAP_NEAREST && !this.isSizePowerOfTwo()) {\n minFilter = gl.LINEAR;\n }\n\n if (filter !== this.filter) {\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, filter);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, minFilter || filter);\n this.filter = filter;\n }\n\n if (wrap !== this.wrap) {\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, wrap);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, wrap);\n this.wrap = wrap;\n }\n }\n\n isSizePowerOfTwo() {\n return this.size[0] === this.size[1] && (Math.log(this.size[0]) / Math.LN2) % 1 === 0;\n }\n\n destroy() {\n const {gl} = this.context;\n gl.deleteTexture(this.texture);\n this.texture = null;\n }\n}\n","import {RGBAImage} from '../util/image';\n\nimport type {Map} from '../ui/map';\n\n/**\n * The sprite data\n */\nexport type SpriteOnDemandStyleImage = {\n width: number;\n height: number;\n x: number;\n y: number;\n context: CanvasRenderingContext2D;\n};\n\n/**\n * The style's image metadata\n */\nexport type StyleImageData = {\n data: RGBAImage;\n version?: number;\n hasRenderCallback?: boolean;\n userImage?: StyleImageInterface;\n spriteData?: SpriteOnDemandStyleImage;\n};\n\n/**\n * The style's image metadata\n */\nexport type StyleImageMetadata = {\n /**\n * The ratio of pixels in the image to physical pixels on the screen\n */\n pixelRatio: number;\n /**\n * Whether the image should be interpreted as an SDF image\n */\n sdf: boolean;\n /**\n * If `icon-text-fit` is used in a layer with this image, this option defines the part(s) of the image that can be stretched horizontally.\n */\n stretchX?: Array<[number, number]>;\n /**\n * If `icon-text-fit` is used in a layer with this image, this option defines the part(s) of the image that can be stretched vertically.\n */\n stretchY?: Array<[number, number]>;\n /**\n * If `icon-text-fit` is used in a layer with this image, this option defines the part of the image that can be covered by the content in `text-field`.\n */\n content?: [number, number, number, number];\n};\n\n/**\n * the style's image, including data and metedata\n */\nexport type StyleImage = StyleImageData & StyleImageMetadata;\n\n/**\n * Interface for dynamically generated style images. This is a specification for\n * implementers to model: it is not an exported method or class.\n *\n * Images implementing this interface can be redrawn for every frame. They can be used to animate\n * icons and patterns or make them respond to user input. Style images can implement a\n * {@link StyleImageInterface#render} method. The method is called every frame and\n * can be used to update the image.\n *\n * @see [Add an animated icon to the map.](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-animated/)\n *\n * @example\n * ```ts\n * let flashingSquare = {\n * width: 64,\n * height: 64,\n * data: new Uint8Array(64 * 64 * 4),\n *\n * onAdd: function(map) {\n * this.map = map;\n * },\n *\n * render: function() {\n * // keep repainting while the icon is on the map\n * this.map.triggerRepaint();\n *\n * // alternate between black and white based on the time\n * let value = Math.round(Date.now() / 1000) % 2 === 0 ? 255 : 0;\n *\n * // check if image needs to be changed\n * if (value !== this.previousValue) {\n * this.previousValue = value;\n *\n * let bytesPerPixel = 4;\n * for (let x = 0; x < this.width; x++) {\n * for (let y = 0; y < this.height; y++) {\n * let offset = (y * this.width + x) * bytesPerPixel;\n * this.data[offset + 0] = value;\n * this.data[offset + 1] = value;\n * this.data[offset + 2] = value;\n * this.data[offset + 3] = 255;\n * }\n * }\n *\n * // return true to indicate that the image changed\n * return true;\n * }\n * }\n * }\n *\n * map.addImage('flashing_square', flashingSquare);\n * ```\n */\n\nexport interface StyleImageInterface {\n width: number;\n height: number;\n data: Uint8Array | Uint8ClampedArray;\n /**\n * This method is called once before every frame where the icon will be used.\n * The method can optionally update the image's `data` member with a new image.\n *\n * If the method updates the image it must return `true` to commit the change.\n * If the method returns `false` or nothing the image is assumed to not have changed.\n *\n * If updates are infrequent it maybe easier to use {@link Map#updateImage} to update\n * the image instead of implementing this method.\n *\n * @returns `true` if this method updated the image. `false` if the image was not changed.\n */\n render?: () => boolean;\n /**\n * Optional method called when the layer has been added to the Map with {@link Map#addImage}.\n *\n * @param map - The Map this custom layer was just added to.\n */\n onAdd?: (map: Map, id: string) => void;\n /**\n * Optional method called when the icon is removed from the map with {@link Map#removeImage}.\n * This gives the image a chance to clean up resources and event listeners.\n */\n onRemove?: () => void;\n}\n\nexport function renderStyleImage(image: StyleImage) {\n const {userImage} = image;\n if (userImage && userImage.render) {\n const updated = userImage.render();\n if (updated) {\n image.data.replace(new Uint8Array(userImage.data.buffer));\n return true;\n }\n }\n return false;\n}\n","/* eslint-disable key-spacing */\nimport potpack from 'potpack';\n\nimport {Event, ErrorEvent, Evented} from '../util/evented';\nimport {RGBAImage} from '../util/image';\nimport {ImagePosition} from './image_atlas';\nimport {Texture} from './texture';\nimport {renderStyleImage} from '../style/style_image';\nimport {warnOnce} from '../util/util';\n\nimport type {StyleImage} from '../style/style_image';\nimport type {Context} from '../gl/context';\nimport type {PotpackBox} from 'potpack';\nimport type {Callback} from '../types/callback';\n\ntype Pattern = {\n bin: PotpackBox;\n position: ImagePosition;\n};\n\n// When copied into the atlas texture, image data is padded by one pixel on each side. Icon\n// images are padded with fully transparent pixels, while pattern images are padded with a\n// copy of the image data wrapped from the opposite side. In both cases, this ensures the\n// correct behavior of GL_LINEAR texture sampling mode.\nconst padding = 1;\n\n/*\n ImageManager does three things:\n\n 1. Tracks requests for icon images from tile workers and sends responses when the requests are fulfilled.\n 2. Builds a texture atlas for pattern images.\n 3. Rerenders renderable images once per frame\n\n These are disparate responsibilities and should eventually be handled by different classes. When we implement\n data-driven support for `*-pattern`, we'll likely use per-bucket pattern atlases, and that would be a good time\n to refactor this.\n*/\nexport class ImageManager extends Evented {\n images: {[_: string]: StyleImage};\n updatedImages: {[_: string]: boolean};\n callbackDispatchedThisFrame: {[_: string]: boolean};\n loaded: boolean;\n requestors: Array<{\n ids: Array;\n callback: Callback<{[_: string]: StyleImage}>;\n }>;\n\n patterns: {[_: string]: Pattern};\n atlasImage: RGBAImage;\n atlasTexture: Texture;\n dirty: boolean;\n\n constructor() {\n super();\n this.images = {};\n this.updatedImages = {};\n this.callbackDispatchedThisFrame = {};\n this.loaded = false;\n this.requestors = [];\n\n this.patterns = {};\n this.atlasImage = new RGBAImage({width: 1, height: 1});\n this.dirty = true;\n }\n\n isLoaded() {\n return this.loaded;\n }\n\n setLoaded(loaded: boolean) {\n if (this.loaded === loaded) {\n return;\n }\n\n this.loaded = loaded;\n\n if (loaded) {\n for (const {ids, callback} of this.requestors) {\n this._notify(ids, callback);\n }\n this.requestors = [];\n }\n }\n\n getImage(id: string): StyleImage {\n const image = this.images[id];\n\n // Extract sprite image data on demand\n if (image && !image.data && image.spriteData) {\n const spriteData = image.spriteData;\n image.data = new RGBAImage({\n width: spriteData.width,\n height: spriteData.height\n }, spriteData.context.getImageData(\n spriteData.x,\n spriteData.y,\n spriteData.width,\n spriteData.height).data);\n image.spriteData = null;\n }\n\n return image;\n }\n\n addImage(id: string, image: StyleImage) {\n if (this.images[id]) throw new Error(`Image id ${id} already exist, use updateImage instead`);\n if (this._validate(id, image)) {\n this.images[id] = image;\n }\n }\n\n _validate(id: string, image: StyleImage) {\n let valid = true;\n const data = image.data || image.spriteData;\n if (!this._validateStretch(image.stretchX, data && data.width)) {\n this.fire(new ErrorEvent(new Error(`Image \"${id}\" has invalid \"stretchX\" value`)));\n valid = false;\n }\n if (!this._validateStretch(image.stretchY, data && data.height)) {\n this.fire(new ErrorEvent(new Error(`Image \"${id}\" has invalid \"stretchY\" value`)));\n valid = false;\n }\n if (!this._validateContent(image.content, image)) {\n this.fire(new ErrorEvent(new Error(`Image \"${id}\" has invalid \"content\" value`)));\n valid = false;\n }\n return valid;\n }\n\n _validateStretch(stretch: Array<[number, number]>, size: number) {\n if (!stretch) return true;\n let last = 0;\n for (const part of stretch) {\n if (part[0] < last || part[1] < part[0] || size < part[1]) return false;\n last = part[1];\n }\n return true;\n }\n\n _validateContent(content: [number, number, number, number], image: StyleImage) {\n if (!content) return true;\n if (content.length !== 4) return false;\n const spriteData = image.spriteData;\n const width = (spriteData && spriteData.width) || image.data.width;\n const height = (spriteData && spriteData.height) || image.data.height;\n if (content[0] < 0 || width < content[0]) return false;\n if (content[1] < 0 || height < content[1]) return false;\n if (content[2] < 0 || width < content[2]) return false;\n if (content[3] < 0 || height < content[3]) return false;\n if (content[2] < content[0]) return false;\n if (content[3] < content[1]) return false;\n return true;\n }\n\n updateImage(id: string, image: StyleImage, validate = true) {\n const oldImage = this.getImage(id);\n if (validate && (oldImage.data.width !== image.data.width || oldImage.data.height !== image.data.height)) {\n throw new Error(`size mismatch between old image (${oldImage.data.width}x${oldImage.data.height}) and new image (${image.data.width}x${image.data.height}).`);\n }\n image.version = oldImage.version + 1;\n this.images[id] = image;\n this.updatedImages[id] = true;\n }\n\n removeImage(id: string) {\n const image = this.images[id];\n delete this.images[id];\n delete this.patterns[id];\n\n if (image.userImage && image.userImage.onRemove) {\n image.userImage.onRemove();\n }\n }\n\n listImages(): Array {\n return Object.keys(this.images);\n }\n\n getImages(ids: Array, callback: Callback<{[_: string]: StyleImage}>) {\n // If the sprite has been loaded, or if all the icon dependencies are already present\n // (i.e. if they've been added via runtime styling), then notify the requestor immediately.\n // Otherwise, delay notification until the sprite is loaded. At that point, if any of the\n // dependencies are still unavailable, we'll just assume they are permanently missing.\n let hasAllDependencies = true;\n if (!this.isLoaded()) {\n for (const id of ids) {\n if (!this.images[id]) {\n hasAllDependencies = false;\n }\n }\n }\n if (this.isLoaded() || hasAllDependencies) {\n this._notify(ids, callback);\n } else {\n this.requestors.push({ids, callback});\n }\n }\n\n _notify(ids: Array, callback: Callback<{[_: string]: StyleImage}>) {\n const response = {};\n\n for (const id of ids) {\n let image = this.getImage(id);\n\n if (!image) {\n this.fire(new Event('styleimagemissing', {id}));\n //Try to acquire image again in case styleimagemissing has populated it\n image = this.getImage(id);\n }\n\n if (image) {\n // Clone the image so that our own copy of its ArrayBuffer doesn't get transferred.\n response[id] = {\n data: image.data.clone(),\n pixelRatio: image.pixelRatio,\n sdf: image.sdf,\n version: image.version,\n stretchX: image.stretchX,\n stretchY: image.stretchY,\n content: image.content,\n hasRenderCallback: Boolean(image.userImage && image.userImage.render)\n };\n } else {\n warnOnce(`Image \"${id}\" could not be loaded. Please make sure you have added the image with map.addImage() or a \"sprite\" property in your style. You can provide missing images by listening for the \"styleimagemissing\" map event.`);\n }\n }\n\n callback(null, response);\n }\n\n // Pattern stuff\n\n getPixelSize() {\n const {width, height} = this.atlasImage;\n return {width, height};\n }\n\n getPattern(id: string): ImagePosition {\n const pattern = this.patterns[id];\n\n const image = this.getImage(id);\n if (!image) {\n return null;\n }\n\n if (pattern && pattern.position.version === image.version) {\n return pattern.position;\n }\n\n if (!pattern) {\n const w = image.data.width + padding * 2;\n const h = image.data.height + padding * 2;\n const bin = {w, h, x: 0, y: 0};\n const position = new ImagePosition(bin, image);\n this.patterns[id] = {bin, position};\n } else {\n pattern.position.version = image.version;\n }\n\n this._updatePatternAtlas();\n\n return this.patterns[id].position;\n }\n\n bind(context: Context) {\n const gl = context.gl;\n if (!this.atlasTexture) {\n this.atlasTexture = new Texture(context, this.atlasImage, gl.RGBA);\n } else if (this.dirty) {\n this.atlasTexture.update(this.atlasImage);\n this.dirty = false;\n }\n\n this.atlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n }\n\n _updatePatternAtlas() {\n const bins = [];\n for (const id in this.patterns) {\n bins.push(this.patterns[id].bin);\n }\n\n const {w, h} = potpack(bins);\n\n const dst = this.atlasImage;\n dst.resize({width: w || 1, height: h || 1});\n\n for (const id in this.patterns) {\n const {bin} = this.patterns[id];\n const x = bin.x + padding;\n const y = bin.y + padding;\n const src = this.getImage(id).data;\n const w = src.width;\n const h = src.height;\n\n RGBAImage.copy(src, dst, {x: 0, y: 0}, {x, y}, {width: w, height: h});\n\n // Add 1 pixel wrapped padding on each side of the image.\n RGBAImage.copy(src, dst, {x: 0, y: h - 1}, {x, y: y - 1}, {width: w, height: 1}); // T\n RGBAImage.copy(src, dst, {x: 0, y: 0}, {x, y: y + h}, {width: w, height: 1}); // B\n RGBAImage.copy(src, dst, {x: w - 1, y: 0}, {x: x - 1, y}, {width: 1, height: h}); // L\n RGBAImage.copy(src, dst, {x: 0, y: 0}, {x: x + w, y}, {width: 1, height: h}); // R\n }\n\n this.dirty = true;\n }\n\n beginFrame() {\n this.callbackDispatchedThisFrame = {};\n }\n\n dispatchRenderCallbacks(ids: Array) {\n for (const id of ids) {\n\n // the callback for the image was already dispatched for a different frame\n if (this.callbackDispatchedThisFrame[id]) continue;\n this.callbackDispatchedThisFrame[id] = true;\n\n const image = this.getImage(id);\n if (!image) warnOnce(`Image with ID: \"${id}\" was not found`);\n\n const updated = renderStyleImage(image);\n if (updated) {\n this.updateImage(id, image);\n }\n }\n }\n}\n","const INF = 1e20;\n\nexport default class TinySDF {\n constructor({\n fontSize = 24,\n buffer = 3,\n radius = 8,\n cutoff = 0.25,\n fontFamily = 'sans-serif',\n fontWeight = 'normal',\n fontStyle = 'normal'\n } = {}) {\n this.buffer = buffer;\n this.cutoff = cutoff;\n this.radius = radius;\n\n // make the canvas size big enough to both have the specified buffer around the glyph\n // for \"halo\", and account for some glyphs possibly being larger than their font size\n const size = this.size = fontSize + buffer * 4;\n\n const canvas = this._createCanvas(size);\n const ctx = this.ctx = canvas.getContext('2d', {willReadFrequently: true});\n ctx.font = `${fontStyle} ${fontWeight} ${fontSize}px ${fontFamily}`;\n\n ctx.textBaseline = 'alphabetic';\n ctx.textAlign = 'left'; // Necessary so that RTL text doesn't have different alignment\n ctx.fillStyle = 'black';\n\n // temporary arrays for the distance transform\n this.gridOuter = new Float64Array(size * size);\n this.gridInner = new Float64Array(size * size);\n this.f = new Float64Array(size);\n this.z = new Float64Array(size + 1);\n this.v = new Uint16Array(size);\n }\n\n _createCanvas(size) {\n const canvas = document.createElement('canvas');\n canvas.width = canvas.height = size;\n return canvas;\n }\n\n draw(char) {\n const {\n width: glyphAdvance,\n actualBoundingBoxAscent,\n actualBoundingBoxDescent,\n actualBoundingBoxLeft,\n actualBoundingBoxRight\n } = this.ctx.measureText(char);\n\n // The integer/pixel part of the top alignment is encoded in metrics.glyphTop\n // The remainder is implicitly encoded in the rasterization\n const glyphTop = Math.ceil(actualBoundingBoxAscent);\n const glyphLeft = 0;\n\n // If the glyph overflows the canvas size, it will be clipped at the bottom/right\n const glyphWidth = Math.max(0, Math.min(this.size - this.buffer, Math.ceil(actualBoundingBoxRight - actualBoundingBoxLeft)));\n const glyphHeight = Math.min(this.size - this.buffer, glyphTop + Math.ceil(actualBoundingBoxDescent));\n\n const width = glyphWidth + 2 * this.buffer;\n const height = glyphHeight + 2 * this.buffer;\n\n const len = Math.max(width * height, 0);\n const data = new Uint8ClampedArray(len);\n const glyph = {data, width, height, glyphWidth, glyphHeight, glyphTop, glyphLeft, glyphAdvance};\n if (glyphWidth === 0 || glyphHeight === 0) return glyph;\n\n const {ctx, buffer, gridInner, gridOuter} = this;\n ctx.clearRect(buffer, buffer, glyphWidth, glyphHeight);\n ctx.fillText(char, buffer, buffer + glyphTop);\n const imgData = ctx.getImageData(buffer, buffer, glyphWidth, glyphHeight);\n\n // Initialize grids outside the glyph range to alpha 0\n gridOuter.fill(INF, 0, len);\n gridInner.fill(0, 0, len);\n\n for (let y = 0; y < glyphHeight; y++) {\n for (let x = 0; x < glyphWidth; x++) {\n const a = imgData.data[4 * (y * glyphWidth + x) + 3] / 255; // alpha value\n if (a === 0) continue; // empty pixels\n\n const j = (y + buffer) * width + x + buffer;\n\n if (a === 1) { // fully drawn pixels\n gridOuter[j] = 0;\n gridInner[j] = INF;\n\n } else { // aliased pixels\n const d = 0.5 - a;\n gridOuter[j] = d > 0 ? d * d : 0;\n gridInner[j] = d < 0 ? d * d : 0;\n }\n }\n }\n\n edt(gridOuter, 0, 0, width, height, width, this.f, this.v, this.z);\n edt(gridInner, buffer, buffer, glyphWidth, glyphHeight, width, this.f, this.v, this.z);\n\n for (let i = 0; i < len; i++) {\n const d = Math.sqrt(gridOuter[i]) - Math.sqrt(gridInner[i]);\n data[i] = Math.round(255 - 255 * (d / this.radius + this.cutoff));\n }\n\n return glyph;\n }\n}\n\n// 2D Euclidean squared distance transform by Felzenszwalb & Huttenlocher https://cs.brown.edu/~pff/papers/dt-final.pdf\nfunction edt(data, x0, y0, width, height, gridSize, f, v, z) {\n for (let x = x0; x < x0 + width; x++) edt1d(data, y0 * gridSize + x, gridSize, height, f, v, z);\n for (let y = y0; y < y0 + height; y++) edt1d(data, y * gridSize + x0, 1, width, f, v, z);\n}\n\n// 1D squared distance transform\nfunction edt1d(grid, offset, stride, length, f, v, z) {\n v[0] = 0;\n z[0] = -INF;\n z[1] = INF;\n f[0] = grid[offset];\n\n for (let q = 1, k = 0, s = 0; q < length; q++) {\n f[q] = grid[offset + q * stride];\n const q2 = q * q;\n do {\n const r = v[k];\n s = (f[q] - f[r] + q2 - r * r) / (q - r) / 2;\n } while (s <= z[k] && --k > -1);\n\n k++;\n v[k] = q;\n z[k] = s;\n z[k + 1] = INF;\n }\n\n for (let q = 0, k = 0; q < length; q++) {\n while (z[k + 1] < q) k++;\n const r = v[k];\n const qr = q - r;\n grid[offset + q * stride] = f[r] + qr * qr;\n }\n}\n","import {loadGlyphRange} from '../style/load_glyph_range';\n\nimport TinySDF from '@mapbox/tiny-sdf';\nimport {unicodeBlockLookup} from '../util/is_char_in_unicode_block';\nimport {asyncAll} from '../util/util';\nimport {AlphaImage} from '../util/image';\n\nimport type {StyleGlyph} from '../style/style_glyph';\nimport type {RequestManager} from '../util/request_manager';\nimport type {Callback} from '../types/callback';\n\ntype Entry = {\n // null means we've requested the range, but the glyph wasn't included in the result.\n glyphs: {\n [id: number]: StyleGlyph | null;\n };\n requests: {\n [range: number]: Array>;\n };\n ranges: {\n [range: number]: boolean | null;\n };\n tinySDF?: TinySDF;\n};\n\nexport class GlyphManager {\n requestManager: RequestManager;\n localIdeographFontFamily: string;\n entries: {\n [_: string]: Entry;\n };\n url: string;\n\n // exposed as statics to enable stubbing in unit tests\n static loadGlyphRange = loadGlyphRange;\n static TinySDF = TinySDF;\n\n constructor(requestManager: RequestManager, localIdeographFontFamily?: string | null) {\n this.requestManager = requestManager;\n this.localIdeographFontFamily = localIdeographFontFamily;\n this.entries = {};\n }\n\n setURL(url?: string | null) {\n this.url = url;\n }\n\n getGlyphs(glyphs: {\n [stack: string]: Array;\n }, callback: Callback<{\n [stack: string]: {\n [id: number]: StyleGlyph;\n };\n }>) {\n const all = [];\n\n for (const stack in glyphs) {\n for (const id of glyphs[stack]) {\n all.push({stack, id});\n }\n }\n\n asyncAll(all, ({stack, id}, callback: Callback<{\n stack: string;\n id: number;\n glyph: StyleGlyph;\n }>) => {\n let entry = this.entries[stack];\n if (!entry) {\n entry = this.entries[stack] = {\n glyphs: {},\n requests: {},\n ranges: {}\n };\n }\n\n let glyph = entry.glyphs[id];\n if (glyph !== undefined) {\n callback(null, {stack, id, glyph});\n return;\n }\n\n glyph = this._tinySDF(entry, stack, id);\n if (glyph) {\n entry.glyphs[id] = glyph;\n callback(null, {stack, id, glyph});\n return;\n }\n\n const range = Math.floor(id / 256);\n if (range * 256 > 65535) {\n callback(new Error('glyphs > 65535 not supported'));\n return;\n }\n\n if (entry.ranges[range]) {\n callback(null, {stack, id, glyph});\n return;\n }\n\n if (!this.url) {\n callback(new Error('glyphsUrl is not set'));\n return;\n }\n\n let requests = entry.requests[range];\n if (!requests) {\n requests = entry.requests[range] = [];\n GlyphManager.loadGlyphRange(stack, range, this.url, this.requestManager,\n (err, response?: {\n [_: number]: StyleGlyph | null;\n } | null) => {\n if (response) {\n for (const id in response) {\n if (!this._doesCharSupportLocalGlyph(+id)) {\n entry.glyphs[+id] = response[+id];\n }\n }\n entry.ranges[range] = true;\n }\n for (const cb of requests) {\n cb(err, response);\n }\n delete entry.requests[range];\n });\n }\n\n requests.push((err, result?: {\n [_: number]: StyleGlyph | null;\n } | null) => {\n if (err) {\n callback(err);\n } else if (result) {\n callback(null, {stack, id, glyph: result[id] || null});\n }\n });\n }, (err, glyphs?: Array<{\n stack: string;\n id: number;\n glyph: StyleGlyph;\n }> | null) => {\n if (err) {\n callback(err);\n } else if (glyphs) {\n const result = {};\n\n for (const {stack, id, glyph} of glyphs) {\n // Clone the glyph so that our own copy of its ArrayBuffer doesn't get transferred.\n (result[stack] || (result[stack] = {}))[id] = glyph && {\n id: glyph.id,\n bitmap: glyph.bitmap.clone(),\n metrics: glyph.metrics\n };\n }\n\n callback(null, result);\n }\n });\n }\n\n _doesCharSupportLocalGlyph(id: number): boolean {\n /* eslint-disable new-cap */\n return !!this.localIdeographFontFamily &&\n (unicodeBlockLookup['CJK Unified Ideographs'](id) ||\n unicodeBlockLookup['Hangul Syllables'](id) ||\n unicodeBlockLookup['Hiragana'](id) ||\n unicodeBlockLookup['Katakana'](id));\n /* eslint-enable new-cap */\n }\n\n _tinySDF(entry: Entry, stack: string, id: number): StyleGlyph {\n const fontFamily = this.localIdeographFontFamily;\n if (!fontFamily) {\n return;\n }\n\n if (!this._doesCharSupportLocalGlyph(id)) {\n return;\n }\n\n // Client-generated glyphs are rendered at 2x texture scale,\n // because CJK glyphs are more detailed than others.\n const textureScale = 2;\n\n let tinySDF = entry.tinySDF;\n if (!tinySDF) {\n let fontWeight = '400';\n if (/bold/i.test(stack)) {\n fontWeight = '900';\n } else if (/medium/i.test(stack)) {\n fontWeight = '500';\n } else if (/light/i.test(stack)) {\n fontWeight = '200';\n }\n tinySDF = entry.tinySDF = new GlyphManager.TinySDF({\n fontSize: 24 * textureScale,\n buffer: 3 * textureScale,\n radius: 8 * textureScale,\n cutoff: 0.25,\n fontFamily,\n fontWeight\n });\n }\n\n const char = tinySDF.draw(String.fromCharCode(id));\n\n /**\n * TinySDF's \"top\" is the distance from the alphabetic baseline to the top of the glyph.\n * Server-generated fonts specify \"top\" relative to an origin above the em box (the origin\n * comes from FreeType, but I'm unclear on exactly how it's derived)\n * ref: https://github.com/mapbox/sdf-glyph-foundry\n *\n * Server fonts don't yet include baseline information, so we can't line up exactly with them\n * (and they don't line up with each other)\n * ref: https://github.com/mapbox/node-fontnik/pull/160\n *\n * To approximately align TinySDF glyphs with server-provided glyphs, we use this baseline adjustment\n * factor calibrated to be in between DIN Pro and Arial Unicode (but closer to Arial Unicode)\n */\n const topAdjustment = 27.5;\n\n const leftAdjustment = 0.5;\n\n return {\n id,\n bitmap: new AlphaImage({width: char.width || 30 * textureScale, height: char.height || 30 * textureScale}, char.data),\n metrics: {\n width: char.glyphWidth / textureScale || 24,\n height: char.glyphHeight / textureScale || 24,\n left: (char.glyphLeft / textureScale + leftAdjustment) || 0,\n top: char.glyphTop / textureScale - topAdjustment || -8,\n advance: char.glyphAdvance / textureScale || 24,\n isDoubleResolution: true\n }\n };\n }\n}\n","import {getArrayBuffer} from '../util/ajax';\nimport {ResourceType} from '../util/request_manager';\n\nimport {parseGlyphPbf} from './parse_glyph_pbf';\n\nimport type {StyleGlyph} from './style_glyph';\nimport type {RequestManager} from '../util/request_manager';\nimport type {Callback} from '../types/callback';\n\nexport function loadGlyphRange(fontstack: string,\n range: number,\n urlTemplate: string,\n requestManager: RequestManager,\n callback: Callback<{\n [_: number]: StyleGlyph | null;\n }>) {\n const begin = range * 256;\n const end = begin + 255;\n\n const request = requestManager.transformRequest(\n urlTemplate.replace('{fontstack}', fontstack).replace('{range}', `${begin}-${end}`),\n ResourceType.Glyphs\n );\n\n getArrayBuffer(request, (err?: Error | null, data?: ArrayBuffer | null) => {\n if (err) {\n callback(err);\n } else if (data) {\n const glyphs = {};\n\n for (const glyph of parseGlyphPbf(data)) {\n glyphs[glyph.id] = glyph;\n }\n\n callback(null, glyphs);\n }\n });\n}\n","import {interpolates, Color, latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {extend, sphericalToCartesian} from '../util/util';\nimport {Evented} from '../util/evented';\nimport {\n validateStyle,\n validateLight,\n emitValidationErrors\n} from './validate_style';\n\nimport type {StylePropertySpecification, LightSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {EvaluationParameters} from './evaluation_parameters';\nimport type {StyleSetterOptions} from '../style/style';\nimport {Properties, Transitionable, Transitioning, PossiblyEvaluated, DataConstantProperty} from './properties';\n\nimport type {\n Property,\n PropertyValue,\n TransitionParameters\n} from './properties';\n\ntype LightPosition = {\n x: number;\n y: number;\n z: number;\n};\n\nclass LightPositionProperty implements Property<[number, number, number], LightPosition> {\n specification: StylePropertySpecification;\n\n constructor() {\n this.specification = styleSpec.light.position as StylePropertySpecification;\n }\n\n possiblyEvaluate(\n value: PropertyValue<[number, number, number], LightPosition>,\n parameters: EvaluationParameters\n ): LightPosition {\n return sphericalToCartesian(value.expression.evaluate(parameters));\n }\n\n interpolate(a: LightPosition, b: LightPosition, t: number): LightPosition {\n return {\n x: interpolates.number(a.x, b.x, t),\n y: interpolates.number(a.y, b.y, t),\n z: interpolates.number(a.z, b.z, t),\n };\n }\n}\n\ntype Props = {\n 'anchor': DataConstantProperty<'map' | 'viewport'>;\n 'position': LightPositionProperty;\n 'color': DataConstantProperty;\n 'intensity': DataConstantProperty;\n};\n\ntype PropsPossiblyEvaluated = {\n 'anchor': 'map' | 'viewport';\n 'position': LightPosition;\n 'color': Color;\n 'intensity': number;\n};\n\nconst TRANSITION_SUFFIX = '-transition';\n\nlet lightProperties: Properties;\n\n/*\n * Represents the light used to light extruded features.\n */\nexport class Light extends Evented {\n _transitionable: Transitionable;\n _transitioning: Transitioning;\n properties: PossiblyEvaluated;\n\n constructor(lightOptions?: LightSpecification) {\n super();\n lightProperties = lightProperties || new Properties({\n 'anchor': new DataConstantProperty(styleSpec.light.anchor as StylePropertySpecification),\n 'position': new LightPositionProperty(),\n 'color': new DataConstantProperty(styleSpec.light.color as StylePropertySpecification),\n 'intensity': new DataConstantProperty(styleSpec.light.intensity as StylePropertySpecification),\n });\n this._transitionable = new Transitionable(lightProperties);\n this.setLight(lightOptions);\n this._transitioning = this._transitionable.untransitioned();\n }\n\n getLight(): LightSpecification {\n return this._transitionable.serialize();\n }\n\n setLight(light?: LightSpecification, options: StyleSetterOptions = {}) {\n if (this._validate(validateLight, light, options)) {\n return;\n }\n\n for (const name in light) {\n const value = light[name];\n if (name.endsWith(TRANSITION_SUFFIX)) {\n this._transitionable.setTransition(name.slice(0, -TRANSITION_SUFFIX.length) as keyof Props, value);\n } else {\n this._transitionable.setValue(name as keyof Props, value);\n }\n }\n }\n\n updateTransitions(parameters: TransitionParameters) {\n this._transitioning = this._transitionable.transitioned(parameters, this._transitioning);\n }\n\n hasTransition() {\n return this._transitioning.hasTransition();\n }\n\n recalculate(parameters: EvaluationParameters) {\n this.properties = this._transitioning.possiblyEvaluate(parameters);\n }\n\n _validate(validate: Function, value: unknown, options?: {\n validate?: boolean;\n }) {\n if (options && options.validate === false) {\n return false;\n }\n\n return emitValidationErrors(this, validate.call(validateStyle, extend({\n value,\n // Workaround for https://github.com/mapbox/mapbox-gl-js/issues/2407\n style: {glyphs: true, sprite: true},\n styleSpec\n })));\n }\n}\n","import {warnOnce} from '../util/util';\n\nimport type {Context} from '../gl/context';\n\n/**\n * A dash entry\n */\ntype DashEntry = {\n y: number;\n height: number;\n width: number;\n}\n\n/**\n * @internal\n * A LineAtlas lets us reuse rendered dashed lines\n * by writing many of them to a texture and then fetching their positions\n * using {@link LineAtlas#getDash}.\n *\n * @param width - the width\n * @param height - the height\n */\nexport class LineAtlas {\n width: number;\n height: number;\n nextRow: number;\n bytes: number;\n data: Uint8Array;\n dashEntry: {[_: string]: DashEntry};\n dirty: boolean;\n texture: WebGLTexture;\n\n constructor(width: number, height: number) {\n this.width = width;\n this.height = height;\n this.nextRow = 0;\n\n this.data = new Uint8Array(this.width * this.height);\n\n this.dashEntry = {};\n }\n\n /**\n * Get or create a dash line pattern.\n *\n * @param dasharray - the key (represented by numbers) to get the dash texture\n * @param round - whether to add circle caps in between dash segments\n * @returns position of dash texture in {@link DashEntry}\n */\n getDash(dasharray: Array, round: boolean) {\n const key = dasharray.join(',') + String(round);\n\n if (!this.dashEntry[key]) {\n this.dashEntry[key] = this.addDash(dasharray, round);\n }\n return this.dashEntry[key];\n }\n\n getDashRanges(dasharray: Array, lineAtlasWidth: number, stretch: number) {\n // If dasharray has an odd length, both the first and last parts\n // are dashes and should be joined seamlessly.\n const oddDashArray = dasharray.length % 2 === 1;\n\n const ranges = [];\n\n let left = oddDashArray ? -dasharray[dasharray.length - 1] * stretch : 0;\n let right = dasharray[0] * stretch;\n let isDash = true;\n\n ranges.push({left, right, isDash, zeroLength: dasharray[0] === 0});\n\n let currentDashLength = dasharray[0];\n for (let i = 1; i < dasharray.length; i++) {\n isDash = !isDash;\n\n const dashLength = dasharray[i];\n left = currentDashLength * stretch;\n currentDashLength += dashLength;\n right = currentDashLength * stretch;\n\n ranges.push({left, right, isDash, zeroLength: dashLength === 0});\n }\n\n return ranges;\n }\n\n addRoundDash(ranges: any, stretch: number, n: number) {\n const halfStretch = stretch / 2;\n\n for (let y = -n; y <= n; y++) {\n const row = this.nextRow + n + y;\n const index = this.width * row;\n let currIndex = 0;\n let range = ranges[currIndex];\n\n for (let x = 0; x < this.width; x++) {\n if (x / range.right > 1) { range = ranges[++currIndex]; }\n\n const distLeft = Math.abs(x - range.left);\n const distRight = Math.abs(x - range.right);\n const minDist = Math.min(distLeft, distRight);\n let signedDistance;\n\n const distMiddle = y / n * (halfStretch + 1);\n if (range.isDash) {\n const distEdge = halfStretch - Math.abs(distMiddle);\n signedDistance = Math.sqrt(minDist * minDist + distEdge * distEdge);\n } else {\n signedDistance = halfStretch - Math.sqrt(minDist * minDist + distMiddle * distMiddle);\n }\n\n this.data[index + x] = Math.max(0, Math.min(255, signedDistance + 128));\n }\n }\n }\n\n addRegularDash(ranges: any) {\n\n // Collapse any zero-length range\n // Collapse neighbouring same-type parts into a single part\n for (let i = ranges.length - 1; i >= 0; --i) {\n const part = ranges[i];\n const next = ranges[i + 1];\n if (part.zeroLength) {\n ranges.splice(i, 1);\n } else if (next && next.isDash === part.isDash) {\n next.left = part.left;\n ranges.splice(i, 1);\n }\n }\n\n // Combine the first and last parts if possible\n const first = ranges[0];\n const last = ranges[ranges.length - 1];\n if (first.isDash === last.isDash) {\n first.left = last.left - this.width;\n last.right = first.right + this.width;\n }\n\n const index = this.width * this.nextRow;\n let currIndex = 0;\n let range = ranges[currIndex];\n\n for (let x = 0; x < this.width; x++) {\n if (x / range.right > 1) {\n range = ranges[++currIndex];\n }\n\n const distLeft = Math.abs(x - range.left);\n const distRight = Math.abs(x - range.right);\n\n const minDist = Math.min(distLeft, distRight);\n const signedDistance = range.isDash ? minDist : -minDist;\n\n this.data[index + x] = Math.max(0, Math.min(255, signedDistance + 128));\n }\n }\n\n addDash(dasharray: Array, round: boolean): DashEntry {\n const n = round ? 7 : 0;\n const height = 2 * n + 1;\n\n if (this.nextRow + height > this.height) {\n warnOnce('LineAtlas out of space');\n return null;\n }\n\n let length = 0;\n for (let i = 0; i < dasharray.length; i++) { length += dasharray[i]; }\n\n if (length !== 0) {\n const stretch = this.width / length;\n const ranges = this.getDashRanges(dasharray, this.width, stretch);\n\n if (round) {\n this.addRoundDash(ranges, stretch, n);\n } else {\n this.addRegularDash(ranges);\n }\n }\n\n const dashEntry = {\n y: (this.nextRow + n + 0.5) / this.height,\n height: 2 * n / this.height,\n width: length\n };\n\n this.nextRow += height;\n this.dirty = true;\n\n return dashEntry;\n }\n\n bind(context: Context) {\n const gl = context.gl;\n if (!this.texture) {\n this.texture = gl.createTexture();\n gl.bindTexture(gl.TEXTURE_2D, this.texture);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.REPEAT);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.REPEAT);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n gl.texImage2D(gl.TEXTURE_2D, 0, gl.ALPHA, this.width, this.height, 0, gl.ALPHA, gl.UNSIGNED_BYTE, this.data);\n\n } else {\n gl.bindTexture(gl.TEXTURE_2D, this.texture);\n\n if (this.dirty) {\n this.dirty = false;\n gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, this.width, this.height, gl.ALPHA, gl.UNSIGNED_BYTE, this.data);\n }\n }\n }\n}\n","/**\n * Invokes the wrapped function in a non-blocking way when trigger() is called. Invocation requests\n * are ignored until the function was actually invoked.\n */\nexport class ThrottledInvoker {\n _channel: MessageChannel;\n _triggered: boolean;\n _callback: Function;\n\n constructor(callback: Function) {\n this._callback = callback;\n this._triggered = false;\n if (typeof MessageChannel !== 'undefined') {\n this._channel = new MessageChannel();\n this._channel.port2.onmessage = () => {\n this._triggered = false;\n this._callback();\n };\n }\n }\n\n trigger() {\n if (!this._triggered) {\n this._triggered = true;\n if (this._channel) {\n this._channel.port1.postMessage(true);\n } else {\n setTimeout(() => {\n this._triggered = false;\n this._callback();\n }, 0);\n }\n }\n }\n\n remove() {\n delete this._channel;\n this._callback = () => {};\n }\n}\n","import {isWorker} from './util';\nimport {serialize, deserialize, Serialized} from './web_worker_transfer';\nimport {ThrottledInvoker} from './throttled_invoker';\n\nimport type {Transferable} from '../types/transferable';\nimport type {Cancelable} from '../types/cancelable';\nimport type {WorkerSource} from '../source/worker_source';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {Callback} from '../types/callback';\nimport type {StyleGlyph} from '../style/style_glyph';\n\nexport interface ActorTarget {\n addEventListener: typeof window.addEventListener;\n removeEventListener: typeof window.removeEventListener;\n postMessage: typeof window.postMessage;\n terminate?: () => void;\n}\n\nexport interface WorkerSourceProvider {\n getWorkerSource(mapId: string | number, sourceType: string, sourceName: string): WorkerSource;\n}\n\nexport interface GlyphsProvider {\n getGlyphs(mapId: string, params: {\n stacks: {[_: string]: Array};\n source: string;\n tileID: OverscaledTileID;\n type: string;\n },\n callback: Callback<{[_: string]: {[_: number]: StyleGlyph}}>\n );\n}\n\nexport type MessageType = '' | '' |\n'geojson.getClusterExpansionZoom' | 'geojson.getClusterChildren' | 'geojson.getClusterLeaves' | 'geojson.loadData' |\n'removeSource' | 'loadWorkerSource' | 'loadDEMTile' | 'removeDEMTile' |\n'removeTile' | 'reloadTile' | 'abortTile' | 'loadTile' | 'getTile' |\n'getGlyphs' | 'getImages' | 'setImages' |\n'syncRTLPluginState' | 'setReferrer' | 'setLayers' | 'updateLayers';\n\nexport type MessageData = {\n id: string;\n type: MessageType;\n data?: Serialized;\n targetMapId?: string | number | null;\n mustQueue?: boolean;\n error?: Serialized | null;\n hasCallback?: boolean;\n sourceMapId: string | number | null;\n}\n\nexport type Message = {\n data: MessageData;\n}\n\n/**\n * An implementation of the [Actor design pattern](http://en.wikipedia.org/wiki/Actor_model)\n * that maintains the relationship between asynchronous tasks and the objects\n * that spin them off - in this case, tasks like parsing parts of styles,\n * owned by the styles\n */\nexport class Actor {\n target: ActorTarget;\n parent: WorkerSourceProvider | GlyphsProvider;\n mapId: string | number | null;\n callbacks: { [x: number]: Function};\n name: string;\n tasks: { [x: number]: MessageData };\n taskQueue: Array;\n cancelCallbacks: { [x: number]: () => void };\n invoker: ThrottledInvoker;\n globalScope: ActorTarget;\n\n /**\n * @param target - The target\n * @param parent - The parent\n * @param mapId - A unique identifier for the Map instance using this Actor.\n */\n constructor(target: ActorTarget, parent: WorkerSourceProvider | GlyphsProvider, mapId?: string | number) {\n this.target = target;\n this.parent = parent;\n this.mapId = mapId;\n this.callbacks = {};\n this.tasks = {};\n this.taskQueue = [];\n this.cancelCallbacks = {};\n this.invoker = new ThrottledInvoker(this.process);\n this.target.addEventListener('message', this.receive, false);\n this.globalScope = isWorker() ? target : window;\n }\n\n /**\n * Sends a message from a main-thread map to a Worker or from a Worker back to\n * a main-thread map instance.\n *\n * @param type - The name of the target method to invoke or '[source-type].[source-name].name' for a method on a WorkerSource.\n * @param targetMapId - A particular mapId to which to send this message.\n */\n send(\n type: MessageType,\n data: unknown,\n callback?: Function | null,\n targetMapId?: string | null,\n mustQueue: boolean = false\n ): Cancelable {\n // We're using a string ID instead of numbers because they are being used as object keys\n // anyway, and thus stringified implicitly. We use random IDs because an actor may receive\n // message from multiple other actors which could run in different execution context. A\n // linearly increasing ID could produce collisions.\n const id = Math.round((Math.random() * 1e18)).toString(36).substring(0, 10);\n if (callback) {\n this.callbacks[id] = callback;\n }\n const buffers: Array = [];\n const message: MessageData = {\n id,\n type,\n hasCallback: !!callback,\n targetMapId,\n mustQueue,\n sourceMapId: this.mapId,\n data: serialize(data, buffers)\n };\n\n this.target.postMessage(message, {transfer: buffers});\n return {\n cancel: () => {\n if (callback) {\n // Set the callback to null so that it never fires after the request is aborted.\n delete this.callbacks[id];\n }\n const cancelMessage: MessageData = {\n id,\n type: '',\n targetMapId,\n sourceMapId: this.mapId\n };\n this.target.postMessage(cancelMessage);\n }\n };\n }\n\n receive = (message: Message) => {\n const data = message.data;\n const id = data.id;\n\n if (!id) {\n return;\n }\n\n if (data.targetMapId && this.mapId !== data.targetMapId) {\n return;\n }\n\n if (data.type === '') {\n // Remove the original request from the queue. This is only possible if it\n // hasn't been kicked off yet. The id will remain in the queue, but because\n // there is no associated task, it will be dropped once it's time to execute it.\n delete this.tasks[id];\n const cancel = this.cancelCallbacks[id];\n delete this.cancelCallbacks[id];\n if (cancel) {\n cancel();\n }\n } else {\n if (isWorker() || data.mustQueue) {\n // In workers, store the tasks that we need to process before actually processing them. This\n // is necessary because we want to keep receiving messages, and in particular,\n // messages. Some tasks may take a while in the worker thread, so before\n // executing the next task in our queue, postMessage preempts this and \n // messages can be processed. We're using a MessageChannel object to get throttle the\n // process() flow to one at a time.\n this.tasks[id] = data;\n this.taskQueue.push(id);\n this.invoker.trigger();\n } else {\n // In the main thread, process messages immediately so that other work does not slip in\n // between getting partial data back from workers.\n this.processTask(id, data);\n }\n }\n };\n\n process = () => {\n if (!this.taskQueue.length) {\n return;\n }\n const id = this.taskQueue.shift();\n const task = this.tasks[id];\n delete this.tasks[id];\n // Schedule another process call if we know there's more to process _before_ invoking the\n // current task. This is necessary so that processing continues even if the current task\n // doesn't execute successfully.\n if (this.taskQueue.length) {\n this.invoker.trigger();\n }\n if (!task) {\n // If the task ID doesn't have associated task data anymore, it was canceled.\n return;\n }\n\n this.processTask(id, task);\n };\n\n processTask(id: string, task: MessageData) {\n if (task.type === '') {\n // The done() function in the counterpart has been called, and we are now\n // firing the callback in the originating actor, if there is one.\n const callback = this.callbacks[id];\n delete this.callbacks[id];\n if (callback) {\n // If we get a response, but don't have a callback, the request was canceled.\n if (task.error) {\n callback(deserialize(task.error));\n } else {\n callback(null, deserialize(task.data));\n }\n }\n } else {\n let completed = false;\n const buffers: Array = [];\n const done = task.hasCallback ? (err: Error, data?: any) => {\n completed = true;\n delete this.cancelCallbacks[id];\n const responseMessage: MessageData = {\n id,\n type: '',\n sourceMapId: this.mapId,\n error: err ? serialize(err) : null,\n data: serialize(data, buffers)\n };\n this.target.postMessage(responseMessage, {transfer: buffers});\n } : (_) => {\n completed = true;\n };\n\n let callback: Cancelable = null;\n const params = deserialize(task.data);\n if (this.parent[task.type]) {\n // task.type == 'loadTile', 'removeTile', etc.\n callback = this.parent[task.type](task.sourceMapId, params, done);\n } else if ('getWorkerSource' in this.parent) {\n // task.type == sourcetype.method\n const keys = task.type.split('.');\n const scope = this.parent.getWorkerSource(task.sourceMapId, keys[0], (params as any).source);\n callback = scope[keys[1]](params, done);\n } else {\n // No function was found.\n done(new Error(`Could not find function ${task.type}`));\n }\n\n if (!completed && callback && callback.cancel) {\n // Allows canceling the task as long as it hasn't been completed yet.\n this.cancelCallbacks[id] = callback.cancel;\n }\n }\n }\n\n remove() {\n this.invoker.remove();\n this.target.removeEventListener('message', this.receive, false);\n }\n}\n","import {asyncAll} from './util';\nimport {Actor, GlyphsProvider, MessageType} from './actor';\n\nimport type {WorkerPool} from './worker_pool';\nimport type {WorkerSource} from '../source/worker_source'; /* eslint-disable-line */ // this is used for the docs' import\n/**\n * Responsible for sending messages from a {@link Source} to an associated\n * {@link WorkerSource}.\n */\nexport class Dispatcher {\n workerPool: WorkerPool;\n actors: Array;\n currentActor: number;\n id: string | number;\n\n constructor(workerPool: WorkerPool, parent: GlyphsProvider, mapId: string | number) {\n this.workerPool = workerPool;\n this.actors = [];\n this.currentActor = 0;\n this.id = mapId;\n const workers = this.workerPool.acquire(mapId);\n for (let i = 0; i < workers.length; i++) {\n const worker = workers[i];\n const actor = new Actor(worker, parent, mapId);\n actor.name = `Worker ${i}`;\n this.actors.push(actor);\n }\n if (!this.actors.length) throw new Error('No actors found');\n }\n\n /**\n * Broadcast a message to all Workers.\n */\n broadcast(type: MessageType, data: unknown, cb?: (...args: any[]) => any) {\n cb = cb || function () {};\n asyncAll(this.actors, (actor, done) => {\n actor.send(type, data, done);\n }, cb);\n }\n\n /**\n * Acquires an actor to dispatch messages to. The actors are distributed in round-robin fashion.\n * @returns An actor object backed by a web worker for processing messages.\n */\n getActor(): Actor {\n this.currentActor = (this.currentActor + 1) % this.actors.length;\n return this.actors[this.currentActor];\n }\n\n remove(mapRemoved: boolean = true) {\n this.actors.forEach((actor) => { actor.remove(); });\n this.actors = [];\n if (mapRemoved) this.workerPool.release(this.id);\n }\n}\n","import {pick, extend} from '../util/util';\n\nimport {getJSON} from '../util/ajax';\nimport {ResourceType} from '../util/request_manager';\nimport {browser} from '../util/browser';\n\nimport type {RequestManager} from '../util/request_manager';\nimport type {Callback} from '../types/callback';\nimport type {TileJSON} from '../types/tilejson';\nimport type {Cancelable} from '../types/cancelable';\nimport type {RasterDEMSourceSpecification, RasterSourceSpecification, VectorSourceSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport function loadTileJson(\n options: RasterSourceSpecification | RasterDEMSourceSpecification | VectorSourceSpecification,\n requestManager: RequestManager,\n callback: Callback\n): Cancelable {\n const loaded = function(err: Error, tileJSON: any) {\n if (err) {\n return callback(err);\n } else if (tileJSON) {\n const result: any = pick(\n // explicit source options take precedence over TileJSON\n extend(tileJSON, options),\n ['tiles', 'minzoom', 'maxzoom', 'attribution', 'bounds', 'scheme', 'tileSize', 'encoding']\n );\n\n if (tileJSON.vector_layers) {\n result.vectorLayers = tileJSON.vector_layers;\n result.vectorLayerIds = result.vectorLayers.map((layer) => { return layer.id; });\n }\n\n callback(null, result);\n }\n };\n\n if (options.url) {\n return getJSON(requestManager.transformRequest(options.url, ResourceType.Source), loaded);\n } else {\n return browser.frame(() => loaded(null, options));\n }\n}\n","import {wrap} from '../util/util';\n\n/*\n* Approximate radius of the earth in meters.\n* Uses the WGS-84 approximation. The radius at the equator is ~6378137 and at the poles is ~6356752. https://en.wikipedia.org/wiki/World_Geodetic_System#WGS84\n* 6371008.8 is one published \"average radius\" see https://en.wikipedia.org/wiki/Earth_radius#Mean_radius, or ftp://athena.fsv.cvut.cz/ZFG/grs80-Moritz.pdf p.4\n*/\nexport const earthRadius = 6371008.8;\n\n/**\n * A {@link LngLat} object, an array of two numbers representing longitude and latitude,\n * or an object with `lng` and `lat` or `lon` and `lat` properties.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let v1 = new maplibregl.LngLat(-122.420679, 37.772537);\n * let v2 = [-122.420679, 37.772537];\n * let v3 = {lon: -122.420679, lat: 37.772537};\n * ```\n */\nexport type LngLatLike = LngLat | {\n lng: number;\n lat: number;\n} | {\n lon: number;\n lat: number;\n} | [number, number];\n\n/**\n * A `LngLat` object represents a given longitude and latitude coordinate, measured in degrees.\n * These coordinates are based on the [WGS84 (EPSG:4326) standard](https://en.wikipedia.org/wiki/World_Geodetic_System#WGS84).\n *\n * MapLibre GL JS uses longitude, latitude coordinate order (as opposed to latitude, longitude) to match the\n * [GeoJSON specification](https://tools.ietf.org/html/rfc7946).\n *\n * Note that any MapLibre GL JS method that accepts a `LngLat` object as an argument or option\n * can also accept an `Array` of two numbers and will perform an implicit conversion.\n * This flexible type is documented as {@link LngLatLike}.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(-123.9749, 40.7736);\n * ll.lng; // = -123.9749\n * ```\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Create a timeline animation](https://maplibre.org/maplibre-gl-js/docs/examples/timeline-animation/)\n */\nexport class LngLat {\n lng: number;\n lat: number;\n\n /**\n * @param lng - Longitude, measured in degrees.\n * @param lat - Latitude, measured in degrees.\n */\n constructor(lng: number, lat: number) {\n if (isNaN(lng) || isNaN(lat)) {\n throw new Error(`Invalid LngLat object: (${lng}, ${lat})`);\n }\n this.lng = +lng;\n this.lat = +lat;\n if (this.lat > 90 || this.lat < -90) {\n throw new Error('Invalid LngLat latitude value: must be between -90 and 90');\n }\n }\n\n /**\n * Returns a new `LngLat` object whose longitude is wrapped to the range (-180, 180).\n *\n * @returns The wrapped `LngLat` object.\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(286.0251, 40.7736);\n * let wrapped = ll.wrap();\n * wrapped.lng; // = -73.9749\n * ```\n */\n wrap() {\n return new LngLat(wrap(this.lng, -180, 180), this.lat);\n }\n\n /**\n * Returns the coordinates represented as an array of two numbers.\n *\n * @returns The coordinates represented as an array of longitude and latitude.\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(-73.9749, 40.7736);\n * ll.toArray(); // = [-73.9749, 40.7736]\n * ```\n */\n toArray(): [number, number] {\n return [this.lng, this.lat];\n }\n\n /**\n * Returns the coordinates represent as a string.\n *\n * @returns The coordinates represented as a string of the format `'LngLat(lng, lat)'`.\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(-73.9749, 40.7736);\n * ll.toString(); // = \"LngLat(-73.9749, 40.7736)\"\n * ```\n */\n toString(): string {\n return `LngLat(${this.lng}, ${this.lat})`;\n }\n\n /**\n * Returns the approximate distance between a pair of coordinates in meters\n * Uses the Haversine Formula (from R.W. Sinnott, \"Virtues of the Haversine\", Sky and Telescope, vol. 68, no. 2, 1984, p. 159)\n *\n * @param lngLat - coordinates to compute the distance to\n * @returns Distance in meters between the two coordinates.\n * @example\n * ```ts\n * let new_york = new maplibregl.LngLat(-74.0060, 40.7128);\n * let los_angeles = new maplibregl.LngLat(-118.2437, 34.0522);\n * new_york.distanceTo(los_angeles); // = 3935751.690893987, \"true distance\" using a non-spherical approximation is ~3966km\n * ```\n */\n distanceTo(lngLat: LngLat): number {\n const rad = Math.PI / 180;\n const lat1 = this.lat * rad;\n const lat2 = lngLat.lat * rad;\n const a = Math.sin(lat1) * Math.sin(lat2) + Math.cos(lat1) * Math.cos(lat2) * Math.cos((lngLat.lng - this.lng) * rad);\n\n const maxMeters = earthRadius * Math.acos(Math.min(a, 1));\n return maxMeters;\n }\n\n /**\n * Converts an array of two numbers or an object with `lng` and `lat` or `lon` and `lat` properties\n * to a `LngLat` object.\n *\n * If a `LngLat` object is passed in, the function returns it unchanged.\n *\n * @param input - An array of two numbers or object to convert, or a `LngLat` object to return.\n * @returns A new `LngLat` object, if a conversion occurred, or the original `LngLat` object.\n * @example\n * ```ts\n * let arr = [-73.9749, 40.7736];\n * let ll = maplibregl.LngLat.convert(arr);\n * ll; // = LngLat {lng: -73.9749, lat: 40.7736}\n * ```\n */\n static convert(input: LngLatLike): LngLat {\n if (input instanceof LngLat) {\n return input;\n }\n if (Array.isArray(input) && (input.length === 2 || input.length === 3)) {\n return new LngLat(Number(input[0]), Number(input[1]));\n }\n if (!Array.isArray(input) && typeof input === 'object' && input !== null) {\n return new LngLat(\n // flow can't refine this to have one of lng or lat, so we have to cast to any\n Number('lng' in input ? (input as any).lng : (input as any).lon),\n Number(input.lat)\n );\n }\n throw new Error('`LngLatLike` argument must be specified as a LngLat instance, an object {lng: , lat: }, an object {lon: , lat: }, or an array of [, ]');\n }\n}\n","import {LngLat} from './lng_lat';\nimport type {LngLatLike} from './lng_lat';\n\n/**\n * A {@link LngLatBounds} object, an array of {@link LngLatLike} objects in [sw, ne] order,\n * or an array of numbers in [west, south, east, north] order.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let v1 = new maplibregl.LngLatBounds(\n * new maplibregl.LngLat(-73.9876, 40.7661),\n * new maplibregl.LngLat(-73.9397, 40.8002)\n * );\n * let v2 = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002])\n * let v3 = [[-73.9876, 40.7661], [-73.9397, 40.8002]];\n * ```\n */\nexport type LngLatBoundsLike = LngLatBounds | [LngLatLike, LngLatLike] | [number, number, number, number];\n\n/**\n * A `LngLatBounds` object represents a geographical bounding box,\n * defined by its southwest and northeast points in longitude and latitude.\n *\n * If no arguments are provided to the constructor, a `null` bounding box is created.\n *\n * Note that any Mapbox GL method that accepts a `LngLatBounds` object as an argument or option\n * can also accept an `Array` of two {@link LngLatLike} constructs and will perform an implicit conversion.\n * This flexible type is documented as {@link LngLatBoundsLike}.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let sw = new maplibregl.LngLat(-73.9876, 40.7661);\n * let ne = new maplibregl.LngLat(-73.9397, 40.8002);\n * let llb = new maplibregl.LngLatBounds(sw, ne);\n * ```\n */\nexport class LngLatBounds {\n _ne: LngLat;\n _sw: LngLat;\n\n /**\n * @param sw - The southwest corner of the bounding box.\n * OR array of 4 numbers in the order of west, south, east, north\n * OR array of 2 LngLatLike: [sw,ne]\n * @param ne - The northeast corner of the bounding box.\n * @example\n * ```ts\n * let sw = new maplibregl.LngLat(-73.9876, 40.7661);\n * let ne = new maplibregl.LngLat(-73.9397, 40.8002);\n * let llb = new maplibregl.LngLatBounds(sw, ne);\n * ```\n * OR\n * ```ts\n * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661, -73.9397, 40.8002]);\n * ```\n * OR\n * ```ts\n * let llb = new maplibregl.LngLatBounds([sw, ne]);\n * ```\n */\n constructor(sw?: LngLatLike | [number, number, number, number] | [LngLatLike, LngLatLike], ne?: LngLatLike) {\n if (!sw) {\n // noop\n } else if (ne) {\n this.setSouthWest(sw).setNorthEast(ne);\n } else if (Array.isArray(sw)) {\n if (sw.length === 4) {\n // 4 element array: west, south, east, north\n this.setSouthWest([sw[0], sw[1]]).setNorthEast([sw[2], sw[3]]);\n } else {\n this.setSouthWest(sw[0] as LngLatLike).setNorthEast(sw[1] as LngLatLike);\n }\n }\n }\n\n /**\n * Set the northeast corner of the bounding box\n *\n * @param ne - a {@link LngLatLike} object describing the northeast corner of the bounding box.\n * @returns `this`\n */\n setNorthEast(ne: LngLatLike): this {\n this._ne = ne instanceof LngLat ? new LngLat(ne.lng, ne.lat) : LngLat.convert(ne);\n return this;\n }\n\n /**\n * Set the southwest corner of the bounding box\n *\n * @param sw - a {@link LngLatLike} object describing the southwest corner of the bounding box.\n * @returns `this`\n */\n setSouthWest(sw: LngLatLike): this {\n this._sw = sw instanceof LngLat ? new LngLat(sw.lng, sw.lat) : LngLat.convert(sw);\n return this;\n }\n\n /**\n * Extend the bounds to include a given LngLatLike or LngLatBoundsLike.\n *\n * @param obj - object to extend to\n * @returns `this`\n */\n extend(obj: LngLatLike | LngLatBoundsLike): this {\n const sw = this._sw,\n ne = this._ne;\n let sw2, ne2;\n\n if (obj instanceof LngLat) {\n sw2 = obj;\n ne2 = obj;\n\n } else if (obj instanceof LngLatBounds) {\n sw2 = obj._sw;\n ne2 = obj._ne;\n\n if (!sw2 || !ne2) return this;\n\n } else {\n if (Array.isArray(obj)) {\n if (obj.length === 4 || (obj as any[]).every(Array.isArray)) {\n const lngLatBoundsObj = (obj as any as LngLatBoundsLike);\n return this.extend(LngLatBounds.convert(lngLatBoundsObj));\n } else {\n const lngLatObj = (obj as any as LngLatLike);\n return this.extend(LngLat.convert(lngLatObj));\n }\n\n } else if (obj && ('lng' in obj || 'lon' in obj) && 'lat' in obj) {\n return this.extend(LngLat.convert(obj));\n }\n\n return this;\n }\n\n if (!sw && !ne) {\n this._sw = new LngLat(sw2.lng, sw2.lat);\n this._ne = new LngLat(ne2.lng, ne2.lat);\n\n } else {\n sw.lng = Math.min(sw2.lng, sw.lng);\n sw.lat = Math.min(sw2.lat, sw.lat);\n ne.lng = Math.max(ne2.lng, ne.lng);\n ne.lat = Math.max(ne2.lat, ne.lat);\n }\n\n return this;\n }\n\n /**\n * Returns the geographical coordinate equidistant from the bounding box's corners.\n *\n * @returns The bounding box's center.\n * @example\n * ```ts\n * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]);\n * llb.getCenter(); // = LngLat {lng: -73.96365, lat: 40.78315}\n * ```\n */\n getCenter(): LngLat {\n return new LngLat((this._sw.lng + this._ne.lng) / 2, (this._sw.lat + this._ne.lat) / 2);\n }\n\n /**\n * Returns the southwest corner of the bounding box.\n *\n * @returns The southwest corner of the bounding box.\n */\n getSouthWest(): LngLat { return this._sw; }\n\n /**\n * Returns the northeast corner of the bounding box.\n *\n * @returns The northeast corner of the bounding box.\n */\n getNorthEast(): LngLat { return this._ne; }\n\n /**\n * Returns the northwest corner of the bounding box.\n *\n * @returns The northwest corner of the bounding box.\n */\n getNorthWest(): LngLat { return new LngLat(this.getWest(), this.getNorth()); }\n\n /**\n * Returns the southeast corner of the bounding box.\n *\n * @returns The southeast corner of the bounding box.\n */\n getSouthEast(): LngLat { return new LngLat(this.getEast(), this.getSouth()); }\n\n /**\n * Returns the west edge of the bounding box.\n *\n * @returns The west edge of the bounding box.\n */\n getWest(): number { return this._sw.lng; }\n\n /**\n * Returns the south edge of the bounding box.\n *\n * @returns The south edge of the bounding box.\n */\n getSouth(): number { return this._sw.lat; }\n\n /**\n * Returns the east edge of the bounding box.\n *\n * @returns The east edge of the bounding box.\n */\n getEast(): number { return this._ne.lng; }\n\n /**\n * Returns the north edge of the bounding box.\n *\n * @returns The north edge of the bounding box.\n */\n getNorth(): number { return this._ne.lat; }\n\n /**\n * Returns the bounding box represented as an array.\n *\n * @returns The bounding box represented as an array, consisting of the\n * southwest and northeast coordinates of the bounding represented as arrays of numbers.\n * @example\n * ```ts\n * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]);\n * llb.toArray(); // = [[-73.9876, 40.7661], [-73.9397, 40.8002]]\n * ```\n */\n toArray() {\n return [this._sw.toArray(), this._ne.toArray()];\n }\n\n /**\n * Return the bounding box represented as a string.\n *\n * @returns The bounding box represents as a string of the format\n * `'LngLatBounds(LngLat(lng, lat), LngLat(lng, lat))'`.\n * @example\n * ```ts\n * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]);\n * llb.toString(); // = \"LngLatBounds(LngLat(-73.9876, 40.7661), LngLat(-73.9397, 40.8002))\"\n * ```\n */\n toString() {\n return `LngLatBounds(${this._sw.toString()}, ${this._ne.toString()})`;\n }\n\n /**\n * Check if the bounding box is an empty/`null`-type box.\n *\n * @returns True if bounds have been defined, otherwise false.\n */\n isEmpty() {\n return !(this._sw && this._ne);\n }\n\n /**\n * Check if the point is within the bounding box.\n *\n * @param lnglat - geographic point to check against.\n * @returns `true` if the point is within the bounding box.\n * @example\n * ```ts\n * let llb = new maplibregl.LngLatBounds(\n * new maplibregl.LngLat(-73.9876, 40.7661),\n * new maplibregl.LngLat(-73.9397, 40.8002)\n * );\n *\n * let ll = new maplibregl.LngLat(-73.9567, 40.7789);\n *\n * console.log(llb.contains(ll)); // = true\n * ```\n */\n contains(lnglat: LngLatLike) {\n const {lng, lat} = LngLat.convert(lnglat);\n\n const containsLatitude = this._sw.lat <= lat && lat <= this._ne.lat;\n let containsLongitude = this._sw.lng <= lng && lng <= this._ne.lng;\n if (this._sw.lng > this._ne.lng) { // wrapped coordinates\n containsLongitude = this._sw.lng >= lng && lng >= this._ne.lng;\n }\n\n return containsLatitude && containsLongitude;\n }\n\n /**\n * Converts an array to a `LngLatBounds` object.\n *\n * If a `LngLatBounds` object is passed in, the function returns it unchanged.\n *\n * Internally, the function calls `LngLat#convert` to convert arrays to `LngLat` values.\n *\n * @param input - An array of two coordinates to convert, or a `LngLatBounds` object to return.\n * @returns A new `LngLatBounds` object, if a conversion occurred, or the original `LngLatBounds` object.\n * @example\n * ```ts\n * let arr = [[-73.9876, 40.7661], [-73.9397, 40.8002]];\n * let llb = maplibregl.LngLatBounds.convert(arr); // = LngLatBounds {_sw: LngLat {lng: -73.9876, lat: 40.7661}, _ne: LngLat {lng: -73.9397, lat: 40.8002}}\n * ```\n */\n static convert(input: LngLatBoundsLike | null): LngLatBounds {\n if (input instanceof LngLatBounds) return input;\n if (!input) return input as null;\n return new LngLatBounds(input);\n }\n\n /**\n * Returns a `LngLatBounds` from the coordinates extended by a given `radius`. The returned `LngLatBounds` completely contains the `radius`.\n *\n * @param center - center coordinates of the new bounds.\n * @param radius - Distance in meters from the coordinates to extend the bounds.\n * @returns A new `LngLatBounds` object representing the coordinates extended by the `radius`.\n * @example\n * ```ts\n * let center = new maplibregl.LngLat(-73.9749, 40.7736);\n * maplibregl.LngLatBounds.fromLngLat(100).toArray(); // = [[-73.97501862141328, 40.77351016847229], [-73.97478137858673, 40.77368983152771]]\n * ```\n */\n static fromLngLat(center: LngLat, radius:number = 0): LngLatBounds {\n const earthCircumferenceInMetersAtEquator = 40075017;\n const latAccuracy = 360 * radius / earthCircumferenceInMetersAtEquator,\n lngAccuracy = latAccuracy / Math.cos((Math.PI / 180) * center.lat);\n\n return new LngLatBounds(new LngLat(center.lng - lngAccuracy, center.lat - latAccuracy),\n new LngLat(center.lng + lngAccuracy, center.lat + latAccuracy));\n }\n}\n","import {LngLat, earthRadius} from '../geo/lng_lat';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport {IMercatorCoordinate} from '@maplibre/maplibre-gl-style-spec';\n\n/*\n * The average circumference of the world in meters.\n */\nconst earthCircumfrence = 2 * Math.PI * earthRadius; // meters\n\n/*\n * The circumference at a line of latitude in meters.\n */\nfunction circumferenceAtLatitude(latitude: number) {\n return earthCircumfrence * Math.cos(latitude * Math.PI / 180);\n}\n\nexport function mercatorXfromLng(lng: number) {\n return (180 + lng) / 360;\n}\n\nexport function mercatorYfromLat(lat: number) {\n return (180 - (180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360)))) / 360;\n}\n\nexport function mercatorZfromAltitude(altitude: number, lat: number) {\n return altitude / circumferenceAtLatitude(lat);\n}\n\nexport function lngFromMercatorX(x: number) {\n return x * 360 - 180;\n}\n\nexport function latFromMercatorY(y: number) {\n const y2 = 180 - y * 360;\n return 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90;\n}\n\nexport function altitudeFromMercatorZ(z: number, y: number) {\n return z * circumferenceAtLatitude(latFromMercatorY(y));\n}\n\n/**\n * Determine the Mercator scale factor for a given latitude, see\n * https://en.wikipedia.org/wiki/Mercator_projection#Scale_factor\n *\n * At the equator the scale factor will be 1, which increases at higher latitudes.\n *\n * @param lat - Latitude\n * @returns scale factor\n */\nexport function mercatorScale(lat: number) {\n return 1 / Math.cos(lat * Math.PI / 180);\n}\n\n/**\n * A `MercatorCoordinate` object represents a projected three dimensional position.\n *\n * `MercatorCoordinate` uses the web mercator projection ([EPSG:3857](https://epsg.io/3857)) with slightly different units:\n * - the size of 1 unit is the width of the projected world instead of the \"mercator meter\"\n * - the origin of the coordinate space is at the north-west corner instead of the middle\n *\n * For example, `MercatorCoordinate(0, 0, 0)` is the north-west corner of the mercator world and\n * `MercatorCoordinate(1, 1, 0)` is the south-east corner. If you are familiar with\n * [vector tiles](https://github.com/mapbox/vector-tile-spec) it may be helpful to think\n * of the coordinate space as the `0/0/0` tile with an extent of `1`.\n *\n * The `z` dimension of `MercatorCoordinate` is conformal. A cube in the mercator coordinate space would be rendered as a cube.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let nullIsland = new maplibregl.MercatorCoordinate(0.5, 0.5, 0);\n * ```\n * @see [Add a custom style layer](https://maplibre.org/maplibre-gl-js/docs/examples/custom-style-layer/)\n */\nexport class MercatorCoordinate implements IMercatorCoordinate {\n x: number;\n y: number;\n z: number;\n\n /**\n * @param x - The x component of the position.\n * @param y - The y component of the position.\n * @param z - The z component of the position.\n */\n constructor(x: number, y: number, z: number = 0) {\n this.x = +x;\n this.y = +y;\n this.z = +z;\n }\n\n /**\n * Project a `LngLat` to a `MercatorCoordinate`.\n *\n * @param lngLatLike - The location to project.\n * @param altitude - The altitude in meters of the position.\n * @returns The projected mercator coordinate.\n * @example\n * ```ts\n * let coord = maplibregl.MercatorCoordinate.fromLngLat({ lng: 0, lat: 0}, 0);\n * coord; // MercatorCoordinate(0.5, 0.5, 0)\n * ```\n */\n static fromLngLat(lngLatLike: LngLatLike, altitude: number = 0): MercatorCoordinate {\n const lngLat = LngLat.convert(lngLatLike);\n\n return new MercatorCoordinate(\n mercatorXfromLng(lngLat.lng),\n mercatorYfromLat(lngLat.lat),\n mercatorZfromAltitude(altitude, lngLat.lat));\n }\n\n /**\n * Returns the `LngLat` for the coordinate.\n *\n * @returns The `LngLat` object.\n * @example\n * ```ts\n * let coord = new maplibregl.MercatorCoordinate(0.5, 0.5, 0);\n * let lngLat = coord.toLngLat(); // LngLat(0, 0)\n * ```\n */\n toLngLat() {\n return new LngLat(\n lngFromMercatorX(this.x),\n latFromMercatorY(this.y));\n }\n\n /**\n * Returns the altitude in meters of the coordinate.\n *\n * @returns The altitude in meters.\n * @example\n * ```ts\n * let coord = new maplibregl.MercatorCoordinate(0, 0, 0.02);\n * coord.toAltitude(); // 6914.281956295339\n * ```\n */\n toAltitude(): number {\n return altitudeFromMercatorZ(this.z, this.y);\n }\n\n /**\n * Returns the distance of 1 meter in `MercatorCoordinate` units at this latitude.\n *\n * For coordinates in real world units using meters, this naturally provides the scale\n * to transform into `MercatorCoordinate`s.\n *\n * @returns Distance of 1 meter in `MercatorCoordinate` units.\n */\n meterInMercatorCoordinateUnits(): number {\n // 1 meter / circumference at equator in meters * Mercator projection scale factor at this latitude\n return 1 / earthCircumfrence * mercatorScale(latFromMercatorY(this.y));\n }\n}\n","import {LngLatBounds, LngLatBoundsLike} from '../geo/lng_lat_bounds';\nimport {mercatorXfromLng, mercatorYfromLat} from '../geo/mercator_coordinate';\n\nimport type {CanonicalTileID} from './tile_id';\n\nexport class TileBounds {\n bounds: LngLatBounds;\n minzoom: number;\n maxzoom: number;\n\n constructor(bounds: [number, number, number, number], minzoom?: number | null, maxzoom?: number | null) {\n this.bounds = LngLatBounds.convert(this.validateBounds(bounds));\n this.minzoom = minzoom || 0;\n this.maxzoom = maxzoom || 24;\n }\n\n validateBounds(bounds: [number, number, number, number]): LngLatBoundsLike {\n // make sure the bounds property contains valid longitude and latitudes\n if (!Array.isArray(bounds) || bounds.length !== 4) return [-180, -90, 180, 90];\n return [Math.max(-180, bounds[0]), Math.max(-90, bounds[1]), Math.min(180, bounds[2]), Math.min(90, bounds[3])];\n }\n\n contains(tileID: CanonicalTileID) {\n const worldSize = Math.pow(2, tileID.z);\n const level = {\n minX: Math.floor(mercatorXfromLng(this.bounds.getWest()) * worldSize),\n minY: Math.floor(mercatorYfromLat(this.bounds.getNorth()) * worldSize),\n maxX: Math.ceil(mercatorXfromLng(this.bounds.getEast()) * worldSize),\n maxY: Math.ceil(mercatorYfromLat(this.bounds.getSouth()) * worldSize)\n };\n const hit = tileID.x >= level.minX && tileID.x < level.maxX && tileID.y >= level.minY && tileID.y < level.maxY;\n return hit;\n }\n}\n","import {Event, ErrorEvent, Evented} from '../util/evented';\n\nimport {extend, pick} from '../util/util';\nimport {loadTileJson} from './load_tilejson';\nimport {TileBounds} from './tile_bounds';\nimport {ResourceType} from '../util/request_manager';\n\nimport type {Source} from './source';\nimport type {OverscaledTileID} from './tile_id';\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\nimport type {VectorSourceSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type VectorTileSourceOptions = VectorSourceSpecification & {\n collectResourceTiming?: boolean;\n}\n\n/**\n * A source containing vector tiles in [Mapbox Vector Tile format](https://docs.mapbox.com/vector-tiles/reference/).\n * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * map.addSource('some id', {\n * type: 'vector',\n * url: 'https://demotiles.maplibre.org/tiles/tiles.json'\n * });\n * ```\n *\n * @example\n * ```ts\n * map.addSource('some id', {\n * type: 'vector',\n * tiles: ['https://d25uarhxywzl1j.cloudfront.net/v0.1/{z}/{x}/{y}.mvt'],\n * minzoom: 6,\n * maxzoom: 14\n * });\n * ```\n *\n * @example\n * ```ts\n * map.getSource('some id').setUrl(\"https://demotiles.maplibre.org/tiles/tiles.json\");\n * ```\n *\n * @example\n * ```ts\n * map.getSource('some id').setTiles(['https://d25uarhxywzl1j.cloudfront.net/v0.1/{z}/{x}/{y}.mvt']);\n * ```\n * @see [Add a vector tile source](https://maplibre.org/maplibre-gl-js/docs/examples/vector-source/)\n */\nexport class VectorTileSource extends Evented implements Source {\n type: 'vector';\n id: string;\n minzoom: number;\n maxzoom: number;\n url: string;\n scheme: string;\n tileSize: number;\n promoteId: PromoteIdSpecification;\n\n _options: VectorSourceSpecification;\n _collectResourceTiming: boolean;\n dispatcher: Dispatcher;\n map: Map;\n bounds: [number, number, number, number];\n tiles: Array;\n tileBounds: TileBounds;\n reparseOverscaled: boolean;\n isTileClipped: boolean;\n _tileJSONRequest: Cancelable;\n _loaded: boolean;\n\n constructor(id: string, options: VectorTileSourceOptions, dispatcher: Dispatcher, eventedParent: Evented) {\n super();\n this.id = id;\n this.dispatcher = dispatcher;\n\n this.type = 'vector';\n this.minzoom = 0;\n this.maxzoom = 22;\n this.scheme = 'xyz';\n this.tileSize = 512;\n this.reparseOverscaled = true;\n this.isTileClipped = true;\n this._loaded = false;\n\n extend(this, pick(options, ['url', 'scheme', 'tileSize', 'promoteId']));\n this._options = extend({type: 'vector'}, options);\n\n this._collectResourceTiming = options.collectResourceTiming;\n\n if (this.tileSize !== 512) {\n throw new Error('vector tile sources must have a tileSize of 512');\n }\n\n this.setEventedParent(eventedParent);\n }\n\n load = () => {\n this._loaded = false;\n this.fire(new Event('dataloading', {dataType: 'source'}));\n this._tileJSONRequest = loadTileJson(this._options, this.map._requestManager, (err, tileJSON) => {\n this._tileJSONRequest = null;\n this._loaded = true;\n this.map.style.sourceCaches[this.id].clearTiles();\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (tileJSON) {\n extend(this, tileJSON);\n if (tileJSON.bounds) this.tileBounds = new TileBounds(tileJSON.bounds, this.minzoom, this.maxzoom);\n\n // `content` is included here to prevent a race condition where `Style#_updateSources` is called\n // before the TileJSON arrives. this makes sure the tiles needed are loaded once TileJSON arrives\n // ref: https://github.com/mapbox/mapbox-gl-js/pull/4347#discussion_r104418088\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'}));\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'}));\n }\n });\n };\n\n loaded(): boolean {\n return this._loaded;\n }\n\n hasTile(tileID: OverscaledTileID) {\n return !this.tileBounds || this.tileBounds.contains(tileID.canonical);\n }\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n }\n\n setSourceProperty(callback: Function) {\n if (this._tileJSONRequest) {\n this._tileJSONRequest.cancel();\n }\n\n callback();\n\n this.load();\n }\n\n /**\n * Sets the source `tiles` property and re-renders the map.\n *\n * @param tiles - An array of one or more tile source URLs, as in the TileJSON spec.\n * @returns `this`\n */\n setTiles(tiles: Array): this {\n this.setSourceProperty(() => {\n this._options.tiles = tiles;\n });\n\n return this;\n }\n\n /**\n * Sets the source `url` property and re-renders the map.\n *\n * @param url - A URL to a TileJSON resource. Supported protocols are `http:` and `https:`.\n * @returns `this`\n */\n setUrl(url: string): this {\n this.setSourceProperty(() => {\n this.url = url;\n this._options.url = url;\n });\n\n return this;\n }\n\n onRemove() {\n if (this._tileJSONRequest) {\n this._tileJSONRequest.cancel();\n this._tileJSONRequest = null;\n }\n }\n\n serialize = (): VectorSourceSpecification => {\n return extend({}, this._options);\n };\n\n loadTile(tile: Tile, callback: Callback) {\n const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme);\n const params = {\n request: this.map._requestManager.transformRequest(url, ResourceType.Tile),\n uid: tile.uid,\n tileID: tile.tileID,\n zoom: tile.tileID.overscaledZ,\n tileSize: this.tileSize * tile.tileID.overscaleFactor(),\n type: this.type,\n source: this.id,\n pixelRatio: this.map.getPixelRatio(),\n showCollisionBoxes: this.map.showCollisionBoxes,\n promoteId: this.promoteId\n };\n params.request.collectResourceTiming = this._collectResourceTiming;\n\n if (!tile.actor || tile.state === 'expired') {\n tile.actor = this.dispatcher.getActor();\n tile.request = tile.actor.send('loadTile', params, done.bind(this));\n } else if (tile.state === 'loading') {\n // schedule tile reloading after it has been loaded\n tile.reloadCallback = callback;\n } else {\n tile.request = tile.actor.send('reloadTile', params, done.bind(this));\n }\n\n function done(err, data) {\n delete tile.request;\n\n if (tile.aborted)\n return callback(null);\n\n if (err && err.status !== 404) {\n return callback(err);\n }\n\n if (data && data.resourceTiming)\n tile.resourceTiming = data.resourceTiming;\n\n if (this.map._refreshExpiredTiles && data) tile.setExpiryData(data);\n tile.loadVectorData(data, this.map.painter);\n\n callback(null);\n\n if (tile.reloadCallback) {\n this.loadTile(tile, tile.reloadCallback);\n tile.reloadCallback = null;\n }\n }\n }\n\n abortTile(tile: Tile) {\n if (tile.request) {\n tile.request.cancel();\n delete tile.request;\n }\n if (tile.actor) {\n tile.actor.send('abortTile', {uid: tile.uid, type: this.type, source: this.id}, undefined);\n }\n }\n\n unloadTile(tile: Tile) {\n tile.unloadVectorData();\n if (tile.actor) {\n tile.actor.send('removeTile', {uid: tile.uid, type: this.type, source: this.id}, undefined);\n }\n }\n\n hasTransition() {\n return false;\n }\n}\n","import {extend, pick} from '../util/util';\n\nimport {ImageRequest} from '../util/image_request';\n\nimport {ResourceType} from '../util/request_manager';\nimport {Event, ErrorEvent, Evented} from '../util/evented';\nimport {loadTileJson} from './load_tilejson';\nimport {TileBounds} from './tile_bounds';\nimport {Texture} from '../render/texture';\n\nimport type {Source} from './source';\nimport type {OverscaledTileID} from './tile_id';\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\nimport type {\n RasterSourceSpecification,\n RasterDEMSourceSpecification\n} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A source containing raster tiles (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * map.addSource('raster-source', {\n * 'type': 'raster',\n * 'tiles': ['https://tiles.stadiamaps.com/tiles/stamen_watercolor/{z}/{x}/{y}.jpg'],\n * 'tileSize': 256,\n * });\n * ```\n *\n * @example\n * ```ts\n * map.addSource('wms-test-source', {\n * 'type': 'raster',\n * // use the tiles option to specify a WMS tile source URL\n * 'tiles': [\n * 'https://img.nj.gov/imagerywms/Natural2015?bbox={bbox-epsg-3857}&format=image/png&service=WMS&version=1.1.1&request=GetMap&srs=EPSG:3857&transparent=true&width=256&height=256&layers=Natural2015'\n * ],\n * 'tileSize': 256\n * });\n * ```\n * @see [Add a raster tile source](https://maplibre.org/maplibre-gl-js/docs/examples/map-tiles/)\n * @see [Add a WMS source](https://maplibre.org/maplibre-gl-js/docs/examples/wms/)\n * @see [Display a satellite map](https://maplibre.org/maplibre-gl-js/docs/examples/satellite-map/)\n */\nexport class RasterTileSource extends Evented implements Source {\n type: 'raster' | 'raster-dem';\n id: string;\n minzoom: number;\n maxzoom: number;\n url: string;\n scheme: string;\n tileSize: number;\n\n bounds: [number, number, number, number];\n tileBounds: TileBounds;\n roundZoom: boolean;\n dispatcher: Dispatcher;\n map: Map;\n tiles: Array;\n\n _loaded: boolean;\n _options: RasterSourceSpecification | RasterDEMSourceSpecification;\n _tileJSONRequest: Cancelable;\n\n constructor(id: string, options: RasterSourceSpecification | RasterDEMSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super();\n this.id = id;\n this.dispatcher = dispatcher;\n this.setEventedParent(eventedParent);\n\n this.type = 'raster';\n this.minzoom = 0;\n this.maxzoom = 22;\n this.roundZoom = true;\n this.scheme = 'xyz';\n this.tileSize = 512;\n this._loaded = false;\n\n this._options = extend({type: 'raster'}, options);\n extend(this, pick(options, ['url', 'scheme', 'tileSize']));\n }\n\n load() {\n this._loaded = false;\n this.fire(new Event('dataloading', {dataType: 'source'}));\n this._tileJSONRequest = loadTileJson(this._options, this.map._requestManager, (err, tileJSON) => {\n this._tileJSONRequest = null;\n this._loaded = true;\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (tileJSON) {\n extend(this, tileJSON);\n if (tileJSON.bounds) this.tileBounds = new TileBounds(tileJSON.bounds, this.minzoom, this.maxzoom);\n\n // `content` is included here to prevent a race condition where `Style#_updateSources` is called\n // before the TileJSON arrives. this makes sure the tiles needed are loaded once TileJSON arrives\n // ref: https://github.com/mapbox/mapbox-gl-js/pull/4347#discussion_r104418088\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'}));\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'}));\n }\n });\n }\n\n loaded(): boolean {\n return this._loaded;\n }\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n }\n\n onRemove() {\n if (this._tileJSONRequest) {\n this._tileJSONRequest.cancel();\n this._tileJSONRequest = null;\n }\n }\n\n setSourceProperty(callback: Function) {\n if (this._tileJSONRequest) {\n this._tileJSONRequest.cancel();\n }\n\n callback();\n\n this.load();\n }\n\n /**\n * Sets the source `tiles` property and re-renders the map.\n *\n * @param tiles - An array of one or more tile source URLs, as in the raster tiles spec (See the [Style Specification](https://maplibre.org/maplibre-style-spec/)\n * @returns `this`\n */\n setTiles(tiles: Array): this {\n this.setSourceProperty(() => {\n this._options.tiles = tiles;\n });\n\n return this;\n }\n\n serialize() {\n return extend({}, this._options);\n }\n\n hasTile(tileID: OverscaledTileID) {\n return !this.tileBounds || this.tileBounds.contains(tileID.canonical);\n }\n\n loadTile(tile: Tile, callback: Callback) {\n const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme);\n tile.request = ImageRequest.getImage(this.map._requestManager.transformRequest(url, ResourceType.Tile), (err, img, expiry) => {\n delete tile.request;\n\n if (tile.aborted) {\n tile.state = 'unloaded';\n callback(null);\n } else if (err) {\n tile.state = 'errored';\n callback(err);\n } else if (img) {\n if (this.map._refreshExpiredTiles && expiry) tile.setExpiryData(expiry);\n\n const context = this.map.painter.context;\n const gl = context.gl;\n tile.texture = this.map.painter.getTileTexture(img.width);\n if (tile.texture) {\n tile.texture.update(img, {useMipmap: true});\n } else {\n tile.texture = new Texture(context, img, gl.RGBA, {useMipmap: true});\n tile.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST);\n\n if (context.extTextureFilterAnisotropic) {\n gl.texParameterf(gl.TEXTURE_2D, context.extTextureFilterAnisotropic.TEXTURE_MAX_ANISOTROPY_EXT, context.extTextureFilterAnisotropicMax);\n }\n }\n\n tile.state = 'loaded';\n\n callback(null);\n }\n }, this.map._refreshExpiredTiles);\n }\n\n abortTile(tile: Tile, callback: Callback) {\n if (tile.request) {\n tile.request.cancel();\n delete tile.request;\n }\n callback();\n }\n\n unloadTile(tile: Tile, callback: Callback) {\n if (tile.texture) this.map.painter.saveTileTexture(tile.texture);\n callback();\n }\n\n hasTransition() {\n return false;\n }\n}\n","export { getURL, getTileBBox, getMercCoords };\n\n\n/**\n * getURL\n *\n * @param {String} baseUrl Base url of the WMS server\n * @param {String} layer Layer name\n * @param {Number} x Tile coordinate x\n * @param {Number} y Tile coordinate y\n * @param {Number} z Tile zoom\n * @param {Object} [options]\n * @param {String} [options.format='image/png']\n * @param {String} [options.service='WMS']\n * @param {String} [options.version='1.1.1']\n * @param {String} [options.request='GetMap']\n * @param {String} [options.srs='EPSG:3857']\n * @param {Number} [options.width='256']\n * @param {Number} [options.height='256']\n * @returns {String} url\n * @example\n * var baseUrl = 'http://geodata.state.nj.us/imagerywms/Natural2015';\n * var layer = 'Natural2015';\n * var url = whoots.getURL(baseUrl, layer, 154308, 197167, 19);\n */\nfunction getURL(baseUrl, layer, x, y, z, options) {\n options = options || {};\n\n var url = baseUrl + '?' + [\n 'bbox=' + getTileBBox(x, y, z),\n 'format=' + (options.format || 'image/png'),\n 'service=' + (options.service || 'WMS'),\n 'version=' + (options.version || '1.1.1'),\n 'request=' + (options.request || 'GetMap'),\n 'srs=' + (options.srs || 'EPSG:3857'),\n 'width=' + (options.width || 256),\n 'height=' + (options.height || 256),\n 'layers=' + layer\n ].join('&');\n\n return url;\n}\n\n\n/**\n * getTileBBox\n *\n * @param {Number} x Tile coordinate x\n * @param {Number} y Tile coordinate y\n * @param {Number} z Tile zoom\n * @returns {String} String of the bounding box\n */\nfunction getTileBBox(x, y, z) {\n // for Google/OSM tile scheme we need to alter the y\n y = (Math.pow(2, z) - y - 1);\n\n var min = getMercCoords(x * 256, y * 256, z),\n max = getMercCoords((x + 1) * 256, (y + 1) * 256, z);\n\n return min[0] + ',' + min[1] + ',' + max[0] + ',' + max[1];\n}\n\n\n/**\n * getMercCoords\n *\n * @param {Number} x Pixel coordinate x\n * @param {Number} y Pixel coordinate y\n * @param {Number} z Tile zoom\n * @returns {Array} [x, y]\n */\nfunction getMercCoords(x, y, z) {\n var resolution = (2 * Math.PI * 6378137 / 256) / Math.pow(2, z),\n merc_x = (x * resolution - 2 * Math.PI * 6378137 / 2.0),\n merc_y = (y * resolution - 2 * Math.PI * 6378137 / 2.0);\n\n return [merc_x, merc_y];\n}\n","import {getTileBBox} from '@mapbox/whoots-js';\nimport {EXTENT} from '../data/extent';\nimport Point from '@mapbox/point-geometry';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\nimport {register} from '../util/web_worker_transfer';\nimport {mat4} from 'gl-matrix';\nimport {ICanonicalTileID, IMercatorCoordinate} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A canonical way to define a tile ID\n */\nexport class CanonicalTileID implements ICanonicalTileID {\n z: number;\n x: number;\n y: number;\n key: string;\n\n constructor(z: number, x: number, y: number) {\n\n if (z < 0 || z > 25 || y < 0 || y >= Math.pow(2, z) || x < 0 || x >= Math.pow(2, z)) {\n throw new Error(`x=${x}, y=${y}, z=${z} outside of bounds. 0<=x<${Math.pow(2, z)}, 0<=y<${Math.pow(2, z)} 0<=z<=25 `);\n }\n\n this.z = z;\n this.x = x;\n this.y = y;\n this.key = calculateKey(0, z, z, x, y);\n }\n\n equals(id: ICanonicalTileID) {\n return this.z === id.z && this.x === id.x && this.y === id.y;\n }\n\n // given a list of urls, choose a url template and return a tile URL\n url(urls: Array, pixelRatio: number, scheme?: string | null) {\n const bbox = getTileBBox(this.x, this.y, this.z);\n const quadkey = getQuadkey(this.z, this.x, this.y);\n\n return urls[(this.x + this.y) % urls.length]\n .replace(/{prefix}/g, (this.x % 16).toString(16) + (this.y % 16).toString(16))\n .replace(/{z}/g, String(this.z))\n .replace(/{x}/g, String(this.x))\n .replace(/{y}/g, String(scheme === 'tms' ? (Math.pow(2, this.z) - this.y - 1) : this.y))\n .replace(/{ratio}/g, pixelRatio > 1 ? '@2x' : '')\n .replace(/{quadkey}/g, quadkey)\n .replace(/{bbox-epsg-3857}/g, bbox);\n }\n\n isChildOf(parent: ICanonicalTileID) {\n const dz = this.z - parent.z;\n return dz > 0 && parent.x === (this.x >> dz) && parent.y === (this.y >> dz);\n }\n\n getTilePoint(coord: IMercatorCoordinate) {\n const tilesAtZoom = Math.pow(2, this.z);\n return new Point(\n (coord.x * tilesAtZoom - this.x) * EXTENT,\n (coord.y * tilesAtZoom - this.y) * EXTENT);\n }\n\n toString() {\n return `${this.z}/${this.x}/${this.y}`;\n }\n}\n\n/**\n * @internal\n * An unwrapped tile identifier\n */\nexport class UnwrappedTileID {\n wrap: number;\n canonical: CanonicalTileID;\n key: string;\n\n constructor(wrap: number, canonical: CanonicalTileID) {\n this.wrap = wrap;\n this.canonical = canonical;\n this.key = calculateKey(wrap, canonical.z, canonical.z, canonical.x, canonical.y);\n }\n}\n\n/**\n * An overscaled tile identifier\n */\nexport class OverscaledTileID {\n overscaledZ: number;\n wrap: number;\n canonical: CanonicalTileID;\n key: string;\n posMatrix: mat4;\n\n constructor(overscaledZ: number, wrap: number, z: number, x: number, y: number) {\n if (overscaledZ < z) throw new Error(`overscaledZ should be >= z; overscaledZ = ${overscaledZ}; z = ${z}`);\n this.overscaledZ = overscaledZ;\n this.wrap = wrap;\n this.canonical = new CanonicalTileID(z, +x, +y);\n this.key = calculateKey(wrap, overscaledZ, z, x, y);\n }\n\n clone() {\n return new OverscaledTileID(this.overscaledZ, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y);\n }\n\n equals(id: OverscaledTileID) {\n return this.overscaledZ === id.overscaledZ && this.wrap === id.wrap && this.canonical.equals(id.canonical);\n }\n\n scaledTo(targetZ: number) {\n if (targetZ > this.overscaledZ) throw new Error(`targetZ > this.overscaledZ; targetZ = ${targetZ}; overscaledZ = ${this.overscaledZ}`);\n const zDifference = this.canonical.z - targetZ;\n if (targetZ > this.canonical.z) {\n return new OverscaledTileID(targetZ, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y);\n } else {\n return new OverscaledTileID(targetZ, this.wrap, targetZ, this.canonical.x >> zDifference, this.canonical.y >> zDifference);\n }\n }\n\n /*\n * calculateScaledKey is an optimization:\n * when withWrap == true, implements the same as this.scaledTo(z).key,\n * when withWrap == false, implements the same as this.scaledTo(z).wrapped().key.\n */\n calculateScaledKey(targetZ: number, withWrap: boolean): string {\n if (targetZ > this.overscaledZ) throw new Error(`targetZ > this.overscaledZ; targetZ = ${targetZ}; overscaledZ = ${this.overscaledZ}`);\n const zDifference = this.canonical.z - targetZ;\n if (targetZ > this.canonical.z) {\n return calculateKey(this.wrap * +withWrap, targetZ, this.canonical.z, this.canonical.x, this.canonical.y);\n } else {\n return calculateKey(this.wrap * +withWrap, targetZ, targetZ, this.canonical.x >> zDifference, this.canonical.y >> zDifference);\n }\n }\n\n isChildOf(parent: OverscaledTileID) {\n if (parent.wrap !== this.wrap) {\n // We can't be a child if we're in a different world copy\n return false;\n }\n const zDifference = this.canonical.z - parent.canonical.z;\n // We're first testing for z == 0, to avoid a 32 bit shift, which is undefined.\n return parent.overscaledZ === 0 || (\n parent.overscaledZ < this.overscaledZ &&\n parent.canonical.x === (this.canonical.x >> zDifference) &&\n parent.canonical.y === (this.canonical.y >> zDifference));\n }\n\n children(sourceMaxZoom: number) {\n if (this.overscaledZ >= sourceMaxZoom) {\n // return a single tile coord representing a an overscaled tile\n return [new OverscaledTileID(this.overscaledZ + 1, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y)];\n }\n\n const z = this.canonical.z + 1;\n const x = this.canonical.x * 2;\n const y = this.canonical.y * 2;\n return [\n new OverscaledTileID(z, this.wrap, z, x, y),\n new OverscaledTileID(z, this.wrap, z, x + 1, y),\n new OverscaledTileID(z, this.wrap, z, x, y + 1),\n new OverscaledTileID(z, this.wrap, z, x + 1, y + 1)\n ];\n }\n\n isLessThan(rhs: OverscaledTileID) {\n if (this.wrap < rhs.wrap) return true;\n if (this.wrap > rhs.wrap) return false;\n\n if (this.overscaledZ < rhs.overscaledZ) return true;\n if (this.overscaledZ > rhs.overscaledZ) return false;\n\n if (this.canonical.x < rhs.canonical.x) return true;\n if (this.canonical.x > rhs.canonical.x) return false;\n\n if (this.canonical.y < rhs.canonical.y) return true;\n return false;\n }\n\n wrapped() {\n return new OverscaledTileID(this.overscaledZ, 0, this.canonical.z, this.canonical.x, this.canonical.y);\n }\n\n unwrapTo(wrap: number) {\n return new OverscaledTileID(this.overscaledZ, wrap, this.canonical.z, this.canonical.x, this.canonical.y);\n }\n\n overscaleFactor() {\n return Math.pow(2, this.overscaledZ - this.canonical.z);\n }\n\n toUnwrapped() {\n return new UnwrappedTileID(this.wrap, this.canonical);\n }\n\n toString() {\n return `${this.overscaledZ}/${this.canonical.x}/${this.canonical.y}`;\n }\n\n getTilePoint(coord: MercatorCoordinate) {\n return this.canonical.getTilePoint(new MercatorCoordinate(coord.x - this.wrap, coord.y));\n }\n}\n\nfunction calculateKey(wrap: number, overscaledZ: number, z: number, x: number, y: number): string {\n wrap *= 2;\n if (wrap < 0) wrap = wrap * -1 - 1;\n const dim = 1 << z;\n return (dim * dim * wrap + dim * y + x).toString(36) + z.toString(36) + overscaledZ.toString(36);\n}\n\nfunction getQuadkey(z, x, y) {\n let quadkey = '', mask;\n for (let i = z; i > 0; i--) {\n mask = 1 << (i - 1);\n quadkey += ((x & mask ? 1 : 0) + (y & mask ? 2 : 0));\n }\n return quadkey;\n}\n\nregister('CanonicalTileID', CanonicalTileID);\nregister('OverscaledTileID', OverscaledTileID, {omit: ['posMatrix']});\n","import {RGBAImage} from '../util/image';\n\nimport {warnOnce} from '../util/util';\nimport {register} from '../util/web_worker_transfer';\n\n// DEMData is a data structure for decoding, backfilling, and storing elevation data for processing in the hillshade shaders\n// data can be populated either from a pngraw image tile or from serliazed data sent back from a worker. When data is initially\n// loaded from a image tile, we decode the pixel values using the appropriate decoding formula, but we store the\n// elevation data as an Int32 value. we add 65536 (2^16) to eliminate negative values and enable the use of\n// integer overflow when creating the texture used in the hillshadePrepare step.\n\n// DEMData also handles the backfilling of data from a tile's neighboring tiles. This is necessary because we use a pixel's 8\n// surrounding pixel values to compute the slope at that pixel, and we cannot accurately calculate the slope at pixels on a\n// tile's edge without backfilling from neighboring tiles.\n\nexport type DEMEncoding = 'mapbox' | 'terrarium' | 'custom'\n\nexport class DEMData {\n uid: string;\n data: Uint32Array;\n stride: number;\n dim: number;\n min: number;\n max: number;\n redFactor: number;\n greenFactor: number;\n blueFactor: number;\n baseShift: number;\n\n // RGBAImage data has uniform 1px padding on all sides: square tile edge size defines stride\n // and dim is calculated as stride - 2.\n constructor(uid: string, data: RGBAImage, encoding: DEMEncoding, redFactor = 1.0, greenFactor = 1.0, blueFactor = 1.0, baseShift = 0.0) {\n this.uid = uid;\n if (data.height !== data.width) throw new RangeError('DEM tiles must be square');\n if (encoding && !['mapbox', 'terrarium', 'custom'].includes(encoding)) {\n warnOnce(`\"${encoding}\" is not a valid encoding type. Valid types include \"mapbox\", \"terrarium\" and \"custom\".`);\n return;\n }\n this.stride = data.height;\n const dim = this.dim = data.height - 2;\n this.data = new Uint32Array(data.data.buffer);\n switch (encoding) {\n case 'terrarium':\n // unpacking formula for mapzen terrarium:\n // https://aws.amazon.com/public-datasets/terrain/\n this.redFactor = 256.0;\n this.greenFactor = 1.0;\n this.blueFactor = 1.0 / 256.0;\n this.baseShift = 32768.0;\n break;\n case 'custom':\n this.redFactor = redFactor;\n this.greenFactor = greenFactor;\n this.blueFactor = blueFactor;\n this.baseShift = baseShift;\n break;\n case 'mapbox':\n default:\n // unpacking formula for mapbox.terrain-rgb:\n // https://www.mapbox.com/help/access-elevation-data/#mapbox-terrain-rgb\n this.redFactor = 6553.6;\n this.greenFactor = 25.6;\n this.blueFactor = 0.1;\n this.baseShift = 10000.0;\n break;\n }\n\n // in order to avoid flashing seams between tiles, here we are initially populating a 1px border of pixels around the image\n // with the data of the nearest pixel from the image. this data is eventually replaced when the tile's neighboring\n // tiles are loaded and the accurate data can be backfilled using DEMData#backfillBorder\n for (let x = 0; x < dim; x++) {\n // left vertical border\n this.data[this._idx(-1, x)] = this.data[this._idx(0, x)];\n // right vertical border\n this.data[this._idx(dim, x)] = this.data[this._idx(dim - 1, x)];\n // left horizontal border\n this.data[this._idx(x, -1)] = this.data[this._idx(x, 0)];\n // right horizontal border\n this.data[this._idx(x, dim)] = this.data[this._idx(x, dim - 1)];\n }\n // corners\n this.data[this._idx(-1, -1)] = this.data[this._idx(0, 0)];\n this.data[this._idx(dim, -1)] = this.data[this._idx(dim - 1, 0)];\n this.data[this._idx(-1, dim)] = this.data[this._idx(0, dim - 1)];\n this.data[this._idx(dim, dim)] = this.data[this._idx(dim - 1, dim - 1)];\n\n // calculate min/max values\n this.min = Number.MAX_SAFE_INTEGER;\n this.max = Number.MIN_SAFE_INTEGER;\n for (let x = 0; x < dim; x++) {\n for (let y = 0; y < dim; y++) {\n const ele = this.get(x, y);\n if (ele > this.max) this.max = ele;\n if (ele < this.min) this.min = ele;\n }\n }\n }\n\n get(x: number, y: number) {\n const pixels = new Uint8Array(this.data.buffer);\n const index = this._idx(x, y) * 4;\n return this.unpack(pixels[index], pixels[index + 1], pixels[index + 2]);\n }\n\n getUnpackVector() {\n return [this.redFactor, this.greenFactor, this.blueFactor, this.baseShift];\n }\n\n _idx(x: number, y: number) {\n if (x < -1 || x >= this.dim + 1 || y < -1 || y >= this.dim + 1) throw new RangeError('out of range source coordinates for DEM data');\n return (y + 1) * this.stride + (x + 1);\n }\n\n unpack(r: number, g: number, b: number) {\n return (r * this.redFactor + g * this.greenFactor + b * this.blueFactor - this.baseShift);\n }\n\n getPixels() {\n return new RGBAImage({width: this.stride, height: this.stride}, new Uint8Array(this.data.buffer));\n }\n\n backfillBorder(borderTile: DEMData, dx: number, dy: number) {\n if (this.dim !== borderTile.dim) throw new Error('dem dimension mismatch');\n\n let xMin = dx * this.dim,\n xMax = dx * this.dim + this.dim,\n yMin = dy * this.dim,\n yMax = dy * this.dim + this.dim;\n\n switch (dx) {\n case -1:\n xMin = xMax - 1;\n break;\n case 1:\n xMax = xMin + 1;\n break;\n }\n\n switch (dy) {\n case -1:\n yMin = yMax - 1;\n break;\n case 1:\n yMax = yMin + 1;\n break;\n }\n\n const ox = -dx * this.dim;\n const oy = -dy * this.dim;\n for (let y = yMin; y < yMax; y++) {\n for (let x = xMin; x < xMax; x++) {\n this.data[this._idx(x, y)] = borderTile.data[this._idx(x + ox, y + oy)];\n }\n }\n }\n}\n\nregister('DEMData', DEMData);\n","import {ImageRequest} from '../util/image_request';\nimport {ResourceType} from '../util/request_manager';\nimport {extend, isImageBitmap, readImageUsingVideoFrame} from '../util/util';\nimport {Evented} from '../util/evented';\nimport {browser} from '../util/browser';\nimport {offscreenCanvasSupported} from '../util/offscreen_canvas_supported';\nimport {OverscaledTileID} from './tile_id';\nimport {RasterTileSource} from './raster_tile_source';\n// ensure DEMData is registered for worker transfer on main thread:\nimport '../data/dem_data';\nimport type {DEMEncoding} from '../data/dem_data';\n\nimport type {Source} from './source';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {RasterDEMSourceSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {ExpiryData} from '../util/ajax';\nimport {isOffscreenCanvasDistorted} from '../util/offscreen_canvas_distorted';\nimport {RGBAImage} from '../util/image';\n\n/**\n * A source containing raster DEM tiles (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.)\n * This source can be used to show hillshading and 3D terrain\n *\n * @group Sources\n *\n * @example\n * ```ts\n * map.addSource('raster-dem-source', {\n * type: 'raster-dem',\n * url: 'https://demotiles.maplibre.org/terrain-tiles/tiles.json',\n * tileSize: 256\n * });\n * ```\n * @see [3D Terrain](https://maplibre.org/maplibre-gl-js/docs/examples/3d-terrain/)\n */\nexport class RasterDEMTileSource extends RasterTileSource implements Source {\n encoding: DEMEncoding;\n redFactor?: number;\n greenFactor?: number;\n blueFactor?: number;\n baseShift?: number;\n\n constructor(id: string, options: RasterDEMSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super(id, options, dispatcher, eventedParent);\n this.type = 'raster-dem';\n this.maxzoom = 22;\n this._options = extend({type: 'raster-dem'}, options);\n this.encoding = options.encoding || 'mapbox';\n this.redFactor = options.redFactor;\n this.greenFactor = options.greenFactor;\n this.blueFactor = options.blueFactor;\n this.baseShift = options.baseShift;\n }\n\n loadTile(tile: Tile, callback: Callback) {\n const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme);\n const request = this.map._requestManager.transformRequest(url, ResourceType.Tile);\n tile.neighboringTiles = this._getNeighboringTiles(tile.tileID);\n tile.request = ImageRequest.getImage(request, async (err: Error, img: (HTMLImageElement | ImageBitmap), expiry: ExpiryData) => {\n delete tile.request;\n if (tile.aborted) {\n tile.state = 'unloaded';\n callback(null);\n } else if (err) {\n tile.state = 'errored';\n callback(err);\n } else if (img) {\n if (this.map._refreshExpiredTiles) tile.setExpiryData(expiry);\n const transfer = isImageBitmap(img) && offscreenCanvasSupported();\n const rawImageData = transfer ? img : await readImageNow(img);\n const params = {\n uid: tile.uid,\n coord: tile.tileID,\n source: this.id,\n rawImageData,\n encoding: this.encoding,\n redFactor: this.redFactor,\n greenFactor: this.greenFactor,\n blueFactor: this.blueFactor,\n baseShift: this.baseShift\n };\n\n if (!tile.actor || tile.state === 'expired') {\n tile.actor = this.dispatcher.getActor();\n tile.actor.send('loadDEMTile', params, done);\n }\n }\n }, this.map._refreshExpiredTiles);\n\n async function readImageNow(img: ImageBitmap | HTMLImageElement): Promise {\n if (typeof VideoFrame !== 'undefined' && isOffscreenCanvasDistorted()) {\n const width = img.width + 2;\n const height = img.height + 2;\n try {\n return new RGBAImage({width, height}, await readImageUsingVideoFrame(img, -1, -1, width, height));\n } catch (e) {\n // fall-back to browser canvas decoding\n }\n }\n return browser.getImageData(img, 1);\n }\n\n function done(err, data) {\n if (err) {\n tile.state = 'errored';\n callback(err);\n }\n\n if (data) {\n tile.dem = data;\n tile.needsHillshadePrepare = true;\n tile.needsTerrainPrepare = true;\n tile.state = 'loaded';\n callback(null);\n }\n }\n }\n\n _getNeighboringTiles(tileID: OverscaledTileID) {\n const canonical = tileID.canonical;\n const dim = Math.pow(2, canonical.z);\n\n const px = (canonical.x - 1 + dim) % dim;\n const pxw = canonical.x === 0 ? tileID.wrap - 1 : tileID.wrap;\n const nx = (canonical.x + 1 + dim) % dim;\n const nxw = canonical.x + 1 === dim ? tileID.wrap + 1 : tileID.wrap;\n\n const neighboringTiles = {};\n // add adjacent tiles\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y).key] = {backfilled: false};\n\n // Add upper neighboringTiles\n if (canonical.y > 0) {\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y - 1).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, tileID.wrap, canonical.z, canonical.x, canonical.y - 1).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y - 1).key] = {backfilled: false};\n }\n // Add lower neighboringTiles\n if (canonical.y + 1 < dim) {\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y + 1).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, tileID.wrap, canonical.z, canonical.x, canonical.y + 1).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y + 1).key] = {backfilled: false};\n }\n\n return neighboringTiles;\n }\n\n unloadTile(tile: Tile) {\n if (tile.demTexture) this.map.painter.saveTileTexture(tile.demTexture);\n if (tile.fbo) {\n tile.fbo.destroy();\n delete tile.fbo;\n }\n if (tile.dem) delete tile.dem;\n delete tile.neighboringTiles;\n\n tile.state = 'unloaded';\n if (tile.actor) {\n tile.actor.send('removeDEMTile', {uid: tile.uid, source: this.id});\n }\n }\n}\n","import {Event, ErrorEvent, Evented} from '../util/evented';\n\nimport {extend} from '../util/util';\nimport {EXTENT} from '../data/extent';\nimport {ResourceType} from '../util/request_manager';\nimport {browser} from '../util/browser';\n\nimport type {Source} from './source';\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Actor} from '../util/actor';\nimport type {Callback} from '../types/callback';\nimport type {GeoJSONSourceSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {GeoJSONSourceDiff} from './geojson_source_diff';\nimport type {Options, ClusterProperties} from 'supercluster';\n\nexport type GeoJSONSourceOptions = GeoJSONSourceSpecification & {\n workerOptions?: WorkerOptions;\n collectResourceTiming?: boolean;\n}\n\nexport type GeoJsonSourceOptions = {\n data?: GeoJSON.GeoJSON | string | undefined;\n cluster?: boolean;\n clusterMaxZoom?: number;\n clusterRadius?: number;\n clusterMinPoints?: number;\n generateId?: boolean;\n}\nexport type WorkerOptions = {\n source?: string;\n cluster?: boolean;\n geojsonVtOptions?: {\n buffer?: number;\n tolerance?: number;\n extent?: number;\n maxZoom?: number;\n linemetrics?: boolean;\n generateId?: boolean;\n };\n superclusterOptions?: Options;\n clusterProperties?: ClusterProperties;\n fliter?: any;\n promoteId?: any;\n collectResourceTiming?: boolean;\n}\n\n/**\n * The cluster options to set\n */\nexport type SetClusterOptions = {\n /**\n * Whether or not to cluster\n */\n cluster?: boolean;\n /**\n * The cluster's max zoom\n */\n clusterMaxZoom?: number;\n /**\n * The cluster's radius\n */\n clusterRadius?: number;\n}\n\n/**\n * A source containing GeoJSON.\n * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-geojson) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * map.addSource('some id', {\n * type: 'geojson',\n * data: 'https://d2ad6b4ur7yvpq.cloudfront.net/naturalearth-3.3.0/ne_10m_ports.geojson'\n * });\n * ```\n *\n * @example\n * ```ts\n * map.addSource('some id', {\n * type: 'geojson',\n * data: {\n * \"type\": \"FeatureCollection\",\n * \"features\": [{\n * \"type\": \"Feature\",\n * \"properties\": {},\n * \"geometry\": {\n * \"type\": \"Point\",\n * \"coordinates\": [\n * -76.53063297271729,\n * 39.18174077994108\n * ]\n * }\n * }]\n * }\n * });\n * ```\n *\n * @example\n * ```ts\n * map.getSource('some id').setData({\n * \"type\": \"FeatureCollection\",\n * \"features\": [{\n * \"type\": \"Feature\",\n * \"properties\": { \"name\": \"Null Island\" },\n * \"geometry\": {\n * \"type\": \"Point\",\n * \"coordinates\": [ 0, 0 ]\n * }\n * }]\n * });\n * ```\n * @see [Draw GeoJSON points](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-markers/)\n * @see [Add a GeoJSON line](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-line/)\n * @see [Create a heatmap from points](https://maplibre.org/maplibre-gl-js/docs/examples/heatmap/)\n * @see [Create and style clusters](https://maplibre.org/maplibre-gl-js/docs/examples/cluster/)\n */\nexport class GeoJSONSource extends Evented implements Source {\n type: 'geojson';\n id: string;\n minzoom: number;\n maxzoom: number;\n tileSize: number;\n attribution: string;\n promoteId: PromoteIdSpecification;\n\n isTileClipped: boolean;\n reparseOverscaled: boolean;\n _data: GeoJSON.GeoJSON | string | undefined;\n _options: GeoJsonSourceOptions;\n workerOptions: WorkerOptions;\n map: Map;\n actor: Actor;\n _pendingLoads: number;\n _collectResourceTiming: boolean;\n _removed: boolean;\n\n /** @internal */\n constructor(id: string, options: GeoJSONSourceOptions, dispatcher: Dispatcher, eventedParent: Evented) {\n super();\n\n this.id = id;\n\n // `type` is a property rather than a constant to make it easy for 3rd\n // parties to use GeoJSONSource to build their own source types.\n this.type = 'geojson';\n\n this.minzoom = 0;\n this.maxzoom = 18;\n this.tileSize = 512;\n this.isTileClipped = true;\n this.reparseOverscaled = true;\n this._removed = false;\n this._pendingLoads = 0;\n\n this.actor = dispatcher.getActor();\n this.setEventedParent(eventedParent);\n\n this._data = (options.data as any);\n this._options = extend({}, options);\n\n this._collectResourceTiming = options.collectResourceTiming;\n\n if (options.maxzoom !== undefined) this.maxzoom = options.maxzoom;\n if (options.type) this.type = options.type;\n if (options.attribution) this.attribution = options.attribution;\n this.promoteId = options.promoteId;\n\n const scale = EXTENT / this.tileSize;\n\n // sent to the worker, along with `url: ...` or `data: literal geojson`,\n // so that it can load/parse/index the geojson data\n // extending with `options.workerOptions` helps to make it easy for\n // third-party sources to hack/reuse GeoJSONSource.\n this.workerOptions = extend({\n source: this.id,\n cluster: options.cluster || false,\n geojsonVtOptions: {\n buffer: (options.buffer !== undefined ? options.buffer : 128) * scale,\n tolerance: (options.tolerance !== undefined ? options.tolerance : 0.375) * scale,\n extent: EXTENT,\n maxZoom: this.maxzoom,\n lineMetrics: options.lineMetrics || false,\n generateId: options.generateId || false\n },\n superclusterOptions: {\n maxZoom: options.clusterMaxZoom !== undefined ? options.clusterMaxZoom : this.maxzoom - 1,\n minPoints: Math.max(2, options.clusterMinPoints || 2),\n extent: EXTENT,\n radius: (options.clusterRadius || 50) * scale,\n log: false,\n generateId: options.generateId || false\n },\n clusterProperties: options.clusterProperties,\n filter: options.filter\n }, options.workerOptions);\n\n // send the promoteId to the worker to have more flexible updates, but only if it is a string\n if (typeof this.promoteId === 'string') {\n this.workerOptions.promoteId = this.promoteId;\n }\n }\n\n load = () => {\n this._updateWorkerData();\n };\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n }\n\n /**\n * Sets the GeoJSON data and re-renders the map.\n *\n * @param data - A GeoJSON data object or a URL to one. The latter is preferable in the case of large GeoJSON files.\n * @returns `this`\n */\n setData(data: GeoJSON.GeoJSON | string): this {\n this._data = data;\n this._updateWorkerData();\n\n return this;\n }\n\n /**\n * Updates the source's GeoJSON, and re-renders the map.\n *\n * For sources with lots of features, this method can be used to make updates more quickly.\n *\n * This approach requires unique IDs for every feature in the source. The IDs can either be specified on the feature,\n * or by using the promoteId option to specify which property should be used as the ID.\n *\n * It is an error to call updateData on a source that did not have unique IDs for each of its features already.\n *\n * Updates are applied on a best-effort basis, updating an ID that does not exist will not result in an error.\n *\n * @param diff - The changes that need to be applied.\n * @returns `this`\n */\n updateData(diff: GeoJSONSourceDiff): this {\n this._updateWorkerData(diff);\n\n return this;\n }\n\n /**\n * To disable/enable clustering on the source options\n * @param options - The options to set\n * @returns `this`\n * @example\n * ```ts\n * map.getSource('some id').setClusterOptions({cluster: false});\n * map.getSource('some id').setClusterOptions({cluster: false, clusterRadius: 50, clusterMaxZoom: 14});\n * ```\n */\n setClusterOptions(options: SetClusterOptions): this {\n this.workerOptions.cluster = options.cluster;\n if (options) {\n if (options.clusterRadius !== undefined) this.workerOptions.superclusterOptions.radius = options.clusterRadius;\n if (options.clusterMaxZoom !== undefined) this.workerOptions.superclusterOptions.maxZoom = options.clusterMaxZoom;\n }\n this._updateWorkerData();\n return this;\n }\n\n /**\n * For clustered sources, fetches the zoom at which the given cluster expands.\n *\n * @param clusterId - The value of the cluster's `cluster_id` property.\n * @param callback - A callback to be called when the zoom value is retrieved (`(error, zoom) => { ... }`).\n * @returns `this`\n */\n getClusterExpansionZoom(clusterId: number, callback: Callback): this {\n this.actor.send('geojson.getClusterExpansionZoom', {clusterId, source: this.id}, callback);\n return this;\n }\n\n /**\n * For clustered sources, fetches the children of the given cluster on the next zoom level (as an array of GeoJSON features).\n *\n * @param clusterId - The value of the cluster's `cluster_id` property.\n * @param callback - A callback to be called when the features are retrieved (`(error, features) => { ... }`).\n * @returns `this`\n */\n getClusterChildren(clusterId: number, callback: Callback>): this {\n this.actor.send('geojson.getClusterChildren', {clusterId, source: this.id}, callback);\n return this;\n }\n\n /**\n * For clustered sources, fetches the original points that belong to the cluster (as an array of GeoJSON features).\n *\n * @param clusterId - The value of the cluster's `cluster_id` property.\n * @param limit - The maximum number of features to return.\n * @param offset - The number of features to skip (e.g. for pagination).\n * @param callback - A callback to be called when the features are retrieved (`(error, features) => { ... }`).\n * @returns `this`\n * @example\n * Retrieve cluster leaves on click\n * ```ts\n * map.on('click', 'clusters', function(e) {\n * let features = map.queryRenderedFeatures(e.point, {\n * layers: ['clusters']\n * });\n *\n * let clusterId = features[0].properties.cluster_id;\n * let pointCount = features[0].properties.point_count;\n * let clusterSource = map.getSource('clusters');\n *\n * clusterSource.getClusterLeaves(clusterId, pointCount, 0, function(error, features) {\n * // Print cluster leaves in the console\n * console.log('Cluster leaves:', error, features);\n * })\n * });\n * ```\n */\n getClusterLeaves(clusterId: number, limit: number, offset: number, callback: Callback>): this {\n this.actor.send('geojson.getClusterLeaves', {\n source: this.id,\n clusterId,\n limit,\n offset\n }, callback);\n return this;\n }\n\n /**\n * Responsible for invoking WorkerSource's geojson.loadData target, which\n * handles loading the geojson data and preparing to serve it up as tiles,\n * using geojson-vt or supercluster as appropriate.\n * @param diff - the diff object\n */\n _updateWorkerData(diff?: GeoJSONSourceDiff) {\n const options = extend({}, this.workerOptions);\n if (diff) {\n options.dataDiff = diff;\n } else if (typeof this._data === 'string') {\n options.request = this.map._requestManager.transformRequest(browser.resolveURL(this._data as string), ResourceType.Source);\n options.request.collectResourceTiming = this._collectResourceTiming;\n } else {\n options.data = JSON.stringify(this._data);\n }\n\n this._pendingLoads++;\n this.fire(new Event('dataloading', {dataType: 'source'}));\n\n // target {this.type}.loadData rather than literally geojson.loadData,\n // so that other geojson-like source types can easily reuse this\n // implementation\n this.actor.send(`${this.type}.loadData`, options, (err, result) => {\n this._pendingLoads--;\n\n if (this._removed || (result && result.abandoned)) {\n this.fire(new Event('dataabort', {dataType: 'source'}));\n return;\n }\n\n let resourceTiming = null;\n if (result && result.resourceTiming && result.resourceTiming[this.id])\n resourceTiming = result.resourceTiming[this.id].slice(0);\n\n if (err) {\n this.fire(new ErrorEvent(err));\n return;\n }\n\n const data: any = {dataType: 'source'};\n if (this._collectResourceTiming && resourceTiming && resourceTiming.length > 0)\n extend(data, {resourceTiming});\n\n // although GeoJSON sources contain no metadata, we fire this event to let the SourceCache\n // know its ok to start requesting tiles.\n this.fire(new Event('data', {...data, sourceDataType: 'metadata'}));\n this.fire(new Event('data', {...data, sourceDataType: 'content'}));\n });\n }\n\n loaded(): boolean {\n return this._pendingLoads === 0;\n }\n\n loadTile(tile: Tile, callback: Callback) {\n const message = !tile.actor ? 'loadTile' : 'reloadTile';\n tile.actor = this.actor;\n const params = {\n type: this.type,\n uid: tile.uid,\n tileID: tile.tileID,\n zoom: tile.tileID.overscaledZ,\n maxZoom: this.maxzoom,\n tileSize: this.tileSize,\n source: this.id,\n pixelRatio: this.map.getPixelRatio(),\n showCollisionBoxes: this.map.showCollisionBoxes,\n promoteId: this.promoteId\n };\n\n tile.request = this.actor.send(message, params, (err, data) => {\n delete tile.request;\n tile.unloadVectorData();\n\n if (tile.aborted) {\n return callback(null);\n }\n\n if (err) {\n return callback(err);\n }\n\n tile.loadVectorData(data, this.map.painter, message === 'reloadTile');\n\n return callback(null);\n });\n }\n\n abortTile(tile: Tile) {\n if (tile.request) {\n tile.request.cancel();\n delete tile.request;\n }\n tile.aborted = true;\n }\n\n unloadTile(tile: Tile) {\n tile.unloadVectorData();\n this.actor.send('removeTile', {uid: tile.uid, type: this.type, source: this.id});\n }\n\n onRemove() {\n this._removed = true;\n this.actor.send('removeSource', {type: this.type, source: this.id});\n }\n\n serialize = (): GeoJSONSourceSpecification => {\n return extend({}, this._options, {\n type: this.type,\n data: this._data\n });\n };\n\n hasTransition() {\n return false;\n }\n}\n","import {createLayout} from '../util/struct_array';\n\nexport default createLayout([\n {name: 'a_pos', type: 'Int16', components: 2},\n {name: 'a_texture_pos', type: 'Int16', components: 2}\n]);\n","import {CanonicalTileID} from './tile_id';\nimport {Event, ErrorEvent, Evented} from '../util/evented';\nimport {ImageRequest} from '../util/image_request';\nimport {ResourceType} from '../util/request_manager';\nimport {EXTENT} from '../data/extent';\nimport {RasterBoundsArray} from '../data/array_types.g';\nimport rasterBoundsAttributes from '../data/raster_bounds_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {Texture} from '../render/texture';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\n\nimport type {Source} from './source';\nimport type {CanvasSourceSpecification} from './canvas_source';\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {\n ImageSourceSpecification,\n VideoSourceSpecification\n} from '@maplibre/maplibre-gl-style-spec';\nimport {Cancelable} from '../types/cancelable';\n\n/**\n * Four geographical coordinates,\n * represented as arrays of longitude and latitude numbers, which define the corners of the image.\n * The coordinates start at the top left corner of the image and proceed in clockwise order.\n * They do not have to represent a rectangle.\n */\nexport type Coordinates = [[number, number], [number, number], [number, number], [number, number]];\n\n/**\n * The options object for the {@link ImageSource#updateImage} method\n */\nexport type UpdateImageOptions = {\n /**\n * Required image URL.\n */\n url: string;\n /**\n * The image coordinates\n */\n coordinates?: Coordinates;\n}\n\n/**\n * A data source containing an image.\n * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-image) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * // add to map\n * map.addSource('some id', {\n * type: 'image',\n * url: 'https://www.maplibre.org/images/foo.png',\n * coordinates: [\n * [-76.54, 39.18],\n * [-76.52, 39.18],\n * [-76.52, 39.17],\n * [-76.54, 39.17]\n * ]\n * });\n *\n * // update coordinates\n * let mySource = map.getSource('some id');\n * mySource.setCoordinates([\n * [-76.54335737228394, 39.18579907229748],\n * [-76.52803659439087, 39.1838364847587],\n * [-76.5295386314392, 39.17683392507606],\n * [-76.54520273208618, 39.17876344106642]\n * ]);\n *\n * // update url and coordinates simultaneously\n * mySource.updateImage({\n * url: 'https://www.maplibre.org/images/bar.png',\n * coordinates: [\n * [-76.54335737228394, 39.18579907229748],\n * [-76.52803659439087, 39.1838364847587],\n * [-76.5295386314392, 39.17683392507606],\n * [-76.54520273208618, 39.17876344106642]\n * ]\n * })\n *\n * map.removeSource('some id'); // remove\n * ```\n */\nexport class ImageSource extends Evented implements Source {\n type: string;\n id: string;\n minzoom: number;\n maxzoom: number;\n tileSize: number;\n url: string;\n\n coordinates: Coordinates;\n tiles: {[_: string]: Tile};\n options: any;\n dispatcher: Dispatcher;\n map: Map;\n texture: Texture | null;\n image: HTMLImageElement | ImageBitmap;\n tileID: CanonicalTileID;\n _boundsArray: RasterBoundsArray;\n boundsBuffer: VertexBuffer;\n boundsSegments: SegmentVector;\n _loaded: boolean;\n _request: Cancelable;\n\n /** @internal */\n constructor(id: string, options: ImageSourceSpecification | VideoSourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super();\n this.id = id;\n this.dispatcher = dispatcher;\n this.coordinates = options.coordinates;\n\n this.type = 'image';\n this.minzoom = 0;\n this.maxzoom = 22;\n this.tileSize = 512;\n this.tiles = {};\n this._loaded = false;\n\n this.setEventedParent(eventedParent);\n\n this.options = options;\n }\n\n load = (newCoordinates?: Coordinates, successCallback?: () => void) => {\n this._loaded = false;\n this.fire(new Event('dataloading', {dataType: 'source'}));\n\n this.url = this.options.url;\n\n this._request = ImageRequest.getImage(this.map._requestManager.transformRequest(this.url, ResourceType.Image), (err, image) => {\n this._request = null;\n this._loaded = true;\n\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (image) {\n this.image = image;\n if (newCoordinates) {\n this.coordinates = newCoordinates;\n }\n if (successCallback) {\n successCallback();\n }\n this._finishLoading();\n }\n });\n };\n\n loaded(): boolean {\n return this._loaded;\n }\n\n /**\n * Updates the image URL and, optionally, the coordinates. To avoid having the image flash after changing,\n * set the `raster-fade-duration` paint property on the raster layer to 0.\n *\n * @param options - The options object.\n * @returns `this`\n */\n updateImage(options: UpdateImageOptions): this {\n if (!options.url) {\n return this;\n }\n\n if (this._request) {\n this._request.cancel();\n this._request = null;\n }\n\n this.options.url = options.url;\n this.load(options.coordinates, () => { this.texture = null; });\n return this;\n }\n\n _finishLoading() {\n if (this.map) {\n this.setCoordinates(this.coordinates);\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'}));\n }\n }\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n }\n\n onRemove() {\n if (this._request) {\n this._request.cancel();\n this._request = null;\n }\n }\n\n /**\n * Sets the image's coordinates and re-renders the map.\n *\n * @param coordinates - Four geographical coordinates,\n * represented as arrays of longitude and latitude numbers, which define the corners of the image.\n * The coordinates start at the top left corner of the image and proceed in clockwise order.\n * They do not have to represent a rectangle.\n * @returns `this`\n */\n setCoordinates(coordinates: Coordinates): this {\n this.coordinates = coordinates;\n\n // Calculate which mercator tile is suitable for rendering the video in\n // and create a buffer with the corner coordinates. These coordinates\n // may be outside the tile, because raster tiles aren't clipped when rendering.\n\n // transform the geo coordinates into (zoom 0) tile space coordinates\n const cornerCoords = coordinates.map(MercatorCoordinate.fromLngLat);\n\n // Compute the coordinates of the tile we'll use to hold this image's\n // render data\n this.tileID = getCoordinatesCenterTileID(cornerCoords);\n\n // Constrain min/max zoom to our tile's zoom level in order to force\n // SourceCache to request this tile (no matter what the map's zoom\n // level)\n this.minzoom = this.maxzoom = this.tileID.z;\n\n // Transform the corner coordinates into the coordinate space of our\n // tile.\n const tileCoords = cornerCoords.map((coord) => this.tileID.getTilePoint(coord)._round());\n\n this._boundsArray = new RasterBoundsArray();\n this._boundsArray.emplaceBack(tileCoords[0].x, tileCoords[0].y, 0, 0);\n this._boundsArray.emplaceBack(tileCoords[1].x, tileCoords[1].y, EXTENT, 0);\n this._boundsArray.emplaceBack(tileCoords[3].x, tileCoords[3].y, 0, EXTENT);\n this._boundsArray.emplaceBack(tileCoords[2].x, tileCoords[2].y, EXTENT, EXTENT);\n\n if (this.boundsBuffer) {\n this.boundsBuffer.destroy();\n delete this.boundsBuffer;\n }\n\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'}));\n return this;\n }\n\n prepare = () => {\n if (Object.keys(this.tiles).length === 0 || !this.image) {\n return;\n }\n\n const context = this.map.painter.context;\n const gl = context.gl;\n\n if (!this.boundsBuffer) {\n this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members);\n }\n\n if (!this.boundsSegments) {\n this.boundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n }\n\n if (!this.texture) {\n this.texture = new Texture(context, this.image, gl.RGBA);\n this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n }\n\n let newTilesLoaded = false;\n for (const w in this.tiles) {\n const tile = this.tiles[w];\n if (tile.state !== 'loaded') {\n tile.state = 'loaded';\n tile.texture = this.texture;\n newTilesLoaded = true;\n }\n }\n\n if (newTilesLoaded) {\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'idle', sourceId: this.id}));\n }\n };\n\n loadTile(tile: Tile, callback: Callback) {\n // We have a single tile -- whose coordinates are this.tileID -- that\n // covers the image we want to render. If that's the one being\n // requested, set it up with the image; otherwise, mark the tile as\n // `errored` to indicate that we have no data for it.\n // If the world wraps, we may have multiple \"wrapped\" copies of the\n // single tile.\n if (this.tileID && this.tileID.equals(tile.tileID.canonical)) {\n this.tiles[String(tile.tileID.wrap)] = tile;\n tile.buckets = {};\n callback(null);\n } else {\n tile.state = 'errored';\n callback(null);\n }\n }\n\n serialize = (): ImageSourceSpecification | VideoSourceSpecification | CanvasSourceSpecification => {\n return {\n type: 'image',\n url: this.options.url,\n coordinates: this.coordinates\n };\n };\n\n hasTransition() {\n return false;\n }\n}\n\n/**\n * Given a list of coordinates, get their center as a coordinate.\n *\n * @returns centerpoint\n * @internal\n */\nexport function getCoordinatesCenterTileID(coords: Array) {\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n\n for (const coord of coords) {\n minX = Math.min(minX, coord.x);\n minY = Math.min(minY, coord.y);\n maxX = Math.max(maxX, coord.x);\n maxY = Math.max(maxY, coord.y);\n }\n\n const dx = maxX - minX;\n const dy = maxY - minY;\n const dMax = Math.max(dx, dy);\n const zoom = Math.max(0, Math.floor(-Math.log(dMax) / Math.LN2));\n const tilesAtZoom = Math.pow(2, zoom);\n\n return new CanonicalTileID(\n zoom,\n Math.floor((minX + maxX) / 2 * tilesAtZoom),\n Math.floor((minY + maxY) / 2 * tilesAtZoom));\n}\n","import {getVideo} from '../util/ajax';\nimport {ResourceType} from '../util/request_manager';\n\nimport {ImageSource} from './image_source';\nimport rasterBoundsAttributes from '../data/raster_bounds_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {Texture} from '../render/texture';\nimport {Event, ErrorEvent} from '../util/evented';\nimport {ValidationError} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Evented} from '../util/evented';\nimport type {VideoSourceSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A data source containing video.\n * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-video) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * // add to map\n * map.addSource('some id', {\n * type: 'video',\n * url: [\n * 'https://www.mapbox.com/blog/assets/baltimore-smoke.mp4',\n * 'https://www.mapbox.com/blog/assets/baltimore-smoke.webm'\n * ],\n * coordinates: [\n * [-76.54, 39.18],\n * [-76.52, 39.18],\n * [-76.52, 39.17],\n * [-76.54, 39.17]\n * ]\n * });\n *\n * // update\n * let mySource = map.getSource('some id');\n * mySource.setCoordinates([\n * [-76.54335737228394, 39.18579907229748],\n * [-76.52803659439087, 39.1838364847587],\n * [-76.5295386314392, 39.17683392507606],\n * [-76.54520273208618, 39.17876344106642]\n * ]);\n *\n * map.removeSource('some id'); // remove\n * ```\n * @see [Add a video](https://maplibre.org/maplibre-gl-js/docs/examples/video-on-a-map/)\n *\n * Note that when rendered as a raster layer, the layer's `raster-fade-duration` property will cause the video to fade in.\n * This happens when playback is started, paused and resumed, or when the video's coordinates are updated. To avoid this behavior,\n * set the layer's `raster-fade-duration` property to `0`.\n */\nexport class VideoSource extends ImageSource {\n options: VideoSourceSpecification;\n urls: Array;\n video: HTMLVideoElement;\n roundZoom: boolean;\n\n constructor(id: string, options: VideoSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super(id, options, dispatcher, eventedParent);\n this.roundZoom = true;\n this.type = 'video';\n this.options = options;\n }\n\n load = () => {\n this._loaded = false;\n const options = this.options;\n\n this.urls = [];\n for (const url of options.urls) {\n this.urls.push(this.map._requestManager.transformRequest(url, ResourceType.Source).url);\n }\n\n getVideo(this.urls, (err, video) => {\n this._loaded = true;\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (video) {\n this.video = video;\n this.video.loop = true;\n\n // Start repainting when video starts playing. hasTransition() will then return\n // true to trigger additional frames as long as the videos continues playing.\n this.video.addEventListener('playing', () => {\n this.map.triggerRepaint();\n });\n\n if (this.map) {\n this.video.play();\n }\n\n this._finishLoading();\n }\n });\n };\n\n /**\n * Pauses the video.\n */\n pause() {\n if (this.video) {\n this.video.pause();\n }\n }\n\n /**\n * Plays the video.\n */\n play() {\n if (this.video) {\n this.video.play();\n }\n }\n\n /**\n * Sets playback to a timestamp, in seconds.\n */\n seek(seconds: number) {\n if (this.video) {\n const seekableRange = this.video.seekable;\n if (seconds < seekableRange.start(0) || seconds > seekableRange.end(0)) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${this.id}`, null, `Playback for this video can be set only between the ${seekableRange.start(0)} and ${seekableRange.end(0)}-second mark.`)));\n } else this.video.currentTime = seconds;\n }\n }\n\n /**\n * Returns the HTML `video` element.\n *\n * @returns The HTML `video` element.\n */\n getVideo(): HTMLVideoElement {\n return this.video;\n }\n\n onAdd(map: Map) {\n if (this.map) return;\n this.map = map;\n this.load();\n if (this.video) {\n this.video.play();\n this.setCoordinates(this.coordinates);\n }\n }\n\n /**\n * Sets the video's coordinates and re-renders the map.\n *\n * @returns `this`\n */\n prepare = (): this => {\n if (Object.keys(this.tiles).length === 0 || this.video.readyState < 2) {\n return; // not enough data for current position\n }\n\n const context = this.map.painter.context;\n const gl = context.gl;\n\n if (!this.boundsBuffer) {\n this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members);\n }\n\n if (!this.boundsSegments) {\n this.boundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n }\n\n if (!this.texture) {\n this.texture = new Texture(context, this.video, gl.RGBA);\n this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n } else if (!this.video.paused) {\n this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, gl.RGBA, gl.UNSIGNED_BYTE, this.video);\n }\n\n let newTilesLoaded = false;\n for (const w in this.tiles) {\n const tile = this.tiles[w];\n if (tile.state !== 'loaded') {\n tile.state = 'loaded';\n tile.texture = this.texture;\n newTilesLoaded = true;\n }\n }\n\n if (newTilesLoaded) {\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'idle', sourceId: this.id}));\n }\n };\n\n serialize = (): VideoSourceSpecification => {\n return {\n type: 'video',\n urls: this.urls,\n coordinates: this.coordinates\n };\n };\n\n hasTransition() {\n return this.video && !this.video.paused;\n }\n}\n","import {ImageSource} from './image_source';\n\nimport rasterBoundsAttributes from '../data/raster_bounds_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {Texture} from '../render/texture';\nimport {Event, ErrorEvent} from '../util/evented';\nimport {ValidationError} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Evented} from '../util/evented';\n\n/**\n * Options to add a canvas source type to the map.\n */\nexport type CanvasSourceSpecification = {\n /**\n * Source type. Must be `\"canvas\"`.\n */\n type: 'canvas';\n /**\n * Four geographical coordinates denoting where to place the corners of the canvas, specified in `[longitude, latitude]` pairs.\n */\n coordinates: [[number, number], [number, number], [number, number], [number, number]];\n /**\n * Whether the canvas source is animated. If the canvas is static (i.e. pixels do not need to be re-read on every frame), `animate` should be set to `false` to improve performance.\n * @defaultValue true\n */\n animate?: boolean;\n /**\n * Canvas source from which to read pixels. Can be a string representing the ID of the canvas element, or the `HTMLCanvasElement` itself.\n */\n canvas?: string | HTMLCanvasElement;\n};\n\n/**\n * A data source containing the contents of an HTML canvas. See {@link CanvasSourceSpecification} for detailed documentation of options.\n *\n * @group Sources\n *\n * @example\n * ```ts\n * // add to map\n * map.addSource('some id', {\n * type: 'canvas',\n * canvas: 'idOfMyHTMLCanvas',\n * animate: true,\n * coordinates: [\n * [-76.54, 39.18],\n * [-76.52, 39.18],\n * [-76.52, 39.17],\n * [-76.54, 39.17]\n * ]\n * });\n *\n * // update\n * let mySource = map.getSource('some id');\n * mySource.setCoordinates([\n * [-76.54335737228394, 39.18579907229748],\n * [-76.52803659439087, 39.1838364847587],\n * [-76.5295386314392, 39.17683392507606],\n * [-76.54520273208618, 39.17876344106642]\n * ]);\n *\n * map.removeSource('some id'); // remove\n * ```\n */\nexport class CanvasSource extends ImageSource {\n options: CanvasSourceSpecification;\n animate: boolean;\n canvas: HTMLCanvasElement;\n width: number;\n height: number;\n /**\n * Enables animation. The image will be copied from the canvas to the map on each frame.\n */\n play: () => void;\n /**\n * Disables animation. The map will display a static copy of the canvas image.\n */\n pause: () => void;\n _playing: boolean;\n\n /** @internal */\n constructor(id: string, options: CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super(id, options, dispatcher, eventedParent);\n\n // We build in some validation here, since canvas sources aren't included in the style spec:\n if (!options.coordinates) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, 'missing required property \"coordinates\"')));\n } else if (!Array.isArray(options.coordinates) || options.coordinates.length !== 4 ||\n options.coordinates.some(c => !Array.isArray(c) || c.length !== 2 || c.some(l => typeof l !== 'number'))) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, '\"coordinates\" property must be an array of 4 longitude/latitude array pairs')));\n }\n\n if (options.animate && typeof options.animate !== 'boolean') {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, 'optional \"animate\" property must be a boolean value')));\n }\n\n if (!options.canvas) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, 'missing required property \"canvas\"')));\n } else if (typeof options.canvas !== 'string' && !(options.canvas instanceof HTMLCanvasElement)) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, '\"canvas\" must be either a string representing the ID of the canvas element from which to read, or an HTMLCanvasElement instance')));\n }\n\n this.options = options;\n this.animate = options.animate !== undefined ? options.animate : true;\n }\n\n load = () => {\n this._loaded = true;\n if (!this.canvas) {\n this.canvas = (this.options.canvas instanceof HTMLCanvasElement) ?\n this.options.canvas :\n document.getElementById(this.options.canvas) as HTMLCanvasElement;\n // cast to HTMLCanvasElement in else of ternary\n // should we do a safety check and throw if it's not actually HTMLCanvasElement?\n }\n this.width = this.canvas.width;\n this.height = this.canvas.height;\n\n if (this._hasInvalidDimensions()) {\n this.fire(new ErrorEvent(new Error('Canvas dimensions cannot be less than or equal to zero.')));\n return;\n }\n\n this.play = function() {\n this._playing = true;\n this.map.triggerRepaint();\n };\n\n this.pause = function() {\n if (this._playing) {\n this.prepare();\n this._playing = false;\n }\n };\n\n this._finishLoading();\n };\n\n /**\n * Returns the HTML `canvas` element.\n *\n * @returns The HTML `canvas` element.\n */\n getCanvas(): HTMLCanvasElement {\n return this.canvas;\n }\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n if (this.canvas) {\n if (this.animate) this.play();\n }\n }\n\n onRemove() {\n this.pause();\n }\n\n prepare = () => {\n let resize = false;\n if (this.canvas.width !== this.width) {\n this.width = this.canvas.width;\n resize = true;\n }\n if (this.canvas.height !== this.height) {\n this.height = this.canvas.height;\n resize = true;\n }\n\n if (this._hasInvalidDimensions()) return;\n\n if (Object.keys(this.tiles).length === 0) return; // not enough data for current position\n\n const context = this.map.painter.context;\n const gl = context.gl;\n\n if (!this.boundsBuffer) {\n this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members);\n }\n\n if (!this.boundsSegments) {\n this.boundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n }\n\n if (!this.texture) {\n this.texture = new Texture(context, this.canvas, gl.RGBA, {premultiply: true});\n } else if (resize || this._playing) {\n this.texture.update(this.canvas, {premultiply: true});\n }\n\n let newTilesLoaded = false;\n for (const w in this.tiles) {\n const tile = this.tiles[w];\n if (tile.state !== 'loaded') {\n tile.state = 'loaded';\n tile.texture = this.texture;\n newTilesLoaded = true;\n }\n }\n\n if (newTilesLoaded) {\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'idle', sourceId: this.id}));\n }\n };\n\n serialize = (): CanvasSourceSpecification => {\n return {\n type: 'canvas',\n coordinates: this.coordinates\n };\n };\n\n hasTransition() {\n return this._playing;\n }\n\n _hasInvalidDimensions() {\n for (const x of [this.canvas.width, this.canvas.height]) {\n if (isNaN(x) || x <= 0) return true;\n }\n return false;\n }\n}\n","import {VectorTileSource} from '../source/vector_tile_source';\nimport {RasterTileSource} from '../source/raster_tile_source';\nimport {RasterDEMTileSource} from '../source/raster_dem_tile_source';\nimport {GeoJSONSource} from '../source/geojson_source';\nimport {VideoSource} from '../source/video_source';\nimport {ImageSource} from '../source/image_source';\nimport {CanvasSource} from '../source/canvas_source';\n\nimport type {SourceSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Event, Evented} from '../util/evented';\nimport type {Map} from '../ui/map';\nimport type {Tile} from './tile';\nimport type {OverscaledTileID, CanonicalTileID} from './tile_id';\nimport type {Callback} from '../types/callback';\nimport type {CanvasSourceSpecification} from '../source/canvas_source';\n\nconst registeredSources = {} as {[key:string]: SourceClass};\n\n/**\n * The `Source` interface must be implemented by each source type, including \"core\" types (`vector`, `raster`,\n * `video`, etc.) and all custom, third-party types.\n *\n * @event `data` - Fired with `{dataType: 'source', sourceDataType: 'metadata'}` to indicate that any necessary metadata\n * has been loaded so that it's okay to call `loadTile`; and with `{dataType: 'source', sourceDataType: 'content'}`\n * to indicate that the source data has changed, so that any current caches should be flushed.\n *\n * @group Sources\n */\nexport interface Source {\n readonly type: string;\n /**\n * The id for the source. Must not be used by any existing source.\n */\n id: string;\n minzoom: number;\n maxzoom: number;\n tileSize: number;\n attribution?: string;\n /**\n * `true` if zoom levels are rounded to the nearest integer in the source data, `false` if they are floor-ed to the nearest integer.\n */\n roundZoom?: boolean;\n /**\n * `false` if tiles can be drawn outside their boundaries, `true` if they cannot.\n */\n isTileClipped?: boolean;\n tileID?: CanonicalTileID;\n /**\n * `true` if tiles should be sent back to the worker for each overzoomed zoom level, `false` if not.\n */\n reparseOverscaled?: boolean;\n vectorLayerIds?: Array;\n hasTransition(): boolean;\n loaded(): boolean;\n fire(event: Event): unknown;\n readonly onAdd?: (map: Map) => void;\n readonly onRemove?: (map: Map) => void;\n loadTile(tile: Tile, callback: Callback): void;\n readonly hasTile?: (tileID: OverscaledTileID) => boolean;\n readonly abortTile?: (tile: Tile, callback: Callback) => void;\n readonly unloadTile?: (tile: Tile, callback: Callback) => void;\n /**\n * @returns A plain (stringifiable) JS object representing the current state of the source.\n * Creating a source using the returned object as the `options` should result in a Source that is\n * equivalent to this one.\n */\n serialize(): any;\n readonly prepare?: () => void;\n}\n\n/**\n * A supporting type to the source definition\n */\ntype SourceStatics = {\n /*\n * An optional URL to a script which, when run by a Worker, registers a {@link WorkerSource}\n * implementation for this Source type by calling `self.registerWorkerSource(workerSource: WorkerSource)`.\n */\n workerSourceURL?: URL;\n};\n\n/**\n * A general definition of a {@link Source} class for factory usage\n */\nexport type SourceClass = {\n new (id: string, specification: SourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented): Source;\n} & SourceStatics;\n\n/**\n * Creates a tiled data source instance given an options object.\n *\n * @param id - The id for the source. Must not be used by any existing source.\n * @param specification - Source options, specific to the source type (except for `options.type`, which is always required).\n * @param source - A source definition object compliant with\n * [`maplibre-gl-style-spec`](https://maplibre.org/maplibre-style-spec/#sources) or, for a third-party source type,\n * with that type's requirements.\n * @param dispatcher - A {@link Dispatcher} instance, which can be used to send messages to the workers.\n * @returns a newly created source\n */\nexport const create = (id: string, specification: SourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented): Source => {\n\n const Class = getSourceType(specification.type);\n const source = new Class(id, specification, dispatcher, eventedParent);\n\n if (source.id !== id) {\n throw new Error(`Expected Source id to be ${id} instead of ${source.id}`);\n }\n\n return source;\n};\n\nexport const getSourceType = (name: string): SourceClass => {\n switch (name) {\n case 'geojson':\n return GeoJSONSource;\n case 'image':\n return ImageSource;\n case 'raster':\n return RasterTileSource;\n case 'raster-dem':\n return RasterDEMTileSource;\n case 'vector':\n return VectorTileSource;\n case 'video':\n return VideoSource;\n case 'canvas':\n return CanvasSource;\n }\n return registeredSources[name];\n};\n\nexport const setSourceType = (name: string, type: SourceClass) => {\n registeredSources[name] = type;\n};\n\nexport interface Actor {\n send(type: string, data: any, callback: Callback): void;\n}\n","import type {SourceCache} from './source_cache';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {CollisionIndex} from '../symbol/collision_index';\nimport type {Transform} from '../geo/transform';\nimport type {RetainedQueryData} from '../symbol/placement';\nimport type {FilterSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\nimport type Point from '@mapbox/point-geometry';\nimport {mat4} from 'gl-matrix';\n\n/**\n * Options to pass to query the map for the rendered features\n */\nexport type QueryRenderedFeaturesOptions = {\n /**\n * An array of [style layer IDs](https://maplibre.org/maplibre-style-spec/#layer-id) for the query to inspect.\n * Only features within these layers will be returned. If this parameter is undefined, all layers will be checked.\n */\n layers?: Array;\n /**\n * A [filter](https://maplibre.org/maplibre-style-spec/layers/#filter) to limit query results.\n */\n filter?: FilterSpecification;\n /**\n * An array of string representing the available images\n */\n availableImages?: Array;\n /**\n * Whether to check if the [options.filter] conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function.\n */\n validate?: boolean;\n};\n\n/**\n * The options object related to the {@link Map#queryRenderedFeatures} method\n */\nexport type QuerySourceFeatureOptions = {\n /**\n * The name of the source layer to query. *For vector tile sources, this parameter is required.* For GeoJSON sources, it is ignored.\n */\n sourceLayer?: string;\n /**\n * A [filter](https://maplibre.org/maplibre-style-spec/layers/#filter)\n * to limit query results.\n */\n filter?: FilterSpecification;\n /**\n * Whether to check if the [parameters.filter] conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function.\n * @defaultValue true\n */\n validate?: boolean;\n}\n\n/*\n * Returns a matrix that can be used to convert from tile coordinates to viewport pixel coordinates.\n */\nfunction getPixelPosMatrix(transform, tileID) {\n const t = mat4.create();\n mat4.translate(t, t, [1, 1, 0]);\n mat4.scale(t, t, [transform.width * 0.5, transform.height * 0.5, 1]);\n return mat4.multiply(t, t, transform.calculatePosMatrix(tileID.toUnwrapped()));\n}\n\nfunction queryIncludes3DLayer(layers: Array, styleLayers: {[_: string]: StyleLayer}, sourceID: string) {\n if (layers) {\n for (const layerID of layers) {\n const layer = styleLayers[layerID];\n if (layer && layer.source === sourceID && layer.type === 'fill-extrusion') {\n return true;\n }\n }\n } else {\n for (const key in styleLayers) {\n const layer = styleLayers[key];\n if (layer.source === sourceID && layer.type === 'fill-extrusion') {\n return true;\n }\n }\n }\n return false;\n}\n\nexport function queryRenderedFeatures(\n sourceCache: SourceCache,\n styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n queryGeometry: Array,\n params: QueryRenderedFeaturesOptions,\n transform: Transform\n): { [key: string]: Array<{featureIndex: number; feature: MapGeoJSONFeature}> } {\n\n const has3DLayer = queryIncludes3DLayer(params && params.layers, styleLayers, sourceCache.id);\n const maxPitchScaleFactor = transform.maxPitchScaleFactor();\n const tilesIn = sourceCache.tilesIn(queryGeometry, maxPitchScaleFactor, has3DLayer);\n\n tilesIn.sort(sortTilesIn);\n const renderedFeatureLayers = [];\n for (const tileIn of tilesIn) {\n renderedFeatureLayers.push({\n wrappedTileID: tileIn.tileID.wrapped().key,\n queryResults: tileIn.tile.queryRenderedFeatures(\n styleLayers,\n serializedLayers,\n sourceCache._state,\n tileIn.queryGeometry,\n tileIn.cameraQueryGeometry,\n tileIn.scale,\n params,\n transform,\n maxPitchScaleFactor,\n getPixelPosMatrix(sourceCache.transform, tileIn.tileID))\n });\n }\n\n const result = mergeRenderedFeatureLayers(renderedFeatureLayers);\n\n // Merge state from SourceCache into the results\n for (const layerID in result) {\n result[layerID].forEach((featureWrapper) => {\n const feature = featureWrapper.feature as MapGeoJSONFeature;\n const state = sourceCache.getFeatureState(feature.layer['source-layer'], feature.id);\n feature.source = feature.layer.source;\n if (feature.layer['source-layer']) {\n feature.sourceLayer = feature.layer['source-layer'];\n }\n feature.state = state;\n });\n }\n return result;\n}\n\nexport function queryRenderedSymbols(styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: StyleLayer},\n sourceCaches: {[_: string]: SourceCache},\n queryGeometry: Array,\n params: QueryRenderedFeaturesOptions,\n collisionIndex: CollisionIndex,\n retainedQueryData: {\n [_: number]: RetainedQueryData;\n }) {\n const result = {};\n const renderedSymbols = collisionIndex.queryRenderedSymbols(queryGeometry);\n const bucketQueryData = [];\n for (const bucketInstanceId of Object.keys(renderedSymbols).map(Number)) {\n bucketQueryData.push(retainedQueryData[bucketInstanceId]);\n }\n bucketQueryData.sort(sortTilesIn);\n\n for (const queryData of bucketQueryData) {\n const bucketSymbols = queryData.featureIndex.lookupSymbolFeatures(\n renderedSymbols[queryData.bucketInstanceId],\n serializedLayers,\n queryData.bucketIndex,\n queryData.sourceLayerIndex,\n params.filter,\n params.layers,\n params.availableImages,\n styleLayers);\n\n for (const layerID in bucketSymbols) {\n const resultFeatures = result[layerID] = result[layerID] || [];\n const layerSymbols = bucketSymbols[layerID];\n layerSymbols.sort((a, b) => {\n // Match topDownFeatureComparator from FeatureIndex, but using\n // most recent sorting of features from bucket.sortFeatures\n const featureSortOrder = queryData.featureSortOrder;\n if (featureSortOrder) {\n // queryRenderedSymbols documentation says we'll return features in\n // \"top-to-bottom\" rendering order (aka last-to-first).\n // Actually there can be multiple symbol instances per feature, so\n // we sort each feature based on the first matching symbol instance.\n const sortedA = featureSortOrder.indexOf(a.featureIndex);\n const sortedB = featureSortOrder.indexOf(b.featureIndex);\n return sortedB - sortedA;\n } else {\n // Bucket hasn't been re-sorted based on angle, so use the\n // reverse of the order the features appeared in the data.\n return b.featureIndex - a.featureIndex;\n }\n });\n for (const symbolFeature of layerSymbols) {\n resultFeatures.push(symbolFeature);\n }\n }\n }\n\n // Merge state from SourceCache into the results\n for (const layerName in result) {\n result[layerName].forEach((featureWrapper) => {\n const feature = featureWrapper.feature;\n const layer = styleLayers[layerName];\n const sourceCache = sourceCaches[layer.source];\n const state = sourceCache.getFeatureState(feature.layer['source-layer'], feature.id);\n feature.source = feature.layer.source;\n if (feature.layer['source-layer']) {\n feature.sourceLayer = feature.layer['source-layer'];\n }\n feature.state = state;\n });\n }\n return result;\n}\n\nexport function querySourceFeatures(sourceCache: SourceCache, params: QuerySourceFeatureOptions) {\n const tiles = sourceCache.getRenderableIds().map((id) => {\n return sourceCache.getTileByID(id);\n });\n\n const result = [];\n\n const dataTiles = {};\n for (let i = 0; i < tiles.length; i++) {\n const tile = tiles[i];\n const dataID = tile.tileID.canonical.key;\n if (!dataTiles[dataID]) {\n dataTiles[dataID] = true;\n tile.querySourceFeatures(result, params);\n }\n }\n\n return result;\n}\n\nfunction sortTilesIn(a, b) {\n const idA = a.tileID;\n const idB = b.tileID;\n return (idA.overscaledZ - idB.overscaledZ) || (idA.canonical.y - idB.canonical.y) || (idA.wrap - idB.wrap) || (idA.canonical.x - idB.canonical.x);\n}\n\nfunction mergeRenderedFeatureLayers(tiles) {\n // Merge results from all tiles, but if two tiles share the same\n // wrapped ID, don't duplicate features between the two tiles\n const result = {};\n const wrappedIDLayerMap = {};\n for (const tile of tiles) {\n const queryResults = tile.queryResults;\n const wrappedID = tile.wrappedTileID;\n const wrappedIDLayers = wrappedIDLayerMap[wrappedID] = wrappedIDLayerMap[wrappedID] || {};\n for (const layerID in queryResults) {\n const tileFeatures = queryResults[layerID];\n const wrappedIDFeatures = wrappedIDLayers[layerID] = wrappedIDLayers[layerID] || {};\n const resultFeatures = result[layerID] = result[layerID] || [];\n for (const tileFeature of tileFeatures) {\n if (!wrappedIDFeatures[tileFeature.featureIndex]) {\n wrappedIDFeatures[tileFeature.featureIndex] = true;\n resultFeatures.push(tileFeature);\n }\n }\n }\n }\n return result;\n}\n","export class DictionaryCoder {\n _stringToNumber: {[_: string]: number};\n _numberToString: Array;\n\n constructor(strings: Array) {\n this._stringToNumber = {};\n this._numberToString = [];\n for (let i = 0; i < strings.length; i++) {\n const string = strings[i];\n this._stringToNumber[string] = i;\n this._numberToString[i] = string;\n }\n }\n\n encode(string: string) {\n return this._stringToNumber[string];\n }\n\n decode(n: number) {\n if (n >= this._numberToString.length) throw new Error(`Out of bounds. Index requested n=${n} can't be >= this._numberToString.length ${this._numberToString.length}`);\n return this._numberToString[n];\n }\n}\n","import type {VectorTileFeature} from '@mapbox/vector-tile';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A helper for type to omit a property from a type\n */\ntype DistributiveKeys = T extends T ? keyof T : never;\n/**\n * A helper for type to omit a property from a type\n */\ntype DistributiveOmit> = T extends unknown\n ? Omit\n : never;\n\n/**\n * An extended geojson feature used by the events to return data to the listener\n */\nexport type MapGeoJSONFeature = GeoJSONFeature & {\n layer: DistributiveOmit & {source: string};\n source: string;\n sourceLayer?: string;\n state: { [key: string]: any };\n}\n\n/**\n * A geojson feature\n */\nexport class GeoJSONFeature {\n type: 'Feature';\n _geometry: GeoJSON.Geometry;\n properties: { [name: string]: any };\n id: number | string | undefined;\n\n _vectorTileFeature: VectorTileFeature;\n\n constructor(vectorTileFeature: VectorTileFeature, z: number, x: number, y: number, id: string | number | undefined) {\n this.type = 'Feature';\n\n this._vectorTileFeature = vectorTileFeature;\n (vectorTileFeature as any)._z = z;\n (vectorTileFeature as any)._x = x;\n (vectorTileFeature as any)._y = y;\n\n this.properties = vectorTileFeature.properties;\n this.id = id;\n }\n\n get geometry(): GeoJSON.Geometry {\n if (this._geometry === undefined) {\n this._geometry = this._vectorTileFeature.toGeoJSON(\n (this._vectorTileFeature as any)._x,\n (this._vectorTileFeature as any)._y,\n (this._vectorTileFeature as any)._z).geometry;\n }\n return this._geometry;\n }\n\n set geometry(g: GeoJSON.Geometry) {\n this._geometry = g;\n }\n\n toJSON() {\n const json: any = {\n geometry: this.geometry\n };\n for (const i in this) {\n if (i === '_geometry' || i === '_vectorTileFeature') continue;\n json[i] = (this)[i];\n }\n return json;\n }\n}\n","import Point from '@mapbox/point-geometry';\nimport {loadGeometry} from './load_geometry';\nimport {toEvaluationFeature} from './evaluation_feature';\nimport {EXTENT} from './extent';\nimport {featureFilter} from '@maplibre/maplibre-gl-style-spec';\nimport {TransferableGridIndex} from '../util/transferable_grid_index';\nimport {DictionaryCoder} from '../util/dictionary_coder';\nimport vt from '@mapbox/vector-tile';\nimport Protobuf from 'pbf';\nimport {GeoJSONFeature} from '../util/vectortile_to_geojson';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\nimport {arraysIntersect, mapObject, extend} from '../util/util';\nimport {OverscaledTileID} from '../source/tile_id';\nimport {register} from '../util/web_worker_transfer';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {SourceFeatureState} from '../source/source_state';\nimport {polygonIntersectsBox} from '../util/intersection_tests';\nimport {PossiblyEvaluated} from '../style/properties';\nimport {FeatureIndexArray} from './array_types.g';\nimport {mat4} from 'gl-matrix';\n\nimport type {StyleLayer} from '../style/style_layer';\nimport type {FeatureFilter, FeatureState, FilterSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Transform} from '../geo/transform';\nimport type {VectorTileFeature, VectorTileLayer} from '@mapbox/vector-tile';\n\ntype QueryParameters = {\n scale: number;\n pixelPosMatrix: mat4;\n transform: Transform;\n tileSize: number;\n queryGeometry: Array;\n cameraQueryGeometry: Array;\n queryPadding: number;\n params: {\n filter: FilterSpecification;\n layers: Array;\n availableImages: Array;\n };\n};\n\n/**\n * @internal\n * An in memory index class to allow fast interaction with features\n */\nexport class FeatureIndex {\n tileID: OverscaledTileID;\n x: number;\n y: number;\n z: number;\n grid: TransferableGridIndex;\n grid3D: TransferableGridIndex;\n featureIndexArray: FeatureIndexArray;\n promoteId?: PromoteIdSpecification;\n\n rawTileData: ArrayBuffer;\n bucketLayerIDs: Array>;\n\n vtLayers: {[_: string]: VectorTileLayer};\n sourceLayerCoder: DictionaryCoder;\n\n constructor(tileID: OverscaledTileID, promoteId?: PromoteIdSpecification | null) {\n this.tileID = tileID;\n this.x = tileID.canonical.x;\n this.y = tileID.canonical.y;\n this.z = tileID.canonical.z;\n this.grid = new TransferableGridIndex(EXTENT, 16, 0);\n this.grid3D = new TransferableGridIndex(EXTENT, 16, 0);\n this.featureIndexArray = new FeatureIndexArray();\n this.promoteId = promoteId;\n }\n\n insert(feature: VectorTileFeature, geometry: Array>, featureIndex: number, sourceLayerIndex: number, bucketIndex: number, is3D?: boolean) {\n const key = this.featureIndexArray.length;\n this.featureIndexArray.emplaceBack(featureIndex, sourceLayerIndex, bucketIndex);\n\n const grid = is3D ? this.grid3D : this.grid;\n\n for (let r = 0; r < geometry.length; r++) {\n const ring = geometry[r];\n\n const bbox = [Infinity, Infinity, -Infinity, -Infinity];\n for (let i = 0; i < ring.length; i++) {\n const p = ring[i];\n bbox[0] = Math.min(bbox[0], p.x);\n bbox[1] = Math.min(bbox[1], p.y);\n bbox[2] = Math.max(bbox[2], p.x);\n bbox[3] = Math.max(bbox[3], p.y);\n }\n\n if (bbox[0] < EXTENT &&\n bbox[1] < EXTENT &&\n bbox[2] >= 0 &&\n bbox[3] >= 0) {\n grid.insert(key, bbox[0], bbox[1], bbox[2], bbox[3]);\n }\n }\n }\n\n loadVTLayers(): {[_: string]: VectorTileLayer} {\n if (!this.vtLayers) {\n this.vtLayers = new vt.VectorTile(new Protobuf(this.rawTileData)).layers;\n this.sourceLayerCoder = new DictionaryCoder(this.vtLayers ? Object.keys(this.vtLayers).sort() : ['_geojsonTileLayer']);\n }\n return this.vtLayers;\n }\n\n // Finds non-symbol features in this tile at a particular position.\n query(\n args: QueryParameters,\n styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n sourceFeatureState: SourceFeatureState\n ): {[_: string]: Array<{featureIndex: number; feature: GeoJSONFeature}>} {\n this.loadVTLayers();\n\n const params = args.params || {} as { filter: any; layers: string[]; availableImages: string[] },\n pixelsToTileUnits = EXTENT / args.tileSize / args.scale,\n filter = featureFilter(params.filter);\n\n const queryGeometry = args.queryGeometry;\n const queryPadding = args.queryPadding * pixelsToTileUnits;\n\n const bounds = getBounds(queryGeometry);\n const matching = this.grid.query(bounds.minX - queryPadding, bounds.minY - queryPadding, bounds.maxX + queryPadding, bounds.maxY + queryPadding);\n\n const cameraBounds = getBounds(args.cameraQueryGeometry);\n const matching3D = this.grid3D.query(\n cameraBounds.minX - queryPadding, cameraBounds.minY - queryPadding, cameraBounds.maxX + queryPadding, cameraBounds.maxY + queryPadding,\n (bx1, by1, bx2, by2) => {\n return polygonIntersectsBox(args.cameraQueryGeometry, bx1 - queryPadding, by1 - queryPadding, bx2 + queryPadding, by2 + queryPadding);\n });\n\n for (const key of matching3D) {\n matching.push(key);\n }\n\n matching.sort(topDownFeatureComparator);\n\n const result = {};\n let previousIndex;\n for (let k = 0; k < matching.length; k++) {\n const index = matching[k];\n\n // don't check the same feature more than once\n if (index === previousIndex) continue;\n previousIndex = index;\n\n const match = this.featureIndexArray.get(index);\n let featureGeometry = null;\n this.loadMatchingFeature(\n result,\n match.bucketIndex,\n match.sourceLayerIndex,\n match.featureIndex,\n filter,\n params.layers,\n params.availableImages,\n styleLayers,\n serializedLayers,\n sourceFeatureState,\n (feature: VectorTileFeature, styleLayer: StyleLayer, featureState: FeatureState) => {\n if (!featureGeometry) {\n featureGeometry = loadGeometry(feature);\n }\n\n return styleLayer.queryIntersectsFeature(queryGeometry, feature, featureState, featureGeometry, this.z, args.transform, pixelsToTileUnits, args.pixelPosMatrix);\n }\n );\n }\n\n return result;\n }\n\n loadMatchingFeature(\n result: {\n [_: string]: Array<{\n featureIndex: number;\n feature: GeoJSONFeature;\n intersectionZ?: boolean | number;\n }>;\n },\n bucketIndex: number,\n sourceLayerIndex: number,\n featureIndex: number,\n filter: FeatureFilter,\n filterLayerIDs: Array,\n availableImages: Array,\n styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n sourceFeatureState?: SourceFeatureState,\n intersectionTest?: (\n feature: VectorTileFeature,\n styleLayer: StyleLayer,\n featureState: any,\n id: string | number | void\n ) => boolean | number) {\n\n const layerIDs = this.bucketLayerIDs[bucketIndex];\n if (filterLayerIDs && !arraysIntersect(filterLayerIDs, layerIDs))\n return;\n\n const sourceLayerName = this.sourceLayerCoder.decode(sourceLayerIndex);\n const sourceLayer = this.vtLayers[sourceLayerName];\n const feature = sourceLayer.feature(featureIndex);\n\n if (filter.needGeometry) {\n const evaluationFeature = toEvaluationFeature(feature, true);\n if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), evaluationFeature, this.tileID.canonical)) {\n return;\n }\n } else if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), feature)) {\n return;\n }\n\n const id = this.getId(feature, sourceLayerName);\n\n for (let l = 0; l < layerIDs.length; l++) {\n const layerID = layerIDs[l];\n\n if (filterLayerIDs && filterLayerIDs.indexOf(layerID) < 0) {\n continue;\n }\n\n const styleLayer = styleLayers[layerID];\n\n if (!styleLayer) continue;\n\n let featureState = {};\n if (id && sourceFeatureState) {\n // `feature-state` expression evaluation requires feature state to be available\n featureState = sourceFeatureState.getState(styleLayer.sourceLayer || '_geojsonTileLayer', id);\n }\n\n const serializedLayer = extend({}, serializedLayers[layerID]);\n\n serializedLayer.paint = evaluateProperties(serializedLayer.paint, styleLayer.paint, feature, featureState, availableImages);\n serializedLayer.layout = evaluateProperties(serializedLayer.layout, styleLayer.layout, feature, featureState, availableImages);\n\n const intersectionZ = !intersectionTest || intersectionTest(feature, styleLayer, featureState);\n if (!intersectionZ) {\n // Only applied for non-symbol features\n continue;\n }\n\n const geojsonFeature = new GeoJSONFeature(feature, this.z, this.x, this.y, id) as MapGeoJSONFeature;\n geojsonFeature.layer = serializedLayer;\n let layerResult = result[layerID];\n if (layerResult === undefined) {\n layerResult = result[layerID] = [];\n }\n layerResult.push({featureIndex, feature: geojsonFeature, intersectionZ});\n }\n }\n\n // Given a set of symbol indexes that have already been looked up,\n // return a matching set of GeoJSONFeatures\n lookupSymbolFeatures(symbolFeatureIndexes: Array,\n serializedLayers: {[_: string]: StyleLayer},\n bucketIndex: number,\n sourceLayerIndex: number,\n filterSpec: FilterSpecification,\n filterLayerIDs: Array,\n availableImages: Array,\n styleLayers: {[_: string]: StyleLayer}) {\n const result = {};\n this.loadVTLayers();\n\n const filter = featureFilter(filterSpec);\n\n for (const symbolFeatureIndex of symbolFeatureIndexes) {\n this.loadMatchingFeature(\n result,\n bucketIndex,\n sourceLayerIndex,\n symbolFeatureIndex,\n filter,\n filterLayerIDs,\n availableImages,\n styleLayers,\n serializedLayers\n );\n\n }\n return result;\n }\n\n hasLayer(id: string) {\n for (const layerIDs of this.bucketLayerIDs) {\n for (const layerID of layerIDs) {\n if (id === layerID) return true;\n }\n }\n\n return false;\n }\n\n getId(feature: VectorTileFeature, sourceLayerId: string): string | number {\n let id: string | number = feature.id;\n if (this.promoteId) {\n const propName = typeof this.promoteId === 'string' ? this.promoteId : this.promoteId[sourceLayerId];\n id = feature.properties[propName] as string | number;\n if (typeof id === 'boolean') id = Number(id);\n }\n return id;\n }\n}\n\nregister(\n 'FeatureIndex',\n FeatureIndex,\n {omit: ['rawTileData', 'sourceLayerCoder']}\n);\n\nfunction evaluateProperties(serializedProperties, styleLayerProperties, feature, featureState, availableImages) {\n return mapObject(serializedProperties, (property, key) => {\n const prop = styleLayerProperties instanceof PossiblyEvaluated ? styleLayerProperties.get(key) : null;\n return prop && prop.evaluate ? prop.evaluate(feature, featureState, availableImages) : prop;\n });\n}\n\nfunction getBounds(geometry: Array) {\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n for (const p of geometry) {\n minX = Math.min(minX, p.x);\n minY = Math.min(minY, p.y);\n maxX = Math.max(maxX, p.x);\n maxY = Math.max(maxY, p.y);\n }\n return {minX, minY, maxX, maxY};\n}\n\nfunction topDownFeatureComparator(a, b) {\n return b - a;\n}\n","import {uniqueId, parseCacheControl} from '../util/util';\nimport {deserialize as deserializeBucket} from '../data/bucket';\nimport '../data/feature_index';\nimport type {FeatureIndex} from '../data/feature_index';\nimport {GeoJSONFeature} from '../util/vectortile_to_geojson';\nimport {featureFilter} from '@maplibre/maplibre-gl-style-spec';\nimport {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {CollisionBoxArray} from '../data/array_types.g';\nimport {Texture} from '../render/texture';\nimport {browser} from '../util/browser';\nimport {toEvaluationFeature} from '../data/evaluation_feature';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {SourceFeatureState} from '../source/source_state';\nimport {lazyLoadRTLTextPlugin} from './rtl_text_plugin';\n\nconst CLOCK_SKEW_RETRY_TIMEOUT = 30000;\n\nimport type {Bucket} from '../data/bucket';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {WorkerTileResult} from './worker_source';\nimport type {Actor} from '../util/actor';\nimport type {DEMData} from '../data/dem_data';\nimport type {AlphaImage} from '../util/image';\nimport type {ImageAtlas} from '../render/image_atlas';\nimport type {ImageManager} from '../render/image_manager';\nimport type {Context} from '../gl/context';\nimport type {OverscaledTileID} from './tile_id';\nimport type {Framebuffer} from '../gl/framebuffer';\nimport type {Transform} from '../geo/transform';\nimport type {LayerFeatureStates} from './source_state';\nimport type {Cancelable} from '../types/cancelable';\nimport type {FilterSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type Point from '@mapbox/point-geometry';\nimport {mat4} from 'gl-matrix';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\nimport {ExpiryData} from '../util/ajax';\n\n/**\n * The tile's state, can be:\n * - `loading` Tile data is in the process of loading.\n * - `loaded` Tile data has been loaded. Tile can be rendered.\n * - `reloading` Tile data has been loaded and is being updated. Tile can be rendered.\n * - `unloaded` Tile data has been deleted.\n * - `errored` Tile data was not loaded because of an error.\n * - `expired` Tile data was previously loaded, but has expired per its HTTP headers and is in the process of refreshing.\n */\nexport type TileState = 'loading' | 'loaded' | 'reloading' | 'unloaded' | 'errored' | 'expired';\n\n/**\n * @internal\n * A tile object is the combination of a Coordinate, which defines\n * its place, as well as a unique ID and data tracking for its content\n */\nexport class Tile {\n tileID: OverscaledTileID;\n uid: number;\n uses: number;\n tileSize: number;\n buckets: {[_: string]: Bucket};\n latestFeatureIndex: FeatureIndex;\n latestRawTileData: ArrayBuffer;\n imageAtlas: ImageAtlas;\n imageAtlasTexture: Texture;\n glyphAtlasImage: AlphaImage;\n glyphAtlasTexture: Texture;\n expirationTime: any;\n expiredRequestCount: number;\n state: TileState;\n timeAdded: number = 0;\n fadeEndTime: number = 0;\n collisionBoxArray: CollisionBoxArray;\n redoWhenDone: boolean;\n showCollisionBoxes: boolean;\n placementSource: any;\n actor: Actor;\n vtLayers: {[_: string]: VectorTileLayer};\n\n neighboringTiles: any;\n dem: DEMData;\n demMatrix: mat4;\n aborted: boolean;\n needsHillshadePrepare: boolean;\n needsTerrainPrepare: boolean;\n request: Cancelable;\n texture: any;\n fbo: Framebuffer;\n demTexture: Texture;\n refreshedUponExpiration: boolean;\n reloadCallback: any;\n resourceTiming: Array;\n queryPadding: number;\n\n symbolFadeHoldUntil: number;\n hasSymbolBuckets: boolean;\n hasRTLText: boolean;\n dependencies: any;\n rtt: Array<{id: number; stamp: number}>;\n rttCoords: {[_:string]: string};\n\n /**\n * @param tileID - the tile ID\n * @param size - The tile size\n */\n constructor(tileID: OverscaledTileID, size: number) {\n this.tileID = tileID;\n this.uid = uniqueId();\n this.uses = 0;\n this.tileSize = size;\n this.buckets = {};\n this.expirationTime = null;\n this.queryPadding = 0;\n this.hasSymbolBuckets = false;\n this.hasRTLText = false;\n this.dependencies = {};\n this.rtt = [];\n this.rttCoords = {};\n\n // Counts the number of times a response was already expired when\n // received. We're using this to add a delay when making a new request\n // so we don't have to keep retrying immediately in case of a server\n // serving expired tiles.\n this.expiredRequestCount = 0;\n\n this.state = 'loading';\n }\n\n registerFadeDuration(duration: number) {\n const fadeEndTime = duration + this.timeAdded;\n\n if (fadeEndTime < this.fadeEndTime) {\n return;\n }\n\n this.fadeEndTime = fadeEndTime;\n }\n\n wasRequested() {\n return this.state === 'errored' || this.state === 'loaded' || this.state === 'reloading';\n }\n\n clearTextures(painter: any) {\n if (this.demTexture) painter.saveTileTexture(this.demTexture);\n this.demTexture = null;\n }\n\n /**\n * Given a data object with a 'buffers' property, load it into\n * this tile's elementGroups and buffers properties and set loaded\n * to true. If the data is null, like in the case of an empty\n * GeoJSON tile, no-op but still set loaded to true.\n * @param data - The data from the worker\n * @param painter - the painter\n * @param justReloaded - `true` to just reload\n */\n loadVectorData(data: WorkerTileResult, painter: any, justReloaded?: boolean | null) {\n if (this.hasData()) {\n this.unloadVectorData();\n }\n\n this.state = 'loaded';\n\n // empty GeoJSON tile\n if (!data) {\n this.collisionBoxArray = new CollisionBoxArray();\n return;\n }\n\n if (data.featureIndex) {\n this.latestFeatureIndex = data.featureIndex;\n if (data.rawTileData) {\n // Only vector tiles have rawTileData, and they won't update it for\n // 'reloadTile'\n this.latestRawTileData = data.rawTileData;\n this.latestFeatureIndex.rawTileData = data.rawTileData;\n } else if (this.latestRawTileData) {\n // If rawTileData hasn't updated, hold onto a pointer to the last\n // one we received\n this.latestFeatureIndex.rawTileData = this.latestRawTileData;\n }\n }\n this.collisionBoxArray = data.collisionBoxArray;\n this.buckets = deserializeBucket(data.buckets, painter.style);\n\n this.hasSymbolBuckets = false;\n for (const id in this.buckets) {\n const bucket = this.buckets[id];\n if (bucket instanceof SymbolBucket) {\n this.hasSymbolBuckets = true;\n if (justReloaded) {\n bucket.justReloaded = true;\n } else {\n break;\n }\n }\n }\n\n this.hasRTLText = false;\n if (this.hasSymbolBuckets) {\n for (const id in this.buckets) {\n const bucket = this.buckets[id];\n if (bucket instanceof SymbolBucket) {\n if (bucket.hasRTLText) {\n this.hasRTLText = true;\n lazyLoadRTLTextPlugin();\n break;\n }\n }\n }\n }\n\n this.queryPadding = 0;\n for (const id in this.buckets) {\n const bucket = this.buckets[id];\n this.queryPadding = Math.max(this.queryPadding, painter.style.getLayer(id).queryRadius(bucket));\n }\n\n if (data.imageAtlas) {\n this.imageAtlas = data.imageAtlas;\n }\n if (data.glyphAtlasImage) {\n this.glyphAtlasImage = data.glyphAtlasImage;\n }\n }\n\n /**\n * Release any data or WebGL resources referenced by this tile.\n */\n unloadVectorData() {\n for (const id in this.buckets) {\n this.buckets[id].destroy();\n }\n this.buckets = {};\n\n if (this.imageAtlasTexture) {\n this.imageAtlasTexture.destroy();\n }\n\n if (this.imageAtlas) {\n this.imageAtlas = null;\n }\n\n if (this.glyphAtlasTexture) {\n this.glyphAtlasTexture.destroy();\n }\n\n this.latestFeatureIndex = null;\n this.state = 'unloaded';\n }\n\n getBucket(layer: StyleLayer) {\n return this.buckets[layer.id];\n }\n\n upload(context: Context) {\n for (const id in this.buckets) {\n const bucket = this.buckets[id];\n if (bucket.uploadPending()) {\n bucket.upload(context);\n }\n }\n\n const gl = context.gl;\n if (this.imageAtlas && !this.imageAtlas.uploaded) {\n this.imageAtlasTexture = new Texture(context, this.imageAtlas.image, gl.RGBA);\n this.imageAtlas.uploaded = true;\n }\n\n if (this.glyphAtlasImage) {\n this.glyphAtlasTexture = new Texture(context, this.glyphAtlasImage, gl.ALPHA);\n this.glyphAtlasImage = null;\n }\n }\n\n prepare(imageManager: ImageManager) {\n if (this.imageAtlas) {\n this.imageAtlas.patchUpdatedImages(imageManager, this.imageAtlasTexture);\n }\n }\n\n // Queries non-symbol features rendered for this tile.\n // Symbol features are queried globally\n queryRenderedFeatures(\n layers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n sourceFeatureState: SourceFeatureState,\n queryGeometry: Array,\n cameraQueryGeometry: Array,\n scale: number,\n params: {\n filter: FilterSpecification;\n layers: Array;\n availableImages: Array;\n },\n transform: Transform,\n maxPitchScaleFactor: number,\n pixelPosMatrix: mat4\n ): {[_: string]: Array<{featureIndex: number; feature: GeoJSONFeature}>} {\n if (!this.latestFeatureIndex || !this.latestFeatureIndex.rawTileData)\n return {};\n\n return this.latestFeatureIndex.query({\n queryGeometry,\n cameraQueryGeometry,\n scale,\n tileSize: this.tileSize,\n pixelPosMatrix,\n transform,\n params,\n queryPadding: this.queryPadding * maxPitchScaleFactor\n }, layers, serializedLayers, sourceFeatureState);\n }\n\n querySourceFeatures(result: Array, params?: {\n sourceLayer?: string;\n filter?: FilterSpecification;\n validate?: boolean;\n }) {\n const featureIndex = this.latestFeatureIndex;\n if (!featureIndex || !featureIndex.rawTileData) return;\n\n const vtLayers = featureIndex.loadVTLayers();\n\n const sourceLayer = params && params.sourceLayer ? params.sourceLayer : '';\n const layer = vtLayers._geojsonTileLayer || vtLayers[sourceLayer];\n\n if (!layer) return;\n\n const filter = featureFilter(params && params.filter);\n const {z, x, y} = this.tileID.canonical;\n const coord = {z, x, y};\n\n for (let i = 0; i < layer.length; i++) {\n const feature = layer.feature(i);\n if (filter.needGeometry) {\n const evaluationFeature = toEvaluationFeature(feature, true);\n if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), evaluationFeature, this.tileID.canonical)) continue;\n } else if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), feature)) {\n continue;\n }\n const id = featureIndex.getId(feature, sourceLayer);\n const geojsonFeature = new GeoJSONFeature(feature, z, x, y, id);\n (geojsonFeature as any).tile = coord;\n result.push(geojsonFeature);\n }\n }\n\n hasData() {\n return this.state === 'loaded' || this.state === 'reloading' || this.state === 'expired';\n }\n\n patternsLoaded() {\n return this.imageAtlas && !!Object.keys(this.imageAtlas.patternPositions).length;\n }\n\n setExpiryData(data: ExpiryData) {\n const prior = this.expirationTime;\n\n if (data.cacheControl) {\n const parsedCC = parseCacheControl(data.cacheControl);\n if (parsedCC['max-age']) this.expirationTime = Date.now() + parsedCC['max-age'] * 1000;\n } else if (data.expires) {\n this.expirationTime = new Date(data.expires).getTime();\n }\n\n if (this.expirationTime) {\n const now = Date.now();\n let isExpired = false;\n\n if (this.expirationTime > now) {\n isExpired = false;\n } else if (!prior) {\n isExpired = true;\n } else if (this.expirationTime < prior) {\n // Expiring date is going backwards:\n // fall back to exponential backoff\n isExpired = true;\n\n } else {\n const delta = this.expirationTime - prior;\n\n if (!delta) {\n // Server is serving the same expired resource over and over: fall\n // back to exponential backoff.\n isExpired = true;\n\n } else {\n // Assume that either the client or the server clock is wrong and\n // try to interpolate a valid expiration date (from the client POV)\n // observing a minimum timeout.\n this.expirationTime = now + Math.max(delta, CLOCK_SKEW_RETRY_TIMEOUT);\n\n }\n }\n\n if (isExpired) {\n this.expiredRequestCount++;\n this.state = 'expired';\n } else {\n this.expiredRequestCount = 0;\n }\n }\n }\n\n getExpiryTimeout() {\n if (this.expirationTime) {\n if (this.expiredRequestCount) {\n return 1000 * (1 << Math.min(this.expiredRequestCount - 1, 31));\n } else {\n // Max value for `setTimeout` implementations is a 32 bit integer; cap this accordingly\n return Math.min(this.expirationTime - new Date().getTime(), Math.pow(2, 31) - 1);\n }\n }\n }\n\n setFeatureState(states: LayerFeatureStates, painter: any) {\n if (!this.latestFeatureIndex ||\n !this.latestFeatureIndex.rawTileData ||\n Object.keys(states).length === 0) {\n return;\n }\n\n const vtLayers = this.latestFeatureIndex.loadVTLayers();\n\n for (const id in this.buckets) {\n if (!painter.style.hasLayer(id)) continue;\n\n const bucket = this.buckets[id];\n // Buckets are grouped by common source-layer\n const sourceLayerId = bucket.layers[0]['sourceLayer'] || '_geojsonTileLayer';\n const sourceLayer = vtLayers[sourceLayerId];\n const sourceLayerStates = states[sourceLayerId];\n if (!sourceLayer || !sourceLayerStates || Object.keys(sourceLayerStates).length === 0) continue;\n\n bucket.update(sourceLayerStates, sourceLayer, this.imageAtlas && this.imageAtlas.patternPositions || {});\n const layer = painter && painter.style && painter.style.getLayer(id);\n if (layer) {\n this.queryPadding = Math.max(this.queryPadding, layer.queryRadius(bucket));\n }\n }\n }\n\n holdingForFade(): boolean {\n return this.symbolFadeHoldUntil !== undefined;\n }\n\n symbolFadeFinished(): boolean {\n return !this.symbolFadeHoldUntil || this.symbolFadeHoldUntil < browser.now();\n }\n\n clearFadeHold() {\n this.symbolFadeHoldUntil = undefined;\n }\n\n setHoldDuration(duration: number) {\n this.symbolFadeHoldUntil = browser.now() + duration;\n }\n\n setDependencies(namespace: string, dependencies: Array) {\n const index = {};\n for (const dep of dependencies) {\n index[dep] = true;\n }\n this.dependencies[namespace] = index;\n }\n\n hasDependency(namespaces: Array, keys: Array) {\n for (const namespace of namespaces) {\n const dependencies = this.dependencies[namespace];\n if (dependencies) {\n for (const key of keys) {\n if (dependencies[key]) {\n return true;\n }\n }\n }\n }\n return false;\n }\n}\n","import type {CollisionBoxArray} from './array_types.g';\nimport type {Style} from '../style/style';\nimport type {TypedStyleLayer} from '../style/style_layer/typed_style_layer';\nimport type {FeatureIndex} from './feature_index';\nimport type {Context} from '../gl/context';\nimport type {FeatureStates} from '../source/source_state';\nimport type {ImagePosition} from '../render/image_atlas';\nimport type {CanonicalTileID} from '../source/tile_id';\nimport type {VectorTileFeature, VectorTileLayer} from '@mapbox/vector-tile';\nimport Point from '@mapbox/point-geometry';\n\nexport type BucketParameters = {\n index: number;\n layers: Array;\n zoom: number;\n pixelRatio: number;\n overscaling: number;\n collisionBoxArray: CollisionBoxArray;\n sourceLayerIndex: number;\n sourceID: string;\n};\n\nexport type PopulateParameters = {\n featureIndex: FeatureIndex;\n iconDependencies: {};\n patternDependencies: {};\n glyphDependencies: {};\n availableImages: Array;\n};\n\nexport type IndexedFeature = {\n feature: VectorTileFeature;\n id: number | string;\n index: number;\n sourceLayerIndex: number;\n};\n\nexport type BucketFeature = {\n index: number;\n sourceLayerIndex: number;\n geometry: Array>;\n properties: any;\n type: 0 | 1 | 2 | 3;\n id?: any;\n readonly patterns: {\n [_: string]: {\n 'min': string;\n 'mid': string;\n 'max': string;\n };\n };\n sortKey?: number;\n};\n\n/**\n * The `Bucket` interface is the single point of knowledge about turning vector\n * tiles into WebGL buffers.\n *\n * `Bucket` is an abstract interface. An implementation exists for each style layer type.\n * Create a bucket via the `StyleLayer#createBucket` method.\n *\n * The concrete bucket types, using layout options from the style layer,\n * transform feature geometries into vertex and index data for use by the\n * vertex shader. They also (via `ProgramConfiguration`) use feature\n * properties and the zoom level to populate the attributes needed for\n * data-driven styling.\n *\n * Buckets are designed to be built on a worker thread and then serialized and\n * transferred back to the main thread for rendering. On the worker side, a\n * bucket's vertex, index, and attribute data is stored in `bucket.arrays: ArrayGroup`.\n * When a bucket's data is serialized and sent back to the main thread,\n * is gets deserialized (using `new Bucket(serializedBucketData)`, with\n * the array data now stored in `bucket.buffers: BufferGroup`. BufferGroups\n * hold the same data as ArrayGroups, but are tuned for consumption by WebGL.\n */\nexport interface Bucket {\n layerIds: Array;\n hasPattern: boolean;\n readonly layers: Array;\n readonly stateDependentLayers: Array;\n readonly stateDependentLayerIds: Array;\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID): void;\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}): void;\n isEmpty(): boolean;\n upload(context: Context): void;\n uploadPending(): boolean;\n /**\n * Release the WebGL resources associated with the buffers. Note that because\n * buckets are shared between layers having the same layout properties, they\n * must be destroyed in groups (all buckets for a tile, or all symbol buckets).\n */\n destroy(): void;\n}\n\nexport function deserialize(input: Array, style: Style): {[_: string]: Bucket} {\n const output = {};\n\n // Guard against the case where the map's style has been set to null while\n // this bucket has been parsing.\n if (!style) return output;\n\n for (const bucket of input) {\n const layers = bucket.layerIds\n .map((id) => style.getLayer(id))\n .filter(Boolean);\n\n if (layers.length === 0) {\n continue;\n }\n\n // look up StyleLayer objects from layer ids (since we don't\n // want to waste time serializing/copying them from the worker)\n (bucket as any).layers = layers;\n if (bucket.stateDependentLayerIds) {\n (bucket as any).stateDependentLayers = bucket.stateDependentLayerIds.map((lId) => layers.filter((l) => l.id === lId)[0]);\n }\n for (const layer of layers) {\n output[layer.id] = bucket;\n }\n }\n\n return output;\n}\n","import {OverscaledTileID} from './tile_id';\nimport type {Tile} from './tile';\n\n/**\n * @internal\n * A [least-recently-used cache](http://en.wikipedia.org/wiki/Cache_algorithms)\n * with hash lookup made possible by keeping a list of keys in parallel to\n * an array of dictionary of values\n */\nexport class TileCache {\n max: number;\n data: {\n [key: string]: Array<{\n value: Tile;\n timeout: ReturnType;\n }>;\n };\n order: Array;\n onRemove: (element: Tile) => void;\n /**\n * @param max - number of permitted values\n * @param onRemove - callback called with items when they expire\n */\n constructor(max: number, onRemove: (element: Tile) => void) {\n this.max = max;\n this.onRemove = onRemove;\n this.reset();\n }\n\n /**\n * Clear the cache\n *\n * @returns this cache\n */\n reset() {\n for (const key in this.data) {\n for (const removedData of this.data[key]) {\n if (removedData.timeout) clearTimeout(removedData.timeout);\n this.onRemove(removedData.value);\n }\n }\n\n this.data = {};\n this.order = [];\n\n return this;\n }\n\n /**\n * Add a key, value combination to the cache, trimming its size if this pushes\n * it over max length.\n *\n * @param tileID - lookup key for the item\n * @param data - tile data\n *\n * @returns this cache\n */\n add(tileID: OverscaledTileID, data: Tile, expiryTimeout: number | void) {\n const key = tileID.wrapped().key;\n if (this.data[key] === undefined) {\n this.data[key] = [];\n }\n\n const dataWrapper = {\n value: data,\n timeout: undefined\n };\n\n if (expiryTimeout !== undefined) {\n dataWrapper.timeout = setTimeout(() => {\n this.remove(tileID, dataWrapper);\n }, expiryTimeout as number);\n }\n\n this.data[key].push(dataWrapper);\n this.order.push(key);\n\n if (this.order.length > this.max) {\n const removedData = this._getAndRemoveByKey(this.order[0]);\n if (removedData) this.onRemove(removedData);\n }\n\n return this;\n }\n\n /**\n * Determine whether the value attached to `key` is present\n *\n * @param tileID - the key to be looked-up\n * @returns whether the cache has this value\n */\n has(tileID: OverscaledTileID): boolean {\n return tileID.wrapped().key in this.data;\n }\n\n /**\n * Get the value attached to a specific key and remove data from cache.\n * If the key is not found, returns `null`\n *\n * @param tileID - the key to look up\n * @returns the tile data, or null if it isn't found\n */\n getAndRemove(tileID: OverscaledTileID): Tile {\n if (!this.has(tileID)) { return null; }\n return this._getAndRemoveByKey(tileID.wrapped().key);\n }\n\n /*\n * Get and remove the value with the specified key.\n */\n _getAndRemoveByKey(key: string): Tile {\n const data = this.data[key].shift();\n if (data.timeout) clearTimeout(data.timeout);\n\n if (this.data[key].length === 0) {\n delete this.data[key];\n }\n this.order.splice(this.order.indexOf(key), 1);\n\n return data.value;\n }\n\n /*\n * Get the value with the specified (wrapped tile) key.\n */\n getByKey(key: string): Tile {\n const data = this.data[key];\n return data ? data[0].value : null;\n }\n\n /**\n * Get the value attached to a specific key without removing data\n * from the cache. If the key is not found, returns `null`\n *\n * @param tileID - the key to look up\n * @returns the tile data, or null if it isn't found\n */\n get(tileID: OverscaledTileID): Tile {\n if (!this.has(tileID)) { return null; }\n\n const data = this.data[tileID.wrapped().key][0];\n return data.value;\n }\n\n /**\n * Remove a key/value combination from the cache.\n *\n * @param tileID - the key for the pair to delete\n * @param value - If a value is provided, remove that exact version of the value.\n * @returns this cache\n */\n remove(tileID: OverscaledTileID, value?: {\n value: Tile;\n timeout: ReturnType;\n }) {\n if (!this.has(tileID)) { return this; }\n const key = tileID.wrapped().key;\n\n const dataIndex = value === undefined ? 0 : this.data[key].indexOf(value);\n const data = this.data[key][dataIndex];\n this.data[key].splice(dataIndex, 1);\n if (data.timeout) clearTimeout(data.timeout);\n if (this.data[key].length === 0) {\n delete this.data[key];\n }\n this.onRemove(data.value);\n this.order.splice(this.order.indexOf(key), 1);\n\n return this;\n }\n\n /**\n * Change the max size of the cache.\n *\n * @param max - the max size of the cache\n * @returns this cache\n */\n setMaxSize(max: number): TileCache {\n this.max = max;\n\n while (this.order.length > this.max) {\n const removedData = this._getAndRemoveByKey(this.order[0]);\n if (removedData) this.onRemove(removedData);\n }\n\n return this;\n }\n\n /**\n * Remove entries that do not pass a filter function. Used for removing\n * stale tiles from the cache.\n *\n * @param filterFn - Determines whether the tile is filtered. If the supplied function returns false, the tile will be filtered out.\n */\n filter(filterFn: (tile: Tile) => boolean) {\n const removed = [];\n for (const key in this.data) {\n for (const entry of this.data[key]) {\n if (!filterFn(entry.value)) {\n removed.push(entry);\n }\n }\n }\n for (const r of removed) {\n this.remove(r.value.tileID, r);\n }\n }\n}\n","import {extend} from '../util/util';\nimport {Tile} from './tile';\nimport type {FeatureState} from '@maplibre/maplibre-gl-style-spec';\n\nexport type FeatureStates = {[featureId: string]: FeatureState};\nexport type LayerFeatureStates = {[layer: string]: FeatureStates};\n\n/**\n * @internal\n * SourceFeatureState manages the state and pending changes\n * to features in a source, separated by source layer.\n * stateChanges and deletedStates batch all changes to the tile (updates and removes, respectively)\n * between coalesce() events. addFeatureState() and removeFeatureState() also update their counterpart's\n * list of changes, such that coalesce() can apply the proper state changes while agnostic to the order of operations.\n * In deletedStates, all null's denote complete removal of state at that scope\n*/\nexport class SourceFeatureState {\n state: LayerFeatureStates;\n stateChanges: LayerFeatureStates;\n deletedStates: {};\n\n constructor() {\n this.state = {};\n this.stateChanges = {};\n this.deletedStates = {};\n }\n\n updateState(sourceLayer: string, featureId: number | string, newState: any) {\n const feature = String(featureId);\n this.stateChanges[sourceLayer] = this.stateChanges[sourceLayer] || {};\n this.stateChanges[sourceLayer][feature] = this.stateChanges[sourceLayer][feature] || {};\n extend(this.stateChanges[sourceLayer][feature], newState);\n\n if (this.deletedStates[sourceLayer] === null) {\n this.deletedStates[sourceLayer] = {};\n for (const ft in this.state[sourceLayer]) {\n if (ft !== feature) this.deletedStates[sourceLayer][ft] = null;\n }\n } else {\n const featureDeletionQueued = this.deletedStates[sourceLayer] && this.deletedStates[sourceLayer][feature] === null;\n if (featureDeletionQueued) {\n this.deletedStates[sourceLayer][feature] = {};\n for (const prop in this.state[sourceLayer][feature]) {\n if (!newState[prop]) this.deletedStates[sourceLayer][feature][prop] = null;\n }\n } else {\n for (const key in newState) {\n const deletionInQueue = this.deletedStates[sourceLayer] && this.deletedStates[sourceLayer][feature] && this.deletedStates[sourceLayer][feature][key] === null;\n if (deletionInQueue) delete this.deletedStates[sourceLayer][feature][key];\n }\n }\n }\n }\n\n removeFeatureState(sourceLayer: string, featureId?: number | string, key?: string) {\n const sourceLayerDeleted = this.deletedStates[sourceLayer] === null;\n if (sourceLayerDeleted) return;\n\n const feature = String(featureId);\n\n this.deletedStates[sourceLayer] = this.deletedStates[sourceLayer] || {};\n\n if (key && featureId !== undefined) {\n if (this.deletedStates[sourceLayer][feature] !== null) {\n this.deletedStates[sourceLayer][feature] = this.deletedStates[sourceLayer][feature] || {};\n this.deletedStates[sourceLayer][feature][key] = null;\n }\n } else if (featureId !== undefined) {\n const updateInQueue = this.stateChanges[sourceLayer] && this.stateChanges[sourceLayer][feature];\n if (updateInQueue) {\n this.deletedStates[sourceLayer][feature] = {};\n for (key in this.stateChanges[sourceLayer][feature]) this.deletedStates[sourceLayer][feature][key] = null;\n\n } else {\n this.deletedStates[sourceLayer][feature] = null;\n }\n } else {\n this.deletedStates[sourceLayer] = null;\n }\n\n }\n\n getState(sourceLayer: string, featureId: number | string) {\n const feature = String(featureId);\n const base = this.state[sourceLayer] || {};\n const changes = this.stateChanges[sourceLayer] || {};\n\n const reconciledState = extend({}, base[feature], changes[feature]);\n\n //return empty object if the whole source layer is awaiting deletion\n if (this.deletedStates[sourceLayer] === null) return {};\n else if (this.deletedStates[sourceLayer]) {\n const featureDeletions = this.deletedStates[sourceLayer][featureId];\n if (featureDeletions === null) return {};\n for (const prop in featureDeletions) delete reconciledState[prop];\n }\n return reconciledState;\n }\n\n initializeTileState(tile: Tile, painter: any) {\n tile.setFeatureState(this.state, painter);\n }\n\n coalesceChanges(tiles: {\n [_ in any]: Tile;\n }, painter: any) {\n //track changes with full state objects, but only for features that got modified\n const featuresChanged: LayerFeatureStates = {};\n\n for (const sourceLayer in this.stateChanges) {\n this.state[sourceLayer] = this.state[sourceLayer] || {};\n const layerStates = {};\n for (const feature in this.stateChanges[sourceLayer]) {\n if (!this.state[sourceLayer][feature]) this.state[sourceLayer][feature] = {};\n extend(this.state[sourceLayer][feature], this.stateChanges[sourceLayer][feature]);\n layerStates[feature] = this.state[sourceLayer][feature];\n }\n featuresChanged[sourceLayer] = layerStates;\n }\n\n for (const sourceLayer in this.deletedStates) {\n this.state[sourceLayer] = this.state[sourceLayer] || {};\n const layerStates = {};\n\n if (this.deletedStates[sourceLayer] === null) {\n for (const ft in this.state[sourceLayer]) {\n layerStates[ft] = {};\n this.state[sourceLayer][ft] = {};\n }\n } else {\n for (const feature in this.deletedStates[sourceLayer]) {\n const deleteWholeFeatureState = this.deletedStates[sourceLayer][feature] === null;\n if (deleteWholeFeatureState) this.state[sourceLayer][feature] = {};\n else {\n for (const key of Object.keys(this.deletedStates[sourceLayer][feature])) {\n delete this.state[sourceLayer][feature][key];\n }\n }\n layerStates[feature] = this.state[sourceLayer][feature];\n }\n }\n\n featuresChanged[sourceLayer] = featuresChanged[sourceLayer] || {};\n extend(featuresChanged[sourceLayer], layerStates);\n }\n\n this.stateChanges = {};\n this.deletedStates = {};\n\n if (Object.keys(featuresChanged).length === 0) return;\n\n for (const id in tiles) {\n const tile = tiles[id];\n tile.setFeatureState(featuresChanged, painter);\n }\n }\n}\n","import {create as createSource} from './source';\n\nimport {Tile} from './tile';\nimport {Event, ErrorEvent, Evented} from '../util/evented';\nimport {TileCache} from './tile_cache';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\nimport {keysDifference} from '../util/util';\nimport {EXTENT} from '../data/extent';\nimport {Context} from '../gl/context';\nimport Point from '@mapbox/point-geometry';\nimport {browser} from '../util/browser';\nimport {OverscaledTileID} from './tile_id';\nimport {SourceFeatureState} from './source_state';\n\nimport type {Source} from './source';\nimport type {Map} from '../ui/map';\nimport type {Style} from '../style/style';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Transform} from '../geo/transform';\nimport type {TileState} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {SourceSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {MapSourceDataEvent} from '../ui/events';\nimport {Terrain} from '../render/terrain';\nimport {config} from '../util/config';\n\n/**\n * @internal\n * `SourceCache` is responsible for\n *\n * - creating an instance of `Source`\n * - forwarding events from `Source`\n * - caching tiles loaded from an instance of `Source`\n * - loading the tiles needed to render a given viewport\n * - unloading the cached tiles not needed to render a given viewport\n */\nexport class SourceCache extends Evented {\n id: string;\n dispatcher: Dispatcher;\n map: Map;\n style: Style;\n\n _source: Source;\n _sourceLoaded: boolean;\n _sourceErrored: boolean;\n _tiles: {[_: string]: Tile};\n _prevLng: number;\n _cache: TileCache;\n _timers: {\n [_ in any]: ReturnType;\n };\n _cacheTimers: {\n [_ in any]: ReturnType;\n };\n _maxTileCacheSize: number;\n _maxTileCacheZoomLevels: number;\n _paused: boolean;\n _shouldReloadOnResume: boolean;\n _coveredTiles: {[_: string]: boolean};\n transform: Transform;\n terrain: Terrain;\n used: boolean;\n usedForTerrain: boolean;\n tileSize: number;\n _state: SourceFeatureState;\n _loadedParentTiles: {[_: string]: Tile};\n _didEmitContent: boolean;\n _updated: boolean;\n\n static maxUnderzooming: number;\n static maxOverzooming: number;\n\n constructor(id: string, options: SourceSpecification, dispatcher: Dispatcher) {\n super();\n this.id = id;\n this.dispatcher = dispatcher;\n\n this.on('data', (e: MapSourceDataEvent) => {\n // this._sourceLoaded signifies that the TileJSON is loaded if applicable.\n // if the source type does not come with a TileJSON, the flag signifies the\n // source data has loaded (i.e geojson has been tiled on the worker and is ready)\n if (e.dataType === 'source' && e.sourceDataType === 'metadata') this._sourceLoaded = true;\n\n // for sources with mutable data, this event fires when the underlying data\n // to a source is changed. (i.e. GeoJSONSource#setData and ImageSource#serCoordinates)\n if (this._sourceLoaded && !this._paused && e.dataType === 'source' && e.sourceDataType === 'content') {\n this.reload();\n if (this.transform) {\n this.update(this.transform, this.terrain);\n }\n\n this._didEmitContent = true;\n }\n });\n\n this.on('dataloading', () => {\n this._sourceErrored = false;\n });\n\n this.on('error', () => {\n // Only set _sourceErrored if the source does not have pending loads.\n this._sourceErrored = this._source.loaded();\n });\n\n this._source = createSource(id, options, dispatcher, this);\n\n this._tiles = {};\n this._cache = new TileCache(0, this._unloadTile.bind(this));\n this._timers = {};\n this._cacheTimers = {};\n this._maxTileCacheSize = null;\n this._maxTileCacheZoomLevels = null;\n this._loadedParentTiles = {};\n\n this._coveredTiles = {};\n this._state = new SourceFeatureState();\n this._didEmitContent = false;\n this._updated = false;\n }\n\n onAdd(map: Map) {\n this.map = map;\n this._maxTileCacheSize = map ? map._maxTileCacheSize : null;\n this._maxTileCacheZoomLevels = map ? map._maxTileCacheZoomLevels : null;\n if (this._source && this._source.onAdd) {\n this._source.onAdd(map);\n }\n }\n\n onRemove(map: Map) {\n this.clearTiles();\n if (this._source && this._source.onRemove) {\n this._source.onRemove(map);\n }\n }\n\n /**\n * Return true if no tile data is pending, tiles will not change unless\n * an additional API call is received.\n */\n loaded(): boolean {\n if (this._sourceErrored) { return true; }\n if (!this._sourceLoaded) { return false; }\n if (!this._source.loaded()) { return false; }\n if ((this.used !== undefined || this.usedForTerrain !== undefined) && !this.used && !this.usedForTerrain) { return true; }\n // do not consider as loaded if the update hasn't been called yet (we do not know if we will have any tiles to fetch)\n if (!this._updated) { return false; }\n\n for (const t in this._tiles) {\n const tile = this._tiles[t];\n if (tile.state !== 'loaded' && tile.state !== 'errored')\n return false;\n }\n return true;\n }\n\n getSource(): Source {\n return this._source;\n }\n\n pause() {\n this._paused = true;\n }\n\n resume() {\n if (!this._paused) return;\n const shouldReload = this._shouldReloadOnResume;\n this._paused = false;\n this._shouldReloadOnResume = false;\n if (shouldReload) this.reload();\n if (this.transform) this.update(this.transform, this.terrain);\n }\n\n _loadTile(tile: Tile, callback: Callback) {\n return this._source.loadTile(tile, callback);\n }\n\n _unloadTile(tile: Tile) {\n if (this._source.unloadTile)\n return this._source.unloadTile(tile, () => {});\n }\n\n _abortTile(tile: Tile) {\n if (this._source.abortTile)\n this._source.abortTile(tile, () => {});\n\n this._source.fire(new Event('dataabort', {tile, coord: tile.tileID, dataType: 'source'}));\n }\n\n serialize() {\n return this._source.serialize();\n }\n\n prepare(context: Context) {\n if (this._source.prepare) {\n this._source.prepare();\n }\n\n this._state.coalesceChanges(this._tiles, this.map ? this.map.painter : null);\n for (const i in this._tiles) {\n const tile = this._tiles[i];\n tile.upload(context);\n tile.prepare(this.map.style.imageManager);\n }\n }\n\n /**\n * Return all tile ids ordered with z-order, and cast to numbers\n */\n getIds(): Array {\n return (Object.values(this._tiles) as any).map((tile: Tile) => tile.tileID).sort(compareTileId).map(id => id.key);\n }\n\n getRenderableIds(symbolLayer?: boolean): Array {\n const renderables: Array = [];\n for (const id in this._tiles) {\n if (this._isIdRenderable(id, symbolLayer)) renderables.push(this._tiles[id]);\n }\n if (symbolLayer) {\n return renderables.sort((a_: Tile, b_: Tile) => {\n const a = a_.tileID;\n const b = b_.tileID;\n const rotatedA = (new Point(a.canonical.x, a.canonical.y))._rotate(this.transform.angle);\n const rotatedB = (new Point(b.canonical.x, b.canonical.y))._rotate(this.transform.angle);\n return a.overscaledZ - b.overscaledZ || rotatedB.y - rotatedA.y || rotatedB.x - rotatedA.x;\n }).map(tile => tile.tileID.key);\n }\n return renderables.map(tile => tile.tileID).sort(compareTileId).map(id => id.key);\n }\n\n hasRenderableParent(tileID: OverscaledTileID) {\n const parentTile = this.findLoadedParent(tileID, 0);\n if (parentTile) {\n return this._isIdRenderable(parentTile.tileID.key);\n }\n return false;\n }\n\n _isIdRenderable(id: string, symbolLayer?: boolean) {\n return this._tiles[id] && this._tiles[id].hasData() &&\n !this._coveredTiles[id] && (symbolLayer || !this._tiles[id].holdingForFade());\n }\n\n reload() {\n if (this._paused) {\n this._shouldReloadOnResume = true;\n return;\n }\n\n this._cache.reset();\n\n for (const i in this._tiles) {\n if (this._tiles[i].state !== 'errored') this._reloadTile(i, 'reloading');\n }\n }\n\n _reloadTile(id: string, state: TileState) {\n const tile = this._tiles[id];\n\n // this potentially does not address all underlying\n // issues https://github.com/mapbox/mapbox-gl-js/issues/4252\n // - hard to tell without repro steps\n if (!tile) return;\n\n // The difference between \"loading\" tiles and \"reloading\" or \"expired\"\n // tiles is that \"reloading\"/\"expired\" tiles are \"renderable\".\n // Therefore, a \"loading\" tile cannot become a \"reloading\" tile without\n // first becoming a \"loaded\" tile.\n if (tile.state !== 'loading') {\n tile.state = state;\n }\n\n this._loadTile(tile, this._tileLoaded.bind(this, tile, id, state));\n }\n\n _tileLoaded(tile: Tile, id: string, previousState: TileState, err?: Error | null) {\n if (err) {\n tile.state = 'errored';\n if ((err as any).status !== 404) this._source.fire(new ErrorEvent(err, {tile}));\n // continue to try loading parent/children tiles if a tile doesn't exist (404)\n else this.update(this.transform, this.terrain);\n return;\n }\n\n tile.timeAdded = browser.now();\n if (previousState === 'expired') tile.refreshedUponExpiration = true;\n this._setTileReloadTimer(id, tile);\n if (this.getSource().type === 'raster-dem' && tile.dem) this._backfillDEM(tile);\n this._state.initializeTileState(tile, this.map ? this.map.painter : null);\n\n if (!tile.aborted) {\n this._source.fire(new Event('data', {dataType: 'source', tile, coord: tile.tileID}));\n }\n }\n\n /**\n * For raster terrain source, backfill DEM to eliminate visible tile boundaries\n */\n _backfillDEM(tile: Tile) {\n const renderables = this.getRenderableIds();\n for (let i = 0; i < renderables.length; i++) {\n const borderId = renderables[i];\n if (tile.neighboringTiles && tile.neighboringTiles[borderId]) {\n const borderTile = this.getTileByID(borderId);\n fillBorder(tile, borderTile);\n fillBorder(borderTile, tile);\n }\n }\n\n function fillBorder(tile, borderTile) {\n tile.needsHillshadePrepare = true;\n tile.needsTerrainPrepare = true;\n let dx = borderTile.tileID.canonical.x - tile.tileID.canonical.x;\n const dy = borderTile.tileID.canonical.y - tile.tileID.canonical.y;\n const dim = Math.pow(2, tile.tileID.canonical.z);\n const borderId = borderTile.tileID.key;\n if (dx === 0 && dy === 0) return;\n\n if (Math.abs(dy) > 1) {\n return;\n }\n if (Math.abs(dx) > 1) {\n // Adjust the delta coordinate for world wraparound.\n if (Math.abs(dx + dim) === 1) {\n dx += dim;\n } else if (Math.abs(dx - dim) === 1) {\n dx -= dim;\n }\n }\n if (!borderTile.dem || !tile.dem) return;\n tile.dem.backfillBorder(borderTile.dem, dx, dy);\n if (tile.neighboringTiles && tile.neighboringTiles[borderId])\n tile.neighboringTiles[borderId].backfilled = true;\n }\n }\n /**\n * Get a specific tile by TileID\n */\n getTile(tileID: OverscaledTileID): Tile {\n return this.getTileByID(tileID.key);\n }\n\n /**\n * Get a specific tile by id\n */\n getTileByID(id: string): Tile {\n return this._tiles[id];\n }\n\n /**\n * For a given set of tiles, retain children that are loaded and have a zoom\n * between `zoom` (exclusive) and `maxCoveringZoom` (inclusive)\n */\n _retainLoadedChildren(\n idealTiles: {\n [_ in any]: OverscaledTileID;\n },\n zoom: number,\n maxCoveringZoom: number,\n retain: {\n [_ in any]: OverscaledTileID;\n }\n ) {\n for (const id in this._tiles) {\n let tile = this._tiles[id];\n\n // only consider renderable tiles up to maxCoveringZoom\n if (retain[id] ||\n !tile.hasData() ||\n tile.tileID.overscaledZ <= zoom ||\n tile.tileID.overscaledZ > maxCoveringZoom\n ) continue;\n\n // loop through parents and retain the topmost loaded one if found\n let topmostLoadedID = tile.tileID;\n while (tile && tile.tileID.overscaledZ > zoom + 1) {\n const parentID = tile.tileID.scaledTo(tile.tileID.overscaledZ - 1);\n\n tile = this._tiles[parentID.key];\n\n if (tile && tile.hasData()) {\n topmostLoadedID = parentID;\n }\n }\n\n // loop through ancestors of the topmost loaded child to see if there's one that needed it\n let tileID = topmostLoadedID;\n while (tileID.overscaledZ > zoom) {\n tileID = tileID.scaledTo(tileID.overscaledZ - 1);\n\n if (idealTiles[tileID.key]) {\n // found a parent that needed a loaded child; retain that child\n retain[topmostLoadedID.key] = topmostLoadedID;\n break;\n }\n }\n }\n }\n\n /**\n * Find a loaded parent of the given tile (up to minCoveringZoom)\n */\n findLoadedParent(tileID: OverscaledTileID, minCoveringZoom: number): Tile {\n if (tileID.key in this._loadedParentTiles) {\n const parent = this._loadedParentTiles[tileID.key];\n if (parent && parent.tileID.overscaledZ >= minCoveringZoom) {\n return parent;\n } else {\n return null;\n }\n }\n for (let z = tileID.overscaledZ - 1; z >= minCoveringZoom; z--) {\n const parentTileID = tileID.scaledTo(z);\n const tile = this._getLoadedTile(parentTileID);\n if (tile) {\n return tile;\n }\n }\n }\n\n _getLoadedTile(tileID: OverscaledTileID): Tile {\n const tile = this._tiles[tileID.key];\n if (tile && tile.hasData()) {\n return tile;\n }\n // TileCache ignores wrap in lookup.\n const cachedTile = this._cache.getByKey(tileID.wrapped().key);\n return cachedTile;\n }\n\n /**\n * Resizes the tile cache based on the current viewport's size\n * or the maxTileCacheSize option passed during map creation\n *\n * Larger viewports use more tiles and need larger caches. Larger viewports\n * are more likely to be found on devices with more memory and on pages where\n * the map is more important.\n */\n updateCacheSize(transform: Transform) {\n const widthInTiles = Math.ceil(transform.width / this._source.tileSize) + 1;\n const heightInTiles = Math.ceil(transform.height / this._source.tileSize) + 1;\n const approxTilesInView = widthInTiles * heightInTiles;\n const commonZoomRange = this._maxTileCacheZoomLevels === null ?\n config.MAX_TILE_CACHE_ZOOM_LEVELS : this._maxTileCacheZoomLevels;\n const viewDependentMaxSize = Math.floor(approxTilesInView * commonZoomRange);\n const maxSize = typeof this._maxTileCacheSize === 'number' ?\n Math.min(this._maxTileCacheSize, viewDependentMaxSize) : viewDependentMaxSize;\n\n this._cache.setMaxSize(maxSize);\n }\n\n handleWrapJump(lng: number) {\n // On top of the regular z/x/y values, TileIDs have a `wrap` value that specify\n // which cppy of the world the tile belongs to. For example, at `lng: 10` you\n // might render z/x/y/0 while at `lng: 370` you would render z/x/y/1.\n //\n // When lng values get wrapped (going from `lng: 370` to `long: 10`) you expect\n // to see the same thing on the screen (370 degrees and 10 degrees is the same\n // place in the world) but all the TileIDs will have different wrap values.\n //\n // In order to make this transition seamless, we calculate the rounded difference of\n // \"worlds\" between the last frame and the current frame. If the map panned by\n // a world, then we can assign all the tiles new TileIDs with updated wrap values.\n // For example, assign z/x/y/1 a new id: z/x/y/0. It is the same tile, just rendered\n // in a different position.\n //\n // This enables us to reuse the tiles at more ideal locations and prevent flickering.\n const prevLng = this._prevLng === undefined ? lng : this._prevLng;\n const lngDifference = lng - prevLng;\n const worldDifference = lngDifference / 360;\n const wrapDelta = Math.round(worldDifference);\n this._prevLng = lng;\n\n if (wrapDelta) {\n const tiles: {[_: string]: Tile} = {};\n for (const key in this._tiles) {\n const tile = this._tiles[key];\n tile.tileID = tile.tileID.unwrapTo(tile.tileID.wrap + wrapDelta);\n tiles[tile.tileID.key] = tile;\n }\n this._tiles = tiles;\n\n // Reset tile reload timers\n for (const id in this._timers) {\n clearTimeout(this._timers[id]);\n delete this._timers[id];\n }\n for (const id in this._tiles) {\n const tile = this._tiles[id];\n this._setTileReloadTimer(id, tile);\n }\n }\n }\n\n /**\n * Removes tiles that are outside the viewport and adds new tiles that\n * are inside the viewport.\n */\n update(transform: Transform, terrain?: Terrain) {\n this.transform = transform;\n this.terrain = terrain;\n if (!this._sourceLoaded || this._paused) { return; }\n\n this.updateCacheSize(transform);\n this.handleWrapJump(this.transform.center.lng);\n\n // Covered is a list of retained tiles who's areas are fully covered by other,\n // better, retained tiles. They are not drawn separately.\n this._coveredTiles = {};\n\n let idealTileIDs;\n if (!this.used && !this.usedForTerrain) {\n idealTileIDs = [];\n } else if (this._source.tileID) {\n idealTileIDs = transform.getVisibleUnwrappedCoordinates(this._source.tileID)\n .map((unwrapped) => new OverscaledTileID(unwrapped.canonical.z, unwrapped.wrap, unwrapped.canonical.z, unwrapped.canonical.x, unwrapped.canonical.y));\n } else {\n idealTileIDs = transform.coveringTiles({\n tileSize: this.usedForTerrain ? this.tileSize : this._source.tileSize,\n minzoom: this._source.minzoom,\n maxzoom: this._source.maxzoom,\n roundZoom: this.usedForTerrain ? false : this._source.roundZoom,\n reparseOverscaled: this._source.reparseOverscaled,\n terrain\n });\n\n if (this._source.hasTile) {\n idealTileIDs = idealTileIDs.filter((coord) => (this._source.hasTile as any)(coord));\n }\n }\n\n // Determine the overzooming/underzooming amounts.\n const zoom = transform.coveringZoomLevel(this._source);\n const minCoveringZoom = Math.max(zoom - SourceCache.maxOverzooming, this._source.minzoom);\n const maxCoveringZoom = Math.max(zoom + SourceCache.maxUnderzooming, this._source.minzoom);\n\n // When sourcecache is used for terrain also load parent tiles to avoid flickering when zooming out\n if (this.usedForTerrain) {\n const parents = {};\n for (const tileID of idealTileIDs) {\n if (tileID.canonical.z > this._source.minzoom) {\n const parent = tileID.scaledTo(tileID.canonical.z - 1);\n parents[parent.key] = parent;\n // load very low zoom to calculate tile visibility in transform.coveringTiles and high zoomlevels correct\n const parent2 = tileID.scaledTo(Math.max(this._source.minzoom, Math.min(tileID.canonical.z, 5)));\n parents[parent2.key] = parent2;\n }\n }\n idealTileIDs = idealTileIDs.concat(Object.values(parents));\n }\n\n const noPendingDataEmissions = idealTileIDs.length === 0 && !this._updated && this._didEmitContent;\n this._updated = true;\n // if we won't have any tiles to fetch and content is already emitted\n // there will be no more data emissions, so we need to emit the event with isSourceLoaded = true\n if (noPendingDataEmissions) {\n this.fire(new Event('data', {sourceDataType: 'idle', dataType: 'source', sourceId: this.id}));\n }\n\n // Retain is a list of tiles that we shouldn't delete, even if they are not\n // the most ideal tile for the current viewport. This may include tiles like\n // parent or child tiles that are *already* loaded.\n const retain = this._updateRetainedTiles(idealTileIDs, zoom);\n\n if (isRasterType(this._source.type)) {\n const parentsForFading: {[_: string]: OverscaledTileID} = {};\n const fadingTiles = {};\n const ids = Object.keys(retain);\n const now = browser.now();\n for (const id of ids) {\n const tileID = retain[id];\n\n const tile = this._tiles[id];\n\n // when fadeEndTime is 0, the tile is created but registerFadeDuration\n // has not been called, therefore must be kept in fadingTiles dictionary\n // for next round of rendering\n if (!tile || (tile.fadeEndTime !== 0 && tile.fadeEndTime <= now)) {\n continue;\n }\n\n // if the tile is loaded but still fading in, find parents to cross-fade with it\n const parentTile = this.findLoadedParent(tileID, minCoveringZoom);\n if (parentTile) {\n this._addTile(parentTile.tileID);\n parentsForFading[parentTile.tileID.key] = parentTile.tileID;\n }\n\n fadingTiles[id] = tileID;\n }\n\n // for tiles that are still fading in, also find children to cross-fade with\n this._retainLoadedChildren(fadingTiles, zoom, maxCoveringZoom, retain);\n\n for (const id in parentsForFading) {\n if (!retain[id]) {\n // If a tile is only needed for fading, mark it as covered so that it isn't rendered on it's own.\n this._coveredTiles[id] = true;\n retain[id] = parentsForFading[id];\n }\n }\n\n // disable fading logic in terrain3D mode to avoid rendering two tiles on the same place\n if (terrain) {\n const idealRasterTileIDs: {[_: string]: OverscaledTileID} = {};\n const missingTileIDs: {[_: string]: OverscaledTileID} = {};\n for (const tileID of idealTileIDs) {\n if (this._tiles[tileID.key].hasData())\n idealRasterTileIDs[tileID.key] = tileID;\n else\n missingTileIDs[tileID.key] = tileID;\n }\n // search for a complete set of children for each missing tile\n for (const key in missingTileIDs) {\n const children = missingTileIDs[key].children(this._source.maxzoom);\n if (this._tiles[children[0].key] && this._tiles[children[1].key] && this._tiles[children[2].key] && this._tiles[children[3].key]) {\n idealRasterTileIDs[children[0].key] = retain[children[0].key] = children[0];\n idealRasterTileIDs[children[1].key] = retain[children[1].key] = children[1];\n idealRasterTileIDs[children[2].key] = retain[children[2].key] = children[2];\n idealRasterTileIDs[children[3].key] = retain[children[3].key] = children[3];\n delete missingTileIDs[key];\n }\n }\n // search for parent for each missing tile\n for (const key in missingTileIDs) {\n const parent = this.findLoadedParent(missingTileIDs[key], this._source.minzoom);\n if (parent) {\n idealRasterTileIDs[parent.tileID.key] = retain[parent.tileID.key] = parent.tileID;\n // remove idealTiles which would be rendered twice\n for (const key in idealRasterTileIDs) {\n if (idealRasterTileIDs[key].isChildOf(parent.tileID)) delete idealRasterTileIDs[key];\n }\n }\n }\n // cover all tiles which are not needed\n for (const key in this._tiles) {\n if (!idealRasterTileIDs[key]) this._coveredTiles[key] = true;\n }\n }\n }\n\n for (const retainedId in retain) {\n // Make sure retained tiles always clear any existing fade holds\n // so that if they're removed again their fade timer starts fresh.\n this._tiles[retainedId].clearFadeHold();\n }\n\n // Remove the tiles we don't need anymore.\n const remove = keysDifference(this._tiles, retain);\n for (const tileID of remove) {\n const tile = this._tiles[tileID];\n if (tile.hasSymbolBuckets && !tile.holdingForFade()) {\n tile.setHoldDuration(this.map._fadeDuration);\n } else if (!tile.hasSymbolBuckets || tile.symbolFadeFinished()) {\n this._removeTile(tileID);\n }\n }\n\n // Construct a cache of loaded parents\n this._updateLoadedParentTileCache();\n }\n\n releaseSymbolFadeTiles() {\n for (const id in this._tiles) {\n if (this._tiles[id].holdingForFade()) {\n this._removeTile(id);\n }\n }\n }\n\n _updateRetainedTiles(idealTileIDs: Array, zoom: number): {[_: string]: OverscaledTileID} {\n const retain: {[_: string]: OverscaledTileID} = {};\n const checked: {[_: string]: boolean} = {};\n const minCoveringZoom = Math.max(zoom - SourceCache.maxOverzooming, this._source.minzoom);\n const maxCoveringZoom = Math.max(zoom + SourceCache.maxUnderzooming, this._source.minzoom);\n\n const missingTiles = {};\n for (const tileID of idealTileIDs) {\n const tile = this._addTile(tileID);\n\n // retain the tile even if it's not loaded because it's an ideal tile.\n retain[tileID.key] = tileID;\n\n if (tile.hasData()) continue;\n\n if (zoom < this._source.maxzoom) {\n // save missing tiles that potentially have loaded children\n missingTiles[tileID.key] = tileID;\n }\n }\n\n // retain any loaded children of ideal tiles up to maxCoveringZoom\n this._retainLoadedChildren(missingTiles, zoom, maxCoveringZoom, retain);\n\n for (const tileID of idealTileIDs) {\n let tile = this._tiles[tileID.key];\n\n if (tile.hasData()) continue;\n\n // The tile we require is not yet loaded or does not exist;\n // Attempt to find children that fully cover it.\n\n if (zoom + 1 > this._source.maxzoom) {\n // We're looking for an overzoomed child tile.\n const childCoord = tileID.children(this._source.maxzoom)[0];\n const childTile = this.getTile(childCoord);\n if (!!childTile && childTile.hasData()) {\n retain[childCoord.key] = childCoord;\n continue; // tile is covered by overzoomed child\n }\n } else {\n // check if all 4 immediate children are loaded (i.e. the missing ideal tile is covered)\n const children = tileID.children(this._source.maxzoom);\n\n if (retain[children[0].key] &&\n retain[children[1].key] &&\n retain[children[2].key] &&\n retain[children[3].key]) continue; // tile is covered by children\n }\n\n // We couldn't find child tiles that entirely cover the ideal tile; look for parents now.\n\n // As we ascend up the tile pyramid of the ideal tile, we check whether the parent\n // tile has been previously requested (and errored because we only loop over tiles with no data)\n // in order to determine if we need to request its parent.\n let parentWasRequested = tile.wasRequested();\n\n for (let overscaledZ = tileID.overscaledZ - 1; overscaledZ >= minCoveringZoom; --overscaledZ) {\n const parentId = tileID.scaledTo(overscaledZ);\n\n // Break parent tile ascent if this route has been previously checked by another child.\n if (checked[parentId.key]) break;\n checked[parentId.key] = true;\n\n tile = this.getTile(parentId);\n if (!tile && parentWasRequested) {\n tile = this._addTile(parentId);\n }\n if (tile) {\n const hasData = tile.hasData();\n if (parentWasRequested || hasData) {\n retain[parentId.key] = parentId;\n }\n // Save the current values, since they're the parent of the next iteration\n // of the parent tile ascent loop.\n parentWasRequested = tile.wasRequested();\n if (hasData) break;\n }\n }\n }\n\n return retain;\n }\n\n _updateLoadedParentTileCache() {\n this._loadedParentTiles = {};\n\n for (const tileKey in this._tiles) {\n const path = [];\n let parentTile: Tile;\n let currentId = this._tiles[tileKey].tileID;\n\n // Find the closest loaded ancestor by traversing the tile tree towards the root and\n // caching results along the way\n while (currentId.overscaledZ > 0) {\n\n // Do we have a cached result from previous traversals?\n if (currentId.key in this._loadedParentTiles) {\n parentTile = this._loadedParentTiles[currentId.key];\n break;\n }\n\n path.push(currentId.key);\n\n // Is the parent loaded?\n const parentId = currentId.scaledTo(currentId.overscaledZ - 1);\n parentTile = this._getLoadedTile(parentId);\n if (parentTile) {\n break;\n }\n\n currentId = parentId;\n }\n\n // Cache the result of this traversal to all newly visited tiles\n for (const key of path) {\n this._loadedParentTiles[key] = parentTile;\n }\n }\n }\n\n /**\n * Add a tile, given its coordinate, to the pyramid.\n */\n _addTile(tileID: OverscaledTileID): Tile {\n let tile = this._tiles[tileID.key];\n if (tile)\n return tile;\n\n tile = this._cache.getAndRemove(tileID);\n if (tile) {\n this._setTileReloadTimer(tileID.key, tile);\n // set the tileID because the cached tile could have had a different wrap value\n tile.tileID = tileID;\n this._state.initializeTileState(tile, this.map ? this.map.painter : null);\n if (this._cacheTimers[tileID.key]) {\n clearTimeout(this._cacheTimers[tileID.key]);\n delete this._cacheTimers[tileID.key];\n this._setTileReloadTimer(tileID.key, tile);\n }\n }\n\n const cached = tile;\n\n if (!tile) {\n tile = new Tile(tileID, this._source.tileSize * tileID.overscaleFactor());\n this._loadTile(tile, this._tileLoaded.bind(this, tile, tileID.key, tile.state));\n }\n\n tile.uses++;\n this._tiles[tileID.key] = tile;\n if (!cached) {\n this._source.fire(new Event('dataloading', {tile, coord: tile.tileID, dataType: 'source'}));\n }\n\n return tile;\n }\n\n _setTileReloadTimer(id: string, tile: Tile) {\n if (id in this._timers) {\n clearTimeout(this._timers[id]);\n delete this._timers[id];\n }\n\n const expiryTimeout = tile.getExpiryTimeout();\n if (expiryTimeout) {\n this._timers[id] = setTimeout(() => {\n this._reloadTile(id, 'expired');\n delete this._timers[id];\n }, expiryTimeout);\n }\n }\n\n /**\n * Remove a tile, given its id, from the pyramid\n */\n _removeTile(id: string) {\n const tile = this._tiles[id];\n if (!tile)\n return;\n\n tile.uses--;\n delete this._tiles[id];\n if (this._timers[id]) {\n clearTimeout(this._timers[id]);\n delete this._timers[id];\n }\n\n if (tile.uses > 0)\n return;\n\n if (tile.hasData() && tile.state !== 'reloading') {\n this._cache.add(tile.tileID, tile, tile.getExpiryTimeout());\n } else {\n tile.aborted = true;\n this._abortTile(tile);\n this._unloadTile(tile);\n }\n }\n\n /**\n * Remove all tiles from this pyramid\n */\n clearTiles() {\n this._shouldReloadOnResume = false;\n this._paused = false;\n\n for (const id in this._tiles)\n this._removeTile(id);\n\n this._cache.reset();\n }\n\n /**\n * Search through our current tiles and attempt to find the tiles that\n * cover the given bounds.\n * @param pointQueryGeometry - coordinates of the corners of bounding rectangle\n * @returns result items have `{tile, minX, maxX, minY, maxY}`, where min/max bounding values are the given bounds transformed in into the coordinate space of this tile.\n */\n tilesIn(pointQueryGeometry: Array, maxPitchScaleFactor: number, has3DLayer: boolean): any[] {\n\n const tileResults = [];\n\n const transform = this.transform;\n if (!transform) return tileResults;\n\n const cameraPointQueryGeometry = has3DLayer ?\n transform.getCameraQueryGeometry(pointQueryGeometry) :\n pointQueryGeometry;\n\n const queryGeometry = pointQueryGeometry.map((p: Point) => transform.pointCoordinate(p, this.terrain));\n const cameraQueryGeometry = cameraPointQueryGeometry.map((p: Point) => transform.pointCoordinate(p, this.terrain));\n\n const ids = this.getIds();\n\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n\n for (const p of cameraQueryGeometry) {\n minX = Math.min(minX, p.x);\n minY = Math.min(minY, p.y);\n maxX = Math.max(maxX, p.x);\n maxY = Math.max(maxY, p.y);\n }\n\n for (let i = 0; i < ids.length; i++) {\n const tile = this._tiles[ids[i]];\n if (tile.holdingForFade()) {\n // Tiles held for fading are covered by tiles that are closer to ideal\n continue;\n }\n const tileID = tile.tileID;\n const scale = Math.pow(2, transform.zoom - tile.tileID.overscaledZ);\n const queryPadding = maxPitchScaleFactor * tile.queryPadding * EXTENT / tile.tileSize / scale;\n\n const tileSpaceBounds = [\n tileID.getTilePoint(new MercatorCoordinate(minX, minY)),\n tileID.getTilePoint(new MercatorCoordinate(maxX, maxY))\n ];\n\n if (tileSpaceBounds[0].x - queryPadding < EXTENT && tileSpaceBounds[0].y - queryPadding < EXTENT &&\n tileSpaceBounds[1].x + queryPadding >= 0 && tileSpaceBounds[1].y + queryPadding >= 0) {\n\n const tileSpaceQueryGeometry: Array = queryGeometry.map((c) => tileID.getTilePoint(c));\n const tileSpaceCameraQueryGeometry = cameraQueryGeometry.map((c) => tileID.getTilePoint(c));\n\n tileResults.push({\n tile,\n tileID,\n queryGeometry: tileSpaceQueryGeometry,\n cameraQueryGeometry: tileSpaceCameraQueryGeometry,\n scale\n });\n }\n }\n\n return tileResults;\n }\n\n getVisibleCoordinates(symbolLayer?: boolean): Array {\n const coords = this.getRenderableIds(symbolLayer).map((id) => this._tiles[id].tileID);\n for (const coord of coords) {\n coord.posMatrix = this.transform.calculatePosMatrix(coord.toUnwrapped());\n }\n return coords;\n }\n\n hasTransition() {\n if (this._source.hasTransition()) {\n return true;\n }\n\n if (isRasterType(this._source.type)) {\n const now = browser.now();\n for (const id in this._tiles) {\n const tile = this._tiles[id];\n if (tile.fadeEndTime >= now) {\n return true;\n }\n }\n }\n\n return false;\n }\n\n /**\n * Set the value of a particular state for a feature\n */\n setFeatureState(sourceLayer: string, featureId: number | string, state: any) {\n sourceLayer = sourceLayer || '_geojsonTileLayer';\n this._state.updateState(sourceLayer, featureId, state);\n }\n\n /**\n * Resets the value of a particular state key for a feature\n */\n removeFeatureState(sourceLayer?: string, featureId?: number | string, key?: string) {\n sourceLayer = sourceLayer || '_geojsonTileLayer';\n this._state.removeFeatureState(sourceLayer, featureId, key);\n }\n\n /**\n * Get the entire state object for a feature\n */\n getFeatureState(sourceLayer: string, featureId: number | string) {\n sourceLayer = sourceLayer || '_geojsonTileLayer';\n return this._state.getState(sourceLayer, featureId);\n }\n\n /**\n * Sets the set of keys that the tile depends on. This allows tiles to\n * be reloaded when their dependencies change.\n */\n setDependencies(tileKey: string, namespace: string, dependencies: Array) {\n const tile = this._tiles[tileKey];\n if (tile) {\n tile.setDependencies(namespace, dependencies);\n }\n }\n\n /**\n * Reloads all tiles that depend on the given keys.\n */\n reloadTilesForDependencies(namespaces: Array, keys: Array) {\n for (const id in this._tiles) {\n const tile = this._tiles[id];\n if (tile.hasDependency(namespaces, keys)) {\n this._reloadTile(id, 'reloading');\n }\n }\n this._cache.filter(tile => !tile.hasDependency(namespaces, keys));\n }\n}\n\nSourceCache.maxOverzooming = 10;\nSourceCache.maxUnderzooming = 3;\n\nfunction compareTileId(a: OverscaledTileID, b: OverscaledTileID): number {\n // Different copies of the world are sorted based on their distance to the center.\n // Wrap values are converted to unsigned distances by reserving odd number for copies\n // with negative wrap and even numbers for copies with positive wrap.\n const aWrap = Math.abs(a.wrap * 2) - +(a.wrap < 0);\n const bWrap = Math.abs(b.wrap * 2) - +(b.wrap < 0);\n return a.overscaledZ - b.overscaledZ || bWrap - aWrap || b.canonical.y - a.canonical.y || b.canonical.x - a.canonical.x;\n}\n\nfunction isRasterType(type) {\n return type === 'raster' || type === 'image' || type === 'video';\n}\n","import {workerFactory} from './web_worker';\nimport {browser} from './browser';\nimport {isSafari} from './util';\nimport {ActorTarget} from './actor';\n\nexport const PRELOAD_POOL_ID = 'mapboxgl_preloaded_worker_pool';\n\n/**\n * Constructs a worker pool.\n */\nexport class WorkerPool {\n static workerCount: number;\n\n active: {\n [_ in number | string]: boolean;\n };\n workers: Array;\n\n constructor() {\n this.active = {};\n }\n\n acquire(mapId: number | string): Array {\n if (!this.workers) {\n // Lazily look up the value of mapboxgl.workerCount so that\n // client code has had a chance to set it.\n this.workers = [];\n while (this.workers.length < WorkerPool.workerCount) {\n this.workers.push(workerFactory());\n }\n }\n\n this.active[mapId] = true;\n return this.workers.slice();\n }\n\n release(mapId: number | string) {\n delete this.active[mapId];\n if (this.numActive() === 0) {\n this.workers.forEach((w) => {\n w.terminate();\n });\n this.workers = null;\n }\n }\n\n isPreloaded(): boolean {\n return !!this.active[PRELOAD_POOL_ID];\n }\n\n numActive(): number {\n return Object.keys(this.active).length;\n }\n}\n\n// Based on results from A/B testing: https://github.com/maplibre/maplibre-gl-js/pull/2354\nconst availableLogicalProcessors = Math.floor(browser.hardwareConcurrency / 2);\nWorkerPool.workerCount = isSafari(globalThis) ? Math.max(Math.min(availableLogicalProcessors, 3), 1) : 1;\n","import {config} from './config';\n\nimport type {WorkerSource} from '../source/worker_source';\n\nexport interface WorkerGlobalScopeInterface {\n importScripts(...urls: Array): void;\n registerWorkerSource: (\n b: string,\n a: {\n new(...args: any): WorkerSource;\n }\n ) => void;\n registerRTLTextPlugin: (_: any) => void;\n}\n\nexport function workerFactory() {\n return new Worker(config.WORKER_URL);\n}\n","import {WorkerPool, PRELOAD_POOL_ID} from './worker_pool';\n\nlet globalWorkerPool;\n\n/**\n * Creates (if necessary) and returns the single, global WorkerPool instance\n * to be shared across each Map\n */\nexport function getGlobalWorkerPool() {\n if (!globalWorkerPool) {\n globalWorkerPool = new WorkerPool();\n }\n return globalWorkerPool;\n}\n\nexport function prewarm() {\n const workerPool = getGlobalWorkerPool();\n workerPool.acquire(PRELOAD_POOL_ID);\n}\n\nexport function clearPrewarmedResources() {\n const pool = globalWorkerPool;\n if (pool) {\n // Remove the pool only if all maps that referenced the preloaded global worker pool have been removed.\n if (pool.isPreloaded() && pool.numActive() === 1) {\n pool.release(PRELOAD_POOL_ID);\n globalWorkerPool = null;\n } else {\n console.warn('Could not clear WebWorkers since there are active Map instances that still reference it. The pre-warmed WebWorker pool can only be cleared when all map instances have been removed with map.remove()');\n }\n }\n}\n","import {clamp} from '../util/util';\nimport Point from '@mapbox/point-geometry';\n\nexport class PathInterpolator {\n points: Array;\n length: number;\n paddedLength: number;\n padding: number;\n _distances: Array;\n\n constructor(points_?: Array | null, padding_?: number | null) {\n this.reset(points_, padding_);\n }\n\n reset(points_?: Array | null, padding_?: number | null) {\n this.points = points_ || [];\n\n // Compute cumulative distance from first point to every other point in the segment.\n // Last entry in the array is total length of the path\n this._distances = [0.0];\n\n for (let i = 1; i < this.points.length; i++) {\n this._distances[i] = this._distances[i - 1] + this.points[i].dist(this.points[i - 1]);\n }\n\n this.length = this._distances[this._distances.length - 1];\n this.padding = Math.min(padding_ || 0, this.length * 0.5);\n this.paddedLength = this.length - this.padding * 2.0;\n }\n\n lerp(t: number): Point {\n if (this.points.length === 1) {\n return this.points[0];\n }\n\n t = clamp(t, 0, 1);\n\n // Find the correct segment [p0, p1] where p0 <= x < p1\n let currentIndex = 1;\n let distOfCurrentIdx = this._distances[currentIndex];\n const distToTarget = t * this.paddedLength + this.padding;\n\n while (distOfCurrentIdx < distToTarget && currentIndex < this._distances.length) {\n distOfCurrentIdx = this._distances[++currentIndex];\n }\n\n // Interpolate between the two points of the segment\n const idxOfPrevPoint = currentIndex - 1;\n const distOfPrevIdx = this._distances[idxOfPrevPoint];\n const segmentLength = distOfCurrentIdx - distOfPrevIdx;\n const segmentT = segmentLength > 0 ? (distToTarget - distOfPrevIdx) / segmentLength : 0;\n\n return this.points[idxOfPrevPoint].mult(1.0 - segmentT).add(this.points[currentIndex].mult(segmentT));\n }\n}\n","import type {OverlapMode} from '../style/style_layer/overlap_mode';\n\ntype QueryArgs = {\n hitTest: boolean;\n overlapMode?: OverlapMode;\n circle?: {\n x: number;\n y: number;\n radius: number;\n };\n seenUids: {\n box: {\n [_: number]: boolean;\n };\n circle: {\n [_: number]: boolean;\n };\n };\n};\n\ntype QueryResult = {\n key: T;\n x1: number;\n y1: number;\n x2: number;\n y2: number;\n};\n\n/**\n * A key for the grid\n */\nexport type GridKey = {\n overlapMode?: OverlapMode;\n}\n\nfunction overlapAllowed(overlapA: OverlapMode, overlapB: OverlapMode): boolean {\n let allowed = true;\n\n if (overlapA === 'always') {\n // symbol A using 'always' overlap - allowed to overlap anything.\n } else if (overlapA === 'never' || overlapB === 'never') {\n // symbol A using 'never' overlap - can't overlap anything\n // symbol A using 'cooperative' overlap - can overlap 'always' or 'cooperative' symbol; can't overlap 'never'\n allowed = false;\n }\n\n return allowed;\n}\n\n/**\n * @internal\n * GridIndex is a data structure for testing the intersection of\n * circles and rectangles in a 2d plane.\n * It is optimized for rapid insertion and querying.\n * GridIndex splits the plane into a set of \"cells\" and keeps track\n * of which geometries intersect with each cell. At query time,\n * full geometry comparisons are only done for items that share\n * at least one cell. As long as the geometries are relatively\n * uniformly distributed across the plane, this greatly reduces\n * the number of comparisons necessary.\n */\nexport class GridIndex {\n circleKeys: Array;\n boxKeys: Array;\n boxCells: Array>;\n circleCells: Array>;\n bboxes: Array;\n circles: Array;\n xCellCount: number;\n yCellCount: number;\n width: number;\n height: number;\n xScale: number;\n yScale: number;\n boxUid: number;\n circleUid: number;\n\n constructor (width: number, height: number, cellSize: number) {\n const boxCells = this.boxCells = [];\n const circleCells = this.circleCells = [];\n\n // More cells -> fewer geometries to check per cell, but items tend\n // to be split across more cells.\n // Sweet spot allows most small items to fit in one cell\n this.xCellCount = Math.ceil(width / cellSize);\n this.yCellCount = Math.ceil(height / cellSize);\n\n for (let i = 0; i < this.xCellCount * this.yCellCount; i++) {\n boxCells.push([]);\n circleCells.push([]);\n }\n this.circleKeys = [];\n this.boxKeys = [];\n this.bboxes = [];\n this.circles = [];\n\n this.width = width;\n this.height = height;\n this.xScale = this.xCellCount / width;\n this.yScale = this.yCellCount / height;\n this.boxUid = 0;\n this.circleUid = 0;\n }\n\n keysLength() {\n return this.boxKeys.length + this.circleKeys.length;\n }\n\n insert(key: T, x1: number, y1: number, x2: number, y2: number) {\n this._forEachCell(x1, y1, x2, y2, this._insertBoxCell, this.boxUid++);\n this.boxKeys.push(key);\n this.bboxes.push(x1);\n this.bboxes.push(y1);\n this.bboxes.push(x2);\n this.bboxes.push(y2);\n }\n\n insertCircle(key: T, x: number, y: number, radius: number) {\n // Insert circle into grid for all cells in the circumscribing square\n // It's more than necessary (by a factor of 4/PI), but fast to insert\n this._forEachCell(x - radius, y - radius, x + radius, y + radius, this._insertCircleCell, this.circleUid++);\n this.circleKeys.push(key);\n this.circles.push(x);\n this.circles.push(y);\n this.circles.push(radius);\n }\n\n private _insertBoxCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) {\n this.boxCells[cellIndex].push(uid);\n }\n\n private _insertCircleCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) {\n this.circleCells[cellIndex].push(uid);\n }\n\n private _query(x1: number, y1: number, x2: number, y2: number, hitTest: boolean, overlapMode: OverlapMode, predicate?: (key: T) => boolean): Array> {\n if (x2 < 0 || x1 > this.width || y2 < 0 || y1 > this.height) {\n return [];\n }\n const result: Array> = [];\n if (x1 <= 0 && y1 <= 0 && this.width <= x2 && this.height <= y2) {\n if (hitTest) {\n // Covers the entire grid, so collides with everything\n return [{\n key: null,\n x1,\n y1,\n x2,\n y2\n }];\n }\n for (let boxUid = 0; boxUid < this.boxKeys.length; boxUid++) {\n result.push({\n key: this.boxKeys[boxUid],\n x1: this.bboxes[boxUid * 4],\n y1: this.bboxes[boxUid * 4 + 1],\n x2: this.bboxes[boxUid * 4 + 2],\n y2: this.bboxes[boxUid * 4 + 3]\n });\n }\n for (let circleUid = 0; circleUid < this.circleKeys.length; circleUid++) {\n const x = this.circles[circleUid * 3];\n const y = this.circles[circleUid * 3 + 1];\n const radius = this.circles[circleUid * 3 + 2];\n result.push({\n key: this.circleKeys[circleUid],\n x1: x - radius,\n y1: y - radius,\n x2: x + radius,\n y2: y + radius\n });\n }\n } else {\n const queryArgs: QueryArgs = {\n hitTest,\n overlapMode,\n seenUids: {box: {}, circle: {}}\n };\n this._forEachCell(x1, y1, x2, y2, this._queryCell, result, queryArgs, predicate);\n }\n\n return result;\n }\n\n query(x1: number, y1: number, x2: number, y2: number): Array> {\n return this._query(x1, y1, x2, y2, false, null);\n }\n\n hitTest(x1: number, y1: number, x2: number, y2: number, overlapMode: OverlapMode, predicate?: (key: T) => boolean): boolean {\n return this._query(x1, y1, x2, y2, true, overlapMode, predicate).length > 0;\n }\n\n hitTestCircle(x: number, y: number, radius: number, overlapMode: OverlapMode, predicate?: (key: T) => boolean): boolean {\n // Insert circle into grid for all cells in the circumscribing square\n // It's more than necessary (by a factor of 4/PI), but fast to insert\n const x1 = x - radius;\n const x2 = x + radius;\n const y1 = y - radius;\n const y2 = y + radius;\n if (x2 < 0 || x1 > this.width || y2 < 0 || y1 > this.height) {\n return false;\n }\n\n // Box query early exits if the bounding box is larger than the grid, but we don't do\n // the equivalent calculation for circle queries because early exit is less likely\n // and the calculation is more expensive\n const result: boolean[] = [];\n const queryArgs: QueryArgs = {\n hitTest: true,\n overlapMode,\n circle: {x, y, radius},\n seenUids: {box: {}, circle: {}}\n };\n this._forEachCell(x1, y1, x2, y2, this._queryCellCircle, result, queryArgs, predicate);\n return result.length > 0;\n }\n\n private _queryCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, result: Array>, queryArgs: QueryArgs, predicate?: (key: T) => boolean): boolean {\n const {seenUids, hitTest, overlapMode} = queryArgs;\n const boxCell = this.boxCells[cellIndex];\n\n if (boxCell !== null) {\n const bboxes = this.bboxes;\n for (const boxUid of boxCell) {\n if (!seenUids.box[boxUid]) {\n seenUids.box[boxUid] = true;\n const offset = boxUid * 4;\n const key = this.boxKeys[boxUid];\n\n if ((x1 <= bboxes[offset + 2]) &&\n (y1 <= bboxes[offset + 3]) &&\n (x2 >= bboxes[offset + 0]) &&\n (y2 >= bboxes[offset + 1]) &&\n (!predicate || predicate(key))) {\n if (!hitTest || !overlapAllowed(overlapMode, key.overlapMode)) {\n result.push({\n key,\n x1: bboxes[offset],\n y1: bboxes[offset + 1],\n x2: bboxes[offset + 2],\n y2: bboxes[offset + 3]\n });\n if (hitTest) {\n // true return value stops the query after first match\n return true;\n }\n }\n }\n }\n }\n }\n const circleCell = this.circleCells[cellIndex];\n if (circleCell !== null) {\n const circles = this.circles;\n for (const circleUid of circleCell) {\n if (!seenUids.circle[circleUid]) {\n seenUids.circle[circleUid] = true;\n const offset = circleUid * 3;\n const key = this.circleKeys[circleUid];\n\n if (this._circleAndRectCollide(\n circles[offset],\n circles[offset + 1],\n circles[offset + 2],\n x1,\n y1,\n x2,\n y2) &&\n (!predicate || predicate(key))) {\n if (!hitTest || !overlapAllowed(overlapMode, key.overlapMode)) {\n const x = circles[offset];\n const y = circles[offset + 1];\n const radius = circles[offset + 2];\n result.push({\n key,\n x1: x - radius,\n y1: y - radius,\n x2: x + radius,\n y2: y + radius\n });\n if (hitTest) {\n // true return value stops the query after first match\n return true;\n }\n }\n }\n }\n }\n }\n\n // false return to continue query\n return false;\n }\n\n private _queryCellCircle(x1: number, y1: number, x2: number, y2: number, cellIndex: number, result: Array, queryArgs: QueryArgs, predicate?: (key: T) => boolean): boolean {\n const {circle, seenUids, overlapMode} = queryArgs;\n const boxCell = this.boxCells[cellIndex];\n\n if (boxCell !== null) {\n const bboxes = this.bboxes;\n for (const boxUid of boxCell) {\n if (!seenUids.box[boxUid]) {\n seenUids.box[boxUid] = true;\n const offset = boxUid * 4;\n const key = this.boxKeys[boxUid];\n if (this._circleAndRectCollide(\n circle.x,\n circle.y,\n circle.radius,\n bboxes[offset + 0],\n bboxes[offset + 1],\n bboxes[offset + 2],\n bboxes[offset + 3]) &&\n (!predicate || predicate(key)) &&\n !overlapAllowed(overlapMode, key.overlapMode)) {\n result.push(true);\n return true;\n }\n }\n }\n }\n\n const circleCell = this.circleCells[cellIndex];\n if (circleCell !== null) {\n const circles = this.circles;\n for (const circleUid of circleCell) {\n if (!seenUids.circle[circleUid]) {\n seenUids.circle[circleUid] = true;\n const offset = circleUid * 3;\n const key = this.circleKeys[circleUid];\n if (this._circlesCollide(\n circles[offset],\n circles[offset + 1],\n circles[offset + 2],\n circle.x,\n circle.y,\n circle.radius) &&\n (!predicate || predicate(key)) &&\n !overlapAllowed(overlapMode, key.overlapMode)) {\n result.push(true);\n return true;\n }\n }\n }\n }\n }\n\n private _forEachCell(\n x1: number,\n y1: number,\n x2: number,\n y2: number,\n fn: (x1: number, y1: number, x2: number, y2: number, cellIndex: number, arg1: TArg, arg2?: QueryArgs, predicate?: (key: T) => boolean) => boolean | void,\n arg1: TArg,\n arg2?: QueryArgs,\n predicate?: (key: T) => boolean) {\n const cx1 = this._convertToXCellCoord(x1);\n const cy1 = this._convertToYCellCoord(y1);\n const cx2 = this._convertToXCellCoord(x2);\n const cy2 = this._convertToYCellCoord(y2);\n\n for (let x = cx1; x <= cx2; x++) {\n for (let y = cy1; y <= cy2; y++) {\n const cellIndex = this.xCellCount * y + x;\n if (fn.call(this, x1, y1, x2, y2, cellIndex, arg1, arg2, predicate)) return;\n }\n }\n }\n\n private _convertToXCellCoord(x: number) {\n return Math.max(0, Math.min(this.xCellCount - 1, Math.floor(x * this.xScale)));\n }\n\n private _convertToYCellCoord(y: number) {\n return Math.max(0, Math.min(this.yCellCount - 1, Math.floor(y * this.yScale)));\n }\n\n private _circlesCollide(x1: number, y1: number, r1: number, x2: number, y2: number, r2: number): boolean {\n const dx = x2 - x1;\n const dy = y2 - y1;\n const bothRadii = r1 + r2;\n return (bothRadii * bothRadii) > (dx * dx + dy * dy);\n }\n\n private _circleAndRectCollide(\n circleX: number,\n circleY: number,\n radius: number,\n x1: number,\n y1: number,\n x2: number,\n y2: number\n ): boolean {\n const halfRectWidth = (x2 - x1) / 2;\n const distX = Math.abs(circleX - (x1 + halfRectWidth));\n if (distX > (halfRectWidth + radius)) {\n return false;\n }\n\n const halfRectHeight = (y2 - y1) / 2;\n const distY = Math.abs(circleY - (y1 + halfRectHeight));\n if (distY > (halfRectHeight + radius)) {\n return false;\n }\n\n if (distX <= halfRectWidth || distY <= halfRectHeight) {\n return true;\n }\n\n const dx = distX - halfRectWidth;\n const dy = distY - halfRectHeight;\n return (dx * dx + dy * dy <= (radius * radius));\n }\n}\n","import Point from '@mapbox/point-geometry';\n\nimport {mat4, vec4} from 'gl-matrix';\nimport * as symbolSize from './symbol_size';\nimport {addDynamicAttributes} from '../data/bucket/symbol_bucket';\n\nimport type {Painter} from '../render/painter';\nimport type {Transform} from '../geo/transform';\nimport type {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport type {\n GlyphOffsetArray,\n SymbolLineVertexArray,\n SymbolDynamicLayoutArray\n} from '../data/array_types.g';\nimport {WritingMode} from '../symbol/shaping';\nimport {findLineIntersection} from '../util/util';\n\nexport {updateLineLabels, hideGlyphs, getLabelPlaneMatrix, getGlCoordMatrix, project, getPerspectiveRatio, placeFirstAndLastGlyph, placeGlyphAlongLine, xyTransformMat4, projectVertexToViewport, findOffsetIntersectionPoint, transformToOffsetNormal};\n\n/*\n * # Overview of coordinate spaces\n *\n * ## Tile coordinate spaces\n * Each label has an anchor. Some labels have corresponding line geometries.\n * The points for both anchors and lines are stored in tile units. Each tile has it's own\n * coordinate space going from (0, 0) at the top left to (EXTENT, EXTENT) at the bottom right.\n *\n * ## GL coordinate space\n * At the end of everything, the vertex shader needs to produce a position in GL coordinate space,\n * which is (-1, 1) at the top left and (1, -1) in the bottom right.\n *\n * ## Map pixel coordinate spaces\n * Each tile has a pixel coordinate space. It's just the tile units scaled so that one unit is\n * whatever counts as 1 pixel at the current zoom.\n * This space is used for pitch-alignment=map, rotation-alignment=map\n *\n * ## Rotated map pixel coordinate spaces\n * Like the above, but rotated so axis of the space are aligned with the viewport instead of the tile.\n * This space is used for pitch-alignment=map, rotation-alignment=viewport\n *\n * ## Viewport pixel coordinate space\n * (0, 0) is at the top left of the canvas and (pixelWidth, pixelHeight) is at the bottom right corner\n * of the canvas. This space is used for pitch-alignment=viewport\n *\n *\n * # Vertex projection\n * It goes roughly like this:\n * 1. project the anchor and line from tile units into the correct label coordinate space\n * - map pixel space pitch-alignment=map rotation-alignment=map\n * - rotated map pixel space pitch-alignment=map rotation-alignment=viewport\n * - viewport pixel space pitch-alignment=viewport rotation-alignment=*\n * 2. if the label follows a line, find the point along the line that is the correct distance from the anchor.\n * 3. add the glyph's corner offset to the point from step 3\n * 4. convert from the label coordinate space to gl coordinates\n *\n * For horizontal labels we want to do step 1 in the shader for performance reasons (no cpu work).\n * This is what `u_label_plane_matrix` is used for.\n * For labels aligned with lines we have to steps 1 and 2 on the cpu since we need access to the line geometry.\n * This is what `updateLineLabels(...)` does.\n * Since the conversion is handled on the cpu we just set `u_label_plane_matrix` to an identity matrix.\n *\n * Steps 3 and 4 are done in the shaders for all labels.\n */\n\n/*\n * Returns a matrix for converting from tile units to the correct label coordinate space.\n */\nfunction getLabelPlaneMatrix(posMatrix: mat4,\n pitchWithMap: boolean,\n rotateWithMap: boolean,\n transform: Transform,\n pixelsToTileUnits: number) {\n const m = mat4.create();\n if (pitchWithMap) {\n mat4.scale(m, m, [1 / pixelsToTileUnits, 1 / pixelsToTileUnits, 1]);\n if (!rotateWithMap) {\n mat4.rotateZ(m, m, transform.angle);\n }\n } else {\n mat4.multiply(m, transform.labelPlaneMatrix, posMatrix);\n }\n return m;\n}\n\n/*\n * Returns a matrix for converting from the correct label coordinate space to gl coords.\n */\nfunction getGlCoordMatrix(posMatrix: mat4,\n pitchWithMap: boolean,\n rotateWithMap: boolean,\n transform: Transform,\n pixelsToTileUnits: number) {\n if (pitchWithMap) {\n const m = mat4.clone(posMatrix);\n mat4.scale(m, m, [pixelsToTileUnits, pixelsToTileUnits, 1]);\n if (!rotateWithMap) {\n mat4.rotateZ(m, m, -transform.angle);\n }\n return m;\n } else {\n return transform.glCoordMatrix;\n }\n}\n\nfunction project(point: Point, matrix: mat4, getElevation?: (x: number, y: number) => number) {\n let pos;\n if (getElevation) { // slow because of handle z-index\n pos = [point.x, point.y, getElevation(point.x, point.y), 1] as vec4;\n vec4.transformMat4(pos, pos, matrix);\n } else { // fast because of ignore z-index\n pos = [point.x, point.y, 0, 1] as vec4;\n xyTransformMat4(pos, pos, matrix);\n }\n const w = pos[3];\n return {\n point: new Point(pos[0] / w, pos[1] / w),\n signedDistanceFromCamera: w\n };\n}\n\nfunction getPerspectiveRatio(cameraToCenterDistance: number, signedDistanceFromCamera: number): number {\n return 0.5 + 0.5 * (cameraToCenterDistance / signedDistanceFromCamera);\n}\n\nfunction isVisible(anchorPos: vec4,\n clippingBuffer: [number, number]) {\n const x = anchorPos[0] / anchorPos[3];\n const y = anchorPos[1] / anchorPos[3];\n const inPaddedViewport = (\n x >= -clippingBuffer[0] &&\n x <= clippingBuffer[0] &&\n y >= -clippingBuffer[1] &&\n y <= clippingBuffer[1]);\n return inPaddedViewport;\n}\n\n/*\n * Update the `dynamicLayoutVertexBuffer` for the buffer with the correct glyph positions for the current map view.\n * This is only run on labels that are aligned with lines. Horizontal labels are handled entirely in the shader.\n */\nfunction updateLineLabels(bucket: SymbolBucket,\n posMatrix: mat4,\n painter: Painter,\n isText: boolean,\n labelPlaneMatrix: mat4,\n glCoordMatrix: mat4,\n pitchWithMap: boolean,\n keepUpright: boolean,\n rotateToLine: boolean,\n getElevation: (x: number, y: number) => number) {\n\n const sizeData = isText ? bucket.textSizeData : bucket.iconSizeData;\n const partiallyEvaluatedSize = symbolSize.evaluateSizeForZoom(sizeData, painter.transform.zoom);\n\n const clippingBuffer: [number, number] = [256 / painter.width * 2 + 1, 256 / painter.height * 2 + 1];\n\n const dynamicLayoutVertexArray = isText ?\n bucket.text.dynamicLayoutVertexArray :\n bucket.icon.dynamicLayoutVertexArray;\n dynamicLayoutVertexArray.clear();\n\n const lineVertexArray = bucket.lineVertexArray;\n const placedSymbols = isText ? bucket.text.placedSymbolArray : bucket.icon.placedSymbolArray;\n\n const aspectRatio = painter.transform.width / painter.transform.height;\n\n let useVertical = false;\n\n for (let s = 0; s < placedSymbols.length; s++) {\n const symbol = placedSymbols.get(s);\n\n // Don't do calculations for vertical glyphs unless the previous symbol was horizontal\n // and we determined that vertical glyphs were necessary.\n // Also don't do calculations for symbols that are collided and fully faded out\n if (symbol.hidden || symbol.writingMode === WritingMode.vertical && !useVertical) {\n hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray);\n continue;\n }\n // Awkward... but we're counting on the paired \"vertical\" symbol coming immediately after its horizontal counterpart\n useVertical = false;\n\n let anchorPos;\n if (getElevation) { // slow because of handle z-index\n anchorPos = [symbol.anchorX, symbol.anchorY, getElevation(symbol.anchorX, symbol.anchorY), 1] as vec4;\n vec4.transformMat4(anchorPos, anchorPos, posMatrix);\n } else { // fast because of ignore z-index\n anchorPos = [symbol.anchorX, symbol.anchorY, 0, 1] as vec4;\n xyTransformMat4(anchorPos, anchorPos, posMatrix);\n }\n\n // Don't bother calculating the correct point for invisible labels.\n if (!isVisible(anchorPos, clippingBuffer)) {\n hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray);\n continue;\n }\n\n const cameraToAnchorDistance = anchorPos[3];\n const perspectiveRatio = getPerspectiveRatio(painter.transform.cameraToCenterDistance, cameraToAnchorDistance);\n\n const fontSize = symbolSize.evaluateSizeForFeature(sizeData, partiallyEvaluatedSize, symbol);\n const pitchScaledFontSize = pitchWithMap ? fontSize / perspectiveRatio : fontSize * perspectiveRatio;\n\n const tileAnchorPoint = new Point(symbol.anchorX, symbol.anchorY);\n const anchorPoint = project(tileAnchorPoint, labelPlaneMatrix, getElevation).point;\n const projectionCache = {projections: {}, offsets: {}};\n\n const placeUnflipped: any = placeGlyphsAlongLine(symbol, pitchScaledFontSize, false /*unflipped*/, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix,\n bucket.glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation);\n\n useVertical = placeUnflipped.useVertical;\n\n if (placeUnflipped.notEnoughRoom || useVertical ||\n (placeUnflipped.needsFlipping &&\n (placeGlyphsAlongLine(symbol, pitchScaledFontSize, true /*flipped*/, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix,\n bucket.glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation) as any).notEnoughRoom)) {\n hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray);\n }\n }\n\n if (isText) {\n bucket.text.dynamicLayoutVertexBuffer.updateData(dynamicLayoutVertexArray);\n } else {\n bucket.icon.dynamicLayoutVertexBuffer.updateData(dynamicLayoutVertexArray);\n }\n}\n\ntype FirstAndLastGlyphPlacement = {\n first: PlacedGlyph;\n last: PlacedGlyph;\n} | null;\n\n/*\n * Place the first and last glyph of a line label, projected to the label plane.\n * This function is called both during collision detection (to determine the label's size)\n * and during line label rendering (to make sure the label fits on the line geometry with\n * the current camera position, which may differ from the position used during collision detection).\n *\n * Calling this function has the effect of populating the \"projectionCache\" with all projected\n * vertex locations the label will need, making future calls to placeGlyphAlongLine (for all the\n * intermediate glyphs) much cheaper.\n *\n * Returns null if the label can't fit on the geometry\n */\nfunction placeFirstAndLastGlyph(fontScale: number, glyphOffsetArray: GlyphOffsetArray, lineOffsetX: number, lineOffsetY: number, flip: boolean, anchorPoint: Point, tileAnchorPoint: Point, symbol: any, lineVertexArray: SymbolLineVertexArray, labelPlaneMatrix: mat4, projectionCache: ProjectionCache, rotateToLine: boolean, getElevation: (x: number, y: number) => number): FirstAndLastGlyphPlacement {\n const glyphEndIndex = symbol.glyphStartIndex + symbol.numGlyphs;\n const lineStartIndex = symbol.lineStartIndex;\n const lineEndIndex = symbol.lineStartIndex + symbol.lineLength;\n\n const firstGlyphOffset = glyphOffsetArray.getoffsetX(symbol.glyphStartIndex);\n const lastGlyphOffset = glyphOffsetArray.getoffsetX(glyphEndIndex - 1);\n\n const firstPlacedGlyph = placeGlyphAlongLine(fontScale * firstGlyphOffset, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment,\n lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation);\n if (!firstPlacedGlyph)\n return null;\n\n const lastPlacedGlyph = placeGlyphAlongLine(fontScale * lastGlyphOffset, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment,\n lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation);\n if (!lastPlacedGlyph)\n return null;\n\n return {first: firstPlacedGlyph, last: lastPlacedGlyph};\n}\n\nfunction requiresOrientationChange(writingMode, firstPoint, lastPoint, aspectRatio) {\n if (writingMode === WritingMode.horizontal) {\n // On top of choosing whether to flip, choose whether to render this version of the glyphs or the alternate\n // vertical glyphs. We can't just filter out vertical glyphs in the horizontal range because the horizontal\n // and vertical versions can have slightly different projections which could lead to angles where both or\n // neither showed.\n const rise = Math.abs(lastPoint.y - firstPoint.y);\n const run = Math.abs(lastPoint.x - firstPoint.x) * aspectRatio;\n if (rise > run) {\n return {useVertical: true};\n }\n }\n\n if (writingMode === WritingMode.vertical ? firstPoint.y < lastPoint.y : firstPoint.x > lastPoint.x) {\n // Includes \"horizontalOnly\" case for labels without vertical glyphs\n return {needsFlipping: true};\n }\n\n return null;\n}\n\n/*\n* Place first and last glyph along the line projected to label plane, and if they fit\n* iterate through all the intermediate glyphs, calculating their label plane positions\n* from the projected line.\n*\n* Finally, add resulting glyph position calculations to dynamicLayoutVertexArray for\n* upload to the GPU\n*/\nfunction placeGlyphsAlongLine(symbol, fontSize, flip, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix, glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation) {\n const fontScale = fontSize / 24;\n const lineOffsetX = symbol.lineOffsetX * fontScale;\n const lineOffsetY = symbol.lineOffsetY * fontScale;\n\n let placedGlyphs;\n if (symbol.numGlyphs > 1) {\n const glyphEndIndex = symbol.glyphStartIndex + symbol.numGlyphs;\n const lineStartIndex = symbol.lineStartIndex;\n const lineEndIndex = symbol.lineStartIndex + symbol.lineLength;\n\n // Place the first and the last glyph in the label first, so we can figure out\n // the overall orientation of the label and determine whether it needs to be flipped in keepUpright mode\n const firstAndLastGlyph = placeFirstAndLastGlyph(fontScale, glyphOffsetArray, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation);\n if (!firstAndLastGlyph) {\n return {notEnoughRoom: true};\n }\n const firstPoint = project(firstAndLastGlyph.first.point, glCoordMatrix, getElevation).point;\n const lastPoint = project(firstAndLastGlyph.last.point, glCoordMatrix, getElevation).point;\n\n if (keepUpright && !flip) {\n const orientationChange = requiresOrientationChange(symbol.writingMode, firstPoint, lastPoint, aspectRatio);\n if (orientationChange) {\n return orientationChange;\n }\n }\n\n placedGlyphs = [firstAndLastGlyph.first];\n for (let glyphIndex = symbol.glyphStartIndex + 1; glyphIndex < glyphEndIndex - 1; glyphIndex++) {\n // Since first and last glyph fit on the line, we're sure that the rest of the glyphs can be placed\n placedGlyphs.push(placeGlyphAlongLine(fontScale * glyphOffsetArray.getoffsetX(glyphIndex), lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment,\n lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation));\n }\n placedGlyphs.push(firstAndLastGlyph.last);\n } else {\n // Only a single glyph to place\n // So, determine whether to flip based on projected angle of the line segment it's on\n if (keepUpright && !flip) {\n const a = project(tileAnchorPoint, posMatrix, getElevation).point;\n const tileVertexIndex = (symbol.lineStartIndex + symbol.segment + 1);\n const tileSegmentEnd = new Point(lineVertexArray.getx(tileVertexIndex), lineVertexArray.gety(tileVertexIndex));\n const projectedVertex = project(tileSegmentEnd, posMatrix, getElevation);\n // We know the anchor will be in the viewport, but the end of the line segment may be\n // behind the plane of the camera, in which case we can use a point at any arbitrary (closer)\n // point on the segment.\n const b = (projectedVertex.signedDistanceFromCamera > 0) ?\n projectedVertex.point :\n projectTruncatedLineSegment(tileAnchorPoint, tileSegmentEnd, a, 1, posMatrix, getElevation);\n\n const orientationChange = requiresOrientationChange(symbol.writingMode, a, b, aspectRatio);\n if (orientationChange) {\n return orientationChange;\n }\n }\n const singleGlyph = placeGlyphAlongLine(fontScale * glyphOffsetArray.getoffsetX(symbol.glyphStartIndex), lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment,\n symbol.lineStartIndex, symbol.lineStartIndex + symbol.lineLength, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation);\n if (!singleGlyph)\n return {notEnoughRoom: true};\n\n placedGlyphs = [singleGlyph];\n }\n\n for (const glyph of placedGlyphs) {\n addDynamicAttributes(dynamicLayoutVertexArray, glyph.point, glyph.angle);\n }\n return {};\n}\n\nfunction projectTruncatedLineSegment(previousTilePoint: Point, currentTilePoint: Point, previousProjectedPoint: Point, minimumLength: number, projectionMatrix: mat4, getElevation: (x: number, y: number) => number) {\n // We are assuming \"previousTilePoint\" won't project to a point within one unit of the camera plane\n // If it did, that would mean our label extended all the way out from within the viewport to a (very distant)\n // point near the plane of the camera. We wouldn't be able to render the label anyway once it crossed the\n // plane of the camera.\n const projectedUnitVertex = project(previousTilePoint.add(previousTilePoint.sub(currentTilePoint)._unit()), projectionMatrix, getElevation).point;\n const projectedUnitSegment = previousProjectedPoint.sub(projectedUnitVertex);\n\n return previousProjectedPoint.add(projectedUnitSegment._mult(minimumLength / projectedUnitSegment.mag()));\n}\n\ntype IndexToPointCache = { [lineIndex: number]: Point };\n\n/**\n * We calculate label-plane projected points for line vertices as we place glyphs along the line\n * Since we will use the same vertices for potentially many glyphs, cache the results for this bucket\n * over the course of the render. Each vertex location also potentially has one offset equivalent\n * for us to hold onto. The vertex indices are per-symbol-bucket.\n */\ntype ProjectionCache = {\n /**\n * tile-unit vertices projected into label-plane units\n */\n projections: IndexToPointCache;\n /**\n * label-plane vertices which have been shifted to follow an offset line\n */\n offsets: IndexToPointCache;\n};\n\n/**\n * Arguments necessary to project a vertex to the label plane\n */\ntype ProjectionArgs = {\n /**\n * Used to cache results, save cost if projecting the same vertex multiple times\n */\n projectionCache: ProjectionCache;\n /**\n * The array of tile-unit vertices transferred from worker\n */\n lineVertexArray: SymbolLineVertexArray;\n /**\n * Label plane projection matrix\n */\n labelPlaneMatrix: mat4;\n /**\n * Function to get elevation at a point\n * @param x - the x coordinate\n * @param y - the y coordinate\n */\n getElevation: (x: number, y: number) => number;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n tileAnchorPoint: Point;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n distanceFromAnchor: number;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n previousVertex: Point;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n direction: number;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n absOffsetX: number;\n};\n\n/**\n * Transform a vertex from tile coordinates to label plane coordinates\n * @param index - index of vertex to project\n * @param projectionArgs - necessary data to project a vertex\n * @returns the vertex projected to the label plane\n */\nfunction projectVertexToViewport(index: number, projectionArgs: ProjectionArgs): Point {\n const {projectionCache, lineVertexArray, labelPlaneMatrix, tileAnchorPoint, distanceFromAnchor, getElevation, previousVertex, direction, absOffsetX} = projectionArgs;\n if (projectionCache.projections[index]) {\n return projectionCache.projections[index];\n }\n const currentVertex = new Point(lineVertexArray.getx(index), lineVertexArray.gety(index));\n const projection = project(currentVertex, labelPlaneMatrix, getElevation);\n if (projection.signedDistanceFromCamera > 0) {\n projectionCache.projections[index] = projection.point;\n return projection.point;\n }\n\n // The vertex is behind the plane of the camera, so we can't project it\n // Instead, we'll create a vertex along the line that's far enough to include the glyph\n const previousLineVertexIndex = index - direction;\n const previousTilePoint = distanceFromAnchor === 0 ?\n tileAnchorPoint :\n new Point(lineVertexArray.getx(previousLineVertexIndex), lineVertexArray.gety(previousLineVertexIndex));\n // Don't cache because the new vertex might not be far enough out for future glyphs on the same segment\n return projectTruncatedLineSegment(previousTilePoint, currentVertex, previousVertex, absOffsetX - distanceFromAnchor + 1, labelPlaneMatrix, getElevation);\n}\n\n/**\n * Calculate the normal vector for a line segment\n * @param segmentVector - will be mutated as a tiny optimization\n * @param offset - magnitude of resulting vector\n * @param direction - direction of line traversal\n * @returns a normal vector from the segment, with magnitude equal to offset amount\n */\nfunction transformToOffsetNormal(segmentVector: Point, offset: number, direction: number): Point {\n return segmentVector._unit()._perp()._mult(offset * direction);\n}\n\n/**\n * Construct offset line segments for the current segment and the next segment, then extend/shrink\n * the segments until they intersect. If the segments are parallel, then they will touch with no modification.\n *\n * @param index - Index of the current vertex\n * @param prevToCurrentOffsetNormal - Normal vector of the line segment from the previous vertex to the current vertex\n * @param currentVertex - Current (non-offset) vertex projected to the label plane\n * @param lineStartIndex - Beginning index for the line this label is on\n * @param lineEndIndex - End index for the line this label is on\n * @param offsetPreviousVertex - The previous vertex projected to the label plane, and then offset along the previous segments normal\n * @param lineOffsetY - Magnitude of the offset\n * @param projectionArgs - Necessary data for tile-to-label-plane projection\n * @returns The point at which the current and next line segments intersect, once offset and extended/shrunk to their meeting point\n */\nfunction findOffsetIntersectionPoint(index: number, prevToCurrentOffsetNormal: Point, currentVertex: Point, lineStartIndex: number, lineEndIndex: number, offsetPreviousVertex: Point, lineOffsetY: number, projectionArgs: ProjectionArgs) {\n const {projectionCache, direction} = projectionArgs;\n if (projectionCache.offsets[index]) {\n return projectionCache.offsets[index];\n }\n\n const offsetCurrentVertex = currentVertex.add(prevToCurrentOffsetNormal);\n\n if (index + direction < lineStartIndex || index + direction >= lineEndIndex) {\n // This is the end of the line, no intersection to calculate\n projectionCache.offsets[index] = offsetCurrentVertex;\n return offsetCurrentVertex;\n }\n // Offset the vertices for the next segment\n const nextVertex = projectVertexToViewport(index + direction, projectionArgs);\n const currentToNextOffsetNormal = transformToOffsetNormal(nextVertex.sub(currentVertex), lineOffsetY, direction);\n const offsetNextSegmentBegin = currentVertex.add(currentToNextOffsetNormal);\n const offsetNextSegmentEnd = nextVertex.add(currentToNextOffsetNormal);\n\n // find the intersection of these two lines\n // if the lines are parallel, offsetCurrent/offsetNextBegin will touch\n projectionCache.offsets[index] = findLineIntersection(offsetPreviousVertex, offsetCurrentVertex, offsetNextSegmentBegin, offsetNextSegmentEnd) || offsetCurrentVertex;\n\n return projectionCache.offsets[index];\n}\n\n/**\n * Placed Glyph type\n */\ntype PlacedGlyph = {\n /**\n * The point at which the glyph should be placed, in label plane coordinates\n */\n point: Point;\n /**\n * The angle at which the glyph should be placed\n */\n angle: number;\n /**\n * The label-plane path used to reach this glyph: used only for collision detection\n */\n path: Array;\n};\n\n/*\n * Place a single glyph along its line, projected into the label plane, by iterating outward\n * from the anchor point until the distance traversed in the label plane equals the glyph's\n * offsetX. Returns null if the glyph can't fit on the line geometry.\n */\nfunction placeGlyphAlongLine(\n offsetX: number,\n lineOffsetX: number,\n lineOffsetY: number,\n flip: boolean,\n anchorPoint: Point,\n tileAnchorPoint: Point,\n anchorSegment: number,\n lineStartIndex: number,\n lineEndIndex: number,\n lineVertexArray: SymbolLineVertexArray,\n labelPlaneMatrix: mat4,\n projectionCache: ProjectionCache,\n rotateToLine: boolean,\n getElevation: (x: number, y: number) => number): PlacedGlyph | null {\n\n const combinedOffsetX = flip ?\n offsetX - lineOffsetX :\n offsetX + lineOffsetX;\n\n let direction = combinedOffsetX > 0 ? 1 : -1;\n\n let angle = 0;\n if (flip) {\n // The label needs to be flipped to keep text upright.\n // Iterate in the reverse direction.\n direction *= -1;\n angle = Math.PI;\n }\n\n if (direction < 0) angle += Math.PI;\n\n let currentIndex = direction > 0 ?\n lineStartIndex + anchorSegment :\n lineStartIndex + anchorSegment + 1;\n\n let currentVertex = anchorPoint;\n let previousVertex = anchorPoint;\n\n // offsetPrev and intersectionPoint are analogous to previousVertex and currentVertex\n // but if there's a line offset they are calculated in parallel as projection happens\n let offsetIntersectionPoint: Point;\n let offsetPreviousVertex: Point;\n\n let distanceFromAnchor = 0;\n let currentSegmentDistance = 0;\n const absOffsetX = Math.abs(combinedOffsetX);\n const pathVertices: Array = [];\n\n let currentLineSegment: Point;\n while (distanceFromAnchor + currentSegmentDistance <= absOffsetX) {\n currentIndex += direction;\n\n // offset does not fit on the projected line\n if (currentIndex < lineStartIndex || currentIndex >= lineEndIndex)\n return null;\n\n // accumulate values from last iteration\n distanceFromAnchor += currentSegmentDistance;\n previousVertex = currentVertex;\n offsetPreviousVertex = offsetIntersectionPoint;\n\n const projectionArgs: ProjectionArgs = {\n projectionCache,\n lineVertexArray,\n labelPlaneMatrix,\n tileAnchorPoint,\n distanceFromAnchor,\n getElevation,\n previousVertex,\n direction,\n absOffsetX\n };\n\n // find next vertex in viewport space\n currentVertex = projectVertexToViewport(currentIndex, projectionArgs);\n if (lineOffsetY === 0) {\n // Store vertices for collision detection and update current segment geometry\n pathVertices.push(previousVertex);\n currentLineSegment = currentVertex.sub(previousVertex);\n } else {\n // Calculate the offset for this section\n let prevToCurrentOffsetNormal;\n const prevToCurrent = currentVertex.sub(previousVertex);\n if (prevToCurrent.mag() === 0) {\n // We are starting with our anchor point directly on the vertex, so look one vertex ahead\n // to calculate a normal\n const nextVertex = projectVertexToViewport(currentIndex + direction, projectionArgs);\n prevToCurrentOffsetNormal = transformToOffsetNormal(nextVertex.sub(currentVertex), lineOffsetY, direction);\n } else {\n prevToCurrentOffsetNormal = transformToOffsetNormal(prevToCurrent, lineOffsetY, direction);\n }\n // Initialize offsetPrev on our first iteration, after that it will be pre-calculated\n if (!offsetPreviousVertex)\n offsetPreviousVertex = previousVertex.add(prevToCurrentOffsetNormal);\n\n offsetIntersectionPoint = findOffsetIntersectionPoint(currentIndex, prevToCurrentOffsetNormal, currentVertex, lineStartIndex, lineEndIndex, offsetPreviousVertex, lineOffsetY, projectionArgs);\n\n pathVertices.push(offsetPreviousVertex);\n currentLineSegment = offsetIntersectionPoint.sub(offsetPreviousVertex);\n }\n currentSegmentDistance = currentLineSegment.mag();\n }\n\n // The point is on the current segment. Interpolate to find it.\n const segmentInterpolationT = (absOffsetX - distanceFromAnchor) / currentSegmentDistance;\n const p = currentLineSegment._mult(segmentInterpolationT)._add(offsetPreviousVertex || previousVertex);\n\n const segmentAngle = angle + Math.atan2(currentVertex.y - previousVertex.y, currentVertex.x - previousVertex.x);\n\n pathVertices.push(p);\n\n return {\n point: p,\n angle: rotateToLine ? segmentAngle : 0.0,\n path: pathVertices\n };\n}\n\nconst hiddenGlyphAttributes = new Float32Array([-Infinity, -Infinity, 0, -Infinity, -Infinity, 0, -Infinity, -Infinity, 0, -Infinity, -Infinity, 0]);\n\n// Hide them by moving them offscreen. We still need to add them to the buffer\n// because the dynamic buffer is paired with a static buffer that doesn't get updated.\nfunction hideGlyphs(num: number, dynamicLayoutVertexArray: SymbolDynamicLayoutArray) {\n for (let i = 0; i < num; i++) {\n const offset = dynamicLayoutVertexArray.length;\n dynamicLayoutVertexArray.resize(offset + 4);\n // Since all hidden glyphs have the same attributes, we can build up the array faster with a single call to Float32Array.set\n // for each set of four vertices, instead of calling addDynamicAttributes for each vertex.\n dynamicLayoutVertexArray.float32.set(hiddenGlyphAttributes, offset * 3);\n }\n}\n\n// For line label layout, we're not using z output and our w input is always 1\n// This custom matrix transformation ignores those components to make projection faster\nfunction xyTransformMat4(out: vec4, a: vec4, m: mat4) {\n const x = a[0], y = a[1];\n out[0] = m[0] * x + m[4] * y + m[12];\n out[1] = m[1] * x + m[5] * y + m[13];\n out[3] = m[3] * x + m[7] * y + m[15];\n return out;\n}\n","import Point from '@mapbox/point-geometry';\nimport {clipLine} from './clip_line';\nimport {PathInterpolator} from './path_interpolator';\n\nimport * as intersectionTests from '../util/intersection_tests';\nimport {GridIndex} from './grid_index';\nimport {mat4, vec4} from 'gl-matrix';\nimport ONE_EM from '../symbol/one_em';\n\nimport * as projection from '../symbol/projection';\n\nimport type {Transform} from '../geo/transform';\nimport type {SingleCollisionBox} from '../data/bucket/symbol_bucket';\nimport type {\n GlyphOffsetArray,\n SymbolLineVertexArray\n} from '../data/array_types.g';\nimport type {OverlapMode} from '../style/style_layer/overlap_mode';\n\n// When a symbol crosses the edge that causes it to be included in\n// collision detection, it will cause changes in the symbols around\n// it. This constant specifies how many pixels to pad the edge of\n// the viewport for collision detection so that the bulk of the changes\n// occur offscreen. Making this constant greater increases label\n// stability, but it's expensive.\nconst viewportPadding = 100;\n\nexport type FeatureKey = {\n bucketInstanceId: number;\n featureIndex: number;\n collisionGroupID: number;\n overlapMode: OverlapMode;\n};\n\n/**\n * @internal\n * A collision index used to prevent symbols from overlapping. It keep tracks of\n * where previous symbols have been placed and is used to check if a new\n * symbol overlaps with any previously added symbols.\n *\n * There are two steps to insertion: first placeCollisionBox/Circles checks if\n * there's room for a symbol, then insertCollisionBox/Circles actually puts the\n * symbol in the index. The two step process allows paired symbols to be inserted\n * together even if they overlap.\n */\nexport class CollisionIndex {\n grid: GridIndex;\n ignoredGrid: GridIndex;\n transform: Transform;\n pitchfactor: number;\n screenRightBoundary: number;\n screenBottomBoundary: number;\n gridRightBoundary: number;\n gridBottomBoundary: number;\n\n // With perspectiveRatio the fontsize is calculated for tilted maps (near = bigger, far = smaller).\n // The cutoff defines a threshold to no longer render labels near the horizon.\n perspectiveRatioCutoff: number;\n\n constructor(\n transform: Transform,\n grid = new GridIndex(transform.width + 2 * viewportPadding, transform.height + 2 * viewportPadding, 25),\n ignoredGrid = new GridIndex(transform.width + 2 * viewportPadding, transform.height + 2 * viewportPadding, 25)\n ) {\n this.transform = transform;\n\n this.grid = grid;\n this.ignoredGrid = ignoredGrid;\n this.pitchfactor = Math.cos(transform._pitch) * transform.cameraToCenterDistance;\n\n this.screenRightBoundary = transform.width + viewportPadding;\n this.screenBottomBoundary = transform.height + viewportPadding;\n this.gridRightBoundary = transform.width + 2 * viewportPadding;\n this.gridBottomBoundary = transform.height + 2 * viewportPadding;\n\n this.perspectiveRatioCutoff = 0.6;\n }\n\n placeCollisionBox(\n collisionBox: SingleCollisionBox,\n overlapMode: OverlapMode,\n textPixelRatio: number,\n posMatrix: mat4,\n collisionGroupPredicate?: (key: FeatureKey) => boolean,\n getElevation?: (x: number, y: number) => number\n ): {\n box: Array;\n offscreen: boolean;\n } {\n const projectedPoint = this.projectAndGetPerspectiveRatio(posMatrix, collisionBox.anchorPointX, collisionBox.anchorPointY, getElevation);\n const tileToViewport = textPixelRatio * projectedPoint.perspectiveRatio;\n const tlX = collisionBox.x1 * tileToViewport + projectedPoint.point.x;\n const tlY = collisionBox.y1 * tileToViewport + projectedPoint.point.y;\n const brX = collisionBox.x2 * tileToViewport + projectedPoint.point.x;\n const brY = collisionBox.y2 * tileToViewport + projectedPoint.point.y;\n\n if (!this.isInsideGrid(tlX, tlY, brX, brY) ||\n (overlapMode !== 'always' && this.grid.hitTest(tlX, tlY, brX, brY, overlapMode, collisionGroupPredicate)) ||\n projectedPoint.perspectiveRatio < this.perspectiveRatioCutoff) {\n return {\n box: [],\n offscreen: false\n };\n }\n\n return {\n box: [tlX, tlY, brX, brY],\n offscreen: this.isOffscreen(tlX, tlY, brX, brY)\n };\n }\n\n placeCollisionCircles(\n overlapMode: OverlapMode,\n symbol: any,\n lineVertexArray: SymbolLineVertexArray,\n glyphOffsetArray: GlyphOffsetArray,\n fontSize: number,\n posMatrix: mat4,\n labelPlaneMatrix: mat4,\n labelToScreenMatrix: mat4,\n showCollisionCircles: boolean,\n pitchWithMap: boolean,\n collisionGroupPredicate: (key: FeatureKey) => boolean,\n circlePixelDiameter: number,\n textPixelPadding: number,\n getElevation: (x: number, y: number) => number\n ): {\n circles: Array;\n offscreen: boolean;\n collisionDetected: boolean;\n } {\n const placedCollisionCircles = [];\n\n const tileUnitAnchorPoint = new Point(symbol.anchorX, symbol.anchorY);\n const screenAnchorPoint = projection.project(tileUnitAnchorPoint, posMatrix, getElevation);\n const perspectiveRatio = projection.getPerspectiveRatio(this.transform.cameraToCenterDistance, screenAnchorPoint.signedDistanceFromCamera);\n const labelPlaneFontSize = pitchWithMap ? fontSize / perspectiveRatio : fontSize * perspectiveRatio;\n const labelPlaneFontScale = labelPlaneFontSize / ONE_EM;\n\n const labelPlaneAnchorPoint = projection.project(tileUnitAnchorPoint, labelPlaneMatrix, getElevation).point;\n\n const projectionCache = {projections: {}, offsets: {}};\n const lineOffsetX = symbol.lineOffsetX * labelPlaneFontScale;\n const lineOffsetY = symbol.lineOffsetY * labelPlaneFontScale;\n\n const firstAndLastGlyph = projection.placeFirstAndLastGlyph(\n labelPlaneFontScale,\n glyphOffsetArray,\n lineOffsetX,\n lineOffsetY,\n /*flip*/ false,\n labelPlaneAnchorPoint,\n tileUnitAnchorPoint,\n symbol,\n lineVertexArray,\n labelPlaneMatrix,\n projectionCache,\n false,\n getElevation);\n\n let collisionDetected = false;\n let inGrid = false;\n let entirelyOffscreen = true;\n\n if (firstAndLastGlyph) {\n const radius = circlePixelDiameter * 0.5 * perspectiveRatio + textPixelPadding;\n const screenPlaneMin = new Point(-viewportPadding, -viewportPadding);\n const screenPlaneMax = new Point(this.screenRightBoundary, this.screenBottomBoundary);\n const interpolator = new PathInterpolator();\n\n // Construct a projected path from projected line vertices. Anchor points are ignored and removed\n const first = firstAndLastGlyph.first;\n const last = firstAndLastGlyph.last;\n\n let projectedPath = [];\n for (let i = first.path.length - 1; i >= 1; i--) {\n projectedPath.push(first.path[i]);\n }\n for (let i = 1; i < last.path.length; i++) {\n projectedPath.push(last.path[i]);\n }\n\n // Tolerate a slightly longer distance than one diameter between two adjacent circles\n const circleDist = radius * 2.5;\n\n // The path might need to be converted into screen space if a pitched map is used as the label space\n if (labelToScreenMatrix) {\n const screenSpacePath = projectedPath.map(p => projection.project(p, labelToScreenMatrix, getElevation));\n\n // Do not try to place collision circles if even of the points is behind the camera.\n // This is a plausible scenario with big camera pitch angles\n if (screenSpacePath.some(point => point.signedDistanceFromCamera <= 0)) {\n projectedPath = [];\n } else {\n projectedPath = screenSpacePath.map(p => p.point);\n }\n }\n\n let segments = [];\n\n if (projectedPath.length > 0) {\n // Quickly check if the path is fully inside or outside of the padded collision region.\n // For overlapping paths we'll only create collision circles for the visible segments\n const minPoint = projectedPath[0].clone();\n const maxPoint = projectedPath[0].clone();\n\n for (let i = 1; i < projectedPath.length; i++) {\n minPoint.x = Math.min(minPoint.x, projectedPath[i].x);\n minPoint.y = Math.min(minPoint.y, projectedPath[i].y);\n maxPoint.x = Math.max(maxPoint.x, projectedPath[i].x);\n maxPoint.y = Math.max(maxPoint.y, projectedPath[i].y);\n }\n\n if (minPoint.x >= screenPlaneMin.x && maxPoint.x <= screenPlaneMax.x &&\n minPoint.y >= screenPlaneMin.y && maxPoint.y <= screenPlaneMax.y) {\n // Quad fully visible\n segments = [projectedPath];\n } else if (maxPoint.x < screenPlaneMin.x || minPoint.x > screenPlaneMax.x ||\n maxPoint.y < screenPlaneMin.y || minPoint.y > screenPlaneMax.y) {\n // Not visible\n segments = [];\n } else {\n segments = clipLine([projectedPath], screenPlaneMin.x, screenPlaneMin.y, screenPlaneMax.x, screenPlaneMax.y);\n }\n }\n\n for (const seg of segments) {\n // interpolate positions for collision circles. Add a small padding to both ends of the segment\n interpolator.reset(seg, radius * 0.25);\n\n let numCircles = 0;\n\n if (interpolator.length <= 0.5 * radius) {\n numCircles = 1;\n } else {\n numCircles = Math.ceil(interpolator.paddedLength / circleDist) + 1;\n }\n\n for (let i = 0; i < numCircles; i++) {\n const t = i / Math.max(numCircles - 1, 1);\n const circlePosition = interpolator.lerp(t);\n\n // add viewport padding to the position and perform initial collision check\n const centerX = circlePosition.x + viewportPadding;\n const centerY = circlePosition.y + viewportPadding;\n\n placedCollisionCircles.push(centerX, centerY, radius, 0);\n\n const x1 = centerX - radius;\n const y1 = centerY - radius;\n const x2 = centerX + radius;\n const y2 = centerY + radius;\n\n entirelyOffscreen = entirelyOffscreen && this.isOffscreen(x1, y1, x2, y2);\n inGrid = inGrid || this.isInsideGrid(x1, y1, x2, y2);\n\n if (overlapMode !== 'always' && this.grid.hitTestCircle(centerX, centerY, radius, overlapMode, collisionGroupPredicate)) {\n // Don't early exit if we're showing the debug circles because we still want to calculate\n // which circles are in use\n collisionDetected = true;\n if (!showCollisionCircles) {\n return {\n circles: [],\n offscreen: false,\n collisionDetected\n };\n }\n }\n }\n }\n }\n\n return {\n circles: ((!showCollisionCircles && collisionDetected) || !inGrid || perspectiveRatio < this.perspectiveRatioCutoff) ? [] : placedCollisionCircles,\n offscreen: entirelyOffscreen,\n collisionDetected\n };\n }\n\n /**\n * Because the geometries in the CollisionIndex are an approximation of the shape of\n * symbols on the map, we use the CollisionIndex to look up the symbol part of\n * `queryRenderedFeatures`.\n */\n queryRenderedSymbols(viewportQueryGeometry: Array) {\n if (viewportQueryGeometry.length === 0 || (this.grid.keysLength() === 0 && this.ignoredGrid.keysLength() === 0)) {\n return {};\n }\n\n const query = [];\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n for (const point of viewportQueryGeometry) {\n const gridPoint = new Point(point.x + viewportPadding, point.y + viewportPadding);\n minX = Math.min(minX, gridPoint.x);\n minY = Math.min(minY, gridPoint.y);\n maxX = Math.max(maxX, gridPoint.x);\n maxY = Math.max(maxY, gridPoint.y);\n query.push(gridPoint);\n }\n\n const features = this.grid.query(minX, minY, maxX, maxY)\n .concat(this.ignoredGrid.query(minX, minY, maxX, maxY));\n\n const seenFeatures = {};\n const result = {};\n\n for (const feature of features) {\n const featureKey = feature.key;\n // Skip already seen features.\n if (seenFeatures[featureKey.bucketInstanceId] === undefined) {\n seenFeatures[featureKey.bucketInstanceId] = {};\n }\n if (seenFeatures[featureKey.bucketInstanceId][featureKey.featureIndex]) {\n continue;\n }\n\n // Check if query intersects with the feature box\n // \"Collision Circles\" for line labels are treated as boxes here\n // Since there's no actual collision taking place, the circle vs. square\n // distinction doesn't matter as much, and box geometry is easier\n // to work with.\n const bbox = [\n new Point(feature.x1, feature.y1),\n new Point(feature.x2, feature.y1),\n new Point(feature.x2, feature.y2),\n new Point(feature.x1, feature.y2)\n ];\n if (!intersectionTests.polygonIntersectsPolygon(query, bbox)) {\n continue;\n }\n\n seenFeatures[featureKey.bucketInstanceId][featureKey.featureIndex] = true;\n if (result[featureKey.bucketInstanceId] === undefined) {\n result[featureKey.bucketInstanceId] = [];\n }\n result[featureKey.bucketInstanceId].push(featureKey.featureIndex);\n }\n\n return result;\n }\n\n insertCollisionBox(collisionBox: Array, overlapMode: OverlapMode, ignorePlacement: boolean, bucketInstanceId: number, featureIndex: number, collisionGroupID: number) {\n const grid = ignorePlacement ? this.ignoredGrid : this.grid;\n\n const key = {bucketInstanceId, featureIndex, collisionGroupID, overlapMode};\n grid.insert(key, collisionBox[0], collisionBox[1], collisionBox[2], collisionBox[3]);\n }\n\n insertCollisionCircles(collisionCircles: Array, overlapMode: OverlapMode, ignorePlacement: boolean, bucketInstanceId: number, featureIndex: number, collisionGroupID: number) {\n const grid = ignorePlacement ? this.ignoredGrid : this.grid;\n\n const key = {bucketInstanceId, featureIndex, collisionGroupID, overlapMode};\n for (let k = 0; k < collisionCircles.length; k += 4) {\n grid.insertCircle(key, collisionCircles[k], collisionCircles[k + 1], collisionCircles[k + 2]);\n }\n }\n\n projectAndGetPerspectiveRatio(posMatrix: mat4, x: number, y: number, getElevation?: (x: number, y: number) => number) {\n let p;\n if (getElevation) { // slow because of handle z-index\n p = [x, y, getElevation(x, y), 1] as vec4;\n vec4.transformMat4(p, p, posMatrix);\n } else { // fast because of ignore z-index\n p = [x, y, 0, 1] as vec4;\n projection.xyTransformMat4(p, p, posMatrix);\n }\n const a = new Point(\n (((p[0] / p[3] + 1) / 2) * this.transform.width) + viewportPadding,\n (((-p[1] / p[3] + 1) / 2) * this.transform.height) + viewportPadding\n );\n return {\n point: a,\n // See perspective ratio comment in symbol_sdf.vertex\n // We're doing collision detection in viewport space so we need\n // to scale down boxes in the distance\n perspectiveRatio: 0.5 + 0.5 * (this.transform.cameraToCenterDistance / p[3])\n };\n }\n\n isOffscreen(x1: number, y1: number, x2: number, y2: number) {\n return x2 < viewportPadding || x1 >= this.screenRightBoundary || y2 < viewportPadding || y1 > this.screenBottomBoundary;\n }\n\n isInsideGrid(x1: number, y1: number, x2: number, y2: number) {\n return x2 >= 0 && x1 < this.gridRightBoundary && y2 >= 0 && y1 < this.gridBottomBoundary;\n }\n\n /*\n * Returns a matrix for transforming collision shapes to viewport coordinate space.\n * Use this function to render e.g. collision circles on the screen.\n * example transformation: clipPos = glCoordMatrix * viewportMatrix * circle_pos\n */\n getViewportMatrix() {\n const m = mat4.identity([] as any);\n mat4.translate(m, m, [-viewportPadding, -viewportPadding, 0.0]);\n return m;\n }\n}\n","import Point from '@mapbox/point-geometry';\n\n/**\n * Returns the part of a multiline that intersects with the provided rectangular box.\n *\n * @param lines - the lines to check\n * @param x1 - the left edge of the box\n * @param y1 - the top edge of the box\n * @param x2 - the right edge of the box\n * @param y2 - the bottom edge of the box\n * @returns lines\n */\nexport function clipLine(lines: Array>, x1: number, y1: number, x2: number, y2: number): Array> {\n const clippedLines = [];\n\n for (let l = 0; l < lines.length; l++) {\n const line = lines[l];\n let clippedLine;\n\n for (let i = 0; i < line.length - 1; i++) {\n let p0 = line[i];\n let p1 = line[i + 1];\n\n if (p0.x < x1 && p1.x < x1) {\n continue;\n } else if (p0.x < x1) {\n p0 = new Point(x1, p0.y + (p1.y - p0.y) * ((x1 - p0.x) / (p1.x - p0.x)))._round();\n } else if (p1.x < x1) {\n p1 = new Point(x1, p0.y + (p1.y - p0.y) * ((x1 - p0.x) / (p1.x - p0.x)))._round();\n }\n\n if (p0.y < y1 && p1.y < y1) {\n continue;\n } else if (p0.y < y1) {\n p0 = new Point(p0.x + (p1.x - p0.x) * ((y1 - p0.y) / (p1.y - p0.y)), y1)._round();\n } else if (p1.y < y1) {\n p1 = new Point(p0.x + (p1.x - p0.x) * ((y1 - p0.y) / (p1.y - p0.y)), y1)._round();\n }\n\n if (p0.x >= x2 && p1.x >= x2) {\n continue;\n } else if (p0.x >= x2) {\n p0 = new Point(x2, p0.y + (p1.y - p0.y) * ((x2 - p0.x) / (p1.x - p0.x)))._round();\n } else if (p1.x >= x2) {\n p1 = new Point(x2, p0.y + (p1.y - p0.y) * ((x2 - p0.x) / (p1.x - p0.x)))._round();\n }\n\n if (p0.y >= y2 && p1.y >= y2) {\n continue;\n } else if (p0.y >= y2) {\n p0 = new Point(p0.x + (p1.x - p0.x) * ((y2 - p0.y) / (p1.y - p0.y)), y2)._round();\n } else if (p1.y >= y2) {\n p1 = new Point(p0.x + (p1.x - p0.x) * ((y2 - p0.y) / (p1.y - p0.y)), y2)._round();\n }\n\n if (!clippedLine || !p0.equals(clippedLine[clippedLine.length - 1])) {\n clippedLine = [p0];\n clippedLines.push(clippedLine);\n }\n\n clippedLine.push(p1);\n }\n }\n\n return clippedLines;\n}\n","import Point from '@mapbox/point-geometry';\n\nimport {register} from '../util/web_worker_transfer';\n\nexport class Anchor extends Point {\n angle: any;\n segment?: number;\n\n constructor(x: number, y: number, angle: number, segment?: number) {\n super(x, y);\n this.angle = angle;\n if (segment !== undefined) {\n this.segment = segment;\n }\n }\n\n clone() {\n return new Anchor(this.x, this.y, this.angle, this.segment);\n }\n}\n\nregister('Anchor', Anchor);\n","import {VariableAnchorOffsetCollection, VariableAnchorOffsetCollectionSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport {SymbolFeature} from '../../data/bucket/symbol_bucket';\nimport {CanonicalTileID} from '../../source/tile_id';\nimport ONE_EM from '../../symbol/one_em';\nimport {SymbolStyleLayer} from './symbol_style_layer';\n\nexport enum TextAnchorEnum {\n 'center' = 1,\n 'left' = 2,\n 'right' = 3,\n 'top' = 4,\n 'bottom' = 5,\n 'top-left' = 6,\n 'top-right' = 7,\n 'bottom-left' = 8,\n 'bottom-right' = 9\n}\n\nexport type TextAnchor = keyof typeof TextAnchorEnum;\n\n// The radial offset is to the edge of the text box\n// In the horizontal direction, the edge of the text box is where glyphs start\n// But in the vertical direction, the glyphs appear to \"start\" at the baseline\n// We don't actually load baseline data, but we assume an offset of ONE_EM - 17\n// (see \"yOffset\" in shaping.js)\nconst baselineOffset = 7;\nexport const INVALID_TEXT_OFFSET = Number.POSITIVE_INFINITY;\n\nexport function evaluateVariableOffset(anchor: TextAnchor, offset: [number, number]): [number, number] {\n\n function fromRadialOffset(anchor: TextAnchor, radialOffset: number): [number, number] {\n let x = 0, y = 0;\n if (radialOffset < 0) radialOffset = 0; // Ignore negative offset.\n // solve for r where r^2 + r^2 = radialOffset^2\n const hypotenuse = radialOffset / Math.SQRT2;\n switch (anchor) {\n case 'top-right':\n case 'top-left':\n y = hypotenuse - baselineOffset;\n break;\n case 'bottom-right':\n case 'bottom-left':\n y = -hypotenuse + baselineOffset;\n break;\n case 'bottom':\n y = -radialOffset + baselineOffset;\n break;\n case 'top':\n y = radialOffset - baselineOffset;\n break;\n }\n\n switch (anchor) {\n case 'top-right':\n case 'bottom-right':\n x = -hypotenuse;\n break;\n case 'top-left':\n case 'bottom-left':\n x = hypotenuse;\n break;\n case 'left':\n x = radialOffset;\n break;\n case 'right':\n x = -radialOffset;\n break;\n }\n\n return [x, y];\n }\n\n function fromTextOffset(anchor: TextAnchor, offsetX: number, offsetY: number): [number, number] {\n let x = 0, y = 0;\n // Use absolute offset values.\n offsetX = Math.abs(offsetX);\n offsetY = Math.abs(offsetY);\n\n switch (anchor) {\n case 'top-right':\n case 'top-left':\n case 'top':\n y = offsetY - baselineOffset;\n break;\n case 'bottom-right':\n case 'bottom-left':\n case 'bottom':\n y = -offsetY + baselineOffset;\n break;\n }\n\n switch (anchor) {\n case 'top-right':\n case 'bottom-right':\n case 'right':\n x = -offsetX;\n break;\n case 'top-left':\n case 'bottom-left':\n case 'left':\n x = offsetX;\n break;\n }\n\n return [x, y];\n }\n\n return (offset[1] !== INVALID_TEXT_OFFSET) ? fromTextOffset(anchor, offset[0], offset[1]) : fromRadialOffset(anchor, offset[0]);\n}\n\n// Helper to support both text-variable-anchor and text-variable-anchor-offset. Offset values converted from EMs to PXs\nexport function getTextVariableAnchorOffset(layer: SymbolStyleLayer, feature: SymbolFeature, canonical: CanonicalTileID): VariableAnchorOffsetCollection | null {\n const layout = layer.layout;\n // If style specifies text-variable-anchor-offset, just return it\n const variableAnchorOffset = layout.get('text-variable-anchor-offset')?.evaluate(feature, {}, canonical);\n\n if (variableAnchorOffset) {\n const sourceValues = variableAnchorOffset.values;\n const destValues: VariableAnchorOffsetCollectionSpecification = [];\n\n // Convert offsets from EM to PX, and apply baseline shift\n for (let i = 0; i < sourceValues.length; i += 2) {\n const anchor = destValues[i] = sourceValues[i] as TextAnchor;\n const offset = (sourceValues[i + 1] as [number, number]).map(t => t * ONE_EM) as [number, number];\n\n if (anchor.startsWith('top')) {\n offset[1] -= baselineOffset;\n } else if (anchor.startsWith('bottom')) {\n offset[1] += baselineOffset;\n }\n\n destValues[i + 1] = offset;\n }\n\n return new VariableAnchorOffsetCollection(destValues);\n }\n\n // If style specifies text-variable-anchor, convert to the new format\n const variableAnchor = layout.get('text-variable-anchor');\n\n if (variableAnchor) {\n let textOffset: [number, number];\n const unevaluatedLayout = layer._unevaluatedLayout;\n\n // The style spec says don't use `text-offset` and `text-radial-offset` together\n // but doesn't actually specify what happens if you use both. We go with the radial offset.\n if (unevaluatedLayout.getValue('text-radial-offset') !== undefined) {\n textOffset = [layout.get('text-radial-offset').evaluate(feature, {}, canonical) * ONE_EM, INVALID_TEXT_OFFSET];\n } else {\n textOffset = layout.get('text-offset').evaluate(feature, {}, canonical).map(t => t * ONE_EM) as [number, number];\n }\n\n const anchorOffsets: VariableAnchorOffsetCollectionSpecification = [];\n\n for (const anchor of variableAnchor) {\n anchorOffsets.push(anchor, evaluateVariableOffset(anchor, textOffset));\n }\n\n return new VariableAnchorOffsetCollection(anchorOffsets);\n }\n\n return null;\n}\n","import {EXTENT} from '../data/extent';\n\nimport type {OverscaledTileID} from './tile_id';\n\n/**\n * Converts a pixel value at a the given zoom level to tile units.\n *\n * The shaders mostly calculate everything in tile units so style\n * properties need to be converted from pixels to tile units using this.\n *\n * For example, a translation by 30 pixels at zoom 6.5 will be a\n * translation by pixelsToTileUnits(30, 6.5) tile units.\n *\n * @returns value in tile units\n */\nexport function pixelsToTileUnits(\n tile: {\n tileID: OverscaledTileID;\n tileSize: number;\n },\n pixelValue: number,\n z: number\n): number {\n return pixelValue * (EXTENT / (tile.tileSize * Math.pow(2, z - tile.tileID.overscaledZ)));\n}\n","import {CollisionIndex} from './collision_index';\nimport type {FeatureKey} from './collision_index';\nimport {EXTENT} from '../data/extent';\nimport * as symbolSize from './symbol_size';\nimport * as projection from './projection';\nimport {getAnchorJustification} from './symbol_layout';\nimport {getAnchorAlignment, WritingMode} from './shaping';\nimport {mat4} from 'gl-matrix';\nimport {pixelsToTileUnits} from '../source/pixels_to_tile_units';\nimport Point from '@mapbox/point-geometry';\nimport type {Transform} from '../geo/transform';\nimport type {StyleLayer} from '../style/style_layer';\nimport {PossiblyEvaluated} from '../style/properties';\nimport type {SymbolLayoutProps, SymbolLayoutPropsPossiblyEvaluated} from '../style/style_layer/symbol_style_layer_properties.g';\nimport {getOverlapMode, OverlapMode} from '../style/style_layer/overlap_mode';\n\nimport type {Tile} from '../source/tile';\nimport {SymbolBucket, CollisionArrays, SingleCollisionBox} from '../data/bucket/symbol_bucket';\n\nimport type {CollisionBoxArray, CollisionVertexArray, SymbolInstance, TextAnchorOffset} from '../data/array_types.g';\nimport type {FeatureIndex} from '../data/feature_index';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport {Terrain} from '../render/terrain';\nimport {warnOnce} from '../util/util';\nimport {TextAnchor, TextAnchorEnum} from '../style/style_layer/variable_text_anchor';\n\nclass OpacityState {\n opacity: number;\n placed: boolean;\n constructor(prevState: OpacityState, increment: number, placed: boolean, skipFade?: boolean | null) {\n if (prevState) {\n this.opacity = Math.max(0, Math.min(1, prevState.opacity + (prevState.placed ? increment : -increment)));\n } else {\n this.opacity = (skipFade && placed) ? 1 : 0;\n }\n this.placed = placed;\n }\n isHidden() {\n return this.opacity === 0 && !this.placed;\n }\n}\n\nclass JointOpacityState {\n text: OpacityState;\n icon: OpacityState;\n constructor(prevState: JointOpacityState, increment: number, placedText: boolean, placedIcon: boolean, skipFade?: boolean | null) {\n this.text = new OpacityState(prevState ? prevState.text : null, increment, placedText, skipFade);\n this.icon = new OpacityState(prevState ? prevState.icon : null, increment, placedIcon, skipFade);\n }\n isHidden() {\n return this.text.isHidden() && this.icon.isHidden();\n }\n}\n\nclass JointPlacement {\n text: boolean;\n icon: boolean;\n // skipFade = outside viewport, but within CollisionIndex::viewportPadding px of the edge\n // Because these symbols aren't onscreen yet, we can skip the \"fade in\" animation,\n // and if a subsequent viewport change brings them into view, they'll be fully\n // visible right away.\n skipFade: boolean;\n constructor(text: boolean, icon: boolean, skipFade: boolean) {\n this.text = text;\n this.icon = icon;\n this.skipFade = skipFade;\n }\n}\n\nclass CollisionCircleArray {\n // Stores collision circles and placement matrices of a bucket for debug rendering.\n invProjMatrix: mat4;\n viewportMatrix: mat4;\n circles: Array;\n\n constructor() {\n this.invProjMatrix = mat4.create();\n this.viewportMatrix = mat4.create();\n this.circles = [];\n }\n}\n\nexport class RetainedQueryData {\n bucketInstanceId: number;\n featureIndex: FeatureIndex;\n sourceLayerIndex: number;\n bucketIndex: number;\n tileID: OverscaledTileID;\n featureSortOrder: Array;\n constructor(bucketInstanceId: number,\n featureIndex: FeatureIndex,\n sourceLayerIndex: number,\n bucketIndex: number,\n tileID: OverscaledTileID) {\n this.bucketInstanceId = bucketInstanceId;\n this.featureIndex = featureIndex;\n this.sourceLayerIndex = sourceLayerIndex;\n this.bucketIndex = bucketIndex;\n this.tileID = tileID;\n }\n}\n\ntype CollisionGroup = {\n ID: number;\n predicate?: (key: FeatureKey) => boolean;\n};\n\nclass CollisionGroups {\n collisionGroups: {[groupName: string]: CollisionGroup};\n maxGroupID: number;\n crossSourceCollisions: boolean;\n\n constructor(crossSourceCollisions: boolean) {\n this.crossSourceCollisions = crossSourceCollisions;\n this.maxGroupID = 0;\n this.collisionGroups = {};\n }\n\n get(sourceID: string) {\n // The predicate/groupID mechanism allows for arbitrary grouping,\n // but the current interface defines one source == one group when\n // crossSourceCollisions == true.\n if (!this.crossSourceCollisions) {\n if (!this.collisionGroups[sourceID]) {\n const nextGroupID = ++this.maxGroupID;\n this.collisionGroups[sourceID] = {\n ID: nextGroupID,\n predicate: (key) => {\n return key.collisionGroupID === nextGroupID;\n }\n };\n }\n return this.collisionGroups[sourceID];\n } else {\n return {ID: 0, predicate: null};\n }\n }\n}\n\nfunction calculateVariableLayoutShift(\n anchor: TextAnchor,\n width: number,\n height: number,\n textOffset: [number, number],\n textBoxScale: number\n): Point {\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(anchor);\n const shiftX = -(horizontalAlign - 0.5) * width;\n const shiftY = -(verticalAlign - 0.5) * height;\n return new Point(\n shiftX + textOffset[0] * textBoxScale,\n shiftY + textOffset[1] * textBoxScale\n );\n}\n\nfunction shiftVariableCollisionBox(collisionBox: SingleCollisionBox,\n shiftX: number, shiftY: number,\n rotateWithMap: boolean, pitchWithMap: boolean,\n angle: number) {\n const {x1, x2, y1, y2, anchorPointX, anchorPointY} = collisionBox;\n const rotatedOffset = new Point(shiftX, shiftY);\n if (rotateWithMap) {\n rotatedOffset._rotate(pitchWithMap ? angle : -angle);\n }\n return {\n x1: x1 + rotatedOffset.x,\n y1: y1 + rotatedOffset.y,\n x2: x2 + rotatedOffset.x,\n y2: y2 + rotatedOffset.y,\n // symbol anchor point stays the same regardless of text-anchor\n anchorPointX,\n anchorPointY\n };\n}\n\nexport type VariableOffset = {\n textOffset: [number, number];\n width: number;\n height: number;\n anchor: TextAnchor;\n textBoxScale: number;\n prevAnchor?: TextAnchor;\n};\n\ntype TileLayerParameters = {\n bucket: SymbolBucket;\n layout: PossiblyEvaluated;\n posMatrix: mat4;\n textLabelPlaneMatrix: mat4;\n labelToScreenMatrix: mat4;\n scale: number;\n textPixelRatio: number;\n holdingForFade: boolean;\n collisionBoxArray: CollisionBoxArray;\n partiallyEvaluatedTextSize: {\n uSize: number;\n uSizeT: number;\n };\n collisionGroup: CollisionGroup;\n};\n\nexport type BucketPart = {\n sortKey?: number | void;\n symbolInstanceStart: number;\n symbolInstanceEnd: number;\n parameters: TileLayerParameters;\n};\n\nexport type CrossTileID = string | number;\n\nexport class Placement {\n transform: Transform;\n terrain: Terrain;\n collisionIndex: CollisionIndex;\n placements: {\n [_ in CrossTileID]: JointPlacement;\n };\n opacities: {\n [_ in CrossTileID]: JointOpacityState;\n };\n variableOffsets: {\n [_ in CrossTileID]: VariableOffset;\n };\n placedOrientations: {\n [_ in CrossTileID]: number;\n };\n commitTime: number;\n prevZoomAdjustment: number;\n lastPlacementChangeTime: number;\n stale: boolean;\n fadeDuration: number;\n retainedQueryData: {\n [_: number]: RetainedQueryData;\n };\n collisionGroups: CollisionGroups;\n prevPlacement: Placement;\n zoomAtLastRecencyCheck: number;\n collisionCircleArrays: {\n [k in any]: CollisionCircleArray;\n };\n\n constructor(transform: Transform, terrain: Terrain, fadeDuration: number, crossSourceCollisions: boolean, prevPlacement?: Placement) {\n this.transform = transform.clone();\n this.terrain = terrain;\n this.collisionIndex = new CollisionIndex(this.transform);\n this.placements = {};\n this.opacities = {};\n this.variableOffsets = {};\n this.stale = false;\n this.commitTime = 0;\n this.fadeDuration = fadeDuration;\n this.retainedQueryData = {};\n this.collisionGroups = new CollisionGroups(crossSourceCollisions);\n this.collisionCircleArrays = {};\n\n this.prevPlacement = prevPlacement;\n if (prevPlacement) {\n prevPlacement.prevPlacement = undefined; // Only hold on to one placement back\n }\n\n this.placedOrientations = {};\n }\n\n getBucketParts(results: Array, styleLayer: StyleLayer, tile: Tile, sortAcrossTiles: boolean) {\n const symbolBucket = (tile.getBucket(styleLayer) as SymbolBucket);\n const bucketFeatureIndex = tile.latestFeatureIndex;\n if (!symbolBucket || !bucketFeatureIndex || styleLayer.id !== symbolBucket.layerIds[0])\n return;\n\n const collisionBoxArray = tile.collisionBoxArray;\n\n const layout = symbolBucket.layers[0].layout;\n\n const scale = Math.pow(2, this.transform.zoom - tile.tileID.overscaledZ);\n const textPixelRatio = tile.tileSize / EXTENT;\n\n const posMatrix = this.transform.calculatePosMatrix(tile.tileID.toUnwrapped());\n\n const pitchWithMap = layout.get('text-pitch-alignment') === 'map';\n const rotateWithMap = layout.get('text-rotation-alignment') === 'map';\n const pixelsToTiles = pixelsToTileUnits(tile, 1, this.transform.zoom);\n\n const textLabelPlaneMatrix = projection.getLabelPlaneMatrix(posMatrix,\n pitchWithMap,\n rotateWithMap,\n this.transform,\n pixelsToTiles);\n\n let labelToScreenMatrix = null;\n\n if (pitchWithMap) {\n const glMatrix = projection.getGlCoordMatrix(\n posMatrix,\n pitchWithMap,\n rotateWithMap,\n this.transform,\n pixelsToTiles);\n\n labelToScreenMatrix = mat4.multiply([] as any, this.transform.labelPlaneMatrix, glMatrix);\n }\n\n // As long as this placement lives, we have to hold onto this bucket's\n // matching FeatureIndex/data for querying purposes\n this.retainedQueryData[symbolBucket.bucketInstanceId] = new RetainedQueryData(\n symbolBucket.bucketInstanceId,\n bucketFeatureIndex,\n symbolBucket.sourceLayerIndex,\n symbolBucket.index,\n tile.tileID\n );\n\n const parameters = {\n bucket: symbolBucket,\n layout,\n posMatrix,\n textLabelPlaneMatrix,\n labelToScreenMatrix,\n scale,\n textPixelRatio,\n holdingForFade: tile.holdingForFade(),\n collisionBoxArray,\n partiallyEvaluatedTextSize: symbolSize.evaluateSizeForZoom(symbolBucket.textSizeData, this.transform.zoom),\n collisionGroup: this.collisionGroups.get(symbolBucket.sourceID)\n };\n\n if (sortAcrossTiles) {\n for (const range of symbolBucket.sortKeyRanges) {\n const {sortKey, symbolInstanceStart, symbolInstanceEnd} = range;\n results.push({sortKey, symbolInstanceStart, symbolInstanceEnd, parameters});\n }\n } else {\n results.push({\n symbolInstanceStart: 0,\n symbolInstanceEnd: symbolBucket.symbolInstances.length,\n parameters\n });\n }\n }\n\n attemptAnchorPlacement(\n textAnchorOffset: TextAnchorOffset,\n textBox: SingleCollisionBox,\n width: number,\n height: number,\n textBoxScale: number,\n rotateWithMap: boolean,\n pitchWithMap: boolean,\n textPixelRatio: number,\n posMatrix: mat4,\n collisionGroup: CollisionGroup,\n textOverlapMode: OverlapMode,\n symbolInstance: SymbolInstance,\n bucket: SymbolBucket,\n orientation: number,\n iconBox?: SingleCollisionBox | null,\n getElevation?: (x: number, y: number) => number\n ): {\n shift: Point;\n placedGlyphBoxes: {\n box: Array;\n offscreen: boolean;\n };\n } {\n\n const anchor = TextAnchorEnum[textAnchorOffset.textAnchor] as TextAnchor;\n const textOffset = [textAnchorOffset.textOffset0, textAnchorOffset.textOffset1] as [number, number];\n const shift = calculateVariableLayoutShift(anchor, width, height, textOffset, textBoxScale);\n\n const placedGlyphBoxes = this.collisionIndex.placeCollisionBox(\n shiftVariableCollisionBox(\n textBox, shift.x, shift.y,\n rotateWithMap, pitchWithMap, this.transform.angle),\n textOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation);\n\n if (iconBox) {\n const placedIconBoxes = this.collisionIndex.placeCollisionBox(\n shiftVariableCollisionBox(\n iconBox, shift.x, shift.y,\n rotateWithMap, pitchWithMap, this.transform.angle),\n textOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation);\n if (placedIconBoxes.box.length === 0) return;\n }\n\n if (placedGlyphBoxes.box.length > 0) {\n let prevAnchor;\n // If this label was placed in the previous placement, record the anchor position\n // to allow us to animate the transition\n if (this.prevPlacement &&\n this.prevPlacement.variableOffsets[symbolInstance.crossTileID] &&\n this.prevPlacement.placements[symbolInstance.crossTileID] &&\n this.prevPlacement.placements[symbolInstance.crossTileID].text) {\n prevAnchor = this.prevPlacement.variableOffsets[symbolInstance.crossTileID].anchor;\n }\n if (symbolInstance.crossTileID === 0) throw new Error('symbolInstance.crossTileID can\\'t be 0');\n this.variableOffsets[symbolInstance.crossTileID] = {\n textOffset,\n width,\n height,\n anchor,\n textBoxScale,\n prevAnchor\n };\n this.markUsedJustification(bucket, anchor, symbolInstance, orientation);\n\n if (bucket.allowVerticalPlacement) {\n this.markUsedOrientation(bucket, orientation, symbolInstance);\n this.placedOrientations[symbolInstance.crossTileID] = orientation;\n }\n\n return {shift, placedGlyphBoxes};\n }\n }\n\n placeLayerBucketPart(bucketPart: BucketPart, seenCrossTileIDs: {\n [k in string | number]: boolean;\n }, showCollisionBoxes: boolean) {\n\n const {\n bucket,\n layout,\n posMatrix,\n textLabelPlaneMatrix,\n labelToScreenMatrix,\n textPixelRatio,\n holdingForFade,\n collisionBoxArray,\n partiallyEvaluatedTextSize,\n collisionGroup\n } = bucketPart.parameters;\n\n const textOptional = layout.get('text-optional');\n const iconOptional = layout.get('icon-optional');\n const textOverlapMode = getOverlapMode(layout, 'text-overlap', 'text-allow-overlap');\n const textAlwaysOverlap = textOverlapMode === 'always';\n const iconOverlapMode = getOverlapMode(layout, 'icon-overlap', 'icon-allow-overlap');\n const iconAlwaysOverlap = iconOverlapMode === 'always';\n const rotateWithMap = layout.get('text-rotation-alignment') === 'map';\n const pitchWithMap = layout.get('text-pitch-alignment') === 'map';\n const hasIconTextFit = layout.get('icon-text-fit') !== 'none';\n const zOrderByViewportY = layout.get('symbol-z-order') === 'viewport-y';\n\n // This logic is similar to the \"defaultOpacityState\" logic below in updateBucketOpacities\n // If we know a symbol is always supposed to show, force it to be marked visible even if\n // it wasn't placed into the collision index (because some or all of it was outside the range\n // of the collision grid).\n // There is a subtle edge case here we're accepting:\n // Symbol A has text-allow-overlap: true, icon-allow-overlap: true, icon-optional: false\n // A's icon is outside the grid, so doesn't get placed\n // A's text would be inside grid, but doesn't get placed because of icon-optional: false\n // We still show A because of the allow-overlap settings.\n // Symbol B has allow-overlap: false, and gets placed where A's text would be\n // On panning in, there is a short period when Symbol B and Symbol A will overlap\n // This is the reverse of our normal policy of \"fade in on pan\", but should look like any other\n // collision and hopefully not be too noticeable.\n // See https://github.com/mapbox/mapbox-gl-js/issues/7172\n const alwaysShowText = textAlwaysOverlap && (iconAlwaysOverlap || !bucket.hasIconData() || iconOptional);\n const alwaysShowIcon = iconAlwaysOverlap && (textAlwaysOverlap || !bucket.hasTextData() || textOptional);\n\n if (!bucket.collisionArrays && collisionBoxArray) {\n bucket.deserializeCollisionBoxes(collisionBoxArray);\n }\n\n const tileID = this.retainedQueryData[bucket.bucketInstanceId].tileID;\n const getElevation = this.terrain ? (x: number, y: number) => this.terrain.getElevation(tileID, x, y) : null;\n\n const placeSymbol = (symbolInstance: SymbolInstance, collisionArrays: CollisionArrays) => {\n if (seenCrossTileIDs[symbolInstance.crossTileID]) return;\n if (holdingForFade) {\n // Mark all symbols from this tile as \"not placed\", but don't add to seenCrossTileIDs, because we don't\n // know yet if we have a duplicate in a parent tile that _should_ be placed.\n this.placements[symbolInstance.crossTileID] = new JointPlacement(false, false, false);\n return;\n }\n\n let placeText = false;\n let placeIcon = false;\n let offscreen = true;\n let shift = null;\n\n let placed = {box: null, offscreen: null};\n let placedVerticalText = {box: null, offscreen: null};\n\n let placedGlyphBoxes = null;\n let placedGlyphCircles = null;\n let placedIconBoxes = null;\n let textFeatureIndex = 0;\n let verticalTextFeatureIndex = 0;\n let iconFeatureIndex = 0;\n\n if (collisionArrays.textFeatureIndex) {\n textFeatureIndex = collisionArrays.textFeatureIndex;\n } else if (symbolInstance.useRuntimeCollisionCircles) {\n textFeatureIndex = symbolInstance.featureIndex;\n }\n if (collisionArrays.verticalTextFeatureIndex) {\n verticalTextFeatureIndex = collisionArrays.verticalTextFeatureIndex;\n }\n\n const textBox = collisionArrays.textBox;\n if (textBox) {\n\n const updatePreviousOrientationIfNotPlaced = (isPlaced) => {\n let previousOrientation = WritingMode.horizontal;\n if (bucket.allowVerticalPlacement && !isPlaced && this.prevPlacement) {\n const prevPlacedOrientation = this.prevPlacement.placedOrientations[symbolInstance.crossTileID];\n if (prevPlacedOrientation) {\n this.placedOrientations[symbolInstance.crossTileID] = prevPlacedOrientation;\n previousOrientation = prevPlacedOrientation;\n this.markUsedOrientation(bucket, previousOrientation, symbolInstance);\n }\n }\n return previousOrientation;\n };\n\n const placeTextForPlacementModes = (placeHorizontalFn, placeVerticalFn) => {\n if (bucket.allowVerticalPlacement && symbolInstance.numVerticalGlyphVertices > 0 && collisionArrays.verticalTextBox) {\n for (const placementMode of bucket.writingModes) {\n if (placementMode === WritingMode.vertical) {\n placed = placeVerticalFn();\n placedVerticalText = placed;\n } else {\n placed = placeHorizontalFn();\n }\n if (placed && placed.box && placed.box.length) break;\n }\n } else {\n placed = placeHorizontalFn();\n }\n };\n\n const textAnchorOffsetStart = symbolInstance.textAnchorOffsetStartIndex;\n const textAnchorOffsetEnd = symbolInstance.textAnchorOffsetEndIndex;\n\n // If start+end indices match, text-variable-anchor is not in play.\n if (textAnchorOffsetEnd === textAnchorOffsetStart) {\n const placeBox = (collisionTextBox, orientation) => {\n const placedFeature = this.collisionIndex.placeCollisionBox(\n collisionTextBox,\n textOverlapMode,\n textPixelRatio,\n posMatrix,\n collisionGroup.predicate,\n getElevation\n );\n if (placedFeature && placedFeature.box && placedFeature.box.length) {\n this.markUsedOrientation(bucket, orientation, symbolInstance);\n this.placedOrientations[symbolInstance.crossTileID] = orientation;\n }\n return placedFeature;\n };\n\n const placeHorizontal = () => {\n return placeBox(textBox, WritingMode.horizontal);\n };\n\n const placeVertical = () => {\n const verticalTextBox = collisionArrays.verticalTextBox;\n if (bucket.allowVerticalPlacement && symbolInstance.numVerticalGlyphVertices > 0 && verticalTextBox) {\n return placeBox(verticalTextBox, WritingMode.vertical);\n }\n return {box: null, offscreen: null};\n };\n\n placeTextForPlacementModes(placeHorizontal, placeVertical);\n updatePreviousOrientationIfNotPlaced(placed && placed.box && placed.box.length);\n\n } else {\n // If this symbol was in the last placement, prefer placement using same anchor, if it's still available\n let prevAnchor = TextAnchorEnum[this.prevPlacement?.variableOffsets[symbolInstance.crossTileID]?.anchor];\n\n const placeBoxForVariableAnchors = (collisionTextBox, collisionIconBox, orientation) => {\n const width = collisionTextBox.x2 - collisionTextBox.x1;\n const height = collisionTextBox.y2 - collisionTextBox.y1;\n const textBoxScale = symbolInstance.textBoxScale;\n const variableIconBox = hasIconTextFit && (iconOverlapMode === 'never') ? collisionIconBox : null;\n\n let placedBox: {\n box: Array;\n offscreen: boolean;\n } = {box: [], offscreen: false};\n let placementPasses = (textOverlapMode === 'never') ? 1 : 2;\n let overlapMode: OverlapMode = 'never';\n\n if (prevAnchor) {\n placementPasses++;\n }\n\n for (let pass = 0; pass < placementPasses; pass++) {\n for (let i = textAnchorOffsetStart; i < textAnchorOffsetEnd; i++) {\n const textAnchorOffset = bucket.textAnchorOffsets.get(i);\n\n if (prevAnchor && textAnchorOffset.textAnchor !== prevAnchor) {\n continue;\n }\n\n const result = this.attemptAnchorPlacement(\n textAnchorOffset, collisionTextBox, width, height,\n textBoxScale, rotateWithMap, pitchWithMap, textPixelRatio, posMatrix,\n collisionGroup, overlapMode, symbolInstance, bucket, orientation, variableIconBox, getElevation);\n\n if (result) {\n placedBox = result.placedGlyphBoxes;\n if (placedBox && placedBox.box && placedBox.box.length) {\n placeText = true;\n shift = result.shift;\n return placedBox;\n }\n }\n }\n\n if (prevAnchor) {\n prevAnchor = null;\n } else {\n overlapMode = textOverlapMode;\n }\n }\n\n return placedBox;\n };\n\n const placeHorizontal = () => {\n return placeBoxForVariableAnchors(textBox, collisionArrays.iconBox, WritingMode.horizontal);\n };\n\n const placeVertical = () => {\n const verticalTextBox = collisionArrays.verticalTextBox;\n const wasPlaced = placed && placed.box && placed.box.length;\n if (bucket.allowVerticalPlacement && !wasPlaced && symbolInstance.numVerticalGlyphVertices > 0 && verticalTextBox) {\n return placeBoxForVariableAnchors(verticalTextBox, collisionArrays.verticalIconBox, WritingMode.vertical);\n }\n return {box: null, offscreen: null};\n };\n\n placeTextForPlacementModes(placeHorizontal, placeVertical);\n\n if (placed) {\n placeText = placed.box;\n offscreen = placed.offscreen;\n }\n\n const prevOrientation = updatePreviousOrientationIfNotPlaced(placed && placed.box);\n\n // If we didn't get placed, we still need to copy our position from the last placement for\n // fade animations\n if (!placeText && this.prevPlacement) {\n const prevOffset = this.prevPlacement.variableOffsets[symbolInstance.crossTileID];\n if (prevOffset) {\n this.variableOffsets[symbolInstance.crossTileID] = prevOffset;\n this.markUsedJustification(bucket, prevOffset.anchor, symbolInstance, prevOrientation);\n }\n }\n\n }\n }\n\n placedGlyphBoxes = placed;\n placeText = placedGlyphBoxes && placedGlyphBoxes.box && placedGlyphBoxes.box.length > 0;\n\n offscreen = placedGlyphBoxes && placedGlyphBoxes.offscreen;\n\n if (symbolInstance.useRuntimeCollisionCircles) {\n const placedSymbol = bucket.text.placedSymbolArray.get(symbolInstance.centerJustifiedTextSymbolIndex);\n const fontSize = symbolSize.evaluateSizeForFeature(bucket.textSizeData, partiallyEvaluatedTextSize, placedSymbol);\n\n const textPixelPadding = layout.get('text-padding');\n const circlePixelDiameter = symbolInstance.collisionCircleDiameter;\n\n placedGlyphCircles = this.collisionIndex.placeCollisionCircles(\n textOverlapMode,\n placedSymbol,\n bucket.lineVertexArray,\n bucket.glyphOffsetArray,\n fontSize,\n posMatrix,\n textLabelPlaneMatrix,\n labelToScreenMatrix,\n showCollisionBoxes,\n pitchWithMap,\n collisionGroup.predicate,\n circlePixelDiameter,\n textPixelPadding,\n getElevation\n );\n\n if (placedGlyphCircles.circles.length && placedGlyphCircles.collisionDetected && !showCollisionBoxes) {\n warnOnce('Collisions detected, but collision boxes are not shown');\n }\n\n // If text-overlap is set to 'always', force \"placedCircles\" to true\n // In theory there should always be at least one circle placed\n // in this case, but for now quirks in text-anchor\n // and text-offset may prevent that from being true.\n placeText = textAlwaysOverlap || (placedGlyphCircles.circles.length > 0 && !placedGlyphCircles.collisionDetected);\n offscreen = offscreen && placedGlyphCircles.offscreen;\n }\n\n if (collisionArrays.iconFeatureIndex) {\n iconFeatureIndex = collisionArrays.iconFeatureIndex;\n }\n\n if (collisionArrays.iconBox) {\n const placeIconFeature = iconBox => {\n const shiftedIconBox = hasIconTextFit && shift ?\n shiftVariableCollisionBox(\n iconBox, shift.x, shift.y,\n rotateWithMap, pitchWithMap, this.transform.angle) :\n iconBox;\n return this.collisionIndex.placeCollisionBox(shiftedIconBox,\n iconOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation);\n };\n\n if (placedVerticalText && placedVerticalText.box && placedVerticalText.box.length && collisionArrays.verticalIconBox) {\n placedIconBoxes = placeIconFeature(collisionArrays.verticalIconBox);\n placeIcon = placedIconBoxes.box.length > 0;\n } else {\n placedIconBoxes = placeIconFeature(collisionArrays.iconBox);\n placeIcon = placedIconBoxes.box.length > 0;\n }\n offscreen = offscreen && placedIconBoxes.offscreen;\n }\n\n const iconWithoutText = textOptional ||\n (symbolInstance.numHorizontalGlyphVertices === 0 && symbolInstance.numVerticalGlyphVertices === 0);\n const textWithoutIcon = iconOptional || symbolInstance.numIconVertices === 0;\n\n // Combine the scales for icons and text.\n if (!iconWithoutText && !textWithoutIcon) {\n placeIcon = placeText = placeIcon && placeText;\n } else if (!textWithoutIcon) {\n placeText = placeIcon && placeText;\n } else if (!iconWithoutText) {\n placeIcon = placeIcon && placeText;\n }\n\n if (placeText && placedGlyphBoxes && placedGlyphBoxes.box) {\n if (placedVerticalText && placedVerticalText.box && verticalTextFeatureIndex) {\n this.collisionIndex.insertCollisionBox(\n placedGlyphBoxes.box,\n textOverlapMode,\n layout.get('text-ignore-placement'),\n bucket.bucketInstanceId,\n verticalTextFeatureIndex,\n collisionGroup.ID);\n } else {\n this.collisionIndex.insertCollisionBox(\n placedGlyphBoxes.box,\n textOverlapMode,\n layout.get('text-ignore-placement'),\n bucket.bucketInstanceId,\n textFeatureIndex,\n collisionGroup.ID);\n }\n\n }\n if (placeIcon && placedIconBoxes) {\n this.collisionIndex.insertCollisionBox(\n placedIconBoxes.box,\n iconOverlapMode,\n layout.get('icon-ignore-placement'),\n bucket.bucketInstanceId,\n iconFeatureIndex,\n collisionGroup.ID);\n }\n if (placedGlyphCircles) {\n if (placeText) {\n this.collisionIndex.insertCollisionCircles(\n placedGlyphCircles.circles,\n textOverlapMode,\n layout.get('text-ignore-placement'),\n bucket.bucketInstanceId,\n textFeatureIndex,\n collisionGroup.ID);\n }\n\n if (showCollisionBoxes) {\n const id = bucket.bucketInstanceId;\n let circleArray = this.collisionCircleArrays[id];\n\n // Group collision circles together by bucket. Circles can't be pushed forward for rendering yet as the symbol placement\n // for a bucket is not guaranteed to be complete before the commit-function has been called\n if (circleArray === undefined)\n circleArray = this.collisionCircleArrays[id] = new CollisionCircleArray();\n\n for (let i = 0; i < placedGlyphCircles.circles.length; i += 4) {\n circleArray.circles.push(placedGlyphCircles.circles[i + 0]); // x\n circleArray.circles.push(placedGlyphCircles.circles[i + 1]); // y\n circleArray.circles.push(placedGlyphCircles.circles[i + 2]); // radius\n circleArray.circles.push(placedGlyphCircles.collisionDetected ? 1 : 0); // collisionDetected-flag\n }\n }\n }\n\n if (symbolInstance.crossTileID === 0) throw new Error('symbolInstance.crossTileID can\\'t be 0');\n if (bucket.bucketInstanceId === 0) throw new Error('bucket.bucketInstanceId can\\'t be 0');\n\n this.placements[symbolInstance.crossTileID] = new JointPlacement(placeText || alwaysShowText, placeIcon || alwaysShowIcon, offscreen || bucket.justReloaded);\n seenCrossTileIDs[symbolInstance.crossTileID] = true;\n };\n\n if (zOrderByViewportY) {\n if (bucketPart.symbolInstanceStart !== 0) throw new Error('bucket.bucketInstanceId should be 0');\n const symbolIndexes = bucket.getSortedSymbolIndexes(this.transform.angle);\n for (let i = symbolIndexes.length - 1; i >= 0; --i) {\n const symbolIndex = symbolIndexes[i];\n placeSymbol(bucket.symbolInstances.get(symbolIndex), bucket.collisionArrays[symbolIndex]);\n }\n } else {\n for (let i = bucketPart.symbolInstanceStart; i < bucketPart.symbolInstanceEnd; i++) {\n placeSymbol(bucket.symbolInstances.get(i), bucket.collisionArrays[i]);\n }\n }\n\n if (showCollisionBoxes && bucket.bucketInstanceId in this.collisionCircleArrays) {\n const circleArray = this.collisionCircleArrays[bucket.bucketInstanceId];\n\n // Store viewport and inverse projection matrices per bucket\n mat4.invert(circleArray.invProjMatrix, posMatrix);\n circleArray.viewportMatrix = this.collisionIndex.getViewportMatrix();\n }\n\n bucket.justReloaded = false;\n }\n\n markUsedJustification(bucket: SymbolBucket, placedAnchor: TextAnchor, symbolInstance: SymbolInstance, orientation: number) {\n const justifications = {\n 'left': symbolInstance.leftJustifiedTextSymbolIndex,\n 'center': symbolInstance.centerJustifiedTextSymbolIndex,\n 'right': symbolInstance.rightJustifiedTextSymbolIndex\n };\n\n let autoIndex;\n if (orientation === WritingMode.vertical) {\n autoIndex = symbolInstance.verticalPlacedTextSymbolIndex;\n } else {\n autoIndex = justifications[getAnchorJustification(placedAnchor)];\n }\n\n const indexes = [\n symbolInstance.leftJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.rightJustifiedTextSymbolIndex,\n symbolInstance.verticalPlacedTextSymbolIndex\n ];\n\n for (const index of indexes) {\n if (index >= 0) {\n if (autoIndex >= 0 && index !== autoIndex) {\n // There are multiple justifications and this one isn't it: shift offscreen\n bucket.text.placedSymbolArray.get(index).crossTileID = 0;\n } else {\n // Either this is the chosen justification or the justification is hardwired: use this one\n bucket.text.placedSymbolArray.get(index).crossTileID = symbolInstance.crossTileID;\n }\n }\n }\n }\n\n markUsedOrientation(bucket: SymbolBucket, orientation: number, symbolInstance: SymbolInstance) {\n const horizontal = (orientation === WritingMode.horizontal || orientation === WritingMode.horizontalOnly) ? orientation : 0;\n const vertical = orientation === WritingMode.vertical ? orientation : 0;\n\n const horizontalIndexes = [\n symbolInstance.leftJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.rightJustifiedTextSymbolIndex\n ];\n\n for (const index of horizontalIndexes) {\n bucket.text.placedSymbolArray.get(index).placedOrientation = horizontal;\n }\n\n if (symbolInstance.verticalPlacedTextSymbolIndex) {\n bucket.text.placedSymbolArray.get(symbolInstance.verticalPlacedTextSymbolIndex).placedOrientation = vertical;\n }\n }\n\n commit(now: number): void {\n this.commitTime = now;\n this.zoomAtLastRecencyCheck = this.transform.zoom;\n\n const prevPlacement = this.prevPlacement;\n let placementChanged = false;\n\n this.prevZoomAdjustment = prevPlacement ? prevPlacement.zoomAdjustment(this.transform.zoom) : 0;\n const increment = prevPlacement ? prevPlacement.symbolFadeChange(now) : 1;\n\n const prevOpacities = prevPlacement ? prevPlacement.opacities : {};\n const prevOffsets = prevPlacement ? prevPlacement.variableOffsets : {};\n const prevOrientations = prevPlacement ? prevPlacement.placedOrientations : {};\n\n // add the opacities from the current placement, and copy their current values from the previous placement\n for (const crossTileID in this.placements) {\n const jointPlacement = this.placements[crossTileID];\n const prevOpacity = prevOpacities[crossTileID];\n if (prevOpacity) {\n this.opacities[crossTileID] = new JointOpacityState(prevOpacity, increment, jointPlacement.text, jointPlacement.icon);\n placementChanged = placementChanged ||\n jointPlacement.text !== prevOpacity.text.placed ||\n jointPlacement.icon !== prevOpacity.icon.placed;\n } else {\n this.opacities[crossTileID] = new JointOpacityState(null, increment, jointPlacement.text, jointPlacement.icon, jointPlacement.skipFade);\n placementChanged = placementChanged || jointPlacement.text || jointPlacement.icon;\n }\n }\n\n // copy and update values from the previous placement that aren't in the current placement but haven't finished fading\n for (const crossTileID in prevOpacities) {\n const prevOpacity = prevOpacities[crossTileID];\n if (!this.opacities[crossTileID]) {\n const jointOpacity = new JointOpacityState(prevOpacity, increment, false, false);\n if (!jointOpacity.isHidden()) {\n this.opacities[crossTileID] = jointOpacity;\n placementChanged = placementChanged || prevOpacity.text.placed || prevOpacity.icon.placed;\n }\n }\n }\n for (const crossTileID in prevOffsets) {\n if (!this.variableOffsets[crossTileID] && this.opacities[crossTileID] && !this.opacities[crossTileID].isHidden()) {\n this.variableOffsets[crossTileID] = prevOffsets[crossTileID];\n }\n }\n\n for (const crossTileID in prevOrientations) {\n if (!this.placedOrientations[crossTileID] && this.opacities[crossTileID] && !this.opacities[crossTileID].isHidden()) {\n this.placedOrientations[crossTileID] = prevOrientations[crossTileID];\n }\n }\n\n // this.lastPlacementChangeTime is the time of the last commit() that\n // resulted in a placement change -- in other words, the start time of\n // the last symbol fade animation\n if (prevPlacement && prevPlacement.lastPlacementChangeTime === undefined) {\n throw new Error('Last placement time for previous placement is not defined');\n }\n if (placementChanged) {\n this.lastPlacementChangeTime = now;\n } else if (typeof this.lastPlacementChangeTime !== 'number') {\n this.lastPlacementChangeTime = prevPlacement ? prevPlacement.lastPlacementChangeTime : now;\n }\n }\n\n updateLayerOpacities(styleLayer: StyleLayer, tiles: Array) {\n const seenCrossTileIDs = {};\n for (const tile of tiles) {\n const symbolBucket = tile.getBucket(styleLayer) as SymbolBucket;\n if (symbolBucket && tile.latestFeatureIndex && styleLayer.id === symbolBucket.layerIds[0]) {\n this.updateBucketOpacities(symbolBucket, seenCrossTileIDs, tile.collisionBoxArray);\n }\n }\n }\n\n updateBucketOpacities(bucket: SymbolBucket, seenCrossTileIDs: {\n [k in string | number]: boolean;\n }, collisionBoxArray?: CollisionBoxArray | null) {\n if (bucket.hasTextData()) {\n bucket.text.opacityVertexArray.clear();\n bucket.text.hasVisibleVertices = false;\n }\n if (bucket.hasIconData()) {\n bucket.icon.opacityVertexArray.clear();\n bucket.icon.hasVisibleVertices = false;\n }\n if (bucket.hasIconCollisionBoxData()) bucket.iconCollisionBox.collisionVertexArray.clear();\n if (bucket.hasTextCollisionBoxData()) bucket.textCollisionBox.collisionVertexArray.clear();\n\n const layer = bucket.layers[0];\n const layout = layer.layout;\n const duplicateOpacityState = new JointOpacityState(null, 0, false, false, true);\n const textAllowOverlap = layout.get('text-allow-overlap');\n const iconAllowOverlap = layout.get('icon-allow-overlap');\n const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset');\n const rotateWithMap = layout.get('text-rotation-alignment') === 'map';\n const pitchWithMap = layout.get('text-pitch-alignment') === 'map';\n const hasIconTextFit = layout.get('icon-text-fit') !== 'none';\n // If allow-overlap is true, we can show symbols before placement runs on them\n // But we have to wait for placement if we potentially depend on a paired icon/text\n // with allow-overlap: false.\n // See https://github.com/mapbox/mapbox-gl-js/issues/7032\n const defaultOpacityState = new JointOpacityState(null, 0,\n textAllowOverlap && (iconAllowOverlap || !bucket.hasIconData() || layout.get('icon-optional')),\n iconAllowOverlap && (textAllowOverlap || !bucket.hasTextData() || layout.get('text-optional')),\n true);\n\n if (!bucket.collisionArrays && collisionBoxArray && ((bucket.hasIconCollisionBoxData() || bucket.hasTextCollisionBoxData()))) {\n bucket.deserializeCollisionBoxes(collisionBoxArray);\n }\n\n const addOpacities = (iconOrText, numVertices: number, opacity: number) => {\n for (let i = 0; i < numVertices / 4; i++) {\n iconOrText.opacityVertexArray.emplaceBack(opacity);\n }\n iconOrText.hasVisibleVertices = iconOrText.hasVisibleVertices || (opacity !== PACKED_HIDDEN_OPACITY);\n };\n\n for (let s = 0; s < bucket.symbolInstances.length; s++) {\n const symbolInstance = bucket.symbolInstances.get(s);\n const {\n numHorizontalGlyphVertices,\n numVerticalGlyphVertices,\n crossTileID\n } = symbolInstance;\n\n const isDuplicate = seenCrossTileIDs[crossTileID];\n\n let opacityState = this.opacities[crossTileID];\n if (isDuplicate) {\n opacityState = duplicateOpacityState;\n } else if (!opacityState) {\n opacityState = defaultOpacityState;\n // store the state so that future placements use it as a starting point\n this.opacities[crossTileID] = opacityState;\n }\n\n seenCrossTileIDs[crossTileID] = true;\n\n const hasText = numHorizontalGlyphVertices > 0 || numVerticalGlyphVertices > 0;\n const hasIcon = symbolInstance.numIconVertices > 0;\n\n const placedOrientation = this.placedOrientations[symbolInstance.crossTileID];\n const horizontalHidden = placedOrientation === WritingMode.vertical;\n const verticalHidden = placedOrientation === WritingMode.horizontal || placedOrientation === WritingMode.horizontalOnly;\n\n if (hasText) {\n const packedOpacity = packOpacity(opacityState.text);\n // Vertical text fades in/out on collision the same way as corresponding\n // horizontal text. Switch between vertical/horizontal should be instantaneous\n const horizontalOpacity = horizontalHidden ? PACKED_HIDDEN_OPACITY : packedOpacity;\n addOpacities(bucket.text, numHorizontalGlyphVertices, horizontalOpacity);\n const verticalOpacity = verticalHidden ? PACKED_HIDDEN_OPACITY : packedOpacity;\n addOpacities(bucket.text, numVerticalGlyphVertices, verticalOpacity);\n\n // If this label is completely faded, mark it so that we don't have to calculate\n // its position at render time. If this layer has variable placement, shift the various\n // symbol instances appropriately so that symbols from buckets that have yet to be placed\n // offset appropriately.\n const symbolHidden = opacityState.text.isHidden();\n [\n symbolInstance.rightJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.leftJustifiedTextSymbolIndex\n ].forEach(index => {\n if (index >= 0) {\n bucket.text.placedSymbolArray.get(index).hidden = symbolHidden || horizontalHidden ? 1 : 0;\n }\n });\n\n if (symbolInstance.verticalPlacedTextSymbolIndex >= 0) {\n bucket.text.placedSymbolArray.get(symbolInstance.verticalPlacedTextSymbolIndex).hidden = symbolHidden || verticalHidden ? 1 : 0;\n }\n\n const prevOffset = this.variableOffsets[symbolInstance.crossTileID];\n if (prevOffset) {\n this.markUsedJustification(bucket, prevOffset.anchor, symbolInstance, placedOrientation);\n }\n\n const prevOrientation = this.placedOrientations[symbolInstance.crossTileID];\n if (prevOrientation) {\n this.markUsedJustification(bucket, 'left', symbolInstance, prevOrientation);\n this.markUsedOrientation(bucket, prevOrientation, symbolInstance);\n }\n }\n\n if (hasIcon) {\n const packedOpacity = packOpacity(opacityState.icon);\n\n const useHorizontal = !(hasIconTextFit && symbolInstance.verticalPlacedIconSymbolIndex && horizontalHidden);\n\n if (symbolInstance.placedIconSymbolIndex >= 0) {\n const horizontalOpacity = useHorizontal ? packedOpacity : PACKED_HIDDEN_OPACITY;\n addOpacities(bucket.icon, symbolInstance.numIconVertices, horizontalOpacity);\n bucket.icon.placedSymbolArray.get(symbolInstance.placedIconSymbolIndex).hidden =\n (opacityState.icon.isHidden() as any);\n }\n\n if (symbolInstance.verticalPlacedIconSymbolIndex >= 0) {\n const verticalOpacity = !useHorizontal ? packedOpacity : PACKED_HIDDEN_OPACITY;\n addOpacities(bucket.icon, symbolInstance.numVerticalIconVertices, verticalOpacity);\n bucket.icon.placedSymbolArray.get(symbolInstance.verticalPlacedIconSymbolIndex).hidden =\n (opacityState.icon.isHidden() as any);\n }\n }\n\n if (bucket.hasIconCollisionBoxData() || bucket.hasTextCollisionBoxData()) {\n const collisionArrays = bucket.collisionArrays[s];\n if (collisionArrays) {\n let shift = new Point(0, 0);\n if (collisionArrays.textBox || collisionArrays.verticalTextBox) {\n let used = true;\n if (hasVariablePlacement) {\n const variableOffset = this.variableOffsets[crossTileID];\n if (variableOffset) {\n // This will show either the currently placed position or the last\n // successfully placed position (so you can visualize what collision\n // just made the symbol disappear, and the most likely place for the\n // symbol to come back)\n shift = calculateVariableLayoutShift(variableOffset.anchor,\n variableOffset.width,\n variableOffset.height,\n variableOffset.textOffset,\n variableOffset.textBoxScale);\n if (rotateWithMap) {\n shift._rotate(pitchWithMap ? this.transform.angle : -this.transform.angle);\n }\n } else {\n // No offset -> this symbol hasn't been placed since coming on-screen\n // No single box is particularly meaningful and all of them would be too noisy\n // Use the center box just to show something's there, but mark it \"not used\"\n used = false;\n }\n }\n\n if (collisionArrays.textBox) {\n updateCollisionVertices(bucket.textCollisionBox.collisionVertexArray, opacityState.text.placed, !used || horizontalHidden, shift.x, shift.y);\n }\n if (collisionArrays.verticalTextBox) {\n updateCollisionVertices(bucket.textCollisionBox.collisionVertexArray, opacityState.text.placed, !used || verticalHidden, shift.x, shift.y);\n }\n }\n\n const verticalIconUsed = Boolean(!verticalHidden && collisionArrays.verticalIconBox);\n\n if (collisionArrays.iconBox) {\n updateCollisionVertices(bucket.iconCollisionBox.collisionVertexArray, opacityState.icon.placed, verticalIconUsed,\n hasIconTextFit ? shift.x : 0,\n hasIconTextFit ? shift.y : 0);\n }\n\n if (collisionArrays.verticalIconBox) {\n updateCollisionVertices(bucket.iconCollisionBox.collisionVertexArray, opacityState.icon.placed, !verticalIconUsed,\n hasIconTextFit ? shift.x : 0,\n hasIconTextFit ? shift.y : 0);\n }\n }\n }\n }\n\n bucket.sortFeatures(this.transform.angle);\n if (this.retainedQueryData[bucket.bucketInstanceId]) {\n this.retainedQueryData[bucket.bucketInstanceId].featureSortOrder = bucket.featureSortOrder;\n }\n\n if (bucket.hasTextData() && bucket.text.opacityVertexBuffer) {\n bucket.text.opacityVertexBuffer.updateData(bucket.text.opacityVertexArray);\n }\n if (bucket.hasIconData() && bucket.icon.opacityVertexBuffer) {\n bucket.icon.opacityVertexBuffer.updateData(bucket.icon.opacityVertexArray);\n }\n if (bucket.hasIconCollisionBoxData() && bucket.iconCollisionBox.collisionVertexBuffer) {\n bucket.iconCollisionBox.collisionVertexBuffer.updateData(bucket.iconCollisionBox.collisionVertexArray);\n }\n if (bucket.hasTextCollisionBoxData() && bucket.textCollisionBox.collisionVertexBuffer) {\n bucket.textCollisionBox.collisionVertexBuffer.updateData(bucket.textCollisionBox.collisionVertexArray);\n }\n\n if (bucket.text.opacityVertexArray.length !== bucket.text.layoutVertexArray.length / 4) throw new Error(`bucket.text.opacityVertexArray.length (= ${bucket.text.opacityVertexArray.length}) !== bucket.text.layoutVertexArray.length (= ${bucket.text.layoutVertexArray.length}) / 4`);\n if (bucket.icon.opacityVertexArray.length !== bucket.icon.layoutVertexArray.length / 4) throw new Error(`bucket.icon.opacityVertexArray.length (= ${bucket.icon.opacityVertexArray.length}) !== bucket.icon.layoutVertexArray.length (= ${bucket.icon.layoutVertexArray.length}) / 4`);\n\n // Push generated collision circles to the bucket for debug rendering\n if (bucket.bucketInstanceId in this.collisionCircleArrays) {\n const instance = this.collisionCircleArrays[bucket.bucketInstanceId];\n\n bucket.placementInvProjMatrix = instance.invProjMatrix;\n bucket.placementViewportMatrix = instance.viewportMatrix;\n bucket.collisionCircleArray = instance.circles;\n\n delete this.collisionCircleArrays[bucket.bucketInstanceId];\n }\n }\n\n symbolFadeChange(now: number) {\n return this.fadeDuration === 0 ?\n 1 :\n ((now - this.commitTime) / this.fadeDuration + this.prevZoomAdjustment);\n }\n\n zoomAdjustment(zoom: number) {\n // When zooming out quickly, labels can overlap each other. This\n // adjustment is used to reduce the interval between placement calculations\n // and to reduce the fade duration when zooming out quickly. Discovering the\n // collisions more quickly and fading them more quickly reduces the unwanted effect.\n return Math.max(0, (this.transform.zoom - zoom) / 1.5);\n }\n\n hasTransitions(now: number) {\n return this.stale ||\n now - this.lastPlacementChangeTime < this.fadeDuration;\n }\n\n stillRecent(now: number, zoom: number) {\n // The adjustment makes placement more frequent when zooming.\n // This condition applies the adjustment only after the map has\n // stopped zooming. This avoids adding extra jank while zooming.\n const durationAdjustment = this.zoomAtLastRecencyCheck === zoom ?\n (1 - this.zoomAdjustment(zoom)) :\n 1;\n this.zoomAtLastRecencyCheck = zoom;\n\n return this.commitTime + this.fadeDuration * durationAdjustment > now;\n }\n\n setStale() {\n this.stale = true;\n }\n}\n\nfunction updateCollisionVertices(collisionVertexArray: CollisionVertexArray, placed: boolean, notUsed: boolean | number, shiftX?: number, shiftY?: number) {\n collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0);\n collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0);\n collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0);\n collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0);\n}\n\n// All four vertices for a glyph will have the same opacity state\n// So we pack the opacity into a uint8, and then repeat it four times\n// to make a single uint32 that we can upload for each glyph in the\n// label.\nconst shift25 = Math.pow(2, 25);\nconst shift24 = Math.pow(2, 24);\nconst shift17 = Math.pow(2, 17);\nconst shift16 = Math.pow(2, 16);\nconst shift9 = Math.pow(2, 9);\nconst shift8 = Math.pow(2, 8);\nconst shift1 = Math.pow(2, 1);\nfunction packOpacity(opacityState: OpacityState): number {\n if (opacityState.opacity === 0 && !opacityState.placed) {\n return 0;\n } else if (opacityState.opacity === 1 && opacityState.placed) {\n return 4294967295;\n }\n const targetBit = opacityState.placed ? 1 : 0;\n const opacityBits = Math.floor(opacityState.opacity * 127);\n return opacityBits * shift25 + targetBit * shift24 +\n opacityBits * shift17 + targetBit * shift16 +\n opacityBits * shift9 + targetBit * shift8 +\n opacityBits * shift1 + targetBit;\n}\n\nconst PACKED_HIDDEN_OPACITY = 0;\n","import {Anchor} from './anchor';\n\nimport {getAnchors, getCenterAnchor} from './get_anchors';\nimport {clipLine} from './clip_line';\nimport {shapeText, shapeIcon, WritingMode, fitIconToText} from './shaping';\nimport {getGlyphQuads, getIconQuads} from './quads';\nimport {CollisionFeature} from './collision_feature';\nimport {warnOnce} from '../util/util';\nimport {\n allowsVerticalWritingMode,\n allowsLetterSpacing\n} from '../util/script_detection';\nimport {findPoleOfInaccessibility} from '../util/find_pole_of_inaccessibility';\nimport {classifyRings} from '../util/classify_rings';\nimport {EXTENT} from '../data/extent';\nimport {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {SIZE_PACK_FACTOR, MAX_PACKED_SIZE, MAX_GLYPH_ICON_SIZE} from './symbol_size';\nimport ONE_EM from './one_em';\nimport type {CanonicalTileID} from '../source/tile_id';\nimport type {Shaping, PositionedIcon, TextJustify} from './shaping';\nimport type {CollisionBoxArray, TextAnchorOffsetArray} from '../data/array_types.g';\nimport type {SymbolFeature} from '../data/bucket/symbol_bucket';\nimport type {StyleImage} from '../style/style_image';\nimport type {StyleGlyph} from '../style/style_glyph';\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\nimport type {ImagePosition} from '../render/image_atlas';\nimport type {GlyphPosition} from '../render/glyph_atlas';\nimport type {PossiblyEvaluatedPropertyValue} from '../style/properties';\n\nimport Point from '@mapbox/point-geometry';\nimport murmur3 from 'murmurhash-js';\nimport {getIconPadding, SymbolPadding} from '../style/style_layer/symbol_style_layer';\nimport {VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {getTextVariableAnchorOffset, evaluateVariableOffset, INVALID_TEXT_OFFSET, TextAnchor, TextAnchorEnum} from '../style/style_layer/variable_text_anchor';\n\n// The symbol layout process needs `text-size` evaluated at up to five different zoom levels, and\n// `icon-size` at up to three:\n//\n// 1. `text-size` at the zoom level of the bucket. Used to calculate a per-feature size for source `text-size`\n// expressions, and to calculate the box dimensions for icon-text-fit.\n// 2. `icon-size` at the zoom level of the bucket. Used to calculate a per-feature size for source `icon-size`\n// expressions.\n// 3. `text-size` and `icon-size` at the zoom level of the bucket, plus one. Used to calculate collision boxes.\n// 4. `text-size` at zoom level 18. Used for something line-symbol-placement-related.\n// 5. For composite `*-size` expressions: two zoom levels of curve stops that \"cover\" the zoom level of the\n// bucket. These go into a vertex buffer and are used by the shader to interpolate the size at render time.\n//\n// (1) and (2) are stored in `bucket.layers[0].layout`. The remainder are below.\n//\ntype Sizes = {\n layoutTextSize: PossiblyEvaluatedPropertyValue; // (3),\n layoutIconSize: PossiblyEvaluatedPropertyValue; // (3),\n textMaxSize: PossiblyEvaluatedPropertyValue; // (4),\n compositeTextSizes: [PossiblyEvaluatedPropertyValue, PossiblyEvaluatedPropertyValue]; // (5),\n compositeIconSizes: [PossiblyEvaluatedPropertyValue, PossiblyEvaluatedPropertyValue]; // (5)\n};\n\ntype ShapedTextOrientations = {\n vertical: Shaping | false;\n horizontal: Record;\n};\n\nexport function performSymbolLayout(args: {\n bucket: SymbolBucket;\n glyphMap: {\n [_: string]: {\n [x: number]: StyleGlyph;\n };\n };\n glyphPositions: {\n [_: string]: {\n [x: number]: GlyphPosition;\n };\n };\n imageMap: {[_: string]: StyleImage};\n imagePositions: {[_: string]: ImagePosition};\n showCollisionBoxes: boolean;\n canonical: CanonicalTileID;\n}) {\n args.bucket.createArrays();\n\n const tileSize = 512 * args.bucket.overscaling;\n args.bucket.tilePixelRatio = EXTENT / tileSize;\n args.bucket.compareText = {};\n args.bucket.iconsNeedLinear = false;\n\n const layer = args.bucket.layers[0];\n const layout = layer.layout;\n const unevaluatedLayoutValues = layer._unevaluatedLayout._values;\n\n const sizes: Sizes = {\n // Filled in below, if *SizeData.kind is 'composite'\n // compositeIconSizes: undefined,\n // compositeTextSizes: undefined,\n layoutIconSize: unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(args.bucket.zoom + 1), args.canonical),\n layoutTextSize: unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(args.bucket.zoom + 1), args.canonical),\n textMaxSize: unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(18))\n } as Sizes;\n\n if (args.bucket.textSizeData.kind === 'composite') {\n const {minZoom, maxZoom} = args.bucket.textSizeData;\n sizes.compositeTextSizes = [\n unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(minZoom), args.canonical),\n unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), args.canonical)\n ];\n }\n\n if (args.bucket.iconSizeData.kind === 'composite') {\n const {minZoom, maxZoom} = args.bucket.iconSizeData;\n sizes.compositeIconSizes = [\n unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(minZoom), args.canonical),\n unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), args.canonical)\n ];\n }\n\n const lineHeight = layout.get('text-line-height') * ONE_EM;\n const textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point';\n const keepUpright = layout.get('text-keep-upright');\n const textSize = layout.get('text-size');\n\n for (const feature of args.bucket.features) {\n const fontstack = layout.get('text-font').evaluate(feature, {}, args.canonical).join(',');\n const layoutTextSizeThisZoom = textSize.evaluate(feature, {}, args.canonical);\n const layoutTextSize = sizes.layoutTextSize.evaluate(feature, {}, args.canonical);\n const layoutIconSize = sizes.layoutIconSize.evaluate(feature, {}, args.canonical);\n\n const shapedTextOrientations: ShapedTextOrientations = {\n horizontal: {} as Record,\n vertical: undefined\n };\n const text = feature.text;\n let textOffset: [number, number] = [0, 0];\n if (text) {\n const unformattedText = text.toString();\n const spacing = layout.get('text-letter-spacing').evaluate(feature, {}, args.canonical) * ONE_EM;\n const spacingIfAllowed = allowsLetterSpacing(unformattedText) ? spacing : 0;\n\n const textAnchor = layout.get('text-anchor').evaluate(feature, {}, args.canonical);\n const variableAnchorOffset = getTextVariableAnchorOffset(layer, feature, args.canonical);\n\n if (!variableAnchorOffset) {\n const radialOffset = layout.get('text-radial-offset').evaluate(feature, {}, args.canonical);\n // Layers with variable anchors use the `text-radial-offset` property and the [x, y] offset vector\n // is calculated at placement time instead of layout time\n if (radialOffset) {\n // The style spec says don't use `text-offset` and `text-radial-offset` together\n // but doesn't actually specify what happens if you use both. We go with the radial offset.\n textOffset = evaluateVariableOffset(textAnchor, [radialOffset * ONE_EM, INVALID_TEXT_OFFSET]);\n } else {\n textOffset = (layout.get('text-offset').evaluate(feature, {}, args.canonical).map(t => t * ONE_EM) as [number, number]);\n }\n }\n\n let textJustify = textAlongLine ?\n 'center' :\n layout.get('text-justify').evaluate(feature, {}, args.canonical);\n\n const symbolPlacement = layout.get('symbol-placement');\n const maxWidth = symbolPlacement === 'point' ?\n layout.get('text-max-width').evaluate(feature, {}, args.canonical) * ONE_EM :\n 0;\n\n const addVerticalShapingForPointLabelIfNeeded = () => {\n if (args.bucket.allowVerticalPlacement && allowsVerticalWritingMode(unformattedText)) {\n // Vertical POI label placement is meant to be used for scripts that support vertical\n // writing mode, thus, default left justification is used. If Latin\n // scripts would need to be supported, this should take into account other justifications.\n shapedTextOrientations.vertical = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor,\n 'left', spacingIfAllowed, textOffset, WritingMode.vertical, true, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n }\n };\n\n // If this layer uses text-variable-anchor, generate shapings for all justification possibilities.\n if (!textAlongLine && variableAnchorOffset) {\n const justifications = new Set();\n\n if (textJustify === 'auto') {\n for (let i = 0; i < variableAnchorOffset.values.length; i += 2) {\n justifications.add(getAnchorJustification(variableAnchorOffset.values[i] as TextAnchor));\n }\n } else {\n justifications.add(textJustify);\n }\n\n let singleLine = false;\n for (const justification of justifications) {\n if (shapedTextOrientations.horizontal[justification]) continue;\n if (singleLine) {\n // If the shaping for the first justification was only a single line, we\n // can re-use it for the other justifications\n shapedTextOrientations.horizontal[justification] = shapedTextOrientations.horizontal[0];\n } else {\n // If using text-variable-anchor for the layer, we use a center anchor for all shapings and apply\n // the offsets for the anchor in the placement step.\n const shaping = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, 'center',\n justification, spacingIfAllowed, textOffset, WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n if (shaping) {\n shapedTextOrientations.horizontal[justification] = shaping;\n singleLine = shaping.positionedLines.length === 1;\n }\n }\n }\n\n addVerticalShapingForPointLabelIfNeeded();\n } else {\n if (textJustify === 'auto') {\n textJustify = getAnchorJustification(textAnchor);\n }\n\n // Horizontal point or line label.\n const shaping = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify, spacingIfAllowed,\n textOffset, WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n if (shaping) shapedTextOrientations.horizontal[textJustify] = shaping;\n\n // Vertical point label (if allowVerticalPlacement is enabled).\n addVerticalShapingForPointLabelIfNeeded();\n\n // Verticalized line label.\n if (allowsVerticalWritingMode(unformattedText) && textAlongLine && keepUpright) {\n shapedTextOrientations.vertical = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify,\n spacingIfAllowed, textOffset, WritingMode.vertical, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n }\n }\n }\n\n let shapedIcon;\n let isSDFIcon = false;\n if (feature.icon && feature.icon.name) {\n const image = args.imageMap[feature.icon.name];\n if (image) {\n shapedIcon = shapeIcon(\n args.imagePositions[feature.icon.name],\n layout.get('icon-offset').evaluate(feature, {}, args.canonical),\n layout.get('icon-anchor').evaluate(feature, {}, args.canonical));\n // null/undefined SDF property treated same as default (false)\n isSDFIcon = !!image.sdf;\n if (args.bucket.sdfIcons === undefined) {\n args.bucket.sdfIcons = isSDFIcon;\n } else if (args.bucket.sdfIcons !== isSDFIcon) {\n warnOnce('Style sheet warning: Cannot mix SDF and non-SDF icons in one buffer');\n }\n if (image.pixelRatio !== args.bucket.pixelRatio) {\n args.bucket.iconsNeedLinear = true;\n } else if (layout.get('icon-rotate').constantOr(1) !== 0) {\n args.bucket.iconsNeedLinear = true;\n }\n }\n }\n\n const shapedText = getDefaultHorizontalShaping(shapedTextOrientations.horizontal) || shapedTextOrientations.vertical;\n args.bucket.iconsInText = shapedText ? shapedText.iconsInText : false;\n if (shapedText || shapedIcon) {\n addFeature(args.bucket, feature, shapedTextOrientations, shapedIcon, args.imageMap, sizes, layoutTextSize, layoutIconSize, textOffset, isSDFIcon, args.canonical);\n }\n }\n\n if (args.showCollisionBoxes) {\n args.bucket.generateCollisionDebugBuffers();\n }\n}\n\n// Choose the justification that matches the direction of the TextAnchor\nexport function getAnchorJustification(anchor: TextAnchor): TextJustify {\n switch (anchor) {\n case 'right':\n case 'top-right':\n case 'bottom-right':\n return 'right';\n case 'left':\n case 'top-left':\n case 'bottom-left':\n return 'left';\n }\n return 'center';\n}\n\n/**\n * Given a feature and its shaped text and icon data, add a 'symbol\n * instance' for each _possible_ placement of the symbol feature.\n * (At render timePlaceSymbols#place() selects which of these instances to\n * show or hide based on collisions with symbols in other layers.)\n */\nfunction addFeature(bucket: SymbolBucket,\n feature: SymbolFeature,\n shapedTextOrientations: ShapedTextOrientations,\n shapedIcon: PositionedIcon,\n imageMap: {[_: string]: StyleImage},\n sizes: Sizes,\n layoutTextSize: number,\n layoutIconSize: number,\n textOffset: [number, number],\n isSDFIcon: boolean, canonical: CanonicalTileID) {\n // To reduce the number of labels that jump around when zooming we need\n // to use a text-size value that is the same for all zoom levels.\n // bucket calculates text-size at a high zoom level so that all tiles can\n // use the same value when calculating anchor positions.\n let textMaxSize = sizes.textMaxSize.evaluate(feature, {});\n if (textMaxSize === undefined) {\n textMaxSize = layoutTextSize;\n }\n const layout = bucket.layers[0].layout;\n const iconOffset = layout.get('icon-offset').evaluate(feature, {}, canonical);\n const defaultHorizontalShaping = getDefaultHorizontalShaping(shapedTextOrientations.horizontal);\n const glyphSize = 24,\n fontScale = layoutTextSize / glyphSize,\n textBoxScale = bucket.tilePixelRatio * fontScale,\n textMaxBoxScale = bucket.tilePixelRatio * textMaxSize / glyphSize,\n iconBoxScale = bucket.tilePixelRatio * layoutIconSize,\n symbolMinDistance = bucket.tilePixelRatio * layout.get('symbol-spacing'),\n textPadding = layout.get('text-padding') * bucket.tilePixelRatio,\n iconPadding = getIconPadding(layout, feature, canonical, bucket.tilePixelRatio),\n textMaxAngle = layout.get('text-max-angle') / 180 * Math.PI,\n textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point',\n iconAlongLine = layout.get('icon-rotation-alignment') === 'map' && layout.get('symbol-placement') !== 'point',\n symbolPlacement = layout.get('symbol-placement'),\n textRepeatDistance = symbolMinDistance / 2;\n\n const iconTextFit = layout.get('icon-text-fit');\n let verticallyShapedIcon;\n // Adjust shaped icon size when icon-text-fit is used.\n if (shapedIcon && iconTextFit !== 'none') {\n if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) {\n verticallyShapedIcon = fitIconToText(shapedIcon, shapedTextOrientations.vertical, iconTextFit,\n layout.get('icon-text-fit-padding'), iconOffset, fontScale);\n }\n if (defaultHorizontalShaping) {\n shapedIcon = fitIconToText(shapedIcon, defaultHorizontalShaping, iconTextFit,\n layout.get('icon-text-fit-padding'), iconOffset, fontScale);\n }\n }\n\n const addSymbolAtAnchor = (line, anchor) => {\n if (anchor.x < 0 || anchor.x >= EXTENT || anchor.y < 0 || anchor.y >= EXTENT) {\n // Symbol layers are drawn across tile boundaries, We filter out symbols\n // outside our tile boundaries (which may be included in vector tile buffers)\n // to prevent double-drawing symbols.\n return;\n }\n\n addSymbol(bucket, anchor, line, shapedTextOrientations, shapedIcon, imageMap, verticallyShapedIcon, bucket.layers[0],\n bucket.collisionBoxArray, feature.index, feature.sourceLayerIndex, bucket.index,\n textBoxScale, [textPadding, textPadding, textPadding, textPadding], textAlongLine, textOffset,\n iconBoxScale, iconPadding, iconAlongLine, iconOffset,\n feature, sizes, isSDFIcon, canonical, layoutTextSize);\n };\n\n if (symbolPlacement === 'line') {\n for (const line of clipLine(feature.geometry, 0, 0, EXTENT, EXTENT)) {\n const anchors = getAnchors(\n line,\n symbolMinDistance,\n textMaxAngle,\n shapedTextOrientations.vertical || defaultHorizontalShaping,\n shapedIcon,\n glyphSize,\n textMaxBoxScale,\n bucket.overscaling,\n EXTENT\n );\n for (const anchor of anchors) {\n const shapedText = defaultHorizontalShaping;\n if (!shapedText || !anchorIsTooClose(bucket, shapedText.text, textRepeatDistance, anchor)) {\n addSymbolAtAnchor(line, anchor);\n }\n }\n }\n } else if (symbolPlacement === 'line-center') {\n // No clipping, multiple lines per feature are allowed\n // \"lines\" with only one point are ignored as in clipLines\n for (const line of feature.geometry) {\n if (line.length > 1) {\n const anchor = getCenterAnchor(\n line,\n textMaxAngle,\n shapedTextOrientations.vertical || defaultHorizontalShaping,\n shapedIcon,\n glyphSize,\n textMaxBoxScale);\n if (anchor) {\n addSymbolAtAnchor(line, anchor);\n }\n }\n }\n } else if (feature.type === 'Polygon') {\n for (const polygon of classifyRings(feature.geometry, 0)) {\n // 16 here represents 2 pixels\n const poi = findPoleOfInaccessibility(polygon, 16);\n addSymbolAtAnchor(polygon[0], new Anchor(poi.x, poi.y, 0));\n }\n } else if (feature.type === 'LineString') {\n // https://github.com/mapbox/mapbox-gl-js/issues/3808\n for (const line of feature.geometry) {\n addSymbolAtAnchor(line, new Anchor(line[0].x, line[0].y, 0));\n }\n } else if (feature.type === 'Point') {\n for (const points of feature.geometry) {\n for (const point of points) {\n addSymbolAtAnchor([point], new Anchor(point.x, point.y, 0));\n }\n }\n }\n}\n\nfunction addTextVariableAnchorOffsets(textAnchorOffsets: TextAnchorOffsetArray, variableAnchorOffset: VariableAnchorOffsetCollection): [number, number] {\n const startIndex = textAnchorOffsets.length;\n const values = variableAnchorOffset?.values;\n\n if (values?.length > 0) {\n for (let i = 0; i < values.length; i += 2) {\n const anchor = TextAnchorEnum[values[i] as TextAnchor];\n const offset = values[i + 1] as [number, number];\n\n textAnchorOffsets.emplaceBack(anchor, offset[0], offset[1]);\n }\n }\n\n return [startIndex, textAnchorOffsets.length];\n}\n\nfunction addTextVertices(bucket: SymbolBucket,\n anchor: Point,\n shapedText: Shaping,\n imageMap: {[_: string]: StyleImage},\n layer: SymbolStyleLayer,\n textAlongLine: boolean,\n feature: SymbolFeature,\n textOffset: [number, number],\n lineArray: {\n lineStartIndex: number;\n lineLength: number;\n },\n writingMode: WritingMode,\n placementTypes: Array<'vertical' | 'center' | 'left' | 'right'>,\n placedTextSymbolIndices: {[_: string]: number},\n placedIconIndex: number,\n sizes: Sizes,\n canonical: CanonicalTileID) {\n const glyphQuads = getGlyphQuads(anchor, shapedText, textOffset,\n layer, textAlongLine, feature, imageMap, bucket.allowVerticalPlacement);\n\n const sizeData = bucket.textSizeData;\n let textSizeData = null;\n\n if (sizeData.kind === 'source') {\n textSizeData = [\n SIZE_PACK_FACTOR * layer.layout.get('text-size').evaluate(feature, {})\n ];\n if (textSizeData[0] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"text-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"text-size\".`);\n }\n } else if (sizeData.kind === 'composite') {\n textSizeData = [\n SIZE_PACK_FACTOR * sizes.compositeTextSizes[0].evaluate(feature, {}, canonical),\n SIZE_PACK_FACTOR * sizes.compositeTextSizes[1].evaluate(feature, {}, canonical)\n ];\n if (textSizeData[0] > MAX_PACKED_SIZE || textSizeData[1] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"text-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"text-size\".`);\n }\n }\n\n bucket.addSymbols(\n bucket.text,\n glyphQuads,\n textSizeData,\n textOffset,\n textAlongLine,\n feature,\n writingMode,\n anchor,\n lineArray.lineStartIndex,\n lineArray.lineLength,\n placedIconIndex,\n canonical);\n\n // The placedSymbolArray is used at render time in drawTileSymbols\n // These indices allow access to the array at collision detection time\n for (const placementType of placementTypes) {\n placedTextSymbolIndices[placementType] = bucket.text.placedSymbolArray.length - 1;\n }\n\n return glyphQuads.length * 4;\n}\n\nfunction getDefaultHorizontalShaping(\n horizontalShaping: Record\n): Shaping | null {\n // We don't care which shaping we get because this is used for collision purposes\n // and all the justifications have the same collision box\n for (const justification in horizontalShaping) {\n return horizontalShaping[justification];\n }\n return null;\n}\n\n/**\n * Add a single label & icon placement.\n */\nfunction addSymbol(bucket: SymbolBucket,\n anchor: Anchor,\n line: Array,\n shapedTextOrientations: ShapedTextOrientations,\n shapedIcon: PositionedIcon | void,\n imageMap: {[_: string]: StyleImage},\n verticallyShapedIcon: PositionedIcon | void,\n layer: SymbolStyleLayer,\n collisionBoxArray: CollisionBoxArray,\n featureIndex: number,\n sourceLayerIndex: number,\n bucketIndex: number,\n textBoxScale: number,\n textPadding: SymbolPadding,\n textAlongLine: boolean,\n textOffset: [number, number],\n iconBoxScale: number,\n iconPadding: SymbolPadding,\n iconAlongLine: boolean,\n iconOffset: [number, number],\n feature: SymbolFeature,\n sizes: Sizes,\n isSDFIcon: boolean,\n canonical: CanonicalTileID,\n layoutTextSize: number) {\n const lineArray = bucket.addToLineVertexArray(anchor, line);\n\n let textCollisionFeature, iconCollisionFeature, verticalTextCollisionFeature, verticalIconCollisionFeature;\n\n let numIconVertices = 0;\n let numVerticalIconVertices = 0;\n let numHorizontalGlyphVertices = 0;\n let numVerticalGlyphVertices = 0;\n let placedIconSymbolIndex = -1;\n let verticalPlacedIconSymbolIndex = -1;\n const placedTextSymbolIndices: {[k: string]: number} = {};\n let key = murmur3('');\n\n if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) {\n const textRotation = layer.layout.get('text-rotate').evaluate(feature, {}, canonical);\n const verticalTextRotation = textRotation + 90.0;\n const verticalShaping = shapedTextOrientations.vertical;\n verticalTextCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticalShaping, textBoxScale, textPadding, textAlongLine, verticalTextRotation);\n\n if (verticallyShapedIcon) {\n verticalIconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticallyShapedIcon, iconBoxScale, iconPadding, textAlongLine, verticalTextRotation);\n }\n }\n\n //Place icon first, so text can have a reference to its index in the placed symbol array.\n //Text symbols can lazily shift at render-time because of variable anchor placement.\n //If the style specifies an `icon-text-fit` then the icon would have to shift along with it.\n // For more info check `updateVariableAnchors` in `draw_symbol.js` .\n if (shapedIcon) {\n const iconRotate = layer.layout.get('icon-rotate').evaluate(feature, {});\n const hasIconTextFit = layer.layout.get('icon-text-fit') !== 'none';\n const iconQuads = getIconQuads(shapedIcon, iconRotate, isSDFIcon, hasIconTextFit);\n const verticalIconQuads = verticallyShapedIcon ? getIconQuads(verticallyShapedIcon, iconRotate, isSDFIcon, hasIconTextFit) : undefined;\n iconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shapedIcon, iconBoxScale, iconPadding, /*align boxes to line*/false, iconRotate);\n\n numIconVertices = iconQuads.length * 4;\n\n const sizeData = bucket.iconSizeData;\n let iconSizeData = null;\n\n if (sizeData.kind === 'source') {\n iconSizeData = [\n SIZE_PACK_FACTOR * layer.layout.get('icon-size').evaluate(feature, {})\n ];\n if (iconSizeData[0] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"icon-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"icon-size\".`);\n }\n } else if (sizeData.kind === 'composite') {\n iconSizeData = [\n SIZE_PACK_FACTOR * sizes.compositeIconSizes[0].evaluate(feature, {}, canonical),\n SIZE_PACK_FACTOR * sizes.compositeIconSizes[1].evaluate(feature, {}, canonical)\n ];\n if (iconSizeData[0] > MAX_PACKED_SIZE || iconSizeData[1] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"icon-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"icon-size\".`);\n }\n }\n\n bucket.addSymbols(\n bucket.icon,\n iconQuads,\n iconSizeData,\n iconOffset,\n iconAlongLine,\n feature,\n WritingMode.none,\n anchor,\n lineArray.lineStartIndex,\n lineArray.lineLength,\n // The icon itself does not have an associated symbol since the text isnt placed yet\n -1, canonical);\n\n placedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1;\n\n if (verticalIconQuads) {\n numVerticalIconVertices = verticalIconQuads.length * 4;\n\n bucket.addSymbols(\n bucket.icon,\n verticalIconQuads,\n iconSizeData,\n iconOffset,\n iconAlongLine,\n feature,\n WritingMode.vertical,\n anchor,\n lineArray.lineStartIndex,\n lineArray.lineLength,\n // The icon itself does not have an associated symbol since the text isnt placed yet\n -1, canonical);\n\n verticalPlacedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1;\n }\n }\n\n const justifications = Object.keys(shapedTextOrientations.horizontal) as TextJustify[];\n for (const justification of justifications) {\n const shaping = shapedTextOrientations.horizontal[justification];\n\n if (!textCollisionFeature) {\n key = murmur3(shaping.text);\n const textRotate = layer.layout.get('text-rotate').evaluate(feature, {}, canonical);\n // As a collision approximation, we can use either the vertical or any of the horizontal versions of the feature\n // We're counting on all versions having similar dimensions\n textCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shaping, textBoxScale, textPadding, textAlongLine, textRotate);\n }\n\n const singleLine = shaping.positionedLines.length === 1;\n numHorizontalGlyphVertices += addTextVertices(\n bucket, anchor, shaping, imageMap, layer, textAlongLine, feature, textOffset, lineArray,\n shapedTextOrientations.vertical ? WritingMode.horizontal : WritingMode.horizontalOnly,\n singleLine ? justifications : [justification],\n placedTextSymbolIndices, placedIconSymbolIndex, sizes, canonical);\n\n if (singleLine) {\n break;\n }\n }\n\n if (shapedTextOrientations.vertical) {\n numVerticalGlyphVertices += addTextVertices(\n bucket, anchor, shapedTextOrientations.vertical, imageMap, layer, textAlongLine, feature,\n textOffset, lineArray, WritingMode.vertical, ['vertical'], placedTextSymbolIndices, verticalPlacedIconSymbolIndex, sizes, canonical);\n }\n\n const textBoxStartIndex = textCollisionFeature ? textCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const textBoxEndIndex = textCollisionFeature ? textCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n const verticalTextBoxStartIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const verticalTextBoxEndIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n const iconBoxStartIndex = iconCollisionFeature ? iconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const iconBoxEndIndex = iconCollisionFeature ? iconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n const verticalIconBoxStartIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const verticalIconBoxEndIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n // Check if runtime collision circles should be used for any of the collision features.\n // It is enough to choose the tallest feature shape as circles are always placed on a line.\n // All measurements are in glyph metrics and later converted into pixels using proper font size \"layoutTextSize\"\n let collisionCircleDiameter = -1;\n\n const getCollisionCircleHeight = (feature: CollisionFeature, prevHeight: number): number => {\n if (feature && feature.circleDiameter)\n return Math.max(feature.circleDiameter, prevHeight);\n return prevHeight;\n };\n\n collisionCircleDiameter = getCollisionCircleHeight(textCollisionFeature, collisionCircleDiameter);\n collisionCircleDiameter = getCollisionCircleHeight(verticalTextCollisionFeature, collisionCircleDiameter);\n collisionCircleDiameter = getCollisionCircleHeight(iconCollisionFeature, collisionCircleDiameter);\n collisionCircleDiameter = getCollisionCircleHeight(verticalIconCollisionFeature, collisionCircleDiameter);\n const useRuntimeCollisionCircles = (collisionCircleDiameter > -1) ? 1 : 0;\n\n // Convert circle collision height into pixels\n if (useRuntimeCollisionCircles)\n collisionCircleDiameter *= layoutTextSize / ONE_EM;\n\n if (bucket.glyphOffsetArray.length >= SymbolBucket.MAX_GLYPHS) warnOnce(\n 'Too many glyphs being rendered in a tile. See https://github.com/mapbox/mapbox-gl-js/issues/2907'\n );\n\n if (feature.sortKey !== undefined) {\n bucket.addToSortKeyRanges(bucket.symbolInstances.length, feature.sortKey as number);\n }\n\n const variableAnchorOffset = getTextVariableAnchorOffset(layer, feature, canonical);\n const [textAnchorOffsetStartIndex, textAnchorOffsetEndIndex] = addTextVariableAnchorOffsets(bucket.textAnchorOffsets, variableAnchorOffset);\n\n bucket.symbolInstances.emplaceBack(\n anchor.x,\n anchor.y,\n placedTextSymbolIndices.right >= 0 ? placedTextSymbolIndices.right : -1,\n placedTextSymbolIndices.center >= 0 ? placedTextSymbolIndices.center : -1,\n placedTextSymbolIndices.left >= 0 ? placedTextSymbolIndices.left : -1,\n placedTextSymbolIndices.vertical || -1,\n placedIconSymbolIndex,\n verticalPlacedIconSymbolIndex,\n key,\n textBoxStartIndex,\n textBoxEndIndex,\n verticalTextBoxStartIndex,\n verticalTextBoxEndIndex,\n iconBoxStartIndex,\n iconBoxEndIndex,\n verticalIconBoxStartIndex,\n verticalIconBoxEndIndex,\n featureIndex,\n numHorizontalGlyphVertices,\n numVerticalGlyphVertices,\n numIconVertices,\n numVerticalIconVertices,\n useRuntimeCollisionCircles,\n 0,\n textBoxScale,\n collisionCircleDiameter,\n textAnchorOffsetStartIndex,\n textAnchorOffsetEndIndex);\n}\n\nfunction anchorIsTooClose(bucket: SymbolBucket, text: string, repeatDistance: number, anchor: Point) {\n const compareText = bucket.compareText;\n if (!(text in compareText)) {\n compareText[text] = [];\n } else {\n const otherAnchors = compareText[text];\n for (let k = otherAnchors.length - 1; k >= 0; k--) {\n if (anchor.dist(otherAnchors[k]) < repeatDistance) {\n // If it's within repeatDistance of one anchor, stop looking\n return true;\n }\n }\n }\n // If anchor is not within repeatDistance of any other anchor, add to array\n compareText[text].push(anchor);\n return false;\n}\n","import {browser} from '../util/browser';\n\nimport {Placement} from '../symbol/placement';\n\nimport type {Transform} from '../geo/transform';\nimport type {StyleLayer} from './style_layer';\nimport type {SymbolStyleLayer} from './style_layer/symbol_style_layer';\nimport type {Tile} from '../source/tile';\nimport type {BucketPart} from '../symbol/placement';\nimport {Terrain} from '../render/terrain';\n\nclass LayerPlacement {\n _sortAcrossTiles: boolean;\n _currentTileIndex: number;\n _currentPartIndex: number;\n _seenCrossTileIDs: {\n [k in string | number]: boolean;\n };\n _bucketParts: Array;\n\n constructor(styleLayer: SymbolStyleLayer) {\n this._sortAcrossTiles = styleLayer.layout.get('symbol-z-order') !== 'viewport-y' &&\n !styleLayer.layout.get('symbol-sort-key').isConstant();\n\n this._currentTileIndex = 0;\n this._currentPartIndex = 0;\n this._seenCrossTileIDs = {};\n this._bucketParts = [];\n }\n\n continuePlacement(tiles: Array, placement: Placement, showCollisionBoxes: boolean, styleLayer: StyleLayer, shouldPausePlacement: () => boolean) {\n\n const bucketParts = this._bucketParts;\n\n while (this._currentTileIndex < tiles.length) {\n const tile = tiles[this._currentTileIndex];\n placement.getBucketParts(bucketParts, styleLayer, tile, this._sortAcrossTiles);\n\n this._currentTileIndex++;\n if (shouldPausePlacement()) {\n return true;\n }\n }\n\n if (this._sortAcrossTiles) {\n this._sortAcrossTiles = false;\n bucketParts.sort((a, b) => (a.sortKey as any as number) - (b.sortKey as any as number));\n }\n\n while (this._currentPartIndex < bucketParts.length) {\n const bucketPart = bucketParts[this._currentPartIndex];\n placement.placeLayerBucketPart(bucketPart, this._seenCrossTileIDs, showCollisionBoxes);\n\n this._currentPartIndex++;\n if (shouldPausePlacement()) {\n return true;\n }\n }\n return false;\n }\n}\n\nexport class PauseablePlacement {\n placement: Placement;\n _done: boolean;\n _currentPlacementIndex: number;\n _forceFullPlacement: boolean;\n _showCollisionBoxes: boolean;\n _inProgressLayer: LayerPlacement;\n\n constructor(\n transform: Transform,\n terrain: Terrain,\n order: Array,\n forceFullPlacement: boolean,\n showCollisionBoxes: boolean,\n fadeDuration: number,\n crossSourceCollisions: boolean,\n prevPlacement?: Placement\n ) {\n this.placement = new Placement(transform, terrain, fadeDuration, crossSourceCollisions, prevPlacement);\n this._currentPlacementIndex = order.length - 1;\n this._forceFullPlacement = forceFullPlacement;\n this._showCollisionBoxes = showCollisionBoxes;\n this._done = false;\n }\n\n isDone() {\n return this._done;\n }\n\n continuePlacement(\n order: Array,\n layers: {[_: string]: StyleLayer},\n layerTiles: {[_: string]: Array}\n ) {\n const startTime = browser.now();\n\n const shouldPausePlacement = () => {\n return this._forceFullPlacement ? false : (browser.now() - startTime) > 2;\n };\n\n while (this._currentPlacementIndex >= 0) {\n const layerId = order[this._currentPlacementIndex];\n const layer = layers[layerId];\n const placementZoom = this.placement.collisionIndex.transform.zoom;\n if (layer.type === 'symbol' &&\n (!layer.minzoom || layer.minzoom <= placementZoom) &&\n (!layer.maxzoom || layer.maxzoom > placementZoom)) {\n\n if (!this._inProgressLayer) {\n this._inProgressLayer = new LayerPlacement(layer as any as SymbolStyleLayer);\n }\n\n const pausePlacement = this._inProgressLayer.continuePlacement(layerTiles[layer.source], this.placement, this._showCollisionBoxes, layer, shouldPausePlacement);\n\n if (pausePlacement) {\n // We didn't finish placing all layers within 2ms,\n // but we can keep rendering with a partial placement\n // We'll resume here on the next frame\n return;\n }\n\n delete this._inProgressLayer;\n }\n\n this._currentPlacementIndex--;\n }\n\n this._done = true;\n }\n\n commit(now: number) {\n this.placement.commit(now);\n return this.placement;\n }\n}\n","\nconst ARRAY_TYPES = [\n Int8Array, Uint8Array, Uint8ClampedArray, Int16Array, Uint16Array,\n Int32Array, Uint32Array, Float32Array, Float64Array\n];\n\n/** @typedef {Int8ArrayConstructor | Uint8ArrayConstructor | Uint8ClampedArrayConstructor | Int16ArrayConstructor | Uint16ArrayConstructor | Int32ArrayConstructor | Uint32ArrayConstructor | Float32ArrayConstructor | Float64ArrayConstructor} TypedArrayConstructor */\n\nconst VERSION = 1; // serialized format version\nconst HEADER_SIZE = 8;\n\nexport default class KDBush {\n\n /**\n * Creates an index from raw `ArrayBuffer` data.\n * @param {ArrayBuffer} data\n */\n static from(data) {\n if (!(data instanceof ArrayBuffer)) {\n throw new Error('Data must be an instance of ArrayBuffer.');\n }\n const [magic, versionAndType] = new Uint8Array(data, 0, 2);\n if (magic !== 0xdb) {\n throw new Error('Data does not appear to be in a KDBush format.');\n }\n const version = versionAndType >> 4;\n if (version !== VERSION) {\n throw new Error(`Got v${version} data when expected v${VERSION}.`);\n }\n const ArrayType = ARRAY_TYPES[versionAndType & 0x0f];\n if (!ArrayType) {\n throw new Error('Unrecognized array type.');\n }\n const [nodeSize] = new Uint16Array(data, 2, 1);\n const [numItems] = new Uint32Array(data, 4, 1);\n\n return new KDBush(numItems, nodeSize, ArrayType, data);\n }\n\n /**\n * Creates an index that will hold a given number of items.\n * @param {number} numItems\n * @param {number} [nodeSize=64] Size of the KD-tree node (64 by default).\n * @param {TypedArrayConstructor} [ArrayType=Float64Array] The array type used for coordinates storage (`Float64Array` by default).\n * @param {ArrayBuffer} [data] (For internal use only)\n */\n constructor(numItems, nodeSize = 64, ArrayType = Float64Array, data) {\n if (isNaN(numItems) || numItems < 0) throw new Error(`Unpexpected numItems value: ${numItems}.`);\n\n this.numItems = +numItems;\n this.nodeSize = Math.min(Math.max(+nodeSize, 2), 65535);\n this.ArrayType = ArrayType;\n this.IndexArrayType = numItems < 65536 ? Uint16Array : Uint32Array;\n\n const arrayTypeIndex = ARRAY_TYPES.indexOf(this.ArrayType);\n const coordsByteSize = numItems * 2 * this.ArrayType.BYTES_PER_ELEMENT;\n const idsByteSize = numItems * this.IndexArrayType.BYTES_PER_ELEMENT;\n const padCoords = (8 - idsByteSize % 8) % 8;\n\n if (arrayTypeIndex < 0) {\n throw new Error(`Unexpected typed array class: ${ArrayType}.`);\n }\n\n if (data && (data instanceof ArrayBuffer)) { // reconstruct an index from a buffer\n this.data = data;\n this.ids = new this.IndexArrayType(this.data, HEADER_SIZE, numItems);\n this.coords = new this.ArrayType(this.data, HEADER_SIZE + idsByteSize + padCoords, numItems * 2);\n this._pos = numItems * 2;\n this._finished = true;\n } else { // initialize a new index\n this.data = new ArrayBuffer(HEADER_SIZE + coordsByteSize + idsByteSize + padCoords);\n this.ids = new this.IndexArrayType(this.data, HEADER_SIZE, numItems);\n this.coords = new this.ArrayType(this.data, HEADER_SIZE + idsByteSize + padCoords, numItems * 2);\n this._pos = 0;\n this._finished = false;\n\n // set header\n new Uint8Array(this.data, 0, 2).set([0xdb, (VERSION << 4) + arrayTypeIndex]);\n new Uint16Array(this.data, 2, 1)[0] = nodeSize;\n new Uint32Array(this.data, 4, 1)[0] = numItems;\n }\n }\n\n /**\n * Add a point to the index.\n * @param {number} x\n * @param {number} y\n * @returns {number} An incremental index associated with the added item (starting from `0`).\n */\n add(x, y) {\n const index = this._pos >> 1;\n this.ids[index] = index;\n this.coords[this._pos++] = x;\n this.coords[this._pos++] = y;\n return index;\n }\n\n /**\n * Perform indexing of the added points.\n */\n finish() {\n const numAdded = this._pos >> 1;\n if (numAdded !== this.numItems) {\n throw new Error(`Added ${numAdded} items when expected ${this.numItems}.`);\n }\n // kd-sort both arrays for efficient search\n sort(this.ids, this.coords, this.nodeSize, 0, this.numItems - 1, 0);\n\n this._finished = true;\n return this;\n }\n\n /**\n * Search the index for items within a given bounding box.\n * @param {number} minX\n * @param {number} minY\n * @param {number} maxX\n * @param {number} maxY\n * @returns {number[]} An array of indices correponding to the found items.\n */\n range(minX, minY, maxX, maxY) {\n if (!this._finished) throw new Error('Data not yet indexed - call index.finish().');\n\n const {ids, coords, nodeSize} = this;\n const stack = [0, ids.length - 1, 0];\n const result = [];\n\n // recursively search for items in range in the kd-sorted arrays\n while (stack.length) {\n const axis = stack.pop() || 0;\n const right = stack.pop() || 0;\n const left = stack.pop() || 0;\n\n // if we reached \"tree node\", search linearly\n if (right - left <= nodeSize) {\n for (let i = left; i <= right; i++) {\n const x = coords[2 * i];\n const y = coords[2 * i + 1];\n if (x >= minX && x <= maxX && y >= minY && y <= maxY) result.push(ids[i]);\n }\n continue;\n }\n\n // otherwise find the middle index\n const m = (left + right) >> 1;\n\n // include the middle item if it's in range\n const x = coords[2 * m];\n const y = coords[2 * m + 1];\n if (x >= minX && x <= maxX && y >= minY && y <= maxY) result.push(ids[m]);\n\n // queue search in halves that intersect the query\n if (axis === 0 ? minX <= x : minY <= y) {\n stack.push(left);\n stack.push(m - 1);\n stack.push(1 - axis);\n }\n if (axis === 0 ? maxX >= x : maxY >= y) {\n stack.push(m + 1);\n stack.push(right);\n stack.push(1 - axis);\n }\n }\n\n return result;\n }\n\n /**\n * Search the index for items within a given radius.\n * @param {number} qx\n * @param {number} qy\n * @param {number} r Query radius.\n * @returns {number[]} An array of indices correponding to the found items.\n */\n within(qx, qy, r) {\n if (!this._finished) throw new Error('Data not yet indexed - call index.finish().');\n\n const {ids, coords, nodeSize} = this;\n const stack = [0, ids.length - 1, 0];\n const result = [];\n const r2 = r * r;\n\n // recursively search for items within radius in the kd-sorted arrays\n while (stack.length) {\n const axis = stack.pop() || 0;\n const right = stack.pop() || 0;\n const left = stack.pop() || 0;\n\n // if we reached \"tree node\", search linearly\n if (right - left <= nodeSize) {\n for (let i = left; i <= right; i++) {\n if (sqDist(coords[2 * i], coords[2 * i + 1], qx, qy) <= r2) result.push(ids[i]);\n }\n continue;\n }\n\n // otherwise find the middle index\n const m = (left + right) >> 1;\n\n // include the middle item if it's in range\n const x = coords[2 * m];\n const y = coords[2 * m + 1];\n if (sqDist(x, y, qx, qy) <= r2) result.push(ids[m]);\n\n // queue search in halves that intersect the query\n if (axis === 0 ? qx - r <= x : qy - r <= y) {\n stack.push(left);\n stack.push(m - 1);\n stack.push(1 - axis);\n }\n if (axis === 0 ? qx + r >= x : qy + r >= y) {\n stack.push(m + 1);\n stack.push(right);\n stack.push(1 - axis);\n }\n }\n\n return result;\n }\n}\n\n/**\n * @param {Uint16Array | Uint32Array} ids\n * @param {InstanceType} coords\n * @param {number} nodeSize\n * @param {number} left\n * @param {number} right\n * @param {number} axis\n */\nfunction sort(ids, coords, nodeSize, left, right, axis) {\n if (right - left <= nodeSize) return;\n\n const m = (left + right) >> 1; // middle index\n\n // sort ids and coords around the middle index so that the halves lie\n // either left/right or top/bottom correspondingly (taking turns)\n select(ids, coords, m, left, right, axis);\n\n // recursively kd-sort first half and second half on the opposite axis\n sort(ids, coords, nodeSize, left, m - 1, 1 - axis);\n sort(ids, coords, nodeSize, m + 1, right, 1 - axis);\n}\n\n/**\n * Custom Floyd-Rivest selection algorithm: sort ids and coords so that\n * [left..k-1] items are smaller than k-th item (on either x or y axis)\n * @param {Uint16Array | Uint32Array} ids\n * @param {InstanceType} coords\n * @param {number} k\n * @param {number} left\n * @param {number} right\n * @param {number} axis\n */\nfunction select(ids, coords, k, left, right, axis) {\n\n while (right > left) {\n if (right - left > 600) {\n const n = right - left + 1;\n const m = k - left + 1;\n const z = Math.log(n);\n const s = 0.5 * Math.exp(2 * z / 3);\n const sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);\n const newLeft = Math.max(left, Math.floor(k - m * s / n + sd));\n const newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));\n select(ids, coords, k, newLeft, newRight, axis);\n }\n\n const t = coords[2 * k + axis];\n let i = left;\n let j = right;\n\n swapItem(ids, coords, left, k);\n if (coords[2 * right + axis] > t) swapItem(ids, coords, left, right);\n\n while (i < j) {\n swapItem(ids, coords, i, j);\n i++;\n j--;\n while (coords[2 * i + axis] < t) i++;\n while (coords[2 * j + axis] > t) j--;\n }\n\n if (coords[2 * left + axis] === t) swapItem(ids, coords, left, j);\n else {\n j++;\n swapItem(ids, coords, j, right);\n }\n\n if (j <= k) left = j + 1;\n if (k <= j) right = j - 1;\n }\n}\n\n/**\n * @param {Uint16Array | Uint32Array} ids\n * @param {InstanceType} coords\n * @param {number} i\n * @param {number} j\n */\nfunction swapItem(ids, coords, i, j) {\n swap(ids, i, j);\n swap(coords, 2 * i, 2 * j);\n swap(coords, 2 * i + 1, 2 * j + 1);\n}\n\n/**\n * @param {InstanceType} arr\n * @param {number} i\n * @param {number} j\n */\nfunction swap(arr, i, j) {\n const tmp = arr[i];\n arr[i] = arr[j];\n arr[j] = tmp;\n}\n\n/**\n * @param {number} ax\n * @param {number} ay\n * @param {number} bx\n * @param {number} by\n */\nfunction sqDist(ax, ay, bx, by) {\n const dx = ax - bx;\n const dy = ay - by;\n return dx * dx + dy * dy;\n}\n","import KDBush from 'kdbush';\nimport {EXTENT} from '../data/extent';\n\nimport {SymbolInstanceArray} from '../data/array_types.g';\n\nimport type {SymbolInstance} from '../data/array_types.g';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {Tile} from '../source/tile';\n\n/*\n The CrossTileSymbolIndex generally works on the assumption that\n a conceptual \"unique symbol\" can be identified by the text of\n the label combined with the anchor point. The goal is to assign\n these conceptual \"unique symbols\" a shared crossTileID that can be\n used by Placement to keep fading opacity states consistent and to\n deduplicate labels.\n\n The CrossTileSymbolIndex indexes all the current symbol instances and\n their crossTileIDs. When a symbol bucket gets added or updated, the\n index assigns a crossTileID to each of it's symbol instances by either\n matching it with an existing id or assigning a new one.\n*/\n\n// Round anchor positions to roughly 4 pixel grid\nconst roundingFactor = 512 / EXTENT / 2;\n\nexport const KDBUSH_THRESHHOLD = 128;\n\ninterface SymbolsByKeyEntry {\n index?: KDBush;\n positions?: {x: number; y: number}[];\n crossTileIDs: number[];\n}\n\nclass TileLayerIndex {\n _symbolsByKey: Record = {};\n\n constructor(public tileID: OverscaledTileID, symbolInstances: SymbolInstanceArray, public bucketInstanceId: number) {\n // group the symbolInstances by key\n const symbolInstancesByKey = new Map();\n for (let i = 0; i < symbolInstances.length; i++) {\n const symbolInstance = symbolInstances.get(i);\n const key = symbolInstance.key;\n const instances = symbolInstancesByKey.get(key);\n if (instances) {\n // This tile may have multiple symbol instances with the same key\n // Store each one along with its coordinates\n instances.push(symbolInstance);\n } else {\n symbolInstancesByKey.set(key, [symbolInstance]);\n }\n }\n\n // index the SymbolInstances in this each bucket\n for (const [key, symbols] of symbolInstancesByKey) {\n const positions = symbols.map(symbolInstance => ({x: Math.floor(symbolInstance.anchorX * roundingFactor), y: Math.floor(symbolInstance.anchorY * roundingFactor)}));\n const crossTileIDs = symbols.map(v => v.crossTileID);\n const entry: SymbolsByKeyEntry = {positions, crossTileIDs};\n\n // once we get too many symbols for a given key, it becomes much faster to index it before queries\n if (entry.positions.length > KDBUSH_THRESHHOLD) {\n\n const index = new KDBush(entry.positions.length, 16, Uint16Array);\n for (const {x, y} of entry.positions) index.add(x, y);\n index.finish();\n\n // clear all references to the original positions data\n delete entry.positions;\n entry.index = index;\n }\n\n this._symbolsByKey[key] = entry;\n }\n }\n\n // Converts the coordinates of the input symbol instance into coordinates that be can compared\n // against other symbols in this index. Coordinates are:\n // (1) local-tile-based (so after correction we get x,y values relative to our local anchorX/Y)\n // (2) converted to the z-scale of this TileLayerIndex\n // (3) down-sampled by \"roundingFactor\" from tile coordinate precision in order to be\n // more tolerant of small differences between tiles.\n getScaledCoordinates(symbolInstance: SymbolInstance, childTileID: OverscaledTileID): {x: number; y: number} {\n const {x: localX, y: localY, z: localZ} = this.tileID.canonical;\n const {x, y, z} = childTileID.canonical;\n\n const zDifference = z - localZ;\n const scale = roundingFactor / Math.pow(2, zDifference);\n const xWorld = (x * EXTENT + symbolInstance.anchorX) * scale;\n const yWorld = (y * EXTENT + symbolInstance.anchorY) * scale;\n const xOffset = localX * EXTENT * roundingFactor;\n const yOffset = localY * EXTENT * roundingFactor;\n const result = {\n x: Math.floor(xWorld - xOffset),\n y: Math.floor(yWorld - yOffset)\n };\n\n return result;\n }\n\n findMatches(symbolInstances: SymbolInstanceArray, newTileID: OverscaledTileID, zoomCrossTileIDs: {\n [crossTileID: number]: boolean;\n }) {\n const tolerance = this.tileID.canonical.z < newTileID.canonical.z ? 1 : Math.pow(2, this.tileID.canonical.z - newTileID.canonical.z);\n\n for (let i = 0; i < symbolInstances.length; i++) {\n const symbolInstance = symbolInstances.get(i);\n if (symbolInstance.crossTileID) {\n // already has a match, skip\n continue;\n }\n\n const entry = this._symbolsByKey[symbolInstance.key];\n if (!entry) {\n // No symbol with this key in this bucket\n continue;\n }\n\n const scaledSymbolCoord = this.getScaledCoordinates(symbolInstance, newTileID);\n\n if (entry.index) {\n // Return any symbol with the same keys whose coordinates are within 1\n // grid unit. (with a 4px grid, this covers a 12px by 12px area)\n const indexes = entry.index.range(\n scaledSymbolCoord.x - tolerance,\n scaledSymbolCoord.y - tolerance,\n scaledSymbolCoord.x + tolerance,\n scaledSymbolCoord.y + tolerance).sort();\n\n for (const i of indexes) {\n const crossTileID = entry.crossTileIDs[i];\n\n if (!zoomCrossTileIDs[crossTileID]) {\n // Once we've marked ourselves duplicate against this parent symbol,\n // don't let any other symbols at the same zoom level duplicate against\n // the same parent (see issue #5993)\n zoomCrossTileIDs[crossTileID] = true;\n symbolInstance.crossTileID = crossTileID;\n break;\n }\n }\n } else if (entry.positions) {\n for (let i = 0; i < entry.positions.length; i++) {\n const thisTileSymbol = entry.positions[i];\n const crossTileID = entry.crossTileIDs[i];\n\n // Return any symbol with the same keys whose coordinates are within 1\n // grid unit. (with a 4px grid, this covers a 12px by 12px area)\n if (Math.abs(thisTileSymbol.x - scaledSymbolCoord.x) <= tolerance &&\n Math.abs(thisTileSymbol.y - scaledSymbolCoord.y) <= tolerance &&\n !zoomCrossTileIDs[crossTileID]) {\n // Once we've marked ourselves duplicate against this parent symbol,\n // don't let any other symbols at the same zoom level duplicate against\n // the same parent (see issue #5993)\n zoomCrossTileIDs[crossTileID] = true;\n symbolInstance.crossTileID = crossTileID;\n break;\n }\n }\n }\n }\n }\n\n getCrossTileIDsLists() {\n return Object.values(this._symbolsByKey).map(({crossTileIDs}) => crossTileIDs);\n }\n}\n\nclass CrossTileIDs {\n maxCrossTileID: number;\n constructor() {\n this.maxCrossTileID = 0;\n }\n generate() {\n return ++this.maxCrossTileID;\n }\n}\n\nclass CrossTileSymbolLayerIndex {\n indexes: {\n [zoom in string | number]: {\n [tileId in string | number]: TileLayerIndex;\n };\n };\n usedCrossTileIDs: {\n [zoom in string | number]: {\n [crossTileID: number]: boolean;\n };\n };\n lng: number;\n\n constructor() {\n this.indexes = {};\n this.usedCrossTileIDs = {};\n this.lng = 0;\n }\n\n /*\n * Sometimes when a user pans across the antimeridian the longitude value gets wrapped.\n * To prevent labels from flashing out and in we adjust the tileID values in the indexes\n * so that they match the new wrapped version of the map.\n */\n handleWrapJump(lng: number) {\n const wrapDelta = Math.round((lng - this.lng) / 360);\n if (wrapDelta !== 0) {\n for (const zoom in this.indexes) {\n const zoomIndexes = this.indexes[zoom];\n const newZoomIndex = {};\n for (const key in zoomIndexes) {\n // change the tileID's wrap and add it to a new index\n const index = zoomIndexes[key];\n index.tileID = index.tileID.unwrapTo(index.tileID.wrap + wrapDelta);\n newZoomIndex[index.tileID.key] = index;\n }\n this.indexes[zoom] = newZoomIndex;\n }\n }\n this.lng = lng;\n }\n\n addBucket(tileID: OverscaledTileID, bucket: SymbolBucket, crossTileIDs: CrossTileIDs) {\n if (this.indexes[tileID.overscaledZ] &&\n this.indexes[tileID.overscaledZ][tileID.key]) {\n if (this.indexes[tileID.overscaledZ][tileID.key].bucketInstanceId ===\n bucket.bucketInstanceId) {\n return false;\n } else {\n // We're replacing this bucket with an updated version\n // Remove the old bucket's \"used crossTileIDs\" now so that\n // the new bucket can claim them.\n // The old index entries themselves stick around until\n // 'removeStaleBuckets' is called.\n this.removeBucketCrossTileIDs(tileID.overscaledZ,\n this.indexes[tileID.overscaledZ][tileID.key]);\n }\n }\n\n for (let i = 0; i < bucket.symbolInstances.length; i++) {\n const symbolInstance = bucket.symbolInstances.get(i);\n symbolInstance.crossTileID = 0;\n }\n\n if (!this.usedCrossTileIDs[tileID.overscaledZ]) {\n this.usedCrossTileIDs[tileID.overscaledZ] = {};\n }\n const zoomCrossTileIDs = this.usedCrossTileIDs[tileID.overscaledZ];\n\n for (const zoom in this.indexes) {\n const zoomIndexes = this.indexes[zoom];\n if (Number(zoom) > tileID.overscaledZ) {\n for (const id in zoomIndexes) {\n const childIndex = zoomIndexes[id];\n if (childIndex.tileID.isChildOf(tileID)) {\n childIndex.findMatches(bucket.symbolInstances, tileID, zoomCrossTileIDs);\n }\n }\n } else {\n const parentCoord = tileID.scaledTo(Number(zoom));\n const parentIndex = zoomIndexes[parentCoord.key];\n if (parentIndex) {\n parentIndex.findMatches(bucket.symbolInstances, tileID, zoomCrossTileIDs);\n }\n }\n }\n\n for (let i = 0; i < bucket.symbolInstances.length; i++) {\n const symbolInstance = bucket.symbolInstances.get(i);\n if (!symbolInstance.crossTileID) {\n // symbol did not match any known symbol, assign a new id\n symbolInstance.crossTileID = crossTileIDs.generate();\n zoomCrossTileIDs[symbolInstance.crossTileID] = true;\n }\n }\n\n if (this.indexes[tileID.overscaledZ] === undefined) {\n this.indexes[tileID.overscaledZ] = {};\n }\n this.indexes[tileID.overscaledZ][tileID.key] = new TileLayerIndex(tileID, bucket.symbolInstances, bucket.bucketInstanceId);\n\n return true;\n }\n\n removeBucketCrossTileIDs(zoom: string | number, removedBucket: TileLayerIndex) {\n for (const crossTileIDs of removedBucket.getCrossTileIDsLists()) {\n for (const crossTileID of crossTileIDs) {\n delete this.usedCrossTileIDs[zoom][crossTileID];\n }\n }\n }\n\n removeStaleBuckets(currentIDs: {\n [k in string | number]: boolean;\n }) {\n let tilesChanged = false;\n for (const z in this.indexes) {\n const zoomIndexes = this.indexes[z];\n for (const tileKey in zoomIndexes) {\n if (!currentIDs[zoomIndexes[tileKey].bucketInstanceId]) {\n this.removeBucketCrossTileIDs(z, zoomIndexes[tileKey]);\n delete zoomIndexes[tileKey];\n tilesChanged = true;\n }\n }\n }\n return tilesChanged;\n }\n}\n\nexport class CrossTileSymbolIndex {\n layerIndexes: {[layerId: string]: CrossTileSymbolLayerIndex};\n crossTileIDs: CrossTileIDs;\n maxBucketInstanceId: number;\n bucketsInCurrentPlacement: {[_: number]: boolean};\n\n constructor() {\n this.layerIndexes = {};\n this.crossTileIDs = new CrossTileIDs();\n this.maxBucketInstanceId = 0;\n this.bucketsInCurrentPlacement = {};\n }\n\n addLayer(styleLayer: StyleLayer, tiles: Array, lng: number) {\n let layerIndex = this.layerIndexes[styleLayer.id];\n if (layerIndex === undefined) {\n layerIndex = this.layerIndexes[styleLayer.id] = new CrossTileSymbolLayerIndex();\n }\n\n let symbolBucketsChanged = false;\n const currentBucketIDs = {};\n\n layerIndex.handleWrapJump(lng);\n\n for (const tile of tiles) {\n const symbolBucket = (tile.getBucket(styleLayer) as any as SymbolBucket);\n if (!symbolBucket || styleLayer.id !== symbolBucket.layerIds[0])\n continue;\n\n if (!symbolBucket.bucketInstanceId) {\n symbolBucket.bucketInstanceId = ++this.maxBucketInstanceId;\n }\n\n if (layerIndex.addBucket(tile.tileID, symbolBucket, this.crossTileIDs)) {\n symbolBucketsChanged = true;\n }\n currentBucketIDs[symbolBucket.bucketInstanceId] = true;\n }\n\n if (layerIndex.removeStaleBuckets(currentBucketIDs)) {\n symbolBucketsChanged = true;\n }\n\n return symbolBucketsChanged;\n }\n\n pruneUnusedLayers(usedLayers: Array) {\n const usedLayerMap = {};\n usedLayers.forEach((usedLayer) => {\n usedLayerMap[usedLayer] = true;\n });\n for (const layerId in this.layerIndexes) {\n if (!usedLayerMap[layerId]) {\n delete this.layerIndexes[layerId];\n }\n }\n }\n}\n","import {Event, ErrorEvent, Evented} from '../util/evented';\nimport {StyleLayer} from './style_layer';\nimport {createStyleLayer} from './create_style_layer';\nimport {loadSprite} from './load_sprite';\nimport {ImageManager} from '../render/image_manager';\nimport {GlyphManager} from '../render/glyph_manager';\nimport {Light} from './light';\nimport {LineAtlas} from '../render/line_atlas';\nimport {pick, clone, extend, deepEqual, filterObject, mapObject} from '../util/util';\nimport {coerceSpriteToArray} from '../util/style';\nimport {getJSON, getReferrer, makeRequest} from '../util/ajax';\nimport {ResourceType} from '../util/request_manager';\nimport {browser} from '../util/browser';\nimport {Dispatcher} from '../util/dispatcher';\nimport {validateStyle, emitValidationErrors as _emitValidationErrors} from './validate_style';\nimport {getSourceType, setSourceType, Source} from '../source/source';\nimport type {SourceClass} from '../source/source';\nimport {QueryRenderedFeaturesOptions, QuerySourceFeatureOptions, queryRenderedFeatures, queryRenderedSymbols, querySourceFeatures} from '../source/query_features';\nimport {SourceCache} from '../source/source_cache';\nimport {GeoJSONSource} from '../source/geojson_source';\nimport {latest as styleSpec, derefLayers as deref, emptyStyle, diff as diffStyles, operations as diffOperations} from '@maplibre/maplibre-gl-style-spec';\nimport {getGlobalWorkerPool} from '../util/global_worker_pool';\nimport {\n registerForPluginStateChange,\n evented as rtlTextPluginEvented,\n triggerPluginCompletionEvent\n} from '../source/rtl_text_plugin';\nimport {PauseablePlacement} from './pauseable_placement';\nimport {ZoomHistory} from './zoom_history';\nimport {CrossTileSymbolIndex} from '../symbol/cross_tile_symbol_index';\nimport {validateCustomStyleLayer} from './style_layer/custom_style_layer';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\n\n// We're skipping validation errors with the `source.canvas` identifier in order\n// to continue to allow canvas sources to be added at runtime/updated in\n// smart setStyle (see https://github.com/mapbox/mapbox-gl-js/pull/6424):\nconst emitValidationErrors = (evented: Evented, errors?: ReadonlyArray<{\n message: string;\n identifier?: string;\n}> | null) =>\n _emitValidationErrors(evented, errors && errors.filter(error => error.identifier !== 'source.canvas'));\n\nimport type {Map} from '../ui/map';\nimport type {Transform} from '../geo/transform';\nimport type {StyleImage} from './style_image';\nimport type {StyleGlyph} from './style_glyph';\nimport type {Callback} from '../types/callback';\nimport type {EvaluationParameters} from './evaluation_parameters';\nimport type {Placement} from '../symbol/placement';\nimport type {Cancelable} from '../types/cancelable';\nimport type {RequestParameters, ResponseCallback} from '../util/ajax';\nimport type {\n LayerSpecification,\n FilterSpecification,\n StyleSpecification,\n LightSpecification,\n SourceSpecification,\n SpriteSpecification,\n} from '@maplibre/maplibre-gl-style-spec';\nimport type {CustomLayerInterface} from './style_layer/custom_style_layer';\nimport type {Validator} from './validate_style';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nconst supportedDiffOperations = pick(diffOperations, [\n 'addLayer',\n 'removeLayer',\n 'setPaintProperty',\n 'setLayoutProperty',\n 'setFilter',\n 'addSource',\n 'removeSource',\n 'setLayerZoomRange',\n 'setLight',\n 'setTransition',\n 'setGeoJSONSourceData',\n 'setGlyphs',\n 'setSprite',\n]);\n\nconst ignoredDiffOperations = pick(diffOperations, [\n 'setCenter',\n 'setZoom',\n 'setBearing',\n 'setPitch'\n]);\n\nconst empty = emptyStyle() as StyleSpecification;\n/**\n * A feature identifier that is bound to a source\n */\nexport type FeatureIdentifier = {\n /**\n * Unique id of the feature.\n */\n id?: string | number | undefined;\n /**\n * The id of the vector or GeoJSON source for the feature.\n */\n source: string;\n /**\n * *For vector tile sources, `sourceLayer` is required.*\n */\n sourceLayer?: string | undefined;\n};\n\n/**\n * The options object related to the {@link Map}'s style related methods\n */\nexport type StyleOptions = {\n /**\n * If false, style validation will be skipped. Useful in production environment.\n */\n validate?: boolean;\n /**\n * Defines a CSS\n * font-family for locally overriding generation of glyphs in the 'CJK Unified Ideographs', 'Hiragana', 'Katakana' and 'Hangul Syllables' ranges.\n * In these ranges, font settings from the map's style will be ignored, except for font-weight keywords (light/regular/medium/bold).\n * Set to `false`, to enable font settings from the map's style for these glyph ranges.\n * Forces a full update.\n */\n localIdeographFontFamily?: string;\n};\n\n/**\n * Supporting type to add validation to another style related type\n */\nexport type StyleSetterOptions = {\n /**\n * Whether to check if the filter conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function.\n */\n validate?: boolean;\n};\n\n/**\n * Part of {@link Map#setStyle} options, transformStyle is a convenience function that allows to modify a style after it is fetched but before it is committed to the map state\n * this function exposes previous and next styles, it can be commonly used to support a range of functionalities like:\n * when previous style carries certain 'state' that needs to be carried over to a new style gracefully\n * when a desired style is a certain combination of previous and incoming style\n * when an incoming style requires modification based on external state\n *\n * @param previousStyle - The current style.\n * @param nextStyle - The next style.\n * @returns resulting style that will to be applied to the map\n *\n * @example\n * ```ts\n * map.setStyle('https://demotiles.maplibre.org/style.json', {\n * transformStyle: (previousStyle, nextStyle) => ({\n * ...nextStyle,\n * sources: {\n * ...nextStyle.sources,\n * // copy a source from previous style\n * 'osm': previousStyle.sources.osm\n * },\n * layers: [\n * // background layer\n * nextStyle.layers[0],\n * // copy a layer from previous style\n * previousStyle.layers[0],\n * // other layers from the next style\n * ...nextStyle.layers.slice(1).map(layer => {\n * // hide the layers we don't need from demotiles style\n * if (layer.id.startsWith('geolines')) {\n * layer.layout = {...layer.layout || {}, visibility: 'none'};\n * // filter out US polygons\n * } else if (layer.id.startsWith('coastline') || layer.id.startsWith('countries')) {\n * layer.filter = ['!=', ['get', 'ADM0_A3'], 'USA'];\n * }\n * return layer;\n * })\n * ]\n * })\n * });\n * ```\n */\nexport type TransformStyleFunction = (previous: StyleSpecification | undefined, next: StyleSpecification) => StyleSpecification;\n\n/**\n * The options object related to the {@link Map}'s style related methods\n */\nexport type StyleSwapOptions = {\n /**\n * If false, force a 'full' update, removing the current style\n * and building the given one instead of attempting a diff-based update.\n */\n diff?: boolean;\n /**\n * TransformStyleFunction is a convenience function\n * that allows to modify a style after it is fetched but before it is committed to the map state. Refer to {@link TransformStyleFunction}.\n */\n transformStyle?: TransformStyleFunction;\n}\n\n/**\n * Specifies a layer to be added to a {@link Style}. In addition to a standard {@link LayerSpecification}\n * or a {@link CustomLayerInterface}, a {@link LayerSpecification} with an embedded {@link SourceSpecification} can also be provided.\n */\nexport type AddLayerObject = LayerSpecification | (Omit & {source: SourceSpecification}) | CustomLayerInterface;\n\n/**\n * The Style base class\n */\nexport class Style extends Evented {\n map: Map;\n stylesheet: StyleSpecification;\n dispatcher: Dispatcher;\n imageManager: ImageManager;\n glyphManager: GlyphManager;\n lineAtlas: LineAtlas;\n light: Light;\n\n _request: Cancelable;\n _spriteRequest: Cancelable;\n _layers: {[_: string]: StyleLayer};\n _serializedLayers: {[_: string]: LayerSpecification};\n _order: Array;\n sourceCaches: {[_: string]: SourceCache};\n zoomHistory: ZoomHistory;\n _loaded: boolean;\n _rtlTextPluginCallback: (a: any) => any;\n _changed: boolean;\n _updatedSources: {[_: string]: 'clear' | 'reload'};\n _updatedLayers: {[_: string]: true};\n _removedLayers: {[_: string]: StyleLayer};\n _changedImages: {[_: string]: true};\n _glyphsDidChange: boolean;\n _updatedPaintProps: {[layer: string]: true};\n _layerOrderChanged: boolean;\n // image ids of images loaded from style's sprite\n _spritesImagesIds: {[spriteId: string]: string[]};\n // image ids of all images loaded (sprite + user)\n _availableImages: Array;\n\n crossTileSymbolIndex: CrossTileSymbolIndex;\n pauseablePlacement: PauseablePlacement;\n placement: Placement;\n z: number;\n\n static registerForPluginStateChange: typeof registerForPluginStateChange;\n\n constructor(map: Map, options: StyleOptions = {}) {\n super();\n\n this.map = map;\n this.dispatcher = new Dispatcher(getGlobalWorkerPool(), this, map._getMapId());\n this.imageManager = new ImageManager();\n this.imageManager.setEventedParent(this);\n this.glyphManager = new GlyphManager(map._requestManager, options.localIdeographFontFamily);\n this.lineAtlas = new LineAtlas(256, 512);\n this.crossTileSymbolIndex = new CrossTileSymbolIndex();\n\n this._spritesImagesIds = {};\n this._layers = {};\n\n this._order = [];\n this.sourceCaches = {};\n this.zoomHistory = new ZoomHistory();\n this._loaded = false;\n this._availableImages = [];\n\n this._resetUpdates();\n\n this.dispatcher.broadcast('setReferrer', getReferrer());\n\n const self = this;\n this._rtlTextPluginCallback = Style.registerForPluginStateChange((event) => {\n const state = {\n pluginStatus: event.pluginStatus,\n pluginURL: event.pluginURL\n };\n self.dispatcher.broadcast('syncRTLPluginState', state, (err, results) => {\n triggerPluginCompletionEvent(err);\n if (results) {\n const allComplete = results.every((elem) => elem);\n if (allComplete) {\n for (const id in self.sourceCaches) {\n const sourceType = self.sourceCaches[id].getSource().type;\n if (sourceType === 'vector' || sourceType === 'geojson') {\n // Non-vector sources don't have any symbols buckets to reload when the RTL text plugin loads\n // They also load more quickly, so they're more likely to have already displaying tiles\n // that would be unnecessarily booted by the plugin load event\n self.sourceCaches[id].reload(); // Should be a no-op if the plugin loads before any tiles load\n }\n }\n }\n }\n\n });\n });\n\n this.on('data', (event) => {\n if (event.dataType !== 'source' || event.sourceDataType !== 'metadata') {\n return;\n }\n\n const sourceCache = this.sourceCaches[event.sourceId];\n if (!sourceCache) {\n return;\n }\n\n const source = sourceCache.getSource();\n if (!source || !source.vectorLayerIds) {\n return;\n }\n\n for (const layerId in this._layers) {\n const layer = this._layers[layerId];\n if (layer.source === source.id) {\n this._validateLayer(layer);\n }\n }\n });\n }\n\n loadURL(url: string, options: StyleSwapOptions & StyleSetterOptions = {}, previousStyle?: StyleSpecification) {\n this.fire(new Event('dataloading', {dataType: 'style'}));\n\n options.validate = typeof options.validate === 'boolean' ?\n options.validate : true;\n\n const request = this.map._requestManager.transformRequest(url, ResourceType.Style);\n this._request = getJSON(request, (error?: Error | null, json?: any | null) => {\n this._request = null;\n if (error) {\n this.fire(new ErrorEvent(error));\n } else if (json) {\n this._load(json, options, previousStyle);\n }\n });\n }\n\n loadJSON(json: StyleSpecification, options: StyleSetterOptions & StyleSwapOptions = {}, previousStyle?: StyleSpecification) {\n this.fire(new Event('dataloading', {dataType: 'style'}));\n\n this._request = browser.frame(() => {\n this._request = null;\n options.validate = options.validate !== false;\n this._load(json, options, previousStyle);\n });\n }\n\n loadEmpty() {\n this.fire(new Event('dataloading', {dataType: 'style'}));\n this._load(empty, {validate: false});\n }\n\n _load(json: StyleSpecification, options: StyleSwapOptions & StyleSetterOptions, previousStyle?: StyleSpecification) {\n const nextState = options.transformStyle ? options.transformStyle(previousStyle, json) : json;\n if (options.validate && emitValidationErrors(this, validateStyle(nextState))) {\n return;\n }\n\n this._loaded = true;\n this.stylesheet = nextState;\n\n for (const id in nextState.sources) {\n this.addSource(id, nextState.sources[id], {validate: false});\n }\n\n if (nextState.sprite) {\n this._loadSprite(nextState.sprite);\n } else {\n this.imageManager.setLoaded(true);\n }\n\n this.glyphManager.setURL(nextState.glyphs);\n this._createLayers();\n\n this.light = new Light(this.stylesheet.light);\n\n this.map.setTerrain(this.stylesheet.terrain ?? null);\n\n this.fire(new Event('data', {dataType: 'style'}));\n this.fire(new Event('style.load'));\n }\n\n private _createLayers() {\n const dereferencedLayers = deref(this.stylesheet.layers);\n\n // Broadcast layers to workers first, so that expensive style processing (createStyleLayer)\n // can happen in parallel on both main and worker threads.\n this.dispatcher.broadcast('setLayers', dereferencedLayers);\n\n this._order = dereferencedLayers.map((layer) => layer.id);\n this._layers = {};\n\n // reset serialization field, to be populated only when needed\n this._serializedLayers = null;\n for (const layer of dereferencedLayers) {\n const styledLayer = createStyleLayer(layer);\n styledLayer.setEventedParent(this, {layer: {id: layer.id}});\n this._layers[layer.id] = styledLayer;\n }\n }\n\n _loadSprite(sprite: SpriteSpecification, isUpdate: boolean = false, completion: (err: Error) => void = undefined) {\n this.imageManager.setLoaded(false);\n\n this._spriteRequest = loadSprite(sprite, this.map._requestManager, this.map.getPixelRatio(), (err, images) => {\n this._spriteRequest = null;\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (images) {\n for (const spriteId in images) {\n this._spritesImagesIds[spriteId] = [];\n\n // remove old sprite's loaded images (for the same sprite id) that are not in new sprite\n const imagesToRemove = this._spritesImagesIds[spriteId] ? this._spritesImagesIds[spriteId].filter(id => !(id in images)) : [];\n for (const id of imagesToRemove) {\n this.imageManager.removeImage(id);\n this._changedImages[id] = true;\n }\n\n for (const id in images[spriteId]) {\n // don't prefix images of the \"default\" sprite\n const imageId = spriteId === 'default' ? id : `${spriteId}:${id}`;\n // save all the sprite's images' ids to be able to delete them in `removeSprite`\n this._spritesImagesIds[spriteId].push(imageId);\n if (imageId in this.imageManager.images) {\n this.imageManager.updateImage(imageId, images[spriteId][id], false);\n } else {\n this.imageManager.addImage(imageId, images[spriteId][id]);\n }\n\n if (isUpdate) {\n this._changedImages[imageId] = true;\n }\n }\n }\n }\n\n this.imageManager.setLoaded(true);\n this._availableImages = this.imageManager.listImages();\n\n if (isUpdate) {\n this._changed = true;\n }\n\n this.dispatcher.broadcast('setImages', this._availableImages);\n this.fire(new Event('data', {dataType: 'style'}));\n\n if (completion) {\n completion(err);\n }\n });\n }\n\n _unloadSprite() {\n for (const id of Object.values(this._spritesImagesIds).flat()) {\n this.imageManager.removeImage(id);\n this._changedImages[id] = true;\n }\n\n this._spritesImagesIds = {};\n this._availableImages = this.imageManager.listImages();\n this._changed = true;\n this.dispatcher.broadcast('setImages', this._availableImages);\n this.fire(new Event('data', {dataType: 'style'}));\n }\n\n _validateLayer(layer: StyleLayer) {\n const sourceCache = this.sourceCaches[layer.source];\n if (!sourceCache) {\n return;\n }\n\n const sourceLayer = layer.sourceLayer;\n if (!sourceLayer) {\n return;\n }\n\n const source = sourceCache.getSource();\n if (source.type === 'geojson' || (source.vectorLayerIds && source.vectorLayerIds.indexOf(sourceLayer) === -1)) {\n this.fire(new ErrorEvent(new Error(\n `Source layer \"${sourceLayer}\" ` +\n `does not exist on source \"${source.id}\" ` +\n `as specified by style layer \"${layer.id}\".`\n )));\n }\n }\n\n loaded() {\n if (!this._loaded)\n return false;\n\n if (Object.keys(this._updatedSources).length)\n return false;\n\n for (const id in this.sourceCaches)\n if (!this.sourceCaches[id].loaded())\n return false;\n\n if (!this.imageManager.isLoaded())\n return false;\n\n return true;\n }\n\n /**\n * take an array of string IDs, and based on this._layers, generate an array of LayerSpecification\n * @param ids - an array of string IDs, for which serialized layers will be generated. If omitted, all serialized layers will be returned\n * @returns generated result\n */\n private _serializeByIds(ids?: Array): Array {\n\n const serializedLayersDictionary = this._serializedAllLayers();\n if (!ids || ids.length === 0) {\n return Object.values(serializedLayersDictionary);\n }\n\n const serializedLayers = [];\n for (const id of ids) {\n // this check will skip all custom layers\n if (serializedLayersDictionary[id]) {\n serializedLayers.push(serializedLayersDictionary[id]);\n }\n }\n\n return serializedLayers;\n }\n\n /**\n * Lazy initialization of this._serializedLayers dictionary and return it\n * @returns this._serializedLayers dictionary\n */\n private _serializedAllLayers(): {[_: string]: LayerSpecification} {\n let serializedLayers = this._serializedLayers;\n if (serializedLayers) {\n return serializedLayers;\n }\n\n serializedLayers = this._serializedLayers = {};\n const allLayerIds: string [] = Object.keys(this._layers);\n for (const layerId of allLayerIds) {\n const layer = this._layers[layerId];\n if (layer.type !== 'custom') {\n serializedLayers[layerId] = layer.serialize();\n }\n }\n\n return serializedLayers;\n }\n\n hasTransitions() {\n if (this.light && this.light.hasTransition()) {\n return true;\n }\n\n for (const id in this.sourceCaches) {\n if (this.sourceCaches[id].hasTransition()) {\n return true;\n }\n }\n\n for (const id in this._layers) {\n if (this._layers[id].hasTransition()) {\n return true;\n }\n }\n\n return false;\n }\n\n _checkLoaded() {\n if (!this._loaded) {\n throw new Error('Style is not done loading.');\n }\n }\n\n /**\n * @internal\n * Apply queued style updates in a batch and recalculate zoom-dependent paint properties.\n */\n update(parameters: EvaluationParameters) {\n if (!this._loaded) {\n return;\n }\n\n const changed = this._changed;\n if (this._changed) {\n const updatedIds = Object.keys(this._updatedLayers);\n const removedIds = Object.keys(this._removedLayers);\n\n if (updatedIds.length || removedIds.length) {\n this._updateWorkerLayers(updatedIds, removedIds);\n }\n for (const id in this._updatedSources) {\n const action = this._updatedSources[id];\n\n if (action === 'reload') {\n this._reloadSource(id);\n } else if (action === 'clear') {\n this._clearSource(id);\n } else {\n throw new Error(`Invalid action ${action}`);\n }\n }\n\n this._updateTilesForChangedImages();\n this._updateTilesForChangedGlyphs();\n\n for (const id in this._updatedPaintProps) {\n this._layers[id].updateTransitions(parameters);\n }\n\n this.light.updateTransitions(parameters);\n\n this._resetUpdates();\n }\n\n const sourcesUsedBefore = {};\n\n for (const sourceId in this.sourceCaches) {\n const sourceCache = this.sourceCaches[sourceId];\n sourcesUsedBefore[sourceId] = sourceCache.used;\n sourceCache.used = false;\n }\n\n for (const layerId of this._order) {\n const layer = this._layers[layerId];\n\n layer.recalculate(parameters, this._availableImages);\n if (!layer.isHidden(parameters.zoom) && layer.source) {\n this.sourceCaches[layer.source].used = true;\n }\n }\n\n for (const sourceId in sourcesUsedBefore) {\n const sourceCache = this.sourceCaches[sourceId];\n if (sourcesUsedBefore[sourceId] !== sourceCache.used) {\n sourceCache.fire(new Event('data', {sourceDataType: 'visibility', dataType: 'source', sourceId}));\n }\n }\n\n this.light.recalculate(parameters);\n this.z = parameters.zoom;\n\n if (changed) {\n this.fire(new Event('data', {dataType: 'style'}));\n }\n\n }\n\n /*\n * Apply any queued image changes.\n */\n _updateTilesForChangedImages() {\n const changedImages = Object.keys(this._changedImages);\n if (changedImages.length) {\n for (const name in this.sourceCaches) {\n this.sourceCaches[name].reloadTilesForDependencies(['icons', 'patterns'], changedImages);\n }\n this._changedImages = {};\n }\n }\n\n _updateTilesForChangedGlyphs() {\n if (this._glyphsDidChange) {\n for (const name in this.sourceCaches) {\n this.sourceCaches[name].reloadTilesForDependencies(['glyphs'], ['']);\n }\n this._glyphsDidChange = false;\n }\n }\n\n _updateWorkerLayers(updatedIds: Array, removedIds: Array) {\n this.dispatcher.broadcast('updateLayers', {\n layers: this._serializeByIds(updatedIds),\n removedIds\n });\n }\n\n _resetUpdates() {\n this._changed = false;\n\n this._updatedLayers = {};\n this._removedLayers = {};\n\n this._updatedSources = {};\n this._updatedPaintProps = {};\n\n this._changedImages = {};\n this._glyphsDidChange = false;\n }\n\n /**\n * Update this style's state to match the given style JSON, performing only\n * the necessary mutations.\n *\n * May throw an Error ('Unimplemented: METHOD') if the mapbox-gl-style-spec\n * diff algorithm produces an operation that is not supported.\n *\n * @returns true if any changes were made; false otherwise\n */\n setState(nextState: StyleSpecification, options: StyleSwapOptions = {}) {\n this._checkLoaded();\n\n const serializedStyle = this.serialize();\n nextState = options.transformStyle ? options.transformStyle(serializedStyle, nextState) : nextState;\n if (emitValidationErrors(this, validateStyle(nextState))) return false;\n\n nextState = clone(nextState);\n nextState.layers = deref(nextState.layers);\n\n const changes = diffStyles(serializedStyle, nextState)\n .filter(op => !(op.command in ignoredDiffOperations));\n\n if (changes.length === 0) {\n return false;\n }\n\n const unimplementedOps = changes.filter(op => !(op.command in supportedDiffOperations));\n if (unimplementedOps.length > 0) {\n throw new Error(`Unimplemented: ${unimplementedOps.map(op => op.command).join(', ')}.`);\n }\n\n for (const op of changes) {\n if (op.command === 'setTransition') {\n // `transition` is always read directly off of\n // `this.stylesheet`, which we update below\n continue;\n }\n (this as any)[op.command].apply(this, op.args);\n }\n\n this.stylesheet = nextState;\n\n // reset serialization field, to be populated only when needed\n this._serializedLayers = null;\n\n return true;\n }\n\n addImage(id: string, image: StyleImage) {\n if (this.getImage(id)) {\n return this.fire(new ErrorEvent(new Error(`An image named \"${id}\" already exists.`)));\n }\n this.imageManager.addImage(id, image);\n this._afterImageUpdated(id);\n }\n\n updateImage(id: string, image: StyleImage) {\n this.imageManager.updateImage(id, image);\n }\n\n getImage(id: string): StyleImage {\n return this.imageManager.getImage(id);\n }\n\n removeImage(id: string) {\n if (!this.getImage(id)) {\n return this.fire(new ErrorEvent(new Error(`An image named \"${id}\" does not exist.`)));\n }\n this.imageManager.removeImage(id);\n this._afterImageUpdated(id);\n }\n\n _afterImageUpdated(id: string) {\n this._availableImages = this.imageManager.listImages();\n this._changedImages[id] = true;\n this._changed = true;\n this.dispatcher.broadcast('setImages', this._availableImages);\n this.fire(new Event('data', {dataType: 'style'}));\n }\n\n listImages() {\n this._checkLoaded();\n\n return this.imageManager.listImages();\n }\n\n addSource(id: string, source: SourceSpecification, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n if (this.sourceCaches[id] !== undefined) {\n throw new Error(`Source \"${id}\" already exists.`);\n }\n\n if (!source.type) {\n throw new Error(`The type property must be defined, but only the following properties were given: ${Object.keys(source).join(', ')}.`);\n }\n\n const builtIns = ['vector', 'raster', 'geojson', 'video', 'image'];\n const shouldValidate = builtIns.indexOf(source.type) >= 0;\n if (shouldValidate && this._validate(validateStyle.source, `sources.${id}`, source, null, options)) return;\n\n if (this.map && this.map._collectResourceTiming) (source as any).collectResourceTiming = true;\n const sourceCache = this.sourceCaches[id] = new SourceCache(id, source, this.dispatcher);\n sourceCache.style = this;\n sourceCache.setEventedParent(this, () => ({\n isSourceLoaded: sourceCache.loaded(),\n source: sourceCache.serialize(),\n sourceId: id\n }));\n\n sourceCache.onAdd(this.map);\n this._changed = true;\n }\n\n /**\n * Remove a source from this stylesheet, given its id.\n * @param id - id of the source to remove\n * @throws if no source is found with the given ID\n * @returns `this`.\n */\n removeSource(id: string): this {\n this._checkLoaded();\n\n if (this.sourceCaches[id] === undefined) {\n throw new Error('There is no source with this ID');\n }\n for (const layerId in this._layers) {\n if (this._layers[layerId].source === id) {\n return this.fire(new ErrorEvent(new Error(`Source \"${id}\" cannot be removed while layer \"${layerId}\" is using it.`)));\n }\n }\n\n const sourceCache = this.sourceCaches[id];\n delete this.sourceCaches[id];\n delete this._updatedSources[id];\n sourceCache.fire(new Event('data', {sourceDataType: 'metadata', dataType: 'source', sourceId: id}));\n sourceCache.setEventedParent(null);\n sourceCache.onRemove(this.map);\n this._changed = true;\n }\n\n /**\n * Set the data of a GeoJSON source, given its id.\n * @param id - id of the source\n * @param data - GeoJSON source\n */\n setGeoJSONSourceData(id: string, data: GeoJSON.GeoJSON | string) {\n this._checkLoaded();\n\n if (this.sourceCaches[id] === undefined) throw new Error(`There is no source with this ID=${id}`);\n const geojsonSource: GeoJSONSource = (this.sourceCaches[id].getSource() as any);\n if (geojsonSource.type !== 'geojson') throw new Error(`geojsonSource.type is ${geojsonSource.type}, which is !== 'geojson`);\n\n geojsonSource.setData(data);\n this._changed = true;\n }\n\n /**\n * Get a source by ID.\n * @param id - ID of the desired source\n * @returns source\n */\n getSource(id: string): Source | undefined {\n return this.sourceCaches[id] && this.sourceCaches[id].getSource();\n }\n\n /**\n * Add a layer to the map style. The layer will be inserted before the layer with\n * ID `before`, or appended if `before` is omitted.\n * @param layerObject - The style layer to add.\n * @param before - ID of an existing layer to insert before\n * @param options - Style setter options.\n * @returns `this`.\n */\n addLayer(layerObject: AddLayerObject, before?: string, options: StyleSetterOptions = {}): this {\n this._checkLoaded();\n\n const id = layerObject.id;\n\n if (this.getLayer(id)) {\n this.fire(new ErrorEvent(new Error(`Layer \"${id}\" already exists on this map.`)));\n return;\n }\n\n let layer: ReturnType;\n if (layerObject.type === 'custom') {\n\n if (emitValidationErrors(this, validateCustomStyleLayer(layerObject))) return;\n\n layer = createStyleLayer(layerObject);\n\n } else {\n if ('source' in layerObject && typeof layerObject.source === 'object') {\n this.addSource(id, layerObject.source);\n layerObject = clone(layerObject);\n layerObject = extend(layerObject, {source: id});\n }\n\n // this layer is not in the style.layers array, so we pass an impossible array index\n if (this._validate(validateStyle.layer,\n `layers.${id}`, layerObject, {arrayIndex: -1}, options)) return;\n\n layer = createStyleLayer(layerObject as LayerSpecification | CustomLayerInterface);\n this._validateLayer(layer);\n\n layer.setEventedParent(this, {layer: {id}});\n }\n\n const index = before ? this._order.indexOf(before) : this._order.length;\n if (before && index === -1) {\n this.fire(new ErrorEvent(new Error(`Cannot add layer \"${id}\" before non-existing layer \"${before}\".`)));\n return;\n }\n\n this._order.splice(index, 0, id);\n this._layerOrderChanged = true;\n\n this._layers[id] = layer;\n\n if (this._removedLayers[id] && layer.source && layer.type !== 'custom') {\n // If, in the current batch, we have already removed this layer\n // and we are now re-adding it with a different `type`, then we\n // need to clear (rather than just reload) the underlying source's\n // tiles. Otherwise, tiles marked 'reloading' will have buckets /\n // buffers that are set up for the _previous_ version of this\n // layer, causing, e.g.:\n // https://github.com/mapbox/mapbox-gl-js/issues/3633\n const removed = this._removedLayers[id];\n delete this._removedLayers[id];\n if (removed.type !== layer.type) {\n this._updatedSources[layer.source] = 'clear';\n } else {\n this._updatedSources[layer.source] = 'reload';\n this.sourceCaches[layer.source].pause();\n }\n }\n this._updateLayer(layer);\n\n if (layer.onAdd) {\n layer.onAdd(this.map);\n }\n }\n\n /**\n * Moves a layer to a different z-position. The layer will be inserted before the layer with\n * ID `before`, or appended if `before` is omitted.\n * @param id - ID of the layer to move\n * @param before - ID of an existing layer to insert before\n */\n moveLayer(id: string, before?: string) {\n this._checkLoaded();\n this._changed = true;\n\n const layer = this._layers[id];\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`The layer '${id}' does not exist in the map's style and cannot be moved.`)));\n return;\n }\n\n if (id === before) {\n return;\n }\n\n const index = this._order.indexOf(id);\n this._order.splice(index, 1);\n\n const newIndex = before ? this._order.indexOf(before) : this._order.length;\n if (before && newIndex === -1) {\n this.fire(new ErrorEvent(new Error(`Cannot move layer \"${id}\" before non-existing layer \"${before}\".`)));\n return;\n }\n this._order.splice(newIndex, 0, id);\n\n this._layerOrderChanged = true;\n }\n\n /**\n * Remove the layer with the given id from the style.\n *\n * If no such layer exists, an `error` event is fired.\n *\n * @param id - id of the layer to remove\n * @event `error` - Fired if the layer does not exist\n */\n removeLayer(id: string) {\n this._checkLoaded();\n\n const layer = this._layers[id];\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot remove non-existing layer \"${id}\".`)));\n return;\n }\n\n layer.setEventedParent(null);\n\n const index = this._order.indexOf(id);\n this._order.splice(index, 1);\n\n this._layerOrderChanged = true;\n this._changed = true;\n this._removedLayers[id] = layer;\n delete this._layers[id];\n\n if (this._serializedLayers) {\n delete this._serializedLayers[id];\n }\n delete this._updatedLayers[id];\n delete this._updatedPaintProps[id];\n\n if (layer.onRemove) {\n layer.onRemove(this.map);\n }\n }\n\n /**\n * Return the style layer object with the given `id`.\n *\n * @param id - id of the desired layer\n * @returns a layer, if one with the given `id` exists\n */\n getLayer(id: string): StyleLayer | undefined {\n return this._layers[id];\n }\n\n /**\n * Return the ids of all layers currently in the style, including custom layers, in order.\n *\n * @returns ids of layers, in order\n */\n getLayersOrder(): string[] {\n return [...this._order];\n }\n\n /**\n * Checks if a specific layer is present within the style.\n *\n * @param id - the id of the desired layer\n * @returns a boolean specifying if the given layer is present\n */\n hasLayer(id: string): boolean {\n return id in this._layers;\n }\n\n setLayerZoomRange(layerId: string, minzoom?: number | null, maxzoom?: number | null) {\n this._checkLoaded();\n\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot set the zoom range of non-existing layer \"${layerId}\".`)));\n return;\n }\n\n if (layer.minzoom === minzoom && layer.maxzoom === maxzoom) return;\n\n if (minzoom != null) {\n layer.minzoom = minzoom;\n }\n if (maxzoom != null) {\n layer.maxzoom = maxzoom;\n }\n this._updateLayer(layer);\n }\n\n setFilter(layerId: string, filter?: FilterSpecification | null, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot filter non-existing layer \"${layerId}\".`)));\n return;\n }\n\n if (deepEqual(layer.filter, filter)) {\n return;\n }\n\n if (filter === null || filter === undefined) {\n layer.filter = undefined;\n this._updateLayer(layer);\n return;\n }\n\n if (this._validate(validateStyle.filter, `layers.${layer.id}.filter`, filter, null, options)) {\n return;\n }\n\n layer.filter = clone(filter);\n this._updateLayer(layer);\n }\n\n /**\n * Get a layer's filter object\n * @param layer - the layer to inspect\n * @returns the layer's filter, if any\n */\n getFilter(layer: string): FilterSpecification | void {\n return clone(this.getLayer(layer).filter);\n }\n\n setLayoutProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot style non-existing layer \"${layerId}\".`)));\n return;\n }\n\n if (deepEqual(layer.getLayoutProperty(name), value)) return;\n\n layer.setLayoutProperty(name, value, options);\n this._updateLayer(layer);\n }\n\n /**\n * Get a layout property's value from a given layer\n * @param layerId - the layer to inspect\n * @param name - the name of the layout property\n * @returns the property value\n */\n getLayoutProperty(layerId: string, name: string) {\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot get style of non-existing layer \"${layerId}\".`)));\n return;\n }\n\n return layer.getLayoutProperty(name);\n }\n\n setPaintProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot style non-existing layer \"${layerId}\".`)));\n return;\n }\n\n if (deepEqual(layer.getPaintProperty(name), value)) return;\n\n const requiresRelayout = layer.setPaintProperty(name, value, options);\n if (requiresRelayout) {\n this._updateLayer(layer);\n }\n\n this._changed = true;\n this._updatedPaintProps[layerId] = true;\n }\n\n getPaintProperty(layer: string, name: string) {\n return this.getLayer(layer).getPaintProperty(name);\n }\n\n setFeatureState(target: FeatureIdentifier, state: any) {\n this._checkLoaded();\n const sourceId = target.source;\n const sourceLayer = target.sourceLayer;\n const sourceCache = this.sourceCaches[sourceId];\n\n if (sourceCache === undefined) {\n this.fire(new ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`)));\n return;\n }\n const sourceType = sourceCache.getSource().type;\n if (sourceType === 'geojson' && sourceLayer) {\n this.fire(new ErrorEvent(new Error('GeoJSON sources cannot have a sourceLayer parameter.')));\n return;\n }\n if (sourceType === 'vector' && !sourceLayer) {\n this.fire(new ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.')));\n return;\n }\n if (target.id === undefined) {\n this.fire(new ErrorEvent(new Error('The feature id parameter must be provided.')));\n }\n\n sourceCache.setFeatureState(sourceLayer, target.id, state);\n }\n\n removeFeatureState(target: FeatureIdentifier, key?: string) {\n this._checkLoaded();\n const sourceId = target.source;\n const sourceCache = this.sourceCaches[sourceId];\n\n if (sourceCache === undefined) {\n this.fire(new ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`)));\n return;\n }\n\n const sourceType = sourceCache.getSource().type;\n const sourceLayer = sourceType === 'vector' ? target.sourceLayer : undefined;\n\n if (sourceType === 'vector' && !sourceLayer) {\n this.fire(new ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.')));\n return;\n }\n\n if (key && (typeof target.id !== 'string' && typeof target.id !== 'number')) {\n this.fire(new ErrorEvent(new Error('A feature id is required to remove its specific state property.')));\n return;\n }\n\n sourceCache.removeFeatureState(sourceLayer, target.id, key);\n }\n\n getFeatureState(target: FeatureIdentifier) {\n this._checkLoaded();\n const sourceId = target.source;\n const sourceLayer = target.sourceLayer;\n const sourceCache = this.sourceCaches[sourceId];\n\n if (sourceCache === undefined) {\n this.fire(new ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`)));\n return;\n }\n const sourceType = sourceCache.getSource().type;\n if (sourceType === 'vector' && !sourceLayer) {\n this.fire(new ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.')));\n return;\n }\n if (target.id === undefined) {\n this.fire(new ErrorEvent(new Error('The feature id parameter must be provided.')));\n }\n\n return sourceCache.getFeatureState(sourceLayer, target.id);\n }\n\n getTransition() {\n return extend({duration: 300, delay: 0}, this.stylesheet && this.stylesheet.transition);\n }\n\n serialize(): StyleSpecification {\n // We return undefined before we're loaded, following the pattern of Map.getStyle() before\n // the Style object is initialized.\n // Internally, Style._validate() calls Style.serialize() but callers are responsible for\n // calling Style._checkLoaded() first if their validation requires the style to be loaded.\n if (!this._loaded) return;\n\n const sources = mapObject(this.sourceCaches, (source) => source.serialize());\n const layers = this._serializeByIds(this._order);\n const terrain = this.map.getTerrain() || undefined;\n const myStyleSheet = this.stylesheet;\n\n return filterObject({\n version: myStyleSheet.version,\n name: myStyleSheet.name,\n metadata: myStyleSheet.metadata,\n light: myStyleSheet.light,\n center: myStyleSheet.center,\n zoom: myStyleSheet.zoom,\n bearing: myStyleSheet.bearing,\n pitch: myStyleSheet.pitch,\n sprite: myStyleSheet.sprite,\n glyphs: myStyleSheet.glyphs,\n transition: myStyleSheet.transition,\n sources,\n layers,\n terrain\n },\n (value) => { return value !== undefined; });\n }\n\n _updateLayer(layer: StyleLayer) {\n this._updatedLayers[layer.id] = true;\n if (layer.source && !this._updatedSources[layer.source] &&\n //Skip for raster layers (https://github.com/mapbox/mapbox-gl-js/issues/7865)\n this.sourceCaches[layer.source].getSource().type !== 'raster') {\n this._updatedSources[layer.source] = 'reload';\n this.sourceCaches[layer.source].pause();\n }\n\n // upon updating, serilized layer dictionary should be reset.\n // When needed, it will be populated with the correct copy again.\n this._serializedLayers = null;\n this._changed = true;\n }\n\n _flattenAndSortRenderedFeatures(sourceResults: Array<{ [key: string]: Array<{featureIndex: number; feature: MapGeoJSONFeature}> }>) {\n // Feature order is complicated.\n // The order between features in two 2D layers is always determined by layer order.\n // The order between features in two 3D layers is always determined by depth.\n // The order between a feature in a 2D layer and a 3D layer is tricky:\n // Most often layer order determines the feature order in this case. If\n // a line layer is above a extrusion layer the line feature will be rendered\n // above the extrusion. If the line layer is below the extrusion layer,\n // it will be rendered below it.\n //\n // There is a weird case though.\n // You have layers in this order: extrusion_layer_a, line_layer, extrusion_layer_b\n // Each layer has a feature that overlaps the other features.\n // The feature in extrusion_layer_a is closer than the feature in extrusion_layer_b so it is rendered above.\n // The feature in line_layer is rendered above extrusion_layer_a.\n // This means that that the line_layer feature is above the extrusion_layer_b feature despite\n // it being in an earlier layer.\n\n const isLayer3D = layerId => this._layers[layerId].type === 'fill-extrusion';\n\n const layerIndex = {};\n const features3D = [];\n for (let l = this._order.length - 1; l >= 0; l--) {\n const layerId = this._order[l];\n if (isLayer3D(layerId)) {\n layerIndex[layerId] = l;\n for (const sourceResult of sourceResults) {\n const layerFeatures = sourceResult[layerId];\n if (layerFeatures) {\n for (const featureWrapper of layerFeatures) {\n features3D.push(featureWrapper);\n }\n }\n }\n }\n }\n\n features3D.sort((a, b) => {\n return b.intersectionZ - a.intersectionZ;\n });\n\n const features = [];\n for (let l = this._order.length - 1; l >= 0; l--) {\n const layerId = this._order[l];\n\n if (isLayer3D(layerId)) {\n // add all 3D features that are in or above the current layer\n for (let i = features3D.length - 1; i >= 0; i--) {\n const topmost3D = features3D[i].feature;\n if (layerIndex[topmost3D.layer.id] < l) break;\n features.push(topmost3D);\n features3D.pop();\n }\n } else {\n for (const sourceResult of sourceResults) {\n const layerFeatures = sourceResult[layerId];\n if (layerFeatures) {\n for (const featureWrapper of layerFeatures) {\n features.push(featureWrapper.feature);\n }\n }\n }\n }\n }\n\n return features;\n }\n\n queryRenderedFeatures(queryGeometry: any, params: QueryRenderedFeaturesOptions, transform: Transform) {\n if (params && params.filter) {\n this._validate(validateStyle.filter, 'queryRenderedFeatures.filter', params.filter, null, params);\n }\n\n const includedSources = {};\n if (params && params.layers) {\n if (!Array.isArray(params.layers)) {\n this.fire(new ErrorEvent(new Error('parameters.layers must be an Array.')));\n return [];\n }\n for (const layerId of params.layers) {\n const layer = this._layers[layerId];\n if (!layer) {\n // this layer is not in the style.layers array\n this.fire(new ErrorEvent(new Error(`The layer '${layerId}' does not exist in the map's style and cannot be queried for features.`)));\n return [];\n }\n includedSources[layer.source] = true;\n }\n }\n\n const sourceResults = [];\n\n params.availableImages = this._availableImages;\n\n // LayerSpecification is serialized StyleLayer, and this casting is safe.\n const serializedLayers = this._serializedAllLayers() as {[_: string]: StyleLayer};\n\n for (const id in this.sourceCaches) {\n if (params.layers && !includedSources[id]) continue;\n sourceResults.push(\n queryRenderedFeatures(\n this.sourceCaches[id],\n this._layers,\n serializedLayers,\n queryGeometry,\n params,\n transform)\n );\n }\n\n if (this.placement) {\n // If a placement has run, query against its CollisionIndex\n // for symbol results, and treat it as an extra source to merge\n sourceResults.push(\n queryRenderedSymbols(\n this._layers,\n serializedLayers,\n this.sourceCaches,\n queryGeometry,\n params,\n this.placement.collisionIndex,\n this.placement.retainedQueryData)\n );\n }\n\n return this._flattenAndSortRenderedFeatures(sourceResults);\n }\n\n querySourceFeatures(\n sourceID: string,\n params?: QuerySourceFeatureOptions\n ) {\n if (params && params.filter) {\n this._validate(validateStyle.filter, 'querySourceFeatures.filter', params.filter, null, params);\n }\n const sourceCache = this.sourceCaches[sourceID];\n return sourceCache ? querySourceFeatures(sourceCache, params) : [];\n }\n\n addSourceType(name: string, SourceType: SourceClass, callback: Callback) {\n if (getSourceType(name)) {\n return callback(new Error(`A source type called \"${name}\" already exists.`));\n }\n\n setSourceType(name, SourceType);\n\n if (!SourceType.workerSourceURL) {\n return callback(null, null);\n }\n\n this.dispatcher.broadcast('loadWorkerSource', {\n name,\n url: SourceType.workerSourceURL\n }, callback);\n }\n\n getLight() {\n return this.light.getLight();\n }\n\n setLight(lightOptions: LightSpecification, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n const light = this.light.getLight();\n let _update = false;\n for (const key in lightOptions) {\n if (!deepEqual(lightOptions[key], light[key])) {\n _update = true;\n break;\n }\n }\n if (!_update) return;\n\n const parameters = {\n now: browser.now(),\n transition: extend({\n duration: 300,\n delay: 0\n }, this.stylesheet.transition)\n };\n\n this.light.setLight(lightOptions, options);\n this.light.updateTransitions(parameters);\n }\n\n _validate(validate: Validator, key: string, value: any, props: any, options: {\n validate?: boolean;\n } = {}) {\n if (options && options.validate === false) {\n return false;\n }\n return emitValidationErrors(this, validate.call(validateStyle, extend({\n key,\n style: this.serialize(),\n value,\n styleSpec\n }, props)));\n }\n\n _remove(mapRemoved: boolean = true) {\n if (this._request) {\n this._request.cancel();\n this._request = null;\n }\n if (this._spriteRequest) {\n this._spriteRequest.cancel();\n this._spriteRequest = null;\n }\n rtlTextPluginEvented.off('pluginStateChange', this._rtlTextPluginCallback);\n for (const layerId in this._layers) {\n const layer: StyleLayer = this._layers[layerId];\n layer.setEventedParent(null);\n }\n for (const id in this.sourceCaches) {\n const sourceCache = this.sourceCaches[id];\n sourceCache.setEventedParent(null);\n sourceCache.onRemove(this.map);\n }\n this.imageManager.setEventedParent(null);\n this.setEventedParent(null);\n this.dispatcher.remove(mapRemoved);\n }\n\n _clearSource(id: string) {\n this.sourceCaches[id].clearTiles();\n }\n\n _reloadSource(id: string) {\n this.sourceCaches[id].resume();\n this.sourceCaches[id].reload();\n }\n\n _updateSources(transform: Transform) {\n for (const id in this.sourceCaches) {\n this.sourceCaches[id].update(transform, this.map.terrain);\n }\n }\n\n _generateCollisionBoxes() {\n for (const id in this.sourceCaches) {\n this._reloadSource(id);\n }\n }\n\n _updatePlacement(transform: Transform, showCollisionBoxes: boolean, fadeDuration: number, crossSourceCollisions: boolean, forceFullPlacement: boolean = false) {\n let symbolBucketsChanged = false;\n let placementCommitted = false;\n\n const layerTiles = {};\n\n for (const layerID of this._order) {\n const styleLayer = this._layers[layerID];\n if (styleLayer.type !== 'symbol') continue;\n\n if (!layerTiles[styleLayer.source]) {\n const sourceCache = this.sourceCaches[styleLayer.source];\n layerTiles[styleLayer.source] = sourceCache.getRenderableIds(true)\n .map((id) => sourceCache.getTileByID(id))\n .sort((a, b) => (b.tileID.overscaledZ - a.tileID.overscaledZ) || (a.tileID.isLessThan(b.tileID) ? -1 : 1));\n }\n\n const layerBucketsChanged = this.crossTileSymbolIndex.addLayer(styleLayer, layerTiles[styleLayer.source], transform.center.lng);\n symbolBucketsChanged = symbolBucketsChanged || layerBucketsChanged;\n }\n this.crossTileSymbolIndex.pruneUnusedLayers(this._order);\n\n // Anything that changes our \"in progress\" layer and tile indices requires us\n // to start over. When we start over, we do a full placement instead of incremental\n // to prevent starvation.\n // We need to restart placement to keep layer indices in sync.\n // Also force full placement when fadeDuration === 0 to ensure that newly loaded\n // tiles will fully display symbols in their first frame\n forceFullPlacement = forceFullPlacement || this._layerOrderChanged || fadeDuration === 0;\n\n if (forceFullPlacement || !this.pauseablePlacement || (this.pauseablePlacement.isDone() && !this.placement.stillRecent(browser.now(), transform.zoom))) {\n this.pauseablePlacement = new PauseablePlacement(transform, this.map.terrain, this._order, forceFullPlacement, showCollisionBoxes, fadeDuration, crossSourceCollisions, this.placement);\n this._layerOrderChanged = false;\n }\n\n if (this.pauseablePlacement.isDone()) {\n // the last placement finished running, but the next one hasn’t\n // started yet because of the `stillRecent` check immediately\n // above, so mark it stale to ensure that we request another\n // render frame\n this.placement.setStale();\n } else {\n this.pauseablePlacement.continuePlacement(this._order, this._layers, layerTiles);\n\n if (this.pauseablePlacement.isDone()) {\n this.placement = this.pauseablePlacement.commit(browser.now());\n placementCommitted = true;\n }\n\n if (symbolBucketsChanged) {\n // since the placement gets split over multiple frames it is possible\n // these buckets were processed before they were changed and so the\n // placement is already stale while it is in progress\n this.pauseablePlacement.placement.setStale();\n }\n }\n\n if (placementCommitted || symbolBucketsChanged) {\n for (const layerID of this._order) {\n const styleLayer = this._layers[layerID];\n if (styleLayer.type !== 'symbol') continue;\n this.placement.updateLayerOpacities(styleLayer, layerTiles[styleLayer.source]);\n }\n }\n\n // needsRender is false when we have just finished a placement that didn't change the visibility of any symbols\n const needsRerender = !this.pauseablePlacement.isDone() || this.placement.hasTransitions(browser.now());\n return needsRerender;\n }\n\n _releaseSymbolFadeTiles() {\n for (const id in this.sourceCaches) {\n this.sourceCaches[id].releaseSymbolFadeTiles();\n }\n }\n\n // Callbacks from web workers\n\n getImages(\n mapId: string,\n params: {\n icons: Array;\n source: string;\n tileID: OverscaledTileID;\n type: string;\n },\n callback: Callback<{[_: string]: StyleImage}>\n ) {\n this.imageManager.getImages(params.icons, callback);\n\n // Apply queued image changes before setting the tile's dependencies so that the tile\n // is not reloaded unnecessarily. Without this forced update the reload could happen in cases\n // like this one:\n // - icons contains \"my-image\"\n // - imageManager.getImages(...) triggers `onstyleimagemissing`\n // - the user adds \"my-image\" within the callback\n // - addImage adds \"my-image\" to this._changedImages\n // - the next frame triggers a reload of this tile even though it already has the latest version\n this._updateTilesForChangedImages();\n\n const sourceCache = this.sourceCaches[params.source];\n if (sourceCache) {\n sourceCache.setDependencies(params.tileID.key, params.type, params.icons);\n }\n }\n\n getGlyphs(\n mapId: string,\n params: {\n stacks: {[_: string]: Array};\n source: string;\n tileID: OverscaledTileID;\n type: string;\n },\n callback: Callback<{[_: string]: {[_: number]: StyleGlyph}}>\n ) {\n this.glyphManager.getGlyphs(params.stacks, callback);\n const sourceCache = this.sourceCaches[params.source];\n if (sourceCache) {\n // we are not setting stacks as dependencies since for now\n // we just need to know which tiles have glyph dependencies\n sourceCache.setDependencies(params.tileID.key, params.type, ['']);\n }\n }\n\n getResource(mapId: string, params: RequestParameters, callback: ResponseCallback): Cancelable {\n return makeRequest(params, callback);\n }\n\n getGlyphsUrl() {\n return this.stylesheet.glyphs || null;\n }\n\n setGlyphs(glyphsUrl: string | null, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n if (glyphsUrl && this._validate(validateStyle.glyphs, 'glyphs', glyphsUrl, null, options)) {\n return;\n }\n\n this._glyphsDidChange = true;\n this.stylesheet.glyphs = glyphsUrl;\n this.glyphManager.entries = {};\n this.glyphManager.setURL(glyphsUrl);\n }\n\n /**\n * Add a sprite.\n *\n * @param id - The id of the desired sprite\n * @param url - The url to load the desired sprite from\n * @param options - The style setter options\n * @param completion - The completion handler\n */\n addSprite(id: string, url: string, options: StyleSetterOptions = {}, completion?: (err: Error) => void) {\n this._checkLoaded();\n\n const spriteToAdd = [{id, url}];\n const updatedSprite = [\n ...coerceSpriteToArray(this.stylesheet.sprite),\n ...spriteToAdd\n ];\n\n if (this._validate(validateStyle.sprite, 'sprite', updatedSprite, null, options)) return;\n\n this.stylesheet.sprite = updatedSprite;\n this._loadSprite(spriteToAdd, true, completion);\n }\n\n /**\n * Remove a sprite by its id. When the last sprite is removed, the whole `this.stylesheet.sprite` object becomes\n * `undefined`. This falsy `undefined` value later prevents attempts to load the sprite when it's absent.\n *\n * @param id - the id of the sprite to remove\n */\n removeSprite(id: string) {\n this._checkLoaded();\n\n const internalSpriteRepresentation = coerceSpriteToArray(this.stylesheet.sprite);\n\n if (!internalSpriteRepresentation.find(sprite => sprite.id === id)) {\n this.fire(new ErrorEvent(new Error(`Sprite \"${id}\" doesn't exists on this map.`)));\n return;\n }\n\n if (this._spritesImagesIds[id]) {\n for (const imageId of this._spritesImagesIds[id]) {\n this.imageManager.removeImage(imageId);\n this._changedImages[imageId] = true;\n }\n }\n\n internalSpriteRepresentation.splice(internalSpriteRepresentation.findIndex(sprite => sprite.id === id), 1);\n this.stylesheet.sprite = internalSpriteRepresentation.length > 0 ? internalSpriteRepresentation : undefined;\n\n delete this._spritesImagesIds[id];\n this._availableImages = this.imageManager.listImages();\n this._changed = true;\n this.dispatcher.broadcast('setImages', this._availableImages);\n this.fire(new Event('data', {dataType: 'style'}));\n }\n\n /**\n * Get the current sprite value.\n *\n * @returns empty array when no sprite is set; id-url pairs otherwise\n */\n getSprite() {\n return coerceSpriteToArray(this.stylesheet.sprite);\n }\n\n /**\n * Set a new value for the style's sprite.\n *\n * @param sprite - new sprite value\n * @param options - style setter options\n * @param completion - the completion handler\n */\n setSprite(sprite: SpriteSpecification, options: StyleSetterOptions = {}, completion?: (err: Error) => void) {\n this._checkLoaded();\n\n if (sprite && this._validate(validateStyle.sprite, 'sprite', sprite, null, options)) {\n return;\n }\n\n this.stylesheet.sprite = sprite;\n\n if (sprite) {\n this._loadSprite(sprite, true, completion);\n } else {\n this._unloadSprite();\n if (completion) {\n completion(null);\n }\n }\n }\n}\n\nStyle.registerForPluginStateChange = registerForPluginStateChange;\n","import {createLayout} from '../util/struct_array';\n\nexport default createLayout([\n {name: 'a_pos', type: 'Int16', components: 2}\n]);\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec3 a_pos3d;uniform mat4 u_matrix;uniform float u_ele_delta;varying vec2 v_texture_pos;varying float v_depth;void main() {float extent=8192.0;float ele_delta=a_pos3d.z==1.0 ? u_ele_delta : 0.0;v_texture_pos=a_pos3d.xy/extent;gl_Position=u_matrix*vec4(a_pos3d.xy,get_elevation(a_pos3d.xy)-ele_delta,1.0);v_depth=gl_Position.z/gl_Position.w;}';\n","\n// Disable Flow annotations here because Flow doesn't support importing GLSL files\n\nimport preludeFrag from './_prelude.fragment.glsl.g';\nimport preludeVert from './_prelude.vertex.glsl.g';\nimport backgroundFrag from './background.fragment.glsl.g';\nimport backgroundVert from './background.vertex.glsl.g';\nimport backgroundPatternFrag from './background_pattern.fragment.glsl.g';\nimport backgroundPatternVert from './background_pattern.vertex.glsl.g';\nimport circleFrag from './circle.fragment.glsl.g';\nimport circleVert from './circle.vertex.glsl.g';\nimport clippingMaskFrag from './clipping_mask.fragment.glsl.g';\nimport clippingMaskVert from './clipping_mask.vertex.glsl.g';\nimport heatmapFrag from './heatmap.fragment.glsl.g';\nimport heatmapVert from './heatmap.vertex.glsl.g';\nimport heatmapTextureFrag from './heatmap_texture.fragment.glsl.g';\nimport heatmapTextureVert from './heatmap_texture.vertex.glsl.g';\nimport collisionBoxFrag from './collision_box.fragment.glsl.g';\nimport collisionBoxVert from './collision_box.vertex.glsl.g';\nimport collisionCircleFrag from './collision_circle.fragment.glsl.g';\nimport collisionCircleVert from './collision_circle.vertex.glsl.g';\nimport debugFrag from './debug.fragment.glsl.g';\nimport debugVert from './debug.vertex.glsl.g';\nimport fillFrag from './fill.fragment.glsl.g';\nimport fillVert from './fill.vertex.glsl.g';\nimport fillOutlineFrag from './fill_outline.fragment.glsl.g';\nimport fillOutlineVert from './fill_outline.vertex.glsl.g';\nimport fillOutlinePatternFrag from './fill_outline_pattern.fragment.glsl.g';\nimport fillOutlinePatternVert from './fill_outline_pattern.vertex.glsl.g';\nimport fillPatternFrag from './fill_pattern.fragment.glsl.g';\nimport fillPatternVert from './fill_pattern.vertex.glsl.g';\nimport fillExtrusionFrag from './fill_extrusion.fragment.glsl.g';\nimport fillExtrusionVert from './fill_extrusion.vertex.glsl.g';\nimport fillExtrusionPatternFrag from './fill_extrusion_pattern.fragment.glsl.g';\nimport fillExtrusionPatternVert from './fill_extrusion_pattern.vertex.glsl.g';\nimport hillshadePrepareFrag from './hillshade_prepare.fragment.glsl.g';\nimport hillshadePrepareVert from './hillshade_prepare.vertex.glsl.g';\nimport hillshadeFrag from './hillshade.fragment.glsl.g';\nimport hillshadeVert from './hillshade.vertex.glsl.g';\nimport lineFrag from './line.fragment.glsl.g';\nimport lineVert from './line.vertex.glsl.g';\nimport lineGradientFrag from './line_gradient.fragment.glsl.g';\nimport lineGradientVert from './line_gradient.vertex.glsl.g';\nimport linePatternFrag from './line_pattern.fragment.glsl.g';\nimport linePatternVert from './line_pattern.vertex.glsl.g';\nimport lineSDFFrag from './line_sdf.fragment.glsl.g';\nimport lineSDFVert from './line_sdf.vertex.glsl.g';\nimport rasterFrag from './raster.fragment.glsl.g';\nimport rasterVert from './raster.vertex.glsl.g';\nimport symbolIconFrag from './symbol_icon.fragment.glsl.g';\nimport symbolIconVert from './symbol_icon.vertex.glsl.g';\nimport symbolSDFFrag from './symbol_sdf.fragment.glsl.g';\nimport symbolSDFVert from './symbol_sdf.vertex.glsl.g';\nimport symbolTextAndIconFrag from './symbol_text_and_icon.fragment.glsl.g';\nimport symbolTextAndIconVert from './symbol_text_and_icon.vertex.glsl.g';\nimport terrainDepthFrag from './terrain_depth.fragment.glsl.g';\nimport terrainCoordsFrag from './terrain_coords.fragment.glsl.g';\nimport terrainFrag from './terrain.fragment.glsl.g';\nimport terrainVert from './terrain.vertex.glsl.g';\n\nexport const shaders = {\n prelude: compile(preludeFrag, preludeVert),\n background: compile(backgroundFrag, backgroundVert),\n backgroundPattern: compile(backgroundPatternFrag, backgroundPatternVert),\n circle: compile(circleFrag, circleVert),\n clippingMask: compile(clippingMaskFrag, clippingMaskVert),\n heatmap: compile(heatmapFrag, heatmapVert),\n heatmapTexture: compile(heatmapTextureFrag, heatmapTextureVert),\n collisionBox: compile(collisionBoxFrag, collisionBoxVert),\n collisionCircle: compile(collisionCircleFrag, collisionCircleVert),\n debug: compile(debugFrag, debugVert),\n fill: compile(fillFrag, fillVert),\n fillOutline: compile(fillOutlineFrag, fillOutlineVert),\n fillOutlinePattern: compile(fillOutlinePatternFrag, fillOutlinePatternVert),\n fillPattern: compile(fillPatternFrag, fillPatternVert),\n fillExtrusion: compile(fillExtrusionFrag, fillExtrusionVert),\n fillExtrusionPattern: compile(fillExtrusionPatternFrag, fillExtrusionPatternVert),\n hillshadePrepare: compile(hillshadePrepareFrag, hillshadePrepareVert),\n hillshade: compile(hillshadeFrag, hillshadeVert),\n line: compile(lineFrag, lineVert),\n lineGradient: compile(lineGradientFrag, lineGradientVert),\n linePattern: compile(linePatternFrag, linePatternVert),\n lineSDF: compile(lineSDFFrag, lineSDFVert),\n raster: compile(rasterFrag, rasterVert),\n symbolIcon: compile(symbolIconFrag, symbolIconVert),\n symbolSDF: compile(symbolSDFFrag, symbolSDFVert),\n symbolTextAndIcon: compile(symbolTextAndIconFrag, symbolTextAndIconVert),\n terrain: compile(terrainFrag, terrainVert),\n terrainDepth: compile(terrainDepthFrag, terrainVert),\n terrainCoords: compile(terrainCoordsFrag, terrainVert)\n};\n\n// Expand #pragmas to #ifdefs.\n\nfunction compile(fragmentSource, vertexSource) {\n const re = /#pragma mapbox: ([\\w]+) ([\\w]+) ([\\w]+) ([\\w]+)/g;\n\n const staticAttributes = vertexSource.match(/attribute ([\\w]+) ([\\w]+)/g);\n const fragmentUniforms = fragmentSource.match(/uniform ([\\w]+) ([\\w]+)([\\s]*)([\\w]*)/g);\n const vertexUniforms = vertexSource.match(/uniform ([\\w]+) ([\\w]+)([\\s]*)([\\w]*)/g);\n const staticUniforms = vertexUniforms ? vertexUniforms.concat(fragmentUniforms) : fragmentUniforms;\n\n const fragmentPragmas = {};\n\n fragmentSource = fragmentSource.replace(re, (match, operation, precision, type, name) => {\n fragmentPragmas[name] = true;\n if (operation === 'define') {\n return `\n#ifndef HAS_UNIFORM_u_${name}\nvarying ${precision} ${type} ${name};\n#else\nuniform ${precision} ${type} u_${name};\n#endif\n`;\n } else /* if (operation === 'initialize') */ {\n return `\n#ifdef HAS_UNIFORM_u_${name}\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n }\n });\n\n vertexSource = vertexSource.replace(re, (match, operation, precision, type, name) => {\n const attrType = type === 'float' ? 'vec2' : 'vec4';\n const unpackType = name.match(/color/) ? 'color' : attrType;\n\n if (fragmentPragmas[name]) {\n if (operation === 'define') {\n return `\n#ifndef HAS_UNIFORM_u_${name}\nuniform lowp float u_${name}_t;\nattribute ${precision} ${attrType} a_${name};\nvarying ${precision} ${type} ${name};\n#else\nuniform ${precision} ${type} u_${name};\n#endif\n`;\n } else /* if (operation === 'initialize') */ {\n if (unpackType === 'vec4') {\n // vec4 attributes are only used for cross-faded properties, and are not packed\n return `\n#ifndef HAS_UNIFORM_u_${name}\n ${name} = a_${name};\n#else\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n } else {\n return `\n#ifndef HAS_UNIFORM_u_${name}\n ${name} = unpack_mix_${unpackType}(a_${name}, u_${name}_t);\n#else\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n }\n }\n } else {\n if (operation === 'define') {\n return `\n#ifndef HAS_UNIFORM_u_${name}\nuniform lowp float u_${name}_t;\nattribute ${precision} ${attrType} a_${name};\n#else\nuniform ${precision} ${type} u_${name};\n#endif\n`;\n } else /* if (operation === 'initialize') */ {\n if (unpackType === 'vec4') {\n // vec4 attributes are only used for cross-faded properties, and are not packed\n return `\n#ifndef HAS_UNIFORM_u_${name}\n ${precision} ${type} ${name} = a_${name};\n#else\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n } else /* */ {\n return `\n#ifndef HAS_UNIFORM_u_${name}\n ${precision} ${type} ${name} = unpack_mix_${unpackType}(a_${name}, u_${name}_t);\n#else\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n }\n }\n }\n });\n\n return {fragmentSource, vertexSource, staticAttributes, staticUniforms};\n}\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision mediump float;\\n#else\\n#if !defined(lowp)\\n#define lowp\\n#endif\\n#if !defined(mediump)\\n#define mediump\\n#endif\\n#if !defined(highp)\\n#define highp\\n#endif\\n#endif\\n';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision highp float;\\n#else\\n#if !defined(lowp)\\n#define lowp\\n#endif\\n#if !defined(mediump)\\n#define mediump\\n#endif\\n#if !defined(highp)\\n#define highp\\n#endif\\n#endif\\nvec2 unpack_float(const float packedValue) {int packedIntValue=int(packedValue);int v0=packedIntValue/256;return vec2(v0,packedIntValue-v0*256);}vec2 unpack_opacity(const float packedOpacity) {int intOpacity=int(packedOpacity)/2;return vec2(float(intOpacity)/127.0,mod(packedOpacity,2.0));}vec4 decode_color(const vec2 encodedColor) {return vec4(unpack_float(encodedColor[0])/255.0,unpack_float(encodedColor[1])/255.0\\n);}float unpack_mix_vec2(const vec2 packedValue,const float t) {return mix(packedValue[0],packedValue[1],t);}vec4 unpack_mix_color(const vec4 packedColors,const float t) {vec4 minColor=decode_color(vec2(packedColors[0],packedColors[1]));vec4 maxColor=decode_color(vec2(packedColors[2],packedColors[3]));return mix(minColor,maxColor,t);}vec2 get_pattern_pos(const vec2 pixel_coord_upper,const vec2 pixel_coord_lower,const vec2 pattern_size,const float tile_units_to_pixels,const vec2 pos) {vec2 offset=mod(mod(mod(pixel_coord_upper,pattern_size)*256.0,pattern_size)*256.0+pixel_coord_lower,pattern_size);return (tile_units_to_pixels*pos+offset)/pattern_size;}\\n#ifdef TERRAIN3D\\nuniform sampler2D u_terrain;uniform float u_terrain_dim;uniform mat4 u_terrain_matrix;uniform vec4 u_terrain_unpack;uniform float u_terrain_exaggeration;uniform highp sampler2D u_depth;\\n#endif\\nconst highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitShifts=vec4(1.)/bitSh;highp float unpack(highp vec4 color) {return dot(color,bitShifts);}highp float depthOpacity(vec3 frag) {\\n#ifdef TERRAIN3D\\nhighp float d=unpack(texture2D(u_depth,frag.xy*0.5+0.5))+0.0001-frag.z;return 1.0-max(0.0,min(1.0,-d*500.0));\\n#else\\nreturn 1.0;\\n#endif\\n}float calculate_visibility(vec4 pos) {\\n#ifdef TERRAIN3D\\nvec3 frag=pos.xyz/pos.w;highp float d=depthOpacity(frag);if (d > 0.95) return 1.0;return (d+depthOpacity(frag+vec3(0.0,0.01,0.0)))/2.0;\\n#else\\nreturn 1.0;\\n#endif\\n}float ele(vec2 pos) {\\n#ifdef TERRAIN3D\\nvec4 rgb=(texture2D(u_terrain,pos)*255.0)*u_terrain_unpack;return rgb.r+rgb.g+rgb.b-u_terrain_unpack.a;\\n#else\\nreturn 0.0;\\n#endif\\n}float get_elevation(vec2 pos) {\\n#ifdef TERRAIN3D\\nvec2 coord=(u_terrain_matrix*vec4(pos,0.0,1.0)).xy*u_terrain_dim+1.0;vec2 f=fract(coord);vec2 c=(floor(coord)+0.5)/(u_terrain_dim+2.0);float d=1.0/(u_terrain_dim+2.0);float tl=ele(c);float tr=ele(c+vec2(d,0.0));float bl=ele(c+vec2(0.0,d));float br=ele(c+vec2(d,d));float elevation=mix(mix(tl,tr,f.x),mix(bl,br,f.x),f.y);return elevation*u_terrain_exaggeration;\\n#else\\nreturn 0.0;\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform vec4 u_color;uniform float u_opacity;void main() {gl_FragColor=u_color*u_opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform vec2 u_pattern_tl_a;uniform vec2 u_pattern_br_a;uniform vec2 u_pattern_tl_b;uniform vec2 u_pattern_br_b;uniform vec2 u_texsize;uniform float u_mix;uniform float u_opacity;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(u_pattern_tl_a/u_texsize,u_pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(u_pattern_tl_b/u_texsize,u_pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_mix)*u_opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_pattern_size_a;uniform vec2 u_pattern_size_b;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_scale_a;uniform float u_scale_b;uniform float u_tile_units_to_pixels;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_a*u_pattern_size_a,u_tile_units_to_pixels,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_b*u_pattern_size_b,u_tile_units_to_pixels,a_pos);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying vec3 v_data;varying float v_visibility;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define mediump float radius\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define highp vec4 stroke_color\\n#pragma mapbox: define mediump float stroke_width\\n#pragma mapbox: define lowp float stroke_opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize mediump float radius\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize highp vec4 stroke_color\\n#pragma mapbox: initialize mediump float stroke_width\\n#pragma mapbox: initialize lowp float stroke_opacity\\nvec2 extrude=v_data.xy;float extrude_length=length(extrude);lowp float antialiasblur=v_data.z;float antialiased_blur=-max(blur,antialiasblur);float opacity_t=smoothstep(0.0,antialiased_blur,extrude_length-1.0);float color_t=stroke_width < 0.01 ? 0.0 : smoothstep(antialiased_blur,0.0,extrude_length-radius/(radius+stroke_width));gl_FragColor=v_visibility*opacity_t*mix(color*opacity,stroke_color*stroke_opacity,color_t);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform bool u_scale_with_map;uniform bool u_pitch_with_map;uniform vec2 u_extrude_scale;uniform lowp float u_device_pixel_ratio;uniform highp float u_camera_to_center_distance;attribute vec2 a_pos;varying vec3 v_data;varying float v_visibility;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define mediump float radius\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define highp vec4 stroke_color\\n#pragma mapbox: define mediump float stroke_width\\n#pragma mapbox: define lowp float stroke_opacity\\nvoid main(void) {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize mediump float radius\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize highp vec4 stroke_color\\n#pragma mapbox: initialize mediump float stroke_width\\n#pragma mapbox: initialize lowp float stroke_opacity\\nvec2 extrude=vec2(mod(a_pos,2.0)*2.0-1.0);vec2 circle_center=floor(a_pos*0.5);float ele=get_elevation(circle_center);v_visibility=calculate_visibility(u_matrix*vec4(circle_center,ele,1.0));if (u_pitch_with_map) {vec2 corner_position=circle_center;if (u_scale_with_map) {corner_position+=extrude*(radius+stroke_width)*u_extrude_scale;} else {vec4 projected_center=u_matrix*vec4(circle_center,0,1);corner_position+=extrude*(radius+stroke_width)*u_extrude_scale*(projected_center.w/u_camera_to_center_distance);}gl_Position=u_matrix*vec4(corner_position,ele,1);} else {gl_Position=u_matrix*vec4(circle_center,ele,1);if (u_scale_with_map) {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*u_camera_to_center_distance;} else {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*gl_Position.w;}}lowp float antialiasblur=1.0/u_device_pixel_ratio/(radius+stroke_width);v_data=vec3(extrude.x,extrude.y,antialiasblur);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'void main() {gl_FragColor=vec4(1.0);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform highp float u_intensity;varying vec2 v_extrude;\\n#pragma mapbox: define highp float weight\\n#define GAUSS_COEF 0.3989422804014327\\nvoid main() {\\n#pragma mapbox: initialize highp float weight\\nfloat d=-0.5*3.0*3.0*dot(v_extrude,v_extrude);float val=weight*u_intensity*GAUSS_COEF*exp(d);gl_FragColor=vec4(val,1.0,1.0,1.0);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform float u_extrude_scale;uniform float u_opacity;uniform float u_intensity;attribute vec2 a_pos;varying vec2 v_extrude;\\n#pragma mapbox: define highp float weight\\n#pragma mapbox: define mediump float radius\\nconst highp float ZERO=1.0/255.0/16.0;\\n#define GAUSS_COEF 0.3989422804014327\\nvoid main(void) {\\n#pragma mapbox: initialize highp float weight\\n#pragma mapbox: initialize mediump float radius\\nvec2 unscaled_extrude=vec2(mod(a_pos,2.0)*2.0-1.0);float S=sqrt(-2.0*log(ZERO/weight/u_intensity/GAUSS_COEF))/3.0;v_extrude=S*unscaled_extrude;vec2 extrude=v_extrude*radius*u_extrude_scale;vec4 pos=vec4(floor(a_pos*0.5)+extrude,0,1);gl_Position=u_matrix*pos;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform sampler2D u_image;uniform sampler2D u_color_ramp;uniform float u_opacity;varying vec2 v_pos;void main() {float t=texture2D(u_image,v_pos).r;vec4 color=texture2D(u_color_ramp,vec2(t,0.5));gl_FragColor=color*u_opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(0.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_world;attribute vec2 a_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos*u_world,0,1);v_pos.x=a_pos.x;v_pos.y=1.0-a_pos.y;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying float v_placed;varying float v_notUsed;void main() {float alpha=0.5;gl_FragColor=vec4(1.0,0.0,0.0,1.0)*alpha;if (v_placed > 0.5) {gl_FragColor=vec4(0.0,0.0,1.0,0.5)*alpha;}if (v_notUsed > 0.5) {gl_FragColor*=.1;}}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;attribute vec2 a_anchor_pos;attribute vec2 a_extrude;attribute vec2 a_placed;attribute vec2 a_shift;uniform mat4 u_matrix;uniform vec2 u_extrude_scale;uniform float u_camera_to_center_distance;varying float v_placed;varying float v_notUsed;void main() {vec4 projectedPoint=u_matrix*vec4(a_anchor_pos,0,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);gl_Position=u_matrix*vec4(a_pos,get_elevation(a_pos),1.0);gl_Position.xy+=(a_extrude+a_shift)*u_extrude_scale*gl_Position.w*collision_perspective_ratio;v_placed=a_placed.x;v_notUsed=a_placed.y;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;void main() {float alpha=0.5*min(v_perspective_ratio,1.0);float stroke_radius=0.9*max(v_perspective_ratio,1.0);float distance_to_center=length(v_extrude);float distance_to_edge=abs(distance_to_center-v_radius);float opacity_t=smoothstep(-stroke_radius,0.0,-distance_to_edge);vec4 color=mix(vec4(0.0,0.0,1.0,0.5),vec4(1.0,0.0,0.0,1.0),v_collision);gl_FragColor=color*alpha*opacity_t;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;attribute float a_radius;attribute vec2 a_flags;uniform mat4 u_matrix;uniform mat4 u_inv_matrix;uniform vec2 u_viewport_size;uniform float u_camera_to_center_distance;varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;vec3 toTilePosition(vec2 screenPos) {vec4 rayStart=u_inv_matrix*vec4(screenPos,-1.0,1.0);vec4 rayEnd =u_inv_matrix*vec4(screenPos, 1.0,1.0);rayStart.xyz/=rayStart.w;rayEnd.xyz /=rayEnd.w;highp float t=(0.0-rayStart.z)/(rayEnd.z-rayStart.z);return mix(rayStart.xyz,rayEnd.xyz,t);}void main() {vec2 quadCenterPos=a_pos;float radius=a_radius;float collision=a_flags.x;float vertexIdx=a_flags.y;vec2 quadVertexOffset=vec2(mix(-1.0,1.0,float(vertexIdx >=2.0)),mix(-1.0,1.0,float(vertexIdx >=1.0 && vertexIdx <=2.0)));vec2 quadVertexExtent=quadVertexOffset*radius;vec3 tilePos=toTilePosition(quadCenterPos);vec4 clipPos=u_matrix*vec4(tilePos,1.0);highp float camera_to_anchor_distance=clipPos.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);float padding_factor=1.2;v_radius=radius;v_extrude=quadVertexExtent*padding_factor;v_perspective_ratio=collision_perspective_ratio;v_collision=collision;gl_Position=vec4(clipPos.xyz/clipPos.w,1.0)+vec4(quadVertexExtent*padding_factor/u_viewport_size*2.0,0.0,0.0);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform highp vec4 u_color;uniform sampler2D u_overlay;varying vec2 v_uv;void main() {vec4 overlay_color=texture2D(u_overlay,v_uv);gl_FragColor=mix(u_color,overlay_color,overlay_color.a);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;varying vec2 v_uv;uniform mat4 u_matrix;uniform float u_overlay_scale;void main() {v_uv=a_pos/8192.0;gl_Position=u_matrix*vec4(a_pos*u_overlay_scale,get_elevation(a_pos),1);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float opacity\\ngl_FragColor=color*opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;uniform mat4 u_matrix;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float opacity\\ngl_Position=u_matrix*vec4(a_pos,0,1);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying vec2 v_pos;\\n#pragma mapbox: define highp vec4 outline_color\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 outline_color\\n#pragma mapbox: initialize lowp float opacity\\nfloat dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=outline_color*(alpha*opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;uniform mat4 u_matrix;uniform vec2 u_world;varying vec2 v_pos;\\n#pragma mapbox: define highp vec4 outline_color\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 outline_color\\n#pragma mapbox: initialize lowp float opacity\\ngl_Position=u_matrix*vec4(a_pos,0,1);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform vec2 u_texsize;uniform sampler2D u_image;uniform float u_fade;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);float dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=mix(color1,color2,u_fade)*alpha*opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_world;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;gl_Position=u_matrix*vec4(a_pos,0,1);vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,a_pos);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision highp float;\\n#endif\\nuniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_fade)*opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileZoomRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileZoomRatio,a_pos);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying vec4 v_color;void main() {gl_FragColor=v_color;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;uniform float u_vertical_gradient;uniform lowp float u_opacity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\\n#ifdef TERRAIN3D\\nattribute vec2 a_centroid;\\n#endif\\nvarying vec4 v_color;\\n#pragma mapbox: define highp float base\\n#pragma mapbox: define highp float height\\n#pragma mapbox: define highp vec4 color\\nvoid main() {\\n#pragma mapbox: initialize highp float base\\n#pragma mapbox: initialize highp float height\\n#pragma mapbox: initialize highp vec4 color\\nvec3 normal=a_normal_ed.xyz;\\n#ifdef TERRAIN3D\\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\\n#else\\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\\n#endif\\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);gl_Position=u_matrix*vec4(a_pos,t > 0.0 ? height : base,1);float colorvalue=color.r*0.2126+color.g*0.7152+color.b*0.0722;v_color=vec4(0.0,0.0,0.0,1.0);vec4 ambientlight=vec4(0.03,0.03,0.03,1.0);color+=ambientlight;float directional=clamp(dot(normal/16384.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((1.0-colorvalue+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_color.r+=clamp(color.r*directional*u_lightcolor.r,mix(0.0,0.3,1.0-u_lightcolor.r),1.0);v_color.g+=clamp(color.g*directional*u_lightcolor.g,mix(0.0,0.3,1.0-u_lightcolor.g),1.0);v_color.b+=clamp(color.b*directional*u_lightcolor.b,mix(0.0,0.3,1.0-u_lightcolor.b),1.0);v_color*=u_opacity;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\\n#pragma mapbox: define lowp float base\\n#pragma mapbox: define lowp float height\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float base\\n#pragma mapbox: initialize lowp float height\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);vec4 mixedColor=mix(color1,color2,u_fade);gl_FragColor=mixedColor*v_lighting;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_height_factor;uniform vec3 u_scale;uniform float u_vertical_gradient;uniform lowp float u_opacity;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\\n#ifdef TERRAIN3D\\nattribute vec2 a_centroid;\\n#endif\\nvarying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\\n#pragma mapbox: define lowp float base\\n#pragma mapbox: define lowp float height\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float base\\n#pragma mapbox: initialize lowp float height\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec3 normal=a_normal_ed.xyz;float edgedistance=a_normal_ed.w;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;\\n#ifdef TERRAIN3D\\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\\n#else\\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\\n#endif\\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);float z=t > 0.0 ? height : base;gl_Position=u_matrix*vec4(a_pos,z,1);vec2 pos=normal.x==1.0 && normal.y==0.0 && normal.z==16384.0\\n? a_pos\\n: vec2(edgedistance,z*u_height_factor);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,pos);v_lighting=vec4(0.0,0.0,0.0,1.0);float directional=clamp(dot(normal/16383.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((0.5+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_lighting.rgb+=clamp(directional*u_lightcolor,mix(vec3(0.0),vec3(0.3),1.0-u_lightcolor),vec3(1.0));v_lighting*=u_opacity;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision highp float;\\n#endif\\nuniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_dimension;uniform float u_zoom;uniform vec4 u_unpack;float getElevation(vec2 coord,float bias) {vec4 data=texture2D(u_image,coord)*255.0;data.a=-1.0;return dot(data,u_unpack)/4.0;}void main() {vec2 epsilon=1.0/u_dimension;float a=getElevation(v_pos+vec2(-epsilon.x,-epsilon.y),0.0);float b=getElevation(v_pos+vec2(0,-epsilon.y),0.0);float c=getElevation(v_pos+vec2(epsilon.x,-epsilon.y),0.0);float d=getElevation(v_pos+vec2(-epsilon.x,0),0.0);float e=getElevation(v_pos,0.0);float f=getElevation(v_pos+vec2(epsilon.x,0),0.0);float g=getElevation(v_pos+vec2(-epsilon.x,epsilon.y),0.0);float h=getElevation(v_pos+vec2(0,epsilon.y),0.0);float i=getElevation(v_pos+vec2(epsilon.x,epsilon.y),0.0);float exaggerationFactor=u_zoom < 2.0 ? 0.4 : u_zoom < 4.5 ? 0.35 : 0.3;float exaggeration=u_zoom < 15.0 ? (u_zoom-15.0)*exaggerationFactor : 0.0;vec2 deriv=vec2((c+f+f+i)-(a+d+d+g),(g+h+h+i)-(a+b+b+c))/pow(2.0,exaggeration+(19.2562-u_zoom));gl_FragColor=clamp(vec4(deriv.x/2.0+0.5,deriv.y/2.0+0.5,1.0,1.0),0.0,1.0);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_dimension;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);highp vec2 epsilon=1.0/u_dimension;float scale=(u_dimension.x-2.0)/u_dimension.x;v_pos=(a_texture_pos/8192.0)*scale+epsilon;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_latrange;uniform vec2 u_light;uniform vec4 u_shadow;uniform vec4 u_highlight;uniform vec4 u_accent;\\n#define PI 3.141592653589793\\nvoid main() {vec4 pixel=texture2D(u_image,v_pos);vec2 deriv=((pixel.rg*2.0)-1.0);float scaleFactor=cos(radians((u_latrange[0]-u_latrange[1])*(1.0-v_pos.y)+u_latrange[1]));float slope=atan(1.25*length(deriv)/scaleFactor);float aspect=deriv.x !=0.0 ? atan(deriv.y,-deriv.x) : PI/2.0*(deriv.y > 0.0 ? 1.0 :-1.0);float intensity=u_light.x;float azimuth=u_light.y+PI;float base=1.875-intensity*1.75;float maxValue=0.5*PI;float scaledSlope=intensity !=0.5 ? ((pow(base,slope)-1.0)/(pow(base,maxValue)-1.0))*maxValue : slope;float accent=cos(scaledSlope);vec4 accent_color=(1.0-accent)*u_accent*clamp(intensity*2.0,0.0,1.0);float shade=abs(mod((aspect+azimuth)/PI+0.5,2.0)-1.0);vec4 shade_color=mix(u_shadow,u_highlight,shade)*sin(scaledSlope)*clamp(intensity*2.0,0.0,1.0);gl_FragColor=accent_color*(1.0-shade_color.a)+shade_color;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos=a_texture_pos/8192.0;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform lowp float u_device_pixel_ratio;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);gl_FragColor=color*(alpha*opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '\\n#define scale 0.015873016\\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform vec2 u_units_to_pixels;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp float v_linesofar;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define mediump float gapwidth\\n#pragma mapbox: define lowp float offset\\n#pragma mapbox: define mediump float width\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump float gapwidth\\n#pragma mapbox: initialize lowp float offset\\n#pragma mapbox: initialize mediump float width\\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;v_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*2.0;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\\n#ifdef TERRAIN3D\\nv_gamma_scale=1.0;\\n#else\\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\\n#endif\\nv_width2=vec2(outset,inset);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;varying highp vec2 v_uv;\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);vec4 color=texture2D(u_image,v_uv);gl_FragColor=color*(alpha*opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '\\n#define scale 0.015873016\\nattribute vec2 a_pos_normal;attribute vec4 a_data;attribute float a_uv_x;attribute float a_split_index;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_units_to_pixels;uniform float u_image_height;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp vec2 v_uv;\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define mediump float gapwidth\\n#pragma mapbox: define lowp float offset\\n#pragma mapbox: define mediump float width\\nvoid main() {\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump float gapwidth\\n#pragma mapbox: initialize lowp float offset\\n#pragma mapbox: initialize mediump float width\\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;highp float texel_height=1.0/u_image_height;highp float half_texel_height=0.5*texel_height;v_uv=vec2(a_uv_x,a_split_index*texel_height-half_texel_height);vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\\n#ifdef TERRAIN3D\\nv_gamma_scale=1.0;\\n#else\\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\\n#endif\\nv_width2=vec2(outset,inset);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision highp float;\\n#endif\\nuniform lowp float u_device_pixel_ratio;uniform vec2 u_texsize;uniform float u_fade;uniform mediump vec3 u_scale;uniform sampler2D u_image;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;vec2 pattern_size_a=vec2(display_size_a.x*fromScale/tileZoomRatio,display_size_a.y);vec2 pattern_size_b=vec2(display_size_b.x*toScale/tileZoomRatio,display_size_b.y);float aspect_a=display_size_a.y/v_width;float aspect_b=display_size_b.y/v_width;float dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float x_a=mod(v_linesofar/pattern_size_a.x*aspect_a,1.0);float x_b=mod(v_linesofar/pattern_size_b.x*aspect_b,1.0);float y=0.5*v_normal.y+0.5;vec2 texel_size=1.0/u_texsize;vec2 pos_a=mix(pattern_tl_a*texel_size-texel_size,pattern_br_a*texel_size+texel_size,vec2(x_a,y));vec2 pos_b=mix(pattern_tl_b*texel_size-texel_size,pattern_br_b*texel_size+texel_size,vec2(x_b,y));vec4 color=mix(texture2D(u_image,pos_a),texture2D(u_image,pos_b),u_fade);gl_FragColor=color*alpha*opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '\\n#define scale 0.015873016\\n#define LINE_DISTANCE_SCALE 2.0\\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform vec2 u_units_to_pixels;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float offset\\n#pragma mapbox: define mediump float gapwidth\\n#pragma mapbox: define mediump float width\\n#pragma mapbox: define lowp float floorwidth\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float offset\\n#pragma mapbox: initialize mediump float gapwidth\\n#pragma mapbox: initialize mediump float width\\n#pragma mapbox: initialize lowp float floorwidth\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\\n#ifdef TERRAIN3D\\nv_gamma_scale=1.0;\\n#else\\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\\n#endif\\nv_linesofar=a_linesofar;v_width2=vec2(outset,inset);v_width=floorwidth;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;uniform float u_sdfgamma;uniform float u_mix;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define mediump float width\\n#pragma mapbox: define lowp float floorwidth\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump float width\\n#pragma mapbox: initialize lowp float floorwidth\\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float sdfdist_a=texture2D(u_image,v_tex_a).a;float sdfdist_b=texture2D(u_image,v_tex_b).a;float sdfdist=mix(sdfdist_a,sdfdist_b,u_mix);alpha*=smoothstep(0.5-u_sdfgamma/floorwidth,0.5+u_sdfgamma/floorwidth,sdfdist);gl_FragColor=color*(alpha*opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '\\n#define scale 0.015873016\\n#define LINE_DISTANCE_SCALE 2.0\\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_patternscale_a;uniform float u_tex_y_a;uniform vec2 u_patternscale_b;uniform float u_tex_y_b;uniform vec2 u_units_to_pixels;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define mediump float gapwidth\\n#pragma mapbox: define lowp float offset\\n#pragma mapbox: define mediump float width\\n#pragma mapbox: define lowp float floorwidth\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump float gapwidth\\n#pragma mapbox: initialize lowp float offset\\n#pragma mapbox: initialize mediump float width\\n#pragma mapbox: initialize lowp float floorwidth\\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\\n#ifdef TERRAIN3D\\nv_gamma_scale=1.0;\\n#else\\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\\n#endif\\nv_tex_a=vec2(a_linesofar*u_patternscale_a.x/floorwidth,normal.y*u_patternscale_a.y+u_tex_y_a);v_tex_b=vec2(a_linesofar*u_patternscale_b.x/floorwidth,normal.y*u_patternscale_b.y+u_tex_y_b);v_width2=vec2(outset,inset);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform float u_fade_t;uniform float u_opacity;uniform sampler2D u_image0;uniform sampler2D u_image1;varying vec2 v_pos0;varying vec2 v_pos1;uniform float u_brightness_low;uniform float u_brightness_high;uniform float u_saturation_factor;uniform float u_contrast_factor;uniform vec3 u_spin_weights;void main() {vec4 color0=texture2D(u_image0,v_pos0);vec4 color1=texture2D(u_image1,v_pos1);if (color0.a > 0.0) {color0.rgb=color0.rgb/color0.a;}if (color1.a > 0.0) {color1.rgb=color1.rgb/color1.a;}vec4 color=mix(color0,color1,u_fade_t);color.a*=u_opacity;vec3 rgb=color.rgb;rgb=vec3(dot(rgb,u_spin_weights.xyz),dot(rgb,u_spin_weights.zxy),dot(rgb,u_spin_weights.yzx));float average=(color.r+color.g+color.b)/3.0;rgb+=(average-rgb)*u_saturation_factor;rgb=(rgb-0.5)*u_contrast_factor+0.5;vec3 u_high_vec=vec3(u_brightness_low,u_brightness_low,u_brightness_low);vec3 u_low_vec=vec3(u_brightness_high,u_brightness_high,u_brightness_high);gl_FragColor=vec4(mix(u_high_vec,u_low_vec,rgb)*color.a,color.a);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_tl_parent;uniform float u_scale_parent;uniform float u_buffer_scale;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos0;varying vec2 v_pos1;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos0=(((a_texture_pos/8192.0)-0.5)/u_buffer_scale )+0.5;v_pos1=(v_pos0*u_scale_parent)+u_tl_parent;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform sampler2D u_texture;varying vec2 v_tex;varying float v_fade_opacity;\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\nlowp float alpha=opacity*v_fade_opacity;gl_FragColor=texture2D(u_texture,v_tex)*alpha;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform highp float u_camera_to_center_distance;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform float u_fade_change;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform vec2 u_texsize;varying vec2 v_tex;varying float v_fade_opacity;\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;vec2 a_minFontScale=a_pixeloffset.zw/256.0;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\\ncamera_to_anchor_distance/u_camera_to_center_distance :\\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*max(a_minFontScale,fontScale)+a_pxoffset/16.0),z,1.0);v_tex=a_tex/u_texsize;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float visibility=calculate_visibility(projectedPoint);v_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#define SDF_PX 8.0\\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;uniform bool u_is_text;varying vec2 v_data0;varying vec3 v_data1;\\n#pragma mapbox: define highp vec4 fill_color\\n#pragma mapbox: define highp vec4 halo_color\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float halo_width\\n#pragma mapbox: define lowp float halo_blur\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 fill_color\\n#pragma mapbox: initialize highp vec4 halo_color\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float halo_width\\n#pragma mapbox: initialize lowp float halo_blur\\nfloat EDGE_GAMMA=0.105/u_device_pixel_ratio;vec2 tex=v_data0.xy;float gamma_scale=v_data1.x;float size=v_data1.y;float fade_opacity=v_data1[2];float fontScale=u_is_text ? size/24.0 : size;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float inner_edge=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);inner_edge=inner_edge+gamma*gamma_scale;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(inner_edge-gamma_scaled,inner_edge+gamma_scaled,dist);if (u_is_halo) {lowp float halo_edge=(6.0-halo_width/fontScale)/SDF_PX;alpha=min(smoothstep(halo_edge-gamma_scaled,halo_edge+gamma_scaled,dist),1.0-alpha);}gl_FragColor=color*(alpha*opacity*fade_opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;varying vec2 v_data0;varying vec3 v_data1;\\n#pragma mapbox: define highp vec4 fill_color\\n#pragma mapbox: define highp vec4 halo_color\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float halo_width\\n#pragma mapbox: define lowp float halo_blur\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 fill_color\\n#pragma mapbox: initialize highp vec4 halo_color\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float halo_width\\n#pragma mapbox: initialize lowp float halo_blur\\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\\ncamera_to_anchor_distance/u_camera_to_center_distance :\\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale+a_pxoffset),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0=a_tex/u_texsize;v_data1=vec3(gamma_scale,size,interpolated_fade_opacity);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#define SDF_PX 8.0\\n#define SDF 1.0\\n#define ICON 0.0\\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform sampler2D u_texture_icon;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;varying vec4 v_data0;varying vec4 v_data1;\\n#pragma mapbox: define highp vec4 fill_color\\n#pragma mapbox: define highp vec4 halo_color\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float halo_width\\n#pragma mapbox: define lowp float halo_blur\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 fill_color\\n#pragma mapbox: initialize highp vec4 halo_color\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float halo_width\\n#pragma mapbox: initialize lowp float halo_blur\\nfloat fade_opacity=v_data1[2];if (v_data1.w==ICON) {vec2 tex_icon=v_data0.zw;lowp float alpha=opacity*fade_opacity;gl_FragColor=texture2D(u_texture_icon,tex_icon)*alpha;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\nreturn;}vec2 tex=v_data0.xy;float EDGE_GAMMA=0.105/u_device_pixel_ratio;float gamma_scale=v_data1.x;float size=v_data1.y;float fontScale=size/24.0;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float buff=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);buff=(6.0-halo_width/fontScale)/SDF_PX;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(buff-gamma_scaled,buff+gamma_scaled,dist);gl_FragColor=color*(alpha*opacity*fade_opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;uniform vec2 u_texsize_icon;varying vec4 v_data0;varying vec4 v_data1;\\n#pragma mapbox: define highp vec4 fill_color\\n#pragma mapbox: define highp vec4 halo_color\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float halo_width\\n#pragma mapbox: define lowp float halo_blur\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 fill_color\\n#pragma mapbox: initialize highp vec4 halo_color\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float halo_width\\n#pragma mapbox: initialize lowp float halo_blur\\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);float is_sdf=a_size[0]-2.0*a_size_min;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\\ncamera_to_anchor_distance/u_camera_to_center_distance :\\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=size/24.0;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0.xy=a_tex/u_texsize;v_data0.zw=a_tex/u_texsize_icon;v_data1=vec4(gamma_scale,size,interpolated_fade_opacity,is_sdf);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform sampler2D u_texture;varying vec2 v_texture_pos;void main() {gl_FragColor=texture2D(u_texture,v_texture_pos);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying float v_depth;const highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitMsk=vec4(0.,vec3(1./256.0));highp vec4 pack(highp float value) {highp vec4 comp=fract(value*bitSh);comp-=comp.xxyz*bitMsk;return comp;}void main() {gl_FragColor=pack(v_depth);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'precision mediump float;uniform sampler2D u_texture;uniform float u_terrain_coords_id;varying vec2 v_texture_pos;void main() {vec4 rgba=texture2D(u_texture,v_texture_pos);gl_FragColor=vec4(rgba.r,rgba.g,rgba.b,u_terrain_coords_id);}';\n","\nimport type {Program} from './program';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {IndexBuffer} from '../gl/index_buffer';\nimport type {Context} from '../gl/context';\n\n/**\n * @internal\n * A vertex array object used to pass data to the webgl code\n */\nexport class VertexArrayObject {\n context: Context;\n boundProgram: Program;\n boundLayoutVertexBuffer: VertexBuffer;\n boundPaintVertexBuffers: Array;\n boundIndexBuffer: IndexBuffer;\n boundVertexOffset: number;\n boundDynamicVertexBuffer: VertexBuffer;\n boundDynamicVertexBuffer2: VertexBuffer;\n boundDynamicVertexBuffer3: VertexBuffer;\n vao: any;\n\n constructor() {\n this.boundProgram = null;\n this.boundLayoutVertexBuffer = null;\n this.boundPaintVertexBuffers = [];\n this.boundIndexBuffer = null;\n this.boundVertexOffset = null;\n this.boundDynamicVertexBuffer = null;\n this.vao = null;\n }\n\n bind(context: Context,\n program: Program,\n layoutVertexBuffer: VertexBuffer,\n paintVertexBuffers: Array,\n indexBuffer?: IndexBuffer | null,\n vertexOffset?: number | null,\n dynamicVertexBuffer?: VertexBuffer | null,\n dynamicVertexBuffer2?: VertexBuffer | null,\n dynamicVertexBuffer3?: VertexBuffer | null) {\n\n this.context = context;\n\n let paintBuffersDiffer = this.boundPaintVertexBuffers.length !== paintVertexBuffers.length;\n for (let i = 0; !paintBuffersDiffer && i < paintVertexBuffers.length; i++) {\n if (this.boundPaintVertexBuffers[i] !== paintVertexBuffers[i]) {\n paintBuffersDiffer = true;\n }\n }\n\n const isFreshBindRequired = (\n !this.vao ||\n this.boundProgram !== program ||\n this.boundLayoutVertexBuffer !== layoutVertexBuffer ||\n paintBuffersDiffer ||\n this.boundIndexBuffer !== indexBuffer ||\n this.boundVertexOffset !== vertexOffset ||\n this.boundDynamicVertexBuffer !== dynamicVertexBuffer ||\n this.boundDynamicVertexBuffer2 !== dynamicVertexBuffer2 ||\n this.boundDynamicVertexBuffer3 !== dynamicVertexBuffer3\n );\n\n if (isFreshBindRequired) {\n this.freshBind(program, layoutVertexBuffer, paintVertexBuffers, indexBuffer, vertexOffset, dynamicVertexBuffer, dynamicVertexBuffer2, dynamicVertexBuffer3);\n } else {\n context.bindVertexArray.set(this.vao);\n\n if (dynamicVertexBuffer) {\n // The buffer may have been updated. Rebind to upload data.\n dynamicVertexBuffer.bind();\n }\n\n if (indexBuffer && indexBuffer.dynamicDraw) {\n indexBuffer.bind();\n }\n\n if (dynamicVertexBuffer2) {\n dynamicVertexBuffer2.bind();\n }\n\n if (dynamicVertexBuffer3) {\n dynamicVertexBuffer3.bind();\n }\n }\n }\n\n freshBind(program: Program,\n layoutVertexBuffer: VertexBuffer,\n paintVertexBuffers: Array,\n indexBuffer?: IndexBuffer | null,\n vertexOffset?: number | null,\n dynamicVertexBuffer?: VertexBuffer | null,\n dynamicVertexBuffer2?: VertexBuffer | null,\n dynamicVertexBuffer3?: VertexBuffer | null) {\n\n const numNextAttributes = program.numAttributes;\n\n const context = this.context;\n const gl = context.gl;\n\n if (this.vao) this.destroy();\n this.vao = context.createVertexArray();\n context.bindVertexArray.set(this.vao);\n\n // store the arguments so that we can verify them when the vao is bound again\n this.boundProgram = program;\n this.boundLayoutVertexBuffer = layoutVertexBuffer;\n this.boundPaintVertexBuffers = paintVertexBuffers;\n this.boundIndexBuffer = indexBuffer;\n this.boundVertexOffset = vertexOffset;\n this.boundDynamicVertexBuffer = dynamicVertexBuffer;\n this.boundDynamicVertexBuffer2 = dynamicVertexBuffer2;\n this.boundDynamicVertexBuffer3 = dynamicVertexBuffer3;\n\n layoutVertexBuffer.enableAttributes(gl, program);\n for (const vertexBuffer of paintVertexBuffers) {\n vertexBuffer.enableAttributes(gl, program);\n }\n\n if (dynamicVertexBuffer) {\n dynamicVertexBuffer.enableAttributes(gl, program);\n }\n if (dynamicVertexBuffer2) {\n dynamicVertexBuffer2.enableAttributes(gl, program);\n }\n if (dynamicVertexBuffer3) {\n dynamicVertexBuffer3.enableAttributes(gl, program);\n }\n\n layoutVertexBuffer.bind();\n layoutVertexBuffer.setVertexAttribPointers(gl, program, vertexOffset);\n for (const vertexBuffer of paintVertexBuffers) {\n vertexBuffer.bind();\n vertexBuffer.setVertexAttribPointers(gl, program, vertexOffset);\n }\n\n if (dynamicVertexBuffer) {\n dynamicVertexBuffer.bind();\n dynamicVertexBuffer.setVertexAttribPointers(gl, program, vertexOffset);\n }\n if (indexBuffer) {\n indexBuffer.bind();\n }\n if (dynamicVertexBuffer2) {\n dynamicVertexBuffer2.bind();\n dynamicVertexBuffer2.setVertexAttribPointers(gl, program, vertexOffset);\n }\n if (dynamicVertexBuffer3) {\n dynamicVertexBuffer3.bind();\n dynamicVertexBuffer3.setVertexAttribPointers(gl, program, vertexOffset);\n }\n\n context.currentNumAttributes = numNextAttributes;\n }\n\n destroy() {\n if (this.vao) {\n this.context.deleteVertexArray(this.vao);\n this.vao = null;\n }\n }\n}\n","import {shaders} from '../shaders/shaders';\nimport {ProgramConfiguration} from '../data/program_configuration';\nimport {VertexArrayObject} from './vertex_array_object';\nimport {Context} from '../gl/context';\n\nimport type {SegmentVector} from '../data/segment';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {IndexBuffer} from '../gl/index_buffer';\nimport type {DepthMode} from '../gl/depth_mode';\nimport type {StencilMode} from '../gl/stencil_mode';\nimport type {ColorMode} from '../gl/color_mode';\nimport type {CullFaceMode} from '../gl/cull_face_mode';\nimport type {UniformBindings, UniformValues, UniformLocations} from './uniform_binding';\nimport type {BinderUniform} from '../data/program_configuration';\nimport {terrainPreludeUniforms, TerrainPreludeUniformsType} from './program/terrain_program';\nimport type {TerrainData} from '../render/terrain';\nimport {Terrain} from '../render/terrain';\n\nexport type DrawMode = WebGLRenderingContextBase['LINES'] | WebGLRenderingContextBase['TRIANGLES'] | WebGL2RenderingContext['LINE_STRIP'];\n\nfunction getTokenizedAttributesAndUniforms(array: Array): Array {\n const result = [];\n\n for (let i = 0; i < array.length; i++) {\n if (array[i] === null) continue;\n const token = array[i].split(' ');\n result.push(token.pop());\n }\n return result;\n}\n\n/**\n * @internal\n * A webgl program to execute in the GPU space\n */\nexport class Program {\n program: WebGLProgram;\n attributes: {[_: string]: number};\n numAttributes: number;\n fixedUniforms: Us;\n terrainUniforms: TerrainPreludeUniformsType;\n binderUniforms: Array;\n failedToCreate: boolean;\n\n constructor(context: Context,\n source: {\n fragmentSource: string;\n vertexSource: string;\n staticAttributes: Array;\n staticUniforms: Array;\n },\n configuration: ProgramConfiguration,\n fixedUniforms: (b: Context, a: UniformLocations) => Us,\n showOverdrawInspector: boolean,\n terrain: Terrain) {\n\n const gl = context.gl;\n this.program = gl.createProgram();\n\n const staticAttrInfo = getTokenizedAttributesAndUniforms(source.staticAttributes);\n const dynamicAttrInfo = configuration ? configuration.getBinderAttributes() : [];\n const allAttrInfo = staticAttrInfo.concat(dynamicAttrInfo);\n\n const preludeUniformsInfo = shaders.prelude.staticUniforms ? getTokenizedAttributesAndUniforms(shaders.prelude.staticUniforms) : [];\n const staticUniformsInfo = source.staticUniforms ? getTokenizedAttributesAndUniforms(source.staticUniforms) : [];\n const dynamicUniformsInfo = configuration ? configuration.getBinderUniforms() : [];\n // remove duplicate uniforms\n const uniformList = preludeUniformsInfo.concat(staticUniformsInfo).concat(dynamicUniformsInfo);\n const allUniformsInfo = [];\n for (const uniform of uniformList) {\n if (allUniformsInfo.indexOf(uniform) < 0) allUniformsInfo.push(uniform);\n }\n\n const defines = configuration ? configuration.defines() : [];\n if (showOverdrawInspector) {\n defines.push('#define OVERDRAW_INSPECTOR;');\n }\n if (terrain) {\n defines.push('#define TERRAIN3D;');\n }\n\n const fragmentSource = defines.concat(shaders.prelude.fragmentSource, source.fragmentSource).join('\\n');\n const vertexSource = defines.concat(shaders.prelude.vertexSource, source.vertexSource).join('\\n');\n\n const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);\n if (gl.isContextLost()) {\n this.failedToCreate = true;\n return;\n }\n gl.shaderSource(fragmentShader, fragmentSource);\n gl.compileShader(fragmentShader);\n\n if (!gl.getShaderParameter(fragmentShader, gl.COMPILE_STATUS)) {\n throw new Error(`Could not compile fragment shader: ${gl.getShaderInfoLog(fragmentShader)}`);\n }\n\n gl.attachShader(this.program, fragmentShader);\n\n const vertexShader = gl.createShader(gl.VERTEX_SHADER);\n if (gl.isContextLost()) {\n this.failedToCreate = true;\n return;\n }\n gl.shaderSource(vertexShader, vertexSource);\n gl.compileShader(vertexShader);\n\n if (!gl.getShaderParameter(vertexShader, gl.COMPILE_STATUS)) {\n throw new Error(`Could not compile vertex shader: ${gl.getShaderInfoLog(vertexShader)}`);\n }\n\n gl.attachShader(this.program, vertexShader);\n\n this.attributes = {};\n const uniformLocations = {};\n\n this.numAttributes = allAttrInfo.length;\n\n for (let i = 0; i < this.numAttributes; i++) {\n if (allAttrInfo[i]) {\n gl.bindAttribLocation(this.program, i, allAttrInfo[i]);\n this.attributes[allAttrInfo[i]] = i;\n }\n }\n\n gl.linkProgram(this.program);\n\n if (!gl.getProgramParameter(this.program, gl.LINK_STATUS)) {\n throw new Error(`Program failed to link: ${gl.getProgramInfoLog(this.program)}`);\n }\n\n gl.deleteShader(vertexShader);\n gl.deleteShader(fragmentShader);\n\n for (let it = 0; it < allUniformsInfo.length; it++) {\n const uniform = allUniformsInfo[it];\n if (uniform && !uniformLocations[uniform]) {\n const uniformLocation = gl.getUniformLocation(this.program, uniform);\n if (uniformLocation) {\n uniformLocations[uniform] = uniformLocation;\n }\n }\n }\n\n this.fixedUniforms = fixedUniforms(context, uniformLocations);\n this.terrainUniforms = terrainPreludeUniforms(context, uniformLocations);\n this.binderUniforms = configuration ? configuration.getUniforms(context, uniformLocations) : [];\n }\n\n draw(context: Context,\n drawMode: DrawMode,\n depthMode: Readonly,\n stencilMode: Readonly,\n colorMode: Readonly,\n cullFaceMode: Readonly,\n uniformValues: UniformValues,\n terrain: TerrainData,\n layerID: string,\n layoutVertexBuffer: VertexBuffer,\n indexBuffer: IndexBuffer,\n segments: SegmentVector,\n currentProperties?: any,\n zoom?: number | null,\n configuration?: ProgramConfiguration | null,\n dynamicLayoutBuffer?: VertexBuffer | null,\n dynamicLayoutBuffer2?: VertexBuffer | null,\n dynamicLayoutBuffer3?: VertexBuffer | null) {\n\n const gl = context.gl;\n\n if (this.failedToCreate) return;\n\n context.program.set(this.program);\n context.setDepthMode(depthMode);\n context.setStencilMode(stencilMode);\n context.setColorMode(colorMode);\n context.setCullFace(cullFaceMode);\n\n // set variables used by the 3d functions defined in _prelude.vertex.glsl\n if (terrain) {\n context.activeTexture.set(gl.TEXTURE2);\n gl.bindTexture(gl.TEXTURE_2D, terrain.depthTexture);\n context.activeTexture.set(gl.TEXTURE3);\n gl.bindTexture(gl.TEXTURE_2D, terrain.texture);\n for (const name in this.terrainUniforms) {\n this.terrainUniforms[name].set(terrain[name]);\n }\n }\n\n for (const name in this.fixedUniforms) {\n this.fixedUniforms[name].set(uniformValues[name]);\n }\n\n if (configuration) {\n configuration.setUniforms(context, this.binderUniforms, currentProperties, {zoom: (zoom as any)});\n }\n\n let primitiveSize = 0;\n switch (drawMode) {\n case gl.LINES:\n primitiveSize = 2;\n break;\n case gl.TRIANGLES:\n primitiveSize = 3;\n break;\n case gl.LINE_STRIP:\n primitiveSize = 1;\n break;\n }\n\n for (const segment of segments.get()) {\n const vaos = segment.vaos || (segment.vaos = {});\n const vao: VertexArrayObject = vaos[layerID] || (vaos[layerID] = new VertexArrayObject());\n\n vao.bind(\n context,\n this,\n layoutVertexBuffer,\n configuration ? configuration.getPaintVertexBuffers() : [],\n indexBuffer,\n segment.vertexOffset,\n dynamicLayoutBuffer,\n dynamicLayoutBuffer2,\n dynamicLayoutBuffer3\n );\n\n gl.drawElements(\n drawMode,\n segment.primitiveLength * primitiveSize,\n gl.UNSIGNED_SHORT,\n segment.primitiveOffset * primitiveSize * 2);\n }\n }\n}\n","import {\n Uniform1i,\n Uniform1f,\n Uniform4f,\n UniformMatrix4f\n} from '../uniform_binding';\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../../render/uniform_binding';\nimport {mat4} from 'gl-matrix';\n\nexport type TerrainPreludeUniformsType = {\n 'u_depth': Uniform1i;\n 'u_terrain': Uniform1i;\n 'u_terrain_dim': Uniform1f;\n 'u_terrain_matrix': UniformMatrix4f;\n 'u_terrain_unpack': Uniform4f;\n 'u_terrain_exaggeration': Uniform1f;\n};\n\nexport type TerrainUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_texture': Uniform1i;\n 'u_ele_delta': Uniform1f;\n};\n\nexport type TerrainDepthUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_ele_delta': Uniform1f;\n};\n\nexport type TerrainCoordsUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_texture': Uniform1i;\n 'u_terrain_coords_id': Uniform1f;\n 'u_ele_delta': Uniform1f;\n};\n\nconst terrainPreludeUniforms = (context: Context, locations: UniformLocations): TerrainPreludeUniformsType => ({\n 'u_depth': new Uniform1i(context, locations.u_depth),\n 'u_terrain': new Uniform1i(context, locations.u_terrain),\n 'u_terrain_dim': new Uniform1f(context, locations.u_terrain_dim),\n 'u_terrain_matrix': new UniformMatrix4f(context, locations.u_terrain_matrix),\n 'u_terrain_unpack': new Uniform4f(context, locations.u_terrain_unpack),\n 'u_terrain_exaggeration': new Uniform1f(context, locations.u_terrain_exaggeration)\n});\n\nconst terrainUniforms = (context: Context, locations: UniformLocations): TerrainUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_texture': new Uniform1i(context, locations.u_texture),\n 'u_ele_delta': new Uniform1f(context, locations.u_ele_delta)\n});\n\nconst terrainDepthUniforms = (context: Context, locations: UniformLocations): TerrainDepthUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_ele_delta': new Uniform1f(context, locations.u_ele_delta)\n});\n\nconst terrainCoordsUniforms = (context: Context, locations: UniformLocations): TerrainCoordsUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_texture': new Uniform1i(context, locations.u_texture),\n 'u_terrain_coords_id': new Uniform1f(context, locations.u_terrain_coords_id),\n 'u_ele_delta': new Uniform1f(context, locations.u_ele_delta)\n});\n\nconst terrainUniformValues = (\n matrix: mat4,\n eleDelta: number\n): UniformValues => ({\n 'u_matrix': matrix,\n 'u_texture': 0,\n 'u_ele_delta': eleDelta\n});\n\nconst terrainDepthUniformValues = (\n matrix: mat4,\n eleDelta: number\n): UniformValues => ({\n 'u_matrix': matrix,\n 'u_ele_delta': eleDelta\n});\n\nconst terrainCoordsUniformValues = (\n matrix: mat4,\n coordsId: number,\n eleDelta: number\n): UniformValues => ({\n 'u_matrix': matrix,\n 'u_terrain_coords_id': coordsId / 255,\n 'u_texture': 0,\n 'u_ele_delta': eleDelta\n});\n\nexport {terrainUniforms, terrainDepthUniforms, terrainCoordsUniforms, terrainPreludeUniforms, terrainUniformValues, terrainDepthUniformValues, terrainCoordsUniformValues};\n","import {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f\n} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\n\nimport type {Painter} from '../painter';\nimport type {OverscaledTileID} from '../../source/tile_id';\nimport type {CrossFaded} from '../../style/properties';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport type {UniformValues} from '../uniform_binding';\nimport type {Tile} from '../../source/tile';\nimport type {ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\n\ntype BackgroundPatternUniformsType = {\n 'u_image': Uniform1i;\n 'u_pattern_tl_a': Uniform2f;\n 'u_pattern_br_a': Uniform2f;\n 'u_pattern_tl_b': Uniform2f;\n 'u_pattern_br_b': Uniform2f;\n 'u_texsize': Uniform2f;\n 'u_mix': Uniform1f;\n 'u_pattern_size_a': Uniform2f;\n 'u_pattern_size_b': Uniform2f;\n 'u_scale_a': Uniform1f;\n 'u_scale_b': Uniform1f;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_tile_units_to_pixels': Uniform1f;\n};\n\nexport type PatternUniformsType = {\n // pattern uniforms:\n 'u_image': Uniform1i;\n 'u_texsize': Uniform2f;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n};\n\nfunction patternUniformValues(crossfade: CrossfadeParameters, painter: Painter, tile: Tile): UniformValues {\n\n const tileRatio = 1 / pixelsToTileUnits(tile, 1, painter.transform.tileZoom);\n\n const numTiles = Math.pow(2, tile.tileID.overscaledZ);\n const tileSizeAtNearestZoom = tile.tileSize * Math.pow(2, painter.transform.tileZoom) / numTiles;\n\n const pixelX = tileSizeAtNearestZoom * (tile.tileID.canonical.x + tile.tileID.wrap * numTiles);\n const pixelY = tileSizeAtNearestZoom * tile.tileID.canonical.y;\n\n return {\n 'u_image': 0,\n 'u_texsize': tile.imageAtlasTexture.size,\n 'u_scale': [tileRatio, crossfade.fromScale, crossfade.toScale],\n 'u_fade': crossfade.t,\n // split the pixel coord into two pairs of 16 bit numbers. The glsl spec only guarantees 16 bits of precision.\n 'u_pixel_coord_upper': [pixelX >> 16, pixelY >> 16],\n 'u_pixel_coord_lower': [pixelX & 0xFFFF, pixelY & 0xFFFF]\n };\n}\n\nfunction bgPatternUniformValues(\n image: CrossFaded,\n crossfade: CrossfadeParameters,\n painter: Painter,\n tile: {\n tileID: OverscaledTileID;\n tileSize: number;\n }\n): UniformValues {\n const imagePosA = painter.imageManager.getPattern(image.from.toString());\n const imagePosB = painter.imageManager.getPattern(image.to.toString());\n const {width, height} = painter.imageManager.getPixelSize();\n\n const numTiles = Math.pow(2, tile.tileID.overscaledZ);\n const tileSizeAtNearestZoom = tile.tileSize * Math.pow(2, painter.transform.tileZoom) / numTiles;\n\n const pixelX = tileSizeAtNearestZoom * (tile.tileID.canonical.x + tile.tileID.wrap * numTiles);\n const pixelY = tileSizeAtNearestZoom * tile.tileID.canonical.y;\n\n return {\n 'u_image': 0,\n 'u_pattern_tl_a': (imagePosA as any).tl,\n 'u_pattern_br_a': (imagePosA as any).br,\n 'u_pattern_tl_b': (imagePosB as any).tl,\n 'u_pattern_br_b': (imagePosB as any).br,\n 'u_texsize': [width, height],\n 'u_mix': crossfade.t,\n 'u_pattern_size_a': (imagePosA as any).displaySize,\n 'u_pattern_size_b': (imagePosB as any).displaySize,\n 'u_scale_a': crossfade.fromScale,\n 'u_scale_b': crossfade.toScale,\n 'u_tile_units_to_pixels': 1 / pixelsToTileUnits(tile, 1, painter.transform.tileZoom),\n // split the pixel coord into two pairs of 16 bit numbers. The glsl spec only guarantees 16 bits of precision.\n 'u_pixel_coord_upper': [pixelX >> 16, pixelY >> 16],\n 'u_pixel_coord_lower': [pixelX & 0xFFFF, pixelY & 0xFFFF]\n };\n}\nexport {bgPatternUniformValues, patternUniformValues};\n","import {patternUniformValues} from './pattern';\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f,\n UniformMatrix4f\n} from '../uniform_binding';\n\nimport {mat3, mat4, vec3} from 'gl-matrix';\nimport {extend} from '../../util/util';\n\nimport type {Context} from '../../gl/context';\nimport type {Painter} from '../painter';\nimport type {OverscaledTileID} from '../../source/tile_id';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport type {Tile} from '../../source/tile';\n\nexport type FillExtrusionUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_lightpos': Uniform3f;\n 'u_lightintensity': Uniform1f;\n 'u_lightcolor': Uniform3f;\n 'u_vertical_gradient': Uniform1f;\n 'u_opacity': Uniform1f;\n};\n\nexport type FillExtrusionPatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_lightpos': Uniform3f;\n 'u_lightintensity': Uniform1f;\n 'u_lightcolor': Uniform3f;\n 'u_height_factor': Uniform1f;\n 'u_vertical_gradient': Uniform1f;\n // pattern uniforms:\n 'u_texsize': Uniform2f;\n 'u_image': Uniform1i;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n 'u_opacity': Uniform1f;\n};\n\nconst fillExtrusionUniforms = (context: Context, locations: UniformLocations): FillExtrusionUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_lightpos': new Uniform3f(context, locations.u_lightpos),\n 'u_lightintensity': new Uniform1f(context, locations.u_lightintensity),\n 'u_lightcolor': new Uniform3f(context, locations.u_lightcolor),\n 'u_vertical_gradient': new Uniform1f(context, locations.u_vertical_gradient),\n 'u_opacity': new Uniform1f(context, locations.u_opacity)\n});\n\nconst fillExtrusionPatternUniforms = (context: Context, locations: UniformLocations): FillExtrusionPatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_lightpos': new Uniform3f(context, locations.u_lightpos),\n 'u_lightintensity': new Uniform1f(context, locations.u_lightintensity),\n 'u_lightcolor': new Uniform3f(context, locations.u_lightcolor),\n 'u_vertical_gradient': new Uniform1f(context, locations.u_vertical_gradient),\n 'u_height_factor': new Uniform1f(context, locations.u_height_factor),\n // pattern uniforms\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper),\n 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower),\n 'u_scale': new Uniform3f(context, locations.u_scale),\n 'u_fade': new Uniform1f(context, locations.u_fade),\n 'u_opacity': new Uniform1f(context, locations.u_opacity)\n});\n\nconst fillExtrusionUniformValues = (\n matrix: mat4,\n painter: Painter,\n shouldUseVerticalGradient: boolean,\n opacity: number\n): UniformValues => {\n const light = painter.style.light;\n const _lp = light.properties.get('position');\n const lightPos = [_lp.x, _lp.y, _lp.z] as vec3;\n const lightMat = mat3.create();\n if (light.properties.get('anchor') === 'viewport') {\n mat3.fromRotation(lightMat, -painter.transform.angle);\n }\n vec3.transformMat3(lightPos, lightPos, lightMat);\n\n const lightColor = light.properties.get('color');\n\n return {\n 'u_matrix': matrix,\n 'u_lightpos': lightPos,\n 'u_lightintensity': light.properties.get('intensity'),\n 'u_lightcolor': [lightColor.r, lightColor.g, lightColor.b],\n 'u_vertical_gradient': +shouldUseVerticalGradient,\n 'u_opacity': opacity\n };\n};\n\nconst fillExtrusionPatternUniformValues = (\n matrix: mat4,\n painter: Painter,\n shouldUseVerticalGradient: boolean,\n opacity: number,\n coord: OverscaledTileID,\n crossfade: CrossfadeParameters,\n tile: Tile\n): UniformValues => {\n return extend(fillExtrusionUniformValues(matrix, painter, shouldUseVerticalGradient, opacity),\n patternUniformValues(crossfade, painter, tile),\n {\n 'u_height_factor': -Math.pow(2, coord.overscaledZ) / tile.tileSize / 8\n });\n};\n\nexport {\n fillExtrusionUniforms,\n fillExtrusionPatternUniforms,\n fillExtrusionUniformValues,\n fillExtrusionPatternUniformValues\n};\n","import * as glMatrix from \"./common.js\";\n/**\n * 3x3 Matrix\n * @module mat3\n */\n\n/**\n * Creates a new identity mat3\n *\n * @returns {mat3} a new 3x3 matrix\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(9);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n }\n\n out[0] = 1;\n out[4] = 1;\n out[8] = 1;\n return out;\n}\n/**\n * Copies the upper-left 3x3 values into the given mat3.\n *\n * @param {mat3} out the receiving 3x3 matrix\n * @param {ReadonlyMat4} a the source 4x4 matrix\n * @returns {mat3} out\n */\n\nexport function fromMat4(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[4];\n out[4] = a[5];\n out[5] = a[6];\n out[6] = a[8];\n out[7] = a[9];\n out[8] = a[10];\n return out;\n}\n/**\n * Creates a new mat3 initialized with values from an existing matrix\n *\n * @param {ReadonlyMat3} a matrix to clone\n * @returns {mat3} a new 3x3 matrix\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(9);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n return out;\n}\n/**\n * Copy the values from one mat3 to another\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the source matrix\n * @returns {mat3} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n return out;\n}\n/**\n * Create a new mat3 with the given values\n *\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m10 Component in column 1, row 0 position (index 3)\n * @param {Number} m11 Component in column 1, row 1 position (index 4)\n * @param {Number} m12 Component in column 1, row 2 position (index 5)\n * @param {Number} m20 Component in column 2, row 0 position (index 6)\n * @param {Number} m21 Component in column 2, row 1 position (index 7)\n * @param {Number} m22 Component in column 2, row 2 position (index 8)\n * @returns {mat3} A new mat3\n */\n\nexport function fromValues(m00, m01, m02, m10, m11, m12, m20, m21, m22) {\n var out = new glMatrix.ARRAY_TYPE(9);\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m10;\n out[4] = m11;\n out[5] = m12;\n out[6] = m20;\n out[7] = m21;\n out[8] = m22;\n return out;\n}\n/**\n * Set the components of a mat3 to the given values\n *\n * @param {mat3} out the receiving matrix\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m10 Component in column 1, row 0 position (index 3)\n * @param {Number} m11 Component in column 1, row 1 position (index 4)\n * @param {Number} m12 Component in column 1, row 2 position (index 5)\n * @param {Number} m20 Component in column 2, row 0 position (index 6)\n * @param {Number} m21 Component in column 2, row 1 position (index 7)\n * @param {Number} m22 Component in column 2, row 2 position (index 8)\n * @returns {mat3} out\n */\n\nexport function set(out, m00, m01, m02, m10, m11, m12, m20, m21, m22) {\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m10;\n out[4] = m11;\n out[5] = m12;\n out[6] = m20;\n out[7] = m21;\n out[8] = m22;\n return out;\n}\n/**\n * Set a mat3 to the identity matrix\n *\n * @param {mat3} out the receiving matrix\n * @returns {mat3} out\n */\n\nexport function identity(out) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 1;\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 1;\n return out;\n}\n/**\n * Transpose the values of a mat3\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the source matrix\n * @returns {mat3} out\n */\n\nexport function transpose(out, a) {\n // If we are transposing ourselves we can skip a few steps but have to cache some values\n if (out === a) {\n var a01 = a[1],\n a02 = a[2],\n a12 = a[5];\n out[1] = a[3];\n out[2] = a[6];\n out[3] = a01;\n out[5] = a[7];\n out[6] = a02;\n out[7] = a12;\n } else {\n out[0] = a[0];\n out[1] = a[3];\n out[2] = a[6];\n out[3] = a[1];\n out[4] = a[4];\n out[5] = a[7];\n out[6] = a[2];\n out[7] = a[5];\n out[8] = a[8];\n }\n\n return out;\n}\n/**\n * Inverts a mat3\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the source matrix\n * @returns {mat3} out\n */\n\nexport function invert(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2];\n var a10 = a[3],\n a11 = a[4],\n a12 = a[5];\n var a20 = a[6],\n a21 = a[7],\n a22 = a[8];\n var b01 = a22 * a11 - a12 * a21;\n var b11 = -a22 * a10 + a12 * a20;\n var b21 = a21 * a10 - a11 * a20; // Calculate the determinant\n\n var det = a00 * b01 + a01 * b11 + a02 * b21;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = b01 * det;\n out[1] = (-a22 * a01 + a02 * a21) * det;\n out[2] = (a12 * a01 - a02 * a11) * det;\n out[3] = b11 * det;\n out[4] = (a22 * a00 - a02 * a20) * det;\n out[5] = (-a12 * a00 + a02 * a10) * det;\n out[6] = b21 * det;\n out[7] = (-a21 * a00 + a01 * a20) * det;\n out[8] = (a11 * a00 - a01 * a10) * det;\n return out;\n}\n/**\n * Calculates the adjugate of a mat3\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the source matrix\n * @returns {mat3} out\n */\n\nexport function adjoint(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2];\n var a10 = a[3],\n a11 = a[4],\n a12 = a[5];\n var a20 = a[6],\n a21 = a[7],\n a22 = a[8];\n out[0] = a11 * a22 - a12 * a21;\n out[1] = a02 * a21 - a01 * a22;\n out[2] = a01 * a12 - a02 * a11;\n out[3] = a12 * a20 - a10 * a22;\n out[4] = a00 * a22 - a02 * a20;\n out[5] = a02 * a10 - a00 * a12;\n out[6] = a10 * a21 - a11 * a20;\n out[7] = a01 * a20 - a00 * a21;\n out[8] = a00 * a11 - a01 * a10;\n return out;\n}\n/**\n * Calculates the determinant of a mat3\n *\n * @param {ReadonlyMat3} a the source matrix\n * @returns {Number} determinant of a\n */\n\nexport function determinant(a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2];\n var a10 = a[3],\n a11 = a[4],\n a12 = a[5];\n var a20 = a[6],\n a21 = a[7],\n a22 = a[8];\n return a00 * (a22 * a11 - a12 * a21) + a01 * (-a22 * a10 + a12 * a20) + a02 * (a21 * a10 - a11 * a20);\n}\n/**\n * Multiplies two mat3's\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the first operand\n * @param {ReadonlyMat3} b the second operand\n * @returns {mat3} out\n */\n\nexport function multiply(out, a, b) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2];\n var a10 = a[3],\n a11 = a[4],\n a12 = a[5];\n var a20 = a[6],\n a21 = a[7],\n a22 = a[8];\n var b00 = b[0],\n b01 = b[1],\n b02 = b[2];\n var b10 = b[3],\n b11 = b[4],\n b12 = b[5];\n var b20 = b[6],\n b21 = b[7],\n b22 = b[8];\n out[0] = b00 * a00 + b01 * a10 + b02 * a20;\n out[1] = b00 * a01 + b01 * a11 + b02 * a21;\n out[2] = b00 * a02 + b01 * a12 + b02 * a22;\n out[3] = b10 * a00 + b11 * a10 + b12 * a20;\n out[4] = b10 * a01 + b11 * a11 + b12 * a21;\n out[5] = b10 * a02 + b11 * a12 + b12 * a22;\n out[6] = b20 * a00 + b21 * a10 + b22 * a20;\n out[7] = b20 * a01 + b21 * a11 + b22 * a21;\n out[8] = b20 * a02 + b21 * a12 + b22 * a22;\n return out;\n}\n/**\n * Translate a mat3 by the given vector\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the matrix to translate\n * @param {ReadonlyVec2} v vector to translate by\n * @returns {mat3} out\n */\n\nexport function translate(out, a, v) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a10 = a[3],\n a11 = a[4],\n a12 = a[5],\n a20 = a[6],\n a21 = a[7],\n a22 = a[8],\n x = v[0],\n y = v[1];\n out[0] = a00;\n out[1] = a01;\n out[2] = a02;\n out[3] = a10;\n out[4] = a11;\n out[5] = a12;\n out[6] = x * a00 + y * a10 + a20;\n out[7] = x * a01 + y * a11 + a21;\n out[8] = x * a02 + y * a12 + a22;\n return out;\n}\n/**\n * Rotates a mat3 by the given angle\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat3} out\n */\n\nexport function rotate(out, a, rad) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a10 = a[3],\n a11 = a[4],\n a12 = a[5],\n a20 = a[6],\n a21 = a[7],\n a22 = a[8],\n s = Math.sin(rad),\n c = Math.cos(rad);\n out[0] = c * a00 + s * a10;\n out[1] = c * a01 + s * a11;\n out[2] = c * a02 + s * a12;\n out[3] = c * a10 - s * a00;\n out[4] = c * a11 - s * a01;\n out[5] = c * a12 - s * a02;\n out[6] = a20;\n out[7] = a21;\n out[8] = a22;\n return out;\n}\n/**\n * Scales the mat3 by the dimensions in the given vec2\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the matrix to rotate\n * @param {ReadonlyVec2} v the vec2 to scale the matrix by\n * @returns {mat3} out\n **/\n\nexport function scale(out, a, v) {\n var x = v[0],\n y = v[1];\n out[0] = x * a[0];\n out[1] = x * a[1];\n out[2] = x * a[2];\n out[3] = y * a[3];\n out[4] = y * a[4];\n out[5] = y * a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n return out;\n}\n/**\n * Creates a matrix from a vector translation\n * This is equivalent to (but much faster than):\n *\n * mat3.identity(dest);\n * mat3.translate(dest, dest, vec);\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {ReadonlyVec2} v Translation vector\n * @returns {mat3} out\n */\n\nexport function fromTranslation(out, v) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 1;\n out[5] = 0;\n out[6] = v[0];\n out[7] = v[1];\n out[8] = 1;\n return out;\n}\n/**\n * Creates a matrix from a given angle\n * This is equivalent to (but much faster than):\n *\n * mat3.identity(dest);\n * mat3.rotate(dest, dest, rad);\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat3} out\n */\n\nexport function fromRotation(out, rad) {\n var s = Math.sin(rad),\n c = Math.cos(rad);\n out[0] = c;\n out[1] = s;\n out[2] = 0;\n out[3] = -s;\n out[4] = c;\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 1;\n return out;\n}\n/**\n * Creates a matrix from a vector scaling\n * This is equivalent to (but much faster than):\n *\n * mat3.identity(dest);\n * mat3.scale(dest, dest, vec);\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {ReadonlyVec2} v Scaling vector\n * @returns {mat3} out\n */\n\nexport function fromScaling(out, v) {\n out[0] = v[0];\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = v[1];\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 1;\n return out;\n}\n/**\n * Copies the values from a mat2d into a mat3\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat2d} a the matrix to copy\n * @returns {mat3} out\n **/\n\nexport function fromMat2d(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = 0;\n out[3] = a[2];\n out[4] = a[3];\n out[5] = 0;\n out[6] = a[4];\n out[7] = a[5];\n out[8] = 1;\n return out;\n}\n/**\n * Calculates a 3x3 matrix from the given quaternion\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {ReadonlyQuat} q Quaternion to create matrix from\n *\n * @returns {mat3} out\n */\n\nexport function fromQuat(out, q) {\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var yx = y * x2;\n var yy = y * y2;\n var zx = z * x2;\n var zy = z * y2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n out[0] = 1 - yy - zz;\n out[3] = yx - wz;\n out[6] = zx + wy;\n out[1] = yx + wz;\n out[4] = 1 - xx - zz;\n out[7] = zy - wx;\n out[2] = zx - wy;\n out[5] = zy + wx;\n out[8] = 1 - xx - yy;\n return out;\n}\n/**\n * Calculates a 3x3 normal matrix (transpose inverse) from the 4x4 matrix\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {ReadonlyMat4} a Mat4 to derive the normal matrix from\n *\n * @returns {mat3} out\n */\n\nexport function normalFromMat4(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n var b00 = a00 * a11 - a01 * a10;\n var b01 = a00 * a12 - a02 * a10;\n var b02 = a00 * a13 - a03 * a10;\n var b03 = a01 * a12 - a02 * a11;\n var b04 = a01 * a13 - a03 * a11;\n var b05 = a02 * a13 - a03 * a12;\n var b06 = a20 * a31 - a21 * a30;\n var b07 = a20 * a32 - a22 * a30;\n var b08 = a20 * a33 - a23 * a30;\n var b09 = a21 * a32 - a22 * a31;\n var b10 = a21 * a33 - a23 * a31;\n var b11 = a22 * a33 - a23 * a32; // Calculate the determinant\n\n var det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det;\n out[1] = (a12 * b08 - a10 * b11 - a13 * b07) * det;\n out[2] = (a10 * b10 - a11 * b08 + a13 * b06) * det;\n out[3] = (a02 * b10 - a01 * b11 - a03 * b09) * det;\n out[4] = (a00 * b11 - a02 * b08 + a03 * b07) * det;\n out[5] = (a01 * b08 - a00 * b10 - a03 * b06) * det;\n out[6] = (a31 * b05 - a32 * b04 + a33 * b03) * det;\n out[7] = (a32 * b02 - a30 * b05 - a33 * b01) * det;\n out[8] = (a30 * b04 - a31 * b02 + a33 * b00) * det;\n return out;\n}\n/**\n * Generates a 2D projection matrix with the given bounds\n *\n * @param {mat3} out mat3 frustum matrix will be written into\n * @param {number} width Width of your gl context\n * @param {number} height Height of gl context\n * @returns {mat3} out\n */\n\nexport function projection(out, width, height) {\n out[0] = 2 / width;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = -2 / height;\n out[5] = 0;\n out[6] = -1;\n out[7] = 1;\n out[8] = 1;\n return out;\n}\n/**\n * Returns a string representation of a mat3\n *\n * @param {ReadonlyMat3} a matrix to represent as a string\n * @returns {String} string representation of the matrix\n */\n\nexport function str(a) {\n return \"mat3(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \", \" + a[4] + \", \" + a[5] + \", \" + a[6] + \", \" + a[7] + \", \" + a[8] + \")\";\n}\n/**\n * Returns Frobenius norm of a mat3\n *\n * @param {ReadonlyMat3} a the matrix to calculate Frobenius norm of\n * @returns {Number} Frobenius norm\n */\n\nexport function frob(a) {\n return Math.hypot(a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8]);\n}\n/**\n * Adds two mat3's\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the first operand\n * @param {ReadonlyMat3} b the second operand\n * @returns {mat3} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n out[4] = a[4] + b[4];\n out[5] = a[5] + b[5];\n out[6] = a[6] + b[6];\n out[7] = a[7] + b[7];\n out[8] = a[8] + b[8];\n return out;\n}\n/**\n * Subtracts matrix b from matrix a\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the first operand\n * @param {ReadonlyMat3} b the second operand\n * @returns {mat3} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n out[4] = a[4] - b[4];\n out[5] = a[5] - b[5];\n out[6] = a[6] - b[6];\n out[7] = a[7] - b[7];\n out[8] = a[8] - b[8];\n return out;\n}\n/**\n * Multiply each element of the matrix by a scalar.\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the matrix to scale\n * @param {Number} b amount to scale the matrix's elements by\n * @returns {mat3} out\n */\n\nexport function multiplyScalar(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n out[4] = a[4] * b;\n out[5] = a[5] * b;\n out[6] = a[6] * b;\n out[7] = a[7] * b;\n out[8] = a[8] * b;\n return out;\n}\n/**\n * Adds two mat3's after multiplying each element of the second operand by a scalar value.\n *\n * @param {mat3} out the receiving vector\n * @param {ReadonlyMat3} a the first operand\n * @param {ReadonlyMat3} b the second operand\n * @param {Number} scale the amount to scale b's elements by before adding\n * @returns {mat3} out\n */\n\nexport function multiplyScalarAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n out[4] = a[4] + b[4] * scale;\n out[5] = a[5] + b[5] * scale;\n out[6] = a[6] + b[6] * scale;\n out[7] = a[7] + b[7] * scale;\n out[8] = a[8] + b[8] * scale;\n return out;\n}\n/**\n * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyMat3} a The first matrix.\n * @param {ReadonlyMat3} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5] && a[6] === b[6] && a[7] === b[7] && a[8] === b[8];\n}\n/**\n * Returns whether or not the matrices have approximately the same elements in the same position.\n *\n * @param {ReadonlyMat3} a The first matrix.\n * @param {ReadonlyMat3} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3],\n a4 = a[4],\n a5 = a[5],\n a6 = a[6],\n a7 = a[7],\n a8 = a[8];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3],\n b4 = b[4],\n b5 = b[5],\n b6 = b[6],\n b7 = b[7],\n b8 = b[8];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5)) && Math.abs(a6 - b6) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a6), Math.abs(b6)) && Math.abs(a7 - b7) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a7), Math.abs(b7)) && Math.abs(a8 - b8) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a8), Math.abs(b8));\n}\n/**\n * Alias for {@link mat3.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link mat3.subtract}\n * @function\n */\n\nexport var sub = subtract;","import {patternUniformValues} from './pattern';\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f,\n UniformMatrix4f\n} from '../uniform_binding';\nimport {extend} from '../../util/util';\n\nimport type {Painter} from '../painter';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Context} from '../../gl/context';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport type {Tile} from '../../source/tile';\nimport {mat4} from 'gl-matrix';\n\nexport type FillUniformsType = {\n 'u_matrix': UniformMatrix4f;\n};\n\nexport type FillOutlineUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_world': Uniform2f;\n};\n\nexport type FillPatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n // pattern uniforms:\n 'u_texsize': Uniform2f;\n 'u_image': Uniform1i;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n};\n\nexport type FillOutlinePatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_world': Uniform2f;\n // pattern uniforms:\n 'u_texsize': Uniform2f;\n 'u_image': Uniform1i;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n};\n\nconst fillUniforms = (context: Context, locations: UniformLocations): FillUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix)\n});\n\nconst fillPatternUniforms = (context: Context, locations: UniformLocations): FillPatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper),\n 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower),\n 'u_scale': new Uniform3f(context, locations.u_scale),\n 'u_fade': new Uniform1f(context, locations.u_fade)\n});\n\nconst fillOutlineUniforms = (context: Context, locations: UniformLocations): FillOutlineUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_world': new Uniform2f(context, locations.u_world)\n});\n\nconst fillOutlinePatternUniforms = (context: Context, locations: UniformLocations): FillOutlinePatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_world': new Uniform2f(context, locations.u_world),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper),\n 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower),\n 'u_scale': new Uniform3f(context, locations.u_scale),\n 'u_fade': new Uniform1f(context, locations.u_fade)\n});\n\nconst fillUniformValues = (matrix: mat4): UniformValues => ({\n 'u_matrix': matrix\n});\n\nconst fillPatternUniformValues = (\n matrix: mat4,\n painter: Painter,\n crossfade: CrossfadeParameters,\n tile: Tile\n): UniformValues => extend(\n fillUniformValues(matrix),\n patternUniformValues(crossfade, painter, tile)\n);\n\nconst fillOutlineUniformValues = (matrix: mat4, drawingBufferSize: [number, number]): UniformValues => ({\n 'u_matrix': matrix,\n 'u_world': drawingBufferSize\n});\n\nconst fillOutlinePatternUniformValues = (\n matrix: mat4,\n painter: Painter,\n crossfade: CrossfadeParameters,\n tile: Tile,\n drawingBufferSize: [number, number]\n): UniformValues => extend(\n fillPatternUniformValues(matrix, painter, crossfade, tile),\n {\n 'u_world': drawingBufferSize\n }\n);\n\nexport {\n fillUniforms,\n fillPatternUniforms,\n fillOutlineUniforms,\n fillOutlinePatternUniforms,\n fillUniformValues,\n fillPatternUniformValues,\n fillOutlineUniformValues,\n fillOutlinePatternUniformValues\n};\n","import {Uniform1i, Uniform1f, Uniform2f, UniformMatrix4f} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {OverscaledTileID} from '../../source/tile_id';\nimport type {Tile} from '../../source/tile';\nimport type {CircleStyleLayer} from '../../style/style_layer/circle_style_layer';\nimport type {Painter} from '../painter';\n\nexport type CircleUniformsType = {\n 'u_camera_to_center_distance': Uniform1f;\n 'u_scale_with_map': Uniform1i;\n 'u_pitch_with_map': Uniform1i;\n 'u_extrude_scale': Uniform2f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n};\n\nconst circleUniforms = (context: Context, locations: UniformLocations): CircleUniformsType => ({\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_scale_with_map': new Uniform1i(context, locations.u_scale_with_map),\n 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map),\n 'u_extrude_scale': new Uniform2f(context, locations.u_extrude_scale),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix)\n});\n\nconst circleUniformValues = (\n painter: Painter,\n coord: OverscaledTileID,\n tile: Tile,\n layer: CircleStyleLayer\n): UniformValues => {\n const transform = painter.transform;\n\n let pitchWithMap: boolean, extrudeScale: [number, number];\n if (layer.paint.get('circle-pitch-alignment') === 'map') {\n const pixelRatio = pixelsToTileUnits(tile, 1, transform.zoom);\n pitchWithMap = true;\n extrudeScale = [pixelRatio, pixelRatio];\n } else {\n pitchWithMap = false;\n extrudeScale = transform.pixelsToGLUnits;\n }\n\n return {\n 'u_camera_to_center_distance': transform.cameraToCenterDistance,\n 'u_scale_with_map': +(layer.paint.get('circle-pitch-scale') === 'map'),\n 'u_matrix': painter.translatePosMatrix(\n coord.posMatrix,\n tile,\n layer.paint.get('circle-translate'),\n layer.paint.get('circle-translate-anchor')),\n 'u_pitch_with_map': +(pitchWithMap),\n 'u_device_pixel_ratio': painter.pixelRatio,\n 'u_extrude_scale': extrudeScale\n };\n};\n\nexport {circleUniforms, circleUniformValues};\n","import {Uniform1f, Uniform2f, UniformMatrix4f} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Transform} from '../../geo/transform';\nimport type {Tile} from '../../source/tile';\nimport {mat4} from 'gl-matrix';\n\nexport type CollisionUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_pixels_to_tile_units': Uniform1f;\n 'u_extrude_scale': Uniform2f;\n 'u_overscale_factor': Uniform1f;\n};\n\nexport type CollisionCircleUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_inv_matrix': UniformMatrix4f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_viewport_size': Uniform2f;\n};\n\nconst collisionUniforms = (context: Context, locations: UniformLocations): CollisionUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_pixels_to_tile_units': new Uniform1f(context, locations.u_pixels_to_tile_units),\n 'u_extrude_scale': new Uniform2f(context, locations.u_extrude_scale),\n 'u_overscale_factor': new Uniform1f(context, locations.u_overscale_factor)\n});\n\nconst collisionCircleUniforms = (context: Context, locations: UniformLocations): CollisionCircleUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_inv_matrix': new UniformMatrix4f(context, locations.u_inv_matrix),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_viewport_size': new Uniform2f(context, locations.u_viewport_size)\n});\n\nconst collisionUniformValues = (matrix: mat4, transform: Transform, tile: Tile): UniformValues => {\n const pixelRatio = pixelsToTileUnits(tile, 1, transform.zoom);\n const scale = Math.pow(2, transform.zoom - tile.tileID.overscaledZ);\n const overscaleFactor = tile.tileID.overscaleFactor();\n return {\n 'u_matrix': matrix,\n 'u_camera_to_center_distance': transform.cameraToCenterDistance,\n 'u_pixels_to_tile_units': pixelRatio,\n 'u_extrude_scale': [transform.pixelsToGLUnits[0] / (pixelRatio * scale),\n transform.pixelsToGLUnits[1] / (pixelRatio * scale)],\n 'u_overscale_factor': overscaleFactor\n };\n};\n\nconst collisionCircleUniformValues = (matrix: mat4, invMatrix: mat4, transform: Transform): UniformValues => {\n return {\n 'u_matrix': matrix,\n 'u_inv_matrix': invMatrix,\n 'u_camera_to_center_distance': transform.cameraToCenterDistance,\n 'u_viewport_size': [transform.width, transform.height]\n };\n};\n\nexport {collisionUniforms, collisionUniformValues, collisionCircleUniforms, collisionCircleUniformValues};\n","import {UniformColor, UniformMatrix4f, Uniform1i, Uniform1f} from '../uniform_binding';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {mat4} from 'gl-matrix';\n\nexport type DebugUniformsType = {\n 'u_color': UniformColor;\n 'u_matrix': UniformMatrix4f;\n 'u_overlay': Uniform1i;\n 'u_overlay_scale': Uniform1f;\n};\n\nconst debugUniforms = (context: Context, locations: UniformLocations): DebugUniformsType => ({\n 'u_color': new UniformColor(context, locations.u_color),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_overlay': new Uniform1i(context, locations.u_overlay),\n 'u_overlay_scale': new Uniform1f(context, locations.u_overlay_scale)\n});\n\nconst debugUniformValues = (matrix: mat4, color: Color, scaleRatio: number = 1): UniformValues => ({\n 'u_matrix': matrix,\n 'u_color': color,\n 'u_overlay': 0,\n 'u_overlay_scale': scaleRatio\n});\n\nexport {debugUniforms, debugUniformValues};\n","import {UniformMatrix4f} from '../uniform_binding';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport {mat4} from 'gl-matrix';\n\nexport type ClippingMaskUniformsType = {\n 'u_matrix': UniformMatrix4f;\n};\n\nconst clippingMaskUniforms = (context: Context, locations: UniformLocations): ClippingMaskUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix)\n});\n\nconst clippingMaskUniformValues = (matrix: mat4): UniformValues => ({\n 'u_matrix': matrix\n});\n\nexport {clippingMaskUniforms, clippingMaskUniformValues};\n","import {mat4} from 'gl-matrix';\n\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n UniformMatrix4f\n} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\n\nimport type {Context} from '../../gl/context';\nimport type {Tile} from '../../source/tile';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Painter} from '../painter';\nimport type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer';\n\nexport type HeatmapUniformsType = {\n 'u_extrude_scale': Uniform1f;\n 'u_intensity': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n};\n\nexport type HeatmapTextureUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_world': Uniform2f;\n 'u_image': Uniform1i;\n 'u_color_ramp': Uniform1i;\n 'u_opacity': Uniform1f;\n};\n\nconst heatmapUniforms = (context: Context, locations: UniformLocations): HeatmapUniformsType => ({\n 'u_extrude_scale': new Uniform1f(context, locations.u_extrude_scale),\n 'u_intensity': new Uniform1f(context, locations.u_intensity),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix)\n});\n\nconst heatmapTextureUniforms = (context: Context, locations: UniformLocations): HeatmapTextureUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_world': new Uniform2f(context, locations.u_world),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_color_ramp': new Uniform1i(context, locations.u_color_ramp),\n 'u_opacity': new Uniform1f(context, locations.u_opacity)\n});\n\nconst heatmapUniformValues = (matrix: mat4, tile: Tile, zoom: number, intensity: number): UniformValues => ({\n 'u_matrix': matrix,\n 'u_extrude_scale': pixelsToTileUnits(tile, 1, zoom),\n 'u_intensity': intensity\n});\n\nconst heatmapTextureUniformValues = (\n painter: Painter,\n layer: HeatmapStyleLayer,\n textureUnit: number,\n colorRampUnit: number\n): UniformValues => {\n const matrix = mat4.create();\n mat4.ortho(matrix, 0, painter.width, painter.height, 0, 0, 1);\n\n const gl = painter.context.gl;\n\n return {\n 'u_matrix': matrix,\n 'u_world': [gl.drawingBufferWidth, gl.drawingBufferHeight],\n 'u_image': textureUnit,\n 'u_color_ramp': colorRampUnit,\n 'u_opacity': layer.paint.get('heatmap-opacity')\n };\n};\n\nexport {\n heatmapUniforms,\n heatmapTextureUniforms,\n heatmapUniformValues,\n heatmapTextureUniformValues\n};\n","import {mat4} from 'gl-matrix';\n\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n UniformColor,\n UniformMatrix4f,\n Uniform4f\n} from '../uniform_binding';\nimport {EXTENT} from '../../data/extent';\nimport {MercatorCoordinate} from '../../geo/mercator_coordinate';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Tile} from '../../source/tile';\nimport type {Painter} from '../painter';\nimport type {HillshadeStyleLayer} from '../../style/style_layer/hillshade_style_layer';\nimport type {DEMData} from '../../data/dem_data';\nimport type {OverscaledTileID} from '../../source/tile_id';\n\nexport type HillshadeUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_image': Uniform1i;\n 'u_latrange': Uniform2f;\n 'u_light': Uniform2f;\n 'u_shadow': UniformColor;\n 'u_highlight': UniformColor;\n 'u_accent': UniformColor;\n};\n\nexport type HillshadePrepareUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_image': Uniform1i;\n 'u_dimension': Uniform2f;\n 'u_zoom': Uniform1f;\n 'u_unpack': Uniform4f;\n};\n\nconst hillshadeUniforms = (context: Context, locations: UniformLocations): HillshadeUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_latrange': new Uniform2f(context, locations.u_latrange),\n 'u_light': new Uniform2f(context, locations.u_light),\n 'u_shadow': new UniformColor(context, locations.u_shadow),\n 'u_highlight': new UniformColor(context, locations.u_highlight),\n 'u_accent': new UniformColor(context, locations.u_accent)\n});\n\nconst hillshadePrepareUniforms = (context: Context, locations: UniformLocations): HillshadePrepareUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_dimension': new Uniform2f(context, locations.u_dimension),\n 'u_zoom': new Uniform1f(context, locations.u_zoom),\n 'u_unpack': new Uniform4f(context, locations.u_unpack)\n});\n\nconst hillshadeUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: HillshadeStyleLayer,\n coord: OverscaledTileID\n): UniformValues => {\n const shadow = layer.paint.get('hillshade-shadow-color');\n const highlight = layer.paint.get('hillshade-highlight-color');\n const accent = layer.paint.get('hillshade-accent-color');\n\n let azimuthal = layer.paint.get('hillshade-illumination-direction') * (Math.PI / 180);\n // modify azimuthal angle by map rotation if light is anchored at the viewport\n if (layer.paint.get('hillshade-illumination-anchor') === 'viewport') {\n azimuthal -= painter.transform.angle;\n }\n const align = !painter.options.moving;\n return {\n 'u_matrix': coord ? coord.posMatrix : painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped(), align),\n 'u_image': 0,\n 'u_latrange': getTileLatRange(painter, tile.tileID),\n 'u_light': [layer.paint.get('hillshade-exaggeration'), azimuthal],\n 'u_shadow': shadow,\n 'u_highlight': highlight,\n 'u_accent': accent\n };\n};\n\nconst hillshadeUniformPrepareValues = (tileID: OverscaledTileID, dem: DEMData): UniformValues => {\n\n const stride = dem.stride;\n const matrix = mat4.create();\n // Flip rendering at y axis.\n mat4.ortho(matrix, 0, EXTENT, -EXTENT, 0, 0, 1);\n mat4.translate(matrix, matrix, [0, -EXTENT, 0]);\n\n return {\n 'u_matrix': matrix,\n 'u_image': 1,\n 'u_dimension': [stride, stride],\n 'u_zoom': tileID.overscaledZ,\n 'u_unpack': dem.getUnpackVector()\n };\n};\n\nfunction getTileLatRange(painter: Painter, tileID: OverscaledTileID) {\n // for scaling the magnitude of a points slope by its latitude\n const tilesAtZoom = Math.pow(2, tileID.canonical.z);\n const y = tileID.canonical.y;\n return [\n new MercatorCoordinate(0, y / tilesAtZoom).toLngLat().lat,\n new MercatorCoordinate(0, (y + 1) / tilesAtZoom).toLngLat().lat];\n}\n\nexport {\n hillshadeUniforms,\n hillshadePrepareUniforms,\n hillshadeUniformValues,\n hillshadeUniformPrepareValues\n};\n","import {Uniform1i, Uniform1f, Uniform2f, Uniform3f, UniformMatrix4f} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\nimport {extend} from '../../util/util';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Transform} from '../../geo/transform';\nimport type {Tile} from '../../source/tile';\nimport type {CrossFaded} from '../../style/properties';\nimport type {LineStyleLayer} from '../../style/style_layer/line_style_layer';\nimport type {Painter} from '../painter';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport {OverscaledTileID} from '../../source/tile_id';\n\nexport type LineUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_ratio': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_units_to_pixels': Uniform2f;\n};\n\nexport type LineGradientUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_ratio': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_units_to_pixels': Uniform2f;\n 'u_image': Uniform1i;\n 'u_image_height': Uniform1f;\n};\n\nexport type LinePatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_texsize': Uniform2f;\n 'u_ratio': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_units_to_pixels': Uniform2f;\n 'u_image': Uniform1i;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n};\n\nexport type LineSDFUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_ratio': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_units_to_pixels': Uniform2f;\n 'u_patternscale_a': Uniform2f;\n 'u_patternscale_b': Uniform2f;\n 'u_sdfgamma': Uniform1f;\n 'u_image': Uniform1i;\n 'u_tex_y_a': Uniform1f;\n 'u_tex_y_b': Uniform1f;\n 'u_mix': Uniform1f;\n};\n\nconst lineUniforms = (context: Context, locations: UniformLocations): LineUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_ratio': new Uniform1f(context, locations.u_ratio),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels)\n});\n\nconst lineGradientUniforms = (context: Context, locations: UniformLocations): LineGradientUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_ratio': new Uniform1f(context, locations.u_ratio),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_image_height': new Uniform1f(context, locations.u_image_height)\n});\n\nconst linePatternUniforms = (context: Context, locations: UniformLocations): LinePatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_ratio': new Uniform1f(context, locations.u_ratio),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels),\n 'u_scale': new Uniform3f(context, locations.u_scale),\n 'u_fade': new Uniform1f(context, locations.u_fade)\n});\n\nconst lineSDFUniforms = (context: Context, locations: UniformLocations): LineSDFUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_ratio': new Uniform1f(context, locations.u_ratio),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels),\n 'u_patternscale_a': new Uniform2f(context, locations.u_patternscale_a),\n 'u_patternscale_b': new Uniform2f(context, locations.u_patternscale_b),\n 'u_sdfgamma': new Uniform1f(context, locations.u_sdfgamma),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_tex_y_a': new Uniform1f(context, locations.u_tex_y_a),\n 'u_tex_y_b': new Uniform1f(context, locations.u_tex_y_b),\n 'u_mix': new Uniform1f(context, locations.u_mix)\n});\n\nconst lineUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: LineStyleLayer,\n coord: OverscaledTileID\n): UniformValues => {\n const transform = painter.transform;\n\n return {\n 'u_matrix': calculateMatrix(painter, tile, layer, coord),\n 'u_ratio': 1 / pixelsToTileUnits(tile, 1, transform.zoom),\n 'u_device_pixel_ratio': painter.pixelRatio,\n 'u_units_to_pixels': [\n 1 / transform.pixelsToGLUnits[0],\n 1 / transform.pixelsToGLUnits[1]\n ]\n };\n};\n\nconst lineGradientUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: LineStyleLayer,\n imageHeight: number,\n coord: OverscaledTileID\n): UniformValues => {\n return extend(lineUniformValues(painter, tile, layer, coord), {\n 'u_image': 0,\n 'u_image_height': imageHeight,\n });\n};\n\nconst linePatternUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: LineStyleLayer,\n crossfade: CrossfadeParameters,\n coord: OverscaledTileID\n): UniformValues => {\n const transform = painter.transform;\n const tileZoomRatio = calculateTileRatio(tile, transform);\n return {\n 'u_matrix': calculateMatrix(painter, tile, layer, coord),\n 'u_texsize': tile.imageAtlasTexture.size,\n // camera zoom ratio\n 'u_ratio': 1 / pixelsToTileUnits(tile, 1, transform.zoom),\n 'u_device_pixel_ratio': painter.pixelRatio,\n 'u_image': 0,\n 'u_scale': [tileZoomRatio, crossfade.fromScale, crossfade.toScale],\n 'u_fade': crossfade.t,\n 'u_units_to_pixels': [\n 1 / transform.pixelsToGLUnits[0],\n 1 / transform.pixelsToGLUnits[1]\n ]\n };\n};\n\nconst lineSDFUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: LineStyleLayer,\n dasharray: CrossFaded>,\n crossfade: CrossfadeParameters,\n coord: OverscaledTileID\n): UniformValues => {\n const transform = painter.transform;\n const lineAtlas = painter.lineAtlas;\n const tileRatio = calculateTileRatio(tile, transform);\n\n const round = layer.layout.get('line-cap') === 'round';\n\n const posA = lineAtlas.getDash(dasharray.from, round);\n const posB = lineAtlas.getDash(dasharray.to, round);\n\n const widthA = posA.width * crossfade.fromScale;\n const widthB = posB.width * crossfade.toScale;\n\n return extend(lineUniformValues(painter, tile, layer, coord), {\n 'u_patternscale_a': [tileRatio / widthA, -posA.height / 2],\n 'u_patternscale_b': [tileRatio / widthB, -posB.height / 2],\n 'u_sdfgamma': lineAtlas.width / (Math.min(widthA, widthB) * 256 * painter.pixelRatio) / 2,\n 'u_image': 0,\n 'u_tex_y_a': posA.y,\n 'u_tex_y_b': posB.y,\n 'u_mix': crossfade.t\n });\n};\n\nfunction calculateTileRatio(tile: Tile, transform: Transform) {\n return 1 / pixelsToTileUnits(tile, 1, transform.tileZoom);\n}\n\nfunction calculateMatrix(painter: Painter, tile: Tile, layer: LineStyleLayer, coord: OverscaledTileID) {\n return painter.translatePosMatrix(\n coord ? coord.posMatrix : tile.tileID.posMatrix,\n tile,\n layer.paint.get('line-translate'),\n layer.paint.get('line-translate-anchor')\n );\n}\n\nexport {\n lineUniforms,\n lineGradientUniforms,\n linePatternUniforms,\n lineSDFUniforms,\n lineUniformValues,\n lineGradientUniformValues,\n linePatternUniformValues,\n lineSDFUniformValues\n};\n","import {Uniform1i, Uniform1f, Uniform2f, Uniform3f, UniformMatrix4f} from '../uniform_binding';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {RasterStyleLayer} from '../../style/style_layer/raster_style_layer';\nimport {mat4} from 'gl-matrix';\n\nexport type RasterUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_tl_parent': Uniform2f;\n 'u_scale_parent': Uniform1f;\n 'u_buffer_scale': Uniform1f;\n 'u_fade_t': Uniform1f;\n 'u_opacity': Uniform1f;\n 'u_image0': Uniform1i;\n 'u_image1': Uniform1i;\n 'u_brightness_low': Uniform1f;\n 'u_brightness_high': Uniform1f;\n 'u_saturation_factor': Uniform1f;\n 'u_contrast_factor': Uniform1f;\n 'u_spin_weights': Uniform3f;\n};\n\nconst rasterUniforms = (context: Context, locations: UniformLocations): RasterUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_tl_parent': new Uniform2f(context, locations.u_tl_parent),\n 'u_scale_parent': new Uniform1f(context, locations.u_scale_parent),\n 'u_buffer_scale': new Uniform1f(context, locations.u_buffer_scale),\n 'u_fade_t': new Uniform1f(context, locations.u_fade_t),\n 'u_opacity': new Uniform1f(context, locations.u_opacity),\n 'u_image0': new Uniform1i(context, locations.u_image0),\n 'u_image1': new Uniform1i(context, locations.u_image1),\n 'u_brightness_low': new Uniform1f(context, locations.u_brightness_low),\n 'u_brightness_high': new Uniform1f(context, locations.u_brightness_high),\n 'u_saturation_factor': new Uniform1f(context, locations.u_saturation_factor),\n 'u_contrast_factor': new Uniform1f(context, locations.u_contrast_factor),\n 'u_spin_weights': new Uniform3f(context, locations.u_spin_weights)\n});\n\nconst rasterUniformValues = (\n matrix: mat4,\n parentTL: [number, number],\n parentScaleBy: number,\n fade: {\n mix: number;\n opacity: number;\n },\n layer: RasterStyleLayer\n): UniformValues => ({\n 'u_matrix': matrix,\n 'u_tl_parent': parentTL,\n 'u_scale_parent': parentScaleBy,\n 'u_buffer_scale': 1,\n 'u_fade_t': fade.mix,\n 'u_opacity': fade.opacity * layer.paint.get('raster-opacity'),\n 'u_image0': 0,\n 'u_image1': 1,\n 'u_brightness_low': layer.paint.get('raster-brightness-min'),\n 'u_brightness_high': layer.paint.get('raster-brightness-max'),\n 'u_saturation_factor': saturationFactor(layer.paint.get('raster-saturation')),\n 'u_contrast_factor': contrastFactor(layer.paint.get('raster-contrast')),\n 'u_spin_weights': spinWeights(layer.paint.get('raster-hue-rotate'))\n});\n\nfunction spinWeights(angle) {\n angle *= Math.PI / 180;\n const s = Math.sin(angle);\n const c = Math.cos(angle);\n return [\n (2 * c + 1) / 3,\n (-Math.sqrt(3) * s - c + 1) / 3,\n (Math.sqrt(3) * s - c + 1) / 3\n ];\n}\n\nfunction contrastFactor(contrast) {\n return contrast > 0 ?\n 1 / (1 - contrast) :\n 1 + contrast;\n}\n\nfunction saturationFactor(saturation) {\n return saturation > 0 ?\n 1 - 1 / (1.001 - saturation) :\n -saturation;\n}\n\nexport {rasterUniforms, rasterUniformValues};\n","import {Uniform1i, Uniform1f, Uniform2f, UniformMatrix4f} from '../uniform_binding';\nimport {extend} from '../../util/util';\n\nimport type {Context} from '../../gl/context';\nimport type {Painter} from '../painter';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport {mat4} from 'gl-matrix';\n\nexport type SymbolIconUniformsType = {\n 'u_is_size_zoom_constant': Uniform1i;\n 'u_is_size_feature_constant': Uniform1i;\n 'u_size_t': Uniform1f;\n 'u_size': Uniform1f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_pitch': Uniform1f;\n 'u_rotate_symbol': Uniform1i;\n 'u_aspect_ratio': Uniform1f;\n 'u_fade_change': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n 'u_label_plane_matrix': UniformMatrix4f;\n 'u_coord_matrix': UniformMatrix4f;\n 'u_is_text': Uniform1i;\n 'u_pitch_with_map': Uniform1i;\n 'u_texsize': Uniform2f;\n 'u_texture': Uniform1i;\n};\n\nexport type SymbolSDFUniformsType = {\n 'u_is_size_zoom_constant': Uniform1i;\n 'u_is_size_feature_constant': Uniform1i;\n 'u_size_t': Uniform1f;\n 'u_size': Uniform1f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_pitch': Uniform1f;\n 'u_rotate_symbol': Uniform1i;\n 'u_aspect_ratio': Uniform1f;\n 'u_fade_change': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n 'u_label_plane_matrix': UniformMatrix4f;\n 'u_coord_matrix': UniformMatrix4f;\n 'u_is_text': Uniform1i;\n 'u_pitch_with_map': Uniform1i;\n 'u_texsize': Uniform2f;\n 'u_texture': Uniform1i;\n 'u_gamma_scale': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_is_halo': Uniform1i;\n};\n\nexport type symbolTextAndIconUniformsType = {\n 'u_is_size_zoom_constant': Uniform1i;\n 'u_is_size_feature_constant': Uniform1i;\n 'u_size_t': Uniform1f;\n 'u_size': Uniform1f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_pitch': Uniform1f;\n 'u_rotate_symbol': Uniform1i;\n 'u_aspect_ratio': Uniform1f;\n 'u_fade_change': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n 'u_label_plane_matrix': UniformMatrix4f;\n 'u_coord_matrix': UniformMatrix4f;\n 'u_is_text': Uniform1i;\n 'u_pitch_with_map': Uniform1i;\n 'u_texsize': Uniform2f;\n 'u_texsize_icon': Uniform2f;\n 'u_texture': Uniform1i;\n 'u_texture_icon': Uniform1i;\n 'u_gamma_scale': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_is_halo': Uniform1i;\n};\n\nconst symbolIconUniforms = (context: Context, locations: UniformLocations): SymbolIconUniformsType => ({\n 'u_is_size_zoom_constant': new Uniform1i(context, locations.u_is_size_zoom_constant),\n 'u_is_size_feature_constant': new Uniform1i(context, locations.u_is_size_feature_constant),\n 'u_size_t': new Uniform1f(context, locations.u_size_t),\n 'u_size': new Uniform1f(context, locations.u_size),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_pitch': new Uniform1f(context, locations.u_pitch),\n 'u_rotate_symbol': new Uniform1i(context, locations.u_rotate_symbol),\n 'u_aspect_ratio': new Uniform1f(context, locations.u_aspect_ratio),\n 'u_fade_change': new Uniform1f(context, locations.u_fade_change),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_label_plane_matrix': new UniformMatrix4f(context, locations.u_label_plane_matrix),\n 'u_coord_matrix': new UniformMatrix4f(context, locations.u_coord_matrix),\n 'u_is_text': new Uniform1i(context, locations.u_is_text),\n 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_texture': new Uniform1i(context, locations.u_texture)\n});\n\nconst symbolSDFUniforms = (context: Context, locations: UniformLocations): SymbolSDFUniformsType => ({\n 'u_is_size_zoom_constant': new Uniform1i(context, locations.u_is_size_zoom_constant),\n 'u_is_size_feature_constant': new Uniform1i(context, locations.u_is_size_feature_constant),\n 'u_size_t': new Uniform1f(context, locations.u_size_t),\n 'u_size': new Uniform1f(context, locations.u_size),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_pitch': new Uniform1f(context, locations.u_pitch),\n 'u_rotate_symbol': new Uniform1i(context, locations.u_rotate_symbol),\n 'u_aspect_ratio': new Uniform1f(context, locations.u_aspect_ratio),\n 'u_fade_change': new Uniform1f(context, locations.u_fade_change),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_label_plane_matrix': new UniformMatrix4f(context, locations.u_label_plane_matrix),\n 'u_coord_matrix': new UniformMatrix4f(context, locations.u_coord_matrix),\n 'u_is_text': new Uniform1i(context, locations.u_is_text),\n 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_texture': new Uniform1i(context, locations.u_texture),\n 'u_gamma_scale': new Uniform1f(context, locations.u_gamma_scale),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_is_halo': new Uniform1i(context, locations.u_is_halo)\n});\n\nconst symbolTextAndIconUniforms = (context: Context, locations: UniformLocations): symbolTextAndIconUniformsType => ({\n 'u_is_size_zoom_constant': new Uniform1i(context, locations.u_is_size_zoom_constant),\n 'u_is_size_feature_constant': new Uniform1i(context, locations.u_is_size_feature_constant),\n 'u_size_t': new Uniform1f(context, locations.u_size_t),\n 'u_size': new Uniform1f(context, locations.u_size),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_pitch': new Uniform1f(context, locations.u_pitch),\n 'u_rotate_symbol': new Uniform1i(context, locations.u_rotate_symbol),\n 'u_aspect_ratio': new Uniform1f(context, locations.u_aspect_ratio),\n 'u_fade_change': new Uniform1f(context, locations.u_fade_change),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_label_plane_matrix': new UniformMatrix4f(context, locations.u_label_plane_matrix),\n 'u_coord_matrix': new UniformMatrix4f(context, locations.u_coord_matrix),\n 'u_is_text': new Uniform1i(context, locations.u_is_text),\n 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_texsize_icon': new Uniform2f(context, locations.u_texsize_icon),\n 'u_texture': new Uniform1i(context, locations.u_texture),\n 'u_texture_icon': new Uniform1i(context, locations.u_texture_icon),\n 'u_gamma_scale': new Uniform1f(context, locations.u_gamma_scale),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_is_halo': new Uniform1i(context, locations.u_is_halo)\n});\n\nconst symbolIconUniformValues = (\n functionType: string,\n size: {\n uSizeT: number;\n uSize: number;\n },\n rotateInShader: boolean,\n pitchWithMap: boolean,\n painter: Painter,\n matrix: mat4,\n labelPlaneMatrix: mat4,\n glCoordMatrix: mat4,\n isText: boolean,\n texSize: [number, number]\n): UniformValues => {\n const transform = painter.transform;\n\n return {\n 'u_is_size_zoom_constant': +(functionType === 'constant' || functionType === 'source'),\n 'u_is_size_feature_constant': +(functionType === 'constant' || functionType === 'camera'),\n 'u_size_t': size ? size.uSizeT : 0,\n 'u_size': size ? size.uSize : 0,\n 'u_camera_to_center_distance': transform.cameraToCenterDistance,\n 'u_pitch': transform.pitch / 360 * 2 * Math.PI,\n 'u_rotate_symbol': +rotateInShader,\n 'u_aspect_ratio': transform.width / transform.height,\n 'u_fade_change': painter.options.fadeDuration ? painter.symbolFadeChange : 1,\n 'u_matrix': matrix,\n 'u_label_plane_matrix': labelPlaneMatrix,\n 'u_coord_matrix': glCoordMatrix,\n 'u_is_text': +isText,\n 'u_pitch_with_map': +pitchWithMap,\n 'u_texsize': texSize,\n 'u_texture': 0\n };\n};\n\nconst symbolSDFUniformValues = (\n functionType: string,\n size: {\n uSizeT: number;\n uSize: number;\n },\n rotateInShader: boolean,\n pitchWithMap: boolean,\n painter: Painter,\n matrix: mat4,\n labelPlaneMatrix: mat4,\n glCoordMatrix: mat4,\n isText: boolean,\n texSize: [number, number],\n isHalo: boolean\n): UniformValues => {\n const transform = painter.transform;\n\n return extend(symbolIconUniformValues(functionType, size,\n rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix,\n glCoordMatrix, isText, texSize), {\n 'u_gamma_scale': (pitchWithMap ? Math.cos(transform._pitch) * transform.cameraToCenterDistance : 1),\n 'u_device_pixel_ratio': painter.pixelRatio,\n 'u_is_halo': +isHalo\n });\n};\n\nconst symbolTextAndIconUniformValues = (\n functionType: string,\n size: {\n uSizeT: number;\n uSize: number;\n },\n rotateInShader: boolean,\n pitchWithMap: boolean,\n painter: Painter,\n matrix: mat4,\n labelPlaneMatrix: mat4,\n glCoordMatrix: mat4,\n texSizeSDF: [number, number],\n texSizeIcon: [number, number]\n): UniformValues => {\n return extend(symbolSDFUniformValues(functionType, size,\n rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix,\n glCoordMatrix, true, texSizeSDF, true), {\n 'u_texsize_icon': texSizeIcon,\n 'u_texture_icon': 1\n });\n};\n\nexport {symbolIconUniforms, symbolSDFUniforms, symbolIconUniformValues, symbolSDFUniformValues, symbolTextAndIconUniformValues, symbolTextAndIconUniforms};\n","import {bgPatternUniformValues} from './pattern';\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n UniformColor,\n UniformMatrix4f\n} from '../uniform_binding';\nimport {extend} from '../../util/util';\n\nimport type {Painter} from '../painter';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Context} from '../../gl/context';\nimport type {Color, ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport type {CrossFaded} from '../../style/properties';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport type {OverscaledTileID} from '../../source/tile_id';\nimport {mat4} from 'gl-matrix';\n\nexport type BackgroundUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_opacity': Uniform1f;\n 'u_color': UniformColor;\n};\n\nexport type BackgroundPatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_opacity': Uniform1f;\n // pattern uniforms:\n 'u_image': Uniform1i;\n 'u_pattern_tl_a': Uniform2f;\n 'u_pattern_br_a': Uniform2f;\n 'u_pattern_tl_b': Uniform2f;\n 'u_pattern_br_b': Uniform2f;\n 'u_texsize': Uniform2f;\n 'u_mix': Uniform1f;\n 'u_pattern_size_a': Uniform2f;\n 'u_pattern_size_b': Uniform2f;\n 'u_scale_a': Uniform1f;\n 'u_scale_b': Uniform1f;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_tile_units_to_pixels': Uniform1f;\n};\n\nconst backgroundUniforms = (context: Context, locations: UniformLocations): BackgroundUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_opacity': new Uniform1f(context, locations.u_opacity),\n 'u_color': new UniformColor(context, locations.u_color)\n});\n\nconst backgroundPatternUniforms = (context: Context, locations: UniformLocations): BackgroundPatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_opacity': new Uniform1f(context, locations.u_opacity),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_pattern_tl_a': new Uniform2f(context, locations.u_pattern_tl_a),\n 'u_pattern_br_a': new Uniform2f(context, locations.u_pattern_br_a),\n 'u_pattern_tl_b': new Uniform2f(context, locations.u_pattern_tl_b),\n 'u_pattern_br_b': new Uniform2f(context, locations.u_pattern_br_b),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_mix': new Uniform1f(context, locations.u_mix),\n 'u_pattern_size_a': new Uniform2f(context, locations.u_pattern_size_a),\n 'u_pattern_size_b': new Uniform2f(context, locations.u_pattern_size_b),\n 'u_scale_a': new Uniform1f(context, locations.u_scale_a),\n 'u_scale_b': new Uniform1f(context, locations.u_scale_b),\n 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper),\n 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower),\n 'u_tile_units_to_pixels': new Uniform1f(context, locations.u_tile_units_to_pixels)\n});\n\nconst backgroundUniformValues = (matrix: mat4, opacity: number, color: Color): UniformValues => ({\n 'u_matrix': matrix,\n 'u_opacity': opacity,\n 'u_color': color\n});\n\nconst backgroundPatternUniformValues = (\n matrix: mat4,\n opacity: number,\n painter: Painter,\n image: CrossFaded,\n tile: {\n tileID: OverscaledTileID;\n tileSize: number;\n },\n crossfade: CrossfadeParameters\n): UniformValues => extend(\n bgPatternUniformValues(image, crossfade, painter, tile),\n {\n 'u_matrix': matrix,\n 'u_opacity': opacity\n }\n);\n\nexport {\n backgroundUniforms,\n backgroundPatternUniforms,\n backgroundUniformValues,\n backgroundPatternUniformValues\n};\n","import {fillExtrusionUniforms, fillExtrusionPatternUniforms} from './fill_extrusion_program';\nimport {fillUniforms, fillPatternUniforms, fillOutlineUniforms, fillOutlinePatternUniforms} from './fill_program';\nimport {circleUniforms} from './circle_program';\nimport {collisionUniforms, collisionCircleUniforms} from './collision_program';\nimport {debugUniforms} from './debug_program';\nimport {clippingMaskUniforms} from './clipping_mask_program';\nimport {heatmapUniforms, heatmapTextureUniforms} from './heatmap_program';\nimport {hillshadeUniforms, hillshadePrepareUniforms} from './hillshade_program';\nimport {lineUniforms, lineGradientUniforms, linePatternUniforms, lineSDFUniforms} from './line_program';\nimport {rasterUniforms} from './raster_program';\nimport {symbolIconUniforms, symbolSDFUniforms, symbolTextAndIconUniforms} from './symbol_program';\nimport {backgroundUniforms, backgroundPatternUniforms} from './background_program';\nimport {terrainUniforms, terrainDepthUniforms, terrainCoordsUniforms} from './terrain_program';\n\nexport const programUniforms = {\n fillExtrusion: fillExtrusionUniforms,\n fillExtrusionPattern: fillExtrusionPatternUniforms,\n fill: fillUniforms,\n fillPattern: fillPatternUniforms,\n fillOutline: fillOutlineUniforms,\n fillOutlinePattern: fillOutlinePatternUniforms,\n circle: circleUniforms,\n collisionBox: collisionUniforms,\n collisionCircle: collisionCircleUniforms,\n debug: debugUniforms,\n clippingMask: clippingMaskUniforms,\n heatmap: heatmapUniforms,\n heatmapTexture: heatmapTextureUniforms,\n hillshade: hillshadeUniforms,\n hillshadePrepare: hillshadePrepareUniforms,\n line: lineUniforms,\n lineGradient: lineGradientUniforms,\n linePattern: linePatternUniforms,\n lineSDF: lineSDFUniforms,\n raster: rasterUniforms,\n symbolIcon: symbolIconUniforms,\n symbolSDF: symbolSDFUniforms,\n symbolTextAndIcon: symbolTextAndIconUniforms,\n background: backgroundUniforms,\n backgroundPattern: backgroundPatternUniforms,\n terrain: terrainUniforms,\n terrainDepth: terrainDepthUniforms,\n terrainCoords: terrainCoordsUniforms\n};\n","\nimport type {StructArray} from '../util/struct_array';\nimport type {TriangleIndexArray, LineIndexArray, LineStripIndexArray} from '../data/index_array_type';\nimport type {Context} from '../gl/context';\n\n/**\n * @internal\n * an index buffer class\n */\nexport class IndexBuffer {\n context: Context;\n buffer: WebGLBuffer;\n dynamicDraw: boolean;\n\n constructor(context: Context, array: TriangleIndexArray | LineIndexArray | LineStripIndexArray, dynamicDraw?: boolean) {\n this.context = context;\n const gl = context.gl;\n this.buffer = gl.createBuffer();\n this.dynamicDraw = Boolean(dynamicDraw);\n\n // The bound index buffer is part of vertex array object state. We don't want to\n // modify whatever VAO happens to be currently bound, so make sure the default\n // vertex array provided by the context is bound instead.\n this.context.unbindVAO();\n\n context.bindElementBuffer.set(this.buffer);\n gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, array.arrayBuffer, this.dynamicDraw ? gl.DYNAMIC_DRAW : gl.STATIC_DRAW);\n\n if (!this.dynamicDraw) {\n delete array.arrayBuffer;\n }\n }\n\n bind() {\n this.context.bindElementBuffer.set(this.buffer);\n }\n\n updateData(array: StructArray) {\n const gl = this.context.gl;\n if (!this.dynamicDraw) throw new Error('Attempted to update data while not in dynamic mode.');\n // The right VAO will get this buffer re-bound later in VertexArrayObject#bind\n // See https://github.com/mapbox/mapbox-gl-js/issues/5620\n this.context.unbindVAO();\n this.bind();\n gl.bufferSubData(gl.ELEMENT_ARRAY_BUFFER, 0, array.arrayBuffer);\n }\n\n destroy() {\n const gl = this.context.gl;\n if (this.buffer) {\n gl.deleteBuffer(this.buffer);\n delete this.buffer;\n }\n }\n}\n","\nimport type {\n StructArray,\n StructArrayMember\n} from '../util/struct_array';\n\nimport type {Program} from '../render/program';\nimport type {Context} from '../gl/context';\n\n/**\n * An Enum for AttributeType\n */\nconst AttributeType = {\n Int8: 'BYTE',\n Uint8: 'UNSIGNED_BYTE',\n Int16: 'SHORT',\n Uint16: 'UNSIGNED_SHORT',\n Int32: 'INT',\n Uint32: 'UNSIGNED_INT',\n Float32: 'FLOAT'\n};\n\n/**\n * @internal\n * The `VertexBuffer` class turns a `StructArray` into a WebGL buffer. Each member of the StructArray's\n * Struct type is converted to a WebGL attribute.\n */\nexport class VertexBuffer {\n length: number;\n attributes: ReadonlyArray;\n itemSize: number;\n dynamicDraw: boolean;\n context: Context;\n buffer: WebGLBuffer;\n\n /**\n * @param dynamicDraw - Whether this buffer will be repeatedly updated.\n */\n constructor(context: Context, array: StructArray, attributes: ReadonlyArray, dynamicDraw?: boolean) {\n this.length = array.length;\n this.attributes = attributes;\n this.itemSize = array.bytesPerElement;\n this.dynamicDraw = dynamicDraw;\n\n this.context = context;\n const gl = context.gl;\n this.buffer = gl.createBuffer();\n context.bindVertexBuffer.set(this.buffer);\n gl.bufferData(gl.ARRAY_BUFFER, array.arrayBuffer, this.dynamicDraw ? gl.DYNAMIC_DRAW : gl.STATIC_DRAW);\n\n if (!this.dynamicDraw) {\n delete array.arrayBuffer;\n }\n }\n\n bind() {\n this.context.bindVertexBuffer.set(this.buffer);\n }\n\n updateData(array: StructArray) {\n if (array.length !== this.length) throw new Error(`Length of new data is ${array.length}, which doesn't match current length of ${this.length}`);\n const gl = this.context.gl;\n this.bind();\n gl.bufferSubData(gl.ARRAY_BUFFER, 0, array.arrayBuffer);\n }\n\n enableAttributes(gl: WebGLRenderingContext|WebGL2RenderingContext, program: Program) {\n for (let j = 0; j < this.attributes.length; j++) {\n const member = this.attributes[j];\n const attribIndex: number | void = program.attributes[member.name];\n if (attribIndex !== undefined) {\n gl.enableVertexAttribArray(attribIndex);\n }\n }\n }\n\n /**\n * Set the attribute pointers in a WebGL context\n * @param gl - The WebGL context\n * @param program - The active WebGL program\n * @param vertexOffset - Index of the starting vertex of the segment\n */\n setVertexAttribPointers(gl: WebGLRenderingContext|WebGL2RenderingContext, program: Program, vertexOffset?: number | null) {\n for (let j = 0; j < this.attributes.length; j++) {\n const member = this.attributes[j];\n const attribIndex: number | void = program.attributes[member.name];\n\n if (attribIndex !== undefined) {\n gl.vertexAttribPointer(\n attribIndex,\n member.components,\n (gl as any)[AttributeType[member.type]],\n false,\n this.itemSize,\n member.offset + (this.itemSize * (vertexOffset || 0))\n );\n }\n }\n }\n\n /**\n * Destroy the GL buffer bound to the given WebGL context\n */\n destroy() {\n const gl = this.context.gl;\n if (this.buffer) {\n gl.deleteBuffer(this.buffer);\n delete this.buffer;\n }\n }\n}\n","const cache = new WeakMap();\nexport function isWebGL2(\n gl: WebGLRenderingContext | WebGL2RenderingContext\n): gl is WebGL2RenderingContext {\n if (cache.has(gl)) {\n return cache.get(gl);\n } else {\n const value = gl.getParameter(gl.VERSION)?.startsWith('WebGL 2.0');\n cache.set(gl, value);\n return value;\n }\n}\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {isWebGL2} from './webgl2';\n\nimport type {Context} from './context';\nimport type {\n BlendFuncType,\n BlendEquationType,\n ColorMaskType,\n DepthRangeType,\n DepthMaskType,\n StencilFuncType,\n StencilOpType,\n DepthFuncType,\n TextureUnitType,\n ViewportType,\n CullFaceModeType,\n FrontFaceType,\n} from './types';\n\nexport interface IValue {\n current: T;\n default: T;\n dirty: boolean;\n get(): T;\n setDefault(): void;\n set(value: T): void;\n}\n\nclass BaseValue implements IValue {\n gl: WebGLRenderingContext|WebGL2RenderingContext;\n current: T;\n default: T;\n dirty: boolean;\n\n constructor(context: Context) {\n this.gl = context.gl;\n this.default = this.getDefault();\n this.current = this.default;\n this.dirty = false;\n }\n\n get(): T {\n return this.current;\n }\n set(value: T) { // eslint-disable-line\n // overridden in child classes;\n }\n\n getDefault(): T {\n return this.default; // overriden in child classes\n }\n setDefault() {\n this.set(this.default);\n }\n}\n\nexport class ClearColor extends BaseValue {\n getDefault(): Color {\n return Color.transparent;\n }\n set(v: Color) {\n const c = this.current;\n if (v.r === c.r && v.g === c.g && v.b === c.b && v.a === c.a && !this.dirty) return;\n this.gl.clearColor(v.r, v.g, v.b, v.a);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ClearDepth extends BaseValue {\n getDefault(): number {\n return 1;\n }\n set(v: number) {\n if (v === this.current && !this.dirty) return;\n this.gl.clearDepth(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ClearStencil extends BaseValue {\n getDefault(): number {\n return 0;\n }\n set(v: number) {\n if (v === this.current && !this.dirty) return;\n this.gl.clearStencil(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ColorMask extends BaseValue {\n getDefault(): ColorMaskType {\n return [true, true, true, true];\n }\n set(v: ColorMaskType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && v[3] === c[3] && !this.dirty) return;\n this.gl.colorMask(v[0], v[1], v[2], v[3]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthMask extends BaseValue {\n getDefault(): DepthMaskType {\n return true;\n }\n set(v: DepthMaskType): void {\n if (v === this.current && !this.dirty) return;\n this.gl.depthMask(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class StencilMask extends BaseValue {\n getDefault(): number {\n return 0xFF;\n }\n set(v: number): void {\n if (v === this.current && !this.dirty) return;\n this.gl.stencilMask(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class StencilFunc extends BaseValue {\n getDefault(): StencilFuncType {\n return {\n func: this.gl.ALWAYS,\n ref: 0,\n mask: 0xFF\n };\n }\n set(v: StencilFuncType): void {\n const c = this.current;\n if (v.func === c.func && v.ref === c.ref && v.mask === c.mask && !this.dirty) return;\n this.gl.stencilFunc(v.func, v.ref, v.mask);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class StencilOp extends BaseValue {\n getDefault(): StencilOpType {\n const gl = this.gl;\n return [gl.KEEP, gl.KEEP, gl.KEEP];\n }\n set(v: StencilOpType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && !this.dirty) return;\n this.gl.stencilOp(v[0], v[1], v[2]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class StencilTest extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n if (v) {\n gl.enable(gl.STENCIL_TEST);\n } else {\n gl.disable(gl.STENCIL_TEST);\n }\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthRange extends BaseValue {\n getDefault(): DepthRangeType {\n return [0, 1];\n }\n set(v: DepthRangeType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && !this.dirty) return;\n this.gl.depthRange(v[0], v[1]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthTest extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n if (v) {\n gl.enable(gl.DEPTH_TEST);\n } else {\n gl.disable(gl.DEPTH_TEST);\n }\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthFunc extends BaseValue {\n getDefault(): DepthFuncType {\n return this.gl.LESS;\n }\n set(v: DepthFuncType) {\n if (v === this.current && !this.dirty) return;\n this.gl.depthFunc(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class Blend extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n if (v) {\n gl.enable(gl.BLEND);\n } else {\n gl.disable(gl.BLEND);\n }\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BlendFunc extends BaseValue {\n getDefault(): BlendFuncType {\n const gl = this.gl;\n return [gl.ONE, gl.ZERO];\n }\n set(v: BlendFuncType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && !this.dirty) return;\n this.gl.blendFunc(v[0], v[1]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BlendColor extends BaseValue {\n getDefault(): Color {\n return Color.transparent;\n }\n set(v: Color) {\n const c = this.current;\n if (v.r === c.r && v.g === c.g && v.b === c.b && v.a === c.a && !this.dirty) return;\n this.gl.blendColor(v.r, v.g, v.b, v.a);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BlendEquation extends BaseValue {\n getDefault(): BlendEquationType {\n return this.gl.FUNC_ADD;\n }\n set(v: BlendEquationType) {\n if (v === this.current && !this.dirty) return;\n this.gl.blendEquation(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class CullFace extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n if (v) {\n gl.enable(gl.CULL_FACE);\n } else {\n gl.disable(gl.CULL_FACE);\n }\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class CullFaceSide extends BaseValue {\n getDefault(): CullFaceModeType {\n return this.gl.BACK;\n }\n set(v: CullFaceModeType) {\n if (v === this.current && !this.dirty) return;\n this.gl.cullFace(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class FrontFace extends BaseValue {\n getDefault(): FrontFaceType {\n return this.gl.CCW;\n }\n set(v: FrontFaceType) {\n if (v === this.current && !this.dirty) return;\n this.gl.frontFace(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ProgramValue extends BaseValue {\n getDefault(): WebGLProgram {\n return null;\n }\n set(v?: WebGLProgram | null) {\n if (v === this.current && !this.dirty) return;\n this.gl.useProgram(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ActiveTextureUnit extends BaseValue {\n getDefault(): TextureUnitType {\n return this.gl.TEXTURE0;\n }\n set(v: TextureUnitType) {\n if (v === this.current && !this.dirty) return;\n this.gl.activeTexture(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class Viewport extends BaseValue {\n getDefault(): ViewportType {\n const gl = this.gl;\n return [0, 0, gl.drawingBufferWidth, gl.drawingBufferHeight];\n }\n set(v: ViewportType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && v[3] === c[3] && !this.dirty) return;\n this.gl.viewport(v[0], v[1], v[2], v[3]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindFramebuffer extends BaseValue {\n getDefault(): WebGLFramebuffer {\n return null;\n }\n set(v?: WebGLFramebuffer | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.bindFramebuffer(gl.FRAMEBUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindRenderbuffer extends BaseValue {\n getDefault(): WebGLRenderbuffer {\n return null;\n }\n set(v?: WebGLRenderbuffer | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.bindRenderbuffer(gl.RENDERBUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindTexture extends BaseValue {\n getDefault(): WebGLTexture {\n return null;\n }\n set(v?: WebGLTexture | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.bindTexture(gl.TEXTURE_2D, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindVertexBuffer extends BaseValue {\n getDefault(): WebGLBuffer {\n return null;\n }\n set(v?: WebGLBuffer | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.bindBuffer(gl.ARRAY_BUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindElementBuffer extends BaseValue {\n getDefault(): WebGLBuffer {\n return null;\n }\n set(v?: WebGLBuffer | null) {\n // Always rebind\n const gl = this.gl;\n gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindVertexArray extends BaseValue {\n getDefault(): WebGLVertexArrayObject | null {\n return null;\n }\n set(v: WebGLVertexArrayObject | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n\n if (isWebGL2(gl)) {\n gl.bindVertexArray(v);\n } else {\n gl.getExtension('OES_vertex_array_object')?.bindVertexArrayOES(v);\n }\n\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class PixelStoreUnpack extends BaseValue {\n getDefault(): number {\n return 4;\n }\n set(v: number) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.pixelStorei(gl.UNPACK_ALIGNMENT, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class PixelStoreUnpackPremultiplyAlpha extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean): void {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, ((v as any)));\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class PixelStoreUnpackFlipY extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean): void {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, ((v as any)));\n this.current = v;\n this.dirty = false;\n }\n}\n\nclass FramebufferAttachment extends BaseValue {\n parent: WebGLFramebuffer;\n context: Context;\n\n constructor(context: Context, parent: WebGLFramebuffer) {\n super(context);\n this.context = context;\n this.parent = parent;\n }\n getDefault() {\n return null;\n }\n}\n\nexport class ColorAttachment extends FramebufferAttachment {\n setDirty() {\n this.dirty = true;\n }\n set(v?: WebGLTexture | null): void {\n if (v === this.current && !this.dirty) return;\n this.context.bindFramebuffer.set(this.parent);\n // note: it's possible to attach a renderbuffer to the color\n // attachment point, but thus far MBGL only uses textures for color\n const gl = this.gl;\n gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, v, 0);\n\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthAttachment extends FramebufferAttachment {\n set(v?: WebGLRenderbuffer | null): void {\n if (v === this.current && !this.dirty) return;\n this.context.bindFramebuffer.set(this.parent);\n // note: it's possible to attach a texture to the depth attachment\n // point, but thus far MBGL only uses renderbuffers for depth\n const gl = this.gl;\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_ATTACHMENT, gl.RENDERBUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthStencilAttachment extends FramebufferAttachment {\n set(v?: WebGLRenderbuffer | null): void {\n if (v === this.current && !this.dirty) return;\n this.context.bindFramebuffer.set(this.parent);\n // note: it's possible to attach a texture to the depth attachment\n // point, but thus far MBGL only uses renderbuffers for depth\n const gl = this.gl;\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n","import {ColorAttachment, DepthAttachment, DepthStencilAttachment} from './value';\n\nimport type {Context} from './context';\n\n/**\n * @internal\n * A framebuffer holder object\n */\nexport class Framebuffer {\n context: Context;\n width: number;\n height: number;\n framebuffer: WebGLFramebuffer;\n colorAttachment: ColorAttachment;\n depthAttachment: DepthAttachment;\n\n constructor(context: Context, width: number, height: number, hasDepth: boolean, hasStencil: boolean) {\n this.context = context;\n this.width = width;\n this.height = height;\n const gl = context.gl;\n const fbo = this.framebuffer = gl.createFramebuffer();\n\n this.colorAttachment = new ColorAttachment(context, fbo);\n if (hasDepth) {\n this.depthAttachment = hasStencil ? new DepthStencilAttachment(context, fbo) : new DepthAttachment(context, fbo);\n } else if (hasStencil) {\n throw new Error('Stencil cannot be setted without depth');\n }\n if (gl.checkFramebufferStatus(gl.FRAMEBUFFER) !== gl.FRAMEBUFFER_COMPLETE) {\n throw new Error('Framebuffer is not complete');\n }\n }\n\n destroy() {\n const gl = this.context.gl;\n\n const texture = this.colorAttachment.get();\n if (texture) gl.deleteTexture(texture);\n\n if (this.depthAttachment) {\n const renderbuffer = this.depthAttachment.get();\n if (renderbuffer) gl.deleteRenderbuffer(renderbuffer);\n }\n\n gl.deleteFramebuffer(this.framebuffer);\n }\n}\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {BlendFuncType, ColorMaskType} from './types';\n\nconst ZERO = 0x0000;\nconst ONE = 0x0001;\nconst ONE_MINUS_SRC_ALPHA = 0x0303;\n\nexport class ColorMode {\n blendFunction: BlendFuncType;\n blendColor: Color;\n mask: ColorMaskType;\n\n constructor(blendFunction: BlendFuncType, blendColor: Color, mask: ColorMaskType) {\n this.blendFunction = blendFunction;\n this.blendColor = blendColor;\n this.mask = mask;\n }\n\n static Replace: BlendFuncType;\n\n static disabled: Readonly;\n static unblended: Readonly;\n static alphaBlended: Readonly;\n}\n\nColorMode.Replace = [ONE, ZERO];\n\nColorMode.disabled = new ColorMode(ColorMode.Replace, Color.transparent, [false, false, false, false]);\nColorMode.unblended = new ColorMode(ColorMode.Replace, Color.transparent, [true, true, true, true]);\nColorMode.alphaBlended = new ColorMode([ONE, ONE_MINUS_SRC_ALPHA], Color.transparent, [true, true, true, true]);\n","import {IndexBuffer} from './index_buffer';\n\nimport {VertexBuffer} from './vertex_buffer';\nimport {Framebuffer} from './framebuffer';\nimport {DepthMode} from './depth_mode';\nimport {StencilMode} from './stencil_mode';\nimport {ColorMode} from './color_mode';\nimport {CullFaceMode} from './cull_face_mode';\nimport {deepEqual} from '../util/util';\nimport {ClearColor, ClearDepth, ClearStencil, ColorMask, DepthMask, StencilMask, StencilFunc, StencilOp, StencilTest, DepthRange, DepthTest, DepthFunc, Blend, BlendFunc, BlendColor, BlendEquation, CullFace, CullFaceSide, FrontFace, ProgramValue, ActiveTextureUnit, Viewport, BindFramebuffer, BindRenderbuffer, BindTexture, BindVertexBuffer, BindElementBuffer, BindVertexArray, PixelStoreUnpack, PixelStoreUnpackPremultiplyAlpha, PixelStoreUnpackFlipY} from './value';\n\nimport type {TriangleIndexArray, LineIndexArray, LineStripIndexArray} from '../data/index_array_type';\nimport type {\n StructArray,\n StructArrayMember\n} from '../util/struct_array';\nimport type {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {isWebGL2} from './webgl2';\n\ntype ClearArgs = {\n color?: Color;\n depth?: number;\n stencil?: number;\n};\n\n/**\n * @internal\n * A webgl wrapper class to allow injection, mocking and abstaction\n */\nexport class Context {\n gl: WebGLRenderingContext | WebGL2RenderingContext;\n\n currentNumAttributes: number;\n maxTextureSize: number;\n\n clearColor: ClearColor;\n clearDepth: ClearDepth;\n clearStencil: ClearStencil;\n colorMask: ColorMask;\n depthMask: DepthMask;\n stencilMask: StencilMask;\n stencilFunc: StencilFunc;\n stencilOp: StencilOp;\n stencilTest: StencilTest;\n depthRange: DepthRange;\n depthTest: DepthTest;\n depthFunc: DepthFunc;\n blend: Blend;\n blendFunc: BlendFunc;\n blendColor: BlendColor;\n blendEquation: BlendEquation;\n cullFace: CullFace;\n cullFaceSide: CullFaceSide;\n frontFace: FrontFace;\n program: ProgramValue;\n activeTexture: ActiveTextureUnit;\n viewport: Viewport;\n bindFramebuffer: BindFramebuffer;\n bindRenderbuffer: BindRenderbuffer;\n bindTexture: BindTexture;\n bindVertexBuffer: BindVertexBuffer;\n bindElementBuffer: BindElementBuffer;\n bindVertexArray: BindVertexArray;\n pixelStoreUnpack: PixelStoreUnpack;\n pixelStoreUnpackPremultiplyAlpha: PixelStoreUnpackPremultiplyAlpha;\n pixelStoreUnpackFlipY: PixelStoreUnpackFlipY;\n\n // eslint-disable-next-line camelcase\n extTextureFilterAnisotropic: EXT_texture_filter_anisotropic | null;\n extTextureFilterAnisotropicMax?: GLfloat;\n HALF_FLOAT?: GLenum;\n RGBA16F?: GLenum;\n RGB16F?: GLenum;\n\n constructor(gl: WebGLRenderingContext | WebGL2RenderingContext) {\n this.gl = gl;\n this.clearColor = new ClearColor(this);\n this.clearDepth = new ClearDepth(this);\n this.clearStencil = new ClearStencil(this);\n this.colorMask = new ColorMask(this);\n this.depthMask = new DepthMask(this);\n this.stencilMask = new StencilMask(this);\n this.stencilFunc = new StencilFunc(this);\n this.stencilOp = new StencilOp(this);\n this.stencilTest = new StencilTest(this);\n this.depthRange = new DepthRange(this);\n this.depthTest = new DepthTest(this);\n this.depthFunc = new DepthFunc(this);\n this.blend = new Blend(this);\n this.blendFunc = new BlendFunc(this);\n this.blendColor = new BlendColor(this);\n this.blendEquation = new BlendEquation(this);\n this.cullFace = new CullFace(this);\n this.cullFaceSide = new CullFaceSide(this);\n this.frontFace = new FrontFace(this);\n this.program = new ProgramValue(this);\n this.activeTexture = new ActiveTextureUnit(this);\n this.viewport = new Viewport(this);\n this.bindFramebuffer = new BindFramebuffer(this);\n this.bindRenderbuffer = new BindRenderbuffer(this);\n this.bindTexture = new BindTexture(this);\n this.bindVertexBuffer = new BindVertexBuffer(this);\n this.bindElementBuffer = new BindElementBuffer(this);\n this.bindVertexArray = new BindVertexArray(this);\n this.pixelStoreUnpack = new PixelStoreUnpack(this);\n this.pixelStoreUnpackPremultiplyAlpha = new PixelStoreUnpackPremultiplyAlpha(this);\n this.pixelStoreUnpackFlipY = new PixelStoreUnpackFlipY(this);\n\n this.extTextureFilterAnisotropic = (\n gl.getExtension('EXT_texture_filter_anisotropic') ||\n gl.getExtension('MOZ_EXT_texture_filter_anisotropic') ||\n gl.getExtension('WEBKIT_EXT_texture_filter_anisotropic')\n );\n\n if (this.extTextureFilterAnisotropic) {\n this.extTextureFilterAnisotropicMax = gl.getParameter(this.extTextureFilterAnisotropic.MAX_TEXTURE_MAX_ANISOTROPY_EXT);\n }\n\n this.maxTextureSize = gl.getParameter(gl.MAX_TEXTURE_SIZE);\n\n if (isWebGL2(gl)) {\n this.HALF_FLOAT = gl.HALF_FLOAT;\n const extColorBufferHalfFloat = gl.getExtension('EXT_color_buffer_half_float');\n this.RGBA16F = gl.RGBA16F ?? extColorBufferHalfFloat?.RGBA16F_EXT;\n this.RGB16F = gl.RGB16F ?? extColorBufferHalfFloat?.RGB16F_EXT;\n gl.getExtension('EXT_color_buffer_float');\n } else {\n gl.getExtension('EXT_color_buffer_half_float');\n gl.getExtension('OES_texture_half_float_linear');\n const extTextureHalfFloat = gl.getExtension('OES_texture_half_float');\n this.HALF_FLOAT = extTextureHalfFloat?.HALF_FLOAT_OES;\n }\n }\n\n setDefault() {\n this.unbindVAO();\n\n this.clearColor.setDefault();\n this.clearDepth.setDefault();\n this.clearStencil.setDefault();\n this.colorMask.setDefault();\n this.depthMask.setDefault();\n this.stencilMask.setDefault();\n this.stencilFunc.setDefault();\n this.stencilOp.setDefault();\n this.stencilTest.setDefault();\n this.depthRange.setDefault();\n this.depthTest.setDefault();\n this.depthFunc.setDefault();\n this.blend.setDefault();\n this.blendFunc.setDefault();\n this.blendColor.setDefault();\n this.blendEquation.setDefault();\n this.cullFace.setDefault();\n this.cullFaceSide.setDefault();\n this.frontFace.setDefault();\n this.program.setDefault();\n this.activeTexture.setDefault();\n this.bindFramebuffer.setDefault();\n this.pixelStoreUnpack.setDefault();\n this.pixelStoreUnpackPremultiplyAlpha.setDefault();\n this.pixelStoreUnpackFlipY.setDefault();\n }\n\n setDirty() {\n this.clearColor.dirty = true;\n this.clearDepth.dirty = true;\n this.clearStencil.dirty = true;\n this.colorMask.dirty = true;\n this.depthMask.dirty = true;\n this.stencilMask.dirty = true;\n this.stencilFunc.dirty = true;\n this.stencilOp.dirty = true;\n this.stencilTest.dirty = true;\n this.depthRange.dirty = true;\n this.depthTest.dirty = true;\n this.depthFunc.dirty = true;\n this.blend.dirty = true;\n this.blendFunc.dirty = true;\n this.blendColor.dirty = true;\n this.blendEquation.dirty = true;\n this.cullFace.dirty = true;\n this.cullFaceSide.dirty = true;\n this.frontFace.dirty = true;\n this.program.dirty = true;\n this.activeTexture.dirty = true;\n this.viewport.dirty = true;\n this.bindFramebuffer.dirty = true;\n this.bindRenderbuffer.dirty = true;\n this.bindTexture.dirty = true;\n this.bindVertexBuffer.dirty = true;\n this.bindElementBuffer.dirty = true;\n this.bindVertexArray.dirty = true;\n this.pixelStoreUnpack.dirty = true;\n this.pixelStoreUnpackPremultiplyAlpha.dirty = true;\n this.pixelStoreUnpackFlipY.dirty = true;\n }\n\n createIndexBuffer(array: TriangleIndexArray | LineIndexArray | LineStripIndexArray, dynamicDraw?: boolean) {\n return new IndexBuffer(this, array, dynamicDraw);\n }\n\n createVertexBuffer(array: StructArray, attributes: ReadonlyArray, dynamicDraw?: boolean) {\n return new VertexBuffer(this, array, attributes, dynamicDraw);\n }\n\n createRenderbuffer(storageFormat: number, width: number, height: number) {\n const gl = this.gl;\n\n const rbo = gl.createRenderbuffer();\n this.bindRenderbuffer.set(rbo);\n gl.renderbufferStorage(gl.RENDERBUFFER, storageFormat, width, height);\n this.bindRenderbuffer.set(null);\n\n return rbo;\n }\n\n createFramebuffer(width: number, height: number, hasDepth: boolean, hasStencil: boolean) {\n return new Framebuffer(this, width, height, hasDepth, hasStencil);\n }\n\n clear({\n color,\n depth,\n stencil\n }: ClearArgs) {\n const gl = this.gl;\n let mask = 0;\n\n if (color) {\n mask |= gl.COLOR_BUFFER_BIT;\n this.clearColor.set(color);\n this.colorMask.set([true, true, true, true]);\n }\n\n if (typeof depth !== 'undefined') {\n mask |= gl.DEPTH_BUFFER_BIT;\n\n // Workaround for platforms where clearDepth doesn't seem to work\n // without resetting the depthRange. See https://github.com/mapbox/mapbox-gl-js/issues/3437\n this.depthRange.set([0, 1]);\n\n this.clearDepth.set(depth);\n this.depthMask.set(true);\n }\n\n if (typeof stencil !== 'undefined') {\n mask |= gl.STENCIL_BUFFER_BIT;\n this.clearStencil.set(stencil);\n this.stencilMask.set(0xFF);\n }\n\n gl.clear(mask);\n }\n\n setCullFace(cullFaceMode: Readonly) {\n if (cullFaceMode.enable === false) {\n this.cullFace.set(false);\n } else {\n this.cullFace.set(true);\n this.cullFaceSide.set(cullFaceMode.mode);\n this.frontFace.set(cullFaceMode.frontFace);\n }\n }\n\n setDepthMode(depthMode: Readonly) {\n if (depthMode.func === this.gl.ALWAYS && !depthMode.mask) {\n this.depthTest.set(false);\n } else {\n this.depthTest.set(true);\n this.depthFunc.set(depthMode.func);\n this.depthMask.set(depthMode.mask);\n this.depthRange.set(depthMode.range);\n }\n }\n\n setStencilMode(stencilMode: Readonly) {\n if (stencilMode.test.func === this.gl.ALWAYS && !stencilMode.mask) {\n this.stencilTest.set(false);\n } else {\n this.stencilTest.set(true);\n this.stencilMask.set(stencilMode.mask);\n this.stencilOp.set([stencilMode.fail, stencilMode.depthFail, stencilMode.pass]);\n this.stencilFunc.set({\n func: stencilMode.test.func,\n ref: stencilMode.ref,\n mask: stencilMode.test.mask\n });\n }\n }\n\n setColorMode(colorMode: Readonly) {\n if (deepEqual(colorMode.blendFunction, ColorMode.Replace)) {\n this.blend.set(false);\n } else {\n this.blend.set(true);\n this.blendFunc.set(colorMode.blendFunction);\n this.blendColor.set(colorMode.blendColor);\n }\n\n this.colorMask.set(colorMode.mask);\n }\n\n createVertexArray(): WebGLVertexArrayObject | undefined {\n if (isWebGL2(this.gl))\n return this.gl.createVertexArray();\n return this.gl.getExtension('OES_vertex_array_object')?.createVertexArrayOES();\n }\n\n deleteVertexArray(x: WebGLVertexArrayObject | undefined) {\n if (isWebGL2(this.gl))\n return this.gl.deleteVertexArray(x);\n return this.gl.getExtension('OES_vertex_array_object')?.deleteVertexArrayOES(x);\n }\n\n unbindVAO() {\n // Unbinding the VAO prevents other things (custom layers, new buffer creation) from\n // unintentionally changing the state of the last VAO used.\n this.bindVertexArray.set(null);\n }\n}\n","import type {DepthFuncType, DepthMaskType, DepthRangeType} from './types';\n\nconst ALWAYS = 0x0207;\n\nexport class DepthMode {\n func: DepthFuncType;\n mask: DepthMaskType;\n range: DepthRangeType;\n\n // DepthMask enums\n static ReadOnly: boolean;\n static ReadWrite: boolean;\n\n constructor(depthFunc: DepthFuncType, depthMask: DepthMaskType, depthRange: DepthRangeType) {\n this.func = depthFunc;\n this.mask = depthMask;\n this.range = depthRange;\n }\n\n static disabled: Readonly;\n}\n\nDepthMode.ReadOnly = false;\nDepthMode.ReadWrite = true;\n\nDepthMode.disabled = new DepthMode(ALWAYS, DepthMode.ReadOnly, [0, 1]);\n","import type {StencilOpConstant, StencilTestGL} from './types';\n\nconst ALWAYS = 0x0207;\nconst KEEP = 0x1E00;\n\nexport class StencilMode {\n test: StencilTestGL;\n ref: number;\n mask: number;\n fail: StencilOpConstant;\n depthFail: StencilOpConstant;\n pass: StencilOpConstant;\n\n constructor(test: StencilTestGL, ref: number, mask: number, fail: StencilOpConstant,\n depthFail: StencilOpConstant, pass: StencilOpConstant) {\n this.test = test;\n this.ref = ref;\n this.mask = mask;\n this.fail = fail;\n this.depthFail = depthFail;\n this.pass = pass;\n }\n\n static disabled: Readonly;\n}\n\nStencilMode.disabled = new StencilMode({func: ALWAYS, mask: 0}, 0, 0, KEEP, KEEP, KEEP);\n","import type {CullFaceModeType, FrontFaceType} from './types';\n\nconst BACK = 0x0405;\nconst CCW = 0x0901;\n\nexport class CullFaceMode {\n enable: boolean;\n mode: CullFaceModeType;\n frontFace: FrontFaceType;\n\n constructor(enable: boolean, mode: CullFaceModeType, frontFace: FrontFaceType) {\n this.enable = enable;\n this.mode = mode;\n this.frontFace = frontFace;\n }\n\n static disabled: Readonly;\n static backCCW: Readonly;\n}\n\nCullFaceMode.disabled = new CullFaceMode(false, BACK, CCW);\nCullFaceMode.backCCW = new CullFaceMode(true, BACK, CCW);\n","import type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {collisionUniformValues, collisionCircleUniformValues} from './program/collision_program';\n\nimport {QuadTriangleArray, CollisionCircleLayoutArray} from '../data/array_types.g';\nimport {collisionCircleLayout} from '../data/bucket/symbol_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {mat4} from 'gl-matrix';\nimport {VertexBuffer} from '../gl/vertex_buffer';\nimport {IndexBuffer} from '../gl/index_buffer';\n\ntype TileBatch = {\n circleArray: Array;\n circleOffset: number;\n transform: mat4;\n invTransform: mat4;\n coord: OverscaledTileID;\n};\n\nlet quadTriangles: QuadTriangleArray;\n\nexport function drawCollisionDebug(painter: Painter, sourceCache: SourceCache, layer: StyleLayer, coords: Array, translate: [number, number], translateAnchor: 'map' | 'viewport', isText: boolean) {\n const context = painter.context;\n const gl = context.gl;\n const program = painter.useProgram('collisionBox');\n const tileBatches: Array = [];\n let circleCount = 0;\n let circleOffset = 0;\n\n for (let i = 0; i < coords.length; i++) {\n const coord = coords[i];\n const tile = sourceCache.getTile(coord);\n const bucket: SymbolBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n let posMatrix = coord.posMatrix;\n if (translate[0] !== 0 || translate[1] !== 0) {\n posMatrix = painter.translatePosMatrix(coord.posMatrix, tile, translate, translateAnchor);\n }\n const buffers = isText ? bucket.textCollisionBox : bucket.iconCollisionBox;\n // Get collision circle data of this bucket\n const circleArray: Array = bucket.collisionCircleArray;\n if (circleArray.length > 0) {\n // We need to know the projection matrix that was used for projecting collision circles to the screen.\n // This might vary between buckets as the symbol placement is a continuous process. This matrix is\n // required for transforming points from previous screen space to the current one\n const invTransform = mat4.create();\n const transform = posMatrix;\n\n mat4.mul(invTransform, bucket.placementInvProjMatrix, painter.transform.glCoordMatrix);\n mat4.mul(invTransform, invTransform, bucket.placementViewportMatrix);\n\n tileBatches.push({\n circleArray,\n circleOffset,\n transform,\n invTransform,\n coord\n });\n\n circleCount += circleArray.length / 4; // 4 values per circle\n circleOffset = circleCount;\n }\n if (!buffers) continue;\n program.draw(context, gl.LINES,\n DepthMode.disabled, StencilMode.disabled,\n painter.colorModeForRenderPass(),\n CullFaceMode.disabled,\n collisionUniformValues(\n posMatrix,\n painter.transform,\n tile),\n painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord),\n layer.id, buffers.layoutVertexBuffer, buffers.indexBuffer,\n buffers.segments, null, painter.transform.zoom, null, null,\n buffers.collisionVertexBuffer);\n }\n\n if (!isText || !tileBatches.length) {\n return;\n }\n\n // Render collision circles\n const circleProgram = painter.useProgram('collisionCircle');\n\n // Construct vertex data\n const vertexData = new CollisionCircleLayoutArray();\n vertexData.resize(circleCount * 4);\n vertexData._trim();\n\n let vertexOffset = 0;\n\n for (const batch of tileBatches) {\n for (let i = 0; i < batch.circleArray.length / 4; i++) {\n const circleIdx = i * 4;\n const x = batch.circleArray[circleIdx + 0];\n const y = batch.circleArray[circleIdx + 1];\n const radius = batch.circleArray[circleIdx + 2];\n const collision = batch.circleArray[circleIdx + 3];\n\n // 4 floats per vertex, 4 vertices per quad\n vertexData.emplace(vertexOffset++, x, y, radius, collision, 0);\n vertexData.emplace(vertexOffset++, x, y, radius, collision, 1);\n vertexData.emplace(vertexOffset++, x, y, radius, collision, 2);\n vertexData.emplace(vertexOffset++, x, y, radius, collision, 3);\n }\n }\n if (!quadTriangles || quadTriangles.length < circleCount * 2) {\n quadTriangles = createQuadTriangles(circleCount);\n }\n\n const indexBuffer: IndexBuffer = context.createIndexBuffer(quadTriangles, true);\n const vertexBuffer: VertexBuffer = context.createVertexBuffer(vertexData, collisionCircleLayout.members, true);\n\n // Render batches\n for (const batch of tileBatches) {\n const uniforms = collisionCircleUniformValues(\n batch.transform,\n batch.invTransform,\n painter.transform\n );\n\n circleProgram.draw(\n context,\n gl.TRIANGLES,\n DepthMode.disabled,\n StencilMode.disabled,\n painter.colorModeForRenderPass(),\n CullFaceMode.disabled,\n uniforms,\n painter.style.map.terrain && painter.style.map.terrain.getTerrainData(batch.coord),\n layer.id,\n vertexBuffer,\n indexBuffer,\n SegmentVector.simpleSegment(0, batch.circleOffset * 2, batch.circleArray.length, batch.circleArray.length / 2),\n null,\n painter.transform.zoom,\n null,\n null,\n null);\n }\n\n vertexBuffer.destroy();\n indexBuffer.destroy();\n}\n\nfunction createQuadTriangles(quadCount: number): QuadTriangleArray {\n const triCount = quadCount * 2;\n const array = new QuadTriangleArray();\n\n array.resize(triCount);\n array._trim();\n\n // Two triangles and 4 vertices per quad.\n for (let i = 0; i < triCount; i++) {\n const idx = i * 6;\n\n array.uint16[idx + 0] = i * 4 + 0;\n array.uint16[idx + 1] = i * 4 + 1;\n array.uint16[idx + 2] = i * 4 + 2;\n array.uint16[idx + 3] = i * 4 + 2;\n array.uint16[idx + 4] = i * 4 + 3;\n array.uint16[idx + 5] = i * 4 + 0;\n }\n\n return array;\n}\n","import Point from '@mapbox/point-geometry';\nimport {drawCollisionDebug} from './draw_collision_debug';\n\nimport {SegmentVector} from '../data/segment';\nimport {pixelsToTileUnits} from '../source/pixels_to_tile_units';\nimport * as symbolProjection from '../symbol/projection';\nimport {EvaluatedZoomSize, evaluateSizeForFeature, evaluateSizeForZoom} from '../symbol/symbol_size';\nimport {mat4} from 'gl-matrix';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {addDynamicAttributes} from '../data/bucket/symbol_bucket';\n\nimport {getAnchorAlignment, WritingMode} from '../symbol/shaping';\nimport ONE_EM from '../symbol/one_em';\n\nimport {\n SymbolIconUniformsType,\n symbolIconUniformValues,\n symbolSDFUniformValues,\n symbolTextAndIconUniformValues\n} from './program/symbol_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\n\nimport type {Texture} from '../render/texture';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {UniformValues} from './uniform_binding';\nimport type {SymbolSDFUniformsType} from '../render/program/symbol_program';\nimport type {CrossTileID, VariableOffset} from '../symbol/placement';\nimport type {SymbolBucket, SymbolBuffers} from '../data/bucket/symbol_bucket';\nimport type {TerrainData} from '../render/terrain';\nimport type {SymbolLayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Transform} from '../geo/transform';\nimport type {ColorMode} from '../gl/color_mode';\nimport type {Program} from './program';\nimport type {TextAnchor} from '../style/style_layer/variable_text_anchor';\n\ntype SymbolTileRenderState = {\n segments: SegmentVector;\n sortKey: number;\n terrainData: TerrainData;\n state: {\n program: Program;\n buffers: SymbolBuffers;\n uniformValues: UniformValues;\n atlasTexture: Texture;\n atlasTextureIcon: Texture | null;\n atlasInterpolation: GLenum;\n atlasInterpolationIcon: GLenum;\n isSDF: boolean;\n hasHalo: boolean;\n };\n};\n\nconst identityMat4 = mat4.identity(new Float32Array(16));\n\nexport function drawSymbols(painter: Painter, sourceCache: SourceCache, layer: SymbolStyleLayer, coords: Array, variableOffsets: {\n [_ in CrossTileID]: VariableOffset;\n}) {\n if (painter.renderPass !== 'translucent') return;\n\n // Disable the stencil test so that labels aren't clipped to tile boundaries.\n const stencilMode = StencilMode.disabled;\n const colorMode = painter.colorModeForRenderPass();\n const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset');\n\n //Compute variable-offsets before painting since icons and text data positioning\n //depend on each other in this case.\n if (hasVariablePlacement) {\n updateVariableAnchors(coords, painter, layer, sourceCache,\n layer.layout.get('text-rotation-alignment'),\n layer.layout.get('text-pitch-alignment'),\n variableOffsets\n );\n }\n\n if (layer.paint.get('icon-opacity').constantOr(1) !== 0) {\n drawLayerSymbols(painter, sourceCache, layer, coords, false,\n layer.paint.get('icon-translate'),\n layer.paint.get('icon-translate-anchor'),\n layer.layout.get('icon-rotation-alignment'),\n layer.layout.get('icon-pitch-alignment'),\n layer.layout.get('icon-keep-upright'),\n stencilMode, colorMode\n );\n }\n\n if (layer.paint.get('text-opacity').constantOr(1) !== 0) {\n drawLayerSymbols(painter, sourceCache, layer, coords, true,\n layer.paint.get('text-translate'),\n layer.paint.get('text-translate-anchor'),\n layer.layout.get('text-rotation-alignment'),\n layer.layout.get('text-pitch-alignment'),\n layer.layout.get('text-keep-upright'),\n stencilMode, colorMode\n );\n }\n\n if (sourceCache.map.showCollisionBoxes) {\n drawCollisionDebug(painter, sourceCache, layer, coords, layer.paint.get('text-translate'),\n layer.paint.get('text-translate-anchor'), true);\n drawCollisionDebug(painter, sourceCache, layer, coords, layer.paint.get('icon-translate'),\n layer.paint.get('icon-translate-anchor'), false);\n }\n}\n\nfunction calculateVariableRenderShift(\n anchor: TextAnchor,\n width: number,\n height: number,\n textOffset: [number, number],\n textBoxScale: number,\n renderTextSize: number): Point {\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(anchor);\n const shiftX = -(horizontalAlign - 0.5) * width;\n const shiftY = -(verticalAlign - 0.5) * height;\n return new Point(\n (shiftX / textBoxScale + textOffset[0]) * renderTextSize,\n (shiftY / textBoxScale + textOffset[1]) * renderTextSize\n );\n}\n\nfunction updateVariableAnchors(coords: Array,\n painter: Painter,\n layer:SymbolStyleLayer, sourceCache: SourceCache,\n rotationAlignment: SymbolLayerSpecification['layout']['text-rotation-alignment'],\n pitchAlignment: SymbolLayerSpecification['layout']['text-pitch-alignment'],\n variableOffsets: {[_ in CrossTileID]: VariableOffset}) {\n const tr = painter.transform;\n const rotateWithMap = rotationAlignment === 'map';\n const pitchWithMap = pitchAlignment === 'map';\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n const bucket = tile.getBucket(layer) as SymbolBucket;\n if (!bucket || !bucket.text || !bucket.text.segments.get().length) continue;\n\n const sizeData = bucket.textSizeData;\n const size = evaluateSizeForZoom(sizeData, tr.zoom);\n\n const pixelToTileScale = pixelsToTileUnits(tile, 1, painter.transform.zoom);\n const labelPlaneMatrix = symbolProjection.getLabelPlaneMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, pixelToTileScale);\n const updateTextFitIcon = layer.layout.get('icon-text-fit') !== 'none' && bucket.hasIconData();\n\n if (size) {\n const tileScale = Math.pow(2, tr.zoom - tile.tileID.overscaledZ);\n const getElevation = painter.style.map.terrain ? (x: number, y: number) => painter.style.map.terrain.getElevation(coord, x, y) : null;\n updateVariableAnchorsForBucket(bucket, rotateWithMap, pitchWithMap, variableOffsets,\n tr, labelPlaneMatrix, coord.posMatrix, tileScale, size, updateTextFitIcon, getElevation);\n }\n }\n}\n\nfunction updateVariableAnchorsForBucket(\n bucket: SymbolBucket,\n rotateWithMap: boolean,\n pitchWithMap: boolean,\n variableOffsets: {[_ in CrossTileID]: VariableOffset},\n transform: Transform,\n labelPlaneMatrix: mat4,\n posMatrix: mat4,\n tileScale: number,\n size: EvaluatedZoomSize,\n updateTextFitIcon: boolean,\n getElevation: (x: number, y: number) => number) {\n const placedSymbols = bucket.text.placedSymbolArray;\n const dynamicTextLayoutVertexArray = bucket.text.dynamicLayoutVertexArray;\n const dynamicIconLayoutVertexArray = bucket.icon.dynamicLayoutVertexArray;\n const placedTextShifts = {};\n\n dynamicTextLayoutVertexArray.clear();\n for (let s = 0; s < placedSymbols.length; s++) {\n const symbol = placedSymbols.get(s);\n const skipOrientation = bucket.allowVerticalPlacement && !symbol.placedOrientation;\n const variableOffset = (!symbol.hidden && symbol.crossTileID && !skipOrientation) ? variableOffsets[symbol.crossTileID] : null;\n\n if (!variableOffset) {\n // These symbols are from a justification that is not being used, or a label that wasn't placed\n // so we don't need to do the extra math to figure out what incremental shift to apply.\n symbolProjection.hideGlyphs(symbol.numGlyphs, dynamicTextLayoutVertexArray);\n } else {\n const tileAnchor = new Point(symbol.anchorX, symbol.anchorY);\n const projectedAnchor = symbolProjection.project(tileAnchor, pitchWithMap ? posMatrix : labelPlaneMatrix, getElevation);\n const perspectiveRatio = symbolProjection.getPerspectiveRatio(transform.cameraToCenterDistance, projectedAnchor.signedDistanceFromCamera);\n let renderTextSize = evaluateSizeForFeature(bucket.textSizeData, size, symbol) * perspectiveRatio / ONE_EM;\n if (pitchWithMap) {\n // Go from size in pixels to equivalent size in tile units\n renderTextSize *= bucket.tilePixelRatio / tileScale;\n }\n\n const {width, height, anchor, textOffset, textBoxScale} = variableOffset;\n\n const shift = calculateVariableRenderShift(\n anchor, width, height, textOffset, textBoxScale, renderTextSize);\n\n // Usual case is that we take the projected anchor and add the pixel-based shift\n // calculated above. In the (somewhat weird) case of pitch-aligned text, we add an equivalent\n // tile-unit based shift to the anchor before projecting to the label plane.\n const shiftedAnchor = pitchWithMap ?\n symbolProjection.project(tileAnchor.add(shift), labelPlaneMatrix, getElevation).point :\n projectedAnchor.point.add(rotateWithMap ?\n shift.rotate(-transform.angle) :\n shift);\n\n const angle = (bucket.allowVerticalPlacement && symbol.placedOrientation === WritingMode.vertical) ? Math.PI / 2 : 0;\n for (let g = 0; g < symbol.numGlyphs; g++) {\n addDynamicAttributes(dynamicTextLayoutVertexArray, shiftedAnchor, angle);\n }\n //Only offset horizontal text icons\n if (updateTextFitIcon && symbol.associatedIconIndex >= 0) {\n placedTextShifts[symbol.associatedIconIndex] = {shiftedAnchor, angle};\n }\n }\n }\n\n if (updateTextFitIcon) {\n dynamicIconLayoutVertexArray.clear();\n const placedIcons = bucket.icon.placedSymbolArray;\n for (let i = 0; i < placedIcons.length; i++) {\n const placedIcon = placedIcons.get(i);\n if (placedIcon.hidden) {\n symbolProjection.hideGlyphs(placedIcon.numGlyphs, dynamicIconLayoutVertexArray);\n } else {\n const shift = placedTextShifts[i];\n if (!shift) {\n symbolProjection.hideGlyphs(placedIcon.numGlyphs, dynamicIconLayoutVertexArray);\n } else {\n for (let g = 0; g < placedIcon.numGlyphs; g++) {\n addDynamicAttributes(dynamicIconLayoutVertexArray, shift.shiftedAnchor, shift.angle);\n }\n }\n }\n }\n bucket.icon.dynamicLayoutVertexBuffer.updateData(dynamicIconLayoutVertexArray);\n }\n bucket.text.dynamicLayoutVertexBuffer.updateData(dynamicTextLayoutVertexArray);\n}\n\nfunction getSymbolProgramName(isSDF: boolean, isText: boolean, bucket: SymbolBucket) {\n if (bucket.iconsInText && isText) {\n return 'symbolTextAndIcon';\n } else if (isSDF) {\n return 'symbolSDF';\n } else {\n return 'symbolIcon';\n }\n}\n\nfunction drawLayerSymbols(\n painter: Painter,\n sourceCache: SourceCache,\n layer: SymbolStyleLayer,\n coords: Array,\n isText: boolean,\n translate: [number, number],\n translateAnchor: 'map' | 'viewport',\n rotationAlignment: SymbolLayerSpecification['layout']['text-rotation-alignment'],\n pitchAlignment: SymbolLayerSpecification['layout']['text-pitch-alignment'],\n keepUpright: boolean,\n stencilMode: StencilMode,\n colorMode: Readonly) {\n\n const context = painter.context;\n const gl = context.gl;\n const tr = painter.transform;\n\n const rotateWithMap = rotationAlignment === 'map';\n const pitchWithMap = pitchAlignment === 'map';\n const alongLine = rotationAlignment !== 'viewport' && layer.layout.get('symbol-placement') !== 'point';\n // Line label rotation happens in `updateLineLabels`\n // Pitched point labels are automatically rotated by the labelPlaneMatrix projection\n // Unpitched point labels need to have their rotation applied after projection\n const rotateInShader = rotateWithMap && !pitchWithMap && !alongLine;\n\n const hasSortKey = !layer.layout.get('symbol-sort-key').isConstant();\n let sortFeaturesByKey = false;\n\n const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n\n const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset');\n\n const tileRenderState: Array = [];\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n const bucket = tile.getBucket(layer) as SymbolBucket;\n if (!bucket) continue;\n const buffers = isText ? bucket.text : bucket.icon;\n\n if (!buffers || !buffers.segments.get().length || !buffers.hasVisibleVertices) continue;\n const programConfiguration = buffers.programConfigurations.get(layer.id);\n\n const isSDF = isText || bucket.sdfIcons;\n\n const sizeData = isText ? bucket.textSizeData : bucket.iconSizeData;\n const transformed = pitchWithMap || tr.pitch !== 0;\n\n const program = painter.useProgram(getSymbolProgramName(isSDF, isText, bucket), programConfiguration);\n const size = evaluateSizeForZoom(sizeData, tr.zoom);\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n let texSize: [number, number];\n let texSizeIcon: [number, number] = [0, 0];\n let atlasTexture: Texture;\n let atlasInterpolation: GLenum;\n let atlasTextureIcon = null;\n let atlasInterpolationIcon: GLenum;\n if (isText) {\n atlasTexture = tile.glyphAtlasTexture;\n atlasInterpolation = gl.LINEAR;\n texSize = tile.glyphAtlasTexture.size;\n if (bucket.iconsInText) {\n texSizeIcon = tile.imageAtlasTexture.size;\n atlasTextureIcon = tile.imageAtlasTexture;\n const zoomDependentSize = sizeData.kind === 'composite' || sizeData.kind === 'camera';\n atlasInterpolationIcon = transformed || painter.options.rotating || painter.options.zooming || zoomDependentSize ? gl.LINEAR : gl.NEAREST;\n }\n } else {\n const iconScaled = layer.layout.get('icon-size').constantOr(0) !== 1 || bucket.iconsNeedLinear;\n atlasTexture = tile.imageAtlasTexture;\n atlasInterpolation = isSDF || painter.options.rotating || painter.options.zooming || iconScaled || transformed ?\n gl.LINEAR :\n gl.NEAREST;\n texSize = tile.imageAtlasTexture.size;\n }\n\n const s = pixelsToTileUnits(tile, 1, painter.transform.zoom);\n const labelPlaneMatrix = symbolProjection.getLabelPlaneMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, s);\n const glCoordMatrix = symbolProjection.getGlCoordMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, s);\n\n const hasVariableAnchors = hasVariablePlacement && bucket.hasTextData();\n const updateTextFitIcon = layer.layout.get('icon-text-fit') !== 'none' &&\n hasVariableAnchors &&\n bucket.hasIconData();\n\n if (alongLine) {\n const getElevation = painter.style.map.terrain ? (x: number, y: number) => painter.style.map.terrain.getElevation(coord, x, y) : null;\n const rotateToLine = layer.layout.get('text-rotation-alignment') === 'map';\n symbolProjection.updateLineLabels(bucket, coord.posMatrix, painter, isText, labelPlaneMatrix, glCoordMatrix, pitchWithMap, keepUpright, rotateToLine, getElevation);\n }\n\n const matrix = painter.translatePosMatrix(coord.posMatrix, tile, translate, translateAnchor),\n uLabelPlaneMatrix = (alongLine || (isText && hasVariablePlacement) || updateTextFitIcon) ? identityMat4 : labelPlaneMatrix,\n uglCoordMatrix = painter.translatePosMatrix(glCoordMatrix, tile, translate, translateAnchor, true);\n\n const hasHalo = isSDF && layer.paint.get(isText ? 'text-halo-width' : 'icon-halo-width').constantOr(1) !== 0;\n\n let uniformValues: UniformValues;\n if (isSDF) {\n if (!bucket.iconsInText) {\n uniformValues = symbolSDFUniformValues(sizeData.kind,\n size, rotateInShader, pitchWithMap, painter, matrix,\n uLabelPlaneMatrix, uglCoordMatrix, isText, texSize, true);\n } else {\n uniformValues = symbolTextAndIconUniformValues(sizeData.kind,\n size, rotateInShader, pitchWithMap, painter, matrix,\n uLabelPlaneMatrix, uglCoordMatrix, texSize, texSizeIcon);\n }\n } else {\n uniformValues = symbolIconUniformValues(sizeData.kind,\n size, rotateInShader, pitchWithMap, painter, matrix,\n uLabelPlaneMatrix, uglCoordMatrix, isText, texSize);\n }\n\n const state = {\n program,\n buffers,\n uniformValues,\n atlasTexture,\n atlasTextureIcon,\n atlasInterpolation,\n atlasInterpolationIcon,\n isSDF,\n hasHalo\n };\n\n if (hasSortKey && bucket.canOverlap) {\n sortFeaturesByKey = true;\n const oldSegments = buffers.segments.get();\n for (const segment of oldSegments) {\n tileRenderState.push({\n segments: new SegmentVector([segment]),\n sortKey: segment.sortKey,\n state,\n terrainData\n });\n }\n } else {\n tileRenderState.push({\n segments: buffers.segments,\n sortKey: 0,\n state,\n terrainData\n });\n }\n }\n\n if (sortFeaturesByKey) {\n tileRenderState.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const segmentState of tileRenderState) {\n const state = segmentState.state;\n\n context.activeTexture.set(gl.TEXTURE0);\n // @ts-ignore\n state.atlasTexture.bind(state.atlasInterpolation, gl.CLAMP_TO_EDGE);\n if (state.atlasTextureIcon) {\n context.activeTexture.set(gl.TEXTURE1);\n if (state.atlasTextureIcon) {\n // @ts-ignore\n state.atlasTextureIcon.bind(state.atlasInterpolationIcon, gl.CLAMP_TO_EDGE);\n }\n }\n\n if (state.isSDF) {\n const uniformValues = state.uniformValues;\n if (state.hasHalo) {\n uniformValues['u_is_halo'] = 1;\n drawSymbolElements(state.buffers, segmentState.segments, layer, painter, state.program, depthMode, stencilMode, colorMode, uniformValues, segmentState.terrainData);\n }\n uniformValues['u_is_halo'] = 0;\n }\n drawSymbolElements(state.buffers, segmentState.segments, layer, painter, state.program, depthMode, stencilMode, colorMode, state.uniformValues, segmentState.terrainData);\n }\n}\n\nfunction drawSymbolElements(\n buffers: SymbolBuffers,\n segments: SegmentVector,\n layer: SymbolStyleLayer,\n painter: Painter,\n program: Program,\n depthMode: Readonly,\n stencilMode: StencilMode,\n colorMode: Readonly,\n uniformValues: UniformValues,\n terrainData: TerrainData) {\n const context = painter.context;\n const gl = context.gl;\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id, buffers.layoutVertexBuffer,\n buffers.indexBuffer, segments, layer.paint,\n painter.transform.zoom, buffers.programConfigurations.get(layer.id),\n buffers.dynamicLayoutVertexBuffer, buffers.opacityVertexBuffer);\n}\n","import type {CrossFaded} from '../style/properties';\nimport type {ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport type {Tile} from '../source/tile';\nimport type {ProgramConfiguration} from '../data/program_configuration';\nimport type {FillExtrusionStyleLayer} from '../style/style_layer/fill_extrusion_style_layer';\nimport type {FillStyleLayer} from '../style/style_layer/fill_style_layer';\n\n/**\n * A simple helper shared by draw_fill and draw_fill_extrusions to find the correct pattern positions AND update program.\n * For transtionable properties, especially 'fill-pattern' and 'fill-extrusion-pattern', while rendering certain frames\n * tile.imageAtlas has been updated by worker to hold the new pattern only, but rendering code is still looking for the previous image.\n * The mismatch was causing setConstantPatternPositions method not being called and pixelRatio was always the\n * default of 1, instead of actual values set by original map.addImage.\n *\n * @param programConfiguration - to be used to set pattern position and device pixel ratio.\n * @param propertyName - 'fill-pattern' or 'fill-extrusion-pattern' property key\n * @param constantPattern - either 'fill-pattern' or 'fill-extrusion-pattern' property value\n * @param tile - current tile being drawn\n * @param layer - current layer being rendered\n */\nexport function updatePatternPositionsInProgram(\n programConfiguration: ProgramConfiguration,\n propertyName: 'fill-pattern' | 'fill-extrusion-pattern',\n constantPattern: CrossFaded,\n tile: Tile,\n layer: FillStyleLayer | FillExtrusionStyleLayer): void {\n\n if (!constantPattern || !tile || !tile.imageAtlas) {\n return;\n }\n\n const patternPositions = tile.imageAtlas.patternPositions;\n let posTo = patternPositions[constantPattern.to.toString()];\n let posFrom = patternPositions[constantPattern.from.toString()];\n\n // https://github.com/maplibre/maplibre-gl-js/issues/3377\n if (!posTo && posFrom) posTo = posFrom;\n if (!posFrom && posTo) posFrom = posTo;\n\n // try again in case patternPositions has been updated by worker\n if (!posTo || !posFrom) {\n const transitioned = layer.getPaintProperty(propertyName) as string;\n posTo = patternPositions[transitioned];\n posFrom = patternPositions[transitioned];\n }\n\n if (posTo && posFrom) {\n programConfiguration.setConstantPatternPositions(posTo, posFrom);\n }\n}\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {\n fillUniformValues,\n fillPatternUniformValues,\n fillOutlineUniformValues,\n fillOutlinePatternUniformValues\n} from './program/fill_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {FillStyleLayer} from '../style/style_layer/fill_style_layer';\nimport type {FillBucket} from '../data/bucket/fill_bucket';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport {updatePatternPositionsInProgram} from './update_pattern_positions_in_program';\n\nexport function drawFill(painter: Painter, sourceCache: SourceCache, layer: FillStyleLayer, coords: Array) {\n const color = layer.paint.get('fill-color');\n const opacity = layer.paint.get('fill-opacity');\n\n if (opacity.constantOr(1) === 0) {\n return;\n }\n\n const colorMode = painter.colorModeForRenderPass();\n\n const pattern = layer.paint.get('fill-pattern');\n const pass = painter.opaquePassEnabledForLayer() &&\n (!pattern.constantOr(1 as any) &&\n color.constantOr(Color.transparent).a === 1 &&\n opacity.constantOr(0) === 1) ? 'opaque' : 'translucent';\n\n // Draw fill\n if (painter.renderPass === pass) {\n const depthMode = painter.depthModeForSublayer(\n 1, painter.renderPass === 'opaque' ? DepthMode.ReadWrite : DepthMode.ReadOnly);\n drawFillTiles(painter, sourceCache, layer, coords, depthMode, colorMode, false);\n }\n\n // Draw stroke\n if (painter.renderPass === 'translucent' && layer.paint.get('fill-antialias')) {\n\n // If we defined a different color for the fill outline, we are\n // going to ignore the bits in 0x07 and just care about the global\n // clipping mask.\n // Otherwise, we only want to drawFill the antialiased parts that are\n // *outside* the current shape. This is important in case the fill\n // or stroke color is translucent. If we wouldn't clip to outside\n // the current shape, some pixels from the outline stroke overlapped\n // the (non-antialiased) fill.\n const depthMode = painter.depthModeForSublayer(\n layer.getPaintProperty('fill-outline-color') ? 2 : 0, DepthMode.ReadOnly);\n drawFillTiles(painter, sourceCache, layer, coords, depthMode, colorMode, true);\n }\n}\n\nfunction drawFillTiles(\n painter: Painter,\n sourceCache: SourceCache,\n layer: FillStyleLayer,\n coords: Array,\n depthMode: Readonly,\n colorMode: Readonly,\n isOutline: boolean) {\n const gl = painter.context.gl;\n const fillPropertyName = 'fill-pattern';\n const patternProperty = layer.paint.get(fillPropertyName);\n const image = patternProperty && patternProperty.constantOr(1 as any);\n const crossfade = layer.getCrossfadeParameters();\n let drawMode, programName, uniformValues, indexBuffer, segments;\n\n if (!isOutline) {\n programName = image ? 'fillPattern' : 'fill';\n drawMode = gl.TRIANGLES;\n } else {\n programName = image && !layer.getPaintProperty('fill-outline-color') ? 'fillOutlinePattern' : 'fillOutline';\n drawMode = gl.LINES;\n }\n\n const constantPattern = patternProperty.constantOr(null);\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n if (image && !tile.patternsLoaded()) continue;\n\n const bucket: FillBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const program = painter.useProgram(programName, programConfiguration);\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n if (image) {\n painter.context.activeTexture.set(gl.TEXTURE0);\n tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n programConfiguration.updatePaintBuffers(crossfade);\n }\n\n updatePatternPositionsInProgram(programConfiguration, fillPropertyName, constantPattern, tile, layer);\n\n const terrainCoord = terrainData ? coord : null;\n const posMatrix = terrainCoord ? terrainCoord.posMatrix : coord.posMatrix;\n const tileMatrix = painter.translatePosMatrix(posMatrix, tile,\n layer.paint.get('fill-translate'), layer.paint.get('fill-translate-anchor'));\n\n if (!isOutline) {\n indexBuffer = bucket.indexBuffer;\n segments = bucket.segments;\n uniformValues = image ?\n fillPatternUniformValues(tileMatrix, painter, crossfade, tile) :\n fillUniformValues(tileMatrix);\n } else {\n indexBuffer = bucket.indexBuffer2;\n segments = bucket.segments2;\n const drawingBufferSize = [gl.drawingBufferWidth, gl.drawingBufferHeight] as [number, number];\n uniformValues = (programName === 'fillOutlinePattern' && image) ?\n fillOutlinePatternUniformValues(tileMatrix, painter, crossfade, tile, drawingBufferSize) :\n fillOutlineUniformValues(tileMatrix, drawingBufferSize);\n }\n\n program.draw(painter.context, drawMode, depthMode,\n painter.stencilModeForClipping(coord), colorMode, CullFaceMode.disabled, uniformValues, terrainData,\n layer.id, bucket.layoutVertexBuffer, indexBuffer, segments,\n layer.paint, painter.transform.zoom, programConfiguration);\n }\n}\n","import {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {\n fillExtrusionUniformValues,\n fillExtrusionPatternUniformValues,\n} from './program/fill_extrusion_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {FillExtrusionStyleLayer} from '../style/style_layer/fill_extrusion_style_layer';\nimport type {FillExtrusionBucket} from '../data/bucket/fill_extrusion_bucket';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nimport {updatePatternPositionsInProgram} from './update_pattern_positions_in_program';\n\nexport function drawFillExtrusion(painter: Painter, source: SourceCache, layer: FillExtrusionStyleLayer, coords: Array) {\n const opacity = layer.paint.get('fill-extrusion-opacity');\n if (opacity === 0) {\n return;\n }\n\n if (painter.renderPass === 'translucent') {\n const depthMode = new DepthMode(painter.context.gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D);\n\n if (opacity === 1 && !layer.paint.get('fill-extrusion-pattern').constantOr(1 as any)) {\n const colorMode = painter.colorModeForRenderPass();\n drawExtrusionTiles(painter, source, layer, coords, depthMode, StencilMode.disabled, colorMode);\n\n } else {\n // Draw transparent buildings in two passes so that only the closest surface is drawn.\n // First draw all the extrusions into only the depth buffer. No colors are drawn.\n drawExtrusionTiles(painter, source, layer, coords, depthMode,\n StencilMode.disabled,\n ColorMode.disabled);\n\n // Then draw all the extrusions a second type, only coloring fragments if they have the\n // same depth value as the closest fragment in the previous pass. Use the stencil buffer\n // to prevent the second draw in cases where we have coincident polygons.\n drawExtrusionTiles(painter, source, layer, coords, depthMode,\n painter.stencilModeFor3D(),\n painter.colorModeForRenderPass());\n }\n }\n}\n\nfunction drawExtrusionTiles(\n painter: Painter,\n source: SourceCache,\n layer: FillExtrusionStyleLayer,\n coords: OverscaledTileID[],\n depthMode: DepthMode,\n stencilMode: Readonly,\n colorMode: Readonly) {\n const context = painter.context;\n const gl = context.gl;\n const fillPropertyName = 'fill-extrusion-pattern';\n const patternProperty = layer.paint.get(fillPropertyName);\n const image = patternProperty.constantOr(1 as any);\n const crossfade = layer.getCrossfadeParameters();\n const opacity = layer.paint.get('fill-extrusion-opacity');\n const constantPattern = patternProperty.constantOr(null);\n for (const coord of coords) {\n const tile = source.getTile(coord);\n const bucket: FillExtrusionBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const program = painter.useProgram(image ? 'fillExtrusionPattern' : 'fillExtrusion', programConfiguration);\n\n if (image) {\n painter.context.activeTexture.set(gl.TEXTURE0);\n tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n programConfiguration.updatePaintBuffers(crossfade);\n }\n\n updatePatternPositionsInProgram(programConfiguration, fillPropertyName, constantPattern, tile, layer);\n\n const matrix = painter.translatePosMatrix(\n coord.posMatrix,\n tile,\n layer.paint.get('fill-extrusion-translate'),\n layer.paint.get('fill-extrusion-translate-anchor'));\n\n const shouldUseVerticalGradient = layer.paint.get('fill-extrusion-vertical-gradient');\n const uniformValues = image ?\n fillExtrusionPatternUniformValues(matrix, painter, shouldUseVerticalGradient, opacity, coord, crossfade, tile) :\n fillExtrusionUniformValues(matrix, painter, shouldUseVerticalGradient, opacity);\n\n program.draw(context, context.gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.backCCW,\n uniformValues, terrainData, layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer,\n bucket.segments, layer.paint, painter.transform.zoom,\n programConfiguration, painter.style.map.terrain && bucket.centroidVertexBuffer);\n }\n}\n","import {Texture} from './texture';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {Tile} from '../source/tile';\nimport {\n hillshadeUniformValues,\n hillshadeUniformPrepareValues\n} from './program/hillshade_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {HillshadeStyleLayer} from '../style/style_layer/hillshade_style_layer';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nexport function drawHillshade(painter: Painter, sourceCache: SourceCache, layer: HillshadeStyleLayer, tileIDs: Array) {\n if (painter.renderPass !== 'offscreen' && painter.renderPass !== 'translucent') return;\n\n const context = painter.context;\n\n const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n const colorMode = painter.colorModeForRenderPass();\n\n const [stencilModes, coords] = painter.renderPass === 'translucent' ?\n painter.stencilConfigForOverlap(tileIDs) : [{}, tileIDs];\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n if (typeof tile.needsHillshadePrepare !== 'undefined' && tile.needsHillshadePrepare && painter.renderPass === 'offscreen') {\n prepareHillshade(painter, tile, layer, depthMode, StencilMode.disabled, colorMode);\n } else if (painter.renderPass === 'translucent') {\n renderHillshade(painter, coord, tile, layer, depthMode, stencilModes[coord.overscaledZ], colorMode);\n }\n }\n\n context.viewport.set([0, 0, painter.width, painter.height]);\n}\n\nfunction renderHillshade(\n painter: Painter,\n coord: OverscaledTileID,\n tile: Tile,\n layer: HillshadeStyleLayer,\n depthMode: Readonly,\n stencilMode: Readonly,\n colorMode: Readonly) {\n const context = painter.context;\n const gl = context.gl;\n const fbo = tile.fbo;\n if (!fbo) return;\n\n const program = painter.useProgram('hillshade');\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n context.activeTexture.set(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get());\n\n const terrainCoord = terrainData ? coord : null;\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n hillshadeUniformValues(painter, tile, layer, terrainCoord), terrainData, layer.id, painter.rasterBoundsBuffer,\n painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments);\n\n}\n\n// hillshade rendering is done in two steps. the prepare step first calculates the slope of the terrain in the x and y\n// directions for each pixel, and saves those values to a framebuffer texture in the r and g channels.\nfunction prepareHillshade(\n painter: Painter,\n tile: Tile,\n layer: HillshadeStyleLayer,\n depthMode: Readonly,\n stencilMode: Readonly,\n colorMode: Readonly) {\n const context = painter.context;\n const gl = context.gl;\n const dem = tile.dem;\n if (dem && dem.data) {\n const tileSize = dem.dim;\n const textureStride = dem.stride;\n\n const pixelData = dem.getPixels();\n context.activeTexture.set(gl.TEXTURE1);\n\n context.pixelStoreUnpackPremultiplyAlpha.set(false);\n tile.demTexture = tile.demTexture || painter.getTileTexture(textureStride);\n if (tile.demTexture) {\n const demTexture = tile.demTexture;\n demTexture.update(pixelData, {premultiply: false});\n demTexture.bind(gl.NEAREST, gl.CLAMP_TO_EDGE);\n } else {\n tile.demTexture = new Texture(context, pixelData, gl.RGBA, {premultiply: false});\n tile.demTexture.bind(gl.NEAREST, gl.CLAMP_TO_EDGE);\n }\n\n context.activeTexture.set(gl.TEXTURE0);\n\n let fbo = tile.fbo;\n\n if (!fbo) {\n const renderTexture = new Texture(context, {width: tileSize, height: tileSize, data: null}, gl.RGBA);\n renderTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n\n fbo = tile.fbo = context.createFramebuffer(tileSize, tileSize, true, false);\n fbo.colorAttachment.set(renderTexture.texture);\n }\n\n context.bindFramebuffer.set(fbo.framebuffer);\n context.viewport.set([0, 0, tileSize, tileSize]);\n\n painter.useProgram('hillshadePrepare').draw(context, gl.TRIANGLES,\n depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n hillshadeUniformPrepareValues(tile.tileID, dem),\n null, layer.id, painter.rasterBoundsBuffer,\n painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments);\n\n tile.needsHillshadePrepare = false;\n }\n}\n","import {clamp} from '../util/util';\n\nimport {ImageSource} from '../source/image_source';\nimport {browser} from '../util/browser';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {rasterUniformValues} from './program/raster_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {RasterStyleLayer} from '../style/style_layer/raster_style_layer';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nexport function drawRaster(painter: Painter, sourceCache: SourceCache, layer: RasterStyleLayer, tileIDs: Array) {\n if (painter.renderPass !== 'translucent') return;\n if (layer.paint.get('raster-opacity') === 0) return;\n if (!tileIDs.length) return;\n\n const context = painter.context;\n const gl = context.gl;\n const source = sourceCache.getSource();\n const program = painter.useProgram('raster');\n\n const colorMode = painter.colorModeForRenderPass();\n\n const [stencilModes, coords] = source instanceof ImageSource ? [{}, tileIDs] :\n painter.stencilConfigForOverlap(tileIDs);\n\n const minTileZ = coords[coords.length - 1].overscaledZ;\n\n const align = !painter.options.moving;\n for (const coord of coords) {\n // Set the lower zoom level to sublayer 0, and higher zoom levels to higher sublayers\n // Use gl.LESS to prevent double drawing in areas where tiles overlap.\n const depthMode = painter.depthModeForSublayer(coord.overscaledZ - minTileZ,\n layer.paint.get('raster-opacity') === 1 ? DepthMode.ReadWrite : DepthMode.ReadOnly, gl.LESS);\n\n const tile = sourceCache.getTile(coord);\n\n tile.registerFadeDuration(layer.paint.get('raster-fade-duration'));\n\n const parentTile = sourceCache.findLoadedParent(coord, 0),\n fade = getFadeValues(tile, parentTile, sourceCache, layer, painter.transform, painter.style.map.terrain);\n\n let parentScaleBy, parentTL;\n\n const textureFilter = layer.paint.get('raster-resampling') === 'nearest' ? gl.NEAREST : gl.LINEAR;\n\n context.activeTexture.set(gl.TEXTURE0);\n tile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST);\n\n context.activeTexture.set(gl.TEXTURE1);\n\n if (parentTile) {\n parentTile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST);\n parentScaleBy = Math.pow(2, parentTile.tileID.overscaledZ - tile.tileID.overscaledZ);\n parentTL = [tile.tileID.canonical.x * parentScaleBy % 1, tile.tileID.canonical.y * parentScaleBy % 1];\n\n } else {\n tile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST);\n }\n\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n const terrainCoord = terrainData ? coord : null;\n const posMatrix = terrainCoord ? terrainCoord.posMatrix : painter.transform.calculatePosMatrix(coord.toUnwrapped(), align);\n const uniformValues = rasterUniformValues(posMatrix, parentTL || [0, 0], parentScaleBy || 1, fade, layer);\n\n if (source instanceof ImageSource) {\n program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id, source.boundsBuffer,\n painter.quadTriangleIndexBuffer, source.boundsSegments);\n } else {\n program.draw(context, gl.TRIANGLES, depthMode, stencilModes[coord.overscaledZ], colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id, painter.rasterBoundsBuffer,\n painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments);\n }\n }\n}\n\nfunction getFadeValues(tile, parentTile, sourceCache, layer, transform, terrain) {\n const fadeDuration = layer.paint.get('raster-fade-duration');\n\n if (!terrain && fadeDuration > 0) {\n const now = browser.now();\n const sinceTile = (now - tile.timeAdded) / fadeDuration;\n const sinceParent = parentTile ? (now - parentTile.timeAdded) / fadeDuration : -1;\n\n const source = sourceCache.getSource();\n const idealZ = transform.coveringZoomLevel({\n tileSize: source.tileSize,\n roundZoom: source.roundZoom\n });\n\n // if no parent or parent is older, fade in; if parent is younger, fade out\n const fadeIn = !parentTile || Math.abs(parentTile.tileID.overscaledZ - idealZ) > Math.abs(tile.tileID.overscaledZ - idealZ);\n\n const childOpacity = (fadeIn && tile.refreshedUponExpiration) ? 1 : clamp(fadeIn ? sinceTile : 1 - sinceParent, 0, 1);\n\n // we don't crossfade tiles that were just refreshed upon expiring:\n // once they're old enough to pass the crossfading threshold\n // (fadeDuration), unset the `refreshedUponExpiration` flag so we don't\n // incorrectly fail to crossfade them when zooming\n if (tile.refreshedUponExpiration && sinceTile >= 1) tile.refreshedUponExpiration = false;\n\n if (parentTile) {\n return {\n opacity: 1,\n mix: 1 - childOpacity\n };\n } else {\n return {\n opacity: childOpacity,\n mix: 0\n };\n }\n } else {\n return {\n opacity: 1,\n mix: 0\n };\n }\n}\n","import {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {debugUniformValues} from './program/debug_program';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {ColorMode} from '../gl/color_mode';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport {Style} from '../style/style';\n\nconst topColor = new Color(1, 0, 0, 1);\nconst btmColor = new Color(0, 1, 0, 1);\nconst leftColor = new Color(0, 0, 1, 1);\nconst rightColor = new Color(1, 0, 1, 1);\nconst centerColor = new Color(0, 1, 1, 1);\n\nexport function drawDebugPadding(painter: Painter) {\n const padding = painter.transform.padding;\n const lineWidth = 3;\n // Top\n drawHorizontalLine(painter, painter.transform.height - (padding.top || 0), lineWidth, topColor);\n // Bottom\n drawHorizontalLine(painter, padding.bottom || 0, lineWidth, btmColor);\n // Left\n drawVerticalLine(painter, padding.left || 0, lineWidth, leftColor);\n // Right\n drawVerticalLine(painter, painter.transform.width - (padding.right || 0), lineWidth, rightColor);\n // Center\n const center = painter.transform.centerPoint;\n drawCrosshair(painter, center.x, painter.transform.height - center.y, centerColor);\n}\n\nfunction drawCrosshair(painter: Painter, x: number, y: number, color: Color) {\n const size = 20;\n const lineWidth = 2;\n //Vertical line\n drawDebugSSRect(painter, x - lineWidth / 2, y - size / 2, lineWidth, size, color);\n //Horizontal line\n drawDebugSSRect(painter, x - size / 2, y - lineWidth / 2, size, lineWidth, color);\n}\n\nfunction drawHorizontalLine(painter: Painter, y: number, lineWidth: number, color: Color) {\n drawDebugSSRect(painter, 0, y + lineWidth / 2, painter.transform.width, lineWidth, color);\n}\n\nfunction drawVerticalLine(painter: Painter, x: number, lineWidth: number, color: Color) {\n drawDebugSSRect(painter, x - lineWidth / 2, 0, lineWidth, painter.transform.height, color);\n}\n\nfunction drawDebugSSRect(painter: Painter, x: number, y: number, width: number, height: number, color: Color) {\n const context = painter.context;\n const gl = context.gl;\n\n gl.enable(gl.SCISSOR_TEST);\n gl.scissor(x * painter.pixelRatio, y * painter.pixelRatio, width * painter.pixelRatio, height * painter.pixelRatio);\n context.clear({color});\n gl.disable(gl.SCISSOR_TEST);\n}\n\nexport function drawDebug(painter: Painter, sourceCache: SourceCache, coords: Array) {\n for (let i = 0; i < coords.length; i++) {\n drawDebugTile(painter, sourceCache, coords[i]);\n }\n}\n\nfunction drawDebugTile(painter: Painter, sourceCache: SourceCache, coord: OverscaledTileID) {\n const context = painter.context;\n const gl = context.gl;\n\n const posMatrix = coord.posMatrix;\n const program = painter.useProgram('debug');\n\n const depthMode = DepthMode.disabled;\n const stencilMode = StencilMode.disabled;\n const colorMode = painter.colorModeForRenderPass();\n const id = '$debug';\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n context.activeTexture.set(gl.TEXTURE0);\n\n const tileRawData = sourceCache.getTileByID(coord.key).latestRawTileData;\n const tileByteLength = (tileRawData && tileRawData.byteLength) || 0;\n const tileSizeKb = Math.floor(tileByteLength / 1024);\n const tileSize = sourceCache.getTile(coord).tileSize;\n const scaleRatio = (512 / Math.min(tileSize, 512) * (coord.overscaledZ / painter.transform.zoom)) * 0.5;\n let tileIdText = coord.canonical.toString();\n if (coord.overscaledZ !== coord.canonical.z) {\n tileIdText += ` => ${coord.overscaledZ}`;\n }\n const tileLabel = `${tileIdText} ${tileSizeKb}kB`;\n drawTextToOverlay(painter, tileLabel);\n\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, ColorMode.alphaBlended, CullFaceMode.disabled,\n debugUniformValues(posMatrix, Color.transparent, scaleRatio), null, id,\n painter.debugBuffer, painter.quadTriangleIndexBuffer, painter.debugSegments);\n program.draw(context, gl.LINE_STRIP, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n debugUniformValues(posMatrix, Color.red), terrainData, id,\n painter.debugBuffer, painter.tileBorderIndexBuffer, painter.debugSegments);\n}\n\nfunction drawTextToOverlay(painter: Painter, text: string) {\n painter.initDebugOverlayCanvas();\n const canvas = painter.debugOverlayCanvas;\n const gl = painter.context.gl;\n const ctx2d = painter.debugOverlayCanvas.getContext('2d');\n ctx2d.clearRect(0, 0, canvas.width, canvas.height);\n\n ctx2d.shadowColor = 'white';\n ctx2d.shadowBlur = 2;\n ctx2d.lineWidth = 1.5;\n ctx2d.strokeStyle = 'white';\n ctx2d.textBaseline = 'top';\n ctx2d.font = `bold ${36}px Open Sans, sans-serif`;\n ctx2d.fillText(text, 5, 5);\n ctx2d.strokeText(text, 5, 5);\n\n painter.debugOverlayTexture.update(canvas);\n painter.debugOverlayTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n}\n\nexport function selectDebugSource(style: Style, zoom: number): SourceCache | null {\n // Use vector source with highest maxzoom\n // Else use source with highest maxzoom of any type\n let selectedSource: SourceCache = null;\n const layers = Object.values(style._layers);\n const sources = layers.flatMap((layer) => {\n if (layer.source && !layer.isHidden(zoom)) {\n const sourceCache = style.sourceCaches[layer.source];\n return [sourceCache];\n } else {\n return [];\n }\n });\n const vectorSources = sources.filter((source) => source.getSource().type === 'vector');\n const otherSources = sources.filter((source) => source.getSource().type !== 'vector');\n const considerSource = (source: SourceCache) => {\n if (!selectedSource || (selectedSource.getSource().maxzoom < source.getSource().maxzoom)) {\n selectedSource = source;\n }\n };\n vectorSources.forEach((source) => considerSource(source));\n if (!selectedSource) {\n otherSources.forEach((source) => considerSource(source));\n }\n return selectedSource;\n}\n","import {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {terrainUniformValues, terrainDepthUniformValues, terrainCoordsUniformValues} from './program/terrain_program';\nimport type {Painter} from './painter';\nimport type {Tile} from '../source/tile';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {ColorMode} from '../gl/color_mode';\nimport {Terrain} from './terrain';\n\n/**\n * Redraw the Depth Framebuffer\n * @param painter - the painter\n * @param terrain - the terrain\n */\nfunction drawDepth(painter: Painter, terrain: Terrain) {\n const context = painter.context;\n const gl = context.gl;\n const colorMode = ColorMode.unblended;\n const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, [0, 1]);\n const mesh = terrain.getTerrainMesh();\n const tiles = terrain.sourceCache.getRenderableTiles();\n const program = painter.useProgram('terrainDepth');\n context.bindFramebuffer.set(terrain.getFramebuffer('depth').framebuffer);\n context.viewport.set([0, 0, painter.width / devicePixelRatio, painter.height / devicePixelRatio]);\n context.clear({color: Color.transparent, depth: 1});\n for (const tile of tiles) {\n const terrainData = terrain.getTerrainData(tile.tileID);\n const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped());\n const uniformValues = terrainDepthUniformValues(posMatrix, terrain.getMeshFrameDelta(painter.transform.zoom));\n program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments);\n }\n context.bindFramebuffer.set(null);\n context.viewport.set([0, 0, painter.width, painter.height]);\n}\n\n/**\n * Redraw the Coords Framebuffers\n * @param painter - the painter\n * @param terrain - the terrain\n */\nfunction drawCoords(painter: Painter, terrain: Terrain) {\n const context = painter.context;\n const gl = context.gl;\n const colorMode = ColorMode.unblended;\n const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, [0, 1]);\n const mesh = terrain.getTerrainMesh();\n const coords = terrain.getCoordsTexture();\n const tiles = terrain.sourceCache.getRenderableTiles();\n\n // draw tile-coords into framebuffer\n const program = painter.useProgram('terrainCoords');\n context.bindFramebuffer.set(terrain.getFramebuffer('coords').framebuffer);\n context.viewport.set([0, 0, painter.width / devicePixelRatio, painter.height / devicePixelRatio]);\n context.clear({color: Color.transparent, depth: 1});\n terrain.coordsIndex = [];\n for (const tile of tiles) {\n const terrainData = terrain.getTerrainData(tile.tileID);\n context.activeTexture.set(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, coords.texture);\n const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped());\n const uniformValues = terrainCoordsUniformValues(posMatrix, 255 - terrain.coordsIndex.length, terrain.getMeshFrameDelta(painter.transform.zoom));\n program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments);\n terrain.coordsIndex.push(tile.tileID.key);\n }\n context.bindFramebuffer.set(null);\n context.viewport.set([0, 0, painter.width, painter.height]);\n}\n\nfunction drawTerrain(painter: Painter, terrain: Terrain, tiles: Array) {\n const context = painter.context;\n const gl = context.gl;\n const colorMode = painter.colorModeForRenderPass();\n const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D);\n const program = painter.useProgram('terrain');\n const mesh = terrain.getTerrainMesh();\n\n context.bindFramebuffer.set(null);\n context.viewport.set([0, 0, painter.width, painter.height]);\n\n for (const tile of tiles) {\n const texture = painter.renderToTexture.getTexture(tile);\n const terrainData = terrain.getTerrainData(tile.tileID);\n context.activeTexture.set(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, texture.texture);\n const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped());\n const uniformValues = terrainUniformValues(posMatrix, terrain.getMeshFrameDelta(painter.transform.zoom));\n program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments);\n }\n\n}\n\nexport {\n drawTerrain,\n drawDepth,\n drawCoords\n};\n","import {browser} from '../util/browser';\nimport {mat4, vec3} from 'gl-matrix';\nimport {SourceCache} from '../source/source_cache';\nimport {EXTENT} from '../data/extent';\nimport {pixelsToTileUnits} from '../source/pixels_to_tile_units';\nimport {SegmentVector} from '../data/segment';\nimport {RasterBoundsArray, PosArray, TriangleIndexArray, LineStripIndexArray} from '../data/array_types.g';\nimport rasterBoundsAttributes from '../data/raster_bounds_attributes';\nimport posAttributes from '../data/pos_attributes';\nimport {ProgramConfiguration} from '../data/program_configuration';\nimport {CrossTileSymbolIndex} from '../symbol/cross_tile_symbol_index';\nimport {shaders} from '../shaders/shaders';\nimport {Program} from './program';\nimport {programUniforms} from './program/program_uniforms';\nimport {Context} from '../gl/context';\nimport {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Texture} from './texture';\nimport {clippingMaskUniformValues} from './program/clipping_mask_program';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {drawSymbols} from './draw_symbol';\nimport {drawCircles} from './draw_circle';\nimport {drawHeatmap} from './draw_heatmap';\nimport {drawLine} from './draw_line';\nimport {drawFill} from './draw_fill';\nimport {drawFillExtrusion} from './draw_fill_extrusion';\nimport {drawHillshade} from './draw_hillshade';\nimport {drawRaster} from './draw_raster';\nimport {drawBackground} from './draw_background';\nimport {drawDebug, drawDebugPadding, selectDebugSource} from './draw_debug';\nimport {drawCustom} from './draw_custom';\nimport {drawDepth, drawCoords} from './draw_terrain';\nimport {OverscaledTileID} from '../source/tile_id';\n\nimport type {Transform} from '../geo/transform';\nimport type {Tile} from '../source/tile';\nimport type {Style} from '../style/style';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {CrossFaded} from '../style/properties';\nimport type {LineAtlas} from './line_atlas';\nimport type {ImageManager} from './image_manager';\nimport type {GlyphManager} from './glyph_manager';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {IndexBuffer} from '../gl/index_buffer';\nimport type {DepthRangeType, DepthMaskType, DepthFuncType} from '../gl/types';\nimport type {ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport {RenderToTexture} from './render_to_texture';\n\nexport type RenderPass = 'offscreen' | 'opaque' | 'translucent';\n\ntype PainterOptions = {\n showOverdrawInspector: boolean;\n showTileBoundaries: boolean;\n showPadding: boolean;\n rotating: boolean;\n zooming: boolean;\n moving: boolean;\n fadeDuration: number;\n};\n\n/**\n * @internal\n * Initialize a new painter object.\n */\nexport class Painter {\n context: Context;\n transform: Transform;\n renderToTexture: RenderToTexture;\n _tileTextures: {\n [_: number]: Array;\n };\n numSublayers: number;\n depthEpsilon: number;\n emptyProgramConfiguration: ProgramConfiguration;\n width: number;\n height: number;\n pixelRatio: number;\n tileExtentBuffer: VertexBuffer;\n tileExtentSegments: SegmentVector;\n debugBuffer: VertexBuffer;\n debugSegments: SegmentVector;\n rasterBoundsBuffer: VertexBuffer;\n rasterBoundsSegments: SegmentVector;\n viewportBuffer: VertexBuffer;\n viewportSegments: SegmentVector;\n quadTriangleIndexBuffer: IndexBuffer;\n tileBorderIndexBuffer: IndexBuffer;\n _tileClippingMaskIDs: {[_: string]: number};\n stencilClearMode: StencilMode;\n style: Style;\n options: PainterOptions;\n lineAtlas: LineAtlas;\n imageManager: ImageManager;\n glyphManager: GlyphManager;\n depthRangeFor3D: DepthRangeType;\n opaquePassCutoff: number;\n renderPass: RenderPass;\n currentLayer: number;\n currentStencilSource: string;\n nextStencilID: number;\n id: string;\n _showOverdrawInspector: boolean;\n cache: {[_: string]: Program};\n crossTileSymbolIndex: CrossTileSymbolIndex;\n symbolFadeChange: number;\n debugOverlayTexture: Texture;\n debugOverlayCanvas: HTMLCanvasElement;\n // this object stores the current camera-matrix and the last render time\n // of the terrain-facilitators. e.g. depth & coords framebuffers\n // every time the camera-matrix changes the terrain-facilitators will be redrawn.\n terrainFacilitator: {dirty: boolean; matrix: mat4; renderTime: number};\n\n constructor(gl: WebGLRenderingContext | WebGL2RenderingContext, transform: Transform) {\n this.context = new Context(gl);\n this.transform = transform;\n this._tileTextures = {};\n this.terrainFacilitator = {dirty: true, matrix: mat4.create(), renderTime: 0};\n\n this.setup();\n\n // Within each layer there are multiple distinct z-planes that can be drawn to.\n // This is implemented using the WebGL depth buffer.\n this.numSublayers = SourceCache.maxUnderzooming + SourceCache.maxOverzooming + 1;\n this.depthEpsilon = 1 / Math.pow(2, 16);\n\n this.crossTileSymbolIndex = new CrossTileSymbolIndex();\n }\n\n /*\n * Update the GL viewport, projection matrix, and transforms to compensate\n * for a new width and height value.\n */\n resize(width: number, height: number, pixelRatio: number) {\n this.width = Math.floor(width * pixelRatio);\n this.height = Math.floor(height * pixelRatio);\n this.pixelRatio = pixelRatio;\n this.context.viewport.set([0, 0, this.width, this.height]);\n\n if (this.style) {\n for (const layerId of this.style._order) {\n this.style._layers[layerId].resize();\n }\n }\n }\n\n setup() {\n const context = this.context;\n\n const tileExtentArray = new PosArray();\n tileExtentArray.emplaceBack(0, 0);\n tileExtentArray.emplaceBack(EXTENT, 0);\n tileExtentArray.emplaceBack(0, EXTENT);\n tileExtentArray.emplaceBack(EXTENT, EXTENT);\n this.tileExtentBuffer = context.createVertexBuffer(tileExtentArray, posAttributes.members);\n this.tileExtentSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n\n const debugArray = new PosArray();\n debugArray.emplaceBack(0, 0);\n debugArray.emplaceBack(EXTENT, 0);\n debugArray.emplaceBack(0, EXTENT);\n debugArray.emplaceBack(EXTENT, EXTENT);\n this.debugBuffer = context.createVertexBuffer(debugArray, posAttributes.members);\n this.debugSegments = SegmentVector.simpleSegment(0, 0, 4, 5);\n\n const rasterBoundsArray = new RasterBoundsArray();\n rasterBoundsArray.emplaceBack(0, 0, 0, 0);\n rasterBoundsArray.emplaceBack(EXTENT, 0, EXTENT, 0);\n rasterBoundsArray.emplaceBack(0, EXTENT, 0, EXTENT);\n rasterBoundsArray.emplaceBack(EXTENT, EXTENT, EXTENT, EXTENT);\n this.rasterBoundsBuffer = context.createVertexBuffer(rasterBoundsArray, rasterBoundsAttributes.members);\n this.rasterBoundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n\n const viewportArray = new PosArray();\n viewportArray.emplaceBack(0, 0);\n viewportArray.emplaceBack(1, 0);\n viewportArray.emplaceBack(0, 1);\n viewportArray.emplaceBack(1, 1);\n this.viewportBuffer = context.createVertexBuffer(viewportArray, posAttributes.members);\n this.viewportSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n\n const tileLineStripIndices = new LineStripIndexArray();\n tileLineStripIndices.emplaceBack(0);\n tileLineStripIndices.emplaceBack(1);\n tileLineStripIndices.emplaceBack(3);\n tileLineStripIndices.emplaceBack(2);\n tileLineStripIndices.emplaceBack(0);\n this.tileBorderIndexBuffer = context.createIndexBuffer(tileLineStripIndices);\n\n const quadTriangleIndices = new TriangleIndexArray();\n quadTriangleIndices.emplaceBack(0, 1, 2);\n quadTriangleIndices.emplaceBack(2, 1, 3);\n this.quadTriangleIndexBuffer = context.createIndexBuffer(quadTriangleIndices);\n\n const gl = this.context.gl;\n this.stencilClearMode = new StencilMode({func: gl.ALWAYS, mask: 0}, 0x0, 0xFF, gl.ZERO, gl.ZERO, gl.ZERO);\n }\n\n /*\n * Reset the drawing canvas by clearing the stencil buffer so that we can draw\n * new tiles at the same location, while retaining previously drawn pixels.\n */\n clearStencil() {\n const context = this.context;\n const gl = context.gl;\n\n this.nextStencilID = 1;\n this.currentStencilSource = undefined;\n\n // As a temporary workaround for https://github.com/mapbox/mapbox-gl-js/issues/5490,\n // pending an upstream fix, we draw a fullscreen stencil=0 clipping mask here,\n // effectively clearing the stencil buffer: once an upstream patch lands, remove\n // this function in favor of context.clear({ stencil: 0x0 })\n\n const matrix = mat4.create();\n mat4.ortho(matrix, 0, this.width, this.height, 0, 0, 1);\n mat4.scale(matrix, matrix, [gl.drawingBufferWidth, gl.drawingBufferHeight, 0]);\n\n this.useProgram('clippingMask').draw(context, gl.TRIANGLES,\n DepthMode.disabled, this.stencilClearMode, ColorMode.disabled, CullFaceMode.disabled,\n clippingMaskUniformValues(matrix), null,\n '$clipping', this.viewportBuffer,\n this.quadTriangleIndexBuffer, this.viewportSegments);\n }\n\n _renderTileClippingMasks(layer: StyleLayer, tileIDs: Array) {\n if (this.currentStencilSource === layer.source || !layer.isTileClipped() || !tileIDs || !tileIDs.length) return;\n\n this.currentStencilSource = layer.source;\n\n const context = this.context;\n const gl = context.gl;\n\n if (this.nextStencilID + tileIDs.length > 256) {\n // we'll run out of fresh IDs so we need to clear and start from scratch\n this.clearStencil();\n }\n\n context.setColorMode(ColorMode.disabled);\n context.setDepthMode(DepthMode.disabled);\n\n const program = this.useProgram('clippingMask');\n\n this._tileClippingMaskIDs = {};\n\n for (const tileID of tileIDs) {\n const id = this._tileClippingMaskIDs[tileID.key] = this.nextStencilID++;\n const terrainData = this.style.map.terrain && this.style.map.terrain.getTerrainData(tileID);\n\n program.draw(context, gl.TRIANGLES, DepthMode.disabled,\n // Tests will always pass, and ref value will be written to stencil buffer.\n new StencilMode({func: gl.ALWAYS, mask: 0}, id, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE),\n ColorMode.disabled, CullFaceMode.disabled, clippingMaskUniformValues(tileID.posMatrix),\n terrainData, '$clipping', this.tileExtentBuffer,\n this.quadTriangleIndexBuffer, this.tileExtentSegments);\n }\n }\n\n stencilModeFor3D(): StencilMode {\n this.currentStencilSource = undefined;\n\n if (this.nextStencilID + 1 > 256) {\n this.clearStencil();\n }\n\n const id = this.nextStencilID++;\n const gl = this.context.gl;\n return new StencilMode({func: gl.NOTEQUAL, mask: 0xFF}, id, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE);\n }\n\n stencilModeForClipping(tileID: OverscaledTileID): StencilMode {\n const gl = this.context.gl;\n return new StencilMode({func: gl.EQUAL, mask: 0xFF}, this._tileClippingMaskIDs[tileID.key], 0x00, gl.KEEP, gl.KEEP, gl.REPLACE);\n }\n\n /*\n * Sort coordinates by Z as drawing tiles is done in Z-descending order.\n * All children with the same Z write the same stencil value. Children\n * stencil values are greater than parent's. This is used only for raster\n * and raster-dem tiles, which are already clipped to tile boundaries, to\n * mask area of tile overlapped by children tiles.\n * Stencil ref values continue range used in _tileClippingMaskIDs.\n *\n * Returns [StencilMode for tile overscaleZ map, sortedCoords].\n */\n stencilConfigForOverlap(tileIDs: Array): [{\n [_: number]: Readonly;\n }, Array] {\n const gl = this.context.gl;\n const coords = tileIDs.sort((a, b) => b.overscaledZ - a.overscaledZ);\n const minTileZ = coords[coords.length - 1].overscaledZ;\n const stencilValues = coords[0].overscaledZ - minTileZ + 1;\n if (stencilValues > 1) {\n this.currentStencilSource = undefined;\n if (this.nextStencilID + stencilValues > 256) {\n this.clearStencil();\n }\n const zToStencilMode = {};\n for (let i = 0; i < stencilValues; i++) {\n zToStencilMode[i + minTileZ] = new StencilMode({func: gl.GEQUAL, mask: 0xFF}, i + this.nextStencilID, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE);\n }\n this.nextStencilID += stencilValues;\n return [zToStencilMode, coords];\n }\n return [{[minTileZ]: StencilMode.disabled}, coords];\n }\n\n colorModeForRenderPass(): Readonly {\n const gl = this.context.gl;\n if (this._showOverdrawInspector) {\n const numOverdrawSteps = 8;\n const a = 1 / numOverdrawSteps;\n\n return new ColorMode([gl.CONSTANT_COLOR, gl.ONE], new Color(a, a, a, 0), [true, true, true, true]);\n } else if (this.renderPass === 'opaque') {\n return ColorMode.unblended;\n } else {\n return ColorMode.alphaBlended;\n }\n }\n\n depthModeForSublayer(n: number, mask: DepthMaskType, func?: DepthFuncType | null): Readonly {\n if (!this.opaquePassEnabledForLayer()) return DepthMode.disabled;\n const depth = 1 - ((1 + this.currentLayer) * this.numSublayers + n) * this.depthEpsilon;\n return new DepthMode(func || this.context.gl.LEQUAL, mask, [depth, depth]);\n }\n\n /*\n * The opaque pass and 3D layers both use the depth buffer.\n * Layers drawn above 3D layers need to be drawn using the\n * painter's algorithm so that they appear above 3D features.\n * This returns true for layers that can be drawn using the\n * opaque pass.\n */\n opaquePassEnabledForLayer() {\n return this.currentLayer < this.opaquePassCutoff;\n }\n\n render(style: Style, options: PainterOptions) {\n this.style = style;\n this.options = options;\n\n this.lineAtlas = style.lineAtlas;\n this.imageManager = style.imageManager;\n this.glyphManager = style.glyphManager;\n\n this.symbolFadeChange = style.placement.symbolFadeChange(browser.now());\n\n this.imageManager.beginFrame();\n\n const layerIds = this.style._order;\n const sourceCaches = this.style.sourceCaches;\n\n const coordsAscending: {[_: string]: Array} = {};\n const coordsDescending: {[_: string]: Array} = {};\n const coordsDescendingSymbol: {[_: string]: Array} = {};\n\n for (const id in sourceCaches) {\n const sourceCache = sourceCaches[id];\n if (sourceCache.used) {\n sourceCache.prepare(this.context);\n }\n\n coordsAscending[id] = sourceCache.getVisibleCoordinates();\n coordsDescending[id] = coordsAscending[id].slice().reverse();\n coordsDescendingSymbol[id] = sourceCache.getVisibleCoordinates(true).reverse();\n }\n\n this.opaquePassCutoff = Infinity;\n for (let i = 0; i < layerIds.length; i++) {\n const layerId = layerIds[i];\n if (this.style._layers[layerId].is3D()) {\n this.opaquePassCutoff = i;\n break;\n }\n }\n\n if (this.renderToTexture) {\n this.renderToTexture.prepareForRender(this.style, this.transform.zoom);\n // this is disabled, because render-to-texture is rendering all layers from bottom to top.\n this.opaquePassCutoff = 0;\n\n // update coords/depth-framebuffer on camera movement, or tile reloading\n const newTiles = this.style.map.terrain.sourceCache.tilesAfterTime(this.terrainFacilitator.renderTime);\n if (this.terrainFacilitator.dirty || !mat4.equals(this.terrainFacilitator.matrix, this.transform.projMatrix) || newTiles.length) {\n mat4.copy(this.terrainFacilitator.matrix, this.transform.projMatrix);\n this.terrainFacilitator.renderTime = Date.now();\n this.terrainFacilitator.dirty = false;\n drawDepth(this, this.style.map.terrain);\n drawCoords(this, this.style.map.terrain);\n }\n }\n\n // Offscreen pass ===============================================\n // We first do all rendering that requires rendering to a separate\n // framebuffer, and then save those for rendering back to the map\n // later: in doing this we avoid doing expensive framebuffer restores.\n this.renderPass = 'offscreen';\n\n for (const layerId of layerIds) {\n const layer = this.style._layers[layerId];\n if (!layer.hasOffscreenPass() || layer.isHidden(this.transform.zoom)) continue;\n\n const coords = coordsDescending[layer.source];\n if (layer.type !== 'custom' && !coords.length) continue;\n\n this.renderLayer(this, sourceCaches[layer.source], layer, coords);\n }\n\n // Rebind the main framebuffer now that all offscreen layers have been rendered:\n this.context.bindFramebuffer.set(null);\n\n // Clear buffers in preparation for drawing to the main framebuffer\n this.context.clear({color: options.showOverdrawInspector ? Color.black : Color.transparent, depth: 1});\n this.clearStencil();\n\n this._showOverdrawInspector = options.showOverdrawInspector;\n this.depthRangeFor3D = [0, 1 - ((style._order.length + 2) * this.numSublayers * this.depthEpsilon)];\n\n // Opaque pass ===============================================\n // Draw opaque layers top-to-bottom first.\n if (!this.renderToTexture) {\n this.renderPass = 'opaque';\n\n for (this.currentLayer = layerIds.length - 1; this.currentLayer >= 0; this.currentLayer--) {\n const layer = this.style._layers[layerIds[this.currentLayer]];\n const sourceCache = sourceCaches[layer.source];\n const coords = coordsAscending[layer.source];\n\n this._renderTileClippingMasks(layer, coords);\n this.renderLayer(this, sourceCache, layer, coords);\n }\n }\n\n // Translucent pass ===============================================\n // Draw all other layers bottom-to-top.\n this.renderPass = 'translucent';\n\n for (this.currentLayer = 0; this.currentLayer < layerIds.length; this.currentLayer++) {\n const layer = this.style._layers[layerIds[this.currentLayer]];\n const sourceCache = sourceCaches[layer.source];\n\n if (this.renderToTexture && this.renderToTexture.renderLayer(layer)) continue;\n\n // For symbol layers in the translucent pass, we add extra tiles to the renderable set\n // for cross-tile symbol fading. Symbol layers don't use tile clipping, so no need to render\n // separate clipping masks\n const coords = (layer.type === 'symbol' ? coordsDescendingSymbol : coordsDescending)[layer.source];\n\n this._renderTileClippingMasks(layer, coordsAscending[layer.source]);\n this.renderLayer(this, sourceCache, layer, coords);\n }\n\n if (this.options.showTileBoundaries) {\n const selectedSource = selectDebugSource(this.style, this.transform.zoom);\n if (selectedSource) {\n drawDebug(this, selectedSource, selectedSource.getVisibleCoordinates());\n }\n }\n\n if (this.options.showPadding) {\n drawDebugPadding(this);\n }\n\n // Set defaults for most GL values so that anyone using the state after the render\n // encounters more expected values.\n this.context.setDefault();\n }\n\n renderLayer(painter: Painter, sourceCache: SourceCache, layer: StyleLayer, coords: Array) {\n if (layer.isHidden(this.transform.zoom)) return;\n if (layer.type !== 'background' && layer.type !== 'custom' && !(coords || []).length) return;\n this.id = layer.id;\n\n switch (layer.type) {\n case 'symbol':\n drawSymbols(painter, sourceCache, layer as any, coords, this.style.placement.variableOffsets);\n break;\n case 'circle':\n drawCircles(painter, sourceCache, layer as any, coords);\n break;\n case 'heatmap':\n drawHeatmap(painter, sourceCache, layer as any, coords);\n break;\n case 'line':\n drawLine(painter, sourceCache, layer as any, coords);\n break;\n case 'fill':\n drawFill(painter, sourceCache, layer as any, coords);\n break;\n case 'fill-extrusion':\n drawFillExtrusion(painter, sourceCache, layer as any, coords);\n break;\n case 'hillshade':\n drawHillshade(painter, sourceCache, layer as any, coords);\n break;\n case 'raster':\n drawRaster(painter, sourceCache, layer as any, coords);\n break;\n case 'background':\n drawBackground(painter, sourceCache, layer as any, coords);\n break;\n case 'custom':\n drawCustom(painter, sourceCache, layer as any);\n break;\n }\n }\n\n /**\n * Transform a matrix to incorporate the *-translate and *-translate-anchor properties into it.\n * @param inViewportPixelUnitsUnits - True when the units accepted by the matrix are in viewport pixels instead of tile units.\n * @returns matrix\n */\n translatePosMatrix(matrix: mat4, tile: Tile, translate: [number, number], translateAnchor: 'map' | 'viewport', inViewportPixelUnitsUnits?: boolean): mat4 {\n if (!translate[0] && !translate[1]) return matrix;\n\n const angle = inViewportPixelUnitsUnits ?\n (translateAnchor === 'map' ? this.transform.angle : 0) :\n (translateAnchor === 'viewport' ? -this.transform.angle : 0);\n\n if (angle) {\n const sinA = Math.sin(angle);\n const cosA = Math.cos(angle);\n translate = [\n translate[0] * cosA - translate[1] * sinA,\n translate[0] * sinA + translate[1] * cosA\n ];\n }\n\n const translation = [\n inViewportPixelUnitsUnits ? translate[0] : pixelsToTileUnits(tile, translate[0], this.transform.zoom),\n inViewportPixelUnitsUnits ? translate[1] : pixelsToTileUnits(tile, translate[1], this.transform.zoom),\n 0\n ] as vec3;\n\n const translatedMatrix = new Float32Array(16);\n mat4.translate(translatedMatrix, matrix, translation);\n return translatedMatrix;\n }\n\n saveTileTexture(texture: Texture) {\n const textures = this._tileTextures[texture.size[0]];\n if (!textures) {\n this._tileTextures[texture.size[0]] = [texture];\n } else {\n textures.push(texture);\n }\n }\n\n getTileTexture(size: number) {\n const textures = this._tileTextures[size];\n return textures && textures.length > 0 ? textures.pop() : null;\n }\n\n /**\n * Checks whether a pattern image is needed, and if it is, whether it is not loaded.\n *\n * @returns true if a needed image is missing and rendering needs to be skipped.\n */\n isPatternMissing(image?: CrossFaded | null): boolean {\n if (!image) return false;\n if (!image.from || !image.to) return true;\n const imagePosA = this.imageManager.getPattern(image.from.toString());\n const imagePosB = this.imageManager.getPattern(image.to.toString());\n return !imagePosA || !imagePosB;\n }\n\n useProgram(name: string, programConfiguration?: ProgramConfiguration | null): Program {\n this.cache = this.cache || {};\n const key = name +\n (programConfiguration ? programConfiguration.cacheKey : '') +\n (this._showOverdrawInspector ? '/overdraw' : '') +\n (this.style.map.terrain ? '/terrain' : '');\n if (!this.cache[key]) {\n this.cache[key] = new Program(\n this.context,\n shaders[name],\n programConfiguration,\n programUniforms[name],\n this._showOverdrawInspector,\n this.style.map.terrain\n );\n }\n return this.cache[key];\n }\n\n /*\n * Reset some GL state to default values to avoid hard-to-debug bugs\n * in custom layers.\n */\n setCustomLayerDefaults() {\n // Prevent custom layers from unintentionally modify the last VAO used.\n // All other state is state is restored on it's own, but for VAOs it's\n // simpler to unbind so that we don't have to track the state of VAOs.\n this.context.unbindVAO();\n\n // The default values for this state is meaningful and often expected.\n // Leaving this state dirty could cause a lot of confusion for users.\n this.context.cullFace.setDefault();\n this.context.activeTexture.setDefault();\n this.context.pixelStoreUnpack.setDefault();\n this.context.pixelStoreUnpackPremultiplyAlpha.setDefault();\n this.context.pixelStoreUnpackFlipY.setDefault();\n }\n\n /*\n * Set GL state that is shared by all layers.\n */\n setBaseState() {\n const gl = this.context.gl;\n this.context.cullFace.set(false);\n this.context.viewport.set([0, 0, this.width, this.height]);\n this.context.blendEquation.set(gl.FUNC_ADD);\n }\n\n initDebugOverlayCanvas() {\n if (this.debugOverlayCanvas == null) {\n this.debugOverlayCanvas = document.createElement('canvas');\n this.debugOverlayCanvas.width = 512;\n this.debugOverlayCanvas.height = 512;\n const gl = this.context.gl;\n this.debugOverlayTexture = new Texture(this.context, this.debugOverlayCanvas, gl.RGBA);\n }\n }\n\n destroy() {\n if (this.debugOverlayTexture) {\n this.debugOverlayTexture.destroy();\n }\n }\n\n /*\n * Return true if drawing buffer size is != from requested size.\n * That means that we've reached GL limits somehow.\n * Note: drawing buffer size changes only when canvas size changes\n */\n overLimit() {\n const {drawingBufferWidth, drawingBufferHeight} = this.context.gl;\n return this.width !== drawingBufferWidth || this.height !== drawingBufferHeight;\n }\n}\n","import {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Program} from './program';\nimport {circleUniformValues} from './program/circle_program';\nimport {SegmentVector} from '../data/segment';\nimport {OverscaledTileID} from '../source/tile_id';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {CircleStyleLayer} from '../style/style_layer/circle_style_layer';\nimport type {CircleBucket} from '../data/bucket/circle_bucket';\nimport type {ProgramConfiguration} from '../data/program_configuration';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {IndexBuffer} from '../gl/index_buffer';\nimport type {UniformValues} from './uniform_binding';\nimport type {CircleUniformsType} from './program/circle_program';\nimport type {TerrainData} from '../render/terrain';\n\ntype TileRenderState = {\n programConfiguration: ProgramConfiguration;\n program: Program;\n layoutVertexBuffer: VertexBuffer;\n indexBuffer: IndexBuffer;\n uniformValues: UniformValues;\n terrainData: TerrainData;\n};\n\ntype SegmentsTileRenderState = {\n segments: SegmentVector;\n sortKey: number;\n state: TileRenderState;\n};\n\nexport function drawCircles(painter: Painter, sourceCache: SourceCache, layer: CircleStyleLayer, coords: Array) {\n if (painter.renderPass !== 'translucent') return;\n\n const opacity = layer.paint.get('circle-opacity');\n const strokeWidth = layer.paint.get('circle-stroke-width');\n const strokeOpacity = layer.paint.get('circle-stroke-opacity');\n const sortFeaturesByKey = !layer.layout.get('circle-sort-key').isConstant();\n\n if (opacity.constantOr(1) === 0 && (strokeWidth.constantOr(1) === 0 || strokeOpacity.constantOr(1) === 0)) {\n return;\n }\n\n const context = painter.context;\n const gl = context.gl;\n\n const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n // Turn off stencil testing to allow circles to be drawn across boundaries,\n // so that large circles are not clipped to tiles\n const stencilMode = StencilMode.disabled;\n const colorMode = painter.colorModeForRenderPass();\n\n const segmentsRenderStates: Array = [];\n\n for (let i = 0; i < coords.length; i++) {\n const coord = coords[i];\n\n const tile = sourceCache.getTile(coord);\n const bucket: CircleBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const program = painter.useProgram('circle', programConfiguration);\n const layoutVertexBuffer = bucket.layoutVertexBuffer;\n const indexBuffer = bucket.indexBuffer;\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n const uniformValues = circleUniformValues(painter, coord, tile, layer);\n\n const state: TileRenderState = {\n programConfiguration,\n program,\n layoutVertexBuffer,\n indexBuffer,\n uniformValues,\n terrainData\n };\n\n if (sortFeaturesByKey) {\n const oldSegments = bucket.segments.get();\n for (const segment of oldSegments) {\n segmentsRenderStates.push({\n segments: new SegmentVector([segment]),\n sortKey: (segment.sortKey as any as number),\n state\n });\n }\n } else {\n segmentsRenderStates.push({\n segments: bucket.segments,\n sortKey: 0,\n state\n });\n }\n\n }\n\n if (sortFeaturesByKey) {\n segmentsRenderStates.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const segmentsState of segmentsRenderStates) {\n const {programConfiguration, program, layoutVertexBuffer, indexBuffer, uniformValues, terrainData} = segmentsState.state;\n const segments = segmentsState.segments;\n\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id,\n layoutVertexBuffer, indexBuffer, segments,\n layer.paint, painter.transform.zoom, programConfiguration);\n }\n}\n","import {Texture} from './texture';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Context} from '../gl/context';\nimport {Framebuffer} from '../gl/framebuffer';\nimport {\n heatmapUniformValues,\n heatmapTextureUniformValues\n} from './program/heatmap_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {HeatmapStyleLayer} from '../style/style_layer/heatmap_style_layer';\nimport type {HeatmapBucket} from '../data/bucket/heatmap_bucket';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nexport function drawHeatmap(painter: Painter, sourceCache: SourceCache, layer: HeatmapStyleLayer, coords: Array) {\n if (layer.paint.get('heatmap-opacity') === 0) {\n return;\n }\n\n if (painter.renderPass === 'offscreen') {\n const context = painter.context;\n const gl = context.gl;\n\n // Allow kernels to be drawn across boundaries, so that\n // large kernels are not clipped to tiles\n const stencilMode = StencilMode.disabled;\n // Turn on additive blending for kernels, which is a key aspect of kernel density estimation formula\n const colorMode = new ColorMode([gl.ONE, gl.ONE], Color.transparent, [true, true, true, true]);\n\n bindFramebuffer(context, painter, layer);\n\n context.clear({color: Color.transparent});\n\n for (let i = 0; i < coords.length; i++) {\n const coord = coords[i];\n\n // Skip tiles that have uncovered parents to avoid flickering; we don't need\n // to use complex tile masking here because the change between zoom levels is subtle,\n // so it's fine to simply render the parent until all its 4 children are loaded\n if (sourceCache.hasRenderableParent(coord)) continue;\n\n const tile = sourceCache.getTile(coord);\n const bucket: HeatmapBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const program = painter.useProgram('heatmap', programConfiguration);\n const {zoom} = painter.transform;\n\n program.draw(context, gl.TRIANGLES, DepthMode.disabled, stencilMode, colorMode, CullFaceMode.disabled,\n heatmapUniformValues(coord.posMatrix, tile, zoom, layer.paint.get('heatmap-intensity')), null,\n layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer,\n bucket.segments, layer.paint, painter.transform.zoom,\n programConfiguration);\n }\n\n context.viewport.set([0, 0, painter.width, painter.height]);\n\n } else if (painter.renderPass === 'translucent') {\n painter.context.setColorMode(painter.colorModeForRenderPass());\n renderTextureToMap(painter, layer);\n }\n}\n\nfunction bindFramebuffer(context: Context, painter: Painter, layer: HeatmapStyleLayer) {\n const gl = context.gl;\n context.activeTexture.set(gl.TEXTURE1);\n\n // Use a 4x downscaled screen texture for better performance\n context.viewport.set([0, 0, painter.width / 4, painter.height / 4]);\n\n let fbo = layer.heatmapFbo;\n\n if (!fbo) {\n const texture = gl.createTexture();\n gl.bindTexture(gl.TEXTURE_2D, texture);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n\n fbo = layer.heatmapFbo = context.createFramebuffer(painter.width / 4, painter.height / 4, false, false);\n\n bindTextureToFramebuffer(context, painter, texture, fbo);\n\n } else {\n gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get());\n context.bindFramebuffer.set(fbo.framebuffer);\n }\n}\n\nfunction bindTextureToFramebuffer(context: Context, painter: Painter, texture: WebGLTexture, fbo: Framebuffer) {\n const gl = context.gl;\n // Use the higher precision half-float texture where available (producing much smoother looking heatmaps);\n // Otherwise, fall back to a low precision texture\n\n const numType = context.HALF_FLOAT ?? gl.UNSIGNED_BYTE;\n const internalFormat = context.RGBA16F ?? gl.RGBA;\n\n gl.texImage2D(gl.TEXTURE_2D, 0, internalFormat, painter.width / 4, painter.height / 4, 0, gl.RGBA, numType, null);\n fbo.colorAttachment.set(texture);\n}\n\nfunction renderTextureToMap(painter: Painter, layer: HeatmapStyleLayer) {\n const context = painter.context;\n const gl = context.gl;\n\n // Here we bind two different textures from which we'll sample in drawing\n // heatmaps: the kernel texture, prepared in the offscreen pass, and a\n // color ramp texture.\n const fbo = layer.heatmapFbo;\n if (!fbo) return;\n context.activeTexture.set(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get());\n\n context.activeTexture.set(gl.TEXTURE1);\n let colorRampTexture = layer.colorRampTexture;\n if (!colorRampTexture) {\n colorRampTexture = layer.colorRampTexture = new Texture(context, layer.colorRamp, gl.RGBA);\n }\n colorRampTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n\n painter.useProgram('heatmapTexture').draw(context, gl.TRIANGLES,\n DepthMode.disabled, StencilMode.disabled, painter.colorModeForRenderPass(), CullFaceMode.disabled,\n heatmapTextureUniformValues(painter, layer, 0, 1), null,\n layer.id, painter.viewportBuffer, painter.quadTriangleIndexBuffer,\n painter.viewportSegments, layer.paint, painter.transform.zoom);\n}\n","import {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Texture} from './texture';\nimport {\n lineUniformValues,\n linePatternUniformValues,\n lineSDFUniformValues,\n lineGradientUniformValues\n} from './program/line_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {LineStyleLayer} from '../style/style_layer/line_style_layer';\nimport type {LineBucket} from '../data/bucket/line_bucket';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport {clamp, nextPowerOfTwo} from '../util/util';\nimport {renderColorRamp} from '../util/color_ramp';\nimport {EXTENT} from '../data/extent';\n\nexport function drawLine(painter: Painter, sourceCache: SourceCache, layer: LineStyleLayer, coords: Array) {\n if (painter.renderPass !== 'translucent') return;\n\n const opacity = layer.paint.get('line-opacity');\n const width = layer.paint.get('line-width');\n if (opacity.constantOr(1) === 0 || width.constantOr(1) === 0) return;\n\n const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n const colorMode = painter.colorModeForRenderPass();\n\n const dasharray = layer.paint.get('line-dasharray');\n const patternProperty = layer.paint.get('line-pattern');\n const image = patternProperty.constantOr(1 as any);\n\n const gradient = layer.paint.get('line-gradient');\n const crossfade = layer.getCrossfadeParameters();\n\n const programId =\n image ? 'linePattern' :\n dasharray ? 'lineSDF' :\n gradient ? 'lineGradient' : 'line';\n\n const context = painter.context;\n const gl = context.gl;\n\n let firstTile = true;\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n\n if (image && !tile.patternsLoaded()) continue;\n\n const bucket: LineBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const prevProgram = painter.context.program.get();\n const program = painter.useProgram(programId, programConfiguration);\n const programChanged = firstTile || program.program !== prevProgram;\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n const constantPattern = patternProperty.constantOr(null);\n if (constantPattern && tile.imageAtlas) {\n const atlas = tile.imageAtlas;\n const posTo = atlas.patternPositions[constantPattern.to.toString()];\n const posFrom = atlas.patternPositions[constantPattern.from.toString()];\n if (posTo && posFrom) programConfiguration.setConstantPatternPositions(posTo, posFrom);\n }\n\n const terrainCoord = terrainData ? coord : null;\n const uniformValues = image ? linePatternUniformValues(painter, tile, layer, crossfade, terrainCoord) :\n dasharray ? lineSDFUniformValues(painter, tile, layer, dasharray, crossfade, terrainCoord) :\n gradient ? lineGradientUniformValues(painter, tile, layer, bucket.lineClipsArray.length, terrainCoord) :\n lineUniformValues(painter, tile, layer, terrainCoord);\n\n if (image) {\n context.activeTexture.set(gl.TEXTURE0);\n tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n programConfiguration.updatePaintBuffers(crossfade);\n } else if (dasharray && (programChanged || painter.lineAtlas.dirty)) {\n context.activeTexture.set(gl.TEXTURE0);\n painter.lineAtlas.bind(context);\n } else if (gradient) {\n const layerGradient = bucket.gradients[layer.id];\n let gradientTexture = layerGradient.texture;\n if (layer.gradientVersion !== layerGradient.version) {\n let textureResolution = 256;\n if (layer.stepInterpolant) {\n const sourceMaxZoom = sourceCache.getSource().maxzoom;\n const potentialOverzoom = coord.canonical.z === sourceMaxZoom ?\n Math.ceil(1 << (painter.transform.maxZoom - coord.canonical.z)) : 1;\n const lineLength = bucket.maxLineLength / EXTENT;\n // Logical pixel tile size is 512px, and 1024px right before current zoom + 1\n const maxTilePixelSize = 1024;\n // Maximum possible texture coverage heuristic, bound by hardware max texture size\n const maxTextureCoverage = lineLength * maxTilePixelSize * potentialOverzoom;\n textureResolution = clamp(nextPowerOfTwo(maxTextureCoverage), 256, context.maxTextureSize);\n }\n layerGradient.gradient = renderColorRamp({\n expression: layer.gradientExpression(),\n evaluationKey: 'lineProgress',\n resolution: textureResolution,\n image: layerGradient.gradient || undefined,\n clips: bucket.lineClipsArray\n });\n if (layerGradient.texture) {\n layerGradient.texture.update(layerGradient.gradient);\n } else {\n layerGradient.texture = new Texture(context, layerGradient.gradient, gl.RGBA);\n }\n layerGradient.version = layer.gradientVersion;\n gradientTexture = layerGradient.texture;\n }\n context.activeTexture.set(gl.TEXTURE0);\n gradientTexture.bind(layer.stepInterpolant ? gl.NEAREST : gl.LINEAR, gl.CLAMP_TO_EDGE);\n }\n\n program.draw(context, gl.TRIANGLES, depthMode,\n painter.stencilModeForClipping(coord), colorMode, CullFaceMode.disabled, uniformValues, terrainData,\n layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer, bucket.segments,\n layer.paint, painter.transform.zoom, programConfiguration, bucket.layoutVertexBuffer2);\n\n firstTile = false;\n // once refactored so that bound texture state is managed, we'll also be able to remove this firstTile/programChanged logic\n }\n}\n","import {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {\n backgroundUniformValues,\n backgroundPatternUniformValues\n} from './program/background_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {BackgroundStyleLayer} from '../style/style_layer/background_style_layer';\nimport {OverscaledTileID} from '../source/tile_id';\n\nexport function drawBackground(painter: Painter, sourceCache: SourceCache, layer: BackgroundStyleLayer, coords?: Array) {\n const color = layer.paint.get('background-color');\n const opacity = layer.paint.get('background-opacity');\n\n if (opacity === 0) return;\n\n const context = painter.context;\n const gl = context.gl;\n const transform = painter.transform;\n const tileSize = transform.tileSize;\n const image = layer.paint.get('background-pattern');\n if (painter.isPatternMissing(image)) return;\n\n const pass = (!image && color.a === 1 && opacity === 1 && painter.opaquePassEnabledForLayer()) ? 'opaque' : 'translucent';\n if (painter.renderPass !== pass) return;\n\n const stencilMode = StencilMode.disabled;\n const depthMode = painter.depthModeForSublayer(0, pass === 'opaque' ? DepthMode.ReadWrite : DepthMode.ReadOnly);\n const colorMode = painter.colorModeForRenderPass();\n const program = painter.useProgram(image ? 'backgroundPattern' : 'background');\n const tileIDs = coords ? coords : transform.coveringTiles({tileSize, terrain: painter.style.map.terrain});\n\n if (image) {\n context.activeTexture.set(gl.TEXTURE0);\n painter.imageManager.bind(painter.context);\n }\n\n const crossfade = layer.getCrossfadeParameters();\n for (const tileID of tileIDs) {\n const matrix = coords ? tileID.posMatrix : painter.transform.calculatePosMatrix(tileID.toUnwrapped());\n const uniformValues = image ?\n backgroundPatternUniformValues(matrix, opacity, painter, image, {tileID, tileSize}, crossfade) :\n backgroundUniformValues(matrix, opacity, color);\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(tileID);\n\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id, painter.tileExtentBuffer,\n painter.quadTriangleIndexBuffer, painter.tileExtentSegments);\n }\n}\n","import {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {CustomStyleLayer} from '../style/style_layer/custom_style_layer';\n\nexport function drawCustom(painter: Painter, sourceCache: SourceCache, layer: CustomStyleLayer) {\n\n const context = painter.context;\n const implementation = layer.implementation;\n\n if (painter.renderPass === 'offscreen') {\n\n const prerender = implementation.prerender;\n if (prerender) {\n painter.setCustomLayerDefaults();\n context.setColorMode(painter.colorModeForRenderPass());\n\n prerender.call(implementation, context.gl, painter.transform.customLayerMatrix());\n\n context.setDirty();\n painter.setBaseState();\n }\n\n } else if (painter.renderPass === 'translucent') {\n\n painter.setCustomLayerDefaults();\n\n context.setColorMode(painter.colorModeForRenderPass());\n context.setStencilMode(StencilMode.disabled);\n\n const depthMode = implementation.renderingMode === '3d' ?\n new DepthMode(painter.context.gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D) :\n painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n\n context.setDepthMode(depthMode);\n\n implementation.render(context.gl, painter.transform.customLayerMatrix());\n\n context.setDirty();\n painter.setBaseState();\n context.bindFramebuffer.set(null);\n }\n}\n","import {mat4, vec3, vec4} from 'gl-matrix';\n\nclass Frustum {\n\n constructor(public points: vec4[], public planes: vec4[]) { }\n\n public static fromInvProjectionMatrix(invProj: mat4, worldSize: number, zoom: number): Frustum {\n const clipSpaceCorners = [\n [-1, 1, -1, 1],\n [1, 1, -1, 1],\n [1, -1, -1, 1],\n [-1, -1, -1, 1],\n [-1, 1, 1, 1],\n [1, 1, 1, 1],\n [1, -1, 1, 1],\n [-1, -1, 1, 1]\n ];\n\n const scale = Math.pow(2, zoom);\n\n // Transform frustum corner points from clip space to tile space, Z to meters\n const frustumCoords = clipSpaceCorners.map(v => {\n v = vec4.transformMat4([] as any, v as any, invProj) as any;\n const s = 1.0 / v[3] / worldSize * scale;\n return vec4.mul(v as any, v as any, [s, s, 1.0 / v[3], s] as vec4);\n });\n\n const frustumPlanePointIndices = [\n [0, 1, 2], // near\n [6, 5, 4], // far\n [0, 3, 7], // left\n [2, 1, 5], // right\n [3, 2, 6], // bottom\n [0, 4, 5] // top\n ];\n\n const frustumPlanes = frustumPlanePointIndices.map((p: number[]) => {\n const a = vec3.sub([] as any, frustumCoords[p[0]] as vec3, frustumCoords[p[1]] as vec3);\n const b = vec3.sub([] as any, frustumCoords[p[2]] as vec3, frustumCoords[p[1]] as vec3);\n const n = vec3.normalize([] as any, vec3.cross([] as any, a, b)) as any;\n const d = -vec3.dot(n, frustumCoords[p[1]] as vec3);\n return n.concat(d);\n });\n\n return new Frustum(frustumCoords, frustumPlanes);\n }\n}\n\nclass Aabb {\n min: vec3;\n max: vec3;\n center: vec3;\n\n constructor(min_: vec3, max_: vec3) {\n this.min = min_;\n this.max = max_;\n this.center = vec3.scale([] as any, vec3.add([] as any, this.min, this.max), 0.5);\n }\n\n quadrant(index: number): Aabb {\n const split = [(index % 2) === 0, index < 2];\n const qMin = vec3.clone(this.min);\n const qMax = vec3.clone(this.max);\n for (let axis = 0; axis < split.length; axis++) {\n qMin[axis] = split[axis] ? this.min[axis] : this.center[axis];\n qMax[axis] = split[axis] ? this.center[axis] : this.max[axis];\n }\n // Elevation is always constant, hence quadrant.max.z = this.max.z\n qMax[2] = this.max[2];\n return new Aabb(qMin, qMax);\n }\n\n distanceX(point: Array): number {\n const pointOnAabb = Math.max(Math.min(this.max[0], point[0]), this.min[0]);\n return pointOnAabb - point[0];\n }\n\n distanceY(point: Array): number {\n const pointOnAabb = Math.max(Math.min(this.max[1], point[1]), this.min[1]);\n return pointOnAabb - point[1];\n }\n\n // Performs a frustum-aabb intersection test. Returns 0 if there's no intersection,\n // 1 if shapes are intersecting and 2 if the aabb if fully inside the frustum.\n intersects(frustum: Frustum): number {\n // Execute separating axis test between two convex objects to find intersections\n // Each frustum plane together with 3 major axes define the separating axes\n\n const aabbPoints = [\n [this.min[0], this.min[1], this.min[2], 1],\n [this.max[0], this.min[1], this.min[2], 1],\n [this.max[0], this.max[1], this.min[2], 1],\n [this.min[0], this.max[1], this.min[2], 1],\n [this.min[0], this.min[1], this.max[2], 1],\n [this.max[0], this.min[1], this.max[2], 1],\n [this.max[0], this.max[1], this.max[2], 1],\n [this.min[0], this.max[1], this.max[2], 1]\n ];\n\n let fullyInside = true;\n\n for (let p = 0; p < frustum.planes.length; p++) {\n const plane = frustum.planes[p];\n let pointsInside = 0;\n\n for (let i = 0; i < aabbPoints.length; i++) {\n if (vec4.dot(plane, aabbPoints[i] as any) >= 0) {\n pointsInside++;\n }\n }\n\n if (pointsInside === 0)\n return 0;\n\n if (pointsInside !== aabbPoints.length)\n fullyInside = false;\n }\n\n if (fullyInside)\n return 2;\n\n for (let axis = 0; axis < 3; axis++) {\n let projMin = Number.MAX_VALUE;\n let projMax = -Number.MAX_VALUE;\n\n for (let p = 0; p < frustum.points.length; p++) {\n const projectedPoint = frustum.points[p][axis] - this.min[axis];\n\n projMin = Math.min(projMin, projectedPoint);\n projMax = Math.max(projMax, projectedPoint);\n }\n\n if (projMax < 0 || projMin > this.max[axis] - this.min[axis])\n return 0;\n }\n\n return 1;\n }\n}\nexport {\n Aabb,\n Frustum\n};\n","import {interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport Point from '@mapbox/point-geometry';\nimport {clamp} from '../util/util';\n\n/**\n * An `EdgeInset` object represents screen space padding applied to the edges of the viewport.\n * This shifts the apprent center or the vanishing point of the map. This is useful for adding floating UI elements\n * on top of the map and having the vanishing point shift as UI elements resize.\n *\n * @group Geography and Geometry\n */\nexport class EdgeInsets {\n /**\n * @defaultValue 0\n */\n top: number;\n /**\n * @defaultValue 0\n */\n bottom: number;\n /**\n * @defaultValue 0\n */\n left: number;\n /**\n * @defaultValue 0\n */\n right: number;\n\n constructor(top: number = 0, bottom: number = 0, left: number = 0, right: number = 0) {\n if (isNaN(top) || top < 0 ||\n isNaN(bottom) || bottom < 0 ||\n isNaN(left) || left < 0 ||\n isNaN(right) || right < 0\n ) {\n throw new Error('Invalid value for edge-insets, top, bottom, left and right must all be numbers');\n }\n\n this.top = top;\n this.bottom = bottom;\n this.left = left;\n this.right = right;\n }\n\n /**\n * Interpolates the inset in-place.\n * This maintains the current inset value for any inset not present in `target`.\n * @param start - interpolation start\n * @param target - interpolation target\n * @param t - interpolation step/weight\n * @returns the insets\n */\n interpolate(start: PaddingOptions | EdgeInsets, target: PaddingOptions, t: number): EdgeInsets {\n if (target.top != null && start.top != null) this.top = interpolates.number(start.top, target.top, t);\n if (target.bottom != null && start.bottom != null) this.bottom = interpolates.number(start.bottom, target.bottom, t);\n if (target.left != null && start.left != null) this.left = interpolates.number(start.left, target.left, t);\n if (target.right != null && start.right != null) this.right = interpolates.number(start.right, target.right, t);\n\n return this;\n }\n\n /**\n * Utility method that computes the new apprent center or vanishing point after applying insets.\n * This is in pixels and with the top left being (0.0) and +y being downwards.\n *\n * @param width - the width\n * @param height - the height\n * @returns the point\n */\n getCenter(width: number, height: number): Point {\n // Clamp insets so they never overflow width/height and always calculate a valid center\n const x = clamp((this.left + width - this.right) / 2, 0, width);\n const y = clamp((this.top + height - this.bottom) / 2, 0, height);\n\n return new Point(x, y);\n }\n\n equals(other: PaddingOptions): boolean {\n return this.top === other.top &&\n this.bottom === other.bottom &&\n this.left === other.left &&\n this.right === other.right;\n }\n\n clone(): EdgeInsets {\n return new EdgeInsets(this.top, this.bottom, this.left, this.right);\n }\n\n /**\n * Returns the current state as json, useful when you want to have a\n * read-only representation of the inset.\n *\n * @returns state as json\n */\n toJSON(): PaddingOptions {\n return {\n top: this.top,\n bottom: this.bottom,\n left: this.left,\n right: this.right\n };\n }\n}\n\n/**\n * Options for setting padding on calls to methods such as {@link Map#fitBounds}, {@link Map#fitScreenCoordinates}, and {@link Map#setPadding}. Adjust these options to set the amount of padding in pixels added to the edges of the canvas. Set a uniform padding on all edges or individual values for each edge. All properties of this object must be\n * non-negative integers.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let bbox = [[-79, 43], [-73, 45]];\n * map.fitBounds(bbox, {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n *\n * @example\n * ```ts\n * let bbox = [[-79, 43], [-73, 45]];\n * map.fitBounds(bbox, {\n * padding: 20\n * });\n * ```\n * @see [Fit to the bounds of a LineString](https://maplibre.org/maplibre-gl-js/docs/examples/zoomto-linestring/)\n * @see [Fit a map to a bounding box](https://maplibre.org/maplibre-gl-js/docs/examples/fitbounds/)\n */\nexport type PaddingOptions = {\n /**\n * Padding in pixels from the top of the map canvas.\n */\n top: number;\n /**\n * Padding in pixels from the bottom of the map canvas.\n */\n bottom: number;\n /**\n * Padding in pixels from the left of the map canvas.\n */\n right: number;\n /**\n * Padding in pixels from the right of the map canvas.\n */\n left: number;\n};\n","import {LngLat} from './lng_lat';\nimport {LngLatBounds} from './lng_lat_bounds';\nimport {MercatorCoordinate, mercatorXfromLng, mercatorYfromLat, mercatorZfromAltitude} from './mercator_coordinate';\nimport Point from '@mapbox/point-geometry';\nimport {wrap, clamp} from '../util/util';\nimport {interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {EXTENT} from '../data/extent';\nimport {vec3, vec4, mat4, mat2, vec2} from 'gl-matrix';\nimport {Aabb, Frustum} from '../util/primitives';\nimport {EdgeInsets} from './edge_insets';\n\nimport {UnwrappedTileID, OverscaledTileID, CanonicalTileID} from '../source/tile_id';\nimport type {PaddingOptions} from './edge_insets';\nimport {Terrain} from '../render/terrain';\n\n/**\n * @internal\n * A single transform, generally used for a single tile to be\n * scaled, rotated, and zoomed.\n */\nexport class Transform {\n tileSize: number;\n tileZoom: number;\n lngRange: [number, number];\n latRange: [number, number];\n maxValidLatitude: number;\n scale: number;\n width: number;\n height: number;\n angle: number;\n rotationMatrix: mat2;\n pixelsToGLUnits: [number, number];\n cameraToCenterDistance: number;\n mercatorMatrix: mat4;\n projMatrix: mat4;\n invProjMatrix: mat4;\n alignedProjMatrix: mat4;\n pixelMatrix: mat4;\n pixelMatrix3D: mat4;\n pixelMatrixInverse: mat4;\n glCoordMatrix: mat4;\n labelPlaneMatrix: mat4;\n _fov: number;\n _pitch: number;\n _zoom: number;\n _unmodified: boolean;\n _renderWorldCopies: boolean;\n _minZoom: number;\n _maxZoom: number;\n _minPitch: number;\n _maxPitch: number;\n _center: LngLat;\n _elevation: number;\n _pixelPerMeter: number;\n _edgeInsets: EdgeInsets;\n _constraining: boolean;\n _posMatrixCache: {[_: string]: mat4};\n _alignedPosMatrixCache: {[_: string]: mat4};\n _minEleveationForCurrentTile: number;\n\n constructor(minZoom?: number, maxZoom?: number, minPitch?: number, maxPitch?: number, renderWorldCopies?: boolean) {\n this.tileSize = 512; // constant\n this.maxValidLatitude = 85.051129; // constant\n\n this._renderWorldCopies = renderWorldCopies === undefined ? true : !!renderWorldCopies;\n this._minZoom = minZoom || 0;\n this._maxZoom = maxZoom || 22;\n\n this._minPitch = (minPitch === undefined || minPitch === null) ? 0 : minPitch;\n this._maxPitch = (maxPitch === undefined || maxPitch === null) ? 60 : maxPitch;\n\n this.setMaxBounds();\n\n this.width = 0;\n this.height = 0;\n this._center = new LngLat(0, 0);\n this._elevation = 0;\n this.zoom = 0;\n this.angle = 0;\n this._fov = 0.6435011087932844;\n this._pitch = 0;\n this._unmodified = true;\n this._edgeInsets = new EdgeInsets();\n this._posMatrixCache = {};\n this._alignedPosMatrixCache = {};\n this._minEleveationForCurrentTile = 0;\n }\n\n clone(): Transform {\n const clone = new Transform(this._minZoom, this._maxZoom, this._minPitch, this.maxPitch, this._renderWorldCopies);\n clone.apply(this);\n return clone;\n }\n\n apply(that: Transform) {\n this.tileSize = that.tileSize;\n this.latRange = that.latRange;\n this.width = that.width;\n this.height = that.height;\n this._center = that._center;\n this._elevation = that._elevation;\n this._minEleveationForCurrentTile = that._minEleveationForCurrentTile;\n this.zoom = that.zoom;\n this.angle = that.angle;\n this._fov = that._fov;\n this._pitch = that._pitch;\n this._unmodified = that._unmodified;\n this._edgeInsets = that._edgeInsets.clone();\n this._calcMatrices();\n }\n\n get minZoom(): number { return this._minZoom; }\n set minZoom(zoom: number) {\n if (this._minZoom === zoom) return;\n this._minZoom = zoom;\n this.zoom = Math.max(this.zoom, zoom);\n }\n\n get maxZoom(): number { return this._maxZoom; }\n set maxZoom(zoom: number) {\n if (this._maxZoom === zoom) return;\n this._maxZoom = zoom;\n this.zoom = Math.min(this.zoom, zoom);\n }\n\n get minPitch(): number { return this._minPitch; }\n set minPitch(pitch: number) {\n if (this._minPitch === pitch) return;\n this._minPitch = pitch;\n this.pitch = Math.max(this.pitch, pitch);\n }\n\n get maxPitch(): number { return this._maxPitch; }\n set maxPitch(pitch: number) {\n if (this._maxPitch === pitch) return;\n this._maxPitch = pitch;\n this.pitch = Math.min(this.pitch, pitch);\n }\n\n get renderWorldCopies(): boolean { return this._renderWorldCopies; }\n set renderWorldCopies(renderWorldCopies: boolean) {\n if (renderWorldCopies === undefined) {\n renderWorldCopies = true;\n } else if (renderWorldCopies === null) {\n renderWorldCopies = false;\n }\n\n this._renderWorldCopies = renderWorldCopies;\n }\n\n get worldSize(): number {\n return this.tileSize * this.scale;\n }\n\n get centerOffset(): Point {\n return this.centerPoint._sub(this.size._div(2));\n }\n\n get size(): Point {\n return new Point(this.width, this.height);\n }\n\n get bearing(): number {\n return -this.angle / Math.PI * 180;\n }\n set bearing(bearing: number) {\n const b = -wrap(bearing, -180, 180) * Math.PI / 180;\n if (this.angle === b) return;\n this._unmodified = false;\n this.angle = b;\n this._calcMatrices();\n\n // 2x2 matrix for rotating points\n this.rotationMatrix = mat2.create();\n mat2.rotate(this.rotationMatrix, this.rotationMatrix, this.angle);\n }\n\n get pitch(): number {\n return this._pitch / Math.PI * 180;\n }\n set pitch(pitch: number) {\n const p = clamp(pitch, this.minPitch, this.maxPitch) / 180 * Math.PI;\n if (this._pitch === p) return;\n this._unmodified = false;\n this._pitch = p;\n this._calcMatrices();\n }\n\n get fov(): number {\n return this._fov / Math.PI * 180;\n }\n set fov(fov: number) {\n fov = Math.max(0.01, Math.min(60, fov));\n if (this._fov === fov) return;\n this._unmodified = false;\n this._fov = fov / 180 * Math.PI;\n this._calcMatrices();\n }\n\n get zoom(): number { return this._zoom; }\n set zoom(zoom: number) {\n const constrainedZoom = Math.min(Math.max(zoom, this.minZoom), this.maxZoom);\n if (this._zoom === constrainedZoom) return;\n this._unmodified = false;\n this._zoom = constrainedZoom;\n this.tileZoom = Math.max(0, Math.floor(constrainedZoom));\n this.scale = this.zoomScale(constrainedZoom);\n this._constrain();\n this._calcMatrices();\n }\n\n get center(): LngLat { return this._center; }\n set center(center: LngLat) {\n if (center.lat === this._center.lat && center.lng === this._center.lng) return;\n this._unmodified = false;\n this._center = center;\n this._constrain();\n this._calcMatrices();\n }\n\n get elevation(): number { return this._elevation; }\n set elevation(elevation: number) {\n if (elevation === this._elevation) return;\n this._elevation = elevation;\n this._constrain();\n this._calcMatrices();\n }\n\n get padding(): PaddingOptions { return this._edgeInsets.toJSON(); }\n set padding(padding: PaddingOptions) {\n if (this._edgeInsets.equals(padding)) return;\n this._unmodified = false;\n //Update edge-insets inplace\n this._edgeInsets.interpolate(this._edgeInsets, padding, 1);\n this._calcMatrices();\n }\n\n /**\n * The center of the screen in pixels with the top-left corner being (0,0)\n * and +y axis pointing downwards. This accounts for padding.\n */\n get centerPoint(): Point {\n return this._edgeInsets.getCenter(this.width, this.height);\n }\n\n /**\n * Returns if the padding params match\n *\n * @param padding - the padding to check against\n * @returns true if they are equal, false otherwise\n */\n isPaddingEqual(padding: PaddingOptions): boolean {\n return this._edgeInsets.equals(padding);\n }\n\n /**\n * Helper method to update edge-insets in place\n *\n * @param start - the starting padding\n * @param target - the target padding\n * @param t - the step/weight\n */\n interpolatePadding(start: PaddingOptions, target: PaddingOptions, t: number) {\n this._unmodified = false;\n this._edgeInsets.interpolate(start, target, t);\n this._constrain();\n this._calcMatrices();\n }\n\n /**\n * Return a zoom level that will cover all tiles the transform\n * @param options - the options\n * @returns zoom level An integer zoom level at which all tiles will be visible.\n */\n coveringZoomLevel(options: {\n /**\n * Target zoom level. If true, the value will be rounded to the closest integer. Otherwise the value will be floored.\n */\n roundZoom?: boolean;\n /**\n * Tile size, expressed in screen pixels.\n */\n tileSize: number;\n }): number {\n const z = (options.roundZoom ? Math.round : Math.floor)(\n this.zoom + this.scaleZoom(this.tileSize / options.tileSize)\n );\n // At negative zoom levels load tiles from z0 because negative tile zoom levels don't exist.\n return Math.max(0, z);\n }\n\n /**\n * Return any \"wrapped\" copies of a given tile coordinate that are visible\n * in the current view.\n */\n getVisibleUnwrappedCoordinates(tileID: CanonicalTileID) {\n const result = [new UnwrappedTileID(0, tileID)];\n if (this._renderWorldCopies) {\n const utl = this.pointCoordinate(new Point(0, 0));\n const utr = this.pointCoordinate(new Point(this.width, 0));\n const ubl = this.pointCoordinate(new Point(this.width, this.height));\n const ubr = this.pointCoordinate(new Point(0, this.height));\n const w0 = Math.floor(Math.min(utl.x, utr.x, ubl.x, ubr.x));\n const w1 = Math.floor(Math.max(utl.x, utr.x, ubl.x, ubr.x));\n\n // Add an extra copy of the world on each side to properly render ImageSources and CanvasSources.\n // Both sources draw outside the tile boundaries of the tile that \"contains them\" so we need\n // to add extra copies on both sides in case offscreen tiles need to draw into on-screen ones.\n const extraWorldCopy = 1;\n\n for (let w = w0 - extraWorldCopy; w <= w1 + extraWorldCopy; w++) {\n if (w === 0) continue;\n result.push(new UnwrappedTileID(w, tileID));\n }\n }\n return result;\n }\n\n /**\n * Return all coordinates that could cover this transform for a covering\n * zoom level.\n * @param options - the options\n * @returns OverscaledTileIDs\n */\n coveringTiles(\n options: {\n tileSize: number;\n minzoom?: number;\n maxzoom?: number;\n roundZoom?: boolean;\n reparseOverscaled?: boolean;\n renderWorldCopies?: boolean;\n terrain?: Terrain;\n }\n ): Array {\n let z = this.coveringZoomLevel(options);\n const actualZ = z;\n\n if (options.minzoom !== undefined && z < options.minzoom) return [];\n if (options.maxzoom !== undefined && z > options.maxzoom) z = options.maxzoom;\n\n const cameraCoord = this.pointCoordinate(this.getCameraPoint());\n const centerCoord = MercatorCoordinate.fromLngLat(this.center);\n const numTiles = Math.pow(2, z);\n const cameraPoint = [numTiles * cameraCoord.x, numTiles * cameraCoord.y, 0];\n const centerPoint = [numTiles * centerCoord.x, numTiles * centerCoord.y, 0];\n const cameraFrustum = Frustum.fromInvProjectionMatrix(this.invProjMatrix, this.worldSize, z);\n\n // No change of LOD behavior for pitch lower than 60 and when there is no top padding: return only tile ids from the requested zoom level\n let minZoom = options.minzoom || 0;\n // Use 0.1 as an epsilon to avoid for explicit == 0.0 floating point checks\n if (!options.terrain && this.pitch <= 60.0 && this._edgeInsets.top < 0.1)\n minZoom = z;\n\n // There should always be a certain number of maximum zoom level tiles surrounding the center location in 2D or in front of the camera in 3D\n const radiusOfMaxLvlLodInTiles = options.terrain ? 2 / Math.min(this.tileSize, options.tileSize) * this.tileSize : 3;\n\n const newRootTile = (wrap: number): any => {\n return {\n aabb: new Aabb([wrap * numTiles, 0, 0], [(wrap + 1) * numTiles, numTiles, 0]),\n zoom: 0,\n x: 0,\n y: 0,\n wrap,\n fullyVisible: false\n };\n };\n\n // Do a depth-first traversal to find visible tiles and proper levels of detail\n const stack = [];\n const result = [];\n const maxZoom = z;\n const overscaledZ = options.reparseOverscaled ? actualZ : z;\n\n if (this._renderWorldCopies) {\n // Render copy of the globe thrice on both sides\n for (let i = 1; i <= 3; i++) {\n stack.push(newRootTile(-i));\n stack.push(newRootTile(i));\n }\n }\n\n stack.push(newRootTile(0));\n\n while (stack.length > 0) {\n const it = stack.pop();\n const x = it.x;\n const y = it.y;\n let fullyVisible = it.fullyVisible;\n\n // Visibility of a tile is not required if any of its ancestor if fully inside the frustum\n if (!fullyVisible) {\n const intersectResult = it.aabb.intersects(cameraFrustum);\n\n if (intersectResult === 0)\n continue;\n\n fullyVisible = intersectResult === 2;\n }\n\n const refPoint = options.terrain ? cameraPoint : centerPoint;\n const distanceX = it.aabb.distanceX(refPoint);\n const distanceY = it.aabb.distanceY(refPoint);\n const longestDim = Math.max(Math.abs(distanceX), Math.abs(distanceY));\n\n // We're using distance based heuristics to determine if a tile should be split into quadrants or not.\n // radiusOfMaxLvlLodInTiles defines that there's always a certain number of maxLevel tiles next to the map center.\n // Using the fact that a parent node in quadtree is twice the size of its children (per dimension)\n // we can define distance thresholds for each relative level:\n // f(k) = offset + 2 + 4 + 8 + 16 + ... + 2^k. This is the same as \"offset+2^(k+1)-2\"\n const distToSplit = radiusOfMaxLvlLodInTiles + (1 << (maxZoom - it.zoom)) - 2;\n\n // Have we reached the target depth or is the tile too far away to be any split further?\n if (it.zoom === maxZoom || (longestDim > distToSplit && it.zoom >= minZoom)) {\n const dz = maxZoom - it.zoom, dx = cameraPoint[0] - 0.5 - (x << dz), dy = cameraPoint[1] - 0.5 - (y << dz);\n result.push({\n tileID: new OverscaledTileID(it.zoom === maxZoom ? overscaledZ : it.zoom, it.wrap, it.zoom, x, y),\n distanceSq: vec2.sqrLen([centerPoint[0] - 0.5 - x, centerPoint[1] - 0.5 - y]),\n // this variable is currently not used, but may be important to reduce the amount of loaded tiles\n tileDistanceToCamera: Math.sqrt(dx * dx + dy * dy)\n });\n continue;\n }\n\n for (let i = 0; i < 4; i++) {\n const childX = (x << 1) + (i % 2);\n const childY = (y << 1) + (i >> 1);\n const childZ = it.zoom + 1;\n let quadrant = it.aabb.quadrant(i);\n if (options.terrain) {\n const tileID = new OverscaledTileID(childZ, it.wrap, childZ, childX, childY);\n const minMax = options.terrain.getMinMaxElevation(tileID);\n const minElevation = minMax.minElevation ?? this.elevation;\n const maxElevation = minMax.maxElevation ?? this.elevation;\n quadrant = new Aabb(\n [quadrant.min[0], quadrant.min[1], minElevation] as vec3,\n [quadrant.max[0], quadrant.max[1], maxElevation] as vec3\n );\n }\n stack.push({aabb: quadrant, zoom: childZ, x: childX, y: childY, wrap: it.wrap, fullyVisible});\n }\n }\n\n return result.sort((a, b) => a.distanceSq - b.distanceSq).map(a => a.tileID);\n }\n\n resize(width: number, height: number) {\n this.width = width;\n this.height = height;\n\n this.pixelsToGLUnits = [2 / width, -2 / height];\n this._constrain();\n this._calcMatrices();\n }\n\n get unmodified(): boolean { return this._unmodified; }\n\n zoomScale(zoom: number) { return Math.pow(2, zoom); }\n scaleZoom(scale: number) { return Math.log(scale) / Math.LN2; }\n\n project(lnglat: LngLat) {\n const lat = clamp(lnglat.lat, -this.maxValidLatitude, this.maxValidLatitude);\n return new Point(\n mercatorXfromLng(lnglat.lng) * this.worldSize,\n mercatorYfromLat(lat) * this.worldSize);\n }\n\n unproject(point: Point): LngLat {\n return new MercatorCoordinate(point.x / this.worldSize, point.y / this.worldSize).toLngLat();\n }\n\n get point(): Point { return this.project(this.center); }\n\n /**\n * get the camera position in LngLat and altitudes in meter\n * @returns An object with lngLat & altitude.\n */\n getCameraPosition(): {\n lngLat: LngLat;\n altitude: number;\n } {\n const lngLat = this.pointLocation(this.getCameraPoint());\n const altitude = Math.cos(this._pitch) * this.cameraToCenterDistance / this._pixelPerMeter;\n return {lngLat, altitude: altitude + this.elevation};\n }\n\n /**\n * This method works in combination with freezeElevation activated.\n * freezeElevtion is enabled during map-panning because during this the camera should sit in constant height.\n * After panning finished, call this method to recalculate the zoomlevel for the current camera-height in current terrain.\n * @param terrain - the terrain\n */\n recalculateZoom(terrain: Terrain) {\n // find position the camera is looking on\n const center = this.pointLocation(this.centerPoint, terrain);\n const elevation = terrain.getElevationForLngLatZoom(center, this.tileZoom);\n const deltaElevation = this.elevation - elevation;\n if (!deltaElevation) return;\n\n // calculate mercator distance between camera & target\n const cameraPosition = this.getCameraPosition();\n const camera = MercatorCoordinate.fromLngLat(cameraPosition.lngLat, cameraPosition.altitude);\n const target = MercatorCoordinate.fromLngLat(center, elevation);\n const dx = camera.x - target.x, dy = camera.y - target.y, dz = camera.z - target.z;\n const distance = Math.sqrt(dx * dx + dy * dy + dz * dz);\n\n // from this distance we calculate the new zoomlevel\n const zoom = this.scaleZoom(this.cameraToCenterDistance / distance / this.tileSize);\n\n // update matrices\n this._elevation = elevation;\n this._center = center;\n this.zoom = zoom;\n }\n\n setLocationAtPoint(lnglat: LngLat, point: Point) {\n const a = this.pointCoordinate(point);\n const b = this.pointCoordinate(this.centerPoint);\n const loc = this.locationCoordinate(lnglat);\n const newCenter = new MercatorCoordinate(\n loc.x - (a.x - b.x),\n loc.y - (a.y - b.y));\n this.center = this.coordinateLocation(newCenter);\n if (this._renderWorldCopies) {\n this.center = this.center.wrap();\n }\n }\n\n /**\n * Given a location, return the screen point that corresponds to it\n * @param lnglat - location\n * @param terrain - optional terrain\n * @returns screen point\n */\n locationPoint(lnglat: LngLat, terrain?: Terrain): Point {\n return terrain ?\n this.coordinatePoint(this.locationCoordinate(lnglat), terrain.getElevationForLngLatZoom(lnglat, this.tileZoom), this.pixelMatrix3D) :\n this.coordinatePoint(this.locationCoordinate(lnglat));\n }\n\n /**\n * Given a point on screen, return its lnglat\n * @param p - screen point\n * @param terrain - optional terrain\n * @returns lnglat location\n */\n pointLocation(p: Point, terrain?: Terrain): LngLat {\n return this.coordinateLocation(this.pointCoordinate(p, terrain));\n }\n\n /**\n * Given a geographical lnglat, return an unrounded\n * coordinate that represents it at this transform's zoom level.\n * @param lnglat - the location\n * @returns The mercator coordinate\n */\n locationCoordinate(lnglat: LngLat): MercatorCoordinate {\n return MercatorCoordinate.fromLngLat(lnglat);\n }\n\n /**\n * Given a Coordinate, return its geographical position.\n * @param coord - mercator coordivates\n * @returns lng and lat\n */\n coordinateLocation(coord: MercatorCoordinate): LngLat {\n return coord && coord.toLngLat();\n }\n\n /**\n * Given a Point, return its mercator coordinate.\n * @param p - the point\n * @param terrain - optional terrain\n * @returns lnglat\n */\n pointCoordinate(p: Point, terrain?: Terrain): MercatorCoordinate {\n // get point-coordinate from terrain coordinates framebuffer\n if (terrain) {\n const coordinate = terrain.pointCoordinate(p);\n if (coordinate != null) {\n return coordinate;\n }\n }\n\n // calculate point-coordinate on flat earth\n const targetZ = 0;\n // since we don't know the correct projected z value for the point,\n // unproject two points to get a line and then find the point on that\n // line with z=0\n\n const coord0 = [p.x, p.y, 0, 1] as any;\n const coord1 = [p.x, p.y, 1, 1] as any;\n\n vec4.transformMat4(coord0, coord0, this.pixelMatrixInverse);\n vec4.transformMat4(coord1, coord1, this.pixelMatrixInverse);\n\n const w0 = coord0[3];\n const w1 = coord1[3];\n const x0 = coord0[0] / w0;\n const x1 = coord1[0] / w1;\n const y0 = coord0[1] / w0;\n const y1 = coord1[1] / w1;\n const z0 = coord0[2] / w0;\n const z1 = coord1[2] / w1;\n\n const t = z0 === z1 ? 0 : (targetZ - z0) / (z1 - z0);\n\n return new MercatorCoordinate(\n interpolates.number(x0, x1, t) / this.worldSize,\n interpolates.number(y0, y1, t) / this.worldSize);\n }\n\n /**\n * Given a coordinate, return the screen point that corresponds to it\n * @param coord - the coordinates\n * @param elevation - the elevation\n * @param pixelMatrix - the pixel matrix\n * @returns screen point\n */\n coordinatePoint(coord: MercatorCoordinate, elevation: number = 0, pixelMatrix = this.pixelMatrix): Point {\n const p = [coord.x * this.worldSize, coord.y * this.worldSize, elevation, 1] as any;\n vec4.transformMat4(p, p, pixelMatrix);\n return new Point(p[0] / p[3], p[1] / p[3]);\n }\n\n /**\n * Returns the map's geographical bounds. When the bearing or pitch is non-zero, the visible region is not\n * an axis-aligned rectangle, and the result is the smallest bounds that encompasses the visible region.\n * @returns Returns a {@link LngLatBounds} object describing the map's geographical bounds.\n */\n getBounds(): LngLatBounds {\n const top = Math.max(0, this.height / 2 - this.getHorizon());\n return new LngLatBounds()\n .extend(this.pointLocation(new Point(0, top)))\n .extend(this.pointLocation(new Point(this.width, top)))\n .extend(this.pointLocation(new Point(this.width, this.height)))\n .extend(this.pointLocation(new Point(0, this.height)));\n }\n\n /**\n * Returns the maximum geographical bounds the map is constrained to, or `null` if none set.\n * @returns max bounds\n */\n getMaxBounds(): LngLatBounds | null {\n if (!this.latRange || this.latRange.length !== 2 ||\n !this.lngRange || this.lngRange.length !== 2) return null;\n\n return new LngLatBounds([this.lngRange[0], this.latRange[0]], [this.lngRange[1], this.latRange[1]]);\n }\n\n /**\n * Calculate pixel height of the visible horizon in relation to map-center (e.g. height/2),\n * multiplied by a static factor to simulate the earth-radius.\n * The calculated value is the horizontal line from the camera-height to sea-level.\n * @returns Horizon above center in pixels.\n */\n getHorizon(): number {\n return Math.tan(Math.PI / 2 - this._pitch) * this.cameraToCenterDistance * 0.85;\n }\n\n /**\n * Sets or clears the map's geographical constraints.\n * @param bounds - A {@link LngLatBounds} object describing the new geographic boundaries of the map.\n */\n setMaxBounds(bounds?: LngLatBounds | null) {\n if (bounds) {\n this.lngRange = [bounds.getWest(), bounds.getEast()];\n this.latRange = [bounds.getSouth(), bounds.getNorth()];\n this._constrain();\n } else {\n this.lngRange = null;\n this.latRange = [-this.maxValidLatitude, this.maxValidLatitude];\n }\n }\n\n /**\n * Calculate the posMatrix that, given a tile coordinate, would be used to display the tile on a map.\n * @param unwrappedTileID - the tile ID\n */\n calculatePosMatrix(unwrappedTileID: UnwrappedTileID, aligned: boolean = false): mat4 {\n const posMatrixKey = unwrappedTileID.key;\n const cache = aligned ? this._alignedPosMatrixCache : this._posMatrixCache;\n if (cache[posMatrixKey]) {\n return cache[posMatrixKey];\n }\n\n const canonical = unwrappedTileID.canonical;\n const scale = this.worldSize / this.zoomScale(canonical.z);\n const unwrappedX = canonical.x + Math.pow(2, canonical.z) * unwrappedTileID.wrap;\n\n const posMatrix = mat4.identity(new Float64Array(16) as any);\n mat4.translate(posMatrix, posMatrix, [unwrappedX * scale, canonical.y * scale, 0]);\n mat4.scale(posMatrix, posMatrix, [scale / EXTENT, scale / EXTENT, 1]);\n mat4.multiply(posMatrix, aligned ? this.alignedProjMatrix : this.projMatrix, posMatrix);\n\n cache[posMatrixKey] = new Float32Array(posMatrix);\n return cache[posMatrixKey];\n }\n\n customLayerMatrix(): mat4 {\n return this.mercatorMatrix.slice() as any;\n }\n\n _constrain() {\n if (!this.center || !this.width || !this.height || this._constraining) return;\n\n this._constraining = true;\n\n let minY = -90;\n let maxY = 90;\n let minX = -180;\n let maxX = 180;\n let sy, sx, x2, y2;\n const size = this.size,\n unmodified = this._unmodified;\n\n if (this.latRange) {\n const latRange = this.latRange;\n minY = mercatorYfromLat(latRange[1]) * this.worldSize;\n maxY = mercatorYfromLat(latRange[0]) * this.worldSize;\n sy = maxY - minY < size.y ? size.y / (maxY - minY) : 0;\n }\n\n if (this.lngRange) {\n const lngRange = this.lngRange;\n\n minX = wrap(\n mercatorXfromLng(lngRange[0]) * this.worldSize,\n 0,\n this.worldSize\n );\n maxX = wrap(\n mercatorXfromLng(lngRange[1]) * this.worldSize,\n 0,\n this.worldSize\n );\n\n if (maxX < minX) maxX += this.worldSize;\n\n sx = maxX - minX < size.x ? size.x / (maxX - minX) : 0;\n }\n\n const point = this.point;\n\n // how much the map should scale to fit the screen into given latitude/longitude ranges\n const s = Math.max(sx || 0, sy || 0);\n\n if (s) {\n this.center = this.unproject(new Point(\n sx ? (maxX + minX) / 2 : point.x,\n sy ? (maxY + minY) / 2 : point.y));\n this.zoom += this.scaleZoom(s);\n this._unmodified = unmodified;\n this._constraining = false;\n return;\n }\n\n if (this.latRange) {\n const y = point.y,\n h2 = size.y / 2;\n\n if (y - h2 < minY) y2 = minY + h2;\n if (y + h2 > maxY) y2 = maxY - h2;\n }\n\n if (this.lngRange) {\n const centerX = (minX + maxX) / 2;\n const x = wrap(point.x, centerX - this.worldSize / 2, centerX + this.worldSize / 2);\n const w2 = size.x / 2;\n\n if (x - w2 < minX) x2 = minX + w2;\n if (x + w2 > maxX) x2 = maxX - w2;\n }\n\n // pan the map if the screen goes off the range\n if (x2 !== undefined || y2 !== undefined) {\n this.center = this.unproject(new Point(\n x2 !== undefined ? x2 : point.x,\n y2 !== undefined ? y2 : point.y)).wrap();\n }\n\n this._unmodified = unmodified;\n this._constraining = false;\n }\n\n _calcMatrices() {\n if (!this.height) return;\n\n const halfFov = this._fov / 2;\n const offset = this.centerOffset;\n const x = this.point.x, y = this.point.y;\n this.cameraToCenterDistance = 0.5 / Math.tan(halfFov) * this.height;\n this._pixelPerMeter = mercatorZfromAltitude(1, this.center.lat) * this.worldSize;\n\n let m = mat4.identity(new Float64Array(16) as any);\n mat4.scale(m, m, [this.width / 2, -this.height / 2, 1]);\n mat4.translate(m, m, [1, -1, 0]);\n this.labelPlaneMatrix = m;\n\n m = mat4.identity(new Float64Array(16) as any);\n mat4.scale(m, m, [1, -1, 1]);\n mat4.translate(m, m, [-1, -1, 0]);\n mat4.scale(m, m, [2 / this.width, 2 / this.height, 1]);\n this.glCoordMatrix = m;\n\n // Calculate the camera to sea-level distance in pixel in respect of terrain\n const cameraToSeaLevelDistance = this.cameraToCenterDistance + this._elevation * this._pixelPerMeter / Math.cos(this._pitch);\n // In case of negative minimum elevation (e.g. the dead see, under the sea maps) use a lower plane for calculation\n const minElevation = Math.min(this.elevation, this._minEleveationForCurrentTile);\n const cameraToLowestPointDistance = cameraToSeaLevelDistance - minElevation * this._pixelPerMeter / Math.cos(this._pitch);\n const lowestPlane = minElevation < 0 ? cameraToLowestPointDistance : cameraToSeaLevelDistance;\n\n // Find the distance from the center point [width/2 + offset.x, height/2 + offset.y] to the\n // center top point [width/2 + offset.x, 0] in Z units, using the law of sines.\n // 1 Z unit is equivalent to 1 horizontal px at the center of the map\n // (the distance between[width/2, height/2] and [width/2 + 1, height/2])\n const groundAngle = Math.PI / 2 + this._pitch;\n const fovAboveCenter = this._fov * (0.5 + offset.y / this.height);\n const topHalfSurfaceDistance = Math.sin(fovAboveCenter) * lowestPlane / Math.sin(clamp(Math.PI - groundAngle - fovAboveCenter, 0.01, Math.PI - 0.01));\n\n // Find the distance from the center point to the horizon\n const horizon = this.getHorizon();\n const horizonAngle = Math.atan(horizon / this.cameraToCenterDistance);\n const fovCenterToHorizon = 2 * horizonAngle * (0.5 + offset.y / (horizon * 2));\n const topHalfSurfaceDistanceHorizon = Math.sin(fovCenterToHorizon) * lowestPlane / Math.sin(clamp(Math.PI - groundAngle - fovCenterToHorizon, 0.01, Math.PI - 0.01));\n\n // Calculate z distance of the farthest fragment that should be rendered.\n // Add a bit extra to avoid precision problems when a fragment's distance is exactly `furthestDistance`\n const topHalfMinDistance = Math.min(topHalfSurfaceDistance, topHalfSurfaceDistanceHorizon);\n const farZ = (Math.cos(Math.PI / 2 - this._pitch) * topHalfMinDistance + lowestPlane) * 1.01;\n\n // The larger the value of nearZ is\n // - the more depth precision is available for features (good)\n // - clipping starts appearing sooner when the camera is close to 3d features (bad)\n //\n // Smaller values worked well for mapbox-gl-js but deckgl was encountering precision issues\n // when rendering it's layers using custom layers. This value was experimentally chosen and\n // seems to solve z-fighting issues in deckgl while not clipping buildings too close to the camera.\n const nearZ = this.height / 50;\n\n // matrix for conversion from location to GL coordinates (-1 .. 1)\n m = new Float64Array(16) as any;\n mat4.perspective(m, this._fov, this.width / this.height, nearZ, farZ);\n\n // Apply center of perspective offset\n m[8] = -offset.x * 2 / this.width;\n m[9] = offset.y * 2 / this.height;\n\n mat4.scale(m, m, [1, -1, 1]);\n mat4.translate(m, m, [0, 0, -this.cameraToCenterDistance]);\n mat4.rotateX(m, m, this._pitch);\n mat4.rotateZ(m, m, this.angle);\n mat4.translate(m, m, [-x, -y, 0]);\n\n // The mercatorMatrix can be used to transform points from mercator coordinates\n // ([0, 0] nw, [1, 1] se) to GL coordinates.\n this.mercatorMatrix = mat4.scale([] as any, m, [this.worldSize, this.worldSize, this.worldSize]);\n\n // scale vertically to meters per pixel (inverse of ground resolution):\n mat4.scale(m, m, [1, 1, this._pixelPerMeter]);\n\n // matrix for conversion from location to screen coordinates in 2D\n this.pixelMatrix = mat4.multiply(new Float64Array(16) as any, this.labelPlaneMatrix, m);\n\n // matrix for conversion from location to GL coordinates (-1 .. 1)\n mat4.translate(m, m, [0, 0, -this.elevation]); // elevate camera over terrain\n this.projMatrix = m;\n this.invProjMatrix = mat4.invert([] as any, m);\n\n // matrix for conversion from location to screen coordinates in 2D\n this.pixelMatrix3D = mat4.multiply(new Float64Array(16) as any, this.labelPlaneMatrix, m);\n\n // Make a second projection matrix that is aligned to a pixel grid for rendering raster tiles.\n // We're rounding the (floating point) x/y values to achieve to avoid rendering raster images to fractional\n // coordinates. Additionally, we adjust by half a pixel in either direction in case that viewport dimension\n // is an odd integer to preserve rendering to the pixel grid. We're rotating this shift based on the angle\n // of the transformation so that 0°, 90°, 180°, and 270° rasters are crisp, and adjust the shift so that\n // it is always <= 0.5 pixels.\n const xShift = (this.width % 2) / 2, yShift = (this.height % 2) / 2,\n angleCos = Math.cos(this.angle), angleSin = Math.sin(this.angle),\n dx = x - Math.round(x) + angleCos * xShift + angleSin * yShift,\n dy = y - Math.round(y) + angleCos * yShift + angleSin * xShift;\n const alignedM = new Float64Array(m) as any as mat4;\n mat4.translate(alignedM, alignedM, [dx > 0.5 ? dx - 1 : dx, dy > 0.5 ? dy - 1 : dy, 0]);\n this.alignedProjMatrix = alignedM;\n\n // inverse matrix for conversion from screen coordinaes to location\n m = mat4.invert(new Float64Array(16) as any, this.pixelMatrix);\n if (!m) throw new Error('failed to invert matrix');\n this.pixelMatrixInverse = m;\n\n this._posMatrixCache = {};\n this._alignedPosMatrixCache = {};\n }\n\n maxPitchScaleFactor() {\n // calcMatrices hasn't run yet\n if (!this.pixelMatrixInverse) return 1;\n\n const coord = this.pointCoordinate(new Point(0, 0));\n const p = [coord.x * this.worldSize, coord.y * this.worldSize, 0, 1] as vec4;\n const topPoint = vec4.transformMat4(p, p, this.pixelMatrix);\n return topPoint[3] / this.cameraToCenterDistance;\n }\n\n /**\n * The camera looks at the map from a 3D (lng, lat, altitude) location. Let's use `cameraLocation`\n * as the name for the location under the camera and on the surface of the earth (lng, lat, 0).\n * `cameraPoint` is the projected position of the `cameraLocation`.\n *\n * This point is useful to us because only fill-extrusions that are between `cameraPoint` and\n * the query point on the surface of the earth can extend and intersect the query.\n *\n * When the map is not pitched the `cameraPoint` is equivalent to the center of the map because\n * the camera is right above the center of the map.\n */\n getCameraPoint() {\n const pitch = this._pitch;\n const yOffset = Math.tan(pitch) * (this.cameraToCenterDistance || 1);\n return this.centerPoint.add(new Point(0, yOffset));\n }\n\n /**\n * When the map is pitched, some of the 3D features that intersect a query will not intersect\n * the query at the surface of the earth. Instead the feature may be closer and only intersect\n * the query because it extrudes into the air.\n * @param queryGeometry - For point queries, the line from the query point to the \"camera point\",\n * for other geometries, the envelope of the query geometry and the \"camera point\"\n * @returns a geometry that includes all of the original query as well as all possible ares of the\n * screen where the *base* of a visible extrusion could be.\n *\n */\n getCameraQueryGeometry(queryGeometry: Array): Array {\n const c = this.getCameraPoint();\n\n if (queryGeometry.length === 1) {\n return [queryGeometry[0], c];\n } else {\n let minX = c.x;\n let minY = c.y;\n let maxX = c.x;\n let maxY = c.y;\n for (const p of queryGeometry) {\n minX = Math.min(minX, p.x);\n minY = Math.min(minY, p.y);\n maxX = Math.max(maxX, p.x);\n maxY = Math.max(maxY, p.y);\n }\n return [\n new Point(minX, minY),\n new Point(maxX, minY),\n new Point(maxX, maxY),\n new Point(minX, maxY),\n new Point(minX, minY)\n ];\n }\n }\n}\n","import * as glMatrix from \"./common.js\";\n/**\n * 2x2 Matrix\n * @module mat2\n */\n\n/**\n * Creates a new identity mat2\n *\n * @returns {mat2} a new 2x2 matrix\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(4);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[1] = 0;\n out[2] = 0;\n }\n\n out[0] = 1;\n out[3] = 1;\n return out;\n}\n/**\n * Creates a new mat2 initialized with values from an existing matrix\n *\n * @param {ReadonlyMat2} a matrix to clone\n * @returns {mat2} a new 2x2 matrix\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Copy the values from one mat2 to another\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the source matrix\n * @returns {mat2} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Set a mat2 to the identity matrix\n *\n * @param {mat2} out the receiving matrix\n * @returns {mat2} out\n */\n\nexport function identity(out) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 1;\n return out;\n}\n/**\n * Create a new mat2 with the given values\n *\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m10 Component in column 1, row 0 position (index 2)\n * @param {Number} m11 Component in column 1, row 1 position (index 3)\n * @returns {mat2} out A new 2x2 matrix\n */\n\nexport function fromValues(m00, m01, m10, m11) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = m00;\n out[1] = m01;\n out[2] = m10;\n out[3] = m11;\n return out;\n}\n/**\n * Set the components of a mat2 to the given values\n *\n * @param {mat2} out the receiving matrix\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m10 Component in column 1, row 0 position (index 2)\n * @param {Number} m11 Component in column 1, row 1 position (index 3)\n * @returns {mat2} out\n */\n\nexport function set(out, m00, m01, m10, m11) {\n out[0] = m00;\n out[1] = m01;\n out[2] = m10;\n out[3] = m11;\n return out;\n}\n/**\n * Transpose the values of a mat2\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the source matrix\n * @returns {mat2} out\n */\n\nexport function transpose(out, a) {\n // If we are transposing ourselves we can skip a few steps but have to cache\n // some values\n if (out === a) {\n var a1 = a[1];\n out[1] = a[2];\n out[2] = a1;\n } else {\n out[0] = a[0];\n out[1] = a[2];\n out[2] = a[1];\n out[3] = a[3];\n }\n\n return out;\n}\n/**\n * Inverts a mat2\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the source matrix\n * @returns {mat2} out\n */\n\nexport function invert(out, a) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3]; // Calculate the determinant\n\n var det = a0 * a3 - a2 * a1;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = a3 * det;\n out[1] = -a1 * det;\n out[2] = -a2 * det;\n out[3] = a0 * det;\n return out;\n}\n/**\n * Calculates the adjugate of a mat2\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the source matrix\n * @returns {mat2} out\n */\n\nexport function adjoint(out, a) {\n // Caching this value is nessecary if out == a\n var a0 = a[0];\n out[0] = a[3];\n out[1] = -a[1];\n out[2] = -a[2];\n out[3] = a0;\n return out;\n}\n/**\n * Calculates the determinant of a mat2\n *\n * @param {ReadonlyMat2} a the source matrix\n * @returns {Number} determinant of a\n */\n\nexport function determinant(a) {\n return a[0] * a[3] - a[2] * a[1];\n}\n/**\n * Multiplies two mat2's\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the first operand\n * @param {ReadonlyMat2} b the second operand\n * @returns {mat2} out\n */\n\nexport function multiply(out, a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n out[0] = a0 * b0 + a2 * b1;\n out[1] = a1 * b0 + a3 * b1;\n out[2] = a0 * b2 + a2 * b3;\n out[3] = a1 * b2 + a3 * b3;\n return out;\n}\n/**\n * Rotates a mat2 by the given angle\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat2} out\n */\n\nexport function rotate(out, a, rad) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n out[0] = a0 * c + a2 * s;\n out[1] = a1 * c + a3 * s;\n out[2] = a0 * -s + a2 * c;\n out[3] = a1 * -s + a3 * c;\n return out;\n}\n/**\n * Scales the mat2 by the dimensions in the given vec2\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the matrix to rotate\n * @param {ReadonlyVec2} v the vec2 to scale the matrix by\n * @returns {mat2} out\n **/\n\nexport function scale(out, a, v) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var v0 = v[0],\n v1 = v[1];\n out[0] = a0 * v0;\n out[1] = a1 * v0;\n out[2] = a2 * v1;\n out[3] = a3 * v1;\n return out;\n}\n/**\n * Creates a matrix from a given angle\n * This is equivalent to (but much faster than):\n *\n * mat2.identity(dest);\n * mat2.rotate(dest, dest, rad);\n *\n * @param {mat2} out mat2 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat2} out\n */\n\nexport function fromRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n out[0] = c;\n out[1] = s;\n out[2] = -s;\n out[3] = c;\n return out;\n}\n/**\n * Creates a matrix from a vector scaling\n * This is equivalent to (but much faster than):\n *\n * mat2.identity(dest);\n * mat2.scale(dest, dest, vec);\n *\n * @param {mat2} out mat2 receiving operation result\n * @param {ReadonlyVec2} v Scaling vector\n * @returns {mat2} out\n */\n\nexport function fromScaling(out, v) {\n out[0] = v[0];\n out[1] = 0;\n out[2] = 0;\n out[3] = v[1];\n return out;\n}\n/**\n * Returns a string representation of a mat2\n *\n * @param {ReadonlyMat2} a matrix to represent as a string\n * @returns {String} string representation of the matrix\n */\n\nexport function str(a) {\n return \"mat2(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \")\";\n}\n/**\n * Returns Frobenius norm of a mat2\n *\n * @param {ReadonlyMat2} a the matrix to calculate Frobenius norm of\n * @returns {Number} Frobenius norm\n */\n\nexport function frob(a) {\n return Math.hypot(a[0], a[1], a[2], a[3]);\n}\n/**\n * Returns L, D and U matrices (Lower triangular, Diagonal and Upper triangular) by factorizing the input matrix\n * @param {ReadonlyMat2} L the lower triangular matrix\n * @param {ReadonlyMat2} D the diagonal matrix\n * @param {ReadonlyMat2} U the upper triangular matrix\n * @param {ReadonlyMat2} a the input matrix to factorize\n */\n\nexport function LDU(L, D, U, a) {\n L[2] = a[2] / a[0];\n U[0] = a[0];\n U[1] = a[1];\n U[3] = a[3] - L[2] * U[1];\n return [L, D, U];\n}\n/**\n * Adds two mat2's\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the first operand\n * @param {ReadonlyMat2} b the second operand\n * @returns {mat2} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n return out;\n}\n/**\n * Subtracts matrix b from matrix a\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the first operand\n * @param {ReadonlyMat2} b the second operand\n * @returns {mat2} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n return out;\n}\n/**\n * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyMat2} a The first matrix.\n * @param {ReadonlyMat2} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3];\n}\n/**\n * Returns whether or not the matrices have approximately the same elements in the same position.\n *\n * @param {ReadonlyMat2} a The first matrix.\n * @param {ReadonlyMat2} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3));\n}\n/**\n * Multiply each element of the matrix by a scalar.\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the matrix to scale\n * @param {Number} b amount to scale the matrix's elements by\n * @returns {mat2} out\n */\n\nexport function multiplyScalar(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n return out;\n}\n/**\n * Adds two mat2's after multiplying each element of the second operand by a scalar value.\n *\n * @param {mat2} out the receiving vector\n * @param {ReadonlyMat2} a the first operand\n * @param {ReadonlyMat2} b the second operand\n * @param {Number} scale the amount to scale b's elements by before adding\n * @returns {mat2} out\n */\n\nexport function multiplyScalarAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n return out;\n}\n/**\n * Alias for {@link mat2.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link mat2.subtract}\n * @function\n */\n\nexport var sub = subtract;","/**\n * Throttle the given function to run at most every `period` milliseconds.\n */\nexport function throttle void>(fn: T, time: number): (...args: Parameters) => ReturnType {\n let pending = false;\n let timerId: ReturnType = null;\n let lastCallContext = null;\n let lastCallArgs: Parameters;\n\n const later = () => {\n timerId = null;\n if (pending) {\n fn.apply(lastCallContext, lastCallArgs);\n timerId = setTimeout(later, time);\n pending = false;\n }\n };\n\n return (...args: Parameters) => {\n pending = true;\n lastCallContext = this;\n lastCallArgs = args;\n if (!timerId) {\n later();\n }\n return timerId;\n };\n}\n","import {throttle} from '../util/throttle';\n\nimport type {Map} from './map';\n\n/**\n * Adds the map's position to its page's location hash.\n * Passed as an option to the map object.\n *\n * @group Markers and Controls\n */\nexport class Hash {\n _map: Map;\n _hashName: string;\n\n constructor(hashName?: string | null) {\n this._hashName = hashName && encodeURIComponent(hashName);\n }\n\n /**\n * Map element to listen for coordinate changes\n *\n * @param map - The map object\n * @returns `this`\n */\n addTo(map: Map) {\n this._map = map;\n addEventListener('hashchange', this._onHashChange, false);\n this._map.on('moveend', this._updateHash);\n return this;\n }\n\n /**\n * Removes hash\n *\n * @returns `this`\n */\n remove() {\n removeEventListener('hashchange', this._onHashChange, false);\n this._map.off('moveend', this._updateHash);\n clearTimeout(this._updateHash());\n\n delete this._map;\n return this;\n }\n\n getHashString(mapFeedback?: boolean) {\n const center = this._map.getCenter(),\n zoom = Math.round(this._map.getZoom() * 100) / 100,\n // derived from equation: 512px * 2^z / 360 / 10^d < 0.5px\n precision = Math.ceil((zoom * Math.LN2 + Math.log(512 / 360 / 0.5)) / Math.LN10),\n m = Math.pow(10, precision),\n lng = Math.round(center.lng * m) / m,\n lat = Math.round(center.lat * m) / m,\n bearing = this._map.getBearing(),\n pitch = this._map.getPitch();\n let hash = '';\n if (mapFeedback) {\n // new map feedback site has some constraints that don't allow\n // us to use the same hash format as we do for the Map hash option.\n hash += `/${lng}/${lat}/${zoom}`;\n } else {\n hash += `${zoom}/${lat}/${lng}`;\n }\n\n if (bearing || pitch) hash += (`/${Math.round(bearing * 10) / 10}`);\n if (pitch) hash += (`/${Math.round(pitch)}`);\n\n if (this._hashName) {\n const hashName = this._hashName;\n let found = false;\n const parts = window.location.hash.slice(1).split('&').map(part => {\n const key = part.split('=')[0];\n if (key === hashName) {\n found = true;\n return `${key}=${hash}`;\n }\n return part;\n }).filter(a => a);\n if (!found) {\n parts.push(`${hashName}=${hash}`);\n }\n return `#${parts.join('&')}`;\n }\n\n return `#${hash}`;\n }\n\n _getCurrentHash = () => {\n // Get the current hash from location, stripped from its number sign\n const hash = window.location.hash.replace('#', '');\n if (this._hashName) {\n // Split the parameter-styled hash into parts and find the value we need\n let keyval;\n hash.split('&').map(\n part => part.split('=')\n ).forEach(part => {\n if (part[0] === this._hashName) {\n keyval = part;\n }\n });\n return (keyval ? keyval[1] || '' : '').split('/');\n }\n return hash.split('/');\n };\n\n _onHashChange = () => {\n const loc = this._getCurrentHash();\n if (loc.length >= 3 && !loc.some(v => isNaN(v))) {\n const bearing = this._map.dragRotate.isEnabled() && this._map.touchZoomRotate.isEnabled() ? +(loc[3] || 0) : this._map.getBearing();\n this._map.jumpTo({\n center: [+loc[2], +loc[1]],\n zoom: +loc[0],\n bearing,\n pitch: +(loc[4] || 0)\n });\n return true;\n }\n return false;\n };\n\n _updateHashUnthrottled = () => {\n // Replace if already present, else append the updated hash string\n const location = window.location.href.replace(/(#.+)?$/, this.getHashString());\n try {\n window.history.replaceState(window.history.state, null, location);\n } catch (SecurityError) {\n // IE11 does not allow this if the page is within an iframe created\n // with iframe.contentWindow.document.write(...).\n // https://github.com/mapbox/mapbox-gl-js/issues/7410\n }\n };\n\n /**\n * Mobile Safari doesn't allow updating the hash more than 100 times per 30 seconds.\n */\n _updateHash: () => ReturnType = throttle(this._updateHashUnthrottled, 30 * 1000 / 100);\n}\n","import {browser} from '../util/browser';\nimport type {Map} from './map';\nimport {bezier, clamp, extend} from '../util/util';\nimport Point from '@mapbox/point-geometry';\nimport type {DragPanOptions} from './handler/shim/drag_pan';\n\nconst defaultInertiaOptions = {\n linearity: 0.3,\n easing: bezier(0, 0, 0.3, 1),\n};\n\nconst defaultPanInertiaOptions = extend({\n deceleration: 2500,\n maxSpeed: 1400\n}, defaultInertiaOptions);\n\nconst defaultZoomInertiaOptions = extend({\n deceleration: 20,\n maxSpeed: 1400\n}, defaultInertiaOptions);\n\nconst defaultBearingInertiaOptions = extend({\n deceleration: 1000,\n maxSpeed: 360\n}, defaultInertiaOptions);\n\nconst defaultPitchInertiaOptions = extend({\n deceleration: 1000,\n maxSpeed: 90\n}, defaultInertiaOptions);\n\nexport type InertiaOptions = {\n linearity: number;\n easing: (t: number) => number;\n deceleration: number;\n maxSpeed: number;\n};\n\nexport type InputEvent = MouseEvent | TouchEvent | KeyboardEvent | WheelEvent;\n\nexport class HandlerInertia {\n _map: Map;\n _inertiaBuffer: Array<{\n time: number;\n settings: any;\n }>;\n\n constructor(map: Map) {\n this._map = map;\n this.clear();\n }\n\n clear() {\n this._inertiaBuffer = [];\n }\n\n record(settings: any) {\n this._drainInertiaBuffer();\n this._inertiaBuffer.push({time: browser.now(), settings});\n }\n\n _drainInertiaBuffer() {\n const inertia = this._inertiaBuffer,\n now = browser.now(),\n cutoff = 160; //msec\n\n while (inertia.length > 0 && now - inertia[0].time > cutoff)\n inertia.shift();\n }\n\n _onMoveEnd(panInertiaOptions?: DragPanOptions | boolean) {\n this._drainInertiaBuffer();\n if (this._inertiaBuffer.length < 2) {\n return;\n }\n\n const deltas = {\n zoom: 0,\n bearing: 0,\n pitch: 0,\n pan: new Point(0, 0),\n pinchAround: undefined,\n around: undefined\n };\n\n for (const {settings} of this._inertiaBuffer) {\n deltas.zoom += settings.zoomDelta || 0;\n deltas.bearing += settings.bearingDelta || 0;\n deltas.pitch += settings.pitchDelta || 0;\n if (settings.panDelta) deltas.pan._add(settings.panDelta);\n if (settings.around) deltas.around = settings.around;\n if (settings.pinchAround) deltas.pinchAround = settings.pinchAround;\n }\n\n const lastEntry = this._inertiaBuffer[this._inertiaBuffer.length - 1];\n const duration = (lastEntry.time - this._inertiaBuffer[0].time);\n\n const easeOptions = {} as any;\n\n if (deltas.pan.mag()) {\n const result = calculateEasing(deltas.pan.mag(), duration, extend({}, defaultPanInertiaOptions, panInertiaOptions || {}));\n easeOptions.offset = deltas.pan.mult(result.amount / deltas.pan.mag());\n easeOptions.center = this._map.transform.center;\n extendDuration(easeOptions, result);\n }\n\n if (deltas.zoom) {\n const result = calculateEasing(deltas.zoom, duration, defaultZoomInertiaOptions);\n easeOptions.zoom = this._map.transform.zoom + result.amount;\n extendDuration(easeOptions, result);\n }\n\n if (deltas.bearing) {\n const result = calculateEasing(deltas.bearing, duration, defaultBearingInertiaOptions);\n easeOptions.bearing = this._map.transform.bearing + clamp(result.amount, -179, 179);\n extendDuration(easeOptions, result);\n }\n\n if (deltas.pitch) {\n const result = calculateEasing(deltas.pitch, duration, defaultPitchInertiaOptions);\n easeOptions.pitch = this._map.transform.pitch + result.amount;\n extendDuration(easeOptions, result);\n }\n\n if (easeOptions.zoom || easeOptions.bearing) {\n const last = deltas.pinchAround === undefined ? deltas.around : deltas.pinchAround;\n easeOptions.around = last ? this._map.unproject(last) : this._map.getCenter();\n }\n\n this.clear();\n return extend(easeOptions, {\n noMoveStart: true\n });\n\n }\n}\n\n// Unfortunately zoom, bearing, etc can't have different durations and easings so\n// we need to choose one. We use the longest duration and it's corresponding easing.\nfunction extendDuration(easeOptions, result) {\n if (!easeOptions.duration || easeOptions.duration < result.duration) {\n easeOptions.duration = result.duration;\n easeOptions.easing = result.easing;\n }\n}\n\nfunction calculateEasing(amount, inertiaDuration: number, inertiaOptions) {\n const {maxSpeed, linearity, deceleration} = inertiaOptions;\n const speed = clamp(\n amount * linearity / (inertiaDuration / 1000),\n -maxSpeed,\n maxSpeed);\n const duration = Math.abs(speed) / (deceleration * linearity);\n return {\n easing: inertiaOptions.easing,\n duration: duration * 1000,\n amount: speed * (duration / 2)\n };\n}\n","import {Event} from '../util/evented';\n\nimport {DOM} from '../util/dom';\nimport Point from '@mapbox/point-geometry';\nimport {extend} from '../util/util';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\n\nimport type {Map} from './map';\nimport type {LngLat} from '../geo/lng_lat';\nimport {SourceSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * An event from the mouse relevant to a specific layer.\n *\n * @group Event Related\n */\nexport type MapLayerMouseEvent = MapMouseEvent & { features?: MapGeoJSONFeature[] };\n\n/**\n * An event from a touch device relevat to a specific layer.\n *\n * @group Event Related\n */\nexport type MapLayerTouchEvent = MapTouchEvent & { features?: MapGeoJSONFeature[] };\n\n/**\n * The source event data type\n */\nexport type MapSourceDataType = 'content' | 'metadata' | 'visibility' | 'idle';\n\n/**\n * `MapLayerEventType` - a mapping between the event name and the event.\n * **Note:** These events are compatible with the optional `layerId` parameter.\n * If `layerId` is included as the second argument in {@link Map#on}, the event listener will fire only when the\n * event action contains a visible portion of the specified layer.\n * The following example can be used for all the events.\n *\n * @group Event Related\n * @example\n * ```ts\n * // Initialize the map\n * let map = new maplibregl.Map({ // map options });\n * // Set an event listener for a specific layer\n * map.on('the-event-name', 'poi-label', function(e) {\n * console.log('An event has occurred on a visible portion of the poi-label layer');\n * });\n * ```\n */\nexport type MapLayerEventType = {\n /**\n * Fired when a pointing device (usually a mouse) is pressed and released contains a visible portion of the specified layer.\n *\n * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/)\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n */\n click: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed and released twice contains a visible portion of the specified layer.\n *\n * **Note:** Under normal conditions, this event will be preceded by two `click` events.\n */\n dblclick: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed while inside a visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n mousedown: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is released while inside a visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n mouseup: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is moved while the cursor is inside a visible portion of the specified layer.\n * As you move the cursor across the layer, the event will fire every time the cursor changes position within that layer.\n *\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on over](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n */\n mousemove: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) enters a visible portion of a specified layer from\n * outside that layer or outside the map canvas.\n *\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n */\n mouseenter: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) leaves a visible portion of a specified layer, or leaves\n * the map canvas.\n *\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n */\n mouseleave: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is moved inside a visible portion of the specified layer.\n *\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n */\n mouseover: MapLayerMouseEvent;\n /**\n * Fired when a point device (usually a mouse) leaves the visible portion of the specified layer.\n */\n mouseout: MapLayerMouseEvent;\n /**\n * Fired when the right button of the mouse is clicked or the context menu key is pressed within visible portion of the specified layer.\n */\n contextmenu: MapLayerMouseEvent;\n /**\n * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchstart: MapLayerTouchEvent;\n /**\n * Fired when a [`touchend`](https://developer.mozilla.org/en-US/docs/Web/Events/touchend) event occurs within the visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchend: MapLayerTouchEvent;\n /**\n * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchcancel: MapLayerTouchEvent;\n};\n\n/**\n * `MapEventType` - a mapping between the event name and the event value.\n * These events are used with the {@link Map#on} method.\n * When using a `layerId` with {@link Map#on} method, please refer to {@link MapLayerEventType}.\n * The following example can be used for all the events.\n *\n * @group Event Related\n * @example\n * ```ts\n * // Initialize the map\n * let map = new maplibregl.Map({ // map options });\n * // Set an event listener\n * map.on('the-event-name', () => {\n * console.log('An event has occurred!');\n * });\n * ```\n */\nexport type MapEventType = {\n /**\n * Fired when an error occurs. This is GL JS's primary error reporting\n * mechanism. We use an event instead of `throw` to better accommodate\n * asynchronous operations. If no listeners are bound to the `error` event, the\n * error will be printed to the console.\n */\n error: ErrorEvent;\n /**\n * @event `load` Fired immediately after all necessary resources have been downloaded\n * and the first visually complete rendering of the map has occurred.\n *\n * @see [Draw GeoJSON points](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-markers/)\n * @see [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/)\n * @see [Animate a point](https://maplibre.org/maplibre-gl-js/docs/examples/animate-point-along-line/)\n */\n load: MapLibreEvent;\n /**\n * Fired after the last frame rendered before the map enters an\n * \"idle\" state:\n *\n * - No camera transitions are in progress\n * - All currently requested tiles have loaded\n * - All fade/transition animations have completed\n */\n idle: MapLibreEvent;\n /**\n * Fired immediately after the map has been removed with {@link Map#remove}.\n */\n remove: MapLibreEvent;\n /**\n * Fired whenever the map is drawn to the screen, as the result of\n *\n * - a change to the map's position, zoom, pitch, or bearing\n * - a change to the map's style\n * - a change to a GeoJSON source\n * - the loading of a vector tile, GeoJSON file, glyph, or sprite\n */\n render: MapLibreEvent;\n /**\n * Fired immediately after the map has been resized.\n */\n resize: MapLibreEvent;\n /**\n * Fired when the WebGL context is lost.\n */\n webglcontextlost: MapContextEvent;\n /**\n * Fired when the WebGL context is restored.\n */\n webglcontextrestored: MapContextEvent;\n /**\n * Fired when any map data (style, source, tile, etc) begins loading or\n * changing asynchronously. All `dataloading` events are followed by a `data`,\n * `dataabort` or `error` event.\n */\n dataloading: MapDataEvent;\n /**\n * Fired when any map data loads or changes. See {@link MapDataEvent} for more information.\n * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/)\n */\n data: MapDataEvent;\n tiledataloading: MapDataEvent;\n /**\n * Fired when one of the map's sources begins loading or changing asynchronously.\n * All `sourcedataloading` events are followed by a `sourcedata`, `sourcedataabort` or `error` event.\n */\n sourcedataloading: MapSourceDataEvent;\n /**\n * Fired when the map's style begins loading or changing asynchronously.\n * All `styledataloading` events are followed by a `styledata`\n * or `error` event.\n */\n styledataloading: MapStyleDataEvent;\n /**\n * Fired when one of the map's sources loads or changes, including if a tile belonging\n * to a source loads or changes.\n */\n sourcedata: MapSourceDataEvent;\n /**\n * Fired when the map's style loads or changes.\n */\n styledata: MapStyleDataEvent;\n /**\n * Fired when an icon or pattern needed by the style is missing. The missing image can\n * be added with {@link Map#addImage} within this event listener callback to prevent the image from\n * being skipped. This event can be used to dynamically generate icons and patterns.\n * @see [Generate and add a missing icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-missing-generated/)\n */\n styleimagemissing: MapStyleImageMissingEvent;\n /**\n * Fired when a request for one of the map's sources' tiles or data is aborted.\n */\n dataabort: MapDataEvent;\n /**\n * Fired when a request for one of the map's sources' data is aborted.\n */\n sourcedataabort: MapSourceDataEvent;\n /**\n * Fired when the user cancels a \"box zoom\" interaction, or when the bounding box does not meet the minimum size threshold.\n * See {@link BoxZoomHandler}.\n */\n boxzoomcancel: MapLibreZoomEvent;\n /**\n * Fired when a \"box zoom\" interaction starts. See {@link BoxZoomHandler}.\n */\n boxzoomstart: MapLibreZoomEvent;\n /**\n * Fired when a \"box zoom\" interaction ends. See {@link BoxZoomHandler}.\n */\n boxzoomend: MapLibreZoomEvent;\n /**\n * Fired when a [`touchcancel`](https://developer.mozilla.org/en-US/docs/Web/Events/touchcancel) event occurs within the map.\n */\n touchcancel: MapTouchEvent;\n /**\n * Fired when a [`touchmove`](https://developer.mozilla.org/en-US/docs/Web/Events/touchmove) event occurs within the map.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchmove: MapTouchEvent;\n /**\n * Fired when a [`touchend`](https://developer.mozilla.org/en-US/docs/Web/Events/touchend) event occurs within the map.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchend: MapTouchEvent;\n /**\n * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the map.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchstart: MapTouchEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed and released at the same point on the map.\n *\n * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/)\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n */\n click: MapMouseEvent;\n /**\n * Fired when the right button of the mouse is clicked or the context menu key is pressed within the map.\n */\n contextmenu: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed and released twice at the same point on the map in rapid succession.\n *\n * **Note:** Under normal conditions, this event will be preceded by two `click` events.\n */\n dblclick: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is moved while the cursor is inside the map.\n * As you move the cursor across the map, the event will fire every time the cursor changes position within the map.\n *\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on over](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n */\n mousemove: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is released within the map.\n *\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n mouseup: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed within the map.\n *\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n mousedown: MapMouseEvent;\n /**\n * Fired when a point device (usually a mouse) leaves the map's canvas.\n */\n mouseout: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is moved within the map.\n * As you move the cursor across a web page containing a map,\n * the event will fire each time it enters the map or any child elements.\n *\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n */\n mouseover: MapMouseEvent;\n /**\n * Fired just before the map begins a transition from one\n * view to another, as the result of either user interaction or methods such as {@link Map#jumpTo}.\n *\n */\n movestart: MapLibreEvent;\n /**\n * Fired repeatedly during an animated transition from one view to\n * another, as the result of either user interaction or methods such as {@link Map#flyTo}.\n *\n * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/)\n */\n move: MapLibreEvent;\n /**\n * Fired just after the map completes a transition from one\n * view to another, as the result of either user interaction or methods such as {@link Map#jumpTo}.\n *\n * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/)\n */\n moveend: MapLibreEvent;\n /**\n * Fired just before the map begins a transition from one zoom level to another,\n * as the result of either user interaction or methods such as {@link Map#flyTo}.\n */\n zoomstart: MapLibreEvent;\n /**\n * Fired repeatedly during an animated transition from one zoom level to another,\n * as the result of either user interaction or methods such as {@link Map#flyTo}.\n */\n zoom: MapLibreEvent;\n /**\n * Fired just after the map completes a transition from one zoom level to another,\n * as the result of either user interaction or methods such as {@link Map#flyTo}.\n */\n zoomend: MapLibreEvent;\n /**\n * Fired when a \"drag to rotate\" interaction starts. See {@link DragRotateHandler}.\n */\n rotatestart: MapLibreEvent;\n /**\n * Fired repeatedly during a \"drag to rotate\" interaction. See {@link DragRotateHandler}.\n */\n rotate: MapLibreEvent;\n /**\n * Fired when a \"drag to rotate\" interaction ends. See {@link DragRotateHandler}.\n */\n rotateend: MapLibreEvent;\n /**\n * Fired when a \"drag to pan\" interaction starts. See {@link DragPanHandler}.\n */\n dragstart: MapLibreEvent;\n /**\n * Fired repeatedly during a \"drag to pan\" interaction. See {@link DragPanHandler}.\n */\n drag: MapLibreEvent;\n /**\n * Fired when a \"drag to pan\" interaction ends. See {@link DragPanHandler}.\n * @see [Create a draggable marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/)\n */\n dragend: MapLibreEvent;\n /**\n * Fired whenever the map's pitch (tilt) begins a change as\n * the result of either user interaction or methods such as {@link Map#flyTo} .\n */\n pitchstart: MapLibreEvent;\n /**\n * Fired repeatedly during the map's pitch (tilt) animation between\n * one state and another as the result of either user interaction\n * or methods such as {@link Map#flyTo}.\n */\n pitch: MapLibreEvent;\n /**\n * Fired immediately after the map's pitch (tilt) finishes changing as\n * the result of either user interaction or methods such as {@link Map#flyTo}.\n */\n pitchend: MapLibreEvent;\n /**\n * Fired when a [`wheel`](https://developer.mozilla.org/en-US/docs/Web/Events/wheel) event occurs within the map.\n */\n wheel: MapWheelEvent;\n /**\n * Fired when terrain is changed\n */\n terrain: MapTerrainEvent;\n};\n\n/**\n * The base event for MapLibre\n *\n * @group Event Related\n */\nexport type MapLibreEvent = {\n type: keyof MapEventType | keyof MapLayerEventType;\n target: Map;\n originalEvent: TOrig;\n}\n\n/**\n * The style data event\n *\n * @group Event Related\n */\nexport type MapStyleDataEvent = MapLibreEvent & {\n dataType: 'style';\n}\n\n/**\n * The source data event interface\n *\n * @group Event Related\n */\nexport type MapSourceDataEvent = MapLibreEvent & {\n dataType: 'source';\n /**\n * True if the event has a `dataType` of `source` and the source has no outstanding network requests.\n */\n isSourceLoaded: boolean;\n /**\n * The [style spec representation of the source](https://maplibre.org/maplibre-style-spec/#sources) if the event has a `dataType` of `source`.\n */\n source: SourceSpecification;\n sourceId: string;\n sourceDataType: MapSourceDataType;\n /**\n * The tile being loaded or changed, if the event has a `dataType` of `source` and\n * the event is related to loading of a tile.\n */\n tile: any;\n}\n/**\n * `MapMouseEvent` is the event type for mouse-related map events.\n * @example\n * ```ts\n * // The `click` event is an example of a `MapMouseEvent`.\n * // Set up an event listener on the map.\n * map.on('click', function(e) {\n * // The event object (e) contains information like the\n * // coordinates of the point on the map that was clicked.\n * console.log('A click event has occurred at ' + e.lngLat);\n * });\n * ```\n */\nexport class MapMouseEvent extends Event implements MapLibreEvent {\n /**\n * The event type\n */\n type: 'mousedown' | 'mouseup' | 'click' | 'dblclick' | 'mousemove' | 'mouseover' | 'mouseenter' | 'mouseleave' | 'mouseout' | 'contextmenu';\n\n /**\n * The `Map` object that fired the event.\n */\n target: Map;\n\n /**\n * The DOM event which caused the map event.\n */\n originalEvent: MouseEvent;\n\n /**\n * The pixel coordinates of the mouse cursor, relative to the map and measured from the top left corner.\n */\n point: Point;\n\n /**\n * The geographic location on the map of the mouse cursor.\n */\n lngLat: LngLat;\n\n /**\n * Prevents subsequent default processing of the event by the map.\n *\n * Calling this method will prevent the following default map behaviors:\n *\n * * On `mousedown` events, the behavior of {@link DragPanHandler}\n * * On `mousedown` events, the behavior of {@link DragRotateHandler}\n * * On `mousedown` events, the behavior of {@link BoxZoomHandler}\n * * On `dblclick` events, the behavior of {@link DoubleClickZoomHandler}\n *\n */\n preventDefault() {\n this._defaultPrevented = true;\n }\n\n /**\n * `true` if `preventDefault` has been called.\n */\n get defaultPrevented(): boolean {\n return this._defaultPrevented;\n }\n\n _defaultPrevented: boolean;\n\n constructor(type: string, map: Map, originalEvent: MouseEvent, data: any = {}) {\n const point = DOM.mousePos(map.getCanvasContainer(), originalEvent);\n const lngLat = map.unproject(point);\n super(type, extend({point, lngLat, originalEvent}, data));\n this._defaultPrevented = false;\n this.target = map;\n }\n}\n\n/**\n * `MapTouchEvent` is the event type for touch-related map events.\n *\n * @group Event Related\n */\nexport class MapTouchEvent extends Event implements MapLibreEvent {\n /**\n * The event type.\n */\n type: 'touchstart' | 'touchmove' | 'touchend' | 'touchcancel';\n\n /**\n * The `Map` object that fired the event.\n */\n target: Map;\n\n /**\n * The DOM event which caused the map event.\n */\n originalEvent: TouchEvent;\n\n /**\n * The geographic location on the map of the center of the touch event points.\n */\n lngLat: LngLat;\n\n /**\n * The pixel coordinates of the center of the touch event points, relative to the map and measured from the top left\n * corner.\n */\n point: Point;\n\n /**\n * The array of pixel coordinates corresponding to a\n * [touch event's `touches`](https://developer.mozilla.org/en-US/docs/Web/API/TouchEvent/touches) property.\n */\n points: Array;\n\n /**\n * The geographical locations on the map corresponding to a\n * [touch event's `touches`](https://developer.mozilla.org/en-US/docs/Web/API/TouchEvent/touches) property.\n */\n lngLats: Array;\n\n /**\n * Prevents subsequent default processing of the event by the map.\n *\n * Calling this method will prevent the following default map behaviors:\n *\n * * On `touchstart` events, the behavior of {@link DragPanHandler}\n * * On `touchstart` events, the behavior of {@link TwoFingersTouchZoomRotateHandler}\n *\n */\n preventDefault() {\n this._defaultPrevented = true;\n }\n\n /**\n * `true` if `preventDefault` has been called.\n */\n get defaultPrevented(): boolean {\n return this._defaultPrevented;\n }\n\n _defaultPrevented: boolean;\n\n constructor(type: string, map: Map, originalEvent: TouchEvent) {\n const touches = type === 'touchend' ? originalEvent.changedTouches : originalEvent.touches;\n const points = DOM.touchPos(map.getCanvasContainer(), touches);\n const lngLats = points.map((t) => map.unproject(t));\n const point = points.reduce((prev, curr, i, arr) => {\n return prev.add(curr.div(arr.length));\n }, new Point(0, 0));\n const lngLat = map.unproject(point);\n super(type, {points, point, lngLats, lngLat, originalEvent});\n this._defaultPrevented = false;\n }\n}\n\n/**\n * `MapWheelEvent` is the event type for the `wheel` map event.\n *\n * @group Event Related\n */\nexport class MapWheelEvent extends Event {\n /**\n * The event type.\n */\n type: 'wheel';\n\n /**\n * The `Map` object that fired the event.\n */\n target: Map;\n\n /**\n * The DOM event which caused the map event.\n */\n originalEvent: WheelEvent;\n\n /**\n * Prevents subsequent default processing of the event by the map.\n *\n * Calling this method will prevent the behavior of {@link ScrollZoomHandler}.\n */\n preventDefault() {\n this._defaultPrevented = true;\n }\n\n /**\n * `true` if `preventDefault` has been called.\n */\n get defaultPrevented(): boolean {\n return this._defaultPrevented;\n }\n\n _defaultPrevented: boolean;\n\n /** */\n constructor(type: string, map: Map, originalEvent: WheelEvent) {\n super(type, {originalEvent});\n this._defaultPrevented = false;\n }\n}\n\n/**\n * A `MapLibreZoomEvent` is the event type for the boxzoom-related map events emitted by the {@link BoxZoomHandler}.\n *\n * @group Event Related\n */\nexport type MapLibreZoomEvent = {\n /**\n * The type of boxzoom event. One of `boxzoomstart`, `boxzoomend` or `boxzoomcancel`\n */\n type: 'boxzoomstart' | 'boxzoomend' | 'boxzoomcancel';\n /**\n * The `Map` instance that triggered the event\n */\n target: Map;\n /**\n * The DOM event that triggered the boxzoom event. Can be a `MouseEvent` or `KeyboardEvent`\n */\n originalEvent: MouseEvent;\n};\n\n/**\n * A `MapDataEvent` object is emitted with the `data`\n * and `dataloading` events. Possible values for\n * `dataType`s are:\n *\n * - `'source'`: The non-tile data associated with any source\n * - `'style'`: The [style](https://maplibre.org/maplibre-style-spec/) used by the map\n *\n * Possible values for `sourceDataType`s are:\n *\n * - `'metadata'`: indicates that any necessary source metadata has been loaded (such as TileJSON) and it is ok to start loading tiles\n * - `'content'`: indicates the source data has changed (such as when source.setData() has been called on GeoJSONSource)\n * - `'visibility'`: send when the source becomes used when at least one of its layers becomes visible in style sense (inside the layer's zoom range and with layout.visibility set to 'visible')\n * - `'idle'`: indicates that no new source data has been fetched (but the source has done loading)\n *\n * @group Event Related\n *\n * @example\n * ```ts\n * // The sourcedata event is an example of MapDataEvent.\n * // Set up an event listener on the map.\n * map.on('sourcedata', function(e) {\n * if (e.isSourceLoaded) {\n * // Do something when the source has finished loading\n * }\n * });\n * ```\n */\nexport type MapDataEvent = {\n /**\n * The event type.\n */\n type: string;\n /**\n * The type of data that has changed. One of `'source'`, `'style'`.\n */\n dataType: string;\n /**\n * Included if the event has a `dataType` of `source` and the event signals that internal data has been received or changed. Possible values are `metadata`, `content`, `visibility` and `idle`.\n */\n sourceDataType: MapSourceDataType;\n};\n\n/**\n * The terrain event\n *\n * @group Event Related\n */\nexport type MapTerrainEvent = {\n type: 'terrain';\n};\n\n/**\n * An event related to the web gl context\n *\n * @group Event Related\n */\nexport type MapContextEvent = {\n type: 'webglcontextlost' | 'webglcontextrestored';\n originalEvent: WebGLContextEvent;\n};\n\n/**\n * The style image missing event\n *\n * @group Event Related\n *\n * @see [Generate and add a missing icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-missing-generated/)\n */\nexport type MapStyleImageMissingEvent = MapLibreEvent & {\n type: 'styleimagemissing';\n id: string;\n}\n","import {MapMouseEvent, MapTouchEvent, MapWheelEvent} from '../events';\nimport {Handler} from '../handler_manager';\nimport type {Map} from '../map';\nimport type Point from '@mapbox/point-geometry';\n\nexport class MapEventHandler implements Handler {\n\n _mousedownPos: Point;\n _clickTolerance: number;\n _map: Map;\n\n constructor(map: Map, options: {\n clickTolerance: number;\n }) {\n this._map = map;\n this._clickTolerance = options.clickTolerance;\n }\n\n reset() {\n delete this._mousedownPos;\n }\n\n wheel(e: WheelEvent) {\n // If mapEvent.preventDefault() is called by the user, prevent handlers such as:\n // - ScrollZoom\n return this._firePreventable(new MapWheelEvent(e.type, this._map, e));\n }\n\n mousedown(e: MouseEvent, point: Point) {\n this._mousedownPos = point;\n // If mapEvent.preventDefault() is called by the user, prevent handlers such as:\n // - MousePan\n // - MouseRotate\n // - MousePitch\n // - DblclickHandler\n return this._firePreventable(new MapMouseEvent(e.type, this._map, e));\n }\n\n mouseup(e: MouseEvent) {\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n click(e: MouseEvent, point: Point) {\n if (this._mousedownPos && this._mousedownPos.dist(point) >= this._clickTolerance) return;\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n dblclick(e: MouseEvent) {\n // If mapEvent.preventDefault() is called by the user, prevent handlers such as:\n // - DblClickZoom\n return this._firePreventable(new MapMouseEvent(e.type, this._map, e));\n }\n\n mouseover(e: MouseEvent) {\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n mouseout(e: MouseEvent) {\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n touchstart(e: TouchEvent) {\n // If mapEvent.preventDefault() is called by the user, prevent handlers such as:\n // - TouchPan\n // - TouchZoom\n // - TouchRotate\n // - TouchPitch\n // - TapZoom\n // - SwipeZoom\n return this._firePreventable(new MapTouchEvent(e.type, this._map, e));\n }\n\n touchmove(e: TouchEvent) {\n this._map.fire(new MapTouchEvent(e.type, this._map, e));\n }\n\n touchend(e: TouchEvent) {\n this._map.fire(new MapTouchEvent(e.type, this._map, e));\n }\n\n touchcancel(e: TouchEvent) {\n this._map.fire(new MapTouchEvent(e.type, this._map, e));\n }\n\n _firePreventable(mapEvent: MapMouseEvent | MapTouchEvent | MapWheelEvent) {\n this._map.fire(mapEvent);\n if (mapEvent.defaultPrevented) {\n // returning an object marks the handler as active and resets other handlers\n return {};\n }\n }\n\n isEnabled() {\n return true;\n }\n\n isActive() {\n return false;\n }\n enable() {}\n disable() {}\n}\n\nexport class BlockableMapEventHandler {\n _map: Map;\n _delayContextMenu: boolean;\n _ignoreContextMenu: boolean;\n _contextMenuEvent: MouseEvent;\n\n constructor(map: Map) {\n this._map = map;\n }\n\n reset() {\n this._delayContextMenu = false;\n this._ignoreContextMenu = true;\n delete this._contextMenuEvent;\n }\n\n mousemove(e: MouseEvent) {\n // mousemove map events should not be fired when interaction handlers (pan, rotate, etc) are active\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n mousedown() {\n this._delayContextMenu = true;\n this._ignoreContextMenu = false;\n }\n\n mouseup() {\n this._delayContextMenu = false;\n if (this._contextMenuEvent) {\n this._map.fire(new MapMouseEvent('contextmenu', this._map, this._contextMenuEvent));\n delete this._contextMenuEvent;\n }\n }\n contextmenu(e: MouseEvent) {\n if (this._delayContextMenu) {\n // Mac: contextmenu fired on mousedown; we save it until mouseup for consistency's sake\n this._contextMenuEvent = e;\n } else if (!this._ignoreContextMenu) {\n // Windows: contextmenu fired on mouseup, so fire event now\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n // prevent browser context menu when necessary\n if (this._map.listens('contextmenu')) {\n e.preventDefault();\n }\n }\n\n isEnabled() {\n return true;\n }\n\n isActive() {\n return false;\n }\n enable() {}\n disable() {}\n}\n","import type {Map} from '../map';\nimport type {PointLike} from '../camera';\nimport type {Transform} from '../../geo/transform';\nimport Point from '@mapbox/point-geometry';\nimport {LngLat} from '../../geo/lng_lat';\n\n/**\n * @internal\n * Shared utilities for the Handler classes to access the correct camera state.\n * If Camera.transformCameraUpdate is specified, the \"desired state\" of camera may differ from the state used for rendering.\n * The handlers need the \"desired state\" to track accumulated changes.\n */\nexport class TransformProvider {\n _map: Map;\n\n constructor(map: Map) {\n this._map = map;\n }\n\n get transform(): Transform {\n return this._map._requestedCameraState || this._map.transform;\n }\n\n get center() {\n return {lng: this.transform.center.lng, lat: this.transform.center.lat};\n }\n\n get zoom() {\n return this.transform.zoom;\n }\n\n get pitch() {\n return this.transform.pitch;\n }\n\n get bearing() {\n return this.transform.bearing;\n }\n\n unproject(point: PointLike): LngLat {\n return this.transform.pointLocation(Point.convert(point), this._map.terrain);\n }\n}\n","import {DOM} from '../../util/dom';\n\nimport {Event} from '../../util/evented';\nimport {TransformProvider} from './transform-provider';\n\nimport type {Map} from '../map';\nimport type Point from '@mapbox/point-geometry';\nimport {Handler} from '../handler_manager';\n\n/**\n * The `BoxZoomHandler` allows the user to zoom the map to fit within a bounding box.\n * The bounding box is defined by clicking and holding `shift` while dragging the cursor.\n *\n * @group Handlers\n */\nexport class BoxZoomHandler implements Handler {\n _map: Map;\n _tr: TransformProvider;\n _el: HTMLElement;\n _container: HTMLElement;\n _enabled: boolean;\n _active: boolean;\n _startPos: Point;\n _lastPos: Point;\n _box: HTMLElement;\n _clickTolerance: number;\n\n /** @internal */\n constructor(map: Map, options: {\n clickTolerance: number;\n }) {\n this._map = map;\n this._tr = new TransformProvider(map);\n this._el = map.getCanvasContainer();\n this._container = map.getContainer();\n this._clickTolerance = options.clickTolerance || 1;\n }\n\n /**\n * Returns a Boolean indicating whether the \"box zoom\" interaction is enabled.\n *\n * @returns `true` if the \"box zoom\" interaction is enabled.\n */\n isEnabled() {\n return !!this._enabled;\n }\n\n /**\n * Returns a Boolean indicating whether the \"box zoom\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"box zoom\" interaction is active.\n */\n isActive() {\n return !!this._active;\n }\n\n /**\n * Enables the \"box zoom\" interaction.\n *\n * @example\n * ```ts\n * map.boxZoom.enable();\n * ```\n */\n enable() {\n if (this.isEnabled()) return;\n this._enabled = true;\n }\n\n /**\n * Disables the \"box zoom\" interaction.\n *\n * @example\n * ```ts\n * map.boxZoom.disable();\n * ```\n */\n disable() {\n if (!this.isEnabled()) return;\n this._enabled = false;\n }\n\n mousedown(e: MouseEvent, point: Point) {\n if (!this.isEnabled()) return;\n if (!(e.shiftKey && e.button === 0)) return;\n\n DOM.disableDrag();\n this._startPos = this._lastPos = point;\n this._active = true;\n }\n\n mousemoveWindow(e: MouseEvent, point: Point) {\n if (!this._active) return;\n\n const pos = point;\n\n if (this._lastPos.equals(pos) || (!this._box && pos.dist(this._startPos) < this._clickTolerance)) {\n return;\n }\n\n const p0 = this._startPos;\n this._lastPos = pos;\n\n if (!this._box) {\n this._box = DOM.create('div', 'maplibregl-boxzoom', this._container);\n this._container.classList.add('maplibregl-crosshair');\n this._fireEvent('boxzoomstart', e);\n }\n\n const minX = Math.min(p0.x, pos.x),\n maxX = Math.max(p0.x, pos.x),\n minY = Math.min(p0.y, pos.y),\n maxY = Math.max(p0.y, pos.y);\n\n DOM.setTransform(this._box, `translate(${minX}px,${minY}px)`);\n\n this._box.style.width = `${maxX - minX}px`;\n this._box.style.height = `${maxY - minY}px`;\n }\n\n mouseupWindow(e: MouseEvent, point: Point) {\n if (!this._active) return;\n\n if (e.button !== 0) return;\n\n const p0 = this._startPos,\n p1 = point;\n\n this.reset();\n\n DOM.suppressClick();\n\n if (p0.x === p1.x && p0.y === p1.y) {\n this._fireEvent('boxzoomcancel', e);\n } else {\n this._map.fire(new Event('boxzoomend', {originalEvent: e}));\n return {\n cameraAnimation: map => map.fitScreenCoordinates(p0, p1, this._tr.bearing, {linear: true})\n };\n }\n }\n\n keydown(e: KeyboardEvent) {\n if (!this._active) return;\n\n if (e.keyCode === 27) {\n this.reset();\n this._fireEvent('boxzoomcancel', e);\n }\n }\n\n reset() {\n this._active = false;\n\n this._container.classList.remove('maplibregl-crosshair');\n\n if (this._box) {\n DOM.remove(this._box);\n this._box = null;\n }\n\n DOM.enableDrag();\n\n delete this._startPos;\n delete this._lastPos;\n }\n\n _fireEvent(type: string, e: any) {\n return this._map.fire(new Event(type, {originalEvent: e}));\n }\n}\n","import Point from '@mapbox/point-geometry';\n\nexport function indexTouches(touches: Array, points: Array) {\n if (touches.length !== points.length) throw new Error(`The number of touches and points are not equal - touches ${touches.length}, points ${points.length}`);\n const obj = {};\n for (let i = 0; i < touches.length; i++) {\n obj[touches[i].identifier] = points[i];\n }\n return obj;\n}\n","import Point from '@mapbox/point-geometry';\nimport {indexTouches} from './handler_util';\n\nfunction getCentroid(points: Array) {\n const sum = new Point(0, 0);\n for (const point of points) {\n sum._add(point);\n }\n return sum.div(points.length);\n}\n\nexport const MAX_TAP_INTERVAL = 500;\nconst MAX_TOUCH_TIME = 500;\nexport const MAX_DIST = 30;\n\nexport class SingleTapRecognizer {\n\n numTouches: number;\n centroid: Point;\n startTime: number;\n aborted: boolean;\n touches: {\n [k in number | string]: Point;\n };\n\n constructor(options: {\n numTouches: number;\n }) {\n this.reset();\n this.numTouches = options.numTouches;\n }\n\n reset() {\n delete this.centroid;\n delete this.startTime;\n delete this.touches;\n this.aborted = false;\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n\n if (this.centroid || mapTouches.length > this.numTouches) {\n this.aborted = true;\n }\n if (this.aborted) {\n return;\n }\n\n if (this.startTime === undefined) {\n this.startTime = e.timeStamp;\n }\n\n if (mapTouches.length === this.numTouches) {\n this.centroid = getCentroid(points);\n this.touches = indexTouches(mapTouches, points);\n }\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n if (this.aborted || !this.centroid) return;\n\n const newTouches = indexTouches(mapTouches, points);\n for (const id in this.touches) {\n const prevPos = this.touches[id];\n const pos = newTouches[id];\n if (!pos || pos.dist(prevPos) > MAX_DIST) {\n this.aborted = true;\n }\n }\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this.centroid || e.timeStamp - this.startTime > MAX_TOUCH_TIME) {\n this.aborted = true;\n }\n\n if (mapTouches.length === 0) {\n const centroid = !this.aborted && this.centroid;\n this.reset();\n if (centroid) return centroid;\n }\n }\n\n}\n\nexport class TapRecognizer {\n\n singleTap: SingleTapRecognizer;\n numTaps: number;\n lastTime: number;\n lastTap: Point;\n count: number;\n\n constructor(options: {\n numTaps: number;\n numTouches: number;\n }) {\n this.singleTap = new SingleTapRecognizer(options);\n this.numTaps = options.numTaps;\n this.reset();\n }\n\n reset() {\n this.lastTime = Infinity;\n delete this.lastTap;\n this.count = 0;\n this.singleTap.reset();\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n this.singleTap.touchstart(e, points, mapTouches);\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n this.singleTap.touchmove(e, points, mapTouches);\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n const tap = this.singleTap.touchend(e, points, mapTouches);\n if (tap) {\n const soonEnough = e.timeStamp - this.lastTime < MAX_TAP_INTERVAL;\n const closeEnough = !this.lastTap || this.lastTap.dist(tap) < MAX_DIST;\n\n if (!soonEnough || !closeEnough) {\n this.reset();\n }\n\n this.count++;\n this.lastTime = e.timeStamp;\n this.lastTap = tap;\n\n if (this.count === this.numTaps) {\n this.reset();\n return tap;\n }\n }\n }\n}\n","import {TapRecognizer} from './tap_recognizer';\nimport type Point from '@mapbox/point-geometry';\nimport type {Map} from '../map';\nimport {TransformProvider} from './transform-provider';\nimport {Handler} from '../handler_manager';\n\nexport class TapZoomHandler implements Handler {\n _tr: TransformProvider;\n _enabled: boolean;\n _active: boolean;\n _zoomIn: TapRecognizer;\n _zoomOut: TapRecognizer;\n\n constructor(map: Map) {\n this._tr = new TransformProvider(map);\n this._zoomIn = new TapRecognizer({\n numTouches: 1,\n numTaps: 2\n });\n\n this._zoomOut = new TapRecognizer({\n numTouches: 2,\n numTaps: 1\n });\n\n this.reset();\n }\n\n reset() {\n this._active = false;\n this._zoomIn.reset();\n this._zoomOut.reset();\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n this._zoomIn.touchstart(e, points, mapTouches);\n this._zoomOut.touchstart(e, points, mapTouches);\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n this._zoomIn.touchmove(e, points, mapTouches);\n this._zoomOut.touchmove(e, points, mapTouches);\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n const zoomInPoint = this._zoomIn.touchend(e, points, mapTouches);\n const zoomOutPoint = this._zoomOut.touchend(e, points, mapTouches);\n const tr = this._tr;\n\n if (zoomInPoint) {\n this._active = true;\n e.preventDefault();\n setTimeout(() => this.reset(), 0);\n return {\n cameraAnimation: (map: Map) => map.easeTo({\n duration: 300,\n zoom: tr.zoom + 1,\n around: tr.unproject(zoomInPoint)\n }, {originalEvent: e})\n };\n } else if (zoomOutPoint) {\n this._active = true;\n e.preventDefault();\n setTimeout(() => this.reset(), 0);\n return {\n cameraAnimation: (map: Map) => map.easeTo({\n duration: 300,\n zoom: tr.zoom - 1,\n around: tr.unproject(zoomOutPoint)\n }, {originalEvent: e})\n };\n }\n }\n\n touchcancel() {\n this.reset();\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n}\n","import {DOM} from '../../util/dom';\nimport type Point from '@mapbox/point-geometry';\nimport {DragMoveStateManager} from './drag_move_state_manager';\nimport {Handler} from '../handler_manager';\n\ninterface DragMovementResult {\n bearingDelta?: number;\n pitchDelta?: number;\n around?: Point;\n panDelta?: Point;\n}\n\nexport interface DragPanResult extends DragMovementResult {\n around: Point;\n panDelta: Point;\n}\n\nexport interface DragRotateResult extends DragMovementResult {\n bearingDelta: number;\n}\n\nexport interface DragPitchResult extends DragMovementResult {\n pitchDelta: number;\n}\n\ntype DragMoveFunction = (lastPoint: Point, point: Point) => T;\n\nexport interface DragMoveHandler extends Handler {\n dragStart: (e: E, point: Point) => void;\n dragMove: (e: E, point: Point) => T | void;\n dragEnd: (e: E) => void;\n getClickTolerance: () => number;\n}\n\nexport type DragMoveHandlerOptions = {\n /**\n * If the movement is shorter than this value, consider it a click.\n */\n clickTolerance: number;\n /**\n * The move function to run on a valid movement.\n */\n move: DragMoveFunction;\n /**\n * A class used to manage the state of the drag event - start, checking valid moves, end. See the class documentation for more details.\n */\n moveStateManager: DragMoveStateManager;\n /**\n * A method used to assign the dragStart, dragMove, and dragEnd methods to the relevant event handlers, as well as assigning the contextmenu handler\n * @param handler - the handler\n */\n assignEvents: (handler: DragMoveHandler) => void;\n /**\n * Should the move start on the \"start\" event, or should it start on the first valid move.\n */\n activateOnStart?: boolean;\n /**\n * If true, handler will be enabled during construction\n */\n enable?: boolean;\n}\n\n/**\n * A generic class to create handlers for drag events, from both mouse and touch events.\n */\nexport class DragHandler implements DragMoveHandler {\n // Event handlers that may be assigned by the implementations of this class\n contextmenu?: Handler['contextmenu'];\n mousedown?: Handler['mousedown'];\n mousemoveWindow?: Handler['mousemoveWindow'];\n mouseup?: Handler['mouseup'];\n touchstart?: Handler['touchstart'];\n touchmoveWindow?: Handler['touchmoveWindow'];\n touchend?: Handler['touchend'];\n\n _clickTolerance: number;\n _moveFunction: DragMoveFunction;\n _activateOnStart: boolean;\n _active: boolean;\n _enabled: boolean;\n _moved: boolean;\n _lastPoint: Point | null;\n _moveStateManager: DragMoveStateManager;\n\n constructor(options: DragMoveHandlerOptions) {\n this._enabled = !!options.enable;\n this._moveStateManager = options.moveStateManager;\n this._clickTolerance = options.clickTolerance || 1;\n this._moveFunction = options.move;\n this._activateOnStart = !!options.activateOnStart;\n\n options.assignEvents(this);\n\n this.reset();\n }\n\n reset(e?: E) {\n this._active = false;\n this._moved = false;\n delete this._lastPoint;\n this._moveStateManager.endMove(e);\n }\n\n _move(...params: Parameters>) {\n const move = this._moveFunction(...params);\n if (move.bearingDelta || move.pitchDelta || move.around || move.panDelta) {\n this._active = true;\n return move;\n }\n }\n\n dragStart(e: E, point: Point);\n dragStart(e: E, point: Point[]);\n dragStart(e: E, point: Point | Point[]) {\n if (!this.isEnabled() || this._lastPoint) return;\n\n if (!this._moveStateManager.isValidStartEvent(e)) return;\n this._moveStateManager.startMove(e);\n\n this._lastPoint = point['length'] ? point[0] : point;\n\n if (this._activateOnStart && this._lastPoint) this._active = true;\n }\n\n dragMove(e: E, point: Point);\n dragMove(e: E, point: Point[]);\n dragMove(e: E, point: Point | Point[]) {\n if (!this.isEnabled()) return;\n const lastPoint = this._lastPoint;\n if (!lastPoint) return;\n e.preventDefault();\n\n if (!this._moveStateManager.isValidMoveEvent(e)) {\n this.reset(e);\n return;\n }\n\n const movePoint = point['length'] ? point[0] : point;\n\n if (!this._moved && movePoint.dist(lastPoint) < this._clickTolerance) return;\n this._moved = true;\n this._lastPoint = movePoint;\n\n return this._move(lastPoint, movePoint);\n }\n\n dragEnd(e: E) {\n if (!this.isEnabled() || !this._lastPoint) return;\n if (!this._moveStateManager.isValidEndEvent(e)) return;\n if (this._moved) DOM.suppressClick();\n this.reset(e);\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n\n getClickTolerance() {\n return this._clickTolerance;\n }\n}\n","import {DOM} from '../../util/dom';\n\nconst LEFT_BUTTON = 0;\nconst RIGHT_BUTTON = 2;\n\n// the values for each button in MouseEvent.buttons\nconst BUTTONS_FLAGS = {\n [LEFT_BUTTON]: 1,\n [RIGHT_BUTTON]: 2\n};\n\nfunction buttonNoLongerPressed(e: MouseEvent, button: number) {\n const flag = BUTTONS_FLAGS[button];\n return e.buttons === undefined || (e.buttons & flag) !== flag;\n}\n\n/*\n * Drag events are initiated by specific interaction which needs to be tracked until it ends.\n * This requires some state management:\n * 1. registering the initiating event,\n * 2. tracking that it was not canceled / not confusing it with another event firing.\n * 3. recognizing the ending event and cleaning up any internal state\n *\n * Concretely, we implement two state managers:\n * 1. MouseMoveStateManager\n * Receives a functions that is used to recognize mouse events that should be registered as the\n * relevant drag interactions - i.e. dragging with the right mouse button, or while CTRL is pressed.\n * 2. OneFingerTouchMoveStateManager\n * Checks if a drag event is using one finger, and continuously tracking that this is the same event\n * (i.e. to make sure not additional finger has started interacting with the screen before raising\n * the first finger).\n */\nexport interface DragMoveStateManager {\n startMove: (e: E) => void;\n endMove: (e?: E) => void;\n isValidStartEvent: (e: E) => boolean;\n isValidMoveEvent: (e: E) => boolean;\n isValidEndEvent: (e?: E) => boolean;\n}\n\nexport class MouseMoveStateManager implements DragMoveStateManager {\n _eventButton: number | undefined;\n _correctEvent: (e: MouseEvent) => boolean;\n\n constructor(options: {\n checkCorrectEvent: (e: MouseEvent) => boolean;\n }) {\n this._correctEvent = options.checkCorrectEvent;\n }\n\n startMove(e: MouseEvent) {\n const eventButton = DOM.mouseButton(e);\n this._eventButton = eventButton;\n }\n\n endMove(_e?: MouseEvent) {\n delete this._eventButton;\n }\n\n isValidStartEvent(e: MouseEvent) {\n return this._correctEvent(e);\n }\n\n isValidMoveEvent(e: MouseEvent) {\n // Some browsers don't fire a `mouseup` when the mouseup occurs outside\n // the window or iframe:\n // https://github.com/mapbox/mapbox-gl-js/issues/4622\n //\n // If the button is no longer pressed during this `mousemove` it may have\n // been released outside of the window or iframe.\n return !buttonNoLongerPressed(e, this._eventButton);\n }\n\n isValidEndEvent(e: MouseEvent) {\n const eventButton = DOM.mouseButton(e);\n return eventButton === this._eventButton;\n }\n}\n\nexport class OneFingerTouchMoveStateManager implements DragMoveStateManager {\n _firstTouch: number | undefined;\n\n constructor() {\n this._firstTouch = undefined;\n }\n\n _isOneFingerTouch(e: TouchEvent) {\n return e.targetTouches.length === 1;\n }\n\n _isSameTouchEvent(e: TouchEvent) {\n return e.targetTouches[0].identifier === this._firstTouch;\n }\n\n startMove(e: TouchEvent) {\n const firstTouch = e.targetTouches[0].identifier;\n this._firstTouch = firstTouch;\n }\n\n endMove(_e?: TouchEvent) {\n delete this._firstTouch;\n }\n\n isValidStartEvent(e: TouchEvent) {\n return this._isOneFingerTouch(e);\n }\n\n isValidMoveEvent(e: TouchEvent) {\n return this._isOneFingerTouch(e) && this._isSameTouchEvent(e);\n }\n\n isValidEndEvent(e: TouchEvent) {\n return this._isOneFingerTouch(e) && this._isSameTouchEvent(e);\n }\n}\n","import type Point from '@mapbox/point-geometry';\n\nimport {DOM} from '../../util/dom';\nimport {DragMoveHandler, DragPanResult, DragRotateResult, DragPitchResult, DragHandler} from './drag_handler';\nimport {MouseMoveStateManager} from './drag_move_state_manager';\n\nexport interface MousePanHandler extends DragMoveHandler {}\nexport interface MouseRotateHandler extends DragMoveHandler {}\nexport interface MousePitchHandler extends DragMoveHandler {}\n\nconst LEFT_BUTTON = 0;\nconst RIGHT_BUTTON = 2;\n\nconst assignEvents = (handler: DragHandler) => {\n handler.mousedown = handler.dragStart;\n handler.mousemoveWindow = handler.dragMove;\n handler.mouseup = handler.dragEnd;\n handler.contextmenu = function(e: MouseEvent) {\n e.preventDefault();\n };\n};\n\nexport const generateMousePanHandler = ({enable, clickTolerance,}: {\n clickTolerance: number;\n enable?: boolean;\n}): MousePanHandler => {\n const mouseMoveStateManager = new MouseMoveStateManager({\n checkCorrectEvent: (e: MouseEvent) => DOM.mouseButton(e) === LEFT_BUTTON && !e.ctrlKey,\n });\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({around: point, panDelta: point.sub(lastPoint)}),\n activateOnStart: true,\n moveStateManager: mouseMoveStateManager,\n enable,\n assignEvents,\n });\n};\n\nexport const generateMouseRotationHandler = ({enable, clickTolerance, bearingDegreesPerPixelMoved = 0.8}: {\n clickTolerance: number;\n bearingDegreesPerPixelMoved?: number;\n enable?: boolean;\n}): MouseRotateHandler => {\n const mouseMoveStateManager = new MouseMoveStateManager({\n checkCorrectEvent: (e: MouseEvent): boolean =>\n (DOM.mouseButton(e) === LEFT_BUTTON && e.ctrlKey) ||\n (DOM.mouseButton(e) === RIGHT_BUTTON),\n });\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({bearingDelta: (point.x - lastPoint.x) * bearingDegreesPerPixelMoved}),\n // prevent browser context menu when necessary; we don't allow it with rotation\n // because we can't discern rotation gesture start from contextmenu on Mac\n moveStateManager: mouseMoveStateManager,\n enable,\n assignEvents,\n });\n};\n\nexport const generateMousePitchHandler = ({enable, clickTolerance, pitchDegreesPerPixelMoved = -0.5}: {\n clickTolerance: number;\n pitchDegreesPerPixelMoved?: number;\n enable?: boolean;\n}): MousePitchHandler => {\n const mouseMoveStateManager = new MouseMoveStateManager({\n checkCorrectEvent: (e: MouseEvent): boolean =>\n (DOM.mouseButton(e) === LEFT_BUTTON && e.ctrlKey) ||\n (DOM.mouseButton(e) === RIGHT_BUTTON),\n });\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({pitchDelta: (point.y - lastPoint.y) * pitchDegreesPerPixelMoved}),\n // prevent browser context menu when necessary; we don't allow it with rotation\n // because we can't discern rotation gesture start from contextmenu on Mac\n moveStateManager: mouseMoveStateManager,\n enable,\n assignEvents,\n });\n};\n","import Point from '@mapbox/point-geometry';\nimport {indexTouches} from './handler_util';\nimport type {Map} from '../map';\nimport {GestureOptions} from '../map';\nimport {Handler} from '../handler_manager';\n\nexport class TouchPanHandler implements Handler {\n\n _enabled: boolean;\n _active: boolean;\n _touches: {\n [k in string | number]: Point;\n };\n _minTouches: number;\n _clickTolerance: number;\n _sum: Point;\n _map: Map;\n _cancelCooperativeMessage: boolean;\n\n constructor(options: {\n clickTolerance: number;\n cooperativeGestures: boolean | GestureOptions;\n }, map: Map) {\n this._minTouches = options.cooperativeGestures ? 2 : 1;\n this._clickTolerance = options.clickTolerance || 1;\n this._map = map;\n this.reset();\n }\n\n reset() {\n this._active = false;\n this._touches = {};\n this._sum = new Point(0, 0);\n\n // Put a delay on the cooperative gesture message so it's less twitchy\n setTimeout(() => {\n this._cancelCooperativeMessage = false;\n }, 200);\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n return this._calculateTransform(e, points, mapTouches);\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n if (this._map._cooperativeGestures) {\n if (this._minTouches === 2 && mapTouches.length < 2 && !this._cancelCooperativeMessage) {\n // If coop gesture enabled, show panning info to user\n this._map._onCooperativeGesture(e, false, mapTouches.length);\n } else if (!this._cancelCooperativeMessage) {\n // If user is successfully navigating, we don't need this warning until the touch resets\n this._cancelCooperativeMessage = true;\n }\n }\n if (!this._active || mapTouches.length < this._minTouches) return;\n e.preventDefault();\n return this._calculateTransform(e, points, mapTouches);\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n this._calculateTransform(e, points, mapTouches);\n\n if (this._active && mapTouches.length < this._minTouches) {\n this.reset();\n }\n }\n\n touchcancel() {\n this.reset();\n }\n\n _calculateTransform(e: TouchEvent, points: Array, mapTouches: Array) {\n if (mapTouches.length > 0) this._active = true;\n\n const touches = indexTouches(mapTouches, points);\n\n const touchPointSum = new Point(0, 0);\n const touchDeltaSum = new Point(0, 0);\n let touchDeltaCount = 0;\n\n for (const identifier in touches) {\n const point = touches[identifier];\n const prevPoint = this._touches[identifier];\n if (prevPoint) {\n touchPointSum._add(point);\n touchDeltaSum._add(point.sub(prevPoint));\n touchDeltaCount++;\n touches[identifier] = point;\n }\n }\n\n this._touches = touches;\n\n if (touchDeltaCount < this._minTouches || !touchDeltaSum.mag()) return;\n\n const panDelta = touchDeltaSum.div(touchDeltaCount);\n this._sum._add(panDelta);\n if (this._sum.mag() < this._clickTolerance) return;\n\n const around = touchPointSum.div(touchDeltaCount);\n\n return {\n around,\n panDelta\n };\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n}\n","import Point from '@mapbox/point-geometry';\nimport {DOM} from '../../util/dom';\nimport type {Map} from '../map';\nimport {Handler} from '../handler_manager';\n\n/**\n * An options object sent to the enable function of some of the handlers\n */\nexport type AroundCenterOptions = {\n /**\n * If \"center\" is passed, map will zoom around the center of map\n */\n around: 'center';\n}\n\n/**\n * The `TwoFingersTouchHandler`s allows the user to zoom, pitch and rotate the map using two fingers\n *\n * @group Handlers\n */\nabstract class TwoFingersTouchHandler implements Handler {\n\n _enabled: boolean;\n _active: boolean;\n _firstTwoTouches: [number, number];\n _vector: Point;\n _startVector: Point;\n _aroundCenter: boolean;\n\n /** @internal */\n constructor() {\n this.reset();\n }\n\n reset() {\n this._active = false;\n delete this._firstTwoTouches;\n }\n\n abstract _start(points: [Point, Point]);\n abstract _move(points: [Point, Point], pinchAround: Point, e: TouchEvent);\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n //log('touchstart', points, e.target.innerHTML, e.targetTouches.length ? e.targetTouches[0].target.innerHTML: undefined);\n if (this._firstTwoTouches || mapTouches.length < 2) return;\n\n this._firstTwoTouches = [\n mapTouches[0].identifier,\n mapTouches[1].identifier\n ];\n\n // implemented by child classes\n this._start([points[0], points[1]]);\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this._firstTwoTouches) return;\n\n e.preventDefault();\n\n const [idA, idB] = this._firstTwoTouches;\n const a = getTouchById(mapTouches, points, idA);\n const b = getTouchById(mapTouches, points, idB);\n if (!a || !b) return;\n const pinchAround = this._aroundCenter ? null : a.add(b).div(2);\n\n // implemented by child classes\n return this._move([a, b], pinchAround, e);\n\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this._firstTwoTouches) return;\n\n const [idA, idB] = this._firstTwoTouches;\n const a = getTouchById(mapTouches, points, idA);\n const b = getTouchById(mapTouches, points, idB);\n if (a && b) return;\n\n if (this._active) DOM.suppressClick();\n\n this.reset();\n }\n\n touchcancel() {\n this.reset();\n }\n\n /**\n * Enables the \"drag to pitch\" interaction.\n *\n * @example\n * ```ts\n * map.touchPitch.enable();\n * ```\n */\n enable(options?: AroundCenterOptions | boolean | null) {\n this._enabled = true;\n this._aroundCenter = !!options && (options as AroundCenterOptions).around === 'center';\n }\n\n /**\n * Disables the \"drag to pitch\" interaction.\n *\n * @example\n * ```ts\n * map.touchPitch.disable();\n * ```\n */\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to pitch\" interaction is enabled.\n *\n * @returns `true` if the \"drag to pitch\" interaction is enabled.\n */\n isEnabled() {\n return this._enabled;\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to pitch\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"drag to pitch\" interaction is active.\n */\n isActive() {\n return this._active;\n }\n}\n\nfunction getTouchById(mapTouches: Array, points: Array, identifier: number) {\n for (let i = 0; i < mapTouches.length; i++) {\n if (mapTouches[i].identifier === identifier) return points[i];\n }\n}\n\n/* ZOOM */\n\nconst ZOOM_THRESHOLD = 0.1;\n\nfunction getZoomDelta(distance, lastDistance) {\n return Math.log(distance / lastDistance) / Math.LN2;\n}\n\n/**\n * The `TwoFingersTouchHandler`s allows the user to zoom the map two fingers\n *\n * @group Handlers\n */\nexport class TwoFingersTouchZoomHandler extends TwoFingersTouchHandler {\n\n _distance: number;\n _startDistance: number;\n\n reset() {\n super.reset();\n delete this._distance;\n delete this._startDistance;\n }\n\n _start(points: [Point, Point]) {\n this._startDistance = this._distance = points[0].dist(points[1]);\n }\n\n _move(points: [Point, Point], pinchAround: Point) {\n const lastDistance = this._distance;\n this._distance = points[0].dist(points[1]);\n if (!this._active && Math.abs(getZoomDelta(this._distance, this._startDistance)) < ZOOM_THRESHOLD) return;\n this._active = true;\n return {\n zoomDelta: getZoomDelta(this._distance, lastDistance),\n pinchAround\n };\n }\n}\n\n/* ROTATE */\n\nconst ROTATION_THRESHOLD = 25; // pixels along circumference of touch circle\n\nfunction getBearingDelta(a, b) {\n return a.angleWith(b) * 180 / Math.PI;\n}\n\n/**\n * The `TwoFingersTouchHandler`s allows the user to rotate the map two fingers\n *\n * @group Handlers\n */\nexport class TwoFingersTouchRotateHandler extends TwoFingersTouchHandler {\n _minDiameter: number;\n\n reset() {\n super.reset();\n delete this._minDiameter;\n delete this._startVector;\n delete this._vector;\n }\n\n _start(points: [Point, Point]) {\n this._startVector = this._vector = points[0].sub(points[1]);\n this._minDiameter = points[0].dist(points[1]);\n }\n\n _move(points: [Point, Point], pinchAround: Point) {\n const lastVector = this._vector;\n this._vector = points[0].sub(points[1]);\n\n if (!this._active && this._isBelowThreshold(this._vector)) return;\n this._active = true;\n\n return {\n bearingDelta: getBearingDelta(this._vector, lastVector),\n pinchAround\n };\n }\n\n _isBelowThreshold(vector: Point) {\n /*\n * The threshold before a rotation actually happens is configured in\n * pixels along the circumference of the circle formed by the two fingers.\n * This makes the threshold in degrees larger when the fingers are close\n * together and smaller when the fingers are far apart.\n *\n * Use the smallest diameter from the whole gesture to reduce sensitivity\n * when pinching in and out.\n */\n\n this._minDiameter = Math.min(this._minDiameter, vector.mag());\n const circumference = Math.PI * this._minDiameter;\n const threshold = ROTATION_THRESHOLD / circumference * 360;\n\n const bearingDeltaSinceStart = getBearingDelta(vector, this._startVector);\n return Math.abs(bearingDeltaSinceStart) < threshold;\n }\n}\n\n/* PITCH */\n\nfunction isVertical(vector) {\n return Math.abs(vector.y) > Math.abs(vector.x);\n}\n\nconst ALLOWED_SINGLE_TOUCH_TIME = 100;\n\n/**\n * The `TwoFingersTouchPitchHandler` allows the user to pitch the map by dragging up and down with two fingers.\n *\n * @group Handlers\n */\nexport class TwoFingersTouchPitchHandler extends TwoFingersTouchHandler {\n\n _valid: boolean | void;\n _firstMove: number;\n _lastPoints: [Point, Point];\n _map: Map;\n _currentTouchCount: number;\n\n constructor(map: Map) {\n super();\n this._map = map;\n }\n\n reset() {\n super.reset();\n this._valid = undefined;\n delete this._firstMove;\n delete this._lastPoints;\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n super.touchstart(e, points, mapTouches);\n this._currentTouchCount = mapTouches.length;\n }\n\n _start(points: [Point, Point]) {\n this._lastPoints = points;\n if (isVertical(points[0].sub(points[1]))) {\n // fingers are more horizontal than vertical\n this._valid = false;\n\n }\n }\n\n _move(points: [Point, Point], center: Point, e: TouchEvent) {\n // If cooperative gestures is enabled, we need a 3-finger minimum for this gesture to register\n if (this._map._cooperativeGestures && this._currentTouchCount < 3) {\n return;\n }\n\n const vectorA = points[0].sub(this._lastPoints[0]);\n const vectorB = points[1].sub(this._lastPoints[1]);\n\n this._valid = this.gestureBeginsVertically(vectorA, vectorB, e.timeStamp);\n if (!this._valid) return;\n\n this._lastPoints = points;\n this._active = true;\n const yDeltaAverage = (vectorA.y + vectorB.y) / 2;\n const degreesPerPixelMoved = -0.5;\n return {\n pitchDelta: yDeltaAverage * degreesPerPixelMoved\n };\n }\n\n gestureBeginsVertically(vectorA: Point, vectorB: Point, timeStamp: number) {\n if (this._valid !== undefined) return this._valid;\n\n const threshold = 2;\n const movedA = vectorA.mag() >= threshold;\n const movedB = vectorB.mag() >= threshold;\n\n // neither finger has moved a meaningful amount, wait\n if (!movedA && !movedB) return;\n\n // One finger has moved and the other has not.\n // If enough time has passed, decide it is not a pitch.\n if (!movedA || !movedB) {\n if (this._firstMove === undefined) {\n this._firstMove = timeStamp;\n }\n\n if (timeStamp - this._firstMove < ALLOWED_SINGLE_TOUCH_TIME) {\n // still waiting for a movement from the second finger\n return undefined;\n } else {\n return false;\n }\n }\n\n const isSameDirection = vectorA.y > 0 === vectorB.y > 0;\n return isVertical(vectorA) && isVertical(vectorB) && isSameDirection;\n }\n}\n","import {Handler} from '../handler_manager';\nimport type {Map} from '../map';\nimport {TransformProvider} from './transform-provider';\n\nconst defaultOptions = {\n panStep: 100,\n bearingStep: 15,\n pitchStep: 10\n};\n\n/**\n * The `KeyboardHandler` allows the user to zoom, rotate, and pan the map using\n * the following keyboard shortcuts:\n *\n * - `=` / `+`: Increase the zoom level by 1.\n * - `Shift-=` / `Shift-+`: Increase the zoom level by 2.\n * - `-`: Decrease the zoom level by 1.\n * - `Shift--`: Decrease the zoom level by 2.\n * - Arrow keys: Pan by 100 pixels.\n * - `Shift+⇢`: Increase the rotation by 15 degrees.\n * - `Shift+⇠`: Decrease the rotation by 15 degrees.\n * - `Shift+⇡`: Increase the pitch by 10 degrees.\n * - `Shift+⇣`: Decrease the pitch by 10 degrees.\n *\n * @group Handlers\n */\nexport class KeyboardHandler implements Handler {\n _tr: TransformProvider;\n _enabled: boolean;\n _active: boolean;\n _panStep: number;\n _bearingStep: number;\n _pitchStep: number;\n _rotationDisabled: boolean;\n\n /** @internal */\n constructor(map: Map) {\n this._tr = new TransformProvider(map);\n const stepOptions = defaultOptions;\n this._panStep = stepOptions.panStep;\n this._bearingStep = stepOptions.bearingStep;\n this._pitchStep = stepOptions.pitchStep;\n this._rotationDisabled = false;\n }\n\n reset() {\n this._active = false;\n }\n\n keydown(e: KeyboardEvent) {\n if (e.altKey || e.ctrlKey || e.metaKey) return;\n\n let zoomDir = 0;\n let bearingDir = 0;\n let pitchDir = 0;\n let xDir = 0;\n let yDir = 0;\n\n switch (e.keyCode) {\n case 61:\n case 107:\n case 171:\n case 187:\n zoomDir = 1;\n break;\n\n case 189:\n case 109:\n case 173:\n zoomDir = -1;\n break;\n\n case 37:\n if (e.shiftKey) {\n bearingDir = -1;\n } else {\n e.preventDefault();\n xDir = -1;\n }\n break;\n\n case 39:\n if (e.shiftKey) {\n bearingDir = 1;\n } else {\n e.preventDefault();\n xDir = 1;\n }\n break;\n\n case 38:\n if (e.shiftKey) {\n pitchDir = 1;\n } else {\n e.preventDefault();\n yDir = -1;\n }\n break;\n\n case 40:\n if (e.shiftKey) {\n pitchDir = -1;\n } else {\n e.preventDefault();\n yDir = 1;\n }\n break;\n\n default:\n return;\n }\n\n if (this._rotationDisabled) {\n bearingDir = 0;\n pitchDir = 0;\n }\n\n return {\n cameraAnimation: (map: Map) => {\n const tr = this._tr;\n map.easeTo({\n duration: 300,\n easeId: 'keyboardHandler',\n easing: easeOut,\n\n zoom: zoomDir ? Math.round(tr.zoom) + zoomDir * (e.shiftKey ? 2 : 1) : tr.zoom,\n bearing: tr.bearing + bearingDir * this._bearingStep,\n pitch: tr.pitch + pitchDir * this._pitchStep,\n offset: [-xDir * this._panStep, -yDir * this._panStep],\n center: tr.center\n }, {originalEvent: e});\n }\n };\n }\n\n /**\n * Enables the \"keyboard rotate and zoom\" interaction.\n *\n * @example\n * ```ts\n * map.keyboard.enable();\n * ```\n */\n enable() {\n this._enabled = true;\n }\n\n /**\n * Disables the \"keyboard rotate and zoom\" interaction.\n *\n * @example\n * ```ts\n * map.keyboard.disable();\n * ```\n */\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n /**\n * Returns a Boolean indicating whether the \"keyboard rotate and zoom\"\n * interaction is enabled.\n *\n * @returns `true` if the \"keyboard rotate and zoom\"\n * interaction is enabled.\n */\n isEnabled() {\n return this._enabled;\n }\n\n /**\n * Returns true if the handler is enabled and has detected the start of a\n * zoom/rotate gesture.\n *\n * @returns `true` if the handler is enabled and has detected the\n * start of a zoom/rotate gesture.\n */\n isActive() {\n return this._active;\n }\n\n /**\n * Disables the \"keyboard pan/rotate\" interaction, leaving the\n * \"keyboard zoom\" interaction enabled.\n *\n * @example\n * ```ts\n * map.keyboard.disableRotation();\n * ```\n */\n disableRotation() {\n this._rotationDisabled = true;\n }\n\n /**\n * Enables the \"keyboard pan/rotate\" interaction.\n *\n * @example\n * ```ts\n * map.keyboard.enable();\n * map.keyboard.enableRotation();\n * ```\n */\n enableRotation() {\n this._rotationDisabled = false;\n }\n}\n\nfunction easeOut(t: number) {\n return t * (2 - t);\n}\n","import {DOM} from '../../util/dom';\n\nimport {defaultEasing, bezier} from '../../util/util';\nimport {browser} from '../../util/browser';\nimport {interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {LngLat} from '../../geo/lng_lat';\nimport {TransformProvider} from './transform-provider';\n\nimport type {Map} from '../map';\nimport type Point from '@mapbox/point-geometry';\nimport type {AroundCenterOptions} from './two_fingers_touch';\nimport {Handler} from '../handler_manager';\n\n// deltaY value for mouse scroll wheel identification\nconst wheelZoomDelta = 4.000244140625;\n\n// These magic numbers control the rate of zoom. Trackpad events fire at a greater\n// frequency than mouse scroll wheel, so reduce the zoom rate per wheel tick\nconst defaultZoomRate = 1 / 100;\nconst wheelZoomRate = 1 / 450;\n\n// upper bound on how much we scale the map in any single render frame; this\n// is used to limit zoom rate in the case of very fast scrolling\nconst maxScalePerFrame = 2;\n\n/**\n * The `ScrollZoomHandler` allows the user to zoom the map by scrolling.\n *\n * @group Handlers\n */\nexport class ScrollZoomHandler implements Handler {\n _map: Map;\n _tr: TransformProvider;\n _el: HTMLElement;\n _enabled: boolean;\n _active: boolean;\n _zooming: boolean;\n _aroundCenter: boolean;\n _around: LngLat;\n _aroundPoint: Point;\n _type: 'wheel' | 'trackpad' | null;\n _lastValue: number;\n _timeout: ReturnType; // used for delayed-handling of a single wheel movement\n _finishTimeout: ReturnType; // used to delay final '{move,zoom}end' events\n\n _lastWheelEvent: any;\n _lastWheelEventTime: number;\n\n _startZoom: number;\n _targetZoom: number;\n _delta: number;\n _easing: ((a: number) => number);\n _prevEase: {\n start: number;\n duration: number;\n easing: (_: number) => number;\n };\n\n _frameId: boolean;\n _triggerRenderFrame: () => void;\n\n _defaultZoomRate: number;\n _wheelZoomRate: number;\n\n /** @internal */\n constructor(map: Map, triggerRenderFrame: () => void) {\n this._map = map;\n this._tr = new TransformProvider(map);\n this._el = map.getCanvasContainer();\n this._triggerRenderFrame = triggerRenderFrame;\n\n this._delta = 0;\n\n this._defaultZoomRate = defaultZoomRate;\n this._wheelZoomRate = wheelZoomRate;\n }\n\n /**\n * Set the zoom rate of a trackpad\n * @param zoomRate - 1/100 The rate used to scale trackpad movement to a zoom value.\n * @example\n * Speed up trackpad zoom\n * ```ts\n * map.scrollZoom.setZoomRate(1/25);\n * ```\n */\n setZoomRate(zoomRate: number) {\n this._defaultZoomRate = zoomRate;\n }\n\n /**\n * Set the zoom rate of a mouse wheel\n * @param wheelZoomRate - 1/450 The rate used to scale mouse wheel movement to a zoom value.\n * @example\n * Slow down zoom of mouse wheel\n * ```ts\n * map.scrollZoom.setWheelZoomRate(1/600);\n * ```\n */\n setWheelZoomRate(wheelZoomRate: number) {\n this._wheelZoomRate = wheelZoomRate;\n }\n\n /**\n * Returns a Boolean indicating whether the \"scroll to zoom\" interaction is enabled.\n * @returns `true` if the \"scroll to zoom\" interaction is enabled.\n */\n isEnabled() {\n return !!this._enabled;\n }\n\n /*\n * Active state is turned on and off with every scroll wheel event and is set back to false before the map\n * render is called, so _active is not a good candidate for determining if a scroll zoom animation is in\n * progress.\n */\n isActive() {\n return !!this._active || this._finishTimeout !== undefined;\n }\n\n isZooming() {\n return !!this._zooming;\n }\n\n /**\n * Enables the \"scroll to zoom\" interaction.\n *\n * @param options - Options object.\n * @example\n * ```ts\n * map.scrollZoom.enable();\n * map.scrollZoom.enable({ around: 'center' })\n * ```\n */\n enable(options?: AroundCenterOptions | boolean) {\n if (this.isEnabled()) return;\n this._enabled = true;\n this._aroundCenter = !!options && (options as AroundCenterOptions).around === 'center';\n }\n\n /**\n * Disables the \"scroll to zoom\" interaction.\n *\n * @example\n * ```ts\n * map.scrollZoom.disable();\n * ```\n */\n disable() {\n if (!this.isEnabled()) return;\n this._enabled = false;\n }\n\n wheel(e: WheelEvent) {\n if (!this.isEnabled()) return;\n if (this._map._cooperativeGestures) {\n if (e[this._map._metaKey]) {\n e.preventDefault();\n } else {\n return;\n }\n }\n let value = e.deltaMode === WheelEvent.DOM_DELTA_LINE ? e.deltaY * 40 : e.deltaY;\n const now = browser.now(),\n timeDelta = now - (this._lastWheelEventTime || 0);\n\n this._lastWheelEventTime = now;\n\n if (value !== 0 && (value % wheelZoomDelta) === 0) {\n // This one is definitely a mouse wheel event.\n this._type = 'wheel';\n\n } else if (value !== 0 && Math.abs(value) < 4) {\n // This one is definitely a trackpad event because it is so small.\n this._type = 'trackpad';\n\n } else if (timeDelta > 400) {\n // This is likely a new scroll action.\n this._type = null;\n this._lastValue = value;\n\n // Start a timeout in case this was a singular event, and dely it by up to 40ms.\n this._timeout = setTimeout(this._onTimeout, 40, e);\n\n } else if (!this._type) {\n // This is a repeating event, but we don't know the type of event just yet.\n // If the delta per time is small, we assume it's a fast trackpad; otherwise we switch into wheel mode.\n this._type = (Math.abs(timeDelta * value) < 200) ? 'trackpad' : 'wheel';\n\n // Make sure our delayed event isn't fired again, because we accumulate\n // the previous event (which was less than 40ms ago) into this event.\n if (this._timeout) {\n clearTimeout(this._timeout);\n this._timeout = null;\n value += this._lastValue;\n }\n }\n\n // Slow down zoom if shift key is held for more precise zooming\n if (e.shiftKey && value) value = value / 4;\n\n // Only fire the callback if we actually know what type of scrolling device the user uses.\n if (this._type) {\n this._lastWheelEvent = e;\n this._delta -= value;\n if (!this._active) {\n this._start(e);\n }\n }\n\n e.preventDefault();\n }\n\n _onTimeout = (initialEvent: MouseEvent) => {\n this._type = 'wheel';\n this._delta -= this._lastValue;\n if (!this._active) {\n this._start(initialEvent);\n }\n };\n\n _start(e: MouseEvent) {\n if (!this._delta) return;\n\n if (this._frameId) {\n this._frameId = null;\n }\n\n this._active = true;\n if (!this.isZooming()) {\n this._zooming = true;\n }\n\n if (this._finishTimeout) {\n clearTimeout(this._finishTimeout);\n delete this._finishTimeout;\n }\n\n const pos = DOM.mousePos(this._el, e);\n const tr = this._tr;\n\n this._around = LngLat.convert(this._aroundCenter ? tr.center : tr.unproject(pos));\n this._aroundPoint = tr.transform.locationPoint(this._around);\n if (!this._frameId) {\n this._frameId = true;\n this._triggerRenderFrame();\n }\n }\n\n renderFrame() {\n if (!this._frameId) return;\n this._frameId = null;\n\n if (!this.isActive()) return;\n const tr = this._tr.transform;\n\n // if we've had scroll events since the last render frame, consume the\n // accumulated delta, and update the target zoom level accordingly\n if (this._delta !== 0) {\n // For trackpad events and single mouse wheel ticks, use the default zoom rate\n const zoomRate = (this._type === 'wheel' && Math.abs(this._delta) > wheelZoomDelta) ? this._wheelZoomRate : this._defaultZoomRate;\n // Scale by sigmoid of scroll wheel delta.\n let scale = maxScalePerFrame / (1 + Math.exp(-Math.abs(this._delta * zoomRate)));\n\n if (this._delta < 0 && scale !== 0) {\n scale = 1 / scale;\n }\n\n const fromScale = typeof this._targetZoom === 'number' ? tr.zoomScale(this._targetZoom) : tr.scale;\n this._targetZoom = Math.min(tr.maxZoom, Math.max(tr.minZoom, tr.scaleZoom(fromScale * scale)));\n\n // if this is a mouse wheel, refresh the starting zoom and easing\n // function we're using to smooth out the zooming between wheel\n // events\n if (this._type === 'wheel') {\n this._startZoom = tr.zoom;\n this._easing = this._smoothOutEasing(200);\n }\n\n this._delta = 0;\n }\n\n const targetZoom = typeof this._targetZoom === 'number' ?\n this._targetZoom : tr.zoom;\n const startZoom = this._startZoom;\n const easing = this._easing;\n\n let finished = false;\n let zoom;\n if (this._type === 'wheel' && startZoom && easing) {\n\n const t = Math.min((browser.now() - this._lastWheelEventTime) / 200, 1);\n const k = easing(t);\n zoom = interpolates.number(startZoom, targetZoom, k);\n if (t < 1) {\n if (!this._frameId) {\n this._frameId = true;\n }\n } else {\n finished = true;\n }\n } else {\n zoom = targetZoom;\n finished = true;\n }\n\n this._active = true;\n\n if (finished) {\n this._active = false;\n this._finishTimeout = setTimeout(() => {\n this._zooming = false;\n this._triggerRenderFrame();\n delete this._targetZoom;\n delete this._finishTimeout;\n }, 200);\n }\n\n return {\n noInertia: true,\n needsRenderFrame: !finished,\n zoomDelta: zoom - tr.zoom,\n around: this._aroundPoint,\n originalEvent: this._lastWheelEvent\n };\n }\n\n _smoothOutEasing(duration: number) {\n let easing = defaultEasing;\n\n if (this._prevEase) {\n const currentEase = this._prevEase;\n const t = (browser.now() - currentEase.start) / currentEase.duration;\n const speed = currentEase.easing(t + 0.01) - currentEase.easing(t);\n\n // Quick hack to make new bezier that is continuous with last\n const x = 0.27 / Math.sqrt(speed * speed + 0.0001) * 0.01;\n const y = Math.sqrt(0.27 * 0.27 - x * x);\n\n easing = bezier(x, y, 0.25, 1);\n }\n\n this._prevEase = {\n start: browser.now(),\n duration,\n easing\n };\n\n return easing;\n }\n\n reset() {\n this._active = false;\n this._zooming = false;\n delete this._targetZoom;\n if (this._finishTimeout) {\n clearTimeout(this._finishTimeout);\n delete this._finishTimeout;\n }\n }\n}\n","import type {ClickZoomHandler} from '../click_zoom';\nimport type {TapZoomHandler} from './../tap_zoom';\n\n/**\n * The `DoubleClickZoomHandler` allows the user to zoom the map at a point by\n * double clicking or double tapping.\n *\n * @group Handlers\n */\nexport class DoubleClickZoomHandler {\n\n _clickZoom: ClickZoomHandler;\n _tapZoom: TapZoomHandler;\n\n /** @internal */\n constructor(clickZoom: ClickZoomHandler, TapZoom: TapZoomHandler) {\n this._clickZoom = clickZoom;\n this._tapZoom = TapZoom;\n }\n\n /**\n * Enables the \"double click to zoom\" interaction.\n *\n * @example\n * ```ts\n * map.doubleClickZoom.enable();\n * ```\n */\n enable() {\n this._clickZoom.enable();\n this._tapZoom.enable();\n }\n\n /**\n * Disables the \"double click to zoom\" interaction.\n *\n * @example\n * ```ts\n * map.doubleClickZoom.disable();\n * ```\n */\n disable() {\n this._clickZoom.disable();\n this._tapZoom.disable();\n }\n\n /**\n * Returns a Boolean indicating whether the \"double click to zoom\" interaction is enabled.\n *\n * @returns `true` if the \"double click to zoom\" interaction is enabled.\n */\n isEnabled() {\n return this._clickZoom.isEnabled() && this._tapZoom.isEnabled();\n }\n\n /**\n * Returns a Boolean indicating whether the \"double click to zoom\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"double click to zoom\" interaction is active.\n */\n isActive() {\n return this._clickZoom.isActive() || this._tapZoom.isActive();\n }\n}\n","import type Point from '@mapbox/point-geometry';\nimport type {Map} from '../map';\nimport {TransformProvider} from './transform-provider';\nimport {Handler} from '../handler_manager';\n\n/**\n * The `ClickZoomHandler` allows the user to zoom the map at a point by double clicking\n * It is used by other handlers\n */\nexport class ClickZoomHandler implements Handler {\n\n _tr: TransformProvider;\n _enabled: boolean;\n _active: boolean;\n\n /** @internal */\n constructor(map: Map) {\n this._tr = new TransformProvider(map);\n this.reset();\n }\n\n reset() {\n this._active = false;\n }\n\n dblclick(e: MouseEvent, point: Point) {\n e.preventDefault();\n return {\n cameraAnimation: (map: Map) => {\n map.easeTo({\n duration: 300,\n zoom: this._tr.zoom + (e.shiftKey ? -1 : 1),\n around: this._tr.unproject(point)\n }, {originalEvent: e});\n }\n };\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n}\n","import {Handler} from '../handler_manager';\nimport {TapRecognizer, MAX_TAP_INTERVAL, MAX_DIST} from './tap_recognizer';\nimport type Point from '@mapbox/point-geometry';\n\nexport class TapDragZoomHandler implements Handler {\n\n _enabled: boolean;\n _active: boolean;\n _swipePoint: Point;\n _swipeTouch: number;\n _tapTime: number;\n _tapPoint: Point;\n _tap: TapRecognizer;\n\n constructor() {\n\n this._tap = new TapRecognizer({\n numTouches: 1,\n numTaps: 1\n });\n\n this.reset();\n }\n\n reset() {\n this._active = false;\n delete this._swipePoint;\n delete this._swipeTouch;\n delete this._tapTime;\n delete this._tapPoint;\n this._tap.reset();\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n if (this._swipePoint) return;\n\n if (!this._tapTime) {\n this._tap.touchstart(e, points, mapTouches);\n } else {\n const swipePoint = points[0];\n\n const soonEnough = e.timeStamp - this._tapTime < MAX_TAP_INTERVAL;\n const closeEnough = this._tapPoint.dist(swipePoint) < MAX_DIST;\n\n if (!soonEnough || !closeEnough) {\n this.reset();\n } else if (mapTouches.length > 0) {\n this._swipePoint = swipePoint;\n this._swipeTouch = mapTouches[0].identifier;\n }\n }\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this._tapTime) {\n this._tap.touchmove(e, points, mapTouches);\n } else if (this._swipePoint) {\n if (mapTouches[0].identifier !== this._swipeTouch) {\n return;\n }\n\n const newSwipePoint = points[0];\n const dist = newSwipePoint.y - this._swipePoint.y;\n this._swipePoint = newSwipePoint;\n\n e.preventDefault();\n this._active = true;\n\n return {\n zoomDelta: dist / 128\n };\n }\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this._tapTime) {\n const point = this._tap.touchend(e, points, mapTouches);\n if (point) {\n this._tapTime = e.timeStamp;\n this._tapPoint = point;\n }\n } else if (this._swipePoint) {\n if (mapTouches.length === 0) {\n this.reset();\n }\n }\n }\n\n touchcancel() {\n this.reset();\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n}\n","import type {MousePanHandler} from '../mouse';\nimport type {TouchPanHandler} from './../touch_pan';\n\n/**\n * A {@link DragPanHandler} options object\n */\nexport type DragPanOptions = {\n /**\n * factor used to scale the drag velocity\n * @defaultValue 0\n */\n linearity?: number;\n /**\n * easing function applled to `map.panTo` when applying the drag.\n * @param t - the easing function\n * @defaultValue bezier(0, 0, 0.3, 1)\n */\n easing?: (t: number) => number;\n /**\n * the maximum value of the drag velocity.\n * @defaultValue 1400\n */\n deceleration?: number;\n /**\n * the rate at which the speed reduces after the pan ends.\n * @defaultValue 2500\n */\n maxSpeed?: number;\n};\n\n/**\n * The `DragPanHandler` allows the user to pan the map by clicking and dragging\n * the cursor.\n *\n * @group Handlers\n */\nexport class DragPanHandler {\n\n _el: HTMLElement;\n _mousePan: MousePanHandler;\n _touchPan: TouchPanHandler;\n _inertiaOptions: DragPanOptions | boolean;\n\n /** @internal */\n constructor(el: HTMLElement, mousePan: MousePanHandler, touchPan: TouchPanHandler) {\n this._el = el;\n this._mousePan = mousePan;\n this._touchPan = touchPan;\n }\n\n /**\n * Enables the \"drag to pan\" interaction.\n *\n * @param options - Options object\n * @example\n * ```ts\n * map.dragPan.enable();\n * map.dragPan.enable({\n * linearity: 0.3,\n * easing: bezier(0, 0, 0.3, 1),\n * maxSpeed: 1400,\n * deceleration: 2500,\n * });\n * ```\n */\n enable(options?: DragPanOptions | boolean) {\n this._inertiaOptions = options || {};\n this._mousePan.enable();\n this._touchPan.enable();\n this._el.classList.add('maplibregl-touch-drag-pan');\n }\n\n /**\n * Disables the \"drag to pan\" interaction.\n *\n * @example\n * ```ts\n * map.dragPan.disable();\n * ```\n */\n disable() {\n this._mousePan.disable();\n this._touchPan.disable();\n this._el.classList.remove('maplibregl-touch-drag-pan');\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to pan\" interaction is enabled.\n *\n * @returns `true` if the \"drag to pan\" interaction is enabled.\n */\n isEnabled() {\n return this._mousePan.isEnabled() && this._touchPan.isEnabled();\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to pan\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"drag to pan\" interaction is active.\n */\n isActive() {\n return this._mousePan.isActive() || this._touchPan.isActive();\n }\n}\n","import type {MousePitchHandler, MouseRotateHandler} from '../mouse';\n\nexport type DragRotateHandlerOptions = {\n /**\n * Control the map pitch in addition to the bearing\n * @defaultValue true\n */\n pitchWithRotate: boolean;\n}\n\n/**\n * The `DragRotateHandler` allows the user to rotate the map by clicking and\n * dragging the cursor while holding the right mouse button or `ctrl` key.\n *\n * @group Handlers\n */\nexport class DragRotateHandler {\n\n _mouseRotate: MouseRotateHandler;\n _mousePitch: MousePitchHandler;\n _pitchWithRotate: boolean;\n\n /** @internal */\n constructor(options: DragRotateHandlerOptions, mouseRotate: MouseRotateHandler, mousePitch: MousePitchHandler) {\n this._pitchWithRotate = options.pitchWithRotate;\n this._mouseRotate = mouseRotate;\n this._mousePitch = mousePitch;\n }\n\n /**\n * Enables the \"drag to rotate\" interaction.\n *\n * @example\n * ```ts\n * map.dragRotate.enable();\n * ```\n */\n enable() {\n this._mouseRotate.enable();\n if (this._pitchWithRotate) this._mousePitch.enable();\n }\n\n /**\n * Disables the \"drag to rotate\" interaction.\n *\n * @example\n * ```ts\n * map.dragRotate.disable();\n * ```\n */\n disable() {\n this._mouseRotate.disable();\n this._mousePitch.disable();\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to rotate\" interaction is enabled.\n *\n * @returns `true` if the \"drag to rotate\" interaction is enabled.\n */\n isEnabled() {\n return this._mouseRotate.isEnabled() && (!this._pitchWithRotate || this._mousePitch.isEnabled());\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to rotate\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"drag to rotate\" interaction is active.\n */\n isActive() {\n return this._mouseRotate.isActive() || this._mousePitch.isActive();\n }\n}\n","import type {TwoFingersTouchZoomHandler, TwoFingersTouchRotateHandler, AroundCenterOptions} from '../two_fingers_touch';\nimport type {TapDragZoomHandler} from '../tap_drag_zoom';\n\n/**\n * The `TwoFingersTouchZoomRotateHandler` allows the user to zoom and rotate the map by\n * pinching on a touchscreen.\n *\n * They can zoom with one finger by double tapping and dragging. On the second tap,\n * hold the finger down and drag up or down to zoom in or out.\n *\n * @group Handlers\n */\nexport class TwoFingersTouchZoomRotateHandler {\n\n _el: HTMLElement;\n _touchZoom: TwoFingersTouchZoomHandler;\n _touchRotate: TwoFingersTouchRotateHandler;\n _tapDragZoom: TapDragZoomHandler;\n _rotationDisabled: boolean;\n _enabled: boolean;\n\n /** @internal */\n constructor(el: HTMLElement, touchZoom: TwoFingersTouchZoomHandler, touchRotate: TwoFingersTouchRotateHandler, tapDragZoom: TapDragZoomHandler) {\n this._el = el;\n this._touchZoom = touchZoom;\n this._touchRotate = touchRotate;\n this._tapDragZoom = tapDragZoom;\n this._rotationDisabled = false;\n this._enabled = true;\n }\n\n /**\n * Enables the \"pinch to rotate and zoom\" interaction.\n *\n * @param options - Options object.\n *\n * @example\n * ```ts\n * map.touchZoomRotate.enable();\n * map.touchZoomRotate.enable({ around: 'center' });\n * ```\n */\n enable(options?: AroundCenterOptions | boolean | null) {\n this._touchZoom.enable(options);\n if (!this._rotationDisabled) this._touchRotate.enable(options);\n this._tapDragZoom.enable();\n this._el.classList.add('maplibregl-touch-zoom-rotate');\n }\n\n /**\n * Disables the \"pinch to rotate and zoom\" interaction.\n *\n * @example\n * ```ts\n * map.touchZoomRotate.disable();\n * ```\n */\n disable() {\n this._touchZoom.disable();\n this._touchRotate.disable();\n this._tapDragZoom.disable();\n this._el.classList.remove('maplibregl-touch-zoom-rotate');\n }\n\n /**\n * Returns a Boolean indicating whether the \"pinch to rotate and zoom\" interaction is enabled.\n *\n * @returns `true` if the \"pinch to rotate and zoom\" interaction is enabled.\n */\n isEnabled() {\n return this._touchZoom.isEnabled() &&\n (this._rotationDisabled || this._touchRotate.isEnabled()) &&\n this._tapDragZoom.isEnabled();\n }\n\n /**\n * Returns true if the handler is enabled and has detected the start of a zoom/rotate gesture.\n *\n * @returns `true` if the handler is active, `false` otherwise\n */\n isActive() {\n return this._touchZoom.isActive() || this._touchRotate.isActive() || this._tapDragZoom.isActive();\n }\n\n /**\n * Disables the \"pinch to rotate\" interaction, leaving the \"pinch to zoom\"\n * interaction enabled.\n *\n * @example\n * ```ts\n * map.touchZoomRotate.disableRotation();\n * ```\n */\n disableRotation() {\n this._rotationDisabled = true;\n this._touchRotate.disable();\n }\n\n /**\n * Enables the \"pinch to rotate\" interaction.\n *\n * @example\n * ```ts\n * map.touchZoomRotate.enable();\n * map.touchZoomRotate.enableRotation();\n * ```\n */\n enableRotation() {\n this._rotationDisabled = false;\n if (this._touchZoom.isEnabled()) this._touchRotate.enable();\n }\n}\n","import {Event} from '../util/evented';\nimport {DOM} from '../util/dom';\nimport {Map, CompleteMapOptions} from './map';\nimport {HandlerInertia} from './handler_inertia';\nimport {MapEventHandler, BlockableMapEventHandler} from './handler/map_event';\nimport {BoxZoomHandler} from './handler/box_zoom';\nimport {TapZoomHandler} from './handler/tap_zoom';\nimport {generateMouseRotationHandler, generateMousePitchHandler, generateMousePanHandler} from './handler/mouse';\nimport {TouchPanHandler} from './handler/touch_pan';\nimport {TwoFingersTouchZoomHandler, TwoFingersTouchRotateHandler, TwoFingersTouchPitchHandler} from './handler/two_fingers_touch';\nimport {KeyboardHandler} from './handler/keyboard';\nimport {ScrollZoomHandler} from './handler/scroll_zoom';\nimport {DoubleClickZoomHandler} from './handler/shim/dblclick_zoom';\nimport {ClickZoomHandler} from './handler/click_zoom';\nimport {TapDragZoomHandler} from './handler/tap_drag_zoom';\nimport {DragPanHandler} from './handler/shim/drag_pan';\nimport {DragRotateHandler} from './handler/shim/drag_rotate';\nimport {TwoFingersTouchZoomRotateHandler} from './handler/shim/two_fingers_touch';\nimport {extend} from '../util/util';\nimport {browser} from '../util/browser';\nimport Point from '@mapbox/point-geometry';\n\nexport type InputEvent = MouseEvent | TouchEvent | KeyboardEvent | WheelEvent;\n\nconst isMoving = p => p.zoom || p.drag || p.pitch || p.rotate;\n\nclass RenderFrameEvent extends Event {\n type: 'renderFrame';\n timeStamp: number;\n}\n\n/**\n * Handlers interpret dom events and return camera changes that should be\n * applied to the map (`HandlerResult`s). The camera changes are all deltas.\n * The handler itself should have no knowledge of the map's current state.\n * This makes it easier to merge multiple results and keeps handlers simpler.\n * For example, if there is a mousedown and mousemove, the mousePan handler\n * would return a `panDelta` on the mousemove.\n */\nexport interface Handler {\n enable(): void;\n disable(): void;\n isEnabled(): boolean;\n isActive(): boolean;\n /**\n * `reset` can be called by the manager at any time and must reset everything to it's original state\n */\n reset(): void;\n // Handlers can optionally implement these methods.\n // They are called with dom events whenever those dom evens are received.\n readonly touchstart?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly touchmove?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly touchmoveWindow?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly touchend?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly touchcancel?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly mousedown?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly mousemove?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly mousemoveWindow?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly mouseup?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly mouseupWindow?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly dblclick?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly contextmenu?: (e: MouseEvent) => HandlerResult | void;\n readonly wheel?: (e: WheelEvent, point: Point) => HandlerResult | void;\n readonly keydown?: (e: KeyboardEvent) => HandlerResult | void;\n readonly keyup?: (e: KeyboardEvent) => HandlerResult | void;\n /**\n * `renderFrame` is the only non-dom event. It is called during render\n * frames and can be used to smooth camera changes (see scroll handler).\n */\n readonly renderFrame?: () => HandlerResult | void;\n}\n\n/**\n * All handler methods that are called with events can optionally return a `HandlerResult`.\n */\nexport type HandlerResult = {\n panDelta?: Point;\n zoomDelta?: number;\n bearingDelta?: number;\n pitchDelta?: number;\n /**\n * the point to not move when changing the camera\n */\n around?: Point | null;\n /**\n * same as above, except for pinch actions, which are given higher priority\n */\n pinchAround?: Point | null;\n /**\n * A method that can fire a one-off easing by directly changing the map's camera.\n */\n cameraAnimation?: (map: Map) => any;\n /**\n * The last three properties are needed by only one handler: scrollzoom.\n * The DOM event to be used as the `originalEvent` on any camera change events.\n */\n originalEvent?: Event;\n /**\n * Makes the manager trigger a frame, allowing the handler to return multiple results over time (see scrollzoom).\n */\n needsRenderFrame?: boolean;\n /**\n * The camera changes won't get recorded for inertial zooming.\n */\n noInertia?: boolean;\n};\n\nexport type EventInProgress = {\n handlerName: string;\n originalEvent: Event;\n}\n\nexport type EventsInProgress = {\n zoom?: EventInProgress;\n pitch?: EventInProgress;\n rotate?: EventInProgress;\n drag?: EventInProgress;\n}\n\nfunction hasChange(result: HandlerResult) {\n return (result.panDelta && result.panDelta.mag()) || result.zoomDelta || result.bearingDelta || result.pitchDelta;\n}\n\nexport class HandlerManager {\n _map: Map;\n _el: HTMLElement;\n _handlers: Array<{\n handlerName: string;\n handler: Handler;\n allowed: Array;\n }>;\n _eventsInProgress: EventsInProgress;\n _frameId: number;\n _inertia: HandlerInertia;\n _bearingSnap: number;\n _handlersById: {[x: string]: Handler};\n _updatingCamera: boolean;\n _changes: Array<[HandlerResult, EventsInProgress, {[handlerName: string]: Event}]>;\n _terrainMovement: boolean;\n _zoom: {handlerName: string};\n _previousActiveHandlers: {[x: string]: Handler};\n _listeners: Array<[Window | Document | HTMLElement, string, {\n passive?: boolean;\n capture?: boolean;\n } | undefined]>;\n\n constructor(map: Map, options: CompleteMapOptions) {\n this._map = map;\n this._el = this._map.getCanvasContainer();\n this._handlers = [];\n this._handlersById = {};\n this._changes = [];\n\n this._inertia = new HandlerInertia(map);\n this._bearingSnap = options.bearingSnap;\n this._previousActiveHandlers = {};\n\n // Track whether map is currently moving, to compute start/move/end events\n this._eventsInProgress = {};\n\n this._addDefaultHandlers(options);\n\n const el = this._el;\n\n this._listeners = [\n // This needs to be `passive: true` so that a double tap fires two\n // pairs of touchstart/end events in iOS Safari 13. If this is set to\n // `passive: false` then the second pair of events is only fired if\n // preventDefault() is called on the first touchstart. Calling preventDefault()\n // undesirably prevents click events.\n [el, 'touchstart', {passive: true}],\n // This needs to be `passive: false` so that scrolls and pinches can be\n // prevented in browsers that don't support `touch-actions: none`, for example iOS Safari 12.\n [el, 'touchmove', {passive: false}],\n [el, 'touchend', undefined],\n [el, 'touchcancel', undefined],\n\n [el, 'mousedown', undefined],\n [el, 'mousemove', undefined],\n [el, 'mouseup', undefined],\n\n // Bind window-level event listeners for move and up/end events. In the absence of\n // the pointer capture API, which is not supported by all necessary platforms,\n // window-level event listeners give us the best shot at capturing events that\n // fall outside the map canvas element. Use `{capture: true}` for the move event\n // to prevent map move events from being fired during a drag.\n [document, 'mousemove', {capture: true}],\n [document, 'mouseup', undefined],\n\n [el, 'mouseover', undefined],\n [el, 'mouseout', undefined],\n [el, 'dblclick', undefined],\n [el, 'click', undefined],\n\n [el, 'keydown', {capture: false}],\n [el, 'keyup', undefined],\n\n [el, 'wheel', {passive: false}],\n [el, 'contextmenu', undefined],\n\n [window, 'blur', undefined]\n ];\n\n for (const [target, type, listenerOptions] of this._listeners) {\n DOM.addEventListener(target, type, target === document ? this.handleWindowEvent : this.handleEvent, listenerOptions);\n }\n }\n\n destroy() {\n for (const [target, type, listenerOptions] of this._listeners) {\n DOM.removeEventListener(target, type, target === document ? this.handleWindowEvent : this.handleEvent, listenerOptions);\n }\n }\n\n _addDefaultHandlers(options: CompleteMapOptions) {\n const map = this._map;\n const el = map.getCanvasContainer();\n this._add('mapEvent', new MapEventHandler(map, options));\n\n const boxZoom = map.boxZoom = new BoxZoomHandler(map, options);\n this._add('boxZoom', boxZoom);\n if (options.interactive && options.boxZoom) {\n boxZoom.enable();\n }\n\n const tapZoom = new TapZoomHandler(map);\n const clickZoom = new ClickZoomHandler(map);\n map.doubleClickZoom = new DoubleClickZoomHandler(clickZoom, tapZoom);\n this._add('tapZoom', tapZoom);\n this._add('clickZoom', clickZoom);\n if (options.interactive && options.doubleClickZoom) {\n map.doubleClickZoom.enable();\n }\n\n const tapDragZoom = new TapDragZoomHandler();\n this._add('tapDragZoom', tapDragZoom);\n\n const touchPitch = map.touchPitch = new TwoFingersTouchPitchHandler(map);\n this._add('touchPitch', touchPitch);\n if (options.interactive && options.touchPitch) {\n map.touchPitch.enable(options.touchPitch);\n }\n\n const mouseRotate = generateMouseRotationHandler(options);\n const mousePitch = generateMousePitchHandler(options);\n map.dragRotate = new DragRotateHandler(options, mouseRotate, mousePitch);\n this._add('mouseRotate', mouseRotate, ['mousePitch']);\n this._add('mousePitch', mousePitch, ['mouseRotate']);\n if (options.interactive && options.dragRotate) {\n map.dragRotate.enable();\n }\n\n const mousePan = generateMousePanHandler(options);\n const touchPan = new TouchPanHandler(options, map);\n map.dragPan = new DragPanHandler(el, mousePan, touchPan);\n this._add('mousePan', mousePan);\n this._add('touchPan', touchPan, ['touchZoom', 'touchRotate']);\n if (options.interactive && options.dragPan) {\n map.dragPan.enable(options.dragPan);\n }\n\n const touchRotate = new TwoFingersTouchRotateHandler();\n const touchZoom = new TwoFingersTouchZoomHandler();\n map.touchZoomRotate = new TwoFingersTouchZoomRotateHandler(el, touchZoom, touchRotate, tapDragZoom);\n this._add('touchRotate', touchRotate, ['touchPan', 'touchZoom']);\n this._add('touchZoom', touchZoom, ['touchPan', 'touchRotate']);\n if (options.interactive && options.touchZoomRotate) {\n map.touchZoomRotate.enable(options.touchZoomRotate);\n }\n\n const scrollZoom = map.scrollZoom = new ScrollZoomHandler(map, () => this._triggerRenderFrame());\n this._add('scrollZoom', scrollZoom, ['mousePan']);\n if (options.interactive && options.scrollZoom) {\n map.scrollZoom.enable(options.scrollZoom);\n }\n\n const keyboard = map.keyboard = new KeyboardHandler(map);\n this._add('keyboard', keyboard);\n if (options.interactive && options.keyboard) {\n map.keyboard.enable();\n }\n\n this._add('blockableMapEvent', new BlockableMapEventHandler(map));\n }\n\n _add(handlerName: string, handler: Handler, allowed?: Array) {\n this._handlers.push({handlerName, handler, allowed});\n this._handlersById[handlerName] = handler;\n }\n\n stop(allowEndAnimation: boolean) {\n // do nothing if this method was triggered by a gesture update\n if (this._updatingCamera) return;\n\n for (const {handler} of this._handlers) {\n handler.reset();\n }\n this._inertia.clear();\n this._fireEvents({}, {}, allowEndAnimation);\n this._changes = [];\n }\n\n isActive() {\n for (const {handler} of this._handlers) {\n if (handler.isActive()) return true;\n }\n return false;\n }\n\n isZooming() {\n return !!this._eventsInProgress.zoom || this._map.scrollZoom.isZooming();\n }\n isRotating() {\n return !!this._eventsInProgress.rotate;\n }\n\n isMoving() {\n return Boolean(isMoving(this._eventsInProgress)) || this.isZooming();\n }\n\n _blockedByActive(activeHandlers: {[x: string]: Handler}, allowed: Array, myName: string) {\n for (const name in activeHandlers) {\n if (name === myName) continue;\n if (!allowed || allowed.indexOf(name) < 0) {\n return true;\n }\n }\n return false;\n }\n\n handleWindowEvent = (e: { type: 'mousemove' | 'mouseup' | 'touchmove'}) => {\n this.handleEvent(e, `${e.type}Window`);\n };\n\n _getMapTouches(touches: TouchList) {\n const mapTouches = [];\n for (const t of touches) {\n const target = (t.target as any as Node);\n if (this._el.contains(target)) {\n mapTouches.push(t);\n }\n }\n return mapTouches as any as TouchList;\n }\n\n handleEvent = (e: Event, eventName?: keyof Handler) => {\n\n if (e.type === 'blur') {\n this.stop(true);\n return;\n }\n\n this._updatingCamera = true;\n\n const inputEvent = e.type === 'renderFrame' ? undefined : e as InputEvent;\n\n /*\n * We don't call e.preventDefault() for any events by default.\n * Handlers are responsible for calling it where necessary.\n */\n\n const mergedHandlerResult: HandlerResult = {needsRenderFrame: false};\n const eventsInProgress: EventsInProgress = {};\n const activeHandlers = {};\n const eventTouches = (e as TouchEvent).touches;\n\n const mapTouches = eventTouches ? this._getMapTouches(eventTouches) : undefined;\n const points = mapTouches ? DOM.touchPos(this._el, mapTouches) : DOM.mousePos(this._el, ((e as MouseEvent)));\n\n for (const {handlerName, handler, allowed} of this._handlers) {\n if (!handler.isEnabled()) continue;\n\n let data: HandlerResult;\n if (this._blockedByActive(activeHandlers, allowed, handlerName)) {\n handler.reset();\n\n } else {\n if (handler[eventName || e.type]) {\n data = handler[eventName || e.type](e, points, mapTouches);\n this.mergeHandlerResult(mergedHandlerResult, eventsInProgress, data, handlerName, inputEvent);\n if (data && data.needsRenderFrame) {\n this._triggerRenderFrame();\n }\n }\n }\n\n if (data || handler.isActive()) {\n activeHandlers[handlerName] = handler;\n }\n }\n\n const deactivatedHandlers: {[handlerName: string]: Event} = {};\n for (const name in this._previousActiveHandlers) {\n if (!activeHandlers[name]) {\n deactivatedHandlers[name] = inputEvent;\n }\n }\n this._previousActiveHandlers = activeHandlers;\n\n if (Object.keys(deactivatedHandlers).length || hasChange(mergedHandlerResult)) {\n this._changes.push([mergedHandlerResult, eventsInProgress, deactivatedHandlers]);\n this._triggerRenderFrame();\n }\n\n if (Object.keys(activeHandlers).length || hasChange(mergedHandlerResult)) {\n this._map._stop(true);\n }\n\n this._updatingCamera = false;\n\n const {cameraAnimation} = mergedHandlerResult;\n if (cameraAnimation) {\n this._inertia.clear();\n this._fireEvents({}, {}, true);\n this._changes = [];\n cameraAnimation(this._map);\n }\n };\n\n mergeHandlerResult(mergedHandlerResult: HandlerResult,\n eventsInProgress: EventsInProgress,\n handlerResult: HandlerResult,\n name: string,\n e?: InputEvent) {\n if (!handlerResult) return;\n\n extend(mergedHandlerResult, handlerResult);\n\n const eventData = {handlerName: name, originalEvent: handlerResult.originalEvent || e};\n\n // track which handler changed which camera property\n if (handlerResult.zoomDelta !== undefined) {\n eventsInProgress.zoom = eventData;\n }\n if (handlerResult.panDelta !== undefined) {\n eventsInProgress.drag = eventData;\n }\n if (handlerResult.pitchDelta !== undefined) {\n eventsInProgress.pitch = eventData;\n }\n if (handlerResult.bearingDelta !== undefined) {\n eventsInProgress.rotate = eventData;\n }\n\n }\n\n _applyChanges() {\n const combined: HandlerResult = {};\n const combinedEventsInProgress: EventsInProgress = {};\n const combinedDeactivatedHandlers = {};\n\n for (const [change, eventsInProgress, deactivatedHandlers] of this._changes) {\n\n if (change.panDelta) combined.panDelta = (combined.panDelta || new Point(0, 0))._add(change.panDelta);\n if (change.zoomDelta) combined.zoomDelta = (combined.zoomDelta || 0) + change.zoomDelta;\n if (change.bearingDelta) combined.bearingDelta = (combined.bearingDelta || 0) + change.bearingDelta;\n if (change.pitchDelta) combined.pitchDelta = (combined.pitchDelta || 0) + change.pitchDelta;\n if (change.around !== undefined) combined.around = change.around;\n if (change.pinchAround !== undefined) combined.pinchAround = change.pinchAround;\n if (change.noInertia) combined.noInertia = change.noInertia;\n\n extend(combinedEventsInProgress, eventsInProgress);\n extend(combinedDeactivatedHandlers, deactivatedHandlers);\n }\n\n this._updateMapTransform(combined, combinedEventsInProgress, combinedDeactivatedHandlers);\n this._changes = [];\n }\n\n _updateMapTransform(combinedResult: HandlerResult,\n combinedEventsInProgress: EventsInProgress,\n deactivatedHandlers: {[handlerName: string]: Event}) {\n const map = this._map;\n const tr = map._getTransformForUpdate();\n const terrain = map.terrain;\n\n if (!hasChange(combinedResult) && !(terrain && this._terrainMovement)) {\n return this._fireEvents(combinedEventsInProgress, deactivatedHandlers, true);\n }\n\n let {panDelta, zoomDelta, bearingDelta, pitchDelta, around, pinchAround} = combinedResult;\n\n if (pinchAround !== undefined) {\n around = pinchAround;\n }\n\n // stop any ongoing camera animations (easeTo, flyTo)\n map._stop(true);\n\n around = around || map.transform.centerPoint;\n const loc = tr.pointLocation(panDelta ? around.sub(panDelta) : around);\n if (bearingDelta) tr.bearing += bearingDelta;\n if (pitchDelta) tr.pitch += pitchDelta;\n if (zoomDelta) tr.zoom += zoomDelta;\n\n if (!terrain) {\n tr.setLocationAtPoint(loc, around);\n } else {\n // when 3d-terrain is enabled act a little different:\n // - dragging do not drag the picked point itself, instead it drags the map by pixel-delta.\n // With this approach it is no longer possible to pick a point from somewhere near\n // the horizon to the center in one move.\n // So this logic avoids the problem, that in such cases you easily loose orientation.\n if (!this._terrainMovement &&\n (combinedEventsInProgress.drag || combinedEventsInProgress.zoom)) {\n // When starting to drag or move, flag it and register moveend to clear flagging\n this._terrainMovement = true;\n this._map._elevationFreeze = true;\n tr.setLocationAtPoint(loc, around);\n this._map.once('moveend', () => {\n this._map._elevationFreeze = false;\n this._terrainMovement = false;\n tr.recalculateZoom(map.terrain);\n });\n } else if (combinedEventsInProgress.drag && this._terrainMovement) {\n // drag map\n tr.center = tr.pointLocation(tr.centerPoint.sub(panDelta));\n } else {\n tr.setLocationAtPoint(loc, around);\n }\n }\n\n map._applyUpdatedTransform(tr);\n\n this._map._update();\n if (!combinedResult.noInertia) this._inertia.record(combinedResult);\n this._fireEvents(combinedEventsInProgress, deactivatedHandlers, true);\n\n }\n\n _fireEvents(newEventsInProgress: EventsInProgress, deactivatedHandlers: {[handlerName: string]: Event}, allowEndAnimation: boolean) {\n\n const wasMoving = isMoving(this._eventsInProgress);\n const nowMoving = isMoving(newEventsInProgress);\n\n const startEvents = {};\n\n for (const eventName in newEventsInProgress) {\n const {originalEvent} = newEventsInProgress[eventName];\n if (!this._eventsInProgress[eventName]) {\n startEvents[`${eventName}start`] = originalEvent;\n }\n this._eventsInProgress[eventName] = newEventsInProgress[eventName];\n }\n\n // fire start events only after this._eventsInProgress has been updated\n if (!wasMoving && nowMoving) {\n this._fireEvent('movestart', nowMoving.originalEvent);\n }\n\n for (const name in startEvents) {\n this._fireEvent(name, startEvents[name]);\n }\n\n if (nowMoving) {\n this._fireEvent('move', nowMoving.originalEvent);\n }\n\n for (const eventName in newEventsInProgress) {\n const {originalEvent} = newEventsInProgress[eventName];\n this._fireEvent(eventName, originalEvent);\n }\n\n const endEvents = {};\n\n let originalEndEvent;\n for (const eventName in this._eventsInProgress) {\n const {handlerName, originalEvent} = this._eventsInProgress[eventName];\n if (!this._handlersById[handlerName].isActive()) {\n delete this._eventsInProgress[eventName];\n originalEndEvent = deactivatedHandlers[handlerName] || originalEvent;\n endEvents[`${eventName}end`] = originalEndEvent;\n }\n }\n\n for (const name in endEvents) {\n this._fireEvent(name, endEvents[name]);\n }\n\n const stillMoving = isMoving(this._eventsInProgress);\n if (allowEndAnimation && (wasMoving || nowMoving) && !stillMoving) {\n this._updatingCamera = true;\n const inertialEase = this._inertia._onMoveEnd(this._map.dragPan._inertiaOptions);\n\n const shouldSnapToNorth = bearing => bearing !== 0 && -this._bearingSnap < bearing && bearing < this._bearingSnap;\n\n if (inertialEase && (inertialEase.essential || !browser.prefersReducedMotion)) {\n if (shouldSnapToNorth(inertialEase.bearing || this._map.getBearing())) {\n inertialEase.bearing = 0;\n }\n inertialEase.freezeElevation = true;\n this._map.easeTo(inertialEase, {originalEvent: originalEndEvent});\n } else {\n this._map.fire(new Event('moveend', {originalEvent: originalEndEvent}));\n if (shouldSnapToNorth(this._map.getBearing())) {\n this._map.resetNorth();\n }\n }\n this._updatingCamera = false;\n }\n\n }\n\n _fireEvent(type: string, e?: Event) {\n this._map.fire(new Event(type, e ? {originalEvent: e} : {}));\n }\n\n _requestFrame() {\n this._map.triggerRepaint();\n return this._map._renderTaskQueue.add(timeStamp => {\n delete this._frameId;\n this.handleEvent(new RenderFrameEvent('renderFrame', {timeStamp}));\n this._applyChanges();\n });\n }\n\n _triggerRenderFrame() {\n if (this._frameId === undefined) {\n this._frameId = this._requestFrame();\n }\n }\n}\n","import {extend, warnOnce, clamp, wrap, defaultEasing, pick} from '../util/util';\nimport {interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {browser} from '../util/browser';\nimport {LngLat} from '../geo/lng_lat';\nimport {LngLatBounds} from '../geo/lng_lat_bounds';\nimport Point from '@mapbox/point-geometry';\nimport {Event, Evented} from '../util/evented';\nimport {Terrain} from '../render/terrain';\n\nimport type {Transform} from '../geo/transform';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport type {LngLatBoundsLike} from '../geo/lng_lat_bounds';\nimport type {TaskID} from '../util/task_queue';\nimport type {PaddingOptions} from '../geo/edge_insets';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\n\n/**\n * A [Point](https://github.com/mapbox/point-geometry) or an array of two numbers representing `x` and `y` screen coordinates in pixels.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let p1 = new Point(-77, 38); // a PointLike which is a Point\n * let p2 = [-77, 38]; // a PointLike which is an array of two numbers\n * ```\n */\nexport type PointLike = Point | [number, number];\n\n/**\n * A helper to allow require of at least one propery\n */\nexport type RequireAtLeastOne = { [K in keyof T]-?: Required> & Partial>>; }[keyof T];\n\n/**\n * Options common to {@link Map#jumpTo}, {@link Map#easeTo}, and {@link Map#flyTo}, controlling the desired location,\n * zoom, bearing, and pitch of the camera. All properties are optional, and when a property is omitted, the current\n * camera value for that property will remain unchanged.\n *\n * @example\n * Set the map's initial perspective with CameraOptions\n * ```ts\n * let map = new maplibregl.Map({\n * container: 'map',\n * style: 'https://demotiles.maplibre.org/style.json',\n * center: [-73.5804, 45.53483],\n * pitch: 60,\n * bearing: -60,\n * zoom: 10\n * });\n * ```\n * @see [Set pitch and bearing](https://maplibre.org/maplibre-gl-js/docs/examples/set-perspective/)\n * @see [Jump to a series of locations](https://maplibre.org/maplibre-gl-js/docs/examples/jump-to/)\n * @see [Fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto/)\n * @see [Display buildings in 3D](https://maplibre.org/maplibre-gl-js/docs/examples/3d-buildings/)\n */\nexport type CameraOptions = CenterZoomBearing & {\n /**\n * The desired pitch in degrees. The pitch is the angle towards the horizon\n * measured in degrees with a range between 0 and 60 degrees. For example, pitch: 0 provides the appearance\n * of looking straight down at the map, while pitch: 60 tilts the user's perspective towards the horizon.\n * Increasing the pitch value is often used to display 3D objects.\n */\n pitch?: number;\n /**\n * If `zoom` is specified, `around` determines the point around which the zoom is centered.\n */\n around?: LngLatLike;\n};\n\n/**\n * Holds center, zoom and bearing properties\n */\nexport type CenterZoomBearing = {\n /**\n * The desired center.\n */\n center?: LngLatLike;\n /**\n * The desired zoom level.\n */\n zoom?: number;\n /**\n * The desired bearing in degrees. The bearing is the compass direction that\n * is \"up\". For example, `bearing: 90` orients the map so that east is up.\n */\n bearing?: number;\n}\n\n/**\n * The options object related to the {@link Map#jumpTo} method\n */\nexport type JumpToOptions = CameraOptions & {\n /**\n * Dimensions in pixels applied on each side of the viewport for shifting the vanishing point.\n */\n padding?: PaddingOptions;\n}\n\n/**\n * A options object for the {@link Map#cameraForBounds} method\n */\nexport type CameraForBoundsOptions = CameraOptions & {\n /**\n * The amount of padding in pixels to add to the given bounds.\n */\n padding?: number | RequireAtLeastOne;\n /**\n * The center of the given bounds relative to the map's center, measured in pixels.\n * @defaultValue [0, 0]\n */\n offset?: PointLike;\n /**\n * The maximum zoom level to allow when the camera would transition to the specified bounds.\n */\n maxZoom?: number;\n}\n\n/**\n * The {@link Map#flyTo} options object\n */\nexport type FlyToOptions = AnimationOptions & CameraOptions & {\n /**\n * The zooming \"curve\" that will occur along the\n * flight path. A high value maximizes zooming for an exaggerated animation, while a low\n * value minimizes zooming for an effect closer to {@link Map#easeTo}. 1.42 is the average\n * value selected by participants in the user study discussed in\n * [van Wijk (2003)](https://www.win.tue.nl/~vanwijk/zoompan.pdf). A value of\n * `Math.pow(6, 0.25)` would be equivalent to the root mean squared average velocity. A\n * value of 1 would produce a circular motion.\n * @defaultValue 1.42\n */\n curve?: number;\n /**\n * The zero-based zoom level at the peak of the flight path. If\n * `options.curve` is specified, this option is ignored.\n */\n minZoom?: number;\n /**\n * The average speed of the animation defined in relation to\n * `options.curve`. A speed of 1.2 means that the map appears to move along the flight path\n * by 1.2 times `options.curve` screenfuls every second. A _screenful_ is the map's visible span.\n * It does not correspond to a fixed physical distance, but varies by zoom level.\n * @defaultValue 1.2\n */\n speed?: number;\n /**\n * The average speed of the animation measured in screenfuls\n * per second, assuming a linear timing curve. If `options.speed` is specified, this option is ignored.\n */\n screenSpeed?: number;\n /**\n * The animation's maximum duration, measured in milliseconds.\n * If duration exceeds maximum duration, it resets to 0.\n */\n maxDuration?: number;\n /**\n * The amount of padding in pixels to add to the given bounds.\n */\n padding?: number | RequireAtLeastOne;\n}\n\nexport type EaseToOptions = AnimationOptions & CameraOptions & {\n delayEndEvents?: number;\n padding?: number | RequireAtLeastOne;\n}\n\n/**\n * Options for {@link Map#fitBounds} method\n */\nexport type FitBoundsOptions = FlyToOptions & {\n /**\n * If `true`, the map transitions using {@link Map#easeTo}. If `false`, the map transitions using {@link Map#flyTo}.\n * See those functions and {@link AnimationOptions} for information about options available.\n * @defaultValue false\n */\n linear?: boolean;\n /**\n * The center of the given bounds relative to the map's center, measured in pixels.\n * @defaultValue [0, 0]\n */\n offset?: PointLike;\n /**\n * The maximum zoom level to allow when the map view transitions to the specified bounds.\n */\n maxZoom?: number;\n}\n\n/**\n * Options common to map movement methods that involve animation, such as {@link Map#panBy} and\n * {@link Map#easeTo}, controlling the duration and easing function of the animation. All properties\n * are optional.\n *\n */\nexport type AnimationOptions = {\n /**\n * The animation's duration, measured in milliseconds.\n */\n duration?: number;\n /**\n * A function taking a time in the range 0..1 and returning a number where 0 is\n * the initial state and 1 is the final state.\n */\n easing?: (_: number) => number;\n /**\n * of the target center relative to real map container center at the end of animation.\n */\n offset?: PointLike;\n /**\n * If `false`, no animation will occur.\n */\n animate?: boolean;\n /**\n * If `true`, then the animation is considered essential and will not be affected by\n * [`prefers-reduced-motion`](https://developer.mozilla.org/en-US/docs/Web/CSS/\\@media/prefers-reduced-motion).\n */\n essential?: boolean;\n /**\n * Default false. Needed in 3D maps to let the camera stay in a constant\n * height based on sea-level. After the animation finished the zoom-level will be recalculated in respect of\n * the distance from the camera to the center-coordinate-altitude.\n */\n freezeElevation?: boolean;\n};\n\n/**\n * A callback hook that allows manipulating the camera and being notified about camera updates before they happen\n */\nexport type CameraUpdateTransformFunction = (next: {\n center: LngLat;\n zoom: number;\n pitch: number;\n bearing: number;\n elevation: number;\n}) => {\n center?: LngLat;\n zoom?: number;\n pitch?: number;\n bearing?: number;\n elevation?: number;\n};\n\nexport abstract class Camera extends Evented {\n transform: Transform;\n terrain: Terrain;\n\n _moving: boolean;\n _zooming: boolean;\n _rotating: boolean;\n _pitching: boolean;\n _padding: boolean;\n\n _bearingSnap: number;\n _easeStart: number;\n _easeOptions: {\n duration?: number;\n easing?: (_: number) => number;\n };\n _easeId: string | void;\n\n _onEaseFrame: (_: number) => void;\n _onEaseEnd: (easeId?: string) => void;\n _easeFrameId: TaskID;\n\n /**\n * @internal\n * holds the geographical coordinate of the target\n */\n _elevationCenter: LngLat;\n /**\n * @internal\n * holds the targ altitude value, = center elevation of the target.\n * This value may changes during flight, because new terrain-tiles loads during flight.\n */\n _elevationTarget: number;\n /**\n * @internal\n * holds the start altitude value, = center elevation before animation begins\n * this value will recalculated during flight in respect of changing _elevationTarget values,\n * so the linear interpolation between start and target keeps smooth and without jumps.\n */\n _elevationStart: number;\n /**\n * @internal\n * Saves the current state of the elevation freeze - this is used during map movement to prevent \"rocky\" camera movement.\n */\n _elevationFreeze: boolean;\n /**\n * @internal\n * Used to track accumulated changes during continuous interaction\n */\n _requestedCameraState?: Transform;\n /**\n * A callback used to defer camera updates or apply arbitrary constraints.\n * If specified, this Camera instance can be used as a stateless component in React etc.\n */\n transformCameraUpdate: CameraUpdateTransformFunction | null;\n\n abstract _requestRenderFrame(a: () => void): TaskID;\n abstract _cancelRenderFrame(_: TaskID): void;\n\n constructor(transform: Transform, options: {\n bearingSnap: number;\n }) {\n super();\n this._moving = false;\n this._zooming = false;\n this.transform = transform;\n this._bearingSnap = options.bearingSnap;\n\n this.on('moveend', () => {\n delete this._requestedCameraState;\n });\n }\n\n /**\n * Returns the map's geographical centerpoint.\n *\n * @returns The map's geographical centerpoint.\n * @example\n * Return a LngLat object such as `{lng: 0, lat: 0}`\n * ```ts\n * let center = map.getCenter();\n * // access longitude and latitude values directly\n * let {lng, lat} = map.getCenter();\n * ```\n */\n getCenter(): LngLat { return new LngLat(this.transform.center.lng, this.transform.center.lat); }\n\n /**\n * Sets the map's geographical centerpoint. Equivalent to `jumpTo({center: center})`.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param center - The centerpoint to set.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * map.setCenter([-74, 38]);\n * ```\n */\n setCenter(center: LngLatLike, eventData?: any) {\n return this.jumpTo({center}, eventData);\n }\n\n /**\n * Pans the map by the specified offset.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param offset - `x` and `y` coordinates by which to pan the map.\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/)\n */\n panBy(offset: PointLike, options?: AnimationOptions, eventData?: any): this {\n offset = Point.convert(offset).mult(-1);\n return this.panTo(this.transform.center, extend({offset}, options), eventData);\n }\n\n /**\n * Pans the map to the specified location with an animated transition.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param lnglat - The location to pan the map to.\n * @param options - Options describing the destination and animation of the transition.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * map.panTo([-74, 38]);\n * // Specify that the panTo animation should last 5000 milliseconds.\n * map.panTo([-74, 38], {duration: 5000});\n * ```\n * @see [Update a feature in realtime](https://maplibre.org/maplibre-gl-js/docs/examples/live-update-feature/)\n */\n panTo(lnglat: LngLatLike, options?: AnimationOptions, eventData?: any): this {\n return this.easeTo(extend({\n center: lnglat\n }, options), eventData);\n }\n\n /**\n * Returns the map's current zoom level.\n *\n * @returns The map's current zoom level.\n * @example\n * ```ts\n * map.getZoom();\n * ```\n */\n getZoom(): number { return this.transform.zoom; }\n\n /**\n * Sets the map's zoom level. Equivalent to `jumpTo({zoom: zoom})`.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`.\n *\n * @param zoom - The zoom level to set (0-20).\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Zoom to the zoom level 5 without an animated transition\n * ```ts\n * map.setZoom(5);\n * ```\n */\n setZoom(zoom: number, eventData?: any): this {\n this.jumpTo({zoom}, eventData);\n return this;\n }\n\n /**\n * Zooms the map to the specified zoom level, with an animated transition.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`.\n *\n * @param zoom - The zoom level to transition to.\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * // Zoom to the zoom level 5 without an animated transition\n * map.zoomTo(5);\n * // Zoom to the zoom level 8 with an animated transition\n * map.zoomTo(8, {\n * duration: 2000,\n * offset: [100, 50]\n * });\n * ```\n */\n zoomTo(zoom: number, options?: AnimationOptions | null, eventData?: any): this {\n return this.easeTo(extend({\n zoom\n }, options), eventData);\n }\n\n /**\n * Increases the map's zoom level by 1.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Zoom the map in one level with a custom animation duration\n * ```ts\n * map.zoomIn({duration: 1000});\n * ```\n */\n zoomIn(options?: AnimationOptions, eventData?: any): this {\n this.zoomTo(this.getZoom() + 1, options, eventData);\n return this;\n }\n\n /**\n * Decreases the map's zoom level by 1.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Zoom the map out one level with a custom animation offset\n * ```ts\n * map.zoomOut({offset: [80, 60]});\n * ```\n */\n zoomOut(options?: AnimationOptions, eventData?: any): this {\n this.zoomTo(this.getZoom() - 1, options, eventData);\n return this;\n }\n\n /**\n * Returns the map's current bearing. The bearing is the compass direction that is \"up\"; for example, a bearing\n * of 90° orients the map so that east is up.\n *\n * @returns The map's current bearing.\n * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/)\n */\n getBearing(): number { return this.transform.bearing; }\n\n /**\n * Sets the map's bearing (rotation). The bearing is the compass direction that is \"up\"; for example, a bearing\n * of 90° orients the map so that east is up.\n *\n * Equivalent to `jumpTo({bearing: bearing})`.\n *\n * Triggers the following events: `movestart`, `moveend`, and `rotate`.\n *\n * @param bearing - The desired bearing.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Rotate the map to 90 degrees\n * ```ts\n * map.setBearing(90);\n * ```\n */\n setBearing(bearing: number, eventData?: any): this {\n this.jumpTo({bearing}, eventData);\n return this;\n }\n\n /**\n * Returns the current padding applied around the map viewport.\n *\n * @returns The current padding around the map viewport.\n */\n getPadding(): PaddingOptions { return this.transform.padding; }\n\n /**\n * Sets the padding in pixels around the viewport.\n *\n * Equivalent to `jumpTo({padding: padding})`.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param padding - The desired padding.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Sets a left padding of 300px, and a top padding of 50px\n * ```ts\n * map.setPadding({ left: 300, top: 50 });\n * ```\n */\n setPadding(padding: PaddingOptions, eventData?: any): this {\n this.jumpTo({padding}, eventData);\n return this;\n }\n\n /**\n * Rotates the map to the specified bearing, with an animated transition. The bearing is the compass direction\n * that is \"up\"; for example, a bearing of 90° orients the map so that east is up.\n *\n * Triggers the following events: `movestart`, `moveend`, and `rotate`.\n *\n * @param bearing - The desired bearing.\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n rotateTo(bearing: number, options?: AnimationOptions, eventData?: any): this {\n return this.easeTo(extend({\n bearing\n }, options), eventData);\n }\n\n /**\n * Rotates the map so that north is up (0° bearing), with an animated transition.\n *\n * Triggers the following events: `movestart`, `moveend`, and `rotate`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n resetNorth(options?: AnimationOptions, eventData?: any): this {\n this.rotateTo(0, extend({duration: 1000}, options), eventData);\n return this;\n }\n\n /**\n * Rotates and pitches the map so that north is up (0° bearing) and pitch is 0°, with an animated transition.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `pitchstart`, `pitch`, `pitchend`, and `rotate`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n resetNorthPitch(options?: AnimationOptions, eventData?: any): this {\n this.easeTo(extend({\n bearing: 0,\n pitch: 0,\n duration: 1000\n }, options), eventData);\n return this;\n }\n\n /**\n * Snaps the map so that north is up (0° bearing), if the current bearing is close enough to it (i.e. within the\n * `bearingSnap` threshold).\n *\n * Triggers the following events: `movestart`, `moveend`, and `rotate`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n snapToNorth(options?: AnimationOptions, eventData?: any): this {\n if (Math.abs(this.getBearing()) < this._bearingSnap) {\n return this.resetNorth(options, eventData);\n }\n return this;\n }\n\n /**\n * Returns the map's current pitch (tilt).\n *\n * @returns The map's current pitch, measured in degrees away from the plane of the screen.\n */\n getPitch(): number { return this.transform.pitch; }\n\n /**\n * Sets the map's pitch (tilt). Equivalent to `jumpTo({pitch: pitch})`.\n *\n * Triggers the following events: `movestart`, `moveend`, `pitchstart`, and `pitchend`.\n *\n * @param pitch - The pitch to set, measured in degrees away from the plane of the screen (0-60).\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n setPitch(pitch: number, eventData?: any): this {\n this.jumpTo({pitch}, eventData);\n return this;\n }\n\n /**\n * @param bounds - Calculate the center for these bounds in the viewport and use\n * the highest zoom level up to and including `Map#getMaxZoom()` that fits\n * in the viewport. LngLatBounds represent a box that is always axis-aligned with bearing 0.\n * @param options - Options object\n * @returns If map is able to fit to provided bounds, returns `center`, `zoom`, and `bearing`.\n * If map is unable to fit, method will warn and return undefined.\n * @example\n * ```ts\n * let bbox = [[-79, 43], [-73, 45]];\n * let newCameraTransform = map.cameraForBounds(bbox, {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n */\n cameraForBounds(bounds: LngLatBoundsLike, options?: CameraForBoundsOptions): CenterZoomBearing {\n bounds = LngLatBounds.convert(bounds);\n const bearing = options && options.bearing || 0;\n return this._cameraForBoxAndBearing(bounds.getNorthWest(), bounds.getSouthEast(), bearing, options);\n }\n\n /**\n * @internal\n * Calculate the center of these two points in the viewport and use\n * the highest zoom level up to and including `Map#getMaxZoom()` that fits\n * the points in the viewport at the specified bearing.\n * @param p0 - First point\n * @param p1 - Second point\n * @param bearing - Desired map bearing at end of animation, in degrees\n * @param options - the camera options\n * @returns If map is able to fit to provided bounds, returns `center`, `zoom`, and `bearing`.\n * If map is unable to fit, method will warn and return undefined.\n * @example\n * ```ts\n * let p0 = [-79, 43];\n * let p1 = [-73, 45];\n * let bearing = 90;\n * let newCameraTransform = map._cameraForBoxAndBearing(p0, p1, bearing, {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n */\n _cameraForBoxAndBearing(p0: LngLatLike, p1: LngLatLike, bearing: number, options?: CameraForBoundsOptions): CenterZoomBearing {\n const defaultPadding = {\n top: 0,\n bottom: 0,\n right: 0,\n left: 0\n };\n options = extend({\n padding: defaultPadding,\n offset: [0, 0],\n maxZoom: this.transform.maxZoom\n }, options);\n\n if (typeof options.padding === 'number') {\n const p = options.padding;\n options.padding = {\n top: p,\n bottom: p,\n right: p,\n left: p\n };\n }\n\n options.padding = extend(defaultPadding, options.padding) as PaddingOptions;\n const tr = this.transform;\n const edgePadding = tr.padding;\n\n // We want to calculate the upper right and lower left of the box defined by p0 and p1\n // in a coordinate system rotate to match the destination bearing.\n const p0world = tr.project(LngLat.convert(p0));\n const p1world = tr.project(LngLat.convert(p1));\n const p0rotated = p0world.rotate(-bearing * Math.PI / 180);\n const p1rotated = p1world.rotate(-bearing * Math.PI / 180);\n\n const upperRight = new Point(Math.max(p0rotated.x, p1rotated.x), Math.max(p0rotated.y, p1rotated.y));\n const lowerLeft = new Point(Math.min(p0rotated.x, p1rotated.x), Math.min(p0rotated.y, p1rotated.y));\n\n // Calculate zoom: consider the original bbox and padding.\n const size = upperRight.sub(lowerLeft);\n const scaleX = (tr.width - (edgePadding.left + edgePadding.right + options.padding.left + options.padding.right)) / size.x;\n const scaleY = (tr.height - (edgePadding.top + edgePadding.bottom + options.padding.top + options.padding.bottom)) / size.y;\n\n if (scaleY < 0 || scaleX < 0) {\n warnOnce(\n 'Map cannot fit within canvas with the given bounds, padding, and/or offset.'\n );\n return undefined;\n }\n\n const zoom = Math.min(tr.scaleZoom(tr.scale * Math.min(scaleX, scaleY)), options.maxZoom);\n\n // Calculate center: apply the zoom, the configured offset, as well as offset that exists as a result of padding.\n const offset = Point.convert(options.offset);\n const paddingOffsetX = (options.padding.left - options.padding.right) / 2;\n const paddingOffsetY = (options.padding.top - options.padding.bottom) / 2;\n const paddingOffset = new Point(paddingOffsetX, paddingOffsetY);\n const rotatedPaddingOffset = paddingOffset.rotate(bearing * Math.PI / 180);\n const offsetAtInitialZoom = offset.add(rotatedPaddingOffset);\n const offsetAtFinalZoom = offsetAtInitialZoom.mult(tr.scale / tr.zoomScale(zoom));\n\n const center = tr.unproject(p0world.add(p1world).div(2).sub(offsetAtFinalZoom));\n\n return {\n center,\n zoom,\n bearing\n };\n }\n\n /**\n * Pans and zooms the map to contain its visible area within the specified geographical bounds.\n * This function will also reset the map's bearing to 0 if bearing is nonzero.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param bounds - Center these bounds in the viewport and use the highest\n * zoom level up to and including `Map#getMaxZoom()` that fits them in the viewport.\n * @param options- Options supports all properties from {@link AnimationOptions} and {@link CameraOptions} in addition to the fields below.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * let bbox = [[-79, 43], [-73, 45]];\n * map.fitBounds(bbox, {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n * @see [Fit a map to a bounding box](https://maplibre.org/maplibre-gl-js/docs/examples/fitbounds/)\n */\n fitBounds(bounds: LngLatBoundsLike, options?: FitBoundsOptions, eventData?: any): this {\n return this._fitInternal(\n this.cameraForBounds(bounds, options),\n options,\n eventData);\n }\n\n /**\n * Pans, rotates and zooms the map to to fit the box made by points p0 and p1\n * once the map is rotated to the specified bearing. To zoom without rotating,\n * pass in the current map bearing.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend` and `rotate`.\n *\n * @param p0 - First point on screen, in pixel coordinates\n * @param p1 - Second point on screen, in pixel coordinates\n * @param bearing - Desired map bearing at end of animation, in degrees\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * let p0 = [220, 400];\n * let p1 = [500, 900];\n * map.fitScreenCoordinates(p0, p1, map.getBearing(), {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n * @see Used by {@link BoxZoomHandler}\n */\n fitScreenCoordinates(p0: PointLike, p1: PointLike, bearing: number, options?: FitBoundsOptions, eventData?: any): this {\n return this._fitInternal(\n this._cameraForBoxAndBearing(\n this.transform.pointLocation(Point.convert(p0)),\n this.transform.pointLocation(Point.convert(p1)),\n bearing,\n options),\n options,\n eventData);\n }\n\n _fitInternal(calculatedOptions?: CenterZoomBearing, options?: FitBoundsOptions, eventData?: any): this {\n // cameraForBounds warns + returns undefined if unable to fit:\n if (!calculatedOptions) return this;\n\n options = extend(calculatedOptions, options);\n // Explicitly remove the padding field because, calculatedOptions already accounts for padding by setting zoom and center accordingly.\n delete options.padding;\n\n return options.linear ?\n this.easeTo(options, eventData) :\n this.flyTo(options, eventData);\n }\n\n /**\n * Changes any combination of center, zoom, bearing, and pitch, without\n * an animated transition. The map will retain its current values for any\n * details not specified in `options`.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`,\n * `pitch`, `pitchend`, and `rotate`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * // jump to coordinates at current zoom\n * map.jumpTo({center: [0, 0]});\n * // jump with zoom, pitch, and bearing options\n * map.jumpTo({\n * center: [0, 0],\n * zoom: 8,\n * pitch: 45,\n * bearing: 90\n * });\n * ```\n * @see [Jump to a series of locations](https://maplibre.org/maplibre-gl-js/docs/examples/jump-to/)\n * @see [Update a feature in realtime](https://maplibre.org/maplibre-gl-js/docs/examples/live-update-feature/)\n */\n jumpTo(options: JumpToOptions, eventData?: any): this {\n this.stop();\n\n const tr = this._getTransformForUpdate();\n let zoomChanged = false,\n bearingChanged = false,\n pitchChanged = false;\n\n if ('zoom' in options && tr.zoom !== +options.zoom) {\n zoomChanged = true;\n tr.zoom = +options.zoom;\n }\n\n if (options.center !== undefined) {\n tr.center = LngLat.convert(options.center);\n }\n\n if ('bearing' in options && tr.bearing !== +options.bearing) {\n bearingChanged = true;\n tr.bearing = +options.bearing;\n }\n\n if ('pitch' in options && tr.pitch !== +options.pitch) {\n pitchChanged = true;\n tr.pitch = +options.pitch;\n }\n\n if (options.padding != null && !tr.isPaddingEqual(options.padding)) {\n tr.padding = options.padding;\n }\n this._applyUpdatedTransform(tr);\n\n this.fire(new Event('movestart', eventData))\n .fire(new Event('move', eventData));\n\n if (zoomChanged) {\n this.fire(new Event('zoomstart', eventData))\n .fire(new Event('zoom', eventData))\n .fire(new Event('zoomend', eventData));\n }\n\n if (bearingChanged) {\n this.fire(new Event('rotatestart', eventData))\n .fire(new Event('rotate', eventData))\n .fire(new Event('rotateend', eventData));\n }\n\n if (pitchChanged) {\n this.fire(new Event('pitchstart', eventData))\n .fire(new Event('pitch', eventData))\n .fire(new Event('pitchend', eventData));\n }\n\n return this.fire(new Event('moveend', eventData));\n }\n\n /**\n * Calculates pitch, zoom and bearing for looking at `newCenter` with the camera position being `newCenter`\n * and returns them as {@link CameraOptions}.\n * @param from - The camera to look from\n * @param altitudeFrom - The altitude of the camera to look from\n * @param to - The center to look at\n * @param altitudeTo - Optional altitude of the center to look at. If none given the ground height will be used.\n * @returns the calculated camera options\n */\n calculateCameraOptionsFromTo(from: LngLat, altitudeFrom: number, to: LngLat, altitudeTo: number = 0): CameraOptions {\n const fromMerc = MercatorCoordinate.fromLngLat(from, altitudeFrom);\n const toMerc = MercatorCoordinate.fromLngLat(to, altitudeTo);\n const dx = toMerc.x - fromMerc.x;\n const dy = toMerc.y - fromMerc.y;\n const dz = toMerc.z - fromMerc.z;\n\n const distance3D = Math.hypot(dx, dy, dz);\n if (distance3D === 0) throw new Error('Can\\'t calculate camera options with same From and To');\n\n const groundDistance = Math.hypot(dx, dy);\n\n const zoom = this.transform.scaleZoom(this.transform.cameraToCenterDistance / distance3D / this.transform.tileSize);\n const bearing = (Math.atan2(dx, -dy) * 180) / Math.PI;\n let pitch = (Math.acos(groundDistance / distance3D) * 180) / Math.PI;\n pitch = dz < 0 ? 90 - pitch : 90 + pitch;\n\n return {\n center: toMerc.toLngLat(),\n zoom,\n pitch,\n bearing\n };\n }\n\n /**\n * Changes any combination of `center`, `zoom`, `bearing`, `pitch`, and `padding` with an animated transition\n * between old and new values. The map will retain its current values for any\n * details not specified in `options`.\n *\n * Note: The transition will happen instantly if the user has enabled\n * the `reduced motion` accessibility feature enabled in their operating system,\n * unless `options` includes `essential: true`.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`,\n * `pitch`, `pitchend`, and `rotate`.\n *\n * @param options - Options describing the destination and animation of the transition.\n * Accepts {@link CameraOptions} and {@link AnimationOptions}.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/)\n */\n easeTo(options: EaseToOptions & {\n easeId?: string;\n noMoveStart?: boolean;\n }, eventData?: any): this {\n this._stop(false, options.easeId);\n\n options = extend({\n offset: [0, 0],\n duration: 500,\n easing: defaultEasing\n }, options);\n\n if (options.animate === false || (!options.essential && browser.prefersReducedMotion)) options.duration = 0;\n\n const tr = this._getTransformForUpdate(),\n startZoom = this.getZoom(),\n startBearing = this.getBearing(),\n startPitch = this.getPitch(),\n startPadding = this.getPadding(),\n\n zoom = 'zoom' in options ? +options.zoom : startZoom,\n bearing = 'bearing' in options ? this._normalizeBearing(options.bearing, startBearing) : startBearing,\n pitch = 'pitch' in options ? +options.pitch : startPitch,\n padding = 'padding' in options ? options.padding : tr.padding;\n\n const offsetAsPoint = Point.convert(options.offset);\n let pointAtOffset = tr.centerPoint.add(offsetAsPoint);\n const locationAtOffset = tr.pointLocation(pointAtOffset);\n const center = LngLat.convert(options.center || locationAtOffset);\n this._normalizeCenter(center);\n\n const from = tr.project(locationAtOffset);\n const delta = tr.project(center).sub(from);\n const finalScale = tr.zoomScale(zoom - startZoom);\n\n let around, aroundPoint;\n\n if (options.around) {\n around = LngLat.convert(options.around);\n aroundPoint = tr.locationPoint(around);\n }\n\n const currently = {\n moving: this._moving,\n zooming: this._zooming,\n rotating: this._rotating,\n pitching: this._pitching\n };\n\n this._zooming = this._zooming || (zoom !== startZoom);\n this._rotating = this._rotating || (startBearing !== bearing);\n this._pitching = this._pitching || (pitch !== startPitch);\n this._padding = !tr.isPaddingEqual(padding as PaddingOptions);\n\n this._easeId = options.easeId;\n this._prepareEase(eventData, options.noMoveStart, currently);\n if (this.terrain) this._prepareElevation(center);\n\n this._ease((k) => {\n if (this._zooming) {\n tr.zoom = interpolates.number(startZoom, zoom, k);\n }\n if (this._rotating) {\n tr.bearing = interpolates.number(startBearing, bearing, k);\n }\n if (this._pitching) {\n tr.pitch = interpolates.number(startPitch, pitch, k);\n }\n if (this._padding) {\n tr.interpolatePadding(startPadding, padding as PaddingOptions, k);\n // When padding is being applied, Transform#centerPoint is changing continuously,\n // thus we need to recalculate offsetPoint every frame\n pointAtOffset = tr.centerPoint.add(offsetAsPoint);\n }\n\n if (this.terrain && !options.freezeElevation) this._updateElevation(k);\n\n if (around) {\n tr.setLocationAtPoint(around, aroundPoint);\n } else {\n const scale = tr.zoomScale(tr.zoom - startZoom);\n const base = zoom > startZoom ?\n Math.min(2, finalScale) :\n Math.max(0.5, finalScale);\n const speedup = Math.pow(base, 1 - k);\n const newCenter = tr.unproject(from.add(delta.mult(k * speedup)).mult(scale));\n tr.setLocationAtPoint(tr.renderWorldCopies ? newCenter.wrap() : newCenter, pointAtOffset);\n }\n\n this._applyUpdatedTransform(tr);\n\n this._fireMoveEvents(eventData);\n\n }, (interruptingEaseId?: string) => {\n if (this.terrain) this._finalizeElevation();\n this._afterEase(eventData, interruptingEaseId);\n }, options as any);\n\n return this;\n }\n\n _prepareEase(eventData: any, noMoveStart: boolean, currently: any = {}) {\n this._moving = true;\n if (!noMoveStart && !currently.moving) {\n this.fire(new Event('movestart', eventData));\n }\n if (this._zooming && !currently.zooming) {\n this.fire(new Event('zoomstart', eventData));\n }\n if (this._rotating && !currently.rotating) {\n this.fire(new Event('rotatestart', eventData));\n }\n if (this._pitching && !currently.pitching) {\n this.fire(new Event('pitchstart', eventData));\n }\n }\n\n _prepareElevation(center: LngLat) {\n this._elevationCenter = center;\n this._elevationStart = this.transform.elevation;\n this._elevationTarget = this.terrain.getElevationForLngLatZoom(center, this.transform.tileZoom);\n this._elevationFreeze = true;\n }\n\n _updateElevation(k: number) {\n this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this._elevationCenter, this.transform.tileZoom);\n const elevation = this.terrain.getElevationForLngLatZoom(this._elevationCenter, this.transform.tileZoom);\n // target terrain updated during flight, slowly move camera to new height\n if (k < 1 && elevation !== this._elevationTarget) {\n const pitch1 = this._elevationTarget - this._elevationStart;\n const pitch2 = (elevation - (pitch1 * k + this._elevationStart)) / (1 - k);\n this._elevationStart += k * (pitch1 - pitch2);\n this._elevationTarget = elevation;\n }\n this.transform.elevation = interpolates.number(this._elevationStart, this._elevationTarget, k);\n }\n\n _finalizeElevation() {\n this._elevationFreeze = false;\n this.transform.recalculateZoom(this.terrain);\n }\n\n /**\n * @internal\n * Called when the camera is about to be manipulated.\n * If `transformCameraUpdate` is specified, a copy of the current transform is created to track the accumulated changes.\n * This underlying transform represents the \"desired state\" proposed by input handlers / animations / UI controls.\n * It may differ from the state used for rendering (`this.transform`).\n * @returns Transform to apply changes to\n */\n _getTransformForUpdate(): Transform {\n if (!this.transformCameraUpdate) return this.transform;\n\n if (!this._requestedCameraState) {\n this._requestedCameraState = this.transform.clone();\n }\n return this._requestedCameraState;\n }\n\n /**\n * @internal\n * Called after the camera is done being manipulated.\n * @param tr - the requested camera end state\n * Call `transformCameraUpdate` if present, and then apply the \"approved\" changes.\n */\n _applyUpdatedTransform(tr: Transform) {\n if (!this.transformCameraUpdate) return;\n\n const nextTransform = tr.clone();\n const {\n center,\n zoom,\n pitch,\n bearing,\n elevation\n } = this.transformCameraUpdate(nextTransform);\n if (center) nextTransform.center = center;\n if (zoom !== undefined) nextTransform.zoom = zoom;\n if (pitch !== undefined) nextTransform.pitch = pitch;\n if (bearing !== undefined) nextTransform.bearing = bearing;\n if (elevation !== undefined) nextTransform.elevation = elevation;\n this.transform.apply(nextTransform);\n }\n\n _fireMoveEvents(eventData?: any) {\n this.fire(new Event('move', eventData));\n if (this._zooming) {\n this.fire(new Event('zoom', eventData));\n }\n if (this._rotating) {\n this.fire(new Event('rotate', eventData));\n }\n if (this._pitching) {\n this.fire(new Event('pitch', eventData));\n }\n }\n\n _afterEase(eventData?: any, easeId?: string) {\n // if this easing is being stopped to start another easing with\n // the same id then don't fire any events to avoid extra start/stop events\n if (this._easeId && easeId && this._easeId === easeId) {\n return;\n }\n delete this._easeId;\n\n const wasZooming = this._zooming;\n const wasRotating = this._rotating;\n const wasPitching = this._pitching;\n this._moving = false;\n this._zooming = false;\n this._rotating = false;\n this._pitching = false;\n this._padding = false;\n\n if (wasZooming) {\n this.fire(new Event('zoomend', eventData));\n }\n if (wasRotating) {\n this.fire(new Event('rotateend', eventData));\n }\n if (wasPitching) {\n this.fire(new Event('pitchend', eventData));\n }\n this.fire(new Event('moveend', eventData));\n }\n\n /**\n * Changes any combination of center, zoom, bearing, and pitch, animating the transition along a curve that\n * evokes flight. The animation seamlessly incorporates zooming and panning to help\n * the user maintain her bearings even after traversing a great distance.\n *\n * Note: The animation will be skipped, and this will behave equivalently to `jumpTo`\n * if the user has the `reduced motion` accessibility feature enabled in their operating system,\n * unless 'options' includes `essential: true`.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`,\n * `pitch`, `pitchend`, and `rotate`.\n *\n * @param options - Options describing the destination and animation of the transition.\n * Accepts {@link CameraOptions}, {@link AnimationOptions},\n * and the following additional options.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * // fly with default options to null island\n * map.flyTo({center: [0, 0], zoom: 9});\n * // using flyTo options\n * map.flyTo({\n * center: [0, 0],\n * zoom: 9,\n * speed: 0.2,\n * curve: 1,\n * easing(t) {\n * return t;\n * }\n * });\n * ```\n * @see [Fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto/)\n * @see [Slowly fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto-options/)\n * @see [Fly to a location based on scroll position](https://maplibre.org/maplibre-gl-js/docs/examples/scroll-fly-to/)\n */\n flyTo(options: FlyToOptions, eventData?: any): this {\n // Fall through to jumpTo if user has set prefers-reduced-motion\n if (!options.essential && browser.prefersReducedMotion) {\n const coercedOptions = pick(options, ['center', 'zoom', 'bearing', 'pitch', 'around']) as CameraOptions;\n return this.jumpTo(coercedOptions, eventData);\n }\n\n // This method implements an “optimal path” animation, as detailed in:\n //\n // Van Wijk, Jarke J.; Nuij, Wim A. A. “Smooth and efficient zooming and panning.” INFOVIS\n // ’03. pp. 15–22. .\n //\n // Where applicable, local variable documentation begins with the associated variable or\n // function in van Wijk (2003).\n\n this.stop();\n\n options = extend({\n offset: [0, 0],\n speed: 1.2,\n curve: 1.42,\n easing: defaultEasing\n }, options);\n\n const tr = this._getTransformForUpdate(),\n startZoom = this.getZoom(),\n startBearing = this.getBearing(),\n startPitch = this.getPitch(),\n startPadding = this.getPadding();\n\n const zoom = 'zoom' in options ? clamp(+options.zoom, tr.minZoom, tr.maxZoom) : startZoom;\n const bearing = 'bearing' in options ? this._normalizeBearing(options.bearing, startBearing) : startBearing;\n const pitch = 'pitch' in options ? +options.pitch : startPitch;\n const padding = 'padding' in options ? options.padding : tr.padding;\n\n const scale = tr.zoomScale(zoom - startZoom);\n const offsetAsPoint = Point.convert(options.offset);\n let pointAtOffset = tr.centerPoint.add(offsetAsPoint);\n const locationAtOffset = tr.pointLocation(pointAtOffset);\n const center = LngLat.convert(options.center || locationAtOffset);\n this._normalizeCenter(center);\n\n const from = tr.project(locationAtOffset);\n const delta = tr.project(center).sub(from);\n\n let rho = options.curve;\n\n // w₀: Initial visible span, measured in pixels at the initial scale.\n const w0 = Math.max(tr.width, tr.height),\n // w₁: Final visible span, measured in pixels with respect to the initial scale.\n w1 = w0 / scale,\n // Length of the flight path as projected onto the ground plane, measured in pixels from\n // the world image origin at the initial scale.\n u1 = delta.mag();\n\n if ('minZoom' in options) {\n const minZoom = clamp(Math.min(options.minZoom, startZoom, zoom), tr.minZoom, tr.maxZoom);\n // wm: Maximum visible span, measured in pixels with respect to the initial\n // scale.\n const wMax = w0 / tr.zoomScale(minZoom - startZoom);\n rho = Math.sqrt(wMax / u1 * 2);\n }\n\n // ρ²\n const rho2 = rho * rho;\n\n /**\n * rᵢ: Returns the zoom-out factor at one end of the animation.\n *\n * @param descent - `true` for the descent, `false` for the ascent\n */\n function zoomOutFactor(descent: boolean) {\n const b = (w1 * w1 - w0 * w0 + (descent ? -1 : 1) * rho2 * rho2 * u1 * u1) / (2 * (descent ? w1 : w0) * rho2 * u1);\n return Math.log(Math.sqrt(b * b + 1) - b);\n }\n\n function sinh(n) { return (Math.exp(n) - Math.exp(-n)) / 2; }\n function cosh(n) { return (Math.exp(n) + Math.exp(-n)) / 2; }\n function tanh(n) { return sinh(n) / cosh(n); }\n\n // r₀: Zoom-out factor during ascent.\n const r0 = zoomOutFactor(false);\n\n // w(s): Returns the visible span on the ground, measured in pixels with respect to the\n // initial scale. Assumes an angular field of view of 2 arctan ½ ≈ 53°.\n let w: (_: number) => number = function (s) {\n return (cosh(r0) / cosh(r0 + rho * s));\n };\n\n // u(s): Returns the distance along the flight path as projected onto the ground plane,\n // measured in pixels from the world image origin at the initial scale.\n let u: (_: number) => number = function (s) {\n return w0 * ((cosh(r0) * tanh(r0 + rho * s) - sinh(r0)) / rho2) / u1;\n };\n\n // S: Total length of the flight path, measured in ρ-screenfuls.\n let S = (zoomOutFactor(true) - r0) / rho;\n\n // When u₀ = u₁, the optimal path doesn’t require both ascent and descent.\n if (Math.abs(u1) < 0.000001 || !isFinite(S)) {\n // Perform a more or less instantaneous transition if the path is too short.\n if (Math.abs(w0 - w1) < 0.000001) return this.easeTo(options, eventData);\n\n const k = w1 < w0 ? -1 : 1;\n S = Math.abs(Math.log(w1 / w0)) / rho;\n\n u = function() { return 0; };\n w = function(s) { return Math.exp(k * rho * s); };\n }\n\n if ('duration' in options) {\n options.duration = +options.duration;\n } else {\n const V = 'screenSpeed' in options ? +options.screenSpeed / rho : +options.speed;\n options.duration = 1000 * S / V;\n }\n\n if (options.maxDuration && options.duration > options.maxDuration) {\n options.duration = 0;\n }\n\n this._zooming = true;\n this._rotating = (startBearing !== bearing);\n this._pitching = (pitch !== startPitch);\n this._padding = !tr.isPaddingEqual(padding as PaddingOptions);\n\n this._prepareEase(eventData, false);\n if (this.terrain) this._prepareElevation(center);\n\n this._ease((k) => {\n // s: The distance traveled along the flight path, measured in ρ-screenfuls.\n const s = k * S;\n const scale = 1 / w(s);\n tr.zoom = k === 1 ? zoom : startZoom + tr.scaleZoom(scale);\n\n if (this._rotating) {\n tr.bearing = interpolates.number(startBearing, bearing, k);\n }\n if (this._pitching) {\n tr.pitch = interpolates.number(startPitch, pitch, k);\n }\n if (this._padding) {\n tr.interpolatePadding(startPadding, padding as PaddingOptions, k);\n // When padding is being applied, Transform#centerPoint is changing continuously,\n // thus we need to recalculate offsetPoint every frame\n pointAtOffset = tr.centerPoint.add(offsetAsPoint);\n }\n\n if (this.terrain && !options.freezeElevation) this._updateElevation(k);\n\n const newCenter = k === 1 ? center : tr.unproject(from.add(delta.mult(u(s))).mult(scale));\n tr.setLocationAtPoint(tr.renderWorldCopies ? newCenter.wrap() : newCenter, pointAtOffset);\n\n this._applyUpdatedTransform(tr);\n\n this._fireMoveEvents(eventData);\n\n }, () => {\n if (this.terrain) this._finalizeElevation();\n this._afterEase(eventData);\n }, options);\n\n return this;\n }\n\n isEasing() {\n return !!this._easeFrameId;\n }\n\n /**\n * Stops any animated transition underway.\n *\n * @returns `this`\n */\n stop(): this {\n return this._stop();\n }\n\n _stop(allowGestures?: boolean, easeId?: string): this {\n if (this._easeFrameId) {\n this._cancelRenderFrame(this._easeFrameId);\n delete this._easeFrameId;\n delete this._onEaseFrame;\n }\n\n if (this._onEaseEnd) {\n // The _onEaseEnd function might emit events which trigger new\n // animation, which sets a new _onEaseEnd. Ensure we don't delete\n // it unintentionally.\n const onEaseEnd = this._onEaseEnd;\n delete this._onEaseEnd;\n onEaseEnd.call(this, easeId);\n }\n if (!allowGestures) {\n const handlers = (this as any).handlers;\n if (handlers) handlers.stop(false);\n }\n return this;\n }\n\n _ease(frame: (_: number) => void,\n finish: () => void,\n options: {\n animate?: boolean;\n duration?: number;\n easing?: (_: number) => number;\n }) {\n if (options.animate === false || options.duration === 0) {\n frame(1);\n finish();\n } else {\n this._easeStart = browser.now();\n this._easeOptions = options;\n this._onEaseFrame = frame;\n this._onEaseEnd = finish;\n this._easeFrameId = this._requestRenderFrame(this._renderFrameCallback);\n }\n }\n\n // Callback for map._requestRenderFrame\n _renderFrameCallback = () => {\n const t = Math.min((browser.now() - this._easeStart) / this._easeOptions.duration, 1);\n this._onEaseFrame(this._easeOptions.easing(t));\n\n // if _stop is called during _onEaseFrame from _fireMoveEvents we should avoid a new _requestRenderFrame, checking it by ensuring _easeFrameId was not deleted\n if (t < 1 && this._easeFrameId) {\n this._easeFrameId = this._requestRenderFrame(this._renderFrameCallback);\n } else {\n this.stop();\n }\n };\n\n // convert bearing so that it's numerically close to the current one so that it interpolates properly\n _normalizeBearing(bearing: number, currentBearing: number) {\n bearing = wrap(bearing, -180, 180);\n const diff = Math.abs(bearing - currentBearing);\n if (Math.abs(bearing - 360 - currentBearing) < diff) bearing -= 360;\n if (Math.abs(bearing + 360 - currentBearing) < diff) bearing += 360;\n return bearing;\n }\n\n // If a path crossing the antimeridian would be shorter, extend the final coordinate so that\n // interpolating between the two endpoints will cross it.\n _normalizeCenter(center: LngLat) {\n const tr = this.transform;\n if (!tr.renderWorldCopies || tr.lngRange) return;\n\n const delta = center.lng - tr.center.lng;\n center.lng +=\n delta > 180 ? -360 :\n delta < -180 ? 360 : 0;\n }\n\n /**\n * Query the current elevation of location. It return null if terrain is not enabled. the elevation is in meters relative to mean sea-level\n * @param lngLatLike - [x,y] or LngLat coordinates of the location\n * @returns elevation in meters\n */\n queryTerrainElevation(lngLatLike: LngLatLike): number | null {\n if (!this.terrain) {\n return null;\n }\n const elevation = this.terrain.getElevationForLngLatZoom(LngLat.convert(lngLatLike), this.transform.tileZoom);\n /**\n * Different zoomlevels with different terrain-tiles the elevation-values are not the same.\n * map.transform.elevation variable with the center-altitude.\n * In maplibre the proj-matrix is translated by this value in negative z-direction.\n * So we need to add this value to the elevation to get the correct value.\n */\n return elevation - this.transform.elevation;\n }\n}\n","import {DOM} from '../../util/dom';\n\nimport type {Map} from '../map';\nimport type {ControlPosition, IControl} from './control';\nimport type {MapDataEvent} from '../events';\nimport type {StyleSpecification} from '@maplibre/maplibre-gl-style-spec';\n/**\n * The {@link AttributionControl} options\n */\ntype AttributionOptions = {\n /**\n * If `true`, the attribution control will always collapse when moving the map. If `false`,\n * force the expanded attribution control. The default is a responsive attribution that collapses when the user moves the map on maps less than 640 pixels wide.\n * **Attribution should not be collapsed if it can comfortably fit on the map. `compact` should only be used to modify default attribution when map size makes it impossible to fit default attribution and when the automatic compact resizing for default settings are not sufficient.**\n */\n compact?: boolean;\n /**\n * Attributions to show in addition to any other attributions.\n */\n customAttribution?: string | Array;\n};\n\n/**\n * An `AttributionControl` control presents the map's attribution information. By default, the attribution control is expanded (regardless of map width).\n * @group Markers and Controls\n * @example\n * ```ts\n * let map = new maplibregl.Map({attributionControl: false})\n * .addControl(new maplibregl.AttributionControl({\n * compact: true\n * }));\n * ```\n */\nexport class AttributionControl implements IControl {\n options: AttributionOptions;\n _map: Map;\n _compact: boolean;\n _container: HTMLElement;\n _innerContainer: HTMLElement;\n _compactButton: HTMLElement;\n _editLink: HTMLAnchorElement;\n _attribHTML: string;\n styleId: string;\n styleOwner: string;\n\n /**\n * @param options - the attribution options\n */\n constructor(options: AttributionOptions = {}) {\n this.options = options;\n }\n\n getDefaultPosition(): ControlPosition {\n return 'bottom-right';\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._compact = this.options && this.options.compact;\n this._container = DOM.create('details', 'maplibregl-ctrl maplibregl-ctrl-attrib');\n this._compactButton = DOM.create('summary', 'maplibregl-ctrl-attrib-button', this._container);\n this._compactButton.addEventListener('click', this._toggleAttribution);\n this._setElementTitle(this._compactButton, 'ToggleAttribution');\n this._innerContainer = DOM.create('div', 'maplibregl-ctrl-attrib-inner', this._container);\n\n this._updateAttributions();\n this._updateCompact();\n\n this._map.on('styledata', this._updateData);\n this._map.on('sourcedata', this._updateData);\n this._map.on('terrain', this._updateData);\n this._map.on('resize', this._updateCompact);\n this._map.on('drag', this._updateCompactMinimize);\n\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._container);\n\n this._map.off('styledata', this._updateData);\n this._map.off('sourcedata', this._updateData);\n this._map.off('terrain', this._updateData);\n this._map.off('resize', this._updateCompact);\n this._map.off('drag', this._updateCompactMinimize);\n\n this._map = undefined;\n this._compact = undefined;\n this._attribHTML = undefined;\n }\n\n _setElementTitle(element: HTMLElement, title: string) {\n const str = this._map._getUIString(`AttributionControl.${title}`);\n element.title = str;\n element.setAttribute('aria-label', str);\n }\n\n _toggleAttribution = () => {\n if (this._container.classList.contains('maplibregl-compact')) {\n if (this._container.classList.contains('maplibregl-compact-show')) {\n this._container.setAttribute('open', '');\n this._container.classList.remove('maplibregl-compact-show');\n } else {\n this._container.classList.add('maplibregl-compact-show');\n this._container.removeAttribute('open');\n }\n }\n };\n\n _updateData = (e: MapDataEvent) => {\n if (e && (e.sourceDataType === 'metadata' || e.sourceDataType === 'visibility' || e.dataType === 'style' || e.type === 'terrain')) {\n this._updateAttributions();\n }\n };\n\n _updateAttributions() {\n if (!this._map.style) return;\n let attributions: Array = [];\n if (this.options.customAttribution) {\n if (Array.isArray(this.options.customAttribution)) {\n attributions = attributions.concat(\n this.options.customAttribution.map(attribution => {\n if (typeof attribution !== 'string') return '';\n return attribution;\n })\n );\n } else if (typeof this.options.customAttribution === 'string') {\n attributions.push(this.options.customAttribution);\n }\n }\n\n if (this._map.style.stylesheet) {\n const stylesheet = this._map.style.stylesheet as StyleSpecification & { owner: string; id: string };\n this.styleOwner = stylesheet.owner;\n this.styleId = stylesheet.id;\n }\n\n const sourceCaches = this._map.style.sourceCaches;\n for (const id in sourceCaches) {\n const sourceCache = sourceCaches[id];\n if (sourceCache.used || sourceCache.usedForTerrain) {\n const source = sourceCache.getSource();\n if (source.attribution && attributions.indexOf(source.attribution) < 0) {\n attributions.push(source.attribution);\n }\n }\n }\n\n // remove any entries that are whitespace\n attributions = attributions.filter(e => String(e).trim());\n\n // remove any entries that are substrings of another entry.\n // first sort by length so that substrings come first\n attributions.sort((a, b) => a.length - b.length);\n attributions = attributions.filter((attrib, i) => {\n for (let j = i + 1; j < attributions.length; j++) {\n if (attributions[j].indexOf(attrib) >= 0) { return false; }\n }\n return true;\n });\n\n // check if attribution string is different to minimize DOM changes\n const attribHTML = attributions.join(' | ');\n if (attribHTML === this._attribHTML) return;\n\n this._attribHTML = attribHTML;\n\n if (attributions.length) {\n this._innerContainer.innerHTML = attribHTML;\n this._container.classList.remove('maplibregl-attrib-empty');\n } else {\n this._container.classList.add('maplibregl-attrib-empty');\n }\n this._updateCompact();\n // remove old DOM node from _editLink\n this._editLink = null;\n }\n\n _updateCompact = () => {\n if (this._map.getCanvasContainer().offsetWidth <= 640 || this._compact) {\n if (this._compact === false) {\n this._container.setAttribute('open', '');\n } else if (!this._container.classList.contains('maplibregl-compact') && !this._container.classList.contains('maplibregl-attrib-empty')) {\n this._container.setAttribute('open', '');\n this._container.classList.add('maplibregl-compact', 'maplibregl-compact-show');\n }\n } else {\n this._container.setAttribute('open', '');\n if (this._container.classList.contains('maplibregl-compact')) {\n this._container.classList.remove('maplibregl-compact', 'maplibregl-compact-show');\n }\n }\n };\n\n _updateCompactMinimize = () => {\n if (this._container.classList.contains('maplibregl-compact')) {\n if (this._container.classList.contains('maplibregl-compact-show')) {\n this._container.classList.remove('maplibregl-compact-show');\n }\n }\n };\n}\n","import {DOM} from '../../util/dom';\n\nimport type {Map} from '../map';\nimport type {ControlPosition, IControl} from './control';\n\n/**\n * The {@link LogoControl} options object\n */\ntype LogoOptions = {\n /**\n * If `true`, force a compact logo.\n * If `false`, force the full logo. The default is a responsive logo that collapses when the map is less than 640 pixels wide.\n */\n compact?: boolean;\n};\n\n/**\n * A `LogoControl` is a control that adds the watermark.\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * map.addControl(new maplibregl.LogoControl({compact: false}));\n * ```\n **/\nexport class LogoControl implements IControl {\n options: LogoOptions;\n _map: Map;\n _compact: boolean;\n _container: HTMLElement;\n\n constructor(options: LogoOptions = {}) {\n this.options = options;\n }\n\n getDefaultPosition(): ControlPosition {\n return 'bottom-left';\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._compact = this.options && this.options.compact;\n this._container = DOM.create('div', 'maplibregl-ctrl');\n const anchor = DOM.create('a', 'maplibregl-ctrl-logo');\n anchor.target = '_blank';\n anchor.rel = 'noopener nofollow';\n anchor.href = 'https://maplibre.org/';\n anchor.setAttribute('aria-label', this._map._getUIString('LogoControl.Title'));\n anchor.setAttribute('rel', 'noopener nofollow');\n this._container.appendChild(anchor);\n this._container.style.display = 'block';\n\n this._map.on('resize', this._updateCompact);\n this._updateCompact();\n\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._container);\n this._map.off('resize', this._updateCompact);\n this._map = undefined;\n this._compact = undefined;\n }\n\n _updateCompact = () => {\n const containerChildren = this._container.children;\n if (containerChildren.length) {\n const anchor = containerChildren[0];\n if (this._map.getCanvasContainer().offsetWidth <= 640 || this._compact) {\n if (this._compact !== false) {\n anchor.classList.add('maplibregl-compact');\n }\n } else {\n anchor.classList.remove('maplibregl-compact');\n }\n }\n };\n\n}\n","export type TaskID = number; // can't mark opaque due to https://github.com/flowtype/flow-remove-types/pull/61\n\ntype Task = {\n callback: (timeStamp: number) => void;\n id: TaskID;\n cancelled: boolean;\n};\n\nexport class TaskQueue {\n _queue: Array;\n _id: TaskID;\n _cleared: boolean;\n _currentlyRunning: Array | false;\n\n constructor() {\n this._queue = [];\n this._id = 0;\n this._cleared = false;\n this._currentlyRunning = false;\n }\n\n add(callback: (timeStamp: number) => void): TaskID {\n const id = ++this._id;\n const queue = this._queue;\n queue.push({callback, id, cancelled: false});\n return id;\n }\n\n remove(id: TaskID) {\n const running = this._currentlyRunning;\n const queue = running ? this._queue.concat(running) : this._queue;\n for (const task of queue) {\n if (task.id === id) {\n task.cancelled = true;\n return;\n }\n }\n }\n\n run(timeStamp: number = 0) {\n if (this._currentlyRunning) throw new Error('Attempting to run(), but is already running.');\n const queue = this._currentlyRunning = this._queue;\n\n // Tasks queued by callbacks in the current queue should be executed\n // on the next run, not the current run.\n this._queue = [];\n\n for (const task of queue) {\n if (task.cancelled) continue;\n task.callback(timeStamp);\n if (this._cleared) break;\n }\n\n this._cleared = false;\n this._currentlyRunning = false;\n }\n\n clear() {\n if (this._currentlyRunning) {\n this._cleared = true;\n }\n this._queue = [];\n }\n}\n","import type {RequestParameters} from '../util/ajax';\n\nexport type PerformanceMetrics = {\n loadTime: number;\n fullLoadTime: number;\n fps: number;\n percentDroppedFrames: number;\n totalFrames: number;\n};\n\nexport enum PerformanceMarkers {\n create = 'create',\n load = 'load',\n fullLoad = 'fullLoad'\n}\n\nlet lastFrameTime = null;\nlet frameTimes = [];\n\nconst minFramerateTarget = 60;\nconst frameTimeTarget = 1000 / minFramerateTarget;\n\nconst loadTimeKey = 'loadTime';\nconst fullLoadTimeKey = 'fullLoadTime';\n\nexport const PerformanceUtils = {\n mark(marker: PerformanceMarkers) {\n performance.mark(marker);\n },\n frame(timestamp: number) {\n const currTimestamp = timestamp;\n if (lastFrameTime != null) {\n const frameTime = currTimestamp - lastFrameTime;\n frameTimes.push(frameTime);\n }\n lastFrameTime = currTimestamp;\n },\n clearMetrics() {\n lastFrameTime = null;\n frameTimes = [];\n performance.clearMeasures(loadTimeKey);\n performance.clearMeasures(fullLoadTimeKey);\n\n for (const marker in PerformanceMarkers) {\n performance.clearMarks(PerformanceMarkers[marker]);\n }\n },\n\n getPerformanceMetrics(): PerformanceMetrics {\n performance.measure(loadTimeKey, PerformanceMarkers.create, PerformanceMarkers.load);\n performance.measure(fullLoadTimeKey, PerformanceMarkers.create, PerformanceMarkers.fullLoad);\n const loadTime = performance.getEntriesByName(loadTimeKey)[0].duration;\n const fullLoadTime = performance.getEntriesByName(fullLoadTimeKey)[0].duration;\n const totalFrames = frameTimes.length;\n\n const avgFrameTime = frameTimes.reduce((prev, curr) => prev + curr, 0) / totalFrames / 1000;\n const fps = 1 / avgFrameTime;\n\n // count frames that missed our framerate target\n const droppedFrames = frameTimes\n .filter((frameTime) => frameTime > frameTimeTarget)\n .reduce((acc, curr) => {\n return acc + (curr - frameTimeTarget) / frameTimeTarget;\n }, 0);\n const percentDroppedFrames = (droppedFrames / (totalFrames + droppedFrames)) * 100;\n\n return {\n loadTime,\n fullLoadTime,\n fps,\n percentDroppedFrames,\n totalFrames\n };\n }\n};\n\n/**\n * @internal\n * Safe wrapper for the performance resource timing API in web workers with graceful degradation\n */\nexport class RequestPerformance {\n _marks: {\n start: string;\n end: string;\n measure: string;\n };\n\n constructor (request: RequestParameters) {\n this._marks = {\n start: [request.url, 'start'].join('#'),\n end: [request.url, 'end'].join('#'),\n measure: request.url.toString()\n };\n\n performance.mark(this._marks.start);\n }\n\n finish() {\n performance.mark(this._marks.end);\n let resourceTimingData = performance.getEntriesByName(this._marks.measure);\n\n // fallback if web worker implementation of perf.getEntriesByName returns empty\n if (resourceTimingData.length === 0) {\n performance.measure(this._marks.measure, this._marks.start, this._marks.end);\n resourceTimingData = performance.getEntriesByName(this._marks.measure);\n\n // cleanup\n performance.clearMarks(this._marks.start);\n performance.clearMarks(this._marks.end);\n performance.clearMeasures(this._marks.measure);\n }\n\n return resourceTimingData;\n }\n}\n\nexport default performance;\n","export const defaultLocale = {\n 'AttributionControl.ToggleAttribution': 'Toggle attribution',\n 'AttributionControl.MapFeedback': 'Map feedback',\n 'FullscreenControl.Enter': 'Enter fullscreen',\n 'FullscreenControl.Exit': 'Exit fullscreen',\n 'GeolocateControl.FindMyLocation': 'Find my location',\n 'GeolocateControl.LocationNotAvailable': 'Location not available',\n 'LogoControl.Title': 'Mapbox logo',\n 'NavigationControl.ResetBearing': 'Reset bearing to north',\n 'NavigationControl.ZoomIn': 'Zoom in',\n 'NavigationControl.ZoomOut': 'Zoom out',\n 'ScaleControl.Feet': 'ft',\n 'ScaleControl.Meters': 'm',\n 'ScaleControl.Kilometers': 'km',\n 'ScaleControl.Miles': 'mi',\n 'ScaleControl.NauticalMiles': 'nm',\n 'TerrainControl.enableTerrain': 'Enable terrain',\n 'TerrainControl.disableTerrain': 'Disable terrain'\n};\n","import {createLayout} from '../util/struct_array';\n\nexport default createLayout([\n {name: 'a_pos3d', type: 'Int16', components: 3}\n]);\n","import {OverscaledTileID} from './tile_id';\nimport {Tile} from './tile';\nimport {EXTENT} from '../data/extent';\nimport {mat4} from 'gl-matrix';\nimport {Evented} from '../util/evented';\nimport type {Transform} from '../geo/transform';\nimport type {SourceCache} from '../source/source_cache';\nimport {Terrain} from '../render/terrain';\n\n/**\n * @internal\n * This class is a helper for the Terrain-class, it:\n * - loads raster-dem tiles\n * - manages all renderToTexture tiles.\n * - caches previous rendered tiles.\n * - finds all necessary renderToTexture tiles for a OverscaledTileID area\n * - finds the corresponding raster-dem tile for OverscaledTileID\n */\nexport class TerrainSourceCache extends Evented {\n /**\n * source-cache for the raster-dem source.\n */\n sourceCache: SourceCache;\n /**\n * stores all render-to-texture tiles.\n */\n _tiles: {[_: string]: Tile};\n /**\n * contains a list of tileID-keys for the current scene. (only for performance)\n */\n _renderableTilesKeys: Array;\n /**\n * raster-dem-tile for a TileID cache.\n */\n _sourceTileCache: {[_: string]: string};\n /**\n * minimum zoomlevel to render the terrain.\n */\n minzoom: number;\n /**\n * maximum zoomlevel to render the terrain.\n */\n maxzoom: number;\n /**\n * render-to-texture tileSize in scene.\n */\n tileSize: number;\n /**\n * raster-dem tiles will load for performance the actualZoom - deltaZoom zoom-level.\n */\n deltaZoom: number;\n\n constructor(sourceCache: SourceCache) {\n super();\n this.sourceCache = sourceCache;\n this._tiles = {};\n this._renderableTilesKeys = [];\n this._sourceTileCache = {};\n this.minzoom = 0;\n this.maxzoom = 22;\n this.tileSize = 512;\n this.deltaZoom = 1;\n sourceCache.usedForTerrain = true;\n sourceCache.tileSize = this.tileSize * 2 ** this.deltaZoom;\n }\n\n destruct() {\n this.sourceCache.usedForTerrain = false;\n this.sourceCache.tileSize = null;\n }\n\n /**\n * Load Terrain Tiles, create internal render-to-texture tiles, free GPU memory.\n * @param transform - the operation to do\n * @param terrain - the terrain\n */\n update(transform: Transform, terrain: Terrain): void {\n // load raster-dem tiles for the current scene.\n this.sourceCache.update(transform, terrain);\n // create internal render-to-texture tiles for the current scene.\n this._renderableTilesKeys = [];\n const keys = {};\n for (const tileID of transform.coveringTiles({\n tileSize: this.tileSize,\n minzoom: this.minzoom,\n maxzoom: this.maxzoom,\n reparseOverscaled: false,\n terrain\n })) {\n keys[tileID.key] = true;\n this._renderableTilesKeys.push(tileID.key);\n if (!this._tiles[tileID.key]) {\n tileID.posMatrix = new Float64Array(16) as any;\n mat4.ortho(tileID.posMatrix, 0, EXTENT, 0, EXTENT, 0, 1);\n this._tiles[tileID.key] = new Tile(tileID, this.tileSize);\n }\n }\n // free unused tiles\n for (const key in this._tiles) {\n if (!keys[key]) delete this._tiles[key];\n }\n }\n\n /**\n * Free render to texture cache\n * @param tileID - optional, free only corresponding to tileID.\n */\n freeRtt(tileID?: OverscaledTileID) {\n for (const key in this._tiles) {\n const tile = this._tiles[key];\n if (!tileID || tile.tileID.equals(tileID) || tile.tileID.isChildOf(tileID) || tileID.isChildOf(tile.tileID))\n tile.rtt = [];\n }\n }\n\n /**\n * get a list of tiles, which are loaded and should be rendered in the current scene\n * @returns the renderable tiles\n */\n getRenderableTiles(): Array {\n return this._renderableTilesKeys.map(key => this.getTileByID(key));\n }\n\n /**\n * get terrain tile by the TileID key\n * @param id - the tile id\n * @returns the tile\n */\n getTileByID(id: string): Tile {\n return this._tiles[id];\n }\n\n /**\n * Searches for the corresponding current renderable terrain-tiles\n * @param tileID - the tile to look for\n * @returns the tiles that were found\n */\n getTerrainCoords(tileID: OverscaledTileID): Record {\n const coords = {};\n for (const key of this._renderableTilesKeys) {\n const _tileID = this._tiles[key].tileID;\n if (_tileID.canonical.equals(tileID.canonical)) {\n const coord = tileID.clone();\n coord.posMatrix = new Float64Array(16) as any;\n mat4.ortho(coord.posMatrix, 0, EXTENT, 0, EXTENT, 0, 1);\n coords[key] = coord;\n } else if (_tileID.canonical.isChildOf(tileID.canonical)) {\n const coord = tileID.clone();\n coord.posMatrix = new Float64Array(16) as any;\n const dz = _tileID.canonical.z - tileID.canonical.z;\n const dx = _tileID.canonical.x - (_tileID.canonical.x >> dz << dz);\n const dy = _tileID.canonical.y - (_tileID.canonical.y >> dz << dz);\n const size = EXTENT >> dz;\n mat4.ortho(coord.posMatrix, 0, size, 0, size, 0, 1);\n mat4.translate(coord.posMatrix, coord.posMatrix, [-dx * size, -dy * size, 0]);\n coords[key] = coord;\n } else if (tileID.canonical.isChildOf(_tileID.canonical)) {\n const coord = tileID.clone();\n coord.posMatrix = new Float64Array(16) as any;\n const dz = tileID.canonical.z - _tileID.canonical.z;\n const dx = tileID.canonical.x - (tileID.canonical.x >> dz << dz);\n const dy = tileID.canonical.y - (tileID.canonical.y >> dz << dz);\n const size = EXTENT >> dz;\n mat4.ortho(coord.posMatrix, 0, EXTENT, 0, EXTENT, 0, 1);\n mat4.translate(coord.posMatrix, coord.posMatrix, [dx * size, dy * size, 0]);\n mat4.scale(coord.posMatrix, coord.posMatrix, [1 / (2 ** dz), 1 / (2 ** dz), 0]);\n coords[key] = coord;\n }\n }\n return coords;\n }\n\n /**\n * find the covering raster-dem tile\n * @param tileID - the tile to look for\n * @param searchForDEM - Optinal parameter to search for (parent) souretiles with loaded dem.\n * @returns the tile\n */\n getSourceTile(tileID: OverscaledTileID, searchForDEM?: boolean): Tile {\n const source = this.sourceCache._source;\n let z = tileID.overscaledZ - this.deltaZoom;\n if (z > source.maxzoom) z = source.maxzoom;\n if (z < source.minzoom) return null;\n // cache for tileID to terrain-tileID\n if (!this._sourceTileCache[tileID.key])\n this._sourceTileCache[tileID.key] = tileID.scaledTo(z).key;\n let tile = this.sourceCache.getTileByID(this._sourceTileCache[tileID.key]);\n // during tile-loading phase look if parent tiles (with loaded dem) are available.\n if (!(tile && tile.dem) && searchForDEM)\n while (z >= source.minzoom && !(tile && tile.dem))\n tile = this.sourceCache.getTileByID(tileID.scaledTo(z--).key);\n return tile;\n }\n\n /**\n * get a list of tiles, loaded after a spezific time. This is used to update depth & coords framebuffers.\n * @param time - the time\n * @returns the relevant tiles\n */\n tilesAfterTime(time = Date.now()): Array {\n return Object.values(this._tiles).filter(t => t.timeAdded >= time);\n }\n}\n","\nimport {Tile} from '../source/tile';\nimport {mat4, vec2} from 'gl-matrix';\nimport {OverscaledTileID} from '../source/tile_id';\nimport {RGBAImage} from '../util/image';\nimport {warnOnce} from '../util/util';\nimport {Pos3dArray, TriangleIndexArray} from '../data/array_types.g';\nimport pos3dAttributes from '../data/pos3d_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {VertexBuffer} from '../gl/vertex_buffer';\nimport {IndexBuffer} from '../gl/index_buffer';\nimport {Painter} from './painter';\nimport {Texture} from '../render/texture';\nimport type {Framebuffer} from '../gl/framebuffer';\nimport Point from '@mapbox/point-geometry';\nimport {MercatorCoordinate, lngFromMercatorX, mercatorXfromLng} from '../geo/mercator_coordinate';\nimport {TerrainSourceCache} from '../source/terrain_source_cache';\nimport {SourceCache} from '../source/source_cache';\nimport {EXTENT} from '../data/extent';\nimport type {TerrainSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport {LngLat, earthRadius} from '../geo/lng_lat';\n\n/**\n * @internal\n * A terrain GPU related object\n */\nexport type TerrainData = {\n 'u_depth': number;\n 'u_terrain': number;\n 'u_terrain_dim': number;\n 'u_terrain_matrix': mat4;\n 'u_terrain_unpack': number[];\n 'u_terrain_exaggeration': number;\n texture: WebGLTexture;\n depthTexture: WebGLTexture;\n tile: Tile;\n}\n\n/**\n * @internal\n * A terrain mesh object\n */\nexport type TerrainMesh = {\n indexBuffer: IndexBuffer;\n vertexBuffer: VertexBuffer;\n segments: SegmentVector;\n}\n\n/**\n * @internal\n * This is the main class which handles most of the 3D Terrain logic. It has the following topics:\n * 1) loads raster-dem tiles via the internal sourceCache this.sourceCache\n * 2) creates a depth-framebuffer, which is used to calculate the visibility of coordinates\n * 3) creates a coords-framebuffer, which is used the get to tile-coordinate for a screen-pixel\n * 4) stores all render-to-texture tiles in the this.sourceCache._tiles\n * 5) calculates the elevation for a specific tile-coordinate\n * 6) creates a terrain-mesh\n *\n * A note about the GPU resource-usage:\n * Framebuffers:\n * - one for the depth & coords framebuffer with the size of the map-div.\n * - one for rendering a tile to texture with the size of tileSize (= 512x512).\n * Textures:\n * - one texture for an empty raster-dem tile with size 1x1\n * - one texture for an empty depth-buffer, when terrain is disabled with size 1x1\n * - one texture for an each loaded raster-dem with size of the source.tileSize\n * - one texture for the coords-framebuffer with the size of the map-div.\n * - one texture for the depth-framebuffer with the size of the map-div.\n * - one texture for the encoded tile-coords with the size 2*tileSize (=1024x1024)\n * - finally for each render-to-texture tile (= this._tiles) a set of textures\n * for each render stack (The stack-concept is documented in painter.ts).\n * Normally there exists 1-3 Textures per tile, depending on the stylesheet.\n * Each Textures has the size 2*tileSize (= 1024x1024). Also there exists a\n * cache of the last 150 newest rendered tiles.\n *\n */\nexport class Terrain {\n /**\n * The style this terrain crresponds to\n */\n painter: Painter;\n /**\n * the sourcecache this terrain is based on\n */\n sourceCache: TerrainSourceCache;\n /**\n * the TerrainSpecification object passed to this instance\n */\n options: TerrainSpecification;\n /**\n * define the meshSize per tile.\n */\n meshSize: number;\n /**\n * multiplicator for the elevation. Used to make terrain more \"extreme\".\n */\n exaggeration: number;\n /**\n * to not see pixels in the render-to-texture tiles it is good to render them bigger\n * this number is the multiplicator (must be a power of 2) for the current tileSize.\n * So to get good results with not too much memory footprint a value of 2 should be fine.\n */\n qualityFactor: number;\n /**\n * holds the framebuffer object in size of the screen to render the coords & depth into a texture.\n */\n _fbo: Framebuffer;\n _fboCoordsTexture: Texture;\n _fboDepthTexture: Texture;\n _emptyDepthTexture: Texture;\n /**\n * GL Objects for the terrain-mesh\n * The mesh is a regular mesh, which has the advantage that it can be reused for all tiles.\n */\n _mesh: TerrainMesh;\n /**\n * coords index contains a list of tileID.keys. This index is used to identify\n * the tile via the alpha-cannel in the coords-texture.\n * As the alpha-channel has 1 Byte a max of 255 tiles can rendered without an error.\n */\n coordsIndex: Array;\n /**\n * tile-coords encoded in the rgb channel, _coordsIndex is in the alpha-channel.\n */\n _coordsTexture: Texture;\n /**\n * accuracy of the coords. 2 * tileSize should be enoughth.\n */\n _coordsTextureSize: number;\n /**\n * variables for an empty dem texture, which is used while the raster-dem tile is loading.\n */\n _emptyDemUnpack: number[];\n _emptyDemTexture: Texture;\n _emptyDemMatrix: mat4;\n /**\n * as of overzooming of raster-dem tiles in high zoomlevels, this cache contains\n * matrices to transform from vector-tile coords to raster-dem-tile coords.\n */\n _demMatrixCache: {[_: string]: { matrix: mat4; coord: OverscaledTileID }};\n\n constructor(painter: Painter, sourceCache: SourceCache, options: TerrainSpecification) {\n this.painter = painter;\n this.sourceCache = new TerrainSourceCache(sourceCache);\n this.options = options;\n this.exaggeration = typeof options.exaggeration === 'number' ? options.exaggeration : 1.0;\n this.qualityFactor = 2;\n this.meshSize = 128;\n this._demMatrixCache = {};\n this.coordsIndex = [];\n this._coordsTextureSize = 1024;\n }\n\n /**\n * get the elevation-value from original dem-data for a given tile-coordinate\n * @param tileID - the tile to get elevation for\n * @param x - between 0 .. EXTENT\n * @param y - between 0 .. EXTENT\n * @param extent - optional, default 8192\n * @returns the elevation\n */\n getDEMElevation(tileID: OverscaledTileID, x: number, y: number, extent: number = EXTENT): number {\n if (!(x >= 0 && x < extent && y >= 0 && y < extent)) return 0;\n const terrain = this.getTerrainData(tileID);\n const dem = terrain.tile?.dem;\n if (!dem)\n return 0;\n\n const pos = vec2.transformMat4([] as any, [x / extent * EXTENT, y / extent * EXTENT], terrain.u_terrain_matrix);\n const coord = [pos[0] * dem.dim, pos[1] * dem.dim];\n\n // bilinear interpolation\n const cx = Math.floor(coord[0]),\n cy = Math.floor(coord[1]),\n tx = coord[0] - cx,\n ty = coord[1] - cy;\n return (\n dem.get(cx, cy) * (1 - tx) * (1 - ty) +\n dem.get(cx + 1, cy) * (tx) * (1 - ty) +\n dem.get(cx, cy + 1) * (1 - tx) * (ty) +\n dem.get(cx + 1, cy + 1) * (tx) * (ty)\n );\n }\n\n /**\n * Get the elevation for given {@link LngLat} in respect of exaggeration.\n * @param lnglat - the location\n * @param zoom - the zoom\n * @returns the elevation\n */\n getElevationForLngLatZoom(lnglat: LngLat, zoom: number) {\n const {tileID, mercatorX, mercatorY} = this._getOverscaledTileIDFromLngLatZoom(lnglat, zoom);\n return this.getElevation(tileID, mercatorX % EXTENT, mercatorY % EXTENT, EXTENT);\n }\n\n /**\n * Get the elevation for given coordinate in respect of exaggeration.\n * @param tileID - the tile id\n * @param x - between 0 .. EXTENT\n * @param y - between 0 .. EXTENT\n * @param extent - optional, default 8192\n * @returns the elevation\n */\n getElevation(tileID: OverscaledTileID, x: number, y: number, extent: number = EXTENT): number {\n return this.getDEMElevation(tileID, x, y, extent) * this.exaggeration;\n }\n\n /**\n * returns a Terrain Object for a tile. Unless the tile corresponds to data (e.g. tile is loading), return a flat dem object\n * @param tileID - the tile to get the terrain for\n * @returns the terrain data to use in the program\n */\n getTerrainData(tileID: OverscaledTileID): TerrainData {\n // create empty DEM Objects, which will used while raster-dem tiles are loading.\n // creates an empty depth-buffer texture which is needed, during the initialization process of the 3d mesh..\n if (!this._emptyDemTexture) {\n const context = this.painter.context;\n const image = new RGBAImage({width: 1, height: 1}, new Uint8Array(1 * 4));\n this._emptyDepthTexture = new Texture(context, image, context.gl.RGBA, {premultiply: false});\n this._emptyDemUnpack = [0, 0, 0, 0];\n this._emptyDemTexture = new Texture(context, new RGBAImage({width: 1, height: 1}), context.gl.RGBA, {premultiply: false});\n this._emptyDemTexture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE);\n this._emptyDemMatrix = mat4.identity([] as any);\n }\n // find covering dem tile and prepare demTexture\n const sourceTile = this.sourceCache.getSourceTile(tileID, true);\n if (sourceTile && sourceTile.dem && (!sourceTile.demTexture || sourceTile.needsTerrainPrepare)) {\n const context = this.painter.context;\n sourceTile.demTexture = this.painter.getTileTexture(sourceTile.dem.stride);\n if (sourceTile.demTexture) sourceTile.demTexture.update(sourceTile.dem.getPixels(), {premultiply: false});\n else sourceTile.demTexture = new Texture(context, sourceTile.dem.getPixels(), context.gl.RGBA, {premultiply: false});\n sourceTile.demTexture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE);\n sourceTile.needsTerrainPrepare = false;\n }\n // create matrix for lookup in dem data\n const matrixKey = sourceTile && (sourceTile + sourceTile.tileID.key) + tileID.key;\n if (matrixKey && !this._demMatrixCache[matrixKey]) {\n const maxzoom = this.sourceCache.sourceCache._source.maxzoom;\n let dz = tileID.canonical.z - sourceTile.tileID.canonical.z;\n if (tileID.overscaledZ > tileID.canonical.z) {\n if (tileID.canonical.z >= maxzoom) dz = tileID.canonical.z - maxzoom;\n else warnOnce('cannot calculate elevation if elevation maxzoom > source.maxzoom');\n }\n const dx = tileID.canonical.x - (tileID.canonical.x >> dz << dz);\n const dy = tileID.canonical.y - (tileID.canonical.y >> dz << dz);\n const demMatrix = mat4.fromScaling(new Float64Array(16) as any, [1 / (EXTENT << dz), 1 / (EXTENT << dz), 0]);\n mat4.translate(demMatrix, demMatrix, [dx * EXTENT, dy * EXTENT, 0]);\n this._demMatrixCache[tileID.key] = {matrix: demMatrix, coord: tileID};\n }\n // return uniform values & textures\n return {\n 'u_depth': 2,\n 'u_terrain': 3,\n 'u_terrain_dim': sourceTile && sourceTile.dem && sourceTile.dem.dim || 1,\n 'u_terrain_matrix': matrixKey ? this._demMatrixCache[tileID.key].matrix : this._emptyDemMatrix,\n 'u_terrain_unpack': sourceTile && sourceTile.dem && sourceTile.dem.getUnpackVector() || this._emptyDemUnpack,\n 'u_terrain_exaggeration': this.exaggeration,\n texture: (sourceTile && sourceTile.demTexture || this._emptyDemTexture).texture,\n depthTexture: (this._fboDepthTexture || this._emptyDepthTexture).texture,\n tile: sourceTile\n };\n }\n\n /**\n * get a framebuffer as big as the map-div, which will be used to render depth & coords into a texture\n * @param texture - the texture\n * @returns the frame buffer\n */\n getFramebuffer(texture: string): Framebuffer {\n const painter = this.painter;\n const width = painter.width / devicePixelRatio;\n const height = painter.height / devicePixelRatio;\n if (this._fbo && (this._fbo.width !== width || this._fbo.height !== height)) {\n this._fbo.destroy();\n this._fboCoordsTexture.destroy();\n this._fboDepthTexture.destroy();\n delete this._fbo;\n delete this._fboDepthTexture;\n delete this._fboCoordsTexture;\n }\n if (!this._fboCoordsTexture) {\n this._fboCoordsTexture = new Texture(painter.context, {width, height, data: null}, painter.context.gl.RGBA, {premultiply: false});\n this._fboCoordsTexture.bind(painter.context.gl.NEAREST, painter.context.gl.CLAMP_TO_EDGE);\n }\n if (!this._fboDepthTexture) {\n this._fboDepthTexture = new Texture(painter.context, {width, height, data: null}, painter.context.gl.RGBA, {premultiply: false});\n this._fboDepthTexture.bind(painter.context.gl.NEAREST, painter.context.gl.CLAMP_TO_EDGE);\n }\n if (!this._fbo) {\n this._fbo = painter.context.createFramebuffer(width, height, true, false);\n this._fbo.depthAttachment.set(painter.context.createRenderbuffer(painter.context.gl.DEPTH_COMPONENT16, width, height));\n }\n this._fbo.colorAttachment.set(texture === 'coords' ? this._fboCoordsTexture.texture : this._fboDepthTexture.texture);\n return this._fbo;\n }\n\n /**\n * create coords texture, needed to grab coordinates from canvas\n * encode coords coordinate into 4 bytes:\n * - 8 lower bits for x\n * - 8 lower bits for y\n * - 4 higher bits for x\n * - 4 higher bits for y\n * - 8 bits for coordsIndex (1 .. 255) (= number of terraintile), is later setted in draw_terrain uniform value\n * @returns the texture\n */\n getCoordsTexture(): Texture {\n const context = this.painter.context;\n if (this._coordsTexture) return this._coordsTexture;\n const data = new Uint8Array(this._coordsTextureSize * this._coordsTextureSize * 4);\n for (let y = 0, i = 0; y < this._coordsTextureSize; y++) for (let x = 0; x < this._coordsTextureSize; x++, i += 4) {\n data[i + 0] = x & 255;\n data[i + 1] = y & 255;\n data[i + 2] = ((x >> 8) << 4) | (y >> 8);\n data[i + 3] = 0;\n }\n const image = new RGBAImage({width: this._coordsTextureSize, height: this._coordsTextureSize}, new Uint8Array(data.buffer));\n const texture = new Texture(context, image, context.gl.RGBA, {premultiply: false});\n texture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE);\n this._coordsTexture = texture;\n return texture;\n }\n\n /**\n * Reads a pixel from the coords-framebuffer and translate this to mercator.\n * @param p - Screen-Coordinate\n * @returns mercator coordinate for a screen pixel\n */\n pointCoordinate(p: Point): MercatorCoordinate {\n const rgba = new Uint8Array(4);\n const context = this.painter.context, gl = context.gl;\n // grab coordinate pixel from coordinates framebuffer\n context.bindFramebuffer.set(this.getFramebuffer('coords').framebuffer);\n gl.readPixels(p.x, this.painter.height / devicePixelRatio - p.y - 1, 1, 1, gl.RGBA, gl.UNSIGNED_BYTE, rgba);\n context.bindFramebuffer.set(null);\n // decode coordinates (encoding see getCoordsTexture)\n const x = rgba[0] + ((rgba[2] >> 4) << 8);\n const y = rgba[1] + ((rgba[2] & 15) << 8);\n const tileID = this.coordsIndex[255 - rgba[3]];\n const tile = tileID && this.sourceCache.getTileByID(tileID);\n if (!tile) return null;\n const coordsSize = this._coordsTextureSize;\n const worldSize = (1 << tile.tileID.canonical.z) * coordsSize;\n const mercatorX = (tile.tileID.canonical.x * coordsSize + x) / worldSize;\n return new MercatorCoordinate(\n this._allowMercatorOverflow(p, mercatorX),\n (tile.tileID.canonical.y * coordsSize + y) / worldSize,\n this.getElevation(tile.tileID, x, y, coordsSize)\n );\n }\n\n /**\n * create a regular mesh which will be used by all terrain-tiles\n * @returns the created regular mesh\n */\n getTerrainMesh(): TerrainMesh {\n if (this._mesh) return this._mesh;\n const context = this.painter.context;\n const vertexArray = new Pos3dArray();\n const indexArray = new TriangleIndexArray();\n const meshSize = this.meshSize;\n const delta = EXTENT / meshSize;\n const meshSize2 = meshSize * meshSize;\n for (let y = 0; y <= meshSize; y++) for (let x = 0; x <= meshSize; x++)\n vertexArray.emplaceBack(x * delta, y * delta, 0);\n for (let y = 0; y < meshSize2; y += meshSize + 1) for (let x = 0; x < meshSize; x++) {\n indexArray.emplaceBack(x + y, meshSize + x + y + 1, meshSize + x + y + 2);\n indexArray.emplaceBack(x + y, meshSize + x + y + 2, x + y + 1);\n }\n // add an extra frame around the mesh to avoid stiching on tile boundaries with different zoomlevels\n // first code-block is for top-bottom frame and second for left-right frame\n const offsetTop = vertexArray.length, offsetBottom = offsetTop + (meshSize + 1) * 2;\n for (const y of [0, 1]) for (let x = 0; x <= meshSize; x++) for (const z of [0, 1])\n vertexArray.emplaceBack(x * delta, y * EXTENT, z);\n for (let x = 0; x < meshSize * 2; x += 2) {\n indexArray.emplaceBack(offsetBottom + x, offsetBottom + x + 1, offsetBottom + x + 3);\n indexArray.emplaceBack(offsetBottom + x, offsetBottom + x + 3, offsetBottom + x + 2);\n indexArray.emplaceBack(offsetTop + x, offsetTop + x + 3, offsetTop + x + 1);\n indexArray.emplaceBack(offsetTop + x, offsetTop + x + 2, offsetTop + x + 3);\n }\n const offsetLeft = vertexArray.length, offsetRight = offsetLeft + (meshSize + 1) * 2;\n for (const x of [0, 1]) for (let y = 0; y <= meshSize; y++) for (const z of [0, 1])\n vertexArray.emplaceBack(x * EXTENT, y * delta, z);\n for (let y = 0; y < meshSize * 2; y += 2) {\n indexArray.emplaceBack(offsetLeft + y, offsetLeft + y + 1, offsetLeft + y + 3);\n indexArray.emplaceBack(offsetLeft + y, offsetLeft + y + 3, offsetLeft + y + 2);\n indexArray.emplaceBack(offsetRight + y, offsetRight + y + 3, offsetRight + y + 1);\n indexArray.emplaceBack(offsetRight + y, offsetRight + y + 2, offsetRight + y + 3);\n }\n this._mesh = {\n indexBuffer: context.createIndexBuffer(indexArray),\n vertexBuffer: context.createVertexBuffer(vertexArray, pos3dAttributes.members),\n segments: SegmentVector.simpleSegment(0, 0, vertexArray.length, indexArray.length)\n };\n return this._mesh;\n }\n\n /**\n * Calculates a height of the frame around the terrain-mesh to avoid stiching between\n * tile boundaries in different zoomlevels.\n * @param zoom - current zoomlevel\n * @returns the elevation delta in meters\n */\n getMeshFrameDelta(zoom: number): number {\n // divide by 5 is evaluated by trial & error to get a frame in the right height\n return 2 * Math.PI * earthRadius / Math.pow(2, zoom) / 5;\n }\n\n getMinTileElevationForLngLatZoom(lnglat: LngLat, zoom: number) {\n const {tileID} = this._getOverscaledTileIDFromLngLatZoom(lnglat, zoom);\n return this.getMinMaxElevation(tileID).minElevation ?? 0;\n }\n\n /**\n * Get the minimum and maximum elevation contained in a tile. This includes any\n * exaggeration included in the terrain.\n *\n * @param tileID - ID of the tile to be used as a source for the min/max elevation\n * @returns the minimum and maximum elevation found in the tile, including the terrain's\n * exaggeration\n */\n getMinMaxElevation(tileID: OverscaledTileID): {minElevation: number | null; maxElevation: number | null} {\n const tile = this.getTerrainData(tileID).tile;\n const minMax = {minElevation: null, maxElevation: null};\n if (tile && tile.dem) {\n minMax.minElevation = tile.dem.min * this.exaggeration;\n minMax.maxElevation = tile.dem.max * this.exaggeration;\n }\n return minMax;\n }\n\n _getOverscaledTileIDFromLngLatZoom(lnglat: LngLat, zoom: number): { tileID: OverscaledTileID; mercatorX: number; mercatorY: number} {\n const mercatorCoordinate = MercatorCoordinate.fromLngLat(lnglat.wrap());\n const worldSize = (1 << zoom) * EXTENT;\n const mercatorX = mercatorCoordinate.x * worldSize;\n const mercatorY = mercatorCoordinate.y * worldSize;\n const tileX = Math.floor(mercatorX / EXTENT), tileY = Math.floor(mercatorY / EXTENT);\n const tileID = new OverscaledTileID(zoom, 0, zoom, tileX, tileY);\n return {\n tileID,\n mercatorX,\n mercatorY\n };\n }\n\n _allowMercatorOverflow(p: Point, mercatorX: number): number {\n const inLeftHalf = p.x < (this.painter.width / 2);\n let lng = lngFromMercatorX(mercatorX);\n const centerLng = this.painter.transform.center.lng;\n if (\n (inLeftHalf && Math.sign(lng) > 0 && Math.sign(centerLng) < 0) ||\n (!inLeftHalf && Math.sign(lng) < 0 && Math.sign(centerLng) > 0)\n ) {\n lng = 360 * Math.sign(centerLng) + lng;\n return mercatorXfromLng(lng);\n }\n return mercatorX;\n }\n}\n","import {Texture} from '../render/texture';\nimport {Context} from './context';\nimport {Framebuffer} from './framebuffer';\n\nexport type PoolObject = {\n id: number;\n fbo: Framebuffer;\n texture: Texture;\n stamp: number;\n inUse: boolean;\n};\n/**\n * @internal\n * `RenderPool` is a resource pool for textures and framebuffers\n */\nexport class RenderPool {\n private _objects: Array;\n /**\n * An index array of recently used pool objects.\n * Items that are used recently are last in the array\n */\n private _recentlyUsed: Array;\n private _stamp: number;\n\n constructor(\n private readonly _context: Context,\n private readonly _size: number,\n private readonly _tileSize: number) {\n this._objects = [];\n this._recentlyUsed = [];\n this._stamp = 0;\n }\n\n public destruct() {\n for (const obj of this._objects) {\n obj.texture.destroy();\n obj.fbo.destroy();\n }\n }\n\n private _createObject(id: number): PoolObject {\n const fbo = this._context.createFramebuffer(this._tileSize, this._tileSize, true, true);\n const texture = new Texture(this._context, {width: this._tileSize, height: this._tileSize, data: null}, this._context.gl.RGBA);\n texture.bind(this._context.gl.LINEAR, this._context.gl.CLAMP_TO_EDGE);\n fbo.depthAttachment.set(this._context.createRenderbuffer(this._context.gl.DEPTH_STENCIL, this._tileSize, this._tileSize));\n fbo.colorAttachment.set(texture.texture);\n return {id, fbo, texture, stamp: -1, inUse: false};\n }\n\n public getObjectForId(id: number): PoolObject {\n return this._objects[id];\n }\n\n public useObject(obj: PoolObject) {\n obj.inUse = true;\n this._recentlyUsed = this._recentlyUsed.filter(id => obj.id !== id);\n this._recentlyUsed.push(obj.id);\n }\n\n public stampObject(obj: PoolObject) {\n obj.stamp = ++this._stamp;\n }\n\n public getOrCreateFreeObject(): PoolObject {\n // check for free existing object\n for (const id of this._recentlyUsed) {\n if (!this._objects[id].inUse)\n return this._objects[id];\n }\n if (this._objects.length >= this._size)\n throw new Error('No free RenderPool available, call freeAllObjects() required!');\n // create new object\n const obj = this._createObject(this._objects.length);\n this._objects.push(obj);\n return obj;\n }\n\n public freeObject(obj: PoolObject) {\n obj.inUse = false;\n }\n\n public freeAllObjects() {\n for (const obj of this._objects)\n this.freeObject(obj);\n }\n\n public isFull(): boolean {\n if (this._objects.length < this._size) {\n return false;\n }\n return this._objects.some(o => !o.inUse) === false;\n }\n}\n","import {Painter} from './painter';\nimport {Tile} from '../source/tile';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {OverscaledTileID} from '../source/tile_id';\nimport {drawTerrain} from './draw_terrain';\nimport {Style} from '../style/style';\nimport {Terrain} from './terrain';\nimport {RenderPool} from '../gl/render_pool';\nimport {Texture} from './texture';\nimport type {StyleLayer} from '../style/style_layer';\n\n/**\n * lookup table which layers should rendered to texture\n */\nconst LAYERS: { [keyof in StyleLayer['type']]?: boolean } = {\n background: true,\n fill: true,\n line: true,\n raster: true,\n hillshade: true\n};\n\n/**\n * @internal\n * A helper class to help define what should be rendered to texture and how\n */\nexport class RenderToTexture {\n painter: Painter;\n terrain: Terrain;\n pool: RenderPool;\n /**\n * coordsDescendingInv contains a list of all tiles which should be rendered for one render-to-texture tile\n * e.g. render 4 raster-tiles with size 256px to the 512px render-to-texture tile\n */\n _coordsDescendingInv: {[_: string]: {[_:string]: Array}};\n /**\n * create a string representation of all to tiles rendered to render-to-texture tiles\n * this string representation is used to check if tile should be re-rendered.\n */\n _coordsDescendingInvStr: {[_: string]: {[_:string]: string}};\n /**\n * store for render-stacks\n * a render stack is a set of layers which should be rendered into one texture\n * every stylesheet can have multiple stacks. A new stack is created if layers which should\n * not rendered to texture sit inbetween layers which should rendered to texture. e.g. hillshading or symbols\n */\n _stacks: Array>;\n /**\n * remember the previous processed layer to check if a new stack is needed\n */\n _prevType: string;\n /**\n * a list of tiles that can potentially rendered\n */\n _renderableTiles: Array;\n /**\n * a list of tiles that should be rendered to screen in the next render-call\n */\n _rttTiles: Array;\n /**\n * a list of all layer-ids which should be rendered\n */\n _renderableLayerIds: Array;\n\n constructor(painter: Painter, terrain: Terrain) {\n this.painter = painter;\n this.terrain = terrain;\n this.pool = new RenderPool(painter.context, 30, terrain.sourceCache.tileSize * terrain.qualityFactor);\n }\n\n destruct() {\n this.pool.destruct();\n }\n\n getTexture(tile: Tile): Texture {\n return this.pool.getObjectForId(tile.rtt[this._stacks.length - 1].id).texture;\n }\n\n prepareForRender(style: Style, zoom: number) {\n this._stacks = [];\n this._prevType = null;\n this._rttTiles = [];\n this._renderableTiles = this.terrain.sourceCache.getRenderableTiles();\n this._renderableLayerIds = style._order.filter(id => !style._layers[id].isHidden(zoom));\n\n this._coordsDescendingInv = {};\n for (const id in style.sourceCaches) {\n this._coordsDescendingInv[id] = {};\n const tileIDs = style.sourceCaches[id].getVisibleCoordinates();\n for (const tileID of tileIDs) {\n const keys = this.terrain.sourceCache.getTerrainCoords(tileID);\n for (const key in keys) {\n if (!this._coordsDescendingInv[id][key]) this._coordsDescendingInv[id][key] = [];\n this._coordsDescendingInv[id][key].push(keys[key]);\n }\n }\n }\n\n this._coordsDescendingInvStr = {};\n for (const id of style._order) {\n const layer = style._layers[id], source = layer.source;\n if (LAYERS[layer.type]) {\n if (!this._coordsDescendingInvStr[source]) {\n this._coordsDescendingInvStr[source] = {};\n for (const key in this._coordsDescendingInv[source])\n this._coordsDescendingInvStr[source][key] = this._coordsDescendingInv[source][key].map(c => c.key).sort().join();\n }\n }\n }\n\n // check tiles to render\n for (const tile of this._renderableTiles) {\n for (const source in this._coordsDescendingInvStr) {\n // rerender if there are more coords to render than in the last rendering\n const coords = this._coordsDescendingInvStr[source][tile.tileID.key];\n if (coords && coords !== tile.rttCoords[source]) tile.rtt = [];\n }\n }\n }\n\n /**\n * due that switching textures is relatively slow, the render\n * layer-by-layer context is not practicable. To bypass this problem\n * this lines of code stack all layers and later render all at once.\n * Because of the stylesheet possibility to mixing render-to-texture layers\n * and 'live'-layers (f.e. symbols) it is necessary to create more stacks. For example\n * a symbol-layer is in between of fill-layers.\n * @param layer - the layer to render\n * @returns if true layer is rendered to texture, otherwise false\n */\n renderLayer(layer: StyleLayer): boolean {\n if (layer.isHidden(this.painter.transform.zoom)) return false;\n\n const type = layer.type;\n const painter = this.painter;\n const isLastLayer = this._renderableLayerIds[this._renderableLayerIds.length - 1] === layer.id;\n\n // remember background, fill, line & raster layer to render into a stack\n if (LAYERS[type]) {\n // create a new stack if previous layer was not rendered to texture (f.e. symbols)\n if (!this._prevType || !LAYERS[this._prevType]) this._stacks.push([]);\n // push current render-to-texture layer to render-stack\n this._prevType = type;\n this._stacks[this._stacks.length - 1].push(layer.id);\n // rendering is done later, all in once\n if (!isLastLayer) return true;\n }\n\n // in case a stack is finished render all collected stack-layers into a texture\n if (LAYERS[this._prevType] || (LAYERS[type] && isLastLayer)) {\n this._prevType = type;\n const stack = this._stacks.length - 1, layers = this._stacks[stack] || [];\n for (const tile of this._renderableTiles) {\n // if render pool is full draw current tiles to screen and free pool\n if (this.pool.isFull()) {\n drawTerrain(this.painter, this.terrain, this._rttTiles);\n this._rttTiles = [];\n this.pool.freeAllObjects();\n }\n this._rttTiles.push(tile);\n // check for cached PoolObject\n if (tile.rtt[stack]) {\n const obj = this.pool.getObjectForId(tile.rtt[stack].id);\n if (obj.stamp === tile.rtt[stack].stamp) {\n this.pool.useObject(obj);\n continue;\n }\n }\n // get free PoolObject\n const obj = this.pool.getOrCreateFreeObject();\n this.pool.useObject(obj);\n this.pool.stampObject(obj);\n tile.rtt[stack] = {id: obj.id, stamp: obj.stamp};\n // prepare PoolObject for rendering\n painter.context.bindFramebuffer.set(obj.fbo.framebuffer);\n painter.context.clear({color: Color.transparent, stencil: 0});\n painter.currentStencilSource = undefined;\n for (let l = 0; l < layers.length; l++) {\n const layer = painter.style._layers[layers[l]];\n const coords = layer.source ? this._coordsDescendingInv[layer.source][tile.tileID.key] : [tile.tileID];\n painter.context.viewport.set([0, 0, obj.fbo.width, obj.fbo.height]);\n painter._renderTileClippingMasks(layer, coords);\n painter.renderLayer(painter, painter.style.sourceCaches[layer.source], layer, coords);\n if (layer.source) tile.rttCoords[layer.source] = this._coordsDescendingInvStr[layer.source][tile.tileID.key];\n }\n }\n drawTerrain(this.painter, this.terrain, this._rttTiles);\n this._rttTiles = [];\n this.pool.freeAllObjects();\n\n return LAYERS[type];\n }\n\n return false;\n }\n\n}\n","import {extend, warnOnce, uniqueId, isImageBitmap} from '../util/util';\nimport {browser} from '../util/browser';\nimport {DOM} from '../util/dom';\nimport packageJSON from '../../package.json' assert {type: 'json'};\n\nimport {getJSON} from '../util/ajax';\nimport {ImageRequest} from '../util/image_request';\nimport type {GetImageCallback} from '../util/image_request';\n\nimport {RequestManager, ResourceType} from '../util/request_manager';\nimport {Style, StyleSwapOptions} from '../style/style';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {Painter} from '../render/painter';\nimport {Transform} from '../geo/transform';\nimport {Hash} from './hash';\nimport {HandlerManager} from './handler_manager';\nimport {Camera, CameraOptions, CameraUpdateTransformFunction, FitBoundsOptions} from './camera';\nimport {LngLat} from '../geo/lng_lat';\nimport {LngLatBounds} from '../geo/lng_lat_bounds';\nimport Point from '@mapbox/point-geometry';\nimport {AttributionControl} from './control/attribution_control';\nimport {LogoControl} from './control/logo_control';\n\nimport {RGBAImage} from '../util/image';\nimport {Event, ErrorEvent, Listener} from '../util/evented';\nimport {MapEventType, MapLayerEventType, MapMouseEvent, MapSourceDataEvent, MapStyleDataEvent} from './events';\nimport {TaskQueue} from '../util/task_queue';\nimport {throttle} from '../util/throttle';\nimport {webpSupported} from '../util/webp_supported';\nimport {PerformanceMarkers, PerformanceUtils} from '../util/performance';\nimport {Source, SourceClass} from '../source/source';\nimport {StyleLayer} from '../style/style_layer';\n\nimport type {RequestTransformFunction} from '../util/request_manager';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport type {LngLatBoundsLike} from '../geo/lng_lat_bounds';\nimport type {AddLayerObject, FeatureIdentifier, StyleOptions, StyleSetterOptions} from '../style/style';\nimport type {MapDataEvent} from './events';\nimport type {StyleImage, StyleImageInterface, StyleImageMetadata} from '../style/style_image';\nimport type {PointLike} from './camera';\nimport type {ScrollZoomHandler} from './handler/scroll_zoom';\nimport type {BoxZoomHandler} from './handler/box_zoom';\nimport type {AroundCenterOptions, TwoFingersTouchPitchHandler} from './handler/two_fingers_touch';\nimport type {DragRotateHandler} from './handler/shim/drag_rotate';\nimport {DragPanHandler, DragPanOptions} from './handler/shim/drag_pan';\n\nimport type {KeyboardHandler} from './handler/keyboard';\nimport type {DoubleClickZoomHandler} from './handler/shim/dblclick_zoom';\nimport type {TwoFingersTouchZoomRotateHandler} from './handler/shim/two_fingers_touch';\nimport {defaultLocale} from './default_locale';\nimport type {TaskID} from '../util/task_queue';\nimport type {Cancelable} from '../types/cancelable';\nimport type {\n FilterSpecification,\n StyleSpecification,\n LightSpecification,\n SourceSpecification,\n TerrainSpecification\n} from '@maplibre/maplibre-gl-style-spec';\n\nimport {Callback} from '../types/callback';\nimport type {ControlPosition, IControl} from './control/control';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\nimport {Terrain} from '../render/terrain';\nimport {RenderToTexture} from '../render/render_to_texture';\nimport {config} from '../util/config';\nimport type {QueryRenderedFeaturesOptions, QuerySourceFeatureOptions} from '../source/query_features';\n\nconst version = packageJSON.version;\n\n/**\n * The {@link Map} options object.\n */\nexport type MapOptions = {\n /**\n * If `true`, the map's position (zoom, center latitude, center longitude, bearing, and pitch) will be synced with the hash fragment of the page's URL.\n * For example, `http://path/to/my/page.html#2.59/39.26/53.07/-24.1/60`.\n * An additional string may optionally be provided to indicate a parameter-styled hash,\n * e.g. http://path/to/my/page.html#map=2.59/39.26/53.07/-24.1/60&foo=bar, where foo\n * is a custom parameter and bar is an arbitrary hash distinct from the map hash.\n * @defaultValue false\n */\n hash?: boolean | string;\n /**\n * If `false`, no mouse, touch, or keyboard listeners will be attached to the map, so it will not respond to interaction.\n * @defaultValue true\n */\n interactive?: boolean;\n /**\n * The HTML element in which MapLibre GL JS will render the map, or the element's string `id`. The specified element must have no children.\n */\n container: HTMLElement | string;\n /**\n * The threshold, measured in degrees, that determines when the map's\n * bearing will snap to north. For example, with a `bearingSnap` of 7, if the user rotates\n * the map within 7 degrees of north, the map will automatically snap to exact north.\n * @defaultValue 7\n */\n bearingSnap?: number;\n /**\n * If `true`, an {@link AttributionControl} will be added to the map.\n * @defaultValue true\n */\n attributionControl?: boolean;\n /**\n * Attribution text to show in an {@link AttributionControl}. Only applicable if `options.attributionControl` is `true`.\n */\n customAttribution?: string | Array;\n /**\n * If `true`, the MapLibre logo will be shown.\n * @defaultValue false\n */\n maplibreLogo?: boolean;\n /**\n * A string representing the position of the MapLibre wordmark on the map. Valid options are `top-left`,`top-right`, `bottom-left`, or `bottom-right`.\n * @defaultValue 'bottom-left'\n */\n logoPosition?: ControlPosition;\n /**\n * If `true`, map creation will fail if the performance of MapLibre GL JS would be dramatically worse than expected\n * (i.e. a software renderer would be used).\n * @defaultValue false\n */\n failIfMajorPerformanceCaveat?: boolean;\n /**\n * If `true`, the map's canvas can be exported to a PNG using `map.getCanvas().toDataURL()`. This is `false` by default as a performance optimization.\n * @defaultValue false\n */\n preserveDrawingBuffer?: boolean;\n /**\n * If `true`, the gl context will be created with MSAA antialiasing, which can be useful for antialiasing custom layers. This is `false` by default as a performance optimization.\n */\n antialias?: boolean;\n /**\n * If `false`, the map won't attempt to re-request tiles once they expire per their HTTP `cacheControl`/`expires` headers.\n * @defaultValue true\n */\n refreshExpiredTiles?: boolean;\n /**\n * If set, the map will be constrained to the given bounds.\n */\n maxBounds?: LngLatBoundsLike;\n /**\n * If `true`, the \"scroll to zoom\" interaction is enabled. {@link AroundCenterOptions} are passed as options to {@link ScrollZoomHandler#enable}.\n * @defaultValue true\n */\n scrollZoom?: boolean | AroundCenterOptions;\n /**\n * The minimum zoom level of the map (0-24).\n * @defaultValue 0\n */\n minZoom?: number | null;\n /**\n * The maximum zoom level of the map (0-24).\n * @defaultValue 22\n */\n maxZoom?: number | null;\n /**\n * The minimum pitch of the map (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * @defaultValue 0\n */\n minPitch?: number | null;\n /**\n * The maximum pitch of the map (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * @defaultValue 60\n */\n maxPitch?: number | null;\n /**\n * If `true`, the \"box zoom\" interaction is enabled (see {@link BoxZoomHandler}).\n * @defaultValue true\n */\n boxZoom?: boolean;\n /**\n * If `true`, the \"drag to rotate\" interaction is enabled (see {@link DragRotateHandler}).\n * @defaultValue true\n */\n dragRotate?: boolean;\n /**\n * If `true`, the \"drag to pan\" interaction is enabled. An `Object` value is passed as options to {@link DragPanHandler#enable}.\n * @defaultValue true\n */\n dragPan?: boolean | DragPanOptions;\n /**\n * If `true`, keyboard shortcuts are enabled (see {@link KeyboardHandler}).\n * @defaultValue true\n */\n keyboard?: boolean;\n /**\n * If `true`, the \"double click to zoom\" interaction is enabled (see {@link DoubleClickZoomHandler}).\n * @defaultValue true\n */\n doubleClickZoom?: boolean;\n /**\n * If `true`, the \"pinch to rotate and zoom\" interaction is enabled. An `Object` value is passed as options to {@link TwoFingersTouchZoomRotateHandler#enable}.\n * @defaultValue true\n */\n touchZoomRotate?: boolean | AroundCenterOptions;\n /**\n * If `true`, the \"drag to pitch\" interaction is enabled. An `Object` value is passed as options to {@link TwoFingersTouchPitchHandler#enable}.\n * @defaultValue true\n */\n touchPitch?: boolean | AroundCenterOptions;\n /**\n * If `true` or set to an options object, the map is only accessible on desktop while holding Command/Ctrl and only accessible on mobile with two fingers. Interacting with the map using normal gestures will trigger an informational screen. With this option enabled, \"drag to pitch\" requires a three-finger gesture. Cooperative gestures are disabled when a map enters fullscreen using {@link FullscreenControl}.\n * @defaultValue undefined\n */\n cooperativeGestures?: boolean | GestureOptions;\n /**\n * If `true`, the map will automatically resize when the browser window resizes.\n * @defaultValue true\n */\n trackResize?: boolean;\n /**\n * The initial geographical centerpoint of the map. If `center` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `[0, 0]` Note: MapLibre GL JS uses longitude, latitude coordinate order (as opposed to latitude, longitude) to match GeoJSON.\n * @defaultValue [0, 0]\n */\n center?: LngLatLike;\n /**\n * The initial zoom level of the map. If `zoom` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`.\n * @defaultValue 0\n */\n zoom?: number;\n /**\n * The initial bearing (rotation) of the map, measured in degrees counter-clockwise from north. If `bearing` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`.\n * @defaultValue 0\n */\n bearing?: number;\n /**\n * The initial pitch (tilt) of the map, measured in degrees away from the plane of the screen (0-85). If `pitch` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`. Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * @defaultValue 0\n */\n pitch?: number;\n /**\n * If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`:\n * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire\n * container, there will be blank space beyond 180 and -180 degrees longitude.\n * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the\n * map and the other on the left edge of the map) at every zoom level.\n * @defaultValue true\n */\n renderWorldCopies?: boolean;\n /**\n * The maximum number of tiles stored in the tile cache for a given source. If omitted, the cache will be dynamically sized based on the current viewport which can be set using `maxTileCacheZoomLevels` constructor options.\n * @defaultValue null\n */\n maxTileCacheSize?: number;\n /**\n * The maximum number of zoom levels for which to store tiles for a given source. Tile cache dynamic size is calculated by multiplying `maxTileCacheZoomLevels` with the approximate number of tiles in the viewport for a given source.\n * @defaultValue 5\n */\n maxTileCacheZoomLevels?: number;\n /**\n * A callback run before the Map makes a request for an external URL. The callback can be used to modify the url, set headers, or set the credentials property for cross-origin requests.\n * Expected to return an object with a `url` property and optionally `headers` and `credentials` properties.\n */\n transformRequest?: RequestTransformFunction;\n /**\n * A callback run before the map's camera is moved due to user input or animation. The callback can be used to modify the new center, zoom, pitch and bearing.\n * Expected to return an object containing center, zoom, pitch or bearing values to overwrite.\n */\n transformCameraUpdate?: CameraUpdateTransformFunction;\n /**\n * A patch to apply to the default localization table for UI strings, e.g. control tooltips. The `locale` object maps namespaced UI string IDs to translated strings in the target language; see `src/ui/default_locale.js` for an example with all supported string IDs. The object may specify all UI strings (thereby adding support for a new translation) or only a subset of strings (thereby patching the default translation table).\n * @defaultValue null\n */\n locale?: any;\n /**\n * Controls the duration of the fade-in/fade-out animation for label collisions after initial map load, in milliseconds. This setting affects all symbol layers. This setting does not affect the duration of runtime styling transitions or raster tile cross-fading.\n * @defaultValue 300\n */\n fadeDuration?: number;\n /**\n * If `true`, symbols from multiple sources can collide with each other during collision detection. If `false`, collision detection is run separately for the symbols in each source.\n * @defaultValue true\n */\n crossSourceCollisions?: boolean;\n /**\n * If `true`, Resource Timing API information will be collected for requests made by GeoJSON and Vector Tile web workers (this information is normally inaccessible from the main Javascript thread). Information will be returned in a `resourceTiming` property of relevant `data` events.\n * @defaultValue false\n */\n collectResourceTiming?: boolean;\n /**\n * The max number of pixels a user can shift the mouse pointer during a click for it to be considered a valid click (as opposed to a mouse drag).\n * @defaultValue true\n */\n clickTolerance?: number;\n /**\n * The initial bounds of the map. If `bounds` is specified, it overrides `center` and `zoom` constructor options.\n */\n bounds?: LngLatBoundsLike;\n /**\n * A {@link FitBoundsOptions} options object to use _only_ when fitting the initial `bounds` provided above.\n */\n fitBoundsOptions?: FitBoundsOptions;\n /**\n * Defines a CSS\n * font-family for locally overriding generation of glyphs in the 'CJK Unified Ideographs', 'Hiragana', 'Katakana' and 'Hangul Syllables' ranges.\n * In these ranges, font settings from the map's style will be ignored, except for font-weight keywords (light/regular/medium/bold).\n * Set to `false`, to enable font settings from the map's style for these glyph ranges.\n * The purpose of this option is to avoid bandwidth-intensive glyph server requests. (See [Use locally generated ideographs](https://maplibre.org/maplibre-gl-js/docs/examples/local-ideographs).)\n * @defaultValue 'sans-serif'\n */\n localIdeographFontFamily?: string;\n /**\n * The map's MapLibre style. This must be a JSON object conforming to\n * the schema described in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/),\n * or a URL to such JSON.\n */\n style: StyleSpecification | string;\n /**\n * If `false`, the map's pitch (tilt) control with \"drag to rotate\" interaction will be disabled.\n * @defaultValue true\n */\n pitchWithRotate?: boolean;\n /**\n * The pixel ratio. The canvas' `width` attribute will be `container.clientWidth * pixelRatio` and its `height` attribute will be `container.clientHeight * pixelRatio`. Defaults to `devicePixelRatio` if not specified.\n */\n pixelRatio?: number;\n /**\n * If false, style validation will be skipped. Useful in production environment.\n * @defaultValue true\n */\n validateStyle?: boolean;\n /**\n * The canvas' `width` and `height` max size. The values are passed as an array where the first element is max width and the second element is max height.\n * You shouldn't set this above WebGl `MAX_TEXTURE_SIZE`. Defaults to [4096, 4096].\n */\n maxCanvasSize?: [number, number];\n};\n\n/**\n * An options object for the gesture settings\n * @example\n * ```ts\n * let options = {\n * windowsHelpText: \"Use Ctrl + scroll to zoom the map\",\n * macHelpText: \"Use ⌘ + scroll to zoom the map\",\n * mobileHelpText: \"Use two fingers to move the map\",\n * }\n * ```\n */\nexport type GestureOptions = {\n windowsHelpText?: string;\n macHelpText?: string;\n mobileHelpText?: string;\n};\n\nexport type AddImageOptions = {\n\n}\n\n// See article here: https://medium.com/terria/typescript-transforming-optional-properties-to-required-properties-that-may-be-undefined-7482cb4e1585\ntype Complete = {\n [P in keyof Required]: Pick extends Required> ? T[P] : (T[P] | undefined);\n}\n\n// This type is used inside map since all properties are assigned a default value.\nexport type CompleteMapOptions = Complete;\n\nconst defaultMinZoom = -2;\nconst defaultMaxZoom = 22;\n\n// the default values, but also the valid range\nconst defaultMinPitch = 0;\nconst defaultMaxPitch = 60;\n\n// use this variable to check maxPitch for validity\nconst maxPitchThreshold = 85;\n\nconst defaultOptions = {\n center: [0, 0],\n zoom: 0,\n bearing: 0,\n pitch: 0,\n\n minZoom: defaultMinZoom,\n maxZoom: defaultMaxZoom,\n\n minPitch: defaultMinPitch,\n maxPitch: defaultMaxPitch,\n\n interactive: true,\n scrollZoom: true,\n boxZoom: true,\n dragRotate: true,\n dragPan: true,\n keyboard: true,\n doubleClickZoom: true,\n touchZoomRotate: true,\n touchPitch: true,\n cooperativeGestures: undefined,\n\n bearingSnap: 7,\n clickTolerance: 3,\n pitchWithRotate: true,\n\n hash: false,\n attributionControl: true,\n maplibreLogo: false,\n\n failIfMajorPerformanceCaveat: false,\n preserveDrawingBuffer: false,\n trackResize: true,\n renderWorldCopies: true,\n refreshExpiredTiles: true,\n maxTileCacheSize: null,\n maxTileCacheZoomLevels: config.MAX_TILE_CACHE_ZOOM_LEVELS,\n localIdeographFontFamily: 'sans-serif',\n transformRequest: null,\n transformCameraUpdate: null,\n fadeDuration: 300,\n crossSourceCollisions: true,\n validateStyle: true,\n /**Because GL MAX_TEXTURE_SIZE is usually at least 4096px. */\n maxCanvasSize: [4096, 4096]\n} as CompleteMapOptions;\n\n/**\n * The `Map` object represents the map on your page. It exposes methods\n * and properties that enable you to programmatically change the map,\n * and fires events as users interact with it.\n *\n * You create a `Map` by specifying a `container` and other options, see {@link MapOptions} for the full list.\n * Then MapLibre GL JS initializes the map on the page and returns your `Map` object.\n *\n * @group Main\n *\n * @example\n * ```ts\n * let map = new maplibregl.Map({\n * container: 'map',\n * center: [-122.420679, 37.772537],\n * zoom: 13,\n * style: style_object,\n * hash: true,\n * transformRequest: (url, resourceType)=> {\n * if(resourceType === 'Source' && url.startsWith('http://myHost')) {\n * return {\n * url: url.replace('http', 'https'),\n * headers: { 'my-custom-header': true},\n * credentials: 'include' // Include cookies for cross-origin requests\n * }\n * }\n * }\n * });\n * ```\n * @see [Display a map](https://maplibre.org/maplibre-gl-js/docs/examples/simple-map/)\n */\nexport class Map extends Camera {\n style: Style;\n painter: Painter;\n handlers: HandlerManager;\n\n _container: HTMLElement;\n _canvasContainer: HTMLElement;\n _controlContainer: HTMLElement;\n _controlPositions: {[_: string]: HTMLElement};\n _interactive: boolean;\n _cooperativeGestures: boolean | GestureOptions;\n _cooperativeGesturesScreen: HTMLElement;\n _metaKey: keyof MouseEvent;\n _showTileBoundaries: boolean;\n _showCollisionBoxes: boolean;\n _showPadding: boolean;\n _showOverdrawInspector: boolean;\n _repaint: boolean;\n _vertices: boolean;\n _canvas: HTMLCanvasElement;\n _maxTileCacheSize: number;\n _maxTileCacheZoomLevels: number;\n _frame: Cancelable;\n _styleDirty: boolean;\n _sourcesDirty: boolean;\n _placementDirty: boolean;\n\n _loaded: boolean;\n _idleTriggered: boolean;\n // accounts for placement finishing as well\n _fullyLoaded: boolean;\n _trackResize: boolean;\n _resizeObserver: ResizeObserver;\n _preserveDrawingBuffer: boolean;\n _failIfMajorPerformanceCaveat: boolean;\n _antialias: boolean;\n _refreshExpiredTiles: boolean;\n _hash: Hash;\n _delegatedListeners: any;\n _fadeDuration: number;\n _crossSourceCollisions: boolean;\n _crossFadingFactor: number;\n _collectResourceTiming: boolean;\n _renderTaskQueue: TaskQueue;\n _controls: Array;\n _mapId: number;\n _localIdeographFontFamily: string;\n _validateStyle: boolean;\n _requestManager: RequestManager;\n _locale: any;\n _removed: boolean;\n _clickTolerance: number;\n _overridePixelRatio: number | null;\n _maxCanvasSize: [number, number];\n _terrainDataCallback: (e: MapStyleDataEvent | MapSourceDataEvent) => void;\n\n /**\n * @internal\n * image queue throttling handle. To be used later when clean up\n */\n _imageQueueHandle: number;\n\n /**\n * The map's {@link ScrollZoomHandler}, which implements zooming in and out with a scroll wheel or trackpad.\n * Find more details and examples using `scrollZoom` in the {@link ScrollZoomHandler} section.\n */\n scrollZoom: ScrollZoomHandler;\n\n /**\n * The map's {@link BoxZoomHandler}, which implements zooming using a drag gesture with the Shift key pressed.\n * Find more details and examples using `boxZoom` in the {@link BoxZoomHandler} section.\n */\n boxZoom: BoxZoomHandler;\n\n /**\n * The map's {@link DragRotateHandler}, which implements rotating the map while dragging with the right\n * mouse button or with the Control key pressed. Find more details and examples using `dragRotate`\n * in the {@link DragRotateHandler} section.\n */\n dragRotate: DragRotateHandler;\n\n /**\n * The map's {@link DragPanHandler}, which implements dragging the map with a mouse or touch gesture.\n * Find more details and examples using `dragPan` in the {@link DragPanHandler} section.\n */\n dragPan: DragPanHandler;\n\n /**\n * The map's {@link KeyboardHandler}, which allows the user to zoom, rotate, and pan the map using keyboard\n * shortcuts. Find more details and examples using `keyboard` in the {@link KeyboardHandler} section.\n */\n keyboard: KeyboardHandler;\n\n /**\n * The map's {@link DoubleClickZoomHandler}, which allows the user to zoom by double clicking.\n * Find more details and examples using `doubleClickZoom` in the {@link DoubleClickZoomHandler} section.\n */\n doubleClickZoom: DoubleClickZoomHandler;\n\n /**\n * The map's {@link TwoFingersTouchZoomRotateHandler}, which allows the user to zoom or rotate the map with touch gestures.\n * Find more details and examples using `touchZoomRotate` in the {@link TwoFingersTouchZoomRotateHandler} section.\n */\n touchZoomRotate: TwoFingersTouchZoomRotateHandler;\n\n /**\n * The map's {@link TwoFingersTouchPitchHandler}, which allows the user to pitch the map with touch gestures.\n * Find more details and examples using `touchPitch` in the {@link TwoFingersTouchPitchHandler} section.\n */\n touchPitch: TwoFingersTouchPitchHandler;\n\n constructor(options: MapOptions) {\n PerformanceUtils.mark(PerformanceMarkers.create);\n\n options = extend({}, defaultOptions, options);\n\n if (options.minZoom != null && options.maxZoom != null && options.minZoom > options.maxZoom) {\n throw new Error('maxZoom must be greater than or equal to minZoom');\n }\n\n if (options.minPitch != null && options.maxPitch != null && options.minPitch > options.maxPitch) {\n throw new Error('maxPitch must be greater than or equal to minPitch');\n }\n\n if (options.minPitch != null && options.minPitch < defaultMinPitch) {\n throw new Error(`minPitch must be greater than or equal to ${defaultMinPitch}`);\n }\n\n if (options.maxPitch != null && options.maxPitch > maxPitchThreshold) {\n throw new Error(`maxPitch must be less than or equal to ${maxPitchThreshold}`);\n }\n\n const transform = new Transform(options.minZoom, options.maxZoom, options.minPitch, options.maxPitch, options.renderWorldCopies);\n super(transform, {bearingSnap: options.bearingSnap});\n\n this._interactive = options.interactive;\n this._cooperativeGestures = options.cooperativeGestures;\n this._metaKey = navigator.platform.indexOf('Mac') === 0 ? 'metaKey' : 'ctrlKey';\n this._maxTileCacheSize = options.maxTileCacheSize;\n this._maxTileCacheZoomLevels = options.maxTileCacheZoomLevels;\n this._failIfMajorPerformanceCaveat = options.failIfMajorPerformanceCaveat;\n this._preserveDrawingBuffer = options.preserveDrawingBuffer;\n this._antialias = options.antialias;\n this._trackResize = options.trackResize;\n this._bearingSnap = options.bearingSnap;\n this._refreshExpiredTiles = options.refreshExpiredTiles;\n this._fadeDuration = options.fadeDuration;\n this._crossSourceCollisions = options.crossSourceCollisions;\n this._crossFadingFactor = 1;\n this._collectResourceTiming = options.collectResourceTiming;\n this._renderTaskQueue = new TaskQueue();\n this._controls = [];\n this._mapId = uniqueId();\n this._locale = extend({}, defaultLocale, options.locale);\n this._clickTolerance = options.clickTolerance;\n this._overridePixelRatio = options.pixelRatio;\n this._maxCanvasSize = options.maxCanvasSize;\n this.transformCameraUpdate = options.transformCameraUpdate;\n\n this._imageQueueHandle = ImageRequest.addThrottleControl(() => this.isMoving());\n\n this._requestManager = new RequestManager(options.transformRequest);\n\n if (typeof options.container === 'string') {\n this._container = document.getElementById(options.container);\n if (!this._container) {\n throw new Error(`Container '${options.container}' not found.`);\n }\n } else if (options.container instanceof HTMLElement) {\n this._container = options.container;\n } else {\n throw new Error('Invalid type: \\'container\\' must be a String or HTMLElement.');\n }\n\n if (options.maxBounds) {\n this.setMaxBounds(options.maxBounds);\n }\n\n this._setupContainer();\n this._setupPainter();\n\n this.on('move', () => this._update(false));\n this.on('moveend', () => this._update(false));\n this.on('zoom', () => this._update(true));\n this.on('terrain', () => {\n this.painter.terrainFacilitator.dirty = true;\n this._update(true);\n });\n this.once('idle', () => { this._idleTriggered = true; });\n\n if (typeof window !== 'undefined') {\n addEventListener('online', this._onWindowOnline, false);\n let initialResizeEventCaptured = false;\n const throttledResizeCallback = throttle((entries: ResizeObserverEntry[]) => {\n if (this._trackResize && !this._removed) {\n this.resize(entries)._update();\n }\n }, 50);\n this._resizeObserver = new ResizeObserver((entries) => {\n if (!initialResizeEventCaptured) {\n initialResizeEventCaptured = true;\n return;\n }\n throttledResizeCallback(entries);\n });\n this._resizeObserver.observe(this._container);\n }\n\n this.handlers = new HandlerManager(this, options as CompleteMapOptions);\n\n if (this._cooperativeGestures) {\n this._setupCooperativeGestures();\n }\n\n const hashName = (typeof options.hash === 'string' && options.hash) || undefined;\n this._hash = options.hash && (new Hash(hashName)).addTo(this);\n // don't set position from options if set through hash\n if (!this._hash || !this._hash._onHashChange()) {\n this.jumpTo({\n center: options.center,\n zoom: options.zoom,\n bearing: options.bearing,\n pitch: options.pitch\n });\n\n if (options.bounds) {\n this.resize();\n this.fitBounds(options.bounds, extend({}, options.fitBoundsOptions, {duration: 0}));\n }\n }\n\n this.resize();\n\n this._localIdeographFontFamily = options.localIdeographFontFamily;\n this._validateStyle = options.validateStyle;\n\n if (options.style) this.setStyle(options.style, {localIdeographFontFamily: options.localIdeographFontFamily});\n\n if (options.attributionControl)\n this.addControl(new AttributionControl({customAttribution: options.customAttribution}));\n\n if (options.maplibreLogo)\n this.addControl(new LogoControl(), options.logoPosition);\n\n this.on('style.load', () => {\n if (this.transform.unmodified) {\n this.jumpTo(this.style.stylesheet as any);\n }\n });\n this.on('data', (event: MapDataEvent) => {\n this._update(event.dataType === 'style');\n this.fire(new Event(`${event.dataType}data`, event));\n });\n this.on('dataloading', (event: MapDataEvent) => {\n this.fire(new Event(`${event.dataType}dataloading`, event));\n });\n this.on('dataabort', (event: MapDataEvent) => {\n this.fire(new Event('sourcedataabort', event));\n });\n }\n\n /**\n * @internal\n * Returns a unique number for this map instance which is used for the MapLoadEvent\n * to make sure we only fire one event per instantiated map object.\n * @returns the uniq map ID\n */\n _getMapId() {\n return this._mapId;\n }\n\n /**\n * Adds an {@link IControl} to the map, calling `control.onAdd(this)`.\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param control - The {@link IControl} to add.\n * @param position - position on the map to which the control will be added.\n * Valid values are `'top-left'`, `'top-right'`, `'bottom-left'`, and `'bottom-right'`. Defaults to `'top-right'`.\n * @returns `this`\n * @example\n * Add zoom and rotation controls to the map.\n * ```ts\n * map.addControl(new maplibregl.NavigationControl());\n * ```\n * @see [Display map navigation controls](https://maplibre.org/maplibre-gl-js/docs/examples/navigation/)\n */\n addControl(control: IControl, position?: ControlPosition): Map {\n if (position === undefined) {\n if (control.getDefaultPosition) {\n position = control.getDefaultPosition();\n } else {\n position = 'top-right';\n }\n }\n if (!control || !control.onAdd) {\n return this.fire(new ErrorEvent(new Error(\n 'Invalid argument to map.addControl(). Argument must be a control with onAdd and onRemove methods.')));\n }\n const controlElement = control.onAdd(this);\n this._controls.push(control);\n\n const positionContainer = this._controlPositions[position];\n if (position.indexOf('bottom') !== -1) {\n positionContainer.insertBefore(controlElement, positionContainer.firstChild);\n } else {\n positionContainer.appendChild(controlElement);\n }\n return this;\n }\n\n /**\n * Removes the control from the map.\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param control - The {@link IControl} to remove.\n * @returns `this`\n * @example\n * ```ts\n * // Define a new navigation control.\n * let navigation = new maplibregl.NavigationControl();\n * // Add zoom and rotation controls to the map.\n * map.addControl(navigation);\n * // Remove zoom and rotation controls from the map.\n * map.removeControl(navigation);\n * ```\n */\n removeControl(control: IControl): Map {\n if (!control || !control.onRemove) {\n return this.fire(new ErrorEvent(new Error(\n 'Invalid argument to map.removeControl(). Argument must be a control with onAdd and onRemove methods.')));\n }\n const ci = this._controls.indexOf(control);\n if (ci > -1) this._controls.splice(ci, 1);\n control.onRemove(this);\n return this;\n }\n\n /**\n * Checks if a control exists on the map.\n *\n * @param control - The {@link IControl} to check.\n * @returns true if map contains control.\n * @example\n * ```ts\n * // Define a new navigation control.\n * let navigation = new maplibregl.NavigationControl();\n * // Add zoom and rotation controls to the map.\n * map.addControl(navigation);\n * // Check that the navigation control exists on the map.\n * map.hasControl(navigation);\n * ```\n */\n hasControl(control: IControl): boolean {\n return this._controls.indexOf(control) > -1;\n }\n\n calculateCameraOptionsFromTo(from: LngLat, altitudeFrom: number, to: LngLat, altitudeTo?: number): CameraOptions {\n if (altitudeTo == null && this.terrain) {\n altitudeTo = this.terrain.getElevationForLngLatZoom(to, this.transform.tileZoom);\n }\n return super.calculateCameraOptionsFromTo(from, altitudeFrom, to, altitudeTo);\n }\n\n /**\n * Resizes the map according to the dimensions of its\n * `container` element.\n *\n * Checks if the map container size changed and updates the map if it has changed.\n * This method must be called after the map's `container` is resized programmatically\n * or when the map is shown after being initially hidden with CSS.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, and `resize`.\n *\n * @param eventData - Additional properties to be passed to `movestart`, `move`, `resize`, and `moveend`\n * events that get triggered as a result of resize. This can be useful for differentiating the\n * source of an event (for example, user-initiated or programmatically-triggered events).\n * @returns `this`\n * @example\n * Resize the map when the map container is shown after being initially hidden with CSS.\n * ```ts\n * let mapDiv = document.getElementById('map');\n * if (mapDiv.style.visibility === true) map.resize();\n * ```\n */\n resize(eventData?: any): Map {\n const dimensions = this._containerDimensions();\n const width = dimensions[0];\n const height = dimensions[1];\n\n const clampedPixelRatio = this._getClampedPixelRatio(width, height);\n this._resizeCanvas(width, height, clampedPixelRatio);\n this.painter.resize(width, height, clampedPixelRatio);\n\n // check if we've reached GL limits, in that case further clamps pixelRatio\n if (this.painter.overLimit()) {\n const gl = this.painter.context.gl;\n // store updated _maxCanvasSize value\n this._maxCanvasSize = [gl.drawingBufferWidth, gl.drawingBufferHeight];\n const clampedPixelRatio = this._getClampedPixelRatio(width, height);\n this._resizeCanvas(width, height, clampedPixelRatio);\n this.painter.resize(width, height, clampedPixelRatio);\n }\n\n this.transform.resize(width, height);\n this._requestedCameraState?.resize(width, height);\n\n const fireMoving = !this._moving;\n if (fireMoving) {\n this.stop();\n this.fire(new Event('movestart', eventData))\n .fire(new Event('move', eventData));\n }\n\n this.fire(new Event('resize', eventData));\n\n if (fireMoving) this.fire(new Event('moveend', eventData));\n\n return this;\n }\n\n /**\n * @internal\n * Return the map's pixel ratio eventually scaled down to respect maxCanvasSize.\n * Internally you should use this and not getPixelRatio().\n */\n _getClampedPixelRatio(width: number, height: number): number {\n const {0: maxCanvasWidth, 1: maxCanvasHeight} = this._maxCanvasSize;\n const pixelRatio = this.getPixelRatio();\n\n const canvasWidth = width * pixelRatio;\n const canvasHeight = height * pixelRatio;\n\n const widthScaleFactor = canvasWidth > maxCanvasWidth ? (maxCanvasWidth / canvasWidth) : 1;\n const heightScaleFactor = canvasHeight > maxCanvasHeight ? (maxCanvasHeight / canvasHeight) : 1;\n\n return Math.min(widthScaleFactor, heightScaleFactor) * pixelRatio;\n }\n\n /**\n * Returns the map's pixel ratio.\n * Note that the pixel ratio actually applied may be lower to respect maxCanvasSize.\n * @returns The pixel ratio.\n */\n getPixelRatio(): number {\n return this._overridePixelRatio ?? devicePixelRatio;\n }\n\n /**\n * Sets the map's pixel ratio. This allows to override `devicePixelRatio`.\n * After this call, the canvas' `width` attribute will be `container.clientWidth * pixelRatio`\n * and its height attribute will be `container.clientHeight * pixelRatio`.\n * Set this to null to disable `devicePixelRatio` override.\n * Note that the pixel ratio actually applied may be lower to respect maxCanvasSize.\n * @param pixelRatio - The pixel ratio.\n */\n setPixelRatio(pixelRatio: number) {\n this._overridePixelRatio = pixelRatio;\n this.resize();\n }\n\n /**\n * Returns the map's geographical bounds. When the bearing or pitch is non-zero, the visible region is not\n * an axis-aligned rectangle, and the result is the smallest bounds that encompasses the visible region.\n * @returns The geographical bounds of the map as {@link LngLatBounds}.\n * @example\n * ```ts\n * let bounds = map.getBounds();\n * ```\n */\n getBounds(): LngLatBounds {\n return this.transform.getBounds();\n }\n\n /**\n * Returns the maximum geographical bounds the map is constrained to, or `null` if none set.\n * @returns The map object.\n * @example\n * ```ts\n * let maxBounds = map.getMaxBounds();\n * ```\n */\n getMaxBounds(): LngLatBounds | null {\n return this.transform.getMaxBounds();\n }\n\n /**\n * Sets or clears the map's geographical bounds.\n *\n * Pan and zoom operations are constrained within these bounds.\n * If a pan or zoom is performed that would\n * display regions outside these bounds, the map will\n * instead display a position and zoom level\n * as close as possible to the operation's request while still\n * remaining within the bounds.\n *\n * @param bounds - The maximum bounds to set. If `null` or `undefined` is provided, the function removes the map's maximum bounds.\n * @returns `this`\n * @example\n * Define bounds that conform to the `LngLatBoundsLike` object as set the max bounds.\n * ```ts\n * let bounds = [\n * [-74.04728, 40.68392], // [west, south]\n * [-73.91058, 40.87764] // [east, north]\n * ];\n * map.setMaxBounds(bounds);\n * ```\n */\n setMaxBounds(bounds?: LngLatBoundsLike | null): Map {\n this.transform.setMaxBounds(LngLatBounds.convert(bounds));\n return this._update();\n }\n\n /**\n * Sets or clears the map's minimum zoom level.\n * If the map's current zoom level is lower than the new minimum,\n * the map will zoom to the new minimum.\n *\n * It is not always possible to zoom out and reach the set `minZoom`.\n * Other factors such as map height may restrict zooming. For example,\n * if the map is 512px tall it will not be possible to zoom below zoom 0\n * no matter what the `minZoom` is set to.\n *\n * A {@link ErrorEvent} event will be fired if minZoom is out of bounds.\n *\n * @param minZoom - The minimum zoom level to set (-2 - 24).\n * If `null` or `undefined` is provided, the function removes the current minimum zoom (i.e. sets it to -2).\n * @returns `this`\n * @example\n * ```ts\n * map.setMinZoom(12.25);\n * ```\n */\n setMinZoom(minZoom?: number | null): Map {\n\n minZoom = minZoom === null || minZoom === undefined ? defaultMinZoom : minZoom;\n\n if (minZoom >= defaultMinZoom && minZoom <= this.transform.maxZoom) {\n this.transform.minZoom = minZoom;\n this._update();\n\n if (this.getZoom() < minZoom) this.setZoom(minZoom);\n\n return this;\n\n } else throw new Error(`minZoom must be between ${defaultMinZoom} and the current maxZoom, inclusive`);\n }\n\n /**\n * Returns the map's minimum allowable zoom level.\n *\n * @returns minZoom\n * @example\n * ```ts\n * let minZoom = map.getMinZoom();\n * ```\n */\n getMinZoom(): number { return this.transform.minZoom; }\n\n /**\n * Sets or clears the map's maximum zoom level.\n * If the map's current zoom level is higher than the new maximum,\n * the map will zoom to the new maximum.\n *\n * A {@link ErrorEvent} event will be fired if minZoom is out of bounds.\n *\n * @param maxZoom - The maximum zoom level to set.\n * If `null` or `undefined` is provided, the function removes the current maximum zoom (sets it to 22).\n * @returns `this`\n * @example\n * ```ts\n * map.setMaxZoom(18.75);\n * ```\n */\n setMaxZoom(maxZoom?: number | null): Map {\n\n maxZoom = maxZoom === null || maxZoom === undefined ? defaultMaxZoom : maxZoom;\n\n if (maxZoom >= this.transform.minZoom) {\n this.transform.maxZoom = maxZoom;\n this._update();\n\n if (this.getZoom() > maxZoom) this.setZoom(maxZoom);\n\n return this;\n\n } else throw new Error('maxZoom must be greater than the current minZoom');\n }\n\n /**\n * Returns the map's maximum allowable zoom level.\n *\n * @returns The maxZoom\n * @example\n * ```ts\n * let maxZoom = map.getMaxZoom();\n * ```\n */\n getMaxZoom(): number { return this.transform.maxZoom; }\n\n /**\n * Sets or clears the map's minimum pitch.\n * If the map's current pitch is lower than the new minimum,\n * the map will pitch to the new minimum.\n *\n * A {@link ErrorEvent} event will be fired if minPitch is out of bounds.\n *\n * @param minPitch - The minimum pitch to set (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * If `null` or `undefined` is provided, the function removes the current minimum pitch (i.e. sets it to 0).\n * @returns `this`\n */\n setMinPitch(minPitch?: number | null): Map {\n\n minPitch = minPitch === null || minPitch === undefined ? defaultMinPitch : minPitch;\n\n if (minPitch < defaultMinPitch) {\n throw new Error(`minPitch must be greater than or equal to ${defaultMinPitch}`);\n }\n\n if (minPitch >= defaultMinPitch && minPitch <= this.transform.maxPitch) {\n this.transform.minPitch = minPitch;\n this._update();\n\n if (this.getPitch() < minPitch) this.setPitch(minPitch);\n\n return this;\n\n } else throw new Error(`minPitch must be between ${defaultMinPitch} and the current maxPitch, inclusive`);\n }\n\n /**\n * Returns the map's minimum allowable pitch.\n *\n * @returns The minPitch\n */\n getMinPitch(): number { return this.transform.minPitch; }\n\n /**\n * Sets or clears the map's maximum pitch.\n * If the map's current pitch is higher than the new maximum,\n * the map will pitch to the new maximum.\n *\n * A {@link ErrorEvent} event will be fired if maxPitch is out of bounds.\n *\n * @param maxPitch - The maximum pitch to set (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * If `null` or `undefined` is provided, the function removes the current maximum pitch (sets it to 60).\n * @returns `this`\n */\n setMaxPitch(maxPitch?: number | null): Map {\n\n maxPitch = maxPitch === null || maxPitch === undefined ? defaultMaxPitch : maxPitch;\n\n if (maxPitch > maxPitchThreshold) {\n throw new Error(`maxPitch must be less than or equal to ${maxPitchThreshold}`);\n }\n\n if (maxPitch >= this.transform.minPitch) {\n this.transform.maxPitch = maxPitch;\n this._update();\n\n if (this.getPitch() > maxPitch) this.setPitch(maxPitch);\n\n return this;\n\n } else throw new Error('maxPitch must be greater than the current minPitch');\n }\n\n /**\n * Returns the map's maximum allowable pitch.\n *\n * @returns The maxPitch\n */\n getMaxPitch(): number { return this.transform.maxPitch; }\n\n /**\n * Returns the state of `renderWorldCopies`. If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`:\n * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire\n * container, there will be blank space beyond 180 and -180 degrees longitude.\n * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the\n * map and the other on the left edge of the map) at every zoom level.\n * @returns The renderWorldCopies\n * @example\n * ```ts\n * let worldCopiesRendered = map.getRenderWorldCopies();\n * ```\n * @see [Render world copies](https://maplibre.org/maplibre-gl-js/docs/examples/render-world-copies/)\n */\n getRenderWorldCopies(): boolean { return this.transform.renderWorldCopies; }\n\n /**\n * Sets the state of `renderWorldCopies`.\n *\n * @param renderWorldCopies - If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`:\n * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire\n * container, there will be blank space beyond 180 and -180 degrees longitude.\n * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the\n * map and the other on the left edge of the map) at every zoom level.\n *\n * `undefined` is treated as `true`, `null` is treated as `false`.\n * @returns `this`\n * @example\n * ```ts\n * map.setRenderWorldCopies(true);\n * ```\n * @see [Render world copies](https://maplibre.org/maplibre-gl-js/docs/examples/render-world-copies/)\n */\n setRenderWorldCopies(renderWorldCopies?: boolean | null): Map {\n this.transform.renderWorldCopies = renderWorldCopies;\n return this._update();\n }\n\n /**\n * Gets the map's cooperativeGestures option\n *\n * @returns The gestureOptions\n */\n getCooperativeGestures(): boolean | GestureOptions {\n return this._cooperativeGestures;\n }\n\n /**\n * Sets or clears the map's cooperativeGestures option\n *\n * @param gestureOptions - If `true` or set to an options object, map is only accessible on desktop while holding Command/Ctrl and only accessible on mobile with two fingers. Interacting with the map using normal gestures will trigger an informational screen. With this option enabled, \"drag to pitch\" requires a three-finger gesture.\n * @returns `this`\n */\n setCooperativeGestures(gestureOptions?: GestureOptions | boolean | null): Map {\n this._cooperativeGestures = gestureOptions;\n if (this._cooperativeGestures) {\n this._setupCooperativeGestures();\n } else {\n this._destroyCooperativeGestures();\n }\n\n return this;\n }\n\n /**\n * Returns a [Point](https://github.com/mapbox/point-geometry) representing pixel coordinates, relative to the map's `container`,\n * that correspond to the specified geographical location.\n *\n * @param lnglat - The geographical location to project.\n * @returns The [Point](https://github.com/mapbox/point-geometry) corresponding to `lnglat`, relative to the map's `container`.\n * @example\n * ```ts\n * let coordinate = [-122.420679, 37.772537];\n * let point = map.project(coordinate);\n * ```\n */\n project(lnglat: LngLatLike): Point {\n return this.transform.locationPoint(LngLat.convert(lnglat), this.style && this.terrain);\n }\n\n /**\n * Returns a {@link LngLat} representing geographical coordinates that correspond\n * to the specified pixel coordinates.\n *\n * @param point - The pixel coordinates to unproject.\n * @returns The {@link LngLat} corresponding to `point`.\n * @example\n * ```ts\n * map.on('click', function(e) {\n * // When the map is clicked, get the geographic coordinate.\n * let coordinate = map.unproject(e.point);\n * });\n * ```\n */\n unproject(point: PointLike): LngLat {\n return this.transform.pointLocation(Point.convert(point), this.terrain);\n }\n\n /**\n * Returns true if the map is panning, zooming, rotating, or pitching due to a camera animation or user gesture.\n * @returns true if the map is moving.\n * @example\n * ```ts\n * let isMoving = map.isMoving();\n * ```\n */\n isMoving(): boolean {\n return this._moving || this.handlers?.isMoving();\n }\n\n /**\n * Returns true if the map is zooming due to a camera animation or user gesture.\n * @returns true if the map is zooming.\n * @example\n * ```ts\n * let isZooming = map.isZooming();\n * ```\n */\n isZooming(): boolean {\n return this._zooming || this.handlers?.isZooming();\n }\n\n /**\n * Returns true if the map is rotating due to a camera animation or user gesture.\n * @returns true if the map is rotating.\n * @example\n * ```ts\n * map.isRotating();\n * ```\n */\n isRotating(): boolean {\n return this._rotating || this.handlers?.isRotating();\n }\n\n _createDelegatedListener(type: keyof MapEventType | string, layerId: string, listener: Listener): {\n layer: string;\n listener: Listener;\n delegates: {[type in keyof MapEventType]?: (e: any) => void};\n } {\n if (type === 'mouseenter' || type === 'mouseover') {\n let mousein = false;\n const mousemove = (e) => {\n const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, {layers: [layerId]}) : [];\n if (!features.length) {\n mousein = false;\n } else if (!mousein) {\n mousein = true;\n listener.call(this, new MapMouseEvent(type, this, e.originalEvent, {features}));\n }\n };\n const mouseout = () => {\n mousein = false;\n };\n return {layer: layerId, listener, delegates: {mousemove, mouseout}};\n } else if (type === 'mouseleave' || type === 'mouseout') {\n let mousein = false;\n const mousemove = (e) => {\n const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, {layers: [layerId]}) : [];\n if (features.length) {\n mousein = true;\n } else if (mousein) {\n mousein = false;\n listener.call(this, new MapMouseEvent(type, this, e.originalEvent));\n }\n };\n const mouseout = (e) => {\n if (mousein) {\n mousein = false;\n listener.call(this, new MapMouseEvent(type, this, e.originalEvent));\n }\n };\n return {layer: layerId, listener, delegates: {mousemove, mouseout}};\n } else {\n const delegate = (e) => {\n const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, {layers: [layerId]}) : [];\n if (features.length) {\n // Here we need to mutate the original event, so that preventDefault works as expected.\n e.features = features;\n listener.call(this, e);\n delete e.features;\n }\n };\n return {layer: layerId, listener, delegates: {[type]: delegate}};\n }\n }\n\n /**\n * @event\n * Adds a listener for events of a specified type, optionally limited to features in a specified style layer.\n * See {@link MapEventType} and {@link MapLayerEventType} for a full list of events and their description.\n *\n * | Event | Compatible with `layerId` |\n * |------------------------|---------------------------|\n * | `mousedown` | yes |\n * | `mouseup` | yes |\n * | `mouseover` | yes |\n * | `mouseout` | yes |\n * | `mousemove` | yes |\n * | `mouseenter` | yes (required) |\n * | `mouseleave` | yes (required) |\n * | `click` | yes |\n * | `dblclick` | yes |\n * | `contextmenu` | yes |\n * | `touchstart` | yes |\n * | `touchend` | yes |\n * | `touchcancel` | yes |\n * | `wheel` | |\n * | `resize` | |\n * | `remove` | |\n * | `touchmove` | |\n * | `movestart` | |\n * | `move` | |\n * | `moveend` | |\n * | `dragstart` | |\n * | `drag` | |\n * | `dragend` | |\n * | `zoomstart` | |\n * | `zoom` | |\n * | `zoomend` | |\n * | `rotatestart` | |\n * | `rotate` | |\n * | `rotateend` | |\n * | `pitchstart` | |\n * | `pitch` | |\n * | `pitchend` | |\n * | `boxzoomstart` | |\n * | `boxzoomend` | |\n * | `boxzoomcancel` | |\n * | `webglcontextlost` | |\n * | `webglcontextrestored` | |\n * | `load` | |\n * | `render` | |\n * | `idle` | |\n * | `error` | |\n * | `data` | |\n * | `styledata` | |\n * | `sourcedata` | |\n * | `dataloading` | |\n * | `styledataloading` | |\n * | `sourcedataloading` | |\n * | `styleimagemissing` | |\n * | `dataabort` | |\n * | `sourcedataabort` | |\n *\n * @param type - The event type to listen for. Events compatible with the optional `layerId` parameter are triggered\n * when the cursor enters a visible portion of the specified layer from outside that layer or outside the map canvas.\n * @param layer - The ID of a style layer or a listener if no ID is provided. Event will only be triggered if its location\n * is within a visible feature in this layer. The event will have a `features` property containing\n * an array of the matching features. If `layer` is not supplied, the event will not have a `features` property.\n * Please note that many event types are not compatible with the optional `layer` parameter.\n * @param listener - The function to be called when the event is fired.\n * @returns `this`\n * @example\n * ```ts\n * // Set an event listener that will fire\n * // when the map has finished loading\n * map.on('load', function() {\n * // Once the map has finished loading,\n * // add a new layer\n * map.addLayer({\n * id: 'points-of-interest',\n * source: {\n * type: 'vector',\n * url: 'https://maplibre.org/maplibre-style-spec/'\n * },\n * 'source-layer': 'poi_label',\n * type: 'circle',\n * paint: {\n * // MapLibre Style Specification paint properties\n * },\n * layout: {\n * // MapLibre Style Specification layout properties\n * }\n * });\n * });\n * ```\n * @example\n * ```ts\n * // Set an event listener that will fire\n * // when a feature on the countries layer of the map is clicked\n * map.on('click', 'countries', (e) => {\n * new maplibregl.Popup()\n * .setLngLat(e.lngLat)\n * .setHTML(`Country name: ${e.features[0].properties.name}`)\n * .addTo(map);\n * });\n * ```\n * @see [Display popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n * @see [Create a hover effect](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Create a draggable marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n on(\n type: T,\n layer: string,\n listener: (ev: MapLayerEventType[T] & Object) => void,\n ): Map;\n /**\n * Overload of the `on` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The listener callback.\n * @returns `this`\n */\n on(type: T, listener: (ev: MapEventType[T] & Object) => void): this;\n /**\n * Overload of the `on` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The listener callback.\n * @returns `this`\n */\n on(type: keyof MapEventType | string, listener: Listener): this;\n on(type: keyof MapEventType | string, layerIdOrListener: string | Listener, listener?: Listener): this {\n if (listener === undefined) {\n return super.on(type, layerIdOrListener as Listener);\n }\n\n const delegatedListener = this._createDelegatedListener(type, layerIdOrListener as string, listener);\n\n this._delegatedListeners = this._delegatedListeners || {};\n this._delegatedListeners[type] = this._delegatedListeners[type] || [];\n this._delegatedListeners[type].push(delegatedListener);\n\n for (const event in delegatedListener.delegates) {\n this.on(event, delegatedListener.delegates[event]);\n }\n\n return this;\n }\n\n /**\n * Adds a listener that will be called only once to a specified event type, optionally limited to features in a specified style layer.\n *\n * @event\n * @param type - The event type to listen for; one of `'mousedown'`, `'mouseup'`, `'click'`, `'dblclick'`,\n * `'mousemove'`, `'mouseenter'`, `'mouseleave'`, `'mouseover'`, `'mouseout'`, `'contextmenu'`, `'touchstart'`,\n * `'touchend'`, or `'touchcancel'`. `mouseenter` and `mouseover` events are triggered when the cursor enters\n * a visible portion of the specified layer from outside that layer or outside the map canvas. `mouseleave`\n * and `mouseout` events are triggered when the cursor leaves a visible portion of the specified layer, or leaves\n * the map canvas.\n * @param layer - The ID of a style layer or a listener if no ID is provided. Only events whose location is within a visible\n * feature in this layer will trigger the listener. The event will have a `features` property containing\n * an array of the matching features.\n * @param listener - The function to be called when the event is fired.\n * @returns `this` if listener is provided, promise otherwise to allow easier usage of async/await\n */\n once(\n type: T,\n layer: string,\n listener?: (ev: MapLayerEventType[T] & Object) => void,\n ): this | Promise;\n /**\n * Overload of the `once` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The listener callback.\n * @returns `this`\n */\n once(type: T, listener?: (ev: MapEventType[T] & Object) => void): this | Promise;\n /**\n * Overload of the `once` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The listener callback.\n * @returns `this`\n */\n once(type: keyof MapEventType | string, listener?: Listener): this | Promise;\n once(type: keyof MapEventType | string, layerIdOrListener: string | Listener, listener?: Listener): this | Promise {\n\n if (listener === undefined) {\n return super.once(type, layerIdOrListener as Listener);\n }\n\n const delegatedListener = this._createDelegatedListener(type, layerIdOrListener as string, listener);\n\n for (const event in delegatedListener.delegates) {\n this.once(event, delegatedListener.delegates[event]);\n }\n\n return this;\n }\n\n /**\n * Removes an event listener for events previously added with `Map#on`.\n *\n * @event\n * @param type - The event type previously used to install the listener.\n * @param layer - The layer ID or listener previously used to install the listener.\n * @param listener - The function previously installed as a listener.\n * @returns `this`\n */\n off(\n type: T,\n layer: string,\n listener: (ev: MapLayerEventType[T] & Object) => void,\n ): this;\n /**\n * Overload of the `off` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The function previously installed as a listener.\n * @returns `this`\n */\n off(type: T, listener: (ev: MapEventType[T] & Object) => void): this;\n /**\n * Overload of the `off` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The function previously installed as a listener.\n * @returns `this`\n */\n off(type: keyof MapEventType | string, listener: Listener): this;\n off(type: keyof MapEventType | string, layerIdOrListener: string | Listener, listener?: Listener): this {\n if (listener === undefined) {\n return super.off(type, layerIdOrListener as Listener);\n }\n\n const removeDelegatedListener = (delegatedListeners) => {\n const listeners = delegatedListeners[type];\n for (let i = 0; i < listeners.length; i++) {\n const delegatedListener = listeners[i];\n if (delegatedListener.layer === layerIdOrListener && delegatedListener.listener === listener) {\n for (const event in delegatedListener.delegates) {\n this.off(((event as any)), delegatedListener.delegates[event]);\n }\n listeners.splice(i, 1);\n return this;\n }\n }\n };\n\n if (this._delegatedListeners && this._delegatedListeners[type]) {\n removeDelegatedListener(this._delegatedListeners);\n }\n\n return this;\n }\n\n /**\n * Returns an array of MapGeoJSONFeature objects\n * representing visible features that satisfy the query parameters.\n *\n * @param geometryOrOptions - (optional) The geometry of the query region:\n * either a single point or southwest and northeast points describing a bounding box.\n * Omitting this parameter (i.e. calling {@link Map#queryRenderedFeatures} with zero arguments,\n * or with only a `options` argument) is equivalent to passing a bounding box encompassing the entire\n * map viewport.\n * The geometryOrOptions can receive a {@link QueryRenderedFeaturesOptions} only to support a situation where the function receives only one parameter which is the options parameter.\n * @param options - (optional) Options object.\n *\n * @returns An array of MapGeoJSONFeature objects.\n *\n * The `properties` value of each returned feature object contains the properties of its source feature. For GeoJSON sources, only\n * string and numeric property values are supported (i.e. `null`, `Array`, and `Object` values are not supported).\n *\n * Each feature includes top-level `layer`, `source`, and `sourceLayer` properties. The `layer` property is an object\n * representing the style layer to which the feature belongs. Layout and paint properties in this object contain values\n * which are fully evaluated for the given zoom level and feature.\n *\n * Only features that are currently rendered are included. Some features will **not** be included, like:\n *\n * - Features from layers whose `visibility` property is `\"none\"`.\n * - Features from layers whose zoom range excludes the current zoom level.\n * - Symbol features that have been hidden due to text or icon collision.\n *\n * Features from all other layers are included, including features that may have no visible\n * contribution to the rendered result; for example, because the layer's opacity or color alpha component is set to\n * 0.\n *\n * The topmost rendered feature appears first in the returned array, and subsequent features are sorted by\n * descending z-order. Features that are rendered multiple times (due to wrapping across the antemeridian at low\n * zoom levels) are returned only once (though subject to the following caveat).\n *\n * Because features come from tiled vector data or GeoJSON data that is converted to tiles internally, feature\n * geometries may be split or duplicated across tile boundaries and, as a result, features may appear multiple\n * times in query results. For example, suppose there is a highway running through the bounding rectangle of a query.\n * The results of the query will be those parts of the highway that lie within the map tiles covering the bounding\n * rectangle, even if the highway extends into other tiles, and the portion of the highway within each map tile\n * will be returned as a separate feature. Similarly, a point feature near a tile boundary may appear in multiple\n * tiles due to tile buffering.\n *\n * @example\n * Find all features at a point\n * ```ts\n * let features = map.queryRenderedFeatures(\n * [20, 35],\n * { layers: ['my-layer-name'] }\n * );\n * ```\n *\n * @example\n * Find all features within a static bounding box\n * ```ts\n * let features = map.queryRenderedFeatures(\n * [[10, 20], [30, 50]],\n * { layers: ['my-layer-name'] }\n * );\n * ```\n *\n * @example\n * Find all features within a bounding box around a point\n * ```ts\n * let width = 10;\n * let height = 20;\n * let features = map.queryRenderedFeatures([\n * [point.x - width / 2, point.y - height / 2],\n * [point.x + width / 2, point.y + height / 2]\n * ], { layers: ['my-layer-name'] });\n * ```\n *\n * @example\n * Query all rendered features from a single layer\n * ```ts\n * let features = map.queryRenderedFeatures({ layers: ['my-layer-name'] });\n * ```\n * @see [Get features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/queryrenderedfeatures/)\n */\n queryRenderedFeatures(geometryOrOptions?: PointLike | [PointLike, PointLike] | QueryRenderedFeaturesOptions, options?: QueryRenderedFeaturesOptions): MapGeoJSONFeature[] {\n if (!this.style) {\n return [];\n }\n let queryGeometry;\n const isGeometry = geometryOrOptions instanceof Point || Array.isArray(geometryOrOptions);\n const geometry = isGeometry ? geometryOrOptions : [[0, 0], [this.transform.width, this.transform.height]];\n options = options || (isGeometry ? {} : geometryOrOptions) || {};\n\n if (geometry instanceof Point || typeof geometry[0] === 'number') {\n queryGeometry = [Point.convert(geometry as PointLike)];\n } else {\n const tl = Point.convert(geometry[0] as PointLike);\n const br = Point.convert(geometry[1] as PointLike);\n queryGeometry = [tl, new Point(br.x, tl.y), br, new Point(tl.x, br.y), tl];\n }\n\n return this.style.queryRenderedFeatures(queryGeometry, options, this.transform);\n }\n\n /**\n * Returns an array of MapGeoJSONFeature objects\n * representing features within the specified vector tile or GeoJSON source that satisfy the query parameters.\n *\n * @param sourceId - The ID of the vector tile or GeoJSON source to query.\n * @param parameters - The options object.\n * @returns An array of MapGeoJSONFeature objects.\n *\n * In contrast to {@link Map#queryRenderedFeatures}, this function returns all features matching the query parameters,\n * whether or not they are rendered by the current style (i.e. visible). The domain of the query includes all currently-loaded\n * vector tiles and GeoJSON source tiles: this function does not check tiles outside the currently\n * visible viewport.\n *\n * Because features come from tiled vector data or GeoJSON data that is converted to tiles internally, feature\n * geometries may be split or duplicated across tile boundaries and, as a result, features may appear multiple\n * times in query results. For example, suppose there is a highway running through the bounding rectangle of a query.\n * The results of the query will be those parts of the highway that lie within the map tiles covering the bounding\n * rectangle, even if the highway extends into other tiles, and the portion of the highway within each map tile\n * will be returned as a separate feature. Similarly, a point feature near a tile boundary may appear in multiple\n * tiles due to tile buffering.\n *\n * @example\n * Find all features in one source layer in a vector source\n * ```ts\n * let features = map.querySourceFeatures('your-source-id', {\n * sourceLayer: 'your-source-layer'\n * });\n * ```\n *\n */\n querySourceFeatures(sourceId: string, parameters?: QuerySourceFeatureOptions | null): MapGeoJSONFeature[] {\n return this.style.querySourceFeatures(sourceId, parameters);\n }\n\n /**\n * Updates the map's MapLibre style object with a new value.\n *\n * If a style is already set when this is used and options.diff is set to true, the map renderer will attempt to compare the given style\n * against the map's current state and perform only the changes necessary to make the map style match the desired state. Changes in sprites\n * (images used for icons and patterns) and glyphs (fonts for label text) **cannot** be diffed. If the sprites or fonts used in the current\n * style and the given style are different in any way, the map renderer will force a full update, removing the current style and building\n * the given one from scratch.\n *\n *\n * @param style - A JSON object conforming to the schema described in the\n * [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/), or a URL to such JSON.\n * @param options - The options object.\n * @returns `this`\n *\n * @example\n * ```ts\n * map.setStyle(\"https://demotiles.maplibre.org/style.json\");\n *\n * map.setStyle('https://demotiles.maplibre.org/style.json', {\n * transformStyle: (previousStyle, nextStyle) => ({\n * ...nextStyle,\n * sources: {\n * ...nextStyle.sources,\n * // copy a source from previous style\n * 'osm': previousStyle.sources.osm\n * },\n * layers: [\n * // background layer\n * nextStyle.layers[0],\n * // copy a layer from previous style\n * previousStyle.layers[0],\n * // other layers from the next style\n * ...nextStyle.layers.slice(1).map(layer => {\n * // hide the layers we don't need from demotiles style\n * if (layer.id.startsWith('geolines')) {\n * layer.layout = {...layer.layout || {}, visibility: 'none'};\n * // filter out US polygons\n * } else if (layer.id.startsWith('coastline') || layer.id.startsWith('countries')) {\n * layer.filter = ['!=', ['get', 'ADM0_A3'], 'USA'];\n * }\n * return layer;\n * })\n * ]\n * })\n * });\n * ```\n */\n setStyle(style: StyleSpecification | string | null, options?: StyleSwapOptions & StyleOptions): this {\n options = extend({},\n {\n localIdeographFontFamily: this._localIdeographFontFamily,\n validate: this._validateStyle\n }, options);\n\n if ((options.diff !== false && options.localIdeographFontFamily === this._localIdeographFontFamily) && this.style && style) {\n this._diffStyle(style, options);\n return this;\n } else {\n this._localIdeographFontFamily = options.localIdeographFontFamily;\n return this._updateStyle(style, options);\n }\n }\n\n /**\n * Updates the requestManager's transform request with a new function\n *\n * @param transformRequest - A callback run before the Map makes a request for an external URL. The callback can be used to modify the url, set headers, or set the credentials property for cross-origin requests.\n * Expected to return an object with a `url` property and optionally `headers` and `credentials` properties\n *\n * @returns `this`\n *\n * @example\n * ```ts\n * map.setTransformRequest((url: string, resourceType: string) => {});\n * ```\n */\n setTransformRequest(transformRequest: RequestTransformFunction): this {\n this._requestManager.setTransformRequest(transformRequest);\n return this;\n }\n\n _getUIString(key: string) {\n const str = this._locale[key];\n if (str == null) {\n throw new Error(`Missing UI string '${key}'`);\n }\n\n return str;\n }\n\n _updateStyle(style: StyleSpecification | string | null, options?: StyleSwapOptions & StyleOptions) {\n // transformStyle relies on having previous style serialized, if it is not loaded yet, delay _updateStyle until previous style is loaded\n if (options.transformStyle && this.style && !this.style._loaded) {\n this.style.once('style.load', () => this._updateStyle(style, options));\n return;\n }\n\n const previousStyle = this.style && options.transformStyle ? this.style.serialize() : undefined;\n if (this.style) {\n this.style.setEventedParent(null);\n\n // Only release workers when map is getting disposed\n this.style._remove(!style);\n }\n\n if (!style) {\n delete this.style;\n return this;\n } else {\n this.style = new Style(this, options || {});\n }\n\n this.style.setEventedParent(this, {style: this.style});\n\n if (typeof style === 'string') {\n this.style.loadURL(style, options, previousStyle);\n } else {\n this.style.loadJSON(style, options, previousStyle);\n }\n\n return this;\n }\n\n _lazyInitEmptyStyle() {\n if (!this.style) {\n this.style = new Style(this, {});\n this.style.setEventedParent(this, {style: this.style});\n this.style.loadEmpty();\n }\n }\n\n _diffStyle(style: StyleSpecification | string, options?: StyleSwapOptions & StyleOptions) {\n if (typeof style === 'string') {\n const url = style;\n const request = this._requestManager.transformRequest(url, ResourceType.Style);\n getJSON(request, (error?: Error | null, json?: any | null) => {\n if (error) {\n this.fire(new ErrorEvent(error));\n } else if (json) {\n this._updateDiff(json, options);\n }\n });\n } else if (typeof style === 'object') {\n this._updateDiff(style, options);\n }\n }\n\n _updateDiff(style: StyleSpecification, options?: StyleSwapOptions & StyleOptions) {\n try {\n if (this.style.setState(style, options)) {\n this._update(true);\n }\n } catch (e) {\n warnOnce(\n `Unable to perform style diff: ${e.message || e.error || e}. Rebuilding the style from scratch.`\n );\n this._updateStyle(style, options);\n }\n }\n\n /**\n * Returns the map's MapLibre style object, a JSON object which can be used to recreate the map's style.\n *\n * @returns The map's style JSON object.\n *\n * @example\n * ```ts\n * let styleJson = map.getStyle();\n * ```\n *\n */\n getStyle(): StyleSpecification {\n if (this.style) {\n return this.style.serialize();\n }\n }\n\n /**\n * Returns a Boolean indicating whether the map's style is fully loaded.\n *\n * @returns A Boolean indicating whether the style is fully loaded.\n *\n * @example\n * ```ts\n * let styleLoadStatus = map.isStyleLoaded();\n * ```\n */\n isStyleLoaded(): boolean | void {\n if (!this.style) return warnOnce('There is no style added to the map.');\n return this.style.loaded();\n }\n\n /**\n * Adds a source to the map's style.\n *\n * Events triggered:\n *\n * Triggers the `source.add` event.\n *\n * @param id - The ID of the source to add. Must not conflict with existing sources.\n * @param source - The source object, conforming to the\n * MapLibre Style Specification's [source definition](https://maplibre.org/maplibre-style-spec/sources) or\n * {@link CanvasSourceSpecification}.\n * @returns `this`\n * @example\n * ```ts\n * map.addSource('my-data', {\n * type: 'vector',\n * url: 'https://demotiles.maplibre.org/tiles/tiles.json'\n * });\n * ```\n * @example\n * ```ts\n * map.addSource('my-data', {\n * \"type\": \"geojson\",\n * \"data\": {\n * \"type\": \"Feature\",\n * \"geometry\": {\n * \"type\": \"Point\",\n * \"coordinates\": [-77.0323, 38.9131]\n * },\n * \"properties\": {\n * \"title\": \"Mapbox DC\",\n * \"marker-symbol\": \"monument\"\n * }\n * }\n * });\n * ```\n * @see GeoJSON source: [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/)\n */\n addSource(id: string, source: SourceSpecification): this {\n this._lazyInitEmptyStyle();\n this.style.addSource(id, source);\n return this._update(true);\n }\n\n /**\n * Returns a Boolean indicating whether the source is loaded. Returns `true` if the source with\n * the given ID in the map's style has no outstanding network requests, otherwise `false`.\n *\n * A {@link ErrorEvent} event will be fired if there is no source wit the specified ID.\n *\n * @param id - The ID of the source to be checked.\n * @returns A Boolean indicating whether the source is loaded.\n * @example\n * ```ts\n * let sourceLoaded = map.isSourceLoaded('bathymetry-data');\n * ```\n */\n isSourceLoaded(id: string): boolean {\n const source = this.style && this.style.sourceCaches[id];\n if (source === undefined) {\n this.fire(new ErrorEvent(new Error(`There is no source with ID '${id}'`)));\n return;\n }\n return source.loaded();\n }\n\n /**\n * Loads a 3D terrain mesh, based on a \"raster-dem\" source.\n *\n * Triggers the `terrain` event.\n *\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setTerrain({ source: 'terrain' });\n * ```\n */\n setTerrain(options: TerrainSpecification | null): this {\n this.style._checkLoaded();\n\n // clear event handlers\n if (this._terrainDataCallback) this.style.off('data', this._terrainDataCallback);\n\n if (!options) {\n // remove terrain\n if (this.terrain) this.terrain.sourceCache.destruct();\n this.terrain = null;\n if (this.painter.renderToTexture) this.painter.renderToTexture.destruct();\n this.painter.renderToTexture = null;\n this.transform._minEleveationForCurrentTile = 0;\n this.transform.elevation = 0;\n } else {\n // add terrain\n const sourceCache = this.style.sourceCaches[options.source];\n if (!sourceCache) throw new Error(`cannot load terrain, because there exists no source with ID: ${options.source}`);\n // Warn once if user is using the same source for hillshade and terrain\n for (const index in this.style._layers) {\n const thisLayer = this.style._layers[index];\n if (thisLayer.type === 'hillshade' && thisLayer.source === options.source) {\n warnOnce('You are using the same source for a hillshade layer and for 3D terrain. Please consider using two separate sources to improve rendering quality.');\n }\n }\n this.terrain = new Terrain(this.painter, sourceCache, options);\n this.painter.renderToTexture = new RenderToTexture(this.painter, this.terrain);\n this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n this._terrainDataCallback = e => {\n if (e.dataType === 'style') {\n this.terrain.sourceCache.freeRtt();\n } else if (e.dataType === 'source' && e.tile) {\n if (e.sourceId === options.source && !this._elevationFreeze) {\n this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n }\n this.terrain.sourceCache.freeRtt(e.tile.tileID);\n }\n };\n this.style.on('data', this._terrainDataCallback);\n }\n\n this.fire(new Event('terrain', {terrain: options}));\n return this;\n }\n\n /**\n * Get the terrain-options if terrain is loaded\n * @returns the TerrainSpecification passed to setTerrain\n * @example\n * ```ts\n * map.getTerrain(); // { source: 'terrain' };\n * ```\n */\n getTerrain(): TerrainSpecification | null {\n return this.terrain?.options ?? null;\n }\n\n /**\n * Returns a Boolean indicating whether all tiles in the viewport from all sources on\n * the style are loaded.\n *\n * @returns A Boolean indicating whether all tiles are loaded.\n * @example\n * ```ts\n * let tilesLoaded = map.areTilesLoaded();\n * ```\n */\n areTilesLoaded(): boolean {\n const sources = this.style && this.style.sourceCaches;\n for (const id in sources) {\n const source = sources[id];\n const tiles = source._tiles;\n for (const t in tiles) {\n const tile = tiles[t];\n if (!(tile.state === 'loaded' || tile.state === 'errored')) return false;\n }\n }\n return true;\n }\n\n /**\n * Adds a [custom source type](#Custom Sources), making it available for use with\n * {@link Map#addSource}.\n * @param name - The name of the source type; source definition objects use this name in the `{type: ...}` field.\n * @param SourceType - A {@link Source} constructor.\n * @param callback - Called when the source type is ready or with an error argument if there is an error.\n */\n addSourceType(name: string, SourceType: SourceClass, callback: Callback) {\n this._lazyInitEmptyStyle();\n return this.style.addSourceType(name, SourceType, callback);\n }\n\n /**\n * Removes a source from the map's style.\n *\n * @param id - The ID of the source to remove.\n * @returns `this`\n * @example\n * ```ts\n * map.removeSource('bathymetry-data');\n * ```\n */\n removeSource(id: string): Map {\n this.style.removeSource(id);\n return this._update(true);\n }\n\n /**\n * Returns the source with the specified ID in the map's style.\n *\n * This method is often used to update a source using the instance members for the relevant\n * source type as defined in [Sources](#sources).\n * For example, setting the `data` for a GeoJSON source or updating the `url` and `coordinates`\n * of an image source.\n *\n * @param id - The ID of the source to get.\n * @returns The style source with the specified ID or `undefined` if the ID\n * corresponds to no existing sources.\n * The shape of the object varies by source type.\n * A list of options for each source type is available on the MapLibre Style Specification's\n * [Sources](https://maplibre.org/maplibre-style-spec/sources/) page.\n * @example\n * ```ts\n * let sourceObject = map.getSource('points');\n * ```\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n * @see [Animate a point](https://maplibre.org/maplibre-gl-js/docs/examples/animate-point-along-line/)\n * @see [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/)\n */\n getSource(id: string): Source | undefined {\n return this.style.getSource(id);\n }\n\n /**\n * Add an image to the style. This image can be displayed on the map like any other icon in the style's\n * sprite using the image's ID with\n * [`icon-image`](https://maplibre.org/maplibre-style-spec/layers/#layout-symbol-icon-image),\n * [`background-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-background-background-pattern),\n * [`fill-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-fill-fill-pattern),\n * or [`line-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-line-line-pattern).\n *\n * A {@link ErrorEvent} event will be fired if the image parameter is invalid or there is not enough space in the sprite to add this image.\n *\n * @param id - The ID of the image.\n * @param image - The image as an `HTMLImageElement`, `ImageData`, `ImageBitmap` or object with `width`, `height`, and `data`\n * properties with the same format as `ImageData`.\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * // If the style's sprite does not already contain an image with ID 'cat',\n * // add the image 'cat-icon.png' to the style's sprite with the ID 'cat'.\n * map.loadImage('https://upload.wikimedia.org/wikipedia/commons/thumb/6/60/Cat_silhouette.svg/400px-Cat_silhouette.svg.png', function(error, image) {\n * if (error) throw error;\n * if (!map.hasImage('cat')) map.addImage('cat', image);\n * });\n *\n * // Add a stretchable image that can be used with `icon-text-fit`\n * // In this example, the image is 600px wide by 400px high.\n * map.loadImage('https://upload.wikimedia.org/wikipedia/commons/8/89/Black_and_White_Boxed_%28bordered%29.png', function(error, image) {\n * if (error) throw error;\n * if (!map.hasImage('border-image')) {\n * map.addImage('border-image', image, {\n * content: [16, 16, 300, 384], // place text over left half of image, avoiding the 16px border\n * stretchX: [[16, 584]], // stretch everything horizontally except the 16px border\n * stretchY: [[16, 384]], // stretch everything vertically except the 16px border\n * });\n * }\n * });\n * ```\n * @see Use `HTMLImageElement`: [Add an icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image/)\n * @see Use `ImageData`: [Add a generated icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-generated/)\n */\n addImage(id: string,\n image: HTMLImageElement | ImageBitmap | ImageData | {\n width: number;\n height: number;\n data: Uint8Array | Uint8ClampedArray;\n } | StyleImageInterface,\n options: Partial = {}): this {\n const {\n pixelRatio = 1,\n sdf = false,\n stretchX,\n stretchY,\n content\n } = options;\n this._lazyInitEmptyStyle();\n const version = 0;\n\n if (image instanceof HTMLImageElement || isImageBitmap(image)) {\n const {width, height, data} = browser.getImageData(image);\n this.style.addImage(id, {data: new RGBAImage({width, height}, data), pixelRatio, stretchX, stretchY, content, sdf, version});\n } else if (image.width === undefined || image.height === undefined) {\n return this.fire(new ErrorEvent(new Error(\n 'Invalid arguments to map.addImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, ' +\n 'or object with `width`, `height`, and `data` properties with the same format as `ImageData`')));\n } else {\n const {width, height, data} = image as ImageData;\n const userImage = (image as any as StyleImageInterface);\n\n this.style.addImage(id, {\n data: new RGBAImage({width, height}, new Uint8Array(data)),\n pixelRatio,\n stretchX,\n stretchY,\n content,\n sdf,\n version,\n userImage\n });\n\n if (userImage.onAdd) {\n userImage.onAdd(this, id);\n }\n return this;\n }\n }\n\n /**\n * Update an existing image in a style. This image can be displayed on the map like any other icon in the style's\n * sprite using the image's ID with\n * [`icon-image`](https://maplibre.org/maplibre-style-spec/layers/#layout-symbol-icon-image),\n * [`background-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-background-background-pattern),\n * [`fill-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-fill-fill-pattern),\n * or [`line-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-line-line-pattern).\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param id - The ID of the image.\n * @param image - The image as an `HTMLImageElement`, `ImageData`, `ImageBitmap` or object with `width`, `height`, and `data`\n * properties with the same format as `ImageData`.\n * @returns `this`\n * @example\n * ```ts\n * // If an image with the ID 'cat' already exists in the style's sprite,\n * // replace that image with a new image, 'other-cat-icon.png'.\n * if (map.hasImage('cat')) map.updateImage('cat', './other-cat-icon.png');\n * ```\n */\n updateImage(id: string,\n image: HTMLImageElement | ImageBitmap | ImageData | {\n width: number;\n height: number;\n data: Uint8Array | Uint8ClampedArray;\n } | StyleImageInterface): this {\n\n const existingImage = this.style.getImage(id);\n if (!existingImage) {\n return this.fire(new ErrorEvent(new Error(\n 'The map has no image with that id. If you are adding a new image use `map.addImage(...)` instead.')));\n }\n const imageData = (image instanceof HTMLImageElement || isImageBitmap(image)) ?\n browser.getImageData(image) :\n image;\n const {width, height, data} = imageData;\n\n if (width === undefined || height === undefined) {\n return this.fire(new ErrorEvent(new Error(\n 'Invalid arguments to map.updateImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, ' +\n 'or object with `width`, `height`, and `data` properties with the same format as `ImageData`')));\n }\n\n if (width !== existingImage.data.width || height !== existingImage.data.height) {\n return this.fire(new ErrorEvent(new Error(\n 'The width and height of the updated image must be that same as the previous version of the image')));\n }\n\n const copy = !(image instanceof HTMLImageElement || isImageBitmap(image));\n existingImage.data.replace(data, copy);\n\n this.style.updateImage(id, existingImage);\n return this;\n }\n\n /**\n * Returns an image, specified by ID, currently available in the map.\n * This includes both images from the style's original sprite\n * and any images that have been added at runtime using {@link Map#addImage}.\n *\n * @param id - The ID of the image.\n * @returns An image in the map with the specified ID.\n *\n * @example\n * ```ts\n * let coffeeShopIcon = map.getImage(\"coffee_cup\");\n * ```\n */\n getImage(id: string): StyleImage {\n return this.style.getImage(id);\n }\n\n /**\n * Check whether or not an image with a specific ID exists in the style. This checks both images\n * in the style's original sprite and any images\n * that have been added at runtime using {@link Map#addImage}.\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param id - The ID of the image.\n *\n * @returns A Boolean indicating whether the image exists.\n * @example\n * Check if an image with the ID 'cat' exists in the style's sprite.\n * ```ts\n * let catIconExists = map.hasImage('cat');\n * ```\n */\n hasImage(id: string): boolean {\n if (!id) {\n this.fire(new ErrorEvent(new Error('Missing required image id')));\n return false;\n }\n\n return !!this.style.getImage(id);\n }\n\n /**\n * Remove an image from a style. This can be an image from the style's original\n * sprite or any images\n * that have been added at runtime using {@link Map#addImage}.\n *\n * @param id - The ID of the image.\n *\n * @example\n * ```ts\n * // If an image with the ID 'cat' exists in\n * // the style's sprite, remove it.\n * if (map.hasImage('cat')) map.removeImage('cat');\n * ```\n */\n removeImage(id: string) {\n this.style.removeImage(id);\n }\n\n /**\n * Load an image from an external URL to be used with {@link Map#addImage}. External\n * domains must support [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/Access_control_CORS).\n *\n * @param url - The URL of the image file. Image file must be in png, webp, or jpg format.\n * @param callback - Expecting `callback(error, data)`. Called when the image has loaded or with an error argument if there is an error.\n *\n * @example\n * Load an image from an external URL.\n * ```ts\n * map.loadImage('http://placekitten.com/50/50', function(error, image) {\n * if (error) throw error;\n * // Add the loaded image to the style's sprite with the ID 'kitten'.\n * map.addImage('kitten', image);\n * });\n * ```\n * @see [Add an icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image/)\n */\n loadImage(url: string, callback: GetImageCallback) {\n ImageRequest.getImage(this._requestManager.transformRequest(url, ResourceType.Image), callback);\n }\n\n /**\n * Returns an Array of strings containing the IDs of all images currently available in the map.\n * This includes both images from the style's original sprite\n * and any images that have been added at runtime using {@link Map#addImage}.\n *\n * @returns An Array of strings containing the names of all sprites/images currently available in the map.\n *\n * @example\n * ```ts\n * let allImages = map.listImages();\n * ```\n */\n listImages(): Array {\n return this.style.listImages();\n }\n\n /**\n * Adds a [MapLibre style layer](https://maplibre.org/maplibre-style-spec/layers)\n * to the map's style.\n *\n * A layer defines how data from a specified source will be styled. Read more about layer types\n * and available paint and layout properties in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/layers).\n *\n * @param layer - The layer to add,\n * conforming to either the MapLibre Style Specification's [layer definition](https://maplibre.org/maplibre-style-spec/layers) or,\n * less commonly, the {@link CustomLayerInterface} specification. Can also be a layer definition with an embedded source definition.\n * The MapLibre Style Specification's layer definition is appropriate for most layers.\n *\n * @param beforeId - The ID of an existing layer to insert the new layer before,\n * resulting in the new layer appearing visually beneath the existing layer.\n * If this argument is not specified, the layer will be appended to the end of the layers array\n * and appear visually above all other layers.\n *\n * @returns `this`\n *\n * @example\n * Add a circle layer with a vector source\n * ```ts\n * map.addLayer({\n * id: 'points-of-interest',\n * source: {\n * type: 'vector',\n * url: 'https://demotiles.maplibre.org/tiles/tiles.json'\n * },\n * 'source-layer': 'poi_label',\n * type: 'circle',\n * paint: {\n * // MapLibre Style Specification paint properties\n * },\n * layout: {\n * // MapLibre Style Specification layout properties\n * }\n * });\n * ```\n *\n * @example\n * Define a source before using it to create a new layer\n * ```ts\n * map.addSource('state-data', {\n * type: 'geojson',\n * data: 'path/to/data.geojson'\n * });\n *\n * map.addLayer({\n * id: 'states',\n * // References the GeoJSON source defined above\n * // and does not require a `source-layer`\n * source: 'state-data',\n * type: 'symbol',\n * layout: {\n * // Set the label content to the\n * // feature's `name` property\n * text-field: ['get', 'name']\n * }\n * });\n * ```\n *\n * @example\n * Add a new symbol layer before an existing layer\n * ```ts\n * map.addLayer({\n * id: 'states',\n * // References a source that's already been defined\n * source: 'state-data',\n * type: 'symbol',\n * layout: {\n * // Set the label content to the\n * // feature's `name` property\n * text-field: ['get', 'name']\n * }\n * // Add the layer before the existing `cities` layer\n * }, 'cities');\n * ```\n * @see [Create and style clusters](https://maplibre.org/maplibre-gl-js/docs/examples/cluster/)\n * @see [Add a vector tile source](https://maplibre.org/maplibre-gl-js/docs/examples/vector-source/)\n * @see [Add a WMS source](https://maplibre.org/maplibre-gl-js/docs/examples/wms/)\n */\n addLayer(layer: AddLayerObject, beforeId?: string) {\n this._lazyInitEmptyStyle();\n this.style.addLayer(layer, beforeId);\n return this._update(true);\n }\n\n /**\n * Moves a layer to a different z-position.\n *\n * @param id - The ID of the layer to move.\n * @param beforeId - The ID of an existing layer to insert the new layer before. When viewing the map, the `id` layer will appear beneath the `beforeId` layer. If `beforeId` is omitted, the layer will be appended to the end of the layers array and appear above all other layers on the map.\n * @returns `this`\n *\n * @example\n * Move a layer with ID 'polygon' before the layer with ID 'country-label'. The `polygon` layer will appear beneath the `country-label` layer on the map.\n * ```ts\n * map.moveLayer('polygon', 'country-label');\n * ```\n */\n moveLayer(id: string, beforeId?: string): this {\n this.style.moveLayer(id, beforeId);\n return this._update(true);\n }\n\n /**\n * Removes the layer with the given ID from the map's style.\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param id - The ID of the layer to remove\n * @returns `this`\n *\n * @example\n * If a layer with ID 'state-data' exists, remove it.\n * ```ts\n * if (map.getLayer('state-data')) map.removeLayer('state-data');\n * ```\n */\n removeLayer(id: string): this {\n this.style.removeLayer(id);\n return this._update(true);\n }\n\n /**\n * Returns the layer with the specified ID in the map's style.\n *\n * @param id - The ID of the layer to get.\n * @returns The layer with the specified ID, or `undefined`\n * if the ID corresponds to no existing layers.\n *\n * @example\n * ```ts\n * let stateDataLayer = map.getLayer('state-data');\n * ```\n * @see [Filter symbols by toggling a list](https://maplibre.org/maplibre-gl-js/docs/examples/filter-markers/)\n * @see [Filter symbols by text input](https://maplibre.org/maplibre-gl-js/docs/examples/filter-markers-by-input/)\n */\n getLayer(id: string): StyleLayer | undefined {\n return this.style.getLayer(id);\n }\n\n /**\n * Return the ids of all layers currently in the style, including custom layers, in order.\n *\n * @returns ids of layers, in order\n *\n * @example\n * ```ts\n * const orderedLayerIds = map.getLayersOrder();\n * ```\n */\n getLayersOrder(): string[] {\n return this.style.getLayersOrder();\n }\n\n /**\n * Sets the zoom extent for the specified style layer. The zoom extent includes the\n * [minimum zoom level](https://maplibre.org/maplibre-style-spec/layers/#minzoom)\n * and [maximum zoom level](https://maplibre.org/maplibre-style-spec/layers/#maxzoom))\n * at which the layer will be rendered.\n *\n * Note: For style layers using vector sources, style layers cannot be rendered at zoom levels lower than the\n * minimum zoom level of the _source layer_ because the data does not exist at those zoom levels. If the minimum\n * zoom level of the source layer is higher than the minimum zoom level defined in the style layer, the style\n * layer will not be rendered at all zoom levels in the zoom range.\n *\n * @param layerId - The ID of the layer to which the zoom extent will be applied.\n * @param minzoom - The minimum zoom to set (0-24).\n * @param maxzoom - The maximum zoom to set (0-24).\n * @returns `this`\n *\n * @example\n * ```ts\n * map.setLayerZoomRange('my-layer', 2, 5);\n * ```\n */\n setLayerZoomRange(layerId: string, minzoom: number, maxzoom: number): this {\n this.style.setLayerZoomRange(layerId, minzoom, maxzoom);\n return this._update(true);\n }\n\n /**\n * Sets the filter for the specified style layer.\n *\n * Filters control which features a style layer renders from its source.\n * Any feature for which the filter expression evaluates to `true` will be\n * rendered on the map. Those that are false will be hidden.\n *\n * Use `setFilter` to show a subset of your source data.\n *\n * To clear the filter, pass `null` or `undefined` as the second parameter.\n *\n * @param layerId - The ID of the layer to which the filter will be applied.\n * @param filter - The filter, conforming to the MapLibre Style Specification's\n * [filter definition](https://maplibre.org/maplibre-style-spec/layers/#filter). If `null` or `undefined` is provided, the function removes any existing filter from the layer.\n * @param options - Options object.\n * @returns `this`\n *\n * @example\n * Display only features with the 'name' property 'USA'\n * ```ts\n * map.setFilter('my-layer', ['==', ['get', 'name'], 'USA']);\n * ```\n * @example\n * Display only features with five or more 'available-spots'\n * ```ts\n * map.setFilter('bike-docks', ['>=', ['get', 'available-spots'], 5]);\n * ```\n * @example\n * Remove the filter for the 'bike-docks' style layer\n * ```ts\n * map.setFilter('bike-docks', null);\n * ```\n * @see [Create a timeline animation](https://maplibre.org/maplibre-gl-js/docs/examples/timeline-animation/)\n */\n setFilter(layerId: string, filter?: FilterSpecification | null, options: StyleSetterOptions = {}) {\n this.style.setFilter(layerId, filter, options);\n return this._update(true);\n }\n\n /**\n * Returns the filter applied to the specified style layer.\n *\n * @param layerId - The ID of the style layer whose filter to get.\n * @returns The layer's filter.\n */\n getFilter(layerId: string): FilterSpecification | void {\n return this.style.getFilter(layerId);\n }\n\n /**\n * Sets the value of a paint property in the specified style layer.\n *\n * @param layerId - The ID of the layer to set the paint property in.\n * @param name - The name of the paint property to set.\n * @param value - The value of the paint property to set.\n * Must be of a type appropriate for the property, as defined in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/).\n * Pass `null` to unset the existing value.\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setPaintProperty('my-layer', 'fill-color', '#faafee');\n * ```\n * @see [Change a layer's color with buttons](https://maplibre.org/maplibre-gl-js/docs/examples/color-switcher/)\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n setPaintProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}): this {\n this.style.setPaintProperty(layerId, name, value, options);\n return this._update(true);\n }\n\n /**\n * Returns the value of a paint property in the specified style layer.\n *\n * @param layerId - The ID of the layer to get the paint property from.\n * @param name - The name of a paint property to get.\n * @returns The value of the specified paint property.\n */\n getPaintProperty(layerId: string, name: string) {\n return this.style.getPaintProperty(layerId, name);\n }\n\n /**\n * Sets the value of a layout property in the specified style layer.\n *\n * @param layerId - The ID of the layer to set the layout property in.\n * @param name - The name of the layout property to set.\n * @param value - The value of the layout property. Must be of a type appropriate for the property, as defined in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/).\n * @param options - The options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setLayoutProperty('my-layer', 'visibility', 'none');\n * ```\n */\n setLayoutProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}): this {\n this.style.setLayoutProperty(layerId, name, value, options);\n return this._update(true);\n }\n\n /**\n * Returns the value of a layout property in the specified style layer.\n *\n * @param layerId - The ID of the layer to get the layout property from.\n * @param name - The name of the layout property to get.\n * @returns The value of the specified layout property.\n */\n getLayoutProperty(layerId: string, name: string) {\n return this.style.getLayoutProperty(layerId, name);\n }\n\n /**\n * Sets the value of the style's glyphs property.\n *\n * @param glyphsUrl - Glyph URL to set. Must conform to the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/glyphs/).\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setGlyphs('https://demotiles.maplibre.org/font/{fontstack}/{range}.pbf');\n * ```\n */\n setGlyphs(glyphsUrl: string | null, options: StyleSetterOptions = {}): this {\n this._lazyInitEmptyStyle();\n this.style.setGlyphs(glyphsUrl, options);\n return this._update(true);\n }\n\n /**\n * Returns the value of the style's glyphs URL\n *\n * @returns glyphs Style's glyphs url\n */\n getGlyphs(): string | null {\n return this.style.getGlyphsUrl();\n }\n\n /**\n * Adds a sprite to the map's style. Fires the `style` event.\n *\n * @param id - The ID of the sprite to add. Must not conflict with existing sprites.\n * @param url - The URL to load the sprite from\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.addSprite('sprite-two', 'http://example.com/sprite-two');\n * ```\n */\n addSprite(id: string, url: string, options: StyleSetterOptions = {}): this {\n this._lazyInitEmptyStyle();\n this.style.addSprite(id, url, options, (err) => {\n if (!err) {\n this._update(true);\n }\n });\n return this;\n }\n\n /**\n * Removes the sprite from the map's style. Fires the `style` event.\n *\n * @param id - The ID of the sprite to remove. If the sprite is declared as a single URL, the ID must be \"default\".\n * @returns `this`\n * @example\n * ```ts\n * map.removeSprite('sprite-two');\n * map.removeSprite('default');\n * ```\n */\n removeSprite(id: string) {\n this._lazyInitEmptyStyle();\n this.style.removeSprite(id);\n return this._update(true);\n }\n\n /**\n * Returns the as-is value of the style's sprite.\n *\n * @returns style's sprite list of id-url pairs\n */\n getSprite(): {id: string; url: string}[] {\n return this.style.getSprite();\n }\n\n /**\n * Sets the value of the style's sprite property.\n *\n * @param spriteUrl - Sprite URL to set.\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setSprite('YOUR_SPRITE_URL');\n * ```\n */\n setSprite(spriteUrl: string | null, options: StyleSetterOptions = {}) {\n this._lazyInitEmptyStyle();\n this.style.setSprite(spriteUrl, options, (err) => {\n if (!err) {\n this._update(true);\n }\n });\n return this;\n }\n\n /**\n * Sets the any combination of light values.\n *\n * @param light - Light properties to set. Must conform to the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/light).\n * @param options - Options object.\n * @returns `this`\n *\n * @example\n * ```ts\n * let layerVisibility = map.getLayoutProperty('my-layer', 'visibility');\n * ```\n */\n setLight(light: LightSpecification, options: StyleSetterOptions = {}) {\n this._lazyInitEmptyStyle();\n this.style.setLight(light, options);\n return this._update(true);\n }\n\n /**\n * Returns the value of the light object.\n *\n * @returns light Light properties of the style.\n */\n getLight(): LightSpecification {\n return this.style.getLight();\n }\n\n /**\n * Sets the `state` of a feature.\n * A feature's `state` is a set of user-defined key-value pairs that are assigned to a feature at runtime.\n * When using this method, the `state` object is merged with any existing key-value pairs in the feature's state.\n * Features are identified by their `feature.id` attribute, which can be any number or string.\n *\n * This method can only be used with sources that have a `feature.id` attribute. The `feature.id` attribute can be defined in three ways:\n * - For vector or GeoJSON sources, including an `id` attribute in the original data file.\n * - For vector or GeoJSON sources, using the [`promoteId`](https://maplibre.org/maplibre-style-spec/sources/#vector-promoteId) option at the time the source is defined.\n * - For GeoJSON sources, using the [`generateId`](https://maplibre.org/maplibre-style-spec/sources/#geojson-generateId) option to auto-assign an `id` based on the feature's index in the source data. If you change feature data using `map.getSource('some id').setData(..)`, you may need to re-apply state taking into account updated `id` values.\n *\n * _Note: You can use the [`feature-state` expression](https://maplibre.org/maplibre-style-spec/expressions/#feature-state) to access the values in a feature's state object for the purposes of styling._\n *\n * @param feature - Feature identifier. Feature objects returned from\n * {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers.\n * @param state - A set of key-value pairs. The values should be valid JSON types.\n * @returns `this`\n *\n * @example\n * ```ts\n * // When the mouse moves over the `my-layer` layer, update\n * // the feature state for the feature under the mouse\n * map.on('mousemove', 'my-layer', function(e) {\n * if (e.features.length > 0) {\n * map.setFeatureState({\n * source: 'my-source',\n * sourceLayer: 'my-source-layer',\n * id: e.features[0].id,\n * }, {\n * hover: true\n * });\n * }\n * });\n * ```\n * @see [Create a hover effect](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n */\n setFeatureState(feature: FeatureIdentifier, state: any): this {\n this.style.setFeatureState(feature, state);\n return this._update();\n }\n\n /**\n * Removes the `state` of a feature, setting it back to the default behavior.\n * If only a `target.source` is specified, it will remove the state for all features from that source.\n * If `target.id` is also specified, it will remove all keys for that feature's state.\n * If `key` is also specified, it removes only that key from that feature's state.\n * Features are identified by their `feature.id` attribute, which can be any number or string.\n *\n * @param target - Identifier of where to remove state. It can be a source, a feature, or a specific key of feature.\n * Feature objects returned from {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers.\n * @param key - (optional) The key in the feature state to reset.\n * @returns `this`\n * @example\n * Reset the entire state object for all features in the `my-source` source\n * ```ts\n * map.removeFeatureState({\n * source: 'my-source'\n * });\n * ```\n *\n * @example\n * When the mouse leaves the `my-layer` layer,\n * reset the entire state object for the\n * feature under the mouse\n * ```ts\n * map.on('mouseleave', 'my-layer', function(e) {\n * map.removeFeatureState({\n * source: 'my-source',\n * sourceLayer: 'my-source-layer',\n * id: e.features[0].id\n * });\n * });\n * ```\n *\n * @example\n * When the mouse leaves the `my-layer` layer,\n * reset only the `hover` key-value pair in the\n * state for the feature under the mouse\n * ```ts\n * map.on('mouseleave', 'my-layer', function(e) {\n * map.removeFeatureState({\n * source: 'my-source',\n * sourceLayer: 'my-source-layer',\n * id: e.features[0].id\n * }, 'hover');\n * });\n * ```\n */\n removeFeatureState(target: FeatureIdentifier, key?: string): this {\n this.style.removeFeatureState(target, key);\n return this._update();\n }\n\n /**\n * Gets the `state` of a feature.\n * A feature's `state` is a set of user-defined key-value pairs that are assigned to a feature at runtime.\n * Features are identified by their `feature.id` attribute, which can be any number or string.\n *\n * _Note: To access the values in a feature's state object for the purposes of styling the feature, use the [`feature-state` expression](https://maplibre.org/maplibre-style-spec/expressions/#feature-state)._\n *\n * @param feature - Feature identifier. Feature objects returned from\n * {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers.\n * @returns The state of the feature: a set of key-value pairs that was assigned to the feature at runtime.\n *\n * @example\n * When the mouse moves over the `my-layer` layer,\n * get the feature state for the feature under the mouse\n * ```ts\n * map.on('mousemove', 'my-layer', function(e) {\n * if (e.features.length > 0) {\n * map.getFeatureState({\n * source: 'my-source',\n * sourceLayer: 'my-source-layer',\n * id: e.features[0].id\n * });\n * }\n * });\n * ```\n */\n getFeatureState(feature: FeatureIdentifier): any {\n return this.style.getFeatureState(feature);\n }\n\n /**\n * Returns the map's containing HTML element.\n *\n * @returns The map's container.\n */\n getContainer(): HTMLElement {\n return this._container;\n }\n\n /**\n * Returns the HTML element containing the map's `` element.\n *\n * If you want to add non-GL overlays to the map, you should append them to this element.\n *\n * This is the element to which event bindings for map interactivity (such as panning and zooming) are\n * attached. It will receive bubbled events from child elements such as the ``, but not from\n * map controls.\n *\n * @returns The container of the map's ``.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n getCanvasContainer(): HTMLElement {\n return this._canvasContainer;\n }\n\n /**\n * Returns the map's `` element.\n *\n * @returns The map's `` element.\n * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n */\n getCanvas(): HTMLCanvasElement {\n return this._canvas;\n }\n\n _containerDimensions() {\n let width = 0;\n let height = 0;\n\n if (this._container) {\n width = this._container.clientWidth || 400;\n height = this._container.clientHeight || 300;\n }\n\n return [width, height];\n }\n\n _setupContainer() {\n const container = this._container;\n container.classList.add('maplibregl-map');\n\n const canvasContainer = this._canvasContainer = DOM.create('div', 'maplibregl-canvas-container', container);\n if (this._interactive) {\n canvasContainer.classList.add('maplibregl-interactive');\n }\n\n this._canvas = DOM.create('canvas', 'maplibregl-canvas', canvasContainer);\n this._canvas.addEventListener('webglcontextlost', this._contextLost, false);\n this._canvas.addEventListener('webglcontextrestored', this._contextRestored, false);\n this._canvas.setAttribute('tabindex', '0');\n this._canvas.setAttribute('aria-label', 'Map');\n this._canvas.setAttribute('role', 'region');\n\n const dimensions = this._containerDimensions();\n const clampedPixelRatio = this._getClampedPixelRatio(dimensions[0], dimensions[1]);\n this._resizeCanvas(dimensions[0], dimensions[1], clampedPixelRatio);\n\n const controlContainer = this._controlContainer = DOM.create('div', 'maplibregl-control-container', container);\n const positions = this._controlPositions = {};\n ['top-left', 'top-right', 'bottom-left', 'bottom-right'].forEach((positionName) => {\n positions[positionName] = DOM.create('div', `maplibregl-ctrl-${positionName} `, controlContainer);\n });\n\n this._container.addEventListener('scroll', this._onMapScroll, false);\n }\n\n _cooperativeGesturesOnWheel = (event: WheelEvent) => {\n this._onCooperativeGesture(event, event[this._metaKey], 1);\n };\n\n _setupCooperativeGestures() {\n const container = this._container;\n this._cooperativeGesturesScreen = DOM.create('div', 'maplibregl-cooperative-gesture-screen', container);\n let desktopMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.windowsHelpText ? this._cooperativeGestures.windowsHelpText : 'Use Ctrl + scroll to zoom the map';\n if (navigator.platform.indexOf('Mac') === 0) {\n desktopMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.macHelpText ? this._cooperativeGestures.macHelpText : 'Use ⌘ + scroll to zoom the map';\n }\n const mobileMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.mobileHelpText ? this._cooperativeGestures.mobileHelpText : 'Use two fingers to move the map';\n this._cooperativeGesturesScreen.innerHTML = `\n
${desktopMessage}
\n
${mobileMessage}
\n `;\n\n // Remove cooperative gesture screen from the accessibility tree since screenreaders cannot interact with the map using gestures\n this._cooperativeGesturesScreen.setAttribute('aria-hidden', 'true');\n\n // Add event to canvas container since gesture container is pointer-events: none\n this._canvasContainer.addEventListener('wheel', this._cooperativeGesturesOnWheel, false);\n\n // Add a cooperative gestures class (enable touch-action: pan-x pan-y;)\n this._canvasContainer.classList.add('maplibregl-cooperative-gestures');\n }\n\n _destroyCooperativeGestures() {\n DOM.remove(this._cooperativeGesturesScreen);\n this._canvasContainer.removeEventListener('wheel', this._cooperativeGesturesOnWheel, false);\n this._canvasContainer.classList.remove('maplibregl-cooperative-gestures');\n }\n\n _resizeCanvas(width: number, height: number, pixelRatio: number) {\n // Request the required canvas size taking the pixelratio into account.\n this._canvas.width = Math.floor(pixelRatio * width);\n this._canvas.height = Math.floor(pixelRatio * height);\n\n // Maintain the same canvas size, potentially downscaling it for HiDPI displays\n this._canvas.style.width = `${width}px`;\n this._canvas.style.height = `${height}px`;\n }\n\n _setupPainter() {\n\n const attributes = {\n alpha: true,\n stencil: true,\n depth: true,\n failIfMajorPerformanceCaveat: this._failIfMajorPerformanceCaveat,\n preserveDrawingBuffer: this._preserveDrawingBuffer,\n antialias: this._antialias || false\n };\n\n let webglcontextcreationerrorDetailObject: any = null;\n this._canvas.addEventListener('webglcontextcreationerror', (args: WebGLContextEvent) => {\n webglcontextcreationerrorDetailObject = {requestedAttributes: attributes};\n if (args) {\n webglcontextcreationerrorDetailObject.statusMessage = args.statusMessage;\n webglcontextcreationerrorDetailObject.type = args.type;\n }\n }, {once: true});\n\n const gl =\n this._canvas.getContext('webgl2', attributes) as WebGL2RenderingContext ||\n this._canvas.getContext('webgl', attributes) as WebGLRenderingContext;\n\n if (!gl) {\n const msg = 'Failed to initialize WebGL';\n if (webglcontextcreationerrorDetailObject) {\n webglcontextcreationerrorDetailObject.message = msg;\n throw new Error(JSON.stringify(webglcontextcreationerrorDetailObject));\n } else {\n throw new Error(msg);\n }\n }\n\n this.painter = new Painter(gl, this.transform);\n\n webpSupported.testSupport(gl);\n }\n\n _contextLost = (event: any) => {\n event.preventDefault();\n if (this._frame) {\n this._frame.cancel();\n this._frame = null;\n }\n this.fire(new Event('webglcontextlost', {originalEvent: event}));\n };\n\n _contextRestored = (event: any) => {\n this._setupPainter();\n this.resize();\n this._update();\n this.fire(new Event('webglcontextrestored', {originalEvent: event}));\n };\n\n _onMapScroll = (event: any) => {\n if (event.target !== this._container) return;\n\n // Revert any scroll which would move the canvas outside of the view\n this._container.scrollTop = 0;\n this._container.scrollLeft = 0;\n return false;\n };\n\n _onCooperativeGesture(event: any, metaPress, touches) {\n if (!metaPress && touches < 2) {\n // Alert user how to scroll/pan\n this._cooperativeGesturesScreen.classList.add('maplibregl-show');\n setTimeout(() => {\n this._cooperativeGesturesScreen.classList.remove('maplibregl-show');\n }, 100);\n }\n return false;\n }\n\n /**\n * Returns a Boolean indicating whether the map is fully loaded.\n *\n * Returns `false` if the style is not yet fully loaded,\n * or if there has been a change to the sources or style that\n * has not yet fully loaded.\n *\n * @returns A Boolean indicating whether the map is fully loaded.\n */\n loaded(): boolean {\n return !this._styleDirty && !this._sourcesDirty && !!this.style && this.style.loaded();\n }\n\n /**\n * @internal\n * Update this map's style and sources, and re-render the map.\n *\n * @param updateStyle - mark the map's style for reprocessing as\n * well as its sources\n * @returns `this`\n */\n _update(updateStyle?: boolean) {\n if (!this.style || !this.style._loaded) return this;\n\n this._styleDirty = this._styleDirty || updateStyle;\n this._sourcesDirty = true;\n this.triggerRepaint();\n\n return this;\n }\n\n /**\n * @internal\n * Request that the given callback be executed during the next render\n * frame. Schedule a render frame if one is not already scheduled.\n *\n * @returns An id that can be used to cancel the callback\n */\n _requestRenderFrame(callback: () => void): TaskID {\n this._update();\n return this._renderTaskQueue.add(callback);\n }\n\n _cancelRenderFrame(id: TaskID) {\n this._renderTaskQueue.remove(id);\n }\n\n /**\n * @internal\n * Call when a (re-)render of the map is required:\n * - The style has changed (`setPaintProperty()`, etc.)\n * - Source data has changed (e.g. tiles have finished loading)\n * - The map has is moving (or just finished moving)\n * - A transition is in progress\n *\n * @param paintStartTimeStamp - The time when the animation frame began executing.\n *\n * @returns `this`\n */\n _render(paintStartTimeStamp: number) {\n const fadeDuration = this._idleTriggered ? this._fadeDuration : 0;\n\n // A custom layer may have used the context asynchronously. Mark the state as dirty.\n this.painter.context.setDirty();\n this.painter.setBaseState();\n\n this._renderTaskQueue.run(paintStartTimeStamp);\n // A task queue callback may have fired a user event which may have removed the map\n if (this._removed) return;\n\n let crossFading = false;\n\n // If the style has changed, the map is being zoomed, or a transition or fade is in progress:\n // - Apply style changes (in a batch)\n // - Recalculate paint properties.\n if (this.style && this._styleDirty) {\n this._styleDirty = false;\n\n const zoom = this.transform.zoom;\n const now = browser.now();\n this.style.zoomHistory.update(zoom, now);\n\n const parameters = new EvaluationParameters(zoom, {\n now,\n fadeDuration,\n zoomHistory: this.style.zoomHistory,\n transition: this.style.getTransition()\n });\n\n const factor = parameters.crossFadingFactor();\n if (factor !== 1 || factor !== this._crossFadingFactor) {\n crossFading = true;\n this._crossFadingFactor = factor;\n }\n\n this.style.update(parameters);\n }\n\n // If we are in _render for any reason other than an in-progress paint\n // transition, update source caches to check for and load any tiles we\n // need for the current transform\n if (this.style && this._sourcesDirty) {\n this._sourcesDirty = false;\n this.style._updateSources(this.transform);\n }\n\n // update terrain stuff\n if (this.terrain) {\n this.terrain.sourceCache.update(this.transform, this.terrain);\n this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n if (!this._elevationFreeze) {\n this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n }\n } else {\n this.transform._minEleveationForCurrentTile = 0;\n this.transform.elevation = 0;\n }\n\n this._placementDirty = this.style && this.style._updatePlacement(this.painter.transform, this.showCollisionBoxes, fadeDuration, this._crossSourceCollisions);\n\n // Actually draw\n this.painter.render(this.style, {\n showTileBoundaries: this.showTileBoundaries,\n showOverdrawInspector: this._showOverdrawInspector,\n rotating: this.isRotating(),\n zooming: this.isZooming(),\n moving: this.isMoving(),\n fadeDuration,\n showPadding: this.showPadding,\n });\n\n this.fire(new Event('render'));\n\n if (this.loaded() && !this._loaded) {\n this._loaded = true;\n PerformanceUtils.mark(PerformanceMarkers.load);\n this.fire(new Event('load'));\n }\n\n if (this.style && (this.style.hasTransitions() || crossFading)) {\n this._styleDirty = true;\n }\n\n if (this.style && !this._placementDirty) {\n // Since no fade operations are in progress, we can release\n // all tiles held for fading. If we didn't do this, the tiles\n // would just sit in the SourceCaches until the next render\n this.style._releaseSymbolFadeTiles();\n }\n\n // Schedule another render frame if it's needed.\n //\n // Even though `_styleDirty` and `_sourcesDirty` are reset in this\n // method, synchronous events fired during Style#update or\n // Style#_updateSources could have caused them to be set again.\n const somethingDirty = this._sourcesDirty || this._styleDirty || this._placementDirty;\n if (somethingDirty || this._repaint) {\n this.triggerRepaint();\n } else if (!this.isMoving() && this.loaded()) {\n this.fire(new Event('idle'));\n }\n\n if (this._loaded && !this._fullyLoaded && !somethingDirty) {\n this._fullyLoaded = true;\n PerformanceUtils.mark(PerformanceMarkers.fullLoad);\n }\n\n return this;\n }\n\n /**\n * Force a synchronous redraw of the map.\n * @returns `this`\n * @example\n * ```ts\n * map.redraw();\n * ```\n */\n redraw(): this {\n if (this.style) {\n // cancel the scheduled update\n if (this._frame) {\n this._frame.cancel();\n this._frame = null;\n }\n this._render(0);\n }\n return this;\n }\n\n /**\n * Clean up and release all internal resources associated with this map.\n *\n * This includes DOM elements, event bindings, web workers, and WebGL resources.\n *\n * Use this method when you are done using the map and wish to ensure that it no\n * longer consumes browser resources. Afterwards, you must not call any other\n * methods on the map.\n */\n remove() {\n if (this._hash) this._hash.remove();\n\n for (const control of this._controls) control.onRemove(this);\n this._controls = [];\n\n if (this._frame) {\n this._frame.cancel();\n this._frame = null;\n }\n this._renderTaskQueue.clear();\n this.painter.destroy();\n this.handlers.destroy();\n delete this.handlers;\n this.setStyle(null);\n if (typeof window !== 'undefined') {\n removeEventListener('online', this._onWindowOnline, false);\n }\n\n ImageRequest.removeThrottleControl(this._imageQueueHandle);\n\n this._resizeObserver?.disconnect();\n const extension = this.painter.context.gl.getExtension('WEBGL_lose_context');\n if (extension) extension.loseContext();\n this._canvas.removeEventListener('webglcontextrestored', this._contextRestored, false);\n this._canvas.removeEventListener('webglcontextlost', this._contextLost, false);\n DOM.remove(this._canvasContainer);\n DOM.remove(this._controlContainer);\n if (this._cooperativeGestures) {\n this._destroyCooperativeGestures();\n }\n this._container.classList.remove('maplibregl-map');\n\n PerformanceUtils.clearMetrics();\n\n this._removed = true;\n this.fire(new Event('remove'));\n }\n\n /**\n * Trigger the rendering of a single frame. Use this method with custom layers to\n * repaint the map when the layer changes. Calling this multiple times before the\n * next frame is rendered will still result in only a single frame being rendered.\n * @example\n * ```ts\n * map.triggerRepaint();\n * ```\n * @see [Add a 3D model](https://maplibre.org/maplibre-gl-js/docs/examples/add-3d-model/)\n * @see [Add an animated icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-animated/)\n */\n triggerRepaint() {\n if (this.style && !this._frame) {\n this._frame = browser.frame((paintStartTimeStamp: number) => {\n PerformanceUtils.frame(paintStartTimeStamp);\n this._frame = null;\n this._render(paintStartTimeStamp);\n });\n }\n }\n\n _onWindowOnline = () => {\n this._update();\n };\n\n /**\n * Gets and sets a Boolean indicating whether the map will render an outline\n * around each tile and the tile ID. These tile boundaries are useful for\n * debugging.\n *\n * The uncompressed file size of the first vector source is drawn in the top left\n * corner of each tile, next to the tile ID.\n *\n * @example\n * ```ts\n * map.showTileBoundaries = true;\n * ```\n */\n get showTileBoundaries(): boolean { return !!this._showTileBoundaries; }\n set showTileBoundaries(value: boolean) {\n if (this._showTileBoundaries === value) return;\n this._showTileBoundaries = value;\n this._update();\n }\n\n /**\n * Gets and sets a Boolean indicating whether the map will visualize\n * the padding offsets.\n */\n get showPadding(): boolean { return !!this._showPadding; }\n set showPadding(value: boolean) {\n if (this._showPadding === value) return;\n this._showPadding = value;\n this._update();\n }\n\n /**\n * Gets and sets a Boolean indicating whether the map will render boxes\n * around all symbols in the data source, revealing which symbols\n * were rendered or which were hidden due to collisions.\n * This information is useful for debugging.\n */\n get showCollisionBoxes(): boolean { return !!this._showCollisionBoxes; }\n set showCollisionBoxes(value: boolean) {\n if (this._showCollisionBoxes === value) return;\n this._showCollisionBoxes = value;\n if (value) {\n // When we turn collision boxes on we have to generate them for existing tiles\n // When we turn them off, there's no cost to leaving existing boxes in place\n this.style._generateCollisionBoxes();\n } else {\n // Otherwise, call an update to remove collision boxes\n this._update();\n }\n }\n\n /**\n * Gets and sets a Boolean indicating whether the map should color-code\n * each fragment to show how many times it has been shaded.\n * White fragments have been shaded 8 or more times.\n * Black fragments have been shaded 0 times.\n * This information is useful for debugging.\n */\n get showOverdrawInspector(): boolean { return !!this._showOverdrawInspector; }\n set showOverdrawInspector(value: boolean) {\n if (this._showOverdrawInspector === value) return;\n this._showOverdrawInspector = value;\n this._update();\n }\n\n /**\n * Gets and sets a Boolean indicating whether the map will\n * continuously repaint. This information is useful for analyzing performance.\n */\n get repaint(): boolean { return !!this._repaint; }\n set repaint(value: boolean) {\n if (this._repaint !== value) {\n this._repaint = value;\n this.triggerRepaint();\n }\n }\n // show vertices\n get vertices(): boolean { return !!this._vertices; }\n set vertices(value: boolean) { this._vertices = value; this._update(); }\n\n /**\n * Returns the package version of the library\n * @returns Package version of the library\n */\n get version(): string {\n return version;\n }\n\n /**\n * Returns the elevation for the point where the camera is looking.\n * This value corresponds to:\n * \"meters above sea level\" * \"exaggeration\"\n * @returns The elevation.\n */\n getCameraTargetElevation(): number {\n return this.transform.elevation;\n }\n}\n","import type Point from '@mapbox/point-geometry';\n\nimport {DragMoveHandler, DragRotateResult, DragPitchResult, DragHandler} from './drag_handler';\nimport {OneFingerTouchMoveStateManager} from './drag_move_state_manager';\n\nexport interface OneFingerTouchRotateHandler extends DragMoveHandler {}\nexport interface OneFingerTouchPitchHandler extends DragMoveHandler {}\n\nconst assignEvents = (handler: DragHandler) => {\n handler.touchstart = handler.dragStart;\n handler.touchmoveWindow = handler.dragMove;\n handler.touchend = handler.dragEnd;\n};\n\nexport const generateOneFingerTouchRotationHandler = ({enable, clickTolerance, bearingDegreesPerPixelMoved = 0.8}: {\n clickTolerance: number;\n bearingDegreesPerPixelMoved?: number;\n enable?: boolean;\n}): OneFingerTouchRotateHandler => {\n const touchMoveStateManager = new OneFingerTouchMoveStateManager();\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({bearingDelta: (point.x - lastPoint.x) * bearingDegreesPerPixelMoved}),\n moveStateManager: touchMoveStateManager,\n enable,\n assignEvents,\n });\n};\n\nexport const generateOneFingerTouchPitchHandler = ({enable, clickTolerance, pitchDegreesPerPixelMoved = -0.5}: {\n clickTolerance: number;\n pitchDegreesPerPixelMoved?: number;\n enable?: boolean;\n}): OneFingerTouchPitchHandler => {\n const touchMoveStateManager = new OneFingerTouchMoveStateManager();\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({pitchDelta: (point.y - lastPoint.y) * pitchDegreesPerPixelMoved}),\n moveStateManager: touchMoveStateManager,\n enable,\n assignEvents,\n });\n};\n","import Point from '@mapbox/point-geometry';\n\nimport {DOM} from '../../util/dom';\nimport {extend} from '../../util/util';\nimport {generateMousePitchHandler, generateMouseRotationHandler, MousePitchHandler, MouseRotateHandler} from '../handler/mouse';\nimport {generateOneFingerTouchPitchHandler, generateOneFingerTouchRotationHandler, OneFingerTouchPitchHandler, OneFingerTouchRotateHandler} from '../handler/one_finger_touch_drag';\n\nimport type {Map} from '../map';\nimport type {IControl} from './control';\n\n/**\n * The {@link NavigationControl} options object\n */\ntype NavigationOptions = {\n /**\n * If `true` the compass button is included.\n */\n showCompass?: boolean;\n /**\n * If `true` the zoom-in and zoom-out buttons are included.\n */\n showZoom?: boolean;\n /**\n * If `true` the pitch is visualized by rotating X-axis of compass.\n */\n visualizePitch?: boolean;\n};\n\nconst defaultOptions: NavigationOptions = {\n showCompass: true,\n showZoom: true,\n visualizePitch: false\n};\n\n/**\n * A `NavigationControl` control contains zoom buttons and a compass.\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * let nav = new maplibregl.NavigationControl();\n * map.addControl(nav, 'top-left');\n * ```\n * @see [Display map navigation controls](https://maplibre.org/maplibre-gl-js/docs/examples/navigation/)\n */\nexport class NavigationControl implements IControl {\n _map: Map;\n options: NavigationOptions;\n _container: HTMLElement;\n _zoomInButton: HTMLButtonElement;\n _zoomOutButton: HTMLButtonElement;\n _compass: HTMLButtonElement;\n _compassIcon: HTMLElement;\n _handler: MouseRotateWrapper;\n\n /**\n * @param options - the control's options\n */\n constructor(options?: NavigationOptions) {\n this.options = extend({}, defaultOptions, options);\n\n this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group');\n this._container.addEventListener('contextmenu', (e) => e.preventDefault());\n\n if (this.options.showZoom) {\n this._zoomInButton = this._createButton('maplibregl-ctrl-zoom-in', (e) => this._map.zoomIn({}, {originalEvent: e}));\n DOM.create('span', 'maplibregl-ctrl-icon', this._zoomInButton).setAttribute('aria-hidden', 'true');\n this._zoomOutButton = this._createButton('maplibregl-ctrl-zoom-out', (e) => this._map.zoomOut({}, {originalEvent: e}));\n DOM.create('span', 'maplibregl-ctrl-icon', this._zoomOutButton).setAttribute('aria-hidden', 'true');\n }\n if (this.options.showCompass) {\n this._compass = this._createButton('maplibregl-ctrl-compass', (e) => {\n if (this.options.visualizePitch) {\n this._map.resetNorthPitch({}, {originalEvent: e});\n } else {\n this._map.resetNorth({}, {originalEvent: e});\n }\n });\n this._compassIcon = DOM.create('span', 'maplibregl-ctrl-icon', this._compass);\n this._compassIcon.setAttribute('aria-hidden', 'true');\n }\n }\n\n _updateZoomButtons = () => {\n const zoom = this._map.getZoom();\n const isMax = zoom === this._map.getMaxZoom();\n const isMin = zoom === this._map.getMinZoom();\n this._zoomInButton.disabled = isMax;\n this._zoomOutButton.disabled = isMin;\n this._zoomInButton.setAttribute('aria-disabled', isMax.toString());\n this._zoomOutButton.setAttribute('aria-disabled', isMin.toString());\n };\n\n _rotateCompassArrow = () => {\n const rotate = this.options.visualizePitch ?\n `scale(${1 / Math.pow(Math.cos(this._map.transform.pitch * (Math.PI / 180)), 0.5)}) rotateX(${this._map.transform.pitch}deg) rotateZ(${this._map.transform.angle * (180 / Math.PI)}deg)` :\n `rotate(${this._map.transform.angle * (180 / Math.PI)}deg)`;\n\n this._compassIcon.style.transform = rotate;\n };\n\n onAdd(map: Map) {\n this._map = map;\n if (this.options.showZoom) {\n this._setButtonTitle(this._zoomInButton, 'ZoomIn');\n this._setButtonTitle(this._zoomOutButton, 'ZoomOut');\n this._map.on('zoom', this._updateZoomButtons);\n this._updateZoomButtons();\n }\n if (this.options.showCompass) {\n this._setButtonTitle(this._compass, 'ResetBearing');\n if (this.options.visualizePitch) {\n this._map.on('pitch', this._rotateCompassArrow);\n }\n this._map.on('rotate', this._rotateCompassArrow);\n this._rotateCompassArrow();\n this._handler = new MouseRotateWrapper(this._map, this._compass, this.options.visualizePitch);\n }\n return this._container;\n }\n\n onRemove() {\n DOM.remove(this._container);\n if (this.options.showZoom) {\n this._map.off('zoom', this._updateZoomButtons);\n }\n if (this.options.showCompass) {\n if (this.options.visualizePitch) {\n this._map.off('pitch', this._rotateCompassArrow);\n }\n this._map.off('rotate', this._rotateCompassArrow);\n this._handler.off();\n delete this._handler;\n }\n\n delete this._map;\n }\n\n _createButton(className: string, fn: (e?: any) => unknown) {\n const a = DOM.create('button', className, this._container) as HTMLButtonElement;\n a.type = 'button';\n a.addEventListener('click', fn);\n return a;\n }\n\n _setButtonTitle = (button: HTMLButtonElement, title: string) => {\n const str = this._map._getUIString(`NavigationControl.${title}`);\n button.title = str;\n button.setAttribute('aria-label', str);\n };\n}\n\nclass MouseRotateWrapper {\n\n map: Map;\n _clickTolerance: number;\n element: HTMLElement;\n // Rotation and pitch handlers are separated due to different _clickTolerance values\n mouseRotate: MouseRotateHandler;\n touchRotate: OneFingerTouchRotateHandler;\n mousePitch: MousePitchHandler;\n touchPitch: OneFingerTouchPitchHandler;\n _startPos: Point;\n _lastPos: Point;\n\n constructor(map: Map, element: HTMLElement, pitch: boolean = false) {\n this._clickTolerance = 10;\n const mapRotateTolerance = map.dragRotate._mouseRotate.getClickTolerance();\n const mapPitchTolerance = map.dragRotate._mousePitch.getClickTolerance();\n this.element = element;\n this.mouseRotate = generateMouseRotationHandler({clickTolerance: mapRotateTolerance, enable: true});\n this.touchRotate = generateOneFingerTouchRotationHandler({clickTolerance: mapRotateTolerance, enable: true});\n this.map = map;\n if (pitch) {\n this.mousePitch = generateMousePitchHandler({clickTolerance: mapPitchTolerance, enable: true});\n this.touchPitch = generateOneFingerTouchPitchHandler({clickTolerance: mapPitchTolerance, enable: true});\n }\n\n DOM.addEventListener(element, 'mousedown', this.mousedown);\n DOM.addEventListener(element, 'touchstart', this.touchstart, {passive: false});\n DOM.addEventListener(element, 'touchcancel', this.reset);\n }\n\n startMouse(e: MouseEvent, point: Point) {\n this.mouseRotate.dragStart(e, point);\n if (this.mousePitch) this.mousePitch.dragStart(e, point);\n DOM.disableDrag();\n }\n\n startTouch(e: TouchEvent, point: Point) {\n this.touchRotate.dragStart(e, point);\n if (this.touchPitch) this.touchPitch.dragStart(e, point);\n DOM.disableDrag();\n }\n\n moveMouse(e: MouseEvent, point: Point) {\n const map = this.map;\n const {bearingDelta} = this.mouseRotate.dragMove(e, point) || {};\n if (bearingDelta) map.setBearing(map.getBearing() + bearingDelta);\n if (this.mousePitch) {\n const {pitchDelta} = this.mousePitch.dragMove(e, point) || {};\n if (pitchDelta) map.setPitch(map.getPitch() + pitchDelta);\n }\n }\n\n moveTouch(e: TouchEvent, point: Point) {\n const map = this.map;\n const {bearingDelta} = this.touchRotate.dragMove(e, point) || {};\n if (bearingDelta) map.setBearing(map.getBearing() + bearingDelta);\n if (this.touchPitch) {\n const {pitchDelta} = this.touchPitch.dragMove(e, point) || {};\n if (pitchDelta) map.setPitch(map.getPitch() + pitchDelta);\n }\n }\n\n off() {\n const element = this.element;\n DOM.removeEventListener(element, 'mousedown', this.mousedown);\n DOM.removeEventListener(element, 'touchstart', this.touchstart, {passive: false});\n DOM.removeEventListener(window, 'touchmove', this.touchmove, {passive: false});\n DOM.removeEventListener(window, 'touchend', this.touchend);\n DOM.removeEventListener(element, 'touchcancel', this.reset);\n this.offTemp();\n }\n\n offTemp() {\n DOM.enableDrag();\n DOM.removeEventListener(window, 'mousemove', this.mousemove);\n DOM.removeEventListener(window, 'mouseup', this.mouseup);\n DOM.removeEventListener(window, 'touchmove', this.touchmove, {passive: false});\n DOM.removeEventListener(window, 'touchend', this.touchend);\n }\n\n mousedown = (e: MouseEvent) => {\n this.startMouse(extend({}, e, {ctrlKey: true, preventDefault: () => e.preventDefault()}), DOM.mousePos(this.element, e));\n DOM.addEventListener(window, 'mousemove', this.mousemove);\n DOM.addEventListener(window, 'mouseup', this.mouseup);\n };\n\n mousemove = (e: MouseEvent) => {\n this.moveMouse(e, DOM.mousePos(this.element, e));\n };\n\n mouseup = (e: MouseEvent) => {\n this.mouseRotate.dragEnd(e);\n if (this.mousePitch) this.mousePitch.dragEnd(e);\n this.offTemp();\n };\n\n touchstart = (e: TouchEvent) => {\n if (e.targetTouches.length !== 1) {\n this.reset();\n } else {\n this._startPos = this._lastPos = DOM.touchPos(this.element, e.targetTouches)[0];\n this.startTouch(e, this._startPos);\n DOM.addEventListener(window, 'touchmove', this.touchmove, {passive: false});\n DOM.addEventListener(window, 'touchend', this.touchend);\n }\n };\n\n touchmove = (e: TouchEvent) => {\n if (e.targetTouches.length !== 1) {\n this.reset();\n } else {\n this._lastPos = DOM.touchPos(this.element, e.targetTouches)[0];\n this.moveTouch(e, this._lastPos);\n }\n };\n\n touchend = (e: TouchEvent) => {\n if (e.targetTouches.length === 0 &&\n this._startPos &&\n this._lastPos &&\n this._startPos.dist(this._lastPos) < this._clickTolerance) {\n this.element.click();\n }\n delete this._startPos;\n delete this._lastPos;\n this.offTemp();\n };\n\n reset = () => {\n this.mouseRotate.reset();\n if (this.mousePitch) this.mousePitch.reset();\n this.touchRotate.reset();\n if (this.touchPitch) this.touchPitch.reset();\n delete this._startPos;\n delete this._lastPos;\n this.offTemp();\n };\n}\n","let supportsGeolocation;\n\nexport function checkGeolocationSupport(callback: (supported: boolean) => void, forceRecalculation = false): void {\n if (supportsGeolocation !== undefined && !forceRecalculation) {\n callback(supportsGeolocation);\n } else if (window.navigator.permissions !== undefined) {\n // navigator.permissions has incomplete browser support\n // http://caniuse.com/#feat=permissions-api\n // Test for the case where a browser disables Geolocation because of an\n // insecure origin\n window.navigator.permissions.query({name: 'geolocation'}).then((p) => {\n supportsGeolocation = p.state !== 'denied';\n callback(supportsGeolocation);\n }).catch(() => {\n // Fix for iOS16 which rejects query but still supports geolocation\n supportsGeolocation = !!window.navigator.geolocation;\n callback(supportsGeolocation);\n });\n\n } else {\n supportsGeolocation = !!window.navigator.geolocation;\n callback(supportsGeolocation);\n }\n}\n","import {LngLat} from '../geo/lng_lat';\n\nimport type Point from '@mapbox/point-geometry';\nimport type {Transform} from '../geo/transform';\n\n/**\n * Given a LngLat, prior projected position, and a transform, return a new LngLat shifted\n * n × 360° east or west for some n ≥ 0 such that:\n *\n * * the projected location of the result is on screen, if possible, and secondarily:\n * * the difference between the projected location of the result and the prior position\n * is minimized.\n *\n * The object is to preserve perceived object constancy for Popups and Markers as much as\n * possible; they should avoid shifting large distances across the screen, even when the\n * map center changes by ±360° due to automatic wrapping, and when about to go off screen,\n * should wrap just enough to avoid doing so.\n */\nexport function smartWrap(lngLat: LngLat, priorPos: Point, transform: Transform): LngLat {\n lngLat = new LngLat(lngLat.lng, lngLat.lat);\n\n // First, try shifting one world in either direction, and see if either is closer to the\n // prior position. This preserves object constancy when the map center is auto-wrapped\n // during animations.\n if (priorPos) {\n const left = new LngLat(lngLat.lng - 360, lngLat.lat);\n const right = new LngLat(lngLat.lng + 360, lngLat.lat);\n const delta = transform.locationPoint(lngLat).distSqr(priorPos);\n if (transform.locationPoint(left).distSqr(priorPos) < delta) {\n lngLat = left;\n } else if (transform.locationPoint(right).distSqr(priorPos) < delta) {\n lngLat = right;\n }\n }\n\n // Second, wrap toward the center until the new position is on screen, or we can't get\n // any closer.\n while (Math.abs(lngLat.lng - transform.center.lng) > 180) {\n const pos = transform.locationPoint(lngLat);\n if (pos.x >= 0 && pos.y >= 0 && pos.x <= transform.width && pos.y <= transform.height) {\n break;\n }\n if (lngLat.lng > transform.center.lng) {\n lngLat.lng -= 360;\n } else {\n lngLat.lng += 360;\n }\n }\n\n return lngLat;\n}\n","/**\n * Where to position the anchor.\n * Used by a popup and a marker.\n */\nexport type PositionAnchor = 'center' | 'top' | 'bottom' | 'left' | 'right' | 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right';\n\nexport const anchorTranslate: {\n [_ in PositionAnchor]: string;\n} = {\n 'center': 'translate(-50%,-50%)',\n 'top': 'translate(-50%,0)',\n 'top-left': 'translate(0,0)',\n 'top-right': 'translate(-100%,0)',\n 'bottom': 'translate(-50%,-100%)',\n 'bottom-left': 'translate(0,-100%)',\n 'bottom-right': 'translate(-100%,-100%)',\n 'left': 'translate(0,-50%)',\n 'right': 'translate(-100%,-50%)'\n};\n\nexport function applyAnchorClass(element: HTMLElement, anchor: PositionAnchor, prefix: string) {\n const classList = element.classList;\n for (const key in anchorTranslate) {\n classList.remove(`maplibregl-${prefix}-anchor-${key}`);\n }\n classList.add(`maplibregl-${prefix}-anchor-${anchor}`);\n}\n","import {DOM} from '../util/dom';\nimport {LngLat} from '../geo/lng_lat';\nimport Point from '@mapbox/point-geometry';\nimport {smartWrap} from '../util/smart_wrap';\nimport {anchorTranslate, applyAnchorClass} from './anchor';\nimport type {PositionAnchor} from './anchor';\nimport {Event, Evented} from '../util/evented';\nimport type {Map} from './map';\nimport {Popup, Offset} from './popup';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport type {MapMouseEvent, MapTouchEvent} from './events';\nimport type {PointLike} from './camera';\n\n/**\n * Alignment options of rotation and pitch\n */\ntype Alignment = 'map' | 'viewport' | 'auto';\n\n/**\n * The {@link Marker} options object\n */\ntype MarkerOptions = {\n /**\n * DOM element to use as a marker. The default is a light blue, droplet-shaped SVG marker.\n */\n element?: HTMLElement;\n /**\n * Space-separated CSS class names to add to marker element.\n */\n className?: string;\n /**\n * The offset in pixels as a {@link PointLike} object to apply relative to the element's center. Negatives indicate left and up.\n */\n offset?: PointLike;\n /**\n * A string indicating the part of the Marker that should be positioned closest to the coordinate set via {@link Marker#setLngLat}.\n * Options are `'center'`, `'top'`, `'bottom'`, `'left'`, `'right'`, `'top-left'`, `'top-right'`, `'bottom-left'`, and `'bottom-right'`.\n * @defaultValue 'center'\n * */\n anchor?: PositionAnchor;\n /**\n * The color to use for the default marker if options.element is not provided. The default is light blue.\n * @defaultValue '#3FB1CE'\n */\n color?: string;\n /**\n * The scale to use for the default marker if options.element is not provided. The default scale corresponds to a height of `41px` and a width of `27px`.\n * @defaultValue 1\n */\n scale?: number;\n /**\n * A boolean indicating whether or not a marker is able to be dragged to a new position on the map.\n * @defaultValue false\n */\n draggable?: boolean;\n /**\n * The max number of pixels a user can shift the mouse pointer during a click on the marker for it to be considered a valid click (as opposed to a marker drag). The default is to inherit map's clickTolerance.\n * @defaultValue 0\n */\n clickTolerance?: number;\n /**\n * The rotation angle of the marker in degrees, relative to its respective `rotationAlignment` setting. A positive value will rotate the marker clockwise.\n * @defaultValue 0\n */\n rotation?: number;\n /**\n * `map` aligns the `Marker`'s rotation relative to the map, maintaining a bearing as the map rotates. `viewport` aligns the `Marker`'s rotation relative to the viewport, agnostic to map rotations. `auto` is equivalent to `viewport`.\n * @defaultValue 'auto'\n */\n rotationAlignment?: Alignment;\n /**\n * `map` aligns the `Marker` to the plane of the map. `viewport` aligns the `Marker` to the plane of the viewport. `auto` automatically matches the value of `rotationAlignment`.\n * @defaultValue 'auto'\n */\n pitchAlignment?: Alignment;\n};\n\n/**\n * Creates a marker component\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([30.5, 50.5])\n * .addTo(map);\n * ```\n *\n * @example\n * Set options\n * ```ts\n * let marker = new maplibregl.Marker({\n * color: \"#FFFFFF\",\n * draggable: true\n * }).setLngLat([30.5, 50.5])\n * .addTo(map);\n * ```\n * @see [Add custom icons with Markers](https://maplibre.org/maplibre-gl-js/docs/examples/custom-marker-icons/)\n * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/)\n *\n * ### Events\n *\n * @event `dragstart` Fired when dragging starts, `marker` object that is being dragged\n *\n * @event `drag` Fired while dragging. `marker` object that is being dragged\n *\n * @event `dragend` Fired when the marker is finished being dragged, `marker` object that was dragged\n */\nexport class Marker extends Evented {\n _map: Map;\n _anchor: PositionAnchor;\n _offset: Point;\n _element: HTMLElement;\n _popup: Popup;\n _lngLat: LngLat;\n _pos: Point;\n _color: string;\n _scale: number;\n _defaultMarker: boolean;\n _draggable: boolean;\n _clickTolerance: number;\n _isDragging: boolean;\n _state: 'inactive' | 'pending' | 'active'; // used for handling drag events\n _positionDelta: Point;\n _pointerdownPos: Point;\n _rotation: number;\n _pitchAlignment: Alignment;\n _rotationAlignment: Alignment;\n _originalTabIndex: string; // original tabindex of _element\n _opacityTimeout: ReturnType;\n\n /**\n * @param options - the options\n */\n constructor(options?: MarkerOptions) {\n super();\n\n this._anchor = options && options.anchor || 'center';\n this._color = options && options.color || '#3FB1CE';\n this._scale = options && options.scale || 1;\n this._draggable = options && options.draggable || false;\n this._clickTolerance = options && options.clickTolerance || 0;\n this._isDragging = false;\n this._state = 'inactive';\n this._rotation = options && options.rotation || 0;\n this._rotationAlignment = options && options.rotationAlignment || 'auto';\n this._pitchAlignment = options && options.pitchAlignment && options.pitchAlignment !== 'auto' ? options.pitchAlignment : this._rotationAlignment;\n\n if (!options || !options.element) {\n this._defaultMarker = true;\n this._element = DOM.create('div');\n this._element.setAttribute('aria-label', 'Map marker');\n\n // create default map marker SVG\n const svg = DOM.createNS('http://www.w3.org/2000/svg', 'svg');\n const defaultHeight = 41;\n const defaultWidth = 27;\n svg.setAttributeNS(null, 'display', 'block');\n svg.setAttributeNS(null, 'height', `${defaultHeight}px`);\n svg.setAttributeNS(null, 'width', `${defaultWidth}px`);\n svg.setAttributeNS(null, 'viewBox', `0 0 ${defaultWidth} ${defaultHeight}`);\n\n const markerLarge = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n markerLarge.setAttributeNS(null, 'stroke', 'none');\n markerLarge.setAttributeNS(null, 'stroke-width', '1');\n markerLarge.setAttributeNS(null, 'fill', 'none');\n markerLarge.setAttributeNS(null, 'fill-rule', 'evenodd');\n\n const page1 = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n page1.setAttributeNS(null, 'fill-rule', 'nonzero');\n\n const shadow = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n shadow.setAttributeNS(null, 'transform', 'translate(3.0, 29.0)');\n shadow.setAttributeNS(null, 'fill', '#000000');\n\n const ellipses = [\n {'rx': '10.5', 'ry': '5.25002273'},\n {'rx': '10.5', 'ry': '5.25002273'},\n {'rx': '9.5', 'ry': '4.77275007'},\n {'rx': '8.5', 'ry': '4.29549936'},\n {'rx': '7.5', 'ry': '3.81822308'},\n {'rx': '6.5', 'ry': '3.34094679'},\n {'rx': '5.5', 'ry': '2.86367051'},\n {'rx': '4.5', 'ry': '2.38636864'}\n ];\n\n for (const data of ellipses) {\n const ellipse = DOM.createNS('http://www.w3.org/2000/svg', 'ellipse');\n ellipse.setAttributeNS(null, 'opacity', '0.04');\n ellipse.setAttributeNS(null, 'cx', '10.5');\n ellipse.setAttributeNS(null, 'cy', '5.80029008');\n ellipse.setAttributeNS(null, 'rx', data['rx']);\n ellipse.setAttributeNS(null, 'ry', data['ry']);\n shadow.appendChild(ellipse);\n }\n\n const background = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n background.setAttributeNS(null, 'fill', this._color);\n\n const bgPath = DOM.createNS('http://www.w3.org/2000/svg', 'path');\n bgPath.setAttributeNS(null, 'd', 'M27,13.5 C27,19.074644 20.250001,27.000002 14.75,34.500002 C14.016665,35.500004 12.983335,35.500004 12.25,34.500002 C6.7499993,27.000002 0,19.222562 0,13.5 C0,6.0441559 6.0441559,0 13.5,0 C20.955844,0 27,6.0441559 27,13.5 Z');\n\n background.appendChild(bgPath);\n\n const border = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n border.setAttributeNS(null, 'opacity', '0.25');\n border.setAttributeNS(null, 'fill', '#000000');\n\n const borderPath = DOM.createNS('http://www.w3.org/2000/svg', 'path');\n borderPath.setAttributeNS(null, 'd', 'M13.5,0 C6.0441559,0 0,6.0441559 0,13.5 C0,19.222562 6.7499993,27 12.25,34.5 C13,35.522727 14.016664,35.500004 14.75,34.5 C20.250001,27 27,19.074644 27,13.5 C27,6.0441559 20.955844,0 13.5,0 Z M13.5,1 C20.415404,1 26,6.584596 26,13.5 C26,15.898657 24.495584,19.181431 22.220703,22.738281 C19.945823,26.295132 16.705119,30.142167 13.943359,33.908203 C13.743445,34.180814 13.612715,34.322738 13.5,34.441406 C13.387285,34.322738 13.256555,34.180814 13.056641,33.908203 C10.284481,30.127985 7.4148684,26.314159 5.015625,22.773438 C2.6163816,19.232715 1,15.953538 1,13.5 C1,6.584596 6.584596,1 13.5,1 Z');\n\n border.appendChild(borderPath);\n\n const maki = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n maki.setAttributeNS(null, 'transform', 'translate(6.0, 7.0)');\n maki.setAttributeNS(null, 'fill', '#FFFFFF');\n\n const circleContainer = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n circleContainer.setAttributeNS(null, 'transform', 'translate(8.0, 8.0)');\n\n const circle1 = DOM.createNS('http://www.w3.org/2000/svg', 'circle');\n circle1.setAttributeNS(null, 'fill', '#000000');\n circle1.setAttributeNS(null, 'opacity', '0.25');\n circle1.setAttributeNS(null, 'cx', '5.5');\n circle1.setAttributeNS(null, 'cy', '5.5');\n circle1.setAttributeNS(null, 'r', '5.4999962');\n\n const circle2 = DOM.createNS('http://www.w3.org/2000/svg', 'circle');\n circle2.setAttributeNS(null, 'fill', '#FFFFFF');\n circle2.setAttributeNS(null, 'cx', '5.5');\n circle2.setAttributeNS(null, 'cy', '5.5');\n circle2.setAttributeNS(null, 'r', '5.4999962');\n\n circleContainer.appendChild(circle1);\n circleContainer.appendChild(circle2);\n\n page1.appendChild(shadow);\n page1.appendChild(background);\n page1.appendChild(border);\n page1.appendChild(maki);\n page1.appendChild(circleContainer);\n\n svg.appendChild(page1);\n\n svg.setAttributeNS(null, 'height', `${defaultHeight * this._scale}px`);\n svg.setAttributeNS(null, 'width', `${defaultWidth * this._scale}px`);\n\n this._element.appendChild(svg);\n\n // if no element and no offset option given apply an offset for the default marker\n // the -14 as the y value of the default marker offset was determined as follows\n //\n // the marker tip is at the center of the shadow ellipse from the default svg\n // the y value of the center of the shadow ellipse relative to the svg top left is \"shadow transform translate-y (29.0) + ellipse cy (5.80029008)\"\n // offset to the svg center \"height (41 / 2)\" gives (29.0 + 5.80029008) - (41 / 2) and rounded for an integer pixel offset gives 14\n // negative is used to move the marker up from the center so the tip is at the Marker lngLat\n this._offset = Point.convert(options && options.offset || [0, -14]);\n } else {\n this._element = options.element;\n this._offset = Point.convert(options && options.offset || [0, 0]);\n }\n\n this._element.classList.add('maplibregl-marker');\n this._element.addEventListener('dragstart', (e: DragEvent) => {\n e.preventDefault();\n });\n this._element.addEventListener('mousedown', (e: MouseEvent) => {\n // prevent focusing on click\n e.preventDefault();\n });\n applyAnchorClass(this._element, this._anchor, 'marker');\n\n if (options && options.className) {\n for (const name of options.className.split(' ')) {\n this._element.classList.add(name);\n }\n }\n\n this._popup = null;\n }\n\n /**\n * Attaches the `Marker` to a `Map` object.\n * @param map - The MapLibre GL JS map to add the marker to.\n * @returns `this`\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([30.5, 50.5])\n * .addTo(map); // add the marker to the map\n * ```\n */\n addTo(map: Map): this {\n this.remove();\n this._map = map;\n map.getCanvasContainer().appendChild(this._element);\n map.on('move', this._update);\n map.on('moveend', this._update);\n map.on('terrain', this._update);\n\n this.setDraggable(this._draggable);\n this._update();\n\n // If we attached the `click` listener to the marker element, the popup\n // would close once the event propagated to `map` due to the\n // `Popup#_onClickClose` listener.\n this._map.on('click', this._onMapClick);\n\n return this;\n }\n\n /**\n * Removes the marker from a map\n * @example\n * ```ts\n * let marker = new maplibregl.Marker().addTo(map);\n * marker.remove();\n * ```\n * @returns `this`\n */\n remove(): this {\n if (this._opacityTimeout) {\n clearTimeout(this._opacityTimeout);\n delete this._opacityTimeout;\n }\n if (this._map) {\n this._map.off('click', this._onMapClick);\n this._map.off('move', this._update);\n this._map.off('moveend', this._update);\n this._map.off('mousedown', this._addDragHandler);\n this._map.off('touchstart', this._addDragHandler);\n this._map.off('mouseup', this._onUp);\n this._map.off('touchend', this._onUp);\n this._map.off('mousemove', this._onMove);\n this._map.off('touchmove', this._onMove);\n delete this._map;\n }\n DOM.remove(this._element);\n if (this._popup) this._popup.remove();\n return this;\n }\n\n /**\n * Get the marker's geographical location.\n *\n * The longitude of the result may differ by a multiple of 360 degrees from the longitude previously\n * set by `setLngLat` because `Marker` wraps the anchor longitude across copies of the world to keep\n * the marker on screen.\n *\n * @returns A {@link LngLat} describing the marker's location.\n * @example\n * ```ts\n * // Store the marker's longitude and latitude coordinates in a variable\n * let lngLat = marker.getLngLat();\n * // Print the marker's longitude and latitude values in the console\n * console.log('Longitude: ' + lngLat.lng + ', Latitude: ' + lngLat.lat )\n * ```\n * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/)\n */\n getLngLat(): LngLat {\n return this._lngLat;\n }\n\n /**\n * Set the marker's geographical position and move it.\n * @param lnglat - A {@link LngLat} describing where the marker should be located.\n * @returns `this`\n * @example\n * Create a new marker, set the longitude and latitude, and add it to the map\n * ```ts\n * new maplibregl.Marker()\n * .setLngLat([-65.017, -16.457])\n * .addTo(map);\n * ```\n * @see [Add custom icons with Markers](https://maplibre.org/maplibre-gl-js/docs/examples/custom-marker-icons/)\n * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/)\n */\n setLngLat(lnglat: LngLatLike): this {\n this._lngLat = LngLat.convert(lnglat);\n this._pos = null;\n if (this._popup) this._popup.setLngLat(this._lngLat);\n this._update();\n return this;\n }\n\n /**\n * Returns the `Marker`'s HTML element.\n * @returns element\n */\n getElement(): HTMLElement {\n return this._element;\n }\n\n /**\n * Binds a {@link Popup} to the {@link Marker}.\n * @param popup - An instance of the {@link Popup} class. If undefined or null, any popup\n * set on this {@link Marker} instance is unset.\n * @returns `this`\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([0, 0])\n * .setPopup(new maplibregl.Popup().setHTML(\"

Hello World!

\")) // add popup\n * .addTo(map);\n * ```\n * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/)\n */\n setPopup(popup?: Popup | null): this {\n if (this._popup) {\n this._popup.remove();\n this._popup = null;\n this._element.removeEventListener('keypress', this._onKeyPress);\n\n if (!this._originalTabIndex) {\n this._element.removeAttribute('tabindex');\n }\n }\n\n if (popup) {\n if (!('offset' in popup.options)) {\n const markerHeight = 41 - (5.8 / 2);\n const markerRadius = 13.5;\n const linearOffset = Math.abs(markerRadius) / Math.SQRT2;\n popup.options.offset = this._defaultMarker ? {\n 'top': [0, 0],\n 'top-left': [0, 0],\n 'top-right': [0, 0],\n 'bottom': [0, -markerHeight],\n 'bottom-left': [linearOffset, (markerHeight - markerRadius + linearOffset) * -1],\n 'bottom-right': [-linearOffset, (markerHeight - markerRadius + linearOffset) * -1],\n 'left': [markerRadius, (markerHeight - markerRadius) * -1],\n 'right': [-markerRadius, (markerHeight - markerRadius) * -1]\n } as Offset : this._offset;\n }\n this._popup = popup;\n if (this._lngLat) this._popup.setLngLat(this._lngLat);\n\n this._originalTabIndex = this._element.getAttribute('tabindex');\n if (!this._originalTabIndex) {\n this._element.setAttribute('tabindex', '0');\n }\n this._element.addEventListener('keypress', this._onKeyPress);\n }\n\n return this;\n }\n\n _onKeyPress = (e: KeyboardEvent) => {\n const code = e.code;\n const legacyCode = e.charCode || e.keyCode;\n\n if (\n (code === 'Space') || (code === 'Enter') ||\n (legacyCode === 32) || (legacyCode === 13) // space or enter\n ) {\n this.togglePopup();\n }\n };\n\n _onMapClick = (e: MapMouseEvent) => {\n const targetElement = e.originalEvent.target;\n const element = this._element;\n\n if (this._popup && (targetElement === element || element.contains(targetElement as any))) {\n this.togglePopup();\n }\n };\n\n /**\n * Returns the {@link Popup} instance that is bound to the {@link Marker}.\n * @returns popup\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([0, 0])\n * .setPopup(new maplibregl.Popup().setHTML(\"

Hello World!

\"))\n * .addTo(map);\n *\n * console.log(marker.getPopup()); // return the popup instance\n * ```\n */\n getPopup(): Popup {\n return this._popup;\n }\n\n /**\n * Opens or closes the {@link Popup} instance that is bound to the {@link Marker}, depending on the current state of the {@link Popup}.\n * @returns `this`\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([0, 0])\n * .setPopup(new maplibregl.Popup().setHTML(\"

Hello World!

\"))\n * .addTo(map);\n *\n * marker.togglePopup(); // toggle popup open or closed\n * ```\n */\n togglePopup(): this {\n const popup = this._popup;\n\n if (!popup) return this;\n else if (popup.isOpen()) popup.remove();\n else popup.addTo(this._map);\n return this;\n }\n\n _update = (e?: { type: 'move' | 'moveend' | 'terrain' | 'render' }) => {\n if (!this._map) return;\n\n const isFullyLoaded = this._map.loaded() && !this._map.isMoving();\n if (e?.type === 'terrain' || (e?.type === 'render' && !isFullyLoaded)) {\n this._map.once('render', this._update);\n }\n\n if (this._map.transform.renderWorldCopies) {\n this._lngLat = smartWrap(this._lngLat, this._pos, this._map.transform);\n }\n\n this._pos = this._map.project(this._lngLat)._add(this._offset);\n\n let rotation = '';\n if (this._rotationAlignment === 'viewport' || this._rotationAlignment === 'auto') {\n rotation = `rotateZ(${this._rotation}deg)`;\n } else if (this._rotationAlignment === 'map') {\n rotation = `rotateZ(${this._rotation - this._map.getBearing()}deg)`;\n }\n\n let pitch = '';\n if (this._pitchAlignment === 'viewport' || this._pitchAlignment === 'auto') {\n pitch = 'rotateX(0deg)';\n } else if (this._pitchAlignment === 'map') {\n pitch = `rotateX(${this._map.getPitch()}deg)`;\n }\n\n // because rounding the coordinates at every `move` event causes stuttered zooming\n // we only round them when _update is called with `moveend` or when its called with\n // no arguments (when the Marker is initialized or Marker#setLngLat is invoked).\n if (!e || e.type === 'moveend') {\n this._pos = this._pos.round();\n }\n\n DOM.setTransform(this._element, `${anchorTranslate[this._anchor]} translate(${this._pos.x}px, ${this._pos.y}px) ${pitch} ${rotation}`);\n\n // in case of 3D, ask the terrain coords-framebuffer for this pos and check if the marker is visible\n // call this logic in setTimeout with a timeout of 100ms to save performance in map-movement\n if (this._map.terrain && !this._opacityTimeout) this._opacityTimeout = setTimeout(() => {\n const lnglat = this._map.unproject(this._pos);\n const metresPerPixel = 40075016.686 * Math.abs(Math.cos(this._lngLat.lat * Math.PI / 180)) / Math.pow(2, this._map.transform.tileZoom + 8);\n this._element.style.opacity = lnglat.distanceTo(this._lngLat) > metresPerPixel * 20 ? '0.2' : '1.0';\n this._opacityTimeout = null;\n }, 100);\n };\n\n /**\n * Get the marker's offset.\n * @returns The marker's screen coordinates in pixels.\n */\n getOffset(): Point {\n return this._offset;\n }\n\n /**\n * Sets the offset of the marker\n * @param offset - The offset in pixels as a {@link PointLike} object to apply relative to the element's center. Negatives indicate left and up.\n * @returns `this`\n */\n setOffset(offset: PointLike): this {\n this._offset = Point.convert(offset);\n this._update();\n return this;\n }\n\n /**\n * Adds a CSS class to the marker element.\n *\n * @param className - on-empty string with CSS class name to add to marker element\n *\n * @example\n * ```\n * let marker = new maplibregl.Marker()\n * marker.addClassName('some-class')\n * ```\n */\n addClassName(className: string) {\n this._element.classList.add(className);\n }\n\n /**\n * Removes a CSS class from the marker element.\n *\n * @param className - Non-empty string with CSS class name to remove from marker element\n *\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * marker.removeClassName('some-class')\n * ```\n */\n removeClassName(className: string) {\n this._element.classList.remove(className);\n }\n\n /**\n * Add or remove the given CSS class on the marker element, depending on whether the element currently has that class.\n *\n * @param className - Non-empty string with CSS class name to add/remove\n *\n * @returns if the class was removed return false, if class was added, then return true\n *\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * marker.toggleClassName('toggleClass')\n * ```\n */\n toggleClassName(className: string): boolean {\n return this._element.classList.toggle(className);\n }\n\n _onMove = (e: MapMouseEvent | MapTouchEvent) => {\n if (!this._isDragging) {\n const clickTolerance = this._clickTolerance || this._map._clickTolerance;\n this._isDragging = e.point.dist(this._pointerdownPos) >= clickTolerance;\n }\n if (!this._isDragging) return;\n\n this._pos = e.point.sub(this._positionDelta);\n this._lngLat = this._map.unproject(this._pos);\n this.setLngLat(this._lngLat);\n // suppress click event so that popups don't toggle on drag\n this._element.style.pointerEvents = 'none';\n\n // make sure dragstart only fires on the first move event after mousedown.\n // this can't be on mousedown because that event doesn't necessarily\n // imply that a drag is about to happen.\n if (this._state === 'pending') {\n this._state = 'active';\n this.fire(new Event('dragstart'));\n }\n this.fire(new Event('drag'));\n };\n\n _onUp = () => {\n // revert to normal pointer event handling\n this._element.style.pointerEvents = 'auto';\n this._positionDelta = null;\n this._pointerdownPos = null;\n this._isDragging = false;\n this._map.off('mousemove', this._onMove);\n this._map.off('touchmove', this._onMove);\n\n // only fire dragend if it was preceded by at least one drag event\n if (this._state === 'active') {\n this.fire(new Event('dragend'));\n }\n\n this._state = 'inactive';\n };\n\n _addDragHandler = (e: MapMouseEvent | MapTouchEvent) => {\n if (this._element.contains(e.originalEvent.target as any)) {\n e.preventDefault();\n\n // We need to calculate the pixel distance between the click point\n // and the marker position, with the offset accounted for. Then we\n // can subtract this distance from the mousemove event's position\n // to calculate the new marker position.\n // If we don't do this, the marker 'jumps' to the click position\n // creating a jarring UX effect.\n this._positionDelta = e.point.sub(this._pos).add(this._offset);\n\n this._pointerdownPos = e.point;\n\n this._state = 'pending';\n this._map.on('mousemove', this._onMove);\n this._map.on('touchmove', this._onMove);\n this._map.once('mouseup', this._onUp);\n this._map.once('touchend', this._onUp);\n }\n };\n\n /**\n * Sets the `draggable` property and functionality of the marker\n * @param shouldBeDraggable - Turns drag functionality on/off\n * @returns `this`\n */\n setDraggable(shouldBeDraggable?: boolean): this {\n this._draggable = !!shouldBeDraggable; // convert possible undefined value to false\n\n // handle case where map may not exist yet\n // e.g. when setDraggable is called before addTo\n if (this._map) {\n if (shouldBeDraggable) {\n this._map.on('mousedown', this._addDragHandler);\n this._map.on('touchstart', this._addDragHandler);\n } else {\n this._map.off('mousedown', this._addDragHandler);\n this._map.off('touchstart', this._addDragHandler);\n }\n }\n\n return this;\n }\n\n /**\n * Returns true if the marker can be dragged\n * @returns True if the marker is draggable.\n */\n isDraggable(): boolean {\n return this._draggable;\n }\n\n /**\n * Sets the `rotation` property of the marker.\n * @param rotation - The rotation angle of the marker (clockwise, in degrees), relative to its respective {@link Marker#setRotationAlignment} setting.\n * @returns `this`\n */\n setRotation(rotation?: number): this {\n this._rotation = rotation || 0;\n this._update();\n return this;\n }\n\n /**\n * Returns the current rotation angle of the marker (in degrees).\n * @returns The current rotation angle of the marker.\n */\n getRotation(): number {\n return this._rotation;\n }\n\n /**\n * Sets the `rotationAlignment` property of the marker.\n * @param alignment - Sets the `rotationAlignment` property of the marker. defaults to 'auto'\n * @returns `this`\n */\n setRotationAlignment(alignment?: Alignment): this {\n this._rotationAlignment = alignment || 'auto';\n this._update();\n return this;\n }\n\n /**\n * Returns the current `rotationAlignment` property of the marker.\n * @returns The current rotational alignment of the marker.\n */\n getRotationAlignment(): Alignment {\n return this._rotationAlignment;\n }\n\n /**\n * Sets the `pitchAlignment` property of the marker.\n * @param alignment - Sets the `pitchAlignment` property of the marker. If alignment is 'auto', it will automatically match `rotationAlignment`.\n * @returns `this`\n */\n setPitchAlignment(alignment?: Alignment): this {\n this._pitchAlignment = alignment && alignment !== 'auto' ? alignment : this._rotationAlignment;\n this._update();\n return this;\n }\n\n /**\n * Returns the current `pitchAlignment` property of the marker.\n * @returns The current pitch alignment of the marker in degrees.\n */\n getPitchAlignment(): Alignment {\n return this._pitchAlignment;\n }\n}\n","import {Event, Evented} from '../../util/evented';\nimport {DOM} from '../../util/dom';\nimport {extend, warnOnce} from '../../util/util';\nimport {checkGeolocationSupport} from '../../util/geolocation_support';\nimport {LngLat} from '../../geo/lng_lat';\nimport {Marker} from '../marker';\n\nimport type {Map} from '../map';\nimport type {FitBoundsOptions} from '../camera';\nimport type {IControl} from './control';\nimport {LngLatBounds} from '../../geo/lng_lat_bounds';\n\n/**\n * The {@link GeolocateControl} options\n */\ntype GeolocateOptions = {\n /**\n * A Geolocation API [PositionOptions](https://developer.mozilla.org/en-US/docs/Web/API/PositionOptions) object.\n * @defaultValue `{enableHighAccuracy: false, timeout: 6000}`\n */\n positionOptions?: PositionOptions;\n /**\n * A options object to use when the map is panned and zoomed to the user's location. The default is to use a `maxZoom` of 15 to limit how far the map will zoom in for very accurate locations.\n */\n fitBoundsOptions?: FitBoundsOptions;\n /**\n * If `true` the Geolocate Control becomes a toggle button and when active the map will receive updates to the user's location as it changes.\n * @defaultValue false\n */\n trackUserLocation?: boolean;\n /**\n * By default, if showUserLocation is `true`, a transparent circle will be drawn around the user location indicating the accuracy (95% confidence level) of the user's location. Set to `false` to disable. Always disabled when showUserLocation is `false`.\n * @defaultValue true\n */\n showAccuracyCircle?: boolean;\n /**\n * By default a dot will be shown on the map at the user's location. Set to `false` to disable.\n * @defaultValue true\n */\n showUserLocation?: boolean;\n};\n\nconst defaultOptions: GeolocateOptions = {\n positionOptions: {\n enableHighAccuracy: false,\n maximumAge: 0,\n timeout: 6000 /* 6 sec */\n },\n fitBoundsOptions: {\n maxZoom: 15\n },\n trackUserLocation: false,\n showAccuracyCircle: true,\n showUserLocation: true\n};\n\nlet numberOfWatches = 0;\nlet noTimeout = false;\n\n/**\n * A `GeolocateControl` control provides a button that uses the browser's geolocation\n * API to locate the user on the map.\n *\n * Not all browsers support geolocation,\n * and some users may disable the feature. Geolocation support for modern\n * browsers including Chrome requires sites to be served over HTTPS. If\n * geolocation support is not available, the GeolocateControl will show\n * as disabled.\n *\n * The zoom level applied will depend on the accuracy of the geolocation provided by the device.\n *\n * The GeolocateControl has two modes. If `trackUserLocation` is `false` (default) the control acts as a button, which when pressed will set the map's camera to target the user location. If the user moves, the map won't update. This is most suited for the desktop. If `trackUserLocation` is `true` the control acts as a toggle button that when active the user's location is actively monitored for changes. In this mode the GeolocateControl has three interaction states:\n * * active - the map's camera automatically updates as the user's location changes, keeping the location dot in the center. Initial state and upon clicking the `GeolocateControl` button.\n * * passive - the user's location dot automatically updates, but the map's camera does not. Occurs upon the user initiating a map movement.\n * * disabled - occurs if Geolocation is not available, disabled or denied.\n *\n * These interaction states can't be controlled programmatically, rather they are set based on user interactions.\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * map.addControl(new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * }));\n * ```\n * @see [Locate the user](https://maplibre.org/maplibre-gl-js/docs/examples/locate-user/)\n *\n * ### Events\n *\n * @event `trackuserlocationend` - Fired when the Geolocate Control changes to the background state, which happens when a user changes the camera during an active position lock. This only applies when trackUserLocation is true. In the background state, the dot on the map will update with location updates but the camera will not.\n *\n * @event `trackuserlocationstart` - Fired when the Geolocate Control changes to the active lock state, which happens either upon first obtaining a successful Geolocation API position for the user (a geolocate event will follow), or the user clicks the geolocate button when in the background state which uses the last known position to recenter the map and enter active lock state (no geolocate event will follow unless the users's location changes).\n *\n * @event `geolocate` - Fired on each Geolocation API position update which returned as success.\n * `data` - The returned [Position](https://developer.mozilla.org/en-US/docs/Web/API/Position) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition).\n *\n * @event `error` - Fired on each Geolocation API position update which returned as an error.\n * `data` - The returned [PositionError](https://developer.mozilla.org/en-US/docs/Web/API/PositionError) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition).\n *\n * @event `outofmaxbounds` Fired on each Geolocation API position update which returned as success but user position is out of map maxBounds.\n * `data` - The returned [Position](https://developer.mozilla.org/en-US/docs/Web/API/Position) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition).\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when a trackuserlocationend event occurs.\n * geolocate.on('trackuserlocationend', function() {\n * console.log('A trackuserlocationend event has occurred.')\n * });\n * ```\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when a trackuserlocationstart event occurs.\n * geolocate.on('trackuserlocationstart', function() {\n * console.log('A trackuserlocationstart event has occurred.')\n * });\n * ```\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when a geolocate event occurs.\n * geolocate.on('geolocate', function() {\n * console.log('A geolocate event has occurred.')\n * });\n * ```\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when an error event occurs.\n * geolocate.on('error', function() {\n * console.log('An error event has occurred.')\n * });\n * ```\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when an outofmaxbounds event occurs.\n * geolocate.on('outofmaxbounds', function() {\n * console.log('An outofmaxbounds event has occurred.')\n * });\n * ```\n */\nexport class GeolocateControl extends Evented implements IControl {\n _map: Map;\n options: GeolocateOptions;\n _container: HTMLElement;\n _dotElement: HTMLElement;\n _circleElement: HTMLElement;\n _geolocateButton: HTMLButtonElement;\n _geolocationWatchID: number;\n _timeoutId: ReturnType;\n /* Geolocate Control Watch States\n * This is the private state of the control.\n *\n * OFF\n * off/inactive\n * WAITING_ACTIVE\n * Geolocate Control was clicked but still waiting for Geolocation API response with user location\n * ACTIVE_LOCK\n * Showing the user location as a dot AND tracking the camera to be fixed to their location. If their location changes the map moves to follow.\n * ACTIVE_ERROR\n * There was en error from the Geolocation API while trying to show and track the user location.\n * BACKGROUND\n * Showing the user location as a dot but the camera doesn't follow their location as it changes.\n * BACKGROUND_ERROR\n * There was an error from the Geolocation API while trying to show (but not track) the user location.\n */\n _watchState: 'OFF' | 'ACTIVE_LOCK' | 'WAITING_ACTIVE' | 'ACTIVE_ERROR' | 'BACKGROUND' | 'BACKGROUND_ERROR';\n _lastKnownPosition: any;\n _userLocationDotMarker: Marker;\n _accuracyCircleMarker: Marker;\n _accuracy: number;\n _setup: boolean; // set to true once the control has been setup\n\n constructor(options: GeolocateOptions) {\n super();\n this.options = extend({}, defaultOptions, options);\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group');\n checkGeolocationSupport(this._setupUI);\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n // clear the geolocation watch if exists\n if (this._geolocationWatchID !== undefined) {\n window.navigator.geolocation.clearWatch(this._geolocationWatchID);\n this._geolocationWatchID = undefined;\n }\n\n // clear the markers from the map\n if (this.options.showUserLocation && this._userLocationDotMarker) {\n this._userLocationDotMarker.remove();\n }\n if (this.options.showAccuracyCircle && this._accuracyCircleMarker) {\n this._accuracyCircleMarker.remove();\n }\n\n DOM.remove(this._container);\n this._map.off('zoom', this._onZoom);\n this._map = undefined;\n numberOfWatches = 0;\n noTimeout = false;\n }\n\n /**\n * Check if the Geolocation API Position is outside the map's maxbounds.\n *\n * @param position - the Geolocation API Position\n * @returns `true` if position is outside the map's maxbounds, otherwise returns `false`.\n */\n _isOutOfMapMaxBounds(position: GeolocationPosition) {\n const bounds = this._map.getMaxBounds();\n const coordinates = position.coords;\n\n return bounds && (\n coordinates.longitude < bounds.getWest() ||\n coordinates.longitude > bounds.getEast() ||\n coordinates.latitude < bounds.getSouth() ||\n coordinates.latitude > bounds.getNorth()\n );\n }\n\n _setErrorState() {\n switch (this._watchState) {\n case 'WAITING_ACTIVE':\n this._watchState = 'ACTIVE_ERROR';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active-error');\n break;\n case 'ACTIVE_LOCK':\n this._watchState = 'ACTIVE_ERROR';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active-error');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting');\n // turn marker grey\n break;\n case 'BACKGROUND':\n this._watchState = 'BACKGROUND_ERROR';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background-error');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting');\n // turn marker grey\n break;\n case 'ACTIVE_ERROR':\n break;\n default:\n throw new Error(`Unexpected watchState ${this._watchState}`);\n }\n }\n\n /**\n * When the Geolocation API returns a new location, update the GeolocateControl.\n *\n * @param position - the Geolocation API Position\n */\n _onSuccess = (position: GeolocationPosition) => {\n if (!this._map) {\n // control has since been removed\n return;\n }\n\n if (this._isOutOfMapMaxBounds(position)) {\n this._setErrorState();\n\n this.fire(new Event('outofmaxbounds', position));\n this._updateMarker();\n this._finish();\n\n return;\n }\n\n if (this.options.trackUserLocation) {\n // keep a record of the position so that if the state is BACKGROUND and the user\n // clicks the button, we can move to ACTIVE_LOCK immediately without waiting for\n // watchPosition to trigger _onSuccess\n this._lastKnownPosition = position;\n\n switch (this._watchState) {\n case 'WAITING_ACTIVE':\n case 'ACTIVE_LOCK':\n case 'ACTIVE_ERROR':\n this._watchState = 'ACTIVE_LOCK';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active');\n break;\n case 'BACKGROUND':\n case 'BACKGROUND_ERROR':\n this._watchState = 'BACKGROUND';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background');\n break;\n default:\n throw new Error(`Unexpected watchState ${this._watchState}`);\n }\n }\n\n // if showUserLocation and the watch state isn't off then update the marker location\n if (this.options.showUserLocation && this._watchState !== 'OFF') {\n this._updateMarker(position);\n }\n\n // if in normal mode (not watch mode), or if in watch mode and the state is active watch\n // then update the camera\n if (!this.options.trackUserLocation || this._watchState === 'ACTIVE_LOCK') {\n this._updateCamera(position);\n }\n\n if (this.options.showUserLocation) {\n this._dotElement.classList.remove('maplibregl-user-location-dot-stale');\n }\n\n this.fire(new Event('geolocate', position));\n this._finish();\n };\n\n /**\n * Update the camera location to center on the current position\n *\n * @param position - the Geolocation API Position\n */\n _updateCamera = (position: GeolocationPosition) => {\n const center = new LngLat(position.coords.longitude, position.coords.latitude);\n const radius = position.coords.accuracy;\n const bearing = this._map.getBearing();\n const options = extend({bearing}, this.options.fitBoundsOptions);\n const newBounds = LngLatBounds.fromLngLat(center, radius);\n\n this._map.fitBounds(newBounds, options, {\n geolocateSource: true // tag this camera change so it won't cause the control to change to background state\n });\n };\n\n /**\n * Update the user location dot Marker to the current position\n *\n * @param position - the Geolocation API Position\n */\n _updateMarker = (position?: GeolocationPosition | null) => {\n if (position) {\n const center = new LngLat(position.coords.longitude, position.coords.latitude);\n this._accuracyCircleMarker.setLngLat(center).addTo(this._map);\n this._userLocationDotMarker.setLngLat(center).addTo(this._map);\n this._accuracy = position.coords.accuracy;\n if (this.options.showUserLocation && this.options.showAccuracyCircle) {\n this._updateCircleRadius();\n }\n } else {\n this._userLocationDotMarker.remove();\n this._accuracyCircleMarker.remove();\n }\n };\n\n _updateCircleRadius() {\n const bounds = this._map.getBounds();\n const southEastPoint = bounds.getSouthEast();\n const northEastPoint = bounds.getNorthEast();\n const mapHeightInMeters = southEastPoint.distanceTo(northEastPoint);\n const mapHeightInPixels = this._map._container.clientHeight;\n const circleDiameter = Math.ceil(2 * (this._accuracy / (mapHeightInMeters / mapHeightInPixels)));\n this._circleElement.style.width = `${circleDiameter}px`;\n this._circleElement.style.height = `${circleDiameter}px`;\n }\n\n _onZoom = () => {\n if (this.options.showUserLocation && this.options.showAccuracyCircle) {\n this._updateCircleRadius();\n }\n };\n\n _onError = (error: GeolocationPositionError) => {\n if (!this._map) {\n // control has since been removed\n return;\n }\n\n if (this.options.trackUserLocation) {\n if (error.code === 1) {\n // PERMISSION_DENIED\n this._watchState = 'OFF';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error');\n this._geolocateButton.disabled = true;\n const title = this._map._getUIString('GeolocateControl.LocationNotAvailable');\n this._geolocateButton.title = title;\n this._geolocateButton.setAttribute('aria-label', title);\n\n if (this._geolocationWatchID !== undefined) {\n this._clearWatch();\n }\n } else if (error.code === 3 && noTimeout) {\n // this represents a forced error state\n // this was triggered to force immediate geolocation when a watch is already present\n // see https://github.com/mapbox/mapbox-gl-js/issues/8214\n // and https://w3c.github.io/geolocation-api/#example-5-forcing-the-user-agent-to-return-a-fresh-cached-position\n return;\n } else {\n this._setErrorState();\n }\n }\n\n if (this._watchState !== 'OFF' && this.options.showUserLocation) {\n this._dotElement.classList.add('maplibregl-user-location-dot-stale');\n }\n\n this.fire(new Event('error', error));\n\n this._finish();\n };\n\n _finish = () => {\n if (this._timeoutId) { clearTimeout(this._timeoutId); }\n this._timeoutId = undefined;\n };\n\n _setupUI = (supported: boolean) => {\n // this method is called asynchronously during onAdd\n // the control could have been removed before reaching here\n if (!this._map) {\n return;\n }\n\n this._container.addEventListener('contextmenu', (e: MouseEvent) => e.preventDefault());\n this._geolocateButton = DOM.create('button', 'maplibregl-ctrl-geolocate', this._container);\n DOM.create('span', 'maplibregl-ctrl-icon', this._geolocateButton).setAttribute('aria-hidden', 'true');\n this._geolocateButton.type = 'button';\n\n if (supported === false) {\n warnOnce('Geolocation support is not available so the GeolocateControl will be disabled.');\n const title = this._map._getUIString('GeolocateControl.LocationNotAvailable');\n this._geolocateButton.disabled = true;\n this._geolocateButton.title = title;\n this._geolocateButton.setAttribute('aria-label', title);\n } else {\n const title = this._map._getUIString('GeolocateControl.FindMyLocation');\n this._geolocateButton.title = title;\n this._geolocateButton.setAttribute('aria-label', title);\n }\n\n if (this.options.trackUserLocation) {\n this._geolocateButton.setAttribute('aria-pressed', 'false');\n this._watchState = 'OFF';\n }\n\n // when showUserLocation is enabled, keep the Geolocate button disabled until the device location marker is setup on the map\n if (this.options.showUserLocation) {\n this._dotElement = DOM.create('div', 'maplibregl-user-location-dot');\n\n this._userLocationDotMarker = new Marker({element: this._dotElement});\n\n this._circleElement = DOM.create('div', 'maplibregl-user-location-accuracy-circle');\n this._accuracyCircleMarker = new Marker({element: this._circleElement, pitchAlignment: 'map'});\n\n if (this.options.trackUserLocation) this._watchState = 'OFF';\n\n this._map.on('zoom', this._onZoom);\n }\n\n this._geolocateButton.addEventListener('click',\n this.trigger.bind(this));\n\n this._setup = true;\n\n // when the camera is changed (and it's not as a result of the Geolocation Control) change\n // the watch mode to background watch, so that the marker is updated but not the camera.\n if (this.options.trackUserLocation) {\n this._map.on('movestart', (event: any) => {\n const fromResize = event.originalEvent && event.originalEvent.type === 'resize';\n if (!event.geolocateSource && this._watchState === 'ACTIVE_LOCK' && !fromResize) {\n this._watchState = 'BACKGROUND';\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n\n this.fire(new Event('trackuserlocationend'));\n }\n });\n }\n };\n\n /**\n * Programmatically request and move the map to the user's location.\n *\n * @returns `false` if called before control was added to a map, otherwise returns `true`.\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * map.on('load', function() {\n * geolocate.trigger();\n * });\n * ```\n */\n trigger(): boolean {\n if (!this._setup) {\n warnOnce('Geolocate control triggered before added to a map');\n return false;\n }\n if (this.options.trackUserLocation) {\n // update watchState and do any outgoing state cleanup\n switch (this._watchState) {\n case 'OFF':\n // turn on the Geolocate Control\n this._watchState = 'WAITING_ACTIVE';\n\n this.fire(new Event('trackuserlocationstart'));\n break;\n case 'WAITING_ACTIVE':\n case 'ACTIVE_LOCK':\n case 'ACTIVE_ERROR':\n case 'BACKGROUND_ERROR':\n // turn off the Geolocate Control\n numberOfWatches--;\n noTimeout = false;\n this._watchState = 'OFF';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error');\n\n this.fire(new Event('trackuserlocationend'));\n break;\n case 'BACKGROUND':\n this._watchState = 'ACTIVE_LOCK';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background');\n // set camera to last known location\n if (this._lastKnownPosition) this._updateCamera(this._lastKnownPosition);\n\n this.fire(new Event('trackuserlocationstart'));\n break;\n default:\n throw new Error(`Unexpected watchState ${this._watchState}`);\n }\n\n // incoming state setup\n switch (this._watchState) {\n case 'WAITING_ACTIVE':\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active');\n break;\n case 'ACTIVE_LOCK':\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active');\n break;\n case 'OFF':\n break;\n default:\n throw new Error(`Unexpected watchState ${this._watchState}`);\n }\n\n // manage geolocation.watchPosition / geolocation.clearWatch\n if (this._watchState === 'OFF' && this._geolocationWatchID !== undefined) {\n // clear watchPosition as we've changed to an OFF state\n this._clearWatch();\n } else if (this._geolocationWatchID === undefined) {\n // enable watchPosition since watchState is not OFF and there is no watchPosition already running\n\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.setAttribute('aria-pressed', 'true');\n\n numberOfWatches++;\n let positionOptions;\n if (numberOfWatches > 1) {\n positionOptions = {maximumAge: 600000, timeout: 0};\n noTimeout = true;\n } else {\n positionOptions = this.options.positionOptions;\n noTimeout = false;\n }\n\n this._geolocationWatchID = window.navigator.geolocation.watchPosition(\n this._onSuccess, this._onError, positionOptions);\n }\n } else {\n window.navigator.geolocation.getCurrentPosition(\n this._onSuccess, this._onError, this.options.positionOptions);\n\n // This timeout ensures that we still call finish() even if\n // the user declines to share their location in Firefox\n this._timeoutId = setTimeout(this._finish, 10000 /* 10sec */);\n }\n\n return true;\n }\n\n _clearWatch() {\n window.navigator.geolocation.clearWatch(this._geolocationWatchID);\n\n this._geolocationWatchID = undefined;\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.setAttribute('aria-pressed', 'false');\n\n if (this.options.showUserLocation) {\n this._updateMarker(null);\n }\n }\n}\n\n","import {DOM} from '../../util/dom';\nimport {extend} from '../../util/util';\n\nimport type {Map} from '../map';\nimport type {ControlPosition, IControl} from './control';\n\n/**\n * The unit type for length to use for the {@link ScaleControl}\n */\nexport type Unit = 'imperial' | 'metric' | 'nautical';\n\n/**\n * The {@link ScaleControl} options object\n */\ntype ScaleOptions = {\n /**\n * The maximum length of the scale control in pixels.\n * @defaultValue 100\n */\n maxWidth?: number;\n /**\n * Unit of the distance (`'imperial'`, `'metric'` or `'nautical'`).\n * @defaultValue 'metric'\n */\n unit?: Unit;\n};\n\nconst defaultOptions: ScaleOptions = {\n maxWidth: 100,\n unit: 'metric'\n};\n\n/**\n * A `ScaleControl` control displays the ratio of a distance on the map to the corresponding distance on the ground.\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * let scale = new maplibregl.ScaleControl({\n * maxWidth: 80,\n * unit: 'imperial'\n * });\n * map.addControl(scale);\n *\n * scale.setUnit('metric');\n * ```\n */\nexport class ScaleControl implements IControl {\n _map: Map;\n _container: HTMLElement;\n options: ScaleOptions;\n\n constructor(options: ScaleOptions) {\n this.options = extend({}, defaultOptions, options);\n }\n\n getDefaultPosition(): ControlPosition {\n return 'bottom-left';\n }\n\n _onMove = () => {\n updateScale(this._map, this._container, this.options);\n };\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-scale', map.getContainer());\n\n this._map.on('move', this._onMove);\n this._onMove();\n\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._container);\n this._map.off('move', this._onMove);\n this._map = undefined;\n }\n\n /**\n * Set the scale's unit of the distance\n *\n * @param unit - Unit of the distance (`'imperial'`, `'metric'` or `'nautical'`).\n */\n setUnit = (unit: Unit) => {\n this.options.unit = unit;\n updateScale(this._map, this._container, this.options);\n };\n}\n\nfunction updateScale(map, container, options) {\n // A horizontal scale is imagined to be present at center of the map\n // container with maximum length (Default) as 100px.\n // Using spherical law of cosines approximation, the real distance is\n // found between the two coordinates.\n const maxWidth = options && options.maxWidth || 100;\n\n const y = map._container.clientHeight / 2;\n const left = map.unproject([0, y]);\n const right = map.unproject([maxWidth, y]);\n const maxMeters = left.distanceTo(right);\n // The real distance corresponding to 100px scale length is rounded off to\n // near pretty number and the scale length for the same is found out.\n // Default unit of the scale is based on User's locale.\n if (options && options.unit === 'imperial') {\n const maxFeet = 3.2808 * maxMeters;\n if (maxFeet > 5280) {\n const maxMiles = maxFeet / 5280;\n setScale(container, maxWidth, maxMiles, map._getUIString('ScaleControl.Miles'));\n } else {\n setScale(container, maxWidth, maxFeet, map._getUIString('ScaleControl.Feet'));\n }\n } else if (options && options.unit === 'nautical') {\n const maxNauticals = maxMeters / 1852;\n setScale(container, maxWidth, maxNauticals, map._getUIString('ScaleControl.NauticalMiles'));\n } else if (maxMeters >= 1000) {\n setScale(container, maxWidth, maxMeters / 1000, map._getUIString('ScaleControl.Kilometers'));\n } else {\n setScale(container, maxWidth, maxMeters, map._getUIString('ScaleControl.Meters'));\n }\n}\n\nfunction setScale(container, maxWidth, maxDistance, unit) {\n const distance = getRoundNum(maxDistance);\n const ratio = distance / maxDistance;\n container.style.width = `${maxWidth * ratio}px`;\n container.innerHTML = `${distance} ${unit}`;\n}\n\nfunction getDecimalRoundNum(d) {\n const multiplier = Math.pow(10, Math.ceil(-Math.log(d) / Math.LN10));\n return Math.round(d * multiplier) / multiplier;\n}\n\nfunction getRoundNum(num) {\n const pow10 = Math.pow(10, (`${Math.floor(num)}`).length - 1);\n let d = num / pow10;\n\n d = d >= 10 ? 10 :\n d >= 5 ? 5 :\n d >= 3 ? 3 :\n d >= 2 ? 2 :\n d >= 1 ? 1 : getDecimalRoundNum(d);\n\n return pow10 * d;\n}\n","import {extend} from '../util/util';\nimport {Event, Evented} from '../util/evented';\nimport {MapMouseEvent} from '../ui/events';\nimport {DOM} from '../util/dom';\nimport {LngLat} from '../geo/lng_lat';\nimport Point from '@mapbox/point-geometry';\nimport {smartWrap} from '../util/smart_wrap';\nimport {anchorTranslate, applyAnchorClass} from './anchor';\n\nimport type {PositionAnchor} from './anchor';\nimport type {Map} from './map';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport type {PointLike} from './camera';\n\nconst defaultOptions = {\n closeButton: true,\n closeOnClick: true,\n focusAfterOpen: true,\n className: '',\n maxWidth: '240px'\n};\n\n/**\n * A pixel offset specified as:\n * - a single number specifying a distance from the location\n * - a {@link PointLike} specifying a constant offset\n * - an object of {@link Point}s specifying an offset for each anchor position\n * Negative offsets indicate left and up.\n */\nexport type Offset = number | PointLike | {\n [_ in PositionAnchor]: PointLike;\n};\n\nexport type PopupOptions = {\n /**\n * If `true`, a close button will appear in the top right corner of the popup.\n * @defaultValue true\n */\n closeButton?: boolean;\n /**\n * If `true`, the popup will closed when the map is clicked.\n * @defaultValue true\n */\n closeOnClick?: boolean;\n /**\n * If `true`, the popup will closed when the map moves.\n * @defaultValue false\n */\n closeOnMove?: boolean;\n /**\n * If `true`, the popup will try to focus the first focusable element inside the popup.\n * @defaultValue true\n */\n focusAfterOpen?: boolean;\n /**\n * A string indicating the part of the Popup that should\n * be positioned closest to the coordinate set via {@link Popup#setLngLat}.\n * Options are `'center'`, `'top'`, `'bottom'`, `'left'`, `'right'`, `'top-left'`,\n * `'top-right'`, `'bottom-left'`, and `'bottom-right'`. If unset the anchor will be\n * dynamically set to ensure the popup falls within the map container with a preference\n * for `'bottom'`.\n */\n anchor?: PositionAnchor;\n /**\n * A pixel offset applied to the popup's location\n */\n offset?: Offset;\n /**\n * Space-separated CSS class names to add to popup container\n */\n className?: string;\n /**\n * A string that sets the CSS property of the popup's maximum width, eg `'300px'`.\n * To ensure the popup resizes to fit its content, set this property to `'none'`.\n * Available values can be found here: https://developer.mozilla.org/en-US/docs/Web/CSS/max-width\n * @defaultValue '240px'\n */\n maxWidth?: string;\n};\n\nconst focusQuerySelector = [\n 'a[href]',\n '[tabindex]:not([tabindex=\\'-1\\'])',\n '[contenteditable]:not([contenteditable=\\'false\\'])',\n 'button:not([disabled])',\n 'input:not([disabled])',\n 'select:not([disabled])',\n 'textarea:not([disabled])',\n].join(', ');\n\n/**\n * A popup component.\n *\n * @group Markers and Controls\n *\n *\n * @example\n * Create a popup\n * ```ts\n * let popup = new maplibregl.Popup();\n * // Set an event listener that will fire\n * // any time the popup is opened\n * popup.on('open', function(){\n * console.log('popup was opened');\n * });\n * ```\n *\n * @example\n * Create a popup\n * ```ts\n * let popup = new maplibregl.Popup();\n * // Set an event listener that will fire\n * // any time the popup is closed\n * popup.on('close', function(){\n * console.log('popup was closed');\n * });\n * ```\n *\n * @example\n * ```ts\n * let markerHeight = 50, markerRadius = 10, linearOffset = 25;\n * let popupOffsets = {\n * 'top': [0, 0],\n * 'top-left': [0,0],\n * 'top-right': [0,0],\n * 'bottom': [0, -markerHeight],\n * 'bottom-left': [linearOffset, (markerHeight - markerRadius + linearOffset) * -1],\n * 'bottom-right': [-linearOffset, (markerHeight - markerRadius + linearOffset) * -1],\n * 'left': [markerRadius, (markerHeight - markerRadius) * -1],\n * 'right': [-markerRadius, (markerHeight - markerRadius) * -1]\n * };\n * let popup = new maplibregl.Popup({offset: popupOffsets, className: 'my-class'})\n * .setLngLat(e.lngLat)\n * .setHTML(\"

Hello World!

\")\n * .setMaxWidth(\"300px\")\n * .addTo(map);\n * ```\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/)\n *\n * ### Events\n *\n * @event `open` Fired when the popup is opened manually or programmatically. `popup` object that was opened\n *\n * @event `close` Fired when the popup is closed manually or programmatically. `popup` object that was closed\n */\nexport class Popup extends Evented {\n _map: Map;\n options: PopupOptions;\n _content: HTMLElement;\n _container: HTMLElement;\n _closeButton: HTMLButtonElement;\n _tip: HTMLElement;\n _lngLat: LngLat;\n _trackPointer: boolean;\n _pos: Point;\n\n constructor(options?: PopupOptions) {\n super();\n this.options = extend(Object.create(defaultOptions), options);\n }\n\n /**\n * Adds the popup to a map.\n *\n * @param map - The MapLibre GL JS map to add the popup to.\n * @returns `this`\n * @example\n * ```ts\n * new maplibregl.Popup()\n * .setLngLat([0, 0])\n * .setHTML(\"

Null Island

\")\n * .addTo(map);\n * ```\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n * @see [Show polygon information on click](https://maplibre.org/maplibre-gl-js/docs/examples/polygon-popup-on-click/)\n */\n addTo(map: Map): this {\n if (this._map) this.remove();\n\n this._map = map;\n if (this.options.closeOnClick) {\n this._map.on('click', this._onClose);\n }\n\n if (this.options.closeOnMove) {\n this._map.on('move', this._onClose);\n }\n\n this._map.on('remove', this.remove);\n this._update();\n this._focusFirstElement();\n\n if (this._trackPointer) {\n this._map.on('mousemove', this._onMouseMove);\n this._map.on('mouseup', this._onMouseUp);\n if (this._container) {\n this._container.classList.add('maplibregl-popup-track-pointer');\n }\n this._map._canvasContainer.classList.add('maplibregl-track-pointer');\n } else {\n this._map.on('move', this._update);\n }\n\n this.fire(new Event('open'));\n\n return this;\n }\n\n /**\n * @returns `true` if the popup is open, `false` if it is closed.\n */\n isOpen() {\n return !!this._map;\n }\n\n /**\n * Removes the popup from the map it has been added to.\n *\n * @example\n * ```ts\n * let popup = new maplibregl.Popup().addTo(map);\n * popup.remove();\n * ```\n * @returns `this`\n */\n remove = (): this => {\n if (this._content) {\n DOM.remove(this._content);\n }\n\n if (this._container) {\n DOM.remove(this._container);\n delete this._container;\n }\n\n if (this._map) {\n this._map.off('move', this._update);\n this._map.off('move', this._onClose);\n this._map.off('click', this._onClose);\n this._map.off('remove', this.remove);\n this._map.off('mousemove', this._onMouseMove);\n this._map.off('mouseup', this._onMouseUp);\n this._map.off('drag', this._onDrag);\n delete this._map;\n }\n\n this.fire(new Event('close'));\n\n return this;\n };\n\n /**\n * Returns the geographical location of the popup's anchor.\n *\n * The longitude of the result may differ by a multiple of 360 degrees from the longitude previously\n * set by `setLngLat` because `Popup` wraps the anchor longitude across copies of the world to keep\n * the popup on screen.\n *\n * @returns The geographical location of the popup's anchor.\n */\n getLngLat(): LngLat {\n return this._lngLat;\n }\n\n /**\n * Sets the geographical location of the popup's anchor, and moves the popup to it. Replaces trackPointer() behavior.\n *\n * @param lnglat - The geographical location to set as the popup's anchor.\n * @returns `this`\n */\n setLngLat(lnglat: LngLatLike): this {\n this._lngLat = LngLat.convert(lnglat);\n this._pos = null;\n\n this._trackPointer = false;\n\n this._update();\n\n if (this._map) {\n this._map.on('move', this._update);\n this._map.off('mousemove', this._onMouseMove);\n if (this._container) {\n this._container.classList.remove('maplibregl-popup-track-pointer');\n }\n this._map._canvasContainer.classList.remove('maplibregl-track-pointer');\n }\n\n return this;\n }\n\n /**\n * Tracks the popup anchor to the cursor position on screens with a pointer device (it will be hidden on touchscreens). Replaces the `setLngLat` behavior.\n * For most use cases, set `closeOnClick` and `closeButton` to `false`.\n * @example\n * ```ts\n * let popup = new maplibregl.Popup({ closeOnClick: false, closeButton: false })\n * .setHTML(\"

Hello World!

\")\n * .trackPointer()\n * .addTo(map);\n * ```\n * @returns `this`\n */\n trackPointer(): this {\n this._trackPointer = true;\n this._pos = null;\n this._update();\n if (this._map) {\n this._map.off('move', this._update);\n this._map.on('mousemove', this._onMouseMove);\n this._map.on('drag', this._onDrag);\n if (this._container) {\n this._container.classList.add('maplibregl-popup-track-pointer');\n }\n this._map._canvasContainer.classList.add('maplibregl-track-pointer');\n }\n\n return this;\n\n }\n\n /**\n * Returns the `Popup`'s HTML element.\n * @example\n * Change the `Popup` element's font size\n * ```ts\n * let popup = new maplibregl.Popup()\n * .setLngLat([-96, 37.8])\n * .setHTML(\"

Hello World!

\")\n * .addTo(map);\n * let popupElem = popup.getElement();\n * popupElem.style.fontSize = \"25px\";\n * ```\n * @returns element\n */\n getElement(): HTMLElement {\n return this._container;\n }\n\n /**\n * Sets the popup's content to a string of text.\n *\n * This function creates a [Text](https://developer.mozilla.org/en-US/docs/Web/API/Text) node in the DOM,\n * so it cannot insert raw HTML. Use this method for security against XSS\n * if the popup content is user-provided.\n *\n * @param text - Textual content for the popup.\n * @returns `this`\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * .setLngLat(e.lngLat)\n * .setText('Hello, world!')\n * .addTo(map);\n * ```\n */\n setText(text: string): this {\n return this.setDOMContent(document.createTextNode(text));\n }\n\n /**\n * Sets the popup's content to the HTML provided as a string.\n *\n * This method does not perform HTML filtering or sanitization, and must be\n * used only with trusted content. Consider {@link Popup#setText} if\n * the content is an untrusted text string.\n *\n * @param html - A string representing HTML content for the popup.\n * @returns `this`\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * .setLngLat(e.lngLat)\n * .setHTML(\"

Hello World!

\")\n * .addTo(map);\n * ```\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/)\n */\n setHTML(html: string): this {\n const frag = document.createDocumentFragment();\n const temp = document.createElement('body');\n let child;\n temp.innerHTML = html;\n while (true) {\n child = temp.firstChild;\n if (!child) break;\n frag.appendChild(child);\n }\n\n return this.setDOMContent(frag);\n }\n\n /**\n * Returns the popup's maximum width.\n *\n * @returns The maximum width of the popup.\n */\n getMaxWidth(): string {\n return this._container?.style.maxWidth;\n }\n\n /**\n * Sets the popup's maximum width. This is setting the CSS property `max-width`.\n * Available values can be found here: https://developer.mozilla.org/en-US/docs/Web/CSS/max-width\n *\n * @param maxWidth - A string representing the value for the maximum width.\n * @returns `this`\n */\n setMaxWidth(maxWidth: string): this {\n this.options.maxWidth = maxWidth;\n this._update();\n return this;\n }\n\n /**\n * Sets the popup's content to the element provided as a DOM node.\n *\n * @param htmlNode - A DOM node to be used as content for the popup.\n * @returns `this`\n * @example\n * Create an element with the popup content\n * ```ts\n * let div = document.createElement('div');\n * div.innerHTML = 'Hello, world!';\n * let popup = new maplibregl.Popup()\n * .setLngLat(e.lngLat)\n * .setDOMContent(div)\n * .addTo(map);\n * ```\n */\n setDOMContent(htmlNode: Node): this {\n if (this._content) {\n // Clear out children first.\n while (this._content.hasChildNodes()) {\n if (this._content.firstChild) {\n this._content.removeChild(this._content.firstChild);\n }\n }\n } else {\n this._content = DOM.create('div', 'maplibregl-popup-content', this._container);\n }\n\n // The close button should be the last tabbable element inside the popup for a good keyboard UX.\n this._content.appendChild(htmlNode);\n this._createCloseButton();\n this._update();\n this._focusFirstElement();\n return this;\n }\n\n /**\n * Adds a CSS class to the popup container element.\n *\n * @param className - Non-empty string with CSS class name to add to popup container\n *\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * popup.addClassName('some-class')\n * ```\n */\n addClassName(className: string) {\n if (this._container) {\n this._container.classList.add(className);\n }\n }\n\n /**\n * Removes a CSS class from the popup container element.\n *\n * @param className - Non-empty string with CSS class name to remove from popup container\n *\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * popup.removeClassName('some-class')\n * ```\n */\n removeClassName(className: string) {\n if (this._container) {\n this._container.classList.remove(className);\n }\n }\n\n /**\n * Sets the popup's offset.\n *\n * @param offset - Sets the popup's offset.\n * @returns `this`\n */\n setOffset (offset?: Offset): this {\n this.options.offset = offset;\n this._update();\n return this;\n }\n\n /**\n * Add or remove the given CSS class on the popup container, depending on whether the container currently has that class.\n *\n * @param className - Non-empty string with CSS class name to add/remove\n *\n * @returns if the class was removed return false, if class was added, then return true, undefined if there is no container\n *\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * popup.toggleClassName('toggleClass')\n * ```\n */\n toggleClassName(className: string): boolean | undefined {\n if (this._container) {\n return this._container.classList.toggle(className);\n }\n }\n\n _createCloseButton() {\n if (this.options.closeButton) {\n this._closeButton = DOM.create('button', 'maplibregl-popup-close-button', this._content);\n this._closeButton.type = 'button';\n this._closeButton.setAttribute('aria-label', 'Close popup');\n this._closeButton.innerHTML = '×';\n this._closeButton.addEventListener('click', this._onClose);\n }\n }\n\n _onMouseUp = (event: MapMouseEvent) => {\n this._update(event.point);\n };\n\n _onMouseMove = (event: MapMouseEvent) => {\n this._update(event.point);\n };\n\n _onDrag = (event: MapMouseEvent) => {\n this._update(event.point);\n };\n\n _update = (cursor?: Point) => {\n const hasPosition = this._lngLat || this._trackPointer;\n\n if (!this._map || !hasPosition || !this._content) { return; }\n\n if (!this._container) {\n this._container = DOM.create('div', 'maplibregl-popup', this._map.getContainer());\n this._tip = DOM.create('div', 'maplibregl-popup-tip', this._container);\n this._container.appendChild(this._content);\n if (this.options.className) {\n for (const name of this.options.className.split(' ')) {\n this._container.classList.add(name);\n }\n }\n\n if (this._trackPointer) {\n this._container.classList.add('maplibregl-popup-track-pointer');\n }\n }\n\n if (this.options.maxWidth && this._container.style.maxWidth !== this.options.maxWidth) {\n this._container.style.maxWidth = this.options.maxWidth;\n }\n\n if (this._map.transform.renderWorldCopies && !this._trackPointer) {\n this._lngLat = smartWrap(this._lngLat, this._pos, this._map.transform);\n }\n\n if (this._trackPointer && !cursor) return;\n\n const pos = this._pos = this._trackPointer && cursor ? cursor : this._map.project(this._lngLat);\n\n let anchor = this.options.anchor;\n const offset = normalizeOffset(this.options.offset);\n\n if (!anchor) {\n const width = this._container.offsetWidth;\n const height = this._container.offsetHeight;\n let anchorComponents;\n\n if (pos.y + offset.bottom.y < height) {\n anchorComponents = ['top'];\n } else if (pos.y > this._map.transform.height - height) {\n anchorComponents = ['bottom'];\n } else {\n anchorComponents = [];\n }\n\n if (pos.x < width / 2) {\n anchorComponents.push('left');\n } else if (pos.x > this._map.transform.width - width / 2) {\n anchorComponents.push('right');\n }\n\n if (anchorComponents.length === 0) {\n anchor = 'bottom';\n } else {\n anchor = (anchorComponents.join('-') as any);\n }\n }\n\n const offsetedPos = pos.add(offset[anchor]).round();\n DOM.setTransform(this._container, `${anchorTranslate[anchor]} translate(${offsetedPos.x}px,${offsetedPos.y}px)`);\n applyAnchorClass(this._container, anchor, 'popup');\n };\n\n _focusFirstElement() {\n if (!this.options.focusAfterOpen || !this._container) return;\n\n const firstFocusable = this._container.querySelector(focusQuerySelector) as HTMLElement;\n\n if (firstFocusable) firstFocusable.focus();\n }\n\n _onClose = () => {\n this.remove();\n };\n}\n\nfunction normalizeOffset(offset?: Offset | null) {\n if (!offset) {\n return normalizeOffset(new Point(0, 0));\n\n } else if (typeof offset === 'number') {\n // input specifies a radius from which to calculate offsets at all positions\n const cornerOffset = Math.round(Math.abs(offset) / Math.SQRT2);\n return {\n 'center': new Point(0, 0),\n 'top': new Point(0, offset),\n 'top-left': new Point(cornerOffset, cornerOffset),\n 'top-right': new Point(-cornerOffset, cornerOffset),\n 'bottom': new Point(0, -offset),\n 'bottom-left': new Point(cornerOffset, -cornerOffset),\n 'bottom-right': new Point(-cornerOffset, -cornerOffset),\n 'left': new Point(offset, 0),\n 'right': new Point(-offset, 0)\n };\n\n } else if (offset instanceof Point || Array.isArray(offset)) {\n // input specifies a single offset to be applied to all positions\n const convertedOffset = Point.convert(offset);\n return {\n 'center': convertedOffset,\n 'top': convertedOffset,\n 'top-left': convertedOffset,\n 'top-right': convertedOffset,\n 'bottom': convertedOffset,\n 'bottom-left': convertedOffset,\n 'bottom-right': convertedOffset,\n 'left': convertedOffset,\n 'right': convertedOffset\n };\n\n } else {\n // input specifies an offset per position\n return {\n 'center': Point.convert(offset['center'] || [0, 0]),\n 'top': Point.convert(offset['top'] || [0, 0]),\n 'top-left': Point.convert(offset['top-left'] || [0, 0]),\n 'top-right': Point.convert(offset['top-right'] || [0, 0]),\n 'bottom': Point.convert(offset['bottom'] || [0, 0]),\n 'bottom-left': Point.convert(offset['bottom-left'] || [0, 0]),\n 'bottom-right': Point.convert(offset['bottom-right'] || [0, 0]),\n 'left': Point.convert(offset['left'] || [0, 0]),\n 'right': Point.convert(offset['right'] || [0, 0])\n };\n }\n}\n","import {extend} from './util';\n\n/**\n * This is a private namespace for utility functions that will get automatically stripped\n * out in production builds.\n */\nexport const Debug = {\n extend(dest: any, ...sources: Array): any {\n return extend(dest, ...sources);\n },\n\n run(fn: () => any) {\n fn();\n },\n\n logToElement(message: string, overwrite: boolean = false, id: string = 'log') {\n const el = window.document.getElementById(id);\n if (el) {\n if (overwrite) el.innerHTML = '';\n el.innerHTML += `
${message}`;\n }\n\n }\n};\n","import packageJSON from '../package.json' assert {type: 'json'};\nimport {Map} from './ui/map';\nimport {NavigationControl} from './ui/control/navigation_control';\nimport {GeolocateControl} from './ui/control/geolocate_control';\nimport {AttributionControl} from './ui/control/attribution_control';\nimport {LogoControl} from './ui/control/logo_control';\nimport {ScaleControl} from './ui/control/scale_control';\nimport {FullscreenControl} from './ui/control/fullscreen_control';\nimport {TerrainControl} from './ui/control/terrain_control';\nimport {Popup} from './ui/popup';\nimport {Marker} from './ui/marker';\nimport {Style} from './style/style';\nimport {LngLat} from './geo/lng_lat';\nimport {LngLatBounds} from './geo/lng_lat_bounds';\nimport Point from '@mapbox/point-geometry';\nimport {MercatorCoordinate} from './geo/mercator_coordinate';\nimport {Evented} from './util/evented';\nimport {config} from './util/config';\nimport {Debug} from './util/debug';\nimport {isSafari} from './util/util';\nimport {setRTLTextPlugin, getRTLTextPluginStatus} from './source/rtl_text_plugin';\nimport {WorkerPool} from './util/worker_pool';\nimport {prewarm, clearPrewarmedResources} from './util/global_worker_pool';\nimport {PerformanceUtils} from './util/performance';\nimport {AJAXError} from './util/ajax';\nimport type {RequestParameters, ResponseCallback} from './util/ajax';\nimport type {Cancelable} from './types/cancelable';\nimport {GeoJSONSource} from './source/geojson_source';\nimport {CanvasSource} from './source/canvas_source';\nimport {ImageSource} from './source/image_source';\nimport {RasterDEMTileSource} from './source/raster_dem_tile_source';\nimport {RasterTileSource} from './source/raster_tile_source';\nimport {VectorTileSource} from './source/vector_tile_source';\nimport {VideoSource} from './source/video_source';\n\nconst version = packageJSON.version;\n\nexport type * from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * `maplibregl` is the global object that allows configurations that are not specific to a map instance\n *\n * @group Main\n */\nclass MapLibreGL {\n static Map = Map;\n static NavigationControl = NavigationControl;\n static GeolocateControl = GeolocateControl;\n static AttributionControl = AttributionControl;\n static LogoControl = LogoControl;\n static ScaleControl = ScaleControl;\n static FullscreenControl = FullscreenControl;\n static TerrainControl = TerrainControl;\n static Popup = Popup;\n static Marker = Marker;\n static Style = Style;\n static LngLat = LngLat;\n static LngLatBounds = LngLatBounds;\n static Point = Point;\n static MercatorCoordinate = MercatorCoordinate;\n static Evented = Evented;\n static AJAXError = AJAXError;\n static config = config;\n static CanvasSource = CanvasSource;\n static GeoJSONSource = GeoJSONSource;\n static ImageSource = ImageSource;\n static RasterDEMTileSource = RasterDEMTileSource;\n static RasterTileSource = RasterTileSource;\n static VectorTileSource = VectorTileSource;\n static VideoSource = VideoSource;\n /**\n * Sets the map's [RTL text plugin](https://www.mapbox.com/mapbox-gl-js/plugins/#mapbox-gl-rtl-text).\n * Necessary for supporting the Arabic and Hebrew languages, which are written right-to-left.\n *\n * @param pluginURL - URL pointing to the Mapbox RTL text plugin source.\n * @param callback - Called with an error argument if there is an error.\n * @param lazy - If set to `true`, mapboxgl will defer loading the plugin until rtl text is encountered,\n * rtl text will then be rendered only after the plugin finishes loading.\n * @example\n * ```ts\n * maplibregl.setRTLTextPlugin('https://unpkg.com/@mapbox/mapbox-gl-rtl-text@0.2.3/mapbox-gl-rtl-text.js');\n * ```\n * @see [Add support for right-to-left scripts](https://maplibre.org/maplibre-gl-js/docs/examples/mapbox-gl-rtl-text/)\n */\n static setRTLTextPlugin = setRTLTextPlugin;\n /**\n * Gets the map's [RTL text plugin](https://www.mapbox.com/mapbox-gl-js/plugins/#mapbox-gl-rtl-text) status.\n * The status can be `unavailable` (i.e. not requested or removed), `loading`, `loaded` or `error`.\n * If the status is `loaded` and the plugin is requested again, an error will be thrown.\n *\n * @example\n * ```ts\n * const pluginStatus = maplibregl.getRTLTextPluginStatus();\n * ```\n */\n static getRTLTextPluginStatus = getRTLTextPluginStatus;\n /**\n * Initializes resources like WebWorkers that can be shared across maps to lower load\n * times in some situations. `maplibregl.workerUrl` and `maplibregl.workerCount`, if being\n * used, must be set before `prewarm()` is called to have an effect.\n *\n * By default, the lifecycle of these resources is managed automatically, and they are\n * lazily initialized when a Map is first created. By invoking `prewarm()`, these\n * resources will be created ahead of time, and will not be cleared when the last Map\n * is removed from the page. This allows them to be re-used by new Map instances that\n * are created later. They can be manually cleared by calling\n * `maplibregl.clearPrewarmedResources()`. This is only necessary if your web page remains\n * active but stops using maps altogether.\n *\n * This is primarily useful when using GL-JS maps in a single page app, wherein a user\n * would navigate between various views that can cause Map instances to constantly be\n * created and destroyed.\n *\n * @example\n * ```ts\n * maplibregl.prewarm()\n * ```\n */\n static prewarm = prewarm;\n /**\n * Clears up resources that have previously been created by `maplibregl.prewarm()`.\n * Note that this is typically not necessary. You should only call this function\n * if you expect the user of your app to not return to a Map view at any point\n * in your application.\n *\n * @example\n * ```ts\n * maplibregl.clearPrewarmedResources()\n * ```\n */\n static clearPrewarmedResources = clearPrewarmedResources;\n /**\n * Returns the package version of the library\n * @returns Package version of the library\n */\n static get version(): string {\n return version;\n }\n\n /**\n * Gets and sets the number of web workers instantiated on a page with GL JS maps.\n * By default, workerCount is 1 except for Safari browser where it is set to half the number of CPU cores (capped at 3).\n * Make sure to set this property before creating any map instances for it to have effect.\n *\n * @returns Number of workers currently configured.\n * @example\n * ```ts\n * maplibregl.workerCount = 2;\n * ```\n */\n static get workerCount(): number {\n return WorkerPool.workerCount;\n }\n\n static set workerCount(count: number) {\n WorkerPool.workerCount = count;\n }\n /**\n * Gets and sets the maximum number of images (raster tiles, sprites, icons) to load in parallel,\n * which affects performance in raster-heavy maps. 16 by default.\n *\n * @returns Number of parallel requests currently configured.\n * @example\n * ```ts\n * maplibregl.maxParallelImageRequests = 10;\n * ```\n */\n static get maxParallelImageRequests(): number {\n return config.MAX_PARALLEL_IMAGE_REQUESTS;\n }\n\n static set maxParallelImageRequests(numRequests: number) {\n config.MAX_PARALLEL_IMAGE_REQUESTS = numRequests;\n }\n\n static get workerUrl(): string {\n return config.WORKER_URL;\n }\n\n static set workerUrl(value: string) {\n config.WORKER_URL = value;\n }\n\n /**\n * Sets a custom load tile function that will be called when using a source that starts with a custom url schema.\n * The example below will be triggered for custom:// urls defined in the sources list in the style definitions.\n * The function passed will receive the request parameters and should call the callback with the resulting request,\n * for example a pbf vector tile, non-compressed, represented as ArrayBuffer.\n *\n * @param customProtocol - the protocol to hook, for example 'custom'\n * @param loadFn - the function to use when trying to fetch a tile specified by the customProtocol\n * @example\n * This will fetch a file using the fetch API (this is obviously a non interesting example...)\n * ```ts\n * maplibregl.addProtocol('custom', (params, callback) => {\n fetch(`https://${params.url.split(\"://\")[1]}`)\n .then(t => {\n if (t.status == 200) {\n t.arrayBuffer().then(arr => {\n callback(null, arr, null, null);\n });\n } else {\n callback(new Error(`Tile fetch error: ${t.statusText}`));\n }\n })\n .catch(e => {\n callback(new Error(e));\n });\n return { cancel: () => { } };\n });\n * // the following is an example of a way to return an error when trying to load a tile\n * maplibregl.addProtocol('custom2', (params, callback) => {\n * callback(new Error('someErrorMessage'));\n * return { cancel: () => { } };\n * });\n * ```\n */\n static addProtocol(customProtocol: string, loadFn: (requestParameters: RequestParameters, callback: ResponseCallback) => Cancelable) {\n config.REGISTERED_PROTOCOLS[customProtocol] = loadFn;\n }\n\n /**\n * Removes a previously added protocol\n *\n * @param customProtocol - the custom protocol to remove registration for\n * @example\n * ```ts\n * maplibregl.removeProtocol('custom');\n * ```\n */\n static removeProtocol(customProtocol: string) {\n delete config.REGISTERED_PROTOCOLS[customProtocol];\n }\n}\n\n//This gets automatically stripped out in production builds.\nDebug.extend(MapLibreGL, {isSafari, getPerformanceMetrics: PerformanceUtils.getPerformanceMetrics});\n\nexport default MapLibreGL;\n","import {DOM} from '../../util/dom';\n\nimport {warnOnce} from '../../util/util';\n\nimport {Event, Evented} from '../../util/evented';\nimport type {Map, GestureOptions} from '../map';\nimport type {IControl} from './control';\n\n/**\n * The {@link FullscreenControl} options\n */\ntype FullscreenOptions = {\n /**\n * `container` is the [compatible DOM element](https://developer.mozilla.org/en-US/docs/Web/API/Element/requestFullScreen#Compatible_elements) which should be made full screen. By default, the map container element will be made full screen.\n */\n container?: HTMLElement;\n};\n\n/**\n * A `FullscreenControl` control contains a button for toggling the map in and out of fullscreen mode.\n * When [requestFullscreen](https://developer.mozilla.org/en-US/docs/Web/API/Element/requestFullscreen) is not supported, fullscreen is handled via CSS properties.\n * The map's `cooperativeGestures` option is temporarily disabled while the map\n * is in fullscreen mode, and is restored when the map exist fullscreen mode.\n *\n * @group Markers and Controls\n * @param options - the full screen control options\n *\n * @example\n * ```ts\n * map.addControl(new maplibregl.FullscreenControl({container: document.querySelector('body')}));\n * ```\n * @see [View a fullscreen map](https://maplibre.org/maplibre-gl-js/docs/examples/fullscreen/)\n *\n * ### Events\n *\n * @event `fullscreenstart` - Fired when fullscreen mode has started\n *\n * @event `fullscreenend` - Fired when fullscreen mode has ended\n */\nexport class FullscreenControl extends Evented implements IControl {\n _map: Map;\n _controlContainer: HTMLElement;\n _fullscreen: boolean;\n _fullscreenchange: string;\n _fullscreenButton: HTMLButtonElement;\n _container: HTMLElement;\n _prevCooperativeGestures: boolean | GestureOptions;\n\n constructor(options: FullscreenOptions = {}) {\n super();\n this._fullscreen = false;\n\n if (options && options.container) {\n if (options.container instanceof HTMLElement) {\n this._container = options.container;\n } else {\n warnOnce('Full screen control \\'container\\' must be a DOM element.');\n }\n }\n\n if ('onfullscreenchange' in document) {\n this._fullscreenchange = 'fullscreenchange';\n } else if ('onmozfullscreenchange' in document) {\n this._fullscreenchange = 'mozfullscreenchange';\n } else if ('onwebkitfullscreenchange' in document) {\n this._fullscreenchange = 'webkitfullscreenchange';\n } else if ('onmsfullscreenchange' in document) {\n this._fullscreenchange = 'MSFullscreenChange';\n }\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map): HTMLElement {\n this._map = map;\n if (!this._container) this._container = this._map.getContainer();\n this._controlContainer = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group');\n this._setupUI();\n return this._controlContainer;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._controlContainer);\n this._map = null;\n window.document.removeEventListener(this._fullscreenchange, this._onFullscreenChange);\n }\n\n _setupUI() {\n const button = this._fullscreenButton = DOM.create('button', (('maplibregl-ctrl-fullscreen')), this._controlContainer);\n DOM.create('span', 'maplibregl-ctrl-icon', button).setAttribute('aria-hidden', 'true');\n button.type = 'button';\n this._updateTitle();\n this._fullscreenButton.addEventListener('click', this._onClickFullscreen);\n window.document.addEventListener(this._fullscreenchange, this._onFullscreenChange);\n }\n\n _updateTitle() {\n const title = this._getTitle();\n this._fullscreenButton.setAttribute('aria-label', title);\n this._fullscreenButton.title = title;\n }\n\n _getTitle() {\n return this._map._getUIString(this._isFullscreen() ? 'FullscreenControl.Exit' : 'FullscreenControl.Enter');\n }\n\n _isFullscreen() {\n return this._fullscreen;\n }\n\n _onFullscreenChange = () => {\n const fullscreenElement =\n window.document.fullscreenElement ||\n (window.document as any).mozFullScreenElement ||\n (window.document as any).webkitFullscreenElement ||\n (window.document as any).msFullscreenElement;\n\n if ((fullscreenElement === this._container) !== this._fullscreen) {\n this._handleFullscreenChange();\n }\n };\n\n _handleFullscreenChange() {\n this._fullscreen = !this._fullscreen;\n this._fullscreenButton.classList.toggle('maplibregl-ctrl-shrink');\n this._fullscreenButton.classList.toggle('maplibregl-ctrl-fullscreen');\n this._updateTitle();\n\n if (this._fullscreen) {\n this.fire(new Event('fullscreenstart'));\n if (this._map._cooperativeGestures) {\n this._prevCooperativeGestures = this._map._cooperativeGestures;\n this._map.setCooperativeGestures();\n }\n } else {\n this.fire(new Event('fullscreenend'));\n if (this._prevCooperativeGestures) {\n this._map.setCooperativeGestures(this._prevCooperativeGestures);\n delete this._prevCooperativeGestures;\n }\n }\n }\n\n _onClickFullscreen = () => {\n if (this._isFullscreen()) {\n this._exitFullscreen();\n } else {\n this._requestFullscreen();\n }\n };\n\n _exitFullscreen() {\n if (window.document.exitFullscreen) {\n (window.document as any).exitFullscreen();\n } else if ((window.document as any).mozCancelFullScreen) {\n (window.document as any).mozCancelFullScreen();\n } else if ((window.document as any).msExitFullscreen) {\n (window.document as any).msExitFullscreen();\n } else if ((window.document as any).webkitCancelFullScreen) {\n (window.document as any).webkitCancelFullScreen();\n } else {\n this._togglePseudoFullScreen();\n }\n }\n\n _requestFullscreen() {\n if (this._container.requestFullscreen) {\n this._container.requestFullscreen();\n } else if ((this._container as any).mozRequestFullScreen) {\n (this._container as any).mozRequestFullScreen();\n } else if ((this._container as any).msRequestFullscreen) {\n (this._container as any).msRequestFullscreen();\n } else if ((this._container as any).webkitRequestFullscreen) {\n (this._container as any).webkitRequestFullscreen();\n } else {\n this._togglePseudoFullScreen();\n }\n }\n\n _togglePseudoFullScreen() {\n this._container.classList.toggle('maplibregl-pseudo-fullscreen');\n this._handleFullscreenChange();\n this._map.resize();\n }\n}\n","import {DOM} from '../../util/dom';\n\nimport type {Map} from '../map';\nimport type {IControl} from './control';\nimport type {TerrainSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A `TerrainControl` control contains a button for turning the terrain on and off.\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * let map = new maplibregl.Map({TerrainControl: false})\n * .addControl(new maplibregl.TerrainControl({\n * source: \"terrain\"\n * }));\n * ```\n */\nexport class TerrainControl implements IControl {\n options: TerrainSpecification;\n _map: Map;\n _container: HTMLElement;\n _terrainButton: HTMLButtonElement;\n\n constructor(options: TerrainSpecification) {\n this.options = options;\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group');\n this._terrainButton = DOM.create('button', 'maplibregl-ctrl-terrain', this._container);\n DOM.create('span', 'maplibregl-ctrl-icon', this._terrainButton).setAttribute('aria-hidden', 'true');\n this._terrainButton.type = 'button';\n this._terrainButton.addEventListener('click', this._toggleTerrain);\n\n this._updateTerrainIcon();\n this._map.on('terrain', this._updateTerrainIcon);\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._container);\n this._map.off('terrain', this._updateTerrainIcon);\n this._map = undefined;\n }\n\n _toggleTerrain = () => {\n if (this._map.getTerrain()) {\n this._map.setTerrain(null);\n } else {\n this._map.setTerrain(this.options);\n }\n this._updateTerrainIcon();\n };\n\n _updateTerrainIcon = () => {\n this._terrainButton.classList.remove('maplibregl-ctrl-terrain');\n this._terrainButton.classList.remove('maplibregl-ctrl-terrain-enabled');\n if (this._map.terrain) {\n this._terrainButton.classList.add('maplibregl-ctrl-terrain-enabled');\n this._terrainButton.title = this._map._getUIString('TerrainControl.disableTerrain');\n } else {\n this._terrainButton.classList.add('maplibregl-ctrl-terrain');\n this._terrainButton.title = this._map._getUIString('TerrainControl.enableTerrain');\n }\n };\n}\n"],"names":["__awaiter","thisArg","_arguments","P","generator","Promise","resolve","reject","fulfilled","value","step","next","e","rejected","result","done","then","apply","SuppressedError","pointGeometry","Point","x","y","this","prototype","clone","add","p","_add","sub","_sub","multByPoint","_multByPoint","divByPoint","_divByPoint","mult","k","_mult","div","_div","rotate","a","_rotate","rotateAround","_rotateAround","matMult","m","_matMult","unit","_unit","perp","_perp","round","_round","mag","Math","sqrt","equals","other","dist","distSqr","dx","dy","angle","atan2","angleTo","b","angleWith","angleWithSep","cos","sin","convert","Array","isArray","unitbezier","UnitBezier","p1x","p1y","p2x","p2y","cx","bx","ax","cy","by","ay","sampleCurveX","t","sampleCurveY","sampleCurveDerivativeX","solveCurveX","epsilon","undefined","i","x2","abs","d2","t0","t1","solve","supportsOffscreenCanvas","offscreenCanvasDistorted","offscreenCanvasSupported","OffscreenCanvas","getContext","createImageBitmap","bezier","defaultEasing","clamp","n","min","max","wrap","d","w","asyncAll","array","fn","callback","length","remaining","results","error","forEach","item","err","extend","dest","sources","src","pick","properties","id","uniqueId","mapObject","input","iterator","context","output","key","call","filterObject","deepEqual","Object","keys","map","warnOnceHistory","warnOnce","message","console","warn","isCounterClockwise","c","calculateSignedArea","ring","sum","p1","p2","len","j","isWorker","WorkerGlobalScope","self","_isSafari","isSafari","scope","userAgent","navigator","safari","test","match","isImageBitmap","image","ImageBitmap","transparentPngUrl","linkEl","reducedMotionQuery","browser","now","performance","bind","Date","frame","requestAnimationFrame","cancel","cancelAnimationFrame","getImageData","img","padding","getImageCanvasContext","width","height","canvas","window","document","createElement","willReadFrequently","Error","drawImage","resolveURL","path","href","hardwareConcurrency","prefersReducedMotion","matchMedia","matches","DOM","static","props","docStyle","tagName","className","container","el","appendChild","namespaceURI","createElementNS","selectProp","userSelect","style","transformProp","target","type","options","addEventListener","capture","removeEventListener","preventDefault","stopPropagation","suppressClickInternal","setTimeout","rect","getBoundingClientRect","clientX","left","clientLeft","clientY","top","clientTop","touches","points","push","button","node","parentNode","removeChild","documentElement","testProp","config","MAX_PARALLEL_IMAGE_REQUESTS","MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME","MAX_TILE_CACHE_ZOOM_LEVELS","REGISTERED_PROTOCOLS","WORKER_URL","AJAXError","constructor","status","statusText","url","body","super","getReferrer","worker","referrer","location","protocol","parent","getProtocolAction","substring","indexOf","makeFetchRequest","requestParameters","controller","AbortController","request","Request","method","credentials","headers","cache","signal","complete","aborted","set","fetch","response","ok","arrayBuffer","json","text","get","catch","finishRequest","blob","code","abort","makeRequest","actor","send","hasOwnProperty","xhr","XMLHttpRequest","open","responseType","setRequestHeader","withCredentials","onerror","onload","data","JSON","parse","getResponseHeader","Blob","makeXMLHttpRequest","getJSON","getArrayBuffer","sameOrigin","inComingUrl","urlObj","URL","locationObj","host","webpSupported","supported","testSupport","gl","webpCheckComplete","webpImgTest","webpImgTestOnloadComplete","testWebpTextureUpload","glForTesting","texture","createTexture","bindTexture","TEXTURE_2D","texImage2D","RGBA","UNSIGNED_BYTE","isContextLost","deleteTexture","ImageRequest","ResourceType","imageRequestQueue","currentParallelImageRequests","throttleControlCallbackHandleCounter","throttleControlCallbacks","resetRequestQueue","addThrottleControl","handle","removeThrottleControl","callbackHandle","processQueue","getImage","supportImageRefresh","accept","cancelled","completed","innerRequest","doImageRequest","itemInQueue","reduce","acc","getImageUsingHtmlImage","cacheControl","expires","onImageResponse","HTMLImageElement","Uint8Array","imgBitmap","arrayBufferToImageBitmap","Image","revokeObjectURL","byteLength","createObjectURL","arrayBufferToImage","arrayBufferToCanvasImageSource","imgErr","imgResult","maxImageRequests","allControlKeys","throttleingRequested","isThrottled","numImageRequests","topItemInQueue","shift","requestCancelled","crossOrigin","fetchPriority","RequestManager","transformRequestFn","_transformRequestFn","transformRequest","normalizeSpriteURL","format","extension","urlObject","parts","urlRe","authority","params","split","parseUrl","obj","join","formatUrl","setTransformRequest","_addEventListener","listener","listenerList","_removeEventListener","index","splice","Event","ErrorEvent","Evented","on","_listeners","off","_oneTimeListeners","once","fire","event","listens","listeners","slice","oneTimeListeners","_eventedParent","_eventedParentData","setEventedParent","v8Spec","$version","$root","version","required","values","name","metadata","center","zoom","bearing","default","period","units","pitch","light","terrain","sprite","glyphs","transition","layers","source","source_vector","vector","tiles","bounds","scheme","xyz","tms","minzoom","maxzoom","attribution","promoteId","volatile","source_raster","raster","tileSize","source_raster_dem","encoding","terrarium","mapbox","custom","redFactor","blueFactor","greenFactor","baseShift","source_geojson","geojson","buffer","maximum","minimum","filter","tolerance","cluster","clusterRadius","clusterMaxZoom","clusterMinPoints","clusterProperties","lineMetrics","generateId","source_video","video","urls","coordinates","source_image","layer","fill","line","symbol","circle","heatmap","hillshade","background","layout","paint","layout_background","visibility","visible","none","layout_fill","expression","interpolated","parameters","layout_circle","layout_heatmap","layout_line","butt","square","bevel","miter","requires","layout_symbol","point","auto","never","always","cooperative","viewport","both","tokens","right","bottom","horizontal","vertical","uppercase","lowercase","layout_raster","layout_hillshade","filter_operator","in","all","any","has","within","geometry_type","LineString","Polygon","function","stops","base","property","identity","exponential","interval","categorical","colorSpace","rgb","lab","hcl","function_stop","anchor","position","color","intensity","exaggeration","paint_fill","paint_line","paint_circle","paint_heatmap","paint_symbol","overridable","paint_raster","linear","nearest","paint_hillshade","paint_background","duration","delay","constant","refProperties","deref","derefLayers","create","ref","operations","setStyle","addLayer","removeLayer","setPaintProperty","setLayoutProperty","setFilter","addSource","removeSource","setGeoJSONSourceData","setLayerZoomRange","setLayerProperty","setCenter","setZoom","setBearing","setPitch","setSprite","setGlyphs","setTransition","setLight","sourceId","after","commands","command","args","sourcesRemoved","updateSource","canUpdateGeoJSON","before","prop","diffLayerPropertyChanges","layerId","klass","pluckId","indexById","group","ValidationError","identifier","__line__","extendBy","inputs","ExpressionParsingError","Scope","bindings","concat","NullType","kind","NumberType","StringType","BooleanType","ColorType","ObjectType","ValueType","CollatorType","FormattedType","PaddingType","ResolvedImageType","VariableAnchorOffsetCollectionType","array$1","itemType","N","toString$1","valueMemberTypes","checkSubtype","expected","memberType","isValidType","provided","allowedTypes","some","isValidNativeType","verifyType","sample","Xn","Zn","t2","t3","deg2rad","PI","rad2deg","constrainAngle","rgbToLab","r","g","alpha","z","xyz2lab","rgb2xyz","l","pow","labToRgb","isNaN","lab2xyz","xyz2rgb","parseHex","hex","parseInt","padEnd","parseAlpha","asPercentage","validateNumbers","Number","namedColors","aliceblue","antiquewhite","aqua","aquamarine","azure","beige","bisque","black","blanchedalmond","blue","blueviolet","brown","burlywood","cadetblue","chartreuse","chocolate","coral","cornflowerblue","cornsilk","crimson","cyan","darkblue","darkcyan","darkgoldenrod","darkgray","darkgreen","darkgrey","darkkhaki","darkmagenta","darkolivegreen","darkorange","darkorchid","darkred","darksalmon","darkseagreen","darkslateblue","darkslategray","darkslategrey","darkturquoise","darkviolet","deeppink","deepskyblue","dimgray","dimgrey","dodgerblue","firebrick","floralwhite","forestgreen","fuchsia","gainsboro","ghostwhite","gold","goldenrod","gray","green","greenyellow","grey","honeydew","hotpink","indianred","indigo","ivory","khaki","lavender","lavenderblush","lawngreen","lemonchiffon","lightblue","lightcoral","lightcyan","lightgoldenrodyellow","lightgray","lightgreen","lightgrey","lightpink","lightsalmon","lightseagreen","lightskyblue","lightslategray","lightslategrey","lightsteelblue","lightyellow","lime","limegreen","linen","magenta","maroon","mediumaquamarine","mediumblue","mediumorchid","mediumpurple","mediumseagreen","mediumslateblue","mediumspringgreen","mediumturquoise","mediumvioletred","midnightblue","mintcream","mistyrose","moccasin","navajowhite","navy","oldlace","olive","olivedrab","orange","orangered","orchid","palegoldenrod","palegreen","paleturquoise","palevioletred","papayawhip","peachpuff","peru","pink","plum","powderblue","purple","rebeccapurple","red","rosybrown","royalblue","saddlebrown","salmon","sandybrown","seagreen","seashell","sienna","silver","skyblue","slateblue","slategray","slategrey","snow","springgreen","steelblue","tan","teal","thistle","tomato","turquoise","violet","wheat","white","whitesmoke","yellow","yellowgreen","Color","premultiplied","overwriteGetter","rgba","toLowerCase","trim","namedColorsMatch","startsWith","rgbMatch","_","rp","f1","gp","f2","bp","f3","ap","argFormat","valFormat","maxValue","hslMatch","h","s","hsla","f","hslToRgb","parseCssColor","Infinity","rgbColor","NaN","rgbToHcl","getterKey","lazyValue","defineProperty","toString","transparent","Collator","caseSensitive","diacriticSensitive","locale","sensitivity","collator","Intl","usage","compare","lhs","rhs","resolvedLocale","resolvedOptions","FormattedSection","scale","fontStack","textColor","Formatted","sections","unformatted","isEmpty","section","fromString","Padding","val","stringify","anchors","Set","VariableAnchorOffsetCollection","anchorValue","offsetValue","ResolvedImage","available","validateRGBA","isValue","mixed","typeOf","String","Literal","expectedType","evaluate","eachChild","outputDefined","RuntimeError","toJSON","types$1","string","number","boolean","object","Assertion","floor","parsed","ctx","every","arg","types","Coercion","Boolean","parseColor","pad","coll","num","geometryTypes","EvaluationContext","globals","feature","featureState","formattedSection","_parseColorCache","availableImages","canonical","geometryType","geometry","canonicalID","cached","ParsingContext","registry","isConstantFunc","errors","part","_isConstant","expr","_parse","annotate","typeAnnotation","op","Expr","actual","ec","CollatorExpression","EXTENT","updateBBox","bbox","coord","boxWithinBox","bbox1","bbox2","getTileCoordinates","log","tilesAtZoom","onBoundary","x1","y1","y2","pointWithinPolygon","rings","inside","len2","pointWithinPolygons","polygons","twoSided","q1","q2","x3","y3","det1","det2","lineIntersectPolygon","polygon","v1","v2","lineStringWithinPolygon","lineStringWithinPolygons","getTilePolygon","getTilePolygons","updatePoint","polyBBox","worldSize","halfWorldSize","getTilePoints","pointBBox","shifts","tilePoints","getTileLines","lineBBox","tileLines","tileLine","Within","geometries","features","polygonGeometry","tilePolygon","tilePolygons","pointsWithinPolygons","linesWithinPolygons","Var","boundExpression","CompoundExpression","_evaluate","definition","definitions","availableOverloads","overloads","signature","signatureContext","isExpressionConstant","parsedArgs","argParseFailed","signatures","stringifySignature","actualTypes","isTypeAnnotation","childrenConstant","child","isFeatureConstant","isGlobalPropertyConstant","isStateConstant","findStopLessThanOrEqualTo","lastIndex","currentValue","nextValue","lowerIndex","upperIndex","currentIndex","Step","labels","outputs","label","outputType","labelKey","valueKey","stopCount","out","from","to","interpolate","spaceKey","hue0","chroma0","light0","alphaF","hue1","chroma1","light1","alphaT","hue","chroma","dh","hclToRgb","variableAnchorOffsetCollection","fromValues","toValues","fx","fy","tx","ty","Interpolate","operator","interpolation","lower","upper","exponentialInterpolation","controlPoints","rest","interpolationFactor","outputLower","outputUpper","lowerValue","upperValue","difference","progress","Coalesce","needsAnnotation","requestedImageName","argCount","Let","binding","At","In","needle","haystack","IndexOf","fromIndex","Match","inputType","cases","otherwise","labelContext","MAX_SAFE_INTEGER","Case","branches","Slice","beginIndex","endIndex","isComparableType","eqCollate","makeComparison","compareBasic","compareWithCollator","isOrderComparison","Comparison","hasUntypedArgument","lt","rt","Equals","NotEquals","LessThan","GreaterThan","LessThanOrEqual","GreaterThanOrEqual","NumberFormat","currency","minFractionDigits","maxFractionDigits","minimumFractionDigits","maximumFractionDigits","FormatExpression","firstArg","nextTokenMayBeObject","font","lastExpression","content","evaluatedContent","ImageExpression","evaluatedImageName","Length","expressions","at","case","coalesce","let","literal","var","v","varargs","success","supportsPropertyExpression","spec","supportsZoomExpression","supportsInterpolation","getType","isFunction","identityFunction","createFunction","propertySpec","isColor","zoomAndFeatureDependent","zoomDependent","parseFn","stop","innerFun","hashedStops","categoricalKeyType","evaluateExponentialFunction","evaluateIntervalFunction","evaluateCategoricalFunction","evaluateIdentityFunction","featureFunctions","zoomStops","featureFunctionStops","interpolationType","coalesce$1","keyType","interp","evaluatedLower","evaluatedUpper","register","typeof","heatmapDensity","lineProgress","accumulated","ln2","LN2","pi","E","log10","LN10","ln","log2","asin","acos","atan","ceil","binarySearch","isSupportedScript","upcase","toUpperCase","downcase","StyleExpression","_warningHistory","_evaluator","_defaultValue","_enumValues","evaluateWithoutErrorHandling","isExpression","createExpression","parser","enum","formatted","resolvedImage","getExpectedType","ZoomConstantExpression","_styleExpression","isStateDependent","ZoomDependentExpression","createPropertyExpression","expressionInput","isFeatureConstantResult","isZoomConstant","zoomCurve","findZoomCurve","StylePropertyFunction","specification","_parameters","_specification","serialized","childResult","isExpressionFilter","filterSpec","createFilter","needGeometry","convertFilter$1","compiled","globalProperties","geometryNeeded","convertComparisonOp$1","convertNegation","filters","convertInOp$1","convertHasOp$1","sort","validateConstants","constants","unbundle","valueOf","deepUnbundle","unbundledValue","validateObject","elementSpecs","valueSpec","elementValidators","objectElementValidators","styleSpec","validateSpec","objectKey","elementSpecKey","elementSpec","validateElement","validateArray","arraySpec","validateArrayElement","arrayElementValidator","arrayElementSpec","arrayIndex","validateNumber","validateFunction","functionValueSpec","functionType","stopKeyType","previousStopDomainValue","previousStopDomainZoom","stopDomainValues","isZoomFunction","isPropertyFunction","isZoomAndPropertyFunction","validateFunctionStop","validateStopDomainValue","reportValue","isFinite","validateExpression","expressionContext","expressionObj","propertyKey","propertyType","validateEnum","validateFilter","validateNonExpressionFilter","validateProperty","layerSpec","layerType","transitionMatch","tokenMatch","exec","validatePaintProperty","validateLayoutProperty","validateLayer","otherLayer","sourceType","validateString","validateSource","replace","_a","sourceName","rasterDEM","rasterDEMSpec","rootType","isCustomEncoding","customEncodingKeys","encodingName","includes","validateRasterDEMSource","mapExpr","reduceExpr","validateLight","lightSpec","validateTerrain","terrainSpec","validateSprite","allSpriteIds","allSpriteURLs","VALIDATORS","validate","validateGlyphsUrl","validateStyleMin","sortErrors","injectValidateSpec","validator","wrapCleanErrors","inner","paintProperty","layoutProperty","validateStyle","emitValidationErrors","emitter","hasErrors","TransferableGridIndex","extent","cells","ArrayBuffer","Int32Array","start","end","subarray","bboxesOffset","bboxes","insert","_insertReadonly","uid","_forEachCell","_insertCell","cellIndex","query","intersectionTest","_queryCell","seenUids","cell","u","offset","arg1","arg2","cx1","_convertToCellCoord","cy1","cx2","cy2","_convertFromCellCoord","toArrayBuffer","metadataLength","totalCellLength","grid","transferables","writeable","omit","shallow","_classRegistryKey","isArrayBuffer","serialize","RegExp","isView","view","ImageData","$name","deserialize","ZoomHistory","first","update","floorZ","lastIntegerZoom","lastIntegerZoomTime","lastZoom","lastFloorZoom","unicodeBlockLookup","char","Arabic","Khmer","Hiragana","Katakana","Bopomofo","Kanbun","allowsVerticalWritingMode","chars","charHasUprightVerticalOrientation","charCodeAt","isChar","charInRTLScript","charInSupportedScript","canRenderRTL","stringContainsRTLText","_completionCallback","pluginStatus","pluginURL","triggerPluginCompletionEvent","sendPluginStateToWorker","evented","getRTLTextPluginStatus","downloadRTLTextPlugin","plugin","applyArabicShaping","processBidirectionalText","processStyledBidirectionalText","isLoaded","isLoading","setState","state","isParsed","getPluginURL","EvaluationParameters","fadeDuration","zoomHistory","str","isStringInSupportedScript","rtlTextPlugin","crossFadingFactor","getCrossfadeParameters","fraction","fromScale","toScale","PropertyValue","normalizePropertyExpression","isDataDriven","possiblyEvaluate","TransitionablePropertyValue","transitioned","prior","TransitioningPropertyValue","untransitioned","Transitionable","_properties","_values","defaultTransitionablePropertyValues","getValue","setValue","getTransition","Transitioning","begin","finalValue","easeCubicInOut","defaultTransitioningPropertyValues","PossiblyEvaluated","hasTransition","Layout","defaultPropertyValues","hasValue","PossiblyEvaluatedPropertyValue","isConstant","constantOr","defaultPossiblyEvaluatedValues","DataConstantProperty","interpolationFn","interpolates","DataDrivenProperty","overrides","interpolatedValue","CrossFadedDataDrivenProperty","evaluatedValue","constantValue","_calculate","cameraVal","mid","CrossFadedProperty","ColorRampProperty","Properties","overridableProperties","defaultPropertyValue","defaultTransitionablePropertyValue","TRANSITION_SUFFIX","StyleLayer","_featureFilter","sourceLayer","_unevaluatedLayout","_transitionablePaint","_transitioningPaint","_crossfadeParameters","getLayoutProperty","_validate","getPaintProperty","endsWith","transitionable","isCrossFadedProperty","wasDataDriven","oldValue","_handleSpecialPaintPropertyUpdate","newValue","_handleOverridablePaintPropertyUpdate","isHidden","updateTransitions","recalculate","is3D","isTileClipped","hasOffscreenPass","resize","viewTypes","Int8","Int8Array","Uint8","Int16","Int16Array","Uint16","Uint16Array","Int32","Uint32","Uint32Array","Float32","Float32Array","Struct","structArray","_structArray","_pos1","size","_pos2","_pos4","_pos8","StructArray","isTransferred","capacity","_trim","bytesPerElement","_refreshViews","clear","reserve","oldUint8Array","uint8","createLayout","members","alignment","maxSize","member","typeSize","BYTES_PER_ELEMENT","memberOffset","align","components","StructArrayLayout2i4","int16","emplaceBack","v0","emplace","o2","StructArrayLayout3i6","StructArrayLayout4i8","v3","StructArrayLayout2i4i12","v4","v5","StructArrayLayout2i4ub8","o1","StructArrayLayout2f8","float32","o4","StructArrayLayout10ui20","uint16","v6","v7","v8","v9","StructArrayLayout4i4ui4i24","v10","v11","StructArrayLayout3f12","StructArrayLayout1ul4","uint32","StructArrayLayout6i1ul2ui20","StructArrayLayout2i2i2i12","StructArrayLayout2f1f2i16","StructArrayLayout2ub2f12","StructArrayLayout3ui6","StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48","v12","v13","v14","v15","v16","StructArrayLayout8i15ui1ul2f2ui64","v17","v18","v19","v20","v21","v22","v23","v24","v25","v26","v27","StructArrayLayout1f4","StructArrayLayout1ui2f12","StructArrayLayout1ul2ui8","StructArrayLayout2ui4","StructArrayLayout1ui2","StructArrayLayout4f16","CollisionBoxStruct","anchorPointX","anchorPointY","featureIndex","sourceLayerIndex","bucketIndex","anchorPoint","CollisionBoxArray","PlacedSymbolStruct","anchorX","anchorY","glyphStartIndex","numGlyphs","vertexStartIndex","lineStartIndex","lineLength","segment","lowerSize","upperSize","lineOffsetX","lineOffsetY","writingMode","placedOrientation","hidden","crossTileID","associatedIconIndex","PlacedSymbolArray","SymbolInstanceStruct","rightJustifiedTextSymbolIndex","centerJustifiedTextSymbolIndex","leftJustifiedTextSymbolIndex","verticalPlacedTextSymbolIndex","placedIconSymbolIndex","verticalPlacedIconSymbolIndex","textBoxStartIndex","textBoxEndIndex","verticalTextBoxStartIndex","verticalTextBoxEndIndex","iconBoxStartIndex","iconBoxEndIndex","verticalIconBoxStartIndex","verticalIconBoxEndIndex","numHorizontalGlyphVertices","numVerticalGlyphVertices","numIconVertices","numVerticalIconVertices","useRuntimeCollisionCircles","textBoxScale","collisionCircleDiameter","textAnchorOffsetStartIndex","textAnchorOffsetEndIndex","SymbolInstanceArray","GlyphOffsetArray","getoffsetX","SymbolLineVertexArray","getx","gety","gettileUnitDistanceFromAnchor","TextAnchorOffsetStruct","textAnchor","textOffset0","textOffset1","TextAnchorOffsetArray","FeatureIndexStruct","FeatureIndexArray","PosArray","Pos3dArray","RasterBoundsArray","CircleLayoutArray","FillLayoutArray","FillExtrusionLayoutArray","LineLayoutArray","LineExtLayoutArray","PatternLayoutArray","SymbolLayoutArray","SymbolDynamicLayoutArray","SymbolOpacityArray","CollisionBoxLayoutArray","CollisionCircleLayoutArray","CollisionVertexArray","QuadTriangleArray","TriangleIndexArray","LineIndexArray","LineStripIndexArray","SegmentVector","segments","prepareSegment","numVertices","layoutVertexArray","indexArray","sortKey","MAX_VERTEX_ARRAY_LENGTH","vertexLength","vertexOffset","primitiveOffset","primitiveLength","destroy","vaos","packUint8ToFloat","patternAttributes","module","seed","remainder","bytes","h1","h1b","c1","c2","k1","murmur3","require$$0","murmur2","murmurhashJsModule","exports","FeaturePositionMap","ids","positions","indexed","getNumericId","getPositions","intId","Float64Array","numValue","pivot","swap","arr","tmp","Uniform","Uniform1i","current","uniform1i","Uniform1f","uniform1f","Uniform2f","uniform2f","Uniform3f","uniform3f","Uniform4f","uniform4f","UniformColor","emptyMat4","UniformMatrix4f","uniformMatrix4fv","packColor","ConstantBinder","names","uniformNames","setUniform","uniform","getBinding","CrossFadedConstantBinder","patternFrom","patternTo","pixelRatioFrom","pixelRatioTo","setConstantPatternPositions","posTo","posFrom","pixelRatio","tlbr","uniformName","pos","substr","SourceExpressionBinder","PaintVertexArray","paintVertexAttributes","paintVertexArray","populatePaintArray","newLength","imagePositions","_setPaintValue","updatePaintArray","upload","paintVertexBuffer","updateData","createVertexBuffer","CompositeExpressionBinder","useIntegerZoom","minColor","maxColor","currentZoom","factor","CrossFadedCompositeBinder","zoomInPaintVertexArray","zoomOutPaintVertexArray","_setPaintValues","patterns","imageMin","imageMid","imageMax","tl","br","zoomInPaintVertexBuffer","zoomOutPaintVertexBuffer","ProgramConfiguration","filterProperties","binders","_buffers","paintAttributeNames","propType","isCrossFaded","StructArrayLayout","layoutType","cacheKey","getMaxValue","binder","populatePaintArrays","updatePaintArrays","featureStates","featureMap","vtLayer","dirty","defines","getBinderAttributes","getBinderUniforms","uniforms","getPaintVertexBuffers","getUniforms","locations","setUniforms","binderUniforms","updatePaintBuffers","crossfade","patternVertexBuffer","ProgramConfigurationSet","programConfigurations","needsUpload","_featureMap","_bufferOffset","binderType","defaultLayouts","composite","layoutException","getLayoutException","MAX","BITS","MIN","loadGeometry","toEvaluationFeature","addCircleVertex","extrudeX","extrudeY","CircleBucket","overscaling","layerIds","hasPattern","stateDependentLayerIds","populate","styleLayer","bucketFeatures","circleSortKey","sortFeaturesByKey","evaluationFeature","bucketFeature","addFeature","states","stateDependentLayers","uploadPending","uploaded","layoutVertexBuffer","layoutAttributes","indexBuffer","createIndexBuffer","polygonIntersectsPolygon","polygonA","polygonB","polygonContainsPoint","lineIntersectsLine","polygonIntersectsBufferedPoint","radius","pointIntersectsBufferedLine","polygonIntersectsMultiPolygon","multiPolygon","multiPolygonContainsPoint","lineIntersectsBufferedLine","lineA","lineB","a0","a1","lineSegmentIntersectsLineSegment","b0","b1","radiusSquared","distToSegmentSquared","l2","edgeIntersectsBox","e1","e2","corners","dir","getMaximumPaintValue","bucket","translateDistance","translate","queryGeometry","translateAnchor","pixelsToTileUnits","pt","translated","properties$8","EPSILON","ARRAY_TYPE","glMatrix.ARRAY_TYPE","invert","a00","a01","a02","a03","a10","a11","a12","a13","a20","a21","a22","a23","a30","a31","a32","a33","b00","b01","b02","b03","b04","b05","b06","b07","b08","b09","b10","b11","det","multiply","b2","b3","rotateZ","rad","hypot","arguments","ortho","near","far","lr","bt","nf","mul","transformMat4","sqrLen","CircleStyleLayer","createBucket","queryRadius","circleBucket","queryIntersectsFeature","transform","pixelPosMatrix","translatedPolygon","alignWithMap","transformedPolygon","projectPoint","projectQueryGeometry","transformedSize","transformedPoint","adjustedSize","projectedCenter","vec4.transformMat4","cameraToCenterDistance","HeatmapBucket","properties$7","createImage","channels","Uint8ClampedArray","RangeError","resizeImage","newImage","copyImage","srcImg","dstImg","srcPt","dstPt","srcData","dstData","srcOffset","dstOffset","AlphaImage","RGBAImage","copy","renderColorRamp","evaluationGlobals","resolution","clips","renderPixel","stride","evaluationKey","pxColor","clip","HeatmapStyleLayer","_updateColorRamp","colorRamp","colorRampTexture","heatmapFbo","properties$6","HillshadeStyleLayer","earcut","holeIndices","dim","minX","minY","maxX","maxY","invSize","hasHoles","outerLen","outerNode","linkedList","triangles","prev","list","queue","steiner","getLeftmost","compareX","eliminateHole","eliminateHoles","earcutLinked","clockwise","last","signedArea","insertNode","removeNode","filterPoints","again","area","ear","pass","zOrder","prevZ","nextZ","q","tail","numMerges","pSize","qSize","inSize","sortLinked","indexCurve","isEarHashed","isEar","cureLocalIntersections","splitEarcut","x0","y0","pointInTriangle","minZ","maxZ","intersects","locallyInside","isValidDiagonal","splitPolygon","hole","bridge","hx","hy","qx","mx","my","tanMin","sectorContainsSector","findHoleBridge","bridgeReverse","leftmost","px","py","intersectsPolygon","middleInside","sign","o3","onSegment","a2","Node","an","earcutModule","deviation","polygonArea","trianglesArea","flatten","vertices","holes","dimensions","holeIndex","quickselect","quickselectStep","defaultCompare","exp","sd","classifyRings","maxRings","ccw","compareAreas","patternDependencies","patternProperty","constantPattern","addPatternDependencies","patternFeature","patternPropertyValue","FillBucket","patternFeatures","indexArray2","segments2","fillSortKey","addFeatures","indexBuffer2","triangleSegment","triangleIndex","flattened","lineSegment","lineIndex","indices","properties$5","FillStyleLayer","outlineColor","centroidAttributes","vectortilefeature","VectorTileFeature","pbf","_pbf","_geometry","_keys","readFields","readFeature","tag","readVarint","readTag","cmd","lines","cmdLen","readSVarint","toGeoJSON","coords","project","vectortilelayer","VectorTileLayer","_features","readLayer","readString","readFloat","readDouble","readVarint64","readBoolean","readValueMessage","readTile","vectorTile","VectorTile","require$$1","require$$2","vectorTileFeatureTypes","mvt","FACTOR","addVertex","vertexArray","nx","ny","nz","FillExtrusionBucket","centroidVertexArray","centroidVertexBuffer","centroid","vertexCount","isEntirelyOutside","edgeDistance","isBoundaryEdge","bottomRight","properties$4","FillExtrusionStyleLayer","projectedQueryGeometry","projected","zBase","zTop","projectedBase","projectedTop","baseXZ","baseYZ","baseZZ","baseWZ","topXZ","topYZ","topZZ","topWZ","ringBase","ringTop","sX","sY","sZ","sW","baseZ","baseW","topX","topY","topZ","topW","projectExtrusion","closestDistance","getIntersectionDistance","topA","face","checkIntersection","dot","projectedFace","ab","ac","dotABAB","dotABAC","dotACAC","dotAPAB","dotAPAC","denom","distance","lineLayoutAttributes","lineLayoutAttributesExt","COS_HALF_SHARP_CORNER","MAX_LINE_DISTANCE","LINE_DISTANCE_BUFFER_BITS","LineBucket","lineClipsArray","gradients","layoutVertexArray2","maxLineLength","lineSortKey","patternBucketFeature","layoutVertexBuffer2","layoutAttributesExt","lineFeatureClips","cap","miterLimit","roundLimit","lineClips","addLine","scaledDistance","totalDistance","updateScaledDistance","isPolygon","sharpCornerOffset","currentVertex","prevVertex","nextVertex","prevNormal","nextNormal","joinNormal","cosAngle","cosHalfAngle","miterLength","approxAngle","isSharpCorner","lineTurnsLeft","prevSegmentLength","newPrevVertex","updateDistance","addCurrentVertex","middleVertex","currentJoin","bevelLength","offsetA","offsetB","extrude","addHalfVertex","nextSegmentLength","newCurrentVertex","normal","endLeft","endRight","rightX","rightY","up","linesofarScaled","properties$3","LineFloorwidthProperty","lineFloorwidthProperty","LineStyleLayer","gradientVersion","gradientExpression","stepInterpolant","isZoomExpression","lineBucket","getLineWidth","halfWidth","lineOffset","newRings","ringIndex","newRing","aToB","bToC","offsetLine","multiLine","polygonIntersectsBufferedMultiLine","lineWidth","lineGapWidth","symbolLayoutAttributes","dynamicLayoutAttributes","collisionVertexAttributes","collisionBoxLayout","collisionCircleLayout","transformText","toLocaleUpperCase","toLocaleLowerCase","transformTextInternal","verticalizedCharacterMap","$","Pbf","ieee754","isLE","mLen","nBytes","eLen","eMax","eBias","nBits","buf","Varint","Fixed64","Bytes","Fixed32","SHIFT_LEFT_32","SHIFT_RIGHT_32","utf8TextDecoder","TextDecoder","readPackedEnd","toNum","low","high","isSigned","makeRoomForExtraLength","startPos","extraLen","realloc","writePackedVarint","writeVarint","writePackedSVarint","writeSVarint","writePackedFloat","writeFloat","writePackedDouble","writeDouble","writePackedBoolean","writeBoolean","writePackedFixed32","writeFixed32","writePackedSFixed32","writeSFixed32","writePackedFixed64","writeFixed64","writePackedSFixed64","writeSFixed64","readUInt32","writeInt32","readInt32","readField","skip","readMessage","readFixed32","readSFixed32","readFixed64","readSFixed64","readVarintRemainder","decode","readUtf8TextDecoder","bytesPerSequence","fromCharCode","readUtf8","readBytes","readPackedVarint","readPackedSVarint","readPackedBoolean","readPackedFloat","readPackedDouble","readPackedFixed32","readPackedSFixed32","readPackedFixed64","readPackedSFixed64","writeTag","finish","writeBigVarintLow","lsb","writeBigVarintHigh","writeBigVarint","writeString","lead","writeUtf8","writeBytes","writeRawMessage","writeMessage","writeBytesField","writeFixed32Field","writeSFixed32Field","writeFixed64Field","writeSFixed64Field","writeVarintField","writeSVarintField","writeStringField","writeFloatField","writeDoubleField","writeBooleanField","border","readFontstacks","readFontstack","bitmap","advance","readGlyph","metrics","glyph","potpack","boxes","maxWidth","box","spaces","space","pop","ImagePosition","paddedRect","stretchX","stretchY","displaySize","IMAGE_PADDING","WritingMode","getAnchorAlignment","horizontalAlign","verticalAlign","icons","iconPositions","patternPositions","haveRenderCallbacks","bins","addImages","bin","images","hasRenderCallback","patchUpdatedImages","imageManager","dispatchRenderCallbacks","updatedImages","patchUpdatedImage","SIZE_PACK_FACTOR","MAX_PACKED_SIZE","getSizeData","tileZoom","layoutSize","minZoom","maxZoom","minSize","evaluateSizeForFeature","sizeData","uSize","uSizeT","evaluateSizeForZoom","getOverlapMode","overlapProp","allowOverlapProp","overlap","shaderOpacityAttributes","ox","oy","sizeVertex","isSDF","pixelOffsetX","pixelOffsetY","minFontScaleX","minFontScaleY","aSizeX","aSizeY","addDynamicAttributes","dynamicLayoutVertexArray","containsRTLText","formattedText","SymbolBuffers","opacityVertexArray","hasVisibleVertices","placedSymbolArray","dynamicIndexBuffer","dynamicLayoutVertexBuffer","opacityVertexBuffer","itemSize","CollisionBuffers","LayoutArray","IndexArray","collisionVertexArray","collisionVertexBuffer","SymbolBucket","collisionBoxArray","hasRTLText","sortKeyRanges","collisionCircleArray","placementInvProjMatrix","mat4.identity","placementViewportMatrix","unevaluatedLayoutValues","textSizeData","iconSizeData","canOverlap","sortFeaturesByY","writingModes","wm","sourceID","createArrays","icon","glyphOffsetArray","lineVertexArray","symbolInstances","textAnchorOffsets","calculateGlyphDependencies","stack","textAlongLine","allowVerticalPlacement","doesAllowVerticalWritingMode","verticalChar","charAt","textFont","textField","iconImage","hasText","hasIcon","symbolSortKey","iconDependencies","stacks","glyphDependencies","resolvedTokens","getValueAndResolveTokens","factory","globalRTLTextPlugin","sectionFont","sectionStack","leftIndex","rightIndex","mergedFeatures","mergedIndex","mergeFromRight","leftKey","rightKey","geom","mergeFromLeft","getKey","onRight","mergeLines","hasDebugData","textCollisionBox","iconCollisionBox","destroyDebugData","addToLineVertexArray","sumForwardLength","sumBackwardLength","tileUnitDistanceFromAnchor","vertex","addSymbols","arrays","quads","alongLine","labelAnchor","glyphOffsetArrayStart","tr","bl","tex","pixelOffsetTL","pixelOffsetBR","glyphOffset","sectionIndex","_addCollisionDebugVertex","addCollisionDebugVertices","boxAnchorPoint","symbolInstance","addDebugCollisionBoxes","startIndex","isText","generateCollisionDebugBuffers","_deserializeCollisionBoxesForSymbol","textStartIndex","textEndIndex","verticalTextStartIndex","verticalTextEndIndex","iconStartIndex","iconEndIndex","verticalIconStartIndex","verticalIconEndIndex","collisionArrays","textBox","textFeatureIndex","verticalTextBox","verticalTextFeatureIndex","iconBox","iconFeatureIndex","verticalIconBox","verticalIconFeatureIndex","deserializeCollisionBoxes","hasTextData","hasIconData","hasTextCollisionBoxData","hasIconCollisionBoxData","addIndicesForPlacedSymbol","iconOrText","placedSymbolIndex","placedSymbol","vertexIndex","getSortedSymbolIndexes","sortedAngle","symbolInstanceIndexes","rotatedYs","featureIndexes","aIndex","bIndex","addToSortKeyRanges","symbolInstanceIndex","symbolInstanceEnd","symbolInstanceStart","sortFeatures","featureSortOrder","MAX_GLYPHS","properties$2","runtimeType","getOverride","o","hasOverride","FormatSectionOverride","defaultValue","SymbolStyleLayer","deduped","_setPaintOverrides","unevaluated","resolveTokens","hasPaintOverride","overriden","override","styleExpression","propertyName","hasOverrides","checkSections","checkExpression","properties$1","BackgroundStyleLayer","RasterStyleLayer","CustomStyleLayer","implementation","onAdd","painter","onRemove","renderingMode","prerender","createStyleLayer","coerceSpriteToArray","resultArray","dedupArray","doOnceCompleted","callbackFunc","jsonsMap","imagesMap","expectedResultCounter","spriteName","sdf","spriteData","Texture","useMipmap","pixelStoreUnpackFlipY","pixelStoreUnpack","pixelStoreUnpackPremultiplyAlpha","premultiply","HTMLCanvasElement","HTMLVideoElement","texSubImage2D","isSizePowerOfTwo","generateMipmap","minFilter","LINEAR_MIPMAP_NEAREST","LINEAR","texParameteri","TEXTURE_MAG_FILTER","TEXTURE_MIN_FILTER","TEXTURE_WRAP_S","TEXTURE_WRAP_T","renderStyleImage","userImage","render","ImageManager","callbackDispatchedThisFrame","loaded","requestors","atlasImage","setLoaded","_notify","addImage","valid","_validateStretch","_validateContent","stretch","updateImage","oldImage","removeImage","listImages","getImages","hasAllDependencies","getPixelSize","getPattern","pattern","_updatePatternAtlas","atlasTexture","CLAMP_TO_EDGE","dst","beginFrame","INF","edt","gridSize","edt1d","qr","GlyphManager","requestManager","localIdeographFontFamily","entries","setURL","getGlyphs","entry","requests","ranges","_tinySDF","range","loadGlyphRange","_doesCharSupportLocalGlyph","cb","fontFamily","tinySDF","fontWeight","TinySDF","fontSize","cutoff","draw","glyphWidth","glyphHeight","glyphLeft","glyphTop","glyphAdvance","isDoubleResolution","fontstack","urlTemplate","Glyphs","Protobuf","parseGlyphPbf","fontStyle","_createCanvas","textBaseline","textAlign","fillStyle","gridOuter","gridInner","actualBoundingBoxAscent","actualBoundingBoxDescent","actualBoundingBoxLeft","actualBoundingBoxRight","measureText","clearRect","fillText","imgData","LightPositionProperty","azimuthal","polar","sphericalToCartesian","lightProperties","Light","lightOptions","_transitionable","_transitioning","getLight","LineAtlas","nextRow","dashEntry","getDash","dasharray","addDash","getDashRanges","lineAtlasWidth","isDash","zeroLength","currentDashLength","dashLength","addRoundDash","halfStretch","currIndex","distLeft","distRight","minDist","signedDistance","distMiddle","distEdge","addRegularDash","ALPHA","REPEAT","ThrottledInvoker","_callback","_triggered","MessageChannel","_channel","port2","onmessage","trigger","port1","postMessage","remove","Actor","mapId","receive","targetMapId","tasks","cancelCallbacks","mustQueue","taskQueue","invoker","processTask","process","task","callbacks","globalScope","random","buffers","hasCallback","sourceMapId","transfer","responseMessage","getWorkerSource","Dispatcher","workerPool","actors","currentActor","workers","acquire","broadcast","getActor","mapRemoved","release","loadTileJson","tileJSON","vector_layers","vectorLayers","vectorLayerIds","Source","earthRadius","LngLat","lng","lat","toArray","distanceTo","lngLat","lat1","lat2","lon","LngLatBounds","sw","ne","setSouthWest","setNorthEast","_ne","_sw","sw2","ne2","getCenter","getSouthWest","getNorthEast","getNorthWest","getWest","getNorth","getSouthEast","getEast","getSouth","contains","lnglat","containsLongitude","latAccuracy","lngAccuracy","earthCircumfrence","circumferenceAtLatitude","latitude","mercatorXfromLng","mercatorYfromLat","mercatorZfromAltitude","altitude","lngFromMercatorX","latFromMercatorY","MercatorCoordinate","lngLatLike","toLngLat","toAltitude","meterInMercatorCoordinateUnits","TileBounds","validateBounds","tileID","level","VectorTileSource","dispatcher","eventedParent","load","_loaded","dataType","_tileJSONRequest","_options","_requestManager","sourceCaches","clearTiles","tileBounds","sourceDataType","reparseOverscaled","_collectResourceTiming","collectResourceTiming","hasTile","setSourceProperty","setTiles","setUrl","loadTile","tile","getPixelRatio","Tile","overscaledZ","overscaleFactor","showCollisionBoxes","resourceTiming","_refreshExpiredTiles","setExpiryData","loadVectorData","reloadCallback","abortTile","unloadTile","unloadVectorData","RasterTileSource","roundZoom","expiry","getTileTexture","extTextureFilterAnisotropic","texParameterf","TEXTURE_MAX_ANISOTROPY_EXT","extTextureFilterAnisotropicMax","saveTileTexture","getMercCoords","CanonicalTileID","calculateKey","quadkey","mask","getQuadkey","isChildOf","dz","getTilePoint","UnwrappedTileID","OverscaledTileID","scaledTo","targetZ","zDifference","calculateScaledKey","withWrap","children","sourceMaxZoom","isLessThan","wrapped","unwrapTo","toUnwrapped","_idx","MIN_SAFE_INTEGER","ele","pixels","unpack","getUnpackVector","getPixels","backfillBorder","borderTile","xMin","xMax","yMin","yMax","RasterDEMTileSource","dem","needsHillshadePrepare","needsTerrainPrepare","neighboringTiles","_getNeighboringTiles","rawImageData","VideoFrame","fillRect","isOffscreenCanvasDistorted","timestamp","swapBR","copyTo","destRowOffset","sourceLeft","sourceTop","computeVideoFrameParameters","close","readImageUsingVideoFrame","readImageNow","pxw","nxw","backfilled","demTexture","fbo","GeoJSONSource","_updateWorkerData","_data","_removed","_pendingLoads","workerOptions","geojsonVtOptions","superclusterOptions","minPoints","setData","diff","setClusterOptions","getClusterExpansionZoom","clusterId","getClusterChildren","getClusterLeaves","limit","dataDiff","abandoned","assign","rasterBoundsAttributes","ImageSource","newCoordinates","successCallback","_request","_finishLoading","prepare","boundsBuffer","_boundsArray","boundsSegments","simpleSegment","newTilesLoaded","setCoordinates","cornerCoords","fromLngLat","dMax","getCoordinatesCenterTileID","tileCoords","buckets","VideoSource","muted","onloadstart","getVideo","loop","triggerRepaint","play","readyState","paused","pause","seek","seconds","seekableRange","seekable","currentTime","CanvasSource","getElementById","_hasInvalidDimensions","_playing","animate","getCanvas","registeredSources","getSourceType","getPixelPosMatrix","mat4.create","mat4.translate","mat4.scale","mat4.multiply","calculatePosMatrix","queryRenderedFeatures","sourceCache","styleLayers","serializedLayers","has3DLayer","layerID","queryIncludes3DLayer","maxPitchScaleFactor","tilesIn","sortTilesIn","renderedFeatureLayers","tileIn","wrappedTileID","queryResults","_state","cameraQueryGeometry","wrappedIDLayerMap","wrappedID","wrappedIDLayers","tileFeatures","wrappedIDFeatures","resultFeatures","tileFeature","mergeRenderedFeatureLayers","featureWrapper","getFeatureState","idA","idB","DictionaryCoder","strings","_stringToNumber","_numberToString","encode","GeoJSONFeature","vectorTileFeature","_vectorTileFeature","_z","_x","_y","evaluateProperties","serializedProperties","styleLayerProperties","getBounds","topDownFeatureComparator","grid3D","featureIndexArray","loadVTLayers","vtLayers","vt","rawTileData","sourceLayerCoder","sourceFeatureState","featureFilter","queryPadding","matching","cameraBounds","matching3D","bx1","by1","bx2","by2","boxX1","boxY1","boxX2","boxY2","corner","polygonIntersectsBox","previousIndex","featureGeometry","loadMatchingFeature","filterLayerIDs","layerIDs","bucketLayerIDs","arraysIntersect","sourceLayerName","getId","getState","serializedLayer","intersectionZ","geojsonFeature","layerResult","lookupSymbolFeatures","symbolFeatureIndexes","symbolFeatureIndex","hasLayer","sourceLayerId","timeAdded","fadeEndTime","uses","expirationTime","hasSymbolBuckets","dependencies","rtt","rttCoords","expiredRequestCount","registerFadeDuration","wasRequested","clearTextures","justReloaded","hasData","latestFeatureIndex","latestRawTileData","getLayer","lId","deserializeBucket","imageAtlas","glyphAtlasImage","imageAtlasTexture","glyphAtlasTexture","getBucket","querySourceFeatures","_geojsonTileLayer","patternsLoaded","parsedCC","header","$0","$1","$2","$3","maxAge","parseCacheControl","getTime","isExpired","delta","getExpiryTimeout","setFeatureState","sourceLayerStates","holdingForFade","symbolFadeHoldUntil","symbolFadeFinished","clearFadeHold","setHoldDuration","setDependencies","namespace","dep","hasDependency","namespaces","TileCache","reset","removedData","timeout","clearTimeout","order","expiryTimeout","dataWrapper","_getAndRemoveByKey","getAndRemove","getByKey","dataIndex","setMaxSize","filterFn","removed","SourceFeatureState","stateChanges","deletedStates","updateState","featureId","newState","ft","removeFeatureState","reconciledState","featureDeletions","initializeTileState","coalesceChanges","featuresChanged","layerStates","SourceCache","_sourceLoaded","_paused","reload","_didEmitContent","_sourceErrored","_source","createSource","_tiles","_cache","_unloadTile","_timers","_cacheTimers","_maxTileCacheSize","_maxTileCacheZoomLevels","_loadedParentTiles","_coveredTiles","_updated","used","usedForTerrain","getSource","resume","shouldReload","_shouldReloadOnResume","_loadTile","_abortTile","getIds","compareTileId","getRenderableIds","symbolLayer","renderables","_isIdRenderable","a_","b_","rotatedA","rotatedB","hasRenderableParent","parentTile","findLoadedParent","_reloadTile","_tileLoaded","previousState","refreshedUponExpiration","_setTileReloadTimer","_backfillDEM","borderId","getTileByID","fillBorder","getTile","_retainLoadedChildren","idealTiles","maxCoveringZoom","retain","topmostLoadedID","parentID","minCoveringZoom","parentTileID","_getLoadedTile","updateCacheSize","widthInTiles","heightInTiles","viewDependentMaxSize","handleWrapJump","wrapDelta","_prevLng","idealTileIDs","getVisibleUnwrappedCoordinates","unwrapped","coveringTiles","coveringZoomLevel","maxOverzooming","maxUnderzooming","parents","parent2","noPendingDataEmissions","_updateRetainedTiles","isRasterType","parentsForFading","fadingTiles","_addTile","idealRasterTileIDs","missingTileIDs","retainedId","keysDifference","_fadeDuration","_removeTile","_updateLoadedParentTileCache","releaseSymbolFadeTiles","checked","missingTiles","childCoord","childTile","parentWasRequested","parentId","tileKey","currentId","pointQueryGeometry","tileResults","cameraPointQueryGeometry","getCameraQueryGeometry","pointCoordinate","tileSpaceBounds","tileSpaceQueryGeometry","tileSpaceCameraQueryGeometry","getVisibleCoordinates","posMatrix","reloadTilesForDependencies","aWrap","bWrap","PRELOAD_POOL_ID","WorkerPool","active","workerCount","Worker","numActive","terminate","isPreloaded","availableLogicalProcessors","globalWorkerPool","getGlobalWorkerPool","globalThis","PathInterpolator","points_","padding_","_distances","paddedLength","lerp","distOfCurrentIdx","distToTarget","idxOfPrevPoint","distOfPrevIdx","segmentLength","segmentT","overlapAllowed","overlapA","overlapB","allowed","GridIndex","cellSize","boxCells","circleCells","xCellCount","yCellCount","circleKeys","boxKeys","circles","xScale","yScale","boxUid","circleUid","keysLength","_insertBoxCell","insertCircle","_insertCircleCell","_query","hitTest","overlapMode","predicate","hitTestCircle","_queryCellCircle","queryArgs","boxCell","circleCell","_circleAndRectCollide","_circlesCollide","_convertToXCellCoord","_convertToYCellCoord","r1","r2","bothRadii","circleX","circleY","halfRectWidth","distX","halfRectHeight","distY","getLabelPlaneMatrix","pitchWithMap","rotateWithMap","mat4.rotateZ","labelPlaneMatrix","getGlCoordMatrix","mat4.clone","glCoordMatrix","matrix","getElevation","xyTransformMat4","signedDistanceFromCamera","getPerspectiveRatio","isVisible","anchorPos","clippingBuffer","updateLineLabels","keepUpright","rotateToLine","partiallyEvaluatedSize","symbolSize.evaluateSizeForZoom","placedSymbols","aspectRatio","useVertical","hideGlyphs","perspectiveRatio","symbolSize.evaluateSizeForFeature","pitchScaledFontSize","tileAnchorPoint","projectionCache","projections","offsets","placeUnflipped","placeGlyphsAlongLine","notEnoughRoom","needsFlipping","placeFirstAndLastGlyph","fontScale","flip","glyphEndIndex","lineEndIndex","firstGlyphOffset","lastGlyphOffset","firstPlacedGlyph","placeGlyphAlongLine","lastPlacedGlyph","requiresOrientationChange","firstPoint","lastPoint","placedGlyphs","firstAndLastGlyph","orientationChange","glyphIndex","tileVertexIndex","tileSegmentEnd","projectedVertex","projectTruncatedLineSegment","singleGlyph","previousTilePoint","currentTilePoint","previousProjectedPoint","minimumLength","projectionMatrix","projectedUnitVertex","projectedUnitSegment","projectVertexToViewport","projectionArgs","distanceFromAnchor","previousVertex","direction","absOffsetX","projection","previousLineVertexIndex","transformToOffsetNormal","segmentVector","findOffsetIntersectionPoint","prevToCurrentOffsetNormal","offsetPreviousVertex","offsetCurrentVertex","currentToNextOffsetNormal","offsetNextSegmentBegin","offsetNextSegmentEnd","aDeltaY","aDeltaX","bDeltaY","bDeltaX","denominator","aInterpolation","findLineIntersection","offsetX","anchorSegment","combinedOffsetX","offsetIntersectionPoint","currentSegmentDistance","pathVertices","currentLineSegment","prevToCurrent","segmentAngle","hiddenGlyphAttributes","viewportPadding","CollisionIndex","ignoredGrid","pitchfactor","_pitch","screenRightBoundary","screenBottomBoundary","gridRightBoundary","gridBottomBoundary","perspectiveRatioCutoff","placeCollisionBox","collisionBox","textPixelRatio","collisionGroupPredicate","projectedPoint","projectAndGetPerspectiveRatio","tileToViewport","tlX","tlY","brX","brY","isInsideGrid","offscreen","isOffscreen","placeCollisionCircles","labelToScreenMatrix","showCollisionCircles","circlePixelDiameter","textPixelPadding","placedCollisionCircles","tileUnitAnchorPoint","screenAnchorPoint","projection.project","projection.getPerspectiveRatio","labelPlaneFontScale","labelPlaneAnchorPoint","projection.placeFirstAndLastGlyph","collisionDetected","inGrid","entirelyOffscreen","screenPlaneMin","screenPlaneMax","interpolator","projectedPath","circleDist","screenSpacePath","minPoint","maxPoint","clippedLines","clippedLine","p0","clipLine","seg","numCircles","circlePosition","centerX","centerY","queryRenderedSymbols","viewportQueryGeometry","gridPoint","seenFeatures","featureKey","bucketInstanceId","intersectionTests.polygonIntersectsPolygon","insertCollisionBox","ignorePlacement","collisionGroupID","insertCollisionCircles","collisionCircles","projection.xyTransformMat4","getViewportMatrix","Anchor","TextAnchorEnum","pixelValue","OpacityState","prevState","increment","placed","skipFade","opacity","JointOpacityState","placedText","placedIcon","JointPlacement","CollisionCircleArray","invProjMatrix","viewportMatrix","RetainedQueryData","CollisionGroups","crossSourceCollisions","maxGroupID","collisionGroups","ID","nextGroupID","calculateVariableLayoutShift","textOffset","shiftVariableCollisionBox","shiftX","shiftY","rotatedOffset","Placement","prevPlacement","collisionIndex","placements","opacities","variableOffsets","stale","commitTime","retainedQueryData","collisionCircleArrays","placedOrientations","getBucketParts","sortAcrossTiles","symbolBucket","bucketFeatureIndex","pixelsToTiles","textLabelPlaneMatrix","projection.getLabelPlaneMatrix","glMatrix","projection.getGlCoordMatrix","partiallyEvaluatedTextSize","collisionGroup","attemptAnchorPlacement","textAnchorOffset","textOverlapMode","orientation","placedGlyphBoxes","prevAnchor","markUsedJustification","markUsedOrientation","placeLayerBucketPart","bucketPart","seenCrossTileIDs","textOptional","iconOptional","textAlwaysOverlap","iconOverlapMode","iconAlwaysOverlap","hasIconTextFit","zOrderByViewportY","alwaysShowText","alwaysShowIcon","placeSymbol","placeText","placeIcon","placedVerticalText","placedGlyphCircles","placedIconBoxes","updatePreviousOrientationIfNotPlaced","isPlaced","previousOrientation","prevPlacedOrientation","placeTextForPlacementModes","placeHorizontalFn","placeVerticalFn","placementMode","textAnchorOffsetStart","textAnchorOffsetEnd","placeBox","collisionTextBox","placedFeature","_b","placeBoxForVariableAnchors","collisionIconBox","variableIconBox","placedBox","placementPasses","prevOrientation","prevOffset","placeIconFeature","shiftedIconBox","iconWithoutText","textWithoutIcon","circleArray","symbolIndexes","symbolIndex","mat4.invert","placedAnchor","autoIndex","getAnchorJustification","indexes","horizontalOnly","horizontalIndexes","commit","zoomAtLastRecencyCheck","placementChanged","prevZoomAdjustment","zoomAdjustment","symbolFadeChange","prevOpacities","prevOffsets","prevOrientations","jointPlacement","prevOpacity","jointOpacity","lastPlacementChangeTime","updateLayerOpacities","updateBucketOpacities","duplicateOpacityState","textAllowOverlap","iconAllowOverlap","hasVariablePlacement","defaultOpacityState","addOpacities","PACKED_HIDDEN_OPACITY","opacityState","horizontalHidden","verticalHidden","packedOpacity","packOpacity","symbolHidden","useHorizontal","variableOffset","updateCollisionVertices","verticalIconUsed","instance","hasTransitions","stillRecent","durationAdjustment","setStale","notUsed","shift25","shift24","shift17","shift16","shift9","shift8","shift1","targetBit","opacityBits","LayerPlacement","_sortAcrossTiles","_currentTileIndex","_currentPartIndex","_seenCrossTileIDs","_bucketParts","continuePlacement","placement","shouldPausePlacement","bucketParts","PauseablePlacement","forceFullPlacement","_currentPlacementIndex","_forceFullPlacement","_showCollisionBoxes","_done","isDone","layerTiles","startTime","placementZoom","_inProgressLayer","ARRAY_TYPES","KDBush","magic","versionAndType","ArrayType","nodeSize","numItems","IndexArrayType","arrayTypeIndex","coordsByteSize","idsByteSize","padCoords","_pos","_finished","numAdded","axis","qy","sqDist","select","swapItem","roundingFactor","TileLayerIndex","_symbolsByKey","symbolInstancesByKey","Map","instances","symbols","crossTileIDs","getScaledCoordinates","childTileID","localX","localY","localZ","yWorld","yOffset","findMatches","newTileID","zoomCrossTileIDs","scaledSymbolCoord","thisTileSymbol","getCrossTileIDsLists","CrossTileIDs","maxCrossTileID","generate","CrossTileSymbolLayerIndex","usedCrossTileIDs","zoomIndexes","newZoomIndex","addBucket","removeBucketCrossTileIDs","childIndex","parentIndex","removedBucket","removeStaleBuckets","currentIDs","tilesChanged","CrossTileSymbolIndex","layerIndexes","maxBucketInstanceId","bucketsInCurrentPlacement","layerIndex","symbolBucketsChanged","currentBucketIDs","pruneUnusedLayers","usedLayers","usedLayerMap","usedLayer","_emitValidationErrors","supportedDiffOperations","diffOperations","ignoredDiffOperations","empty","styleKey","emptyStyle","Style","_getMapId","glyphManager","lineAtlas","crossTileSymbolIndex","_spritesImagesIds","_layers","_order","_availableImages","_resetUpdates","_rtlTextPluginCallback","registerForPluginStateChange","elem","_validateLayer","loadURL","previousStyle","_load","loadJSON","loadEmpty","nextState","transformStyle","stylesheet","_loadSprite","_createLayers","setTerrain","dereferencedLayers","_serializedLayers","styledLayer","isUpdate","completion","_spriteRequest","originalSprite","spriteArray","spriteArrayLength","combinedRequestsMap","jsonRequestParameters","SpriteJSON","jsonRequestKey","imageRequestParameters","SpriteImage","imageRequestKey","requst","loadSprite","spriteId","imagesToRemove","_changedImages","imageId","_changed","_unloadSprite","flat","_updatedSources","_serializeByIds","serializedLayersDictionary","_serializedAllLayers","allLayerIds","_checkLoaded","changed","updatedIds","_updatedLayers","removedIds","_removedLayers","_updateWorkerLayers","action","_reloadSource","_clearSource","_updateTilesForChangedImages","_updateTilesForChangedGlyphs","_updatedPaintProps","sourcesUsedBefore","changedImages","_glyphsDidChange","serializedStyle","changes","removeOrAddSourceCommands","diffSources","beforeLayers","beforeOrder","afterOrder","beforeIndex","afterIndex","tracker","clean","beforeLayer","afterLayer","insertBeforeLayerId","lastIndexOf","diffLayers","diffStyles","unimplementedOps","_afterImageUpdated","isSourceLoaded","geojsonSource","layerObject","validateCustomStyleLayer","_layerOrderChanged","_updateLayer","moveLayer","newIndex","getLayersOrder","getFilter","getTerrain","myStyleSheet","_flattenAndSortRenderedFeatures","sourceResults","isLayer3D","features3D","sourceResult","layerFeatures","topmost3D","includedSources","renderedSymbols","bucketQueryData","queryData","bucketSymbols","layerSymbols","sortedA","symbolFeature","layerName","dataTiles","dataID","addSourceType","SourceType","setSourceType","workerSourceURL","_update","_remove","rtlTextPluginEvented","_updateSources","_generateCollisionBoxes","_updatePlacement","placementCommitted","layerBucketsChanged","pauseablePlacement","_releaseSymbolFadeTiles","getResource","getGlyphsUrl","glyphsUrl","addSprite","spriteToAdd","updatedSprite","removeSprite","internalSpriteRepresentation","find","findIndex","getSprite","posAttributes","terrainVert","shaders","prelude","compile","backgroundPattern","clippingMask","heatmapTexture","collisionCircle","debug","fillOutline","fillOutlinePattern","fillPattern","fillExtrusion","fillExtrusionPattern","hillshadePrepare","lineGradient","linePattern","lineSDF","symbolIcon","symbolSDF","symbolTextAndIcon","terrainDepth","terrainCoords","fragmentSource","vertexSource","re","staticAttributes","fragmentUniforms","vertexUniforms","staticUniforms","fragmentPragmas","operation","precision","attrType","unpackType","VertexArrayObject","boundProgram","boundLayoutVertexBuffer","boundPaintVertexBuffers","boundIndexBuffer","boundVertexOffset","boundDynamicVertexBuffer","vao","program","paintVertexBuffers","dynamicVertexBuffer","dynamicVertexBuffer2","dynamicVertexBuffer3","paintBuffersDiffer","boundDynamicVertexBuffer2","boundDynamicVertexBuffer3","freshBind","bindVertexArray","dynamicDraw","numNextAttributes","numAttributes","createVertexArray","enableAttributes","vertexBuffer","setVertexAttribPointers","currentNumAttributes","deleteVertexArray","getTokenizedAttributesAndUniforms","token","Program","configuration","fixedUniforms","showOverdrawInspector","createProgram","staticAttrInfo","dynamicAttrInfo","allAttrInfo","preludeUniformsInfo","staticUniformsInfo","dynamicUniformsInfo","uniformList","allUniformsInfo","fragmentShader","createShader","FRAGMENT_SHADER","failedToCreate","shaderSource","compileShader","getShaderParameter","COMPILE_STATUS","getShaderInfoLog","attachShader","vertexShader","VERTEX_SHADER","attributes","uniformLocations","bindAttribLocation","linkProgram","getProgramParameter","LINK_STATUS","getProgramInfoLog","deleteShader","it","uniformLocation","getUniformLocation","terrainUniforms","u_depth","u_terrain","u_terrain_dim","u_terrain_matrix","u_terrain_unpack","u_terrain_exaggeration","terrainPreludeUniforms","drawMode","depthMode","stencilMode","colorMode","cullFaceMode","uniformValues","currentProperties","dynamicLayoutBuffer","dynamicLayoutBuffer2","dynamicLayoutBuffer3","setDepthMode","setStencilMode","setColorMode","setCullFace","activeTexture","TEXTURE2","depthTexture","TEXTURE3","primitiveSize","LINES","TRIANGLES","LINE_STRIP","drawElements","UNSIGNED_SHORT","patternUniformValues","tileRatio","numTiles","tileSizeAtNearestZoom","pixelX","pixelY","u_image","u_texsize","u_scale","u_fade","u_pixel_coord_upper","u_pixel_coord_lower","fillExtrusionUniformValues","shouldUseVerticalGradient","_lp","lightPos","lightMat","mat3.create","mat3.fromRotation","vec3.transformMat3","lightColor","u_matrix","u_lightpos","u_lightintensity","u_lightcolor","u_vertical_gradient","u_opacity","fillExtrusionPatternUniformValues","u_height_factor","fillUniformValues","fillPatternUniformValues","fillOutlineUniformValues","drawingBufferSize","u_world","fillOutlinePatternUniformValues","circleUniformValues","extrudeScale","pixelsToGLUnits","u_camera_to_center_distance","u_scale_with_map","translatePosMatrix","u_pitch_with_map","u_device_pixel_ratio","u_extrude_scale","collisionUniformValues","u_pixels_to_tile_units","u_overscale_factor","debugUniformValues","scaleRatio","u_color","u_overlay","u_overlay_scale","clippingMaskUniformValues","heatmapUniformValues","u_intensity","getTileLatRange","lineUniformValues","calculateMatrix","u_ratio","u_units_to_pixels","lineGradientUniformValues","imageHeight","u_image_height","linePatternUniformValues","tileZoomRatio","calculateTileRatio","lineSDFUniformValues","posA","posB","widthA","widthB","u_patternscale_a","u_patternscale_b","u_sdfgamma","u_tex_y_a","u_tex_y_b","u_mix","rasterUniformValues","parentTL","parentScaleBy","fade","u_tl_parent","u_scale_parent","u_buffer_scale","u_fade_t","mix","u_image0","u_image1","u_brightness_low","u_brightness_high","u_saturation_factor","saturation","u_contrast_factor","contrast","u_spin_weights","spinWeights","symbolIconUniformValues","rotateInShader","texSize","u_is_size_zoom_constant","u_is_size_feature_constant","u_size_t","u_size","u_pitch","u_rotate_symbol","u_aspect_ratio","u_fade_change","u_label_plane_matrix","u_coord_matrix","u_is_text","u_texture","symbolSDFUniformValues","isHalo","u_gamma_scale","u_is_halo","symbolTextAndIconUniformValues","texSizeSDF","texSizeIcon","u_texsize_icon","u_texture_icon","backgroundUniformValues","backgroundPatternUniformValues","imagePosA","imagePosB","u_pattern_tl_a","u_pattern_br_a","u_pattern_tl_b","u_pattern_br_b","u_pattern_size_a","u_pattern_size_b","u_scale_a","u_scale_b","u_tile_units_to_pixels","bgPatternUniformValues","programUniforms","u_inv_matrix","u_viewport_size","u_color_ramp","u_latrange","u_light","u_shadow","u_highlight","u_accent","u_dimension","u_zoom","u_unpack","u_ele_delta","u_terrain_coords_id","IndexBuffer","createBuffer","unbindVAO","bindElementBuffer","bufferData","ELEMENT_ARRAY_BUFFER","DYNAMIC_DRAW","STATIC_DRAW","bufferSubData","deleteBuffer","AttributeType","VertexBuffer","bindVertexBuffer","ARRAY_BUFFER","attribIndex","enableVertexAttribArray","vertexAttribPointer","WeakMap","isWebGL2","getParameter","VERSION","BaseValue","getDefault","setDefault","ClearColor","clearColor","ClearDepth","clearDepth","ClearStencil","clearStencil","ColorMask","colorMask","DepthMask","depthMask","StencilMask","stencilMask","StencilFunc","func","ALWAYS","stencilFunc","StencilOp","KEEP","stencilOp","StencilTest","enable","STENCIL_TEST","disable","DepthRange","depthRange","DepthTest","DEPTH_TEST","DepthFunc","LESS","depthFunc","Blend","BLEND","BlendFunc","ONE","ZERO","blendFunc","BlendColor","blendColor","BlendEquation","FUNC_ADD","blendEquation","CullFace","CULL_FACE","CullFaceSide","BACK","cullFace","FrontFace","CCW","frontFace","ProgramValue","useProgram","ActiveTextureUnit","TEXTURE0","Viewport","drawingBufferWidth","drawingBufferHeight","BindFramebuffer","bindFramebuffer","FRAMEBUFFER","BindRenderbuffer","bindRenderbuffer","RENDERBUFFER","BindTexture","BindVertexBuffer","bindBuffer","BindElementBuffer","BindVertexArray","getExtension","bindVertexArrayOES","PixelStoreUnpack","pixelStorei","UNPACK_ALIGNMENT","PixelStoreUnpackPremultiplyAlpha","UNPACK_PREMULTIPLY_ALPHA_WEBGL","PixelStoreUnpackFlipY","UNPACK_FLIP_Y_WEBGL","FramebufferAttachment","ColorAttachment","setDirty","framebufferTexture2D","COLOR_ATTACHMENT0","DepthAttachment","framebufferRenderbuffer","DEPTH_ATTACHMENT","DepthStencilAttachment","DEPTH_STENCIL_ATTACHMENT","Framebuffer","hasDepth","hasStencil","framebuffer","createFramebuffer","colorAttachment","depthAttachment","checkFramebufferStatus","FRAMEBUFFER_COMPLETE","renderbuffer","deleteRenderbuffer","deleteFramebuffer","ColorMode","blendFunction","Replace","disabled","unblended","alphaBlended","Context","stencilTest","depthTest","blend","cullFaceSide","MAX_TEXTURE_MAX_ANISOTROPY_EXT","maxTextureSize","MAX_TEXTURE_SIZE","HALF_FLOAT","extColorBufferHalfFloat","RGBA16F","RGBA16F_EXT","RGB16F","RGB16F_EXT","extTextureHalfFloat","HALF_FLOAT_OES","createRenderbuffer","storageFormat","rbo","renderbufferStorage","depth","stencil","COLOR_BUFFER_BIT","DEPTH_BUFFER_BIT","STENCIL_BUFFER_BIT","mode","fail","depthFail","createVertexArrayOES","deleteVertexArrayOES","DepthMode","ReadOnly","ReadWrite","StencilMode","CullFaceMode","quadTriangles","drawCollisionDebug","tileBatches","circleCount","circleOffset","invTransform","mat4.mul","colorModeForRenderPass","getTerrainData","circleProgram","vertexData","batch","circleIdx","collision","quadCount","triCount","idx","createQuadTriangles","backCCW","identityMat4","calculateVariableRenderShift","renderTextSize","updateVariableAnchorsForBucket","tileScale","updateTextFitIcon","dynamicTextLayoutVertexArray","dynamicIconLayoutVertexArray","placedTextShifts","tileAnchor","projectedAnchor","symbolProjection.project","symbolProjection.getPerspectiveRatio","tilePixelRatio","shiftedAnchor","symbolProjection.hideGlyphs","placedIcons","getSymbolProgramName","iconsInText","drawLayerSymbols","rotationAlignment","pitchAlignment","hasSortKey","depthModeForSublayer","tileRenderState","programConfiguration","sdfIcons","transformed","terrainData","atlasInterpolation","atlasInterpolationIcon","atlasTextureIcon","rotating","zooming","NEAREST","iconScaled","iconsNeedLinear","symbolProjection.getLabelPlaneMatrix","symbolProjection.getGlCoordMatrix","hasVariableAnchors","symbolProjection.updateLineLabels","uLabelPlaneMatrix","uglCoordMatrix","hasHalo","oldSegments","segmentState","TEXTURE1","drawSymbolElements","updatePatternPositionsInProgram","drawFillTiles","isOutline","fillPropertyName","programName","terrainCoord","tileMatrix","stencilModeForClipping","drawExtrusionTiles","renderHillshade","shadow","highlight","accent","moving","hillshadeUniformValues","rasterBoundsBuffer","quadTriangleIndexBuffer","rasterBoundsSegments","prepareHillshade","textureStride","pixelData","renderTexture","mat4.ortho","hillshadeUniformPrepareValues","getFadeValues","sinceTile","sinceParent","idealZ","fadeIn","childOpacity","topColor","btmColor","leftColor","rightColor","centerColor","drawHorizontalLine","drawDebugSSRect","drawVerticalLine","SCISSOR_TEST","scissor","drawDebugTile","tileRawData","tileSizeKb","tileIdText","initDebugOverlayCanvas","debugOverlayCanvas","ctx2d","shadowColor","shadowBlur","strokeStyle","strokeText","debugOverlayTexture","drawTextToOverlay","debugBuffer","debugSegments","tileBorderIndexBuffer","drawTerrain","LEQUAL","depthRangeFor3D","mesh","getTerrainMesh","renderToTexture","getTexture","getMeshFrameDelta","Painter","_tileTextures","terrainFacilitator","renderTime","setup","numSublayers","depthEpsilon","tileExtentArray","tileExtentBuffer","tileExtentSegments","debugArray","rasterBoundsArray","viewportArray","viewportBuffer","viewportSegments","tileLineStripIndices","quadTriangleIndices","stencilClearMode","nextStencilID","currentStencilSource","_renderTileClippingMasks","tileIDs","_tileClippingMaskIDs","REPLACE","stencilModeFor3D","NOTEQUAL","EQUAL","stencilConfigForOverlap","minTileZ","stencilValues","zToStencilMode","GEQUAL","_showOverdrawInspector","CONSTANT_COLOR","renderPass","opaquePassEnabledForLayer","currentLayer","opaquePassCutoff","coordsAscending","coordsDescending","coordsDescendingSymbol","reverse","prepareForRender","newTiles","tilesAfterTime","a3","a4","a5","a6","a7","a8","a9","a14","a15","projMatrix","b4","b5","b6","b7","b8","b9","b12","b13","b14","b15","glMatrix.EPSILON","mat4.copy","getRenderableTiles","getFramebuffer","devicePixelRatio","drawDepth","getCoordsTexture","coordsIndex","drawCoords","renderLayer","showTileBoundaries","selectedSource","flatMap","vectorSources","otherSources","considerSource","selectDebugSource","drawDebug","showPadding","centerPoint","drawCrosshair","drawDebugPadding","pixelToTileScale","updateVariableAnchors","drawSymbols","strokeWidth","strokeOpacity","segmentsRenderStates","segmentsState","drawCircles","numType","internalFormat","bindTextureToFramebuffer","textureUnit","colorRampUnit","heatmapTextureUniformValues","renderTextureToMap","drawHeatmap","gradient","programId","firstTile","prevProgram","programChanged","atlas","layerGradient","gradientTexture","textureResolution","potentialOverzoom","drawLine","drawFill","drawFillExtrusion","stencilModes","drawHillshade","textureFilter","drawRaster","isPatternMissing","drawBackground","setCustomLayerDefaults","customLayerMatrix","setBaseState","drawCustom","inViewportPixelUnitsUnits","sinA","cosA","translation","translatedMatrix","textures","overLimit","Frustum","planes","invProj","frustumCoords","vec4.mul","frustumPlanes","vec3.normalize","az","bz","vec3.cross","vec3.sub","Aabb","min_","max_","vec3.scale","vec3.add","quadrant","qMin","vec3.clone","qMax","distanceX","distanceY","frustum","aabbPoints","fullyInside","plane","pointsInside","projMin","MAX_VALUE","projMax","EdgeInsets","Transform","minPitch","maxPitch","renderWorldCopies","maxValidLatitude","_renderWorldCopies","_minZoom","_maxZoom","_minPitch","_maxPitch","setMaxBounds","_center","_elevation","_fov","_unmodified","_edgeInsets","_posMatrixCache","_alignedPosMatrixCache","_minEleveationForCurrentTile","that","latRange","_calcMatrices","centerOffset","rotationMatrix","mat2.create","mat2.rotate","fov","_zoom","constrainedZoom","zoomScale","_constrain","elevation","isPaddingEqual","interpolatePadding","scaleZoom","utl","utr","ubl","ubr","w0","w1","extraWorldCopy","actualZ","cameraCoord","getCameraPoint","centerCoord","cameraPoint","cameraFrustum","fromInvProjectionMatrix","radiusOfMaxLvlLodInTiles","newRootTile","aabb","fullyVisible","intersectResult","refPoint","longestDim","distanceSq","vec2.sqrLen","tileDistanceToCamera","childX","childY","childZ","minMax","getMinMaxElevation","minElevation","maxElevation","unmodified","unproject","getCameraPosition","pointLocation","_pixelPerMeter","recalculateZoom","getElevationForLngLatZoom","cameraPosition","camera","setLocationAtPoint","loc","locationCoordinate","newCenter","coordinateLocation","locationPoint","coordinatePoint","pixelMatrix3D","coordinate","coord0","coord1","pixelMatrixInverse","z0","z1","pixelMatrix","getHorizon","getMaxBounds","lngRange","unwrappedTileID","aligned","posMatrixKey","unwrappedX","alignedProjMatrix","mercatorMatrix","_constraining","sy","sx","h2","w2","cameraToSeaLevelDistance","cameraToLowestPointDistance","lowestPlane","groundAngle","fovAboveCenter","topHalfSurfaceDistance","horizon","fovCenterToHorizon","topHalfSurfaceDistanceHorizon","topHalfMinDistance","farZ","nearZ","fovy","aspect","mat4.perspective","mat4.rotateX","xShift","yShift","angleCos","angleSin","alignedM","throttle","time","lastCallArgs","pending","timerId","lastCallContext","later","Hash","hashName","_getCurrentHash","hash","_hashName","keyval","_onHashChange","_map","dragRotate","isEnabled","touchZoomRotate","getBearing","jumpTo","_updateHashUnthrottled","getHashString","history","replaceState","SecurityError","_updateHash","encodeURIComponent","addTo","mapFeedback","getZoom","getPitch","found","defaultInertiaOptions","linearity","easing","defaultPanInertiaOptions","deceleration","maxSpeed","defaultZoomInertiaOptions","defaultBearingInertiaOptions","defaultPitchInertiaOptions","HandlerInertia","_inertiaBuffer","record","settings","_drainInertiaBuffer","inertia","_onMoveEnd","panInertiaOptions","deltas","pan","pinchAround","around","zoomDelta","bearingDelta","pitchDelta","panDelta","easeOptions","calculateEasing","amount","extendDuration","noMoveStart","inertiaDuration","inertiaOptions","speed","MapMouseEvent","_defaultPrevented","defaultPrevented","originalEvent","mousePos","getCanvasContainer","MapTouchEvent","changedTouches","touchPos","lngLats","curr","MapWheelEvent","MapEventHandler","_clickTolerance","clickTolerance","_mousedownPos","wheel","_firePreventable","mousedown","mouseup","click","dblclick","mouseover","mouseout","touchstart","touchmove","touchend","touchcancel","mapEvent","isActive","BlockableMapEventHandler","_delayContextMenu","_ignoreContextMenu","_contextMenuEvent","mousemove","contextmenu","TransformProvider","_requestedCameraState","BoxZoomHandler","_tr","_el","_container","getContainer","_enabled","_active","shiftKey","disableDrag","_startPos","_lastPos","mousemoveWindow","_box","classList","_fireEvent","setTransform","mouseupWindow","suppressClick","cameraAnimation","fitScreenCoordinates","keydown","keyCode","enableDrag","indexTouches","SingleTapRecognizer","numTouches","mapTouches","timeStamp","getCentroid","newTouches","TapRecognizer","singleTap","numTaps","lastTime","lastTap","count","tap","soonEnough","closeEnough","TapZoomHandler","_zoomIn","_zoomOut","zoomInPoint","zoomOutPoint","easeTo","DragHandler","_moveStateManager","moveStateManager","_moveFunction","move","_activateOnStart","activateOnStart","assignEvents","_moved","_lastPoint","endMove","_move","dragStart","isValidStartEvent","startMove","dragMove","isValidMoveEvent","movePoint","dragEnd","isValidEndEvent","getClickTolerance","BUTTONS_FLAGS","MouseMoveStateManager","_correctEvent","checkCorrectEvent","eventButton","mouseButton","_eventButton","_e","flag","buttons","buttonNoLongerPressed","OneFingerTouchMoveStateManager","_firstTouch","_isOneFingerTouch","targetTouches","_isSameTouchEvent","handler","generateMouseRotationHandler","bearingDegreesPerPixelMoved","mouseMoveStateManager","ctrlKey","generateMousePitchHandler","pitchDegreesPerPixelMoved","TouchPanHandler","_minTouches","cooperativeGestures","_touches","_sum","_cancelCooperativeMessage","_calculateTransform","_cooperativeGestures","_onCooperativeGesture","touchPointSum","touchDeltaSum","touchDeltaCount","prevPoint","TwoFingersTouchHandler","_firstTwoTouches","_start","getTouchById","_aroundCenter","getZoomDelta","lastDistance","TwoFingersTouchZoomHandler","_distance","_startDistance","getBearingDelta","TwoFingersTouchRotateHandler","_minDiameter","_startVector","_vector","lastVector","_isBelowThreshold","threshold","bearingDeltaSinceStart","isVertical","TwoFingersTouchPitchHandler","_valid","_firstMove","_lastPoints","_currentTouchCount","vectorA","vectorB","gestureBeginsVertically","movedA","movedB","isSameDirection","defaultOptions","panStep","bearingStep","pitchStep","KeyboardHandler","stepOptions","_panStep","_bearingStep","_pitchStep","_rotationDisabled","altKey","metaKey","zoomDir","bearingDir","pitchDir","xDir","yDir","easeId","easeOut","disableRotation","enableRotation","wheelZoomDelta","ScrollZoomHandler","triggerRenderFrame","_onTimeout","initialEvent","_type","_delta","_lastValue","_triggerRenderFrame","_defaultZoomRate","_wheelZoomRate","setZoomRate","zoomRate","setWheelZoomRate","wheelZoomRate","_finishTimeout","isZooming","_zooming","_metaKey","deltaMode","WheelEvent","DOM_DELTA_LINE","deltaY","timeDelta","_lastWheelEventTime","_timeout","_lastWheelEvent","_frameId","_around","_aroundPoint","renderFrame","_targetZoom","_startZoom","_easing","_smoothOutEasing","targetZoom","startZoom","finished","noInertia","needsRenderFrame","_prevEase","currentEase","DoubleClickZoomHandler","clickZoom","TapZoom","_clickZoom","_tapZoom","ClickZoomHandler","TapDragZoomHandler","_tap","_swipePoint","_swipeTouch","_tapTime","_tapPoint","swipePoint","newSwipePoint","DragPanHandler","mousePan","touchPan","_mousePan","_touchPan","_inertiaOptions","DragRotateHandler","mouseRotate","mousePitch","_pitchWithRotate","pitchWithRotate","_mouseRotate","_mousePitch","TwoFingersTouchZoomRotateHandler","touchZoom","touchRotate","tapDragZoom","_touchZoom","_touchRotate","_tapDragZoom","isMoving","drag","RenderFrameEvent","hasChange","HandlerManager","handleWindowEvent","handleEvent","eventName","_updatingCamera","inputEvent","mergedHandlerResult","eventsInProgress","activeHandlers","eventTouches","_getMapTouches","handlerName","_handlers","_blockedByActive","mergeHandlerResult","deactivatedHandlers","_previousActiveHandlers","_changes","_stop","_inertia","_fireEvents","_handlersById","_bearingSnap","bearingSnap","_eventsInProgress","_addDefaultHandlers","passive","listenerOptions","boxZoom","interactive","tapZoom","doubleClickZoom","touchPitch","generateMousePanHandler","dragPan","scrollZoom","keyboard","allowEndAnimation","isRotating","myName","handlerResult","eventData","_applyChanges","combined","combinedEventsInProgress","combinedDeactivatedHandlers","change","_updateMapTransform","combinedResult","_getTransformForUpdate","_terrainMovement","_elevationFreeze","_applyUpdatedTransform","newEventsInProgress","wasMoving","nowMoving","startEvents","endEvents","originalEndEvent","stillMoving","inertialEase","shouldSnapToNorth","essential","resetNorth","freezeElevation","_requestFrame","_renderTaskQueue","Camera","_renderFrameCallback","_easeStart","_easeOptions","_onEaseFrame","_easeFrameId","_requestRenderFrame","_moving","panBy","panTo","zoomTo","zoomIn","zoomOut","getPadding","setPadding","rotateTo","resetNorthPitch","snapToNorth","cameraForBounds","_cameraForBoxAndBearing","defaultPadding","edgePadding","p0world","p1world","p0rotated","p1rotated","upperRight","lowerLeft","scaleX","scaleY","rotatedPaddingOffset","offsetAtFinalZoom","fitBounds","_fitInternal","calculatedOptions","flyTo","zoomChanged","bearingChanged","pitchChanged","calculateCameraOptionsFromTo","altitudeFrom","altitudeTo","fromMerc","toMerc","distance3D","groundDistance","startBearing","startPitch","startPadding","_normalizeBearing","offsetAsPoint","pointAtOffset","locationAtOffset","_normalizeCenter","finalScale","aroundPoint","currently","_rotating","pitching","_pitching","_padding","_easeId","_prepareEase","_prepareElevation","_ease","_updateElevation","speedup","_fireMoveEvents","interruptingEaseId","_finalizeElevation","_afterEase","_elevationCenter","_elevationStart","_elevationTarget","getMinTileElevationForLngLatZoom","pitch1","transformCameraUpdate","nextTransform","wasZooming","wasRotating","wasPitching","coercedOptions","curve","rho","u1","wMax","rho2","zoomOutFactor","descent","sinh","cosh","r0","S","screenSpeed","maxDuration","isEasing","allowGestures","_cancelRenderFrame","_onEaseEnd","onEaseEnd","handlers","currentBearing","queryTerrainElevation","AttributionControl","_toggleAttribution","setAttribute","removeAttribute","_updateData","_updateAttributions","_updateCompact","offsetWidth","_compact","_updateCompactMinimize","getDefaultPosition","compact","_compactButton","_setElementTitle","_innerContainer","_attribHTML","element","title","_getUIString","attributions","customAttribution","styleOwner","owner","styleId","attrib","attribHTML","innerHTML","_editLink","LogoControl","containerChildren","rel","display","TaskQueue","_queue","_id","_cleared","_currentlyRunning","running","run","PerformanceMarkers","lastFrameTime","frameTimes","frameTimeTarget","loadTimeKey","fullLoadTimeKey","PerformanceUtils","mark","marker","currTimestamp","clearMetrics","clearMeasures","clearMarks","getPerformanceMetrics","measure","fullLoad","loadTime","getEntriesByName","fullLoadTime","totalFrames","fps","droppedFrames","frameTime","percentDroppedFrames","defaultLocale","pos3dAttributes","TerrainSourceCache","_renderableTilesKeys","_sourceTileCache","deltaZoom","destruct","freeRtt","getTerrainCoords","_tileID","getSourceTile","searchForDEM","Terrain","qualityFactor","meshSize","_demMatrixCache","_coordsTextureSize","getDEMElevation","vec2.transformMat4","mercatorX","mercatorY","_getOverscaledTileIDFromLngLatZoom","_emptyDemTexture","_emptyDepthTexture","_emptyDemUnpack","_emptyDemMatrix","sourceTile","matrixKey","demMatrix","mat4.fromScaling","_fboDepthTexture","_fbo","_fboCoordsTexture","DEPTH_COMPONENT16","_coordsTexture","readPixels","coordsSize","_allowMercatorOverflow","_mesh","meshSize2","offsetTop","offsetBottom","offsetLeft","offsetRight","mercatorCoordinate","tileX","tileY","inLeftHalf","centerLng","RenderPool","_context","_size","_tileSize","_objects","_recentlyUsed","_stamp","_createObject","DEPTH_STENCIL","stamp","inUse","getObjectForId","useObject","stampObject","getOrCreateFreeObject","freeObject","freeAllObjects","isFull","LAYERS","RenderToTexture","pool","_stacks","_prevType","_rttTiles","_renderableTiles","_renderableLayerIds","_coordsDescendingInv","_coordsDescendingInvStr","isLastLayer","packageJSON","attributionControl","maplibreLogo","failIfMajorPerformanceCaveat","preserveDrawingBuffer","trackResize","refreshExpiredTiles","maxTileCacheSize","maxTileCacheZoomLevels","maxCanvasSize","touchmoveWindow","showCompass","showZoom","visualizePitch","MouseRotateWrapper","startMouse","moveMouse","offTemp","startTouch","moveTouch","mapRotateTolerance","mapPitchTolerance","touchMoveStateManager","generateOneFingerTouchRotationHandler","generateOneFingerTouchPitchHandler","supportsGeolocation","smartWrap","priorPos","anchorTranslate","applyAnchorClass","prefix","Marker","_onKeyPress","legacyCode","charCode","togglePopup","_onMapClick","targetElement","_element","_popup","isFullyLoaded","_lngLat","_offset","rotation","_rotationAlignment","_rotation","_pitchAlignment","_anchor","_opacityTimeout","metresPerPixel","_onMove","_isDragging","_pointerdownPos","_positionDelta","setLngLat","pointerEvents","_onUp","_addDragHandler","_color","_scale","_draggable","draggable","_defaultMarker","svg","createNS","defaultHeight","defaultWidth","setAttributeNS","markerLarge","page1","ellipses","rx","ry","ellipse","bgPath","borderPath","maki","circleContainer","circle1","circle2","setDraggable","getLngLat","getElement","setPopup","popup","_originalTabIndex","markerHeight","markerRadius","linearOffset","SQRT2","getAttribute","getPopup","isOpen","getOffset","setOffset","addClassName","removeClassName","toggleClassName","toggle","shouldBeDraggable","isDraggable","setRotation","getRotation","setRotationAlignment","getRotationAlignment","setPitchAlignment","getPitchAlignment","positionOptions","enableHighAccuracy","maximumAge","fitBoundsOptions","trackUserLocation","showAccuracyCircle","showUserLocation","numberOfWatches","noTimeout","updateScale","clientHeight","maxMeters","maxFeet","setScale","maxDistance","pow10","multiplier","getDecimalRoundNum","getRoundNum","closeButton","closeOnClick","focusAfterOpen","focusQuerySelector","normalizeOffset","cornerOffset","convertedOffset","Debug","logToElement","overwrite","MapLibreGL","maxParallelImageRequests","numRequests","workerUrl","customProtocol","loadFn","_cooperativeGesturesOnWheel","_contextLost","_frame","_contextRestored","_setupPainter","_onMapScroll","scrollTop","scrollLeft","_onWindowOnline","_interactive","platform","_failIfMajorPerformanceCaveat","_preserveDrawingBuffer","_antialias","antialias","_trackResize","_crossSourceCollisions","_crossFadingFactor","_controls","_mapId","_locale","_overridePixelRatio","_maxCanvasSize","_imageQueueHandle","HTMLElement","maxBounds","_setupContainer","_idleTriggered","initialResizeEventCaptured","throttledResizeCallback","_resizeObserver","ResizeObserver","observe","_setupCooperativeGestures","_hash","_localIdeographFontFamily","_validateStyle","addControl","logoPosition","control","controlElement","positionContainer","_controlPositions","insertBefore","firstChild","removeControl","ci","hasControl","_containerDimensions","clampedPixelRatio","_getClampedPixelRatio","_resizeCanvas","fireMoving","maxCanvasWidth","maxCanvasHeight","canvasWidth","canvasHeight","setPixelRatio","setMinZoom","getMinZoom","setMaxZoom","getMaxZoom","setMinPitch","getMinPitch","setMaxPitch","getMaxPitch","getRenderWorldCopies","setRenderWorldCopies","getCooperativeGestures","setCooperativeGestures","gestureOptions","_destroyCooperativeGestures","_createDelegatedListener","mousein","delegates","delegate","layerIdOrListener","delegatedListener","_delegatedListeners","delegatedListeners","removeDelegatedListener","geometryOrOptions","isGeometry","_diffStyle","_updateStyle","_lazyInitEmptyStyle","_updateDiff","getStyle","isStyleLoaded","_terrainDataCallback","thisLayer","areTilesLoaded","existingImage","imageData","hasImage","loadImage","beforeId","spriteUrl","_canvasContainer","_canvas","clientWidth","canvasContainer","controlContainer","_controlContainer","positionName","_cooperativeGesturesScreen","desktopMessage","windowsHelpText","macHelpText","mobileHelpText","webglcontextcreationerrorDetailObject","requestedAttributes","statusMessage","msg","metaPress","_styleDirty","_sourcesDirty","updateStyle","_render","paintStartTimeStamp","crossFading","_placementDirty","somethingDirty","_repaint","_fullyLoaded","redraw","disconnect","loseContext","_showTileBoundaries","_showPadding","repaint","_vertices","getCameraTargetElevation","NavigationControl","_updateZoomButtons","isMax","isMin","_zoomInButton","_zoomOutButton","_rotateCompassArrow","_compassIcon","_setButtonTitle","_createButton","_compass","_handler","GeolocateControl","_onSuccess","_isOutOfMapMaxBounds","_setErrorState","_updateMarker","_finish","_lastKnownPosition","_watchState","_geolocateButton","_updateCamera","_dotElement","longitude","accuracy","newBounds","geolocateSource","_accuracyCircleMarker","_userLocationDotMarker","_accuracy","_updateCircleRadius","_onZoom","_onError","_geolocationWatchID","_clearWatch","_timeoutId","_setupUI","_circleElement","_setup","forceRecalculation","permissions","geolocation","checkGeolocationSupport","clearWatch","southEastPoint","northEastPoint","mapHeightInMeters","circleDiameter","watchPosition","getCurrentPosition","ScaleControl","setUnit","FullscreenControl","_onFullscreenChange","fullscreenElement","mozFullScreenElement","webkitFullscreenElement","msFullscreenElement","_fullscreen","_handleFullscreenChange","_onClickFullscreen","_isFullscreen","_exitFullscreen","_requestFullscreen","_fullscreenchange","_fullscreenButton","_updateTitle","_getTitle","_prevCooperativeGestures","exitFullscreen","mozCancelFullScreen","msExitFullscreen","webkitCancelFullScreen","_togglePseudoFullScreen","requestFullscreen","mozRequestFullScreen","msRequestFullscreen","webkitRequestFullscreen","TerrainControl","_toggleTerrain","_updateTerrainIcon","_terrainButton","Popup","_content","_onClose","_onMouseMove","_onMouseUp","_onDrag","cursor","_trackPointer","_tip","offsetHeight","anchorComponents","offsetedPos","closeOnMove","_focusFirstElement","trackPointer","setText","setDOMContent","createTextNode","setHTML","html","frag","createDocumentFragment","temp","getMaxWidth","setMaxWidth","htmlNode","hasChildNodes","_createCloseButton","_closeButton","firstFocusable","querySelector","focus","setRTLTextPlugin","deferred","prewarm","clearPrewarmedResources"],"mappings":"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","x_google_ignoreList":[0,1,2,14,31,32,33,44,45,46,47,48,58,59,66,67,68,69,82,83,85,107,122,155,221,262]} \ No newline at end of file diff --git a/web/libraries/maplibre-gl/dist/maplibre-gl-dev.js b/web/libraries/maplibre-gl/dist/maplibre-gl-dev.js new file mode 100644 index 00000000..3d6880e7 --- /dev/null +++ b/web/libraries/maplibre-gl/dist/maplibre-gl-dev.js @@ -0,0 +1,58062 @@ +/* MapLibre GL JS is licensed under the 3-Clause BSD License. Full text of license: https://github.com/maplibre/maplibre-gl-js/blob/v3.6.2/LICENSE.txt */ +(function (global, factory) { +typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() : +typeof define === 'function' && define.amd ? define(factory) : +(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.maplibregl = factory()); +})(this, (function () { 'use strict'; + +/* eslint-disable */ + +var shared, worker, maplibregl; +// define gets called three times: one for each chunk. we rely on the order +// they're imported to know which is which +function define(_, chunk) { + if (!shared) { + shared = chunk; + } else if (!worker) { + worker = chunk; + } else { + var workerBundleString = 'var sharedChunk = {}; (' + shared + ')(sharedChunk); (' + worker + ')(sharedChunk);' + + var sharedChunk = {}; + shared(sharedChunk); + maplibregl = chunk(sharedChunk); + if (typeof window !== 'undefined') { + maplibregl.workerUrl = window.URL.createObjectURL(new Blob([workerBundleString], { type: 'text/javascript' })); + } + } +} + + +define(['exports'], (function (exports) { 'use strict'; + +/****************************************************************************** +Copyright (c) Microsoft Corporation. + +Permission to use, copy, modify, and/or distribute this software for any +purpose with or without fee is hereby granted. + +THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH +REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY +AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, +INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM +LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR +OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR +PERFORMANCE OF THIS SOFTWARE. +***************************************************************************** */ +/* global Reflect, Promise, SuppressedError, Symbol */ + +var extendStatics = function(d, b) { + extendStatics = Object.setPrototypeOf || + ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || + function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; }; + return extendStatics(d, b); +}; + +function __extends(d, b) { + if (typeof b !== "function" && b !== null) + throw new TypeError("Class extends value " + String(b) + " is not a constructor or null"); + extendStatics(d, b); + function __() { this.constructor = d; } + d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); +} + +var __assign = function() { + __assign = Object.assign || function __assign(t) { + for (var s, i = 1, n = arguments.length; i < n; i++) { + s = arguments[i]; + for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p]; + } + return t; + }; + return __assign.apply(this, arguments); +}; + +function __rest(s, e) { + var t = {}; + for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0) + t[p] = s[p]; + if (s != null && typeof Object.getOwnPropertySymbols === "function") + for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) { + if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i])) + t[p[i]] = s[p[i]]; + } + return t; +} + +function __decorate(decorators, target, key, desc) { + var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; + if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); + else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; + return c > 3 && r && Object.defineProperty(target, key, r), r; +} + +function __param(paramIndex, decorator) { + return function (target, key) { decorator(target, key, paramIndex); } +} + +function __esDecorate(ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) { + function accept(f) { if (f !== void 0 && typeof f !== "function") throw new TypeError("Function expected"); return f; } + var kind = contextIn.kind, key = kind === "getter" ? "get" : kind === "setter" ? "set" : "value"; + var target = !descriptorIn && ctor ? contextIn["static"] ? ctor : ctor.prototype : null; + var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {}); + var _, done = false; + for (var i = decorators.length - 1; i >= 0; i--) { + var context = {}; + for (var p in contextIn) context[p] = p === "access" ? {} : contextIn[p]; + for (var p in contextIn.access) context.access[p] = contextIn.access[p]; + context.addInitializer = function (f) { if (done) throw new TypeError("Cannot add initializers after decoration has completed"); extraInitializers.push(accept(f || null)); }; + var result = (0, decorators[i])(kind === "accessor" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context); + if (kind === "accessor") { + if (result === void 0) continue; + if (result === null || typeof result !== "object") throw new TypeError("Object expected"); + if (_ = accept(result.get)) descriptor.get = _; + if (_ = accept(result.set)) descriptor.set = _; + if (_ = accept(result.init)) initializers.unshift(_); + } + else if (_ = accept(result)) { + if (kind === "field") initializers.unshift(_); + else descriptor[key] = _; + } + } + if (target) Object.defineProperty(target, contextIn.name, descriptor); + done = true; +}; + +function __runInitializers(thisArg, initializers, value) { + var useValue = arguments.length > 2; + for (var i = 0; i < initializers.length; i++) { + value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg); + } + return useValue ? value : void 0; +}; + +function __propKey(x) { + return typeof x === "symbol" ? x : "".concat(x); +}; + +function __setFunctionName(f, name, prefix) { + if (typeof name === "symbol") name = name.description ? "[".concat(name.description, "]") : ""; + return Object.defineProperty(f, "name", { configurable: true, value: prefix ? "".concat(prefix, " ", name) : name }); +}; + +function __metadata(metadataKey, metadataValue) { + if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(metadataKey, metadataValue); +} + +function __awaiter(thisArg, _arguments, P, generator) { + function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } + return new (P || (P = Promise))(function (resolve, reject) { + function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } + function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } + function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } + step((generator = generator.apply(thisArg, _arguments || [])).next()); + }); +} + +function __generator(thisArg, body) { + var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g; + return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g; + function verb(n) { return function (v) { return step([n, v]); }; } + function step(op) { + if (f) throw new TypeError("Generator is already executing."); + while (g && (g = 0, op[0] && (_ = 0)), _) try { + if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t; + if (y = 0, t) op = [op[0] & 2, t.value]; + switch (op[0]) { + case 0: case 1: t = op; break; + case 4: _.label++; return { value: op[1], done: false }; + case 5: _.label++; y = op[1]; op = [0]; continue; + case 7: op = _.ops.pop(); _.trys.pop(); continue; + default: + if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; } + if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; } + if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; } + if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; } + if (t[2]) _.ops.pop(); + _.trys.pop(); continue; + } + op = body.call(thisArg, _); + } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; } + if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true }; + } +} + +var __createBinding = Object.create ? (function(o, m, k, k2) { + if (k2 === undefined) k2 = k; + var desc = Object.getOwnPropertyDescriptor(m, k); + if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { + desc = { enumerable: true, get: function() { return m[k]; } }; + } + Object.defineProperty(o, k2, desc); +}) : (function(o, m, k, k2) { + if (k2 === undefined) k2 = k; + o[k2] = m[k]; +}); + +function __exportStar(m, o) { + for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p); +} + +function __values(o) { + var s = typeof Symbol === "function" && Symbol.iterator, m = s && o[s], i = 0; + if (m) return m.call(o); + if (o && typeof o.length === "number") return { + next: function () { + if (o && i >= o.length) o = void 0; + return { value: o && o[i++], done: !o }; + } + }; + throw new TypeError(s ? "Object is not iterable." : "Symbol.iterator is not defined."); +} + +function __read(o, n) { + var m = typeof Symbol === "function" && o[Symbol.iterator]; + if (!m) return o; + var i = m.call(o), r, ar = [], e; + try { + while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value); + } + catch (error) { e = { error: error }; } + finally { + try { + if (r && !r.done && (m = i["return"])) m.call(i); + } + finally { if (e) throw e.error; } + } + return ar; +} + +/** @deprecated */ +function __spread() { + for (var ar = [], i = 0; i < arguments.length; i++) + ar = ar.concat(__read(arguments[i])); + return ar; +} + +/** @deprecated */ +function __spreadArrays() { + for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length; + for (var r = Array(s), k = 0, i = 0; i < il; i++) + for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++) + r[k] = a[j]; + return r; +} + +function __spreadArray(to, from, pack) { + if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) { + if (ar || !(i in from)) { + if (!ar) ar = Array.prototype.slice.call(from, 0, i); + ar[i] = from[i]; + } + } + return to.concat(ar || Array.prototype.slice.call(from)); +} + +function __await(v) { + return this instanceof __await ? (this.v = v, this) : new __await(v); +} + +function __asyncGenerator(thisArg, _arguments, generator) { + if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined."); + var g = generator.apply(thisArg, _arguments || []), i, q = []; + return i = {}, verb("next"), verb("throw"), verb("return"), i[Symbol.asyncIterator] = function () { return this; }, i; + function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; } + function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } } + function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); } + function fulfill(value) { resume("next", value); } + function reject(value) { resume("throw", value); } + function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); } +} + +function __asyncDelegator(o) { + var i, p; + return i = {}, verb("next"), verb("throw", function (e) { throw e; }), verb("return"), i[Symbol.iterator] = function () { return this; }, i; + function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: false } : f ? f(v) : v; } : f; } +} + +function __asyncValues(o) { + if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined."); + var m = o[Symbol.asyncIterator], i; + return m ? m.call(o) : (o = typeof __values === "function" ? __values(o) : o[Symbol.iterator](), i = {}, verb("next"), verb("throw"), verb("return"), i[Symbol.asyncIterator] = function () { return this; }, i); + function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; } + function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); } +} + +function __makeTemplateObject(cooked, raw) { + if (Object.defineProperty) { Object.defineProperty(cooked, "raw", { value: raw }); } else { cooked.raw = raw; } + return cooked; +}; + +var __setModuleDefault = Object.create ? (function(o, v) { + Object.defineProperty(o, "default", { enumerable: true, value: v }); +}) : function(o, v) { + o["default"] = v; +}; + +function __importStar(mod) { + if (mod && mod.__esModule) return mod; + var result = {}; + if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); + __setModuleDefault(result, mod); + return result; +} + +function __importDefault(mod) { + return (mod && mod.__esModule) ? mod : { default: mod }; +} + +function __classPrivateFieldGet(receiver, state, kind, f) { + if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter"); + if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it"); + return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver); +} + +function __classPrivateFieldSet(receiver, state, value, kind, f) { + if (kind === "m") throw new TypeError("Private method is not writable"); + if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter"); + if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it"); + return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value; +} + +function __classPrivateFieldIn(state, receiver) { + if (receiver === null || (typeof receiver !== "object" && typeof receiver !== "function")) throw new TypeError("Cannot use 'in' operator on non-object"); + return typeof state === "function" ? receiver === state : state.has(receiver); +} + +function __addDisposableResource(env, value, async) { + if (value !== null && value !== void 0) { + if (typeof value !== "object" && typeof value !== "function") throw new TypeError("Object expected."); + var dispose; + if (async) { + if (!Symbol.asyncDispose) throw new TypeError("Symbol.asyncDispose is not defined."); + dispose = value[Symbol.asyncDispose]; + } + if (dispose === void 0) { + if (!Symbol.dispose) throw new TypeError("Symbol.dispose is not defined."); + dispose = value[Symbol.dispose]; + } + if (typeof dispose !== "function") throw new TypeError("Object not disposable."); + env.stack.push({ value: value, dispose: dispose, async: async }); + } + else if (async) { + env.stack.push({ async: true }); + } + return value; +} + +var _SuppressedError = typeof SuppressedError === "function" ? SuppressedError : function (error, suppressed, message) { + var e = new Error(message); + return e.name = "SuppressedError", e.error = error, e.suppressed = suppressed, e; +}; + +function __disposeResources(env) { + function fail(e) { + env.error = env.hasError ? new _SuppressedError(e, env.error, "An error was suppressed during disposal.") : e; + env.hasError = true; + } + function next() { + while (env.stack.length) { + var rec = env.stack.pop(); + try { + var result = rec.dispose && rec.dispose.call(rec.value); + if (rec.async) return Promise.resolve(result).then(next, function(e) { fail(e); return next(); }); + } + catch (e) { + fail(e); + } + } + if (env.hasError) throw env.error; + } + return next(); +} + +var tslib_es6 = { + __extends: __extends, + __assign: __assign, + __rest: __rest, + __decorate: __decorate, + __param: __param, + __metadata: __metadata, + __awaiter: __awaiter, + __generator: __generator, + __createBinding: __createBinding, + __exportStar: __exportStar, + __values: __values, + __read: __read, + __spread: __spread, + __spreadArrays: __spreadArrays, + __spreadArray: __spreadArray, + __await: __await, + __asyncGenerator: __asyncGenerator, + __asyncDelegator: __asyncDelegator, + __asyncValues: __asyncValues, + __makeTemplateObject: __makeTemplateObject, + __importStar: __importStar, + __importDefault: __importDefault, + __classPrivateFieldGet: __classPrivateFieldGet, + __classPrivateFieldSet: __classPrivateFieldSet, + __classPrivateFieldIn: __classPrivateFieldIn, + __addDisposableResource: __addDisposableResource, + __disposeResources: __disposeResources, +}; + +var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {}; + +function getDefaultExportFromCjs (x) { + return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x; +} + +function getDefaultExportFromNamespaceIfPresent (n) { + return n && Object.prototype.hasOwnProperty.call(n, 'default') ? n['default'] : n; +} + +function getDefaultExportFromNamespaceIfNotNamed (n) { + return n && Object.prototype.hasOwnProperty.call(n, 'default') && Object.keys(n).length === 1 ? n['default'] : n; +} + +function getAugmentedNamespace(n) { + if (n.__esModule) return n; + var f = n.default; + if (typeof f == "function") { + var a = function a () { + if (this instanceof a) { + return Reflect.construct(f, arguments, this.constructor); + } + return f.apply(this, arguments); + }; + a.prototype = f.prototype; + } else a = {}; + Object.defineProperty(a, '__esModule', {value: true}); + Object.keys(n).forEach(function (k) { + var d = Object.getOwnPropertyDescriptor(n, k); + Object.defineProperty(a, k, d.get ? d : { + enumerable: true, + get: function () { + return n[k]; + } + }); + }); + return a; +} + +'use strict'; + +var pointGeometry = Point$1; + +/** + * A standalone point geometry with useful accessor, comparison, and + * modification methods. + * + * @class Point + * @param {Number} x the x-coordinate. this could be longitude or screen + * pixels, or any other sort of unit. + * @param {Number} y the y-coordinate. this could be latitude or screen + * pixels, or any other sort of unit. + * @example + * var point = new Point(-77, 38); + */ +function Point$1(x, y) { + this.x = x; + this.y = y; +} + +Point$1.prototype = { + + /** + * Clone this point, returning a new point that can be modified + * without affecting the old one. + * @return {Point} the clone + */ + clone: function() { return new Point$1(this.x, this.y); }, + + /** + * Add this point's x & y coordinates to another point, + * yielding a new point. + * @param {Point} p the other point + * @return {Point} output point + */ + add: function(p) { return this.clone()._add(p); }, + + /** + * Subtract this point's x & y coordinates to from point, + * yielding a new point. + * @param {Point} p the other point + * @return {Point} output point + */ + sub: function(p) { return this.clone()._sub(p); }, + + /** + * Multiply this point's x & y coordinates by point, + * yielding a new point. + * @param {Point} p the other point + * @return {Point} output point + */ + multByPoint: function(p) { return this.clone()._multByPoint(p); }, + + /** + * Divide this point's x & y coordinates by point, + * yielding a new point. + * @param {Point} p the other point + * @return {Point} output point + */ + divByPoint: function(p) { return this.clone()._divByPoint(p); }, + + /** + * Multiply this point's x & y coordinates by a factor, + * yielding a new point. + * @param {Point} k factor + * @return {Point} output point + */ + mult: function(k) { return this.clone()._mult(k); }, + + /** + * Divide this point's x & y coordinates by a factor, + * yielding a new point. + * @param {Point} k factor + * @return {Point} output point + */ + div: function(k) { return this.clone()._div(k); }, + + /** + * Rotate this point around the 0, 0 origin by an angle a, + * given in radians + * @param {Number} a angle to rotate around, in radians + * @return {Point} output point + */ + rotate: function(a) { return this.clone()._rotate(a); }, + + /** + * Rotate this point around p point by an angle a, + * given in radians + * @param {Number} a angle to rotate around, in radians + * @param {Point} p Point to rotate around + * @return {Point} output point + */ + rotateAround: function(a,p) { return this.clone()._rotateAround(a,p); }, + + /** + * Multiply this point by a 4x1 transformation matrix + * @param {Array} m transformation matrix + * @return {Point} output point + */ + matMult: function(m) { return this.clone()._matMult(m); }, + + /** + * Calculate this point but as a unit vector from 0, 0, meaning + * that the distance from the resulting point to the 0, 0 + * coordinate will be equal to 1 and the angle from the resulting + * point to the 0, 0 coordinate will be the same as before. + * @return {Point} unit vector point + */ + unit: function() { return this.clone()._unit(); }, + + /** + * Compute a perpendicular point, where the new y coordinate + * is the old x coordinate and the new x coordinate is the old y + * coordinate multiplied by -1 + * @return {Point} perpendicular point + */ + perp: function() { return this.clone()._perp(); }, + + /** + * Return a version of this point with the x & y coordinates + * rounded to integers. + * @return {Point} rounded point + */ + round: function() { return this.clone()._round(); }, + + /** + * Return the magitude of this point: this is the Euclidean + * distance from the 0, 0 coordinate to this point's x and y + * coordinates. + * @return {Number} magnitude + */ + mag: function() { + return Math.sqrt(this.x * this.x + this.y * this.y); + }, + + /** + * Judge whether this point is equal to another point, returning + * true or false. + * @param {Point} other the other point + * @return {boolean} whether the points are equal + */ + equals: function(other) { + return this.x === other.x && + this.y === other.y; + }, + + /** + * Calculate the distance from this point to another point + * @param {Point} p the other point + * @return {Number} distance + */ + dist: function(p) { + return Math.sqrt(this.distSqr(p)); + }, + + /** + * Calculate the distance from this point to another point, + * without the square root step. Useful if you're comparing + * relative distances. + * @param {Point} p the other point + * @return {Number} distance + */ + distSqr: function(p) { + var dx = p.x - this.x, + dy = p.y - this.y; + return dx * dx + dy * dy; + }, + + /** + * Get the angle from the 0, 0 coordinate to this point, in radians + * coordinates. + * @return {Number} angle + */ + angle: function() { + return Math.atan2(this.y, this.x); + }, + + /** + * Get the angle from this point to another point, in radians + * @param {Point} b the other point + * @return {Number} angle + */ + angleTo: function(b) { + return Math.atan2(this.y - b.y, this.x - b.x); + }, + + /** + * Get the angle between this point and another point, in radians + * @param {Point} b the other point + * @return {Number} angle + */ + angleWith: function(b) { + return this.angleWithSep(b.x, b.y); + }, + + /* + * Find the angle of the two vectors, solving the formula for + * the cross product a x b = |a||b|sin(θ) for θ. + * @param {Number} x the x-coordinate + * @param {Number} y the y-coordinate + * @return {Number} the angle in radians + */ + angleWithSep: function(x, y) { + return Math.atan2( + this.x * y - this.y * x, + this.x * x + this.y * y); + }, + + _matMult: function(m) { + var x = m[0] * this.x + m[1] * this.y, + y = m[2] * this.x + m[3] * this.y; + this.x = x; + this.y = y; + return this; + }, + + _add: function(p) { + this.x += p.x; + this.y += p.y; + return this; + }, + + _sub: function(p) { + this.x -= p.x; + this.y -= p.y; + return this; + }, + + _mult: function(k) { + this.x *= k; + this.y *= k; + return this; + }, + + _div: function(k) { + this.x /= k; + this.y /= k; + return this; + }, + + _multByPoint: function(p) { + this.x *= p.x; + this.y *= p.y; + return this; + }, + + _divByPoint: function(p) { + this.x /= p.x; + this.y /= p.y; + return this; + }, + + _unit: function() { + this._div(this.mag()); + return this; + }, + + _perp: function() { + var y = this.y; + this.y = this.x; + this.x = -y; + return this; + }, + + _rotate: function(angle) { + var cos = Math.cos(angle), + sin = Math.sin(angle), + x = cos * this.x - sin * this.y, + y = sin * this.x + cos * this.y; + this.x = x; + this.y = y; + return this; + }, + + _rotateAround: function(angle, p) { + var cos = Math.cos(angle), + sin = Math.sin(angle), + x = p.x + cos * (this.x - p.x) - sin * (this.y - p.y), + y = p.y + sin * (this.x - p.x) + cos * (this.y - p.y); + this.x = x; + this.y = y; + return this; + }, + + _round: function() { + this.x = Math.round(this.x); + this.y = Math.round(this.y); + return this; + } +}; + +/** + * Construct a point from an array if necessary, otherwise if the input + * is already a Point, or an unknown type, return it unchanged + * @param {Array|Point|*} a any kind of input value + * @return {Point} constructed point, or passed-through value. + * @example + * // this + * var point = Point.convert([0, 1]); + * // is equivalent to + * var point = new Point(0, 1); + */ +Point$1.convert = function (a) { + if (a instanceof Point$1) { + return a; + } + if (Array.isArray(a)) { + return new Point$1(a[0], a[1]); + } + return a; +}; + +var Point$2 = /*@__PURE__*/getDefaultExportFromCjs(pointGeometry); + +'use strict'; + +var unitbezier = UnitBezier; + +function UnitBezier(p1x, p1y, p2x, p2y) { + // Calculate the polynomial coefficients, implicit first and last control points are (0,0) and (1,1). + this.cx = 3.0 * p1x; + this.bx = 3.0 * (p2x - p1x) - this.cx; + this.ax = 1.0 - this.cx - this.bx; + + this.cy = 3.0 * p1y; + this.by = 3.0 * (p2y - p1y) - this.cy; + this.ay = 1.0 - this.cy - this.by; + + this.p1x = p1x; + this.p1y = p1y; + this.p2x = p2x; + this.p2y = p2y; +} + +UnitBezier.prototype = { + sampleCurveX: function (t) { + // `ax t^3 + bx t^2 + cx t' expanded using Horner's rule. + return ((this.ax * t + this.bx) * t + this.cx) * t; + }, + + sampleCurveY: function (t) { + return ((this.ay * t + this.by) * t + this.cy) * t; + }, + + sampleCurveDerivativeX: function (t) { + return (3.0 * this.ax * t + 2.0 * this.bx) * t + this.cx; + }, + + solveCurveX: function (x, epsilon) { + if (epsilon === undefined) epsilon = 1e-6; + + if (x < 0.0) return 0.0; + if (x > 1.0) return 1.0; + + var t = x; + + // First try a few iterations of Newton's method - normally very fast. + for (var i = 0; i < 8; i++) { + var x2 = this.sampleCurveX(t) - x; + if (Math.abs(x2) < epsilon) return t; + + var d2 = this.sampleCurveDerivativeX(t); + if (Math.abs(d2) < 1e-6) break; + + t = t - x2 / d2; + } + + // Fall back to the bisection method for reliability. + var t0 = 0.0; + var t1 = 1.0; + t = x; + + for (i = 0; i < 20; i++) { + x2 = this.sampleCurveX(t); + if (Math.abs(x2 - x) < epsilon) break; + + if (x > x2) { + t0 = t; + } else { + t1 = t; + } + + t = (t1 - t0) * 0.5 + t0; + } + + return t; + }, + + solve: function (x, epsilon) { + return this.sampleCurveY(this.solveCurveX(x, epsilon)); + } +}; + +var UnitBezier$1 = /*@__PURE__*/getDefaultExportFromCjs(unitbezier); + +let supportsOffscreenCanvas; +function offscreenCanvasSupported() { + if (supportsOffscreenCanvas == null) { + supportsOffscreenCanvas = typeof OffscreenCanvas !== 'undefined' && + new OffscreenCanvas(1, 1).getContext('2d') && + typeof createImageBitmap === 'function'; + } + return supportsOffscreenCanvas; +} + +let offscreenCanvasDistorted; +/** + * Some browsers don't return the exact pixels from a canvas to prevent user fingerprinting (see #3185). + * This function writes pixels to an OffscreenCanvas and reads them back using getImageData, returning false + * if they don't match. + * + * @returns true if the browser supports OffscreenCanvas but it distorts getImageData results, false otherwise. + */ +function isOffscreenCanvasDistorted() { + if (offscreenCanvasDistorted == null) { + offscreenCanvasDistorted = false; + if (offscreenCanvasSupported()) { + const size = 5; + const canvas = new OffscreenCanvas(size, size); + const context = canvas.getContext('2d', { willReadFrequently: true }); + if (context) { + // fill each pixel with an RGB value that should make the byte at index i equal to i (except alpha channel): + // [0, 1, 2, 255, 4, 5, 6, 255, 8, 9, 10, 255, ...] + for (let i = 0; i < size * size; i++) { + const base = i * 4; + context.fillStyle = `rgb(${base},${base + 1},${base + 2})`; + context.fillRect(i % size, Math.floor(i / size), 1, 1); + } + const data = context.getImageData(0, 0, size, size).data; + for (let i = 0; i < size * size * 4; i++) { + if (i % 4 !== 3 && data[i] !== i) { + offscreenCanvasDistorted = true; + break; + } + } + } + } + } + return offscreenCanvasDistorted || false; +} + +/** + * Given a value `t` that varies between 0 and 1, return + * an interpolation function that eases between 0 and 1 in a pleasing + * cubic in-out fashion. + */ +function easeCubicInOut(t) { + if (t <= 0) + return 0; + if (t >= 1) + return 1; + const t2 = t * t, t3 = t2 * t; + return 4 * (t < 0.5 ? t3 : 3 * (t - t2) + t3 - 0.75); +} +/** + * Given given (x, y), (x1, y1) control points for a bezier curve, + * return a function that interpolates along that curve. + * + * @param p1x - control point 1 x coordinate + * @param p1y - control point 1 y coordinate + * @param p2x - control point 2 x coordinate + * @param p2y - control point 2 y coordinate + */ +function bezier$1(p1x, p1y, p2x, p2y) { + const bezier = new UnitBezier$1(p1x, p1y, p2x, p2y); + return function (t) { + return bezier.solve(t); + }; +} +/** + * A default bezier-curve powered easing function with + * control points (0.25, 0.1) and (0.25, 1) + */ +const defaultEasing = bezier$1(0.25, 0.1, 0.25, 1); +/** + * constrain n to the given range via min + max + * + * @param n - value + * @param min - the minimum value to be returned + * @param max - the maximum value to be returned + * @returns the clamped value + */ +function clamp$1(n, min, max) { + return Math.min(max, Math.max(min, n)); +} +/** + * constrain n to the given range, excluding the minimum, via modular arithmetic + * + * @param n - value + * @param min - the minimum value to be returned, exclusive + * @param max - the maximum value to be returned, inclusive + * @returns constrained number + */ +function wrap(n, min, max) { + const d = max - min; + const w = ((n - min) % d + d) % d + min; + return (w === min) ? max : w; +} +/** + * Call an asynchronous function on an array of arguments, + * calling `callback` with the completed results of all calls. + * + * @param array - input to each call of the async function. + * @param fn - an async function with signature (data, callback) + * @param callback - a callback run after all async work is done. + * called with an array, containing the results of each async call. + */ +function asyncAll(array, fn, callback) { + if (!array.length) { + return callback(null, []); + } + let remaining = array.length; + const results = new Array(array.length); + let error = null; + array.forEach((item, i) => { + fn(item, (err, result) => { + if (err) + error = err; + results[i] = result; // https://github.com/facebook/flow/issues/2123 + if (--remaining === 0) + callback(error, results); + }); + }); +} +/** + * Compute the difference between the keys in one object and the keys + * in another object. + * + * @returns keys difference + */ +function keysDifference(obj, other) { + const difference = []; + for (const i in obj) { + if (!(i in other)) { + difference.push(i); + } + } + return difference; +} +/** + * Given a destination object and optionally many source objects, + * copy all properties from the source objects into the destination. + * The last source object given overrides properties from previous + * source objects. + * + * @param dest - destination object + * @param sources - sources from which properties are pulled + */ +function extend(dest, ...sources) { + for (const src of sources) { + for (const k in src) { + dest[k] = src[k]; + } + } + return dest; +} +/** + * Given an object and a number of properties as strings, return version + * of that object with only those properties. + * + * @param src - the object + * @param properties - an array of property names chosen + * to appear on the resulting object. + * @returns object with limited properties. + * @example + * ```ts + * let foo = { name: 'Charlie', age: 10 }; + * let justName = pick(foo, ['name']); // justName = { name: 'Charlie' } + * ``` + */ +function pick(src, properties) { + const result = {}; + for (let i = 0; i < properties.length; i++) { + const k = properties[i]; + if (k in src) { + result[k] = src[k]; + } + } + return result; +} +let id = 1; +/** + * Return a unique numeric id, starting at 1 and incrementing with + * each call. + * + * @returns unique numeric id. + */ +function uniqueId() { + return id++; +} +/** + * Return whether a given value is a power of two + */ +function isPowerOfTwo(value) { + return (Math.log(value) / Math.LN2) % 1 === 0; +} +/** + * Return the next power of two, or the input value if already a power of two + */ +function nextPowerOfTwo(value) { + if (value <= 1) + return 1; + return Math.pow(2, Math.ceil(Math.log(value) / Math.LN2)); +} +/** + * Create an object by mapping all the values of an existing object while + * preserving their keys. + */ +function mapObject(input, iterator, context) { + const output = {}; + for (const key in input) { + output[key] = iterator.call(context || this, input[key], key, input); + } + return output; +} +/** + * Create an object by filtering out values of an existing object. + */ +function filterObject(input, iterator, context) { + const output = {}; + for (const key in input) { + if (iterator.call(context || this, input[key], key, input)) { + output[key] = input[key]; + } + } + return output; +} +/** + * Deeply compares two object literals. + * @param a - first object literal to be compared + * @param b - second object literal to be compared + * @returns true if the two object literals are deeply equal, false otherwise + */ +function deepEqual$1(a, b) { + if (Array.isArray(a)) { + if (!Array.isArray(b) || a.length !== b.length) + return false; + for (let i = 0; i < a.length; i++) { + if (!deepEqual$1(a[i], b[i])) + return false; + } + return true; + } + if (typeof a === 'object' && a !== null && b !== null) { + if (!(typeof b === 'object')) + return false; + const keys = Object.keys(a); + if (keys.length !== Object.keys(b).length) + return false; + for (const key in a) { + if (!deepEqual$1(a[key], b[key])) + return false; + } + return true; + } + return a === b; +} +/** + * Deeply clones two objects. + */ +function clone$9(input) { + if (Array.isArray(input)) { + return input.map(clone$9); + } + else if (typeof input === 'object' && input) { + return mapObject(input, clone$9); + } + else { + return input; + } +} +/** + * Check if two arrays have at least one common element. + */ +function arraysIntersect(a, b) { + for (let l = 0; l < a.length; l++) { + if (b.indexOf(a[l]) >= 0) + return true; + } + return false; +} +/** + * Print a warning message to the console and ensure duplicate warning messages + * are not printed. + */ +const warnOnceHistory = {}; +function warnOnce(message) { + if (!warnOnceHistory[message]) { + // console isn't defined in some WebWorkers, see #2558 + if (typeof console !== 'undefined') + console.warn(message); + warnOnceHistory[message] = true; + } +} +/** + * Indicates if the provided Points are in a counter clockwise (true) or clockwise (false) order + * + * @returns true for a counter clockwise set of points + */ +// http://bryceboe.com/2006/10/23/line-segment-intersection-algorithm/ +function isCounterClockwise(a, b, c) { + return (c.y - a.y) * (b.x - a.x) > (b.y - a.y) * (c.x - a.x); +} +/** + * For two lines a and b in 2d space, defined by any two points along the lines, + * find the intersection point, or return null if the lines are parallel + * + * @param a1 - First point on line a + * @param a2 - Second point on line a + * @param b1 - First point on line b + * @param b2 - Second point on line b + * + * @returns the intersection point of the two lines or null if they are parallel + */ +function findLineIntersection(a1, a2, b1, b2) { + const aDeltaY = a2.y - a1.y; + const aDeltaX = a2.x - a1.x; + const bDeltaY = b2.y - b1.y; + const bDeltaX = b2.x - b1.x; + const denominator = (bDeltaY * aDeltaX) - (bDeltaX * aDeltaY); + if (denominator === 0) { + // Lines are parallel + return null; + } + const originDeltaY = a1.y - b1.y; + const originDeltaX = a1.x - b1.x; + const aInterpolation = (bDeltaX * originDeltaY - bDeltaY * originDeltaX) / denominator; + // Find intersection by projecting out from origin of first segment + return new Point$2(a1.x + (aInterpolation * aDeltaX), a1.y + (aInterpolation * aDeltaY)); +} +/** + * Returns the signed area for the polygon ring. Positive areas are exterior rings and + * have a clockwise winding. Negative areas are interior rings and have a counter clockwise + * ordering. + * + * @param ring - Exterior or interior ring + */ +function calculateSignedArea(ring) { + let sum = 0; + for (let i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) { + p1 = ring[i]; + p2 = ring[j]; + sum += (p2.x - p1.x) * (p1.y + p2.y); + } + return sum; +} +/** + * Detects closed polygons, first + last point are equal + * + * @param points - array of points + * @returns `true` if the points are a closed polygon + */ +function isClosedPolygon(points) { + // If it is 2 points that are the same then it is a point + // If it is 3 points with start and end the same then it is a line + if (points.length < 4) + return false; + const p1 = points[0]; + const p2 = points[points.length - 1]; + if (Math.abs(p1.x - p2.x) > 0 || + Math.abs(p1.y - p2.y) > 0) { + return false; + } + // polygon simplification can produce polygons with zero area and more than 3 points + return Math.abs(calculateSignedArea(points)) > 0.01; +} +/** + * Converts spherical coordinates to cartesian coordinates. + * + * @param spherical - Spherical coordinates, in [radial, azimuthal, polar] + * @returns cartesian coordinates in [x, y, z] + */ +function sphericalToCartesian([r, azimuthal, polar]) { + // We abstract "north"/"up" (compass-wise) to be 0° when really this is 90° (π/2): + // correct for that here + azimuthal += 90; + // Convert azimuthal and polar angles to radians + azimuthal *= Math.PI / 180; + polar *= Math.PI / 180; + return { + x: r * Math.cos(azimuthal) * Math.sin(polar), + y: r * Math.sin(azimuthal) * Math.sin(polar), + z: r * Math.cos(polar) + }; +} +/** + * Returns true if the when run in the web-worker context. + * + * @returns `true` if the when run in the web-worker context. + */ +function isWorker() { + // @ts-ignore + return typeof WorkerGlobalScope !== 'undefined' && typeof self !== 'undefined' && self instanceof WorkerGlobalScope; +} +/** + * Parses data from 'Cache-Control' headers. + * + * @param cacheControl - Value of 'Cache-Control' header + * @returns object containing parsed header info. + */ +function parseCacheControl(cacheControl) { + // Taken from [Wreck](https://github.com/hapijs/wreck) + const re = /(?:^|(?:\s*\,\s*))([^\x00-\x20\(\)<>@\,;\:\\"\/\[\]\?\=\{\}\x7F]+)(?:\=(?:([^\x00-\x20\(\)<>@\,;\:\\"\/\[\]\?\=\{\}\x7F]+)|(?:\"((?:[^"\\]|\\.)*)\")))?/g; + const header = {}; + cacheControl.replace(re, ($0, $1, $2, $3) => { + const value = $2 || $3; + header[$1] = value ? value.toLowerCase() : true; + return ''; + }); + if (header['max-age']) { + const maxAge = parseInt(header['max-age'], 10); + if (isNaN(maxAge)) + delete header['max-age']; + else + header['max-age'] = maxAge; + } + return header; +} +let _isSafari = null; +/** + * Returns true when run in WebKit derived browsers. + * This is used as a workaround for a memory leak in Safari caused by using Transferable objects to + * transfer data between WebWorkers and the main thread. + * https://github.com/mapbox/mapbox-gl-js/issues/8771 + * + * This should be removed once the underlying Safari issue is fixed. + * + * @param scope - Since this function is used both on the main thread and WebWorker context, + * let the calling scope pass in the global scope object. + * @returns `true` when run in WebKit derived browsers. + */ +function isSafari(scope) { + if (_isSafari == null) { + const userAgent = scope.navigator ? scope.navigator.userAgent : null; + _isSafari = !!scope.safari || + !!(userAgent && (/\b(iPad|iPhone|iPod)\b/.test(userAgent) || (!!userAgent.match('Safari') && !userAgent.match('Chrome')))); + } + return _isSafari; +} +function storageAvailable(type) { + try { + const storage = window[type]; + storage.setItem('_mapbox_test_', 1); + storage.removeItem('_mapbox_test_'); + return true; + } + catch (e) { + return false; + } +} +// The following methods are from https://developer.mozilla.org/en-US/docs/Web/API/WindowBase64/Base64_encoding_and_decoding#The_Unicode_Problem +//Unicode compliant base64 encoder for strings +function b64EncodeUnicode(str) { + return btoa(encodeURIComponent(str).replace(/%([0-9A-F]{2})/g, (match, p1) => { + return String.fromCharCode(Number('0x' + p1)); //eslint-disable-line + })); +} +// Unicode compliant decoder for base64-encoded strings +function b64DecodeUnicode(str) { + return decodeURIComponent(atob(str).split('').map((c) => { + return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2); //eslint-disable-line + }).join('')); +} +function isImageBitmap(image) { + return typeof ImageBitmap !== 'undefined' && image instanceof ImageBitmap; +} +/** + * Converts an ArrayBuffer to an ImageBitmap. + * + * Used mostly for testing purposes only, because mocking libs don't know how to work with ArrayBuffers, but work + * perfectly fine with ImageBitmaps. Might also be used for environments (other than testing) not supporting + * ArrayBuffers. + * + * @param data - Data to convert + * @param callback - A callback executed after the conversion is finished. Invoked with error (if any) as the first argument and resulting image bitmap (when no error) as the second + */ +function arrayBufferToImageBitmap(data, callback) { + const blob = new Blob([new Uint8Array(data)], { type: 'image/png' }); + createImageBitmap(blob).then((imgBitmap) => { + callback(null, imgBitmap); + }).catch((e) => { + callback(new Error(`Could not load image because of ${e.message}. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.`)); + }); +} +const transparentPngUrl = ''; +/** + * Converts an ArrayBuffer to an HTMLImageElement. + * + * Used mostly for testing purposes only, because mocking libs don't know how to work with ArrayBuffers, but work + * perfectly fine with ImageBitmaps. Might also be used for environments (other than testing) not supporting + * ArrayBuffers. + * + * @param data - Data to convert + * @param callback - A callback executed after the conversion is finished. Invoked with error (if any) as the first argument and resulting image element (when no error) as the second + */ +function arrayBufferToImage(data, callback) { + const img = new Image(); + img.onload = () => { + callback(null, img); + URL.revokeObjectURL(img.src); + // prevent image dataURI memory leak in Safari; + // but don't free the image immediately because it might be uploaded in the next frame + // https://github.com/mapbox/mapbox-gl-js/issues/10226 + img.onload = null; + window.requestAnimationFrame(() => { img.src = transparentPngUrl; }); + }; + img.onerror = () => callback(new Error('Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.')); + const blob = new Blob([new Uint8Array(data)], { type: 'image/png' }); + img.src = data.byteLength ? URL.createObjectURL(blob) : transparentPngUrl; +} +/** + * Computes the webcodecs VideoFrame API options to select a rectangle out of + * an image and write it into the destination rectangle. + * + * Rect (x/y/width/height) select the overlapping rectangle from the source image + * and layout (offset/stride) write that overlapping rectangle to the correct place + * in the destination image. + * + * Offset is the byte offset in the dest image that the first pixel appears at + * and stride is the number of bytes to the start of the next row: + * ┌───────────┐ + * │ dest │ + * │ ┌───┼───────┐ + * │offset→│▓▓▓│ source│ + * │ │▓▓▓│ │ + * │ └───┼───────┘ + * │stride ⇠╌╌╌│ + * │╌╌╌╌╌╌→ │ + * └───────────┘ + * + * @param image - source image containing a width and height attribute + * @param x - top-left x coordinate to read from the image + * @param y - top-left y coordinate to read from the image + * @param width - width of the rectangle to read from the image + * @param height - height of the rectangle to read from the image + * @returns the layout and rect options to pass into VideoFrame API + */ +function computeVideoFrameParameters(image, x, y, width, height) { + const destRowOffset = Math.max(-x, 0) * 4; + const firstSourceRow = Math.max(0, y); + const firstDestRow = firstSourceRow - y; + const offset = firstDestRow * width * 4 + destRowOffset; + const stride = width * 4; + const sourceLeft = Math.max(0, x); + const sourceTop = Math.max(0, y); + const sourceRight = Math.min(image.width, x + width); + const sourceBottom = Math.min(image.height, y + height); + return { + rect: { + x: sourceLeft, + y: sourceTop, + width: sourceRight - sourceLeft, + height: sourceBottom - sourceTop + }, + layout: [{ offset, stride }] + }; +} +/** + * Reads pixels from an ImageBitmap/Image/canvas using webcodec VideoFrame API. + * + * @param data - image, imagebitmap, or canvas to parse + * @param x - top-left x coordinate to read from the image + * @param y - top-left y coordinate to read from the image + * @param width - width of the rectangle to read from the image + * @param height - height of the rectangle to read from the image + * @returns a promise containing the parsed RGBA pixel values of the image, or the error if an error occurred + */ +function readImageUsingVideoFrame(image, x, y, width, height) { + return __awaiter(this, void 0, void 0, function* () { + if (typeof VideoFrame === 'undefined') { + throw new Error('VideoFrame not supported'); + } + const frame = new VideoFrame(image, { timestamp: 0 }); + try { + const format = frame === null || frame === void 0 ? void 0 : frame.format; + if (!format || !(format.startsWith('BGR') || format.startsWith('RGB'))) { + throw new Error(`Unrecognized format ${format}`); + } + const swapBR = format.startsWith('BGR'); + const result = new Uint8ClampedArray(width * height * 4); + yield frame.copyTo(result, computeVideoFrameParameters(image, x, y, width, height)); + if (swapBR) { + for (let i = 0; i < result.length; i += 4) { + const tmp = result[i]; + result[i] = result[i + 2]; + result[i + 2] = tmp; + } + } + return result; + } + finally { + frame.close(); + } + }); +} +let offscreenCanvas; +let offscreenCanvasContext; +/** + * Reads pixels from an ImageBitmap/Image/canvas using OffscreenCanvas + * + * @param data - image, imagebitmap, or canvas to parse + * @param x - top-left x coordinate to read from the image + * @param y - top-left y coordinate to read from the image + * @param width - width of the rectangle to read from the image + * @param height - height of the rectangle to read from the image + * @returns a promise containing the parsed RGBA pixel values of the image, or the error if an error occurred + */ +function readImageDataUsingOffscreenCanvas(imgBitmap, x, y, width, height) { + const origWidth = imgBitmap.width; + const origHeight = imgBitmap.height; + // Lazily initialize OffscreenCanvas + if (!offscreenCanvas || !offscreenCanvasContext) { + // Dem tiles are typically 256x256 + offscreenCanvas = new OffscreenCanvas(origWidth, origHeight); + offscreenCanvasContext = offscreenCanvas.getContext('2d', { willReadFrequently: true }); + } + offscreenCanvas.width = origWidth; + offscreenCanvas.height = origHeight; + offscreenCanvasContext.drawImage(imgBitmap, 0, 0, origWidth, origHeight); + const imgData = offscreenCanvasContext.getImageData(x, y, width, height); + offscreenCanvasContext.clearRect(0, 0, origWidth, origHeight); + return imgData.data; +} +/** + * Reads RGBA pixels from an preferring OffscreenCanvas, but falling back to VideoFrame if supported and + * the browser is mangling OffscreenCanvas getImageData results. + * + * @param data - image, imagebitmap, or canvas to parse + * @param x - top-left x coordinate to read from the image + * @param y - top-left y coordinate to read from the image + * @param width - width of the rectangle to read from the image + * @param height - height of the rectangle to read from the image + * @returns a promise containing the parsed RGBA pixel values of the image + */ +function getImageData(image, x, y, width, height) { + return __awaiter(this, void 0, void 0, function* () { + if (isOffscreenCanvasDistorted()) { + try { + return yield readImageUsingVideoFrame(image, x, y, width, height); + } + catch (e) { + // fall back to OffscreenCanvas + } + } + return readImageDataUsingOffscreenCanvas(image, x, y, width, height); + }); +} + +const now = typeof performance !== 'undefined' && performance && performance.now ? + performance.now.bind(performance) : + Date.now.bind(Date); +let linkEl; +let reducedMotionQuery; +/** */ +const browser = { + /** + * Provides a function that outputs milliseconds: either performance.now() + * or a fallback to Date.now() + */ + now, + frame(fn) { + const frame = requestAnimationFrame(fn); + return { cancel: () => cancelAnimationFrame(frame) }; + }, + getImageData(img, padding = 0) { + const context = this.getImageCanvasContext(img); + return context.getImageData(-padding, -padding, img.width + 2 * padding, img.height + 2 * padding); + }, + getImageCanvasContext(img) { + const canvas = window.document.createElement('canvas'); + const context = canvas.getContext('2d', { willReadFrequently: true }); + if (!context) { + throw new Error('failed to create canvas 2d context'); + } + canvas.width = img.width; + canvas.height = img.height; + context.drawImage(img, 0, 0, img.width, img.height); + return context; + }, + resolveURL(path) { + if (!linkEl) + linkEl = document.createElement('a'); + linkEl.href = path; + return linkEl.href; + }, + hardwareConcurrency: typeof navigator !== 'undefined' && navigator.hardwareConcurrency || 4, + get prefersReducedMotion() { + // In case your test crashes when checking matchMedia, call setMatchMedia from 'src/util/test/util' + if (!matchMedia) + return false; + //Lazily initialize media query + if (reducedMotionQuery == null) { + reducedMotionQuery = matchMedia('(prefers-reduced-motion: reduce)'); + } + return reducedMotionQuery.matches; + }, +}; + +const config = { + MAX_PARALLEL_IMAGE_REQUESTS: 16, + MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME: 8, + MAX_TILE_CACHE_ZOOM_LEVELS: 5, + REGISTERED_PROTOCOLS: {}, + WORKER_URL: '' +}; + +/** + * An error thrown when a HTTP request results in an error response. + */ +class AJAXError extends Error { + /** + * @param status - The response's HTTP status code. + * @param statusText - The response's HTTP status text. + * @param url - The request's URL. + * @param body - The response's body. + */ + constructor(status, statusText, url, body) { + super(`AJAXError: ${statusText} (${status}): ${url}`); + this.status = status; + this.statusText = statusText; + this.url = url; + this.body = body; + } +} +// Ensure that we're sending the correct referrer from blob URL worker bundles. +// For files loaded from the local file system, `location.origin` will be set +// to the string(!) "null" (Firefox), or "file://" (Chrome, Safari, Edge, IE), +// and we will set an empty referrer. Otherwise, we're using the document's URL. +/* global self */ +const getReferrer = isWorker() ? + () => self.worker && self.worker.referrer : + () => (window.location.protocol === 'blob:' ? window.parent : window).location.href; +const getProtocolAction = url => config.REGISTERED_PROTOCOLS[url.substring(0, url.indexOf('://'))]; +// Determines whether a URL is a file:// URL. This is obviously the case if it begins +// with file://. Relative URLs are also file:// URLs iff the original document was loaded +// via a file:// URL. +const isFileURL = url => /^file:/.test(url) || (/^file:/.test(getReferrer()) && !/^\w+:/.test(url)); +function makeFetchRequest(requestParameters, callback) { + const controller = new AbortController(); + const request = new Request(requestParameters.url, { + method: requestParameters.method || 'GET', + body: requestParameters.body, + credentials: requestParameters.credentials, + headers: requestParameters.headers, + cache: requestParameters.cache, + referrer: getReferrer(), + signal: controller.signal + }); + let complete = false; + let aborted = false; + if (requestParameters.type === 'json') { + request.headers.set('Accept', 'application/json'); + } + const validateOrFetch = (err, cachedResponse, responseIsFresh) => { + if (aborted) + return; + if (err) { + // Do fetch in case of cache error. + // HTTP pages in Edge trigger a security error that can be ignored. + if (err.message !== 'SecurityError') { + warnOnce(err); + } + } + if (cachedResponse && responseIsFresh) { + return finishRequest(cachedResponse); + } + if (cachedResponse) { + // We can't do revalidation with 'If-None-Match' because then the + // request doesn't have simple cors headers. + } + fetch(request).then(response => { + if (response.ok) { + return finishRequest(response); + } + else { + return response.blob().then(body => callback(new AJAXError(response.status, response.statusText, requestParameters.url, body))); + } + }).catch(error => { + if (error.code === 20) { + // silence expected AbortError + return; + } + callback(new Error(error.message)); + }); + }; + const finishRequest = (response) => { + ((requestParameters.type === 'arrayBuffer' || requestParameters.type === 'image') ? response.arrayBuffer() : + requestParameters.type === 'json' ? response.json() : + response.text()).then(result => { + if (aborted) + return; + complete = true; + callback(null, result, response.headers.get('Cache-Control'), response.headers.get('Expires')); + }).catch(err => { + if (!aborted) + callback(new Error(err.message)); + }); + }; + validateOrFetch(null, null); + return { cancel: () => { + aborted = true; + if (!complete) + controller.abort(); + } }; +} +function makeXMLHttpRequest(requestParameters, callback) { + const xhr = new XMLHttpRequest(); + xhr.open(requestParameters.method || 'GET', requestParameters.url, true); + if (requestParameters.type === 'arrayBuffer' || requestParameters.type === 'image') { + xhr.responseType = 'arraybuffer'; + } + for (const k in requestParameters.headers) { + xhr.setRequestHeader(k, requestParameters.headers[k]); + } + if (requestParameters.type === 'json') { + xhr.responseType = 'text'; + xhr.setRequestHeader('Accept', 'application/json'); + } + xhr.withCredentials = requestParameters.credentials === 'include'; + xhr.onerror = () => { + callback(new Error(xhr.statusText)); + }; + xhr.onload = () => { + if (((xhr.status >= 200 && xhr.status < 300) || xhr.status === 0) && xhr.response !== null) { + let data = xhr.response; + if (requestParameters.type === 'json') { + // We're manually parsing JSON here to get better error messages. + try { + data = JSON.parse(xhr.response); + } + catch (err) { + return callback(err); + } + } + callback(null, data, xhr.getResponseHeader('Cache-Control'), xhr.getResponseHeader('Expires')); + } + else { + const body = new Blob([xhr.response], { type: xhr.getResponseHeader('Content-Type') }); + callback(new AJAXError(xhr.status, xhr.statusText, requestParameters.url, body)); + } + }; + xhr.send(requestParameters.body); + return { cancel: () => xhr.abort() }; +} +const makeRequest = function (requestParameters, callback) { + // We're trying to use the Fetch API if possible. However, in some situations we can't use it: + // - IE11 doesn't support it at all. In this case, we dispatch the request to the main thread so + // that we can get an accruate referrer header. + // - Safari exposes window.AbortController, but it doesn't work actually abort any requests in + // some versions (see https://bugs.webkit.org/show_bug.cgi?id=174980#c2) + // - Requests for resources with the file:// URI scheme don't work with the Fetch API either. In + // this case we unconditionally use XHR on the current thread since referrers don't matter. + if (/:\/\//.test(requestParameters.url) && !(/^https?:|^file:/.test(requestParameters.url))) { + if (isWorker() && self.worker && self.worker.actor) { + return self.worker.actor.send('getResource', requestParameters, callback); + } + if (!isWorker()) { + const action = getProtocolAction(requestParameters.url) || makeFetchRequest; + return action(requestParameters, callback); + } + } + if (!isFileURL(requestParameters.url)) { + if (fetch && Request && AbortController && Object.prototype.hasOwnProperty.call(Request.prototype, 'signal')) { + return makeFetchRequest(requestParameters, callback); + } + if (isWorker() && self.worker && self.worker.actor) { + const queueOnMainThread = true; + return self.worker.actor.send('getResource', requestParameters, callback, undefined, queueOnMainThread); + } + } + return makeXMLHttpRequest(requestParameters, callback); +}; +const getJSON = function (requestParameters, callback) { + return makeRequest(extend(requestParameters, { type: 'json' }), callback); +}; +const getArrayBuffer = function (requestParameters, callback) { + return makeRequest(extend(requestParameters, { type: 'arrayBuffer' }), callback); +}; +const postData = function (requestParameters, callback) { + return makeRequest(extend(requestParameters, { method: 'POST' }), callback); +}; +function sameOrigin(inComingUrl) { + // URL class should be available everywhere + // https://developer.mozilla.org/en-US/docs/Web/API/URL + // In addtion, a relative URL "/foo" or "./foo" will throw exception in its ctor, + // try-catch is expansive so just use a heuristic check to avoid it + // also check data URL + if (!inComingUrl || + inComingUrl.indexOf('://') <= 0 || // relative URL + inComingUrl.indexOf('data:image/') === 0 || // data image URL + inComingUrl.indexOf('blob:') === 0) { // blob + return true; + } + const urlObj = new URL(inComingUrl); + const locationObj = window.location; + return urlObj.protocol === locationObj.protocol && urlObj.host === locationObj.host; +} +const getVideo = function (urls, callback) { + const video = window.document.createElement('video'); + video.muted = true; + video.onloadstart = function () { + callback(null, video); + }; + for (let i = 0; i < urls.length; i++) { + const s = window.document.createElement('source'); + if (!sameOrigin(urls[i])) { + video.crossOrigin = 'Anonymous'; + } + s.src = urls[i]; + video.appendChild(s); + } + return { cancel: () => { } }; +}; + +function _addEventListener(type, listener, listenerList) { + const listenerExists = listenerList[type] && listenerList[type].indexOf(listener) !== -1; + if (!listenerExists) { + listenerList[type] = listenerList[type] || []; + listenerList[type].push(listener); + } +} +function _removeEventListener(type, listener, listenerList) { + if (listenerList && listenerList[type]) { + const index = listenerList[type].indexOf(listener); + if (index !== -1) { + listenerList[type].splice(index, 1); + } + } +} +/** + * The event class + */ +class Event { + constructor(type, data = {}) { + extend(this, data); + this.type = type; + } +} +/** + * An error event + */ +class ErrorEvent extends Event { + constructor(error, data = {}) { + super('error', extend({ error }, data)); + } +} +/** + * Methods mixed in to other classes for event capabilities. + * + * @group Event Related + */ +class Evented { + /** + * Adds a listener to a specified event type. + * + * @param type - The event type to add a listen for. + * @param listener - The function to be called when the event is fired. + * The listener function is called with the data object passed to `fire`, + * extended with `target` and `type` properties. + * @returns `this` + */ + on(type, listener) { + this._listeners = this._listeners || {}; + _addEventListener(type, listener, this._listeners); + return this; + } + /** + * Removes a previously registered event listener. + * + * @param type - The event type to remove listeners for. + * @param listener - The listener function to remove. + * @returns `this` + */ + off(type, listener) { + _removeEventListener(type, listener, this._listeners); + _removeEventListener(type, listener, this._oneTimeListeners); + return this; + } + /** + * Adds a listener that will be called only once to a specified event type. + * + * The listener will be called first time the event fires after the listener is registered. + * + * @param type - The event type to listen for. + * @param listener - The function to be called when the event is fired the first time. + * @returns `this` or a promise if a listener is not provided + */ + once(type, listener) { + if (!listener) { + return new Promise((resolve) => this.once(type, resolve)); + } + this._oneTimeListeners = this._oneTimeListeners || {}; + _addEventListener(type, listener, this._oneTimeListeners); + return this; + } + fire(event, properties) { + // Compatibility with (type: string, properties: Object) signature from previous versions. + // See https://github.com/mapbox/mapbox-gl-js/issues/6522, + // https://github.com/mapbox/mapbox-gl-draw/issues/766 + if (typeof event === 'string') { + event = new Event(event, properties || {}); + } + const type = event.type; + if (this.listens(type)) { + event.target = this; + // make sure adding or removing listeners inside other listeners won't cause an infinite loop + const listeners = this._listeners && this._listeners[type] ? this._listeners[type].slice() : []; + for (const listener of listeners) { + listener.call(this, event); + } + const oneTimeListeners = this._oneTimeListeners && this._oneTimeListeners[type] ? this._oneTimeListeners[type].slice() : []; + for (const listener of oneTimeListeners) { + _removeEventListener(type, listener, this._oneTimeListeners); + listener.call(this, event); + } + const parent = this._eventedParent; + if (parent) { + extend(event, typeof this._eventedParentData === 'function' ? this._eventedParentData() : this._eventedParentData); + parent.fire(event); + } + // To ensure that no error events are dropped, print them to the + // console if they have no listeners. + } + else if (event instanceof ErrorEvent) { + console.error(event.error); + } + return this; + } + /** + * Returns a true if this instance of Evented or any forwardeed instances of Evented have a listener for the specified type. + * + * @param type - The event type + * @returns `true` if there is at least one registered listener for specified event type, `false` otherwise + */ + listens(type) { + return ((this._listeners && this._listeners[type] && this._listeners[type].length > 0) || + (this._oneTimeListeners && this._oneTimeListeners[type] && this._oneTimeListeners[type].length > 0) || + (this._eventedParent && this._eventedParent.listens(type))); + } + /** + * Bubble all events fired by this instance of Evented to this parent instance of Evented. + * @returns `this` + */ + setEventedParent(parent, data) { + this._eventedParent = parent; + this._eventedParentData = data; + return this; + } +} + +var $version = 8; +var $root = { + version: { + required: true, + type: "enum", + values: [ + 8 + ] + }, + name: { + type: "string" + }, + metadata: { + type: "*" + }, + center: { + type: "array", + value: "number" + }, + zoom: { + type: "number" + }, + bearing: { + type: "number", + "default": 0, + period: 360, + units: "degrees" + }, + pitch: { + type: "number", + "default": 0, + units: "degrees" + }, + light: { + type: "light" + }, + terrain: { + type: "terrain" + }, + sources: { + required: true, + type: "sources" + }, + sprite: { + type: "sprite" + }, + glyphs: { + type: "string" + }, + transition: { + type: "transition" + }, + layers: { + required: true, + type: "array", + value: "layer" + } +}; +var sources = { + "*": { + type: "source" + } +}; +var source = [ + "source_vector", + "source_raster", + "source_raster_dem", + "source_geojson", + "source_video", + "source_image" +]; +var source_vector = { + type: { + required: true, + type: "enum", + values: { + vector: { + } + } + }, + url: { + type: "string" + }, + tiles: { + type: "array", + value: "string" + }, + bounds: { + type: "array", + value: "number", + length: 4, + "default": [ + -180, + -85.051129, + 180, + 85.051129 + ] + }, + scheme: { + type: "enum", + values: { + xyz: { + }, + tms: { + } + }, + "default": "xyz" + }, + minzoom: { + type: "number", + "default": 0 + }, + maxzoom: { + type: "number", + "default": 22 + }, + attribution: { + type: "string" + }, + promoteId: { + type: "promoteId" + }, + volatile: { + type: "boolean", + "default": false + }, + "*": { + type: "*" + } +}; +var source_raster = { + type: { + required: true, + type: "enum", + values: { + raster: { + } + } + }, + url: { + type: "string" + }, + tiles: { + type: "array", + value: "string" + }, + bounds: { + type: "array", + value: "number", + length: 4, + "default": [ + -180, + -85.051129, + 180, + 85.051129 + ] + }, + minzoom: { + type: "number", + "default": 0 + }, + maxzoom: { + type: "number", + "default": 22 + }, + tileSize: { + type: "number", + "default": 512, + units: "pixels" + }, + scheme: { + type: "enum", + values: { + xyz: { + }, + tms: { + } + }, + "default": "xyz" + }, + attribution: { + type: "string" + }, + volatile: { + type: "boolean", + "default": false + }, + "*": { + type: "*" + } +}; +var source_raster_dem = { + type: { + required: true, + type: "enum", + values: { + "raster-dem": { + } + } + }, + url: { + type: "string" + }, + tiles: { + type: "array", + value: "string" + }, + bounds: { + type: "array", + value: "number", + length: 4, + "default": [ + -180, + -85.051129, + 180, + 85.051129 + ] + }, + minzoom: { + type: "number", + "default": 0 + }, + maxzoom: { + type: "number", + "default": 22 + }, + tileSize: { + type: "number", + "default": 512, + units: "pixels" + }, + attribution: { + type: "string" + }, + encoding: { + type: "enum", + values: { + terrarium: { + }, + mapbox: { + }, + custom: { + } + }, + "default": "mapbox" + }, + redFactor: { + type: "number", + "default": 1 + }, + blueFactor: { + type: "number", + "default": 1 + }, + greenFactor: { + type: "number", + "default": 1 + }, + baseShift: { + type: "number", + "default": 0 + }, + volatile: { + type: "boolean", + "default": false + }, + "*": { + type: "*" + } +}; +var source_geojson = { + type: { + required: true, + type: "enum", + values: { + geojson: { + } + } + }, + data: { + required: true, + type: "*" + }, + maxzoom: { + type: "number", + "default": 18 + }, + attribution: { + type: "string" + }, + buffer: { + type: "number", + "default": 128, + maximum: 512, + minimum: 0 + }, + filter: { + type: "*" + }, + tolerance: { + type: "number", + "default": 0.375 + }, + cluster: { + type: "boolean", + "default": false + }, + clusterRadius: { + type: "number", + "default": 50, + minimum: 0 + }, + clusterMaxZoom: { + type: "number" + }, + clusterMinPoints: { + type: "number" + }, + clusterProperties: { + type: "*" + }, + lineMetrics: { + type: "boolean", + "default": false + }, + generateId: { + type: "boolean", + "default": false + }, + promoteId: { + type: "promoteId" + } +}; +var source_video = { + type: { + required: true, + type: "enum", + values: { + video: { + } + } + }, + urls: { + required: true, + type: "array", + value: "string" + }, + coordinates: { + required: true, + type: "array", + length: 4, + value: { + type: "array", + length: 2, + value: "number" + } + } +}; +var source_image = { + type: { + required: true, + type: "enum", + values: { + image: { + } + } + }, + url: { + required: true, + type: "string" + }, + coordinates: { + required: true, + type: "array", + length: 4, + value: { + type: "array", + length: 2, + value: "number" + } + } +}; +var layer = { + id: { + type: "string", + required: true + }, + type: { + type: "enum", + values: { + fill: { + }, + line: { + }, + symbol: { + }, + circle: { + }, + heatmap: { + }, + "fill-extrusion": { + }, + raster: { + }, + hillshade: { + }, + background: { + } + }, + required: true + }, + metadata: { + type: "*" + }, + source: { + type: "string" + }, + "source-layer": { + type: "string" + }, + minzoom: { + type: "number", + minimum: 0, + maximum: 24 + }, + maxzoom: { + type: "number", + minimum: 0, + maximum: 24 + }, + filter: { + type: "filter" + }, + layout: { + type: "layout" + }, + paint: { + type: "paint" + } +}; +var layout$7 = [ + "layout_fill", + "layout_line", + "layout_circle", + "layout_heatmap", + "layout_fill-extrusion", + "layout_symbol", + "layout_raster", + "layout_hillshade", + "layout_background" +]; +var layout_background = { + visibility: { + type: "enum", + values: { + visible: { + }, + none: { + } + }, + "default": "visible", + "property-type": "constant" + } +}; +var layout_fill = { + "fill-sort-key": { + type: "number", + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + visibility: { + type: "enum", + values: { + visible: { + }, + none: { + } + }, + "default": "visible", + "property-type": "constant" + } +}; +var layout_circle = { + "circle-sort-key": { + type: "number", + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + visibility: { + type: "enum", + values: { + visible: { + }, + none: { + } + }, + "default": "visible", + "property-type": "constant" + } +}; +var layout_heatmap = { + visibility: { + type: "enum", + values: { + visible: { + }, + none: { + } + }, + "default": "visible", + "property-type": "constant" + } +}; +var layout_line = { + "line-cap": { + type: "enum", + values: { + butt: { + }, + round: { + }, + square: { + } + }, + "default": "butt", + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "line-join": { + type: "enum", + values: { + bevel: { + }, + round: { + }, + miter: { + } + }, + "default": "miter", + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "line-miter-limit": { + type: "number", + "default": 2, + requires: [ + { + "line-join": "miter" + } + ], + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "line-round-limit": { + type: "number", + "default": 1.05, + requires: [ + { + "line-join": "round" + } + ], + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "line-sort-key": { + type: "number", + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + visibility: { + type: "enum", + values: { + visible: { + }, + none: { + } + }, + "default": "visible", + "property-type": "constant" + } +}; +var layout_symbol = { + "symbol-placement": { + type: "enum", + values: { + point: { + }, + line: { + }, + "line-center": { + } + }, + "default": "point", + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "symbol-spacing": { + type: "number", + "default": 250, + minimum: 1, + units: "pixels", + requires: [ + { + "symbol-placement": "line" + } + ], + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "symbol-avoid-edges": { + type: "boolean", + "default": false, + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "symbol-sort-key": { + type: "number", + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "symbol-z-order": { + type: "enum", + values: { + auto: { + }, + "viewport-y": { + }, + source: { + } + }, + "default": "auto", + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "icon-allow-overlap": { + type: "boolean", + "default": false, + requires: [ + "icon-image", + { + "!": "icon-overlap" + } + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "icon-overlap": { + type: "enum", + values: { + never: { + }, + always: { + }, + cooperative: { + } + }, + requires: [ + "icon-image" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "icon-ignore-placement": { + type: "boolean", + "default": false, + requires: [ + "icon-image" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "icon-optional": { + type: "boolean", + "default": false, + requires: [ + "icon-image", + "text-field" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "icon-rotation-alignment": { + type: "enum", + values: { + map: { + }, + viewport: { + }, + auto: { + } + }, + "default": "auto", + requires: [ + "icon-image" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "icon-size": { + type: "number", + "default": 1, + minimum: 0, + units: "factor of the original icon size", + requires: [ + "icon-image" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "icon-text-fit": { + type: "enum", + values: { + none: { + }, + width: { + }, + height: { + }, + both: { + } + }, + "default": "none", + requires: [ + "icon-image", + "text-field" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "icon-text-fit-padding": { + type: "array", + value: "number", + length: 4, + "default": [ + 0, + 0, + 0, + 0 + ], + units: "pixels", + requires: [ + "icon-image", + "text-field", + { + "icon-text-fit": [ + "both", + "width", + "height" + ] + } + ], + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "icon-image": { + type: "resolvedImage", + tokens: true, + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "icon-rotate": { + type: "number", + "default": 0, + period: 360, + units: "degrees", + requires: [ + "icon-image" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "icon-padding": { + type: "padding", + "default": [ + 2 + ], + units: "pixels", + requires: [ + "icon-image" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "icon-keep-upright": { + type: "boolean", + "default": false, + requires: [ + "icon-image", + { + "icon-rotation-alignment": "map" + }, + { + "symbol-placement": [ + "line", + "line-center" + ] + } + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "icon-offset": { + type: "array", + value: "number", + length: 2, + "default": [ + 0, + 0 + ], + requires: [ + "icon-image" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "icon-anchor": { + type: "enum", + values: { + center: { + }, + left: { + }, + right: { + }, + top: { + }, + bottom: { + }, + "top-left": { + }, + "top-right": { + }, + "bottom-left": { + }, + "bottom-right": { + } + }, + "default": "center", + requires: [ + "icon-image" + ], + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "icon-pitch-alignment": { + type: "enum", + values: { + map: { + }, + viewport: { + }, + auto: { + } + }, + "default": "auto", + requires: [ + "icon-image" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-pitch-alignment": { + type: "enum", + values: { + map: { + }, + viewport: { + }, + auto: { + } + }, + "default": "auto", + requires: [ + "text-field" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-rotation-alignment": { + type: "enum", + values: { + map: { + }, + viewport: { + }, + "viewport-glyph": { + }, + auto: { + } + }, + "default": "auto", + requires: [ + "text-field" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-field": { + type: "formatted", + "default": "", + tokens: true, + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "text-font": { + type: "array", + value: "string", + "default": [ + "Open Sans Regular", + "Arial Unicode MS Regular" + ], + requires: [ + "text-field" + ], + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "text-size": { + type: "number", + "default": 16, + minimum: 0, + units: "pixels", + requires: [ + "text-field" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "text-max-width": { + type: "number", + "default": 10, + minimum: 0, + units: "ems", + requires: [ + "text-field" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "text-line-height": { + type: "number", + "default": 1.2, + units: "ems", + requires: [ + "text-field" + ], + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-letter-spacing": { + type: "number", + "default": 0, + units: "ems", + requires: [ + "text-field" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "text-justify": { + type: "enum", + values: { + auto: { + }, + left: { + }, + center: { + }, + right: { + } + }, + "default": "center", + requires: [ + "text-field" + ], + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "text-radial-offset": { + type: "number", + units: "ems", + "default": 0, + requires: [ + "text-field" + ], + "property-type": "data-driven", + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature" + ] + } + }, + "text-variable-anchor": { + type: "array", + value: "enum", + values: { + center: { + }, + left: { + }, + right: { + }, + top: { + }, + bottom: { + }, + "top-left": { + }, + "top-right": { + }, + "bottom-left": { + }, + "bottom-right": { + } + }, + requires: [ + "text-field", + { + "symbol-placement": [ + "point" + ] + } + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-variable-anchor-offset": { + type: "variableAnchorOffsetCollection", + requires: [ + "text-field", + { + "symbol-placement": [ + "point" + ] + } + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "text-anchor": { + type: "enum", + values: { + center: { + }, + left: { + }, + right: { + }, + top: { + }, + bottom: { + }, + "top-left": { + }, + "top-right": { + }, + "bottom-left": { + }, + "bottom-right": { + } + }, + "default": "center", + requires: [ + "text-field", + { + "!": "text-variable-anchor" + } + ], + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "text-max-angle": { + type: "number", + "default": 45, + units: "degrees", + requires: [ + "text-field", + { + "symbol-placement": [ + "line", + "line-center" + ] + } + ], + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-writing-mode": { + type: "array", + value: "enum", + values: { + horizontal: { + }, + vertical: { + } + }, + requires: [ + "text-field", + { + "symbol-placement": [ + "point" + ] + } + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-rotate": { + type: "number", + "default": 0, + period: 360, + units: "degrees", + requires: [ + "text-field" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "text-padding": { + type: "number", + "default": 2, + minimum: 0, + units: "pixels", + requires: [ + "text-field" + ], + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-keep-upright": { + type: "boolean", + "default": true, + requires: [ + "text-field", + { + "text-rotation-alignment": "map" + }, + { + "symbol-placement": [ + "line", + "line-center" + ] + } + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-transform": { + type: "enum", + values: { + none: { + }, + uppercase: { + }, + lowercase: { + } + }, + "default": "none", + requires: [ + "text-field" + ], + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "text-offset": { + type: "array", + value: "number", + units: "ems", + length: 2, + "default": [ + 0, + 0 + ], + requires: [ + "text-field", + { + "!": "text-radial-offset" + } + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "data-driven" + }, + "text-allow-overlap": { + type: "boolean", + "default": false, + requires: [ + "text-field", + { + "!": "text-overlap" + } + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-overlap": { + type: "enum", + values: { + never: { + }, + always: { + }, + cooperative: { + } + }, + requires: [ + "text-field" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-ignore-placement": { + type: "boolean", + "default": false, + requires: [ + "text-field" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-optional": { + type: "boolean", + "default": false, + requires: [ + "text-field", + "icon-image" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + visibility: { + type: "enum", + values: { + visible: { + }, + none: { + } + }, + "default": "visible", + "property-type": "constant" + } +}; +var layout_raster = { + visibility: { + type: "enum", + values: { + visible: { + }, + none: { + } + }, + "default": "visible", + "property-type": "constant" + } +}; +var layout_hillshade = { + visibility: { + type: "enum", + values: { + visible: { + }, + none: { + } + }, + "default": "visible", + "property-type": "constant" + } +}; +var filter = { + type: "array", + value: "*" +}; +var filter_operator = { + type: "enum", + values: { + "==": { + }, + "!=": { + }, + ">": { + }, + ">=": { + }, + "<": { + }, + "<=": { + }, + "in": { + }, + "!in": { + }, + all: { + }, + any: { + }, + none: { + }, + has: { + }, + "!has": { + }, + within: { + } + } +}; +var geometry_type = { + type: "enum", + values: { + Point: { + }, + LineString: { + }, + Polygon: { + } + } +}; +var function_stop = { + type: "array", + minimum: 0, + maximum: 24, + value: [ + "number", + "color" + ], + length: 2 +}; +var expression$1 = { + type: "array", + value: "*", + minimum: 1 +}; +var light = { + anchor: { + type: "enum", + "default": "viewport", + values: { + map: { + }, + viewport: { + } + }, + "property-type": "data-constant", + transition: false, + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + } + }, + position: { + type: "array", + "default": [ + 1.15, + 210, + 30 + ], + length: 3, + value: "number", + "property-type": "data-constant", + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + } + }, + color: { + type: "color", + "property-type": "data-constant", + "default": "#ffffff", + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + transition: true + }, + intensity: { + type: "number", + "property-type": "data-constant", + "default": 0.5, + minimum: 0, + maximum: 1, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + transition: true + } +}; +var terrain = { + source: { + type: "string", + required: true + }, + exaggeration: { + type: "number", + minimum: 0, + "default": 1 + } +}; +var paint$9 = [ + "paint_fill", + "paint_line", + "paint_circle", + "paint_heatmap", + "paint_fill-extrusion", + "paint_symbol", + "paint_raster", + "paint_hillshade", + "paint_background" +]; +var paint_fill = { + "fill-antialias": { + type: "boolean", + "default": true, + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "fill-opacity": { + type: "number", + "default": 1, + minimum: 0, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "fill-color": { + type: "color", + "default": "#000000", + transition: true, + requires: [ + { + "!": "fill-pattern" + } + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "fill-outline-color": { + type: "color", + transition: true, + requires: [ + { + "!": "fill-pattern" + }, + { + "fill-antialias": true + } + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "fill-translate": { + type: "array", + value: "number", + length: 2, + "default": [ + 0, + 0 + ], + transition: true, + units: "pixels", + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "fill-translate-anchor": { + type: "enum", + values: { + map: { + }, + viewport: { + } + }, + "default": "map", + requires: [ + "fill-translate" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "fill-pattern": { + type: "resolvedImage", + transition: true, + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "cross-faded-data-driven" + } +}; +var paint_line = { + "line-opacity": { + type: "number", + "default": 1, + minimum: 0, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "line-color": { + type: "color", + "default": "#000000", + transition: true, + requires: [ + { + "!": "line-pattern" + } + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "line-translate": { + type: "array", + value: "number", + length: 2, + "default": [ + 0, + 0 + ], + transition: true, + units: "pixels", + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "line-translate-anchor": { + type: "enum", + values: { + map: { + }, + viewport: { + } + }, + "default": "map", + requires: [ + "line-translate" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "line-width": { + type: "number", + "default": 1, + minimum: 0, + transition: true, + units: "pixels", + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "line-gap-width": { + type: "number", + "default": 0, + minimum: 0, + transition: true, + units: "pixels", + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "line-offset": { + type: "number", + "default": 0, + transition: true, + units: "pixels", + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "line-blur": { + type: "number", + "default": 0, + minimum: 0, + transition: true, + units: "pixels", + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "line-dasharray": { + type: "array", + value: "number", + minimum: 0, + transition: true, + units: "line widths", + requires: [ + { + "!": "line-pattern" + } + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "cross-faded" + }, + "line-pattern": { + type: "resolvedImage", + transition: true, + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "cross-faded-data-driven" + }, + "line-gradient": { + type: "color", + transition: false, + requires: [ + { + "!": "line-dasharray" + }, + { + "!": "line-pattern" + }, + { + source: "geojson", + has: { + lineMetrics: true + } + } + ], + expression: { + interpolated: true, + parameters: [ + "line-progress" + ] + }, + "property-type": "color-ramp" + } +}; +var paint_circle = { + "circle-radius": { + type: "number", + "default": 5, + minimum: 0, + transition: true, + units: "pixels", + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "circle-color": { + type: "color", + "default": "#000000", + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "circle-blur": { + type: "number", + "default": 0, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "circle-opacity": { + type: "number", + "default": 1, + minimum: 0, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "circle-translate": { + type: "array", + value: "number", + length: 2, + "default": [ + 0, + 0 + ], + transition: true, + units: "pixels", + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "circle-translate-anchor": { + type: "enum", + values: { + map: { + }, + viewport: { + } + }, + "default": "map", + requires: [ + "circle-translate" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "circle-pitch-scale": { + type: "enum", + values: { + map: { + }, + viewport: { + } + }, + "default": "map", + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "circle-pitch-alignment": { + type: "enum", + values: { + map: { + }, + viewport: { + } + }, + "default": "viewport", + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "circle-stroke-width": { + type: "number", + "default": 0, + minimum: 0, + transition: true, + units: "pixels", + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "circle-stroke-color": { + type: "color", + "default": "#000000", + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "circle-stroke-opacity": { + type: "number", + "default": 1, + minimum: 0, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + } +}; +var paint_heatmap = { + "heatmap-radius": { + type: "number", + "default": 30, + minimum: 1, + transition: true, + units: "pixels", + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "heatmap-weight": { + type: "number", + "default": 1, + minimum: 0, + transition: false, + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "heatmap-intensity": { + type: "number", + "default": 1, + minimum: 0, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "heatmap-color": { + type: "color", + "default": [ + "interpolate", + [ + "linear" + ], + [ + "heatmap-density" + ], + 0, + "rgba(0, 0, 255, 0)", + 0.1, + "royalblue", + 0.3, + "cyan", + 0.5, + "lime", + 0.7, + "yellow", + 1, + "red" + ], + transition: false, + expression: { + interpolated: true, + parameters: [ + "heatmap-density" + ] + }, + "property-type": "color-ramp" + }, + "heatmap-opacity": { + type: "number", + "default": 1, + minimum: 0, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + } +}; +var paint_symbol = { + "icon-opacity": { + type: "number", + "default": 1, + minimum: 0, + maximum: 1, + transition: true, + requires: [ + "icon-image" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "icon-color": { + type: "color", + "default": "#000000", + transition: true, + requires: [ + "icon-image" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "icon-halo-color": { + type: "color", + "default": "rgba(0, 0, 0, 0)", + transition: true, + requires: [ + "icon-image" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "icon-halo-width": { + type: "number", + "default": 0, + minimum: 0, + transition: true, + units: "pixels", + requires: [ + "icon-image" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "icon-halo-blur": { + type: "number", + "default": 0, + minimum: 0, + transition: true, + units: "pixels", + requires: [ + "icon-image" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "icon-translate": { + type: "array", + value: "number", + length: 2, + "default": [ + 0, + 0 + ], + transition: true, + units: "pixels", + requires: [ + "icon-image" + ], + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "icon-translate-anchor": { + type: "enum", + values: { + map: { + }, + viewport: { + } + }, + "default": "map", + requires: [ + "icon-image", + "icon-translate" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-opacity": { + type: "number", + "default": 1, + minimum: 0, + maximum: 1, + transition: true, + requires: [ + "text-field" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "text-color": { + type: "color", + "default": "#000000", + transition: true, + overridable: true, + requires: [ + "text-field" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "text-halo-color": { + type: "color", + "default": "rgba(0, 0, 0, 0)", + transition: true, + requires: [ + "text-field" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "text-halo-width": { + type: "number", + "default": 0, + minimum: 0, + transition: true, + units: "pixels", + requires: [ + "text-field" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "text-halo-blur": { + type: "number", + "default": 0, + minimum: 0, + transition: true, + units: "pixels", + requires: [ + "text-field" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "text-translate": { + type: "array", + value: "number", + length: 2, + "default": [ + 0, + 0 + ], + transition: true, + units: "pixels", + requires: [ + "text-field" + ], + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "text-translate-anchor": { + type: "enum", + values: { + map: { + }, + viewport: { + } + }, + "default": "map", + requires: [ + "text-field", + "text-translate" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + } +}; +var paint_raster = { + "raster-opacity": { + type: "number", + "default": 1, + minimum: 0, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "raster-hue-rotate": { + type: "number", + "default": 0, + period: 360, + transition: true, + units: "degrees", + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "raster-brightness-min": { + type: "number", + "default": 0, + minimum: 0, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "raster-brightness-max": { + type: "number", + "default": 1, + minimum: 0, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "raster-saturation": { + type: "number", + "default": 0, + minimum: -1, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "raster-contrast": { + type: "number", + "default": 0, + minimum: -1, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "raster-resampling": { + type: "enum", + values: { + linear: { + }, + nearest: { + } + }, + "default": "linear", + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "raster-fade-duration": { + type: "number", + "default": 300, + minimum: 0, + transition: false, + units: "milliseconds", + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + } +}; +var paint_hillshade = { + "hillshade-illumination-direction": { + type: "number", + "default": 335, + minimum: 0, + maximum: 359, + transition: false, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "hillshade-illumination-anchor": { + type: "enum", + values: { + map: { + }, + viewport: { + } + }, + "default": "viewport", + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "hillshade-exaggeration": { + type: "number", + "default": 0.5, + minimum: 0, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "hillshade-shadow-color": { + type: "color", + "default": "#000000", + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "hillshade-highlight-color": { + type: "color", + "default": "#FFFFFF", + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "hillshade-accent-color": { + type: "color", + "default": "#000000", + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + } +}; +var paint_background = { + "background-color": { + type: "color", + "default": "#000000", + transition: true, + requires: [ + { + "!": "background-pattern" + } + ], + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "background-pattern": { + type: "resolvedImage", + transition: true, + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "cross-faded" + }, + "background-opacity": { + type: "number", + "default": 1, + minimum: 0, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + } +}; +var transition = { + duration: { + type: "number", + "default": 300, + minimum: 0, + units: "milliseconds" + }, + delay: { + type: "number", + "default": 0, + minimum: 0, + units: "milliseconds" + } +}; +var promoteId = { + "*": { + type: "string" + } +}; +var v8Spec = { + $version: $version, + $root: $root, + sources: sources, + source: source, + source_vector: source_vector, + source_raster: source_raster, + source_raster_dem: source_raster_dem, + source_geojson: source_geojson, + source_video: source_video, + source_image: source_image, + layer: layer, + layout: layout$7, + layout_background: layout_background, + layout_fill: layout_fill, + layout_circle: layout_circle, + layout_heatmap: layout_heatmap, + "layout_fill-extrusion": { + visibility: { + type: "enum", + values: { + visible: { + }, + none: { + } + }, + "default": "visible", + "property-type": "constant" + } +}, + layout_line: layout_line, + layout_symbol: layout_symbol, + layout_raster: layout_raster, + layout_hillshade: layout_hillshade, + filter: filter, + filter_operator: filter_operator, + geometry_type: geometry_type, + "function": { + expression: { + type: "expression" + }, + stops: { + type: "array", + value: "function_stop" + }, + base: { + type: "number", + "default": 1, + minimum: 0 + }, + property: { + type: "string", + "default": "$zoom" + }, + type: { + type: "enum", + values: { + identity: { + }, + exponential: { + }, + interval: { + }, + categorical: { + } + }, + "default": "exponential" + }, + colorSpace: { + type: "enum", + values: { + rgb: { + }, + lab: { + }, + hcl: { + } + }, + "default": "rgb" + }, + "default": { + type: "*", + required: false + } +}, + function_stop: function_stop, + expression: expression$1, + light: light, + terrain: terrain, + paint: paint$9, + paint_fill: paint_fill, + "paint_fill-extrusion": { + "fill-extrusion-opacity": { + type: "number", + "default": 1, + minimum: 0, + maximum: 1, + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "fill-extrusion-color": { + type: "color", + "default": "#000000", + transition: true, + requires: [ + { + "!": "fill-extrusion-pattern" + } + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "fill-extrusion-translate": { + type: "array", + value: "number", + length: 2, + "default": [ + 0, + 0 + ], + transition: true, + units: "pixels", + expression: { + interpolated: true, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "fill-extrusion-translate-anchor": { + type: "enum", + values: { + map: { + }, + viewport: { + } + }, + "default": "map", + requires: [ + "fill-extrusion-translate" + ], + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + }, + "fill-extrusion-pattern": { + type: "resolvedImage", + transition: true, + expression: { + interpolated: false, + parameters: [ + "zoom", + "feature" + ] + }, + "property-type": "cross-faded-data-driven" + }, + "fill-extrusion-height": { + type: "number", + "default": 0, + minimum: 0, + units: "meters", + transition: true, + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "fill-extrusion-base": { + type: "number", + "default": 0, + minimum: 0, + units: "meters", + transition: true, + requires: [ + "fill-extrusion-height" + ], + expression: { + interpolated: true, + parameters: [ + "zoom", + "feature", + "feature-state" + ] + }, + "property-type": "data-driven" + }, + "fill-extrusion-vertical-gradient": { + type: "boolean", + "default": true, + transition: false, + expression: { + interpolated: false, + parameters: [ + "zoom" + ] + }, + "property-type": "data-constant" + } +}, + paint_line: paint_line, + paint_circle: paint_circle, + paint_heatmap: paint_heatmap, + paint_symbol: paint_symbol, + paint_raster: paint_raster, + paint_hillshade: paint_hillshade, + paint_background: paint_background, + transition: transition, + "property-type": { + "data-driven": { + type: "property-type" + }, + "cross-faded": { + type: "property-type" + }, + "cross-faded-data-driven": { + type: "property-type" + }, + "color-ramp": { + type: "property-type" + }, + "data-constant": { + type: "property-type" + }, + constant: { + type: "property-type" + } +}, + promoteId: promoteId +}; + +const refProperties = ['type', 'source', 'source-layer', 'minzoom', 'maxzoom', 'filter', 'layout']; + +function deref(layer, parent) { + const result = {}; + for (const k in layer) { + if (k !== 'ref') { + result[k] = layer[k]; + } + } + refProperties.forEach((k) => { + if (k in parent) { + result[k] = parent[k]; + } + }); + return result; +} +/** + * Given an array of layers, some of which may contain `ref` properties + * whose value is the `id` of another property, return a new array where + * such layers have been augmented with the 'type', 'source', etc. properties + * from the parent layer, and the `ref` property has been removed. + * + * The input is not modified. The output may contain references to portions + * of the input. + * + * @private + * @param {Array} layers + * @returns {Array} + */ +function derefLayers(layers) { + layers = layers.slice(); + const map = Object.create(null); + for (let i = 0; i < layers.length; i++) { + map[layers[i].id] = layers[i]; + } + for (let i = 0; i < layers.length; i++) { + if ('ref' in layers[i]) { + layers[i] = deref(layers[i], map[layers[i].ref]); + } + } + return layers; +} + +/** + * Deeply compares two object literals. + * + * @private + */ +function deepEqual(a, b) { + if (Array.isArray(a)) { + if (!Array.isArray(b) || a.length !== b.length) + return false; + for (let i = 0; i < a.length; i++) { + if (!deepEqual(a[i], b[i])) + return false; + } + return true; + } + if (typeof a === 'object' && a !== null && b !== null) { + if (!(typeof b === 'object')) + return false; + const keys = Object.keys(a); + if (keys.length !== Object.keys(b).length) + return false; + for (const key in a) { + if (!deepEqual(a[key], b[key])) + return false; + } + return true; + } + return a === b; +} + +const operations = { + /* + * { command: 'setStyle', args: [stylesheet] } + */ + setStyle: 'setStyle', + /* + * { command: 'addLayer', args: [layer, 'beforeLayerId'] } + */ + addLayer: 'addLayer', + /* + * { command: 'removeLayer', args: ['layerId'] } + */ + removeLayer: 'removeLayer', + /* + * { command: 'setPaintProperty', args: ['layerId', 'prop', value] } + */ + setPaintProperty: 'setPaintProperty', + /* + * { command: 'setLayoutProperty', args: ['layerId', 'prop', value] } + */ + setLayoutProperty: 'setLayoutProperty', + /* + * { command: 'setFilter', args: ['layerId', filter] } + */ + setFilter: 'setFilter', + /* + * { command: 'addSource', args: ['sourceId', source] } + */ + addSource: 'addSource', + /* + * { command: 'removeSource', args: ['sourceId'] } + */ + removeSource: 'removeSource', + /* + * { command: 'setGeoJSONSourceData', args: ['sourceId', data] } + */ + setGeoJSONSourceData: 'setGeoJSONSourceData', + /* + * { command: 'setLayerZoomRange', args: ['layerId', 0, 22] } + */ + setLayerZoomRange: 'setLayerZoomRange', + /* + * { command: 'setLayerProperty', args: ['layerId', 'prop', value] } + */ + setLayerProperty: 'setLayerProperty', + /* + * { command: 'setCenter', args: [[lon, lat]] } + */ + setCenter: 'setCenter', + /* + * { command: 'setZoom', args: [zoom] } + */ + setZoom: 'setZoom', + /* + * { command: 'setBearing', args: [bearing] } + */ + setBearing: 'setBearing', + /* + * { command: 'setPitch', args: [pitch] } + */ + setPitch: 'setPitch', + /* + * { command: 'setSprite', args: ['spriteUrl'] } + */ + setSprite: 'setSprite', + /* + * { command: 'setGlyphs', args: ['glyphsUrl'] } + */ + setGlyphs: 'setGlyphs', + /* + * { command: 'setTransition', args: [transition] } + */ + setTransition: 'setTransition', + /* + * { command: 'setLighting', args: [lightProperties] } + */ + setLight: 'setLight' +}; +function addSource(sourceId, after, commands) { + commands.push({ command: operations.addSource, args: [sourceId, after[sourceId]] }); +} +function removeSource(sourceId, commands, sourcesRemoved) { + commands.push({ command: operations.removeSource, args: [sourceId] }); + sourcesRemoved[sourceId] = true; +} +function updateSource(sourceId, after, commands, sourcesRemoved) { + removeSource(sourceId, commands, sourcesRemoved); + addSource(sourceId, after, commands); +} +function canUpdateGeoJSON(before, after, sourceId) { + let prop; + for (prop in before[sourceId]) { + if (!Object.prototype.hasOwnProperty.call(before[sourceId], prop)) + continue; + if (prop !== 'data' && !deepEqual(before[sourceId][prop], after[sourceId][prop])) { + return false; + } + } + for (prop in after[sourceId]) { + if (!Object.prototype.hasOwnProperty.call(after[sourceId], prop)) + continue; + if (prop !== 'data' && !deepEqual(before[sourceId][prop], after[sourceId][prop])) { + return false; + } + } + return true; +} +function diffSources(before, after, commands, sourcesRemoved) { + before = before || {}; + after = after || {}; + let sourceId; + // look for sources to remove + for (sourceId in before) { + if (!Object.prototype.hasOwnProperty.call(before, sourceId)) + continue; + if (!Object.prototype.hasOwnProperty.call(after, sourceId)) { + removeSource(sourceId, commands, sourcesRemoved); + } + } + // look for sources to add/update + for (sourceId in after) { + if (!Object.prototype.hasOwnProperty.call(after, sourceId)) + continue; + if (!Object.prototype.hasOwnProperty.call(before, sourceId)) { + addSource(sourceId, after, commands); + } + else if (!deepEqual(before[sourceId], after[sourceId])) { + if (before[sourceId].type === 'geojson' && after[sourceId].type === 'geojson' && canUpdateGeoJSON(before, after, sourceId)) { + commands.push({ command: operations.setGeoJSONSourceData, args: [sourceId, after[sourceId].data] }); + } + else { + // no update command, must remove then add + updateSource(sourceId, after, commands, sourcesRemoved); + } + } + } +} +function diffLayerPropertyChanges(before, after, commands, layerId, klass, command) { + before = before || {}; + after = after || {}; + let prop; + for (prop in before) { + if (!Object.prototype.hasOwnProperty.call(before, prop)) + continue; + if (!deepEqual(before[prop], after[prop])) { + commands.push({ command, args: [layerId, prop, after[prop], klass] }); + } + } + for (prop in after) { + if (!Object.prototype.hasOwnProperty.call(after, prop) || Object.prototype.hasOwnProperty.call(before, prop)) + continue; + if (!deepEqual(before[prop], after[prop])) { + commands.push({ command, args: [layerId, prop, after[prop], klass] }); + } + } +} +function pluckId(layer) { + return layer.id; +} +function indexById(group, layer) { + group[layer.id] = layer; + return group; +} +function diffLayers(before, after, commands) { + before = before || []; + after = after || []; + // order of layers by id + const beforeOrder = before.map(pluckId); + const afterOrder = after.map(pluckId); + // index of layer by id + const beforeIndex = before.reduce(indexById, {}); + const afterIndex = after.reduce(indexById, {}); + // track order of layers as if they have been mutated + const tracker = beforeOrder.slice(); + // layers that have been added do not need to be diffed + const clean = Object.create(null); + let i, d, layerId, beforeLayer, afterLayer, insertBeforeLayerId, prop; + // remove layers + for (i = 0, d = 0; i < beforeOrder.length; i++) { + layerId = beforeOrder[i]; + if (!Object.prototype.hasOwnProperty.call(afterIndex, layerId)) { + commands.push({ command: operations.removeLayer, args: [layerId] }); + tracker.splice(tracker.indexOf(layerId, d), 1); + } + else { + // limit where in tracker we need to look for a match + d++; + } + } + // add/reorder layers + for (i = 0, d = 0; i < afterOrder.length; i++) { + // work backwards as insert is before an existing layer + layerId = afterOrder[afterOrder.length - 1 - i]; + if (tracker[tracker.length - 1 - i] === layerId) + continue; + if (Object.prototype.hasOwnProperty.call(beforeIndex, layerId)) { + // remove the layer before we insert at the correct position + commands.push({ command: operations.removeLayer, args: [layerId] }); + tracker.splice(tracker.lastIndexOf(layerId, tracker.length - d), 1); + } + else { + // limit where in tracker we need to look for a match + d++; + } + // add layer at correct position + insertBeforeLayerId = tracker[tracker.length - i]; + commands.push({ command: operations.addLayer, args: [afterIndex[layerId], insertBeforeLayerId] }); + tracker.splice(tracker.length - i, 0, layerId); + clean[layerId] = true; + } + // update layers + for (i = 0; i < afterOrder.length; i++) { + layerId = afterOrder[i]; + beforeLayer = beforeIndex[layerId]; + afterLayer = afterIndex[layerId]; + // no need to update if previously added (new or moved) + if (clean[layerId] || deepEqual(beforeLayer, afterLayer)) + continue; + // If source, source-layer, or type have changes, then remove the layer + // and add it back 'from scratch'. + if (!deepEqual(beforeLayer.source, afterLayer.source) || !deepEqual(beforeLayer['source-layer'], afterLayer['source-layer']) || !deepEqual(beforeLayer.type, afterLayer.type)) { + commands.push({ command: operations.removeLayer, args: [layerId] }); + // we add the layer back at the same position it was already in, so + // there's no need to update the `tracker` + insertBeforeLayerId = tracker[tracker.lastIndexOf(layerId) + 1]; + commands.push({ command: operations.addLayer, args: [afterLayer, insertBeforeLayerId] }); + continue; + } + // layout, paint, filter, minzoom, maxzoom + diffLayerPropertyChanges(beforeLayer.layout, afterLayer.layout, commands, layerId, null, operations.setLayoutProperty); + diffLayerPropertyChanges(beforeLayer.paint, afterLayer.paint, commands, layerId, null, operations.setPaintProperty); + if (!deepEqual(beforeLayer.filter, afterLayer.filter)) { + commands.push({ command: operations.setFilter, args: [layerId, afterLayer.filter] }); + } + if (!deepEqual(beforeLayer.minzoom, afterLayer.minzoom) || !deepEqual(beforeLayer.maxzoom, afterLayer.maxzoom)) { + commands.push({ command: operations.setLayerZoomRange, args: [layerId, afterLayer.minzoom, afterLayer.maxzoom] }); + } + // handle all other layer props, including paint.* + for (prop in beforeLayer) { + if (!Object.prototype.hasOwnProperty.call(beforeLayer, prop)) + continue; + if (prop === 'layout' || prop === 'paint' || prop === 'filter' || + prop === 'metadata' || prop === 'minzoom' || prop === 'maxzoom') + continue; + if (prop.indexOf('paint.') === 0) { + diffLayerPropertyChanges(beforeLayer[prop], afterLayer[prop], commands, layerId, prop.slice(6), operations.setPaintProperty); + } + else if (!deepEqual(beforeLayer[prop], afterLayer[prop])) { + commands.push({ command: operations.setLayerProperty, args: [layerId, prop, afterLayer[prop]] }); + } + } + for (prop in afterLayer) { + if (!Object.prototype.hasOwnProperty.call(afterLayer, prop) || Object.prototype.hasOwnProperty.call(beforeLayer, prop)) + continue; + if (prop === 'layout' || prop === 'paint' || prop === 'filter' || + prop === 'metadata' || prop === 'minzoom' || prop === 'maxzoom') + continue; + if (prop.indexOf('paint.') === 0) { + diffLayerPropertyChanges(beforeLayer[prop], afterLayer[prop], commands, layerId, prop.slice(6), operations.setPaintProperty); + } + else if (!deepEqual(beforeLayer[prop], afterLayer[prop])) { + commands.push({ command: operations.setLayerProperty, args: [layerId, prop, afterLayer[prop]] }); + } + } + } +} +/** + * Diff two stylesheet + * + * Creates semanticly aware diffs that can easily be applied at runtime. + * Operations produced by the diff closely resemble the maplibre-gl-js API. Any + * error creating the diff will fall back to the 'setStyle' operation. + * + * Example diff: + * [ + * { command: 'setConstant', args: ['@water', '#0000FF'] }, + * { command: 'setPaintProperty', args: ['background', 'background-color', 'black'] } + * ] + * + * @private + * @param {*} [before] stylesheet to compare from + * @param {*} after stylesheet to compare to + * @returns Array list of changes + */ +function diffStyles(before, after) { + if (!before) + return [{ command: operations.setStyle, args: [after] }]; + let commands = []; + try { + // Handle changes to top-level properties + if (!deepEqual(before.version, after.version)) { + return [{ command: operations.setStyle, args: [after] }]; + } + if (!deepEqual(before.center, after.center)) { + commands.push({ command: operations.setCenter, args: [after.center] }); + } + if (!deepEqual(before.zoom, after.zoom)) { + commands.push({ command: operations.setZoom, args: [after.zoom] }); + } + if (!deepEqual(before.bearing, after.bearing)) { + commands.push({ command: operations.setBearing, args: [after.bearing] }); + } + if (!deepEqual(before.pitch, after.pitch)) { + commands.push({ command: operations.setPitch, args: [after.pitch] }); + } + if (!deepEqual(before.sprite, after.sprite)) { + commands.push({ command: operations.setSprite, args: [after.sprite] }); + } + if (!deepEqual(before.glyphs, after.glyphs)) { + commands.push({ command: operations.setGlyphs, args: [after.glyphs] }); + } + if (!deepEqual(before.transition, after.transition)) { + commands.push({ command: operations.setTransition, args: [after.transition] }); + } + if (!deepEqual(before.light, after.light)) { + commands.push({ command: operations.setLight, args: [after.light] }); + } + // Handle changes to `sources` + // If a source is to be removed, we also--before the removeSource + // command--need to remove all the style layers that depend on it. + const sourcesRemoved = {}; + // First collect the {add,remove}Source commands + const removeOrAddSourceCommands = []; + diffSources(before.sources, after.sources, removeOrAddSourceCommands, sourcesRemoved); + // Push a removeLayer command for each style layer that depends on a + // source that's being removed. + // Also, exclude any such layers them from the input to `diffLayers` + // below, so that diffLayers produces the appropriate `addLayers` + // command + const beforeLayers = []; + if (before.layers) { + before.layers.forEach((layer) => { + if (sourcesRemoved[layer.source]) { + commands.push({ command: operations.removeLayer, args: [layer.id] }); + } + else { + beforeLayers.push(layer); + } + }); + } + commands = commands.concat(removeOrAddSourceCommands); + // Handle changes to `layers` + diffLayers(beforeLayers, after.layers, commands); + } + catch (e) { + // fall back to setStyle + console.warn('Unable to compute style diff:', e); + commands = [{ command: operations.setStyle, args: [after] }]; + } + return commands; +} + +// Note: Do not inherit from Error. It breaks when transpiling to ES5. +class ValidationError { + constructor(key, value, message, identifier) { + this.message = (key ? `${key}: ` : '') + message; + if (identifier) + this.identifier = identifier; + if (value !== null && value !== undefined && value.__line__) { + this.line = value.__line__; + } + } +} + +// Note: Do not inherit from Error. It breaks when transpiling to ES5. +class ParsingError { + constructor(error) { + this.error = error; + this.message = error.message; + const match = error.message.match(/line (\d+)/); + this.line = match ? parseInt(match[1], 10) : 0; + } +} + +function extendBy(output, ...inputs) { + for (const input of inputs) { + for (const k in input) { + output[k] = input[k]; + } + } + return output; +} + +class ExpressionParsingError extends Error { + constructor(key, message) { + super(message); + this.message = message; + this.key = key; + } +} + +/** + * Tracks `let` bindings during expression parsing. + * @private + */ +class Scope { + constructor(parent, bindings = []) { + this.parent = parent; + this.bindings = {}; + for (const [name, expression] of bindings) { + this.bindings[name] = expression; + } + } + concat(bindings) { + return new Scope(this, bindings); + } + get(name) { + if (this.bindings[name]) { + return this.bindings[name]; + } + if (this.parent) { + return this.parent.get(name); + } + throw new Error(`${name} not found in scope.`); + } + has(name) { + if (this.bindings[name]) + return true; + return this.parent ? this.parent.has(name) : false; + } +} + +const NullType = { kind: 'null' }; +const NumberType = { kind: 'number' }; +const StringType = { kind: 'string' }; +const BooleanType = { kind: 'boolean' }; +const ColorType = { kind: 'color' }; +const ObjectType = { kind: 'object' }; +const ValueType = { kind: 'value' }; +const ErrorType = { kind: 'error' }; +const CollatorType = { kind: 'collator' }; +const FormattedType = { kind: 'formatted' }; +const PaddingType = { kind: 'padding' }; +const ResolvedImageType = { kind: 'resolvedImage' }; +const VariableAnchorOffsetCollectionType = { kind: 'variableAnchorOffsetCollection' }; +function array$1(itemType, N) { + return { + kind: 'array', + itemType, + N + }; +} +function toString$1(type) { + if (type.kind === 'array') { + const itemType = toString$1(type.itemType); + return typeof type.N === 'number' ? + `array<${itemType}, ${type.N}>` : + type.itemType.kind === 'value' ? 'array' : `array<${itemType}>`; + } + else { + return type.kind; + } +} +const valueMemberTypes = [ + NullType, + NumberType, + StringType, + BooleanType, + ColorType, + FormattedType, + ObjectType, + array$1(ValueType), + PaddingType, + ResolvedImageType, + VariableAnchorOffsetCollectionType +]; +/** + * Returns null if `t` is a subtype of `expected`; otherwise returns an + * error message. + * @private + */ +function checkSubtype(expected, t) { + if (t.kind === 'error') { + // Error is a subtype of every type + return null; + } + else if (expected.kind === 'array') { + if (t.kind === 'array' && + ((t.N === 0 && t.itemType.kind === 'value') || !checkSubtype(expected.itemType, t.itemType)) && + (typeof expected.N !== 'number' || expected.N === t.N)) { + return null; + } + } + else if (expected.kind === t.kind) { + return null; + } + else if (expected.kind === 'value') { + for (const memberType of valueMemberTypes) { + if (!checkSubtype(memberType, t)) { + return null; + } + } + } + return `Expected ${toString$1(expected)} but found ${toString$1(t)} instead.`; +} +function isValidType(provided, allowedTypes) { + return allowedTypes.some(t => t.kind === provided.kind); +} +function isValidNativeType(provided, allowedTypes) { + return allowedTypes.some(t => { + if (t === 'null') { + return provided === null; + } + else if (t === 'array') { + return Array.isArray(provided); + } + else if (t === 'object') { + return provided && !Array.isArray(provided) && typeof provided === 'object'; + } + else { + return t === typeof provided; + } + }); +} +/** + * Verify whether the specified type is of the same type as the specified sample. + * + * @param provided Type to verify + * @param sample Sample type to reference + * @returns `true` if both objects are of the same type, `false` otherwise + * @example basic types + * if (verifyType(outputType, ValueType)) { + * // type narrowed to: + * outputType.kind; // 'value' + * } + * @example array types + * if (verifyType(outputType, array(NumberType))) { + * // type narrowed to: + * outputType.kind; // 'array' + * outputType.itemType; // NumberTypeT + * outputType.itemType.kind; // 'number' + * } + */ +function verifyType(provided, sample) { + if (provided.kind === 'array' && sample.kind === 'array') { + return provided.itemType.kind === sample.itemType.kind && typeof provided.N === 'number'; + } + return provided.kind === sample.kind; +} + +// See https://observablehq.com/@mbostock/lab-and-rgb +const Xn = 0.96422, Yn = 1, Zn = 0.82521, t0 = 4 / 29, t1 = 6 / 29, t2 = 3 * t1 * t1, t3 = t1 * t1 * t1, deg2rad = Math.PI / 180, rad2deg = 180 / Math.PI; +function constrainAngle(angle) { + angle = angle % 360; + if (angle < 0) { + angle += 360; + } + return angle; +} +function rgbToLab([r, g, b, alpha]) { + r = rgb2xyz(r); + g = rgb2xyz(g); + b = rgb2xyz(b); + let x, z; + const y = xyz2lab((0.2225045 * r + 0.7168786 * g + 0.0606169 * b) / Yn); + if (r === g && g === b) { + x = z = y; + } + else { + x = xyz2lab((0.4360747 * r + 0.3850649 * g + 0.1430804 * b) / Xn); + z = xyz2lab((0.0139322 * r + 0.0971045 * g + 0.7141733 * b) / Zn); + } + const l = 116 * y - 16; + return [(l < 0) ? 0 : l, 500 * (x - y), 200 * (y - z), alpha]; +} +function rgb2xyz(x) { + return (x <= 0.04045) ? x / 12.92 : Math.pow((x + 0.055) / 1.055, 2.4); +} +function xyz2lab(t) { + return (t > t3) ? Math.pow(t, 1 / 3) : t / t2 + t0; +} +function labToRgb([l, a, b, alpha]) { + let y = (l + 16) / 116, x = isNaN(a) ? y : y + a / 500, z = isNaN(b) ? y : y - b / 200; + y = Yn * lab2xyz(y); + x = Xn * lab2xyz(x); + z = Zn * lab2xyz(z); + return [ + xyz2rgb(3.1338561 * x - 1.6168667 * y - 0.4906146 * z), + xyz2rgb(-0.9787684 * x + 1.9161415 * y + 0.0334540 * z), + xyz2rgb(0.0719453 * x - 0.2289914 * y + 1.4052427 * z), + alpha, + ]; +} +function xyz2rgb(x) { + x = (x <= 0.00304) ? 12.92 * x : 1.055 * Math.pow(x, 1 / 2.4) - 0.055; + return (x < 0) ? 0 : (x > 1) ? 1 : x; // clip to 0..1 range +} +function lab2xyz(t) { + return (t > t1) ? t * t * t : t2 * (t - t0); +} +function rgbToHcl(rgbColor) { + const [l, a, b, alpha] = rgbToLab(rgbColor); + const c = Math.sqrt(a * a + b * b); + const h = Math.round(c * 10000) ? constrainAngle(Math.atan2(b, a) * rad2deg) : NaN; + return [h, c, l, alpha]; +} +function hclToRgb([h, c, l, alpha]) { + h = isNaN(h) ? 0 : h * deg2rad; + return labToRgb([l, Math.cos(h) * c, Math.sin(h) * c, alpha]); +} +// https://drafts.csswg.org/css-color-4/#hsl-to-rgb +function hslToRgb([h, s, l, alpha]) { + h = constrainAngle(h); + s /= 100; + l /= 100; + function f(n) { + const k = (n + h / 30) % 12; + const a = s * Math.min(l, 1 - l); + return l - a * Math.max(-1, Math.min(k - 3, 9 - k, 1)); + } + return [f(0), f(8), f(4), alpha]; +} + +/** + * CSS color parser compliant with CSS Color 4 Specification. + * Supports: named colors, `transparent` keyword, all rgb hex notations, + * rgb(), rgba(), hsl() and hsla() functions. + * Does not round the parsed values to integers from the range 0..255. + * + * Syntax: + * + * = | + * = | + * + * rgb() = rgb( {3} [ / ]? ) | rgb( {3} [ / ]? ) + * rgb() = rgb( #{3} , ? ) | rgb( #{3} , ? ) + * + * hsl() = hsl( [ / ]? ) + * hsl() = hsl( , , , ? ) + * + * Caveats: + * - - with optional `deg` suffix; `grad`, `rad`, `turn` are not supported + * - `none` keyword is not supported + * - comments inside rgb()/hsl() are not supported + * - legacy color syntax rgba() is supported with an identical grammar and behavior to rgb() + * - legacy color syntax hsla() is supported with an identical grammar and behavior to hsl() + * + * @param input CSS color string to parse. + * @returns Color in sRGB color space, with `red`, `green`, `blue` + * and `alpha` channels normalized to the range 0..1, + * or `undefined` if the input is not a valid color string. + */ +function parseCssColor(input) { + input = input.toLowerCase().trim(); + if (input === 'transparent') { + return [0, 0, 0, 0]; + } + // 'white', 'black', 'blue' + const namedColorsMatch = namedColors[input]; + if (namedColorsMatch) { + const [r, g, b] = namedColorsMatch; + return [r / 255, g / 255, b / 255, 1]; + } + // #f0c, #f0cf, #ff00cc, #ff00ccff + if (input.startsWith('#')) { + const hexRegexp = /^#(?:[0-9a-f]{3,4}|[0-9a-f]{6}|[0-9a-f]{8})$/; + if (hexRegexp.test(input)) { + const step = input.length < 6 ? 1 : 2; + let i = 1; + return [ + parseHex(input.slice(i, i += step)), + parseHex(input.slice(i, i += step)), + parseHex(input.slice(i, i += step)), + parseHex(input.slice(i, i + step) || 'ff'), + ]; + } + } + // rgb(128 0 0), rgb(50% 0% 0%), rgba(255,0,255,0.6), rgb(255 0 255 / 60%), rgb(100% 0% 100% /.6) + if (input.startsWith('rgb')) { + const rgbRegExp = /^rgba?\(\s*([\de.+-]+)(%)?(?:\s+|\s*(,)\s*)([\de.+-]+)(%)?(?:\s+|\s*(,)\s*)([\de.+-]+)(%)?(?:\s*([,\/])\s*([\de.+-]+)(%)?)?\s*\)$/; + const rgbMatch = input.match(rgbRegExp); + if (rgbMatch) { + const [_, // eslint-disable-line @typescript-eslint/no-unused-vars + r, // + rp, // % (optional) + f1, // , (optional) + g, // + gp, // % (optional) + f2, // , (optional) + b, // + bp, // % (optional) + f3, // ,|/ (optional) + a, // (optional) + ap, // % (optional) + ] = rgbMatch; + const argFormat = [f1 || ' ', f2 || ' ', f3].join(''); + if (argFormat === ' ' || + argFormat === ' /' || + argFormat === ',,' || + argFormat === ',,,') { + const valFormat = [rp, gp, bp].join(''); + const maxValue = (valFormat === '%%%') ? 100 : + (valFormat === '') ? 255 : 0; + if (maxValue) { + const rgba = [ + clamp(+r / maxValue, 0, 1), + clamp(+g / maxValue, 0, 1), + clamp(+b / maxValue, 0, 1), + a ? parseAlpha(+a, ap) : 1, + ]; + if (validateNumbers(rgba)) { + return rgba; + } + // invalid numbers + } + // values must be all numbers or all percentages + } + return; // comma optional syntax requires no commas at all + } + } + // hsl(120 50% 80%), hsla(120deg,50%,80%,.9), hsl(12e1 50% 80% / 90%) + const hslRegExp = /^hsla?\(\s*([\de.+-]+)(?:deg)?(?:\s+|\s*(,)\s*)([\de.+-]+)%(?:\s+|\s*(,)\s*)([\de.+-]+)%(?:\s*([,\/])\s*([\de.+-]+)(%)?)?\s*\)$/; + const hslMatch = input.match(hslRegExp); + if (hslMatch) { + const [_, // eslint-disable-line @typescript-eslint/no-unused-vars + h, // + f1, // , (optional) + s, // + f2, // , (optional) + l, // + f3, // ,|/ (optional) + a, // (optional) + ap, // % (optional) + ] = hslMatch; + const argFormat = [f1 || ' ', f2 || ' ', f3].join(''); + if (argFormat === ' ' || + argFormat === ' /' || + argFormat === ',,' || + argFormat === ',,,') { + const hsla = [ + +h, + clamp(+s, 0, 100), + clamp(+l, 0, 100), + a ? parseAlpha(+a, ap) : 1, + ]; + if (validateNumbers(hsla)) { + return hslToRgb(hsla); + } + // invalid numbers + } + // comma optional syntax requires no commas at all + } +} +function parseHex(hex) { + return parseInt(hex.padEnd(2, hex), 16) / 255; +} +function parseAlpha(a, asPercentage) { + return clamp(asPercentage ? (a / 100) : a, 0, 1); +} +function clamp(n, min, max) { + return Math.min(Math.max(min, n), max); +} +/** + * The regular expression for numeric values is not super specific, and it may + * happen that it will accept a value that is not a valid number. In order to + * detect and eliminate such values this function exists. + * + * @param array Array of uncertain numbers. + * @returns `true` if the specified array contains only valid numbers, `false` otherwise. + */ +function validateNumbers(array) { + return !array.some(Number.isNaN); +} +/** + * To generate: + * - visit {@link https://www.w3.org/TR/css-color-4/#named-colors} + * - run in the console: + * @example + * copy(`{\n${[...document.querySelector('.named-color-table tbody').children].map((tr) => `${tr.cells[2].textContent.trim()}: [${tr.cells[4].textContent.trim().split(/\s+/).join(', ')}],`).join('\n')}\n}`); + */ +const namedColors = { + aliceblue: [240, 248, 255], + antiquewhite: [250, 235, 215], + aqua: [0, 255, 255], + aquamarine: [127, 255, 212], + azure: [240, 255, 255], + beige: [245, 245, 220], + bisque: [255, 228, 196], + black: [0, 0, 0], + blanchedalmond: [255, 235, 205], + blue: [0, 0, 255], + blueviolet: [138, 43, 226], + brown: [165, 42, 42], + burlywood: [222, 184, 135], + cadetblue: [95, 158, 160], + chartreuse: [127, 255, 0], + chocolate: [210, 105, 30], + coral: [255, 127, 80], + cornflowerblue: [100, 149, 237], + cornsilk: [255, 248, 220], + crimson: [220, 20, 60], + cyan: [0, 255, 255], + darkblue: [0, 0, 139], + darkcyan: [0, 139, 139], + darkgoldenrod: [184, 134, 11], + darkgray: [169, 169, 169], + darkgreen: [0, 100, 0], + darkgrey: [169, 169, 169], + darkkhaki: [189, 183, 107], + darkmagenta: [139, 0, 139], + darkolivegreen: [85, 107, 47], + darkorange: [255, 140, 0], + darkorchid: [153, 50, 204], + darkred: [139, 0, 0], + darksalmon: [233, 150, 122], + darkseagreen: [143, 188, 143], + darkslateblue: [72, 61, 139], + darkslategray: [47, 79, 79], + darkslategrey: [47, 79, 79], + darkturquoise: [0, 206, 209], + darkviolet: [148, 0, 211], + deeppink: [255, 20, 147], + deepskyblue: [0, 191, 255], + dimgray: [105, 105, 105], + dimgrey: [105, 105, 105], + dodgerblue: [30, 144, 255], + firebrick: [178, 34, 34], + floralwhite: [255, 250, 240], + forestgreen: [34, 139, 34], + fuchsia: [255, 0, 255], + gainsboro: [220, 220, 220], + ghostwhite: [248, 248, 255], + gold: [255, 215, 0], + goldenrod: [218, 165, 32], + gray: [128, 128, 128], + green: [0, 128, 0], + greenyellow: [173, 255, 47], + grey: [128, 128, 128], + honeydew: [240, 255, 240], + hotpink: [255, 105, 180], + indianred: [205, 92, 92], + indigo: [75, 0, 130], + ivory: [255, 255, 240], + khaki: [240, 230, 140], + lavender: [230, 230, 250], + lavenderblush: [255, 240, 245], + lawngreen: [124, 252, 0], + lemonchiffon: [255, 250, 205], + lightblue: [173, 216, 230], + lightcoral: [240, 128, 128], + lightcyan: [224, 255, 255], + lightgoldenrodyellow: [250, 250, 210], + lightgray: [211, 211, 211], + lightgreen: [144, 238, 144], + lightgrey: [211, 211, 211], + lightpink: [255, 182, 193], + lightsalmon: [255, 160, 122], + lightseagreen: [32, 178, 170], + lightskyblue: [135, 206, 250], + lightslategray: [119, 136, 153], + lightslategrey: [119, 136, 153], + lightsteelblue: [176, 196, 222], + lightyellow: [255, 255, 224], + lime: [0, 255, 0], + limegreen: [50, 205, 50], + linen: [250, 240, 230], + magenta: [255, 0, 255], + maroon: [128, 0, 0], + mediumaquamarine: [102, 205, 170], + mediumblue: [0, 0, 205], + mediumorchid: [186, 85, 211], + mediumpurple: [147, 112, 219], + mediumseagreen: [60, 179, 113], + mediumslateblue: [123, 104, 238], + mediumspringgreen: [0, 250, 154], + mediumturquoise: [72, 209, 204], + mediumvioletred: [199, 21, 133], + midnightblue: [25, 25, 112], + mintcream: [245, 255, 250], + mistyrose: [255, 228, 225], + moccasin: [255, 228, 181], + navajowhite: [255, 222, 173], + navy: [0, 0, 128], + oldlace: [253, 245, 230], + olive: [128, 128, 0], + olivedrab: [107, 142, 35], + orange: [255, 165, 0], + orangered: [255, 69, 0], + orchid: [218, 112, 214], + palegoldenrod: [238, 232, 170], + palegreen: [152, 251, 152], + paleturquoise: [175, 238, 238], + palevioletred: [219, 112, 147], + papayawhip: [255, 239, 213], + peachpuff: [255, 218, 185], + peru: [205, 133, 63], + pink: [255, 192, 203], + plum: [221, 160, 221], + powderblue: [176, 224, 230], + purple: [128, 0, 128], + rebeccapurple: [102, 51, 153], + red: [255, 0, 0], + rosybrown: [188, 143, 143], + royalblue: [65, 105, 225], + saddlebrown: [139, 69, 19], + salmon: [250, 128, 114], + sandybrown: [244, 164, 96], + seagreen: [46, 139, 87], + seashell: [255, 245, 238], + sienna: [160, 82, 45], + silver: [192, 192, 192], + skyblue: [135, 206, 235], + slateblue: [106, 90, 205], + slategray: [112, 128, 144], + slategrey: [112, 128, 144], + snow: [255, 250, 250], + springgreen: [0, 255, 127], + steelblue: [70, 130, 180], + tan: [210, 180, 140], + teal: [0, 128, 128], + thistle: [216, 191, 216], + tomato: [255, 99, 71], + turquoise: [64, 224, 208], + violet: [238, 130, 238], + wheat: [245, 222, 179], + white: [255, 255, 255], + whitesmoke: [245, 245, 245], + yellow: [255, 255, 0], + yellowgreen: [154, 205, 50], +}; + +/** + * Color representation used by WebGL. + * Defined in sRGB color space and pre-blended with alpha. + * @private + */ +class Color { + /** + * @param r Red component premultiplied by `alpha` 0..1 + * @param g Green component premultiplied by `alpha` 0..1 + * @param b Blue component premultiplied by `alpha` 0..1 + * @param [alpha=1] Alpha component 0..1 + * @param [premultiplied=true] Whether the `r`, `g` and `b` values have already + * been multiplied by alpha. If `true` nothing happens if `false` then they will + * be multiplied automatically. + */ + constructor(r, g, b, alpha = 1, premultiplied = true) { + this.r = r; + this.g = g; + this.b = b; + this.a = alpha; + if (!premultiplied) { + this.r *= alpha; + this.g *= alpha; + this.b *= alpha; + if (!alpha) { + // alpha = 0 erases completely rgb channels. This behavior is not desirable + // if this particular color is later used in color interpolation. + // Because of that, a reference to original color is saved. + this.overwriteGetter('rgb', [r, g, b, alpha]); + } + } + } + /** + * Parses CSS color strings and converts colors to sRGB color space if needed. + * Officially supported color formats: + * - keyword, e.g. 'aquamarine' or 'steelblue' + * - hex (with 3, 4, 6 or 8 digits), e.g. '#f0f' or '#e9bebea9' + * - rgb and rgba, e.g. 'rgb(0,240,120)' or 'rgba(0%,94%,47%,0.1)' or 'rgb(0 240 120 / .3)' + * - hsl and hsla, e.g. 'hsl(0,0%,83%)' or 'hsla(0,0%,83%,.5)' or 'hsl(0 0% 83% / 20%)' + * + * @param input CSS color string to parse. + * @returns A `Color` instance, or `undefined` if the input is not a valid color string. + */ + static parse(input) { + // in zoom-and-property function input could be an instance of Color class + if (input instanceof Color) { + return input; + } + if (typeof input !== 'string') { + return; + } + const rgba = parseCssColor(input); + if (rgba) { + return new Color(...rgba, false); + } + } + /** + * Used in color interpolation and by 'to-rgba' expression. + * + * @returns Gien color, with reversed alpha blending, in sRGB color space. + */ + get rgb() { + const { r, g, b, a } = this; + const f = a || Infinity; // reverse alpha blending factor + return this.overwriteGetter('rgb', [r / f, g / f, b / f, a]); + } + /** + * Used in color interpolation. + * + * @returns Gien color, with reversed alpha blending, in HCL color space. + */ + get hcl() { + return this.overwriteGetter('hcl', rgbToHcl(this.rgb)); + } + /** + * Used in color interpolation. + * + * @returns Gien color, with reversed alpha blending, in LAB color space. + */ + get lab() { + return this.overwriteGetter('lab', rgbToLab(this.rgb)); + } + /** + * Lazy getter pattern. When getter is called for the first time lazy value + * is calculated and then overwrites getter function in given object instance. + * + * @example: + * const redColor = Color.parse('red'); + * let x = redColor.hcl; // this will invoke `get hcl()`, which will calculate + * // the value of red in HCL space and invoke this `overwriteGetter` function + * // which in turn will set a field with a key 'hcl' in the `redColor` object. + * // In other words it will override `get hcl()` from its `Color` prototype + * // with its own property: hcl = [calculated red value in hcl]. + * let y = redColor.hcl; // next call will no longer invoke getter but simply + * // return the previously calculated value + * x === y; // true - `x` is exactly the same object as `y` + * + * @param getterKey Getter key + * @param lazyValue Lazily calculated value to be memoized by current instance + * @private + */ + overwriteGetter(getterKey, lazyValue) { + Object.defineProperty(this, getterKey, { value: lazyValue }); + return lazyValue; + } + /** + * Used by 'to-string' expression. + * + * @returns Serialized color in format `rgba(r,g,b,a)` + * where r,g,b are numbers within 0..255 and alpha is number within 1..0 + * + * @example + * var purple = new Color.parse('purple'); + * purple.toString; // = "rgba(128,0,128,1)" + * var translucentGreen = new Color.parse('rgba(26, 207, 26, .73)'); + * translucentGreen.toString(); // = "rgba(26,207,26,0.73)" + */ + toString() { + const [r, g, b, a] = this.rgb; + return `rgba(${[r, g, b].map(n => Math.round(n * 255)).join(',')},${a})`; + } +} +Color.black = new Color(0, 0, 0, 1); +Color.white = new Color(1, 1, 1, 1); +Color.transparent = new Color(0, 0, 0, 0); +Color.red = new Color(1, 0, 0, 1); + +// Flow type declarations for Intl cribbed from +// https://github.com/facebook/flow/issues/1270 +class Collator { + constructor(caseSensitive, diacriticSensitive, locale) { + if (caseSensitive) + this.sensitivity = diacriticSensitive ? 'variant' : 'case'; + else + this.sensitivity = diacriticSensitive ? 'accent' : 'base'; + this.locale = locale; + this.collator = new Intl.Collator(this.locale ? this.locale : [], { sensitivity: this.sensitivity, usage: 'search' }); + } + compare(lhs, rhs) { + return this.collator.compare(lhs, rhs); + } + resolvedLocale() { + // We create a Collator without "usage: search" because we don't want + // the search options encoded in our result (e.g. "en-u-co-search") + return new Intl.Collator(this.locale ? this.locale : []) + .resolvedOptions().locale; + } +} + +class FormattedSection { + constructor(text, image, scale, fontStack, textColor) { + this.text = text; + this.image = image; + this.scale = scale; + this.fontStack = fontStack; + this.textColor = textColor; + } +} +class Formatted { + constructor(sections) { + this.sections = sections; + } + static fromString(unformatted) { + return new Formatted([new FormattedSection(unformatted, null, null, null, null)]); + } + isEmpty() { + if (this.sections.length === 0) + return true; + return !this.sections.some(section => section.text.length !== 0 || + (section.image && section.image.name.length !== 0)); + } + static factory(text) { + if (text instanceof Formatted) { + return text; + } + else { + return Formatted.fromString(text); + } + } + toString() { + if (this.sections.length === 0) + return ''; + return this.sections.map(section => section.text).join(''); + } +} + +/** + * A set of four numbers representing padding around a box. Create instances from + * bare arrays or numeric values using the static method `Padding.parse`. + * @private + */ +class Padding { + constructor(values) { + this.values = values.slice(); + } + /** + * Numeric padding values + * @param input A padding value + * @returns A `Padding` instance, or `undefined` if the input is not a valid padding value. + */ + static parse(input) { + if (input instanceof Padding) { + return input; + } + // Backwards compatibility: bare number is treated the same as array with single value. + // Padding applies to all four sides. + if (typeof input === 'number') { + return new Padding([input, input, input, input]); + } + if (!Array.isArray(input)) { + return undefined; + } + if (input.length < 1 || input.length > 4) { + return undefined; + } + for (const val of input) { + if (typeof val !== 'number') { + return undefined; + } + } + // Expand shortcut properties into explicit 4-sided values + switch (input.length) { + case 1: + input = [input[0], input[0], input[0], input[0]]; + break; + case 2: + input = [input[0], input[1], input[0], input[1]]; + break; + case 3: + input = [input[0], input[1], input[2], input[1]]; + break; + } + return new Padding(input); + } + toString() { + return JSON.stringify(this.values); + } +} + +/** Set of valid anchor positions, as a set for validation */ +const anchors = new Set(['center', 'left', 'right', 'top', 'bottom', 'top-left', 'top-right', 'bottom-left', 'bottom-right']); +/** + * Utility class to assist managing values for text-variable-anchor-offset property. Create instances from + * bare arrays using the static method `VariableAnchorOffsetCollection.parse`. + * @private + */ +class VariableAnchorOffsetCollection { + constructor(values) { + this.values = values.slice(); + } + static parse(input) { + if (input instanceof VariableAnchorOffsetCollection) { + return input; + } + if (!Array.isArray(input) || + input.length < 1 || + input.length % 2 !== 0) { + return undefined; + } + for (let i = 0; i < input.length; i += 2) { + // Elements in even positions should be anchor positions; Elements in odd positions should be offset values + const anchorValue = input[i]; + const offsetValue = input[i + 1]; + if (typeof anchorValue !== 'string' || !anchors.has(anchorValue)) { + return undefined; + } + if (!Array.isArray(offsetValue) || offsetValue.length !== 2 || typeof offsetValue[0] !== 'number' || typeof offsetValue[1] !== 'number') { + return undefined; + } + } + return new VariableAnchorOffsetCollection(input); + } + toString() { + return JSON.stringify(this.values); + } +} + +class ResolvedImage { + constructor(options) { + this.name = options.name; + this.available = options.available; + } + toString() { + return this.name; + } + static fromString(name) { + if (!name) + return null; // treat empty values as no image + return new ResolvedImage({ name, available: false }); + } +} + +function validateRGBA(r, g, b, a) { + if (!(typeof r === 'number' && r >= 0 && r <= 255 && + typeof g === 'number' && g >= 0 && g <= 255 && + typeof b === 'number' && b >= 0 && b <= 255)) { + const value = typeof a === 'number' ? [r, g, b, a] : [r, g, b]; + return `Invalid rgba value [${value.join(', ')}]: 'r', 'g', and 'b' must be between 0 and 255.`; + } + if (!(typeof a === 'undefined' || (typeof a === 'number' && a >= 0 && a <= 1))) { + return `Invalid rgba value [${[r, g, b, a].join(', ')}]: 'a' must be between 0 and 1.`; + } + return null; +} +function isValue(mixed) { + if (mixed === null || + typeof mixed === 'string' || + typeof mixed === 'boolean' || + typeof mixed === 'number' || + mixed instanceof Color || + mixed instanceof Collator || + mixed instanceof Formatted || + mixed instanceof Padding || + mixed instanceof VariableAnchorOffsetCollection || + mixed instanceof ResolvedImage) { + return true; + } + else if (Array.isArray(mixed)) { + for (const item of mixed) { + if (!isValue(item)) { + return false; + } + } + return true; + } + else if (typeof mixed === 'object') { + for (const key in mixed) { + if (!isValue(mixed[key])) { + return false; + } + } + return true; + } + else { + return false; + } +} +function typeOf(value) { + if (value === null) { + return NullType; + } + else if (typeof value === 'string') { + return StringType; + } + else if (typeof value === 'boolean') { + return BooleanType; + } + else if (typeof value === 'number') { + return NumberType; + } + else if (value instanceof Color) { + return ColorType; + } + else if (value instanceof Collator) { + return CollatorType; + } + else if (value instanceof Formatted) { + return FormattedType; + } + else if (value instanceof Padding) { + return PaddingType; + } + else if (value instanceof VariableAnchorOffsetCollection) { + return VariableAnchorOffsetCollectionType; + } + else if (value instanceof ResolvedImage) { + return ResolvedImageType; + } + else if (Array.isArray(value)) { + const length = value.length; + let itemType; + for (const item of value) { + const t = typeOf(item); + if (!itemType) { + itemType = t; + } + else if (itemType === t) { + continue; + } + else { + itemType = ValueType; + break; + } + } + return array$1(itemType || ValueType, length); + } + else { + return ObjectType; + } +} +function toString(value) { + const type = typeof value; + if (value === null) { + return ''; + } + else if (type === 'string' || type === 'number' || type === 'boolean') { + return String(value); + } + else if (value instanceof Color || value instanceof Formatted || value instanceof Padding || value instanceof VariableAnchorOffsetCollection || value instanceof ResolvedImage) { + return value.toString(); + } + else { + return JSON.stringify(value); + } +} + +class Literal { + constructor(type, value) { + this.type = type; + this.value = value; + } + static parse(args, context) { + if (args.length !== 2) + return context.error(`'literal' expression requires exactly one argument, but found ${args.length - 1} instead.`); + if (!isValue(args[1])) + return context.error('invalid value'); + const value = args[1]; + let type = typeOf(value); + // special case: infer the item type if possible for zero-length arrays + const expected = context.expectedType; + if (type.kind === 'array' && + type.N === 0 && + expected && + expected.kind === 'array' && + (typeof expected.N !== 'number' || expected.N === 0)) { + type = expected; + } + return new Literal(type, value); + } + evaluate() { + return this.value; + } + eachChild() { } + outputDefined() { + return true; + } +} + +class RuntimeError { + constructor(message) { + this.name = 'ExpressionEvaluationError'; + this.message = message; + } + toJSON() { + return this.message; + } +} + +const types$1 = { + string: StringType, + number: NumberType, + boolean: BooleanType, + object: ObjectType +}; +class Assertion { + constructor(type, args) { + this.type = type; + this.args = args; + } + static parse(args, context) { + if (args.length < 2) + return context.error('Expected at least one argument.'); + let i = 1; + let type; + const name = args[0]; + if (name === 'array') { + let itemType; + if (args.length > 2) { + const type = args[1]; + if (typeof type !== 'string' || !(type in types$1) || type === 'object') + return context.error('The item type argument of "array" must be one of string, number, boolean', 1); + itemType = types$1[type]; + i++; + } + else { + itemType = ValueType; + } + let N; + if (args.length > 3) { + if (args[2] !== null && + (typeof args[2] !== 'number' || + args[2] < 0 || + args[2] !== Math.floor(args[2]))) { + return context.error('The length argument to "array" must be a positive integer literal', 2); + } + N = args[2]; + i++; + } + type = array$1(itemType, N); + } + else { + if (!types$1[name]) + throw new Error(`Types doesn't contain name = ${name}`); + type = types$1[name]; + } + const parsed = []; + for (; i < args.length; i++) { + const input = context.parse(args[i], i, ValueType); + if (!input) + return null; + parsed.push(input); + } + return new Assertion(type, parsed); + } + evaluate(ctx) { + for (let i = 0; i < this.args.length; i++) { + const value = this.args[i].evaluate(ctx); + const error = checkSubtype(this.type, typeOf(value)); + if (!error) { + return value; + } + else if (i === this.args.length - 1) { + throw new RuntimeError(`Expected value to be of type ${toString$1(this.type)}, but found ${toString$1(typeOf(value))} instead.`); + } + } + throw new Error(); + } + eachChild(fn) { + this.args.forEach(fn); + } + outputDefined() { + return this.args.every(arg => arg.outputDefined()); + } +} + +const types = { + 'to-boolean': BooleanType, + 'to-color': ColorType, + 'to-number': NumberType, + 'to-string': StringType +}; +/** + * Special form for error-coalescing coercion expressions "to-number", + * "to-color". Since these coercions can fail at runtime, they accept multiple + * arguments, only evaluating one at a time until one succeeds. + * + * @private + */ +class Coercion { + constructor(type, args) { + this.type = type; + this.args = args; + } + static parse(args, context) { + if (args.length < 2) + return context.error('Expected at least one argument.'); + const name = args[0]; + if (!types[name]) + throw new Error(`Can't parse ${name} as it is not part of the known types`); + if ((name === 'to-boolean' || name === 'to-string') && args.length !== 2) + return context.error('Expected one argument.'); + const type = types[name]; + const parsed = []; + for (let i = 1; i < args.length; i++) { + const input = context.parse(args[i], i, ValueType); + if (!input) + return null; + parsed.push(input); + } + return new Coercion(type, parsed); + } + evaluate(ctx) { + switch (this.type.kind) { + case 'boolean': + return Boolean(this.args[0].evaluate(ctx)); + case 'color': { + let input; + let error; + for (const arg of this.args) { + input = arg.evaluate(ctx); + error = null; + if (input instanceof Color) { + return input; + } + else if (typeof input === 'string') { + const c = ctx.parseColor(input); + if (c) + return c; + } + else if (Array.isArray(input)) { + if (input.length < 3 || input.length > 4) { + error = `Invalid rbga value ${JSON.stringify(input)}: expected an array containing either three or four numeric values.`; + } + else { + error = validateRGBA(input[0], input[1], input[2], input[3]); + } + if (!error) { + return new Color(input[0] / 255, input[1] / 255, input[2] / 255, input[3]); + } + } + } + throw new RuntimeError(error || `Could not parse color from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`); + } + case 'padding': { + let input; + for (const arg of this.args) { + input = arg.evaluate(ctx); + const pad = Padding.parse(input); + if (pad) { + return pad; + } + } + throw new RuntimeError(`Could not parse padding from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`); + } + case 'variableAnchorOffsetCollection': { + let input; + for (const arg of this.args) { + input = arg.evaluate(ctx); + const coll = VariableAnchorOffsetCollection.parse(input); + if (coll) { + return coll; + } + } + throw new RuntimeError(`Could not parse variableAnchorOffsetCollection from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`); + } + case 'number': { + let value = null; + for (const arg of this.args) { + value = arg.evaluate(ctx); + if (value === null) + return 0; + const num = Number(value); + if (isNaN(num)) + continue; + return num; + } + throw new RuntimeError(`Could not convert ${JSON.stringify(value)} to number.`); + } + case 'formatted': + // There is no explicit 'to-formatted' but this coercion can be implicitly + // created by properties that expect the 'formatted' type. + return Formatted.fromString(toString(this.args[0].evaluate(ctx))); + case 'resolvedImage': + return ResolvedImage.fromString(toString(this.args[0].evaluate(ctx))); + default: + return toString(this.args[0].evaluate(ctx)); + } + } + eachChild(fn) { + this.args.forEach(fn); + } + outputDefined() { + return this.args.every(arg => arg.outputDefined()); + } +} + +const geometryTypes = ['Unknown', 'Point', 'LineString', 'Polygon']; +class EvaluationContext { + constructor() { + this.globals = null; + this.feature = null; + this.featureState = null; + this.formattedSection = null; + this._parseColorCache = {}; + this.availableImages = null; + this.canonical = null; + } + id() { + return this.feature && 'id' in this.feature ? this.feature.id : null; + } + geometryType() { + return this.feature ? typeof this.feature.type === 'number' ? geometryTypes[this.feature.type] : this.feature.type : null; + } + geometry() { + return this.feature && 'geometry' in this.feature ? this.feature.geometry : null; + } + canonicalID() { + return this.canonical; + } + properties() { + return this.feature && this.feature.properties || {}; + } + parseColor(input) { + let cached = this._parseColorCache[input]; + if (!cached) { + cached = this._parseColorCache[input] = Color.parse(input); + } + return cached; + } +} + +/** + * State associated parsing at a given point in an expression tree. + * @private + */ +class ParsingContext { + constructor(registry, isConstantFunc, path = [], expectedType, scope = new Scope(), errors = []) { + this.registry = registry; + this.path = path; + this.key = path.map(part => `[${part}]`).join(''); + this.scope = scope; + this.errors = errors; + this.expectedType = expectedType; + this._isConstant = isConstantFunc; + } + /** + * @param expr the JSON expression to parse + * @param index the optional argument index if this expression is an argument of a parent expression that's being parsed + * @param options + * @param options.omitTypeAnnotations set true to omit inferred type annotations. Caller beware: with this option set, the parsed expression's type will NOT satisfy `expectedType` if it would normally be wrapped in an inferred annotation. + * @private + */ + parse(expr, index, expectedType, bindings, options = {}) { + if (index) { + return this.concat(index, expectedType, bindings)._parse(expr, options); + } + return this._parse(expr, options); + } + _parse(expr, options) { + if (expr === null || typeof expr === 'string' || typeof expr === 'boolean' || typeof expr === 'number') { + expr = ['literal', expr]; + } + function annotate(parsed, type, typeAnnotation) { + if (typeAnnotation === 'assert') { + return new Assertion(type, [parsed]); + } + else if (typeAnnotation === 'coerce') { + return new Coercion(type, [parsed]); + } + else { + return parsed; + } + } + if (Array.isArray(expr)) { + if (expr.length === 0) { + return this.error('Expected an array with at least one element. If you wanted a literal array, use ["literal", []].'); + } + const op = expr[0]; + if (typeof op !== 'string') { + this.error(`Expression name must be a string, but found ${typeof op} instead. If you wanted a literal array, use ["literal", [...]].`, 0); + return null; + } + const Expr = this.registry[op]; + if (Expr) { + let parsed = Expr.parse(expr, this); + if (!parsed) + return null; + if (this.expectedType) { + const expected = this.expectedType; + const actual = parsed.type; + // When we expect a number, string, boolean, or array but have a value, wrap it in an assertion. + // When we expect a color or formatted string, but have a string or value, wrap it in a coercion. + // Otherwise, we do static type-checking. + // + // These behaviors are overridable for: + // * The "coalesce" operator, which needs to omit type annotations. + // * String-valued properties (e.g. `text-field`), where coercion is more convenient than assertion. + // + if ((expected.kind === 'string' || expected.kind === 'number' || expected.kind === 'boolean' || expected.kind === 'object' || expected.kind === 'array') && actual.kind === 'value') { + parsed = annotate(parsed, expected, options.typeAnnotation || 'assert'); + } + else if ((expected.kind === 'color' || expected.kind === 'formatted' || expected.kind === 'resolvedImage') && (actual.kind === 'value' || actual.kind === 'string')) { + parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce'); + } + else if (expected.kind === 'padding' && (actual.kind === 'value' || actual.kind === 'number' || actual.kind === 'array')) { + parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce'); + } + else if (expected.kind === 'variableAnchorOffsetCollection' && (actual.kind === 'value' || actual.kind === 'array')) { + parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce'); + } + else if (this.checkSubtype(expected, actual)) { + return null; + } + } + // If an expression's arguments are all literals, we can evaluate + // it immediately and replace it with a literal value in the + // parsed/compiled result. Expressions that expect an image should + // not be resolved here so we can later get the available images. + if (!(parsed instanceof Literal) && (parsed.type.kind !== 'resolvedImage') && this._isConstant(parsed)) { + const ec = new EvaluationContext(); + try { + parsed = new Literal(parsed.type, parsed.evaluate(ec)); + } + catch (e) { + this.error(e.message); + return null; + } + } + return parsed; + } + return this.error(`Unknown expression "${op}". If you wanted a literal array, use ["literal", [...]].`, 0); + } + else if (typeof expr === 'undefined') { + return this.error('\'undefined\' value invalid. Use null instead.'); + } + else if (typeof expr === 'object') { + return this.error('Bare objects invalid. Use ["literal", {...}] instead.'); + } + else { + return this.error(`Expected an array, but found ${typeof expr} instead.`); + } + } + /** + * Returns a copy of this context suitable for parsing the subexpression at + * index `index`, optionally appending to 'let' binding map. + * + * Note that `errors` property, intended for collecting errors while + * parsing, is copied by reference rather than cloned. + * @private + */ + concat(index, expectedType, bindings) { + const path = typeof index === 'number' ? this.path.concat(index) : this.path; + const scope = bindings ? this.scope.concat(bindings) : this.scope; + return new ParsingContext(this.registry, this._isConstant, path, expectedType || null, scope, this.errors); + } + /** + * Push a parsing (or type checking) error into the `this.errors` + * @param error The message + * @param keys Optionally specify the source of the error at a child + * of the current expression at `this.key`. + * @private + */ + error(error, ...keys) { + const key = `${this.key}${keys.map(k => `[${k}]`).join('')}`; + this.errors.push(new ExpressionParsingError(key, error)); + } + /** + * Returns null if `t` is a subtype of `expected`; otherwise returns an + * error message and also pushes it to `this.errors`. + * @param expected The expected type + * @param t The actual type + * @returns null if `t` is a subtype of `expected`; otherwise returns an error message + */ + checkSubtype(expected, t) { + const error = checkSubtype(expected, t); + if (error) + this.error(error); + return error; + } +} + +class CollatorExpression { + constructor(caseSensitive, diacriticSensitive, locale) { + this.type = CollatorType; + this.locale = locale; + this.caseSensitive = caseSensitive; + this.diacriticSensitive = diacriticSensitive; + } + static parse(args, context) { + if (args.length !== 2) + return context.error('Expected one argument.'); + const options = args[1]; + if (typeof options !== 'object' || Array.isArray(options)) + return context.error('Collator options argument must be an object.'); + const caseSensitive = context.parse(options['case-sensitive'] === undefined ? false : options['case-sensitive'], 1, BooleanType); + if (!caseSensitive) + return null; + const diacriticSensitive = context.parse(options['diacritic-sensitive'] === undefined ? false : options['diacritic-sensitive'], 1, BooleanType); + if (!diacriticSensitive) + return null; + let locale = null; + if (options['locale']) { + locale = context.parse(options['locale'], 1, StringType); + if (!locale) + return null; + } + return new CollatorExpression(caseSensitive, diacriticSensitive, locale); + } + evaluate(ctx) { + return new Collator(this.caseSensitive.evaluate(ctx), this.diacriticSensitive.evaluate(ctx), this.locale ? this.locale.evaluate(ctx) : null); + } + eachChild(fn) { + fn(this.caseSensitive); + fn(this.diacriticSensitive); + if (this.locale) { + fn(this.locale); + } + } + outputDefined() { + // Technically the set of possible outputs is the combinatoric set of Collators produced + // by all possible outputs of locale/caseSensitive/diacriticSensitive + // But for the primary use of Collators in comparison operators, we ignore the Collator's + // possible outputs anyway, so we can get away with leaving this false for now. + return false; + } +} + +const EXTENT$1 = 8192; +function updateBBox(bbox, coord) { + bbox[0] = Math.min(bbox[0], coord[0]); + bbox[1] = Math.min(bbox[1], coord[1]); + bbox[2] = Math.max(bbox[2], coord[0]); + bbox[3] = Math.max(bbox[3], coord[1]); +} +function mercatorXfromLng$1(lng) { + return (180 + lng) / 360; +} +function mercatorYfromLat$1(lat) { + return (180 - (180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360)))) / 360; +} +function boxWithinBox(bbox1, bbox2) { + if (bbox1[0] <= bbox2[0]) + return false; + if (bbox1[2] >= bbox2[2]) + return false; + if (bbox1[1] <= bbox2[1]) + return false; + if (bbox1[3] >= bbox2[3]) + return false; + return true; +} +function getTileCoordinates(p, canonical) { + const x = mercatorXfromLng$1(p[0]); + const y = mercatorYfromLat$1(p[1]); + const tilesAtZoom = Math.pow(2, canonical.z); + return [Math.round(x * tilesAtZoom * EXTENT$1), Math.round(y * tilesAtZoom * EXTENT$1)]; +} +function onBoundary(p, p1, p2) { + const x1 = p[0] - p1[0]; + const y1 = p[1] - p1[1]; + const x2 = p[0] - p2[0]; + const y2 = p[1] - p2[1]; + return (x1 * y2 - x2 * y1 === 0) && (x1 * x2 <= 0) && (y1 * y2 <= 0); +} +function rayIntersect(p, p1, p2) { + return ((p1[1] > p[1]) !== (p2[1] > p[1])) && (p[0] < (p2[0] - p1[0]) * (p[1] - p1[1]) / (p2[1] - p1[1]) + p1[0]); +} +// ray casting algorithm for detecting if point is in polygon +function pointWithinPolygon(point, rings) { + let inside = false; + for (let i = 0, len = rings.length; i < len; i++) { + const ring = rings[i]; + for (let j = 0, len2 = ring.length; j < len2 - 1; j++) { + if (onBoundary(point, ring[j], ring[j + 1])) + return false; + if (rayIntersect(point, ring[j], ring[j + 1])) + inside = !inside; + } + } + return inside; +} +function pointWithinPolygons(point, polygons) { + for (let i = 0; i < polygons.length; i++) { + if (pointWithinPolygon(point, polygons[i])) + return true; + } + return false; +} +function perp(v1, v2) { + return (v1[0] * v2[1] - v1[1] * v2[0]); +} +// check if p1 and p2 are in different sides of line segment q1->q2 +function twoSided(p1, p2, q1, q2) { + // q1->p1 (x1, y1), q1->p2 (x2, y2), q1->q2 (x3, y3) + const x1 = p1[0] - q1[0]; + const y1 = p1[1] - q1[1]; + const x2 = p2[0] - q1[0]; + const y2 = p2[1] - q1[1]; + const x3 = q2[0] - q1[0]; + const y3 = q2[1] - q1[1]; + const det1 = (x1 * y3 - x3 * y1); + const det2 = (x2 * y3 - x3 * y2); + if ((det1 > 0 && det2 < 0) || (det1 < 0 && det2 > 0)) + return true; + return false; +} +// a, b are end points for line segment1, c and d are end points for line segment2 +function lineIntersectLine(a, b, c, d) { + // check if two segments are parallel or not + // precondition is end point a, b is inside polygon, if line a->b is + // parallel to polygon edge c->d, then a->b won't intersect with c->d + const vectorP = [b[0] - a[0], b[1] - a[1]]; + const vectorQ = [d[0] - c[0], d[1] - c[1]]; + if (perp(vectorQ, vectorP) === 0) + return false; + // If lines are intersecting with each other, the relative location should be: + // a and b lie in different sides of segment c->d + // c and d lie in different sides of segment a->b + if (twoSided(a, b, c, d) && twoSided(c, d, a, b)) + return true; + return false; +} +function lineIntersectPolygon(p1, p2, polygon) { + for (const ring of polygon) { + // loop through every edge of the ring + for (let j = 0; j < ring.length - 1; ++j) { + if (lineIntersectLine(p1, p2, ring[j], ring[j + 1])) { + return true; + } + } + } + return false; +} +function lineStringWithinPolygon(line, polygon) { + // First, check if geometry points of line segments are all inside polygon + for (let i = 0; i < line.length; ++i) { + if (!pointWithinPolygon(line[i], polygon)) { + return false; + } + } + // Second, check if there is line segment intersecting polygon edge + for (let i = 0; i < line.length - 1; ++i) { + if (lineIntersectPolygon(line[i], line[i + 1], polygon)) { + return false; + } + } + return true; +} +function lineStringWithinPolygons(line, polygons) { + for (let i = 0; i < polygons.length; i++) { + if (lineStringWithinPolygon(line, polygons[i])) + return true; + } + return false; +} +function getTilePolygon(coordinates, bbox, canonical) { + const polygon = []; + for (let i = 0; i < coordinates.length; i++) { + const ring = []; + for (let j = 0; j < coordinates[i].length; j++) { + const coord = getTileCoordinates(coordinates[i][j], canonical); + updateBBox(bbox, coord); + ring.push(coord); + } + polygon.push(ring); + } + return polygon; +} +function getTilePolygons(coordinates, bbox, canonical) { + const polygons = []; + for (let i = 0; i < coordinates.length; i++) { + const polygon = getTilePolygon(coordinates[i], bbox, canonical); + polygons.push(polygon); + } + return polygons; +} +function updatePoint(p, bbox, polyBBox, worldSize) { + if (p[0] < polyBBox[0] || p[0] > polyBBox[2]) { + const halfWorldSize = worldSize * 0.5; + let shift = (p[0] - polyBBox[0] > halfWorldSize) ? -worldSize : (polyBBox[0] - p[0] > halfWorldSize) ? worldSize : 0; + if (shift === 0) { + shift = (p[0] - polyBBox[2] > halfWorldSize) ? -worldSize : (polyBBox[2] - p[0] > halfWorldSize) ? worldSize : 0; + } + p[0] += shift; + } + updateBBox(bbox, p); +} +function resetBBox(bbox) { + bbox[0] = bbox[1] = Infinity; + bbox[2] = bbox[3] = -Infinity; +} +function getTilePoints(geometry, pointBBox, polyBBox, canonical) { + const worldSize = Math.pow(2, canonical.z) * EXTENT$1; + const shifts = [canonical.x * EXTENT$1, canonical.y * EXTENT$1]; + const tilePoints = []; + for (const points of geometry) { + for (const point of points) { + const p = [point.x + shifts[0], point.y + shifts[1]]; + updatePoint(p, pointBBox, polyBBox, worldSize); + tilePoints.push(p); + } + } + return tilePoints; +} +function getTileLines(geometry, lineBBox, polyBBox, canonical) { + const worldSize = Math.pow(2, canonical.z) * EXTENT$1; + const shifts = [canonical.x * EXTENT$1, canonical.y * EXTENT$1]; + const tileLines = []; + for (const line of geometry) { + const tileLine = []; + for (const point of line) { + const p = [point.x + shifts[0], point.y + shifts[1]]; + updateBBox(lineBBox, p); + tileLine.push(p); + } + tileLines.push(tileLine); + } + if (lineBBox[2] - lineBBox[0] <= worldSize / 2) { + resetBBox(lineBBox); + for (const line of tileLines) { + for (const p of line) { + updatePoint(p, lineBBox, polyBBox, worldSize); + } + } + } + return tileLines; +} +function pointsWithinPolygons(ctx, polygonGeometry) { + const pointBBox = [Infinity, Infinity, -Infinity, -Infinity]; + const polyBBox = [Infinity, Infinity, -Infinity, -Infinity]; + const canonical = ctx.canonicalID(); + if (polygonGeometry.type === 'Polygon') { + const tilePolygon = getTilePolygon(polygonGeometry.coordinates, polyBBox, canonical); + const tilePoints = getTilePoints(ctx.geometry(), pointBBox, polyBBox, canonical); + if (!boxWithinBox(pointBBox, polyBBox)) + return false; + for (const point of tilePoints) { + if (!pointWithinPolygon(point, tilePolygon)) + return false; + } + } + if (polygonGeometry.type === 'MultiPolygon') { + const tilePolygons = getTilePolygons(polygonGeometry.coordinates, polyBBox, canonical); + const tilePoints = getTilePoints(ctx.geometry(), pointBBox, polyBBox, canonical); + if (!boxWithinBox(pointBBox, polyBBox)) + return false; + for (const point of tilePoints) { + if (!pointWithinPolygons(point, tilePolygons)) + return false; + } + } + return true; +} +function linesWithinPolygons(ctx, polygonGeometry) { + const lineBBox = [Infinity, Infinity, -Infinity, -Infinity]; + const polyBBox = [Infinity, Infinity, -Infinity, -Infinity]; + const canonical = ctx.canonicalID(); + if (polygonGeometry.type === 'Polygon') { + const tilePolygon = getTilePolygon(polygonGeometry.coordinates, polyBBox, canonical); + const tileLines = getTileLines(ctx.geometry(), lineBBox, polyBBox, canonical); + if (!boxWithinBox(lineBBox, polyBBox)) + return false; + for (const line of tileLines) { + if (!lineStringWithinPolygon(line, tilePolygon)) + return false; + } + } + if (polygonGeometry.type === 'MultiPolygon') { + const tilePolygons = getTilePolygons(polygonGeometry.coordinates, polyBBox, canonical); + const tileLines = getTileLines(ctx.geometry(), lineBBox, polyBBox, canonical); + if (!boxWithinBox(lineBBox, polyBBox)) + return false; + for (const line of tileLines) { + if (!lineStringWithinPolygons(line, tilePolygons)) + return false; + } + } + return true; +} +class Within { + constructor(geojson, geometries) { + this.type = BooleanType; + this.geojson = geojson; + this.geometries = geometries; + } + static parse(args, context) { + if (args.length !== 2) + return context.error(`'within' expression requires exactly one argument, but found ${args.length - 1} instead.`); + if (isValue(args[1])) { + const geojson = args[1]; + if (geojson.type === 'FeatureCollection') { + for (let i = 0; i < geojson.features.length; ++i) { + const type = geojson.features[i].geometry.type; + if (type === 'Polygon' || type === 'MultiPolygon') { + return new Within(geojson, geojson.features[i].geometry); + } + } + } + else if (geojson.type === 'Feature') { + const type = geojson.geometry.type; + if (type === 'Polygon' || type === 'MultiPolygon') { + return new Within(geojson, geojson.geometry); + } + } + else if (geojson.type === 'Polygon' || geojson.type === 'MultiPolygon') { + return new Within(geojson, geojson); + } + } + return context.error('\'within\' expression requires valid geojson object that contains polygon geometry type.'); + } + evaluate(ctx) { + if (ctx.geometry() != null && ctx.canonicalID() != null) { + if (ctx.geometryType() === 'Point') { + return pointsWithinPolygons(ctx, this.geometries); + } + else if (ctx.geometryType() === 'LineString') { + return linesWithinPolygons(ctx, this.geometries); + } + } + return false; + } + eachChild() { } + outputDefined() { + return true; + } +} + +class Var { + constructor(name, boundExpression) { + this.type = boundExpression.type; + this.name = name; + this.boundExpression = boundExpression; + } + static parse(args, context) { + if (args.length !== 2 || typeof args[1] !== 'string') + return context.error('\'var\' expression requires exactly one string literal argument.'); + const name = args[1]; + if (!context.scope.has(name)) { + return context.error(`Unknown variable "${name}". Make sure "${name}" has been bound in an enclosing "let" expression before using it.`, 1); + } + return new Var(name, context.scope.get(name)); + } + evaluate(ctx) { + return this.boundExpression.evaluate(ctx); + } + eachChild() { } + outputDefined() { + return false; + } +} + +class CompoundExpression { + constructor(name, type, evaluate, args) { + this.name = name; + this.type = type; + this._evaluate = evaluate; + this.args = args; + } + evaluate(ctx) { + return this._evaluate(ctx, this.args); + } + eachChild(fn) { + this.args.forEach(fn); + } + outputDefined() { + return false; + } + static parse(args, context) { + const op = args[0]; + const definition = CompoundExpression.definitions[op]; + if (!definition) { + return context.error(`Unknown expression "${op}". If you wanted a literal array, use ["literal", [...]].`, 0); + } + // Now check argument types against each signature + const type = Array.isArray(definition) ? + definition[0] : definition.type; + const availableOverloads = Array.isArray(definition) ? + [[definition[1], definition[2]]] : + definition.overloads; + const overloads = availableOverloads.filter(([signature]) => (!Array.isArray(signature) || // varags + signature.length === args.length - 1 // correct param count + )); + let signatureContext = null; + for (const [params, evaluate] of overloads) { + // Use a fresh context for each attempted signature so that, if + // we eventually succeed, we haven't polluted `context.errors`. + signatureContext = new ParsingContext(context.registry, isExpressionConstant, context.path, null, context.scope); + // First parse all the args, potentially coercing to the + // types expected by this overload. + const parsedArgs = []; + let argParseFailed = false; + for (let i = 1; i < args.length; i++) { + const arg = args[i]; + const expectedType = Array.isArray(params) ? + params[i - 1] : + params.type; + const parsed = signatureContext.parse(arg, 1 + parsedArgs.length, expectedType); + if (!parsed) { + argParseFailed = true; + break; + } + parsedArgs.push(parsed); + } + if (argParseFailed) { + // Couldn't coerce args of this overload to expected type, move + // on to next one. + continue; + } + if (Array.isArray(params)) { + if (params.length !== parsedArgs.length) { + signatureContext.error(`Expected ${params.length} arguments, but found ${parsedArgs.length} instead.`); + continue; + } + } + for (let i = 0; i < parsedArgs.length; i++) { + const expected = Array.isArray(params) ? params[i] : params.type; + const arg = parsedArgs[i]; + signatureContext.concat(i + 1).checkSubtype(expected, arg.type); + } + if (signatureContext.errors.length === 0) { + return new CompoundExpression(op, type, evaluate, parsedArgs); + } + } + if (overloads.length === 1) { + context.errors.push(...signatureContext.errors); + } + else { + const expected = overloads.length ? overloads : availableOverloads; + const signatures = expected + .map(([params]) => stringifySignature(params)) + .join(' | '); + const actualTypes = []; + // For error message, re-parse arguments without trying to + // apply any coercions + for (let i = 1; i < args.length; i++) { + const parsed = context.parse(args[i], 1 + actualTypes.length); + if (!parsed) + return null; + actualTypes.push(toString$1(parsed.type)); + } + context.error(`Expected arguments of type ${signatures}, but found (${actualTypes.join(', ')}) instead.`); + } + return null; + } + static register(registry, definitions) { + CompoundExpression.definitions = definitions; + for (const name in definitions) { + registry[name] = CompoundExpression; + } + } +} +function stringifySignature(signature) { + if (Array.isArray(signature)) { + return `(${signature.map(toString$1).join(', ')})`; + } + else { + return `(${toString$1(signature.type)}...)`; + } +} +function isExpressionConstant(expression) { + if (expression instanceof Var) { + return isExpressionConstant(expression.boundExpression); + } + else if (expression instanceof CompoundExpression && expression.name === 'error') { + return false; + } + else if (expression instanceof CollatorExpression) { + // Although the results of a Collator expression with fixed arguments + // generally shouldn't change between executions, we can't serialize them + // as constant expressions because results change based on environment. + return false; + } + else if (expression instanceof Within) { + return false; + } + const isTypeAnnotation = expression instanceof Coercion || + expression instanceof Assertion; + let childrenConstant = true; + expression.eachChild(child => { + // We can _almost_ assume that if `expressions` children are constant, + // they would already have been evaluated to Literal values when they + // were parsed. Type annotations are the exception, because they might + // have been inferred and added after a child was parsed. + // So we recurse into isConstant() for the children of type annotations, + // but otherwise simply check whether they are Literals. + if (isTypeAnnotation) { + childrenConstant = childrenConstant && isExpressionConstant(child); + } + else { + childrenConstant = childrenConstant && child instanceof Literal; + } + }); + if (!childrenConstant) { + return false; + } + return isFeatureConstant(expression) && + isGlobalPropertyConstant(expression, ['zoom', 'heatmap-density', 'line-progress', 'accumulated', 'is-supported-script']); +} +function isFeatureConstant(e) { + if (e instanceof CompoundExpression) { + if (e.name === 'get' && e.args.length === 1) { + return false; + } + else if (e.name === 'feature-state') { + return false; + } + else if (e.name === 'has' && e.args.length === 1) { + return false; + } + else if (e.name === 'properties' || + e.name === 'geometry-type' || + e.name === 'id') { + return false; + } + else if (/^filter-/.test(e.name)) { + return false; + } + } + if (e instanceof Within) { + return false; + } + let result = true; + e.eachChild(arg => { + if (result && !isFeatureConstant(arg)) { + result = false; + } + }); + return result; +} +function isStateConstant(e) { + if (e instanceof CompoundExpression) { + if (e.name === 'feature-state') { + return false; + } + } + let result = true; + e.eachChild(arg => { + if (result && !isStateConstant(arg)) { + result = false; + } + }); + return result; +} +function isGlobalPropertyConstant(e, properties) { + if (e instanceof CompoundExpression && properties.indexOf(e.name) >= 0) { + return false; + } + let result = true; + e.eachChild((arg) => { + if (result && !isGlobalPropertyConstant(arg, properties)) { + result = false; + } + }); + return result; +} + +/** + * Returns the index of the last stop <= input, or 0 if it doesn't exist. + * @private + */ +function findStopLessThanOrEqualTo(stops, input) { + const lastIndex = stops.length - 1; + let lowerIndex = 0; + let upperIndex = lastIndex; + let currentIndex = 0; + let currentValue, nextValue; + while (lowerIndex <= upperIndex) { + currentIndex = Math.floor((lowerIndex + upperIndex) / 2); + currentValue = stops[currentIndex]; + nextValue = stops[currentIndex + 1]; + if (currentValue <= input) { + if (currentIndex === lastIndex || input < nextValue) { // Search complete + return currentIndex; + } + lowerIndex = currentIndex + 1; + } + else if (currentValue > input) { + upperIndex = currentIndex - 1; + } + else { + throw new RuntimeError('Input is not a number.'); + } + } + return 0; +} + +class Step { + constructor(type, input, stops) { + this.type = type; + this.input = input; + this.labels = []; + this.outputs = []; + for (const [label, expression] of stops) { + this.labels.push(label); + this.outputs.push(expression); + } + } + static parse(args, context) { + if (args.length - 1 < 4) { + return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`); + } + if ((args.length - 1) % 2 !== 0) { + return context.error('Expected an even number of arguments.'); + } + const input = context.parse(args[1], 1, NumberType); + if (!input) + return null; + const stops = []; + let outputType = null; + if (context.expectedType && context.expectedType.kind !== 'value') { + outputType = context.expectedType; + } + for (let i = 1; i < args.length; i += 2) { + const label = i === 1 ? -Infinity : args[i]; + const value = args[i + 1]; + const labelKey = i; + const valueKey = i + 1; + if (typeof label !== 'number') { + return context.error('Input/output pairs for "step" expressions must be defined using literal numeric values (not computed expressions) for the input values.', labelKey); + } + if (stops.length && stops[stops.length - 1][0] >= label) { + return context.error('Input/output pairs for "step" expressions must be arranged with input values in strictly ascending order.', labelKey); + } + const parsed = context.parse(value, valueKey, outputType); + if (!parsed) + return null; + outputType = outputType || parsed.type; + stops.push([label, parsed]); + } + return new Step(outputType, input, stops); + } + evaluate(ctx) { + const labels = this.labels; + const outputs = this.outputs; + if (labels.length === 1) { + return outputs[0].evaluate(ctx); + } + const value = this.input.evaluate(ctx); + if (value <= labels[0]) { + return outputs[0].evaluate(ctx); + } + const stopCount = labels.length; + if (value >= labels[stopCount - 1]) { + return outputs[stopCount - 1].evaluate(ctx); + } + const index = findStopLessThanOrEqualTo(labels, value); + return outputs[index].evaluate(ctx); + } + eachChild(fn) { + fn(this.input); + for (const expression of this.outputs) { + fn(expression); + } + } + outputDefined() { + return this.outputs.every(out => out.outputDefined()); + } +} + +/** + * Checks whether the specified color space is one of the supported interpolation color spaces. + * + * @param colorSpace Color space key to verify. + * @returns `true` if the specified color space is one of the supported + * interpolation color spaces, `false` otherwise + */ +function isSupportedInterpolationColorSpace(colorSpace) { + return colorSpace === 'rgb' || colorSpace === 'hcl' || colorSpace === 'lab'; +} +/** + * @param interpolationType Interpolation type + * @returns interpolation fn + * @deprecated use `interpolate[type]` instead + */ +const interpolateFactory = (interpolationType) => { + switch (interpolationType) { + case 'number': return number; + case 'color': return color; + case 'array': return array; + case 'padding': return padding; + case 'variableAnchorOffsetCollection': return variableAnchorOffsetCollection; + } +}; +function number(from, to, t) { + return from + t * (to - from); +} +function color(from, to, t, spaceKey = 'rgb') { + switch (spaceKey) { + case 'rgb': { + const [r, g, b, alpha] = array(from.rgb, to.rgb, t); + return new Color(r, g, b, alpha, false); + } + case 'hcl': { + const [hue0, chroma0, light0, alphaF] = from.hcl; + const [hue1, chroma1, light1, alphaT] = to.hcl; + // https://github.com/gka/chroma.js/blob/cd1b3c0926c7a85cbdc3b1453b3a94006de91a92/src/interpolator/_hsx.js + let hue, chroma; + if (!isNaN(hue0) && !isNaN(hue1)) { + let dh = hue1 - hue0; + if (hue1 > hue0 && dh > 180) { + dh -= 360; + } + else if (hue1 < hue0 && hue0 - hue1 > 180) { + dh += 360; + } + hue = hue0 + t * dh; + } + else if (!isNaN(hue0)) { + hue = hue0; + if (light1 === 1 || light1 === 0) + chroma = chroma0; + } + else if (!isNaN(hue1)) { + hue = hue1; + if (light0 === 1 || light0 === 0) + chroma = chroma1; + } + else { + hue = NaN; + } + const [r, g, b, alpha] = hclToRgb([ + hue, + chroma !== null && chroma !== void 0 ? chroma : number(chroma0, chroma1, t), + number(light0, light1, t), + number(alphaF, alphaT, t), + ]); + return new Color(r, g, b, alpha, false); + } + case 'lab': { + const [r, g, b, alpha] = labToRgb(array(from.lab, to.lab, t)); + return new Color(r, g, b, alpha, false); + } + } +} +function array(from, to, t) { + return from.map((d, i) => { + return number(d, to[i], t); + }); +} +function padding(from, to, t) { + return new Padding(array(from.values, to.values, t)); +} +function variableAnchorOffsetCollection(from, to, t) { + const fromValues = from.values; + const toValues = to.values; + if (fromValues.length !== toValues.length) { + throw new RuntimeError(`Cannot interpolate values of different length. from: ${from.toString()}, to: ${to.toString()}`); + } + const output = []; + for (let i = 0; i < fromValues.length; i += 2) { + // Anchor entries must match + if (fromValues[i] !== toValues[i]) { + throw new RuntimeError(`Cannot interpolate values containing mismatched anchors. from[${i}]: ${fromValues[i]}, to[${i}]: ${toValues[i]}`); + } + output.push(fromValues[i]); + // Interpolate the offset values for each anchor + const [fx, fy] = fromValues[i + 1]; + const [tx, ty] = toValues[i + 1]; + output.push([number(fx, tx, t), number(fy, ty, t)]); + } + return new VariableAnchorOffsetCollection(output); +} +const interpolate = { + number, + color, + array, + padding, + variableAnchorOffsetCollection +}; + +class Interpolate { + constructor(type, operator, interpolation, input, stops) { + this.type = type; + this.operator = operator; + this.interpolation = interpolation; + this.input = input; + this.labels = []; + this.outputs = []; + for (const [label, expression] of stops) { + this.labels.push(label); + this.outputs.push(expression); + } + } + static interpolationFactor(interpolation, input, lower, upper) { + let t = 0; + if (interpolation.name === 'exponential') { + t = exponentialInterpolation(input, interpolation.base, lower, upper); + } + else if (interpolation.name === 'linear') { + t = exponentialInterpolation(input, 1, lower, upper); + } + else if (interpolation.name === 'cubic-bezier') { + const c = interpolation.controlPoints; + const ub = new UnitBezier$1(c[0], c[1], c[2], c[3]); + t = ub.solve(exponentialInterpolation(input, 1, lower, upper)); + } + return t; + } + static parse(args, context) { + let [operator, interpolation, input, ...rest] = args; + if (!Array.isArray(interpolation) || interpolation.length === 0) { + return context.error('Expected an interpolation type expression.', 1); + } + if (interpolation[0] === 'linear') { + interpolation = { name: 'linear' }; + } + else if (interpolation[0] === 'exponential') { + const base = interpolation[1]; + if (typeof base !== 'number') + return context.error('Exponential interpolation requires a numeric base.', 1, 1); + interpolation = { + name: 'exponential', + base + }; + } + else if (interpolation[0] === 'cubic-bezier') { + const controlPoints = interpolation.slice(1); + if (controlPoints.length !== 4 || + controlPoints.some(t => typeof t !== 'number' || t < 0 || t > 1)) { + return context.error('Cubic bezier interpolation requires four numeric arguments with values between 0 and 1.', 1); + } + interpolation = { + name: 'cubic-bezier', + controlPoints: controlPoints + }; + } + else { + return context.error(`Unknown interpolation type ${String(interpolation[0])}`, 1, 0); + } + if (args.length - 1 < 4) { + return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`); + } + if ((args.length - 1) % 2 !== 0) { + return context.error('Expected an even number of arguments.'); + } + input = context.parse(input, 2, NumberType); + if (!input) + return null; + const stops = []; + let outputType = null; + if (operator === 'interpolate-hcl' || operator === 'interpolate-lab') { + outputType = ColorType; + } + else if (context.expectedType && context.expectedType.kind !== 'value') { + outputType = context.expectedType; + } + for (let i = 0; i < rest.length; i += 2) { + const label = rest[i]; + const value = rest[i + 1]; + const labelKey = i + 3; + const valueKey = i + 4; + if (typeof label !== 'number') { + return context.error('Input/output pairs for "interpolate" expressions must be defined using literal numeric values (not computed expressions) for the input values.', labelKey); + } + if (stops.length && stops[stops.length - 1][0] >= label) { + return context.error('Input/output pairs for "interpolate" expressions must be arranged with input values in strictly ascending order.', labelKey); + } + const parsed = context.parse(value, valueKey, outputType); + if (!parsed) + return null; + outputType = outputType || parsed.type; + stops.push([label, parsed]); + } + if (!verifyType(outputType, NumberType) && + !verifyType(outputType, ColorType) && + !verifyType(outputType, PaddingType) && + !verifyType(outputType, VariableAnchorOffsetCollectionType) && + !verifyType(outputType, array$1(NumberType))) { + return context.error(`Type ${toString$1(outputType)} is not interpolatable.`); + } + return new Interpolate(outputType, operator, interpolation, input, stops); + } + evaluate(ctx) { + const labels = this.labels; + const outputs = this.outputs; + if (labels.length === 1) { + return outputs[0].evaluate(ctx); + } + const value = this.input.evaluate(ctx); + if (value <= labels[0]) { + return outputs[0].evaluate(ctx); + } + const stopCount = labels.length; + if (value >= labels[stopCount - 1]) { + return outputs[stopCount - 1].evaluate(ctx); + } + const index = findStopLessThanOrEqualTo(labels, value); + const lower = labels[index]; + const upper = labels[index + 1]; + const t = Interpolate.interpolationFactor(this.interpolation, value, lower, upper); + const outputLower = outputs[index].evaluate(ctx); + const outputUpper = outputs[index + 1].evaluate(ctx); + switch (this.operator) { + case 'interpolate': + return interpolate[this.type.kind](outputLower, outputUpper, t); + case 'interpolate-hcl': + return interpolate.color(outputLower, outputUpper, t, 'hcl'); + case 'interpolate-lab': + return interpolate.color(outputLower, outputUpper, t, 'lab'); + } + } + eachChild(fn) { + fn(this.input); + for (const expression of this.outputs) { + fn(expression); + } + } + outputDefined() { + return this.outputs.every(out => out.outputDefined()); + } +} +/** + * Returns a ratio that can be used to interpolate between exponential function + * stops. + * How it works: Two consecutive stop values define a (scaled and shifted) exponential function `f(x) = a * base^x + b`, where `base` is the user-specified base, + * and `a` and `b` are constants affording sufficient degrees of freedom to fit + * the function to the given stops. + * + * Here's a bit of algebra that lets us compute `f(x)` directly from the stop + * values without explicitly solving for `a` and `b`: + * + * First stop value: `f(x0) = y0 = a * base^x0 + b` + * Second stop value: `f(x1) = y1 = a * base^x1 + b` + * => `y1 - y0 = a(base^x1 - base^x0)` + * => `a = (y1 - y0)/(base^x1 - base^x0)` + * + * Desired value: `f(x) = y = a * base^x + b` + * => `f(x) = y0 + a * (base^x - base^x0)` + * + * From the above, we can replace the `a` in `a * (base^x - base^x0)` and do a + * little algebra: + * ``` + * a * (base^x - base^x0) = (y1 - y0)/(base^x1 - base^x0) * (base^x - base^x0) + * = (y1 - y0) * (base^x - base^x0) / (base^x1 - base^x0) + * ``` + * + * If we let `(base^x - base^x0) / (base^x1 base^x0)`, then we have + * `f(x) = y0 + (y1 - y0) * ratio`. In other words, `ratio` may be treated as + * an interpolation factor between the two stops' output values. + * + * (Note: a slightly different form for `ratio`, + * `(base^(x-x0) - 1) / (base^(x1-x0) - 1) `, is equivalent, but requires fewer + * expensive `Math.pow()` operations.) + * + * @private +*/ +function exponentialInterpolation(input, base, lowerValue, upperValue) { + const difference = upperValue - lowerValue; + const progress = input - lowerValue; + if (difference === 0) { + return 0; + } + else if (base === 1) { + return progress / difference; + } + else { + return (Math.pow(base, progress) - 1) / (Math.pow(base, difference) - 1); + } +} + +class Coalesce { + constructor(type, args) { + this.type = type; + this.args = args; + } + static parse(args, context) { + if (args.length < 2) { + return context.error('Expectected at least one argument.'); + } + let outputType = null; + const expectedType = context.expectedType; + if (expectedType && expectedType.kind !== 'value') { + outputType = expectedType; + } + const parsedArgs = []; + for (const arg of args.slice(1)) { + const parsed = context.parse(arg, 1 + parsedArgs.length, outputType, undefined, { typeAnnotation: 'omit' }); + if (!parsed) + return null; + outputType = outputType || parsed.type; + parsedArgs.push(parsed); + } + if (!outputType) + throw new Error('No output type'); + // Above, we parse arguments without inferred type annotation so that + // they don't produce a runtime error for `null` input, which would + // preempt the desired null-coalescing behavior. + // Thus, if any of our arguments would have needed an annotation, we + // need to wrap the enclosing coalesce expression with it instead. + const needsAnnotation = expectedType && + parsedArgs.some(arg => checkSubtype(expectedType, arg.type)); + return needsAnnotation ? + new Coalesce(ValueType, parsedArgs) : + new Coalesce(outputType, parsedArgs); + } + evaluate(ctx) { + let result = null; + let argCount = 0; + let requestedImageName; + for (const arg of this.args) { + argCount++; + result = arg.evaluate(ctx); + // we need to keep track of the first requested image in a coalesce statement + // if coalesce can't find a valid image, we return the first image name so styleimagemissing can fire + if (result && result instanceof ResolvedImage && !result.available) { + if (!requestedImageName) { + requestedImageName = result.name; + } + result = null; + if (argCount === this.args.length) { + result = requestedImageName; + } + } + if (result !== null) + break; + } + return result; + } + eachChild(fn) { + this.args.forEach(fn); + } + outputDefined() { + return this.args.every(arg => arg.outputDefined()); + } +} + +class Let { + constructor(bindings, result) { + this.type = result.type; + this.bindings = [].concat(bindings); + this.result = result; + } + evaluate(ctx) { + return this.result.evaluate(ctx); + } + eachChild(fn) { + for (const binding of this.bindings) { + fn(binding[1]); + } + fn(this.result); + } + static parse(args, context) { + if (args.length < 4) + return context.error(`Expected at least 3 arguments, but found ${args.length - 1} instead.`); + const bindings = []; + for (let i = 1; i < args.length - 1; i += 2) { + const name = args[i]; + if (typeof name !== 'string') { + return context.error(`Expected string, but found ${typeof name} instead.`, i); + } + if (/[^a-zA-Z0-9_]/.test(name)) { + return context.error('Variable names must contain only alphanumeric characters or \'_\'.', i); + } + const value = context.parse(args[i + 1], i + 1); + if (!value) + return null; + bindings.push([name, value]); + } + const result = context.parse(args[args.length - 1], args.length - 1, context.expectedType, bindings); + if (!result) + return null; + return new Let(bindings, result); + } + outputDefined() { + return this.result.outputDefined(); + } +} + +class At { + constructor(type, index, input) { + this.type = type; + this.index = index; + this.input = input; + } + static parse(args, context) { + if (args.length !== 3) + return context.error(`Expected 2 arguments, but found ${args.length - 1} instead.`); + const index = context.parse(args[1], 1, NumberType); + const input = context.parse(args[2], 2, array$1(context.expectedType || ValueType)); + if (!index || !input) + return null; + const t = input.type; + return new At(t.itemType, index, input); + } + evaluate(ctx) { + const index = this.index.evaluate(ctx); + const array = this.input.evaluate(ctx); + if (index < 0) { + throw new RuntimeError(`Array index out of bounds: ${index} < 0.`); + } + if (index >= array.length) { + throw new RuntimeError(`Array index out of bounds: ${index} > ${array.length - 1}.`); + } + if (index !== Math.floor(index)) { + throw new RuntimeError(`Array index must be an integer, but found ${index} instead.`); + } + return array[index]; + } + eachChild(fn) { + fn(this.index); + fn(this.input); + } + outputDefined() { + return false; + } +} + +class In { + constructor(needle, haystack) { + this.type = BooleanType; + this.needle = needle; + this.haystack = haystack; + } + static parse(args, context) { + if (args.length !== 3) { + return context.error(`Expected 2 arguments, but found ${args.length - 1} instead.`); + } + const needle = context.parse(args[1], 1, ValueType); + const haystack = context.parse(args[2], 2, ValueType); + if (!needle || !haystack) + return null; + if (!isValidType(needle.type, [BooleanType, StringType, NumberType, NullType, ValueType])) { + return context.error(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(needle.type)} instead`); + } + return new In(needle, haystack); + } + evaluate(ctx) { + const needle = this.needle.evaluate(ctx); + const haystack = this.haystack.evaluate(ctx); + if (!haystack) + return false; + if (!isValidNativeType(needle, ['boolean', 'string', 'number', 'null'])) { + throw new RuntimeError(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(typeOf(needle))} instead.`); + } + if (!isValidNativeType(haystack, ['string', 'array'])) { + throw new RuntimeError(`Expected second argument to be of type array or string, but found ${toString$1(typeOf(haystack))} instead.`); + } + return haystack.indexOf(needle) >= 0; + } + eachChild(fn) { + fn(this.needle); + fn(this.haystack); + } + outputDefined() { + return true; + } +} + +class IndexOf { + constructor(needle, haystack, fromIndex) { + this.type = NumberType; + this.needle = needle; + this.haystack = haystack; + this.fromIndex = fromIndex; + } + static parse(args, context) { + if (args.length <= 2 || args.length >= 5) { + return context.error(`Expected 3 or 4 arguments, but found ${args.length - 1} instead.`); + } + const needle = context.parse(args[1], 1, ValueType); + const haystack = context.parse(args[2], 2, ValueType); + if (!needle || !haystack) + return null; + if (!isValidType(needle.type, [BooleanType, StringType, NumberType, NullType, ValueType])) { + return context.error(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(needle.type)} instead`); + } + if (args.length === 4) { + const fromIndex = context.parse(args[3], 3, NumberType); + if (!fromIndex) + return null; + return new IndexOf(needle, haystack, fromIndex); + } + else { + return new IndexOf(needle, haystack); + } + } + evaluate(ctx) { + const needle = this.needle.evaluate(ctx); + const haystack = this.haystack.evaluate(ctx); + if (!isValidNativeType(needle, ['boolean', 'string', 'number', 'null'])) { + throw new RuntimeError(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(typeOf(needle))} instead.`); + } + if (!isValidNativeType(haystack, ['string', 'array'])) { + throw new RuntimeError(`Expected second argument to be of type array or string, but found ${toString$1(typeOf(haystack))} instead.`); + } + if (this.fromIndex) { + const fromIndex = this.fromIndex.evaluate(ctx); + return haystack.indexOf(needle, fromIndex); + } + return haystack.indexOf(needle); + } + eachChild(fn) { + fn(this.needle); + fn(this.haystack); + if (this.fromIndex) { + fn(this.fromIndex); + } + } + outputDefined() { + return false; + } +} + +class Match { + constructor(inputType, outputType, input, cases, outputs, otherwise) { + this.inputType = inputType; + this.type = outputType; + this.input = input; + this.cases = cases; + this.outputs = outputs; + this.otherwise = otherwise; + } + static parse(args, context) { + if (args.length < 5) + return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`); + if (args.length % 2 !== 1) + return context.error('Expected an even number of arguments.'); + let inputType; + let outputType; + if (context.expectedType && context.expectedType.kind !== 'value') { + outputType = context.expectedType; + } + const cases = {}; + const outputs = []; + for (let i = 2; i < args.length - 1; i += 2) { + let labels = args[i]; + const value = args[i + 1]; + if (!Array.isArray(labels)) { + labels = [labels]; + } + const labelContext = context.concat(i); + if (labels.length === 0) { + return labelContext.error('Expected at least one branch label.'); + } + for (const label of labels) { + if (typeof label !== 'number' && typeof label !== 'string') { + return labelContext.error('Branch labels must be numbers or strings.'); + } + else if (typeof label === 'number' && Math.abs(label) > Number.MAX_SAFE_INTEGER) { + return labelContext.error(`Branch labels must be integers no larger than ${Number.MAX_SAFE_INTEGER}.`); + } + else if (typeof label === 'number' && Math.floor(label) !== label) { + return labelContext.error('Numeric branch labels must be integer values.'); + } + else if (!inputType) { + inputType = typeOf(label); + } + else if (labelContext.checkSubtype(inputType, typeOf(label))) { + return null; + } + if (typeof cases[String(label)] !== 'undefined') { + return labelContext.error('Branch labels must be unique.'); + } + cases[String(label)] = outputs.length; + } + const result = context.parse(value, i, outputType); + if (!result) + return null; + outputType = outputType || result.type; + outputs.push(result); + } + const input = context.parse(args[1], 1, ValueType); + if (!input) + return null; + const otherwise = context.parse(args[args.length - 1], args.length - 1, outputType); + if (!otherwise) + return null; + if (input.type.kind !== 'value' && context.concat(1).checkSubtype(inputType, input.type)) { + return null; + } + return new Match(inputType, outputType, input, cases, outputs, otherwise); + } + evaluate(ctx) { + const input = this.input.evaluate(ctx); + const output = (typeOf(input) === this.inputType && this.outputs[this.cases[input]]) || this.otherwise; + return output.evaluate(ctx); + } + eachChild(fn) { + fn(this.input); + this.outputs.forEach(fn); + fn(this.otherwise); + } + outputDefined() { + return this.outputs.every(out => out.outputDefined()) && this.otherwise.outputDefined(); + } +} + +class Case { + constructor(type, branches, otherwise) { + this.type = type; + this.branches = branches; + this.otherwise = otherwise; + } + static parse(args, context) { + if (args.length < 4) + return context.error(`Expected at least 3 arguments, but found only ${args.length - 1}.`); + if (args.length % 2 !== 0) + return context.error('Expected an odd number of arguments.'); + let outputType; + if (context.expectedType && context.expectedType.kind !== 'value') { + outputType = context.expectedType; + } + const branches = []; + for (let i = 1; i < args.length - 1; i += 2) { + const test = context.parse(args[i], i, BooleanType); + if (!test) + return null; + const result = context.parse(args[i + 1], i + 1, outputType); + if (!result) + return null; + branches.push([test, result]); + outputType = outputType || result.type; + } + const otherwise = context.parse(args[args.length - 1], args.length - 1, outputType); + if (!otherwise) + return null; + if (!outputType) + throw new Error('Can\'t infer output type'); + return new Case(outputType, branches, otherwise); + } + evaluate(ctx) { + for (const [test, expression] of this.branches) { + if (test.evaluate(ctx)) { + return expression.evaluate(ctx); + } + } + return this.otherwise.evaluate(ctx); + } + eachChild(fn) { + for (const [test, expression] of this.branches) { + fn(test); + fn(expression); + } + fn(this.otherwise); + } + outputDefined() { + return this.branches.every(([_, out]) => out.outputDefined()) && this.otherwise.outputDefined(); + } +} + +class Slice { + constructor(type, input, beginIndex, endIndex) { + this.type = type; + this.input = input; + this.beginIndex = beginIndex; + this.endIndex = endIndex; + } + static parse(args, context) { + if (args.length <= 2 || args.length >= 5) { + return context.error(`Expected 3 or 4 arguments, but found ${args.length - 1} instead.`); + } + const input = context.parse(args[1], 1, ValueType); + const beginIndex = context.parse(args[2], 2, NumberType); + if (!input || !beginIndex) + return null; + if (!isValidType(input.type, [array$1(ValueType), StringType, ValueType])) { + return context.error(`Expected first argument to be of type array or string, but found ${toString$1(input.type)} instead`); + } + if (args.length === 4) { + const endIndex = context.parse(args[3], 3, NumberType); + if (!endIndex) + return null; + return new Slice(input.type, input, beginIndex, endIndex); + } + else { + return new Slice(input.type, input, beginIndex); + } + } + evaluate(ctx) { + const input = this.input.evaluate(ctx); + const beginIndex = this.beginIndex.evaluate(ctx); + if (!isValidNativeType(input, ['string', 'array'])) { + throw new RuntimeError(`Expected first argument to be of type array or string, but found ${toString$1(typeOf(input))} instead.`); + } + if (this.endIndex) { + const endIndex = this.endIndex.evaluate(ctx); + return input.slice(beginIndex, endIndex); + } + return input.slice(beginIndex); + } + eachChild(fn) { + fn(this.input); + fn(this.beginIndex); + if (this.endIndex) { + fn(this.endIndex); + } + } + outputDefined() { + return false; + } +} + +function isComparableType(op, type) { + if (op === '==' || op === '!=') { + // equality operator + return type.kind === 'boolean' || + type.kind === 'string' || + type.kind === 'number' || + type.kind === 'null' || + type.kind === 'value'; + } + else { + // ordering operator + return type.kind === 'string' || + type.kind === 'number' || + type.kind === 'value'; + } +} +function eq(ctx, a, b) { return a === b; } +function neq(ctx, a, b) { return a !== b; } +function lt(ctx, a, b) { return a < b; } +function gt(ctx, a, b) { return a > b; } +function lteq(ctx, a, b) { return a <= b; } +function gteq(ctx, a, b) { return a >= b; } +function eqCollate(ctx, a, b, c) { return c.compare(a, b) === 0; } +function neqCollate(ctx, a, b, c) { return !eqCollate(ctx, a, b, c); } +function ltCollate(ctx, a, b, c) { return c.compare(a, b) < 0; } +function gtCollate(ctx, a, b, c) { return c.compare(a, b) > 0; } +function lteqCollate(ctx, a, b, c) { return c.compare(a, b) <= 0; } +function gteqCollate(ctx, a, b, c) { return c.compare(a, b) >= 0; } +/** + * Special form for comparison operators, implementing the signatures: + * - (T, T, ?Collator) => boolean + * - (T, value, ?Collator) => boolean + * - (value, T, ?Collator) => boolean + * + * For inequalities, T must be either value, string, or number. For ==/!=, it + * can also be boolean or null. + * + * Equality semantics are equivalent to Javascript's strict equality (===/!==) + * -- i.e., when the arguments' types don't match, == evaluates to false, != to + * true. + * + * When types don't match in an ordering comparison, a runtime error is thrown. + * + * @private + */ +function makeComparison(op, compareBasic, compareWithCollator) { + const isOrderComparison = op !== '==' && op !== '!='; + return class Comparison { + constructor(lhs, rhs, collator) { + this.type = BooleanType; + this.lhs = lhs; + this.rhs = rhs; + this.collator = collator; + this.hasUntypedArgument = lhs.type.kind === 'value' || rhs.type.kind === 'value'; + } + static parse(args, context) { + if (args.length !== 3 && args.length !== 4) + return context.error('Expected two or three arguments.'); + const op = args[0]; + let lhs = context.parse(args[1], 1, ValueType); + if (!lhs) + return null; + if (!isComparableType(op, lhs.type)) { + return context.concat(1).error(`"${op}" comparisons are not supported for type '${toString$1(lhs.type)}'.`); + } + let rhs = context.parse(args[2], 2, ValueType); + if (!rhs) + return null; + if (!isComparableType(op, rhs.type)) { + return context.concat(2).error(`"${op}" comparisons are not supported for type '${toString$1(rhs.type)}'.`); + } + if (lhs.type.kind !== rhs.type.kind && + lhs.type.kind !== 'value' && + rhs.type.kind !== 'value') { + return context.error(`Cannot compare types '${toString$1(lhs.type)}' and '${toString$1(rhs.type)}'.`); + } + if (isOrderComparison) { + // typing rules specific to less/greater than operators + if (lhs.type.kind === 'value' && rhs.type.kind !== 'value') { + // (value, T) + lhs = new Assertion(rhs.type, [lhs]); + } + else if (lhs.type.kind !== 'value' && rhs.type.kind === 'value') { + // (T, value) + rhs = new Assertion(lhs.type, [rhs]); + } + } + let collator = null; + if (args.length === 4) { + if (lhs.type.kind !== 'string' && + rhs.type.kind !== 'string' && + lhs.type.kind !== 'value' && + rhs.type.kind !== 'value') { + return context.error('Cannot use collator to compare non-string types.'); + } + collator = context.parse(args[3], 3, CollatorType); + if (!collator) + return null; + } + return new Comparison(lhs, rhs, collator); + } + evaluate(ctx) { + const lhs = this.lhs.evaluate(ctx); + const rhs = this.rhs.evaluate(ctx); + if (isOrderComparison && this.hasUntypedArgument) { + const lt = typeOf(lhs); + const rt = typeOf(rhs); + // check that type is string or number, and equal + if (lt.kind !== rt.kind || !(lt.kind === 'string' || lt.kind === 'number')) { + throw new RuntimeError(`Expected arguments for "${op}" to be (string, string) or (number, number), but found (${lt.kind}, ${rt.kind}) instead.`); + } + } + if (this.collator && !isOrderComparison && this.hasUntypedArgument) { + const lt = typeOf(lhs); + const rt = typeOf(rhs); + if (lt.kind !== 'string' || rt.kind !== 'string') { + return compareBasic(ctx, lhs, rhs); + } + } + return this.collator ? + compareWithCollator(ctx, lhs, rhs, this.collator.evaluate(ctx)) : + compareBasic(ctx, lhs, rhs); + } + eachChild(fn) { + fn(this.lhs); + fn(this.rhs); + if (this.collator) { + fn(this.collator); + } + } + outputDefined() { + return true; + } + }; +} +const Equals = makeComparison('==', eq, eqCollate); +const NotEquals = makeComparison('!=', neq, neqCollate); +const LessThan = makeComparison('<', lt, ltCollate); +const GreaterThan = makeComparison('>', gt, gtCollate); +const LessThanOrEqual = makeComparison('<=', lteq, lteqCollate); +const GreaterThanOrEqual = makeComparison('>=', gteq, gteqCollate); + +class NumberFormat { + constructor(number, locale, currency, minFractionDigits, maxFractionDigits) { + this.type = StringType; + this.number = number; + this.locale = locale; + this.currency = currency; + this.minFractionDigits = minFractionDigits; + this.maxFractionDigits = maxFractionDigits; + } + static parse(args, context) { + if (args.length !== 3) + return context.error('Expected two arguments.'); + const number = context.parse(args[1], 1, NumberType); + if (!number) + return null; + const options = args[2]; + if (typeof options !== 'object' || Array.isArray(options)) + return context.error('NumberFormat options argument must be an object.'); + let locale = null; + if (options['locale']) { + locale = context.parse(options['locale'], 1, StringType); + if (!locale) + return null; + } + let currency = null; + if (options['currency']) { + currency = context.parse(options['currency'], 1, StringType); + if (!currency) + return null; + } + let minFractionDigits = null; + if (options['min-fraction-digits']) { + minFractionDigits = context.parse(options['min-fraction-digits'], 1, NumberType); + if (!minFractionDigits) + return null; + } + let maxFractionDigits = null; + if (options['max-fraction-digits']) { + maxFractionDigits = context.parse(options['max-fraction-digits'], 1, NumberType); + if (!maxFractionDigits) + return null; + } + return new NumberFormat(number, locale, currency, minFractionDigits, maxFractionDigits); + } + evaluate(ctx) { + return new Intl.NumberFormat(this.locale ? this.locale.evaluate(ctx) : [], { + style: this.currency ? 'currency' : 'decimal', + currency: this.currency ? this.currency.evaluate(ctx) : undefined, + minimumFractionDigits: this.minFractionDigits ? this.minFractionDigits.evaluate(ctx) : undefined, + maximumFractionDigits: this.maxFractionDigits ? this.maxFractionDigits.evaluate(ctx) : undefined, + }).format(this.number.evaluate(ctx)); + } + eachChild(fn) { + fn(this.number); + if (this.locale) { + fn(this.locale); + } + if (this.currency) { + fn(this.currency); + } + if (this.minFractionDigits) { + fn(this.minFractionDigits); + } + if (this.maxFractionDigits) { + fn(this.maxFractionDigits); + } + } + outputDefined() { + return false; + } +} + +class FormatExpression { + constructor(sections) { + this.type = FormattedType; + this.sections = sections; + } + static parse(args, context) { + if (args.length < 2) { + return context.error('Expected at least one argument.'); + } + const firstArg = args[1]; + if (!Array.isArray(firstArg) && typeof firstArg === 'object') { + return context.error('First argument must be an image or text section.'); + } + const sections = []; + let nextTokenMayBeObject = false; + for (let i = 1; i <= args.length - 1; ++i) { + const arg = args[i]; + if (nextTokenMayBeObject && typeof arg === 'object' && !Array.isArray(arg)) { + nextTokenMayBeObject = false; + let scale = null; + if (arg['font-scale']) { + scale = context.parse(arg['font-scale'], 1, NumberType); + if (!scale) + return null; + } + let font = null; + if (arg['text-font']) { + font = context.parse(arg['text-font'], 1, array$1(StringType)); + if (!font) + return null; + } + let textColor = null; + if (arg['text-color']) { + textColor = context.parse(arg['text-color'], 1, ColorType); + if (!textColor) + return null; + } + const lastExpression = sections[sections.length - 1]; + lastExpression.scale = scale; + lastExpression.font = font; + lastExpression.textColor = textColor; + } + else { + const content = context.parse(args[i], 1, ValueType); + if (!content) + return null; + const kind = content.type.kind; + if (kind !== 'string' && kind !== 'value' && kind !== 'null' && kind !== 'resolvedImage') + return context.error('Formatted text type must be \'string\', \'value\', \'image\' or \'null\'.'); + nextTokenMayBeObject = true; + sections.push({ content, scale: null, font: null, textColor: null }); + } + } + return new FormatExpression(sections); + } + evaluate(ctx) { + const evaluateSection = section => { + const evaluatedContent = section.content.evaluate(ctx); + if (typeOf(evaluatedContent) === ResolvedImageType) { + return new FormattedSection('', evaluatedContent, null, null, null); + } + return new FormattedSection(toString(evaluatedContent), null, section.scale ? section.scale.evaluate(ctx) : null, section.font ? section.font.evaluate(ctx).join(',') : null, section.textColor ? section.textColor.evaluate(ctx) : null); + }; + return new Formatted(this.sections.map(evaluateSection)); + } + eachChild(fn) { + for (const section of this.sections) { + fn(section.content); + if (section.scale) { + fn(section.scale); + } + if (section.font) { + fn(section.font); + } + if (section.textColor) { + fn(section.textColor); + } + } + } + outputDefined() { + // Technically the combinatoric set of all children + // Usually, this.text will be undefined anyway + return false; + } +} + +class ImageExpression { + constructor(input) { + this.type = ResolvedImageType; + this.input = input; + } + static parse(args, context) { + if (args.length !== 2) { + return context.error('Expected two arguments.'); + } + const name = context.parse(args[1], 1, StringType); + if (!name) + return context.error('No image name provided.'); + return new ImageExpression(name); + } + evaluate(ctx) { + const evaluatedImageName = this.input.evaluate(ctx); + const value = ResolvedImage.fromString(evaluatedImageName); + if (value && ctx.availableImages) + value.available = ctx.availableImages.indexOf(evaluatedImageName) > -1; + return value; + } + eachChild(fn) { + fn(this.input); + } + outputDefined() { + // The output of image is determined by the list of available images in the evaluation context + return false; + } +} + +class Length { + constructor(input) { + this.type = NumberType; + this.input = input; + } + static parse(args, context) { + if (args.length !== 2) + return context.error(`Expected 1 argument, but found ${args.length - 1} instead.`); + const input = context.parse(args[1], 1); + if (!input) + return null; + if (input.type.kind !== 'array' && input.type.kind !== 'string' && input.type.kind !== 'value') + return context.error(`Expected argument of type string or array, but found ${toString$1(input.type)} instead.`); + return new Length(input); + } + evaluate(ctx) { + const input = this.input.evaluate(ctx); + if (typeof input === 'string') { + return input.length; + } + else if (Array.isArray(input)) { + return input.length; + } + else { + throw new RuntimeError(`Expected value to be of type string or array, but found ${toString$1(typeOf(input))} instead.`); + } + } + eachChild(fn) { + fn(this.input); + } + outputDefined() { + return false; + } +} + +const expressions = { + // special forms + '==': Equals, + '!=': NotEquals, + '>': GreaterThan, + '<': LessThan, + '>=': GreaterThanOrEqual, + '<=': LessThanOrEqual, + 'array': Assertion, + 'at': At, + 'boolean': Assertion, + 'case': Case, + 'coalesce': Coalesce, + 'collator': CollatorExpression, + 'format': FormatExpression, + 'image': ImageExpression, + 'in': In, + 'index-of': IndexOf, + 'interpolate': Interpolate, + 'interpolate-hcl': Interpolate, + 'interpolate-lab': Interpolate, + 'length': Length, + 'let': Let, + 'literal': Literal, + 'match': Match, + 'number': Assertion, + 'number-format': NumberFormat, + 'object': Assertion, + 'slice': Slice, + 'step': Step, + 'string': Assertion, + 'to-boolean': Coercion, + 'to-color': Coercion, + 'to-number': Coercion, + 'to-string': Coercion, + 'var': Var, + 'within': Within +}; +function rgba(ctx, [r, g, b, a]) { + r = r.evaluate(ctx); + g = g.evaluate(ctx); + b = b.evaluate(ctx); + const alpha = a ? a.evaluate(ctx) : 1; + const error = validateRGBA(r, g, b, alpha); + if (error) + throw new RuntimeError(error); + return new Color(r / 255, g / 255, b / 255, alpha, false); +} +function has(key, obj) { + return key in obj; +} +function get(key, obj) { + const v = obj[key]; + return typeof v === 'undefined' ? null : v; +} +function binarySearch(v, a, i, j) { + while (i <= j) { + const m = (i + j) >> 1; + if (a[m] === v) + return true; + if (a[m] > v) + j = m - 1; + else + i = m + 1; + } + return false; +} +function varargs(type) { + return { type }; +} +CompoundExpression.register(expressions, { + 'error': [ + ErrorType, + [StringType], + (ctx, [v]) => { throw new RuntimeError(v.evaluate(ctx)); } + ], + 'typeof': [ + StringType, + [ValueType], + (ctx, [v]) => toString$1(typeOf(v.evaluate(ctx))) + ], + 'to-rgba': [ + array$1(NumberType, 4), + [ColorType], + (ctx, [v]) => { + const [r, g, b, a] = v.evaluate(ctx).rgb; + return [r * 255, g * 255, b * 255, a]; + }, + ], + 'rgb': [ + ColorType, + [NumberType, NumberType, NumberType], + rgba + ], + 'rgba': [ + ColorType, + [NumberType, NumberType, NumberType, NumberType], + rgba + ], + 'has': { + type: BooleanType, + overloads: [ + [ + [StringType], + (ctx, [key]) => has(key.evaluate(ctx), ctx.properties()) + ], [ + [StringType, ObjectType], + (ctx, [key, obj]) => has(key.evaluate(ctx), obj.evaluate(ctx)) + ] + ] + }, + 'get': { + type: ValueType, + overloads: [ + [ + [StringType], + (ctx, [key]) => get(key.evaluate(ctx), ctx.properties()) + ], [ + [StringType, ObjectType], + (ctx, [key, obj]) => get(key.evaluate(ctx), obj.evaluate(ctx)) + ] + ] + }, + 'feature-state': [ + ValueType, + [StringType], + (ctx, [key]) => get(key.evaluate(ctx), ctx.featureState || {}) + ], + 'properties': [ + ObjectType, + [], + (ctx) => ctx.properties() + ], + 'geometry-type': [ + StringType, + [], + (ctx) => ctx.geometryType() + ], + 'id': [ + ValueType, + [], + (ctx) => ctx.id() + ], + 'zoom': [ + NumberType, + [], + (ctx) => ctx.globals.zoom + ], + 'heatmap-density': [ + NumberType, + [], + (ctx) => ctx.globals.heatmapDensity || 0 + ], + 'line-progress': [ + NumberType, + [], + (ctx) => ctx.globals.lineProgress || 0 + ], + 'accumulated': [ + ValueType, + [], + (ctx) => ctx.globals.accumulated === undefined ? null : ctx.globals.accumulated + ], + '+': [ + NumberType, + varargs(NumberType), + (ctx, args) => { + let result = 0; + for (const arg of args) { + result += arg.evaluate(ctx); + } + return result; + } + ], + '*': [ + NumberType, + varargs(NumberType), + (ctx, args) => { + let result = 1; + for (const arg of args) { + result *= arg.evaluate(ctx); + } + return result; + } + ], + '-': { + type: NumberType, + overloads: [ + [ + [NumberType, NumberType], + (ctx, [a, b]) => a.evaluate(ctx) - b.evaluate(ctx) + ], [ + [NumberType], + (ctx, [a]) => -a.evaluate(ctx) + ] + ] + }, + '/': [ + NumberType, + [NumberType, NumberType], + (ctx, [a, b]) => a.evaluate(ctx) / b.evaluate(ctx) + ], + '%': [ + NumberType, + [NumberType, NumberType], + (ctx, [a, b]) => a.evaluate(ctx) % b.evaluate(ctx) + ], + 'ln2': [ + NumberType, + [], + () => Math.LN2 + ], + 'pi': [ + NumberType, + [], + () => Math.PI + ], + 'e': [ + NumberType, + [], + () => Math.E + ], + '^': [ + NumberType, + [NumberType, NumberType], + (ctx, [b, e]) => Math.pow(b.evaluate(ctx), e.evaluate(ctx)) + ], + 'sqrt': [ + NumberType, + [NumberType], + (ctx, [x]) => Math.sqrt(x.evaluate(ctx)) + ], + 'log10': [ + NumberType, + [NumberType], + (ctx, [n]) => Math.log(n.evaluate(ctx)) / Math.LN10 + ], + 'ln': [ + NumberType, + [NumberType], + (ctx, [n]) => Math.log(n.evaluate(ctx)) + ], + 'log2': [ + NumberType, + [NumberType], + (ctx, [n]) => Math.log(n.evaluate(ctx)) / Math.LN2 + ], + 'sin': [ + NumberType, + [NumberType], + (ctx, [n]) => Math.sin(n.evaluate(ctx)) + ], + 'cos': [ + NumberType, + [NumberType], + (ctx, [n]) => Math.cos(n.evaluate(ctx)) + ], + 'tan': [ + NumberType, + [NumberType], + (ctx, [n]) => Math.tan(n.evaluate(ctx)) + ], + 'asin': [ + NumberType, + [NumberType], + (ctx, [n]) => Math.asin(n.evaluate(ctx)) + ], + 'acos': [ + NumberType, + [NumberType], + (ctx, [n]) => Math.acos(n.evaluate(ctx)) + ], + 'atan': [ + NumberType, + [NumberType], + (ctx, [n]) => Math.atan(n.evaluate(ctx)) + ], + 'min': [ + NumberType, + varargs(NumberType), + (ctx, args) => Math.min(...args.map(arg => arg.evaluate(ctx))) + ], + 'max': [ + NumberType, + varargs(NumberType), + (ctx, args) => Math.max(...args.map(arg => arg.evaluate(ctx))) + ], + 'abs': [ + NumberType, + [NumberType], + (ctx, [n]) => Math.abs(n.evaluate(ctx)) + ], + 'round': [ + NumberType, + [NumberType], + (ctx, [n]) => { + const v = n.evaluate(ctx); + // Javascript's Math.round() rounds towards +Infinity for halfway + // values, even when they're negative. It's more common to round + // away from 0 (e.g., this is what python and C++ do) + return v < 0 ? -Math.round(-v) : Math.round(v); + } + ], + 'floor': [ + NumberType, + [NumberType], + (ctx, [n]) => Math.floor(n.evaluate(ctx)) + ], + 'ceil': [ + NumberType, + [NumberType], + (ctx, [n]) => Math.ceil(n.evaluate(ctx)) + ], + 'filter-==': [ + BooleanType, + [StringType, ValueType], + (ctx, [k, v]) => ctx.properties()[k.value] === v.value + ], + 'filter-id-==': [ + BooleanType, + [ValueType], + (ctx, [v]) => ctx.id() === v.value + ], + 'filter-type-==': [ + BooleanType, + [StringType], + (ctx, [v]) => ctx.geometryType() === v.value + ], + 'filter-<': [ + BooleanType, + [StringType, ValueType], + (ctx, [k, v]) => { + const a = ctx.properties()[k.value]; + const b = v.value; + return typeof a === typeof b && a < b; + } + ], + 'filter-id-<': [ + BooleanType, + [ValueType], + (ctx, [v]) => { + const a = ctx.id(); + const b = v.value; + return typeof a === typeof b && a < b; + } + ], + 'filter->': [ + BooleanType, + [StringType, ValueType], + (ctx, [k, v]) => { + const a = ctx.properties()[k.value]; + const b = v.value; + return typeof a === typeof b && a > b; + } + ], + 'filter-id->': [ + BooleanType, + [ValueType], + (ctx, [v]) => { + const a = ctx.id(); + const b = v.value; + return typeof a === typeof b && a > b; + } + ], + 'filter-<=': [ + BooleanType, + [StringType, ValueType], + (ctx, [k, v]) => { + const a = ctx.properties()[k.value]; + const b = v.value; + return typeof a === typeof b && a <= b; + } + ], + 'filter-id-<=': [ + BooleanType, + [ValueType], + (ctx, [v]) => { + const a = ctx.id(); + const b = v.value; + return typeof a === typeof b && a <= b; + } + ], + 'filter->=': [ + BooleanType, + [StringType, ValueType], + (ctx, [k, v]) => { + const a = ctx.properties()[k.value]; + const b = v.value; + return typeof a === typeof b && a >= b; + } + ], + 'filter-id->=': [ + BooleanType, + [ValueType], + (ctx, [v]) => { + const a = ctx.id(); + const b = v.value; + return typeof a === typeof b && a >= b; + } + ], + 'filter-has': [ + BooleanType, + [ValueType], + (ctx, [k]) => k.value in ctx.properties() + ], + 'filter-has-id': [ + BooleanType, + [], + (ctx) => (ctx.id() !== null && ctx.id() !== undefined) + ], + 'filter-type-in': [ + BooleanType, + [array$1(StringType)], + (ctx, [v]) => v.value.indexOf(ctx.geometryType()) >= 0 + ], + 'filter-id-in': [ + BooleanType, + [array$1(ValueType)], + (ctx, [v]) => v.value.indexOf(ctx.id()) >= 0 + ], + 'filter-in-small': [ + BooleanType, + [StringType, array$1(ValueType)], + // assumes v is an array literal + (ctx, [k, v]) => v.value.indexOf(ctx.properties()[k.value]) >= 0 + ], + 'filter-in-large': [ + BooleanType, + [StringType, array$1(ValueType)], + // assumes v is a array literal with values sorted in ascending order and of a single type + (ctx, [k, v]) => binarySearch(ctx.properties()[k.value], v.value, 0, v.value.length - 1) + ], + 'all': { + type: BooleanType, + overloads: [ + [ + [BooleanType, BooleanType], + (ctx, [a, b]) => a.evaluate(ctx) && b.evaluate(ctx) + ], + [ + varargs(BooleanType), + (ctx, args) => { + for (const arg of args) { + if (!arg.evaluate(ctx)) + return false; + } + return true; + } + ] + ] + }, + 'any': { + type: BooleanType, + overloads: [ + [ + [BooleanType, BooleanType], + (ctx, [a, b]) => a.evaluate(ctx) || b.evaluate(ctx) + ], + [ + varargs(BooleanType), + (ctx, args) => { + for (const arg of args) { + if (arg.evaluate(ctx)) + return true; + } + return false; + } + ] + ] + }, + '!': [ + BooleanType, + [BooleanType], + (ctx, [b]) => !b.evaluate(ctx) + ], + 'is-supported-script': [ + BooleanType, + [StringType], + // At parse time this will always return true, so we need to exclude this expression with isGlobalPropertyConstant + (ctx, [s]) => { + const isSupportedScript = ctx.globals && ctx.globals.isSupportedScript; + if (isSupportedScript) { + return isSupportedScript(s.evaluate(ctx)); + } + return true; + } + ], + 'upcase': [ + StringType, + [StringType], + (ctx, [s]) => s.evaluate(ctx).toUpperCase() + ], + 'downcase': [ + StringType, + [StringType], + (ctx, [s]) => s.evaluate(ctx).toLowerCase() + ], + 'concat': [ + StringType, + varargs(ValueType), + (ctx, args) => args.map(arg => toString(arg.evaluate(ctx))).join('') + ], + 'resolved-locale': [ + StringType, + [CollatorType], + (ctx, [collator]) => collator.evaluate(ctx).resolvedLocale() + ] +}); + +function success(value) { + return { result: 'success', value }; +} +function error(value) { + return { result: 'error', value }; +} + +function supportsPropertyExpression(spec) { + return spec['property-type'] === 'data-driven' || spec['property-type'] === 'cross-faded-data-driven'; +} +function supportsZoomExpression(spec) { + return !!spec.expression && spec.expression.parameters.indexOf('zoom') > -1; +} +function supportsInterpolation(spec) { + return !!spec.expression && spec.expression.interpolated; +} + +function getType(val) { + if (val instanceof Number) { + return 'number'; + } + else if (val instanceof String) { + return 'string'; + } + else if (val instanceof Boolean) { + return 'boolean'; + } + else if (Array.isArray(val)) { + return 'array'; + } + else if (val === null) { + return 'null'; + } + else { + return typeof val; + } +} + +function isFunction(value) { + return typeof value === 'object' && value !== null && !Array.isArray(value); +} +function identityFunction(x) { + return x; +} +function createFunction(parameters, propertySpec) { + const isColor = propertySpec.type === 'color'; + const zoomAndFeatureDependent = parameters.stops && typeof parameters.stops[0][0] === 'object'; + const featureDependent = zoomAndFeatureDependent || parameters.property !== undefined; + const zoomDependent = zoomAndFeatureDependent || !featureDependent; + const type = parameters.type || (supportsInterpolation(propertySpec) ? 'exponential' : 'interval'); + if (isColor || propertySpec.type === 'padding') { + const parseFn = isColor ? Color.parse : Padding.parse; + parameters = extendBy({}, parameters); + if (parameters.stops) { + parameters.stops = parameters.stops.map((stop) => { + return [stop[0], parseFn(stop[1])]; + }); + } + if (parameters.default) { + parameters.default = parseFn(parameters.default); + } + else { + parameters.default = parseFn(propertySpec.default); + } + } + if (parameters.colorSpace && !isSupportedInterpolationColorSpace(parameters.colorSpace)) { + throw new Error(`Unknown color space: "${parameters.colorSpace}"`); + } + let innerFun; + let hashedStops; + let categoricalKeyType; + if (type === 'exponential') { + innerFun = evaluateExponentialFunction; + } + else if (type === 'interval') { + innerFun = evaluateIntervalFunction; + } + else if (type === 'categorical') { + innerFun = evaluateCategoricalFunction; + // For categorical functions, generate an Object as a hashmap of the stops for fast searching + hashedStops = Object.create(null); + for (const stop of parameters.stops) { + hashedStops[stop[0]] = stop[1]; + } + // Infer key type based on first stop key-- used to encforce strict type checking later + categoricalKeyType = typeof parameters.stops[0][0]; + } + else if (type === 'identity') { + innerFun = evaluateIdentityFunction; + } + else { + throw new Error(`Unknown function type "${type}"`); + } + if (zoomAndFeatureDependent) { + const featureFunctions = {}; + const zoomStops = []; + for (let s = 0; s < parameters.stops.length; s++) { + const stop = parameters.stops[s]; + const zoom = stop[0].zoom; + if (featureFunctions[zoom] === undefined) { + featureFunctions[zoom] = { + zoom, + type: parameters.type, + property: parameters.property, + default: parameters.default, + stops: [] + }; + zoomStops.push(zoom); + } + featureFunctions[zoom].stops.push([stop[0].value, stop[1]]); + } + const featureFunctionStops = []; + for (const z of zoomStops) { + featureFunctionStops.push([featureFunctions[z].zoom, createFunction(featureFunctions[z], propertySpec)]); + } + const interpolationType = { name: 'linear' }; + return { + kind: 'composite', + interpolationType, + interpolationFactor: Interpolate.interpolationFactor.bind(undefined, interpolationType), + zoomStops: featureFunctionStops.map(s => s[0]), + evaluate({ zoom }, properties) { + return evaluateExponentialFunction({ + stops: featureFunctionStops, + base: parameters.base + }, propertySpec, zoom).evaluate(zoom, properties); + } + }; + } + else if (zoomDependent) { + const interpolationType = type === 'exponential' ? + { name: 'exponential', base: parameters.base !== undefined ? parameters.base : 1 } : null; + return { + kind: 'camera', + interpolationType, + interpolationFactor: Interpolate.interpolationFactor.bind(undefined, interpolationType), + zoomStops: parameters.stops.map(s => s[0]), + evaluate: ({ zoom }) => innerFun(parameters, propertySpec, zoom, hashedStops, categoricalKeyType) + }; + } + else { + return { + kind: 'source', + evaluate(_, feature) { + const value = feature && feature.properties ? feature.properties[parameters.property] : undefined; + if (value === undefined) { + return coalesce$1(parameters.default, propertySpec.default); + } + return innerFun(parameters, propertySpec, value, hashedStops, categoricalKeyType); + } + }; + } +} +function coalesce$1(a, b, c) { + if (a !== undefined) + return a; + if (b !== undefined) + return b; + if (c !== undefined) + return c; +} +function evaluateCategoricalFunction(parameters, propertySpec, input, hashedStops, keyType) { + const evaluated = typeof input === keyType ? hashedStops[input] : undefined; // Enforce strict typing on input + return coalesce$1(evaluated, parameters.default, propertySpec.default); +} +function evaluateIntervalFunction(parameters, propertySpec, input) { + // Edge cases + if (getType(input) !== 'number') + return coalesce$1(parameters.default, propertySpec.default); + const n = parameters.stops.length; + if (n === 1) + return parameters.stops[0][1]; + if (input <= parameters.stops[0][0]) + return parameters.stops[0][1]; + if (input >= parameters.stops[n - 1][0]) + return parameters.stops[n - 1][1]; + const index = findStopLessThanOrEqualTo(parameters.stops.map((stop) => stop[0]), input); + return parameters.stops[index][1]; +} +function evaluateExponentialFunction(parameters, propertySpec, input) { + const base = parameters.base !== undefined ? parameters.base : 1; + // Edge cases + if (getType(input) !== 'number') + return coalesce$1(parameters.default, propertySpec.default); + const n = parameters.stops.length; + if (n === 1) + return parameters.stops[0][1]; + if (input <= parameters.stops[0][0]) + return parameters.stops[0][1]; + if (input >= parameters.stops[n - 1][0]) + return parameters.stops[n - 1][1]; + const index = findStopLessThanOrEqualTo(parameters.stops.map((stop) => stop[0]), input); + const t = interpolationFactor(input, base, parameters.stops[index][0], parameters.stops[index + 1][0]); + const outputLower = parameters.stops[index][1]; + const outputUpper = parameters.stops[index + 1][1]; + const interp = interpolate[propertySpec.type] || identityFunction; + if (typeof outputLower.evaluate === 'function') { + return { + evaluate(...args) { + const evaluatedLower = outputLower.evaluate.apply(undefined, args); + const evaluatedUpper = outputUpper.evaluate.apply(undefined, args); + // Special case for fill-outline-color, which has no spec default. + if (evaluatedLower === undefined || evaluatedUpper === undefined) { + return undefined; + } + return interp(evaluatedLower, evaluatedUpper, t, parameters.colorSpace); + } + }; + } + return interp(outputLower, outputUpper, t, parameters.colorSpace); +} +function evaluateIdentityFunction(parameters, propertySpec, input) { + switch (propertySpec.type) { + case 'color': + input = Color.parse(input); + break; + case 'formatted': + input = Formatted.fromString(input.toString()); + break; + case 'resolvedImage': + input = ResolvedImage.fromString(input.toString()); + break; + case 'padding': + input = Padding.parse(input); + break; + default: + if (getType(input) !== propertySpec.type && (propertySpec.type !== 'enum' || !propertySpec.values[input])) { + input = undefined; + } + } + return coalesce$1(input, parameters.default, propertySpec.default); +} +/** + * Returns a ratio that can be used to interpolate between exponential function + * stops. + * + * How it works: + * Two consecutive stop values define a (scaled and shifted) exponential + * function `f(x) = a * base^x + b`, where `base` is the user-specified base, + * and `a` and `b` are constants affording sufficient degrees of freedom to fit + * the function to the given stops. + * + * Here's a bit of algebra that lets us compute `f(x)` directly from the stop + * values without explicitly solving for `a` and `b`: + * + * First stop value: `f(x0) = y0 = a * base^x0 + b` + * Second stop value: `f(x1) = y1 = a * base^x1 + b` + * => `y1 - y0 = a(base^x1 - base^x0)` + * => `a = (y1 - y0)/(base^x1 - base^x0)` + * + * Desired value: `f(x) = y = a * base^x + b` + * => `f(x) = y0 + a * (base^x - base^x0)` + * + * From the above, we can replace the `a` in `a * (base^x - base^x0)` and do a + * little algebra: + * ``` + * a * (base^x - base^x0) = (y1 - y0)/(base^x1 - base^x0) * (base^x - base^x0) + * = (y1 - y0) * (base^x - base^x0) / (base^x1 - base^x0) + * ``` + * + * If we let `(base^x - base^x0) / (base^x1 base^x0)`, then we have + * `f(x) = y0 + (y1 - y0) * ratio`. In other words, `ratio` may be treated as + * an interpolation factor between the two stops' output values. + * + * (Note: a slightly different form for `ratio`, + * `(base^(x-x0) - 1) / (base^(x1-x0) - 1) `, is equivalent, but requires fewer + * expensive `Math.pow()` operations.) + * + * @private + */ +function interpolationFactor(input, base, lowerValue, upperValue) { + const difference = upperValue - lowerValue; + const progress = input - lowerValue; + if (difference === 0) { + return 0; + } + else if (base === 1) { + return progress / difference; + } + else { + return (Math.pow(base, progress) - 1) / (Math.pow(base, difference) - 1); + } +} + +class StyleExpression { + constructor(expression, propertySpec) { + this.expression = expression; + this._warningHistory = {}; + this._evaluator = new EvaluationContext(); + this._defaultValue = propertySpec ? getDefaultValue(propertySpec) : null; + this._enumValues = propertySpec && propertySpec.type === 'enum' ? propertySpec.values : null; + } + evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) { + this._evaluator.globals = globals; + this._evaluator.feature = feature; + this._evaluator.featureState = featureState; + this._evaluator.canonical = canonical; + this._evaluator.availableImages = availableImages || null; + this._evaluator.formattedSection = formattedSection; + return this.expression.evaluate(this._evaluator); + } + evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) { + this._evaluator.globals = globals; + this._evaluator.feature = feature || null; + this._evaluator.featureState = featureState || null; + this._evaluator.canonical = canonical; + this._evaluator.availableImages = availableImages || null; + this._evaluator.formattedSection = formattedSection || null; + try { + const val = this.expression.evaluate(this._evaluator); + // eslint-disable-next-line no-self-compare + if (val === null || val === undefined || (typeof val === 'number' && val !== val)) { + return this._defaultValue; + } + if (this._enumValues && !(val in this._enumValues)) { + throw new RuntimeError(`Expected value to be one of ${Object.keys(this._enumValues).map(v => JSON.stringify(v)).join(', ')}, but found ${JSON.stringify(val)} instead.`); + } + return val; + } + catch (e) { + if (!this._warningHistory[e.message]) { + this._warningHistory[e.message] = true; + if (typeof console !== 'undefined') { + console.warn(e.message); + } + } + return this._defaultValue; + } + } +} +function isExpression(expression) { + return Array.isArray(expression) && expression.length > 0 && + typeof expression[0] === 'string' && expression[0] in expressions; +} +/** + * Parse and typecheck the given style spec JSON expression. If + * options.defaultValue is provided, then the resulting StyleExpression's + * `evaluate()` method will handle errors by logging a warning (once per + * message) and returning the default value. Otherwise, it will throw + * evaluation errors. + * + * @private + */ +function createExpression(expression, propertySpec) { + const parser = new ParsingContext(expressions, isExpressionConstant, [], propertySpec ? getExpectedType(propertySpec) : undefined); + // For string-valued properties, coerce to string at the top level rather than asserting. + const parsed = parser.parse(expression, undefined, undefined, undefined, propertySpec && propertySpec.type === 'string' ? { typeAnnotation: 'coerce' } : undefined); + if (!parsed) { + return error(parser.errors); + } + return success(new StyleExpression(parsed, propertySpec)); +} +class ZoomConstantExpression { + constructor(kind, expression) { + this.kind = kind; + this._styleExpression = expression; + this.isStateDependent = kind !== 'constant' && !isStateConstant(expression.expression); + } + evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) { + return this._styleExpression.evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection); + } + evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) { + return this._styleExpression.evaluate(globals, feature, featureState, canonical, availableImages, formattedSection); + } +} +class ZoomDependentExpression { + constructor(kind, expression, zoomStops, interpolationType) { + this.kind = kind; + this.zoomStops = zoomStops; + this._styleExpression = expression; + this.isStateDependent = kind !== 'camera' && !isStateConstant(expression.expression); + this.interpolationType = interpolationType; + } + evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) { + return this._styleExpression.evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection); + } + evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) { + return this._styleExpression.evaluate(globals, feature, featureState, canonical, availableImages, formattedSection); + } + interpolationFactor(input, lower, upper) { + if (this.interpolationType) { + return Interpolate.interpolationFactor(this.interpolationType, input, lower, upper); + } + else { + return 0; + } + } +} +function isZoomExpression(expression) { + return expression._styleExpression !== undefined; +} +function createPropertyExpression(expressionInput, propertySpec) { + const expression = createExpression(expressionInput, propertySpec); + if (expression.result === 'error') { + return expression; + } + const parsed = expression.value.expression; + const isFeatureConstantResult = isFeatureConstant(parsed); + if (!isFeatureConstantResult && !supportsPropertyExpression(propertySpec)) { + return error([new ExpressionParsingError('', 'data expressions not supported')]); + } + const isZoomConstant = isGlobalPropertyConstant(parsed, ['zoom']); + if (!isZoomConstant && !supportsZoomExpression(propertySpec)) { + return error([new ExpressionParsingError('', 'zoom expressions not supported')]); + } + const zoomCurve = findZoomCurve(parsed); + if (!zoomCurve && !isZoomConstant) { + return error([new ExpressionParsingError('', '"zoom" expression may only be used as input to a top-level "step" or "interpolate" expression.')]); + } + else if (zoomCurve instanceof ExpressionParsingError) { + return error([zoomCurve]); + } + else if (zoomCurve instanceof Interpolate && !supportsInterpolation(propertySpec)) { + return error([new ExpressionParsingError('', '"interpolate" expressions cannot be used with this property')]); + } + if (!zoomCurve) { + return success(isFeatureConstantResult ? + new ZoomConstantExpression('constant', expression.value) : + new ZoomConstantExpression('source', expression.value)); + } + const interpolationType = zoomCurve instanceof Interpolate ? zoomCurve.interpolation : undefined; + return success(isFeatureConstantResult ? + new ZoomDependentExpression('camera', expression.value, zoomCurve.labels, interpolationType) : + new ZoomDependentExpression('composite', expression.value, zoomCurve.labels, interpolationType)); +} +// serialization wrapper for old-style stop functions normalized to the +// expression interface +class StylePropertyFunction { + constructor(parameters, specification) { + this._parameters = parameters; + this._specification = specification; + extendBy(this, createFunction(this._parameters, this._specification)); + } + static deserialize(serialized) { + return new StylePropertyFunction(serialized._parameters, serialized._specification); + } + static serialize(input) { + return { + _parameters: input._parameters, + _specification: input._specification + }; + } +} +function normalizePropertyExpression(value, specification) { + if (isFunction(value)) { + return new StylePropertyFunction(value, specification); + } + else if (isExpression(value)) { + const expression = createPropertyExpression(value, specification); + if (expression.result === 'error') { + // this should have been caught in validation + throw new Error(expression.value.map(err => `${err.key}: ${err.message}`).join(', ')); + } + return expression.value; + } + else { + let constant = value; + if (specification.type === 'color' && typeof value === 'string') { + constant = Color.parse(value); + } + else if (specification.type === 'padding' && (typeof value === 'number' || Array.isArray(value))) { + constant = Padding.parse(value); + } + else if (specification.type === 'variableAnchorOffsetCollection' && Array.isArray(value)) { + constant = VariableAnchorOffsetCollection.parse(value); + } + return { + kind: 'constant', + evaluate: () => constant + }; + } +} +// Zoom-dependent expressions may only use ["zoom"] as the input to a top-level "step" or "interpolate" +// expression (collectively referred to as a "curve"). The curve may be wrapped in one or more "let" or +// "coalesce" expressions. +function findZoomCurve(expression) { + let result = null; + if (expression instanceof Let) { + result = findZoomCurve(expression.result); + } + else if (expression instanceof Coalesce) { + for (const arg of expression.args) { + result = findZoomCurve(arg); + if (result) { + break; + } + } + } + else if ((expression instanceof Step || expression instanceof Interpolate) && + expression.input instanceof CompoundExpression && + expression.input.name === 'zoom') { + result = expression; + } + if (result instanceof ExpressionParsingError) { + return result; + } + expression.eachChild((child) => { + const childResult = findZoomCurve(child); + if (childResult instanceof ExpressionParsingError) { + result = childResult; + } + else if (!result && childResult) { + result = new ExpressionParsingError('', '"zoom" expression may only be used as input to a top-level "step" or "interpolate" expression.'); + } + else if (result && childResult && result !== childResult) { + result = new ExpressionParsingError('', 'Only one zoom-based "step" or "interpolate" subexpression may be used in an expression.'); + } + }); + return result; +} +function getExpectedType(spec) { + const types = { + color: ColorType, + string: StringType, + number: NumberType, + enum: StringType, + boolean: BooleanType, + formatted: FormattedType, + padding: PaddingType, + resolvedImage: ResolvedImageType, + variableAnchorOffsetCollection: VariableAnchorOffsetCollectionType + }; + if (spec.type === 'array') { + return array$1(types[spec.value] || ValueType, spec.length); + } + return types[spec.type]; +} +function getDefaultValue(spec) { + if (spec.type === 'color' && isFunction(spec.default)) { + // Special case for heatmap-color: it uses the 'default:' to define a + // default color ramp, but createExpression expects a simple value to fall + // back to in case of runtime errors + return new Color(0, 0, 0, 0); + } + else if (spec.type === 'color') { + return Color.parse(spec.default) || null; + } + else if (spec.type === 'padding') { + return Padding.parse(spec.default) || null; + } + else if (spec.type === 'variableAnchorOffsetCollection') { + return VariableAnchorOffsetCollection.parse(spec.default) || null; + } + else if (spec.default === undefined) { + return null; + } + else { + return spec.default; + } +} + +function isExpressionFilter(filter) { + if (filter === true || filter === false) { + return true; + } + if (!Array.isArray(filter) || filter.length === 0) { + return false; + } + switch (filter[0]) { + case 'has': + return filter.length >= 2 && filter[1] !== '$id' && filter[1] !== '$type'; + case 'in': + return filter.length >= 3 && (typeof filter[1] !== 'string' || Array.isArray(filter[2])); + case '!in': + case '!has': + case 'none': + return false; + case '==': + case '!=': + case '>': + case '>=': + case '<': + case '<=': + return filter.length !== 3 || (Array.isArray(filter[1]) || Array.isArray(filter[2])); + case 'any': + case 'all': + for (const f of filter.slice(1)) { + if (!isExpressionFilter(f) && typeof f !== 'boolean') { + return false; + } + } + return true; + default: + return true; + } +} +const filterSpec = { + 'type': 'boolean', + 'default': false, + 'transition': false, + 'property-type': 'data-driven', + 'expression': { + 'interpolated': false, + 'parameters': ['zoom', 'feature'] + } +}; +/** + * Given a filter expressed as nested arrays, return a new function + * that evaluates whether a given feature (with a .properties or .tags property) + * passes its test. + * + * @private + * @param {Array} filter MapLibre filter + * @returns {Function} filter-evaluating function + */ +function createFilter(filter) { + if (filter === null || filter === undefined) { + return { filter: () => true, needGeometry: false }; + } + if (!isExpressionFilter(filter)) { + filter = convertFilter$1(filter); + } + const compiled = createExpression(filter, filterSpec); + if (compiled.result === 'error') { + throw new Error(compiled.value.map(err => `${err.key}: ${err.message}`).join(', ')); + } + else { + const needGeometry = geometryNeeded(filter); + return { filter: (globalProperties, feature, canonical) => compiled.value.evaluate(globalProperties, feature, {}, canonical), + needGeometry }; + } +} +// Comparison function to sort numbers and strings +function compare(a, b) { + return a < b ? -1 : a > b ? 1 : 0; +} +function geometryNeeded(filter) { + if (!Array.isArray(filter)) + return false; + if (filter[0] === 'within') + return true; + for (let index = 1; index < filter.length; index++) { + if (geometryNeeded(filter[index])) + return true; + } + return false; +} +function convertFilter$1(filter) { + if (!filter) + return true; + const op = filter[0]; + if (filter.length <= 1) + return (op !== 'any'); + const converted = op === '==' ? convertComparisonOp$1(filter[1], filter[2], '==') : + op === '!=' ? convertNegation(convertComparisonOp$1(filter[1], filter[2], '==')) : + op === '<' || + op === '>' || + op === '<=' || + op === '>=' ? convertComparisonOp$1(filter[1], filter[2], op) : + op === 'any' ? convertDisjunctionOp(filter.slice(1)) : + op === 'all' ? ['all'].concat(filter.slice(1).map(convertFilter$1)) : + op === 'none' ? ['all'].concat(filter.slice(1).map(convertFilter$1).map(convertNegation)) : + op === 'in' ? convertInOp$1(filter[1], filter.slice(2)) : + op === '!in' ? convertNegation(convertInOp$1(filter[1], filter.slice(2))) : + op === 'has' ? convertHasOp$1(filter[1]) : + op === '!has' ? convertNegation(convertHasOp$1(filter[1])) : + op === 'within' ? filter : + true; + return converted; +} +function convertComparisonOp$1(property, value, op) { + switch (property) { + case '$type': + return [`filter-type-${op}`, value]; + case '$id': + return [`filter-id-${op}`, value]; + default: + return [`filter-${op}`, property, value]; + } +} +function convertDisjunctionOp(filters) { + return ['any'].concat(filters.map(convertFilter$1)); +} +function convertInOp$1(property, values) { + if (values.length === 0) { + return false; + } + switch (property) { + case '$type': + return ['filter-type-in', ['literal', values]]; + case '$id': + return ['filter-id-in', ['literal', values]]; + default: + if (values.length > 200 && !values.some(v => typeof v !== typeof values[0])) { + return ['filter-in-large', property, ['literal', values.sort(compare)]]; + } + else { + return ['filter-in-small', property, ['literal', values]]; + } + } +} +function convertHasOp$1(property) { + switch (property) { + case '$type': + return true; + case '$id': + return ['filter-has-id']; + default: + return ['filter-has', property]; + } +} +function convertNegation(filter) { + return ['!', filter]; +} + +/* + * Convert the given filter to an expression, storing the expected types for + * any feature properties referenced in expectedTypes. + * + * These expected types are needed in order to construct preflight type checks + * needed for handling 'any' filters. A preflight type check is necessary in + * order to mimic legacy filters' semantics around expected type mismatches. + * For example, consider the legacy filter: + * + * ["any", ["all", [">", "y", 0], [">", "y", 0]], [">", "x", 0]] + * + * Naively, we might convert this to the expression: + * + * ["any", ["all", [">", ["get", "y"], 0], [">", ["get", "z"], 0]], [">", ["get", "x"], 0]] + * + * But if we tried to evaluate this against, say `{x: 1, y: null, z: 0}`, the + * [">", ["get", "y"], 0] would cause an evaluation error, leading to the + * entire filter returning false. Legacy filter semantics, though, ask for + * [">", "y", 0] to simply return `false` when `y` is of the wrong type, + * allowing the subsequent terms of the outer "any" expression to be evaluated + * (resulting, in this case, in a `true` value, because x > 0). + * + * We account for this by inserting a preflight type-checking expression before + * each "any" term, allowing us to avoid evaluating the actual converted filter + * if any type mismatches would cause it to produce an evalaution error: + * + * ["any", + * ["case", + * ["all", ["==", ["typeof", ["get", "y"]], "number"], ["==", ["typeof", ["get", "z"], "number]], + * ["all", [">", ["get", "y"], 0], [">", ["get", "z"], 0]], + * false + * ], + * ["case", + * ["==", ["typeof", ["get", "x"], "number"]], + * [">", ["get", "x"], 0], + * false + * ] + * ] + * + * An alternative, possibly more direct approach would be to use type checks + * in the conversion of each comparison operator, so that the converted version + * of each individual ==, >=, etc. would mimic the legacy filter semantics. The + * downside of this approach is that it can lead to many more type checks than + * would otherwise be necessary: outside the context of an "any" expression, + * bailing out due to a runtime type error (expression semantics) and returning + * false (legacy filter semantics) are equivalent: they cause the filter to + * produce a `false` result. + */ +function convertFilter(filter, expectedTypes = {}) { + if (isExpressionFilter(filter)) + return filter; + if (!filter) + return true; + const legacyFilter = filter; + const legacyOp = legacyFilter[0]; + if (filter.length <= 1) + return (legacyOp !== 'any'); + switch (legacyOp) { + case '==': + case '!=': + case '<': + case '>': + case '<=': + case '>=': { + const [, property, value] = filter; + return convertComparisonOp(property, value, legacyOp, expectedTypes); + } + case 'any': { + const [, ...conditions] = legacyFilter; + const children = conditions.map((f) => { + const types = {}; + const child = convertFilter(f, types); + const typechecks = runtimeTypeChecks(types); + return typechecks === true ? child : ['case', typechecks, child, false]; + }); + return ['any', ...children]; + } + case 'all': { + const [, ...conditions] = legacyFilter; + const children = conditions.map(f => convertFilter(f, expectedTypes)); + return children.length > 1 ? ['all', ...children] : children[0]; + } + case 'none': { + const [, ...conditions] = legacyFilter; + return ['!', convertFilter(['any', ...conditions], {})]; + } + case 'in': { + const [, property, ...values] = legacyFilter; + return convertInOp(property, values); + } + case '!in': { + const [, property, ...values] = legacyFilter; + return convertInOp(property, values, true); + } + case 'has': + return convertHasOp(legacyFilter[1]); + case '!has': + return ['!', convertHasOp(legacyFilter[1])]; + default: + return true; + } +} +// Given a set of feature properties and an expected type for each one, +// construct an boolean expression that tests whether each property has the +// right type. +// E.g.: for {name: 'string', population: 'number'}, return +// [ 'all', +// ['==', ['typeof', ['get', 'name'], 'string']], +// ['==', ['typeof', ['get', 'population'], 'number]] +// ] +function runtimeTypeChecks(expectedTypes) { + const conditions = []; + for (const property in expectedTypes) { + const get = property === '$id' ? ['id'] : ['get', property]; + conditions.push(['==', ['typeof', get], expectedTypes[property]]); + } + if (conditions.length === 0) + return true; + if (conditions.length === 1) + return conditions[0]; + return ['all', ...conditions]; +} +function convertComparisonOp(property, value, op, expectedTypes) { + let get; + if (property === '$type') { + return [op, ['geometry-type'], value]; + } + else if (property === '$id') { + get = ['id']; + } + else { + get = ['get', property]; + } + if (expectedTypes && value !== null) { + const type = typeof value; + expectedTypes[property] = type; + } + if (op === '==' && property !== '$id' && value === null) { + return [ + 'all', + ['has', property], + ['==', get, null] + ]; + } + else if (op === '!=' && property !== '$id' && value === null) { + return [ + 'any', + ['!', ['has', property]], + ['!=', get, null] + ]; + } + return [op, get, value]; +} +function convertInOp(property, values, negate = false) { + if (values.length === 0) + return negate; + let get; + if (property === '$type') { + get = ['geometry-type']; + } + else if (property === '$id') { + get = ['id']; + } + else { + get = ['get', property]; + } + // Determine if the list of values to be searched is homogenously typed. + // If so (and if the type is string or number), then we can use a + // [match, input, [...values], true, false] construction rather than a + // bunch of `==` tests. + let uniformTypes = true; + const type = typeof values[0]; + for (const value of values) { + if (typeof value !== type) { + uniformTypes = false; + break; + } + } + if (uniformTypes && (type === 'string' || type === 'number')) { + // Match expressions must have unique values. + const uniqueValues = values.sort().filter((v, i) => i === 0 || values[i - 1] !== v); + return ['match', get, uniqueValues, !negate, negate]; + } + if (negate) { + return ['all', ...values.map(v => ['!=', get, v])]; + } + else { + return ['any', ...values.map(v => ['==', get, v])]; + } +} +function convertHasOp(property) { + if (property === '$type') { + return true; + } + else if (property === '$id') { + return ['!=', ['id'], null]; + } + else { + return ['has', property]; + } +} + +function convertLiteral(value) { + return typeof value === 'object' ? ['literal', value] : value; +} +function convertFunction(parameters, propertySpec) { + let stops = parameters.stops; + if (!stops) { + // identity function + return convertIdentityFunction(parameters, propertySpec); + } + const zoomAndFeatureDependent = stops && typeof stops[0][0] === 'object'; + const featureDependent = zoomAndFeatureDependent || parameters.property !== undefined; + const zoomDependent = zoomAndFeatureDependent || !featureDependent; + stops = stops.map((stop) => { + if (!featureDependent && propertySpec.tokens && typeof stop[1] === 'string') { + return [stop[0], convertTokenString(stop[1])]; + } + return [stop[0], convertLiteral(stop[1])]; + }); + if (zoomAndFeatureDependent) { + return convertZoomAndPropertyFunction(parameters, propertySpec, stops); + } + else if (zoomDependent) { + return convertZoomFunction(parameters, propertySpec, stops); + } + else { + return convertPropertyFunction(parameters, propertySpec, stops); + } +} +function convertIdentityFunction(parameters, propertySpec) { + const get = ['get', parameters.property]; + if (parameters.default === undefined) { + // By default, expressions for string-valued properties get coerced. To preserve + // legacy function semantics, insert an explicit assertion instead. + return propertySpec.type === 'string' ? ['string', get] : get; + } + else if (propertySpec.type === 'enum') { + return [ + 'match', + get, + Object.keys(propertySpec.values), + get, + parameters.default + ]; + } + else { + const expression = [propertySpec.type === 'color' ? 'to-color' : propertySpec.type, get, convertLiteral(parameters.default)]; + if (propertySpec.type === 'array') { + expression.splice(1, 0, propertySpec.value, propertySpec.length || null); + } + return expression; + } +} +function getInterpolateOperator(parameters) { + switch (parameters.colorSpace) { + case 'hcl': return 'interpolate-hcl'; + case 'lab': return 'interpolate-lab'; + default: return 'interpolate'; + } +} +function convertZoomAndPropertyFunction(parameters, propertySpec, stops) { + const featureFunctionParameters = {}; + const featureFunctionStops = {}; + const zoomStops = []; + for (let s = 0; s < stops.length; s++) { + const stop = stops[s]; + const zoom = stop[0].zoom; + if (featureFunctionParameters[zoom] === undefined) { + featureFunctionParameters[zoom] = { + zoom, + type: parameters.type, + property: parameters.property, + default: parameters.default, + }; + featureFunctionStops[zoom] = []; + zoomStops.push(zoom); + } + featureFunctionStops[zoom].push([stop[0].value, stop[1]]); + } + // the interpolation type for the zoom dimension of a zoom-and-property + // function is determined directly from the style property specification + // for which it's being used: linear for interpolatable properties, step + // otherwise. + const functionType = getFunctionType({}, propertySpec); + if (functionType === 'exponential') { + const expression = [getInterpolateOperator(parameters), ['linear'], ['zoom']]; + for (const z of zoomStops) { + const output = convertPropertyFunction(featureFunctionParameters[z], propertySpec, featureFunctionStops[z]); + appendStopPair(expression, z, output, false); + } + return expression; + } + else { + const expression = ['step', ['zoom']]; + for (const z of zoomStops) { + const output = convertPropertyFunction(featureFunctionParameters[z], propertySpec, featureFunctionStops[z]); + appendStopPair(expression, z, output, true); + } + fixupDegenerateStepCurve(expression); + return expression; + } +} +function coalesce(a, b) { + if (a !== undefined) + return a; + if (b !== undefined) + return b; +} +function getFallback(parameters, propertySpec) { + const defaultValue = convertLiteral(coalesce(parameters.default, propertySpec.default)); + /* + * Some fields with type: resolvedImage have an undefined default. + * Because undefined is an invalid value for resolvedImage, set fallback to + * an empty string instead of undefined to ensure output + * passes validation. + */ + if (defaultValue === undefined && propertySpec.type === 'resolvedImage') { + return ''; + } + return defaultValue; +} +function convertPropertyFunction(parameters, propertySpec, stops) { + const type = getFunctionType(parameters, propertySpec); + const get = ['get', parameters.property]; + if (type === 'categorical' && typeof stops[0][0] === 'boolean') { + const expression = ['case']; + for (const stop of stops) { + expression.push(['==', get, stop[0]], stop[1]); + } + expression.push(getFallback(parameters, propertySpec)); + return expression; + } + else if (type === 'categorical') { + const expression = ['match', get]; + for (const stop of stops) { + appendStopPair(expression, stop[0], stop[1], false); + } + expression.push(getFallback(parameters, propertySpec)); + return expression; + } + else if (type === 'interval') { + const expression = ['step', ['number', get]]; + for (const stop of stops) { + appendStopPair(expression, stop[0], stop[1], true); + } + fixupDegenerateStepCurve(expression); + return parameters.default === undefined ? expression : [ + 'case', + ['==', ['typeof', get], 'number'], + expression, + convertLiteral(parameters.default) + ]; + } + else if (type === 'exponential') { + const base = parameters.base !== undefined ? parameters.base : 1; + const expression = [ + getInterpolateOperator(parameters), + base === 1 ? ['linear'] : ['exponential', base], + ['number', get] + ]; + for (const stop of stops) { + appendStopPair(expression, stop[0], stop[1], false); + } + return parameters.default === undefined ? expression : [ + 'case', + ['==', ['typeof', get], 'number'], + expression, + convertLiteral(parameters.default) + ]; + } + else { + throw new Error(`Unknown property function type ${type}`); + } +} +function convertZoomFunction(parameters, propertySpec, stops, input = ['zoom']) { + const type = getFunctionType(parameters, propertySpec); + let expression; + let isStep = false; + if (type === 'interval') { + expression = ['step', input]; + isStep = true; + } + else if (type === 'exponential') { + const base = parameters.base !== undefined ? parameters.base : 1; + expression = [getInterpolateOperator(parameters), base === 1 ? ['linear'] : ['exponential', base], input]; + } + else { + throw new Error(`Unknown zoom function type "${type}"`); + } + for (const stop of stops) { + appendStopPair(expression, stop[0], stop[1], isStep); + } + fixupDegenerateStepCurve(expression); + return expression; +} +function fixupDegenerateStepCurve(expression) { + // degenerate step curve (i.e. a constant function): add a noop stop + if (expression[0] === 'step' && expression.length === 3) { + expression.push(0); + expression.push(expression[3]); + } +} +function appendStopPair(curve, input, output, isStep) { + // Skip duplicate stop values. They were not validated for functions, but they are for expressions. + // https://github.com/mapbox/mapbox-gl-js/issues/4107 + if (curve.length > 3 && input === curve[curve.length - 2]) { + return; + } + // step curves don't get the first input value, as it is redundant. + if (!(isStep && curve.length === 2)) { + curve.push(input); + } + curve.push(output); +} +function getFunctionType(parameters, propertySpec) { + if (parameters.type) { + return parameters.type; + } + else { + return propertySpec.expression.interpolated ? 'exponential' : 'interval'; + } +} +// "String with {name} token" => ["concat", "String with ", ["get", "name"], " token"] +function convertTokenString(s) { + const result = ['concat']; + const re = /{([^{}]+)}/g; + let pos = 0; + for (let match = re.exec(s); match !== null; match = re.exec(s)) { + const literal = s.slice(pos, re.lastIndex - match[0].length); + pos = re.lastIndex; + if (literal.length > 0) + result.push(literal); + result.push(['get', match[1]]); + } + if (result.length === 1) { + return s; + } + if (pos < s.length) { + result.push(s.slice(pos)); + } + else if (result.length === 2) { + return ['to-string', result[1]]; + } + return result; +} + +function getPropertyReference(propertyName) { + for (let i = 0; i < v8Spec.layout.length; i++) { + for (const key in v8Spec[v8Spec.layout[i]]) { + if (key === propertyName) + return v8Spec[v8Spec.layout[i]][key]; + } + } + for (let i = 0; i < v8Spec.paint.length; i++) { + for (const key in v8Spec[v8Spec.paint[i]]) { + if (key === propertyName) + return v8Spec[v8Spec.paint[i]][key]; + } + } + return null; +} +function eachSource(style, callback) { + for (const k in style.sources) { + callback(style.sources[k]); + } +} +function eachLayer(style, callback) { + for (const layer of style.layers) { + callback(layer); + } +} +function eachProperty(style, options, callback) { + function inner(layer, propertyType) { + const properties = layer[propertyType]; + if (!properties) + return; + Object.keys(properties).forEach((key) => { + callback({ + path: [layer.id, propertyType, key], + key, + value: properties[key], + reference: getPropertyReference(key), + set(x) { + properties[key] = x; + } + }); + }); + } + eachLayer(style, (layer) => { + if (options.paint) { + inner(layer, 'paint'); + } + if (options.layout) { + inner(layer, 'layout'); + } + }); +} + +function stringify(obj) { + const type = typeof obj; + if (type === 'number' || type === 'boolean' || type === 'string' || obj === undefined || obj === null) + return JSON.stringify(obj); + if (Array.isArray(obj)) { + let str = '['; + for (const val of obj) { + str += `${stringify(val)},`; + } + return `${str}]`; + } + const keys = Object.keys(obj).sort(); + let str = '{'; + for (let i = 0; i < keys.length; i++) { + str += `${JSON.stringify(keys[i])}:${stringify(obj[keys[i]])},`; + } + return `${str}}`; +} +function getKey(layer) { + let key = ''; + for (const k of refProperties) { + key += `/${stringify(layer[k])}`; + } + return key; +} +/** + * Given an array of layers, return an array of arrays of layers where all + * layers in each group have identical layout-affecting properties. These + * are the properties that were formerly used by explicit `ref` mechanism + * for layers: 'type', 'source', 'source-layer', 'minzoom', 'maxzoom', + * 'filter', and 'layout'. + * + * The input is not modified. The output layers are references to the + * input layers. + * + * @private + * @param {Array} layers + * @param {Object} [cachedKeys] - an object to keep already calculated keys. + * @returns {Array>} + */ +function groupByLayout(layers, cachedKeys) { + const groups = {}; + for (let i = 0; i < layers.length; i++) { + const k = (cachedKeys && cachedKeys[layers[i].id]) || getKey(layers[i]); + // update the cache if there is one + if (cachedKeys) + cachedKeys[layers[i].id] = k; + let group = groups[k]; + if (!group) { + group = groups[k] = []; + } + group.push(layers[i]); + } + const result = []; + for (const k in groups) { + result.push(groups[k]); + } + return result; +} + +function emptyStyle() { + const style = {}; + const version = v8Spec['$version']; + for (const styleKey in v8Spec['$root']) { + const spec = v8Spec['$root'][styleKey]; + if (spec.required) { + let value = null; + if (styleKey === 'version') { + value = version; + } + else { + if (spec.type === 'array') { + value = []; + } + else { + value = {}; + } + } + if (value != null) { + style[styleKey] = value; + } + } + } + return style; +} + +function validateConstants(options) { + const key = options.key; + const constants = options.value; + if (constants) { + return [new ValidationError(key, constants, 'constants have been deprecated as of v8')]; + } + else { + return []; + } +} + +// Turn jsonlint-lines-primitives objects into primitive objects +function unbundle(value) { + if (value instanceof Number || value instanceof String || value instanceof Boolean) { + return value.valueOf(); + } + else { + return value; + } +} +function deepUnbundle(value) { + if (Array.isArray(value)) { + return value.map(deepUnbundle); + } + else if (value instanceof Object && !(value instanceof Number || value instanceof String || value instanceof Boolean)) { + const unbundledValue = {}; + for (const key in value) { + unbundledValue[key] = deepUnbundle(value[key]); + } + return unbundledValue; + } + return unbundle(value); +} + +function validateObject(options) { + const key = options.key; + const object = options.value; + const elementSpecs = options.valueSpec || {}; + const elementValidators = options.objectElementValidators || {}; + const style = options.style; + const styleSpec = options.styleSpec; + const validateSpec = options.validateSpec; + let errors = []; + const type = getType(object); + if (type !== 'object') { + return [new ValidationError(key, object, `object expected, ${type} found`)]; + } + for (const objectKey in object) { + const elementSpecKey = objectKey.split('.')[0]; // treat 'paint.*' as 'paint' + const elementSpec = elementSpecs[elementSpecKey] || elementSpecs['*']; + let validateElement; + if (elementValidators[elementSpecKey]) { + validateElement = elementValidators[elementSpecKey]; + } + else if (elementSpecs[elementSpecKey]) { + validateElement = validateSpec; + } + else if (elementValidators['*']) { + validateElement = elementValidators['*']; + } + else if (elementSpecs['*']) { + validateElement = validateSpec; + } + else { + errors.push(new ValidationError(key, object[objectKey], `unknown property "${objectKey}"`)); + continue; + } + errors = errors.concat(validateElement({ + key: (key ? `${key}.` : key) + objectKey, + value: object[objectKey], + valueSpec: elementSpec, + style, + styleSpec, + object, + objectKey, + validateSpec, + }, object)); + } + for (const elementSpecKey in elementSpecs) { + // Don't check `required` when there's a custom validator for that property. + if (elementValidators[elementSpecKey]) { + continue; + } + if (elementSpecs[elementSpecKey].required && elementSpecs[elementSpecKey]['default'] === undefined && object[elementSpecKey] === undefined) { + errors.push(new ValidationError(key, object, `missing required property "${elementSpecKey}"`)); + } + } + return errors; +} + +function validateArray(options) { + const array = options.value; + const arraySpec = options.valueSpec; + const validateSpec = options.validateSpec; + const style = options.style; + const styleSpec = options.styleSpec; + const key = options.key; + const validateArrayElement = options.arrayElementValidator || validateSpec; + if (getType(array) !== 'array') { + return [new ValidationError(key, array, `array expected, ${getType(array)} found`)]; + } + if (arraySpec.length && array.length !== arraySpec.length) { + return [new ValidationError(key, array, `array length ${arraySpec.length} expected, length ${array.length} found`)]; + } + if (arraySpec['min-length'] && array.length < arraySpec['min-length']) { + return [new ValidationError(key, array, `array length at least ${arraySpec['min-length']} expected, length ${array.length} found`)]; + } + let arrayElementSpec = { + 'type': arraySpec.value, + 'values': arraySpec.values + }; + if (styleSpec.$version < 7) { + arrayElementSpec['function'] = arraySpec.function; + } + if (getType(arraySpec.value) === 'object') { + arrayElementSpec = arraySpec.value; + } + let errors = []; + for (let i = 0; i < array.length; i++) { + errors = errors.concat(validateArrayElement({ + array, + arrayIndex: i, + value: array[i], + valueSpec: arrayElementSpec, + validateSpec: options.validateSpec, + style, + styleSpec, + key: `${key}[${i}]` + })); + } + return errors; +} + +function validateNumber(options) { + const key = options.key; + const value = options.value; + const valueSpec = options.valueSpec; + let type = getType(value); + // eslint-disable-next-line no-self-compare + if (type === 'number' && value !== value) { + type = 'NaN'; + } + if (type !== 'number') { + return [new ValidationError(key, value, `number expected, ${type} found`)]; + } + if ('minimum' in valueSpec && value < valueSpec.minimum) { + return [new ValidationError(key, value, `${value} is less than the minimum value ${valueSpec.minimum}`)]; + } + if ('maximum' in valueSpec && value > valueSpec.maximum) { + return [new ValidationError(key, value, `${value} is greater than the maximum value ${valueSpec.maximum}`)]; + } + return []; +} + +function validateFunction(options) { + const functionValueSpec = options.valueSpec; + const functionType = unbundle(options.value.type); + let stopKeyType; + let stopDomainValues = {}; + let previousStopDomainValue; + let previousStopDomainZoom; + const isZoomFunction = functionType !== 'categorical' && options.value.property === undefined; + const isPropertyFunction = !isZoomFunction; + const isZoomAndPropertyFunction = getType(options.value.stops) === 'array' && + getType(options.value.stops[0]) === 'array' && + getType(options.value.stops[0][0]) === 'object'; + const errors = validateObject({ + key: options.key, + value: options.value, + valueSpec: options.styleSpec.function, + validateSpec: options.validateSpec, + style: options.style, + styleSpec: options.styleSpec, + objectElementValidators: { + stops: validateFunctionStops, + default: validateFunctionDefault + } + }); + if (functionType === 'identity' && isZoomFunction) { + errors.push(new ValidationError(options.key, options.value, 'missing required property "property"')); + } + if (functionType !== 'identity' && !options.value.stops) { + errors.push(new ValidationError(options.key, options.value, 'missing required property "stops"')); + } + if (functionType === 'exponential' && options.valueSpec.expression && !supportsInterpolation(options.valueSpec)) { + errors.push(new ValidationError(options.key, options.value, 'exponential functions not supported')); + } + if (options.styleSpec.$version >= 8) { + if (isPropertyFunction && !supportsPropertyExpression(options.valueSpec)) { + errors.push(new ValidationError(options.key, options.value, 'property functions not supported')); + } + else if (isZoomFunction && !supportsZoomExpression(options.valueSpec)) { + errors.push(new ValidationError(options.key, options.value, 'zoom functions not supported')); + } + } + if ((functionType === 'categorical' || isZoomAndPropertyFunction) && options.value.property === undefined) { + errors.push(new ValidationError(options.key, options.value, '"property" property is required')); + } + return errors; + function validateFunctionStops(options) { + if (functionType === 'identity') { + return [new ValidationError(options.key, options.value, 'identity function may not have a "stops" property')]; + } + let errors = []; + const value = options.value; + errors = errors.concat(validateArray({ + key: options.key, + value, + valueSpec: options.valueSpec, + validateSpec: options.validateSpec, + style: options.style, + styleSpec: options.styleSpec, + arrayElementValidator: validateFunctionStop + })); + if (getType(value) === 'array' && value.length === 0) { + errors.push(new ValidationError(options.key, value, 'array must have at least one stop')); + } + return errors; + } + function validateFunctionStop(options) { + let errors = []; + const value = options.value; + const key = options.key; + if (getType(value) !== 'array') { + return [new ValidationError(key, value, `array expected, ${getType(value)} found`)]; + } + if (value.length !== 2) { + return [new ValidationError(key, value, `array length 2 expected, length ${value.length} found`)]; + } + if (isZoomAndPropertyFunction) { + if (getType(value[0]) !== 'object') { + return [new ValidationError(key, value, `object expected, ${getType(value[0])} found`)]; + } + if (value[0].zoom === undefined) { + return [new ValidationError(key, value, 'object stop key must have zoom')]; + } + if (value[0].value === undefined) { + return [new ValidationError(key, value, 'object stop key must have value')]; + } + if (previousStopDomainZoom && previousStopDomainZoom > unbundle(value[0].zoom)) { + return [new ValidationError(key, value[0].zoom, 'stop zoom values must appear in ascending order')]; + } + if (unbundle(value[0].zoom) !== previousStopDomainZoom) { + previousStopDomainZoom = unbundle(value[0].zoom); + previousStopDomainValue = undefined; + stopDomainValues = {}; + } + errors = errors.concat(validateObject({ + key: `${key}[0]`, + value: value[0], + valueSpec: { zoom: {} }, + validateSpec: options.validateSpec, + style: options.style, + styleSpec: options.styleSpec, + objectElementValidators: { zoom: validateNumber, value: validateStopDomainValue } + })); + } + else { + errors = errors.concat(validateStopDomainValue({ + key: `${key}[0]`, + value: value[0], + valueSpec: {}, + validateSpec: options.validateSpec, + style: options.style, + styleSpec: options.styleSpec + }, value)); + } + if (isExpression(deepUnbundle(value[1]))) { + return errors.concat([new ValidationError(`${key}[1]`, value[1], 'expressions are not allowed in function stops.')]); + } + return errors.concat(options.validateSpec({ + key: `${key}[1]`, + value: value[1], + valueSpec: functionValueSpec, + validateSpec: options.validateSpec, + style: options.style, + styleSpec: options.styleSpec + })); + } + function validateStopDomainValue(options, stop) { + const type = getType(options.value); + const value = unbundle(options.value); + const reportValue = options.value !== null ? options.value : stop; + if (!stopKeyType) { + stopKeyType = type; + } + else if (type !== stopKeyType) { + return [new ValidationError(options.key, reportValue, `${type} stop domain type must match previous stop domain type ${stopKeyType}`)]; + } + if (type !== 'number' && type !== 'string' && type !== 'boolean') { + return [new ValidationError(options.key, reportValue, 'stop domain value must be a number, string, or boolean')]; + } + if (type !== 'number' && functionType !== 'categorical') { + let message = `number expected, ${type} found`; + if (supportsPropertyExpression(functionValueSpec) && functionType === undefined) { + message += '\nIf you intended to use a categorical function, specify `"type": "categorical"`.'; + } + return [new ValidationError(options.key, reportValue, message)]; + } + if (functionType === 'categorical' && type === 'number' && (!isFinite(value) || Math.floor(value) !== value)) { + return [new ValidationError(options.key, reportValue, `integer expected, found ${value}`)]; + } + if (functionType !== 'categorical' && type === 'number' && previousStopDomainValue !== undefined && value < previousStopDomainValue) { + return [new ValidationError(options.key, reportValue, 'stop domain values must appear in ascending order')]; + } + else { + previousStopDomainValue = value; + } + if (functionType === 'categorical' && value in stopDomainValues) { + return [new ValidationError(options.key, reportValue, 'stop domain values must be unique')]; + } + else { + stopDomainValues[value] = true; + } + return []; + } + function validateFunctionDefault(options) { + return options.validateSpec({ + key: options.key, + value: options.value, + valueSpec: functionValueSpec, + validateSpec: options.validateSpec, + style: options.style, + styleSpec: options.styleSpec + }); + } +} + +function validateExpression(options) { + const expression = (options.expressionContext === 'property' ? createPropertyExpression : createExpression)(deepUnbundle(options.value), options.valueSpec); + if (expression.result === 'error') { + return expression.value.map((error) => { + return new ValidationError(`${options.key}${error.key}`, options.value, error.message); + }); + } + const expressionObj = expression.value.expression || expression.value._styleExpression.expression; + if (options.expressionContext === 'property' && (options.propertyKey === 'text-font') && + !expressionObj.outputDefined()) { + return [new ValidationError(options.key, options.value, `Invalid data expression for "${options.propertyKey}". Output values must be contained as literals within the expression.`)]; + } + if (options.expressionContext === 'property' && options.propertyType === 'layout' && + (!isStateConstant(expressionObj))) { + return [new ValidationError(options.key, options.value, '"feature-state" data expressions are not supported with layout properties.')]; + } + if (options.expressionContext === 'filter' && !isStateConstant(expressionObj)) { + return [new ValidationError(options.key, options.value, '"feature-state" data expressions are not supported with filters.')]; + } + if (options.expressionContext && options.expressionContext.indexOf('cluster') === 0) { + if (!isGlobalPropertyConstant(expressionObj, ['zoom', 'feature-state'])) { + return [new ValidationError(options.key, options.value, '"zoom" and "feature-state" expressions are not supported with cluster properties.')]; + } + if (options.expressionContext === 'cluster-initial' && !isFeatureConstant(expressionObj)) { + return [new ValidationError(options.key, options.value, 'Feature data expressions are not supported with initial expression part of cluster properties.')]; + } + } + return []; +} + +function validateBoolean(options) { + const value = options.value; + const key = options.key; + const type = getType(value); + if (type !== 'boolean') { + return [new ValidationError(key, value, `boolean expected, ${type} found`)]; + } + return []; +} + +function validateColor(options) { + const key = options.key; + const value = options.value; + const type = getType(value); + if (type !== 'string') { + return [new ValidationError(key, value, `color expected, ${type} found`)]; + } + if (!Color.parse(String(value))) { // cast String object to string primitive + return [new ValidationError(key, value, `color expected, "${value}" found`)]; + } + return []; +} + +function validateEnum(options) { + const key = options.key; + const value = options.value; + const valueSpec = options.valueSpec; + const errors = []; + if (Array.isArray(valueSpec.values)) { // <=v7 + if (valueSpec.values.indexOf(unbundle(value)) === -1) { + errors.push(new ValidationError(key, value, `expected one of [${valueSpec.values.join(', ')}], ${JSON.stringify(value)} found`)); + } + } + else { // >=v8 + if (Object.keys(valueSpec.values).indexOf(unbundle(value)) === -1) { + errors.push(new ValidationError(key, value, `expected one of [${Object.keys(valueSpec.values).join(', ')}], ${JSON.stringify(value)} found`)); + } + } + return errors; +} + +function validateFilter$1(options) { + if (isExpressionFilter(deepUnbundle(options.value))) { + return validateExpression(extendBy({}, options, { + expressionContext: 'filter', + valueSpec: { value: 'boolean' } + })); + } + else { + return validateNonExpressionFilter(options); + } +} +function validateNonExpressionFilter(options) { + const value = options.value; + const key = options.key; + if (getType(value) !== 'array') { + return [new ValidationError(key, value, `array expected, ${getType(value)} found`)]; + } + const styleSpec = options.styleSpec; + let type; + let errors = []; + if (value.length < 1) { + return [new ValidationError(key, value, 'filter array must have at least 1 element')]; + } + errors = errors.concat(validateEnum({ + key: `${key}[0]`, + value: value[0], + valueSpec: styleSpec.filter_operator, + style: options.style, + styleSpec: options.styleSpec + })); + switch (unbundle(value[0])) { + case '<': + case '<=': + case '>': + case '>=': + if (value.length >= 2 && unbundle(value[1]) === '$type') { + errors.push(new ValidationError(key, value, `"$type" cannot be use with operator "${value[0]}"`)); + } + /* falls through */ + case '==': + case '!=': + if (value.length !== 3) { + errors.push(new ValidationError(key, value, `filter array for operator "${value[0]}" must have 3 elements`)); + } + /* falls through */ + case 'in': + case '!in': + if (value.length >= 2) { + type = getType(value[1]); + if (type !== 'string') { + errors.push(new ValidationError(`${key}[1]`, value[1], `string expected, ${type} found`)); + } + } + for (let i = 2; i < value.length; i++) { + type = getType(value[i]); + if (unbundle(value[1]) === '$type') { + errors = errors.concat(validateEnum({ + key: `${key}[${i}]`, + value: value[i], + valueSpec: styleSpec.geometry_type, + style: options.style, + styleSpec: options.styleSpec + })); + } + else if (type !== 'string' && type !== 'number' && type !== 'boolean') { + errors.push(new ValidationError(`${key}[${i}]`, value[i], `string, number, or boolean expected, ${type} found`)); + } + } + break; + case 'any': + case 'all': + case 'none': + for (let i = 1; i < value.length; i++) { + errors = errors.concat(validateNonExpressionFilter({ + key: `${key}[${i}]`, + value: value[i], + style: options.style, + styleSpec: options.styleSpec + })); + } + break; + case 'has': + case '!has': + type = getType(value[1]); + if (value.length !== 2) { + errors.push(new ValidationError(key, value, `filter array for "${value[0]}" operator must have 2 elements`)); + } + else if (type !== 'string') { + errors.push(new ValidationError(`${key}[1]`, value[1], `string expected, ${type} found`)); + } + break; + case 'within': + type = getType(value[1]); + if (value.length !== 2) { + errors.push(new ValidationError(key, value, `filter array for "${value[0]}" operator must have 2 elements`)); + } + else if (type !== 'object') { + errors.push(new ValidationError(`${key}[1]`, value[1], `object expected, ${type} found`)); + } + break; + } + return errors; +} + +function validateProperty(options, propertyType) { + const key = options.key; + const validateSpec = options.validateSpec; + const style = options.style; + const styleSpec = options.styleSpec; + const value = options.value; + const propertyKey = options.objectKey; + const layerSpec = styleSpec[`${propertyType}_${options.layerType}`]; + if (!layerSpec) + return []; + const transitionMatch = propertyKey.match(/^(.*)-transition$/); + if (propertyType === 'paint' && transitionMatch && layerSpec[transitionMatch[1]] && layerSpec[transitionMatch[1]].transition) { + return validateSpec({ + key, + value, + valueSpec: styleSpec.transition, + style, + styleSpec + }); + } + const valueSpec = options.valueSpec || layerSpec[propertyKey]; + if (!valueSpec) { + return [new ValidationError(key, value, `unknown property "${propertyKey}"`)]; + } + let tokenMatch; + if (getType(value) === 'string' && supportsPropertyExpression(valueSpec) && !valueSpec.tokens && (tokenMatch = /^{([^}]+)}$/.exec(value))) { + return [new ValidationError(key, value, `"${propertyKey}" does not support interpolation syntax\n` + + `Use an identity property function instead: \`{ "type": "identity", "property": ${JSON.stringify(tokenMatch[1])} }\`.`)]; + } + const errors = []; + if (options.layerType === 'symbol') { + if (propertyKey === 'text-field' && style && !style.glyphs) { + errors.push(new ValidationError(key, value, 'use of "text-field" requires a style "glyphs" property')); + } + if (propertyKey === 'text-font' && isFunction(deepUnbundle(value)) && unbundle(value.type) === 'identity') { + errors.push(new ValidationError(key, value, '"text-font" does not support identity functions')); + } + } + return errors.concat(validateSpec({ + key: options.key, + value, + valueSpec, + style, + styleSpec, + expressionContext: 'property', + propertyType, + propertyKey + })); +} + +function validatePaintProperty$1(options) { + return validateProperty(options, 'paint'); +} + +function validateLayoutProperty$1(options) { + return validateProperty(options, 'layout'); +} + +function validateLayer(options) { + let errors = []; + const layer = options.value; + const key = options.key; + const style = options.style; + const styleSpec = options.styleSpec; + if (!layer.type && !layer.ref) { + errors.push(new ValidationError(key, layer, 'either "type" or "ref" is required')); + } + let type = unbundle(layer.type); + const ref = unbundle(layer.ref); + if (layer.id) { + const layerId = unbundle(layer.id); + for (let i = 0; i < options.arrayIndex; i++) { + const otherLayer = style.layers[i]; + if (unbundle(otherLayer.id) === layerId) { + errors.push(new ValidationError(key, layer.id, `duplicate layer id "${layer.id}", previously used at line ${otherLayer.id.__line__}`)); + } + } + } + if ('ref' in layer) { + ['type', 'source', 'source-layer', 'filter', 'layout'].forEach((p) => { + if (p in layer) { + errors.push(new ValidationError(key, layer[p], `"${p}" is prohibited for ref layers`)); + } + }); + let parent; + style.layers.forEach((layer) => { + if (unbundle(layer.id) === ref) + parent = layer; + }); + if (!parent) { + errors.push(new ValidationError(key, layer.ref, `ref layer "${ref}" not found`)); + } + else if (parent.ref) { + errors.push(new ValidationError(key, layer.ref, 'ref cannot reference another ref layer')); + } + else { + type = unbundle(parent.type); + } + } + else if (type !== 'background') { + if (!layer.source) { + errors.push(new ValidationError(key, layer, 'missing required property "source"')); + } + else { + const source = style.sources && style.sources[layer.source]; + const sourceType = source && unbundle(source.type); + if (!source) { + errors.push(new ValidationError(key, layer.source, `source "${layer.source}" not found`)); + } + else if (sourceType === 'vector' && type === 'raster') { + errors.push(new ValidationError(key, layer.source, `layer "${layer.id}" requires a raster source`)); + } + else if (sourceType !== 'raster-dem' && type === 'hillshade') { + errors.push(new ValidationError(key, layer.source, `layer "${layer.id}" requires a raster-dem source`)); + } + else if (sourceType === 'raster' && type !== 'raster') { + errors.push(new ValidationError(key, layer.source, `layer "${layer.id}" requires a vector source`)); + } + else if (sourceType === 'vector' && !layer['source-layer']) { + errors.push(new ValidationError(key, layer, `layer "${layer.id}" must specify a "source-layer"`)); + } + else if (sourceType === 'raster-dem' && type !== 'hillshade') { + errors.push(new ValidationError(key, layer.source, 'raster-dem source can only be used with layer type \'hillshade\'.')); + } + else if (type === 'line' && layer.paint && layer.paint['line-gradient'] && + (sourceType !== 'geojson' || !source.lineMetrics)) { + errors.push(new ValidationError(key, layer, `layer "${layer.id}" specifies a line-gradient, which requires a GeoJSON source with \`lineMetrics\` enabled.`)); + } + } + } + errors = errors.concat(validateObject({ + key, + value: layer, + valueSpec: styleSpec.layer, + style: options.style, + styleSpec: options.styleSpec, + validateSpec: options.validateSpec, + objectElementValidators: { + '*'() { + return []; + }, + // We don't want to enforce the spec's `"requires": true` for backward compatibility with refs; + // the actual requirement is validated above. See https://github.com/mapbox/mapbox-gl-js/issues/5772. + type() { + return options.validateSpec({ + key: `${key}.type`, + value: layer.type, + valueSpec: styleSpec.layer.type, + style: options.style, + styleSpec: options.styleSpec, + validateSpec: options.validateSpec, + object: layer, + objectKey: 'type' + }); + }, + filter: validateFilter$1, + layout(options) { + return validateObject({ + layer, + key: options.key, + value: options.value, + style: options.style, + styleSpec: options.styleSpec, + validateSpec: options.validateSpec, + objectElementValidators: { + '*'(options) { + return validateLayoutProperty$1(extendBy({ layerType: type }, options)); + } + } + }); + }, + paint(options) { + return validateObject({ + layer, + key: options.key, + value: options.value, + style: options.style, + styleSpec: options.styleSpec, + validateSpec: options.validateSpec, + objectElementValidators: { + '*'(options) { + return validatePaintProperty$1(extendBy({ layerType: type }, options)); + } + } + }); + } + } + })); + return errors; +} + +function validateString(options) { + const value = options.value; + const key = options.key; + const type = getType(value); + if (type !== 'string') { + return [new ValidationError(key, value, `string expected, ${type} found`)]; + } + return []; +} + +function validateRasterDEMSource(options) { + var _a; + const sourceName = (_a = options.sourceName) !== null && _a !== void 0 ? _a : ''; + const rasterDEM = options.value; + const styleSpec = options.styleSpec; + const rasterDEMSpec = styleSpec.source_raster_dem; + const style = options.style; + let errors = []; + const rootType = getType(rasterDEM); + if (rasterDEM === undefined) { + return errors; + } + else if (rootType !== 'object') { + errors.push(new ValidationError('source_raster_dem', rasterDEM, `object expected, ${rootType} found`)); + return errors; + } + const encoding = unbundle(rasterDEM.encoding); + const isCustomEncoding = encoding === 'custom'; + const customEncodingKeys = ['redFactor', 'greenFactor', 'blueFactor', 'baseShift']; + const encodingName = options.value.encoding ? `"${options.value.encoding}"` : 'Default'; + for (const key in rasterDEM) { + if (!isCustomEncoding && customEncodingKeys.includes(key)) { + errors.push(new ValidationError(key, rasterDEM[key], `In "${sourceName}": "${key}" is only valid when "encoding" is set to "custom". ${encodingName} encoding found`)); + } + else if (rasterDEMSpec[key]) { + errors = errors.concat(options.validateSpec({ + key, + value: rasterDEM[key], + valueSpec: rasterDEMSpec[key], + validateSpec: options.validateSpec, + style, + styleSpec + })); + } + else { + errors.push(new ValidationError(key, rasterDEM[key], `unknown property "${key}"`)); + } + } + return errors; +} + +const objectElementValidators = { + promoteId: validatePromoteId +}; +function validateSource$1(options) { + const value = options.value; + const key = options.key; + const styleSpec = options.styleSpec; + const style = options.style; + const validateSpec = options.validateSpec; + if (!value.type) { + return [new ValidationError(key, value, '"type" is required')]; + } + const type = unbundle(value.type); + let errors; + switch (type) { + case 'vector': + case 'raster': + errors = validateObject({ + key, + value, + valueSpec: styleSpec[`source_${type.replace('-', '_')}`], + style: options.style, + styleSpec, + objectElementValidators, + validateSpec, + }); + return errors; + case 'raster-dem': + errors = validateRasterDEMSource({ + sourceName: key, + value, + style: options.style, + styleSpec, + validateSpec, + }); + return errors; + case 'geojson': + errors = validateObject({ + key, + value, + valueSpec: styleSpec.source_geojson, + style, + styleSpec, + validateSpec, + objectElementValidators + }); + if (value.cluster) { + for (const prop in value.clusterProperties) { + const [operator, mapExpr] = value.clusterProperties[prop]; + const reduceExpr = typeof operator === 'string' ? [operator, ['accumulated'], ['get', prop]] : operator; + errors.push(...validateExpression({ + key: `${key}.${prop}.map`, + value: mapExpr, + validateSpec, + expressionContext: 'cluster-map' + })); + errors.push(...validateExpression({ + key: `${key}.${prop}.reduce`, + value: reduceExpr, + validateSpec, + expressionContext: 'cluster-reduce' + })); + } + } + return errors; + case 'video': + return validateObject({ + key, + value, + valueSpec: styleSpec.source_video, + style, + validateSpec, + styleSpec + }); + case 'image': + return validateObject({ + key, + value, + valueSpec: styleSpec.source_image, + style, + validateSpec, + styleSpec + }); + case 'canvas': + return [new ValidationError(key, null, 'Please use runtime APIs to add canvas sources, rather than including them in stylesheets.', 'source.canvas')]; + default: + return validateEnum({ + key: `${key}.type`, + value: value.type, + valueSpec: { values: ['vector', 'raster', 'raster-dem', 'geojson', 'video', 'image'] }, + style, + validateSpec, + styleSpec + }); + } +} +function validatePromoteId({ key, value }) { + if (getType(value) === 'string') { + return validateString({ key, value }); + } + else { + const errors = []; + for (const prop in value) { + errors.push(...validateString({ key: `${key}.${prop}`, value: value[prop] })); + } + return errors; + } +} + +function validateLight$1(options) { + const light = options.value; + const styleSpec = options.styleSpec; + const lightSpec = styleSpec.light; + const style = options.style; + let errors = []; + const rootType = getType(light); + if (light === undefined) { + return errors; + } + else if (rootType !== 'object') { + errors = errors.concat([new ValidationError('light', light, `object expected, ${rootType} found`)]); + return errors; + } + for (const key in light) { + const transitionMatch = key.match(/^(.*)-transition$/); + if (transitionMatch && lightSpec[transitionMatch[1]] && lightSpec[transitionMatch[1]].transition) { + errors = errors.concat(options.validateSpec({ + key, + value: light[key], + valueSpec: styleSpec.transition, + validateSpec: options.validateSpec, + style, + styleSpec + })); + } + else if (lightSpec[key]) { + errors = errors.concat(options.validateSpec({ + key, + value: light[key], + valueSpec: lightSpec[key], + validateSpec: options.validateSpec, + style, + styleSpec + })); + } + else { + errors = errors.concat([new ValidationError(key, light[key], `unknown property "${key}"`)]); + } + } + return errors; +} + +function validateTerrain$1(options) { + const terrain = options.value; + const styleSpec = options.styleSpec; + const terrainSpec = styleSpec.terrain; + const style = options.style; + let errors = []; + const rootType = getType(terrain); + if (terrain === undefined) { + return errors; + } + else if (rootType !== 'object') { + errors = errors.concat([new ValidationError('terrain', terrain, `object expected, ${rootType} found`)]); + return errors; + } + for (const key in terrain) { + if (terrainSpec[key]) { + errors = errors.concat(options.validateSpec({ + key, + value: terrain[key], + valueSpec: terrainSpec[key], + validateSpec: options.validateSpec, + style, + styleSpec + })); + } + else { + errors = errors.concat([new ValidationError(key, terrain[key], `unknown property "${key}"`)]); + } + } + return errors; +} + +function validateFormatted(options) { + if (validateString(options).length === 0) { + return []; + } + return validateExpression(options); +} + +function validateImage(options) { + if (validateString(options).length === 0) { + return []; + } + return validateExpression(options); +} + +function validatePadding(options) { + const key = options.key; + const value = options.value; + const type = getType(value); + if (type === 'array') { + if (value.length < 1 || value.length > 4) { + return [new ValidationError(key, value, `padding requires 1 to 4 values; ${value.length} values found`)]; + } + const arrayElementSpec = { + type: 'number' + }; + let errors = []; + for (let i = 0; i < value.length; i++) { + errors = errors.concat(options.validateSpec({ + key: `${key}[${i}]`, + value: value[i], + validateSpec: options.validateSpec, + valueSpec: arrayElementSpec + })); + } + return errors; + } + else { + return validateNumber({ + key, + value, + valueSpec: {} + }); + } +} + +function validateVariableAnchorOffsetCollection(options) { + const key = options.key; + const value = options.value; + const type = getType(value); + const styleSpec = options.styleSpec; + if (type !== 'array' || value.length < 1 || value.length % 2 !== 0) { + return [new ValidationError(key, value, 'variableAnchorOffsetCollection requires a non-empty array of even length')]; + } + let errors = []; + for (let i = 0; i < value.length; i += 2) { + // Elements in even positions should be values from text-anchor enum + errors = errors.concat(validateEnum({ + key: `${key}[${i}]`, + value: value[i], + valueSpec: styleSpec['layout_symbol']['text-anchor'] + })); + // Elements in odd positions should be points (2-element numeric arrays) + errors = errors.concat(validateArray({ + key: `${key}[${i + 1}]`, + value: value[i + 1], + valueSpec: { + length: 2, + value: 'number' + }, + validateSpec: options.validateSpec, + style: options.style, + styleSpec + })); + } + return errors; +} + +function validateSprite(options) { + let errors = []; + const sprite = options.value; + const key = options.key; + if (!Array.isArray(sprite)) { + return validateString({ + key, + value: sprite + }); + } + else { + const allSpriteIds = []; + const allSpriteURLs = []; + for (const i in sprite) { + if (sprite[i].id && allSpriteIds.includes(sprite[i].id)) + errors.push(new ValidationError(key, sprite, `all the sprites' ids must be unique, but ${sprite[i].id} is duplicated`)); + allSpriteIds.push(sprite[i].id); + if (sprite[i].url && allSpriteURLs.includes(sprite[i].url)) + errors.push(new ValidationError(key, sprite, `all the sprites' URLs must be unique, but ${sprite[i].url} is duplicated`)); + allSpriteURLs.push(sprite[i].url); + const pairSpec = { + id: { + type: 'string', + required: true, + }, + url: { + type: 'string', + required: true, + } + }; + errors = errors.concat(validateObject({ + key: `${key}[${i}]`, + value: sprite[i], + valueSpec: pairSpec, + validateSpec: options.validateSpec, + })); + } + return errors; + } +} + +const VALIDATORS = { + '*'() { + return []; + }, + 'array': validateArray, + 'boolean': validateBoolean, + 'number': validateNumber, + 'color': validateColor, + 'constants': validateConstants, + 'enum': validateEnum, + 'filter': validateFilter$1, + 'function': validateFunction, + 'layer': validateLayer, + 'object': validateObject, + 'source': validateSource$1, + 'light': validateLight$1, + 'terrain': validateTerrain$1, + 'string': validateString, + 'formatted': validateFormatted, + 'resolvedImage': validateImage, + 'padding': validatePadding, + 'variableAnchorOffsetCollection': validateVariableAnchorOffsetCollection, + 'sprite': validateSprite, +}; +// Main recursive validation function. Tracks: +// +// - key: string representing location of validation in style tree. Used only +// for more informative error reporting. +// - value: current value from style being evaluated. May be anything from a +// high level object that needs to be descended into deeper or a simple +// scalar value. +// - valueSpec: current spec being evaluated. Tracks value. +// - styleSpec: current full spec being evaluated. +function validate(options) { + const value = options.value; + const valueSpec = options.valueSpec; + const styleSpec = options.styleSpec; + options.validateSpec = validate; + if (valueSpec.expression && isFunction(unbundle(value))) { + return validateFunction(options); + } + else if (valueSpec.expression && isExpression(deepUnbundle(value))) { + return validateExpression(options); + } + else if (valueSpec.type && VALIDATORS[valueSpec.type]) { + return VALIDATORS[valueSpec.type](options); + } + else { + const valid = validateObject(extendBy({}, options, { + valueSpec: valueSpec.type ? styleSpec[valueSpec.type] : valueSpec + })); + return valid; + } +} + +function validateGlyphsUrl(options) { + const value = options.value; + const key = options.key; + const errors = validateString(options); + if (errors.length) + return errors; + if (value.indexOf('{fontstack}') === -1) { + errors.push(new ValidationError(key, value, '"glyphs" url must include a "{fontstack}" token')); + } + if (value.indexOf('{range}') === -1) { + errors.push(new ValidationError(key, value, '"glyphs" url must include a "{range}" token')); + } + return errors; +} + +/** + * Validate a MapLibre style against the style specification. This entrypoint, + * `maplibre-gl-style-spec/lib/validate_style.min`, is designed to produce as + * small a browserify bundle as possible by omitting unnecessary functionality + * and legacy style specifications. + * + * @private + * @param {Object} style The style to be validated. + * @param {Object} [styleSpec] The style specification to validate against. + * If omitted, the latest style spec is used. + * @returns {Array} + * @example + * var validate = require('maplibre-gl-style-spec/lib/validate_style.min'); + * var errors = validate(style); + */ +function validateStyleMin(style, styleSpec = v8Spec) { + let errors = []; + errors = errors.concat(validate({ + key: '', + value: style, + valueSpec: styleSpec.$root, + styleSpec, + style, + validateSpec: validate, + objectElementValidators: { + glyphs: validateGlyphsUrl, + '*'() { + return []; + } + } + })); + if (style['constants']) { + errors = errors.concat(validateConstants({ + key: 'constants', + value: style['constants'], + style, + styleSpec, + validateSpec: validate, + })); + } + return sortErrors(errors); +} +validateStyleMin.source = wrapCleanErrors(injectValidateSpec(validateSource$1)); +validateStyleMin.sprite = wrapCleanErrors(injectValidateSpec(validateSprite)); +validateStyleMin.glyphs = wrapCleanErrors(injectValidateSpec(validateGlyphsUrl)); +validateStyleMin.light = wrapCleanErrors(injectValidateSpec(validateLight$1)); +validateStyleMin.terrain = wrapCleanErrors(injectValidateSpec(validateTerrain$1)); +validateStyleMin.layer = wrapCleanErrors(injectValidateSpec(validateLayer)); +validateStyleMin.filter = wrapCleanErrors(injectValidateSpec(validateFilter$1)); +validateStyleMin.paintProperty = wrapCleanErrors(injectValidateSpec(validatePaintProperty$1)); +validateStyleMin.layoutProperty = wrapCleanErrors(injectValidateSpec(validateLayoutProperty$1)); +function injectValidateSpec(validator) { + return function (options) { + return validator({ + ...options, + validateSpec: validate, + }); + }; +} +function sortErrors(errors) { + return [].concat(errors).sort((a, b) => { + return a.line - b.line; + }); +} +function wrapCleanErrors(inner) { + return function (...args) { + return sortErrors(inner.apply(this, args)); + }; +} + +const v8 = v8Spec; +const expression = { + StyleExpression, + StylePropertyFunction, + ZoomConstantExpression, + ZoomDependentExpression, + createExpression, + createPropertyExpression, + isExpression, + isExpressionFilter, + isZoomExpression, + normalizePropertyExpression, +}; +const styleFunction = { + convertFunction, + createFunction, + isFunction +}; +const visit = { eachLayer, eachProperty, eachSource }; + +const validateStyle = validateStyleMin; +const validateSource = validateStyle.source; +const validateLight = validateStyle.light; +const validateTerrain = validateStyle.terrain; +const validateFilter = validateStyle.filter; +const validatePaintProperty = validateStyle.paintProperty; +const validateLayoutProperty = validateStyle.layoutProperty; +function emitValidationErrors(emitter, errors) { + let hasErrors = false; + if (errors && errors.length) { + for (const error of errors) { + emitter.fire(new ErrorEvent(new Error(error.message))); + hasErrors = true; + } + } + return hasErrors; +} + +/* +This file was copied from https://github.com/mapbox/grid-index and was +migrated from JavaScript to TypeScript. + +Copyright (c) 2016, Mapbox + +Permission to use, copy, modify, and/or distribute this software for any purpose +with or without fee is hereby granted, provided that the above copyright notice +and this permission notice appear in all copies. + +THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH +REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND +FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, +INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS +OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER +TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF +THIS SOFTWARE. +*/ +const NUM_PARAMS = 3; +class TransferableGridIndex { + constructor(extent, n, padding) { + const cells = this.cells = []; + if (extent instanceof ArrayBuffer) { + this.arrayBuffer = extent; + const array = new Int32Array(this.arrayBuffer); + extent = array[0]; + n = array[1]; + padding = array[2]; + this.d = n + 2 * padding; + for (let k = 0; k < this.d * this.d; k++) { + const start = array[NUM_PARAMS + k]; + const end = array[NUM_PARAMS + k + 1]; + cells.push(start === end ? null : array.subarray(start, end)); + } + const keysOffset = array[NUM_PARAMS + cells.length]; + const bboxesOffset = array[NUM_PARAMS + cells.length + 1]; + this.keys = array.subarray(keysOffset, bboxesOffset); + this.bboxes = array.subarray(bboxesOffset); + this.insert = this._insertReadonly; + } + else { + this.d = n + 2 * padding; + for (let i = 0; i < this.d * this.d; i++) { + cells.push([]); + } + this.keys = []; + this.bboxes = []; + } + this.n = n; + this.extent = extent; + this.padding = padding; + this.scale = n / extent; + this.uid = 0; + const p = (padding / n) * extent; + this.min = -p; + this.max = extent + p; + } + insert(key, x1, y1, x2, y2) { + this._forEachCell(x1, y1, x2, y2, this._insertCell, this.uid++, undefined, undefined); + this.keys.push(key); + this.bboxes.push(x1); + this.bboxes.push(y1); + this.bboxes.push(x2); + this.bboxes.push(y2); + } + _insertReadonly() { + throw new Error('Cannot insert into a GridIndex created from an ArrayBuffer.'); + } + _insertCell(x1, y1, x2, y2, cellIndex, uid) { + this.cells[cellIndex].push(uid); + } + query(x1, y1, x2, y2, intersectionTest) { + const min = this.min; + const max = this.max; + if (x1 <= min && y1 <= min && max <= x2 && max <= y2 && !intersectionTest) { + // We use `Array#slice` because `this.keys` may be a `Int32Array` and + // some browsers (Safari and IE) do not support `TypedArray#slice` + // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/TypedArray/slice#Browser_compatibility + return Array.prototype.slice.call(this.keys); + } + else { + const result = []; + const seenUids = {}; + this._forEachCell(x1, y1, x2, y2, this._queryCell, result, seenUids, intersectionTest); + return result; + } + } + _queryCell(x1, y1, x2, y2, cellIndex, result, seenUids, intersectionTest) { + const cell = this.cells[cellIndex]; + if (cell !== null) { + const keys = this.keys; + const bboxes = this.bboxes; + for (let u = 0; u < cell.length; u++) { + const uid = cell[u]; + if (seenUids[uid] === undefined) { + const offset = uid * 4; + if (intersectionTest ? + intersectionTest(bboxes[offset + 0], bboxes[offset + 1], bboxes[offset + 2], bboxes[offset + 3]) : + ((x1 <= bboxes[offset + 2]) && + (y1 <= bboxes[offset + 3]) && + (x2 >= bboxes[offset + 0]) && + (y2 >= bboxes[offset + 1]))) { + seenUids[uid] = true; + result.push(keys[uid]); + } + else { + seenUids[uid] = false; + } + } + } + } + } + _forEachCell(x1, y1, x2, y2, fn, arg1, arg2, intersectionTest) { + const cx1 = this._convertToCellCoord(x1); + const cy1 = this._convertToCellCoord(y1); + const cx2 = this._convertToCellCoord(x2); + const cy2 = this._convertToCellCoord(y2); + for (let x = cx1; x <= cx2; x++) { + for (let y = cy1; y <= cy2; y++) { + const cellIndex = this.d * y + x; + if (intersectionTest && !intersectionTest(this._convertFromCellCoord(x), this._convertFromCellCoord(y), this._convertFromCellCoord(x + 1), this._convertFromCellCoord(y + 1))) + continue; + if (fn.call(this, x1, y1, x2, y2, cellIndex, arg1, arg2, intersectionTest)) + return; + } + } + } + _convertFromCellCoord(x) { + return (x - this.padding) / this.scale; + } + _convertToCellCoord(x) { + return Math.max(0, Math.min(this.d - 1, Math.floor(x * this.scale) + this.padding)); + } + toArrayBuffer() { + if (this.arrayBuffer) + return this.arrayBuffer; + const cells = this.cells; + const metadataLength = NUM_PARAMS + this.cells.length + 1 + 1; + let totalCellLength = 0; + for (let i = 0; i < this.cells.length; i++) { + totalCellLength += this.cells[i].length; + } + const array = new Int32Array(metadataLength + totalCellLength + this.keys.length + this.bboxes.length); + array[0] = this.extent; + array[1] = this.n; + array[2] = this.padding; + let offset = metadataLength; + for (let k = 0; k < cells.length; k++) { + const cell = cells[k]; + array[NUM_PARAMS + k] = offset; + array.set(cell, offset); + offset += cell.length; + } + array[NUM_PARAMS + cells.length] = offset; + array.set(this.keys, offset); + offset += this.keys.length; + array[NUM_PARAMS + cells.length + 1] = offset; + array.set(this.bboxes, offset); + offset += this.bboxes.length; + return array.buffer; + } + static serialize(grid, transferables) { + const buffer = grid.toArrayBuffer(); + if (transferables) { + transferables.push(buffer); + } + return { buffer }; + } + static deserialize(serialized) { + return new TransferableGridIndex(serialized.buffer); + } +} + +const registry = {}; +/** + * Register the given class as serializable. + * + * @param options - the registration options + */ +function register(name, klass, options = {}) { + if (registry[name]) + throw new Error(`${name} is already registered.`); + Object.defineProperty(klass, '_classRegistryKey', { + value: name, + writeable: false + }); + registry[name] = { + klass, + omit: options.omit || [], + shallow: options.shallow || [] + }; +} +register('Object', Object); +register('TransferableGridIndex', TransferableGridIndex); +register('Color', Color); +register('Error', Error); +register('AJAXError', AJAXError); +register('ResolvedImage', ResolvedImage); +register('StylePropertyFunction', StylePropertyFunction); +register('StyleExpression', StyleExpression, { omit: ['_evaluator'] }); +register('ZoomDependentExpression', ZoomDependentExpression); +register('ZoomConstantExpression', ZoomConstantExpression); +register('CompoundExpression', CompoundExpression, { omit: ['_evaluate'] }); +for (const name in expressions) { + if (expressions[name]._classRegistryKey) + continue; + register(`Expression_${name}`, expressions[name]); +} +function isArrayBuffer(value) { + return value && typeof ArrayBuffer !== 'undefined' && + (value instanceof ArrayBuffer || (value.constructor && value.constructor.name === 'ArrayBuffer')); +} +/** + * Serialize the given object for transfer to or from a web worker. + * + * For non-builtin types, recursively serialize each property (possibly + * omitting certain properties - see register()), and package the result along + * with the constructor's `name` so that the appropriate constructor can be + * looked up in `deserialize()`. + * + * If a `transferables` array is provided, add any transferable objects (i.e., + * any ArrayBuffers or ArrayBuffer views) to the list. (If a copy is needed, + * this should happen in the client code, before using serialize().) + */ +function serialize(input, transferables) { + if (input === null || + input === undefined || + typeof input === 'boolean' || + typeof input === 'number' || + typeof input === 'string' || + input instanceof Boolean || + input instanceof Number || + input instanceof String || + input instanceof Date || + input instanceof RegExp || + input instanceof Blob) { + return input; + } + if (isArrayBuffer(input)) { + if (transferables) { + transferables.push(input); + } + return input; + } + if (isImageBitmap(input)) { + if (transferables) { + transferables.push(input); + } + return input; + } + if (ArrayBuffer.isView(input)) { + const view = input; + if (transferables) { + transferables.push(view.buffer); + } + return view; + } + if (input instanceof ImageData) { + if (transferables) { + transferables.push(input.data.buffer); + } + return input; + } + if (Array.isArray(input)) { + const serialized = []; + for (const item of input) { + serialized.push(serialize(item, transferables)); + } + return serialized; + } + if (typeof input === 'object') { + const klass = input.constructor; + const name = klass._classRegistryKey; + if (!name) { + throw new Error('can\'t serialize object of unregistered class'); + } + if (!registry[name]) + throw new Error(`${name} is not registered.`); + const properties = klass.serialize ? + // (Temporary workaround) allow a class to provide static + // `serialize()` and `deserialize()` methods to bypass the generic + // approach. + // This temporary workaround lets us use the generic serialization + // approach for objects whose members include instances of dynamic + // StructArray types. Once we refactor StructArray to be static, + // we can remove this complexity. + klass.serialize(input, transferables) : {}; + if (!klass.serialize) { + for (const key in input) { + if (!input.hasOwnProperty(key)) + continue; // eslint-disable-line no-prototype-builtins + if (registry[name].omit.indexOf(key) >= 0) + continue; + const property = input[key]; + properties[key] = registry[name].shallow.indexOf(key) >= 0 ? + property : + serialize(property, transferables); + } + if (input instanceof Error) { + properties.message = input.message; + } + } + else { + if (transferables && properties === transferables[transferables.length - 1]) { + throw new Error('statically serialized object won\'t survive transfer of $name property'); + } + } + if (properties.$name) { + throw new Error('$name property is reserved for worker serialization logic.'); + } + if (name !== 'Object') { + properties.$name = name; + } + return properties; + } + throw new Error(`can't serialize object of type ${typeof input}`); +} +function deserialize(input) { + if (input === null || + input === undefined || + typeof input === 'boolean' || + typeof input === 'number' || + typeof input === 'string' || + input instanceof Boolean || + input instanceof Number || + input instanceof String || + input instanceof Date || + input instanceof RegExp || + input instanceof Blob || + isArrayBuffer(input) || + isImageBitmap(input) || + ArrayBuffer.isView(input) || + input instanceof ImageData) { + return input; + } + if (Array.isArray(input)) { + return input.map(deserialize); + } + if (typeof input === 'object') { + const name = input.$name || 'Object'; + if (!registry[name]) { + throw new Error(`can't deserialize unregistered class ${name}`); + } + const { klass } = registry[name]; + if (!klass) { + throw new Error(`can't deserialize unregistered class ${name}`); + } + if (klass.deserialize) { + return klass.deserialize(input); + } + const result = Object.create(klass.prototype); + for (const key of Object.keys(input)) { + if (key === '$name') + continue; + const value = input[key]; + result[key] = registry[name].shallow.indexOf(key) >= 0 ? value : deserialize(value); + } + return result; + } + throw new Error(`can't deserialize object of type ${typeof input}`); +} + +class ZoomHistory { + constructor() { + this.first = true; + } + update(z, now) { + const floorZ = Math.floor(z); + if (this.first) { + this.first = false; + this.lastIntegerZoom = floorZ; + this.lastIntegerZoomTime = 0; + this.lastZoom = z; + this.lastFloorZoom = floorZ; + return true; + } + if (this.lastFloorZoom > floorZ) { + this.lastIntegerZoom = floorZ + 1; + this.lastIntegerZoomTime = now; + } + else if (this.lastFloorZoom < floorZ) { + this.lastIntegerZoom = floorZ; + this.lastIntegerZoomTime = now; + } + if (z !== this.lastZoom) { + this.lastZoom = z; + this.lastFloorZoom = floorZ; + return true; + } + return false; + } +} + +// The following table comes from . +// Keep it synchronized with . +const unicodeBlockLookup = { + // 'Basic Latin': (char) => char >= 0x0000 && char <= 0x007F, + 'Latin-1 Supplement': (char) => char >= 0x0080 && char <= 0x00FF, + // 'Latin Extended-A': (char) => char >= 0x0100 && char <= 0x017F, + // 'Latin Extended-B': (char) => char >= 0x0180 && char <= 0x024F, + // 'IPA Extensions': (char) => char >= 0x0250 && char <= 0x02AF, + // 'Spacing Modifier Letters': (char) => char >= 0x02B0 && char <= 0x02FF, + // 'Combining Diacritical Marks': (char) => char >= 0x0300 && char <= 0x036F, + // 'Greek and Coptic': (char) => char >= 0x0370 && char <= 0x03FF, + // 'Cyrillic': (char) => char >= 0x0400 && char <= 0x04FF, + // 'Cyrillic Supplement': (char) => char >= 0x0500 && char <= 0x052F, + // 'Armenian': (char) => char >= 0x0530 && char <= 0x058F, + //'Hebrew': (char) => char >= 0x0590 && char <= 0x05FF, + 'Arabic': (char) => char >= 0x0600 && char <= 0x06FF, + //'Syriac': (char) => char >= 0x0700 && char <= 0x074F, + 'Arabic Supplement': (char) => char >= 0x0750 && char <= 0x077F, + // 'Thaana': (char) => char >= 0x0780 && char <= 0x07BF, + // 'NKo': (char) => char >= 0x07C0 && char <= 0x07FF, + // 'Samaritan': (char) => char >= 0x0800 && char <= 0x083F, + // 'Mandaic': (char) => char >= 0x0840 && char <= 0x085F, + // 'Syriac Supplement': (char) => char >= 0x0860 && char <= 0x086F, + 'Arabic Extended-A': (char) => char >= 0x08A0 && char <= 0x08FF, + // 'Devanagari': (char) => char >= 0x0900 && char <= 0x097F, + // 'Bengali': (char) => char >= 0x0980 && char <= 0x09FF, + // 'Gurmukhi': (char) => char >= 0x0A00 && char <= 0x0A7F, + // 'Gujarati': (char) => char >= 0x0A80 && char <= 0x0AFF, + // 'Oriya': (char) => char >= 0x0B00 && char <= 0x0B7F, + // 'Tamil': (char) => char >= 0x0B80 && char <= 0x0BFF, + // 'Telugu': (char) => char >= 0x0C00 && char <= 0x0C7F, + // 'Kannada': (char) => char >= 0x0C80 && char <= 0x0CFF, + // 'Malayalam': (char) => char >= 0x0D00 && char <= 0x0D7F, + // 'Sinhala': (char) => char >= 0x0D80 && char <= 0x0DFF, + // 'Thai': (char) => char >= 0x0E00 && char <= 0x0E7F, + // 'Lao': (char) => char >= 0x0E80 && char <= 0x0EFF, + // 'Tibetan': (char) => char >= 0x0F00 && char <= 0x0FFF, + // 'Myanmar': (char) => char >= 0x1000 && char <= 0x109F, + // 'Georgian': (char) => char >= 0x10A0 && char <= 0x10FF, + 'Hangul Jamo': (char) => char >= 0x1100 && char <= 0x11FF, + // 'Ethiopic': (char) => char >= 0x1200 && char <= 0x137F, + // 'Ethiopic Supplement': (char) => char >= 0x1380 && char <= 0x139F, + // 'Cherokee': (char) => char >= 0x13A0 && char <= 0x13FF, + 'Unified Canadian Aboriginal Syllabics': (char) => char >= 0x1400 && char <= 0x167F, + // 'Ogham': (char) => char >= 0x1680 && char <= 0x169F, + // 'Runic': (char) => char >= 0x16A0 && char <= 0x16FF, + // 'Tagalog': (char) => char >= 0x1700 && char <= 0x171F, + // 'Hanunoo': (char) => char >= 0x1720 && char <= 0x173F, + // 'Buhid': (char) => char >= 0x1740 && char <= 0x175F, + // 'Tagbanwa': (char) => char >= 0x1760 && char <= 0x177F, + 'Khmer': (char) => char >= 0x1780 && char <= 0x17FF, + // 'Mongolian': (char) => char >= 0x1800 && char <= 0x18AF, + 'Unified Canadian Aboriginal Syllabics Extended': (char) => char >= 0x18B0 && char <= 0x18FF, + // 'Limbu': (char) => char >= 0x1900 && char <= 0x194F, + // 'Tai Le': (char) => char >= 0x1950 && char <= 0x197F, + // 'New Tai Lue': (char) => char >= 0x1980 && char <= 0x19DF, + // 'Khmer Symbols': (char) => char >= 0x19E0 && char <= 0x19FF, + // 'Buginese': (char) => char >= 0x1A00 && char <= 0x1A1F, + // 'Tai Tham': (char) => char >= 0x1A20 && char <= 0x1AAF, + // 'Combining Diacritical Marks Extended': (char) => char >= 0x1AB0 && char <= 0x1AFF, + // 'Balinese': (char) => char >= 0x1B00 && char <= 0x1B7F, + // 'Sundanese': (char) => char >= 0x1B80 && char <= 0x1BBF, + // 'Batak': (char) => char >= 0x1BC0 && char <= 0x1BFF, + // 'Lepcha': (char) => char >= 0x1C00 && char <= 0x1C4F, + // 'Ol Chiki': (char) => char >= 0x1C50 && char <= 0x1C7F, + // 'Cyrillic Extended-C': (char) => char >= 0x1C80 && char <= 0x1C8F, + // 'Georgian Extended': (char) => char >= 0x1C90 && char <= 0x1CBF, + // 'Sundanese Supplement': (char) => char >= 0x1CC0 && char <= 0x1CCF, + // 'Vedic Extensions': (char) => char >= 0x1CD0 && char <= 0x1CFF, + // 'Phonetic Extensions': (char) => char >= 0x1D00 && char <= 0x1D7F, + // 'Phonetic Extensions Supplement': (char) => char >= 0x1D80 && char <= 0x1DBF, + // 'Combining Diacritical Marks Supplement': (char) => char >= 0x1DC0 && char <= 0x1DFF, + // 'Latin Extended Additional': (char) => char >= 0x1E00 && char <= 0x1EFF, + // 'Greek Extended': (char) => char >= 0x1F00 && char <= 0x1FFF, + 'General Punctuation': (char) => char >= 0x2000 && char <= 0x206F, + // 'Superscripts and Subscripts': (char) => char >= 0x2070 && char <= 0x209F, + // 'Currency Symbols': (char) => char >= 0x20A0 && char <= 0x20CF, + // 'Combining Diacritical Marks for Symbols': (char) => char >= 0x20D0 && char <= 0x20FF, + 'Letterlike Symbols': (char) => char >= 0x2100 && char <= 0x214F, + 'Number Forms': (char) => char >= 0x2150 && char <= 0x218F, + // 'Arrows': (char) => char >= 0x2190 && char <= 0x21FF, + // 'Mathematical Operators': (char) => char >= 0x2200 && char <= 0x22FF, + 'Miscellaneous Technical': (char) => char >= 0x2300 && char <= 0x23FF, + 'Control Pictures': (char) => char >= 0x2400 && char <= 0x243F, + 'Optical Character Recognition': (char) => char >= 0x2440 && char <= 0x245F, + 'Enclosed Alphanumerics': (char) => char >= 0x2460 && char <= 0x24FF, + // 'Box Drawing': (char) => char >= 0x2500 && char <= 0x257F, + // 'Block Elements': (char) => char >= 0x2580 && char <= 0x259F, + 'Geometric Shapes': (char) => char >= 0x25A0 && char <= 0x25FF, + 'Miscellaneous Symbols': (char) => char >= 0x2600 && char <= 0x26FF, + // 'Dingbats': (char) => char >= 0x2700 && char <= 0x27BF, + // 'Miscellaneous Mathematical Symbols-A': (char) => char >= 0x27C0 && char <= 0x27EF, + // 'Supplemental Arrows-A': (char) => char >= 0x27F0 && char <= 0x27FF, + // 'Braille Patterns': (char) => char >= 0x2800 && char <= 0x28FF, + // 'Supplemental Arrows-B': (char) => char >= 0x2900 && char <= 0x297F, + // 'Miscellaneous Mathematical Symbols-B': (char) => char >= 0x2980 && char <= 0x29FF, + // 'Supplemental Mathematical Operators': (char) => char >= 0x2A00 && char <= 0x2AFF, + 'Miscellaneous Symbols and Arrows': (char) => char >= 0x2B00 && char <= 0x2BFF, + // 'Glagolitic': (char) => char >= 0x2C00 && char <= 0x2C5F, + // 'Latin Extended-C': (char) => char >= 0x2C60 && char <= 0x2C7F, + // 'Coptic': (char) => char >= 0x2C80 && char <= 0x2CFF, + // 'Georgian Supplement': (char) => char >= 0x2D00 && char <= 0x2D2F, + // 'Tifinagh': (char) => char >= 0x2D30 && char <= 0x2D7F, + // 'Ethiopic Extended': (char) => char >= 0x2D80 && char <= 0x2DDF, + // 'Cyrillic Extended-A': (char) => char >= 0x2DE0 && char <= 0x2DFF, + // 'Supplemental Punctuation': (char) => char >= 0x2E00 && char <= 0x2E7F, + 'CJK Radicals Supplement': (char) => char >= 0x2E80 && char <= 0x2EFF, + 'Kangxi Radicals': (char) => char >= 0x2F00 && char <= 0x2FDF, + 'Ideographic Description Characters': (char) => char >= 0x2FF0 && char <= 0x2FFF, + 'CJK Symbols and Punctuation': (char) => char >= 0x3000 && char <= 0x303F, + 'Hiragana': (char) => char >= 0x3040 && char <= 0x309F, + 'Katakana': (char) => char >= 0x30A0 && char <= 0x30FF, + 'Bopomofo': (char) => char >= 0x3100 && char <= 0x312F, + 'Hangul Compatibility Jamo': (char) => char >= 0x3130 && char <= 0x318F, + 'Kanbun': (char) => char >= 0x3190 && char <= 0x319F, + 'Bopomofo Extended': (char) => char >= 0x31A0 && char <= 0x31BF, + 'CJK Strokes': (char) => char >= 0x31C0 && char <= 0x31EF, + 'Katakana Phonetic Extensions': (char) => char >= 0x31F0 && char <= 0x31FF, + 'Enclosed CJK Letters and Months': (char) => char >= 0x3200 && char <= 0x32FF, + 'CJK Compatibility': (char) => char >= 0x3300 && char <= 0x33FF, + 'CJK Unified Ideographs Extension A': (char) => char >= 0x3400 && char <= 0x4DBF, + 'Yijing Hexagram Symbols': (char) => char >= 0x4DC0 && char <= 0x4DFF, + 'CJK Unified Ideographs': (char) => char >= 0x4E00 && char <= 0x9FFF, + 'Yi Syllables': (char) => char >= 0xA000 && char <= 0xA48F, + 'Yi Radicals': (char) => char >= 0xA490 && char <= 0xA4CF, + // 'Lisu': (char) => char >= 0xA4D0 && char <= 0xA4FF, + // 'Vai': (char) => char >= 0xA500 && char <= 0xA63F, + // 'Cyrillic Extended-B': (char) => char >= 0xA640 && char <= 0xA69F, + // 'Bamum': (char) => char >= 0xA6A0 && char <= 0xA6FF, + // 'Modifier Tone Letters': (char) => char >= 0xA700 && char <= 0xA71F, + // 'Latin Extended-D': (char) => char >= 0xA720 && char <= 0xA7FF, + // 'Syloti Nagri': (char) => char >= 0xA800 && char <= 0xA82F, + // 'Common Indic Number Forms': (char) => char >= 0xA830 && char <= 0xA83F, + // 'Phags-pa': (char) => char >= 0xA840 && char <= 0xA87F, + // 'Saurashtra': (char) => char >= 0xA880 && char <= 0xA8DF, + // 'Devanagari Extended': (char) => char >= 0xA8E0 && char <= 0xA8FF, + // 'Kayah Li': (char) => char >= 0xA900 && char <= 0xA92F, + // 'Rejang': (char) => char >= 0xA930 && char <= 0xA95F, + 'Hangul Jamo Extended-A': (char) => char >= 0xA960 && char <= 0xA97F, + // 'Javanese': (char) => char >= 0xA980 && char <= 0xA9DF, + // 'Myanmar Extended-B': (char) => char >= 0xA9E0 && char <= 0xA9FF, + // 'Cham': (char) => char >= 0xAA00 && char <= 0xAA5F, + // 'Myanmar Extended-A': (char) => char >= 0xAA60 && char <= 0xAA7F, + // 'Tai Viet': (char) => char >= 0xAA80 && char <= 0xAADF, + // 'Meetei Mayek Extensions': (char) => char >= 0xAAE0 && char <= 0xAAFF, + // 'Ethiopic Extended-A': (char) => char >= 0xAB00 && char <= 0xAB2F, + // 'Latin Extended-E': (char) => char >= 0xAB30 && char <= 0xAB6F, + // 'Cherokee Supplement': (char) => char >= 0xAB70 && char <= 0xABBF, + // 'Meetei Mayek': (char) => char >= 0xABC0 && char <= 0xABFF, + 'Hangul Syllables': (char) => char >= 0xAC00 && char <= 0xD7AF, + 'Hangul Jamo Extended-B': (char) => char >= 0xD7B0 && char <= 0xD7FF, + // 'High Surrogates': (char) => char >= 0xD800 && char <= 0xDB7F, + // 'High Private Use Surrogates': (char) => char >= 0xDB80 && char <= 0xDBFF, + // 'Low Surrogates': (char) => char >= 0xDC00 && char <= 0xDFFF, + 'Private Use Area': (char) => char >= 0xE000 && char <= 0xF8FF, + 'CJK Compatibility Ideographs': (char) => char >= 0xF900 && char <= 0xFAFF, + // 'Alphabetic Presentation Forms': (char) => char >= 0xFB00 && char <= 0xFB4F, + 'Arabic Presentation Forms-A': (char) => char >= 0xFB50 && char <= 0xFDFF, + // 'Variation Selectors': (char) => char >= 0xFE00 && char <= 0xFE0F, + 'Vertical Forms': (char) => char >= 0xFE10 && char <= 0xFE1F, + // 'Combining Half Marks': (char) => char >= 0xFE20 && char <= 0xFE2F, + 'CJK Compatibility Forms': (char) => char >= 0xFE30 && char <= 0xFE4F, + 'Small Form Variants': (char) => char >= 0xFE50 && char <= 0xFE6F, + 'Arabic Presentation Forms-B': (char) => char >= 0xFE70 && char <= 0xFEFF, + 'Halfwidth and Fullwidth Forms': (char) => char >= 0xFF00 && char <= 0xFFEF + // 'Specials': (char) => char >= 0xFFF0 && char <= 0xFFFF, + // 'Linear B Syllabary': (char) => char >= 0x10000 && char <= 0x1007F, + // 'Linear B Ideograms': (char) => char >= 0x10080 && char <= 0x100FF, + // 'Aegean Numbers': (char) => char >= 0x10100 && char <= 0x1013F, + // 'Ancient Greek Numbers': (char) => char >= 0x10140 && char <= 0x1018F, + // 'Ancient Symbols': (char) => char >= 0x10190 && char <= 0x101CF, + // 'Phaistos Disc': (char) => char >= 0x101D0 && char <= 0x101FF, + // 'Lycian': (char) => char >= 0x10280 && char <= 0x1029F, + // 'Carian': (char) => char >= 0x102A0 && char <= 0x102DF, + // 'Coptic Epact Numbers': (char) => char >= 0x102E0 && char <= 0x102FF, + // 'Old Italic': (char) => char >= 0x10300 && char <= 0x1032F, + // 'Gothic': (char) => char >= 0x10330 && char <= 0x1034F, + // 'Old Permic': (char) => char >= 0x10350 && char <= 0x1037F, + // 'Ugaritic': (char) => char >= 0x10380 && char <= 0x1039F, + // 'Old Persian': (char) => char >= 0x103A0 && char <= 0x103DF, + // 'Deseret': (char) => char >= 0x10400 && char <= 0x1044F, + // 'Shavian': (char) => char >= 0x10450 && char <= 0x1047F, + // 'Osmanya': (char) => char >= 0x10480 && char <= 0x104AF, + // 'Osage': (char) => char >= 0x104B0 && char <= 0x104FF, + // 'Elbasan': (char) => char >= 0x10500 && char <= 0x1052F, + // 'Caucasian Albanian': (char) => char >= 0x10530 && char <= 0x1056F, + // 'Linear A': (char) => char >= 0x10600 && char <= 0x1077F, + // 'Cypriot Syllabary': (char) => char >= 0x10800 && char <= 0x1083F, + // 'Imperial Aramaic': (char) => char >= 0x10840 && char <= 0x1085F, + // 'Palmyrene': (char) => char >= 0x10860 && char <= 0x1087F, + // 'Nabataean': (char) => char >= 0x10880 && char <= 0x108AF, + // 'Hatran': (char) => char >= 0x108E0 && char <= 0x108FF, + // 'Phoenician': (char) => char >= 0x10900 && char <= 0x1091F, + // 'Lydian': (char) => char >= 0x10920 && char <= 0x1093F, + // 'Meroitic Hieroglyphs': (char) => char >= 0x10980 && char <= 0x1099F, + // 'Meroitic Cursive': (char) => char >= 0x109A0 && char <= 0x109FF, + // 'Kharoshthi': (char) => char >= 0x10A00 && char <= 0x10A5F, + // 'Old South Arabian': (char) => char >= 0x10A60 && char <= 0x10A7F, + // 'Old North Arabian': (char) => char >= 0x10A80 && char <= 0x10A9F, + // 'Manichaean': (char) => char >= 0x10AC0 && char <= 0x10AFF, + // 'Avestan': (char) => char >= 0x10B00 && char <= 0x10B3F, + // 'Inscriptional Parthian': (char) => char >= 0x10B40 && char <= 0x10B5F, + // 'Inscriptional Pahlavi': (char) => char >= 0x10B60 && char <= 0x10B7F, + // 'Psalter Pahlavi': (char) => char >= 0x10B80 && char <= 0x10BAF, + // 'Old Turkic': (char) => char >= 0x10C00 && char <= 0x10C4F, + // 'Old Hungarian': (char) => char >= 0x10C80 && char <= 0x10CFF, + // 'Hanifi Rohingya': (char) => char >= 0x10D00 && char <= 0x10D3F, + // 'Rumi Numeral Symbols': (char) => char >= 0x10E60 && char <= 0x10E7F, + // 'Old Sogdian': (char) => char >= 0x10F00 && char <= 0x10F2F, + // 'Sogdian': (char) => char >= 0x10F30 && char <= 0x10F6F, + // 'Elymaic': (char) => char >= 0x10FE0 && char <= 0x10FFF, + // 'Brahmi': (char) => char >= 0x11000 && char <= 0x1107F, + // 'Kaithi': (char) => char >= 0x11080 && char <= 0x110CF, + // 'Sora Sompeng': (char) => char >= 0x110D0 && char <= 0x110FF, + // 'Chakma': (char) => char >= 0x11100 && char <= 0x1114F, + // 'Mahajani': (char) => char >= 0x11150 && char <= 0x1117F, + // 'Sharada': (char) => char >= 0x11180 && char <= 0x111DF, + // 'Sinhala Archaic Numbers': (char) => char >= 0x111E0 && char <= 0x111FF, + // 'Khojki': (char) => char >= 0x11200 && char <= 0x1124F, + // 'Multani': (char) => char >= 0x11280 && char <= 0x112AF, + // 'Khudawadi': (char) => char >= 0x112B0 && char <= 0x112FF, + // 'Grantha': (char) => char >= 0x11300 && char <= 0x1137F, + // 'Newa': (char) => char >= 0x11400 && char <= 0x1147F, + // 'Tirhuta': (char) => char >= 0x11480 && char <= 0x114DF, + // 'Siddham': (char) => char >= 0x11580 && char <= 0x115FF, + // 'Modi': (char) => char >= 0x11600 && char <= 0x1165F, + // 'Mongolian Supplement': (char) => char >= 0x11660 && char <= 0x1167F, + // 'Takri': (char) => char >= 0x11680 && char <= 0x116CF, + // 'Ahom': (char) => char >= 0x11700 && char <= 0x1173F, + // 'Dogra': (char) => char >= 0x11800 && char <= 0x1184F, + // 'Warang Citi': (char) => char >= 0x118A0 && char <= 0x118FF, + // 'Nandinagari': (char) => char >= 0x119A0 && char <= 0x119FF, + // 'Zanabazar Square': (char) => char >= 0x11A00 && char <= 0x11A4F, + // 'Soyombo': (char) => char >= 0x11A50 && char <= 0x11AAF, + // 'Pau Cin Hau': (char) => char >= 0x11AC0 && char <= 0x11AFF, + // 'Bhaiksuki': (char) => char >= 0x11C00 && char <= 0x11C6F, + // 'Marchen': (char) => char >= 0x11C70 && char <= 0x11CBF, + // 'Masaram Gondi': (char) => char >= 0x11D00 && char <= 0x11D5F, + // 'Gunjala Gondi': (char) => char >= 0x11D60 && char <= 0x11DAF, + // 'Makasar': (char) => char >= 0x11EE0 && char <= 0x11EFF, + // 'Tamil Supplement': (char) => char >= 0x11FC0 && char <= 0x11FFF, + // 'Cuneiform': (char) => char >= 0x12000 && char <= 0x123FF, + // 'Cuneiform Numbers and Punctuation': (char) => char >= 0x12400 && char <= 0x1247F, + // 'Early Dynastic Cuneiform': (char) => char >= 0x12480 && char <= 0x1254F, + // 'Egyptian Hieroglyphs': (char) => char >= 0x13000 && char <= 0x1342F, + // 'Egyptian Hieroglyph Format Controls': (char) => char >= 0x13430 && char <= 0x1343F, + // 'Anatolian Hieroglyphs': (char) => char >= 0x14400 && char <= 0x1467F, + // 'Bamum Supplement': (char) => char >= 0x16800 && char <= 0x16A3F, + // 'Mro': (char) => char >= 0x16A40 && char <= 0x16A6F, + // 'Bassa Vah': (char) => char >= 0x16AD0 && char <= 0x16AFF, + // 'Pahawh Hmong': (char) => char >= 0x16B00 && char <= 0x16B8F, + // 'Medefaidrin': (char) => char >= 0x16E40 && char <= 0x16E9F, + // 'Miao': (char) => char >= 0x16F00 && char <= 0x16F9F, + // 'Ideographic Symbols and Punctuation': (char) => char >= 0x16FE0 && char <= 0x16FFF, + // 'Tangut': (char) => char >= 0x17000 && char <= 0x187FF, + // 'Tangut Components': (char) => char >= 0x18800 && char <= 0x18AFF, + // 'Kana Supplement': (char) => char >= 0x1B000 && char <= 0x1B0FF, + // 'Kana Extended-A': (char) => char >= 0x1B100 && char <= 0x1B12F, + // 'Small Kana Extension': (char) => char >= 0x1B130 && char <= 0x1B16F, + // 'Nushu': (char) => char >= 0x1B170 && char <= 0x1B2FF, + // 'Duployan': (char) => char >= 0x1BC00 && char <= 0x1BC9F, + // 'Shorthand Format Controls': (char) => char >= 0x1BCA0 && char <= 0x1BCAF, + // 'Byzantine Musical Symbols': (char) => char >= 0x1D000 && char <= 0x1D0FF, + // 'Musical Symbols': (char) => char >= 0x1D100 && char <= 0x1D1FF, + // 'Ancient Greek Musical Notation': (char) => char >= 0x1D200 && char <= 0x1D24F, + // 'Mayan Numerals': (char) => char >= 0x1D2E0 && char <= 0x1D2FF, + // 'Tai Xuan Jing Symbols': (char) => char >= 0x1D300 && char <= 0x1D35F, + // 'Counting Rod Numerals': (char) => char >= 0x1D360 && char <= 0x1D37F, + // 'Mathematical Alphanumeric Symbols': (char) => char >= 0x1D400 && char <= 0x1D7FF, + // 'Sutton SignWriting': (char) => char >= 0x1D800 && char <= 0x1DAAF, + // 'Glagolitic Supplement': (char) => char >= 0x1E000 && char <= 0x1E02F, + // 'Nyiakeng Puachue Hmong': (char) => char >= 0x1E100 && char <= 0x1E14F, + // 'Wancho': (char) => char >= 0x1E2C0 && char <= 0x1E2FF, + // 'Mende Kikakui': (char) => char >= 0x1E800 && char <= 0x1E8DF, + // 'Adlam': (char) => char >= 0x1E900 && char <= 0x1E95F, + // 'Indic Siyaq Numbers': (char) => char >= 0x1EC70 && char <= 0x1ECBF, + // 'Ottoman Siyaq Numbers': (char) => char >= 0x1ED00 && char <= 0x1ED4F, + // 'Arabic Mathematical Alphabetic Symbols': (char) => char >= 0x1EE00 && char <= 0x1EEFF, + // 'Mahjong Tiles': (char) => char >= 0x1F000 && char <= 0x1F02F, + // 'Domino Tiles': (char) => char >= 0x1F030 && char <= 0x1F09F, + // 'Playing Cards': (char) => char >= 0x1F0A0 && char <= 0x1F0FF, + // 'Enclosed Alphanumeric Supplement': (char) => char >= 0x1F100 && char <= 0x1F1FF, + // 'Enclosed Ideographic Supplement': (char) => char >= 0x1F200 && char <= 0x1F2FF, + // 'Miscellaneous Symbols and Pictographs': (char) => char >= 0x1F300 && char <= 0x1F5FF, + // 'Emoticons': (char) => char >= 0x1F600 && char <= 0x1F64F, + // 'Ornamental Dingbats': (char) => char >= 0x1F650 && char <= 0x1F67F, + // 'Transport and Map Symbols': (char) => char >= 0x1F680 && char <= 0x1F6FF, + // 'Alchemical Symbols': (char) => char >= 0x1F700 && char <= 0x1F77F, + // 'Geometric Shapes Extended': (char) => char >= 0x1F780 && char <= 0x1F7FF, + // 'Supplemental Arrows-C': (char) => char >= 0x1F800 && char <= 0x1F8FF, + // 'Supplemental Symbols and Pictographs': (char) => char >= 0x1F900 && char <= 0x1F9FF, + // 'Chess Symbols': (char) => char >= 0x1FA00 && char <= 0x1FA6F, + // 'Symbols and Pictographs Extended-A': (char) => char >= 0x1FA70 && char <= 0x1FAFF, + // 'CJK Unified Ideographs Extension B': (char) => char >= 0x20000 && char <= 0x2A6DF, + // 'CJK Unified Ideographs Extension C': (char) => char >= 0x2A700 && char <= 0x2B73F, + // 'CJK Unified Ideographs Extension D': (char) => char >= 0x2B740 && char <= 0x2B81F, + // 'CJK Unified Ideographs Extension E': (char) => char >= 0x2B820 && char <= 0x2CEAF, + // 'CJK Unified Ideographs Extension F': (char) => char >= 0x2CEB0 && char <= 0x2EBEF, + // 'CJK Compatibility Ideographs Supplement': (char) => char >= 0x2F800 && char <= 0x2FA1F, + // 'Tags': (char) => char >= 0xE0000 && char <= 0xE007F, + // 'Variation Selectors Supplement': (char) => char >= 0xE0100 && char <= 0xE01EF, + // 'Supplementary Private Use Area-A': (char) => char >= 0xF0000 && char <= 0xFFFFF, + // 'Supplementary Private Use Area-B': (char) => char >= 0x100000 && char <= 0x10FFFF, +}; + +/* eslint-disable new-cap */ +function allowsIdeographicBreaking(chars) { + for (const char of chars) { + if (!charAllowsIdeographicBreaking(char.charCodeAt(0))) + return false; + } + return true; +} +function allowsVerticalWritingMode(chars) { + for (const char of chars) { + if (charHasUprightVerticalOrientation(char.charCodeAt(0))) + return true; + } + return false; +} +function allowsLetterSpacing(chars) { + for (const char of chars) { + if (!charAllowsLetterSpacing(char.charCodeAt(0))) + return false; + } + return true; +} +function charAllowsLetterSpacing(char) { + if (unicodeBlockLookup['Arabic'](char)) + return false; + if (unicodeBlockLookup['Arabic Supplement'](char)) + return false; + if (unicodeBlockLookup['Arabic Extended-A'](char)) + return false; + if (unicodeBlockLookup['Arabic Presentation Forms-A'](char)) + return false; + if (unicodeBlockLookup['Arabic Presentation Forms-B'](char)) + return false; + return true; +} +function charAllowsIdeographicBreaking(char) { + // Return early for characters outside all ideographic ranges. + if (char < 0x2E80) + return false; + if (unicodeBlockLookup['Bopomofo Extended'](char)) + return true; + if (unicodeBlockLookup['Bopomofo'](char)) + return true; + if (unicodeBlockLookup['CJK Compatibility Forms'](char)) + return true; + if (unicodeBlockLookup['CJK Compatibility Ideographs'](char)) + return true; + if (unicodeBlockLookup['CJK Compatibility'](char)) + return true; + if (unicodeBlockLookup['CJK Radicals Supplement'](char)) + return true; + if (unicodeBlockLookup['CJK Strokes'](char)) + return true; + if (unicodeBlockLookup['CJK Symbols and Punctuation'](char)) + return true; + if (unicodeBlockLookup['CJK Unified Ideographs Extension A'](char)) + return true; + if (unicodeBlockLookup['CJK Unified Ideographs'](char)) + return true; + if (unicodeBlockLookup['Enclosed CJK Letters and Months'](char)) + return true; + if (unicodeBlockLookup['Halfwidth and Fullwidth Forms'](char)) + return true; + if (unicodeBlockLookup['Hiragana'](char)) + return true; + if (unicodeBlockLookup['Ideographic Description Characters'](char)) + return true; + if (unicodeBlockLookup['Kangxi Radicals'](char)) + return true; + if (unicodeBlockLookup['Katakana Phonetic Extensions'](char)) + return true; + if (unicodeBlockLookup['Katakana'](char)) + return true; + if (unicodeBlockLookup['Vertical Forms'](char)) + return true; + if (unicodeBlockLookup['Yi Radicals'](char)) + return true; + if (unicodeBlockLookup['Yi Syllables'](char)) + return true; + return false; +} +// The following logic comes from +// . +// Keep it synchronized with +// . +// The data file denotes with “U” or “Tu” any codepoint that may be drawn +// upright in vertical text but does not distinguish between upright and +// “neutral” characters. +// Blocks in the Unicode supplementary planes are excluded from this module due +// to . +/** + * Returns true if the given Unicode codepoint identifies a character with + * upright orientation. + * + * A character has upright orientation if it is drawn upright (unrotated) + * whether the line is oriented horizontally or vertically, even if both + * adjacent characters can be rotated. For example, a Chinese character is + * always drawn upright. An uprightly oriented character causes an adjacent + * “neutral” character to be drawn upright as well. + */ +function charHasUprightVerticalOrientation(char) { + if (char === 0x02EA /* modifier letter yin departing tone mark */ || + char === 0x02EB /* modifier letter yang departing tone mark */) { + return true; + } + // Return early for characters outside all ranges whose characters remain + // upright in vertical writing mode. + if (char < 0x1100) + return false; + if (unicodeBlockLookup['Bopomofo Extended'](char)) + return true; + if (unicodeBlockLookup['Bopomofo'](char)) + return true; + if (unicodeBlockLookup['CJK Compatibility Forms'](char)) { + if (!((char >= 0xFE49 /* dashed overline */ && char <= 0xFE4F) /* wavy low line */)) { + return true; + } + } + if (unicodeBlockLookup['CJK Compatibility Ideographs'](char)) + return true; + if (unicodeBlockLookup['CJK Compatibility'](char)) + return true; + if (unicodeBlockLookup['CJK Radicals Supplement'](char)) + return true; + if (unicodeBlockLookup['CJK Strokes'](char)) + return true; + if (unicodeBlockLookup['CJK Symbols and Punctuation'](char)) { + if (!((char >= 0x3008 /* left angle bracket */ && char <= 0x3011) /* right black lenticular bracket */) && + !((char >= 0x3014 /* left tortoise shell bracket */ && char <= 0x301F) /* low double prime quotation mark */) && + char !== 0x3030 /* wavy dash */) { + return true; + } + } + if (unicodeBlockLookup['CJK Unified Ideographs Extension A'](char)) + return true; + if (unicodeBlockLookup['CJK Unified Ideographs'](char)) + return true; + if (unicodeBlockLookup['Enclosed CJK Letters and Months'](char)) + return true; + if (unicodeBlockLookup['Hangul Compatibility Jamo'](char)) + return true; + if (unicodeBlockLookup['Hangul Jamo Extended-A'](char)) + return true; + if (unicodeBlockLookup['Hangul Jamo Extended-B'](char)) + return true; + if (unicodeBlockLookup['Hangul Jamo'](char)) + return true; + if (unicodeBlockLookup['Hangul Syllables'](char)) + return true; + if (unicodeBlockLookup['Hiragana'](char)) + return true; + if (unicodeBlockLookup['Ideographic Description Characters'](char)) + return true; + if (unicodeBlockLookup['Kanbun'](char)) + return true; + if (unicodeBlockLookup['Kangxi Radicals'](char)) + return true; + if (unicodeBlockLookup['Katakana Phonetic Extensions'](char)) + return true; + if (unicodeBlockLookup['Katakana'](char)) { + if (char !== 0x30FC /* katakana-hiragana prolonged sound mark */) { + return true; + } + } + if (unicodeBlockLookup['Halfwidth and Fullwidth Forms'](char)) { + if (char !== 0xFF08 /* fullwidth left parenthesis */ && + char !== 0xFF09 /* fullwidth right parenthesis */ && + char !== 0xFF0D /* fullwidth hyphen-minus */ && + !((char >= 0xFF1A /* fullwidth colon */ && char <= 0xFF1E) /* fullwidth greater-than sign */) && + char !== 0xFF3B /* fullwidth left square bracket */ && + char !== 0xFF3D /* fullwidth right square bracket */ && + char !== 0xFF3F /* fullwidth low line */ && + !(char >= 0xFF5B /* fullwidth left curly bracket */ && char <= 0xFFDF) && + char !== 0xFFE3 /* fullwidth macron */ && + !(char >= 0xFFE8 /* halfwidth forms light vertical */ && char <= 0xFFEF)) { + return true; + } + } + if (unicodeBlockLookup['Small Form Variants'](char)) { + if (!((char >= 0xFE58 /* small em dash */ && char <= 0xFE5E) /* small right tortoise shell bracket */) && + !((char >= 0xFE63 /* small hyphen-minus */ && char <= 0xFE66) /* small equals sign */)) { + return true; + } + } + if (unicodeBlockLookup['Unified Canadian Aboriginal Syllabics'](char)) + return true; + if (unicodeBlockLookup['Unified Canadian Aboriginal Syllabics Extended'](char)) + return true; + if (unicodeBlockLookup['Vertical Forms'](char)) + return true; + if (unicodeBlockLookup['Yijing Hexagram Symbols'](char)) + return true; + if (unicodeBlockLookup['Yi Syllables'](char)) + return true; + if (unicodeBlockLookup['Yi Radicals'](char)) + return true; + return false; +} +/** + * Returns true if the given Unicode codepoint identifies a character with + * neutral orientation. + * + * A character has neutral orientation if it may be drawn rotated or unrotated + * when the line is oriented vertically, depending on the orientation of the + * adjacent characters. For example, along a verticlly oriented line, the vulgar + * fraction ½ is drawn upright among Chinese characters but rotated among Latin + * letters. A neutrally oriented character does not influence whether an + * adjacent character is drawn upright or rotated. + */ +function charHasNeutralVerticalOrientation(char) { + if (unicodeBlockLookup['Latin-1 Supplement'](char)) { + if (char === 0x00A7 /* section sign */ || + char === 0x00A9 /* copyright sign */ || + char === 0x00AE /* registered sign */ || + char === 0x00B1 /* plus-minus sign */ || + char === 0x00BC /* vulgar fraction one quarter */ || + char === 0x00BD /* vulgar fraction one half */ || + char === 0x00BE /* vulgar fraction three quarters */ || + char === 0x00D7 /* multiplication sign */ || + char === 0x00F7 /* division sign */) { + return true; + } + } + if (unicodeBlockLookup['General Punctuation'](char)) { + if (char === 0x2016 /* double vertical line */ || + char === 0x2020 /* dagger */ || + char === 0x2021 /* double dagger */ || + char === 0x2030 /* per mille sign */ || + char === 0x2031 /* per ten thousand sign */ || + char === 0x203B /* reference mark */ || + char === 0x203C /* double exclamation mark */ || + char === 0x2042 /* asterism */ || + char === 0x2047 /* double question mark */ || + char === 0x2048 /* question exclamation mark */ || + char === 0x2049 /* exclamation question mark */ || + char === 0x2051 /* two asterisks aligned vertically */) { + return true; + } + } + if (unicodeBlockLookup['Letterlike Symbols'](char)) + return true; + if (unicodeBlockLookup['Number Forms'](char)) + return true; + if (unicodeBlockLookup['Miscellaneous Technical'](char)) { + if ((char >= 0x2300 /* diameter sign */ && char <= 0x2307 /* wavy line */) || + (char >= 0x230C /* bottom right crop */ && char <= 0x231F /* bottom right corner */) || + (char >= 0x2324 /* up arrowhead between two horizontal bars */ && char <= 0x2328 /* keyboard */) || + char === 0x232B /* erase to the left */ || + (char >= 0x237D /* shouldered open box */ && char <= 0x239A /* clear screen symbol */) || + (char >= 0x23BE /* dentistry symbol light vertical and top right */ && char <= 0x23CD /* square foot */) || + char === 0x23CF /* eject symbol */ || + (char >= 0x23D1 /* metrical breve */ && char <= 0x23DB /* fuse */) || + (char >= 0x23E2 /* white trapezium */ && char <= 0x23FF)) { + return true; + } + } + if (unicodeBlockLookup['Control Pictures'](char) && char !== 0x2423 /* open box */) + return true; + if (unicodeBlockLookup['Optical Character Recognition'](char)) + return true; + if (unicodeBlockLookup['Enclosed Alphanumerics'](char)) + return true; + if (unicodeBlockLookup['Geometric Shapes'](char)) + return true; + if (unicodeBlockLookup['Miscellaneous Symbols'](char)) { + if (!((char >= 0x261A /* black left pointing index */ && char <= 0x261F) /* white down pointing index */)) { + return true; + } + } + if (unicodeBlockLookup['Miscellaneous Symbols and Arrows'](char)) { + if ((char >= 0x2B12 /* square with top half black */ && char <= 0x2B2F /* white vertical ellipse */) || + (char >= 0x2B50 /* white medium star */ && char <= 0x2B59 /* heavy circled saltire */) || + (char >= 0x2BB8 /* upwards white arrow from bar with horizontal bar */ && char <= 0x2BEB)) { + return true; + } + } + if (unicodeBlockLookup['CJK Symbols and Punctuation'](char)) + return true; + if (unicodeBlockLookup['Katakana'](char)) + return true; + if (unicodeBlockLookup['Private Use Area'](char)) + return true; + if (unicodeBlockLookup['CJK Compatibility Forms'](char)) + return true; + if (unicodeBlockLookup['Small Form Variants'](char)) + return true; + if (unicodeBlockLookup['Halfwidth and Fullwidth Forms'](char)) + return true; + if (char === 0x221E /* infinity */ || + char === 0x2234 /* therefore */ || + char === 0x2235 /* because */ || + (char >= 0x2700 /* black safety scissors */ && char <= 0x2767 /* rotated floral heart bullet */) || + (char >= 0x2776 /* dingbat negative circled digit one */ && char <= 0x2793 /* dingbat negative circled sans-serif number ten */) || + char === 0xFFFC /* object replacement character */ || + char === 0xFFFD /* replacement character */) { + return true; + } + return false; +} +/** + * Returns true if the given Unicode codepoint identifies a character with + * rotated orientation. + * + * A character has rotated orientation if it is drawn rotated when the line is + * oriented vertically, even if both adjacent characters are upright. For + * example, a Latin letter is drawn rotated along a vertical line. A rotated + * character causes an adjacent “neutral” character to be drawn rotated as well. + */ +function charHasRotatedVerticalOrientation(char) { + return !(charHasUprightVerticalOrientation(char) || + charHasNeutralVerticalOrientation(char)); +} +function charInComplexShapingScript(char) { + return unicodeBlockLookup['Arabic'](char) || + unicodeBlockLookup['Arabic Supplement'](char) || + unicodeBlockLookup['Arabic Extended-A'](char) || + unicodeBlockLookup['Arabic Presentation Forms-A'](char) || + unicodeBlockLookup['Arabic Presentation Forms-B'](char); +} +function charInRTLScript(char) { + // Main blocks for Hebrew, Arabic, Thaana and other RTL scripts + return (char >= 0x0590 && char <= 0x08FF) || + unicodeBlockLookup['Arabic Presentation Forms-A'](char) || + unicodeBlockLookup['Arabic Presentation Forms-B'](char); +} +function charInSupportedScript(char, canRenderRTL) { + // This is a rough heuristic: whether we "can render" a script + // actually depends on the properties of the font being used + // and whether differences from the ideal rendering are considered + // semantically significant. + // Even in Latin script, we "can't render" combinations such as the fi + // ligature, but we don't consider that semantically significant. + if (!canRenderRTL && charInRTLScript(char)) { + return false; + } + if ((char >= 0x0900 && char <= 0x0DFF) || + // Main blocks for Indic scripts and Sinhala + (char >= 0x0F00 && char <= 0x109F) || + // Main blocks for Tibetan and Myanmar + unicodeBlockLookup['Khmer'](char)) { + // These blocks cover common scripts that require + // complex text shaping, based on unicode script metadata: + // http://www.unicode.org/repos/cldr/trunk/common/properties/scriptMetadata.txt + // where "Web Rank <= 32" "Shaping Required = YES" + return false; + } + return true; +} +function stringContainsRTLText(chars) { + for (const char of chars) { + if (charInRTLScript(char.charCodeAt(0))) { + return true; + } + } + return false; +} +function isStringInSupportedScript(chars, canRenderRTL) { + for (const char of chars) { + if (!charInSupportedScript(char.charCodeAt(0), canRenderRTL)) { + return false; + } + } + return true; +} + +const status = { + unavailable: 'unavailable', + deferred: 'deferred', + loading: 'loading', + loaded: 'loaded', + error: 'error' +}; +let _completionCallback = null; +//Variables defining the current state of the plugin +let pluginStatus = status.unavailable; +let pluginURL = null; +const triggerPluginCompletionEvent = function (error) { + // NetworkError's are not correctly reflected by the plugin status which prevents reloading plugin + if (error && typeof error === 'string' && error.indexOf('NetworkError') > -1) { + pluginStatus = status.error; + } + if (_completionCallback) { + _completionCallback(error); + } +}; +function sendPluginStateToWorker() { + evented.fire(new Event('pluginStateChange', { pluginStatus, pluginURL })); +} +const evented = new Evented(); +const getRTLTextPluginStatus = function () { + return pluginStatus; +}; +const registerForPluginStateChange = function (callback) { + // Do an initial sync of the state + callback({ pluginStatus, pluginURL }); + // Listen for all future state changes + evented.on('pluginStateChange', callback); + return callback; +}; +const clearRTLTextPlugin = function () { + pluginStatus = status.unavailable; + pluginURL = null; + _completionCallback = null; +}; +const setRTLTextPlugin = function (url, callback, deferred = false) { + if (pluginStatus === status.deferred || pluginStatus === status.loading || pluginStatus === status.loaded) { + throw new Error('setRTLTextPlugin cannot be called multiple times.'); + } + pluginURL = browser.resolveURL(url); + pluginStatus = status.deferred; + _completionCallback = callback; + sendPluginStateToWorker(); + //Start downloading the plugin immediately if not intending to lazy-load + if (!deferred) { + downloadRTLTextPlugin(); + } +}; +const downloadRTLTextPlugin = function () { + if (pluginStatus !== status.deferred || !pluginURL) { + throw new Error('rtl-text-plugin cannot be downloaded unless a pluginURL is specified'); + } + pluginStatus = status.loading; + sendPluginStateToWorker(); + if (pluginURL) { + getArrayBuffer({ url: pluginURL }, (error) => { + if (error) { + triggerPluginCompletionEvent(error); + } + else { + pluginStatus = status.loaded; + sendPluginStateToWorker(); + } + }); + } +}; +const plugin = { + applyArabicShaping: null, + processBidirectionalText: null, + processStyledBidirectionalText: null, + isLoaded() { + return pluginStatus === status.loaded || // Main Thread: loaded if the completion callback returned successfully + plugin.applyArabicShaping != null; // Web-worker: loaded if the plugin functions have been compiled + }, + isLoading() { + return pluginStatus === status.loading; + }, + setState(state) { + if (!isWorker()) + throw new Error('Cannot set the state of the rtl-text-plugin when not in the web-worker context'); + pluginStatus = state.pluginStatus; + pluginURL = state.pluginURL; + }, + isParsed() { + if (!isWorker()) + throw new Error('rtl-text-plugin is only parsed on the worker-threads'); + return plugin.applyArabicShaping != null && + plugin.processBidirectionalText != null && + plugin.processStyledBidirectionalText != null; + }, + getPluginURL() { + if (!isWorker()) + throw new Error('rtl-text-plugin url can only be queried from the worker threads'); + return pluginURL; + } +}; +const lazyLoadRTLTextPlugin = function () { + if (!plugin.isLoading() && + !plugin.isLoaded() && + getRTLTextPluginStatus() === 'deferred') { + downloadRTLTextPlugin(); + } +}; + +/** + * @internal + * A parameter that can be evaluated to a value + */ +class EvaluationParameters { + // "options" may also be another EvaluationParameters to copy, see CrossFadedProperty.possiblyEvaluate + constructor(zoom, options) { + this.zoom = zoom; + if (options) { + this.now = options.now; + this.fadeDuration = options.fadeDuration; + this.zoomHistory = options.zoomHistory; + this.transition = options.transition; + } + else { + this.now = 0; + this.fadeDuration = 0; + this.zoomHistory = new ZoomHistory(); + this.transition = {}; + } + } + isSupportedScript(str) { + return isStringInSupportedScript(str, plugin.isLoaded()); + } + crossFadingFactor() { + if (this.fadeDuration === 0) { + return 1; + } + else { + return Math.min((this.now - this.zoomHistory.lastIntegerZoomTime) / this.fadeDuration, 1); + } + } + getCrossfadeParameters() { + const z = this.zoom; + const fraction = z - Math.floor(z); + const t = this.crossFadingFactor(); + return z > this.zoomHistory.lastIntegerZoom ? + { fromScale: 2, toScale: 1, t: fraction + (1 - fraction) * t } : + { fromScale: 0.5, toScale: 1, t: 1 - (1 - t) * fraction }; + } +} + +/** + * @internal + * `PropertyValue` represents the value part of a property key-value unit. It's used to represent both + * paint and layout property values, and regardless of whether or not their property supports data-driven + * expressions. + * + * `PropertyValue` stores the raw input value as seen in a style or a runtime styling API call, i.e. one of the + * following: + * + * * A constant value of the type appropriate for the property + * * A function which produces a value of that type (but functions are quasi-deprecated in favor of expressions) + * * An expression which produces a value of that type + * * "undefined"/"not present", in which case the property is assumed to take on its default value. + * + * In addition to storing the original input value, `PropertyValue` also stores a normalized representation, + * effectively treating functions as if they are expressions, and constant or default values as if they are + * (constant) expressions. + */ +class PropertyValue { + constructor(property, value) { + this.property = property; + this.value = value; + this.expression = normalizePropertyExpression(value === undefined ? property.specification.default : value, property.specification); + } + isDataDriven() { + return this.expression.kind === 'source' || this.expression.kind === 'composite'; + } + possiblyEvaluate(parameters, canonical, availableImages) { + return this.property.possiblyEvaluate(this, parameters, canonical, availableImages); + } +} +/** + * @internal + * Paint properties are _transitionable_: they can change in a fluid manner, interpolating or cross-fading between + * old and new value. The duration of the transition, and the delay before it begins, is configurable. + * + * `TransitionablePropertyValue` is a compositional class that stores both the property value and that transition + * configuration. + * + * A `TransitionablePropertyValue` can calculate the next step in the evaluation chain for paint property values: + * `TransitioningPropertyValue`. + */ +class TransitionablePropertyValue { + constructor(property) { + this.property = property; + this.value = new PropertyValue(property, undefined); + } + transitioned(parameters, prior) { + return new TransitioningPropertyValue(this.property, this.value, prior, extend({}, parameters.transition, this.transition), parameters.now); + } + untransitioned() { + return new TransitioningPropertyValue(this.property, this.value, null, {}, 0); + } +} +/** + * @internal + * `Transitionable` stores a map of all (property name, `TransitionablePropertyValue`) pairs for paint properties of a + * given layer type. It can calculate the `TransitioningPropertyValue`s for all of them at once, producing a + * `Transitioning` instance for the same set of properties. + */ +class Transitionable { + constructor(properties) { + this._properties = properties; + this._values = Object.create(properties.defaultTransitionablePropertyValues); + } + getValue(name) { + return clone$9(this._values[name].value.value); + } + setValue(name, value) { + if (!Object.prototype.hasOwnProperty.call(this._values, name)) { + this._values[name] = new TransitionablePropertyValue(this._values[name].property); + } + // Note that we do not _remove_ an own property in the case where a value is being reset + // to the default: the transition might still be non-default. + this._values[name].value = new PropertyValue(this._values[name].property, value === null ? undefined : clone$9(value)); + } + getTransition(name) { + return clone$9(this._values[name].transition); + } + setTransition(name, value) { + if (!Object.prototype.hasOwnProperty.call(this._values, name)) { + this._values[name] = new TransitionablePropertyValue(this._values[name].property); + } + this._values[name].transition = clone$9(value) || undefined; + } + serialize() { + const result = {}; + for (const property of Object.keys(this._values)) { + const value = this.getValue(property); + if (value !== undefined) { + result[property] = value; + } + const transition = this.getTransition(property); + if (transition !== undefined) { + result[`${property}-transition`] = transition; + } + } + return result; + } + transitioned(parameters, prior) { + const result = new Transitioning(this._properties); + for (const property of Object.keys(this._values)) { + result._values[property] = this._values[property].transitioned(parameters, prior._values[property]); + } + return result; + } + untransitioned() { + const result = new Transitioning(this._properties); + for (const property of Object.keys(this._values)) { + result._values[property] = this._values[property].untransitioned(); + } + return result; + } +} +/** + * @internal + * `TransitioningPropertyValue` implements the first of two intermediate steps in the evaluation chain of a paint + * property value. In this step, transitions between old and new values are handled: as long as the transition is in + * progress, `TransitioningPropertyValue` maintains a reference to the prior value, and interpolates between it and + * the new value based on the current time and the configured transition duration and delay. The product is the next + * step in the evaluation chain: the "possibly evaluated" result type `R`. See below for more on this concept. + */ +class TransitioningPropertyValue { + constructor(property, value, prior, transition, now) { + this.property = property; + this.value = value; + this.begin = now + transition.delay || 0; + this.end = this.begin + transition.duration || 0; + if (property.specification.transition && (transition.delay || transition.duration)) { + this.prior = prior; + } + } + possiblyEvaluate(parameters, canonical, availableImages) { + const now = parameters.now || 0; + const finalValue = this.value.possiblyEvaluate(parameters, canonical, availableImages); + const prior = this.prior; + if (!prior) { + // No prior value. + return finalValue; + } + else if (now > this.end) { + // Transition from prior value is now complete. + this.prior = null; + return finalValue; + } + else if (this.value.isDataDriven()) { + // Transitions to data-driven properties are not supported. + // We snap immediately to the data-driven value so that, when we perform layout, + // we see the data-driven function and can use it to populate vertex buffers. + this.prior = null; + return finalValue; + } + else if (now < this.begin) { + // Transition hasn't started yet. + return prior.possiblyEvaluate(parameters, canonical, availableImages); + } + else { + // Interpolate between recursively-calculated prior value and final. + const t = (now - this.begin) / (this.end - this.begin); + return this.property.interpolate(prior.possiblyEvaluate(parameters, canonical, availableImages), finalValue, easeCubicInOut(t)); + } + } +} +/** + * @internal + * `Transitioning` stores a map of all (property name, `TransitioningPropertyValue`) pairs for paint properties of a + * given layer type. It can calculate the possibly-evaluated values for all of them at once, producing a + * `PossiblyEvaluated` instance for the same set of properties. + */ +class Transitioning { + constructor(properties) { + this._properties = properties; + this._values = Object.create(properties.defaultTransitioningPropertyValues); + } + possiblyEvaluate(parameters, canonical, availableImages) { + const result = new PossiblyEvaluated(this._properties); + for (const property of Object.keys(this._values)) { + result._values[property] = this._values[property].possiblyEvaluate(parameters, canonical, availableImages); + } + return result; + } + hasTransition() { + for (const property of Object.keys(this._values)) { + if (this._values[property].prior) { + return true; + } + } + return false; + } +} +// ------- Layout ------- +/** + * Because layout properties are not transitionable, they have a simpler representation and evaluation chain than + * paint properties: `PropertyValue`s are possibly evaluated, producing possibly evaluated values, which are then + * fully evaluated. + * + * `Layout` stores a map of all (property name, `PropertyValue`) pairs for layout properties of a + * given layer type. It can calculate the possibly-evaluated values for all of them at once, producing a + * `PossiblyEvaluated` instance for the same set of properties. + */ +class Layout { + constructor(properties) { + this._properties = properties; + this._values = Object.create(properties.defaultPropertyValues); + } + hasValue(name) { + return this._values[name].value !== undefined; + } + getValue(name) { + return clone$9(this._values[name].value); + } + setValue(name, value) { + this._values[name] = new PropertyValue(this._values[name].property, value === null ? undefined : clone$9(value)); + } + serialize() { + const result = {}; + for (const property of Object.keys(this._values)) { + const value = this.getValue(property); + if (value !== undefined) { + result[property] = value; + } + } + return result; + } + possiblyEvaluate(parameters, canonical, availableImages) { + const result = new PossiblyEvaluated(this._properties); + for (const property of Object.keys(this._values)) { + result._values[property] = this._values[property].possiblyEvaluate(parameters, canonical, availableImages); + } + return result; + } +} +/** + * @internal + * `PossiblyEvaluatedPropertyValue` is used for data-driven paint and layout property values. It holds a + * `PossiblyEvaluatedValue` and the `GlobalProperties` that were used to generate it. You're not allowed to supply + * a different set of `GlobalProperties` when performing the final evaluation because they would be ignored in the + * case where the input value was a constant or camera function. + */ +class PossiblyEvaluatedPropertyValue { + constructor(property, value, parameters) { + this.property = property; + this.value = value; + this.parameters = parameters; + } + isConstant() { + return this.value.kind === 'constant'; + } + constantOr(value) { + if (this.value.kind === 'constant') { + return this.value.value; + } + else { + return value; + } + } + evaluate(feature, featureState, canonical, availableImages) { + return this.property.evaluate(this.value, this.parameters, feature, featureState, canonical, availableImages); + } +} +/** + * @internal + * `PossiblyEvaluated` stores a map of all (property name, `R`) pairs for paint or layout properties of a + * given layer type. + */ +class PossiblyEvaluated { + constructor(properties) { + this._properties = properties; + this._values = Object.create(properties.defaultPossiblyEvaluatedValues); + } + get(name) { + return this._values[name]; + } +} +/** + * @internal + * An implementation of `Property` for properties that do not permit data-driven (source or composite) expressions. + * This restriction allows us to declare statically that the result of possibly evaluating this kind of property + * is in fact always the scalar type `T`, and can be used without further evaluating the value on a per-feature basis. + */ +class DataConstantProperty { + constructor(specification) { + this.specification = specification; + } + possiblyEvaluate(value, parameters) { + if (value.isDataDriven()) + throw new Error('Value should not be data driven'); + return value.expression.evaluate(parameters); + } + interpolate(a, b, t) { + const interpolationType = this.specification.type; + const interpolationFn = interpolate[interpolationType]; + if (interpolationFn) { + return interpolationFn(a, b, t); + } + else { + return a; + } + } +} +/** + * @internal + * An implementation of `Property` for properties that permit data-driven (source or composite) expressions. + * The result of possibly evaluating this kind of property is `PossiblyEvaluatedPropertyValue`; obtaining + * a scalar value `T` requires further evaluation on a per-feature basis. + */ +class DataDrivenProperty { + constructor(specification, overrides) { + this.specification = specification; + this.overrides = overrides; + } + possiblyEvaluate(value, parameters, canonical, availableImages) { + if (value.expression.kind === 'constant' || value.expression.kind === 'camera') { + return new PossiblyEvaluatedPropertyValue(this, { kind: 'constant', value: value.expression.evaluate(parameters, null, {}, canonical, availableImages) }, parameters); + } + else { + return new PossiblyEvaluatedPropertyValue(this, value.expression, parameters); + } + } + interpolate(a, b, t) { + // If either possibly-evaluated value is non-constant, give up: we aren't able to interpolate data-driven values. + if (a.value.kind !== 'constant' || b.value.kind !== 'constant') { + return a; + } + // Special case hack solely for fill-outline-color. The undefined value is subsequently handled in + // FillStyleLayer#recalculate, which sets fill-outline-color to the fill-color value if the former + // is a PossiblyEvaluatedPropertyValue containing a constant undefined value. In addition to the + // return value here, the other source of a PossiblyEvaluatedPropertyValue containing a constant + // undefined value is the "default value" for fill-outline-color held in + // `Properties#defaultPossiblyEvaluatedValues`, which serves as the prototype of + // `PossiblyEvaluated#_values`. + if (a.value.value === undefined || b.value.value === undefined) { + return new PossiblyEvaluatedPropertyValue(this, { kind: 'constant', value: undefined }, a.parameters); + } + const interpolationType = this.specification.type; + const interpolationFn = interpolate[interpolationType]; + if (interpolationFn) { + const interpolatedValue = interpolationFn(a.value.value, b.value.value, t); + return new PossiblyEvaluatedPropertyValue(this, { kind: 'constant', value: interpolatedValue }, a.parameters); + } + else { + return a; + } + } + evaluate(value, parameters, feature, featureState, canonical, availableImages) { + if (value.kind === 'constant') { + return value.value; + } + else { + return value.evaluate(parameters, feature, featureState, canonical, availableImages); + } + } +} +/** + * @internal + * An implementation of `Property` for data driven `line-pattern` which are transitioned by cross-fading + * rather than interpolation. + */ +class CrossFadedDataDrivenProperty extends DataDrivenProperty { + possiblyEvaluate(value, parameters, canonical, availableImages) { + if (value.value === undefined) { + return new PossiblyEvaluatedPropertyValue(this, { kind: 'constant', value: undefined }, parameters); + } + else if (value.expression.kind === 'constant') { + const evaluatedValue = value.expression.evaluate(parameters, null, {}, canonical, availableImages); + const isImageExpression = value.property.specification.type === 'resolvedImage'; + const constantValue = isImageExpression && typeof evaluatedValue !== 'string' ? evaluatedValue.name : evaluatedValue; + const constant = this._calculate(constantValue, constantValue, constantValue, parameters); + return new PossiblyEvaluatedPropertyValue(this, { kind: 'constant', value: constant }, parameters); + } + else if (value.expression.kind === 'camera') { + const cameraVal = this._calculate(value.expression.evaluate({ zoom: parameters.zoom - 1.0 }), value.expression.evaluate({ zoom: parameters.zoom }), value.expression.evaluate({ zoom: parameters.zoom + 1.0 }), parameters); + return new PossiblyEvaluatedPropertyValue(this, { kind: 'constant', value: cameraVal }, parameters); + } + else { + // source or composite expression + return new PossiblyEvaluatedPropertyValue(this, value.expression, parameters); + } + } + evaluate(value, globals, feature, featureState, canonical, availableImages) { + if (value.kind === 'source') { + const constant = value.evaluate(globals, feature, featureState, canonical, availableImages); + return this._calculate(constant, constant, constant, globals); + } + else if (value.kind === 'composite') { + return this._calculate(value.evaluate({ zoom: Math.floor(globals.zoom) - 1.0 }, feature, featureState), value.evaluate({ zoom: Math.floor(globals.zoom) }, feature, featureState), value.evaluate({ zoom: Math.floor(globals.zoom) + 1.0 }, feature, featureState), globals); + } + else { + return value.value; + } + } + _calculate(min, mid, max, parameters) { + const z = parameters.zoom; + return z > parameters.zoomHistory.lastIntegerZoom ? { from: min, to: mid } : { from: max, to: mid }; + } + interpolate(a) { + return a; + } +} +/** + * @internal + * An implementation of `Property` for `*-pattern` and `line-dasharray`, which are transitioned by cross-fading + * rather than interpolation. + */ +class CrossFadedProperty { + constructor(specification) { + this.specification = specification; + } + possiblyEvaluate(value, parameters, canonical, availableImages) { + if (value.value === undefined) { + return undefined; + } + else if (value.expression.kind === 'constant') { + const constant = value.expression.evaluate(parameters, null, {}, canonical, availableImages); + return this._calculate(constant, constant, constant, parameters); + } + else { + return this._calculate(value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom - 1.0), parameters)), value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom), parameters)), value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom + 1.0), parameters)), parameters); + } + } + _calculate(min, mid, max, parameters) { + const z = parameters.zoom; + return z > parameters.zoomHistory.lastIntegerZoom ? { from: min, to: mid } : { from: max, to: mid }; + } + interpolate(a) { + return a; + } +} +/** + * @internal + * An implementation of `Property` for `heatmap-color` and `line-gradient`. Interpolation is a no-op, and + * evaluation returns a boolean value in order to indicate its presence, but the real + * evaluation happens in StyleLayer classes. + */ +class ColorRampProperty { + constructor(specification) { + this.specification = specification; + } + possiblyEvaluate(value, parameters, canonical, availableImages) { + return !!value.expression.evaluate(parameters, null, {}, canonical, availableImages); + } + interpolate() { return false; } +} +/** + * @internal + * `Properties` holds objects containing default values for the layout or paint property set of a given + * layer type. These objects are immutable, and they are used as the prototypes for the `_values` members of + * `Transitionable`, `Transitioning`, `Layout`, and `PossiblyEvaluated`. This allows these classes to avoid + * doing work in the common case where a property has no explicit value set and should be considered to take + * on the default value: using `for (const property of Object.keys(this._values))`, they can iterate over + * only the _own_ properties of `_values`, skipping repeated calculation of transitions and possible/final + * evaluations for defaults, the result of which will always be the same. + */ +class Properties { + constructor(properties) { + this.properties = properties; + this.defaultPropertyValues = {}; + this.defaultTransitionablePropertyValues = {}; + this.defaultTransitioningPropertyValues = {}; + this.defaultPossiblyEvaluatedValues = {}; + this.overridableProperties = []; + for (const property in properties) { + const prop = properties[property]; + if (prop.specification.overridable) { + this.overridableProperties.push(property); + } + const defaultPropertyValue = this.defaultPropertyValues[property] = + new PropertyValue(prop, undefined); + const defaultTransitionablePropertyValue = this.defaultTransitionablePropertyValues[property] = + new TransitionablePropertyValue(prop); + this.defaultTransitioningPropertyValues[property] = + defaultTransitionablePropertyValue.untransitioned(); + this.defaultPossiblyEvaluatedValues[property] = + defaultPropertyValue.possiblyEvaluate({}); + } + } +} +register('DataDrivenProperty', DataDrivenProperty); +register('DataConstantProperty', DataConstantProperty); +register('CrossFadedDataDrivenProperty', CrossFadedDataDrivenProperty); +register('CrossFadedProperty', CrossFadedProperty); +register('ColorRampProperty', ColorRampProperty); + +const TRANSITION_SUFFIX = '-transition'; +/** + * A base class for style layers + */ +class StyleLayer extends Evented { + constructor(layer, properties) { + super(); + this.id = layer.id; + this.type = layer.type; + this._featureFilter = { filter: () => true, needGeometry: false }; + if (layer.type === 'custom') + return; + layer = layer; + this.metadata = layer.metadata; + this.minzoom = layer.minzoom; + this.maxzoom = layer.maxzoom; + if (layer.type !== 'background') { + this.source = layer.source; + this.sourceLayer = layer['source-layer']; + this.filter = layer.filter; + } + if (properties.layout) { + this._unevaluatedLayout = new Layout(properties.layout); + } + if (properties.paint) { + this._transitionablePaint = new Transitionable(properties.paint); + for (const property in layer.paint) { + this.setPaintProperty(property, layer.paint[property], { validate: false }); + } + for (const property in layer.layout) { + this.setLayoutProperty(property, layer.layout[property], { validate: false }); + } + this._transitioningPaint = this._transitionablePaint.untransitioned(); + //$FlowFixMe + this.paint = new PossiblyEvaluated(properties.paint); + } + } + getCrossfadeParameters() { + return this._crossfadeParameters; + } + getLayoutProperty(name) { + if (name === 'visibility') { + return this.visibility; + } + return this._unevaluatedLayout.getValue(name); + } + setLayoutProperty(name, value, options = {}) { + if (value !== null && value !== undefined) { + const key = `layers.${this.id}.layout.${name}`; + if (this._validate(validateLayoutProperty, key, name, value, options)) { + return; + } + } + if (name === 'visibility') { + this.visibility = value; + return; + } + this._unevaluatedLayout.setValue(name, value); + } + getPaintProperty(name) { + if (name.endsWith(TRANSITION_SUFFIX)) { + return this._transitionablePaint.getTransition(name.slice(0, -TRANSITION_SUFFIX.length)); + } + else { + return this._transitionablePaint.getValue(name); + } + } + setPaintProperty(name, value, options = {}) { + if (value !== null && value !== undefined) { + const key = `layers.${this.id}.paint.${name}`; + if (this._validate(validatePaintProperty, key, name, value, options)) { + return false; + } + } + if (name.endsWith(TRANSITION_SUFFIX)) { + this._transitionablePaint.setTransition(name.slice(0, -TRANSITION_SUFFIX.length), value || undefined); + return false; + } + else { + const transitionable = this._transitionablePaint._values[name]; + const isCrossFadedProperty = transitionable.property.specification['property-type'] === 'cross-faded-data-driven'; + const wasDataDriven = transitionable.value.isDataDriven(); + const oldValue = transitionable.value; + this._transitionablePaint.setValue(name, value); + this._handleSpecialPaintPropertyUpdate(name); + const newValue = this._transitionablePaint._values[name].value; + const isDataDriven = newValue.isDataDriven(); + // if a cross-faded value is changed, we need to make sure the new icons get added to each tile's iconAtlas + // so a call to _updateLayer is necessary, and we return true from this function so it gets called in + // Style#setPaintProperty + return isDataDriven || wasDataDriven || isCrossFadedProperty || this._handleOverridablePaintPropertyUpdate(name, oldValue, newValue); + } + } + _handleSpecialPaintPropertyUpdate(_) { + // No-op; can be overridden by derived classes. + } + // eslint-disable-next-line @typescript-eslint/no-unused-vars + _handleOverridablePaintPropertyUpdate(name, oldValue, newValue) { + // No-op; can be overridden by derived classes. + return false; + } + isHidden(zoom) { + if (this.minzoom && zoom < this.minzoom) + return true; + if (this.maxzoom && zoom >= this.maxzoom) + return true; + return this.visibility === 'none'; + } + updateTransitions(parameters) { + this._transitioningPaint = this._transitionablePaint.transitioned(parameters, this._transitioningPaint); + } + hasTransition() { + return this._transitioningPaint.hasTransition(); + } + recalculate(parameters, availableImages) { + if (parameters.getCrossfadeParameters) { + this._crossfadeParameters = parameters.getCrossfadeParameters(); + } + if (this._unevaluatedLayout) { + this.layout = this._unevaluatedLayout.possiblyEvaluate(parameters, undefined, availableImages); + } + this.paint = this._transitioningPaint.possiblyEvaluate(parameters, undefined, availableImages); + } + serialize() { + const output = { + 'id': this.id, + 'type': this.type, + 'source': this.source, + 'source-layer': this.sourceLayer, + 'metadata': this.metadata, + 'minzoom': this.minzoom, + 'maxzoom': this.maxzoom, + 'filter': this.filter, + 'layout': this._unevaluatedLayout && this._unevaluatedLayout.serialize(), + 'paint': this._transitionablePaint && this._transitionablePaint.serialize() + }; + if (this.visibility) { + output.layout = output.layout || {}; + output.layout.visibility = this.visibility; + } + return filterObject(output, (value, key) => { + return value !== undefined && + !(key === 'layout' && !Object.keys(value).length) && + !(key === 'paint' && !Object.keys(value).length); + }); + } + _validate(validate, key, name, value, options = {}) { + if (options && options.validate === false) { + return false; + } + return emitValidationErrors(this, validate.call(validateStyle, { + key, + layerType: this.type, + objectKey: name, + value, + styleSpec: v8Spec, + // Workaround for https://github.com/mapbox/mapbox-gl-js/issues/2407 + style: { glyphs: true, sprite: true } + })); + } + is3D() { + return false; + } + isTileClipped() { + return false; + } + hasOffscreenPass() { + return false; + } + resize() { + // noop + } + isStateDependent() { + for (const property in this.paint._values) { + const value = this.paint.get(property); + if (!(value instanceof PossiblyEvaluatedPropertyValue) || !supportsPropertyExpression(value.property.specification)) { + continue; + } + if ((value.value.kind === 'source' || value.value.kind === 'composite') && + value.value.isStateDependent) { + return true; + } + } + return false; + } +} + +// Note: all "sizes" are measured in bytes +/** + * @internal + * A view type size + */ +const viewTypes = { + 'Int8': Int8Array, + 'Uint8': Uint8Array, + 'Int16': Int16Array, + 'Uint16': Uint16Array, + 'Int32': Int32Array, + 'Uint32': Uint32Array, + 'Float32': Float32Array +}; +/** @internal */ +class Struct { + /** + * @param structArray - The StructArray the struct is stored in + * @param index - The index of the struct in the StructArray. + */ + constructor(structArray, index) { + this._structArray = structArray; + this._pos1 = index * this.size; + this._pos2 = this._pos1 / 2; + this._pos4 = this._pos1 / 4; + this._pos8 = this._pos1 / 8; + } +} +const DEFAULT_CAPACITY = 128; +const RESIZE_MULTIPLIER = 5; +/** + * @internal + * `StructArray` provides an abstraction over `ArrayBuffer` and `TypedArray` + * making it behave like an array of typed structs. + * + * Conceptually, a StructArray is comprised of elements, i.e., instances of its + * associated struct type. Each particular struct type, together with an + * alignment size, determines the memory layout of a StructArray whose elements + * are of that type. Thus, for each such layout that we need, we have + * a corresponding StructArrayLayout class, inheriting from StructArray and + * implementing `emplaceBack()` and `_refreshViews()`. + * + * In some cases, where we need to access particular elements of a StructArray, + * we implement a more specific subclass that inherits from one of the + * StructArrayLayouts and adds a `get(i): T` accessor that returns a structured + * object whose properties are proxies into the underlying memory space for the + * i-th element. This affords the convience of working with (seemingly) plain + * Javascript objects without the overhead of serializing/deserializing them + * into ArrayBuffers for efficient web worker transfer. + */ +class StructArray { + constructor() { + this.isTransferred = false; + this.capacity = -1; + this.resize(0); + } + /** + * Serialize a StructArray instance. Serializes both the raw data and the + * metadata needed to reconstruct the StructArray base class during + * deserialization. + */ + static serialize(array, transferables) { + array._trim(); + if (transferables) { + array.isTransferred = true; + transferables.push(array.arrayBuffer); + } + return { + length: array.length, + arrayBuffer: array.arrayBuffer, + }; + } + static deserialize(input) { + const structArray = Object.create(this.prototype); + structArray.arrayBuffer = input.arrayBuffer; + structArray.length = input.length; + structArray.capacity = input.arrayBuffer.byteLength / structArray.bytesPerElement; + structArray._refreshViews(); + return structArray; + } + /** + * Resize the array to discard unused capacity. + */ + _trim() { + if (this.length !== this.capacity) { + this.capacity = this.length; + this.arrayBuffer = this.arrayBuffer.slice(0, this.length * this.bytesPerElement); + this._refreshViews(); + } + } + /** + * Resets the length of the array to 0 without de-allocating capcacity. + */ + clear() { + this.length = 0; + } + /** + * Resize the array. + * If `n` is greater than the current length then additional elements with undefined values are added. + * If `n` is less than the current length then the array will be reduced to the first `n` elements. + * @param n - The new size of the array. + */ + resize(n) { + this.reserve(n); + this.length = n; + } + /** + * Indicate a planned increase in size, so that any necessary allocation may + * be done once, ahead of time. + * @param n - The expected size of the array. + */ + reserve(n) { + if (n > this.capacity) { + this.capacity = Math.max(n, Math.floor(this.capacity * RESIZE_MULTIPLIER), DEFAULT_CAPACITY); + this.arrayBuffer = new ArrayBuffer(this.capacity * this.bytesPerElement); + const oldUint8Array = this.uint8; + this._refreshViews(); + if (oldUint8Array) + this.uint8.set(oldUint8Array); + } + } + /** + * Create TypedArray views for the current ArrayBuffer. + */ + _refreshViews() { + throw new Error('_refreshViews() must be implemented by each concrete StructArray layout'); + } +} +/** + * Given a list of member fields, create a full StructArrayLayout, in + * particular calculating the correct byte offset for each field. This data + * is used at build time to generate StructArrayLayout_*#emplaceBack() and + * other accessors, and at runtime for binding vertex buffer attributes. + */ +function createLayout(members, alignment = 1) { + let offset = 0; + let maxSize = 0; + const layoutMembers = members.map((member) => { + const typeSize = sizeOf(member.type); + const memberOffset = offset = align$1(offset, Math.max(alignment, typeSize)); + const components = member.components || 1; + maxSize = Math.max(maxSize, typeSize); + offset += typeSize * components; + return { + name: member.name, + type: member.type, + components, + offset: memberOffset, + }; + }); + const size = align$1(offset, Math.max(maxSize, alignment)); + return { + members: layoutMembers, + size, + alignment + }; +} +function sizeOf(type) { + return viewTypes[type].BYTES_PER_ELEMENT; +} +function align$1(offset, size) { + return Math.ceil(offset / size) * size; +} + +// This file is generated. Edit build/generate-struct-arrays.ts, then run `npm run codegen`. +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[2] + * + */ +class StructArrayLayout2i4 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + emplaceBack(v0, v1) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1); + } + emplace(i, v0, v1) { + const o2 = i * 2; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + return i; + } +} +StructArrayLayout2i4.prototype.bytesPerElement = 4; +register('StructArrayLayout2i4', StructArrayLayout2i4); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[3] + * + */ +class StructArrayLayout3i6 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2); + } + emplace(i, v0, v1, v2) { + const o2 = i * 3; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + return i; + } +} +StructArrayLayout3i6.prototype.bytesPerElement = 6; +register('StructArrayLayout3i6', StructArrayLayout3i6); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[4] + * + */ +class StructArrayLayout4i8 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2, v3) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3); + } + emplace(i, v0, v1, v2, v3) { + const o2 = i * 4; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + this.int16[o2 + 3] = v3; + return i; + } +} +StructArrayLayout4i8.prototype.bytesPerElement = 8; +register('StructArrayLayout4i8', StructArrayLayout4i8); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[2] + * [4]: Int16[4] + * + */ +class StructArrayLayout2i4i12 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2, v3, v4, v5) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5); + } + emplace(i, v0, v1, v2, v3, v4, v5) { + const o2 = i * 6; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + this.int16[o2 + 3] = v3; + this.int16[o2 + 4] = v4; + this.int16[o2 + 5] = v5; + return i; + } +} +StructArrayLayout2i4i12.prototype.bytesPerElement = 12; +register('StructArrayLayout2i4i12', StructArrayLayout2i4i12); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[2] + * [4]: Uint8[4] + * + */ +class StructArrayLayout2i4ub8 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2, v3, v4, v5) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5); + } + emplace(i, v0, v1, v2, v3, v4, v5) { + const o2 = i * 4; + const o1 = i * 8; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.uint8[o1 + 4] = v2; + this.uint8[o1 + 5] = v3; + this.uint8[o1 + 6] = v4; + this.uint8[o1 + 7] = v5; + return i; + } +} +StructArrayLayout2i4ub8.prototype.bytesPerElement = 8; +register('StructArrayLayout2i4ub8', StructArrayLayout2i4ub8); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Float32[2] + * + */ +class StructArrayLayout2f8 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + emplaceBack(v0, v1) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1); + } + emplace(i, v0, v1) { + const o4 = i * 2; + this.float32[o4 + 0] = v0; + this.float32[o4 + 1] = v1; + return i; + } +} +StructArrayLayout2f8.prototype.bytesPerElement = 8; +register('StructArrayLayout2f8', StructArrayLayout2f8); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint16[10] + * + */ +class StructArrayLayout10ui20 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2, v3, v4, v5, v6, v7, v8, v9) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9); + } + emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9) { + const o2 = i * 10; + this.uint16[o2 + 0] = v0; + this.uint16[o2 + 1] = v1; + this.uint16[o2 + 2] = v2; + this.uint16[o2 + 3] = v3; + this.uint16[o2 + 4] = v4; + this.uint16[o2 + 5] = v5; + this.uint16[o2 + 6] = v6; + this.uint16[o2 + 7] = v7; + this.uint16[o2 + 8] = v8; + this.uint16[o2 + 9] = v9; + return i; + } +} +StructArrayLayout10ui20.prototype.bytesPerElement = 20; +register('StructArrayLayout10ui20', StructArrayLayout10ui20); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[4] + * [8]: Uint16[4] + * [16]: Int16[4] + * + */ +class StructArrayLayout4i4ui4i24 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11); + } + emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11) { + const o2 = i * 12; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + this.int16[o2 + 3] = v3; + this.uint16[o2 + 4] = v4; + this.uint16[o2 + 5] = v5; + this.uint16[o2 + 6] = v6; + this.uint16[o2 + 7] = v7; + this.int16[o2 + 8] = v8; + this.int16[o2 + 9] = v9; + this.int16[o2 + 10] = v10; + this.int16[o2 + 11] = v11; + return i; + } +} +StructArrayLayout4i4ui4i24.prototype.bytesPerElement = 24; +register('StructArrayLayout4i4ui4i24', StructArrayLayout4i4ui4i24); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Float32[3] + * + */ +class StructArrayLayout3f12 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2); + } + emplace(i, v0, v1, v2) { + const o4 = i * 3; + this.float32[o4 + 0] = v0; + this.float32[o4 + 1] = v1; + this.float32[o4 + 2] = v2; + return i; + } +} +StructArrayLayout3f12.prototype.bytesPerElement = 12; +register('StructArrayLayout3f12', StructArrayLayout3f12); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint32[1] + * + */ +class StructArrayLayout1ul4 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint32 = new Uint32Array(this.arrayBuffer); + } + emplaceBack(v0) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0); + } + emplace(i, v0) { + const o4 = i * 1; + this.uint32[o4 + 0] = v0; + return i; + } +} +StructArrayLayout1ul4.prototype.bytesPerElement = 4; +register('StructArrayLayout1ul4', StructArrayLayout1ul4); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[6] + * [12]: Uint32[1] + * [16]: Uint16[2] + * + */ +class StructArrayLayout6i1ul2ui20 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + this.uint32 = new Uint32Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2, v3, v4, v5, v6, v7, v8) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8); + } + emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8) { + const o2 = i * 10; + const o4 = i * 5; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + this.int16[o2 + 3] = v3; + this.int16[o2 + 4] = v4; + this.int16[o2 + 5] = v5; + this.uint32[o4 + 3] = v6; + this.uint16[o2 + 8] = v7; + this.uint16[o2 + 9] = v8; + return i; + } +} +StructArrayLayout6i1ul2ui20.prototype.bytesPerElement = 20; +register('StructArrayLayout6i1ul2ui20', StructArrayLayout6i1ul2ui20); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[2] + * [4]: Int16[2] + * [8]: Int16[2] + * + */ +class StructArrayLayout2i2i2i12 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2, v3, v4, v5) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5); + } + emplace(i, v0, v1, v2, v3, v4, v5) { + const o2 = i * 6; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + this.int16[o2 + 3] = v3; + this.int16[o2 + 4] = v4; + this.int16[o2 + 5] = v5; + return i; + } +} +StructArrayLayout2i2i2i12.prototype.bytesPerElement = 12; +register('StructArrayLayout2i2i2i12', StructArrayLayout2i2i2i12); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Float32[2] + * [8]: Float32[1] + * [12]: Int16[2] + * + */ +class StructArrayLayout2f1f2i16 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2, v3, v4) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4); + } + emplace(i, v0, v1, v2, v3, v4) { + const o4 = i * 4; + const o2 = i * 8; + this.float32[o4 + 0] = v0; + this.float32[o4 + 1] = v1; + this.float32[o4 + 2] = v2; + this.int16[o2 + 6] = v3; + this.int16[o2 + 7] = v4; + return i; + } +} +StructArrayLayout2f1f2i16.prototype.bytesPerElement = 16; +register('StructArrayLayout2f1f2i16', StructArrayLayout2f1f2i16); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint8[2] + * [4]: Float32[2] + * + */ +class StructArrayLayout2ub2f12 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2, v3) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3); + } + emplace(i, v0, v1, v2, v3) { + const o1 = i * 12; + const o4 = i * 3; + this.uint8[o1 + 0] = v0; + this.uint8[o1 + 1] = v1; + this.float32[o4 + 1] = v2; + this.float32[o4 + 2] = v3; + return i; + } +} +StructArrayLayout2ub2f12.prototype.bytesPerElement = 12; +register('StructArrayLayout2ub2f12', StructArrayLayout2ub2f12); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint16[3] + * + */ +class StructArrayLayout3ui6 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2); + } + emplace(i, v0, v1, v2) { + const o2 = i * 3; + this.uint16[o2 + 0] = v0; + this.uint16[o2 + 1] = v1; + this.uint16[o2 + 2] = v2; + return i; + } +} +StructArrayLayout3ui6.prototype.bytesPerElement = 6; +register('StructArrayLayout3ui6', StructArrayLayout3ui6); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[2] + * [4]: Uint16[2] + * [8]: Uint32[3] + * [20]: Uint16[3] + * [28]: Float32[2] + * [36]: Uint8[3] + * [40]: Uint32[1] + * [44]: Int16[1] + * + */ +class StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + this.uint32 = new Uint32Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16); + } + emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16) { + const o2 = i * 24; + const o4 = i * 12; + const o1 = i * 48; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.uint16[o2 + 2] = v2; + this.uint16[o2 + 3] = v3; + this.uint32[o4 + 2] = v4; + this.uint32[o4 + 3] = v5; + this.uint32[o4 + 4] = v6; + this.uint16[o2 + 10] = v7; + this.uint16[o2 + 11] = v8; + this.uint16[o2 + 12] = v9; + this.float32[o4 + 7] = v10; + this.float32[o4 + 8] = v11; + this.uint8[o1 + 36] = v12; + this.uint8[o1 + 37] = v13; + this.uint8[o1 + 38] = v14; + this.uint32[o4 + 10] = v15; + this.int16[o2 + 22] = v16; + return i; + } +} +StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48.prototype.bytesPerElement = 48; +register('StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48', StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[8] + * [16]: Uint16[15] + * [48]: Uint32[1] + * [52]: Float32[2] + * [60]: Uint16[2] + * + */ +class StructArrayLayout8i15ui1ul2f2ui64 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + this.uint32 = new Uint32Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27); + } + emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27) { + const o2 = i * 32; + const o4 = i * 16; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + this.int16[o2 + 3] = v3; + this.int16[o2 + 4] = v4; + this.int16[o2 + 5] = v5; + this.int16[o2 + 6] = v6; + this.int16[o2 + 7] = v7; + this.uint16[o2 + 8] = v8; + this.uint16[o2 + 9] = v9; + this.uint16[o2 + 10] = v10; + this.uint16[o2 + 11] = v11; + this.uint16[o2 + 12] = v12; + this.uint16[o2 + 13] = v13; + this.uint16[o2 + 14] = v14; + this.uint16[o2 + 15] = v15; + this.uint16[o2 + 16] = v16; + this.uint16[o2 + 17] = v17; + this.uint16[o2 + 18] = v18; + this.uint16[o2 + 19] = v19; + this.uint16[o2 + 20] = v20; + this.uint16[o2 + 21] = v21; + this.uint16[o2 + 22] = v22; + this.uint32[o4 + 12] = v23; + this.float32[o4 + 13] = v24; + this.float32[o4 + 14] = v25; + this.uint16[o2 + 30] = v26; + this.uint16[o2 + 31] = v27; + return i; + } +} +StructArrayLayout8i15ui1ul2f2ui64.prototype.bytesPerElement = 64; +register('StructArrayLayout8i15ui1ul2f2ui64', StructArrayLayout8i15ui1ul2f2ui64); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Float32[1] + * + */ +class StructArrayLayout1f4 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + emplaceBack(v0) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0); + } + emplace(i, v0) { + const o4 = i * 1; + this.float32[o4 + 0] = v0; + return i; + } +} +StructArrayLayout1f4.prototype.bytesPerElement = 4; +register('StructArrayLayout1f4', StructArrayLayout1f4); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint16[1] + * [4]: Float32[2] + * + */ +class StructArrayLayout1ui2f12 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2); + } + emplace(i, v0, v1, v2) { + const o2 = i * 6; + const o4 = i * 3; + this.uint16[o2 + 0] = v0; + this.float32[o4 + 1] = v1; + this.float32[o4 + 2] = v2; + return i; + } +} +StructArrayLayout1ui2f12.prototype.bytesPerElement = 12; +register('StructArrayLayout1ui2f12', StructArrayLayout1ui2f12); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint32[1] + * [4]: Uint16[2] + * + */ +class StructArrayLayout1ul2ui8 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint32 = new Uint32Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2); + } + emplace(i, v0, v1, v2) { + const o4 = i * 2; + const o2 = i * 4; + this.uint32[o4 + 0] = v0; + this.uint16[o2 + 2] = v1; + this.uint16[o2 + 3] = v2; + return i; + } +} +StructArrayLayout1ul2ui8.prototype.bytesPerElement = 8; +register('StructArrayLayout1ul2ui8', StructArrayLayout1ul2ui8); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint16[2] + * + */ +class StructArrayLayout2ui4 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + emplaceBack(v0, v1) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1); + } + emplace(i, v0, v1) { + const o2 = i * 2; + this.uint16[o2 + 0] = v0; + this.uint16[o2 + 1] = v1; + return i; + } +} +StructArrayLayout2ui4.prototype.bytesPerElement = 4; +register('StructArrayLayout2ui4', StructArrayLayout2ui4); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint16[1] + * + */ +class StructArrayLayout1ui2 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + emplaceBack(v0) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0); + } + emplace(i, v0) { + const o2 = i * 1; + this.uint16[o2 + 0] = v0; + return i; + } +} +StructArrayLayout1ui2.prototype.bytesPerElement = 2; +register('StructArrayLayout1ui2', StructArrayLayout1ui2); +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Float32[4] + * + */ +class StructArrayLayout4f16 extends StructArray { + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + emplaceBack(v0, v1, v2, v3) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3); + } + emplace(i, v0, v1, v2, v3) { + const o4 = i * 4; + this.float32[o4 + 0] = v0; + this.float32[o4 + 1] = v1; + this.float32[o4 + 2] = v2; + this.float32[o4 + 3] = v3; + return i; + } +} +StructArrayLayout4f16.prototype.bytesPerElement = 16; +register('StructArrayLayout4f16', StructArrayLayout4f16); +/** @internal */ +class CollisionBoxStruct extends Struct { + get anchorPointX() { return this._structArray.int16[this._pos2 + 0]; } + get anchorPointY() { return this._structArray.int16[this._pos2 + 1]; } + get x1() { return this._structArray.int16[this._pos2 + 2]; } + get y1() { return this._structArray.int16[this._pos2 + 3]; } + get x2() { return this._structArray.int16[this._pos2 + 4]; } + get y2() { return this._structArray.int16[this._pos2 + 5]; } + get featureIndex() { return this._structArray.uint32[this._pos4 + 3]; } + get sourceLayerIndex() { return this._structArray.uint16[this._pos2 + 8]; } + get bucketIndex() { return this._structArray.uint16[this._pos2 + 9]; } + get anchorPoint() { return new Point$2(this.anchorPointX, this.anchorPointY); } +} +CollisionBoxStruct.prototype.size = 20; +/** @internal */ +class CollisionBoxArray extends StructArrayLayout6i1ul2ui20 { + /** + * Return the CollisionBoxStruct at the given location in the array. + * @param index The index of the element. + */ + get(index) { + return new CollisionBoxStruct(this, index); + } +} +register('CollisionBoxArray', CollisionBoxArray); +/** @internal */ +class PlacedSymbolStruct extends Struct { + get anchorX() { return this._structArray.int16[this._pos2 + 0]; } + get anchorY() { return this._structArray.int16[this._pos2 + 1]; } + get glyphStartIndex() { return this._structArray.uint16[this._pos2 + 2]; } + get numGlyphs() { return this._structArray.uint16[this._pos2 + 3]; } + get vertexStartIndex() { return this._structArray.uint32[this._pos4 + 2]; } + get lineStartIndex() { return this._structArray.uint32[this._pos4 + 3]; } + get lineLength() { return this._structArray.uint32[this._pos4 + 4]; } + get segment() { return this._structArray.uint16[this._pos2 + 10]; } + get lowerSize() { return this._structArray.uint16[this._pos2 + 11]; } + get upperSize() { return this._structArray.uint16[this._pos2 + 12]; } + get lineOffsetX() { return this._structArray.float32[this._pos4 + 7]; } + get lineOffsetY() { return this._structArray.float32[this._pos4 + 8]; } + get writingMode() { return this._structArray.uint8[this._pos1 + 36]; } + get placedOrientation() { return this._structArray.uint8[this._pos1 + 37]; } + set placedOrientation(x) { this._structArray.uint8[this._pos1 + 37] = x; } + get hidden() { return this._structArray.uint8[this._pos1 + 38]; } + set hidden(x) { this._structArray.uint8[this._pos1 + 38] = x; } + get crossTileID() { return this._structArray.uint32[this._pos4 + 10]; } + set crossTileID(x) { this._structArray.uint32[this._pos4 + 10] = x; } + get associatedIconIndex() { return this._structArray.int16[this._pos2 + 22]; } +} +PlacedSymbolStruct.prototype.size = 48; +/** @internal */ +class PlacedSymbolArray extends StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 { + /** + * Return the PlacedSymbolStruct at the given location in the array. + * @param index The index of the element. + */ + get(index) { + return new PlacedSymbolStruct(this, index); + } +} +register('PlacedSymbolArray', PlacedSymbolArray); +/** @internal */ +class SymbolInstanceStruct extends Struct { + get anchorX() { return this._structArray.int16[this._pos2 + 0]; } + get anchorY() { return this._structArray.int16[this._pos2 + 1]; } + get rightJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 2]; } + get centerJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 3]; } + get leftJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 4]; } + get verticalPlacedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 5]; } + get placedIconSymbolIndex() { return this._structArray.int16[this._pos2 + 6]; } + get verticalPlacedIconSymbolIndex() { return this._structArray.int16[this._pos2 + 7]; } + get key() { return this._structArray.uint16[this._pos2 + 8]; } + get textBoxStartIndex() { return this._structArray.uint16[this._pos2 + 9]; } + get textBoxEndIndex() { return this._structArray.uint16[this._pos2 + 10]; } + get verticalTextBoxStartIndex() { return this._structArray.uint16[this._pos2 + 11]; } + get verticalTextBoxEndIndex() { return this._structArray.uint16[this._pos2 + 12]; } + get iconBoxStartIndex() { return this._structArray.uint16[this._pos2 + 13]; } + get iconBoxEndIndex() { return this._structArray.uint16[this._pos2 + 14]; } + get verticalIconBoxStartIndex() { return this._structArray.uint16[this._pos2 + 15]; } + get verticalIconBoxEndIndex() { return this._structArray.uint16[this._pos2 + 16]; } + get featureIndex() { return this._structArray.uint16[this._pos2 + 17]; } + get numHorizontalGlyphVertices() { return this._structArray.uint16[this._pos2 + 18]; } + get numVerticalGlyphVertices() { return this._structArray.uint16[this._pos2 + 19]; } + get numIconVertices() { return this._structArray.uint16[this._pos2 + 20]; } + get numVerticalIconVertices() { return this._structArray.uint16[this._pos2 + 21]; } + get useRuntimeCollisionCircles() { return this._structArray.uint16[this._pos2 + 22]; } + get crossTileID() { return this._structArray.uint32[this._pos4 + 12]; } + set crossTileID(x) { this._structArray.uint32[this._pos4 + 12] = x; } + get textBoxScale() { return this._structArray.float32[this._pos4 + 13]; } + get collisionCircleDiameter() { return this._structArray.float32[this._pos4 + 14]; } + get textAnchorOffsetStartIndex() { return this._structArray.uint16[this._pos2 + 30]; } + get textAnchorOffsetEndIndex() { return this._structArray.uint16[this._pos2 + 31]; } +} +SymbolInstanceStruct.prototype.size = 64; +/** @internal */ +class SymbolInstanceArray extends StructArrayLayout8i15ui1ul2f2ui64 { + /** + * Return the SymbolInstanceStruct at the given location in the array. + * @param index The index of the element. + */ + get(index) { + return new SymbolInstanceStruct(this, index); + } +} +register('SymbolInstanceArray', SymbolInstanceArray); +/** @internal */ +class GlyphOffsetArray extends StructArrayLayout1f4 { + getoffsetX(index) { return this.float32[index * 1 + 0]; } +} +register('GlyphOffsetArray', GlyphOffsetArray); +/** @internal */ +class SymbolLineVertexArray extends StructArrayLayout3i6 { + getx(index) { return this.int16[index * 3 + 0]; } + gety(index) { return this.int16[index * 3 + 1]; } + gettileUnitDistanceFromAnchor(index) { return this.int16[index * 3 + 2]; } +} +register('SymbolLineVertexArray', SymbolLineVertexArray); +/** @internal */ +class TextAnchorOffsetStruct extends Struct { + get textAnchor() { return this._structArray.uint16[this._pos2 + 0]; } + get textOffset0() { return this._structArray.float32[this._pos4 + 1]; } + get textOffset1() { return this._structArray.float32[this._pos4 + 2]; } +} +TextAnchorOffsetStruct.prototype.size = 12; +/** @internal */ +class TextAnchorOffsetArray extends StructArrayLayout1ui2f12 { + /** + * Return the TextAnchorOffsetStruct at the given location in the array. + * @param index The index of the element. + */ + get(index) { + return new TextAnchorOffsetStruct(this, index); + } +} +register('TextAnchorOffsetArray', TextAnchorOffsetArray); +/** @internal */ +class FeatureIndexStruct extends Struct { + get featureIndex() { return this._structArray.uint32[this._pos4 + 0]; } + get sourceLayerIndex() { return this._structArray.uint16[this._pos2 + 2]; } + get bucketIndex() { return this._structArray.uint16[this._pos2 + 3]; } +} +FeatureIndexStruct.prototype.size = 8; +/** @internal */ +class FeatureIndexArray extends StructArrayLayout1ul2ui8 { + /** + * Return the FeatureIndexStruct at the given location in the array. + * @param index The index of the element. + */ + get(index) { + return new FeatureIndexStruct(this, index); + } +} +register('FeatureIndexArray', FeatureIndexArray); +class PosArray extends StructArrayLayout2i4 { +} +class Pos3dArray extends StructArrayLayout3i6 { +} +class RasterBoundsArray extends StructArrayLayout4i8 { +} +class CircleLayoutArray extends StructArrayLayout2i4 { +} +class FillLayoutArray extends StructArrayLayout2i4 { +} +class FillExtrusionLayoutArray extends StructArrayLayout2i4i12 { +} +class HeatmapLayoutArray extends StructArrayLayout2i4 { +} +class LineLayoutArray extends StructArrayLayout2i4ub8 { +} +class LineExtLayoutArray extends StructArrayLayout2f8 { +} +class PatternLayoutArray extends StructArrayLayout10ui20 { +} +class SymbolLayoutArray extends StructArrayLayout4i4ui4i24 { +} +class SymbolDynamicLayoutArray extends StructArrayLayout3f12 { +} +class SymbolOpacityArray extends StructArrayLayout1ul4 { +} +class CollisionBoxLayoutArray extends StructArrayLayout2i2i2i12 { +} +class CollisionCircleLayoutArray extends StructArrayLayout2f1f2i16 { +} +class CollisionVertexArray extends StructArrayLayout2ub2f12 { +} +class QuadTriangleArray extends StructArrayLayout3ui6 { +} +class TriangleIndexArray extends StructArrayLayout3ui6 { +} +class LineIndexArray extends StructArrayLayout2ui4 { +} +class LineStripIndexArray extends StructArrayLayout1ui2 { +} + +const layout$6 = createLayout([ + { name: 'a_pos', components: 2, type: 'Int16' } +], 4); +const { members: members$4, size: size$4, alignment: alignment$4 } = layout$6; + +/** + * @internal + * Used for calculations on vector segments + */ +class SegmentVector { + constructor(segments = []) { + this.segments = segments; + } + prepareSegment(numVertices, layoutVertexArray, indexArray, sortKey) { + let segment = this.segments[this.segments.length - 1]; + if (numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) + warnOnce(`Max vertices per segment is ${SegmentVector.MAX_VERTEX_ARRAY_LENGTH}: bucket requested ${numVertices}`); + if (!segment || segment.vertexLength + numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH || segment.sortKey !== sortKey) { + segment = { + vertexOffset: layoutVertexArray.length, + primitiveOffset: indexArray.length, + vertexLength: 0, + primitiveLength: 0 + }; + if (sortKey !== undefined) + segment.sortKey = sortKey; + this.segments.push(segment); + } + return segment; + } + get() { + return this.segments; + } + destroy() { + for (const segment of this.segments) { + for (const k in segment.vaos) { + segment.vaos[k].destroy(); + } + } + } + static simpleSegment(vertexOffset, primitiveOffset, vertexLength, primitiveLength) { + return new SegmentVector([{ + vertexOffset, + primitiveOffset, + vertexLength, + primitiveLength, + vaos: {}, + sortKey: 0 + }]); + } +} +/** + * The maximum size of a vertex array. This limit is imposed by WebGL's 16 bit + * addressing of vertex buffers. + */ +SegmentVector.MAX_VERTEX_ARRAY_LENGTH = Math.pow(2, 16) - 1; +register('SegmentVector', SegmentVector); + +/** + * Packs two numbers, interpreted as 8-bit unsigned integers, into a single + * float. Unpack them in the shader using the `unpack_float()` function, + * defined in _prelude.vertex.glsl + */ +function packUint8ToFloat(a, b) { + // coerce a and b to 8-bit ints + a = clamp$1(Math.floor(a), 0, 255); + b = clamp$1(Math.floor(b), 0, 255); + return 256 * a + b; +} + +const patternAttributes = createLayout([ + // [tl.x, tl.y, br.x, br.y] + { name: 'a_pattern_from', components: 4, type: 'Uint16' }, + { name: 'a_pattern_to', components: 4, type: 'Uint16' }, + { name: 'a_pixel_ratio_from', components: 1, type: 'Uint16' }, + { name: 'a_pixel_ratio_to', components: 1, type: 'Uint16' }, +]); + +var murmurhashJs$1 = {exports: {}}; + +var murmurhash3_gc$2 = {exports: {}}; + +/** + * JS Implementation of MurmurHash3 (r136) (as of May 20, 2011) + * + * @author Gary Court + * @see http://github.com/garycourt/murmurhash-js + * @author Austin Appleby + * @see http://sites.google.com/site/murmurhash/ + * + * @param {string} key ASCII only + * @param {number} seed Positive integer only + * @return {number} 32-bit positive integer hash + */ +var murmurhash3_gc = murmurhash3_gc$2.exports; + +(function (module) { + function murmurhash3_32_gc(key, seed) { + var remainder, bytes, h1, h1b, c1, c1b, c2, c2b, k1, i; + + remainder = key.length & 3; // key.length % 4 + bytes = key.length - remainder; + h1 = seed; + c1 = 0xcc9e2d51; + c2 = 0x1b873593; + i = 0; + + while (i < bytes) { + k1 = + ((key.charCodeAt(i) & 0xff)) | + ((key.charCodeAt(++i) & 0xff) << 8) | + ((key.charCodeAt(++i) & 0xff) << 16) | + ((key.charCodeAt(++i) & 0xff) << 24); + ++i; + + k1 = ((((k1 & 0xffff) * c1) + ((((k1 >>> 16) * c1) & 0xffff) << 16))) & 0xffffffff; + k1 = (k1 << 15) | (k1 >>> 17); + k1 = ((((k1 & 0xffff) * c2) + ((((k1 >>> 16) * c2) & 0xffff) << 16))) & 0xffffffff; + + h1 ^= k1; + h1 = (h1 << 13) | (h1 >>> 19); + h1b = ((((h1 & 0xffff) * 5) + ((((h1 >>> 16) * 5) & 0xffff) << 16))) & 0xffffffff; + h1 = (((h1b & 0xffff) + 0x6b64) + ((((h1b >>> 16) + 0xe654) & 0xffff) << 16)); + } + + k1 = 0; + + switch (remainder) { + case 3: k1 ^= (key.charCodeAt(i + 2) & 0xff) << 16; + case 2: k1 ^= (key.charCodeAt(i + 1) & 0xff) << 8; + case 1: k1 ^= (key.charCodeAt(i) & 0xff); + + k1 = (((k1 & 0xffff) * c1) + ((((k1 >>> 16) * c1) & 0xffff) << 16)) & 0xffffffff; + k1 = (k1 << 15) | (k1 >>> 17); + k1 = (((k1 & 0xffff) * c2) + ((((k1 >>> 16) * c2) & 0xffff) << 16)) & 0xffffffff; + h1 ^= k1; + } + + h1 ^= key.length; + + h1 ^= h1 >>> 16; + h1 = (((h1 & 0xffff) * 0x85ebca6b) + ((((h1 >>> 16) * 0x85ebca6b) & 0xffff) << 16)) & 0xffffffff; + h1 ^= h1 >>> 13; + h1 = ((((h1 & 0xffff) * 0xc2b2ae35) + ((((h1 >>> 16) * 0xc2b2ae35) & 0xffff) << 16))) & 0xffffffff; + h1 ^= h1 >>> 16; + + return h1 >>> 0; + } + + if('object' !== "undefined") { + module.exports = murmurhash3_32_gc; + } +} (murmurhash3_gc$2)); + +var murmurhash3_gcExports = murmurhash3_gc$2.exports; +var murmurhash3_gc$1 = /*@__PURE__*/getDefaultExportFromCjs(murmurhash3_gcExports); + +var murmurhash2_gc$2 = {exports: {}}; + +/** + * JS Implementation of MurmurHash2 + * + * @author Gary Court + * @see http://github.com/garycourt/murmurhash-js + * @author Austin Appleby + * @see http://sites.google.com/site/murmurhash/ + * + * @param {string} str ASCII only + * @param {number} seed Positive integer only + * @return {number} 32-bit positive integer hash + */ +var murmurhash2_gc = murmurhash2_gc$2.exports; + +(function (module) { + function murmurhash2_32_gc(str, seed) { + var + l = str.length, + h = seed ^ l, + i = 0, + k; + + while (l >= 4) { + k = + ((str.charCodeAt(i) & 0xff)) | + ((str.charCodeAt(++i) & 0xff) << 8) | + ((str.charCodeAt(++i) & 0xff) << 16) | + ((str.charCodeAt(++i) & 0xff) << 24); + + k = (((k & 0xffff) * 0x5bd1e995) + ((((k >>> 16) * 0x5bd1e995) & 0xffff) << 16)); + k ^= k >>> 24; + k = (((k & 0xffff) * 0x5bd1e995) + ((((k >>> 16) * 0x5bd1e995) & 0xffff) << 16)); + + h = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16)) ^ k; + + l -= 4; + ++i; + } + + switch (l) { + case 3: h ^= (str.charCodeAt(i + 2) & 0xff) << 16; + case 2: h ^= (str.charCodeAt(i + 1) & 0xff) << 8; + case 1: h ^= (str.charCodeAt(i) & 0xff); + h = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16)); + } + + h ^= h >>> 13; + h = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16)); + h ^= h >>> 15; + + return h >>> 0; + } + + if('object' !== undefined) { + module.exports = murmurhash2_32_gc; + } +} (murmurhash2_gc$2)); + +var murmurhash2_gcExports = murmurhash2_gc$2.exports; +var murmurhash2_gc$1 = /*@__PURE__*/getDefaultExportFromCjs(murmurhash2_gcExports); + +var murmurhashJs = murmurhashJs$1.exports; + +var murmur3 = murmurhash3_gcExports; +var murmur2 = murmurhash2_gcExports; + +murmurhashJs$1.exports = murmur3; +var murmur3_1 = murmurhashJs$1.exports.murmur3 = murmur3; +var murmur2_1 = murmurhashJs$1.exports.murmur2 = murmur2; + +var murmurhashJsExports = murmurhashJs$1.exports; +var murmur3$1 = /*@__PURE__*/getDefaultExportFromCjs(murmurhashJsExports); + +// A transferable data structure that maps feature ids to their indices and buffer offsets +class FeaturePositionMap { + constructor() { + this.ids = []; + this.positions = []; + this.indexed = false; + } + add(id, index, start, end) { + this.ids.push(getNumericId(id)); + this.positions.push(index, start, end); + } + getPositions(id) { + if (!this.indexed) + throw new Error('Trying to get index, but feature positions are not indexed'); + const intId = getNumericId(id); + // binary search for the first occurrence of id in this.ids; + // relies on ids/positions being sorted by id, which happens in serialization + let i = 0; + let j = this.ids.length - 1; + while (i < j) { + const m = (i + j) >> 1; + if (this.ids[m] >= intId) { + j = m; + } + else { + i = m + 1; + } + } + const positions = []; + while (this.ids[i] === intId) { + const index = this.positions[3 * i]; + const start = this.positions[3 * i + 1]; + const end = this.positions[3 * i + 2]; + positions.push({ index, start, end }); + i++; + } + return positions; + } + static serialize(map, transferables) { + const ids = new Float64Array(map.ids); + const positions = new Uint32Array(map.positions); + sort$1(ids, positions, 0, ids.length - 1); + if (transferables) { + transferables.push(ids.buffer, positions.buffer); + } + return { ids, positions }; + } + static deserialize(obj) { + const map = new FeaturePositionMap(); + // after transferring, we only use these arrays statically (no pushes), + // so TypedArray vs Array distinction that flow points out doesn't matter + map.ids = obj.ids; + map.positions = obj.positions; + map.indexed = true; + return map; + } +} +function getNumericId(value) { + const numValue = +value; + if (!isNaN(numValue) && numValue <= Number.MAX_SAFE_INTEGER) { + return numValue; + } + return murmur3$1(String(value)); +} +// custom quicksort that sorts ids, indices and offsets together (by ids) +// uses Hoare partitioning & manual tail call optimization to avoid worst case scenarios +function sort$1(ids, positions, left, right) { + while (left < right) { + const pivot = ids[(left + right) >> 1]; + let i = left - 1; + let j = right + 1; + while (true) { + do + i++; + while (ids[i] < pivot); + do + j--; + while (ids[j] > pivot); + if (i >= j) + break; + swap$2(ids, i, j); + swap$2(positions, 3 * i, 3 * j); + swap$2(positions, 3 * i + 1, 3 * j + 1); + swap$2(positions, 3 * i + 2, 3 * j + 2); + } + if (j - left < right - j) { + sort$1(ids, positions, left, j); + left = j + 1; + } + else { + sort$1(ids, positions, j + 1, right); + right = j; + } + } +} +function swap$2(arr, i, j) { + const tmp = arr[i]; + arr[i] = arr[j]; + arr[j] = tmp; +} +register('FeaturePositionMap', FeaturePositionMap); + +/** + * @internal + * A base uniform abstract class + */ +class Uniform { + constructor(context, location) { + this.gl = context.gl; + this.location = location; + } +} +class Uniform1i extends Uniform { + constructor(context, location) { + super(context, location); + this.current = 0; + } + set(v) { + if (this.current !== v) { + this.current = v; + this.gl.uniform1i(this.location, v); + } + } +} +class Uniform1f extends Uniform { + constructor(context, location) { + super(context, location); + this.current = 0; + } + set(v) { + if (this.current !== v) { + this.current = v; + this.gl.uniform1f(this.location, v); + } + } +} +class Uniform2f extends Uniform { + constructor(context, location) { + super(context, location); + this.current = [0, 0]; + } + set(v) { + if (v[0] !== this.current[0] || v[1] !== this.current[1]) { + this.current = v; + this.gl.uniform2f(this.location, v[0], v[1]); + } + } +} +class Uniform3f extends Uniform { + constructor(context, location) { + super(context, location); + this.current = [0, 0, 0]; + } + set(v) { + if (v[0] !== this.current[0] || v[1] !== this.current[1] || v[2] !== this.current[2]) { + this.current = v; + this.gl.uniform3f(this.location, v[0], v[1], v[2]); + } + } +} +class Uniform4f extends Uniform { + constructor(context, location) { + super(context, location); + this.current = [0, 0, 0, 0]; + } + set(v) { + if (v[0] !== this.current[0] || v[1] !== this.current[1] || + v[2] !== this.current[2] || v[3] !== this.current[3]) { + this.current = v; + this.gl.uniform4f(this.location, v[0], v[1], v[2], v[3]); + } + } +} +class UniformColor extends Uniform { + constructor(context, location) { + super(context, location); + this.current = Color.transparent; + } + set(v) { + if (v.r !== this.current.r || v.g !== this.current.g || + v.b !== this.current.b || v.a !== this.current.a) { + this.current = v; + this.gl.uniform4f(this.location, v.r, v.g, v.b, v.a); + } + } +} +const emptyMat4 = new Float32Array(16); +class UniformMatrix4f extends Uniform { + constructor(context, location) { + super(context, location); + this.current = emptyMat4; + } + set(v) { + // The vast majority of matrix comparisons that will trip this set + // happen at i=12 or i=0, so we check those first to avoid lots of + // unnecessary iteration: + if (v[12] !== this.current[12] || v[0] !== this.current[0]) { + this.current = v; + this.gl.uniformMatrix4fv(this.location, false, v); + return; + } + for (let i = 1; i < 16; i++) { + if (v[i] !== this.current[i]) { + this.current = v; + this.gl.uniformMatrix4fv(this.location, false, v); + break; + } + } + } +} + +function packColor(color) { + return [ + packUint8ToFloat(255 * color.r, 255 * color.g), + packUint8ToFloat(255 * color.b, 255 * color.a) + ]; +} +class ConstantBinder { + constructor(value, names, type) { + this.value = value; + this.uniformNames = names.map(name => `u_${name}`); + this.type = type; + } + setUniform(uniform, globals, currentValue) { + uniform.set(currentValue.constantOr(this.value)); + } + getBinding(context, location, _) { + return (this.type === 'color') ? + new UniformColor(context, location) : + new Uniform1f(context, location); + } +} +class CrossFadedConstantBinder { + constructor(value, names) { + this.uniformNames = names.map(name => `u_${name}`); + this.patternFrom = null; + this.patternTo = null; + this.pixelRatioFrom = 1.0; + this.pixelRatioTo = 1.0; + } + setConstantPatternPositions(posTo, posFrom) { + this.pixelRatioFrom = posFrom.pixelRatio; + this.pixelRatioTo = posTo.pixelRatio; + this.patternFrom = posFrom.tlbr; + this.patternTo = posTo.tlbr; + } + setUniform(uniform, globals, currentValue, uniformName) { + const pos = uniformName === 'u_pattern_to' ? this.patternTo : + uniformName === 'u_pattern_from' ? this.patternFrom : + uniformName === 'u_pixel_ratio_to' ? this.pixelRatioTo : + uniformName === 'u_pixel_ratio_from' ? this.pixelRatioFrom : null; + if (pos) + uniform.set(pos); + } + getBinding(context, location, name) { + return name.substr(0, 9) === 'u_pattern' ? + new Uniform4f(context, location) : + new Uniform1f(context, location); + } +} +class SourceExpressionBinder { + constructor(expression, names, type, PaintVertexArray) { + this.expression = expression; + this.type = type; + this.maxValue = 0; + this.paintVertexAttributes = names.map((name) => ({ + name: `a_${name}`, + type: 'Float32', + components: type === 'color' ? 2 : 1, + offset: 0 + })); + this.paintVertexArray = new PaintVertexArray(); + } + populatePaintArray(newLength, feature, imagePositions, canonical, formattedSection) { + const start = this.paintVertexArray.length; + const value = this.expression.evaluate(new EvaluationParameters(0), feature, {}, canonical, [], formattedSection); + this.paintVertexArray.resize(newLength); + this._setPaintValue(start, newLength, value); + } + updatePaintArray(start, end, feature, featureState) { + const value = this.expression.evaluate({ zoom: 0 }, feature, featureState); + this._setPaintValue(start, end, value); + } + _setPaintValue(start, end, value) { + if (this.type === 'color') { + const color = packColor(value); + for (let i = start; i < end; i++) { + this.paintVertexArray.emplace(i, color[0], color[1]); + } + } + else { + for (let i = start; i < end; i++) { + this.paintVertexArray.emplace(i, value); + } + this.maxValue = Math.max(this.maxValue, Math.abs(value)); + } + } + upload(context) { + if (this.paintVertexArray && this.paintVertexArray.arrayBuffer) { + if (this.paintVertexBuffer && this.paintVertexBuffer.buffer) { + this.paintVertexBuffer.updateData(this.paintVertexArray); + } + else { + this.paintVertexBuffer = context.createVertexBuffer(this.paintVertexArray, this.paintVertexAttributes, this.expression.isStateDependent); + } + } + } + destroy() { + if (this.paintVertexBuffer) { + this.paintVertexBuffer.destroy(); + } + } +} +class CompositeExpressionBinder { + constructor(expression, names, type, useIntegerZoom, zoom, PaintVertexArray) { + this.expression = expression; + this.uniformNames = names.map(name => `u_${name}_t`); + this.type = type; + this.useIntegerZoom = useIntegerZoom; + this.zoom = zoom; + this.maxValue = 0; + this.paintVertexAttributes = names.map((name) => ({ + name: `a_${name}`, + type: 'Float32', + components: type === 'color' ? 4 : 2, + offset: 0 + })); + this.paintVertexArray = new PaintVertexArray(); + } + populatePaintArray(newLength, feature, imagePositions, canonical, formattedSection) { + const min = this.expression.evaluate(new EvaluationParameters(this.zoom), feature, {}, canonical, [], formattedSection); + const max = this.expression.evaluate(new EvaluationParameters(this.zoom + 1), feature, {}, canonical, [], formattedSection); + const start = this.paintVertexArray.length; + this.paintVertexArray.resize(newLength); + this._setPaintValue(start, newLength, min, max); + } + updatePaintArray(start, end, feature, featureState) { + const min = this.expression.evaluate({ zoom: this.zoom }, feature, featureState); + const max = this.expression.evaluate({ zoom: this.zoom + 1 }, feature, featureState); + this._setPaintValue(start, end, min, max); + } + _setPaintValue(start, end, min, max) { + if (this.type === 'color') { + const minColor = packColor(min); + const maxColor = packColor(max); + for (let i = start; i < end; i++) { + this.paintVertexArray.emplace(i, minColor[0], minColor[1], maxColor[0], maxColor[1]); + } + } + else { + for (let i = start; i < end; i++) { + this.paintVertexArray.emplace(i, min, max); + } + this.maxValue = Math.max(this.maxValue, Math.abs(min), Math.abs(max)); + } + } + upload(context) { + if (this.paintVertexArray && this.paintVertexArray.arrayBuffer) { + if (this.paintVertexBuffer && this.paintVertexBuffer.buffer) { + this.paintVertexBuffer.updateData(this.paintVertexArray); + } + else { + this.paintVertexBuffer = context.createVertexBuffer(this.paintVertexArray, this.paintVertexAttributes, this.expression.isStateDependent); + } + } + } + destroy() { + if (this.paintVertexBuffer) { + this.paintVertexBuffer.destroy(); + } + } + setUniform(uniform, globals) { + const currentZoom = this.useIntegerZoom ? Math.floor(globals.zoom) : globals.zoom; + const factor = clamp$1(this.expression.interpolationFactor(currentZoom, this.zoom, this.zoom + 1), 0, 1); + uniform.set(factor); + } + getBinding(context, location, _) { + return new Uniform1f(context, location); + } +} +class CrossFadedCompositeBinder { + constructor(expression, type, useIntegerZoom, zoom, PaintVertexArray, layerId) { + this.expression = expression; + this.type = type; + this.useIntegerZoom = useIntegerZoom; + this.zoom = zoom; + this.layerId = layerId; + this.zoomInPaintVertexArray = new PaintVertexArray(); + this.zoomOutPaintVertexArray = new PaintVertexArray(); + } + populatePaintArray(length, feature, imagePositions) { + const start = this.zoomInPaintVertexArray.length; + this.zoomInPaintVertexArray.resize(length); + this.zoomOutPaintVertexArray.resize(length); + this._setPaintValues(start, length, feature.patterns && feature.patterns[this.layerId], imagePositions); + } + updatePaintArray(start, end, feature, featureState, imagePositions) { + this._setPaintValues(start, end, feature.patterns && feature.patterns[this.layerId], imagePositions); + } + _setPaintValues(start, end, patterns, positions) { + if (!positions || !patterns) + return; + const { min, mid, max } = patterns; + const imageMin = positions[min]; + const imageMid = positions[mid]; + const imageMax = positions[max]; + if (!imageMin || !imageMid || !imageMax) + return; + // We populate two paint arrays because, for cross-faded properties, we don't know which direction + // we're cross-fading to at layout time. In order to keep vertex attributes to a minimum and not pass + // unnecessary vertex data to the shaders, we determine which to upload at draw time. + for (let i = start; i < end; i++) { + this.zoomInPaintVertexArray.emplace(i, imageMid.tl[0], imageMid.tl[1], imageMid.br[0], imageMid.br[1], imageMin.tl[0], imageMin.tl[1], imageMin.br[0], imageMin.br[1], imageMid.pixelRatio, imageMin.pixelRatio); + this.zoomOutPaintVertexArray.emplace(i, imageMid.tl[0], imageMid.tl[1], imageMid.br[0], imageMid.br[1], imageMax.tl[0], imageMax.tl[1], imageMax.br[0], imageMax.br[1], imageMid.pixelRatio, imageMax.pixelRatio); + } + } + upload(context) { + if (this.zoomInPaintVertexArray && this.zoomInPaintVertexArray.arrayBuffer && this.zoomOutPaintVertexArray && this.zoomOutPaintVertexArray.arrayBuffer) { + this.zoomInPaintVertexBuffer = context.createVertexBuffer(this.zoomInPaintVertexArray, patternAttributes.members, this.expression.isStateDependent); + this.zoomOutPaintVertexBuffer = context.createVertexBuffer(this.zoomOutPaintVertexArray, patternAttributes.members, this.expression.isStateDependent); + } + } + destroy() { + if (this.zoomOutPaintVertexBuffer) + this.zoomOutPaintVertexBuffer.destroy(); + if (this.zoomInPaintVertexBuffer) + this.zoomInPaintVertexBuffer.destroy(); + } +} +/** + * @internal + * ProgramConfiguration contains the logic for binding style layer properties and tile + * layer feature data into GL program uniforms and vertex attributes. + * + * Non-data-driven property values are bound to shader uniforms. Data-driven property + * values are bound to vertex attributes. In order to support a uniform GLSL syntax over + * both, [Mapbox GL Shaders](https://github.com/mapbox/mapbox-gl-shaders) defines a `#pragma` + * abstraction, which ProgramConfiguration is responsible for implementing. At runtime, + * it examines the attributes of a particular layer, combines this with fixed knowledge + * about how layers of the particular type are implemented, and determines which uniforms + * and vertex attributes will be required. It can then substitute the appropriate text + * into the shader source code, create and link a program, and bind the uniforms and + * vertex attributes in preparation for drawing. + * + * When a vector tile is parsed, this same configuration information is used to + * populate the attribute buffers needed for data-driven styling using the zoom + * level and feature property data. + */ +class ProgramConfiguration { + constructor(layer, zoom, filterProperties) { + this.binders = {}; + this._buffers = []; + const keys = []; + for (const property in layer.paint._values) { + if (!filterProperties(property)) + continue; + const value = layer.paint.get(property); + if (!(value instanceof PossiblyEvaluatedPropertyValue) || !supportsPropertyExpression(value.property.specification)) { + continue; + } + const names = paintAttributeNames(property, layer.type); + const expression = value.value; + const type = value.property.specification.type; + const useIntegerZoom = value.property.useIntegerZoom; + const propType = value.property.specification['property-type']; + const isCrossFaded = propType === 'cross-faded' || propType === 'cross-faded-data-driven'; + if (expression.kind === 'constant') { + this.binders[property] = isCrossFaded ? + new CrossFadedConstantBinder(expression.value, names) : + new ConstantBinder(expression.value, names, type); + keys.push(`/u_${property}`); + } + else if (expression.kind === 'source' || isCrossFaded) { + const StructArrayLayout = layoutType(property, type, 'source'); + this.binders[property] = isCrossFaded ? + new CrossFadedCompositeBinder(expression, type, useIntegerZoom, zoom, StructArrayLayout, layer.id) : + new SourceExpressionBinder(expression, names, type, StructArrayLayout); + keys.push(`/a_${property}`); + } + else { + const StructArrayLayout = layoutType(property, type, 'composite'); + this.binders[property] = new CompositeExpressionBinder(expression, names, type, useIntegerZoom, zoom, StructArrayLayout); + keys.push(`/z_${property}`); + } + } + this.cacheKey = keys.sort().join(''); + } + getMaxValue(property) { + const binder = this.binders[property]; + return binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder ? binder.maxValue : 0; + } + populatePaintArrays(newLength, feature, imagePositions, canonical, formattedSection) { + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder) + binder.populatePaintArray(newLength, feature, imagePositions, canonical, formattedSection); + } + } + setConstantPatternPositions(posTo, posFrom) { + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof CrossFadedConstantBinder) + binder.setConstantPatternPositions(posTo, posFrom); + } + } + updatePaintArrays(featureStates, featureMap, vtLayer, layer, imagePositions) { + let dirty = false; + for (const id in featureStates) { + const positions = featureMap.getPositions(id); + for (const pos of positions) { + const feature = vtLayer.feature(pos.index); + for (const property in this.binders) { + const binder = this.binders[property]; + if ((binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || + binder instanceof CrossFadedCompositeBinder) && binder.expression.isStateDependent === true) { + //AHM: Remove after https://github.com/mapbox/mapbox-gl-js/issues/6255 + const value = layer.paint.get(property); + binder.expression = value.value; + binder.updatePaintArray(pos.start, pos.end, feature, featureStates[id], imagePositions); + dirty = true; + } + } + } + } + return dirty; + } + defines() { + const result = []; + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder) { + result.push(...binder.uniformNames.map(name => `#define HAS_UNIFORM_${name}`)); + } + } + return result; + } + getBinderAttributes() { + const result = []; + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder) { + for (let i = 0; i < binder.paintVertexAttributes.length; i++) { + result.push(binder.paintVertexAttributes[i].name); + } + } + else if (binder instanceof CrossFadedCompositeBinder) { + for (let i = 0; i < patternAttributes.members.length; i++) { + result.push(patternAttributes.members[i].name); + } + } + } + return result; + } + getBinderUniforms() { + const uniforms = []; + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder || binder instanceof CompositeExpressionBinder) { + for (const uniformName of binder.uniformNames) { + uniforms.push(uniformName); + } + } + } + return uniforms; + } + getPaintVertexBuffers() { + return this._buffers; + } + getUniforms(context, locations) { + const uniforms = []; + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder || binder instanceof CompositeExpressionBinder) { + for (const name of binder.uniformNames) { + if (locations[name]) { + const binding = binder.getBinding(context, locations[name], name); + uniforms.push({ name, property, binding }); + } + } + } + } + return uniforms; + } + setUniforms(context, binderUniforms, properties, globals) { + // Uniform state bindings are owned by the Program, but we set them + // from within the ProgramConfiguraton's binder members. + for (const { name, property, binding } of binderUniforms) { + this.binders[property].setUniform(binding, globals, properties.get(property), name); + } + } + updatePaintBuffers(crossfade) { + this._buffers = []; + for (const property in this.binders) { + const binder = this.binders[property]; + if (crossfade && binder instanceof CrossFadedCompositeBinder) { + const patternVertexBuffer = crossfade.fromScale === 2 ? binder.zoomInPaintVertexBuffer : binder.zoomOutPaintVertexBuffer; + if (patternVertexBuffer) + this._buffers.push(patternVertexBuffer); + } + else if ((binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder) && binder.paintVertexBuffer) { + this._buffers.push(binder.paintVertexBuffer); + } + } + } + upload(context) { + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder) + binder.upload(context); + } + this.updatePaintBuffers(); + } + destroy() { + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder) + binder.destroy(); + } + } +} +class ProgramConfigurationSet { + constructor(layers, zoom, filterProperties = () => true) { + this.programConfigurations = {}; + for (const layer of layers) { + this.programConfigurations[layer.id] = new ProgramConfiguration(layer, zoom, filterProperties); + } + this.needsUpload = false; + this._featureMap = new FeaturePositionMap(); + this._bufferOffset = 0; + } + populatePaintArrays(length, feature, index, imagePositions, canonical, formattedSection) { + for (const key in this.programConfigurations) { + this.programConfigurations[key].populatePaintArrays(length, feature, imagePositions, canonical, formattedSection); + } + if (feature.id !== undefined) { + this._featureMap.add(feature.id, index, this._bufferOffset, length); + } + this._bufferOffset = length; + this.needsUpload = true; + } + updatePaintArrays(featureStates, vtLayer, layers, imagePositions) { + for (const layer of layers) { + this.needsUpload = this.programConfigurations[layer.id].updatePaintArrays(featureStates, this._featureMap, vtLayer, layer, imagePositions) || this.needsUpload; + } + } + get(layerId) { + return this.programConfigurations[layerId]; + } + upload(context) { + if (!this.needsUpload) + return; + for (const layerId in this.programConfigurations) { + this.programConfigurations[layerId].upload(context); + } + this.needsUpload = false; + } + destroy() { + for (const layerId in this.programConfigurations) { + this.programConfigurations[layerId].destroy(); + } + } +} +function paintAttributeNames(property, type) { + const attributeNameExceptions = { + 'text-opacity': ['opacity'], + 'icon-opacity': ['opacity'], + 'text-color': ['fill_color'], + 'icon-color': ['fill_color'], + 'text-halo-color': ['halo_color'], + 'icon-halo-color': ['halo_color'], + 'text-halo-blur': ['halo_blur'], + 'icon-halo-blur': ['halo_blur'], + 'text-halo-width': ['halo_width'], + 'icon-halo-width': ['halo_width'], + 'line-gap-width': ['gapwidth'], + 'line-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'], + 'fill-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'], + 'fill-extrusion-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'], + }; + return attributeNameExceptions[property] || [property.replace(`${type}-`, '').replace(/-/g, '_')]; +} +function getLayoutException(property) { + const propertyExceptions = { + 'line-pattern': { + 'source': PatternLayoutArray, + 'composite': PatternLayoutArray + }, + 'fill-pattern': { + 'source': PatternLayoutArray, + 'composite': PatternLayoutArray + }, + 'fill-extrusion-pattern': { + 'source': PatternLayoutArray, + 'composite': PatternLayoutArray + } + }; + return propertyExceptions[property]; +} +function layoutType(property, type, binderType) { + const defaultLayouts = { + 'color': { + 'source': StructArrayLayout2f8, + 'composite': StructArrayLayout4f16 + }, + 'number': { + 'source': StructArrayLayout1f4, + 'composite': StructArrayLayout2f8 + } + }; + const layoutException = getLayoutException(property); + return layoutException && layoutException[binderType] || defaultLayouts[type][binderType]; +} +register('ConstantBinder', ConstantBinder); +register('CrossFadedConstantBinder', CrossFadedConstantBinder); +register('SourceExpressionBinder', SourceExpressionBinder); +register('CrossFadedCompositeBinder', CrossFadedCompositeBinder); +register('CompositeExpressionBinder', CompositeExpressionBinder); +register('ProgramConfiguration', ProgramConfiguration, { omit: ['_buffers'] }); +register('ProgramConfigurationSet', ProgramConfigurationSet); + +/** + * The maximum value of a coordinate in the internal tile coordinate system. Coordinates of + * all source features normalized to this extent upon load. + * + * The value is a consequence of the following: + * + * * Vertex buffer store positions as signed 16 bit integers. + * * One bit is lost for signedness to support tile buffers. + * * One bit is lost because the line vertex buffer used to pack 1 bit of other data into the int. + * * One bit is lost to support features extending past the extent on the right edge of the tile. + * * This leaves us with 2^13 = 8192 + */ +const EXTENT = 8192; + +// These bounds define the minimum and maximum supported coordinate values. +// While visible coordinates are within [0, EXTENT], tiles may theoretically +// contain coordinates within [-Infinity, Infinity]. Our range is limited by the +// number of bits used to represent the coordinate. +const BITS = 15; +const MAX = Math.pow(2, BITS - 1) - 1; +const MIN = -MAX - 1; +/** + * Loads a geometry from a VectorTileFeature and scales it to the common extent + * used internally. + * @param feature - the vector tile feature to load + */ +function loadGeometry(feature) { + const scale = EXTENT / feature.extent; + const geometry = feature.loadGeometry(); + for (let r = 0; r < geometry.length; r++) { + const ring = geometry[r]; + for (let p = 0; p < ring.length; p++) { + const point = ring[p]; + // round here because mapbox-gl-native uses integers to represent + // points and we need to do the same to avoid renering differences. + const x = Math.round(point.x * scale); + const y = Math.round(point.y * scale); + point.x = clamp$1(x, MIN, MAX); + point.y = clamp$1(y, MIN, MAX); + if (x < point.x || x > point.x + 1 || y < point.y || y > point.y + 1) { + // warn when exceeding allowed extent except for the 1-px-off case + // https://github.com/mapbox/mapbox-gl-js/issues/8992 + warnOnce('Geometry exceeds allowed extent, reduce your vector tile buffer size'); + } + } + } + return geometry; +} + +/** + * Construct a new feature based on a VectorTileFeature for expression evaluation, the geometry of which + * will be loaded based on necessity. + * @param feature - the feature to evaluate + * @param needGeometry - if set to true this will load the geometry + */ +function toEvaluationFeature(feature, needGeometry) { + return { type: feature.type, + id: feature.id, + properties: feature.properties, + geometry: needGeometry ? loadGeometry(feature) : [] }; +} + +function addCircleVertex(layoutVertexArray, x, y, extrudeX, extrudeY) { + layoutVertexArray.emplaceBack((x * 2) + ((extrudeX + 1) / 2), (y * 2) + ((extrudeY + 1) / 2)); +} +/** + * @internal + * Circles are represented by two triangles. + * + * Each corner has a pos that is the center of the circle and an extrusion + * vector that is where it points. + */ +class CircleBucket { + constructor(options) { + this.zoom = options.zoom; + this.overscaling = options.overscaling; + this.layers = options.layers; + this.layerIds = this.layers.map(layer => layer.id); + this.index = options.index; + this.hasPattern = false; + this.layoutVertexArray = new CircleLayoutArray(); + this.indexArray = new TriangleIndexArray(); + this.segments = new SegmentVector(); + this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom); + this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id); + } + populate(features, options, canonical) { + const styleLayer = this.layers[0]; + const bucketFeatures = []; + let circleSortKey = null; + let sortFeaturesByKey = false; + // Heatmap layers are handled in this bucket and have no evaluated properties, so we check our access + if (styleLayer.type === 'circle') { + circleSortKey = styleLayer.layout.get('circle-sort-key'); + sortFeaturesByKey = !circleSortKey.isConstant(); + } + for (const { feature, id, index, sourceLayerIndex } of features) { + const needGeometry = this.layers[0]._featureFilter.needGeometry; + const evaluationFeature = toEvaluationFeature(feature, needGeometry); + if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) + continue; + const sortKey = sortFeaturesByKey ? + circleSortKey.evaluate(evaluationFeature, {}, canonical) : + undefined; + const bucketFeature = { + id, + properties: feature.properties, + type: feature.type, + sourceLayerIndex, + index, + geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature), + patterns: {}, + sortKey + }; + bucketFeatures.push(bucketFeature); + } + if (sortFeaturesByKey) { + bucketFeatures.sort((a, b) => a.sortKey - b.sortKey); + } + for (const bucketFeature of bucketFeatures) { + const { geometry, index, sourceLayerIndex } = bucketFeature; + const feature = features[index].feature; + this.addFeature(bucketFeature, geometry, index, canonical); + options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index); + } + } + update(states, vtLayer, imagePositions) { + if (!this.stateDependentLayers.length) + return; + this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions); + } + isEmpty() { + return this.layoutVertexArray.length === 0; + } + uploadPending() { + return !this.uploaded || this.programConfigurations.needsUpload; + } + upload(context) { + if (!this.uploaded) { + this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, members$4); + this.indexBuffer = context.createIndexBuffer(this.indexArray); + } + this.programConfigurations.upload(context); + this.uploaded = true; + } + destroy() { + if (!this.layoutVertexBuffer) + return; + this.layoutVertexBuffer.destroy(); + this.indexBuffer.destroy(); + this.programConfigurations.destroy(); + this.segments.destroy(); + } + addFeature(feature, geometry, index, canonical) { + for (const ring of geometry) { + for (const point of ring) { + const x = point.x; + const y = point.y; + // Do not include points that are outside the tile boundaries. + if (x < 0 || x >= EXTENT || y < 0 || y >= EXTENT) + continue; + // this geometry will be of the Point type, and we'll derive + // two triangles from it. + // + // ┌─────────┐ + // │ 3 2 │ + // │ │ + // │ 0 1 │ + // └─────────┘ + const segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray, feature.sortKey); + const index = segment.vertexLength; + addCircleVertex(this.layoutVertexArray, x, y, -1, -1); + addCircleVertex(this.layoutVertexArray, x, y, 1, -1); + addCircleVertex(this.layoutVertexArray, x, y, 1, 1); + addCircleVertex(this.layoutVertexArray, x, y, -1, 1); + this.indexArray.emplaceBack(index, index + 1, index + 2); + this.indexArray.emplaceBack(index, index + 3, index + 2); + segment.vertexLength += 4; + segment.primitiveLength += 2; + } + } + this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, {}, canonical); + } +} +register('CircleBucket', CircleBucket, { omit: ['layers'] }); + +function polygonIntersectsPolygon(polygonA, polygonB) { + for (let i = 0; i < polygonA.length; i++) { + if (polygonContainsPoint(polygonB, polygonA[i])) + return true; + } + for (let i = 0; i < polygonB.length; i++) { + if (polygonContainsPoint(polygonA, polygonB[i])) + return true; + } + if (lineIntersectsLine(polygonA, polygonB)) + return true; + return false; +} +function polygonIntersectsBufferedPoint(polygon, point, radius) { + if (polygonContainsPoint(polygon, point)) + return true; + if (pointIntersectsBufferedLine(point, polygon, radius)) + return true; + return false; +} +function polygonIntersectsMultiPolygon(polygon, multiPolygon) { + if (polygon.length === 1) { + return multiPolygonContainsPoint(multiPolygon, polygon[0]); + } + for (let m = 0; m < multiPolygon.length; m++) { + const ring = multiPolygon[m]; + for (let n = 0; n < ring.length; n++) { + if (polygonContainsPoint(polygon, ring[n])) + return true; + } + } + for (let i = 0; i < polygon.length; i++) { + if (multiPolygonContainsPoint(multiPolygon, polygon[i])) + return true; + } + for (let k = 0; k < multiPolygon.length; k++) { + if (lineIntersectsLine(polygon, multiPolygon[k])) + return true; + } + return false; +} +function polygonIntersectsBufferedMultiLine(polygon, multiLine, radius) { + for (let i = 0; i < multiLine.length; i++) { + const line = multiLine[i]; + if (polygon.length >= 3) { + for (let k = 0; k < line.length; k++) { + if (polygonContainsPoint(polygon, line[k])) + return true; + } + } + if (lineIntersectsBufferedLine(polygon, line, radius)) + return true; + } + return false; +} +function lineIntersectsBufferedLine(lineA, lineB, radius) { + if (lineA.length > 1) { + if (lineIntersectsLine(lineA, lineB)) + return true; + // Check whether any point in either line is within radius of the other line + for (let j = 0; j < lineB.length; j++) { + if (pointIntersectsBufferedLine(lineB[j], lineA, radius)) + return true; + } + } + for (let k = 0; k < lineA.length; k++) { + if (pointIntersectsBufferedLine(lineA[k], lineB, radius)) + return true; + } + return false; +} +function lineIntersectsLine(lineA, lineB) { + if (lineA.length === 0 || lineB.length === 0) + return false; + for (let i = 0; i < lineA.length - 1; i++) { + const a0 = lineA[i]; + const a1 = lineA[i + 1]; + for (let j = 0; j < lineB.length - 1; j++) { + const b0 = lineB[j]; + const b1 = lineB[j + 1]; + if (lineSegmentIntersectsLineSegment(a0, a1, b0, b1)) + return true; + } + } + return false; +} +function lineSegmentIntersectsLineSegment(a0, a1, b0, b1) { + return isCounterClockwise(a0, b0, b1) !== isCounterClockwise(a1, b0, b1) && + isCounterClockwise(a0, a1, b0) !== isCounterClockwise(a0, a1, b1); +} +function pointIntersectsBufferedLine(p, line, radius) { + const radiusSquared = radius * radius; + if (line.length === 1) + return p.distSqr(line[0]) < radiusSquared; + for (let i = 1; i < line.length; i++) { + // Find line segments that have a distance <= radius^2 to p + // In that case, we treat the line as "containing point p". + const v = line[i - 1], w = line[i]; + if (distToSegmentSquared(p, v, w) < radiusSquared) + return true; + } + return false; +} +// Code from http://stackoverflow.com/a/1501725/331379. +function distToSegmentSquared(p, v, w) { + const l2 = v.distSqr(w); + if (l2 === 0) + return p.distSqr(v); + const t = ((p.x - v.x) * (w.x - v.x) + (p.y - v.y) * (w.y - v.y)) / l2; + if (t < 0) + return p.distSqr(v); + if (t > 1) + return p.distSqr(w); + return p.distSqr(w.sub(v)._mult(t)._add(v)); +} +// point in polygon ray casting algorithm +function multiPolygonContainsPoint(rings, p) { + let c = false, ring, p1, p2; + for (let k = 0; k < rings.length; k++) { + ring = rings[k]; + for (let i = 0, j = ring.length - 1; i < ring.length; j = i++) { + p1 = ring[i]; + p2 = ring[j]; + if (((p1.y > p.y) !== (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) { + c = !c; + } + } + } + return c; +} +function polygonContainsPoint(ring, p) { + let c = false; + for (let i = 0, j = ring.length - 1; i < ring.length; j = i++) { + const p1 = ring[i]; + const p2 = ring[j]; + if (((p1.y > p.y) !== (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) { + c = !c; + } + } + return c; +} +function polygonIntersectsBox(ring, boxX1, boxY1, boxX2, boxY2) { + for (const p of ring) { + if (boxX1 <= p.x && + boxY1 <= p.y && + boxX2 >= p.x && + boxY2 >= p.y) + return true; + } + const corners = [ + new Point$2(boxX1, boxY1), + new Point$2(boxX1, boxY2), + new Point$2(boxX2, boxY2), + new Point$2(boxX2, boxY1) + ]; + if (ring.length > 2) { + for (const corner of corners) { + if (polygonContainsPoint(ring, corner)) + return true; + } + } + for (let i = 0; i < ring.length - 1; i++) { + const p1 = ring[i]; + const p2 = ring[i + 1]; + if (edgeIntersectsBox(p1, p2, corners)) + return true; + } + return false; +} +function edgeIntersectsBox(e1, e2, corners) { + const tl = corners[0]; + const br = corners[2]; + // the edge and box do not intersect in either the x or y dimensions + if (((e1.x < tl.x) && (e2.x < tl.x)) || + ((e1.x > br.x) && (e2.x > br.x)) || + ((e1.y < tl.y) && (e2.y < tl.y)) || + ((e1.y > br.y) && (e2.y > br.y))) + return false; + // check if all corners of the box are on the same side of the edge + const dir = isCounterClockwise(e1, e2, corners[0]); + return dir !== isCounterClockwise(e1, e2, corners[1]) || + dir !== isCounterClockwise(e1, e2, corners[2]) || + dir !== isCounterClockwise(e1, e2, corners[3]); +} + +function getMaximumPaintValue(property, layer, bucket) { + const value = layer.paint.get(property).value; + if (value.kind === 'constant') { + return value.value; + } + else { + return bucket.programConfigurations.get(layer.id).getMaxValue(property); + } +} +function translateDistance(translate) { + return Math.sqrt(translate[0] * translate[0] + translate[1] * translate[1]); +} +function translate$4(queryGeometry, translate, translateAnchor, bearing, pixelsToTileUnits) { + if (!translate[0] && !translate[1]) { + return queryGeometry; + } + const pt = Point$2.convert(translate)._mult(pixelsToTileUnits); + if (translateAnchor === 'viewport') { + pt._rotate(-bearing); + } + const translated = []; + for (let i = 0; i < queryGeometry.length; i++) { + const point = queryGeometry[i]; + translated.push(point.sub(pt)); + } + return translated; +} +function offsetLine(rings, offset) { + const newRings = []; + for (let ringIndex = 0; ringIndex < rings.length; ringIndex++) { + const ring = rings[ringIndex]; + const newRing = []; + for (let index = 0; index < ring.length; index++) { + const a = ring[index - 1]; + const b = ring[index]; + const c = ring[index + 1]; + const aToB = index === 0 ? new Point$2(0, 0) : b.sub(a)._unit()._perp(); + const bToC = index === ring.length - 1 ? new Point$2(0, 0) : c.sub(b)._unit()._perp(); + const extrude = aToB._add(bToC)._unit(); + const cosHalfAngle = extrude.x * bToC.x + extrude.y * bToC.y; + if (cosHalfAngle !== 0) { + extrude._mult(1 / cosHalfAngle); + } + newRing.push(extrude._mult(offset)._add(b)); + } + newRings.push(newRing); + } + return newRings; +} + +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ +let layout$5; +const getLayout$3 = () => layout$5 = layout$5 || new Properties({ + "circle-sort-key": new DataDrivenProperty(v8Spec["layout_circle"]["circle-sort-key"]), +}); +let paint$8; +const getPaint$8 = () => paint$8 = paint$8 || new Properties({ + "circle-radius": new DataDrivenProperty(v8Spec["paint_circle"]["circle-radius"]), + "circle-color": new DataDrivenProperty(v8Spec["paint_circle"]["circle-color"]), + "circle-blur": new DataDrivenProperty(v8Spec["paint_circle"]["circle-blur"]), + "circle-opacity": new DataDrivenProperty(v8Spec["paint_circle"]["circle-opacity"]), + "circle-translate": new DataConstantProperty(v8Spec["paint_circle"]["circle-translate"]), + "circle-translate-anchor": new DataConstantProperty(v8Spec["paint_circle"]["circle-translate-anchor"]), + "circle-pitch-scale": new DataConstantProperty(v8Spec["paint_circle"]["circle-pitch-scale"]), + "circle-pitch-alignment": new DataConstantProperty(v8Spec["paint_circle"]["circle-pitch-alignment"]), + "circle-stroke-width": new DataDrivenProperty(v8Spec["paint_circle"]["circle-stroke-width"]), + "circle-stroke-color": new DataDrivenProperty(v8Spec["paint_circle"]["circle-stroke-color"]), + "circle-stroke-opacity": new DataDrivenProperty(v8Spec["paint_circle"]["circle-stroke-opacity"]), +}); +var properties$8 = ({ get paint() { return getPaint$8(); }, get layout() { return getLayout$3(); } }); + +/** + * Common utilities + * @module glMatrix + */ +// Configuration Constants +var EPSILON = 0.000001; +var ARRAY_TYPE = typeof Float32Array !== 'undefined' ? Float32Array : Array; +var RANDOM = Math.random; +/** + * Sets the type of array used when creating new vectors and matrices + * + * @param {Float32ArrayConstructor | ArrayConstructor} type Array type, such as Float32Array or Array + */ + +function setMatrixArrayType(type) { + ARRAY_TYPE = type; +} +var degree = Math.PI / 180; +/** + * Convert Degree To Radian + * + * @param {Number} a Angle in Degrees + */ + +function toRadian(a) { + return a * degree; +} +/** + * Tests whether or not the arguments have approximately the same value, within an absolute + * or relative tolerance of glMatrix.EPSILON (an absolute tolerance is used for values less + * than or equal to 1.0, and a relative tolerance is used for larger values) + * + * @param {Number} a The first number to test. + * @param {Number} b The second number to test. + * @returns {Boolean} True if the numbers are approximately equal, false otherwise. + */ + +function equals$a(a, b) { + return Math.abs(a - b) <= EPSILON * Math.max(1.0, Math.abs(a), Math.abs(b)); +} +if (!Math.hypot) Math.hypot = function () { + var y = 0, + i = arguments.length; + + while (i--) { + y += arguments[i] * arguments[i]; + } + + return Math.sqrt(y); +}; + +var common = /*#__PURE__*/Object.freeze({ +__proto__: null, +get ARRAY_TYPE () { return ARRAY_TYPE; }, +EPSILON: EPSILON, +RANDOM: RANDOM, +equals: equals$a, +setMatrixArrayType: setMatrixArrayType, +toRadian: toRadian +}); + +/** + * 2x2 Matrix + * @module mat2 + */ + +/** + * Creates a new identity mat2 + * + * @returns {mat2} a new 2x2 matrix + */ + +function create$8() { + var out = new ARRAY_TYPE(4); + + if (ARRAY_TYPE != Float32Array) { + out[1] = 0; + out[2] = 0; + } + + out[0] = 1; + out[3] = 1; + return out; +} +/** + * Creates a new mat2 initialized with values from an existing matrix + * + * @param {ReadonlyMat2} a matrix to clone + * @returns {mat2} a new 2x2 matrix + */ + +function clone$8(a) { + var out = new ARRAY_TYPE(4); + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[3]; + return out; +} +/** + * Copy the values from one mat2 to another + * + * @param {mat2} out the receiving matrix + * @param {ReadonlyMat2} a the source matrix + * @returns {mat2} out + */ + +function copy$8(out, a) { + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[3]; + return out; +} +/** + * Set a mat2 to the identity matrix + * + * @param {mat2} out the receiving matrix + * @returns {mat2} out + */ + +function identity$5(out) { + out[0] = 1; + out[1] = 0; + out[2] = 0; + out[3] = 1; + return out; +} +/** + * Create a new mat2 with the given values + * + * @param {Number} m00 Component in column 0, row 0 position (index 0) + * @param {Number} m01 Component in column 0, row 1 position (index 1) + * @param {Number} m10 Component in column 1, row 0 position (index 2) + * @param {Number} m11 Component in column 1, row 1 position (index 3) + * @returns {mat2} out A new 2x2 matrix + */ + +function fromValues$8(m00, m01, m10, m11) { + var out = new ARRAY_TYPE(4); + out[0] = m00; + out[1] = m01; + out[2] = m10; + out[3] = m11; + return out; +} +/** + * Set the components of a mat2 to the given values + * + * @param {mat2} out the receiving matrix + * @param {Number} m00 Component in column 0, row 0 position (index 0) + * @param {Number} m01 Component in column 0, row 1 position (index 1) + * @param {Number} m10 Component in column 1, row 0 position (index 2) + * @param {Number} m11 Component in column 1, row 1 position (index 3) + * @returns {mat2} out + */ + +function set$8(out, m00, m01, m10, m11) { + out[0] = m00; + out[1] = m01; + out[2] = m10; + out[3] = m11; + return out; +} +/** + * Transpose the values of a mat2 + * + * @param {mat2} out the receiving matrix + * @param {ReadonlyMat2} a the source matrix + * @returns {mat2} out + */ + +function transpose$2(out, a) { + // If we are transposing ourselves we can skip a few steps but have to cache + // some values + if (out === a) { + var a1 = a[1]; + out[1] = a[2]; + out[2] = a1; + } else { + out[0] = a[0]; + out[1] = a[2]; + out[2] = a[1]; + out[3] = a[3]; + } + + return out; +} +/** + * Inverts a mat2 + * + * @param {mat2} out the receiving matrix + * @param {ReadonlyMat2} a the source matrix + * @returns {mat2} out + */ + +function invert$5(out, a) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3]; // Calculate the determinant + + var det = a0 * a3 - a2 * a1; + + if (!det) { + return null; + } + + det = 1.0 / det; + out[0] = a3 * det; + out[1] = -a1 * det; + out[2] = -a2 * det; + out[3] = a0 * det; + return out; +} +/** + * Calculates the adjugate of a mat2 + * + * @param {mat2} out the receiving matrix + * @param {ReadonlyMat2} a the source matrix + * @returns {mat2} out + */ + +function adjoint$2(out, a) { + // Caching this value is nessecary if out == a + var a0 = a[0]; + out[0] = a[3]; + out[1] = -a[1]; + out[2] = -a[2]; + out[3] = a0; + return out; +} +/** + * Calculates the determinant of a mat2 + * + * @param {ReadonlyMat2} a the source matrix + * @returns {Number} determinant of a + */ + +function determinant$3(a) { + return a[0] * a[3] - a[2] * a[1]; +} +/** + * Multiplies two mat2's + * + * @param {mat2} out the receiving matrix + * @param {ReadonlyMat2} a the first operand + * @param {ReadonlyMat2} b the second operand + * @returns {mat2} out + */ + +function multiply$8(out, a, b) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3]; + var b0 = b[0], + b1 = b[1], + b2 = b[2], + b3 = b[3]; + out[0] = a0 * b0 + a2 * b1; + out[1] = a1 * b0 + a3 * b1; + out[2] = a0 * b2 + a2 * b3; + out[3] = a1 * b2 + a3 * b3; + return out; +} +/** + * Rotates a mat2 by the given angle + * + * @param {mat2} out the receiving matrix + * @param {ReadonlyMat2} a the matrix to rotate + * @param {Number} rad the angle to rotate the matrix by + * @returns {mat2} out + */ + +function rotate$4(out, a, rad) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3]; + var s = Math.sin(rad); + var c = Math.cos(rad); + out[0] = a0 * c + a2 * s; + out[1] = a1 * c + a3 * s; + out[2] = a0 * -s + a2 * c; + out[3] = a1 * -s + a3 * c; + return out; +} +/** + * Scales the mat2 by the dimensions in the given vec2 + * + * @param {mat2} out the receiving matrix + * @param {ReadonlyMat2} a the matrix to rotate + * @param {ReadonlyVec2} v the vec2 to scale the matrix by + * @returns {mat2} out + **/ + +function scale$8(out, a, v) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3]; + var v0 = v[0], + v1 = v[1]; + out[0] = a0 * v0; + out[1] = a1 * v0; + out[2] = a2 * v1; + out[3] = a3 * v1; + return out; +} +/** + * Creates a matrix from a given angle + * This is equivalent to (but much faster than): + * + * mat2.identity(dest); + * mat2.rotate(dest, dest, rad); + * + * @param {mat2} out mat2 receiving operation result + * @param {Number} rad the angle to rotate the matrix by + * @returns {mat2} out + */ + +function fromRotation$4(out, rad) { + var s = Math.sin(rad); + var c = Math.cos(rad); + out[0] = c; + out[1] = s; + out[2] = -s; + out[3] = c; + return out; +} +/** + * Creates a matrix from a vector scaling + * This is equivalent to (but much faster than): + * + * mat2.identity(dest); + * mat2.scale(dest, dest, vec); + * + * @param {mat2} out mat2 receiving operation result + * @param {ReadonlyVec2} v Scaling vector + * @returns {mat2} out + */ + +function fromScaling$3(out, v) { + out[0] = v[0]; + out[1] = 0; + out[2] = 0; + out[3] = v[1]; + return out; +} +/** + * Returns a string representation of a mat2 + * + * @param {ReadonlyMat2} a matrix to represent as a string + * @returns {String} string representation of the matrix + */ + +function str$8(a) { + return "mat2(" + a[0] + ", " + a[1] + ", " + a[2] + ", " + a[3] + ")"; +} +/** + * Returns Frobenius norm of a mat2 + * + * @param {ReadonlyMat2} a the matrix to calculate Frobenius norm of + * @returns {Number} Frobenius norm + */ + +function frob$3(a) { + return Math.hypot(a[0], a[1], a[2], a[3]); +} +/** + * Returns L, D and U matrices (Lower triangular, Diagonal and Upper triangular) by factorizing the input matrix + * @param {ReadonlyMat2} L the lower triangular matrix + * @param {ReadonlyMat2} D the diagonal matrix + * @param {ReadonlyMat2} U the upper triangular matrix + * @param {ReadonlyMat2} a the input matrix to factorize + */ + +function LDU(L, D, U, a) { + L[2] = a[2] / a[0]; + U[0] = a[0]; + U[1] = a[1]; + U[3] = a[3] - L[2] * U[1]; + return [L, D, U]; +} +/** + * Adds two mat2's + * + * @param {mat2} out the receiving matrix + * @param {ReadonlyMat2} a the first operand + * @param {ReadonlyMat2} b the second operand + * @returns {mat2} out + */ + +function add$8(out, a, b) { + out[0] = a[0] + b[0]; + out[1] = a[1] + b[1]; + out[2] = a[2] + b[2]; + out[3] = a[3] + b[3]; + return out; +} +/** + * Subtracts matrix b from matrix a + * + * @param {mat2} out the receiving matrix + * @param {ReadonlyMat2} a the first operand + * @param {ReadonlyMat2} b the second operand + * @returns {mat2} out + */ + +function subtract$6(out, a, b) { + out[0] = a[0] - b[0]; + out[1] = a[1] - b[1]; + out[2] = a[2] - b[2]; + out[3] = a[3] - b[3]; + return out; +} +/** + * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===) + * + * @param {ReadonlyMat2} a The first matrix. + * @param {ReadonlyMat2} b The second matrix. + * @returns {Boolean} True if the matrices are equal, false otherwise. + */ + +function exactEquals$8(a, b) { + return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3]; +} +/** + * Returns whether or not the matrices have approximately the same elements in the same position. + * + * @param {ReadonlyMat2} a The first matrix. + * @param {ReadonlyMat2} b The second matrix. + * @returns {Boolean} True if the matrices are equal, false otherwise. + */ + +function equals$9(a, b) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3]; + var b0 = b[0], + b1 = b[1], + b2 = b[2], + b3 = b[3]; + return Math.abs(a0 - b0) <= EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)); +} +/** + * Multiply each element of the matrix by a scalar. + * + * @param {mat2} out the receiving matrix + * @param {ReadonlyMat2} a the matrix to scale + * @param {Number} b amount to scale the matrix's elements by + * @returns {mat2} out + */ + +function multiplyScalar$3(out, a, b) { + out[0] = a[0] * b; + out[1] = a[1] * b; + out[2] = a[2] * b; + out[3] = a[3] * b; + return out; +} +/** + * Adds two mat2's after multiplying each element of the second operand by a scalar value. + * + * @param {mat2} out the receiving vector + * @param {ReadonlyMat2} a the first operand + * @param {ReadonlyMat2} b the second operand + * @param {Number} scale the amount to scale b's elements by before adding + * @returns {mat2} out + */ + +function multiplyScalarAndAdd$3(out, a, b, scale) { + out[0] = a[0] + b[0] * scale; + out[1] = a[1] + b[1] * scale; + out[2] = a[2] + b[2] * scale; + out[3] = a[3] + b[3] * scale; + return out; +} +/** + * Alias for {@link mat2.multiply} + * @function + */ + +var mul$8 = multiply$8; +/** + * Alias for {@link mat2.subtract} + * @function + */ + +var sub$6 = subtract$6; + +var mat2 = /*#__PURE__*/Object.freeze({ +__proto__: null, +LDU: LDU, +add: add$8, +adjoint: adjoint$2, +clone: clone$8, +copy: copy$8, +create: create$8, +determinant: determinant$3, +equals: equals$9, +exactEquals: exactEquals$8, +frob: frob$3, +fromRotation: fromRotation$4, +fromScaling: fromScaling$3, +fromValues: fromValues$8, +identity: identity$5, +invert: invert$5, +mul: mul$8, +multiply: multiply$8, +multiplyScalar: multiplyScalar$3, +multiplyScalarAndAdd: multiplyScalarAndAdd$3, +rotate: rotate$4, +scale: scale$8, +set: set$8, +str: str$8, +sub: sub$6, +subtract: subtract$6, +transpose: transpose$2 +}); + +/** + * 2x3 Matrix + * @module mat2d + * @description + * A mat2d contains six elements defined as: + *
+ * [a, b,
+ *  c, d,
+ *  tx, ty]
+ * 
+ * This is a short form for the 3x3 matrix: + *
+ * [a, b, 0,
+ *  c, d, 0,
+ *  tx, ty, 1]
+ * 
+ * The last column is ignored so the array is shorter and operations are faster. + */ + +/** + * Creates a new identity mat2d + * + * @returns {mat2d} a new 2x3 matrix + */ + +function create$7() { + var out = new ARRAY_TYPE(6); + + if (ARRAY_TYPE != Float32Array) { + out[1] = 0; + out[2] = 0; + out[4] = 0; + out[5] = 0; + } + + out[0] = 1; + out[3] = 1; + return out; +} +/** + * Creates a new mat2d initialized with values from an existing matrix + * + * @param {ReadonlyMat2d} a matrix to clone + * @returns {mat2d} a new 2x3 matrix + */ + +function clone$7(a) { + var out = new ARRAY_TYPE(6); + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[3]; + out[4] = a[4]; + out[5] = a[5]; + return out; +} +/** + * Copy the values from one mat2d to another + * + * @param {mat2d} out the receiving matrix + * @param {ReadonlyMat2d} a the source matrix + * @returns {mat2d} out + */ + +function copy$7(out, a) { + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[3]; + out[4] = a[4]; + out[5] = a[5]; + return out; +} +/** + * Set a mat2d to the identity matrix + * + * @param {mat2d} out the receiving matrix + * @returns {mat2d} out + */ + +function identity$4(out) { + out[0] = 1; + out[1] = 0; + out[2] = 0; + out[3] = 1; + out[4] = 0; + out[5] = 0; + return out; +} +/** + * Create a new mat2d with the given values + * + * @param {Number} a Component A (index 0) + * @param {Number} b Component B (index 1) + * @param {Number} c Component C (index 2) + * @param {Number} d Component D (index 3) + * @param {Number} tx Component TX (index 4) + * @param {Number} ty Component TY (index 5) + * @returns {mat2d} A new mat2d + */ + +function fromValues$7(a, b, c, d, tx, ty) { + var out = new ARRAY_TYPE(6); + out[0] = a; + out[1] = b; + out[2] = c; + out[3] = d; + out[4] = tx; + out[5] = ty; + return out; +} +/** + * Set the components of a mat2d to the given values + * + * @param {mat2d} out the receiving matrix + * @param {Number} a Component A (index 0) + * @param {Number} b Component B (index 1) + * @param {Number} c Component C (index 2) + * @param {Number} d Component D (index 3) + * @param {Number} tx Component TX (index 4) + * @param {Number} ty Component TY (index 5) + * @returns {mat2d} out + */ + +function set$7(out, a, b, c, d, tx, ty) { + out[0] = a; + out[1] = b; + out[2] = c; + out[3] = d; + out[4] = tx; + out[5] = ty; + return out; +} +/** + * Inverts a mat2d + * + * @param {mat2d} out the receiving matrix + * @param {ReadonlyMat2d} a the source matrix + * @returns {mat2d} out + */ + +function invert$4(out, a) { + var aa = a[0], + ab = a[1], + ac = a[2], + ad = a[3]; + var atx = a[4], + aty = a[5]; + var det = aa * ad - ab * ac; + + if (!det) { + return null; + } + + det = 1.0 / det; + out[0] = ad * det; + out[1] = -ab * det; + out[2] = -ac * det; + out[3] = aa * det; + out[4] = (ac * aty - ad * atx) * det; + out[5] = (ab * atx - aa * aty) * det; + return out; +} +/** + * Calculates the determinant of a mat2d + * + * @param {ReadonlyMat2d} a the source matrix + * @returns {Number} determinant of a + */ + +function determinant$2(a) { + return a[0] * a[3] - a[1] * a[2]; +} +/** + * Multiplies two mat2d's + * + * @param {mat2d} out the receiving matrix + * @param {ReadonlyMat2d} a the first operand + * @param {ReadonlyMat2d} b the second operand + * @returns {mat2d} out + */ + +function multiply$7(out, a, b) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3], + a4 = a[4], + a5 = a[5]; + var b0 = b[0], + b1 = b[1], + b2 = b[2], + b3 = b[3], + b4 = b[4], + b5 = b[5]; + out[0] = a0 * b0 + a2 * b1; + out[1] = a1 * b0 + a3 * b1; + out[2] = a0 * b2 + a2 * b3; + out[3] = a1 * b2 + a3 * b3; + out[4] = a0 * b4 + a2 * b5 + a4; + out[5] = a1 * b4 + a3 * b5 + a5; + return out; +} +/** + * Rotates a mat2d by the given angle + * + * @param {mat2d} out the receiving matrix + * @param {ReadonlyMat2d} a the matrix to rotate + * @param {Number} rad the angle to rotate the matrix by + * @returns {mat2d} out + */ + +function rotate$3(out, a, rad) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3], + a4 = a[4], + a5 = a[5]; + var s = Math.sin(rad); + var c = Math.cos(rad); + out[0] = a0 * c + a2 * s; + out[1] = a1 * c + a3 * s; + out[2] = a0 * -s + a2 * c; + out[3] = a1 * -s + a3 * c; + out[4] = a4; + out[5] = a5; + return out; +} +/** + * Scales the mat2d by the dimensions in the given vec2 + * + * @param {mat2d} out the receiving matrix + * @param {ReadonlyMat2d} a the matrix to translate + * @param {ReadonlyVec2} v the vec2 to scale the matrix by + * @returns {mat2d} out + **/ + +function scale$7(out, a, v) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3], + a4 = a[4], + a5 = a[5]; + var v0 = v[0], + v1 = v[1]; + out[0] = a0 * v0; + out[1] = a1 * v0; + out[2] = a2 * v1; + out[3] = a3 * v1; + out[4] = a4; + out[5] = a5; + return out; +} +/** + * Translates the mat2d by the dimensions in the given vec2 + * + * @param {mat2d} out the receiving matrix + * @param {ReadonlyMat2d} a the matrix to translate + * @param {ReadonlyVec2} v the vec2 to translate the matrix by + * @returns {mat2d} out + **/ + +function translate$3(out, a, v) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3], + a4 = a[4], + a5 = a[5]; + var v0 = v[0], + v1 = v[1]; + out[0] = a0; + out[1] = a1; + out[2] = a2; + out[3] = a3; + out[4] = a0 * v0 + a2 * v1 + a4; + out[5] = a1 * v0 + a3 * v1 + a5; + return out; +} +/** + * Creates a matrix from a given angle + * This is equivalent to (but much faster than): + * + * mat2d.identity(dest); + * mat2d.rotate(dest, dest, rad); + * + * @param {mat2d} out mat2d receiving operation result + * @param {Number} rad the angle to rotate the matrix by + * @returns {mat2d} out + */ + +function fromRotation$3(out, rad) { + var s = Math.sin(rad), + c = Math.cos(rad); + out[0] = c; + out[1] = s; + out[2] = -s; + out[3] = c; + out[4] = 0; + out[5] = 0; + return out; +} +/** + * Creates a matrix from a vector scaling + * This is equivalent to (but much faster than): + * + * mat2d.identity(dest); + * mat2d.scale(dest, dest, vec); + * + * @param {mat2d} out mat2d receiving operation result + * @param {ReadonlyVec2} v Scaling vector + * @returns {mat2d} out + */ + +function fromScaling$2(out, v) { + out[0] = v[0]; + out[1] = 0; + out[2] = 0; + out[3] = v[1]; + out[4] = 0; + out[5] = 0; + return out; +} +/** + * Creates a matrix from a vector translation + * This is equivalent to (but much faster than): + * + * mat2d.identity(dest); + * mat2d.translate(dest, dest, vec); + * + * @param {mat2d} out mat2d receiving operation result + * @param {ReadonlyVec2} v Translation vector + * @returns {mat2d} out + */ + +function fromTranslation$3(out, v) { + out[0] = 1; + out[1] = 0; + out[2] = 0; + out[3] = 1; + out[4] = v[0]; + out[5] = v[1]; + return out; +} +/** + * Returns a string representation of a mat2d + * + * @param {ReadonlyMat2d} a matrix to represent as a string + * @returns {String} string representation of the matrix + */ + +function str$7(a) { + return "mat2d(" + a[0] + ", " + a[1] + ", " + a[2] + ", " + a[3] + ", " + a[4] + ", " + a[5] + ")"; +} +/** + * Returns Frobenius norm of a mat2d + * + * @param {ReadonlyMat2d} a the matrix to calculate Frobenius norm of + * @returns {Number} Frobenius norm + */ + +function frob$2(a) { + return Math.hypot(a[0], a[1], a[2], a[3], a[4], a[5], 1); +} +/** + * Adds two mat2d's + * + * @param {mat2d} out the receiving matrix + * @param {ReadonlyMat2d} a the first operand + * @param {ReadonlyMat2d} b the second operand + * @returns {mat2d} out + */ + +function add$7(out, a, b) { + out[0] = a[0] + b[0]; + out[1] = a[1] + b[1]; + out[2] = a[2] + b[2]; + out[3] = a[3] + b[3]; + out[4] = a[4] + b[4]; + out[5] = a[5] + b[5]; + return out; +} +/** + * Subtracts matrix b from matrix a + * + * @param {mat2d} out the receiving matrix + * @param {ReadonlyMat2d} a the first operand + * @param {ReadonlyMat2d} b the second operand + * @returns {mat2d} out + */ + +function subtract$5(out, a, b) { + out[0] = a[0] - b[0]; + out[1] = a[1] - b[1]; + out[2] = a[2] - b[2]; + out[3] = a[3] - b[3]; + out[4] = a[4] - b[4]; + out[5] = a[5] - b[5]; + return out; +} +/** + * Multiply each element of the matrix by a scalar. + * + * @param {mat2d} out the receiving matrix + * @param {ReadonlyMat2d} a the matrix to scale + * @param {Number} b amount to scale the matrix's elements by + * @returns {mat2d} out + */ + +function multiplyScalar$2(out, a, b) { + out[0] = a[0] * b; + out[1] = a[1] * b; + out[2] = a[2] * b; + out[3] = a[3] * b; + out[4] = a[4] * b; + out[5] = a[5] * b; + return out; +} +/** + * Adds two mat2d's after multiplying each element of the second operand by a scalar value. + * + * @param {mat2d} out the receiving vector + * @param {ReadonlyMat2d} a the first operand + * @param {ReadonlyMat2d} b the second operand + * @param {Number} scale the amount to scale b's elements by before adding + * @returns {mat2d} out + */ + +function multiplyScalarAndAdd$2(out, a, b, scale) { + out[0] = a[0] + b[0] * scale; + out[1] = a[1] + b[1] * scale; + out[2] = a[2] + b[2] * scale; + out[3] = a[3] + b[3] * scale; + out[4] = a[4] + b[4] * scale; + out[5] = a[5] + b[5] * scale; + return out; +} +/** + * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===) + * + * @param {ReadonlyMat2d} a The first matrix. + * @param {ReadonlyMat2d} b The second matrix. + * @returns {Boolean} True if the matrices are equal, false otherwise. + */ + +function exactEquals$7(a, b) { + return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5]; +} +/** + * Returns whether or not the matrices have approximately the same elements in the same position. + * + * @param {ReadonlyMat2d} a The first matrix. + * @param {ReadonlyMat2d} b The second matrix. + * @returns {Boolean} True if the matrices are equal, false otherwise. + */ + +function equals$8(a, b) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3], + a4 = a[4], + a5 = a[5]; + var b0 = b[0], + b1 = b[1], + b2 = b[2], + b3 = b[3], + b4 = b[4], + b5 = b[5]; + return Math.abs(a0 - b0) <= EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5)); +} +/** + * Alias for {@link mat2d.multiply} + * @function + */ + +var mul$7 = multiply$7; +/** + * Alias for {@link mat2d.subtract} + * @function + */ + +var sub$5 = subtract$5; + +var mat2d = /*#__PURE__*/Object.freeze({ +__proto__: null, +add: add$7, +clone: clone$7, +copy: copy$7, +create: create$7, +determinant: determinant$2, +equals: equals$8, +exactEquals: exactEquals$7, +frob: frob$2, +fromRotation: fromRotation$3, +fromScaling: fromScaling$2, +fromTranslation: fromTranslation$3, +fromValues: fromValues$7, +identity: identity$4, +invert: invert$4, +mul: mul$7, +multiply: multiply$7, +multiplyScalar: multiplyScalar$2, +multiplyScalarAndAdd: multiplyScalarAndAdd$2, +rotate: rotate$3, +scale: scale$7, +set: set$7, +str: str$7, +sub: sub$5, +subtract: subtract$5, +translate: translate$3 +}); + +/** + * 3x3 Matrix + * @module mat3 + */ + +/** + * Creates a new identity mat3 + * + * @returns {mat3} a new 3x3 matrix + */ + +function create$6() { + var out = new ARRAY_TYPE(9); + + if (ARRAY_TYPE != Float32Array) { + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[5] = 0; + out[6] = 0; + out[7] = 0; + } + + out[0] = 1; + out[4] = 1; + out[8] = 1; + return out; +} +/** + * Copies the upper-left 3x3 values into the given mat3. + * + * @param {mat3} out the receiving 3x3 matrix + * @param {ReadonlyMat4} a the source 4x4 matrix + * @returns {mat3} out + */ + +function fromMat4$1(out, a) { + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[4]; + out[4] = a[5]; + out[5] = a[6]; + out[6] = a[8]; + out[7] = a[9]; + out[8] = a[10]; + return out; +} +/** + * Creates a new mat3 initialized with values from an existing matrix + * + * @param {ReadonlyMat3} a matrix to clone + * @returns {mat3} a new 3x3 matrix + */ + +function clone$6(a) { + var out = new ARRAY_TYPE(9); + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[3]; + out[4] = a[4]; + out[5] = a[5]; + out[6] = a[6]; + out[7] = a[7]; + out[8] = a[8]; + return out; +} +/** + * Copy the values from one mat3 to another + * + * @param {mat3} out the receiving matrix + * @param {ReadonlyMat3} a the source matrix + * @returns {mat3} out + */ + +function copy$6(out, a) { + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[3]; + out[4] = a[4]; + out[5] = a[5]; + out[6] = a[6]; + out[7] = a[7]; + out[8] = a[8]; + return out; +} +/** + * Create a new mat3 with the given values + * + * @param {Number} m00 Component in column 0, row 0 position (index 0) + * @param {Number} m01 Component in column 0, row 1 position (index 1) + * @param {Number} m02 Component in column 0, row 2 position (index 2) + * @param {Number} m10 Component in column 1, row 0 position (index 3) + * @param {Number} m11 Component in column 1, row 1 position (index 4) + * @param {Number} m12 Component in column 1, row 2 position (index 5) + * @param {Number} m20 Component in column 2, row 0 position (index 6) + * @param {Number} m21 Component in column 2, row 1 position (index 7) + * @param {Number} m22 Component in column 2, row 2 position (index 8) + * @returns {mat3} A new mat3 + */ + +function fromValues$6(m00, m01, m02, m10, m11, m12, m20, m21, m22) { + var out = new ARRAY_TYPE(9); + out[0] = m00; + out[1] = m01; + out[2] = m02; + out[3] = m10; + out[4] = m11; + out[5] = m12; + out[6] = m20; + out[7] = m21; + out[8] = m22; + return out; +} +/** + * Set the components of a mat3 to the given values + * + * @param {mat3} out the receiving matrix + * @param {Number} m00 Component in column 0, row 0 position (index 0) + * @param {Number} m01 Component in column 0, row 1 position (index 1) + * @param {Number} m02 Component in column 0, row 2 position (index 2) + * @param {Number} m10 Component in column 1, row 0 position (index 3) + * @param {Number} m11 Component in column 1, row 1 position (index 4) + * @param {Number} m12 Component in column 1, row 2 position (index 5) + * @param {Number} m20 Component in column 2, row 0 position (index 6) + * @param {Number} m21 Component in column 2, row 1 position (index 7) + * @param {Number} m22 Component in column 2, row 2 position (index 8) + * @returns {mat3} out + */ + +function set$6(out, m00, m01, m02, m10, m11, m12, m20, m21, m22) { + out[0] = m00; + out[1] = m01; + out[2] = m02; + out[3] = m10; + out[4] = m11; + out[5] = m12; + out[6] = m20; + out[7] = m21; + out[8] = m22; + return out; +} +/** + * Set a mat3 to the identity matrix + * + * @param {mat3} out the receiving matrix + * @returns {mat3} out + */ + +function identity$3(out) { + out[0] = 1; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = 1; + out[5] = 0; + out[6] = 0; + out[7] = 0; + out[8] = 1; + return out; +} +/** + * Transpose the values of a mat3 + * + * @param {mat3} out the receiving matrix + * @param {ReadonlyMat3} a the source matrix + * @returns {mat3} out + */ + +function transpose$1(out, a) { + // If we are transposing ourselves we can skip a few steps but have to cache some values + if (out === a) { + var a01 = a[1], + a02 = a[2], + a12 = a[5]; + out[1] = a[3]; + out[2] = a[6]; + out[3] = a01; + out[5] = a[7]; + out[6] = a02; + out[7] = a12; + } else { + out[0] = a[0]; + out[1] = a[3]; + out[2] = a[6]; + out[3] = a[1]; + out[4] = a[4]; + out[5] = a[7]; + out[6] = a[2]; + out[7] = a[5]; + out[8] = a[8]; + } + + return out; +} +/** + * Inverts a mat3 + * + * @param {mat3} out the receiving matrix + * @param {ReadonlyMat3} a the source matrix + * @returns {mat3} out + */ + +function invert$3(out, a) { + var a00 = a[0], + a01 = a[1], + a02 = a[2]; + var a10 = a[3], + a11 = a[4], + a12 = a[5]; + var a20 = a[6], + a21 = a[7], + a22 = a[8]; + var b01 = a22 * a11 - a12 * a21; + var b11 = -a22 * a10 + a12 * a20; + var b21 = a21 * a10 - a11 * a20; // Calculate the determinant + + var det = a00 * b01 + a01 * b11 + a02 * b21; + + if (!det) { + return null; + } + + det = 1.0 / det; + out[0] = b01 * det; + out[1] = (-a22 * a01 + a02 * a21) * det; + out[2] = (a12 * a01 - a02 * a11) * det; + out[3] = b11 * det; + out[4] = (a22 * a00 - a02 * a20) * det; + out[5] = (-a12 * a00 + a02 * a10) * det; + out[6] = b21 * det; + out[7] = (-a21 * a00 + a01 * a20) * det; + out[8] = (a11 * a00 - a01 * a10) * det; + return out; +} +/** + * Calculates the adjugate of a mat3 + * + * @param {mat3} out the receiving matrix + * @param {ReadonlyMat3} a the source matrix + * @returns {mat3} out + */ + +function adjoint$1(out, a) { + var a00 = a[0], + a01 = a[1], + a02 = a[2]; + var a10 = a[3], + a11 = a[4], + a12 = a[5]; + var a20 = a[6], + a21 = a[7], + a22 = a[8]; + out[0] = a11 * a22 - a12 * a21; + out[1] = a02 * a21 - a01 * a22; + out[2] = a01 * a12 - a02 * a11; + out[3] = a12 * a20 - a10 * a22; + out[4] = a00 * a22 - a02 * a20; + out[5] = a02 * a10 - a00 * a12; + out[6] = a10 * a21 - a11 * a20; + out[7] = a01 * a20 - a00 * a21; + out[8] = a00 * a11 - a01 * a10; + return out; +} +/** + * Calculates the determinant of a mat3 + * + * @param {ReadonlyMat3} a the source matrix + * @returns {Number} determinant of a + */ + +function determinant$1(a) { + var a00 = a[0], + a01 = a[1], + a02 = a[2]; + var a10 = a[3], + a11 = a[4], + a12 = a[5]; + var a20 = a[6], + a21 = a[7], + a22 = a[8]; + return a00 * (a22 * a11 - a12 * a21) + a01 * (-a22 * a10 + a12 * a20) + a02 * (a21 * a10 - a11 * a20); +} +/** + * Multiplies two mat3's + * + * @param {mat3} out the receiving matrix + * @param {ReadonlyMat3} a the first operand + * @param {ReadonlyMat3} b the second operand + * @returns {mat3} out + */ + +function multiply$6(out, a, b) { + var a00 = a[0], + a01 = a[1], + a02 = a[2]; + var a10 = a[3], + a11 = a[4], + a12 = a[5]; + var a20 = a[6], + a21 = a[7], + a22 = a[8]; + var b00 = b[0], + b01 = b[1], + b02 = b[2]; + var b10 = b[3], + b11 = b[4], + b12 = b[5]; + var b20 = b[6], + b21 = b[7], + b22 = b[8]; + out[0] = b00 * a00 + b01 * a10 + b02 * a20; + out[1] = b00 * a01 + b01 * a11 + b02 * a21; + out[2] = b00 * a02 + b01 * a12 + b02 * a22; + out[3] = b10 * a00 + b11 * a10 + b12 * a20; + out[4] = b10 * a01 + b11 * a11 + b12 * a21; + out[5] = b10 * a02 + b11 * a12 + b12 * a22; + out[6] = b20 * a00 + b21 * a10 + b22 * a20; + out[7] = b20 * a01 + b21 * a11 + b22 * a21; + out[8] = b20 * a02 + b21 * a12 + b22 * a22; + return out; +} +/** + * Translate a mat3 by the given vector + * + * @param {mat3} out the receiving matrix + * @param {ReadonlyMat3} a the matrix to translate + * @param {ReadonlyVec2} v vector to translate by + * @returns {mat3} out + */ + +function translate$2(out, a, v) { + var a00 = a[0], + a01 = a[1], + a02 = a[2], + a10 = a[3], + a11 = a[4], + a12 = a[5], + a20 = a[6], + a21 = a[7], + a22 = a[8], + x = v[0], + y = v[1]; + out[0] = a00; + out[1] = a01; + out[2] = a02; + out[3] = a10; + out[4] = a11; + out[5] = a12; + out[6] = x * a00 + y * a10 + a20; + out[7] = x * a01 + y * a11 + a21; + out[8] = x * a02 + y * a12 + a22; + return out; +} +/** + * Rotates a mat3 by the given angle + * + * @param {mat3} out the receiving matrix + * @param {ReadonlyMat3} a the matrix to rotate + * @param {Number} rad the angle to rotate the matrix by + * @returns {mat3} out + */ + +function rotate$2(out, a, rad) { + var a00 = a[0], + a01 = a[1], + a02 = a[2], + a10 = a[3], + a11 = a[4], + a12 = a[5], + a20 = a[6], + a21 = a[7], + a22 = a[8], + s = Math.sin(rad), + c = Math.cos(rad); + out[0] = c * a00 + s * a10; + out[1] = c * a01 + s * a11; + out[2] = c * a02 + s * a12; + out[3] = c * a10 - s * a00; + out[4] = c * a11 - s * a01; + out[5] = c * a12 - s * a02; + out[6] = a20; + out[7] = a21; + out[8] = a22; + return out; +} +/** + * Scales the mat3 by the dimensions in the given vec2 + * + * @param {mat3} out the receiving matrix + * @param {ReadonlyMat3} a the matrix to rotate + * @param {ReadonlyVec2} v the vec2 to scale the matrix by + * @returns {mat3} out + **/ + +function scale$6(out, a, v) { + var x = v[0], + y = v[1]; + out[0] = x * a[0]; + out[1] = x * a[1]; + out[2] = x * a[2]; + out[3] = y * a[3]; + out[4] = y * a[4]; + out[5] = y * a[5]; + out[6] = a[6]; + out[7] = a[7]; + out[8] = a[8]; + return out; +} +/** + * Creates a matrix from a vector translation + * This is equivalent to (but much faster than): + * + * mat3.identity(dest); + * mat3.translate(dest, dest, vec); + * + * @param {mat3} out mat3 receiving operation result + * @param {ReadonlyVec2} v Translation vector + * @returns {mat3} out + */ + +function fromTranslation$2(out, v) { + out[0] = 1; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = 1; + out[5] = 0; + out[6] = v[0]; + out[7] = v[1]; + out[8] = 1; + return out; +} +/** + * Creates a matrix from a given angle + * This is equivalent to (but much faster than): + * + * mat3.identity(dest); + * mat3.rotate(dest, dest, rad); + * + * @param {mat3} out mat3 receiving operation result + * @param {Number} rad the angle to rotate the matrix by + * @returns {mat3} out + */ + +function fromRotation$2(out, rad) { + var s = Math.sin(rad), + c = Math.cos(rad); + out[0] = c; + out[1] = s; + out[2] = 0; + out[3] = -s; + out[4] = c; + out[5] = 0; + out[6] = 0; + out[7] = 0; + out[8] = 1; + return out; +} +/** + * Creates a matrix from a vector scaling + * This is equivalent to (but much faster than): + * + * mat3.identity(dest); + * mat3.scale(dest, dest, vec); + * + * @param {mat3} out mat3 receiving operation result + * @param {ReadonlyVec2} v Scaling vector + * @returns {mat3} out + */ + +function fromScaling$1(out, v) { + out[0] = v[0]; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = v[1]; + out[5] = 0; + out[6] = 0; + out[7] = 0; + out[8] = 1; + return out; +} +/** + * Copies the values from a mat2d into a mat3 + * + * @param {mat3} out the receiving matrix + * @param {ReadonlyMat2d} a the matrix to copy + * @returns {mat3} out + **/ + +function fromMat2d(out, a) { + out[0] = a[0]; + out[1] = a[1]; + out[2] = 0; + out[3] = a[2]; + out[4] = a[3]; + out[5] = 0; + out[6] = a[4]; + out[7] = a[5]; + out[8] = 1; + return out; +} +/** + * Calculates a 3x3 matrix from the given quaternion + * + * @param {mat3} out mat3 receiving operation result + * @param {ReadonlyQuat} q Quaternion to create matrix from + * + * @returns {mat3} out + */ + +function fromQuat$1(out, q) { + var x = q[0], + y = q[1], + z = q[2], + w = q[3]; + var x2 = x + x; + var y2 = y + y; + var z2 = z + z; + var xx = x * x2; + var yx = y * x2; + var yy = y * y2; + var zx = z * x2; + var zy = z * y2; + var zz = z * z2; + var wx = w * x2; + var wy = w * y2; + var wz = w * z2; + out[0] = 1 - yy - zz; + out[3] = yx - wz; + out[6] = zx + wy; + out[1] = yx + wz; + out[4] = 1 - xx - zz; + out[7] = zy - wx; + out[2] = zx - wy; + out[5] = zy + wx; + out[8] = 1 - xx - yy; + return out; +} +/** + * Calculates a 3x3 normal matrix (transpose inverse) from the 4x4 matrix + * + * @param {mat3} out mat3 receiving operation result + * @param {ReadonlyMat4} a Mat4 to derive the normal matrix from + * + * @returns {mat3} out + */ + +function normalFromMat4(out, a) { + var a00 = a[0], + a01 = a[1], + a02 = a[2], + a03 = a[3]; + var a10 = a[4], + a11 = a[5], + a12 = a[6], + a13 = a[7]; + var a20 = a[8], + a21 = a[9], + a22 = a[10], + a23 = a[11]; + var a30 = a[12], + a31 = a[13], + a32 = a[14], + a33 = a[15]; + var b00 = a00 * a11 - a01 * a10; + var b01 = a00 * a12 - a02 * a10; + var b02 = a00 * a13 - a03 * a10; + var b03 = a01 * a12 - a02 * a11; + var b04 = a01 * a13 - a03 * a11; + var b05 = a02 * a13 - a03 * a12; + var b06 = a20 * a31 - a21 * a30; + var b07 = a20 * a32 - a22 * a30; + var b08 = a20 * a33 - a23 * a30; + var b09 = a21 * a32 - a22 * a31; + var b10 = a21 * a33 - a23 * a31; + var b11 = a22 * a33 - a23 * a32; // Calculate the determinant + + var det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06; + + if (!det) { + return null; + } + + det = 1.0 / det; + out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det; + out[1] = (a12 * b08 - a10 * b11 - a13 * b07) * det; + out[2] = (a10 * b10 - a11 * b08 + a13 * b06) * det; + out[3] = (a02 * b10 - a01 * b11 - a03 * b09) * det; + out[4] = (a00 * b11 - a02 * b08 + a03 * b07) * det; + out[5] = (a01 * b08 - a00 * b10 - a03 * b06) * det; + out[6] = (a31 * b05 - a32 * b04 + a33 * b03) * det; + out[7] = (a32 * b02 - a30 * b05 - a33 * b01) * det; + out[8] = (a30 * b04 - a31 * b02 + a33 * b00) * det; + return out; +} +/** + * Generates a 2D projection matrix with the given bounds + * + * @param {mat3} out mat3 frustum matrix will be written into + * @param {number} width Width of your gl context + * @param {number} height Height of gl context + * @returns {mat3} out + */ + +function projection(out, width, height) { + out[0] = 2 / width; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = -2 / height; + out[5] = 0; + out[6] = -1; + out[7] = 1; + out[8] = 1; + return out; +} +/** + * Returns a string representation of a mat3 + * + * @param {ReadonlyMat3} a matrix to represent as a string + * @returns {String} string representation of the matrix + */ + +function str$6(a) { + return "mat3(" + a[0] + ", " + a[1] + ", " + a[2] + ", " + a[3] + ", " + a[4] + ", " + a[5] + ", " + a[6] + ", " + a[7] + ", " + a[8] + ")"; +} +/** + * Returns Frobenius norm of a mat3 + * + * @param {ReadonlyMat3} a the matrix to calculate Frobenius norm of + * @returns {Number} Frobenius norm + */ + +function frob$1(a) { + return Math.hypot(a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8]); +} +/** + * Adds two mat3's + * + * @param {mat3} out the receiving matrix + * @param {ReadonlyMat3} a the first operand + * @param {ReadonlyMat3} b the second operand + * @returns {mat3} out + */ + +function add$6(out, a, b) { + out[0] = a[0] + b[0]; + out[1] = a[1] + b[1]; + out[2] = a[2] + b[2]; + out[3] = a[3] + b[3]; + out[4] = a[4] + b[4]; + out[5] = a[5] + b[5]; + out[6] = a[6] + b[6]; + out[7] = a[7] + b[7]; + out[8] = a[8] + b[8]; + return out; +} +/** + * Subtracts matrix b from matrix a + * + * @param {mat3} out the receiving matrix + * @param {ReadonlyMat3} a the first operand + * @param {ReadonlyMat3} b the second operand + * @returns {mat3} out + */ + +function subtract$4(out, a, b) { + out[0] = a[0] - b[0]; + out[1] = a[1] - b[1]; + out[2] = a[2] - b[2]; + out[3] = a[3] - b[3]; + out[4] = a[4] - b[4]; + out[5] = a[5] - b[5]; + out[6] = a[6] - b[6]; + out[7] = a[7] - b[7]; + out[8] = a[8] - b[8]; + return out; +} +/** + * Multiply each element of the matrix by a scalar. + * + * @param {mat3} out the receiving matrix + * @param {ReadonlyMat3} a the matrix to scale + * @param {Number} b amount to scale the matrix's elements by + * @returns {mat3} out + */ + +function multiplyScalar$1(out, a, b) { + out[0] = a[0] * b; + out[1] = a[1] * b; + out[2] = a[2] * b; + out[3] = a[3] * b; + out[4] = a[4] * b; + out[5] = a[5] * b; + out[6] = a[6] * b; + out[7] = a[7] * b; + out[8] = a[8] * b; + return out; +} +/** + * Adds two mat3's after multiplying each element of the second operand by a scalar value. + * + * @param {mat3} out the receiving vector + * @param {ReadonlyMat3} a the first operand + * @param {ReadonlyMat3} b the second operand + * @param {Number} scale the amount to scale b's elements by before adding + * @returns {mat3} out + */ + +function multiplyScalarAndAdd$1(out, a, b, scale) { + out[0] = a[0] + b[0] * scale; + out[1] = a[1] + b[1] * scale; + out[2] = a[2] + b[2] * scale; + out[3] = a[3] + b[3] * scale; + out[4] = a[4] + b[4] * scale; + out[5] = a[5] + b[5] * scale; + out[6] = a[6] + b[6] * scale; + out[7] = a[7] + b[7] * scale; + out[8] = a[8] + b[8] * scale; + return out; +} +/** + * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===) + * + * @param {ReadonlyMat3} a The first matrix. + * @param {ReadonlyMat3} b The second matrix. + * @returns {Boolean} True if the matrices are equal, false otherwise. + */ + +function exactEquals$6(a, b) { + return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5] && a[6] === b[6] && a[7] === b[7] && a[8] === b[8]; +} +/** + * Returns whether or not the matrices have approximately the same elements in the same position. + * + * @param {ReadonlyMat3} a The first matrix. + * @param {ReadonlyMat3} b The second matrix. + * @returns {Boolean} True if the matrices are equal, false otherwise. + */ + +function equals$7(a, b) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3], + a4 = a[4], + a5 = a[5], + a6 = a[6], + a7 = a[7], + a8 = a[8]; + var b0 = b[0], + b1 = b[1], + b2 = b[2], + b3 = b[3], + b4 = b[4], + b5 = b[5], + b6 = b[6], + b7 = b[7], + b8 = b[8]; + return Math.abs(a0 - b0) <= EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5)) && Math.abs(a6 - b6) <= EPSILON * Math.max(1.0, Math.abs(a6), Math.abs(b6)) && Math.abs(a7 - b7) <= EPSILON * Math.max(1.0, Math.abs(a7), Math.abs(b7)) && Math.abs(a8 - b8) <= EPSILON * Math.max(1.0, Math.abs(a8), Math.abs(b8)); +} +/** + * Alias for {@link mat3.multiply} + * @function + */ + +var mul$6 = multiply$6; +/** + * Alias for {@link mat3.subtract} + * @function + */ + +var sub$4 = subtract$4; + +var mat3 = /*#__PURE__*/Object.freeze({ +__proto__: null, +add: add$6, +adjoint: adjoint$1, +clone: clone$6, +copy: copy$6, +create: create$6, +determinant: determinant$1, +equals: equals$7, +exactEquals: exactEquals$6, +frob: frob$1, +fromMat2d: fromMat2d, +fromMat4: fromMat4$1, +fromQuat: fromQuat$1, +fromRotation: fromRotation$2, +fromScaling: fromScaling$1, +fromTranslation: fromTranslation$2, +fromValues: fromValues$6, +identity: identity$3, +invert: invert$3, +mul: mul$6, +multiply: multiply$6, +multiplyScalar: multiplyScalar$1, +multiplyScalarAndAdd: multiplyScalarAndAdd$1, +normalFromMat4: normalFromMat4, +projection: projection, +rotate: rotate$2, +scale: scale$6, +set: set$6, +str: str$6, +sub: sub$4, +subtract: subtract$4, +translate: translate$2, +transpose: transpose$1 +}); + +/** + * 4x4 Matrix
Format: column-major, when typed out it looks like row-major
The matrices are being post multiplied. + * @module mat4 + */ + +/** + * Creates a new identity mat4 + * + * @returns {mat4} a new 4x4 matrix + */ + +function create$5() { + var out = new ARRAY_TYPE(16); + + if (ARRAY_TYPE != Float32Array) { + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = 0; + out[6] = 0; + out[7] = 0; + out[8] = 0; + out[9] = 0; + out[11] = 0; + out[12] = 0; + out[13] = 0; + out[14] = 0; + } + + out[0] = 1; + out[5] = 1; + out[10] = 1; + out[15] = 1; + return out; +} +/** + * Creates a new mat4 initialized with values from an existing matrix + * + * @param {ReadonlyMat4} a matrix to clone + * @returns {mat4} a new 4x4 matrix + */ + +function clone$5(a) { + var out = new ARRAY_TYPE(16); + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[3]; + out[4] = a[4]; + out[5] = a[5]; + out[6] = a[6]; + out[7] = a[7]; + out[8] = a[8]; + out[9] = a[9]; + out[10] = a[10]; + out[11] = a[11]; + out[12] = a[12]; + out[13] = a[13]; + out[14] = a[14]; + out[15] = a[15]; + return out; +} +/** + * Copy the values from one mat4 to another + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the source matrix + * @returns {mat4} out + */ + +function copy$5(out, a) { + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[3]; + out[4] = a[4]; + out[5] = a[5]; + out[6] = a[6]; + out[7] = a[7]; + out[8] = a[8]; + out[9] = a[9]; + out[10] = a[10]; + out[11] = a[11]; + out[12] = a[12]; + out[13] = a[13]; + out[14] = a[14]; + out[15] = a[15]; + return out; +} +/** + * Create a new mat4 with the given values + * + * @param {Number} m00 Component in column 0, row 0 position (index 0) + * @param {Number} m01 Component in column 0, row 1 position (index 1) + * @param {Number} m02 Component in column 0, row 2 position (index 2) + * @param {Number} m03 Component in column 0, row 3 position (index 3) + * @param {Number} m10 Component in column 1, row 0 position (index 4) + * @param {Number} m11 Component in column 1, row 1 position (index 5) + * @param {Number} m12 Component in column 1, row 2 position (index 6) + * @param {Number} m13 Component in column 1, row 3 position (index 7) + * @param {Number} m20 Component in column 2, row 0 position (index 8) + * @param {Number} m21 Component in column 2, row 1 position (index 9) + * @param {Number} m22 Component in column 2, row 2 position (index 10) + * @param {Number} m23 Component in column 2, row 3 position (index 11) + * @param {Number} m30 Component in column 3, row 0 position (index 12) + * @param {Number} m31 Component in column 3, row 1 position (index 13) + * @param {Number} m32 Component in column 3, row 2 position (index 14) + * @param {Number} m33 Component in column 3, row 3 position (index 15) + * @returns {mat4} A new mat4 + */ + +function fromValues$5(m00, m01, m02, m03, m10, m11, m12, m13, m20, m21, m22, m23, m30, m31, m32, m33) { + var out = new ARRAY_TYPE(16); + out[0] = m00; + out[1] = m01; + out[2] = m02; + out[3] = m03; + out[4] = m10; + out[5] = m11; + out[6] = m12; + out[7] = m13; + out[8] = m20; + out[9] = m21; + out[10] = m22; + out[11] = m23; + out[12] = m30; + out[13] = m31; + out[14] = m32; + out[15] = m33; + return out; +} +/** + * Set the components of a mat4 to the given values + * + * @param {mat4} out the receiving matrix + * @param {Number} m00 Component in column 0, row 0 position (index 0) + * @param {Number} m01 Component in column 0, row 1 position (index 1) + * @param {Number} m02 Component in column 0, row 2 position (index 2) + * @param {Number} m03 Component in column 0, row 3 position (index 3) + * @param {Number} m10 Component in column 1, row 0 position (index 4) + * @param {Number} m11 Component in column 1, row 1 position (index 5) + * @param {Number} m12 Component in column 1, row 2 position (index 6) + * @param {Number} m13 Component in column 1, row 3 position (index 7) + * @param {Number} m20 Component in column 2, row 0 position (index 8) + * @param {Number} m21 Component in column 2, row 1 position (index 9) + * @param {Number} m22 Component in column 2, row 2 position (index 10) + * @param {Number} m23 Component in column 2, row 3 position (index 11) + * @param {Number} m30 Component in column 3, row 0 position (index 12) + * @param {Number} m31 Component in column 3, row 1 position (index 13) + * @param {Number} m32 Component in column 3, row 2 position (index 14) + * @param {Number} m33 Component in column 3, row 3 position (index 15) + * @returns {mat4} out + */ + +function set$5(out, m00, m01, m02, m03, m10, m11, m12, m13, m20, m21, m22, m23, m30, m31, m32, m33) { + out[0] = m00; + out[1] = m01; + out[2] = m02; + out[3] = m03; + out[4] = m10; + out[5] = m11; + out[6] = m12; + out[7] = m13; + out[8] = m20; + out[9] = m21; + out[10] = m22; + out[11] = m23; + out[12] = m30; + out[13] = m31; + out[14] = m32; + out[15] = m33; + return out; +} +/** + * Set a mat4 to the identity matrix + * + * @param {mat4} out the receiving matrix + * @returns {mat4} out + */ + +function identity$2(out) { + out[0] = 1; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = 0; + out[5] = 1; + out[6] = 0; + out[7] = 0; + out[8] = 0; + out[9] = 0; + out[10] = 1; + out[11] = 0; + out[12] = 0; + out[13] = 0; + out[14] = 0; + out[15] = 1; + return out; +} +/** + * Transpose the values of a mat4 + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the source matrix + * @returns {mat4} out + */ + +function transpose(out, a) { + // If we are transposing ourselves we can skip a few steps but have to cache some values + if (out === a) { + var a01 = a[1], + a02 = a[2], + a03 = a[3]; + var a12 = a[6], + a13 = a[7]; + var a23 = a[11]; + out[1] = a[4]; + out[2] = a[8]; + out[3] = a[12]; + out[4] = a01; + out[6] = a[9]; + out[7] = a[13]; + out[8] = a02; + out[9] = a12; + out[11] = a[14]; + out[12] = a03; + out[13] = a13; + out[14] = a23; + } else { + out[0] = a[0]; + out[1] = a[4]; + out[2] = a[8]; + out[3] = a[12]; + out[4] = a[1]; + out[5] = a[5]; + out[6] = a[9]; + out[7] = a[13]; + out[8] = a[2]; + out[9] = a[6]; + out[10] = a[10]; + out[11] = a[14]; + out[12] = a[3]; + out[13] = a[7]; + out[14] = a[11]; + out[15] = a[15]; + } + + return out; +} +/** + * Inverts a mat4 + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the source matrix + * @returns {mat4} out + */ + +function invert$2(out, a) { + var a00 = a[0], + a01 = a[1], + a02 = a[2], + a03 = a[3]; + var a10 = a[4], + a11 = a[5], + a12 = a[6], + a13 = a[7]; + var a20 = a[8], + a21 = a[9], + a22 = a[10], + a23 = a[11]; + var a30 = a[12], + a31 = a[13], + a32 = a[14], + a33 = a[15]; + var b00 = a00 * a11 - a01 * a10; + var b01 = a00 * a12 - a02 * a10; + var b02 = a00 * a13 - a03 * a10; + var b03 = a01 * a12 - a02 * a11; + var b04 = a01 * a13 - a03 * a11; + var b05 = a02 * a13 - a03 * a12; + var b06 = a20 * a31 - a21 * a30; + var b07 = a20 * a32 - a22 * a30; + var b08 = a20 * a33 - a23 * a30; + var b09 = a21 * a32 - a22 * a31; + var b10 = a21 * a33 - a23 * a31; + var b11 = a22 * a33 - a23 * a32; // Calculate the determinant + + var det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06; + + if (!det) { + return null; + } + + det = 1.0 / det; + out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det; + out[1] = (a02 * b10 - a01 * b11 - a03 * b09) * det; + out[2] = (a31 * b05 - a32 * b04 + a33 * b03) * det; + out[3] = (a22 * b04 - a21 * b05 - a23 * b03) * det; + out[4] = (a12 * b08 - a10 * b11 - a13 * b07) * det; + out[5] = (a00 * b11 - a02 * b08 + a03 * b07) * det; + out[6] = (a32 * b02 - a30 * b05 - a33 * b01) * det; + out[7] = (a20 * b05 - a22 * b02 + a23 * b01) * det; + out[8] = (a10 * b10 - a11 * b08 + a13 * b06) * det; + out[9] = (a01 * b08 - a00 * b10 - a03 * b06) * det; + out[10] = (a30 * b04 - a31 * b02 + a33 * b00) * det; + out[11] = (a21 * b02 - a20 * b04 - a23 * b00) * det; + out[12] = (a11 * b07 - a10 * b09 - a12 * b06) * det; + out[13] = (a00 * b09 - a01 * b07 + a02 * b06) * det; + out[14] = (a31 * b01 - a30 * b03 - a32 * b00) * det; + out[15] = (a20 * b03 - a21 * b01 + a22 * b00) * det; + return out; +} +/** + * Calculates the adjugate of a mat4 + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the source matrix + * @returns {mat4} out + */ + +function adjoint(out, a) { + var a00 = a[0], + a01 = a[1], + a02 = a[2], + a03 = a[3]; + var a10 = a[4], + a11 = a[5], + a12 = a[6], + a13 = a[7]; + var a20 = a[8], + a21 = a[9], + a22 = a[10], + a23 = a[11]; + var a30 = a[12], + a31 = a[13], + a32 = a[14], + a33 = a[15]; + out[0] = a11 * (a22 * a33 - a23 * a32) - a21 * (a12 * a33 - a13 * a32) + a31 * (a12 * a23 - a13 * a22); + out[1] = -(a01 * (a22 * a33 - a23 * a32) - a21 * (a02 * a33 - a03 * a32) + a31 * (a02 * a23 - a03 * a22)); + out[2] = a01 * (a12 * a33 - a13 * a32) - a11 * (a02 * a33 - a03 * a32) + a31 * (a02 * a13 - a03 * a12); + out[3] = -(a01 * (a12 * a23 - a13 * a22) - a11 * (a02 * a23 - a03 * a22) + a21 * (a02 * a13 - a03 * a12)); + out[4] = -(a10 * (a22 * a33 - a23 * a32) - a20 * (a12 * a33 - a13 * a32) + a30 * (a12 * a23 - a13 * a22)); + out[5] = a00 * (a22 * a33 - a23 * a32) - a20 * (a02 * a33 - a03 * a32) + a30 * (a02 * a23 - a03 * a22); + out[6] = -(a00 * (a12 * a33 - a13 * a32) - a10 * (a02 * a33 - a03 * a32) + a30 * (a02 * a13 - a03 * a12)); + out[7] = a00 * (a12 * a23 - a13 * a22) - a10 * (a02 * a23 - a03 * a22) + a20 * (a02 * a13 - a03 * a12); + out[8] = a10 * (a21 * a33 - a23 * a31) - a20 * (a11 * a33 - a13 * a31) + a30 * (a11 * a23 - a13 * a21); + out[9] = -(a00 * (a21 * a33 - a23 * a31) - a20 * (a01 * a33 - a03 * a31) + a30 * (a01 * a23 - a03 * a21)); + out[10] = a00 * (a11 * a33 - a13 * a31) - a10 * (a01 * a33 - a03 * a31) + a30 * (a01 * a13 - a03 * a11); + out[11] = -(a00 * (a11 * a23 - a13 * a21) - a10 * (a01 * a23 - a03 * a21) + a20 * (a01 * a13 - a03 * a11)); + out[12] = -(a10 * (a21 * a32 - a22 * a31) - a20 * (a11 * a32 - a12 * a31) + a30 * (a11 * a22 - a12 * a21)); + out[13] = a00 * (a21 * a32 - a22 * a31) - a20 * (a01 * a32 - a02 * a31) + a30 * (a01 * a22 - a02 * a21); + out[14] = -(a00 * (a11 * a32 - a12 * a31) - a10 * (a01 * a32 - a02 * a31) + a30 * (a01 * a12 - a02 * a11)); + out[15] = a00 * (a11 * a22 - a12 * a21) - a10 * (a01 * a22 - a02 * a21) + a20 * (a01 * a12 - a02 * a11); + return out; +} +/** + * Calculates the determinant of a mat4 + * + * @param {ReadonlyMat4} a the source matrix + * @returns {Number} determinant of a + */ + +function determinant(a) { + var a00 = a[0], + a01 = a[1], + a02 = a[2], + a03 = a[3]; + var a10 = a[4], + a11 = a[5], + a12 = a[6], + a13 = a[7]; + var a20 = a[8], + a21 = a[9], + a22 = a[10], + a23 = a[11]; + var a30 = a[12], + a31 = a[13], + a32 = a[14], + a33 = a[15]; + var b00 = a00 * a11 - a01 * a10; + var b01 = a00 * a12 - a02 * a10; + var b02 = a00 * a13 - a03 * a10; + var b03 = a01 * a12 - a02 * a11; + var b04 = a01 * a13 - a03 * a11; + var b05 = a02 * a13 - a03 * a12; + var b06 = a20 * a31 - a21 * a30; + var b07 = a20 * a32 - a22 * a30; + var b08 = a20 * a33 - a23 * a30; + var b09 = a21 * a32 - a22 * a31; + var b10 = a21 * a33 - a23 * a31; + var b11 = a22 * a33 - a23 * a32; // Calculate the determinant + + return b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06; +} +/** + * Multiplies two mat4s + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the first operand + * @param {ReadonlyMat4} b the second operand + * @returns {mat4} out + */ + +function multiply$5(out, a, b) { + var a00 = a[0], + a01 = a[1], + a02 = a[2], + a03 = a[3]; + var a10 = a[4], + a11 = a[5], + a12 = a[6], + a13 = a[7]; + var a20 = a[8], + a21 = a[9], + a22 = a[10], + a23 = a[11]; + var a30 = a[12], + a31 = a[13], + a32 = a[14], + a33 = a[15]; // Cache only the current line of the second matrix + + var b0 = b[0], + b1 = b[1], + b2 = b[2], + b3 = b[3]; + out[0] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30; + out[1] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31; + out[2] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32; + out[3] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33; + b0 = b[4]; + b1 = b[5]; + b2 = b[6]; + b3 = b[7]; + out[4] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30; + out[5] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31; + out[6] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32; + out[7] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33; + b0 = b[8]; + b1 = b[9]; + b2 = b[10]; + b3 = b[11]; + out[8] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30; + out[9] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31; + out[10] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32; + out[11] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33; + b0 = b[12]; + b1 = b[13]; + b2 = b[14]; + b3 = b[15]; + out[12] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30; + out[13] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31; + out[14] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32; + out[15] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33; + return out; +} +/** + * Translate a mat4 by the given vector + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the matrix to translate + * @param {ReadonlyVec3} v vector to translate by + * @returns {mat4} out + */ + +function translate$1(out, a, v) { + var x = v[0], + y = v[1], + z = v[2]; + var a00, a01, a02, a03; + var a10, a11, a12, a13; + var a20, a21, a22, a23; + + if (a === out) { + out[12] = a[0] * x + a[4] * y + a[8] * z + a[12]; + out[13] = a[1] * x + a[5] * y + a[9] * z + a[13]; + out[14] = a[2] * x + a[6] * y + a[10] * z + a[14]; + out[15] = a[3] * x + a[7] * y + a[11] * z + a[15]; + } else { + a00 = a[0]; + a01 = a[1]; + a02 = a[2]; + a03 = a[3]; + a10 = a[4]; + a11 = a[5]; + a12 = a[6]; + a13 = a[7]; + a20 = a[8]; + a21 = a[9]; + a22 = a[10]; + a23 = a[11]; + out[0] = a00; + out[1] = a01; + out[2] = a02; + out[3] = a03; + out[4] = a10; + out[5] = a11; + out[6] = a12; + out[7] = a13; + out[8] = a20; + out[9] = a21; + out[10] = a22; + out[11] = a23; + out[12] = a00 * x + a10 * y + a20 * z + a[12]; + out[13] = a01 * x + a11 * y + a21 * z + a[13]; + out[14] = a02 * x + a12 * y + a22 * z + a[14]; + out[15] = a03 * x + a13 * y + a23 * z + a[15]; + } + + return out; +} +/** + * Scales the mat4 by the dimensions in the given vec3 not using vectorization + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the matrix to scale + * @param {ReadonlyVec3} v the vec3 to scale the matrix by + * @returns {mat4} out + **/ + +function scale$5(out, a, v) { + var x = v[0], + y = v[1], + z = v[2]; + out[0] = a[0] * x; + out[1] = a[1] * x; + out[2] = a[2] * x; + out[3] = a[3] * x; + out[4] = a[4] * y; + out[5] = a[5] * y; + out[6] = a[6] * y; + out[7] = a[7] * y; + out[8] = a[8] * z; + out[9] = a[9] * z; + out[10] = a[10] * z; + out[11] = a[11] * z; + out[12] = a[12]; + out[13] = a[13]; + out[14] = a[14]; + out[15] = a[15]; + return out; +} +/** + * Rotates a mat4 by the given angle around the given axis + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the matrix to rotate + * @param {Number} rad the angle to rotate the matrix by + * @param {ReadonlyVec3} axis the axis to rotate around + * @returns {mat4} out + */ + +function rotate$1(out, a, rad, axis) { + var x = axis[0], + y = axis[1], + z = axis[2]; + var len = Math.hypot(x, y, z); + var s, c, t; + var a00, a01, a02, a03; + var a10, a11, a12, a13; + var a20, a21, a22, a23; + var b00, b01, b02; + var b10, b11, b12; + var b20, b21, b22; + + if (len < EPSILON) { + return null; + } + + len = 1 / len; + x *= len; + y *= len; + z *= len; + s = Math.sin(rad); + c = Math.cos(rad); + t = 1 - c; + a00 = a[0]; + a01 = a[1]; + a02 = a[2]; + a03 = a[3]; + a10 = a[4]; + a11 = a[5]; + a12 = a[6]; + a13 = a[7]; + a20 = a[8]; + a21 = a[9]; + a22 = a[10]; + a23 = a[11]; // Construct the elements of the rotation matrix + + b00 = x * x * t + c; + b01 = y * x * t + z * s; + b02 = z * x * t - y * s; + b10 = x * y * t - z * s; + b11 = y * y * t + c; + b12 = z * y * t + x * s; + b20 = x * z * t + y * s; + b21 = y * z * t - x * s; + b22 = z * z * t + c; // Perform rotation-specific matrix multiplication + + out[0] = a00 * b00 + a10 * b01 + a20 * b02; + out[1] = a01 * b00 + a11 * b01 + a21 * b02; + out[2] = a02 * b00 + a12 * b01 + a22 * b02; + out[3] = a03 * b00 + a13 * b01 + a23 * b02; + out[4] = a00 * b10 + a10 * b11 + a20 * b12; + out[5] = a01 * b10 + a11 * b11 + a21 * b12; + out[6] = a02 * b10 + a12 * b11 + a22 * b12; + out[7] = a03 * b10 + a13 * b11 + a23 * b12; + out[8] = a00 * b20 + a10 * b21 + a20 * b22; + out[9] = a01 * b20 + a11 * b21 + a21 * b22; + out[10] = a02 * b20 + a12 * b21 + a22 * b22; + out[11] = a03 * b20 + a13 * b21 + a23 * b22; + + if (a !== out) { + // If the source and destination differ, copy the unchanged last row + out[12] = a[12]; + out[13] = a[13]; + out[14] = a[14]; + out[15] = a[15]; + } + + return out; +} +/** + * Rotates a matrix by the given angle around the X axis + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the matrix to rotate + * @param {Number} rad the angle to rotate the matrix by + * @returns {mat4} out + */ + +function rotateX$3(out, a, rad) { + var s = Math.sin(rad); + var c = Math.cos(rad); + var a10 = a[4]; + var a11 = a[5]; + var a12 = a[6]; + var a13 = a[7]; + var a20 = a[8]; + var a21 = a[9]; + var a22 = a[10]; + var a23 = a[11]; + + if (a !== out) { + // If the source and destination differ, copy the unchanged rows + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[3]; + out[12] = a[12]; + out[13] = a[13]; + out[14] = a[14]; + out[15] = a[15]; + } // Perform axis-specific matrix multiplication + + + out[4] = a10 * c + a20 * s; + out[5] = a11 * c + a21 * s; + out[6] = a12 * c + a22 * s; + out[7] = a13 * c + a23 * s; + out[8] = a20 * c - a10 * s; + out[9] = a21 * c - a11 * s; + out[10] = a22 * c - a12 * s; + out[11] = a23 * c - a13 * s; + return out; +} +/** + * Rotates a matrix by the given angle around the Y axis + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the matrix to rotate + * @param {Number} rad the angle to rotate the matrix by + * @returns {mat4} out + */ + +function rotateY$3(out, a, rad) { + var s = Math.sin(rad); + var c = Math.cos(rad); + var a00 = a[0]; + var a01 = a[1]; + var a02 = a[2]; + var a03 = a[3]; + var a20 = a[8]; + var a21 = a[9]; + var a22 = a[10]; + var a23 = a[11]; + + if (a !== out) { + // If the source and destination differ, copy the unchanged rows + out[4] = a[4]; + out[5] = a[5]; + out[6] = a[6]; + out[7] = a[7]; + out[12] = a[12]; + out[13] = a[13]; + out[14] = a[14]; + out[15] = a[15]; + } // Perform axis-specific matrix multiplication + + + out[0] = a00 * c - a20 * s; + out[1] = a01 * c - a21 * s; + out[2] = a02 * c - a22 * s; + out[3] = a03 * c - a23 * s; + out[8] = a00 * s + a20 * c; + out[9] = a01 * s + a21 * c; + out[10] = a02 * s + a22 * c; + out[11] = a03 * s + a23 * c; + return out; +} +/** + * Rotates a matrix by the given angle around the Z axis + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the matrix to rotate + * @param {Number} rad the angle to rotate the matrix by + * @returns {mat4} out + */ + +function rotateZ$3(out, a, rad) { + var s = Math.sin(rad); + var c = Math.cos(rad); + var a00 = a[0]; + var a01 = a[1]; + var a02 = a[2]; + var a03 = a[3]; + var a10 = a[4]; + var a11 = a[5]; + var a12 = a[6]; + var a13 = a[7]; + + if (a !== out) { + // If the source and destination differ, copy the unchanged last row + out[8] = a[8]; + out[9] = a[9]; + out[10] = a[10]; + out[11] = a[11]; + out[12] = a[12]; + out[13] = a[13]; + out[14] = a[14]; + out[15] = a[15]; + } // Perform axis-specific matrix multiplication + + + out[0] = a00 * c + a10 * s; + out[1] = a01 * c + a11 * s; + out[2] = a02 * c + a12 * s; + out[3] = a03 * c + a13 * s; + out[4] = a10 * c - a00 * s; + out[5] = a11 * c - a01 * s; + out[6] = a12 * c - a02 * s; + out[7] = a13 * c - a03 * s; + return out; +} +/** + * Creates a matrix from a vector translation + * This is equivalent to (but much faster than): + * + * mat4.identity(dest); + * mat4.translate(dest, dest, vec); + * + * @param {mat4} out mat4 receiving operation result + * @param {ReadonlyVec3} v Translation vector + * @returns {mat4} out + */ + +function fromTranslation$1(out, v) { + out[0] = 1; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = 0; + out[5] = 1; + out[6] = 0; + out[7] = 0; + out[8] = 0; + out[9] = 0; + out[10] = 1; + out[11] = 0; + out[12] = v[0]; + out[13] = v[1]; + out[14] = v[2]; + out[15] = 1; + return out; +} +/** + * Creates a matrix from a vector scaling + * This is equivalent to (but much faster than): + * + * mat4.identity(dest); + * mat4.scale(dest, dest, vec); + * + * @param {mat4} out mat4 receiving operation result + * @param {ReadonlyVec3} v Scaling vector + * @returns {mat4} out + */ + +function fromScaling(out, v) { + out[0] = v[0]; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = 0; + out[5] = v[1]; + out[6] = 0; + out[7] = 0; + out[8] = 0; + out[9] = 0; + out[10] = v[2]; + out[11] = 0; + out[12] = 0; + out[13] = 0; + out[14] = 0; + out[15] = 1; + return out; +} +/** + * Creates a matrix from a given angle around a given axis + * This is equivalent to (but much faster than): + * + * mat4.identity(dest); + * mat4.rotate(dest, dest, rad, axis); + * + * @param {mat4} out mat4 receiving operation result + * @param {Number} rad the angle to rotate the matrix by + * @param {ReadonlyVec3} axis the axis to rotate around + * @returns {mat4} out + */ + +function fromRotation$1(out, rad, axis) { + var x = axis[0], + y = axis[1], + z = axis[2]; + var len = Math.hypot(x, y, z); + var s, c, t; + + if (len < EPSILON) { + return null; + } + + len = 1 / len; + x *= len; + y *= len; + z *= len; + s = Math.sin(rad); + c = Math.cos(rad); + t = 1 - c; // Perform rotation-specific matrix multiplication + + out[0] = x * x * t + c; + out[1] = y * x * t + z * s; + out[2] = z * x * t - y * s; + out[3] = 0; + out[4] = x * y * t - z * s; + out[5] = y * y * t + c; + out[6] = z * y * t + x * s; + out[7] = 0; + out[8] = x * z * t + y * s; + out[9] = y * z * t - x * s; + out[10] = z * z * t + c; + out[11] = 0; + out[12] = 0; + out[13] = 0; + out[14] = 0; + out[15] = 1; + return out; +} +/** + * Creates a matrix from the given angle around the X axis + * This is equivalent to (but much faster than): + * + * mat4.identity(dest); + * mat4.rotateX(dest, dest, rad); + * + * @param {mat4} out mat4 receiving operation result + * @param {Number} rad the angle to rotate the matrix by + * @returns {mat4} out + */ + +function fromXRotation(out, rad) { + var s = Math.sin(rad); + var c = Math.cos(rad); // Perform axis-specific matrix multiplication + + out[0] = 1; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = 0; + out[5] = c; + out[6] = s; + out[7] = 0; + out[8] = 0; + out[9] = -s; + out[10] = c; + out[11] = 0; + out[12] = 0; + out[13] = 0; + out[14] = 0; + out[15] = 1; + return out; +} +/** + * Creates a matrix from the given angle around the Y axis + * This is equivalent to (but much faster than): + * + * mat4.identity(dest); + * mat4.rotateY(dest, dest, rad); + * + * @param {mat4} out mat4 receiving operation result + * @param {Number} rad the angle to rotate the matrix by + * @returns {mat4} out + */ + +function fromYRotation(out, rad) { + var s = Math.sin(rad); + var c = Math.cos(rad); // Perform axis-specific matrix multiplication + + out[0] = c; + out[1] = 0; + out[2] = -s; + out[3] = 0; + out[4] = 0; + out[5] = 1; + out[6] = 0; + out[7] = 0; + out[8] = s; + out[9] = 0; + out[10] = c; + out[11] = 0; + out[12] = 0; + out[13] = 0; + out[14] = 0; + out[15] = 1; + return out; +} +/** + * Creates a matrix from the given angle around the Z axis + * This is equivalent to (but much faster than): + * + * mat4.identity(dest); + * mat4.rotateZ(dest, dest, rad); + * + * @param {mat4} out mat4 receiving operation result + * @param {Number} rad the angle to rotate the matrix by + * @returns {mat4} out + */ + +function fromZRotation(out, rad) { + var s = Math.sin(rad); + var c = Math.cos(rad); // Perform axis-specific matrix multiplication + + out[0] = c; + out[1] = s; + out[2] = 0; + out[3] = 0; + out[4] = -s; + out[5] = c; + out[6] = 0; + out[7] = 0; + out[8] = 0; + out[9] = 0; + out[10] = 1; + out[11] = 0; + out[12] = 0; + out[13] = 0; + out[14] = 0; + out[15] = 1; + return out; +} +/** + * Creates a matrix from a quaternion rotation and vector translation + * This is equivalent to (but much faster than): + * + * mat4.identity(dest); + * mat4.translate(dest, vec); + * let quatMat = mat4.create(); + * quat4.toMat4(quat, quatMat); + * mat4.multiply(dest, quatMat); + * + * @param {mat4} out mat4 receiving operation result + * @param {quat4} q Rotation quaternion + * @param {ReadonlyVec3} v Translation vector + * @returns {mat4} out + */ + +function fromRotationTranslation$1(out, q, v) { + // Quaternion math + var x = q[0], + y = q[1], + z = q[2], + w = q[3]; + var x2 = x + x; + var y2 = y + y; + var z2 = z + z; + var xx = x * x2; + var xy = x * y2; + var xz = x * z2; + var yy = y * y2; + var yz = y * z2; + var zz = z * z2; + var wx = w * x2; + var wy = w * y2; + var wz = w * z2; + out[0] = 1 - (yy + zz); + out[1] = xy + wz; + out[2] = xz - wy; + out[3] = 0; + out[4] = xy - wz; + out[5] = 1 - (xx + zz); + out[6] = yz + wx; + out[7] = 0; + out[8] = xz + wy; + out[9] = yz - wx; + out[10] = 1 - (xx + yy); + out[11] = 0; + out[12] = v[0]; + out[13] = v[1]; + out[14] = v[2]; + out[15] = 1; + return out; +} +/** + * Creates a new mat4 from a dual quat. + * + * @param {mat4} out Matrix + * @param {ReadonlyQuat2} a Dual Quaternion + * @returns {mat4} mat4 receiving operation result + */ + +function fromQuat2(out, a) { + var translation = new ARRAY_TYPE(3); + var bx = -a[0], + by = -a[1], + bz = -a[2], + bw = a[3], + ax = a[4], + ay = a[5], + az = a[6], + aw = a[7]; + var magnitude = bx * bx + by * by + bz * bz + bw * bw; //Only scale if it makes sense + + if (magnitude > 0) { + translation[0] = (ax * bw + aw * bx + ay * bz - az * by) * 2 / magnitude; + translation[1] = (ay * bw + aw * by + az * bx - ax * bz) * 2 / magnitude; + translation[2] = (az * bw + aw * bz + ax * by - ay * bx) * 2 / magnitude; + } else { + translation[0] = (ax * bw + aw * bx + ay * bz - az * by) * 2; + translation[1] = (ay * bw + aw * by + az * bx - ax * bz) * 2; + translation[2] = (az * bw + aw * bz + ax * by - ay * bx) * 2; + } + + fromRotationTranslation$1(out, a, translation); + return out; +} +/** + * Returns the translation vector component of a transformation + * matrix. If a matrix is built with fromRotationTranslation, + * the returned vector will be the same as the translation vector + * originally supplied. + * @param {vec3} out Vector to receive translation component + * @param {ReadonlyMat4} mat Matrix to be decomposed (input) + * @return {vec3} out + */ + +function getTranslation$1(out, mat) { + out[0] = mat[12]; + out[1] = mat[13]; + out[2] = mat[14]; + return out; +} +/** + * Returns the scaling factor component of a transformation + * matrix. If a matrix is built with fromRotationTranslationScale + * with a normalized Quaternion paramter, the returned vector will be + * the same as the scaling vector + * originally supplied. + * @param {vec3} out Vector to receive scaling factor component + * @param {ReadonlyMat4} mat Matrix to be decomposed (input) + * @return {vec3} out + */ + +function getScaling(out, mat) { + var m11 = mat[0]; + var m12 = mat[1]; + var m13 = mat[2]; + var m21 = mat[4]; + var m22 = mat[5]; + var m23 = mat[6]; + var m31 = mat[8]; + var m32 = mat[9]; + var m33 = mat[10]; + out[0] = Math.hypot(m11, m12, m13); + out[1] = Math.hypot(m21, m22, m23); + out[2] = Math.hypot(m31, m32, m33); + return out; +} +/** + * Returns a quaternion representing the rotational component + * of a transformation matrix. If a matrix is built with + * fromRotationTranslation, the returned quaternion will be the + * same as the quaternion originally supplied. + * @param {quat} out Quaternion to receive the rotation component + * @param {ReadonlyMat4} mat Matrix to be decomposed (input) + * @return {quat} out + */ + +function getRotation(out, mat) { + var scaling = new ARRAY_TYPE(3); + getScaling(scaling, mat); + var is1 = 1 / scaling[0]; + var is2 = 1 / scaling[1]; + var is3 = 1 / scaling[2]; + var sm11 = mat[0] * is1; + var sm12 = mat[1] * is2; + var sm13 = mat[2] * is3; + var sm21 = mat[4] * is1; + var sm22 = mat[5] * is2; + var sm23 = mat[6] * is3; + var sm31 = mat[8] * is1; + var sm32 = mat[9] * is2; + var sm33 = mat[10] * is3; + var trace = sm11 + sm22 + sm33; + var S = 0; + + if (trace > 0) { + S = Math.sqrt(trace + 1.0) * 2; + out[3] = 0.25 * S; + out[0] = (sm23 - sm32) / S; + out[1] = (sm31 - sm13) / S; + out[2] = (sm12 - sm21) / S; + } else if (sm11 > sm22 && sm11 > sm33) { + S = Math.sqrt(1.0 + sm11 - sm22 - sm33) * 2; + out[3] = (sm23 - sm32) / S; + out[0] = 0.25 * S; + out[1] = (sm12 + sm21) / S; + out[2] = (sm31 + sm13) / S; + } else if (sm22 > sm33) { + S = Math.sqrt(1.0 + sm22 - sm11 - sm33) * 2; + out[3] = (sm31 - sm13) / S; + out[0] = (sm12 + sm21) / S; + out[1] = 0.25 * S; + out[2] = (sm23 + sm32) / S; + } else { + S = Math.sqrt(1.0 + sm33 - sm11 - sm22) * 2; + out[3] = (sm12 - sm21) / S; + out[0] = (sm31 + sm13) / S; + out[1] = (sm23 + sm32) / S; + out[2] = 0.25 * S; + } + + return out; +} +/** + * Creates a matrix from a quaternion rotation, vector translation and vector scale + * This is equivalent to (but much faster than): + * + * mat4.identity(dest); + * mat4.translate(dest, vec); + * let quatMat = mat4.create(); + * quat4.toMat4(quat, quatMat); + * mat4.multiply(dest, quatMat); + * mat4.scale(dest, scale) + * + * @param {mat4} out mat4 receiving operation result + * @param {quat4} q Rotation quaternion + * @param {ReadonlyVec3} v Translation vector + * @param {ReadonlyVec3} s Scaling vector + * @returns {mat4} out + */ + +function fromRotationTranslationScale(out, q, v, s) { + // Quaternion math + var x = q[0], + y = q[1], + z = q[2], + w = q[3]; + var x2 = x + x; + var y2 = y + y; + var z2 = z + z; + var xx = x * x2; + var xy = x * y2; + var xz = x * z2; + var yy = y * y2; + var yz = y * z2; + var zz = z * z2; + var wx = w * x2; + var wy = w * y2; + var wz = w * z2; + var sx = s[0]; + var sy = s[1]; + var sz = s[2]; + out[0] = (1 - (yy + zz)) * sx; + out[1] = (xy + wz) * sx; + out[2] = (xz - wy) * sx; + out[3] = 0; + out[4] = (xy - wz) * sy; + out[5] = (1 - (xx + zz)) * sy; + out[6] = (yz + wx) * sy; + out[7] = 0; + out[8] = (xz + wy) * sz; + out[9] = (yz - wx) * sz; + out[10] = (1 - (xx + yy)) * sz; + out[11] = 0; + out[12] = v[0]; + out[13] = v[1]; + out[14] = v[2]; + out[15] = 1; + return out; +} +/** + * Creates a matrix from a quaternion rotation, vector translation and vector scale, rotating and scaling around the given origin + * This is equivalent to (but much faster than): + * + * mat4.identity(dest); + * mat4.translate(dest, vec); + * mat4.translate(dest, origin); + * let quatMat = mat4.create(); + * quat4.toMat4(quat, quatMat); + * mat4.multiply(dest, quatMat); + * mat4.scale(dest, scale) + * mat4.translate(dest, negativeOrigin); + * + * @param {mat4} out mat4 receiving operation result + * @param {quat4} q Rotation quaternion + * @param {ReadonlyVec3} v Translation vector + * @param {ReadonlyVec3} s Scaling vector + * @param {ReadonlyVec3} o The origin vector around which to scale and rotate + * @returns {mat4} out + */ + +function fromRotationTranslationScaleOrigin(out, q, v, s, o) { + // Quaternion math + var x = q[0], + y = q[1], + z = q[2], + w = q[3]; + var x2 = x + x; + var y2 = y + y; + var z2 = z + z; + var xx = x * x2; + var xy = x * y2; + var xz = x * z2; + var yy = y * y2; + var yz = y * z2; + var zz = z * z2; + var wx = w * x2; + var wy = w * y2; + var wz = w * z2; + var sx = s[0]; + var sy = s[1]; + var sz = s[2]; + var ox = o[0]; + var oy = o[1]; + var oz = o[2]; + var out0 = (1 - (yy + zz)) * sx; + var out1 = (xy + wz) * sx; + var out2 = (xz - wy) * sx; + var out4 = (xy - wz) * sy; + var out5 = (1 - (xx + zz)) * sy; + var out6 = (yz + wx) * sy; + var out8 = (xz + wy) * sz; + var out9 = (yz - wx) * sz; + var out10 = (1 - (xx + yy)) * sz; + out[0] = out0; + out[1] = out1; + out[2] = out2; + out[3] = 0; + out[4] = out4; + out[5] = out5; + out[6] = out6; + out[7] = 0; + out[8] = out8; + out[9] = out9; + out[10] = out10; + out[11] = 0; + out[12] = v[0] + ox - (out0 * ox + out4 * oy + out8 * oz); + out[13] = v[1] + oy - (out1 * ox + out5 * oy + out9 * oz); + out[14] = v[2] + oz - (out2 * ox + out6 * oy + out10 * oz); + out[15] = 1; + return out; +} +/** + * Calculates a 4x4 matrix from the given quaternion + * + * @param {mat4} out mat4 receiving operation result + * @param {ReadonlyQuat} q Quaternion to create matrix from + * + * @returns {mat4} out + */ + +function fromQuat(out, q) { + var x = q[0], + y = q[1], + z = q[2], + w = q[3]; + var x2 = x + x; + var y2 = y + y; + var z2 = z + z; + var xx = x * x2; + var yx = y * x2; + var yy = y * y2; + var zx = z * x2; + var zy = z * y2; + var zz = z * z2; + var wx = w * x2; + var wy = w * y2; + var wz = w * z2; + out[0] = 1 - yy - zz; + out[1] = yx + wz; + out[2] = zx - wy; + out[3] = 0; + out[4] = yx - wz; + out[5] = 1 - xx - zz; + out[6] = zy + wx; + out[7] = 0; + out[8] = zx + wy; + out[9] = zy - wx; + out[10] = 1 - xx - yy; + out[11] = 0; + out[12] = 0; + out[13] = 0; + out[14] = 0; + out[15] = 1; + return out; +} +/** + * Generates a frustum matrix with the given bounds + * + * @param {mat4} out mat4 frustum matrix will be written into + * @param {Number} left Left bound of the frustum + * @param {Number} right Right bound of the frustum + * @param {Number} bottom Bottom bound of the frustum + * @param {Number} top Top bound of the frustum + * @param {Number} near Near bound of the frustum + * @param {Number} far Far bound of the frustum + * @returns {mat4} out + */ + +function frustum(out, left, right, bottom, top, near, far) { + var rl = 1 / (right - left); + var tb = 1 / (top - bottom); + var nf = 1 / (near - far); + out[0] = near * 2 * rl; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = 0; + out[5] = near * 2 * tb; + out[6] = 0; + out[7] = 0; + out[8] = (right + left) * rl; + out[9] = (top + bottom) * tb; + out[10] = (far + near) * nf; + out[11] = -1; + out[12] = 0; + out[13] = 0; + out[14] = far * near * 2 * nf; + out[15] = 0; + return out; +} +/** + * Generates a perspective projection matrix with the given bounds. + * The near/far clip planes correspond to a normalized device coordinate Z range of [-1, 1], + * which matches WebGL/OpenGL's clip volume. + * Passing null/undefined/no value for far will generate infinite projection matrix. + * + * @param {mat4} out mat4 frustum matrix will be written into + * @param {number} fovy Vertical field of view in radians + * @param {number} aspect Aspect ratio. typically viewport width/height + * @param {number} near Near bound of the frustum + * @param {number} far Far bound of the frustum, can be null or Infinity + * @returns {mat4} out + */ + +function perspectiveNO(out, fovy, aspect, near, far) { + var f = 1.0 / Math.tan(fovy / 2), + nf; + out[0] = f / aspect; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = 0; + out[5] = f; + out[6] = 0; + out[7] = 0; + out[8] = 0; + out[9] = 0; + out[11] = -1; + out[12] = 0; + out[13] = 0; + out[15] = 0; + + if (far != null && far !== Infinity) { + nf = 1 / (near - far); + out[10] = (far + near) * nf; + out[14] = 2 * far * near * nf; + } else { + out[10] = -1; + out[14] = -2 * near; + } + + return out; +} +/** + * Alias for {@link mat4.perspectiveNO} + * @function + */ + +var perspective = perspectiveNO; +/** + * Generates a perspective projection matrix suitable for WebGPU with the given bounds. + * The near/far clip planes correspond to a normalized device coordinate Z range of [0, 1], + * which matches WebGPU/Vulkan/DirectX/Metal's clip volume. + * Passing null/undefined/no value for far will generate infinite projection matrix. + * + * @param {mat4} out mat4 frustum matrix will be written into + * @param {number} fovy Vertical field of view in radians + * @param {number} aspect Aspect ratio. typically viewport width/height + * @param {number} near Near bound of the frustum + * @param {number} far Far bound of the frustum, can be null or Infinity + * @returns {mat4} out + */ + +function perspectiveZO(out, fovy, aspect, near, far) { + var f = 1.0 / Math.tan(fovy / 2), + nf; + out[0] = f / aspect; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = 0; + out[5] = f; + out[6] = 0; + out[7] = 0; + out[8] = 0; + out[9] = 0; + out[11] = -1; + out[12] = 0; + out[13] = 0; + out[15] = 0; + + if (far != null && far !== Infinity) { + nf = 1 / (near - far); + out[10] = far * nf; + out[14] = far * near * nf; + } else { + out[10] = -1; + out[14] = -near; + } + + return out; +} +/** + * Generates a perspective projection matrix with the given field of view. + * This is primarily useful for generating projection matrices to be used + * with the still experiemental WebVR API. + * + * @param {mat4} out mat4 frustum matrix will be written into + * @param {Object} fov Object containing the following values: upDegrees, downDegrees, leftDegrees, rightDegrees + * @param {number} near Near bound of the frustum + * @param {number} far Far bound of the frustum + * @returns {mat4} out + */ + +function perspectiveFromFieldOfView(out, fov, near, far) { + var upTan = Math.tan(fov.upDegrees * Math.PI / 180.0); + var downTan = Math.tan(fov.downDegrees * Math.PI / 180.0); + var leftTan = Math.tan(fov.leftDegrees * Math.PI / 180.0); + var rightTan = Math.tan(fov.rightDegrees * Math.PI / 180.0); + var xScale = 2.0 / (leftTan + rightTan); + var yScale = 2.0 / (upTan + downTan); + out[0] = xScale; + out[1] = 0.0; + out[2] = 0.0; + out[3] = 0.0; + out[4] = 0.0; + out[5] = yScale; + out[6] = 0.0; + out[7] = 0.0; + out[8] = -((leftTan - rightTan) * xScale * 0.5); + out[9] = (upTan - downTan) * yScale * 0.5; + out[10] = far / (near - far); + out[11] = -1.0; + out[12] = 0.0; + out[13] = 0.0; + out[14] = far * near / (near - far); + out[15] = 0.0; + return out; +} +/** + * Generates a orthogonal projection matrix with the given bounds. + * The near/far clip planes correspond to a normalized device coordinate Z range of [-1, 1], + * which matches WebGL/OpenGL's clip volume. + * + * @param {mat4} out mat4 frustum matrix will be written into + * @param {number} left Left bound of the frustum + * @param {number} right Right bound of the frustum + * @param {number} bottom Bottom bound of the frustum + * @param {number} top Top bound of the frustum + * @param {number} near Near bound of the frustum + * @param {number} far Far bound of the frustum + * @returns {mat4} out + */ + +function orthoNO(out, left, right, bottom, top, near, far) { + var lr = 1 / (left - right); + var bt = 1 / (bottom - top); + var nf = 1 / (near - far); + out[0] = -2 * lr; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = 0; + out[5] = -2 * bt; + out[6] = 0; + out[7] = 0; + out[8] = 0; + out[9] = 0; + out[10] = 2 * nf; + out[11] = 0; + out[12] = (left + right) * lr; + out[13] = (top + bottom) * bt; + out[14] = (far + near) * nf; + out[15] = 1; + return out; +} +/** + * Alias for {@link mat4.orthoNO} + * @function + */ + +var ortho = orthoNO; +/** + * Generates a orthogonal projection matrix with the given bounds. + * The near/far clip planes correspond to a normalized device coordinate Z range of [0, 1], + * which matches WebGPU/Vulkan/DirectX/Metal's clip volume. + * + * @param {mat4} out mat4 frustum matrix will be written into + * @param {number} left Left bound of the frustum + * @param {number} right Right bound of the frustum + * @param {number} bottom Bottom bound of the frustum + * @param {number} top Top bound of the frustum + * @param {number} near Near bound of the frustum + * @param {number} far Far bound of the frustum + * @returns {mat4} out + */ + +function orthoZO(out, left, right, bottom, top, near, far) { + var lr = 1 / (left - right); + var bt = 1 / (bottom - top); + var nf = 1 / (near - far); + out[0] = -2 * lr; + out[1] = 0; + out[2] = 0; + out[3] = 0; + out[4] = 0; + out[5] = -2 * bt; + out[6] = 0; + out[7] = 0; + out[8] = 0; + out[9] = 0; + out[10] = nf; + out[11] = 0; + out[12] = (left + right) * lr; + out[13] = (top + bottom) * bt; + out[14] = near * nf; + out[15] = 1; + return out; +} +/** + * Generates a look-at matrix with the given eye position, focal point, and up axis. + * If you want a matrix that actually makes an object look at another object, you should use targetTo instead. + * + * @param {mat4} out mat4 frustum matrix will be written into + * @param {ReadonlyVec3} eye Position of the viewer + * @param {ReadonlyVec3} center Point the viewer is looking at + * @param {ReadonlyVec3} up vec3 pointing up + * @returns {mat4} out + */ + +function lookAt(out, eye, center, up) { + var x0, x1, x2, y0, y1, y2, z0, z1, z2, len; + var eyex = eye[0]; + var eyey = eye[1]; + var eyez = eye[2]; + var upx = up[0]; + var upy = up[1]; + var upz = up[2]; + var centerx = center[0]; + var centery = center[1]; + var centerz = center[2]; + + if (Math.abs(eyex - centerx) < EPSILON && Math.abs(eyey - centery) < EPSILON && Math.abs(eyez - centerz) < EPSILON) { + return identity$2(out); + } + + z0 = eyex - centerx; + z1 = eyey - centery; + z2 = eyez - centerz; + len = 1 / Math.hypot(z0, z1, z2); + z0 *= len; + z1 *= len; + z2 *= len; + x0 = upy * z2 - upz * z1; + x1 = upz * z0 - upx * z2; + x2 = upx * z1 - upy * z0; + len = Math.hypot(x0, x1, x2); + + if (!len) { + x0 = 0; + x1 = 0; + x2 = 0; + } else { + len = 1 / len; + x0 *= len; + x1 *= len; + x2 *= len; + } + + y0 = z1 * x2 - z2 * x1; + y1 = z2 * x0 - z0 * x2; + y2 = z0 * x1 - z1 * x0; + len = Math.hypot(y0, y1, y2); + + if (!len) { + y0 = 0; + y1 = 0; + y2 = 0; + } else { + len = 1 / len; + y0 *= len; + y1 *= len; + y2 *= len; + } + + out[0] = x0; + out[1] = y0; + out[2] = z0; + out[3] = 0; + out[4] = x1; + out[5] = y1; + out[6] = z1; + out[7] = 0; + out[8] = x2; + out[9] = y2; + out[10] = z2; + out[11] = 0; + out[12] = -(x0 * eyex + x1 * eyey + x2 * eyez); + out[13] = -(y0 * eyex + y1 * eyey + y2 * eyez); + out[14] = -(z0 * eyex + z1 * eyey + z2 * eyez); + out[15] = 1; + return out; +} +/** + * Generates a matrix that makes something look at something else. + * + * @param {mat4} out mat4 frustum matrix will be written into + * @param {ReadonlyVec3} eye Position of the viewer + * @param {ReadonlyVec3} center Point the viewer is looking at + * @param {ReadonlyVec3} up vec3 pointing up + * @returns {mat4} out + */ + +function targetTo(out, eye, target, up) { + var eyex = eye[0], + eyey = eye[1], + eyez = eye[2], + upx = up[0], + upy = up[1], + upz = up[2]; + var z0 = eyex - target[0], + z1 = eyey - target[1], + z2 = eyez - target[2]; + var len = z0 * z0 + z1 * z1 + z2 * z2; + + if (len > 0) { + len = 1 / Math.sqrt(len); + z0 *= len; + z1 *= len; + z2 *= len; + } + + var x0 = upy * z2 - upz * z1, + x1 = upz * z0 - upx * z2, + x2 = upx * z1 - upy * z0; + len = x0 * x0 + x1 * x1 + x2 * x2; + + if (len > 0) { + len = 1 / Math.sqrt(len); + x0 *= len; + x1 *= len; + x2 *= len; + } + + out[0] = x0; + out[1] = x1; + out[2] = x2; + out[3] = 0; + out[4] = z1 * x2 - z2 * x1; + out[5] = z2 * x0 - z0 * x2; + out[6] = z0 * x1 - z1 * x0; + out[7] = 0; + out[8] = z0; + out[9] = z1; + out[10] = z2; + out[11] = 0; + out[12] = eyex; + out[13] = eyey; + out[14] = eyez; + out[15] = 1; + return out; +} +/** + * Returns a string representation of a mat4 + * + * @param {ReadonlyMat4} a matrix to represent as a string + * @returns {String} string representation of the matrix + */ + +function str$5(a) { + return "mat4(" + a[0] + ", " + a[1] + ", " + a[2] + ", " + a[3] + ", " + a[4] + ", " + a[5] + ", " + a[6] + ", " + a[7] + ", " + a[8] + ", " + a[9] + ", " + a[10] + ", " + a[11] + ", " + a[12] + ", " + a[13] + ", " + a[14] + ", " + a[15] + ")"; +} +/** + * Returns Frobenius norm of a mat4 + * + * @param {ReadonlyMat4} a the matrix to calculate Frobenius norm of + * @returns {Number} Frobenius norm + */ + +function frob(a) { + return Math.hypot(a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8], a[9], a[10], a[11], a[12], a[13], a[14], a[15]); +} +/** + * Adds two mat4's + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the first operand + * @param {ReadonlyMat4} b the second operand + * @returns {mat4} out + */ + +function add$5(out, a, b) { + out[0] = a[0] + b[0]; + out[1] = a[1] + b[1]; + out[2] = a[2] + b[2]; + out[3] = a[3] + b[3]; + out[4] = a[4] + b[4]; + out[5] = a[5] + b[5]; + out[6] = a[6] + b[6]; + out[7] = a[7] + b[7]; + out[8] = a[8] + b[8]; + out[9] = a[9] + b[9]; + out[10] = a[10] + b[10]; + out[11] = a[11] + b[11]; + out[12] = a[12] + b[12]; + out[13] = a[13] + b[13]; + out[14] = a[14] + b[14]; + out[15] = a[15] + b[15]; + return out; +} +/** + * Subtracts matrix b from matrix a + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the first operand + * @param {ReadonlyMat4} b the second operand + * @returns {mat4} out + */ + +function subtract$3(out, a, b) { + out[0] = a[0] - b[0]; + out[1] = a[1] - b[1]; + out[2] = a[2] - b[2]; + out[3] = a[3] - b[3]; + out[4] = a[4] - b[4]; + out[5] = a[5] - b[5]; + out[6] = a[6] - b[6]; + out[7] = a[7] - b[7]; + out[8] = a[8] - b[8]; + out[9] = a[9] - b[9]; + out[10] = a[10] - b[10]; + out[11] = a[11] - b[11]; + out[12] = a[12] - b[12]; + out[13] = a[13] - b[13]; + out[14] = a[14] - b[14]; + out[15] = a[15] - b[15]; + return out; +} +/** + * Multiply each element of the matrix by a scalar. + * + * @param {mat4} out the receiving matrix + * @param {ReadonlyMat4} a the matrix to scale + * @param {Number} b amount to scale the matrix's elements by + * @returns {mat4} out + */ + +function multiplyScalar(out, a, b) { + out[0] = a[0] * b; + out[1] = a[1] * b; + out[2] = a[2] * b; + out[3] = a[3] * b; + out[4] = a[4] * b; + out[5] = a[5] * b; + out[6] = a[6] * b; + out[7] = a[7] * b; + out[8] = a[8] * b; + out[9] = a[9] * b; + out[10] = a[10] * b; + out[11] = a[11] * b; + out[12] = a[12] * b; + out[13] = a[13] * b; + out[14] = a[14] * b; + out[15] = a[15] * b; + return out; +} +/** + * Adds two mat4's after multiplying each element of the second operand by a scalar value. + * + * @param {mat4} out the receiving vector + * @param {ReadonlyMat4} a the first operand + * @param {ReadonlyMat4} b the second operand + * @param {Number} scale the amount to scale b's elements by before adding + * @returns {mat4} out + */ + +function multiplyScalarAndAdd(out, a, b, scale) { + out[0] = a[0] + b[0] * scale; + out[1] = a[1] + b[1] * scale; + out[2] = a[2] + b[2] * scale; + out[3] = a[3] + b[3] * scale; + out[4] = a[4] + b[4] * scale; + out[5] = a[5] + b[5] * scale; + out[6] = a[6] + b[6] * scale; + out[7] = a[7] + b[7] * scale; + out[8] = a[8] + b[8] * scale; + out[9] = a[9] + b[9] * scale; + out[10] = a[10] + b[10] * scale; + out[11] = a[11] + b[11] * scale; + out[12] = a[12] + b[12] * scale; + out[13] = a[13] + b[13] * scale; + out[14] = a[14] + b[14] * scale; + out[15] = a[15] + b[15] * scale; + return out; +} +/** + * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===) + * + * @param {ReadonlyMat4} a The first matrix. + * @param {ReadonlyMat4} b The second matrix. + * @returns {Boolean} True if the matrices are equal, false otherwise. + */ + +function exactEquals$5(a, b) { + return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5] && a[6] === b[6] && a[7] === b[7] && a[8] === b[8] && a[9] === b[9] && a[10] === b[10] && a[11] === b[11] && a[12] === b[12] && a[13] === b[13] && a[14] === b[14] && a[15] === b[15]; +} +/** + * Returns whether or not the matrices have approximately the same elements in the same position. + * + * @param {ReadonlyMat4} a The first matrix. + * @param {ReadonlyMat4} b The second matrix. + * @returns {Boolean} True if the matrices are equal, false otherwise. + */ + +function equals$6(a, b) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3]; + var a4 = a[4], + a5 = a[5], + a6 = a[6], + a7 = a[7]; + var a8 = a[8], + a9 = a[9], + a10 = a[10], + a11 = a[11]; + var a12 = a[12], + a13 = a[13], + a14 = a[14], + a15 = a[15]; + var b0 = b[0], + b1 = b[1], + b2 = b[2], + b3 = b[3]; + var b4 = b[4], + b5 = b[5], + b6 = b[6], + b7 = b[7]; + var b8 = b[8], + b9 = b[9], + b10 = b[10], + b11 = b[11]; + var b12 = b[12], + b13 = b[13], + b14 = b[14], + b15 = b[15]; + return Math.abs(a0 - b0) <= EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5)) && Math.abs(a6 - b6) <= EPSILON * Math.max(1.0, Math.abs(a6), Math.abs(b6)) && Math.abs(a7 - b7) <= EPSILON * Math.max(1.0, Math.abs(a7), Math.abs(b7)) && Math.abs(a8 - b8) <= EPSILON * Math.max(1.0, Math.abs(a8), Math.abs(b8)) && Math.abs(a9 - b9) <= EPSILON * Math.max(1.0, Math.abs(a9), Math.abs(b9)) && Math.abs(a10 - b10) <= EPSILON * Math.max(1.0, Math.abs(a10), Math.abs(b10)) && Math.abs(a11 - b11) <= EPSILON * Math.max(1.0, Math.abs(a11), Math.abs(b11)) && Math.abs(a12 - b12) <= EPSILON * Math.max(1.0, Math.abs(a12), Math.abs(b12)) && Math.abs(a13 - b13) <= EPSILON * Math.max(1.0, Math.abs(a13), Math.abs(b13)) && Math.abs(a14 - b14) <= EPSILON * Math.max(1.0, Math.abs(a14), Math.abs(b14)) && Math.abs(a15 - b15) <= EPSILON * Math.max(1.0, Math.abs(a15), Math.abs(b15)); +} +/** + * Alias for {@link mat4.multiply} + * @function + */ + +var mul$5 = multiply$5; +/** + * Alias for {@link mat4.subtract} + * @function + */ + +var sub$3 = subtract$3; + +var mat4 = /*#__PURE__*/Object.freeze({ +__proto__: null, +add: add$5, +adjoint: adjoint, +clone: clone$5, +copy: copy$5, +create: create$5, +determinant: determinant, +equals: equals$6, +exactEquals: exactEquals$5, +frob: frob, +fromQuat: fromQuat, +fromQuat2: fromQuat2, +fromRotation: fromRotation$1, +fromRotationTranslation: fromRotationTranslation$1, +fromRotationTranslationScale: fromRotationTranslationScale, +fromRotationTranslationScaleOrigin: fromRotationTranslationScaleOrigin, +fromScaling: fromScaling, +fromTranslation: fromTranslation$1, +fromValues: fromValues$5, +fromXRotation: fromXRotation, +fromYRotation: fromYRotation, +fromZRotation: fromZRotation, +frustum: frustum, +getRotation: getRotation, +getScaling: getScaling, +getTranslation: getTranslation$1, +identity: identity$2, +invert: invert$2, +lookAt: lookAt, +mul: mul$5, +multiply: multiply$5, +multiplyScalar: multiplyScalar, +multiplyScalarAndAdd: multiplyScalarAndAdd, +ortho: ortho, +orthoNO: orthoNO, +orthoZO: orthoZO, +perspective: perspective, +perspectiveFromFieldOfView: perspectiveFromFieldOfView, +perspectiveNO: perspectiveNO, +perspectiveZO: perspectiveZO, +rotate: rotate$1, +rotateX: rotateX$3, +rotateY: rotateY$3, +rotateZ: rotateZ$3, +scale: scale$5, +set: set$5, +str: str$5, +sub: sub$3, +subtract: subtract$3, +targetTo: targetTo, +translate: translate$1, +transpose: transpose +}); + +/** + * 3 Dimensional Vector + * @module vec3 + */ + +/** + * Creates a new, empty vec3 + * + * @returns {vec3} a new 3D vector + */ + +function create$4() { + var out = new ARRAY_TYPE(3); + + if (ARRAY_TYPE != Float32Array) { + out[0] = 0; + out[1] = 0; + out[2] = 0; + } + + return out; +} +/** + * Creates a new vec3 initialized with values from an existing vector + * + * @param {ReadonlyVec3} a vector to clone + * @returns {vec3} a new 3D vector + */ + +function clone$4(a) { + var out = new ARRAY_TYPE(3); + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + return out; +} +/** + * Calculates the length of a vec3 + * + * @param {ReadonlyVec3} a vector to calculate length of + * @returns {Number} length of a + */ + +function length$4(a) { + var x = a[0]; + var y = a[1]; + var z = a[2]; + return Math.hypot(x, y, z); +} +/** + * Creates a new vec3 initialized with the given values + * + * @param {Number} x X component + * @param {Number} y Y component + * @param {Number} z Z component + * @returns {vec3} a new 3D vector + */ + +function fromValues$4(x, y, z) { + var out = new ARRAY_TYPE(3); + out[0] = x; + out[1] = y; + out[2] = z; + return out; +} +/** + * Copy the values from one vec3 to another + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the source vector + * @returns {vec3} out + */ + +function copy$4(out, a) { + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + return out; +} +/** + * Set the components of a vec3 to the given values + * + * @param {vec3} out the receiving vector + * @param {Number} x X component + * @param {Number} y Y component + * @param {Number} z Z component + * @returns {vec3} out + */ + +function set$4(out, x, y, z) { + out[0] = x; + out[1] = y; + out[2] = z; + return out; +} +/** + * Adds two vec3's + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @returns {vec3} out + */ + +function add$4(out, a, b) { + out[0] = a[0] + b[0]; + out[1] = a[1] + b[1]; + out[2] = a[2] + b[2]; + return out; +} +/** + * Subtracts vector b from vector a + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @returns {vec3} out + */ + +function subtract$2(out, a, b) { + out[0] = a[0] - b[0]; + out[1] = a[1] - b[1]; + out[2] = a[2] - b[2]; + return out; +} +/** + * Multiplies two vec3's + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @returns {vec3} out + */ + +function multiply$4(out, a, b) { + out[0] = a[0] * b[0]; + out[1] = a[1] * b[1]; + out[2] = a[2] * b[2]; + return out; +} +/** + * Divides two vec3's + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @returns {vec3} out + */ + +function divide$2(out, a, b) { + out[0] = a[0] / b[0]; + out[1] = a[1] / b[1]; + out[2] = a[2] / b[2]; + return out; +} +/** + * Math.ceil the components of a vec3 + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a vector to ceil + * @returns {vec3} out + */ + +function ceil$2(out, a) { + out[0] = Math.ceil(a[0]); + out[1] = Math.ceil(a[1]); + out[2] = Math.ceil(a[2]); + return out; +} +/** + * Math.floor the components of a vec3 + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a vector to floor + * @returns {vec3} out + */ + +function floor$2(out, a) { + out[0] = Math.floor(a[0]); + out[1] = Math.floor(a[1]); + out[2] = Math.floor(a[2]); + return out; +} +/** + * Returns the minimum of two vec3's + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @returns {vec3} out + */ + +function min$2(out, a, b) { + out[0] = Math.min(a[0], b[0]); + out[1] = Math.min(a[1], b[1]); + out[2] = Math.min(a[2], b[2]); + return out; +} +/** + * Returns the maximum of two vec3's + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @returns {vec3} out + */ + +function max$2(out, a, b) { + out[0] = Math.max(a[0], b[0]); + out[1] = Math.max(a[1], b[1]); + out[2] = Math.max(a[2], b[2]); + return out; +} +/** + * Math.round the components of a vec3 + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a vector to round + * @returns {vec3} out + */ + +function round$2(out, a) { + out[0] = Math.round(a[0]); + out[1] = Math.round(a[1]); + out[2] = Math.round(a[2]); + return out; +} +/** + * Scales a vec3 by a scalar number + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the vector to scale + * @param {Number} b amount to scale the vector by + * @returns {vec3} out + */ + +function scale$4(out, a, b) { + out[0] = a[0] * b; + out[1] = a[1] * b; + out[2] = a[2] * b; + return out; +} +/** + * Adds two vec3's after scaling the second operand by a scalar value + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @param {Number} scale the amount to scale b by before adding + * @returns {vec3} out + */ + +function scaleAndAdd$2(out, a, b, scale) { + out[0] = a[0] + b[0] * scale; + out[1] = a[1] + b[1] * scale; + out[2] = a[2] + b[2] * scale; + return out; +} +/** + * Calculates the euclidian distance between two vec3's + * + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @returns {Number} distance between a and b + */ + +function distance$2(a, b) { + var x = b[0] - a[0]; + var y = b[1] - a[1]; + var z = b[2] - a[2]; + return Math.hypot(x, y, z); +} +/** + * Calculates the squared euclidian distance between two vec3's + * + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @returns {Number} squared distance between a and b + */ + +function squaredDistance$2(a, b) { + var x = b[0] - a[0]; + var y = b[1] - a[1]; + var z = b[2] - a[2]; + return x * x + y * y + z * z; +} +/** + * Calculates the squared length of a vec3 + * + * @param {ReadonlyVec3} a vector to calculate squared length of + * @returns {Number} squared length of a + */ + +function squaredLength$4(a) { + var x = a[0]; + var y = a[1]; + var z = a[2]; + return x * x + y * y + z * z; +} +/** + * Negates the components of a vec3 + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a vector to negate + * @returns {vec3} out + */ + +function negate$2(out, a) { + out[0] = -a[0]; + out[1] = -a[1]; + out[2] = -a[2]; + return out; +} +/** + * Returns the inverse of the components of a vec3 + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a vector to invert + * @returns {vec3} out + */ + +function inverse$2(out, a) { + out[0] = 1.0 / a[0]; + out[1] = 1.0 / a[1]; + out[2] = 1.0 / a[2]; + return out; +} +/** + * Normalize a vec3 + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a vector to normalize + * @returns {vec3} out + */ + +function normalize$4(out, a) { + var x = a[0]; + var y = a[1]; + var z = a[2]; + var len = x * x + y * y + z * z; + + if (len > 0) { + //TODO: evaluate use of glm_invsqrt here? + len = 1 / Math.sqrt(len); + } + + out[0] = a[0] * len; + out[1] = a[1] * len; + out[2] = a[2] * len; + return out; +} +/** + * Calculates the dot product of two vec3's + * + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @returns {Number} dot product of a and b + */ + +function dot$5(a, b) { + return a[0] * b[0] + a[1] * b[1] + a[2] * b[2]; +} +/** + * Computes the cross product of two vec3's + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @returns {vec3} out + */ + +function cross$2(out, a, b) { + var ax = a[0], + ay = a[1], + az = a[2]; + var bx = b[0], + by = b[1], + bz = b[2]; + out[0] = ay * bz - az * by; + out[1] = az * bx - ax * bz; + out[2] = ax * by - ay * bx; + return out; +} +/** + * Performs a linear interpolation between two vec3's + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @param {Number} t interpolation amount, in the range [0-1], between the two inputs + * @returns {vec3} out + */ + +function lerp$4(out, a, b, t) { + var ax = a[0]; + var ay = a[1]; + var az = a[2]; + out[0] = ax + t * (b[0] - ax); + out[1] = ay + t * (b[1] - ay); + out[2] = az + t * (b[2] - az); + return out; +} +/** + * Performs a hermite interpolation with two control points + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @param {ReadonlyVec3} c the third operand + * @param {ReadonlyVec3} d the fourth operand + * @param {Number} t interpolation amount, in the range [0-1], between the two inputs + * @returns {vec3} out + */ + +function hermite(out, a, b, c, d, t) { + var factorTimes2 = t * t; + var factor1 = factorTimes2 * (2 * t - 3) + 1; + var factor2 = factorTimes2 * (t - 2) + t; + var factor3 = factorTimes2 * (t - 1); + var factor4 = factorTimes2 * (3 - 2 * t); + out[0] = a[0] * factor1 + b[0] * factor2 + c[0] * factor3 + d[0] * factor4; + out[1] = a[1] * factor1 + b[1] * factor2 + c[1] * factor3 + d[1] * factor4; + out[2] = a[2] * factor1 + b[2] * factor2 + c[2] * factor3 + d[2] * factor4; + return out; +} +/** + * Performs a bezier interpolation with two control points + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the first operand + * @param {ReadonlyVec3} b the second operand + * @param {ReadonlyVec3} c the third operand + * @param {ReadonlyVec3} d the fourth operand + * @param {Number} t interpolation amount, in the range [0-1], between the two inputs + * @returns {vec3} out + */ + +function bezier(out, a, b, c, d, t) { + var inverseFactor = 1 - t; + var inverseFactorTimesTwo = inverseFactor * inverseFactor; + var factorTimes2 = t * t; + var factor1 = inverseFactorTimesTwo * inverseFactor; + var factor2 = 3 * t * inverseFactorTimesTwo; + var factor3 = 3 * factorTimes2 * inverseFactor; + var factor4 = factorTimes2 * t; + out[0] = a[0] * factor1 + b[0] * factor2 + c[0] * factor3 + d[0] * factor4; + out[1] = a[1] * factor1 + b[1] * factor2 + c[1] * factor3 + d[1] * factor4; + out[2] = a[2] * factor1 + b[2] * factor2 + c[2] * factor3 + d[2] * factor4; + return out; +} +/** + * Generates a random vector with the given scale + * + * @param {vec3} out the receiving vector + * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned + * @returns {vec3} out + */ + +function random$3(out, scale) { + scale = scale || 1.0; + var r = RANDOM() * 2.0 * Math.PI; + var z = RANDOM() * 2.0 - 1.0; + var zScale = Math.sqrt(1.0 - z * z) * scale; + out[0] = Math.cos(r) * zScale; + out[1] = Math.sin(r) * zScale; + out[2] = z * scale; + return out; +} +/** + * Transforms the vec3 with a mat4. + * 4th vector component is implicitly '1' + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the vector to transform + * @param {ReadonlyMat4} m matrix to transform with + * @returns {vec3} out + */ + +function transformMat4$2(out, a, m) { + var x = a[0], + y = a[1], + z = a[2]; + var w = m[3] * x + m[7] * y + m[11] * z + m[15]; + w = w || 1.0; + out[0] = (m[0] * x + m[4] * y + m[8] * z + m[12]) / w; + out[1] = (m[1] * x + m[5] * y + m[9] * z + m[13]) / w; + out[2] = (m[2] * x + m[6] * y + m[10] * z + m[14]) / w; + return out; +} +/** + * Transforms the vec3 with a mat3. + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the vector to transform + * @param {ReadonlyMat3} m the 3x3 matrix to transform with + * @returns {vec3} out + */ + +function transformMat3$1(out, a, m) { + var x = a[0], + y = a[1], + z = a[2]; + out[0] = x * m[0] + y * m[3] + z * m[6]; + out[1] = x * m[1] + y * m[4] + z * m[7]; + out[2] = x * m[2] + y * m[5] + z * m[8]; + return out; +} +/** + * Transforms the vec3 with a quat + * Can also be used for dual quaternions. (Multiply it with the real part) + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec3} a the vector to transform + * @param {ReadonlyQuat} q quaternion to transform with + * @returns {vec3} out + */ + +function transformQuat$1(out, a, q) { + // benchmarks: https://jsperf.com/quaternion-transform-vec3-implementations-fixed + var qx = q[0], + qy = q[1], + qz = q[2], + qw = q[3]; + var x = a[0], + y = a[1], + z = a[2]; // var qvec = [qx, qy, qz]; + // var uv = vec3.cross([], qvec, a); + + var uvx = qy * z - qz * y, + uvy = qz * x - qx * z, + uvz = qx * y - qy * x; // var uuv = vec3.cross([], qvec, uv); + + var uuvx = qy * uvz - qz * uvy, + uuvy = qz * uvx - qx * uvz, + uuvz = qx * uvy - qy * uvx; // vec3.scale(uv, uv, 2 * w); + + var w2 = qw * 2; + uvx *= w2; + uvy *= w2; + uvz *= w2; // vec3.scale(uuv, uuv, 2); + + uuvx *= 2; + uuvy *= 2; + uuvz *= 2; // return vec3.add(out, a, vec3.add(out, uv, uuv)); + + out[0] = x + uvx + uuvx; + out[1] = y + uvy + uuvy; + out[2] = z + uvz + uuvz; + return out; +} +/** + * Rotate a 3D vector around the x-axis + * @param {vec3} out The receiving vec3 + * @param {ReadonlyVec3} a The vec3 point to rotate + * @param {ReadonlyVec3} b The origin of the rotation + * @param {Number} rad The angle of rotation in radians + * @returns {vec3} out + */ + +function rotateX$2(out, a, b, rad) { + var p = [], + r = []; //Translate point to the origin + + p[0] = a[0] - b[0]; + p[1] = a[1] - b[1]; + p[2] = a[2] - b[2]; //perform rotation + + r[0] = p[0]; + r[1] = p[1] * Math.cos(rad) - p[2] * Math.sin(rad); + r[2] = p[1] * Math.sin(rad) + p[2] * Math.cos(rad); //translate to correct position + + out[0] = r[0] + b[0]; + out[1] = r[1] + b[1]; + out[2] = r[2] + b[2]; + return out; +} +/** + * Rotate a 3D vector around the y-axis + * @param {vec3} out The receiving vec3 + * @param {ReadonlyVec3} a The vec3 point to rotate + * @param {ReadonlyVec3} b The origin of the rotation + * @param {Number} rad The angle of rotation in radians + * @returns {vec3} out + */ + +function rotateY$2(out, a, b, rad) { + var p = [], + r = []; //Translate point to the origin + + p[0] = a[0] - b[0]; + p[1] = a[1] - b[1]; + p[2] = a[2] - b[2]; //perform rotation + + r[0] = p[2] * Math.sin(rad) + p[0] * Math.cos(rad); + r[1] = p[1]; + r[2] = p[2] * Math.cos(rad) - p[0] * Math.sin(rad); //translate to correct position + + out[0] = r[0] + b[0]; + out[1] = r[1] + b[1]; + out[2] = r[2] + b[2]; + return out; +} +/** + * Rotate a 3D vector around the z-axis + * @param {vec3} out The receiving vec3 + * @param {ReadonlyVec3} a The vec3 point to rotate + * @param {ReadonlyVec3} b The origin of the rotation + * @param {Number} rad The angle of rotation in radians + * @returns {vec3} out + */ + +function rotateZ$2(out, a, b, rad) { + var p = [], + r = []; //Translate point to the origin + + p[0] = a[0] - b[0]; + p[1] = a[1] - b[1]; + p[2] = a[2] - b[2]; //perform rotation + + r[0] = p[0] * Math.cos(rad) - p[1] * Math.sin(rad); + r[1] = p[0] * Math.sin(rad) + p[1] * Math.cos(rad); + r[2] = p[2]; //translate to correct position + + out[0] = r[0] + b[0]; + out[1] = r[1] + b[1]; + out[2] = r[2] + b[2]; + return out; +} +/** + * Get the angle between two 3D vectors + * @param {ReadonlyVec3} a The first operand + * @param {ReadonlyVec3} b The second operand + * @returns {Number} The angle in radians + */ + +function angle$1(a, b) { + var ax = a[0], + ay = a[1], + az = a[2], + bx = b[0], + by = b[1], + bz = b[2], + mag1 = Math.sqrt(ax * ax + ay * ay + az * az), + mag2 = Math.sqrt(bx * bx + by * by + bz * bz), + mag = mag1 * mag2, + cosine = mag && dot$5(a, b) / mag; + return Math.acos(Math.min(Math.max(cosine, -1), 1)); +} +/** + * Set the components of a vec3 to zero + * + * @param {vec3} out the receiving vector + * @returns {vec3} out + */ + +function zero$2(out) { + out[0] = 0.0; + out[1] = 0.0; + out[2] = 0.0; + return out; +} +/** + * Returns a string representation of a vector + * + * @param {ReadonlyVec3} a vector to represent as a string + * @returns {String} string representation of the vector + */ + +function str$4(a) { + return "vec3(" + a[0] + ", " + a[1] + ", " + a[2] + ")"; +} +/** + * Returns whether or not the vectors have exactly the same elements in the same position (when compared with ===) + * + * @param {ReadonlyVec3} a The first vector. + * @param {ReadonlyVec3} b The second vector. + * @returns {Boolean} True if the vectors are equal, false otherwise. + */ + +function exactEquals$4(a, b) { + return a[0] === b[0] && a[1] === b[1] && a[2] === b[2]; +} +/** + * Returns whether or not the vectors have approximately the same elements in the same position. + * + * @param {ReadonlyVec3} a The first vector. + * @param {ReadonlyVec3} b The second vector. + * @returns {Boolean} True if the vectors are equal, false otherwise. + */ + +function equals$5(a, b) { + var a0 = a[0], + a1 = a[1], + a2 = a[2]; + var b0 = b[0], + b1 = b[1], + b2 = b[2]; + return Math.abs(a0 - b0) <= EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)); +} +/** + * Alias for {@link vec3.subtract} + * @function + */ + +var sub$2 = subtract$2; +/** + * Alias for {@link vec3.multiply} + * @function + */ + +var mul$4 = multiply$4; +/** + * Alias for {@link vec3.divide} + * @function + */ + +var div$2 = divide$2; +/** + * Alias for {@link vec3.distance} + * @function + */ + +var dist$2 = distance$2; +/** + * Alias for {@link vec3.squaredDistance} + * @function + */ + +var sqrDist$2 = squaredDistance$2; +/** + * Alias for {@link vec3.length} + * @function + */ + +var len$4 = length$4; +/** + * Alias for {@link vec3.squaredLength} + * @function + */ + +var sqrLen$4 = squaredLength$4; +/** + * Perform some operation over an array of vec3s. + * + * @param {Array} a the array of vectors to iterate over + * @param {Number} stride Number of elements between the start of each vec3. If 0 assumes tightly packed + * @param {Number} offset Number of elements to skip at the beginning of the array + * @param {Number} count Number of vec3s to iterate over. If 0 iterates over entire array + * @param {Function} fn Function to call for each vector in the array + * @param {Object} [arg] additional argument to pass to fn + * @returns {Array} a + * @function + */ + +var forEach$2 = function () { + var vec = create$4(); + return function (a, stride, offset, count, fn, arg) { + var i, l; + + if (!stride) { + stride = 3; + } + + if (!offset) { + offset = 0; + } + + if (count) { + l = Math.min(count * stride + offset, a.length); + } else { + l = a.length; + } + + for (i = offset; i < l; i += stride) { + vec[0] = a[i]; + vec[1] = a[i + 1]; + vec[2] = a[i + 2]; + fn(vec, vec, arg); + a[i] = vec[0]; + a[i + 1] = vec[1]; + a[i + 2] = vec[2]; + } + + return a; + }; +}(); + +var vec3 = /*#__PURE__*/Object.freeze({ +__proto__: null, +add: add$4, +angle: angle$1, +bezier: bezier, +ceil: ceil$2, +clone: clone$4, +copy: copy$4, +create: create$4, +cross: cross$2, +dist: dist$2, +distance: distance$2, +div: div$2, +divide: divide$2, +dot: dot$5, +equals: equals$5, +exactEquals: exactEquals$4, +floor: floor$2, +forEach: forEach$2, +fromValues: fromValues$4, +hermite: hermite, +inverse: inverse$2, +len: len$4, +length: length$4, +lerp: lerp$4, +max: max$2, +min: min$2, +mul: mul$4, +multiply: multiply$4, +negate: negate$2, +normalize: normalize$4, +random: random$3, +rotateX: rotateX$2, +rotateY: rotateY$2, +rotateZ: rotateZ$2, +round: round$2, +scale: scale$4, +scaleAndAdd: scaleAndAdd$2, +set: set$4, +sqrDist: sqrDist$2, +sqrLen: sqrLen$4, +squaredDistance: squaredDistance$2, +squaredLength: squaredLength$4, +str: str$4, +sub: sub$2, +subtract: subtract$2, +transformMat3: transformMat3$1, +transformMat4: transformMat4$2, +transformQuat: transformQuat$1, +zero: zero$2 +}); + +/** + * 4 Dimensional Vector + * @module vec4 + */ + +/** + * Creates a new, empty vec4 + * + * @returns {vec4} a new 4D vector + */ + +function create$3() { + var out = new ARRAY_TYPE(4); + + if (ARRAY_TYPE != Float32Array) { + out[0] = 0; + out[1] = 0; + out[2] = 0; + out[3] = 0; + } + + return out; +} +/** + * Creates a new vec4 initialized with values from an existing vector + * + * @param {ReadonlyVec4} a vector to clone + * @returns {vec4} a new 4D vector + */ + +function clone$3(a) { + var out = new ARRAY_TYPE(4); + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[3]; + return out; +} +/** + * Creates a new vec4 initialized with the given values + * + * @param {Number} x X component + * @param {Number} y Y component + * @param {Number} z Z component + * @param {Number} w W component + * @returns {vec4} a new 4D vector + */ + +function fromValues$3(x, y, z, w) { + var out = new ARRAY_TYPE(4); + out[0] = x; + out[1] = y; + out[2] = z; + out[3] = w; + return out; +} +/** + * Copy the values from one vec4 to another + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a the source vector + * @returns {vec4} out + */ + +function copy$3(out, a) { + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[3]; + return out; +} +/** + * Set the components of a vec4 to the given values + * + * @param {vec4} out the receiving vector + * @param {Number} x X component + * @param {Number} y Y component + * @param {Number} z Z component + * @param {Number} w W component + * @returns {vec4} out + */ + +function set$3(out, x, y, z, w) { + out[0] = x; + out[1] = y; + out[2] = z; + out[3] = w; + return out; +} +/** + * Adds two vec4's + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a the first operand + * @param {ReadonlyVec4} b the second operand + * @returns {vec4} out + */ + +function add$3(out, a, b) { + out[0] = a[0] + b[0]; + out[1] = a[1] + b[1]; + out[2] = a[2] + b[2]; + out[3] = a[3] + b[3]; + return out; +} +/** + * Subtracts vector b from vector a + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a the first operand + * @param {ReadonlyVec4} b the second operand + * @returns {vec4} out + */ + +function subtract$1(out, a, b) { + out[0] = a[0] - b[0]; + out[1] = a[1] - b[1]; + out[2] = a[2] - b[2]; + out[3] = a[3] - b[3]; + return out; +} +/** + * Multiplies two vec4's + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a the first operand + * @param {ReadonlyVec4} b the second operand + * @returns {vec4} out + */ + +function multiply$3(out, a, b) { + out[0] = a[0] * b[0]; + out[1] = a[1] * b[1]; + out[2] = a[2] * b[2]; + out[3] = a[3] * b[3]; + return out; +} +/** + * Divides two vec4's + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a the first operand + * @param {ReadonlyVec4} b the second operand + * @returns {vec4} out + */ + +function divide$1(out, a, b) { + out[0] = a[0] / b[0]; + out[1] = a[1] / b[1]; + out[2] = a[2] / b[2]; + out[3] = a[3] / b[3]; + return out; +} +/** + * Math.ceil the components of a vec4 + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a vector to ceil + * @returns {vec4} out + */ + +function ceil$1(out, a) { + out[0] = Math.ceil(a[0]); + out[1] = Math.ceil(a[1]); + out[2] = Math.ceil(a[2]); + out[3] = Math.ceil(a[3]); + return out; +} +/** + * Math.floor the components of a vec4 + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a vector to floor + * @returns {vec4} out + */ + +function floor$1(out, a) { + out[0] = Math.floor(a[0]); + out[1] = Math.floor(a[1]); + out[2] = Math.floor(a[2]); + out[3] = Math.floor(a[3]); + return out; +} +/** + * Returns the minimum of two vec4's + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a the first operand + * @param {ReadonlyVec4} b the second operand + * @returns {vec4} out + */ + +function min$1(out, a, b) { + out[0] = Math.min(a[0], b[0]); + out[1] = Math.min(a[1], b[1]); + out[2] = Math.min(a[2], b[2]); + out[3] = Math.min(a[3], b[3]); + return out; +} +/** + * Returns the maximum of two vec4's + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a the first operand + * @param {ReadonlyVec4} b the second operand + * @returns {vec4} out + */ + +function max$1(out, a, b) { + out[0] = Math.max(a[0], b[0]); + out[1] = Math.max(a[1], b[1]); + out[2] = Math.max(a[2], b[2]); + out[3] = Math.max(a[3], b[3]); + return out; +} +/** + * Math.round the components of a vec4 + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a vector to round + * @returns {vec4} out + */ + +function round$1(out, a) { + out[0] = Math.round(a[0]); + out[1] = Math.round(a[1]); + out[2] = Math.round(a[2]); + out[3] = Math.round(a[3]); + return out; +} +/** + * Scales a vec4 by a scalar number + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a the vector to scale + * @param {Number} b amount to scale the vector by + * @returns {vec4} out + */ + +function scale$3(out, a, b) { + out[0] = a[0] * b; + out[1] = a[1] * b; + out[2] = a[2] * b; + out[3] = a[3] * b; + return out; +} +/** + * Adds two vec4's after scaling the second operand by a scalar value + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a the first operand + * @param {ReadonlyVec4} b the second operand + * @param {Number} scale the amount to scale b by before adding + * @returns {vec4} out + */ + +function scaleAndAdd$1(out, a, b, scale) { + out[0] = a[0] + b[0] * scale; + out[1] = a[1] + b[1] * scale; + out[2] = a[2] + b[2] * scale; + out[3] = a[3] + b[3] * scale; + return out; +} +/** + * Calculates the euclidian distance between two vec4's + * + * @param {ReadonlyVec4} a the first operand + * @param {ReadonlyVec4} b the second operand + * @returns {Number} distance between a and b + */ + +function distance$1(a, b) { + var x = b[0] - a[0]; + var y = b[1] - a[1]; + var z = b[2] - a[2]; + var w = b[3] - a[3]; + return Math.hypot(x, y, z, w); +} +/** + * Calculates the squared euclidian distance between two vec4's + * + * @param {ReadonlyVec4} a the first operand + * @param {ReadonlyVec4} b the second operand + * @returns {Number} squared distance between a and b + */ + +function squaredDistance$1(a, b) { + var x = b[0] - a[0]; + var y = b[1] - a[1]; + var z = b[2] - a[2]; + var w = b[3] - a[3]; + return x * x + y * y + z * z + w * w; +} +/** + * Calculates the length of a vec4 + * + * @param {ReadonlyVec4} a vector to calculate length of + * @returns {Number} length of a + */ + +function length$3(a) { + var x = a[0]; + var y = a[1]; + var z = a[2]; + var w = a[3]; + return Math.hypot(x, y, z, w); +} +/** + * Calculates the squared length of a vec4 + * + * @param {ReadonlyVec4} a vector to calculate squared length of + * @returns {Number} squared length of a + */ + +function squaredLength$3(a) { + var x = a[0]; + var y = a[1]; + var z = a[2]; + var w = a[3]; + return x * x + y * y + z * z + w * w; +} +/** + * Negates the components of a vec4 + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a vector to negate + * @returns {vec4} out + */ + +function negate$1(out, a) { + out[0] = -a[0]; + out[1] = -a[1]; + out[2] = -a[2]; + out[3] = -a[3]; + return out; +} +/** + * Returns the inverse of the components of a vec4 + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a vector to invert + * @returns {vec4} out + */ + +function inverse$1(out, a) { + out[0] = 1.0 / a[0]; + out[1] = 1.0 / a[1]; + out[2] = 1.0 / a[2]; + out[3] = 1.0 / a[3]; + return out; +} +/** + * Normalize a vec4 + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a vector to normalize + * @returns {vec4} out + */ + +function normalize$3(out, a) { + var x = a[0]; + var y = a[1]; + var z = a[2]; + var w = a[3]; + var len = x * x + y * y + z * z + w * w; + + if (len > 0) { + len = 1 / Math.sqrt(len); + } + + out[0] = x * len; + out[1] = y * len; + out[2] = z * len; + out[3] = w * len; + return out; +} +/** + * Calculates the dot product of two vec4's + * + * @param {ReadonlyVec4} a the first operand + * @param {ReadonlyVec4} b the second operand + * @returns {Number} dot product of a and b + */ + +function dot$4(a, b) { + return a[0] * b[0] + a[1] * b[1] + a[2] * b[2] + a[3] * b[3]; +} +/** + * Returns the cross-product of three vectors in a 4-dimensional space + * + * @param {ReadonlyVec4} result the receiving vector + * @param {ReadonlyVec4} U the first vector + * @param {ReadonlyVec4} V the second vector + * @param {ReadonlyVec4} W the third vector + * @returns {vec4} result + */ + +function cross$1(out, u, v, w) { + var A = v[0] * w[1] - v[1] * w[0], + B = v[0] * w[2] - v[2] * w[0], + C = v[0] * w[3] - v[3] * w[0], + D = v[1] * w[2] - v[2] * w[1], + E = v[1] * w[3] - v[3] * w[1], + F = v[2] * w[3] - v[3] * w[2]; + var G = u[0]; + var H = u[1]; + var I = u[2]; + var J = u[3]; + out[0] = H * F - I * E + J * D; + out[1] = -(G * F) + I * C - J * B; + out[2] = G * E - H * C + J * A; + out[3] = -(G * D) + H * B - I * A; + return out; +} +/** + * Performs a linear interpolation between two vec4's + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a the first operand + * @param {ReadonlyVec4} b the second operand + * @param {Number} t interpolation amount, in the range [0-1], between the two inputs + * @returns {vec4} out + */ + +function lerp$3(out, a, b, t) { + var ax = a[0]; + var ay = a[1]; + var az = a[2]; + var aw = a[3]; + out[0] = ax + t * (b[0] - ax); + out[1] = ay + t * (b[1] - ay); + out[2] = az + t * (b[2] - az); + out[3] = aw + t * (b[3] - aw); + return out; +} +/** + * Generates a random vector with the given scale + * + * @param {vec4} out the receiving vector + * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned + * @returns {vec4} out + */ + +function random$2(out, scale) { + scale = scale || 1.0; // Marsaglia, George. Choosing a Point from the Surface of a + // Sphere. Ann. Math. Statist. 43 (1972), no. 2, 645--646. + // http://projecteuclid.org/euclid.aoms/1177692644; + + var v1, v2, v3, v4; + var s1, s2; + + do { + v1 = RANDOM() * 2 - 1; + v2 = RANDOM() * 2 - 1; + s1 = v1 * v1 + v2 * v2; + } while (s1 >= 1); + + do { + v3 = RANDOM() * 2 - 1; + v4 = RANDOM() * 2 - 1; + s2 = v3 * v3 + v4 * v4; + } while (s2 >= 1); + + var d = Math.sqrt((1 - s1) / s2); + out[0] = scale * v1; + out[1] = scale * v2; + out[2] = scale * v3 * d; + out[3] = scale * v4 * d; + return out; +} +/** + * Transforms the vec4 with a mat4. + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a the vector to transform + * @param {ReadonlyMat4} m matrix to transform with + * @returns {vec4} out + */ + +function transformMat4$1(out, a, m) { + var x = a[0], + y = a[1], + z = a[2], + w = a[3]; + out[0] = m[0] * x + m[4] * y + m[8] * z + m[12] * w; + out[1] = m[1] * x + m[5] * y + m[9] * z + m[13] * w; + out[2] = m[2] * x + m[6] * y + m[10] * z + m[14] * w; + out[3] = m[3] * x + m[7] * y + m[11] * z + m[15] * w; + return out; +} +/** + * Transforms the vec4 with a quat + * + * @param {vec4} out the receiving vector + * @param {ReadonlyVec4} a the vector to transform + * @param {ReadonlyQuat} q quaternion to transform with + * @returns {vec4} out + */ + +function transformQuat(out, a, q) { + var x = a[0], + y = a[1], + z = a[2]; + var qx = q[0], + qy = q[1], + qz = q[2], + qw = q[3]; // calculate quat * vec + + var ix = qw * x + qy * z - qz * y; + var iy = qw * y + qz * x - qx * z; + var iz = qw * z + qx * y - qy * x; + var iw = -qx * x - qy * y - qz * z; // calculate result * inverse quat + + out[0] = ix * qw + iw * -qx + iy * -qz - iz * -qy; + out[1] = iy * qw + iw * -qy + iz * -qx - ix * -qz; + out[2] = iz * qw + iw * -qz + ix * -qy - iy * -qx; + out[3] = a[3]; + return out; +} +/** + * Set the components of a vec4 to zero + * + * @param {vec4} out the receiving vector + * @returns {vec4} out + */ + +function zero$1(out) { + out[0] = 0.0; + out[1] = 0.0; + out[2] = 0.0; + out[3] = 0.0; + return out; +} +/** + * Returns a string representation of a vector + * + * @param {ReadonlyVec4} a vector to represent as a string + * @returns {String} string representation of the vector + */ + +function str$3(a) { + return "vec4(" + a[0] + ", " + a[1] + ", " + a[2] + ", " + a[3] + ")"; +} +/** + * Returns whether or not the vectors have exactly the same elements in the same position (when compared with ===) + * + * @param {ReadonlyVec4} a The first vector. + * @param {ReadonlyVec4} b The second vector. + * @returns {Boolean} True if the vectors are equal, false otherwise. + */ + +function exactEquals$3(a, b) { + return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3]; +} +/** + * Returns whether or not the vectors have approximately the same elements in the same position. + * + * @param {ReadonlyVec4} a The first vector. + * @param {ReadonlyVec4} b The second vector. + * @returns {Boolean} True if the vectors are equal, false otherwise. + */ + +function equals$4(a, b) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3]; + var b0 = b[0], + b1 = b[1], + b2 = b[2], + b3 = b[3]; + return Math.abs(a0 - b0) <= EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)); +} +/** + * Alias for {@link vec4.subtract} + * @function + */ + +var sub$1 = subtract$1; +/** + * Alias for {@link vec4.multiply} + * @function + */ + +var mul$3 = multiply$3; +/** + * Alias for {@link vec4.divide} + * @function + */ + +var div$1 = divide$1; +/** + * Alias for {@link vec4.distance} + * @function + */ + +var dist$1 = distance$1; +/** + * Alias for {@link vec4.squaredDistance} + * @function + */ + +var sqrDist$1 = squaredDistance$1; +/** + * Alias for {@link vec4.length} + * @function + */ + +var len$3 = length$3; +/** + * Alias for {@link vec4.squaredLength} + * @function + */ + +var sqrLen$3 = squaredLength$3; +/** + * Perform some operation over an array of vec4s. + * + * @param {Array} a the array of vectors to iterate over + * @param {Number} stride Number of elements between the start of each vec4. If 0 assumes tightly packed + * @param {Number} offset Number of elements to skip at the beginning of the array + * @param {Number} count Number of vec4s to iterate over. If 0 iterates over entire array + * @param {Function} fn Function to call for each vector in the array + * @param {Object} [arg] additional argument to pass to fn + * @returns {Array} a + * @function + */ + +var forEach$1 = function () { + var vec = create$3(); + return function (a, stride, offset, count, fn, arg) { + var i, l; + + if (!stride) { + stride = 4; + } + + if (!offset) { + offset = 0; + } + + if (count) { + l = Math.min(count * stride + offset, a.length); + } else { + l = a.length; + } + + for (i = offset; i < l; i += stride) { + vec[0] = a[i]; + vec[1] = a[i + 1]; + vec[2] = a[i + 2]; + vec[3] = a[i + 3]; + fn(vec, vec, arg); + a[i] = vec[0]; + a[i + 1] = vec[1]; + a[i + 2] = vec[2]; + a[i + 3] = vec[3]; + } + + return a; + }; +}(); + +var vec4 = /*#__PURE__*/Object.freeze({ +__proto__: null, +add: add$3, +ceil: ceil$1, +clone: clone$3, +copy: copy$3, +create: create$3, +cross: cross$1, +dist: dist$1, +distance: distance$1, +div: div$1, +divide: divide$1, +dot: dot$4, +equals: equals$4, +exactEquals: exactEquals$3, +floor: floor$1, +forEach: forEach$1, +fromValues: fromValues$3, +inverse: inverse$1, +len: len$3, +length: length$3, +lerp: lerp$3, +max: max$1, +min: min$1, +mul: mul$3, +multiply: multiply$3, +negate: negate$1, +normalize: normalize$3, +random: random$2, +round: round$1, +scale: scale$3, +scaleAndAdd: scaleAndAdd$1, +set: set$3, +sqrDist: sqrDist$1, +sqrLen: sqrLen$3, +squaredDistance: squaredDistance$1, +squaredLength: squaredLength$3, +str: str$3, +sub: sub$1, +subtract: subtract$1, +transformMat4: transformMat4$1, +transformQuat: transformQuat, +zero: zero$1 +}); + +/** + * Quaternion + * @module quat + */ + +/** + * Creates a new identity quat + * + * @returns {quat} a new quaternion + */ + +function create$2() { + var out = new ARRAY_TYPE(4); + + if (ARRAY_TYPE != Float32Array) { + out[0] = 0; + out[1] = 0; + out[2] = 0; + } + + out[3] = 1; + return out; +} +/** + * Set a quat to the identity quaternion + * + * @param {quat} out the receiving quaternion + * @returns {quat} out + */ + +function identity$1(out) { + out[0] = 0; + out[1] = 0; + out[2] = 0; + out[3] = 1; + return out; +} +/** + * Sets a quat from the given angle and rotation axis, + * then returns it. + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyVec3} axis the axis around which to rotate + * @param {Number} rad the angle in radians + * @returns {quat} out + **/ + +function setAxisAngle(out, axis, rad) { + rad = rad * 0.5; + var s = Math.sin(rad); + out[0] = s * axis[0]; + out[1] = s * axis[1]; + out[2] = s * axis[2]; + out[3] = Math.cos(rad); + return out; +} +/** + * Gets the rotation axis and angle for a given + * quaternion. If a quaternion is created with + * setAxisAngle, this method will return the same + * values as providied in the original parameter list + * OR functionally equivalent values. + * Example: The quaternion formed by axis [0, 0, 1] and + * angle -90 is the same as the quaternion formed by + * [0, 0, 1] and 270. This method favors the latter. + * @param {vec3} out_axis Vector receiving the axis of rotation + * @param {ReadonlyQuat} q Quaternion to be decomposed + * @return {Number} Angle, in radians, of the rotation + */ + +function getAxisAngle(out_axis, q) { + var rad = Math.acos(q[3]) * 2.0; + var s = Math.sin(rad / 2.0); + + if (s > EPSILON) { + out_axis[0] = q[0] / s; + out_axis[1] = q[1] / s; + out_axis[2] = q[2] / s; + } else { + // If s is zero, return any axis (no rotation - axis does not matter) + out_axis[0] = 1; + out_axis[1] = 0; + out_axis[2] = 0; + } + + return rad; +} +/** + * Gets the angular distance between two unit quaternions + * + * @param {ReadonlyQuat} a Origin unit quaternion + * @param {ReadonlyQuat} b Destination unit quaternion + * @return {Number} Angle, in radians, between the two quaternions + */ + +function getAngle(a, b) { + var dotproduct = dot$3(a, b); + return Math.acos(2 * dotproduct * dotproduct - 1); +} +/** + * Multiplies two quat's + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a the first operand + * @param {ReadonlyQuat} b the second operand + * @returns {quat} out + */ + +function multiply$2(out, a, b) { + var ax = a[0], + ay = a[1], + az = a[2], + aw = a[3]; + var bx = b[0], + by = b[1], + bz = b[2], + bw = b[3]; + out[0] = ax * bw + aw * bx + ay * bz - az * by; + out[1] = ay * bw + aw * by + az * bx - ax * bz; + out[2] = az * bw + aw * bz + ax * by - ay * bx; + out[3] = aw * bw - ax * bx - ay * by - az * bz; + return out; +} +/** + * Rotates a quaternion by the given angle about the X axis + * + * @param {quat} out quat receiving operation result + * @param {ReadonlyQuat} a quat to rotate + * @param {number} rad angle (in radians) to rotate + * @returns {quat} out + */ + +function rotateX$1(out, a, rad) { + rad *= 0.5; + var ax = a[0], + ay = a[1], + az = a[2], + aw = a[3]; + var bx = Math.sin(rad), + bw = Math.cos(rad); + out[0] = ax * bw + aw * bx; + out[1] = ay * bw + az * bx; + out[2] = az * bw - ay * bx; + out[3] = aw * bw - ax * bx; + return out; +} +/** + * Rotates a quaternion by the given angle about the Y axis + * + * @param {quat} out quat receiving operation result + * @param {ReadonlyQuat} a quat to rotate + * @param {number} rad angle (in radians) to rotate + * @returns {quat} out + */ + +function rotateY$1(out, a, rad) { + rad *= 0.5; + var ax = a[0], + ay = a[1], + az = a[2], + aw = a[3]; + var by = Math.sin(rad), + bw = Math.cos(rad); + out[0] = ax * bw - az * by; + out[1] = ay * bw + aw * by; + out[2] = az * bw + ax * by; + out[3] = aw * bw - ay * by; + return out; +} +/** + * Rotates a quaternion by the given angle about the Z axis + * + * @param {quat} out quat receiving operation result + * @param {ReadonlyQuat} a quat to rotate + * @param {number} rad angle (in radians) to rotate + * @returns {quat} out + */ + +function rotateZ$1(out, a, rad) { + rad *= 0.5; + var ax = a[0], + ay = a[1], + az = a[2], + aw = a[3]; + var bz = Math.sin(rad), + bw = Math.cos(rad); + out[0] = ax * bw + ay * bz; + out[1] = ay * bw - ax * bz; + out[2] = az * bw + aw * bz; + out[3] = aw * bw - az * bz; + return out; +} +/** + * Calculates the W component of a quat from the X, Y, and Z components. + * Assumes that quaternion is 1 unit in length. + * Any existing W component will be ignored. + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a quat to calculate W component of + * @returns {quat} out + */ + +function calculateW(out, a) { + var x = a[0], + y = a[1], + z = a[2]; + out[0] = x; + out[1] = y; + out[2] = z; + out[3] = Math.sqrt(Math.abs(1.0 - x * x - y * y - z * z)); + return out; +} +/** + * Calculate the exponential of a unit quaternion. + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a quat to calculate the exponential of + * @returns {quat} out + */ + +function exp(out, a) { + var x = a[0], + y = a[1], + z = a[2], + w = a[3]; + var r = Math.sqrt(x * x + y * y + z * z); + var et = Math.exp(w); + var s = r > 0 ? et * Math.sin(r) / r : 0; + out[0] = x * s; + out[1] = y * s; + out[2] = z * s; + out[3] = et * Math.cos(r); + return out; +} +/** + * Calculate the natural logarithm of a unit quaternion. + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a quat to calculate the exponential of + * @returns {quat} out + */ + +function ln(out, a) { + var x = a[0], + y = a[1], + z = a[2], + w = a[3]; + var r = Math.sqrt(x * x + y * y + z * z); + var t = r > 0 ? Math.atan2(r, w) / r : 0; + out[0] = x * t; + out[1] = y * t; + out[2] = z * t; + out[3] = 0.5 * Math.log(x * x + y * y + z * z + w * w); + return out; +} +/** + * Calculate the scalar power of a unit quaternion. + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a quat to calculate the exponential of + * @param {Number} b amount to scale the quaternion by + * @returns {quat} out + */ + +function pow(out, a, b) { + ln(out, a); + scale$2(out, out, b); + exp(out, out); + return out; +} +/** + * Performs a spherical linear interpolation between two quat + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a the first operand + * @param {ReadonlyQuat} b the second operand + * @param {Number} t interpolation amount, in the range [0-1], between the two inputs + * @returns {quat} out + */ + +function slerp(out, a, b, t) { + // benchmarks: + // http://jsperf.com/quaternion-slerp-implementations + var ax = a[0], + ay = a[1], + az = a[2], + aw = a[3]; + var bx = b[0], + by = b[1], + bz = b[2], + bw = b[3]; + var omega, cosom, sinom, scale0, scale1; // calc cosine + + cosom = ax * bx + ay * by + az * bz + aw * bw; // adjust signs (if necessary) + + if (cosom < 0.0) { + cosom = -cosom; + bx = -bx; + by = -by; + bz = -bz; + bw = -bw; + } // calculate coefficients + + + if (1.0 - cosom > EPSILON) { + // standard case (slerp) + omega = Math.acos(cosom); + sinom = Math.sin(omega); + scale0 = Math.sin((1.0 - t) * omega) / sinom; + scale1 = Math.sin(t * omega) / sinom; + } else { + // "from" and "to" quaternions are very close + // ... so we can do a linear interpolation + scale0 = 1.0 - t; + scale1 = t; + } // calculate final values + + + out[0] = scale0 * ax + scale1 * bx; + out[1] = scale0 * ay + scale1 * by; + out[2] = scale0 * az + scale1 * bz; + out[3] = scale0 * aw + scale1 * bw; + return out; +} +/** + * Generates a random unit quaternion + * + * @param {quat} out the receiving quaternion + * @returns {quat} out + */ + +function random$1(out) { + // Implementation of http://planning.cs.uiuc.edu/node198.html + // TODO: Calling random 3 times is probably not the fastest solution + var u1 = RANDOM(); + var u2 = RANDOM(); + var u3 = RANDOM(); + var sqrt1MinusU1 = Math.sqrt(1 - u1); + var sqrtU1 = Math.sqrt(u1); + out[0] = sqrt1MinusU1 * Math.sin(2.0 * Math.PI * u2); + out[1] = sqrt1MinusU1 * Math.cos(2.0 * Math.PI * u2); + out[2] = sqrtU1 * Math.sin(2.0 * Math.PI * u3); + out[3] = sqrtU1 * Math.cos(2.0 * Math.PI * u3); + return out; +} +/** + * Calculates the inverse of a quat + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a quat to calculate inverse of + * @returns {quat} out + */ + +function invert$1(out, a) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3]; + var dot = a0 * a0 + a1 * a1 + a2 * a2 + a3 * a3; + var invDot = dot ? 1.0 / dot : 0; // TODO: Would be faster to return [0,0,0,0] immediately if dot == 0 + + out[0] = -a0 * invDot; + out[1] = -a1 * invDot; + out[2] = -a2 * invDot; + out[3] = a3 * invDot; + return out; +} +/** + * Calculates the conjugate of a quat + * If the quaternion is normalized, this function is faster than quat.inverse and produces the same result. + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a quat to calculate conjugate of + * @returns {quat} out + */ + +function conjugate$1(out, a) { + out[0] = -a[0]; + out[1] = -a[1]; + out[2] = -a[2]; + out[3] = a[3]; + return out; +} +/** + * Creates a quaternion from the given 3x3 rotation matrix. + * + * NOTE: The resultant quaternion is not normalized, so you should be sure + * to renormalize the quaternion yourself where necessary. + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyMat3} m rotation matrix + * @returns {quat} out + * @function + */ + +function fromMat3(out, m) { + // Algorithm in Ken Shoemake's article in 1987 SIGGRAPH course notes + // article "Quaternion Calculus and Fast Animation". + var fTrace = m[0] + m[4] + m[8]; + var fRoot; + + if (fTrace > 0.0) { + // |w| > 1/2, may as well choose w > 1/2 + fRoot = Math.sqrt(fTrace + 1.0); // 2w + + out[3] = 0.5 * fRoot; + fRoot = 0.5 / fRoot; // 1/(4w) + + out[0] = (m[5] - m[7]) * fRoot; + out[1] = (m[6] - m[2]) * fRoot; + out[2] = (m[1] - m[3]) * fRoot; + } else { + // |w| <= 1/2 + var i = 0; + if (m[4] > m[0]) i = 1; + if (m[8] > m[i * 3 + i]) i = 2; + var j = (i + 1) % 3; + var k = (i + 2) % 3; + fRoot = Math.sqrt(m[i * 3 + i] - m[j * 3 + j] - m[k * 3 + k] + 1.0); + out[i] = 0.5 * fRoot; + fRoot = 0.5 / fRoot; + out[3] = (m[j * 3 + k] - m[k * 3 + j]) * fRoot; + out[j] = (m[j * 3 + i] + m[i * 3 + j]) * fRoot; + out[k] = (m[k * 3 + i] + m[i * 3 + k]) * fRoot; + } + + return out; +} +/** + * Creates a quaternion from the given euler angle x, y, z. + * + * @param {quat} out the receiving quaternion + * @param {x} Angle to rotate around X axis in degrees. + * @param {y} Angle to rotate around Y axis in degrees. + * @param {z} Angle to rotate around Z axis in degrees. + * @returns {quat} out + * @function + */ + +function fromEuler(out, x, y, z) { + var halfToRad = 0.5 * Math.PI / 180.0; + x *= halfToRad; + y *= halfToRad; + z *= halfToRad; + var sx = Math.sin(x); + var cx = Math.cos(x); + var sy = Math.sin(y); + var cy = Math.cos(y); + var sz = Math.sin(z); + var cz = Math.cos(z); + out[0] = sx * cy * cz - cx * sy * sz; + out[1] = cx * sy * cz + sx * cy * sz; + out[2] = cx * cy * sz - sx * sy * cz; + out[3] = cx * cy * cz + sx * sy * sz; + return out; +} +/** + * Returns a string representation of a quatenion + * + * @param {ReadonlyQuat} a vector to represent as a string + * @returns {String} string representation of the vector + */ + +function str$2(a) { + return "quat(" + a[0] + ", " + a[1] + ", " + a[2] + ", " + a[3] + ")"; +} +/** + * Creates a new quat initialized with values from an existing quaternion + * + * @param {ReadonlyQuat} a quaternion to clone + * @returns {quat} a new quaternion + * @function + */ + +var clone$2 = clone$3; +/** + * Creates a new quat initialized with the given values + * + * @param {Number} x X component + * @param {Number} y Y component + * @param {Number} z Z component + * @param {Number} w W component + * @returns {quat} a new quaternion + * @function + */ + +var fromValues$2 = fromValues$3; +/** + * Copy the values from one quat to another + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a the source quaternion + * @returns {quat} out + * @function + */ + +var copy$2 = copy$3; +/** + * Set the components of a quat to the given values + * + * @param {quat} out the receiving quaternion + * @param {Number} x X component + * @param {Number} y Y component + * @param {Number} z Z component + * @param {Number} w W component + * @returns {quat} out + * @function + */ + +var set$2 = set$3; +/** + * Adds two quat's + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a the first operand + * @param {ReadonlyQuat} b the second operand + * @returns {quat} out + * @function + */ + +var add$2 = add$3; +/** + * Alias for {@link quat.multiply} + * @function + */ + +var mul$2 = multiply$2; +/** + * Scales a quat by a scalar number + * + * @param {quat} out the receiving vector + * @param {ReadonlyQuat} a the vector to scale + * @param {Number} b amount to scale the vector by + * @returns {quat} out + * @function + */ + +var scale$2 = scale$3; +/** + * Calculates the dot product of two quat's + * + * @param {ReadonlyQuat} a the first operand + * @param {ReadonlyQuat} b the second operand + * @returns {Number} dot product of a and b + * @function + */ + +var dot$3 = dot$4; +/** + * Performs a linear interpolation between two quat's + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a the first operand + * @param {ReadonlyQuat} b the second operand + * @param {Number} t interpolation amount, in the range [0-1], between the two inputs + * @returns {quat} out + * @function + */ + +var lerp$2 = lerp$3; +/** + * Calculates the length of a quat + * + * @param {ReadonlyQuat} a vector to calculate length of + * @returns {Number} length of a + */ + +var length$2 = length$3; +/** + * Alias for {@link quat.length} + * @function + */ + +var len$2 = length$2; +/** + * Calculates the squared length of a quat + * + * @param {ReadonlyQuat} a vector to calculate squared length of + * @returns {Number} squared length of a + * @function + */ + +var squaredLength$2 = squaredLength$3; +/** + * Alias for {@link quat.squaredLength} + * @function + */ + +var sqrLen$2 = squaredLength$2; +/** + * Normalize a quat + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a quaternion to normalize + * @returns {quat} out + * @function + */ + +var normalize$2 = normalize$3; +/** + * Returns whether or not the quaternions have exactly the same elements in the same position (when compared with ===) + * + * @param {ReadonlyQuat} a The first quaternion. + * @param {ReadonlyQuat} b The second quaternion. + * @returns {Boolean} True if the vectors are equal, false otherwise. + */ + +var exactEquals$2 = exactEquals$3; +/** + * Returns whether or not the quaternions have approximately the same elements in the same position. + * + * @param {ReadonlyQuat} a The first vector. + * @param {ReadonlyQuat} b The second vector. + * @returns {Boolean} True if the vectors are equal, false otherwise. + */ + +var equals$3 = equals$4; +/** + * Sets a quaternion to represent the shortest rotation from one + * vector to another. + * + * Both vectors are assumed to be unit length. + * + * @param {quat} out the receiving quaternion. + * @param {ReadonlyVec3} a the initial vector + * @param {ReadonlyVec3} b the destination vector + * @returns {quat} out + */ + +var rotationTo = function () { + var tmpvec3 = create$4(); + var xUnitVec3 = fromValues$4(1, 0, 0); + var yUnitVec3 = fromValues$4(0, 1, 0); + return function (out, a, b) { + var dot = dot$5(a, b); + + if (dot < -0.999999) { + cross$2(tmpvec3, xUnitVec3, a); + if (len$4(tmpvec3) < 0.000001) cross$2(tmpvec3, yUnitVec3, a); + normalize$4(tmpvec3, tmpvec3); + setAxisAngle(out, tmpvec3, Math.PI); + return out; + } else if (dot > 0.999999) { + out[0] = 0; + out[1] = 0; + out[2] = 0; + out[3] = 1; + return out; + } else { + cross$2(tmpvec3, a, b); + out[0] = tmpvec3[0]; + out[1] = tmpvec3[1]; + out[2] = tmpvec3[2]; + out[3] = 1 + dot; + return normalize$2(out, out); + } + }; +}(); +/** + * Performs a spherical linear interpolation with two control points + * + * @param {quat} out the receiving quaternion + * @param {ReadonlyQuat} a the first operand + * @param {ReadonlyQuat} b the second operand + * @param {ReadonlyQuat} c the third operand + * @param {ReadonlyQuat} d the fourth operand + * @param {Number} t interpolation amount, in the range [0-1], between the two inputs + * @returns {quat} out + */ + +var sqlerp = function () { + var temp1 = create$2(); + var temp2 = create$2(); + return function (out, a, b, c, d, t) { + slerp(temp1, a, d, t); + slerp(temp2, b, c, t); + slerp(out, temp1, temp2, 2 * t * (1 - t)); + return out; + }; +}(); +/** + * Sets the specified quaternion with values corresponding to the given + * axes. Each axis is a vec3 and is expected to be unit length and + * perpendicular to all other specified axes. + * + * @param {ReadonlyVec3} view the vector representing the viewing direction + * @param {ReadonlyVec3} right the vector representing the local "right" direction + * @param {ReadonlyVec3} up the vector representing the local "up" direction + * @returns {quat} out + */ + +var setAxes = function () { + var matr = create$6(); + return function (out, view, right, up) { + matr[0] = right[0]; + matr[3] = right[1]; + matr[6] = right[2]; + matr[1] = up[0]; + matr[4] = up[1]; + matr[7] = up[2]; + matr[2] = -view[0]; + matr[5] = -view[1]; + matr[8] = -view[2]; + return normalize$2(out, fromMat3(out, matr)); + }; +}(); + +var quat = /*#__PURE__*/Object.freeze({ +__proto__: null, +add: add$2, +calculateW: calculateW, +clone: clone$2, +conjugate: conjugate$1, +copy: copy$2, +create: create$2, +dot: dot$3, +equals: equals$3, +exactEquals: exactEquals$2, +exp: exp, +fromEuler: fromEuler, +fromMat3: fromMat3, +fromValues: fromValues$2, +getAngle: getAngle, +getAxisAngle: getAxisAngle, +identity: identity$1, +invert: invert$1, +len: len$2, +length: length$2, +lerp: lerp$2, +ln: ln, +mul: mul$2, +multiply: multiply$2, +normalize: normalize$2, +pow: pow, +random: random$1, +rotateX: rotateX$1, +rotateY: rotateY$1, +rotateZ: rotateZ$1, +rotationTo: rotationTo, +scale: scale$2, +set: set$2, +setAxes: setAxes, +setAxisAngle: setAxisAngle, +slerp: slerp, +sqlerp: sqlerp, +sqrLen: sqrLen$2, +squaredLength: squaredLength$2, +str: str$2 +}); + +/** + * Dual Quaternion
+ * Format: [real, dual]
+ * Quaternion format: XYZW
+ * Make sure to have normalized dual quaternions, otherwise the functions may not work as intended.
+ * @module quat2 + */ + +/** + * Creates a new identity dual quat + * + * @returns {quat2} a new dual quaternion [real -> rotation, dual -> translation] + */ + +function create$1() { + var dq = new ARRAY_TYPE(8); + + if (ARRAY_TYPE != Float32Array) { + dq[0] = 0; + dq[1] = 0; + dq[2] = 0; + dq[4] = 0; + dq[5] = 0; + dq[6] = 0; + dq[7] = 0; + } + + dq[3] = 1; + return dq; +} +/** + * Creates a new quat initialized with values from an existing quaternion + * + * @param {ReadonlyQuat2} a dual quaternion to clone + * @returns {quat2} new dual quaternion + * @function + */ + +function clone$1(a) { + var dq = new ARRAY_TYPE(8); + dq[0] = a[0]; + dq[1] = a[1]; + dq[2] = a[2]; + dq[3] = a[3]; + dq[4] = a[4]; + dq[5] = a[5]; + dq[6] = a[6]; + dq[7] = a[7]; + return dq; +} +/** + * Creates a new dual quat initialized with the given values + * + * @param {Number} x1 X component + * @param {Number} y1 Y component + * @param {Number} z1 Z component + * @param {Number} w1 W component + * @param {Number} x2 X component + * @param {Number} y2 Y component + * @param {Number} z2 Z component + * @param {Number} w2 W component + * @returns {quat2} new dual quaternion + * @function + */ + +function fromValues$1(x1, y1, z1, w1, x2, y2, z2, w2) { + var dq = new ARRAY_TYPE(8); + dq[0] = x1; + dq[1] = y1; + dq[2] = z1; + dq[3] = w1; + dq[4] = x2; + dq[5] = y2; + dq[6] = z2; + dq[7] = w2; + return dq; +} +/** + * Creates a new dual quat from the given values (quat and translation) + * + * @param {Number} x1 X component + * @param {Number} y1 Y component + * @param {Number} z1 Z component + * @param {Number} w1 W component + * @param {Number} x2 X component (translation) + * @param {Number} y2 Y component (translation) + * @param {Number} z2 Z component (translation) + * @returns {quat2} new dual quaternion + * @function + */ + +function fromRotationTranslationValues(x1, y1, z1, w1, x2, y2, z2) { + var dq = new ARRAY_TYPE(8); + dq[0] = x1; + dq[1] = y1; + dq[2] = z1; + dq[3] = w1; + var ax = x2 * 0.5, + ay = y2 * 0.5, + az = z2 * 0.5; + dq[4] = ax * w1 + ay * z1 - az * y1; + dq[5] = ay * w1 + az * x1 - ax * z1; + dq[6] = az * w1 + ax * y1 - ay * x1; + dq[7] = -ax * x1 - ay * y1 - az * z1; + return dq; +} +/** + * Creates a dual quat from a quaternion and a translation + * + * @param {ReadonlyQuat2} dual quaternion receiving operation result + * @param {ReadonlyQuat} q a normalized quaternion + * @param {ReadonlyVec3} t tranlation vector + * @returns {quat2} dual quaternion receiving operation result + * @function + */ + +function fromRotationTranslation(out, q, t) { + var ax = t[0] * 0.5, + ay = t[1] * 0.5, + az = t[2] * 0.5, + bx = q[0], + by = q[1], + bz = q[2], + bw = q[3]; + out[0] = bx; + out[1] = by; + out[2] = bz; + out[3] = bw; + out[4] = ax * bw + ay * bz - az * by; + out[5] = ay * bw + az * bx - ax * bz; + out[6] = az * bw + ax * by - ay * bx; + out[7] = -ax * bx - ay * by - az * bz; + return out; +} +/** + * Creates a dual quat from a translation + * + * @param {ReadonlyQuat2} dual quaternion receiving operation result + * @param {ReadonlyVec3} t translation vector + * @returns {quat2} dual quaternion receiving operation result + * @function + */ + +function fromTranslation(out, t) { + out[0] = 0; + out[1] = 0; + out[2] = 0; + out[3] = 1; + out[4] = t[0] * 0.5; + out[5] = t[1] * 0.5; + out[6] = t[2] * 0.5; + out[7] = 0; + return out; +} +/** + * Creates a dual quat from a quaternion + * + * @param {ReadonlyQuat2} dual quaternion receiving operation result + * @param {ReadonlyQuat} q the quaternion + * @returns {quat2} dual quaternion receiving operation result + * @function + */ + +function fromRotation(out, q) { + out[0] = q[0]; + out[1] = q[1]; + out[2] = q[2]; + out[3] = q[3]; + out[4] = 0; + out[5] = 0; + out[6] = 0; + out[7] = 0; + return out; +} +/** + * Creates a new dual quat from a matrix (4x4) + * + * @param {quat2} out the dual quaternion + * @param {ReadonlyMat4} a the matrix + * @returns {quat2} dual quat receiving operation result + * @function + */ + +function fromMat4(out, a) { + //TODO Optimize this + var outer = create$2(); + getRotation(outer, a); + var t = new ARRAY_TYPE(3); + getTranslation$1(t, a); + fromRotationTranslation(out, outer, t); + return out; +} +/** + * Copy the values from one dual quat to another + * + * @param {quat2} out the receiving dual quaternion + * @param {ReadonlyQuat2} a the source dual quaternion + * @returns {quat2} out + * @function + */ + +function copy$1(out, a) { + out[0] = a[0]; + out[1] = a[1]; + out[2] = a[2]; + out[3] = a[3]; + out[4] = a[4]; + out[5] = a[5]; + out[6] = a[6]; + out[7] = a[7]; + return out; +} +/** + * Set a dual quat to the identity dual quaternion + * + * @param {quat2} out the receiving quaternion + * @returns {quat2} out + */ + +function identity(out) { + out[0] = 0; + out[1] = 0; + out[2] = 0; + out[3] = 1; + out[4] = 0; + out[5] = 0; + out[6] = 0; + out[7] = 0; + return out; +} +/** + * Set the components of a dual quat to the given values + * + * @param {quat2} out the receiving quaternion + * @param {Number} x1 X component + * @param {Number} y1 Y component + * @param {Number} z1 Z component + * @param {Number} w1 W component + * @param {Number} x2 X component + * @param {Number} y2 Y component + * @param {Number} z2 Z component + * @param {Number} w2 W component + * @returns {quat2} out + * @function + */ + +function set$1(out, x1, y1, z1, w1, x2, y2, z2, w2) { + out[0] = x1; + out[1] = y1; + out[2] = z1; + out[3] = w1; + out[4] = x2; + out[5] = y2; + out[6] = z2; + out[7] = w2; + return out; +} +/** + * Gets the real part of a dual quat + * @param {quat} out real part + * @param {ReadonlyQuat2} a Dual Quaternion + * @return {quat} real part + */ + +var getReal = copy$2; +/** + * Gets the dual part of a dual quat + * @param {quat} out dual part + * @param {ReadonlyQuat2} a Dual Quaternion + * @return {quat} dual part + */ + +function getDual(out, a) { + out[0] = a[4]; + out[1] = a[5]; + out[2] = a[6]; + out[3] = a[7]; + return out; +} +/** + * Set the real component of a dual quat to the given quaternion + * + * @param {quat2} out the receiving quaternion + * @param {ReadonlyQuat} q a quaternion representing the real part + * @returns {quat2} out + * @function + */ + +var setReal = copy$2; +/** + * Set the dual component of a dual quat to the given quaternion + * + * @param {quat2} out the receiving quaternion + * @param {ReadonlyQuat} q a quaternion representing the dual part + * @returns {quat2} out + * @function + */ + +function setDual(out, q) { + out[4] = q[0]; + out[5] = q[1]; + out[6] = q[2]; + out[7] = q[3]; + return out; +} +/** + * Gets the translation of a normalized dual quat + * @param {vec3} out translation + * @param {ReadonlyQuat2} a Dual Quaternion to be decomposed + * @return {vec3} translation + */ + +function getTranslation(out, a) { + var ax = a[4], + ay = a[5], + az = a[6], + aw = a[7], + bx = -a[0], + by = -a[1], + bz = -a[2], + bw = a[3]; + out[0] = (ax * bw + aw * bx + ay * bz - az * by) * 2; + out[1] = (ay * bw + aw * by + az * bx - ax * bz) * 2; + out[2] = (az * bw + aw * bz + ax * by - ay * bx) * 2; + return out; +} +/** + * Translates a dual quat by the given vector + * + * @param {quat2} out the receiving dual quaternion + * @param {ReadonlyQuat2} a the dual quaternion to translate + * @param {ReadonlyVec3} v vector to translate by + * @returns {quat2} out + */ + +function translate(out, a, v) { + var ax1 = a[0], + ay1 = a[1], + az1 = a[2], + aw1 = a[3], + bx1 = v[0] * 0.5, + by1 = v[1] * 0.5, + bz1 = v[2] * 0.5, + ax2 = a[4], + ay2 = a[5], + az2 = a[6], + aw2 = a[7]; + out[0] = ax1; + out[1] = ay1; + out[2] = az1; + out[3] = aw1; + out[4] = aw1 * bx1 + ay1 * bz1 - az1 * by1 + ax2; + out[5] = aw1 * by1 + az1 * bx1 - ax1 * bz1 + ay2; + out[6] = aw1 * bz1 + ax1 * by1 - ay1 * bx1 + az2; + out[7] = -ax1 * bx1 - ay1 * by1 - az1 * bz1 + aw2; + return out; +} +/** + * Rotates a dual quat around the X axis + * + * @param {quat2} out the receiving dual quaternion + * @param {ReadonlyQuat2} a the dual quaternion to rotate + * @param {number} rad how far should the rotation be + * @returns {quat2} out + */ + +function rotateX(out, a, rad) { + var bx = -a[0], + by = -a[1], + bz = -a[2], + bw = a[3], + ax = a[4], + ay = a[5], + az = a[6], + aw = a[7], + ax1 = ax * bw + aw * bx + ay * bz - az * by, + ay1 = ay * bw + aw * by + az * bx - ax * bz, + az1 = az * bw + aw * bz + ax * by - ay * bx, + aw1 = aw * bw - ax * bx - ay * by - az * bz; + rotateX$1(out, a, rad); + bx = out[0]; + by = out[1]; + bz = out[2]; + bw = out[3]; + out[4] = ax1 * bw + aw1 * bx + ay1 * bz - az1 * by; + out[5] = ay1 * bw + aw1 * by + az1 * bx - ax1 * bz; + out[6] = az1 * bw + aw1 * bz + ax1 * by - ay1 * bx; + out[7] = aw1 * bw - ax1 * bx - ay1 * by - az1 * bz; + return out; +} +/** + * Rotates a dual quat around the Y axis + * + * @param {quat2} out the receiving dual quaternion + * @param {ReadonlyQuat2} a the dual quaternion to rotate + * @param {number} rad how far should the rotation be + * @returns {quat2} out + */ + +function rotateY(out, a, rad) { + var bx = -a[0], + by = -a[1], + bz = -a[2], + bw = a[3], + ax = a[4], + ay = a[5], + az = a[6], + aw = a[7], + ax1 = ax * bw + aw * bx + ay * bz - az * by, + ay1 = ay * bw + aw * by + az * bx - ax * bz, + az1 = az * bw + aw * bz + ax * by - ay * bx, + aw1 = aw * bw - ax * bx - ay * by - az * bz; + rotateY$1(out, a, rad); + bx = out[0]; + by = out[1]; + bz = out[2]; + bw = out[3]; + out[4] = ax1 * bw + aw1 * bx + ay1 * bz - az1 * by; + out[5] = ay1 * bw + aw1 * by + az1 * bx - ax1 * bz; + out[6] = az1 * bw + aw1 * bz + ax1 * by - ay1 * bx; + out[7] = aw1 * bw - ax1 * bx - ay1 * by - az1 * bz; + return out; +} +/** + * Rotates a dual quat around the Z axis + * + * @param {quat2} out the receiving dual quaternion + * @param {ReadonlyQuat2} a the dual quaternion to rotate + * @param {number} rad how far should the rotation be + * @returns {quat2} out + */ + +function rotateZ(out, a, rad) { + var bx = -a[0], + by = -a[1], + bz = -a[2], + bw = a[3], + ax = a[4], + ay = a[5], + az = a[6], + aw = a[7], + ax1 = ax * bw + aw * bx + ay * bz - az * by, + ay1 = ay * bw + aw * by + az * bx - ax * bz, + az1 = az * bw + aw * bz + ax * by - ay * bx, + aw1 = aw * bw - ax * bx - ay * by - az * bz; + rotateZ$1(out, a, rad); + bx = out[0]; + by = out[1]; + bz = out[2]; + bw = out[3]; + out[4] = ax1 * bw + aw1 * bx + ay1 * bz - az1 * by; + out[5] = ay1 * bw + aw1 * by + az1 * bx - ax1 * bz; + out[6] = az1 * bw + aw1 * bz + ax1 * by - ay1 * bx; + out[7] = aw1 * bw - ax1 * bx - ay1 * by - az1 * bz; + return out; +} +/** + * Rotates a dual quat by a given quaternion (a * q) + * + * @param {quat2} out the receiving dual quaternion + * @param {ReadonlyQuat2} a the dual quaternion to rotate + * @param {ReadonlyQuat} q quaternion to rotate by + * @returns {quat2} out + */ + +function rotateByQuatAppend(out, a, q) { + var qx = q[0], + qy = q[1], + qz = q[2], + qw = q[3], + ax = a[0], + ay = a[1], + az = a[2], + aw = a[3]; + out[0] = ax * qw + aw * qx + ay * qz - az * qy; + out[1] = ay * qw + aw * qy + az * qx - ax * qz; + out[2] = az * qw + aw * qz + ax * qy - ay * qx; + out[3] = aw * qw - ax * qx - ay * qy - az * qz; + ax = a[4]; + ay = a[5]; + az = a[6]; + aw = a[7]; + out[4] = ax * qw + aw * qx + ay * qz - az * qy; + out[5] = ay * qw + aw * qy + az * qx - ax * qz; + out[6] = az * qw + aw * qz + ax * qy - ay * qx; + out[7] = aw * qw - ax * qx - ay * qy - az * qz; + return out; +} +/** + * Rotates a dual quat by a given quaternion (q * a) + * + * @param {quat2} out the receiving dual quaternion + * @param {ReadonlyQuat} q quaternion to rotate by + * @param {ReadonlyQuat2} a the dual quaternion to rotate + * @returns {quat2} out + */ + +function rotateByQuatPrepend(out, q, a) { + var qx = q[0], + qy = q[1], + qz = q[2], + qw = q[3], + bx = a[0], + by = a[1], + bz = a[2], + bw = a[3]; + out[0] = qx * bw + qw * bx + qy * bz - qz * by; + out[1] = qy * bw + qw * by + qz * bx - qx * bz; + out[2] = qz * bw + qw * bz + qx * by - qy * bx; + out[3] = qw * bw - qx * bx - qy * by - qz * bz; + bx = a[4]; + by = a[5]; + bz = a[6]; + bw = a[7]; + out[4] = qx * bw + qw * bx + qy * bz - qz * by; + out[5] = qy * bw + qw * by + qz * bx - qx * bz; + out[6] = qz * bw + qw * bz + qx * by - qy * bx; + out[7] = qw * bw - qx * bx - qy * by - qz * bz; + return out; +} +/** + * Rotates a dual quat around a given axis. Does the normalisation automatically + * + * @param {quat2} out the receiving dual quaternion + * @param {ReadonlyQuat2} a the dual quaternion to rotate + * @param {ReadonlyVec3} axis the axis to rotate around + * @param {Number} rad how far the rotation should be + * @returns {quat2} out + */ + +function rotateAroundAxis(out, a, axis, rad) { + //Special case for rad = 0 + if (Math.abs(rad) < EPSILON) { + return copy$1(out, a); + } + + var axisLength = Math.hypot(axis[0], axis[1], axis[2]); + rad = rad * 0.5; + var s = Math.sin(rad); + var bx = s * axis[0] / axisLength; + var by = s * axis[1] / axisLength; + var bz = s * axis[2] / axisLength; + var bw = Math.cos(rad); + var ax1 = a[0], + ay1 = a[1], + az1 = a[2], + aw1 = a[3]; + out[0] = ax1 * bw + aw1 * bx + ay1 * bz - az1 * by; + out[1] = ay1 * bw + aw1 * by + az1 * bx - ax1 * bz; + out[2] = az1 * bw + aw1 * bz + ax1 * by - ay1 * bx; + out[3] = aw1 * bw - ax1 * bx - ay1 * by - az1 * bz; + var ax = a[4], + ay = a[5], + az = a[6], + aw = a[7]; + out[4] = ax * bw + aw * bx + ay * bz - az * by; + out[5] = ay * bw + aw * by + az * bx - ax * bz; + out[6] = az * bw + aw * bz + ax * by - ay * bx; + out[7] = aw * bw - ax * bx - ay * by - az * bz; + return out; +} +/** + * Adds two dual quat's + * + * @param {quat2} out the receiving dual quaternion + * @param {ReadonlyQuat2} a the first operand + * @param {ReadonlyQuat2} b the second operand + * @returns {quat2} out + * @function + */ + +function add$1(out, a, b) { + out[0] = a[0] + b[0]; + out[1] = a[1] + b[1]; + out[2] = a[2] + b[2]; + out[3] = a[3] + b[3]; + out[4] = a[4] + b[4]; + out[5] = a[5] + b[5]; + out[6] = a[6] + b[6]; + out[7] = a[7] + b[7]; + return out; +} +/** + * Multiplies two dual quat's + * + * @param {quat2} out the receiving dual quaternion + * @param {ReadonlyQuat2} a the first operand + * @param {ReadonlyQuat2} b the second operand + * @returns {quat2} out + */ + +function multiply$1(out, a, b) { + var ax0 = a[0], + ay0 = a[1], + az0 = a[2], + aw0 = a[3], + bx1 = b[4], + by1 = b[5], + bz1 = b[6], + bw1 = b[7], + ax1 = a[4], + ay1 = a[5], + az1 = a[6], + aw1 = a[7], + bx0 = b[0], + by0 = b[1], + bz0 = b[2], + bw0 = b[3]; + out[0] = ax0 * bw0 + aw0 * bx0 + ay0 * bz0 - az0 * by0; + out[1] = ay0 * bw0 + aw0 * by0 + az0 * bx0 - ax0 * bz0; + out[2] = az0 * bw0 + aw0 * bz0 + ax0 * by0 - ay0 * bx0; + out[3] = aw0 * bw0 - ax0 * bx0 - ay0 * by0 - az0 * bz0; + out[4] = ax0 * bw1 + aw0 * bx1 + ay0 * bz1 - az0 * by1 + ax1 * bw0 + aw1 * bx0 + ay1 * bz0 - az1 * by0; + out[5] = ay0 * bw1 + aw0 * by1 + az0 * bx1 - ax0 * bz1 + ay1 * bw0 + aw1 * by0 + az1 * bx0 - ax1 * bz0; + out[6] = az0 * bw1 + aw0 * bz1 + ax0 * by1 - ay0 * bx1 + az1 * bw0 + aw1 * bz0 + ax1 * by0 - ay1 * bx0; + out[7] = aw0 * bw1 - ax0 * bx1 - ay0 * by1 - az0 * bz1 + aw1 * bw0 - ax1 * bx0 - ay1 * by0 - az1 * bz0; + return out; +} +/** + * Alias for {@link quat2.multiply} + * @function + */ + +var mul$1 = multiply$1; +/** + * Scales a dual quat by a scalar number + * + * @param {quat2} out the receiving dual quat + * @param {ReadonlyQuat2} a the dual quat to scale + * @param {Number} b amount to scale the dual quat by + * @returns {quat2} out + * @function + */ + +function scale$1(out, a, b) { + out[0] = a[0] * b; + out[1] = a[1] * b; + out[2] = a[2] * b; + out[3] = a[3] * b; + out[4] = a[4] * b; + out[5] = a[5] * b; + out[6] = a[6] * b; + out[7] = a[7] * b; + return out; +} +/** + * Calculates the dot product of two dual quat's (The dot product of the real parts) + * + * @param {ReadonlyQuat2} a the first operand + * @param {ReadonlyQuat2} b the second operand + * @returns {Number} dot product of a and b + * @function + */ + +var dot$2 = dot$3; +/** + * Performs a linear interpolation between two dual quats's + * NOTE: The resulting dual quaternions won't always be normalized (The error is most noticeable when t = 0.5) + * + * @param {quat2} out the receiving dual quat + * @param {ReadonlyQuat2} a the first operand + * @param {ReadonlyQuat2} b the second operand + * @param {Number} t interpolation amount, in the range [0-1], between the two inputs + * @returns {quat2} out + */ + +function lerp$1(out, a, b, t) { + var mt = 1 - t; + if (dot$2(a, b) < 0) t = -t; + out[0] = a[0] * mt + b[0] * t; + out[1] = a[1] * mt + b[1] * t; + out[2] = a[2] * mt + b[2] * t; + out[3] = a[3] * mt + b[3] * t; + out[4] = a[4] * mt + b[4] * t; + out[5] = a[5] * mt + b[5] * t; + out[6] = a[6] * mt + b[6] * t; + out[7] = a[7] * mt + b[7] * t; + return out; +} +/** + * Calculates the inverse of a dual quat. If they are normalized, conjugate is cheaper + * + * @param {quat2} out the receiving dual quaternion + * @param {ReadonlyQuat2} a dual quat to calculate inverse of + * @returns {quat2} out + */ + +function invert(out, a) { + var sqlen = squaredLength$1(a); + out[0] = -a[0] / sqlen; + out[1] = -a[1] / sqlen; + out[2] = -a[2] / sqlen; + out[3] = a[3] / sqlen; + out[4] = -a[4] / sqlen; + out[5] = -a[5] / sqlen; + out[6] = -a[6] / sqlen; + out[7] = a[7] / sqlen; + return out; +} +/** + * Calculates the conjugate of a dual quat + * If the dual quaternion is normalized, this function is faster than quat2.inverse and produces the same result. + * + * @param {quat2} out the receiving quaternion + * @param {ReadonlyQuat2} a quat to calculate conjugate of + * @returns {quat2} out + */ + +function conjugate(out, a) { + out[0] = -a[0]; + out[1] = -a[1]; + out[2] = -a[2]; + out[3] = a[3]; + out[4] = -a[4]; + out[5] = -a[5]; + out[6] = -a[6]; + out[7] = a[7]; + return out; +} +/** + * Calculates the length of a dual quat + * + * @param {ReadonlyQuat2} a dual quat to calculate length of + * @returns {Number} length of a + * @function + */ + +var length$1 = length$2; +/** + * Alias for {@link quat2.length} + * @function + */ + +var len$1 = length$1; +/** + * Calculates the squared length of a dual quat + * + * @param {ReadonlyQuat2} a dual quat to calculate squared length of + * @returns {Number} squared length of a + * @function + */ + +var squaredLength$1 = squaredLength$2; +/** + * Alias for {@link quat2.squaredLength} + * @function + */ + +var sqrLen$1 = squaredLength$1; +/** + * Normalize a dual quat + * + * @param {quat2} out the receiving dual quaternion + * @param {ReadonlyQuat2} a dual quaternion to normalize + * @returns {quat2} out + * @function + */ + +function normalize$1(out, a) { + var magnitude = squaredLength$1(a); + + if (magnitude > 0) { + magnitude = Math.sqrt(magnitude); + var a0 = a[0] / magnitude; + var a1 = a[1] / magnitude; + var a2 = a[2] / magnitude; + var a3 = a[3] / magnitude; + var b0 = a[4]; + var b1 = a[5]; + var b2 = a[6]; + var b3 = a[7]; + var a_dot_b = a0 * b0 + a1 * b1 + a2 * b2 + a3 * b3; + out[0] = a0; + out[1] = a1; + out[2] = a2; + out[3] = a3; + out[4] = (b0 - a0 * a_dot_b) / magnitude; + out[5] = (b1 - a1 * a_dot_b) / magnitude; + out[6] = (b2 - a2 * a_dot_b) / magnitude; + out[7] = (b3 - a3 * a_dot_b) / magnitude; + } + + return out; +} +/** + * Returns a string representation of a dual quatenion + * + * @param {ReadonlyQuat2} a dual quaternion to represent as a string + * @returns {String} string representation of the dual quat + */ + +function str$1(a) { + return "quat2(" + a[0] + ", " + a[1] + ", " + a[2] + ", " + a[3] + ", " + a[4] + ", " + a[5] + ", " + a[6] + ", " + a[7] + ")"; +} +/** + * Returns whether or not the dual quaternions have exactly the same elements in the same position (when compared with ===) + * + * @param {ReadonlyQuat2} a the first dual quaternion. + * @param {ReadonlyQuat2} b the second dual quaternion. + * @returns {Boolean} true if the dual quaternions are equal, false otherwise. + */ + +function exactEquals$1(a, b) { + return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5] && a[6] === b[6] && a[7] === b[7]; +} +/** + * Returns whether or not the dual quaternions have approximately the same elements in the same position. + * + * @param {ReadonlyQuat2} a the first dual quat. + * @param {ReadonlyQuat2} b the second dual quat. + * @returns {Boolean} true if the dual quats are equal, false otherwise. + */ + +function equals$2(a, b) { + var a0 = a[0], + a1 = a[1], + a2 = a[2], + a3 = a[3], + a4 = a[4], + a5 = a[5], + a6 = a[6], + a7 = a[7]; + var b0 = b[0], + b1 = b[1], + b2 = b[2], + b3 = b[3], + b4 = b[4], + b5 = b[5], + b6 = b[6], + b7 = b[7]; + return Math.abs(a0 - b0) <= EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5)) && Math.abs(a6 - b6) <= EPSILON * Math.max(1.0, Math.abs(a6), Math.abs(b6)) && Math.abs(a7 - b7) <= EPSILON * Math.max(1.0, Math.abs(a7), Math.abs(b7)); +} + +var quat2 = /*#__PURE__*/Object.freeze({ +__proto__: null, +add: add$1, +clone: clone$1, +conjugate: conjugate, +copy: copy$1, +create: create$1, +dot: dot$2, +equals: equals$2, +exactEquals: exactEquals$1, +fromMat4: fromMat4, +fromRotation: fromRotation, +fromRotationTranslation: fromRotationTranslation, +fromRotationTranslationValues: fromRotationTranslationValues, +fromTranslation: fromTranslation, +fromValues: fromValues$1, +getDual: getDual, +getReal: getReal, +getTranslation: getTranslation, +identity: identity, +invert: invert, +len: len$1, +length: length$1, +lerp: lerp$1, +mul: mul$1, +multiply: multiply$1, +normalize: normalize$1, +rotateAroundAxis: rotateAroundAxis, +rotateByQuatAppend: rotateByQuatAppend, +rotateByQuatPrepend: rotateByQuatPrepend, +rotateX: rotateX, +rotateY: rotateY, +rotateZ: rotateZ, +scale: scale$1, +set: set$1, +setDual: setDual, +setReal: setReal, +sqrLen: sqrLen$1, +squaredLength: squaredLength$1, +str: str$1, +translate: translate +}); + +/** + * 2 Dimensional Vector + * @module vec2 + */ + +/** + * Creates a new, empty vec2 + * + * @returns {vec2} a new 2D vector + */ + +function create() { + var out = new ARRAY_TYPE(2); + + if (ARRAY_TYPE != Float32Array) { + out[0] = 0; + out[1] = 0; + } + + return out; +} +/** + * Creates a new vec2 initialized with values from an existing vector + * + * @param {ReadonlyVec2} a vector to clone + * @returns {vec2} a new 2D vector + */ + +function clone(a) { + var out = new ARRAY_TYPE(2); + out[0] = a[0]; + out[1] = a[1]; + return out; +} +/** + * Creates a new vec2 initialized with the given values + * + * @param {Number} x X component + * @param {Number} y Y component + * @returns {vec2} a new 2D vector + */ + +function fromValues(x, y) { + var out = new ARRAY_TYPE(2); + out[0] = x; + out[1] = y; + return out; +} +/** + * Copy the values from one vec2 to another + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the source vector + * @returns {vec2} out + */ + +function copy(out, a) { + out[0] = a[0]; + out[1] = a[1]; + return out; +} +/** + * Set the components of a vec2 to the given values + * + * @param {vec2} out the receiving vector + * @param {Number} x X component + * @param {Number} y Y component + * @returns {vec2} out + */ + +function set(out, x, y) { + out[0] = x; + out[1] = y; + return out; +} +/** + * Adds two vec2's + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the first operand + * @param {ReadonlyVec2} b the second operand + * @returns {vec2} out + */ + +function add(out, a, b) { + out[0] = a[0] + b[0]; + out[1] = a[1] + b[1]; + return out; +} +/** + * Subtracts vector b from vector a + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the first operand + * @param {ReadonlyVec2} b the second operand + * @returns {vec2} out + */ + +function subtract(out, a, b) { + out[0] = a[0] - b[0]; + out[1] = a[1] - b[1]; + return out; +} +/** + * Multiplies two vec2's + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the first operand + * @param {ReadonlyVec2} b the second operand + * @returns {vec2} out + */ + +function multiply(out, a, b) { + out[0] = a[0] * b[0]; + out[1] = a[1] * b[1]; + return out; +} +/** + * Divides two vec2's + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the first operand + * @param {ReadonlyVec2} b the second operand + * @returns {vec2} out + */ + +function divide(out, a, b) { + out[0] = a[0] / b[0]; + out[1] = a[1] / b[1]; + return out; +} +/** + * Math.ceil the components of a vec2 + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a vector to ceil + * @returns {vec2} out + */ + +function ceil(out, a) { + out[0] = Math.ceil(a[0]); + out[1] = Math.ceil(a[1]); + return out; +} +/** + * Math.floor the components of a vec2 + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a vector to floor + * @returns {vec2} out + */ + +function floor(out, a) { + out[0] = Math.floor(a[0]); + out[1] = Math.floor(a[1]); + return out; +} +/** + * Returns the minimum of two vec2's + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the first operand + * @param {ReadonlyVec2} b the second operand + * @returns {vec2} out + */ + +function min(out, a, b) { + out[0] = Math.min(a[0], b[0]); + out[1] = Math.min(a[1], b[1]); + return out; +} +/** + * Returns the maximum of two vec2's + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the first operand + * @param {ReadonlyVec2} b the second operand + * @returns {vec2} out + */ + +function max(out, a, b) { + out[0] = Math.max(a[0], b[0]); + out[1] = Math.max(a[1], b[1]); + return out; +} +/** + * Math.round the components of a vec2 + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a vector to round + * @returns {vec2} out + */ + +function round(out, a) { + out[0] = Math.round(a[0]); + out[1] = Math.round(a[1]); + return out; +} +/** + * Scales a vec2 by a scalar number + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the vector to scale + * @param {Number} b amount to scale the vector by + * @returns {vec2} out + */ + +function scale(out, a, b) { + out[0] = a[0] * b; + out[1] = a[1] * b; + return out; +} +/** + * Adds two vec2's after scaling the second operand by a scalar value + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the first operand + * @param {ReadonlyVec2} b the second operand + * @param {Number} scale the amount to scale b by before adding + * @returns {vec2} out + */ + +function scaleAndAdd(out, a, b, scale) { + out[0] = a[0] + b[0] * scale; + out[1] = a[1] + b[1] * scale; + return out; +} +/** + * Calculates the euclidian distance between two vec2's + * + * @param {ReadonlyVec2} a the first operand + * @param {ReadonlyVec2} b the second operand + * @returns {Number} distance between a and b + */ + +function distance(a, b) { + var x = b[0] - a[0], + y = b[1] - a[1]; + return Math.hypot(x, y); +} +/** + * Calculates the squared euclidian distance between two vec2's + * + * @param {ReadonlyVec2} a the first operand + * @param {ReadonlyVec2} b the second operand + * @returns {Number} squared distance between a and b + */ + +function squaredDistance(a, b) { + var x = b[0] - a[0], + y = b[1] - a[1]; + return x * x + y * y; +} +/** + * Calculates the length of a vec2 + * + * @param {ReadonlyVec2} a vector to calculate length of + * @returns {Number} length of a + */ + +function length(a) { + var x = a[0], + y = a[1]; + return Math.hypot(x, y); +} +/** + * Calculates the squared length of a vec2 + * + * @param {ReadonlyVec2} a vector to calculate squared length of + * @returns {Number} squared length of a + */ + +function squaredLength(a) { + var x = a[0], + y = a[1]; + return x * x + y * y; +} +/** + * Negates the components of a vec2 + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a vector to negate + * @returns {vec2} out + */ + +function negate(out, a) { + out[0] = -a[0]; + out[1] = -a[1]; + return out; +} +/** + * Returns the inverse of the components of a vec2 + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a vector to invert + * @returns {vec2} out + */ + +function inverse(out, a) { + out[0] = 1.0 / a[0]; + out[1] = 1.0 / a[1]; + return out; +} +/** + * Normalize a vec2 + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a vector to normalize + * @returns {vec2} out + */ + +function normalize(out, a) { + var x = a[0], + y = a[1]; + var len = x * x + y * y; + + if (len > 0) { + //TODO: evaluate use of glm_invsqrt here? + len = 1 / Math.sqrt(len); + } + + out[0] = a[0] * len; + out[1] = a[1] * len; + return out; +} +/** + * Calculates the dot product of two vec2's + * + * @param {ReadonlyVec2} a the first operand + * @param {ReadonlyVec2} b the second operand + * @returns {Number} dot product of a and b + */ + +function dot$1(a, b) { + return a[0] * b[0] + a[1] * b[1]; +} +/** + * Computes the cross product of two vec2's + * Note that the cross product must by definition produce a 3D vector + * + * @param {vec3} out the receiving vector + * @param {ReadonlyVec2} a the first operand + * @param {ReadonlyVec2} b the second operand + * @returns {vec3} out + */ + +function cross(out, a, b) { + var z = a[0] * b[1] - a[1] * b[0]; + out[0] = out[1] = 0; + out[2] = z; + return out; +} +/** + * Performs a linear interpolation between two vec2's + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the first operand + * @param {ReadonlyVec2} b the second operand + * @param {Number} t interpolation amount, in the range [0-1], between the two inputs + * @returns {vec2} out + */ + +function lerp(out, a, b, t) { + var ax = a[0], + ay = a[1]; + out[0] = ax + t * (b[0] - ax); + out[1] = ay + t * (b[1] - ay); + return out; +} +/** + * Generates a random vector with the given scale + * + * @param {vec2} out the receiving vector + * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned + * @returns {vec2} out + */ + +function random(out, scale) { + scale = scale || 1.0; + var r = RANDOM() * 2.0 * Math.PI; + out[0] = Math.cos(r) * scale; + out[1] = Math.sin(r) * scale; + return out; +} +/** + * Transforms the vec2 with a mat2 + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the vector to transform + * @param {ReadonlyMat2} m matrix to transform with + * @returns {vec2} out + */ + +function transformMat2(out, a, m) { + var x = a[0], + y = a[1]; + out[0] = m[0] * x + m[2] * y; + out[1] = m[1] * x + m[3] * y; + return out; +} +/** + * Transforms the vec2 with a mat2d + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the vector to transform + * @param {ReadonlyMat2d} m matrix to transform with + * @returns {vec2} out + */ + +function transformMat2d(out, a, m) { + var x = a[0], + y = a[1]; + out[0] = m[0] * x + m[2] * y + m[4]; + out[1] = m[1] * x + m[3] * y + m[5]; + return out; +} +/** + * Transforms the vec2 with a mat3 + * 3rd vector component is implicitly '1' + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the vector to transform + * @param {ReadonlyMat3} m matrix to transform with + * @returns {vec2} out + */ + +function transformMat3(out, a, m) { + var x = a[0], + y = a[1]; + out[0] = m[0] * x + m[3] * y + m[6]; + out[1] = m[1] * x + m[4] * y + m[7]; + return out; +} +/** + * Transforms the vec2 with a mat4 + * 3rd vector component is implicitly '0' + * 4th vector component is implicitly '1' + * + * @param {vec2} out the receiving vector + * @param {ReadonlyVec2} a the vector to transform + * @param {ReadonlyMat4} m matrix to transform with + * @returns {vec2} out + */ + +function transformMat4(out, a, m) { + var x = a[0]; + var y = a[1]; + out[0] = m[0] * x + m[4] * y + m[12]; + out[1] = m[1] * x + m[5] * y + m[13]; + return out; +} +/** + * Rotate a 2D vector + * @param {vec2} out The receiving vec2 + * @param {ReadonlyVec2} a The vec2 point to rotate + * @param {ReadonlyVec2} b The origin of the rotation + * @param {Number} rad The angle of rotation in radians + * @returns {vec2} out + */ + +function rotate(out, a, b, rad) { + //Translate point to the origin + var p0 = a[0] - b[0], + p1 = a[1] - b[1], + sinC = Math.sin(rad), + cosC = Math.cos(rad); //perform rotation and translate to correct position + + out[0] = p0 * cosC - p1 * sinC + b[0]; + out[1] = p0 * sinC + p1 * cosC + b[1]; + return out; +} +/** + * Get the angle between two 2D vectors + * @param {ReadonlyVec2} a The first operand + * @param {ReadonlyVec2} b The second operand + * @returns {Number} The angle in radians + */ + +function angle(a, b) { + var x1 = a[0], + y1 = a[1], + x2 = b[0], + y2 = b[1], + // mag is the product of the magnitudes of a and b + mag = Math.sqrt(x1 * x1 + y1 * y1) * Math.sqrt(x2 * x2 + y2 * y2), + // mag &&.. short circuits if mag == 0 + cosine = mag && (x1 * x2 + y1 * y2) / mag; // Math.min(Math.max(cosine, -1), 1) clamps the cosine between -1 and 1 + + return Math.acos(Math.min(Math.max(cosine, -1), 1)); +} +/** + * Set the components of a vec2 to zero + * + * @param {vec2} out the receiving vector + * @returns {vec2} out + */ + +function zero(out) { + out[0] = 0.0; + out[1] = 0.0; + return out; +} +/** + * Returns a string representation of a vector + * + * @param {ReadonlyVec2} a vector to represent as a string + * @returns {String} string representation of the vector + */ + +function str(a) { + return "vec2(" + a[0] + ", " + a[1] + ")"; +} +/** + * Returns whether or not the vectors exactly have the same elements in the same position (when compared with ===) + * + * @param {ReadonlyVec2} a The first vector. + * @param {ReadonlyVec2} b The second vector. + * @returns {Boolean} True if the vectors are equal, false otherwise. + */ + +function exactEquals(a, b) { + return a[0] === b[0] && a[1] === b[1]; +} +/** + * Returns whether or not the vectors have approximately the same elements in the same position. + * + * @param {ReadonlyVec2} a The first vector. + * @param {ReadonlyVec2} b The second vector. + * @returns {Boolean} True if the vectors are equal, false otherwise. + */ + +function equals$1(a, b) { + var a0 = a[0], + a1 = a[1]; + var b0 = b[0], + b1 = b[1]; + return Math.abs(a0 - b0) <= EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)); +} +/** + * Alias for {@link vec2.length} + * @function + */ + +var len = length; +/** + * Alias for {@link vec2.subtract} + * @function + */ + +var sub = subtract; +/** + * Alias for {@link vec2.multiply} + * @function + */ + +var mul = multiply; +/** + * Alias for {@link vec2.divide} + * @function + */ + +var div = divide; +/** + * Alias for {@link vec2.distance} + * @function + */ + +var dist = distance; +/** + * Alias for {@link vec2.squaredDistance} + * @function + */ + +var sqrDist = squaredDistance; +/** + * Alias for {@link vec2.squaredLength} + * @function + */ + +var sqrLen = squaredLength; +/** + * Perform some operation over an array of vec2s. + * + * @param {Array} a the array of vectors to iterate over + * @param {Number} stride Number of elements between the start of each vec2. If 0 assumes tightly packed + * @param {Number} offset Number of elements to skip at the beginning of the array + * @param {Number} count Number of vec2s to iterate over. If 0 iterates over entire array + * @param {Function} fn Function to call for each vector in the array + * @param {Object} [arg] additional argument to pass to fn + * @returns {Array} a + * @function + */ + +var forEach = function () { + var vec = create(); + return function (a, stride, offset, count, fn, arg) { + var i, l; + + if (!stride) { + stride = 2; + } + + if (!offset) { + offset = 0; + } + + if (count) { + l = Math.min(count * stride + offset, a.length); + } else { + l = a.length; + } + + for (i = offset; i < l; i += stride) { + vec[0] = a[i]; + vec[1] = a[i + 1]; + fn(vec, vec, arg); + a[i] = vec[0]; + a[i + 1] = vec[1]; + } + + return a; + }; +}(); + +var vec2 = /*#__PURE__*/Object.freeze({ +__proto__: null, +add: add, +angle: angle, +ceil: ceil, +clone: clone, +copy: copy, +create: create, +cross: cross, +dist: dist, +distance: distance, +div: div, +divide: divide, +dot: dot$1, +equals: equals$1, +exactEquals: exactEquals, +floor: floor, +forEach: forEach, +fromValues: fromValues, +inverse: inverse, +len: len, +length: length, +lerp: lerp, +max: max, +min: min, +mul: mul, +multiply: multiply, +negate: negate, +normalize: normalize, +random: random, +rotate: rotate, +round: round, +scale: scale, +scaleAndAdd: scaleAndAdd, +set: set, +sqrDist: sqrDist, +sqrLen: sqrLen, +squaredDistance: squaredDistance, +squaredLength: squaredLength, +str: str, +sub: sub, +subtract: subtract, +transformMat2: transformMat2, +transformMat2d: transformMat2d, +transformMat3: transformMat3, +transformMat4: transformMat4, +zero: zero +}); + +/** + * A style layer that defines a circle + */ +class CircleStyleLayer extends StyleLayer { + constructor(layer) { + super(layer, properties$8); + } + createBucket(parameters) { + return new CircleBucket(parameters); + } + queryRadius(bucket) { + const circleBucket = bucket; + return getMaximumPaintValue('circle-radius', this, circleBucket) + + getMaximumPaintValue('circle-stroke-width', this, circleBucket) + + translateDistance(this.paint.get('circle-translate')); + } + queryIntersectsFeature(queryGeometry, feature, featureState, geometry, zoom, transform, pixelsToTileUnits, pixelPosMatrix) { + const translatedPolygon = translate$4(queryGeometry, this.paint.get('circle-translate'), this.paint.get('circle-translate-anchor'), transform.angle, pixelsToTileUnits); + const radius = this.paint.get('circle-radius').evaluate(feature, featureState); + const stroke = this.paint.get('circle-stroke-width').evaluate(feature, featureState); + const size = radius + stroke; + // For pitch-alignment: map, compare feature geometry to query geometry in the plane of the tile + // // Otherwise, compare geometry in the plane of the viewport + // // A circle with fixed scaling relative to the viewport gets larger in tile space as it moves into the distance + // // A circle with fixed scaling relative to the map gets smaller in viewport space as it moves into the distance + const alignWithMap = this.paint.get('circle-pitch-alignment') === 'map'; + const transformedPolygon = alignWithMap ? translatedPolygon : projectQueryGeometry$1(translatedPolygon, pixelPosMatrix); + const transformedSize = alignWithMap ? size * pixelsToTileUnits : size; + for (const ring of geometry) { + for (const point of ring) { + const transformedPoint = alignWithMap ? point : projectPoint(point, pixelPosMatrix); + let adjustedSize = transformedSize; + const projectedCenter = transformMat4$1([], [point.x, point.y, 0, 1], pixelPosMatrix); + if (this.paint.get('circle-pitch-scale') === 'viewport' && this.paint.get('circle-pitch-alignment') === 'map') { + adjustedSize *= projectedCenter[3] / transform.cameraToCenterDistance; + } + else if (this.paint.get('circle-pitch-scale') === 'map' && this.paint.get('circle-pitch-alignment') === 'viewport') { + adjustedSize *= transform.cameraToCenterDistance / projectedCenter[3]; + } + if (polygonIntersectsBufferedPoint(transformedPolygon, transformedPoint, adjustedSize)) + return true; + } + } + return false; + } +} +function projectPoint(p, pixelPosMatrix) { + const point = transformMat4$1([], [p.x, p.y, 0, 1], pixelPosMatrix); + return new Point$2(point[0] / point[3], point[1] / point[3]); +} +function projectQueryGeometry$1(queryGeometry, pixelPosMatrix) { + return queryGeometry.map((p) => { + return projectPoint(p, pixelPosMatrix); + }); +} + +class HeatmapBucket extends CircleBucket { +} +register('HeatmapBucket', HeatmapBucket, { omit: ['layers'] }); + +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ +let paint$7; +const getPaint$7 = () => paint$7 = paint$7 || new Properties({ + "heatmap-radius": new DataDrivenProperty(v8Spec["paint_heatmap"]["heatmap-radius"]), + "heatmap-weight": new DataDrivenProperty(v8Spec["paint_heatmap"]["heatmap-weight"]), + "heatmap-intensity": new DataConstantProperty(v8Spec["paint_heatmap"]["heatmap-intensity"]), + "heatmap-color": new ColorRampProperty(v8Spec["paint_heatmap"]["heatmap-color"]), + "heatmap-opacity": new DataConstantProperty(v8Spec["paint_heatmap"]["heatmap-opacity"]), +}); +var properties$7 = ({ get paint() { return getPaint$7(); } }); + +function createImage(image, { width, height }, channels, data) { + if (!data) { + data = new Uint8Array(width * height * channels); + } + else if (data instanceof Uint8ClampedArray) { + data = new Uint8Array(data.buffer); + } + else if (data.length !== width * height * channels) { + throw new RangeError(`mismatched image size. expected: ${data.length} but got: ${width * height * channels}`); + } + image.width = width; + image.height = height; + image.data = data; + return image; +} +function resizeImage(image, { width, height }, channels) { + if (width === image.width && height === image.height) { + return; + } + const newImage = createImage({}, { width, height }, channels); + copyImage(image, newImage, { x: 0, y: 0 }, { x: 0, y: 0 }, { + width: Math.min(image.width, width), + height: Math.min(image.height, height) + }, channels); + image.width = width; + image.height = height; + image.data = newImage.data; +} +function copyImage(srcImg, dstImg, srcPt, dstPt, size, channels) { + if (size.width === 0 || size.height === 0) { + return dstImg; + } + if (size.width > srcImg.width || + size.height > srcImg.height || + srcPt.x > srcImg.width - size.width || + srcPt.y > srcImg.height - size.height) { + throw new RangeError('out of range source coordinates for image copy'); + } + if (size.width > dstImg.width || + size.height > dstImg.height || + dstPt.x > dstImg.width - size.width || + dstPt.y > dstImg.height - size.height) { + throw new RangeError('out of range destination coordinates for image copy'); + } + const srcData = srcImg.data; + const dstData = dstImg.data; + if (srcData === dstData) + throw new Error('srcData equals dstData, so image is already copied'); + for (let y = 0; y < size.height; y++) { + const srcOffset = ((srcPt.y + y) * srcImg.width + srcPt.x) * channels; + const dstOffset = ((dstPt.y + y) * dstImg.width + dstPt.x) * channels; + for (let i = 0; i < size.width * channels; i++) { + dstData[dstOffset + i] = srcData[srcOffset + i]; + } + } + return dstImg; +} +/** + * @internal + * An image with alpha color value + */ +class AlphaImage { + constructor(size, data) { + createImage(this, size, 1, data); + } + resize(size) { + resizeImage(this, size, 1); + } + clone() { + return new AlphaImage({ width: this.width, height: this.height }, new Uint8Array(this.data)); + } + static copy(srcImg, dstImg, srcPt, dstPt, size) { + copyImage(srcImg, dstImg, srcPt, dstPt, size, 1); + } +} +/** + * An object to store image data not premultiplied, because ImageData is not premultiplied. + * UNPACK_PREMULTIPLY_ALPHA_WEBGL must be used when uploading to a texture. + */ +class RGBAImage { + constructor(size, data) { + createImage(this, size, 4, data); + } + resize(size) { + resizeImage(this, size, 4); + } + replace(data, copy) { + if (copy) { + this.data.set(data); + } + else if (data instanceof Uint8ClampedArray) { + this.data = new Uint8Array(data.buffer); + } + else { + this.data = data; + } + } + clone() { + return new RGBAImage({ width: this.width, height: this.height }, new Uint8Array(this.data)); + } + static copy(srcImg, dstImg, srcPt, dstPt, size) { + copyImage(srcImg, dstImg, srcPt, dstPt, size, 4); + } +} +register('AlphaImage', AlphaImage); +register('RGBAImage', RGBAImage); + +/** + * Given an expression that should evaluate to a color ramp, + * return a RGBA image representing that ramp expression. + */ +function renderColorRamp(params) { + const evaluationGlobals = {}; + const width = params.resolution || 256; + const height = params.clips ? params.clips.length : 1; + const image = params.image || new RGBAImage({ width, height }); + if (!isPowerOfTwo(width)) + throw new Error(`width is not a power of 2 - ${width}`); + const renderPixel = (stride, index, progress) => { + evaluationGlobals[params.evaluationKey] = progress; + const pxColor = params.expression.evaluate(evaluationGlobals); + // the colors are being unpremultiplied because Color uses + // premultiplied values, and the Texture class expects unpremultiplied ones + image.data[stride + index + 0] = Math.floor(pxColor.r * 255 / pxColor.a); + image.data[stride + index + 1] = Math.floor(pxColor.g * 255 / pxColor.a); + image.data[stride + index + 2] = Math.floor(pxColor.b * 255 / pxColor.a); + image.data[stride + index + 3] = Math.floor(pxColor.a * 255); + }; + if (!params.clips) { + for (let i = 0, j = 0; i < width; i++, j += 4) { + const progress = i / (width - 1); + renderPixel(0, j, progress); + } + } + else { + for (let clip = 0, stride = 0; clip < height; ++clip, stride += width * 4) { + for (let i = 0, j = 0; i < width; i++, j += 4) { + // Remap progress between clips + const progress = i / (width - 1); + const { start, end } = params.clips[clip]; + const evaluationProgress = start * (1 - progress) + end * progress; + renderPixel(stride, j, evaluationProgress); + } + } + } + return image; +} + +/** + * A style layer that defines a heatmap + */ +class HeatmapStyleLayer extends StyleLayer { + createBucket(options) { + return new HeatmapBucket(options); + } + constructor(layer) { + super(layer, properties$7); + // make sure color ramp texture is generated for default heatmap color too + this._updateColorRamp(); + } + _handleSpecialPaintPropertyUpdate(name) { + if (name === 'heatmap-color') { + this._updateColorRamp(); + } + } + _updateColorRamp() { + const expression = this._transitionablePaint._values['heatmap-color'].value.expression; + this.colorRamp = renderColorRamp({ + expression, + evaluationKey: 'heatmapDensity', + image: this.colorRamp + }); + this.colorRampTexture = null; + } + resize() { + if (this.heatmapFbo) { + this.heatmapFbo.destroy(); + this.heatmapFbo = null; + } + } + queryRadius() { + return 0; + } + queryIntersectsFeature() { + return false; + } + hasOffscreenPass() { + return this.paint.get('heatmap-opacity') !== 0 && this.visibility !== 'none'; + } +} + +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ +let paint$6; +const getPaint$6 = () => paint$6 = paint$6 || new Properties({ + "hillshade-illumination-direction": new DataConstantProperty(v8Spec["paint_hillshade"]["hillshade-illumination-direction"]), + "hillshade-illumination-anchor": new DataConstantProperty(v8Spec["paint_hillshade"]["hillshade-illumination-anchor"]), + "hillshade-exaggeration": new DataConstantProperty(v8Spec["paint_hillshade"]["hillshade-exaggeration"]), + "hillshade-shadow-color": new DataConstantProperty(v8Spec["paint_hillshade"]["hillshade-shadow-color"]), + "hillshade-highlight-color": new DataConstantProperty(v8Spec["paint_hillshade"]["hillshade-highlight-color"]), + "hillshade-accent-color": new DataConstantProperty(v8Spec["paint_hillshade"]["hillshade-accent-color"]), +}); +var properties$6 = ({ get paint() { return getPaint$6(); } }); + +class HillshadeStyleLayer extends StyleLayer { + constructor(layer) { + super(layer, properties$6); + } + hasOffscreenPass() { + return this.paint.get('hillshade-exaggeration') !== 0 && this.visibility !== 'none'; + } +} + +const layout$4 = createLayout([ + { name: 'a_pos', components: 2, type: 'Int16' } +], 4); +const { members: members$3, size: size$3, alignment: alignment$3 } = layout$4; + +var earcut$2 = {exports: {}}; + +var earcut_1 = earcut$2.exports; + +'use strict'; + +earcut$2.exports = earcut; +var _default = earcut$2.exports.default = earcut; + +function earcut(data, holeIndices, dim) { + + dim = dim || 2; + + var hasHoles = holeIndices && holeIndices.length, + outerLen = hasHoles ? holeIndices[0] * dim : data.length, + outerNode = linkedList(data, 0, outerLen, dim, true), + triangles = []; + + if (!outerNode || outerNode.next === outerNode.prev) return triangles; + + var minX, minY, maxX, maxY, x, y, invSize; + + if (hasHoles) outerNode = eliminateHoles(data, holeIndices, outerNode, dim); + + // if the shape is not too simple, we'll use z-order curve hash later; calculate polygon bbox + if (data.length > 80 * dim) { + minX = maxX = data[0]; + minY = maxY = data[1]; + + for (var i = dim; i < outerLen; i += dim) { + x = data[i]; + y = data[i + 1]; + if (x < minX) minX = x; + if (y < minY) minY = y; + if (x > maxX) maxX = x; + if (y > maxY) maxY = y; + } + + // minX, minY and invSize are later used to transform coords into integers for z-order calculation + invSize = Math.max(maxX - minX, maxY - minY); + invSize = invSize !== 0 ? 32767 / invSize : 0; + } + + earcutLinked(outerNode, triangles, dim, minX, minY, invSize, 0); + + return triangles; +} + +// create a circular doubly linked list from polygon points in the specified winding order +function linkedList(data, start, end, dim, clockwise) { + var i, last; + + if (clockwise === (signedArea$1(data, start, end, dim) > 0)) { + for (i = start; i < end; i += dim) last = insertNode(i, data[i], data[i + 1], last); + } else { + for (i = end - dim; i >= start; i -= dim) last = insertNode(i, data[i], data[i + 1], last); + } + + if (last && equals(last, last.next)) { + removeNode(last); + last = last.next; + } + + return last; +} + +// eliminate colinear or duplicate points +function filterPoints(start, end) { + if (!start) return start; + if (!end) end = start; + + var p = start, + again; + do { + again = false; + + if (!p.steiner && (equals(p, p.next) || area(p.prev, p, p.next) === 0)) { + removeNode(p); + p = end = p.prev; + if (p === p.next) break; + again = true; + + } else { + p = p.next; + } + } while (again || p !== end); + + return end; +} + +// main ear slicing loop which triangulates a polygon (given as a linked list) +function earcutLinked(ear, triangles, dim, minX, minY, invSize, pass) { + if (!ear) return; + + // interlink polygon nodes in z-order + if (!pass && invSize) indexCurve(ear, minX, minY, invSize); + + var stop = ear, + prev, next; + + // iterate through ears, slicing them one by one + while (ear.prev !== ear.next) { + prev = ear.prev; + next = ear.next; + + if (invSize ? isEarHashed(ear, minX, minY, invSize) : isEar(ear)) { + // cut off the triangle + triangles.push(prev.i / dim | 0); + triangles.push(ear.i / dim | 0); + triangles.push(next.i / dim | 0); + + removeNode(ear); + + // skipping the next vertex leads to less sliver triangles + ear = next.next; + stop = next.next; + + continue; + } + + ear = next; + + // if we looped through the whole remaining polygon and can't find any more ears + if (ear === stop) { + // try filtering points and slicing again + if (!pass) { + earcutLinked(filterPoints(ear), triangles, dim, minX, minY, invSize, 1); + + // if this didn't work, try curing all small self-intersections locally + } else if (pass === 1) { + ear = cureLocalIntersections(filterPoints(ear), triangles, dim); + earcutLinked(ear, triangles, dim, minX, minY, invSize, 2); + + // as a last resort, try splitting the remaining polygon into two + } else if (pass === 2) { + splitEarcut(ear, triangles, dim, minX, minY, invSize); + } + + break; + } + } +} + +// check whether a polygon node forms a valid ear with adjacent nodes +function isEar(ear) { + var a = ear.prev, + b = ear, + c = ear.next; + + if (area(a, b, c) >= 0) return false; // reflex, can't be an ear + + // now make sure we don't have other points inside the potential ear + var ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y; + + // triangle bbox; min & max are calculated like this for speed + var x0 = ax < bx ? (ax < cx ? ax : cx) : (bx < cx ? bx : cx), + y0 = ay < by ? (ay < cy ? ay : cy) : (by < cy ? by : cy), + x1 = ax > bx ? (ax > cx ? ax : cx) : (bx > cx ? bx : cx), + y1 = ay > by ? (ay > cy ? ay : cy) : (by > cy ? by : cy); + + var p = c.next; + while (p !== a) { + if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 && + pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) && + area(p.prev, p, p.next) >= 0) return false; + p = p.next; + } + + return true; +} + +function isEarHashed(ear, minX, minY, invSize) { + var a = ear.prev, + b = ear, + c = ear.next; + + if (area(a, b, c) >= 0) return false; // reflex, can't be an ear + + var ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y; + + // triangle bbox; min & max are calculated like this for speed + var x0 = ax < bx ? (ax < cx ? ax : cx) : (bx < cx ? bx : cx), + y0 = ay < by ? (ay < cy ? ay : cy) : (by < cy ? by : cy), + x1 = ax > bx ? (ax > cx ? ax : cx) : (bx > cx ? bx : cx), + y1 = ay > by ? (ay > cy ? ay : cy) : (by > cy ? by : cy); + + // z-order range for the current triangle bbox; + var minZ = zOrder(x0, y0, minX, minY, invSize), + maxZ = zOrder(x1, y1, minX, minY, invSize); + + var p = ear.prevZ, + n = ear.nextZ; + + // look for points inside the triangle in both directions + while (p && p.z >= minZ && n && n.z <= maxZ) { + if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 && p !== a && p !== c && + pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) && area(p.prev, p, p.next) >= 0) return false; + p = p.prevZ; + + if (n.x >= x0 && n.x <= x1 && n.y >= y0 && n.y <= y1 && n !== a && n !== c && + pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) && area(n.prev, n, n.next) >= 0) return false; + n = n.nextZ; + } + + // look for remaining points in decreasing z-order + while (p && p.z >= minZ) { + if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 && p !== a && p !== c && + pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) && area(p.prev, p, p.next) >= 0) return false; + p = p.prevZ; + } + + // look for remaining points in increasing z-order + while (n && n.z <= maxZ) { + if (n.x >= x0 && n.x <= x1 && n.y >= y0 && n.y <= y1 && n !== a && n !== c && + pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) && area(n.prev, n, n.next) >= 0) return false; + n = n.nextZ; + } + + return true; +} + +// go through all polygon nodes and cure small local self-intersections +function cureLocalIntersections(start, triangles, dim) { + var p = start; + do { + var a = p.prev, + b = p.next.next; + + if (!equals(a, b) && intersects(a, p, p.next, b) && locallyInside(a, b) && locallyInside(b, a)) { + + triangles.push(a.i / dim | 0); + triangles.push(p.i / dim | 0); + triangles.push(b.i / dim | 0); + + // remove two nodes involved + removeNode(p); + removeNode(p.next); + + p = start = b; + } + p = p.next; + } while (p !== start); + + return filterPoints(p); +} + +// try splitting polygon into two and triangulate them independently +function splitEarcut(start, triangles, dim, minX, minY, invSize) { + // look for a valid diagonal that divides the polygon into two + var a = start; + do { + var b = a.next.next; + while (b !== a.prev) { + if (a.i !== b.i && isValidDiagonal(a, b)) { + // split the polygon in two by the diagonal + var c = splitPolygon(a, b); + + // filter colinear points around the cuts + a = filterPoints(a, a.next); + c = filterPoints(c, c.next); + + // run earcut on each half + earcutLinked(a, triangles, dim, minX, minY, invSize, 0); + earcutLinked(c, triangles, dim, minX, minY, invSize, 0); + return; + } + b = b.next; + } + a = a.next; + } while (a !== start); +} + +// link every hole into the outer loop, producing a single-ring polygon without holes +function eliminateHoles(data, holeIndices, outerNode, dim) { + var queue = [], + i, len, start, end, list; + + for (i = 0, len = holeIndices.length; i < len; i++) { + start = holeIndices[i] * dim; + end = i < len - 1 ? holeIndices[i + 1] * dim : data.length; + list = linkedList(data, start, end, dim, false); + if (list === list.next) list.steiner = true; + queue.push(getLeftmost(list)); + } + + queue.sort(compareX); + + // process holes from left to right + for (i = 0; i < queue.length; i++) { + outerNode = eliminateHole(queue[i], outerNode); + } + + return outerNode; +} + +function compareX(a, b) { + return a.x - b.x; +} + +// find a bridge between vertices that connects hole with an outer ring and and link it +function eliminateHole(hole, outerNode) { + var bridge = findHoleBridge(hole, outerNode); + if (!bridge) { + return outerNode; + } + + var bridgeReverse = splitPolygon(bridge, hole); + + // filter collinear points around the cuts + filterPoints(bridgeReverse, bridgeReverse.next); + return filterPoints(bridge, bridge.next); +} + +// David Eberly's algorithm for finding a bridge between hole and outer polygon +function findHoleBridge(hole, outerNode) { + var p = outerNode, + hx = hole.x, + hy = hole.y, + qx = -Infinity, + m; + + // find a segment intersected by a ray from the hole's leftmost point to the left; + // segment's endpoint with lesser x will be potential connection point + do { + if (hy <= p.y && hy >= p.next.y && p.next.y !== p.y) { + var x = p.x + (hy - p.y) * (p.next.x - p.x) / (p.next.y - p.y); + if (x <= hx && x > qx) { + qx = x; + m = p.x < p.next.x ? p : p.next; + if (x === hx) return m; // hole touches outer segment; pick leftmost endpoint + } + } + p = p.next; + } while (p !== outerNode); + + if (!m) return null; + + // look for points inside the triangle of hole point, segment intersection and endpoint; + // if there are no points found, we have a valid connection; + // otherwise choose the point of the minimum angle with the ray as connection point + + var stop = m, + mx = m.x, + my = m.y, + tanMin = Infinity, + tan; + + p = m; + + do { + if (hx >= p.x && p.x >= mx && hx !== p.x && + pointInTriangle(hy < my ? hx : qx, hy, mx, my, hy < my ? qx : hx, hy, p.x, p.y)) { + + tan = Math.abs(hy - p.y) / (hx - p.x); // tangential + + if (locallyInside(p, hole) && + (tan < tanMin || (tan === tanMin && (p.x > m.x || (p.x === m.x && sectorContainsSector(m, p)))))) { + m = p; + tanMin = tan; + } + } + + p = p.next; + } while (p !== stop); + + return m; +} + +// whether sector in vertex m contains sector in vertex p in the same coordinates +function sectorContainsSector(m, p) { + return area(m.prev, m, p.prev) < 0 && area(p.next, m, m.next) < 0; +} + +// interlink polygon nodes in z-order +function indexCurve(start, minX, minY, invSize) { + var p = start; + do { + if (p.z === 0) p.z = zOrder(p.x, p.y, minX, minY, invSize); + p.prevZ = p.prev; + p.nextZ = p.next; + p = p.next; + } while (p !== start); + + p.prevZ.nextZ = null; + p.prevZ = null; + + sortLinked(p); +} + +// Simon Tatham's linked list merge sort algorithm +// http://www.chiark.greenend.org.uk/~sgtatham/algorithms/listsort.html +function sortLinked(list) { + var i, p, q, e, tail, numMerges, pSize, qSize, + inSize = 1; + + do { + p = list; + list = null; + tail = null; + numMerges = 0; + + while (p) { + numMerges++; + q = p; + pSize = 0; + for (i = 0; i < inSize; i++) { + pSize++; + q = q.nextZ; + if (!q) break; + } + qSize = inSize; + + while (pSize > 0 || (qSize > 0 && q)) { + + if (pSize !== 0 && (qSize === 0 || !q || p.z <= q.z)) { + e = p; + p = p.nextZ; + pSize--; + } else { + e = q; + q = q.nextZ; + qSize--; + } + + if (tail) tail.nextZ = e; + else list = e; + + e.prevZ = tail; + tail = e; + } + + p = q; + } + + tail.nextZ = null; + inSize *= 2; + + } while (numMerges > 1); + + return list; +} + +// z-order of a point given coords and inverse of the longer side of data bbox +function zOrder(x, y, minX, minY, invSize) { + // coords are transformed into non-negative 15-bit integer range + x = (x - minX) * invSize | 0; + y = (y - minY) * invSize | 0; + + x = (x | (x << 8)) & 0x00FF00FF; + x = (x | (x << 4)) & 0x0F0F0F0F; + x = (x | (x << 2)) & 0x33333333; + x = (x | (x << 1)) & 0x55555555; + + y = (y | (y << 8)) & 0x00FF00FF; + y = (y | (y << 4)) & 0x0F0F0F0F; + y = (y | (y << 2)) & 0x33333333; + y = (y | (y << 1)) & 0x55555555; + + return x | (y << 1); +} + +// find the leftmost node of a polygon ring +function getLeftmost(start) { + var p = start, + leftmost = start; + do { + if (p.x < leftmost.x || (p.x === leftmost.x && p.y < leftmost.y)) leftmost = p; + p = p.next; + } while (p !== start); + + return leftmost; +} + +// check if a point lies within a convex triangle +function pointInTriangle(ax, ay, bx, by, cx, cy, px, py) { + return (cx - px) * (ay - py) >= (ax - px) * (cy - py) && + (ax - px) * (by - py) >= (bx - px) * (ay - py) && + (bx - px) * (cy - py) >= (cx - px) * (by - py); +} + +// check if a diagonal between two polygon nodes is valid (lies in polygon interior) +function isValidDiagonal(a, b) { + return a.next.i !== b.i && a.prev.i !== b.i && !intersectsPolygon(a, b) && // dones't intersect other edges + (locallyInside(a, b) && locallyInside(b, a) && middleInside(a, b) && // locally visible + (area(a.prev, a, b.prev) || area(a, b.prev, b)) || // does not create opposite-facing sectors + equals(a, b) && area(a.prev, a, a.next) > 0 && area(b.prev, b, b.next) > 0); // special zero-length case +} + +// signed area of a triangle +function area(p, q, r) { + return (q.y - p.y) * (r.x - q.x) - (q.x - p.x) * (r.y - q.y); +} + +// check if two points are equal +function equals(p1, p2) { + return p1.x === p2.x && p1.y === p2.y; +} + +// check if two segments intersect +function intersects(p1, q1, p2, q2) { + var o1 = sign(area(p1, q1, p2)); + var o2 = sign(area(p1, q1, q2)); + var o3 = sign(area(p2, q2, p1)); + var o4 = sign(area(p2, q2, q1)); + + if (o1 !== o2 && o3 !== o4) return true; // general case + + if (o1 === 0 && onSegment(p1, p2, q1)) return true; // p1, q1 and p2 are collinear and p2 lies on p1q1 + if (o2 === 0 && onSegment(p1, q2, q1)) return true; // p1, q1 and q2 are collinear and q2 lies on p1q1 + if (o3 === 0 && onSegment(p2, p1, q2)) return true; // p2, q2 and p1 are collinear and p1 lies on p2q2 + if (o4 === 0 && onSegment(p2, q1, q2)) return true; // p2, q2 and q1 are collinear and q1 lies on p2q2 + + return false; +} + +// for collinear points p, q, r, check if point q lies on segment pr +function onSegment(p, q, r) { + return q.x <= Math.max(p.x, r.x) && q.x >= Math.min(p.x, r.x) && q.y <= Math.max(p.y, r.y) && q.y >= Math.min(p.y, r.y); +} + +function sign(num) { + return num > 0 ? 1 : num < 0 ? -1 : 0; +} + +// check if a polygon diagonal intersects any polygon segments +function intersectsPolygon(a, b) { + var p = a; + do { + if (p.i !== a.i && p.next.i !== a.i && p.i !== b.i && p.next.i !== b.i && + intersects(p, p.next, a, b)) return true; + p = p.next; + } while (p !== a); + + return false; +} + +// check if a polygon diagonal is locally inside the polygon +function locallyInside(a, b) { + return area(a.prev, a, a.next) < 0 ? + area(a, b, a.next) >= 0 && area(a, a.prev, b) >= 0 : + area(a, b, a.prev) < 0 || area(a, a.next, b) < 0; +} + +// check if the middle point of a polygon diagonal is inside the polygon +function middleInside(a, b) { + var p = a, + inside = false, + px = (a.x + b.x) / 2, + py = (a.y + b.y) / 2; + do { + if (((p.y > py) !== (p.next.y > py)) && p.next.y !== p.y && + (px < (p.next.x - p.x) * (py - p.y) / (p.next.y - p.y) + p.x)) + inside = !inside; + p = p.next; + } while (p !== a); + + return inside; +} + +// link two polygon vertices with a bridge; if the vertices belong to the same ring, it splits polygon into two; +// if one belongs to the outer ring and another to a hole, it merges it into a single ring +function splitPolygon(a, b) { + var a2 = new Node(a.i, a.x, a.y), + b2 = new Node(b.i, b.x, b.y), + an = a.next, + bp = b.prev; + + a.next = b; + b.prev = a; + + a2.next = an; + an.prev = a2; + + b2.next = a2; + a2.prev = b2; + + bp.next = b2; + b2.prev = bp; + + return b2; +} + +// create a node and optionally link it with previous one (in a circular doubly linked list) +function insertNode(i, x, y, last) { + var p = new Node(i, x, y); + + if (!last) { + p.prev = p; + p.next = p; + + } else { + p.next = last.next; + p.prev = last; + last.next.prev = p; + last.next = p; + } + return p; +} + +function removeNode(p) { + p.next.prev = p.prev; + p.prev.next = p.next; + + if (p.prevZ) p.prevZ.nextZ = p.nextZ; + if (p.nextZ) p.nextZ.prevZ = p.prevZ; +} + +function Node(i, x, y) { + // vertex index in coordinates array + this.i = i; + + // vertex coordinates + this.x = x; + this.y = y; + + // previous and next vertex nodes in a polygon ring + this.prev = null; + this.next = null; + + // z-order curve value + this.z = 0; + + // previous and next nodes in z-order + this.prevZ = null; + this.nextZ = null; + + // indicates whether this is a steiner point + this.steiner = false; +} + +// return a percentage difference between the polygon area and its triangulation area; +// used to verify correctness of triangulation +earcut.deviation = function (data, holeIndices, dim, triangles) { + var hasHoles = holeIndices && holeIndices.length; + var outerLen = hasHoles ? holeIndices[0] * dim : data.length; + + var polygonArea = Math.abs(signedArea$1(data, 0, outerLen, dim)); + if (hasHoles) { + for (var i = 0, len = holeIndices.length; i < len; i++) { + var start = holeIndices[i] * dim; + var end = i < len - 1 ? holeIndices[i + 1] * dim : data.length; + polygonArea -= Math.abs(signedArea$1(data, start, end, dim)); + } + } + + var trianglesArea = 0; + for (i = 0; i < triangles.length; i += 3) { + var a = triangles[i] * dim; + var b = triangles[i + 1] * dim; + var c = triangles[i + 2] * dim; + trianglesArea += Math.abs( + (data[a] - data[c]) * (data[b + 1] - data[a + 1]) - + (data[a] - data[b]) * (data[c + 1] - data[a + 1])); + } + + return polygonArea === 0 && trianglesArea === 0 ? 0 : + Math.abs((trianglesArea - polygonArea) / polygonArea); +}; + +function signedArea$1(data, start, end, dim) { + var sum = 0; + for (var i = start, j = end - dim; i < end; i += dim) { + sum += (data[j] - data[i]) * (data[i + 1] + data[j + 1]); + j = i; + } + return sum; +} + +// turn a polygon in a multi-dimensional array form (e.g. as in GeoJSON) into a form Earcut accepts +earcut.flatten = function (data) { + var dim = data[0][0].length, + result = {vertices: [], holes: [], dimensions: dim}, + holeIndex = 0; + + for (var i = 0; i < data.length; i++) { + for (var j = 0; j < data[i].length; j++) { + for (var d = 0; d < dim; d++) result.vertices.push(data[i][j][d]); + } + if (i > 0) { + holeIndex += data[i - 1].length; + result.holes.push(holeIndex); + } + } + return result; +}; + +var earcutExports = earcut$2.exports; +var earcut$1 = /*@__PURE__*/getDefaultExportFromCjs(earcutExports); + +function quickselect(arr, k, left, right, compare) { + quickselectStep(arr, k, left || 0, right || (arr.length - 1), compare || defaultCompare$1); +} + +function quickselectStep(arr, k, left, right, compare) { + + while (right > left) { + if (right - left > 600) { + var n = right - left + 1; + var m = k - left + 1; + var z = Math.log(n); + var s = 0.5 * Math.exp(2 * z / 3); + var sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1); + var newLeft = Math.max(left, Math.floor(k - m * s / n + sd)); + var newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd)); + quickselectStep(arr, k, newLeft, newRight, compare); + } + + var t = arr[k]; + var i = left; + var j = right; + + swap$1(arr, left, k); + if (compare(arr[right], t) > 0) swap$1(arr, left, right); + + while (i < j) { + swap$1(arr, i, j); + i++; + j--; + while (compare(arr[i], t) < 0) i++; + while (compare(arr[j], t) > 0) j--; + } + + if (compare(arr[left], t) === 0) swap$1(arr, left, j); + else { + j++; + swap$1(arr, j, right); + } + + if (j <= k) left = j + 1; + if (k <= j) right = j - 1; + } +} + +function swap$1(arr, i, j) { + var tmp = arr[i]; + arr[i] = arr[j]; + arr[j] = tmp; +} + +function defaultCompare$1(a, b) { + return a < b ? -1 : a > b ? 1 : 0; +} + +// classifies an array of rings into polygons with outer rings and holes +function classifyRings$1(rings, maxRings) { + const len = rings.length; + if (len <= 1) + return [rings]; + const polygons = []; + let polygon, ccw; + for (let i = 0; i < len; i++) { + const area = calculateSignedArea(rings[i]); + if (area === 0) + continue; + rings[i].area = Math.abs(area); + if (ccw === undefined) + ccw = area < 0; + if (ccw === area < 0) { + if (polygon) + polygons.push(polygon); + polygon = [rings[i]]; + } + else { + polygon.push(rings[i]); + } + } + if (polygon) + polygons.push(polygon); + // Earcut performance degrades with the # of rings in a polygon. For this + // reason, we limit strip out all but the `maxRings` largest rings. + if (maxRings > 1) { + for (let j = 0; j < polygons.length; j++) { + if (polygons[j].length <= maxRings) + continue; + quickselect(polygons[j], maxRings, 1, polygons[j].length - 1, compareAreas); + polygons[j] = polygons[j].slice(0, maxRings); + } + } + return polygons; +} +function compareAreas(a, b) { + return b.area - a.area; +} + +function hasPattern(type, layers, options) { + const patterns = options.patternDependencies; + let hasPattern = false; + for (const layer of layers) { + const patternProperty = layer.paint.get(`${type}-pattern`); + if (!patternProperty.isConstant()) { + hasPattern = true; + } + const constantPattern = patternProperty.constantOr(null); + if (constantPattern) { + hasPattern = true; + patterns[constantPattern.to] = true; + patterns[constantPattern.from] = true; + } + } + return hasPattern; +} +function addPatternDependencies(type, layers, patternFeature, zoom, options) { + const patterns = options.patternDependencies; + for (const layer of layers) { + const patternProperty = layer.paint.get(`${type}-pattern`); + const patternPropertyValue = patternProperty.value; + if (patternPropertyValue.kind !== 'constant') { + let min = patternPropertyValue.evaluate({ zoom: zoom - 1 }, patternFeature, {}, options.availableImages); + let mid = patternPropertyValue.evaluate({ zoom }, patternFeature, {}, options.availableImages); + let max = patternPropertyValue.evaluate({ zoom: zoom + 1 }, patternFeature, {}, options.availableImages); + min = min && min.name ? min.name : min; + mid = mid && mid.name ? mid.name : mid; + max = max && max.name ? max.name : max; + // add to patternDependencies + patterns[min] = true; + patterns[mid] = true; + patterns[max] = true; + // save for layout + patternFeature.patterns[layer.id] = { min, mid, max }; + } + } + return patternFeature; +} + +const EARCUT_MAX_RINGS$1 = 500; +class FillBucket { + constructor(options) { + this.zoom = options.zoom; + this.overscaling = options.overscaling; + this.layers = options.layers; + this.layerIds = this.layers.map(layer => layer.id); + this.index = options.index; + this.hasPattern = false; + this.patternFeatures = []; + this.layoutVertexArray = new FillLayoutArray(); + this.indexArray = new TriangleIndexArray(); + this.indexArray2 = new LineIndexArray(); + this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom); + this.segments = new SegmentVector(); + this.segments2 = new SegmentVector(); + this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id); + } + populate(features, options, canonical) { + this.hasPattern = hasPattern('fill', this.layers, options); + const fillSortKey = this.layers[0].layout.get('fill-sort-key'); + const sortFeaturesByKey = !fillSortKey.isConstant(); + const bucketFeatures = []; + for (const { feature, id, index, sourceLayerIndex } of features) { + const needGeometry = this.layers[0]._featureFilter.needGeometry; + const evaluationFeature = toEvaluationFeature(feature, needGeometry); + if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) + continue; + const sortKey = sortFeaturesByKey ? + fillSortKey.evaluate(evaluationFeature, {}, canonical, options.availableImages) : + undefined; + const bucketFeature = { + id, + properties: feature.properties, + type: feature.type, + sourceLayerIndex, + index, + geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature), + patterns: {}, + sortKey + }; + bucketFeatures.push(bucketFeature); + } + if (sortFeaturesByKey) { + bucketFeatures.sort((a, b) => a.sortKey - b.sortKey); + } + for (const bucketFeature of bucketFeatures) { + const { geometry, index, sourceLayerIndex } = bucketFeature; + if (this.hasPattern) { + const patternFeature = addPatternDependencies('fill', this.layers, bucketFeature, this.zoom, options); + // pattern features are added only once the pattern is loaded into the image atlas + // so are stored during populate until later updated with positions by tile worker in addFeatures + this.patternFeatures.push(patternFeature); + } + else { + this.addFeature(bucketFeature, geometry, index, canonical, {}); + } + const feature = features[index].feature; + options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index); + } + } + update(states, vtLayer, imagePositions) { + if (!this.stateDependentLayers.length) + return; + this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions); + } + addFeatures(options, canonical, imagePositions) { + for (const feature of this.patternFeatures) { + this.addFeature(feature, feature.geometry, feature.index, canonical, imagePositions); + } + } + isEmpty() { + return this.layoutVertexArray.length === 0; + } + uploadPending() { + return !this.uploaded || this.programConfigurations.needsUpload; + } + upload(context) { + if (!this.uploaded) { + this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, members$3); + this.indexBuffer = context.createIndexBuffer(this.indexArray); + this.indexBuffer2 = context.createIndexBuffer(this.indexArray2); + } + this.programConfigurations.upload(context); + this.uploaded = true; + } + destroy() { + if (!this.layoutVertexBuffer) + return; + this.layoutVertexBuffer.destroy(); + this.indexBuffer.destroy(); + this.indexBuffer2.destroy(); + this.programConfigurations.destroy(); + this.segments.destroy(); + this.segments2.destroy(); + } + addFeature(feature, geometry, index, canonical, imagePositions) { + for (const polygon of classifyRings$1(geometry, EARCUT_MAX_RINGS$1)) { + let numVertices = 0; + for (const ring of polygon) { + numVertices += ring.length; + } + const triangleSegment = this.segments.prepareSegment(numVertices, this.layoutVertexArray, this.indexArray); + const triangleIndex = triangleSegment.vertexLength; + const flattened = []; + const holeIndices = []; + for (const ring of polygon) { + if (ring.length === 0) { + continue; + } + if (ring !== polygon[0]) { + holeIndices.push(flattened.length / 2); + } + const lineSegment = this.segments2.prepareSegment(ring.length, this.layoutVertexArray, this.indexArray2); + const lineIndex = lineSegment.vertexLength; + this.layoutVertexArray.emplaceBack(ring[0].x, ring[0].y); + this.indexArray2.emplaceBack(lineIndex + ring.length - 1, lineIndex); + flattened.push(ring[0].x); + flattened.push(ring[0].y); + for (let i = 1; i < ring.length; i++) { + this.layoutVertexArray.emplaceBack(ring[i].x, ring[i].y); + this.indexArray2.emplaceBack(lineIndex + i - 1, lineIndex + i); + flattened.push(ring[i].x); + flattened.push(ring[i].y); + } + lineSegment.vertexLength += ring.length; + lineSegment.primitiveLength += ring.length; + } + const indices = earcut$1(flattened, holeIndices); + for (let i = 0; i < indices.length; i += 3) { + this.indexArray.emplaceBack(triangleIndex + indices[i], triangleIndex + indices[i + 1], triangleIndex + indices[i + 2]); + } + triangleSegment.vertexLength += numVertices; + triangleSegment.primitiveLength += indices.length / 3; + } + this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical); + } +} +register('FillBucket', FillBucket, { omit: ['layers', 'patternFeatures'] }); + +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ +let layout$3; +const getLayout$2 = () => layout$3 = layout$3 || new Properties({ + "fill-sort-key": new DataDrivenProperty(v8Spec["layout_fill"]["fill-sort-key"]), +}); +let paint$5; +const getPaint$5 = () => paint$5 = paint$5 || new Properties({ + "fill-antialias": new DataConstantProperty(v8Spec["paint_fill"]["fill-antialias"]), + "fill-opacity": new DataDrivenProperty(v8Spec["paint_fill"]["fill-opacity"]), + "fill-color": new DataDrivenProperty(v8Spec["paint_fill"]["fill-color"]), + "fill-outline-color": new DataDrivenProperty(v8Spec["paint_fill"]["fill-outline-color"]), + "fill-translate": new DataConstantProperty(v8Spec["paint_fill"]["fill-translate"]), + "fill-translate-anchor": new DataConstantProperty(v8Spec["paint_fill"]["fill-translate-anchor"]), + "fill-pattern": new CrossFadedDataDrivenProperty(v8Spec["paint_fill"]["fill-pattern"]), +}); +var properties$5 = ({ get paint() { return getPaint$5(); }, get layout() { return getLayout$2(); } }); + +class FillStyleLayer extends StyleLayer { + constructor(layer) { + super(layer, properties$5); + } + recalculate(parameters, availableImages) { + super.recalculate(parameters, availableImages); + const outlineColor = this.paint._values['fill-outline-color']; + if (outlineColor.value.kind === 'constant' && outlineColor.value.value === undefined) { + this.paint._values['fill-outline-color'] = this.paint._values['fill-color']; + } + } + createBucket(parameters) { + return new FillBucket(parameters); + } + queryRadius() { + return translateDistance(this.paint.get('fill-translate')); + } + queryIntersectsFeature(queryGeometry, feature, featureState, geometry, zoom, transform, pixelsToTileUnits) { + const translatedPolygon = translate$4(queryGeometry, this.paint.get('fill-translate'), this.paint.get('fill-translate-anchor'), transform.angle, pixelsToTileUnits); + return polygonIntersectsMultiPolygon(translatedPolygon, geometry); + } + isTileClipped() { + return true; + } +} + +const layout$2 = createLayout([ + { name: 'a_pos', components: 2, type: 'Int16' }, + { name: 'a_normal_ed', components: 4, type: 'Int16' }, +], 4); +const centroidAttributes = createLayout([ + { name: 'a_centroid', components: 2, type: 'Int16' } +], 4); +const { members: members$2, size: size$2, alignment: alignment$2 } = layout$2; + +var vectorTile = {}; + +'use strict'; + +var Point = pointGeometry; + +var vectortilefeature = VectorTileFeature$2; + +function VectorTileFeature$2(pbf, end, extent, keys, values) { + // Public + this.properties = {}; + this.extent = extent; + this.type = 0; + + // Private + this._pbf = pbf; + this._geometry = -1; + this._keys = keys; + this._values = values; + + pbf.readFields(readFeature, this, end); +} + +function readFeature(tag, feature, pbf) { + if (tag == 1) feature.id = pbf.readVarint(); + else if (tag == 2) readTag(pbf, feature); + else if (tag == 3) feature.type = pbf.readVarint(); + else if (tag == 4) feature._geometry = pbf.pos; +} + +function readTag(pbf, feature) { + var end = pbf.readVarint() + pbf.pos; + + while (pbf.pos < end) { + var key = feature._keys[pbf.readVarint()], + value = feature._values[pbf.readVarint()]; + feature.properties[key] = value; + } +} + +VectorTileFeature$2.types = ['Unknown', 'Point', 'LineString', 'Polygon']; + +VectorTileFeature$2.prototype.loadGeometry = function() { + var pbf = this._pbf; + pbf.pos = this._geometry; + + var end = pbf.readVarint() + pbf.pos, + cmd = 1, + length = 0, + x = 0, + y = 0, + lines = [], + line; + + while (pbf.pos < end) { + if (length <= 0) { + var cmdLen = pbf.readVarint(); + cmd = cmdLen & 0x7; + length = cmdLen >> 3; + } + + length--; + + if (cmd === 1 || cmd === 2) { + x += pbf.readSVarint(); + y += pbf.readSVarint(); + + if (cmd === 1) { // moveTo + if (line) lines.push(line); + line = []; + } + + line.push(new Point(x, y)); + + } else if (cmd === 7) { + + // Workaround for https://github.com/mapbox/mapnik-vector-tile/issues/90 + if (line) { + line.push(line[0].clone()); // closePolygon + } + + } else { + throw new Error('unknown command ' + cmd); + } + } + + if (line) lines.push(line); + + return lines; +}; + +VectorTileFeature$2.prototype.bbox = function() { + var pbf = this._pbf; + pbf.pos = this._geometry; + + var end = pbf.readVarint() + pbf.pos, + cmd = 1, + length = 0, + x = 0, + y = 0, + x1 = Infinity, + x2 = -Infinity, + y1 = Infinity, + y2 = -Infinity; + + while (pbf.pos < end) { + if (length <= 0) { + var cmdLen = pbf.readVarint(); + cmd = cmdLen & 0x7; + length = cmdLen >> 3; + } + + length--; + + if (cmd === 1 || cmd === 2) { + x += pbf.readSVarint(); + y += pbf.readSVarint(); + if (x < x1) x1 = x; + if (x > x2) x2 = x; + if (y < y1) y1 = y; + if (y > y2) y2 = y; + + } else if (cmd !== 7) { + throw new Error('unknown command ' + cmd); + } + } + + return [x1, y1, x2, y2]; +}; + +VectorTileFeature$2.prototype.toGeoJSON = function(x, y, z) { + var size = this.extent * Math.pow(2, z), + x0 = this.extent * x, + y0 = this.extent * y, + coords = this.loadGeometry(), + type = VectorTileFeature$2.types[this.type], + i, j; + + function project(line) { + for (var j = 0; j < line.length; j++) { + var p = line[j], y2 = 180 - (p.y + y0) * 360 / size; + line[j] = [ + (p.x + x0) * 360 / size - 180, + 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90 + ]; + } + } + + switch (this.type) { + case 1: + var points = []; + for (i = 0; i < coords.length; i++) { + points[i] = coords[i][0]; + } + coords = points; + project(coords); + break; + + case 2: + for (i = 0; i < coords.length; i++) { + project(coords[i]); + } + break; + + case 3: + coords = classifyRings(coords); + for (i = 0; i < coords.length; i++) { + for (j = 0; j < coords[i].length; j++) { + project(coords[i][j]); + } + } + break; + } + + if (coords.length === 1) { + coords = coords[0]; + } else { + type = 'Multi' + type; + } + + var result = { + type: "Feature", + geometry: { + type: type, + coordinates: coords + }, + properties: this.properties + }; + + if ('id' in this) { + result.id = this.id; + } + + return result; +}; + +// classifies an array of rings into polygons with outer rings and holes + +function classifyRings(rings) { + var len = rings.length; + + if (len <= 1) return [rings]; + + var polygons = [], + polygon, + ccw; + + for (var i = 0; i < len; i++) { + var area = signedArea(rings[i]); + if (area === 0) continue; + + if (ccw === undefined) ccw = area < 0; + + if (ccw === area < 0) { + if (polygon) polygons.push(polygon); + polygon = [rings[i]]; + + } else { + polygon.push(rings[i]); + } + } + if (polygon) polygons.push(polygon); + + return polygons; +} + +function signedArea(ring) { + var sum = 0; + for (var i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) { + p1 = ring[i]; + p2 = ring[j]; + sum += (p2.x - p1.x) * (p1.y + p2.y); + } + return sum; +} + +var vectortilefeature$1 = /*@__PURE__*/getDefaultExportFromCjs(vectortilefeature); + +'use strict'; + +var VectorTileFeature$1 = vectortilefeature; + +var vectortilelayer = VectorTileLayer$2; + +function VectorTileLayer$2(pbf, end) { + // Public + this.version = 1; + this.name = null; + this.extent = 4096; + this.length = 0; + + // Private + this._pbf = pbf; + this._keys = []; + this._values = []; + this._features = []; + + pbf.readFields(readLayer, this, end); + + this.length = this._features.length; +} + +function readLayer(tag, layer, pbf) { + if (tag === 15) layer.version = pbf.readVarint(); + else if (tag === 1) layer.name = pbf.readString(); + else if (tag === 5) layer.extent = pbf.readVarint(); + else if (tag === 2) layer._features.push(pbf.pos); + else if (tag === 3) layer._keys.push(pbf.readString()); + else if (tag === 4) layer._values.push(readValueMessage(pbf)); +} + +function readValueMessage(pbf) { + var value = null, + end = pbf.readVarint() + pbf.pos; + + while (pbf.pos < end) { + var tag = pbf.readVarint() >> 3; + + value = tag === 1 ? pbf.readString() : + tag === 2 ? pbf.readFloat() : + tag === 3 ? pbf.readDouble() : + tag === 4 ? pbf.readVarint64() : + tag === 5 ? pbf.readVarint() : + tag === 6 ? pbf.readSVarint() : + tag === 7 ? pbf.readBoolean() : null; + } + + return value; +} + +// return feature `i` from this layer as a `VectorTileFeature` +VectorTileLayer$2.prototype.feature = function(i) { + if (i < 0 || i >= this._features.length) throw new Error('feature index out of bounds'); + + this._pbf.pos = this._features[i]; + + var end = this._pbf.readVarint() + this._pbf.pos; + return new VectorTileFeature$1(this._pbf, end, this.extent, this._keys, this._values); +}; + +var vectortilelayer$1 = /*@__PURE__*/getDefaultExportFromCjs(vectortilelayer); + +'use strict'; + +var VectorTileLayer$1 = vectortilelayer; + +var vectortile = VectorTile$1; + +function VectorTile$1(pbf, end) { + this.layers = pbf.readFields(readTile, {}, end); +} + +function readTile(tag, layers, pbf) { + if (tag === 3) { + var layer = new VectorTileLayer$1(pbf, pbf.readVarint() + pbf.pos); + if (layer.length) layers[layer.name] = layer; + } +} + +var vectortile$1 = /*@__PURE__*/getDefaultExportFromCjs(vectortile); + +var VectorTile = vectorTile.VectorTile = vectortile; +var VectorTileFeature = vectorTile.VectorTileFeature = vectortilefeature; +var VectorTileLayer = vectorTile.VectorTileLayer = vectortilelayer; + +const vectorTileFeatureTypes$2 = vectorTile.VectorTileFeature.types; +const EARCUT_MAX_RINGS = 500; +const FACTOR = Math.pow(2, 13); +function addVertex$1(vertexArray, x, y, nx, ny, nz, t, e) { + vertexArray.emplaceBack( + // a_pos + x, y, + // a_normal_ed: 3-component normal and 1-component edgedistance + Math.floor(nx * FACTOR) * 2 + t, ny * FACTOR * 2, nz * FACTOR * 2, + // edgedistance (used for wrapping patterns around extrusion sides) + Math.round(e)); +} +class FillExtrusionBucket { + constructor(options) { + this.zoom = options.zoom; + this.overscaling = options.overscaling; + this.layers = options.layers; + this.layerIds = this.layers.map(layer => layer.id); + this.index = options.index; + this.hasPattern = false; + this.layoutVertexArray = new FillExtrusionLayoutArray(); + this.centroidVertexArray = new PosArray(); + this.indexArray = new TriangleIndexArray(); + this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom); + this.segments = new SegmentVector(); + this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id); + } + populate(features, options, canonical) { + this.features = []; + this.hasPattern = hasPattern('fill-extrusion', this.layers, options); + for (const { feature, id, index, sourceLayerIndex } of features) { + const needGeometry = this.layers[0]._featureFilter.needGeometry; + const evaluationFeature = toEvaluationFeature(feature, needGeometry); + if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) + continue; + const bucketFeature = { + id, + sourceLayerIndex, + index, + geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature), + properties: feature.properties, + type: feature.type, + patterns: {} + }; + if (this.hasPattern) { + this.features.push(addPatternDependencies('fill-extrusion', this.layers, bucketFeature, this.zoom, options)); + } + else { + this.addFeature(bucketFeature, bucketFeature.geometry, index, canonical, {}); + } + options.featureIndex.insert(feature, bucketFeature.geometry, index, sourceLayerIndex, this.index, true); + } + } + addFeatures(options, canonical, imagePositions) { + for (const feature of this.features) { + const { geometry } = feature; + this.addFeature(feature, geometry, feature.index, canonical, imagePositions); + } + } + update(states, vtLayer, imagePositions) { + if (!this.stateDependentLayers.length) + return; + this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions); + } + isEmpty() { + return this.layoutVertexArray.length === 0 && this.centroidVertexArray.length === 0; + } + uploadPending() { + return !this.uploaded || this.programConfigurations.needsUpload; + } + upload(context) { + if (!this.uploaded) { + this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, members$2); + this.centroidVertexBuffer = context.createVertexBuffer(this.centroidVertexArray, centroidAttributes.members, true); + this.indexBuffer = context.createIndexBuffer(this.indexArray); + } + this.programConfigurations.upload(context); + this.uploaded = true; + } + destroy() { + if (!this.layoutVertexBuffer) + return; + this.layoutVertexBuffer.destroy(); + this.indexBuffer.destroy(); + this.programConfigurations.destroy(); + this.segments.destroy(); + this.centroidVertexBuffer.destroy(); + } + addFeature(feature, geometry, index, canonical, imagePositions) { + const centroid = { x: 0, y: 0, vertexCount: 0 }; + for (const polygon of classifyRings$1(geometry, EARCUT_MAX_RINGS)) { + let numVertices = 0; + for (const ring of polygon) { + numVertices += ring.length; + } + let segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray); + for (const ring of polygon) { + if (ring.length === 0) { + continue; + } + if (isEntirelyOutside(ring)) { + continue; + } + let edgeDistance = 0; + for (let p = 0; p < ring.length; p++) { + const p1 = ring[p]; + if (p >= 1) { + const p2 = ring[p - 1]; + if (!isBoundaryEdge(p1, p2)) { + if (segment.vertexLength + 4 > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) { + segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray); + } + const perp = p1.sub(p2)._perp()._unit(); + const dist = p2.dist(p1); + if (edgeDistance + dist > 32768) + edgeDistance = 0; + addVertex$1(this.layoutVertexArray, p1.x, p1.y, perp.x, perp.y, 0, 0, edgeDistance); + addVertex$1(this.layoutVertexArray, p1.x, p1.y, perp.x, perp.y, 0, 1, edgeDistance); + centroid.x += 2 * p1.x; + centroid.y += 2 * p1.y; + centroid.vertexCount += 2; + edgeDistance += dist; + addVertex$1(this.layoutVertexArray, p2.x, p2.y, perp.x, perp.y, 0, 0, edgeDistance); + addVertex$1(this.layoutVertexArray, p2.x, p2.y, perp.x, perp.y, 0, 1, edgeDistance); + centroid.x += 2 * p2.x; + centroid.y += 2 * p2.y; + centroid.vertexCount += 2; + const bottomRight = segment.vertexLength; + // ┌──────┐ + // │ 0 1 │ Counter-clockwise winding order. + // │ │ Triangle 1: 0 => 2 => 1 + // │ 2 3 │ Triangle 2: 1 => 2 => 3 + // └──────┘ + this.indexArray.emplaceBack(bottomRight, bottomRight + 2, bottomRight + 1); + this.indexArray.emplaceBack(bottomRight + 1, bottomRight + 2, bottomRight + 3); + segment.vertexLength += 4; + segment.primitiveLength += 2; + } + } + } + } + if (segment.vertexLength + numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) { + segment = this.segments.prepareSegment(numVertices, this.layoutVertexArray, this.indexArray); + } + //Only triangulate and draw the area of the feature if it is a polygon + //Other feature types (e.g. LineString) do not have area, so triangulation is pointless / undefined + if (vectorTileFeatureTypes$2[feature.type] !== 'Polygon') + continue; + const flattened = []; + const holeIndices = []; + const triangleIndex = segment.vertexLength; + for (const ring of polygon) { + if (ring.length === 0) { + continue; + } + if (ring !== polygon[0]) { + holeIndices.push(flattened.length / 2); + } + for (let i = 0; i < ring.length; i++) { + const p = ring[i]; + addVertex$1(this.layoutVertexArray, p.x, p.y, 0, 0, 1, 1, 0); + centroid.x += p.x; + centroid.y += p.y; + centroid.vertexCount += 1; + flattened.push(p.x); + flattened.push(p.y); + } + } + const indices = earcut$1(flattened, holeIndices); + for (let j = 0; j < indices.length; j += 3) { + // Counter-clockwise winding order. + this.indexArray.emplaceBack(triangleIndex + indices[j], triangleIndex + indices[j + 2], triangleIndex + indices[j + 1]); + } + segment.primitiveLength += indices.length / 3; + segment.vertexLength += numVertices; + } + // remember polygon centroid to calculate elevation in GPU + for (let i = 0; i < centroid.vertexCount; i++) { + this.centroidVertexArray.emplaceBack(Math.floor(centroid.x / centroid.vertexCount), Math.floor(centroid.y / centroid.vertexCount)); + } + this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical); + } +} +register('FillExtrusionBucket', FillExtrusionBucket, { omit: ['layers', 'features'] }); +function isBoundaryEdge(p1, p2) { + return (p1.x === p2.x && (p1.x < 0 || p1.x > EXTENT)) || + (p1.y === p2.y && (p1.y < 0 || p1.y > EXTENT)); +} +function isEntirelyOutside(ring) { + return ring.every(p => p.x < 0) || + ring.every(p => p.x > EXTENT) || + ring.every(p => p.y < 0) || + ring.every(p => p.y > EXTENT); +} + +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ +let paint$4; +const getPaint$4 = () => paint$4 = paint$4 || new Properties({ + "fill-extrusion-opacity": new DataConstantProperty(v8Spec["paint_fill-extrusion"]["fill-extrusion-opacity"]), + "fill-extrusion-color": new DataDrivenProperty(v8Spec["paint_fill-extrusion"]["fill-extrusion-color"]), + "fill-extrusion-translate": new DataConstantProperty(v8Spec["paint_fill-extrusion"]["fill-extrusion-translate"]), + "fill-extrusion-translate-anchor": new DataConstantProperty(v8Spec["paint_fill-extrusion"]["fill-extrusion-translate-anchor"]), + "fill-extrusion-pattern": new CrossFadedDataDrivenProperty(v8Spec["paint_fill-extrusion"]["fill-extrusion-pattern"]), + "fill-extrusion-height": new DataDrivenProperty(v8Spec["paint_fill-extrusion"]["fill-extrusion-height"]), + "fill-extrusion-base": new DataDrivenProperty(v8Spec["paint_fill-extrusion"]["fill-extrusion-base"]), + "fill-extrusion-vertical-gradient": new DataConstantProperty(v8Spec["paint_fill-extrusion"]["fill-extrusion-vertical-gradient"]), +}); +var properties$4 = ({ get paint() { return getPaint$4(); } }); + +class Point3D extends Point$2 { +} +class FillExtrusionStyleLayer extends StyleLayer { + constructor(layer) { + super(layer, properties$4); + } + createBucket(parameters) { + return new FillExtrusionBucket(parameters); + } + queryRadius() { + return translateDistance(this.paint.get('fill-extrusion-translate')); + } + is3D() { + return true; + } + queryIntersectsFeature(queryGeometry, feature, featureState, geometry, zoom, transform, pixelsToTileUnits, pixelPosMatrix) { + const translatedPolygon = translate$4(queryGeometry, this.paint.get('fill-extrusion-translate'), this.paint.get('fill-extrusion-translate-anchor'), transform.angle, pixelsToTileUnits); + const height = this.paint.get('fill-extrusion-height').evaluate(feature, featureState); + const base = this.paint.get('fill-extrusion-base').evaluate(feature, featureState); + const projectedQueryGeometry = projectQueryGeometry(translatedPolygon, pixelPosMatrix, transform, 0); + const projected = projectExtrusion(geometry, base, height, pixelPosMatrix); + const projectedBase = projected[0]; + const projectedTop = projected[1]; + return checkIntersection(projectedBase, projectedTop, projectedQueryGeometry); + } +} +function dot(a, b) { + return a.x * b.x + a.y * b.y; +} +function getIntersectionDistance(projectedQueryGeometry, projectedFace) { + if (projectedQueryGeometry.length === 1) { + // For point queries calculate the z at which the point intersects the face + // using barycentric coordinates. + // Find the barycentric coordinates of the projected point within the first + // triangle of the face, using only the xy plane. It doesn't matter if the + // point is outside the first triangle because all the triangles in the face + // are in the same plane. + // + // Check whether points are coincident and use other points if they are. + let i = 0; + const a = projectedFace[i++]; + let b; + while (!b || a.equals(b)) { + b = projectedFace[i++]; + if (!b) + return Infinity; + } + // Loop until point `c` is not colinear with points `a` and `b`. + for (; i < projectedFace.length; i++) { + const c = projectedFace[i]; + const p = projectedQueryGeometry[0]; + const ab = b.sub(a); + const ac = c.sub(a); + const ap = p.sub(a); + const dotABAB = dot(ab, ab); + const dotABAC = dot(ab, ac); + const dotACAC = dot(ac, ac); + const dotAPAB = dot(ap, ab); + const dotAPAC = dot(ap, ac); + const denom = dotABAB * dotACAC - dotABAC * dotABAC; + const v = (dotACAC * dotAPAB - dotABAC * dotAPAC) / denom; + const w = (dotABAB * dotAPAC - dotABAC * dotAPAB) / denom; + const u = 1 - v - w; + // Use the barycentric weighting along with the original triangle z coordinates to get the point of intersection. + const distance = a.z * u + b.z * v + c.z * w; + if (isFinite(distance)) + return distance; + } + return Infinity; + } + else { + // The counts as closest is less clear when the query is a box. This + // returns the distance to the nearest point on the face, whether it is + // within the query or not. It could be more correct to return the + // distance to the closest point within the query box but this would be + // more complicated and expensive to calculate with little benefit. + let closestDistance = Infinity; + for (const p of projectedFace) { + closestDistance = Math.min(closestDistance, p.z); + } + return closestDistance; + } +} +function checkIntersection(projectedBase, projectedTop, projectedQueryGeometry) { + let closestDistance = Infinity; + if (polygonIntersectsMultiPolygon(projectedQueryGeometry, projectedTop)) { + closestDistance = getIntersectionDistance(projectedQueryGeometry, projectedTop[0]); + } + for (let r = 0; r < projectedTop.length; r++) { + const ringTop = projectedTop[r]; + const ringBase = projectedBase[r]; + for (let p = 0; p < ringTop.length - 1; p++) { + const topA = ringTop[p]; + const topB = ringTop[p + 1]; + const baseA = ringBase[p]; + const baseB = ringBase[p + 1]; + const face = [topA, topB, baseB, baseA, topA]; + if (polygonIntersectsPolygon(projectedQueryGeometry, face)) { + closestDistance = Math.min(closestDistance, getIntersectionDistance(projectedQueryGeometry, face)); + } + } + } + return closestDistance === Infinity ? false : closestDistance; +} +/* + * Project the geometry using matrix `m`. This is essentially doing + * `vec4.transformMat4([], [p.x, p.y, z, 1], m)` but the multiplication + * is inlined so that parts of the projection that are the same across + * different points can only be done once. This produced a measurable + * performance improvement. + */ +function projectExtrusion(geometry, zBase, zTop, m) { + const projectedBase = []; + const projectedTop = []; + const baseXZ = m[8] * zBase; + const baseYZ = m[9] * zBase; + const baseZZ = m[10] * zBase; + const baseWZ = m[11] * zBase; + const topXZ = m[8] * zTop; + const topYZ = m[9] * zTop; + const topZZ = m[10] * zTop; + const topWZ = m[11] * zTop; + for (const r of geometry) { + const ringBase = []; + const ringTop = []; + for (const p of r) { + const x = p.x; + const y = p.y; + const sX = m[0] * x + m[4] * y + m[12]; + const sY = m[1] * x + m[5] * y + m[13]; + const sZ = m[2] * x + m[6] * y + m[14]; + const sW = m[3] * x + m[7] * y + m[15]; + const baseX = sX + baseXZ; + const baseY = sY + baseYZ; + const baseZ = sZ + baseZZ; + const baseW = sW + baseWZ; + const topX = sX + topXZ; + const topY = sY + topYZ; + const topZ = sZ + topZZ; + const topW = sW + topWZ; + const b = new Point$2(baseX / baseW, baseY / baseW); + b.z = baseZ / baseW; + ringBase.push(b); + const t = new Point$2(topX / topW, topY / topW); + t.z = topZ / topW; + ringTop.push(t); + } + projectedBase.push(ringBase); + projectedTop.push(ringTop); + } + return [projectedBase, projectedTop]; +} +function projectQueryGeometry(queryGeometry, pixelPosMatrix, transform, z) { + const projectedQueryGeometry = []; + for (const p of queryGeometry) { + const v = [p.x, p.y, z, 1]; + transformMat4$1(v, v, pixelPosMatrix); + projectedQueryGeometry.push(new Point$2(v[0] / v[3], v[1] / v[3])); + } + return projectedQueryGeometry; +} + +const lineLayoutAttributes = createLayout([ + { name: 'a_pos_normal', components: 2, type: 'Int16' }, + { name: 'a_data', components: 4, type: 'Uint8' } +], 4); +const { members: members$1, size: size$1, alignment: alignment$1 } = lineLayoutAttributes; + +const lineLayoutAttributesExt = createLayout([ + { name: 'a_uv_x', components: 1, type: 'Float32' }, + { name: 'a_split_index', components: 1, type: 'Float32' }, +]); +const { members, size, alignment } = lineLayoutAttributesExt; + +const vectorTileFeatureTypes$1 = vectorTile.VectorTileFeature.types; +// NOTE ON EXTRUDE SCALE: +// scale the extrusion vector so that the normal length is this value. +// contains the "texture" normals (-1..1). this is distinct from the extrude +// normals for line joins, because the x-value remains 0 for the texture +// normal array, while the extrude normal actually moves the vertex to create +// the acute/bevelled line join. +const EXTRUDE_SCALE = 63; +/* + * Sharp corners cause dashed lines to tilt because the distance along the line + * is the same at both the inner and outer corners. To improve the appearance of + * dashed lines we add extra points near sharp corners so that a smaller part + * of the line is tilted. + * + * COS_HALF_SHARP_CORNER controls how sharp a corner has to be for us to add an + * extra vertex. The default is 75 degrees. + * + * The newly created vertices are placed SHARP_CORNER_OFFSET pixels from the corner. + */ +const COS_HALF_SHARP_CORNER = Math.cos(75 / 2 * (Math.PI / 180)); +const SHARP_CORNER_OFFSET = 15; +// Angle per triangle for approximating round line joins. +const DEG_PER_TRIANGLE = 20; +// The number of bits that is used to store the line distance in the buffer. +const LINE_DISTANCE_BUFFER_BITS = 15; +// We don't have enough bits for the line distance as we'd like to have, so +// use this value to scale the line distance (in tile units) down to a smaller +// value. This lets us store longer distances while sacrificing precision. +const LINE_DISTANCE_SCALE = 1 / 2; +// The maximum line distance, in tile units, that fits in the buffer. +const MAX_LINE_DISTANCE = Math.pow(2, LINE_DISTANCE_BUFFER_BITS - 1) / LINE_DISTANCE_SCALE; +/** + * @internal + * Line bucket class + */ +class LineBucket { + constructor(options) { + this.zoom = options.zoom; + this.overscaling = options.overscaling; + this.layers = options.layers; + this.layerIds = this.layers.map(layer => layer.id); + this.index = options.index; + this.hasPattern = false; + this.patternFeatures = []; + this.lineClipsArray = []; + this.gradients = {}; + this.layers.forEach(layer => { + this.gradients[layer.id] = {}; + }); + this.layoutVertexArray = new LineLayoutArray(); + this.layoutVertexArray2 = new LineExtLayoutArray(); + this.indexArray = new TriangleIndexArray(); + this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom); + this.segments = new SegmentVector(); + this.maxLineLength = 0; + this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id); + } + populate(features, options, canonical) { + this.hasPattern = hasPattern('line', this.layers, options); + const lineSortKey = this.layers[0].layout.get('line-sort-key'); + const sortFeaturesByKey = !lineSortKey.isConstant(); + const bucketFeatures = []; + for (const { feature, id, index, sourceLayerIndex } of features) { + const needGeometry = this.layers[0]._featureFilter.needGeometry; + const evaluationFeature = toEvaluationFeature(feature, needGeometry); + if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) + continue; + const sortKey = sortFeaturesByKey ? + lineSortKey.evaluate(evaluationFeature, {}, canonical) : + undefined; + const bucketFeature = { + id, + properties: feature.properties, + type: feature.type, + sourceLayerIndex, + index, + geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature), + patterns: {}, + sortKey + }; + bucketFeatures.push(bucketFeature); + } + if (sortFeaturesByKey) { + bucketFeatures.sort((a, b) => { + return (a.sortKey) - (b.sortKey); + }); + } + for (const bucketFeature of bucketFeatures) { + const { geometry, index, sourceLayerIndex } = bucketFeature; + if (this.hasPattern) { + const patternBucketFeature = addPatternDependencies('line', this.layers, bucketFeature, this.zoom, options); + // pattern features are added only once the pattern is loaded into the image atlas + // so are stored during populate until later updated with positions by tile worker in addFeatures + this.patternFeatures.push(patternBucketFeature); + } + else { + this.addFeature(bucketFeature, geometry, index, canonical, {}); + } + const feature = features[index].feature; + options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index); + } + } + update(states, vtLayer, imagePositions) { + if (!this.stateDependentLayers.length) + return; + this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions); + } + addFeatures(options, canonical, imagePositions) { + for (const feature of this.patternFeatures) { + this.addFeature(feature, feature.geometry, feature.index, canonical, imagePositions); + } + } + isEmpty() { + return this.layoutVertexArray.length === 0; + } + uploadPending() { + return !this.uploaded || this.programConfigurations.needsUpload; + } + upload(context) { + if (!this.uploaded) { + if (this.layoutVertexArray2.length !== 0) { + this.layoutVertexBuffer2 = context.createVertexBuffer(this.layoutVertexArray2, members); + } + this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, members$1); + this.indexBuffer = context.createIndexBuffer(this.indexArray); + } + this.programConfigurations.upload(context); + this.uploaded = true; + } + destroy() { + if (!this.layoutVertexBuffer) + return; + this.layoutVertexBuffer.destroy(); + this.indexBuffer.destroy(); + this.programConfigurations.destroy(); + this.segments.destroy(); + } + lineFeatureClips(feature) { + if (!!feature.properties && Object.prototype.hasOwnProperty.call(feature.properties, 'mapbox_clip_start') && Object.prototype.hasOwnProperty.call(feature.properties, 'mapbox_clip_end')) { + const start = +feature.properties['mapbox_clip_start']; + const end = +feature.properties['mapbox_clip_end']; + return { start, end }; + } + } + addFeature(feature, geometry, index, canonical, imagePositions) { + const layout = this.layers[0].layout; + const join = layout.get('line-join').evaluate(feature, {}); + const cap = layout.get('line-cap'); + const miterLimit = layout.get('line-miter-limit'); + const roundLimit = layout.get('line-round-limit'); + this.lineClips = this.lineFeatureClips(feature); + for (const line of geometry) { + this.addLine(line, feature, join, cap, miterLimit, roundLimit); + } + this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical); + } + addLine(vertices, feature, join, cap, miterLimit, roundLimit) { + this.distance = 0; + this.scaledDistance = 0; + this.totalDistance = 0; + if (this.lineClips) { + this.lineClipsArray.push(this.lineClips); + // Calculate the total distance, in tile units, of this tiled line feature + for (let i = 0; i < vertices.length - 1; i++) { + this.totalDistance += vertices[i].dist(vertices[i + 1]); + } + this.updateScaledDistance(); + this.maxLineLength = Math.max(this.maxLineLength, this.totalDistance); + } + const isPolygon = vectorTileFeatureTypes$1[feature.type] === 'Polygon'; + // If the line has duplicate vertices at the ends, adjust start/length to remove them. + let len = vertices.length; + while (len >= 2 && vertices[len - 1].equals(vertices[len - 2])) { + len--; + } + let first = 0; + while (first < len - 1 && vertices[first].equals(vertices[first + 1])) { + first++; + } + // Ignore invalid geometry. + if (len < (isPolygon ? 3 : 2)) + return; + if (join === 'bevel') + miterLimit = 1.05; + const sharpCornerOffset = this.overscaling <= 16 ? + SHARP_CORNER_OFFSET * EXTENT / (512 * this.overscaling) : + 0; + // we could be more precise, but it would only save a negligible amount of space + const segment = this.segments.prepareSegment(len * 10, this.layoutVertexArray, this.indexArray); + let currentVertex; + let prevVertex; + let nextVertex; + let prevNormal; + let nextNormal; + // the last two vertices added + this.e1 = this.e2 = -1; + if (isPolygon) { + currentVertex = vertices[len - 2]; + nextNormal = vertices[first].sub(currentVertex)._unit()._perp(); + } + for (let i = first; i < len; i++) { + nextVertex = i === len - 1 ? + (isPolygon ? vertices[first + 1] : undefined) : // if it's a polygon, treat the last vertex like the first + vertices[i + 1]; // just the next vertex + // if two consecutive vertices exist, skip the current one + if (nextVertex && vertices[i].equals(nextVertex)) + continue; + if (nextNormal) + prevNormal = nextNormal; + if (currentVertex) + prevVertex = currentVertex; + currentVertex = vertices[i]; + // Calculate the normal towards the next vertex in this line. In case + // there is no next vertex, pretend that the line is continuing straight, + // meaning that we are just using the previous normal. + nextNormal = nextVertex ? nextVertex.sub(currentVertex)._unit()._perp() : prevNormal; + // If we still don't have a previous normal, this is the beginning of a + // non-closed line, so we're doing a straight "join". + prevNormal = prevNormal || nextNormal; + // Determine the normal of the join extrusion. It is the angle bisector + // of the segments between the previous line and the next line. + // In the case of 180° angles, the prev and next normals cancel each other out: + // prevNormal + nextNormal = (0, 0), its magnitude is 0, so the unit vector would be + // undefined. In that case, we're keeping the joinNormal at (0, 0), so that the cosHalfAngle + // below will also become 0 and miterLength will become Infinity. + let joinNormal = prevNormal.add(nextNormal); + if (joinNormal.x !== 0 || joinNormal.y !== 0) { + joinNormal._unit(); + } + /* joinNormal prevNormal + * ↖ ↑ + * .________. prevVertex + * | + * nextNormal ← | currentVertex + * | + * nextVertex ! + * + */ + // calculate cosines of the angle (and its half) using dot product + const cosAngle = prevNormal.x * nextNormal.x + prevNormal.y * nextNormal.y; + const cosHalfAngle = joinNormal.x * nextNormal.x + joinNormal.y * nextNormal.y; + // Calculate the length of the miter (the ratio of the miter to the width) + // as the inverse of cosine of the angle between next and join normals + const miterLength = cosHalfAngle !== 0 ? 1 / cosHalfAngle : Infinity; + // approximate angle from cosine + const approxAngle = 2 * Math.sqrt(2 - 2 * cosHalfAngle); + const isSharpCorner = cosHalfAngle < COS_HALF_SHARP_CORNER && prevVertex && nextVertex; + const lineTurnsLeft = prevNormal.x * nextNormal.y - prevNormal.y * nextNormal.x > 0; + if (isSharpCorner && i > first) { + const prevSegmentLength = currentVertex.dist(prevVertex); + if (prevSegmentLength > 2 * sharpCornerOffset) { + const newPrevVertex = currentVertex.sub(currentVertex.sub(prevVertex)._mult(sharpCornerOffset / prevSegmentLength)._round()); + this.updateDistance(prevVertex, newPrevVertex); + this.addCurrentVertex(newPrevVertex, prevNormal, 0, 0, segment); + prevVertex = newPrevVertex; + } + } + // The join if a middle vertex, otherwise the cap. + const middleVertex = prevVertex && nextVertex; + let currentJoin = middleVertex ? join : isPolygon ? 'butt' : cap; + if (middleVertex && currentJoin === 'round') { + if (miterLength < roundLimit) { + currentJoin = 'miter'; + } + else if (miterLength <= 2) { + currentJoin = 'fakeround'; + } + } + if (currentJoin === 'miter' && miterLength > miterLimit) { + currentJoin = 'bevel'; + } + if (currentJoin === 'bevel') { + // The maximum extrude length is 128 / 63 = 2 times the width of the line + // so if miterLength >= 2 we need to draw a different type of bevel here. + if (miterLength > 2) + currentJoin = 'flipbevel'; + // If the miterLength is really small and the line bevel wouldn't be visible, + // just draw a miter join to save a triangle. + if (miterLength < miterLimit) + currentJoin = 'miter'; + } + // Calculate how far along the line the currentVertex is + if (prevVertex) + this.updateDistance(prevVertex, currentVertex); + if (currentJoin === 'miter') { + joinNormal._mult(miterLength); + this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment); + } + else if (currentJoin === 'flipbevel') { + // miter is too big, flip the direction to make a beveled join + if (miterLength > 100) { + // Almost parallel lines + joinNormal = nextNormal.mult(-1); + } + else { + const bevelLength = miterLength * prevNormal.add(nextNormal).mag() / prevNormal.sub(nextNormal).mag(); + joinNormal._perp()._mult(bevelLength * (lineTurnsLeft ? -1 : 1)); + } + this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment); + this.addCurrentVertex(currentVertex, joinNormal.mult(-1), 0, 0, segment); + } + else if (currentJoin === 'bevel' || currentJoin === 'fakeround') { + const offset = -Math.sqrt(miterLength * miterLength - 1); + const offsetA = lineTurnsLeft ? offset : 0; + const offsetB = lineTurnsLeft ? 0 : offset; + // Close previous segment with a bevel + if (prevVertex) { + this.addCurrentVertex(currentVertex, prevNormal, offsetA, offsetB, segment); + } + if (currentJoin === 'fakeround') { + // The join angle is sharp enough that a round join would be visible. + // Bevel joins fill the gap between segments with a single pie slice triangle. + // Create a round join by adding multiple pie slices. The join isn't actually round, but + // it looks like it is at the sizes we render lines at. + // pick the number of triangles for approximating round join by based on the angle between normals + const n = Math.round((approxAngle * 180 / Math.PI) / DEG_PER_TRIANGLE); + for (let m = 1; m < n; m++) { + let t = m / n; + if (t !== 0.5) { + // approximate spherical interpolation https://observablehq.com/@mourner/approximating-geometric-slerp + const t2 = t - 0.5; + const A = 1.0904 + cosAngle * (-3.2452 + cosAngle * (3.55645 - cosAngle * 1.43519)); + const B = 0.848013 + cosAngle * (-1.06021 + cosAngle * 0.215638); + t = t + t * t2 * (t - 1) * (A * t2 * t2 + B); + } + const extrude = nextNormal.sub(prevNormal)._mult(t)._add(prevNormal)._unit()._mult(lineTurnsLeft ? -1 : 1); + this.addHalfVertex(currentVertex, extrude.x, extrude.y, false, lineTurnsLeft, 0, segment); + } + } + if (nextVertex) { + // Start next segment + this.addCurrentVertex(currentVertex, nextNormal, -offsetA, -offsetB, segment); + } + } + else if (currentJoin === 'butt') { + this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment); // butt cap + } + else if (currentJoin === 'square') { + const offset = prevVertex ? 1 : -1; // closing or starting square cap + this.addCurrentVertex(currentVertex, joinNormal, offset, offset, segment); + } + else if (currentJoin === 'round') { + if (prevVertex) { + // Close previous segment with butt + this.addCurrentVertex(currentVertex, prevNormal, 0, 0, segment); + // Add round cap or linejoin at end of segment + this.addCurrentVertex(currentVertex, prevNormal, 1, 1, segment, true); + } + if (nextVertex) { + // Add round cap before first segment + this.addCurrentVertex(currentVertex, nextNormal, -1, -1, segment, true); + // Start next segment with a butt + this.addCurrentVertex(currentVertex, nextNormal, 0, 0, segment); + } + } + if (isSharpCorner && i < len - 1) { + const nextSegmentLength = currentVertex.dist(nextVertex); + if (nextSegmentLength > 2 * sharpCornerOffset) { + const newCurrentVertex = currentVertex.add(nextVertex.sub(currentVertex)._mult(sharpCornerOffset / nextSegmentLength)._round()); + this.updateDistance(currentVertex, newCurrentVertex); + this.addCurrentVertex(newCurrentVertex, nextNormal, 0, 0, segment); + currentVertex = newCurrentVertex; + } + } + } + } + /** + * Add two vertices to the buffers. + * + * @param p - the line vertex to add buffer vertices for + * @param normal - vertex normal + * @param endLeft - extrude to shift the left vertex along the line + * @param endRight - extrude to shift the left vertex along the line + * @param segment - the segment object to add the vertex to + * @param round - whether this is a round cap + */ + addCurrentVertex(p, normal, endLeft, endRight, segment, round = false) { + // left and right extrude vectors, perpendicularly shifted by endLeft/endRight + const leftX = normal.x + normal.y * endLeft; + const leftY = normal.y - normal.x * endLeft; + const rightX = -normal.x + normal.y * endRight; + const rightY = -normal.y - normal.x * endRight; + this.addHalfVertex(p, leftX, leftY, round, false, endLeft, segment); + this.addHalfVertex(p, rightX, rightY, round, true, -endRight, segment); + // There is a maximum "distance along the line" that we can store in the buffers. + // When we get close to the distance, reset it to zero and add the vertex again with + // a distance of zero. The max distance is determined by the number of bits we allocate + // to `linesofar`. + if (this.distance > MAX_LINE_DISTANCE / 2 && this.totalDistance === 0) { + this.distance = 0; + this.updateScaledDistance(); + this.addCurrentVertex(p, normal, endLeft, endRight, segment, round); + } + } + addHalfVertex({ x, y }, extrudeX, extrudeY, round, up, dir, segment) { + const totalDistance = this.lineClips ? this.scaledDistance * (MAX_LINE_DISTANCE - 1) : this.scaledDistance; + // scale down so that we can store longer distances while sacrificing precision. + const linesofarScaled = totalDistance * LINE_DISTANCE_SCALE; + this.layoutVertexArray.emplaceBack( + // a_pos_normal + // Encode round/up the least significant bits + (x << 1) + (round ? 1 : 0), (y << 1) + (up ? 1 : 0), + // a_data + // add 128 to store a byte in an unsigned byte + Math.round(EXTRUDE_SCALE * extrudeX) + 128, Math.round(EXTRUDE_SCALE * extrudeY) + 128, + // Encode the -1/0/1 direction value into the first two bits of .z of a_data. + // Combine it with the lower 6 bits of `linesofarScaled` (shifted by 2 bits to make + // room for the direction value). The upper 8 bits of `linesofarScaled` are placed in + // the `w` component. + ((dir === 0 ? 0 : (dir < 0 ? -1 : 1)) + 1) | ((linesofarScaled & 0x3F) << 2), linesofarScaled >> 6); + // Constructs a second vertex buffer with higher precision line progress + if (this.lineClips) { + const progressRealigned = this.scaledDistance - this.lineClips.start; + const endClipRealigned = this.lineClips.end - this.lineClips.start; + const uvX = progressRealigned / endClipRealigned; + this.layoutVertexArray2.emplaceBack(uvX, this.lineClipsArray.length); + } + const e = segment.vertexLength++; + if (this.e1 >= 0 && this.e2 >= 0) { + this.indexArray.emplaceBack(this.e1, this.e2, e); + segment.primitiveLength++; + } + if (up) { + this.e2 = e; + } + else { + this.e1 = e; + } + } + updateScaledDistance() { + // Knowing the ratio of the full linestring covered by this tiled feature, as well + // as the total distance (in tile units) of this tiled feature, and the distance + // (in tile units) of the current vertex, we can determine the relative distance + // of this vertex along the full linestring feature and scale it to [0, 2^15) + this.scaledDistance = this.lineClips ? + this.lineClips.start + (this.lineClips.end - this.lineClips.start) * this.distance / this.totalDistance : + this.distance; + } + updateDistance(prev, next) { + this.distance += prev.dist(next); + this.updateScaledDistance(); + } +} +register('LineBucket', LineBucket, { omit: ['layers', 'patternFeatures'] }); + +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ +let layout$1; +const getLayout$1 = () => layout$1 = layout$1 || new Properties({ + "line-cap": new DataConstantProperty(v8Spec["layout_line"]["line-cap"]), + "line-join": new DataDrivenProperty(v8Spec["layout_line"]["line-join"]), + "line-miter-limit": new DataConstantProperty(v8Spec["layout_line"]["line-miter-limit"]), + "line-round-limit": new DataConstantProperty(v8Spec["layout_line"]["line-round-limit"]), + "line-sort-key": new DataDrivenProperty(v8Spec["layout_line"]["line-sort-key"]), +}); +let paint$3; +const getPaint$3 = () => paint$3 = paint$3 || new Properties({ + "line-opacity": new DataDrivenProperty(v8Spec["paint_line"]["line-opacity"]), + "line-color": new DataDrivenProperty(v8Spec["paint_line"]["line-color"]), + "line-translate": new DataConstantProperty(v8Spec["paint_line"]["line-translate"]), + "line-translate-anchor": new DataConstantProperty(v8Spec["paint_line"]["line-translate-anchor"]), + "line-width": new DataDrivenProperty(v8Spec["paint_line"]["line-width"]), + "line-gap-width": new DataDrivenProperty(v8Spec["paint_line"]["line-gap-width"]), + "line-offset": new DataDrivenProperty(v8Spec["paint_line"]["line-offset"]), + "line-blur": new DataDrivenProperty(v8Spec["paint_line"]["line-blur"]), + "line-dasharray": new CrossFadedProperty(v8Spec["paint_line"]["line-dasharray"]), + "line-pattern": new CrossFadedDataDrivenProperty(v8Spec["paint_line"]["line-pattern"]), + "line-gradient": new ColorRampProperty(v8Spec["paint_line"]["line-gradient"]), +}); +var properties$3 = ({ get paint() { return getPaint$3(); }, get layout() { return getLayout$1(); } }); + +class LineFloorwidthProperty extends DataDrivenProperty { + possiblyEvaluate(value, parameters) { + parameters = new EvaluationParameters(Math.floor(parameters.zoom), { + now: parameters.now, + fadeDuration: parameters.fadeDuration, + zoomHistory: parameters.zoomHistory, + transition: parameters.transition + }); + return super.possiblyEvaluate(value, parameters); + } + evaluate(value, globals, feature, featureState) { + globals = extend({}, globals, { zoom: Math.floor(globals.zoom) }); + return super.evaluate(value, globals, feature, featureState); + } +} +let lineFloorwidthProperty; +class LineStyleLayer extends StyleLayer { + constructor(layer) { + super(layer, properties$3); + this.gradientVersion = 0; + if (!lineFloorwidthProperty) { + lineFloorwidthProperty = + new LineFloorwidthProperty(properties$3.paint.properties['line-width'].specification); + lineFloorwidthProperty.useIntegerZoom = true; + } + } + _handleSpecialPaintPropertyUpdate(name) { + if (name === 'line-gradient') { + const expression = this.gradientExpression(); + if (isZoomExpression(expression)) { + this.stepInterpolant = expression._styleExpression.expression instanceof Step; + } + else { + this.stepInterpolant = false; + } + this.gradientVersion = (this.gradientVersion + 1) % Number.MAX_SAFE_INTEGER; + } + } + gradientExpression() { + return this._transitionablePaint._values['line-gradient'].value.expression; + } + recalculate(parameters, availableImages) { + super.recalculate(parameters, availableImages); + this.paint._values['line-floorwidth'] = + lineFloorwidthProperty.possiblyEvaluate(this._transitioningPaint._values['line-width'].value, parameters); + } + createBucket(parameters) { + return new LineBucket(parameters); + } + queryRadius(bucket) { + const lineBucket = bucket; + const width = getLineWidth(getMaximumPaintValue('line-width', this, lineBucket), getMaximumPaintValue('line-gap-width', this, lineBucket)); + const offset = getMaximumPaintValue('line-offset', this, lineBucket); + return width / 2 + Math.abs(offset) + translateDistance(this.paint.get('line-translate')); + } + queryIntersectsFeature(queryGeometry, feature, featureState, geometry, zoom, transform, pixelsToTileUnits) { + const translatedPolygon = translate$4(queryGeometry, this.paint.get('line-translate'), this.paint.get('line-translate-anchor'), transform.angle, pixelsToTileUnits); + const halfWidth = pixelsToTileUnits / 2 * getLineWidth(this.paint.get('line-width').evaluate(feature, featureState), this.paint.get('line-gap-width').evaluate(feature, featureState)); + const lineOffset = this.paint.get('line-offset').evaluate(feature, featureState); + if (lineOffset) { + geometry = offsetLine(geometry, lineOffset * pixelsToTileUnits); + } + return polygonIntersectsBufferedMultiLine(translatedPolygon, geometry, halfWidth); + } + isTileClipped() { + return true; + } +} +function getLineWidth(lineWidth, lineGapWidth) { + if (lineGapWidth > 0) { + return lineGapWidth + 2 * lineWidth; + } + else { + return lineWidth; + } +} + +const symbolLayoutAttributes = createLayout([ + { name: 'a_pos_offset', components: 4, type: 'Int16' }, + { name: 'a_data', components: 4, type: 'Uint16' }, + { name: 'a_pixeloffset', components: 4, type: 'Int16' } +], 4); +const dynamicLayoutAttributes = createLayout([ + { name: 'a_projected_pos', components: 3, type: 'Float32' } +], 4); +const placementOpacityAttributes = createLayout([ + { name: 'a_fade_opacity', components: 1, type: 'Uint32' } +], 4); +const collisionVertexAttributes = createLayout([ + { name: 'a_placed', components: 2, type: 'Uint8' }, + { name: 'a_shift', components: 2, type: 'Float32' } +]); +const collisionBox = createLayout([ + // the box is centered around the anchor point + { type: 'Int16', name: 'anchorPointX' }, + { type: 'Int16', name: 'anchorPointY' }, + // distances to the edges from the anchor + { type: 'Int16', name: 'x1' }, + { type: 'Int16', name: 'y1' }, + { type: 'Int16', name: 'x2' }, + { type: 'Int16', name: 'y2' }, + // the index of the feature in the original vectortile + { type: 'Uint32', name: 'featureIndex' }, + // the source layer the feature appears in + { type: 'Uint16', name: 'sourceLayerIndex' }, + // the bucket the feature appears in + { type: 'Uint16', name: 'bucketIndex' }, +]); +const collisionBoxLayout = createLayout([ + { name: 'a_pos', components: 2, type: 'Int16' }, + { name: 'a_anchor_pos', components: 2, type: 'Int16' }, + { name: 'a_extrude', components: 2, type: 'Int16' } +], 4); +const collisionCircleLayout = createLayout([ + { name: 'a_pos', components: 2, type: 'Float32' }, + { name: 'a_radius', components: 1, type: 'Float32' }, + { name: 'a_flags', components: 2, type: 'Int16' } +], 4); +const quadTriangle = createLayout([ + { name: 'triangle', components: 3, type: 'Uint16' }, +]); +const placement = createLayout([ + { type: 'Int16', name: 'anchorX' }, + { type: 'Int16', name: 'anchorY' }, + { type: 'Uint16', name: 'glyphStartIndex' }, + { type: 'Uint16', name: 'numGlyphs' }, + { type: 'Uint32', name: 'vertexStartIndex' }, + { type: 'Uint32', name: 'lineStartIndex' }, + { type: 'Uint32', name: 'lineLength' }, + { type: 'Uint16', name: 'segment' }, + { type: 'Uint16', name: 'lowerSize' }, + { type: 'Uint16', name: 'upperSize' }, + { type: 'Float32', name: 'lineOffsetX' }, + { type: 'Float32', name: 'lineOffsetY' }, + { type: 'Uint8', name: 'writingMode' }, + { type: 'Uint8', name: 'placedOrientation' }, + { type: 'Uint8', name: 'hidden' }, + { type: 'Uint32', name: 'crossTileID' }, + { type: 'Int16', name: 'associatedIconIndex' } +]); +const symbolInstance = createLayout([ + { type: 'Int16', name: 'anchorX' }, + { type: 'Int16', name: 'anchorY' }, + { type: 'Int16', name: 'rightJustifiedTextSymbolIndex' }, + { type: 'Int16', name: 'centerJustifiedTextSymbolIndex' }, + { type: 'Int16', name: 'leftJustifiedTextSymbolIndex' }, + { type: 'Int16', name: 'verticalPlacedTextSymbolIndex' }, + { type: 'Int16', name: 'placedIconSymbolIndex' }, + { type: 'Int16', name: 'verticalPlacedIconSymbolIndex' }, + { type: 'Uint16', name: 'key' }, + { type: 'Uint16', name: 'textBoxStartIndex' }, + { type: 'Uint16', name: 'textBoxEndIndex' }, + { type: 'Uint16', name: 'verticalTextBoxStartIndex' }, + { type: 'Uint16', name: 'verticalTextBoxEndIndex' }, + { type: 'Uint16', name: 'iconBoxStartIndex' }, + { type: 'Uint16', name: 'iconBoxEndIndex' }, + { type: 'Uint16', name: 'verticalIconBoxStartIndex' }, + { type: 'Uint16', name: 'verticalIconBoxEndIndex' }, + { type: 'Uint16', name: 'featureIndex' }, + { type: 'Uint16', name: 'numHorizontalGlyphVertices' }, + { type: 'Uint16', name: 'numVerticalGlyphVertices' }, + { type: 'Uint16', name: 'numIconVertices' }, + { type: 'Uint16', name: 'numVerticalIconVertices' }, + { type: 'Uint16', name: 'useRuntimeCollisionCircles' }, + { type: 'Uint32', name: 'crossTileID' }, + { type: 'Float32', name: 'textBoxScale' }, + { type: 'Float32', name: 'collisionCircleDiameter' }, + { type: 'Uint16', name: 'textAnchorOffsetStartIndex' }, + { type: 'Uint16', name: 'textAnchorOffsetEndIndex' } +]); +const glyphOffset = createLayout([ + { type: 'Float32', name: 'offsetX' } +]); +const lineVertex = createLayout([ + { type: 'Int16', name: 'x' }, + { type: 'Int16', name: 'y' }, + { type: 'Int16', name: 'tileUnitDistanceFromAnchor' } +]); +const textAnchorOffset = createLayout([ + { type: 'Uint16', name: 'textAnchor' }, + { type: 'Float32', components: 2, name: 'textOffset' } +]); + +function transformTextInternal(text, layer, feature) { + const transform = layer.layout.get('text-transform').evaluate(feature, {}); + if (transform === 'uppercase') { + text = text.toLocaleUpperCase(); + } + else if (transform === 'lowercase') { + text = text.toLocaleLowerCase(); + } + if (plugin.applyArabicShaping) { + text = plugin.applyArabicShaping(text); + } + return text; +} +function transformText(text, layer, feature) { + text.sections.forEach(section => { + section.text = transformTextInternal(section.text, layer, feature); + }); + return text; +} + +function mergeLines(features) { + const leftIndex = {}; + const rightIndex = {}; + const mergedFeatures = []; + let mergedIndex = 0; + function add(k) { + mergedFeatures.push(features[k]); + mergedIndex++; + } + function mergeFromRight(leftKey, rightKey, geom) { + const i = rightIndex[leftKey]; + delete rightIndex[leftKey]; + rightIndex[rightKey] = i; + mergedFeatures[i].geometry[0].pop(); + mergedFeatures[i].geometry[0] = mergedFeatures[i].geometry[0].concat(geom[0]); + return i; + } + function mergeFromLeft(leftKey, rightKey, geom) { + const i = leftIndex[rightKey]; + delete leftIndex[rightKey]; + leftIndex[leftKey] = i; + mergedFeatures[i].geometry[0].shift(); + mergedFeatures[i].geometry[0] = geom[0].concat(mergedFeatures[i].geometry[0]); + return i; + } + function getKey(text, geom, onRight) { + const point = onRight ? geom[0][geom[0].length - 1] : geom[0][0]; + return `${text}:${point.x}:${point.y}`; + } + for (let k = 0; k < features.length; k++) { + const feature = features[k]; + const geom = feature.geometry; + const text = feature.text ? feature.text.toString() : null; + if (!text) { + add(k); + continue; + } + const leftKey = getKey(text, geom), rightKey = getKey(text, geom, true); + if ((leftKey in rightIndex) && (rightKey in leftIndex) && (rightIndex[leftKey] !== leftIndex[rightKey])) { + // found lines with the same text adjacent to both ends of the current line, merge all three + const j = mergeFromLeft(leftKey, rightKey, geom); + const i = mergeFromRight(leftKey, rightKey, mergedFeatures[j].geometry); + delete leftIndex[leftKey]; + delete rightIndex[rightKey]; + rightIndex[getKey(text, mergedFeatures[i].geometry, true)] = i; + mergedFeatures[j].geometry = null; + } + else if (leftKey in rightIndex) { + // found mergeable line adjacent to the start of the current line, merge + mergeFromRight(leftKey, rightKey, geom); + } + else if (rightKey in leftIndex) { + // found mergeable line adjacent to the end of the current line, merge + mergeFromLeft(leftKey, rightKey, geom); + } + else { + // no adjacent lines, add as a new item + add(k); + leftIndex[leftKey] = mergedIndex - 1; + rightIndex[rightKey] = mergedIndex - 1; + } + } + return mergedFeatures.filter((f) => f.geometry); +} + +const verticalizedCharacterMap = { + '!': '︕', + '#': '#', + '$': '$', + '%': '%', + '&': '&', + '(': '︵', + ')': '︶', + '*': '*', + '+': '+', + ',': '︐', + '-': '︲', + '.': '・', + '/': '/', + ':': '︓', + ';': '︔', + '<': '︿', + '=': '=', + '>': '﹀', + '?': '︖', + '@': '@', + '[': '﹇', + '\\': '\', + ']': '﹈', + '^': '^', + '_': '︳', + '`': '`', + '{': '︷', + '|': '―', + '}': '︸', + '~': '~', + '¢': '¢', + '£': '£', + '¥': '¥', + '¦': '¦', + '¬': '¬', + '¯': ' ̄', + '–': '︲', + '—': '︱', + '‘': '﹃', + '’': '﹄', + '“': '﹁', + '”': '﹂', + '…': '︙', + '‧': '・', + '₩': '₩', + '、': '︑', + '。': '︒', + '〈': '︿', + '〉': '﹀', + '《': '︽', + '》': '︾', + '「': '﹁', + '」': '﹂', + '『': '﹃', + '』': '﹄', + '【': '︻', + '】': '︼', + '〔': '︹', + '〕': '︺', + '〖': '︗', + '〗': '︘', + '!': '︕', + '(': '︵', + ')': '︶', + ',': '︐', + '-': '︲', + '.': '・', + ':': '︓', + ';': '︔', + '<': '︿', + '>': '﹀', + '?': '︖', + '[': '﹇', + ']': '﹈', + '_': '︳', + '{': '︷', + '|': '―', + '}': '︸', + '⦅': '︵', + '⦆': '︶', + '。': '︒', + '「': '﹁', + '」': '﹂' +}; +function verticalizePunctuation(input) { + let output = ''; + for (let i = 0; i < input.length; i++) { + const nextCharCode = input.charCodeAt(i + 1) || null; + const prevCharCode = input.charCodeAt(i - 1) || null; + const canReplacePunctuation = ((!nextCharCode || !charHasRotatedVerticalOrientation(nextCharCode) || verticalizedCharacterMap[input[i + 1]]) && + (!prevCharCode || !charHasRotatedVerticalOrientation(prevCharCode) || verticalizedCharacterMap[input[i - 1]])); + if (canReplacePunctuation && verticalizedCharacterMap[input[i]]) { + output += verticalizedCharacterMap[input[i]]; + } + else { + output += input[i]; + } + } + return output; +} + +// ONE_EM constant used to go between "em" units used in style spec and "points" used internally for layout +var ONE_EM = 24; + +var ieee754$1 = {}; + +/*! ieee754. BSD-3-Clause License. Feross Aboukhadijeh */ + +var read = ieee754$1.read = function (buffer, offset, isLE, mLen, nBytes) { + var e, m; + var eLen = (nBytes * 8) - mLen - 1; + var eMax = (1 << eLen) - 1; + var eBias = eMax >> 1; + var nBits = -7; + var i = isLE ? (nBytes - 1) : 0; + var d = isLE ? -1 : 1; + var s = buffer[offset + i]; + + i += d; + + e = s & ((1 << (-nBits)) - 1); + s >>= (-nBits); + nBits += eLen; + for (; nBits > 0; e = (e * 256) + buffer[offset + i], i += d, nBits -= 8) {} + + m = e & ((1 << (-nBits)) - 1); + e >>= (-nBits); + nBits += mLen; + for (; nBits > 0; m = (m * 256) + buffer[offset + i], i += d, nBits -= 8) {} + + if (e === 0) { + e = 1 - eBias; + } else if (e === eMax) { + return m ? NaN : ((s ? -1 : 1) * Infinity) + } else { + m = m + Math.pow(2, mLen); + e = e - eBias; + } + return (s ? -1 : 1) * m * Math.pow(2, e - mLen) +}; + +var write = ieee754$1.write = function (buffer, value, offset, isLE, mLen, nBytes) { + var e, m, c; + var eLen = (nBytes * 8) - mLen - 1; + var eMax = (1 << eLen) - 1; + var eBias = eMax >> 1; + var rt = (mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0); + var i = isLE ? 0 : (nBytes - 1); + var d = isLE ? 1 : -1; + var s = value < 0 || (value === 0 && 1 / value < 0) ? 1 : 0; + + value = Math.abs(value); + + if (isNaN(value) || value === Infinity) { + m = isNaN(value) ? 1 : 0; + e = eMax; + } else { + e = Math.floor(Math.log(value) / Math.LN2); + if (value * (c = Math.pow(2, -e)) < 1) { + e--; + c *= 2; + } + if (e + eBias >= 1) { + value += rt / c; + } else { + value += rt * Math.pow(2, 1 - eBias); + } + if (value * c >= 2) { + e++; + c /= 2; + } + + if (e + eBias >= eMax) { + m = 0; + e = eMax; + } else if (e + eBias >= 1) { + m = ((value * c) - 1) * Math.pow(2, mLen); + e = e + eBias; + } else { + m = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen); + e = 0; + } + } + + for (; mLen >= 8; buffer[offset + i] = m & 0xff, i += d, m /= 256, mLen -= 8) {} + + e = (e << mLen) | m; + eLen += mLen; + for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8) {} + + buffer[offset + i - d] |= s * 128; +}; + +'use strict'; + +var pbf = Pbf; + +var ieee754 = ieee754$1; + +function Pbf(buf) { + this.buf = ArrayBuffer.isView && ArrayBuffer.isView(buf) ? buf : new Uint8Array(buf || 0); + this.pos = 0; + this.type = 0; + this.length = this.buf.length; +} + +Pbf.Varint = 0; // varint: int32, int64, uint32, uint64, sint32, sint64, bool, enum +Pbf.Fixed64 = 1; // 64-bit: double, fixed64, sfixed64 +Pbf.Bytes = 2; // length-delimited: string, bytes, embedded messages, packed repeated fields +Pbf.Fixed32 = 5; // 32-bit: float, fixed32, sfixed32 + +var SHIFT_LEFT_32 = (1 << 16) * (1 << 16), + SHIFT_RIGHT_32 = 1 / SHIFT_LEFT_32; + +// Threshold chosen based on both benchmarking and knowledge about browser string +// data structures (which currently switch structure types at 12 bytes or more) +var TEXT_DECODER_MIN_LENGTH = 12; +var utf8TextDecoder = typeof TextDecoder === 'undefined' ? null : new TextDecoder('utf8'); + +Pbf.prototype = { + + destroy: function() { + this.buf = null; + }, + + // === READING ================================================================= + + readFields: function(readField, result, end) { + end = end || this.length; + + while (this.pos < end) { + var val = this.readVarint(), + tag = val >> 3, + startPos = this.pos; + + this.type = val & 0x7; + readField(tag, result, this); + + if (this.pos === startPos) this.skip(val); + } + return result; + }, + + readMessage: function(readField, result) { + return this.readFields(readField, result, this.readVarint() + this.pos); + }, + + readFixed32: function() { + var val = readUInt32(this.buf, this.pos); + this.pos += 4; + return val; + }, + + readSFixed32: function() { + var val = readInt32(this.buf, this.pos); + this.pos += 4; + return val; + }, + + // 64-bit int handling is based on github.com/dpw/node-buffer-more-ints (MIT-licensed) + + readFixed64: function() { + var val = readUInt32(this.buf, this.pos) + readUInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32; + this.pos += 8; + return val; + }, + + readSFixed64: function() { + var val = readUInt32(this.buf, this.pos) + readInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32; + this.pos += 8; + return val; + }, + + readFloat: function() { + var val = ieee754.read(this.buf, this.pos, true, 23, 4); + this.pos += 4; + return val; + }, + + readDouble: function() { + var val = ieee754.read(this.buf, this.pos, true, 52, 8); + this.pos += 8; + return val; + }, + + readVarint: function(isSigned) { + var buf = this.buf, + val, b; + + b = buf[this.pos++]; val = b & 0x7f; if (b < 0x80) return val; + b = buf[this.pos++]; val |= (b & 0x7f) << 7; if (b < 0x80) return val; + b = buf[this.pos++]; val |= (b & 0x7f) << 14; if (b < 0x80) return val; + b = buf[this.pos++]; val |= (b & 0x7f) << 21; if (b < 0x80) return val; + b = buf[this.pos]; val |= (b & 0x0f) << 28; + + return readVarintRemainder(val, isSigned, this); + }, + + readVarint64: function() { // for compatibility with v2.0.1 + return this.readVarint(true); + }, + + readSVarint: function() { + var num = this.readVarint(); + return num % 2 === 1 ? (num + 1) / -2 : num / 2; // zigzag encoding + }, + + readBoolean: function() { + return Boolean(this.readVarint()); + }, + + readString: function() { + var end = this.readVarint() + this.pos; + var pos = this.pos; + this.pos = end; + + if (end - pos >= TEXT_DECODER_MIN_LENGTH && utf8TextDecoder) { + // longer strings are fast with the built-in browser TextDecoder API + return readUtf8TextDecoder(this.buf, pos, end); + } + // short strings are fast with our custom implementation + return readUtf8(this.buf, pos, end); + }, + + readBytes: function() { + var end = this.readVarint() + this.pos, + buffer = this.buf.subarray(this.pos, end); + this.pos = end; + return buffer; + }, + + // verbose for performance reasons; doesn't affect gzipped size + + readPackedVarint: function(arr, isSigned) { + if (this.type !== Pbf.Bytes) return arr.push(this.readVarint(isSigned)); + var end = readPackedEnd(this); + arr = arr || []; + while (this.pos < end) arr.push(this.readVarint(isSigned)); + return arr; + }, + readPackedSVarint: function(arr) { + if (this.type !== Pbf.Bytes) return arr.push(this.readSVarint()); + var end = readPackedEnd(this); + arr = arr || []; + while (this.pos < end) arr.push(this.readSVarint()); + return arr; + }, + readPackedBoolean: function(arr) { + if (this.type !== Pbf.Bytes) return arr.push(this.readBoolean()); + var end = readPackedEnd(this); + arr = arr || []; + while (this.pos < end) arr.push(this.readBoolean()); + return arr; + }, + readPackedFloat: function(arr) { + if (this.type !== Pbf.Bytes) return arr.push(this.readFloat()); + var end = readPackedEnd(this); + arr = arr || []; + while (this.pos < end) arr.push(this.readFloat()); + return arr; + }, + readPackedDouble: function(arr) { + if (this.type !== Pbf.Bytes) return arr.push(this.readDouble()); + var end = readPackedEnd(this); + arr = arr || []; + while (this.pos < end) arr.push(this.readDouble()); + return arr; + }, + readPackedFixed32: function(arr) { + if (this.type !== Pbf.Bytes) return arr.push(this.readFixed32()); + var end = readPackedEnd(this); + arr = arr || []; + while (this.pos < end) arr.push(this.readFixed32()); + return arr; + }, + readPackedSFixed32: function(arr) { + if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed32()); + var end = readPackedEnd(this); + arr = arr || []; + while (this.pos < end) arr.push(this.readSFixed32()); + return arr; + }, + readPackedFixed64: function(arr) { + if (this.type !== Pbf.Bytes) return arr.push(this.readFixed64()); + var end = readPackedEnd(this); + arr = arr || []; + while (this.pos < end) arr.push(this.readFixed64()); + return arr; + }, + readPackedSFixed64: function(arr) { + if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed64()); + var end = readPackedEnd(this); + arr = arr || []; + while (this.pos < end) arr.push(this.readSFixed64()); + return arr; + }, + + skip: function(val) { + var type = val & 0x7; + if (type === Pbf.Varint) while (this.buf[this.pos++] > 0x7f) {} + else if (type === Pbf.Bytes) this.pos = this.readVarint() + this.pos; + else if (type === Pbf.Fixed32) this.pos += 4; + else if (type === Pbf.Fixed64) this.pos += 8; + else throw new Error('Unimplemented type: ' + type); + }, + + // === WRITING ================================================================= + + writeTag: function(tag, type) { + this.writeVarint((tag << 3) | type); + }, + + realloc: function(min) { + var length = this.length || 16; + + while (length < this.pos + min) length *= 2; + + if (length !== this.length) { + var buf = new Uint8Array(length); + buf.set(this.buf); + this.buf = buf; + this.length = length; + } + }, + + finish: function() { + this.length = this.pos; + this.pos = 0; + return this.buf.subarray(0, this.length); + }, + + writeFixed32: function(val) { + this.realloc(4); + writeInt32(this.buf, val, this.pos); + this.pos += 4; + }, + + writeSFixed32: function(val) { + this.realloc(4); + writeInt32(this.buf, val, this.pos); + this.pos += 4; + }, + + writeFixed64: function(val) { + this.realloc(8); + writeInt32(this.buf, val & -1, this.pos); + writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4); + this.pos += 8; + }, + + writeSFixed64: function(val) { + this.realloc(8); + writeInt32(this.buf, val & -1, this.pos); + writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4); + this.pos += 8; + }, + + writeVarint: function(val) { + val = +val || 0; + + if (val > 0xfffffff || val < 0) { + writeBigVarint(val, this); + return; + } + + this.realloc(4); + + this.buf[this.pos++] = val & 0x7f | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return; + this.buf[this.pos++] = ((val >>>= 7) & 0x7f) | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return; + this.buf[this.pos++] = ((val >>>= 7) & 0x7f) | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return; + this.buf[this.pos++] = (val >>> 7) & 0x7f; + }, + + writeSVarint: function(val) { + this.writeVarint(val < 0 ? -val * 2 - 1 : val * 2); + }, + + writeBoolean: function(val) { + this.writeVarint(Boolean(val)); + }, + + writeString: function(str) { + str = String(str); + this.realloc(str.length * 4); + + this.pos++; // reserve 1 byte for short string length + + var startPos = this.pos; + // write the string directly to the buffer and see how much was written + this.pos = writeUtf8(this.buf, str, this.pos); + var len = this.pos - startPos; + + if (len >= 0x80) makeRoomForExtraLength(startPos, len, this); + + // finally, write the message length in the reserved place and restore the position + this.pos = startPos - 1; + this.writeVarint(len); + this.pos += len; + }, + + writeFloat: function(val) { + this.realloc(4); + ieee754.write(this.buf, val, this.pos, true, 23, 4); + this.pos += 4; + }, + + writeDouble: function(val) { + this.realloc(8); + ieee754.write(this.buf, val, this.pos, true, 52, 8); + this.pos += 8; + }, + + writeBytes: function(buffer) { + var len = buffer.length; + this.writeVarint(len); + this.realloc(len); + for (var i = 0; i < len; i++) this.buf[this.pos++] = buffer[i]; + }, + + writeRawMessage: function(fn, obj) { + this.pos++; // reserve 1 byte for short message length + + // write the message directly to the buffer and see how much was written + var startPos = this.pos; + fn(obj, this); + var len = this.pos - startPos; + + if (len >= 0x80) makeRoomForExtraLength(startPos, len, this); + + // finally, write the message length in the reserved place and restore the position + this.pos = startPos - 1; + this.writeVarint(len); + this.pos += len; + }, + + writeMessage: function(tag, fn, obj) { + this.writeTag(tag, Pbf.Bytes); + this.writeRawMessage(fn, obj); + }, + + writePackedVarint: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedVarint, arr); }, + writePackedSVarint: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSVarint, arr); }, + writePackedBoolean: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedBoolean, arr); }, + writePackedFloat: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFloat, arr); }, + writePackedDouble: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedDouble, arr); }, + writePackedFixed32: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFixed32, arr); }, + writePackedSFixed32: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSFixed32, arr); }, + writePackedFixed64: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFixed64, arr); }, + writePackedSFixed64: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSFixed64, arr); }, + + writeBytesField: function(tag, buffer) { + this.writeTag(tag, Pbf.Bytes); + this.writeBytes(buffer); + }, + writeFixed32Field: function(tag, val) { + this.writeTag(tag, Pbf.Fixed32); + this.writeFixed32(val); + }, + writeSFixed32Field: function(tag, val) { + this.writeTag(tag, Pbf.Fixed32); + this.writeSFixed32(val); + }, + writeFixed64Field: function(tag, val) { + this.writeTag(tag, Pbf.Fixed64); + this.writeFixed64(val); + }, + writeSFixed64Field: function(tag, val) { + this.writeTag(tag, Pbf.Fixed64); + this.writeSFixed64(val); + }, + writeVarintField: function(tag, val) { + this.writeTag(tag, Pbf.Varint); + this.writeVarint(val); + }, + writeSVarintField: function(tag, val) { + this.writeTag(tag, Pbf.Varint); + this.writeSVarint(val); + }, + writeStringField: function(tag, str) { + this.writeTag(tag, Pbf.Bytes); + this.writeString(str); + }, + writeFloatField: function(tag, val) { + this.writeTag(tag, Pbf.Fixed32); + this.writeFloat(val); + }, + writeDoubleField: function(tag, val) { + this.writeTag(tag, Pbf.Fixed64); + this.writeDouble(val); + }, + writeBooleanField: function(tag, val) { + this.writeVarintField(tag, Boolean(val)); + } +}; + +function readVarintRemainder(l, s, p) { + var buf = p.buf, + h, b; + + b = buf[p.pos++]; h = (b & 0x70) >> 4; if (b < 0x80) return toNum(l, h, s); + b = buf[p.pos++]; h |= (b & 0x7f) << 3; if (b < 0x80) return toNum(l, h, s); + b = buf[p.pos++]; h |= (b & 0x7f) << 10; if (b < 0x80) return toNum(l, h, s); + b = buf[p.pos++]; h |= (b & 0x7f) << 17; if (b < 0x80) return toNum(l, h, s); + b = buf[p.pos++]; h |= (b & 0x7f) << 24; if (b < 0x80) return toNum(l, h, s); + b = buf[p.pos++]; h |= (b & 0x01) << 31; if (b < 0x80) return toNum(l, h, s); + + throw new Error('Expected varint not more than 10 bytes'); +} + +function readPackedEnd(pbf) { + return pbf.type === Pbf.Bytes ? + pbf.readVarint() + pbf.pos : pbf.pos + 1; +} + +function toNum(low, high, isSigned) { + if (isSigned) { + return high * 0x100000000 + (low >>> 0); + } + + return ((high >>> 0) * 0x100000000) + (low >>> 0); +} + +function writeBigVarint(val, pbf) { + var low, high; + + if (val >= 0) { + low = (val % 0x100000000) | 0; + high = (val / 0x100000000) | 0; + } else { + low = ~(-val % 0x100000000); + high = ~(-val / 0x100000000); + + if (low ^ 0xffffffff) { + low = (low + 1) | 0; + } else { + low = 0; + high = (high + 1) | 0; + } + } + + if (val >= 0x10000000000000000 || val < -0x10000000000000000) { + throw new Error('Given varint doesn\'t fit into 10 bytes'); + } + + pbf.realloc(10); + + writeBigVarintLow(low, high, pbf); + writeBigVarintHigh(high, pbf); +} + +function writeBigVarintLow(low, high, pbf) { + pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7; + pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7; + pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7; + pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7; + pbf.buf[pbf.pos] = low & 0x7f; +} + +function writeBigVarintHigh(high, pbf) { + var lsb = (high & 0x07) << 4; + + pbf.buf[pbf.pos++] |= lsb | ((high >>>= 3) ? 0x80 : 0); if (!high) return; + pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return; + pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return; + pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return; + pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return; + pbf.buf[pbf.pos++] = high & 0x7f; +} + +function makeRoomForExtraLength(startPos, len, pbf) { + var extraLen = + len <= 0x3fff ? 1 : + len <= 0x1fffff ? 2 : + len <= 0xfffffff ? 3 : Math.floor(Math.log(len) / (Math.LN2 * 7)); + + // if 1 byte isn't enough for encoding message length, shift the data to the right + pbf.realloc(extraLen); + for (var i = pbf.pos - 1; i >= startPos; i--) pbf.buf[i + extraLen] = pbf.buf[i]; +} + +function writePackedVarint(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeVarint(arr[i]); } +function writePackedSVarint(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSVarint(arr[i]); } +function writePackedFloat(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFloat(arr[i]); } +function writePackedDouble(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeDouble(arr[i]); } +function writePackedBoolean(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeBoolean(arr[i]); } +function writePackedFixed32(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFixed32(arr[i]); } +function writePackedSFixed32(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSFixed32(arr[i]); } +function writePackedFixed64(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFixed64(arr[i]); } +function writePackedSFixed64(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSFixed64(arr[i]); } + +// Buffer code below from https://github.com/feross/buffer, MIT-licensed + +function readUInt32(buf, pos) { + return ((buf[pos]) | + (buf[pos + 1] << 8) | + (buf[pos + 2] << 16)) + + (buf[pos + 3] * 0x1000000); +} + +function writeInt32(buf, val, pos) { + buf[pos] = val; + buf[pos + 1] = (val >>> 8); + buf[pos + 2] = (val >>> 16); + buf[pos + 3] = (val >>> 24); +} + +function readInt32(buf, pos) { + return ((buf[pos]) | + (buf[pos + 1] << 8) | + (buf[pos + 2] << 16)) + + (buf[pos + 3] << 24); +} + +function readUtf8(buf, pos, end) { + var str = ''; + var i = pos; + + while (i < end) { + var b0 = buf[i]; + var c = null; // codepoint + var bytesPerSequence = + b0 > 0xEF ? 4 : + b0 > 0xDF ? 3 : + b0 > 0xBF ? 2 : 1; + + if (i + bytesPerSequence > end) break; + + var b1, b2, b3; + + if (bytesPerSequence === 1) { + if (b0 < 0x80) { + c = b0; + } + } else if (bytesPerSequence === 2) { + b1 = buf[i + 1]; + if ((b1 & 0xC0) === 0x80) { + c = (b0 & 0x1F) << 0x6 | (b1 & 0x3F); + if (c <= 0x7F) { + c = null; + } + } + } else if (bytesPerSequence === 3) { + b1 = buf[i + 1]; + b2 = buf[i + 2]; + if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80) { + c = (b0 & 0xF) << 0xC | (b1 & 0x3F) << 0x6 | (b2 & 0x3F); + if (c <= 0x7FF || (c >= 0xD800 && c <= 0xDFFF)) { + c = null; + } + } + } else if (bytesPerSequence === 4) { + b1 = buf[i + 1]; + b2 = buf[i + 2]; + b3 = buf[i + 3]; + if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80 && (b3 & 0xC0) === 0x80) { + c = (b0 & 0xF) << 0x12 | (b1 & 0x3F) << 0xC | (b2 & 0x3F) << 0x6 | (b3 & 0x3F); + if (c <= 0xFFFF || c >= 0x110000) { + c = null; + } + } + } + + if (c === null) { + c = 0xFFFD; + bytesPerSequence = 1; + + } else if (c > 0xFFFF) { + c -= 0x10000; + str += String.fromCharCode(c >>> 10 & 0x3FF | 0xD800); + c = 0xDC00 | c & 0x3FF; + } + + str += String.fromCharCode(c); + i += bytesPerSequence; + } + + return str; +} + +function readUtf8TextDecoder(buf, pos, end) { + return utf8TextDecoder.decode(buf.subarray(pos, end)); +} + +function writeUtf8(buf, str, pos) { + for (var i = 0, c, lead; i < str.length; i++) { + c = str.charCodeAt(i); // code point + + if (c > 0xD7FF && c < 0xE000) { + if (lead) { + if (c < 0xDC00) { + buf[pos++] = 0xEF; + buf[pos++] = 0xBF; + buf[pos++] = 0xBD; + lead = c; + continue; + } else { + c = lead - 0xD800 << 10 | c - 0xDC00 | 0x10000; + lead = null; + } + } else { + if (c > 0xDBFF || (i + 1 === str.length)) { + buf[pos++] = 0xEF; + buf[pos++] = 0xBF; + buf[pos++] = 0xBD; + } else { + lead = c; + } + continue; + } + } else if (lead) { + buf[pos++] = 0xEF; + buf[pos++] = 0xBF; + buf[pos++] = 0xBD; + lead = null; + } + + if (c < 0x80) { + buf[pos++] = c; + } else { + if (c < 0x800) { + buf[pos++] = c >> 0x6 | 0xC0; + } else { + if (c < 0x10000) { + buf[pos++] = c >> 0xC | 0xE0; + } else { + buf[pos++] = c >> 0x12 | 0xF0; + buf[pos++] = c >> 0xC & 0x3F | 0x80; + } + buf[pos++] = c >> 0x6 & 0x3F | 0x80; + } + buf[pos++] = c & 0x3F | 0x80; + } + } + return pos; +} + +var Protobuf = /*@__PURE__*/getDefaultExportFromCjs(pbf); + +const border$1 = 3; +function readFontstacks(tag, glyphs, pbf) { + if (tag === 1) { + pbf.readMessage(readFontstack, glyphs); + } +} +function readFontstack(tag, glyphs, pbf) { + if (tag === 3) { + const { id, bitmap, width, height, left, top, advance } = pbf.readMessage(readGlyph, {}); + glyphs.push({ + id, + bitmap: new AlphaImage({ + width: width + 2 * border$1, + height: height + 2 * border$1 + }, bitmap), + metrics: { width, height, left, top, advance } + }); + } +} +function readGlyph(tag, glyph, pbf) { + if (tag === 1) + glyph.id = pbf.readVarint(); + else if (tag === 2) + glyph.bitmap = pbf.readBytes(); + else if (tag === 3) + glyph.width = pbf.readVarint(); + else if (tag === 4) + glyph.height = pbf.readVarint(); + else if (tag === 5) + glyph.left = pbf.readSVarint(); + else if (tag === 6) + glyph.top = pbf.readSVarint(); + else if (tag === 7) + glyph.advance = pbf.readVarint(); +} +function parseGlyphPbf(data) { + return new Protobuf(data).readFields(readFontstacks, []); +} +const GLYPH_PBF_BORDER = border$1; + +function potpack(boxes) { + + // calculate total box area and maximum box width + let area = 0; + let maxWidth = 0; + + for (const box of boxes) { + area += box.w * box.h; + maxWidth = Math.max(maxWidth, box.w); + } + + // sort the boxes for insertion by height, descending + boxes.sort((a, b) => b.h - a.h); + + // aim for a squarish resulting container, + // slightly adjusted for sub-100% space utilization + const startWidth = Math.max(Math.ceil(Math.sqrt(area / 0.95)), maxWidth); + + // start with a single empty space, unbounded at the bottom + const spaces = [{x: 0, y: 0, w: startWidth, h: Infinity}]; + + let width = 0; + let height = 0; + + for (const box of boxes) { + // look through spaces backwards so that we check smaller spaces first + for (let i = spaces.length - 1; i >= 0; i--) { + const space = spaces[i]; + + // look for empty spaces that can accommodate the current box + if (box.w > space.w || box.h > space.h) continue; + + // found the space; add the box to its top-left corner + // |-------|-------| + // | box | | + // |_______| | + // | space | + // |_______________| + box.x = space.x; + box.y = space.y; + + height = Math.max(height, box.y + box.h); + width = Math.max(width, box.x + box.w); + + if (box.w === space.w && box.h === space.h) { + // space matches the box exactly; remove it + const last = spaces.pop(); + if (i < spaces.length) spaces[i] = last; + + } else if (box.h === space.h) { + // space matches the box height; update it accordingly + // |-------|---------------| + // | box | updated space | + // |_______|_______________| + space.x += box.w; + space.w -= box.w; + + } else if (box.w === space.w) { + // space matches the box width; update it accordingly + // |---------------| + // | box | + // |_______________| + // | updated space | + // |_______________| + space.y += box.h; + space.h -= box.h; + + } else { + // otherwise the box splits the space into two spaces + // |-------|-----------| + // | box | new space | + // |_______|___________| + // | updated space | + // |___________________| + spaces.push({ + x: space.x + box.w, + y: space.y, + w: space.w - box.w, + h: box.h + }); + space.y += box.h; + space.h -= box.h; + } + break; + } + } + + return { + w: width, // container width + h: height, // container height + fill: (area / (width * height)) || 0 // space utilization + }; +} + +/* eslint-disable key-spacing */ +const IMAGE_PADDING = 1; +class ImagePosition { + constructor(paddedRect, { pixelRatio, version, stretchX, stretchY, content }) { + this.paddedRect = paddedRect; + this.pixelRatio = pixelRatio; + this.stretchX = stretchX; + this.stretchY = stretchY; + this.content = content; + this.version = version; + } + get tl() { + return [ + this.paddedRect.x + IMAGE_PADDING, + this.paddedRect.y + IMAGE_PADDING + ]; + } + get br() { + return [ + this.paddedRect.x + this.paddedRect.w - IMAGE_PADDING, + this.paddedRect.y + this.paddedRect.h - IMAGE_PADDING + ]; + } + get tlbr() { + return this.tl.concat(this.br); + } + get displaySize() { + return [ + (this.paddedRect.w - IMAGE_PADDING * 2) / this.pixelRatio, + (this.paddedRect.h - IMAGE_PADDING * 2) / this.pixelRatio + ]; + } +} +/** + * @internal + * A class holding all the images + */ +class ImageAtlas { + constructor(icons, patterns) { + const iconPositions = {}, patternPositions = {}; + this.haveRenderCallbacks = []; + const bins = []; + this.addImages(icons, iconPositions, bins); + this.addImages(patterns, patternPositions, bins); + const { w, h } = potpack(bins); + const image = new RGBAImage({ width: w || 1, height: h || 1 }); + for (const id in icons) { + const src = icons[id]; + const bin = iconPositions[id].paddedRect; + RGBAImage.copy(src.data, image, { x: 0, y: 0 }, { x: bin.x + IMAGE_PADDING, y: bin.y + IMAGE_PADDING }, src.data); + } + for (const id in patterns) { + const src = patterns[id]; + const bin = patternPositions[id].paddedRect; + const x = bin.x + IMAGE_PADDING, y = bin.y + IMAGE_PADDING, w = src.data.width, h = src.data.height; + RGBAImage.copy(src.data, image, { x: 0, y: 0 }, { x, y }, src.data); + // Add 1 pixel wrapped padding on each side of the image. + RGBAImage.copy(src.data, image, { x: 0, y: h - 1 }, { x, y: y - 1 }, { width: w, height: 1 }); // T + RGBAImage.copy(src.data, image, { x: 0, y: 0 }, { x, y: y + h }, { width: w, height: 1 }); // B + RGBAImage.copy(src.data, image, { x: w - 1, y: 0 }, { x: x - 1, y }, { width: 1, height: h }); // L + RGBAImage.copy(src.data, image, { x: 0, y: 0 }, { x: x + w, y }, { width: 1, height: h }); // R + } + this.image = image; + this.iconPositions = iconPositions; + this.patternPositions = patternPositions; + } + addImages(images, positions, bins) { + for (const id in images) { + const src = images[id]; + const bin = { + x: 0, + y: 0, + w: src.data.width + 2 * IMAGE_PADDING, + h: src.data.height + 2 * IMAGE_PADDING, + }; + bins.push(bin); + positions[id] = new ImagePosition(bin, src); + if (src.hasRenderCallback) { + this.haveRenderCallbacks.push(id); + } + } + } + patchUpdatedImages(imageManager, texture) { + imageManager.dispatchRenderCallbacks(this.haveRenderCallbacks); + for (const name in imageManager.updatedImages) { + this.patchUpdatedImage(this.iconPositions[name], imageManager.getImage(name), texture); + this.patchUpdatedImage(this.patternPositions[name], imageManager.getImage(name), texture); + } + } + patchUpdatedImage(position, image, texture) { + if (!position || !image) + return; + if (position.version === image.version) + return; + position.version = image.version; + const [x, y] = position.tl; + texture.update(image.data, undefined, { x, y }); + } +} +register('ImagePosition', ImagePosition); +register('ImageAtlas', ImageAtlas); + +exports.WritingMode = void 0; +(function (WritingMode) { + WritingMode[WritingMode["none"] = 0] = "none"; + WritingMode[WritingMode["horizontal"] = 1] = "horizontal"; + WritingMode[WritingMode["vertical"] = 2] = "vertical"; + WritingMode[WritingMode["horizontalOnly"] = 3] = "horizontalOnly"; +})(exports.WritingMode || (exports.WritingMode = {})); +const SHAPING_DEFAULT_OFFSET = -17; +function isEmpty(positionedLines) { + for (const line of positionedLines) { + if (line.positionedGlyphs.length !== 0) { + return false; + } + } + return true; +} +// Max number of images in label is 6401 U+E000–U+F8FF that covers +// Basic Multilingual Plane Unicode Private Use Area (PUA). +const PUAbegin = 0xE000; +const PUAend = 0xF8FF; +class SectionOptions { + constructor() { + this.scale = 1.0; + this.fontStack = ''; + this.imageName = null; + } + static forText(scale, fontStack) { + const textOptions = new SectionOptions(); + textOptions.scale = scale || 1; + textOptions.fontStack = fontStack; + return textOptions; + } + static forImage(imageName) { + const imageOptions = new SectionOptions(); + imageOptions.imageName = imageName; + return imageOptions; + } +} +class TaggedString { + constructor() { + this.text = ''; + this.sectionIndex = []; + this.sections = []; + this.imageSectionID = null; + } + static fromFeature(text, defaultFontStack) { + const result = new TaggedString(); + for (let i = 0; i < text.sections.length; i++) { + const section = text.sections[i]; + if (!section.image) { + result.addTextSection(section, defaultFontStack); + } + else { + result.addImageSection(section); + } + } + return result; + } + length() { + return this.text.length; + } + getSection(index) { + return this.sections[this.sectionIndex[index]]; + } + getSectionIndex(index) { + return this.sectionIndex[index]; + } + getCharCode(index) { + return this.text.charCodeAt(index); + } + verticalizePunctuation() { + this.text = verticalizePunctuation(this.text); + } + trim() { + let beginningWhitespace = 0; + for (let i = 0; i < this.text.length && whitespace[this.text.charCodeAt(i)]; i++) { + beginningWhitespace++; + } + let trailingWhitespace = this.text.length; + for (let i = this.text.length - 1; i >= 0 && i >= beginningWhitespace && whitespace[this.text.charCodeAt(i)]; i--) { + trailingWhitespace--; + } + this.text = this.text.substring(beginningWhitespace, trailingWhitespace); + this.sectionIndex = this.sectionIndex.slice(beginningWhitespace, trailingWhitespace); + } + substring(start, end) { + const substring = new TaggedString(); + substring.text = this.text.substring(start, end); + substring.sectionIndex = this.sectionIndex.slice(start, end); + substring.sections = this.sections; + return substring; + } + toString() { + return this.text; + } + getMaxScale() { + return this.sectionIndex.reduce((max, index) => Math.max(max, this.sections[index].scale), 0); + } + addTextSection(section, defaultFontStack) { + this.text += section.text; + this.sections.push(SectionOptions.forText(section.scale, section.fontStack || defaultFontStack)); + const index = this.sections.length - 1; + for (let i = 0; i < section.text.length; ++i) { + this.sectionIndex.push(index); + } + } + addImageSection(section) { + const imageName = section.image ? section.image.name : ''; + if (imageName.length === 0) { + warnOnce('Can\'t add FormattedSection with an empty image.'); + return; + } + const nextImageSectionCharCode = this.getNextImageSectionCharCode(); + if (!nextImageSectionCharCode) { + warnOnce(`Reached maximum number of images ${PUAend - PUAbegin + 2}`); + return; + } + this.text += String.fromCharCode(nextImageSectionCharCode); + this.sections.push(SectionOptions.forImage(imageName)); + this.sectionIndex.push(this.sections.length - 1); + } + getNextImageSectionCharCode() { + if (!this.imageSectionID) { + this.imageSectionID = PUAbegin; + return this.imageSectionID; + } + if (this.imageSectionID >= PUAend) + return null; + return ++this.imageSectionID; + } +} +function breakLines(input, lineBreakPoints) { + const lines = []; + const text = input.text; + let start = 0; + for (const lineBreak of lineBreakPoints) { + lines.push(input.substring(start, lineBreak)); + start = lineBreak; + } + if (start < text.length) { + lines.push(input.substring(start, text.length)); + } + return lines; +} +function shapeText(text, glyphMap, glyphPositions, imagePositions, defaultFontStack, maxWidth, lineHeight, textAnchor, textJustify, spacing, translate, writingMode, allowVerticalPlacement, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom) { + const logicalInput = TaggedString.fromFeature(text, defaultFontStack); + if (writingMode === exports.WritingMode.vertical) { + logicalInput.verticalizePunctuation(); + } + let lines; + const { processBidirectionalText, processStyledBidirectionalText } = plugin; + if (processBidirectionalText && logicalInput.sections.length === 1) { + // Bidi doesn't have to be style-aware + lines = []; + const untaggedLines = processBidirectionalText(logicalInput.toString(), determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize)); + for (const line of untaggedLines) { + const taggedLine = new TaggedString(); + taggedLine.text = line; + taggedLine.sections = logicalInput.sections; + for (let i = 0; i < line.length; i++) { + taggedLine.sectionIndex.push(0); + } + lines.push(taggedLine); + } + } + else if (processStyledBidirectionalText) { + // Need version of mapbox-gl-rtl-text with style support for combining RTL text + // with formatting + lines = []; + const processedLines = processStyledBidirectionalText(logicalInput.text, logicalInput.sectionIndex, determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize)); + for (const line of processedLines) { + const taggedLine = new TaggedString(); + taggedLine.text = line[0]; + taggedLine.sectionIndex = line[1]; + taggedLine.sections = logicalInput.sections; + lines.push(taggedLine); + } + } + else { + lines = breakLines(logicalInput, determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize)); + } + const positionedLines = []; + const shaping = { + positionedLines, + text: logicalInput.toString(), + top: translate[1], + bottom: translate[1], + left: translate[0], + right: translate[0], + writingMode, + iconsInText: false, + verticalizable: false + }; + shapeLines(shaping, glyphMap, glyphPositions, imagePositions, lines, lineHeight, textAnchor, textJustify, writingMode, spacing, allowVerticalPlacement, layoutTextSizeThisZoom); + if (isEmpty(positionedLines)) + return false; + return shaping; +} +// using computed properties due to https://github.com/facebook/flow/issues/380 +/* eslint no-useless-computed-key: 0 */ +const whitespace = { + [0x09]: true, + [0x0a]: true, + [0x0b]: true, + [0x0c]: true, + [0x0d]: true, + [0x20]: true, // space +}; +const breakable = { + [0x0a]: true, + [0x20]: true, + [0x26]: true, + [0x28]: true, + [0x29]: true, + [0x2b]: true, + [0x2d]: true, + [0x2f]: true, + [0xad]: true, + [0xb7]: true, + [0x200b]: true, + [0x2010]: true, + [0x2013]: true, + [0x2027]: true // interpunct + // Many other characters may be reasonable breakpoints + // Consider "neutral orientation" characters at scriptDetection.charHasNeutralVerticalOrientation + // See https://github.com/mapbox/mapbox-gl-js/issues/3658 +}; +function getGlyphAdvance(codePoint, section, glyphMap, imagePositions, spacing, layoutTextSize) { + if (!section.imageName) { + const positions = glyphMap[section.fontStack]; + const glyph = positions && positions[codePoint]; + if (!glyph) + return 0; + return glyph.metrics.advance * section.scale + spacing; + } + else { + const imagePosition = imagePositions[section.imageName]; + if (!imagePosition) + return 0; + return imagePosition.displaySize[0] * section.scale * ONE_EM / layoutTextSize + spacing; + } +} +function determineAverageLineWidth(logicalInput, spacing, maxWidth, glyphMap, imagePositions, layoutTextSize) { + let totalWidth = 0; + for (let index = 0; index < logicalInput.length(); index++) { + const section = logicalInput.getSection(index); + totalWidth += getGlyphAdvance(logicalInput.getCharCode(index), section, glyphMap, imagePositions, spacing, layoutTextSize); + } + const lineCount = Math.max(1, Math.ceil(totalWidth / maxWidth)); + return totalWidth / lineCount; +} +function calculateBadness(lineWidth, targetWidth, penalty, isLastBreak) { + const raggedness = Math.pow(lineWidth - targetWidth, 2); + if (isLastBreak) { + // Favor finals lines shorter than average over longer than average + if (lineWidth < targetWidth) { + return raggedness / 2; + } + else { + return raggedness * 2; + } + } + return raggedness + Math.abs(penalty) * penalty; +} +function calculatePenalty(codePoint, nextCodePoint, penalizableIdeographicBreak) { + let penalty = 0; + // Force break on newline + if (codePoint === 0x0a) { + penalty -= 10000; + } + // Penalize breaks between characters that allow ideographic breaking because + // they are less preferable than breaks at spaces (or zero width spaces). + if (penalizableIdeographicBreak) { + penalty += 150; + } + // Penalize open parenthesis at end of line + if (codePoint === 0x28 || codePoint === 0xff08) { + penalty += 50; + } + // Penalize close parenthesis at beginning of line + if (nextCodePoint === 0x29 || nextCodePoint === 0xff09) { + penalty += 50; + } + return penalty; +} +function evaluateBreak(breakIndex, breakX, targetWidth, potentialBreaks, penalty, isLastBreak) { + // We could skip evaluating breaks where the line length (breakX - priorBreak.x) > maxWidth + // ...but in fact we allow lines longer than maxWidth (if there's no break points) + // ...and when targetWidth and maxWidth are close, strictly enforcing maxWidth can give + // more lopsided results. + let bestPriorBreak = null; + let bestBreakBadness = calculateBadness(breakX, targetWidth, penalty, isLastBreak); + for (const potentialBreak of potentialBreaks) { + const lineWidth = breakX - potentialBreak.x; + const breakBadness = calculateBadness(lineWidth, targetWidth, penalty, isLastBreak) + potentialBreak.badness; + if (breakBadness <= bestBreakBadness) { + bestPriorBreak = potentialBreak; + bestBreakBadness = breakBadness; + } + } + return { + index: breakIndex, + x: breakX, + priorBreak: bestPriorBreak, + badness: bestBreakBadness + }; +} +function leastBadBreaks(lastLineBreak) { + if (!lastLineBreak) { + return []; + } + return leastBadBreaks(lastLineBreak.priorBreak).concat(lastLineBreak.index); +} +function determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize) { + if (symbolPlacement !== 'point') + return []; + if (!logicalInput) + return []; + const potentialLineBreaks = []; + const targetWidth = determineAverageLineWidth(logicalInput, spacing, maxWidth, glyphMap, imagePositions, layoutTextSize); + const hasServerSuggestedBreakpoints = logicalInput.text.indexOf('\u200b') >= 0; + let currentX = 0; + for (let i = 0; i < logicalInput.length(); i++) { + const section = logicalInput.getSection(i); + const codePoint = logicalInput.getCharCode(i); + if (!whitespace[codePoint]) + currentX += getGlyphAdvance(codePoint, section, glyphMap, imagePositions, spacing, layoutTextSize); + // Ideographic characters, spaces, and word-breaking punctuation that often appear without + // surrounding spaces. + if ((i < logicalInput.length() - 1)) { + const ideographicBreak = charAllowsIdeographicBreaking(codePoint); + if (breakable[codePoint] || ideographicBreak || section.imageName) { + potentialLineBreaks.push(evaluateBreak(i + 1, currentX, targetWidth, potentialLineBreaks, calculatePenalty(codePoint, logicalInput.getCharCode(i + 1), ideographicBreak && hasServerSuggestedBreakpoints), false)); + } + } + } + return leastBadBreaks(evaluateBreak(logicalInput.length(), currentX, targetWidth, potentialLineBreaks, 0, true)); +} +function getAnchorAlignment(anchor) { + let horizontalAlign = 0.5, verticalAlign = 0.5; + switch (anchor) { + case 'right': + case 'top-right': + case 'bottom-right': + horizontalAlign = 1; + break; + case 'left': + case 'top-left': + case 'bottom-left': + horizontalAlign = 0; + break; + } + switch (anchor) { + case 'bottom': + case 'bottom-right': + case 'bottom-left': + verticalAlign = 1; + break; + case 'top': + case 'top-right': + case 'top-left': + verticalAlign = 0; + break; + } + return { horizontalAlign, verticalAlign }; +} +function shapeLines(shaping, glyphMap, glyphPositions, imagePositions, lines, lineHeight, textAnchor, textJustify, writingMode, spacing, allowVerticalPlacement, layoutTextSizeThisZoom) { + let x = 0; + let y = SHAPING_DEFAULT_OFFSET; + let maxLineLength = 0; + let maxLineHeight = 0; + const justify = textJustify === 'right' ? 1 : + textJustify === 'left' ? 0 : 0.5; + let lineIndex = 0; + for (const line of lines) { + line.trim(); + const lineMaxScale = line.getMaxScale(); + const maxLineOffset = (lineMaxScale - 1) * ONE_EM; + const positionedLine = { positionedGlyphs: [], lineOffset: 0 }; + shaping.positionedLines[lineIndex] = positionedLine; + const positionedGlyphs = positionedLine.positionedGlyphs; + let lineOffset = 0.0; + if (!line.length()) { + y += lineHeight; // Still need a line feed after empty line + ++lineIndex; + continue; + } + for (let i = 0; i < line.length(); i++) { + const section = line.getSection(i); + const sectionIndex = line.getSectionIndex(i); + const codePoint = line.getCharCode(i); + let baselineOffset = 0.0; + let metrics = null; + let rect = null; + let imageName = null; + let verticalAdvance = ONE_EM; + const vertical = !(writingMode === exports.WritingMode.horizontal || + // Don't verticalize glyphs that have no upright orientation if vertical placement is disabled. + (!allowVerticalPlacement && !charHasUprightVerticalOrientation(codePoint)) || + // If vertical placement is enabled, don't verticalize glyphs that + // are from complex text layout script, or whitespaces. + (allowVerticalPlacement && (whitespace[codePoint] || charInComplexShapingScript(codePoint)))); + if (!section.imageName) { + const positions = glyphPositions[section.fontStack]; + const glyphPosition = positions && positions[codePoint]; + if (glyphPosition && glyphPosition.rect) { + rect = glyphPosition.rect; + metrics = glyphPosition.metrics; + } + else { + const glyphs = glyphMap[section.fontStack]; + const glyph = glyphs && glyphs[codePoint]; + if (!glyph) + continue; + metrics = glyph.metrics; + } + // We don't know the baseline, but since we're laying out + // at 24 points, we can calculate how much it will move when + // we scale up or down. + baselineOffset = (lineMaxScale - section.scale) * ONE_EM; + } + else { + const imagePosition = imagePositions[section.imageName]; + if (!imagePosition) + continue; + imageName = section.imageName; + shaping.iconsInText = shaping.iconsInText || true; + rect = imagePosition.paddedRect; + const size = imagePosition.displaySize; + // If needed, allow to set scale factor for an image using + // alias "image-scale" that could be alias for "font-scale" + // when FormattedSection is an image section. + section.scale = section.scale * ONE_EM / layoutTextSizeThisZoom; + metrics = { width: size[0], + height: size[1], + left: IMAGE_PADDING, + top: -GLYPH_PBF_BORDER, + advance: vertical ? size[1] : size[0] }; + // Difference between one EM and an image size. + // Aligns bottom of an image to a baseline level. + const imageOffset = ONE_EM - size[1] * section.scale; + baselineOffset = maxLineOffset + imageOffset; + verticalAdvance = metrics.advance; + // Difference between height of an image and one EM at max line scale. + // Pushes current line down if an image size is over 1 EM at max line scale. + const offset = vertical ? size[0] * section.scale - ONE_EM * lineMaxScale : + size[1] * section.scale - ONE_EM * lineMaxScale; + if (offset > 0 && offset > lineOffset) { + lineOffset = offset; + } + } + if (!vertical) { + positionedGlyphs.push({ glyph: codePoint, imageName, x, y: y + baselineOffset, vertical, scale: section.scale, fontStack: section.fontStack, sectionIndex, metrics, rect }); + x += metrics.advance * section.scale + spacing; + } + else { + shaping.verticalizable = true; + positionedGlyphs.push({ glyph: codePoint, imageName, x, y: y + baselineOffset, vertical, scale: section.scale, fontStack: section.fontStack, sectionIndex, metrics, rect }); + x += verticalAdvance * section.scale + spacing; + } + } + // Only justify if we placed at least one glyph + if (positionedGlyphs.length !== 0) { + const lineLength = x - spacing; + maxLineLength = Math.max(lineLength, maxLineLength); + justifyLine(positionedGlyphs, 0, positionedGlyphs.length - 1, justify, lineOffset); + } + x = 0; + const currentLineHeight = lineHeight * lineMaxScale + lineOffset; + positionedLine.lineOffset = Math.max(lineOffset, maxLineOffset); + y += currentLineHeight; + maxLineHeight = Math.max(currentLineHeight, maxLineHeight); + ++lineIndex; + } + // Calculate the bounding box and justify / align text block. + const height = y - SHAPING_DEFAULT_OFFSET; + const { horizontalAlign, verticalAlign } = getAnchorAlignment(textAnchor); + align(shaping.positionedLines, justify, horizontalAlign, verticalAlign, maxLineLength, maxLineHeight, lineHeight, height, lines.length); + shaping.top += -verticalAlign * height; + shaping.bottom = shaping.top + height; + shaping.left += -horizontalAlign * maxLineLength; + shaping.right = shaping.left + maxLineLength; +} +// justify right = 1, left = 0, center = 0.5 +function justifyLine(positionedGlyphs, start, end, justify, lineOffset) { + if (!justify && !lineOffset) + return; + const lastPositionedGlyph = positionedGlyphs[end]; + const lastAdvance = lastPositionedGlyph.metrics.advance * lastPositionedGlyph.scale; + const lineIndent = (positionedGlyphs[end].x + lastAdvance) * justify; + for (let j = start; j <= end; j++) { + positionedGlyphs[j].x -= lineIndent; + positionedGlyphs[j].y += lineOffset; + } +} +function align(positionedLines, justify, horizontalAlign, verticalAlign, maxLineLength, maxLineHeight, lineHeight, blockHeight, lineCount) { + const shiftX = (justify - horizontalAlign) * maxLineLength; + let shiftY = 0; + if (maxLineHeight !== lineHeight) { + shiftY = -blockHeight * verticalAlign - SHAPING_DEFAULT_OFFSET; + } + else { + shiftY = (-verticalAlign * lineCount + 0.5) * lineHeight; + } + for (const line of positionedLines) { + for (const positionedGlyph of line.positionedGlyphs) { + positionedGlyph.x += shiftX; + positionedGlyph.y += shiftY; + } + } +} +function shapeIcon(image, iconOffset, iconAnchor) { + const { horizontalAlign, verticalAlign } = getAnchorAlignment(iconAnchor); + const dx = iconOffset[0]; + const dy = iconOffset[1]; + const x1 = dx - image.displaySize[0] * horizontalAlign; + const x2 = x1 + image.displaySize[0]; + const y1 = dy - image.displaySize[1] * verticalAlign; + const y2 = y1 + image.displaySize[1]; + return { image, top: y1, bottom: y2, left: x1, right: x2 }; +} +function fitIconToText(shapedIcon, shapedText, textFit, padding, iconOffset, fontScale) { + const image = shapedIcon.image; + let collisionPadding; + if (image.content) { + const content = image.content; + const pixelRatio = image.pixelRatio || 1; + collisionPadding = [ + content[0] / pixelRatio, + content[1] / pixelRatio, + image.displaySize[0] - content[2] / pixelRatio, + image.displaySize[1] - content[3] / pixelRatio + ]; + } + // We don't respect the icon-anchor, because icon-text-fit is set. Instead, + // the icon will be centered on the text, then stretched in the given + // dimensions. + const textLeft = shapedText.left * fontScale; + const textRight = shapedText.right * fontScale; + let top, right, bottom, left; + if (textFit === 'width' || textFit === 'both') { + // Stretched horizontally to the text width + left = iconOffset[0] + textLeft - padding[3]; + right = iconOffset[0] + textRight + padding[1]; + } + else { + // Centered on the text + left = iconOffset[0] + (textLeft + textRight - image.displaySize[0]) / 2; + right = left + image.displaySize[0]; + } + const textTop = shapedText.top * fontScale; + const textBottom = shapedText.bottom * fontScale; + if (textFit === 'height' || textFit === 'both') { + // Stretched vertically to the text height + top = iconOffset[1] + textTop - padding[0]; + bottom = iconOffset[1] + textBottom + padding[2]; + } + else { + // Centered on the text + top = iconOffset[1] + (textTop + textBottom - image.displaySize[1]) / 2; + bottom = top + image.displaySize[1]; + } + return { image, top, right, bottom, left, collisionPadding }; +} + +const MAX_GLYPH_ICON_SIZE = 255; +const SIZE_PACK_FACTOR = 128; +const MAX_PACKED_SIZE = MAX_GLYPH_ICON_SIZE * SIZE_PACK_FACTOR; +// For {text,icon}-size, get the bucket-level data that will be needed by +// the painter to set symbol-size-related uniforms +function getSizeData(tileZoom, value) { + const { expression } = value; + if (expression.kind === 'constant') { + const layoutSize = expression.evaluate(new EvaluationParameters(tileZoom + 1)); + return { kind: 'constant', layoutSize }; + } + else if (expression.kind === 'source') { + return { kind: 'source' }; + } + else { + const { zoomStops, interpolationType } = expression; + // calculate covering zoom stops for zoom-dependent values + let lower = 0; + while (lower < zoomStops.length && zoomStops[lower] <= tileZoom) + lower++; + lower = Math.max(0, lower - 1); + let upper = lower; + while (upper < zoomStops.length && zoomStops[upper] < tileZoom + 1) + upper++; + upper = Math.min(zoomStops.length - 1, upper); + const minZoom = zoomStops[lower]; + const maxZoom = zoomStops[upper]; + // We'd like to be able to use CameraExpression or CompositeExpression in these + // return types rather than ExpressionSpecification, but the former are not + // transferrable across Web Worker boundaries. + if (expression.kind === 'composite') { + return { kind: 'composite', minZoom, maxZoom, interpolationType }; + } + // for camera functions, also save off the function values + // evaluated at the covering zoom levels + const minSize = expression.evaluate(new EvaluationParameters(minZoom)); + const maxSize = expression.evaluate(new EvaluationParameters(maxZoom)); + return { kind: 'camera', minZoom, maxZoom, minSize, maxSize, interpolationType }; + } +} +function evaluateSizeForFeature(sizeData, { uSize, uSizeT }, { lowerSize, upperSize }) { + if (sizeData.kind === 'source') { + return lowerSize / SIZE_PACK_FACTOR; + } + else if (sizeData.kind === 'composite') { + return interpolate.number(lowerSize / SIZE_PACK_FACTOR, upperSize / SIZE_PACK_FACTOR, uSizeT); + } + return uSize; +} +function evaluateSizeForZoom(sizeData, zoom) { + let uSizeT = 0; + let uSize = 0; + if (sizeData.kind === 'constant') { + uSize = sizeData.layoutSize; + } + else if (sizeData.kind !== 'source') { + const { interpolationType, minZoom, maxZoom } = sizeData; + // Even though we could get the exact value of the camera function + // at z = tr.zoom, we intentionally do not: instead, we interpolate + // between the camera function values at a pair of zoom stops covering + // [tileZoom, tileZoom + 1] in order to be consistent with this + // restriction on composite functions + const t = !interpolationType ? 0 : clamp$1(Interpolate.interpolationFactor(interpolationType, zoom, minZoom, maxZoom), 0, 1); + if (sizeData.kind === 'camera') { + uSize = interpolate.number(sizeData.minSize, sizeData.maxSize, t); + } + else { + uSizeT = t; + } + } + return { uSizeT, uSize }; +} + +function getOverlapMode(layout, overlapProp, allowOverlapProp) { + let result = 'never'; + const overlap = layout.get(overlapProp); + if (overlap) { + // if -overlap is set, use it + result = overlap; + } + else if (layout.get(allowOverlapProp)) { + // fall back to -allow-overlap, with false='never', true='always' + result = 'always'; + } + return result; +} + +const vectorTileFeatureTypes = vectorTile.VectorTileFeature.types; +// Opacity arrays are frequently updated but don't contain a lot of information, so we pack them +// tight. Each Uint32 is actually four duplicate Uint8s for the four corners of a glyph +// 7 bits are for the current opacity, and the lowest bit is the target opacity +// actually defined in symbol_attributes.js +// const placementOpacityAttributes = [ +// { name: 'a_fade_opacity', components: 1, type: 'Uint32' } +// ]; +const shaderOpacityAttributes = [ + { name: 'a_fade_opacity', components: 1, type: 'Uint8', offset: 0 } +]; +function addVertex(array, anchorX, anchorY, ox, oy, tx, ty, sizeVertex, isSDF, pixelOffsetX, pixelOffsetY, minFontScaleX, minFontScaleY) { + const aSizeX = sizeVertex ? Math.min(MAX_PACKED_SIZE, Math.round(sizeVertex[0])) : 0; + const aSizeY = sizeVertex ? Math.min(MAX_PACKED_SIZE, Math.round(sizeVertex[1])) : 0; + array.emplaceBack( + // a_pos_offset + anchorX, anchorY, Math.round(ox * 32), Math.round(oy * 32), + // a_data + tx, // x coordinate of symbol on glyph atlas texture + ty, // y coordinate of symbol on glyph atlas texture + (aSizeX << 1) + (isSDF ? 1 : 0), aSizeY, pixelOffsetX * 16, pixelOffsetY * 16, minFontScaleX * 256, minFontScaleY * 256); +} +function addDynamicAttributes(dynamicLayoutVertexArray, p, angle) { + dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle); + dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle); + dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle); + dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle); +} +function containsRTLText(formattedText) { + for (const section of formattedText.sections) { + if (stringContainsRTLText(section.text)) { + return true; + } + } + return false; +} +class SymbolBuffers { + constructor(programConfigurations) { + this.layoutVertexArray = new SymbolLayoutArray(); + this.indexArray = new TriangleIndexArray(); + this.programConfigurations = programConfigurations; + this.segments = new SegmentVector(); + this.dynamicLayoutVertexArray = new SymbolDynamicLayoutArray(); + this.opacityVertexArray = new SymbolOpacityArray(); + this.hasVisibleVertices = false; + this.placedSymbolArray = new PlacedSymbolArray(); + } + isEmpty() { + return this.layoutVertexArray.length === 0 && + this.indexArray.length === 0 && + this.dynamicLayoutVertexArray.length === 0 && + this.opacityVertexArray.length === 0; + } + upload(context, dynamicIndexBuffer, upload, update) { + if (this.isEmpty()) { + return; + } + if (upload) { + this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, symbolLayoutAttributes.members); + this.indexBuffer = context.createIndexBuffer(this.indexArray, dynamicIndexBuffer); + this.dynamicLayoutVertexBuffer = context.createVertexBuffer(this.dynamicLayoutVertexArray, dynamicLayoutAttributes.members, true); + this.opacityVertexBuffer = context.createVertexBuffer(this.opacityVertexArray, shaderOpacityAttributes, true); + // This is a performance hack so that we can write to opacityVertexArray with uint32s + // even though the shaders read uint8s + this.opacityVertexBuffer.itemSize = 1; + } + if (upload || update) { + this.programConfigurations.upload(context); + } + } + destroy() { + if (!this.layoutVertexBuffer) + return; + this.layoutVertexBuffer.destroy(); + this.indexBuffer.destroy(); + this.programConfigurations.destroy(); + this.segments.destroy(); + this.dynamicLayoutVertexBuffer.destroy(); + this.opacityVertexBuffer.destroy(); + } +} +register('SymbolBuffers', SymbolBuffers); +class CollisionBuffers { + constructor(LayoutArray, layoutAttributes, IndexArray) { + this.layoutVertexArray = new LayoutArray(); + this.layoutAttributes = layoutAttributes; + this.indexArray = new IndexArray(); + this.segments = new SegmentVector(); + this.collisionVertexArray = new CollisionVertexArray(); + } + upload(context) { + this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, this.layoutAttributes); + this.indexBuffer = context.createIndexBuffer(this.indexArray); + this.collisionVertexBuffer = context.createVertexBuffer(this.collisionVertexArray, collisionVertexAttributes.members, true); + } + destroy() { + if (!this.layoutVertexBuffer) + return; + this.layoutVertexBuffer.destroy(); + this.indexBuffer.destroy(); + this.segments.destroy(); + this.collisionVertexBuffer.destroy(); + } +} +register('CollisionBuffers', CollisionBuffers); +/** + * @internal + * Unlike other buckets, which simply implement #addFeature with type-specific + * logic for (essentially) triangulating feature geometries, SymbolBucket + * requires specialized behavior: + * + * 1. WorkerTile#parse(), the logical owner of the bucket creation process, + * calls SymbolBucket#populate(), which resolves text and icon tokens on + * each feature, adds each glyphs and symbols needed to the passed-in + * collections options.glyphDependencies and options.iconDependencies, and + * stores the feature data for use in subsequent step (this.features). + * + * 2. WorkerTile asynchronously requests from the main thread all of the glyphs + * and icons needed (by this bucket and any others). When glyphs and icons + * have been received, the WorkerTile creates a CollisionIndex and invokes: + * + * 3. performSymbolLayout(bucket, stacks, icons) perform texts shaping and + * layout on a Symbol Bucket. This step populates: + * `this.symbolInstances`: metadata on generated symbols + * `this.collisionBoxArray`: collision data for use by foreground + * `this.text`: SymbolBuffers for text symbols + * `this.icons`: SymbolBuffers for icons + * `this.iconCollisionBox`: Debug SymbolBuffers for icon collision boxes + * `this.textCollisionBox`: Debug SymbolBuffers for text collision boxes + * The results are sent to the foreground for rendering + * + * 4. performSymbolPlacement(bucket, collisionIndex) is run on the foreground, + * and uses the CollisionIndex along with current camera settings to determine + * which symbols can actually show on the map. Collided symbols are hidden + * using a dynamic "OpacityVertexArray". + */ +class SymbolBucket { + constructor(options) { + this.collisionBoxArray = options.collisionBoxArray; + this.zoom = options.zoom; + this.overscaling = options.overscaling; + this.layers = options.layers; + this.layerIds = this.layers.map(layer => layer.id); + this.index = options.index; + this.pixelRatio = options.pixelRatio; + this.sourceLayerIndex = options.sourceLayerIndex; + this.hasPattern = false; + this.hasRTLText = false; + this.sortKeyRanges = []; + this.collisionCircleArray = []; + this.placementInvProjMatrix = identity$2([]); + this.placementViewportMatrix = identity$2([]); + const layer = this.layers[0]; + const unevaluatedLayoutValues = layer._unevaluatedLayout._values; + this.textSizeData = getSizeData(this.zoom, unevaluatedLayoutValues['text-size']); + this.iconSizeData = getSizeData(this.zoom, unevaluatedLayoutValues['icon-size']); + const layout = this.layers[0].layout; + const sortKey = layout.get('symbol-sort-key'); + const zOrder = layout.get('symbol-z-order'); + this.canOverlap = + getOverlapMode(layout, 'text-overlap', 'text-allow-overlap') !== 'never' || + getOverlapMode(layout, 'icon-overlap', 'icon-allow-overlap') !== 'never' || + layout.get('text-ignore-placement') || + layout.get('icon-ignore-placement'); + this.sortFeaturesByKey = zOrder !== 'viewport-y' && !sortKey.isConstant(); + const zOrderByViewportY = zOrder === 'viewport-y' || (zOrder === 'auto' && !this.sortFeaturesByKey); + this.sortFeaturesByY = zOrderByViewportY && this.canOverlap; + if (layout.get('symbol-placement') === 'point') { + this.writingModes = layout.get('text-writing-mode').map(wm => exports.WritingMode[wm]); + } + this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id); + this.sourceID = options.sourceID; + } + createArrays() { + this.text = new SymbolBuffers(new ProgramConfigurationSet(this.layers, this.zoom, property => /^text/.test(property))); + this.icon = new SymbolBuffers(new ProgramConfigurationSet(this.layers, this.zoom, property => /^icon/.test(property))); + this.glyphOffsetArray = new GlyphOffsetArray(); + this.lineVertexArray = new SymbolLineVertexArray(); + this.symbolInstances = new SymbolInstanceArray(); + this.textAnchorOffsets = new TextAnchorOffsetArray(); + } + calculateGlyphDependencies(text, stack, textAlongLine, allowVerticalPlacement, doesAllowVerticalWritingMode) { + for (let i = 0; i < text.length; i++) { + stack[text.charCodeAt(i)] = true; + if ((textAlongLine || allowVerticalPlacement) && doesAllowVerticalWritingMode) { + const verticalChar = verticalizedCharacterMap[text.charAt(i)]; + if (verticalChar) { + stack[verticalChar.charCodeAt(0)] = true; + } + } + } + } + populate(features, options, canonical) { + const layer = this.layers[0]; + const layout = layer.layout; + const textFont = layout.get('text-font'); + const textField = layout.get('text-field'); + const iconImage = layout.get('icon-image'); + const hasText = (textField.value.kind !== 'constant' || + (textField.value.value instanceof Formatted && !textField.value.value.isEmpty()) || + textField.value.value.toString().length > 0) && + (textFont.value.kind !== 'constant' || textFont.value.value.length > 0); + // we should always resolve the icon-image value if the property was defined in the style + // this allows us to fire the styleimagemissing event if image evaluation returns null + // the only way to distinguish between null returned from a coalesce statement with no valid images + // and null returned because icon-image wasn't defined is to check whether or not iconImage.parameters is an empty object + const hasIcon = iconImage.value.kind !== 'constant' || !!iconImage.value.value || Object.keys(iconImage.parameters).length > 0; + const symbolSortKey = layout.get('symbol-sort-key'); + this.features = []; + if (!hasText && !hasIcon) { + return; + } + const icons = options.iconDependencies; + const stacks = options.glyphDependencies; + const availableImages = options.availableImages; + const globalProperties = new EvaluationParameters(this.zoom); + for (const { feature, id, index, sourceLayerIndex } of features) { + const needGeometry = layer._featureFilter.needGeometry; + const evaluationFeature = toEvaluationFeature(feature, needGeometry); + if (!layer._featureFilter.filter(globalProperties, evaluationFeature, canonical)) { + continue; + } + if (!needGeometry) + evaluationFeature.geometry = loadGeometry(feature); + let text; + if (hasText) { + // Expression evaluation will automatically coerce to Formatted + // but plain string token evaluation skips that pathway so do the + // conversion here. + const resolvedTokens = layer.getValueAndResolveTokens('text-field', evaluationFeature, canonical, availableImages); + const formattedText = Formatted.factory(resolvedTokens); + if (containsRTLText(formattedText)) { + this.hasRTLText = true; + } + if (!this.hasRTLText || // non-rtl text so can proceed safely + getRTLTextPluginStatus() === 'unavailable' || // We don't intend to lazy-load the rtl text plugin, so proceed with incorrect shaping + this.hasRTLText && plugin.isParsed() // Use the rtlText plugin to shape text + ) { + text = transformText(formattedText, layer, evaluationFeature); + } + } + let icon; + if (hasIcon) { + // Expression evaluation will automatically coerce to Image + // but plain string token evaluation skips that pathway so do the + // conversion here. + const resolvedTokens = layer.getValueAndResolveTokens('icon-image', evaluationFeature, canonical, availableImages); + if (resolvedTokens instanceof ResolvedImage) { + icon = resolvedTokens; + } + else { + icon = ResolvedImage.fromString(resolvedTokens); + } + } + if (!text && !icon) { + continue; + } + const sortKey = this.sortFeaturesByKey ? + symbolSortKey.evaluate(evaluationFeature, {}, canonical) : + undefined; + const symbolFeature = { + id, + text, + icon, + index, + sourceLayerIndex, + geometry: evaluationFeature.geometry, + properties: feature.properties, + type: vectorTileFeatureTypes[feature.type], + sortKey + }; + this.features.push(symbolFeature); + if (icon) { + icons[icon.name] = true; + } + if (text) { + const fontStack = textFont.evaluate(evaluationFeature, {}, canonical).join(','); + const textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point'; + this.allowVerticalPlacement = this.writingModes && this.writingModes.indexOf(exports.WritingMode.vertical) >= 0; + for (const section of text.sections) { + if (!section.image) { + const doesAllowVerticalWritingMode = allowsVerticalWritingMode(text.toString()); + const sectionFont = section.fontStack || fontStack; + const sectionStack = stacks[sectionFont] = stacks[sectionFont] || {}; + this.calculateGlyphDependencies(section.text, sectionStack, textAlongLine, this.allowVerticalPlacement, doesAllowVerticalWritingMode); + } + else { + // Add section image to the list of dependencies. + icons[section.image.name] = true; + } + } + } + } + if (layout.get('symbol-placement') === 'line') { + // Merge adjacent lines with the same text to improve labelling. + // It's better to place labels on one long line than on many short segments. + this.features = mergeLines(this.features); + } + if (this.sortFeaturesByKey) { + this.features.sort((a, b) => { + // a.sortKey is always a number when sortFeaturesByKey is true + return a.sortKey - b.sortKey; + }); + } + } + update(states, vtLayer, imagePositions) { + if (!this.stateDependentLayers.length) + return; + this.text.programConfigurations.updatePaintArrays(states, vtLayer, this.layers, imagePositions); + this.icon.programConfigurations.updatePaintArrays(states, vtLayer, this.layers, imagePositions); + } + isEmpty() { + // When the bucket encounters only rtl-text but the plugin isnt loaded, no symbol instances will be created. + // In order for the bucket to be serialized, and not discarded as an empty bucket both checks are necessary. + return this.symbolInstances.length === 0 && !this.hasRTLText; + } + uploadPending() { + return !this.uploaded || this.text.programConfigurations.needsUpload || this.icon.programConfigurations.needsUpload; + } + upload(context) { + if (!this.uploaded && this.hasDebugData()) { + this.textCollisionBox.upload(context); + this.iconCollisionBox.upload(context); + } + this.text.upload(context, this.sortFeaturesByY, !this.uploaded, this.text.programConfigurations.needsUpload); + this.icon.upload(context, this.sortFeaturesByY, !this.uploaded, this.icon.programConfigurations.needsUpload); + this.uploaded = true; + } + destroyDebugData() { + this.textCollisionBox.destroy(); + this.iconCollisionBox.destroy(); + } + destroy() { + this.text.destroy(); + this.icon.destroy(); + if (this.hasDebugData()) { + this.destroyDebugData(); + } + } + addToLineVertexArray(anchor, line) { + const lineStartIndex = this.lineVertexArray.length; + if (anchor.segment !== undefined) { + let sumForwardLength = anchor.dist(line[anchor.segment + 1]); + let sumBackwardLength = anchor.dist(line[anchor.segment]); + const vertices = {}; + for (let i = anchor.segment + 1; i < line.length; i++) { + vertices[i] = { x: line[i].x, y: line[i].y, tileUnitDistanceFromAnchor: sumForwardLength }; + if (i < line.length - 1) { + sumForwardLength += line[i + 1].dist(line[i]); + } + } + for (let i = anchor.segment || 0; i >= 0; i--) { + vertices[i] = { x: line[i].x, y: line[i].y, tileUnitDistanceFromAnchor: sumBackwardLength }; + if (i > 0) { + sumBackwardLength += line[i - 1].dist(line[i]); + } + } + for (let i = 0; i < line.length; i++) { + const vertex = vertices[i]; + this.lineVertexArray.emplaceBack(vertex.x, vertex.y, vertex.tileUnitDistanceFromAnchor); + } + } + return { + lineStartIndex, + lineLength: this.lineVertexArray.length - lineStartIndex + }; + } + addSymbols(arrays, quads, sizeVertex, lineOffset, alongLine, feature, writingMode, labelAnchor, lineStartIndex, lineLength, associatedIconIndex, canonical) { + const indexArray = arrays.indexArray; + const layoutVertexArray = arrays.layoutVertexArray; + const segment = arrays.segments.prepareSegment(4 * quads.length, layoutVertexArray, indexArray, this.canOverlap ? feature.sortKey : undefined); + const glyphOffsetArrayStart = this.glyphOffsetArray.length; + const vertexStartIndex = segment.vertexLength; + const angle = (this.allowVerticalPlacement && writingMode === exports.WritingMode.vertical) ? Math.PI / 2 : 0; + const sections = feature.text && feature.text.sections; + for (let i = 0; i < quads.length; i++) { + const { tl, tr, bl, br, tex, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY, glyphOffset, isSDF, sectionIndex } = quads[i]; + const index = segment.vertexLength; + const y = glyphOffset[1]; + addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, tl.x, y + tl.y, tex.x, tex.y, sizeVertex, isSDF, pixelOffsetTL.x, pixelOffsetTL.y, minFontScaleX, minFontScaleY); + addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, tr.x, y + tr.y, tex.x + tex.w, tex.y, sizeVertex, isSDF, pixelOffsetBR.x, pixelOffsetTL.y, minFontScaleX, minFontScaleY); + addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, bl.x, y + bl.y, tex.x, tex.y + tex.h, sizeVertex, isSDF, pixelOffsetTL.x, pixelOffsetBR.y, minFontScaleX, minFontScaleY); + addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, br.x, y + br.y, tex.x + tex.w, tex.y + tex.h, sizeVertex, isSDF, pixelOffsetBR.x, pixelOffsetBR.y, minFontScaleX, minFontScaleY); + addDynamicAttributes(arrays.dynamicLayoutVertexArray, labelAnchor, angle); + indexArray.emplaceBack(index, index + 1, index + 2); + indexArray.emplaceBack(index + 1, index + 2, index + 3); + segment.vertexLength += 4; + segment.primitiveLength += 2; + this.glyphOffsetArray.emplaceBack(glyphOffset[0]); + if (i === quads.length - 1 || sectionIndex !== quads[i + 1].sectionIndex) { + arrays.programConfigurations.populatePaintArrays(layoutVertexArray.length, feature, feature.index, {}, canonical, sections && sections[sectionIndex]); + } + } + arrays.placedSymbolArray.emplaceBack(labelAnchor.x, labelAnchor.y, glyphOffsetArrayStart, this.glyphOffsetArray.length - glyphOffsetArrayStart, vertexStartIndex, lineStartIndex, lineLength, labelAnchor.segment, sizeVertex ? sizeVertex[0] : 0, sizeVertex ? sizeVertex[1] : 0, lineOffset[0], lineOffset[1], writingMode, + // placedOrientation is null initially; will be updated to horizontal(1)/vertical(2) if placed + 0, false, + // The crossTileID is only filled/used on the foreground for dynamic text anchors + 0, associatedIconIndex); + } + _addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, point, anchorX, anchorY, extrude) { + collisionVertexArray.emplaceBack(0, 0); + return layoutVertexArray.emplaceBack( + // pos + point.x, point.y, + // a_anchor_pos + anchorX, anchorY, + // extrude + Math.round(extrude.x), Math.round(extrude.y)); + } + addCollisionDebugVertices(x1, y1, x2, y2, arrays, boxAnchorPoint, symbolInstance) { + const segment = arrays.segments.prepareSegment(4, arrays.layoutVertexArray, arrays.indexArray); + const index = segment.vertexLength; + const layoutVertexArray = arrays.layoutVertexArray; + const collisionVertexArray = arrays.collisionVertexArray; + const anchorX = symbolInstance.anchorX; + const anchorY = symbolInstance.anchorY; + this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point$2(x1, y1)); + this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point$2(x2, y1)); + this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point$2(x2, y2)); + this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point$2(x1, y2)); + segment.vertexLength += 4; + const indexArray = arrays.indexArray; + indexArray.emplaceBack(index, index + 1); + indexArray.emplaceBack(index + 1, index + 2); + indexArray.emplaceBack(index + 2, index + 3); + indexArray.emplaceBack(index + 3, index); + segment.primitiveLength += 4; + } + addDebugCollisionBoxes(startIndex, endIndex, symbolInstance, isText) { + for (let b = startIndex; b < endIndex; b++) { + const box = this.collisionBoxArray.get(b); + const x1 = box.x1; + const y1 = box.y1; + const x2 = box.x2; + const y2 = box.y2; + this.addCollisionDebugVertices(x1, y1, x2, y2, isText ? this.textCollisionBox : this.iconCollisionBox, box.anchorPoint, symbolInstance); + } + } + generateCollisionDebugBuffers() { + if (this.hasDebugData()) { + this.destroyDebugData(); + } + this.textCollisionBox = new CollisionBuffers(CollisionBoxLayoutArray, collisionBoxLayout.members, LineIndexArray); + this.iconCollisionBox = new CollisionBuffers(CollisionBoxLayoutArray, collisionBoxLayout.members, LineIndexArray); + for (let i = 0; i < this.symbolInstances.length; i++) { + const symbolInstance = this.symbolInstances.get(i); + this.addDebugCollisionBoxes(symbolInstance.textBoxStartIndex, symbolInstance.textBoxEndIndex, symbolInstance, true); + this.addDebugCollisionBoxes(symbolInstance.verticalTextBoxStartIndex, symbolInstance.verticalTextBoxEndIndex, symbolInstance, true); + this.addDebugCollisionBoxes(symbolInstance.iconBoxStartIndex, symbolInstance.iconBoxEndIndex, symbolInstance, false); + this.addDebugCollisionBoxes(symbolInstance.verticalIconBoxStartIndex, symbolInstance.verticalIconBoxEndIndex, symbolInstance, false); + } + } + // These flat arrays are meant to be quicker to iterate over than the source + // CollisionBoxArray + _deserializeCollisionBoxesForSymbol(collisionBoxArray, textStartIndex, textEndIndex, verticalTextStartIndex, verticalTextEndIndex, iconStartIndex, iconEndIndex, verticalIconStartIndex, verticalIconEndIndex) { + const collisionArrays = {}; + for (let k = textStartIndex; k < textEndIndex; k++) { + const box = collisionBoxArray.get(k); + collisionArrays.textBox = { x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY }; + collisionArrays.textFeatureIndex = box.featureIndex; + break; // Only one box allowed per instance + } + for (let k = verticalTextStartIndex; k < verticalTextEndIndex; k++) { + const box = collisionBoxArray.get(k); + collisionArrays.verticalTextBox = { x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY }; + collisionArrays.verticalTextFeatureIndex = box.featureIndex; + break; // Only one box allowed per instance + } + for (let k = iconStartIndex; k < iconEndIndex; k++) { + // An icon can only have one box now, so this indexing is a bit vestigial... + const box = collisionBoxArray.get(k); + collisionArrays.iconBox = { x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY }; + collisionArrays.iconFeatureIndex = box.featureIndex; + break; // Only one box allowed per instance + } + for (let k = verticalIconStartIndex; k < verticalIconEndIndex; k++) { + // An icon can only have one box now, so this indexing is a bit vestigial... + const box = collisionBoxArray.get(k); + collisionArrays.verticalIconBox = { x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY }; + collisionArrays.verticalIconFeatureIndex = box.featureIndex; + break; // Only one box allowed per instance + } + return collisionArrays; + } + deserializeCollisionBoxes(collisionBoxArray) { + this.collisionArrays = []; + for (let i = 0; i < this.symbolInstances.length; i++) { + const symbolInstance = this.symbolInstances.get(i); + this.collisionArrays.push(this._deserializeCollisionBoxesForSymbol(collisionBoxArray, symbolInstance.textBoxStartIndex, symbolInstance.textBoxEndIndex, symbolInstance.verticalTextBoxStartIndex, symbolInstance.verticalTextBoxEndIndex, symbolInstance.iconBoxStartIndex, symbolInstance.iconBoxEndIndex, symbolInstance.verticalIconBoxStartIndex, symbolInstance.verticalIconBoxEndIndex)); + } + } + hasTextData() { + return this.text.segments.get().length > 0; + } + hasIconData() { + return this.icon.segments.get().length > 0; + } + hasDebugData() { + return this.textCollisionBox && this.iconCollisionBox; + } + hasTextCollisionBoxData() { + return this.hasDebugData() && this.textCollisionBox.segments.get().length > 0; + } + hasIconCollisionBoxData() { + return this.hasDebugData() && this.iconCollisionBox.segments.get().length > 0; + } + addIndicesForPlacedSymbol(iconOrText, placedSymbolIndex) { + const placedSymbol = iconOrText.placedSymbolArray.get(placedSymbolIndex); + const endIndex = placedSymbol.vertexStartIndex + placedSymbol.numGlyphs * 4; + for (let vertexIndex = placedSymbol.vertexStartIndex; vertexIndex < endIndex; vertexIndex += 4) { + iconOrText.indexArray.emplaceBack(vertexIndex, vertexIndex + 1, vertexIndex + 2); + iconOrText.indexArray.emplaceBack(vertexIndex + 1, vertexIndex + 2, vertexIndex + 3); + } + } + getSortedSymbolIndexes(angle) { + if (this.sortedAngle === angle && this.symbolInstanceIndexes !== undefined) { + return this.symbolInstanceIndexes; + } + const sin = Math.sin(angle); + const cos = Math.cos(angle); + const rotatedYs = []; + const featureIndexes = []; + const result = []; + for (let i = 0; i < this.symbolInstances.length; ++i) { + result.push(i); + const symbolInstance = this.symbolInstances.get(i); + rotatedYs.push(Math.round(sin * symbolInstance.anchorX + cos * symbolInstance.anchorY) | 0); + featureIndexes.push(symbolInstance.featureIndex); + } + result.sort((aIndex, bIndex) => { + return (rotatedYs[aIndex] - rotatedYs[bIndex]) || + (featureIndexes[bIndex] - featureIndexes[aIndex]); + }); + return result; + } + addToSortKeyRanges(symbolInstanceIndex, sortKey) { + const last = this.sortKeyRanges[this.sortKeyRanges.length - 1]; + if (last && last.sortKey === sortKey) { + last.symbolInstanceEnd = symbolInstanceIndex + 1; + } + else { + this.sortKeyRanges.push({ + sortKey, + symbolInstanceStart: symbolInstanceIndex, + symbolInstanceEnd: symbolInstanceIndex + 1 + }); + } + } + sortFeatures(angle) { + if (!this.sortFeaturesByY) + return; + if (this.sortedAngle === angle) + return; + // The current approach to sorting doesn't sort across segments so don't try. + // Sorting within segments separately seemed not to be worth the complexity. + if (this.text.segments.get().length > 1 || this.icon.segments.get().length > 1) + return; + // If the symbols are allowed to overlap sort them by their vertical screen position. + // The index array buffer is rewritten to reference the (unchanged) vertices in the + // sorted order. + // To avoid sorting the actual symbolInstance array we sort an array of indexes. + this.symbolInstanceIndexes = this.getSortedSymbolIndexes(angle); + this.sortedAngle = angle; + this.text.indexArray.clear(); + this.icon.indexArray.clear(); + this.featureSortOrder = []; + for (const i of this.symbolInstanceIndexes) { + const symbolInstance = this.symbolInstances.get(i); + this.featureSortOrder.push(symbolInstance.featureIndex); + [ + symbolInstance.rightJustifiedTextSymbolIndex, + symbolInstance.centerJustifiedTextSymbolIndex, + symbolInstance.leftJustifiedTextSymbolIndex + ].forEach((index, i, array) => { + // Only add a given index the first time it shows up, + // to avoid duplicate opacity entries when multiple justifications + // share the same glyphs. + if (index >= 0 && array.indexOf(index) === i) { + this.addIndicesForPlacedSymbol(this.text, index); + } + }); + if (symbolInstance.verticalPlacedTextSymbolIndex >= 0) { + this.addIndicesForPlacedSymbol(this.text, symbolInstance.verticalPlacedTextSymbolIndex); + } + if (symbolInstance.placedIconSymbolIndex >= 0) { + this.addIndicesForPlacedSymbol(this.icon, symbolInstance.placedIconSymbolIndex); + } + if (symbolInstance.verticalPlacedIconSymbolIndex >= 0) { + this.addIndicesForPlacedSymbol(this.icon, symbolInstance.verticalPlacedIconSymbolIndex); + } + } + if (this.text.indexBuffer) + this.text.indexBuffer.updateData(this.text.indexArray); + if (this.icon.indexBuffer) + this.icon.indexBuffer.updateData(this.icon.indexArray); + } +} +register('SymbolBucket', SymbolBucket, { + omit: ['layers', 'collisionBoxArray', 'features', 'compareText'] +}); +// this constant is based on the size of StructArray indexes used in a symbol +// bucket--namely, glyphOffsetArrayStart +// eg the max valid UInt16 is 65,535 +// See https://github.com/mapbox/mapbox-gl-js/issues/2907 for motivation +// lineStartIndex and textBoxStartIndex could potentially be concerns +// but we expect there to be many fewer boxes/lines than glyphs +SymbolBucket.MAX_GLYPHS = 65535; +SymbolBucket.addDynamicAttributes = addDynamicAttributes; + +/** + * Replace tokens in a string template with values in an object + * + * @param properties - a key/value relationship between tokens and replacements + * @param text - the template string + * @returns the template with tokens replaced + */ +function resolveTokens(properties, text) { + return text.replace(/{([^{}]+)}/g, (match, key) => { + return properties && key in properties ? String(properties[key]) : ''; + }); +} + +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ +let layout; +const getLayout = () => layout = layout || new Properties({ + "symbol-placement": new DataConstantProperty(v8Spec["layout_symbol"]["symbol-placement"]), + "symbol-spacing": new DataConstantProperty(v8Spec["layout_symbol"]["symbol-spacing"]), + "symbol-avoid-edges": new DataConstantProperty(v8Spec["layout_symbol"]["symbol-avoid-edges"]), + "symbol-sort-key": new DataDrivenProperty(v8Spec["layout_symbol"]["symbol-sort-key"]), + "symbol-z-order": new DataConstantProperty(v8Spec["layout_symbol"]["symbol-z-order"]), + "icon-allow-overlap": new DataConstantProperty(v8Spec["layout_symbol"]["icon-allow-overlap"]), + "icon-overlap": new DataConstantProperty(v8Spec["layout_symbol"]["icon-overlap"]), + "icon-ignore-placement": new DataConstantProperty(v8Spec["layout_symbol"]["icon-ignore-placement"]), + "icon-optional": new DataConstantProperty(v8Spec["layout_symbol"]["icon-optional"]), + "icon-rotation-alignment": new DataConstantProperty(v8Spec["layout_symbol"]["icon-rotation-alignment"]), + "icon-size": new DataDrivenProperty(v8Spec["layout_symbol"]["icon-size"]), + "icon-text-fit": new DataConstantProperty(v8Spec["layout_symbol"]["icon-text-fit"]), + "icon-text-fit-padding": new DataConstantProperty(v8Spec["layout_symbol"]["icon-text-fit-padding"]), + "icon-image": new DataDrivenProperty(v8Spec["layout_symbol"]["icon-image"]), + "icon-rotate": new DataDrivenProperty(v8Spec["layout_symbol"]["icon-rotate"]), + "icon-padding": new DataDrivenProperty(v8Spec["layout_symbol"]["icon-padding"]), + "icon-keep-upright": new DataConstantProperty(v8Spec["layout_symbol"]["icon-keep-upright"]), + "icon-offset": new DataDrivenProperty(v8Spec["layout_symbol"]["icon-offset"]), + "icon-anchor": new DataDrivenProperty(v8Spec["layout_symbol"]["icon-anchor"]), + "icon-pitch-alignment": new DataConstantProperty(v8Spec["layout_symbol"]["icon-pitch-alignment"]), + "text-pitch-alignment": new DataConstantProperty(v8Spec["layout_symbol"]["text-pitch-alignment"]), + "text-rotation-alignment": new DataConstantProperty(v8Spec["layout_symbol"]["text-rotation-alignment"]), + "text-field": new DataDrivenProperty(v8Spec["layout_symbol"]["text-field"]), + "text-font": new DataDrivenProperty(v8Spec["layout_symbol"]["text-font"]), + "text-size": new DataDrivenProperty(v8Spec["layout_symbol"]["text-size"]), + "text-max-width": new DataDrivenProperty(v8Spec["layout_symbol"]["text-max-width"]), + "text-line-height": new DataConstantProperty(v8Spec["layout_symbol"]["text-line-height"]), + "text-letter-spacing": new DataDrivenProperty(v8Spec["layout_symbol"]["text-letter-spacing"]), + "text-justify": new DataDrivenProperty(v8Spec["layout_symbol"]["text-justify"]), + "text-radial-offset": new DataDrivenProperty(v8Spec["layout_symbol"]["text-radial-offset"]), + "text-variable-anchor": new DataConstantProperty(v8Spec["layout_symbol"]["text-variable-anchor"]), + "text-variable-anchor-offset": new DataDrivenProperty(v8Spec["layout_symbol"]["text-variable-anchor-offset"]), + "text-anchor": new DataDrivenProperty(v8Spec["layout_symbol"]["text-anchor"]), + "text-max-angle": new DataConstantProperty(v8Spec["layout_symbol"]["text-max-angle"]), + "text-writing-mode": new DataConstantProperty(v8Spec["layout_symbol"]["text-writing-mode"]), + "text-rotate": new DataDrivenProperty(v8Spec["layout_symbol"]["text-rotate"]), + "text-padding": new DataConstantProperty(v8Spec["layout_symbol"]["text-padding"]), + "text-keep-upright": new DataConstantProperty(v8Spec["layout_symbol"]["text-keep-upright"]), + "text-transform": new DataDrivenProperty(v8Spec["layout_symbol"]["text-transform"]), + "text-offset": new DataDrivenProperty(v8Spec["layout_symbol"]["text-offset"]), + "text-allow-overlap": new DataConstantProperty(v8Spec["layout_symbol"]["text-allow-overlap"]), + "text-overlap": new DataConstantProperty(v8Spec["layout_symbol"]["text-overlap"]), + "text-ignore-placement": new DataConstantProperty(v8Spec["layout_symbol"]["text-ignore-placement"]), + "text-optional": new DataConstantProperty(v8Spec["layout_symbol"]["text-optional"]), +}); +let paint$2; +const getPaint$2 = () => paint$2 = paint$2 || new Properties({ + "icon-opacity": new DataDrivenProperty(v8Spec["paint_symbol"]["icon-opacity"]), + "icon-color": new DataDrivenProperty(v8Spec["paint_symbol"]["icon-color"]), + "icon-halo-color": new DataDrivenProperty(v8Spec["paint_symbol"]["icon-halo-color"]), + "icon-halo-width": new DataDrivenProperty(v8Spec["paint_symbol"]["icon-halo-width"]), + "icon-halo-blur": new DataDrivenProperty(v8Spec["paint_symbol"]["icon-halo-blur"]), + "icon-translate": new DataConstantProperty(v8Spec["paint_symbol"]["icon-translate"]), + "icon-translate-anchor": new DataConstantProperty(v8Spec["paint_symbol"]["icon-translate-anchor"]), + "text-opacity": new DataDrivenProperty(v8Spec["paint_symbol"]["text-opacity"]), + "text-color": new DataDrivenProperty(v8Spec["paint_symbol"]["text-color"], { runtimeType: ColorType, getOverride: (o) => o.textColor, hasOverride: (o) => !!o.textColor }), + "text-halo-color": new DataDrivenProperty(v8Spec["paint_symbol"]["text-halo-color"]), + "text-halo-width": new DataDrivenProperty(v8Spec["paint_symbol"]["text-halo-width"]), + "text-halo-blur": new DataDrivenProperty(v8Spec["paint_symbol"]["text-halo-blur"]), + "text-translate": new DataConstantProperty(v8Spec["paint_symbol"]["text-translate"]), + "text-translate-anchor": new DataConstantProperty(v8Spec["paint_symbol"]["text-translate-anchor"]), +}); +var properties$2 = ({ get paint() { return getPaint$2(); }, get layout() { return getLayout(); } }); + +// This is an internal expression class. It is only used in GL JS and +// has GL JS dependencies which can break the standalone style-spec module +class FormatSectionOverride { + constructor(defaultValue) { + if (defaultValue.property.overrides === undefined) + throw new Error('overrides must be provided to instantiate FormatSectionOverride class'); + this.type = defaultValue.property.overrides ? defaultValue.property.overrides.runtimeType : NullType; + this.defaultValue = defaultValue; + } + evaluate(ctx) { + if (ctx.formattedSection) { + const overrides = this.defaultValue.property.overrides; + if (overrides && overrides.hasOverride(ctx.formattedSection)) { + return overrides.getOverride(ctx.formattedSection); + } + } + if (ctx.feature && ctx.featureState) { + return this.defaultValue.evaluate(ctx.feature, ctx.featureState); + } + return this.defaultValue.property.specification.default; + } + eachChild(fn) { + if (!this.defaultValue.isConstant()) { + const expr = this.defaultValue.value; + fn(expr._styleExpression.expression); + } + } + // Cannot be statically evaluated, as the output depends on the evaluation context. + outputDefined() { + return false; + } + serialize() { + return null; + } +} +register('FormatSectionOverride', FormatSectionOverride, { omit: ['defaultValue'] }); + +class SymbolStyleLayer extends StyleLayer { + constructor(layer) { + super(layer, properties$2); + } + recalculate(parameters, availableImages) { + super.recalculate(parameters, availableImages); + if (this.layout.get('icon-rotation-alignment') === 'auto') { + if (this.layout.get('symbol-placement') !== 'point') { + this.layout._values['icon-rotation-alignment'] = 'map'; + } + else { + this.layout._values['icon-rotation-alignment'] = 'viewport'; + } + } + if (this.layout.get('text-rotation-alignment') === 'auto') { + if (this.layout.get('symbol-placement') !== 'point') { + this.layout._values['text-rotation-alignment'] = 'map'; + } + else { + this.layout._values['text-rotation-alignment'] = 'viewport'; + } + } + // If unspecified, `*-pitch-alignment` inherits `*-rotation-alignment` + if (this.layout.get('text-pitch-alignment') === 'auto') { + this.layout._values['text-pitch-alignment'] = this.layout.get('text-rotation-alignment') === 'map' ? 'map' : 'viewport'; + } + if (this.layout.get('icon-pitch-alignment') === 'auto') { + this.layout._values['icon-pitch-alignment'] = this.layout.get('icon-rotation-alignment'); + } + if (this.layout.get('symbol-placement') === 'point') { + const writingModes = this.layout.get('text-writing-mode'); + if (writingModes) { + // remove duplicates, preserving order + const deduped = []; + for (const m of writingModes) { + if (deduped.indexOf(m) < 0) + deduped.push(m); + } + this.layout._values['text-writing-mode'] = deduped; + } + else { + this.layout._values['text-writing-mode'] = ['horizontal']; + } + } + this._setPaintOverrides(); + } + getValueAndResolveTokens(name, feature, canonical, availableImages) { + const value = this.layout.get(name).evaluate(feature, {}, canonical, availableImages); + const unevaluated = this._unevaluatedLayout._values[name]; + if (!unevaluated.isDataDriven() && !isExpression(unevaluated.value) && value) { + return resolveTokens(feature.properties, value); + } + return value; + } + createBucket(parameters) { + return new SymbolBucket(parameters); + } + queryRadius() { + return 0; + } + queryIntersectsFeature() { + throw new Error('Should take a different path in FeatureIndex'); + } + _setPaintOverrides() { + for (const overridable of properties$2.paint.overridableProperties) { + if (!SymbolStyleLayer.hasPaintOverride(this.layout, overridable)) { + continue; + } + const overriden = this.paint.get(overridable); + const override = new FormatSectionOverride(overriden); + const styleExpression = new StyleExpression(override, overriden.property.specification); + let expression = null; + if (overriden.value.kind === 'constant' || overriden.value.kind === 'source') { + expression = new ZoomConstantExpression('source', styleExpression); + } + else { + expression = new ZoomDependentExpression('composite', styleExpression, overriden.value.zoomStops); + } + this.paint._values[overridable] = new PossiblyEvaluatedPropertyValue(overriden.property, expression, overriden.parameters); + } + } + _handleOverridablePaintPropertyUpdate(name, oldValue, newValue) { + if (!this.layout || oldValue.isDataDriven() || newValue.isDataDriven()) { + return false; + } + return SymbolStyleLayer.hasPaintOverride(this.layout, name); + } + static hasPaintOverride(layout, propertyName) { + const textField = layout.get('text-field'); + const property = properties$2.paint.properties[propertyName]; + let hasOverrides = false; + const checkSections = (sections) => { + for (const section of sections) { + if (property.overrides && property.overrides.hasOverride(section)) { + hasOverrides = true; + return; + } + } + }; + if (textField.value.kind === 'constant' && textField.value.value instanceof Formatted) { + checkSections(textField.value.value.sections); + } + else if (textField.value.kind === 'source') { + const checkExpression = (expression) => { + if (hasOverrides) + return; + if (expression instanceof Literal && typeOf(expression.value) === FormattedType) { + const formatted = expression.value; + checkSections(formatted.sections); + } + else if (expression instanceof FormatExpression) { + checkSections(expression.sections); + } + else { + expression.eachChild(checkExpression); + } + }; + const expr = textField.value; + if (expr._styleExpression) { + checkExpression(expr._styleExpression.expression); + } + } + return hasOverrides; + } +} +function getIconPadding(layout, feature, canonical, pixelRatio = 1) { + // Support text-padding in addition to icon-padding? Unclear how to apply asymmetric text-padding to the radius for collision circles. + const result = layout.get('icon-padding').evaluate(feature, {}, canonical); + const values = result && result.values; + return [ + values[0] * pixelRatio, + values[1] * pixelRatio, + values[2] * pixelRatio, + values[3] * pixelRatio, + ]; +} + +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ +let paint$1; +const getPaint$1 = () => paint$1 = paint$1 || new Properties({ + "background-color": new DataConstantProperty(v8Spec["paint_background"]["background-color"]), + "background-pattern": new CrossFadedProperty(v8Spec["paint_background"]["background-pattern"]), + "background-opacity": new DataConstantProperty(v8Spec["paint_background"]["background-opacity"]), +}); +var properties$1 = ({ get paint() { return getPaint$1(); } }); + +class BackgroundStyleLayer extends StyleLayer { + constructor(layer) { + super(layer, properties$1); + } +} + +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ +let paint; +const getPaint = () => paint = paint || new Properties({ + "raster-opacity": new DataConstantProperty(v8Spec["paint_raster"]["raster-opacity"]), + "raster-hue-rotate": new DataConstantProperty(v8Spec["paint_raster"]["raster-hue-rotate"]), + "raster-brightness-min": new DataConstantProperty(v8Spec["paint_raster"]["raster-brightness-min"]), + "raster-brightness-max": new DataConstantProperty(v8Spec["paint_raster"]["raster-brightness-max"]), + "raster-saturation": new DataConstantProperty(v8Spec["paint_raster"]["raster-saturation"]), + "raster-contrast": new DataConstantProperty(v8Spec["paint_raster"]["raster-contrast"]), + "raster-resampling": new DataConstantProperty(v8Spec["paint_raster"]["raster-resampling"]), + "raster-fade-duration": new DataConstantProperty(v8Spec["paint_raster"]["raster-fade-duration"]), +}); +var properties = ({ get paint() { return getPaint(); } }); + +class RasterStyleLayer extends StyleLayer { + constructor(layer) { + super(layer, properties); + } +} + +function validateCustomStyleLayer(layerObject) { + const errors = []; + const id = layerObject.id; + if (id === undefined) { + errors.push({ + message: `layers.${id}: missing required property "id"` + }); + } + if (layerObject.render === undefined) { + errors.push({ + message: `layers.${id}: missing required method "render"` + }); + } + if (layerObject.renderingMode && + layerObject.renderingMode !== '2d' && + layerObject.renderingMode !== '3d') { + errors.push({ + message: `layers.${id}: property "renderingMode" must be either "2d" or "3d"` + }); + } + return errors; +} +class CustomStyleLayer extends StyleLayer { + constructor(implementation) { + super(implementation, {}); + this.onAdd = (map) => { + if (this.implementation.onAdd) { + this.implementation.onAdd(map, map.painter.context.gl); + } + }; + this.onRemove = (map) => { + if (this.implementation.onRemove) { + this.implementation.onRemove(map, map.painter.context.gl); + } + }; + this.implementation = implementation; + } + is3D() { + return this.implementation.renderingMode === '3d'; + } + hasOffscreenPass() { + return this.implementation.prerender !== undefined; + } + recalculate() { } + updateTransitions() { } + hasTransition() { return false; } + serialize() { + throw new Error('Custom layers cannot be serialized'); + } +} + +function createStyleLayer(layer) { + if (layer.type === 'custom') { + return new CustomStyleLayer(layer); + } + switch (layer.type) { + case 'background': + return new BackgroundStyleLayer(layer); + case 'circle': + return new CircleStyleLayer(layer); + case 'fill': + return new FillStyleLayer(layer); + case 'fill-extrusion': + return new FillExtrusionStyleLayer(layer); + case 'heatmap': + return new HeatmapStyleLayer(layer); + case 'hillshade': + return new HillshadeStyleLayer(layer); + case 'line': + return new LineStyleLayer(layer); + case 'raster': + return new RasterStyleLayer(layer); + case 'symbol': + return new SymbolStyleLayer(layer); + } +} + +/** + * Invokes the wrapped function in a non-blocking way when trigger() is called. Invocation requests + * are ignored until the function was actually invoked. + */ +class ThrottledInvoker { + constructor(callback) { + this._callback = callback; + this._triggered = false; + if (typeof MessageChannel !== 'undefined') { + this._channel = new MessageChannel(); + this._channel.port2.onmessage = () => { + this._triggered = false; + this._callback(); + }; + } + } + trigger() { + if (!this._triggered) { + this._triggered = true; + if (this._channel) { + this._channel.port1.postMessage(true); + } + else { + setTimeout(() => { + this._triggered = false; + this._callback(); + }, 0); + } + } + } + remove() { + delete this._channel; + this._callback = () => { }; + } +} + +/** + * An implementation of the [Actor design pattern](http://en.wikipedia.org/wiki/Actor_model) + * that maintains the relationship between asynchronous tasks and the objects + * that spin them off - in this case, tasks like parsing parts of styles, + * owned by the styles + */ +class Actor { + /** + * @param target - The target + * @param parent - The parent + * @param mapId - A unique identifier for the Map instance using this Actor. + */ + constructor(target, parent, mapId) { + this.receive = (message) => { + const data = message.data; + const id = data.id; + if (!id) { + return; + } + if (data.targetMapId && this.mapId !== data.targetMapId) { + return; + } + if (data.type === '') { + // Remove the original request from the queue. This is only possible if it + // hasn't been kicked off yet. The id will remain in the queue, but because + // there is no associated task, it will be dropped once it's time to execute it. + delete this.tasks[id]; + const cancel = this.cancelCallbacks[id]; + delete this.cancelCallbacks[id]; + if (cancel) { + cancel(); + } + } + else { + if (isWorker() || data.mustQueue) { + // In workers, store the tasks that we need to process before actually processing them. This + // is necessary because we want to keep receiving messages, and in particular, + // messages. Some tasks may take a while in the worker thread, so before + // executing the next task in our queue, postMessage preempts this and + // messages can be processed. We're using a MessageChannel object to get throttle the + // process() flow to one at a time. + this.tasks[id] = data; + this.taskQueue.push(id); + this.invoker.trigger(); + } + else { + // In the main thread, process messages immediately so that other work does not slip in + // between getting partial data back from workers. + this.processTask(id, data); + } + } + }; + this.process = () => { + if (!this.taskQueue.length) { + return; + } + const id = this.taskQueue.shift(); + const task = this.tasks[id]; + delete this.tasks[id]; + // Schedule another process call if we know there's more to process _before_ invoking the + // current task. This is necessary so that processing continues even if the current task + // doesn't execute successfully. + if (this.taskQueue.length) { + this.invoker.trigger(); + } + if (!task) { + // If the task ID doesn't have associated task data anymore, it was canceled. + return; + } + this.processTask(id, task); + }; + this.target = target; + this.parent = parent; + this.mapId = mapId; + this.callbacks = {}; + this.tasks = {}; + this.taskQueue = []; + this.cancelCallbacks = {}; + this.invoker = new ThrottledInvoker(this.process); + this.target.addEventListener('message', this.receive, false); + this.globalScope = isWorker() ? target : window; + } + /** + * Sends a message from a main-thread map to a Worker or from a Worker back to + * a main-thread map instance. + * + * @param type - The name of the target method to invoke or '[source-type].[source-name].name' for a method on a WorkerSource. + * @param targetMapId - A particular mapId to which to send this message. + */ + send(type, data, callback, targetMapId, mustQueue = false) { + // We're using a string ID instead of numbers because they are being used as object keys + // anyway, and thus stringified implicitly. We use random IDs because an actor may receive + // message from multiple other actors which could run in different execution context. A + // linearly increasing ID could produce collisions. + const id = Math.round((Math.random() * 1e18)).toString(36).substring(0, 10); + if (callback) { + this.callbacks[id] = callback; + } + const buffers = []; + const message = { + id, + type, + hasCallback: !!callback, + targetMapId, + mustQueue, + sourceMapId: this.mapId, + data: serialize(data, buffers) + }; + this.target.postMessage(message, { transfer: buffers }); + return { + cancel: () => { + if (callback) { + // Set the callback to null so that it never fires after the request is aborted. + delete this.callbacks[id]; + } + const cancelMessage = { + id, + type: '', + targetMapId, + sourceMapId: this.mapId + }; + this.target.postMessage(cancelMessage); + } + }; + } + processTask(id, task) { + if (task.type === '') { + // The done() function in the counterpart has been called, and we are now + // firing the callback in the originating actor, if there is one. + const callback = this.callbacks[id]; + delete this.callbacks[id]; + if (callback) { + // If we get a response, but don't have a callback, the request was canceled. + if (task.error) { + callback(deserialize(task.error)); + } + else { + callback(null, deserialize(task.data)); + } + } + } + else { + let completed = false; + const buffers = []; + const done = task.hasCallback ? (err, data) => { + completed = true; + delete this.cancelCallbacks[id]; + const responseMessage = { + id, + type: '', + sourceMapId: this.mapId, + error: err ? serialize(err) : null, + data: serialize(data, buffers) + }; + this.target.postMessage(responseMessage, { transfer: buffers }); + } : (_) => { + completed = true; + }; + let callback = null; + const params = deserialize(task.data); + if (this.parent[task.type]) { + // task.type == 'loadTile', 'removeTile', etc. + callback = this.parent[task.type](task.sourceMapId, params, done); + } + else if ('getWorkerSource' in this.parent) { + // task.type == sourcetype.method + const keys = task.type.split('.'); + const scope = this.parent.getWorkerSource(task.sourceMapId, keys[0], params.source); + callback = scope[keys[1]](params, done); + } + else { + // No function was found. + done(new Error(`Could not find function ${task.type}`)); + } + if (!completed && callback && callback.cancel) { + // Allows canceling the task as long as it hasn't been completed yet. + this.cancelCallbacks[id] = callback.cancel; + } + } + } + remove() { + this.invoker.remove(); + this.target.removeEventListener('message', this.receive, false); + } +} + +/* +* Approximate radius of the earth in meters. +* Uses the WGS-84 approximation. The radius at the equator is ~6378137 and at the poles is ~6356752. https://en.wikipedia.org/wiki/World_Geodetic_System#WGS84 +* 6371008.8 is one published "average radius" see https://en.wikipedia.org/wiki/Earth_radius#Mean_radius, or ftp://athena.fsv.cvut.cz/ZFG/grs80-Moritz.pdf p.4 +*/ +const earthRadius = 6371008.8; +/** + * A `LngLat` object represents a given longitude and latitude coordinate, measured in degrees. + * These coordinates are based on the [WGS84 (EPSG:4326) standard](https://en.wikipedia.org/wiki/World_Geodetic_System#WGS84). + * + * MapLibre GL JS uses longitude, latitude coordinate order (as opposed to latitude, longitude) to match the + * [GeoJSON specification](https://tools.ietf.org/html/rfc7946). + * + * Note that any MapLibre GL JS method that accepts a `LngLat` object as an argument or option + * can also accept an `Array` of two numbers and will perform an implicit conversion. + * This flexible type is documented as {@link LngLatLike}. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let ll = new maplibregl.LngLat(-123.9749, 40.7736); + * ll.lng; // = -123.9749 + * ``` + * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/) + * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/) + * @see [Create a timeline animation](https://maplibre.org/maplibre-gl-js/docs/examples/timeline-animation/) + */ +class LngLat { + /** + * @param lng - Longitude, measured in degrees. + * @param lat - Latitude, measured in degrees. + */ + constructor(lng, lat) { + if (isNaN(lng) || isNaN(lat)) { + throw new Error(`Invalid LngLat object: (${lng}, ${lat})`); + } + this.lng = +lng; + this.lat = +lat; + if (this.lat > 90 || this.lat < -90) { + throw new Error('Invalid LngLat latitude value: must be between -90 and 90'); + } + } + /** + * Returns a new `LngLat` object whose longitude is wrapped to the range (-180, 180). + * + * @returns The wrapped `LngLat` object. + * @example + * ```ts + * let ll = new maplibregl.LngLat(286.0251, 40.7736); + * let wrapped = ll.wrap(); + * wrapped.lng; // = -73.9749 + * ``` + */ + wrap() { + return new LngLat(wrap(this.lng, -180, 180), this.lat); + } + /** + * Returns the coordinates represented as an array of two numbers. + * + * @returns The coordinates represented as an array of longitude and latitude. + * @example + * ```ts + * let ll = new maplibregl.LngLat(-73.9749, 40.7736); + * ll.toArray(); // = [-73.9749, 40.7736] + * ``` + */ + toArray() { + return [this.lng, this.lat]; + } + /** + * Returns the coordinates represent as a string. + * + * @returns The coordinates represented as a string of the format `'LngLat(lng, lat)'`. + * @example + * ```ts + * let ll = new maplibregl.LngLat(-73.9749, 40.7736); + * ll.toString(); // = "LngLat(-73.9749, 40.7736)" + * ``` + */ + toString() { + return `LngLat(${this.lng}, ${this.lat})`; + } + /** + * Returns the approximate distance between a pair of coordinates in meters + * Uses the Haversine Formula (from R.W. Sinnott, "Virtues of the Haversine", Sky and Telescope, vol. 68, no. 2, 1984, p. 159) + * + * @param lngLat - coordinates to compute the distance to + * @returns Distance in meters between the two coordinates. + * @example + * ```ts + * let new_york = new maplibregl.LngLat(-74.0060, 40.7128); + * let los_angeles = new maplibregl.LngLat(-118.2437, 34.0522); + * new_york.distanceTo(los_angeles); // = 3935751.690893987, "true distance" using a non-spherical approximation is ~3966km + * ``` + */ + distanceTo(lngLat) { + const rad = Math.PI / 180; + const lat1 = this.lat * rad; + const lat2 = lngLat.lat * rad; + const a = Math.sin(lat1) * Math.sin(lat2) + Math.cos(lat1) * Math.cos(lat2) * Math.cos((lngLat.lng - this.lng) * rad); + const maxMeters = earthRadius * Math.acos(Math.min(a, 1)); + return maxMeters; + } + /** + * Converts an array of two numbers or an object with `lng` and `lat` or `lon` and `lat` properties + * to a `LngLat` object. + * + * If a `LngLat` object is passed in, the function returns it unchanged. + * + * @param input - An array of two numbers or object to convert, or a `LngLat` object to return. + * @returns A new `LngLat` object, if a conversion occurred, or the original `LngLat` object. + * @example + * ```ts + * let arr = [-73.9749, 40.7736]; + * let ll = maplibregl.LngLat.convert(arr); + * ll; // = LngLat {lng: -73.9749, lat: 40.7736} + * ``` + */ + static convert(input) { + if (input instanceof LngLat) { + return input; + } + if (Array.isArray(input) && (input.length === 2 || input.length === 3)) { + return new LngLat(Number(input[0]), Number(input[1])); + } + if (!Array.isArray(input) && typeof input === 'object' && input !== null) { + return new LngLat( + // flow can't refine this to have one of lng or lat, so we have to cast to any + Number('lng' in input ? input.lng : input.lon), Number(input.lat)); + } + throw new Error('`LngLatLike` argument must be specified as a LngLat instance, an object {lng: , lat: }, an object {lon: , lat: }, or an array of [, ]'); + } +} + +/* + * The average circumference of the world in meters. + */ +const earthCircumfrence = 2 * Math.PI * earthRadius; // meters +/* + * The circumference at a line of latitude in meters. + */ +function circumferenceAtLatitude(latitude) { + return earthCircumfrence * Math.cos(latitude * Math.PI / 180); +} +function mercatorXfromLng(lng) { + return (180 + lng) / 360; +} +function mercatorYfromLat(lat) { + return (180 - (180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360)))) / 360; +} +function mercatorZfromAltitude(altitude, lat) { + return altitude / circumferenceAtLatitude(lat); +} +function lngFromMercatorX(x) { + return x * 360 - 180; +} +function latFromMercatorY(y) { + const y2 = 180 - y * 360; + return 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90; +} +function altitudeFromMercatorZ(z, y) { + return z * circumferenceAtLatitude(latFromMercatorY(y)); +} +/** + * Determine the Mercator scale factor for a given latitude, see + * https://en.wikipedia.org/wiki/Mercator_projection#Scale_factor + * + * At the equator the scale factor will be 1, which increases at higher latitudes. + * + * @param lat - Latitude + * @returns scale factor + */ +function mercatorScale(lat) { + return 1 / Math.cos(lat * Math.PI / 180); +} +/** + * A `MercatorCoordinate` object represents a projected three dimensional position. + * + * `MercatorCoordinate` uses the web mercator projection ([EPSG:3857](https://epsg.io/3857)) with slightly different units: + * - the size of 1 unit is the width of the projected world instead of the "mercator meter" + * - the origin of the coordinate space is at the north-west corner instead of the middle + * + * For example, `MercatorCoordinate(0, 0, 0)` is the north-west corner of the mercator world and + * `MercatorCoordinate(1, 1, 0)` is the south-east corner. If you are familiar with + * [vector tiles](https://github.com/mapbox/vector-tile-spec) it may be helpful to think + * of the coordinate space as the `0/0/0` tile with an extent of `1`. + * + * The `z` dimension of `MercatorCoordinate` is conformal. A cube in the mercator coordinate space would be rendered as a cube. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let nullIsland = new maplibregl.MercatorCoordinate(0.5, 0.5, 0); + * ``` + * @see [Add a custom style layer](https://maplibre.org/maplibre-gl-js/docs/examples/custom-style-layer/) + */ +class MercatorCoordinate { + /** + * @param x - The x component of the position. + * @param y - The y component of the position. + * @param z - The z component of the position. + */ + constructor(x, y, z = 0) { + this.x = +x; + this.y = +y; + this.z = +z; + } + /** + * Project a `LngLat` to a `MercatorCoordinate`. + * + * @param lngLatLike - The location to project. + * @param altitude - The altitude in meters of the position. + * @returns The projected mercator coordinate. + * @example + * ```ts + * let coord = maplibregl.MercatorCoordinate.fromLngLat({ lng: 0, lat: 0}, 0); + * coord; // MercatorCoordinate(0.5, 0.5, 0) + * ``` + */ + static fromLngLat(lngLatLike, altitude = 0) { + const lngLat = LngLat.convert(lngLatLike); + return new MercatorCoordinate(mercatorXfromLng(lngLat.lng), mercatorYfromLat(lngLat.lat), mercatorZfromAltitude(altitude, lngLat.lat)); + } + /** + * Returns the `LngLat` for the coordinate. + * + * @returns The `LngLat` object. + * @example + * ```ts + * let coord = new maplibregl.MercatorCoordinate(0.5, 0.5, 0); + * let lngLat = coord.toLngLat(); // LngLat(0, 0) + * ``` + */ + toLngLat() { + return new LngLat(lngFromMercatorX(this.x), latFromMercatorY(this.y)); + } + /** + * Returns the altitude in meters of the coordinate. + * + * @returns The altitude in meters. + * @example + * ```ts + * let coord = new maplibregl.MercatorCoordinate(0, 0, 0.02); + * coord.toAltitude(); // 6914.281956295339 + * ``` + */ + toAltitude() { + return altitudeFromMercatorZ(this.z, this.y); + } + /** + * Returns the distance of 1 meter in `MercatorCoordinate` units at this latitude. + * + * For coordinates in real world units using meters, this naturally provides the scale + * to transform into `MercatorCoordinate`s. + * + * @returns Distance of 1 meter in `MercatorCoordinate` units. + */ + meterInMercatorCoordinateUnits() { + // 1 meter / circumference at equator in meters * Mercator projection scale factor at this latitude + return 1 / earthCircumfrence * mercatorScale(latFromMercatorY(this.y)); + } +} + +/** + * getURL + * + * @param {String} baseUrl Base url of the WMS server + * @param {String} layer Layer name + * @param {Number} x Tile coordinate x + * @param {Number} y Tile coordinate y + * @param {Number} z Tile zoom + * @param {Object} [options] + * @param {String} [options.format='image/png'] + * @param {String} [options.service='WMS'] + * @param {String} [options.version='1.1.1'] + * @param {String} [options.request='GetMap'] + * @param {String} [options.srs='EPSG:3857'] + * @param {Number} [options.width='256'] + * @param {Number} [options.height='256'] + * @returns {String} url + * @example + * var baseUrl = 'http://geodata.state.nj.us/imagerywms/Natural2015'; + * var layer = 'Natural2015'; + * var url = whoots.getURL(baseUrl, layer, 154308, 197167, 19); + */ +function getURL(baseUrl, layer, x, y, z, options) { + options = options || {}; + + var url = baseUrl + '?' + [ + 'bbox=' + getTileBBox(x, y, z), + 'format=' + (options.format || 'image/png'), + 'service=' + (options.service || 'WMS'), + 'version=' + (options.version || '1.1.1'), + 'request=' + (options.request || 'GetMap'), + 'srs=' + (options.srs || 'EPSG:3857'), + 'width=' + (options.width || 256), + 'height=' + (options.height || 256), + 'layers=' + layer + ].join('&'); + + return url; +} + + +/** + * getTileBBox + * + * @param {Number} x Tile coordinate x + * @param {Number} y Tile coordinate y + * @param {Number} z Tile zoom + * @returns {String} String of the bounding box + */ +function getTileBBox(x, y, z) { + // for Google/OSM tile scheme we need to alter the y + y = (Math.pow(2, z) - y - 1); + + var min = getMercCoords(x * 256, y * 256, z), + max = getMercCoords((x + 1) * 256, (y + 1) * 256, z); + + return min[0] + ',' + min[1] + ',' + max[0] + ',' + max[1]; +} + + +/** + * getMercCoords + * + * @param {Number} x Pixel coordinate x + * @param {Number} y Pixel coordinate y + * @param {Number} z Tile zoom + * @returns {Array} [x, y] + */ +function getMercCoords(x, y, z) { + var resolution = (2 * Math.PI * 6378137 / 256) / Math.pow(2, z), + merc_x = (x * resolution - 2 * Math.PI * 6378137 / 2.0), + merc_y = (y * resolution - 2 * Math.PI * 6378137 / 2.0); + + return [merc_x, merc_y]; +} + +/** + * A canonical way to define a tile ID + */ +class CanonicalTileID { + constructor(z, x, y) { + if (z < 0 || z > 25 || y < 0 || y >= Math.pow(2, z) || x < 0 || x >= Math.pow(2, z)) { + throw new Error(`x=${x}, y=${y}, z=${z} outside of bounds. 0<=x<${Math.pow(2, z)}, 0<=y<${Math.pow(2, z)} 0<=z<=25 `); + } + this.z = z; + this.x = x; + this.y = y; + this.key = calculateKey(0, z, z, x, y); + } + equals(id) { + return this.z === id.z && this.x === id.x && this.y === id.y; + } + // given a list of urls, choose a url template and return a tile URL + url(urls, pixelRatio, scheme) { + const bbox = getTileBBox(this.x, this.y, this.z); + const quadkey = getQuadkey(this.z, this.x, this.y); + return urls[(this.x + this.y) % urls.length] + .replace(/{prefix}/g, (this.x % 16).toString(16) + (this.y % 16).toString(16)) + .replace(/{z}/g, String(this.z)) + .replace(/{x}/g, String(this.x)) + .replace(/{y}/g, String(scheme === 'tms' ? (Math.pow(2, this.z) - this.y - 1) : this.y)) + .replace(/{ratio}/g, pixelRatio > 1 ? '@2x' : '') + .replace(/{quadkey}/g, quadkey) + .replace(/{bbox-epsg-3857}/g, bbox); + } + isChildOf(parent) { + const dz = this.z - parent.z; + return dz > 0 && parent.x === (this.x >> dz) && parent.y === (this.y >> dz); + } + getTilePoint(coord) { + const tilesAtZoom = Math.pow(2, this.z); + return new Point$2((coord.x * tilesAtZoom - this.x) * EXTENT, (coord.y * tilesAtZoom - this.y) * EXTENT); + } + toString() { + return `${this.z}/${this.x}/${this.y}`; + } +} +/** + * @internal + * An unwrapped tile identifier + */ +class UnwrappedTileID { + constructor(wrap, canonical) { + this.wrap = wrap; + this.canonical = canonical; + this.key = calculateKey(wrap, canonical.z, canonical.z, canonical.x, canonical.y); + } +} +/** + * An overscaled tile identifier + */ +class OverscaledTileID { + constructor(overscaledZ, wrap, z, x, y) { + if (overscaledZ < z) + throw new Error(`overscaledZ should be >= z; overscaledZ = ${overscaledZ}; z = ${z}`); + this.overscaledZ = overscaledZ; + this.wrap = wrap; + this.canonical = new CanonicalTileID(z, +x, +y); + this.key = calculateKey(wrap, overscaledZ, z, x, y); + } + clone() { + return new OverscaledTileID(this.overscaledZ, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y); + } + equals(id) { + return this.overscaledZ === id.overscaledZ && this.wrap === id.wrap && this.canonical.equals(id.canonical); + } + scaledTo(targetZ) { + if (targetZ > this.overscaledZ) + throw new Error(`targetZ > this.overscaledZ; targetZ = ${targetZ}; overscaledZ = ${this.overscaledZ}`); + const zDifference = this.canonical.z - targetZ; + if (targetZ > this.canonical.z) { + return new OverscaledTileID(targetZ, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y); + } + else { + return new OverscaledTileID(targetZ, this.wrap, targetZ, this.canonical.x >> zDifference, this.canonical.y >> zDifference); + } + } + /* + * calculateScaledKey is an optimization: + * when withWrap == true, implements the same as this.scaledTo(z).key, + * when withWrap == false, implements the same as this.scaledTo(z).wrapped().key. + */ + calculateScaledKey(targetZ, withWrap) { + if (targetZ > this.overscaledZ) + throw new Error(`targetZ > this.overscaledZ; targetZ = ${targetZ}; overscaledZ = ${this.overscaledZ}`); + const zDifference = this.canonical.z - targetZ; + if (targetZ > this.canonical.z) { + return calculateKey(this.wrap * +withWrap, targetZ, this.canonical.z, this.canonical.x, this.canonical.y); + } + else { + return calculateKey(this.wrap * +withWrap, targetZ, targetZ, this.canonical.x >> zDifference, this.canonical.y >> zDifference); + } + } + isChildOf(parent) { + if (parent.wrap !== this.wrap) { + // We can't be a child if we're in a different world copy + return false; + } + const zDifference = this.canonical.z - parent.canonical.z; + // We're first testing for z == 0, to avoid a 32 bit shift, which is undefined. + return parent.overscaledZ === 0 || (parent.overscaledZ < this.overscaledZ && + parent.canonical.x === (this.canonical.x >> zDifference) && + parent.canonical.y === (this.canonical.y >> zDifference)); + } + children(sourceMaxZoom) { + if (this.overscaledZ >= sourceMaxZoom) { + // return a single tile coord representing a an overscaled tile + return [new OverscaledTileID(this.overscaledZ + 1, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y)]; + } + const z = this.canonical.z + 1; + const x = this.canonical.x * 2; + const y = this.canonical.y * 2; + return [ + new OverscaledTileID(z, this.wrap, z, x, y), + new OverscaledTileID(z, this.wrap, z, x + 1, y), + new OverscaledTileID(z, this.wrap, z, x, y + 1), + new OverscaledTileID(z, this.wrap, z, x + 1, y + 1) + ]; + } + isLessThan(rhs) { + if (this.wrap < rhs.wrap) + return true; + if (this.wrap > rhs.wrap) + return false; + if (this.overscaledZ < rhs.overscaledZ) + return true; + if (this.overscaledZ > rhs.overscaledZ) + return false; + if (this.canonical.x < rhs.canonical.x) + return true; + if (this.canonical.x > rhs.canonical.x) + return false; + if (this.canonical.y < rhs.canonical.y) + return true; + return false; + } + wrapped() { + return new OverscaledTileID(this.overscaledZ, 0, this.canonical.z, this.canonical.x, this.canonical.y); + } + unwrapTo(wrap) { + return new OverscaledTileID(this.overscaledZ, wrap, this.canonical.z, this.canonical.x, this.canonical.y); + } + overscaleFactor() { + return Math.pow(2, this.overscaledZ - this.canonical.z); + } + toUnwrapped() { + return new UnwrappedTileID(this.wrap, this.canonical); + } + toString() { + return `${this.overscaledZ}/${this.canonical.x}/${this.canonical.y}`; + } + getTilePoint(coord) { + return this.canonical.getTilePoint(new MercatorCoordinate(coord.x - this.wrap, coord.y)); + } +} +function calculateKey(wrap, overscaledZ, z, x, y) { + wrap *= 2; + if (wrap < 0) + wrap = wrap * -1 - 1; + const dim = 1 << z; + return (dim * dim * wrap + dim * y + x).toString(36) + z.toString(36) + overscaledZ.toString(36); +} +function getQuadkey(z, x, y) { + let quadkey = '', mask; + for (let i = z; i > 0; i--) { + mask = 1 << (i - 1); + quadkey += ((x & mask ? 1 : 0) + (y & mask ? 2 : 0)); + } + return quadkey; +} +register('CanonicalTileID', CanonicalTileID); +register('OverscaledTileID', OverscaledTileID, { omit: ['posMatrix'] }); + +class DEMData { + // RGBAImage data has uniform 1px padding on all sides: square tile edge size defines stride + // and dim is calculated as stride - 2. + constructor(uid, data, encoding, redFactor = 1.0, greenFactor = 1.0, blueFactor = 1.0, baseShift = 0.0) { + this.uid = uid; + if (data.height !== data.width) + throw new RangeError('DEM tiles must be square'); + if (encoding && !['mapbox', 'terrarium', 'custom'].includes(encoding)) { + warnOnce(`"${encoding}" is not a valid encoding type. Valid types include "mapbox", "terrarium" and "custom".`); + return; + } + this.stride = data.height; + const dim = this.dim = data.height - 2; + this.data = new Uint32Array(data.data.buffer); + switch (encoding) { + case 'terrarium': + // unpacking formula for mapzen terrarium: + // https://aws.amazon.com/public-datasets/terrain/ + this.redFactor = 256.0; + this.greenFactor = 1.0; + this.blueFactor = 1.0 / 256.0; + this.baseShift = 32768.0; + break; + case 'custom': + this.redFactor = redFactor; + this.greenFactor = greenFactor; + this.blueFactor = blueFactor; + this.baseShift = baseShift; + break; + case 'mapbox': + default: + // unpacking formula for mapbox.terrain-rgb: + // https://www.mapbox.com/help/access-elevation-data/#mapbox-terrain-rgb + this.redFactor = 6553.6; + this.greenFactor = 25.6; + this.blueFactor = 0.1; + this.baseShift = 10000.0; + break; + } + // in order to avoid flashing seams between tiles, here we are initially populating a 1px border of pixels around the image + // with the data of the nearest pixel from the image. this data is eventually replaced when the tile's neighboring + // tiles are loaded and the accurate data can be backfilled using DEMData#backfillBorder + for (let x = 0; x < dim; x++) { + // left vertical border + this.data[this._idx(-1, x)] = this.data[this._idx(0, x)]; + // right vertical border + this.data[this._idx(dim, x)] = this.data[this._idx(dim - 1, x)]; + // left horizontal border + this.data[this._idx(x, -1)] = this.data[this._idx(x, 0)]; + // right horizontal border + this.data[this._idx(x, dim)] = this.data[this._idx(x, dim - 1)]; + } + // corners + this.data[this._idx(-1, -1)] = this.data[this._idx(0, 0)]; + this.data[this._idx(dim, -1)] = this.data[this._idx(dim - 1, 0)]; + this.data[this._idx(-1, dim)] = this.data[this._idx(0, dim - 1)]; + this.data[this._idx(dim, dim)] = this.data[this._idx(dim - 1, dim - 1)]; + // calculate min/max values + this.min = Number.MAX_SAFE_INTEGER; + this.max = Number.MIN_SAFE_INTEGER; + for (let x = 0; x < dim; x++) { + for (let y = 0; y < dim; y++) { + const ele = this.get(x, y); + if (ele > this.max) + this.max = ele; + if (ele < this.min) + this.min = ele; + } + } + } + get(x, y) { + const pixels = new Uint8Array(this.data.buffer); + const index = this._idx(x, y) * 4; + return this.unpack(pixels[index], pixels[index + 1], pixels[index + 2]); + } + getUnpackVector() { + return [this.redFactor, this.greenFactor, this.blueFactor, this.baseShift]; + } + _idx(x, y) { + if (x < -1 || x >= this.dim + 1 || y < -1 || y >= this.dim + 1) + throw new RangeError('out of range source coordinates for DEM data'); + return (y + 1) * this.stride + (x + 1); + } + unpack(r, g, b) { + return (r * this.redFactor + g * this.greenFactor + b * this.blueFactor - this.baseShift); + } + getPixels() { + return new RGBAImage({ width: this.stride, height: this.stride }, new Uint8Array(this.data.buffer)); + } + backfillBorder(borderTile, dx, dy) { + if (this.dim !== borderTile.dim) + throw new Error('dem dimension mismatch'); + let xMin = dx * this.dim, xMax = dx * this.dim + this.dim, yMin = dy * this.dim, yMax = dy * this.dim + this.dim; + switch (dx) { + case -1: + xMin = xMax - 1; + break; + case 1: + xMax = xMin + 1; + break; + } + switch (dy) { + case -1: + yMin = yMax - 1; + break; + case 1: + yMax = yMin + 1; + break; + } + const ox = -dx * this.dim; + const oy = -dy * this.dim; + for (let y = yMin; y < yMax; y++) { + for (let x = xMin; x < xMax; x++) { + this.data[this._idx(x, y)] = borderTile.data[this._idx(x + ox, y + oy)]; + } + } + } +} +register('DEMData', DEMData); + +class DictionaryCoder { + constructor(strings) { + this._stringToNumber = {}; + this._numberToString = []; + for (let i = 0; i < strings.length; i++) { + const string = strings[i]; + this._stringToNumber[string] = i; + this._numberToString[i] = string; + } + } + encode(string) { + return this._stringToNumber[string]; + } + decode(n) { + if (n >= this._numberToString.length) + throw new Error(`Out of bounds. Index requested n=${n} can't be >= this._numberToString.length ${this._numberToString.length}`); + return this._numberToString[n]; + } +} + +/** + * A geojson feature + */ +class GeoJSONFeature { + constructor(vectorTileFeature, z, x, y, id) { + this.type = 'Feature'; + this._vectorTileFeature = vectorTileFeature; + vectorTileFeature._z = z; + vectorTileFeature._x = x; + vectorTileFeature._y = y; + this.properties = vectorTileFeature.properties; + this.id = id; + } + get geometry() { + if (this._geometry === undefined) { + this._geometry = this._vectorTileFeature.toGeoJSON(this._vectorTileFeature._x, this._vectorTileFeature._y, this._vectorTileFeature._z).geometry; + } + return this._geometry; + } + set geometry(g) { + this._geometry = g; + } + toJSON() { + const json = { + geometry: this.geometry + }; + for (const i in this) { + if (i === '_geometry' || i === '_vectorTileFeature') + continue; + json[i] = (this)[i]; + } + return json; + } +} + +/** + * @internal + * An in memory index class to allow fast interaction with features + */ +class FeatureIndex { + constructor(tileID, promoteId) { + this.tileID = tileID; + this.x = tileID.canonical.x; + this.y = tileID.canonical.y; + this.z = tileID.canonical.z; + this.grid = new TransferableGridIndex(EXTENT, 16, 0); + this.grid3D = new TransferableGridIndex(EXTENT, 16, 0); + this.featureIndexArray = new FeatureIndexArray(); + this.promoteId = promoteId; + } + insert(feature, geometry, featureIndex, sourceLayerIndex, bucketIndex, is3D) { + const key = this.featureIndexArray.length; + this.featureIndexArray.emplaceBack(featureIndex, sourceLayerIndex, bucketIndex); + const grid = is3D ? this.grid3D : this.grid; + for (let r = 0; r < geometry.length; r++) { + const ring = geometry[r]; + const bbox = [Infinity, Infinity, -Infinity, -Infinity]; + for (let i = 0; i < ring.length; i++) { + const p = ring[i]; + bbox[0] = Math.min(bbox[0], p.x); + bbox[1] = Math.min(bbox[1], p.y); + bbox[2] = Math.max(bbox[2], p.x); + bbox[3] = Math.max(bbox[3], p.y); + } + if (bbox[0] < EXTENT && + bbox[1] < EXTENT && + bbox[2] >= 0 && + bbox[3] >= 0) { + grid.insert(key, bbox[0], bbox[1], bbox[2], bbox[3]); + } + } + } + loadVTLayers() { + if (!this.vtLayers) { + this.vtLayers = new vectorTile.VectorTile(new Protobuf(this.rawTileData)).layers; + this.sourceLayerCoder = new DictionaryCoder(this.vtLayers ? Object.keys(this.vtLayers).sort() : ['_geojsonTileLayer']); + } + return this.vtLayers; + } + // Finds non-symbol features in this tile at a particular position. + query(args, styleLayers, serializedLayers, sourceFeatureState) { + this.loadVTLayers(); + const params = args.params || {}, pixelsToTileUnits = EXTENT / args.tileSize / args.scale, filter = createFilter(params.filter); + const queryGeometry = args.queryGeometry; + const queryPadding = args.queryPadding * pixelsToTileUnits; + const bounds = getBounds(queryGeometry); + const matching = this.grid.query(bounds.minX - queryPadding, bounds.minY - queryPadding, bounds.maxX + queryPadding, bounds.maxY + queryPadding); + const cameraBounds = getBounds(args.cameraQueryGeometry); + const matching3D = this.grid3D.query(cameraBounds.minX - queryPadding, cameraBounds.minY - queryPadding, cameraBounds.maxX + queryPadding, cameraBounds.maxY + queryPadding, (bx1, by1, bx2, by2) => { + return polygonIntersectsBox(args.cameraQueryGeometry, bx1 - queryPadding, by1 - queryPadding, bx2 + queryPadding, by2 + queryPadding); + }); + for (const key of matching3D) { + matching.push(key); + } + matching.sort(topDownFeatureComparator); + const result = {}; + let previousIndex; + for (let k = 0; k < matching.length; k++) { + const index = matching[k]; + // don't check the same feature more than once + if (index === previousIndex) + continue; + previousIndex = index; + const match = this.featureIndexArray.get(index); + let featureGeometry = null; + this.loadMatchingFeature(result, match.bucketIndex, match.sourceLayerIndex, match.featureIndex, filter, params.layers, params.availableImages, styleLayers, serializedLayers, sourceFeatureState, (feature, styleLayer, featureState) => { + if (!featureGeometry) { + featureGeometry = loadGeometry(feature); + } + return styleLayer.queryIntersectsFeature(queryGeometry, feature, featureState, featureGeometry, this.z, args.transform, pixelsToTileUnits, args.pixelPosMatrix); + }); + } + return result; + } + loadMatchingFeature(result, bucketIndex, sourceLayerIndex, featureIndex, filter, filterLayerIDs, availableImages, styleLayers, serializedLayers, sourceFeatureState, intersectionTest) { + const layerIDs = this.bucketLayerIDs[bucketIndex]; + if (filterLayerIDs && !arraysIntersect(filterLayerIDs, layerIDs)) + return; + const sourceLayerName = this.sourceLayerCoder.decode(sourceLayerIndex); + const sourceLayer = this.vtLayers[sourceLayerName]; + const feature = sourceLayer.feature(featureIndex); + if (filter.needGeometry) { + const evaluationFeature = toEvaluationFeature(feature, true); + if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), evaluationFeature, this.tileID.canonical)) { + return; + } + } + else if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), feature)) { + return; + } + const id = this.getId(feature, sourceLayerName); + for (let l = 0; l < layerIDs.length; l++) { + const layerID = layerIDs[l]; + if (filterLayerIDs && filterLayerIDs.indexOf(layerID) < 0) { + continue; + } + const styleLayer = styleLayers[layerID]; + if (!styleLayer) + continue; + let featureState = {}; + if (id && sourceFeatureState) { + // `feature-state` expression evaluation requires feature state to be available + featureState = sourceFeatureState.getState(styleLayer.sourceLayer || '_geojsonTileLayer', id); + } + const serializedLayer = extend({}, serializedLayers[layerID]); + serializedLayer.paint = evaluateProperties(serializedLayer.paint, styleLayer.paint, feature, featureState, availableImages); + serializedLayer.layout = evaluateProperties(serializedLayer.layout, styleLayer.layout, feature, featureState, availableImages); + const intersectionZ = !intersectionTest || intersectionTest(feature, styleLayer, featureState); + if (!intersectionZ) { + // Only applied for non-symbol features + continue; + } + const geojsonFeature = new GeoJSONFeature(feature, this.z, this.x, this.y, id); + geojsonFeature.layer = serializedLayer; + let layerResult = result[layerID]; + if (layerResult === undefined) { + layerResult = result[layerID] = []; + } + layerResult.push({ featureIndex, feature: geojsonFeature, intersectionZ }); + } + } + // Given a set of symbol indexes that have already been looked up, + // return a matching set of GeoJSONFeatures + lookupSymbolFeatures(symbolFeatureIndexes, serializedLayers, bucketIndex, sourceLayerIndex, filterSpec, filterLayerIDs, availableImages, styleLayers) { + const result = {}; + this.loadVTLayers(); + const filter = createFilter(filterSpec); + for (const symbolFeatureIndex of symbolFeatureIndexes) { + this.loadMatchingFeature(result, bucketIndex, sourceLayerIndex, symbolFeatureIndex, filter, filterLayerIDs, availableImages, styleLayers, serializedLayers); + } + return result; + } + hasLayer(id) { + for (const layerIDs of this.bucketLayerIDs) { + for (const layerID of layerIDs) { + if (id === layerID) + return true; + } + } + return false; + } + getId(feature, sourceLayerId) { + let id = feature.id; + if (this.promoteId) { + const propName = typeof this.promoteId === 'string' ? this.promoteId : this.promoteId[sourceLayerId]; + id = feature.properties[propName]; + if (typeof id === 'boolean') + id = Number(id); + } + return id; + } +} +register('FeatureIndex', FeatureIndex, { omit: ['rawTileData', 'sourceLayerCoder'] }); +function evaluateProperties(serializedProperties, styleLayerProperties, feature, featureState, availableImages) { + return mapObject(serializedProperties, (property, key) => { + const prop = styleLayerProperties instanceof PossiblyEvaluated ? styleLayerProperties.get(key) : null; + return prop && prop.evaluate ? prop.evaluate(feature, featureState, availableImages) : prop; + }); +} +function getBounds(geometry) { + let minX = Infinity; + let minY = Infinity; + let maxX = -Infinity; + let maxY = -Infinity; + for (const p of geometry) { + minX = Math.min(minX, p.x); + minY = Math.min(minY, p.y); + maxX = Math.max(maxX, p.x); + maxY = Math.max(maxY, p.y); + } + return { minX, minY, maxX, maxY }; +} +function topDownFeatureComparator(a, b) { + return b - a; +} + +/** + * Returns the part of a multiline that intersects with the provided rectangular box. + * + * @param lines - the lines to check + * @param x1 - the left edge of the box + * @param y1 - the top edge of the box + * @param x2 - the right edge of the box + * @param y2 - the bottom edge of the box + * @returns lines + */ +function clipLine(lines, x1, y1, x2, y2) { + const clippedLines = []; + for (let l = 0; l < lines.length; l++) { + const line = lines[l]; + let clippedLine; + for (let i = 0; i < line.length - 1; i++) { + let p0 = line[i]; + let p1 = line[i + 1]; + if (p0.x < x1 && p1.x < x1) { + continue; + } + else if (p0.x < x1) { + p0 = new Point$2(x1, p0.y + (p1.y - p0.y) * ((x1 - p0.x) / (p1.x - p0.x)))._round(); + } + else if (p1.x < x1) { + p1 = new Point$2(x1, p0.y + (p1.y - p0.y) * ((x1 - p0.x) / (p1.x - p0.x)))._round(); + } + if (p0.y < y1 && p1.y < y1) { + continue; + } + else if (p0.y < y1) { + p0 = new Point$2(p0.x + (p1.x - p0.x) * ((y1 - p0.y) / (p1.y - p0.y)), y1)._round(); + } + else if (p1.y < y1) { + p1 = new Point$2(p0.x + (p1.x - p0.x) * ((y1 - p0.y) / (p1.y - p0.y)), y1)._round(); + } + if (p0.x >= x2 && p1.x >= x2) { + continue; + } + else if (p0.x >= x2) { + p0 = new Point$2(x2, p0.y + (p1.y - p0.y) * ((x2 - p0.x) / (p1.x - p0.x)))._round(); + } + else if (p1.x >= x2) { + p1 = new Point$2(x2, p0.y + (p1.y - p0.y) * ((x2 - p0.x) / (p1.x - p0.x)))._round(); + } + if (p0.y >= y2 && p1.y >= y2) { + continue; + } + else if (p0.y >= y2) { + p0 = new Point$2(p0.x + (p1.x - p0.x) * ((y2 - p0.y) / (p1.y - p0.y)), y2)._round(); + } + else if (p1.y >= y2) { + p1 = new Point$2(p0.x + (p1.x - p0.x) * ((y2 - p0.y) / (p1.y - p0.y)), y2)._round(); + } + if (!clippedLine || !p0.equals(clippedLine[clippedLine.length - 1])) { + clippedLine = [p0]; + clippedLines.push(clippedLine); + } + clippedLine.push(p1); + } + } + return clippedLines; +} + +class Anchor extends Point$2 { + constructor(x, y, angle, segment) { + super(x, y); + this.angle = angle; + if (segment !== undefined) { + this.segment = segment; + } + } + clone() { + return new Anchor(this.x, this.y, this.angle, this.segment); + } +} +register('Anchor', Anchor); + +/** + * Labels placed around really sharp angles aren't readable. Check if any + * part of the potential label has a combined angle that is too big. + * + * @param line - The line to check + * @param anchor - The point on the line around which the label is anchored. + * @param labelLength - The length of the label in geometry units. + * @param windowSize - The check fails if the combined angles within a part of the line that is `windowSize` long is too big. + * @param maxAngle - The maximum combined angle that any window along the label is allowed to have. + * + * @returns whether the label should be placed + */ +function checkMaxAngle(line, anchor, labelLength, windowSize, maxAngle) { + // horizontal labels and labels with length 0 always pass + if (anchor.segment === undefined || labelLength === 0) + return true; + let p = anchor; + let index = anchor.segment + 1; + let anchorDistance = 0; + // move backwards along the line to the first segment the label appears on + while (anchorDistance > -labelLength / 2) { + index--; + // there isn't enough room for the label after the beginning of the line + if (index < 0) + return false; + anchorDistance -= line[index].dist(p); + p = line[index]; + } + anchorDistance += line[index].dist(line[index + 1]); + index++; + // store recent corners and their total angle difference + const recentCorners = []; + let recentAngleDelta = 0; + // move forwards by the length of the label and check angles along the way + while (anchorDistance < labelLength / 2) { + const prev = line[index - 1]; + const current = line[index]; + const next = line[index + 1]; + // there isn't enough room for the label before the end of the line + if (!next) + return false; + let angleDelta = prev.angleTo(current) - current.angleTo(next); + // restrict angle to -pi..pi range + angleDelta = Math.abs(((angleDelta + 3 * Math.PI) % (Math.PI * 2)) - Math.PI); + recentCorners.push({ + distance: anchorDistance, + angleDelta + }); + recentAngleDelta += angleDelta; + // remove corners that are far enough away from the list of recent anchors + while (anchorDistance - recentCorners[0].distance > windowSize) { + recentAngleDelta -= recentCorners.shift().angleDelta; + } + // the sum of angles within the window area exceeds the maximum allowed value. check fails. + if (recentAngleDelta > maxAngle) + return false; + index++; + anchorDistance += current.dist(next); + } + // no part of the line had an angle greater than the maximum allowed. check passes. + return true; +} + +function getLineLength(line) { + let lineLength = 0; + for (let k = 0; k < line.length - 1; k++) { + lineLength += line[k].dist(line[k + 1]); + } + return lineLength; +} +function getAngleWindowSize(shapedText, glyphSize, boxScale) { + return shapedText ? + 3 / 5 * glyphSize * boxScale : + 0; +} +function getShapedLabelLength(shapedText, shapedIcon) { + return Math.max(shapedText ? shapedText.right - shapedText.left : 0, shapedIcon ? shapedIcon.right - shapedIcon.left : 0); +} +function getCenterAnchor(line, maxAngle, shapedText, shapedIcon, glyphSize, boxScale) { + const angleWindowSize = getAngleWindowSize(shapedText, glyphSize, boxScale); + const labelLength = getShapedLabelLength(shapedText, shapedIcon) * boxScale; + let prevDistance = 0; + const centerDistance = getLineLength(line) / 2; + for (let i = 0; i < line.length - 1; i++) { + const a = line[i], b = line[i + 1]; + const segmentDistance = a.dist(b); + if (prevDistance + segmentDistance > centerDistance) { + // The center is on this segment + const t = (centerDistance - prevDistance) / segmentDistance, x = interpolate.number(a.x, b.x, t), y = interpolate.number(a.y, b.y, t); + const anchor = new Anchor(x, y, b.angleTo(a), i); + anchor._round(); + if (!angleWindowSize || checkMaxAngle(line, anchor, labelLength, angleWindowSize, maxAngle)) { + return anchor; + } + else { + return; + } + } + prevDistance += segmentDistance; + } +} +function getAnchors(line, spacing, maxAngle, shapedText, shapedIcon, glyphSize, boxScale, overscaling, tileExtent) { + // Resample a line to get anchor points for labels and check that each + // potential label passes text-max-angle check and has enough froom to fit + // on the line. + const angleWindowSize = getAngleWindowSize(shapedText, glyphSize, boxScale); + const shapedLabelLength = getShapedLabelLength(shapedText, shapedIcon); + const labelLength = shapedLabelLength * boxScale; + // Is the line continued from outside the tile boundary? + const isLineContinued = line[0].x === 0 || line[0].x === tileExtent || line[0].y === 0 || line[0].y === tileExtent; + // Is the label long, relative to the spacing? + // If so, adjust the spacing so there is always a minimum space of `spacing / 4` between label edges. + if (spacing - labelLength < spacing / 4) { + spacing = labelLength + spacing / 4; + } + // Offset the first anchor by: + // Either half the label length plus a fixed extra offset if the line is not continued + // Or half the spacing if the line is continued. + // For non-continued lines, add a bit of fixed extra offset to avoid collisions at T intersections. + const fixedExtraOffset = glyphSize * 2; + const offset = !isLineContinued ? + ((shapedLabelLength / 2 + fixedExtraOffset) * boxScale * overscaling) % spacing : + (spacing / 2 * overscaling) % spacing; + return resample(line, offset, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, false, tileExtent); +} +function resample(line, offset, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, placeAtMiddle, tileExtent) { + const halfLabelLength = labelLength / 2; + const lineLength = getLineLength(line); + let distance = 0, markedDistance = offset - spacing; + let anchors = []; + for (let i = 0; i < line.length - 1; i++) { + const a = line[i], b = line[i + 1]; + const segmentDist = a.dist(b), angle = b.angleTo(a); + while (markedDistance + spacing < distance + segmentDist) { + markedDistance += spacing; + const t = (markedDistance - distance) / segmentDist, x = interpolate.number(a.x, b.x, t), y = interpolate.number(a.y, b.y, t); + // Check that the point is within the tile boundaries and that + // the label would fit before the beginning and end of the line + // if placed at this point. + if (x >= 0 && x < tileExtent && y >= 0 && y < tileExtent && + markedDistance - halfLabelLength >= 0 && + markedDistance + halfLabelLength <= lineLength) { + const anchor = new Anchor(x, y, angle, i); + anchor._round(); + if (!angleWindowSize || checkMaxAngle(line, anchor, labelLength, angleWindowSize, maxAngle)) { + anchors.push(anchor); + } + } + } + distance += segmentDist; + } + if (!placeAtMiddle && !anchors.length && !isLineContinued) { + // The first attempt at finding anchors at which labels can be placed failed. + // Try again, but this time just try placing one anchor at the middle of the line. + // This has the most effect for short lines in overscaled tiles, since the + // initial offset used in overscaled tiles is calculated to align labels with positions in + // parent tiles instead of placing the label as close to the beginning as possible. + anchors = resample(line, distance / 2, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, true, tileExtent); + } + return anchors; +} + +// If you have a 10px icon that isn't perfectly aligned to the pixel grid it will cover 11 actual +// pixels. The quad needs to be padded to account for this, otherwise they'll look slightly clipped +// on one edge in some cases. +const border = IMAGE_PADDING; +/** + * Create the quads used for rendering an icon. + */ +function getIconQuads(shapedIcon, iconRotate, isSDFIcon, hasIconTextFit) { + const quads = []; + const image = shapedIcon.image; + const pixelRatio = image.pixelRatio; + const imageWidth = image.paddedRect.w - 2 * border; + const imageHeight = image.paddedRect.h - 2 * border; + const iconWidth = shapedIcon.right - shapedIcon.left; + const iconHeight = shapedIcon.bottom - shapedIcon.top; + const stretchX = image.stretchX || [[0, imageWidth]]; + const stretchY = image.stretchY || [[0, imageHeight]]; + const reduceRanges = (sum, range) => sum + range[1] - range[0]; + const stretchWidth = stretchX.reduce(reduceRanges, 0); + const stretchHeight = stretchY.reduce(reduceRanges, 0); + const fixedWidth = imageWidth - stretchWidth; + const fixedHeight = imageHeight - stretchHeight; + let stretchOffsetX = 0; + let stretchContentWidth = stretchWidth; + let stretchOffsetY = 0; + let stretchContentHeight = stretchHeight; + let fixedOffsetX = 0; + let fixedContentWidth = fixedWidth; + let fixedOffsetY = 0; + let fixedContentHeight = fixedHeight; + if (image.content && hasIconTextFit) { + const content = image.content; + stretchOffsetX = sumWithinRange(stretchX, 0, content[0]); + stretchOffsetY = sumWithinRange(stretchY, 0, content[1]); + stretchContentWidth = sumWithinRange(stretchX, content[0], content[2]); + stretchContentHeight = sumWithinRange(stretchY, content[1], content[3]); + fixedOffsetX = content[0] - stretchOffsetX; + fixedOffsetY = content[1] - stretchOffsetY; + fixedContentWidth = content[2] - content[0] - stretchContentWidth; + fixedContentHeight = content[3] - content[1] - stretchContentHeight; + } + const makeBox = (left, top, right, bottom) => { + const leftEm = getEmOffset(left.stretch - stretchOffsetX, stretchContentWidth, iconWidth, shapedIcon.left); + const leftPx = getPxOffset(left.fixed - fixedOffsetX, fixedContentWidth, left.stretch, stretchWidth); + const topEm = getEmOffset(top.stretch - stretchOffsetY, stretchContentHeight, iconHeight, shapedIcon.top); + const topPx = getPxOffset(top.fixed - fixedOffsetY, fixedContentHeight, top.stretch, stretchHeight); + const rightEm = getEmOffset(right.stretch - stretchOffsetX, stretchContentWidth, iconWidth, shapedIcon.left); + const rightPx = getPxOffset(right.fixed - fixedOffsetX, fixedContentWidth, right.stretch, stretchWidth); + const bottomEm = getEmOffset(bottom.stretch - stretchOffsetY, stretchContentHeight, iconHeight, shapedIcon.top); + const bottomPx = getPxOffset(bottom.fixed - fixedOffsetY, fixedContentHeight, bottom.stretch, stretchHeight); + const tl = new Point$2(leftEm, topEm); + const tr = new Point$2(rightEm, topEm); + const br = new Point$2(rightEm, bottomEm); + const bl = new Point$2(leftEm, bottomEm); + const pixelOffsetTL = new Point$2(leftPx / pixelRatio, topPx / pixelRatio); + const pixelOffsetBR = new Point$2(rightPx / pixelRatio, bottomPx / pixelRatio); + const angle = iconRotate * Math.PI / 180; + if (angle) { + const sin = Math.sin(angle), cos = Math.cos(angle), matrix = [cos, -sin, sin, cos]; + tl._matMult(matrix); + tr._matMult(matrix); + bl._matMult(matrix); + br._matMult(matrix); + } + const x1 = left.stretch + left.fixed; + const x2 = right.stretch + right.fixed; + const y1 = top.stretch + top.fixed; + const y2 = bottom.stretch + bottom.fixed; + const subRect = { + x: image.paddedRect.x + border + x1, + y: image.paddedRect.y + border + y1, + w: x2 - x1, + h: y2 - y1 + }; + const minFontScaleX = fixedContentWidth / pixelRatio / iconWidth; + const minFontScaleY = fixedContentHeight / pixelRatio / iconHeight; + // Icon quad is padded, so texture coordinates also need to be padded. + return { tl, tr, bl, br, tex: subRect, writingMode: undefined, glyphOffset: [0, 0], sectionIndex: 0, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY, isSDF: isSDFIcon }; + }; + if (!hasIconTextFit || (!image.stretchX && !image.stretchY)) { + quads.push(makeBox({ fixed: 0, stretch: -1 }, { fixed: 0, stretch: -1 }, { fixed: 0, stretch: imageWidth + 1 }, { fixed: 0, stretch: imageHeight + 1 })); + } + else { + const xCuts = stretchZonesToCuts(stretchX, fixedWidth, stretchWidth); + const yCuts = stretchZonesToCuts(stretchY, fixedHeight, stretchHeight); + for (let xi = 0; xi < xCuts.length - 1; xi++) { + const x1 = xCuts[xi]; + const x2 = xCuts[xi + 1]; + for (let yi = 0; yi < yCuts.length - 1; yi++) { + const y1 = yCuts[yi]; + const y2 = yCuts[yi + 1]; + quads.push(makeBox(x1, y1, x2, y2)); + } + } + } + return quads; +} +function sumWithinRange(ranges, min, max) { + let sum = 0; + for (const range of ranges) { + sum += Math.max(min, Math.min(max, range[1])) - Math.max(min, Math.min(max, range[0])); + } + return sum; +} +function stretchZonesToCuts(stretchZones, fixedSize, stretchSize) { + const cuts = [{ fixed: -border, stretch: 0 }]; + for (const [c1, c2] of stretchZones) { + const last = cuts[cuts.length - 1]; + cuts.push({ + fixed: c1 - last.stretch, + stretch: last.stretch + }); + cuts.push({ + fixed: c1 - last.stretch, + stretch: last.stretch + (c2 - c1) + }); + } + cuts.push({ + fixed: fixedSize + border, + stretch: stretchSize + }); + return cuts; +} +function getEmOffset(stretchOffset, stretchSize, iconSize, iconOffset) { + return stretchOffset / stretchSize * iconSize + iconOffset; +} +function getPxOffset(fixedOffset, fixedSize, stretchOffset, stretchSize) { + return fixedOffset - fixedSize * stretchOffset / stretchSize; +} +/** + * Create the quads used for rendering a text label. + */ +function getGlyphQuads(anchor, shaping, textOffset, layer, alongLine, feature, imageMap, allowVerticalPlacement) { + const textRotate = layer.layout.get('text-rotate').evaluate(feature, {}) * Math.PI / 180; + const quads = []; + for (const line of shaping.positionedLines) { + for (const positionedGlyph of line.positionedGlyphs) { + if (!positionedGlyph.rect) + continue; + const textureRect = positionedGlyph.rect || {}; + // The rects have an additional buffer that is not included in their size. + const glyphPadding = 1.0; + let rectBuffer = GLYPH_PBF_BORDER + glyphPadding; + let isSDF = true; + let pixelRatio = 1.0; + let lineOffset = 0.0; + const rotateVerticalGlyph = (alongLine || allowVerticalPlacement) && positionedGlyph.vertical; + const halfAdvance = positionedGlyph.metrics.advance * positionedGlyph.scale / 2; + // Align images and scaled glyphs in the middle of a vertical line. + if (allowVerticalPlacement && shaping.verticalizable) { + const scaledGlyphOffset = (positionedGlyph.scale - 1) * ONE_EM; + const imageOffset = (ONE_EM - positionedGlyph.metrics.width * positionedGlyph.scale) / 2; + lineOffset = line.lineOffset / 2 - (positionedGlyph.imageName ? -imageOffset : scaledGlyphOffset); + } + if (positionedGlyph.imageName) { + const image = imageMap[positionedGlyph.imageName]; + isSDF = image.sdf; + pixelRatio = image.pixelRatio; + rectBuffer = IMAGE_PADDING / pixelRatio; + } + const glyphOffset = alongLine ? + [positionedGlyph.x + halfAdvance, positionedGlyph.y] : + [0, 0]; + let builtInOffset = alongLine ? + [0, 0] : + [positionedGlyph.x + halfAdvance + textOffset[0], positionedGlyph.y + textOffset[1] - lineOffset]; + let verticalizedLabelOffset = [0, 0]; + if (rotateVerticalGlyph) { + // Vertical POI labels that are rotated 90deg CW and whose glyphs must preserve upright orientation + // need to be rotated 90deg CCW. After a quad is rotated, it is translated to the original built-in offset. + verticalizedLabelOffset = builtInOffset; + builtInOffset = [0, 0]; + } + const textureScale = positionedGlyph.metrics.isDoubleResolution ? 2 : 1; + const x1 = (positionedGlyph.metrics.left - rectBuffer) * positionedGlyph.scale - halfAdvance + builtInOffset[0]; + const y1 = (-positionedGlyph.metrics.top - rectBuffer) * positionedGlyph.scale + builtInOffset[1]; + const x2 = x1 + textureRect.w / textureScale * positionedGlyph.scale / pixelRatio; + const y2 = y1 + textureRect.h / textureScale * positionedGlyph.scale / pixelRatio; + const tl = new Point$2(x1, y1); + const tr = new Point$2(x2, y1); + const bl = new Point$2(x1, y2); + const br = new Point$2(x2, y2); + if (rotateVerticalGlyph) { + // Vertical-supporting glyphs are laid out in 24x24 point boxes (1 square em) + // In horizontal orientation, the y values for glyphs are below the midline + // and we use a "yOffset" of -17 to pull them up to the middle. + // By rotating counter-clockwise around the point at the center of the left + // edge of a 24x24 layout box centered below the midline, we align the center + // of the glyphs with the horizontal midline, so the yOffset is no longer + // necessary, but we also pull the glyph to the left along the x axis. + // The y coordinate includes baseline yOffset, thus needs to be accounted + // for when glyph is rotated and translated. + const center = new Point$2(-halfAdvance, halfAdvance - SHAPING_DEFAULT_OFFSET); + const verticalRotation = -Math.PI / 2; + // xHalfWidthOffsetCorrection is a difference between full-width and half-width + // advance, should be 0 for full-width glyphs and will pull up half-width glyphs. + const xHalfWidthOffsetCorrection = ONE_EM / 2 - halfAdvance; + const yImageOffsetCorrection = positionedGlyph.imageName ? xHalfWidthOffsetCorrection : 0.0; + const halfWidthOffsetCorrection = new Point$2(5 - SHAPING_DEFAULT_OFFSET - xHalfWidthOffsetCorrection, -yImageOffsetCorrection); + const verticalOffsetCorrection = new Point$2(...verticalizedLabelOffset); + tl._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection); + tr._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection); + bl._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection); + br._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection); + } + if (textRotate) { + const sin = Math.sin(textRotate), cos = Math.cos(textRotate), matrix = [cos, -sin, sin, cos]; + tl._matMult(matrix); + tr._matMult(matrix); + bl._matMult(matrix); + br._matMult(matrix); + } + const pixelOffsetTL = new Point$2(0, 0); + const pixelOffsetBR = new Point$2(0, 0); + const minFontScaleX = 0; + const minFontScaleY = 0; + quads.push({ tl, tr, bl, br, tex: textureRect, writingMode: shaping.writingMode, glyphOffset, sectionIndex: positionedGlyph.sectionIndex, isSDF, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY }); + } + } + return quads; +} + +/** + * A CollisionFeature represents the area of the tile covered by a single label. + * It is used with CollisionIndex to check if the label overlaps with any + * previous labels. A CollisionFeature is mostly just a set of CollisionBox + * objects. + */ +class CollisionFeature { + /** + * Create a CollisionFeature, adding its collision box data to the given collisionBoxArray in the process. + * For line aligned labels a collision circle diameter is computed instead. + * + * @param anchor - The point along the line around which the label is anchored. + * @param shaped - The text or icon shaping results. + * @param boxScale - A magic number used to convert from glyph metrics units to geometry units. + * @param padding - The amount of padding to add around the label edges. + * @param alignLine - Whether the label is aligned with the line or the viewport. + */ + constructor(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shaped, boxScale, padding, alignLine, rotate) { + this.boxStartIndex = collisionBoxArray.length; + if (alignLine) { + // Compute height of the shape in glyph metrics and apply collision padding. + // Note that the pixel based 'text-padding' is applied at runtime + let top = shaped.top; + let bottom = shaped.bottom; + const collisionPadding = shaped.collisionPadding; + if (collisionPadding) { + top -= collisionPadding[1]; + bottom += collisionPadding[3]; + } + let height = bottom - top; + if (height > 0) { + // set minimum box height to avoid very many small labels + height = Math.max(10, height); + this.circleDiameter = height; + } + } + else { + // margin is in CSS order: [top, right, bottom, left] + let y1 = shaped.top * boxScale - padding[0]; + let y2 = shaped.bottom * boxScale + padding[2]; + let x1 = shaped.left * boxScale - padding[3]; + let x2 = shaped.right * boxScale + padding[1]; + const collisionPadding = shaped.collisionPadding; + if (collisionPadding) { + x1 -= collisionPadding[0] * boxScale; + y1 -= collisionPadding[1] * boxScale; + x2 += collisionPadding[2] * boxScale; + y2 += collisionPadding[3] * boxScale; + } + if (rotate) { + // Account for *-rotate in point collision boxes + // See https://github.com/mapbox/mapbox-gl-js/issues/6075 + // Doesn't account for icon-text-fit + const tl = new Point$2(x1, y1); + const tr = new Point$2(x2, y1); + const bl = new Point$2(x1, y2); + const br = new Point$2(x2, y2); + const rotateRadians = rotate * Math.PI / 180; + tl._rotate(rotateRadians); + tr._rotate(rotateRadians); + bl._rotate(rotateRadians); + br._rotate(rotateRadians); + // Collision features require an "on-axis" geometry, + // so take the envelope of the rotated geometry + // (may be quite large for wide labels rotated 45 degrees) + x1 = Math.min(tl.x, tr.x, bl.x, br.x); + x2 = Math.max(tl.x, tr.x, bl.x, br.x); + y1 = Math.min(tl.y, tr.y, bl.y, br.y); + y2 = Math.max(tl.y, tr.y, bl.y, br.y); + } + collisionBoxArray.emplaceBack(anchor.x, anchor.y, x1, y1, x2, y2, featureIndex, sourceLayerIndex, bucketIndex); + } + this.boxEndIndex = collisionBoxArray.length; + } +} + +class TinyQueue { + constructor(data = [], compare = defaultCompare) { + this.data = data; + this.length = this.data.length; + this.compare = compare; + + if (this.length > 0) { + for (let i = (this.length >> 1) - 1; i >= 0; i--) this._down(i); + } + } + + push(item) { + this.data.push(item); + this.length++; + this._up(this.length - 1); + } + + pop() { + if (this.length === 0) return undefined; + + const top = this.data[0]; + const bottom = this.data.pop(); + this.length--; + + if (this.length > 0) { + this.data[0] = bottom; + this._down(0); + } + + return top; + } + + peek() { + return this.data[0]; + } + + _up(pos) { + const {data, compare} = this; + const item = data[pos]; + + while (pos > 0) { + const parent = (pos - 1) >> 1; + const current = data[parent]; + if (compare(item, current) >= 0) break; + data[pos] = current; + pos = parent; + } + + data[pos] = item; + } + + _down(pos) { + const {data, compare} = this; + const halfLength = this.length >> 1; + const item = data[pos]; + + while (pos < halfLength) { + let left = (pos << 1) + 1; + let best = data[left]; + const right = left + 1; + + if (right < this.length && compare(data[right], best) < 0) { + left = right; + best = data[right]; + } + if (compare(best, item) >= 0) break; + + data[pos] = best; + pos = left; + } + + data[pos] = item; + } +} + +function defaultCompare(a, b) { + return a < b ? -1 : a > b ? 1 : 0; +} + +/** + * Finds an approximation of a polygon's Pole Of Inaccessibiliy https://en.wikipedia.org/wiki/Pole_of_inaccessibility + * This is a copy of http://github.com/mapbox/polylabel adapted to use Points + * + * @param polygonRings - first item in array is the outer ring followed optionally by the list of holes, should be an element of the result of util/classify_rings + * @param precision - Specified in input coordinate units. If 0 returns after first run, if `> 0` repeatedly narrows the search space until the radius of the area searched for the best pole is less than precision + * @param debug - Print some statistics to the console during execution + * @returns Pole of Inaccessibiliy. + */ +function findPoleOfInaccessibility(polygonRings, precision = 1, debug = false) { + // find the bounding box of the outer ring + let minX = Infinity, minY = Infinity, maxX = -Infinity, maxY = -Infinity; + const outerRing = polygonRings[0]; + for (let i = 0; i < outerRing.length; i++) { + const p = outerRing[i]; + if (!i || p.x < minX) + minX = p.x; + if (!i || p.y < minY) + minY = p.y; + if (!i || p.x > maxX) + maxX = p.x; + if (!i || p.y > maxY) + maxY = p.y; + } + const width = maxX - minX; + const height = maxY - minY; + const cellSize = Math.min(width, height); + let h = cellSize / 2; + // a priority queue of cells in order of their "potential" (max distance to polygon) + const cellQueue = new TinyQueue([], compareMax); + if (cellSize === 0) + return new Point$2(minX, minY); + // cover polygon with initial cells + for (let x = minX; x < maxX; x += cellSize) { + for (let y = minY; y < maxY; y += cellSize) { + cellQueue.push(new Cell(x + h, y + h, h, polygonRings)); + } + } + // take centroid as the first best guess + let bestCell = getCentroidCell(polygonRings); + let numProbes = cellQueue.length; + while (cellQueue.length) { + // pick the most promising cell from the queue + const cell = cellQueue.pop(); + // update the best cell if we found a better one + if (cell.d > bestCell.d || !bestCell.d) { + bestCell = cell; + if (debug) + console.log('found best %d after %d probes', Math.round(1e4 * cell.d) / 1e4, numProbes); + } + // do not drill down further if there's no chance of a better solution + if (cell.max - bestCell.d <= precision) + continue; + // split the cell into four cells + h = cell.h / 2; + cellQueue.push(new Cell(cell.p.x - h, cell.p.y - h, h, polygonRings)); + cellQueue.push(new Cell(cell.p.x + h, cell.p.y - h, h, polygonRings)); + cellQueue.push(new Cell(cell.p.x - h, cell.p.y + h, h, polygonRings)); + cellQueue.push(new Cell(cell.p.x + h, cell.p.y + h, h, polygonRings)); + numProbes += 4; + } + if (debug) { + console.log(`num probes: ${numProbes}`); + console.log(`best distance: ${bestCell.d}`); + } + return bestCell.p; +} +function compareMax(a, b) { + return b.max - a.max; +} +function Cell(x, y, h, polygon) { + this.p = new Point$2(x, y); + this.h = h; // half the cell size + this.d = pointToPolygonDist(this.p, polygon); // distance from cell center to polygon + this.max = this.d + this.h * Math.SQRT2; // max distance to polygon within a cell +} +// signed distance from point to polygon outline (negative if point is outside) +function pointToPolygonDist(p, polygon) { + let inside = false; + let minDistSq = Infinity; + for (let k = 0; k < polygon.length; k++) { + const ring = polygon[k]; + for (let i = 0, len = ring.length, j = len - 1; i < len; j = i++) { + const a = ring[i]; + const b = ring[j]; + if ((a.y > p.y !== b.y > p.y) && + (p.x < (b.x - a.x) * (p.y - a.y) / (b.y - a.y) + a.x)) + inside = !inside; + minDistSq = Math.min(minDistSq, distToSegmentSquared(p, a, b)); + } + } + return (inside ? 1 : -1) * Math.sqrt(minDistSq); +} +// get polygon centroid +function getCentroidCell(polygon) { + let area = 0; + let x = 0; + let y = 0; + const points = polygon[0]; + for (let i = 0, len = points.length, j = len - 1; i < len; j = i++) { + const a = points[i]; + const b = points[j]; + const f = a.x * b.y - b.x * a.y; + x += (a.x + b.x) * f; + y += (a.y + b.y) * f; + area += f * 3; + } + return new Cell(x / area, y / area, 0, polygon); +} + +exports.TextAnchorEnum = void 0; +(function (TextAnchorEnum) { + TextAnchorEnum[TextAnchorEnum["center"] = 1] = "center"; + TextAnchorEnum[TextAnchorEnum["left"] = 2] = "left"; + TextAnchorEnum[TextAnchorEnum["right"] = 3] = "right"; + TextAnchorEnum[TextAnchorEnum["top"] = 4] = "top"; + TextAnchorEnum[TextAnchorEnum["bottom"] = 5] = "bottom"; + TextAnchorEnum[TextAnchorEnum["top-left"] = 6] = "top-left"; + TextAnchorEnum[TextAnchorEnum["top-right"] = 7] = "top-right"; + TextAnchorEnum[TextAnchorEnum["bottom-left"] = 8] = "bottom-left"; + TextAnchorEnum[TextAnchorEnum["bottom-right"] = 9] = "bottom-right"; +})(exports.TextAnchorEnum || (exports.TextAnchorEnum = {})); +// The radial offset is to the edge of the text box +// In the horizontal direction, the edge of the text box is where glyphs start +// But in the vertical direction, the glyphs appear to "start" at the baseline +// We don't actually load baseline data, but we assume an offset of ONE_EM - 17 +// (see "yOffset" in shaping.js) +const baselineOffset = 7; +const INVALID_TEXT_OFFSET = Number.POSITIVE_INFINITY; +function evaluateVariableOffset(anchor, offset) { + function fromRadialOffset(anchor, radialOffset) { + let x = 0, y = 0; + if (radialOffset < 0) + radialOffset = 0; // Ignore negative offset. + // solve for r where r^2 + r^2 = radialOffset^2 + const hypotenuse = radialOffset / Math.SQRT2; + switch (anchor) { + case 'top-right': + case 'top-left': + y = hypotenuse - baselineOffset; + break; + case 'bottom-right': + case 'bottom-left': + y = -hypotenuse + baselineOffset; + break; + case 'bottom': + y = -radialOffset + baselineOffset; + break; + case 'top': + y = radialOffset - baselineOffset; + break; + } + switch (anchor) { + case 'top-right': + case 'bottom-right': + x = -hypotenuse; + break; + case 'top-left': + case 'bottom-left': + x = hypotenuse; + break; + case 'left': + x = radialOffset; + break; + case 'right': + x = -radialOffset; + break; + } + return [x, y]; + } + function fromTextOffset(anchor, offsetX, offsetY) { + let x = 0, y = 0; + // Use absolute offset values. + offsetX = Math.abs(offsetX); + offsetY = Math.abs(offsetY); + switch (anchor) { + case 'top-right': + case 'top-left': + case 'top': + y = offsetY - baselineOffset; + break; + case 'bottom-right': + case 'bottom-left': + case 'bottom': + y = -offsetY + baselineOffset; + break; + } + switch (anchor) { + case 'top-right': + case 'bottom-right': + case 'right': + x = -offsetX; + break; + case 'top-left': + case 'bottom-left': + case 'left': + x = offsetX; + break; + } + return [x, y]; + } + return (offset[1] !== INVALID_TEXT_OFFSET) ? fromTextOffset(anchor, offset[0], offset[1]) : fromRadialOffset(anchor, offset[0]); +} +// Helper to support both text-variable-anchor and text-variable-anchor-offset. Offset values converted from EMs to PXs +function getTextVariableAnchorOffset(layer, feature, canonical) { + var _a; + const layout = layer.layout; + // If style specifies text-variable-anchor-offset, just return it + const variableAnchorOffset = (_a = layout.get('text-variable-anchor-offset')) === null || _a === void 0 ? void 0 : _a.evaluate(feature, {}, canonical); + if (variableAnchorOffset) { + const sourceValues = variableAnchorOffset.values; + const destValues = []; + // Convert offsets from EM to PX, and apply baseline shift + for (let i = 0; i < sourceValues.length; i += 2) { + const anchor = destValues[i] = sourceValues[i]; + const offset = sourceValues[i + 1].map(t => t * ONE_EM); + if (anchor.startsWith('top')) { + offset[1] -= baselineOffset; + } + else if (anchor.startsWith('bottom')) { + offset[1] += baselineOffset; + } + destValues[i + 1] = offset; + } + return new VariableAnchorOffsetCollection(destValues); + } + // If style specifies text-variable-anchor, convert to the new format + const variableAnchor = layout.get('text-variable-anchor'); + if (variableAnchor) { + let textOffset; + const unevaluatedLayout = layer._unevaluatedLayout; + // The style spec says don't use `text-offset` and `text-radial-offset` together + // but doesn't actually specify what happens if you use both. We go with the radial offset. + if (unevaluatedLayout.getValue('text-radial-offset') !== undefined) { + textOffset = [layout.get('text-radial-offset').evaluate(feature, {}, canonical) * ONE_EM, INVALID_TEXT_OFFSET]; + } + else { + textOffset = layout.get('text-offset').evaluate(feature, {}, canonical).map(t => t * ONE_EM); + } + const anchorOffsets = []; + for (const anchor of variableAnchor) { + anchorOffsets.push(anchor, evaluateVariableOffset(anchor, textOffset)); + } + return new VariableAnchorOffsetCollection(anchorOffsets); + } + return null; +} + +function performSymbolLayout(args) { + args.bucket.createArrays(); + const tileSize = 512 * args.bucket.overscaling; + args.bucket.tilePixelRatio = EXTENT / tileSize; + args.bucket.compareText = {}; + args.bucket.iconsNeedLinear = false; + const layer = args.bucket.layers[0]; + const layout = layer.layout; + const unevaluatedLayoutValues = layer._unevaluatedLayout._values; + const sizes = { + // Filled in below, if *SizeData.kind is 'composite' + // compositeIconSizes: undefined, + // compositeTextSizes: undefined, + layoutIconSize: unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(args.bucket.zoom + 1), args.canonical), + layoutTextSize: unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(args.bucket.zoom + 1), args.canonical), + textMaxSize: unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(18)) + }; + if (args.bucket.textSizeData.kind === 'composite') { + const { minZoom, maxZoom } = args.bucket.textSizeData; + sizes.compositeTextSizes = [ + unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(minZoom), args.canonical), + unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), args.canonical) + ]; + } + if (args.bucket.iconSizeData.kind === 'composite') { + const { minZoom, maxZoom } = args.bucket.iconSizeData; + sizes.compositeIconSizes = [ + unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(minZoom), args.canonical), + unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), args.canonical) + ]; + } + const lineHeight = layout.get('text-line-height') * ONE_EM; + const textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point'; + const keepUpright = layout.get('text-keep-upright'); + const textSize = layout.get('text-size'); + for (const feature of args.bucket.features) { + const fontstack = layout.get('text-font').evaluate(feature, {}, args.canonical).join(','); + const layoutTextSizeThisZoom = textSize.evaluate(feature, {}, args.canonical); + const layoutTextSize = sizes.layoutTextSize.evaluate(feature, {}, args.canonical); + const layoutIconSize = sizes.layoutIconSize.evaluate(feature, {}, args.canonical); + const shapedTextOrientations = { + horizontal: {}, + vertical: undefined + }; + const text = feature.text; + let textOffset = [0, 0]; + if (text) { + const unformattedText = text.toString(); + const spacing = layout.get('text-letter-spacing').evaluate(feature, {}, args.canonical) * ONE_EM; + const spacingIfAllowed = allowsLetterSpacing(unformattedText) ? spacing : 0; + const textAnchor = layout.get('text-anchor').evaluate(feature, {}, args.canonical); + const variableAnchorOffset = getTextVariableAnchorOffset(layer, feature, args.canonical); + if (!variableAnchorOffset) { + const radialOffset = layout.get('text-radial-offset').evaluate(feature, {}, args.canonical); + // Layers with variable anchors use the `text-radial-offset` property and the [x, y] offset vector + // is calculated at placement time instead of layout time + if (radialOffset) { + // The style spec says don't use `text-offset` and `text-radial-offset` together + // but doesn't actually specify what happens if you use both. We go with the radial offset. + textOffset = evaluateVariableOffset(textAnchor, [radialOffset * ONE_EM, INVALID_TEXT_OFFSET]); + } + else { + textOffset = layout.get('text-offset').evaluate(feature, {}, args.canonical).map(t => t * ONE_EM); + } + } + let textJustify = textAlongLine ? + 'center' : + layout.get('text-justify').evaluate(feature, {}, args.canonical); + const symbolPlacement = layout.get('symbol-placement'); + const maxWidth = symbolPlacement === 'point' ? + layout.get('text-max-width').evaluate(feature, {}, args.canonical) * ONE_EM : + 0; + const addVerticalShapingForPointLabelIfNeeded = () => { + if (args.bucket.allowVerticalPlacement && allowsVerticalWritingMode(unformattedText)) { + // Vertical POI label placement is meant to be used for scripts that support vertical + // writing mode, thus, default left justification is used. If Latin + // scripts would need to be supported, this should take into account other justifications. + shapedTextOrientations.vertical = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, 'left', spacingIfAllowed, textOffset, exports.WritingMode.vertical, true, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom); + } + }; + // If this layer uses text-variable-anchor, generate shapings for all justification possibilities. + if (!textAlongLine && variableAnchorOffset) { + const justifications = new Set(); + if (textJustify === 'auto') { + for (let i = 0; i < variableAnchorOffset.values.length; i += 2) { + justifications.add(getAnchorJustification(variableAnchorOffset.values[i])); + } + } + else { + justifications.add(textJustify); + } + let singleLine = false; + for (const justification of justifications) { + if (shapedTextOrientations.horizontal[justification]) + continue; + if (singleLine) { + // If the shaping for the first justification was only a single line, we + // can re-use it for the other justifications + shapedTextOrientations.horizontal[justification] = shapedTextOrientations.horizontal[0]; + } + else { + // If using text-variable-anchor for the layer, we use a center anchor for all shapings and apply + // the offsets for the anchor in the placement step. + const shaping = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, 'center', justification, spacingIfAllowed, textOffset, exports.WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom); + if (shaping) { + shapedTextOrientations.horizontal[justification] = shaping; + singleLine = shaping.positionedLines.length === 1; + } + } + } + addVerticalShapingForPointLabelIfNeeded(); + } + else { + if (textJustify === 'auto') { + textJustify = getAnchorJustification(textAnchor); + } + // Horizontal point or line label. + const shaping = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify, spacingIfAllowed, textOffset, exports.WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom); + if (shaping) + shapedTextOrientations.horizontal[textJustify] = shaping; + // Vertical point label (if allowVerticalPlacement is enabled). + addVerticalShapingForPointLabelIfNeeded(); + // Verticalized line label. + if (allowsVerticalWritingMode(unformattedText) && textAlongLine && keepUpright) { + shapedTextOrientations.vertical = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify, spacingIfAllowed, textOffset, exports.WritingMode.vertical, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom); + } + } + } + let shapedIcon; + let isSDFIcon = false; + if (feature.icon && feature.icon.name) { + const image = args.imageMap[feature.icon.name]; + if (image) { + shapedIcon = shapeIcon(args.imagePositions[feature.icon.name], layout.get('icon-offset').evaluate(feature, {}, args.canonical), layout.get('icon-anchor').evaluate(feature, {}, args.canonical)); + // null/undefined SDF property treated same as default (false) + isSDFIcon = !!image.sdf; + if (args.bucket.sdfIcons === undefined) { + args.bucket.sdfIcons = isSDFIcon; + } + else if (args.bucket.sdfIcons !== isSDFIcon) { + warnOnce('Style sheet warning: Cannot mix SDF and non-SDF icons in one buffer'); + } + if (image.pixelRatio !== args.bucket.pixelRatio) { + args.bucket.iconsNeedLinear = true; + } + else if (layout.get('icon-rotate').constantOr(1) !== 0) { + args.bucket.iconsNeedLinear = true; + } + } + } + const shapedText = getDefaultHorizontalShaping(shapedTextOrientations.horizontal) || shapedTextOrientations.vertical; + args.bucket.iconsInText = shapedText ? shapedText.iconsInText : false; + if (shapedText || shapedIcon) { + addFeature(args.bucket, feature, shapedTextOrientations, shapedIcon, args.imageMap, sizes, layoutTextSize, layoutIconSize, textOffset, isSDFIcon, args.canonical); + } + } + if (args.showCollisionBoxes) { + args.bucket.generateCollisionDebugBuffers(); + } +} +// Choose the justification that matches the direction of the TextAnchor +function getAnchorJustification(anchor) { + switch (anchor) { + case 'right': + case 'top-right': + case 'bottom-right': + return 'right'; + case 'left': + case 'top-left': + case 'bottom-left': + return 'left'; + } + return 'center'; +} +/** + * Given a feature and its shaped text and icon data, add a 'symbol + * instance' for each _possible_ placement of the symbol feature. + * (At render timePlaceSymbols#place() selects which of these instances to + * show or hide based on collisions with symbols in other layers.) + */ +function addFeature(bucket, feature, shapedTextOrientations, shapedIcon, imageMap, sizes, layoutTextSize, layoutIconSize, textOffset, isSDFIcon, canonical) { + // To reduce the number of labels that jump around when zooming we need + // to use a text-size value that is the same for all zoom levels. + // bucket calculates text-size at a high zoom level so that all tiles can + // use the same value when calculating anchor positions. + let textMaxSize = sizes.textMaxSize.evaluate(feature, {}); + if (textMaxSize === undefined) { + textMaxSize = layoutTextSize; + } + const layout = bucket.layers[0].layout; + const iconOffset = layout.get('icon-offset').evaluate(feature, {}, canonical); + const defaultHorizontalShaping = getDefaultHorizontalShaping(shapedTextOrientations.horizontal); + const glyphSize = 24, fontScale = layoutTextSize / glyphSize, textBoxScale = bucket.tilePixelRatio * fontScale, textMaxBoxScale = bucket.tilePixelRatio * textMaxSize / glyphSize, iconBoxScale = bucket.tilePixelRatio * layoutIconSize, symbolMinDistance = bucket.tilePixelRatio * layout.get('symbol-spacing'), textPadding = layout.get('text-padding') * bucket.tilePixelRatio, iconPadding = getIconPadding(layout, feature, canonical, bucket.tilePixelRatio), textMaxAngle = layout.get('text-max-angle') / 180 * Math.PI, textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point', iconAlongLine = layout.get('icon-rotation-alignment') === 'map' && layout.get('symbol-placement') !== 'point', symbolPlacement = layout.get('symbol-placement'), textRepeatDistance = symbolMinDistance / 2; + const iconTextFit = layout.get('icon-text-fit'); + let verticallyShapedIcon; + // Adjust shaped icon size when icon-text-fit is used. + if (shapedIcon && iconTextFit !== 'none') { + if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) { + verticallyShapedIcon = fitIconToText(shapedIcon, shapedTextOrientations.vertical, iconTextFit, layout.get('icon-text-fit-padding'), iconOffset, fontScale); + } + if (defaultHorizontalShaping) { + shapedIcon = fitIconToText(shapedIcon, defaultHorizontalShaping, iconTextFit, layout.get('icon-text-fit-padding'), iconOffset, fontScale); + } + } + const addSymbolAtAnchor = (line, anchor) => { + if (anchor.x < 0 || anchor.x >= EXTENT || anchor.y < 0 || anchor.y >= EXTENT) { + // Symbol layers are drawn across tile boundaries, We filter out symbols + // outside our tile boundaries (which may be included in vector tile buffers) + // to prevent double-drawing symbols. + return; + } + addSymbol(bucket, anchor, line, shapedTextOrientations, shapedIcon, imageMap, verticallyShapedIcon, bucket.layers[0], bucket.collisionBoxArray, feature.index, feature.sourceLayerIndex, bucket.index, textBoxScale, [textPadding, textPadding, textPadding, textPadding], textAlongLine, textOffset, iconBoxScale, iconPadding, iconAlongLine, iconOffset, feature, sizes, isSDFIcon, canonical, layoutTextSize); + }; + if (symbolPlacement === 'line') { + for (const line of clipLine(feature.geometry, 0, 0, EXTENT, EXTENT)) { + const anchors = getAnchors(line, symbolMinDistance, textMaxAngle, shapedTextOrientations.vertical || defaultHorizontalShaping, shapedIcon, glyphSize, textMaxBoxScale, bucket.overscaling, EXTENT); + for (const anchor of anchors) { + const shapedText = defaultHorizontalShaping; + if (!shapedText || !anchorIsTooClose(bucket, shapedText.text, textRepeatDistance, anchor)) { + addSymbolAtAnchor(line, anchor); + } + } + } + } + else if (symbolPlacement === 'line-center') { + // No clipping, multiple lines per feature are allowed + // "lines" with only one point are ignored as in clipLines + for (const line of feature.geometry) { + if (line.length > 1) { + const anchor = getCenterAnchor(line, textMaxAngle, shapedTextOrientations.vertical || defaultHorizontalShaping, shapedIcon, glyphSize, textMaxBoxScale); + if (anchor) { + addSymbolAtAnchor(line, anchor); + } + } + } + } + else if (feature.type === 'Polygon') { + for (const polygon of classifyRings$1(feature.geometry, 0)) { + // 16 here represents 2 pixels + const poi = findPoleOfInaccessibility(polygon, 16); + addSymbolAtAnchor(polygon[0], new Anchor(poi.x, poi.y, 0)); + } + } + else if (feature.type === 'LineString') { + // https://github.com/mapbox/mapbox-gl-js/issues/3808 + for (const line of feature.geometry) { + addSymbolAtAnchor(line, new Anchor(line[0].x, line[0].y, 0)); + } + } + else if (feature.type === 'Point') { + for (const points of feature.geometry) { + for (const point of points) { + addSymbolAtAnchor([point], new Anchor(point.x, point.y, 0)); + } + } + } +} +function addTextVariableAnchorOffsets(textAnchorOffsets, variableAnchorOffset) { + const startIndex = textAnchorOffsets.length; + const values = variableAnchorOffset === null || variableAnchorOffset === void 0 ? void 0 : variableAnchorOffset.values; + if ((values === null || values === void 0 ? void 0 : values.length) > 0) { + for (let i = 0; i < values.length; i += 2) { + const anchor = exports.TextAnchorEnum[values[i]]; + const offset = values[i + 1]; + textAnchorOffsets.emplaceBack(anchor, offset[0], offset[1]); + } + } + return [startIndex, textAnchorOffsets.length]; +} +function addTextVertices(bucket, anchor, shapedText, imageMap, layer, textAlongLine, feature, textOffset, lineArray, writingMode, placementTypes, placedTextSymbolIndices, placedIconIndex, sizes, canonical) { + const glyphQuads = getGlyphQuads(anchor, shapedText, textOffset, layer, textAlongLine, feature, imageMap, bucket.allowVerticalPlacement); + const sizeData = bucket.textSizeData; + let textSizeData = null; + if (sizeData.kind === 'source') { + textSizeData = [ + SIZE_PACK_FACTOR * layer.layout.get('text-size').evaluate(feature, {}) + ]; + if (textSizeData[0] > MAX_PACKED_SIZE) { + warnOnce(`${bucket.layerIds[0]}: Value for "text-size" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your "text-size".`); + } + } + else if (sizeData.kind === 'composite') { + textSizeData = [ + SIZE_PACK_FACTOR * sizes.compositeTextSizes[0].evaluate(feature, {}, canonical), + SIZE_PACK_FACTOR * sizes.compositeTextSizes[1].evaluate(feature, {}, canonical) + ]; + if (textSizeData[0] > MAX_PACKED_SIZE || textSizeData[1] > MAX_PACKED_SIZE) { + warnOnce(`${bucket.layerIds[0]}: Value for "text-size" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your "text-size".`); + } + } + bucket.addSymbols(bucket.text, glyphQuads, textSizeData, textOffset, textAlongLine, feature, writingMode, anchor, lineArray.lineStartIndex, lineArray.lineLength, placedIconIndex, canonical); + // The placedSymbolArray is used at render time in drawTileSymbols + // These indices allow access to the array at collision detection time + for (const placementType of placementTypes) { + placedTextSymbolIndices[placementType] = bucket.text.placedSymbolArray.length - 1; + } + return glyphQuads.length * 4; +} +function getDefaultHorizontalShaping(horizontalShaping) { + // We don't care which shaping we get because this is used for collision purposes + // and all the justifications have the same collision box + for (const justification in horizontalShaping) { + return horizontalShaping[justification]; + } + return null; +} +/** + * Add a single label & icon placement. + */ +function addSymbol(bucket, anchor, line, shapedTextOrientations, shapedIcon, imageMap, verticallyShapedIcon, layer, collisionBoxArray, featureIndex, sourceLayerIndex, bucketIndex, textBoxScale, textPadding, textAlongLine, textOffset, iconBoxScale, iconPadding, iconAlongLine, iconOffset, feature, sizes, isSDFIcon, canonical, layoutTextSize) { + const lineArray = bucket.addToLineVertexArray(anchor, line); + let textCollisionFeature, iconCollisionFeature, verticalTextCollisionFeature, verticalIconCollisionFeature; + let numIconVertices = 0; + let numVerticalIconVertices = 0; + let numHorizontalGlyphVertices = 0; + let numVerticalGlyphVertices = 0; + let placedIconSymbolIndex = -1; + let verticalPlacedIconSymbolIndex = -1; + const placedTextSymbolIndices = {}; + let key = murmur3$1(''); + if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) { + const textRotation = layer.layout.get('text-rotate').evaluate(feature, {}, canonical); + const verticalTextRotation = textRotation + 90.0; + const verticalShaping = shapedTextOrientations.vertical; + verticalTextCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticalShaping, textBoxScale, textPadding, textAlongLine, verticalTextRotation); + if (verticallyShapedIcon) { + verticalIconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticallyShapedIcon, iconBoxScale, iconPadding, textAlongLine, verticalTextRotation); + } + } + //Place icon first, so text can have a reference to its index in the placed symbol array. + //Text symbols can lazily shift at render-time because of variable anchor placement. + //If the style specifies an `icon-text-fit` then the icon would have to shift along with it. + // For more info check `updateVariableAnchors` in `draw_symbol.js` . + if (shapedIcon) { + const iconRotate = layer.layout.get('icon-rotate').evaluate(feature, {}); + const hasIconTextFit = layer.layout.get('icon-text-fit') !== 'none'; + const iconQuads = getIconQuads(shapedIcon, iconRotate, isSDFIcon, hasIconTextFit); + const verticalIconQuads = verticallyShapedIcon ? getIconQuads(verticallyShapedIcon, iconRotate, isSDFIcon, hasIconTextFit) : undefined; + iconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shapedIcon, iconBoxScale, iconPadding, /*align boxes to line*/ false, iconRotate); + numIconVertices = iconQuads.length * 4; + const sizeData = bucket.iconSizeData; + let iconSizeData = null; + if (sizeData.kind === 'source') { + iconSizeData = [ + SIZE_PACK_FACTOR * layer.layout.get('icon-size').evaluate(feature, {}) + ]; + if (iconSizeData[0] > MAX_PACKED_SIZE) { + warnOnce(`${bucket.layerIds[0]}: Value for "icon-size" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your "icon-size".`); + } + } + else if (sizeData.kind === 'composite') { + iconSizeData = [ + SIZE_PACK_FACTOR * sizes.compositeIconSizes[0].evaluate(feature, {}, canonical), + SIZE_PACK_FACTOR * sizes.compositeIconSizes[1].evaluate(feature, {}, canonical) + ]; + if (iconSizeData[0] > MAX_PACKED_SIZE || iconSizeData[1] > MAX_PACKED_SIZE) { + warnOnce(`${bucket.layerIds[0]}: Value for "icon-size" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your "icon-size".`); + } + } + bucket.addSymbols(bucket.icon, iconQuads, iconSizeData, iconOffset, iconAlongLine, feature, exports.WritingMode.none, anchor, lineArray.lineStartIndex, lineArray.lineLength, + // The icon itself does not have an associated symbol since the text isnt placed yet + -1, canonical); + placedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1; + if (verticalIconQuads) { + numVerticalIconVertices = verticalIconQuads.length * 4; + bucket.addSymbols(bucket.icon, verticalIconQuads, iconSizeData, iconOffset, iconAlongLine, feature, exports.WritingMode.vertical, anchor, lineArray.lineStartIndex, lineArray.lineLength, + // The icon itself does not have an associated symbol since the text isnt placed yet + -1, canonical); + verticalPlacedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1; + } + } + const justifications = Object.keys(shapedTextOrientations.horizontal); + for (const justification of justifications) { + const shaping = shapedTextOrientations.horizontal[justification]; + if (!textCollisionFeature) { + key = murmur3$1(shaping.text); + const textRotate = layer.layout.get('text-rotate').evaluate(feature, {}, canonical); + // As a collision approximation, we can use either the vertical or any of the horizontal versions of the feature + // We're counting on all versions having similar dimensions + textCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shaping, textBoxScale, textPadding, textAlongLine, textRotate); + } + const singleLine = shaping.positionedLines.length === 1; + numHorizontalGlyphVertices += addTextVertices(bucket, anchor, shaping, imageMap, layer, textAlongLine, feature, textOffset, lineArray, shapedTextOrientations.vertical ? exports.WritingMode.horizontal : exports.WritingMode.horizontalOnly, singleLine ? justifications : [justification], placedTextSymbolIndices, placedIconSymbolIndex, sizes, canonical); + if (singleLine) { + break; + } + } + if (shapedTextOrientations.vertical) { + numVerticalGlyphVertices += addTextVertices(bucket, anchor, shapedTextOrientations.vertical, imageMap, layer, textAlongLine, feature, textOffset, lineArray, exports.WritingMode.vertical, ['vertical'], placedTextSymbolIndices, verticalPlacedIconSymbolIndex, sizes, canonical); + } + const textBoxStartIndex = textCollisionFeature ? textCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length; + const textBoxEndIndex = textCollisionFeature ? textCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length; + const verticalTextBoxStartIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length; + const verticalTextBoxEndIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length; + const iconBoxStartIndex = iconCollisionFeature ? iconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length; + const iconBoxEndIndex = iconCollisionFeature ? iconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length; + const verticalIconBoxStartIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length; + const verticalIconBoxEndIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length; + // Check if runtime collision circles should be used for any of the collision features. + // It is enough to choose the tallest feature shape as circles are always placed on a line. + // All measurements are in glyph metrics and later converted into pixels using proper font size "layoutTextSize" + let collisionCircleDiameter = -1; + const getCollisionCircleHeight = (feature, prevHeight) => { + if (feature && feature.circleDiameter) + return Math.max(feature.circleDiameter, prevHeight); + return prevHeight; + }; + collisionCircleDiameter = getCollisionCircleHeight(textCollisionFeature, collisionCircleDiameter); + collisionCircleDiameter = getCollisionCircleHeight(verticalTextCollisionFeature, collisionCircleDiameter); + collisionCircleDiameter = getCollisionCircleHeight(iconCollisionFeature, collisionCircleDiameter); + collisionCircleDiameter = getCollisionCircleHeight(verticalIconCollisionFeature, collisionCircleDiameter); + const useRuntimeCollisionCircles = (collisionCircleDiameter > -1) ? 1 : 0; + // Convert circle collision height into pixels + if (useRuntimeCollisionCircles) + collisionCircleDiameter *= layoutTextSize / ONE_EM; + if (bucket.glyphOffsetArray.length >= SymbolBucket.MAX_GLYPHS) + warnOnce('Too many glyphs being rendered in a tile. See https://github.com/mapbox/mapbox-gl-js/issues/2907'); + if (feature.sortKey !== undefined) { + bucket.addToSortKeyRanges(bucket.symbolInstances.length, feature.sortKey); + } + const variableAnchorOffset = getTextVariableAnchorOffset(layer, feature, canonical); + const [textAnchorOffsetStartIndex, textAnchorOffsetEndIndex] = addTextVariableAnchorOffsets(bucket.textAnchorOffsets, variableAnchorOffset); + bucket.symbolInstances.emplaceBack(anchor.x, anchor.y, placedTextSymbolIndices.right >= 0 ? placedTextSymbolIndices.right : -1, placedTextSymbolIndices.center >= 0 ? placedTextSymbolIndices.center : -1, placedTextSymbolIndices.left >= 0 ? placedTextSymbolIndices.left : -1, placedTextSymbolIndices.vertical || -1, placedIconSymbolIndex, verticalPlacedIconSymbolIndex, key, textBoxStartIndex, textBoxEndIndex, verticalTextBoxStartIndex, verticalTextBoxEndIndex, iconBoxStartIndex, iconBoxEndIndex, verticalIconBoxStartIndex, verticalIconBoxEndIndex, featureIndex, numHorizontalGlyphVertices, numVerticalGlyphVertices, numIconVertices, numVerticalIconVertices, useRuntimeCollisionCircles, 0, textBoxScale, collisionCircleDiameter, textAnchorOffsetStartIndex, textAnchorOffsetEndIndex); +} +function anchorIsTooClose(bucket, text, repeatDistance, anchor) { + const compareText = bucket.compareText; + if (!(text in compareText)) { + compareText[text] = []; + } + else { + const otherAnchors = compareText[text]; + for (let k = otherAnchors.length - 1; k >= 0; k--) { + if (anchor.dist(otherAnchors[k]) < repeatDistance) { + // If it's within repeatDistance of one anchor, stop looking + return true; + } + } + } + // If anchor is not within repeatDistance of any other anchor, add to array + compareText[text].push(anchor); + return false; +} + +const ARRAY_TYPES = [ + Int8Array, Uint8Array, Uint8ClampedArray, Int16Array, Uint16Array, + Int32Array, Uint32Array, Float32Array, Float64Array +]; + +/** @typedef {Int8ArrayConstructor | Uint8ArrayConstructor | Uint8ClampedArrayConstructor | Int16ArrayConstructor | Uint16ArrayConstructor | Int32ArrayConstructor | Uint32ArrayConstructor | Float32ArrayConstructor | Float64ArrayConstructor} TypedArrayConstructor */ + +const VERSION = 1; // serialized format version +const HEADER_SIZE = 8; + +class KDBush { + + /** + * Creates an index from raw `ArrayBuffer` data. + * @param {ArrayBuffer} data + */ + static from(data) { + if (!(data instanceof ArrayBuffer)) { + throw new Error('Data must be an instance of ArrayBuffer.'); + } + const [magic, versionAndType] = new Uint8Array(data, 0, 2); + if (magic !== 0xdb) { + throw new Error('Data does not appear to be in a KDBush format.'); + } + const version = versionAndType >> 4; + if (version !== VERSION) { + throw new Error(`Got v${version} data when expected v${VERSION}.`); + } + const ArrayType = ARRAY_TYPES[versionAndType & 0x0f]; + if (!ArrayType) { + throw new Error('Unrecognized array type.'); + } + const [nodeSize] = new Uint16Array(data, 2, 1); + const [numItems] = new Uint32Array(data, 4, 1); + + return new KDBush(numItems, nodeSize, ArrayType, data); + } + + /** + * Creates an index that will hold a given number of items. + * @param {number} numItems + * @param {number} [nodeSize=64] Size of the KD-tree node (64 by default). + * @param {TypedArrayConstructor} [ArrayType=Float64Array] The array type used for coordinates storage (`Float64Array` by default). + * @param {ArrayBuffer} [data] (For internal use only) + */ + constructor(numItems, nodeSize = 64, ArrayType = Float64Array, data) { + if (isNaN(numItems) || numItems < 0) throw new Error(`Unpexpected numItems value: ${numItems}.`); + + this.numItems = +numItems; + this.nodeSize = Math.min(Math.max(+nodeSize, 2), 65535); + this.ArrayType = ArrayType; + this.IndexArrayType = numItems < 65536 ? Uint16Array : Uint32Array; + + const arrayTypeIndex = ARRAY_TYPES.indexOf(this.ArrayType); + const coordsByteSize = numItems * 2 * this.ArrayType.BYTES_PER_ELEMENT; + const idsByteSize = numItems * this.IndexArrayType.BYTES_PER_ELEMENT; + const padCoords = (8 - idsByteSize % 8) % 8; + + if (arrayTypeIndex < 0) { + throw new Error(`Unexpected typed array class: ${ArrayType}.`); + } + + if (data && (data instanceof ArrayBuffer)) { // reconstruct an index from a buffer + this.data = data; + this.ids = new this.IndexArrayType(this.data, HEADER_SIZE, numItems); + this.coords = new this.ArrayType(this.data, HEADER_SIZE + idsByteSize + padCoords, numItems * 2); + this._pos = numItems * 2; + this._finished = true; + } else { // initialize a new index + this.data = new ArrayBuffer(HEADER_SIZE + coordsByteSize + idsByteSize + padCoords); + this.ids = new this.IndexArrayType(this.data, HEADER_SIZE, numItems); + this.coords = new this.ArrayType(this.data, HEADER_SIZE + idsByteSize + padCoords, numItems * 2); + this._pos = 0; + this._finished = false; + + // set header + new Uint8Array(this.data, 0, 2).set([0xdb, (VERSION << 4) + arrayTypeIndex]); + new Uint16Array(this.data, 2, 1)[0] = nodeSize; + new Uint32Array(this.data, 4, 1)[0] = numItems; + } + } + + /** + * Add a point to the index. + * @param {number} x + * @param {number} y + * @returns {number} An incremental index associated with the added item (starting from `0`). + */ + add(x, y) { + const index = this._pos >> 1; + this.ids[index] = index; + this.coords[this._pos++] = x; + this.coords[this._pos++] = y; + return index; + } + + /** + * Perform indexing of the added points. + */ + finish() { + const numAdded = this._pos >> 1; + if (numAdded !== this.numItems) { + throw new Error(`Added ${numAdded} items when expected ${this.numItems}.`); + } + // kd-sort both arrays for efficient search + sort(this.ids, this.coords, this.nodeSize, 0, this.numItems - 1, 0); + + this._finished = true; + return this; + } + + /** + * Search the index for items within a given bounding box. + * @param {number} minX + * @param {number} minY + * @param {number} maxX + * @param {number} maxY + * @returns {number[]} An array of indices correponding to the found items. + */ + range(minX, minY, maxX, maxY) { + if (!this._finished) throw new Error('Data not yet indexed - call index.finish().'); + + const {ids, coords, nodeSize} = this; + const stack = [0, ids.length - 1, 0]; + const result = []; + + // recursively search for items in range in the kd-sorted arrays + while (stack.length) { + const axis = stack.pop() || 0; + const right = stack.pop() || 0; + const left = stack.pop() || 0; + + // if we reached "tree node", search linearly + if (right - left <= nodeSize) { + for (let i = left; i <= right; i++) { + const x = coords[2 * i]; + const y = coords[2 * i + 1]; + if (x >= minX && x <= maxX && y >= minY && y <= maxY) result.push(ids[i]); + } + continue; + } + + // otherwise find the middle index + const m = (left + right) >> 1; + + // include the middle item if it's in range + const x = coords[2 * m]; + const y = coords[2 * m + 1]; + if (x >= minX && x <= maxX && y >= minY && y <= maxY) result.push(ids[m]); + + // queue search in halves that intersect the query + if (axis === 0 ? minX <= x : minY <= y) { + stack.push(left); + stack.push(m - 1); + stack.push(1 - axis); + } + if (axis === 0 ? maxX >= x : maxY >= y) { + stack.push(m + 1); + stack.push(right); + stack.push(1 - axis); + } + } + + return result; + } + + /** + * Search the index for items within a given radius. + * @param {number} qx + * @param {number} qy + * @param {number} r Query radius. + * @returns {number[]} An array of indices correponding to the found items. + */ + within(qx, qy, r) { + if (!this._finished) throw new Error('Data not yet indexed - call index.finish().'); + + const {ids, coords, nodeSize} = this; + const stack = [0, ids.length - 1, 0]; + const result = []; + const r2 = r * r; + + // recursively search for items within radius in the kd-sorted arrays + while (stack.length) { + const axis = stack.pop() || 0; + const right = stack.pop() || 0; + const left = stack.pop() || 0; + + // if we reached "tree node", search linearly + if (right - left <= nodeSize) { + for (let i = left; i <= right; i++) { + if (sqDist(coords[2 * i], coords[2 * i + 1], qx, qy) <= r2) result.push(ids[i]); + } + continue; + } + + // otherwise find the middle index + const m = (left + right) >> 1; + + // include the middle item if it's in range + const x = coords[2 * m]; + const y = coords[2 * m + 1]; + if (sqDist(x, y, qx, qy) <= r2) result.push(ids[m]); + + // queue search in halves that intersect the query + if (axis === 0 ? qx - r <= x : qy - r <= y) { + stack.push(left); + stack.push(m - 1); + stack.push(1 - axis); + } + if (axis === 0 ? qx + r >= x : qy + r >= y) { + stack.push(m + 1); + stack.push(right); + stack.push(1 - axis); + } + } + + return result; + } +} + +/** + * @param {Uint16Array | Uint32Array} ids + * @param {InstanceType} coords + * @param {number} nodeSize + * @param {number} left + * @param {number} right + * @param {number} axis + */ +function sort(ids, coords, nodeSize, left, right, axis) { + if (right - left <= nodeSize) return; + + const m = (left + right) >> 1; // middle index + + // sort ids and coords around the middle index so that the halves lie + // either left/right or top/bottom correspondingly (taking turns) + select(ids, coords, m, left, right, axis); + + // recursively kd-sort first half and second half on the opposite axis + sort(ids, coords, nodeSize, left, m - 1, 1 - axis); + sort(ids, coords, nodeSize, m + 1, right, 1 - axis); +} + +/** + * Custom Floyd-Rivest selection algorithm: sort ids and coords so that + * [left..k-1] items are smaller than k-th item (on either x or y axis) + * @param {Uint16Array | Uint32Array} ids + * @param {InstanceType} coords + * @param {number} k + * @param {number} left + * @param {number} right + * @param {number} axis + */ +function select(ids, coords, k, left, right, axis) { + + while (right > left) { + if (right - left > 600) { + const n = right - left + 1; + const m = k - left + 1; + const z = Math.log(n); + const s = 0.5 * Math.exp(2 * z / 3); + const sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1); + const newLeft = Math.max(left, Math.floor(k - m * s / n + sd)); + const newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd)); + select(ids, coords, k, newLeft, newRight, axis); + } + + const t = coords[2 * k + axis]; + let i = left; + let j = right; + + swapItem(ids, coords, left, k); + if (coords[2 * right + axis] > t) swapItem(ids, coords, left, right); + + while (i < j) { + swapItem(ids, coords, i, j); + i++; + j--; + while (coords[2 * i + axis] < t) i++; + while (coords[2 * j + axis] > t) j--; + } + + if (coords[2 * left + axis] === t) swapItem(ids, coords, left, j); + else { + j++; + swapItem(ids, coords, j, right); + } + + if (j <= k) left = j + 1; + if (k <= j) right = j - 1; + } +} + +/** + * @param {Uint16Array | Uint32Array} ids + * @param {InstanceType} coords + * @param {number} i + * @param {number} j + */ +function swapItem(ids, coords, i, j) { + swap(ids, i, j); + swap(coords, 2 * i, 2 * j); + swap(coords, 2 * i + 1, 2 * j + 1); +} + +/** + * @param {InstanceType} arr + * @param {number} i + * @param {number} j + */ +function swap(arr, i, j) { + const tmp = arr[i]; + arr[i] = arr[j]; + arr[j] = tmp; +} + +/** + * @param {number} ax + * @param {number} ay + * @param {number} bx + * @param {number} by + */ +function sqDist(ax, ay, bx, by) { + const dx = ax - bx; + const dy = ay - by; + return dx * dx + dy * dy; +} + +exports.PerformanceMarkers = void 0; +(function (PerformanceMarkers) { + PerformanceMarkers["create"] = "create"; + PerformanceMarkers["load"] = "load"; + PerformanceMarkers["fullLoad"] = "fullLoad"; +})(exports.PerformanceMarkers || (exports.PerformanceMarkers = {})); +let lastFrameTime = null; +let frameTimes = []; +const minFramerateTarget = 60; +const frameTimeTarget = 1000 / minFramerateTarget; +const loadTimeKey = 'loadTime'; +const fullLoadTimeKey = 'fullLoadTime'; +const PerformanceUtils = { + mark(marker) { + performance.mark(marker); + }, + frame(timestamp) { + const currTimestamp = timestamp; + if (lastFrameTime != null) { + const frameTime = currTimestamp - lastFrameTime; + frameTimes.push(frameTime); + } + lastFrameTime = currTimestamp; + }, + clearMetrics() { + lastFrameTime = null; + frameTimes = []; + performance.clearMeasures(loadTimeKey); + performance.clearMeasures(fullLoadTimeKey); + for (const marker in exports.PerformanceMarkers) { + performance.clearMarks(exports.PerformanceMarkers[marker]); + } + }, + getPerformanceMetrics() { + performance.measure(loadTimeKey, exports.PerformanceMarkers.create, exports.PerformanceMarkers.load); + performance.measure(fullLoadTimeKey, exports.PerformanceMarkers.create, exports.PerformanceMarkers.fullLoad); + const loadTime = performance.getEntriesByName(loadTimeKey)[0].duration; + const fullLoadTime = performance.getEntriesByName(fullLoadTimeKey)[0].duration; + const totalFrames = frameTimes.length; + const avgFrameTime = frameTimes.reduce((prev, curr) => prev + curr, 0) / totalFrames / 1000; + const fps = 1 / avgFrameTime; + // count frames that missed our framerate target + const droppedFrames = frameTimes + .filter((frameTime) => frameTime > frameTimeTarget) + .reduce((acc, curr) => { + return acc + (curr - frameTimeTarget) / frameTimeTarget; + }, 0); + const percentDroppedFrames = (droppedFrames / (totalFrames + droppedFrames)) * 100; + return { + loadTime, + fullLoadTime, + fps, + percentDroppedFrames, + totalFrames + }; + } +}; +/** + * @internal + * Safe wrapper for the performance resource timing API in web workers with graceful degradation + */ +class RequestPerformance { + constructor(request) { + this._marks = { + start: [request.url, 'start'].join('#'), + end: [request.url, 'end'].join('#'), + measure: request.url.toString() + }; + performance.mark(this._marks.start); + } + finish() { + performance.mark(this._marks.end); + let resourceTimingData = performance.getEntriesByName(this._marks.measure); + // fallback if web worker implementation of perf.getEntriesByName returns empty + if (resourceTimingData.length === 0) { + performance.measure(this._marks.measure, this._marks.start, this._marks.end); + resourceTimingData = performance.getEntriesByName(this._marks.measure); + // cleanup + performance.clearMarks(this._marks.start); + performance.clearMarks(this._marks.end); + performance.clearMeasures(this._marks.measure); + } + return resourceTimingData; + } +} +var performance$1 = performance; + +exports.AJAXError = AJAXError; +exports.Actor = Actor; +exports.AlphaImage = AlphaImage; +exports.CanonicalTileID = CanonicalTileID; +exports.CollisionBoxArray = CollisionBoxArray; +exports.CollisionCircleLayoutArray = CollisionCircleLayoutArray; +exports.Color = Color; +exports.DEMData = DEMData; +exports.DataConstantProperty = DataConstantProperty; +exports.DictionaryCoder = DictionaryCoder; +exports.EXTENT = EXTENT; +exports.ErrorEvent = ErrorEvent; +exports.EvaluationParameters = EvaluationParameters; +exports.Event = Event; +exports.Evented = Evented; +exports.FeatureIndex = FeatureIndex; +exports.FillBucket = FillBucket; +exports.FillExtrusionBucket = FillExtrusionBucket; +exports.GeoJSONFeature = GeoJSONFeature; +exports.ImageAtlas = ImageAtlas; +exports.ImagePosition = ImagePosition; +exports.KDBush = KDBush; +exports.LineBucket = LineBucket; +exports.LineStripIndexArray = LineStripIndexArray; +exports.LngLat = LngLat; +exports.MercatorCoordinate = MercatorCoordinate; +exports.ONE_EM = ONE_EM; +exports.OverscaledTileID = OverscaledTileID; +exports.PerformanceUtils = PerformanceUtils; +exports.Point = Point$2; +exports.Pos3dArray = Pos3dArray; +exports.PosArray = PosArray; +exports.Properties = Properties; +exports.Protobuf = Protobuf; +exports.QuadTriangleArray = QuadTriangleArray; +exports.RGBAImage = RGBAImage; +exports.RasterBoundsArray = RasterBoundsArray; +exports.RequestPerformance = RequestPerformance; +exports.SegmentVector = SegmentVector; +exports.SymbolBucket = SymbolBucket; +exports.Transitionable = Transitionable; +exports.TriangleIndexArray = TriangleIndexArray; +exports.Uniform1f = Uniform1f; +exports.Uniform1i = Uniform1i; +exports.Uniform2f = Uniform2f; +exports.Uniform3f = Uniform3f; +exports.Uniform4f = Uniform4f; +exports.UniformColor = UniformColor; +exports.UniformMatrix4f = UniformMatrix4f; +exports.UnwrappedTileID = UnwrappedTileID; +exports.ValidationError = ValidationError; +exports.ZoomHistory = ZoomHistory; +exports.__awaiter = __awaiter; +exports.add = add$4; +exports.addDynamicAttributes = addDynamicAttributes; +exports.arrayBufferToImage = arrayBufferToImage; +exports.arrayBufferToImageBitmap = arrayBufferToImageBitmap; +exports.asyncAll = asyncAll; +exports.bezier = bezier$1; +exports.browser = browser; +exports.clamp = clamp$1; +exports.clipLine = clipLine; +exports.clone = clone$5; +exports.clone$1 = clone$9; +exports.clone$2 = clone$4; +exports.collisionCircleLayout = collisionCircleLayout; +exports.config = config; +exports.copy = copy$5; +exports.create = create$5; +exports.create$1 = create$6; +exports.create$2 = create$8; +exports.createExpression = createExpression; +exports.createFilter = createFilter; +exports.createLayout = createLayout; +exports.createStyleLayer = createStyleLayer; +exports.cross = cross$2; +exports.deepEqual = deepEqual$1; +exports.defaultEasing = defaultEasing; +exports.derefLayers = derefLayers; +exports.diffStyles = diffStyles; +exports.dot = dot$5; +exports.dot$1 = dot$4; +exports.earthRadius = earthRadius; +exports.emitValidationErrors = emitValidationErrors; +exports.emptyStyle = emptyStyle; +exports.equals = equals$6; +exports.evaluateSizeForFeature = evaluateSizeForFeature; +exports.evaluateSizeForZoom = evaluateSizeForZoom; +exports.evented = evented; +exports.extend = extend; +exports.filterObject = filterObject; +exports.findLineIntersection = findLineIntersection; +exports.fromRotation = fromRotation$2; +exports.fromScaling = fromScaling; +exports.getAnchorAlignment = getAnchorAlignment; +exports.getAnchorJustification = getAnchorJustification; +exports.getArrayBuffer = getArrayBuffer; +exports.getDefaultExportFromCjs = getDefaultExportFromCjs; +exports.getImageData = getImageData; +exports.getJSON = getJSON; +exports.getOverlapMode = getOverlapMode; +exports.getProtocolAction = getProtocolAction; +exports.getRTLTextPluginStatus = getRTLTextPluginStatus; +exports.getReferrer = getReferrer; +exports.getVideo = getVideo; +exports.groupByLayout = groupByLayout; +exports.identity = identity$2; +exports.interpolate = interpolate; +exports.invert = invert$2; +exports.isImageBitmap = isImageBitmap; +exports.isOffscreenCanvasDistorted = isOffscreenCanvasDistorted; +exports.isSafari = isSafari; +exports.isWorker = isWorker; +exports.keysDifference = keysDifference; +exports.lazyLoadRTLTextPlugin = lazyLoadRTLTextPlugin; +exports.lngFromMercatorX = lngFromMercatorX; +exports.makeRequest = makeRequest; +exports.mapObject = mapObject; +exports.mercatorXfromLng = mercatorXfromLng; +exports.mercatorYfromLat = mercatorYfromLat; +exports.mercatorZfromAltitude = mercatorZfromAltitude; +exports.mul = mul$5; +exports.mul$1 = mul$3; +exports.multiply = multiply$5; +exports.nextPowerOfTwo = nextPowerOfTwo; +exports.normalize = normalize$4; +exports.offscreenCanvasSupported = offscreenCanvasSupported; +exports.operations = operations; +exports.ortho = ortho; +exports.parseCacheControl = parseCacheControl; +exports.parseGlyphPbf = parseGlyphPbf; +exports.pbf = pbf; +exports.performSymbolLayout = performSymbolLayout; +exports.perspective = perspective; +exports.pick = pick; +exports.plugin = plugin; +exports.pointGeometry = pointGeometry; +exports.polygonIntersectsPolygon = polygonIntersectsPolygon; +exports.potpack = potpack; +exports.readImageUsingVideoFrame = readImageUsingVideoFrame; +exports.register = register; +exports.registerForPluginStateChange = registerForPluginStateChange; +exports.renderColorRamp = renderColorRamp; +exports.rotate = rotate$4; +exports.rotateX = rotateX$3; +exports.rotateZ = rotateZ$3; +exports.sameOrigin = sameOrigin; +exports.scale = scale$5; +exports.scale$1 = scale$4; +exports.setRTLTextPlugin = setRTLTextPlugin; +exports.sphericalToCartesian = sphericalToCartesian; +exports.sqrLen = sqrLen; +exports.sub = sub$2; +exports.toEvaluationFeature = toEvaluationFeature; +exports.transformMat3 = transformMat3$1; +exports.transformMat4 = transformMat4$1; +exports.transformMat4$1 = transformMat4; +exports.translate = translate$1; +exports.triggerPluginCompletionEvent = triggerPluginCompletionEvent; +exports.unicodeBlockLookup = unicodeBlockLookup; +exports.uniqueId = uniqueId; +exports.v8Spec = v8Spec; +exports.validateCustomStyleLayer = validateCustomStyleLayer; +exports.validateLight = validateLight; +exports.validateStyle = validateStyle; +exports.vectorTile = vectorTile; +exports.warnOnce = warnOnce; +exports.wrap = wrap; + +})); + +define(['./shared'], (function (performance) { 'use strict'; + +class StyleLayerIndex { + constructor(layerConfigs) { + this.keyCache = {}; + if (layerConfigs) { + this.replace(layerConfigs); + } + } + replace(layerConfigs) { + this._layerConfigs = {}; + this._layers = {}; + this.update(layerConfigs, []); + } + update(layerConfigs, removedIds) { + for (const layerConfig of layerConfigs) { + this._layerConfigs[layerConfig.id] = layerConfig; + const layer = this._layers[layerConfig.id] = performance.createStyleLayer(layerConfig); + layer._featureFilter = performance.createFilter(layer.filter); + if (this.keyCache[layerConfig.id]) + delete this.keyCache[layerConfig.id]; + } + for (const id of removedIds) { + delete this.keyCache[id]; + delete this._layerConfigs[id]; + delete this._layers[id]; + } + this.familiesBySource = {}; + const groups = performance.groupByLayout(Object.values(this._layerConfigs), this.keyCache); + for (const layerConfigs of groups) { + const layers = layerConfigs.map((layerConfig) => this._layers[layerConfig.id]); + const layer = layers[0]; + if (layer.visibility === 'none') { + continue; + } + const sourceId = layer.source || ''; + let sourceGroup = this.familiesBySource[sourceId]; + if (!sourceGroup) { + sourceGroup = this.familiesBySource[sourceId] = {}; + } + const sourceLayerId = layer.sourceLayer || '_geojsonTileLayer'; + let sourceLayerFamilies = sourceGroup[sourceLayerId]; + if (!sourceLayerFamilies) { + sourceLayerFamilies = sourceGroup[sourceLayerId] = []; + } + sourceLayerFamilies.push(layers); + } + } +} + +const padding = 1; +class GlyphAtlas { + constructor(stacks) { + const positions = {}; + const bins = []; + for (const stack in stacks) { + const glyphs = stacks[stack]; + const stackPositions = positions[stack] = {}; + for (const id in glyphs) { + const src = glyphs[+id]; + if (!src || src.bitmap.width === 0 || src.bitmap.height === 0) + continue; + const bin = { + x: 0, + y: 0, + w: src.bitmap.width + 2 * padding, + h: src.bitmap.height + 2 * padding + }; + bins.push(bin); + stackPositions[id] = { rect: bin, metrics: src.metrics }; + } + } + const { w, h } = performance.potpack(bins); + const image = new performance.AlphaImage({ width: w || 1, height: h || 1 }); + for (const stack in stacks) { + const glyphs = stacks[stack]; + for (const id in glyphs) { + const src = glyphs[+id]; + if (!src || src.bitmap.width === 0 || src.bitmap.height === 0) + continue; + const bin = positions[stack][id].rect; + performance.AlphaImage.copy(src.bitmap, image, { x: 0, y: 0 }, { x: bin.x + padding, y: bin.y + padding }, src.bitmap); + } + } + this.image = image; + this.positions = positions; + } +} +performance.register('GlyphAtlas', GlyphAtlas); + +class WorkerTile { + constructor(params) { + this.tileID = new performance.OverscaledTileID(params.tileID.overscaledZ, params.tileID.wrap, params.tileID.canonical.z, params.tileID.canonical.x, params.tileID.canonical.y); + this.uid = params.uid; + this.zoom = params.zoom; + this.pixelRatio = params.pixelRatio; + this.tileSize = params.tileSize; + this.source = params.source; + this.overscaling = this.tileID.overscaleFactor(); + this.showCollisionBoxes = params.showCollisionBoxes; + this.collectResourceTiming = !!params.collectResourceTiming; + this.returnDependencies = !!params.returnDependencies; + this.promoteId = params.promoteId; + this.inFlightDependencies = []; + this.dependencySentinel = -1; + } + parse(data, layerIndex, availableImages, actor, callback) { + this.status = 'parsing'; + this.data = data; + this.collisionBoxArray = new performance.CollisionBoxArray(); + const sourceLayerCoder = new performance.DictionaryCoder(Object.keys(data.layers).sort()); + const featureIndex = new performance.FeatureIndex(this.tileID, this.promoteId); + featureIndex.bucketLayerIDs = []; + const buckets = {}; + const options = { + featureIndex, + iconDependencies: {}, + patternDependencies: {}, + glyphDependencies: {}, + availableImages + }; + const layerFamilies = layerIndex.familiesBySource[this.source]; + for (const sourceLayerId in layerFamilies) { + const sourceLayer = data.layers[sourceLayerId]; + if (!sourceLayer) { + continue; + } + if (sourceLayer.version === 1) { + performance.warnOnce(`Vector tile source "${this.source}" layer "${sourceLayerId}" ` + + 'does not use vector tile spec v2 and therefore may have some rendering errors.'); + } + const sourceLayerIndex = sourceLayerCoder.encode(sourceLayerId); + const features = []; + for (let index = 0; index < sourceLayer.length; index++) { + const feature = sourceLayer.feature(index); + const id = featureIndex.getId(feature, sourceLayerId); + features.push({ feature, id, index, sourceLayerIndex }); + } + for (const family of layerFamilies[sourceLayerId]) { + const layer = family[0]; + if (layer.source !== this.source) { + performance.warnOnce(`layer.source = ${layer.source} does not equal this.source = ${this.source}`); + } + if (layer.minzoom && this.zoom < Math.floor(layer.minzoom)) + continue; + if (layer.maxzoom && this.zoom >= layer.maxzoom) + continue; + if (layer.visibility === 'none') + continue; + recalculateLayers(family, this.zoom, availableImages); + const bucket = buckets[layer.id] = layer.createBucket({ + index: featureIndex.bucketLayerIDs.length, + layers: family, + zoom: this.zoom, + pixelRatio: this.pixelRatio, + overscaling: this.overscaling, + collisionBoxArray: this.collisionBoxArray, + sourceLayerIndex, + sourceID: this.source + }); + bucket.populate(features, options, this.tileID.canonical); + featureIndex.bucketLayerIDs.push(family.map((l) => l.id)); + } + } + let error; + let glyphMap; + let iconMap; + let patternMap; + const stacks = performance.mapObject(options.glyphDependencies, (glyphs) => Object.keys(glyphs).map(Number)); + this.inFlightDependencies.forEach((request) => request === null || request === void 0 ? void 0 : request.cancel()); + this.inFlightDependencies = []; + // cancelling seems to be not sufficient, we seems to still manage to get a callback hit, so use a sentinel to drop stale results + const dependencySentinel = ++this.dependencySentinel; + if (Object.keys(stacks).length) { + this.inFlightDependencies.push(actor.send('getGlyphs', { uid: this.uid, stacks, source: this.source, tileID: this.tileID, type: 'glyphs' }, (err, result) => { + if (dependencySentinel !== this.dependencySentinel) { + return; + } + if (!error) { + error = err; + glyphMap = result; + maybePrepare.call(this); + } + })); + } + else { + glyphMap = {}; + } + const icons = Object.keys(options.iconDependencies); + if (icons.length) { + this.inFlightDependencies.push(actor.send('getImages', { icons, source: this.source, tileID: this.tileID, type: 'icons' }, (err, result) => { + if (dependencySentinel !== this.dependencySentinel) { + return; + } + if (!error) { + error = err; + iconMap = result; + maybePrepare.call(this); + } + })); + } + else { + iconMap = {}; + } + const patterns = Object.keys(options.patternDependencies); + if (patterns.length) { + this.inFlightDependencies.push(actor.send('getImages', { icons: patterns, source: this.source, tileID: this.tileID, type: 'patterns' }, (err, result) => { + if (dependencySentinel !== this.dependencySentinel) { + return; + } + if (!error) { + error = err; + patternMap = result; + maybePrepare.call(this); + } + })); + } + else { + patternMap = {}; + } + maybePrepare.call(this); + function maybePrepare() { + if (error) { + return callback(error); + } + else if (glyphMap && iconMap && patternMap) { + const glyphAtlas = new GlyphAtlas(glyphMap); + const imageAtlas = new performance.ImageAtlas(iconMap, patternMap); + for (const key in buckets) { + const bucket = buckets[key]; + if (bucket instanceof performance.SymbolBucket) { + recalculateLayers(bucket.layers, this.zoom, availableImages); + performance.performSymbolLayout({ + bucket, + glyphMap, + glyphPositions: glyphAtlas.positions, + imageMap: iconMap, + imagePositions: imageAtlas.iconPositions, + showCollisionBoxes: this.showCollisionBoxes, + canonical: this.tileID.canonical + }); + } + else if (bucket.hasPattern && + (bucket instanceof performance.LineBucket || + bucket instanceof performance.FillBucket || + bucket instanceof performance.FillExtrusionBucket)) { + recalculateLayers(bucket.layers, this.zoom, availableImages); + bucket.addFeatures(options, this.tileID.canonical, imageAtlas.patternPositions); + } + } + this.status = 'done'; + callback(null, { + buckets: Object.values(buckets).filter(b => !b.isEmpty()), + featureIndex, + collisionBoxArray: this.collisionBoxArray, + glyphAtlasImage: glyphAtlas.image, + imageAtlas, + // Only used for benchmarking: + glyphMap: this.returnDependencies ? glyphMap : null, + iconMap: this.returnDependencies ? iconMap : null, + glyphPositions: this.returnDependencies ? glyphAtlas.positions : null + }); + } + } + } +} +function recalculateLayers(layers, zoom, availableImages) { + // Layers are shared and may have been used by a WorkerTile with a different zoom. + const parameters = new performance.EvaluationParameters(zoom); + for (const layer of layers) { + layer.recalculate(parameters, availableImages); + } +} + +/** + * Loads a vector tile + */ +function loadVectorTile(params, callback) { + const request = performance.getArrayBuffer(params.request, (err, data, cacheControl, expires) => { + if (err) { + callback(err); + } + else if (data) { + try { + const vectorTile = new performance.vectorTile.VectorTile(new performance.Protobuf(data)); + callback(null, { + vectorTile, + rawData: data, + cacheControl, + expires + }); + } + catch (ex) { + const bytes = new Uint8Array(data); + const isGzipped = bytes[0] === 0x1f && bytes[1] === 0x8b; + let errorMessage = `Unable to parse the tile at ${params.request.url}, `; + if (isGzipped) { + errorMessage += 'please make sure the data is not gzipped and that you have configured the relevant header in the server'; + } + else { + errorMessage += `got error: ${ex.messge}`; + } + callback(new Error(errorMessage)); + } + } + }); + return () => { + request.cancel(); + callback(); + }; +} +/** + * The {@link WorkerSource} implementation that supports {@link VectorTileSource}. + * This class is designed to be easily reused to support custom source types + * for data formats that can be parsed/converted into an in-memory VectorTile + * representation. To do so, create it with + * `new VectorTileWorkerSource(actor, styleLayers, customLoadVectorDataFunction)`. + */ +class VectorTileWorkerSource { + /** + * @param loadVectorData - Optional method for custom loading of a VectorTile + * object based on parameters passed from the main-thread Source. See + * {@link VectorTileWorkerSource#loadTile}. The default implementation simply + * loads the pbf at `params.url`. + */ + constructor(actor, layerIndex, availableImages, loadVectorData) { + this.actor = actor; + this.layerIndex = layerIndex; + this.availableImages = availableImages; + this.loadVectorData = loadVectorData || loadVectorTile; + this.fetching = {}; + this.loading = {}; + this.loaded = {}; + } + /** + * Implements {@link WorkerSource#loadTile}. Delegates to + * {@link VectorTileWorkerSource#loadVectorData} (which by default expects + * a `params.url` property) for fetching and producing a VectorTile object. + */ + loadTile(params, callback) { + const uid = params.uid; + if (!this.loading) + this.loading = {}; + const perf = (params && params.request && params.request.collectResourceTiming) ? + new performance.RequestPerformance(params.request) : false; + const workerTile = this.loading[uid] = new WorkerTile(params); + workerTile.abort = this.loadVectorData(params, (err, response) => { + delete this.loading[uid]; + if (err || !response) { + workerTile.status = 'done'; + this.loaded[uid] = workerTile; + return callback(err); + } + const rawTileData = response.rawData; + const cacheControl = {}; + if (response.expires) + cacheControl.expires = response.expires; + if (response.cacheControl) + cacheControl.cacheControl = response.cacheControl; + const resourceTiming = {}; + if (perf) { + const resourceTimingData = perf.finish(); + // it's necessary to eval the result of getEntriesByName() here via parse/stringify + // late evaluation in the main thread causes TypeError: illegal invocation + if (resourceTimingData) + resourceTiming.resourceTiming = JSON.parse(JSON.stringify(resourceTimingData)); + } + workerTile.vectorTile = response.vectorTile; + workerTile.parse(response.vectorTile, this.layerIndex, this.availableImages, this.actor, (err, result) => { + delete this.fetching[uid]; + if (err || !result) + return callback(err); + // Transferring a copy of rawTileData because the worker needs to retain its copy. + callback(null, performance.extend({ rawTileData: rawTileData.slice(0) }, result, cacheControl, resourceTiming)); + }); + this.loaded = this.loaded || {}; + this.loaded[uid] = workerTile; + // keep the original fetching state so that reload tile can pick it up if the original parse is cancelled by reloads' parse + this.fetching[uid] = { rawTileData, cacheControl, resourceTiming }; + }); + } + /** + * Implements {@link WorkerSource#reloadTile}. + */ + reloadTile(params, callback) { + const loaded = this.loaded; + const uid = params.uid; + if (loaded && loaded[uid]) { + const workerTile = loaded[uid]; + workerTile.showCollisionBoxes = params.showCollisionBoxes; + if (workerTile.status === 'parsing') { + workerTile.parse(workerTile.vectorTile, this.layerIndex, this.availableImages, this.actor, (err, result) => { + if (err || !result) + return callback(err, result); + // if we have cancelled the original parse, make sure to pass the rawTileData from the original fetch + let parseResult; + if (this.fetching[uid]) { + const { rawTileData, cacheControl, resourceTiming } = this.fetching[uid]; + delete this.fetching[uid]; + parseResult = performance.extend({ rawTileData: rawTileData.slice(0) }, result, cacheControl, resourceTiming); + } + else { + parseResult = result; + } + callback(null, parseResult); + }); + } + else if (workerTile.status === 'done') { + // if there was no vector tile data on the initial load, don't try and re-parse tile + if (workerTile.vectorTile) { + workerTile.parse(workerTile.vectorTile, this.layerIndex, this.availableImages, this.actor, callback); + } + else { + callback(); + } + } + } + } + /** + * Implements {@link WorkerSource#abortTile}. + * + * @param params - The tile parameters + * @param callback - The callback + */ + abortTile(params, callback) { + const loading = this.loading, uid = params.uid; + if (loading && loading[uid] && loading[uid].abort) { + loading[uid].abort(); + delete loading[uid]; + } + callback(); + } + /** + * Implements {@link WorkerSource#removeTile}. + * + * @param params - The tile parameters + * @param callback - The callback + */ + removeTile(params, callback) { + const loaded = this.loaded, uid = params.uid; + if (loaded && loaded[uid]) { + delete loaded[uid]; + } + callback(); + } +} + +class RasterDEMTileWorkerSource { + constructor() { + this.loaded = {}; + } + loadTile(params, callback) { + return performance.__awaiter(this, void 0, void 0, function* () { + const { uid, encoding, rawImageData, redFactor, greenFactor, blueFactor, baseShift } = params; + const width = rawImageData.width + 2; + const height = rawImageData.height + 2; + const imagePixels = performance.isImageBitmap(rawImageData) ? + new performance.RGBAImage({ width, height }, yield performance.getImageData(rawImageData, -1, -1, width, height)) : + rawImageData; + const dem = new performance.DEMData(uid, imagePixels, encoding, redFactor, greenFactor, blueFactor, baseShift); + this.loaded = this.loaded || {}; + this.loaded[uid] = dem; + callback(null, dem); + }); + } + removeTile(params) { + const loaded = this.loaded, uid = params.uid; + if (loaded && loaded[uid]) { + delete loaded[uid]; + } + } +} + +var geojsonRewind = rewind$1; + +function rewind$1(gj, outer) { + var type = gj && gj.type, i; + + if (type === 'FeatureCollection') { + for (i = 0; i < gj.features.length; i++) rewind$1(gj.features[i], outer); + + } else if (type === 'GeometryCollection') { + for (i = 0; i < gj.geometries.length; i++) rewind$1(gj.geometries[i], outer); + + } else if (type === 'Feature') { + rewind$1(gj.geometry, outer); + + } else if (type === 'Polygon') { + rewindRings(gj.coordinates, outer); + + } else if (type === 'MultiPolygon') { + for (i = 0; i < gj.coordinates.length; i++) rewindRings(gj.coordinates[i], outer); + } + + return gj; +} + +function rewindRings(rings, outer) { + if (rings.length === 0) return; + + rewindRing(rings[0], outer); + for (var i = 1; i < rings.length; i++) { + rewindRing(rings[i], !outer); + } +} + +function rewindRing(ring, dir) { + var area = 0, err = 0; + for (var i = 0, len = ring.length, j = len - 1; i < len; j = i++) { + var k = (ring[i][0] - ring[j][0]) * (ring[j][1] + ring[i][1]); + var m = area + k; + err += Math.abs(area) >= Math.abs(k) ? area - m + k : k - m + area; + area = m; + } + if (area + err >= 0 !== !!dir) ring.reverse(); +} + +var rewind$2 = /*@__PURE__*/performance.getDefaultExportFromCjs(geojsonRewind); + +const toGeoJSON = performance.vectorTile.VectorTileFeature.prototype.toGeoJSON; +let FeatureWrapper$1 = class FeatureWrapper { + constructor(feature) { + this._feature = feature; + this.extent = performance.EXTENT; + this.type = feature.type; + this.properties = feature.tags; + // If the feature has a top-level `id` property, copy it over, but only + // if it can be coerced to an integer, because this wrapper is used for + // serializing geojson feature data into vector tile PBF data, and the + // vector tile spec only supports integer values for feature ids -- + // allowing non-integer values here results in a non-compliant PBF + // that causes an exception when it is parsed with vector-tile-js + if ('id' in feature && !isNaN(feature.id)) { + this.id = parseInt(feature.id, 10); + } + } + loadGeometry() { + if (this._feature.type === 1) { + const geometry = []; + for (const point of this._feature.geometry) { + geometry.push([new performance.Point(point[0], point[1])]); + } + return geometry; + } + else { + const geometry = []; + for (const ring of this._feature.geometry) { + const newRing = []; + for (const point of ring) { + newRing.push(new performance.Point(point[0], point[1])); + } + geometry.push(newRing); + } + return geometry; + } + } + toGeoJSON(x, y, z) { + return toGeoJSON.call(this, x, y, z); + } +}; +let GeoJSONWrapper$2 = class GeoJSONWrapper { + constructor(features) { + this.layers = { '_geojsonTileLayer': this }; + this.name = '_geojsonTileLayer'; + this.extent = performance.EXTENT; + this.length = features.length; + this._features = features; + } + feature(i) { + return new FeatureWrapper$1(this._features[i]); + } +}; + +var vtPbf$1 = {exports: {}}; + +'use strict'; + +var Point = performance.pointGeometry; +var VectorTileFeature = performance.vectorTile.VectorTileFeature; + +var geojson_wrapper = GeoJSONWrapper$1; + +// conform to vectortile api +function GeoJSONWrapper$1 (features, options) { + this.options = options || {}; + this.features = features; + this.length = features.length; +} + +GeoJSONWrapper$1.prototype.feature = function (i) { + return new FeatureWrapper(this.features[i], this.options.extent) +}; + +function FeatureWrapper (feature, extent) { + this.id = typeof feature.id === 'number' ? feature.id : undefined; + this.type = feature.type; + this.rawGeometry = feature.type === 1 ? [feature.geometry] : feature.geometry; + this.properties = feature.tags; + this.extent = extent || 4096; +} + +FeatureWrapper.prototype.loadGeometry = function () { + var rings = this.rawGeometry; + this.geometry = []; + + for (var i = 0; i < rings.length; i++) { + var ring = rings[i]; + var newRing = []; + for (var j = 0; j < ring.length; j++) { + newRing.push(new Point(ring[j][0], ring[j][1])); + } + this.geometry.push(newRing); + } + return this.geometry +}; + +FeatureWrapper.prototype.bbox = function () { + if (!this.geometry) this.loadGeometry(); + + var rings = this.geometry; + var x1 = Infinity; + var x2 = -Infinity; + var y1 = Infinity; + var y2 = -Infinity; + + for (var i = 0; i < rings.length; i++) { + var ring = rings[i]; + + for (var j = 0; j < ring.length; j++) { + var coord = ring[j]; + + x1 = Math.min(x1, coord.x); + x2 = Math.max(x2, coord.x); + y1 = Math.min(y1, coord.y); + y2 = Math.max(y2, coord.y); + } + } + + return [x1, y1, x2, y2] +}; + +FeatureWrapper.prototype.toGeoJSON = VectorTileFeature.prototype.toGeoJSON; + +var geojson_wrapper$1 = /*@__PURE__*/performance.getDefaultExportFromCjs(geojson_wrapper); + +var vtPbf = vtPbf$1.exports; + +var Pbf = performance.pbf; +var GeoJSONWrapper = geojson_wrapper; + +vtPbf$1.exports = fromVectorTileJs; +var fromVectorTileJs_1 = vtPbf$1.exports.fromVectorTileJs = fromVectorTileJs; +var fromGeojsonVt_1 = vtPbf$1.exports.fromGeojsonVt = fromGeojsonVt; +var GeoJSONWrapper_1 = vtPbf$1.exports.GeoJSONWrapper = GeoJSONWrapper; + +/** + * Serialize a vector-tile-js-created tile to pbf + * + * @param {Object} tile + * @return {Buffer} uncompressed, pbf-serialized tile data + */ +function fromVectorTileJs (tile) { + var out = new Pbf(); + writeTile(tile, out); + return out.finish() +} + +/** + * Serialized a geojson-vt-created tile to pbf. + * + * @param {Object} layers - An object mapping layer names to geojson-vt-created vector tile objects + * @param {Object} [options] - An object specifying the vector-tile specification version and extent that were used to create `layers`. + * @param {Number} [options.version=1] - Version of vector-tile spec used + * @param {Number} [options.extent=4096] - Extent of the vector tile + * @return {Buffer} uncompressed, pbf-serialized tile data + */ +function fromGeojsonVt (layers, options) { + options = options || {}; + var l = {}; + for (var k in layers) { + l[k] = new GeoJSONWrapper(layers[k].features, options); + l[k].name = k; + l[k].version = options.version; + l[k].extent = options.extent; + } + return fromVectorTileJs({ layers: l }) +} + +function writeTile (tile, pbf) { + for (var key in tile.layers) { + pbf.writeMessage(3, writeLayer, tile.layers[key]); + } +} + +function writeLayer (layer, pbf) { + pbf.writeVarintField(15, layer.version || 1); + pbf.writeStringField(1, layer.name || ''); + pbf.writeVarintField(5, layer.extent || 4096); + + var i; + var context = { + keys: [], + values: [], + keycache: {}, + valuecache: {} + }; + + for (i = 0; i < layer.length; i++) { + context.feature = layer.feature(i); + pbf.writeMessage(2, writeFeature, context); + } + + var keys = context.keys; + for (i = 0; i < keys.length; i++) { + pbf.writeStringField(3, keys[i]); + } + + var values = context.values; + for (i = 0; i < values.length; i++) { + pbf.writeMessage(4, writeValue, values[i]); + } +} + +function writeFeature (context, pbf) { + var feature = context.feature; + + if (feature.id !== undefined) { + pbf.writeVarintField(1, feature.id); + } + + pbf.writeMessage(2, writeProperties, context); + pbf.writeVarintField(3, feature.type); + pbf.writeMessage(4, writeGeometry, feature); +} + +function writeProperties (context, pbf) { + var feature = context.feature; + var keys = context.keys; + var values = context.values; + var keycache = context.keycache; + var valuecache = context.valuecache; + + for (var key in feature.properties) { + var value = feature.properties[key]; + + var keyIndex = keycache[key]; + if (value === null) continue // don't encode null value properties + + if (typeof keyIndex === 'undefined') { + keys.push(key); + keyIndex = keys.length - 1; + keycache[key] = keyIndex; + } + pbf.writeVarint(keyIndex); + + var type = typeof value; + if (type !== 'string' && type !== 'boolean' && type !== 'number') { + value = JSON.stringify(value); + } + var valueKey = type + ':' + value; + var valueIndex = valuecache[valueKey]; + if (typeof valueIndex === 'undefined') { + values.push(value); + valueIndex = values.length - 1; + valuecache[valueKey] = valueIndex; + } + pbf.writeVarint(valueIndex); + } +} + +function command (cmd, length) { + return (length << 3) + (cmd & 0x7) +} + +function zigzag (num) { + return (num << 1) ^ (num >> 31) +} + +function writeGeometry (feature, pbf) { + var geometry = feature.loadGeometry(); + var type = feature.type; + var x = 0; + var y = 0; + var rings = geometry.length; + for (var r = 0; r < rings; r++) { + var ring = geometry[r]; + var count = 1; + if (type === 1) { + count = ring.length; + } + pbf.writeVarint(command(1, count)); // moveto + // do not write polygon closing path as lineto + var lineCount = type === 3 ? ring.length - 1 : ring.length; + for (var i = 0; i < lineCount; i++) { + if (i === 1 && type !== 1) { + pbf.writeVarint(command(2, lineCount - 1)); // lineto + } + var dx = ring[i].x - x; + var dy = ring[i].y - y; + pbf.writeVarint(zigzag(dx)); + pbf.writeVarint(zigzag(dy)); + x += dx; + y += dy; + } + if (type === 3) { + pbf.writeVarint(command(7, 1)); // closepath + } + } +} + +function writeValue (value, pbf) { + var type = typeof value; + if (type === 'string') { + pbf.writeStringField(1, value); + } else if (type === 'boolean') { + pbf.writeBooleanField(7, value); + } else if (type === 'number') { + if (value % 1 !== 0) { + pbf.writeDoubleField(3, value); + } else if (value < 0) { + pbf.writeSVarintField(6, value); + } else { + pbf.writeVarintField(5, value); + } + } +} + +var vtPbfExports = vtPbf$1.exports; +var vtpbf = /*@__PURE__*/performance.getDefaultExportFromCjs(vtPbfExports); + +const defaultOptions = { + minZoom: 0, // min zoom to generate clusters on + maxZoom: 16, // max zoom level to cluster the points on + minPoints: 2, // minimum points to form a cluster + radius: 40, // cluster radius in pixels + extent: 512, // tile extent (radius is calculated relative to it) + nodeSize: 64, // size of the KD-tree leaf node, affects performance + log: false, // whether to log timing info + + // whether to generate numeric ids for input features (in vector tiles) + generateId: false, + + // a reduce function for calculating custom cluster properties + reduce: null, // (accumulated, props) => { accumulated.sum += props.sum; } + + // properties to use for individual points when running the reducer + map: props => props // props => ({sum: props.my_value}) +}; + +const fround = Math.fround || (tmp => ((x) => { tmp[0] = +x; return tmp[0]; }))(new Float32Array(1)); + +const OFFSET_ZOOM = 2; +const OFFSET_ID = 3; +const OFFSET_PARENT = 4; +const OFFSET_NUM = 5; +const OFFSET_PROP = 6; + +class Supercluster { + constructor(options) { + this.options = Object.assign(Object.create(defaultOptions), options); + this.trees = new Array(this.options.maxZoom + 1); + this.stride = this.options.reduce ? 7 : 6; + this.clusterProps = []; + } + + load(points) { + const {log, minZoom, maxZoom} = this.options; + + if (log) console.time('total time'); + + const timerId = `prepare ${ points.length } points`; + if (log) console.time(timerId); + + this.points = points; + + // generate a cluster object for each point and index input points into a KD-tree + const data = []; + + for (let i = 0; i < points.length; i++) { + const p = points[i]; + if (!p.geometry) continue; + + const [lng, lat] = p.geometry.coordinates; + const x = fround(lngX(lng)); + const y = fround(latY(lat)); + // store internal point/cluster data in flat numeric arrays for performance + data.push( + x, y, // projected point coordinates + Infinity, // the last zoom the point was processed at + i, // index of the source feature in the original input array + -1, // parent cluster id + 1 // number of points in a cluster + ); + if (this.options.reduce) data.push(0); // noop + } + let tree = this.trees[maxZoom + 1] = this._createTree(data); + + if (log) console.timeEnd(timerId); + + // cluster points on max zoom, then cluster the results on previous zoom, etc.; + // results in a cluster hierarchy across zoom levels + for (let z = maxZoom; z >= minZoom; z--) { + const now = +Date.now(); + + // create a new set of clusters for the zoom and index them with a KD-tree + tree = this.trees[z] = this._createTree(this._cluster(tree, z)); + + if (log) console.log('z%d: %d clusters in %dms', z, tree.numItems, +Date.now() - now); + } + + if (log) console.timeEnd('total time'); + + return this; + } + + getClusters(bbox, zoom) { + let minLng = ((bbox[0] + 180) % 360 + 360) % 360 - 180; + const minLat = Math.max(-90, Math.min(90, bbox[1])); + let maxLng = bbox[2] === 180 ? 180 : ((bbox[2] + 180) % 360 + 360) % 360 - 180; + const maxLat = Math.max(-90, Math.min(90, bbox[3])); + + if (bbox[2] - bbox[0] >= 360) { + minLng = -180; + maxLng = 180; + } else if (minLng > maxLng) { + const easternHem = this.getClusters([minLng, minLat, 180, maxLat], zoom); + const westernHem = this.getClusters([-180, minLat, maxLng, maxLat], zoom); + return easternHem.concat(westernHem); + } + + const tree = this.trees[this._limitZoom(zoom)]; + const ids = tree.range(lngX(minLng), latY(maxLat), lngX(maxLng), latY(minLat)); + const data = tree.data; + const clusters = []; + for (const id of ids) { + const k = this.stride * id; + clusters.push(data[k + OFFSET_NUM] > 1 ? getClusterJSON(data, k, this.clusterProps) : this.points[data[k + OFFSET_ID]]); + } + return clusters; + } + + getChildren(clusterId) { + const originId = this._getOriginId(clusterId); + const originZoom = this._getOriginZoom(clusterId); + const errorMsg = 'No cluster with the specified id.'; + + const tree = this.trees[originZoom]; + if (!tree) throw new Error(errorMsg); + + const data = tree.data; + if (originId * this.stride >= data.length) throw new Error(errorMsg); + + const r = this.options.radius / (this.options.extent * Math.pow(2, originZoom - 1)); + const x = data[originId * this.stride]; + const y = data[originId * this.stride + 1]; + const ids = tree.within(x, y, r); + const children = []; + for (const id of ids) { + const k = id * this.stride; + if (data[k + OFFSET_PARENT] === clusterId) { + children.push(data[k + OFFSET_NUM] > 1 ? getClusterJSON(data, k, this.clusterProps) : this.points[data[k + OFFSET_ID]]); + } + } + + if (children.length === 0) throw new Error(errorMsg); + + return children; + } + + getLeaves(clusterId, limit, offset) { + limit = limit || 10; + offset = offset || 0; + + const leaves = []; + this._appendLeaves(leaves, clusterId, limit, offset, 0); + + return leaves; + } + + getTile(z, x, y) { + const tree = this.trees[this._limitZoom(z)]; + const z2 = Math.pow(2, z); + const {extent, radius} = this.options; + const p = radius / extent; + const top = (y - p) / z2; + const bottom = (y + 1 + p) / z2; + + const tile = { + features: [] + }; + + this._addTileFeatures( + tree.range((x - p) / z2, top, (x + 1 + p) / z2, bottom), + tree.data, x, y, z2, tile); + + if (x === 0) { + this._addTileFeatures( + tree.range(1 - p / z2, top, 1, bottom), + tree.data, z2, y, z2, tile); + } + if (x === z2 - 1) { + this._addTileFeatures( + tree.range(0, top, p / z2, bottom), + tree.data, -1, y, z2, tile); + } + + return tile.features.length ? tile : null; + } + + getClusterExpansionZoom(clusterId) { + let expansionZoom = this._getOriginZoom(clusterId) - 1; + while (expansionZoom <= this.options.maxZoom) { + const children = this.getChildren(clusterId); + expansionZoom++; + if (children.length !== 1) break; + clusterId = children[0].properties.cluster_id; + } + return expansionZoom; + } + + _appendLeaves(result, clusterId, limit, offset, skipped) { + const children = this.getChildren(clusterId); + + for (const child of children) { + const props = child.properties; + + if (props && props.cluster) { + if (skipped + props.point_count <= offset) { + // skip the whole cluster + skipped += props.point_count; + } else { + // enter the cluster + skipped = this._appendLeaves(result, props.cluster_id, limit, offset, skipped); + // exit the cluster + } + } else if (skipped < offset) { + // skip a single point + skipped++; + } else { + // add a single point + result.push(child); + } + if (result.length === limit) break; + } + + return skipped; + } + + _createTree(data) { + const tree = new performance.KDBush(data.length / this.stride | 0, this.options.nodeSize, Float32Array); + for (let i = 0; i < data.length; i += this.stride) tree.add(data[i], data[i + 1]); + tree.finish(); + tree.data = data; + return tree; + } + + _addTileFeatures(ids, data, x, y, z2, tile) { + for (const i of ids) { + const k = i * this.stride; + const isCluster = data[k + OFFSET_NUM] > 1; + + let tags, px, py; + if (isCluster) { + tags = getClusterProperties(data, k, this.clusterProps); + px = data[k]; + py = data[k + 1]; + } else { + const p = this.points[data[k + OFFSET_ID]]; + tags = p.properties; + const [lng, lat] = p.geometry.coordinates; + px = lngX(lng); + py = latY(lat); + } + + const f = { + type: 1, + geometry: [[ + Math.round(this.options.extent * (px * z2 - x)), + Math.round(this.options.extent * (py * z2 - y)) + ]], + tags + }; + + // assign id + let id; + if (isCluster || this.options.generateId) { + // optionally generate id for points + id = data[k + OFFSET_ID]; + } else { + // keep id if already assigned + id = this.points[data[k + OFFSET_ID]].id; + } + + if (id !== undefined) f.id = id; + + tile.features.push(f); + } + } + + _limitZoom(z) { + return Math.max(this.options.minZoom, Math.min(Math.floor(+z), this.options.maxZoom + 1)); + } + + _cluster(tree, zoom) { + const {radius, extent, reduce, minPoints} = this.options; + const r = radius / (extent * Math.pow(2, zoom)); + const data = tree.data; + const nextData = []; + const stride = this.stride; + + // loop through each point + for (let i = 0; i < data.length; i += stride) { + // if we've already visited the point at this zoom level, skip it + if (data[i + OFFSET_ZOOM] <= zoom) continue; + data[i + OFFSET_ZOOM] = zoom; + + // find all nearby points + const x = data[i]; + const y = data[i + 1]; + const neighborIds = tree.within(data[i], data[i + 1], r); + + const numPointsOrigin = data[i + OFFSET_NUM]; + let numPoints = numPointsOrigin; + + // count the number of points in a potential cluster + for (const neighborId of neighborIds) { + const k = neighborId * stride; + // filter out neighbors that are already processed + if (data[k + OFFSET_ZOOM] > zoom) numPoints += data[k + OFFSET_NUM]; + } + + // if there were neighbors to merge, and there are enough points to form a cluster + if (numPoints > numPointsOrigin && numPoints >= minPoints) { + let wx = x * numPointsOrigin; + let wy = y * numPointsOrigin; + + let clusterProperties; + let clusterPropIndex = -1; + + // encode both zoom and point index on which the cluster originated -- offset by total length of features + const id = ((i / stride | 0) << 5) + (zoom + 1) + this.points.length; + + for (const neighborId of neighborIds) { + const k = neighborId * stride; + + if (data[k + OFFSET_ZOOM] <= zoom) continue; + data[k + OFFSET_ZOOM] = zoom; // save the zoom (so it doesn't get processed twice) + + const numPoints2 = data[k + OFFSET_NUM]; + wx += data[k] * numPoints2; // accumulate coordinates for calculating weighted center + wy += data[k + 1] * numPoints2; + + data[k + OFFSET_PARENT] = id; + + if (reduce) { + if (!clusterProperties) { + clusterProperties = this._map(data, i, true); + clusterPropIndex = this.clusterProps.length; + this.clusterProps.push(clusterProperties); + } + reduce(clusterProperties, this._map(data, k)); + } + } + + data[i + OFFSET_PARENT] = id; + nextData.push(wx / numPoints, wy / numPoints, Infinity, id, -1, numPoints); + if (reduce) nextData.push(clusterPropIndex); + + } else { // left points as unclustered + for (let j = 0; j < stride; j++) nextData.push(data[i + j]); + + if (numPoints > 1) { + for (const neighborId of neighborIds) { + const k = neighborId * stride; + if (data[k + OFFSET_ZOOM] <= zoom) continue; + data[k + OFFSET_ZOOM] = zoom; + for (let j = 0; j < stride; j++) nextData.push(data[k + j]); + } + } + } + } + + return nextData; + } + + // get index of the point from which the cluster originated + _getOriginId(clusterId) { + return (clusterId - this.points.length) >> 5; + } + + // get zoom of the point from which the cluster originated + _getOriginZoom(clusterId) { + return (clusterId - this.points.length) % 32; + } + + _map(data, i, clone) { + if (data[i + OFFSET_NUM] > 1) { + const props = this.clusterProps[data[i + OFFSET_PROP]]; + return clone ? Object.assign({}, props) : props; + } + const original = this.points[data[i + OFFSET_ID]].properties; + const result = this.options.map(original); + return clone && result === original ? Object.assign({}, result) : result; + } +} + +function getClusterJSON(data, i, clusterProps) { + return { + type: 'Feature', + id: data[i + OFFSET_ID], + properties: getClusterProperties(data, i, clusterProps), + geometry: { + type: 'Point', + coordinates: [xLng(data[i]), yLat(data[i + 1])] + } + }; +} + +function getClusterProperties(data, i, clusterProps) { + const count = data[i + OFFSET_NUM]; + const abbrev = + count >= 10000 ? `${Math.round(count / 1000) }k` : + count >= 1000 ? `${Math.round(count / 100) / 10 }k` : count; + const propIndex = data[i + OFFSET_PROP]; + const properties = propIndex === -1 ? {} : Object.assign({}, clusterProps[propIndex]); + return Object.assign(properties, { + cluster: true, + cluster_id: data[i + OFFSET_ID], + point_count: count, + point_count_abbreviated: abbrev + }); +} + +// longitude/latitude to spherical mercator in [0..1] range +function lngX(lng) { + return lng / 360 + 0.5; +} +function latY(lat) { + const sin = Math.sin(lat * Math.PI / 180); + const y = (0.5 - 0.25 * Math.log((1 + sin) / (1 - sin)) / Math.PI); + return y < 0 ? 0 : y > 1 ? 1 : y; +} + +// spherical mercator to longitude/latitude +function xLng(x) { + return (x - 0.5) * 360; +} +function yLat(y) { + const y2 = (180 - y * 360) * Math.PI / 180; + return 360 * Math.atan(Math.exp(y2)) / Math.PI - 90; +} + +// calculate simplification data using optimized Douglas-Peucker algorithm + +function simplify(coords, first, last, sqTolerance) { + var maxSqDist = sqTolerance; + var mid = (last - first) >> 1; + var minPosToMid = last - first; + var index; + + var ax = coords[first]; + var ay = coords[first + 1]; + var bx = coords[last]; + var by = coords[last + 1]; + + for (var i = first + 3; i < last; i += 3) { + var d = getSqSegDist(coords[i], coords[i + 1], ax, ay, bx, by); + + if (d > maxSqDist) { + index = i; + maxSqDist = d; + + } else if (d === maxSqDist) { + // a workaround to ensure we choose a pivot close to the middle of the list, + // reducing recursion depth, for certain degenerate inputs + // https://github.com/mapbox/geojson-vt/issues/104 + var posToMid = Math.abs(i - mid); + if (posToMid < minPosToMid) { + index = i; + minPosToMid = posToMid; + } + } + } + + if (maxSqDist > sqTolerance) { + if (index - first > 3) simplify(coords, first, index, sqTolerance); + coords[index + 2] = maxSqDist; + if (last - index > 3) simplify(coords, index, last, sqTolerance); + } +} + +// square distance from a point to a segment +function getSqSegDist(px, py, x, y, bx, by) { + + var dx = bx - x; + var dy = by - y; + + if (dx !== 0 || dy !== 0) { + + var t = ((px - x) * dx + (py - y) * dy) / (dx * dx + dy * dy); + + if (t > 1) { + x = bx; + y = by; + + } else if (t > 0) { + x += dx * t; + y += dy * t; + } + } + + dx = px - x; + dy = py - y; + + return dx * dx + dy * dy; +} + +function createFeature(id, type, geom, tags) { + var feature = { + id: typeof id === 'undefined' ? null : id, + type: type, + geometry: geom, + tags: tags, + minX: Infinity, + minY: Infinity, + maxX: -Infinity, + maxY: -Infinity + }; + calcBBox(feature); + return feature; +} + +function calcBBox(feature) { + var geom = feature.geometry; + var type = feature.type; + + if (type === 'Point' || type === 'MultiPoint' || type === 'LineString') { + calcLineBBox(feature, geom); + + } else if (type === 'Polygon' || type === 'MultiLineString') { + for (var i = 0; i < geom.length; i++) { + calcLineBBox(feature, geom[i]); + } + + } else if (type === 'MultiPolygon') { + for (i = 0; i < geom.length; i++) { + for (var j = 0; j < geom[i].length; j++) { + calcLineBBox(feature, geom[i][j]); + } + } + } +} + +function calcLineBBox(feature, geom) { + for (var i = 0; i < geom.length; i += 3) { + feature.minX = Math.min(feature.minX, geom[i]); + feature.minY = Math.min(feature.minY, geom[i + 1]); + feature.maxX = Math.max(feature.maxX, geom[i]); + feature.maxY = Math.max(feature.maxY, geom[i + 1]); + } +} + +// converts GeoJSON feature into an intermediate projected JSON vector format with simplification data + +function convert(data, options) { + var features = []; + if (data.type === 'FeatureCollection') { + for (var i = 0; i < data.features.length; i++) { + convertFeature(features, data.features[i], options, i); + } + + } else if (data.type === 'Feature') { + convertFeature(features, data, options); + + } else { + // single geometry or a geometry collection + convertFeature(features, {geometry: data}, options); + } + + return features; +} + +function convertFeature(features, geojson, options, index) { + if (!geojson.geometry) return; + + var coords = geojson.geometry.coordinates; + var type = geojson.geometry.type; + var tolerance = Math.pow(options.tolerance / ((1 << options.maxZoom) * options.extent), 2); + var geometry = []; + var id = geojson.id; + if (options.promoteId) { + id = geojson.properties[options.promoteId]; + } else if (options.generateId) { + id = index || 0; + } + if (type === 'Point') { + convertPoint(coords, geometry); + + } else if (type === 'MultiPoint') { + for (var i = 0; i < coords.length; i++) { + convertPoint(coords[i], geometry); + } + + } else if (type === 'LineString') { + convertLine(coords, geometry, tolerance, false); + + } else if (type === 'MultiLineString') { + if (options.lineMetrics) { + // explode into linestrings to be able to track metrics + for (i = 0; i < coords.length; i++) { + geometry = []; + convertLine(coords[i], geometry, tolerance, false); + features.push(createFeature(id, 'LineString', geometry, geojson.properties)); + } + return; + } else { + convertLines(coords, geometry, tolerance, false); + } + + } else if (type === 'Polygon') { + convertLines(coords, geometry, tolerance, true); + + } else if (type === 'MultiPolygon') { + for (i = 0; i < coords.length; i++) { + var polygon = []; + convertLines(coords[i], polygon, tolerance, true); + geometry.push(polygon); + } + } else if (type === 'GeometryCollection') { + for (i = 0; i < geojson.geometry.geometries.length; i++) { + convertFeature(features, { + id: id, + geometry: geojson.geometry.geometries[i], + properties: geojson.properties + }, options, index); + } + return; + } else { + throw new Error('Input data is not a valid GeoJSON object.'); + } + + features.push(createFeature(id, type, geometry, geojson.properties)); +} + +function convertPoint(coords, out) { + out.push(projectX(coords[0])); + out.push(projectY(coords[1])); + out.push(0); +} + +function convertLine(ring, out, tolerance, isPolygon) { + var x0, y0; + var size = 0; + + for (var j = 0; j < ring.length; j++) { + var x = projectX(ring[j][0]); + var y = projectY(ring[j][1]); + + out.push(x); + out.push(y); + out.push(0); + + if (j > 0) { + if (isPolygon) { + size += (x0 * y - x * y0) / 2; // area + } else { + size += Math.sqrt(Math.pow(x - x0, 2) + Math.pow(y - y0, 2)); // length + } + } + x0 = x; + y0 = y; + } + + var last = out.length - 3; + out[2] = 1; + simplify(out, 0, last, tolerance); + out[last + 2] = 1; + + out.size = Math.abs(size); + out.start = 0; + out.end = out.size; +} + +function convertLines(rings, out, tolerance, isPolygon) { + for (var i = 0; i < rings.length; i++) { + var geom = []; + convertLine(rings[i], geom, tolerance, isPolygon); + out.push(geom); + } +} + +function projectX(x) { + return x / 360 + 0.5; +} + +function projectY(y) { + var sin = Math.sin(y * Math.PI / 180); + var y2 = 0.5 - 0.25 * Math.log((1 + sin) / (1 - sin)) / Math.PI; + return y2 < 0 ? 0 : y2 > 1 ? 1 : y2; +} + +/* clip features between two axis-parallel lines: + * | | + * ___|___ | / + * / | \____|____/ + * | | + */ + +function clip(features, scale, k1, k2, axis, minAll, maxAll, options) { + + k1 /= scale; + k2 /= scale; + + if (minAll >= k1 && maxAll < k2) return features; // trivial accept + else if (maxAll < k1 || minAll >= k2) return null; // trivial reject + + var clipped = []; + + for (var i = 0; i < features.length; i++) { + + var feature = features[i]; + var geometry = feature.geometry; + var type = feature.type; + + var min = axis === 0 ? feature.minX : feature.minY; + var max = axis === 0 ? feature.maxX : feature.maxY; + + if (min >= k1 && max < k2) { // trivial accept + clipped.push(feature); + continue; + } else if (max < k1 || min >= k2) { // trivial reject + continue; + } + + var newGeometry = []; + + if (type === 'Point' || type === 'MultiPoint') { + clipPoints(geometry, newGeometry, k1, k2, axis); + + } else if (type === 'LineString') { + clipLine(geometry, newGeometry, k1, k2, axis, false, options.lineMetrics); + + } else if (type === 'MultiLineString') { + clipLines(geometry, newGeometry, k1, k2, axis, false); + + } else if (type === 'Polygon') { + clipLines(geometry, newGeometry, k1, k2, axis, true); + + } else if (type === 'MultiPolygon') { + for (var j = 0; j < geometry.length; j++) { + var polygon = []; + clipLines(geometry[j], polygon, k1, k2, axis, true); + if (polygon.length) { + newGeometry.push(polygon); + } + } + } + + if (newGeometry.length) { + if (options.lineMetrics && type === 'LineString') { + for (j = 0; j < newGeometry.length; j++) { + clipped.push(createFeature(feature.id, type, newGeometry[j], feature.tags)); + } + continue; + } + + if (type === 'LineString' || type === 'MultiLineString') { + if (newGeometry.length === 1) { + type = 'LineString'; + newGeometry = newGeometry[0]; + } else { + type = 'MultiLineString'; + } + } + if (type === 'Point' || type === 'MultiPoint') { + type = newGeometry.length === 3 ? 'Point' : 'MultiPoint'; + } + + clipped.push(createFeature(feature.id, type, newGeometry, feature.tags)); + } + } + + return clipped.length ? clipped : null; +} + +function clipPoints(geom, newGeom, k1, k2, axis) { + for (var i = 0; i < geom.length; i += 3) { + var a = geom[i + axis]; + + if (a >= k1 && a <= k2) { + newGeom.push(geom[i]); + newGeom.push(geom[i + 1]); + newGeom.push(geom[i + 2]); + } + } +} + +function clipLine(geom, newGeom, k1, k2, axis, isPolygon, trackMetrics) { + + var slice = newSlice(geom); + var intersect = axis === 0 ? intersectX : intersectY; + var len = geom.start; + var segLen, t; + + for (var i = 0; i < geom.length - 3; i += 3) { + var ax = geom[i]; + var ay = geom[i + 1]; + var az = geom[i + 2]; + var bx = geom[i + 3]; + var by = geom[i + 4]; + var a = axis === 0 ? ax : ay; + var b = axis === 0 ? bx : by; + var exited = false; + + if (trackMetrics) segLen = Math.sqrt(Math.pow(ax - bx, 2) + Math.pow(ay - by, 2)); + + if (a < k1) { + // ---|--> | (line enters the clip region from the left) + if (b > k1) { + t = intersect(slice, ax, ay, bx, by, k1); + if (trackMetrics) slice.start = len + segLen * t; + } + } else if (a > k2) { + // | <--|--- (line enters the clip region from the right) + if (b < k2) { + t = intersect(slice, ax, ay, bx, by, k2); + if (trackMetrics) slice.start = len + segLen * t; + } + } else { + addPoint(slice, ax, ay, az); + } + if (b < k1 && a >= k1) { + // <--|--- | or <--|-----|--- (line exits the clip region on the left) + t = intersect(slice, ax, ay, bx, by, k1); + exited = true; + } + if (b > k2 && a <= k2) { + // | ---|--> or ---|-----|--> (line exits the clip region on the right) + t = intersect(slice, ax, ay, bx, by, k2); + exited = true; + } + + if (!isPolygon && exited) { + if (trackMetrics) slice.end = len + segLen * t; + newGeom.push(slice); + slice = newSlice(geom); + } + + if (trackMetrics) len += segLen; + } + + // add the last point + var last = geom.length - 3; + ax = geom[last]; + ay = geom[last + 1]; + az = geom[last + 2]; + a = axis === 0 ? ax : ay; + if (a >= k1 && a <= k2) addPoint(slice, ax, ay, az); + + // close the polygon if its endpoints are not the same after clipping + last = slice.length - 3; + if (isPolygon && last >= 3 && (slice[last] !== slice[0] || slice[last + 1] !== slice[1])) { + addPoint(slice, slice[0], slice[1], slice[2]); + } + + // add the final slice + if (slice.length) { + newGeom.push(slice); + } +} + +function newSlice(line) { + var slice = []; + slice.size = line.size; + slice.start = line.start; + slice.end = line.end; + return slice; +} + +function clipLines(geom, newGeom, k1, k2, axis, isPolygon) { + for (var i = 0; i < geom.length; i++) { + clipLine(geom[i], newGeom, k1, k2, axis, isPolygon, false); + } +} + +function addPoint(out, x, y, z) { + out.push(x); + out.push(y); + out.push(z); +} + +function intersectX(out, ax, ay, bx, by, x) { + var t = (x - ax) / (bx - ax); + out.push(x); + out.push(ay + (by - ay) * t); + out.push(1); + return t; +} + +function intersectY(out, ax, ay, bx, by, y) { + var t = (y - ay) / (by - ay); + out.push(ax + (bx - ax) * t); + out.push(y); + out.push(1); + return t; +} + +function wrap(features, options) { + var buffer = options.buffer / options.extent; + var merged = features; + var left = clip(features, 1, -1 - buffer, buffer, 0, -1, 2, options); // left world copy + var right = clip(features, 1, 1 - buffer, 2 + buffer, 0, -1, 2, options); // right world copy + + if (left || right) { + merged = clip(features, 1, -buffer, 1 + buffer, 0, -1, 2, options) || []; // center world copy + + if (left) merged = shiftFeatureCoords(left, 1).concat(merged); // merge left into center + if (right) merged = merged.concat(shiftFeatureCoords(right, -1)); // merge right into center + } + + return merged; +} + +function shiftFeatureCoords(features, offset) { + var newFeatures = []; + + for (var i = 0; i < features.length; i++) { + var feature = features[i], + type = feature.type; + + var newGeometry; + + if (type === 'Point' || type === 'MultiPoint' || type === 'LineString') { + newGeometry = shiftCoords(feature.geometry, offset); + + } else if (type === 'MultiLineString' || type === 'Polygon') { + newGeometry = []; + for (var j = 0; j < feature.geometry.length; j++) { + newGeometry.push(shiftCoords(feature.geometry[j], offset)); + } + } else if (type === 'MultiPolygon') { + newGeometry = []; + for (j = 0; j < feature.geometry.length; j++) { + var newPolygon = []; + for (var k = 0; k < feature.geometry[j].length; k++) { + newPolygon.push(shiftCoords(feature.geometry[j][k], offset)); + } + newGeometry.push(newPolygon); + } + } + + newFeatures.push(createFeature(feature.id, type, newGeometry, feature.tags)); + } + + return newFeatures; +} + +function shiftCoords(points, offset) { + var newPoints = []; + newPoints.size = points.size; + + if (points.start !== undefined) { + newPoints.start = points.start; + newPoints.end = points.end; + } + + for (var i = 0; i < points.length; i += 3) { + newPoints.push(points[i] + offset, points[i + 1], points[i + 2]); + } + return newPoints; +} + +// Transforms the coordinates of each feature in the given tile from +// mercator-projected space into (extent x extent) tile space. +function transformTile(tile, extent) { + if (tile.transformed) return tile; + + var z2 = 1 << tile.z, + tx = tile.x, + ty = tile.y, + i, j, k; + + for (i = 0; i < tile.features.length; i++) { + var feature = tile.features[i], + geom = feature.geometry, + type = feature.type; + + feature.geometry = []; + + if (type === 1) { + for (j = 0; j < geom.length; j += 2) { + feature.geometry.push(transformPoint(geom[j], geom[j + 1], extent, z2, tx, ty)); + } + } else { + for (j = 0; j < geom.length; j++) { + var ring = []; + for (k = 0; k < geom[j].length; k += 2) { + ring.push(transformPoint(geom[j][k], geom[j][k + 1], extent, z2, tx, ty)); + } + feature.geometry.push(ring); + } + } + } + + tile.transformed = true; + + return tile; +} + +function transformPoint(x, y, extent, z2, tx, ty) { + return [ + Math.round(extent * (x * z2 - tx)), + Math.round(extent * (y * z2 - ty))]; +} + +function createTile(features, z, tx, ty, options) { + var tolerance = z === options.maxZoom ? 0 : options.tolerance / ((1 << z) * options.extent); + var tile = { + features: [], + numPoints: 0, + numSimplified: 0, + numFeatures: 0, + source: null, + x: tx, + y: ty, + z: z, + transformed: false, + minX: 2, + minY: 1, + maxX: -1, + maxY: 0 + }; + for (var i = 0; i < features.length; i++) { + tile.numFeatures++; + addFeature(tile, features[i], tolerance, options); + + var minX = features[i].minX; + var minY = features[i].minY; + var maxX = features[i].maxX; + var maxY = features[i].maxY; + + if (minX < tile.minX) tile.minX = minX; + if (minY < tile.minY) tile.minY = minY; + if (maxX > tile.maxX) tile.maxX = maxX; + if (maxY > tile.maxY) tile.maxY = maxY; + } + return tile; +} + +function addFeature(tile, feature, tolerance, options) { + + var geom = feature.geometry, + type = feature.type, + simplified = []; + + if (type === 'Point' || type === 'MultiPoint') { + for (var i = 0; i < geom.length; i += 3) { + simplified.push(geom[i]); + simplified.push(geom[i + 1]); + tile.numPoints++; + tile.numSimplified++; + } + + } else if (type === 'LineString') { + addLine(simplified, geom, tile, tolerance, false, false); + + } else if (type === 'MultiLineString' || type === 'Polygon') { + for (i = 0; i < geom.length; i++) { + addLine(simplified, geom[i], tile, tolerance, type === 'Polygon', i === 0); + } + + } else if (type === 'MultiPolygon') { + + for (var k = 0; k < geom.length; k++) { + var polygon = geom[k]; + for (i = 0; i < polygon.length; i++) { + addLine(simplified, polygon[i], tile, tolerance, true, i === 0); + } + } + } + + if (simplified.length) { + var tags = feature.tags || null; + if (type === 'LineString' && options.lineMetrics) { + tags = {}; + for (var key in feature.tags) tags[key] = feature.tags[key]; + tags['mapbox_clip_start'] = geom.start / geom.size; + tags['mapbox_clip_end'] = geom.end / geom.size; + } + var tileFeature = { + geometry: simplified, + type: type === 'Polygon' || type === 'MultiPolygon' ? 3 : + type === 'LineString' || type === 'MultiLineString' ? 2 : 1, + tags: tags + }; + if (feature.id !== null) { + tileFeature.id = feature.id; + } + tile.features.push(tileFeature); + } +} + +function addLine(result, geom, tile, tolerance, isPolygon, isOuter) { + var sqTolerance = tolerance * tolerance; + + if (tolerance > 0 && (geom.size < (isPolygon ? sqTolerance : tolerance))) { + tile.numPoints += geom.length / 3; + return; + } + + var ring = []; + + for (var i = 0; i < geom.length; i += 3) { + if (tolerance === 0 || geom[i + 2] > sqTolerance) { + tile.numSimplified++; + ring.push(geom[i]); + ring.push(geom[i + 1]); + } + tile.numPoints++; + } + + if (isPolygon) rewind(ring, isOuter); + + result.push(ring); +} + +function rewind(ring, clockwise) { + var area = 0; + for (var i = 0, len = ring.length, j = len - 2; i < len; j = i, i += 2) { + area += (ring[i] - ring[j]) * (ring[i + 1] + ring[j + 1]); + } + if (area > 0 === clockwise) { + for (i = 0, len = ring.length; i < len / 2; i += 2) { + var x = ring[i]; + var y = ring[i + 1]; + ring[i] = ring[len - 2 - i]; + ring[i + 1] = ring[len - 1 - i]; + ring[len - 2 - i] = x; + ring[len - 1 - i] = y; + } + } +} + +function geojsonvt(data, options) { + return new GeoJSONVT(data, options); +} + +function GeoJSONVT(data, options) { + options = this.options = extend(Object.create(this.options), options); + + var debug = options.debug; + + if (debug) console.time('preprocess data'); + + if (options.maxZoom < 0 || options.maxZoom > 24) throw new Error('maxZoom should be in the 0-24 range'); + if (options.promoteId && options.generateId) throw new Error('promoteId and generateId cannot be used together.'); + + var features = convert(data, options); + + this.tiles = {}; + this.tileCoords = []; + + if (debug) { + console.timeEnd('preprocess data'); + console.log('index: maxZoom: %d, maxPoints: %d', options.indexMaxZoom, options.indexMaxPoints); + console.time('generate tiles'); + this.stats = {}; + this.total = 0; + } + + features = wrap(features, options); + + // start slicing from the top tile down + if (features.length) this.splitTile(features, 0, 0, 0); + + if (debug) { + if (features.length) console.log('features: %d, points: %d', this.tiles[0].numFeatures, this.tiles[0].numPoints); + console.timeEnd('generate tiles'); + console.log('tiles generated:', this.total, JSON.stringify(this.stats)); + } +} + +GeoJSONVT.prototype.options = { + maxZoom: 14, // max zoom to preserve detail on + indexMaxZoom: 5, // max zoom in the tile index + indexMaxPoints: 100000, // max number of points per tile in the tile index + tolerance: 3, // simplification tolerance (higher means simpler) + extent: 4096, // tile extent + buffer: 64, // tile buffer on each side + lineMetrics: false, // whether to calculate line metrics + promoteId: null, // name of a feature property to be promoted to feature.id + generateId: false, // whether to generate feature ids. Cannot be used with promoteId + debug: 0 // logging level (0, 1 or 2) +}; + +GeoJSONVT.prototype.splitTile = function (features, z, x, y, cz, cx, cy) { + + var stack = [features, z, x, y], + options = this.options, + debug = options.debug; + + // avoid recursion by using a processing queue + while (stack.length) { + y = stack.pop(); + x = stack.pop(); + z = stack.pop(); + features = stack.pop(); + + var z2 = 1 << z, + id = toID(z, x, y), + tile = this.tiles[id]; + + if (!tile) { + if (debug > 1) console.time('creation'); + + tile = this.tiles[id] = createTile(features, z, x, y, options); + this.tileCoords.push({z: z, x: x, y: y}); + + if (debug) { + if (debug > 1) { + console.log('tile z%d-%d-%d (features: %d, points: %d, simplified: %d)', + z, x, y, tile.numFeatures, tile.numPoints, tile.numSimplified); + console.timeEnd('creation'); + } + var key = 'z' + z; + this.stats[key] = (this.stats[key] || 0) + 1; + this.total++; + } + } + + // save reference to original geometry in tile so that we can drill down later if we stop now + tile.source = features; + + // if it's the first-pass tiling + if (!cz) { + // stop tiling if we reached max zoom, or if the tile is too simple + if (z === options.indexMaxZoom || tile.numPoints <= options.indexMaxPoints) continue; + + // if a drilldown to a specific tile + } else { + // stop tiling if we reached base zoom or our target tile zoom + if (z === options.maxZoom || z === cz) continue; + + // stop tiling if it's not an ancestor of the target tile + var m = 1 << (cz - z); + if (x !== Math.floor(cx / m) || y !== Math.floor(cy / m)) continue; + } + + // if we slice further down, no need to keep source geometry + tile.source = null; + + if (features.length === 0) continue; + + if (debug > 1) console.time('clipping'); + + // values we'll use for clipping + var k1 = 0.5 * options.buffer / options.extent, + k2 = 0.5 - k1, + k3 = 0.5 + k1, + k4 = 1 + k1, + tl, bl, tr, br, left, right; + + tl = bl = tr = br = null; + + left = clip(features, z2, x - k1, x + k3, 0, tile.minX, tile.maxX, options); + right = clip(features, z2, x + k2, x + k4, 0, tile.minX, tile.maxX, options); + features = null; + + if (left) { + tl = clip(left, z2, y - k1, y + k3, 1, tile.minY, tile.maxY, options); + bl = clip(left, z2, y + k2, y + k4, 1, tile.minY, tile.maxY, options); + left = null; + } + + if (right) { + tr = clip(right, z2, y - k1, y + k3, 1, tile.minY, tile.maxY, options); + br = clip(right, z2, y + k2, y + k4, 1, tile.minY, tile.maxY, options); + right = null; + } + + if (debug > 1) console.timeEnd('clipping'); + + stack.push(tl || [], z + 1, x * 2, y * 2); + stack.push(bl || [], z + 1, x * 2, y * 2 + 1); + stack.push(tr || [], z + 1, x * 2 + 1, y * 2); + stack.push(br || [], z + 1, x * 2 + 1, y * 2 + 1); + } +}; + +GeoJSONVT.prototype.getTile = function (z, x, y) { + var options = this.options, + extent = options.extent, + debug = options.debug; + + if (z < 0 || z > 24) return null; + + var z2 = 1 << z; + x = ((x % z2) + z2) % z2; // wrap tile x coordinate + + var id = toID(z, x, y); + if (this.tiles[id]) return transformTile(this.tiles[id], extent); + + if (debug > 1) console.log('drilling down to z%d-%d-%d', z, x, y); + + var z0 = z, + x0 = x, + y0 = y, + parent; + + while (!parent && z0 > 0) { + z0--; + x0 = Math.floor(x0 / 2); + y0 = Math.floor(y0 / 2); + parent = this.tiles[toID(z0, x0, y0)]; + } + + if (!parent || !parent.source) return null; + + // if we found a parent tile containing the original geometry, we can drill down from it + if (debug > 1) console.log('found parent tile z%d-%d-%d', z0, x0, y0); + + if (debug > 1) console.time('drilling down'); + this.splitTile(parent.source, z0, x0, y0, z, x, y); + if (debug > 1) console.timeEnd('drilling down'); + + return this.tiles[id] ? transformTile(this.tiles[id], extent) : null; +}; + +function toID(z, x, y) { + return (((1 << z) * y + x) * 32) + z; +} + +function extend(dest, src) { + for (var i in src) dest[i] = src[i]; + return dest; +} + +function getFeatureId(feature, promoteId) { + return promoteId ? feature.properties[promoteId] : feature.id; +} +function isUpdateableGeoJSON(data, promoteId) { + // null can be updated + if (data == null) { + return true; + } + // a single feature with an id can be updated, need to explicitly check against null because 0 is a valid feature id that is falsy + if (data.type === 'Feature') { + return getFeatureId(data, promoteId) != null; + } + // a feature collection can be updated if every feature has an id, and the ids are all unique + // this prevents us from silently dropping features if ids get reused + if (data.type === 'FeatureCollection') { + const seenIds = new Set(); + for (const feature of data.features) { + const id = getFeatureId(feature, promoteId); + if (id == null) { + return false; + } + if (seenIds.has(id)) { + return false; + } + seenIds.add(id); + } + return true; + } + return false; +} +function toUpdateable(data, promoteId) { + const result = new Map(); + if (data == null) { + // empty result + } + else if (data.type === 'Feature') { + result.set(getFeatureId(data, promoteId), data); + } + else { + for (const feature of data.features) { + result.set(getFeatureId(feature, promoteId), feature); + } + } + return result; +} +// mutates updateable +function applySourceDiff(updateable, diff, promoteId) { + var _a, _b, _c, _d; + if (diff.removeAll) { + updateable.clear(); + } + if (diff.remove) { + for (const id of diff.remove) { + updateable.delete(id); + } + } + if (diff.add) { + for (const feature of diff.add) { + const id = getFeatureId(feature, promoteId); + if (id != null) { + updateable.set(id, feature); + } + } + } + if (diff.update) { + for (const update of diff.update) { + let feature = updateable.get(update.id); + if (feature == null) { + continue; + } + // be careful to clone the feature and/or properties objects to avoid mutating our input + const cloneFeature = update.newGeometry || update.removeAllProperties; + // note: removeAllProperties gives us a new properties object, so we can skip the clone step + const cloneProperties = !update.removeAllProperties && (((_a = update.removeProperties) === null || _a === void 0 ? void 0 : _a.length) > 0 || ((_b = update.addOrUpdateProperties) === null || _b === void 0 ? void 0 : _b.length) > 0); + if (cloneFeature || cloneProperties) { + feature = Object.assign({}, feature); + updateable.set(update.id, feature); + if (cloneProperties) { + feature.properties = Object.assign({}, feature.properties); + } + } + if (update.newGeometry) { + feature.geometry = update.newGeometry; + } + if (update.removeAllProperties) { + feature.properties = {}; + } + else if (((_c = update.removeProperties) === null || _c === void 0 ? void 0 : _c.length) > 0) { + for (const prop of update.removeProperties) { + if (Object.prototype.hasOwnProperty.call(feature.properties, prop)) { + delete feature.properties[prop]; + } + } + } + if (((_d = update.addOrUpdateProperties) === null || _d === void 0 ? void 0 : _d.length) > 0) { + for (const { key, value } of update.addOrUpdateProperties) { + feature.properties[key] = value; + } + } + } + } +} + +/** + * The {@link WorkerSource} implementation that supports {@link GeoJSONSource}. + * This class is designed to be easily reused to support custom source types + * for data formats that can be parsed/converted into an in-memory GeoJSON + * representation. To do so, create it with + * `new GeoJSONWorkerSource(actor, layerIndex, customLoadGeoJSONFunction)`. + * For a full example, see [mapbox-gl-topojson](https://github.com/developmentseed/mapbox-gl-topojson). + */ +class GeoJSONWorkerSource extends VectorTileWorkerSource { + /** + * @param loadGeoJSON - Optional method for custom loading/parsing of + * GeoJSON based on parameters passed from the main-thread Source. + * See {@link GeoJSONWorkerSource#loadGeoJSON}. + */ + constructor(actor, layerIndex, availableImages, loadGeoJSON) { + super(actor, layerIndex, availableImages); + this._dataUpdateable = new Map(); + /** + * Fetch and parse GeoJSON according to the given params. Calls `callback` + * with `(err, data)`, where `data` is a parsed GeoJSON object. + * + * GeoJSON is loaded and parsed from `params.url` if it exists, or else + * expected as a literal (string or object) `params.data`. + * + * @param params - the parameters + * @param callback - the callback for completion or error + * @returns A Cancelable object. + */ + this.loadGeoJSON = (params, callback) => { + const { promoteId } = params; + // Because of same origin issues, urls must either include an explicit + // origin or absolute path. + // ie: /foo/bar.json or http://example.com/bar.json + // but not ../foo/bar.json + if (params.request) { + return performance.getJSON(params.request, (error, data, cacheControl, expires) => { + this._dataUpdateable = isUpdateableGeoJSON(data, promoteId) ? toUpdateable(data, promoteId) : undefined; + callback(error, data, cacheControl, expires); + }); + } + else if (typeof params.data === 'string') { + try { + const parsed = JSON.parse(params.data); + this._dataUpdateable = isUpdateableGeoJSON(parsed, promoteId) ? toUpdateable(parsed, promoteId) : undefined; + callback(null, parsed); + } + catch (e) { + callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`)); + } + } + else if (params.dataDiff) { + if (this._dataUpdateable) { + applySourceDiff(this._dataUpdateable, params.dataDiff, promoteId); + callback(null, { type: 'FeatureCollection', features: Array.from(this._dataUpdateable.values()) }); + } + else { + callback(new Error(`Cannot update existing geojson data in ${params.source}`)); + } + } + else { + callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`)); + } + return { cancel: () => { } }; + }; + this.loadVectorData = this.loadGeoJSONTile; + if (loadGeoJSON) { + this.loadGeoJSON = loadGeoJSON; + } + } + loadGeoJSONTile(params, callback) { + const canonical = params.tileID.canonical; + if (!this._geoJSONIndex) { + return callback(null, null); // we couldn't load the file + } + const geoJSONTile = this._geoJSONIndex.getTile(canonical.z, canonical.x, canonical.y); + if (!geoJSONTile) { + return callback(null, null); // nothing in the given tile + } + const geojsonWrapper = new GeoJSONWrapper$2(geoJSONTile.features); + // Encode the geojson-vt tile into binary vector tile form. This + // is a convenience that allows `FeatureIndex` to operate the same way + // across `VectorTileSource` and `GeoJSONSource` data. + let pbf = vtpbf(geojsonWrapper); + if (pbf.byteOffset !== 0 || pbf.byteLength !== pbf.buffer.byteLength) { + // Compatibility with node Buffer (https://github.com/mapbox/pbf/issues/35) + pbf = new Uint8Array(pbf); + } + callback(null, { + vectorTile: geojsonWrapper, + rawData: pbf.buffer + }); + } + /** + * Fetches (if appropriate), parses, and index geojson data into tiles. This + * preparatory method must be called before {@link GeoJSONWorkerSource#loadTile} + * can correctly serve up tiles. + * + * Defers to {@link GeoJSONWorkerSource#loadGeoJSON} for the fetching/parsing, + * expecting `callback(error, data)` to be called with either an error or a + * parsed GeoJSON object. + * + * When a `loadData` request comes in while a previous one is being processed, + * the previous one is aborted. + * + * @param params - the parameters + * @param callback - the callback for completion or error + */ + loadData(params, callback) { + var _a; + (_a = this._pendingRequest) === null || _a === void 0 ? void 0 : _a.cancel(); + if (this._pendingCallback) { + // Tell the foreground the previous call has been abandoned + this._pendingCallback(null, { abandoned: true }); + } + const perf = (params && params.request && params.request.collectResourceTiming) ? + new performance.RequestPerformance(params.request) : false; + this._pendingCallback = callback; + this._pendingRequest = this.loadGeoJSON(params, (err, data) => { + delete this._pendingCallback; + delete this._pendingRequest; + if (err || !data) { + return callback(err); + } + else if (typeof data !== 'object') { + return callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`)); + } + else { + rewind$2(data, true); + try { + if (params.filter) { + const compiled = performance.createExpression(params.filter, { type: 'boolean', 'property-type': 'data-driven', overridable: false, transition: false }); + if (compiled.result === 'error') + throw new Error(compiled.value.map(err => `${err.key}: ${err.message}`).join(', ')); + const features = data.features.filter(feature => compiled.value.evaluate({ zoom: 0 }, feature)); + data = { type: 'FeatureCollection', features }; + } + this._geoJSONIndex = params.cluster ? + new Supercluster(getSuperclusterOptions(params)).load(data.features) : + geojsonvt(data, params.geojsonVtOptions); + } + catch (err) { + return callback(err); + } + this.loaded = {}; + const result = {}; + if (perf) { + const resourceTimingData = perf.finish(); + // it's necessary to eval the result of getEntriesByName() here via parse/stringify + // late evaluation in the main thread causes TypeError: illegal invocation + if (resourceTimingData) { + result.resourceTiming = {}; + result.resourceTiming[params.source] = JSON.parse(JSON.stringify(resourceTimingData)); + } + } + callback(null, result); + } + }); + } + /** + * Implements {@link WorkerSource#reloadTile}. + * + * If the tile is loaded, uses the implementation in VectorTileWorkerSource. + * Otherwise, such as after a setData() call, we load the tile fresh. + * + * @param params - the parameters + * @param callback - the callback for completion or error + */ + reloadTile(params, callback) { + const loaded = this.loaded, uid = params.uid; + if (loaded && loaded[uid]) { + return super.reloadTile(params, callback); + } + else { + return this.loadTile(params, callback); + } + } + removeSource(params, callback) { + if (this._pendingCallback) { + // Don't leak callbacks + this._pendingCallback(null, { abandoned: true }); + } + callback(); + } + getClusterExpansionZoom(params, callback) { + try { + callback(null, this._geoJSONIndex.getClusterExpansionZoom(params.clusterId)); + } + catch (e) { + callback(e); + } + } + getClusterChildren(params, callback) { + try { + callback(null, this._geoJSONIndex.getChildren(params.clusterId)); + } + catch (e) { + callback(e); + } + } + getClusterLeaves(params, callback) { + try { + callback(null, this._geoJSONIndex.getLeaves(params.clusterId, params.limit, params.offset)); + } + catch (e) { + callback(e); + } + } +} +function getSuperclusterOptions({ superclusterOptions, clusterProperties }) { + if (!clusterProperties || !superclusterOptions) + return superclusterOptions; + const mapExpressions = {}; + const reduceExpressions = {}; + const globals = { accumulated: null, zoom: 0 }; + const feature = { properties: null }; + const propertyNames = Object.keys(clusterProperties); + for (const key of propertyNames) { + const [operator, mapExpression] = clusterProperties[key]; + const mapExpressionParsed = performance.createExpression(mapExpression); + const reduceExpressionParsed = performance.createExpression(typeof operator === 'string' ? [operator, ['accumulated'], ['get', key]] : operator); + mapExpressions[key] = mapExpressionParsed.value; + reduceExpressions[key] = reduceExpressionParsed.value; + } + superclusterOptions.map = (pointProperties) => { + feature.properties = pointProperties; + const properties = {}; + for (const key of propertyNames) { + properties[key] = mapExpressions[key].evaluate(globals, feature); + } + return properties; + }; + superclusterOptions.reduce = (accumulated, clusterProperties) => { + feature.properties = clusterProperties; + for (const key of propertyNames) { + globals.accumulated = accumulated[key]; + accumulated[key] = reduceExpressions[key].evaluate(globals, feature); + } + }; + return superclusterOptions; +} + +/** + * The Worker class responsidble for background thread related execution + */ +class Worker { + constructor(self) { + this.self = self; + this.actor = new performance.Actor(self, this); + this.layerIndexes = {}; + this.availableImages = {}; + this.workerSourceTypes = { + vector: VectorTileWorkerSource, + geojson: GeoJSONWorkerSource + }; + // [mapId][sourceType][sourceName] => worker source instance + this.workerSources = {}; + this.demWorkerSources = {}; + this.self.registerWorkerSource = (name, WorkerSource) => { + if (this.workerSourceTypes[name]) { + throw new Error(`Worker source with name "${name}" already registered.`); + } + this.workerSourceTypes[name] = WorkerSource; + }; + // This is invoked by the RTL text plugin when the download via the `importScripts` call has finished, and the code has been parsed. + this.self.registerRTLTextPlugin = (rtlTextPlugin) => { + if (performance.plugin.isParsed()) { + throw new Error('RTL text plugin already registered.'); + } + performance.plugin['applyArabicShaping'] = rtlTextPlugin.applyArabicShaping; + performance.plugin['processBidirectionalText'] = rtlTextPlugin.processBidirectionalText; + performance.plugin['processStyledBidirectionalText'] = rtlTextPlugin.processStyledBidirectionalText; + }; + } + setReferrer(mapID, referrer) { + this.referrer = referrer; + } + setImages(mapId, images, callback) { + this.availableImages[mapId] = images; + for (const workerSource in this.workerSources[mapId]) { + const ws = this.workerSources[mapId][workerSource]; + for (const source in ws) { + ws[source].availableImages = images; + } + } + callback(); + } + setLayers(mapId, layers, callback) { + this.getLayerIndex(mapId).replace(layers); + callback(); + } + updateLayers(mapId, params, callback) { + this.getLayerIndex(mapId).update(params.layers, params.removedIds); + callback(); + } + loadTile(mapId, params, callback) { + this.getWorkerSource(mapId, params.type, params.source).loadTile(params, callback); + } + loadDEMTile(mapId, params, callback) { + this.getDEMWorkerSource(mapId, params.source).loadTile(params, callback); + } + reloadTile(mapId, params, callback) { + this.getWorkerSource(mapId, params.type, params.source).reloadTile(params, callback); + } + abortTile(mapId, params, callback) { + this.getWorkerSource(mapId, params.type, params.source).abortTile(params, callback); + } + removeTile(mapId, params, callback) { + this.getWorkerSource(mapId, params.type, params.source).removeTile(params, callback); + } + removeDEMTile(mapId, params) { + this.getDEMWorkerSource(mapId, params.source).removeTile(params); + } + removeSource(mapId, params, callback) { + if (!this.workerSources[mapId] || + !this.workerSources[mapId][params.type] || + !this.workerSources[mapId][params.type][params.source]) { + return; + } + const worker = this.workerSources[mapId][params.type][params.source]; + delete this.workerSources[mapId][params.type][params.source]; + if (worker.removeSource !== undefined) { + worker.removeSource(params, callback); + } + else { + callback(); + } + } + /** + * Load a {@link WorkerSource} script at params.url. The script is run + * (using importScripts) with `registerWorkerSource` in scope, which is a + * function taking `(name, workerSourceObject)`. + */ + loadWorkerSource(map, params, callback) { + try { + this.self.importScripts(params.url); + callback(); + } + catch (e) { + callback(e.toString()); + } + } + syncRTLPluginState(map, state, callback) { + try { + performance.plugin.setState(state); + const pluginURL = performance.plugin.getPluginURL(); + if (performance.plugin.isLoaded() && + !performance.plugin.isParsed() && + pluginURL != null // Not possible when `isLoaded` is true, but keeps flow happy + ) { + this.self.importScripts(pluginURL); + const complete = performance.plugin.isParsed(); + const error = complete ? undefined : new Error(`RTL Text Plugin failed to import scripts from ${pluginURL}`); + callback(error, complete); + } + } + catch (e) { + callback(e.toString()); + } + } + getAvailableImages(mapId) { + let availableImages = this.availableImages[mapId]; + if (!availableImages) { + availableImages = []; + } + return availableImages; + } + getLayerIndex(mapId) { + let layerIndexes = this.layerIndexes[mapId]; + if (!layerIndexes) { + layerIndexes = this.layerIndexes[mapId] = new StyleLayerIndex(); + } + return layerIndexes; + } + getWorkerSource(mapId, sourceType, sourceName) { + if (!this.workerSources[mapId]) + this.workerSources[mapId] = {}; + if (!this.workerSources[mapId][sourceType]) + this.workerSources[mapId][sourceType] = {}; + if (!this.workerSources[mapId][sourceType][sourceName]) { + // use a wrapped actor so that we can attach a target mapId param + // to any messages invoked by the WorkerSource + const actor = { + send: (type, data, callback) => { + this.actor.send(type, data, callback, mapId); + } + }; + this.workerSources[mapId][sourceType][sourceName] = new this.workerSourceTypes[sourceType](actor, this.getLayerIndex(mapId), this.getAvailableImages(mapId)); + } + return this.workerSources[mapId][sourceType][sourceName]; + } + getDEMWorkerSource(mapId, source) { + if (!this.demWorkerSources[mapId]) + this.demWorkerSources[mapId] = {}; + if (!this.demWorkerSources[mapId][source]) { + this.demWorkerSources[mapId][source] = new RasterDEMTileWorkerSource(); + } + return this.demWorkerSources[mapId][source]; + } +} +if (performance.isWorker()) { + self.worker = new Worker(self); +} + +return Worker; + +})); + +define(['./shared'], (function (performance) { 'use strict'; + +var name = "maplibre-gl"; +var description = "BSD licensed community fork of mapbox-gl, a WebGL interactive maps library"; +var version$2 = "3.6.2"; +var main = "dist/maplibre-gl.js"; +var style = "dist/maplibre-gl.css"; +var license = "BSD-3-Clause"; +var funding = "https://github.com/maplibre/maplibre-gl-js?sponsor=1"; +var repository = { + type: "git", + url: "git://github.com/maplibre/maplibre-gl-js.git" +}; +var types = "dist/maplibre-gl.d.ts"; +var type = "module"; +var dependencies = { + "@mapbox/geojson-rewind": "^0.5.2", + "@mapbox/jsonlint-lines-primitives": "^2.0.2", + "@mapbox/point-geometry": "^0.1.0", + "@mapbox/tiny-sdf": "^2.0.6", + "@mapbox/unitbezier": "^0.0.1", + "@mapbox/vector-tile": "^1.3.1", + "@mapbox/whoots-js": "^3.1.0", + "@maplibre/maplibre-gl-style-spec": "^19.3.3", + "@types/geojson": "^7946.0.13", + "@types/mapbox__point-geometry": "^0.1.4", + "@types/mapbox__vector-tile": "^1.3.4", + "@types/pbf": "^3.0.5", + "@types/supercluster": "^7.1.3", + earcut: "^2.2.4", + "geojson-vt": "^3.2.1", + "gl-matrix": "^3.4.3", + "global-prefix": "^3.0.0", + kdbush: "^4.0.2", + "murmurhash-js": "^1.0.0", + pbf: "^3.2.1", + potpack: "^2.0.0", + quickselect: "^2.0.0", + supercluster: "^8.0.1", + tinyqueue: "^2.0.3", + "vt-pbf": "^3.1.3" +}; +var devDependencies = { + "@mapbox/mapbox-gl-rtl-text": "^0.2.3", + "@mapbox/mvt-fixtures": "^3.10.0", + "@rollup/plugin-commonjs": "^25.0.7", + "@rollup/plugin-json": "^6.0.1", + "@rollup/plugin-node-resolve": "^15.2.3", + "@rollup/plugin-replace": "^5.0.5", + "@rollup/plugin-strip": "^3.0.4", + "@rollup/plugin-terser": "^0.4.4", + "@rollup/plugin-typescript": "^11.1.5", + "@types/benchmark": "^2.1.5", + "@types/cssnano": "^5.0.0", + "@types/d3": "^7.4.3", + "@types/diff": "^5.0.8", + "@types/earcut": "^2.1.4", + "@types/eslint": "^8.44.7", + "@types/geojson-vt": "3.2.4", + "@types/gl": "^6.0.5", + "@types/glob": "^8.1.0", + "@types/jest": "^29.5.8", + "@types/jsdom": "^21.1.5", + "@types/minimist": "^1.2.5", + "@types/murmurhash-js": "^1.0.6", + "@types/nise": "^1.4.4", + "@types/node": "^20.9.2", + "@types/offscreencanvas": "^2019.7.3", + "@types/pixelmatch": "^5.2.6", + "@types/pngjs": "^6.0.4", + "@types/react": "^18.2.37", + "@types/react-dom": "^18.2.15", + "@types/request": "^2.48.12", + "@types/shuffle-seed": "^1.1.2", + "@types/window-or-global": "^1.0.6", + "@typescript-eslint/eslint-plugin": "^6.11.0", + "@typescript-eslint/parser": "^6.11.0", + address: "^2.0.1", + benchmark: "^2.1.4", + canvas: "^2.11.2", + cssnano: "^6.0.1", + d3: "^7.8.5", + "d3-queue": "^3.0.7", + "devtools-protocol": "^0.0.1226504", + diff: "^5.1.0", + "dts-bundle-generator": "^8.1.2", + eslint: "^8.54.0", + "eslint-config-mourner": "^3.0.0", + "eslint-plugin-html": "^7.1.0", + "eslint-plugin-import": "^2.29.0", + "eslint-plugin-jest": "^27.6.0", + "eslint-plugin-react": "^7.33.2", + "eslint-plugin-tsdoc": "0.2.17", + expect: "^29.7.0", + gl: "^6.0.2", + glob: "^10.3.10", + "is-builtin-module": "^3.2.1", + jest: "^29.7.0", + "jest-canvas-mock": "^2.5.2", + "jest-environment-jsdom": "^29.7.0", + jsdom: "^22.1.0", + "json-stringify-pretty-compact": "^4.0.0", + minimist: "^1.2.8", + "mock-geolocation": "^1.0.11", + nise: "^5.1.5", + "npm-font-open-sans": "^1.1.0", + "npm-run-all": "^4.1.5", + "pdf-merger-js": "^4.3.0", + pixelmatch: "^5.3.0", + pngjs: "^7.0.0", + postcss: "^8.4.31", + "postcss-cli": "^10.1.0", + "postcss-inline-svg": "^6.0.0", + "pretty-bytes": "^6.1.1", + puppeteer: "^21.5.2", + react: "^18.2.0", + "react-dom": "^18.2.0", + rollup: "^4.5.0", + "rollup-plugin-sourcemaps": "^0.6.3", + rw: "^1.3.3", + semver: "^7.5.4", + "shuffle-seed": "^1.1.6", + "source-map-explorer": "^2.5.3", + st: "^3.0.0", + stylelint: "^15.11.0", + "stylelint-config-standard": "^34.0.0", + "ts-jest": "^29.1.1", + "ts-node": "^10.9.1", + tslib: "^2.6.2", + typedoc: "^0.25.3", + "typedoc-plugin-markdown": "^3.17.1", + "typedoc-plugin-missing-exports": "^2.1.0", + typescript: "^5.2.2" +}; +var overrides = { + "postcss-inline-svg": { + "css-select": "^5.1.0", + "dom-serializer": "^2.0.0", + htmlparser2: "^8.0.1", + "postcss-value-parser": "^4.2.0" + } +}; +var scripts = { + "generate-dist-package": "npm run tsnode build/generate-dist-package.js", + "generate-shaders": "npm run tsnode build/generate-shaders.ts", + "generate-struct-arrays": "npm run tsnode build/generate-struct-arrays.ts", + "generate-style-code": "npm run tsnode build/generate-style-code.ts", + "generate-typings": "npm run tsnode build/generate-typings.ts", + "generate-docs": "typedoc && npm run tsnode build/generate-docs.ts", + "generate-images": "npm run tsnode build/generate-doc-images.ts", + "build-dist": "run-p --print-label generate-typings build-dev build-prod build-csp build-csp-dev build-css", + "build-dev": "rollup --configPlugin @rollup/plugin-typescript -c --environment BUILD:dev", + "watch-dev": "rollup --configPlugin @rollup/plugin-typescript -c --environment BUILD:dev --watch", + "build-prod": "rollup --configPlugin @rollup/plugin-typescript -c --environment BUILD:production", + "build-csp": "rollup --configPlugin @rollup/plugin-typescript -c rollup.config.csp.ts", + "build-csp-dev": "rollup --configPlugin @rollup/plugin-typescript -c rollup.config.csp.ts --environment BUILD:dev", + "build-css": "postcss -o dist/maplibre-gl.css src/css/maplibre-gl.css", + "watch-css": "postcss --watch -o dist/maplibre-gl.css src/css/maplibre-gl.css", + "build-benchmarks": "npm run build-dev && rollup --configPlugin @rollup/plugin-typescript -c test/bench/rollup_config_benchmarks.ts", + "watch-benchmarks": "rollup --configPlugin @rollup/plugin-typescript -c test/bench/rollup_config_benchmarks.ts --watch", + "start-server": "st --no-cache -H 0.0.0.0 --port 9966 .", + "start-docs": "docker run --rm -it -p 8000:8000 -v ${PWD}:/docs squidfunk/mkdocs-material", + start: "run-p watch-css watch-dev start-server", + "start-bench": "run-p watch-css watch-benchmarks start-server", + lint: "eslint --cache --ext .ts,.tsx,.js,.html --ignore-path .gitignore .", + "lint-css": "stylelint src/css/maplibre-gl.css", + test: "run-p lint lint-css test-render jest", + jest: "jest", + "jest-ci": "jest --reporters=github-actions --reporters=summary", + "test-build": "jest --selectProjects=build", + "test-integration": "jest --selectProjects=integration", + "test-render": "npm run tsnode test/integration/render/run_render_tests.ts", + "test-unit": "jest --selectProjects=unit", + "test-watch-roots": "jest --watch", + codegen: "run-p generate-dist-package generate-style-code generate-struct-arrays generate-shaders", + benchmark: "npm run tsnode test/bench/run-benchmarks.ts", + "gl-stats": "npm run tsnode test/bench/gl-stats.ts", + prepare: "npm run codegen", + typecheck: "tsc --noEmit && tsc --project tsconfig.dist.json", + tsnode: "node --experimental-loader=ts-node/esm --no-warnings" +}; +var files = [ + "build/", + "dist/*", + "src/" +]; +var engines = { + npm: ">=8.1.0", + node: ">=16.14.0" +}; +var packageJSON = { + name: name, + description: description, + version: version$2, + main: main, + style: style, + license: license, + funding: funding, + repository: repository, + types: types, + type: type, + dependencies: dependencies, + devDependencies: devDependencies, + overrides: overrides, + scripts: scripts, + files: files, + engines: engines +}; + +class DOM { + static testProp(props) { + if (!DOM.docStyle) + return props[0]; + for (let i = 0; i < props.length; i++) { + if (props[i] in DOM.docStyle) { + return props[i]; + } + } + return props[0]; + } + static create(tagName, className, container) { + const el = window.document.createElement(tagName); + if (className !== undefined) + el.className = className; + if (container) + container.appendChild(el); + return el; + } + static createNS(namespaceURI, tagName) { + const el = window.document.createElementNS(namespaceURI, tagName); + return el; + } + static disableDrag() { + if (DOM.docStyle && DOM.selectProp) { + DOM.userSelect = DOM.docStyle[DOM.selectProp]; + DOM.docStyle[DOM.selectProp] = 'none'; + } + } + static enableDrag() { + if (DOM.docStyle && DOM.selectProp) { + DOM.docStyle[DOM.selectProp] = DOM.userSelect; + } + } + static setTransform(el, value) { + el.style[DOM.transformProp] = value; + } + static addEventListener(target, type, callback, options = {}) { + if ('passive' in options) { + target.addEventListener(type, callback, options); + } + else { + target.addEventListener(type, callback, options.capture); + } + } + static removeEventListener(target, type, callback, options = {}) { + if ('passive' in options) { + target.removeEventListener(type, callback, options); + } + else { + target.removeEventListener(type, callback, options.capture); + } + } + // Suppress the next click, but only if it's immediate. + static suppressClickInternal(e) { + e.preventDefault(); + e.stopPropagation(); + window.removeEventListener('click', DOM.suppressClickInternal, true); + } + static suppressClick() { + window.addEventListener('click', DOM.suppressClickInternal, true); + window.setTimeout(() => { + window.removeEventListener('click', DOM.suppressClickInternal, true); + }, 0); + } + static mousePos(el, e) { + const rect = el.getBoundingClientRect(); + return new performance.Point(e.clientX - rect.left - el.clientLeft, e.clientY - rect.top - el.clientTop); + } + static touchPos(el, touches) { + const rect = el.getBoundingClientRect(); + const points = []; + for (let i = 0; i < touches.length; i++) { + points.push(new performance.Point(touches[i].clientX - rect.left - el.clientLeft, touches[i].clientY - rect.top - el.clientTop)); + } + return points; + } + static mouseButton(e) { + return e.button; + } + static remove(node) { + if (node.parentNode) { + node.parentNode.removeChild(node); + } + } +} +DOM.docStyle = typeof window !== 'undefined' && window.document && window.document.documentElement.style; +DOM.selectProp = DOM.testProp(['userSelect', 'MozUserSelect', 'WebkitUserSelect', 'msUserSelect']); +DOM.transformProp = DOM.testProp(['transform', 'WebkitTransform']); + +const webpSupported = { + supported: false, + testSupport +}; +let glForTesting; +let webpCheckComplete = false; +let webpImgTest; +let webpImgTestOnloadComplete = false; +if (typeof document !== 'undefined') { + webpImgTest = document.createElement('img'); + webpImgTest.onload = function () { + if (glForTesting) + testWebpTextureUpload(glForTesting); + glForTesting = null; + webpImgTestOnloadComplete = true; + }; + webpImgTest.onerror = function () { + webpCheckComplete = true; + glForTesting = null; + }; + webpImgTest.src = ''; +} +function testSupport(gl) { + if (webpCheckComplete || !webpImgTest) + return; + // HTMLImageElement.complete is set when an image is done loading it's source + // regardless of whether the load was successful or not. + // It's possible for an error to set HTMLImageElement.complete to true which would trigger + // testWebpTextureUpload and mistakenly set exported.supported to true in browsers which don't support webp + // To avoid this, we set a flag in the image's onload handler and only call testWebpTextureUpload + // after a successful image load event. + if (webpImgTestOnloadComplete) { + testWebpTextureUpload(gl); + } + else { + glForTesting = gl; + } +} +function testWebpTextureUpload(gl) { + // Edge 18 supports WebP but not uploading a WebP image to a gl texture + // Test support for this before allowing WebP images. + // https://github.com/mapbox/mapbox-gl-js/issues/7671 + const texture = gl.createTexture(); + gl.bindTexture(gl.TEXTURE_2D, texture); + try { + gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, webpImgTest); + // The error does not get triggered in Edge if the context is lost + if (gl.isContextLost()) + return; + webpSupported.supported = true; + } + catch (e) { + // Catch "Unspecified Error." in Edge 18. + } + gl.deleteTexture(texture); + webpCheckComplete = true; +} + +/** + * By default, the image queue is self driven, meaning as soon as one requested item is processed, + * it will move on to next one as quickly as it can while limiting + * the number of concurrent requests to MAX_PARALLEL_IMAGE_REQUESTS. The default behavior + * ensures that static views of the map can be rendered with minimal delay. + * + * However, the default behavior can prevent dynamic views of the map from rendering + * smoothly in that many requests can finish in one render frame, putting too much pressure on GPU. + * + * When the view of the map is moving dynamically, smoother frame rates can be achieved + * by throttling the number of items processed by the queue per frame. This can be + * accomplished by using {@link addThrottleControl} to allow the caller to + * use a lambda function to determine when the queue should be throttled (e.g. when isMoving()) + * and manually calling {@link processQueue} in the render loop. + */ +var ImageRequest; +(function (ImageRequest) { + let imageRequestQueue; + let currentParallelImageRequests; + let throttleControlCallbackHandleCounter; + let throttleControlCallbacks; + /** + * Reset the image request queue, removing all pending requests. + */ + ImageRequest.resetRequestQueue = () => { + imageRequestQueue = []; + currentParallelImageRequests = 0; + throttleControlCallbackHandleCounter = 0; + throttleControlCallbacks = {}; + }; + /** + * Install a callback to control when image queue throttling is desired. + * (e.g. when the map view is moving) + * @param callback - The callback function to install + * @returns handle that identifies the installed callback. + */ + ImageRequest.addThrottleControl = (callback) => { + const handle = throttleControlCallbackHandleCounter++; + throttleControlCallbacks[handle] = callback; + return handle; + }; + /** + * Remove a previously installed callback by passing in the handle returned + * by {@link addThrottleControl}. + * @param callbackHandle - The handle for the callback to remove. + */ + ImageRequest.removeThrottleControl = (callbackHandle) => { + delete throttleControlCallbacks[callbackHandle]; + // Try updating the queue + processQueue(); + }; + /** + * Check to see if any of the installed callbacks are requesting the queue + * to be throttled. + * @returns `true` if any callback is causing the queue to be throttled. + */ + const isThrottled = () => { + const allControlKeys = Object.keys(throttleControlCallbacks); + let throttleingRequested = false; + if (allControlKeys.length > 0) { + for (const key of allControlKeys) { + throttleingRequested = throttleControlCallbacks[key](); + if (throttleingRequested) { + break; + } + } + } + return throttleingRequested; + }; + /** + * Request to load an image. + * @param requestParameters - Request parameters. + * @param callback - Callback to issue when the request completes. + * @param supportImageRefresh - `true`, if the image request need to support refresh based on cache headers. + * @returns Cancelable request. + */ + ImageRequest.getImage = (requestParameters, callback, supportImageRefresh = true) => { + if (webpSupported.supported) { + if (!requestParameters.headers) { + requestParameters.headers = {}; + } + requestParameters.headers.accept = 'image/webp,*/*'; + } + const request = { + requestParameters, + supportImageRefresh, + callback, + cancelled: false, + completed: false, + cancel: () => { + if (!request.completed && !request.cancelled) { + request.cancelled = true; + // Only reduce currentParallelImageRequests, if the image request was issued. + if (request.innerRequest) { + request.innerRequest.cancel(); + currentParallelImageRequests--; + } + // in the case of cancelling, it WILL move on + processQueue(); + } + } + }; + imageRequestQueue.push(request); + processQueue(); + return request; + }; + const arrayBufferToCanvasImageSource = (data, callback) => { + const imageBitmapSupported = typeof createImageBitmap === 'function'; + if (imageBitmapSupported) { + performance.arrayBufferToImageBitmap(data, callback); + } + else { + performance.arrayBufferToImage(data, callback); + } + }; + const doImageRequest = (itemInQueue) => { + const { requestParameters, supportImageRefresh, callback } = itemInQueue; + performance.extend(requestParameters, { type: 'image' }); + // - If refreshExpiredTiles is false, then we can use HTMLImageElement to download raster images. + // - Fetch/XHR (via MakeRequest API) will be used to download images for following scenarios: + // 1. Style image sprite will had a issue with HTMLImageElement as described + // here: https://github.com/mapbox/mapbox-gl-js/issues/1470 + // 2. If refreshExpiredTiles is true (default), then in order to read the image cache header, + // fetch/XHR request will be required + // - For any special case handling like use of AddProtocol, worker initiated request or additional headers + // let makeRequest handle it. + // - HtmlImageElement request automatically adds accept header for all the browser supported images + const canUseHTMLImageElement = supportImageRefresh === false && + !performance.isWorker() && + !performance.getProtocolAction(requestParameters.url) && + (!requestParameters.headers || + Object.keys(requestParameters.headers).reduce((acc, item) => acc && item === 'accept', true)); + const action = canUseHTMLImageElement ? getImageUsingHtmlImage : performance.makeRequest; + return action(requestParameters, (err, data, cacheControl, expires) => { + onImageResponse(itemInQueue, callback, err, data, cacheControl, expires); + }); + }; + const onImageResponse = (itemInQueue, callback, err, data, cacheControl, expires) => { + if (err) { + callback(err); + } + else if (data instanceof HTMLImageElement || performance.isImageBitmap(data)) { + // User using addProtocol can directly return HTMLImageElement/ImageBitmap type + // If HtmlImageElement is used to get image then response type will be HTMLImageElement + callback(null, data); + } + else if (data) { + const decoratedCallback = (imgErr, imgResult) => { + if (imgErr != null) { + callback(imgErr); + } + else if (imgResult != null) { + callback(null, imgResult, { cacheControl, expires }); + } + }; + arrayBufferToCanvasImageSource(data, decoratedCallback); + } + if (!itemInQueue.cancelled) { + itemInQueue.completed = true; + currentParallelImageRequests--; + processQueue(); + } + }; + /** + * Process some number of items in the image request queue. + */ + const processQueue = () => { + const maxImageRequests = isThrottled() ? + performance.config.MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME : + performance.config.MAX_PARALLEL_IMAGE_REQUESTS; + // limit concurrent image loads to help with raster sources performance on big screens + for (let numImageRequests = currentParallelImageRequests; numImageRequests < maxImageRequests && imageRequestQueue.length > 0; numImageRequests++) { + const topItemInQueue = imageRequestQueue.shift(); + if (topItemInQueue.cancelled) { + numImageRequests--; + continue; + } + const innerRequest = doImageRequest(topItemInQueue); + currentParallelImageRequests++; + topItemInQueue.innerRequest = innerRequest; + } + }; + const getImageUsingHtmlImage = (requestParameters, callback) => { + const image = new Image(); + const url = requestParameters.url; + let requestCancelled = false; + const credentials = requestParameters.credentials; + if (credentials && credentials === 'include') { + image.crossOrigin = 'use-credentials'; + } + else if ((credentials && credentials === 'same-origin') || !performance.sameOrigin(url)) { + image.crossOrigin = 'anonymous'; + } + image.fetchPriority = 'high'; + image.onload = () => { + callback(null, image); + image.onerror = image.onload = null; + }; + image.onerror = () => { + if (!requestCancelled) { + callback(new Error('Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.')); + } + image.onerror = image.onload = null; + }; + image.src = url; + return { + cancel: () => { + requestCancelled = true; + // Set src to '' to actually cancel the request + image.src = ''; + } + }; + }; +})(ImageRequest || (ImageRequest = {})); +ImageRequest.resetRequestQueue(); + +/** + * A type of MapLibre resource. + */ +var ResourceType; +(function (ResourceType) { + ResourceType["Glyphs"] = "Glyphs"; + ResourceType["Image"] = "Image"; + ResourceType["Source"] = "Source"; + ResourceType["SpriteImage"] = "SpriteImage"; + ResourceType["SpriteJSON"] = "SpriteJSON"; + ResourceType["Style"] = "Style"; + ResourceType["Tile"] = "Tile"; + ResourceType["Unknown"] = "Unknown"; +})(ResourceType || (ResourceType = {})); +class RequestManager { + constructor(transformRequestFn) { + this._transformRequestFn = transformRequestFn; + } + transformRequest(url, type) { + if (this._transformRequestFn) { + return this._transformRequestFn(url, type) || { url }; + } + return { url }; + } + normalizeSpriteURL(url, format, extension) { + const urlObject = parseUrl(url); + urlObject.path += `${format}${extension}`; + return formatUrl(urlObject); + } + setTransformRequest(transformRequest) { + this._transformRequestFn = transformRequest; + } +} +const urlRe = /^(\w+):\/\/([^/?]*)(\/[^?]+)?\??(.+)?/; +function parseUrl(url) { + const parts = url.match(urlRe); + if (!parts) { + throw new Error(`Unable to parse URL "${url}"`); + } + return { + protocol: parts[1], + authority: parts[2], + path: parts[3] || '/', + params: parts[4] ? parts[4].split('&') : [] + }; +} +function formatUrl(obj) { + const params = obj.params.length ? `?${obj.params.join('&')}` : ''; + return `${obj.protocol}://${obj.authority}${obj.path}${params}`; +} + +/** + * Takes a SpriteSpecification value and returns it in its array form. If `undefined` is passed as an input value, an + * empty array is returned. + * duplicated entries with identical id/url will be removed in returned array + * @param sprite - optional sprite to coerce + * @returns an empty array in case `undefined` is passed; id-url pairs otherwise + */ +function coerceSpriteToArray(sprite) { + const resultArray = []; + if (typeof sprite === 'string') { + resultArray.push({ id: 'default', url: sprite }); + } + else if (sprite && sprite.length > 0) { + const dedupArray = []; + for (const { id, url } of sprite) { + const key = `${id}${url}`; + if (dedupArray.indexOf(key) === -1) { + dedupArray.push(key); + resultArray.push({ id, url }); + } + } + } + return resultArray; +} + +function loadSprite(originalSprite, requestManager, pixelRatio, callback) { + const spriteArray = coerceSpriteToArray(originalSprite); + const spriteArrayLength = spriteArray.length; + const format = pixelRatio > 1 ? '@2x' : ''; + const combinedRequestsMap = {}; + const jsonsMap = {}; + const imagesMap = {}; + for (const { id, url } of spriteArray) { + const jsonRequestParameters = requestManager.transformRequest(requestManager.normalizeSpriteURL(url, format, '.json'), ResourceType.SpriteJSON); + const jsonRequestKey = `${id}_${jsonRequestParameters.url}`; // use id_url as requestMap key to make sure it is unique + combinedRequestsMap[jsonRequestKey] = performance.getJSON(jsonRequestParameters, (err, data) => { + delete combinedRequestsMap[jsonRequestKey]; + jsonsMap[id] = data; + doOnceCompleted(callback, jsonsMap, imagesMap, err, spriteArrayLength); + }); + const imageRequestParameters = requestManager.transformRequest(requestManager.normalizeSpriteURL(url, format, '.png'), ResourceType.SpriteImage); + const imageRequestKey = `${id}_${imageRequestParameters.url}`; // use id_url as requestMap key to make sure it is unique + combinedRequestsMap[imageRequestKey] = ImageRequest.getImage(imageRequestParameters, (err, img) => { + delete combinedRequestsMap[imageRequestKey]; + imagesMap[id] = img; + doOnceCompleted(callback, jsonsMap, imagesMap, err, spriteArrayLength); + }); + } + return { + cancel() { + for (const requst of Object.values(combinedRequestsMap)) { + requst.cancel(); + } + } + }; +} +/** + * @param callbackFunc - the callback function (both erro and success) + * @param jsonsMap - JSON data map + * @param imagesMap - image data map + * @param err - error object + * @param expectedResultCounter - number of expected JSON or Image results when everything is finished, respectively. + */ +function doOnceCompleted(callbackFunc, jsonsMap, imagesMap, err, expectedResultCounter) { + if (err) { + callbackFunc(err); + return; + } + if (expectedResultCounter !== Object.values(jsonsMap).length || expectedResultCounter !== Object.values(imagesMap).length) { + // not done yet, nothing to do + return; + } + const result = {}; + for (const spriteName in jsonsMap) { + result[spriteName] = {}; + const context = performance.browser.getImageCanvasContext(imagesMap[spriteName]); + const json = jsonsMap[spriteName]; + for (const id in json) { + const { width, height, x, y, sdf, pixelRatio, stretchX, stretchY, content } = json[id]; + const spriteData = { width, height, x, y, context }; + result[spriteName][id] = { data: null, pixelRatio, sdf, stretchX, stretchY, content, spriteData }; + } + } + callbackFunc(null, result); +} + +/** + * @internal + * A `Texture` GL related object + */ +class Texture { + constructor(context, image, format, options) { + this.context = context; + this.format = format; + this.texture = context.gl.createTexture(); + this.update(image, options); + } + update(image, options, position) { + const { width, height } = image; + const resize = (!this.size || this.size[0] !== width || this.size[1] !== height) && !position; + const { context } = this; + const { gl } = context; + this.useMipmap = Boolean(options && options.useMipmap); + gl.bindTexture(gl.TEXTURE_2D, this.texture); + context.pixelStoreUnpackFlipY.set(false); + context.pixelStoreUnpack.set(1); + context.pixelStoreUnpackPremultiplyAlpha.set(this.format === gl.RGBA && (!options || options.premultiply !== false)); + if (resize) { + this.size = [width, height]; + if (image instanceof HTMLImageElement || image instanceof HTMLCanvasElement || image instanceof HTMLVideoElement || image instanceof ImageData || performance.isImageBitmap(image)) { + gl.texImage2D(gl.TEXTURE_2D, 0, this.format, this.format, gl.UNSIGNED_BYTE, image); + } + else { + gl.texImage2D(gl.TEXTURE_2D, 0, this.format, width, height, 0, this.format, gl.UNSIGNED_BYTE, image.data); + } + } + else { + const { x, y } = position || { x: 0, y: 0 }; + if (image instanceof HTMLImageElement || image instanceof HTMLCanvasElement || image instanceof HTMLVideoElement || image instanceof ImageData || performance.isImageBitmap(image)) { + gl.texSubImage2D(gl.TEXTURE_2D, 0, x, y, gl.RGBA, gl.UNSIGNED_BYTE, image); + } + else { + gl.texSubImage2D(gl.TEXTURE_2D, 0, x, y, width, height, gl.RGBA, gl.UNSIGNED_BYTE, image.data); + } + } + if (this.useMipmap && this.isSizePowerOfTwo()) { + gl.generateMipmap(gl.TEXTURE_2D); + } + } + bind(filter, wrap, minFilter) { + const { context } = this; + const { gl } = context; + gl.bindTexture(gl.TEXTURE_2D, this.texture); + if (minFilter === gl.LINEAR_MIPMAP_NEAREST && !this.isSizePowerOfTwo()) { + minFilter = gl.LINEAR; + } + if (filter !== this.filter) { + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, filter); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, minFilter || filter); + this.filter = filter; + } + if (wrap !== this.wrap) { + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, wrap); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, wrap); + this.wrap = wrap; + } + } + isSizePowerOfTwo() { + return this.size[0] === this.size[1] && (Math.log(this.size[0]) / Math.LN2) % 1 === 0; + } + destroy() { + const { gl } = this.context; + gl.deleteTexture(this.texture); + this.texture = null; + } +} + +function renderStyleImage(image) { + const { userImage } = image; + if (userImage && userImage.render) { + const updated = userImage.render(); + if (updated) { + image.data.replace(new Uint8Array(userImage.data.buffer)); + return true; + } + } + return false; +} + +/* eslint-disable key-spacing */ +// When copied into the atlas texture, image data is padded by one pixel on each side. Icon +// images are padded with fully transparent pixels, while pattern images are padded with a +// copy of the image data wrapped from the opposite side. In both cases, this ensures the +// correct behavior of GL_LINEAR texture sampling mode. +const padding = 1; +/* + ImageManager does three things: + + 1. Tracks requests for icon images from tile workers and sends responses when the requests are fulfilled. + 2. Builds a texture atlas for pattern images. + 3. Rerenders renderable images once per frame + + These are disparate responsibilities and should eventually be handled by different classes. When we implement + data-driven support for `*-pattern`, we'll likely use per-bucket pattern atlases, and that would be a good time + to refactor this. +*/ +class ImageManager extends performance.Evented { + constructor() { + super(); + this.images = {}; + this.updatedImages = {}; + this.callbackDispatchedThisFrame = {}; + this.loaded = false; + this.requestors = []; + this.patterns = {}; + this.atlasImage = new performance.RGBAImage({ width: 1, height: 1 }); + this.dirty = true; + } + isLoaded() { + return this.loaded; + } + setLoaded(loaded) { + if (this.loaded === loaded) { + return; + } + this.loaded = loaded; + if (loaded) { + for (const { ids, callback } of this.requestors) { + this._notify(ids, callback); + } + this.requestors = []; + } + } + getImage(id) { + const image = this.images[id]; + // Extract sprite image data on demand + if (image && !image.data && image.spriteData) { + const spriteData = image.spriteData; + image.data = new performance.RGBAImage({ + width: spriteData.width, + height: spriteData.height + }, spriteData.context.getImageData(spriteData.x, spriteData.y, spriteData.width, spriteData.height).data); + image.spriteData = null; + } + return image; + } + addImage(id, image) { + if (this.images[id]) + throw new Error(`Image id ${id} already exist, use updateImage instead`); + if (this._validate(id, image)) { + this.images[id] = image; + } + } + _validate(id, image) { + let valid = true; + const data = image.data || image.spriteData; + if (!this._validateStretch(image.stretchX, data && data.width)) { + this.fire(new performance.ErrorEvent(new Error(`Image "${id}" has invalid "stretchX" value`))); + valid = false; + } + if (!this._validateStretch(image.stretchY, data && data.height)) { + this.fire(new performance.ErrorEvent(new Error(`Image "${id}" has invalid "stretchY" value`))); + valid = false; + } + if (!this._validateContent(image.content, image)) { + this.fire(new performance.ErrorEvent(new Error(`Image "${id}" has invalid "content" value`))); + valid = false; + } + return valid; + } + _validateStretch(stretch, size) { + if (!stretch) + return true; + let last = 0; + for (const part of stretch) { + if (part[0] < last || part[1] < part[0] || size < part[1]) + return false; + last = part[1]; + } + return true; + } + _validateContent(content, image) { + if (!content) + return true; + if (content.length !== 4) + return false; + const spriteData = image.spriteData; + const width = (spriteData && spriteData.width) || image.data.width; + const height = (spriteData && spriteData.height) || image.data.height; + if (content[0] < 0 || width < content[0]) + return false; + if (content[1] < 0 || height < content[1]) + return false; + if (content[2] < 0 || width < content[2]) + return false; + if (content[3] < 0 || height < content[3]) + return false; + if (content[2] < content[0]) + return false; + if (content[3] < content[1]) + return false; + return true; + } + updateImage(id, image, validate = true) { + const oldImage = this.getImage(id); + if (validate && (oldImage.data.width !== image.data.width || oldImage.data.height !== image.data.height)) { + throw new Error(`size mismatch between old image (${oldImage.data.width}x${oldImage.data.height}) and new image (${image.data.width}x${image.data.height}).`); + } + image.version = oldImage.version + 1; + this.images[id] = image; + this.updatedImages[id] = true; + } + removeImage(id) { + const image = this.images[id]; + delete this.images[id]; + delete this.patterns[id]; + if (image.userImage && image.userImage.onRemove) { + image.userImage.onRemove(); + } + } + listImages() { + return Object.keys(this.images); + } + getImages(ids, callback) { + // If the sprite has been loaded, or if all the icon dependencies are already present + // (i.e. if they've been added via runtime styling), then notify the requestor immediately. + // Otherwise, delay notification until the sprite is loaded. At that point, if any of the + // dependencies are still unavailable, we'll just assume they are permanently missing. + let hasAllDependencies = true; + if (!this.isLoaded()) { + for (const id of ids) { + if (!this.images[id]) { + hasAllDependencies = false; + } + } + } + if (this.isLoaded() || hasAllDependencies) { + this._notify(ids, callback); + } + else { + this.requestors.push({ ids, callback }); + } + } + _notify(ids, callback) { + const response = {}; + for (const id of ids) { + let image = this.getImage(id); + if (!image) { + this.fire(new performance.Event('styleimagemissing', { id })); + //Try to acquire image again in case styleimagemissing has populated it + image = this.getImage(id); + } + if (image) { + // Clone the image so that our own copy of its ArrayBuffer doesn't get transferred. + response[id] = { + data: image.data.clone(), + pixelRatio: image.pixelRatio, + sdf: image.sdf, + version: image.version, + stretchX: image.stretchX, + stretchY: image.stretchY, + content: image.content, + hasRenderCallback: Boolean(image.userImage && image.userImage.render) + }; + } + else { + performance.warnOnce(`Image "${id}" could not be loaded. Please make sure you have added the image with map.addImage() or a "sprite" property in your style. You can provide missing images by listening for the "styleimagemissing" map event.`); + } + } + callback(null, response); + } + // Pattern stuff + getPixelSize() { + const { width, height } = this.atlasImage; + return { width, height }; + } + getPattern(id) { + const pattern = this.patterns[id]; + const image = this.getImage(id); + if (!image) { + return null; + } + if (pattern && pattern.position.version === image.version) { + return pattern.position; + } + if (!pattern) { + const w = image.data.width + padding * 2; + const h = image.data.height + padding * 2; + const bin = { w, h, x: 0, y: 0 }; + const position = new performance.ImagePosition(bin, image); + this.patterns[id] = { bin, position }; + } + else { + pattern.position.version = image.version; + } + this._updatePatternAtlas(); + return this.patterns[id].position; + } + bind(context) { + const gl = context.gl; + if (!this.atlasTexture) { + this.atlasTexture = new Texture(context, this.atlasImage, gl.RGBA); + } + else if (this.dirty) { + this.atlasTexture.update(this.atlasImage); + this.dirty = false; + } + this.atlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + } + _updatePatternAtlas() { + const bins = []; + for (const id in this.patterns) { + bins.push(this.patterns[id].bin); + } + const { w, h } = performance.potpack(bins); + const dst = this.atlasImage; + dst.resize({ width: w || 1, height: h || 1 }); + for (const id in this.patterns) { + const { bin } = this.patterns[id]; + const x = bin.x + padding; + const y = bin.y + padding; + const src = this.getImage(id).data; + const w = src.width; + const h = src.height; + performance.RGBAImage.copy(src, dst, { x: 0, y: 0 }, { x, y }, { width: w, height: h }); + // Add 1 pixel wrapped padding on each side of the image. + performance.RGBAImage.copy(src, dst, { x: 0, y: h - 1 }, { x, y: y - 1 }, { width: w, height: 1 }); // T + performance.RGBAImage.copy(src, dst, { x: 0, y: 0 }, { x, y: y + h }, { width: w, height: 1 }); // B + performance.RGBAImage.copy(src, dst, { x: w - 1, y: 0 }, { x: x - 1, y }, { width: 1, height: h }); // L + performance.RGBAImage.copy(src, dst, { x: 0, y: 0 }, { x: x + w, y }, { width: 1, height: h }); // R + } + this.dirty = true; + } + beginFrame() { + this.callbackDispatchedThisFrame = {}; + } + dispatchRenderCallbacks(ids) { + for (const id of ids) { + // the callback for the image was already dispatched for a different frame + if (this.callbackDispatchedThisFrame[id]) + continue; + this.callbackDispatchedThisFrame[id] = true; + const image = this.getImage(id); + if (!image) + performance.warnOnce(`Image with ID: "${id}" was not found`); + const updated = renderStyleImage(image); + if (updated) { + this.updateImage(id, image); + } + } + } +} + +function loadGlyphRange(fontstack, range, urlTemplate, requestManager, callback) { + const begin = range * 256; + const end = begin + 255; + const request = requestManager.transformRequest(urlTemplate.replace('{fontstack}', fontstack).replace('{range}', `${begin}-${end}`), ResourceType.Glyphs); + performance.getArrayBuffer(request, (err, data) => { + if (err) { + callback(err); + } + else if (data) { + const glyphs = {}; + for (const glyph of performance.parseGlyphPbf(data)) { + glyphs[glyph.id] = glyph; + } + callback(null, glyphs); + } + }); +} + +const INF = 1e20; + +class TinySDF { + constructor({ + fontSize = 24, + buffer = 3, + radius = 8, + cutoff = 0.25, + fontFamily = 'sans-serif', + fontWeight = 'normal', + fontStyle = 'normal' + } = {}) { + this.buffer = buffer; + this.cutoff = cutoff; + this.radius = radius; + + // make the canvas size big enough to both have the specified buffer around the glyph + // for "halo", and account for some glyphs possibly being larger than their font size + const size = this.size = fontSize + buffer * 4; + + const canvas = this._createCanvas(size); + const ctx = this.ctx = canvas.getContext('2d', {willReadFrequently: true}); + ctx.font = `${fontStyle} ${fontWeight} ${fontSize}px ${fontFamily}`; + + ctx.textBaseline = 'alphabetic'; + ctx.textAlign = 'left'; // Necessary so that RTL text doesn't have different alignment + ctx.fillStyle = 'black'; + + // temporary arrays for the distance transform + this.gridOuter = new Float64Array(size * size); + this.gridInner = new Float64Array(size * size); + this.f = new Float64Array(size); + this.z = new Float64Array(size + 1); + this.v = new Uint16Array(size); + } + + _createCanvas(size) { + const canvas = document.createElement('canvas'); + canvas.width = canvas.height = size; + return canvas; + } + + draw(char) { + const { + width: glyphAdvance, + actualBoundingBoxAscent, + actualBoundingBoxDescent, + actualBoundingBoxLeft, + actualBoundingBoxRight + } = this.ctx.measureText(char); + + // The integer/pixel part of the top alignment is encoded in metrics.glyphTop + // The remainder is implicitly encoded in the rasterization + const glyphTop = Math.ceil(actualBoundingBoxAscent); + const glyphLeft = 0; + + // If the glyph overflows the canvas size, it will be clipped at the bottom/right + const glyphWidth = Math.max(0, Math.min(this.size - this.buffer, Math.ceil(actualBoundingBoxRight - actualBoundingBoxLeft))); + const glyphHeight = Math.min(this.size - this.buffer, glyphTop + Math.ceil(actualBoundingBoxDescent)); + + const width = glyphWidth + 2 * this.buffer; + const height = glyphHeight + 2 * this.buffer; + + const len = Math.max(width * height, 0); + const data = new Uint8ClampedArray(len); + const glyph = {data, width, height, glyphWidth, glyphHeight, glyphTop, glyphLeft, glyphAdvance}; + if (glyphWidth === 0 || glyphHeight === 0) return glyph; + + const {ctx, buffer, gridInner, gridOuter} = this; + ctx.clearRect(buffer, buffer, glyphWidth, glyphHeight); + ctx.fillText(char, buffer, buffer + glyphTop); + const imgData = ctx.getImageData(buffer, buffer, glyphWidth, glyphHeight); + + // Initialize grids outside the glyph range to alpha 0 + gridOuter.fill(INF, 0, len); + gridInner.fill(0, 0, len); + + for (let y = 0; y < glyphHeight; y++) { + for (let x = 0; x < glyphWidth; x++) { + const a = imgData.data[4 * (y * glyphWidth + x) + 3] / 255; // alpha value + if (a === 0) continue; // empty pixels + + const j = (y + buffer) * width + x + buffer; + + if (a === 1) { // fully drawn pixels + gridOuter[j] = 0; + gridInner[j] = INF; + + } else { // aliased pixels + const d = 0.5 - a; + gridOuter[j] = d > 0 ? d * d : 0; + gridInner[j] = d < 0 ? d * d : 0; + } + } + } + + edt(gridOuter, 0, 0, width, height, width, this.f, this.v, this.z); + edt(gridInner, buffer, buffer, glyphWidth, glyphHeight, width, this.f, this.v, this.z); + + for (let i = 0; i < len; i++) { + const d = Math.sqrt(gridOuter[i]) - Math.sqrt(gridInner[i]); + data[i] = Math.round(255 - 255 * (d / this.radius + this.cutoff)); + } + + return glyph; + } +} + +// 2D Euclidean squared distance transform by Felzenszwalb & Huttenlocher https://cs.brown.edu/~pff/papers/dt-final.pdf +function edt(data, x0, y0, width, height, gridSize, f, v, z) { + for (let x = x0; x < x0 + width; x++) edt1d(data, y0 * gridSize + x, gridSize, height, f, v, z); + for (let y = y0; y < y0 + height; y++) edt1d(data, y * gridSize + x0, 1, width, f, v, z); +} + +// 1D squared distance transform +function edt1d(grid, offset, stride, length, f, v, z) { + v[0] = 0; + z[0] = -INF; + z[1] = INF; + f[0] = grid[offset]; + + for (let q = 1, k = 0, s = 0; q < length; q++) { + f[q] = grid[offset + q * stride]; + const q2 = q * q; + do { + const r = v[k]; + s = (f[q] - f[r] + q2 - r * r) / (q - r) / 2; + } while (s <= z[k] && --k > -1); + + k++; + v[k] = q; + z[k] = s; + z[k + 1] = INF; + } + + for (let q = 0, k = 0; q < length; q++) { + while (z[k + 1] < q) k++; + const r = v[k]; + const qr = q - r; + grid[offset + q * stride] = f[r] + qr * qr; + } +} + +class GlyphManager { + constructor(requestManager, localIdeographFontFamily) { + this.requestManager = requestManager; + this.localIdeographFontFamily = localIdeographFontFamily; + this.entries = {}; + } + setURL(url) { + this.url = url; + } + getGlyphs(glyphs, callback) { + const all = []; + for (const stack in glyphs) { + for (const id of glyphs[stack]) { + all.push({ stack, id }); + } + } + performance.asyncAll(all, ({ stack, id }, callback) => { + let entry = this.entries[stack]; + if (!entry) { + entry = this.entries[stack] = { + glyphs: {}, + requests: {}, + ranges: {} + }; + } + let glyph = entry.glyphs[id]; + if (glyph !== undefined) { + callback(null, { stack, id, glyph }); + return; + } + glyph = this._tinySDF(entry, stack, id); + if (glyph) { + entry.glyphs[id] = glyph; + callback(null, { stack, id, glyph }); + return; + } + const range = Math.floor(id / 256); + if (range * 256 > 65535) { + callback(new Error('glyphs > 65535 not supported')); + return; + } + if (entry.ranges[range]) { + callback(null, { stack, id, glyph }); + return; + } + if (!this.url) { + callback(new Error('glyphsUrl is not set')); + return; + } + let requests = entry.requests[range]; + if (!requests) { + requests = entry.requests[range] = []; + GlyphManager.loadGlyphRange(stack, range, this.url, this.requestManager, (err, response) => { + if (response) { + for (const id in response) { + if (!this._doesCharSupportLocalGlyph(+id)) { + entry.glyphs[+id] = response[+id]; + } + } + entry.ranges[range] = true; + } + for (const cb of requests) { + cb(err, response); + } + delete entry.requests[range]; + }); + } + requests.push((err, result) => { + if (err) { + callback(err); + } + else if (result) { + callback(null, { stack, id, glyph: result[id] || null }); + } + }); + }, (err, glyphs) => { + if (err) { + callback(err); + } + else if (glyphs) { + const result = {}; + for (const { stack, id, glyph } of glyphs) { + // Clone the glyph so that our own copy of its ArrayBuffer doesn't get transferred. + (result[stack] || (result[stack] = {}))[id] = glyph && { + id: glyph.id, + bitmap: glyph.bitmap.clone(), + metrics: glyph.metrics + }; + } + callback(null, result); + } + }); + } + _doesCharSupportLocalGlyph(id) { + /* eslint-disable new-cap */ + return !!this.localIdeographFontFamily && + (performance.unicodeBlockLookup['CJK Unified Ideographs'](id) || + performance.unicodeBlockLookup['Hangul Syllables'](id) || + performance.unicodeBlockLookup['Hiragana'](id) || + performance.unicodeBlockLookup['Katakana'](id)); + /* eslint-enable new-cap */ + } + _tinySDF(entry, stack, id) { + const fontFamily = this.localIdeographFontFamily; + if (!fontFamily) { + return; + } + if (!this._doesCharSupportLocalGlyph(id)) { + return; + } + // Client-generated glyphs are rendered at 2x texture scale, + // because CJK glyphs are more detailed than others. + const textureScale = 2; + let tinySDF = entry.tinySDF; + if (!tinySDF) { + let fontWeight = '400'; + if (/bold/i.test(stack)) { + fontWeight = '900'; + } + else if (/medium/i.test(stack)) { + fontWeight = '500'; + } + else if (/light/i.test(stack)) { + fontWeight = '200'; + } + tinySDF = entry.tinySDF = new GlyphManager.TinySDF({ + fontSize: 24 * textureScale, + buffer: 3 * textureScale, + radius: 8 * textureScale, + cutoff: 0.25, + fontFamily, + fontWeight + }); + } + const char = tinySDF.draw(String.fromCharCode(id)); + /** + * TinySDF's "top" is the distance from the alphabetic baseline to the top of the glyph. + * Server-generated fonts specify "top" relative to an origin above the em box (the origin + * comes from FreeType, but I'm unclear on exactly how it's derived) + * ref: https://github.com/mapbox/sdf-glyph-foundry + * + * Server fonts don't yet include baseline information, so we can't line up exactly with them + * (and they don't line up with each other) + * ref: https://github.com/mapbox/node-fontnik/pull/160 + * + * To approximately align TinySDF glyphs with server-provided glyphs, we use this baseline adjustment + * factor calibrated to be in between DIN Pro and Arial Unicode (but closer to Arial Unicode) + */ + const topAdjustment = 27.5; + const leftAdjustment = 0.5; + return { + id, + bitmap: new performance.AlphaImage({ width: char.width || 30 * textureScale, height: char.height || 30 * textureScale }, char.data), + metrics: { + width: char.glyphWidth / textureScale || 24, + height: char.glyphHeight / textureScale || 24, + left: (char.glyphLeft / textureScale + leftAdjustment) || 0, + top: char.glyphTop / textureScale - topAdjustment || -8, + advance: char.glyphAdvance / textureScale || 24, + isDoubleResolution: true + } + }; + } +} +// exposed as statics to enable stubbing in unit tests +GlyphManager.loadGlyphRange = loadGlyphRange; +GlyphManager.TinySDF = TinySDF; + +class LightPositionProperty { + constructor() { + this.specification = performance.v8Spec.light.position; + } + possiblyEvaluate(value, parameters) { + return performance.sphericalToCartesian(value.expression.evaluate(parameters)); + } + interpolate(a, b, t) { + return { + x: performance.interpolate.number(a.x, b.x, t), + y: performance.interpolate.number(a.y, b.y, t), + z: performance.interpolate.number(a.z, b.z, t), + }; + } +} +const TRANSITION_SUFFIX = '-transition'; +let lightProperties; +/* + * Represents the light used to light extruded features. + */ +class Light extends performance.Evented { + constructor(lightOptions) { + super(); + lightProperties = lightProperties || new performance.Properties({ + 'anchor': new performance.DataConstantProperty(performance.v8Spec.light.anchor), + 'position': new LightPositionProperty(), + 'color': new performance.DataConstantProperty(performance.v8Spec.light.color), + 'intensity': new performance.DataConstantProperty(performance.v8Spec.light.intensity), + }); + this._transitionable = new performance.Transitionable(lightProperties); + this.setLight(lightOptions); + this._transitioning = this._transitionable.untransitioned(); + } + getLight() { + return this._transitionable.serialize(); + } + setLight(light, options = {}) { + if (this._validate(performance.validateLight, light, options)) { + return; + } + for (const name in light) { + const value = light[name]; + if (name.endsWith(TRANSITION_SUFFIX)) { + this._transitionable.setTransition(name.slice(0, -TRANSITION_SUFFIX.length), value); + } + else { + this._transitionable.setValue(name, value); + } + } + } + updateTransitions(parameters) { + this._transitioning = this._transitionable.transitioned(parameters, this._transitioning); + } + hasTransition() { + return this._transitioning.hasTransition(); + } + recalculate(parameters) { + this.properties = this._transitioning.possiblyEvaluate(parameters); + } + _validate(validate, value, options) { + if (options && options.validate === false) { + return false; + } + return performance.emitValidationErrors(this, validate.call(performance.validateStyle, performance.extend({ + value, + // Workaround for https://github.com/mapbox/mapbox-gl-js/issues/2407 + style: { glyphs: true, sprite: true }, + styleSpec: performance.v8Spec + }))); + } +} + +/** + * @internal + * A LineAtlas lets us reuse rendered dashed lines + * by writing many of them to a texture and then fetching their positions + * using {@link LineAtlas#getDash}. + * + * @param width - the width + * @param height - the height + */ +class LineAtlas { + constructor(width, height) { + this.width = width; + this.height = height; + this.nextRow = 0; + this.data = new Uint8Array(this.width * this.height); + this.dashEntry = {}; + } + /** + * Get or create a dash line pattern. + * + * @param dasharray - the key (represented by numbers) to get the dash texture + * @param round - whether to add circle caps in between dash segments + * @returns position of dash texture in {@link DashEntry} + */ + getDash(dasharray, round) { + const key = dasharray.join(',') + String(round); + if (!this.dashEntry[key]) { + this.dashEntry[key] = this.addDash(dasharray, round); + } + return this.dashEntry[key]; + } + getDashRanges(dasharray, lineAtlasWidth, stretch) { + // If dasharray has an odd length, both the first and last parts + // are dashes and should be joined seamlessly. + const oddDashArray = dasharray.length % 2 === 1; + const ranges = []; + let left = oddDashArray ? -dasharray[dasharray.length - 1] * stretch : 0; + let right = dasharray[0] * stretch; + let isDash = true; + ranges.push({ left, right, isDash, zeroLength: dasharray[0] === 0 }); + let currentDashLength = dasharray[0]; + for (let i = 1; i < dasharray.length; i++) { + isDash = !isDash; + const dashLength = dasharray[i]; + left = currentDashLength * stretch; + currentDashLength += dashLength; + right = currentDashLength * stretch; + ranges.push({ left, right, isDash, zeroLength: dashLength === 0 }); + } + return ranges; + } + addRoundDash(ranges, stretch, n) { + const halfStretch = stretch / 2; + for (let y = -n; y <= n; y++) { + const row = this.nextRow + n + y; + const index = this.width * row; + let currIndex = 0; + let range = ranges[currIndex]; + for (let x = 0; x < this.width; x++) { + if (x / range.right > 1) { + range = ranges[++currIndex]; + } + const distLeft = Math.abs(x - range.left); + const distRight = Math.abs(x - range.right); + const minDist = Math.min(distLeft, distRight); + let signedDistance; + const distMiddle = y / n * (halfStretch + 1); + if (range.isDash) { + const distEdge = halfStretch - Math.abs(distMiddle); + signedDistance = Math.sqrt(minDist * minDist + distEdge * distEdge); + } + else { + signedDistance = halfStretch - Math.sqrt(minDist * minDist + distMiddle * distMiddle); + } + this.data[index + x] = Math.max(0, Math.min(255, signedDistance + 128)); + } + } + } + addRegularDash(ranges) { + // Collapse any zero-length range + // Collapse neighbouring same-type parts into a single part + for (let i = ranges.length - 1; i >= 0; --i) { + const part = ranges[i]; + const next = ranges[i + 1]; + if (part.zeroLength) { + ranges.splice(i, 1); + } + else if (next && next.isDash === part.isDash) { + next.left = part.left; + ranges.splice(i, 1); + } + } + // Combine the first and last parts if possible + const first = ranges[0]; + const last = ranges[ranges.length - 1]; + if (first.isDash === last.isDash) { + first.left = last.left - this.width; + last.right = first.right + this.width; + } + const index = this.width * this.nextRow; + let currIndex = 0; + let range = ranges[currIndex]; + for (let x = 0; x < this.width; x++) { + if (x / range.right > 1) { + range = ranges[++currIndex]; + } + const distLeft = Math.abs(x - range.left); + const distRight = Math.abs(x - range.right); + const minDist = Math.min(distLeft, distRight); + const signedDistance = range.isDash ? minDist : -minDist; + this.data[index + x] = Math.max(0, Math.min(255, signedDistance + 128)); + } + } + addDash(dasharray, round) { + const n = round ? 7 : 0; + const height = 2 * n + 1; + if (this.nextRow + height > this.height) { + performance.warnOnce('LineAtlas out of space'); + return null; + } + let length = 0; + for (let i = 0; i < dasharray.length; i++) { + length += dasharray[i]; + } + if (length !== 0) { + const stretch = this.width / length; + const ranges = this.getDashRanges(dasharray, this.width, stretch); + if (round) { + this.addRoundDash(ranges, stretch, n); + } + else { + this.addRegularDash(ranges); + } + } + const dashEntry = { + y: (this.nextRow + n + 0.5) / this.height, + height: 2 * n / this.height, + width: length + }; + this.nextRow += height; + this.dirty = true; + return dashEntry; + } + bind(context) { + const gl = context.gl; + if (!this.texture) { + this.texture = gl.createTexture(); + gl.bindTexture(gl.TEXTURE_2D, this.texture); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.REPEAT); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.REPEAT); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR); + gl.texImage2D(gl.TEXTURE_2D, 0, gl.ALPHA, this.width, this.height, 0, gl.ALPHA, gl.UNSIGNED_BYTE, this.data); + } + else { + gl.bindTexture(gl.TEXTURE_2D, this.texture); + if (this.dirty) { + this.dirty = false; + gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, this.width, this.height, gl.ALPHA, gl.UNSIGNED_BYTE, this.data); + } + } + } +} + +/** + * Responsible for sending messages from a {@link Source} to an associated + * {@link WorkerSource}. + */ +class Dispatcher { + constructor(workerPool, parent, mapId) { + this.workerPool = workerPool; + this.actors = []; + this.currentActor = 0; + this.id = mapId; + const workers = this.workerPool.acquire(mapId); + for (let i = 0; i < workers.length; i++) { + const worker = workers[i]; + const actor = new performance.Actor(worker, parent, mapId); + actor.name = `Worker ${i}`; + this.actors.push(actor); + } + if (!this.actors.length) + throw new Error('No actors found'); + } + /** + * Broadcast a message to all Workers. + */ + broadcast(type, data, cb) { + cb = cb || function () { }; + performance.asyncAll(this.actors, (actor, done) => { + actor.send(type, data, done); + }, cb); + } + /** + * Acquires an actor to dispatch messages to. The actors are distributed in round-robin fashion. + * @returns An actor object backed by a web worker for processing messages. + */ + getActor() { + this.currentActor = (this.currentActor + 1) % this.actors.length; + return this.actors[this.currentActor]; + } + remove(mapRemoved = true) { + this.actors.forEach((actor) => { actor.remove(); }); + this.actors = []; + if (mapRemoved) + this.workerPool.release(this.id); + } +} + +function loadTileJson(options, requestManager, callback) { + const loaded = function (err, tileJSON) { + if (err) { + return callback(err); + } + else if (tileJSON) { + const result = performance.pick( + // explicit source options take precedence over TileJSON + performance.extend(tileJSON, options), ['tiles', 'minzoom', 'maxzoom', 'attribution', 'bounds', 'scheme', 'tileSize', 'encoding']); + if (tileJSON.vector_layers) { + result.vectorLayers = tileJSON.vector_layers; + result.vectorLayerIds = result.vectorLayers.map((layer) => { return layer.id; }); + } + callback(null, result); + } + }; + if (options.url) { + return performance.getJSON(requestManager.transformRequest(options.url, ResourceType.Source), loaded); + } + else { + return performance.browser.frame(() => loaded(null, options)); + } +} + +/** + * A `LngLatBounds` object represents a geographical bounding box, + * defined by its southwest and northeast points in longitude and latitude. + * + * If no arguments are provided to the constructor, a `null` bounding box is created. + * + * Note that any Mapbox GL method that accepts a `LngLatBounds` object as an argument or option + * can also accept an `Array` of two {@link LngLatLike} constructs and will perform an implicit conversion. + * This flexible type is documented as {@link LngLatBoundsLike}. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let sw = new maplibregl.LngLat(-73.9876, 40.7661); + * let ne = new maplibregl.LngLat(-73.9397, 40.8002); + * let llb = new maplibregl.LngLatBounds(sw, ne); + * ``` + */ +class LngLatBounds { + /** + * @param sw - The southwest corner of the bounding box. + * OR array of 4 numbers in the order of west, south, east, north + * OR array of 2 LngLatLike: [sw,ne] + * @param ne - The northeast corner of the bounding box. + * @example + * ```ts + * let sw = new maplibregl.LngLat(-73.9876, 40.7661); + * let ne = new maplibregl.LngLat(-73.9397, 40.8002); + * let llb = new maplibregl.LngLatBounds(sw, ne); + * ``` + * OR + * ```ts + * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661, -73.9397, 40.8002]); + * ``` + * OR + * ```ts + * let llb = new maplibregl.LngLatBounds([sw, ne]); + * ``` + */ + constructor(sw, ne) { + if (!sw) { + // noop + } + else if (ne) { + this.setSouthWest(sw).setNorthEast(ne); + } + else if (Array.isArray(sw)) { + if (sw.length === 4) { + // 4 element array: west, south, east, north + this.setSouthWest([sw[0], sw[1]]).setNorthEast([sw[2], sw[3]]); + } + else { + this.setSouthWest(sw[0]).setNorthEast(sw[1]); + } + } + } + /** + * Set the northeast corner of the bounding box + * + * @param ne - a {@link LngLatLike} object describing the northeast corner of the bounding box. + * @returns `this` + */ + setNorthEast(ne) { + this._ne = ne instanceof performance.LngLat ? new performance.LngLat(ne.lng, ne.lat) : performance.LngLat.convert(ne); + return this; + } + /** + * Set the southwest corner of the bounding box + * + * @param sw - a {@link LngLatLike} object describing the southwest corner of the bounding box. + * @returns `this` + */ + setSouthWest(sw) { + this._sw = sw instanceof performance.LngLat ? new performance.LngLat(sw.lng, sw.lat) : performance.LngLat.convert(sw); + return this; + } + /** + * Extend the bounds to include a given LngLatLike or LngLatBoundsLike. + * + * @param obj - object to extend to + * @returns `this` + */ + extend(obj) { + const sw = this._sw, ne = this._ne; + let sw2, ne2; + if (obj instanceof performance.LngLat) { + sw2 = obj; + ne2 = obj; + } + else if (obj instanceof LngLatBounds) { + sw2 = obj._sw; + ne2 = obj._ne; + if (!sw2 || !ne2) + return this; + } + else { + if (Array.isArray(obj)) { + if (obj.length === 4 || obj.every(Array.isArray)) { + const lngLatBoundsObj = obj; + return this.extend(LngLatBounds.convert(lngLatBoundsObj)); + } + else { + const lngLatObj = obj; + return this.extend(performance.LngLat.convert(lngLatObj)); + } + } + else if (obj && ('lng' in obj || 'lon' in obj) && 'lat' in obj) { + return this.extend(performance.LngLat.convert(obj)); + } + return this; + } + if (!sw && !ne) { + this._sw = new performance.LngLat(sw2.lng, sw2.lat); + this._ne = new performance.LngLat(ne2.lng, ne2.lat); + } + else { + sw.lng = Math.min(sw2.lng, sw.lng); + sw.lat = Math.min(sw2.lat, sw.lat); + ne.lng = Math.max(ne2.lng, ne.lng); + ne.lat = Math.max(ne2.lat, ne.lat); + } + return this; + } + /** + * Returns the geographical coordinate equidistant from the bounding box's corners. + * + * @returns The bounding box's center. + * @example + * ```ts + * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]); + * llb.getCenter(); // = LngLat {lng: -73.96365, lat: 40.78315} + * ``` + */ + getCenter() { + return new performance.LngLat((this._sw.lng + this._ne.lng) / 2, (this._sw.lat + this._ne.lat) / 2); + } + /** + * Returns the southwest corner of the bounding box. + * + * @returns The southwest corner of the bounding box. + */ + getSouthWest() { return this._sw; } + /** + * Returns the northeast corner of the bounding box. + * + * @returns The northeast corner of the bounding box. + */ + getNorthEast() { return this._ne; } + /** + * Returns the northwest corner of the bounding box. + * + * @returns The northwest corner of the bounding box. + */ + getNorthWest() { return new performance.LngLat(this.getWest(), this.getNorth()); } + /** + * Returns the southeast corner of the bounding box. + * + * @returns The southeast corner of the bounding box. + */ + getSouthEast() { return new performance.LngLat(this.getEast(), this.getSouth()); } + /** + * Returns the west edge of the bounding box. + * + * @returns The west edge of the bounding box. + */ + getWest() { return this._sw.lng; } + /** + * Returns the south edge of the bounding box. + * + * @returns The south edge of the bounding box. + */ + getSouth() { return this._sw.lat; } + /** + * Returns the east edge of the bounding box. + * + * @returns The east edge of the bounding box. + */ + getEast() { return this._ne.lng; } + /** + * Returns the north edge of the bounding box. + * + * @returns The north edge of the bounding box. + */ + getNorth() { return this._ne.lat; } + /** + * Returns the bounding box represented as an array. + * + * @returns The bounding box represented as an array, consisting of the + * southwest and northeast coordinates of the bounding represented as arrays of numbers. + * @example + * ```ts + * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]); + * llb.toArray(); // = [[-73.9876, 40.7661], [-73.9397, 40.8002]] + * ``` + */ + toArray() { + return [this._sw.toArray(), this._ne.toArray()]; + } + /** + * Return the bounding box represented as a string. + * + * @returns The bounding box represents as a string of the format + * `'LngLatBounds(LngLat(lng, lat), LngLat(lng, lat))'`. + * @example + * ```ts + * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]); + * llb.toString(); // = "LngLatBounds(LngLat(-73.9876, 40.7661), LngLat(-73.9397, 40.8002))" + * ``` + */ + toString() { + return `LngLatBounds(${this._sw.toString()}, ${this._ne.toString()})`; + } + /** + * Check if the bounding box is an empty/`null`-type box. + * + * @returns True if bounds have been defined, otherwise false. + */ + isEmpty() { + return !(this._sw && this._ne); + } + /** + * Check if the point is within the bounding box. + * + * @param lnglat - geographic point to check against. + * @returns `true` if the point is within the bounding box. + * @example + * ```ts + * let llb = new maplibregl.LngLatBounds( + * new maplibregl.LngLat(-73.9876, 40.7661), + * new maplibregl.LngLat(-73.9397, 40.8002) + * ); + * + * let ll = new maplibregl.LngLat(-73.9567, 40.7789); + * + * console.log(llb.contains(ll)); // = true + * ``` + */ + contains(lnglat) { + const { lng, lat } = performance.LngLat.convert(lnglat); + const containsLatitude = this._sw.lat <= lat && lat <= this._ne.lat; + let containsLongitude = this._sw.lng <= lng && lng <= this._ne.lng; + if (this._sw.lng > this._ne.lng) { // wrapped coordinates + containsLongitude = this._sw.lng >= lng && lng >= this._ne.lng; + } + return containsLatitude && containsLongitude; + } + /** + * Converts an array to a `LngLatBounds` object. + * + * If a `LngLatBounds` object is passed in, the function returns it unchanged. + * + * Internally, the function calls `LngLat#convert` to convert arrays to `LngLat` values. + * + * @param input - An array of two coordinates to convert, or a `LngLatBounds` object to return. + * @returns A new `LngLatBounds` object, if a conversion occurred, or the original `LngLatBounds` object. + * @example + * ```ts + * let arr = [[-73.9876, 40.7661], [-73.9397, 40.8002]]; + * let llb = maplibregl.LngLatBounds.convert(arr); // = LngLatBounds {_sw: LngLat {lng: -73.9876, lat: 40.7661}, _ne: LngLat {lng: -73.9397, lat: 40.8002}} + * ``` + */ + static convert(input) { + if (input instanceof LngLatBounds) + return input; + if (!input) + return input; + return new LngLatBounds(input); + } + /** + * Returns a `LngLatBounds` from the coordinates extended by a given `radius`. The returned `LngLatBounds` completely contains the `radius`. + * + * @param center - center coordinates of the new bounds. + * @param radius - Distance in meters from the coordinates to extend the bounds. + * @returns A new `LngLatBounds` object representing the coordinates extended by the `radius`. + * @example + * ```ts + * let center = new maplibregl.LngLat(-73.9749, 40.7736); + * maplibregl.LngLatBounds.fromLngLat(100).toArray(); // = [[-73.97501862141328, 40.77351016847229], [-73.97478137858673, 40.77368983152771]] + * ``` + */ + static fromLngLat(center, radius = 0) { + const earthCircumferenceInMetersAtEquator = 40075017; + const latAccuracy = 360 * radius / earthCircumferenceInMetersAtEquator, lngAccuracy = latAccuracy / Math.cos((Math.PI / 180) * center.lat); + return new LngLatBounds(new performance.LngLat(center.lng - lngAccuracy, center.lat - latAccuracy), new performance.LngLat(center.lng + lngAccuracy, center.lat + latAccuracy)); + } +} + +class TileBounds { + constructor(bounds, minzoom, maxzoom) { + this.bounds = LngLatBounds.convert(this.validateBounds(bounds)); + this.minzoom = minzoom || 0; + this.maxzoom = maxzoom || 24; + } + validateBounds(bounds) { + // make sure the bounds property contains valid longitude and latitudes + if (!Array.isArray(bounds) || bounds.length !== 4) + return [-180, -90, 180, 90]; + return [Math.max(-180, bounds[0]), Math.max(-90, bounds[1]), Math.min(180, bounds[2]), Math.min(90, bounds[3])]; + } + contains(tileID) { + const worldSize = Math.pow(2, tileID.z); + const level = { + minX: Math.floor(performance.mercatorXfromLng(this.bounds.getWest()) * worldSize), + minY: Math.floor(performance.mercatorYfromLat(this.bounds.getNorth()) * worldSize), + maxX: Math.ceil(performance.mercatorXfromLng(this.bounds.getEast()) * worldSize), + maxY: Math.ceil(performance.mercatorYfromLat(this.bounds.getSouth()) * worldSize) + }; + const hit = tileID.x >= level.minX && tileID.x < level.maxX && tileID.y >= level.minY && tileID.y < level.maxY; + return hit; + } +} + +/** + * A source containing vector tiles in [Mapbox Vector Tile format](https://docs.mapbox.com/vector-tiles/reference/). + * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.) + * + * @group Sources + * + * @example + * ```ts + * map.addSource('some id', { + * type: 'vector', + * url: 'https://demotiles.maplibre.org/tiles/tiles.json' + * }); + * ``` + * + * @example + * ```ts + * map.addSource('some id', { + * type: 'vector', + * tiles: ['https://d25uarhxywzl1j.cloudfront.net/v0.1/{z}/{x}/{y}.mvt'], + * minzoom: 6, + * maxzoom: 14 + * }); + * ``` + * + * @example + * ```ts + * map.getSource('some id').setUrl("https://demotiles.maplibre.org/tiles/tiles.json"); + * ``` + * + * @example + * ```ts + * map.getSource('some id').setTiles(['https://d25uarhxywzl1j.cloudfront.net/v0.1/{z}/{x}/{y}.mvt']); + * ``` + * @see [Add a vector tile source](https://maplibre.org/maplibre-gl-js/docs/examples/vector-source/) + */ +class VectorTileSource extends performance.Evented { + constructor(id, options, dispatcher, eventedParent) { + super(); + this.load = () => { + this._loaded = false; + this.fire(new performance.Event('dataloading', { dataType: 'source' })); + this._tileJSONRequest = loadTileJson(this._options, this.map._requestManager, (err, tileJSON) => { + this._tileJSONRequest = null; + this._loaded = true; + this.map.style.sourceCaches[this.id].clearTiles(); + if (err) { + this.fire(new performance.ErrorEvent(err)); + } + else if (tileJSON) { + performance.extend(this, tileJSON); + if (tileJSON.bounds) + this.tileBounds = new TileBounds(tileJSON.bounds, this.minzoom, this.maxzoom); + // `content` is included here to prevent a race condition where `Style#_updateSources` is called + // before the TileJSON arrives. this makes sure the tiles needed are loaded once TileJSON arrives + // ref: https://github.com/mapbox/mapbox-gl-js/pull/4347#discussion_r104418088 + this.fire(new performance.Event('data', { dataType: 'source', sourceDataType: 'metadata' })); + this.fire(new performance.Event('data', { dataType: 'source', sourceDataType: 'content' })); + } + }); + }; + this.serialize = () => { + return performance.extend({}, this._options); + }; + this.id = id; + this.dispatcher = dispatcher; + this.type = 'vector'; + this.minzoom = 0; + this.maxzoom = 22; + this.scheme = 'xyz'; + this.tileSize = 512; + this.reparseOverscaled = true; + this.isTileClipped = true; + this._loaded = false; + performance.extend(this, performance.pick(options, ['url', 'scheme', 'tileSize', 'promoteId'])); + this._options = performance.extend({ type: 'vector' }, options); + this._collectResourceTiming = options.collectResourceTiming; + if (this.tileSize !== 512) { + throw new Error('vector tile sources must have a tileSize of 512'); + } + this.setEventedParent(eventedParent); + } + loaded() { + return this._loaded; + } + hasTile(tileID) { + return !this.tileBounds || this.tileBounds.contains(tileID.canonical); + } + onAdd(map) { + this.map = map; + this.load(); + } + setSourceProperty(callback) { + if (this._tileJSONRequest) { + this._tileJSONRequest.cancel(); + } + callback(); + this.load(); + } + /** + * Sets the source `tiles` property and re-renders the map. + * + * @param tiles - An array of one or more tile source URLs, as in the TileJSON spec. + * @returns `this` + */ + setTiles(tiles) { + this.setSourceProperty(() => { + this._options.tiles = tiles; + }); + return this; + } + /** + * Sets the source `url` property and re-renders the map. + * + * @param url - A URL to a TileJSON resource. Supported protocols are `http:` and `https:`. + * @returns `this` + */ + setUrl(url) { + this.setSourceProperty(() => { + this.url = url; + this._options.url = url; + }); + return this; + } + onRemove() { + if (this._tileJSONRequest) { + this._tileJSONRequest.cancel(); + this._tileJSONRequest = null; + } + } + loadTile(tile, callback) { + const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme); + const params = { + request: this.map._requestManager.transformRequest(url, ResourceType.Tile), + uid: tile.uid, + tileID: tile.tileID, + zoom: tile.tileID.overscaledZ, + tileSize: this.tileSize * tile.tileID.overscaleFactor(), + type: this.type, + source: this.id, + pixelRatio: this.map.getPixelRatio(), + showCollisionBoxes: this.map.showCollisionBoxes, + promoteId: this.promoteId + }; + params.request.collectResourceTiming = this._collectResourceTiming; + if (!tile.actor || tile.state === 'expired') { + tile.actor = this.dispatcher.getActor(); + tile.request = tile.actor.send('loadTile', params, done.bind(this)); + } + else if (tile.state === 'loading') { + // schedule tile reloading after it has been loaded + tile.reloadCallback = callback; + } + else { + tile.request = tile.actor.send('reloadTile', params, done.bind(this)); + } + function done(err, data) { + delete tile.request; + if (tile.aborted) + return callback(null); + if (err && err.status !== 404) { + return callback(err); + } + if (data && data.resourceTiming) + tile.resourceTiming = data.resourceTiming; + if (this.map._refreshExpiredTiles && data) + tile.setExpiryData(data); + tile.loadVectorData(data, this.map.painter); + callback(null); + if (tile.reloadCallback) { + this.loadTile(tile, tile.reloadCallback); + tile.reloadCallback = null; + } + } + } + abortTile(tile) { + if (tile.request) { + tile.request.cancel(); + delete tile.request; + } + if (tile.actor) { + tile.actor.send('abortTile', { uid: tile.uid, type: this.type, source: this.id }, undefined); + } + } + unloadTile(tile) { + tile.unloadVectorData(); + if (tile.actor) { + tile.actor.send('removeTile', { uid: tile.uid, type: this.type, source: this.id }, undefined); + } + } + hasTransition() { + return false; + } +} + +/** + * A source containing raster tiles (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.) + * + * @group Sources + * + * @example + * ```ts + * map.addSource('raster-source', { + * 'type': 'raster', + * 'tiles': ['https://tiles.stadiamaps.com/tiles/stamen_watercolor/{z}/{x}/{y}.jpg'], + * 'tileSize': 256, + * }); + * ``` + * + * @example + * ```ts + * map.addSource('wms-test-source', { + * 'type': 'raster', + * // use the tiles option to specify a WMS tile source URL + * 'tiles': [ + * 'https://img.nj.gov/imagerywms/Natural2015?bbox={bbox-epsg-3857}&format=image/png&service=WMS&version=1.1.1&request=GetMap&srs=EPSG:3857&transparent=true&width=256&height=256&layers=Natural2015' + * ], + * 'tileSize': 256 + * }); + * ``` + * @see [Add a raster tile source](https://maplibre.org/maplibre-gl-js/docs/examples/map-tiles/) + * @see [Add a WMS source](https://maplibre.org/maplibre-gl-js/docs/examples/wms/) + * @see [Display a satellite map](https://maplibre.org/maplibre-gl-js/docs/examples/satellite-map/) + */ +class RasterTileSource extends performance.Evented { + constructor(id, options, dispatcher, eventedParent) { + super(); + this.id = id; + this.dispatcher = dispatcher; + this.setEventedParent(eventedParent); + this.type = 'raster'; + this.minzoom = 0; + this.maxzoom = 22; + this.roundZoom = true; + this.scheme = 'xyz'; + this.tileSize = 512; + this._loaded = false; + this._options = performance.extend({ type: 'raster' }, options); + performance.extend(this, performance.pick(options, ['url', 'scheme', 'tileSize'])); + } + load() { + this._loaded = false; + this.fire(new performance.Event('dataloading', { dataType: 'source' })); + this._tileJSONRequest = loadTileJson(this._options, this.map._requestManager, (err, tileJSON) => { + this._tileJSONRequest = null; + this._loaded = true; + if (err) { + this.fire(new performance.ErrorEvent(err)); + } + else if (tileJSON) { + performance.extend(this, tileJSON); + if (tileJSON.bounds) + this.tileBounds = new TileBounds(tileJSON.bounds, this.minzoom, this.maxzoom); + // `content` is included here to prevent a race condition where `Style#_updateSources` is called + // before the TileJSON arrives. this makes sure the tiles needed are loaded once TileJSON arrives + // ref: https://github.com/mapbox/mapbox-gl-js/pull/4347#discussion_r104418088 + this.fire(new performance.Event('data', { dataType: 'source', sourceDataType: 'metadata' })); + this.fire(new performance.Event('data', { dataType: 'source', sourceDataType: 'content' })); + } + }); + } + loaded() { + return this._loaded; + } + onAdd(map) { + this.map = map; + this.load(); + } + onRemove() { + if (this._tileJSONRequest) { + this._tileJSONRequest.cancel(); + this._tileJSONRequest = null; + } + } + setSourceProperty(callback) { + if (this._tileJSONRequest) { + this._tileJSONRequest.cancel(); + } + callback(); + this.load(); + } + /** + * Sets the source `tiles` property and re-renders the map. + * + * @param tiles - An array of one or more tile source URLs, as in the raster tiles spec (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) + * @returns `this` + */ + setTiles(tiles) { + this.setSourceProperty(() => { + this._options.tiles = tiles; + }); + return this; + } + serialize() { + return performance.extend({}, this._options); + } + hasTile(tileID) { + return !this.tileBounds || this.tileBounds.contains(tileID.canonical); + } + loadTile(tile, callback) { + const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme); + tile.request = ImageRequest.getImage(this.map._requestManager.transformRequest(url, ResourceType.Tile), (err, img, expiry) => { + delete tile.request; + if (tile.aborted) { + tile.state = 'unloaded'; + callback(null); + } + else if (err) { + tile.state = 'errored'; + callback(err); + } + else if (img) { + if (this.map._refreshExpiredTiles && expiry) + tile.setExpiryData(expiry); + const context = this.map.painter.context; + const gl = context.gl; + tile.texture = this.map.painter.getTileTexture(img.width); + if (tile.texture) { + tile.texture.update(img, { useMipmap: true }); + } + else { + tile.texture = new Texture(context, img, gl.RGBA, { useMipmap: true }); + tile.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST); + if (context.extTextureFilterAnisotropic) { + gl.texParameterf(gl.TEXTURE_2D, context.extTextureFilterAnisotropic.TEXTURE_MAX_ANISOTROPY_EXT, context.extTextureFilterAnisotropicMax); + } + } + tile.state = 'loaded'; + callback(null); + } + }, this.map._refreshExpiredTiles); + } + abortTile(tile, callback) { + if (tile.request) { + tile.request.cancel(); + delete tile.request; + } + callback(); + } + unloadTile(tile, callback) { + if (tile.texture) + this.map.painter.saveTileTexture(tile.texture); + callback(); + } + hasTransition() { + return false; + } +} + +/** + * A source containing raster DEM tiles (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.) + * This source can be used to show hillshading and 3D terrain + * + * @group Sources + * + * @example + * ```ts + * map.addSource('raster-dem-source', { + * type: 'raster-dem', + * url: 'https://demotiles.maplibre.org/terrain-tiles/tiles.json', + * tileSize: 256 + * }); + * ``` + * @see [3D Terrain](https://maplibre.org/maplibre-gl-js/docs/examples/3d-terrain/) + */ +class RasterDEMTileSource extends RasterTileSource { + constructor(id, options, dispatcher, eventedParent) { + super(id, options, dispatcher, eventedParent); + this.type = 'raster-dem'; + this.maxzoom = 22; + this._options = performance.extend({ type: 'raster-dem' }, options); + this.encoding = options.encoding || 'mapbox'; + this.redFactor = options.redFactor; + this.greenFactor = options.greenFactor; + this.blueFactor = options.blueFactor; + this.baseShift = options.baseShift; + } + loadTile(tile, callback) { + const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme); + const request = this.map._requestManager.transformRequest(url, ResourceType.Tile); + tile.neighboringTiles = this._getNeighboringTiles(tile.tileID); + tile.request = ImageRequest.getImage(request, (err, img, expiry) => performance.__awaiter(this, void 0, void 0, function* () { + delete tile.request; + if (tile.aborted) { + tile.state = 'unloaded'; + callback(null); + } + else if (err) { + tile.state = 'errored'; + callback(err); + } + else if (img) { + if (this.map._refreshExpiredTiles) + tile.setExpiryData(expiry); + const transfer = performance.isImageBitmap(img) && performance.offscreenCanvasSupported(); + const rawImageData = transfer ? img : yield readImageNow(img); + const params = { + uid: tile.uid, + coord: tile.tileID, + source: this.id, + rawImageData, + encoding: this.encoding, + redFactor: this.redFactor, + greenFactor: this.greenFactor, + blueFactor: this.blueFactor, + baseShift: this.baseShift + }; + if (!tile.actor || tile.state === 'expired') { + tile.actor = this.dispatcher.getActor(); + tile.actor.send('loadDEMTile', params, done); + } + } + }), this.map._refreshExpiredTiles); + function readImageNow(img) { + return performance.__awaiter(this, void 0, void 0, function* () { + if (typeof VideoFrame !== 'undefined' && performance.isOffscreenCanvasDistorted()) { + const width = img.width + 2; + const height = img.height + 2; + try { + return new performance.RGBAImage({ width, height }, yield performance.readImageUsingVideoFrame(img, -1, -1, width, height)); + } + catch (e) { + // fall-back to browser canvas decoding + } + } + return performance.browser.getImageData(img, 1); + }); + } + function done(err, data) { + if (err) { + tile.state = 'errored'; + callback(err); + } + if (data) { + tile.dem = data; + tile.needsHillshadePrepare = true; + tile.needsTerrainPrepare = true; + tile.state = 'loaded'; + callback(null); + } + } + } + _getNeighboringTiles(tileID) { + const canonical = tileID.canonical; + const dim = Math.pow(2, canonical.z); + const px = (canonical.x - 1 + dim) % dim; + const pxw = canonical.x === 0 ? tileID.wrap - 1 : tileID.wrap; + const nx = (canonical.x + 1 + dim) % dim; + const nxw = canonical.x + 1 === dim ? tileID.wrap + 1 : tileID.wrap; + const neighboringTiles = {}; + // add adjacent tiles + neighboringTiles[new performance.OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y).key] = { backfilled: false }; + neighboringTiles[new performance.OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y).key] = { backfilled: false }; + // Add upper neighboringTiles + if (canonical.y > 0) { + neighboringTiles[new performance.OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y - 1).key] = { backfilled: false }; + neighboringTiles[new performance.OverscaledTileID(tileID.overscaledZ, tileID.wrap, canonical.z, canonical.x, canonical.y - 1).key] = { backfilled: false }; + neighboringTiles[new performance.OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y - 1).key] = { backfilled: false }; + } + // Add lower neighboringTiles + if (canonical.y + 1 < dim) { + neighboringTiles[new performance.OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y + 1).key] = { backfilled: false }; + neighboringTiles[new performance.OverscaledTileID(tileID.overscaledZ, tileID.wrap, canonical.z, canonical.x, canonical.y + 1).key] = { backfilled: false }; + neighboringTiles[new performance.OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y + 1).key] = { backfilled: false }; + } + return neighboringTiles; + } + unloadTile(tile) { + if (tile.demTexture) + this.map.painter.saveTileTexture(tile.demTexture); + if (tile.fbo) { + tile.fbo.destroy(); + delete tile.fbo; + } + if (tile.dem) + delete tile.dem; + delete tile.neighboringTiles; + tile.state = 'unloaded'; + if (tile.actor) { + tile.actor.send('removeDEMTile', { uid: tile.uid, source: this.id }); + } + } +} + +/** + * A source containing GeoJSON. + * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-geojson) for detailed documentation of options.) + * + * @group Sources + * + * @example + * ```ts + * map.addSource('some id', { + * type: 'geojson', + * data: 'https://d2ad6b4ur7yvpq.cloudfront.net/naturalearth-3.3.0/ne_10m_ports.geojson' + * }); + * ``` + * + * @example + * ```ts + * map.addSource('some id', { + * type: 'geojson', + * data: { + * "type": "FeatureCollection", + * "features": [{ + * "type": "Feature", + * "properties": {}, + * "geometry": { + * "type": "Point", + * "coordinates": [ + * -76.53063297271729, + * 39.18174077994108 + * ] + * } + * }] + * } + * }); + * ``` + * + * @example + * ```ts + * map.getSource('some id').setData({ + * "type": "FeatureCollection", + * "features": [{ + * "type": "Feature", + * "properties": { "name": "Null Island" }, + * "geometry": { + * "type": "Point", + * "coordinates": [ 0, 0 ] + * } + * }] + * }); + * ``` + * @see [Draw GeoJSON points](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-markers/) + * @see [Add a GeoJSON line](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-line/) + * @see [Create a heatmap from points](https://maplibre.org/maplibre-gl-js/docs/examples/heatmap/) + * @see [Create and style clusters](https://maplibre.org/maplibre-gl-js/docs/examples/cluster/) + */ +class GeoJSONSource extends performance.Evented { + /** @internal */ + constructor(id, options, dispatcher, eventedParent) { + super(); + this.load = () => { + this._updateWorkerData(); + }; + this.serialize = () => { + return performance.extend({}, this._options, { + type: this.type, + data: this._data + }); + }; + this.id = id; + // `type` is a property rather than a constant to make it easy for 3rd + // parties to use GeoJSONSource to build their own source types. + this.type = 'geojson'; + this.minzoom = 0; + this.maxzoom = 18; + this.tileSize = 512; + this.isTileClipped = true; + this.reparseOverscaled = true; + this._removed = false; + this._pendingLoads = 0; + this.actor = dispatcher.getActor(); + this.setEventedParent(eventedParent); + this._data = options.data; + this._options = performance.extend({}, options); + this._collectResourceTiming = options.collectResourceTiming; + if (options.maxzoom !== undefined) + this.maxzoom = options.maxzoom; + if (options.type) + this.type = options.type; + if (options.attribution) + this.attribution = options.attribution; + this.promoteId = options.promoteId; + const scale = performance.EXTENT / this.tileSize; + // sent to the worker, along with `url: ...` or `data: literal geojson`, + // so that it can load/parse/index the geojson data + // extending with `options.workerOptions` helps to make it easy for + // third-party sources to hack/reuse GeoJSONSource. + this.workerOptions = performance.extend({ + source: this.id, + cluster: options.cluster || false, + geojsonVtOptions: { + buffer: (options.buffer !== undefined ? options.buffer : 128) * scale, + tolerance: (options.tolerance !== undefined ? options.tolerance : 0.375) * scale, + extent: performance.EXTENT, + maxZoom: this.maxzoom, + lineMetrics: options.lineMetrics || false, + generateId: options.generateId || false + }, + superclusterOptions: { + maxZoom: options.clusterMaxZoom !== undefined ? options.clusterMaxZoom : this.maxzoom - 1, + minPoints: Math.max(2, options.clusterMinPoints || 2), + extent: performance.EXTENT, + radius: (options.clusterRadius || 50) * scale, + log: false, + generateId: options.generateId || false + }, + clusterProperties: options.clusterProperties, + filter: options.filter + }, options.workerOptions); + // send the promoteId to the worker to have more flexible updates, but only if it is a string + if (typeof this.promoteId === 'string') { + this.workerOptions.promoteId = this.promoteId; + } + } + onAdd(map) { + this.map = map; + this.load(); + } + /** + * Sets the GeoJSON data and re-renders the map. + * + * @param data - A GeoJSON data object or a URL to one. The latter is preferable in the case of large GeoJSON files. + * @returns `this` + */ + setData(data) { + this._data = data; + this._updateWorkerData(); + return this; + } + /** + * Updates the source's GeoJSON, and re-renders the map. + * + * For sources with lots of features, this method can be used to make updates more quickly. + * + * This approach requires unique IDs for every feature in the source. The IDs can either be specified on the feature, + * or by using the promoteId option to specify which property should be used as the ID. + * + * It is an error to call updateData on a source that did not have unique IDs for each of its features already. + * + * Updates are applied on a best-effort basis, updating an ID that does not exist will not result in an error. + * + * @param diff - The changes that need to be applied. + * @returns `this` + */ + updateData(diff) { + this._updateWorkerData(diff); + return this; + } + /** + * To disable/enable clustering on the source options + * @param options - The options to set + * @returns `this` + * @example + * ```ts + * map.getSource('some id').setClusterOptions({cluster: false}); + * map.getSource('some id').setClusterOptions({cluster: false, clusterRadius: 50, clusterMaxZoom: 14}); + * ``` + */ + setClusterOptions(options) { + this.workerOptions.cluster = options.cluster; + if (options) { + if (options.clusterRadius !== undefined) + this.workerOptions.superclusterOptions.radius = options.clusterRadius; + if (options.clusterMaxZoom !== undefined) + this.workerOptions.superclusterOptions.maxZoom = options.clusterMaxZoom; + } + this._updateWorkerData(); + return this; + } + /** + * For clustered sources, fetches the zoom at which the given cluster expands. + * + * @param clusterId - The value of the cluster's `cluster_id` property. + * @param callback - A callback to be called when the zoom value is retrieved (`(error, zoom) => { ... }`). + * @returns `this` + */ + getClusterExpansionZoom(clusterId, callback) { + this.actor.send('geojson.getClusterExpansionZoom', { clusterId, source: this.id }, callback); + return this; + } + /** + * For clustered sources, fetches the children of the given cluster on the next zoom level (as an array of GeoJSON features). + * + * @param clusterId - The value of the cluster's `cluster_id` property. + * @param callback - A callback to be called when the features are retrieved (`(error, features) => { ... }`). + * @returns `this` + */ + getClusterChildren(clusterId, callback) { + this.actor.send('geojson.getClusterChildren', { clusterId, source: this.id }, callback); + return this; + } + /** + * For clustered sources, fetches the original points that belong to the cluster (as an array of GeoJSON features). + * + * @param clusterId - The value of the cluster's `cluster_id` property. + * @param limit - The maximum number of features to return. + * @param offset - The number of features to skip (e.g. for pagination). + * @param callback - A callback to be called when the features are retrieved (`(error, features) => { ... }`). + * @returns `this` + * @example + * Retrieve cluster leaves on click + * ```ts + * map.on('click', 'clusters', function(e) { + * let features = map.queryRenderedFeatures(e.point, { + * layers: ['clusters'] + * }); + * + * let clusterId = features[0].properties.cluster_id; + * let pointCount = features[0].properties.point_count; + * let clusterSource = map.getSource('clusters'); + * + * clusterSource.getClusterLeaves(clusterId, pointCount, 0, function(error, features) { + * // Print cluster leaves in the console + * console.log('Cluster leaves:', error, features); + * }) + * }); + * ``` + */ + getClusterLeaves(clusterId, limit, offset, callback) { + this.actor.send('geojson.getClusterLeaves', { + source: this.id, + clusterId, + limit, + offset + }, callback); + return this; + } + /** + * Responsible for invoking WorkerSource's geojson.loadData target, which + * handles loading the geojson data and preparing to serve it up as tiles, + * using geojson-vt or supercluster as appropriate. + * @param diff - the diff object + */ + _updateWorkerData(diff) { + const options = performance.extend({}, this.workerOptions); + if (diff) { + options.dataDiff = diff; + } + else if (typeof this._data === 'string') { + options.request = this.map._requestManager.transformRequest(performance.browser.resolveURL(this._data), ResourceType.Source); + options.request.collectResourceTiming = this._collectResourceTiming; + } + else { + options.data = JSON.stringify(this._data); + } + this._pendingLoads++; + this.fire(new performance.Event('dataloading', { dataType: 'source' })); + // target {this.type}.loadData rather than literally geojson.loadData, + // so that other geojson-like source types can easily reuse this + // implementation + this.actor.send(`${this.type}.loadData`, options, (err, result) => { + this._pendingLoads--; + if (this._removed || (result && result.abandoned)) { + this.fire(new performance.Event('dataabort', { dataType: 'source' })); + return; + } + let resourceTiming = null; + if (result && result.resourceTiming && result.resourceTiming[this.id]) + resourceTiming = result.resourceTiming[this.id].slice(0); + if (err) { + this.fire(new performance.ErrorEvent(err)); + return; + } + const data = { dataType: 'source' }; + if (this._collectResourceTiming && resourceTiming && resourceTiming.length > 0) + performance.extend(data, { resourceTiming }); + // although GeoJSON sources contain no metadata, we fire this event to let the SourceCache + // know its ok to start requesting tiles. + this.fire(new performance.Event('data', Object.assign(Object.assign({}, data), { sourceDataType: 'metadata' }))); + this.fire(new performance.Event('data', Object.assign(Object.assign({}, data), { sourceDataType: 'content' }))); + }); + } + loaded() { + return this._pendingLoads === 0; + } + loadTile(tile, callback) { + const message = !tile.actor ? 'loadTile' : 'reloadTile'; + tile.actor = this.actor; + const params = { + type: this.type, + uid: tile.uid, + tileID: tile.tileID, + zoom: tile.tileID.overscaledZ, + maxZoom: this.maxzoom, + tileSize: this.tileSize, + source: this.id, + pixelRatio: this.map.getPixelRatio(), + showCollisionBoxes: this.map.showCollisionBoxes, + promoteId: this.promoteId + }; + tile.request = this.actor.send(message, params, (err, data) => { + delete tile.request; + tile.unloadVectorData(); + if (tile.aborted) { + return callback(null); + } + if (err) { + return callback(err); + } + tile.loadVectorData(data, this.map.painter, message === 'reloadTile'); + return callback(null); + }); + } + abortTile(tile) { + if (tile.request) { + tile.request.cancel(); + delete tile.request; + } + tile.aborted = true; + } + unloadTile(tile) { + tile.unloadVectorData(); + this.actor.send('removeTile', { uid: tile.uid, type: this.type, source: this.id }); + } + onRemove() { + this._removed = true; + this.actor.send('removeSource', { type: this.type, source: this.id }); + } + hasTransition() { + return false; + } +} + +var rasterBoundsAttributes = performance.createLayout([ + { name: 'a_pos', type: 'Int16', components: 2 }, + { name: 'a_texture_pos', type: 'Int16', components: 2 } +]); + +/** + * A data source containing an image. + * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-image) for detailed documentation of options.) + * + * @group Sources + * + * @example + * ```ts + * // add to map + * map.addSource('some id', { + * type: 'image', + * url: 'https://www.maplibre.org/images/foo.png', + * coordinates: [ + * [-76.54, 39.18], + * [-76.52, 39.18], + * [-76.52, 39.17], + * [-76.54, 39.17] + * ] + * }); + * + * // update coordinates + * let mySource = map.getSource('some id'); + * mySource.setCoordinates([ + * [-76.54335737228394, 39.18579907229748], + * [-76.52803659439087, 39.1838364847587], + * [-76.5295386314392, 39.17683392507606], + * [-76.54520273208618, 39.17876344106642] + * ]); + * + * // update url and coordinates simultaneously + * mySource.updateImage({ + * url: 'https://www.maplibre.org/images/bar.png', + * coordinates: [ + * [-76.54335737228394, 39.18579907229748], + * [-76.52803659439087, 39.1838364847587], + * [-76.5295386314392, 39.17683392507606], + * [-76.54520273208618, 39.17876344106642] + * ] + * }) + * + * map.removeSource('some id'); // remove + * ``` + */ +class ImageSource extends performance.Evented { + /** @internal */ + constructor(id, options, dispatcher, eventedParent) { + super(); + this.load = (newCoordinates, successCallback) => { + this._loaded = false; + this.fire(new performance.Event('dataloading', { dataType: 'source' })); + this.url = this.options.url; + this._request = ImageRequest.getImage(this.map._requestManager.transformRequest(this.url, ResourceType.Image), (err, image) => { + this._request = null; + this._loaded = true; + if (err) { + this.fire(new performance.ErrorEvent(err)); + } + else if (image) { + this.image = image; + if (newCoordinates) { + this.coordinates = newCoordinates; + } + if (successCallback) { + successCallback(); + } + this._finishLoading(); + } + }); + }; + this.prepare = () => { + if (Object.keys(this.tiles).length === 0 || !this.image) { + return; + } + const context = this.map.painter.context; + const gl = context.gl; + if (!this.boundsBuffer) { + this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members); + } + if (!this.boundsSegments) { + this.boundsSegments = performance.SegmentVector.simpleSegment(0, 0, 4, 2); + } + if (!this.texture) { + this.texture = new Texture(context, this.image, gl.RGBA); + this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + } + let newTilesLoaded = false; + for (const w in this.tiles) { + const tile = this.tiles[w]; + if (tile.state !== 'loaded') { + tile.state = 'loaded'; + tile.texture = this.texture; + newTilesLoaded = true; + } + } + if (newTilesLoaded) { + this.fire(new performance.Event('data', { dataType: 'source', sourceDataType: 'idle', sourceId: this.id })); + } + }; + this.serialize = () => { + return { + type: 'image', + url: this.options.url, + coordinates: this.coordinates + }; + }; + this.id = id; + this.dispatcher = dispatcher; + this.coordinates = options.coordinates; + this.type = 'image'; + this.minzoom = 0; + this.maxzoom = 22; + this.tileSize = 512; + this.tiles = {}; + this._loaded = false; + this.setEventedParent(eventedParent); + this.options = options; + } + loaded() { + return this._loaded; + } + /** + * Updates the image URL and, optionally, the coordinates. To avoid having the image flash after changing, + * set the `raster-fade-duration` paint property on the raster layer to 0. + * + * @param options - The options object. + * @returns `this` + */ + updateImage(options) { + if (!options.url) { + return this; + } + if (this._request) { + this._request.cancel(); + this._request = null; + } + this.options.url = options.url; + this.load(options.coordinates, () => { this.texture = null; }); + return this; + } + _finishLoading() { + if (this.map) { + this.setCoordinates(this.coordinates); + this.fire(new performance.Event('data', { dataType: 'source', sourceDataType: 'metadata' })); + } + } + onAdd(map) { + this.map = map; + this.load(); + } + onRemove() { + if (this._request) { + this._request.cancel(); + this._request = null; + } + } + /** + * Sets the image's coordinates and re-renders the map. + * + * @param coordinates - Four geographical coordinates, + * represented as arrays of longitude and latitude numbers, which define the corners of the image. + * The coordinates start at the top left corner of the image and proceed in clockwise order. + * They do not have to represent a rectangle. + * @returns `this` + */ + setCoordinates(coordinates) { + this.coordinates = coordinates; + // Calculate which mercator tile is suitable for rendering the video in + // and create a buffer with the corner coordinates. These coordinates + // may be outside the tile, because raster tiles aren't clipped when rendering. + // transform the geo coordinates into (zoom 0) tile space coordinates + const cornerCoords = coordinates.map(performance.MercatorCoordinate.fromLngLat); + // Compute the coordinates of the tile we'll use to hold this image's + // render data + this.tileID = getCoordinatesCenterTileID(cornerCoords); + // Constrain min/max zoom to our tile's zoom level in order to force + // SourceCache to request this tile (no matter what the map's zoom + // level) + this.minzoom = this.maxzoom = this.tileID.z; + // Transform the corner coordinates into the coordinate space of our + // tile. + const tileCoords = cornerCoords.map((coord) => this.tileID.getTilePoint(coord)._round()); + this._boundsArray = new performance.RasterBoundsArray(); + this._boundsArray.emplaceBack(tileCoords[0].x, tileCoords[0].y, 0, 0); + this._boundsArray.emplaceBack(tileCoords[1].x, tileCoords[1].y, performance.EXTENT, 0); + this._boundsArray.emplaceBack(tileCoords[3].x, tileCoords[3].y, 0, performance.EXTENT); + this._boundsArray.emplaceBack(tileCoords[2].x, tileCoords[2].y, performance.EXTENT, performance.EXTENT); + if (this.boundsBuffer) { + this.boundsBuffer.destroy(); + delete this.boundsBuffer; + } + this.fire(new performance.Event('data', { dataType: 'source', sourceDataType: 'content' })); + return this; + } + loadTile(tile, callback) { + // We have a single tile -- whose coordinates are this.tileID -- that + // covers the image we want to render. If that's the one being + // requested, set it up with the image; otherwise, mark the tile as + // `errored` to indicate that we have no data for it. + // If the world wraps, we may have multiple "wrapped" copies of the + // single tile. + if (this.tileID && this.tileID.equals(tile.tileID.canonical)) { + this.tiles[String(tile.tileID.wrap)] = tile; + tile.buckets = {}; + callback(null); + } + else { + tile.state = 'errored'; + callback(null); + } + } + hasTransition() { + return false; + } +} +/** + * Given a list of coordinates, get their center as a coordinate. + * + * @returns centerpoint + * @internal + */ +function getCoordinatesCenterTileID(coords) { + let minX = Infinity; + let minY = Infinity; + let maxX = -Infinity; + let maxY = -Infinity; + for (const coord of coords) { + minX = Math.min(minX, coord.x); + minY = Math.min(minY, coord.y); + maxX = Math.max(maxX, coord.x); + maxY = Math.max(maxY, coord.y); + } + const dx = maxX - minX; + const dy = maxY - minY; + const dMax = Math.max(dx, dy); + const zoom = Math.max(0, Math.floor(-Math.log(dMax) / Math.LN2)); + const tilesAtZoom = Math.pow(2, zoom); + return new performance.CanonicalTileID(zoom, Math.floor((minX + maxX) / 2 * tilesAtZoom), Math.floor((minY + maxY) / 2 * tilesAtZoom)); +} + +/** + * A data source containing video. + * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-video) for detailed documentation of options.) + * + * @group Sources + * + * @example + * ```ts + * // add to map + * map.addSource('some id', { + * type: 'video', + * url: [ + * 'https://www.mapbox.com/blog/assets/baltimore-smoke.mp4', + * 'https://www.mapbox.com/blog/assets/baltimore-smoke.webm' + * ], + * coordinates: [ + * [-76.54, 39.18], + * [-76.52, 39.18], + * [-76.52, 39.17], + * [-76.54, 39.17] + * ] + * }); + * + * // update + * let mySource = map.getSource('some id'); + * mySource.setCoordinates([ + * [-76.54335737228394, 39.18579907229748], + * [-76.52803659439087, 39.1838364847587], + * [-76.5295386314392, 39.17683392507606], + * [-76.54520273208618, 39.17876344106642] + * ]); + * + * map.removeSource('some id'); // remove + * ``` + * @see [Add a video](https://maplibre.org/maplibre-gl-js/docs/examples/video-on-a-map/) + * + * Note that when rendered as a raster layer, the layer's `raster-fade-duration` property will cause the video to fade in. + * This happens when playback is started, paused and resumed, or when the video's coordinates are updated. To avoid this behavior, + * set the layer's `raster-fade-duration` property to `0`. + */ +class VideoSource extends ImageSource { + constructor(id, options, dispatcher, eventedParent) { + super(id, options, dispatcher, eventedParent); + this.load = () => { + this._loaded = false; + const options = this.options; + this.urls = []; + for (const url of options.urls) { + this.urls.push(this.map._requestManager.transformRequest(url, ResourceType.Source).url); + } + performance.getVideo(this.urls, (err, video) => { + this._loaded = true; + if (err) { + this.fire(new performance.ErrorEvent(err)); + } + else if (video) { + this.video = video; + this.video.loop = true; + // Start repainting when video starts playing. hasTransition() will then return + // true to trigger additional frames as long as the videos continues playing. + this.video.addEventListener('playing', () => { + this.map.triggerRepaint(); + }); + if (this.map) { + this.video.play(); + } + this._finishLoading(); + } + }); + }; + /** + * Sets the video's coordinates and re-renders the map. + * + * @returns `this` + */ + this.prepare = () => { + if (Object.keys(this.tiles).length === 0 || this.video.readyState < 2) { + return; // not enough data for current position + } + const context = this.map.painter.context; + const gl = context.gl; + if (!this.boundsBuffer) { + this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members); + } + if (!this.boundsSegments) { + this.boundsSegments = performance.SegmentVector.simpleSegment(0, 0, 4, 2); + } + if (!this.texture) { + this.texture = new Texture(context, this.video, gl.RGBA); + this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + } + else if (!this.video.paused) { + this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, gl.RGBA, gl.UNSIGNED_BYTE, this.video); + } + let newTilesLoaded = false; + for (const w in this.tiles) { + const tile = this.tiles[w]; + if (tile.state !== 'loaded') { + tile.state = 'loaded'; + tile.texture = this.texture; + newTilesLoaded = true; + } + } + if (newTilesLoaded) { + this.fire(new performance.Event('data', { dataType: 'source', sourceDataType: 'idle', sourceId: this.id })); + } + }; + this.serialize = () => { + return { + type: 'video', + urls: this.urls, + coordinates: this.coordinates + }; + }; + this.roundZoom = true; + this.type = 'video'; + this.options = options; + } + /** + * Pauses the video. + */ + pause() { + if (this.video) { + this.video.pause(); + } + } + /** + * Plays the video. + */ + play() { + if (this.video) { + this.video.play(); + } + } + /** + * Sets playback to a timestamp, in seconds. + */ + seek(seconds) { + if (this.video) { + const seekableRange = this.video.seekable; + if (seconds < seekableRange.start(0) || seconds > seekableRange.end(0)) { + this.fire(new performance.ErrorEvent(new performance.ValidationError(`sources.${this.id}`, null, `Playback for this video can be set only between the ${seekableRange.start(0)} and ${seekableRange.end(0)}-second mark.`))); + } + else + this.video.currentTime = seconds; + } + } + /** + * Returns the HTML `video` element. + * + * @returns The HTML `video` element. + */ + getVideo() { + return this.video; + } + onAdd(map) { + if (this.map) + return; + this.map = map; + this.load(); + if (this.video) { + this.video.play(); + this.setCoordinates(this.coordinates); + } + } + hasTransition() { + return this.video && !this.video.paused; + } +} + +/** + * A data source containing the contents of an HTML canvas. See {@link CanvasSourceSpecification} for detailed documentation of options. + * + * @group Sources + * + * @example + * ```ts + * // add to map + * map.addSource('some id', { + * type: 'canvas', + * canvas: 'idOfMyHTMLCanvas', + * animate: true, + * coordinates: [ + * [-76.54, 39.18], + * [-76.52, 39.18], + * [-76.52, 39.17], + * [-76.54, 39.17] + * ] + * }); + * + * // update + * let mySource = map.getSource('some id'); + * mySource.setCoordinates([ + * [-76.54335737228394, 39.18579907229748], + * [-76.52803659439087, 39.1838364847587], + * [-76.5295386314392, 39.17683392507606], + * [-76.54520273208618, 39.17876344106642] + * ]); + * + * map.removeSource('some id'); // remove + * ``` + */ +class CanvasSource extends ImageSource { + /** @internal */ + constructor(id, options, dispatcher, eventedParent) { + super(id, options, dispatcher, eventedParent); + this.load = () => { + this._loaded = true; + if (!this.canvas) { + this.canvas = (this.options.canvas instanceof HTMLCanvasElement) ? + this.options.canvas : + document.getElementById(this.options.canvas); + // cast to HTMLCanvasElement in else of ternary + // should we do a safety check and throw if it's not actually HTMLCanvasElement? + } + this.width = this.canvas.width; + this.height = this.canvas.height; + if (this._hasInvalidDimensions()) { + this.fire(new performance.ErrorEvent(new Error('Canvas dimensions cannot be less than or equal to zero.'))); + return; + } + this.play = function () { + this._playing = true; + this.map.triggerRepaint(); + }; + this.pause = function () { + if (this._playing) { + this.prepare(); + this._playing = false; + } + }; + this._finishLoading(); + }; + this.prepare = () => { + let resize = false; + if (this.canvas.width !== this.width) { + this.width = this.canvas.width; + resize = true; + } + if (this.canvas.height !== this.height) { + this.height = this.canvas.height; + resize = true; + } + if (this._hasInvalidDimensions()) + return; + if (Object.keys(this.tiles).length === 0) + return; // not enough data for current position + const context = this.map.painter.context; + const gl = context.gl; + if (!this.boundsBuffer) { + this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members); + } + if (!this.boundsSegments) { + this.boundsSegments = performance.SegmentVector.simpleSegment(0, 0, 4, 2); + } + if (!this.texture) { + this.texture = new Texture(context, this.canvas, gl.RGBA, { premultiply: true }); + } + else if (resize || this._playing) { + this.texture.update(this.canvas, { premultiply: true }); + } + let newTilesLoaded = false; + for (const w in this.tiles) { + const tile = this.tiles[w]; + if (tile.state !== 'loaded') { + tile.state = 'loaded'; + tile.texture = this.texture; + newTilesLoaded = true; + } + } + if (newTilesLoaded) { + this.fire(new performance.Event('data', { dataType: 'source', sourceDataType: 'idle', sourceId: this.id })); + } + }; + this.serialize = () => { + return { + type: 'canvas', + coordinates: this.coordinates + }; + }; + // We build in some validation here, since canvas sources aren't included in the style spec: + if (!options.coordinates) { + this.fire(new performance.ErrorEvent(new performance.ValidationError(`sources.${id}`, null, 'missing required property "coordinates"'))); + } + else if (!Array.isArray(options.coordinates) || options.coordinates.length !== 4 || + options.coordinates.some(c => !Array.isArray(c) || c.length !== 2 || c.some(l => typeof l !== 'number'))) { + this.fire(new performance.ErrorEvent(new performance.ValidationError(`sources.${id}`, null, '"coordinates" property must be an array of 4 longitude/latitude array pairs'))); + } + if (options.animate && typeof options.animate !== 'boolean') { + this.fire(new performance.ErrorEvent(new performance.ValidationError(`sources.${id}`, null, 'optional "animate" property must be a boolean value'))); + } + if (!options.canvas) { + this.fire(new performance.ErrorEvent(new performance.ValidationError(`sources.${id}`, null, 'missing required property "canvas"'))); + } + else if (typeof options.canvas !== 'string' && !(options.canvas instanceof HTMLCanvasElement)) { + this.fire(new performance.ErrorEvent(new performance.ValidationError(`sources.${id}`, null, '"canvas" must be either a string representing the ID of the canvas element from which to read, or an HTMLCanvasElement instance'))); + } + this.options = options; + this.animate = options.animate !== undefined ? options.animate : true; + } + /** + * Returns the HTML `canvas` element. + * + * @returns The HTML `canvas` element. + */ + getCanvas() { + return this.canvas; + } + onAdd(map) { + this.map = map; + this.load(); + if (this.canvas) { + if (this.animate) + this.play(); + } + } + onRemove() { + this.pause(); + } + hasTransition() { + return this._playing; + } + _hasInvalidDimensions() { + for (const x of [this.canvas.width, this.canvas.height]) { + if (isNaN(x) || x <= 0) + return true; + } + return false; + } +} + +const registeredSources = {}; +/** + * Creates a tiled data source instance given an options object. + * + * @param id - The id for the source. Must not be used by any existing source. + * @param specification - Source options, specific to the source type (except for `options.type`, which is always required). + * @param source - A source definition object compliant with + * [`maplibre-gl-style-spec`](https://maplibre.org/maplibre-style-spec/#sources) or, for a third-party source type, + * with that type's requirements. + * @param dispatcher - A {@link Dispatcher} instance, which can be used to send messages to the workers. + * @returns a newly created source + */ +const create = (id, specification, dispatcher, eventedParent) => { + const Class = getSourceType(specification.type); + const source = new Class(id, specification, dispatcher, eventedParent); + if (source.id !== id) { + throw new Error(`Expected Source id to be ${id} instead of ${source.id}`); + } + return source; +}; +const getSourceType = (name) => { + switch (name) { + case 'geojson': + return GeoJSONSource; + case 'image': + return ImageSource; + case 'raster': + return RasterTileSource; + case 'raster-dem': + return RasterDEMTileSource; + case 'vector': + return VectorTileSource; + case 'video': + return VideoSource; + case 'canvas': + return CanvasSource; + } + return registeredSources[name]; +}; +const setSourceType = (name, type) => { + registeredSources[name] = type; +}; + +/* + * Returns a matrix that can be used to convert from tile coordinates to viewport pixel coordinates. + */ +function getPixelPosMatrix(transform, tileID) { + const t = performance.create(); + performance.translate(t, t, [1, 1, 0]); + performance.scale(t, t, [transform.width * 0.5, transform.height * 0.5, 1]); + return performance.multiply(t, t, transform.calculatePosMatrix(tileID.toUnwrapped())); +} +function queryIncludes3DLayer(layers, styleLayers, sourceID) { + if (layers) { + for (const layerID of layers) { + const layer = styleLayers[layerID]; + if (layer && layer.source === sourceID && layer.type === 'fill-extrusion') { + return true; + } + } + } + else { + for (const key in styleLayers) { + const layer = styleLayers[key]; + if (layer.source === sourceID && layer.type === 'fill-extrusion') { + return true; + } + } + } + return false; +} +function queryRenderedFeatures(sourceCache, styleLayers, serializedLayers, queryGeometry, params, transform) { + const has3DLayer = queryIncludes3DLayer(params && params.layers, styleLayers, sourceCache.id); + const maxPitchScaleFactor = transform.maxPitchScaleFactor(); + const tilesIn = sourceCache.tilesIn(queryGeometry, maxPitchScaleFactor, has3DLayer); + tilesIn.sort(sortTilesIn); + const renderedFeatureLayers = []; + for (const tileIn of tilesIn) { + renderedFeatureLayers.push({ + wrappedTileID: tileIn.tileID.wrapped().key, + queryResults: tileIn.tile.queryRenderedFeatures(styleLayers, serializedLayers, sourceCache._state, tileIn.queryGeometry, tileIn.cameraQueryGeometry, tileIn.scale, params, transform, maxPitchScaleFactor, getPixelPosMatrix(sourceCache.transform, tileIn.tileID)) + }); + } + const result = mergeRenderedFeatureLayers(renderedFeatureLayers); + // Merge state from SourceCache into the results + for (const layerID in result) { + result[layerID].forEach((featureWrapper) => { + const feature = featureWrapper.feature; + const state = sourceCache.getFeatureState(feature.layer['source-layer'], feature.id); + feature.source = feature.layer.source; + if (feature.layer['source-layer']) { + feature.sourceLayer = feature.layer['source-layer']; + } + feature.state = state; + }); + } + return result; +} +function queryRenderedSymbols(styleLayers, serializedLayers, sourceCaches, queryGeometry, params, collisionIndex, retainedQueryData) { + const result = {}; + const renderedSymbols = collisionIndex.queryRenderedSymbols(queryGeometry); + const bucketQueryData = []; + for (const bucketInstanceId of Object.keys(renderedSymbols).map(Number)) { + bucketQueryData.push(retainedQueryData[bucketInstanceId]); + } + bucketQueryData.sort(sortTilesIn); + for (const queryData of bucketQueryData) { + const bucketSymbols = queryData.featureIndex.lookupSymbolFeatures(renderedSymbols[queryData.bucketInstanceId], serializedLayers, queryData.bucketIndex, queryData.sourceLayerIndex, params.filter, params.layers, params.availableImages, styleLayers); + for (const layerID in bucketSymbols) { + const resultFeatures = result[layerID] = result[layerID] || []; + const layerSymbols = bucketSymbols[layerID]; + layerSymbols.sort((a, b) => { + // Match topDownFeatureComparator from FeatureIndex, but using + // most recent sorting of features from bucket.sortFeatures + const featureSortOrder = queryData.featureSortOrder; + if (featureSortOrder) { + // queryRenderedSymbols documentation says we'll return features in + // "top-to-bottom" rendering order (aka last-to-first). + // Actually there can be multiple symbol instances per feature, so + // we sort each feature based on the first matching symbol instance. + const sortedA = featureSortOrder.indexOf(a.featureIndex); + const sortedB = featureSortOrder.indexOf(b.featureIndex); + return sortedB - sortedA; + } + else { + // Bucket hasn't been re-sorted based on angle, so use the + // reverse of the order the features appeared in the data. + return b.featureIndex - a.featureIndex; + } + }); + for (const symbolFeature of layerSymbols) { + resultFeatures.push(symbolFeature); + } + } + } + // Merge state from SourceCache into the results + for (const layerName in result) { + result[layerName].forEach((featureWrapper) => { + const feature = featureWrapper.feature; + const layer = styleLayers[layerName]; + const sourceCache = sourceCaches[layer.source]; + const state = sourceCache.getFeatureState(feature.layer['source-layer'], feature.id); + feature.source = feature.layer.source; + if (feature.layer['source-layer']) { + feature.sourceLayer = feature.layer['source-layer']; + } + feature.state = state; + }); + } + return result; +} +function querySourceFeatures(sourceCache, params) { + const tiles = sourceCache.getRenderableIds().map((id) => { + return sourceCache.getTileByID(id); + }); + const result = []; + const dataTiles = {}; + for (let i = 0; i < tiles.length; i++) { + const tile = tiles[i]; + const dataID = tile.tileID.canonical.key; + if (!dataTiles[dataID]) { + dataTiles[dataID] = true; + tile.querySourceFeatures(result, params); + } + } + return result; +} +function sortTilesIn(a, b) { + const idA = a.tileID; + const idB = b.tileID; + return (idA.overscaledZ - idB.overscaledZ) || (idA.canonical.y - idB.canonical.y) || (idA.wrap - idB.wrap) || (idA.canonical.x - idB.canonical.x); +} +function mergeRenderedFeatureLayers(tiles) { + // Merge results from all tiles, but if two tiles share the same + // wrapped ID, don't duplicate features between the two tiles + const result = {}; + const wrappedIDLayerMap = {}; + for (const tile of tiles) { + const queryResults = tile.queryResults; + const wrappedID = tile.wrappedTileID; + const wrappedIDLayers = wrappedIDLayerMap[wrappedID] = wrappedIDLayerMap[wrappedID] || {}; + for (const layerID in queryResults) { + const tileFeatures = queryResults[layerID]; + const wrappedIDFeatures = wrappedIDLayers[layerID] = wrappedIDLayers[layerID] || {}; + const resultFeatures = result[layerID] = result[layerID] || []; + for (const tileFeature of tileFeatures) { + if (!wrappedIDFeatures[tileFeature.featureIndex]) { + wrappedIDFeatures[tileFeature.featureIndex] = true; + resultFeatures.push(tileFeature); + } + } + } + } + return result; +} + +function deserialize(input, style) { + const output = {}; + // Guard against the case where the map's style has been set to null while + // this bucket has been parsing. + if (!style) + return output; + for (const bucket of input) { + const layers = bucket.layerIds + .map((id) => style.getLayer(id)) + .filter(Boolean); + if (layers.length === 0) { + continue; + } + // look up StyleLayer objects from layer ids (since we don't + // want to waste time serializing/copying them from the worker) + bucket.layers = layers; + if (bucket.stateDependentLayerIds) { + bucket.stateDependentLayers = bucket.stateDependentLayerIds.map((lId) => layers.filter((l) => l.id === lId)[0]); + } + for (const layer of layers) { + output[layer.id] = bucket; + } + } + return output; +} + +const CLOCK_SKEW_RETRY_TIMEOUT = 30000; +/** + * @internal + * A tile object is the combination of a Coordinate, which defines + * its place, as well as a unique ID and data tracking for its content + */ +class Tile { + /** + * @param tileID - the tile ID + * @param size - The tile size + */ + constructor(tileID, size) { + this.timeAdded = 0; + this.fadeEndTime = 0; + this.tileID = tileID; + this.uid = performance.uniqueId(); + this.uses = 0; + this.tileSize = size; + this.buckets = {}; + this.expirationTime = null; + this.queryPadding = 0; + this.hasSymbolBuckets = false; + this.hasRTLText = false; + this.dependencies = {}; + this.rtt = []; + this.rttCoords = {}; + // Counts the number of times a response was already expired when + // received. We're using this to add a delay when making a new request + // so we don't have to keep retrying immediately in case of a server + // serving expired tiles. + this.expiredRequestCount = 0; + this.state = 'loading'; + } + registerFadeDuration(duration) { + const fadeEndTime = duration + this.timeAdded; + if (fadeEndTime < this.fadeEndTime) { + return; + } + this.fadeEndTime = fadeEndTime; + } + wasRequested() { + return this.state === 'errored' || this.state === 'loaded' || this.state === 'reloading'; + } + clearTextures(painter) { + if (this.demTexture) + painter.saveTileTexture(this.demTexture); + this.demTexture = null; + } + /** + * Given a data object with a 'buffers' property, load it into + * this tile's elementGroups and buffers properties and set loaded + * to true. If the data is null, like in the case of an empty + * GeoJSON tile, no-op but still set loaded to true. + * @param data - The data from the worker + * @param painter - the painter + * @param justReloaded - `true` to just reload + */ + loadVectorData(data, painter, justReloaded) { + if (this.hasData()) { + this.unloadVectorData(); + } + this.state = 'loaded'; + // empty GeoJSON tile + if (!data) { + this.collisionBoxArray = new performance.CollisionBoxArray(); + return; + } + if (data.featureIndex) { + this.latestFeatureIndex = data.featureIndex; + if (data.rawTileData) { + // Only vector tiles have rawTileData, and they won't update it for + // 'reloadTile' + this.latestRawTileData = data.rawTileData; + this.latestFeatureIndex.rawTileData = data.rawTileData; + } + else if (this.latestRawTileData) { + // If rawTileData hasn't updated, hold onto a pointer to the last + // one we received + this.latestFeatureIndex.rawTileData = this.latestRawTileData; + } + } + this.collisionBoxArray = data.collisionBoxArray; + this.buckets = deserialize(data.buckets, painter.style); + this.hasSymbolBuckets = false; + for (const id in this.buckets) { + const bucket = this.buckets[id]; + if (bucket instanceof performance.SymbolBucket) { + this.hasSymbolBuckets = true; + if (justReloaded) { + bucket.justReloaded = true; + } + else { + break; + } + } + } + this.hasRTLText = false; + if (this.hasSymbolBuckets) { + for (const id in this.buckets) { + const bucket = this.buckets[id]; + if (bucket instanceof performance.SymbolBucket) { + if (bucket.hasRTLText) { + this.hasRTLText = true; + performance.lazyLoadRTLTextPlugin(); + break; + } + } + } + } + this.queryPadding = 0; + for (const id in this.buckets) { + const bucket = this.buckets[id]; + this.queryPadding = Math.max(this.queryPadding, painter.style.getLayer(id).queryRadius(bucket)); + } + if (data.imageAtlas) { + this.imageAtlas = data.imageAtlas; + } + if (data.glyphAtlasImage) { + this.glyphAtlasImage = data.glyphAtlasImage; + } + } + /** + * Release any data or WebGL resources referenced by this tile. + */ + unloadVectorData() { + for (const id in this.buckets) { + this.buckets[id].destroy(); + } + this.buckets = {}; + if (this.imageAtlasTexture) { + this.imageAtlasTexture.destroy(); + } + if (this.imageAtlas) { + this.imageAtlas = null; + } + if (this.glyphAtlasTexture) { + this.glyphAtlasTexture.destroy(); + } + this.latestFeatureIndex = null; + this.state = 'unloaded'; + } + getBucket(layer) { + return this.buckets[layer.id]; + } + upload(context) { + for (const id in this.buckets) { + const bucket = this.buckets[id]; + if (bucket.uploadPending()) { + bucket.upload(context); + } + } + const gl = context.gl; + if (this.imageAtlas && !this.imageAtlas.uploaded) { + this.imageAtlasTexture = new Texture(context, this.imageAtlas.image, gl.RGBA); + this.imageAtlas.uploaded = true; + } + if (this.glyphAtlasImage) { + this.glyphAtlasTexture = new Texture(context, this.glyphAtlasImage, gl.ALPHA); + this.glyphAtlasImage = null; + } + } + prepare(imageManager) { + if (this.imageAtlas) { + this.imageAtlas.patchUpdatedImages(imageManager, this.imageAtlasTexture); + } + } + // Queries non-symbol features rendered for this tile. + // Symbol features are queried globally + queryRenderedFeatures(layers, serializedLayers, sourceFeatureState, queryGeometry, cameraQueryGeometry, scale, params, transform, maxPitchScaleFactor, pixelPosMatrix) { + if (!this.latestFeatureIndex || !this.latestFeatureIndex.rawTileData) + return {}; + return this.latestFeatureIndex.query({ + queryGeometry, + cameraQueryGeometry, + scale, + tileSize: this.tileSize, + pixelPosMatrix, + transform, + params, + queryPadding: this.queryPadding * maxPitchScaleFactor + }, layers, serializedLayers, sourceFeatureState); + } + querySourceFeatures(result, params) { + const featureIndex = this.latestFeatureIndex; + if (!featureIndex || !featureIndex.rawTileData) + return; + const vtLayers = featureIndex.loadVTLayers(); + const sourceLayer = params && params.sourceLayer ? params.sourceLayer : ''; + const layer = vtLayers._geojsonTileLayer || vtLayers[sourceLayer]; + if (!layer) + return; + const filter = performance.createFilter(params && params.filter); + const { z, x, y } = this.tileID.canonical; + const coord = { z, x, y }; + for (let i = 0; i < layer.length; i++) { + const feature = layer.feature(i); + if (filter.needGeometry) { + const evaluationFeature = performance.toEvaluationFeature(feature, true); + if (!filter.filter(new performance.EvaluationParameters(this.tileID.overscaledZ), evaluationFeature, this.tileID.canonical)) + continue; + } + else if (!filter.filter(new performance.EvaluationParameters(this.tileID.overscaledZ), feature)) { + continue; + } + const id = featureIndex.getId(feature, sourceLayer); + const geojsonFeature = new performance.GeoJSONFeature(feature, z, x, y, id); + geojsonFeature.tile = coord; + result.push(geojsonFeature); + } + } + hasData() { + return this.state === 'loaded' || this.state === 'reloading' || this.state === 'expired'; + } + patternsLoaded() { + return this.imageAtlas && !!Object.keys(this.imageAtlas.patternPositions).length; + } + setExpiryData(data) { + const prior = this.expirationTime; + if (data.cacheControl) { + const parsedCC = performance.parseCacheControl(data.cacheControl); + if (parsedCC['max-age']) + this.expirationTime = Date.now() + parsedCC['max-age'] * 1000; + } + else if (data.expires) { + this.expirationTime = new Date(data.expires).getTime(); + } + if (this.expirationTime) { + const now = Date.now(); + let isExpired = false; + if (this.expirationTime > now) { + isExpired = false; + } + else if (!prior) { + isExpired = true; + } + else if (this.expirationTime < prior) { + // Expiring date is going backwards: + // fall back to exponential backoff + isExpired = true; + } + else { + const delta = this.expirationTime - prior; + if (!delta) { + // Server is serving the same expired resource over and over: fall + // back to exponential backoff. + isExpired = true; + } + else { + // Assume that either the client or the server clock is wrong and + // try to interpolate a valid expiration date (from the client POV) + // observing a minimum timeout. + this.expirationTime = now + Math.max(delta, CLOCK_SKEW_RETRY_TIMEOUT); + } + } + if (isExpired) { + this.expiredRequestCount++; + this.state = 'expired'; + } + else { + this.expiredRequestCount = 0; + } + } + } + getExpiryTimeout() { + if (this.expirationTime) { + if (this.expiredRequestCount) { + return 1000 * (1 << Math.min(this.expiredRequestCount - 1, 31)); + } + else { + // Max value for `setTimeout` implementations is a 32 bit integer; cap this accordingly + return Math.min(this.expirationTime - new Date().getTime(), Math.pow(2, 31) - 1); + } + } + } + setFeatureState(states, painter) { + if (!this.latestFeatureIndex || + !this.latestFeatureIndex.rawTileData || + Object.keys(states).length === 0) { + return; + } + const vtLayers = this.latestFeatureIndex.loadVTLayers(); + for (const id in this.buckets) { + if (!painter.style.hasLayer(id)) + continue; + const bucket = this.buckets[id]; + // Buckets are grouped by common source-layer + const sourceLayerId = bucket.layers[0]['sourceLayer'] || '_geojsonTileLayer'; + const sourceLayer = vtLayers[sourceLayerId]; + const sourceLayerStates = states[sourceLayerId]; + if (!sourceLayer || !sourceLayerStates || Object.keys(sourceLayerStates).length === 0) + continue; + bucket.update(sourceLayerStates, sourceLayer, this.imageAtlas && this.imageAtlas.patternPositions || {}); + const layer = painter && painter.style && painter.style.getLayer(id); + if (layer) { + this.queryPadding = Math.max(this.queryPadding, layer.queryRadius(bucket)); + } + } + } + holdingForFade() { + return this.symbolFadeHoldUntil !== undefined; + } + symbolFadeFinished() { + return !this.symbolFadeHoldUntil || this.symbolFadeHoldUntil < performance.browser.now(); + } + clearFadeHold() { + this.symbolFadeHoldUntil = undefined; + } + setHoldDuration(duration) { + this.symbolFadeHoldUntil = performance.browser.now() + duration; + } + setDependencies(namespace, dependencies) { + const index = {}; + for (const dep of dependencies) { + index[dep] = true; + } + this.dependencies[namespace] = index; + } + hasDependency(namespaces, keys) { + for (const namespace of namespaces) { + const dependencies = this.dependencies[namespace]; + if (dependencies) { + for (const key of keys) { + if (dependencies[key]) { + return true; + } + } + } + } + return false; + } +} + +/** + * @internal + * A [least-recently-used cache](http://en.wikipedia.org/wiki/Cache_algorithms) + * with hash lookup made possible by keeping a list of keys in parallel to + * an array of dictionary of values + */ +class TileCache { + /** + * @param max - number of permitted values + * @param onRemove - callback called with items when they expire + */ + constructor(max, onRemove) { + this.max = max; + this.onRemove = onRemove; + this.reset(); + } + /** + * Clear the cache + * + * @returns this cache + */ + reset() { + for (const key in this.data) { + for (const removedData of this.data[key]) { + if (removedData.timeout) + clearTimeout(removedData.timeout); + this.onRemove(removedData.value); + } + } + this.data = {}; + this.order = []; + return this; + } + /** + * Add a key, value combination to the cache, trimming its size if this pushes + * it over max length. + * + * @param tileID - lookup key for the item + * @param data - tile data + * + * @returns this cache + */ + add(tileID, data, expiryTimeout) { + const key = tileID.wrapped().key; + if (this.data[key] === undefined) { + this.data[key] = []; + } + const dataWrapper = { + value: data, + timeout: undefined + }; + if (expiryTimeout !== undefined) { + dataWrapper.timeout = setTimeout(() => { + this.remove(tileID, dataWrapper); + }, expiryTimeout); + } + this.data[key].push(dataWrapper); + this.order.push(key); + if (this.order.length > this.max) { + const removedData = this._getAndRemoveByKey(this.order[0]); + if (removedData) + this.onRemove(removedData); + } + return this; + } + /** + * Determine whether the value attached to `key` is present + * + * @param tileID - the key to be looked-up + * @returns whether the cache has this value + */ + has(tileID) { + return tileID.wrapped().key in this.data; + } + /** + * Get the value attached to a specific key and remove data from cache. + * If the key is not found, returns `null` + * + * @param tileID - the key to look up + * @returns the tile data, or null if it isn't found + */ + getAndRemove(tileID) { + if (!this.has(tileID)) { + return null; + } + return this._getAndRemoveByKey(tileID.wrapped().key); + } + /* + * Get and remove the value with the specified key. + */ + _getAndRemoveByKey(key) { + const data = this.data[key].shift(); + if (data.timeout) + clearTimeout(data.timeout); + if (this.data[key].length === 0) { + delete this.data[key]; + } + this.order.splice(this.order.indexOf(key), 1); + return data.value; + } + /* + * Get the value with the specified (wrapped tile) key. + */ + getByKey(key) { + const data = this.data[key]; + return data ? data[0].value : null; + } + /** + * Get the value attached to a specific key without removing data + * from the cache. If the key is not found, returns `null` + * + * @param tileID - the key to look up + * @returns the tile data, or null if it isn't found + */ + get(tileID) { + if (!this.has(tileID)) { + return null; + } + const data = this.data[tileID.wrapped().key][0]; + return data.value; + } + /** + * Remove a key/value combination from the cache. + * + * @param tileID - the key for the pair to delete + * @param value - If a value is provided, remove that exact version of the value. + * @returns this cache + */ + remove(tileID, value) { + if (!this.has(tileID)) { + return this; + } + const key = tileID.wrapped().key; + const dataIndex = value === undefined ? 0 : this.data[key].indexOf(value); + const data = this.data[key][dataIndex]; + this.data[key].splice(dataIndex, 1); + if (data.timeout) + clearTimeout(data.timeout); + if (this.data[key].length === 0) { + delete this.data[key]; + } + this.onRemove(data.value); + this.order.splice(this.order.indexOf(key), 1); + return this; + } + /** + * Change the max size of the cache. + * + * @param max - the max size of the cache + * @returns this cache + */ + setMaxSize(max) { + this.max = max; + while (this.order.length > this.max) { + const removedData = this._getAndRemoveByKey(this.order[0]); + if (removedData) + this.onRemove(removedData); + } + return this; + } + /** + * Remove entries that do not pass a filter function. Used for removing + * stale tiles from the cache. + * + * @param filterFn - Determines whether the tile is filtered. If the supplied function returns false, the tile will be filtered out. + */ + filter(filterFn) { + const removed = []; + for (const key in this.data) { + for (const entry of this.data[key]) { + if (!filterFn(entry.value)) { + removed.push(entry); + } + } + } + for (const r of removed) { + this.remove(r.value.tileID, r); + } + } +} + +/** + * @internal + * SourceFeatureState manages the state and pending changes + * to features in a source, separated by source layer. + * stateChanges and deletedStates batch all changes to the tile (updates and removes, respectively) + * between coalesce() events. addFeatureState() and removeFeatureState() also update their counterpart's + * list of changes, such that coalesce() can apply the proper state changes while agnostic to the order of operations. + * In deletedStates, all null's denote complete removal of state at that scope +*/ +class SourceFeatureState { + constructor() { + this.state = {}; + this.stateChanges = {}; + this.deletedStates = {}; + } + updateState(sourceLayer, featureId, newState) { + const feature = String(featureId); + this.stateChanges[sourceLayer] = this.stateChanges[sourceLayer] || {}; + this.stateChanges[sourceLayer][feature] = this.stateChanges[sourceLayer][feature] || {}; + performance.extend(this.stateChanges[sourceLayer][feature], newState); + if (this.deletedStates[sourceLayer] === null) { + this.deletedStates[sourceLayer] = {}; + for (const ft in this.state[sourceLayer]) { + if (ft !== feature) + this.deletedStates[sourceLayer][ft] = null; + } + } + else { + const featureDeletionQueued = this.deletedStates[sourceLayer] && this.deletedStates[sourceLayer][feature] === null; + if (featureDeletionQueued) { + this.deletedStates[sourceLayer][feature] = {}; + for (const prop in this.state[sourceLayer][feature]) { + if (!newState[prop]) + this.deletedStates[sourceLayer][feature][prop] = null; + } + } + else { + for (const key in newState) { + const deletionInQueue = this.deletedStates[sourceLayer] && this.deletedStates[sourceLayer][feature] && this.deletedStates[sourceLayer][feature][key] === null; + if (deletionInQueue) + delete this.deletedStates[sourceLayer][feature][key]; + } + } + } + } + removeFeatureState(sourceLayer, featureId, key) { + const sourceLayerDeleted = this.deletedStates[sourceLayer] === null; + if (sourceLayerDeleted) + return; + const feature = String(featureId); + this.deletedStates[sourceLayer] = this.deletedStates[sourceLayer] || {}; + if (key && featureId !== undefined) { + if (this.deletedStates[sourceLayer][feature] !== null) { + this.deletedStates[sourceLayer][feature] = this.deletedStates[sourceLayer][feature] || {}; + this.deletedStates[sourceLayer][feature][key] = null; + } + } + else if (featureId !== undefined) { + const updateInQueue = this.stateChanges[sourceLayer] && this.stateChanges[sourceLayer][feature]; + if (updateInQueue) { + this.deletedStates[sourceLayer][feature] = {}; + for (key in this.stateChanges[sourceLayer][feature]) + this.deletedStates[sourceLayer][feature][key] = null; + } + else { + this.deletedStates[sourceLayer][feature] = null; + } + } + else { + this.deletedStates[sourceLayer] = null; + } + } + getState(sourceLayer, featureId) { + const feature = String(featureId); + const base = this.state[sourceLayer] || {}; + const changes = this.stateChanges[sourceLayer] || {}; + const reconciledState = performance.extend({}, base[feature], changes[feature]); + //return empty object if the whole source layer is awaiting deletion + if (this.deletedStates[sourceLayer] === null) + return {}; + else if (this.deletedStates[sourceLayer]) { + const featureDeletions = this.deletedStates[sourceLayer][featureId]; + if (featureDeletions === null) + return {}; + for (const prop in featureDeletions) + delete reconciledState[prop]; + } + return reconciledState; + } + initializeTileState(tile, painter) { + tile.setFeatureState(this.state, painter); + } + coalesceChanges(tiles, painter) { + //track changes with full state objects, but only for features that got modified + const featuresChanged = {}; + for (const sourceLayer in this.stateChanges) { + this.state[sourceLayer] = this.state[sourceLayer] || {}; + const layerStates = {}; + for (const feature in this.stateChanges[sourceLayer]) { + if (!this.state[sourceLayer][feature]) + this.state[sourceLayer][feature] = {}; + performance.extend(this.state[sourceLayer][feature], this.stateChanges[sourceLayer][feature]); + layerStates[feature] = this.state[sourceLayer][feature]; + } + featuresChanged[sourceLayer] = layerStates; + } + for (const sourceLayer in this.deletedStates) { + this.state[sourceLayer] = this.state[sourceLayer] || {}; + const layerStates = {}; + if (this.deletedStates[sourceLayer] === null) { + for (const ft in this.state[sourceLayer]) { + layerStates[ft] = {}; + this.state[sourceLayer][ft] = {}; + } + } + else { + for (const feature in this.deletedStates[sourceLayer]) { + const deleteWholeFeatureState = this.deletedStates[sourceLayer][feature] === null; + if (deleteWholeFeatureState) + this.state[sourceLayer][feature] = {}; + else { + for (const key of Object.keys(this.deletedStates[sourceLayer][feature])) { + delete this.state[sourceLayer][feature][key]; + } + } + layerStates[feature] = this.state[sourceLayer][feature]; + } + } + featuresChanged[sourceLayer] = featuresChanged[sourceLayer] || {}; + performance.extend(featuresChanged[sourceLayer], layerStates); + } + this.stateChanges = {}; + this.deletedStates = {}; + if (Object.keys(featuresChanged).length === 0) + return; + for (const id in tiles) { + const tile = tiles[id]; + tile.setFeatureState(featuresChanged, painter); + } + } +} + +/** + * @internal + * `SourceCache` is responsible for + * + * - creating an instance of `Source` + * - forwarding events from `Source` + * - caching tiles loaded from an instance of `Source` + * - loading the tiles needed to render a given viewport + * - unloading the cached tiles not needed to render a given viewport + */ +class SourceCache extends performance.Evented { + constructor(id, options, dispatcher) { + super(); + this.id = id; + this.dispatcher = dispatcher; + this.on('data', (e) => { + // this._sourceLoaded signifies that the TileJSON is loaded if applicable. + // if the source type does not come with a TileJSON, the flag signifies the + // source data has loaded (i.e geojson has been tiled on the worker and is ready) + if (e.dataType === 'source' && e.sourceDataType === 'metadata') + this._sourceLoaded = true; + // for sources with mutable data, this event fires when the underlying data + // to a source is changed. (i.e. GeoJSONSource#setData and ImageSource#serCoordinates) + if (this._sourceLoaded && !this._paused && e.dataType === 'source' && e.sourceDataType === 'content') { + this.reload(); + if (this.transform) { + this.update(this.transform, this.terrain); + } + this._didEmitContent = true; + } + }); + this.on('dataloading', () => { + this._sourceErrored = false; + }); + this.on('error', () => { + // Only set _sourceErrored if the source does not have pending loads. + this._sourceErrored = this._source.loaded(); + }); + this._source = create(id, options, dispatcher, this); + this._tiles = {}; + this._cache = new TileCache(0, this._unloadTile.bind(this)); + this._timers = {}; + this._cacheTimers = {}; + this._maxTileCacheSize = null; + this._maxTileCacheZoomLevels = null; + this._loadedParentTiles = {}; + this._coveredTiles = {}; + this._state = new SourceFeatureState(); + this._didEmitContent = false; + this._updated = false; + } + onAdd(map) { + this.map = map; + this._maxTileCacheSize = map ? map._maxTileCacheSize : null; + this._maxTileCacheZoomLevels = map ? map._maxTileCacheZoomLevels : null; + if (this._source && this._source.onAdd) { + this._source.onAdd(map); + } + } + onRemove(map) { + this.clearTiles(); + if (this._source && this._source.onRemove) { + this._source.onRemove(map); + } + } + /** + * Return true if no tile data is pending, tiles will not change unless + * an additional API call is received. + */ + loaded() { + if (this._sourceErrored) { + return true; + } + if (!this._sourceLoaded) { + return false; + } + if (!this._source.loaded()) { + return false; + } + if ((this.used !== undefined || this.usedForTerrain !== undefined) && !this.used && !this.usedForTerrain) { + return true; + } + // do not consider as loaded if the update hasn't been called yet (we do not know if we will have any tiles to fetch) + if (!this._updated) { + return false; + } + for (const t in this._tiles) { + const tile = this._tiles[t]; + if (tile.state !== 'loaded' && tile.state !== 'errored') + return false; + } + return true; + } + getSource() { + return this._source; + } + pause() { + this._paused = true; + } + resume() { + if (!this._paused) + return; + const shouldReload = this._shouldReloadOnResume; + this._paused = false; + this._shouldReloadOnResume = false; + if (shouldReload) + this.reload(); + if (this.transform) + this.update(this.transform, this.terrain); + } + _loadTile(tile, callback) { + return this._source.loadTile(tile, callback); + } + _unloadTile(tile) { + if (this._source.unloadTile) + return this._source.unloadTile(tile, () => { }); + } + _abortTile(tile) { + if (this._source.abortTile) + this._source.abortTile(tile, () => { }); + this._source.fire(new performance.Event('dataabort', { tile, coord: tile.tileID, dataType: 'source' })); + } + serialize() { + return this._source.serialize(); + } + prepare(context) { + if (this._source.prepare) { + this._source.prepare(); + } + this._state.coalesceChanges(this._tiles, this.map ? this.map.painter : null); + for (const i in this._tiles) { + const tile = this._tiles[i]; + tile.upload(context); + tile.prepare(this.map.style.imageManager); + } + } + /** + * Return all tile ids ordered with z-order, and cast to numbers + */ + getIds() { + return Object.values(this._tiles).map((tile) => tile.tileID).sort(compareTileId).map(id => id.key); + } + getRenderableIds(symbolLayer) { + const renderables = []; + for (const id in this._tiles) { + if (this._isIdRenderable(id, symbolLayer)) + renderables.push(this._tiles[id]); + } + if (symbolLayer) { + return renderables.sort((a_, b_) => { + const a = a_.tileID; + const b = b_.tileID; + const rotatedA = (new performance.Point(a.canonical.x, a.canonical.y))._rotate(this.transform.angle); + const rotatedB = (new performance.Point(b.canonical.x, b.canonical.y))._rotate(this.transform.angle); + return a.overscaledZ - b.overscaledZ || rotatedB.y - rotatedA.y || rotatedB.x - rotatedA.x; + }).map(tile => tile.tileID.key); + } + return renderables.map(tile => tile.tileID).sort(compareTileId).map(id => id.key); + } + hasRenderableParent(tileID) { + const parentTile = this.findLoadedParent(tileID, 0); + if (parentTile) { + return this._isIdRenderable(parentTile.tileID.key); + } + return false; + } + _isIdRenderable(id, symbolLayer) { + return this._tiles[id] && this._tiles[id].hasData() && + !this._coveredTiles[id] && (symbolLayer || !this._tiles[id].holdingForFade()); + } + reload() { + if (this._paused) { + this._shouldReloadOnResume = true; + return; + } + this._cache.reset(); + for (const i in this._tiles) { + if (this._tiles[i].state !== 'errored') + this._reloadTile(i, 'reloading'); + } + } + _reloadTile(id, state) { + const tile = this._tiles[id]; + // this potentially does not address all underlying + // issues https://github.com/mapbox/mapbox-gl-js/issues/4252 + // - hard to tell without repro steps + if (!tile) + return; + // The difference between "loading" tiles and "reloading" or "expired" + // tiles is that "reloading"/"expired" tiles are "renderable". + // Therefore, a "loading" tile cannot become a "reloading" tile without + // first becoming a "loaded" tile. + if (tile.state !== 'loading') { + tile.state = state; + } + this._loadTile(tile, this._tileLoaded.bind(this, tile, id, state)); + } + _tileLoaded(tile, id, previousState, err) { + if (err) { + tile.state = 'errored'; + if (err.status !== 404) + this._source.fire(new performance.ErrorEvent(err, { tile })); + // continue to try loading parent/children tiles if a tile doesn't exist (404) + else + this.update(this.transform, this.terrain); + return; + } + tile.timeAdded = performance.browser.now(); + if (previousState === 'expired') + tile.refreshedUponExpiration = true; + this._setTileReloadTimer(id, tile); + if (this.getSource().type === 'raster-dem' && tile.dem) + this._backfillDEM(tile); + this._state.initializeTileState(tile, this.map ? this.map.painter : null); + if (!tile.aborted) { + this._source.fire(new performance.Event('data', { dataType: 'source', tile, coord: tile.tileID })); + } + } + /** + * For raster terrain source, backfill DEM to eliminate visible tile boundaries + */ + _backfillDEM(tile) { + const renderables = this.getRenderableIds(); + for (let i = 0; i < renderables.length; i++) { + const borderId = renderables[i]; + if (tile.neighboringTiles && tile.neighboringTiles[borderId]) { + const borderTile = this.getTileByID(borderId); + fillBorder(tile, borderTile); + fillBorder(borderTile, tile); + } + } + function fillBorder(tile, borderTile) { + tile.needsHillshadePrepare = true; + tile.needsTerrainPrepare = true; + let dx = borderTile.tileID.canonical.x - tile.tileID.canonical.x; + const dy = borderTile.tileID.canonical.y - tile.tileID.canonical.y; + const dim = Math.pow(2, tile.tileID.canonical.z); + const borderId = borderTile.tileID.key; + if (dx === 0 && dy === 0) + return; + if (Math.abs(dy) > 1) { + return; + } + if (Math.abs(dx) > 1) { + // Adjust the delta coordinate for world wraparound. + if (Math.abs(dx + dim) === 1) { + dx += dim; + } + else if (Math.abs(dx - dim) === 1) { + dx -= dim; + } + } + if (!borderTile.dem || !tile.dem) + return; + tile.dem.backfillBorder(borderTile.dem, dx, dy); + if (tile.neighboringTiles && tile.neighboringTiles[borderId]) + tile.neighboringTiles[borderId].backfilled = true; + } + } + /** + * Get a specific tile by TileID + */ + getTile(tileID) { + return this.getTileByID(tileID.key); + } + /** + * Get a specific tile by id + */ + getTileByID(id) { + return this._tiles[id]; + } + /** + * For a given set of tiles, retain children that are loaded and have a zoom + * between `zoom` (exclusive) and `maxCoveringZoom` (inclusive) + */ + _retainLoadedChildren(idealTiles, zoom, maxCoveringZoom, retain) { + for (const id in this._tiles) { + let tile = this._tiles[id]; + // only consider renderable tiles up to maxCoveringZoom + if (retain[id] || + !tile.hasData() || + tile.tileID.overscaledZ <= zoom || + tile.tileID.overscaledZ > maxCoveringZoom) + continue; + // loop through parents and retain the topmost loaded one if found + let topmostLoadedID = tile.tileID; + while (tile && tile.tileID.overscaledZ > zoom + 1) { + const parentID = tile.tileID.scaledTo(tile.tileID.overscaledZ - 1); + tile = this._tiles[parentID.key]; + if (tile && tile.hasData()) { + topmostLoadedID = parentID; + } + } + // loop through ancestors of the topmost loaded child to see if there's one that needed it + let tileID = topmostLoadedID; + while (tileID.overscaledZ > zoom) { + tileID = tileID.scaledTo(tileID.overscaledZ - 1); + if (idealTiles[tileID.key]) { + // found a parent that needed a loaded child; retain that child + retain[topmostLoadedID.key] = topmostLoadedID; + break; + } + } + } + } + /** + * Find a loaded parent of the given tile (up to minCoveringZoom) + */ + findLoadedParent(tileID, minCoveringZoom) { + if (tileID.key in this._loadedParentTiles) { + const parent = this._loadedParentTiles[tileID.key]; + if (parent && parent.tileID.overscaledZ >= minCoveringZoom) { + return parent; + } + else { + return null; + } + } + for (let z = tileID.overscaledZ - 1; z >= minCoveringZoom; z--) { + const parentTileID = tileID.scaledTo(z); + const tile = this._getLoadedTile(parentTileID); + if (tile) { + return tile; + } + } + } + _getLoadedTile(tileID) { + const tile = this._tiles[tileID.key]; + if (tile && tile.hasData()) { + return tile; + } + // TileCache ignores wrap in lookup. + const cachedTile = this._cache.getByKey(tileID.wrapped().key); + return cachedTile; + } + /** + * Resizes the tile cache based on the current viewport's size + * or the maxTileCacheSize option passed during map creation + * + * Larger viewports use more tiles and need larger caches. Larger viewports + * are more likely to be found on devices with more memory and on pages where + * the map is more important. + */ + updateCacheSize(transform) { + const widthInTiles = Math.ceil(transform.width / this._source.tileSize) + 1; + const heightInTiles = Math.ceil(transform.height / this._source.tileSize) + 1; + const approxTilesInView = widthInTiles * heightInTiles; + const commonZoomRange = this._maxTileCacheZoomLevels === null ? + performance.config.MAX_TILE_CACHE_ZOOM_LEVELS : this._maxTileCacheZoomLevels; + const viewDependentMaxSize = Math.floor(approxTilesInView * commonZoomRange); + const maxSize = typeof this._maxTileCacheSize === 'number' ? + Math.min(this._maxTileCacheSize, viewDependentMaxSize) : viewDependentMaxSize; + this._cache.setMaxSize(maxSize); + } + handleWrapJump(lng) { + // On top of the regular z/x/y values, TileIDs have a `wrap` value that specify + // which cppy of the world the tile belongs to. For example, at `lng: 10` you + // might render z/x/y/0 while at `lng: 370` you would render z/x/y/1. + // + // When lng values get wrapped (going from `lng: 370` to `long: 10`) you expect + // to see the same thing on the screen (370 degrees and 10 degrees is the same + // place in the world) but all the TileIDs will have different wrap values. + // + // In order to make this transition seamless, we calculate the rounded difference of + // "worlds" between the last frame and the current frame. If the map panned by + // a world, then we can assign all the tiles new TileIDs with updated wrap values. + // For example, assign z/x/y/1 a new id: z/x/y/0. It is the same tile, just rendered + // in a different position. + // + // This enables us to reuse the tiles at more ideal locations and prevent flickering. + const prevLng = this._prevLng === undefined ? lng : this._prevLng; + const lngDifference = lng - prevLng; + const worldDifference = lngDifference / 360; + const wrapDelta = Math.round(worldDifference); + this._prevLng = lng; + if (wrapDelta) { + const tiles = {}; + for (const key in this._tiles) { + const tile = this._tiles[key]; + tile.tileID = tile.tileID.unwrapTo(tile.tileID.wrap + wrapDelta); + tiles[tile.tileID.key] = tile; + } + this._tiles = tiles; + // Reset tile reload timers + for (const id in this._timers) { + clearTimeout(this._timers[id]); + delete this._timers[id]; + } + for (const id in this._tiles) { + const tile = this._tiles[id]; + this._setTileReloadTimer(id, tile); + } + } + } + /** + * Removes tiles that are outside the viewport and adds new tiles that + * are inside the viewport. + */ + update(transform, terrain) { + this.transform = transform; + this.terrain = terrain; + if (!this._sourceLoaded || this._paused) { + return; + } + this.updateCacheSize(transform); + this.handleWrapJump(this.transform.center.lng); + // Covered is a list of retained tiles who's areas are fully covered by other, + // better, retained tiles. They are not drawn separately. + this._coveredTiles = {}; + let idealTileIDs; + if (!this.used && !this.usedForTerrain) { + idealTileIDs = []; + } + else if (this._source.tileID) { + idealTileIDs = transform.getVisibleUnwrappedCoordinates(this._source.tileID) + .map((unwrapped) => new performance.OverscaledTileID(unwrapped.canonical.z, unwrapped.wrap, unwrapped.canonical.z, unwrapped.canonical.x, unwrapped.canonical.y)); + } + else { + idealTileIDs = transform.coveringTiles({ + tileSize: this.usedForTerrain ? this.tileSize : this._source.tileSize, + minzoom: this._source.minzoom, + maxzoom: this._source.maxzoom, + roundZoom: this.usedForTerrain ? false : this._source.roundZoom, + reparseOverscaled: this._source.reparseOverscaled, + terrain + }); + if (this._source.hasTile) { + idealTileIDs = idealTileIDs.filter((coord) => this._source.hasTile(coord)); + } + } + // Determine the overzooming/underzooming amounts. + const zoom = transform.coveringZoomLevel(this._source); + const minCoveringZoom = Math.max(zoom - SourceCache.maxOverzooming, this._source.minzoom); + const maxCoveringZoom = Math.max(zoom + SourceCache.maxUnderzooming, this._source.minzoom); + // When sourcecache is used for terrain also load parent tiles to avoid flickering when zooming out + if (this.usedForTerrain) { + const parents = {}; + for (const tileID of idealTileIDs) { + if (tileID.canonical.z > this._source.minzoom) { + const parent = tileID.scaledTo(tileID.canonical.z - 1); + parents[parent.key] = parent; + // load very low zoom to calculate tile visibility in transform.coveringTiles and high zoomlevels correct + const parent2 = tileID.scaledTo(Math.max(this._source.minzoom, Math.min(tileID.canonical.z, 5))); + parents[parent2.key] = parent2; + } + } + idealTileIDs = idealTileIDs.concat(Object.values(parents)); + } + const noPendingDataEmissions = idealTileIDs.length === 0 && !this._updated && this._didEmitContent; + this._updated = true; + // if we won't have any tiles to fetch and content is already emitted + // there will be no more data emissions, so we need to emit the event with isSourceLoaded = true + if (noPendingDataEmissions) { + this.fire(new performance.Event('data', { sourceDataType: 'idle', dataType: 'source', sourceId: this.id })); + } + // Retain is a list of tiles that we shouldn't delete, even if they are not + // the most ideal tile for the current viewport. This may include tiles like + // parent or child tiles that are *already* loaded. + const retain = this._updateRetainedTiles(idealTileIDs, zoom); + if (isRasterType(this._source.type)) { + const parentsForFading = {}; + const fadingTiles = {}; + const ids = Object.keys(retain); + const now = performance.browser.now(); + for (const id of ids) { + const tileID = retain[id]; + const tile = this._tiles[id]; + // when fadeEndTime is 0, the tile is created but registerFadeDuration + // has not been called, therefore must be kept in fadingTiles dictionary + // for next round of rendering + if (!tile || (tile.fadeEndTime !== 0 && tile.fadeEndTime <= now)) { + continue; + } + // if the tile is loaded but still fading in, find parents to cross-fade with it + const parentTile = this.findLoadedParent(tileID, minCoveringZoom); + if (parentTile) { + this._addTile(parentTile.tileID); + parentsForFading[parentTile.tileID.key] = parentTile.tileID; + } + fadingTiles[id] = tileID; + } + // for tiles that are still fading in, also find children to cross-fade with + this._retainLoadedChildren(fadingTiles, zoom, maxCoveringZoom, retain); + for (const id in parentsForFading) { + if (!retain[id]) { + // If a tile is only needed for fading, mark it as covered so that it isn't rendered on it's own. + this._coveredTiles[id] = true; + retain[id] = parentsForFading[id]; + } + } + // disable fading logic in terrain3D mode to avoid rendering two tiles on the same place + if (terrain) { + const idealRasterTileIDs = {}; + const missingTileIDs = {}; + for (const tileID of idealTileIDs) { + if (this._tiles[tileID.key].hasData()) + idealRasterTileIDs[tileID.key] = tileID; + else + missingTileIDs[tileID.key] = tileID; + } + // search for a complete set of children for each missing tile + for (const key in missingTileIDs) { + const children = missingTileIDs[key].children(this._source.maxzoom); + if (this._tiles[children[0].key] && this._tiles[children[1].key] && this._tiles[children[2].key] && this._tiles[children[3].key]) { + idealRasterTileIDs[children[0].key] = retain[children[0].key] = children[0]; + idealRasterTileIDs[children[1].key] = retain[children[1].key] = children[1]; + idealRasterTileIDs[children[2].key] = retain[children[2].key] = children[2]; + idealRasterTileIDs[children[3].key] = retain[children[3].key] = children[3]; + delete missingTileIDs[key]; + } + } + // search for parent for each missing tile + for (const key in missingTileIDs) { + const parent = this.findLoadedParent(missingTileIDs[key], this._source.minzoom); + if (parent) { + idealRasterTileIDs[parent.tileID.key] = retain[parent.tileID.key] = parent.tileID; + // remove idealTiles which would be rendered twice + for (const key in idealRasterTileIDs) { + if (idealRasterTileIDs[key].isChildOf(parent.tileID)) + delete idealRasterTileIDs[key]; + } + } + } + // cover all tiles which are not needed + for (const key in this._tiles) { + if (!idealRasterTileIDs[key]) + this._coveredTiles[key] = true; + } + } + } + for (const retainedId in retain) { + // Make sure retained tiles always clear any existing fade holds + // so that if they're removed again their fade timer starts fresh. + this._tiles[retainedId].clearFadeHold(); + } + // Remove the tiles we don't need anymore. + const remove = performance.keysDifference(this._tiles, retain); + for (const tileID of remove) { + const tile = this._tiles[tileID]; + if (tile.hasSymbolBuckets && !tile.holdingForFade()) { + tile.setHoldDuration(this.map._fadeDuration); + } + else if (!tile.hasSymbolBuckets || tile.symbolFadeFinished()) { + this._removeTile(tileID); + } + } + // Construct a cache of loaded parents + this._updateLoadedParentTileCache(); + } + releaseSymbolFadeTiles() { + for (const id in this._tiles) { + if (this._tiles[id].holdingForFade()) { + this._removeTile(id); + } + } + } + _updateRetainedTiles(idealTileIDs, zoom) { + const retain = {}; + const checked = {}; + const minCoveringZoom = Math.max(zoom - SourceCache.maxOverzooming, this._source.minzoom); + const maxCoveringZoom = Math.max(zoom + SourceCache.maxUnderzooming, this._source.minzoom); + const missingTiles = {}; + for (const tileID of idealTileIDs) { + const tile = this._addTile(tileID); + // retain the tile even if it's not loaded because it's an ideal tile. + retain[tileID.key] = tileID; + if (tile.hasData()) + continue; + if (zoom < this._source.maxzoom) { + // save missing tiles that potentially have loaded children + missingTiles[tileID.key] = tileID; + } + } + // retain any loaded children of ideal tiles up to maxCoveringZoom + this._retainLoadedChildren(missingTiles, zoom, maxCoveringZoom, retain); + for (const tileID of idealTileIDs) { + let tile = this._tiles[tileID.key]; + if (tile.hasData()) + continue; + // The tile we require is not yet loaded or does not exist; + // Attempt to find children that fully cover it. + if (zoom + 1 > this._source.maxzoom) { + // We're looking for an overzoomed child tile. + const childCoord = tileID.children(this._source.maxzoom)[0]; + const childTile = this.getTile(childCoord); + if (!!childTile && childTile.hasData()) { + retain[childCoord.key] = childCoord; + continue; // tile is covered by overzoomed child + } + } + else { + // check if all 4 immediate children are loaded (i.e. the missing ideal tile is covered) + const children = tileID.children(this._source.maxzoom); + if (retain[children[0].key] && + retain[children[1].key] && + retain[children[2].key] && + retain[children[3].key]) + continue; // tile is covered by children + } + // We couldn't find child tiles that entirely cover the ideal tile; look for parents now. + // As we ascend up the tile pyramid of the ideal tile, we check whether the parent + // tile has been previously requested (and errored because we only loop over tiles with no data) + // in order to determine if we need to request its parent. + let parentWasRequested = tile.wasRequested(); + for (let overscaledZ = tileID.overscaledZ - 1; overscaledZ >= minCoveringZoom; --overscaledZ) { + const parentId = tileID.scaledTo(overscaledZ); + // Break parent tile ascent if this route has been previously checked by another child. + if (checked[parentId.key]) + break; + checked[parentId.key] = true; + tile = this.getTile(parentId); + if (!tile && parentWasRequested) { + tile = this._addTile(parentId); + } + if (tile) { + const hasData = tile.hasData(); + if (parentWasRequested || hasData) { + retain[parentId.key] = parentId; + } + // Save the current values, since they're the parent of the next iteration + // of the parent tile ascent loop. + parentWasRequested = tile.wasRequested(); + if (hasData) + break; + } + } + } + return retain; + } + _updateLoadedParentTileCache() { + this._loadedParentTiles = {}; + for (const tileKey in this._tiles) { + const path = []; + let parentTile; + let currentId = this._tiles[tileKey].tileID; + // Find the closest loaded ancestor by traversing the tile tree towards the root and + // caching results along the way + while (currentId.overscaledZ > 0) { + // Do we have a cached result from previous traversals? + if (currentId.key in this._loadedParentTiles) { + parentTile = this._loadedParentTiles[currentId.key]; + break; + } + path.push(currentId.key); + // Is the parent loaded? + const parentId = currentId.scaledTo(currentId.overscaledZ - 1); + parentTile = this._getLoadedTile(parentId); + if (parentTile) { + break; + } + currentId = parentId; + } + // Cache the result of this traversal to all newly visited tiles + for (const key of path) { + this._loadedParentTiles[key] = parentTile; + } + } + } + /** + * Add a tile, given its coordinate, to the pyramid. + */ + _addTile(tileID) { + let tile = this._tiles[tileID.key]; + if (tile) + return tile; + tile = this._cache.getAndRemove(tileID); + if (tile) { + this._setTileReloadTimer(tileID.key, tile); + // set the tileID because the cached tile could have had a different wrap value + tile.tileID = tileID; + this._state.initializeTileState(tile, this.map ? this.map.painter : null); + if (this._cacheTimers[tileID.key]) { + clearTimeout(this._cacheTimers[tileID.key]); + delete this._cacheTimers[tileID.key]; + this._setTileReloadTimer(tileID.key, tile); + } + } + const cached = tile; + if (!tile) { + tile = new Tile(tileID, this._source.tileSize * tileID.overscaleFactor()); + this._loadTile(tile, this._tileLoaded.bind(this, tile, tileID.key, tile.state)); + } + tile.uses++; + this._tiles[tileID.key] = tile; + if (!cached) { + this._source.fire(new performance.Event('dataloading', { tile, coord: tile.tileID, dataType: 'source' })); + } + return tile; + } + _setTileReloadTimer(id, tile) { + if (id in this._timers) { + clearTimeout(this._timers[id]); + delete this._timers[id]; + } + const expiryTimeout = tile.getExpiryTimeout(); + if (expiryTimeout) { + this._timers[id] = setTimeout(() => { + this._reloadTile(id, 'expired'); + delete this._timers[id]; + }, expiryTimeout); + } + } + /** + * Remove a tile, given its id, from the pyramid + */ + _removeTile(id) { + const tile = this._tiles[id]; + if (!tile) + return; + tile.uses--; + delete this._tiles[id]; + if (this._timers[id]) { + clearTimeout(this._timers[id]); + delete this._timers[id]; + } + if (tile.uses > 0) + return; + if (tile.hasData() && tile.state !== 'reloading') { + this._cache.add(tile.tileID, tile, tile.getExpiryTimeout()); + } + else { + tile.aborted = true; + this._abortTile(tile); + this._unloadTile(tile); + } + } + /** + * Remove all tiles from this pyramid + */ + clearTiles() { + this._shouldReloadOnResume = false; + this._paused = false; + for (const id in this._tiles) + this._removeTile(id); + this._cache.reset(); + } + /** + * Search through our current tiles and attempt to find the tiles that + * cover the given bounds. + * @param pointQueryGeometry - coordinates of the corners of bounding rectangle + * @returns result items have `{tile, minX, maxX, minY, maxY}`, where min/max bounding values are the given bounds transformed in into the coordinate space of this tile. + */ + tilesIn(pointQueryGeometry, maxPitchScaleFactor, has3DLayer) { + const tileResults = []; + const transform = this.transform; + if (!transform) + return tileResults; + const cameraPointQueryGeometry = has3DLayer ? + transform.getCameraQueryGeometry(pointQueryGeometry) : + pointQueryGeometry; + const queryGeometry = pointQueryGeometry.map((p) => transform.pointCoordinate(p, this.terrain)); + const cameraQueryGeometry = cameraPointQueryGeometry.map((p) => transform.pointCoordinate(p, this.terrain)); + const ids = this.getIds(); + let minX = Infinity; + let minY = Infinity; + let maxX = -Infinity; + let maxY = -Infinity; + for (const p of cameraQueryGeometry) { + minX = Math.min(minX, p.x); + minY = Math.min(minY, p.y); + maxX = Math.max(maxX, p.x); + maxY = Math.max(maxY, p.y); + } + for (let i = 0; i < ids.length; i++) { + const tile = this._tiles[ids[i]]; + if (tile.holdingForFade()) { + // Tiles held for fading are covered by tiles that are closer to ideal + continue; + } + const tileID = tile.tileID; + const scale = Math.pow(2, transform.zoom - tile.tileID.overscaledZ); + const queryPadding = maxPitchScaleFactor * tile.queryPadding * performance.EXTENT / tile.tileSize / scale; + const tileSpaceBounds = [ + tileID.getTilePoint(new performance.MercatorCoordinate(minX, minY)), + tileID.getTilePoint(new performance.MercatorCoordinate(maxX, maxY)) + ]; + if (tileSpaceBounds[0].x - queryPadding < performance.EXTENT && tileSpaceBounds[0].y - queryPadding < performance.EXTENT && + tileSpaceBounds[1].x + queryPadding >= 0 && tileSpaceBounds[1].y + queryPadding >= 0) { + const tileSpaceQueryGeometry = queryGeometry.map((c) => tileID.getTilePoint(c)); + const tileSpaceCameraQueryGeometry = cameraQueryGeometry.map((c) => tileID.getTilePoint(c)); + tileResults.push({ + tile, + tileID, + queryGeometry: tileSpaceQueryGeometry, + cameraQueryGeometry: tileSpaceCameraQueryGeometry, + scale + }); + } + } + return tileResults; + } + getVisibleCoordinates(symbolLayer) { + const coords = this.getRenderableIds(symbolLayer).map((id) => this._tiles[id].tileID); + for (const coord of coords) { + coord.posMatrix = this.transform.calculatePosMatrix(coord.toUnwrapped()); + } + return coords; + } + hasTransition() { + if (this._source.hasTransition()) { + return true; + } + if (isRasterType(this._source.type)) { + const now = performance.browser.now(); + for (const id in this._tiles) { + const tile = this._tiles[id]; + if (tile.fadeEndTime >= now) { + return true; + } + } + } + return false; + } + /** + * Set the value of a particular state for a feature + */ + setFeatureState(sourceLayer, featureId, state) { + sourceLayer = sourceLayer || '_geojsonTileLayer'; + this._state.updateState(sourceLayer, featureId, state); + } + /** + * Resets the value of a particular state key for a feature + */ + removeFeatureState(sourceLayer, featureId, key) { + sourceLayer = sourceLayer || '_geojsonTileLayer'; + this._state.removeFeatureState(sourceLayer, featureId, key); + } + /** + * Get the entire state object for a feature + */ + getFeatureState(sourceLayer, featureId) { + sourceLayer = sourceLayer || '_geojsonTileLayer'; + return this._state.getState(sourceLayer, featureId); + } + /** + * Sets the set of keys that the tile depends on. This allows tiles to + * be reloaded when their dependencies change. + */ + setDependencies(tileKey, namespace, dependencies) { + const tile = this._tiles[tileKey]; + if (tile) { + tile.setDependencies(namespace, dependencies); + } + } + /** + * Reloads all tiles that depend on the given keys. + */ + reloadTilesForDependencies(namespaces, keys) { + for (const id in this._tiles) { + const tile = this._tiles[id]; + if (tile.hasDependency(namespaces, keys)) { + this._reloadTile(id, 'reloading'); + } + } + this._cache.filter(tile => !tile.hasDependency(namespaces, keys)); + } +} +SourceCache.maxOverzooming = 10; +SourceCache.maxUnderzooming = 3; +function compareTileId(a, b) { + // Different copies of the world are sorted based on their distance to the center. + // Wrap values are converted to unsigned distances by reserving odd number for copies + // with negative wrap and even numbers for copies with positive wrap. + const aWrap = Math.abs(a.wrap * 2) - +(a.wrap < 0); + const bWrap = Math.abs(b.wrap * 2) - +(b.wrap < 0); + return a.overscaledZ - b.overscaledZ || bWrap - aWrap || b.canonical.y - a.canonical.y || b.canonical.x - a.canonical.x; +} +function isRasterType(type) { + return type === 'raster' || type === 'image' || type === 'video'; +} + +function workerFactory() { + return new Worker(performance.config.WORKER_URL); +} + +const PRELOAD_POOL_ID = 'mapboxgl_preloaded_worker_pool'; +/** + * Constructs a worker pool. + */ +class WorkerPool { + constructor() { + this.active = {}; + } + acquire(mapId) { + if (!this.workers) { + // Lazily look up the value of mapboxgl.workerCount so that + // client code has had a chance to set it. + this.workers = []; + while (this.workers.length < WorkerPool.workerCount) { + this.workers.push(workerFactory()); + } + } + this.active[mapId] = true; + return this.workers.slice(); + } + release(mapId) { + delete this.active[mapId]; + if (this.numActive() === 0) { + this.workers.forEach((w) => { + w.terminate(); + }); + this.workers = null; + } + } + isPreloaded() { + return !!this.active[PRELOAD_POOL_ID]; + } + numActive() { + return Object.keys(this.active).length; + } +} +// Based on results from A/B testing: https://github.com/maplibre/maplibre-gl-js/pull/2354 +const availableLogicalProcessors = Math.floor(performance.browser.hardwareConcurrency / 2); +WorkerPool.workerCount = performance.isSafari(globalThis) ? Math.max(Math.min(availableLogicalProcessors, 3), 1) : 1; + +let globalWorkerPool; +/** + * Creates (if necessary) and returns the single, global WorkerPool instance + * to be shared across each Map + */ +function getGlobalWorkerPool() { + if (!globalWorkerPool) { + globalWorkerPool = new WorkerPool(); + } + return globalWorkerPool; +} +function prewarm() { + const workerPool = getGlobalWorkerPool(); + workerPool.acquire(PRELOAD_POOL_ID); +} +function clearPrewarmedResources() { + const pool = globalWorkerPool; + if (pool) { + // Remove the pool only if all maps that referenced the preloaded global worker pool have been removed. + if (pool.isPreloaded() && pool.numActive() === 1) { + pool.release(PRELOAD_POOL_ID); + globalWorkerPool = null; + } + else { + console.warn('Could not clear WebWorkers since there are active Map instances that still reference it. The pre-warmed WebWorker pool can only be cleared when all map instances have been removed with map.remove()'); + } + } +} + +class PathInterpolator { + constructor(points_, padding_) { + this.reset(points_, padding_); + } + reset(points_, padding_) { + this.points = points_ || []; + // Compute cumulative distance from first point to every other point in the segment. + // Last entry in the array is total length of the path + this._distances = [0.0]; + for (let i = 1; i < this.points.length; i++) { + this._distances[i] = this._distances[i - 1] + this.points[i].dist(this.points[i - 1]); + } + this.length = this._distances[this._distances.length - 1]; + this.padding = Math.min(padding_ || 0, this.length * 0.5); + this.paddedLength = this.length - this.padding * 2.0; + } + lerp(t) { + if (this.points.length === 1) { + return this.points[0]; + } + t = performance.clamp(t, 0, 1); + // Find the correct segment [p0, p1] where p0 <= x < p1 + let currentIndex = 1; + let distOfCurrentIdx = this._distances[currentIndex]; + const distToTarget = t * this.paddedLength + this.padding; + while (distOfCurrentIdx < distToTarget && currentIndex < this._distances.length) { + distOfCurrentIdx = this._distances[++currentIndex]; + } + // Interpolate between the two points of the segment + const idxOfPrevPoint = currentIndex - 1; + const distOfPrevIdx = this._distances[idxOfPrevPoint]; + const segmentLength = distOfCurrentIdx - distOfPrevIdx; + const segmentT = segmentLength > 0 ? (distToTarget - distOfPrevIdx) / segmentLength : 0; + return this.points[idxOfPrevPoint].mult(1.0 - segmentT).add(this.points[currentIndex].mult(segmentT)); + } +} + +function overlapAllowed(overlapA, overlapB) { + let allowed = true; + if (overlapA === 'always') { + // symbol A using 'always' overlap - allowed to overlap anything. + } + else if (overlapA === 'never' || overlapB === 'never') { + // symbol A using 'never' overlap - can't overlap anything + // symbol A using 'cooperative' overlap - can overlap 'always' or 'cooperative' symbol; can't overlap 'never' + allowed = false; + } + return allowed; +} +/** + * @internal + * GridIndex is a data structure for testing the intersection of + * circles and rectangles in a 2d plane. + * It is optimized for rapid insertion and querying. + * GridIndex splits the plane into a set of "cells" and keeps track + * of which geometries intersect with each cell. At query time, + * full geometry comparisons are only done for items that share + * at least one cell. As long as the geometries are relatively + * uniformly distributed across the plane, this greatly reduces + * the number of comparisons necessary. + */ +class GridIndex { + constructor(width, height, cellSize) { + const boxCells = this.boxCells = []; + const circleCells = this.circleCells = []; + // More cells -> fewer geometries to check per cell, but items tend + // to be split across more cells. + // Sweet spot allows most small items to fit in one cell + this.xCellCount = Math.ceil(width / cellSize); + this.yCellCount = Math.ceil(height / cellSize); + for (let i = 0; i < this.xCellCount * this.yCellCount; i++) { + boxCells.push([]); + circleCells.push([]); + } + this.circleKeys = []; + this.boxKeys = []; + this.bboxes = []; + this.circles = []; + this.width = width; + this.height = height; + this.xScale = this.xCellCount / width; + this.yScale = this.yCellCount / height; + this.boxUid = 0; + this.circleUid = 0; + } + keysLength() { + return this.boxKeys.length + this.circleKeys.length; + } + insert(key, x1, y1, x2, y2) { + this._forEachCell(x1, y1, x2, y2, this._insertBoxCell, this.boxUid++); + this.boxKeys.push(key); + this.bboxes.push(x1); + this.bboxes.push(y1); + this.bboxes.push(x2); + this.bboxes.push(y2); + } + insertCircle(key, x, y, radius) { + // Insert circle into grid for all cells in the circumscribing square + // It's more than necessary (by a factor of 4/PI), but fast to insert + this._forEachCell(x - radius, y - radius, x + radius, y + radius, this._insertCircleCell, this.circleUid++); + this.circleKeys.push(key); + this.circles.push(x); + this.circles.push(y); + this.circles.push(radius); + } + _insertBoxCell(x1, y1, x2, y2, cellIndex, uid) { + this.boxCells[cellIndex].push(uid); + } + _insertCircleCell(x1, y1, x2, y2, cellIndex, uid) { + this.circleCells[cellIndex].push(uid); + } + _query(x1, y1, x2, y2, hitTest, overlapMode, predicate) { + if (x2 < 0 || x1 > this.width || y2 < 0 || y1 > this.height) { + return []; + } + const result = []; + if (x1 <= 0 && y1 <= 0 && this.width <= x2 && this.height <= y2) { + if (hitTest) { + // Covers the entire grid, so collides with everything + return [{ + key: null, + x1, + y1, + x2, + y2 + }]; + } + for (let boxUid = 0; boxUid < this.boxKeys.length; boxUid++) { + result.push({ + key: this.boxKeys[boxUid], + x1: this.bboxes[boxUid * 4], + y1: this.bboxes[boxUid * 4 + 1], + x2: this.bboxes[boxUid * 4 + 2], + y2: this.bboxes[boxUid * 4 + 3] + }); + } + for (let circleUid = 0; circleUid < this.circleKeys.length; circleUid++) { + const x = this.circles[circleUid * 3]; + const y = this.circles[circleUid * 3 + 1]; + const radius = this.circles[circleUid * 3 + 2]; + result.push({ + key: this.circleKeys[circleUid], + x1: x - radius, + y1: y - radius, + x2: x + radius, + y2: y + radius + }); + } + } + else { + const queryArgs = { + hitTest, + overlapMode, + seenUids: { box: {}, circle: {} } + }; + this._forEachCell(x1, y1, x2, y2, this._queryCell, result, queryArgs, predicate); + } + return result; + } + query(x1, y1, x2, y2) { + return this._query(x1, y1, x2, y2, false, null); + } + hitTest(x1, y1, x2, y2, overlapMode, predicate) { + return this._query(x1, y1, x2, y2, true, overlapMode, predicate).length > 0; + } + hitTestCircle(x, y, radius, overlapMode, predicate) { + // Insert circle into grid for all cells in the circumscribing square + // It's more than necessary (by a factor of 4/PI), but fast to insert + const x1 = x - radius; + const x2 = x + radius; + const y1 = y - radius; + const y2 = y + radius; + if (x2 < 0 || x1 > this.width || y2 < 0 || y1 > this.height) { + return false; + } + // Box query early exits if the bounding box is larger than the grid, but we don't do + // the equivalent calculation for circle queries because early exit is less likely + // and the calculation is more expensive + const result = []; + const queryArgs = { + hitTest: true, + overlapMode, + circle: { x, y, radius }, + seenUids: { box: {}, circle: {} } + }; + this._forEachCell(x1, y1, x2, y2, this._queryCellCircle, result, queryArgs, predicate); + return result.length > 0; + } + _queryCell(x1, y1, x2, y2, cellIndex, result, queryArgs, predicate) { + const { seenUids, hitTest, overlapMode } = queryArgs; + const boxCell = this.boxCells[cellIndex]; + if (boxCell !== null) { + const bboxes = this.bboxes; + for (const boxUid of boxCell) { + if (!seenUids.box[boxUid]) { + seenUids.box[boxUid] = true; + const offset = boxUid * 4; + const key = this.boxKeys[boxUid]; + if ((x1 <= bboxes[offset + 2]) && + (y1 <= bboxes[offset + 3]) && + (x2 >= bboxes[offset + 0]) && + (y2 >= bboxes[offset + 1]) && + (!predicate || predicate(key))) { + if (!hitTest || !overlapAllowed(overlapMode, key.overlapMode)) { + result.push({ + key, + x1: bboxes[offset], + y1: bboxes[offset + 1], + x2: bboxes[offset + 2], + y2: bboxes[offset + 3] + }); + if (hitTest) { + // true return value stops the query after first match + return true; + } + } + } + } + } + } + const circleCell = this.circleCells[cellIndex]; + if (circleCell !== null) { + const circles = this.circles; + for (const circleUid of circleCell) { + if (!seenUids.circle[circleUid]) { + seenUids.circle[circleUid] = true; + const offset = circleUid * 3; + const key = this.circleKeys[circleUid]; + if (this._circleAndRectCollide(circles[offset], circles[offset + 1], circles[offset + 2], x1, y1, x2, y2) && + (!predicate || predicate(key))) { + if (!hitTest || !overlapAllowed(overlapMode, key.overlapMode)) { + const x = circles[offset]; + const y = circles[offset + 1]; + const radius = circles[offset + 2]; + result.push({ + key, + x1: x - radius, + y1: y - radius, + x2: x + radius, + y2: y + radius + }); + if (hitTest) { + // true return value stops the query after first match + return true; + } + } + } + } + } + } + // false return to continue query + return false; + } + _queryCellCircle(x1, y1, x2, y2, cellIndex, result, queryArgs, predicate) { + const { circle, seenUids, overlapMode } = queryArgs; + const boxCell = this.boxCells[cellIndex]; + if (boxCell !== null) { + const bboxes = this.bboxes; + for (const boxUid of boxCell) { + if (!seenUids.box[boxUid]) { + seenUids.box[boxUid] = true; + const offset = boxUid * 4; + const key = this.boxKeys[boxUid]; + if (this._circleAndRectCollide(circle.x, circle.y, circle.radius, bboxes[offset + 0], bboxes[offset + 1], bboxes[offset + 2], bboxes[offset + 3]) && + (!predicate || predicate(key)) && + !overlapAllowed(overlapMode, key.overlapMode)) { + result.push(true); + return true; + } + } + } + } + const circleCell = this.circleCells[cellIndex]; + if (circleCell !== null) { + const circles = this.circles; + for (const circleUid of circleCell) { + if (!seenUids.circle[circleUid]) { + seenUids.circle[circleUid] = true; + const offset = circleUid * 3; + const key = this.circleKeys[circleUid]; + if (this._circlesCollide(circles[offset], circles[offset + 1], circles[offset + 2], circle.x, circle.y, circle.radius) && + (!predicate || predicate(key)) && + !overlapAllowed(overlapMode, key.overlapMode)) { + result.push(true); + return true; + } + } + } + } + } + _forEachCell(x1, y1, x2, y2, fn, arg1, arg2, predicate) { + const cx1 = this._convertToXCellCoord(x1); + const cy1 = this._convertToYCellCoord(y1); + const cx2 = this._convertToXCellCoord(x2); + const cy2 = this._convertToYCellCoord(y2); + for (let x = cx1; x <= cx2; x++) { + for (let y = cy1; y <= cy2; y++) { + const cellIndex = this.xCellCount * y + x; + if (fn.call(this, x1, y1, x2, y2, cellIndex, arg1, arg2, predicate)) + return; + } + } + } + _convertToXCellCoord(x) { + return Math.max(0, Math.min(this.xCellCount - 1, Math.floor(x * this.xScale))); + } + _convertToYCellCoord(y) { + return Math.max(0, Math.min(this.yCellCount - 1, Math.floor(y * this.yScale))); + } + _circlesCollide(x1, y1, r1, x2, y2, r2) { + const dx = x2 - x1; + const dy = y2 - y1; + const bothRadii = r1 + r2; + return (bothRadii * bothRadii) > (dx * dx + dy * dy); + } + _circleAndRectCollide(circleX, circleY, radius, x1, y1, x2, y2) { + const halfRectWidth = (x2 - x1) / 2; + const distX = Math.abs(circleX - (x1 + halfRectWidth)); + if (distX > (halfRectWidth + radius)) { + return false; + } + const halfRectHeight = (y2 - y1) / 2; + const distY = Math.abs(circleY - (y1 + halfRectHeight)); + if (distY > (halfRectHeight + radius)) { + return false; + } + if (distX <= halfRectWidth || distY <= halfRectHeight) { + return true; + } + const dx = distX - halfRectWidth; + const dy = distY - halfRectHeight; + return (dx * dx + dy * dy <= (radius * radius)); + } +} + +/* + * # Overview of coordinate spaces + * + * ## Tile coordinate spaces + * Each label has an anchor. Some labels have corresponding line geometries. + * The points for both anchors and lines are stored in tile units. Each tile has it's own + * coordinate space going from (0, 0) at the top left to (EXTENT, EXTENT) at the bottom right. + * + * ## GL coordinate space + * At the end of everything, the vertex shader needs to produce a position in GL coordinate space, + * which is (-1, 1) at the top left and (1, -1) in the bottom right. + * + * ## Map pixel coordinate spaces + * Each tile has a pixel coordinate space. It's just the tile units scaled so that one unit is + * whatever counts as 1 pixel at the current zoom. + * This space is used for pitch-alignment=map, rotation-alignment=map + * + * ## Rotated map pixel coordinate spaces + * Like the above, but rotated so axis of the space are aligned with the viewport instead of the tile. + * This space is used for pitch-alignment=map, rotation-alignment=viewport + * + * ## Viewport pixel coordinate space + * (0, 0) is at the top left of the canvas and (pixelWidth, pixelHeight) is at the bottom right corner + * of the canvas. This space is used for pitch-alignment=viewport + * + * + * # Vertex projection + * It goes roughly like this: + * 1. project the anchor and line from tile units into the correct label coordinate space + * - map pixel space pitch-alignment=map rotation-alignment=map + * - rotated map pixel space pitch-alignment=map rotation-alignment=viewport + * - viewport pixel space pitch-alignment=viewport rotation-alignment=* + * 2. if the label follows a line, find the point along the line that is the correct distance from the anchor. + * 3. add the glyph's corner offset to the point from step 3 + * 4. convert from the label coordinate space to gl coordinates + * + * For horizontal labels we want to do step 1 in the shader for performance reasons (no cpu work). + * This is what `u_label_plane_matrix` is used for. + * For labels aligned with lines we have to steps 1 and 2 on the cpu since we need access to the line geometry. + * This is what `updateLineLabels(...)` does. + * Since the conversion is handled on the cpu we just set `u_label_plane_matrix` to an identity matrix. + * + * Steps 3 and 4 are done in the shaders for all labels. + */ +/* + * Returns a matrix for converting from tile units to the correct label coordinate space. + */ +function getLabelPlaneMatrix(posMatrix, pitchWithMap, rotateWithMap, transform, pixelsToTileUnits) { + const m = performance.create(); + if (pitchWithMap) { + performance.scale(m, m, [1 / pixelsToTileUnits, 1 / pixelsToTileUnits, 1]); + if (!rotateWithMap) { + performance.rotateZ(m, m, transform.angle); + } + } + else { + performance.multiply(m, transform.labelPlaneMatrix, posMatrix); + } + return m; +} +/* + * Returns a matrix for converting from the correct label coordinate space to gl coords. + */ +function getGlCoordMatrix(posMatrix, pitchWithMap, rotateWithMap, transform, pixelsToTileUnits) { + if (pitchWithMap) { + const m = performance.clone(posMatrix); + performance.scale(m, m, [pixelsToTileUnits, pixelsToTileUnits, 1]); + if (!rotateWithMap) { + performance.rotateZ(m, m, -transform.angle); + } + return m; + } + else { + return transform.glCoordMatrix; + } +} +function project(point, matrix, getElevation) { + let pos; + if (getElevation) { // slow because of handle z-index + pos = [point.x, point.y, getElevation(point.x, point.y), 1]; + performance.transformMat4(pos, pos, matrix); + } + else { // fast because of ignore z-index + pos = [point.x, point.y, 0, 1]; + xyTransformMat4(pos, pos, matrix); + } + const w = pos[3]; + return { + point: new performance.Point(pos[0] / w, pos[1] / w), + signedDistanceFromCamera: w + }; +} +function getPerspectiveRatio(cameraToCenterDistance, signedDistanceFromCamera) { + return 0.5 + 0.5 * (cameraToCenterDistance / signedDistanceFromCamera); +} +function isVisible(anchorPos, clippingBuffer) { + const x = anchorPos[0] / anchorPos[3]; + const y = anchorPos[1] / anchorPos[3]; + const inPaddedViewport = (x >= -clippingBuffer[0] && + x <= clippingBuffer[0] && + y >= -clippingBuffer[1] && + y <= clippingBuffer[1]); + return inPaddedViewport; +} +/* + * Update the `dynamicLayoutVertexBuffer` for the buffer with the correct glyph positions for the current map view. + * This is only run on labels that are aligned with lines. Horizontal labels are handled entirely in the shader. + */ +function updateLineLabels(bucket, posMatrix, painter, isText, labelPlaneMatrix, glCoordMatrix, pitchWithMap, keepUpright, rotateToLine, getElevation) { + const sizeData = isText ? bucket.textSizeData : bucket.iconSizeData; + const partiallyEvaluatedSize = performance.evaluateSizeForZoom(sizeData, painter.transform.zoom); + const clippingBuffer = [256 / painter.width * 2 + 1, 256 / painter.height * 2 + 1]; + const dynamicLayoutVertexArray = isText ? + bucket.text.dynamicLayoutVertexArray : + bucket.icon.dynamicLayoutVertexArray; + dynamicLayoutVertexArray.clear(); + const lineVertexArray = bucket.lineVertexArray; + const placedSymbols = isText ? bucket.text.placedSymbolArray : bucket.icon.placedSymbolArray; + const aspectRatio = painter.transform.width / painter.transform.height; + let useVertical = false; + for (let s = 0; s < placedSymbols.length; s++) { + const symbol = placedSymbols.get(s); + // Don't do calculations for vertical glyphs unless the previous symbol was horizontal + // and we determined that vertical glyphs were necessary. + // Also don't do calculations for symbols that are collided and fully faded out + if (symbol.hidden || symbol.writingMode === performance.WritingMode.vertical && !useVertical) { + hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray); + continue; + } + // Awkward... but we're counting on the paired "vertical" symbol coming immediately after its horizontal counterpart + useVertical = false; + let anchorPos; + if (getElevation) { // slow because of handle z-index + anchorPos = [symbol.anchorX, symbol.anchorY, getElevation(symbol.anchorX, symbol.anchorY), 1]; + performance.transformMat4(anchorPos, anchorPos, posMatrix); + } + else { // fast because of ignore z-index + anchorPos = [symbol.anchorX, symbol.anchorY, 0, 1]; + xyTransformMat4(anchorPos, anchorPos, posMatrix); + } + // Don't bother calculating the correct point for invisible labels. + if (!isVisible(anchorPos, clippingBuffer)) { + hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray); + continue; + } + const cameraToAnchorDistance = anchorPos[3]; + const perspectiveRatio = getPerspectiveRatio(painter.transform.cameraToCenterDistance, cameraToAnchorDistance); + const fontSize = performance.evaluateSizeForFeature(sizeData, partiallyEvaluatedSize, symbol); + const pitchScaledFontSize = pitchWithMap ? fontSize / perspectiveRatio : fontSize * perspectiveRatio; + const tileAnchorPoint = new performance.Point(symbol.anchorX, symbol.anchorY); + const anchorPoint = project(tileAnchorPoint, labelPlaneMatrix, getElevation).point; + const projectionCache = { projections: {}, offsets: {} }; + const placeUnflipped = placeGlyphsAlongLine(symbol, pitchScaledFontSize, false /*unflipped*/, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix, bucket.glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation); + useVertical = placeUnflipped.useVertical; + if (placeUnflipped.notEnoughRoom || useVertical || + (placeUnflipped.needsFlipping && + placeGlyphsAlongLine(symbol, pitchScaledFontSize, true /*flipped*/, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix, bucket.glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation).notEnoughRoom)) { + hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray); + } + } + if (isText) { + bucket.text.dynamicLayoutVertexBuffer.updateData(dynamicLayoutVertexArray); + } + else { + bucket.icon.dynamicLayoutVertexBuffer.updateData(dynamicLayoutVertexArray); + } +} +/* + * Place the first and last glyph of a line label, projected to the label plane. + * This function is called both during collision detection (to determine the label's size) + * and during line label rendering (to make sure the label fits on the line geometry with + * the current camera position, which may differ from the position used during collision detection). + * + * Calling this function has the effect of populating the "projectionCache" with all projected + * vertex locations the label will need, making future calls to placeGlyphAlongLine (for all the + * intermediate glyphs) much cheaper. + * + * Returns null if the label can't fit on the geometry + */ +function placeFirstAndLastGlyph(fontScale, glyphOffsetArray, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation) { + const glyphEndIndex = symbol.glyphStartIndex + symbol.numGlyphs; + const lineStartIndex = symbol.lineStartIndex; + const lineEndIndex = symbol.lineStartIndex + symbol.lineLength; + const firstGlyphOffset = glyphOffsetArray.getoffsetX(symbol.glyphStartIndex); + const lastGlyphOffset = glyphOffsetArray.getoffsetX(glyphEndIndex - 1); + const firstPlacedGlyph = placeGlyphAlongLine(fontScale * firstGlyphOffset, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment, lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation); + if (!firstPlacedGlyph) + return null; + const lastPlacedGlyph = placeGlyphAlongLine(fontScale * lastGlyphOffset, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment, lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation); + if (!lastPlacedGlyph) + return null; + return { first: firstPlacedGlyph, last: lastPlacedGlyph }; +} +function requiresOrientationChange(writingMode, firstPoint, lastPoint, aspectRatio) { + if (writingMode === performance.WritingMode.horizontal) { + // On top of choosing whether to flip, choose whether to render this version of the glyphs or the alternate + // vertical glyphs. We can't just filter out vertical glyphs in the horizontal range because the horizontal + // and vertical versions can have slightly different projections which could lead to angles where both or + // neither showed. + const rise = Math.abs(lastPoint.y - firstPoint.y); + const run = Math.abs(lastPoint.x - firstPoint.x) * aspectRatio; + if (rise > run) { + return { useVertical: true }; + } + } + if (writingMode === performance.WritingMode.vertical ? firstPoint.y < lastPoint.y : firstPoint.x > lastPoint.x) { + // Includes "horizontalOnly" case for labels without vertical glyphs + return { needsFlipping: true }; + } + return null; +} +/* +* Place first and last glyph along the line projected to label plane, and if they fit +* iterate through all the intermediate glyphs, calculating their label plane positions +* from the projected line. +* +* Finally, add resulting glyph position calculations to dynamicLayoutVertexArray for +* upload to the GPU +*/ +function placeGlyphsAlongLine(symbol, fontSize, flip, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix, glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation) { + const fontScale = fontSize / 24; + const lineOffsetX = symbol.lineOffsetX * fontScale; + const lineOffsetY = symbol.lineOffsetY * fontScale; + let placedGlyphs; + if (symbol.numGlyphs > 1) { + const glyphEndIndex = symbol.glyphStartIndex + symbol.numGlyphs; + const lineStartIndex = symbol.lineStartIndex; + const lineEndIndex = symbol.lineStartIndex + symbol.lineLength; + // Place the first and the last glyph in the label first, so we can figure out + // the overall orientation of the label and determine whether it needs to be flipped in keepUpright mode + const firstAndLastGlyph = placeFirstAndLastGlyph(fontScale, glyphOffsetArray, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation); + if (!firstAndLastGlyph) { + return { notEnoughRoom: true }; + } + const firstPoint = project(firstAndLastGlyph.first.point, glCoordMatrix, getElevation).point; + const lastPoint = project(firstAndLastGlyph.last.point, glCoordMatrix, getElevation).point; + if (keepUpright && !flip) { + const orientationChange = requiresOrientationChange(symbol.writingMode, firstPoint, lastPoint, aspectRatio); + if (orientationChange) { + return orientationChange; + } + } + placedGlyphs = [firstAndLastGlyph.first]; + for (let glyphIndex = symbol.glyphStartIndex + 1; glyphIndex < glyphEndIndex - 1; glyphIndex++) { + // Since first and last glyph fit on the line, we're sure that the rest of the glyphs can be placed + placedGlyphs.push(placeGlyphAlongLine(fontScale * glyphOffsetArray.getoffsetX(glyphIndex), lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment, lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation)); + } + placedGlyphs.push(firstAndLastGlyph.last); + } + else { + // Only a single glyph to place + // So, determine whether to flip based on projected angle of the line segment it's on + if (keepUpright && !flip) { + const a = project(tileAnchorPoint, posMatrix, getElevation).point; + const tileVertexIndex = (symbol.lineStartIndex + symbol.segment + 1); + const tileSegmentEnd = new performance.Point(lineVertexArray.getx(tileVertexIndex), lineVertexArray.gety(tileVertexIndex)); + const projectedVertex = project(tileSegmentEnd, posMatrix, getElevation); + // We know the anchor will be in the viewport, but the end of the line segment may be + // behind the plane of the camera, in which case we can use a point at any arbitrary (closer) + // point on the segment. + const b = (projectedVertex.signedDistanceFromCamera > 0) ? + projectedVertex.point : + projectTruncatedLineSegment(tileAnchorPoint, tileSegmentEnd, a, 1, posMatrix, getElevation); + const orientationChange = requiresOrientationChange(symbol.writingMode, a, b, aspectRatio); + if (orientationChange) { + return orientationChange; + } + } + const singleGlyph = placeGlyphAlongLine(fontScale * glyphOffsetArray.getoffsetX(symbol.glyphStartIndex), lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment, symbol.lineStartIndex, symbol.lineStartIndex + symbol.lineLength, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation); + if (!singleGlyph) + return { notEnoughRoom: true }; + placedGlyphs = [singleGlyph]; + } + for (const glyph of placedGlyphs) { + performance.addDynamicAttributes(dynamicLayoutVertexArray, glyph.point, glyph.angle); + } + return {}; +} +function projectTruncatedLineSegment(previousTilePoint, currentTilePoint, previousProjectedPoint, minimumLength, projectionMatrix, getElevation) { + // We are assuming "previousTilePoint" won't project to a point within one unit of the camera plane + // If it did, that would mean our label extended all the way out from within the viewport to a (very distant) + // point near the plane of the camera. We wouldn't be able to render the label anyway once it crossed the + // plane of the camera. + const projectedUnitVertex = project(previousTilePoint.add(previousTilePoint.sub(currentTilePoint)._unit()), projectionMatrix, getElevation).point; + const projectedUnitSegment = previousProjectedPoint.sub(projectedUnitVertex); + return previousProjectedPoint.add(projectedUnitSegment._mult(minimumLength / projectedUnitSegment.mag())); +} +/** + * Transform a vertex from tile coordinates to label plane coordinates + * @param index - index of vertex to project + * @param projectionArgs - necessary data to project a vertex + * @returns the vertex projected to the label plane + */ +function projectVertexToViewport(index, projectionArgs) { + const { projectionCache, lineVertexArray, labelPlaneMatrix, tileAnchorPoint, distanceFromAnchor, getElevation, previousVertex, direction, absOffsetX } = projectionArgs; + if (projectionCache.projections[index]) { + return projectionCache.projections[index]; + } + const currentVertex = new performance.Point(lineVertexArray.getx(index), lineVertexArray.gety(index)); + const projection = project(currentVertex, labelPlaneMatrix, getElevation); + if (projection.signedDistanceFromCamera > 0) { + projectionCache.projections[index] = projection.point; + return projection.point; + } + // The vertex is behind the plane of the camera, so we can't project it + // Instead, we'll create a vertex along the line that's far enough to include the glyph + const previousLineVertexIndex = index - direction; + const previousTilePoint = distanceFromAnchor === 0 ? + tileAnchorPoint : + new performance.Point(lineVertexArray.getx(previousLineVertexIndex), lineVertexArray.gety(previousLineVertexIndex)); + // Don't cache because the new vertex might not be far enough out for future glyphs on the same segment + return projectTruncatedLineSegment(previousTilePoint, currentVertex, previousVertex, absOffsetX - distanceFromAnchor + 1, labelPlaneMatrix, getElevation); +} +/** + * Calculate the normal vector for a line segment + * @param segmentVector - will be mutated as a tiny optimization + * @param offset - magnitude of resulting vector + * @param direction - direction of line traversal + * @returns a normal vector from the segment, with magnitude equal to offset amount + */ +function transformToOffsetNormal(segmentVector, offset, direction) { + return segmentVector._unit()._perp()._mult(offset * direction); +} +/** + * Construct offset line segments for the current segment and the next segment, then extend/shrink + * the segments until they intersect. If the segments are parallel, then they will touch with no modification. + * + * @param index - Index of the current vertex + * @param prevToCurrentOffsetNormal - Normal vector of the line segment from the previous vertex to the current vertex + * @param currentVertex - Current (non-offset) vertex projected to the label plane + * @param lineStartIndex - Beginning index for the line this label is on + * @param lineEndIndex - End index for the line this label is on + * @param offsetPreviousVertex - The previous vertex projected to the label plane, and then offset along the previous segments normal + * @param lineOffsetY - Magnitude of the offset + * @param projectionArgs - Necessary data for tile-to-label-plane projection + * @returns The point at which the current and next line segments intersect, once offset and extended/shrunk to their meeting point + */ +function findOffsetIntersectionPoint(index, prevToCurrentOffsetNormal, currentVertex, lineStartIndex, lineEndIndex, offsetPreviousVertex, lineOffsetY, projectionArgs) { + const { projectionCache, direction } = projectionArgs; + if (projectionCache.offsets[index]) { + return projectionCache.offsets[index]; + } + const offsetCurrentVertex = currentVertex.add(prevToCurrentOffsetNormal); + if (index + direction < lineStartIndex || index + direction >= lineEndIndex) { + // This is the end of the line, no intersection to calculate + projectionCache.offsets[index] = offsetCurrentVertex; + return offsetCurrentVertex; + } + // Offset the vertices for the next segment + const nextVertex = projectVertexToViewport(index + direction, projectionArgs); + const currentToNextOffsetNormal = transformToOffsetNormal(nextVertex.sub(currentVertex), lineOffsetY, direction); + const offsetNextSegmentBegin = currentVertex.add(currentToNextOffsetNormal); + const offsetNextSegmentEnd = nextVertex.add(currentToNextOffsetNormal); + // find the intersection of these two lines + // if the lines are parallel, offsetCurrent/offsetNextBegin will touch + projectionCache.offsets[index] = performance.findLineIntersection(offsetPreviousVertex, offsetCurrentVertex, offsetNextSegmentBegin, offsetNextSegmentEnd) || offsetCurrentVertex; + return projectionCache.offsets[index]; +} +/* + * Place a single glyph along its line, projected into the label plane, by iterating outward + * from the anchor point until the distance traversed in the label plane equals the glyph's + * offsetX. Returns null if the glyph can't fit on the line geometry. + */ +function placeGlyphAlongLine(offsetX, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, anchorSegment, lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation) { + const combinedOffsetX = flip ? + offsetX - lineOffsetX : + offsetX + lineOffsetX; + let direction = combinedOffsetX > 0 ? 1 : -1; + let angle = 0; + if (flip) { + // The label needs to be flipped to keep text upright. + // Iterate in the reverse direction. + direction *= -1; + angle = Math.PI; + } + if (direction < 0) + angle += Math.PI; + let currentIndex = direction > 0 ? + lineStartIndex + anchorSegment : + lineStartIndex + anchorSegment + 1; + let currentVertex = anchorPoint; + let previousVertex = anchorPoint; + // offsetPrev and intersectionPoint are analogous to previousVertex and currentVertex + // but if there's a line offset they are calculated in parallel as projection happens + let offsetIntersectionPoint; + let offsetPreviousVertex; + let distanceFromAnchor = 0; + let currentSegmentDistance = 0; + const absOffsetX = Math.abs(combinedOffsetX); + const pathVertices = []; + let currentLineSegment; + while (distanceFromAnchor + currentSegmentDistance <= absOffsetX) { + currentIndex += direction; + // offset does not fit on the projected line + if (currentIndex < lineStartIndex || currentIndex >= lineEndIndex) + return null; + // accumulate values from last iteration + distanceFromAnchor += currentSegmentDistance; + previousVertex = currentVertex; + offsetPreviousVertex = offsetIntersectionPoint; + const projectionArgs = { + projectionCache, + lineVertexArray, + labelPlaneMatrix, + tileAnchorPoint, + distanceFromAnchor, + getElevation, + previousVertex, + direction, + absOffsetX + }; + // find next vertex in viewport space + currentVertex = projectVertexToViewport(currentIndex, projectionArgs); + if (lineOffsetY === 0) { + // Store vertices for collision detection and update current segment geometry + pathVertices.push(previousVertex); + currentLineSegment = currentVertex.sub(previousVertex); + } + else { + // Calculate the offset for this section + let prevToCurrentOffsetNormal; + const prevToCurrent = currentVertex.sub(previousVertex); + if (prevToCurrent.mag() === 0) { + // We are starting with our anchor point directly on the vertex, so look one vertex ahead + // to calculate a normal + const nextVertex = projectVertexToViewport(currentIndex + direction, projectionArgs); + prevToCurrentOffsetNormal = transformToOffsetNormal(nextVertex.sub(currentVertex), lineOffsetY, direction); + } + else { + prevToCurrentOffsetNormal = transformToOffsetNormal(prevToCurrent, lineOffsetY, direction); + } + // Initialize offsetPrev on our first iteration, after that it will be pre-calculated + if (!offsetPreviousVertex) + offsetPreviousVertex = previousVertex.add(prevToCurrentOffsetNormal); + offsetIntersectionPoint = findOffsetIntersectionPoint(currentIndex, prevToCurrentOffsetNormal, currentVertex, lineStartIndex, lineEndIndex, offsetPreviousVertex, lineOffsetY, projectionArgs); + pathVertices.push(offsetPreviousVertex); + currentLineSegment = offsetIntersectionPoint.sub(offsetPreviousVertex); + } + currentSegmentDistance = currentLineSegment.mag(); + } + // The point is on the current segment. Interpolate to find it. + const segmentInterpolationT = (absOffsetX - distanceFromAnchor) / currentSegmentDistance; + const p = currentLineSegment._mult(segmentInterpolationT)._add(offsetPreviousVertex || previousVertex); + const segmentAngle = angle + Math.atan2(currentVertex.y - previousVertex.y, currentVertex.x - previousVertex.x); + pathVertices.push(p); + return { + point: p, + angle: rotateToLine ? segmentAngle : 0.0, + path: pathVertices + }; +} +const hiddenGlyphAttributes = new Float32Array([-Infinity, -Infinity, 0, -Infinity, -Infinity, 0, -Infinity, -Infinity, 0, -Infinity, -Infinity, 0]); +// Hide them by moving them offscreen. We still need to add them to the buffer +// because the dynamic buffer is paired with a static buffer that doesn't get updated. +function hideGlyphs(num, dynamicLayoutVertexArray) { + for (let i = 0; i < num; i++) { + const offset = dynamicLayoutVertexArray.length; + dynamicLayoutVertexArray.resize(offset + 4); + // Since all hidden glyphs have the same attributes, we can build up the array faster with a single call to Float32Array.set + // for each set of four vertices, instead of calling addDynamicAttributes for each vertex. + dynamicLayoutVertexArray.float32.set(hiddenGlyphAttributes, offset * 3); + } +} +// For line label layout, we're not using z output and our w input is always 1 +// This custom matrix transformation ignores those components to make projection faster +function xyTransformMat4(out, a, m) { + const x = a[0], y = a[1]; + out[0] = m[0] * x + m[4] * y + m[12]; + out[1] = m[1] * x + m[5] * y + m[13]; + out[3] = m[3] * x + m[7] * y + m[15]; + return out; +} + +// When a symbol crosses the edge that causes it to be included in +// collision detection, it will cause changes in the symbols around +// it. This constant specifies how many pixels to pad the edge of +// the viewport for collision detection so that the bulk of the changes +// occur offscreen. Making this constant greater increases label +// stability, but it's expensive. +const viewportPadding = 100; +/** + * @internal + * A collision index used to prevent symbols from overlapping. It keep tracks of + * where previous symbols have been placed and is used to check if a new + * symbol overlaps with any previously added symbols. + * + * There are two steps to insertion: first placeCollisionBox/Circles checks if + * there's room for a symbol, then insertCollisionBox/Circles actually puts the + * symbol in the index. The two step process allows paired symbols to be inserted + * together even if they overlap. + */ +class CollisionIndex { + constructor(transform, grid = new GridIndex(transform.width + 2 * viewportPadding, transform.height + 2 * viewportPadding, 25), ignoredGrid = new GridIndex(transform.width + 2 * viewportPadding, transform.height + 2 * viewportPadding, 25)) { + this.transform = transform; + this.grid = grid; + this.ignoredGrid = ignoredGrid; + this.pitchfactor = Math.cos(transform._pitch) * transform.cameraToCenterDistance; + this.screenRightBoundary = transform.width + viewportPadding; + this.screenBottomBoundary = transform.height + viewportPadding; + this.gridRightBoundary = transform.width + 2 * viewportPadding; + this.gridBottomBoundary = transform.height + 2 * viewportPadding; + this.perspectiveRatioCutoff = 0.6; + } + placeCollisionBox(collisionBox, overlapMode, textPixelRatio, posMatrix, collisionGroupPredicate, getElevation) { + const projectedPoint = this.projectAndGetPerspectiveRatio(posMatrix, collisionBox.anchorPointX, collisionBox.anchorPointY, getElevation); + const tileToViewport = textPixelRatio * projectedPoint.perspectiveRatio; + const tlX = collisionBox.x1 * tileToViewport + projectedPoint.point.x; + const tlY = collisionBox.y1 * tileToViewport + projectedPoint.point.y; + const brX = collisionBox.x2 * tileToViewport + projectedPoint.point.x; + const brY = collisionBox.y2 * tileToViewport + projectedPoint.point.y; + if (!this.isInsideGrid(tlX, tlY, brX, brY) || + (overlapMode !== 'always' && this.grid.hitTest(tlX, tlY, brX, brY, overlapMode, collisionGroupPredicate)) || + projectedPoint.perspectiveRatio < this.perspectiveRatioCutoff) { + return { + box: [], + offscreen: false + }; + } + return { + box: [tlX, tlY, brX, brY], + offscreen: this.isOffscreen(tlX, tlY, brX, brY) + }; + } + placeCollisionCircles(overlapMode, symbol, lineVertexArray, glyphOffsetArray, fontSize, posMatrix, labelPlaneMatrix, labelToScreenMatrix, showCollisionCircles, pitchWithMap, collisionGroupPredicate, circlePixelDiameter, textPixelPadding, getElevation) { + const placedCollisionCircles = []; + const tileUnitAnchorPoint = new performance.Point(symbol.anchorX, symbol.anchorY); + const screenAnchorPoint = project(tileUnitAnchorPoint, posMatrix, getElevation); + const perspectiveRatio = getPerspectiveRatio(this.transform.cameraToCenterDistance, screenAnchorPoint.signedDistanceFromCamera); + const labelPlaneFontSize = pitchWithMap ? fontSize / perspectiveRatio : fontSize * perspectiveRatio; + const labelPlaneFontScale = labelPlaneFontSize / performance.ONE_EM; + const labelPlaneAnchorPoint = project(tileUnitAnchorPoint, labelPlaneMatrix, getElevation).point; + const projectionCache = { projections: {}, offsets: {} }; + const lineOffsetX = symbol.lineOffsetX * labelPlaneFontScale; + const lineOffsetY = symbol.lineOffsetY * labelPlaneFontScale; + const firstAndLastGlyph = placeFirstAndLastGlyph(labelPlaneFontScale, glyphOffsetArray, lineOffsetX, lineOffsetY, + /*flip*/ false, labelPlaneAnchorPoint, tileUnitAnchorPoint, symbol, lineVertexArray, labelPlaneMatrix, projectionCache, false, getElevation); + let collisionDetected = false; + let inGrid = false; + let entirelyOffscreen = true; + if (firstAndLastGlyph) { + const radius = circlePixelDiameter * 0.5 * perspectiveRatio + textPixelPadding; + const screenPlaneMin = new performance.Point(-viewportPadding, -viewportPadding); + const screenPlaneMax = new performance.Point(this.screenRightBoundary, this.screenBottomBoundary); + const interpolator = new PathInterpolator(); + // Construct a projected path from projected line vertices. Anchor points are ignored and removed + const first = firstAndLastGlyph.first; + const last = firstAndLastGlyph.last; + let projectedPath = []; + for (let i = first.path.length - 1; i >= 1; i--) { + projectedPath.push(first.path[i]); + } + for (let i = 1; i < last.path.length; i++) { + projectedPath.push(last.path[i]); + } + // Tolerate a slightly longer distance than one diameter between two adjacent circles + const circleDist = radius * 2.5; + // The path might need to be converted into screen space if a pitched map is used as the label space + if (labelToScreenMatrix) { + const screenSpacePath = projectedPath.map(p => project(p, labelToScreenMatrix, getElevation)); + // Do not try to place collision circles if even of the points is behind the camera. + // This is a plausible scenario with big camera pitch angles + if (screenSpacePath.some(point => point.signedDistanceFromCamera <= 0)) { + projectedPath = []; + } + else { + projectedPath = screenSpacePath.map(p => p.point); + } + } + let segments = []; + if (projectedPath.length > 0) { + // Quickly check if the path is fully inside or outside of the padded collision region. + // For overlapping paths we'll only create collision circles for the visible segments + const minPoint = projectedPath[0].clone(); + const maxPoint = projectedPath[0].clone(); + for (let i = 1; i < projectedPath.length; i++) { + minPoint.x = Math.min(minPoint.x, projectedPath[i].x); + minPoint.y = Math.min(minPoint.y, projectedPath[i].y); + maxPoint.x = Math.max(maxPoint.x, projectedPath[i].x); + maxPoint.y = Math.max(maxPoint.y, projectedPath[i].y); + } + if (minPoint.x >= screenPlaneMin.x && maxPoint.x <= screenPlaneMax.x && + minPoint.y >= screenPlaneMin.y && maxPoint.y <= screenPlaneMax.y) { + // Quad fully visible + segments = [projectedPath]; + } + else if (maxPoint.x < screenPlaneMin.x || minPoint.x > screenPlaneMax.x || + maxPoint.y < screenPlaneMin.y || minPoint.y > screenPlaneMax.y) { + // Not visible + segments = []; + } + else { + segments = performance.clipLine([projectedPath], screenPlaneMin.x, screenPlaneMin.y, screenPlaneMax.x, screenPlaneMax.y); + } + } + for (const seg of segments) { + // interpolate positions for collision circles. Add a small padding to both ends of the segment + interpolator.reset(seg, radius * 0.25); + let numCircles = 0; + if (interpolator.length <= 0.5 * radius) { + numCircles = 1; + } + else { + numCircles = Math.ceil(interpolator.paddedLength / circleDist) + 1; + } + for (let i = 0; i < numCircles; i++) { + const t = i / Math.max(numCircles - 1, 1); + const circlePosition = interpolator.lerp(t); + // add viewport padding to the position and perform initial collision check + const centerX = circlePosition.x + viewportPadding; + const centerY = circlePosition.y + viewportPadding; + placedCollisionCircles.push(centerX, centerY, radius, 0); + const x1 = centerX - radius; + const y1 = centerY - radius; + const x2 = centerX + radius; + const y2 = centerY + radius; + entirelyOffscreen = entirelyOffscreen && this.isOffscreen(x1, y1, x2, y2); + inGrid = inGrid || this.isInsideGrid(x1, y1, x2, y2); + if (overlapMode !== 'always' && this.grid.hitTestCircle(centerX, centerY, radius, overlapMode, collisionGroupPredicate)) { + // Don't early exit if we're showing the debug circles because we still want to calculate + // which circles are in use + collisionDetected = true; + if (!showCollisionCircles) { + return { + circles: [], + offscreen: false, + collisionDetected + }; + } + } + } + } + } + return { + circles: ((!showCollisionCircles && collisionDetected) || !inGrid || perspectiveRatio < this.perspectiveRatioCutoff) ? [] : placedCollisionCircles, + offscreen: entirelyOffscreen, + collisionDetected + }; + } + /** + * Because the geometries in the CollisionIndex are an approximation of the shape of + * symbols on the map, we use the CollisionIndex to look up the symbol part of + * `queryRenderedFeatures`. + */ + queryRenderedSymbols(viewportQueryGeometry) { + if (viewportQueryGeometry.length === 0 || (this.grid.keysLength() === 0 && this.ignoredGrid.keysLength() === 0)) { + return {}; + } + const query = []; + let minX = Infinity; + let minY = Infinity; + let maxX = -Infinity; + let maxY = -Infinity; + for (const point of viewportQueryGeometry) { + const gridPoint = new performance.Point(point.x + viewportPadding, point.y + viewportPadding); + minX = Math.min(minX, gridPoint.x); + minY = Math.min(minY, gridPoint.y); + maxX = Math.max(maxX, gridPoint.x); + maxY = Math.max(maxY, gridPoint.y); + query.push(gridPoint); + } + const features = this.grid.query(minX, minY, maxX, maxY) + .concat(this.ignoredGrid.query(minX, minY, maxX, maxY)); + const seenFeatures = {}; + const result = {}; + for (const feature of features) { + const featureKey = feature.key; + // Skip already seen features. + if (seenFeatures[featureKey.bucketInstanceId] === undefined) { + seenFeatures[featureKey.bucketInstanceId] = {}; + } + if (seenFeatures[featureKey.bucketInstanceId][featureKey.featureIndex]) { + continue; + } + // Check if query intersects with the feature box + // "Collision Circles" for line labels are treated as boxes here + // Since there's no actual collision taking place, the circle vs. square + // distinction doesn't matter as much, and box geometry is easier + // to work with. + const bbox = [ + new performance.Point(feature.x1, feature.y1), + new performance.Point(feature.x2, feature.y1), + new performance.Point(feature.x2, feature.y2), + new performance.Point(feature.x1, feature.y2) + ]; + if (!performance.polygonIntersectsPolygon(query, bbox)) { + continue; + } + seenFeatures[featureKey.bucketInstanceId][featureKey.featureIndex] = true; + if (result[featureKey.bucketInstanceId] === undefined) { + result[featureKey.bucketInstanceId] = []; + } + result[featureKey.bucketInstanceId].push(featureKey.featureIndex); + } + return result; + } + insertCollisionBox(collisionBox, overlapMode, ignorePlacement, bucketInstanceId, featureIndex, collisionGroupID) { + const grid = ignorePlacement ? this.ignoredGrid : this.grid; + const key = { bucketInstanceId, featureIndex, collisionGroupID, overlapMode }; + grid.insert(key, collisionBox[0], collisionBox[1], collisionBox[2], collisionBox[3]); + } + insertCollisionCircles(collisionCircles, overlapMode, ignorePlacement, bucketInstanceId, featureIndex, collisionGroupID) { + const grid = ignorePlacement ? this.ignoredGrid : this.grid; + const key = { bucketInstanceId, featureIndex, collisionGroupID, overlapMode }; + for (let k = 0; k < collisionCircles.length; k += 4) { + grid.insertCircle(key, collisionCircles[k], collisionCircles[k + 1], collisionCircles[k + 2]); + } + } + projectAndGetPerspectiveRatio(posMatrix, x, y, getElevation) { + let p; + if (getElevation) { // slow because of handle z-index + p = [x, y, getElevation(x, y), 1]; + performance.transformMat4(p, p, posMatrix); + } + else { // fast because of ignore z-index + p = [x, y, 0, 1]; + xyTransformMat4(p, p, posMatrix); + } + const a = new performance.Point((((p[0] / p[3] + 1) / 2) * this.transform.width) + viewportPadding, (((-p[1] / p[3] + 1) / 2) * this.transform.height) + viewportPadding); + return { + point: a, + // See perspective ratio comment in symbol_sdf.vertex + // We're doing collision detection in viewport space so we need + // to scale down boxes in the distance + perspectiveRatio: 0.5 + 0.5 * (this.transform.cameraToCenterDistance / p[3]) + }; + } + isOffscreen(x1, y1, x2, y2) { + return x2 < viewportPadding || x1 >= this.screenRightBoundary || y2 < viewportPadding || y1 > this.screenBottomBoundary; + } + isInsideGrid(x1, y1, x2, y2) { + return x2 >= 0 && x1 < this.gridRightBoundary && y2 >= 0 && y1 < this.gridBottomBoundary; + } + /* + * Returns a matrix for transforming collision shapes to viewport coordinate space. + * Use this function to render e.g. collision circles on the screen. + * example transformation: clipPos = glCoordMatrix * viewportMatrix * circle_pos + */ + getViewportMatrix() { + const m = performance.identity([]); + performance.translate(m, m, [-viewportPadding, -viewportPadding, 0.0]); + return m; + } +} + +/** + * Converts a pixel value at a the given zoom level to tile units. + * + * The shaders mostly calculate everything in tile units so style + * properties need to be converted from pixels to tile units using this. + * + * For example, a translation by 30 pixels at zoom 6.5 will be a + * translation by pixelsToTileUnits(30, 6.5) tile units. + * + * @returns value in tile units + */ +function pixelsToTileUnits(tile, pixelValue, z) { + return pixelValue * (performance.EXTENT / (tile.tileSize * Math.pow(2, z - tile.tileID.overscaledZ))); +} + +class OpacityState { + constructor(prevState, increment, placed, skipFade) { + if (prevState) { + this.opacity = Math.max(0, Math.min(1, prevState.opacity + (prevState.placed ? increment : -increment))); + } + else { + this.opacity = (skipFade && placed) ? 1 : 0; + } + this.placed = placed; + } + isHidden() { + return this.opacity === 0 && !this.placed; + } +} +class JointOpacityState { + constructor(prevState, increment, placedText, placedIcon, skipFade) { + this.text = new OpacityState(prevState ? prevState.text : null, increment, placedText, skipFade); + this.icon = new OpacityState(prevState ? prevState.icon : null, increment, placedIcon, skipFade); + } + isHidden() { + return this.text.isHidden() && this.icon.isHidden(); + } +} +class JointPlacement { + constructor(text, icon, skipFade) { + this.text = text; + this.icon = icon; + this.skipFade = skipFade; + } +} +class CollisionCircleArray { + constructor() { + this.invProjMatrix = performance.create(); + this.viewportMatrix = performance.create(); + this.circles = []; + } +} +class RetainedQueryData { + constructor(bucketInstanceId, featureIndex, sourceLayerIndex, bucketIndex, tileID) { + this.bucketInstanceId = bucketInstanceId; + this.featureIndex = featureIndex; + this.sourceLayerIndex = sourceLayerIndex; + this.bucketIndex = bucketIndex; + this.tileID = tileID; + } +} +class CollisionGroups { + constructor(crossSourceCollisions) { + this.crossSourceCollisions = crossSourceCollisions; + this.maxGroupID = 0; + this.collisionGroups = {}; + } + get(sourceID) { + // The predicate/groupID mechanism allows for arbitrary grouping, + // but the current interface defines one source == one group when + // crossSourceCollisions == true. + if (!this.crossSourceCollisions) { + if (!this.collisionGroups[sourceID]) { + const nextGroupID = ++this.maxGroupID; + this.collisionGroups[sourceID] = { + ID: nextGroupID, + predicate: (key) => { + return key.collisionGroupID === nextGroupID; + } + }; + } + return this.collisionGroups[sourceID]; + } + else { + return { ID: 0, predicate: null }; + } + } +} +function calculateVariableLayoutShift(anchor, width, height, textOffset, textBoxScale) { + const { horizontalAlign, verticalAlign } = performance.getAnchorAlignment(anchor); + const shiftX = -(horizontalAlign - 0.5) * width; + const shiftY = -(verticalAlign - 0.5) * height; + return new performance.Point(shiftX + textOffset[0] * textBoxScale, shiftY + textOffset[1] * textBoxScale); +} +function shiftVariableCollisionBox(collisionBox, shiftX, shiftY, rotateWithMap, pitchWithMap, angle) { + const { x1, x2, y1, y2, anchorPointX, anchorPointY } = collisionBox; + const rotatedOffset = new performance.Point(shiftX, shiftY); + if (rotateWithMap) { + rotatedOffset._rotate(pitchWithMap ? angle : -angle); + } + return { + x1: x1 + rotatedOffset.x, + y1: y1 + rotatedOffset.y, + x2: x2 + rotatedOffset.x, + y2: y2 + rotatedOffset.y, + // symbol anchor point stays the same regardless of text-anchor + anchorPointX, + anchorPointY + }; +} +class Placement { + constructor(transform, terrain, fadeDuration, crossSourceCollisions, prevPlacement) { + this.transform = transform.clone(); + this.terrain = terrain; + this.collisionIndex = new CollisionIndex(this.transform); + this.placements = {}; + this.opacities = {}; + this.variableOffsets = {}; + this.stale = false; + this.commitTime = 0; + this.fadeDuration = fadeDuration; + this.retainedQueryData = {}; + this.collisionGroups = new CollisionGroups(crossSourceCollisions); + this.collisionCircleArrays = {}; + this.prevPlacement = prevPlacement; + if (prevPlacement) { + prevPlacement.prevPlacement = undefined; // Only hold on to one placement back + } + this.placedOrientations = {}; + } + getBucketParts(results, styleLayer, tile, sortAcrossTiles) { + const symbolBucket = tile.getBucket(styleLayer); + const bucketFeatureIndex = tile.latestFeatureIndex; + if (!symbolBucket || !bucketFeatureIndex || styleLayer.id !== symbolBucket.layerIds[0]) + return; + const collisionBoxArray = tile.collisionBoxArray; + const layout = symbolBucket.layers[0].layout; + const scale = Math.pow(2, this.transform.zoom - tile.tileID.overscaledZ); + const textPixelRatio = tile.tileSize / performance.EXTENT; + const posMatrix = this.transform.calculatePosMatrix(tile.tileID.toUnwrapped()); + const pitchWithMap = layout.get('text-pitch-alignment') === 'map'; + const rotateWithMap = layout.get('text-rotation-alignment') === 'map'; + const pixelsToTiles = pixelsToTileUnits(tile, 1, this.transform.zoom); + const textLabelPlaneMatrix = getLabelPlaneMatrix(posMatrix, pitchWithMap, rotateWithMap, this.transform, pixelsToTiles); + let labelToScreenMatrix = null; + if (pitchWithMap) { + const glMatrix = getGlCoordMatrix(posMatrix, pitchWithMap, rotateWithMap, this.transform, pixelsToTiles); + labelToScreenMatrix = performance.multiply([], this.transform.labelPlaneMatrix, glMatrix); + } + // As long as this placement lives, we have to hold onto this bucket's + // matching FeatureIndex/data for querying purposes + this.retainedQueryData[symbolBucket.bucketInstanceId] = new RetainedQueryData(symbolBucket.bucketInstanceId, bucketFeatureIndex, symbolBucket.sourceLayerIndex, symbolBucket.index, tile.tileID); + const parameters = { + bucket: symbolBucket, + layout, + posMatrix, + textLabelPlaneMatrix, + labelToScreenMatrix, + scale, + textPixelRatio, + holdingForFade: tile.holdingForFade(), + collisionBoxArray, + partiallyEvaluatedTextSize: performance.evaluateSizeForZoom(symbolBucket.textSizeData, this.transform.zoom), + collisionGroup: this.collisionGroups.get(symbolBucket.sourceID) + }; + if (sortAcrossTiles) { + for (const range of symbolBucket.sortKeyRanges) { + const { sortKey, symbolInstanceStart, symbolInstanceEnd } = range; + results.push({ sortKey, symbolInstanceStart, symbolInstanceEnd, parameters }); + } + } + else { + results.push({ + symbolInstanceStart: 0, + symbolInstanceEnd: symbolBucket.symbolInstances.length, + parameters + }); + } + } + attemptAnchorPlacement(textAnchorOffset, textBox, width, height, textBoxScale, rotateWithMap, pitchWithMap, textPixelRatio, posMatrix, collisionGroup, textOverlapMode, symbolInstance, bucket, orientation, iconBox, getElevation) { + const anchor = performance.TextAnchorEnum[textAnchorOffset.textAnchor]; + const textOffset = [textAnchorOffset.textOffset0, textAnchorOffset.textOffset1]; + const shift = calculateVariableLayoutShift(anchor, width, height, textOffset, textBoxScale); + const placedGlyphBoxes = this.collisionIndex.placeCollisionBox(shiftVariableCollisionBox(textBox, shift.x, shift.y, rotateWithMap, pitchWithMap, this.transform.angle), textOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation); + if (iconBox) { + const placedIconBoxes = this.collisionIndex.placeCollisionBox(shiftVariableCollisionBox(iconBox, shift.x, shift.y, rotateWithMap, pitchWithMap, this.transform.angle), textOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation); + if (placedIconBoxes.box.length === 0) + return; + } + if (placedGlyphBoxes.box.length > 0) { + let prevAnchor; + // If this label was placed in the previous placement, record the anchor position + // to allow us to animate the transition + if (this.prevPlacement && + this.prevPlacement.variableOffsets[symbolInstance.crossTileID] && + this.prevPlacement.placements[symbolInstance.crossTileID] && + this.prevPlacement.placements[symbolInstance.crossTileID].text) { + prevAnchor = this.prevPlacement.variableOffsets[symbolInstance.crossTileID].anchor; + } + if (symbolInstance.crossTileID === 0) + throw new Error('symbolInstance.crossTileID can\'t be 0'); + this.variableOffsets[symbolInstance.crossTileID] = { + textOffset, + width, + height, + anchor, + textBoxScale, + prevAnchor + }; + this.markUsedJustification(bucket, anchor, symbolInstance, orientation); + if (bucket.allowVerticalPlacement) { + this.markUsedOrientation(bucket, orientation, symbolInstance); + this.placedOrientations[symbolInstance.crossTileID] = orientation; + } + return { shift, placedGlyphBoxes }; + } + } + placeLayerBucketPart(bucketPart, seenCrossTileIDs, showCollisionBoxes) { + const { bucket, layout, posMatrix, textLabelPlaneMatrix, labelToScreenMatrix, textPixelRatio, holdingForFade, collisionBoxArray, partiallyEvaluatedTextSize, collisionGroup } = bucketPart.parameters; + const textOptional = layout.get('text-optional'); + const iconOptional = layout.get('icon-optional'); + const textOverlapMode = performance.getOverlapMode(layout, 'text-overlap', 'text-allow-overlap'); + const textAlwaysOverlap = textOverlapMode === 'always'; + const iconOverlapMode = performance.getOverlapMode(layout, 'icon-overlap', 'icon-allow-overlap'); + const iconAlwaysOverlap = iconOverlapMode === 'always'; + const rotateWithMap = layout.get('text-rotation-alignment') === 'map'; + const pitchWithMap = layout.get('text-pitch-alignment') === 'map'; + const hasIconTextFit = layout.get('icon-text-fit') !== 'none'; + const zOrderByViewportY = layout.get('symbol-z-order') === 'viewport-y'; + // This logic is similar to the "defaultOpacityState" logic below in updateBucketOpacities + // If we know a symbol is always supposed to show, force it to be marked visible even if + // it wasn't placed into the collision index (because some or all of it was outside the range + // of the collision grid). + // There is a subtle edge case here we're accepting: + // Symbol A has text-allow-overlap: true, icon-allow-overlap: true, icon-optional: false + // A's icon is outside the grid, so doesn't get placed + // A's text would be inside grid, but doesn't get placed because of icon-optional: false + // We still show A because of the allow-overlap settings. + // Symbol B has allow-overlap: false, and gets placed where A's text would be + // On panning in, there is a short period when Symbol B and Symbol A will overlap + // This is the reverse of our normal policy of "fade in on pan", but should look like any other + // collision and hopefully not be too noticeable. + // See https://github.com/mapbox/mapbox-gl-js/issues/7172 + const alwaysShowText = textAlwaysOverlap && (iconAlwaysOverlap || !bucket.hasIconData() || iconOptional); + const alwaysShowIcon = iconAlwaysOverlap && (textAlwaysOverlap || !bucket.hasTextData() || textOptional); + if (!bucket.collisionArrays && collisionBoxArray) { + bucket.deserializeCollisionBoxes(collisionBoxArray); + } + const tileID = this.retainedQueryData[bucket.bucketInstanceId].tileID; + const getElevation = this.terrain ? (x, y) => this.terrain.getElevation(tileID, x, y) : null; + const placeSymbol = (symbolInstance, collisionArrays) => { + var _a, _b; + if (seenCrossTileIDs[symbolInstance.crossTileID]) + return; + if (holdingForFade) { + // Mark all symbols from this tile as "not placed", but don't add to seenCrossTileIDs, because we don't + // know yet if we have a duplicate in a parent tile that _should_ be placed. + this.placements[symbolInstance.crossTileID] = new JointPlacement(false, false, false); + return; + } + let placeText = false; + let placeIcon = false; + let offscreen = true; + let shift = null; + let placed = { box: null, offscreen: null }; + let placedVerticalText = { box: null, offscreen: null }; + let placedGlyphBoxes = null; + let placedGlyphCircles = null; + let placedIconBoxes = null; + let textFeatureIndex = 0; + let verticalTextFeatureIndex = 0; + let iconFeatureIndex = 0; + if (collisionArrays.textFeatureIndex) { + textFeatureIndex = collisionArrays.textFeatureIndex; + } + else if (symbolInstance.useRuntimeCollisionCircles) { + textFeatureIndex = symbolInstance.featureIndex; + } + if (collisionArrays.verticalTextFeatureIndex) { + verticalTextFeatureIndex = collisionArrays.verticalTextFeatureIndex; + } + const textBox = collisionArrays.textBox; + if (textBox) { + const updatePreviousOrientationIfNotPlaced = (isPlaced) => { + let previousOrientation = performance.WritingMode.horizontal; + if (bucket.allowVerticalPlacement && !isPlaced && this.prevPlacement) { + const prevPlacedOrientation = this.prevPlacement.placedOrientations[symbolInstance.crossTileID]; + if (prevPlacedOrientation) { + this.placedOrientations[symbolInstance.crossTileID] = prevPlacedOrientation; + previousOrientation = prevPlacedOrientation; + this.markUsedOrientation(bucket, previousOrientation, symbolInstance); + } + } + return previousOrientation; + }; + const placeTextForPlacementModes = (placeHorizontalFn, placeVerticalFn) => { + if (bucket.allowVerticalPlacement && symbolInstance.numVerticalGlyphVertices > 0 && collisionArrays.verticalTextBox) { + for (const placementMode of bucket.writingModes) { + if (placementMode === performance.WritingMode.vertical) { + placed = placeVerticalFn(); + placedVerticalText = placed; + } + else { + placed = placeHorizontalFn(); + } + if (placed && placed.box && placed.box.length) + break; + } + } + else { + placed = placeHorizontalFn(); + } + }; + const textAnchorOffsetStart = symbolInstance.textAnchorOffsetStartIndex; + const textAnchorOffsetEnd = symbolInstance.textAnchorOffsetEndIndex; + // If start+end indices match, text-variable-anchor is not in play. + if (textAnchorOffsetEnd === textAnchorOffsetStart) { + const placeBox = (collisionTextBox, orientation) => { + const placedFeature = this.collisionIndex.placeCollisionBox(collisionTextBox, textOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation); + if (placedFeature && placedFeature.box && placedFeature.box.length) { + this.markUsedOrientation(bucket, orientation, symbolInstance); + this.placedOrientations[symbolInstance.crossTileID] = orientation; + } + return placedFeature; + }; + const placeHorizontal = () => { + return placeBox(textBox, performance.WritingMode.horizontal); + }; + const placeVertical = () => { + const verticalTextBox = collisionArrays.verticalTextBox; + if (bucket.allowVerticalPlacement && symbolInstance.numVerticalGlyphVertices > 0 && verticalTextBox) { + return placeBox(verticalTextBox, performance.WritingMode.vertical); + } + return { box: null, offscreen: null }; + }; + placeTextForPlacementModes(placeHorizontal, placeVertical); + updatePreviousOrientationIfNotPlaced(placed && placed.box && placed.box.length); + } + else { + // If this symbol was in the last placement, prefer placement using same anchor, if it's still available + let prevAnchor = performance.TextAnchorEnum[(_b = (_a = this.prevPlacement) === null || _a === void 0 ? void 0 : _a.variableOffsets[symbolInstance.crossTileID]) === null || _b === void 0 ? void 0 : _b.anchor]; + const placeBoxForVariableAnchors = (collisionTextBox, collisionIconBox, orientation) => { + const width = collisionTextBox.x2 - collisionTextBox.x1; + const height = collisionTextBox.y2 - collisionTextBox.y1; + const textBoxScale = symbolInstance.textBoxScale; + const variableIconBox = hasIconTextFit && (iconOverlapMode === 'never') ? collisionIconBox : null; + let placedBox = { box: [], offscreen: false }; + let placementPasses = (textOverlapMode === 'never') ? 1 : 2; + let overlapMode = 'never'; + if (prevAnchor) { + placementPasses++; + } + for (let pass = 0; pass < placementPasses; pass++) { + for (let i = textAnchorOffsetStart; i < textAnchorOffsetEnd; i++) { + const textAnchorOffset = bucket.textAnchorOffsets.get(i); + if (prevAnchor && textAnchorOffset.textAnchor !== prevAnchor) { + continue; + } + const result = this.attemptAnchorPlacement(textAnchorOffset, collisionTextBox, width, height, textBoxScale, rotateWithMap, pitchWithMap, textPixelRatio, posMatrix, collisionGroup, overlapMode, symbolInstance, bucket, orientation, variableIconBox, getElevation); + if (result) { + placedBox = result.placedGlyphBoxes; + if (placedBox && placedBox.box && placedBox.box.length) { + placeText = true; + shift = result.shift; + return placedBox; + } + } + } + if (prevAnchor) { + prevAnchor = null; + } + else { + overlapMode = textOverlapMode; + } + } + return placedBox; + }; + const placeHorizontal = () => { + return placeBoxForVariableAnchors(textBox, collisionArrays.iconBox, performance.WritingMode.horizontal); + }; + const placeVertical = () => { + const verticalTextBox = collisionArrays.verticalTextBox; + const wasPlaced = placed && placed.box && placed.box.length; + if (bucket.allowVerticalPlacement && !wasPlaced && symbolInstance.numVerticalGlyphVertices > 0 && verticalTextBox) { + return placeBoxForVariableAnchors(verticalTextBox, collisionArrays.verticalIconBox, performance.WritingMode.vertical); + } + return { box: null, offscreen: null }; + }; + placeTextForPlacementModes(placeHorizontal, placeVertical); + if (placed) { + placeText = placed.box; + offscreen = placed.offscreen; + } + const prevOrientation = updatePreviousOrientationIfNotPlaced(placed && placed.box); + // If we didn't get placed, we still need to copy our position from the last placement for + // fade animations + if (!placeText && this.prevPlacement) { + const prevOffset = this.prevPlacement.variableOffsets[symbolInstance.crossTileID]; + if (prevOffset) { + this.variableOffsets[symbolInstance.crossTileID] = prevOffset; + this.markUsedJustification(bucket, prevOffset.anchor, symbolInstance, prevOrientation); + } + } + } + } + placedGlyphBoxes = placed; + placeText = placedGlyphBoxes && placedGlyphBoxes.box && placedGlyphBoxes.box.length > 0; + offscreen = placedGlyphBoxes && placedGlyphBoxes.offscreen; + if (symbolInstance.useRuntimeCollisionCircles) { + const placedSymbol = bucket.text.placedSymbolArray.get(symbolInstance.centerJustifiedTextSymbolIndex); + const fontSize = performance.evaluateSizeForFeature(bucket.textSizeData, partiallyEvaluatedTextSize, placedSymbol); + const textPixelPadding = layout.get('text-padding'); + const circlePixelDiameter = symbolInstance.collisionCircleDiameter; + placedGlyphCircles = this.collisionIndex.placeCollisionCircles(textOverlapMode, placedSymbol, bucket.lineVertexArray, bucket.glyphOffsetArray, fontSize, posMatrix, textLabelPlaneMatrix, labelToScreenMatrix, showCollisionBoxes, pitchWithMap, collisionGroup.predicate, circlePixelDiameter, textPixelPadding, getElevation); + if (placedGlyphCircles.circles.length && placedGlyphCircles.collisionDetected && !showCollisionBoxes) { + performance.warnOnce('Collisions detected, but collision boxes are not shown'); + } + // If text-overlap is set to 'always', force "placedCircles" to true + // In theory there should always be at least one circle placed + // in this case, but for now quirks in text-anchor + // and text-offset may prevent that from being true. + placeText = textAlwaysOverlap || (placedGlyphCircles.circles.length > 0 && !placedGlyphCircles.collisionDetected); + offscreen = offscreen && placedGlyphCircles.offscreen; + } + if (collisionArrays.iconFeatureIndex) { + iconFeatureIndex = collisionArrays.iconFeatureIndex; + } + if (collisionArrays.iconBox) { + const placeIconFeature = iconBox => { + const shiftedIconBox = hasIconTextFit && shift ? + shiftVariableCollisionBox(iconBox, shift.x, shift.y, rotateWithMap, pitchWithMap, this.transform.angle) : + iconBox; + return this.collisionIndex.placeCollisionBox(shiftedIconBox, iconOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation); + }; + if (placedVerticalText && placedVerticalText.box && placedVerticalText.box.length && collisionArrays.verticalIconBox) { + placedIconBoxes = placeIconFeature(collisionArrays.verticalIconBox); + placeIcon = placedIconBoxes.box.length > 0; + } + else { + placedIconBoxes = placeIconFeature(collisionArrays.iconBox); + placeIcon = placedIconBoxes.box.length > 0; + } + offscreen = offscreen && placedIconBoxes.offscreen; + } + const iconWithoutText = textOptional || + (symbolInstance.numHorizontalGlyphVertices === 0 && symbolInstance.numVerticalGlyphVertices === 0); + const textWithoutIcon = iconOptional || symbolInstance.numIconVertices === 0; + // Combine the scales for icons and text. + if (!iconWithoutText && !textWithoutIcon) { + placeIcon = placeText = placeIcon && placeText; + } + else if (!textWithoutIcon) { + placeText = placeIcon && placeText; + } + else if (!iconWithoutText) { + placeIcon = placeIcon && placeText; + } + if (placeText && placedGlyphBoxes && placedGlyphBoxes.box) { + if (placedVerticalText && placedVerticalText.box && verticalTextFeatureIndex) { + this.collisionIndex.insertCollisionBox(placedGlyphBoxes.box, textOverlapMode, layout.get('text-ignore-placement'), bucket.bucketInstanceId, verticalTextFeatureIndex, collisionGroup.ID); + } + else { + this.collisionIndex.insertCollisionBox(placedGlyphBoxes.box, textOverlapMode, layout.get('text-ignore-placement'), bucket.bucketInstanceId, textFeatureIndex, collisionGroup.ID); + } + } + if (placeIcon && placedIconBoxes) { + this.collisionIndex.insertCollisionBox(placedIconBoxes.box, iconOverlapMode, layout.get('icon-ignore-placement'), bucket.bucketInstanceId, iconFeatureIndex, collisionGroup.ID); + } + if (placedGlyphCircles) { + if (placeText) { + this.collisionIndex.insertCollisionCircles(placedGlyphCircles.circles, textOverlapMode, layout.get('text-ignore-placement'), bucket.bucketInstanceId, textFeatureIndex, collisionGroup.ID); + } + if (showCollisionBoxes) { + const id = bucket.bucketInstanceId; + let circleArray = this.collisionCircleArrays[id]; + // Group collision circles together by bucket. Circles can't be pushed forward for rendering yet as the symbol placement + // for a bucket is not guaranteed to be complete before the commit-function has been called + if (circleArray === undefined) + circleArray = this.collisionCircleArrays[id] = new CollisionCircleArray(); + for (let i = 0; i < placedGlyphCircles.circles.length; i += 4) { + circleArray.circles.push(placedGlyphCircles.circles[i + 0]); // x + circleArray.circles.push(placedGlyphCircles.circles[i + 1]); // y + circleArray.circles.push(placedGlyphCircles.circles[i + 2]); // radius + circleArray.circles.push(placedGlyphCircles.collisionDetected ? 1 : 0); // collisionDetected-flag + } + } + } + if (symbolInstance.crossTileID === 0) + throw new Error('symbolInstance.crossTileID can\'t be 0'); + if (bucket.bucketInstanceId === 0) + throw new Error('bucket.bucketInstanceId can\'t be 0'); + this.placements[symbolInstance.crossTileID] = new JointPlacement(placeText || alwaysShowText, placeIcon || alwaysShowIcon, offscreen || bucket.justReloaded); + seenCrossTileIDs[symbolInstance.crossTileID] = true; + }; + if (zOrderByViewportY) { + if (bucketPart.symbolInstanceStart !== 0) + throw new Error('bucket.bucketInstanceId should be 0'); + const symbolIndexes = bucket.getSortedSymbolIndexes(this.transform.angle); + for (let i = symbolIndexes.length - 1; i >= 0; --i) { + const symbolIndex = symbolIndexes[i]; + placeSymbol(bucket.symbolInstances.get(symbolIndex), bucket.collisionArrays[symbolIndex]); + } + } + else { + for (let i = bucketPart.symbolInstanceStart; i < bucketPart.symbolInstanceEnd; i++) { + placeSymbol(bucket.symbolInstances.get(i), bucket.collisionArrays[i]); + } + } + if (showCollisionBoxes && bucket.bucketInstanceId in this.collisionCircleArrays) { + const circleArray = this.collisionCircleArrays[bucket.bucketInstanceId]; + // Store viewport and inverse projection matrices per bucket + performance.invert(circleArray.invProjMatrix, posMatrix); + circleArray.viewportMatrix = this.collisionIndex.getViewportMatrix(); + } + bucket.justReloaded = false; + } + markUsedJustification(bucket, placedAnchor, symbolInstance, orientation) { + const justifications = { + 'left': symbolInstance.leftJustifiedTextSymbolIndex, + 'center': symbolInstance.centerJustifiedTextSymbolIndex, + 'right': symbolInstance.rightJustifiedTextSymbolIndex + }; + let autoIndex; + if (orientation === performance.WritingMode.vertical) { + autoIndex = symbolInstance.verticalPlacedTextSymbolIndex; + } + else { + autoIndex = justifications[performance.getAnchorJustification(placedAnchor)]; + } + const indexes = [ + symbolInstance.leftJustifiedTextSymbolIndex, + symbolInstance.centerJustifiedTextSymbolIndex, + symbolInstance.rightJustifiedTextSymbolIndex, + symbolInstance.verticalPlacedTextSymbolIndex + ]; + for (const index of indexes) { + if (index >= 0) { + if (autoIndex >= 0 && index !== autoIndex) { + // There are multiple justifications and this one isn't it: shift offscreen + bucket.text.placedSymbolArray.get(index).crossTileID = 0; + } + else { + // Either this is the chosen justification or the justification is hardwired: use this one + bucket.text.placedSymbolArray.get(index).crossTileID = symbolInstance.crossTileID; + } + } + } + } + markUsedOrientation(bucket, orientation, symbolInstance) { + const horizontal = (orientation === performance.WritingMode.horizontal || orientation === performance.WritingMode.horizontalOnly) ? orientation : 0; + const vertical = orientation === performance.WritingMode.vertical ? orientation : 0; + const horizontalIndexes = [ + symbolInstance.leftJustifiedTextSymbolIndex, + symbolInstance.centerJustifiedTextSymbolIndex, + symbolInstance.rightJustifiedTextSymbolIndex + ]; + for (const index of horizontalIndexes) { + bucket.text.placedSymbolArray.get(index).placedOrientation = horizontal; + } + if (symbolInstance.verticalPlacedTextSymbolIndex) { + bucket.text.placedSymbolArray.get(symbolInstance.verticalPlacedTextSymbolIndex).placedOrientation = vertical; + } + } + commit(now) { + this.commitTime = now; + this.zoomAtLastRecencyCheck = this.transform.zoom; + const prevPlacement = this.prevPlacement; + let placementChanged = false; + this.prevZoomAdjustment = prevPlacement ? prevPlacement.zoomAdjustment(this.transform.zoom) : 0; + const increment = prevPlacement ? prevPlacement.symbolFadeChange(now) : 1; + const prevOpacities = prevPlacement ? prevPlacement.opacities : {}; + const prevOffsets = prevPlacement ? prevPlacement.variableOffsets : {}; + const prevOrientations = prevPlacement ? prevPlacement.placedOrientations : {}; + // add the opacities from the current placement, and copy their current values from the previous placement + for (const crossTileID in this.placements) { + const jointPlacement = this.placements[crossTileID]; + const prevOpacity = prevOpacities[crossTileID]; + if (prevOpacity) { + this.opacities[crossTileID] = new JointOpacityState(prevOpacity, increment, jointPlacement.text, jointPlacement.icon); + placementChanged = placementChanged || + jointPlacement.text !== prevOpacity.text.placed || + jointPlacement.icon !== prevOpacity.icon.placed; + } + else { + this.opacities[crossTileID] = new JointOpacityState(null, increment, jointPlacement.text, jointPlacement.icon, jointPlacement.skipFade); + placementChanged = placementChanged || jointPlacement.text || jointPlacement.icon; + } + } + // copy and update values from the previous placement that aren't in the current placement but haven't finished fading + for (const crossTileID in prevOpacities) { + const prevOpacity = prevOpacities[crossTileID]; + if (!this.opacities[crossTileID]) { + const jointOpacity = new JointOpacityState(prevOpacity, increment, false, false); + if (!jointOpacity.isHidden()) { + this.opacities[crossTileID] = jointOpacity; + placementChanged = placementChanged || prevOpacity.text.placed || prevOpacity.icon.placed; + } + } + } + for (const crossTileID in prevOffsets) { + if (!this.variableOffsets[crossTileID] && this.opacities[crossTileID] && !this.opacities[crossTileID].isHidden()) { + this.variableOffsets[crossTileID] = prevOffsets[crossTileID]; + } + } + for (const crossTileID in prevOrientations) { + if (!this.placedOrientations[crossTileID] && this.opacities[crossTileID] && !this.opacities[crossTileID].isHidden()) { + this.placedOrientations[crossTileID] = prevOrientations[crossTileID]; + } + } + // this.lastPlacementChangeTime is the time of the last commit() that + // resulted in a placement change -- in other words, the start time of + // the last symbol fade animation + if (prevPlacement && prevPlacement.lastPlacementChangeTime === undefined) { + throw new Error('Last placement time for previous placement is not defined'); + } + if (placementChanged) { + this.lastPlacementChangeTime = now; + } + else if (typeof this.lastPlacementChangeTime !== 'number') { + this.lastPlacementChangeTime = prevPlacement ? prevPlacement.lastPlacementChangeTime : now; + } + } + updateLayerOpacities(styleLayer, tiles) { + const seenCrossTileIDs = {}; + for (const tile of tiles) { + const symbolBucket = tile.getBucket(styleLayer); + if (symbolBucket && tile.latestFeatureIndex && styleLayer.id === symbolBucket.layerIds[0]) { + this.updateBucketOpacities(symbolBucket, seenCrossTileIDs, tile.collisionBoxArray); + } + } + } + updateBucketOpacities(bucket, seenCrossTileIDs, collisionBoxArray) { + if (bucket.hasTextData()) { + bucket.text.opacityVertexArray.clear(); + bucket.text.hasVisibleVertices = false; + } + if (bucket.hasIconData()) { + bucket.icon.opacityVertexArray.clear(); + bucket.icon.hasVisibleVertices = false; + } + if (bucket.hasIconCollisionBoxData()) + bucket.iconCollisionBox.collisionVertexArray.clear(); + if (bucket.hasTextCollisionBoxData()) + bucket.textCollisionBox.collisionVertexArray.clear(); + const layer = bucket.layers[0]; + const layout = layer.layout; + const duplicateOpacityState = new JointOpacityState(null, 0, false, false, true); + const textAllowOverlap = layout.get('text-allow-overlap'); + const iconAllowOverlap = layout.get('icon-allow-overlap'); + const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset'); + const rotateWithMap = layout.get('text-rotation-alignment') === 'map'; + const pitchWithMap = layout.get('text-pitch-alignment') === 'map'; + const hasIconTextFit = layout.get('icon-text-fit') !== 'none'; + // If allow-overlap is true, we can show symbols before placement runs on them + // But we have to wait for placement if we potentially depend on a paired icon/text + // with allow-overlap: false. + // See https://github.com/mapbox/mapbox-gl-js/issues/7032 + const defaultOpacityState = new JointOpacityState(null, 0, textAllowOverlap && (iconAllowOverlap || !bucket.hasIconData() || layout.get('icon-optional')), iconAllowOverlap && (textAllowOverlap || !bucket.hasTextData() || layout.get('text-optional')), true); + if (!bucket.collisionArrays && collisionBoxArray && ((bucket.hasIconCollisionBoxData() || bucket.hasTextCollisionBoxData()))) { + bucket.deserializeCollisionBoxes(collisionBoxArray); + } + const addOpacities = (iconOrText, numVertices, opacity) => { + for (let i = 0; i < numVertices / 4; i++) { + iconOrText.opacityVertexArray.emplaceBack(opacity); + } + iconOrText.hasVisibleVertices = iconOrText.hasVisibleVertices || (opacity !== PACKED_HIDDEN_OPACITY); + }; + for (let s = 0; s < bucket.symbolInstances.length; s++) { + const symbolInstance = bucket.symbolInstances.get(s); + const { numHorizontalGlyphVertices, numVerticalGlyphVertices, crossTileID } = symbolInstance; + const isDuplicate = seenCrossTileIDs[crossTileID]; + let opacityState = this.opacities[crossTileID]; + if (isDuplicate) { + opacityState = duplicateOpacityState; + } + else if (!opacityState) { + opacityState = defaultOpacityState; + // store the state so that future placements use it as a starting point + this.opacities[crossTileID] = opacityState; + } + seenCrossTileIDs[crossTileID] = true; + const hasText = numHorizontalGlyphVertices > 0 || numVerticalGlyphVertices > 0; + const hasIcon = symbolInstance.numIconVertices > 0; + const placedOrientation = this.placedOrientations[symbolInstance.crossTileID]; + const horizontalHidden = placedOrientation === performance.WritingMode.vertical; + const verticalHidden = placedOrientation === performance.WritingMode.horizontal || placedOrientation === performance.WritingMode.horizontalOnly; + if (hasText) { + const packedOpacity = packOpacity(opacityState.text); + // Vertical text fades in/out on collision the same way as corresponding + // horizontal text. Switch between vertical/horizontal should be instantaneous + const horizontalOpacity = horizontalHidden ? PACKED_HIDDEN_OPACITY : packedOpacity; + addOpacities(bucket.text, numHorizontalGlyphVertices, horizontalOpacity); + const verticalOpacity = verticalHidden ? PACKED_HIDDEN_OPACITY : packedOpacity; + addOpacities(bucket.text, numVerticalGlyphVertices, verticalOpacity); + // If this label is completely faded, mark it so that we don't have to calculate + // its position at render time. If this layer has variable placement, shift the various + // symbol instances appropriately so that symbols from buckets that have yet to be placed + // offset appropriately. + const symbolHidden = opacityState.text.isHidden(); + [ + symbolInstance.rightJustifiedTextSymbolIndex, + symbolInstance.centerJustifiedTextSymbolIndex, + symbolInstance.leftJustifiedTextSymbolIndex + ].forEach(index => { + if (index >= 0) { + bucket.text.placedSymbolArray.get(index).hidden = symbolHidden || horizontalHidden ? 1 : 0; + } + }); + if (symbolInstance.verticalPlacedTextSymbolIndex >= 0) { + bucket.text.placedSymbolArray.get(symbolInstance.verticalPlacedTextSymbolIndex).hidden = symbolHidden || verticalHidden ? 1 : 0; + } + const prevOffset = this.variableOffsets[symbolInstance.crossTileID]; + if (prevOffset) { + this.markUsedJustification(bucket, prevOffset.anchor, symbolInstance, placedOrientation); + } + const prevOrientation = this.placedOrientations[symbolInstance.crossTileID]; + if (prevOrientation) { + this.markUsedJustification(bucket, 'left', symbolInstance, prevOrientation); + this.markUsedOrientation(bucket, prevOrientation, symbolInstance); + } + } + if (hasIcon) { + const packedOpacity = packOpacity(opacityState.icon); + const useHorizontal = !(hasIconTextFit && symbolInstance.verticalPlacedIconSymbolIndex && horizontalHidden); + if (symbolInstance.placedIconSymbolIndex >= 0) { + const horizontalOpacity = useHorizontal ? packedOpacity : PACKED_HIDDEN_OPACITY; + addOpacities(bucket.icon, symbolInstance.numIconVertices, horizontalOpacity); + bucket.icon.placedSymbolArray.get(symbolInstance.placedIconSymbolIndex).hidden = + opacityState.icon.isHidden(); + } + if (symbolInstance.verticalPlacedIconSymbolIndex >= 0) { + const verticalOpacity = !useHorizontal ? packedOpacity : PACKED_HIDDEN_OPACITY; + addOpacities(bucket.icon, symbolInstance.numVerticalIconVertices, verticalOpacity); + bucket.icon.placedSymbolArray.get(symbolInstance.verticalPlacedIconSymbolIndex).hidden = + opacityState.icon.isHidden(); + } + } + if (bucket.hasIconCollisionBoxData() || bucket.hasTextCollisionBoxData()) { + const collisionArrays = bucket.collisionArrays[s]; + if (collisionArrays) { + let shift = new performance.Point(0, 0); + if (collisionArrays.textBox || collisionArrays.verticalTextBox) { + let used = true; + if (hasVariablePlacement) { + const variableOffset = this.variableOffsets[crossTileID]; + if (variableOffset) { + // This will show either the currently placed position or the last + // successfully placed position (so you can visualize what collision + // just made the symbol disappear, and the most likely place for the + // symbol to come back) + shift = calculateVariableLayoutShift(variableOffset.anchor, variableOffset.width, variableOffset.height, variableOffset.textOffset, variableOffset.textBoxScale); + if (rotateWithMap) { + shift._rotate(pitchWithMap ? this.transform.angle : -this.transform.angle); + } + } + else { + // No offset -> this symbol hasn't been placed since coming on-screen + // No single box is particularly meaningful and all of them would be too noisy + // Use the center box just to show something's there, but mark it "not used" + used = false; + } + } + if (collisionArrays.textBox) { + updateCollisionVertices(bucket.textCollisionBox.collisionVertexArray, opacityState.text.placed, !used || horizontalHidden, shift.x, shift.y); + } + if (collisionArrays.verticalTextBox) { + updateCollisionVertices(bucket.textCollisionBox.collisionVertexArray, opacityState.text.placed, !used || verticalHidden, shift.x, shift.y); + } + } + const verticalIconUsed = Boolean(!verticalHidden && collisionArrays.verticalIconBox); + if (collisionArrays.iconBox) { + updateCollisionVertices(bucket.iconCollisionBox.collisionVertexArray, opacityState.icon.placed, verticalIconUsed, hasIconTextFit ? shift.x : 0, hasIconTextFit ? shift.y : 0); + } + if (collisionArrays.verticalIconBox) { + updateCollisionVertices(bucket.iconCollisionBox.collisionVertexArray, opacityState.icon.placed, !verticalIconUsed, hasIconTextFit ? shift.x : 0, hasIconTextFit ? shift.y : 0); + } + } + } + } + bucket.sortFeatures(this.transform.angle); + if (this.retainedQueryData[bucket.bucketInstanceId]) { + this.retainedQueryData[bucket.bucketInstanceId].featureSortOrder = bucket.featureSortOrder; + } + if (bucket.hasTextData() && bucket.text.opacityVertexBuffer) { + bucket.text.opacityVertexBuffer.updateData(bucket.text.opacityVertexArray); + } + if (bucket.hasIconData() && bucket.icon.opacityVertexBuffer) { + bucket.icon.opacityVertexBuffer.updateData(bucket.icon.opacityVertexArray); + } + if (bucket.hasIconCollisionBoxData() && bucket.iconCollisionBox.collisionVertexBuffer) { + bucket.iconCollisionBox.collisionVertexBuffer.updateData(bucket.iconCollisionBox.collisionVertexArray); + } + if (bucket.hasTextCollisionBoxData() && bucket.textCollisionBox.collisionVertexBuffer) { + bucket.textCollisionBox.collisionVertexBuffer.updateData(bucket.textCollisionBox.collisionVertexArray); + } + if (bucket.text.opacityVertexArray.length !== bucket.text.layoutVertexArray.length / 4) + throw new Error(`bucket.text.opacityVertexArray.length (= ${bucket.text.opacityVertexArray.length}) !== bucket.text.layoutVertexArray.length (= ${bucket.text.layoutVertexArray.length}) / 4`); + if (bucket.icon.opacityVertexArray.length !== bucket.icon.layoutVertexArray.length / 4) + throw new Error(`bucket.icon.opacityVertexArray.length (= ${bucket.icon.opacityVertexArray.length}) !== bucket.icon.layoutVertexArray.length (= ${bucket.icon.layoutVertexArray.length}) / 4`); + // Push generated collision circles to the bucket for debug rendering + if (bucket.bucketInstanceId in this.collisionCircleArrays) { + const instance = this.collisionCircleArrays[bucket.bucketInstanceId]; + bucket.placementInvProjMatrix = instance.invProjMatrix; + bucket.placementViewportMatrix = instance.viewportMatrix; + bucket.collisionCircleArray = instance.circles; + delete this.collisionCircleArrays[bucket.bucketInstanceId]; + } + } + symbolFadeChange(now) { + return this.fadeDuration === 0 ? + 1 : + ((now - this.commitTime) / this.fadeDuration + this.prevZoomAdjustment); + } + zoomAdjustment(zoom) { + // When zooming out quickly, labels can overlap each other. This + // adjustment is used to reduce the interval between placement calculations + // and to reduce the fade duration when zooming out quickly. Discovering the + // collisions more quickly and fading them more quickly reduces the unwanted effect. + return Math.max(0, (this.transform.zoom - zoom) / 1.5); + } + hasTransitions(now) { + return this.stale || + now - this.lastPlacementChangeTime < this.fadeDuration; + } + stillRecent(now, zoom) { + // The adjustment makes placement more frequent when zooming. + // This condition applies the adjustment only after the map has + // stopped zooming. This avoids adding extra jank while zooming. + const durationAdjustment = this.zoomAtLastRecencyCheck === zoom ? + (1 - this.zoomAdjustment(zoom)) : + 1; + this.zoomAtLastRecencyCheck = zoom; + return this.commitTime + this.fadeDuration * durationAdjustment > now; + } + setStale() { + this.stale = true; + } +} +function updateCollisionVertices(collisionVertexArray, placed, notUsed, shiftX, shiftY) { + collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0); + collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0); + collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0); + collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0); +} +// All four vertices for a glyph will have the same opacity state +// So we pack the opacity into a uint8, and then repeat it four times +// to make a single uint32 that we can upload for each glyph in the +// label. +const shift25 = Math.pow(2, 25); +const shift24 = Math.pow(2, 24); +const shift17 = Math.pow(2, 17); +const shift16 = Math.pow(2, 16); +const shift9 = Math.pow(2, 9); +const shift8 = Math.pow(2, 8); +const shift1 = Math.pow(2, 1); +function packOpacity(opacityState) { + if (opacityState.opacity === 0 && !opacityState.placed) { + return 0; + } + else if (opacityState.opacity === 1 && opacityState.placed) { + return 4294967295; + } + const targetBit = opacityState.placed ? 1 : 0; + const opacityBits = Math.floor(opacityState.opacity * 127); + return opacityBits * shift25 + targetBit * shift24 + + opacityBits * shift17 + targetBit * shift16 + + opacityBits * shift9 + targetBit * shift8 + + opacityBits * shift1 + targetBit; +} +const PACKED_HIDDEN_OPACITY = 0; + +class LayerPlacement { + constructor(styleLayer) { + this._sortAcrossTiles = styleLayer.layout.get('symbol-z-order') !== 'viewport-y' && + !styleLayer.layout.get('symbol-sort-key').isConstant(); + this._currentTileIndex = 0; + this._currentPartIndex = 0; + this._seenCrossTileIDs = {}; + this._bucketParts = []; + } + continuePlacement(tiles, placement, showCollisionBoxes, styleLayer, shouldPausePlacement) { + const bucketParts = this._bucketParts; + while (this._currentTileIndex < tiles.length) { + const tile = tiles[this._currentTileIndex]; + placement.getBucketParts(bucketParts, styleLayer, tile, this._sortAcrossTiles); + this._currentTileIndex++; + if (shouldPausePlacement()) { + return true; + } + } + if (this._sortAcrossTiles) { + this._sortAcrossTiles = false; + bucketParts.sort((a, b) => a.sortKey - b.sortKey); + } + while (this._currentPartIndex < bucketParts.length) { + const bucketPart = bucketParts[this._currentPartIndex]; + placement.placeLayerBucketPart(bucketPart, this._seenCrossTileIDs, showCollisionBoxes); + this._currentPartIndex++; + if (shouldPausePlacement()) { + return true; + } + } + return false; + } +} +class PauseablePlacement { + constructor(transform, terrain, order, forceFullPlacement, showCollisionBoxes, fadeDuration, crossSourceCollisions, prevPlacement) { + this.placement = new Placement(transform, terrain, fadeDuration, crossSourceCollisions, prevPlacement); + this._currentPlacementIndex = order.length - 1; + this._forceFullPlacement = forceFullPlacement; + this._showCollisionBoxes = showCollisionBoxes; + this._done = false; + } + isDone() { + return this._done; + } + continuePlacement(order, layers, layerTiles) { + const startTime = performance.browser.now(); + const shouldPausePlacement = () => { + return this._forceFullPlacement ? false : (performance.browser.now() - startTime) > 2; + }; + while (this._currentPlacementIndex >= 0) { + const layerId = order[this._currentPlacementIndex]; + const layer = layers[layerId]; + const placementZoom = this.placement.collisionIndex.transform.zoom; + if (layer.type === 'symbol' && + (!layer.minzoom || layer.minzoom <= placementZoom) && + (!layer.maxzoom || layer.maxzoom > placementZoom)) { + if (!this._inProgressLayer) { + this._inProgressLayer = new LayerPlacement(layer); + } + const pausePlacement = this._inProgressLayer.continuePlacement(layerTiles[layer.source], this.placement, this._showCollisionBoxes, layer, shouldPausePlacement); + if (pausePlacement) { + // We didn't finish placing all layers within 2ms, + // but we can keep rendering with a partial placement + // We'll resume here on the next frame + return; + } + delete this._inProgressLayer; + } + this._currentPlacementIndex--; + } + this._done = true; + } + commit(now) { + this.placement.commit(now); + return this.placement; + } +} + +/* + The CrossTileSymbolIndex generally works on the assumption that + a conceptual "unique symbol" can be identified by the text of + the label combined with the anchor point. The goal is to assign + these conceptual "unique symbols" a shared crossTileID that can be + used by Placement to keep fading opacity states consistent and to + deduplicate labels. + + The CrossTileSymbolIndex indexes all the current symbol instances and + their crossTileIDs. When a symbol bucket gets added or updated, the + index assigns a crossTileID to each of it's symbol instances by either + matching it with an existing id or assigning a new one. +*/ +// Round anchor positions to roughly 4 pixel grid +const roundingFactor = 512 / performance.EXTENT / 2; +const KDBUSH_THRESHHOLD = 128; +class TileLayerIndex { + constructor(tileID, symbolInstances, bucketInstanceId) { + this.tileID = tileID; + this.bucketInstanceId = bucketInstanceId; + this._symbolsByKey = {}; + // group the symbolInstances by key + const symbolInstancesByKey = new Map(); + for (let i = 0; i < symbolInstances.length; i++) { + const symbolInstance = symbolInstances.get(i); + const key = symbolInstance.key; + const instances = symbolInstancesByKey.get(key); + if (instances) { + // This tile may have multiple symbol instances with the same key + // Store each one along with its coordinates + instances.push(symbolInstance); + } + else { + symbolInstancesByKey.set(key, [symbolInstance]); + } + } + // index the SymbolInstances in this each bucket + for (const [key, symbols] of symbolInstancesByKey) { + const positions = symbols.map(symbolInstance => ({ x: Math.floor(symbolInstance.anchorX * roundingFactor), y: Math.floor(symbolInstance.anchorY * roundingFactor) })); + const crossTileIDs = symbols.map(v => v.crossTileID); + const entry = { positions, crossTileIDs }; + // once we get too many symbols for a given key, it becomes much faster to index it before queries + if (entry.positions.length > KDBUSH_THRESHHOLD) { + const index = new performance.KDBush(entry.positions.length, 16, Uint16Array); + for (const { x, y } of entry.positions) + index.add(x, y); + index.finish(); + // clear all references to the original positions data + delete entry.positions; + entry.index = index; + } + this._symbolsByKey[key] = entry; + } + } + // Converts the coordinates of the input symbol instance into coordinates that be can compared + // against other symbols in this index. Coordinates are: + // (1) local-tile-based (so after correction we get x,y values relative to our local anchorX/Y) + // (2) converted to the z-scale of this TileLayerIndex + // (3) down-sampled by "roundingFactor" from tile coordinate precision in order to be + // more tolerant of small differences between tiles. + getScaledCoordinates(symbolInstance, childTileID) { + const { x: localX, y: localY, z: localZ } = this.tileID.canonical; + const { x, y, z } = childTileID.canonical; + const zDifference = z - localZ; + const scale = roundingFactor / Math.pow(2, zDifference); + const xWorld = (x * performance.EXTENT + symbolInstance.anchorX) * scale; + const yWorld = (y * performance.EXTENT + symbolInstance.anchorY) * scale; + const xOffset = localX * performance.EXTENT * roundingFactor; + const yOffset = localY * performance.EXTENT * roundingFactor; + const result = { + x: Math.floor(xWorld - xOffset), + y: Math.floor(yWorld - yOffset) + }; + return result; + } + findMatches(symbolInstances, newTileID, zoomCrossTileIDs) { + const tolerance = this.tileID.canonical.z < newTileID.canonical.z ? 1 : Math.pow(2, this.tileID.canonical.z - newTileID.canonical.z); + for (let i = 0; i < symbolInstances.length; i++) { + const symbolInstance = symbolInstances.get(i); + if (symbolInstance.crossTileID) { + // already has a match, skip + continue; + } + const entry = this._symbolsByKey[symbolInstance.key]; + if (!entry) { + // No symbol with this key in this bucket + continue; + } + const scaledSymbolCoord = this.getScaledCoordinates(symbolInstance, newTileID); + if (entry.index) { + // Return any symbol with the same keys whose coordinates are within 1 + // grid unit. (with a 4px grid, this covers a 12px by 12px area) + const indexes = entry.index.range(scaledSymbolCoord.x - tolerance, scaledSymbolCoord.y - tolerance, scaledSymbolCoord.x + tolerance, scaledSymbolCoord.y + tolerance).sort(); + for (const i of indexes) { + const crossTileID = entry.crossTileIDs[i]; + if (!zoomCrossTileIDs[crossTileID]) { + // Once we've marked ourselves duplicate against this parent symbol, + // don't let any other symbols at the same zoom level duplicate against + // the same parent (see issue #5993) + zoomCrossTileIDs[crossTileID] = true; + symbolInstance.crossTileID = crossTileID; + break; + } + } + } + else if (entry.positions) { + for (let i = 0; i < entry.positions.length; i++) { + const thisTileSymbol = entry.positions[i]; + const crossTileID = entry.crossTileIDs[i]; + // Return any symbol with the same keys whose coordinates are within 1 + // grid unit. (with a 4px grid, this covers a 12px by 12px area) + if (Math.abs(thisTileSymbol.x - scaledSymbolCoord.x) <= tolerance && + Math.abs(thisTileSymbol.y - scaledSymbolCoord.y) <= tolerance && + !zoomCrossTileIDs[crossTileID]) { + // Once we've marked ourselves duplicate against this parent symbol, + // don't let any other symbols at the same zoom level duplicate against + // the same parent (see issue #5993) + zoomCrossTileIDs[crossTileID] = true; + symbolInstance.crossTileID = crossTileID; + break; + } + } + } + } + } + getCrossTileIDsLists() { + return Object.values(this._symbolsByKey).map(({ crossTileIDs }) => crossTileIDs); + } +} +class CrossTileIDs { + constructor() { + this.maxCrossTileID = 0; + } + generate() { + return ++this.maxCrossTileID; + } +} +class CrossTileSymbolLayerIndex { + constructor() { + this.indexes = {}; + this.usedCrossTileIDs = {}; + this.lng = 0; + } + /* + * Sometimes when a user pans across the antimeridian the longitude value gets wrapped. + * To prevent labels from flashing out and in we adjust the tileID values in the indexes + * so that they match the new wrapped version of the map. + */ + handleWrapJump(lng) { + const wrapDelta = Math.round((lng - this.lng) / 360); + if (wrapDelta !== 0) { + for (const zoom in this.indexes) { + const zoomIndexes = this.indexes[zoom]; + const newZoomIndex = {}; + for (const key in zoomIndexes) { + // change the tileID's wrap and add it to a new index + const index = zoomIndexes[key]; + index.tileID = index.tileID.unwrapTo(index.tileID.wrap + wrapDelta); + newZoomIndex[index.tileID.key] = index; + } + this.indexes[zoom] = newZoomIndex; + } + } + this.lng = lng; + } + addBucket(tileID, bucket, crossTileIDs) { + if (this.indexes[tileID.overscaledZ] && + this.indexes[tileID.overscaledZ][tileID.key]) { + if (this.indexes[tileID.overscaledZ][tileID.key].bucketInstanceId === + bucket.bucketInstanceId) { + return false; + } + else { + // We're replacing this bucket with an updated version + // Remove the old bucket's "used crossTileIDs" now so that + // the new bucket can claim them. + // The old index entries themselves stick around until + // 'removeStaleBuckets' is called. + this.removeBucketCrossTileIDs(tileID.overscaledZ, this.indexes[tileID.overscaledZ][tileID.key]); + } + } + for (let i = 0; i < bucket.symbolInstances.length; i++) { + const symbolInstance = bucket.symbolInstances.get(i); + symbolInstance.crossTileID = 0; + } + if (!this.usedCrossTileIDs[tileID.overscaledZ]) { + this.usedCrossTileIDs[tileID.overscaledZ] = {}; + } + const zoomCrossTileIDs = this.usedCrossTileIDs[tileID.overscaledZ]; + for (const zoom in this.indexes) { + const zoomIndexes = this.indexes[zoom]; + if (Number(zoom) > tileID.overscaledZ) { + for (const id in zoomIndexes) { + const childIndex = zoomIndexes[id]; + if (childIndex.tileID.isChildOf(tileID)) { + childIndex.findMatches(bucket.symbolInstances, tileID, zoomCrossTileIDs); + } + } + } + else { + const parentCoord = tileID.scaledTo(Number(zoom)); + const parentIndex = zoomIndexes[parentCoord.key]; + if (parentIndex) { + parentIndex.findMatches(bucket.symbolInstances, tileID, zoomCrossTileIDs); + } + } + } + for (let i = 0; i < bucket.symbolInstances.length; i++) { + const symbolInstance = bucket.symbolInstances.get(i); + if (!symbolInstance.crossTileID) { + // symbol did not match any known symbol, assign a new id + symbolInstance.crossTileID = crossTileIDs.generate(); + zoomCrossTileIDs[symbolInstance.crossTileID] = true; + } + } + if (this.indexes[tileID.overscaledZ] === undefined) { + this.indexes[tileID.overscaledZ] = {}; + } + this.indexes[tileID.overscaledZ][tileID.key] = new TileLayerIndex(tileID, bucket.symbolInstances, bucket.bucketInstanceId); + return true; + } + removeBucketCrossTileIDs(zoom, removedBucket) { + for (const crossTileIDs of removedBucket.getCrossTileIDsLists()) { + for (const crossTileID of crossTileIDs) { + delete this.usedCrossTileIDs[zoom][crossTileID]; + } + } + } + removeStaleBuckets(currentIDs) { + let tilesChanged = false; + for (const z in this.indexes) { + const zoomIndexes = this.indexes[z]; + for (const tileKey in zoomIndexes) { + if (!currentIDs[zoomIndexes[tileKey].bucketInstanceId]) { + this.removeBucketCrossTileIDs(z, zoomIndexes[tileKey]); + delete zoomIndexes[tileKey]; + tilesChanged = true; + } + } + } + return tilesChanged; + } +} +class CrossTileSymbolIndex { + constructor() { + this.layerIndexes = {}; + this.crossTileIDs = new CrossTileIDs(); + this.maxBucketInstanceId = 0; + this.bucketsInCurrentPlacement = {}; + } + addLayer(styleLayer, tiles, lng) { + let layerIndex = this.layerIndexes[styleLayer.id]; + if (layerIndex === undefined) { + layerIndex = this.layerIndexes[styleLayer.id] = new CrossTileSymbolLayerIndex(); + } + let symbolBucketsChanged = false; + const currentBucketIDs = {}; + layerIndex.handleWrapJump(lng); + for (const tile of tiles) { + const symbolBucket = tile.getBucket(styleLayer); + if (!symbolBucket || styleLayer.id !== symbolBucket.layerIds[0]) + continue; + if (!symbolBucket.bucketInstanceId) { + symbolBucket.bucketInstanceId = ++this.maxBucketInstanceId; + } + if (layerIndex.addBucket(tile.tileID, symbolBucket, this.crossTileIDs)) { + symbolBucketsChanged = true; + } + currentBucketIDs[symbolBucket.bucketInstanceId] = true; + } + if (layerIndex.removeStaleBuckets(currentBucketIDs)) { + symbolBucketsChanged = true; + } + return symbolBucketsChanged; + } + pruneUnusedLayers(usedLayers) { + const usedLayerMap = {}; + usedLayers.forEach((usedLayer) => { + usedLayerMap[usedLayer] = true; + }); + for (const layerId in this.layerIndexes) { + if (!usedLayerMap[layerId]) { + delete this.layerIndexes[layerId]; + } + } + } +} + +// We're skipping validation errors with the `source.canvas` identifier in order +// to continue to allow canvas sources to be added at runtime/updated in +// smart setStyle (see https://github.com/mapbox/mapbox-gl-js/pull/6424): +const emitValidationErrors = (evented, errors) => performance.emitValidationErrors(evented, errors && errors.filter(error => error.identifier !== 'source.canvas')); +const supportedDiffOperations = performance.pick(performance.operations, [ + 'addLayer', + 'removeLayer', + 'setPaintProperty', + 'setLayoutProperty', + 'setFilter', + 'addSource', + 'removeSource', + 'setLayerZoomRange', + 'setLight', + 'setTransition', + 'setGeoJSONSourceData', + 'setGlyphs', + 'setSprite', +]); +const ignoredDiffOperations = performance.pick(performance.operations, [ + 'setCenter', + 'setZoom', + 'setBearing', + 'setPitch' +]); +const empty = performance.emptyStyle(); +/** + * The Style base class + */ +class Style extends performance.Evented { + constructor(map, options = {}) { + super(); + this.map = map; + this.dispatcher = new Dispatcher(getGlobalWorkerPool(), this, map._getMapId()); + this.imageManager = new ImageManager(); + this.imageManager.setEventedParent(this); + this.glyphManager = new GlyphManager(map._requestManager, options.localIdeographFontFamily); + this.lineAtlas = new LineAtlas(256, 512); + this.crossTileSymbolIndex = new CrossTileSymbolIndex(); + this._spritesImagesIds = {}; + this._layers = {}; + this._order = []; + this.sourceCaches = {}; + this.zoomHistory = new performance.ZoomHistory(); + this._loaded = false; + this._availableImages = []; + this._resetUpdates(); + this.dispatcher.broadcast('setReferrer', performance.getReferrer()); + const self = this; + this._rtlTextPluginCallback = Style.registerForPluginStateChange((event) => { + const state = { + pluginStatus: event.pluginStatus, + pluginURL: event.pluginURL + }; + self.dispatcher.broadcast('syncRTLPluginState', state, (err, results) => { + performance.triggerPluginCompletionEvent(err); + if (results) { + const allComplete = results.every((elem) => elem); + if (allComplete) { + for (const id in self.sourceCaches) { + const sourceType = self.sourceCaches[id].getSource().type; + if (sourceType === 'vector' || sourceType === 'geojson') { + // Non-vector sources don't have any symbols buckets to reload when the RTL text plugin loads + // They also load more quickly, so they're more likely to have already displaying tiles + // that would be unnecessarily booted by the plugin load event + self.sourceCaches[id].reload(); // Should be a no-op if the plugin loads before any tiles load + } + } + } + } + }); + }); + this.on('data', (event) => { + if (event.dataType !== 'source' || event.sourceDataType !== 'metadata') { + return; + } + const sourceCache = this.sourceCaches[event.sourceId]; + if (!sourceCache) { + return; + } + const source = sourceCache.getSource(); + if (!source || !source.vectorLayerIds) { + return; + } + for (const layerId in this._layers) { + const layer = this._layers[layerId]; + if (layer.source === source.id) { + this._validateLayer(layer); + } + } + }); + } + loadURL(url, options = {}, previousStyle) { + this.fire(new performance.Event('dataloading', { dataType: 'style' })); + options.validate = typeof options.validate === 'boolean' ? + options.validate : true; + const request = this.map._requestManager.transformRequest(url, ResourceType.Style); + this._request = performance.getJSON(request, (error, json) => { + this._request = null; + if (error) { + this.fire(new performance.ErrorEvent(error)); + } + else if (json) { + this._load(json, options, previousStyle); + } + }); + } + loadJSON(json, options = {}, previousStyle) { + this.fire(new performance.Event('dataloading', { dataType: 'style' })); + this._request = performance.browser.frame(() => { + this._request = null; + options.validate = options.validate !== false; + this._load(json, options, previousStyle); + }); + } + loadEmpty() { + this.fire(new performance.Event('dataloading', { dataType: 'style' })); + this._load(empty, { validate: false }); + } + _load(json, options, previousStyle) { + var _a; + const nextState = options.transformStyle ? options.transformStyle(previousStyle, json) : json; + if (options.validate && emitValidationErrors(this, performance.validateStyle(nextState))) { + return; + } + this._loaded = true; + this.stylesheet = nextState; + for (const id in nextState.sources) { + this.addSource(id, nextState.sources[id], { validate: false }); + } + if (nextState.sprite) { + this._loadSprite(nextState.sprite); + } + else { + this.imageManager.setLoaded(true); + } + this.glyphManager.setURL(nextState.glyphs); + this._createLayers(); + this.light = new Light(this.stylesheet.light); + this.map.setTerrain((_a = this.stylesheet.terrain) !== null && _a !== void 0 ? _a : null); + this.fire(new performance.Event('data', { dataType: 'style' })); + this.fire(new performance.Event('style.load')); + } + _createLayers() { + const dereferencedLayers = performance.derefLayers(this.stylesheet.layers); + // Broadcast layers to workers first, so that expensive style processing (createStyleLayer) + // can happen in parallel on both main and worker threads. + this.dispatcher.broadcast('setLayers', dereferencedLayers); + this._order = dereferencedLayers.map((layer) => layer.id); + this._layers = {}; + // reset serialization field, to be populated only when needed + this._serializedLayers = null; + for (const layer of dereferencedLayers) { + const styledLayer = performance.createStyleLayer(layer); + styledLayer.setEventedParent(this, { layer: { id: layer.id } }); + this._layers[layer.id] = styledLayer; + } + } + _loadSprite(sprite, isUpdate = false, completion = undefined) { + this.imageManager.setLoaded(false); + this._spriteRequest = loadSprite(sprite, this.map._requestManager, this.map.getPixelRatio(), (err, images) => { + this._spriteRequest = null; + if (err) { + this.fire(new performance.ErrorEvent(err)); + } + else if (images) { + for (const spriteId in images) { + this._spritesImagesIds[spriteId] = []; + // remove old sprite's loaded images (for the same sprite id) that are not in new sprite + const imagesToRemove = this._spritesImagesIds[spriteId] ? this._spritesImagesIds[spriteId].filter(id => !(id in images)) : []; + for (const id of imagesToRemove) { + this.imageManager.removeImage(id); + this._changedImages[id] = true; + } + for (const id in images[spriteId]) { + // don't prefix images of the "default" sprite + const imageId = spriteId === 'default' ? id : `${spriteId}:${id}`; + // save all the sprite's images' ids to be able to delete them in `removeSprite` + this._spritesImagesIds[spriteId].push(imageId); + if (imageId in this.imageManager.images) { + this.imageManager.updateImage(imageId, images[spriteId][id], false); + } + else { + this.imageManager.addImage(imageId, images[spriteId][id]); + } + if (isUpdate) { + this._changedImages[imageId] = true; + } + } + } + } + this.imageManager.setLoaded(true); + this._availableImages = this.imageManager.listImages(); + if (isUpdate) { + this._changed = true; + } + this.dispatcher.broadcast('setImages', this._availableImages); + this.fire(new performance.Event('data', { dataType: 'style' })); + if (completion) { + completion(err); + } + }); + } + _unloadSprite() { + for (const id of Object.values(this._spritesImagesIds).flat()) { + this.imageManager.removeImage(id); + this._changedImages[id] = true; + } + this._spritesImagesIds = {}; + this._availableImages = this.imageManager.listImages(); + this._changed = true; + this.dispatcher.broadcast('setImages', this._availableImages); + this.fire(new performance.Event('data', { dataType: 'style' })); + } + _validateLayer(layer) { + const sourceCache = this.sourceCaches[layer.source]; + if (!sourceCache) { + return; + } + const sourceLayer = layer.sourceLayer; + if (!sourceLayer) { + return; + } + const source = sourceCache.getSource(); + if (source.type === 'geojson' || (source.vectorLayerIds && source.vectorLayerIds.indexOf(sourceLayer) === -1)) { + this.fire(new performance.ErrorEvent(new Error(`Source layer "${sourceLayer}" ` + + `does not exist on source "${source.id}" ` + + `as specified by style layer "${layer.id}".`))); + } + } + loaded() { + if (!this._loaded) + return false; + if (Object.keys(this._updatedSources).length) + return false; + for (const id in this.sourceCaches) + if (!this.sourceCaches[id].loaded()) + return false; + if (!this.imageManager.isLoaded()) + return false; + return true; + } + /** + * take an array of string IDs, and based on this._layers, generate an array of LayerSpecification + * @param ids - an array of string IDs, for which serialized layers will be generated. If omitted, all serialized layers will be returned + * @returns generated result + */ + _serializeByIds(ids) { + const serializedLayersDictionary = this._serializedAllLayers(); + if (!ids || ids.length === 0) { + return Object.values(serializedLayersDictionary); + } + const serializedLayers = []; + for (const id of ids) { + // this check will skip all custom layers + if (serializedLayersDictionary[id]) { + serializedLayers.push(serializedLayersDictionary[id]); + } + } + return serializedLayers; + } + /** + * Lazy initialization of this._serializedLayers dictionary and return it + * @returns this._serializedLayers dictionary + */ + _serializedAllLayers() { + let serializedLayers = this._serializedLayers; + if (serializedLayers) { + return serializedLayers; + } + serializedLayers = this._serializedLayers = {}; + const allLayerIds = Object.keys(this._layers); + for (const layerId of allLayerIds) { + const layer = this._layers[layerId]; + if (layer.type !== 'custom') { + serializedLayers[layerId] = layer.serialize(); + } + } + return serializedLayers; + } + hasTransitions() { + if (this.light && this.light.hasTransition()) { + return true; + } + for (const id in this.sourceCaches) { + if (this.sourceCaches[id].hasTransition()) { + return true; + } + } + for (const id in this._layers) { + if (this._layers[id].hasTransition()) { + return true; + } + } + return false; + } + _checkLoaded() { + if (!this._loaded) { + throw new Error('Style is not done loading.'); + } + } + /** + * @internal + * Apply queued style updates in a batch and recalculate zoom-dependent paint properties. + */ + update(parameters) { + if (!this._loaded) { + return; + } + const changed = this._changed; + if (this._changed) { + const updatedIds = Object.keys(this._updatedLayers); + const removedIds = Object.keys(this._removedLayers); + if (updatedIds.length || removedIds.length) { + this._updateWorkerLayers(updatedIds, removedIds); + } + for (const id in this._updatedSources) { + const action = this._updatedSources[id]; + if (action === 'reload') { + this._reloadSource(id); + } + else if (action === 'clear') { + this._clearSource(id); + } + else { + throw new Error(`Invalid action ${action}`); + } + } + this._updateTilesForChangedImages(); + this._updateTilesForChangedGlyphs(); + for (const id in this._updatedPaintProps) { + this._layers[id].updateTransitions(parameters); + } + this.light.updateTransitions(parameters); + this._resetUpdates(); + } + const sourcesUsedBefore = {}; + for (const sourceId in this.sourceCaches) { + const sourceCache = this.sourceCaches[sourceId]; + sourcesUsedBefore[sourceId] = sourceCache.used; + sourceCache.used = false; + } + for (const layerId of this._order) { + const layer = this._layers[layerId]; + layer.recalculate(parameters, this._availableImages); + if (!layer.isHidden(parameters.zoom) && layer.source) { + this.sourceCaches[layer.source].used = true; + } + } + for (const sourceId in sourcesUsedBefore) { + const sourceCache = this.sourceCaches[sourceId]; + if (sourcesUsedBefore[sourceId] !== sourceCache.used) { + sourceCache.fire(new performance.Event('data', { sourceDataType: 'visibility', dataType: 'source', sourceId })); + } + } + this.light.recalculate(parameters); + this.z = parameters.zoom; + if (changed) { + this.fire(new performance.Event('data', { dataType: 'style' })); + } + } + /* + * Apply any queued image changes. + */ + _updateTilesForChangedImages() { + const changedImages = Object.keys(this._changedImages); + if (changedImages.length) { + for (const name in this.sourceCaches) { + this.sourceCaches[name].reloadTilesForDependencies(['icons', 'patterns'], changedImages); + } + this._changedImages = {}; + } + } + _updateTilesForChangedGlyphs() { + if (this._glyphsDidChange) { + for (const name in this.sourceCaches) { + this.sourceCaches[name].reloadTilesForDependencies(['glyphs'], ['']); + } + this._glyphsDidChange = false; + } + } + _updateWorkerLayers(updatedIds, removedIds) { + this.dispatcher.broadcast('updateLayers', { + layers: this._serializeByIds(updatedIds), + removedIds + }); + } + _resetUpdates() { + this._changed = false; + this._updatedLayers = {}; + this._removedLayers = {}; + this._updatedSources = {}; + this._updatedPaintProps = {}; + this._changedImages = {}; + this._glyphsDidChange = false; + } + /** + * Update this style's state to match the given style JSON, performing only + * the necessary mutations. + * + * May throw an Error ('Unimplemented: METHOD') if the mapbox-gl-style-spec + * diff algorithm produces an operation that is not supported. + * + * @returns true if any changes were made; false otherwise + */ + setState(nextState, options = {}) { + this._checkLoaded(); + const serializedStyle = this.serialize(); + nextState = options.transformStyle ? options.transformStyle(serializedStyle, nextState) : nextState; + if (emitValidationErrors(this, performance.validateStyle(nextState))) + return false; + nextState = performance.clone$1(nextState); + nextState.layers = performance.derefLayers(nextState.layers); + const changes = performance.diffStyles(serializedStyle, nextState) + .filter(op => !(op.command in ignoredDiffOperations)); + if (changes.length === 0) { + return false; + } + const unimplementedOps = changes.filter(op => !(op.command in supportedDiffOperations)); + if (unimplementedOps.length > 0) { + throw new Error(`Unimplemented: ${unimplementedOps.map(op => op.command).join(', ')}.`); + } + for (const op of changes) { + if (op.command === 'setTransition') { + // `transition` is always read directly off of + // `this.stylesheet`, which we update below + continue; + } + this[op.command].apply(this, op.args); + } + this.stylesheet = nextState; + // reset serialization field, to be populated only when needed + this._serializedLayers = null; + return true; + } + addImage(id, image) { + if (this.getImage(id)) { + return this.fire(new performance.ErrorEvent(new Error(`An image named "${id}" already exists.`))); + } + this.imageManager.addImage(id, image); + this._afterImageUpdated(id); + } + updateImage(id, image) { + this.imageManager.updateImage(id, image); + } + getImage(id) { + return this.imageManager.getImage(id); + } + removeImage(id) { + if (!this.getImage(id)) { + return this.fire(new performance.ErrorEvent(new Error(`An image named "${id}" does not exist.`))); + } + this.imageManager.removeImage(id); + this._afterImageUpdated(id); + } + _afterImageUpdated(id) { + this._availableImages = this.imageManager.listImages(); + this._changedImages[id] = true; + this._changed = true; + this.dispatcher.broadcast('setImages', this._availableImages); + this.fire(new performance.Event('data', { dataType: 'style' })); + } + listImages() { + this._checkLoaded(); + return this.imageManager.listImages(); + } + addSource(id, source, options = {}) { + this._checkLoaded(); + if (this.sourceCaches[id] !== undefined) { + throw new Error(`Source "${id}" already exists.`); + } + if (!source.type) { + throw new Error(`The type property must be defined, but only the following properties were given: ${Object.keys(source).join(', ')}.`); + } + const builtIns = ['vector', 'raster', 'geojson', 'video', 'image']; + const shouldValidate = builtIns.indexOf(source.type) >= 0; + if (shouldValidate && this._validate(performance.validateStyle.source, `sources.${id}`, source, null, options)) + return; + if (this.map && this.map._collectResourceTiming) + source.collectResourceTiming = true; + const sourceCache = this.sourceCaches[id] = new SourceCache(id, source, this.dispatcher); + sourceCache.style = this; + sourceCache.setEventedParent(this, () => ({ + isSourceLoaded: sourceCache.loaded(), + source: sourceCache.serialize(), + sourceId: id + })); + sourceCache.onAdd(this.map); + this._changed = true; + } + /** + * Remove a source from this stylesheet, given its id. + * @param id - id of the source to remove + * @throws if no source is found with the given ID + * @returns `this`. + */ + removeSource(id) { + this._checkLoaded(); + if (this.sourceCaches[id] === undefined) { + throw new Error('There is no source with this ID'); + } + for (const layerId in this._layers) { + if (this._layers[layerId].source === id) { + return this.fire(new performance.ErrorEvent(new Error(`Source "${id}" cannot be removed while layer "${layerId}" is using it.`))); + } + } + const sourceCache = this.sourceCaches[id]; + delete this.sourceCaches[id]; + delete this._updatedSources[id]; + sourceCache.fire(new performance.Event('data', { sourceDataType: 'metadata', dataType: 'source', sourceId: id })); + sourceCache.setEventedParent(null); + sourceCache.onRemove(this.map); + this._changed = true; + } + /** + * Set the data of a GeoJSON source, given its id. + * @param id - id of the source + * @param data - GeoJSON source + */ + setGeoJSONSourceData(id, data) { + this._checkLoaded(); + if (this.sourceCaches[id] === undefined) + throw new Error(`There is no source with this ID=${id}`); + const geojsonSource = this.sourceCaches[id].getSource(); + if (geojsonSource.type !== 'geojson') + throw new Error(`geojsonSource.type is ${geojsonSource.type}, which is !== 'geojson`); + geojsonSource.setData(data); + this._changed = true; + } + /** + * Get a source by ID. + * @param id - ID of the desired source + * @returns source + */ + getSource(id) { + return this.sourceCaches[id] && this.sourceCaches[id].getSource(); + } + /** + * Add a layer to the map style. The layer will be inserted before the layer with + * ID `before`, or appended if `before` is omitted. + * @param layerObject - The style layer to add. + * @param before - ID of an existing layer to insert before + * @param options - Style setter options. + * @returns `this`. + */ + addLayer(layerObject, before, options = {}) { + this._checkLoaded(); + const id = layerObject.id; + if (this.getLayer(id)) { + this.fire(new performance.ErrorEvent(new Error(`Layer "${id}" already exists on this map.`))); + return; + } + let layer; + if (layerObject.type === 'custom') { + if (emitValidationErrors(this, performance.validateCustomStyleLayer(layerObject))) + return; + layer = performance.createStyleLayer(layerObject); + } + else { + if ('source' in layerObject && typeof layerObject.source === 'object') { + this.addSource(id, layerObject.source); + layerObject = performance.clone$1(layerObject); + layerObject = performance.extend(layerObject, { source: id }); + } + // this layer is not in the style.layers array, so we pass an impossible array index + if (this._validate(performance.validateStyle.layer, `layers.${id}`, layerObject, { arrayIndex: -1 }, options)) + return; + layer = performance.createStyleLayer(layerObject); + this._validateLayer(layer); + layer.setEventedParent(this, { layer: { id } }); + } + const index = before ? this._order.indexOf(before) : this._order.length; + if (before && index === -1) { + this.fire(new performance.ErrorEvent(new Error(`Cannot add layer "${id}" before non-existing layer "${before}".`))); + return; + } + this._order.splice(index, 0, id); + this._layerOrderChanged = true; + this._layers[id] = layer; + if (this._removedLayers[id] && layer.source && layer.type !== 'custom') { + // If, in the current batch, we have already removed this layer + // and we are now re-adding it with a different `type`, then we + // need to clear (rather than just reload) the underlying source's + // tiles. Otherwise, tiles marked 'reloading' will have buckets / + // buffers that are set up for the _previous_ version of this + // layer, causing, e.g.: + // https://github.com/mapbox/mapbox-gl-js/issues/3633 + const removed = this._removedLayers[id]; + delete this._removedLayers[id]; + if (removed.type !== layer.type) { + this._updatedSources[layer.source] = 'clear'; + } + else { + this._updatedSources[layer.source] = 'reload'; + this.sourceCaches[layer.source].pause(); + } + } + this._updateLayer(layer); + if (layer.onAdd) { + layer.onAdd(this.map); + } + } + /** + * Moves a layer to a different z-position. The layer will be inserted before the layer with + * ID `before`, or appended if `before` is omitted. + * @param id - ID of the layer to move + * @param before - ID of an existing layer to insert before + */ + moveLayer(id, before) { + this._checkLoaded(); + this._changed = true; + const layer = this._layers[id]; + if (!layer) { + this.fire(new performance.ErrorEvent(new Error(`The layer '${id}' does not exist in the map's style and cannot be moved.`))); + return; + } + if (id === before) { + return; + } + const index = this._order.indexOf(id); + this._order.splice(index, 1); + const newIndex = before ? this._order.indexOf(before) : this._order.length; + if (before && newIndex === -1) { + this.fire(new performance.ErrorEvent(new Error(`Cannot move layer "${id}" before non-existing layer "${before}".`))); + return; + } + this._order.splice(newIndex, 0, id); + this._layerOrderChanged = true; + } + /** + * Remove the layer with the given id from the style. + * + * If no such layer exists, an `error` event is fired. + * + * @param id - id of the layer to remove + * @event `error` - Fired if the layer does not exist + */ + removeLayer(id) { + this._checkLoaded(); + const layer = this._layers[id]; + if (!layer) { + this.fire(new performance.ErrorEvent(new Error(`Cannot remove non-existing layer "${id}".`))); + return; + } + layer.setEventedParent(null); + const index = this._order.indexOf(id); + this._order.splice(index, 1); + this._layerOrderChanged = true; + this._changed = true; + this._removedLayers[id] = layer; + delete this._layers[id]; + if (this._serializedLayers) { + delete this._serializedLayers[id]; + } + delete this._updatedLayers[id]; + delete this._updatedPaintProps[id]; + if (layer.onRemove) { + layer.onRemove(this.map); + } + } + /** + * Return the style layer object with the given `id`. + * + * @param id - id of the desired layer + * @returns a layer, if one with the given `id` exists + */ + getLayer(id) { + return this._layers[id]; + } + /** + * Return the ids of all layers currently in the style, including custom layers, in order. + * + * @returns ids of layers, in order + */ + getLayersOrder() { + return [...this._order]; + } + /** + * Checks if a specific layer is present within the style. + * + * @param id - the id of the desired layer + * @returns a boolean specifying if the given layer is present + */ + hasLayer(id) { + return id in this._layers; + } + setLayerZoomRange(layerId, minzoom, maxzoom) { + this._checkLoaded(); + const layer = this.getLayer(layerId); + if (!layer) { + this.fire(new performance.ErrorEvent(new Error(`Cannot set the zoom range of non-existing layer "${layerId}".`))); + return; + } + if (layer.minzoom === minzoom && layer.maxzoom === maxzoom) + return; + if (minzoom != null) { + layer.minzoom = minzoom; + } + if (maxzoom != null) { + layer.maxzoom = maxzoom; + } + this._updateLayer(layer); + } + setFilter(layerId, filter, options = {}) { + this._checkLoaded(); + const layer = this.getLayer(layerId); + if (!layer) { + this.fire(new performance.ErrorEvent(new Error(`Cannot filter non-existing layer "${layerId}".`))); + return; + } + if (performance.deepEqual(layer.filter, filter)) { + return; + } + if (filter === null || filter === undefined) { + layer.filter = undefined; + this._updateLayer(layer); + return; + } + if (this._validate(performance.validateStyle.filter, `layers.${layer.id}.filter`, filter, null, options)) { + return; + } + layer.filter = performance.clone$1(filter); + this._updateLayer(layer); + } + /** + * Get a layer's filter object + * @param layer - the layer to inspect + * @returns the layer's filter, if any + */ + getFilter(layer) { + return performance.clone$1(this.getLayer(layer).filter); + } + setLayoutProperty(layerId, name, value, options = {}) { + this._checkLoaded(); + const layer = this.getLayer(layerId); + if (!layer) { + this.fire(new performance.ErrorEvent(new Error(`Cannot style non-existing layer "${layerId}".`))); + return; + } + if (performance.deepEqual(layer.getLayoutProperty(name), value)) + return; + layer.setLayoutProperty(name, value, options); + this._updateLayer(layer); + } + /** + * Get a layout property's value from a given layer + * @param layerId - the layer to inspect + * @param name - the name of the layout property + * @returns the property value + */ + getLayoutProperty(layerId, name) { + const layer = this.getLayer(layerId); + if (!layer) { + this.fire(new performance.ErrorEvent(new Error(`Cannot get style of non-existing layer "${layerId}".`))); + return; + } + return layer.getLayoutProperty(name); + } + setPaintProperty(layerId, name, value, options = {}) { + this._checkLoaded(); + const layer = this.getLayer(layerId); + if (!layer) { + this.fire(new performance.ErrorEvent(new Error(`Cannot style non-existing layer "${layerId}".`))); + return; + } + if (performance.deepEqual(layer.getPaintProperty(name), value)) + return; + const requiresRelayout = layer.setPaintProperty(name, value, options); + if (requiresRelayout) { + this._updateLayer(layer); + } + this._changed = true; + this._updatedPaintProps[layerId] = true; + } + getPaintProperty(layer, name) { + return this.getLayer(layer).getPaintProperty(name); + } + setFeatureState(target, state) { + this._checkLoaded(); + const sourceId = target.source; + const sourceLayer = target.sourceLayer; + const sourceCache = this.sourceCaches[sourceId]; + if (sourceCache === undefined) { + this.fire(new performance.ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`))); + return; + } + const sourceType = sourceCache.getSource().type; + if (sourceType === 'geojson' && sourceLayer) { + this.fire(new performance.ErrorEvent(new Error('GeoJSON sources cannot have a sourceLayer parameter.'))); + return; + } + if (sourceType === 'vector' && !sourceLayer) { + this.fire(new performance.ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.'))); + return; + } + if (target.id === undefined) { + this.fire(new performance.ErrorEvent(new Error('The feature id parameter must be provided.'))); + } + sourceCache.setFeatureState(sourceLayer, target.id, state); + } + removeFeatureState(target, key) { + this._checkLoaded(); + const sourceId = target.source; + const sourceCache = this.sourceCaches[sourceId]; + if (sourceCache === undefined) { + this.fire(new performance.ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`))); + return; + } + const sourceType = sourceCache.getSource().type; + const sourceLayer = sourceType === 'vector' ? target.sourceLayer : undefined; + if (sourceType === 'vector' && !sourceLayer) { + this.fire(new performance.ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.'))); + return; + } + if (key && (typeof target.id !== 'string' && typeof target.id !== 'number')) { + this.fire(new performance.ErrorEvent(new Error('A feature id is required to remove its specific state property.'))); + return; + } + sourceCache.removeFeatureState(sourceLayer, target.id, key); + } + getFeatureState(target) { + this._checkLoaded(); + const sourceId = target.source; + const sourceLayer = target.sourceLayer; + const sourceCache = this.sourceCaches[sourceId]; + if (sourceCache === undefined) { + this.fire(new performance.ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`))); + return; + } + const sourceType = sourceCache.getSource().type; + if (sourceType === 'vector' && !sourceLayer) { + this.fire(new performance.ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.'))); + return; + } + if (target.id === undefined) { + this.fire(new performance.ErrorEvent(new Error('The feature id parameter must be provided.'))); + } + return sourceCache.getFeatureState(sourceLayer, target.id); + } + getTransition() { + return performance.extend({ duration: 300, delay: 0 }, this.stylesheet && this.stylesheet.transition); + } + serialize() { + // We return undefined before we're loaded, following the pattern of Map.getStyle() before + // the Style object is initialized. + // Internally, Style._validate() calls Style.serialize() but callers are responsible for + // calling Style._checkLoaded() first if their validation requires the style to be loaded. + if (!this._loaded) + return; + const sources = performance.mapObject(this.sourceCaches, (source) => source.serialize()); + const layers = this._serializeByIds(this._order); + const terrain = this.map.getTerrain() || undefined; + const myStyleSheet = this.stylesheet; + return performance.filterObject({ + version: myStyleSheet.version, + name: myStyleSheet.name, + metadata: myStyleSheet.metadata, + light: myStyleSheet.light, + center: myStyleSheet.center, + zoom: myStyleSheet.zoom, + bearing: myStyleSheet.bearing, + pitch: myStyleSheet.pitch, + sprite: myStyleSheet.sprite, + glyphs: myStyleSheet.glyphs, + transition: myStyleSheet.transition, + sources, + layers, + terrain + }, (value) => { return value !== undefined; }); + } + _updateLayer(layer) { + this._updatedLayers[layer.id] = true; + if (layer.source && !this._updatedSources[layer.source] && + //Skip for raster layers (https://github.com/mapbox/mapbox-gl-js/issues/7865) + this.sourceCaches[layer.source].getSource().type !== 'raster') { + this._updatedSources[layer.source] = 'reload'; + this.sourceCaches[layer.source].pause(); + } + // upon updating, serilized layer dictionary should be reset. + // When needed, it will be populated with the correct copy again. + this._serializedLayers = null; + this._changed = true; + } + _flattenAndSortRenderedFeatures(sourceResults) { + // Feature order is complicated. + // The order between features in two 2D layers is always determined by layer order. + // The order between features in two 3D layers is always determined by depth. + // The order between a feature in a 2D layer and a 3D layer is tricky: + // Most often layer order determines the feature order in this case. If + // a line layer is above a extrusion layer the line feature will be rendered + // above the extrusion. If the line layer is below the extrusion layer, + // it will be rendered below it. + // + // There is a weird case though. + // You have layers in this order: extrusion_layer_a, line_layer, extrusion_layer_b + // Each layer has a feature that overlaps the other features. + // The feature in extrusion_layer_a is closer than the feature in extrusion_layer_b so it is rendered above. + // The feature in line_layer is rendered above extrusion_layer_a. + // This means that that the line_layer feature is above the extrusion_layer_b feature despite + // it being in an earlier layer. + const isLayer3D = layerId => this._layers[layerId].type === 'fill-extrusion'; + const layerIndex = {}; + const features3D = []; + for (let l = this._order.length - 1; l >= 0; l--) { + const layerId = this._order[l]; + if (isLayer3D(layerId)) { + layerIndex[layerId] = l; + for (const sourceResult of sourceResults) { + const layerFeatures = sourceResult[layerId]; + if (layerFeatures) { + for (const featureWrapper of layerFeatures) { + features3D.push(featureWrapper); + } + } + } + } + } + features3D.sort((a, b) => { + return b.intersectionZ - a.intersectionZ; + }); + const features = []; + for (let l = this._order.length - 1; l >= 0; l--) { + const layerId = this._order[l]; + if (isLayer3D(layerId)) { + // add all 3D features that are in or above the current layer + for (let i = features3D.length - 1; i >= 0; i--) { + const topmost3D = features3D[i].feature; + if (layerIndex[topmost3D.layer.id] < l) + break; + features.push(topmost3D); + features3D.pop(); + } + } + else { + for (const sourceResult of sourceResults) { + const layerFeatures = sourceResult[layerId]; + if (layerFeatures) { + for (const featureWrapper of layerFeatures) { + features.push(featureWrapper.feature); + } + } + } + } + } + return features; + } + queryRenderedFeatures(queryGeometry, params, transform) { + if (params && params.filter) { + this._validate(performance.validateStyle.filter, 'queryRenderedFeatures.filter', params.filter, null, params); + } + const includedSources = {}; + if (params && params.layers) { + if (!Array.isArray(params.layers)) { + this.fire(new performance.ErrorEvent(new Error('parameters.layers must be an Array.'))); + return []; + } + for (const layerId of params.layers) { + const layer = this._layers[layerId]; + if (!layer) { + // this layer is not in the style.layers array + this.fire(new performance.ErrorEvent(new Error(`The layer '${layerId}' does not exist in the map's style and cannot be queried for features.`))); + return []; + } + includedSources[layer.source] = true; + } + } + const sourceResults = []; + params.availableImages = this._availableImages; + // LayerSpecification is serialized StyleLayer, and this casting is safe. + const serializedLayers = this._serializedAllLayers(); + for (const id in this.sourceCaches) { + if (params.layers && !includedSources[id]) + continue; + sourceResults.push(queryRenderedFeatures(this.sourceCaches[id], this._layers, serializedLayers, queryGeometry, params, transform)); + } + if (this.placement) { + // If a placement has run, query against its CollisionIndex + // for symbol results, and treat it as an extra source to merge + sourceResults.push(queryRenderedSymbols(this._layers, serializedLayers, this.sourceCaches, queryGeometry, params, this.placement.collisionIndex, this.placement.retainedQueryData)); + } + return this._flattenAndSortRenderedFeatures(sourceResults); + } + querySourceFeatures(sourceID, params) { + if (params && params.filter) { + this._validate(performance.validateStyle.filter, 'querySourceFeatures.filter', params.filter, null, params); + } + const sourceCache = this.sourceCaches[sourceID]; + return sourceCache ? querySourceFeatures(sourceCache, params) : []; + } + addSourceType(name, SourceType, callback) { + if (getSourceType(name)) { + return callback(new Error(`A source type called "${name}" already exists.`)); + } + setSourceType(name, SourceType); + if (!SourceType.workerSourceURL) { + return callback(null, null); + } + this.dispatcher.broadcast('loadWorkerSource', { + name, + url: SourceType.workerSourceURL + }, callback); + } + getLight() { + return this.light.getLight(); + } + setLight(lightOptions, options = {}) { + this._checkLoaded(); + const light = this.light.getLight(); + let _update = false; + for (const key in lightOptions) { + if (!performance.deepEqual(lightOptions[key], light[key])) { + _update = true; + break; + } + } + if (!_update) + return; + const parameters = { + now: performance.browser.now(), + transition: performance.extend({ + duration: 300, + delay: 0 + }, this.stylesheet.transition) + }; + this.light.setLight(lightOptions, options); + this.light.updateTransitions(parameters); + } + _validate(validate, key, value, props, options = {}) { + if (options && options.validate === false) { + return false; + } + return emitValidationErrors(this, validate.call(performance.validateStyle, performance.extend({ + key, + style: this.serialize(), + value, + styleSpec: performance.v8Spec + }, props))); + } + _remove(mapRemoved = true) { + if (this._request) { + this._request.cancel(); + this._request = null; + } + if (this._spriteRequest) { + this._spriteRequest.cancel(); + this._spriteRequest = null; + } + performance.evented.off('pluginStateChange', this._rtlTextPluginCallback); + for (const layerId in this._layers) { + const layer = this._layers[layerId]; + layer.setEventedParent(null); + } + for (const id in this.sourceCaches) { + const sourceCache = this.sourceCaches[id]; + sourceCache.setEventedParent(null); + sourceCache.onRemove(this.map); + } + this.imageManager.setEventedParent(null); + this.setEventedParent(null); + this.dispatcher.remove(mapRemoved); + } + _clearSource(id) { + this.sourceCaches[id].clearTiles(); + } + _reloadSource(id) { + this.sourceCaches[id].resume(); + this.sourceCaches[id].reload(); + } + _updateSources(transform) { + for (const id in this.sourceCaches) { + this.sourceCaches[id].update(transform, this.map.terrain); + } + } + _generateCollisionBoxes() { + for (const id in this.sourceCaches) { + this._reloadSource(id); + } + } + _updatePlacement(transform, showCollisionBoxes, fadeDuration, crossSourceCollisions, forceFullPlacement = false) { + let symbolBucketsChanged = false; + let placementCommitted = false; + const layerTiles = {}; + for (const layerID of this._order) { + const styleLayer = this._layers[layerID]; + if (styleLayer.type !== 'symbol') + continue; + if (!layerTiles[styleLayer.source]) { + const sourceCache = this.sourceCaches[styleLayer.source]; + layerTiles[styleLayer.source] = sourceCache.getRenderableIds(true) + .map((id) => sourceCache.getTileByID(id)) + .sort((a, b) => (b.tileID.overscaledZ - a.tileID.overscaledZ) || (a.tileID.isLessThan(b.tileID) ? -1 : 1)); + } + const layerBucketsChanged = this.crossTileSymbolIndex.addLayer(styleLayer, layerTiles[styleLayer.source], transform.center.lng); + symbolBucketsChanged = symbolBucketsChanged || layerBucketsChanged; + } + this.crossTileSymbolIndex.pruneUnusedLayers(this._order); + // Anything that changes our "in progress" layer and tile indices requires us + // to start over. When we start over, we do a full placement instead of incremental + // to prevent starvation. + // We need to restart placement to keep layer indices in sync. + // Also force full placement when fadeDuration === 0 to ensure that newly loaded + // tiles will fully display symbols in their first frame + forceFullPlacement = forceFullPlacement || this._layerOrderChanged || fadeDuration === 0; + if (forceFullPlacement || !this.pauseablePlacement || (this.pauseablePlacement.isDone() && !this.placement.stillRecent(performance.browser.now(), transform.zoom))) { + this.pauseablePlacement = new PauseablePlacement(transform, this.map.terrain, this._order, forceFullPlacement, showCollisionBoxes, fadeDuration, crossSourceCollisions, this.placement); + this._layerOrderChanged = false; + } + if (this.pauseablePlacement.isDone()) { + // the last placement finished running, but the next one hasn’t + // started yet because of the `stillRecent` check immediately + // above, so mark it stale to ensure that we request another + // render frame + this.placement.setStale(); + } + else { + this.pauseablePlacement.continuePlacement(this._order, this._layers, layerTiles); + if (this.pauseablePlacement.isDone()) { + this.placement = this.pauseablePlacement.commit(performance.browser.now()); + placementCommitted = true; + } + if (symbolBucketsChanged) { + // since the placement gets split over multiple frames it is possible + // these buckets were processed before they were changed and so the + // placement is already stale while it is in progress + this.pauseablePlacement.placement.setStale(); + } + } + if (placementCommitted || symbolBucketsChanged) { + for (const layerID of this._order) { + const styleLayer = this._layers[layerID]; + if (styleLayer.type !== 'symbol') + continue; + this.placement.updateLayerOpacities(styleLayer, layerTiles[styleLayer.source]); + } + } + // needsRender is false when we have just finished a placement that didn't change the visibility of any symbols + const needsRerender = !this.pauseablePlacement.isDone() || this.placement.hasTransitions(performance.browser.now()); + return needsRerender; + } + _releaseSymbolFadeTiles() { + for (const id in this.sourceCaches) { + this.sourceCaches[id].releaseSymbolFadeTiles(); + } + } + // Callbacks from web workers + getImages(mapId, params, callback) { + this.imageManager.getImages(params.icons, callback); + // Apply queued image changes before setting the tile's dependencies so that the tile + // is not reloaded unnecessarily. Without this forced update the reload could happen in cases + // like this one: + // - icons contains "my-image" + // - imageManager.getImages(...) triggers `onstyleimagemissing` + // - the user adds "my-image" within the callback + // - addImage adds "my-image" to this._changedImages + // - the next frame triggers a reload of this tile even though it already has the latest version + this._updateTilesForChangedImages(); + const sourceCache = this.sourceCaches[params.source]; + if (sourceCache) { + sourceCache.setDependencies(params.tileID.key, params.type, params.icons); + } + } + getGlyphs(mapId, params, callback) { + this.glyphManager.getGlyphs(params.stacks, callback); + const sourceCache = this.sourceCaches[params.source]; + if (sourceCache) { + // we are not setting stacks as dependencies since for now + // we just need to know which tiles have glyph dependencies + sourceCache.setDependencies(params.tileID.key, params.type, ['']); + } + } + getResource(mapId, params, callback) { + return performance.makeRequest(params, callback); + } + getGlyphsUrl() { + return this.stylesheet.glyphs || null; + } + setGlyphs(glyphsUrl, options = {}) { + this._checkLoaded(); + if (glyphsUrl && this._validate(performance.validateStyle.glyphs, 'glyphs', glyphsUrl, null, options)) { + return; + } + this._glyphsDidChange = true; + this.stylesheet.glyphs = glyphsUrl; + this.glyphManager.entries = {}; + this.glyphManager.setURL(glyphsUrl); + } + /** + * Add a sprite. + * + * @param id - The id of the desired sprite + * @param url - The url to load the desired sprite from + * @param options - The style setter options + * @param completion - The completion handler + */ + addSprite(id, url, options = {}, completion) { + this._checkLoaded(); + const spriteToAdd = [{ id, url }]; + const updatedSprite = [ + ...coerceSpriteToArray(this.stylesheet.sprite), + ...spriteToAdd + ]; + if (this._validate(performance.validateStyle.sprite, 'sprite', updatedSprite, null, options)) + return; + this.stylesheet.sprite = updatedSprite; + this._loadSprite(spriteToAdd, true, completion); + } + /** + * Remove a sprite by its id. When the last sprite is removed, the whole `this.stylesheet.sprite` object becomes + * `undefined`. This falsy `undefined` value later prevents attempts to load the sprite when it's absent. + * + * @param id - the id of the sprite to remove + */ + removeSprite(id) { + this._checkLoaded(); + const internalSpriteRepresentation = coerceSpriteToArray(this.stylesheet.sprite); + if (!internalSpriteRepresentation.find(sprite => sprite.id === id)) { + this.fire(new performance.ErrorEvent(new Error(`Sprite "${id}" doesn't exists on this map.`))); + return; + } + if (this._spritesImagesIds[id]) { + for (const imageId of this._spritesImagesIds[id]) { + this.imageManager.removeImage(imageId); + this._changedImages[imageId] = true; + } + } + internalSpriteRepresentation.splice(internalSpriteRepresentation.findIndex(sprite => sprite.id === id), 1); + this.stylesheet.sprite = internalSpriteRepresentation.length > 0 ? internalSpriteRepresentation : undefined; + delete this._spritesImagesIds[id]; + this._availableImages = this.imageManager.listImages(); + this._changed = true; + this.dispatcher.broadcast('setImages', this._availableImages); + this.fire(new performance.Event('data', { dataType: 'style' })); + } + /** + * Get the current sprite value. + * + * @returns empty array when no sprite is set; id-url pairs otherwise + */ + getSprite() { + return coerceSpriteToArray(this.stylesheet.sprite); + } + /** + * Set a new value for the style's sprite. + * + * @param sprite - new sprite value + * @param options - style setter options + * @param completion - the completion handler + */ + setSprite(sprite, options = {}, completion) { + this._checkLoaded(); + if (sprite && this._validate(performance.validateStyle.sprite, 'sprite', sprite, null, options)) { + return; + } + this.stylesheet.sprite = sprite; + if (sprite) { + this._loadSprite(sprite, true, completion); + } + else { + this._unloadSprite(); + if (completion) { + completion(null); + } + } + } +} +Style.registerForPluginStateChange = performance.registerForPluginStateChange; + +var posAttributes = performance.createLayout([ + { name: 'a_pos', type: 'Int16', components: 2 } +]); + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var preludeFrag = '#ifdef GL_ES\nprecision mediump float;\n#else\n#if !defined(lowp)\n#define lowp\n#endif\n#if !defined(mediump)\n#define mediump\n#endif\n#if !defined(highp)\n#define highp\n#endif\n#endif\n'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var preludeVert = '#ifdef GL_ES\nprecision highp float;\n#else\n#if !defined(lowp)\n#define lowp\n#endif\n#if !defined(mediump)\n#define mediump\n#endif\n#if !defined(highp)\n#define highp\n#endif\n#endif\nvec2 unpack_float(const float packedValue) {int packedIntValue=int(packedValue);int v0=packedIntValue/256;return vec2(v0,packedIntValue-v0*256);}vec2 unpack_opacity(const float packedOpacity) {int intOpacity=int(packedOpacity)/2;return vec2(float(intOpacity)/127.0,mod(packedOpacity,2.0));}vec4 decode_color(const vec2 encodedColor) {return vec4(unpack_float(encodedColor[0])/255.0,unpack_float(encodedColor[1])/255.0\n);}float unpack_mix_vec2(const vec2 packedValue,const float t) {return mix(packedValue[0],packedValue[1],t);}vec4 unpack_mix_color(const vec4 packedColors,const float t) {vec4 minColor=decode_color(vec2(packedColors[0],packedColors[1]));vec4 maxColor=decode_color(vec2(packedColors[2],packedColors[3]));return mix(minColor,maxColor,t);}vec2 get_pattern_pos(const vec2 pixel_coord_upper,const vec2 pixel_coord_lower,const vec2 pattern_size,const float tile_units_to_pixels,const vec2 pos) {vec2 offset=mod(mod(mod(pixel_coord_upper,pattern_size)*256.0,pattern_size)*256.0+pixel_coord_lower,pattern_size);return (tile_units_to_pixels*pos+offset)/pattern_size;}\n#ifdef TERRAIN3D\nuniform sampler2D u_terrain;uniform float u_terrain_dim;uniform mat4 u_terrain_matrix;uniform vec4 u_terrain_unpack;uniform float u_terrain_exaggeration;uniform highp sampler2D u_depth;\n#endif\nconst highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitShifts=vec4(1.)/bitSh;highp float unpack(highp vec4 color) {return dot(color,bitShifts);}highp float depthOpacity(vec3 frag) {\n#ifdef TERRAIN3D\nhighp float d=unpack(texture2D(u_depth,frag.xy*0.5+0.5))+0.0001-frag.z;return 1.0-max(0.0,min(1.0,-d*500.0));\n#else\nreturn 1.0;\n#endif\n}float calculate_visibility(vec4 pos) {\n#ifdef TERRAIN3D\nvec3 frag=pos.xyz/pos.w;highp float d=depthOpacity(frag);if (d > 0.95) return 1.0;return (d+depthOpacity(frag+vec3(0.0,0.01,0.0)))/2.0;\n#else\nreturn 1.0;\n#endif\n}float ele(vec2 pos) {\n#ifdef TERRAIN3D\nvec4 rgb=(texture2D(u_terrain,pos)*255.0)*u_terrain_unpack;return rgb.r+rgb.g+rgb.b-u_terrain_unpack.a;\n#else\nreturn 0.0;\n#endif\n}float get_elevation(vec2 pos) {\n#ifdef TERRAIN3D\nvec2 coord=(u_terrain_matrix*vec4(pos,0.0,1.0)).xy*u_terrain_dim+1.0;vec2 f=fract(coord);vec2 c=(floor(coord)+0.5)/(u_terrain_dim+2.0);float d=1.0/(u_terrain_dim+2.0);float tl=ele(c);float tr=ele(c+vec2(d,0.0));float bl=ele(c+vec2(0.0,d));float br=ele(c+vec2(d,d));float elevation=mix(mix(tl,tr,f.x),mix(bl,br,f.x),f.y);return elevation*u_terrain_exaggeration;\n#else\nreturn 0.0;\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var backgroundFrag = 'uniform vec4 u_color;uniform float u_opacity;void main() {gl_FragColor=u_color*u_opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var backgroundVert = 'attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var backgroundPatternFrag = 'uniform vec2 u_pattern_tl_a;uniform vec2 u_pattern_br_a;uniform vec2 u_pattern_tl_b;uniform vec2 u_pattern_br_b;uniform vec2 u_texsize;uniform float u_mix;uniform float u_opacity;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(u_pattern_tl_a/u_texsize,u_pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(u_pattern_tl_b/u_texsize,u_pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_mix)*u_opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var backgroundPatternVert = 'uniform mat4 u_matrix;uniform vec2 u_pattern_size_a;uniform vec2 u_pattern_size_b;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_scale_a;uniform float u_scale_b;uniform float u_tile_units_to_pixels;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_a*u_pattern_size_a,u_tile_units_to_pixels,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_b*u_pattern_size_b,u_tile_units_to_pixels,a_pos);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var circleFrag = 'varying vec3 v_data;varying float v_visibility;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define mediump float radius\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define highp vec4 stroke_color\n#pragma mapbox: define mediump float stroke_width\n#pragma mapbox: define lowp float stroke_opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize mediump float radius\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize highp vec4 stroke_color\n#pragma mapbox: initialize mediump float stroke_width\n#pragma mapbox: initialize lowp float stroke_opacity\nvec2 extrude=v_data.xy;float extrude_length=length(extrude);lowp float antialiasblur=v_data.z;float antialiased_blur=-max(blur,antialiasblur);float opacity_t=smoothstep(0.0,antialiased_blur,extrude_length-1.0);float color_t=stroke_width < 0.01 ? 0.0 : smoothstep(antialiased_blur,0.0,extrude_length-radius/(radius+stroke_width));gl_FragColor=v_visibility*opacity_t*mix(color*opacity,stroke_color*stroke_opacity,color_t);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var circleVert = 'uniform mat4 u_matrix;uniform bool u_scale_with_map;uniform bool u_pitch_with_map;uniform vec2 u_extrude_scale;uniform lowp float u_device_pixel_ratio;uniform highp float u_camera_to_center_distance;attribute vec2 a_pos;varying vec3 v_data;varying float v_visibility;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define mediump float radius\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define highp vec4 stroke_color\n#pragma mapbox: define mediump float stroke_width\n#pragma mapbox: define lowp float stroke_opacity\nvoid main(void) {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize mediump float radius\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize highp vec4 stroke_color\n#pragma mapbox: initialize mediump float stroke_width\n#pragma mapbox: initialize lowp float stroke_opacity\nvec2 extrude=vec2(mod(a_pos,2.0)*2.0-1.0);vec2 circle_center=floor(a_pos*0.5);float ele=get_elevation(circle_center);v_visibility=calculate_visibility(u_matrix*vec4(circle_center,ele,1.0));if (u_pitch_with_map) {vec2 corner_position=circle_center;if (u_scale_with_map) {corner_position+=extrude*(radius+stroke_width)*u_extrude_scale;} else {vec4 projected_center=u_matrix*vec4(circle_center,0,1);corner_position+=extrude*(radius+stroke_width)*u_extrude_scale*(projected_center.w/u_camera_to_center_distance);}gl_Position=u_matrix*vec4(corner_position,ele,1);} else {gl_Position=u_matrix*vec4(circle_center,ele,1);if (u_scale_with_map) {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*u_camera_to_center_distance;} else {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*gl_Position.w;}}lowp float antialiasblur=1.0/u_device_pixel_ratio/(radius+stroke_width);v_data=vec3(extrude.x,extrude.y,antialiasblur);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var clippingMaskFrag = 'void main() {gl_FragColor=vec4(1.0);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var clippingMaskVert = 'attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var heatmapFrag = 'uniform highp float u_intensity;varying vec2 v_extrude;\n#pragma mapbox: define highp float weight\n#define GAUSS_COEF 0.3989422804014327\nvoid main() {\n#pragma mapbox: initialize highp float weight\nfloat d=-0.5*3.0*3.0*dot(v_extrude,v_extrude);float val=weight*u_intensity*GAUSS_COEF*exp(d);gl_FragColor=vec4(val,1.0,1.0,1.0);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var heatmapVert = 'uniform mat4 u_matrix;uniform float u_extrude_scale;uniform float u_opacity;uniform float u_intensity;attribute vec2 a_pos;varying vec2 v_extrude;\n#pragma mapbox: define highp float weight\n#pragma mapbox: define mediump float radius\nconst highp float ZERO=1.0/255.0/16.0;\n#define GAUSS_COEF 0.3989422804014327\nvoid main(void) {\n#pragma mapbox: initialize highp float weight\n#pragma mapbox: initialize mediump float radius\nvec2 unscaled_extrude=vec2(mod(a_pos,2.0)*2.0-1.0);float S=sqrt(-2.0*log(ZERO/weight/u_intensity/GAUSS_COEF))/3.0;v_extrude=S*unscaled_extrude;vec2 extrude=v_extrude*radius*u_extrude_scale;vec4 pos=vec4(floor(a_pos*0.5)+extrude,0,1);gl_Position=u_matrix*pos;}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var heatmapTextureFrag = 'uniform sampler2D u_image;uniform sampler2D u_color_ramp;uniform float u_opacity;varying vec2 v_pos;void main() {float t=texture2D(u_image,v_pos).r;vec4 color=texture2D(u_color_ramp,vec2(t,0.5));gl_FragColor=color*u_opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(0.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var heatmapTextureVert = 'uniform mat4 u_matrix;uniform vec2 u_world;attribute vec2 a_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos*u_world,0,1);v_pos.x=a_pos.x;v_pos.y=1.0-a_pos.y;}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var collisionBoxFrag = 'varying float v_placed;varying float v_notUsed;void main() {float alpha=0.5;gl_FragColor=vec4(1.0,0.0,0.0,1.0)*alpha;if (v_placed > 0.5) {gl_FragColor=vec4(0.0,0.0,1.0,0.5)*alpha;}if (v_notUsed > 0.5) {gl_FragColor*=.1;}}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var collisionBoxVert = 'attribute vec2 a_pos;attribute vec2 a_anchor_pos;attribute vec2 a_extrude;attribute vec2 a_placed;attribute vec2 a_shift;uniform mat4 u_matrix;uniform vec2 u_extrude_scale;uniform float u_camera_to_center_distance;varying float v_placed;varying float v_notUsed;void main() {vec4 projectedPoint=u_matrix*vec4(a_anchor_pos,0,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);gl_Position=u_matrix*vec4(a_pos,get_elevation(a_pos),1.0);gl_Position.xy+=(a_extrude+a_shift)*u_extrude_scale*gl_Position.w*collision_perspective_ratio;v_placed=a_placed.x;v_notUsed=a_placed.y;}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var collisionCircleFrag = 'varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;void main() {float alpha=0.5*min(v_perspective_ratio,1.0);float stroke_radius=0.9*max(v_perspective_ratio,1.0);float distance_to_center=length(v_extrude);float distance_to_edge=abs(distance_to_center-v_radius);float opacity_t=smoothstep(-stroke_radius,0.0,-distance_to_edge);vec4 color=mix(vec4(0.0,0.0,1.0,0.5),vec4(1.0,0.0,0.0,1.0),v_collision);gl_FragColor=color*alpha*opacity_t;}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var collisionCircleVert = 'attribute vec2 a_pos;attribute float a_radius;attribute vec2 a_flags;uniform mat4 u_matrix;uniform mat4 u_inv_matrix;uniform vec2 u_viewport_size;uniform float u_camera_to_center_distance;varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;vec3 toTilePosition(vec2 screenPos) {vec4 rayStart=u_inv_matrix*vec4(screenPos,-1.0,1.0);vec4 rayEnd =u_inv_matrix*vec4(screenPos, 1.0,1.0);rayStart.xyz/=rayStart.w;rayEnd.xyz /=rayEnd.w;highp float t=(0.0-rayStart.z)/(rayEnd.z-rayStart.z);return mix(rayStart.xyz,rayEnd.xyz,t);}void main() {vec2 quadCenterPos=a_pos;float radius=a_radius;float collision=a_flags.x;float vertexIdx=a_flags.y;vec2 quadVertexOffset=vec2(mix(-1.0,1.0,float(vertexIdx >=2.0)),mix(-1.0,1.0,float(vertexIdx >=1.0 && vertexIdx <=2.0)));vec2 quadVertexExtent=quadVertexOffset*radius;vec3 tilePos=toTilePosition(quadCenterPos);vec4 clipPos=u_matrix*vec4(tilePos,1.0);highp float camera_to_anchor_distance=clipPos.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);float padding_factor=1.2;v_radius=radius;v_extrude=quadVertexExtent*padding_factor;v_perspective_ratio=collision_perspective_ratio;v_collision=collision;gl_Position=vec4(clipPos.xyz/clipPos.w,1.0)+vec4(quadVertexExtent*padding_factor/u_viewport_size*2.0,0.0,0.0);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var debugFrag = 'uniform highp vec4 u_color;uniform sampler2D u_overlay;varying vec2 v_uv;void main() {vec4 overlay_color=texture2D(u_overlay,v_uv);gl_FragColor=mix(u_color,overlay_color,overlay_color.a);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var debugVert = 'attribute vec2 a_pos;varying vec2 v_uv;uniform mat4 u_matrix;uniform float u_overlay_scale;void main() {v_uv=a_pos/8192.0;gl_Position=u_matrix*vec4(a_pos*u_overlay_scale,get_elevation(a_pos),1);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var fillFrag = '#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float opacity\ngl_FragColor=color*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var fillVert = 'attribute vec2 a_pos;uniform mat4 u_matrix;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float opacity\ngl_Position=u_matrix*vec4(a_pos,0,1);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var fillOutlineFrag = 'varying vec2 v_pos;\n#pragma mapbox: define highp vec4 outline_color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 outline_color\n#pragma mapbox: initialize lowp float opacity\nfloat dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=outline_color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var fillOutlineVert = 'attribute vec2 a_pos;uniform mat4 u_matrix;uniform vec2 u_world;varying vec2 v_pos;\n#pragma mapbox: define highp vec4 outline_color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 outline_color\n#pragma mapbox: initialize lowp float opacity\ngl_Position=u_matrix*vec4(a_pos,0,1);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var fillOutlinePatternFrag = 'uniform vec2 u_texsize;uniform sampler2D u_image;uniform float u_fade;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);float dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=mix(color1,color2,u_fade)*alpha*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var fillOutlinePatternVert = 'uniform mat4 u_matrix;uniform vec2 u_world;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;gl_Position=u_matrix*vec4(a_pos,0,1);vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,a_pos);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var fillPatternFrag = '#ifdef GL_ES\nprecision highp float;\n#endif\nuniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_fade)*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var fillPatternVert = 'uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileZoomRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileZoomRatio,a_pos);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var fillExtrusionFrag = 'varying vec4 v_color;void main() {gl_FragColor=v_color;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var fillExtrusionVert = 'uniform mat4 u_matrix;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;uniform float u_vertical_gradient;uniform lowp float u_opacity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\n#ifdef TERRAIN3D\nattribute vec2 a_centroid;\n#endif\nvarying vec4 v_color;\n#pragma mapbox: define highp float base\n#pragma mapbox: define highp float height\n#pragma mapbox: define highp vec4 color\nvoid main() {\n#pragma mapbox: initialize highp float base\n#pragma mapbox: initialize highp float height\n#pragma mapbox: initialize highp vec4 color\nvec3 normal=a_normal_ed.xyz;\n#ifdef TERRAIN3D\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\n#else\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\n#endif\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);gl_Position=u_matrix*vec4(a_pos,t > 0.0 ? height : base,1);float colorvalue=color.r*0.2126+color.g*0.7152+color.b*0.0722;v_color=vec4(0.0,0.0,0.0,1.0);vec4 ambientlight=vec4(0.03,0.03,0.03,1.0);color+=ambientlight;float directional=clamp(dot(normal/16384.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((1.0-colorvalue+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_color.r+=clamp(color.r*directional*u_lightcolor.r,mix(0.0,0.3,1.0-u_lightcolor.r),1.0);v_color.g+=clamp(color.g*directional*u_lightcolor.g,mix(0.0,0.3,1.0-u_lightcolor.g),1.0);v_color.b+=clamp(color.b*directional*u_lightcolor.b,mix(0.0,0.3,1.0-u_lightcolor.b),1.0);v_color*=u_opacity;}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var fillExtrusionPatternFrag = 'uniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\n#pragma mapbox: define lowp float base\n#pragma mapbox: define lowp float height\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float base\n#pragma mapbox: initialize lowp float height\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);vec4 mixedColor=mix(color1,color2,u_fade);gl_FragColor=mixedColor*v_lighting;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var fillExtrusionPatternVert = 'uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_height_factor;uniform vec3 u_scale;uniform float u_vertical_gradient;uniform lowp float u_opacity;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\n#ifdef TERRAIN3D\nattribute vec2 a_centroid;\n#endif\nvarying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\n#pragma mapbox: define lowp float base\n#pragma mapbox: define lowp float height\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float base\n#pragma mapbox: initialize lowp float height\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec3 normal=a_normal_ed.xyz;float edgedistance=a_normal_ed.w;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;\n#ifdef TERRAIN3D\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\n#else\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\n#endif\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);float z=t > 0.0 ? height : base;gl_Position=u_matrix*vec4(a_pos,z,1);vec2 pos=normal.x==1.0 && normal.y==0.0 && normal.z==16384.0\n? a_pos\n: vec2(edgedistance,z*u_height_factor);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,pos);v_lighting=vec4(0.0,0.0,0.0,1.0);float directional=clamp(dot(normal/16383.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((0.5+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_lighting.rgb+=clamp(directional*u_lightcolor,mix(vec3(0.0),vec3(0.3),1.0-u_lightcolor),vec3(1.0));v_lighting*=u_opacity;}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var hillshadePrepareFrag = '#ifdef GL_ES\nprecision highp float;\n#endif\nuniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_dimension;uniform float u_zoom;uniform vec4 u_unpack;float getElevation(vec2 coord,float bias) {vec4 data=texture2D(u_image,coord)*255.0;data.a=-1.0;return dot(data,u_unpack)/4.0;}void main() {vec2 epsilon=1.0/u_dimension;float a=getElevation(v_pos+vec2(-epsilon.x,-epsilon.y),0.0);float b=getElevation(v_pos+vec2(0,-epsilon.y),0.0);float c=getElevation(v_pos+vec2(epsilon.x,-epsilon.y),0.0);float d=getElevation(v_pos+vec2(-epsilon.x,0),0.0);float e=getElevation(v_pos,0.0);float f=getElevation(v_pos+vec2(epsilon.x,0),0.0);float g=getElevation(v_pos+vec2(-epsilon.x,epsilon.y),0.0);float h=getElevation(v_pos+vec2(0,epsilon.y),0.0);float i=getElevation(v_pos+vec2(epsilon.x,epsilon.y),0.0);float exaggerationFactor=u_zoom < 2.0 ? 0.4 : u_zoom < 4.5 ? 0.35 : 0.3;float exaggeration=u_zoom < 15.0 ? (u_zoom-15.0)*exaggerationFactor : 0.0;vec2 deriv=vec2((c+f+f+i)-(a+d+d+g),(g+h+h+i)-(a+b+b+c))/pow(2.0,exaggeration+(19.2562-u_zoom));gl_FragColor=clamp(vec4(deriv.x/2.0+0.5,deriv.y/2.0+0.5,1.0,1.0),0.0,1.0);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var hillshadePrepareVert = 'uniform mat4 u_matrix;uniform vec2 u_dimension;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);highp vec2 epsilon=1.0/u_dimension;float scale=(u_dimension.x-2.0)/u_dimension.x;v_pos=(a_texture_pos/8192.0)*scale+epsilon;}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var hillshadeFrag = 'uniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_latrange;uniform vec2 u_light;uniform vec4 u_shadow;uniform vec4 u_highlight;uniform vec4 u_accent;\n#define PI 3.141592653589793\nvoid main() {vec4 pixel=texture2D(u_image,v_pos);vec2 deriv=((pixel.rg*2.0)-1.0);float scaleFactor=cos(radians((u_latrange[0]-u_latrange[1])*(1.0-v_pos.y)+u_latrange[1]));float slope=atan(1.25*length(deriv)/scaleFactor);float aspect=deriv.x !=0.0 ? atan(deriv.y,-deriv.x) : PI/2.0*(deriv.y > 0.0 ? 1.0 :-1.0);float intensity=u_light.x;float azimuth=u_light.y+PI;float base=1.875-intensity*1.75;float maxValue=0.5*PI;float scaledSlope=intensity !=0.5 ? ((pow(base,slope)-1.0)/(pow(base,maxValue)-1.0))*maxValue : slope;float accent=cos(scaledSlope);vec4 accent_color=(1.0-accent)*u_accent*clamp(intensity*2.0,0.0,1.0);float shade=abs(mod((aspect+azimuth)/PI+0.5,2.0)-1.0);vec4 shade_color=mix(u_shadow,u_highlight,shade)*sin(scaledSlope)*clamp(intensity*2.0,0.0,1.0);gl_FragColor=accent_color*(1.0-shade_color.a)+shade_color;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var hillshadeVert = 'uniform mat4 u_matrix;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos=a_texture_pos/8192.0;}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var lineFrag = 'uniform lowp float u_device_pixel_ratio;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);gl_FragColor=color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var lineVert = '\n#define scale 0.015873016\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform vec2 u_units_to_pixels;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp float v_linesofar;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float width\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float width\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;v_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*2.0;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_width2=vec2(outset,inset);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var lineGradientFrag = 'uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;varying highp vec2 v_uv;\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);vec4 color=texture2D(u_image,v_uv);gl_FragColor=color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var lineGradientVert = '\n#define scale 0.015873016\nattribute vec2 a_pos_normal;attribute vec4 a_data;attribute float a_uv_x;attribute float a_split_index;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_units_to_pixels;uniform float u_image_height;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp vec2 v_uv;\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float width\nvoid main() {\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float width\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;highp float texel_height=1.0/u_image_height;highp float half_texel_height=0.5*texel_height;v_uv=vec2(a_uv_x,a_split_index*texel_height-half_texel_height);vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_width2=vec2(outset,inset);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var linePatternFrag = '#ifdef GL_ES\nprecision highp float;\n#endif\nuniform lowp float u_device_pixel_ratio;uniform vec2 u_texsize;uniform float u_fade;uniform mediump vec3 u_scale;uniform sampler2D u_image;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;vec2 pattern_size_a=vec2(display_size_a.x*fromScale/tileZoomRatio,display_size_a.y);vec2 pattern_size_b=vec2(display_size_b.x*toScale/tileZoomRatio,display_size_b.y);float aspect_a=display_size_a.y/v_width;float aspect_b=display_size_b.y/v_width;float dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float x_a=mod(v_linesofar/pattern_size_a.x*aspect_a,1.0);float x_b=mod(v_linesofar/pattern_size_b.x*aspect_b,1.0);float y=0.5*v_normal.y+0.5;vec2 texel_size=1.0/u_texsize;vec2 pos_a=mix(pattern_tl_a*texel_size-texel_size,pattern_br_a*texel_size+texel_size,vec2(x_a,y));vec2 pos_b=mix(pattern_tl_b*texel_size-texel_size,pattern_br_b*texel_size+texel_size,vec2(x_b,y));vec4 color=mix(texture2D(u_image,pos_a),texture2D(u_image,pos_b),u_fade);gl_FragColor=color*alpha*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var linePatternVert = '\n#define scale 0.015873016\n#define LINE_DISTANCE_SCALE 2.0\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform vec2 u_units_to_pixels;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define mediump float width\n#pragma mapbox: define lowp float floorwidth\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize mediump float width\n#pragma mapbox: initialize lowp float floorwidth\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_linesofar=a_linesofar;v_width2=vec2(outset,inset);v_width=floorwidth;}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var lineSDFFrag = 'uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;uniform float u_sdfgamma;uniform float u_mix;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float width\n#pragma mapbox: define lowp float floorwidth\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float width\n#pragma mapbox: initialize lowp float floorwidth\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float sdfdist_a=texture2D(u_image,v_tex_a).a;float sdfdist_b=texture2D(u_image,v_tex_b).a;float sdfdist=mix(sdfdist_a,sdfdist_b,u_mix);alpha*=smoothstep(0.5-u_sdfgamma/floorwidth,0.5+u_sdfgamma/floorwidth,sdfdist);gl_FragColor=color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var lineSDFVert = '\n#define scale 0.015873016\n#define LINE_DISTANCE_SCALE 2.0\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_patternscale_a;uniform float u_tex_y_a;uniform vec2 u_patternscale_b;uniform float u_tex_y_b;uniform vec2 u_units_to_pixels;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float width\n#pragma mapbox: define lowp float floorwidth\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float width\n#pragma mapbox: initialize lowp float floorwidth\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_tex_a=vec2(a_linesofar*u_patternscale_a.x/floorwidth,normal.y*u_patternscale_a.y+u_tex_y_a);v_tex_b=vec2(a_linesofar*u_patternscale_b.x/floorwidth,normal.y*u_patternscale_b.y+u_tex_y_b);v_width2=vec2(outset,inset);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var rasterFrag = 'uniform float u_fade_t;uniform float u_opacity;uniform sampler2D u_image0;uniform sampler2D u_image1;varying vec2 v_pos0;varying vec2 v_pos1;uniform float u_brightness_low;uniform float u_brightness_high;uniform float u_saturation_factor;uniform float u_contrast_factor;uniform vec3 u_spin_weights;void main() {vec4 color0=texture2D(u_image0,v_pos0);vec4 color1=texture2D(u_image1,v_pos1);if (color0.a > 0.0) {color0.rgb=color0.rgb/color0.a;}if (color1.a > 0.0) {color1.rgb=color1.rgb/color1.a;}vec4 color=mix(color0,color1,u_fade_t);color.a*=u_opacity;vec3 rgb=color.rgb;rgb=vec3(dot(rgb,u_spin_weights.xyz),dot(rgb,u_spin_weights.zxy),dot(rgb,u_spin_weights.yzx));float average=(color.r+color.g+color.b)/3.0;rgb+=(average-rgb)*u_saturation_factor;rgb=(rgb-0.5)*u_contrast_factor+0.5;vec3 u_high_vec=vec3(u_brightness_low,u_brightness_low,u_brightness_low);vec3 u_low_vec=vec3(u_brightness_high,u_brightness_high,u_brightness_high);gl_FragColor=vec4(mix(u_high_vec,u_low_vec,rgb)*color.a,color.a);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var rasterVert = 'uniform mat4 u_matrix;uniform vec2 u_tl_parent;uniform float u_scale_parent;uniform float u_buffer_scale;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos0;varying vec2 v_pos1;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos0=(((a_texture_pos/8192.0)-0.5)/u_buffer_scale )+0.5;v_pos1=(v_pos0*u_scale_parent)+u_tl_parent;}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var symbolIconFrag = 'uniform sampler2D u_texture;varying vec2 v_tex;varying float v_fade_opacity;\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\nlowp float alpha=opacity*v_fade_opacity;gl_FragColor=texture2D(u_texture,v_tex)*alpha;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var symbolIconVert = 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform highp float u_camera_to_center_distance;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform float u_fade_change;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform vec2 u_texsize;varying vec2 v_tex;varying float v_fade_opacity;\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;vec2 a_minFontScale=a_pixeloffset.zw/256.0;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\ncamera_to_anchor_distance/u_camera_to_center_distance :\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*max(a_minFontScale,fontScale)+a_pxoffset/16.0),z,1.0);v_tex=a_tex/u_texsize;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float visibility=calculate_visibility(projectedPoint);v_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var symbolSDFFrag = '#define SDF_PX 8.0\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;uniform bool u_is_text;varying vec2 v_data0;varying vec3 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nfloat EDGE_GAMMA=0.105/u_device_pixel_ratio;vec2 tex=v_data0.xy;float gamma_scale=v_data1.x;float size=v_data1.y;float fade_opacity=v_data1[2];float fontScale=u_is_text ? size/24.0 : size;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float inner_edge=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);inner_edge=inner_edge+gamma*gamma_scale;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(inner_edge-gamma_scaled,inner_edge+gamma_scaled,dist);if (u_is_halo) {lowp float halo_edge=(6.0-halo_width/fontScale)/SDF_PX;alpha=min(smoothstep(halo_edge-gamma_scaled,halo_edge+gamma_scaled,dist),1.0-alpha);}gl_FragColor=color*(alpha*opacity*fade_opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var symbolSDFVert = 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;varying vec2 v_data0;varying vec3 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\ncamera_to_anchor_distance/u_camera_to_center_distance :\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale+a_pxoffset),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0=a_tex/u_texsize;v_data1=vec3(gamma_scale,size,interpolated_fade_opacity);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var symbolTextAndIconFrag = '#define SDF_PX 8.0\n#define SDF 1.0\n#define ICON 0.0\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform sampler2D u_texture_icon;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;varying vec4 v_data0;varying vec4 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nfloat fade_opacity=v_data1[2];if (v_data1.w==ICON) {vec2 tex_icon=v_data0.zw;lowp float alpha=opacity*fade_opacity;gl_FragColor=texture2D(u_texture_icon,tex_icon)*alpha;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\nreturn;}vec2 tex=v_data0.xy;float EDGE_GAMMA=0.105/u_device_pixel_ratio;float gamma_scale=v_data1.x;float size=v_data1.y;float fontScale=size/24.0;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float buff=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);buff=(6.0-halo_width/fontScale)/SDF_PX;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(buff-gamma_scaled,buff+gamma_scaled,dist);gl_FragColor=color*(alpha*opacity*fade_opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var symbolTextAndIconVert = 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;uniform vec2 u_texsize_icon;varying vec4 v_data0;varying vec4 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);float is_sdf=a_size[0]-2.0*a_size_min;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\ncamera_to_anchor_distance/u_camera_to_center_distance :\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=size/24.0;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0.xy=a_tex/u_texsize;v_data0.zw=a_tex/u_texsize_icon;v_data1=vec4(gamma_scale,size,interpolated_fade_opacity,is_sdf);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var terrainDepthFrag = 'varying float v_depth;const highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitMsk=vec4(0.,vec3(1./256.0));highp vec4 pack(highp float value) {highp vec4 comp=fract(value*bitSh);comp-=comp.xxyz*bitMsk;return comp;}void main() {gl_FragColor=pack(v_depth);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var terrainCoordsFrag = 'precision mediump float;uniform sampler2D u_texture;uniform float u_terrain_coords_id;varying vec2 v_texture_pos;void main() {vec4 rgba=texture2D(u_texture,v_texture_pos);gl_FragColor=vec4(rgba.r,rgba.g,rgba.b,u_terrain_coords_id);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var terrainFrag = 'uniform sampler2D u_texture;varying vec2 v_texture_pos;void main() {gl_FragColor=texture2D(u_texture,v_texture_pos);}'; + +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +var terrainVert = 'attribute vec3 a_pos3d;uniform mat4 u_matrix;uniform float u_ele_delta;varying vec2 v_texture_pos;varying float v_depth;void main() {float extent=8192.0;float ele_delta=a_pos3d.z==1.0 ? u_ele_delta : 0.0;v_texture_pos=a_pos3d.xy/extent;gl_Position=u_matrix*vec4(a_pos3d.xy,get_elevation(a_pos3d.xy)-ele_delta,1.0);v_depth=gl_Position.z/gl_Position.w;}'; + +// Disable Flow annotations here because Flow doesn't support importing GLSL files +const shaders = { + prelude: compile(preludeFrag, preludeVert), + background: compile(backgroundFrag, backgroundVert), + backgroundPattern: compile(backgroundPatternFrag, backgroundPatternVert), + circle: compile(circleFrag, circleVert), + clippingMask: compile(clippingMaskFrag, clippingMaskVert), + heatmap: compile(heatmapFrag, heatmapVert), + heatmapTexture: compile(heatmapTextureFrag, heatmapTextureVert), + collisionBox: compile(collisionBoxFrag, collisionBoxVert), + collisionCircle: compile(collisionCircleFrag, collisionCircleVert), + debug: compile(debugFrag, debugVert), + fill: compile(fillFrag, fillVert), + fillOutline: compile(fillOutlineFrag, fillOutlineVert), + fillOutlinePattern: compile(fillOutlinePatternFrag, fillOutlinePatternVert), + fillPattern: compile(fillPatternFrag, fillPatternVert), + fillExtrusion: compile(fillExtrusionFrag, fillExtrusionVert), + fillExtrusionPattern: compile(fillExtrusionPatternFrag, fillExtrusionPatternVert), + hillshadePrepare: compile(hillshadePrepareFrag, hillshadePrepareVert), + hillshade: compile(hillshadeFrag, hillshadeVert), + line: compile(lineFrag, lineVert), + lineGradient: compile(lineGradientFrag, lineGradientVert), + linePattern: compile(linePatternFrag, linePatternVert), + lineSDF: compile(lineSDFFrag, lineSDFVert), + raster: compile(rasterFrag, rasterVert), + symbolIcon: compile(symbolIconFrag, symbolIconVert), + symbolSDF: compile(symbolSDFFrag, symbolSDFVert), + symbolTextAndIcon: compile(symbolTextAndIconFrag, symbolTextAndIconVert), + terrain: compile(terrainFrag, terrainVert), + terrainDepth: compile(terrainDepthFrag, terrainVert), + terrainCoords: compile(terrainCoordsFrag, terrainVert) +}; +// Expand #pragmas to #ifdefs. +function compile(fragmentSource, vertexSource) { + const re = /#pragma mapbox: ([\w]+) ([\w]+) ([\w]+) ([\w]+)/g; + const staticAttributes = vertexSource.match(/attribute ([\w]+) ([\w]+)/g); + const fragmentUniforms = fragmentSource.match(/uniform ([\w]+) ([\w]+)([\s]*)([\w]*)/g); + const vertexUniforms = vertexSource.match(/uniform ([\w]+) ([\w]+)([\s]*)([\w]*)/g); + const staticUniforms = vertexUniforms ? vertexUniforms.concat(fragmentUniforms) : fragmentUniforms; + const fragmentPragmas = {}; + fragmentSource = fragmentSource.replace(re, (match, operation, precision, type, name) => { + fragmentPragmas[name] = true; + if (operation === 'define') { + return ` +#ifndef HAS_UNIFORM_u_${name} +varying ${precision} ${type} ${name}; +#else +uniform ${precision} ${type} u_${name}; +#endif +`; + } + else /* if (operation === 'initialize') */ { + return ` +#ifdef HAS_UNIFORM_u_${name} + ${precision} ${type} ${name} = u_${name}; +#endif +`; + } + }); + vertexSource = vertexSource.replace(re, (match, operation, precision, type, name) => { + const attrType = type === 'float' ? 'vec2' : 'vec4'; + const unpackType = name.match(/color/) ? 'color' : attrType; + if (fragmentPragmas[name]) { + if (operation === 'define') { + return ` +#ifndef HAS_UNIFORM_u_${name} +uniform lowp float u_${name}_t; +attribute ${precision} ${attrType} a_${name}; +varying ${precision} ${type} ${name}; +#else +uniform ${precision} ${type} u_${name}; +#endif +`; + } + else /* if (operation === 'initialize') */ { + if (unpackType === 'vec4') { + // vec4 attributes are only used for cross-faded properties, and are not packed + return ` +#ifndef HAS_UNIFORM_u_${name} + ${name} = a_${name}; +#else + ${precision} ${type} ${name} = u_${name}; +#endif +`; + } + else { + return ` +#ifndef HAS_UNIFORM_u_${name} + ${name} = unpack_mix_${unpackType}(a_${name}, u_${name}_t); +#else + ${precision} ${type} ${name} = u_${name}; +#endif +`; + } + } + } + else { + if (operation === 'define') { + return ` +#ifndef HAS_UNIFORM_u_${name} +uniform lowp float u_${name}_t; +attribute ${precision} ${attrType} a_${name}; +#else +uniform ${precision} ${type} u_${name}; +#endif +`; + } + else /* if (operation === 'initialize') */ { + if (unpackType === 'vec4') { + // vec4 attributes are only used for cross-faded properties, and are not packed + return ` +#ifndef HAS_UNIFORM_u_${name} + ${precision} ${type} ${name} = a_${name}; +#else + ${precision} ${type} ${name} = u_${name}; +#endif +`; + } + else /* */ { + return ` +#ifndef HAS_UNIFORM_u_${name} + ${precision} ${type} ${name} = unpack_mix_${unpackType}(a_${name}, u_${name}_t); +#else + ${precision} ${type} ${name} = u_${name}; +#endif +`; + } + } + } + }); + return { fragmentSource, vertexSource, staticAttributes, staticUniforms }; +} + +/** + * @internal + * A vertex array object used to pass data to the webgl code + */ +class VertexArrayObject { + constructor() { + this.boundProgram = null; + this.boundLayoutVertexBuffer = null; + this.boundPaintVertexBuffers = []; + this.boundIndexBuffer = null; + this.boundVertexOffset = null; + this.boundDynamicVertexBuffer = null; + this.vao = null; + } + bind(context, program, layoutVertexBuffer, paintVertexBuffers, indexBuffer, vertexOffset, dynamicVertexBuffer, dynamicVertexBuffer2, dynamicVertexBuffer3) { + this.context = context; + let paintBuffersDiffer = this.boundPaintVertexBuffers.length !== paintVertexBuffers.length; + for (let i = 0; !paintBuffersDiffer && i < paintVertexBuffers.length; i++) { + if (this.boundPaintVertexBuffers[i] !== paintVertexBuffers[i]) { + paintBuffersDiffer = true; + } + } + const isFreshBindRequired = (!this.vao || + this.boundProgram !== program || + this.boundLayoutVertexBuffer !== layoutVertexBuffer || + paintBuffersDiffer || + this.boundIndexBuffer !== indexBuffer || + this.boundVertexOffset !== vertexOffset || + this.boundDynamicVertexBuffer !== dynamicVertexBuffer || + this.boundDynamicVertexBuffer2 !== dynamicVertexBuffer2 || + this.boundDynamicVertexBuffer3 !== dynamicVertexBuffer3); + if (isFreshBindRequired) { + this.freshBind(program, layoutVertexBuffer, paintVertexBuffers, indexBuffer, vertexOffset, dynamicVertexBuffer, dynamicVertexBuffer2, dynamicVertexBuffer3); + } + else { + context.bindVertexArray.set(this.vao); + if (dynamicVertexBuffer) { + // The buffer may have been updated. Rebind to upload data. + dynamicVertexBuffer.bind(); + } + if (indexBuffer && indexBuffer.dynamicDraw) { + indexBuffer.bind(); + } + if (dynamicVertexBuffer2) { + dynamicVertexBuffer2.bind(); + } + if (dynamicVertexBuffer3) { + dynamicVertexBuffer3.bind(); + } + } + } + freshBind(program, layoutVertexBuffer, paintVertexBuffers, indexBuffer, vertexOffset, dynamicVertexBuffer, dynamicVertexBuffer2, dynamicVertexBuffer3) { + const numNextAttributes = program.numAttributes; + const context = this.context; + const gl = context.gl; + if (this.vao) + this.destroy(); + this.vao = context.createVertexArray(); + context.bindVertexArray.set(this.vao); + // store the arguments so that we can verify them when the vao is bound again + this.boundProgram = program; + this.boundLayoutVertexBuffer = layoutVertexBuffer; + this.boundPaintVertexBuffers = paintVertexBuffers; + this.boundIndexBuffer = indexBuffer; + this.boundVertexOffset = vertexOffset; + this.boundDynamicVertexBuffer = dynamicVertexBuffer; + this.boundDynamicVertexBuffer2 = dynamicVertexBuffer2; + this.boundDynamicVertexBuffer3 = dynamicVertexBuffer3; + layoutVertexBuffer.enableAttributes(gl, program); + for (const vertexBuffer of paintVertexBuffers) { + vertexBuffer.enableAttributes(gl, program); + } + if (dynamicVertexBuffer) { + dynamicVertexBuffer.enableAttributes(gl, program); + } + if (dynamicVertexBuffer2) { + dynamicVertexBuffer2.enableAttributes(gl, program); + } + if (dynamicVertexBuffer3) { + dynamicVertexBuffer3.enableAttributes(gl, program); + } + layoutVertexBuffer.bind(); + layoutVertexBuffer.setVertexAttribPointers(gl, program, vertexOffset); + for (const vertexBuffer of paintVertexBuffers) { + vertexBuffer.bind(); + vertexBuffer.setVertexAttribPointers(gl, program, vertexOffset); + } + if (dynamicVertexBuffer) { + dynamicVertexBuffer.bind(); + dynamicVertexBuffer.setVertexAttribPointers(gl, program, vertexOffset); + } + if (indexBuffer) { + indexBuffer.bind(); + } + if (dynamicVertexBuffer2) { + dynamicVertexBuffer2.bind(); + dynamicVertexBuffer2.setVertexAttribPointers(gl, program, vertexOffset); + } + if (dynamicVertexBuffer3) { + dynamicVertexBuffer3.bind(); + dynamicVertexBuffer3.setVertexAttribPointers(gl, program, vertexOffset); + } + context.currentNumAttributes = numNextAttributes; + } + destroy() { + if (this.vao) { + this.context.deleteVertexArray(this.vao); + this.vao = null; + } + } +} + +const terrainPreludeUniforms = (context, locations) => ({ + 'u_depth': new performance.Uniform1i(context, locations.u_depth), + 'u_terrain': new performance.Uniform1i(context, locations.u_terrain), + 'u_terrain_dim': new performance.Uniform1f(context, locations.u_terrain_dim), + 'u_terrain_matrix': new performance.UniformMatrix4f(context, locations.u_terrain_matrix), + 'u_terrain_unpack': new performance.Uniform4f(context, locations.u_terrain_unpack), + 'u_terrain_exaggeration': new performance.Uniform1f(context, locations.u_terrain_exaggeration) +}); +const terrainUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_texture': new performance.Uniform1i(context, locations.u_texture), + 'u_ele_delta': new performance.Uniform1f(context, locations.u_ele_delta) +}); +const terrainDepthUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_ele_delta': new performance.Uniform1f(context, locations.u_ele_delta) +}); +const terrainCoordsUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_texture': new performance.Uniform1i(context, locations.u_texture), + 'u_terrain_coords_id': new performance.Uniform1f(context, locations.u_terrain_coords_id), + 'u_ele_delta': new performance.Uniform1f(context, locations.u_ele_delta) +}); +const terrainUniformValues = (matrix, eleDelta) => ({ + 'u_matrix': matrix, + 'u_texture': 0, + 'u_ele_delta': eleDelta +}); +const terrainDepthUniformValues = (matrix, eleDelta) => ({ + 'u_matrix': matrix, + 'u_ele_delta': eleDelta +}); +const terrainCoordsUniformValues = (matrix, coordsId, eleDelta) => ({ + 'u_matrix': matrix, + 'u_terrain_coords_id': coordsId / 255, + 'u_texture': 0, + 'u_ele_delta': eleDelta +}); + +function getTokenizedAttributesAndUniforms(array) { + const result = []; + for (let i = 0; i < array.length; i++) { + if (array[i] === null) + continue; + const token = array[i].split(' '); + result.push(token.pop()); + } + return result; +} +/** + * @internal + * A webgl program to execute in the GPU space + */ +class Program { + constructor(context, source, configuration, fixedUniforms, showOverdrawInspector, terrain) { + const gl = context.gl; + this.program = gl.createProgram(); + const staticAttrInfo = getTokenizedAttributesAndUniforms(source.staticAttributes); + const dynamicAttrInfo = configuration ? configuration.getBinderAttributes() : []; + const allAttrInfo = staticAttrInfo.concat(dynamicAttrInfo); + const preludeUniformsInfo = shaders.prelude.staticUniforms ? getTokenizedAttributesAndUniforms(shaders.prelude.staticUniforms) : []; + const staticUniformsInfo = source.staticUniforms ? getTokenizedAttributesAndUniforms(source.staticUniforms) : []; + const dynamicUniformsInfo = configuration ? configuration.getBinderUniforms() : []; + // remove duplicate uniforms + const uniformList = preludeUniformsInfo.concat(staticUniformsInfo).concat(dynamicUniformsInfo); + const allUniformsInfo = []; + for (const uniform of uniformList) { + if (allUniformsInfo.indexOf(uniform) < 0) + allUniformsInfo.push(uniform); + } + const defines = configuration ? configuration.defines() : []; + if (showOverdrawInspector) { + defines.push('#define OVERDRAW_INSPECTOR;'); + } + if (terrain) { + defines.push('#define TERRAIN3D;'); + } + const fragmentSource = defines.concat(shaders.prelude.fragmentSource, source.fragmentSource).join('\n'); + const vertexSource = defines.concat(shaders.prelude.vertexSource, source.vertexSource).join('\n'); + const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER); + if (gl.isContextLost()) { + this.failedToCreate = true; + return; + } + gl.shaderSource(fragmentShader, fragmentSource); + gl.compileShader(fragmentShader); + if (!gl.getShaderParameter(fragmentShader, gl.COMPILE_STATUS)) { + throw new Error(`Could not compile fragment shader: ${gl.getShaderInfoLog(fragmentShader)}`); + } + gl.attachShader(this.program, fragmentShader); + const vertexShader = gl.createShader(gl.VERTEX_SHADER); + if (gl.isContextLost()) { + this.failedToCreate = true; + return; + } + gl.shaderSource(vertexShader, vertexSource); + gl.compileShader(vertexShader); + if (!gl.getShaderParameter(vertexShader, gl.COMPILE_STATUS)) { + throw new Error(`Could not compile vertex shader: ${gl.getShaderInfoLog(vertexShader)}`); + } + gl.attachShader(this.program, vertexShader); + this.attributes = {}; + const uniformLocations = {}; + this.numAttributes = allAttrInfo.length; + for (let i = 0; i < this.numAttributes; i++) { + if (allAttrInfo[i]) { + gl.bindAttribLocation(this.program, i, allAttrInfo[i]); + this.attributes[allAttrInfo[i]] = i; + } + } + gl.linkProgram(this.program); + if (!gl.getProgramParameter(this.program, gl.LINK_STATUS)) { + throw new Error(`Program failed to link: ${gl.getProgramInfoLog(this.program)}`); + } + gl.deleteShader(vertexShader); + gl.deleteShader(fragmentShader); + for (let it = 0; it < allUniformsInfo.length; it++) { + const uniform = allUniformsInfo[it]; + if (uniform && !uniformLocations[uniform]) { + const uniformLocation = gl.getUniformLocation(this.program, uniform); + if (uniformLocation) { + uniformLocations[uniform] = uniformLocation; + } + } + } + this.fixedUniforms = fixedUniforms(context, uniformLocations); + this.terrainUniforms = terrainPreludeUniforms(context, uniformLocations); + this.binderUniforms = configuration ? configuration.getUniforms(context, uniformLocations) : []; + } + draw(context, drawMode, depthMode, stencilMode, colorMode, cullFaceMode, uniformValues, terrain, layerID, layoutVertexBuffer, indexBuffer, segments, currentProperties, zoom, configuration, dynamicLayoutBuffer, dynamicLayoutBuffer2, dynamicLayoutBuffer3) { + const gl = context.gl; + if (this.failedToCreate) + return; + context.program.set(this.program); + context.setDepthMode(depthMode); + context.setStencilMode(stencilMode); + context.setColorMode(colorMode); + context.setCullFace(cullFaceMode); + // set variables used by the 3d functions defined in _prelude.vertex.glsl + if (terrain) { + context.activeTexture.set(gl.TEXTURE2); + gl.bindTexture(gl.TEXTURE_2D, terrain.depthTexture); + context.activeTexture.set(gl.TEXTURE3); + gl.bindTexture(gl.TEXTURE_2D, terrain.texture); + for (const name in this.terrainUniforms) { + this.terrainUniforms[name].set(terrain[name]); + } + } + for (const name in this.fixedUniforms) { + this.fixedUniforms[name].set(uniformValues[name]); + } + if (configuration) { + configuration.setUniforms(context, this.binderUniforms, currentProperties, { zoom: zoom }); + } + let primitiveSize = 0; + switch (drawMode) { + case gl.LINES: + primitiveSize = 2; + break; + case gl.TRIANGLES: + primitiveSize = 3; + break; + case gl.LINE_STRIP: + primitiveSize = 1; + break; + } + for (const segment of segments.get()) { + const vaos = segment.vaos || (segment.vaos = {}); + const vao = vaos[layerID] || (vaos[layerID] = new VertexArrayObject()); + vao.bind(context, this, layoutVertexBuffer, configuration ? configuration.getPaintVertexBuffers() : [], indexBuffer, segment.vertexOffset, dynamicLayoutBuffer, dynamicLayoutBuffer2, dynamicLayoutBuffer3); + gl.drawElements(drawMode, segment.primitiveLength * primitiveSize, gl.UNSIGNED_SHORT, segment.primitiveOffset * primitiveSize * 2); + } + } +} + +function patternUniformValues(crossfade, painter, tile) { + const tileRatio = 1 / pixelsToTileUnits(tile, 1, painter.transform.tileZoom); + const numTiles = Math.pow(2, tile.tileID.overscaledZ); + const tileSizeAtNearestZoom = tile.tileSize * Math.pow(2, painter.transform.tileZoom) / numTiles; + const pixelX = tileSizeAtNearestZoom * (tile.tileID.canonical.x + tile.tileID.wrap * numTiles); + const pixelY = tileSizeAtNearestZoom * tile.tileID.canonical.y; + return { + 'u_image': 0, + 'u_texsize': tile.imageAtlasTexture.size, + 'u_scale': [tileRatio, crossfade.fromScale, crossfade.toScale], + 'u_fade': crossfade.t, + // split the pixel coord into two pairs of 16 bit numbers. The glsl spec only guarantees 16 bits of precision. + 'u_pixel_coord_upper': [pixelX >> 16, pixelY >> 16], + 'u_pixel_coord_lower': [pixelX & 0xFFFF, pixelY & 0xFFFF] + }; +} +function bgPatternUniformValues(image, crossfade, painter, tile) { + const imagePosA = painter.imageManager.getPattern(image.from.toString()); + const imagePosB = painter.imageManager.getPattern(image.to.toString()); + const { width, height } = painter.imageManager.getPixelSize(); + const numTiles = Math.pow(2, tile.tileID.overscaledZ); + const tileSizeAtNearestZoom = tile.tileSize * Math.pow(2, painter.transform.tileZoom) / numTiles; + const pixelX = tileSizeAtNearestZoom * (tile.tileID.canonical.x + tile.tileID.wrap * numTiles); + const pixelY = tileSizeAtNearestZoom * tile.tileID.canonical.y; + return { + 'u_image': 0, + 'u_pattern_tl_a': imagePosA.tl, + 'u_pattern_br_a': imagePosA.br, + 'u_pattern_tl_b': imagePosB.tl, + 'u_pattern_br_b': imagePosB.br, + 'u_texsize': [width, height], + 'u_mix': crossfade.t, + 'u_pattern_size_a': imagePosA.displaySize, + 'u_pattern_size_b': imagePosB.displaySize, + 'u_scale_a': crossfade.fromScale, + 'u_scale_b': crossfade.toScale, + 'u_tile_units_to_pixels': 1 / pixelsToTileUnits(tile, 1, painter.transform.tileZoom), + // split the pixel coord into two pairs of 16 bit numbers. The glsl spec only guarantees 16 bits of precision. + 'u_pixel_coord_upper': [pixelX >> 16, pixelY >> 16], + 'u_pixel_coord_lower': [pixelX & 0xFFFF, pixelY & 0xFFFF] + }; +} + +const fillExtrusionUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_lightpos': new performance.Uniform3f(context, locations.u_lightpos), + 'u_lightintensity': new performance.Uniform1f(context, locations.u_lightintensity), + 'u_lightcolor': new performance.Uniform3f(context, locations.u_lightcolor), + 'u_vertical_gradient': new performance.Uniform1f(context, locations.u_vertical_gradient), + 'u_opacity': new performance.Uniform1f(context, locations.u_opacity) +}); +const fillExtrusionPatternUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_lightpos': new performance.Uniform3f(context, locations.u_lightpos), + 'u_lightintensity': new performance.Uniform1f(context, locations.u_lightintensity), + 'u_lightcolor': new performance.Uniform3f(context, locations.u_lightcolor), + 'u_vertical_gradient': new performance.Uniform1f(context, locations.u_vertical_gradient), + 'u_height_factor': new performance.Uniform1f(context, locations.u_height_factor), + // pattern uniforms + 'u_image': new performance.Uniform1i(context, locations.u_image), + 'u_texsize': new performance.Uniform2f(context, locations.u_texsize), + 'u_pixel_coord_upper': new performance.Uniform2f(context, locations.u_pixel_coord_upper), + 'u_pixel_coord_lower': new performance.Uniform2f(context, locations.u_pixel_coord_lower), + 'u_scale': new performance.Uniform3f(context, locations.u_scale), + 'u_fade': new performance.Uniform1f(context, locations.u_fade), + 'u_opacity': new performance.Uniform1f(context, locations.u_opacity) +}); +const fillExtrusionUniformValues = (matrix, painter, shouldUseVerticalGradient, opacity) => { + const light = painter.style.light; + const _lp = light.properties.get('position'); + const lightPos = [_lp.x, _lp.y, _lp.z]; + const lightMat = performance.create$1(); + if (light.properties.get('anchor') === 'viewport') { + performance.fromRotation(lightMat, -painter.transform.angle); + } + performance.transformMat3(lightPos, lightPos, lightMat); + const lightColor = light.properties.get('color'); + return { + 'u_matrix': matrix, + 'u_lightpos': lightPos, + 'u_lightintensity': light.properties.get('intensity'), + 'u_lightcolor': [lightColor.r, lightColor.g, lightColor.b], + 'u_vertical_gradient': +shouldUseVerticalGradient, + 'u_opacity': opacity + }; +}; +const fillExtrusionPatternUniformValues = (matrix, painter, shouldUseVerticalGradient, opacity, coord, crossfade, tile) => { + return performance.extend(fillExtrusionUniformValues(matrix, painter, shouldUseVerticalGradient, opacity), patternUniformValues(crossfade, painter, tile), { + 'u_height_factor': -Math.pow(2, coord.overscaledZ) / tile.tileSize / 8 + }); +}; + +const fillUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix) +}); +const fillPatternUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_image': new performance.Uniform1i(context, locations.u_image), + 'u_texsize': new performance.Uniform2f(context, locations.u_texsize), + 'u_pixel_coord_upper': new performance.Uniform2f(context, locations.u_pixel_coord_upper), + 'u_pixel_coord_lower': new performance.Uniform2f(context, locations.u_pixel_coord_lower), + 'u_scale': new performance.Uniform3f(context, locations.u_scale), + 'u_fade': new performance.Uniform1f(context, locations.u_fade) +}); +const fillOutlineUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_world': new performance.Uniform2f(context, locations.u_world) +}); +const fillOutlinePatternUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_world': new performance.Uniform2f(context, locations.u_world), + 'u_image': new performance.Uniform1i(context, locations.u_image), + 'u_texsize': new performance.Uniform2f(context, locations.u_texsize), + 'u_pixel_coord_upper': new performance.Uniform2f(context, locations.u_pixel_coord_upper), + 'u_pixel_coord_lower': new performance.Uniform2f(context, locations.u_pixel_coord_lower), + 'u_scale': new performance.Uniform3f(context, locations.u_scale), + 'u_fade': new performance.Uniform1f(context, locations.u_fade) +}); +const fillUniformValues = (matrix) => ({ + 'u_matrix': matrix +}); +const fillPatternUniformValues = (matrix, painter, crossfade, tile) => performance.extend(fillUniformValues(matrix), patternUniformValues(crossfade, painter, tile)); +const fillOutlineUniformValues = (matrix, drawingBufferSize) => ({ + 'u_matrix': matrix, + 'u_world': drawingBufferSize +}); +const fillOutlinePatternUniformValues = (matrix, painter, crossfade, tile, drawingBufferSize) => performance.extend(fillPatternUniformValues(matrix, painter, crossfade, tile), { + 'u_world': drawingBufferSize +}); + +const circleUniforms = (context, locations) => ({ + 'u_camera_to_center_distance': new performance.Uniform1f(context, locations.u_camera_to_center_distance), + 'u_scale_with_map': new performance.Uniform1i(context, locations.u_scale_with_map), + 'u_pitch_with_map': new performance.Uniform1i(context, locations.u_pitch_with_map), + 'u_extrude_scale': new performance.Uniform2f(context, locations.u_extrude_scale), + 'u_device_pixel_ratio': new performance.Uniform1f(context, locations.u_device_pixel_ratio), + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix) +}); +const circleUniformValues = (painter, coord, tile, layer) => { + const transform = painter.transform; + let pitchWithMap, extrudeScale; + if (layer.paint.get('circle-pitch-alignment') === 'map') { + const pixelRatio = pixelsToTileUnits(tile, 1, transform.zoom); + pitchWithMap = true; + extrudeScale = [pixelRatio, pixelRatio]; + } + else { + pitchWithMap = false; + extrudeScale = transform.pixelsToGLUnits; + } + return { + 'u_camera_to_center_distance': transform.cameraToCenterDistance, + 'u_scale_with_map': +(layer.paint.get('circle-pitch-scale') === 'map'), + 'u_matrix': painter.translatePosMatrix(coord.posMatrix, tile, layer.paint.get('circle-translate'), layer.paint.get('circle-translate-anchor')), + 'u_pitch_with_map': +(pitchWithMap), + 'u_device_pixel_ratio': painter.pixelRatio, + 'u_extrude_scale': extrudeScale + }; +}; + +const collisionUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_camera_to_center_distance': new performance.Uniform1f(context, locations.u_camera_to_center_distance), + 'u_pixels_to_tile_units': new performance.Uniform1f(context, locations.u_pixels_to_tile_units), + 'u_extrude_scale': new performance.Uniform2f(context, locations.u_extrude_scale), + 'u_overscale_factor': new performance.Uniform1f(context, locations.u_overscale_factor) +}); +const collisionCircleUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_inv_matrix': new performance.UniformMatrix4f(context, locations.u_inv_matrix), + 'u_camera_to_center_distance': new performance.Uniform1f(context, locations.u_camera_to_center_distance), + 'u_viewport_size': new performance.Uniform2f(context, locations.u_viewport_size) +}); +const collisionUniformValues = (matrix, transform, tile) => { + const pixelRatio = pixelsToTileUnits(tile, 1, transform.zoom); + const scale = Math.pow(2, transform.zoom - tile.tileID.overscaledZ); + const overscaleFactor = tile.tileID.overscaleFactor(); + return { + 'u_matrix': matrix, + 'u_camera_to_center_distance': transform.cameraToCenterDistance, + 'u_pixels_to_tile_units': pixelRatio, + 'u_extrude_scale': [transform.pixelsToGLUnits[0] / (pixelRatio * scale), + transform.pixelsToGLUnits[1] / (pixelRatio * scale)], + 'u_overscale_factor': overscaleFactor + }; +}; +const collisionCircleUniformValues = (matrix, invMatrix, transform) => { + return { + 'u_matrix': matrix, + 'u_inv_matrix': invMatrix, + 'u_camera_to_center_distance': transform.cameraToCenterDistance, + 'u_viewport_size': [transform.width, transform.height] + }; +}; + +const debugUniforms = (context, locations) => ({ + 'u_color': new performance.UniformColor(context, locations.u_color), + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_overlay': new performance.Uniform1i(context, locations.u_overlay), + 'u_overlay_scale': new performance.Uniform1f(context, locations.u_overlay_scale) +}); +const debugUniformValues = (matrix, color, scaleRatio = 1) => ({ + 'u_matrix': matrix, + 'u_color': color, + 'u_overlay': 0, + 'u_overlay_scale': scaleRatio +}); + +const clippingMaskUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix) +}); +const clippingMaskUniformValues = (matrix) => ({ + 'u_matrix': matrix +}); + +const heatmapUniforms = (context, locations) => ({ + 'u_extrude_scale': new performance.Uniform1f(context, locations.u_extrude_scale), + 'u_intensity': new performance.Uniform1f(context, locations.u_intensity), + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix) +}); +const heatmapTextureUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_world': new performance.Uniform2f(context, locations.u_world), + 'u_image': new performance.Uniform1i(context, locations.u_image), + 'u_color_ramp': new performance.Uniform1i(context, locations.u_color_ramp), + 'u_opacity': new performance.Uniform1f(context, locations.u_opacity) +}); +const heatmapUniformValues = (matrix, tile, zoom, intensity) => ({ + 'u_matrix': matrix, + 'u_extrude_scale': pixelsToTileUnits(tile, 1, zoom), + 'u_intensity': intensity +}); +const heatmapTextureUniformValues = (painter, layer, textureUnit, colorRampUnit) => { + const matrix = performance.create(); + performance.ortho(matrix, 0, painter.width, painter.height, 0, 0, 1); + const gl = painter.context.gl; + return { + 'u_matrix': matrix, + 'u_world': [gl.drawingBufferWidth, gl.drawingBufferHeight], + 'u_image': textureUnit, + 'u_color_ramp': colorRampUnit, + 'u_opacity': layer.paint.get('heatmap-opacity') + }; +}; + +const hillshadeUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_image': new performance.Uniform1i(context, locations.u_image), + 'u_latrange': new performance.Uniform2f(context, locations.u_latrange), + 'u_light': new performance.Uniform2f(context, locations.u_light), + 'u_shadow': new performance.UniformColor(context, locations.u_shadow), + 'u_highlight': new performance.UniformColor(context, locations.u_highlight), + 'u_accent': new performance.UniformColor(context, locations.u_accent) +}); +const hillshadePrepareUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_image': new performance.Uniform1i(context, locations.u_image), + 'u_dimension': new performance.Uniform2f(context, locations.u_dimension), + 'u_zoom': new performance.Uniform1f(context, locations.u_zoom), + 'u_unpack': new performance.Uniform4f(context, locations.u_unpack) +}); +const hillshadeUniformValues = (painter, tile, layer, coord) => { + const shadow = layer.paint.get('hillshade-shadow-color'); + const highlight = layer.paint.get('hillshade-highlight-color'); + const accent = layer.paint.get('hillshade-accent-color'); + let azimuthal = layer.paint.get('hillshade-illumination-direction') * (Math.PI / 180); + // modify azimuthal angle by map rotation if light is anchored at the viewport + if (layer.paint.get('hillshade-illumination-anchor') === 'viewport') { + azimuthal -= painter.transform.angle; + } + const align = !painter.options.moving; + return { + 'u_matrix': coord ? coord.posMatrix : painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped(), align), + 'u_image': 0, + 'u_latrange': getTileLatRange(painter, tile.tileID), + 'u_light': [layer.paint.get('hillshade-exaggeration'), azimuthal], + 'u_shadow': shadow, + 'u_highlight': highlight, + 'u_accent': accent + }; +}; +const hillshadeUniformPrepareValues = (tileID, dem) => { + const stride = dem.stride; + const matrix = performance.create(); + // Flip rendering at y axis. + performance.ortho(matrix, 0, performance.EXTENT, -performance.EXTENT, 0, 0, 1); + performance.translate(matrix, matrix, [0, -performance.EXTENT, 0]); + return { + 'u_matrix': matrix, + 'u_image': 1, + 'u_dimension': [stride, stride], + 'u_zoom': tileID.overscaledZ, + 'u_unpack': dem.getUnpackVector() + }; +}; +function getTileLatRange(painter, tileID) { + // for scaling the magnitude of a points slope by its latitude + const tilesAtZoom = Math.pow(2, tileID.canonical.z); + const y = tileID.canonical.y; + return [ + new performance.MercatorCoordinate(0, y / tilesAtZoom).toLngLat().lat, + new performance.MercatorCoordinate(0, (y + 1) / tilesAtZoom).toLngLat().lat + ]; +} + +const lineUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_ratio': new performance.Uniform1f(context, locations.u_ratio), + 'u_device_pixel_ratio': new performance.Uniform1f(context, locations.u_device_pixel_ratio), + 'u_units_to_pixels': new performance.Uniform2f(context, locations.u_units_to_pixels) +}); +const lineGradientUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_ratio': new performance.Uniform1f(context, locations.u_ratio), + 'u_device_pixel_ratio': new performance.Uniform1f(context, locations.u_device_pixel_ratio), + 'u_units_to_pixels': new performance.Uniform2f(context, locations.u_units_to_pixels), + 'u_image': new performance.Uniform1i(context, locations.u_image), + 'u_image_height': new performance.Uniform1f(context, locations.u_image_height) +}); +const linePatternUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_texsize': new performance.Uniform2f(context, locations.u_texsize), + 'u_ratio': new performance.Uniform1f(context, locations.u_ratio), + 'u_device_pixel_ratio': new performance.Uniform1f(context, locations.u_device_pixel_ratio), + 'u_image': new performance.Uniform1i(context, locations.u_image), + 'u_units_to_pixels': new performance.Uniform2f(context, locations.u_units_to_pixels), + 'u_scale': new performance.Uniform3f(context, locations.u_scale), + 'u_fade': new performance.Uniform1f(context, locations.u_fade) +}); +const lineSDFUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_ratio': new performance.Uniform1f(context, locations.u_ratio), + 'u_device_pixel_ratio': new performance.Uniform1f(context, locations.u_device_pixel_ratio), + 'u_units_to_pixels': new performance.Uniform2f(context, locations.u_units_to_pixels), + 'u_patternscale_a': new performance.Uniform2f(context, locations.u_patternscale_a), + 'u_patternscale_b': new performance.Uniform2f(context, locations.u_patternscale_b), + 'u_sdfgamma': new performance.Uniform1f(context, locations.u_sdfgamma), + 'u_image': new performance.Uniform1i(context, locations.u_image), + 'u_tex_y_a': new performance.Uniform1f(context, locations.u_tex_y_a), + 'u_tex_y_b': new performance.Uniform1f(context, locations.u_tex_y_b), + 'u_mix': new performance.Uniform1f(context, locations.u_mix) +}); +const lineUniformValues = (painter, tile, layer, coord) => { + const transform = painter.transform; + return { + 'u_matrix': calculateMatrix(painter, tile, layer, coord), + 'u_ratio': 1 / pixelsToTileUnits(tile, 1, transform.zoom), + 'u_device_pixel_ratio': painter.pixelRatio, + 'u_units_to_pixels': [ + 1 / transform.pixelsToGLUnits[0], + 1 / transform.pixelsToGLUnits[1] + ] + }; +}; +const lineGradientUniformValues = (painter, tile, layer, imageHeight, coord) => { + return performance.extend(lineUniformValues(painter, tile, layer, coord), { + 'u_image': 0, + 'u_image_height': imageHeight, + }); +}; +const linePatternUniformValues = (painter, tile, layer, crossfade, coord) => { + const transform = painter.transform; + const tileZoomRatio = calculateTileRatio(tile, transform); + return { + 'u_matrix': calculateMatrix(painter, tile, layer, coord), + 'u_texsize': tile.imageAtlasTexture.size, + // camera zoom ratio + 'u_ratio': 1 / pixelsToTileUnits(tile, 1, transform.zoom), + 'u_device_pixel_ratio': painter.pixelRatio, + 'u_image': 0, + 'u_scale': [tileZoomRatio, crossfade.fromScale, crossfade.toScale], + 'u_fade': crossfade.t, + 'u_units_to_pixels': [ + 1 / transform.pixelsToGLUnits[0], + 1 / transform.pixelsToGLUnits[1] + ] + }; +}; +const lineSDFUniformValues = (painter, tile, layer, dasharray, crossfade, coord) => { + const transform = painter.transform; + const lineAtlas = painter.lineAtlas; + const tileRatio = calculateTileRatio(tile, transform); + const round = layer.layout.get('line-cap') === 'round'; + const posA = lineAtlas.getDash(dasharray.from, round); + const posB = lineAtlas.getDash(dasharray.to, round); + const widthA = posA.width * crossfade.fromScale; + const widthB = posB.width * crossfade.toScale; + return performance.extend(lineUniformValues(painter, tile, layer, coord), { + 'u_patternscale_a': [tileRatio / widthA, -posA.height / 2], + 'u_patternscale_b': [tileRatio / widthB, -posB.height / 2], + 'u_sdfgamma': lineAtlas.width / (Math.min(widthA, widthB) * 256 * painter.pixelRatio) / 2, + 'u_image': 0, + 'u_tex_y_a': posA.y, + 'u_tex_y_b': posB.y, + 'u_mix': crossfade.t + }); +}; +function calculateTileRatio(tile, transform) { + return 1 / pixelsToTileUnits(tile, 1, transform.tileZoom); +} +function calculateMatrix(painter, tile, layer, coord) { + return painter.translatePosMatrix(coord ? coord.posMatrix : tile.tileID.posMatrix, tile, layer.paint.get('line-translate'), layer.paint.get('line-translate-anchor')); +} + +const rasterUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_tl_parent': new performance.Uniform2f(context, locations.u_tl_parent), + 'u_scale_parent': new performance.Uniform1f(context, locations.u_scale_parent), + 'u_buffer_scale': new performance.Uniform1f(context, locations.u_buffer_scale), + 'u_fade_t': new performance.Uniform1f(context, locations.u_fade_t), + 'u_opacity': new performance.Uniform1f(context, locations.u_opacity), + 'u_image0': new performance.Uniform1i(context, locations.u_image0), + 'u_image1': new performance.Uniform1i(context, locations.u_image1), + 'u_brightness_low': new performance.Uniform1f(context, locations.u_brightness_low), + 'u_brightness_high': new performance.Uniform1f(context, locations.u_brightness_high), + 'u_saturation_factor': new performance.Uniform1f(context, locations.u_saturation_factor), + 'u_contrast_factor': new performance.Uniform1f(context, locations.u_contrast_factor), + 'u_spin_weights': new performance.Uniform3f(context, locations.u_spin_weights) +}); +const rasterUniformValues = (matrix, parentTL, parentScaleBy, fade, layer) => ({ + 'u_matrix': matrix, + 'u_tl_parent': parentTL, + 'u_scale_parent': parentScaleBy, + 'u_buffer_scale': 1, + 'u_fade_t': fade.mix, + 'u_opacity': fade.opacity * layer.paint.get('raster-opacity'), + 'u_image0': 0, + 'u_image1': 1, + 'u_brightness_low': layer.paint.get('raster-brightness-min'), + 'u_brightness_high': layer.paint.get('raster-brightness-max'), + 'u_saturation_factor': saturationFactor(layer.paint.get('raster-saturation')), + 'u_contrast_factor': contrastFactor(layer.paint.get('raster-contrast')), + 'u_spin_weights': spinWeights(layer.paint.get('raster-hue-rotate')) +}); +function spinWeights(angle) { + angle *= Math.PI / 180; + const s = Math.sin(angle); + const c = Math.cos(angle); + return [ + (2 * c + 1) / 3, + (-Math.sqrt(3) * s - c + 1) / 3, + (Math.sqrt(3) * s - c + 1) / 3 + ]; +} +function contrastFactor(contrast) { + return contrast > 0 ? + 1 / (1 - contrast) : + 1 + contrast; +} +function saturationFactor(saturation) { + return saturation > 0 ? + 1 - 1 / (1.001 - saturation) : + -saturation; +} + +const symbolIconUniforms = (context, locations) => ({ + 'u_is_size_zoom_constant': new performance.Uniform1i(context, locations.u_is_size_zoom_constant), + 'u_is_size_feature_constant': new performance.Uniform1i(context, locations.u_is_size_feature_constant), + 'u_size_t': new performance.Uniform1f(context, locations.u_size_t), + 'u_size': new performance.Uniform1f(context, locations.u_size), + 'u_camera_to_center_distance': new performance.Uniform1f(context, locations.u_camera_to_center_distance), + 'u_pitch': new performance.Uniform1f(context, locations.u_pitch), + 'u_rotate_symbol': new performance.Uniform1i(context, locations.u_rotate_symbol), + 'u_aspect_ratio': new performance.Uniform1f(context, locations.u_aspect_ratio), + 'u_fade_change': new performance.Uniform1f(context, locations.u_fade_change), + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_label_plane_matrix': new performance.UniformMatrix4f(context, locations.u_label_plane_matrix), + 'u_coord_matrix': new performance.UniformMatrix4f(context, locations.u_coord_matrix), + 'u_is_text': new performance.Uniform1i(context, locations.u_is_text), + 'u_pitch_with_map': new performance.Uniform1i(context, locations.u_pitch_with_map), + 'u_texsize': new performance.Uniform2f(context, locations.u_texsize), + 'u_texture': new performance.Uniform1i(context, locations.u_texture) +}); +const symbolSDFUniforms = (context, locations) => ({ + 'u_is_size_zoom_constant': new performance.Uniform1i(context, locations.u_is_size_zoom_constant), + 'u_is_size_feature_constant': new performance.Uniform1i(context, locations.u_is_size_feature_constant), + 'u_size_t': new performance.Uniform1f(context, locations.u_size_t), + 'u_size': new performance.Uniform1f(context, locations.u_size), + 'u_camera_to_center_distance': new performance.Uniform1f(context, locations.u_camera_to_center_distance), + 'u_pitch': new performance.Uniform1f(context, locations.u_pitch), + 'u_rotate_symbol': new performance.Uniform1i(context, locations.u_rotate_symbol), + 'u_aspect_ratio': new performance.Uniform1f(context, locations.u_aspect_ratio), + 'u_fade_change': new performance.Uniform1f(context, locations.u_fade_change), + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_label_plane_matrix': new performance.UniformMatrix4f(context, locations.u_label_plane_matrix), + 'u_coord_matrix': new performance.UniformMatrix4f(context, locations.u_coord_matrix), + 'u_is_text': new performance.Uniform1i(context, locations.u_is_text), + 'u_pitch_with_map': new performance.Uniform1i(context, locations.u_pitch_with_map), + 'u_texsize': new performance.Uniform2f(context, locations.u_texsize), + 'u_texture': new performance.Uniform1i(context, locations.u_texture), + 'u_gamma_scale': new performance.Uniform1f(context, locations.u_gamma_scale), + 'u_device_pixel_ratio': new performance.Uniform1f(context, locations.u_device_pixel_ratio), + 'u_is_halo': new performance.Uniform1i(context, locations.u_is_halo) +}); +const symbolTextAndIconUniforms = (context, locations) => ({ + 'u_is_size_zoom_constant': new performance.Uniform1i(context, locations.u_is_size_zoom_constant), + 'u_is_size_feature_constant': new performance.Uniform1i(context, locations.u_is_size_feature_constant), + 'u_size_t': new performance.Uniform1f(context, locations.u_size_t), + 'u_size': new performance.Uniform1f(context, locations.u_size), + 'u_camera_to_center_distance': new performance.Uniform1f(context, locations.u_camera_to_center_distance), + 'u_pitch': new performance.Uniform1f(context, locations.u_pitch), + 'u_rotate_symbol': new performance.Uniform1i(context, locations.u_rotate_symbol), + 'u_aspect_ratio': new performance.Uniform1f(context, locations.u_aspect_ratio), + 'u_fade_change': new performance.Uniform1f(context, locations.u_fade_change), + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_label_plane_matrix': new performance.UniformMatrix4f(context, locations.u_label_plane_matrix), + 'u_coord_matrix': new performance.UniformMatrix4f(context, locations.u_coord_matrix), + 'u_is_text': new performance.Uniform1i(context, locations.u_is_text), + 'u_pitch_with_map': new performance.Uniform1i(context, locations.u_pitch_with_map), + 'u_texsize': new performance.Uniform2f(context, locations.u_texsize), + 'u_texsize_icon': new performance.Uniform2f(context, locations.u_texsize_icon), + 'u_texture': new performance.Uniform1i(context, locations.u_texture), + 'u_texture_icon': new performance.Uniform1i(context, locations.u_texture_icon), + 'u_gamma_scale': new performance.Uniform1f(context, locations.u_gamma_scale), + 'u_device_pixel_ratio': new performance.Uniform1f(context, locations.u_device_pixel_ratio), + 'u_is_halo': new performance.Uniform1i(context, locations.u_is_halo) +}); +const symbolIconUniformValues = (functionType, size, rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix, glCoordMatrix, isText, texSize) => { + const transform = painter.transform; + return { + 'u_is_size_zoom_constant': +(functionType === 'constant' || functionType === 'source'), + 'u_is_size_feature_constant': +(functionType === 'constant' || functionType === 'camera'), + 'u_size_t': size ? size.uSizeT : 0, + 'u_size': size ? size.uSize : 0, + 'u_camera_to_center_distance': transform.cameraToCenterDistance, + 'u_pitch': transform.pitch / 360 * 2 * Math.PI, + 'u_rotate_symbol': +rotateInShader, + 'u_aspect_ratio': transform.width / transform.height, + 'u_fade_change': painter.options.fadeDuration ? painter.symbolFadeChange : 1, + 'u_matrix': matrix, + 'u_label_plane_matrix': labelPlaneMatrix, + 'u_coord_matrix': glCoordMatrix, + 'u_is_text': +isText, + 'u_pitch_with_map': +pitchWithMap, + 'u_texsize': texSize, + 'u_texture': 0 + }; +}; +const symbolSDFUniformValues = (functionType, size, rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix, glCoordMatrix, isText, texSize, isHalo) => { + const transform = painter.transform; + return performance.extend(symbolIconUniformValues(functionType, size, rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix, glCoordMatrix, isText, texSize), { + 'u_gamma_scale': (pitchWithMap ? Math.cos(transform._pitch) * transform.cameraToCenterDistance : 1), + 'u_device_pixel_ratio': painter.pixelRatio, + 'u_is_halo': +isHalo + }); +}; +const symbolTextAndIconUniformValues = (functionType, size, rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix, glCoordMatrix, texSizeSDF, texSizeIcon) => { + return performance.extend(symbolSDFUniformValues(functionType, size, rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix, glCoordMatrix, true, texSizeSDF, true), { + 'u_texsize_icon': texSizeIcon, + 'u_texture_icon': 1 + }); +}; + +const backgroundUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_opacity': new performance.Uniform1f(context, locations.u_opacity), + 'u_color': new performance.UniformColor(context, locations.u_color) +}); +const backgroundPatternUniforms = (context, locations) => ({ + 'u_matrix': new performance.UniformMatrix4f(context, locations.u_matrix), + 'u_opacity': new performance.Uniform1f(context, locations.u_opacity), + 'u_image': new performance.Uniform1i(context, locations.u_image), + 'u_pattern_tl_a': new performance.Uniform2f(context, locations.u_pattern_tl_a), + 'u_pattern_br_a': new performance.Uniform2f(context, locations.u_pattern_br_a), + 'u_pattern_tl_b': new performance.Uniform2f(context, locations.u_pattern_tl_b), + 'u_pattern_br_b': new performance.Uniform2f(context, locations.u_pattern_br_b), + 'u_texsize': new performance.Uniform2f(context, locations.u_texsize), + 'u_mix': new performance.Uniform1f(context, locations.u_mix), + 'u_pattern_size_a': new performance.Uniform2f(context, locations.u_pattern_size_a), + 'u_pattern_size_b': new performance.Uniform2f(context, locations.u_pattern_size_b), + 'u_scale_a': new performance.Uniform1f(context, locations.u_scale_a), + 'u_scale_b': new performance.Uniform1f(context, locations.u_scale_b), + 'u_pixel_coord_upper': new performance.Uniform2f(context, locations.u_pixel_coord_upper), + 'u_pixel_coord_lower': new performance.Uniform2f(context, locations.u_pixel_coord_lower), + 'u_tile_units_to_pixels': new performance.Uniform1f(context, locations.u_tile_units_to_pixels) +}); +const backgroundUniformValues = (matrix, opacity, color) => ({ + 'u_matrix': matrix, + 'u_opacity': opacity, + 'u_color': color +}); +const backgroundPatternUniformValues = (matrix, opacity, painter, image, tile, crossfade) => performance.extend(bgPatternUniformValues(image, crossfade, painter, tile), { + 'u_matrix': matrix, + 'u_opacity': opacity +}); + +const programUniforms = { + fillExtrusion: fillExtrusionUniforms, + fillExtrusionPattern: fillExtrusionPatternUniforms, + fill: fillUniforms, + fillPattern: fillPatternUniforms, + fillOutline: fillOutlineUniforms, + fillOutlinePattern: fillOutlinePatternUniforms, + circle: circleUniforms, + collisionBox: collisionUniforms, + collisionCircle: collisionCircleUniforms, + debug: debugUniforms, + clippingMask: clippingMaskUniforms, + heatmap: heatmapUniforms, + heatmapTexture: heatmapTextureUniforms, + hillshade: hillshadeUniforms, + hillshadePrepare: hillshadePrepareUniforms, + line: lineUniforms, + lineGradient: lineGradientUniforms, + linePattern: linePatternUniforms, + lineSDF: lineSDFUniforms, + raster: rasterUniforms, + symbolIcon: symbolIconUniforms, + symbolSDF: symbolSDFUniforms, + symbolTextAndIcon: symbolTextAndIconUniforms, + background: backgroundUniforms, + backgroundPattern: backgroundPatternUniforms, + terrain: terrainUniforms, + terrainDepth: terrainDepthUniforms, + terrainCoords: terrainCoordsUniforms +}; + +/** + * @internal + * an index buffer class + */ +class IndexBuffer { + constructor(context, array, dynamicDraw) { + this.context = context; + const gl = context.gl; + this.buffer = gl.createBuffer(); + this.dynamicDraw = Boolean(dynamicDraw); + // The bound index buffer is part of vertex array object state. We don't want to + // modify whatever VAO happens to be currently bound, so make sure the default + // vertex array provided by the context is bound instead. + this.context.unbindVAO(); + context.bindElementBuffer.set(this.buffer); + gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, array.arrayBuffer, this.dynamicDraw ? gl.DYNAMIC_DRAW : gl.STATIC_DRAW); + if (!this.dynamicDraw) { + delete array.arrayBuffer; + } + } + bind() { + this.context.bindElementBuffer.set(this.buffer); + } + updateData(array) { + const gl = this.context.gl; + if (!this.dynamicDraw) + throw new Error('Attempted to update data while not in dynamic mode.'); + // The right VAO will get this buffer re-bound later in VertexArrayObject#bind + // See https://github.com/mapbox/mapbox-gl-js/issues/5620 + this.context.unbindVAO(); + this.bind(); + gl.bufferSubData(gl.ELEMENT_ARRAY_BUFFER, 0, array.arrayBuffer); + } + destroy() { + const gl = this.context.gl; + if (this.buffer) { + gl.deleteBuffer(this.buffer); + delete this.buffer; + } + } +} + +/** + * An Enum for AttributeType + */ +const AttributeType = { + Int8: 'BYTE', + Uint8: 'UNSIGNED_BYTE', + Int16: 'SHORT', + Uint16: 'UNSIGNED_SHORT', + Int32: 'INT', + Uint32: 'UNSIGNED_INT', + Float32: 'FLOAT' +}; +/** + * @internal + * The `VertexBuffer` class turns a `StructArray` into a WebGL buffer. Each member of the StructArray's + * Struct type is converted to a WebGL attribute. + */ +class VertexBuffer { + /** + * @param dynamicDraw - Whether this buffer will be repeatedly updated. + */ + constructor(context, array, attributes, dynamicDraw) { + this.length = array.length; + this.attributes = attributes; + this.itemSize = array.bytesPerElement; + this.dynamicDraw = dynamicDraw; + this.context = context; + const gl = context.gl; + this.buffer = gl.createBuffer(); + context.bindVertexBuffer.set(this.buffer); + gl.bufferData(gl.ARRAY_BUFFER, array.arrayBuffer, this.dynamicDraw ? gl.DYNAMIC_DRAW : gl.STATIC_DRAW); + if (!this.dynamicDraw) { + delete array.arrayBuffer; + } + } + bind() { + this.context.bindVertexBuffer.set(this.buffer); + } + updateData(array) { + if (array.length !== this.length) + throw new Error(`Length of new data is ${array.length}, which doesn't match current length of ${this.length}`); + const gl = this.context.gl; + this.bind(); + gl.bufferSubData(gl.ARRAY_BUFFER, 0, array.arrayBuffer); + } + enableAttributes(gl, program) { + for (let j = 0; j < this.attributes.length; j++) { + const member = this.attributes[j]; + const attribIndex = program.attributes[member.name]; + if (attribIndex !== undefined) { + gl.enableVertexAttribArray(attribIndex); + } + } + } + /** + * Set the attribute pointers in a WebGL context + * @param gl - The WebGL context + * @param program - The active WebGL program + * @param vertexOffset - Index of the starting vertex of the segment + */ + setVertexAttribPointers(gl, program, vertexOffset) { + for (let j = 0; j < this.attributes.length; j++) { + const member = this.attributes[j]; + const attribIndex = program.attributes[member.name]; + if (attribIndex !== undefined) { + gl.vertexAttribPointer(attribIndex, member.components, gl[AttributeType[member.type]], false, this.itemSize, member.offset + (this.itemSize * (vertexOffset || 0))); + } + } + } + /** + * Destroy the GL buffer bound to the given WebGL context + */ + destroy() { + const gl = this.context.gl; + if (this.buffer) { + gl.deleteBuffer(this.buffer); + delete this.buffer; + } + } +} + +const cache = new WeakMap(); +function isWebGL2(gl) { + var _a; + if (cache.has(gl)) { + return cache.get(gl); + } + else { + const value = (_a = gl.getParameter(gl.VERSION)) === null || _a === void 0 ? void 0 : _a.startsWith('WebGL 2.0'); + cache.set(gl, value); + return value; + } +} + +class BaseValue { + constructor(context) { + this.gl = context.gl; + this.default = this.getDefault(); + this.current = this.default; + this.dirty = false; + } + get() { + return this.current; + } + set(value) { + // overridden in child classes; + } + getDefault() { + return this.default; // overriden in child classes + } + setDefault() { + this.set(this.default); + } +} +class ClearColor extends BaseValue { + getDefault() { + return performance.Color.transparent; + } + set(v) { + const c = this.current; + if (v.r === c.r && v.g === c.g && v.b === c.b && v.a === c.a && !this.dirty) + return; + this.gl.clearColor(v.r, v.g, v.b, v.a); + this.current = v; + this.dirty = false; + } +} +class ClearDepth extends BaseValue { + getDefault() { + return 1; + } + set(v) { + if (v === this.current && !this.dirty) + return; + this.gl.clearDepth(v); + this.current = v; + this.dirty = false; + } +} +class ClearStencil extends BaseValue { + getDefault() { + return 0; + } + set(v) { + if (v === this.current && !this.dirty) + return; + this.gl.clearStencil(v); + this.current = v; + this.dirty = false; + } +} +class ColorMask extends BaseValue { + getDefault() { + return [true, true, true, true]; + } + set(v) { + const c = this.current; + if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && v[3] === c[3] && !this.dirty) + return; + this.gl.colorMask(v[0], v[1], v[2], v[3]); + this.current = v; + this.dirty = false; + } +} +class DepthMask extends BaseValue { + getDefault() { + return true; + } + set(v) { + if (v === this.current && !this.dirty) + return; + this.gl.depthMask(v); + this.current = v; + this.dirty = false; + } +} +class StencilMask extends BaseValue { + getDefault() { + return 0xFF; + } + set(v) { + if (v === this.current && !this.dirty) + return; + this.gl.stencilMask(v); + this.current = v; + this.dirty = false; + } +} +class StencilFunc extends BaseValue { + getDefault() { + return { + func: this.gl.ALWAYS, + ref: 0, + mask: 0xFF + }; + } + set(v) { + const c = this.current; + if (v.func === c.func && v.ref === c.ref && v.mask === c.mask && !this.dirty) + return; + this.gl.stencilFunc(v.func, v.ref, v.mask); + this.current = v; + this.dirty = false; + } +} +class StencilOp extends BaseValue { + getDefault() { + const gl = this.gl; + return [gl.KEEP, gl.KEEP, gl.KEEP]; + } + set(v) { + const c = this.current; + if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && !this.dirty) + return; + this.gl.stencilOp(v[0], v[1], v[2]); + this.current = v; + this.dirty = false; + } +} +class StencilTest extends BaseValue { + getDefault() { + return false; + } + set(v) { + if (v === this.current && !this.dirty) + return; + const gl = this.gl; + if (v) { + gl.enable(gl.STENCIL_TEST); + } + else { + gl.disable(gl.STENCIL_TEST); + } + this.current = v; + this.dirty = false; + } +} +class DepthRange extends BaseValue { + getDefault() { + return [0, 1]; + } + set(v) { + const c = this.current; + if (v[0] === c[0] && v[1] === c[1] && !this.dirty) + return; + this.gl.depthRange(v[0], v[1]); + this.current = v; + this.dirty = false; + } +} +class DepthTest extends BaseValue { + getDefault() { + return false; + } + set(v) { + if (v === this.current && !this.dirty) + return; + const gl = this.gl; + if (v) { + gl.enable(gl.DEPTH_TEST); + } + else { + gl.disable(gl.DEPTH_TEST); + } + this.current = v; + this.dirty = false; + } +} +class DepthFunc extends BaseValue { + getDefault() { + return this.gl.LESS; + } + set(v) { + if (v === this.current && !this.dirty) + return; + this.gl.depthFunc(v); + this.current = v; + this.dirty = false; + } +} +class Blend extends BaseValue { + getDefault() { + return false; + } + set(v) { + if (v === this.current && !this.dirty) + return; + const gl = this.gl; + if (v) { + gl.enable(gl.BLEND); + } + else { + gl.disable(gl.BLEND); + } + this.current = v; + this.dirty = false; + } +} +class BlendFunc extends BaseValue { + getDefault() { + const gl = this.gl; + return [gl.ONE, gl.ZERO]; + } + set(v) { + const c = this.current; + if (v[0] === c[0] && v[1] === c[1] && !this.dirty) + return; + this.gl.blendFunc(v[0], v[1]); + this.current = v; + this.dirty = false; + } +} +class BlendColor extends BaseValue { + getDefault() { + return performance.Color.transparent; + } + set(v) { + const c = this.current; + if (v.r === c.r && v.g === c.g && v.b === c.b && v.a === c.a && !this.dirty) + return; + this.gl.blendColor(v.r, v.g, v.b, v.a); + this.current = v; + this.dirty = false; + } +} +class BlendEquation extends BaseValue { + getDefault() { + return this.gl.FUNC_ADD; + } + set(v) { + if (v === this.current && !this.dirty) + return; + this.gl.blendEquation(v); + this.current = v; + this.dirty = false; + } +} +class CullFace extends BaseValue { + getDefault() { + return false; + } + set(v) { + if (v === this.current && !this.dirty) + return; + const gl = this.gl; + if (v) { + gl.enable(gl.CULL_FACE); + } + else { + gl.disable(gl.CULL_FACE); + } + this.current = v; + this.dirty = false; + } +} +class CullFaceSide extends BaseValue { + getDefault() { + return this.gl.BACK; + } + set(v) { + if (v === this.current && !this.dirty) + return; + this.gl.cullFace(v); + this.current = v; + this.dirty = false; + } +} +class FrontFace extends BaseValue { + getDefault() { + return this.gl.CCW; + } + set(v) { + if (v === this.current && !this.dirty) + return; + this.gl.frontFace(v); + this.current = v; + this.dirty = false; + } +} +class ProgramValue extends BaseValue { + getDefault() { + return null; + } + set(v) { + if (v === this.current && !this.dirty) + return; + this.gl.useProgram(v); + this.current = v; + this.dirty = false; + } +} +class ActiveTextureUnit extends BaseValue { + getDefault() { + return this.gl.TEXTURE0; + } + set(v) { + if (v === this.current && !this.dirty) + return; + this.gl.activeTexture(v); + this.current = v; + this.dirty = false; + } +} +class Viewport extends BaseValue { + getDefault() { + const gl = this.gl; + return [0, 0, gl.drawingBufferWidth, gl.drawingBufferHeight]; + } + set(v) { + const c = this.current; + if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && v[3] === c[3] && !this.dirty) + return; + this.gl.viewport(v[0], v[1], v[2], v[3]); + this.current = v; + this.dirty = false; + } +} +class BindFramebuffer extends BaseValue { + getDefault() { + return null; + } + set(v) { + if (v === this.current && !this.dirty) + return; + const gl = this.gl; + gl.bindFramebuffer(gl.FRAMEBUFFER, v); + this.current = v; + this.dirty = false; + } +} +class BindRenderbuffer extends BaseValue { + getDefault() { + return null; + } + set(v) { + if (v === this.current && !this.dirty) + return; + const gl = this.gl; + gl.bindRenderbuffer(gl.RENDERBUFFER, v); + this.current = v; + this.dirty = false; + } +} +class BindTexture extends BaseValue { + getDefault() { + return null; + } + set(v) { + if (v === this.current && !this.dirty) + return; + const gl = this.gl; + gl.bindTexture(gl.TEXTURE_2D, v); + this.current = v; + this.dirty = false; + } +} +class BindVertexBuffer extends BaseValue { + getDefault() { + return null; + } + set(v) { + if (v === this.current && !this.dirty) + return; + const gl = this.gl; + gl.bindBuffer(gl.ARRAY_BUFFER, v); + this.current = v; + this.dirty = false; + } +} +class BindElementBuffer extends BaseValue { + getDefault() { + return null; + } + set(v) { + // Always rebind + const gl = this.gl; + gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, v); + this.current = v; + this.dirty = false; + } +} +class BindVertexArray extends BaseValue { + getDefault() { + return null; + } + set(v) { + var _a; + if (v === this.current && !this.dirty) + return; + const gl = this.gl; + if (isWebGL2(gl)) { + gl.bindVertexArray(v); + } + else { + (_a = gl.getExtension('OES_vertex_array_object')) === null || _a === void 0 ? void 0 : _a.bindVertexArrayOES(v); + } + this.current = v; + this.dirty = false; + } +} +class PixelStoreUnpack extends BaseValue { + getDefault() { + return 4; + } + set(v) { + if (v === this.current && !this.dirty) + return; + const gl = this.gl; + gl.pixelStorei(gl.UNPACK_ALIGNMENT, v); + this.current = v; + this.dirty = false; + } +} +class PixelStoreUnpackPremultiplyAlpha extends BaseValue { + getDefault() { + return false; + } + set(v) { + if (v === this.current && !this.dirty) + return; + const gl = this.gl; + gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, v); + this.current = v; + this.dirty = false; + } +} +class PixelStoreUnpackFlipY extends BaseValue { + getDefault() { + return false; + } + set(v) { + if (v === this.current && !this.dirty) + return; + const gl = this.gl; + gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, v); + this.current = v; + this.dirty = false; + } +} +class FramebufferAttachment extends BaseValue { + constructor(context, parent) { + super(context); + this.context = context; + this.parent = parent; + } + getDefault() { + return null; + } +} +class ColorAttachment extends FramebufferAttachment { + setDirty() { + this.dirty = true; + } + set(v) { + if (v === this.current && !this.dirty) + return; + this.context.bindFramebuffer.set(this.parent); + // note: it's possible to attach a renderbuffer to the color + // attachment point, but thus far MBGL only uses textures for color + const gl = this.gl; + gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, v, 0); + this.current = v; + this.dirty = false; + } +} +class DepthAttachment extends FramebufferAttachment { + set(v) { + if (v === this.current && !this.dirty) + return; + this.context.bindFramebuffer.set(this.parent); + // note: it's possible to attach a texture to the depth attachment + // point, but thus far MBGL only uses renderbuffers for depth + const gl = this.gl; + gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_ATTACHMENT, gl.RENDERBUFFER, v); + this.current = v; + this.dirty = false; + } +} +class DepthStencilAttachment extends FramebufferAttachment { + set(v) { + if (v === this.current && !this.dirty) + return; + this.context.bindFramebuffer.set(this.parent); + // note: it's possible to attach a texture to the depth attachment + // point, but thus far MBGL only uses renderbuffers for depth + const gl = this.gl; + gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, v); + this.current = v; + this.dirty = false; + } +} + +/** + * @internal + * A framebuffer holder object + */ +class Framebuffer { + constructor(context, width, height, hasDepth, hasStencil) { + this.context = context; + this.width = width; + this.height = height; + const gl = context.gl; + const fbo = this.framebuffer = gl.createFramebuffer(); + this.colorAttachment = new ColorAttachment(context, fbo); + if (hasDepth) { + this.depthAttachment = hasStencil ? new DepthStencilAttachment(context, fbo) : new DepthAttachment(context, fbo); + } + else if (hasStencil) { + throw new Error('Stencil cannot be setted without depth'); + } + if (gl.checkFramebufferStatus(gl.FRAMEBUFFER) !== gl.FRAMEBUFFER_COMPLETE) { + throw new Error('Framebuffer is not complete'); + } + } + destroy() { + const gl = this.context.gl; + const texture = this.colorAttachment.get(); + if (texture) + gl.deleteTexture(texture); + if (this.depthAttachment) { + const renderbuffer = this.depthAttachment.get(); + if (renderbuffer) + gl.deleteRenderbuffer(renderbuffer); + } + gl.deleteFramebuffer(this.framebuffer); + } +} + +const ZERO = 0x0000; +const ONE = 0x0001; +const ONE_MINUS_SRC_ALPHA = 0x0303; +class ColorMode { + constructor(blendFunction, blendColor, mask) { + this.blendFunction = blendFunction; + this.blendColor = blendColor; + this.mask = mask; + } +} +ColorMode.Replace = [ONE, ZERO]; +ColorMode.disabled = new ColorMode(ColorMode.Replace, performance.Color.transparent, [false, false, false, false]); +ColorMode.unblended = new ColorMode(ColorMode.Replace, performance.Color.transparent, [true, true, true, true]); +ColorMode.alphaBlended = new ColorMode([ONE, ONE_MINUS_SRC_ALPHA], performance.Color.transparent, [true, true, true, true]); + +/** + * @internal + * A webgl wrapper class to allow injection, mocking and abstaction + */ +class Context { + constructor(gl) { + var _a, _b; + this.gl = gl; + this.clearColor = new ClearColor(this); + this.clearDepth = new ClearDepth(this); + this.clearStencil = new ClearStencil(this); + this.colorMask = new ColorMask(this); + this.depthMask = new DepthMask(this); + this.stencilMask = new StencilMask(this); + this.stencilFunc = new StencilFunc(this); + this.stencilOp = new StencilOp(this); + this.stencilTest = new StencilTest(this); + this.depthRange = new DepthRange(this); + this.depthTest = new DepthTest(this); + this.depthFunc = new DepthFunc(this); + this.blend = new Blend(this); + this.blendFunc = new BlendFunc(this); + this.blendColor = new BlendColor(this); + this.blendEquation = new BlendEquation(this); + this.cullFace = new CullFace(this); + this.cullFaceSide = new CullFaceSide(this); + this.frontFace = new FrontFace(this); + this.program = new ProgramValue(this); + this.activeTexture = new ActiveTextureUnit(this); + this.viewport = new Viewport(this); + this.bindFramebuffer = new BindFramebuffer(this); + this.bindRenderbuffer = new BindRenderbuffer(this); + this.bindTexture = new BindTexture(this); + this.bindVertexBuffer = new BindVertexBuffer(this); + this.bindElementBuffer = new BindElementBuffer(this); + this.bindVertexArray = new BindVertexArray(this); + this.pixelStoreUnpack = new PixelStoreUnpack(this); + this.pixelStoreUnpackPremultiplyAlpha = new PixelStoreUnpackPremultiplyAlpha(this); + this.pixelStoreUnpackFlipY = new PixelStoreUnpackFlipY(this); + this.extTextureFilterAnisotropic = (gl.getExtension('EXT_texture_filter_anisotropic') || + gl.getExtension('MOZ_EXT_texture_filter_anisotropic') || + gl.getExtension('WEBKIT_EXT_texture_filter_anisotropic')); + if (this.extTextureFilterAnisotropic) { + this.extTextureFilterAnisotropicMax = gl.getParameter(this.extTextureFilterAnisotropic.MAX_TEXTURE_MAX_ANISOTROPY_EXT); + } + this.maxTextureSize = gl.getParameter(gl.MAX_TEXTURE_SIZE); + if (isWebGL2(gl)) { + this.HALF_FLOAT = gl.HALF_FLOAT; + const extColorBufferHalfFloat = gl.getExtension('EXT_color_buffer_half_float'); + this.RGBA16F = (_a = gl.RGBA16F) !== null && _a !== void 0 ? _a : extColorBufferHalfFloat === null || extColorBufferHalfFloat === void 0 ? void 0 : extColorBufferHalfFloat.RGBA16F_EXT; + this.RGB16F = (_b = gl.RGB16F) !== null && _b !== void 0 ? _b : extColorBufferHalfFloat === null || extColorBufferHalfFloat === void 0 ? void 0 : extColorBufferHalfFloat.RGB16F_EXT; + gl.getExtension('EXT_color_buffer_float'); + } + else { + gl.getExtension('EXT_color_buffer_half_float'); + gl.getExtension('OES_texture_half_float_linear'); + const extTextureHalfFloat = gl.getExtension('OES_texture_half_float'); + this.HALF_FLOAT = extTextureHalfFloat === null || extTextureHalfFloat === void 0 ? void 0 : extTextureHalfFloat.HALF_FLOAT_OES; + } + } + setDefault() { + this.unbindVAO(); + this.clearColor.setDefault(); + this.clearDepth.setDefault(); + this.clearStencil.setDefault(); + this.colorMask.setDefault(); + this.depthMask.setDefault(); + this.stencilMask.setDefault(); + this.stencilFunc.setDefault(); + this.stencilOp.setDefault(); + this.stencilTest.setDefault(); + this.depthRange.setDefault(); + this.depthTest.setDefault(); + this.depthFunc.setDefault(); + this.blend.setDefault(); + this.blendFunc.setDefault(); + this.blendColor.setDefault(); + this.blendEquation.setDefault(); + this.cullFace.setDefault(); + this.cullFaceSide.setDefault(); + this.frontFace.setDefault(); + this.program.setDefault(); + this.activeTexture.setDefault(); + this.bindFramebuffer.setDefault(); + this.pixelStoreUnpack.setDefault(); + this.pixelStoreUnpackPremultiplyAlpha.setDefault(); + this.pixelStoreUnpackFlipY.setDefault(); + } + setDirty() { + this.clearColor.dirty = true; + this.clearDepth.dirty = true; + this.clearStencil.dirty = true; + this.colorMask.dirty = true; + this.depthMask.dirty = true; + this.stencilMask.dirty = true; + this.stencilFunc.dirty = true; + this.stencilOp.dirty = true; + this.stencilTest.dirty = true; + this.depthRange.dirty = true; + this.depthTest.dirty = true; + this.depthFunc.dirty = true; + this.blend.dirty = true; + this.blendFunc.dirty = true; + this.blendColor.dirty = true; + this.blendEquation.dirty = true; + this.cullFace.dirty = true; + this.cullFaceSide.dirty = true; + this.frontFace.dirty = true; + this.program.dirty = true; + this.activeTexture.dirty = true; + this.viewport.dirty = true; + this.bindFramebuffer.dirty = true; + this.bindRenderbuffer.dirty = true; + this.bindTexture.dirty = true; + this.bindVertexBuffer.dirty = true; + this.bindElementBuffer.dirty = true; + this.bindVertexArray.dirty = true; + this.pixelStoreUnpack.dirty = true; + this.pixelStoreUnpackPremultiplyAlpha.dirty = true; + this.pixelStoreUnpackFlipY.dirty = true; + } + createIndexBuffer(array, dynamicDraw) { + return new IndexBuffer(this, array, dynamicDraw); + } + createVertexBuffer(array, attributes, dynamicDraw) { + return new VertexBuffer(this, array, attributes, dynamicDraw); + } + createRenderbuffer(storageFormat, width, height) { + const gl = this.gl; + const rbo = gl.createRenderbuffer(); + this.bindRenderbuffer.set(rbo); + gl.renderbufferStorage(gl.RENDERBUFFER, storageFormat, width, height); + this.bindRenderbuffer.set(null); + return rbo; + } + createFramebuffer(width, height, hasDepth, hasStencil) { + return new Framebuffer(this, width, height, hasDepth, hasStencil); + } + clear({ color, depth, stencil }) { + const gl = this.gl; + let mask = 0; + if (color) { + mask |= gl.COLOR_BUFFER_BIT; + this.clearColor.set(color); + this.colorMask.set([true, true, true, true]); + } + if (typeof depth !== 'undefined') { + mask |= gl.DEPTH_BUFFER_BIT; + // Workaround for platforms where clearDepth doesn't seem to work + // without resetting the depthRange. See https://github.com/mapbox/mapbox-gl-js/issues/3437 + this.depthRange.set([0, 1]); + this.clearDepth.set(depth); + this.depthMask.set(true); + } + if (typeof stencil !== 'undefined') { + mask |= gl.STENCIL_BUFFER_BIT; + this.clearStencil.set(stencil); + this.stencilMask.set(0xFF); + } + gl.clear(mask); + } + setCullFace(cullFaceMode) { + if (cullFaceMode.enable === false) { + this.cullFace.set(false); + } + else { + this.cullFace.set(true); + this.cullFaceSide.set(cullFaceMode.mode); + this.frontFace.set(cullFaceMode.frontFace); + } + } + setDepthMode(depthMode) { + if (depthMode.func === this.gl.ALWAYS && !depthMode.mask) { + this.depthTest.set(false); + } + else { + this.depthTest.set(true); + this.depthFunc.set(depthMode.func); + this.depthMask.set(depthMode.mask); + this.depthRange.set(depthMode.range); + } + } + setStencilMode(stencilMode) { + if (stencilMode.test.func === this.gl.ALWAYS && !stencilMode.mask) { + this.stencilTest.set(false); + } + else { + this.stencilTest.set(true); + this.stencilMask.set(stencilMode.mask); + this.stencilOp.set([stencilMode.fail, stencilMode.depthFail, stencilMode.pass]); + this.stencilFunc.set({ + func: stencilMode.test.func, + ref: stencilMode.ref, + mask: stencilMode.test.mask + }); + } + } + setColorMode(colorMode) { + if (performance.deepEqual(colorMode.blendFunction, ColorMode.Replace)) { + this.blend.set(false); + } + else { + this.blend.set(true); + this.blendFunc.set(colorMode.blendFunction); + this.blendColor.set(colorMode.blendColor); + } + this.colorMask.set(colorMode.mask); + } + createVertexArray() { + var _a; + if (isWebGL2(this.gl)) + return this.gl.createVertexArray(); + return (_a = this.gl.getExtension('OES_vertex_array_object')) === null || _a === void 0 ? void 0 : _a.createVertexArrayOES(); + } + deleteVertexArray(x) { + var _a; + if (isWebGL2(this.gl)) + return this.gl.deleteVertexArray(x); + return (_a = this.gl.getExtension('OES_vertex_array_object')) === null || _a === void 0 ? void 0 : _a.deleteVertexArrayOES(x); + } + unbindVAO() { + // Unbinding the VAO prevents other things (custom layers, new buffer creation) from + // unintentionally changing the state of the last VAO used. + this.bindVertexArray.set(null); + } +} + +const ALWAYS$1 = 0x0207; +class DepthMode { + constructor(depthFunc, depthMask, depthRange) { + this.func = depthFunc; + this.mask = depthMask; + this.range = depthRange; + } +} +DepthMode.ReadOnly = false; +DepthMode.ReadWrite = true; +DepthMode.disabled = new DepthMode(ALWAYS$1, DepthMode.ReadOnly, [0, 1]); + +const ALWAYS = 0x0207; +const KEEP = 0x1E00; +class StencilMode { + constructor(test, ref, mask, fail, depthFail, pass) { + this.test = test; + this.ref = ref; + this.mask = mask; + this.fail = fail; + this.depthFail = depthFail; + this.pass = pass; + } +} +StencilMode.disabled = new StencilMode({ func: ALWAYS, mask: 0 }, 0, 0, KEEP, KEEP, KEEP); + +const BACK = 0x0405; +const CCW = 0x0901; +class CullFaceMode { + constructor(enable, mode, frontFace) { + this.enable = enable; + this.mode = mode; + this.frontFace = frontFace; + } +} +CullFaceMode.disabled = new CullFaceMode(false, BACK, CCW); +CullFaceMode.backCCW = new CullFaceMode(true, BACK, CCW); + +let quadTriangles; +function drawCollisionDebug(painter, sourceCache, layer, coords, translate, translateAnchor, isText) { + const context = painter.context; + const gl = context.gl; + const program = painter.useProgram('collisionBox'); + const tileBatches = []; + let circleCount = 0; + let circleOffset = 0; + for (let i = 0; i < coords.length; i++) { + const coord = coords[i]; + const tile = sourceCache.getTile(coord); + const bucket = tile.getBucket(layer); + if (!bucket) + continue; + let posMatrix = coord.posMatrix; + if (translate[0] !== 0 || translate[1] !== 0) { + posMatrix = painter.translatePosMatrix(coord.posMatrix, tile, translate, translateAnchor); + } + const buffers = isText ? bucket.textCollisionBox : bucket.iconCollisionBox; + // Get collision circle data of this bucket + const circleArray = bucket.collisionCircleArray; + if (circleArray.length > 0) { + // We need to know the projection matrix that was used for projecting collision circles to the screen. + // This might vary between buckets as the symbol placement is a continuous process. This matrix is + // required for transforming points from previous screen space to the current one + const invTransform = performance.create(); + const transform = posMatrix; + performance.mul(invTransform, bucket.placementInvProjMatrix, painter.transform.glCoordMatrix); + performance.mul(invTransform, invTransform, bucket.placementViewportMatrix); + tileBatches.push({ + circleArray, + circleOffset, + transform, + invTransform, + coord + }); + circleCount += circleArray.length / 4; // 4 values per circle + circleOffset = circleCount; + } + if (!buffers) + continue; + program.draw(context, gl.LINES, DepthMode.disabled, StencilMode.disabled, painter.colorModeForRenderPass(), CullFaceMode.disabled, collisionUniformValues(posMatrix, painter.transform, tile), painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord), layer.id, buffers.layoutVertexBuffer, buffers.indexBuffer, buffers.segments, null, painter.transform.zoom, null, null, buffers.collisionVertexBuffer); + } + if (!isText || !tileBatches.length) { + return; + } + // Render collision circles + const circleProgram = painter.useProgram('collisionCircle'); + // Construct vertex data + const vertexData = new performance.CollisionCircleLayoutArray(); + vertexData.resize(circleCount * 4); + vertexData._trim(); + let vertexOffset = 0; + for (const batch of tileBatches) { + for (let i = 0; i < batch.circleArray.length / 4; i++) { + const circleIdx = i * 4; + const x = batch.circleArray[circleIdx + 0]; + const y = batch.circleArray[circleIdx + 1]; + const radius = batch.circleArray[circleIdx + 2]; + const collision = batch.circleArray[circleIdx + 3]; + // 4 floats per vertex, 4 vertices per quad + vertexData.emplace(vertexOffset++, x, y, radius, collision, 0); + vertexData.emplace(vertexOffset++, x, y, radius, collision, 1); + vertexData.emplace(vertexOffset++, x, y, radius, collision, 2); + vertexData.emplace(vertexOffset++, x, y, radius, collision, 3); + } + } + if (!quadTriangles || quadTriangles.length < circleCount * 2) { + quadTriangles = createQuadTriangles(circleCount); + } + const indexBuffer = context.createIndexBuffer(quadTriangles, true); + const vertexBuffer = context.createVertexBuffer(vertexData, performance.collisionCircleLayout.members, true); + // Render batches + for (const batch of tileBatches) { + const uniforms = collisionCircleUniformValues(batch.transform, batch.invTransform, painter.transform); + circleProgram.draw(context, gl.TRIANGLES, DepthMode.disabled, StencilMode.disabled, painter.colorModeForRenderPass(), CullFaceMode.disabled, uniforms, painter.style.map.terrain && painter.style.map.terrain.getTerrainData(batch.coord), layer.id, vertexBuffer, indexBuffer, performance.SegmentVector.simpleSegment(0, batch.circleOffset * 2, batch.circleArray.length, batch.circleArray.length / 2), null, painter.transform.zoom, null, null, null); + } + vertexBuffer.destroy(); + indexBuffer.destroy(); +} +function createQuadTriangles(quadCount) { + const triCount = quadCount * 2; + const array = new performance.QuadTriangleArray(); + array.resize(triCount); + array._trim(); + // Two triangles and 4 vertices per quad. + for (let i = 0; i < triCount; i++) { + const idx = i * 6; + array.uint16[idx + 0] = i * 4 + 0; + array.uint16[idx + 1] = i * 4 + 1; + array.uint16[idx + 2] = i * 4 + 2; + array.uint16[idx + 3] = i * 4 + 2; + array.uint16[idx + 4] = i * 4 + 3; + array.uint16[idx + 5] = i * 4 + 0; + } + return array; +} + +const identityMat4 = performance.identity(new Float32Array(16)); +function drawSymbols(painter, sourceCache, layer, coords, variableOffsets) { + if (painter.renderPass !== 'translucent') + return; + // Disable the stencil test so that labels aren't clipped to tile boundaries. + const stencilMode = StencilMode.disabled; + const colorMode = painter.colorModeForRenderPass(); + const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset'); + //Compute variable-offsets before painting since icons and text data positioning + //depend on each other in this case. + if (hasVariablePlacement) { + updateVariableAnchors(coords, painter, layer, sourceCache, layer.layout.get('text-rotation-alignment'), layer.layout.get('text-pitch-alignment'), variableOffsets); + } + if (layer.paint.get('icon-opacity').constantOr(1) !== 0) { + drawLayerSymbols(painter, sourceCache, layer, coords, false, layer.paint.get('icon-translate'), layer.paint.get('icon-translate-anchor'), layer.layout.get('icon-rotation-alignment'), layer.layout.get('icon-pitch-alignment'), layer.layout.get('icon-keep-upright'), stencilMode, colorMode); + } + if (layer.paint.get('text-opacity').constantOr(1) !== 0) { + drawLayerSymbols(painter, sourceCache, layer, coords, true, layer.paint.get('text-translate'), layer.paint.get('text-translate-anchor'), layer.layout.get('text-rotation-alignment'), layer.layout.get('text-pitch-alignment'), layer.layout.get('text-keep-upright'), stencilMode, colorMode); + } + if (sourceCache.map.showCollisionBoxes) { + drawCollisionDebug(painter, sourceCache, layer, coords, layer.paint.get('text-translate'), layer.paint.get('text-translate-anchor'), true); + drawCollisionDebug(painter, sourceCache, layer, coords, layer.paint.get('icon-translate'), layer.paint.get('icon-translate-anchor'), false); + } +} +function calculateVariableRenderShift(anchor, width, height, textOffset, textBoxScale, renderTextSize) { + const { horizontalAlign, verticalAlign } = performance.getAnchorAlignment(anchor); + const shiftX = -(horizontalAlign - 0.5) * width; + const shiftY = -(verticalAlign - 0.5) * height; + return new performance.Point((shiftX / textBoxScale + textOffset[0]) * renderTextSize, (shiftY / textBoxScale + textOffset[1]) * renderTextSize); +} +function updateVariableAnchors(coords, painter, layer, sourceCache, rotationAlignment, pitchAlignment, variableOffsets) { + const tr = painter.transform; + const rotateWithMap = rotationAlignment === 'map'; + const pitchWithMap = pitchAlignment === 'map'; + for (const coord of coords) { + const tile = sourceCache.getTile(coord); + const bucket = tile.getBucket(layer); + if (!bucket || !bucket.text || !bucket.text.segments.get().length) + continue; + const sizeData = bucket.textSizeData; + const size = performance.evaluateSizeForZoom(sizeData, tr.zoom); + const pixelToTileScale = pixelsToTileUnits(tile, 1, painter.transform.zoom); + const labelPlaneMatrix = getLabelPlaneMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, pixelToTileScale); + const updateTextFitIcon = layer.layout.get('icon-text-fit') !== 'none' && bucket.hasIconData(); + if (size) { + const tileScale = Math.pow(2, tr.zoom - tile.tileID.overscaledZ); + const getElevation = painter.style.map.terrain ? (x, y) => painter.style.map.terrain.getElevation(coord, x, y) : null; + updateVariableAnchorsForBucket(bucket, rotateWithMap, pitchWithMap, variableOffsets, tr, labelPlaneMatrix, coord.posMatrix, tileScale, size, updateTextFitIcon, getElevation); + } + } +} +function updateVariableAnchorsForBucket(bucket, rotateWithMap, pitchWithMap, variableOffsets, transform, labelPlaneMatrix, posMatrix, tileScale, size, updateTextFitIcon, getElevation) { + const placedSymbols = bucket.text.placedSymbolArray; + const dynamicTextLayoutVertexArray = bucket.text.dynamicLayoutVertexArray; + const dynamicIconLayoutVertexArray = bucket.icon.dynamicLayoutVertexArray; + const placedTextShifts = {}; + dynamicTextLayoutVertexArray.clear(); + for (let s = 0; s < placedSymbols.length; s++) { + const symbol = placedSymbols.get(s); + const skipOrientation = bucket.allowVerticalPlacement && !symbol.placedOrientation; + const variableOffset = (!symbol.hidden && symbol.crossTileID && !skipOrientation) ? variableOffsets[symbol.crossTileID] : null; + if (!variableOffset) { + // These symbols are from a justification that is not being used, or a label that wasn't placed + // so we don't need to do the extra math to figure out what incremental shift to apply. + hideGlyphs(symbol.numGlyphs, dynamicTextLayoutVertexArray); + } + else { + const tileAnchor = new performance.Point(symbol.anchorX, symbol.anchorY); + const projectedAnchor = project(tileAnchor, pitchWithMap ? posMatrix : labelPlaneMatrix, getElevation); + const perspectiveRatio = getPerspectiveRatio(transform.cameraToCenterDistance, projectedAnchor.signedDistanceFromCamera); + let renderTextSize = performance.evaluateSizeForFeature(bucket.textSizeData, size, symbol) * perspectiveRatio / performance.ONE_EM; + if (pitchWithMap) { + // Go from size in pixels to equivalent size in tile units + renderTextSize *= bucket.tilePixelRatio / tileScale; + } + const { width, height, anchor, textOffset, textBoxScale } = variableOffset; + const shift = calculateVariableRenderShift(anchor, width, height, textOffset, textBoxScale, renderTextSize); + // Usual case is that we take the projected anchor and add the pixel-based shift + // calculated above. In the (somewhat weird) case of pitch-aligned text, we add an equivalent + // tile-unit based shift to the anchor before projecting to the label plane. + const shiftedAnchor = pitchWithMap ? + project(tileAnchor.add(shift), labelPlaneMatrix, getElevation).point : + projectedAnchor.point.add(rotateWithMap ? + shift.rotate(-transform.angle) : + shift); + const angle = (bucket.allowVerticalPlacement && symbol.placedOrientation === performance.WritingMode.vertical) ? Math.PI / 2 : 0; + for (let g = 0; g < symbol.numGlyphs; g++) { + performance.addDynamicAttributes(dynamicTextLayoutVertexArray, shiftedAnchor, angle); + } + //Only offset horizontal text icons + if (updateTextFitIcon && symbol.associatedIconIndex >= 0) { + placedTextShifts[symbol.associatedIconIndex] = { shiftedAnchor, angle }; + } + } + } + if (updateTextFitIcon) { + dynamicIconLayoutVertexArray.clear(); + const placedIcons = bucket.icon.placedSymbolArray; + for (let i = 0; i < placedIcons.length; i++) { + const placedIcon = placedIcons.get(i); + if (placedIcon.hidden) { + hideGlyphs(placedIcon.numGlyphs, dynamicIconLayoutVertexArray); + } + else { + const shift = placedTextShifts[i]; + if (!shift) { + hideGlyphs(placedIcon.numGlyphs, dynamicIconLayoutVertexArray); + } + else { + for (let g = 0; g < placedIcon.numGlyphs; g++) { + performance.addDynamicAttributes(dynamicIconLayoutVertexArray, shift.shiftedAnchor, shift.angle); + } + } + } + } + bucket.icon.dynamicLayoutVertexBuffer.updateData(dynamicIconLayoutVertexArray); + } + bucket.text.dynamicLayoutVertexBuffer.updateData(dynamicTextLayoutVertexArray); +} +function getSymbolProgramName(isSDF, isText, bucket) { + if (bucket.iconsInText && isText) { + return 'symbolTextAndIcon'; + } + else if (isSDF) { + return 'symbolSDF'; + } + else { + return 'symbolIcon'; + } +} +function drawLayerSymbols(painter, sourceCache, layer, coords, isText, translate, translateAnchor, rotationAlignment, pitchAlignment, keepUpright, stencilMode, colorMode) { + const context = painter.context; + const gl = context.gl; + const tr = painter.transform; + const rotateWithMap = rotationAlignment === 'map'; + const pitchWithMap = pitchAlignment === 'map'; + const alongLine = rotationAlignment !== 'viewport' && layer.layout.get('symbol-placement') !== 'point'; + // Line label rotation happens in `updateLineLabels` + // Pitched point labels are automatically rotated by the labelPlaneMatrix projection + // Unpitched point labels need to have their rotation applied after projection + const rotateInShader = rotateWithMap && !pitchWithMap && !alongLine; + const hasSortKey = !layer.layout.get('symbol-sort-key').isConstant(); + let sortFeaturesByKey = false; + const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly); + const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset'); + const tileRenderState = []; + for (const coord of coords) { + const tile = sourceCache.getTile(coord); + const bucket = tile.getBucket(layer); + if (!bucket) + continue; + const buffers = isText ? bucket.text : bucket.icon; + if (!buffers || !buffers.segments.get().length || !buffers.hasVisibleVertices) + continue; + const programConfiguration = buffers.programConfigurations.get(layer.id); + const isSDF = isText || bucket.sdfIcons; + const sizeData = isText ? bucket.textSizeData : bucket.iconSizeData; + const transformed = pitchWithMap || tr.pitch !== 0; + const program = painter.useProgram(getSymbolProgramName(isSDF, isText, bucket), programConfiguration); + const size = performance.evaluateSizeForZoom(sizeData, tr.zoom); + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + let texSize; + let texSizeIcon = [0, 0]; + let atlasTexture; + let atlasInterpolation; + let atlasTextureIcon = null; + let atlasInterpolationIcon; + if (isText) { + atlasTexture = tile.glyphAtlasTexture; + atlasInterpolation = gl.LINEAR; + texSize = tile.glyphAtlasTexture.size; + if (bucket.iconsInText) { + texSizeIcon = tile.imageAtlasTexture.size; + atlasTextureIcon = tile.imageAtlasTexture; + const zoomDependentSize = sizeData.kind === 'composite' || sizeData.kind === 'camera'; + atlasInterpolationIcon = transformed || painter.options.rotating || painter.options.zooming || zoomDependentSize ? gl.LINEAR : gl.NEAREST; + } + } + else { + const iconScaled = layer.layout.get('icon-size').constantOr(0) !== 1 || bucket.iconsNeedLinear; + atlasTexture = tile.imageAtlasTexture; + atlasInterpolation = isSDF || painter.options.rotating || painter.options.zooming || iconScaled || transformed ? + gl.LINEAR : + gl.NEAREST; + texSize = tile.imageAtlasTexture.size; + } + const s = pixelsToTileUnits(tile, 1, painter.transform.zoom); + const labelPlaneMatrix = getLabelPlaneMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, s); + const glCoordMatrix = getGlCoordMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, s); + const hasVariableAnchors = hasVariablePlacement && bucket.hasTextData(); + const updateTextFitIcon = layer.layout.get('icon-text-fit') !== 'none' && + hasVariableAnchors && + bucket.hasIconData(); + if (alongLine) { + const getElevation = painter.style.map.terrain ? (x, y) => painter.style.map.terrain.getElevation(coord, x, y) : null; + const rotateToLine = layer.layout.get('text-rotation-alignment') === 'map'; + updateLineLabels(bucket, coord.posMatrix, painter, isText, labelPlaneMatrix, glCoordMatrix, pitchWithMap, keepUpright, rotateToLine, getElevation); + } + const matrix = painter.translatePosMatrix(coord.posMatrix, tile, translate, translateAnchor), uLabelPlaneMatrix = (alongLine || (isText && hasVariablePlacement) || updateTextFitIcon) ? identityMat4 : labelPlaneMatrix, uglCoordMatrix = painter.translatePosMatrix(glCoordMatrix, tile, translate, translateAnchor, true); + const hasHalo = isSDF && layer.paint.get(isText ? 'text-halo-width' : 'icon-halo-width').constantOr(1) !== 0; + let uniformValues; + if (isSDF) { + if (!bucket.iconsInText) { + uniformValues = symbolSDFUniformValues(sizeData.kind, size, rotateInShader, pitchWithMap, painter, matrix, uLabelPlaneMatrix, uglCoordMatrix, isText, texSize, true); + } + else { + uniformValues = symbolTextAndIconUniformValues(sizeData.kind, size, rotateInShader, pitchWithMap, painter, matrix, uLabelPlaneMatrix, uglCoordMatrix, texSize, texSizeIcon); + } + } + else { + uniformValues = symbolIconUniformValues(sizeData.kind, size, rotateInShader, pitchWithMap, painter, matrix, uLabelPlaneMatrix, uglCoordMatrix, isText, texSize); + } + const state = { + program, + buffers, + uniformValues, + atlasTexture, + atlasTextureIcon, + atlasInterpolation, + atlasInterpolationIcon, + isSDF, + hasHalo + }; + if (hasSortKey && bucket.canOverlap) { + sortFeaturesByKey = true; + const oldSegments = buffers.segments.get(); + for (const segment of oldSegments) { + tileRenderState.push({ + segments: new performance.SegmentVector([segment]), + sortKey: segment.sortKey, + state, + terrainData + }); + } + } + else { + tileRenderState.push({ + segments: buffers.segments, + sortKey: 0, + state, + terrainData + }); + } + } + if (sortFeaturesByKey) { + tileRenderState.sort((a, b) => a.sortKey - b.sortKey); + } + for (const segmentState of tileRenderState) { + const state = segmentState.state; + context.activeTexture.set(gl.TEXTURE0); + // @ts-ignore + state.atlasTexture.bind(state.atlasInterpolation, gl.CLAMP_TO_EDGE); + if (state.atlasTextureIcon) { + context.activeTexture.set(gl.TEXTURE1); + if (state.atlasTextureIcon) { + // @ts-ignore + state.atlasTextureIcon.bind(state.atlasInterpolationIcon, gl.CLAMP_TO_EDGE); + } + } + if (state.isSDF) { + const uniformValues = state.uniformValues; + if (state.hasHalo) { + uniformValues['u_is_halo'] = 1; + drawSymbolElements(state.buffers, segmentState.segments, layer, painter, state.program, depthMode, stencilMode, colorMode, uniformValues, segmentState.terrainData); + } + uniformValues['u_is_halo'] = 0; + } + drawSymbolElements(state.buffers, segmentState.segments, layer, painter, state.program, depthMode, stencilMode, colorMode, state.uniformValues, segmentState.terrainData); + } +} +function drawSymbolElements(buffers, segments, layer, painter, program, depthMode, stencilMode, colorMode, uniformValues, terrainData) { + const context = painter.context; + const gl = context.gl; + program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled, uniformValues, terrainData, layer.id, buffers.layoutVertexBuffer, buffers.indexBuffer, segments, layer.paint, painter.transform.zoom, buffers.programConfigurations.get(layer.id), buffers.dynamicLayoutVertexBuffer, buffers.opacityVertexBuffer); +} + +function drawCircles(painter, sourceCache, layer, coords) { + if (painter.renderPass !== 'translucent') + return; + const opacity = layer.paint.get('circle-opacity'); + const strokeWidth = layer.paint.get('circle-stroke-width'); + const strokeOpacity = layer.paint.get('circle-stroke-opacity'); + const sortFeaturesByKey = !layer.layout.get('circle-sort-key').isConstant(); + if (opacity.constantOr(1) === 0 && (strokeWidth.constantOr(1) === 0 || strokeOpacity.constantOr(1) === 0)) { + return; + } + const context = painter.context; + const gl = context.gl; + const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly); + // Turn off stencil testing to allow circles to be drawn across boundaries, + // so that large circles are not clipped to tiles + const stencilMode = StencilMode.disabled; + const colorMode = painter.colorModeForRenderPass(); + const segmentsRenderStates = []; + for (let i = 0; i < coords.length; i++) { + const coord = coords[i]; + const tile = sourceCache.getTile(coord); + const bucket = tile.getBucket(layer); + if (!bucket) + continue; + const programConfiguration = bucket.programConfigurations.get(layer.id); + const program = painter.useProgram('circle', programConfiguration); + const layoutVertexBuffer = bucket.layoutVertexBuffer; + const indexBuffer = bucket.indexBuffer; + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + const uniformValues = circleUniformValues(painter, coord, tile, layer); + const state = { + programConfiguration, + program, + layoutVertexBuffer, + indexBuffer, + uniformValues, + terrainData + }; + if (sortFeaturesByKey) { + const oldSegments = bucket.segments.get(); + for (const segment of oldSegments) { + segmentsRenderStates.push({ + segments: new performance.SegmentVector([segment]), + sortKey: segment.sortKey, + state + }); + } + } + else { + segmentsRenderStates.push({ + segments: bucket.segments, + sortKey: 0, + state + }); + } + } + if (sortFeaturesByKey) { + segmentsRenderStates.sort((a, b) => a.sortKey - b.sortKey); + } + for (const segmentsState of segmentsRenderStates) { + const { programConfiguration, program, layoutVertexBuffer, indexBuffer, uniformValues, terrainData } = segmentsState.state; + const segments = segmentsState.segments; + program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled, uniformValues, terrainData, layer.id, layoutVertexBuffer, indexBuffer, segments, layer.paint, painter.transform.zoom, programConfiguration); + } +} + +function drawHeatmap(painter, sourceCache, layer, coords) { + if (layer.paint.get('heatmap-opacity') === 0) { + return; + } + if (painter.renderPass === 'offscreen') { + const context = painter.context; + const gl = context.gl; + // Allow kernels to be drawn across boundaries, so that + // large kernels are not clipped to tiles + const stencilMode = StencilMode.disabled; + // Turn on additive blending for kernels, which is a key aspect of kernel density estimation formula + const colorMode = new ColorMode([gl.ONE, gl.ONE], performance.Color.transparent, [true, true, true, true]); + bindFramebuffer(context, painter, layer); + context.clear({ color: performance.Color.transparent }); + for (let i = 0; i < coords.length; i++) { + const coord = coords[i]; + // Skip tiles that have uncovered parents to avoid flickering; we don't need + // to use complex tile masking here because the change between zoom levels is subtle, + // so it's fine to simply render the parent until all its 4 children are loaded + if (sourceCache.hasRenderableParent(coord)) + continue; + const tile = sourceCache.getTile(coord); + const bucket = tile.getBucket(layer); + if (!bucket) + continue; + const programConfiguration = bucket.programConfigurations.get(layer.id); + const program = painter.useProgram('heatmap', programConfiguration); + const { zoom } = painter.transform; + program.draw(context, gl.TRIANGLES, DepthMode.disabled, stencilMode, colorMode, CullFaceMode.disabled, heatmapUniformValues(coord.posMatrix, tile, zoom, layer.paint.get('heatmap-intensity')), null, layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer, bucket.segments, layer.paint, painter.transform.zoom, programConfiguration); + } + context.viewport.set([0, 0, painter.width, painter.height]); + } + else if (painter.renderPass === 'translucent') { + painter.context.setColorMode(painter.colorModeForRenderPass()); + renderTextureToMap(painter, layer); + } +} +function bindFramebuffer(context, painter, layer) { + const gl = context.gl; + context.activeTexture.set(gl.TEXTURE1); + // Use a 4x downscaled screen texture for better performance + context.viewport.set([0, 0, painter.width / 4, painter.height / 4]); + let fbo = layer.heatmapFbo; + if (!fbo) { + const texture = gl.createTexture(); + gl.bindTexture(gl.TEXTURE_2D, texture); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR); + fbo = layer.heatmapFbo = context.createFramebuffer(painter.width / 4, painter.height / 4, false, false); + bindTextureToFramebuffer(context, painter, texture, fbo); + } + else { + gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get()); + context.bindFramebuffer.set(fbo.framebuffer); + } +} +function bindTextureToFramebuffer(context, painter, texture, fbo) { + var _a, _b; + const gl = context.gl; + // Use the higher precision half-float texture where available (producing much smoother looking heatmaps); + // Otherwise, fall back to a low precision texture + const numType = (_a = context.HALF_FLOAT) !== null && _a !== void 0 ? _a : gl.UNSIGNED_BYTE; + const internalFormat = (_b = context.RGBA16F) !== null && _b !== void 0 ? _b : gl.RGBA; + gl.texImage2D(gl.TEXTURE_2D, 0, internalFormat, painter.width / 4, painter.height / 4, 0, gl.RGBA, numType, null); + fbo.colorAttachment.set(texture); +} +function renderTextureToMap(painter, layer) { + const context = painter.context; + const gl = context.gl; + // Here we bind two different textures from which we'll sample in drawing + // heatmaps: the kernel texture, prepared in the offscreen pass, and a + // color ramp texture. + const fbo = layer.heatmapFbo; + if (!fbo) + return; + context.activeTexture.set(gl.TEXTURE0); + gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get()); + context.activeTexture.set(gl.TEXTURE1); + let colorRampTexture = layer.colorRampTexture; + if (!colorRampTexture) { + colorRampTexture = layer.colorRampTexture = new Texture(context, layer.colorRamp, gl.RGBA); + } + colorRampTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + painter.useProgram('heatmapTexture').draw(context, gl.TRIANGLES, DepthMode.disabled, StencilMode.disabled, painter.colorModeForRenderPass(), CullFaceMode.disabled, heatmapTextureUniformValues(painter, layer, 0, 1), null, layer.id, painter.viewportBuffer, painter.quadTriangleIndexBuffer, painter.viewportSegments, layer.paint, painter.transform.zoom); +} + +function drawLine(painter, sourceCache, layer, coords) { + if (painter.renderPass !== 'translucent') + return; + const opacity = layer.paint.get('line-opacity'); + const width = layer.paint.get('line-width'); + if (opacity.constantOr(1) === 0 || width.constantOr(1) === 0) + return; + const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly); + const colorMode = painter.colorModeForRenderPass(); + const dasharray = layer.paint.get('line-dasharray'); + const patternProperty = layer.paint.get('line-pattern'); + const image = patternProperty.constantOr(1); + const gradient = layer.paint.get('line-gradient'); + const crossfade = layer.getCrossfadeParameters(); + const programId = image ? 'linePattern' : + dasharray ? 'lineSDF' : + gradient ? 'lineGradient' : 'line'; + const context = painter.context; + const gl = context.gl; + let firstTile = true; + for (const coord of coords) { + const tile = sourceCache.getTile(coord); + if (image && !tile.patternsLoaded()) + continue; + const bucket = tile.getBucket(layer); + if (!bucket) + continue; + const programConfiguration = bucket.programConfigurations.get(layer.id); + const prevProgram = painter.context.program.get(); + const program = painter.useProgram(programId, programConfiguration); + const programChanged = firstTile || program.program !== prevProgram; + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + const constantPattern = patternProperty.constantOr(null); + if (constantPattern && tile.imageAtlas) { + const atlas = tile.imageAtlas; + const posTo = atlas.patternPositions[constantPattern.to.toString()]; + const posFrom = atlas.patternPositions[constantPattern.from.toString()]; + if (posTo && posFrom) + programConfiguration.setConstantPatternPositions(posTo, posFrom); + } + const terrainCoord = terrainData ? coord : null; + const uniformValues = image ? linePatternUniformValues(painter, tile, layer, crossfade, terrainCoord) : + dasharray ? lineSDFUniformValues(painter, tile, layer, dasharray, crossfade, terrainCoord) : + gradient ? lineGradientUniformValues(painter, tile, layer, bucket.lineClipsArray.length, terrainCoord) : + lineUniformValues(painter, tile, layer, terrainCoord); + if (image) { + context.activeTexture.set(gl.TEXTURE0); + tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + programConfiguration.updatePaintBuffers(crossfade); + } + else if (dasharray && (programChanged || painter.lineAtlas.dirty)) { + context.activeTexture.set(gl.TEXTURE0); + painter.lineAtlas.bind(context); + } + else if (gradient) { + const layerGradient = bucket.gradients[layer.id]; + let gradientTexture = layerGradient.texture; + if (layer.gradientVersion !== layerGradient.version) { + let textureResolution = 256; + if (layer.stepInterpolant) { + const sourceMaxZoom = sourceCache.getSource().maxzoom; + const potentialOverzoom = coord.canonical.z === sourceMaxZoom ? + Math.ceil(1 << (painter.transform.maxZoom - coord.canonical.z)) : 1; + const lineLength = bucket.maxLineLength / performance.EXTENT; + // Logical pixel tile size is 512px, and 1024px right before current zoom + 1 + const maxTilePixelSize = 1024; + // Maximum possible texture coverage heuristic, bound by hardware max texture size + const maxTextureCoverage = lineLength * maxTilePixelSize * potentialOverzoom; + textureResolution = performance.clamp(performance.nextPowerOfTwo(maxTextureCoverage), 256, context.maxTextureSize); + } + layerGradient.gradient = performance.renderColorRamp({ + expression: layer.gradientExpression(), + evaluationKey: 'lineProgress', + resolution: textureResolution, + image: layerGradient.gradient || undefined, + clips: bucket.lineClipsArray + }); + if (layerGradient.texture) { + layerGradient.texture.update(layerGradient.gradient); + } + else { + layerGradient.texture = new Texture(context, layerGradient.gradient, gl.RGBA); + } + layerGradient.version = layer.gradientVersion; + gradientTexture = layerGradient.texture; + } + context.activeTexture.set(gl.TEXTURE0); + gradientTexture.bind(layer.stepInterpolant ? gl.NEAREST : gl.LINEAR, gl.CLAMP_TO_EDGE); + } + program.draw(context, gl.TRIANGLES, depthMode, painter.stencilModeForClipping(coord), colorMode, CullFaceMode.disabled, uniformValues, terrainData, layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer, bucket.segments, layer.paint, painter.transform.zoom, programConfiguration, bucket.layoutVertexBuffer2); + firstTile = false; + // once refactored so that bound texture state is managed, we'll also be able to remove this firstTile/programChanged logic + } +} + +/** + * A simple helper shared by draw_fill and draw_fill_extrusions to find the correct pattern positions AND update program. + * For transtionable properties, especially 'fill-pattern' and 'fill-extrusion-pattern', while rendering certain frames + * tile.imageAtlas has been updated by worker to hold the new pattern only, but rendering code is still looking for the previous image. + * The mismatch was causing setConstantPatternPositions method not being called and pixelRatio was always the + * default of 1, instead of actual values set by original map.addImage. + * + * @param programConfiguration - to be used to set pattern position and device pixel ratio. + * @param propertyName - 'fill-pattern' or 'fill-extrusion-pattern' property key + * @param constantPattern - either 'fill-pattern' or 'fill-extrusion-pattern' property value + * @param tile - current tile being drawn + * @param layer - current layer being rendered + */ +function updatePatternPositionsInProgram(programConfiguration, propertyName, constantPattern, tile, layer) { + if (!constantPattern || !tile || !tile.imageAtlas) { + return; + } + const patternPositions = tile.imageAtlas.patternPositions; + let posTo = patternPositions[constantPattern.to.toString()]; + let posFrom = patternPositions[constantPattern.from.toString()]; + // https://github.com/maplibre/maplibre-gl-js/issues/3377 + if (!posTo && posFrom) + posTo = posFrom; + if (!posFrom && posTo) + posFrom = posTo; + // try again in case patternPositions has been updated by worker + if (!posTo || !posFrom) { + const transitioned = layer.getPaintProperty(propertyName); + posTo = patternPositions[transitioned]; + posFrom = patternPositions[transitioned]; + } + if (posTo && posFrom) { + programConfiguration.setConstantPatternPositions(posTo, posFrom); + } +} + +function drawFill(painter, sourceCache, layer, coords) { + const color = layer.paint.get('fill-color'); + const opacity = layer.paint.get('fill-opacity'); + if (opacity.constantOr(1) === 0) { + return; + } + const colorMode = painter.colorModeForRenderPass(); + const pattern = layer.paint.get('fill-pattern'); + const pass = painter.opaquePassEnabledForLayer() && + (!pattern.constantOr(1) && + color.constantOr(performance.Color.transparent).a === 1 && + opacity.constantOr(0) === 1) ? 'opaque' : 'translucent'; + // Draw fill + if (painter.renderPass === pass) { + const depthMode = painter.depthModeForSublayer(1, painter.renderPass === 'opaque' ? DepthMode.ReadWrite : DepthMode.ReadOnly); + drawFillTiles(painter, sourceCache, layer, coords, depthMode, colorMode, false); + } + // Draw stroke + if (painter.renderPass === 'translucent' && layer.paint.get('fill-antialias')) { + // If we defined a different color for the fill outline, we are + // going to ignore the bits in 0x07 and just care about the global + // clipping mask. + // Otherwise, we only want to drawFill the antialiased parts that are + // *outside* the current shape. This is important in case the fill + // or stroke color is translucent. If we wouldn't clip to outside + // the current shape, some pixels from the outline stroke overlapped + // the (non-antialiased) fill. + const depthMode = painter.depthModeForSublayer(layer.getPaintProperty('fill-outline-color') ? 2 : 0, DepthMode.ReadOnly); + drawFillTiles(painter, sourceCache, layer, coords, depthMode, colorMode, true); + } +} +function drawFillTiles(painter, sourceCache, layer, coords, depthMode, colorMode, isOutline) { + const gl = painter.context.gl; + const fillPropertyName = 'fill-pattern'; + const patternProperty = layer.paint.get(fillPropertyName); + const image = patternProperty && patternProperty.constantOr(1); + const crossfade = layer.getCrossfadeParameters(); + let drawMode, programName, uniformValues, indexBuffer, segments; + if (!isOutline) { + programName = image ? 'fillPattern' : 'fill'; + drawMode = gl.TRIANGLES; + } + else { + programName = image && !layer.getPaintProperty('fill-outline-color') ? 'fillOutlinePattern' : 'fillOutline'; + drawMode = gl.LINES; + } + const constantPattern = patternProperty.constantOr(null); + for (const coord of coords) { + const tile = sourceCache.getTile(coord); + if (image && !tile.patternsLoaded()) + continue; + const bucket = tile.getBucket(layer); + if (!bucket) + continue; + const programConfiguration = bucket.programConfigurations.get(layer.id); + const program = painter.useProgram(programName, programConfiguration); + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + if (image) { + painter.context.activeTexture.set(gl.TEXTURE0); + tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + programConfiguration.updatePaintBuffers(crossfade); + } + updatePatternPositionsInProgram(programConfiguration, fillPropertyName, constantPattern, tile, layer); + const terrainCoord = terrainData ? coord : null; + const posMatrix = terrainCoord ? terrainCoord.posMatrix : coord.posMatrix; + const tileMatrix = painter.translatePosMatrix(posMatrix, tile, layer.paint.get('fill-translate'), layer.paint.get('fill-translate-anchor')); + if (!isOutline) { + indexBuffer = bucket.indexBuffer; + segments = bucket.segments; + uniformValues = image ? + fillPatternUniformValues(tileMatrix, painter, crossfade, tile) : + fillUniformValues(tileMatrix); + } + else { + indexBuffer = bucket.indexBuffer2; + segments = bucket.segments2; + const drawingBufferSize = [gl.drawingBufferWidth, gl.drawingBufferHeight]; + uniformValues = (programName === 'fillOutlinePattern' && image) ? + fillOutlinePatternUniformValues(tileMatrix, painter, crossfade, tile, drawingBufferSize) : + fillOutlineUniformValues(tileMatrix, drawingBufferSize); + } + program.draw(painter.context, drawMode, depthMode, painter.stencilModeForClipping(coord), colorMode, CullFaceMode.disabled, uniformValues, terrainData, layer.id, bucket.layoutVertexBuffer, indexBuffer, segments, layer.paint, painter.transform.zoom, programConfiguration); + } +} + +function drawFillExtrusion(painter, source, layer, coords) { + const opacity = layer.paint.get('fill-extrusion-opacity'); + if (opacity === 0) { + return; + } + if (painter.renderPass === 'translucent') { + const depthMode = new DepthMode(painter.context.gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D); + if (opacity === 1 && !layer.paint.get('fill-extrusion-pattern').constantOr(1)) { + const colorMode = painter.colorModeForRenderPass(); + drawExtrusionTiles(painter, source, layer, coords, depthMode, StencilMode.disabled, colorMode); + } + else { + // Draw transparent buildings in two passes so that only the closest surface is drawn. + // First draw all the extrusions into only the depth buffer. No colors are drawn. + drawExtrusionTiles(painter, source, layer, coords, depthMode, StencilMode.disabled, ColorMode.disabled); + // Then draw all the extrusions a second type, only coloring fragments if they have the + // same depth value as the closest fragment in the previous pass. Use the stencil buffer + // to prevent the second draw in cases where we have coincident polygons. + drawExtrusionTiles(painter, source, layer, coords, depthMode, painter.stencilModeFor3D(), painter.colorModeForRenderPass()); + } + } +} +function drawExtrusionTiles(painter, source, layer, coords, depthMode, stencilMode, colorMode) { + const context = painter.context; + const gl = context.gl; + const fillPropertyName = 'fill-extrusion-pattern'; + const patternProperty = layer.paint.get(fillPropertyName); + const image = patternProperty.constantOr(1); + const crossfade = layer.getCrossfadeParameters(); + const opacity = layer.paint.get('fill-extrusion-opacity'); + const constantPattern = patternProperty.constantOr(null); + for (const coord of coords) { + const tile = source.getTile(coord); + const bucket = tile.getBucket(layer); + if (!bucket) + continue; + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + const programConfiguration = bucket.programConfigurations.get(layer.id); + const program = painter.useProgram(image ? 'fillExtrusionPattern' : 'fillExtrusion', programConfiguration); + if (image) { + painter.context.activeTexture.set(gl.TEXTURE0); + tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + programConfiguration.updatePaintBuffers(crossfade); + } + updatePatternPositionsInProgram(programConfiguration, fillPropertyName, constantPattern, tile, layer); + const matrix = painter.translatePosMatrix(coord.posMatrix, tile, layer.paint.get('fill-extrusion-translate'), layer.paint.get('fill-extrusion-translate-anchor')); + const shouldUseVerticalGradient = layer.paint.get('fill-extrusion-vertical-gradient'); + const uniformValues = image ? + fillExtrusionPatternUniformValues(matrix, painter, shouldUseVerticalGradient, opacity, coord, crossfade, tile) : + fillExtrusionUniformValues(matrix, painter, shouldUseVerticalGradient, opacity); + program.draw(context, context.gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer, bucket.segments, layer.paint, painter.transform.zoom, programConfiguration, painter.style.map.terrain && bucket.centroidVertexBuffer); + } +} + +function drawHillshade(painter, sourceCache, layer, tileIDs) { + if (painter.renderPass !== 'offscreen' && painter.renderPass !== 'translucent') + return; + const context = painter.context; + const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly); + const colorMode = painter.colorModeForRenderPass(); + const [stencilModes, coords] = painter.renderPass === 'translucent' ? + painter.stencilConfigForOverlap(tileIDs) : [{}, tileIDs]; + for (const coord of coords) { + const tile = sourceCache.getTile(coord); + if (typeof tile.needsHillshadePrepare !== 'undefined' && tile.needsHillshadePrepare && painter.renderPass === 'offscreen') { + prepareHillshade(painter, tile, layer, depthMode, StencilMode.disabled, colorMode); + } + else if (painter.renderPass === 'translucent') { + renderHillshade(painter, coord, tile, layer, depthMode, stencilModes[coord.overscaledZ], colorMode); + } + } + context.viewport.set([0, 0, painter.width, painter.height]); +} +function renderHillshade(painter, coord, tile, layer, depthMode, stencilMode, colorMode) { + const context = painter.context; + const gl = context.gl; + const fbo = tile.fbo; + if (!fbo) + return; + const program = painter.useProgram('hillshade'); + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + context.activeTexture.set(gl.TEXTURE0); + gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get()); + const terrainCoord = terrainData ? coord : null; + program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled, hillshadeUniformValues(painter, tile, layer, terrainCoord), terrainData, layer.id, painter.rasterBoundsBuffer, painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments); +} +// hillshade rendering is done in two steps. the prepare step first calculates the slope of the terrain in the x and y +// directions for each pixel, and saves those values to a framebuffer texture in the r and g channels. +function prepareHillshade(painter, tile, layer, depthMode, stencilMode, colorMode) { + const context = painter.context; + const gl = context.gl; + const dem = tile.dem; + if (dem && dem.data) { + const tileSize = dem.dim; + const textureStride = dem.stride; + const pixelData = dem.getPixels(); + context.activeTexture.set(gl.TEXTURE1); + context.pixelStoreUnpackPremultiplyAlpha.set(false); + tile.demTexture = tile.demTexture || painter.getTileTexture(textureStride); + if (tile.demTexture) { + const demTexture = tile.demTexture; + demTexture.update(pixelData, { premultiply: false }); + demTexture.bind(gl.NEAREST, gl.CLAMP_TO_EDGE); + } + else { + tile.demTexture = new Texture(context, pixelData, gl.RGBA, { premultiply: false }); + tile.demTexture.bind(gl.NEAREST, gl.CLAMP_TO_EDGE); + } + context.activeTexture.set(gl.TEXTURE0); + let fbo = tile.fbo; + if (!fbo) { + const renderTexture = new Texture(context, { width: tileSize, height: tileSize, data: null }, gl.RGBA); + renderTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + fbo = tile.fbo = context.createFramebuffer(tileSize, tileSize, true, false); + fbo.colorAttachment.set(renderTexture.texture); + } + context.bindFramebuffer.set(fbo.framebuffer); + context.viewport.set([0, 0, tileSize, tileSize]); + painter.useProgram('hillshadePrepare').draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled, hillshadeUniformPrepareValues(tile.tileID, dem), null, layer.id, painter.rasterBoundsBuffer, painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments); + tile.needsHillshadePrepare = false; + } +} + +function drawRaster(painter, sourceCache, layer, tileIDs) { + if (painter.renderPass !== 'translucent') + return; + if (layer.paint.get('raster-opacity') === 0) + return; + if (!tileIDs.length) + return; + const context = painter.context; + const gl = context.gl; + const source = sourceCache.getSource(); + const program = painter.useProgram('raster'); + const colorMode = painter.colorModeForRenderPass(); + const [stencilModes, coords] = source instanceof ImageSource ? [{}, tileIDs] : + painter.stencilConfigForOverlap(tileIDs); + const minTileZ = coords[coords.length - 1].overscaledZ; + const align = !painter.options.moving; + for (const coord of coords) { + // Set the lower zoom level to sublayer 0, and higher zoom levels to higher sublayers + // Use gl.LESS to prevent double drawing in areas where tiles overlap. + const depthMode = painter.depthModeForSublayer(coord.overscaledZ - minTileZ, layer.paint.get('raster-opacity') === 1 ? DepthMode.ReadWrite : DepthMode.ReadOnly, gl.LESS); + const tile = sourceCache.getTile(coord); + tile.registerFadeDuration(layer.paint.get('raster-fade-duration')); + const parentTile = sourceCache.findLoadedParent(coord, 0), fade = getFadeValues(tile, parentTile, sourceCache, layer, painter.transform, painter.style.map.terrain); + let parentScaleBy, parentTL; + const textureFilter = layer.paint.get('raster-resampling') === 'nearest' ? gl.NEAREST : gl.LINEAR; + context.activeTexture.set(gl.TEXTURE0); + tile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST); + context.activeTexture.set(gl.TEXTURE1); + if (parentTile) { + parentTile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST); + parentScaleBy = Math.pow(2, parentTile.tileID.overscaledZ - tile.tileID.overscaledZ); + parentTL = [tile.tileID.canonical.x * parentScaleBy % 1, tile.tileID.canonical.y * parentScaleBy % 1]; + } + else { + tile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST); + } + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + const terrainCoord = terrainData ? coord : null; + const posMatrix = terrainCoord ? terrainCoord.posMatrix : painter.transform.calculatePosMatrix(coord.toUnwrapped(), align); + const uniformValues = rasterUniformValues(posMatrix, parentTL || [0, 0], parentScaleBy || 1, fade, layer); + if (source instanceof ImageSource) { + program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.disabled, uniformValues, terrainData, layer.id, source.boundsBuffer, painter.quadTriangleIndexBuffer, source.boundsSegments); + } + else { + program.draw(context, gl.TRIANGLES, depthMode, stencilModes[coord.overscaledZ], colorMode, CullFaceMode.disabled, uniformValues, terrainData, layer.id, painter.rasterBoundsBuffer, painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments); + } + } +} +function getFadeValues(tile, parentTile, sourceCache, layer, transform, terrain) { + const fadeDuration = layer.paint.get('raster-fade-duration'); + if (!terrain && fadeDuration > 0) { + const now = performance.browser.now(); + const sinceTile = (now - tile.timeAdded) / fadeDuration; + const sinceParent = parentTile ? (now - parentTile.timeAdded) / fadeDuration : -1; + const source = sourceCache.getSource(); + const idealZ = transform.coveringZoomLevel({ + tileSize: source.tileSize, + roundZoom: source.roundZoom + }); + // if no parent or parent is older, fade in; if parent is younger, fade out + const fadeIn = !parentTile || Math.abs(parentTile.tileID.overscaledZ - idealZ) > Math.abs(tile.tileID.overscaledZ - idealZ); + const childOpacity = (fadeIn && tile.refreshedUponExpiration) ? 1 : performance.clamp(fadeIn ? sinceTile : 1 - sinceParent, 0, 1); + // we don't crossfade tiles that were just refreshed upon expiring: + // once they're old enough to pass the crossfading threshold + // (fadeDuration), unset the `refreshedUponExpiration` flag so we don't + // incorrectly fail to crossfade them when zooming + if (tile.refreshedUponExpiration && sinceTile >= 1) + tile.refreshedUponExpiration = false; + if (parentTile) { + return { + opacity: 1, + mix: 1 - childOpacity + }; + } + else { + return { + opacity: childOpacity, + mix: 0 + }; + } + } + else { + return { + opacity: 1, + mix: 0 + }; + } +} + +function drawBackground(painter, sourceCache, layer, coords) { + const color = layer.paint.get('background-color'); + const opacity = layer.paint.get('background-opacity'); + if (opacity === 0) + return; + const context = painter.context; + const gl = context.gl; + const transform = painter.transform; + const tileSize = transform.tileSize; + const image = layer.paint.get('background-pattern'); + if (painter.isPatternMissing(image)) + return; + const pass = (!image && color.a === 1 && opacity === 1 && painter.opaquePassEnabledForLayer()) ? 'opaque' : 'translucent'; + if (painter.renderPass !== pass) + return; + const stencilMode = StencilMode.disabled; + const depthMode = painter.depthModeForSublayer(0, pass === 'opaque' ? DepthMode.ReadWrite : DepthMode.ReadOnly); + const colorMode = painter.colorModeForRenderPass(); + const program = painter.useProgram(image ? 'backgroundPattern' : 'background'); + const tileIDs = coords ? coords : transform.coveringTiles({ tileSize, terrain: painter.style.map.terrain }); + if (image) { + context.activeTexture.set(gl.TEXTURE0); + painter.imageManager.bind(painter.context); + } + const crossfade = layer.getCrossfadeParameters(); + for (const tileID of tileIDs) { + const matrix = coords ? tileID.posMatrix : painter.transform.calculatePosMatrix(tileID.toUnwrapped()); + const uniformValues = image ? + backgroundPatternUniformValues(matrix, opacity, painter, image, { tileID, tileSize }, crossfade) : + backgroundUniformValues(matrix, opacity, color); + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(tileID); + program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled, uniformValues, terrainData, layer.id, painter.tileExtentBuffer, painter.quadTriangleIndexBuffer, painter.tileExtentSegments); + } +} + +const topColor = new performance.Color(1, 0, 0, 1); +const btmColor = new performance.Color(0, 1, 0, 1); +const leftColor = new performance.Color(0, 0, 1, 1); +const rightColor = new performance.Color(1, 0, 1, 1); +const centerColor = new performance.Color(0, 1, 1, 1); +function drawDebugPadding(painter) { + const padding = painter.transform.padding; + const lineWidth = 3; + // Top + drawHorizontalLine(painter, painter.transform.height - (padding.top || 0), lineWidth, topColor); + // Bottom + drawHorizontalLine(painter, padding.bottom || 0, lineWidth, btmColor); + // Left + drawVerticalLine(painter, padding.left || 0, lineWidth, leftColor); + // Right + drawVerticalLine(painter, painter.transform.width - (padding.right || 0), lineWidth, rightColor); + // Center + const center = painter.transform.centerPoint; + drawCrosshair(painter, center.x, painter.transform.height - center.y, centerColor); +} +function drawCrosshair(painter, x, y, color) { + const size = 20; + const lineWidth = 2; + //Vertical line + drawDebugSSRect(painter, x - lineWidth / 2, y - size / 2, lineWidth, size, color); + //Horizontal line + drawDebugSSRect(painter, x - size / 2, y - lineWidth / 2, size, lineWidth, color); +} +function drawHorizontalLine(painter, y, lineWidth, color) { + drawDebugSSRect(painter, 0, y + lineWidth / 2, painter.transform.width, lineWidth, color); +} +function drawVerticalLine(painter, x, lineWidth, color) { + drawDebugSSRect(painter, x - lineWidth / 2, 0, lineWidth, painter.transform.height, color); +} +function drawDebugSSRect(painter, x, y, width, height, color) { + const context = painter.context; + const gl = context.gl; + gl.enable(gl.SCISSOR_TEST); + gl.scissor(x * painter.pixelRatio, y * painter.pixelRatio, width * painter.pixelRatio, height * painter.pixelRatio); + context.clear({ color }); + gl.disable(gl.SCISSOR_TEST); +} +function drawDebug(painter, sourceCache, coords) { + for (let i = 0; i < coords.length; i++) { + drawDebugTile(painter, sourceCache, coords[i]); + } +} +function drawDebugTile(painter, sourceCache, coord) { + const context = painter.context; + const gl = context.gl; + const posMatrix = coord.posMatrix; + const program = painter.useProgram('debug'); + const depthMode = DepthMode.disabled; + const stencilMode = StencilMode.disabled; + const colorMode = painter.colorModeForRenderPass(); + const id = '$debug'; + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + context.activeTexture.set(gl.TEXTURE0); + const tileRawData = sourceCache.getTileByID(coord.key).latestRawTileData; + const tileByteLength = (tileRawData && tileRawData.byteLength) || 0; + const tileSizeKb = Math.floor(tileByteLength / 1024); + const tileSize = sourceCache.getTile(coord).tileSize; + const scaleRatio = (512 / Math.min(tileSize, 512) * (coord.overscaledZ / painter.transform.zoom)) * 0.5; + let tileIdText = coord.canonical.toString(); + if (coord.overscaledZ !== coord.canonical.z) { + tileIdText += ` => ${coord.overscaledZ}`; + } + const tileLabel = `${tileIdText} ${tileSizeKb}kB`; + drawTextToOverlay(painter, tileLabel); + program.draw(context, gl.TRIANGLES, depthMode, stencilMode, ColorMode.alphaBlended, CullFaceMode.disabled, debugUniformValues(posMatrix, performance.Color.transparent, scaleRatio), null, id, painter.debugBuffer, painter.quadTriangleIndexBuffer, painter.debugSegments); + program.draw(context, gl.LINE_STRIP, depthMode, stencilMode, colorMode, CullFaceMode.disabled, debugUniformValues(posMatrix, performance.Color.red), terrainData, id, painter.debugBuffer, painter.tileBorderIndexBuffer, painter.debugSegments); +} +function drawTextToOverlay(painter, text) { + painter.initDebugOverlayCanvas(); + const canvas = painter.debugOverlayCanvas; + const gl = painter.context.gl; + const ctx2d = painter.debugOverlayCanvas.getContext('2d'); + ctx2d.clearRect(0, 0, canvas.width, canvas.height); + ctx2d.shadowColor = 'white'; + ctx2d.shadowBlur = 2; + ctx2d.lineWidth = 1.5; + ctx2d.strokeStyle = 'white'; + ctx2d.textBaseline = 'top'; + ctx2d.font = `bold ${36}px Open Sans, sans-serif`; + ctx2d.fillText(text, 5, 5); + ctx2d.strokeText(text, 5, 5); + painter.debugOverlayTexture.update(canvas); + painter.debugOverlayTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); +} +function selectDebugSource(style, zoom) { + // Use vector source with highest maxzoom + // Else use source with highest maxzoom of any type + let selectedSource = null; + const layers = Object.values(style._layers); + const sources = layers.flatMap((layer) => { + if (layer.source && !layer.isHidden(zoom)) { + const sourceCache = style.sourceCaches[layer.source]; + return [sourceCache]; + } + else { + return []; + } + }); + const vectorSources = sources.filter((source) => source.getSource().type === 'vector'); + const otherSources = sources.filter((source) => source.getSource().type !== 'vector'); + const considerSource = (source) => { + if (!selectedSource || (selectedSource.getSource().maxzoom < source.getSource().maxzoom)) { + selectedSource = source; + } + }; + vectorSources.forEach((source) => considerSource(source)); + if (!selectedSource) { + otherSources.forEach((source) => considerSource(source)); + } + return selectedSource; +} + +function drawCustom(painter, sourceCache, layer) { + const context = painter.context; + const implementation = layer.implementation; + if (painter.renderPass === 'offscreen') { + const prerender = implementation.prerender; + if (prerender) { + painter.setCustomLayerDefaults(); + context.setColorMode(painter.colorModeForRenderPass()); + prerender.call(implementation, context.gl, painter.transform.customLayerMatrix()); + context.setDirty(); + painter.setBaseState(); + } + } + else if (painter.renderPass === 'translucent') { + painter.setCustomLayerDefaults(); + context.setColorMode(painter.colorModeForRenderPass()); + context.setStencilMode(StencilMode.disabled); + const depthMode = implementation.renderingMode === '3d' ? + new DepthMode(painter.context.gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D) : + painter.depthModeForSublayer(0, DepthMode.ReadOnly); + context.setDepthMode(depthMode); + implementation.render(context.gl, painter.transform.customLayerMatrix()); + context.setDirty(); + painter.setBaseState(); + context.bindFramebuffer.set(null); + } +} + +/** + * Redraw the Depth Framebuffer + * @param painter - the painter + * @param terrain - the terrain + */ +function drawDepth(painter, terrain) { + const context = painter.context; + const gl = context.gl; + const colorMode = ColorMode.unblended; + const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, [0, 1]); + const mesh = terrain.getTerrainMesh(); + const tiles = terrain.sourceCache.getRenderableTiles(); + const program = painter.useProgram('terrainDepth'); + context.bindFramebuffer.set(terrain.getFramebuffer('depth').framebuffer); + context.viewport.set([0, 0, painter.width / devicePixelRatio, painter.height / devicePixelRatio]); + context.clear({ color: performance.Color.transparent, depth: 1 }); + for (const tile of tiles) { + const terrainData = terrain.getTerrainData(tile.tileID); + const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped()); + const uniformValues = terrainDepthUniformValues(posMatrix, terrain.getMeshFrameDelta(painter.transform.zoom)); + program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments); + } + context.bindFramebuffer.set(null); + context.viewport.set([0, 0, painter.width, painter.height]); +} +/** + * Redraw the Coords Framebuffers + * @param painter - the painter + * @param terrain - the terrain + */ +function drawCoords(painter, terrain) { + const context = painter.context; + const gl = context.gl; + const colorMode = ColorMode.unblended; + const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, [0, 1]); + const mesh = terrain.getTerrainMesh(); + const coords = terrain.getCoordsTexture(); + const tiles = terrain.sourceCache.getRenderableTiles(); + // draw tile-coords into framebuffer + const program = painter.useProgram('terrainCoords'); + context.bindFramebuffer.set(terrain.getFramebuffer('coords').framebuffer); + context.viewport.set([0, 0, painter.width / devicePixelRatio, painter.height / devicePixelRatio]); + context.clear({ color: performance.Color.transparent, depth: 1 }); + terrain.coordsIndex = []; + for (const tile of tiles) { + const terrainData = terrain.getTerrainData(tile.tileID); + context.activeTexture.set(gl.TEXTURE0); + gl.bindTexture(gl.TEXTURE_2D, coords.texture); + const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped()); + const uniformValues = terrainCoordsUniformValues(posMatrix, 255 - terrain.coordsIndex.length, terrain.getMeshFrameDelta(painter.transform.zoom)); + program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments); + terrain.coordsIndex.push(tile.tileID.key); + } + context.bindFramebuffer.set(null); + context.viewport.set([0, 0, painter.width, painter.height]); +} +function drawTerrain(painter, terrain, tiles) { + const context = painter.context; + const gl = context.gl; + const colorMode = painter.colorModeForRenderPass(); + const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D); + const program = painter.useProgram('terrain'); + const mesh = terrain.getTerrainMesh(); + context.bindFramebuffer.set(null); + context.viewport.set([0, 0, painter.width, painter.height]); + for (const tile of tiles) { + const texture = painter.renderToTexture.getTexture(tile); + const terrainData = terrain.getTerrainData(tile.tileID); + context.activeTexture.set(gl.TEXTURE0); + gl.bindTexture(gl.TEXTURE_2D, texture.texture); + const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped()); + const uniformValues = terrainUniformValues(posMatrix, terrain.getMeshFrameDelta(painter.transform.zoom)); + program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments); + } +} + +/** + * @internal + * Initialize a new painter object. + */ +class Painter { + constructor(gl, transform) { + this.context = new Context(gl); + this.transform = transform; + this._tileTextures = {}; + this.terrainFacilitator = { dirty: true, matrix: performance.create(), renderTime: 0 }; + this.setup(); + // Within each layer there are multiple distinct z-planes that can be drawn to. + // This is implemented using the WebGL depth buffer. + this.numSublayers = SourceCache.maxUnderzooming + SourceCache.maxOverzooming + 1; + this.depthEpsilon = 1 / Math.pow(2, 16); + this.crossTileSymbolIndex = new CrossTileSymbolIndex(); + } + /* + * Update the GL viewport, projection matrix, and transforms to compensate + * for a new width and height value. + */ + resize(width, height, pixelRatio) { + this.width = Math.floor(width * pixelRatio); + this.height = Math.floor(height * pixelRatio); + this.pixelRatio = pixelRatio; + this.context.viewport.set([0, 0, this.width, this.height]); + if (this.style) { + for (const layerId of this.style._order) { + this.style._layers[layerId].resize(); + } + } + } + setup() { + const context = this.context; + const tileExtentArray = new performance.PosArray(); + tileExtentArray.emplaceBack(0, 0); + tileExtentArray.emplaceBack(performance.EXTENT, 0); + tileExtentArray.emplaceBack(0, performance.EXTENT); + tileExtentArray.emplaceBack(performance.EXTENT, performance.EXTENT); + this.tileExtentBuffer = context.createVertexBuffer(tileExtentArray, posAttributes.members); + this.tileExtentSegments = performance.SegmentVector.simpleSegment(0, 0, 4, 2); + const debugArray = new performance.PosArray(); + debugArray.emplaceBack(0, 0); + debugArray.emplaceBack(performance.EXTENT, 0); + debugArray.emplaceBack(0, performance.EXTENT); + debugArray.emplaceBack(performance.EXTENT, performance.EXTENT); + this.debugBuffer = context.createVertexBuffer(debugArray, posAttributes.members); + this.debugSegments = performance.SegmentVector.simpleSegment(0, 0, 4, 5); + const rasterBoundsArray = new performance.RasterBoundsArray(); + rasterBoundsArray.emplaceBack(0, 0, 0, 0); + rasterBoundsArray.emplaceBack(performance.EXTENT, 0, performance.EXTENT, 0); + rasterBoundsArray.emplaceBack(0, performance.EXTENT, 0, performance.EXTENT); + rasterBoundsArray.emplaceBack(performance.EXTENT, performance.EXTENT, performance.EXTENT, performance.EXTENT); + this.rasterBoundsBuffer = context.createVertexBuffer(rasterBoundsArray, rasterBoundsAttributes.members); + this.rasterBoundsSegments = performance.SegmentVector.simpleSegment(0, 0, 4, 2); + const viewportArray = new performance.PosArray(); + viewportArray.emplaceBack(0, 0); + viewportArray.emplaceBack(1, 0); + viewportArray.emplaceBack(0, 1); + viewportArray.emplaceBack(1, 1); + this.viewportBuffer = context.createVertexBuffer(viewportArray, posAttributes.members); + this.viewportSegments = performance.SegmentVector.simpleSegment(0, 0, 4, 2); + const tileLineStripIndices = new performance.LineStripIndexArray(); + tileLineStripIndices.emplaceBack(0); + tileLineStripIndices.emplaceBack(1); + tileLineStripIndices.emplaceBack(3); + tileLineStripIndices.emplaceBack(2); + tileLineStripIndices.emplaceBack(0); + this.tileBorderIndexBuffer = context.createIndexBuffer(tileLineStripIndices); + const quadTriangleIndices = new performance.TriangleIndexArray(); + quadTriangleIndices.emplaceBack(0, 1, 2); + quadTriangleIndices.emplaceBack(2, 1, 3); + this.quadTriangleIndexBuffer = context.createIndexBuffer(quadTriangleIndices); + const gl = this.context.gl; + this.stencilClearMode = new StencilMode({ func: gl.ALWAYS, mask: 0 }, 0x0, 0xFF, gl.ZERO, gl.ZERO, gl.ZERO); + } + /* + * Reset the drawing canvas by clearing the stencil buffer so that we can draw + * new tiles at the same location, while retaining previously drawn pixels. + */ + clearStencil() { + const context = this.context; + const gl = context.gl; + this.nextStencilID = 1; + this.currentStencilSource = undefined; + // As a temporary workaround for https://github.com/mapbox/mapbox-gl-js/issues/5490, + // pending an upstream fix, we draw a fullscreen stencil=0 clipping mask here, + // effectively clearing the stencil buffer: once an upstream patch lands, remove + // this function in favor of context.clear({ stencil: 0x0 }) + const matrix = performance.create(); + performance.ortho(matrix, 0, this.width, this.height, 0, 0, 1); + performance.scale(matrix, matrix, [gl.drawingBufferWidth, gl.drawingBufferHeight, 0]); + this.useProgram('clippingMask').draw(context, gl.TRIANGLES, DepthMode.disabled, this.stencilClearMode, ColorMode.disabled, CullFaceMode.disabled, clippingMaskUniformValues(matrix), null, '$clipping', this.viewportBuffer, this.quadTriangleIndexBuffer, this.viewportSegments); + } + _renderTileClippingMasks(layer, tileIDs) { + if (this.currentStencilSource === layer.source || !layer.isTileClipped() || !tileIDs || !tileIDs.length) + return; + this.currentStencilSource = layer.source; + const context = this.context; + const gl = context.gl; + if (this.nextStencilID + tileIDs.length > 256) { + // we'll run out of fresh IDs so we need to clear and start from scratch + this.clearStencil(); + } + context.setColorMode(ColorMode.disabled); + context.setDepthMode(DepthMode.disabled); + const program = this.useProgram('clippingMask'); + this._tileClippingMaskIDs = {}; + for (const tileID of tileIDs) { + const id = this._tileClippingMaskIDs[tileID.key] = this.nextStencilID++; + const terrainData = this.style.map.terrain && this.style.map.terrain.getTerrainData(tileID); + program.draw(context, gl.TRIANGLES, DepthMode.disabled, + // Tests will always pass, and ref value will be written to stencil buffer. + new StencilMode({ func: gl.ALWAYS, mask: 0 }, id, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE), ColorMode.disabled, CullFaceMode.disabled, clippingMaskUniformValues(tileID.posMatrix), terrainData, '$clipping', this.tileExtentBuffer, this.quadTriangleIndexBuffer, this.tileExtentSegments); + } + } + stencilModeFor3D() { + this.currentStencilSource = undefined; + if (this.nextStencilID + 1 > 256) { + this.clearStencil(); + } + const id = this.nextStencilID++; + const gl = this.context.gl; + return new StencilMode({ func: gl.NOTEQUAL, mask: 0xFF }, id, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE); + } + stencilModeForClipping(tileID) { + const gl = this.context.gl; + return new StencilMode({ func: gl.EQUAL, mask: 0xFF }, this._tileClippingMaskIDs[tileID.key], 0x00, gl.KEEP, gl.KEEP, gl.REPLACE); + } + /* + * Sort coordinates by Z as drawing tiles is done in Z-descending order. + * All children with the same Z write the same stencil value. Children + * stencil values are greater than parent's. This is used only for raster + * and raster-dem tiles, which are already clipped to tile boundaries, to + * mask area of tile overlapped by children tiles. + * Stencil ref values continue range used in _tileClippingMaskIDs. + * + * Returns [StencilMode for tile overscaleZ map, sortedCoords]. + */ + stencilConfigForOverlap(tileIDs) { + const gl = this.context.gl; + const coords = tileIDs.sort((a, b) => b.overscaledZ - a.overscaledZ); + const minTileZ = coords[coords.length - 1].overscaledZ; + const stencilValues = coords[0].overscaledZ - minTileZ + 1; + if (stencilValues > 1) { + this.currentStencilSource = undefined; + if (this.nextStencilID + stencilValues > 256) { + this.clearStencil(); + } + const zToStencilMode = {}; + for (let i = 0; i < stencilValues; i++) { + zToStencilMode[i + minTileZ] = new StencilMode({ func: gl.GEQUAL, mask: 0xFF }, i + this.nextStencilID, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE); + } + this.nextStencilID += stencilValues; + return [zToStencilMode, coords]; + } + return [{ [minTileZ]: StencilMode.disabled }, coords]; + } + colorModeForRenderPass() { + const gl = this.context.gl; + if (this._showOverdrawInspector) { + const numOverdrawSteps = 8; + const a = 1 / numOverdrawSteps; + return new ColorMode([gl.CONSTANT_COLOR, gl.ONE], new performance.Color(a, a, a, 0), [true, true, true, true]); + } + else if (this.renderPass === 'opaque') { + return ColorMode.unblended; + } + else { + return ColorMode.alphaBlended; + } + } + depthModeForSublayer(n, mask, func) { + if (!this.opaquePassEnabledForLayer()) + return DepthMode.disabled; + const depth = 1 - ((1 + this.currentLayer) * this.numSublayers + n) * this.depthEpsilon; + return new DepthMode(func || this.context.gl.LEQUAL, mask, [depth, depth]); + } + /* + * The opaque pass and 3D layers both use the depth buffer. + * Layers drawn above 3D layers need to be drawn using the + * painter's algorithm so that they appear above 3D features. + * This returns true for layers that can be drawn using the + * opaque pass. + */ + opaquePassEnabledForLayer() { + return this.currentLayer < this.opaquePassCutoff; + } + render(style, options) { + this.style = style; + this.options = options; + this.lineAtlas = style.lineAtlas; + this.imageManager = style.imageManager; + this.glyphManager = style.glyphManager; + this.symbolFadeChange = style.placement.symbolFadeChange(performance.browser.now()); + this.imageManager.beginFrame(); + const layerIds = this.style._order; + const sourceCaches = this.style.sourceCaches; + const coordsAscending = {}; + const coordsDescending = {}; + const coordsDescendingSymbol = {}; + for (const id in sourceCaches) { + const sourceCache = sourceCaches[id]; + if (sourceCache.used) { + sourceCache.prepare(this.context); + } + coordsAscending[id] = sourceCache.getVisibleCoordinates(); + coordsDescending[id] = coordsAscending[id].slice().reverse(); + coordsDescendingSymbol[id] = sourceCache.getVisibleCoordinates(true).reverse(); + } + this.opaquePassCutoff = Infinity; + for (let i = 0; i < layerIds.length; i++) { + const layerId = layerIds[i]; + if (this.style._layers[layerId].is3D()) { + this.opaquePassCutoff = i; + break; + } + } + if (this.renderToTexture) { + this.renderToTexture.prepareForRender(this.style, this.transform.zoom); + // this is disabled, because render-to-texture is rendering all layers from bottom to top. + this.opaquePassCutoff = 0; + // update coords/depth-framebuffer on camera movement, or tile reloading + const newTiles = this.style.map.terrain.sourceCache.tilesAfterTime(this.terrainFacilitator.renderTime); + if (this.terrainFacilitator.dirty || !performance.equals(this.terrainFacilitator.matrix, this.transform.projMatrix) || newTiles.length) { + performance.copy(this.terrainFacilitator.matrix, this.transform.projMatrix); + this.terrainFacilitator.renderTime = Date.now(); + this.terrainFacilitator.dirty = false; + drawDepth(this, this.style.map.terrain); + drawCoords(this, this.style.map.terrain); + } + } + // Offscreen pass =============================================== + // We first do all rendering that requires rendering to a separate + // framebuffer, and then save those for rendering back to the map + // later: in doing this we avoid doing expensive framebuffer restores. + this.renderPass = 'offscreen'; + for (const layerId of layerIds) { + const layer = this.style._layers[layerId]; + if (!layer.hasOffscreenPass() || layer.isHidden(this.transform.zoom)) + continue; + const coords = coordsDescending[layer.source]; + if (layer.type !== 'custom' && !coords.length) + continue; + this.renderLayer(this, sourceCaches[layer.source], layer, coords); + } + // Rebind the main framebuffer now that all offscreen layers have been rendered: + this.context.bindFramebuffer.set(null); + // Clear buffers in preparation for drawing to the main framebuffer + this.context.clear({ color: options.showOverdrawInspector ? performance.Color.black : performance.Color.transparent, depth: 1 }); + this.clearStencil(); + this._showOverdrawInspector = options.showOverdrawInspector; + this.depthRangeFor3D = [0, 1 - ((style._order.length + 2) * this.numSublayers * this.depthEpsilon)]; + // Opaque pass =============================================== + // Draw opaque layers top-to-bottom first. + if (!this.renderToTexture) { + this.renderPass = 'opaque'; + for (this.currentLayer = layerIds.length - 1; this.currentLayer >= 0; this.currentLayer--) { + const layer = this.style._layers[layerIds[this.currentLayer]]; + const sourceCache = sourceCaches[layer.source]; + const coords = coordsAscending[layer.source]; + this._renderTileClippingMasks(layer, coords); + this.renderLayer(this, sourceCache, layer, coords); + } + } + // Translucent pass =============================================== + // Draw all other layers bottom-to-top. + this.renderPass = 'translucent'; + for (this.currentLayer = 0; this.currentLayer < layerIds.length; this.currentLayer++) { + const layer = this.style._layers[layerIds[this.currentLayer]]; + const sourceCache = sourceCaches[layer.source]; + if (this.renderToTexture && this.renderToTexture.renderLayer(layer)) + continue; + // For symbol layers in the translucent pass, we add extra tiles to the renderable set + // for cross-tile symbol fading. Symbol layers don't use tile clipping, so no need to render + // separate clipping masks + const coords = (layer.type === 'symbol' ? coordsDescendingSymbol : coordsDescending)[layer.source]; + this._renderTileClippingMasks(layer, coordsAscending[layer.source]); + this.renderLayer(this, sourceCache, layer, coords); + } + if (this.options.showTileBoundaries) { + const selectedSource = selectDebugSource(this.style, this.transform.zoom); + if (selectedSource) { + drawDebug(this, selectedSource, selectedSource.getVisibleCoordinates()); + } + } + if (this.options.showPadding) { + drawDebugPadding(this); + } + // Set defaults for most GL values so that anyone using the state after the render + // encounters more expected values. + this.context.setDefault(); + } + renderLayer(painter, sourceCache, layer, coords) { + if (layer.isHidden(this.transform.zoom)) + return; + if (layer.type !== 'background' && layer.type !== 'custom' && !(coords || []).length) + return; + this.id = layer.id; + switch (layer.type) { + case 'symbol': + drawSymbols(painter, sourceCache, layer, coords, this.style.placement.variableOffsets); + break; + case 'circle': + drawCircles(painter, sourceCache, layer, coords); + break; + case 'heatmap': + drawHeatmap(painter, sourceCache, layer, coords); + break; + case 'line': + drawLine(painter, sourceCache, layer, coords); + break; + case 'fill': + drawFill(painter, sourceCache, layer, coords); + break; + case 'fill-extrusion': + drawFillExtrusion(painter, sourceCache, layer, coords); + break; + case 'hillshade': + drawHillshade(painter, sourceCache, layer, coords); + break; + case 'raster': + drawRaster(painter, sourceCache, layer, coords); + break; + case 'background': + drawBackground(painter, sourceCache, layer, coords); + break; + case 'custom': + drawCustom(painter, sourceCache, layer); + break; + } + } + /** + * Transform a matrix to incorporate the *-translate and *-translate-anchor properties into it. + * @param inViewportPixelUnitsUnits - True when the units accepted by the matrix are in viewport pixels instead of tile units. + * @returns matrix + */ + translatePosMatrix(matrix, tile, translate, translateAnchor, inViewportPixelUnitsUnits) { + if (!translate[0] && !translate[1]) + return matrix; + const angle = inViewportPixelUnitsUnits ? + (translateAnchor === 'map' ? this.transform.angle : 0) : + (translateAnchor === 'viewport' ? -this.transform.angle : 0); + if (angle) { + const sinA = Math.sin(angle); + const cosA = Math.cos(angle); + translate = [ + translate[0] * cosA - translate[1] * sinA, + translate[0] * sinA + translate[1] * cosA + ]; + } + const translation = [ + inViewportPixelUnitsUnits ? translate[0] : pixelsToTileUnits(tile, translate[0], this.transform.zoom), + inViewportPixelUnitsUnits ? translate[1] : pixelsToTileUnits(tile, translate[1], this.transform.zoom), + 0 + ]; + const translatedMatrix = new Float32Array(16); + performance.translate(translatedMatrix, matrix, translation); + return translatedMatrix; + } + saveTileTexture(texture) { + const textures = this._tileTextures[texture.size[0]]; + if (!textures) { + this._tileTextures[texture.size[0]] = [texture]; + } + else { + textures.push(texture); + } + } + getTileTexture(size) { + const textures = this._tileTextures[size]; + return textures && textures.length > 0 ? textures.pop() : null; + } + /** + * Checks whether a pattern image is needed, and if it is, whether it is not loaded. + * + * @returns true if a needed image is missing and rendering needs to be skipped. + */ + isPatternMissing(image) { + if (!image) + return false; + if (!image.from || !image.to) + return true; + const imagePosA = this.imageManager.getPattern(image.from.toString()); + const imagePosB = this.imageManager.getPattern(image.to.toString()); + return !imagePosA || !imagePosB; + } + useProgram(name, programConfiguration) { + this.cache = this.cache || {}; + const key = name + + (programConfiguration ? programConfiguration.cacheKey : '') + + (this._showOverdrawInspector ? '/overdraw' : '') + + (this.style.map.terrain ? '/terrain' : ''); + if (!this.cache[key]) { + this.cache[key] = new Program(this.context, shaders[name], programConfiguration, programUniforms[name], this._showOverdrawInspector, this.style.map.terrain); + } + return this.cache[key]; + } + /* + * Reset some GL state to default values to avoid hard-to-debug bugs + * in custom layers. + */ + setCustomLayerDefaults() { + // Prevent custom layers from unintentionally modify the last VAO used. + // All other state is state is restored on it's own, but for VAOs it's + // simpler to unbind so that we don't have to track the state of VAOs. + this.context.unbindVAO(); + // The default values for this state is meaningful and often expected. + // Leaving this state dirty could cause a lot of confusion for users. + this.context.cullFace.setDefault(); + this.context.activeTexture.setDefault(); + this.context.pixelStoreUnpack.setDefault(); + this.context.pixelStoreUnpackPremultiplyAlpha.setDefault(); + this.context.pixelStoreUnpackFlipY.setDefault(); + } + /* + * Set GL state that is shared by all layers. + */ + setBaseState() { + const gl = this.context.gl; + this.context.cullFace.set(false); + this.context.viewport.set([0, 0, this.width, this.height]); + this.context.blendEquation.set(gl.FUNC_ADD); + } + initDebugOverlayCanvas() { + if (this.debugOverlayCanvas == null) { + this.debugOverlayCanvas = document.createElement('canvas'); + this.debugOverlayCanvas.width = 512; + this.debugOverlayCanvas.height = 512; + const gl = this.context.gl; + this.debugOverlayTexture = new Texture(this.context, this.debugOverlayCanvas, gl.RGBA); + } + } + destroy() { + if (this.debugOverlayTexture) { + this.debugOverlayTexture.destroy(); + } + } + /* + * Return true if drawing buffer size is != from requested size. + * That means that we've reached GL limits somehow. + * Note: drawing buffer size changes only when canvas size changes + */ + overLimit() { + const { drawingBufferWidth, drawingBufferHeight } = this.context.gl; + return this.width !== drawingBufferWidth || this.height !== drawingBufferHeight; + } +} + +class Frustum { + constructor(points, planes) { + this.points = points; + this.planes = planes; + } + static fromInvProjectionMatrix(invProj, worldSize, zoom) { + const clipSpaceCorners = [ + [-1, 1, -1, 1], + [1, 1, -1, 1], + [1, -1, -1, 1], + [-1, -1, -1, 1], + [-1, 1, 1, 1], + [1, 1, 1, 1], + [1, -1, 1, 1], + [-1, -1, 1, 1] + ]; + const scale = Math.pow(2, zoom); + // Transform frustum corner points from clip space to tile space, Z to meters + const frustumCoords = clipSpaceCorners.map(v => { + v = performance.transformMat4([], v, invProj); + const s = 1.0 / v[3] / worldSize * scale; + return performance.mul$1(v, v, [s, s, 1.0 / v[3], s]); + }); + const frustumPlanePointIndices = [ + [0, 1, 2], + [6, 5, 4], + [0, 3, 7], + [2, 1, 5], + [3, 2, 6], + [0, 4, 5] // top + ]; + const frustumPlanes = frustumPlanePointIndices.map((p) => { + const a = performance.sub([], frustumCoords[p[0]], frustumCoords[p[1]]); + const b = performance.sub([], frustumCoords[p[2]], frustumCoords[p[1]]); + const n = performance.normalize([], performance.cross([], a, b)); + const d = -performance.dot(n, frustumCoords[p[1]]); + return n.concat(d); + }); + return new Frustum(frustumCoords, frustumPlanes); + } +} +class Aabb { + constructor(min_, max_) { + this.min = min_; + this.max = max_; + this.center = performance.scale$1([], performance.add([], this.min, this.max), 0.5); + } + quadrant(index) { + const split = [(index % 2) === 0, index < 2]; + const qMin = performance.clone$2(this.min); + const qMax = performance.clone$2(this.max); + for (let axis = 0; axis < split.length; axis++) { + qMin[axis] = split[axis] ? this.min[axis] : this.center[axis]; + qMax[axis] = split[axis] ? this.center[axis] : this.max[axis]; + } + // Elevation is always constant, hence quadrant.max.z = this.max.z + qMax[2] = this.max[2]; + return new Aabb(qMin, qMax); + } + distanceX(point) { + const pointOnAabb = Math.max(Math.min(this.max[0], point[0]), this.min[0]); + return pointOnAabb - point[0]; + } + distanceY(point) { + const pointOnAabb = Math.max(Math.min(this.max[1], point[1]), this.min[1]); + return pointOnAabb - point[1]; + } + // Performs a frustum-aabb intersection test. Returns 0 if there's no intersection, + // 1 if shapes are intersecting and 2 if the aabb if fully inside the frustum. + intersects(frustum) { + // Execute separating axis test between two convex objects to find intersections + // Each frustum plane together with 3 major axes define the separating axes + const aabbPoints = [ + [this.min[0], this.min[1], this.min[2], 1], + [this.max[0], this.min[1], this.min[2], 1], + [this.max[0], this.max[1], this.min[2], 1], + [this.min[0], this.max[1], this.min[2], 1], + [this.min[0], this.min[1], this.max[2], 1], + [this.max[0], this.min[1], this.max[2], 1], + [this.max[0], this.max[1], this.max[2], 1], + [this.min[0], this.max[1], this.max[2], 1] + ]; + let fullyInside = true; + for (let p = 0; p < frustum.planes.length; p++) { + const plane = frustum.planes[p]; + let pointsInside = 0; + for (let i = 0; i < aabbPoints.length; i++) { + if (performance.dot$1(plane, aabbPoints[i]) >= 0) { + pointsInside++; + } + } + if (pointsInside === 0) + return 0; + if (pointsInside !== aabbPoints.length) + fullyInside = false; + } + if (fullyInside) + return 2; + for (let axis = 0; axis < 3; axis++) { + let projMin = Number.MAX_VALUE; + let projMax = -Number.MAX_VALUE; + for (let p = 0; p < frustum.points.length; p++) { + const projectedPoint = frustum.points[p][axis] - this.min[axis]; + projMin = Math.min(projMin, projectedPoint); + projMax = Math.max(projMax, projectedPoint); + } + if (projMax < 0 || projMin > this.max[axis] - this.min[axis]) + return 0; + } + return 1; + } +} + +/** + * An `EdgeInset` object represents screen space padding applied to the edges of the viewport. + * This shifts the apprent center or the vanishing point of the map. This is useful for adding floating UI elements + * on top of the map and having the vanishing point shift as UI elements resize. + * + * @group Geography and Geometry + */ +class EdgeInsets { + constructor(top = 0, bottom = 0, left = 0, right = 0) { + if (isNaN(top) || top < 0 || + isNaN(bottom) || bottom < 0 || + isNaN(left) || left < 0 || + isNaN(right) || right < 0) { + throw new Error('Invalid value for edge-insets, top, bottom, left and right must all be numbers'); + } + this.top = top; + this.bottom = bottom; + this.left = left; + this.right = right; + } + /** + * Interpolates the inset in-place. + * This maintains the current inset value for any inset not present in `target`. + * @param start - interpolation start + * @param target - interpolation target + * @param t - interpolation step/weight + * @returns the insets + */ + interpolate(start, target, t) { + if (target.top != null && start.top != null) + this.top = performance.interpolate.number(start.top, target.top, t); + if (target.bottom != null && start.bottom != null) + this.bottom = performance.interpolate.number(start.bottom, target.bottom, t); + if (target.left != null && start.left != null) + this.left = performance.interpolate.number(start.left, target.left, t); + if (target.right != null && start.right != null) + this.right = performance.interpolate.number(start.right, target.right, t); + return this; + } + /** + * Utility method that computes the new apprent center or vanishing point after applying insets. + * This is in pixels and with the top left being (0.0) and +y being downwards. + * + * @param width - the width + * @param height - the height + * @returns the point + */ + getCenter(width, height) { + // Clamp insets so they never overflow width/height and always calculate a valid center + const x = performance.clamp((this.left + width - this.right) / 2, 0, width); + const y = performance.clamp((this.top + height - this.bottom) / 2, 0, height); + return new performance.Point(x, y); + } + equals(other) { + return this.top === other.top && + this.bottom === other.bottom && + this.left === other.left && + this.right === other.right; + } + clone() { + return new EdgeInsets(this.top, this.bottom, this.left, this.right); + } + /** + * Returns the current state as json, useful when you want to have a + * read-only representation of the inset. + * + * @returns state as json + */ + toJSON() { + return { + top: this.top, + bottom: this.bottom, + left: this.left, + right: this.right + }; + } +} + +/** + * @internal + * A single transform, generally used for a single tile to be + * scaled, rotated, and zoomed. + */ +class Transform { + constructor(minZoom, maxZoom, minPitch, maxPitch, renderWorldCopies) { + this.tileSize = 512; // constant + this.maxValidLatitude = 85.051129; // constant + this._renderWorldCopies = renderWorldCopies === undefined ? true : !!renderWorldCopies; + this._minZoom = minZoom || 0; + this._maxZoom = maxZoom || 22; + this._minPitch = (minPitch === undefined || minPitch === null) ? 0 : minPitch; + this._maxPitch = (maxPitch === undefined || maxPitch === null) ? 60 : maxPitch; + this.setMaxBounds(); + this.width = 0; + this.height = 0; + this._center = new performance.LngLat(0, 0); + this._elevation = 0; + this.zoom = 0; + this.angle = 0; + this._fov = 0.6435011087932844; + this._pitch = 0; + this._unmodified = true; + this._edgeInsets = new EdgeInsets(); + this._posMatrixCache = {}; + this._alignedPosMatrixCache = {}; + this._minEleveationForCurrentTile = 0; + } + clone() { + const clone = new Transform(this._minZoom, this._maxZoom, this._minPitch, this.maxPitch, this._renderWorldCopies); + clone.apply(this); + return clone; + } + apply(that) { + this.tileSize = that.tileSize; + this.latRange = that.latRange; + this.width = that.width; + this.height = that.height; + this._center = that._center; + this._elevation = that._elevation; + this._minEleveationForCurrentTile = that._minEleveationForCurrentTile; + this.zoom = that.zoom; + this.angle = that.angle; + this._fov = that._fov; + this._pitch = that._pitch; + this._unmodified = that._unmodified; + this._edgeInsets = that._edgeInsets.clone(); + this._calcMatrices(); + } + get minZoom() { return this._minZoom; } + set minZoom(zoom) { + if (this._minZoom === zoom) + return; + this._minZoom = zoom; + this.zoom = Math.max(this.zoom, zoom); + } + get maxZoom() { return this._maxZoom; } + set maxZoom(zoom) { + if (this._maxZoom === zoom) + return; + this._maxZoom = zoom; + this.zoom = Math.min(this.zoom, zoom); + } + get minPitch() { return this._minPitch; } + set minPitch(pitch) { + if (this._minPitch === pitch) + return; + this._minPitch = pitch; + this.pitch = Math.max(this.pitch, pitch); + } + get maxPitch() { return this._maxPitch; } + set maxPitch(pitch) { + if (this._maxPitch === pitch) + return; + this._maxPitch = pitch; + this.pitch = Math.min(this.pitch, pitch); + } + get renderWorldCopies() { return this._renderWorldCopies; } + set renderWorldCopies(renderWorldCopies) { + if (renderWorldCopies === undefined) { + renderWorldCopies = true; + } + else if (renderWorldCopies === null) { + renderWorldCopies = false; + } + this._renderWorldCopies = renderWorldCopies; + } + get worldSize() { + return this.tileSize * this.scale; + } + get centerOffset() { + return this.centerPoint._sub(this.size._div(2)); + } + get size() { + return new performance.Point(this.width, this.height); + } + get bearing() { + return -this.angle / Math.PI * 180; + } + set bearing(bearing) { + const b = -performance.wrap(bearing, -180, 180) * Math.PI / 180; + if (this.angle === b) + return; + this._unmodified = false; + this.angle = b; + this._calcMatrices(); + // 2x2 matrix for rotating points + this.rotationMatrix = performance.create$2(); + performance.rotate(this.rotationMatrix, this.rotationMatrix, this.angle); + } + get pitch() { + return this._pitch / Math.PI * 180; + } + set pitch(pitch) { + const p = performance.clamp(pitch, this.minPitch, this.maxPitch) / 180 * Math.PI; + if (this._pitch === p) + return; + this._unmodified = false; + this._pitch = p; + this._calcMatrices(); + } + get fov() { + return this._fov / Math.PI * 180; + } + set fov(fov) { + fov = Math.max(0.01, Math.min(60, fov)); + if (this._fov === fov) + return; + this._unmodified = false; + this._fov = fov / 180 * Math.PI; + this._calcMatrices(); + } + get zoom() { return this._zoom; } + set zoom(zoom) { + const constrainedZoom = Math.min(Math.max(zoom, this.minZoom), this.maxZoom); + if (this._zoom === constrainedZoom) + return; + this._unmodified = false; + this._zoom = constrainedZoom; + this.tileZoom = Math.max(0, Math.floor(constrainedZoom)); + this.scale = this.zoomScale(constrainedZoom); + this._constrain(); + this._calcMatrices(); + } + get center() { return this._center; } + set center(center) { + if (center.lat === this._center.lat && center.lng === this._center.lng) + return; + this._unmodified = false; + this._center = center; + this._constrain(); + this._calcMatrices(); + } + get elevation() { return this._elevation; } + set elevation(elevation) { + if (elevation === this._elevation) + return; + this._elevation = elevation; + this._constrain(); + this._calcMatrices(); + } + get padding() { return this._edgeInsets.toJSON(); } + set padding(padding) { + if (this._edgeInsets.equals(padding)) + return; + this._unmodified = false; + //Update edge-insets inplace + this._edgeInsets.interpolate(this._edgeInsets, padding, 1); + this._calcMatrices(); + } + /** + * The center of the screen in pixels with the top-left corner being (0,0) + * and +y axis pointing downwards. This accounts for padding. + */ + get centerPoint() { + return this._edgeInsets.getCenter(this.width, this.height); + } + /** + * Returns if the padding params match + * + * @param padding - the padding to check against + * @returns true if they are equal, false otherwise + */ + isPaddingEqual(padding) { + return this._edgeInsets.equals(padding); + } + /** + * Helper method to update edge-insets in place + * + * @param start - the starting padding + * @param target - the target padding + * @param t - the step/weight + */ + interpolatePadding(start, target, t) { + this._unmodified = false; + this._edgeInsets.interpolate(start, target, t); + this._constrain(); + this._calcMatrices(); + } + /** + * Return a zoom level that will cover all tiles the transform + * @param options - the options + * @returns zoom level An integer zoom level at which all tiles will be visible. + */ + coveringZoomLevel(options) { + const z = (options.roundZoom ? Math.round : Math.floor)(this.zoom + this.scaleZoom(this.tileSize / options.tileSize)); + // At negative zoom levels load tiles from z0 because negative tile zoom levels don't exist. + return Math.max(0, z); + } + /** + * Return any "wrapped" copies of a given tile coordinate that are visible + * in the current view. + */ + getVisibleUnwrappedCoordinates(tileID) { + const result = [new performance.UnwrappedTileID(0, tileID)]; + if (this._renderWorldCopies) { + const utl = this.pointCoordinate(new performance.Point(0, 0)); + const utr = this.pointCoordinate(new performance.Point(this.width, 0)); + const ubl = this.pointCoordinate(new performance.Point(this.width, this.height)); + const ubr = this.pointCoordinate(new performance.Point(0, this.height)); + const w0 = Math.floor(Math.min(utl.x, utr.x, ubl.x, ubr.x)); + const w1 = Math.floor(Math.max(utl.x, utr.x, ubl.x, ubr.x)); + // Add an extra copy of the world on each side to properly render ImageSources and CanvasSources. + // Both sources draw outside the tile boundaries of the tile that "contains them" so we need + // to add extra copies on both sides in case offscreen tiles need to draw into on-screen ones. + const extraWorldCopy = 1; + for (let w = w0 - extraWorldCopy; w <= w1 + extraWorldCopy; w++) { + if (w === 0) + continue; + result.push(new performance.UnwrappedTileID(w, tileID)); + } + } + return result; + } + /** + * Return all coordinates that could cover this transform for a covering + * zoom level. + * @param options - the options + * @returns OverscaledTileIDs + */ + coveringTiles(options) { + var _a, _b; + let z = this.coveringZoomLevel(options); + const actualZ = z; + if (options.minzoom !== undefined && z < options.minzoom) + return []; + if (options.maxzoom !== undefined && z > options.maxzoom) + z = options.maxzoom; + const cameraCoord = this.pointCoordinate(this.getCameraPoint()); + const centerCoord = performance.MercatorCoordinate.fromLngLat(this.center); + const numTiles = Math.pow(2, z); + const cameraPoint = [numTiles * cameraCoord.x, numTiles * cameraCoord.y, 0]; + const centerPoint = [numTiles * centerCoord.x, numTiles * centerCoord.y, 0]; + const cameraFrustum = Frustum.fromInvProjectionMatrix(this.invProjMatrix, this.worldSize, z); + // No change of LOD behavior for pitch lower than 60 and when there is no top padding: return only tile ids from the requested zoom level + let minZoom = options.minzoom || 0; + // Use 0.1 as an epsilon to avoid for explicit == 0.0 floating point checks + if (!options.terrain && this.pitch <= 60.0 && this._edgeInsets.top < 0.1) + minZoom = z; + // There should always be a certain number of maximum zoom level tiles surrounding the center location in 2D or in front of the camera in 3D + const radiusOfMaxLvlLodInTiles = options.terrain ? 2 / Math.min(this.tileSize, options.tileSize) * this.tileSize : 3; + const newRootTile = (wrap) => { + return { + aabb: new Aabb([wrap * numTiles, 0, 0], [(wrap + 1) * numTiles, numTiles, 0]), + zoom: 0, + x: 0, + y: 0, + wrap, + fullyVisible: false + }; + }; + // Do a depth-first traversal to find visible tiles and proper levels of detail + const stack = []; + const result = []; + const maxZoom = z; + const overscaledZ = options.reparseOverscaled ? actualZ : z; + if (this._renderWorldCopies) { + // Render copy of the globe thrice on both sides + for (let i = 1; i <= 3; i++) { + stack.push(newRootTile(-i)); + stack.push(newRootTile(i)); + } + } + stack.push(newRootTile(0)); + while (stack.length > 0) { + const it = stack.pop(); + const x = it.x; + const y = it.y; + let fullyVisible = it.fullyVisible; + // Visibility of a tile is not required if any of its ancestor if fully inside the frustum + if (!fullyVisible) { + const intersectResult = it.aabb.intersects(cameraFrustum); + if (intersectResult === 0) + continue; + fullyVisible = intersectResult === 2; + } + const refPoint = options.terrain ? cameraPoint : centerPoint; + const distanceX = it.aabb.distanceX(refPoint); + const distanceY = it.aabb.distanceY(refPoint); + const longestDim = Math.max(Math.abs(distanceX), Math.abs(distanceY)); + // We're using distance based heuristics to determine if a tile should be split into quadrants or not. + // radiusOfMaxLvlLodInTiles defines that there's always a certain number of maxLevel tiles next to the map center. + // Using the fact that a parent node in quadtree is twice the size of its children (per dimension) + // we can define distance thresholds for each relative level: + // f(k) = offset + 2 + 4 + 8 + 16 + ... + 2^k. This is the same as "offset+2^(k+1)-2" + const distToSplit = radiusOfMaxLvlLodInTiles + (1 << (maxZoom - it.zoom)) - 2; + // Have we reached the target depth or is the tile too far away to be any split further? + if (it.zoom === maxZoom || (longestDim > distToSplit && it.zoom >= minZoom)) { + const dz = maxZoom - it.zoom, dx = cameraPoint[0] - 0.5 - (x << dz), dy = cameraPoint[1] - 0.5 - (y << dz); + result.push({ + tileID: new performance.OverscaledTileID(it.zoom === maxZoom ? overscaledZ : it.zoom, it.wrap, it.zoom, x, y), + distanceSq: performance.sqrLen([centerPoint[0] - 0.5 - x, centerPoint[1] - 0.5 - y]), + // this variable is currently not used, but may be important to reduce the amount of loaded tiles + tileDistanceToCamera: Math.sqrt(dx * dx + dy * dy) + }); + continue; + } + for (let i = 0; i < 4; i++) { + const childX = (x << 1) + (i % 2); + const childY = (y << 1) + (i >> 1); + const childZ = it.zoom + 1; + let quadrant = it.aabb.quadrant(i); + if (options.terrain) { + const tileID = new performance.OverscaledTileID(childZ, it.wrap, childZ, childX, childY); + const minMax = options.terrain.getMinMaxElevation(tileID); + const minElevation = (_a = minMax.minElevation) !== null && _a !== void 0 ? _a : this.elevation; + const maxElevation = (_b = minMax.maxElevation) !== null && _b !== void 0 ? _b : this.elevation; + quadrant = new Aabb([quadrant.min[0], quadrant.min[1], minElevation], [quadrant.max[0], quadrant.max[1], maxElevation]); + } + stack.push({ aabb: quadrant, zoom: childZ, x: childX, y: childY, wrap: it.wrap, fullyVisible }); + } + } + return result.sort((a, b) => a.distanceSq - b.distanceSq).map(a => a.tileID); + } + resize(width, height) { + this.width = width; + this.height = height; + this.pixelsToGLUnits = [2 / width, -2 / height]; + this._constrain(); + this._calcMatrices(); + } + get unmodified() { return this._unmodified; } + zoomScale(zoom) { return Math.pow(2, zoom); } + scaleZoom(scale) { return Math.log(scale) / Math.LN2; } + project(lnglat) { + const lat = performance.clamp(lnglat.lat, -this.maxValidLatitude, this.maxValidLatitude); + return new performance.Point(performance.mercatorXfromLng(lnglat.lng) * this.worldSize, performance.mercatorYfromLat(lat) * this.worldSize); + } + unproject(point) { + return new performance.MercatorCoordinate(point.x / this.worldSize, point.y / this.worldSize).toLngLat(); + } + get point() { return this.project(this.center); } + /** + * get the camera position in LngLat and altitudes in meter + * @returns An object with lngLat & altitude. + */ + getCameraPosition() { + const lngLat = this.pointLocation(this.getCameraPoint()); + const altitude = Math.cos(this._pitch) * this.cameraToCenterDistance / this._pixelPerMeter; + return { lngLat, altitude: altitude + this.elevation }; + } + /** + * This method works in combination with freezeElevation activated. + * freezeElevtion is enabled during map-panning because during this the camera should sit in constant height. + * After panning finished, call this method to recalculate the zoomlevel for the current camera-height in current terrain. + * @param terrain - the terrain + */ + recalculateZoom(terrain) { + // find position the camera is looking on + const center = this.pointLocation(this.centerPoint, terrain); + const elevation = terrain.getElevationForLngLatZoom(center, this.tileZoom); + const deltaElevation = this.elevation - elevation; + if (!deltaElevation) + return; + // calculate mercator distance between camera & target + const cameraPosition = this.getCameraPosition(); + const camera = performance.MercatorCoordinate.fromLngLat(cameraPosition.lngLat, cameraPosition.altitude); + const target = performance.MercatorCoordinate.fromLngLat(center, elevation); + const dx = camera.x - target.x, dy = camera.y - target.y, dz = camera.z - target.z; + const distance = Math.sqrt(dx * dx + dy * dy + dz * dz); + // from this distance we calculate the new zoomlevel + const zoom = this.scaleZoom(this.cameraToCenterDistance / distance / this.tileSize); + // update matrices + this._elevation = elevation; + this._center = center; + this.zoom = zoom; + } + setLocationAtPoint(lnglat, point) { + const a = this.pointCoordinate(point); + const b = this.pointCoordinate(this.centerPoint); + const loc = this.locationCoordinate(lnglat); + const newCenter = new performance.MercatorCoordinate(loc.x - (a.x - b.x), loc.y - (a.y - b.y)); + this.center = this.coordinateLocation(newCenter); + if (this._renderWorldCopies) { + this.center = this.center.wrap(); + } + } + /** + * Given a location, return the screen point that corresponds to it + * @param lnglat - location + * @param terrain - optional terrain + * @returns screen point + */ + locationPoint(lnglat, terrain) { + return terrain ? + this.coordinatePoint(this.locationCoordinate(lnglat), terrain.getElevationForLngLatZoom(lnglat, this.tileZoom), this.pixelMatrix3D) : + this.coordinatePoint(this.locationCoordinate(lnglat)); + } + /** + * Given a point on screen, return its lnglat + * @param p - screen point + * @param terrain - optional terrain + * @returns lnglat location + */ + pointLocation(p, terrain) { + return this.coordinateLocation(this.pointCoordinate(p, terrain)); + } + /** + * Given a geographical lnglat, return an unrounded + * coordinate that represents it at this transform's zoom level. + * @param lnglat - the location + * @returns The mercator coordinate + */ + locationCoordinate(lnglat) { + return performance.MercatorCoordinate.fromLngLat(lnglat); + } + /** + * Given a Coordinate, return its geographical position. + * @param coord - mercator coordivates + * @returns lng and lat + */ + coordinateLocation(coord) { + return coord && coord.toLngLat(); + } + /** + * Given a Point, return its mercator coordinate. + * @param p - the point + * @param terrain - optional terrain + * @returns lnglat + */ + pointCoordinate(p, terrain) { + // get point-coordinate from terrain coordinates framebuffer + if (terrain) { + const coordinate = terrain.pointCoordinate(p); + if (coordinate != null) { + return coordinate; + } + } + // calculate point-coordinate on flat earth + const targetZ = 0; + // since we don't know the correct projected z value for the point, + // unproject two points to get a line and then find the point on that + // line with z=0 + const coord0 = [p.x, p.y, 0, 1]; + const coord1 = [p.x, p.y, 1, 1]; + performance.transformMat4(coord0, coord0, this.pixelMatrixInverse); + performance.transformMat4(coord1, coord1, this.pixelMatrixInverse); + const w0 = coord0[3]; + const w1 = coord1[3]; + const x0 = coord0[0] / w0; + const x1 = coord1[0] / w1; + const y0 = coord0[1] / w0; + const y1 = coord1[1] / w1; + const z0 = coord0[2] / w0; + const z1 = coord1[2] / w1; + const t = z0 === z1 ? 0 : (targetZ - z0) / (z1 - z0); + return new performance.MercatorCoordinate(performance.interpolate.number(x0, x1, t) / this.worldSize, performance.interpolate.number(y0, y1, t) / this.worldSize); + } + /** + * Given a coordinate, return the screen point that corresponds to it + * @param coord - the coordinates + * @param elevation - the elevation + * @param pixelMatrix - the pixel matrix + * @returns screen point + */ + coordinatePoint(coord, elevation = 0, pixelMatrix = this.pixelMatrix) { + const p = [coord.x * this.worldSize, coord.y * this.worldSize, elevation, 1]; + performance.transformMat4(p, p, pixelMatrix); + return new performance.Point(p[0] / p[3], p[1] / p[3]); + } + /** + * Returns the map's geographical bounds. When the bearing or pitch is non-zero, the visible region is not + * an axis-aligned rectangle, and the result is the smallest bounds that encompasses the visible region. + * @returns Returns a {@link LngLatBounds} object describing the map's geographical bounds. + */ + getBounds() { + const top = Math.max(0, this.height / 2 - this.getHorizon()); + return new LngLatBounds() + .extend(this.pointLocation(new performance.Point(0, top))) + .extend(this.pointLocation(new performance.Point(this.width, top))) + .extend(this.pointLocation(new performance.Point(this.width, this.height))) + .extend(this.pointLocation(new performance.Point(0, this.height))); + } + /** + * Returns the maximum geographical bounds the map is constrained to, or `null` if none set. + * @returns max bounds + */ + getMaxBounds() { + if (!this.latRange || this.latRange.length !== 2 || + !this.lngRange || this.lngRange.length !== 2) + return null; + return new LngLatBounds([this.lngRange[0], this.latRange[0]], [this.lngRange[1], this.latRange[1]]); + } + /** + * Calculate pixel height of the visible horizon in relation to map-center (e.g. height/2), + * multiplied by a static factor to simulate the earth-radius. + * The calculated value is the horizontal line from the camera-height to sea-level. + * @returns Horizon above center in pixels. + */ + getHorizon() { + return Math.tan(Math.PI / 2 - this._pitch) * this.cameraToCenterDistance * 0.85; + } + /** + * Sets or clears the map's geographical constraints. + * @param bounds - A {@link LngLatBounds} object describing the new geographic boundaries of the map. + */ + setMaxBounds(bounds) { + if (bounds) { + this.lngRange = [bounds.getWest(), bounds.getEast()]; + this.latRange = [bounds.getSouth(), bounds.getNorth()]; + this._constrain(); + } + else { + this.lngRange = null; + this.latRange = [-this.maxValidLatitude, this.maxValidLatitude]; + } + } + /** + * Calculate the posMatrix that, given a tile coordinate, would be used to display the tile on a map. + * @param unwrappedTileID - the tile ID + */ + calculatePosMatrix(unwrappedTileID, aligned = false) { + const posMatrixKey = unwrappedTileID.key; + const cache = aligned ? this._alignedPosMatrixCache : this._posMatrixCache; + if (cache[posMatrixKey]) { + return cache[posMatrixKey]; + } + const canonical = unwrappedTileID.canonical; + const scale = this.worldSize / this.zoomScale(canonical.z); + const unwrappedX = canonical.x + Math.pow(2, canonical.z) * unwrappedTileID.wrap; + const posMatrix = performance.identity(new Float64Array(16)); + performance.translate(posMatrix, posMatrix, [unwrappedX * scale, canonical.y * scale, 0]); + performance.scale(posMatrix, posMatrix, [scale / performance.EXTENT, scale / performance.EXTENT, 1]); + performance.multiply(posMatrix, aligned ? this.alignedProjMatrix : this.projMatrix, posMatrix); + cache[posMatrixKey] = new Float32Array(posMatrix); + return cache[posMatrixKey]; + } + customLayerMatrix() { + return this.mercatorMatrix.slice(); + } + _constrain() { + if (!this.center || !this.width || !this.height || this._constraining) + return; + this._constraining = true; + let minY = -90; + let maxY = 90; + let minX = -180; + let maxX = 180; + let sy, sx, x2, y2; + const size = this.size, unmodified = this._unmodified; + if (this.latRange) { + const latRange = this.latRange; + minY = performance.mercatorYfromLat(latRange[1]) * this.worldSize; + maxY = performance.mercatorYfromLat(latRange[0]) * this.worldSize; + sy = maxY - minY < size.y ? size.y / (maxY - minY) : 0; + } + if (this.lngRange) { + const lngRange = this.lngRange; + minX = performance.wrap(performance.mercatorXfromLng(lngRange[0]) * this.worldSize, 0, this.worldSize); + maxX = performance.wrap(performance.mercatorXfromLng(lngRange[1]) * this.worldSize, 0, this.worldSize); + if (maxX < minX) + maxX += this.worldSize; + sx = maxX - minX < size.x ? size.x / (maxX - minX) : 0; + } + const point = this.point; + // how much the map should scale to fit the screen into given latitude/longitude ranges + const s = Math.max(sx || 0, sy || 0); + if (s) { + this.center = this.unproject(new performance.Point(sx ? (maxX + minX) / 2 : point.x, sy ? (maxY + minY) / 2 : point.y)); + this.zoom += this.scaleZoom(s); + this._unmodified = unmodified; + this._constraining = false; + return; + } + if (this.latRange) { + const y = point.y, h2 = size.y / 2; + if (y - h2 < minY) + y2 = minY + h2; + if (y + h2 > maxY) + y2 = maxY - h2; + } + if (this.lngRange) { + const centerX = (minX + maxX) / 2; + const x = performance.wrap(point.x, centerX - this.worldSize / 2, centerX + this.worldSize / 2); + const w2 = size.x / 2; + if (x - w2 < minX) + x2 = minX + w2; + if (x + w2 > maxX) + x2 = maxX - w2; + } + // pan the map if the screen goes off the range + if (x2 !== undefined || y2 !== undefined) { + this.center = this.unproject(new performance.Point(x2 !== undefined ? x2 : point.x, y2 !== undefined ? y2 : point.y)).wrap(); + } + this._unmodified = unmodified; + this._constraining = false; + } + _calcMatrices() { + if (!this.height) + return; + const halfFov = this._fov / 2; + const offset = this.centerOffset; + const x = this.point.x, y = this.point.y; + this.cameraToCenterDistance = 0.5 / Math.tan(halfFov) * this.height; + this._pixelPerMeter = performance.mercatorZfromAltitude(1, this.center.lat) * this.worldSize; + let m = performance.identity(new Float64Array(16)); + performance.scale(m, m, [this.width / 2, -this.height / 2, 1]); + performance.translate(m, m, [1, -1, 0]); + this.labelPlaneMatrix = m; + m = performance.identity(new Float64Array(16)); + performance.scale(m, m, [1, -1, 1]); + performance.translate(m, m, [-1, -1, 0]); + performance.scale(m, m, [2 / this.width, 2 / this.height, 1]); + this.glCoordMatrix = m; + // Calculate the camera to sea-level distance in pixel in respect of terrain + const cameraToSeaLevelDistance = this.cameraToCenterDistance + this._elevation * this._pixelPerMeter / Math.cos(this._pitch); + // In case of negative minimum elevation (e.g. the dead see, under the sea maps) use a lower plane for calculation + const minElevation = Math.min(this.elevation, this._minEleveationForCurrentTile); + const cameraToLowestPointDistance = cameraToSeaLevelDistance - minElevation * this._pixelPerMeter / Math.cos(this._pitch); + const lowestPlane = minElevation < 0 ? cameraToLowestPointDistance : cameraToSeaLevelDistance; + // Find the distance from the center point [width/2 + offset.x, height/2 + offset.y] to the + // center top point [width/2 + offset.x, 0] in Z units, using the law of sines. + // 1 Z unit is equivalent to 1 horizontal px at the center of the map + // (the distance between[width/2, height/2] and [width/2 + 1, height/2]) + const groundAngle = Math.PI / 2 + this._pitch; + const fovAboveCenter = this._fov * (0.5 + offset.y / this.height); + const topHalfSurfaceDistance = Math.sin(fovAboveCenter) * lowestPlane / Math.sin(performance.clamp(Math.PI - groundAngle - fovAboveCenter, 0.01, Math.PI - 0.01)); + // Find the distance from the center point to the horizon + const horizon = this.getHorizon(); + const horizonAngle = Math.atan(horizon / this.cameraToCenterDistance); + const fovCenterToHorizon = 2 * horizonAngle * (0.5 + offset.y / (horizon * 2)); + const topHalfSurfaceDistanceHorizon = Math.sin(fovCenterToHorizon) * lowestPlane / Math.sin(performance.clamp(Math.PI - groundAngle - fovCenterToHorizon, 0.01, Math.PI - 0.01)); + // Calculate z distance of the farthest fragment that should be rendered. + // Add a bit extra to avoid precision problems when a fragment's distance is exactly `furthestDistance` + const topHalfMinDistance = Math.min(topHalfSurfaceDistance, topHalfSurfaceDistanceHorizon); + const farZ = (Math.cos(Math.PI / 2 - this._pitch) * topHalfMinDistance + lowestPlane) * 1.01; + // The larger the value of nearZ is + // - the more depth precision is available for features (good) + // - clipping starts appearing sooner when the camera is close to 3d features (bad) + // + // Smaller values worked well for mapbox-gl-js but deckgl was encountering precision issues + // when rendering it's layers using custom layers. This value was experimentally chosen and + // seems to solve z-fighting issues in deckgl while not clipping buildings too close to the camera. + const nearZ = this.height / 50; + // matrix for conversion from location to GL coordinates (-1 .. 1) + m = new Float64Array(16); + performance.perspective(m, this._fov, this.width / this.height, nearZ, farZ); + // Apply center of perspective offset + m[8] = -offset.x * 2 / this.width; + m[9] = offset.y * 2 / this.height; + performance.scale(m, m, [1, -1, 1]); + performance.translate(m, m, [0, 0, -this.cameraToCenterDistance]); + performance.rotateX(m, m, this._pitch); + performance.rotateZ(m, m, this.angle); + performance.translate(m, m, [-x, -y, 0]); + // The mercatorMatrix can be used to transform points from mercator coordinates + // ([0, 0] nw, [1, 1] se) to GL coordinates. + this.mercatorMatrix = performance.scale([], m, [this.worldSize, this.worldSize, this.worldSize]); + // scale vertically to meters per pixel (inverse of ground resolution): + performance.scale(m, m, [1, 1, this._pixelPerMeter]); + // matrix for conversion from location to screen coordinates in 2D + this.pixelMatrix = performance.multiply(new Float64Array(16), this.labelPlaneMatrix, m); + // matrix for conversion from location to GL coordinates (-1 .. 1) + performance.translate(m, m, [0, 0, -this.elevation]); // elevate camera over terrain + this.projMatrix = m; + this.invProjMatrix = performance.invert([], m); + // matrix for conversion from location to screen coordinates in 2D + this.pixelMatrix3D = performance.multiply(new Float64Array(16), this.labelPlaneMatrix, m); + // Make a second projection matrix that is aligned to a pixel grid for rendering raster tiles. + // We're rounding the (floating point) x/y values to achieve to avoid rendering raster images to fractional + // coordinates. Additionally, we adjust by half a pixel in either direction in case that viewport dimension + // is an odd integer to preserve rendering to the pixel grid. We're rotating this shift based on the angle + // of the transformation so that 0°, 90°, 180°, and 270° rasters are crisp, and adjust the shift so that + // it is always <= 0.5 pixels. + const xShift = (this.width % 2) / 2, yShift = (this.height % 2) / 2, angleCos = Math.cos(this.angle), angleSin = Math.sin(this.angle), dx = x - Math.round(x) + angleCos * xShift + angleSin * yShift, dy = y - Math.round(y) + angleCos * yShift + angleSin * xShift; + const alignedM = new Float64Array(m); + performance.translate(alignedM, alignedM, [dx > 0.5 ? dx - 1 : dx, dy > 0.5 ? dy - 1 : dy, 0]); + this.alignedProjMatrix = alignedM; + // inverse matrix for conversion from screen coordinaes to location + m = performance.invert(new Float64Array(16), this.pixelMatrix); + if (!m) + throw new Error('failed to invert matrix'); + this.pixelMatrixInverse = m; + this._posMatrixCache = {}; + this._alignedPosMatrixCache = {}; + } + maxPitchScaleFactor() { + // calcMatrices hasn't run yet + if (!this.pixelMatrixInverse) + return 1; + const coord = this.pointCoordinate(new performance.Point(0, 0)); + const p = [coord.x * this.worldSize, coord.y * this.worldSize, 0, 1]; + const topPoint = performance.transformMat4(p, p, this.pixelMatrix); + return topPoint[3] / this.cameraToCenterDistance; + } + /** + * The camera looks at the map from a 3D (lng, lat, altitude) location. Let's use `cameraLocation` + * as the name for the location under the camera and on the surface of the earth (lng, lat, 0). + * `cameraPoint` is the projected position of the `cameraLocation`. + * + * This point is useful to us because only fill-extrusions that are between `cameraPoint` and + * the query point on the surface of the earth can extend and intersect the query. + * + * When the map is not pitched the `cameraPoint` is equivalent to the center of the map because + * the camera is right above the center of the map. + */ + getCameraPoint() { + const pitch = this._pitch; + const yOffset = Math.tan(pitch) * (this.cameraToCenterDistance || 1); + return this.centerPoint.add(new performance.Point(0, yOffset)); + } + /** + * When the map is pitched, some of the 3D features that intersect a query will not intersect + * the query at the surface of the earth. Instead the feature may be closer and only intersect + * the query because it extrudes into the air. + * @param queryGeometry - For point queries, the line from the query point to the "camera point", + * for other geometries, the envelope of the query geometry and the "camera point" + * @returns a geometry that includes all of the original query as well as all possible ares of the + * screen where the *base* of a visible extrusion could be. + * + */ + getCameraQueryGeometry(queryGeometry) { + const c = this.getCameraPoint(); + if (queryGeometry.length === 1) { + return [queryGeometry[0], c]; + } + else { + let minX = c.x; + let minY = c.y; + let maxX = c.x; + let maxY = c.y; + for (const p of queryGeometry) { + minX = Math.min(minX, p.x); + minY = Math.min(minY, p.y); + maxX = Math.max(maxX, p.x); + maxY = Math.max(maxY, p.y); + } + return [ + new performance.Point(minX, minY), + new performance.Point(maxX, minY), + new performance.Point(maxX, maxY), + new performance.Point(minX, maxY), + new performance.Point(minX, minY) + ]; + } + } +} + +/** + * Throttle the given function to run at most every `period` milliseconds. + */ +function throttle(fn, time) { + let pending = false; + let timerId = null; + let lastCallContext = null; + let lastCallArgs; + const later = () => { + timerId = null; + if (pending) { + fn.apply(lastCallContext, lastCallArgs); + timerId = setTimeout(later, time); + pending = false; + } + }; + return (...args) => { + pending = true; + lastCallContext = this; + lastCallArgs = args; + if (!timerId) { + later(); + } + return timerId; + }; +} + +/** + * Adds the map's position to its page's location hash. + * Passed as an option to the map object. + * + * @group Markers and Controls + */ +class Hash { + constructor(hashName) { + this._getCurrentHash = () => { + // Get the current hash from location, stripped from its number sign + const hash = window.location.hash.replace('#', ''); + if (this._hashName) { + // Split the parameter-styled hash into parts and find the value we need + let keyval; + hash.split('&').map(part => part.split('=')).forEach(part => { + if (part[0] === this._hashName) { + keyval = part; + } + }); + return (keyval ? keyval[1] || '' : '').split('/'); + } + return hash.split('/'); + }; + this._onHashChange = () => { + const loc = this._getCurrentHash(); + if (loc.length >= 3 && !loc.some(v => isNaN(v))) { + const bearing = this._map.dragRotate.isEnabled() && this._map.touchZoomRotate.isEnabled() ? +(loc[3] || 0) : this._map.getBearing(); + this._map.jumpTo({ + center: [+loc[2], +loc[1]], + zoom: +loc[0], + bearing, + pitch: +(loc[4] || 0) + }); + return true; + } + return false; + }; + this._updateHashUnthrottled = () => { + // Replace if already present, else append the updated hash string + const location = window.location.href.replace(/(#.+)?$/, this.getHashString()); + try { + window.history.replaceState(window.history.state, null, location); + } + catch (SecurityError) { + // IE11 does not allow this if the page is within an iframe created + // with iframe.contentWindow.document.write(...). + // https://github.com/mapbox/mapbox-gl-js/issues/7410 + } + }; + /** + * Mobile Safari doesn't allow updating the hash more than 100 times per 30 seconds. + */ + this._updateHash = throttle(this._updateHashUnthrottled, 30 * 1000 / 100); + this._hashName = hashName && encodeURIComponent(hashName); + } + /** + * Map element to listen for coordinate changes + * + * @param map - The map object + * @returns `this` + */ + addTo(map) { + this._map = map; + addEventListener('hashchange', this._onHashChange, false); + this._map.on('moveend', this._updateHash); + return this; + } + /** + * Removes hash + * + * @returns `this` + */ + remove() { + removeEventListener('hashchange', this._onHashChange, false); + this._map.off('moveend', this._updateHash); + clearTimeout(this._updateHash()); + delete this._map; + return this; + } + getHashString(mapFeedback) { + const center = this._map.getCenter(), zoom = Math.round(this._map.getZoom() * 100) / 100, + // derived from equation: 512px * 2^z / 360 / 10^d < 0.5px + precision = Math.ceil((zoom * Math.LN2 + Math.log(512 / 360 / 0.5)) / Math.LN10), m = Math.pow(10, precision), lng = Math.round(center.lng * m) / m, lat = Math.round(center.lat * m) / m, bearing = this._map.getBearing(), pitch = this._map.getPitch(); + let hash = ''; + if (mapFeedback) { + // new map feedback site has some constraints that don't allow + // us to use the same hash format as we do for the Map hash option. + hash += `/${lng}/${lat}/${zoom}`; + } + else { + hash += `${zoom}/${lat}/${lng}`; + } + if (bearing || pitch) + hash += (`/${Math.round(bearing * 10) / 10}`); + if (pitch) + hash += (`/${Math.round(pitch)}`); + if (this._hashName) { + const hashName = this._hashName; + let found = false; + const parts = window.location.hash.slice(1).split('&').map(part => { + const key = part.split('=')[0]; + if (key === hashName) { + found = true; + return `${key}=${hash}`; + } + return part; + }).filter(a => a); + if (!found) { + parts.push(`${hashName}=${hash}`); + } + return `#${parts.join('&')}`; + } + return `#${hash}`; + } +} + +const defaultInertiaOptions = { + linearity: 0.3, + easing: performance.bezier(0, 0, 0.3, 1), +}; +const defaultPanInertiaOptions = performance.extend({ + deceleration: 2500, + maxSpeed: 1400 +}, defaultInertiaOptions); +const defaultZoomInertiaOptions = performance.extend({ + deceleration: 20, + maxSpeed: 1400 +}, defaultInertiaOptions); +const defaultBearingInertiaOptions = performance.extend({ + deceleration: 1000, + maxSpeed: 360 +}, defaultInertiaOptions); +const defaultPitchInertiaOptions = performance.extend({ + deceleration: 1000, + maxSpeed: 90 +}, defaultInertiaOptions); +class HandlerInertia { + constructor(map) { + this._map = map; + this.clear(); + } + clear() { + this._inertiaBuffer = []; + } + record(settings) { + this._drainInertiaBuffer(); + this._inertiaBuffer.push({ time: performance.browser.now(), settings }); + } + _drainInertiaBuffer() { + const inertia = this._inertiaBuffer, now = performance.browser.now(), cutoff = 160; //msec + while (inertia.length > 0 && now - inertia[0].time > cutoff) + inertia.shift(); + } + _onMoveEnd(panInertiaOptions) { + this._drainInertiaBuffer(); + if (this._inertiaBuffer.length < 2) { + return; + } + const deltas = { + zoom: 0, + bearing: 0, + pitch: 0, + pan: new performance.Point(0, 0), + pinchAround: undefined, + around: undefined + }; + for (const { settings } of this._inertiaBuffer) { + deltas.zoom += settings.zoomDelta || 0; + deltas.bearing += settings.bearingDelta || 0; + deltas.pitch += settings.pitchDelta || 0; + if (settings.panDelta) + deltas.pan._add(settings.panDelta); + if (settings.around) + deltas.around = settings.around; + if (settings.pinchAround) + deltas.pinchAround = settings.pinchAround; + } + const lastEntry = this._inertiaBuffer[this._inertiaBuffer.length - 1]; + const duration = (lastEntry.time - this._inertiaBuffer[0].time); + const easeOptions = {}; + if (deltas.pan.mag()) { + const result = calculateEasing(deltas.pan.mag(), duration, performance.extend({}, defaultPanInertiaOptions, panInertiaOptions || {})); + easeOptions.offset = deltas.pan.mult(result.amount / deltas.pan.mag()); + easeOptions.center = this._map.transform.center; + extendDuration(easeOptions, result); + } + if (deltas.zoom) { + const result = calculateEasing(deltas.zoom, duration, defaultZoomInertiaOptions); + easeOptions.zoom = this._map.transform.zoom + result.amount; + extendDuration(easeOptions, result); + } + if (deltas.bearing) { + const result = calculateEasing(deltas.bearing, duration, defaultBearingInertiaOptions); + easeOptions.bearing = this._map.transform.bearing + performance.clamp(result.amount, -179, 179); + extendDuration(easeOptions, result); + } + if (deltas.pitch) { + const result = calculateEasing(deltas.pitch, duration, defaultPitchInertiaOptions); + easeOptions.pitch = this._map.transform.pitch + result.amount; + extendDuration(easeOptions, result); + } + if (easeOptions.zoom || easeOptions.bearing) { + const last = deltas.pinchAround === undefined ? deltas.around : deltas.pinchAround; + easeOptions.around = last ? this._map.unproject(last) : this._map.getCenter(); + } + this.clear(); + return performance.extend(easeOptions, { + noMoveStart: true + }); + } +} +// Unfortunately zoom, bearing, etc can't have different durations and easings so +// we need to choose one. We use the longest duration and it's corresponding easing. +function extendDuration(easeOptions, result) { + if (!easeOptions.duration || easeOptions.duration < result.duration) { + easeOptions.duration = result.duration; + easeOptions.easing = result.easing; + } +} +function calculateEasing(amount, inertiaDuration, inertiaOptions) { + const { maxSpeed, linearity, deceleration } = inertiaOptions; + const speed = performance.clamp(amount * linearity / (inertiaDuration / 1000), -maxSpeed, maxSpeed); + const duration = Math.abs(speed) / (deceleration * linearity); + return { + easing: inertiaOptions.easing, + duration: duration * 1000, + amount: speed * (duration / 2) + }; +} + +/** + * `MapMouseEvent` is the event type for mouse-related map events. + * @example + * ```ts + * // The `click` event is an example of a `MapMouseEvent`. + * // Set up an event listener on the map. + * map.on('click', function(e) { + * // The event object (e) contains information like the + * // coordinates of the point on the map that was clicked. + * console.log('A click event has occurred at ' + e.lngLat); + * }); + * ``` + */ +class MapMouseEvent extends performance.Event { + /** + * Prevents subsequent default processing of the event by the map. + * + * Calling this method will prevent the following default map behaviors: + * + * * On `mousedown` events, the behavior of {@link DragPanHandler} + * * On `mousedown` events, the behavior of {@link DragRotateHandler} + * * On `mousedown` events, the behavior of {@link BoxZoomHandler} + * * On `dblclick` events, the behavior of {@link DoubleClickZoomHandler} + * + */ + preventDefault() { + this._defaultPrevented = true; + } + /** + * `true` if `preventDefault` has been called. + */ + get defaultPrevented() { + return this._defaultPrevented; + } + constructor(type, map, originalEvent, data = {}) { + const point = DOM.mousePos(map.getCanvasContainer(), originalEvent); + const lngLat = map.unproject(point); + super(type, performance.extend({ point, lngLat, originalEvent }, data)); + this._defaultPrevented = false; + this.target = map; + } +} +/** + * `MapTouchEvent` is the event type for touch-related map events. + * + * @group Event Related + */ +class MapTouchEvent extends performance.Event { + /** + * Prevents subsequent default processing of the event by the map. + * + * Calling this method will prevent the following default map behaviors: + * + * * On `touchstart` events, the behavior of {@link DragPanHandler} + * * On `touchstart` events, the behavior of {@link TwoFingersTouchZoomRotateHandler} + * + */ + preventDefault() { + this._defaultPrevented = true; + } + /** + * `true` if `preventDefault` has been called. + */ + get defaultPrevented() { + return this._defaultPrevented; + } + constructor(type, map, originalEvent) { + const touches = type === 'touchend' ? originalEvent.changedTouches : originalEvent.touches; + const points = DOM.touchPos(map.getCanvasContainer(), touches); + const lngLats = points.map((t) => map.unproject(t)); + const point = points.reduce((prev, curr, i, arr) => { + return prev.add(curr.div(arr.length)); + }, new performance.Point(0, 0)); + const lngLat = map.unproject(point); + super(type, { points, point, lngLats, lngLat, originalEvent }); + this._defaultPrevented = false; + } +} +/** + * `MapWheelEvent` is the event type for the `wheel` map event. + * + * @group Event Related + */ +class MapWheelEvent extends performance.Event { + /** + * Prevents subsequent default processing of the event by the map. + * + * Calling this method will prevent the behavior of {@link ScrollZoomHandler}. + */ + preventDefault() { + this._defaultPrevented = true; + } + /** + * `true` if `preventDefault` has been called. + */ + get defaultPrevented() { + return this._defaultPrevented; + } + /** */ + constructor(type, map, originalEvent) { + super(type, { originalEvent }); + this._defaultPrevented = false; + } +} + +class MapEventHandler { + constructor(map, options) { + this._map = map; + this._clickTolerance = options.clickTolerance; + } + reset() { + delete this._mousedownPos; + } + wheel(e) { + // If mapEvent.preventDefault() is called by the user, prevent handlers such as: + // - ScrollZoom + return this._firePreventable(new MapWheelEvent(e.type, this._map, e)); + } + mousedown(e, point) { + this._mousedownPos = point; + // If mapEvent.preventDefault() is called by the user, prevent handlers such as: + // - MousePan + // - MouseRotate + // - MousePitch + // - DblclickHandler + return this._firePreventable(new MapMouseEvent(e.type, this._map, e)); + } + mouseup(e) { + this._map.fire(new MapMouseEvent(e.type, this._map, e)); + } + click(e, point) { + if (this._mousedownPos && this._mousedownPos.dist(point) >= this._clickTolerance) + return; + this._map.fire(new MapMouseEvent(e.type, this._map, e)); + } + dblclick(e) { + // If mapEvent.preventDefault() is called by the user, prevent handlers such as: + // - DblClickZoom + return this._firePreventable(new MapMouseEvent(e.type, this._map, e)); + } + mouseover(e) { + this._map.fire(new MapMouseEvent(e.type, this._map, e)); + } + mouseout(e) { + this._map.fire(new MapMouseEvent(e.type, this._map, e)); + } + touchstart(e) { + // If mapEvent.preventDefault() is called by the user, prevent handlers such as: + // - TouchPan + // - TouchZoom + // - TouchRotate + // - TouchPitch + // - TapZoom + // - SwipeZoom + return this._firePreventable(new MapTouchEvent(e.type, this._map, e)); + } + touchmove(e) { + this._map.fire(new MapTouchEvent(e.type, this._map, e)); + } + touchend(e) { + this._map.fire(new MapTouchEvent(e.type, this._map, e)); + } + touchcancel(e) { + this._map.fire(new MapTouchEvent(e.type, this._map, e)); + } + _firePreventable(mapEvent) { + this._map.fire(mapEvent); + if (mapEvent.defaultPrevented) { + // returning an object marks the handler as active and resets other handlers + return {}; + } + } + isEnabled() { + return true; + } + isActive() { + return false; + } + enable() { } + disable() { } +} +class BlockableMapEventHandler { + constructor(map) { + this._map = map; + } + reset() { + this._delayContextMenu = false; + this._ignoreContextMenu = true; + delete this._contextMenuEvent; + } + mousemove(e) { + // mousemove map events should not be fired when interaction handlers (pan, rotate, etc) are active + this._map.fire(new MapMouseEvent(e.type, this._map, e)); + } + mousedown() { + this._delayContextMenu = true; + this._ignoreContextMenu = false; + } + mouseup() { + this._delayContextMenu = false; + if (this._contextMenuEvent) { + this._map.fire(new MapMouseEvent('contextmenu', this._map, this._contextMenuEvent)); + delete this._contextMenuEvent; + } + } + contextmenu(e) { + if (this._delayContextMenu) { + // Mac: contextmenu fired on mousedown; we save it until mouseup for consistency's sake + this._contextMenuEvent = e; + } + else if (!this._ignoreContextMenu) { + // Windows: contextmenu fired on mouseup, so fire event now + this._map.fire(new MapMouseEvent(e.type, this._map, e)); + } + // prevent browser context menu when necessary + if (this._map.listens('contextmenu')) { + e.preventDefault(); + } + } + isEnabled() { + return true; + } + isActive() { + return false; + } + enable() { } + disable() { } +} + +/** + * @internal + * Shared utilities for the Handler classes to access the correct camera state. + * If Camera.transformCameraUpdate is specified, the "desired state" of camera may differ from the state used for rendering. + * The handlers need the "desired state" to track accumulated changes. + */ +class TransformProvider { + constructor(map) { + this._map = map; + } + get transform() { + return this._map._requestedCameraState || this._map.transform; + } + get center() { + return { lng: this.transform.center.lng, lat: this.transform.center.lat }; + } + get zoom() { + return this.transform.zoom; + } + get pitch() { + return this.transform.pitch; + } + get bearing() { + return this.transform.bearing; + } + unproject(point) { + return this.transform.pointLocation(performance.Point.convert(point), this._map.terrain); + } +} + +/** + * The `BoxZoomHandler` allows the user to zoom the map to fit within a bounding box. + * The bounding box is defined by clicking and holding `shift` while dragging the cursor. + * + * @group Handlers + */ +class BoxZoomHandler { + /** @internal */ + constructor(map, options) { + this._map = map; + this._tr = new TransformProvider(map); + this._el = map.getCanvasContainer(); + this._container = map.getContainer(); + this._clickTolerance = options.clickTolerance || 1; + } + /** + * Returns a Boolean indicating whether the "box zoom" interaction is enabled. + * + * @returns `true` if the "box zoom" interaction is enabled. + */ + isEnabled() { + return !!this._enabled; + } + /** + * Returns a Boolean indicating whether the "box zoom" interaction is active, i.e. currently being used. + * + * @returns `true` if the "box zoom" interaction is active. + */ + isActive() { + return !!this._active; + } + /** + * Enables the "box zoom" interaction. + * + * @example + * ```ts + * map.boxZoom.enable(); + * ``` + */ + enable() { + if (this.isEnabled()) + return; + this._enabled = true; + } + /** + * Disables the "box zoom" interaction. + * + * @example + * ```ts + * map.boxZoom.disable(); + * ``` + */ + disable() { + if (!this.isEnabled()) + return; + this._enabled = false; + } + mousedown(e, point) { + if (!this.isEnabled()) + return; + if (!(e.shiftKey && e.button === 0)) + return; + DOM.disableDrag(); + this._startPos = this._lastPos = point; + this._active = true; + } + mousemoveWindow(e, point) { + if (!this._active) + return; + const pos = point; + if (this._lastPos.equals(pos) || (!this._box && pos.dist(this._startPos) < this._clickTolerance)) { + return; + } + const p0 = this._startPos; + this._lastPos = pos; + if (!this._box) { + this._box = DOM.create('div', 'maplibregl-boxzoom', this._container); + this._container.classList.add('maplibregl-crosshair'); + this._fireEvent('boxzoomstart', e); + } + const minX = Math.min(p0.x, pos.x), maxX = Math.max(p0.x, pos.x), minY = Math.min(p0.y, pos.y), maxY = Math.max(p0.y, pos.y); + DOM.setTransform(this._box, `translate(${minX}px,${minY}px)`); + this._box.style.width = `${maxX - minX}px`; + this._box.style.height = `${maxY - minY}px`; + } + mouseupWindow(e, point) { + if (!this._active) + return; + if (e.button !== 0) + return; + const p0 = this._startPos, p1 = point; + this.reset(); + DOM.suppressClick(); + if (p0.x === p1.x && p0.y === p1.y) { + this._fireEvent('boxzoomcancel', e); + } + else { + this._map.fire(new performance.Event('boxzoomend', { originalEvent: e })); + return { + cameraAnimation: map => map.fitScreenCoordinates(p0, p1, this._tr.bearing, { linear: true }) + }; + } + } + keydown(e) { + if (!this._active) + return; + if (e.keyCode === 27) { + this.reset(); + this._fireEvent('boxzoomcancel', e); + } + } + reset() { + this._active = false; + this._container.classList.remove('maplibregl-crosshair'); + if (this._box) { + DOM.remove(this._box); + this._box = null; + } + DOM.enableDrag(); + delete this._startPos; + delete this._lastPos; + } + _fireEvent(type, e) { + return this._map.fire(new performance.Event(type, { originalEvent: e })); + } +} + +function indexTouches(touches, points) { + if (touches.length !== points.length) + throw new Error(`The number of touches and points are not equal - touches ${touches.length}, points ${points.length}`); + const obj = {}; + for (let i = 0; i < touches.length; i++) { + obj[touches[i].identifier] = points[i]; + } + return obj; +} + +function getCentroid(points) { + const sum = new performance.Point(0, 0); + for (const point of points) { + sum._add(point); + } + return sum.div(points.length); +} +const MAX_TAP_INTERVAL = 500; +const MAX_TOUCH_TIME = 500; +const MAX_DIST = 30; +class SingleTapRecognizer { + constructor(options) { + this.reset(); + this.numTouches = options.numTouches; + } + reset() { + delete this.centroid; + delete this.startTime; + delete this.touches; + this.aborted = false; + } + touchstart(e, points, mapTouches) { + if (this.centroid || mapTouches.length > this.numTouches) { + this.aborted = true; + } + if (this.aborted) { + return; + } + if (this.startTime === undefined) { + this.startTime = e.timeStamp; + } + if (mapTouches.length === this.numTouches) { + this.centroid = getCentroid(points); + this.touches = indexTouches(mapTouches, points); + } + } + touchmove(e, points, mapTouches) { + if (this.aborted || !this.centroid) + return; + const newTouches = indexTouches(mapTouches, points); + for (const id in this.touches) { + const prevPos = this.touches[id]; + const pos = newTouches[id]; + if (!pos || pos.dist(prevPos) > MAX_DIST) { + this.aborted = true; + } + } + } + touchend(e, points, mapTouches) { + if (!this.centroid || e.timeStamp - this.startTime > MAX_TOUCH_TIME) { + this.aborted = true; + } + if (mapTouches.length === 0) { + const centroid = !this.aborted && this.centroid; + this.reset(); + if (centroid) + return centroid; + } + } +} +class TapRecognizer { + constructor(options) { + this.singleTap = new SingleTapRecognizer(options); + this.numTaps = options.numTaps; + this.reset(); + } + reset() { + this.lastTime = Infinity; + delete this.lastTap; + this.count = 0; + this.singleTap.reset(); + } + touchstart(e, points, mapTouches) { + this.singleTap.touchstart(e, points, mapTouches); + } + touchmove(e, points, mapTouches) { + this.singleTap.touchmove(e, points, mapTouches); + } + touchend(e, points, mapTouches) { + const tap = this.singleTap.touchend(e, points, mapTouches); + if (tap) { + const soonEnough = e.timeStamp - this.lastTime < MAX_TAP_INTERVAL; + const closeEnough = !this.lastTap || this.lastTap.dist(tap) < MAX_DIST; + if (!soonEnough || !closeEnough) { + this.reset(); + } + this.count++; + this.lastTime = e.timeStamp; + this.lastTap = tap; + if (this.count === this.numTaps) { + this.reset(); + return tap; + } + } + } +} + +class TapZoomHandler { + constructor(map) { + this._tr = new TransformProvider(map); + this._zoomIn = new TapRecognizer({ + numTouches: 1, + numTaps: 2 + }); + this._zoomOut = new TapRecognizer({ + numTouches: 2, + numTaps: 1 + }); + this.reset(); + } + reset() { + this._active = false; + this._zoomIn.reset(); + this._zoomOut.reset(); + } + touchstart(e, points, mapTouches) { + this._zoomIn.touchstart(e, points, mapTouches); + this._zoomOut.touchstart(e, points, mapTouches); + } + touchmove(e, points, mapTouches) { + this._zoomIn.touchmove(e, points, mapTouches); + this._zoomOut.touchmove(e, points, mapTouches); + } + touchend(e, points, mapTouches) { + const zoomInPoint = this._zoomIn.touchend(e, points, mapTouches); + const zoomOutPoint = this._zoomOut.touchend(e, points, mapTouches); + const tr = this._tr; + if (zoomInPoint) { + this._active = true; + e.preventDefault(); + setTimeout(() => this.reset(), 0); + return { + cameraAnimation: (map) => map.easeTo({ + duration: 300, + zoom: tr.zoom + 1, + around: tr.unproject(zoomInPoint) + }, { originalEvent: e }) + }; + } + else if (zoomOutPoint) { + this._active = true; + e.preventDefault(); + setTimeout(() => this.reset(), 0); + return { + cameraAnimation: (map) => map.easeTo({ + duration: 300, + zoom: tr.zoom - 1, + around: tr.unproject(zoomOutPoint) + }, { originalEvent: e }) + }; + } + } + touchcancel() { + this.reset(); + } + enable() { + this._enabled = true; + } + disable() { + this._enabled = false; + this.reset(); + } + isEnabled() { + return this._enabled; + } + isActive() { + return this._active; + } +} + +/** + * A generic class to create handlers for drag events, from both mouse and touch events. + */ +class DragHandler { + constructor(options) { + this._enabled = !!options.enable; + this._moveStateManager = options.moveStateManager; + this._clickTolerance = options.clickTolerance || 1; + this._moveFunction = options.move; + this._activateOnStart = !!options.activateOnStart; + options.assignEvents(this); + this.reset(); + } + reset(e) { + this._active = false; + this._moved = false; + delete this._lastPoint; + this._moveStateManager.endMove(e); + } + _move(...params) { + const move = this._moveFunction(...params); + if (move.bearingDelta || move.pitchDelta || move.around || move.panDelta) { + this._active = true; + return move; + } + } + dragStart(e, point) { + if (!this.isEnabled() || this._lastPoint) + return; + if (!this._moveStateManager.isValidStartEvent(e)) + return; + this._moveStateManager.startMove(e); + this._lastPoint = point['length'] ? point[0] : point; + if (this._activateOnStart && this._lastPoint) + this._active = true; + } + dragMove(e, point) { + if (!this.isEnabled()) + return; + const lastPoint = this._lastPoint; + if (!lastPoint) + return; + e.preventDefault(); + if (!this._moveStateManager.isValidMoveEvent(e)) { + this.reset(e); + return; + } + const movePoint = point['length'] ? point[0] : point; + if (!this._moved && movePoint.dist(lastPoint) < this._clickTolerance) + return; + this._moved = true; + this._lastPoint = movePoint; + return this._move(lastPoint, movePoint); + } + dragEnd(e) { + if (!this.isEnabled() || !this._lastPoint) + return; + if (!this._moveStateManager.isValidEndEvent(e)) + return; + if (this._moved) + DOM.suppressClick(); + this.reset(e); + } + enable() { + this._enabled = true; + } + disable() { + this._enabled = false; + this.reset(); + } + isEnabled() { + return this._enabled; + } + isActive() { + return this._active; + } + getClickTolerance() { + return this._clickTolerance; + } +} + +const LEFT_BUTTON$1 = 0; +const RIGHT_BUTTON$1 = 2; +// the values for each button in MouseEvent.buttons +const BUTTONS_FLAGS = { + [LEFT_BUTTON$1]: 1, + [RIGHT_BUTTON$1]: 2 +}; +function buttonNoLongerPressed(e, button) { + const flag = BUTTONS_FLAGS[button]; + return e.buttons === undefined || (e.buttons & flag) !== flag; +} +class MouseMoveStateManager { + constructor(options) { + this._correctEvent = options.checkCorrectEvent; + } + startMove(e) { + const eventButton = DOM.mouseButton(e); + this._eventButton = eventButton; + } + endMove(_e) { + delete this._eventButton; + } + isValidStartEvent(e) { + return this._correctEvent(e); + } + isValidMoveEvent(e) { + // Some browsers don't fire a `mouseup` when the mouseup occurs outside + // the window or iframe: + // https://github.com/mapbox/mapbox-gl-js/issues/4622 + // + // If the button is no longer pressed during this `mousemove` it may have + // been released outside of the window or iframe. + return !buttonNoLongerPressed(e, this._eventButton); + } + isValidEndEvent(e) { + const eventButton = DOM.mouseButton(e); + return eventButton === this._eventButton; + } +} +class OneFingerTouchMoveStateManager { + constructor() { + this._firstTouch = undefined; + } + _isOneFingerTouch(e) { + return e.targetTouches.length === 1; + } + _isSameTouchEvent(e) { + return e.targetTouches[0].identifier === this._firstTouch; + } + startMove(e) { + const firstTouch = e.targetTouches[0].identifier; + this._firstTouch = firstTouch; + } + endMove(_e) { + delete this._firstTouch; + } + isValidStartEvent(e) { + return this._isOneFingerTouch(e); + } + isValidMoveEvent(e) { + return this._isOneFingerTouch(e) && this._isSameTouchEvent(e); + } + isValidEndEvent(e) { + return this._isOneFingerTouch(e) && this._isSameTouchEvent(e); + } +} + +const LEFT_BUTTON = 0; +const RIGHT_BUTTON = 2; +const assignEvents$1 = (handler) => { + handler.mousedown = handler.dragStart; + handler.mousemoveWindow = handler.dragMove; + handler.mouseup = handler.dragEnd; + handler.contextmenu = function (e) { + e.preventDefault(); + }; +}; +const generateMousePanHandler = ({ enable, clickTolerance, }) => { + const mouseMoveStateManager = new MouseMoveStateManager({ + checkCorrectEvent: (e) => DOM.mouseButton(e) === LEFT_BUTTON && !e.ctrlKey, + }); + return new DragHandler({ + clickTolerance, + move: (lastPoint, point) => ({ around: point, panDelta: point.sub(lastPoint) }), + activateOnStart: true, + moveStateManager: mouseMoveStateManager, + enable, + assignEvents: assignEvents$1, + }); +}; +const generateMouseRotationHandler = ({ enable, clickTolerance, bearingDegreesPerPixelMoved = 0.8 }) => { + const mouseMoveStateManager = new MouseMoveStateManager({ + checkCorrectEvent: (e) => (DOM.mouseButton(e) === LEFT_BUTTON && e.ctrlKey) || + (DOM.mouseButton(e) === RIGHT_BUTTON), + }); + return new DragHandler({ + clickTolerance, + move: (lastPoint, point) => ({ bearingDelta: (point.x - lastPoint.x) * bearingDegreesPerPixelMoved }), + // prevent browser context menu when necessary; we don't allow it with rotation + // because we can't discern rotation gesture start from contextmenu on Mac + moveStateManager: mouseMoveStateManager, + enable, + assignEvents: assignEvents$1, + }); +}; +const generateMousePitchHandler = ({ enable, clickTolerance, pitchDegreesPerPixelMoved = -0.5 }) => { + const mouseMoveStateManager = new MouseMoveStateManager({ + checkCorrectEvent: (e) => (DOM.mouseButton(e) === LEFT_BUTTON && e.ctrlKey) || + (DOM.mouseButton(e) === RIGHT_BUTTON), + }); + return new DragHandler({ + clickTolerance, + move: (lastPoint, point) => ({ pitchDelta: (point.y - lastPoint.y) * pitchDegreesPerPixelMoved }), + // prevent browser context menu when necessary; we don't allow it with rotation + // because we can't discern rotation gesture start from contextmenu on Mac + moveStateManager: mouseMoveStateManager, + enable, + assignEvents: assignEvents$1, + }); +}; + +class TouchPanHandler { + constructor(options, map) { + this._minTouches = options.cooperativeGestures ? 2 : 1; + this._clickTolerance = options.clickTolerance || 1; + this._map = map; + this.reset(); + } + reset() { + this._active = false; + this._touches = {}; + this._sum = new performance.Point(0, 0); + // Put a delay on the cooperative gesture message so it's less twitchy + setTimeout(() => { + this._cancelCooperativeMessage = false; + }, 200); + } + touchstart(e, points, mapTouches) { + return this._calculateTransform(e, points, mapTouches); + } + touchmove(e, points, mapTouches) { + if (this._map._cooperativeGestures) { + if (this._minTouches === 2 && mapTouches.length < 2 && !this._cancelCooperativeMessage) { + // If coop gesture enabled, show panning info to user + this._map._onCooperativeGesture(e, false, mapTouches.length); + } + else if (!this._cancelCooperativeMessage) { + // If user is successfully navigating, we don't need this warning until the touch resets + this._cancelCooperativeMessage = true; + } + } + if (!this._active || mapTouches.length < this._minTouches) + return; + e.preventDefault(); + return this._calculateTransform(e, points, mapTouches); + } + touchend(e, points, mapTouches) { + this._calculateTransform(e, points, mapTouches); + if (this._active && mapTouches.length < this._minTouches) { + this.reset(); + } + } + touchcancel() { + this.reset(); + } + _calculateTransform(e, points, mapTouches) { + if (mapTouches.length > 0) + this._active = true; + const touches = indexTouches(mapTouches, points); + const touchPointSum = new performance.Point(0, 0); + const touchDeltaSum = new performance.Point(0, 0); + let touchDeltaCount = 0; + for (const identifier in touches) { + const point = touches[identifier]; + const prevPoint = this._touches[identifier]; + if (prevPoint) { + touchPointSum._add(point); + touchDeltaSum._add(point.sub(prevPoint)); + touchDeltaCount++; + touches[identifier] = point; + } + } + this._touches = touches; + if (touchDeltaCount < this._minTouches || !touchDeltaSum.mag()) + return; + const panDelta = touchDeltaSum.div(touchDeltaCount); + this._sum._add(panDelta); + if (this._sum.mag() < this._clickTolerance) + return; + const around = touchPointSum.div(touchDeltaCount); + return { + around, + panDelta + }; + } + enable() { + this._enabled = true; + } + disable() { + this._enabled = false; + this.reset(); + } + isEnabled() { + return this._enabled; + } + isActive() { + return this._active; + } +} + +/** + * The `TwoFingersTouchHandler`s allows the user to zoom, pitch and rotate the map using two fingers + * + * @group Handlers + */ +class TwoFingersTouchHandler { + /** @internal */ + constructor() { + this.reset(); + } + reset() { + this._active = false; + delete this._firstTwoTouches; + } + touchstart(e, points, mapTouches) { + //log('touchstart', points, e.target.innerHTML, e.targetTouches.length ? e.targetTouches[0].target.innerHTML: undefined); + if (this._firstTwoTouches || mapTouches.length < 2) + return; + this._firstTwoTouches = [ + mapTouches[0].identifier, + mapTouches[1].identifier + ]; + // implemented by child classes + this._start([points[0], points[1]]); + } + touchmove(e, points, mapTouches) { + if (!this._firstTwoTouches) + return; + e.preventDefault(); + const [idA, idB] = this._firstTwoTouches; + const a = getTouchById(mapTouches, points, idA); + const b = getTouchById(mapTouches, points, idB); + if (!a || !b) + return; + const pinchAround = this._aroundCenter ? null : a.add(b).div(2); + // implemented by child classes + return this._move([a, b], pinchAround, e); + } + touchend(e, points, mapTouches) { + if (!this._firstTwoTouches) + return; + const [idA, idB] = this._firstTwoTouches; + const a = getTouchById(mapTouches, points, idA); + const b = getTouchById(mapTouches, points, idB); + if (a && b) + return; + if (this._active) + DOM.suppressClick(); + this.reset(); + } + touchcancel() { + this.reset(); + } + /** + * Enables the "drag to pitch" interaction. + * + * @example + * ```ts + * map.touchPitch.enable(); + * ``` + */ + enable(options) { + this._enabled = true; + this._aroundCenter = !!options && options.around === 'center'; + } + /** + * Disables the "drag to pitch" interaction. + * + * @example + * ```ts + * map.touchPitch.disable(); + * ``` + */ + disable() { + this._enabled = false; + this.reset(); + } + /** + * Returns a Boolean indicating whether the "drag to pitch" interaction is enabled. + * + * @returns `true` if the "drag to pitch" interaction is enabled. + */ + isEnabled() { + return this._enabled; + } + /** + * Returns a Boolean indicating whether the "drag to pitch" interaction is active, i.e. currently being used. + * + * @returns `true` if the "drag to pitch" interaction is active. + */ + isActive() { + return this._active; + } +} +function getTouchById(mapTouches, points, identifier) { + for (let i = 0; i < mapTouches.length; i++) { + if (mapTouches[i].identifier === identifier) + return points[i]; + } +} +/* ZOOM */ +const ZOOM_THRESHOLD = 0.1; +function getZoomDelta(distance, lastDistance) { + return Math.log(distance / lastDistance) / Math.LN2; +} +/** + * The `TwoFingersTouchHandler`s allows the user to zoom the map two fingers + * + * @group Handlers + */ +class TwoFingersTouchZoomHandler extends TwoFingersTouchHandler { + reset() { + super.reset(); + delete this._distance; + delete this._startDistance; + } + _start(points) { + this._startDistance = this._distance = points[0].dist(points[1]); + } + _move(points, pinchAround) { + const lastDistance = this._distance; + this._distance = points[0].dist(points[1]); + if (!this._active && Math.abs(getZoomDelta(this._distance, this._startDistance)) < ZOOM_THRESHOLD) + return; + this._active = true; + return { + zoomDelta: getZoomDelta(this._distance, lastDistance), + pinchAround + }; + } +} +/* ROTATE */ +const ROTATION_THRESHOLD = 25; // pixels along circumference of touch circle +function getBearingDelta(a, b) { + return a.angleWith(b) * 180 / Math.PI; +} +/** + * The `TwoFingersTouchHandler`s allows the user to rotate the map two fingers + * + * @group Handlers + */ +class TwoFingersTouchRotateHandler extends TwoFingersTouchHandler { + reset() { + super.reset(); + delete this._minDiameter; + delete this._startVector; + delete this._vector; + } + _start(points) { + this._startVector = this._vector = points[0].sub(points[1]); + this._minDiameter = points[0].dist(points[1]); + } + _move(points, pinchAround) { + const lastVector = this._vector; + this._vector = points[0].sub(points[1]); + if (!this._active && this._isBelowThreshold(this._vector)) + return; + this._active = true; + return { + bearingDelta: getBearingDelta(this._vector, lastVector), + pinchAround + }; + } + _isBelowThreshold(vector) { + /* + * The threshold before a rotation actually happens is configured in + * pixels along the circumference of the circle formed by the two fingers. + * This makes the threshold in degrees larger when the fingers are close + * together and smaller when the fingers are far apart. + * + * Use the smallest diameter from the whole gesture to reduce sensitivity + * when pinching in and out. + */ + this._minDiameter = Math.min(this._minDiameter, vector.mag()); + const circumference = Math.PI * this._minDiameter; + const threshold = ROTATION_THRESHOLD / circumference * 360; + const bearingDeltaSinceStart = getBearingDelta(vector, this._startVector); + return Math.abs(bearingDeltaSinceStart) < threshold; + } +} +/* PITCH */ +function isVertical(vector) { + return Math.abs(vector.y) > Math.abs(vector.x); +} +const ALLOWED_SINGLE_TOUCH_TIME = 100; +/** + * The `TwoFingersTouchPitchHandler` allows the user to pitch the map by dragging up and down with two fingers. + * + * @group Handlers + */ +class TwoFingersTouchPitchHandler extends TwoFingersTouchHandler { + constructor(map) { + super(); + this._map = map; + } + reset() { + super.reset(); + this._valid = undefined; + delete this._firstMove; + delete this._lastPoints; + } + touchstart(e, points, mapTouches) { + super.touchstart(e, points, mapTouches); + this._currentTouchCount = mapTouches.length; + } + _start(points) { + this._lastPoints = points; + if (isVertical(points[0].sub(points[1]))) { + // fingers are more horizontal than vertical + this._valid = false; + } + } + _move(points, center, e) { + // If cooperative gestures is enabled, we need a 3-finger minimum for this gesture to register + if (this._map._cooperativeGestures && this._currentTouchCount < 3) { + return; + } + const vectorA = points[0].sub(this._lastPoints[0]); + const vectorB = points[1].sub(this._lastPoints[1]); + this._valid = this.gestureBeginsVertically(vectorA, vectorB, e.timeStamp); + if (!this._valid) + return; + this._lastPoints = points; + this._active = true; + const yDeltaAverage = (vectorA.y + vectorB.y) / 2; + const degreesPerPixelMoved = -0.5; + return { + pitchDelta: yDeltaAverage * degreesPerPixelMoved + }; + } + gestureBeginsVertically(vectorA, vectorB, timeStamp) { + if (this._valid !== undefined) + return this._valid; + const threshold = 2; + const movedA = vectorA.mag() >= threshold; + const movedB = vectorB.mag() >= threshold; + // neither finger has moved a meaningful amount, wait + if (!movedA && !movedB) + return; + // One finger has moved and the other has not. + // If enough time has passed, decide it is not a pitch. + if (!movedA || !movedB) { + if (this._firstMove === undefined) { + this._firstMove = timeStamp; + } + if (timeStamp - this._firstMove < ALLOWED_SINGLE_TOUCH_TIME) { + // still waiting for a movement from the second finger + return undefined; + } + else { + return false; + } + } + const isSameDirection = vectorA.y > 0 === vectorB.y > 0; + return isVertical(vectorA) && isVertical(vectorB) && isSameDirection; + } +} + +const defaultOptions$5 = { + panStep: 100, + bearingStep: 15, + pitchStep: 10 +}; +/** + * The `KeyboardHandler` allows the user to zoom, rotate, and pan the map using + * the following keyboard shortcuts: + * + * - `=` / `+`: Increase the zoom level by 1. + * - `Shift-=` / `Shift-+`: Increase the zoom level by 2. + * - `-`: Decrease the zoom level by 1. + * - `Shift--`: Decrease the zoom level by 2. + * - Arrow keys: Pan by 100 pixels. + * - `Shift+⇢`: Increase the rotation by 15 degrees. + * - `Shift+⇠`: Decrease the rotation by 15 degrees. + * - `Shift+⇡`: Increase the pitch by 10 degrees. + * - `Shift+⇣`: Decrease the pitch by 10 degrees. + * + * @group Handlers + */ +class KeyboardHandler { + /** @internal */ + constructor(map) { + this._tr = new TransformProvider(map); + const stepOptions = defaultOptions$5; + this._panStep = stepOptions.panStep; + this._bearingStep = stepOptions.bearingStep; + this._pitchStep = stepOptions.pitchStep; + this._rotationDisabled = false; + } + reset() { + this._active = false; + } + keydown(e) { + if (e.altKey || e.ctrlKey || e.metaKey) + return; + let zoomDir = 0; + let bearingDir = 0; + let pitchDir = 0; + let xDir = 0; + let yDir = 0; + switch (e.keyCode) { + case 61: + case 107: + case 171: + case 187: + zoomDir = 1; + break; + case 189: + case 109: + case 173: + zoomDir = -1; + break; + case 37: + if (e.shiftKey) { + bearingDir = -1; + } + else { + e.preventDefault(); + xDir = -1; + } + break; + case 39: + if (e.shiftKey) { + bearingDir = 1; + } + else { + e.preventDefault(); + xDir = 1; + } + break; + case 38: + if (e.shiftKey) { + pitchDir = 1; + } + else { + e.preventDefault(); + yDir = -1; + } + break; + case 40: + if (e.shiftKey) { + pitchDir = -1; + } + else { + e.preventDefault(); + yDir = 1; + } + break; + default: + return; + } + if (this._rotationDisabled) { + bearingDir = 0; + pitchDir = 0; + } + return { + cameraAnimation: (map) => { + const tr = this._tr; + map.easeTo({ + duration: 300, + easeId: 'keyboardHandler', + easing: easeOut, + zoom: zoomDir ? Math.round(tr.zoom) + zoomDir * (e.shiftKey ? 2 : 1) : tr.zoom, + bearing: tr.bearing + bearingDir * this._bearingStep, + pitch: tr.pitch + pitchDir * this._pitchStep, + offset: [-xDir * this._panStep, -yDir * this._panStep], + center: tr.center + }, { originalEvent: e }); + } + }; + } + /** + * Enables the "keyboard rotate and zoom" interaction. + * + * @example + * ```ts + * map.keyboard.enable(); + * ``` + */ + enable() { + this._enabled = true; + } + /** + * Disables the "keyboard rotate and zoom" interaction. + * + * @example + * ```ts + * map.keyboard.disable(); + * ``` + */ + disable() { + this._enabled = false; + this.reset(); + } + /** + * Returns a Boolean indicating whether the "keyboard rotate and zoom" + * interaction is enabled. + * + * @returns `true` if the "keyboard rotate and zoom" + * interaction is enabled. + */ + isEnabled() { + return this._enabled; + } + /** + * Returns true if the handler is enabled and has detected the start of a + * zoom/rotate gesture. + * + * @returns `true` if the handler is enabled and has detected the + * start of a zoom/rotate gesture. + */ + isActive() { + return this._active; + } + /** + * Disables the "keyboard pan/rotate" interaction, leaving the + * "keyboard zoom" interaction enabled. + * + * @example + * ```ts + * map.keyboard.disableRotation(); + * ``` + */ + disableRotation() { + this._rotationDisabled = true; + } + /** + * Enables the "keyboard pan/rotate" interaction. + * + * @example + * ```ts + * map.keyboard.enable(); + * map.keyboard.enableRotation(); + * ``` + */ + enableRotation() { + this._rotationDisabled = false; + } +} +function easeOut(t) { + return t * (2 - t); +} + +// deltaY value for mouse scroll wheel identification +const wheelZoomDelta = 4.000244140625; +// These magic numbers control the rate of zoom. Trackpad events fire at a greater +// frequency than mouse scroll wheel, so reduce the zoom rate per wheel tick +const defaultZoomRate = 1 / 100; +const wheelZoomRate = 1 / 450; +// upper bound on how much we scale the map in any single render frame; this +// is used to limit zoom rate in the case of very fast scrolling +const maxScalePerFrame = 2; +/** + * The `ScrollZoomHandler` allows the user to zoom the map by scrolling. + * + * @group Handlers + */ +class ScrollZoomHandler { + /** @internal */ + constructor(map, triggerRenderFrame) { + this._onTimeout = (initialEvent) => { + this._type = 'wheel'; + this._delta -= this._lastValue; + if (!this._active) { + this._start(initialEvent); + } + }; + this._map = map; + this._tr = new TransformProvider(map); + this._el = map.getCanvasContainer(); + this._triggerRenderFrame = triggerRenderFrame; + this._delta = 0; + this._defaultZoomRate = defaultZoomRate; + this._wheelZoomRate = wheelZoomRate; + } + /** + * Set the zoom rate of a trackpad + * @param zoomRate - 1/100 The rate used to scale trackpad movement to a zoom value. + * @example + * Speed up trackpad zoom + * ```ts + * map.scrollZoom.setZoomRate(1/25); + * ``` + */ + setZoomRate(zoomRate) { + this._defaultZoomRate = zoomRate; + } + /** + * Set the zoom rate of a mouse wheel + * @param wheelZoomRate - 1/450 The rate used to scale mouse wheel movement to a zoom value. + * @example + * Slow down zoom of mouse wheel + * ```ts + * map.scrollZoom.setWheelZoomRate(1/600); + * ``` + */ + setWheelZoomRate(wheelZoomRate) { + this._wheelZoomRate = wheelZoomRate; + } + /** + * Returns a Boolean indicating whether the "scroll to zoom" interaction is enabled. + * @returns `true` if the "scroll to zoom" interaction is enabled. + */ + isEnabled() { + return !!this._enabled; + } + /* + * Active state is turned on and off with every scroll wheel event and is set back to false before the map + * render is called, so _active is not a good candidate for determining if a scroll zoom animation is in + * progress. + */ + isActive() { + return !!this._active || this._finishTimeout !== undefined; + } + isZooming() { + return !!this._zooming; + } + /** + * Enables the "scroll to zoom" interaction. + * + * @param options - Options object. + * @example + * ```ts + * map.scrollZoom.enable(); + * map.scrollZoom.enable({ around: 'center' }) + * ``` + */ + enable(options) { + if (this.isEnabled()) + return; + this._enabled = true; + this._aroundCenter = !!options && options.around === 'center'; + } + /** + * Disables the "scroll to zoom" interaction. + * + * @example + * ```ts + * map.scrollZoom.disable(); + * ``` + */ + disable() { + if (!this.isEnabled()) + return; + this._enabled = false; + } + wheel(e) { + if (!this.isEnabled()) + return; + if (this._map._cooperativeGestures) { + if (e[this._map._metaKey]) { + e.preventDefault(); + } + else { + return; + } + } + let value = e.deltaMode === WheelEvent.DOM_DELTA_LINE ? e.deltaY * 40 : e.deltaY; + const now = performance.browser.now(), timeDelta = now - (this._lastWheelEventTime || 0); + this._lastWheelEventTime = now; + if (value !== 0 && (value % wheelZoomDelta) === 0) { + // This one is definitely a mouse wheel event. + this._type = 'wheel'; + } + else if (value !== 0 && Math.abs(value) < 4) { + // This one is definitely a trackpad event because it is so small. + this._type = 'trackpad'; + } + else if (timeDelta > 400) { + // This is likely a new scroll action. + this._type = null; + this._lastValue = value; + // Start a timeout in case this was a singular event, and dely it by up to 40ms. + this._timeout = setTimeout(this._onTimeout, 40, e); + } + else if (!this._type) { + // This is a repeating event, but we don't know the type of event just yet. + // If the delta per time is small, we assume it's a fast trackpad; otherwise we switch into wheel mode. + this._type = (Math.abs(timeDelta * value) < 200) ? 'trackpad' : 'wheel'; + // Make sure our delayed event isn't fired again, because we accumulate + // the previous event (which was less than 40ms ago) into this event. + if (this._timeout) { + clearTimeout(this._timeout); + this._timeout = null; + value += this._lastValue; + } + } + // Slow down zoom if shift key is held for more precise zooming + if (e.shiftKey && value) + value = value / 4; + // Only fire the callback if we actually know what type of scrolling device the user uses. + if (this._type) { + this._lastWheelEvent = e; + this._delta -= value; + if (!this._active) { + this._start(e); + } + } + e.preventDefault(); + } + _start(e) { + if (!this._delta) + return; + if (this._frameId) { + this._frameId = null; + } + this._active = true; + if (!this.isZooming()) { + this._zooming = true; + } + if (this._finishTimeout) { + clearTimeout(this._finishTimeout); + delete this._finishTimeout; + } + const pos = DOM.mousePos(this._el, e); + const tr = this._tr; + this._around = performance.LngLat.convert(this._aroundCenter ? tr.center : tr.unproject(pos)); + this._aroundPoint = tr.transform.locationPoint(this._around); + if (!this._frameId) { + this._frameId = true; + this._triggerRenderFrame(); + } + } + renderFrame() { + if (!this._frameId) + return; + this._frameId = null; + if (!this.isActive()) + return; + const tr = this._tr.transform; + // if we've had scroll events since the last render frame, consume the + // accumulated delta, and update the target zoom level accordingly + if (this._delta !== 0) { + // For trackpad events and single mouse wheel ticks, use the default zoom rate + const zoomRate = (this._type === 'wheel' && Math.abs(this._delta) > wheelZoomDelta) ? this._wheelZoomRate : this._defaultZoomRate; + // Scale by sigmoid of scroll wheel delta. + let scale = maxScalePerFrame / (1 + Math.exp(-Math.abs(this._delta * zoomRate))); + if (this._delta < 0 && scale !== 0) { + scale = 1 / scale; + } + const fromScale = typeof this._targetZoom === 'number' ? tr.zoomScale(this._targetZoom) : tr.scale; + this._targetZoom = Math.min(tr.maxZoom, Math.max(tr.minZoom, tr.scaleZoom(fromScale * scale))); + // if this is a mouse wheel, refresh the starting zoom and easing + // function we're using to smooth out the zooming between wheel + // events + if (this._type === 'wheel') { + this._startZoom = tr.zoom; + this._easing = this._smoothOutEasing(200); + } + this._delta = 0; + } + const targetZoom = typeof this._targetZoom === 'number' ? + this._targetZoom : tr.zoom; + const startZoom = this._startZoom; + const easing = this._easing; + let finished = false; + let zoom; + if (this._type === 'wheel' && startZoom && easing) { + const t = Math.min((performance.browser.now() - this._lastWheelEventTime) / 200, 1); + const k = easing(t); + zoom = performance.interpolate.number(startZoom, targetZoom, k); + if (t < 1) { + if (!this._frameId) { + this._frameId = true; + } + } + else { + finished = true; + } + } + else { + zoom = targetZoom; + finished = true; + } + this._active = true; + if (finished) { + this._active = false; + this._finishTimeout = setTimeout(() => { + this._zooming = false; + this._triggerRenderFrame(); + delete this._targetZoom; + delete this._finishTimeout; + }, 200); + } + return { + noInertia: true, + needsRenderFrame: !finished, + zoomDelta: zoom - tr.zoom, + around: this._aroundPoint, + originalEvent: this._lastWheelEvent + }; + } + _smoothOutEasing(duration) { + let easing = performance.defaultEasing; + if (this._prevEase) { + const currentEase = this._prevEase; + const t = (performance.browser.now() - currentEase.start) / currentEase.duration; + const speed = currentEase.easing(t + 0.01) - currentEase.easing(t); + // Quick hack to make new bezier that is continuous with last + const x = 0.27 / Math.sqrt(speed * speed + 0.0001) * 0.01; + const y = Math.sqrt(0.27 * 0.27 - x * x); + easing = performance.bezier(x, y, 0.25, 1); + } + this._prevEase = { + start: performance.browser.now(), + duration, + easing + }; + return easing; + } + reset() { + this._active = false; + this._zooming = false; + delete this._targetZoom; + if (this._finishTimeout) { + clearTimeout(this._finishTimeout); + delete this._finishTimeout; + } + } +} + +/** + * The `DoubleClickZoomHandler` allows the user to zoom the map at a point by + * double clicking or double tapping. + * + * @group Handlers + */ +class DoubleClickZoomHandler { + /** @internal */ + constructor(clickZoom, TapZoom) { + this._clickZoom = clickZoom; + this._tapZoom = TapZoom; + } + /** + * Enables the "double click to zoom" interaction. + * + * @example + * ```ts + * map.doubleClickZoom.enable(); + * ``` + */ + enable() { + this._clickZoom.enable(); + this._tapZoom.enable(); + } + /** + * Disables the "double click to zoom" interaction. + * + * @example + * ```ts + * map.doubleClickZoom.disable(); + * ``` + */ + disable() { + this._clickZoom.disable(); + this._tapZoom.disable(); + } + /** + * Returns a Boolean indicating whether the "double click to zoom" interaction is enabled. + * + * @returns `true` if the "double click to zoom" interaction is enabled. + */ + isEnabled() { + return this._clickZoom.isEnabled() && this._tapZoom.isEnabled(); + } + /** + * Returns a Boolean indicating whether the "double click to zoom" interaction is active, i.e. currently being used. + * + * @returns `true` if the "double click to zoom" interaction is active. + */ + isActive() { + return this._clickZoom.isActive() || this._tapZoom.isActive(); + } +} + +/** + * The `ClickZoomHandler` allows the user to zoom the map at a point by double clicking + * It is used by other handlers + */ +class ClickZoomHandler { + /** @internal */ + constructor(map) { + this._tr = new TransformProvider(map); + this.reset(); + } + reset() { + this._active = false; + } + dblclick(e, point) { + e.preventDefault(); + return { + cameraAnimation: (map) => { + map.easeTo({ + duration: 300, + zoom: this._tr.zoom + (e.shiftKey ? -1 : 1), + around: this._tr.unproject(point) + }, { originalEvent: e }); + } + }; + } + enable() { + this._enabled = true; + } + disable() { + this._enabled = false; + this.reset(); + } + isEnabled() { + return this._enabled; + } + isActive() { + return this._active; + } +} + +class TapDragZoomHandler { + constructor() { + this._tap = new TapRecognizer({ + numTouches: 1, + numTaps: 1 + }); + this.reset(); + } + reset() { + this._active = false; + delete this._swipePoint; + delete this._swipeTouch; + delete this._tapTime; + delete this._tapPoint; + this._tap.reset(); + } + touchstart(e, points, mapTouches) { + if (this._swipePoint) + return; + if (!this._tapTime) { + this._tap.touchstart(e, points, mapTouches); + } + else { + const swipePoint = points[0]; + const soonEnough = e.timeStamp - this._tapTime < MAX_TAP_INTERVAL; + const closeEnough = this._tapPoint.dist(swipePoint) < MAX_DIST; + if (!soonEnough || !closeEnough) { + this.reset(); + } + else if (mapTouches.length > 0) { + this._swipePoint = swipePoint; + this._swipeTouch = mapTouches[0].identifier; + } + } + } + touchmove(e, points, mapTouches) { + if (!this._tapTime) { + this._tap.touchmove(e, points, mapTouches); + } + else if (this._swipePoint) { + if (mapTouches[0].identifier !== this._swipeTouch) { + return; + } + const newSwipePoint = points[0]; + const dist = newSwipePoint.y - this._swipePoint.y; + this._swipePoint = newSwipePoint; + e.preventDefault(); + this._active = true; + return { + zoomDelta: dist / 128 + }; + } + } + touchend(e, points, mapTouches) { + if (!this._tapTime) { + const point = this._tap.touchend(e, points, mapTouches); + if (point) { + this._tapTime = e.timeStamp; + this._tapPoint = point; + } + } + else if (this._swipePoint) { + if (mapTouches.length === 0) { + this.reset(); + } + } + } + touchcancel() { + this.reset(); + } + enable() { + this._enabled = true; + } + disable() { + this._enabled = false; + this.reset(); + } + isEnabled() { + return this._enabled; + } + isActive() { + return this._active; + } +} + +/** + * The `DragPanHandler` allows the user to pan the map by clicking and dragging + * the cursor. + * + * @group Handlers + */ +class DragPanHandler { + /** @internal */ + constructor(el, mousePan, touchPan) { + this._el = el; + this._mousePan = mousePan; + this._touchPan = touchPan; + } + /** + * Enables the "drag to pan" interaction. + * + * @param options - Options object + * @example + * ```ts + * map.dragPan.enable(); + * map.dragPan.enable({ + * linearity: 0.3, + * easing: bezier(0, 0, 0.3, 1), + * maxSpeed: 1400, + * deceleration: 2500, + * }); + * ``` + */ + enable(options) { + this._inertiaOptions = options || {}; + this._mousePan.enable(); + this._touchPan.enable(); + this._el.classList.add('maplibregl-touch-drag-pan'); + } + /** + * Disables the "drag to pan" interaction. + * + * @example + * ```ts + * map.dragPan.disable(); + * ``` + */ + disable() { + this._mousePan.disable(); + this._touchPan.disable(); + this._el.classList.remove('maplibregl-touch-drag-pan'); + } + /** + * Returns a Boolean indicating whether the "drag to pan" interaction is enabled. + * + * @returns `true` if the "drag to pan" interaction is enabled. + */ + isEnabled() { + return this._mousePan.isEnabled() && this._touchPan.isEnabled(); + } + /** + * Returns a Boolean indicating whether the "drag to pan" interaction is active, i.e. currently being used. + * + * @returns `true` if the "drag to pan" interaction is active. + */ + isActive() { + return this._mousePan.isActive() || this._touchPan.isActive(); + } +} + +/** + * The `DragRotateHandler` allows the user to rotate the map by clicking and + * dragging the cursor while holding the right mouse button or `ctrl` key. + * + * @group Handlers + */ +class DragRotateHandler { + /** @internal */ + constructor(options, mouseRotate, mousePitch) { + this._pitchWithRotate = options.pitchWithRotate; + this._mouseRotate = mouseRotate; + this._mousePitch = mousePitch; + } + /** + * Enables the "drag to rotate" interaction. + * + * @example + * ```ts + * map.dragRotate.enable(); + * ``` + */ + enable() { + this._mouseRotate.enable(); + if (this._pitchWithRotate) + this._mousePitch.enable(); + } + /** + * Disables the "drag to rotate" interaction. + * + * @example + * ```ts + * map.dragRotate.disable(); + * ``` + */ + disable() { + this._mouseRotate.disable(); + this._mousePitch.disable(); + } + /** + * Returns a Boolean indicating whether the "drag to rotate" interaction is enabled. + * + * @returns `true` if the "drag to rotate" interaction is enabled. + */ + isEnabled() { + return this._mouseRotate.isEnabled() && (!this._pitchWithRotate || this._mousePitch.isEnabled()); + } + /** + * Returns a Boolean indicating whether the "drag to rotate" interaction is active, i.e. currently being used. + * + * @returns `true` if the "drag to rotate" interaction is active. + */ + isActive() { + return this._mouseRotate.isActive() || this._mousePitch.isActive(); + } +} + +/** + * The `TwoFingersTouchZoomRotateHandler` allows the user to zoom and rotate the map by + * pinching on a touchscreen. + * + * They can zoom with one finger by double tapping and dragging. On the second tap, + * hold the finger down and drag up or down to zoom in or out. + * + * @group Handlers + */ +class TwoFingersTouchZoomRotateHandler { + /** @internal */ + constructor(el, touchZoom, touchRotate, tapDragZoom) { + this._el = el; + this._touchZoom = touchZoom; + this._touchRotate = touchRotate; + this._tapDragZoom = tapDragZoom; + this._rotationDisabled = false; + this._enabled = true; + } + /** + * Enables the "pinch to rotate and zoom" interaction. + * + * @param options - Options object. + * + * @example + * ```ts + * map.touchZoomRotate.enable(); + * map.touchZoomRotate.enable({ around: 'center' }); + * ``` + */ + enable(options) { + this._touchZoom.enable(options); + if (!this._rotationDisabled) + this._touchRotate.enable(options); + this._tapDragZoom.enable(); + this._el.classList.add('maplibregl-touch-zoom-rotate'); + } + /** + * Disables the "pinch to rotate and zoom" interaction. + * + * @example + * ```ts + * map.touchZoomRotate.disable(); + * ``` + */ + disable() { + this._touchZoom.disable(); + this._touchRotate.disable(); + this._tapDragZoom.disable(); + this._el.classList.remove('maplibregl-touch-zoom-rotate'); + } + /** + * Returns a Boolean indicating whether the "pinch to rotate and zoom" interaction is enabled. + * + * @returns `true` if the "pinch to rotate and zoom" interaction is enabled. + */ + isEnabled() { + return this._touchZoom.isEnabled() && + (this._rotationDisabled || this._touchRotate.isEnabled()) && + this._tapDragZoom.isEnabled(); + } + /** + * Returns true if the handler is enabled and has detected the start of a zoom/rotate gesture. + * + * @returns `true` if the handler is active, `false` otherwise + */ + isActive() { + return this._touchZoom.isActive() || this._touchRotate.isActive() || this._tapDragZoom.isActive(); + } + /** + * Disables the "pinch to rotate" interaction, leaving the "pinch to zoom" + * interaction enabled. + * + * @example + * ```ts + * map.touchZoomRotate.disableRotation(); + * ``` + */ + disableRotation() { + this._rotationDisabled = true; + this._touchRotate.disable(); + } + /** + * Enables the "pinch to rotate" interaction. + * + * @example + * ```ts + * map.touchZoomRotate.enable(); + * map.touchZoomRotate.enableRotation(); + * ``` + */ + enableRotation() { + this._rotationDisabled = false; + if (this._touchZoom.isEnabled()) + this._touchRotate.enable(); + } +} + +const isMoving = p => p.zoom || p.drag || p.pitch || p.rotate; +class RenderFrameEvent extends performance.Event { +} +function hasChange(result) { + return (result.panDelta && result.panDelta.mag()) || result.zoomDelta || result.bearingDelta || result.pitchDelta; +} +class HandlerManager { + constructor(map, options) { + this.handleWindowEvent = (e) => { + this.handleEvent(e, `${e.type}Window`); + }; + this.handleEvent = (e, eventName) => { + if (e.type === 'blur') { + this.stop(true); + return; + } + this._updatingCamera = true; + const inputEvent = e.type === 'renderFrame' ? undefined : e; + /* + * We don't call e.preventDefault() for any events by default. + * Handlers are responsible for calling it where necessary. + */ + const mergedHandlerResult = { needsRenderFrame: false }; + const eventsInProgress = {}; + const activeHandlers = {}; + const eventTouches = e.touches; + const mapTouches = eventTouches ? this._getMapTouches(eventTouches) : undefined; + const points = mapTouches ? DOM.touchPos(this._el, mapTouches) : DOM.mousePos(this._el, e); + for (const { handlerName, handler, allowed } of this._handlers) { + if (!handler.isEnabled()) + continue; + let data; + if (this._blockedByActive(activeHandlers, allowed, handlerName)) { + handler.reset(); + } + else { + if (handler[eventName || e.type]) { + data = handler[eventName || e.type](e, points, mapTouches); + this.mergeHandlerResult(mergedHandlerResult, eventsInProgress, data, handlerName, inputEvent); + if (data && data.needsRenderFrame) { + this._triggerRenderFrame(); + } + } + } + if (data || handler.isActive()) { + activeHandlers[handlerName] = handler; + } + } + const deactivatedHandlers = {}; + for (const name in this._previousActiveHandlers) { + if (!activeHandlers[name]) { + deactivatedHandlers[name] = inputEvent; + } + } + this._previousActiveHandlers = activeHandlers; + if (Object.keys(deactivatedHandlers).length || hasChange(mergedHandlerResult)) { + this._changes.push([mergedHandlerResult, eventsInProgress, deactivatedHandlers]); + this._triggerRenderFrame(); + } + if (Object.keys(activeHandlers).length || hasChange(mergedHandlerResult)) { + this._map._stop(true); + } + this._updatingCamera = false; + const { cameraAnimation } = mergedHandlerResult; + if (cameraAnimation) { + this._inertia.clear(); + this._fireEvents({}, {}, true); + this._changes = []; + cameraAnimation(this._map); + } + }; + this._map = map; + this._el = this._map.getCanvasContainer(); + this._handlers = []; + this._handlersById = {}; + this._changes = []; + this._inertia = new HandlerInertia(map); + this._bearingSnap = options.bearingSnap; + this._previousActiveHandlers = {}; + // Track whether map is currently moving, to compute start/move/end events + this._eventsInProgress = {}; + this._addDefaultHandlers(options); + const el = this._el; + this._listeners = [ + // This needs to be `passive: true` so that a double tap fires two + // pairs of touchstart/end events in iOS Safari 13. If this is set to + // `passive: false` then the second pair of events is only fired if + // preventDefault() is called on the first touchstart. Calling preventDefault() + // undesirably prevents click events. + [el, 'touchstart', { passive: true }], + // This needs to be `passive: false` so that scrolls and pinches can be + // prevented in browsers that don't support `touch-actions: none`, for example iOS Safari 12. + [el, 'touchmove', { passive: false }], + [el, 'touchend', undefined], + [el, 'touchcancel', undefined], + [el, 'mousedown', undefined], + [el, 'mousemove', undefined], + [el, 'mouseup', undefined], + // Bind window-level event listeners for move and up/end events. In the absence of + // the pointer capture API, which is not supported by all necessary platforms, + // window-level event listeners give us the best shot at capturing events that + // fall outside the map canvas element. Use `{capture: true}` for the move event + // to prevent map move events from being fired during a drag. + [document, 'mousemove', { capture: true }], + [document, 'mouseup', undefined], + [el, 'mouseover', undefined], + [el, 'mouseout', undefined], + [el, 'dblclick', undefined], + [el, 'click', undefined], + [el, 'keydown', { capture: false }], + [el, 'keyup', undefined], + [el, 'wheel', { passive: false }], + [el, 'contextmenu', undefined], + [window, 'blur', undefined] + ]; + for (const [target, type, listenerOptions] of this._listeners) { + DOM.addEventListener(target, type, target === document ? this.handleWindowEvent : this.handleEvent, listenerOptions); + } + } + destroy() { + for (const [target, type, listenerOptions] of this._listeners) { + DOM.removeEventListener(target, type, target === document ? this.handleWindowEvent : this.handleEvent, listenerOptions); + } + } + _addDefaultHandlers(options) { + const map = this._map; + const el = map.getCanvasContainer(); + this._add('mapEvent', new MapEventHandler(map, options)); + const boxZoom = map.boxZoom = new BoxZoomHandler(map, options); + this._add('boxZoom', boxZoom); + if (options.interactive && options.boxZoom) { + boxZoom.enable(); + } + const tapZoom = new TapZoomHandler(map); + const clickZoom = new ClickZoomHandler(map); + map.doubleClickZoom = new DoubleClickZoomHandler(clickZoom, tapZoom); + this._add('tapZoom', tapZoom); + this._add('clickZoom', clickZoom); + if (options.interactive && options.doubleClickZoom) { + map.doubleClickZoom.enable(); + } + const tapDragZoom = new TapDragZoomHandler(); + this._add('tapDragZoom', tapDragZoom); + const touchPitch = map.touchPitch = new TwoFingersTouchPitchHandler(map); + this._add('touchPitch', touchPitch); + if (options.interactive && options.touchPitch) { + map.touchPitch.enable(options.touchPitch); + } + const mouseRotate = generateMouseRotationHandler(options); + const mousePitch = generateMousePitchHandler(options); + map.dragRotate = new DragRotateHandler(options, mouseRotate, mousePitch); + this._add('mouseRotate', mouseRotate, ['mousePitch']); + this._add('mousePitch', mousePitch, ['mouseRotate']); + if (options.interactive && options.dragRotate) { + map.dragRotate.enable(); + } + const mousePan = generateMousePanHandler(options); + const touchPan = new TouchPanHandler(options, map); + map.dragPan = new DragPanHandler(el, mousePan, touchPan); + this._add('mousePan', mousePan); + this._add('touchPan', touchPan, ['touchZoom', 'touchRotate']); + if (options.interactive && options.dragPan) { + map.dragPan.enable(options.dragPan); + } + const touchRotate = new TwoFingersTouchRotateHandler(); + const touchZoom = new TwoFingersTouchZoomHandler(); + map.touchZoomRotate = new TwoFingersTouchZoomRotateHandler(el, touchZoom, touchRotate, tapDragZoom); + this._add('touchRotate', touchRotate, ['touchPan', 'touchZoom']); + this._add('touchZoom', touchZoom, ['touchPan', 'touchRotate']); + if (options.interactive && options.touchZoomRotate) { + map.touchZoomRotate.enable(options.touchZoomRotate); + } + const scrollZoom = map.scrollZoom = new ScrollZoomHandler(map, () => this._triggerRenderFrame()); + this._add('scrollZoom', scrollZoom, ['mousePan']); + if (options.interactive && options.scrollZoom) { + map.scrollZoom.enable(options.scrollZoom); + } + const keyboard = map.keyboard = new KeyboardHandler(map); + this._add('keyboard', keyboard); + if (options.interactive && options.keyboard) { + map.keyboard.enable(); + } + this._add('blockableMapEvent', new BlockableMapEventHandler(map)); + } + _add(handlerName, handler, allowed) { + this._handlers.push({ handlerName, handler, allowed }); + this._handlersById[handlerName] = handler; + } + stop(allowEndAnimation) { + // do nothing if this method was triggered by a gesture update + if (this._updatingCamera) + return; + for (const { handler } of this._handlers) { + handler.reset(); + } + this._inertia.clear(); + this._fireEvents({}, {}, allowEndAnimation); + this._changes = []; + } + isActive() { + for (const { handler } of this._handlers) { + if (handler.isActive()) + return true; + } + return false; + } + isZooming() { + return !!this._eventsInProgress.zoom || this._map.scrollZoom.isZooming(); + } + isRotating() { + return !!this._eventsInProgress.rotate; + } + isMoving() { + return Boolean(isMoving(this._eventsInProgress)) || this.isZooming(); + } + _blockedByActive(activeHandlers, allowed, myName) { + for (const name in activeHandlers) { + if (name === myName) + continue; + if (!allowed || allowed.indexOf(name) < 0) { + return true; + } + } + return false; + } + _getMapTouches(touches) { + const mapTouches = []; + for (const t of touches) { + const target = t.target; + if (this._el.contains(target)) { + mapTouches.push(t); + } + } + return mapTouches; + } + mergeHandlerResult(mergedHandlerResult, eventsInProgress, handlerResult, name, e) { + if (!handlerResult) + return; + performance.extend(mergedHandlerResult, handlerResult); + const eventData = { handlerName: name, originalEvent: handlerResult.originalEvent || e }; + // track which handler changed which camera property + if (handlerResult.zoomDelta !== undefined) { + eventsInProgress.zoom = eventData; + } + if (handlerResult.panDelta !== undefined) { + eventsInProgress.drag = eventData; + } + if (handlerResult.pitchDelta !== undefined) { + eventsInProgress.pitch = eventData; + } + if (handlerResult.bearingDelta !== undefined) { + eventsInProgress.rotate = eventData; + } + } + _applyChanges() { + const combined = {}; + const combinedEventsInProgress = {}; + const combinedDeactivatedHandlers = {}; + for (const [change, eventsInProgress, deactivatedHandlers] of this._changes) { + if (change.panDelta) + combined.panDelta = (combined.panDelta || new performance.Point(0, 0))._add(change.panDelta); + if (change.zoomDelta) + combined.zoomDelta = (combined.zoomDelta || 0) + change.zoomDelta; + if (change.bearingDelta) + combined.bearingDelta = (combined.bearingDelta || 0) + change.bearingDelta; + if (change.pitchDelta) + combined.pitchDelta = (combined.pitchDelta || 0) + change.pitchDelta; + if (change.around !== undefined) + combined.around = change.around; + if (change.pinchAround !== undefined) + combined.pinchAround = change.pinchAround; + if (change.noInertia) + combined.noInertia = change.noInertia; + performance.extend(combinedEventsInProgress, eventsInProgress); + performance.extend(combinedDeactivatedHandlers, deactivatedHandlers); + } + this._updateMapTransform(combined, combinedEventsInProgress, combinedDeactivatedHandlers); + this._changes = []; + } + _updateMapTransform(combinedResult, combinedEventsInProgress, deactivatedHandlers) { + const map = this._map; + const tr = map._getTransformForUpdate(); + const terrain = map.terrain; + if (!hasChange(combinedResult) && !(terrain && this._terrainMovement)) { + return this._fireEvents(combinedEventsInProgress, deactivatedHandlers, true); + } + let { panDelta, zoomDelta, bearingDelta, pitchDelta, around, pinchAround } = combinedResult; + if (pinchAround !== undefined) { + around = pinchAround; + } + // stop any ongoing camera animations (easeTo, flyTo) + map._stop(true); + around = around || map.transform.centerPoint; + const loc = tr.pointLocation(panDelta ? around.sub(panDelta) : around); + if (bearingDelta) + tr.bearing += bearingDelta; + if (pitchDelta) + tr.pitch += pitchDelta; + if (zoomDelta) + tr.zoom += zoomDelta; + if (!terrain) { + tr.setLocationAtPoint(loc, around); + } + else { + // when 3d-terrain is enabled act a little different: + // - dragging do not drag the picked point itself, instead it drags the map by pixel-delta. + // With this approach it is no longer possible to pick a point from somewhere near + // the horizon to the center in one move. + // So this logic avoids the problem, that in such cases you easily loose orientation. + if (!this._terrainMovement && + (combinedEventsInProgress.drag || combinedEventsInProgress.zoom)) { + // When starting to drag or move, flag it and register moveend to clear flagging + this._terrainMovement = true; + this._map._elevationFreeze = true; + tr.setLocationAtPoint(loc, around); + this._map.once('moveend', () => { + this._map._elevationFreeze = false; + this._terrainMovement = false; + tr.recalculateZoom(map.terrain); + }); + } + else if (combinedEventsInProgress.drag && this._terrainMovement) { + // drag map + tr.center = tr.pointLocation(tr.centerPoint.sub(panDelta)); + } + else { + tr.setLocationAtPoint(loc, around); + } + } + map._applyUpdatedTransform(tr); + this._map._update(); + if (!combinedResult.noInertia) + this._inertia.record(combinedResult); + this._fireEvents(combinedEventsInProgress, deactivatedHandlers, true); + } + _fireEvents(newEventsInProgress, deactivatedHandlers, allowEndAnimation) { + const wasMoving = isMoving(this._eventsInProgress); + const nowMoving = isMoving(newEventsInProgress); + const startEvents = {}; + for (const eventName in newEventsInProgress) { + const { originalEvent } = newEventsInProgress[eventName]; + if (!this._eventsInProgress[eventName]) { + startEvents[`${eventName}start`] = originalEvent; + } + this._eventsInProgress[eventName] = newEventsInProgress[eventName]; + } + // fire start events only after this._eventsInProgress has been updated + if (!wasMoving && nowMoving) { + this._fireEvent('movestart', nowMoving.originalEvent); + } + for (const name in startEvents) { + this._fireEvent(name, startEvents[name]); + } + if (nowMoving) { + this._fireEvent('move', nowMoving.originalEvent); + } + for (const eventName in newEventsInProgress) { + const { originalEvent } = newEventsInProgress[eventName]; + this._fireEvent(eventName, originalEvent); + } + const endEvents = {}; + let originalEndEvent; + for (const eventName in this._eventsInProgress) { + const { handlerName, originalEvent } = this._eventsInProgress[eventName]; + if (!this._handlersById[handlerName].isActive()) { + delete this._eventsInProgress[eventName]; + originalEndEvent = deactivatedHandlers[handlerName] || originalEvent; + endEvents[`${eventName}end`] = originalEndEvent; + } + } + for (const name in endEvents) { + this._fireEvent(name, endEvents[name]); + } + const stillMoving = isMoving(this._eventsInProgress); + if (allowEndAnimation && (wasMoving || nowMoving) && !stillMoving) { + this._updatingCamera = true; + const inertialEase = this._inertia._onMoveEnd(this._map.dragPan._inertiaOptions); + const shouldSnapToNorth = bearing => bearing !== 0 && -this._bearingSnap < bearing && bearing < this._bearingSnap; + if (inertialEase && (inertialEase.essential || !performance.browser.prefersReducedMotion)) { + if (shouldSnapToNorth(inertialEase.bearing || this._map.getBearing())) { + inertialEase.bearing = 0; + } + inertialEase.freezeElevation = true; + this._map.easeTo(inertialEase, { originalEvent: originalEndEvent }); + } + else { + this._map.fire(new performance.Event('moveend', { originalEvent: originalEndEvent })); + if (shouldSnapToNorth(this._map.getBearing())) { + this._map.resetNorth(); + } + } + this._updatingCamera = false; + } + } + _fireEvent(type, e) { + this._map.fire(new performance.Event(type, e ? { originalEvent: e } : {})); + } + _requestFrame() { + this._map.triggerRepaint(); + return this._map._renderTaskQueue.add(timeStamp => { + delete this._frameId; + this.handleEvent(new RenderFrameEvent('renderFrame', { timeStamp })); + this._applyChanges(); + }); + } + _triggerRenderFrame() { + if (this._frameId === undefined) { + this._frameId = this._requestFrame(); + } + } +} + +class Camera extends performance.Evented { + constructor(transform, options) { + super(); + // Callback for map._requestRenderFrame + this._renderFrameCallback = () => { + const t = Math.min((performance.browser.now() - this._easeStart) / this._easeOptions.duration, 1); + this._onEaseFrame(this._easeOptions.easing(t)); + // if _stop is called during _onEaseFrame from _fireMoveEvents we should avoid a new _requestRenderFrame, checking it by ensuring _easeFrameId was not deleted + if (t < 1 && this._easeFrameId) { + this._easeFrameId = this._requestRenderFrame(this._renderFrameCallback); + } + else { + this.stop(); + } + }; + this._moving = false; + this._zooming = false; + this.transform = transform; + this._bearingSnap = options.bearingSnap; + this.on('moveend', () => { + delete this._requestedCameraState; + }); + } + /** + * Returns the map's geographical centerpoint. + * + * @returns The map's geographical centerpoint. + * @example + * Return a LngLat object such as `{lng: 0, lat: 0}` + * ```ts + * let center = map.getCenter(); + * // access longitude and latitude values directly + * let {lng, lat} = map.getCenter(); + * ``` + */ + getCenter() { return new performance.LngLat(this.transform.center.lng, this.transform.center.lat); } + /** + * Sets the map's geographical centerpoint. Equivalent to `jumpTo({center: center})`. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param center - The centerpoint to set. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * map.setCenter([-74, 38]); + * ``` + */ + setCenter(center, eventData) { + return this.jumpTo({ center }, eventData); + } + /** + * Pans the map by the specified offset. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param offset - `x` and `y` coordinates by which to pan the map. + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/) + */ + panBy(offset, options, eventData) { + offset = performance.Point.convert(offset).mult(-1); + return this.panTo(this.transform.center, performance.extend({ offset }, options), eventData); + } + /** + * Pans the map to the specified location with an animated transition. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param lnglat - The location to pan the map to. + * @param options - Options describing the destination and animation of the transition. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * map.panTo([-74, 38]); + * // Specify that the panTo animation should last 5000 milliseconds. + * map.panTo([-74, 38], {duration: 5000}); + * ``` + * @see [Update a feature in realtime](https://maplibre.org/maplibre-gl-js/docs/examples/live-update-feature/) + */ + panTo(lnglat, options, eventData) { + return this.easeTo(performance.extend({ + center: lnglat + }, options), eventData); + } + /** + * Returns the map's current zoom level. + * + * @returns The map's current zoom level. + * @example + * ```ts + * map.getZoom(); + * ``` + */ + getZoom() { return this.transform.zoom; } + /** + * Sets the map's zoom level. Equivalent to `jumpTo({zoom: zoom})`. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`. + * + * @param zoom - The zoom level to set (0-20). + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Zoom to the zoom level 5 without an animated transition + * ```ts + * map.setZoom(5); + * ``` + */ + setZoom(zoom, eventData) { + this.jumpTo({ zoom }, eventData); + return this; + } + /** + * Zooms the map to the specified zoom level, with an animated transition. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`. + * + * @param zoom - The zoom level to transition to. + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * // Zoom to the zoom level 5 without an animated transition + * map.zoomTo(5); + * // Zoom to the zoom level 8 with an animated transition + * map.zoomTo(8, { + * duration: 2000, + * offset: [100, 50] + * }); + * ``` + */ + zoomTo(zoom, options, eventData) { + return this.easeTo(performance.extend({ + zoom + }, options), eventData); + } + /** + * Increases the map's zoom level by 1. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Zoom the map in one level with a custom animation duration + * ```ts + * map.zoomIn({duration: 1000}); + * ``` + */ + zoomIn(options, eventData) { + this.zoomTo(this.getZoom() + 1, options, eventData); + return this; + } + /** + * Decreases the map's zoom level by 1. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Zoom the map out one level with a custom animation offset + * ```ts + * map.zoomOut({offset: [80, 60]}); + * ``` + */ + zoomOut(options, eventData) { + this.zoomTo(this.getZoom() - 1, options, eventData); + return this; + } + /** + * Returns the map's current bearing. The bearing is the compass direction that is "up"; for example, a bearing + * of 90° orients the map so that east is up. + * + * @returns The map's current bearing. + * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/) + */ + getBearing() { return this.transform.bearing; } + /** + * Sets the map's bearing (rotation). The bearing is the compass direction that is "up"; for example, a bearing + * of 90° orients the map so that east is up. + * + * Equivalent to `jumpTo({bearing: bearing})`. + * + * Triggers the following events: `movestart`, `moveend`, and `rotate`. + * + * @param bearing - The desired bearing. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Rotate the map to 90 degrees + * ```ts + * map.setBearing(90); + * ``` + */ + setBearing(bearing, eventData) { + this.jumpTo({ bearing }, eventData); + return this; + } + /** + * Returns the current padding applied around the map viewport. + * + * @returns The current padding around the map viewport. + */ + getPadding() { return this.transform.padding; } + /** + * Sets the padding in pixels around the viewport. + * + * Equivalent to `jumpTo({padding: padding})`. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param padding - The desired padding. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Sets a left padding of 300px, and a top padding of 50px + * ```ts + * map.setPadding({ left: 300, top: 50 }); + * ``` + */ + setPadding(padding, eventData) { + this.jumpTo({ padding }, eventData); + return this; + } + /** + * Rotates the map to the specified bearing, with an animated transition. The bearing is the compass direction + * that is "up"; for example, a bearing of 90° orients the map so that east is up. + * + * Triggers the following events: `movestart`, `moveend`, and `rotate`. + * + * @param bearing - The desired bearing. + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + rotateTo(bearing, options, eventData) { + return this.easeTo(performance.extend({ + bearing + }, options), eventData); + } + /** + * Rotates the map so that north is up (0° bearing), with an animated transition. + * + * Triggers the following events: `movestart`, `moveend`, and `rotate`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + resetNorth(options, eventData) { + this.rotateTo(0, performance.extend({ duration: 1000 }, options), eventData); + return this; + } + /** + * Rotates and pitches the map so that north is up (0° bearing) and pitch is 0°, with an animated transition. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `pitchstart`, `pitch`, `pitchend`, and `rotate`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + resetNorthPitch(options, eventData) { + this.easeTo(performance.extend({ + bearing: 0, + pitch: 0, + duration: 1000 + }, options), eventData); + return this; + } + /** + * Snaps the map so that north is up (0° bearing), if the current bearing is close enough to it (i.e. within the + * `bearingSnap` threshold). + * + * Triggers the following events: `movestart`, `moveend`, and `rotate`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + snapToNorth(options, eventData) { + if (Math.abs(this.getBearing()) < this._bearingSnap) { + return this.resetNorth(options, eventData); + } + return this; + } + /** + * Returns the map's current pitch (tilt). + * + * @returns The map's current pitch, measured in degrees away from the plane of the screen. + */ + getPitch() { return this.transform.pitch; } + /** + * Sets the map's pitch (tilt). Equivalent to `jumpTo({pitch: pitch})`. + * + * Triggers the following events: `movestart`, `moveend`, `pitchstart`, and `pitchend`. + * + * @param pitch - The pitch to set, measured in degrees away from the plane of the screen (0-60). + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + setPitch(pitch, eventData) { + this.jumpTo({ pitch }, eventData); + return this; + } + /** + * @param bounds - Calculate the center for these bounds in the viewport and use + * the highest zoom level up to and including `Map#getMaxZoom()` that fits + * in the viewport. LngLatBounds represent a box that is always axis-aligned with bearing 0. + * @param options - Options object + * @returns If map is able to fit to provided bounds, returns `center`, `zoom`, and `bearing`. + * If map is unable to fit, method will warn and return undefined. + * @example + * ```ts + * let bbox = [[-79, 43], [-73, 45]]; + * let newCameraTransform = map.cameraForBounds(bbox, { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + */ + cameraForBounds(bounds, options) { + bounds = LngLatBounds.convert(bounds); + const bearing = options && options.bearing || 0; + return this._cameraForBoxAndBearing(bounds.getNorthWest(), bounds.getSouthEast(), bearing, options); + } + /** + * @internal + * Calculate the center of these two points in the viewport and use + * the highest zoom level up to and including `Map#getMaxZoom()` that fits + * the points in the viewport at the specified bearing. + * @param p0 - First point + * @param p1 - Second point + * @param bearing - Desired map bearing at end of animation, in degrees + * @param options - the camera options + * @returns If map is able to fit to provided bounds, returns `center`, `zoom`, and `bearing`. + * If map is unable to fit, method will warn and return undefined. + * @example + * ```ts + * let p0 = [-79, 43]; + * let p1 = [-73, 45]; + * let bearing = 90; + * let newCameraTransform = map._cameraForBoxAndBearing(p0, p1, bearing, { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + */ + _cameraForBoxAndBearing(p0, p1, bearing, options) { + const defaultPadding = { + top: 0, + bottom: 0, + right: 0, + left: 0 + }; + options = performance.extend({ + padding: defaultPadding, + offset: [0, 0], + maxZoom: this.transform.maxZoom + }, options); + if (typeof options.padding === 'number') { + const p = options.padding; + options.padding = { + top: p, + bottom: p, + right: p, + left: p + }; + } + options.padding = performance.extend(defaultPadding, options.padding); + const tr = this.transform; + const edgePadding = tr.padding; + // We want to calculate the upper right and lower left of the box defined by p0 and p1 + // in a coordinate system rotate to match the destination bearing. + const p0world = tr.project(performance.LngLat.convert(p0)); + const p1world = tr.project(performance.LngLat.convert(p1)); + const p0rotated = p0world.rotate(-bearing * Math.PI / 180); + const p1rotated = p1world.rotate(-bearing * Math.PI / 180); + const upperRight = new performance.Point(Math.max(p0rotated.x, p1rotated.x), Math.max(p0rotated.y, p1rotated.y)); + const lowerLeft = new performance.Point(Math.min(p0rotated.x, p1rotated.x), Math.min(p0rotated.y, p1rotated.y)); + // Calculate zoom: consider the original bbox and padding. + const size = upperRight.sub(lowerLeft); + const scaleX = (tr.width - (edgePadding.left + edgePadding.right + options.padding.left + options.padding.right)) / size.x; + const scaleY = (tr.height - (edgePadding.top + edgePadding.bottom + options.padding.top + options.padding.bottom)) / size.y; + if (scaleY < 0 || scaleX < 0) { + performance.warnOnce('Map cannot fit within canvas with the given bounds, padding, and/or offset.'); + return undefined; + } + const zoom = Math.min(tr.scaleZoom(tr.scale * Math.min(scaleX, scaleY)), options.maxZoom); + // Calculate center: apply the zoom, the configured offset, as well as offset that exists as a result of padding. + const offset = performance.Point.convert(options.offset); + const paddingOffsetX = (options.padding.left - options.padding.right) / 2; + const paddingOffsetY = (options.padding.top - options.padding.bottom) / 2; + const paddingOffset = new performance.Point(paddingOffsetX, paddingOffsetY); + const rotatedPaddingOffset = paddingOffset.rotate(bearing * Math.PI / 180); + const offsetAtInitialZoom = offset.add(rotatedPaddingOffset); + const offsetAtFinalZoom = offsetAtInitialZoom.mult(tr.scale / tr.zoomScale(zoom)); + const center = tr.unproject(p0world.add(p1world).div(2).sub(offsetAtFinalZoom)); + return { + center, + zoom, + bearing + }; + } + /** + * Pans and zooms the map to contain its visible area within the specified geographical bounds. + * This function will also reset the map's bearing to 0 if bearing is nonzero. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param bounds - Center these bounds in the viewport and use the highest + * zoom level up to and including `Map#getMaxZoom()` that fits them in the viewport. + * @param options- Options supports all properties from {@link AnimationOptions} and {@link CameraOptions} in addition to the fields below. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * let bbox = [[-79, 43], [-73, 45]]; + * map.fitBounds(bbox, { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + * @see [Fit a map to a bounding box](https://maplibre.org/maplibre-gl-js/docs/examples/fitbounds/) + */ + fitBounds(bounds, options, eventData) { + return this._fitInternal(this.cameraForBounds(bounds, options), options, eventData); + } + /** + * Pans, rotates and zooms the map to to fit the box made by points p0 and p1 + * once the map is rotated to the specified bearing. To zoom without rotating, + * pass in the current map bearing. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend` and `rotate`. + * + * @param p0 - First point on screen, in pixel coordinates + * @param p1 - Second point on screen, in pixel coordinates + * @param bearing - Desired map bearing at end of animation, in degrees + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * let p0 = [220, 400]; + * let p1 = [500, 900]; + * map.fitScreenCoordinates(p0, p1, map.getBearing(), { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + * @see Used by {@link BoxZoomHandler} + */ + fitScreenCoordinates(p0, p1, bearing, options, eventData) { + return this._fitInternal(this._cameraForBoxAndBearing(this.transform.pointLocation(performance.Point.convert(p0)), this.transform.pointLocation(performance.Point.convert(p1)), bearing, options), options, eventData); + } + _fitInternal(calculatedOptions, options, eventData) { + // cameraForBounds warns + returns undefined if unable to fit: + if (!calculatedOptions) + return this; + options = performance.extend(calculatedOptions, options); + // Explicitly remove the padding field because, calculatedOptions already accounts for padding by setting zoom and center accordingly. + delete options.padding; + return options.linear ? + this.easeTo(options, eventData) : + this.flyTo(options, eventData); + } + /** + * Changes any combination of center, zoom, bearing, and pitch, without + * an animated transition. The map will retain its current values for any + * details not specified in `options`. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`, + * `pitch`, `pitchend`, and `rotate`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * // jump to coordinates at current zoom + * map.jumpTo({center: [0, 0]}); + * // jump with zoom, pitch, and bearing options + * map.jumpTo({ + * center: [0, 0], + * zoom: 8, + * pitch: 45, + * bearing: 90 + * }); + * ``` + * @see [Jump to a series of locations](https://maplibre.org/maplibre-gl-js/docs/examples/jump-to/) + * @see [Update a feature in realtime](https://maplibre.org/maplibre-gl-js/docs/examples/live-update-feature/) + */ + jumpTo(options, eventData) { + this.stop(); + const tr = this._getTransformForUpdate(); + let zoomChanged = false, bearingChanged = false, pitchChanged = false; + if ('zoom' in options && tr.zoom !== +options.zoom) { + zoomChanged = true; + tr.zoom = +options.zoom; + } + if (options.center !== undefined) { + tr.center = performance.LngLat.convert(options.center); + } + if ('bearing' in options && tr.bearing !== +options.bearing) { + bearingChanged = true; + tr.bearing = +options.bearing; + } + if ('pitch' in options && tr.pitch !== +options.pitch) { + pitchChanged = true; + tr.pitch = +options.pitch; + } + if (options.padding != null && !tr.isPaddingEqual(options.padding)) { + tr.padding = options.padding; + } + this._applyUpdatedTransform(tr); + this.fire(new performance.Event('movestart', eventData)) + .fire(new performance.Event('move', eventData)); + if (zoomChanged) { + this.fire(new performance.Event('zoomstart', eventData)) + .fire(new performance.Event('zoom', eventData)) + .fire(new performance.Event('zoomend', eventData)); + } + if (bearingChanged) { + this.fire(new performance.Event('rotatestart', eventData)) + .fire(new performance.Event('rotate', eventData)) + .fire(new performance.Event('rotateend', eventData)); + } + if (pitchChanged) { + this.fire(new performance.Event('pitchstart', eventData)) + .fire(new performance.Event('pitch', eventData)) + .fire(new performance.Event('pitchend', eventData)); + } + return this.fire(new performance.Event('moveend', eventData)); + } + /** + * Calculates pitch, zoom and bearing for looking at `newCenter` with the camera position being `newCenter` + * and returns them as {@link CameraOptions}. + * @param from - The camera to look from + * @param altitudeFrom - The altitude of the camera to look from + * @param to - The center to look at + * @param altitudeTo - Optional altitude of the center to look at. If none given the ground height will be used. + * @returns the calculated camera options + */ + calculateCameraOptionsFromTo(from, altitudeFrom, to, altitudeTo = 0) { + const fromMerc = performance.MercatorCoordinate.fromLngLat(from, altitudeFrom); + const toMerc = performance.MercatorCoordinate.fromLngLat(to, altitudeTo); + const dx = toMerc.x - fromMerc.x; + const dy = toMerc.y - fromMerc.y; + const dz = toMerc.z - fromMerc.z; + const distance3D = Math.hypot(dx, dy, dz); + if (distance3D === 0) + throw new Error('Can\'t calculate camera options with same From and To'); + const groundDistance = Math.hypot(dx, dy); + const zoom = this.transform.scaleZoom(this.transform.cameraToCenterDistance / distance3D / this.transform.tileSize); + const bearing = (Math.atan2(dx, -dy) * 180) / Math.PI; + let pitch = (Math.acos(groundDistance / distance3D) * 180) / Math.PI; + pitch = dz < 0 ? 90 - pitch : 90 + pitch; + return { + center: toMerc.toLngLat(), + zoom, + pitch, + bearing + }; + } + /** + * Changes any combination of `center`, `zoom`, `bearing`, `pitch`, and `padding` with an animated transition + * between old and new values. The map will retain its current values for any + * details not specified in `options`. + * + * Note: The transition will happen instantly if the user has enabled + * the `reduced motion` accessibility feature enabled in their operating system, + * unless `options` includes `essential: true`. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`, + * `pitch`, `pitchend`, and `rotate`. + * + * @param options - Options describing the destination and animation of the transition. + * Accepts {@link CameraOptions} and {@link AnimationOptions}. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/) + */ + easeTo(options, eventData) { + this._stop(false, options.easeId); + options = performance.extend({ + offset: [0, 0], + duration: 500, + easing: performance.defaultEasing + }, options); + if (options.animate === false || (!options.essential && performance.browser.prefersReducedMotion)) + options.duration = 0; + const tr = this._getTransformForUpdate(), startZoom = this.getZoom(), startBearing = this.getBearing(), startPitch = this.getPitch(), startPadding = this.getPadding(), zoom = 'zoom' in options ? +options.zoom : startZoom, bearing = 'bearing' in options ? this._normalizeBearing(options.bearing, startBearing) : startBearing, pitch = 'pitch' in options ? +options.pitch : startPitch, padding = 'padding' in options ? options.padding : tr.padding; + const offsetAsPoint = performance.Point.convert(options.offset); + let pointAtOffset = tr.centerPoint.add(offsetAsPoint); + const locationAtOffset = tr.pointLocation(pointAtOffset); + const center = performance.LngLat.convert(options.center || locationAtOffset); + this._normalizeCenter(center); + const from = tr.project(locationAtOffset); + const delta = tr.project(center).sub(from); + const finalScale = tr.zoomScale(zoom - startZoom); + let around, aroundPoint; + if (options.around) { + around = performance.LngLat.convert(options.around); + aroundPoint = tr.locationPoint(around); + } + const currently = { + moving: this._moving, + zooming: this._zooming, + rotating: this._rotating, + pitching: this._pitching + }; + this._zooming = this._zooming || (zoom !== startZoom); + this._rotating = this._rotating || (startBearing !== bearing); + this._pitching = this._pitching || (pitch !== startPitch); + this._padding = !tr.isPaddingEqual(padding); + this._easeId = options.easeId; + this._prepareEase(eventData, options.noMoveStart, currently); + if (this.terrain) + this._prepareElevation(center); + this._ease((k) => { + if (this._zooming) { + tr.zoom = performance.interpolate.number(startZoom, zoom, k); + } + if (this._rotating) { + tr.bearing = performance.interpolate.number(startBearing, bearing, k); + } + if (this._pitching) { + tr.pitch = performance.interpolate.number(startPitch, pitch, k); + } + if (this._padding) { + tr.interpolatePadding(startPadding, padding, k); + // When padding is being applied, Transform#centerPoint is changing continuously, + // thus we need to recalculate offsetPoint every frame + pointAtOffset = tr.centerPoint.add(offsetAsPoint); + } + if (this.terrain && !options.freezeElevation) + this._updateElevation(k); + if (around) { + tr.setLocationAtPoint(around, aroundPoint); + } + else { + const scale = tr.zoomScale(tr.zoom - startZoom); + const base = zoom > startZoom ? + Math.min(2, finalScale) : + Math.max(0.5, finalScale); + const speedup = Math.pow(base, 1 - k); + const newCenter = tr.unproject(from.add(delta.mult(k * speedup)).mult(scale)); + tr.setLocationAtPoint(tr.renderWorldCopies ? newCenter.wrap() : newCenter, pointAtOffset); + } + this._applyUpdatedTransform(tr); + this._fireMoveEvents(eventData); + }, (interruptingEaseId) => { + if (this.terrain) + this._finalizeElevation(); + this._afterEase(eventData, interruptingEaseId); + }, options); + return this; + } + _prepareEase(eventData, noMoveStart, currently = {}) { + this._moving = true; + if (!noMoveStart && !currently.moving) { + this.fire(new performance.Event('movestart', eventData)); + } + if (this._zooming && !currently.zooming) { + this.fire(new performance.Event('zoomstart', eventData)); + } + if (this._rotating && !currently.rotating) { + this.fire(new performance.Event('rotatestart', eventData)); + } + if (this._pitching && !currently.pitching) { + this.fire(new performance.Event('pitchstart', eventData)); + } + } + _prepareElevation(center) { + this._elevationCenter = center; + this._elevationStart = this.transform.elevation; + this._elevationTarget = this.terrain.getElevationForLngLatZoom(center, this.transform.tileZoom); + this._elevationFreeze = true; + } + _updateElevation(k) { + this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this._elevationCenter, this.transform.tileZoom); + const elevation = this.terrain.getElevationForLngLatZoom(this._elevationCenter, this.transform.tileZoom); + // target terrain updated during flight, slowly move camera to new height + if (k < 1 && elevation !== this._elevationTarget) { + const pitch1 = this._elevationTarget - this._elevationStart; + const pitch2 = (elevation - (pitch1 * k + this._elevationStart)) / (1 - k); + this._elevationStart += k * (pitch1 - pitch2); + this._elevationTarget = elevation; + } + this.transform.elevation = performance.interpolate.number(this._elevationStart, this._elevationTarget, k); + } + _finalizeElevation() { + this._elevationFreeze = false; + this.transform.recalculateZoom(this.terrain); + } + /** + * @internal + * Called when the camera is about to be manipulated. + * If `transformCameraUpdate` is specified, a copy of the current transform is created to track the accumulated changes. + * This underlying transform represents the "desired state" proposed by input handlers / animations / UI controls. + * It may differ from the state used for rendering (`this.transform`). + * @returns Transform to apply changes to + */ + _getTransformForUpdate() { + if (!this.transformCameraUpdate) + return this.transform; + if (!this._requestedCameraState) { + this._requestedCameraState = this.transform.clone(); + } + return this._requestedCameraState; + } + /** + * @internal + * Called after the camera is done being manipulated. + * @param tr - the requested camera end state + * Call `transformCameraUpdate` if present, and then apply the "approved" changes. + */ + _applyUpdatedTransform(tr) { + if (!this.transformCameraUpdate) + return; + const nextTransform = tr.clone(); + const { center, zoom, pitch, bearing, elevation } = this.transformCameraUpdate(nextTransform); + if (center) + nextTransform.center = center; + if (zoom !== undefined) + nextTransform.zoom = zoom; + if (pitch !== undefined) + nextTransform.pitch = pitch; + if (bearing !== undefined) + nextTransform.bearing = bearing; + if (elevation !== undefined) + nextTransform.elevation = elevation; + this.transform.apply(nextTransform); + } + _fireMoveEvents(eventData) { + this.fire(new performance.Event('move', eventData)); + if (this._zooming) { + this.fire(new performance.Event('zoom', eventData)); + } + if (this._rotating) { + this.fire(new performance.Event('rotate', eventData)); + } + if (this._pitching) { + this.fire(new performance.Event('pitch', eventData)); + } + } + _afterEase(eventData, easeId) { + // if this easing is being stopped to start another easing with + // the same id then don't fire any events to avoid extra start/stop events + if (this._easeId && easeId && this._easeId === easeId) { + return; + } + delete this._easeId; + const wasZooming = this._zooming; + const wasRotating = this._rotating; + const wasPitching = this._pitching; + this._moving = false; + this._zooming = false; + this._rotating = false; + this._pitching = false; + this._padding = false; + if (wasZooming) { + this.fire(new performance.Event('zoomend', eventData)); + } + if (wasRotating) { + this.fire(new performance.Event('rotateend', eventData)); + } + if (wasPitching) { + this.fire(new performance.Event('pitchend', eventData)); + } + this.fire(new performance.Event('moveend', eventData)); + } + /** + * Changes any combination of center, zoom, bearing, and pitch, animating the transition along a curve that + * evokes flight. The animation seamlessly incorporates zooming and panning to help + * the user maintain her bearings even after traversing a great distance. + * + * Note: The animation will be skipped, and this will behave equivalently to `jumpTo` + * if the user has the `reduced motion` accessibility feature enabled in their operating system, + * unless 'options' includes `essential: true`. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`, + * `pitch`, `pitchend`, and `rotate`. + * + * @param options - Options describing the destination and animation of the transition. + * Accepts {@link CameraOptions}, {@link AnimationOptions}, + * and the following additional options. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * // fly with default options to null island + * map.flyTo({center: [0, 0], zoom: 9}); + * // using flyTo options + * map.flyTo({ + * center: [0, 0], + * zoom: 9, + * speed: 0.2, + * curve: 1, + * easing(t) { + * return t; + * } + * }); + * ``` + * @see [Fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto/) + * @see [Slowly fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto-options/) + * @see [Fly to a location based on scroll position](https://maplibre.org/maplibre-gl-js/docs/examples/scroll-fly-to/) + */ + flyTo(options, eventData) { + // Fall through to jumpTo if user has set prefers-reduced-motion + if (!options.essential && performance.browser.prefersReducedMotion) { + const coercedOptions = performance.pick(options, ['center', 'zoom', 'bearing', 'pitch', 'around']); + return this.jumpTo(coercedOptions, eventData); + } + // This method implements an “optimal path” animation, as detailed in: + // + // Van Wijk, Jarke J.; Nuij, Wim A. A. “Smooth and efficient zooming and panning.” INFOVIS + // ’03. pp. 15–22. . + // + // Where applicable, local variable documentation begins with the associated variable or + // function in van Wijk (2003). + this.stop(); + options = performance.extend({ + offset: [0, 0], + speed: 1.2, + curve: 1.42, + easing: performance.defaultEasing + }, options); + const tr = this._getTransformForUpdate(), startZoom = this.getZoom(), startBearing = this.getBearing(), startPitch = this.getPitch(), startPadding = this.getPadding(); + const zoom = 'zoom' in options ? performance.clamp(+options.zoom, tr.minZoom, tr.maxZoom) : startZoom; + const bearing = 'bearing' in options ? this._normalizeBearing(options.bearing, startBearing) : startBearing; + const pitch = 'pitch' in options ? +options.pitch : startPitch; + const padding = 'padding' in options ? options.padding : tr.padding; + const scale = tr.zoomScale(zoom - startZoom); + const offsetAsPoint = performance.Point.convert(options.offset); + let pointAtOffset = tr.centerPoint.add(offsetAsPoint); + const locationAtOffset = tr.pointLocation(pointAtOffset); + const center = performance.LngLat.convert(options.center || locationAtOffset); + this._normalizeCenter(center); + const from = tr.project(locationAtOffset); + const delta = tr.project(center).sub(from); + let rho = options.curve; + // w₀: Initial visible span, measured in pixels at the initial scale. + const w0 = Math.max(tr.width, tr.height), + // w₁: Final visible span, measured in pixels with respect to the initial scale. + w1 = w0 / scale, + // Length of the flight path as projected onto the ground plane, measured in pixels from + // the world image origin at the initial scale. + u1 = delta.mag(); + if ('minZoom' in options) { + const minZoom = performance.clamp(Math.min(options.minZoom, startZoom, zoom), tr.minZoom, tr.maxZoom); + // wm: Maximum visible span, measured in pixels with respect to the initial + // scale. + const wMax = w0 / tr.zoomScale(minZoom - startZoom); + rho = Math.sqrt(wMax / u1 * 2); + } + // ρ² + const rho2 = rho * rho; + /** + * rᵢ: Returns the zoom-out factor at one end of the animation. + * + * @param descent - `true` for the descent, `false` for the ascent + */ + function zoomOutFactor(descent) { + const b = (w1 * w1 - w0 * w0 + (descent ? -1 : 1) * rho2 * rho2 * u1 * u1) / (2 * (descent ? w1 : w0) * rho2 * u1); + return Math.log(Math.sqrt(b * b + 1) - b); + } + function sinh(n) { return (Math.exp(n) - Math.exp(-n)) / 2; } + function cosh(n) { return (Math.exp(n) + Math.exp(-n)) / 2; } + function tanh(n) { return sinh(n) / cosh(n); } + // r₀: Zoom-out factor during ascent. + const r0 = zoomOutFactor(false); + // w(s): Returns the visible span on the ground, measured in pixels with respect to the + // initial scale. Assumes an angular field of view of 2 arctan ½ ≈ 53°. + let w = function (s) { + return (cosh(r0) / cosh(r0 + rho * s)); + }; + // u(s): Returns the distance along the flight path as projected onto the ground plane, + // measured in pixels from the world image origin at the initial scale. + let u = function (s) { + return w0 * ((cosh(r0) * tanh(r0 + rho * s) - sinh(r0)) / rho2) / u1; + }; + // S: Total length of the flight path, measured in ρ-screenfuls. + let S = (zoomOutFactor(true) - r0) / rho; + // When u₀ = u₁, the optimal path doesn’t require both ascent and descent. + if (Math.abs(u1) < 0.000001 || !isFinite(S)) { + // Perform a more or less instantaneous transition if the path is too short. + if (Math.abs(w0 - w1) < 0.000001) + return this.easeTo(options, eventData); + const k = w1 < w0 ? -1 : 1; + S = Math.abs(Math.log(w1 / w0)) / rho; + u = function () { return 0; }; + w = function (s) { return Math.exp(k * rho * s); }; + } + if ('duration' in options) { + options.duration = +options.duration; + } + else { + const V = 'screenSpeed' in options ? +options.screenSpeed / rho : +options.speed; + options.duration = 1000 * S / V; + } + if (options.maxDuration && options.duration > options.maxDuration) { + options.duration = 0; + } + this._zooming = true; + this._rotating = (startBearing !== bearing); + this._pitching = (pitch !== startPitch); + this._padding = !tr.isPaddingEqual(padding); + this._prepareEase(eventData, false); + if (this.terrain) + this._prepareElevation(center); + this._ease((k) => { + // s: The distance traveled along the flight path, measured in ρ-screenfuls. + const s = k * S; + const scale = 1 / w(s); + tr.zoom = k === 1 ? zoom : startZoom + tr.scaleZoom(scale); + if (this._rotating) { + tr.bearing = performance.interpolate.number(startBearing, bearing, k); + } + if (this._pitching) { + tr.pitch = performance.interpolate.number(startPitch, pitch, k); + } + if (this._padding) { + tr.interpolatePadding(startPadding, padding, k); + // When padding is being applied, Transform#centerPoint is changing continuously, + // thus we need to recalculate offsetPoint every frame + pointAtOffset = tr.centerPoint.add(offsetAsPoint); + } + if (this.terrain && !options.freezeElevation) + this._updateElevation(k); + const newCenter = k === 1 ? center : tr.unproject(from.add(delta.mult(u(s))).mult(scale)); + tr.setLocationAtPoint(tr.renderWorldCopies ? newCenter.wrap() : newCenter, pointAtOffset); + this._applyUpdatedTransform(tr); + this._fireMoveEvents(eventData); + }, () => { + if (this.terrain) + this._finalizeElevation(); + this._afterEase(eventData); + }, options); + return this; + } + isEasing() { + return !!this._easeFrameId; + } + /** + * Stops any animated transition underway. + * + * @returns `this` + */ + stop() { + return this._stop(); + } + _stop(allowGestures, easeId) { + if (this._easeFrameId) { + this._cancelRenderFrame(this._easeFrameId); + delete this._easeFrameId; + delete this._onEaseFrame; + } + if (this._onEaseEnd) { + // The _onEaseEnd function might emit events which trigger new + // animation, which sets a new _onEaseEnd. Ensure we don't delete + // it unintentionally. + const onEaseEnd = this._onEaseEnd; + delete this._onEaseEnd; + onEaseEnd.call(this, easeId); + } + if (!allowGestures) { + const handlers = this.handlers; + if (handlers) + handlers.stop(false); + } + return this; + } + _ease(frame, finish, options) { + if (options.animate === false || options.duration === 0) { + frame(1); + finish(); + } + else { + this._easeStart = performance.browser.now(); + this._easeOptions = options; + this._onEaseFrame = frame; + this._onEaseEnd = finish; + this._easeFrameId = this._requestRenderFrame(this._renderFrameCallback); + } + } + // convert bearing so that it's numerically close to the current one so that it interpolates properly + _normalizeBearing(bearing, currentBearing) { + bearing = performance.wrap(bearing, -180, 180); + const diff = Math.abs(bearing - currentBearing); + if (Math.abs(bearing - 360 - currentBearing) < diff) + bearing -= 360; + if (Math.abs(bearing + 360 - currentBearing) < diff) + bearing += 360; + return bearing; + } + // If a path crossing the antimeridian would be shorter, extend the final coordinate so that + // interpolating between the two endpoints will cross it. + _normalizeCenter(center) { + const tr = this.transform; + if (!tr.renderWorldCopies || tr.lngRange) + return; + const delta = center.lng - tr.center.lng; + center.lng += + delta > 180 ? -360 : + delta < -180 ? 360 : 0; + } + /** + * Query the current elevation of location. It return null if terrain is not enabled. the elevation is in meters relative to mean sea-level + * @param lngLatLike - [x,y] or LngLat coordinates of the location + * @returns elevation in meters + */ + queryTerrainElevation(lngLatLike) { + if (!this.terrain) { + return null; + } + const elevation = this.terrain.getElevationForLngLatZoom(performance.LngLat.convert(lngLatLike), this.transform.tileZoom); + /** + * Different zoomlevels with different terrain-tiles the elevation-values are not the same. + * map.transform.elevation variable with the center-altitude. + * In maplibre the proj-matrix is translated by this value in negative z-direction. + * So we need to add this value to the elevation to get the correct value. + */ + return elevation - this.transform.elevation; + } +} + +/** + * An `AttributionControl` control presents the map's attribution information. By default, the attribution control is expanded (regardless of map width). + * @group Markers and Controls + * @example + * ```ts + * let map = new maplibregl.Map({attributionControl: false}) + * .addControl(new maplibregl.AttributionControl({ + * compact: true + * })); + * ``` + */ +class AttributionControl { + /** + * @param options - the attribution options + */ + constructor(options = {}) { + this._toggleAttribution = () => { + if (this._container.classList.contains('maplibregl-compact')) { + if (this._container.classList.contains('maplibregl-compact-show')) { + this._container.setAttribute('open', ''); + this._container.classList.remove('maplibregl-compact-show'); + } + else { + this._container.classList.add('maplibregl-compact-show'); + this._container.removeAttribute('open'); + } + } + }; + this._updateData = (e) => { + if (e && (e.sourceDataType === 'metadata' || e.sourceDataType === 'visibility' || e.dataType === 'style' || e.type === 'terrain')) { + this._updateAttributions(); + } + }; + this._updateCompact = () => { + if (this._map.getCanvasContainer().offsetWidth <= 640 || this._compact) { + if (this._compact === false) { + this._container.setAttribute('open', ''); + } + else if (!this._container.classList.contains('maplibregl-compact') && !this._container.classList.contains('maplibregl-attrib-empty')) { + this._container.setAttribute('open', ''); + this._container.classList.add('maplibregl-compact', 'maplibregl-compact-show'); + } + } + else { + this._container.setAttribute('open', ''); + if (this._container.classList.contains('maplibregl-compact')) { + this._container.classList.remove('maplibregl-compact', 'maplibregl-compact-show'); + } + } + }; + this._updateCompactMinimize = () => { + if (this._container.classList.contains('maplibregl-compact')) { + if (this._container.classList.contains('maplibregl-compact-show')) { + this._container.classList.remove('maplibregl-compact-show'); + } + } + }; + this.options = options; + } + getDefaultPosition() { + return 'bottom-right'; + } + /** {@inheritDoc IControl.onAdd} */ + onAdd(map) { + this._map = map; + this._compact = this.options && this.options.compact; + this._container = DOM.create('details', 'maplibregl-ctrl maplibregl-ctrl-attrib'); + this._compactButton = DOM.create('summary', 'maplibregl-ctrl-attrib-button', this._container); + this._compactButton.addEventListener('click', this._toggleAttribution); + this._setElementTitle(this._compactButton, 'ToggleAttribution'); + this._innerContainer = DOM.create('div', 'maplibregl-ctrl-attrib-inner', this._container); + this._updateAttributions(); + this._updateCompact(); + this._map.on('styledata', this._updateData); + this._map.on('sourcedata', this._updateData); + this._map.on('terrain', this._updateData); + this._map.on('resize', this._updateCompact); + this._map.on('drag', this._updateCompactMinimize); + return this._container; + } + /** {@inheritDoc IControl.onRemove} */ + onRemove() { + DOM.remove(this._container); + this._map.off('styledata', this._updateData); + this._map.off('sourcedata', this._updateData); + this._map.off('terrain', this._updateData); + this._map.off('resize', this._updateCompact); + this._map.off('drag', this._updateCompactMinimize); + this._map = undefined; + this._compact = undefined; + this._attribHTML = undefined; + } + _setElementTitle(element, title) { + const str = this._map._getUIString(`AttributionControl.${title}`); + element.title = str; + element.setAttribute('aria-label', str); + } + _updateAttributions() { + if (!this._map.style) + return; + let attributions = []; + if (this.options.customAttribution) { + if (Array.isArray(this.options.customAttribution)) { + attributions = attributions.concat(this.options.customAttribution.map(attribution => { + if (typeof attribution !== 'string') + return ''; + return attribution; + })); + } + else if (typeof this.options.customAttribution === 'string') { + attributions.push(this.options.customAttribution); + } + } + if (this._map.style.stylesheet) { + const stylesheet = this._map.style.stylesheet; + this.styleOwner = stylesheet.owner; + this.styleId = stylesheet.id; + } + const sourceCaches = this._map.style.sourceCaches; + for (const id in sourceCaches) { + const sourceCache = sourceCaches[id]; + if (sourceCache.used || sourceCache.usedForTerrain) { + const source = sourceCache.getSource(); + if (source.attribution && attributions.indexOf(source.attribution) < 0) { + attributions.push(source.attribution); + } + } + } + // remove any entries that are whitespace + attributions = attributions.filter(e => String(e).trim()); + // remove any entries that are substrings of another entry. + // first sort by length so that substrings come first + attributions.sort((a, b) => a.length - b.length); + attributions = attributions.filter((attrib, i) => { + for (let j = i + 1; j < attributions.length; j++) { + if (attributions[j].indexOf(attrib) >= 0) { + return false; + } + } + return true; + }); + // check if attribution string is different to minimize DOM changes + const attribHTML = attributions.join(' | '); + if (attribHTML === this._attribHTML) + return; + this._attribHTML = attribHTML; + if (attributions.length) { + this._innerContainer.innerHTML = attribHTML; + this._container.classList.remove('maplibregl-attrib-empty'); + } + else { + this._container.classList.add('maplibregl-attrib-empty'); + } + this._updateCompact(); + // remove old DOM node from _editLink + this._editLink = null; + } +} + +/** + * A `LogoControl` is a control that adds the watermark. + * + * @group Markers and Controls + * + * @example + * ```ts + * map.addControl(new maplibregl.LogoControl({compact: false})); + * ``` + **/ +class LogoControl { + constructor(options = {}) { + this._updateCompact = () => { + const containerChildren = this._container.children; + if (containerChildren.length) { + const anchor = containerChildren[0]; + if (this._map.getCanvasContainer().offsetWidth <= 640 || this._compact) { + if (this._compact !== false) { + anchor.classList.add('maplibregl-compact'); + } + } + else { + anchor.classList.remove('maplibregl-compact'); + } + } + }; + this.options = options; + } + getDefaultPosition() { + return 'bottom-left'; + } + /** {@inheritDoc IControl.onAdd} */ + onAdd(map) { + this._map = map; + this._compact = this.options && this.options.compact; + this._container = DOM.create('div', 'maplibregl-ctrl'); + const anchor = DOM.create('a', 'maplibregl-ctrl-logo'); + anchor.target = '_blank'; + anchor.rel = 'noopener nofollow'; + anchor.href = 'https://maplibre.org/'; + anchor.setAttribute('aria-label', this._map._getUIString('LogoControl.Title')); + anchor.setAttribute('rel', 'noopener nofollow'); + this._container.appendChild(anchor); + this._container.style.display = 'block'; + this._map.on('resize', this._updateCompact); + this._updateCompact(); + return this._container; + } + /** {@inheritDoc IControl.onRemove} */ + onRemove() { + DOM.remove(this._container); + this._map.off('resize', this._updateCompact); + this._map = undefined; + this._compact = undefined; + } +} + +class TaskQueue { + constructor() { + this._queue = []; + this._id = 0; + this._cleared = false; + this._currentlyRunning = false; + } + add(callback) { + const id = ++this._id; + const queue = this._queue; + queue.push({ callback, id, cancelled: false }); + return id; + } + remove(id) { + const running = this._currentlyRunning; + const queue = running ? this._queue.concat(running) : this._queue; + for (const task of queue) { + if (task.id === id) { + task.cancelled = true; + return; + } + } + } + run(timeStamp = 0) { + if (this._currentlyRunning) + throw new Error('Attempting to run(), but is already running.'); + const queue = this._currentlyRunning = this._queue; + // Tasks queued by callbacks in the current queue should be executed + // on the next run, not the current run. + this._queue = []; + for (const task of queue) { + if (task.cancelled) + continue; + task.callback(timeStamp); + if (this._cleared) + break; + } + this._cleared = false; + this._currentlyRunning = false; + } + clear() { + if (this._currentlyRunning) { + this._cleared = true; + } + this._queue = []; + } +} + +const defaultLocale = { + 'AttributionControl.ToggleAttribution': 'Toggle attribution', + 'AttributionControl.MapFeedback': 'Map feedback', + 'FullscreenControl.Enter': 'Enter fullscreen', + 'FullscreenControl.Exit': 'Exit fullscreen', + 'GeolocateControl.FindMyLocation': 'Find my location', + 'GeolocateControl.LocationNotAvailable': 'Location not available', + 'LogoControl.Title': 'Mapbox logo', + 'NavigationControl.ResetBearing': 'Reset bearing to north', + 'NavigationControl.ZoomIn': 'Zoom in', + 'NavigationControl.ZoomOut': 'Zoom out', + 'ScaleControl.Feet': 'ft', + 'ScaleControl.Meters': 'm', + 'ScaleControl.Kilometers': 'km', + 'ScaleControl.Miles': 'mi', + 'ScaleControl.NauticalMiles': 'nm', + 'TerrainControl.enableTerrain': 'Enable terrain', + 'TerrainControl.disableTerrain': 'Disable terrain' +}; + +var pos3dAttributes = performance.createLayout([ + { name: 'a_pos3d', type: 'Int16', components: 3 } +]); + +/** + * @internal + * This class is a helper for the Terrain-class, it: + * - loads raster-dem tiles + * - manages all renderToTexture tiles. + * - caches previous rendered tiles. + * - finds all necessary renderToTexture tiles for a OverscaledTileID area + * - finds the corresponding raster-dem tile for OverscaledTileID + */ +class TerrainSourceCache extends performance.Evented { + constructor(sourceCache) { + super(); + this.sourceCache = sourceCache; + this._tiles = {}; + this._renderableTilesKeys = []; + this._sourceTileCache = {}; + this.minzoom = 0; + this.maxzoom = 22; + this.tileSize = 512; + this.deltaZoom = 1; + sourceCache.usedForTerrain = true; + sourceCache.tileSize = this.tileSize * 2 ** this.deltaZoom; + } + destruct() { + this.sourceCache.usedForTerrain = false; + this.sourceCache.tileSize = null; + } + /** + * Load Terrain Tiles, create internal render-to-texture tiles, free GPU memory. + * @param transform - the operation to do + * @param terrain - the terrain + */ + update(transform, terrain) { + // load raster-dem tiles for the current scene. + this.sourceCache.update(transform, terrain); + // create internal render-to-texture tiles for the current scene. + this._renderableTilesKeys = []; + const keys = {}; + for (const tileID of transform.coveringTiles({ + tileSize: this.tileSize, + minzoom: this.minzoom, + maxzoom: this.maxzoom, + reparseOverscaled: false, + terrain + })) { + keys[tileID.key] = true; + this._renderableTilesKeys.push(tileID.key); + if (!this._tiles[tileID.key]) { + tileID.posMatrix = new Float64Array(16); + performance.ortho(tileID.posMatrix, 0, performance.EXTENT, 0, performance.EXTENT, 0, 1); + this._tiles[tileID.key] = new Tile(tileID, this.tileSize); + } + } + // free unused tiles + for (const key in this._tiles) { + if (!keys[key]) + delete this._tiles[key]; + } + } + /** + * Free render to texture cache + * @param tileID - optional, free only corresponding to tileID. + */ + freeRtt(tileID) { + for (const key in this._tiles) { + const tile = this._tiles[key]; + if (!tileID || tile.tileID.equals(tileID) || tile.tileID.isChildOf(tileID) || tileID.isChildOf(tile.tileID)) + tile.rtt = []; + } + } + /** + * get a list of tiles, which are loaded and should be rendered in the current scene + * @returns the renderable tiles + */ + getRenderableTiles() { + return this._renderableTilesKeys.map(key => this.getTileByID(key)); + } + /** + * get terrain tile by the TileID key + * @param id - the tile id + * @returns the tile + */ + getTileByID(id) { + return this._tiles[id]; + } + /** + * Searches for the corresponding current renderable terrain-tiles + * @param tileID - the tile to look for + * @returns the tiles that were found + */ + getTerrainCoords(tileID) { + const coords = {}; + for (const key of this._renderableTilesKeys) { + const _tileID = this._tiles[key].tileID; + if (_tileID.canonical.equals(tileID.canonical)) { + const coord = tileID.clone(); + coord.posMatrix = new Float64Array(16); + performance.ortho(coord.posMatrix, 0, performance.EXTENT, 0, performance.EXTENT, 0, 1); + coords[key] = coord; + } + else if (_tileID.canonical.isChildOf(tileID.canonical)) { + const coord = tileID.clone(); + coord.posMatrix = new Float64Array(16); + const dz = _tileID.canonical.z - tileID.canonical.z; + const dx = _tileID.canonical.x - (_tileID.canonical.x >> dz << dz); + const dy = _tileID.canonical.y - (_tileID.canonical.y >> dz << dz); + const size = performance.EXTENT >> dz; + performance.ortho(coord.posMatrix, 0, size, 0, size, 0, 1); + performance.translate(coord.posMatrix, coord.posMatrix, [-dx * size, -dy * size, 0]); + coords[key] = coord; + } + else if (tileID.canonical.isChildOf(_tileID.canonical)) { + const coord = tileID.clone(); + coord.posMatrix = new Float64Array(16); + const dz = tileID.canonical.z - _tileID.canonical.z; + const dx = tileID.canonical.x - (tileID.canonical.x >> dz << dz); + const dy = tileID.canonical.y - (tileID.canonical.y >> dz << dz); + const size = performance.EXTENT >> dz; + performance.ortho(coord.posMatrix, 0, performance.EXTENT, 0, performance.EXTENT, 0, 1); + performance.translate(coord.posMatrix, coord.posMatrix, [dx * size, dy * size, 0]); + performance.scale(coord.posMatrix, coord.posMatrix, [1 / (2 ** dz), 1 / (2 ** dz), 0]); + coords[key] = coord; + } + } + return coords; + } + /** + * find the covering raster-dem tile + * @param tileID - the tile to look for + * @param searchForDEM - Optinal parameter to search for (parent) souretiles with loaded dem. + * @returns the tile + */ + getSourceTile(tileID, searchForDEM) { + const source = this.sourceCache._source; + let z = tileID.overscaledZ - this.deltaZoom; + if (z > source.maxzoom) + z = source.maxzoom; + if (z < source.minzoom) + return null; + // cache for tileID to terrain-tileID + if (!this._sourceTileCache[tileID.key]) + this._sourceTileCache[tileID.key] = tileID.scaledTo(z).key; + let tile = this.sourceCache.getTileByID(this._sourceTileCache[tileID.key]); + // during tile-loading phase look if parent tiles (with loaded dem) are available. + if (!(tile && tile.dem) && searchForDEM) + while (z >= source.minzoom && !(tile && tile.dem)) + tile = this.sourceCache.getTileByID(tileID.scaledTo(z--).key); + return tile; + } + /** + * get a list of tiles, loaded after a spezific time. This is used to update depth & coords framebuffers. + * @param time - the time + * @returns the relevant tiles + */ + tilesAfterTime(time = Date.now()) { + return Object.values(this._tiles).filter(t => t.timeAdded >= time); + } +} + +/** + * @internal + * This is the main class which handles most of the 3D Terrain logic. It has the following topics: + * 1) loads raster-dem tiles via the internal sourceCache this.sourceCache + * 2) creates a depth-framebuffer, which is used to calculate the visibility of coordinates + * 3) creates a coords-framebuffer, which is used the get to tile-coordinate for a screen-pixel + * 4) stores all render-to-texture tiles in the this.sourceCache._tiles + * 5) calculates the elevation for a specific tile-coordinate + * 6) creates a terrain-mesh + * + * A note about the GPU resource-usage: + * Framebuffers: + * - one for the depth & coords framebuffer with the size of the map-div. + * - one for rendering a tile to texture with the size of tileSize (= 512x512). + * Textures: + * - one texture for an empty raster-dem tile with size 1x1 + * - one texture for an empty depth-buffer, when terrain is disabled with size 1x1 + * - one texture for an each loaded raster-dem with size of the source.tileSize + * - one texture for the coords-framebuffer with the size of the map-div. + * - one texture for the depth-framebuffer with the size of the map-div. + * - one texture for the encoded tile-coords with the size 2*tileSize (=1024x1024) + * - finally for each render-to-texture tile (= this._tiles) a set of textures + * for each render stack (The stack-concept is documented in painter.ts). + * Normally there exists 1-3 Textures per tile, depending on the stylesheet. + * Each Textures has the size 2*tileSize (= 1024x1024). Also there exists a + * cache of the last 150 newest rendered tiles. + * + */ +class Terrain { + constructor(painter, sourceCache, options) { + this.painter = painter; + this.sourceCache = new TerrainSourceCache(sourceCache); + this.options = options; + this.exaggeration = typeof options.exaggeration === 'number' ? options.exaggeration : 1.0; + this.qualityFactor = 2; + this.meshSize = 128; + this._demMatrixCache = {}; + this.coordsIndex = []; + this._coordsTextureSize = 1024; + } + /** + * get the elevation-value from original dem-data for a given tile-coordinate + * @param tileID - the tile to get elevation for + * @param x - between 0 .. EXTENT + * @param y - between 0 .. EXTENT + * @param extent - optional, default 8192 + * @returns the elevation + */ + getDEMElevation(tileID, x, y, extent = performance.EXTENT) { + var _a; + if (!(x >= 0 && x < extent && y >= 0 && y < extent)) + return 0; + const terrain = this.getTerrainData(tileID); + const dem = (_a = terrain.tile) === null || _a === void 0 ? void 0 : _a.dem; + if (!dem) + return 0; + const pos = performance.transformMat4$1([], [x / extent * performance.EXTENT, y / extent * performance.EXTENT], terrain.u_terrain_matrix); + const coord = [pos[0] * dem.dim, pos[1] * dem.dim]; + // bilinear interpolation + const cx = Math.floor(coord[0]), cy = Math.floor(coord[1]), tx = coord[0] - cx, ty = coord[1] - cy; + return (dem.get(cx, cy) * (1 - tx) * (1 - ty) + + dem.get(cx + 1, cy) * (tx) * (1 - ty) + + dem.get(cx, cy + 1) * (1 - tx) * (ty) + + dem.get(cx + 1, cy + 1) * (tx) * (ty)); + } + /** + * Get the elevation for given {@link LngLat} in respect of exaggeration. + * @param lnglat - the location + * @param zoom - the zoom + * @returns the elevation + */ + getElevationForLngLatZoom(lnglat, zoom) { + const { tileID, mercatorX, mercatorY } = this._getOverscaledTileIDFromLngLatZoom(lnglat, zoom); + return this.getElevation(tileID, mercatorX % performance.EXTENT, mercatorY % performance.EXTENT, performance.EXTENT); + } + /** + * Get the elevation for given coordinate in respect of exaggeration. + * @param tileID - the tile id + * @param x - between 0 .. EXTENT + * @param y - between 0 .. EXTENT + * @param extent - optional, default 8192 + * @returns the elevation + */ + getElevation(tileID, x, y, extent = performance.EXTENT) { + return this.getDEMElevation(tileID, x, y, extent) * this.exaggeration; + } + /** + * returns a Terrain Object for a tile. Unless the tile corresponds to data (e.g. tile is loading), return a flat dem object + * @param tileID - the tile to get the terrain for + * @returns the terrain data to use in the program + */ + getTerrainData(tileID) { + // create empty DEM Objects, which will used while raster-dem tiles are loading. + // creates an empty depth-buffer texture which is needed, during the initialization process of the 3d mesh.. + if (!this._emptyDemTexture) { + const context = this.painter.context; + const image = new performance.RGBAImage({ width: 1, height: 1 }, new Uint8Array(1 * 4)); + this._emptyDepthTexture = new Texture(context, image, context.gl.RGBA, { premultiply: false }); + this._emptyDemUnpack = [0, 0, 0, 0]; + this._emptyDemTexture = new Texture(context, new performance.RGBAImage({ width: 1, height: 1 }), context.gl.RGBA, { premultiply: false }); + this._emptyDemTexture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE); + this._emptyDemMatrix = performance.identity([]); + } + // find covering dem tile and prepare demTexture + const sourceTile = this.sourceCache.getSourceTile(tileID, true); + if (sourceTile && sourceTile.dem && (!sourceTile.demTexture || sourceTile.needsTerrainPrepare)) { + const context = this.painter.context; + sourceTile.demTexture = this.painter.getTileTexture(sourceTile.dem.stride); + if (sourceTile.demTexture) + sourceTile.demTexture.update(sourceTile.dem.getPixels(), { premultiply: false }); + else + sourceTile.demTexture = new Texture(context, sourceTile.dem.getPixels(), context.gl.RGBA, { premultiply: false }); + sourceTile.demTexture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE); + sourceTile.needsTerrainPrepare = false; + } + // create matrix for lookup in dem data + const matrixKey = sourceTile && (sourceTile + sourceTile.tileID.key) + tileID.key; + if (matrixKey && !this._demMatrixCache[matrixKey]) { + const maxzoom = this.sourceCache.sourceCache._source.maxzoom; + let dz = tileID.canonical.z - sourceTile.tileID.canonical.z; + if (tileID.overscaledZ > tileID.canonical.z) { + if (tileID.canonical.z >= maxzoom) + dz = tileID.canonical.z - maxzoom; + else + performance.warnOnce('cannot calculate elevation if elevation maxzoom > source.maxzoom'); + } + const dx = tileID.canonical.x - (tileID.canonical.x >> dz << dz); + const dy = tileID.canonical.y - (tileID.canonical.y >> dz << dz); + const demMatrix = performance.fromScaling(new Float64Array(16), [1 / (performance.EXTENT << dz), 1 / (performance.EXTENT << dz), 0]); + performance.translate(demMatrix, demMatrix, [dx * performance.EXTENT, dy * performance.EXTENT, 0]); + this._demMatrixCache[tileID.key] = { matrix: demMatrix, coord: tileID }; + } + // return uniform values & textures + return { + 'u_depth': 2, + 'u_terrain': 3, + 'u_terrain_dim': sourceTile && sourceTile.dem && sourceTile.dem.dim || 1, + 'u_terrain_matrix': matrixKey ? this._demMatrixCache[tileID.key].matrix : this._emptyDemMatrix, + 'u_terrain_unpack': sourceTile && sourceTile.dem && sourceTile.dem.getUnpackVector() || this._emptyDemUnpack, + 'u_terrain_exaggeration': this.exaggeration, + texture: (sourceTile && sourceTile.demTexture || this._emptyDemTexture).texture, + depthTexture: (this._fboDepthTexture || this._emptyDepthTexture).texture, + tile: sourceTile + }; + } + /** + * get a framebuffer as big as the map-div, which will be used to render depth & coords into a texture + * @param texture - the texture + * @returns the frame buffer + */ + getFramebuffer(texture) { + const painter = this.painter; + const width = painter.width / devicePixelRatio; + const height = painter.height / devicePixelRatio; + if (this._fbo && (this._fbo.width !== width || this._fbo.height !== height)) { + this._fbo.destroy(); + this._fboCoordsTexture.destroy(); + this._fboDepthTexture.destroy(); + delete this._fbo; + delete this._fboDepthTexture; + delete this._fboCoordsTexture; + } + if (!this._fboCoordsTexture) { + this._fboCoordsTexture = new Texture(painter.context, { width, height, data: null }, painter.context.gl.RGBA, { premultiply: false }); + this._fboCoordsTexture.bind(painter.context.gl.NEAREST, painter.context.gl.CLAMP_TO_EDGE); + } + if (!this._fboDepthTexture) { + this._fboDepthTexture = new Texture(painter.context, { width, height, data: null }, painter.context.gl.RGBA, { premultiply: false }); + this._fboDepthTexture.bind(painter.context.gl.NEAREST, painter.context.gl.CLAMP_TO_EDGE); + } + if (!this._fbo) { + this._fbo = painter.context.createFramebuffer(width, height, true, false); + this._fbo.depthAttachment.set(painter.context.createRenderbuffer(painter.context.gl.DEPTH_COMPONENT16, width, height)); + } + this._fbo.colorAttachment.set(texture === 'coords' ? this._fboCoordsTexture.texture : this._fboDepthTexture.texture); + return this._fbo; + } + /** + * create coords texture, needed to grab coordinates from canvas + * encode coords coordinate into 4 bytes: + * - 8 lower bits for x + * - 8 lower bits for y + * - 4 higher bits for x + * - 4 higher bits for y + * - 8 bits for coordsIndex (1 .. 255) (= number of terraintile), is later setted in draw_terrain uniform value + * @returns the texture + */ + getCoordsTexture() { + const context = this.painter.context; + if (this._coordsTexture) + return this._coordsTexture; + const data = new Uint8Array(this._coordsTextureSize * this._coordsTextureSize * 4); + for (let y = 0, i = 0; y < this._coordsTextureSize; y++) + for (let x = 0; x < this._coordsTextureSize; x++, i += 4) { + data[i + 0] = x & 255; + data[i + 1] = y & 255; + data[i + 2] = ((x >> 8) << 4) | (y >> 8); + data[i + 3] = 0; + } + const image = new performance.RGBAImage({ width: this._coordsTextureSize, height: this._coordsTextureSize }, new Uint8Array(data.buffer)); + const texture = new Texture(context, image, context.gl.RGBA, { premultiply: false }); + texture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE); + this._coordsTexture = texture; + return texture; + } + /** + * Reads a pixel from the coords-framebuffer and translate this to mercator. + * @param p - Screen-Coordinate + * @returns mercator coordinate for a screen pixel + */ + pointCoordinate(p) { + const rgba = new Uint8Array(4); + const context = this.painter.context, gl = context.gl; + // grab coordinate pixel from coordinates framebuffer + context.bindFramebuffer.set(this.getFramebuffer('coords').framebuffer); + gl.readPixels(p.x, this.painter.height / devicePixelRatio - p.y - 1, 1, 1, gl.RGBA, gl.UNSIGNED_BYTE, rgba); + context.bindFramebuffer.set(null); + // decode coordinates (encoding see getCoordsTexture) + const x = rgba[0] + ((rgba[2] >> 4) << 8); + const y = rgba[1] + ((rgba[2] & 15) << 8); + const tileID = this.coordsIndex[255 - rgba[3]]; + const tile = tileID && this.sourceCache.getTileByID(tileID); + if (!tile) + return null; + const coordsSize = this._coordsTextureSize; + const worldSize = (1 << tile.tileID.canonical.z) * coordsSize; + const mercatorX = (tile.tileID.canonical.x * coordsSize + x) / worldSize; + return new performance.MercatorCoordinate(this._allowMercatorOverflow(p, mercatorX), (tile.tileID.canonical.y * coordsSize + y) / worldSize, this.getElevation(tile.tileID, x, y, coordsSize)); + } + /** + * create a regular mesh which will be used by all terrain-tiles + * @returns the created regular mesh + */ + getTerrainMesh() { + if (this._mesh) + return this._mesh; + const context = this.painter.context; + const vertexArray = new performance.Pos3dArray(); + const indexArray = new performance.TriangleIndexArray(); + const meshSize = this.meshSize; + const delta = performance.EXTENT / meshSize; + const meshSize2 = meshSize * meshSize; + for (let y = 0; y <= meshSize; y++) + for (let x = 0; x <= meshSize; x++) + vertexArray.emplaceBack(x * delta, y * delta, 0); + for (let y = 0; y < meshSize2; y += meshSize + 1) + for (let x = 0; x < meshSize; x++) { + indexArray.emplaceBack(x + y, meshSize + x + y + 1, meshSize + x + y + 2); + indexArray.emplaceBack(x + y, meshSize + x + y + 2, x + y + 1); + } + // add an extra frame around the mesh to avoid stiching on tile boundaries with different zoomlevels + // first code-block is for top-bottom frame and second for left-right frame + const offsetTop = vertexArray.length, offsetBottom = offsetTop + (meshSize + 1) * 2; + for (const y of [0, 1]) + for (let x = 0; x <= meshSize; x++) + for (const z of [0, 1]) + vertexArray.emplaceBack(x * delta, y * performance.EXTENT, z); + for (let x = 0; x < meshSize * 2; x += 2) { + indexArray.emplaceBack(offsetBottom + x, offsetBottom + x + 1, offsetBottom + x + 3); + indexArray.emplaceBack(offsetBottom + x, offsetBottom + x + 3, offsetBottom + x + 2); + indexArray.emplaceBack(offsetTop + x, offsetTop + x + 3, offsetTop + x + 1); + indexArray.emplaceBack(offsetTop + x, offsetTop + x + 2, offsetTop + x + 3); + } + const offsetLeft = vertexArray.length, offsetRight = offsetLeft + (meshSize + 1) * 2; + for (const x of [0, 1]) + for (let y = 0; y <= meshSize; y++) + for (const z of [0, 1]) + vertexArray.emplaceBack(x * performance.EXTENT, y * delta, z); + for (let y = 0; y < meshSize * 2; y += 2) { + indexArray.emplaceBack(offsetLeft + y, offsetLeft + y + 1, offsetLeft + y + 3); + indexArray.emplaceBack(offsetLeft + y, offsetLeft + y + 3, offsetLeft + y + 2); + indexArray.emplaceBack(offsetRight + y, offsetRight + y + 3, offsetRight + y + 1); + indexArray.emplaceBack(offsetRight + y, offsetRight + y + 2, offsetRight + y + 3); + } + this._mesh = { + indexBuffer: context.createIndexBuffer(indexArray), + vertexBuffer: context.createVertexBuffer(vertexArray, pos3dAttributes.members), + segments: performance.SegmentVector.simpleSegment(0, 0, vertexArray.length, indexArray.length) + }; + return this._mesh; + } + /** + * Calculates a height of the frame around the terrain-mesh to avoid stiching between + * tile boundaries in different zoomlevels. + * @param zoom - current zoomlevel + * @returns the elevation delta in meters + */ + getMeshFrameDelta(zoom) { + // divide by 5 is evaluated by trial & error to get a frame in the right height + return 2 * Math.PI * performance.earthRadius / Math.pow(2, zoom) / 5; + } + getMinTileElevationForLngLatZoom(lnglat, zoom) { + var _a; + const { tileID } = this._getOverscaledTileIDFromLngLatZoom(lnglat, zoom); + return (_a = this.getMinMaxElevation(tileID).minElevation) !== null && _a !== void 0 ? _a : 0; + } + /** + * Get the minimum and maximum elevation contained in a tile. This includes any + * exaggeration included in the terrain. + * + * @param tileID - ID of the tile to be used as a source for the min/max elevation + * @returns the minimum and maximum elevation found in the tile, including the terrain's + * exaggeration + */ + getMinMaxElevation(tileID) { + const tile = this.getTerrainData(tileID).tile; + const minMax = { minElevation: null, maxElevation: null }; + if (tile && tile.dem) { + minMax.minElevation = tile.dem.min * this.exaggeration; + minMax.maxElevation = tile.dem.max * this.exaggeration; + } + return minMax; + } + _getOverscaledTileIDFromLngLatZoom(lnglat, zoom) { + const mercatorCoordinate = performance.MercatorCoordinate.fromLngLat(lnglat.wrap()); + const worldSize = (1 << zoom) * performance.EXTENT; + const mercatorX = mercatorCoordinate.x * worldSize; + const mercatorY = mercatorCoordinate.y * worldSize; + const tileX = Math.floor(mercatorX / performance.EXTENT), tileY = Math.floor(mercatorY / performance.EXTENT); + const tileID = new performance.OverscaledTileID(zoom, 0, zoom, tileX, tileY); + return { + tileID, + mercatorX, + mercatorY + }; + } + _allowMercatorOverflow(p, mercatorX) { + const inLeftHalf = p.x < (this.painter.width / 2); + let lng = performance.lngFromMercatorX(mercatorX); + const centerLng = this.painter.transform.center.lng; + if ((inLeftHalf && Math.sign(lng) > 0 && Math.sign(centerLng) < 0) || + (!inLeftHalf && Math.sign(lng) < 0 && Math.sign(centerLng) > 0)) { + lng = 360 * Math.sign(centerLng) + lng; + return performance.mercatorXfromLng(lng); + } + return mercatorX; + } +} + +/** + * @internal + * `RenderPool` is a resource pool for textures and framebuffers + */ +class RenderPool { + constructor(_context, _size, _tileSize) { + this._context = _context; + this._size = _size; + this._tileSize = _tileSize; + this._objects = []; + this._recentlyUsed = []; + this._stamp = 0; + } + destruct() { + for (const obj of this._objects) { + obj.texture.destroy(); + obj.fbo.destroy(); + } + } + _createObject(id) { + const fbo = this._context.createFramebuffer(this._tileSize, this._tileSize, true, true); + const texture = new Texture(this._context, { width: this._tileSize, height: this._tileSize, data: null }, this._context.gl.RGBA); + texture.bind(this._context.gl.LINEAR, this._context.gl.CLAMP_TO_EDGE); + fbo.depthAttachment.set(this._context.createRenderbuffer(this._context.gl.DEPTH_STENCIL, this._tileSize, this._tileSize)); + fbo.colorAttachment.set(texture.texture); + return { id, fbo, texture, stamp: -1, inUse: false }; + } + getObjectForId(id) { + return this._objects[id]; + } + useObject(obj) { + obj.inUse = true; + this._recentlyUsed = this._recentlyUsed.filter(id => obj.id !== id); + this._recentlyUsed.push(obj.id); + } + stampObject(obj) { + obj.stamp = ++this._stamp; + } + getOrCreateFreeObject() { + // check for free existing object + for (const id of this._recentlyUsed) { + if (!this._objects[id].inUse) + return this._objects[id]; + } + if (this._objects.length >= this._size) + throw new Error('No free RenderPool available, call freeAllObjects() required!'); + // create new object + const obj = this._createObject(this._objects.length); + this._objects.push(obj); + return obj; + } + freeObject(obj) { + obj.inUse = false; + } + freeAllObjects() { + for (const obj of this._objects) + this.freeObject(obj); + } + isFull() { + if (this._objects.length < this._size) { + return false; + } + return this._objects.some(o => !o.inUse) === false; + } +} + +/** + * lookup table which layers should rendered to texture + */ +const LAYERS = { + background: true, + fill: true, + line: true, + raster: true, + hillshade: true +}; +/** + * @internal + * A helper class to help define what should be rendered to texture and how + */ +class RenderToTexture { + constructor(painter, terrain) { + this.painter = painter; + this.terrain = terrain; + this.pool = new RenderPool(painter.context, 30, terrain.sourceCache.tileSize * terrain.qualityFactor); + } + destruct() { + this.pool.destruct(); + } + getTexture(tile) { + return this.pool.getObjectForId(tile.rtt[this._stacks.length - 1].id).texture; + } + prepareForRender(style, zoom) { + this._stacks = []; + this._prevType = null; + this._rttTiles = []; + this._renderableTiles = this.terrain.sourceCache.getRenderableTiles(); + this._renderableLayerIds = style._order.filter(id => !style._layers[id].isHidden(zoom)); + this._coordsDescendingInv = {}; + for (const id in style.sourceCaches) { + this._coordsDescendingInv[id] = {}; + const tileIDs = style.sourceCaches[id].getVisibleCoordinates(); + for (const tileID of tileIDs) { + const keys = this.terrain.sourceCache.getTerrainCoords(tileID); + for (const key in keys) { + if (!this._coordsDescendingInv[id][key]) + this._coordsDescendingInv[id][key] = []; + this._coordsDescendingInv[id][key].push(keys[key]); + } + } + } + this._coordsDescendingInvStr = {}; + for (const id of style._order) { + const layer = style._layers[id], source = layer.source; + if (LAYERS[layer.type]) { + if (!this._coordsDescendingInvStr[source]) { + this._coordsDescendingInvStr[source] = {}; + for (const key in this._coordsDescendingInv[source]) + this._coordsDescendingInvStr[source][key] = this._coordsDescendingInv[source][key].map(c => c.key).sort().join(); + } + } + } + // check tiles to render + for (const tile of this._renderableTiles) { + for (const source in this._coordsDescendingInvStr) { + // rerender if there are more coords to render than in the last rendering + const coords = this._coordsDescendingInvStr[source][tile.tileID.key]; + if (coords && coords !== tile.rttCoords[source]) + tile.rtt = []; + } + } + } + /** + * due that switching textures is relatively slow, the render + * layer-by-layer context is not practicable. To bypass this problem + * this lines of code stack all layers and later render all at once. + * Because of the stylesheet possibility to mixing render-to-texture layers + * and 'live'-layers (f.e. symbols) it is necessary to create more stacks. For example + * a symbol-layer is in between of fill-layers. + * @param layer - the layer to render + * @returns if true layer is rendered to texture, otherwise false + */ + renderLayer(layer) { + if (layer.isHidden(this.painter.transform.zoom)) + return false; + const type = layer.type; + const painter = this.painter; + const isLastLayer = this._renderableLayerIds[this._renderableLayerIds.length - 1] === layer.id; + // remember background, fill, line & raster layer to render into a stack + if (LAYERS[type]) { + // create a new stack if previous layer was not rendered to texture (f.e. symbols) + if (!this._prevType || !LAYERS[this._prevType]) + this._stacks.push([]); + // push current render-to-texture layer to render-stack + this._prevType = type; + this._stacks[this._stacks.length - 1].push(layer.id); + // rendering is done later, all in once + if (!isLastLayer) + return true; + } + // in case a stack is finished render all collected stack-layers into a texture + if (LAYERS[this._prevType] || (LAYERS[type] && isLastLayer)) { + this._prevType = type; + const stack = this._stacks.length - 1, layers = this._stacks[stack] || []; + for (const tile of this._renderableTiles) { + // if render pool is full draw current tiles to screen and free pool + if (this.pool.isFull()) { + drawTerrain(this.painter, this.terrain, this._rttTiles); + this._rttTiles = []; + this.pool.freeAllObjects(); + } + this._rttTiles.push(tile); + // check for cached PoolObject + if (tile.rtt[stack]) { + const obj = this.pool.getObjectForId(tile.rtt[stack].id); + if (obj.stamp === tile.rtt[stack].stamp) { + this.pool.useObject(obj); + continue; + } + } + // get free PoolObject + const obj = this.pool.getOrCreateFreeObject(); + this.pool.useObject(obj); + this.pool.stampObject(obj); + tile.rtt[stack] = { id: obj.id, stamp: obj.stamp }; + // prepare PoolObject for rendering + painter.context.bindFramebuffer.set(obj.fbo.framebuffer); + painter.context.clear({ color: performance.Color.transparent, stencil: 0 }); + painter.currentStencilSource = undefined; + for (let l = 0; l < layers.length; l++) { + const layer = painter.style._layers[layers[l]]; + const coords = layer.source ? this._coordsDescendingInv[layer.source][tile.tileID.key] : [tile.tileID]; + painter.context.viewport.set([0, 0, obj.fbo.width, obj.fbo.height]); + painter._renderTileClippingMasks(layer, coords); + painter.renderLayer(painter, painter.style.sourceCaches[layer.source], layer, coords); + if (layer.source) + tile.rttCoords[layer.source] = this._coordsDescendingInvStr[layer.source][tile.tileID.key]; + } + } + drawTerrain(this.painter, this.terrain, this._rttTiles); + this._rttTiles = []; + this.pool.freeAllObjects(); + return LAYERS[type]; + } + return false; + } +} + +const version$1 = packageJSON.version; +const defaultMinZoom = -2; +const defaultMaxZoom = 22; +// the default values, but also the valid range +const defaultMinPitch = 0; +const defaultMaxPitch = 60; +// use this variable to check maxPitch for validity +const maxPitchThreshold = 85; +const defaultOptions$4 = { + center: [0, 0], + zoom: 0, + bearing: 0, + pitch: 0, + minZoom: defaultMinZoom, + maxZoom: defaultMaxZoom, + minPitch: defaultMinPitch, + maxPitch: defaultMaxPitch, + interactive: true, + scrollZoom: true, + boxZoom: true, + dragRotate: true, + dragPan: true, + keyboard: true, + doubleClickZoom: true, + touchZoomRotate: true, + touchPitch: true, + cooperativeGestures: undefined, + bearingSnap: 7, + clickTolerance: 3, + pitchWithRotate: true, + hash: false, + attributionControl: true, + maplibreLogo: false, + failIfMajorPerformanceCaveat: false, + preserveDrawingBuffer: false, + trackResize: true, + renderWorldCopies: true, + refreshExpiredTiles: true, + maxTileCacheSize: null, + maxTileCacheZoomLevels: performance.config.MAX_TILE_CACHE_ZOOM_LEVELS, + localIdeographFontFamily: 'sans-serif', + transformRequest: null, + transformCameraUpdate: null, + fadeDuration: 300, + crossSourceCollisions: true, + validateStyle: true, + /**Because GL MAX_TEXTURE_SIZE is usually at least 4096px. */ + maxCanvasSize: [4096, 4096] +}; +/** + * The `Map` object represents the map on your page. It exposes methods + * and properties that enable you to programmatically change the map, + * and fires events as users interact with it. + * + * You create a `Map` by specifying a `container` and other options, see {@link MapOptions} for the full list. + * Then MapLibre GL JS initializes the map on the page and returns your `Map` object. + * + * @group Main + * + * @example + * ```ts + * let map = new maplibregl.Map({ + * container: 'map', + * center: [-122.420679, 37.772537], + * zoom: 13, + * style: style_object, + * hash: true, + * transformRequest: (url, resourceType)=> { + * if(resourceType === 'Source' && url.startsWith('http://myHost')) { + * return { + * url: url.replace('http', 'https'), + * headers: { 'my-custom-header': true}, + * credentials: 'include' // Include cookies for cross-origin requests + * } + * } + * } + * }); + * ``` + * @see [Display a map](https://maplibre.org/maplibre-gl-js/docs/examples/simple-map/) + */ +let Map$1 = class Map extends Camera { + constructor(options) { + performance.PerformanceUtils.mark(performance.PerformanceMarkers.create); + options = performance.extend({}, defaultOptions$4, options); + if (options.minZoom != null && options.maxZoom != null && options.minZoom > options.maxZoom) { + throw new Error('maxZoom must be greater than or equal to minZoom'); + } + if (options.minPitch != null && options.maxPitch != null && options.minPitch > options.maxPitch) { + throw new Error('maxPitch must be greater than or equal to minPitch'); + } + if (options.minPitch != null && options.minPitch < defaultMinPitch) { + throw new Error(`minPitch must be greater than or equal to ${defaultMinPitch}`); + } + if (options.maxPitch != null && options.maxPitch > maxPitchThreshold) { + throw new Error(`maxPitch must be less than or equal to ${maxPitchThreshold}`); + } + const transform = new Transform(options.minZoom, options.maxZoom, options.minPitch, options.maxPitch, options.renderWorldCopies); + super(transform, { bearingSnap: options.bearingSnap }); + this._cooperativeGesturesOnWheel = (event) => { + this._onCooperativeGesture(event, event[this._metaKey], 1); + }; + this._contextLost = (event) => { + event.preventDefault(); + if (this._frame) { + this._frame.cancel(); + this._frame = null; + } + this.fire(new performance.Event('webglcontextlost', { originalEvent: event })); + }; + this._contextRestored = (event) => { + this._setupPainter(); + this.resize(); + this._update(); + this.fire(new performance.Event('webglcontextrestored', { originalEvent: event })); + }; + this._onMapScroll = (event) => { + if (event.target !== this._container) + return; + // Revert any scroll which would move the canvas outside of the view + this._container.scrollTop = 0; + this._container.scrollLeft = 0; + return false; + }; + this._onWindowOnline = () => { + this._update(); + }; + this._interactive = options.interactive; + this._cooperativeGestures = options.cooperativeGestures; + this._metaKey = navigator.platform.indexOf('Mac') === 0 ? 'metaKey' : 'ctrlKey'; + this._maxTileCacheSize = options.maxTileCacheSize; + this._maxTileCacheZoomLevels = options.maxTileCacheZoomLevels; + this._failIfMajorPerformanceCaveat = options.failIfMajorPerformanceCaveat; + this._preserveDrawingBuffer = options.preserveDrawingBuffer; + this._antialias = options.antialias; + this._trackResize = options.trackResize; + this._bearingSnap = options.bearingSnap; + this._refreshExpiredTiles = options.refreshExpiredTiles; + this._fadeDuration = options.fadeDuration; + this._crossSourceCollisions = options.crossSourceCollisions; + this._crossFadingFactor = 1; + this._collectResourceTiming = options.collectResourceTiming; + this._renderTaskQueue = new TaskQueue(); + this._controls = []; + this._mapId = performance.uniqueId(); + this._locale = performance.extend({}, defaultLocale, options.locale); + this._clickTolerance = options.clickTolerance; + this._overridePixelRatio = options.pixelRatio; + this._maxCanvasSize = options.maxCanvasSize; + this.transformCameraUpdate = options.transformCameraUpdate; + this._imageQueueHandle = ImageRequest.addThrottleControl(() => this.isMoving()); + this._requestManager = new RequestManager(options.transformRequest); + if (typeof options.container === 'string') { + this._container = document.getElementById(options.container); + if (!this._container) { + throw new Error(`Container '${options.container}' not found.`); + } + } + else if (options.container instanceof HTMLElement) { + this._container = options.container; + } + else { + throw new Error('Invalid type: \'container\' must be a String or HTMLElement.'); + } + if (options.maxBounds) { + this.setMaxBounds(options.maxBounds); + } + this._setupContainer(); + this._setupPainter(); + this.on('move', () => this._update(false)); + this.on('moveend', () => this._update(false)); + this.on('zoom', () => this._update(true)); + this.on('terrain', () => { + this.painter.terrainFacilitator.dirty = true; + this._update(true); + }); + this.once('idle', () => { this._idleTriggered = true; }); + if (typeof window !== 'undefined') { + addEventListener('online', this._onWindowOnline, false); + let initialResizeEventCaptured = false; + const throttledResizeCallback = throttle((entries) => { + if (this._trackResize && !this._removed) { + this.resize(entries)._update(); + } + }, 50); + this._resizeObserver = new ResizeObserver((entries) => { + if (!initialResizeEventCaptured) { + initialResizeEventCaptured = true; + return; + } + throttledResizeCallback(entries); + }); + this._resizeObserver.observe(this._container); + } + this.handlers = new HandlerManager(this, options); + if (this._cooperativeGestures) { + this._setupCooperativeGestures(); + } + const hashName = (typeof options.hash === 'string' && options.hash) || undefined; + this._hash = options.hash && (new Hash(hashName)).addTo(this); + // don't set position from options if set through hash + if (!this._hash || !this._hash._onHashChange()) { + this.jumpTo({ + center: options.center, + zoom: options.zoom, + bearing: options.bearing, + pitch: options.pitch + }); + if (options.bounds) { + this.resize(); + this.fitBounds(options.bounds, performance.extend({}, options.fitBoundsOptions, { duration: 0 })); + } + } + this.resize(); + this._localIdeographFontFamily = options.localIdeographFontFamily; + this._validateStyle = options.validateStyle; + if (options.style) + this.setStyle(options.style, { localIdeographFontFamily: options.localIdeographFontFamily }); + if (options.attributionControl) + this.addControl(new AttributionControl({ customAttribution: options.customAttribution })); + if (options.maplibreLogo) + this.addControl(new LogoControl(), options.logoPosition); + this.on('style.load', () => { + if (this.transform.unmodified) { + this.jumpTo(this.style.stylesheet); + } + }); + this.on('data', (event) => { + this._update(event.dataType === 'style'); + this.fire(new performance.Event(`${event.dataType}data`, event)); + }); + this.on('dataloading', (event) => { + this.fire(new performance.Event(`${event.dataType}dataloading`, event)); + }); + this.on('dataabort', (event) => { + this.fire(new performance.Event('sourcedataabort', event)); + }); + } + /** + * @internal + * Returns a unique number for this map instance which is used for the MapLoadEvent + * to make sure we only fire one event per instantiated map object. + * @returns the uniq map ID + */ + _getMapId() { + return this._mapId; + } + /** + * Adds an {@link IControl} to the map, calling `control.onAdd(this)`. + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param control - The {@link IControl} to add. + * @param position - position on the map to which the control will be added. + * Valid values are `'top-left'`, `'top-right'`, `'bottom-left'`, and `'bottom-right'`. Defaults to `'top-right'`. + * @returns `this` + * @example + * Add zoom and rotation controls to the map. + * ```ts + * map.addControl(new maplibregl.NavigationControl()); + * ``` + * @see [Display map navigation controls](https://maplibre.org/maplibre-gl-js/docs/examples/navigation/) + */ + addControl(control, position) { + if (position === undefined) { + if (control.getDefaultPosition) { + position = control.getDefaultPosition(); + } + else { + position = 'top-right'; + } + } + if (!control || !control.onAdd) { + return this.fire(new performance.ErrorEvent(new Error('Invalid argument to map.addControl(). Argument must be a control with onAdd and onRemove methods.'))); + } + const controlElement = control.onAdd(this); + this._controls.push(control); + const positionContainer = this._controlPositions[position]; + if (position.indexOf('bottom') !== -1) { + positionContainer.insertBefore(controlElement, positionContainer.firstChild); + } + else { + positionContainer.appendChild(controlElement); + } + return this; + } + /** + * Removes the control from the map. + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param control - The {@link IControl} to remove. + * @returns `this` + * @example + * ```ts + * // Define a new navigation control. + * let navigation = new maplibregl.NavigationControl(); + * // Add zoom and rotation controls to the map. + * map.addControl(navigation); + * // Remove zoom and rotation controls from the map. + * map.removeControl(navigation); + * ``` + */ + removeControl(control) { + if (!control || !control.onRemove) { + return this.fire(new performance.ErrorEvent(new Error('Invalid argument to map.removeControl(). Argument must be a control with onAdd and onRemove methods.'))); + } + const ci = this._controls.indexOf(control); + if (ci > -1) + this._controls.splice(ci, 1); + control.onRemove(this); + return this; + } + /** + * Checks if a control exists on the map. + * + * @param control - The {@link IControl} to check. + * @returns true if map contains control. + * @example + * ```ts + * // Define a new navigation control. + * let navigation = new maplibregl.NavigationControl(); + * // Add zoom and rotation controls to the map. + * map.addControl(navigation); + * // Check that the navigation control exists on the map. + * map.hasControl(navigation); + * ``` + */ + hasControl(control) { + return this._controls.indexOf(control) > -1; + } + calculateCameraOptionsFromTo(from, altitudeFrom, to, altitudeTo) { + if (altitudeTo == null && this.terrain) { + altitudeTo = this.terrain.getElevationForLngLatZoom(to, this.transform.tileZoom); + } + return super.calculateCameraOptionsFromTo(from, altitudeFrom, to, altitudeTo); + } + /** + * Resizes the map according to the dimensions of its + * `container` element. + * + * Checks if the map container size changed and updates the map if it has changed. + * This method must be called after the map's `container` is resized programmatically + * or when the map is shown after being initially hidden with CSS. + * + * Triggers the following events: `movestart`, `move`, `moveend`, and `resize`. + * + * @param eventData - Additional properties to be passed to `movestart`, `move`, `resize`, and `moveend` + * events that get triggered as a result of resize. This can be useful for differentiating the + * source of an event (for example, user-initiated or programmatically-triggered events). + * @returns `this` + * @example + * Resize the map when the map container is shown after being initially hidden with CSS. + * ```ts + * let mapDiv = document.getElementById('map'); + * if (mapDiv.style.visibility === true) map.resize(); + * ``` + */ + resize(eventData) { + var _a; + const dimensions = this._containerDimensions(); + const width = dimensions[0]; + const height = dimensions[1]; + const clampedPixelRatio = this._getClampedPixelRatio(width, height); + this._resizeCanvas(width, height, clampedPixelRatio); + this.painter.resize(width, height, clampedPixelRatio); + // check if we've reached GL limits, in that case further clamps pixelRatio + if (this.painter.overLimit()) { + const gl = this.painter.context.gl; + // store updated _maxCanvasSize value + this._maxCanvasSize = [gl.drawingBufferWidth, gl.drawingBufferHeight]; + const clampedPixelRatio = this._getClampedPixelRatio(width, height); + this._resizeCanvas(width, height, clampedPixelRatio); + this.painter.resize(width, height, clampedPixelRatio); + } + this.transform.resize(width, height); + (_a = this._requestedCameraState) === null || _a === void 0 ? void 0 : _a.resize(width, height); + const fireMoving = !this._moving; + if (fireMoving) { + this.stop(); + this.fire(new performance.Event('movestart', eventData)) + .fire(new performance.Event('move', eventData)); + } + this.fire(new performance.Event('resize', eventData)); + if (fireMoving) + this.fire(new performance.Event('moveend', eventData)); + return this; + } + /** + * @internal + * Return the map's pixel ratio eventually scaled down to respect maxCanvasSize. + * Internally you should use this and not getPixelRatio(). + */ + _getClampedPixelRatio(width, height) { + const { 0: maxCanvasWidth, 1: maxCanvasHeight } = this._maxCanvasSize; + const pixelRatio = this.getPixelRatio(); + const canvasWidth = width * pixelRatio; + const canvasHeight = height * pixelRatio; + const widthScaleFactor = canvasWidth > maxCanvasWidth ? (maxCanvasWidth / canvasWidth) : 1; + const heightScaleFactor = canvasHeight > maxCanvasHeight ? (maxCanvasHeight / canvasHeight) : 1; + return Math.min(widthScaleFactor, heightScaleFactor) * pixelRatio; + } + /** + * Returns the map's pixel ratio. + * Note that the pixel ratio actually applied may be lower to respect maxCanvasSize. + * @returns The pixel ratio. + */ + getPixelRatio() { + var _a; + return (_a = this._overridePixelRatio) !== null && _a !== void 0 ? _a : devicePixelRatio; + } + /** + * Sets the map's pixel ratio. This allows to override `devicePixelRatio`. + * After this call, the canvas' `width` attribute will be `container.clientWidth * pixelRatio` + * and its height attribute will be `container.clientHeight * pixelRatio`. + * Set this to null to disable `devicePixelRatio` override. + * Note that the pixel ratio actually applied may be lower to respect maxCanvasSize. + * @param pixelRatio - The pixel ratio. + */ + setPixelRatio(pixelRatio) { + this._overridePixelRatio = pixelRatio; + this.resize(); + } + /** + * Returns the map's geographical bounds. When the bearing or pitch is non-zero, the visible region is not + * an axis-aligned rectangle, and the result is the smallest bounds that encompasses the visible region. + * @returns The geographical bounds of the map as {@link LngLatBounds}. + * @example + * ```ts + * let bounds = map.getBounds(); + * ``` + */ + getBounds() { + return this.transform.getBounds(); + } + /** + * Returns the maximum geographical bounds the map is constrained to, or `null` if none set. + * @returns The map object. + * @example + * ```ts + * let maxBounds = map.getMaxBounds(); + * ``` + */ + getMaxBounds() { + return this.transform.getMaxBounds(); + } + /** + * Sets or clears the map's geographical bounds. + * + * Pan and zoom operations are constrained within these bounds. + * If a pan or zoom is performed that would + * display regions outside these bounds, the map will + * instead display a position and zoom level + * as close as possible to the operation's request while still + * remaining within the bounds. + * + * @param bounds - The maximum bounds to set. If `null` or `undefined` is provided, the function removes the map's maximum bounds. + * @returns `this` + * @example + * Define bounds that conform to the `LngLatBoundsLike` object as set the max bounds. + * ```ts + * let bounds = [ + * [-74.04728, 40.68392], // [west, south] + * [-73.91058, 40.87764] // [east, north] + * ]; + * map.setMaxBounds(bounds); + * ``` + */ + setMaxBounds(bounds) { + this.transform.setMaxBounds(LngLatBounds.convert(bounds)); + return this._update(); + } + /** + * Sets or clears the map's minimum zoom level. + * If the map's current zoom level is lower than the new minimum, + * the map will zoom to the new minimum. + * + * It is not always possible to zoom out and reach the set `minZoom`. + * Other factors such as map height may restrict zooming. For example, + * if the map is 512px tall it will not be possible to zoom below zoom 0 + * no matter what the `minZoom` is set to. + * + * A {@link ErrorEvent} event will be fired if minZoom is out of bounds. + * + * @param minZoom - The minimum zoom level to set (-2 - 24). + * If `null` or `undefined` is provided, the function removes the current minimum zoom (i.e. sets it to -2). + * @returns `this` + * @example + * ```ts + * map.setMinZoom(12.25); + * ``` + */ + setMinZoom(minZoom) { + minZoom = minZoom === null || minZoom === undefined ? defaultMinZoom : minZoom; + if (minZoom >= defaultMinZoom && minZoom <= this.transform.maxZoom) { + this.transform.minZoom = minZoom; + this._update(); + if (this.getZoom() < minZoom) + this.setZoom(minZoom); + return this; + } + else + throw new Error(`minZoom must be between ${defaultMinZoom} and the current maxZoom, inclusive`); + } + /** + * Returns the map's minimum allowable zoom level. + * + * @returns minZoom + * @example + * ```ts + * let minZoom = map.getMinZoom(); + * ``` + */ + getMinZoom() { return this.transform.minZoom; } + /** + * Sets or clears the map's maximum zoom level. + * If the map's current zoom level is higher than the new maximum, + * the map will zoom to the new maximum. + * + * A {@link ErrorEvent} event will be fired if minZoom is out of bounds. + * + * @param maxZoom - The maximum zoom level to set. + * If `null` or `undefined` is provided, the function removes the current maximum zoom (sets it to 22). + * @returns `this` + * @example + * ```ts + * map.setMaxZoom(18.75); + * ``` + */ + setMaxZoom(maxZoom) { + maxZoom = maxZoom === null || maxZoom === undefined ? defaultMaxZoom : maxZoom; + if (maxZoom >= this.transform.minZoom) { + this.transform.maxZoom = maxZoom; + this._update(); + if (this.getZoom() > maxZoom) + this.setZoom(maxZoom); + return this; + } + else + throw new Error('maxZoom must be greater than the current minZoom'); + } + /** + * Returns the map's maximum allowable zoom level. + * + * @returns The maxZoom + * @example + * ```ts + * let maxZoom = map.getMaxZoom(); + * ``` + */ + getMaxZoom() { return this.transform.maxZoom; } + /** + * Sets or clears the map's minimum pitch. + * If the map's current pitch is lower than the new minimum, + * the map will pitch to the new minimum. + * + * A {@link ErrorEvent} event will be fired if minPitch is out of bounds. + * + * @param minPitch - The minimum pitch to set (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project. + * If `null` or `undefined` is provided, the function removes the current minimum pitch (i.e. sets it to 0). + * @returns `this` + */ + setMinPitch(minPitch) { + minPitch = minPitch === null || minPitch === undefined ? defaultMinPitch : minPitch; + if (minPitch < defaultMinPitch) { + throw new Error(`minPitch must be greater than or equal to ${defaultMinPitch}`); + } + if (minPitch >= defaultMinPitch && minPitch <= this.transform.maxPitch) { + this.transform.minPitch = minPitch; + this._update(); + if (this.getPitch() < minPitch) + this.setPitch(minPitch); + return this; + } + else + throw new Error(`minPitch must be between ${defaultMinPitch} and the current maxPitch, inclusive`); + } + /** + * Returns the map's minimum allowable pitch. + * + * @returns The minPitch + */ + getMinPitch() { return this.transform.minPitch; } + /** + * Sets or clears the map's maximum pitch. + * If the map's current pitch is higher than the new maximum, + * the map will pitch to the new maximum. + * + * A {@link ErrorEvent} event will be fired if maxPitch is out of bounds. + * + * @param maxPitch - The maximum pitch to set (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project. + * If `null` or `undefined` is provided, the function removes the current maximum pitch (sets it to 60). + * @returns `this` + */ + setMaxPitch(maxPitch) { + maxPitch = maxPitch === null || maxPitch === undefined ? defaultMaxPitch : maxPitch; + if (maxPitch > maxPitchThreshold) { + throw new Error(`maxPitch must be less than or equal to ${maxPitchThreshold}`); + } + if (maxPitch >= this.transform.minPitch) { + this.transform.maxPitch = maxPitch; + this._update(); + if (this.getPitch() > maxPitch) + this.setPitch(maxPitch); + return this; + } + else + throw new Error('maxPitch must be greater than the current minPitch'); + } + /** + * Returns the map's maximum allowable pitch. + * + * @returns The maxPitch + */ + getMaxPitch() { return this.transform.maxPitch; } + /** + * Returns the state of `renderWorldCopies`. If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`: + * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire + * container, there will be blank space beyond 180 and -180 degrees longitude. + * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the + * map and the other on the left edge of the map) at every zoom level. + * @returns The renderWorldCopies + * @example + * ```ts + * let worldCopiesRendered = map.getRenderWorldCopies(); + * ``` + * @see [Render world copies](https://maplibre.org/maplibre-gl-js/docs/examples/render-world-copies/) + */ + getRenderWorldCopies() { return this.transform.renderWorldCopies; } + /** + * Sets the state of `renderWorldCopies`. + * + * @param renderWorldCopies - If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`: + * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire + * container, there will be blank space beyond 180 and -180 degrees longitude. + * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the + * map and the other on the left edge of the map) at every zoom level. + * + * `undefined` is treated as `true`, `null` is treated as `false`. + * @returns `this` + * @example + * ```ts + * map.setRenderWorldCopies(true); + * ``` + * @see [Render world copies](https://maplibre.org/maplibre-gl-js/docs/examples/render-world-copies/) + */ + setRenderWorldCopies(renderWorldCopies) { + this.transform.renderWorldCopies = renderWorldCopies; + return this._update(); + } + /** + * Gets the map's cooperativeGestures option + * + * @returns The gestureOptions + */ + getCooperativeGestures() { + return this._cooperativeGestures; + } + /** + * Sets or clears the map's cooperativeGestures option + * + * @param gestureOptions - If `true` or set to an options object, map is only accessible on desktop while holding Command/Ctrl and only accessible on mobile with two fingers. Interacting with the map using normal gestures will trigger an informational screen. With this option enabled, "drag to pitch" requires a three-finger gesture. + * @returns `this` + */ + setCooperativeGestures(gestureOptions) { + this._cooperativeGestures = gestureOptions; + if (this._cooperativeGestures) { + this._setupCooperativeGestures(); + } + else { + this._destroyCooperativeGestures(); + } + return this; + } + /** + * Returns a [Point](https://github.com/mapbox/point-geometry) representing pixel coordinates, relative to the map's `container`, + * that correspond to the specified geographical location. + * + * @param lnglat - The geographical location to project. + * @returns The [Point](https://github.com/mapbox/point-geometry) corresponding to `lnglat`, relative to the map's `container`. + * @example + * ```ts + * let coordinate = [-122.420679, 37.772537]; + * let point = map.project(coordinate); + * ``` + */ + project(lnglat) { + return this.transform.locationPoint(performance.LngLat.convert(lnglat), this.style && this.terrain); + } + /** + * Returns a {@link LngLat} representing geographical coordinates that correspond + * to the specified pixel coordinates. + * + * @param point - The pixel coordinates to unproject. + * @returns The {@link LngLat} corresponding to `point`. + * @example + * ```ts + * map.on('click', function(e) { + * // When the map is clicked, get the geographic coordinate. + * let coordinate = map.unproject(e.point); + * }); + * ``` + */ + unproject(point) { + return this.transform.pointLocation(performance.Point.convert(point), this.terrain); + } + /** + * Returns true if the map is panning, zooming, rotating, or pitching due to a camera animation or user gesture. + * @returns true if the map is moving. + * @example + * ```ts + * let isMoving = map.isMoving(); + * ``` + */ + isMoving() { + var _a; + return this._moving || ((_a = this.handlers) === null || _a === void 0 ? void 0 : _a.isMoving()); + } + /** + * Returns true if the map is zooming due to a camera animation or user gesture. + * @returns true if the map is zooming. + * @example + * ```ts + * let isZooming = map.isZooming(); + * ``` + */ + isZooming() { + var _a; + return this._zooming || ((_a = this.handlers) === null || _a === void 0 ? void 0 : _a.isZooming()); + } + /** + * Returns true if the map is rotating due to a camera animation or user gesture. + * @returns true if the map is rotating. + * @example + * ```ts + * map.isRotating(); + * ``` + */ + isRotating() { + var _a; + return this._rotating || ((_a = this.handlers) === null || _a === void 0 ? void 0 : _a.isRotating()); + } + _createDelegatedListener(type, layerId, listener) { + if (type === 'mouseenter' || type === 'mouseover') { + let mousein = false; + const mousemove = (e) => { + const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, { layers: [layerId] }) : []; + if (!features.length) { + mousein = false; + } + else if (!mousein) { + mousein = true; + listener.call(this, new MapMouseEvent(type, this, e.originalEvent, { features })); + } + }; + const mouseout = () => { + mousein = false; + }; + return { layer: layerId, listener, delegates: { mousemove, mouseout } }; + } + else if (type === 'mouseleave' || type === 'mouseout') { + let mousein = false; + const mousemove = (e) => { + const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, { layers: [layerId] }) : []; + if (features.length) { + mousein = true; + } + else if (mousein) { + mousein = false; + listener.call(this, new MapMouseEvent(type, this, e.originalEvent)); + } + }; + const mouseout = (e) => { + if (mousein) { + mousein = false; + listener.call(this, new MapMouseEvent(type, this, e.originalEvent)); + } + }; + return { layer: layerId, listener, delegates: { mousemove, mouseout } }; + } + else { + const delegate = (e) => { + const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, { layers: [layerId] }) : []; + if (features.length) { + // Here we need to mutate the original event, so that preventDefault works as expected. + e.features = features; + listener.call(this, e); + delete e.features; + } + }; + return { layer: layerId, listener, delegates: { [type]: delegate } }; + } + } + on(type, layerIdOrListener, listener) { + if (listener === undefined) { + return super.on(type, layerIdOrListener); + } + const delegatedListener = this._createDelegatedListener(type, layerIdOrListener, listener); + this._delegatedListeners = this._delegatedListeners || {}; + this._delegatedListeners[type] = this._delegatedListeners[type] || []; + this._delegatedListeners[type].push(delegatedListener); + for (const event in delegatedListener.delegates) { + this.on(event, delegatedListener.delegates[event]); + } + return this; + } + once(type, layerIdOrListener, listener) { + if (listener === undefined) { + return super.once(type, layerIdOrListener); + } + const delegatedListener = this._createDelegatedListener(type, layerIdOrListener, listener); + for (const event in delegatedListener.delegates) { + this.once(event, delegatedListener.delegates[event]); + } + return this; + } + off(type, layerIdOrListener, listener) { + if (listener === undefined) { + return super.off(type, layerIdOrListener); + } + const removeDelegatedListener = (delegatedListeners) => { + const listeners = delegatedListeners[type]; + for (let i = 0; i < listeners.length; i++) { + const delegatedListener = listeners[i]; + if (delegatedListener.layer === layerIdOrListener && delegatedListener.listener === listener) { + for (const event in delegatedListener.delegates) { + this.off(event, delegatedListener.delegates[event]); + } + listeners.splice(i, 1); + return this; + } + } + }; + if (this._delegatedListeners && this._delegatedListeners[type]) { + removeDelegatedListener(this._delegatedListeners); + } + return this; + } + /** + * Returns an array of MapGeoJSONFeature objects + * representing visible features that satisfy the query parameters. + * + * @param geometryOrOptions - (optional) The geometry of the query region: + * either a single point or southwest and northeast points describing a bounding box. + * Omitting this parameter (i.e. calling {@link Map#queryRenderedFeatures} with zero arguments, + * or with only a `options` argument) is equivalent to passing a bounding box encompassing the entire + * map viewport. + * The geometryOrOptions can receive a {@link QueryRenderedFeaturesOptions} only to support a situation where the function receives only one parameter which is the options parameter. + * @param options - (optional) Options object. + * + * @returns An array of MapGeoJSONFeature objects. + * + * The `properties` value of each returned feature object contains the properties of its source feature. For GeoJSON sources, only + * string and numeric property values are supported (i.e. `null`, `Array`, and `Object` values are not supported). + * + * Each feature includes top-level `layer`, `source`, and `sourceLayer` properties. The `layer` property is an object + * representing the style layer to which the feature belongs. Layout and paint properties in this object contain values + * which are fully evaluated for the given zoom level and feature. + * + * Only features that are currently rendered are included. Some features will **not** be included, like: + * + * - Features from layers whose `visibility` property is `"none"`. + * - Features from layers whose zoom range excludes the current zoom level. + * - Symbol features that have been hidden due to text or icon collision. + * + * Features from all other layers are included, including features that may have no visible + * contribution to the rendered result; for example, because the layer's opacity or color alpha component is set to + * 0. + * + * The topmost rendered feature appears first in the returned array, and subsequent features are sorted by + * descending z-order. Features that are rendered multiple times (due to wrapping across the antemeridian at low + * zoom levels) are returned only once (though subject to the following caveat). + * + * Because features come from tiled vector data or GeoJSON data that is converted to tiles internally, feature + * geometries may be split or duplicated across tile boundaries and, as a result, features may appear multiple + * times in query results. For example, suppose there is a highway running through the bounding rectangle of a query. + * The results of the query will be those parts of the highway that lie within the map tiles covering the bounding + * rectangle, even if the highway extends into other tiles, and the portion of the highway within each map tile + * will be returned as a separate feature. Similarly, a point feature near a tile boundary may appear in multiple + * tiles due to tile buffering. + * + * @example + * Find all features at a point + * ```ts + * let features = map.queryRenderedFeatures( + * [20, 35], + * { layers: ['my-layer-name'] } + * ); + * ``` + * + * @example + * Find all features within a static bounding box + * ```ts + * let features = map.queryRenderedFeatures( + * [[10, 20], [30, 50]], + * { layers: ['my-layer-name'] } + * ); + * ``` + * + * @example + * Find all features within a bounding box around a point + * ```ts + * let width = 10; + * let height = 20; + * let features = map.queryRenderedFeatures([ + * [point.x - width / 2, point.y - height / 2], + * [point.x + width / 2, point.y + height / 2] + * ], { layers: ['my-layer-name'] }); + * ``` + * + * @example + * Query all rendered features from a single layer + * ```ts + * let features = map.queryRenderedFeatures({ layers: ['my-layer-name'] }); + * ``` + * @see [Get features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/queryrenderedfeatures/) + */ + queryRenderedFeatures(geometryOrOptions, options) { + if (!this.style) { + return []; + } + let queryGeometry; + const isGeometry = geometryOrOptions instanceof performance.Point || Array.isArray(geometryOrOptions); + const geometry = isGeometry ? geometryOrOptions : [[0, 0], [this.transform.width, this.transform.height]]; + options = options || (isGeometry ? {} : geometryOrOptions) || {}; + if (geometry instanceof performance.Point || typeof geometry[0] === 'number') { + queryGeometry = [performance.Point.convert(geometry)]; + } + else { + const tl = performance.Point.convert(geometry[0]); + const br = performance.Point.convert(geometry[1]); + queryGeometry = [tl, new performance.Point(br.x, tl.y), br, new performance.Point(tl.x, br.y), tl]; + } + return this.style.queryRenderedFeatures(queryGeometry, options, this.transform); + } + /** + * Returns an array of MapGeoJSONFeature objects + * representing features within the specified vector tile or GeoJSON source that satisfy the query parameters. + * + * @param sourceId - The ID of the vector tile or GeoJSON source to query. + * @param parameters - The options object. + * @returns An array of MapGeoJSONFeature objects. + * + * In contrast to {@link Map#queryRenderedFeatures}, this function returns all features matching the query parameters, + * whether or not they are rendered by the current style (i.e. visible). The domain of the query includes all currently-loaded + * vector tiles and GeoJSON source tiles: this function does not check tiles outside the currently + * visible viewport. + * + * Because features come from tiled vector data or GeoJSON data that is converted to tiles internally, feature + * geometries may be split or duplicated across tile boundaries and, as a result, features may appear multiple + * times in query results. For example, suppose there is a highway running through the bounding rectangle of a query. + * The results of the query will be those parts of the highway that lie within the map tiles covering the bounding + * rectangle, even if the highway extends into other tiles, and the portion of the highway within each map tile + * will be returned as a separate feature. Similarly, a point feature near a tile boundary may appear in multiple + * tiles due to tile buffering. + * + * @example + * Find all features in one source layer in a vector source + * ```ts + * let features = map.querySourceFeatures('your-source-id', { + * sourceLayer: 'your-source-layer' + * }); + * ``` + * + */ + querySourceFeatures(sourceId, parameters) { + return this.style.querySourceFeatures(sourceId, parameters); + } + /** + * Updates the map's MapLibre style object with a new value. + * + * If a style is already set when this is used and options.diff is set to true, the map renderer will attempt to compare the given style + * against the map's current state and perform only the changes necessary to make the map style match the desired state. Changes in sprites + * (images used for icons and patterns) and glyphs (fonts for label text) **cannot** be diffed. If the sprites or fonts used in the current + * style and the given style are different in any way, the map renderer will force a full update, removing the current style and building + * the given one from scratch. + * + * + * @param style - A JSON object conforming to the schema described in the + * [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/), or a URL to such JSON. + * @param options - The options object. + * @returns `this` + * + * @example + * ```ts + * map.setStyle("https://demotiles.maplibre.org/style.json"); + * + * map.setStyle('https://demotiles.maplibre.org/style.json', { + * transformStyle: (previousStyle, nextStyle) => ({ + * ...nextStyle, + * sources: { + * ...nextStyle.sources, + * // copy a source from previous style + * 'osm': previousStyle.sources.osm + * }, + * layers: [ + * // background layer + * nextStyle.layers[0], + * // copy a layer from previous style + * previousStyle.layers[0], + * // other layers from the next style + * ...nextStyle.layers.slice(1).map(layer => { + * // hide the layers we don't need from demotiles style + * if (layer.id.startsWith('geolines')) { + * layer.layout = {...layer.layout || {}, visibility: 'none'}; + * // filter out US polygons + * } else if (layer.id.startsWith('coastline') || layer.id.startsWith('countries')) { + * layer.filter = ['!=', ['get', 'ADM0_A3'], 'USA']; + * } + * return layer; + * }) + * ] + * }) + * }); + * ``` + */ + setStyle(style, options) { + options = performance.extend({}, { + localIdeographFontFamily: this._localIdeographFontFamily, + validate: this._validateStyle + }, options); + if ((options.diff !== false && options.localIdeographFontFamily === this._localIdeographFontFamily) && this.style && style) { + this._diffStyle(style, options); + return this; + } + else { + this._localIdeographFontFamily = options.localIdeographFontFamily; + return this._updateStyle(style, options); + } + } + /** + * Updates the requestManager's transform request with a new function + * + * @param transformRequest - A callback run before the Map makes a request for an external URL. The callback can be used to modify the url, set headers, or set the credentials property for cross-origin requests. + * Expected to return an object with a `url` property and optionally `headers` and `credentials` properties + * + * @returns `this` + * + * @example + * ```ts + * map.setTransformRequest((url: string, resourceType: string) => {}); + * ``` + */ + setTransformRequest(transformRequest) { + this._requestManager.setTransformRequest(transformRequest); + return this; + } + _getUIString(key) { + const str = this._locale[key]; + if (str == null) { + throw new Error(`Missing UI string '${key}'`); + } + return str; + } + _updateStyle(style, options) { + // transformStyle relies on having previous style serialized, if it is not loaded yet, delay _updateStyle until previous style is loaded + if (options.transformStyle && this.style && !this.style._loaded) { + this.style.once('style.load', () => this._updateStyle(style, options)); + return; + } + const previousStyle = this.style && options.transformStyle ? this.style.serialize() : undefined; + if (this.style) { + this.style.setEventedParent(null); + // Only release workers when map is getting disposed + this.style._remove(!style); + } + if (!style) { + delete this.style; + return this; + } + else { + this.style = new Style(this, options || {}); + } + this.style.setEventedParent(this, { style: this.style }); + if (typeof style === 'string') { + this.style.loadURL(style, options, previousStyle); + } + else { + this.style.loadJSON(style, options, previousStyle); + } + return this; + } + _lazyInitEmptyStyle() { + if (!this.style) { + this.style = new Style(this, {}); + this.style.setEventedParent(this, { style: this.style }); + this.style.loadEmpty(); + } + } + _diffStyle(style, options) { + if (typeof style === 'string') { + const url = style; + const request = this._requestManager.transformRequest(url, ResourceType.Style); + performance.getJSON(request, (error, json) => { + if (error) { + this.fire(new performance.ErrorEvent(error)); + } + else if (json) { + this._updateDiff(json, options); + } + }); + } + else if (typeof style === 'object') { + this._updateDiff(style, options); + } + } + _updateDiff(style, options) { + try { + if (this.style.setState(style, options)) { + this._update(true); + } + } + catch (e) { + performance.warnOnce(`Unable to perform style diff: ${e.message || e.error || e}. Rebuilding the style from scratch.`); + this._updateStyle(style, options); + } + } + /** + * Returns the map's MapLibre style object, a JSON object which can be used to recreate the map's style. + * + * @returns The map's style JSON object. + * + * @example + * ```ts + * let styleJson = map.getStyle(); + * ``` + * + */ + getStyle() { + if (this.style) { + return this.style.serialize(); + } + } + /** + * Returns a Boolean indicating whether the map's style is fully loaded. + * + * @returns A Boolean indicating whether the style is fully loaded. + * + * @example + * ```ts + * let styleLoadStatus = map.isStyleLoaded(); + * ``` + */ + isStyleLoaded() { + if (!this.style) + return performance.warnOnce('There is no style added to the map.'); + return this.style.loaded(); + } + /** + * Adds a source to the map's style. + * + * Events triggered: + * + * Triggers the `source.add` event. + * + * @param id - The ID of the source to add. Must not conflict with existing sources. + * @param source - The source object, conforming to the + * MapLibre Style Specification's [source definition](https://maplibre.org/maplibre-style-spec/sources) or + * {@link CanvasSourceSpecification}. + * @returns `this` + * @example + * ```ts + * map.addSource('my-data', { + * type: 'vector', + * url: 'https://demotiles.maplibre.org/tiles/tiles.json' + * }); + * ``` + * @example + * ```ts + * map.addSource('my-data', { + * "type": "geojson", + * "data": { + * "type": "Feature", + * "geometry": { + * "type": "Point", + * "coordinates": [-77.0323, 38.9131] + * }, + * "properties": { + * "title": "Mapbox DC", + * "marker-symbol": "monument" + * } + * } + * }); + * ``` + * @see GeoJSON source: [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/) + */ + addSource(id, source) { + this._lazyInitEmptyStyle(); + this.style.addSource(id, source); + return this._update(true); + } + /** + * Returns a Boolean indicating whether the source is loaded. Returns `true` if the source with + * the given ID in the map's style has no outstanding network requests, otherwise `false`. + * + * A {@link ErrorEvent} event will be fired if there is no source wit the specified ID. + * + * @param id - The ID of the source to be checked. + * @returns A Boolean indicating whether the source is loaded. + * @example + * ```ts + * let sourceLoaded = map.isSourceLoaded('bathymetry-data'); + * ``` + */ + isSourceLoaded(id) { + const source = this.style && this.style.sourceCaches[id]; + if (source === undefined) { + this.fire(new performance.ErrorEvent(new Error(`There is no source with ID '${id}'`))); + return; + } + return source.loaded(); + } + /** + * Loads a 3D terrain mesh, based on a "raster-dem" source. + * + * Triggers the `terrain` event. + * + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.setTerrain({ source: 'terrain' }); + * ``` + */ + setTerrain(options) { + this.style._checkLoaded(); + // clear event handlers + if (this._terrainDataCallback) + this.style.off('data', this._terrainDataCallback); + if (!options) { + // remove terrain + if (this.terrain) + this.terrain.sourceCache.destruct(); + this.terrain = null; + if (this.painter.renderToTexture) + this.painter.renderToTexture.destruct(); + this.painter.renderToTexture = null; + this.transform._minEleveationForCurrentTile = 0; + this.transform.elevation = 0; + } + else { + // add terrain + const sourceCache = this.style.sourceCaches[options.source]; + if (!sourceCache) + throw new Error(`cannot load terrain, because there exists no source with ID: ${options.source}`); + // Warn once if user is using the same source for hillshade and terrain + for (const index in this.style._layers) { + const thisLayer = this.style._layers[index]; + if (thisLayer.type === 'hillshade' && thisLayer.source === options.source) { + performance.warnOnce('You are using the same source for a hillshade layer and for 3D terrain. Please consider using two separate sources to improve rendering quality.'); + } + } + this.terrain = new Terrain(this.painter, sourceCache, options); + this.painter.renderToTexture = new RenderToTexture(this.painter, this.terrain); + this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom); + this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom); + this._terrainDataCallback = e => { + if (e.dataType === 'style') { + this.terrain.sourceCache.freeRtt(); + } + else if (e.dataType === 'source' && e.tile) { + if (e.sourceId === options.source && !this._elevationFreeze) { + this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom); + this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom); + } + this.terrain.sourceCache.freeRtt(e.tile.tileID); + } + }; + this.style.on('data', this._terrainDataCallback); + } + this.fire(new performance.Event('terrain', { terrain: options })); + return this; + } + /** + * Get the terrain-options if terrain is loaded + * @returns the TerrainSpecification passed to setTerrain + * @example + * ```ts + * map.getTerrain(); // { source: 'terrain' }; + * ``` + */ + getTerrain() { + var _a, _b; + return (_b = (_a = this.terrain) === null || _a === void 0 ? void 0 : _a.options) !== null && _b !== void 0 ? _b : null; + } + /** + * Returns a Boolean indicating whether all tiles in the viewport from all sources on + * the style are loaded. + * + * @returns A Boolean indicating whether all tiles are loaded. + * @example + * ```ts + * let tilesLoaded = map.areTilesLoaded(); + * ``` + */ + areTilesLoaded() { + const sources = this.style && this.style.sourceCaches; + for (const id in sources) { + const source = sources[id]; + const tiles = source._tiles; + for (const t in tiles) { + const tile = tiles[t]; + if (!(tile.state === 'loaded' || tile.state === 'errored')) + return false; + } + } + return true; + } + /** + * Adds a [custom source type](#Custom Sources), making it available for use with + * {@link Map#addSource}. + * @param name - The name of the source type; source definition objects use this name in the `{type: ...}` field. + * @param SourceType - A {@link Source} constructor. + * @param callback - Called when the source type is ready or with an error argument if there is an error. + */ + addSourceType(name, SourceType, callback) { + this._lazyInitEmptyStyle(); + return this.style.addSourceType(name, SourceType, callback); + } + /** + * Removes a source from the map's style. + * + * @param id - The ID of the source to remove. + * @returns `this` + * @example + * ```ts + * map.removeSource('bathymetry-data'); + * ``` + */ + removeSource(id) { + this.style.removeSource(id); + return this._update(true); + } + /** + * Returns the source with the specified ID in the map's style. + * + * This method is often used to update a source using the instance members for the relevant + * source type as defined in [Sources](#sources). + * For example, setting the `data` for a GeoJSON source or updating the `url` and `coordinates` + * of an image source. + * + * @param id - The ID of the source to get. + * @returns The style source with the specified ID or `undefined` if the ID + * corresponds to no existing sources. + * The shape of the object varies by source type. + * A list of options for each source type is available on the MapLibre Style Specification's + * [Sources](https://maplibre.org/maplibre-style-spec/sources/) page. + * @example + * ```ts + * let sourceObject = map.getSource('points'); + * ``` + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + * @see [Animate a point](https://maplibre.org/maplibre-gl-js/docs/examples/animate-point-along-line/) + * @see [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/) + */ + getSource(id) { + return this.style.getSource(id); + } + /** + * Add an image to the style. This image can be displayed on the map like any other icon in the style's + * sprite using the image's ID with + * [`icon-image`](https://maplibre.org/maplibre-style-spec/layers/#layout-symbol-icon-image), + * [`background-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-background-background-pattern), + * [`fill-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-fill-fill-pattern), + * or [`line-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-line-line-pattern). + * + * A {@link ErrorEvent} event will be fired if the image parameter is invalid or there is not enough space in the sprite to add this image. + * + * @param id - The ID of the image. + * @param image - The image as an `HTMLImageElement`, `ImageData`, `ImageBitmap` or object with `width`, `height`, and `data` + * properties with the same format as `ImageData`. + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * // If the style's sprite does not already contain an image with ID 'cat', + * // add the image 'cat-icon.png' to the style's sprite with the ID 'cat'. + * map.loadImage('https://upload.wikimedia.org/wikipedia/commons/thumb/6/60/Cat_silhouette.svg/400px-Cat_silhouette.svg.png', function(error, image) { + * if (error) throw error; + * if (!map.hasImage('cat')) map.addImage('cat', image); + * }); + * + * // Add a stretchable image that can be used with `icon-text-fit` + * // In this example, the image is 600px wide by 400px high. + * map.loadImage('https://upload.wikimedia.org/wikipedia/commons/8/89/Black_and_White_Boxed_%28bordered%29.png', function(error, image) { + * if (error) throw error; + * if (!map.hasImage('border-image')) { + * map.addImage('border-image', image, { + * content: [16, 16, 300, 384], // place text over left half of image, avoiding the 16px border + * stretchX: [[16, 584]], // stretch everything horizontally except the 16px border + * stretchY: [[16, 384]], // stretch everything vertically except the 16px border + * }); + * } + * }); + * ``` + * @see Use `HTMLImageElement`: [Add an icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image/) + * @see Use `ImageData`: [Add a generated icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-generated/) + */ + addImage(id, image, options = {}) { + const { pixelRatio = 1, sdf = false, stretchX, stretchY, content } = options; + this._lazyInitEmptyStyle(); + const version = 0; + if (image instanceof HTMLImageElement || performance.isImageBitmap(image)) { + const { width, height, data } = performance.browser.getImageData(image); + this.style.addImage(id, { data: new performance.RGBAImage({ width, height }, data), pixelRatio, stretchX, stretchY, content, sdf, version }); + } + else if (image.width === undefined || image.height === undefined) { + return this.fire(new performance.ErrorEvent(new Error('Invalid arguments to map.addImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, ' + + 'or object with `width`, `height`, and `data` properties with the same format as `ImageData`'))); + } + else { + const { width, height, data } = image; + const userImage = image; + this.style.addImage(id, { + data: new performance.RGBAImage({ width, height }, new Uint8Array(data)), + pixelRatio, + stretchX, + stretchY, + content, + sdf, + version, + userImage + }); + if (userImage.onAdd) { + userImage.onAdd(this, id); + } + return this; + } + } + /** + * Update an existing image in a style. This image can be displayed on the map like any other icon in the style's + * sprite using the image's ID with + * [`icon-image`](https://maplibre.org/maplibre-style-spec/layers/#layout-symbol-icon-image), + * [`background-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-background-background-pattern), + * [`fill-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-fill-fill-pattern), + * or [`line-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-line-line-pattern). + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param id - The ID of the image. + * @param image - The image as an `HTMLImageElement`, `ImageData`, `ImageBitmap` or object with `width`, `height`, and `data` + * properties with the same format as `ImageData`. + * @returns `this` + * @example + * ```ts + * // If an image with the ID 'cat' already exists in the style's sprite, + * // replace that image with a new image, 'other-cat-icon.png'. + * if (map.hasImage('cat')) map.updateImage('cat', './other-cat-icon.png'); + * ``` + */ + updateImage(id, image) { + const existingImage = this.style.getImage(id); + if (!existingImage) { + return this.fire(new performance.ErrorEvent(new Error('The map has no image with that id. If you are adding a new image use `map.addImage(...)` instead.'))); + } + const imageData = (image instanceof HTMLImageElement || performance.isImageBitmap(image)) ? + performance.browser.getImageData(image) : + image; + const { width, height, data } = imageData; + if (width === undefined || height === undefined) { + return this.fire(new performance.ErrorEvent(new Error('Invalid arguments to map.updateImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, ' + + 'or object with `width`, `height`, and `data` properties with the same format as `ImageData`'))); + } + if (width !== existingImage.data.width || height !== existingImage.data.height) { + return this.fire(new performance.ErrorEvent(new Error('The width and height of the updated image must be that same as the previous version of the image'))); + } + const copy = !(image instanceof HTMLImageElement || performance.isImageBitmap(image)); + existingImage.data.replace(data, copy); + this.style.updateImage(id, existingImage); + return this; + } + /** + * Returns an image, specified by ID, currently available in the map. + * This includes both images from the style's original sprite + * and any images that have been added at runtime using {@link Map#addImage}. + * + * @param id - The ID of the image. + * @returns An image in the map with the specified ID. + * + * @example + * ```ts + * let coffeeShopIcon = map.getImage("coffee_cup"); + * ``` + */ + getImage(id) { + return this.style.getImage(id); + } + /** + * Check whether or not an image with a specific ID exists in the style. This checks both images + * in the style's original sprite and any images + * that have been added at runtime using {@link Map#addImage}. + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param id - The ID of the image. + * + * @returns A Boolean indicating whether the image exists. + * @example + * Check if an image with the ID 'cat' exists in the style's sprite. + * ```ts + * let catIconExists = map.hasImage('cat'); + * ``` + */ + hasImage(id) { + if (!id) { + this.fire(new performance.ErrorEvent(new Error('Missing required image id'))); + return false; + } + return !!this.style.getImage(id); + } + /** + * Remove an image from a style. This can be an image from the style's original + * sprite or any images + * that have been added at runtime using {@link Map#addImage}. + * + * @param id - The ID of the image. + * + * @example + * ```ts + * // If an image with the ID 'cat' exists in + * // the style's sprite, remove it. + * if (map.hasImage('cat')) map.removeImage('cat'); + * ``` + */ + removeImage(id) { + this.style.removeImage(id); + } + /** + * Load an image from an external URL to be used with {@link Map#addImage}. External + * domains must support [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/Access_control_CORS). + * + * @param url - The URL of the image file. Image file must be in png, webp, or jpg format. + * @param callback - Expecting `callback(error, data)`. Called when the image has loaded or with an error argument if there is an error. + * + * @example + * Load an image from an external URL. + * ```ts + * map.loadImage('http://placekitten.com/50/50', function(error, image) { + * if (error) throw error; + * // Add the loaded image to the style's sprite with the ID 'kitten'. + * map.addImage('kitten', image); + * }); + * ``` + * @see [Add an icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image/) + */ + loadImage(url, callback) { + ImageRequest.getImage(this._requestManager.transformRequest(url, ResourceType.Image), callback); + } + /** + * Returns an Array of strings containing the IDs of all images currently available in the map. + * This includes both images from the style's original sprite + * and any images that have been added at runtime using {@link Map#addImage}. + * + * @returns An Array of strings containing the names of all sprites/images currently available in the map. + * + * @example + * ```ts + * let allImages = map.listImages(); + * ``` + */ + listImages() { + return this.style.listImages(); + } + /** + * Adds a [MapLibre style layer](https://maplibre.org/maplibre-style-spec/layers) + * to the map's style. + * + * A layer defines how data from a specified source will be styled. Read more about layer types + * and available paint and layout properties in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/layers). + * + * @param layer - The layer to add, + * conforming to either the MapLibre Style Specification's [layer definition](https://maplibre.org/maplibre-style-spec/layers) or, + * less commonly, the {@link CustomLayerInterface} specification. Can also be a layer definition with an embedded source definition. + * The MapLibre Style Specification's layer definition is appropriate for most layers. + * + * @param beforeId - The ID of an existing layer to insert the new layer before, + * resulting in the new layer appearing visually beneath the existing layer. + * If this argument is not specified, the layer will be appended to the end of the layers array + * and appear visually above all other layers. + * + * @returns `this` + * + * @example + * Add a circle layer with a vector source + * ```ts + * map.addLayer({ + * id: 'points-of-interest', + * source: { + * type: 'vector', + * url: 'https://demotiles.maplibre.org/tiles/tiles.json' + * }, + * 'source-layer': 'poi_label', + * type: 'circle', + * paint: { + * // MapLibre Style Specification paint properties + * }, + * layout: { + * // MapLibre Style Specification layout properties + * } + * }); + * ``` + * + * @example + * Define a source before using it to create a new layer + * ```ts + * map.addSource('state-data', { + * type: 'geojson', + * data: 'path/to/data.geojson' + * }); + * + * map.addLayer({ + * id: 'states', + * // References the GeoJSON source defined above + * // and does not require a `source-layer` + * source: 'state-data', + * type: 'symbol', + * layout: { + * // Set the label content to the + * // feature's `name` property + * text-field: ['get', 'name'] + * } + * }); + * ``` + * + * @example + * Add a new symbol layer before an existing layer + * ```ts + * map.addLayer({ + * id: 'states', + * // References a source that's already been defined + * source: 'state-data', + * type: 'symbol', + * layout: { + * // Set the label content to the + * // feature's `name` property + * text-field: ['get', 'name'] + * } + * // Add the layer before the existing `cities` layer + * }, 'cities'); + * ``` + * @see [Create and style clusters](https://maplibre.org/maplibre-gl-js/docs/examples/cluster/) + * @see [Add a vector tile source](https://maplibre.org/maplibre-gl-js/docs/examples/vector-source/) + * @see [Add a WMS source](https://maplibre.org/maplibre-gl-js/docs/examples/wms/) + */ + addLayer(layer, beforeId) { + this._lazyInitEmptyStyle(); + this.style.addLayer(layer, beforeId); + return this._update(true); + } + /** + * Moves a layer to a different z-position. + * + * @param id - The ID of the layer to move. + * @param beforeId - The ID of an existing layer to insert the new layer before. When viewing the map, the `id` layer will appear beneath the `beforeId` layer. If `beforeId` is omitted, the layer will be appended to the end of the layers array and appear above all other layers on the map. + * @returns `this` + * + * @example + * Move a layer with ID 'polygon' before the layer with ID 'country-label'. The `polygon` layer will appear beneath the `country-label` layer on the map. + * ```ts + * map.moveLayer('polygon', 'country-label'); + * ``` + */ + moveLayer(id, beforeId) { + this.style.moveLayer(id, beforeId); + return this._update(true); + } + /** + * Removes the layer with the given ID from the map's style. + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param id - The ID of the layer to remove + * @returns `this` + * + * @example + * If a layer with ID 'state-data' exists, remove it. + * ```ts + * if (map.getLayer('state-data')) map.removeLayer('state-data'); + * ``` + */ + removeLayer(id) { + this.style.removeLayer(id); + return this._update(true); + } + /** + * Returns the layer with the specified ID in the map's style. + * + * @param id - The ID of the layer to get. + * @returns The layer with the specified ID, or `undefined` + * if the ID corresponds to no existing layers. + * + * @example + * ```ts + * let stateDataLayer = map.getLayer('state-data'); + * ``` + * @see [Filter symbols by toggling a list](https://maplibre.org/maplibre-gl-js/docs/examples/filter-markers/) + * @see [Filter symbols by text input](https://maplibre.org/maplibre-gl-js/docs/examples/filter-markers-by-input/) + */ + getLayer(id) { + return this.style.getLayer(id); + } + /** + * Return the ids of all layers currently in the style, including custom layers, in order. + * + * @returns ids of layers, in order + * + * @example + * ```ts + * const orderedLayerIds = map.getLayersOrder(); + * ``` + */ + getLayersOrder() { + return this.style.getLayersOrder(); + } + /** + * Sets the zoom extent for the specified style layer. The zoom extent includes the + * [minimum zoom level](https://maplibre.org/maplibre-style-spec/layers/#minzoom) + * and [maximum zoom level](https://maplibre.org/maplibre-style-spec/layers/#maxzoom)) + * at which the layer will be rendered. + * + * Note: For style layers using vector sources, style layers cannot be rendered at zoom levels lower than the + * minimum zoom level of the _source layer_ because the data does not exist at those zoom levels. If the minimum + * zoom level of the source layer is higher than the minimum zoom level defined in the style layer, the style + * layer will not be rendered at all zoom levels in the zoom range. + * + * @param layerId - The ID of the layer to which the zoom extent will be applied. + * @param minzoom - The minimum zoom to set (0-24). + * @param maxzoom - The maximum zoom to set (0-24). + * @returns `this` + * + * @example + * ```ts + * map.setLayerZoomRange('my-layer', 2, 5); + * ``` + */ + setLayerZoomRange(layerId, minzoom, maxzoom) { + this.style.setLayerZoomRange(layerId, minzoom, maxzoom); + return this._update(true); + } + /** + * Sets the filter for the specified style layer. + * + * Filters control which features a style layer renders from its source. + * Any feature for which the filter expression evaluates to `true` will be + * rendered on the map. Those that are false will be hidden. + * + * Use `setFilter` to show a subset of your source data. + * + * To clear the filter, pass `null` or `undefined` as the second parameter. + * + * @param layerId - The ID of the layer to which the filter will be applied. + * @param filter - The filter, conforming to the MapLibre Style Specification's + * [filter definition](https://maplibre.org/maplibre-style-spec/layers/#filter). If `null` or `undefined` is provided, the function removes any existing filter from the layer. + * @param options - Options object. + * @returns `this` + * + * @example + * Display only features with the 'name' property 'USA' + * ```ts + * map.setFilter('my-layer', ['==', ['get', 'name'], 'USA']); + * ``` + * @example + * Display only features with five or more 'available-spots' + * ```ts + * map.setFilter('bike-docks', ['>=', ['get', 'available-spots'], 5]); + * ``` + * @example + * Remove the filter for the 'bike-docks' style layer + * ```ts + * map.setFilter('bike-docks', null); + * ``` + * @see [Create a timeline animation](https://maplibre.org/maplibre-gl-js/docs/examples/timeline-animation/) + */ + setFilter(layerId, filter, options = {}) { + this.style.setFilter(layerId, filter, options); + return this._update(true); + } + /** + * Returns the filter applied to the specified style layer. + * + * @param layerId - The ID of the style layer whose filter to get. + * @returns The layer's filter. + */ + getFilter(layerId) { + return this.style.getFilter(layerId); + } + /** + * Sets the value of a paint property in the specified style layer. + * + * @param layerId - The ID of the layer to set the paint property in. + * @param name - The name of the paint property to set. + * @param value - The value of the paint property to set. + * Must be of a type appropriate for the property, as defined in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/). + * Pass `null` to unset the existing value. + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.setPaintProperty('my-layer', 'fill-color', '#faafee'); + * ``` + * @see [Change a layer's color with buttons](https://maplibre.org/maplibre-gl-js/docs/examples/color-switcher/) + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + setPaintProperty(layerId, name, value, options = {}) { + this.style.setPaintProperty(layerId, name, value, options); + return this._update(true); + } + /** + * Returns the value of a paint property in the specified style layer. + * + * @param layerId - The ID of the layer to get the paint property from. + * @param name - The name of a paint property to get. + * @returns The value of the specified paint property. + */ + getPaintProperty(layerId, name) { + return this.style.getPaintProperty(layerId, name); + } + /** + * Sets the value of a layout property in the specified style layer. + * + * @param layerId - The ID of the layer to set the layout property in. + * @param name - The name of the layout property to set. + * @param value - The value of the layout property. Must be of a type appropriate for the property, as defined in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/). + * @param options - The options object. + * @returns `this` + * @example + * ```ts + * map.setLayoutProperty('my-layer', 'visibility', 'none'); + * ``` + */ + setLayoutProperty(layerId, name, value, options = {}) { + this.style.setLayoutProperty(layerId, name, value, options); + return this._update(true); + } + /** + * Returns the value of a layout property in the specified style layer. + * + * @param layerId - The ID of the layer to get the layout property from. + * @param name - The name of the layout property to get. + * @returns The value of the specified layout property. + */ + getLayoutProperty(layerId, name) { + return this.style.getLayoutProperty(layerId, name); + } + /** + * Sets the value of the style's glyphs property. + * + * @param glyphsUrl - Glyph URL to set. Must conform to the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/glyphs/). + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.setGlyphs('https://demotiles.maplibre.org/font/{fontstack}/{range}.pbf'); + * ``` + */ + setGlyphs(glyphsUrl, options = {}) { + this._lazyInitEmptyStyle(); + this.style.setGlyphs(glyphsUrl, options); + return this._update(true); + } + /** + * Returns the value of the style's glyphs URL + * + * @returns glyphs Style's glyphs url + */ + getGlyphs() { + return this.style.getGlyphsUrl(); + } + /** + * Adds a sprite to the map's style. Fires the `style` event. + * + * @param id - The ID of the sprite to add. Must not conflict with existing sprites. + * @param url - The URL to load the sprite from + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.addSprite('sprite-two', 'http://example.com/sprite-two'); + * ``` + */ + addSprite(id, url, options = {}) { + this._lazyInitEmptyStyle(); + this.style.addSprite(id, url, options, (err) => { + if (!err) { + this._update(true); + } + }); + return this; + } + /** + * Removes the sprite from the map's style. Fires the `style` event. + * + * @param id - The ID of the sprite to remove. If the sprite is declared as a single URL, the ID must be "default". + * @returns `this` + * @example + * ```ts + * map.removeSprite('sprite-two'); + * map.removeSprite('default'); + * ``` + */ + removeSprite(id) { + this._lazyInitEmptyStyle(); + this.style.removeSprite(id); + return this._update(true); + } + /** + * Returns the as-is value of the style's sprite. + * + * @returns style's sprite list of id-url pairs + */ + getSprite() { + return this.style.getSprite(); + } + /** + * Sets the value of the style's sprite property. + * + * @param spriteUrl - Sprite URL to set. + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.setSprite('YOUR_SPRITE_URL'); + * ``` + */ + setSprite(spriteUrl, options = {}) { + this._lazyInitEmptyStyle(); + this.style.setSprite(spriteUrl, options, (err) => { + if (!err) { + this._update(true); + } + }); + return this; + } + /** + * Sets the any combination of light values. + * + * @param light - Light properties to set. Must conform to the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/light). + * @param options - Options object. + * @returns `this` + * + * @example + * ```ts + * let layerVisibility = map.getLayoutProperty('my-layer', 'visibility'); + * ``` + */ + setLight(light, options = {}) { + this._lazyInitEmptyStyle(); + this.style.setLight(light, options); + return this._update(true); + } + /** + * Returns the value of the light object. + * + * @returns light Light properties of the style. + */ + getLight() { + return this.style.getLight(); + } + /** + * Sets the `state` of a feature. + * A feature's `state` is a set of user-defined key-value pairs that are assigned to a feature at runtime. + * When using this method, the `state` object is merged with any existing key-value pairs in the feature's state. + * Features are identified by their `feature.id` attribute, which can be any number or string. + * + * This method can only be used with sources that have a `feature.id` attribute. The `feature.id` attribute can be defined in three ways: + * - For vector or GeoJSON sources, including an `id` attribute in the original data file. + * - For vector or GeoJSON sources, using the [`promoteId`](https://maplibre.org/maplibre-style-spec/sources/#vector-promoteId) option at the time the source is defined. + * - For GeoJSON sources, using the [`generateId`](https://maplibre.org/maplibre-style-spec/sources/#geojson-generateId) option to auto-assign an `id` based on the feature's index in the source data. If you change feature data using `map.getSource('some id').setData(..)`, you may need to re-apply state taking into account updated `id` values. + * + * _Note: You can use the [`feature-state` expression](https://maplibre.org/maplibre-style-spec/expressions/#feature-state) to access the values in a feature's state object for the purposes of styling._ + * + * @param feature - Feature identifier. Feature objects returned from + * {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers. + * @param state - A set of key-value pairs. The values should be valid JSON types. + * @returns `this` + * + * @example + * ```ts + * // When the mouse moves over the `my-layer` layer, update + * // the feature state for the feature under the mouse + * map.on('mousemove', 'my-layer', function(e) { + * if (e.features.length > 0) { + * map.setFeatureState({ + * source: 'my-source', + * sourceLayer: 'my-source-layer', + * id: e.features[0].id, + * }, { + * hover: true + * }); + * } + * }); + * ``` + * @see [Create a hover effect](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + */ + setFeatureState(feature, state) { + this.style.setFeatureState(feature, state); + return this._update(); + } + /** + * Removes the `state` of a feature, setting it back to the default behavior. + * If only a `target.source` is specified, it will remove the state for all features from that source. + * If `target.id` is also specified, it will remove all keys for that feature's state. + * If `key` is also specified, it removes only that key from that feature's state. + * Features are identified by their `feature.id` attribute, which can be any number or string. + * + * @param target - Identifier of where to remove state. It can be a source, a feature, or a specific key of feature. + * Feature objects returned from {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers. + * @param key - (optional) The key in the feature state to reset. + * @returns `this` + * @example + * Reset the entire state object for all features in the `my-source` source + * ```ts + * map.removeFeatureState({ + * source: 'my-source' + * }); + * ``` + * + * @example + * When the mouse leaves the `my-layer` layer, + * reset the entire state object for the + * feature under the mouse + * ```ts + * map.on('mouseleave', 'my-layer', function(e) { + * map.removeFeatureState({ + * source: 'my-source', + * sourceLayer: 'my-source-layer', + * id: e.features[0].id + * }); + * }); + * ``` + * + * @example + * When the mouse leaves the `my-layer` layer, + * reset only the `hover` key-value pair in the + * state for the feature under the mouse + * ```ts + * map.on('mouseleave', 'my-layer', function(e) { + * map.removeFeatureState({ + * source: 'my-source', + * sourceLayer: 'my-source-layer', + * id: e.features[0].id + * }, 'hover'); + * }); + * ``` + */ + removeFeatureState(target, key) { + this.style.removeFeatureState(target, key); + return this._update(); + } + /** + * Gets the `state` of a feature. + * A feature's `state` is a set of user-defined key-value pairs that are assigned to a feature at runtime. + * Features are identified by their `feature.id` attribute, which can be any number or string. + * + * _Note: To access the values in a feature's state object for the purposes of styling the feature, use the [`feature-state` expression](https://maplibre.org/maplibre-style-spec/expressions/#feature-state)._ + * + * @param feature - Feature identifier. Feature objects returned from + * {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers. + * @returns The state of the feature: a set of key-value pairs that was assigned to the feature at runtime. + * + * @example + * When the mouse moves over the `my-layer` layer, + * get the feature state for the feature under the mouse + * ```ts + * map.on('mousemove', 'my-layer', function(e) { + * if (e.features.length > 0) { + * map.getFeatureState({ + * source: 'my-source', + * sourceLayer: 'my-source-layer', + * id: e.features[0].id + * }); + * } + * }); + * ``` + */ + getFeatureState(feature) { + return this.style.getFeatureState(feature); + } + /** + * Returns the map's containing HTML element. + * + * @returns The map's container. + */ + getContainer() { + return this._container; + } + /** + * Returns the HTML element containing the map's `` element. + * + * If you want to add non-GL overlays to the map, you should append them to this element. + * + * This is the element to which event bindings for map interactivity (such as panning and zooming) are + * attached. It will receive bubbled events from child elements such as the ``, but not from + * map controls. + * + * @returns The container of the map's ``. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + getCanvasContainer() { + return this._canvasContainer; + } + /** + * Returns the map's `` element. + * + * @returns The map's `` element. + * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/) + */ + getCanvas() { + return this._canvas; + } + _containerDimensions() { + let width = 0; + let height = 0; + if (this._container) { + width = this._container.clientWidth || 400; + height = this._container.clientHeight || 300; + } + return [width, height]; + } + _setupContainer() { + const container = this._container; + container.classList.add('maplibregl-map'); + const canvasContainer = this._canvasContainer = DOM.create('div', 'maplibregl-canvas-container', container); + if (this._interactive) { + canvasContainer.classList.add('maplibregl-interactive'); + } + this._canvas = DOM.create('canvas', 'maplibregl-canvas', canvasContainer); + this._canvas.addEventListener('webglcontextlost', this._contextLost, false); + this._canvas.addEventListener('webglcontextrestored', this._contextRestored, false); + this._canvas.setAttribute('tabindex', '0'); + this._canvas.setAttribute('aria-label', 'Map'); + this._canvas.setAttribute('role', 'region'); + const dimensions = this._containerDimensions(); + const clampedPixelRatio = this._getClampedPixelRatio(dimensions[0], dimensions[1]); + this._resizeCanvas(dimensions[0], dimensions[1], clampedPixelRatio); + const controlContainer = this._controlContainer = DOM.create('div', 'maplibregl-control-container', container); + const positions = this._controlPositions = {}; + ['top-left', 'top-right', 'bottom-left', 'bottom-right'].forEach((positionName) => { + positions[positionName] = DOM.create('div', `maplibregl-ctrl-${positionName} `, controlContainer); + }); + this._container.addEventListener('scroll', this._onMapScroll, false); + } + _setupCooperativeGestures() { + const container = this._container; + this._cooperativeGesturesScreen = DOM.create('div', 'maplibregl-cooperative-gesture-screen', container); + let desktopMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.windowsHelpText ? this._cooperativeGestures.windowsHelpText : 'Use Ctrl + scroll to zoom the map'; + if (navigator.platform.indexOf('Mac') === 0) { + desktopMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.macHelpText ? this._cooperativeGestures.macHelpText : 'Use ⌘ + scroll to zoom the map'; + } + const mobileMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.mobileHelpText ? this._cooperativeGestures.mobileHelpText : 'Use two fingers to move the map'; + this._cooperativeGesturesScreen.innerHTML = ` +
${desktopMessage}
+
${mobileMessage}
+ `; + // Remove cooperative gesture screen from the accessibility tree since screenreaders cannot interact with the map using gestures + this._cooperativeGesturesScreen.setAttribute('aria-hidden', 'true'); + // Add event to canvas container since gesture container is pointer-events: none + this._canvasContainer.addEventListener('wheel', this._cooperativeGesturesOnWheel, false); + // Add a cooperative gestures class (enable touch-action: pan-x pan-y;) + this._canvasContainer.classList.add('maplibregl-cooperative-gestures'); + } + _destroyCooperativeGestures() { + DOM.remove(this._cooperativeGesturesScreen); + this._canvasContainer.removeEventListener('wheel', this._cooperativeGesturesOnWheel, false); + this._canvasContainer.classList.remove('maplibregl-cooperative-gestures'); + } + _resizeCanvas(width, height, pixelRatio) { + // Request the required canvas size taking the pixelratio into account. + this._canvas.width = Math.floor(pixelRatio * width); + this._canvas.height = Math.floor(pixelRatio * height); + // Maintain the same canvas size, potentially downscaling it for HiDPI displays + this._canvas.style.width = `${width}px`; + this._canvas.style.height = `${height}px`; + } + _setupPainter() { + const attributes = { + alpha: true, + stencil: true, + depth: true, + failIfMajorPerformanceCaveat: this._failIfMajorPerformanceCaveat, + preserveDrawingBuffer: this._preserveDrawingBuffer, + antialias: this._antialias || false + }; + let webglcontextcreationerrorDetailObject = null; + this._canvas.addEventListener('webglcontextcreationerror', (args) => { + webglcontextcreationerrorDetailObject = { requestedAttributes: attributes }; + if (args) { + webglcontextcreationerrorDetailObject.statusMessage = args.statusMessage; + webglcontextcreationerrorDetailObject.type = args.type; + } + }, { once: true }); + const gl = this._canvas.getContext('webgl2', attributes) || + this._canvas.getContext('webgl', attributes); + if (!gl) { + const msg = 'Failed to initialize WebGL'; + if (webglcontextcreationerrorDetailObject) { + webglcontextcreationerrorDetailObject.message = msg; + throw new Error(JSON.stringify(webglcontextcreationerrorDetailObject)); + } + else { + throw new Error(msg); + } + } + this.painter = new Painter(gl, this.transform); + webpSupported.testSupport(gl); + } + _onCooperativeGesture(event, metaPress, touches) { + if (!metaPress && touches < 2) { + // Alert user how to scroll/pan + this._cooperativeGesturesScreen.classList.add('maplibregl-show'); + setTimeout(() => { + this._cooperativeGesturesScreen.classList.remove('maplibregl-show'); + }, 100); + } + return false; + } + /** + * Returns a Boolean indicating whether the map is fully loaded. + * + * Returns `false` if the style is not yet fully loaded, + * or if there has been a change to the sources or style that + * has not yet fully loaded. + * + * @returns A Boolean indicating whether the map is fully loaded. + */ + loaded() { + return !this._styleDirty && !this._sourcesDirty && !!this.style && this.style.loaded(); + } + /** + * @internal + * Update this map's style and sources, and re-render the map. + * + * @param updateStyle - mark the map's style for reprocessing as + * well as its sources + * @returns `this` + */ + _update(updateStyle) { + if (!this.style || !this.style._loaded) + return this; + this._styleDirty = this._styleDirty || updateStyle; + this._sourcesDirty = true; + this.triggerRepaint(); + return this; + } + /** + * @internal + * Request that the given callback be executed during the next render + * frame. Schedule a render frame if one is not already scheduled. + * + * @returns An id that can be used to cancel the callback + */ + _requestRenderFrame(callback) { + this._update(); + return this._renderTaskQueue.add(callback); + } + _cancelRenderFrame(id) { + this._renderTaskQueue.remove(id); + } + /** + * @internal + * Call when a (re-)render of the map is required: + * - The style has changed (`setPaintProperty()`, etc.) + * - Source data has changed (e.g. tiles have finished loading) + * - The map has is moving (or just finished moving) + * - A transition is in progress + * + * @param paintStartTimeStamp - The time when the animation frame began executing. + * + * @returns `this` + */ + _render(paintStartTimeStamp) { + const fadeDuration = this._idleTriggered ? this._fadeDuration : 0; + // A custom layer may have used the context asynchronously. Mark the state as dirty. + this.painter.context.setDirty(); + this.painter.setBaseState(); + this._renderTaskQueue.run(paintStartTimeStamp); + // A task queue callback may have fired a user event which may have removed the map + if (this._removed) + return; + let crossFading = false; + // If the style has changed, the map is being zoomed, or a transition or fade is in progress: + // - Apply style changes (in a batch) + // - Recalculate paint properties. + if (this.style && this._styleDirty) { + this._styleDirty = false; + const zoom = this.transform.zoom; + const now = performance.browser.now(); + this.style.zoomHistory.update(zoom, now); + const parameters = new performance.EvaluationParameters(zoom, { + now, + fadeDuration, + zoomHistory: this.style.zoomHistory, + transition: this.style.getTransition() + }); + const factor = parameters.crossFadingFactor(); + if (factor !== 1 || factor !== this._crossFadingFactor) { + crossFading = true; + this._crossFadingFactor = factor; + } + this.style.update(parameters); + } + // If we are in _render for any reason other than an in-progress paint + // transition, update source caches to check for and load any tiles we + // need for the current transform + if (this.style && this._sourcesDirty) { + this._sourcesDirty = false; + this.style._updateSources(this.transform); + } + // update terrain stuff + if (this.terrain) { + this.terrain.sourceCache.update(this.transform, this.terrain); + this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom); + if (!this._elevationFreeze) { + this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom); + } + } + else { + this.transform._minEleveationForCurrentTile = 0; + this.transform.elevation = 0; + } + this._placementDirty = this.style && this.style._updatePlacement(this.painter.transform, this.showCollisionBoxes, fadeDuration, this._crossSourceCollisions); + // Actually draw + this.painter.render(this.style, { + showTileBoundaries: this.showTileBoundaries, + showOverdrawInspector: this._showOverdrawInspector, + rotating: this.isRotating(), + zooming: this.isZooming(), + moving: this.isMoving(), + fadeDuration, + showPadding: this.showPadding, + }); + this.fire(new performance.Event('render')); + if (this.loaded() && !this._loaded) { + this._loaded = true; + performance.PerformanceUtils.mark(performance.PerformanceMarkers.load); + this.fire(new performance.Event('load')); + } + if (this.style && (this.style.hasTransitions() || crossFading)) { + this._styleDirty = true; + } + if (this.style && !this._placementDirty) { + // Since no fade operations are in progress, we can release + // all tiles held for fading. If we didn't do this, the tiles + // would just sit in the SourceCaches until the next render + this.style._releaseSymbolFadeTiles(); + } + // Schedule another render frame if it's needed. + // + // Even though `_styleDirty` and `_sourcesDirty` are reset in this + // method, synchronous events fired during Style#update or + // Style#_updateSources could have caused them to be set again. + const somethingDirty = this._sourcesDirty || this._styleDirty || this._placementDirty; + if (somethingDirty || this._repaint) { + this.triggerRepaint(); + } + else if (!this.isMoving() && this.loaded()) { + this.fire(new performance.Event('idle')); + } + if (this._loaded && !this._fullyLoaded && !somethingDirty) { + this._fullyLoaded = true; + performance.PerformanceUtils.mark(performance.PerformanceMarkers.fullLoad); + } + return this; + } + /** + * Force a synchronous redraw of the map. + * @returns `this` + * @example + * ```ts + * map.redraw(); + * ``` + */ + redraw() { + if (this.style) { + // cancel the scheduled update + if (this._frame) { + this._frame.cancel(); + this._frame = null; + } + this._render(0); + } + return this; + } + /** + * Clean up and release all internal resources associated with this map. + * + * This includes DOM elements, event bindings, web workers, and WebGL resources. + * + * Use this method when you are done using the map and wish to ensure that it no + * longer consumes browser resources. Afterwards, you must not call any other + * methods on the map. + */ + remove() { + var _a; + if (this._hash) + this._hash.remove(); + for (const control of this._controls) + control.onRemove(this); + this._controls = []; + if (this._frame) { + this._frame.cancel(); + this._frame = null; + } + this._renderTaskQueue.clear(); + this.painter.destroy(); + this.handlers.destroy(); + delete this.handlers; + this.setStyle(null); + if (typeof window !== 'undefined') { + removeEventListener('online', this._onWindowOnline, false); + } + ImageRequest.removeThrottleControl(this._imageQueueHandle); + (_a = this._resizeObserver) === null || _a === void 0 ? void 0 : _a.disconnect(); + const extension = this.painter.context.gl.getExtension('WEBGL_lose_context'); + if (extension) + extension.loseContext(); + this._canvas.removeEventListener('webglcontextrestored', this._contextRestored, false); + this._canvas.removeEventListener('webglcontextlost', this._contextLost, false); + DOM.remove(this._canvasContainer); + DOM.remove(this._controlContainer); + if (this._cooperativeGestures) { + this._destroyCooperativeGestures(); + } + this._container.classList.remove('maplibregl-map'); + performance.PerformanceUtils.clearMetrics(); + this._removed = true; + this.fire(new performance.Event('remove')); + } + /** + * Trigger the rendering of a single frame. Use this method with custom layers to + * repaint the map when the layer changes. Calling this multiple times before the + * next frame is rendered will still result in only a single frame being rendered. + * @example + * ```ts + * map.triggerRepaint(); + * ``` + * @see [Add a 3D model](https://maplibre.org/maplibre-gl-js/docs/examples/add-3d-model/) + * @see [Add an animated icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-animated/) + */ + triggerRepaint() { + if (this.style && !this._frame) { + this._frame = performance.browser.frame((paintStartTimeStamp) => { + performance.PerformanceUtils.frame(paintStartTimeStamp); + this._frame = null; + this._render(paintStartTimeStamp); + }); + } + } + /** + * Gets and sets a Boolean indicating whether the map will render an outline + * around each tile and the tile ID. These tile boundaries are useful for + * debugging. + * + * The uncompressed file size of the first vector source is drawn in the top left + * corner of each tile, next to the tile ID. + * + * @example + * ```ts + * map.showTileBoundaries = true; + * ``` + */ + get showTileBoundaries() { return !!this._showTileBoundaries; } + set showTileBoundaries(value) { + if (this._showTileBoundaries === value) + return; + this._showTileBoundaries = value; + this._update(); + } + /** + * Gets and sets a Boolean indicating whether the map will visualize + * the padding offsets. + */ + get showPadding() { return !!this._showPadding; } + set showPadding(value) { + if (this._showPadding === value) + return; + this._showPadding = value; + this._update(); + } + /** + * Gets and sets a Boolean indicating whether the map will render boxes + * around all symbols in the data source, revealing which symbols + * were rendered or which were hidden due to collisions. + * This information is useful for debugging. + */ + get showCollisionBoxes() { return !!this._showCollisionBoxes; } + set showCollisionBoxes(value) { + if (this._showCollisionBoxes === value) + return; + this._showCollisionBoxes = value; + if (value) { + // When we turn collision boxes on we have to generate them for existing tiles + // When we turn them off, there's no cost to leaving existing boxes in place + this.style._generateCollisionBoxes(); + } + else { + // Otherwise, call an update to remove collision boxes + this._update(); + } + } + /** + * Gets and sets a Boolean indicating whether the map should color-code + * each fragment to show how many times it has been shaded. + * White fragments have been shaded 8 or more times. + * Black fragments have been shaded 0 times. + * This information is useful for debugging. + */ + get showOverdrawInspector() { return !!this._showOverdrawInspector; } + set showOverdrawInspector(value) { + if (this._showOverdrawInspector === value) + return; + this._showOverdrawInspector = value; + this._update(); + } + /** + * Gets and sets a Boolean indicating whether the map will + * continuously repaint. This information is useful for analyzing performance. + */ + get repaint() { return !!this._repaint; } + set repaint(value) { + if (this._repaint !== value) { + this._repaint = value; + this.triggerRepaint(); + } + } + // show vertices + get vertices() { return !!this._vertices; } + set vertices(value) { this._vertices = value; this._update(); } + /** + * Returns the package version of the library + * @returns Package version of the library + */ + get version() { + return version$1; + } + /** + * Returns the elevation for the point where the camera is looking. + * This value corresponds to: + * "meters above sea level" * "exaggeration" + * @returns The elevation. + */ + getCameraTargetElevation() { + return this.transform.elevation; + } +}; + +const assignEvents = (handler) => { + handler.touchstart = handler.dragStart; + handler.touchmoveWindow = handler.dragMove; + handler.touchend = handler.dragEnd; +}; +const generateOneFingerTouchRotationHandler = ({ enable, clickTolerance, bearingDegreesPerPixelMoved = 0.8 }) => { + const touchMoveStateManager = new OneFingerTouchMoveStateManager(); + return new DragHandler({ + clickTolerance, + move: (lastPoint, point) => ({ bearingDelta: (point.x - lastPoint.x) * bearingDegreesPerPixelMoved }), + moveStateManager: touchMoveStateManager, + enable, + assignEvents, + }); +}; +const generateOneFingerTouchPitchHandler = ({ enable, clickTolerance, pitchDegreesPerPixelMoved = -0.5 }) => { + const touchMoveStateManager = new OneFingerTouchMoveStateManager(); + return new DragHandler({ + clickTolerance, + move: (lastPoint, point) => ({ pitchDelta: (point.y - lastPoint.y) * pitchDegreesPerPixelMoved }), + moveStateManager: touchMoveStateManager, + enable, + assignEvents, + }); +}; + +const defaultOptions$3 = { + showCompass: true, + showZoom: true, + visualizePitch: false +}; +/** + * A `NavigationControl` control contains zoom buttons and a compass. + * + * @group Markers and Controls + * + * @example + * ```ts + * let nav = new maplibregl.NavigationControl(); + * map.addControl(nav, 'top-left'); + * ``` + * @see [Display map navigation controls](https://maplibre.org/maplibre-gl-js/docs/examples/navigation/) + */ +class NavigationControl { + /** + * @param options - the control's options + */ + constructor(options) { + this._updateZoomButtons = () => { + const zoom = this._map.getZoom(); + const isMax = zoom === this._map.getMaxZoom(); + const isMin = zoom === this._map.getMinZoom(); + this._zoomInButton.disabled = isMax; + this._zoomOutButton.disabled = isMin; + this._zoomInButton.setAttribute('aria-disabled', isMax.toString()); + this._zoomOutButton.setAttribute('aria-disabled', isMin.toString()); + }; + this._rotateCompassArrow = () => { + const rotate = this.options.visualizePitch ? + `scale(${1 / Math.pow(Math.cos(this._map.transform.pitch * (Math.PI / 180)), 0.5)}) rotateX(${this._map.transform.pitch}deg) rotateZ(${this._map.transform.angle * (180 / Math.PI)}deg)` : + `rotate(${this._map.transform.angle * (180 / Math.PI)}deg)`; + this._compassIcon.style.transform = rotate; + }; + this._setButtonTitle = (button, title) => { + const str = this._map._getUIString(`NavigationControl.${title}`); + button.title = str; + button.setAttribute('aria-label', str); + }; + this.options = performance.extend({}, defaultOptions$3, options); + this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group'); + this._container.addEventListener('contextmenu', (e) => e.preventDefault()); + if (this.options.showZoom) { + this._zoomInButton = this._createButton('maplibregl-ctrl-zoom-in', (e) => this._map.zoomIn({}, { originalEvent: e })); + DOM.create('span', 'maplibregl-ctrl-icon', this._zoomInButton).setAttribute('aria-hidden', 'true'); + this._zoomOutButton = this._createButton('maplibregl-ctrl-zoom-out', (e) => this._map.zoomOut({}, { originalEvent: e })); + DOM.create('span', 'maplibregl-ctrl-icon', this._zoomOutButton).setAttribute('aria-hidden', 'true'); + } + if (this.options.showCompass) { + this._compass = this._createButton('maplibregl-ctrl-compass', (e) => { + if (this.options.visualizePitch) { + this._map.resetNorthPitch({}, { originalEvent: e }); + } + else { + this._map.resetNorth({}, { originalEvent: e }); + } + }); + this._compassIcon = DOM.create('span', 'maplibregl-ctrl-icon', this._compass); + this._compassIcon.setAttribute('aria-hidden', 'true'); + } + } + onAdd(map) { + this._map = map; + if (this.options.showZoom) { + this._setButtonTitle(this._zoomInButton, 'ZoomIn'); + this._setButtonTitle(this._zoomOutButton, 'ZoomOut'); + this._map.on('zoom', this._updateZoomButtons); + this._updateZoomButtons(); + } + if (this.options.showCompass) { + this._setButtonTitle(this._compass, 'ResetBearing'); + if (this.options.visualizePitch) { + this._map.on('pitch', this._rotateCompassArrow); + } + this._map.on('rotate', this._rotateCompassArrow); + this._rotateCompassArrow(); + this._handler = new MouseRotateWrapper(this._map, this._compass, this.options.visualizePitch); + } + return this._container; + } + onRemove() { + DOM.remove(this._container); + if (this.options.showZoom) { + this._map.off('zoom', this._updateZoomButtons); + } + if (this.options.showCompass) { + if (this.options.visualizePitch) { + this._map.off('pitch', this._rotateCompassArrow); + } + this._map.off('rotate', this._rotateCompassArrow); + this._handler.off(); + delete this._handler; + } + delete this._map; + } + _createButton(className, fn) { + const a = DOM.create('button', className, this._container); + a.type = 'button'; + a.addEventListener('click', fn); + return a; + } +} +class MouseRotateWrapper { + constructor(map, element, pitch = false) { + this.mousedown = (e) => { + this.startMouse(performance.extend({}, e, { ctrlKey: true, preventDefault: () => e.preventDefault() }), DOM.mousePos(this.element, e)); + DOM.addEventListener(window, 'mousemove', this.mousemove); + DOM.addEventListener(window, 'mouseup', this.mouseup); + }; + this.mousemove = (e) => { + this.moveMouse(e, DOM.mousePos(this.element, e)); + }; + this.mouseup = (e) => { + this.mouseRotate.dragEnd(e); + if (this.mousePitch) + this.mousePitch.dragEnd(e); + this.offTemp(); + }; + this.touchstart = (e) => { + if (e.targetTouches.length !== 1) { + this.reset(); + } + else { + this._startPos = this._lastPos = DOM.touchPos(this.element, e.targetTouches)[0]; + this.startTouch(e, this._startPos); + DOM.addEventListener(window, 'touchmove', this.touchmove, { passive: false }); + DOM.addEventListener(window, 'touchend', this.touchend); + } + }; + this.touchmove = (e) => { + if (e.targetTouches.length !== 1) { + this.reset(); + } + else { + this._lastPos = DOM.touchPos(this.element, e.targetTouches)[0]; + this.moveTouch(e, this._lastPos); + } + }; + this.touchend = (e) => { + if (e.targetTouches.length === 0 && + this._startPos && + this._lastPos && + this._startPos.dist(this._lastPos) < this._clickTolerance) { + this.element.click(); + } + delete this._startPos; + delete this._lastPos; + this.offTemp(); + }; + this.reset = () => { + this.mouseRotate.reset(); + if (this.mousePitch) + this.mousePitch.reset(); + this.touchRotate.reset(); + if (this.touchPitch) + this.touchPitch.reset(); + delete this._startPos; + delete this._lastPos; + this.offTemp(); + }; + this._clickTolerance = 10; + const mapRotateTolerance = map.dragRotate._mouseRotate.getClickTolerance(); + const mapPitchTolerance = map.dragRotate._mousePitch.getClickTolerance(); + this.element = element; + this.mouseRotate = generateMouseRotationHandler({ clickTolerance: mapRotateTolerance, enable: true }); + this.touchRotate = generateOneFingerTouchRotationHandler({ clickTolerance: mapRotateTolerance, enable: true }); + this.map = map; + if (pitch) { + this.mousePitch = generateMousePitchHandler({ clickTolerance: mapPitchTolerance, enable: true }); + this.touchPitch = generateOneFingerTouchPitchHandler({ clickTolerance: mapPitchTolerance, enable: true }); + } + DOM.addEventListener(element, 'mousedown', this.mousedown); + DOM.addEventListener(element, 'touchstart', this.touchstart, { passive: false }); + DOM.addEventListener(element, 'touchcancel', this.reset); + } + startMouse(e, point) { + this.mouseRotate.dragStart(e, point); + if (this.mousePitch) + this.mousePitch.dragStart(e, point); + DOM.disableDrag(); + } + startTouch(e, point) { + this.touchRotate.dragStart(e, point); + if (this.touchPitch) + this.touchPitch.dragStart(e, point); + DOM.disableDrag(); + } + moveMouse(e, point) { + const map = this.map; + const { bearingDelta } = this.mouseRotate.dragMove(e, point) || {}; + if (bearingDelta) + map.setBearing(map.getBearing() + bearingDelta); + if (this.mousePitch) { + const { pitchDelta } = this.mousePitch.dragMove(e, point) || {}; + if (pitchDelta) + map.setPitch(map.getPitch() + pitchDelta); + } + } + moveTouch(e, point) { + const map = this.map; + const { bearingDelta } = this.touchRotate.dragMove(e, point) || {}; + if (bearingDelta) + map.setBearing(map.getBearing() + bearingDelta); + if (this.touchPitch) { + const { pitchDelta } = this.touchPitch.dragMove(e, point) || {}; + if (pitchDelta) + map.setPitch(map.getPitch() + pitchDelta); + } + } + off() { + const element = this.element; + DOM.removeEventListener(element, 'mousedown', this.mousedown); + DOM.removeEventListener(element, 'touchstart', this.touchstart, { passive: false }); + DOM.removeEventListener(window, 'touchmove', this.touchmove, { passive: false }); + DOM.removeEventListener(window, 'touchend', this.touchend); + DOM.removeEventListener(element, 'touchcancel', this.reset); + this.offTemp(); + } + offTemp() { + DOM.enableDrag(); + DOM.removeEventListener(window, 'mousemove', this.mousemove); + DOM.removeEventListener(window, 'mouseup', this.mouseup); + DOM.removeEventListener(window, 'touchmove', this.touchmove, { passive: false }); + DOM.removeEventListener(window, 'touchend', this.touchend); + } +} + +let supportsGeolocation; +function checkGeolocationSupport(callback, forceRecalculation = false) { + if (supportsGeolocation !== undefined && !forceRecalculation) { + callback(supportsGeolocation); + } + else if (window.navigator.permissions !== undefined) { + // navigator.permissions has incomplete browser support + // http://caniuse.com/#feat=permissions-api + // Test for the case where a browser disables Geolocation because of an + // insecure origin + window.navigator.permissions.query({ name: 'geolocation' }).then((p) => { + supportsGeolocation = p.state !== 'denied'; + callback(supportsGeolocation); + }).catch(() => { + // Fix for iOS16 which rejects query but still supports geolocation + supportsGeolocation = !!window.navigator.geolocation; + callback(supportsGeolocation); + }); + } + else { + supportsGeolocation = !!window.navigator.geolocation; + callback(supportsGeolocation); + } +} + +/** + * Given a LngLat, prior projected position, and a transform, return a new LngLat shifted + * n × 360° east or west for some n ≥ 0 such that: + * + * * the projected location of the result is on screen, if possible, and secondarily: + * * the difference between the projected location of the result and the prior position + * is minimized. + * + * The object is to preserve perceived object constancy for Popups and Markers as much as + * possible; they should avoid shifting large distances across the screen, even when the + * map center changes by ±360° due to automatic wrapping, and when about to go off screen, + * should wrap just enough to avoid doing so. + */ +function smartWrap(lngLat, priorPos, transform) { + lngLat = new performance.LngLat(lngLat.lng, lngLat.lat); + // First, try shifting one world in either direction, and see if either is closer to the + // prior position. This preserves object constancy when the map center is auto-wrapped + // during animations. + if (priorPos) { + const left = new performance.LngLat(lngLat.lng - 360, lngLat.lat); + const right = new performance.LngLat(lngLat.lng + 360, lngLat.lat); + const delta = transform.locationPoint(lngLat).distSqr(priorPos); + if (transform.locationPoint(left).distSqr(priorPos) < delta) { + lngLat = left; + } + else if (transform.locationPoint(right).distSqr(priorPos) < delta) { + lngLat = right; + } + } + // Second, wrap toward the center until the new position is on screen, or we can't get + // any closer. + while (Math.abs(lngLat.lng - transform.center.lng) > 180) { + const pos = transform.locationPoint(lngLat); + if (pos.x >= 0 && pos.y >= 0 && pos.x <= transform.width && pos.y <= transform.height) { + break; + } + if (lngLat.lng > transform.center.lng) { + lngLat.lng -= 360; + } + else { + lngLat.lng += 360; + } + } + return lngLat; +} + +const anchorTranslate = { + 'center': 'translate(-50%,-50%)', + 'top': 'translate(-50%,0)', + 'top-left': 'translate(0,0)', + 'top-right': 'translate(-100%,0)', + 'bottom': 'translate(-50%,-100%)', + 'bottom-left': 'translate(0,-100%)', + 'bottom-right': 'translate(-100%,-100%)', + 'left': 'translate(0,-50%)', + 'right': 'translate(-100%,-50%)' +}; +function applyAnchorClass(element, anchor, prefix) { + const classList = element.classList; + for (const key in anchorTranslate) { + classList.remove(`maplibregl-${prefix}-anchor-${key}`); + } + classList.add(`maplibregl-${prefix}-anchor-${anchor}`); +} + +/** + * Creates a marker component + * + * @group Markers and Controls + * + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([30.5, 50.5]) + * .addTo(map); + * ``` + * + * @example + * Set options + * ```ts + * let marker = new maplibregl.Marker({ + * color: "#FFFFFF", + * draggable: true + * }).setLngLat([30.5, 50.5]) + * .addTo(map); + * ``` + * @see [Add custom icons with Markers](https://maplibre.org/maplibre-gl-js/docs/examples/custom-marker-icons/) + * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/) + * + * ### Events + * + * @event `dragstart` Fired when dragging starts, `marker` object that is being dragged + * + * @event `drag` Fired while dragging. `marker` object that is being dragged + * + * @event `dragend` Fired when the marker is finished being dragged, `marker` object that was dragged + */ +class Marker extends performance.Evented { + /** + * @param options - the options + */ + constructor(options) { + super(); + this._onKeyPress = (e) => { + const code = e.code; + const legacyCode = e.charCode || e.keyCode; + if ((code === 'Space') || (code === 'Enter') || + (legacyCode === 32) || (legacyCode === 13) // space or enter + ) { + this.togglePopup(); + } + }; + this._onMapClick = (e) => { + const targetElement = e.originalEvent.target; + const element = this._element; + if (this._popup && (targetElement === element || element.contains(targetElement))) { + this.togglePopup(); + } + }; + this._update = (e) => { + if (!this._map) + return; + const isFullyLoaded = this._map.loaded() && !this._map.isMoving(); + if ((e === null || e === void 0 ? void 0 : e.type) === 'terrain' || ((e === null || e === void 0 ? void 0 : e.type) === 'render' && !isFullyLoaded)) { + this._map.once('render', this._update); + } + if (this._map.transform.renderWorldCopies) { + this._lngLat = smartWrap(this._lngLat, this._pos, this._map.transform); + } + this._pos = this._map.project(this._lngLat)._add(this._offset); + let rotation = ''; + if (this._rotationAlignment === 'viewport' || this._rotationAlignment === 'auto') { + rotation = `rotateZ(${this._rotation}deg)`; + } + else if (this._rotationAlignment === 'map') { + rotation = `rotateZ(${this._rotation - this._map.getBearing()}deg)`; + } + let pitch = ''; + if (this._pitchAlignment === 'viewport' || this._pitchAlignment === 'auto') { + pitch = 'rotateX(0deg)'; + } + else if (this._pitchAlignment === 'map') { + pitch = `rotateX(${this._map.getPitch()}deg)`; + } + // because rounding the coordinates at every `move` event causes stuttered zooming + // we only round them when _update is called with `moveend` or when its called with + // no arguments (when the Marker is initialized or Marker#setLngLat is invoked). + if (!e || e.type === 'moveend') { + this._pos = this._pos.round(); + } + DOM.setTransform(this._element, `${anchorTranslate[this._anchor]} translate(${this._pos.x}px, ${this._pos.y}px) ${pitch} ${rotation}`); + // in case of 3D, ask the terrain coords-framebuffer for this pos and check if the marker is visible + // call this logic in setTimeout with a timeout of 100ms to save performance in map-movement + if (this._map.terrain && !this._opacityTimeout) + this._opacityTimeout = setTimeout(() => { + const lnglat = this._map.unproject(this._pos); + const metresPerPixel = 40075016.686 * Math.abs(Math.cos(this._lngLat.lat * Math.PI / 180)) / Math.pow(2, this._map.transform.tileZoom + 8); + this._element.style.opacity = lnglat.distanceTo(this._lngLat) > metresPerPixel * 20 ? '0.2' : '1.0'; + this._opacityTimeout = null; + }, 100); + }; + this._onMove = (e) => { + if (!this._isDragging) { + const clickTolerance = this._clickTolerance || this._map._clickTolerance; + this._isDragging = e.point.dist(this._pointerdownPos) >= clickTolerance; + } + if (!this._isDragging) + return; + this._pos = e.point.sub(this._positionDelta); + this._lngLat = this._map.unproject(this._pos); + this.setLngLat(this._lngLat); + // suppress click event so that popups don't toggle on drag + this._element.style.pointerEvents = 'none'; + // make sure dragstart only fires on the first move event after mousedown. + // this can't be on mousedown because that event doesn't necessarily + // imply that a drag is about to happen. + if (this._state === 'pending') { + this._state = 'active'; + this.fire(new performance.Event('dragstart')); + } + this.fire(new performance.Event('drag')); + }; + this._onUp = () => { + // revert to normal pointer event handling + this._element.style.pointerEvents = 'auto'; + this._positionDelta = null; + this._pointerdownPos = null; + this._isDragging = false; + this._map.off('mousemove', this._onMove); + this._map.off('touchmove', this._onMove); + // only fire dragend if it was preceded by at least one drag event + if (this._state === 'active') { + this.fire(new performance.Event('dragend')); + } + this._state = 'inactive'; + }; + this._addDragHandler = (e) => { + if (this._element.contains(e.originalEvent.target)) { + e.preventDefault(); + // We need to calculate the pixel distance between the click point + // and the marker position, with the offset accounted for. Then we + // can subtract this distance from the mousemove event's position + // to calculate the new marker position. + // If we don't do this, the marker 'jumps' to the click position + // creating a jarring UX effect. + this._positionDelta = e.point.sub(this._pos).add(this._offset); + this._pointerdownPos = e.point; + this._state = 'pending'; + this._map.on('mousemove', this._onMove); + this._map.on('touchmove', this._onMove); + this._map.once('mouseup', this._onUp); + this._map.once('touchend', this._onUp); + } + }; + this._anchor = options && options.anchor || 'center'; + this._color = options && options.color || '#3FB1CE'; + this._scale = options && options.scale || 1; + this._draggable = options && options.draggable || false; + this._clickTolerance = options && options.clickTolerance || 0; + this._isDragging = false; + this._state = 'inactive'; + this._rotation = options && options.rotation || 0; + this._rotationAlignment = options && options.rotationAlignment || 'auto'; + this._pitchAlignment = options && options.pitchAlignment && options.pitchAlignment !== 'auto' ? options.pitchAlignment : this._rotationAlignment; + if (!options || !options.element) { + this._defaultMarker = true; + this._element = DOM.create('div'); + this._element.setAttribute('aria-label', 'Map marker'); + // create default map marker SVG + const svg = DOM.createNS('http://www.w3.org/2000/svg', 'svg'); + const defaultHeight = 41; + const defaultWidth = 27; + svg.setAttributeNS(null, 'display', 'block'); + svg.setAttributeNS(null, 'height', `${defaultHeight}px`); + svg.setAttributeNS(null, 'width', `${defaultWidth}px`); + svg.setAttributeNS(null, 'viewBox', `0 0 ${defaultWidth} ${defaultHeight}`); + const markerLarge = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + markerLarge.setAttributeNS(null, 'stroke', 'none'); + markerLarge.setAttributeNS(null, 'stroke-width', '1'); + markerLarge.setAttributeNS(null, 'fill', 'none'); + markerLarge.setAttributeNS(null, 'fill-rule', 'evenodd'); + const page1 = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + page1.setAttributeNS(null, 'fill-rule', 'nonzero'); + const shadow = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + shadow.setAttributeNS(null, 'transform', 'translate(3.0, 29.0)'); + shadow.setAttributeNS(null, 'fill', '#000000'); + const ellipses = [ + { 'rx': '10.5', 'ry': '5.25002273' }, + { 'rx': '10.5', 'ry': '5.25002273' }, + { 'rx': '9.5', 'ry': '4.77275007' }, + { 'rx': '8.5', 'ry': '4.29549936' }, + { 'rx': '7.5', 'ry': '3.81822308' }, + { 'rx': '6.5', 'ry': '3.34094679' }, + { 'rx': '5.5', 'ry': '2.86367051' }, + { 'rx': '4.5', 'ry': '2.38636864' } + ]; + for (const data of ellipses) { + const ellipse = DOM.createNS('http://www.w3.org/2000/svg', 'ellipse'); + ellipse.setAttributeNS(null, 'opacity', '0.04'); + ellipse.setAttributeNS(null, 'cx', '10.5'); + ellipse.setAttributeNS(null, 'cy', '5.80029008'); + ellipse.setAttributeNS(null, 'rx', data['rx']); + ellipse.setAttributeNS(null, 'ry', data['ry']); + shadow.appendChild(ellipse); + } + const background = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + background.setAttributeNS(null, 'fill', this._color); + const bgPath = DOM.createNS('http://www.w3.org/2000/svg', 'path'); + bgPath.setAttributeNS(null, 'd', 'M27,13.5 C27,19.074644 20.250001,27.000002 14.75,34.500002 C14.016665,35.500004 12.983335,35.500004 12.25,34.500002 C6.7499993,27.000002 0,19.222562 0,13.5 C0,6.0441559 6.0441559,0 13.5,0 C20.955844,0 27,6.0441559 27,13.5 Z'); + background.appendChild(bgPath); + const border = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + border.setAttributeNS(null, 'opacity', '0.25'); + border.setAttributeNS(null, 'fill', '#000000'); + const borderPath = DOM.createNS('http://www.w3.org/2000/svg', 'path'); + borderPath.setAttributeNS(null, 'd', 'M13.5,0 C6.0441559,0 0,6.0441559 0,13.5 C0,19.222562 6.7499993,27 12.25,34.5 C13,35.522727 14.016664,35.500004 14.75,34.5 C20.250001,27 27,19.074644 27,13.5 C27,6.0441559 20.955844,0 13.5,0 Z M13.5,1 C20.415404,1 26,6.584596 26,13.5 C26,15.898657 24.495584,19.181431 22.220703,22.738281 C19.945823,26.295132 16.705119,30.142167 13.943359,33.908203 C13.743445,34.180814 13.612715,34.322738 13.5,34.441406 C13.387285,34.322738 13.256555,34.180814 13.056641,33.908203 C10.284481,30.127985 7.4148684,26.314159 5.015625,22.773438 C2.6163816,19.232715 1,15.953538 1,13.5 C1,6.584596 6.584596,1 13.5,1 Z'); + border.appendChild(borderPath); + const maki = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + maki.setAttributeNS(null, 'transform', 'translate(6.0, 7.0)'); + maki.setAttributeNS(null, 'fill', '#FFFFFF'); + const circleContainer = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + circleContainer.setAttributeNS(null, 'transform', 'translate(8.0, 8.0)'); + const circle1 = DOM.createNS('http://www.w3.org/2000/svg', 'circle'); + circle1.setAttributeNS(null, 'fill', '#000000'); + circle1.setAttributeNS(null, 'opacity', '0.25'); + circle1.setAttributeNS(null, 'cx', '5.5'); + circle1.setAttributeNS(null, 'cy', '5.5'); + circle1.setAttributeNS(null, 'r', '5.4999962'); + const circle2 = DOM.createNS('http://www.w3.org/2000/svg', 'circle'); + circle2.setAttributeNS(null, 'fill', '#FFFFFF'); + circle2.setAttributeNS(null, 'cx', '5.5'); + circle2.setAttributeNS(null, 'cy', '5.5'); + circle2.setAttributeNS(null, 'r', '5.4999962'); + circleContainer.appendChild(circle1); + circleContainer.appendChild(circle2); + page1.appendChild(shadow); + page1.appendChild(background); + page1.appendChild(border); + page1.appendChild(maki); + page1.appendChild(circleContainer); + svg.appendChild(page1); + svg.setAttributeNS(null, 'height', `${defaultHeight * this._scale}px`); + svg.setAttributeNS(null, 'width', `${defaultWidth * this._scale}px`); + this._element.appendChild(svg); + // if no element and no offset option given apply an offset for the default marker + // the -14 as the y value of the default marker offset was determined as follows + // + // the marker tip is at the center of the shadow ellipse from the default svg + // the y value of the center of the shadow ellipse relative to the svg top left is "shadow transform translate-y (29.0) + ellipse cy (5.80029008)" + // offset to the svg center "height (41 / 2)" gives (29.0 + 5.80029008) - (41 / 2) and rounded for an integer pixel offset gives 14 + // negative is used to move the marker up from the center so the tip is at the Marker lngLat + this._offset = performance.Point.convert(options && options.offset || [0, -14]); + } + else { + this._element = options.element; + this._offset = performance.Point.convert(options && options.offset || [0, 0]); + } + this._element.classList.add('maplibregl-marker'); + this._element.addEventListener('dragstart', (e) => { + e.preventDefault(); + }); + this._element.addEventListener('mousedown', (e) => { + // prevent focusing on click + e.preventDefault(); + }); + applyAnchorClass(this._element, this._anchor, 'marker'); + if (options && options.className) { + for (const name of options.className.split(' ')) { + this._element.classList.add(name); + } + } + this._popup = null; + } + /** + * Attaches the `Marker` to a `Map` object. + * @param map - The MapLibre GL JS map to add the marker to. + * @returns `this` + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([30.5, 50.5]) + * .addTo(map); // add the marker to the map + * ``` + */ + addTo(map) { + this.remove(); + this._map = map; + map.getCanvasContainer().appendChild(this._element); + map.on('move', this._update); + map.on('moveend', this._update); + map.on('terrain', this._update); + this.setDraggable(this._draggable); + this._update(); + // If we attached the `click` listener to the marker element, the popup + // would close once the event propagated to `map` due to the + // `Popup#_onClickClose` listener. + this._map.on('click', this._onMapClick); + return this; + } + /** + * Removes the marker from a map + * @example + * ```ts + * let marker = new maplibregl.Marker().addTo(map); + * marker.remove(); + * ``` + * @returns `this` + */ + remove() { + if (this._opacityTimeout) { + clearTimeout(this._opacityTimeout); + delete this._opacityTimeout; + } + if (this._map) { + this._map.off('click', this._onMapClick); + this._map.off('move', this._update); + this._map.off('moveend', this._update); + this._map.off('mousedown', this._addDragHandler); + this._map.off('touchstart', this._addDragHandler); + this._map.off('mouseup', this._onUp); + this._map.off('touchend', this._onUp); + this._map.off('mousemove', this._onMove); + this._map.off('touchmove', this._onMove); + delete this._map; + } + DOM.remove(this._element); + if (this._popup) + this._popup.remove(); + return this; + } + /** + * Get the marker's geographical location. + * + * The longitude of the result may differ by a multiple of 360 degrees from the longitude previously + * set by `setLngLat` because `Marker` wraps the anchor longitude across copies of the world to keep + * the marker on screen. + * + * @returns A {@link LngLat} describing the marker's location. + * @example + * ```ts + * // Store the marker's longitude and latitude coordinates in a variable + * let lngLat = marker.getLngLat(); + * // Print the marker's longitude and latitude values in the console + * console.log('Longitude: ' + lngLat.lng + ', Latitude: ' + lngLat.lat ) + * ``` + * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/) + */ + getLngLat() { + return this._lngLat; + } + /** + * Set the marker's geographical position and move it. + * @param lnglat - A {@link LngLat} describing where the marker should be located. + * @returns `this` + * @example + * Create a new marker, set the longitude and latitude, and add it to the map + * ```ts + * new maplibregl.Marker() + * .setLngLat([-65.017, -16.457]) + * .addTo(map); + * ``` + * @see [Add custom icons with Markers](https://maplibre.org/maplibre-gl-js/docs/examples/custom-marker-icons/) + * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/) + */ + setLngLat(lnglat) { + this._lngLat = performance.LngLat.convert(lnglat); + this._pos = null; + if (this._popup) + this._popup.setLngLat(this._lngLat); + this._update(); + return this; + } + /** + * Returns the `Marker`'s HTML element. + * @returns element + */ + getElement() { + return this._element; + } + /** + * Binds a {@link Popup} to the {@link Marker}. + * @param popup - An instance of the {@link Popup} class. If undefined or null, any popup + * set on this {@link Marker} instance is unset. + * @returns `this` + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([0, 0]) + * .setPopup(new maplibregl.Popup().setHTML("

Hello World!

")) // add popup + * .addTo(map); + * ``` + * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/) + */ + setPopup(popup) { + if (this._popup) { + this._popup.remove(); + this._popup = null; + this._element.removeEventListener('keypress', this._onKeyPress); + if (!this._originalTabIndex) { + this._element.removeAttribute('tabindex'); + } + } + if (popup) { + if (!('offset' in popup.options)) { + const markerHeight = 41 - (5.8 / 2); + const markerRadius = 13.5; + const linearOffset = Math.abs(markerRadius) / Math.SQRT2; + popup.options.offset = this._defaultMarker ? { + 'top': [0, 0], + 'top-left': [0, 0], + 'top-right': [0, 0], + 'bottom': [0, -markerHeight], + 'bottom-left': [linearOffset, (markerHeight - markerRadius + linearOffset) * -1], + 'bottom-right': [-linearOffset, (markerHeight - markerRadius + linearOffset) * -1], + 'left': [markerRadius, (markerHeight - markerRadius) * -1], + 'right': [-markerRadius, (markerHeight - markerRadius) * -1] + } : this._offset; + } + this._popup = popup; + if (this._lngLat) + this._popup.setLngLat(this._lngLat); + this._originalTabIndex = this._element.getAttribute('tabindex'); + if (!this._originalTabIndex) { + this._element.setAttribute('tabindex', '0'); + } + this._element.addEventListener('keypress', this._onKeyPress); + } + return this; + } + /** + * Returns the {@link Popup} instance that is bound to the {@link Marker}. + * @returns popup + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([0, 0]) + * .setPopup(new maplibregl.Popup().setHTML("

Hello World!

")) + * .addTo(map); + * + * console.log(marker.getPopup()); // return the popup instance + * ``` + */ + getPopup() { + return this._popup; + } + /** + * Opens or closes the {@link Popup} instance that is bound to the {@link Marker}, depending on the current state of the {@link Popup}. + * @returns `this` + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([0, 0]) + * .setPopup(new maplibregl.Popup().setHTML("

Hello World!

")) + * .addTo(map); + * + * marker.togglePopup(); // toggle popup open or closed + * ``` + */ + togglePopup() { + const popup = this._popup; + if (!popup) + return this; + else if (popup.isOpen()) + popup.remove(); + else + popup.addTo(this._map); + return this; + } + /** + * Get the marker's offset. + * @returns The marker's screen coordinates in pixels. + */ + getOffset() { + return this._offset; + } + /** + * Sets the offset of the marker + * @param offset - The offset in pixels as a {@link PointLike} object to apply relative to the element's center. Negatives indicate left and up. + * @returns `this` + */ + setOffset(offset) { + this._offset = performance.Point.convert(offset); + this._update(); + return this; + } + /** + * Adds a CSS class to the marker element. + * + * @param className - on-empty string with CSS class name to add to marker element + * + * @example + * ``` + * let marker = new maplibregl.Marker() + * marker.addClassName('some-class') + * ``` + */ + addClassName(className) { + this._element.classList.add(className); + } + /** + * Removes a CSS class from the marker element. + * + * @param className - Non-empty string with CSS class name to remove from marker element + * + * @example + * ```ts + * let marker = new maplibregl.Marker() + * marker.removeClassName('some-class') + * ``` + */ + removeClassName(className) { + this._element.classList.remove(className); + } + /** + * Add or remove the given CSS class on the marker element, depending on whether the element currently has that class. + * + * @param className - Non-empty string with CSS class name to add/remove + * + * @returns if the class was removed return false, if class was added, then return true + * + * @example + * ```ts + * let marker = new maplibregl.Marker() + * marker.toggleClassName('toggleClass') + * ``` + */ + toggleClassName(className) { + return this._element.classList.toggle(className); + } + /** + * Sets the `draggable` property and functionality of the marker + * @param shouldBeDraggable - Turns drag functionality on/off + * @returns `this` + */ + setDraggable(shouldBeDraggable) { + this._draggable = !!shouldBeDraggable; // convert possible undefined value to false + // handle case where map may not exist yet + // e.g. when setDraggable is called before addTo + if (this._map) { + if (shouldBeDraggable) { + this._map.on('mousedown', this._addDragHandler); + this._map.on('touchstart', this._addDragHandler); + } + else { + this._map.off('mousedown', this._addDragHandler); + this._map.off('touchstart', this._addDragHandler); + } + } + return this; + } + /** + * Returns true if the marker can be dragged + * @returns True if the marker is draggable. + */ + isDraggable() { + return this._draggable; + } + /** + * Sets the `rotation` property of the marker. + * @param rotation - The rotation angle of the marker (clockwise, in degrees), relative to its respective {@link Marker#setRotationAlignment} setting. + * @returns `this` + */ + setRotation(rotation) { + this._rotation = rotation || 0; + this._update(); + return this; + } + /** + * Returns the current rotation angle of the marker (in degrees). + * @returns The current rotation angle of the marker. + */ + getRotation() { + return this._rotation; + } + /** + * Sets the `rotationAlignment` property of the marker. + * @param alignment - Sets the `rotationAlignment` property of the marker. defaults to 'auto' + * @returns `this` + */ + setRotationAlignment(alignment) { + this._rotationAlignment = alignment || 'auto'; + this._update(); + return this; + } + /** + * Returns the current `rotationAlignment` property of the marker. + * @returns The current rotational alignment of the marker. + */ + getRotationAlignment() { + return this._rotationAlignment; + } + /** + * Sets the `pitchAlignment` property of the marker. + * @param alignment - Sets the `pitchAlignment` property of the marker. If alignment is 'auto', it will automatically match `rotationAlignment`. + * @returns `this` + */ + setPitchAlignment(alignment) { + this._pitchAlignment = alignment && alignment !== 'auto' ? alignment : this._rotationAlignment; + this._update(); + return this; + } + /** + * Returns the current `pitchAlignment` property of the marker. + * @returns The current pitch alignment of the marker in degrees. + */ + getPitchAlignment() { + return this._pitchAlignment; + } +} + +const defaultOptions$2 = { + positionOptions: { + enableHighAccuracy: false, + maximumAge: 0, + timeout: 6000 /* 6 sec */ + }, + fitBoundsOptions: { + maxZoom: 15 + }, + trackUserLocation: false, + showAccuracyCircle: true, + showUserLocation: true +}; +let numberOfWatches = 0; +let noTimeout = false; +/** + * A `GeolocateControl` control provides a button that uses the browser's geolocation + * API to locate the user on the map. + * + * Not all browsers support geolocation, + * and some users may disable the feature. Geolocation support for modern + * browsers including Chrome requires sites to be served over HTTPS. If + * geolocation support is not available, the GeolocateControl will show + * as disabled. + * + * The zoom level applied will depend on the accuracy of the geolocation provided by the device. + * + * The GeolocateControl has two modes. If `trackUserLocation` is `false` (default) the control acts as a button, which when pressed will set the map's camera to target the user location. If the user moves, the map won't update. This is most suited for the desktop. If `trackUserLocation` is `true` the control acts as a toggle button that when active the user's location is actively monitored for changes. In this mode the GeolocateControl has three interaction states: + * * active - the map's camera automatically updates as the user's location changes, keeping the location dot in the center. Initial state and upon clicking the `GeolocateControl` button. + * * passive - the user's location dot automatically updates, but the map's camera does not. Occurs upon the user initiating a map movement. + * * disabled - occurs if Geolocation is not available, disabled or denied. + * + * These interaction states can't be controlled programmatically, rather they are set based on user interactions. + * @group Markers and Controls + * + * @example + * ```ts + * map.addControl(new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * })); + * ``` + * @see [Locate the user](https://maplibre.org/maplibre-gl-js/docs/examples/locate-user/) + * + * ### Events + * + * @event `trackuserlocationend` - Fired when the Geolocate Control changes to the background state, which happens when a user changes the camera during an active position lock. This only applies when trackUserLocation is true. In the background state, the dot on the map will update with location updates but the camera will not. + * + * @event `trackuserlocationstart` - Fired when the Geolocate Control changes to the active lock state, which happens either upon first obtaining a successful Geolocation API position for the user (a geolocate event will follow), or the user clicks the geolocate button when in the background state which uses the last known position to recenter the map and enter active lock state (no geolocate event will follow unless the users's location changes). + * + * @event `geolocate` - Fired on each Geolocation API position update which returned as success. + * `data` - The returned [Position](https://developer.mozilla.org/en-US/docs/Web/API/Position) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition). + * + * @event `error` - Fired on each Geolocation API position update which returned as an error. + * `data` - The returned [PositionError](https://developer.mozilla.org/en-US/docs/Web/API/PositionError) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition). + * + * @event `outofmaxbounds` Fired on each Geolocation API position update which returned as success but user position is out of map maxBounds. + * `data` - The returned [Position](https://developer.mozilla.org/en-US/docs/Web/API/Position) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition). + * + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * // Set an event listener that fires + * // when a trackuserlocationend event occurs. + * geolocate.on('trackuserlocationend', function() { + * console.log('A trackuserlocationend event has occurred.') + * }); + * ``` + * + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * // Set an event listener that fires + * // when a trackuserlocationstart event occurs. + * geolocate.on('trackuserlocationstart', function() { + * console.log('A trackuserlocationstart event has occurred.') + * }); + * ``` + * + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * // Set an event listener that fires + * // when a geolocate event occurs. + * geolocate.on('geolocate', function() { + * console.log('A geolocate event has occurred.') + * }); + * ``` + * + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * // Set an event listener that fires + * // when an error event occurs. + * geolocate.on('error', function() { + * console.log('An error event has occurred.') + * }); + * ``` + * + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * // Set an event listener that fires + * // when an outofmaxbounds event occurs. + * geolocate.on('outofmaxbounds', function() { + * console.log('An outofmaxbounds event has occurred.') + * }); + * ``` + */ +class GeolocateControl extends performance.Evented { + constructor(options) { + super(); + /** + * When the Geolocation API returns a new location, update the GeolocateControl. + * + * @param position - the Geolocation API Position + */ + this._onSuccess = (position) => { + if (!this._map) { + // control has since been removed + return; + } + if (this._isOutOfMapMaxBounds(position)) { + this._setErrorState(); + this.fire(new performance.Event('outofmaxbounds', position)); + this._updateMarker(); + this._finish(); + return; + } + if (this.options.trackUserLocation) { + // keep a record of the position so that if the state is BACKGROUND and the user + // clicks the button, we can move to ACTIVE_LOCK immediately without waiting for + // watchPosition to trigger _onSuccess + this._lastKnownPosition = position; + switch (this._watchState) { + case 'WAITING_ACTIVE': + case 'ACTIVE_LOCK': + case 'ACTIVE_ERROR': + this._watchState = 'ACTIVE_LOCK'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active'); + break; + case 'BACKGROUND': + case 'BACKGROUND_ERROR': + this._watchState = 'BACKGROUND'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background'); + break; + default: + throw new Error(`Unexpected watchState ${this._watchState}`); + } + } + // if showUserLocation and the watch state isn't off then update the marker location + if (this.options.showUserLocation && this._watchState !== 'OFF') { + this._updateMarker(position); + } + // if in normal mode (not watch mode), or if in watch mode and the state is active watch + // then update the camera + if (!this.options.trackUserLocation || this._watchState === 'ACTIVE_LOCK') { + this._updateCamera(position); + } + if (this.options.showUserLocation) { + this._dotElement.classList.remove('maplibregl-user-location-dot-stale'); + } + this.fire(new performance.Event('geolocate', position)); + this._finish(); + }; + /** + * Update the camera location to center on the current position + * + * @param position - the Geolocation API Position + */ + this._updateCamera = (position) => { + const center = new performance.LngLat(position.coords.longitude, position.coords.latitude); + const radius = position.coords.accuracy; + const bearing = this._map.getBearing(); + const options = performance.extend({ bearing }, this.options.fitBoundsOptions); + const newBounds = LngLatBounds.fromLngLat(center, radius); + this._map.fitBounds(newBounds, options, { + geolocateSource: true // tag this camera change so it won't cause the control to change to background state + }); + }; + /** + * Update the user location dot Marker to the current position + * + * @param position - the Geolocation API Position + */ + this._updateMarker = (position) => { + if (position) { + const center = new performance.LngLat(position.coords.longitude, position.coords.latitude); + this._accuracyCircleMarker.setLngLat(center).addTo(this._map); + this._userLocationDotMarker.setLngLat(center).addTo(this._map); + this._accuracy = position.coords.accuracy; + if (this.options.showUserLocation && this.options.showAccuracyCircle) { + this._updateCircleRadius(); + } + } + else { + this._userLocationDotMarker.remove(); + this._accuracyCircleMarker.remove(); + } + }; + this._onZoom = () => { + if (this.options.showUserLocation && this.options.showAccuracyCircle) { + this._updateCircleRadius(); + } + }; + this._onError = (error) => { + if (!this._map) { + // control has since been removed + return; + } + if (this.options.trackUserLocation) { + if (error.code === 1) { + // PERMISSION_DENIED + this._watchState = 'OFF'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error'); + this._geolocateButton.disabled = true; + const title = this._map._getUIString('GeolocateControl.LocationNotAvailable'); + this._geolocateButton.title = title; + this._geolocateButton.setAttribute('aria-label', title); + if (this._geolocationWatchID !== undefined) { + this._clearWatch(); + } + } + else if (error.code === 3 && noTimeout) { + // this represents a forced error state + // this was triggered to force immediate geolocation when a watch is already present + // see https://github.com/mapbox/mapbox-gl-js/issues/8214 + // and https://w3c.github.io/geolocation-api/#example-5-forcing-the-user-agent-to-return-a-fresh-cached-position + return; + } + else { + this._setErrorState(); + } + } + if (this._watchState !== 'OFF' && this.options.showUserLocation) { + this._dotElement.classList.add('maplibregl-user-location-dot-stale'); + } + this.fire(new performance.Event('error', error)); + this._finish(); + }; + this._finish = () => { + if (this._timeoutId) { + clearTimeout(this._timeoutId); + } + this._timeoutId = undefined; + }; + this._setupUI = (supported) => { + // this method is called asynchronously during onAdd + // the control could have been removed before reaching here + if (!this._map) { + return; + } + this._container.addEventListener('contextmenu', (e) => e.preventDefault()); + this._geolocateButton = DOM.create('button', 'maplibregl-ctrl-geolocate', this._container); + DOM.create('span', 'maplibregl-ctrl-icon', this._geolocateButton).setAttribute('aria-hidden', 'true'); + this._geolocateButton.type = 'button'; + if (supported === false) { + performance.warnOnce('Geolocation support is not available so the GeolocateControl will be disabled.'); + const title = this._map._getUIString('GeolocateControl.LocationNotAvailable'); + this._geolocateButton.disabled = true; + this._geolocateButton.title = title; + this._geolocateButton.setAttribute('aria-label', title); + } + else { + const title = this._map._getUIString('GeolocateControl.FindMyLocation'); + this._geolocateButton.title = title; + this._geolocateButton.setAttribute('aria-label', title); + } + if (this.options.trackUserLocation) { + this._geolocateButton.setAttribute('aria-pressed', 'false'); + this._watchState = 'OFF'; + } + // when showUserLocation is enabled, keep the Geolocate button disabled until the device location marker is setup on the map + if (this.options.showUserLocation) { + this._dotElement = DOM.create('div', 'maplibregl-user-location-dot'); + this._userLocationDotMarker = new Marker({ element: this._dotElement }); + this._circleElement = DOM.create('div', 'maplibregl-user-location-accuracy-circle'); + this._accuracyCircleMarker = new Marker({ element: this._circleElement, pitchAlignment: 'map' }); + if (this.options.trackUserLocation) + this._watchState = 'OFF'; + this._map.on('zoom', this._onZoom); + } + this._geolocateButton.addEventListener('click', this.trigger.bind(this)); + this._setup = true; + // when the camera is changed (and it's not as a result of the Geolocation Control) change + // the watch mode to background watch, so that the marker is updated but not the camera. + if (this.options.trackUserLocation) { + this._map.on('movestart', (event) => { + const fromResize = event.originalEvent && event.originalEvent.type === 'resize'; + if (!event.geolocateSource && this._watchState === 'ACTIVE_LOCK' && !fromResize) { + this._watchState = 'BACKGROUND'; + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active'); + this.fire(new performance.Event('trackuserlocationend')); + } + }); + } + }; + this.options = performance.extend({}, defaultOptions$2, options); + } + /** {@inheritDoc IControl.onAdd} */ + onAdd(map) { + this._map = map; + this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group'); + checkGeolocationSupport(this._setupUI); + return this._container; + } + /** {@inheritDoc IControl.onRemove} */ + onRemove() { + // clear the geolocation watch if exists + if (this._geolocationWatchID !== undefined) { + window.navigator.geolocation.clearWatch(this._geolocationWatchID); + this._geolocationWatchID = undefined; + } + // clear the markers from the map + if (this.options.showUserLocation && this._userLocationDotMarker) { + this._userLocationDotMarker.remove(); + } + if (this.options.showAccuracyCircle && this._accuracyCircleMarker) { + this._accuracyCircleMarker.remove(); + } + DOM.remove(this._container); + this._map.off('zoom', this._onZoom); + this._map = undefined; + numberOfWatches = 0; + noTimeout = false; + } + /** + * Check if the Geolocation API Position is outside the map's maxbounds. + * + * @param position - the Geolocation API Position + * @returns `true` if position is outside the map's maxbounds, otherwise returns `false`. + */ + _isOutOfMapMaxBounds(position) { + const bounds = this._map.getMaxBounds(); + const coordinates = position.coords; + return bounds && (coordinates.longitude < bounds.getWest() || + coordinates.longitude > bounds.getEast() || + coordinates.latitude < bounds.getSouth() || + coordinates.latitude > bounds.getNorth()); + } + _setErrorState() { + switch (this._watchState) { + case 'WAITING_ACTIVE': + this._watchState = 'ACTIVE_ERROR'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active-error'); + break; + case 'ACTIVE_LOCK': + this._watchState = 'ACTIVE_ERROR'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active-error'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting'); + // turn marker grey + break; + case 'BACKGROUND': + this._watchState = 'BACKGROUND_ERROR'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background-error'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting'); + // turn marker grey + break; + case 'ACTIVE_ERROR': + break; + default: + throw new Error(`Unexpected watchState ${this._watchState}`); + } + } + _updateCircleRadius() { + const bounds = this._map.getBounds(); + const southEastPoint = bounds.getSouthEast(); + const northEastPoint = bounds.getNorthEast(); + const mapHeightInMeters = southEastPoint.distanceTo(northEastPoint); + const mapHeightInPixels = this._map._container.clientHeight; + const circleDiameter = Math.ceil(2 * (this._accuracy / (mapHeightInMeters / mapHeightInPixels))); + this._circleElement.style.width = `${circleDiameter}px`; + this._circleElement.style.height = `${circleDiameter}px`; + } + /** + * Programmatically request and move the map to the user's location. + * + * @returns `false` if called before control was added to a map, otherwise returns `true`. + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * map.on('load', function() { + * geolocate.trigger(); + * }); + * ``` + */ + trigger() { + if (!this._setup) { + performance.warnOnce('Geolocate control triggered before added to a map'); + return false; + } + if (this.options.trackUserLocation) { + // update watchState and do any outgoing state cleanup + switch (this._watchState) { + case 'OFF': + // turn on the Geolocate Control + this._watchState = 'WAITING_ACTIVE'; + this.fire(new performance.Event('trackuserlocationstart')); + break; + case 'WAITING_ACTIVE': + case 'ACTIVE_LOCK': + case 'ACTIVE_ERROR': + case 'BACKGROUND_ERROR': + // turn off the Geolocate Control + numberOfWatches--; + noTimeout = false; + this._watchState = 'OFF'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error'); + this.fire(new performance.Event('trackuserlocationend')); + break; + case 'BACKGROUND': + this._watchState = 'ACTIVE_LOCK'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background'); + // set camera to last known location + if (this._lastKnownPosition) + this._updateCamera(this._lastKnownPosition); + this.fire(new performance.Event('trackuserlocationstart')); + break; + default: + throw new Error(`Unexpected watchState ${this._watchState}`); + } + // incoming state setup + switch (this._watchState) { + case 'WAITING_ACTIVE': + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active'); + break; + case 'ACTIVE_LOCK': + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active'); + break; + case 'OFF': + break; + default: + throw new Error(`Unexpected watchState ${this._watchState}`); + } + // manage geolocation.watchPosition / geolocation.clearWatch + if (this._watchState === 'OFF' && this._geolocationWatchID !== undefined) { + // clear watchPosition as we've changed to an OFF state + this._clearWatch(); + } + else if (this._geolocationWatchID === undefined) { + // enable watchPosition since watchState is not OFF and there is no watchPosition already running + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.setAttribute('aria-pressed', 'true'); + numberOfWatches++; + let positionOptions; + if (numberOfWatches > 1) { + positionOptions = { maximumAge: 600000, timeout: 0 }; + noTimeout = true; + } + else { + positionOptions = this.options.positionOptions; + noTimeout = false; + } + this._geolocationWatchID = window.navigator.geolocation.watchPosition(this._onSuccess, this._onError, positionOptions); + } + } + else { + window.navigator.geolocation.getCurrentPosition(this._onSuccess, this._onError, this.options.positionOptions); + // This timeout ensures that we still call finish() even if + // the user declines to share their location in Firefox + this._timeoutId = setTimeout(this._finish, 10000 /* 10sec */); + } + return true; + } + _clearWatch() { + window.navigator.geolocation.clearWatch(this._geolocationWatchID); + this._geolocationWatchID = undefined; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.setAttribute('aria-pressed', 'false'); + if (this.options.showUserLocation) { + this._updateMarker(null); + } + } +} + +const defaultOptions$1 = { + maxWidth: 100, + unit: 'metric' +}; +/** + * A `ScaleControl` control displays the ratio of a distance on the map to the corresponding distance on the ground. + * + * @group Markers and Controls + * + * @example + * ```ts + * let scale = new maplibregl.ScaleControl({ + * maxWidth: 80, + * unit: 'imperial' + * }); + * map.addControl(scale); + * + * scale.setUnit('metric'); + * ``` + */ +class ScaleControl { + constructor(options) { + this._onMove = () => { + updateScale(this._map, this._container, this.options); + }; + /** + * Set the scale's unit of the distance + * + * @param unit - Unit of the distance (`'imperial'`, `'metric'` or `'nautical'`). + */ + this.setUnit = (unit) => { + this.options.unit = unit; + updateScale(this._map, this._container, this.options); + }; + this.options = performance.extend({}, defaultOptions$1, options); + } + getDefaultPosition() { + return 'bottom-left'; + } + /** {@inheritDoc IControl.onAdd} */ + onAdd(map) { + this._map = map; + this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-scale', map.getContainer()); + this._map.on('move', this._onMove); + this._onMove(); + return this._container; + } + /** {@inheritDoc IControl.onRemove} */ + onRemove() { + DOM.remove(this._container); + this._map.off('move', this._onMove); + this._map = undefined; + } +} +function updateScale(map, container, options) { + // A horizontal scale is imagined to be present at center of the map + // container with maximum length (Default) as 100px. + // Using spherical law of cosines approximation, the real distance is + // found between the two coordinates. + const maxWidth = options && options.maxWidth || 100; + const y = map._container.clientHeight / 2; + const left = map.unproject([0, y]); + const right = map.unproject([maxWidth, y]); + const maxMeters = left.distanceTo(right); + // The real distance corresponding to 100px scale length is rounded off to + // near pretty number and the scale length for the same is found out. + // Default unit of the scale is based on User's locale. + if (options && options.unit === 'imperial') { + const maxFeet = 3.2808 * maxMeters; + if (maxFeet > 5280) { + const maxMiles = maxFeet / 5280; + setScale(container, maxWidth, maxMiles, map._getUIString('ScaleControl.Miles')); + } + else { + setScale(container, maxWidth, maxFeet, map._getUIString('ScaleControl.Feet')); + } + } + else if (options && options.unit === 'nautical') { + const maxNauticals = maxMeters / 1852; + setScale(container, maxWidth, maxNauticals, map._getUIString('ScaleControl.NauticalMiles')); + } + else if (maxMeters >= 1000) { + setScale(container, maxWidth, maxMeters / 1000, map._getUIString('ScaleControl.Kilometers')); + } + else { + setScale(container, maxWidth, maxMeters, map._getUIString('ScaleControl.Meters')); + } +} +function setScale(container, maxWidth, maxDistance, unit) { + const distance = getRoundNum(maxDistance); + const ratio = distance / maxDistance; + container.style.width = `${maxWidth * ratio}px`; + container.innerHTML = `${distance} ${unit}`; +} +function getDecimalRoundNum(d) { + const multiplier = Math.pow(10, Math.ceil(-Math.log(d) / Math.LN10)); + return Math.round(d * multiplier) / multiplier; +} +function getRoundNum(num) { + const pow10 = Math.pow(10, (`${Math.floor(num)}`).length - 1); + let d = num / pow10; + d = d >= 10 ? 10 : + d >= 5 ? 5 : + d >= 3 ? 3 : + d >= 2 ? 2 : + d >= 1 ? 1 : getDecimalRoundNum(d); + return pow10 * d; +} + +/** + * A `FullscreenControl` control contains a button for toggling the map in and out of fullscreen mode. + * When [requestFullscreen](https://developer.mozilla.org/en-US/docs/Web/API/Element/requestFullscreen) is not supported, fullscreen is handled via CSS properties. + * The map's `cooperativeGestures` option is temporarily disabled while the map + * is in fullscreen mode, and is restored when the map exist fullscreen mode. + * + * @group Markers and Controls + * @param options - the full screen control options + * + * @example + * ```ts + * map.addControl(new maplibregl.FullscreenControl({container: document.querySelector('body')})); + * ``` + * @see [View a fullscreen map](https://maplibre.org/maplibre-gl-js/docs/examples/fullscreen/) + * + * ### Events + * + * @event `fullscreenstart` - Fired when fullscreen mode has started + * + * @event `fullscreenend` - Fired when fullscreen mode has ended + */ +class FullscreenControl extends performance.Evented { + constructor(options = {}) { + super(); + this._onFullscreenChange = () => { + const fullscreenElement = window.document.fullscreenElement || + window.document.mozFullScreenElement || + window.document.webkitFullscreenElement || + window.document.msFullscreenElement; + if ((fullscreenElement === this._container) !== this._fullscreen) { + this._handleFullscreenChange(); + } + }; + this._onClickFullscreen = () => { + if (this._isFullscreen()) { + this._exitFullscreen(); + } + else { + this._requestFullscreen(); + } + }; + this._fullscreen = false; + if (options && options.container) { + if (options.container instanceof HTMLElement) { + this._container = options.container; + } + else { + performance.warnOnce('Full screen control \'container\' must be a DOM element.'); + } + } + if ('onfullscreenchange' in document) { + this._fullscreenchange = 'fullscreenchange'; + } + else if ('onmozfullscreenchange' in document) { + this._fullscreenchange = 'mozfullscreenchange'; + } + else if ('onwebkitfullscreenchange' in document) { + this._fullscreenchange = 'webkitfullscreenchange'; + } + else if ('onmsfullscreenchange' in document) { + this._fullscreenchange = 'MSFullscreenChange'; + } + } + /** {@inheritDoc IControl.onAdd} */ + onAdd(map) { + this._map = map; + if (!this._container) + this._container = this._map.getContainer(); + this._controlContainer = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group'); + this._setupUI(); + return this._controlContainer; + } + /** {@inheritDoc IControl.onRemove} */ + onRemove() { + DOM.remove(this._controlContainer); + this._map = null; + window.document.removeEventListener(this._fullscreenchange, this._onFullscreenChange); + } + _setupUI() { + const button = this._fullscreenButton = DOM.create('button', (('maplibregl-ctrl-fullscreen')), this._controlContainer); + DOM.create('span', 'maplibregl-ctrl-icon', button).setAttribute('aria-hidden', 'true'); + button.type = 'button'; + this._updateTitle(); + this._fullscreenButton.addEventListener('click', this._onClickFullscreen); + window.document.addEventListener(this._fullscreenchange, this._onFullscreenChange); + } + _updateTitle() { + const title = this._getTitle(); + this._fullscreenButton.setAttribute('aria-label', title); + this._fullscreenButton.title = title; + } + _getTitle() { + return this._map._getUIString(this._isFullscreen() ? 'FullscreenControl.Exit' : 'FullscreenControl.Enter'); + } + _isFullscreen() { + return this._fullscreen; + } + _handleFullscreenChange() { + this._fullscreen = !this._fullscreen; + this._fullscreenButton.classList.toggle('maplibregl-ctrl-shrink'); + this._fullscreenButton.classList.toggle('maplibregl-ctrl-fullscreen'); + this._updateTitle(); + if (this._fullscreen) { + this.fire(new performance.Event('fullscreenstart')); + if (this._map._cooperativeGestures) { + this._prevCooperativeGestures = this._map._cooperativeGestures; + this._map.setCooperativeGestures(); + } + } + else { + this.fire(new performance.Event('fullscreenend')); + if (this._prevCooperativeGestures) { + this._map.setCooperativeGestures(this._prevCooperativeGestures); + delete this._prevCooperativeGestures; + } + } + } + _exitFullscreen() { + if (window.document.exitFullscreen) { + window.document.exitFullscreen(); + } + else if (window.document.mozCancelFullScreen) { + window.document.mozCancelFullScreen(); + } + else if (window.document.msExitFullscreen) { + window.document.msExitFullscreen(); + } + else if (window.document.webkitCancelFullScreen) { + window.document.webkitCancelFullScreen(); + } + else { + this._togglePseudoFullScreen(); + } + } + _requestFullscreen() { + if (this._container.requestFullscreen) { + this._container.requestFullscreen(); + } + else if (this._container.mozRequestFullScreen) { + this._container.mozRequestFullScreen(); + } + else if (this._container.msRequestFullscreen) { + this._container.msRequestFullscreen(); + } + else if (this._container.webkitRequestFullscreen) { + this._container.webkitRequestFullscreen(); + } + else { + this._togglePseudoFullScreen(); + } + } + _togglePseudoFullScreen() { + this._container.classList.toggle('maplibregl-pseudo-fullscreen'); + this._handleFullscreenChange(); + this._map.resize(); + } +} + +/** + * A `TerrainControl` control contains a button for turning the terrain on and off. + * + * @group Markers and Controls + * + * @example + * ```ts + * let map = new maplibregl.Map({TerrainControl: false}) + * .addControl(new maplibregl.TerrainControl({ + * source: "terrain" + * })); + * ``` + */ +class TerrainControl { + constructor(options) { + this._toggleTerrain = () => { + if (this._map.getTerrain()) { + this._map.setTerrain(null); + } + else { + this._map.setTerrain(this.options); + } + this._updateTerrainIcon(); + }; + this._updateTerrainIcon = () => { + this._terrainButton.classList.remove('maplibregl-ctrl-terrain'); + this._terrainButton.classList.remove('maplibregl-ctrl-terrain-enabled'); + if (this._map.terrain) { + this._terrainButton.classList.add('maplibregl-ctrl-terrain-enabled'); + this._terrainButton.title = this._map._getUIString('TerrainControl.disableTerrain'); + } + else { + this._terrainButton.classList.add('maplibregl-ctrl-terrain'); + this._terrainButton.title = this._map._getUIString('TerrainControl.enableTerrain'); + } + }; + this.options = options; + } + /** {@inheritDoc IControl.onAdd} */ + onAdd(map) { + this._map = map; + this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group'); + this._terrainButton = DOM.create('button', 'maplibregl-ctrl-terrain', this._container); + DOM.create('span', 'maplibregl-ctrl-icon', this._terrainButton).setAttribute('aria-hidden', 'true'); + this._terrainButton.type = 'button'; + this._terrainButton.addEventListener('click', this._toggleTerrain); + this._updateTerrainIcon(); + this._map.on('terrain', this._updateTerrainIcon); + return this._container; + } + /** {@inheritDoc IControl.onRemove} */ + onRemove() { + DOM.remove(this._container); + this._map.off('terrain', this._updateTerrainIcon); + this._map = undefined; + } +} + +const defaultOptions = { + closeButton: true, + closeOnClick: true, + focusAfterOpen: true, + className: '', + maxWidth: '240px' +}; +const focusQuerySelector = [ + 'a[href]', + '[tabindex]:not([tabindex=\'-1\'])', + '[contenteditable]:not([contenteditable=\'false\'])', + 'button:not([disabled])', + 'input:not([disabled])', + 'select:not([disabled])', + 'textarea:not([disabled])', +].join(', '); +/** + * A popup component. + * + * @group Markers and Controls + * + * + * @example + * Create a popup + * ```ts + * let popup = new maplibregl.Popup(); + * // Set an event listener that will fire + * // any time the popup is opened + * popup.on('open', function(){ + * console.log('popup was opened'); + * }); + * ``` + * + * @example + * Create a popup + * ```ts + * let popup = new maplibregl.Popup(); + * // Set an event listener that will fire + * // any time the popup is closed + * popup.on('close', function(){ + * console.log('popup was closed'); + * }); + * ``` + * + * @example + * ```ts + * let markerHeight = 50, markerRadius = 10, linearOffset = 25; + * let popupOffsets = { + * 'top': [0, 0], + * 'top-left': [0,0], + * 'top-right': [0,0], + * 'bottom': [0, -markerHeight], + * 'bottom-left': [linearOffset, (markerHeight - markerRadius + linearOffset) * -1], + * 'bottom-right': [-linearOffset, (markerHeight - markerRadius + linearOffset) * -1], + * 'left': [markerRadius, (markerHeight - markerRadius) * -1], + * 'right': [-markerRadius, (markerHeight - markerRadius) * -1] + * }; + * let popup = new maplibregl.Popup({offset: popupOffsets, className: 'my-class'}) + * .setLngLat(e.lngLat) + * .setHTML("

Hello World!

") + * .setMaxWidth("300px") + * .addTo(map); + * ``` + * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/) + * + * ### Events + * + * @event `open` Fired when the popup is opened manually or programmatically. `popup` object that was opened + * + * @event `close` Fired when the popup is closed manually or programmatically. `popup` object that was closed + */ +class Popup extends performance.Evented { + constructor(options) { + super(); + /** + * Removes the popup from the map it has been added to. + * + * @example + * ```ts + * let popup = new maplibregl.Popup().addTo(map); + * popup.remove(); + * ``` + * @returns `this` + */ + this.remove = () => { + if (this._content) { + DOM.remove(this._content); + } + if (this._container) { + DOM.remove(this._container); + delete this._container; + } + if (this._map) { + this._map.off('move', this._update); + this._map.off('move', this._onClose); + this._map.off('click', this._onClose); + this._map.off('remove', this.remove); + this._map.off('mousemove', this._onMouseMove); + this._map.off('mouseup', this._onMouseUp); + this._map.off('drag', this._onDrag); + delete this._map; + } + this.fire(new performance.Event('close')); + return this; + }; + this._onMouseUp = (event) => { + this._update(event.point); + }; + this._onMouseMove = (event) => { + this._update(event.point); + }; + this._onDrag = (event) => { + this._update(event.point); + }; + this._update = (cursor) => { + const hasPosition = this._lngLat || this._trackPointer; + if (!this._map || !hasPosition || !this._content) { + return; + } + if (!this._container) { + this._container = DOM.create('div', 'maplibregl-popup', this._map.getContainer()); + this._tip = DOM.create('div', 'maplibregl-popup-tip', this._container); + this._container.appendChild(this._content); + if (this.options.className) { + for (const name of this.options.className.split(' ')) { + this._container.classList.add(name); + } + } + if (this._trackPointer) { + this._container.classList.add('maplibregl-popup-track-pointer'); + } + } + if (this.options.maxWidth && this._container.style.maxWidth !== this.options.maxWidth) { + this._container.style.maxWidth = this.options.maxWidth; + } + if (this._map.transform.renderWorldCopies && !this._trackPointer) { + this._lngLat = smartWrap(this._lngLat, this._pos, this._map.transform); + } + if (this._trackPointer && !cursor) + return; + const pos = this._pos = this._trackPointer && cursor ? cursor : this._map.project(this._lngLat); + let anchor = this.options.anchor; + const offset = normalizeOffset(this.options.offset); + if (!anchor) { + const width = this._container.offsetWidth; + const height = this._container.offsetHeight; + let anchorComponents; + if (pos.y + offset.bottom.y < height) { + anchorComponents = ['top']; + } + else if (pos.y > this._map.transform.height - height) { + anchorComponents = ['bottom']; + } + else { + anchorComponents = []; + } + if (pos.x < width / 2) { + anchorComponents.push('left'); + } + else if (pos.x > this._map.transform.width - width / 2) { + anchorComponents.push('right'); + } + if (anchorComponents.length === 0) { + anchor = 'bottom'; + } + else { + anchor = anchorComponents.join('-'); + } + } + const offsetedPos = pos.add(offset[anchor]).round(); + DOM.setTransform(this._container, `${anchorTranslate[anchor]} translate(${offsetedPos.x}px,${offsetedPos.y}px)`); + applyAnchorClass(this._container, anchor, 'popup'); + }; + this._onClose = () => { + this.remove(); + }; + this.options = performance.extend(Object.create(defaultOptions), options); + } + /** + * Adds the popup to a map. + * + * @param map - The MapLibre GL JS map to add the popup to. + * @returns `this` + * @example + * ```ts + * new maplibregl.Popup() + * .setLngLat([0, 0]) + * .setHTML("

Null Island

") + * .addTo(map); + * ``` + * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + * @see [Show polygon information on click](https://maplibre.org/maplibre-gl-js/docs/examples/polygon-popup-on-click/) + */ + addTo(map) { + if (this._map) + this.remove(); + this._map = map; + if (this.options.closeOnClick) { + this._map.on('click', this._onClose); + } + if (this.options.closeOnMove) { + this._map.on('move', this._onClose); + } + this._map.on('remove', this.remove); + this._update(); + this._focusFirstElement(); + if (this._trackPointer) { + this._map.on('mousemove', this._onMouseMove); + this._map.on('mouseup', this._onMouseUp); + if (this._container) { + this._container.classList.add('maplibregl-popup-track-pointer'); + } + this._map._canvasContainer.classList.add('maplibregl-track-pointer'); + } + else { + this._map.on('move', this._update); + } + this.fire(new performance.Event('open')); + return this; + } + /** + * @returns `true` if the popup is open, `false` if it is closed. + */ + isOpen() { + return !!this._map; + } + /** + * Returns the geographical location of the popup's anchor. + * + * The longitude of the result may differ by a multiple of 360 degrees from the longitude previously + * set by `setLngLat` because `Popup` wraps the anchor longitude across copies of the world to keep + * the popup on screen. + * + * @returns The geographical location of the popup's anchor. + */ + getLngLat() { + return this._lngLat; + } + /** + * Sets the geographical location of the popup's anchor, and moves the popup to it. Replaces trackPointer() behavior. + * + * @param lnglat - The geographical location to set as the popup's anchor. + * @returns `this` + */ + setLngLat(lnglat) { + this._lngLat = performance.LngLat.convert(lnglat); + this._pos = null; + this._trackPointer = false; + this._update(); + if (this._map) { + this._map.on('move', this._update); + this._map.off('mousemove', this._onMouseMove); + if (this._container) { + this._container.classList.remove('maplibregl-popup-track-pointer'); + } + this._map._canvasContainer.classList.remove('maplibregl-track-pointer'); + } + return this; + } + /** + * Tracks the popup anchor to the cursor position on screens with a pointer device (it will be hidden on touchscreens). Replaces the `setLngLat` behavior. + * For most use cases, set `closeOnClick` and `closeButton` to `false`. + * @example + * ```ts + * let popup = new maplibregl.Popup({ closeOnClick: false, closeButton: false }) + * .setHTML("

Hello World!

") + * .trackPointer() + * .addTo(map); + * ``` + * @returns `this` + */ + trackPointer() { + this._trackPointer = true; + this._pos = null; + this._update(); + if (this._map) { + this._map.off('move', this._update); + this._map.on('mousemove', this._onMouseMove); + this._map.on('drag', this._onDrag); + if (this._container) { + this._container.classList.add('maplibregl-popup-track-pointer'); + } + this._map._canvasContainer.classList.add('maplibregl-track-pointer'); + } + return this; + } + /** + * Returns the `Popup`'s HTML element. + * @example + * Change the `Popup` element's font size + * ```ts + * let popup = new maplibregl.Popup() + * .setLngLat([-96, 37.8]) + * .setHTML("

Hello World!

") + * .addTo(map); + * let popupElem = popup.getElement(); + * popupElem.style.fontSize = "25px"; + * ``` + * @returns element + */ + getElement() { + return this._container; + } + /** + * Sets the popup's content to a string of text. + * + * This function creates a [Text](https://developer.mozilla.org/en-US/docs/Web/API/Text) node in the DOM, + * so it cannot insert raw HTML. Use this method for security against XSS + * if the popup content is user-provided. + * + * @param text - Textual content for the popup. + * @returns `this` + * @example + * ```ts + * let popup = new maplibregl.Popup() + * .setLngLat(e.lngLat) + * .setText('Hello, world!') + * .addTo(map); + * ``` + */ + setText(text) { + return this.setDOMContent(document.createTextNode(text)); + } + /** + * Sets the popup's content to the HTML provided as a string. + * + * This method does not perform HTML filtering or sanitization, and must be + * used only with trusted content. Consider {@link Popup#setText} if + * the content is an untrusted text string. + * + * @param html - A string representing HTML content for the popup. + * @returns `this` + * @example + * ```ts + * let popup = new maplibregl.Popup() + * .setLngLat(e.lngLat) + * .setHTML("

Hello World!

") + * .addTo(map); + * ``` + * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/) + */ + setHTML(html) { + const frag = document.createDocumentFragment(); + const temp = document.createElement('body'); + let child; + temp.innerHTML = html; + while (true) { + child = temp.firstChild; + if (!child) + break; + frag.appendChild(child); + } + return this.setDOMContent(frag); + } + /** + * Returns the popup's maximum width. + * + * @returns The maximum width of the popup. + */ + getMaxWidth() { + var _a; + return (_a = this._container) === null || _a === void 0 ? void 0 : _a.style.maxWidth; + } + /** + * Sets the popup's maximum width. This is setting the CSS property `max-width`. + * Available values can be found here: https://developer.mozilla.org/en-US/docs/Web/CSS/max-width + * + * @param maxWidth - A string representing the value for the maximum width. + * @returns `this` + */ + setMaxWidth(maxWidth) { + this.options.maxWidth = maxWidth; + this._update(); + return this; + } + /** + * Sets the popup's content to the element provided as a DOM node. + * + * @param htmlNode - A DOM node to be used as content for the popup. + * @returns `this` + * @example + * Create an element with the popup content + * ```ts + * let div = document.createElement('div'); + * div.innerHTML = 'Hello, world!'; + * let popup = new maplibregl.Popup() + * .setLngLat(e.lngLat) + * .setDOMContent(div) + * .addTo(map); + * ``` + */ + setDOMContent(htmlNode) { + if (this._content) { + // Clear out children first. + while (this._content.hasChildNodes()) { + if (this._content.firstChild) { + this._content.removeChild(this._content.firstChild); + } + } + } + else { + this._content = DOM.create('div', 'maplibregl-popup-content', this._container); + } + // The close button should be the last tabbable element inside the popup for a good keyboard UX. + this._content.appendChild(htmlNode); + this._createCloseButton(); + this._update(); + this._focusFirstElement(); + return this; + } + /** + * Adds a CSS class to the popup container element. + * + * @param className - Non-empty string with CSS class name to add to popup container + * + * @example + * ```ts + * let popup = new maplibregl.Popup() + * popup.addClassName('some-class') + * ``` + */ + addClassName(className) { + if (this._container) { + this._container.classList.add(className); + } + } + /** + * Removes a CSS class from the popup container element. + * + * @param className - Non-empty string with CSS class name to remove from popup container + * + * @example + * ```ts + * let popup = new maplibregl.Popup() + * popup.removeClassName('some-class') + * ``` + */ + removeClassName(className) { + if (this._container) { + this._container.classList.remove(className); + } + } + /** + * Sets the popup's offset. + * + * @param offset - Sets the popup's offset. + * @returns `this` + */ + setOffset(offset) { + this.options.offset = offset; + this._update(); + return this; + } + /** + * Add or remove the given CSS class on the popup container, depending on whether the container currently has that class. + * + * @param className - Non-empty string with CSS class name to add/remove + * + * @returns if the class was removed return false, if class was added, then return true, undefined if there is no container + * + * @example + * ```ts + * let popup = new maplibregl.Popup() + * popup.toggleClassName('toggleClass') + * ``` + */ + toggleClassName(className) { + if (this._container) { + return this._container.classList.toggle(className); + } + } + _createCloseButton() { + if (this.options.closeButton) { + this._closeButton = DOM.create('button', 'maplibregl-popup-close-button', this._content); + this._closeButton.type = 'button'; + this._closeButton.setAttribute('aria-label', 'Close popup'); + this._closeButton.innerHTML = '×'; + this._closeButton.addEventListener('click', this._onClose); + } + } + _focusFirstElement() { + if (!this.options.focusAfterOpen || !this._container) + return; + const firstFocusable = this._container.querySelector(focusQuerySelector); + if (firstFocusable) + firstFocusable.focus(); + } +} +function normalizeOffset(offset) { + if (!offset) { + return normalizeOffset(new performance.Point(0, 0)); + } + else if (typeof offset === 'number') { + // input specifies a radius from which to calculate offsets at all positions + const cornerOffset = Math.round(Math.abs(offset) / Math.SQRT2); + return { + 'center': new performance.Point(0, 0), + 'top': new performance.Point(0, offset), + 'top-left': new performance.Point(cornerOffset, cornerOffset), + 'top-right': new performance.Point(-cornerOffset, cornerOffset), + 'bottom': new performance.Point(0, -offset), + 'bottom-left': new performance.Point(cornerOffset, -cornerOffset), + 'bottom-right': new performance.Point(-cornerOffset, -cornerOffset), + 'left': new performance.Point(offset, 0), + 'right': new performance.Point(-offset, 0) + }; + } + else if (offset instanceof performance.Point || Array.isArray(offset)) { + // input specifies a single offset to be applied to all positions + const convertedOffset = performance.Point.convert(offset); + return { + 'center': convertedOffset, + 'top': convertedOffset, + 'top-left': convertedOffset, + 'top-right': convertedOffset, + 'bottom': convertedOffset, + 'bottom-left': convertedOffset, + 'bottom-right': convertedOffset, + 'left': convertedOffset, + 'right': convertedOffset + }; + } + else { + // input specifies an offset per position + return { + 'center': performance.Point.convert(offset['center'] || [0, 0]), + 'top': performance.Point.convert(offset['top'] || [0, 0]), + 'top-left': performance.Point.convert(offset['top-left'] || [0, 0]), + 'top-right': performance.Point.convert(offset['top-right'] || [0, 0]), + 'bottom': performance.Point.convert(offset['bottom'] || [0, 0]), + 'bottom-left': performance.Point.convert(offset['bottom-left'] || [0, 0]), + 'bottom-right': performance.Point.convert(offset['bottom-right'] || [0, 0]), + 'left': performance.Point.convert(offset['left'] || [0, 0]), + 'right': performance.Point.convert(offset['right'] || [0, 0]) + }; + } +} + +/** + * This is a private namespace for utility functions that will get automatically stripped + * out in production builds. + */ +const Debug = { + extend(dest, ...sources) { + return performance.extend(dest, ...sources); + }, + run(fn) { + fn(); + }, + logToElement(message, overwrite = false, id = 'log') { + const el = window.document.getElementById(id); + if (el) { + if (overwrite) + el.innerHTML = ''; + el.innerHTML += `
${message}`; + } + } +}; + +const version = packageJSON.version; +/** + * `maplibregl` is the global object that allows configurations that are not specific to a map instance + * + * @group Main + */ +class MapLibreGL { + /** + * Returns the package version of the library + * @returns Package version of the library + */ + static get version() { + return version; + } + /** + * Gets and sets the number of web workers instantiated on a page with GL JS maps. + * By default, workerCount is 1 except for Safari browser where it is set to half the number of CPU cores (capped at 3). + * Make sure to set this property before creating any map instances for it to have effect. + * + * @returns Number of workers currently configured. + * @example + * ```ts + * maplibregl.workerCount = 2; + * ``` + */ + static get workerCount() { + return WorkerPool.workerCount; + } + static set workerCount(count) { + WorkerPool.workerCount = count; + } + /** + * Gets and sets the maximum number of images (raster tiles, sprites, icons) to load in parallel, + * which affects performance in raster-heavy maps. 16 by default. + * + * @returns Number of parallel requests currently configured. + * @example + * ```ts + * maplibregl.maxParallelImageRequests = 10; + * ``` + */ + static get maxParallelImageRequests() { + return performance.config.MAX_PARALLEL_IMAGE_REQUESTS; + } + static set maxParallelImageRequests(numRequests) { + performance.config.MAX_PARALLEL_IMAGE_REQUESTS = numRequests; + } + static get workerUrl() { + return performance.config.WORKER_URL; + } + static set workerUrl(value) { + performance.config.WORKER_URL = value; + } + /** + * Sets a custom load tile function that will be called when using a source that starts with a custom url schema. + * The example below will be triggered for custom:// urls defined in the sources list in the style definitions. + * The function passed will receive the request parameters and should call the callback with the resulting request, + * for example a pbf vector tile, non-compressed, represented as ArrayBuffer. + * + * @param customProtocol - the protocol to hook, for example 'custom' + * @param loadFn - the function to use when trying to fetch a tile specified by the customProtocol + * @example + * This will fetch a file using the fetch API (this is obviously a non interesting example...) + * ```ts + * maplibregl.addProtocol('custom', (params, callback) => { + fetch(`https://${params.url.split("://")[1]}`) + .then(t => { + if (t.status == 200) { + t.arrayBuffer().then(arr => { + callback(null, arr, null, null); + }); + } else { + callback(new Error(`Tile fetch error: ${t.statusText}`)); + } + }) + .catch(e => { + callback(new Error(e)); + }); + return { cancel: () => { } }; + }); + * // the following is an example of a way to return an error when trying to load a tile + * maplibregl.addProtocol('custom2', (params, callback) => { + * callback(new Error('someErrorMessage')); + * return { cancel: () => { } }; + * }); + * ``` + */ + static addProtocol(customProtocol, loadFn) { + performance.config.REGISTERED_PROTOCOLS[customProtocol] = loadFn; + } + /** + * Removes a previously added protocol + * + * @param customProtocol - the custom protocol to remove registration for + * @example + * ```ts + * maplibregl.removeProtocol('custom'); + * ``` + */ + static removeProtocol(customProtocol) { + delete performance.config.REGISTERED_PROTOCOLS[customProtocol]; + } +} +MapLibreGL.Map = Map$1; +MapLibreGL.NavigationControl = NavigationControl; +MapLibreGL.GeolocateControl = GeolocateControl; +MapLibreGL.AttributionControl = AttributionControl; +MapLibreGL.LogoControl = LogoControl; +MapLibreGL.ScaleControl = ScaleControl; +MapLibreGL.FullscreenControl = FullscreenControl; +MapLibreGL.TerrainControl = TerrainControl; +MapLibreGL.Popup = Popup; +MapLibreGL.Marker = Marker; +MapLibreGL.Style = Style; +MapLibreGL.LngLat = performance.LngLat; +MapLibreGL.LngLatBounds = LngLatBounds; +MapLibreGL.Point = performance.Point; +MapLibreGL.MercatorCoordinate = performance.MercatorCoordinate; +MapLibreGL.Evented = performance.Evented; +MapLibreGL.AJAXError = performance.AJAXError; +MapLibreGL.config = performance.config; +MapLibreGL.CanvasSource = CanvasSource; +MapLibreGL.GeoJSONSource = GeoJSONSource; +MapLibreGL.ImageSource = ImageSource; +MapLibreGL.RasterDEMTileSource = RasterDEMTileSource; +MapLibreGL.RasterTileSource = RasterTileSource; +MapLibreGL.VectorTileSource = VectorTileSource; +MapLibreGL.VideoSource = VideoSource; +/** + * Sets the map's [RTL text plugin](https://www.mapbox.com/mapbox-gl-js/plugins/#mapbox-gl-rtl-text). + * Necessary for supporting the Arabic and Hebrew languages, which are written right-to-left. + * + * @param pluginURL - URL pointing to the Mapbox RTL text plugin source. + * @param callback - Called with an error argument if there is an error. + * @param lazy - If set to `true`, mapboxgl will defer loading the plugin until rtl text is encountered, + * rtl text will then be rendered only after the plugin finishes loading. + * @example + * ```ts + * maplibregl.setRTLTextPlugin('https://unpkg.com/@mapbox/mapbox-gl-rtl-text@0.2.3/mapbox-gl-rtl-text.js'); + * ``` + * @see [Add support for right-to-left scripts](https://maplibre.org/maplibre-gl-js/docs/examples/mapbox-gl-rtl-text/) + */ +MapLibreGL.setRTLTextPlugin = performance.setRTLTextPlugin; +/** + * Gets the map's [RTL text plugin](https://www.mapbox.com/mapbox-gl-js/plugins/#mapbox-gl-rtl-text) status. + * The status can be `unavailable` (i.e. not requested or removed), `loading`, `loaded` or `error`. + * If the status is `loaded` and the plugin is requested again, an error will be thrown. + * + * @example + * ```ts + * const pluginStatus = maplibregl.getRTLTextPluginStatus(); + * ``` + */ +MapLibreGL.getRTLTextPluginStatus = performance.getRTLTextPluginStatus; +/** + * Initializes resources like WebWorkers that can be shared across maps to lower load + * times in some situations. `maplibregl.workerUrl` and `maplibregl.workerCount`, if being + * used, must be set before `prewarm()` is called to have an effect. + * + * By default, the lifecycle of these resources is managed automatically, and they are + * lazily initialized when a Map is first created. By invoking `prewarm()`, these + * resources will be created ahead of time, and will not be cleared when the last Map + * is removed from the page. This allows them to be re-used by new Map instances that + * are created later. They can be manually cleared by calling + * `maplibregl.clearPrewarmedResources()`. This is only necessary if your web page remains + * active but stops using maps altogether. + * + * This is primarily useful when using GL-JS maps in a single page app, wherein a user + * would navigate between various views that can cause Map instances to constantly be + * created and destroyed. + * + * @example + * ```ts + * maplibregl.prewarm() + * ``` + */ +MapLibreGL.prewarm = prewarm; +/** + * Clears up resources that have previously been created by `maplibregl.prewarm()`. + * Note that this is typically not necessary. You should only call this function + * if you expect the user of your app to not return to a Map view at any point + * in your application. + * + * @example + * ```ts + * maplibregl.clearPrewarmedResources() + * ``` + */ +MapLibreGL.clearPrewarmedResources = clearPrewarmedResources; +//This gets automatically stripped out in production builds. +Debug.extend(MapLibreGL, { isSafari: performance.isSafari, getPerformanceMetrics: performance.PerformanceUtils.getPerformanceMetrics }); + +return MapLibreGL; + +})); + +// +// Our custom intro provides a specialized "define()" function, called by the +// AMD modules below, that sets up the worker blob URL and then executes the +// main module, storing its exported value as 'maplibregl' + + +var maplibregl$1 = maplibregl; + +return maplibregl$1; + +})); +//# sourceMappingURL=maplibre-gl-dev.js.map diff --git a/web/libraries/maplibre-gl/dist/maplibre-gl-dev.js.map b/web/libraries/maplibre-gl/dist/maplibre-gl-dev.js.map new file mode 100644 index 00000000..7145d8c9 --- /dev/null +++ b/web/libraries/maplibre-gl/dist/maplibre-gl-dev.js.map @@ -0,0 +1 @@ +{"version":3,"file":"maplibre-gl-dev.js","sources":["../node_modules/tslib/tslib.es6.js","../node_modules/@mapbox/point-geometry/index.js","../node_modules/@mapbox/unitbezier/index.js","../src/util/offscreen_canvas_supported.ts","../src/util/offscreen_canvas_distorted.ts","../src/util/util.ts","../src/util/browser.ts","../src/util/config.ts","../src/util/ajax.ts","../src/util/evented.ts","../node_modules/@maplibre/maplibre-gl-style-spec/dist/index.mjs","../src/style/validate_style.ts","../src/util/transferable_grid_index.ts","../src/util/web_worker_transfer.ts","../src/style/zoom_history.ts","../src/util/is_char_in_unicode_block.ts","../src/util/script_detection.ts","../src/source/rtl_text_plugin.ts","../src/style/evaluation_parameters.ts","../src/style/properties.ts","../src/style/style_layer.ts","../src/util/struct_array.ts","../src/data/array_types.g.ts","../src/data/bucket/circle_attributes.ts","../src/data/segment.ts","../src/shaders/encode_attribute.ts","../src/data/bucket/pattern_attributes.ts","../node_modules/murmurhash-js/murmurhash3_gc.js","../node_modules/murmurhash-js/murmurhash2_gc.js","../node_modules/murmurhash-js/index.js","../src/data/feature_position_map.ts","../src/render/uniform_binding.ts","../src/data/program_configuration.ts","../src/data/extent.ts","../src/data/load_geometry.ts","../src/data/evaluation_feature.ts","../src/data/bucket/circle_bucket.ts","../src/util/intersection_tests.ts","../src/style/query_utils.ts","../src/style/style_layer/circle_style_layer_properties.g.ts","../node_modules/gl-matrix/esm/common.js","../node_modules/gl-matrix/esm/mat2.js","../node_modules/gl-matrix/esm/mat2d.js","../node_modules/gl-matrix/esm/mat3.js","../node_modules/gl-matrix/esm/mat4.js","../node_modules/gl-matrix/esm/vec3.js","../node_modules/gl-matrix/esm/vec4.js","../node_modules/gl-matrix/esm/quat.js","../node_modules/gl-matrix/esm/quat2.js","../node_modules/gl-matrix/esm/vec2.js","../src/style/style_layer/circle_style_layer.ts","../src/data/bucket/heatmap_bucket.ts","../src/style/style_layer/heatmap_style_layer_properties.g.ts","../src/util/image.ts","../src/util/color_ramp.ts","../src/style/style_layer/heatmap_style_layer.ts","../src/style/style_layer/hillshade_style_layer_properties.g.ts","../src/style/style_layer/hillshade_style_layer.ts","../src/data/bucket/fill_attributes.ts","../node_modules/earcut/src/earcut.js","../node_modules/quickselect/index.js","../src/util/classify_rings.ts","../src/data/bucket/pattern_bucket_features.ts","../src/data/bucket/fill_bucket.ts","../src/style/style_layer/fill_style_layer_properties.g.ts","../src/style/style_layer/fill_style_layer.ts","../src/data/bucket/fill_extrusion_attributes.ts","../node_modules/@mapbox/vector-tile/lib/vectortilefeature.js","../node_modules/@mapbox/vector-tile/lib/vectortilelayer.js","../node_modules/@mapbox/vector-tile/lib/vectortile.js","../node_modules/@mapbox/vector-tile/index.js","../src/data/bucket/fill_extrusion_bucket.ts","../src/style/style_layer/fill_extrusion_style_layer_properties.g.ts","../src/style/style_layer/fill_extrusion_style_layer.ts","../src/data/bucket/line_attributes.ts","../src/data/bucket/line_attributes_ext.ts","../src/data/bucket/line_bucket.ts","../src/style/style_layer/line_style_layer_properties.g.ts","../src/style/style_layer/line_style_layer.ts","../src/data/bucket/symbol_attributes.ts","../src/symbol/transform_text.ts","../src/symbol/merge_lines.ts","../src/util/verticalize_punctuation.ts","../src/symbol/one_em.ts","../node_modules/ieee754/index.js","../node_modules/pbf/index.js","../src/style/parse_glyph_pbf.ts","../node_modules/potpack/index.js","../src/render/image_atlas.ts","../src/symbol/shaping.ts","../src/symbol/symbol_size.ts","../src/style/style_layer/overlap_mode.ts","../src/data/bucket/symbol_bucket.ts","../src/util/resolve_tokens.ts","../src/style/style_layer/symbol_style_layer_properties.g.ts","../src/style/format_section_override.ts","../src/style/style_layer/symbol_style_layer.ts","../src/style/style_layer/background_style_layer_properties.g.ts","../src/style/style_layer/background_style_layer.ts","../src/style/style_layer/raster_style_layer_properties.g.ts","../src/style/style_layer/raster_style_layer.ts","../src/style/style_layer/custom_style_layer.ts","../src/style/create_style_layer.ts","../src/util/throttled_invoker.ts","../src/util/actor.ts","../src/geo/lng_lat.ts","../src/geo/mercator_coordinate.ts","../node_modules/@mapbox/whoots-js/index.mjs","../src/source/tile_id.ts","../src/data/dem_data.ts","../src/util/dictionary_coder.ts","../src/util/vectortile_to_geojson.ts","../src/data/feature_index.ts","../src/symbol/clip_line.ts","../src/symbol/anchor.ts","../src/symbol/check_max_angle.ts","../src/symbol/get_anchors.ts","../src/symbol/quads.ts","../src/symbol/collision_feature.ts","../node_modules/tinyqueue/index.js","../src/util/find_pole_of_inaccessibility.ts","../src/style/style_layer/variable_text_anchor.ts","../src/symbol/symbol_layout.ts","../node_modules/kdbush/index.js","../src/util/performance.ts","../src/style/style_layer_index.ts","../src/render/glyph_atlas.ts","../src/source/worker_tile.ts","../src/source/vector_tile_worker_source.ts","../src/source/raster_dem_tile_worker_source.ts","../node_modules/@mapbox/geojson-rewind/index.js","../src/source/geojson_wrapper.ts","../node_modules/vt-pbf/lib/geojson_wrapper.js","../node_modules/vt-pbf/index.js","../node_modules/supercluster/index.js","../node_modules/geojson-vt/src/simplify.js","../node_modules/geojson-vt/src/feature.js","../node_modules/geojson-vt/src/convert.js","../node_modules/geojson-vt/src/clip.js","../node_modules/geojson-vt/src/wrap.js","../node_modules/geojson-vt/src/transform.js","../node_modules/geojson-vt/src/tile.js","../node_modules/geojson-vt/src/index.js","../src/source/geojson_source_diff.ts","../src/source/geojson_worker_source.ts","../src/source/worker.ts","../src/util/dom.ts","../src/util/webp_supported.ts","../src/util/image_request.ts","../src/util/request_manager.ts","../src/util/style.ts","../src/style/load_sprite.ts","../src/render/texture.ts","../src/style/style_image.ts","../src/render/image_manager.ts","../src/style/load_glyph_range.ts","../node_modules/@mapbox/tiny-sdf/index.js","../src/render/glyph_manager.ts","../src/style/light.ts","../src/render/line_atlas.ts","../src/util/dispatcher.ts","../src/source/load_tilejson.ts","../src/geo/lng_lat_bounds.ts","../src/source/tile_bounds.ts","../src/source/vector_tile_source.ts","../src/source/raster_tile_source.ts","../src/source/raster_dem_tile_source.ts","../src/source/geojson_source.ts","../src/data/raster_bounds_attributes.ts","../src/source/image_source.ts","../src/source/video_source.ts","../src/source/canvas_source.ts","../src/source/source.ts","../src/source/query_features.ts","../src/data/bucket.ts","../src/source/tile.ts","../src/source/tile_cache.ts","../src/source/source_state.ts","../src/source/source_cache.ts","../src/util/web_worker.ts","../src/util/worker_pool.ts","../src/util/global_worker_pool.ts","../src/symbol/path_interpolator.ts","../src/symbol/grid_index.ts","../src/symbol/projection.ts","../src/symbol/collision_index.ts","../src/source/pixels_to_tile_units.ts","../src/symbol/placement.ts","../src/style/pauseable_placement.ts","../src/symbol/cross_tile_symbol_index.ts","../src/style/style.ts","../src/data/pos_attributes.ts","../src/shaders/_prelude.fragment.glsl.g.ts","../src/shaders/_prelude.vertex.glsl.g.ts","../src/shaders/background.fragment.glsl.g.ts","../src/shaders/background.vertex.glsl.g.ts","../src/shaders/background_pattern.fragment.glsl.g.ts","../src/shaders/background_pattern.vertex.glsl.g.ts","../src/shaders/circle.fragment.glsl.g.ts","../src/shaders/circle.vertex.glsl.g.ts","../src/shaders/clipping_mask.fragment.glsl.g.ts","../src/shaders/clipping_mask.vertex.glsl.g.ts","../src/shaders/heatmap.fragment.glsl.g.ts","../src/shaders/heatmap.vertex.glsl.g.ts","../src/shaders/heatmap_texture.fragment.glsl.g.ts","../src/shaders/heatmap_texture.vertex.glsl.g.ts","../src/shaders/collision_box.fragment.glsl.g.ts","../src/shaders/collision_box.vertex.glsl.g.ts","../src/shaders/collision_circle.fragment.glsl.g.ts","../src/shaders/collision_circle.vertex.glsl.g.ts","../src/shaders/debug.fragment.glsl.g.ts","../src/shaders/debug.vertex.glsl.g.ts","../src/shaders/fill.fragment.glsl.g.ts","../src/shaders/fill.vertex.glsl.g.ts","../src/shaders/fill_outline.fragment.glsl.g.ts","../src/shaders/fill_outline.vertex.glsl.g.ts","../src/shaders/fill_outline_pattern.fragment.glsl.g.ts","../src/shaders/fill_outline_pattern.vertex.glsl.g.ts","../src/shaders/fill_pattern.fragment.glsl.g.ts","../src/shaders/fill_pattern.vertex.glsl.g.ts","../src/shaders/fill_extrusion.fragment.glsl.g.ts","../src/shaders/fill_extrusion.vertex.glsl.g.ts","../src/shaders/fill_extrusion_pattern.fragment.glsl.g.ts","../src/shaders/fill_extrusion_pattern.vertex.glsl.g.ts","../src/shaders/hillshade_prepare.fragment.glsl.g.ts","../src/shaders/hillshade_prepare.vertex.glsl.g.ts","../src/shaders/hillshade.fragment.glsl.g.ts","../src/shaders/hillshade.vertex.glsl.g.ts","../src/shaders/line.fragment.glsl.g.ts","../src/shaders/line.vertex.glsl.g.ts","../src/shaders/line_gradient.fragment.glsl.g.ts","../src/shaders/line_gradient.vertex.glsl.g.ts","../src/shaders/line_pattern.fragment.glsl.g.ts","../src/shaders/line_pattern.vertex.glsl.g.ts","../src/shaders/line_sdf.fragment.glsl.g.ts","../src/shaders/line_sdf.vertex.glsl.g.ts","../src/shaders/raster.fragment.glsl.g.ts","../src/shaders/raster.vertex.glsl.g.ts","../src/shaders/symbol_icon.fragment.glsl.g.ts","../src/shaders/symbol_icon.vertex.glsl.g.ts","../src/shaders/symbol_sdf.fragment.glsl.g.ts","../src/shaders/symbol_sdf.vertex.glsl.g.ts","../src/shaders/symbol_text_and_icon.fragment.glsl.g.ts","../src/shaders/symbol_text_and_icon.vertex.glsl.g.ts","../src/shaders/terrain_depth.fragment.glsl.g.ts","../src/shaders/terrain_coords.fragment.glsl.g.ts","../src/shaders/terrain.fragment.glsl.g.ts","../src/shaders/terrain.vertex.glsl.g.ts","../src/shaders/shaders.ts","../src/render/vertex_array_object.ts","../src/render/program/terrain_program.ts","../src/render/program.ts","../src/render/program/pattern.ts","../src/render/program/fill_extrusion_program.ts","../src/render/program/fill_program.ts","../src/render/program/circle_program.ts","../src/render/program/collision_program.ts","../src/render/program/debug_program.ts","../src/render/program/clipping_mask_program.ts","../src/render/program/heatmap_program.ts","../src/render/program/hillshade_program.ts","../src/render/program/line_program.ts","../src/render/program/raster_program.ts","../src/render/program/symbol_program.ts","../src/render/program/background_program.ts","../src/render/program/program_uniforms.ts","../src/gl/index_buffer.ts","../src/gl/vertex_buffer.ts","../src/gl/webgl2.ts","../src/gl/value.ts","../src/gl/framebuffer.ts","../src/gl/color_mode.ts","../src/gl/context.ts","../src/gl/depth_mode.ts","../src/gl/stencil_mode.ts","../src/gl/cull_face_mode.ts","../src/render/draw_collision_debug.ts","../src/render/draw_symbol.ts","../src/render/draw_circle.ts","../src/render/draw_heatmap.ts","../src/render/draw_line.ts","../src/render/update_pattern_positions_in_program.ts","../src/render/draw_fill.ts","../src/render/draw_fill_extrusion.ts","../src/render/draw_hillshade.ts","../src/render/draw_raster.ts","../src/render/draw_background.ts","../src/render/draw_debug.ts","../src/render/draw_custom.ts","../src/render/draw_terrain.ts","../src/render/painter.ts","../src/util/primitives.ts","../src/geo/edge_insets.ts","../src/geo/transform.ts","../src/util/throttle.ts","../src/ui/hash.ts","../src/ui/handler_inertia.ts","../src/ui/events.ts","../src/ui/handler/map_event.ts","../src/ui/handler/transform-provider.ts","../src/ui/handler/box_zoom.ts","../src/ui/handler/handler_util.ts","../src/ui/handler/tap_recognizer.ts","../src/ui/handler/tap_zoom.ts","../src/ui/handler/drag_handler.ts","../src/ui/handler/drag_move_state_manager.ts","../src/ui/handler/mouse.ts","../src/ui/handler/touch_pan.ts","../src/ui/handler/two_fingers_touch.ts","../src/ui/handler/keyboard.ts","../src/ui/handler/scroll_zoom.ts","../src/ui/handler/shim/dblclick_zoom.ts","../src/ui/handler/click_zoom.ts","../src/ui/handler/tap_drag_zoom.ts","../src/ui/handler/shim/drag_pan.ts","../src/ui/handler/shim/drag_rotate.ts","../src/ui/handler/shim/two_fingers_touch.ts","../src/ui/handler_manager.ts","../src/ui/camera.ts","../src/ui/control/attribution_control.ts","../src/ui/control/logo_control.ts","../src/util/task_queue.ts","../src/ui/default_locale.ts","../src/data/pos3d_attributes.ts","../src/source/terrain_source_cache.ts","../src/render/terrain.ts","../src/gl/render_pool.ts","../src/render/render_to_texture.ts","../src/ui/map.ts","../src/ui/handler/one_finger_touch_drag.ts","../src/ui/control/navigation_control.ts","../src/util/geolocation_support.ts","../src/util/smart_wrap.ts","../src/ui/anchor.ts","../src/ui/marker.ts","../src/ui/control/geolocate_control.ts","../src/ui/control/scale_control.ts","../src/ui/control/fullscreen_control.ts","../src/ui/control/terrain_control.ts","../src/ui/popup.ts","../src/util/debug.ts","../src/index.ts","../build/rollup/maplibregl.js"],"sourcesContent":["/******************************************************************************\r\nCopyright (c) Microsoft Corporation.\r\n\r\nPermission to use, copy, modify, and/or distribute this software for any\r\npurpose with or without fee is hereby granted.\r\n\r\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\r\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\r\nAND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\r\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\r\nLOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\r\nOTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\r\nPERFORMANCE OF THIS SOFTWARE.\r\n***************************************************************************** */\r\n/* global Reflect, Promise, SuppressedError, Symbol */\r\n\r\nvar extendStatics = function(d, b) {\r\n extendStatics = Object.setPrototypeOf ||\r\n ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\r\n function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };\r\n return extendStatics(d, b);\r\n};\r\n\r\nexport function __extends(d, b) {\r\n if (typeof b !== \"function\" && b !== null)\r\n throw new TypeError(\"Class extends value \" + String(b) + \" is not a constructor or null\");\r\n extendStatics(d, b);\r\n function __() { this.constructor = d; }\r\n d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\r\n}\r\n\r\nexport var __assign = function() {\r\n __assign = Object.assign || function __assign(t) {\r\n for (var s, i = 1, n = arguments.length; i < n; i++) {\r\n s = arguments[i];\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];\r\n }\r\n return t;\r\n }\r\n return __assign.apply(this, arguments);\r\n}\r\n\r\nexport function __rest(s, e) {\r\n var t = {};\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\r\n t[p] = s[p];\r\n if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\r\n for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\r\n if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\r\n t[p[i]] = s[p[i]];\r\n }\r\n return t;\r\n}\r\n\r\nexport function __decorate(decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n}\r\n\r\nexport function __param(paramIndex, decorator) {\r\n return function (target, key) { decorator(target, key, paramIndex); }\r\n}\r\n\r\nexport function __esDecorate(ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {\r\n function accept(f) { if (f !== void 0 && typeof f !== \"function\") throw new TypeError(\"Function expected\"); return f; }\r\n var kind = contextIn.kind, key = kind === \"getter\" ? \"get\" : kind === \"setter\" ? \"set\" : \"value\";\r\n var target = !descriptorIn && ctor ? contextIn[\"static\"] ? ctor : ctor.prototype : null;\r\n var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});\r\n var _, done = false;\r\n for (var i = decorators.length - 1; i >= 0; i--) {\r\n var context = {};\r\n for (var p in contextIn) context[p] = p === \"access\" ? {} : contextIn[p];\r\n for (var p in contextIn.access) context.access[p] = contextIn.access[p];\r\n context.addInitializer = function (f) { if (done) throw new TypeError(\"Cannot add initializers after decoration has completed\"); extraInitializers.push(accept(f || null)); };\r\n var result = (0, decorators[i])(kind === \"accessor\" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);\r\n if (kind === \"accessor\") {\r\n if (result === void 0) continue;\r\n if (result === null || typeof result !== \"object\") throw new TypeError(\"Object expected\");\r\n if (_ = accept(result.get)) descriptor.get = _;\r\n if (_ = accept(result.set)) descriptor.set = _;\r\n if (_ = accept(result.init)) initializers.unshift(_);\r\n }\r\n else if (_ = accept(result)) {\r\n if (kind === \"field\") initializers.unshift(_);\r\n else descriptor[key] = _;\r\n }\r\n }\r\n if (target) Object.defineProperty(target, contextIn.name, descriptor);\r\n done = true;\r\n};\r\n\r\nexport function __runInitializers(thisArg, initializers, value) {\r\n var useValue = arguments.length > 2;\r\n for (var i = 0; i < initializers.length; i++) {\r\n value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);\r\n }\r\n return useValue ? value : void 0;\r\n};\r\n\r\nexport function __propKey(x) {\r\n return typeof x === \"symbol\" ? x : \"\".concat(x);\r\n};\r\n\r\nexport function __setFunctionName(f, name, prefix) {\r\n if (typeof name === \"symbol\") name = name.description ? \"[\".concat(name.description, \"]\") : \"\";\r\n return Object.defineProperty(f, \"name\", { configurable: true, value: prefix ? \"\".concat(prefix, \" \", name) : name });\r\n};\r\n\r\nexport function __metadata(metadataKey, metadataValue) {\r\n if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(metadataKey, metadataValue);\r\n}\r\n\r\nexport function __awaiter(thisArg, _arguments, P, generator) {\r\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\r\n return new (P || (P = Promise))(function (resolve, reject) {\r\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\r\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\r\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\r\n step((generator = generator.apply(thisArg, _arguments || [])).next());\r\n });\r\n}\r\n\r\nexport function __generator(thisArg, body) {\r\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\r\n return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\r\n function verb(n) { return function (v) { return step([n, v]); }; }\r\n function step(op) {\r\n if (f) throw new TypeError(\"Generator is already executing.\");\r\n while (g && (g = 0, op[0] && (_ = 0)), _) try {\r\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\r\n if (y = 0, t) op = [op[0] & 2, t.value];\r\n switch (op[0]) {\r\n case 0: case 1: t = op; break;\r\n case 4: _.label++; return { value: op[1], done: false };\r\n case 5: _.label++; y = op[1]; op = [0]; continue;\r\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\r\n default:\r\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\r\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\r\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\r\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\r\n if (t[2]) _.ops.pop();\r\n _.trys.pop(); continue;\r\n }\r\n op = body.call(thisArg, _);\r\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\r\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\r\n }\r\n}\r\n\r\nexport var __createBinding = Object.create ? (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n var desc = Object.getOwnPropertyDescriptor(m, k);\r\n if (!desc || (\"get\" in desc ? !m.__esModule : desc.writable || desc.configurable)) {\r\n desc = { enumerable: true, get: function() { return m[k]; } };\r\n }\r\n Object.defineProperty(o, k2, desc);\r\n}) : (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n o[k2] = m[k];\r\n});\r\n\r\nexport function __exportStar(m, o) {\r\n for (var p in m) if (p !== \"default\" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p);\r\n}\r\n\r\nexport function __values(o) {\r\n var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\r\n if (m) return m.call(o);\r\n if (o && typeof o.length === \"number\") return {\r\n next: function () {\r\n if (o && i >= o.length) o = void 0;\r\n return { value: o && o[i++], done: !o };\r\n }\r\n };\r\n throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\r\n}\r\n\r\nexport function __read(o, n) {\r\n var m = typeof Symbol === \"function\" && o[Symbol.iterator];\r\n if (!m) return o;\r\n var i = m.call(o), r, ar = [], e;\r\n try {\r\n while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\r\n }\r\n catch (error) { e = { error: error }; }\r\n finally {\r\n try {\r\n if (r && !r.done && (m = i[\"return\"])) m.call(i);\r\n }\r\n finally { if (e) throw e.error; }\r\n }\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spread() {\r\n for (var ar = [], i = 0; i < arguments.length; i++)\r\n ar = ar.concat(__read(arguments[i]));\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spreadArrays() {\r\n for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\r\n for (var r = Array(s), k = 0, i = 0; i < il; i++)\r\n for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\r\n r[k] = a[j];\r\n return r;\r\n}\r\n\r\nexport function __spreadArray(to, from, pack) {\r\n if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {\r\n if (ar || !(i in from)) {\r\n if (!ar) ar = Array.prototype.slice.call(from, 0, i);\r\n ar[i] = from[i];\r\n }\r\n }\r\n return to.concat(ar || Array.prototype.slice.call(from));\r\n}\r\n\r\nexport function __await(v) {\r\n return this instanceof __await ? (this.v = v, this) : new __await(v);\r\n}\r\n\r\nexport function __asyncGenerator(thisArg, _arguments, generator) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var g = generator.apply(thisArg, _arguments || []), i, q = [];\r\n return i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i;\r\n function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }\r\n function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\r\n function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\r\n function fulfill(value) { resume(\"next\", value); }\r\n function reject(value) { resume(\"throw\", value); }\r\n function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\r\n}\r\n\r\nexport function __asyncDelegator(o) {\r\n var i, p;\r\n return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\r\n function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: false } : f ? f(v) : v; } : f; }\r\n}\r\n\r\nexport function __asyncValues(o) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var m = o[Symbol.asyncIterator], i;\r\n return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\r\n function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\r\n function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\r\n}\r\n\r\nexport function __makeTemplateObject(cooked, raw) {\r\n if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\r\n return cooked;\r\n};\r\n\r\nvar __setModuleDefault = Object.create ? (function(o, v) {\r\n Object.defineProperty(o, \"default\", { enumerable: true, value: v });\r\n}) : function(o, v) {\r\n o[\"default\"] = v;\r\n};\r\n\r\nexport function __importStar(mod) {\r\n if (mod && mod.__esModule) return mod;\r\n var result = {};\r\n if (mod != null) for (var k in mod) if (k !== \"default\" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);\r\n __setModuleDefault(result, mod);\r\n return result;\r\n}\r\n\r\nexport function __importDefault(mod) {\r\n return (mod && mod.__esModule) ? mod : { default: mod };\r\n}\r\n\r\nexport function __classPrivateFieldGet(receiver, state, kind, f) {\r\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a getter\");\r\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot read private member from an object whose class did not declare it\");\r\n return kind === \"m\" ? f : kind === \"a\" ? f.call(receiver) : f ? f.value : state.get(receiver);\r\n}\r\n\r\nexport function __classPrivateFieldSet(receiver, state, value, kind, f) {\r\n if (kind === \"m\") throw new TypeError(\"Private method is not writable\");\r\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a setter\");\r\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot write private member to an object whose class did not declare it\");\r\n return (kind === \"a\" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;\r\n}\r\n\r\nexport function __classPrivateFieldIn(state, receiver) {\r\n if (receiver === null || (typeof receiver !== \"object\" && typeof receiver !== \"function\")) throw new TypeError(\"Cannot use 'in' operator on non-object\");\r\n return typeof state === \"function\" ? receiver === state : state.has(receiver);\r\n}\r\n\r\nexport function __addDisposableResource(env, value, async) {\r\n if (value !== null && value !== void 0) {\r\n if (typeof value !== \"object\" && typeof value !== \"function\") throw new TypeError(\"Object expected.\");\r\n var dispose;\r\n if (async) {\r\n if (!Symbol.asyncDispose) throw new TypeError(\"Symbol.asyncDispose is not defined.\");\r\n dispose = value[Symbol.asyncDispose];\r\n }\r\n if (dispose === void 0) {\r\n if (!Symbol.dispose) throw new TypeError(\"Symbol.dispose is not defined.\");\r\n dispose = value[Symbol.dispose];\r\n }\r\n if (typeof dispose !== \"function\") throw new TypeError(\"Object not disposable.\");\r\n env.stack.push({ value: value, dispose: dispose, async: async });\r\n }\r\n else if (async) {\r\n env.stack.push({ async: true });\r\n }\r\n return value;\r\n}\r\n\r\nvar _SuppressedError = typeof SuppressedError === \"function\" ? SuppressedError : function (error, suppressed, message) {\r\n var e = new Error(message);\r\n return e.name = \"SuppressedError\", e.error = error, e.suppressed = suppressed, e;\r\n};\r\n\r\nexport function __disposeResources(env) {\r\n function fail(e) {\r\n env.error = env.hasError ? new _SuppressedError(e, env.error, \"An error was suppressed during disposal.\") : e;\r\n env.hasError = true;\r\n }\r\n function next() {\r\n while (env.stack.length) {\r\n var rec = env.stack.pop();\r\n try {\r\n var result = rec.dispose && rec.dispose.call(rec.value);\r\n if (rec.async) return Promise.resolve(result).then(next, function(e) { fail(e); return next(); });\r\n }\r\n catch (e) {\r\n fail(e);\r\n }\r\n }\r\n if (env.hasError) throw env.error;\r\n }\r\n return next();\r\n}\r\n\r\nexport default {\r\n __extends: __extends,\r\n __assign: __assign,\r\n __rest: __rest,\r\n __decorate: __decorate,\r\n __param: __param,\r\n __metadata: __metadata,\r\n __awaiter: __awaiter,\r\n __generator: __generator,\r\n __createBinding: __createBinding,\r\n __exportStar: __exportStar,\r\n __values: __values,\r\n __read: __read,\r\n __spread: __spread,\r\n __spreadArrays: __spreadArrays,\r\n __spreadArray: __spreadArray,\r\n __await: __await,\r\n __asyncGenerator: __asyncGenerator,\r\n __asyncDelegator: __asyncDelegator,\r\n __asyncValues: __asyncValues,\r\n __makeTemplateObject: __makeTemplateObject,\r\n __importStar: __importStar,\r\n __importDefault: __importDefault,\r\n __classPrivateFieldGet: __classPrivateFieldGet,\r\n __classPrivateFieldSet: __classPrivateFieldSet,\r\n __classPrivateFieldIn: __classPrivateFieldIn,\r\n __addDisposableResource: __addDisposableResource,\r\n __disposeResources: __disposeResources,\r\n};\r\n","'use strict';\n\nmodule.exports = Point;\n\n/**\n * A standalone point geometry with useful accessor, comparison, and\n * modification methods.\n *\n * @class Point\n * @param {Number} x the x-coordinate. this could be longitude or screen\n * pixels, or any other sort of unit.\n * @param {Number} y the y-coordinate. this could be latitude or screen\n * pixels, or any other sort of unit.\n * @example\n * var point = new Point(-77, 38);\n */\nfunction Point(x, y) {\n this.x = x;\n this.y = y;\n}\n\nPoint.prototype = {\n\n /**\n * Clone this point, returning a new point that can be modified\n * without affecting the old one.\n * @return {Point} the clone\n */\n clone: function() { return new Point(this.x, this.y); },\n\n /**\n * Add this point's x & y coordinates to another point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n add: function(p) { return this.clone()._add(p); },\n\n /**\n * Subtract this point's x & y coordinates to from point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n sub: function(p) { return this.clone()._sub(p); },\n\n /**\n * Multiply this point's x & y coordinates by point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n multByPoint: function(p) { return this.clone()._multByPoint(p); },\n\n /**\n * Divide this point's x & y coordinates by point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n divByPoint: function(p) { return this.clone()._divByPoint(p); },\n\n /**\n * Multiply this point's x & y coordinates by a factor,\n * yielding a new point.\n * @param {Point} k factor\n * @return {Point} output point\n */\n mult: function(k) { return this.clone()._mult(k); },\n\n /**\n * Divide this point's x & y coordinates by a factor,\n * yielding a new point.\n * @param {Point} k factor\n * @return {Point} output point\n */\n div: function(k) { return this.clone()._div(k); },\n\n /**\n * Rotate this point around the 0, 0 origin by an angle a,\n * given in radians\n * @param {Number} a angle to rotate around, in radians\n * @return {Point} output point\n */\n rotate: function(a) { return this.clone()._rotate(a); },\n\n /**\n * Rotate this point around p point by an angle a,\n * given in radians\n * @param {Number} a angle to rotate around, in radians\n * @param {Point} p Point to rotate around\n * @return {Point} output point\n */\n rotateAround: function(a,p) { return this.clone()._rotateAround(a,p); },\n\n /**\n * Multiply this point by a 4x1 transformation matrix\n * @param {Array} m transformation matrix\n * @return {Point} output point\n */\n matMult: function(m) { return this.clone()._matMult(m); },\n\n /**\n * Calculate this point but as a unit vector from 0, 0, meaning\n * that the distance from the resulting point to the 0, 0\n * coordinate will be equal to 1 and the angle from the resulting\n * point to the 0, 0 coordinate will be the same as before.\n * @return {Point} unit vector point\n */\n unit: function() { return this.clone()._unit(); },\n\n /**\n * Compute a perpendicular point, where the new y coordinate\n * is the old x coordinate and the new x coordinate is the old y\n * coordinate multiplied by -1\n * @return {Point} perpendicular point\n */\n perp: function() { return this.clone()._perp(); },\n\n /**\n * Return a version of this point with the x & y coordinates\n * rounded to integers.\n * @return {Point} rounded point\n */\n round: function() { return this.clone()._round(); },\n\n /**\n * Return the magitude of this point: this is the Euclidean\n * distance from the 0, 0 coordinate to this point's x and y\n * coordinates.\n * @return {Number} magnitude\n */\n mag: function() {\n return Math.sqrt(this.x * this.x + this.y * this.y);\n },\n\n /**\n * Judge whether this point is equal to another point, returning\n * true or false.\n * @param {Point} other the other point\n * @return {boolean} whether the points are equal\n */\n equals: function(other) {\n return this.x === other.x &&\n this.y === other.y;\n },\n\n /**\n * Calculate the distance from this point to another point\n * @param {Point} p the other point\n * @return {Number} distance\n */\n dist: function(p) {\n return Math.sqrt(this.distSqr(p));\n },\n\n /**\n * Calculate the distance from this point to another point,\n * without the square root step. Useful if you're comparing\n * relative distances.\n * @param {Point} p the other point\n * @return {Number} distance\n */\n distSqr: function(p) {\n var dx = p.x - this.x,\n dy = p.y - this.y;\n return dx * dx + dy * dy;\n },\n\n /**\n * Get the angle from the 0, 0 coordinate to this point, in radians\n * coordinates.\n * @return {Number} angle\n */\n angle: function() {\n return Math.atan2(this.y, this.x);\n },\n\n /**\n * Get the angle from this point to another point, in radians\n * @param {Point} b the other point\n * @return {Number} angle\n */\n angleTo: function(b) {\n return Math.atan2(this.y - b.y, this.x - b.x);\n },\n\n /**\n * Get the angle between this point and another point, in radians\n * @param {Point} b the other point\n * @return {Number} angle\n */\n angleWith: function(b) {\n return this.angleWithSep(b.x, b.y);\n },\n\n /*\n * Find the angle of the two vectors, solving the formula for\n * the cross product a x b = |a||b|sin(θ) for θ.\n * @param {Number} x the x-coordinate\n * @param {Number} y the y-coordinate\n * @return {Number} the angle in radians\n */\n angleWithSep: function(x, y) {\n return Math.atan2(\n this.x * y - this.y * x,\n this.x * x + this.y * y);\n },\n\n _matMult: function(m) {\n var x = m[0] * this.x + m[1] * this.y,\n y = m[2] * this.x + m[3] * this.y;\n this.x = x;\n this.y = y;\n return this;\n },\n\n _add: function(p) {\n this.x += p.x;\n this.y += p.y;\n return this;\n },\n\n _sub: function(p) {\n this.x -= p.x;\n this.y -= p.y;\n return this;\n },\n\n _mult: function(k) {\n this.x *= k;\n this.y *= k;\n return this;\n },\n\n _div: function(k) {\n this.x /= k;\n this.y /= k;\n return this;\n },\n\n _multByPoint: function(p) {\n this.x *= p.x;\n this.y *= p.y;\n return this;\n },\n\n _divByPoint: function(p) {\n this.x /= p.x;\n this.y /= p.y;\n return this;\n },\n\n _unit: function() {\n this._div(this.mag());\n return this;\n },\n\n _perp: function() {\n var y = this.y;\n this.y = this.x;\n this.x = -y;\n return this;\n },\n\n _rotate: function(angle) {\n var cos = Math.cos(angle),\n sin = Math.sin(angle),\n x = cos * this.x - sin * this.y,\n y = sin * this.x + cos * this.y;\n this.x = x;\n this.y = y;\n return this;\n },\n\n _rotateAround: function(angle, p) {\n var cos = Math.cos(angle),\n sin = Math.sin(angle),\n x = p.x + cos * (this.x - p.x) - sin * (this.y - p.y),\n y = p.y + sin * (this.x - p.x) + cos * (this.y - p.y);\n this.x = x;\n this.y = y;\n return this;\n },\n\n _round: function() {\n this.x = Math.round(this.x);\n this.y = Math.round(this.y);\n return this;\n }\n};\n\n/**\n * Construct a point from an array if necessary, otherwise if the input\n * is already a Point, or an unknown type, return it unchanged\n * @param {Array|Point|*} a any kind of input value\n * @return {Point} constructed point, or passed-through value.\n * @example\n * // this\n * var point = Point.convert([0, 1]);\n * // is equivalent to\n * var point = new Point(0, 1);\n */\nPoint.convert = function (a) {\n if (a instanceof Point) {\n return a;\n }\n if (Array.isArray(a)) {\n return new Point(a[0], a[1]);\n }\n return a;\n};\n","'use strict';\n\nmodule.exports = UnitBezier;\n\nfunction UnitBezier(p1x, p1y, p2x, p2y) {\n // Calculate the polynomial coefficients, implicit first and last control points are (0,0) and (1,1).\n this.cx = 3.0 * p1x;\n this.bx = 3.0 * (p2x - p1x) - this.cx;\n this.ax = 1.0 - this.cx - this.bx;\n\n this.cy = 3.0 * p1y;\n this.by = 3.0 * (p2y - p1y) - this.cy;\n this.ay = 1.0 - this.cy - this.by;\n\n this.p1x = p1x;\n this.p1y = p1y;\n this.p2x = p2x;\n this.p2y = p2y;\n}\n\nUnitBezier.prototype = {\n sampleCurveX: function (t) {\n // `ax t^3 + bx t^2 + cx t' expanded using Horner's rule.\n return ((this.ax * t + this.bx) * t + this.cx) * t;\n },\n\n sampleCurveY: function (t) {\n return ((this.ay * t + this.by) * t + this.cy) * t;\n },\n\n sampleCurveDerivativeX: function (t) {\n return (3.0 * this.ax * t + 2.0 * this.bx) * t + this.cx;\n },\n\n solveCurveX: function (x, epsilon) {\n if (epsilon === undefined) epsilon = 1e-6;\n\n if (x < 0.0) return 0.0;\n if (x > 1.0) return 1.0;\n\n var t = x;\n\n // First try a few iterations of Newton's method - normally very fast.\n for (var i = 0; i < 8; i++) {\n var x2 = this.sampleCurveX(t) - x;\n if (Math.abs(x2) < epsilon) return t;\n\n var d2 = this.sampleCurveDerivativeX(t);\n if (Math.abs(d2) < 1e-6) break;\n\n t = t - x2 / d2;\n }\n\n // Fall back to the bisection method for reliability.\n var t0 = 0.0;\n var t1 = 1.0;\n t = x;\n\n for (i = 0; i < 20; i++) {\n x2 = this.sampleCurveX(t);\n if (Math.abs(x2 - x) < epsilon) break;\n\n if (x > x2) {\n t0 = t;\n } else {\n t1 = t;\n }\n\n t = (t1 - t0) * 0.5 + t0;\n }\n\n return t;\n },\n\n solve: function (x, epsilon) {\n return this.sampleCurveY(this.solveCurveX(x, epsilon));\n }\n};\n","let supportsOffscreenCanvas: boolean;\n\nexport function offscreenCanvasSupported(): boolean {\n if (supportsOffscreenCanvas == null) {\n supportsOffscreenCanvas = typeof OffscreenCanvas !== 'undefined' &&\n new OffscreenCanvas(1, 1).getContext('2d') &&\n typeof createImageBitmap === 'function';\n }\n\n return supportsOffscreenCanvas;\n}\n","import {offscreenCanvasSupported} from './offscreen_canvas_supported';\n\nlet offscreenCanvasDistorted: boolean;\n\n/**\n * Some browsers don't return the exact pixels from a canvas to prevent user fingerprinting (see #3185).\n * This function writes pixels to an OffscreenCanvas and reads them back using getImageData, returning false\n * if they don't match.\n *\n * @returns true if the browser supports OffscreenCanvas but it distorts getImageData results, false otherwise.\n */\nexport function isOffscreenCanvasDistorted(): boolean {\n if (offscreenCanvasDistorted == null) {\n offscreenCanvasDistorted = false;\n if (offscreenCanvasSupported()) {\n const size = 5;\n const canvas = new OffscreenCanvas(size, size);\n const context = canvas.getContext('2d', {willReadFrequently: true});\n if (context) {\n // fill each pixel with an RGB value that should make the byte at index i equal to i (except alpha channel):\n // [0, 1, 2, 255, 4, 5, 6, 255, 8, 9, 10, 255, ...]\n for (let i = 0; i < size * size; i++) {\n const base = i * 4;\n context.fillStyle = `rgb(${base},${base + 1},${base + 2})`;\n context.fillRect(i % size, Math.floor(i / size), 1, 1);\n }\n const data = context.getImageData(0, 0, size, size).data;\n for (let i = 0; i < size * size * 4; i++) {\n if (i % 4 !== 3 && data[i] !== i) {\n offscreenCanvasDistorted = true;\n break;\n }\n }\n }\n }\n }\n\n return offscreenCanvasDistorted || false;\n}\n","import Point from '@mapbox/point-geometry';\nimport UnitBezier from '@mapbox/unitbezier';\nimport type {Callback} from '../types/callback';\nimport {isOffscreenCanvasDistorted} from './offscreen_canvas_distorted';\nimport type {Size} from './image';\n\n/**\n * Given a value `t` that varies between 0 and 1, return\n * an interpolation function that eases between 0 and 1 in a pleasing\n * cubic in-out fashion.\n */\nexport function easeCubicInOut(t: number): number {\n if (t <= 0) return 0;\n if (t >= 1) return 1;\n const t2 = t * t,\n t3 = t2 * t;\n return 4 * (t < 0.5 ? t3 : 3 * (t - t2) + t3 - 0.75);\n}\n\n/**\n * Given given (x, y), (x1, y1) control points for a bezier curve,\n * return a function that interpolates along that curve.\n *\n * @param p1x - control point 1 x coordinate\n * @param p1y - control point 1 y coordinate\n * @param p2x - control point 2 x coordinate\n * @param p2y - control point 2 y coordinate\n */\nexport function bezier(p1x: number, p1y: number, p2x: number, p2y: number): (t: number) => number {\n const bezier = new UnitBezier(p1x, p1y, p2x, p2y);\n return function(t: number) {\n return bezier.solve(t);\n };\n}\n\n/**\n * A default bezier-curve powered easing function with\n * control points (0.25, 0.1) and (0.25, 1)\n */\nexport const defaultEasing = bezier(0.25, 0.1, 0.25, 1);\n\n/**\n * constrain n to the given range via min + max\n *\n * @param n - value\n * @param min - the minimum value to be returned\n * @param max - the maximum value to be returned\n * @returns the clamped value\n */\nexport function clamp(n: number, min: number, max: number): number {\n return Math.min(max, Math.max(min, n));\n}\n\n/**\n * constrain n to the given range, excluding the minimum, via modular arithmetic\n *\n * @param n - value\n * @param min - the minimum value to be returned, exclusive\n * @param max - the maximum value to be returned, inclusive\n * @returns constrained number\n */\nexport function wrap(n: number, min: number, max: number): number {\n const d = max - min;\n const w = ((n - min) % d + d) % d + min;\n return (w === min) ? max : w;\n}\n\n/**\n * Call an asynchronous function on an array of arguments,\n * calling `callback` with the completed results of all calls.\n *\n * @param array - input to each call of the async function.\n * @param fn - an async function with signature (data, callback)\n * @param callback - a callback run after all async work is done.\n * called with an array, containing the results of each async call.\n */\nexport function asyncAll(\n array: Array,\n fn: (item: Item, fnCallback: Callback) => void,\n callback: Callback>\n) {\n if (!array.length) { return callback(null, []); }\n let remaining = array.length;\n const results = new Array(array.length);\n let error = null;\n array.forEach((item, i) => {\n fn(item, (err, result) => {\n if (err) error = err;\n results[i] = (result as any as Result); // https://github.com/facebook/flow/issues/2123\n if (--remaining === 0) callback(error, results);\n });\n });\n}\n\n/**\n * Compute the difference between the keys in one object and the keys\n * in another object.\n *\n * @returns keys difference\n */\nexport function keysDifference(\n obj: {[key: string]: S},\n other: {[key: string]: T}\n): Array {\n const difference = [];\n for (const i in obj) {\n if (!(i in other)) {\n difference.push(i);\n }\n }\n return difference;\n}\n\n/**\n * Given a destination object and optionally many source objects,\n * copy all properties from the source objects into the destination.\n * The last source object given overrides properties from previous\n * source objects.\n *\n * @param dest - destination object\n * @param sources - sources from which properties are pulled\n */\nexport function extend(dest: any, ...sources: Array): any {\n for (const src of sources) {\n for (const k in src) {\n dest[k] = src[k];\n }\n }\n return dest;\n}\n\n/**\n * Given an object and a number of properties as strings, return version\n * of that object with only those properties.\n *\n * @param src - the object\n * @param properties - an array of property names chosen\n * to appear on the resulting object.\n * @returns object with limited properties.\n * @example\n * ```ts\n * let foo = { name: 'Charlie', age: 10 };\n * let justName = pick(foo, ['name']); // justName = { name: 'Charlie' }\n * ```\n */\nexport function pick(src: any, properties: Array): any {\n const result = {};\n for (let i = 0; i < properties.length; i++) {\n const k = properties[i];\n if (k in src) {\n result[k] = src[k];\n }\n }\n return result;\n}\n\nlet id = 1;\n\n/**\n * Return a unique numeric id, starting at 1 and incrementing with\n * each call.\n *\n * @returns unique numeric id.\n */\nexport function uniqueId(): number {\n return id++;\n}\n\n/**\n * Return whether a given value is a power of two\n */\nexport function isPowerOfTwo(value: number): boolean {\n return (Math.log(value) / Math.LN2) % 1 === 0;\n}\n\n/**\n * Return the next power of two, or the input value if already a power of two\n */\nexport function nextPowerOfTwo(value: number): number {\n if (value <= 1) return 1;\n return Math.pow(2, Math.ceil(Math.log(value) / Math.LN2));\n}\n\n/**\n * Create an object by mapping all the values of an existing object while\n * preserving their keys.\n */\nexport function mapObject(input: any, iterator: Function, context?: any): any {\n const output = {};\n for (const key in input) {\n output[key] = iterator.call(context || this, input[key], key, input);\n }\n return output;\n}\n\n/**\n * Create an object by filtering out values of an existing object.\n */\nexport function filterObject(input: any, iterator: Function, context?: any): any {\n const output = {};\n for (const key in input) {\n if (iterator.call(context || this, input[key], key, input)) {\n output[key] = input[key];\n }\n }\n return output;\n}\n\n/**\n * Deeply compares two object literals.\n * @param a - first object literal to be compared\n * @param b - second object literal to be compared\n * @returns true if the two object literals are deeply equal, false otherwise\n */\nexport function deepEqual(a?: unknown | null, b?: unknown | null): boolean {\n if (Array.isArray(a)) {\n if (!Array.isArray(b) || a.length !== b.length) return false;\n for (let i = 0; i < a.length; i++) {\n if (!deepEqual(a[i], b[i])) return false;\n }\n return true;\n }\n if (typeof a === 'object' && a !== null && b !== null) {\n if (!(typeof b === 'object')) return false;\n const keys = Object.keys(a);\n if (keys.length !== Object.keys(b).length) return false;\n for (const key in a) {\n if (!deepEqual(a[key], b[key])) return false;\n }\n return true;\n }\n return a === b;\n}\n\n/**\n * Deeply clones two objects.\n */\nexport function clone(input: T): T {\n if (Array.isArray(input)) {\n return input.map(clone) as any as T;\n } else if (typeof input === 'object' && input) {\n return mapObject(input, clone) as any as T;\n } else {\n return input;\n }\n}\n\n/**\n * Check if two arrays have at least one common element.\n */\nexport function arraysIntersect(a: Array, b: Array): boolean {\n for (let l = 0; l < a.length; l++) {\n if (b.indexOf(a[l]) >= 0) return true;\n }\n return false;\n}\n\n/**\n * Print a warning message to the console and ensure duplicate warning messages\n * are not printed.\n */\nconst warnOnceHistory: {[key: string]: boolean} = {};\n\nexport function warnOnce(message: string): void {\n if (!warnOnceHistory[message]) {\n // console isn't defined in some WebWorkers, see #2558\n if (typeof console !== 'undefined') console.warn(message);\n warnOnceHistory[message] = true;\n }\n}\n\n/**\n * Indicates if the provided Points are in a counter clockwise (true) or clockwise (false) order\n *\n * @returns true for a counter clockwise set of points\n */\n// http://bryceboe.com/2006/10/23/line-segment-intersection-algorithm/\nexport function isCounterClockwise(a: Point, b: Point, c: Point): boolean {\n return (c.y - a.y) * (b.x - a.x) > (b.y - a.y) * (c.x - a.x);\n}\n\n/**\n * For two lines a and b in 2d space, defined by any two points along the lines,\n * find the intersection point, or return null if the lines are parallel\n *\n * @param a1 - First point on line a\n * @param a2 - Second point on line a\n * @param b1 - First point on line b\n * @param b2 - Second point on line b\n *\n * @returns the intersection point of the two lines or null if they are parallel\n */\nexport function findLineIntersection(a1: Point, a2: Point, b1: Point, b2: Point): Point | null {\n const aDeltaY = a2.y - a1.y;\n const aDeltaX = a2.x - a1.x;\n const bDeltaY = b2.y - b1.y;\n const bDeltaX = b2.x - b1.x;\n\n const denominator = (bDeltaY * aDeltaX) - (bDeltaX * aDeltaY);\n\n if (denominator === 0) {\n // Lines are parallel\n return null;\n }\n\n const originDeltaY = a1.y - b1.y;\n const originDeltaX = a1.x - b1.x;\n const aInterpolation = (bDeltaX * originDeltaY - bDeltaY * originDeltaX) / denominator;\n\n // Find intersection by projecting out from origin of first segment\n return new Point(a1.x + (aInterpolation * aDeltaX), a1.y + (aInterpolation * aDeltaY));\n}\n\n/**\n * Returns the signed area for the polygon ring. Positive areas are exterior rings and\n * have a clockwise winding. Negative areas are interior rings and have a counter clockwise\n * ordering.\n *\n * @param ring - Exterior or interior ring\n */\nexport function calculateSignedArea(ring: Array): number {\n let sum = 0;\n for (let i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) {\n p1 = ring[i];\n p2 = ring[j];\n sum += (p2.x - p1.x) * (p1.y + p2.y);\n }\n return sum;\n}\n\n/**\n * Detects closed polygons, first + last point are equal\n *\n * @param points - array of points\n * @returns `true` if the points are a closed polygon\n */\nexport function isClosedPolygon(points: Array): boolean {\n // If it is 2 points that are the same then it is a point\n // If it is 3 points with start and end the same then it is a line\n if (points.length < 4)\n return false;\n\n const p1 = points[0];\n const p2 = points[points.length - 1];\n\n if (Math.abs(p1.x - p2.x) > 0 ||\n Math.abs(p1.y - p2.y) > 0) {\n return false;\n }\n\n // polygon simplification can produce polygons with zero area and more than 3 points\n return Math.abs(calculateSignedArea(points)) > 0.01;\n}\n\n/**\n * Converts spherical coordinates to cartesian coordinates.\n *\n * @param spherical - Spherical coordinates, in [radial, azimuthal, polar]\n * @returns cartesian coordinates in [x, y, z]\n */\n\nexport function sphericalToCartesian([r, azimuthal, polar]: [number, number, number]): {\n x: number;\n y: number;\n z: number;\n} {\n // We abstract \"north\"/\"up\" (compass-wise) to be 0° when really this is 90° (π/2):\n // correct for that here\n azimuthal += 90;\n\n // Convert azimuthal and polar angles to radians\n azimuthal *= Math.PI / 180;\n polar *= Math.PI / 180;\n\n return {\n x: r * Math.cos(azimuthal) * Math.sin(polar),\n y: r * Math.sin(azimuthal) * Math.sin(polar),\n z: r * Math.cos(polar)\n };\n}\n\n/**\n * Returns true if the when run in the web-worker context.\n *\n * @returns `true` if the when run in the web-worker context.\n */\nexport function isWorker(): boolean {\n // @ts-ignore\n return typeof WorkerGlobalScope !== 'undefined' && typeof self !== 'undefined' && self instanceof WorkerGlobalScope;\n}\n\n/**\n * Parses data from 'Cache-Control' headers.\n *\n * @param cacheControl - Value of 'Cache-Control' header\n * @returns object containing parsed header info.\n */\n\nexport function parseCacheControl(cacheControl: string): any {\n // Taken from [Wreck](https://github.com/hapijs/wreck)\n const re = /(?:^|(?:\\s*\\,\\s*))([^\\x00-\\x20\\(\\)<>@\\,;\\:\\\\\"\\/\\[\\]\\?\\=\\{\\}\\x7F]+)(?:\\=(?:([^\\x00-\\x20\\(\\)<>@\\,;\\:\\\\\"\\/\\[\\]\\?\\=\\{\\}\\x7F]+)|(?:\\\"((?:[^\"\\\\]|\\\\.)*)\\\")))?/g;\n\n const header = {};\n cacheControl.replace(re, ($0, $1, $2, $3) => {\n const value = $2 || $3;\n header[$1] = value ? value.toLowerCase() : true;\n return '';\n });\n\n if (header['max-age']) {\n const maxAge = parseInt(header['max-age'], 10);\n if (isNaN(maxAge)) delete header['max-age'];\n else header['max-age'] = maxAge;\n }\n\n return header;\n}\n\nlet _isSafari = null;\n\n/**\n * Returns true when run in WebKit derived browsers.\n * This is used as a workaround for a memory leak in Safari caused by using Transferable objects to\n * transfer data between WebWorkers and the main thread.\n * https://github.com/mapbox/mapbox-gl-js/issues/8771\n *\n * This should be removed once the underlying Safari issue is fixed.\n *\n * @param scope - Since this function is used both on the main thread and WebWorker context,\n * let the calling scope pass in the global scope object.\n * @returns `true` when run in WebKit derived browsers.\n */\nexport function isSafari(scope: any): boolean {\n if (_isSafari == null) {\n const userAgent = scope.navigator ? scope.navigator.userAgent : null;\n _isSafari = !!scope.safari ||\n !!(userAgent && (/\\b(iPad|iPhone|iPod)\\b/.test(userAgent) || (!!userAgent.match('Safari') && !userAgent.match('Chrome'))));\n }\n return _isSafari;\n}\n\nexport function storageAvailable(type: string): boolean {\n try {\n const storage = window[type];\n storage.setItem('_mapbox_test_', 1);\n storage.removeItem('_mapbox_test_');\n return true;\n } catch (e) {\n return false;\n }\n}\n\n// The following methods are from https://developer.mozilla.org/en-US/docs/Web/API/WindowBase64/Base64_encoding_and_decoding#The_Unicode_Problem\n//Unicode compliant base64 encoder for strings\nexport function b64EncodeUnicode(str: string) {\n return btoa(\n encodeURIComponent(str).replace(/%([0-9A-F]{2})/g,\n (match, p1) => {\n return String.fromCharCode(Number('0x' + p1)); //eslint-disable-line\n }\n )\n );\n}\n\n// Unicode compliant decoder for base64-encoded strings\nexport function b64DecodeUnicode(str: string) {\n return decodeURIComponent(atob(str).split('').map((c) => {\n return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2); //eslint-disable-line\n }).join(''));\n}\n\nexport function isImageBitmap(image: any): image is ImageBitmap {\n return typeof ImageBitmap !== 'undefined' && image instanceof ImageBitmap;\n}\n\n/**\n * Converts an ArrayBuffer to an ImageBitmap.\n *\n * Used mostly for testing purposes only, because mocking libs don't know how to work with ArrayBuffers, but work\n * perfectly fine with ImageBitmaps. Might also be used for environments (other than testing) not supporting\n * ArrayBuffers.\n *\n * @param data - Data to convert\n * @param callback - A callback executed after the conversion is finished. Invoked with error (if any) as the first argument and resulting image bitmap (when no error) as the second\n */\nexport function arrayBufferToImageBitmap(data: ArrayBuffer, callback: (err?: Error | null, image?: ImageBitmap | null) => void) {\n const blob: Blob = new Blob([new Uint8Array(data)], {type: 'image/png'});\n createImageBitmap(blob).then((imgBitmap) => {\n callback(null, imgBitmap);\n }).catch((e) => {\n callback(new Error(`Could not load image because of ${e.message}. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.`));\n });\n}\n\nconst transparentPngUrl = '';\n\n/**\n * Converts an ArrayBuffer to an HTMLImageElement.\n *\n * Used mostly for testing purposes only, because mocking libs don't know how to work with ArrayBuffers, but work\n * perfectly fine with ImageBitmaps. Might also be used for environments (other than testing) not supporting\n * ArrayBuffers.\n *\n * @param data - Data to convert\n * @param callback - A callback executed after the conversion is finished. Invoked with error (if any) as the first argument and resulting image element (when no error) as the second\n */\nexport function arrayBufferToImage(data: ArrayBuffer, callback: (err?: Error | null, image?: HTMLImageElement | null) => void) {\n const img: HTMLImageElement = new Image();\n img.onload = () => {\n callback(null, img);\n URL.revokeObjectURL(img.src);\n // prevent image dataURI memory leak in Safari;\n // but don't free the image immediately because it might be uploaded in the next frame\n // https://github.com/mapbox/mapbox-gl-js/issues/10226\n img.onload = null;\n window.requestAnimationFrame(() => { img.src = transparentPngUrl; });\n };\n img.onerror = () => callback(new Error('Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.'));\n const blob: Blob = new Blob([new Uint8Array(data)], {type: 'image/png'});\n img.src = data.byteLength ? URL.createObjectURL(blob) : transparentPngUrl;\n}\n\n/**\n * Computes the webcodecs VideoFrame API options to select a rectangle out of\n * an image and write it into the destination rectangle.\n *\n * Rect (x/y/width/height) select the overlapping rectangle from the source image\n * and layout (offset/stride) write that overlapping rectangle to the correct place\n * in the destination image.\n *\n * Offset is the byte offset in the dest image that the first pixel appears at\n * and stride is the number of bytes to the start of the next row:\n * ┌───────────┐\n * │ dest │\n * │ ┌───┼───────┐\n * │offset→│▓▓▓│ source│\n * │ │▓▓▓│ │\n * │ └───┼───────┘\n * │stride ⇠╌╌╌│\n * │╌╌╌╌╌╌→ │\n * └───────────┘\n *\n * @param image - source image containing a width and height attribute\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns the layout and rect options to pass into VideoFrame API\n */\nfunction computeVideoFrameParameters(image: Size, x: number, y: number, width: number, height: number): VideoFrameCopyToOptions {\n const destRowOffset = Math.max(-x, 0) * 4;\n const firstSourceRow = Math.max(0, y);\n const firstDestRow = firstSourceRow - y;\n const offset = firstDestRow * width * 4 + destRowOffset;\n const stride = width * 4;\n\n const sourceLeft = Math.max(0, x);\n const sourceTop = Math.max(0, y);\n const sourceRight = Math.min(image.width, x + width);\n const sourceBottom = Math.min(image.height, y + height);\n return {\n rect: {\n x: sourceLeft,\n y: sourceTop,\n width: sourceRight - sourceLeft,\n height: sourceBottom - sourceTop\n },\n layout: [{offset, stride}]\n };\n}\n\n/**\n * Reads pixels from an ImageBitmap/Image/canvas using webcodec VideoFrame API.\n *\n * @param data - image, imagebitmap, or canvas to parse\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns a promise containing the parsed RGBA pixel values of the image, or the error if an error occurred\n */\nexport async function readImageUsingVideoFrame(\n image: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas,\n x: number, y: number, width: number, height: number\n): Promise {\n if (typeof VideoFrame === 'undefined') {\n throw new Error('VideoFrame not supported');\n }\n const frame = new VideoFrame(image, {timestamp: 0});\n try {\n const format = frame?.format;\n if (!format || !(format.startsWith('BGR') || format.startsWith('RGB'))) {\n throw new Error(`Unrecognized format ${format}`);\n }\n const swapBR = format.startsWith('BGR');\n const result = new Uint8ClampedArray(width * height * 4);\n await frame.copyTo(result, computeVideoFrameParameters(image, x, y, width, height));\n if (swapBR) {\n for (let i = 0; i < result.length; i += 4) {\n const tmp = result[i];\n result[i] = result[i + 2];\n result[i + 2] = tmp;\n }\n }\n return result;\n } finally {\n frame.close();\n }\n}\n\nlet offscreenCanvas: OffscreenCanvas;\nlet offscreenCanvasContext: OffscreenCanvasRenderingContext2D;\n\n/**\n * Reads pixels from an ImageBitmap/Image/canvas using OffscreenCanvas\n *\n * @param data - image, imagebitmap, or canvas to parse\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns a promise containing the parsed RGBA pixel values of the image, or the error if an error occurred\n */\nexport function readImageDataUsingOffscreenCanvas(\n imgBitmap: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas,\n x: number, y: number, width: number, height: number\n): Uint8ClampedArray {\n const origWidth = imgBitmap.width;\n const origHeight = imgBitmap.height;\n // Lazily initialize OffscreenCanvas\n if (!offscreenCanvas || !offscreenCanvasContext) {\n // Dem tiles are typically 256x256\n offscreenCanvas = new OffscreenCanvas(origWidth, origHeight);\n offscreenCanvasContext = offscreenCanvas.getContext('2d', {willReadFrequently: true});\n }\n\n offscreenCanvas.width = origWidth;\n offscreenCanvas.height = origHeight;\n\n offscreenCanvasContext.drawImage(imgBitmap, 0, 0, origWidth, origHeight);\n const imgData = offscreenCanvasContext.getImageData(x, y, width, height);\n offscreenCanvasContext.clearRect(0, 0, origWidth, origHeight);\n return imgData.data;\n}\n\n/**\n * Reads RGBA pixels from an preferring OffscreenCanvas, but falling back to VideoFrame if supported and\n * the browser is mangling OffscreenCanvas getImageData results.\n *\n * @param data - image, imagebitmap, or canvas to parse\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns a promise containing the parsed RGBA pixel values of the image\n */\nexport async function getImageData(\n image: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas,\n x: number, y: number, width: number, height: number\n): Promise {\n if (isOffscreenCanvasDistorted()) {\n try {\n return await readImageUsingVideoFrame(image, x, y, width, height);\n } catch (e) {\n // fall back to OffscreenCanvas\n }\n }\n return readImageDataUsingOffscreenCanvas(image, x, y, width, height);\n}\n","import type {Cancelable} from '../types/cancelable';\n\nconst now = typeof performance !== 'undefined' && performance && performance.now ?\n performance.now.bind(performance) :\n Date.now.bind(Date);\n\nlet linkEl;\n\nlet reducedMotionQuery: MediaQueryList;\n\n/** */\nexport const browser = {\n /**\n * Provides a function that outputs milliseconds: either performance.now()\n * or a fallback to Date.now()\n */\n now,\n\n frame(fn: (paintStartTimestamp: number) => void): Cancelable {\n const frame = requestAnimationFrame(fn);\n return {cancel: () => cancelAnimationFrame(frame)};\n },\n\n getImageData(img: HTMLImageElement | ImageBitmap, padding: number = 0): ImageData {\n const context = this.getImageCanvasContext(img);\n return context.getImageData(-padding, -padding, img.width as number + 2 * padding, img.height as number + 2 * padding);\n },\n\n getImageCanvasContext(img: HTMLImageElement | ImageBitmap): CanvasRenderingContext2D {\n const canvas = window.document.createElement('canvas') as HTMLCanvasElement;\n const context = canvas.getContext('2d', {willReadFrequently: true});\n if (!context) {\n throw new Error('failed to create canvas 2d context');\n }\n canvas.width = img.width as number;\n canvas.height = img.height as number;\n context.drawImage(img, 0, 0, img.width as number, img.height as number);\n return context;\n },\n\n resolveURL(path: string) {\n if (!linkEl) linkEl = document.createElement('a');\n linkEl.href = path;\n return linkEl.href;\n },\n\n hardwareConcurrency: typeof navigator !== 'undefined' && navigator.hardwareConcurrency || 4,\n\n get prefersReducedMotion(): boolean {\n // In case your test crashes when checking matchMedia, call setMatchMedia from 'src/util/test/util'\n if (!matchMedia) return false;\n //Lazily initialize media query\n if (reducedMotionQuery == null) {\n reducedMotionQuery = matchMedia('(prefers-reduced-motion: reduce)');\n }\n return reducedMotionQuery.matches;\n },\n};\n","import type {Cancelable} from '../types/cancelable';\nimport type {RequestParameters, ResponseCallback} from './ajax';\n\n/**\n * This is a global config object used to store the configuration\n * It is available in the workers as well.\n * Only serializable data should be stored in it.\n */\ntype Config = {\n MAX_PARALLEL_IMAGE_REQUESTS: number;\n MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME: number;\n MAX_TILE_CACHE_ZOOM_LEVELS: number;\n REGISTERED_PROTOCOLS: {[x: string]: (requestParameters: RequestParameters, callback: ResponseCallback) => Cancelable};\n WORKER_URL: string;\n};\n\nexport const config: Config = {\n MAX_PARALLEL_IMAGE_REQUESTS: 16,\n MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME: 8,\n MAX_TILE_CACHE_ZOOM_LEVELS: 5,\n REGISTERED_PROTOCOLS: {},\n WORKER_URL: ''\n};\n","import {extend, warnOnce, isWorker} from './util';\nimport {config} from './config';\n\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\n\n/**\n * A `RequestParameters` object to be returned from Map.options.transformRequest callbacks.\n * @example\n * ```ts\n * // use transformRequest to modify requests that begin with `http://myHost`\n * transformRequest: function(url, resourceType) {\n * if (resourceType === 'Source' && url.indexOf('http://myHost') > -1) {\n * return {\n * url: url.replace('http', 'https'),\n * headers: { 'my-custom-header': true },\n * credentials: 'include' // Include cookies for cross-origin requests\n * }\n * }\n * }\n * ```\n */\nexport type RequestParameters = {\n /**\n * The URL to be requested.\n */\n url: string;\n /**\n * The headers to be sent with the request.\n */\n headers?: any;\n /**\n * Request method `'GET' | 'POST' | 'PUT'`.\n */\n method?: 'GET' | 'POST' | 'PUT';\n /**\n * Request body.\n */\n body?: string;\n /**\n * Response body type to be returned `'string' | 'json' | 'arrayBuffer'`.\n */\n type?: 'string' | 'json' | 'arrayBuffer' | 'image';\n /**\n * `'same-origin'|'include'` Use 'include' to send cookies with cross-origin requests.\n */\n credentials?: 'same-origin' | 'include';\n /**\n * If `true`, Resource Timing API information will be collected for these transformed requests and returned in a resourceTiming property of relevant data events.\n */\n collectResourceTiming?: boolean;\n /**\n * Parameters supported only by browser fetch API. Property of the Request interface contains the cache mode of the request. It controls how the request will interact with the browser's HTTP cache. (https://developer.mozilla.org/en-US/docs/Web/API/Request/cache)\n */\n cache?: RequestCache;\n};\n\n/**\n * The response callback used in various places\n */\nexport type ResponseCallback = (\n error?: Error | null,\n data?: T | null,\n cacheControl?: string | null,\n expires?: string | null\n) => void;\n\n/**\n * An error thrown when a HTTP request results in an error response.\n */\nexport class AJAXError extends Error {\n /**\n * The response's HTTP status code.\n */\n status: number;\n\n /**\n * The response's HTTP status text.\n */\n statusText: string;\n\n /**\n * The request's URL.\n */\n url: string;\n\n /**\n * The response's body.\n */\n body: Blob;\n\n /**\n * @param status - The response's HTTP status code.\n * @param statusText - The response's HTTP status text.\n * @param url - The request's URL.\n * @param body - The response's body.\n */\n constructor(status: number, statusText: string, url: string, body: Blob) {\n super(`AJAXError: ${statusText} (${status}): ${url}`);\n this.status = status;\n this.statusText = statusText;\n this.url = url;\n this.body = body;\n }\n}\n\n// Ensure that we're sending the correct referrer from blob URL worker bundles.\n// For files loaded from the local file system, `location.origin` will be set\n// to the string(!) \"null\" (Firefox), or \"file://\" (Chrome, Safari, Edge, IE),\n// and we will set an empty referrer. Otherwise, we're using the document's URL.\n/* global self */\nexport const getReferrer = isWorker() ?\n () => (self as any).worker && (self as any).worker.referrer :\n () => (window.location.protocol === 'blob:' ? window.parent : window).location.href;\n\nexport const getProtocolAction = url => config.REGISTERED_PROTOCOLS[url.substring(0, url.indexOf('://'))];\n\n// Determines whether a URL is a file:// URL. This is obviously the case if it begins\n// with file://. Relative URLs are also file:// URLs iff the original document was loaded\n// via a file:// URL.\nconst isFileURL = url => /^file:/.test(url) || (/^file:/.test(getReferrer()) && !/^\\w+:/.test(url));\n\nfunction makeFetchRequest(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n const controller = new AbortController();\n const request = new Request(requestParameters.url, {\n method: requestParameters.method || 'GET',\n body: requestParameters.body,\n credentials: requestParameters.credentials,\n headers: requestParameters.headers,\n cache: requestParameters.cache,\n referrer: getReferrer(),\n signal: controller.signal\n });\n let complete = false;\n let aborted = false;\n\n if (requestParameters.type === 'json') {\n request.headers.set('Accept', 'application/json');\n }\n\n const validateOrFetch = (err, cachedResponse?, responseIsFresh?) => {\n if (aborted) return;\n\n if (err) {\n // Do fetch in case of cache error.\n // HTTP pages in Edge trigger a security error that can be ignored.\n if (err.message !== 'SecurityError') {\n warnOnce(err);\n }\n }\n\n if (cachedResponse && responseIsFresh) {\n return finishRequest(cachedResponse);\n }\n\n if (cachedResponse) {\n // We can't do revalidation with 'If-None-Match' because then the\n // request doesn't have simple cors headers.\n }\n\n fetch(request).then(response => {\n if (response.ok) {\n return finishRequest(response);\n\n } else {\n return response.blob().then(body => callback(new AJAXError(response.status, response.statusText, requestParameters.url, body)));\n }\n }).catch(error => {\n if (error.code === 20) {\n // silence expected AbortError\n return;\n }\n callback(new Error(error.message));\n });\n };\n\n const finishRequest = (response) => {\n (\n (requestParameters.type === 'arrayBuffer' || requestParameters.type === 'image') ? response.arrayBuffer() :\n requestParameters.type === 'json' ? response.json() :\n response.text()\n ).then(result => {\n if (aborted) return;\n complete = true;\n callback(null, result, response.headers.get('Cache-Control'), response.headers.get('Expires'));\n }).catch(err => {\n if (!aborted) callback(new Error(err.message));\n });\n };\n\n validateOrFetch(null, null);\n\n return {cancel: () => {\n aborted = true;\n if (!complete) controller.abort();\n }};\n}\n\nfunction makeXMLHttpRequest(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n const xhr: XMLHttpRequest = new XMLHttpRequest();\n\n xhr.open(requestParameters.method || 'GET', requestParameters.url, true);\n if (requestParameters.type === 'arrayBuffer' || requestParameters.type === 'image') {\n xhr.responseType = 'arraybuffer';\n }\n for (const k in requestParameters.headers) {\n xhr.setRequestHeader(k, requestParameters.headers[k]);\n }\n if (requestParameters.type === 'json') {\n xhr.responseType = 'text';\n xhr.setRequestHeader('Accept', 'application/json');\n }\n xhr.withCredentials = requestParameters.credentials === 'include';\n xhr.onerror = () => {\n callback(new Error(xhr.statusText));\n };\n xhr.onload = () => {\n if (((xhr.status >= 200 && xhr.status < 300) || xhr.status === 0) && xhr.response !== null) {\n let data: unknown = xhr.response;\n if (requestParameters.type === 'json') {\n // We're manually parsing JSON here to get better error messages.\n try {\n data = JSON.parse(xhr.response);\n } catch (err) {\n return callback(err);\n }\n }\n callback(null, data, xhr.getResponseHeader('Cache-Control'), xhr.getResponseHeader('Expires'));\n } else {\n const body = new Blob([xhr.response], {type: xhr.getResponseHeader('Content-Type')});\n callback(new AJAXError(xhr.status, xhr.statusText, requestParameters.url, body));\n }\n };\n xhr.send(requestParameters.body);\n return {cancel: () => xhr.abort()};\n}\n\nexport const makeRequest = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n // We're trying to use the Fetch API if possible. However, in some situations we can't use it:\n // - IE11 doesn't support it at all. In this case, we dispatch the request to the main thread so\n // that we can get an accruate referrer header.\n // - Safari exposes window.AbortController, but it doesn't work actually abort any requests in\n // some versions (see https://bugs.webkit.org/show_bug.cgi?id=174980#c2)\n // - Requests for resources with the file:// URI scheme don't work with the Fetch API either. In\n // this case we unconditionally use XHR on the current thread since referrers don't matter.\n if (/:\\/\\//.test(requestParameters.url) && !(/^https?:|^file:/.test(requestParameters.url))) {\n if (isWorker() && (self as any).worker && (self as any).worker.actor) {\n return (self as any).worker.actor.send('getResource', requestParameters, callback);\n }\n if (!isWorker()) {\n const action = getProtocolAction(requestParameters.url) || makeFetchRequest;\n return action(requestParameters, callback);\n }\n }\n if (!isFileURL(requestParameters.url)) {\n if (fetch && Request && AbortController && Object.prototype.hasOwnProperty.call(Request.prototype, 'signal')) {\n return makeFetchRequest(requestParameters, callback);\n }\n if (isWorker() && (self as any).worker && (self as any).worker.actor) {\n const queueOnMainThread = true;\n return (self as any).worker.actor.send('getResource', requestParameters, callback, undefined, queueOnMainThread);\n }\n }\n return makeXMLHttpRequest(requestParameters, callback);\n};\n\nexport const getJSON = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n return makeRequest(extend(requestParameters, {type: 'json'}), callback);\n};\n\nexport const getArrayBuffer = function(\n requestParameters: RequestParameters,\n callback: ResponseCallback\n): Cancelable {\n return makeRequest(extend(requestParameters, {type: 'arrayBuffer'}), callback);\n};\n\nexport const postData = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n return makeRequest(extend(requestParameters, {method: 'POST'}), callback);\n};\n\nexport function sameOrigin(inComingUrl: string) {\n // URL class should be available everywhere\n // https://developer.mozilla.org/en-US/docs/Web/API/URL\n // In addtion, a relative URL \"/foo\" or \"./foo\" will throw exception in its ctor,\n // try-catch is expansive so just use a heuristic check to avoid it\n // also check data URL\n if (!inComingUrl ||\n inComingUrl.indexOf('://') <= 0 || // relative URL\n inComingUrl.indexOf('data:image/') === 0 || // data image URL\n inComingUrl.indexOf('blob:') === 0) { // blob\n return true;\n }\n const urlObj = new URL(inComingUrl);\n const locationObj = window.location;\n return urlObj.protocol === locationObj.protocol && urlObj.host === locationObj.host;\n}\n/**\n * A type used to store the tile's expiration date and cache control definition\n */\nexport type ExpiryData = {cacheControl?: string | null; expires?: Date | string | null};\nexport const getVideo = function(urls: Array, callback: Callback): Cancelable {\n const video: HTMLVideoElement = window.document.createElement('video');\n video.muted = true;\n video.onloadstart = function() {\n callback(null, video);\n };\n for (let i = 0; i < urls.length; i++) {\n const s: HTMLSourceElement = window.document.createElement('source');\n if (!sameOrigin(urls[i])) {\n video.crossOrigin = 'Anonymous';\n }\n s.src = urls[i];\n video.appendChild(s);\n }\n return {cancel: () => {}};\n};\n","import {extend} from './util';\n\n/**\n * A listener method used as a callback to events\n */\nexport type Listener = (a: any) => any;\n\ntype Listeners = {[_: string]: Array};\n\nfunction _addEventListener(type: string, listener: Listener, listenerList: Listeners) {\n const listenerExists = listenerList[type] && listenerList[type].indexOf(listener) !== -1;\n if (!listenerExists) {\n listenerList[type] = listenerList[type] || [];\n listenerList[type].push(listener);\n }\n}\n\nfunction _removeEventListener(type: string, listener: Listener, listenerList: Listeners) {\n if (listenerList && listenerList[type]) {\n const index = listenerList[type].indexOf(listener);\n if (index !== -1) {\n listenerList[type].splice(index, 1);\n }\n }\n}\n\n/**\n * The event class\n */\nexport class Event {\n readonly type: string;\n\n constructor(type: string, data: any = {}) {\n extend(this, data);\n this.type = type;\n }\n}\n\ninterface ErrorLike {\n message: string;\n}\n\n/**\n * An error event\n */\nexport class ErrorEvent extends Event {\n error: ErrorLike;\n\n constructor(error: ErrorLike, data: any = {}) {\n super('error', extend({error}, data));\n }\n}\n\n/**\n * Methods mixed in to other classes for event capabilities.\n *\n * @group Event Related\n */\nexport class Evented {\n _listeners: Listeners;\n _oneTimeListeners: Listeners;\n _eventedParent: Evented;\n _eventedParentData: any | (() => any);\n\n /**\n * Adds a listener to a specified event type.\n *\n * @param type - The event type to add a listen for.\n * @param listener - The function to be called when the event is fired.\n * The listener function is called with the data object passed to `fire`,\n * extended with `target` and `type` properties.\n * @returns `this`\n */\n on(type: string, listener: Listener): this {\n this._listeners = this._listeners || {};\n _addEventListener(type, listener, this._listeners);\n\n return this;\n }\n\n /**\n * Removes a previously registered event listener.\n *\n * @param type - The event type to remove listeners for.\n * @param listener - The listener function to remove.\n * @returns `this`\n */\n off(type: string, listener: Listener) {\n _removeEventListener(type, listener, this._listeners);\n _removeEventListener(type, listener, this._oneTimeListeners);\n\n return this;\n }\n\n /**\n * Adds a listener that will be called only once to a specified event type.\n *\n * The listener will be called first time the event fires after the listener is registered.\n *\n * @param type - The event type to listen for.\n * @param listener - The function to be called when the event is fired the first time.\n * @returns `this` or a promise if a listener is not provided\n */\n once(type: string, listener?: Listener): this | Promise {\n if (!listener) {\n return new Promise((resolve) => this.once(type, resolve));\n }\n this._oneTimeListeners = this._oneTimeListeners || {};\n _addEventListener(type, listener, this._oneTimeListeners);\n\n return this;\n }\n\n fire(event: Event | string, properties?: any) {\n // Compatibility with (type: string, properties: Object) signature from previous versions.\n // See https://github.com/mapbox/mapbox-gl-js/issues/6522,\n // https://github.com/mapbox/mapbox-gl-draw/issues/766\n if (typeof event === 'string') {\n event = new Event(event, properties || {});\n }\n\n const type = event.type;\n\n if (this.listens(type)) {\n (event as any).target = this;\n\n // make sure adding or removing listeners inside other listeners won't cause an infinite loop\n const listeners = this._listeners && this._listeners[type] ? this._listeners[type].slice() : [];\n for (const listener of listeners) {\n listener.call(this, event);\n }\n\n const oneTimeListeners = this._oneTimeListeners && this._oneTimeListeners[type] ? this._oneTimeListeners[type].slice() : [];\n for (const listener of oneTimeListeners) {\n _removeEventListener(type, listener, this._oneTimeListeners);\n listener.call(this, event);\n }\n\n const parent = this._eventedParent;\n if (parent) {\n extend(\n event,\n typeof this._eventedParentData === 'function' ? this._eventedParentData() : this._eventedParentData\n );\n parent.fire(event);\n }\n\n // To ensure that no error events are dropped, print them to the\n // console if they have no listeners.\n } else if (event instanceof ErrorEvent) {\n console.error(event.error);\n }\n\n return this;\n }\n\n /**\n * Returns a true if this instance of Evented or any forwardeed instances of Evented have a listener for the specified type.\n *\n * @param type - The event type\n * @returns `true` if there is at least one registered listener for specified event type, `false` otherwise\n */\n listens(type: string): boolean {\n return (\n (this._listeners && this._listeners[type] && this._listeners[type].length > 0) ||\n (this._oneTimeListeners && this._oneTimeListeners[type] && this._oneTimeListeners[type].length > 0) ||\n (this._eventedParent && this._eventedParent.listens(type))\n );\n }\n\n /**\n * Bubble all events fired by this instance of Evented to this parent instance of Evented.\n * @returns `this`\n */\n setEventedParent(parent?: Evented | null, data?: any | (() => any)) {\n this._eventedParent = parent;\n this._eventedParentData = data;\n\n return this;\n }\n}\n","import UnitBezier from '@mapbox/unitbezier';\n\nvar $version = 8;\nvar $root = {\n\tversion: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: [\n\t\t\t8\n\t\t]\n\t},\n\tname: {\n\t\ttype: \"string\"\n\t},\n\tmetadata: {\n\t\ttype: \"*\"\n\t},\n\tcenter: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\"\n\t},\n\tzoom: {\n\t\ttype: \"number\"\n\t},\n\tbearing: {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\tunits: \"degrees\"\n\t},\n\tpitch: {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tunits: \"degrees\"\n\t},\n\tlight: {\n\t\ttype: \"light\"\n\t},\n\tterrain: {\n\t\ttype: \"terrain\"\n\t},\n\tsources: {\n\t\trequired: true,\n\t\ttype: \"sources\"\n\t},\n\tsprite: {\n\t\ttype: \"sprite\"\n\t},\n\tglyphs: {\n\t\ttype: \"string\"\n\t},\n\ttransition: {\n\t\ttype: \"transition\"\n\t},\n\tlayers: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tvalue: \"layer\"\n\t}\n};\nvar sources = {\n\t\"*\": {\n\t\ttype: \"source\"\n\t}\n};\nvar source = [\n\t\"source_vector\",\n\t\"source_raster\",\n\t\"source_raster_dem\",\n\t\"source_geojson\",\n\t\"source_video\",\n\t\"source_image\"\n];\nvar source_vector = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvector: {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\ttype: \"string\"\n\t},\n\ttiles: {\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tbounds: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t-180,\n\t\t\t-85.051129,\n\t\t\t180,\n\t\t\t85.051129\n\t\t]\n\t},\n\tscheme: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\txyz: {\n\t\t\t},\n\t\t\ttms: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"xyz\"\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 22\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tpromoteId: {\n\t\ttype: \"promoteId\"\n\t},\n\tvolatile: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\t\"*\": {\n\t\ttype: \"*\"\n\t}\n};\nvar source_raster = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\traster: {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\ttype: \"string\"\n\t},\n\ttiles: {\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tbounds: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t-180,\n\t\t\t-85.051129,\n\t\t\t180,\n\t\t\t85.051129\n\t\t]\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 22\n\t},\n\ttileSize: {\n\t\ttype: \"number\",\n\t\t\"default\": 512,\n\t\tunits: \"pixels\"\n\t},\n\tscheme: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\txyz: {\n\t\t\t},\n\t\t\ttms: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"xyz\"\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tvolatile: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\t\"*\": {\n\t\ttype: \"*\"\n\t}\n};\nvar source_raster_dem = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\t\"raster-dem\": {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\ttype: \"string\"\n\t},\n\ttiles: {\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tbounds: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t-180,\n\t\t\t-85.051129,\n\t\t\t180,\n\t\t\t85.051129\n\t\t]\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 22\n\t},\n\ttileSize: {\n\t\ttype: \"number\",\n\t\t\"default\": 512,\n\t\tunits: \"pixels\"\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tencoding: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tterrarium: {\n\t\t\t},\n\t\t\tmapbox: {\n\t\t\t},\n\t\t\tcustom: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"mapbox\"\n\t},\n\tredFactor: {\n\t\ttype: \"number\",\n\t\t\"default\": 1\n\t},\n\tblueFactor: {\n\t\ttype: \"number\",\n\t\t\"default\": 1\n\t},\n\tgreenFactor: {\n\t\ttype: \"number\",\n\t\t\"default\": 1\n\t},\n\tbaseShift: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tvolatile: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\t\"*\": {\n\t\ttype: \"*\"\n\t}\n};\nvar source_geojson = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tgeojson: {\n\t\t\t}\n\t\t}\n\t},\n\tdata: {\n\t\trequired: true,\n\t\ttype: \"*\"\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 18\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tbuffer: {\n\t\ttype: \"number\",\n\t\t\"default\": 128,\n\t\tmaximum: 512,\n\t\tminimum: 0\n\t},\n\tfilter: {\n\t\ttype: \"*\"\n\t},\n\ttolerance: {\n\t\ttype: \"number\",\n\t\t\"default\": 0.375\n\t},\n\tcluster: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\tclusterRadius: {\n\t\ttype: \"number\",\n\t\t\"default\": 50,\n\t\tminimum: 0\n\t},\n\tclusterMaxZoom: {\n\t\ttype: \"number\"\n\t},\n\tclusterMinPoints: {\n\t\ttype: \"number\"\n\t},\n\tclusterProperties: {\n\t\ttype: \"*\"\n\t},\n\tlineMetrics: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\tgenerateId: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\tpromoteId: {\n\t\ttype: \"promoteId\"\n\t}\n};\nvar source_video = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvideo: {\n\t\t\t}\n\t\t}\n\t},\n\turls: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tcoordinates: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tlength: 4,\n\t\tvalue: {\n\t\t\ttype: \"array\",\n\t\t\tlength: 2,\n\t\t\tvalue: \"number\"\n\t\t}\n\t}\n};\nvar source_image = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\timage: {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\trequired: true,\n\t\ttype: \"string\"\n\t},\n\tcoordinates: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tlength: 4,\n\t\tvalue: {\n\t\t\ttype: \"array\",\n\t\t\tlength: 2,\n\t\t\tvalue: \"number\"\n\t\t}\n\t}\n};\nvar layer = {\n\tid: {\n\t\ttype: \"string\",\n\t\trequired: true\n\t},\n\ttype: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tfill: {\n\t\t\t},\n\t\t\tline: {\n\t\t\t},\n\t\t\tsymbol: {\n\t\t\t},\n\t\t\tcircle: {\n\t\t\t},\n\t\t\theatmap: {\n\t\t\t},\n\t\t\t\"fill-extrusion\": {\n\t\t\t},\n\t\t\traster: {\n\t\t\t},\n\t\t\thillshade: {\n\t\t\t},\n\t\t\tbackground: {\n\t\t\t}\n\t\t},\n\t\trequired: true\n\t},\n\tmetadata: {\n\t\ttype: \"*\"\n\t},\n\tsource: {\n\t\ttype: \"string\"\n\t},\n\t\"source-layer\": {\n\t\ttype: \"string\"\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\tminimum: 0,\n\t\tmaximum: 24\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\tminimum: 0,\n\t\tmaximum: 24\n\t},\n\tfilter: {\n\t\ttype: \"filter\"\n\t},\n\tlayout: {\n\t\ttype: \"layout\"\n\t},\n\tpaint: {\n\t\ttype: \"paint\"\n\t}\n};\nvar layout = [\n\t\"layout_fill\",\n\t\"layout_line\",\n\t\"layout_circle\",\n\t\"layout_heatmap\",\n\t\"layout_fill-extrusion\",\n\t\"layout_symbol\",\n\t\"layout_raster\",\n\t\"layout_hillshade\",\n\t\"layout_background\"\n];\nvar layout_background = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_fill = {\n\t\"fill-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_circle = {\n\t\"circle-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_heatmap = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_line = {\n\t\"line-cap\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tbutt: {\n\t\t\t},\n\t\t\tround: {\n\t\t\t},\n\t\t\tsquare: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"butt\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-join\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tbevel: {\n\t\t\t},\n\t\t\tround: {\n\t\t\t},\n\t\t\tmiter: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"miter\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-miter-limit\": {\n\t\ttype: \"number\",\n\t\t\"default\": 2,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"line-join\": \"miter\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-round-limit\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1.05,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"line-join\": \"round\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_symbol = {\n\t\"symbol-placement\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tpoint: {\n\t\t\t},\n\t\t\tline: {\n\t\t\t},\n\t\t\t\"line-center\": {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"point\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"symbol-spacing\": {\n\t\ttype: \"number\",\n\t\t\"default\": 250,\n\t\tminimum: 1,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"symbol-placement\": \"line\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"symbol-avoid-edges\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"symbol-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"symbol-z-order\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tauto: {\n\t\t\t},\n\t\t\t\"viewport-y\": {\n\t\t\t},\n\t\t\tsource: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-allow-overlap\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t{\n\t\t\t\t\"!\": \"icon-overlap\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-overlap\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnever: {\n\t\t\t},\n\t\t\talways: {\n\t\t\t},\n\t\t\tcooperative: {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-ignore-placement\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-optional\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-rotation-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-size\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tunits: \"factor of the original icon size\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-text-fit\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnone: {\n\t\t\t},\n\t\t\twidth: {\n\t\t\t},\n\t\t\theight: {\n\t\t\t},\n\t\t\tboth: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"none\",\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-text-fit-padding\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0,\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"icon-text-fit\": [\n\t\t\t\t\t\"both\",\n\t\t\t\t\t\"width\",\n\t\t\t\t\t\"height\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-image\": {\n\t\ttype: \"resolvedImage\",\n\t\ttokens: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-rotate\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\tunits: \"degrees\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-padding\": {\n\t\ttype: \"padding\",\n\t\t\"default\": [\n\t\t\t2\n\t\t],\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-keep-upright\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t{\n\t\t\t\t\"icon-rotation-alignment\": \"map\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"line\",\n\t\t\t\t\t\"line-center\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-offset\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t},\n\t\t\ttop: {\n\t\t\t},\n\t\t\tbottom: {\n\t\t\t},\n\t\t\t\"top-left\": {\n\t\t\t},\n\t\t\t\"top-right\": {\n\t\t\t},\n\t\t\t\"bottom-left\": {\n\t\t\t},\n\t\t\t\"bottom-right\": {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"center\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-pitch-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-pitch-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-rotation-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\t\"viewport-glyph\": {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-field\": {\n\t\ttype: \"formatted\",\n\t\t\"default\": \"\",\n\t\ttokens: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-font\": {\n\t\ttype: \"array\",\n\t\tvalue: \"string\",\n\t\t\"default\": [\n\t\t\t\"Open Sans Regular\",\n\t\t\t\"Arial Unicode MS Regular\"\n\t\t],\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-size\": {\n\t\ttype: \"number\",\n\t\t\"default\": 16,\n\t\tminimum: 0,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-max-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 10,\n\t\tminimum: 0,\n\t\tunits: \"ems\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-line-height\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1.2,\n\t\tunits: \"ems\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-letter-spacing\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tunits: \"ems\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-justify\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tauto: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"center\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-radial-offset\": {\n\t\ttype: \"number\",\n\t\tunits: \"ems\",\n\t\t\"default\": 0,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\t\"property-type\": \"data-driven\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t}\n\t},\n\t\"text-variable-anchor\": {\n\t\ttype: \"array\",\n\t\tvalue: \"enum\",\n\t\tvalues: {\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t},\n\t\t\ttop: {\n\t\t\t},\n\t\t\tbottom: {\n\t\t\t},\n\t\t\t\"top-left\": {\n\t\t\t},\n\t\t\t\"top-right\": {\n\t\t\t},\n\t\t\t\"bottom-left\": {\n\t\t\t},\n\t\t\t\"bottom-right\": {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"point\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-variable-anchor-offset\": {\n\t\ttype: \"variableAnchorOffsetCollection\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"point\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t},\n\t\t\ttop: {\n\t\t\t},\n\t\t\tbottom: {\n\t\t\t},\n\t\t\t\"top-left\": {\n\t\t\t},\n\t\t\t\"top-right\": {\n\t\t\t},\n\t\t\t\"bottom-left\": {\n\t\t\t},\n\t\t\t\"bottom-right\": {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"center\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"!\": \"text-variable-anchor\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-max-angle\": {\n\t\ttype: \"number\",\n\t\t\"default\": 45,\n\t\tunits: \"degrees\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"line\",\n\t\t\t\t\t\"line-center\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-writing-mode\": {\n\t\ttype: \"array\",\n\t\tvalue: \"enum\",\n\t\tvalues: {\n\t\t\thorizontal: {\n\t\t\t},\n\t\t\tvertical: {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"point\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-rotate\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\tunits: \"degrees\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-padding\": {\n\t\ttype: \"number\",\n\t\t\"default\": 2,\n\t\tminimum: 0,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-keep-upright\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": true,\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"text-rotation-alignment\": \"map\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"line\",\n\t\t\t\t\t\"line-center\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-transform\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnone: {\n\t\t\t},\n\t\t\tuppercase: {\n\t\t\t},\n\t\t\tlowercase: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"none\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-offset\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tunits: \"ems\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"!\": \"text-radial-offset\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-allow-overlap\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"!\": \"text-overlap\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-overlap\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnever: {\n\t\t\t},\n\t\t\talways: {\n\t\t\t},\n\t\t\tcooperative: {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-ignore-placement\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-optional\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_raster = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_hillshade = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar filter = {\n\ttype: \"array\",\n\tvalue: \"*\"\n};\nvar filter_operator = {\n\ttype: \"enum\",\n\tvalues: {\n\t\t\"==\": {\n\t\t},\n\t\t\"!=\": {\n\t\t},\n\t\t\">\": {\n\t\t},\n\t\t\">=\": {\n\t\t},\n\t\t\"<\": {\n\t\t},\n\t\t\"<=\": {\n\t\t},\n\t\t\"in\": {\n\t\t},\n\t\t\"!in\": {\n\t\t},\n\t\tall: {\n\t\t},\n\t\tany: {\n\t\t},\n\t\tnone: {\n\t\t},\n\t\thas: {\n\t\t},\n\t\t\"!has\": {\n\t\t},\n\t\twithin: {\n\t\t}\n\t}\n};\nvar geometry_type = {\n\ttype: \"enum\",\n\tvalues: {\n\t\tPoint: {\n\t\t},\n\t\tLineString: {\n\t\t},\n\t\tPolygon: {\n\t\t}\n\t}\n};\nvar function_stop = {\n\ttype: \"array\",\n\tminimum: 0,\n\tmaximum: 24,\n\tvalue: [\n\t\t\"number\",\n\t\t\"color\"\n\t],\n\tlength: 2\n};\nvar expression$1 = {\n\ttype: \"array\",\n\tvalue: \"*\",\n\tminimum: 1\n};\nvar light = {\n\tanchor: {\n\t\ttype: \"enum\",\n\t\t\"default\": \"viewport\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"property-type\": \"data-constant\",\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t}\n\t},\n\tposition: {\n\t\ttype: \"array\",\n\t\t\"default\": [\n\t\t\t1.15,\n\t\t\t210,\n\t\t\t30\n\t\t],\n\t\tlength: 3,\n\t\tvalue: \"number\",\n\t\t\"property-type\": \"data-constant\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t}\n\t},\n\tcolor: {\n\t\ttype: \"color\",\n\t\t\"property-type\": \"data-constant\",\n\t\t\"default\": \"#ffffff\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\ttransition: true\n\t},\n\tintensity: {\n\t\ttype: \"number\",\n\t\t\"property-type\": \"data-constant\",\n\t\t\"default\": 0.5,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\ttransition: true\n\t}\n};\nvar terrain = {\n\tsource: {\n\t\ttype: \"string\",\n\t\trequired: true\n\t},\n\texaggeration: {\n\t\ttype: \"number\",\n\t\tminimum: 0,\n\t\t\"default\": 1\n\t}\n};\nvar paint = [\n\t\"paint_fill\",\n\t\"paint_line\",\n\t\"paint_circle\",\n\t\"paint_heatmap\",\n\t\"paint_fill-extrusion\",\n\t\"paint_symbol\",\n\t\"paint_raster\",\n\t\"paint_hillshade\",\n\t\"paint_background\"\n];\nvar paint_fill = {\n\t\"fill-antialias\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"fill-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-outline-color\": {\n\t\ttype: \"color\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"fill-pattern\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"fill-antialias\": true\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"fill-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded-data-driven\"\n\t}\n};\nvar paint_line = {\n\t\"line-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"line-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"line-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-gap-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-offset\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-dasharray\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"line widths\",\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"line-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded\"\n\t},\n\t\"line-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded-data-driven\"\n\t},\n\t\"line-gradient\": {\n\t\ttype: \"color\",\n\t\ttransition: false,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"line-dasharray\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"!\": \"line-pattern\"\n\t\t\t},\n\t\t\t{\n\t\t\t\tsource: \"geojson\",\n\t\t\t\thas: {\n\t\t\t\t\tlineMetrics: true\n\t\t\t\t}\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"line-progress\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"color-ramp\"\n\t}\n};\nvar paint_circle = {\n\t\"circle-radius\": {\n\t\ttype: \"number\",\n\t\t\"default\": 5,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"circle-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-pitch-scale\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-pitch-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"viewport\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-stroke-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-stroke-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-stroke-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t}\n};\nvar paint_heatmap = {\n\t\"heatmap-radius\": {\n\t\ttype: \"number\",\n\t\t\"default\": 30,\n\t\tminimum: 1,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"heatmap-weight\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"heatmap-intensity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"heatmap-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": [\n\t\t\t\"interpolate\",\n\t\t\t[\n\t\t\t\t\"linear\"\n\t\t\t],\n\t\t\t[\n\t\t\t\t\"heatmap-density\"\n\t\t\t],\n\t\t\t0,\n\t\t\t\"rgba(0, 0, 255, 0)\",\n\t\t\t0.1,\n\t\t\t\"royalblue\",\n\t\t\t0.3,\n\t\t\t\"cyan\",\n\t\t\t0.5,\n\t\t\t\"lime\",\n\t\t\t0.7,\n\t\t\t\"yellow\",\n\t\t\t1,\n\t\t\t\"red\"\n\t\t],\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"heatmap-density\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"color-ramp\"\n\t},\n\t\"heatmap-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_symbol = {\n\t\"icon-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-halo-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"rgba(0, 0, 0, 0)\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-halo-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-halo-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"icon-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\toverridable: true,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-halo-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"rgba(0, 0, 0, 0)\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-halo-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-halo-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t\"text-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_raster = {\n\t\"raster-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-hue-rotate\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\ttransition: true,\n\t\tunits: \"degrees\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-brightness-min\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-brightness-max\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-saturation\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: -1,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-contrast\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: -1,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-resampling\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tlinear: {\n\t\t\t},\n\t\t\tnearest: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"linear\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-fade-duration\": {\n\t\ttype: \"number\",\n\t\t\"default\": 300,\n\t\tminimum: 0,\n\t\ttransition: false,\n\t\tunits: \"milliseconds\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_hillshade = {\n\t\"hillshade-illumination-direction\": {\n\t\ttype: \"number\",\n\t\t\"default\": 335,\n\t\tminimum: 0,\n\t\tmaximum: 359,\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-illumination-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"viewport\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-exaggeration\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0.5,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-shadow-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-highlight-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#FFFFFF\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-accent-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_background = {\n\t\"background-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"background-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"background-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded\"\n\t},\n\t\"background-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar transition = {\n\tduration: {\n\t\ttype: \"number\",\n\t\t\"default\": 300,\n\t\tminimum: 0,\n\t\tunits: \"milliseconds\"\n\t},\n\tdelay: {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tunits: \"milliseconds\"\n\t}\n};\nvar promoteId = {\n\t\"*\": {\n\t\ttype: \"string\"\n\t}\n};\nvar v8Spec = {\n\t$version: $version,\n\t$root: $root,\n\tsources: sources,\n\tsource: source,\n\tsource_vector: source_vector,\n\tsource_raster: source_raster,\n\tsource_raster_dem: source_raster_dem,\n\tsource_geojson: source_geojson,\n\tsource_video: source_video,\n\tsource_image: source_image,\n\tlayer: layer,\n\tlayout: layout,\n\tlayout_background: layout_background,\n\tlayout_fill: layout_fill,\n\tlayout_circle: layout_circle,\n\tlayout_heatmap: layout_heatmap,\n\t\"layout_fill-extrusion\": {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n},\n\tlayout_line: layout_line,\n\tlayout_symbol: layout_symbol,\n\tlayout_raster: layout_raster,\n\tlayout_hillshade: layout_hillshade,\n\tfilter: filter,\n\tfilter_operator: filter_operator,\n\tgeometry_type: geometry_type,\n\t\"function\": {\n\texpression: {\n\t\ttype: \"expression\"\n\t},\n\tstops: {\n\t\ttype: \"array\",\n\t\tvalue: \"function_stop\"\n\t},\n\tbase: {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0\n\t},\n\tproperty: {\n\t\ttype: \"string\",\n\t\t\"default\": \"$zoom\"\n\t},\n\ttype: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tidentity: {\n\t\t\t},\n\t\t\texponential: {\n\t\t\t},\n\t\t\tinterval: {\n\t\t\t},\n\t\t\tcategorical: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"exponential\"\n\t},\n\tcolorSpace: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\trgb: {\n\t\t\t},\n\t\t\tlab: {\n\t\t\t},\n\t\t\thcl: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"rgb\"\n\t},\n\t\"default\": {\n\t\ttype: \"*\",\n\t\trequired: false\n\t}\n},\n\tfunction_stop: function_stop,\n\texpression: expression$1,\n\tlight: light,\n\tterrain: terrain,\n\tpaint: paint,\n\tpaint_fill: paint_fill,\n\t\"paint_fill-extrusion\": {\n\t\"fill-extrusion-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-extrusion-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"fill-extrusion-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-extrusion-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-extrusion-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"fill-extrusion-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-extrusion-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded-data-driven\"\n\t},\n\t\"fill-extrusion-height\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tunits: \"meters\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-extrusion-base\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tunits: \"meters\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"fill-extrusion-height\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-extrusion-vertical-gradient\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": true,\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n},\n\tpaint_line: paint_line,\n\tpaint_circle: paint_circle,\n\tpaint_heatmap: paint_heatmap,\n\tpaint_symbol: paint_symbol,\n\tpaint_raster: paint_raster,\n\tpaint_hillshade: paint_hillshade,\n\tpaint_background: paint_background,\n\ttransition: transition,\n\t\"property-type\": {\n\t\"data-driven\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"cross-faded\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"cross-faded-data-driven\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"color-ramp\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"data-constant\": {\n\t\ttype: \"property-type\"\n\t},\n\tconstant: {\n\t\ttype: \"property-type\"\n\t}\n},\n\tpromoteId: promoteId\n};\n\nconst refProperties = ['type', 'source', 'source-layer', 'minzoom', 'maxzoom', 'filter', 'layout'];\n\nfunction deref(layer, parent) {\n const result = {};\n for (const k in layer) {\n if (k !== 'ref') {\n result[k] = layer[k];\n }\n }\n refProperties.forEach((k) => {\n if (k in parent) {\n result[k] = parent[k];\n }\n });\n return result;\n}\n/**\n * Given an array of layers, some of which may contain `ref` properties\n * whose value is the `id` of another property, return a new array where\n * such layers have been augmented with the 'type', 'source', etc. properties\n * from the parent layer, and the `ref` property has been removed.\n *\n * The input is not modified. The output may contain references to portions\n * of the input.\n *\n * @private\n * @param {Array} layers\n * @returns {Array}\n */\nfunction derefLayers(layers) {\n layers = layers.slice();\n const map = Object.create(null);\n for (let i = 0; i < layers.length; i++) {\n map[layers[i].id] = layers[i];\n }\n for (let i = 0; i < layers.length; i++) {\n if ('ref' in layers[i]) {\n layers[i] = deref(layers[i], map[layers[i].ref]);\n }\n }\n return layers;\n}\n\n/**\n * Deeply compares two object literals.\n *\n * @private\n */\nfunction deepEqual(a, b) {\n if (Array.isArray(a)) {\n if (!Array.isArray(b) || a.length !== b.length)\n return false;\n for (let i = 0; i < a.length; i++) {\n if (!deepEqual(a[i], b[i]))\n return false;\n }\n return true;\n }\n if (typeof a === 'object' && a !== null && b !== null) {\n if (!(typeof b === 'object'))\n return false;\n const keys = Object.keys(a);\n if (keys.length !== Object.keys(b).length)\n return false;\n for (const key in a) {\n if (!deepEqual(a[key], b[key]))\n return false;\n }\n return true;\n }\n return a === b;\n}\n\nconst operations = {\n /*\n * { command: 'setStyle', args: [stylesheet] }\n */\n setStyle: 'setStyle',\n /*\n * { command: 'addLayer', args: [layer, 'beforeLayerId'] }\n */\n addLayer: 'addLayer',\n /*\n * { command: 'removeLayer', args: ['layerId'] }\n */\n removeLayer: 'removeLayer',\n /*\n * { command: 'setPaintProperty', args: ['layerId', 'prop', value] }\n */\n setPaintProperty: 'setPaintProperty',\n /*\n * { command: 'setLayoutProperty', args: ['layerId', 'prop', value] }\n */\n setLayoutProperty: 'setLayoutProperty',\n /*\n * { command: 'setFilter', args: ['layerId', filter] }\n */\n setFilter: 'setFilter',\n /*\n * { command: 'addSource', args: ['sourceId', source] }\n */\n addSource: 'addSource',\n /*\n * { command: 'removeSource', args: ['sourceId'] }\n */\n removeSource: 'removeSource',\n /*\n * { command: 'setGeoJSONSourceData', args: ['sourceId', data] }\n */\n setGeoJSONSourceData: 'setGeoJSONSourceData',\n /*\n * { command: 'setLayerZoomRange', args: ['layerId', 0, 22] }\n */\n setLayerZoomRange: 'setLayerZoomRange',\n /*\n * { command: 'setLayerProperty', args: ['layerId', 'prop', value] }\n */\n setLayerProperty: 'setLayerProperty',\n /*\n * { command: 'setCenter', args: [[lon, lat]] }\n */\n setCenter: 'setCenter',\n /*\n * { command: 'setZoom', args: [zoom] }\n */\n setZoom: 'setZoom',\n /*\n * { command: 'setBearing', args: [bearing] }\n */\n setBearing: 'setBearing',\n /*\n * { command: 'setPitch', args: [pitch] }\n */\n setPitch: 'setPitch',\n /*\n * { command: 'setSprite', args: ['spriteUrl'] }\n */\n setSprite: 'setSprite',\n /*\n * { command: 'setGlyphs', args: ['glyphsUrl'] }\n */\n setGlyphs: 'setGlyphs',\n /*\n * { command: 'setTransition', args: [transition] }\n */\n setTransition: 'setTransition',\n /*\n * { command: 'setLighting', args: [lightProperties] }\n */\n setLight: 'setLight'\n};\nfunction addSource(sourceId, after, commands) {\n commands.push({ command: operations.addSource, args: [sourceId, after[sourceId]] });\n}\nfunction removeSource(sourceId, commands, sourcesRemoved) {\n commands.push({ command: operations.removeSource, args: [sourceId] });\n sourcesRemoved[sourceId] = true;\n}\nfunction updateSource(sourceId, after, commands, sourcesRemoved) {\n removeSource(sourceId, commands, sourcesRemoved);\n addSource(sourceId, after, commands);\n}\nfunction canUpdateGeoJSON(before, after, sourceId) {\n let prop;\n for (prop in before[sourceId]) {\n if (!Object.prototype.hasOwnProperty.call(before[sourceId], prop))\n continue;\n if (prop !== 'data' && !deepEqual(before[sourceId][prop], after[sourceId][prop])) {\n return false;\n }\n }\n for (prop in after[sourceId]) {\n if (!Object.prototype.hasOwnProperty.call(after[sourceId], prop))\n continue;\n if (prop !== 'data' && !deepEqual(before[sourceId][prop], after[sourceId][prop])) {\n return false;\n }\n }\n return true;\n}\nfunction diffSources(before, after, commands, sourcesRemoved) {\n before = before || {};\n after = after || {};\n let sourceId;\n // look for sources to remove\n for (sourceId in before) {\n if (!Object.prototype.hasOwnProperty.call(before, sourceId))\n continue;\n if (!Object.prototype.hasOwnProperty.call(after, sourceId)) {\n removeSource(sourceId, commands, sourcesRemoved);\n }\n }\n // look for sources to add/update\n for (sourceId in after) {\n if (!Object.prototype.hasOwnProperty.call(after, sourceId))\n continue;\n if (!Object.prototype.hasOwnProperty.call(before, sourceId)) {\n addSource(sourceId, after, commands);\n }\n else if (!deepEqual(before[sourceId], after[sourceId])) {\n if (before[sourceId].type === 'geojson' && after[sourceId].type === 'geojson' && canUpdateGeoJSON(before, after, sourceId)) {\n commands.push({ command: operations.setGeoJSONSourceData, args: [sourceId, after[sourceId].data] });\n }\n else {\n // no update command, must remove then add\n updateSource(sourceId, after, commands, sourcesRemoved);\n }\n }\n }\n}\nfunction diffLayerPropertyChanges(before, after, commands, layerId, klass, command) {\n before = before || {};\n after = after || {};\n let prop;\n for (prop in before) {\n if (!Object.prototype.hasOwnProperty.call(before, prop))\n continue;\n if (!deepEqual(before[prop], after[prop])) {\n commands.push({ command, args: [layerId, prop, after[prop], klass] });\n }\n }\n for (prop in after) {\n if (!Object.prototype.hasOwnProperty.call(after, prop) || Object.prototype.hasOwnProperty.call(before, prop))\n continue;\n if (!deepEqual(before[prop], after[prop])) {\n commands.push({ command, args: [layerId, prop, after[prop], klass] });\n }\n }\n}\nfunction pluckId(layer) {\n return layer.id;\n}\nfunction indexById(group, layer) {\n group[layer.id] = layer;\n return group;\n}\nfunction diffLayers(before, after, commands) {\n before = before || [];\n after = after || [];\n // order of layers by id\n const beforeOrder = before.map(pluckId);\n const afterOrder = after.map(pluckId);\n // index of layer by id\n const beforeIndex = before.reduce(indexById, {});\n const afterIndex = after.reduce(indexById, {});\n // track order of layers as if they have been mutated\n const tracker = beforeOrder.slice();\n // layers that have been added do not need to be diffed\n const clean = Object.create(null);\n let i, d, layerId, beforeLayer, afterLayer, insertBeforeLayerId, prop;\n // remove layers\n for (i = 0, d = 0; i < beforeOrder.length; i++) {\n layerId = beforeOrder[i];\n if (!Object.prototype.hasOwnProperty.call(afterIndex, layerId)) {\n commands.push({ command: operations.removeLayer, args: [layerId] });\n tracker.splice(tracker.indexOf(layerId, d), 1);\n }\n else {\n // limit where in tracker we need to look for a match\n d++;\n }\n }\n // add/reorder layers\n for (i = 0, d = 0; i < afterOrder.length; i++) {\n // work backwards as insert is before an existing layer\n layerId = afterOrder[afterOrder.length - 1 - i];\n if (tracker[tracker.length - 1 - i] === layerId)\n continue;\n if (Object.prototype.hasOwnProperty.call(beforeIndex, layerId)) {\n // remove the layer before we insert at the correct position\n commands.push({ command: operations.removeLayer, args: [layerId] });\n tracker.splice(tracker.lastIndexOf(layerId, tracker.length - d), 1);\n }\n else {\n // limit where in tracker we need to look for a match\n d++;\n }\n // add layer at correct position\n insertBeforeLayerId = tracker[tracker.length - i];\n commands.push({ command: operations.addLayer, args: [afterIndex[layerId], insertBeforeLayerId] });\n tracker.splice(tracker.length - i, 0, layerId);\n clean[layerId] = true;\n }\n // update layers\n for (i = 0; i < afterOrder.length; i++) {\n layerId = afterOrder[i];\n beforeLayer = beforeIndex[layerId];\n afterLayer = afterIndex[layerId];\n // no need to update if previously added (new or moved)\n if (clean[layerId] || deepEqual(beforeLayer, afterLayer))\n continue;\n // If source, source-layer, or type have changes, then remove the layer\n // and add it back 'from scratch'.\n if (!deepEqual(beforeLayer.source, afterLayer.source) || !deepEqual(beforeLayer['source-layer'], afterLayer['source-layer']) || !deepEqual(beforeLayer.type, afterLayer.type)) {\n commands.push({ command: operations.removeLayer, args: [layerId] });\n // we add the layer back at the same position it was already in, so\n // there's no need to update the `tracker`\n insertBeforeLayerId = tracker[tracker.lastIndexOf(layerId) + 1];\n commands.push({ command: operations.addLayer, args: [afterLayer, insertBeforeLayerId] });\n continue;\n }\n // layout, paint, filter, minzoom, maxzoom\n diffLayerPropertyChanges(beforeLayer.layout, afterLayer.layout, commands, layerId, null, operations.setLayoutProperty);\n diffLayerPropertyChanges(beforeLayer.paint, afterLayer.paint, commands, layerId, null, operations.setPaintProperty);\n if (!deepEqual(beforeLayer.filter, afterLayer.filter)) {\n commands.push({ command: operations.setFilter, args: [layerId, afterLayer.filter] });\n }\n if (!deepEqual(beforeLayer.minzoom, afterLayer.minzoom) || !deepEqual(beforeLayer.maxzoom, afterLayer.maxzoom)) {\n commands.push({ command: operations.setLayerZoomRange, args: [layerId, afterLayer.minzoom, afterLayer.maxzoom] });\n }\n // handle all other layer props, including paint.*\n for (prop in beforeLayer) {\n if (!Object.prototype.hasOwnProperty.call(beforeLayer, prop))\n continue;\n if (prop === 'layout' || prop === 'paint' || prop === 'filter' ||\n prop === 'metadata' || prop === 'minzoom' || prop === 'maxzoom')\n continue;\n if (prop.indexOf('paint.') === 0) {\n diffLayerPropertyChanges(beforeLayer[prop], afterLayer[prop], commands, layerId, prop.slice(6), operations.setPaintProperty);\n }\n else if (!deepEqual(beforeLayer[prop], afterLayer[prop])) {\n commands.push({ command: operations.setLayerProperty, args: [layerId, prop, afterLayer[prop]] });\n }\n }\n for (prop in afterLayer) {\n if (!Object.prototype.hasOwnProperty.call(afterLayer, prop) || Object.prototype.hasOwnProperty.call(beforeLayer, prop))\n continue;\n if (prop === 'layout' || prop === 'paint' || prop === 'filter' ||\n prop === 'metadata' || prop === 'minzoom' || prop === 'maxzoom')\n continue;\n if (prop.indexOf('paint.') === 0) {\n diffLayerPropertyChanges(beforeLayer[prop], afterLayer[prop], commands, layerId, prop.slice(6), operations.setPaintProperty);\n }\n else if (!deepEqual(beforeLayer[prop], afterLayer[prop])) {\n commands.push({ command: operations.setLayerProperty, args: [layerId, prop, afterLayer[prop]] });\n }\n }\n }\n}\n/**\n * Diff two stylesheet\n *\n * Creates semanticly aware diffs that can easily be applied at runtime.\n * Operations produced by the diff closely resemble the maplibre-gl-js API. Any\n * error creating the diff will fall back to the 'setStyle' operation.\n *\n * Example diff:\n * [\n * { command: 'setConstant', args: ['@water', '#0000FF'] },\n * { command: 'setPaintProperty', args: ['background', 'background-color', 'black'] }\n * ]\n *\n * @private\n * @param {*} [before] stylesheet to compare from\n * @param {*} after stylesheet to compare to\n * @returns Array list of changes\n */\nfunction diffStyles(before, after) {\n if (!before)\n return [{ command: operations.setStyle, args: [after] }];\n let commands = [];\n try {\n // Handle changes to top-level properties\n if (!deepEqual(before.version, after.version)) {\n return [{ command: operations.setStyle, args: [after] }];\n }\n if (!deepEqual(before.center, after.center)) {\n commands.push({ command: operations.setCenter, args: [after.center] });\n }\n if (!deepEqual(before.zoom, after.zoom)) {\n commands.push({ command: operations.setZoom, args: [after.zoom] });\n }\n if (!deepEqual(before.bearing, after.bearing)) {\n commands.push({ command: operations.setBearing, args: [after.bearing] });\n }\n if (!deepEqual(before.pitch, after.pitch)) {\n commands.push({ command: operations.setPitch, args: [after.pitch] });\n }\n if (!deepEqual(before.sprite, after.sprite)) {\n commands.push({ command: operations.setSprite, args: [after.sprite] });\n }\n if (!deepEqual(before.glyphs, after.glyphs)) {\n commands.push({ command: operations.setGlyphs, args: [after.glyphs] });\n }\n if (!deepEqual(before.transition, after.transition)) {\n commands.push({ command: operations.setTransition, args: [after.transition] });\n }\n if (!deepEqual(before.light, after.light)) {\n commands.push({ command: operations.setLight, args: [after.light] });\n }\n // Handle changes to `sources`\n // If a source is to be removed, we also--before the removeSource\n // command--need to remove all the style layers that depend on it.\n const sourcesRemoved = {};\n // First collect the {add,remove}Source commands\n const removeOrAddSourceCommands = [];\n diffSources(before.sources, after.sources, removeOrAddSourceCommands, sourcesRemoved);\n // Push a removeLayer command for each style layer that depends on a\n // source that's being removed.\n // Also, exclude any such layers them from the input to `diffLayers`\n // below, so that diffLayers produces the appropriate `addLayers`\n // command\n const beforeLayers = [];\n if (before.layers) {\n before.layers.forEach((layer) => {\n if (sourcesRemoved[layer.source]) {\n commands.push({ command: operations.removeLayer, args: [layer.id] });\n }\n else {\n beforeLayers.push(layer);\n }\n });\n }\n commands = commands.concat(removeOrAddSourceCommands);\n // Handle changes to `layers`\n diffLayers(beforeLayers, after.layers, commands);\n }\n catch (e) {\n // fall back to setStyle\n console.warn('Unable to compute style diff:', e);\n commands = [{ command: operations.setStyle, args: [after] }];\n }\n return commands;\n}\n\n// Note: Do not inherit from Error. It breaks when transpiling to ES5.\nclass ValidationError {\n constructor(key, value, message, identifier) {\n this.message = (key ? `${key}: ` : '') + message;\n if (identifier)\n this.identifier = identifier;\n if (value !== null && value !== undefined && value.__line__) {\n this.line = value.__line__;\n }\n }\n}\n\n// Note: Do not inherit from Error. It breaks when transpiling to ES5.\nclass ParsingError {\n constructor(error) {\n this.error = error;\n this.message = error.message;\n const match = error.message.match(/line (\\d+)/);\n this.line = match ? parseInt(match[1], 10) : 0;\n }\n}\n\nfunction extendBy(output, ...inputs) {\n for (const input of inputs) {\n for (const k in input) {\n output[k] = input[k];\n }\n }\n return output;\n}\n\nclass ExpressionParsingError extends Error {\n constructor(key, message) {\n super(message);\n this.message = message;\n this.key = key;\n }\n}\n\n/**\n * Tracks `let` bindings during expression parsing.\n * @private\n */\nclass Scope {\n constructor(parent, bindings = []) {\n this.parent = parent;\n this.bindings = {};\n for (const [name, expression] of bindings) {\n this.bindings[name] = expression;\n }\n }\n concat(bindings) {\n return new Scope(this, bindings);\n }\n get(name) {\n if (this.bindings[name]) {\n return this.bindings[name];\n }\n if (this.parent) {\n return this.parent.get(name);\n }\n throw new Error(`${name} not found in scope.`);\n }\n has(name) {\n if (this.bindings[name])\n return true;\n return this.parent ? this.parent.has(name) : false;\n }\n}\n\nconst NullType = { kind: 'null' };\nconst NumberType = { kind: 'number' };\nconst StringType = { kind: 'string' };\nconst BooleanType = { kind: 'boolean' };\nconst ColorType = { kind: 'color' };\nconst ObjectType = { kind: 'object' };\nconst ValueType = { kind: 'value' };\nconst ErrorType = { kind: 'error' };\nconst CollatorType = { kind: 'collator' };\nconst FormattedType = { kind: 'formatted' };\nconst PaddingType = { kind: 'padding' };\nconst ResolvedImageType = { kind: 'resolvedImage' };\nconst VariableAnchorOffsetCollectionType = { kind: 'variableAnchorOffsetCollection' };\nfunction array$1(itemType, N) {\n return {\n kind: 'array',\n itemType,\n N\n };\n}\nfunction toString$1(type) {\n if (type.kind === 'array') {\n const itemType = toString$1(type.itemType);\n return typeof type.N === 'number' ?\n `array<${itemType}, ${type.N}>` :\n type.itemType.kind === 'value' ? 'array' : `array<${itemType}>`;\n }\n else {\n return type.kind;\n }\n}\nconst valueMemberTypes = [\n NullType,\n NumberType,\n StringType,\n BooleanType,\n ColorType,\n FormattedType,\n ObjectType,\n array$1(ValueType),\n PaddingType,\n ResolvedImageType,\n VariableAnchorOffsetCollectionType\n];\n/**\n * Returns null if `t` is a subtype of `expected`; otherwise returns an\n * error message.\n * @private\n */\nfunction checkSubtype(expected, t) {\n if (t.kind === 'error') {\n // Error is a subtype of every type\n return null;\n }\n else if (expected.kind === 'array') {\n if (t.kind === 'array' &&\n ((t.N === 0 && t.itemType.kind === 'value') || !checkSubtype(expected.itemType, t.itemType)) &&\n (typeof expected.N !== 'number' || expected.N === t.N)) {\n return null;\n }\n }\n else if (expected.kind === t.kind) {\n return null;\n }\n else if (expected.kind === 'value') {\n for (const memberType of valueMemberTypes) {\n if (!checkSubtype(memberType, t)) {\n return null;\n }\n }\n }\n return `Expected ${toString$1(expected)} but found ${toString$1(t)} instead.`;\n}\nfunction isValidType(provided, allowedTypes) {\n return allowedTypes.some(t => t.kind === provided.kind);\n}\nfunction isValidNativeType(provided, allowedTypes) {\n return allowedTypes.some(t => {\n if (t === 'null') {\n return provided === null;\n }\n else if (t === 'array') {\n return Array.isArray(provided);\n }\n else if (t === 'object') {\n return provided && !Array.isArray(provided) && typeof provided === 'object';\n }\n else {\n return t === typeof provided;\n }\n });\n}\n/**\n * Verify whether the specified type is of the same type as the specified sample.\n *\n * @param provided Type to verify\n * @param sample Sample type to reference\n * @returns `true` if both objects are of the same type, `false` otherwise\n * @example basic types\n * if (verifyType(outputType, ValueType)) {\n * // type narrowed to:\n * outputType.kind; // 'value'\n * }\n * @example array types\n * if (verifyType(outputType, array(NumberType))) {\n * // type narrowed to:\n * outputType.kind; // 'array'\n * outputType.itemType; // NumberTypeT\n * outputType.itemType.kind; // 'number'\n * }\n */\nfunction verifyType(provided, sample) {\n if (provided.kind === 'array' && sample.kind === 'array') {\n return provided.itemType.kind === sample.itemType.kind && typeof provided.N === 'number';\n }\n return provided.kind === sample.kind;\n}\n\n// See https://observablehq.com/@mbostock/lab-and-rgb\nconst Xn = 0.96422, Yn = 1, Zn = 0.82521, t0 = 4 / 29, t1 = 6 / 29, t2 = 3 * t1 * t1, t3 = t1 * t1 * t1, deg2rad = Math.PI / 180, rad2deg = 180 / Math.PI;\nfunction constrainAngle(angle) {\n angle = angle % 360;\n if (angle < 0) {\n angle += 360;\n }\n return angle;\n}\nfunction rgbToLab([r, g, b, alpha]) {\n r = rgb2xyz(r);\n g = rgb2xyz(g);\n b = rgb2xyz(b);\n let x, z;\n const y = xyz2lab((0.2225045 * r + 0.7168786 * g + 0.0606169 * b) / Yn);\n if (r === g && g === b) {\n x = z = y;\n }\n else {\n x = xyz2lab((0.4360747 * r + 0.3850649 * g + 0.1430804 * b) / Xn);\n z = xyz2lab((0.0139322 * r + 0.0971045 * g + 0.7141733 * b) / Zn);\n }\n const l = 116 * y - 16;\n return [(l < 0) ? 0 : l, 500 * (x - y), 200 * (y - z), alpha];\n}\nfunction rgb2xyz(x) {\n return (x <= 0.04045) ? x / 12.92 : Math.pow((x + 0.055) / 1.055, 2.4);\n}\nfunction xyz2lab(t) {\n return (t > t3) ? Math.pow(t, 1 / 3) : t / t2 + t0;\n}\nfunction labToRgb([l, a, b, alpha]) {\n let y = (l + 16) / 116, x = isNaN(a) ? y : y + a / 500, z = isNaN(b) ? y : y - b / 200;\n y = Yn * lab2xyz(y);\n x = Xn * lab2xyz(x);\n z = Zn * lab2xyz(z);\n return [\n xyz2rgb(3.1338561 * x - 1.6168667 * y - 0.4906146 * z),\n xyz2rgb(-0.9787684 * x + 1.9161415 * y + 0.0334540 * z),\n xyz2rgb(0.0719453 * x - 0.2289914 * y + 1.4052427 * z),\n alpha,\n ];\n}\nfunction xyz2rgb(x) {\n x = (x <= 0.00304) ? 12.92 * x : 1.055 * Math.pow(x, 1 / 2.4) - 0.055;\n return (x < 0) ? 0 : (x > 1) ? 1 : x; // clip to 0..1 range\n}\nfunction lab2xyz(t) {\n return (t > t1) ? t * t * t : t2 * (t - t0);\n}\nfunction rgbToHcl(rgbColor) {\n const [l, a, b, alpha] = rgbToLab(rgbColor);\n const c = Math.sqrt(a * a + b * b);\n const h = Math.round(c * 10000) ? constrainAngle(Math.atan2(b, a) * rad2deg) : NaN;\n return [h, c, l, alpha];\n}\nfunction hclToRgb([h, c, l, alpha]) {\n h = isNaN(h) ? 0 : h * deg2rad;\n return labToRgb([l, Math.cos(h) * c, Math.sin(h) * c, alpha]);\n}\n// https://drafts.csswg.org/css-color-4/#hsl-to-rgb\nfunction hslToRgb([h, s, l, alpha]) {\n h = constrainAngle(h);\n s /= 100;\n l /= 100;\n function f(n) {\n const k = (n + h / 30) % 12;\n const a = s * Math.min(l, 1 - l);\n return l - a * Math.max(-1, Math.min(k - 3, 9 - k, 1));\n }\n return [f(0), f(8), f(4), alpha];\n}\n\n/**\n * CSS color parser compliant with CSS Color 4 Specification.\n * Supports: named colors, `transparent` keyword, all rgb hex notations,\n * rgb(), rgba(), hsl() and hsla() functions.\n * Does not round the parsed values to integers from the range 0..255.\n *\n * Syntax:\n *\n * = | \n * = | \n *\n * rgb() = rgb( {3} [ / ]? ) | rgb( {3} [ / ]? )\n * rgb() = rgb( #{3} , ? ) | rgb( #{3} , ? )\n *\n * hsl() = hsl( [ / ]? )\n * hsl() = hsl( , , , ? )\n *\n * Caveats:\n * - - with optional `deg` suffix; `grad`, `rad`, `turn` are not supported\n * - `none` keyword is not supported\n * - comments inside rgb()/hsl() are not supported\n * - legacy color syntax rgba() is supported with an identical grammar and behavior to rgb()\n * - legacy color syntax hsla() is supported with an identical grammar and behavior to hsl()\n *\n * @param input CSS color string to parse.\n * @returns Color in sRGB color space, with `red`, `green`, `blue`\n * and `alpha` channels normalized to the range 0..1,\n * or `undefined` if the input is not a valid color string.\n */\nfunction parseCssColor(input) {\n input = input.toLowerCase().trim();\n if (input === 'transparent') {\n return [0, 0, 0, 0];\n }\n // 'white', 'black', 'blue'\n const namedColorsMatch = namedColors[input];\n if (namedColorsMatch) {\n const [r, g, b] = namedColorsMatch;\n return [r / 255, g / 255, b / 255, 1];\n }\n // #f0c, #f0cf, #ff00cc, #ff00ccff\n if (input.startsWith('#')) {\n const hexRegexp = /^#(?:[0-9a-f]{3,4}|[0-9a-f]{6}|[0-9a-f]{8})$/;\n if (hexRegexp.test(input)) {\n const step = input.length < 6 ? 1 : 2;\n let i = 1;\n return [\n parseHex(input.slice(i, i += step)),\n parseHex(input.slice(i, i += step)),\n parseHex(input.slice(i, i += step)),\n parseHex(input.slice(i, i + step) || 'ff'),\n ];\n }\n }\n // rgb(128 0 0), rgb(50% 0% 0%), rgba(255,0,255,0.6), rgb(255 0 255 / 60%), rgb(100% 0% 100% /.6)\n if (input.startsWith('rgb')) {\n const rgbRegExp = /^rgba?\\(\\s*([\\de.+-]+)(%)?(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)(%)?(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)(%)?(?:\\s*([,\\/])\\s*([\\de.+-]+)(%)?)?\\s*\\)$/;\n const rgbMatch = input.match(rgbRegExp);\n if (rgbMatch) {\n const [_, // eslint-disable-line @typescript-eslint/no-unused-vars\n r, // \n rp, // % (optional)\n f1, // , (optional)\n g, // \n gp, // % (optional)\n f2, // , (optional)\n b, // \n bp, // % (optional)\n f3, // ,|/ (optional)\n a, // (optional)\n ap, // % (optional)\n ] = rgbMatch;\n const argFormat = [f1 || ' ', f2 || ' ', f3].join('');\n if (argFormat === ' ' ||\n argFormat === ' /' ||\n argFormat === ',,' ||\n argFormat === ',,,') {\n const valFormat = [rp, gp, bp].join('');\n const maxValue = (valFormat === '%%%') ? 100 :\n (valFormat === '') ? 255 : 0;\n if (maxValue) {\n const rgba = [\n clamp(+r / maxValue, 0, 1),\n clamp(+g / maxValue, 0, 1),\n clamp(+b / maxValue, 0, 1),\n a ? parseAlpha(+a, ap) : 1,\n ];\n if (validateNumbers(rgba)) {\n return rgba;\n }\n // invalid numbers\n }\n // values must be all numbers or all percentages\n }\n return; // comma optional syntax requires no commas at all\n }\n }\n // hsl(120 50% 80%), hsla(120deg,50%,80%,.9), hsl(12e1 50% 80% / 90%)\n const hslRegExp = /^hsla?\\(\\s*([\\de.+-]+)(?:deg)?(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)%(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)%(?:\\s*([,\\/])\\s*([\\de.+-]+)(%)?)?\\s*\\)$/;\n const hslMatch = input.match(hslRegExp);\n if (hslMatch) {\n const [_, // eslint-disable-line @typescript-eslint/no-unused-vars\n h, // \n f1, // , (optional)\n s, // \n f2, // , (optional)\n l, // \n f3, // ,|/ (optional)\n a, // (optional)\n ap, // % (optional)\n ] = hslMatch;\n const argFormat = [f1 || ' ', f2 || ' ', f3].join('');\n if (argFormat === ' ' ||\n argFormat === ' /' ||\n argFormat === ',,' ||\n argFormat === ',,,') {\n const hsla = [\n +h,\n clamp(+s, 0, 100),\n clamp(+l, 0, 100),\n a ? parseAlpha(+a, ap) : 1,\n ];\n if (validateNumbers(hsla)) {\n return hslToRgb(hsla);\n }\n // invalid numbers\n }\n // comma optional syntax requires no commas at all\n }\n}\nfunction parseHex(hex) {\n return parseInt(hex.padEnd(2, hex), 16) / 255;\n}\nfunction parseAlpha(a, asPercentage) {\n return clamp(asPercentage ? (a / 100) : a, 0, 1);\n}\nfunction clamp(n, min, max) {\n return Math.min(Math.max(min, n), max);\n}\n/**\n * The regular expression for numeric values is not super specific, and it may\n * happen that it will accept a value that is not a valid number. In order to\n * detect and eliminate such values this function exists.\n *\n * @param array Array of uncertain numbers.\n * @returns `true` if the specified array contains only valid numbers, `false` otherwise.\n */\nfunction validateNumbers(array) {\n return !array.some(Number.isNaN);\n}\n/**\n * To generate:\n * - visit {@link https://www.w3.org/TR/css-color-4/#named-colors}\n * - run in the console:\n * @example\n * copy(`{\\n${[...document.querySelector('.named-color-table tbody').children].map((tr) => `${tr.cells[2].textContent.trim()}: [${tr.cells[4].textContent.trim().split(/\\s+/).join(', ')}],`).join('\\n')}\\n}`);\n */\nconst namedColors = {\n aliceblue: [240, 248, 255],\n antiquewhite: [250, 235, 215],\n aqua: [0, 255, 255],\n aquamarine: [127, 255, 212],\n azure: [240, 255, 255],\n beige: [245, 245, 220],\n bisque: [255, 228, 196],\n black: [0, 0, 0],\n blanchedalmond: [255, 235, 205],\n blue: [0, 0, 255],\n blueviolet: [138, 43, 226],\n brown: [165, 42, 42],\n burlywood: [222, 184, 135],\n cadetblue: [95, 158, 160],\n chartreuse: [127, 255, 0],\n chocolate: [210, 105, 30],\n coral: [255, 127, 80],\n cornflowerblue: [100, 149, 237],\n cornsilk: [255, 248, 220],\n crimson: [220, 20, 60],\n cyan: [0, 255, 255],\n darkblue: [0, 0, 139],\n darkcyan: [0, 139, 139],\n darkgoldenrod: [184, 134, 11],\n darkgray: [169, 169, 169],\n darkgreen: [0, 100, 0],\n darkgrey: [169, 169, 169],\n darkkhaki: [189, 183, 107],\n darkmagenta: [139, 0, 139],\n darkolivegreen: [85, 107, 47],\n darkorange: [255, 140, 0],\n darkorchid: [153, 50, 204],\n darkred: [139, 0, 0],\n darksalmon: [233, 150, 122],\n darkseagreen: [143, 188, 143],\n darkslateblue: [72, 61, 139],\n darkslategray: [47, 79, 79],\n darkslategrey: [47, 79, 79],\n darkturquoise: [0, 206, 209],\n darkviolet: [148, 0, 211],\n deeppink: [255, 20, 147],\n deepskyblue: [0, 191, 255],\n dimgray: [105, 105, 105],\n dimgrey: [105, 105, 105],\n dodgerblue: [30, 144, 255],\n firebrick: [178, 34, 34],\n floralwhite: [255, 250, 240],\n forestgreen: [34, 139, 34],\n fuchsia: [255, 0, 255],\n gainsboro: [220, 220, 220],\n ghostwhite: [248, 248, 255],\n gold: [255, 215, 0],\n goldenrod: [218, 165, 32],\n gray: [128, 128, 128],\n green: [0, 128, 0],\n greenyellow: [173, 255, 47],\n grey: [128, 128, 128],\n honeydew: [240, 255, 240],\n hotpink: [255, 105, 180],\n indianred: [205, 92, 92],\n indigo: [75, 0, 130],\n ivory: [255, 255, 240],\n khaki: [240, 230, 140],\n lavender: [230, 230, 250],\n lavenderblush: [255, 240, 245],\n lawngreen: [124, 252, 0],\n lemonchiffon: [255, 250, 205],\n lightblue: [173, 216, 230],\n lightcoral: [240, 128, 128],\n lightcyan: [224, 255, 255],\n lightgoldenrodyellow: [250, 250, 210],\n lightgray: [211, 211, 211],\n lightgreen: [144, 238, 144],\n lightgrey: [211, 211, 211],\n lightpink: [255, 182, 193],\n lightsalmon: [255, 160, 122],\n lightseagreen: [32, 178, 170],\n lightskyblue: [135, 206, 250],\n lightslategray: [119, 136, 153],\n lightslategrey: [119, 136, 153],\n lightsteelblue: [176, 196, 222],\n lightyellow: [255, 255, 224],\n lime: [0, 255, 0],\n limegreen: [50, 205, 50],\n linen: [250, 240, 230],\n magenta: [255, 0, 255],\n maroon: [128, 0, 0],\n mediumaquamarine: [102, 205, 170],\n mediumblue: [0, 0, 205],\n mediumorchid: [186, 85, 211],\n mediumpurple: [147, 112, 219],\n mediumseagreen: [60, 179, 113],\n mediumslateblue: [123, 104, 238],\n mediumspringgreen: [0, 250, 154],\n mediumturquoise: [72, 209, 204],\n mediumvioletred: [199, 21, 133],\n midnightblue: [25, 25, 112],\n mintcream: [245, 255, 250],\n mistyrose: [255, 228, 225],\n moccasin: [255, 228, 181],\n navajowhite: [255, 222, 173],\n navy: [0, 0, 128],\n oldlace: [253, 245, 230],\n olive: [128, 128, 0],\n olivedrab: [107, 142, 35],\n orange: [255, 165, 0],\n orangered: [255, 69, 0],\n orchid: [218, 112, 214],\n palegoldenrod: [238, 232, 170],\n palegreen: [152, 251, 152],\n paleturquoise: [175, 238, 238],\n palevioletred: [219, 112, 147],\n papayawhip: [255, 239, 213],\n peachpuff: [255, 218, 185],\n peru: [205, 133, 63],\n pink: [255, 192, 203],\n plum: [221, 160, 221],\n powderblue: [176, 224, 230],\n purple: [128, 0, 128],\n rebeccapurple: [102, 51, 153],\n red: [255, 0, 0],\n rosybrown: [188, 143, 143],\n royalblue: [65, 105, 225],\n saddlebrown: [139, 69, 19],\n salmon: [250, 128, 114],\n sandybrown: [244, 164, 96],\n seagreen: [46, 139, 87],\n seashell: [255, 245, 238],\n sienna: [160, 82, 45],\n silver: [192, 192, 192],\n skyblue: [135, 206, 235],\n slateblue: [106, 90, 205],\n slategray: [112, 128, 144],\n slategrey: [112, 128, 144],\n snow: [255, 250, 250],\n springgreen: [0, 255, 127],\n steelblue: [70, 130, 180],\n tan: [210, 180, 140],\n teal: [0, 128, 128],\n thistle: [216, 191, 216],\n tomato: [255, 99, 71],\n turquoise: [64, 224, 208],\n violet: [238, 130, 238],\n wheat: [245, 222, 179],\n white: [255, 255, 255],\n whitesmoke: [245, 245, 245],\n yellow: [255, 255, 0],\n yellowgreen: [154, 205, 50],\n};\n\n/**\n * Color representation used by WebGL.\n * Defined in sRGB color space and pre-blended with alpha.\n * @private\n */\nclass Color {\n /**\n * @param r Red component premultiplied by `alpha` 0..1\n * @param g Green component premultiplied by `alpha` 0..1\n * @param b Blue component premultiplied by `alpha` 0..1\n * @param [alpha=1] Alpha component 0..1\n * @param [premultiplied=true] Whether the `r`, `g` and `b` values have already\n * been multiplied by alpha. If `true` nothing happens if `false` then they will\n * be multiplied automatically.\n */\n constructor(r, g, b, alpha = 1, premultiplied = true) {\n this.r = r;\n this.g = g;\n this.b = b;\n this.a = alpha;\n if (!premultiplied) {\n this.r *= alpha;\n this.g *= alpha;\n this.b *= alpha;\n if (!alpha) {\n // alpha = 0 erases completely rgb channels. This behavior is not desirable\n // if this particular color is later used in color interpolation.\n // Because of that, a reference to original color is saved.\n this.overwriteGetter('rgb', [r, g, b, alpha]);\n }\n }\n }\n /**\n * Parses CSS color strings and converts colors to sRGB color space if needed.\n * Officially supported color formats:\n * - keyword, e.g. 'aquamarine' or 'steelblue'\n * - hex (with 3, 4, 6 or 8 digits), e.g. '#f0f' or '#e9bebea9'\n * - rgb and rgba, e.g. 'rgb(0,240,120)' or 'rgba(0%,94%,47%,0.1)' or 'rgb(0 240 120 / .3)'\n * - hsl and hsla, e.g. 'hsl(0,0%,83%)' or 'hsla(0,0%,83%,.5)' or 'hsl(0 0% 83% / 20%)'\n *\n * @param input CSS color string to parse.\n * @returns A `Color` instance, or `undefined` if the input is not a valid color string.\n */\n static parse(input) {\n // in zoom-and-property function input could be an instance of Color class\n if (input instanceof Color) {\n return input;\n }\n if (typeof input !== 'string') {\n return;\n }\n const rgba = parseCssColor(input);\n if (rgba) {\n return new Color(...rgba, false);\n }\n }\n /**\n * Used in color interpolation and by 'to-rgba' expression.\n *\n * @returns Gien color, with reversed alpha blending, in sRGB color space.\n */\n get rgb() {\n const { r, g, b, a } = this;\n const f = a || Infinity; // reverse alpha blending factor\n return this.overwriteGetter('rgb', [r / f, g / f, b / f, a]);\n }\n /**\n * Used in color interpolation.\n *\n * @returns Gien color, with reversed alpha blending, in HCL color space.\n */\n get hcl() {\n return this.overwriteGetter('hcl', rgbToHcl(this.rgb));\n }\n /**\n * Used in color interpolation.\n *\n * @returns Gien color, with reversed alpha blending, in LAB color space.\n */\n get lab() {\n return this.overwriteGetter('lab', rgbToLab(this.rgb));\n }\n /**\n * Lazy getter pattern. When getter is called for the first time lazy value\n * is calculated and then overwrites getter function in given object instance.\n *\n * @example:\n * const redColor = Color.parse('red');\n * let x = redColor.hcl; // this will invoke `get hcl()`, which will calculate\n * // the value of red in HCL space and invoke this `overwriteGetter` function\n * // which in turn will set a field with a key 'hcl' in the `redColor` object.\n * // In other words it will override `get hcl()` from its `Color` prototype\n * // with its own property: hcl = [calculated red value in hcl].\n * let y = redColor.hcl; // next call will no longer invoke getter but simply\n * // return the previously calculated value\n * x === y; // true - `x` is exactly the same object as `y`\n *\n * @param getterKey Getter key\n * @param lazyValue Lazily calculated value to be memoized by current instance\n * @private\n */\n overwriteGetter(getterKey, lazyValue) {\n Object.defineProperty(this, getterKey, { value: lazyValue });\n return lazyValue;\n }\n /**\n * Used by 'to-string' expression.\n *\n * @returns Serialized color in format `rgba(r,g,b,a)`\n * where r,g,b are numbers within 0..255 and alpha is number within 1..0\n *\n * @example\n * var purple = new Color.parse('purple');\n * purple.toString; // = \"rgba(128,0,128,1)\"\n * var translucentGreen = new Color.parse('rgba(26, 207, 26, .73)');\n * translucentGreen.toString(); // = \"rgba(26,207,26,0.73)\"\n */\n toString() {\n const [r, g, b, a] = this.rgb;\n return `rgba(${[r, g, b].map(n => Math.round(n * 255)).join(',')},${a})`;\n }\n}\nColor.black = new Color(0, 0, 0, 1);\nColor.white = new Color(1, 1, 1, 1);\nColor.transparent = new Color(0, 0, 0, 0);\nColor.red = new Color(1, 0, 0, 1);\n\n// Flow type declarations for Intl cribbed from\n// https://github.com/facebook/flow/issues/1270\nclass Collator {\n constructor(caseSensitive, diacriticSensitive, locale) {\n if (caseSensitive)\n this.sensitivity = diacriticSensitive ? 'variant' : 'case';\n else\n this.sensitivity = diacriticSensitive ? 'accent' : 'base';\n this.locale = locale;\n this.collator = new Intl.Collator(this.locale ? this.locale : [], { sensitivity: this.sensitivity, usage: 'search' });\n }\n compare(lhs, rhs) {\n return this.collator.compare(lhs, rhs);\n }\n resolvedLocale() {\n // We create a Collator without \"usage: search\" because we don't want\n // the search options encoded in our result (e.g. \"en-u-co-search\")\n return new Intl.Collator(this.locale ? this.locale : [])\n .resolvedOptions().locale;\n }\n}\n\nclass FormattedSection {\n constructor(text, image, scale, fontStack, textColor) {\n this.text = text;\n this.image = image;\n this.scale = scale;\n this.fontStack = fontStack;\n this.textColor = textColor;\n }\n}\nclass Formatted {\n constructor(sections) {\n this.sections = sections;\n }\n static fromString(unformatted) {\n return new Formatted([new FormattedSection(unformatted, null, null, null, null)]);\n }\n isEmpty() {\n if (this.sections.length === 0)\n return true;\n return !this.sections.some(section => section.text.length !== 0 ||\n (section.image && section.image.name.length !== 0));\n }\n static factory(text) {\n if (text instanceof Formatted) {\n return text;\n }\n else {\n return Formatted.fromString(text);\n }\n }\n toString() {\n if (this.sections.length === 0)\n return '';\n return this.sections.map(section => section.text).join('');\n }\n}\n\n/**\n * A set of four numbers representing padding around a box. Create instances from\n * bare arrays or numeric values using the static method `Padding.parse`.\n * @private\n */\nclass Padding {\n constructor(values) {\n this.values = values.slice();\n }\n /**\n * Numeric padding values\n * @param input A padding value\n * @returns A `Padding` instance, or `undefined` if the input is not a valid padding value.\n */\n static parse(input) {\n if (input instanceof Padding) {\n return input;\n }\n // Backwards compatibility: bare number is treated the same as array with single value.\n // Padding applies to all four sides.\n if (typeof input === 'number') {\n return new Padding([input, input, input, input]);\n }\n if (!Array.isArray(input)) {\n return undefined;\n }\n if (input.length < 1 || input.length > 4) {\n return undefined;\n }\n for (const val of input) {\n if (typeof val !== 'number') {\n return undefined;\n }\n }\n // Expand shortcut properties into explicit 4-sided values\n switch (input.length) {\n case 1:\n input = [input[0], input[0], input[0], input[0]];\n break;\n case 2:\n input = [input[0], input[1], input[0], input[1]];\n break;\n case 3:\n input = [input[0], input[1], input[2], input[1]];\n break;\n }\n return new Padding(input);\n }\n toString() {\n return JSON.stringify(this.values);\n }\n}\n\n/** Set of valid anchor positions, as a set for validation */\nconst anchors = new Set(['center', 'left', 'right', 'top', 'bottom', 'top-left', 'top-right', 'bottom-left', 'bottom-right']);\n/**\n * Utility class to assist managing values for text-variable-anchor-offset property. Create instances from\n * bare arrays using the static method `VariableAnchorOffsetCollection.parse`.\n * @private\n */\nclass VariableAnchorOffsetCollection {\n constructor(values) {\n this.values = values.slice();\n }\n static parse(input) {\n if (input instanceof VariableAnchorOffsetCollection) {\n return input;\n }\n if (!Array.isArray(input) ||\n input.length < 1 ||\n input.length % 2 !== 0) {\n return undefined;\n }\n for (let i = 0; i < input.length; i += 2) {\n // Elements in even positions should be anchor positions; Elements in odd positions should be offset values\n const anchorValue = input[i];\n const offsetValue = input[i + 1];\n if (typeof anchorValue !== 'string' || !anchors.has(anchorValue)) {\n return undefined;\n }\n if (!Array.isArray(offsetValue) || offsetValue.length !== 2 || typeof offsetValue[0] !== 'number' || typeof offsetValue[1] !== 'number') {\n return undefined;\n }\n }\n return new VariableAnchorOffsetCollection(input);\n }\n toString() {\n return JSON.stringify(this.values);\n }\n}\n\nclass ResolvedImage {\n constructor(options) {\n this.name = options.name;\n this.available = options.available;\n }\n toString() {\n return this.name;\n }\n static fromString(name) {\n if (!name)\n return null; // treat empty values as no image\n return new ResolvedImage({ name, available: false });\n }\n}\n\nfunction validateRGBA(r, g, b, a) {\n if (!(typeof r === 'number' && r >= 0 && r <= 255 &&\n typeof g === 'number' && g >= 0 && g <= 255 &&\n typeof b === 'number' && b >= 0 && b <= 255)) {\n const value = typeof a === 'number' ? [r, g, b, a] : [r, g, b];\n return `Invalid rgba value [${value.join(', ')}]: 'r', 'g', and 'b' must be between 0 and 255.`;\n }\n if (!(typeof a === 'undefined' || (typeof a === 'number' && a >= 0 && a <= 1))) {\n return `Invalid rgba value [${[r, g, b, a].join(', ')}]: 'a' must be between 0 and 1.`;\n }\n return null;\n}\nfunction isValue(mixed) {\n if (mixed === null ||\n typeof mixed === 'string' ||\n typeof mixed === 'boolean' ||\n typeof mixed === 'number' ||\n mixed instanceof Color ||\n mixed instanceof Collator ||\n mixed instanceof Formatted ||\n mixed instanceof Padding ||\n mixed instanceof VariableAnchorOffsetCollection ||\n mixed instanceof ResolvedImage) {\n return true;\n }\n else if (Array.isArray(mixed)) {\n for (const item of mixed) {\n if (!isValue(item)) {\n return false;\n }\n }\n return true;\n }\n else if (typeof mixed === 'object') {\n for (const key in mixed) {\n if (!isValue(mixed[key])) {\n return false;\n }\n }\n return true;\n }\n else {\n return false;\n }\n}\nfunction typeOf(value) {\n if (value === null) {\n return NullType;\n }\n else if (typeof value === 'string') {\n return StringType;\n }\n else if (typeof value === 'boolean') {\n return BooleanType;\n }\n else if (typeof value === 'number') {\n return NumberType;\n }\n else if (value instanceof Color) {\n return ColorType;\n }\n else if (value instanceof Collator) {\n return CollatorType;\n }\n else if (value instanceof Formatted) {\n return FormattedType;\n }\n else if (value instanceof Padding) {\n return PaddingType;\n }\n else if (value instanceof VariableAnchorOffsetCollection) {\n return VariableAnchorOffsetCollectionType;\n }\n else if (value instanceof ResolvedImage) {\n return ResolvedImageType;\n }\n else if (Array.isArray(value)) {\n const length = value.length;\n let itemType;\n for (const item of value) {\n const t = typeOf(item);\n if (!itemType) {\n itemType = t;\n }\n else if (itemType === t) {\n continue;\n }\n else {\n itemType = ValueType;\n break;\n }\n }\n return array$1(itemType || ValueType, length);\n }\n else {\n return ObjectType;\n }\n}\nfunction toString(value) {\n const type = typeof value;\n if (value === null) {\n return '';\n }\n else if (type === 'string' || type === 'number' || type === 'boolean') {\n return String(value);\n }\n else if (value instanceof Color || value instanceof Formatted || value instanceof Padding || value instanceof VariableAnchorOffsetCollection || value instanceof ResolvedImage) {\n return value.toString();\n }\n else {\n return JSON.stringify(value);\n }\n}\n\nclass Literal {\n constructor(type, value) {\n this.type = type;\n this.value = value;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error(`'literal' expression requires exactly one argument, but found ${args.length - 1} instead.`);\n if (!isValue(args[1]))\n return context.error('invalid value');\n const value = args[1];\n let type = typeOf(value);\n // special case: infer the item type if possible for zero-length arrays\n const expected = context.expectedType;\n if (type.kind === 'array' &&\n type.N === 0 &&\n expected &&\n expected.kind === 'array' &&\n (typeof expected.N !== 'number' || expected.N === 0)) {\n type = expected;\n }\n return new Literal(type, value);\n }\n evaluate() {\n return this.value;\n }\n eachChild() { }\n outputDefined() {\n return true;\n }\n}\n\nclass RuntimeError {\n constructor(message) {\n this.name = 'ExpressionEvaluationError';\n this.message = message;\n }\n toJSON() {\n return this.message;\n }\n}\n\nconst types$1 = {\n string: StringType,\n number: NumberType,\n boolean: BooleanType,\n object: ObjectType\n};\nclass Assertion {\n constructor(type, args) {\n this.type = type;\n this.args = args;\n }\n static parse(args, context) {\n if (args.length < 2)\n return context.error('Expected at least one argument.');\n let i = 1;\n let type;\n const name = args[0];\n if (name === 'array') {\n let itemType;\n if (args.length > 2) {\n const type = args[1];\n if (typeof type !== 'string' || !(type in types$1) || type === 'object')\n return context.error('The item type argument of \"array\" must be one of string, number, boolean', 1);\n itemType = types$1[type];\n i++;\n }\n else {\n itemType = ValueType;\n }\n let N;\n if (args.length > 3) {\n if (args[2] !== null &&\n (typeof args[2] !== 'number' ||\n args[2] < 0 ||\n args[2] !== Math.floor(args[2]))) {\n return context.error('The length argument to \"array\" must be a positive integer literal', 2);\n }\n N = args[2];\n i++;\n }\n type = array$1(itemType, N);\n }\n else {\n if (!types$1[name])\n throw new Error(`Types doesn't contain name = ${name}`);\n type = types$1[name];\n }\n const parsed = [];\n for (; i < args.length; i++) {\n const input = context.parse(args[i], i, ValueType);\n if (!input)\n return null;\n parsed.push(input);\n }\n return new Assertion(type, parsed);\n }\n evaluate(ctx) {\n for (let i = 0; i < this.args.length; i++) {\n const value = this.args[i].evaluate(ctx);\n const error = checkSubtype(this.type, typeOf(value));\n if (!error) {\n return value;\n }\n else if (i === this.args.length - 1) {\n throw new RuntimeError(`Expected value to be of type ${toString$1(this.type)}, but found ${toString$1(typeOf(value))} instead.`);\n }\n }\n throw new Error();\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return this.args.every(arg => arg.outputDefined());\n }\n}\n\nconst types = {\n 'to-boolean': BooleanType,\n 'to-color': ColorType,\n 'to-number': NumberType,\n 'to-string': StringType\n};\n/**\n * Special form for error-coalescing coercion expressions \"to-number\",\n * \"to-color\". Since these coercions can fail at runtime, they accept multiple\n * arguments, only evaluating one at a time until one succeeds.\n *\n * @private\n */\nclass Coercion {\n constructor(type, args) {\n this.type = type;\n this.args = args;\n }\n static parse(args, context) {\n if (args.length < 2)\n return context.error('Expected at least one argument.');\n const name = args[0];\n if (!types[name])\n throw new Error(`Can't parse ${name} as it is not part of the known types`);\n if ((name === 'to-boolean' || name === 'to-string') && args.length !== 2)\n return context.error('Expected one argument.');\n const type = types[name];\n const parsed = [];\n for (let i = 1; i < args.length; i++) {\n const input = context.parse(args[i], i, ValueType);\n if (!input)\n return null;\n parsed.push(input);\n }\n return new Coercion(type, parsed);\n }\n evaluate(ctx) {\n switch (this.type.kind) {\n case 'boolean':\n return Boolean(this.args[0].evaluate(ctx));\n case 'color': {\n let input;\n let error;\n for (const arg of this.args) {\n input = arg.evaluate(ctx);\n error = null;\n if (input instanceof Color) {\n return input;\n }\n else if (typeof input === 'string') {\n const c = ctx.parseColor(input);\n if (c)\n return c;\n }\n else if (Array.isArray(input)) {\n if (input.length < 3 || input.length > 4) {\n error = `Invalid rbga value ${JSON.stringify(input)}: expected an array containing either three or four numeric values.`;\n }\n else {\n error = validateRGBA(input[0], input[1], input[2], input[3]);\n }\n if (!error) {\n return new Color(input[0] / 255, input[1] / 255, input[2] / 255, input[3]);\n }\n }\n }\n throw new RuntimeError(error || `Could not parse color from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`);\n }\n case 'padding': {\n let input;\n for (const arg of this.args) {\n input = arg.evaluate(ctx);\n const pad = Padding.parse(input);\n if (pad) {\n return pad;\n }\n }\n throw new RuntimeError(`Could not parse padding from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`);\n }\n case 'variableAnchorOffsetCollection': {\n let input;\n for (const arg of this.args) {\n input = arg.evaluate(ctx);\n const coll = VariableAnchorOffsetCollection.parse(input);\n if (coll) {\n return coll;\n }\n }\n throw new RuntimeError(`Could not parse variableAnchorOffsetCollection from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`);\n }\n case 'number': {\n let value = null;\n for (const arg of this.args) {\n value = arg.evaluate(ctx);\n if (value === null)\n return 0;\n const num = Number(value);\n if (isNaN(num))\n continue;\n return num;\n }\n throw new RuntimeError(`Could not convert ${JSON.stringify(value)} to number.`);\n }\n case 'formatted':\n // There is no explicit 'to-formatted' but this coercion can be implicitly\n // created by properties that expect the 'formatted' type.\n return Formatted.fromString(toString(this.args[0].evaluate(ctx)));\n case 'resolvedImage':\n return ResolvedImage.fromString(toString(this.args[0].evaluate(ctx)));\n default:\n return toString(this.args[0].evaluate(ctx));\n }\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return this.args.every(arg => arg.outputDefined());\n }\n}\n\nconst geometryTypes = ['Unknown', 'Point', 'LineString', 'Polygon'];\nclass EvaluationContext {\n constructor() {\n this.globals = null;\n this.feature = null;\n this.featureState = null;\n this.formattedSection = null;\n this._parseColorCache = {};\n this.availableImages = null;\n this.canonical = null;\n }\n id() {\n return this.feature && 'id' in this.feature ? this.feature.id : null;\n }\n geometryType() {\n return this.feature ? typeof this.feature.type === 'number' ? geometryTypes[this.feature.type] : this.feature.type : null;\n }\n geometry() {\n return this.feature && 'geometry' in this.feature ? this.feature.geometry : null;\n }\n canonicalID() {\n return this.canonical;\n }\n properties() {\n return this.feature && this.feature.properties || {};\n }\n parseColor(input) {\n let cached = this._parseColorCache[input];\n if (!cached) {\n cached = this._parseColorCache[input] = Color.parse(input);\n }\n return cached;\n }\n}\n\n/**\n * State associated parsing at a given point in an expression tree.\n * @private\n */\nclass ParsingContext {\n constructor(registry, isConstantFunc, path = [], expectedType, scope = new Scope(), errors = []) {\n this.registry = registry;\n this.path = path;\n this.key = path.map(part => `[${part}]`).join('');\n this.scope = scope;\n this.errors = errors;\n this.expectedType = expectedType;\n this._isConstant = isConstantFunc;\n }\n /**\n * @param expr the JSON expression to parse\n * @param index the optional argument index if this expression is an argument of a parent expression that's being parsed\n * @param options\n * @param options.omitTypeAnnotations set true to omit inferred type annotations. Caller beware: with this option set, the parsed expression's type will NOT satisfy `expectedType` if it would normally be wrapped in an inferred annotation.\n * @private\n */\n parse(expr, index, expectedType, bindings, options = {}) {\n if (index) {\n return this.concat(index, expectedType, bindings)._parse(expr, options);\n }\n return this._parse(expr, options);\n }\n _parse(expr, options) {\n if (expr === null || typeof expr === 'string' || typeof expr === 'boolean' || typeof expr === 'number') {\n expr = ['literal', expr];\n }\n function annotate(parsed, type, typeAnnotation) {\n if (typeAnnotation === 'assert') {\n return new Assertion(type, [parsed]);\n }\n else if (typeAnnotation === 'coerce') {\n return new Coercion(type, [parsed]);\n }\n else {\n return parsed;\n }\n }\n if (Array.isArray(expr)) {\n if (expr.length === 0) {\n return this.error('Expected an array with at least one element. If you wanted a literal array, use [\"literal\", []].');\n }\n const op = expr[0];\n if (typeof op !== 'string') {\n this.error(`Expression name must be a string, but found ${typeof op} instead. If you wanted a literal array, use [\"literal\", [...]].`, 0);\n return null;\n }\n const Expr = this.registry[op];\n if (Expr) {\n let parsed = Expr.parse(expr, this);\n if (!parsed)\n return null;\n if (this.expectedType) {\n const expected = this.expectedType;\n const actual = parsed.type;\n // When we expect a number, string, boolean, or array but have a value, wrap it in an assertion.\n // When we expect a color or formatted string, but have a string or value, wrap it in a coercion.\n // Otherwise, we do static type-checking.\n //\n // These behaviors are overridable for:\n // * The \"coalesce\" operator, which needs to omit type annotations.\n // * String-valued properties (e.g. `text-field`), where coercion is more convenient than assertion.\n //\n if ((expected.kind === 'string' || expected.kind === 'number' || expected.kind === 'boolean' || expected.kind === 'object' || expected.kind === 'array') && actual.kind === 'value') {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'assert');\n }\n else if ((expected.kind === 'color' || expected.kind === 'formatted' || expected.kind === 'resolvedImage') && (actual.kind === 'value' || actual.kind === 'string')) {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce');\n }\n else if (expected.kind === 'padding' && (actual.kind === 'value' || actual.kind === 'number' || actual.kind === 'array')) {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce');\n }\n else if (expected.kind === 'variableAnchorOffsetCollection' && (actual.kind === 'value' || actual.kind === 'array')) {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce');\n }\n else if (this.checkSubtype(expected, actual)) {\n return null;\n }\n }\n // If an expression's arguments are all literals, we can evaluate\n // it immediately and replace it with a literal value in the\n // parsed/compiled result. Expressions that expect an image should\n // not be resolved here so we can later get the available images.\n if (!(parsed instanceof Literal) && (parsed.type.kind !== 'resolvedImage') && this._isConstant(parsed)) {\n const ec = new EvaluationContext();\n try {\n parsed = new Literal(parsed.type, parsed.evaluate(ec));\n }\n catch (e) {\n this.error(e.message);\n return null;\n }\n }\n return parsed;\n }\n return this.error(`Unknown expression \"${op}\". If you wanted a literal array, use [\"literal\", [...]].`, 0);\n }\n else if (typeof expr === 'undefined') {\n return this.error('\\'undefined\\' value invalid. Use null instead.');\n }\n else if (typeof expr === 'object') {\n return this.error('Bare objects invalid. Use [\"literal\", {...}] instead.');\n }\n else {\n return this.error(`Expected an array, but found ${typeof expr} instead.`);\n }\n }\n /**\n * Returns a copy of this context suitable for parsing the subexpression at\n * index `index`, optionally appending to 'let' binding map.\n *\n * Note that `errors` property, intended for collecting errors while\n * parsing, is copied by reference rather than cloned.\n * @private\n */\n concat(index, expectedType, bindings) {\n const path = typeof index === 'number' ? this.path.concat(index) : this.path;\n const scope = bindings ? this.scope.concat(bindings) : this.scope;\n return new ParsingContext(this.registry, this._isConstant, path, expectedType || null, scope, this.errors);\n }\n /**\n * Push a parsing (or type checking) error into the `this.errors`\n * @param error The message\n * @param keys Optionally specify the source of the error at a child\n * of the current expression at `this.key`.\n * @private\n */\n error(error, ...keys) {\n const key = `${this.key}${keys.map(k => `[${k}]`).join('')}`;\n this.errors.push(new ExpressionParsingError(key, error));\n }\n /**\n * Returns null if `t` is a subtype of `expected`; otherwise returns an\n * error message and also pushes it to `this.errors`.\n * @param expected The expected type\n * @param t The actual type\n * @returns null if `t` is a subtype of `expected`; otherwise returns an error message\n */\n checkSubtype(expected, t) {\n const error = checkSubtype(expected, t);\n if (error)\n this.error(error);\n return error;\n }\n}\n\nclass CollatorExpression {\n constructor(caseSensitive, diacriticSensitive, locale) {\n this.type = CollatorType;\n this.locale = locale;\n this.caseSensitive = caseSensitive;\n this.diacriticSensitive = diacriticSensitive;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error('Expected one argument.');\n const options = args[1];\n if (typeof options !== 'object' || Array.isArray(options))\n return context.error('Collator options argument must be an object.');\n const caseSensitive = context.parse(options['case-sensitive'] === undefined ? false : options['case-sensitive'], 1, BooleanType);\n if (!caseSensitive)\n return null;\n const diacriticSensitive = context.parse(options['diacritic-sensitive'] === undefined ? false : options['diacritic-sensitive'], 1, BooleanType);\n if (!diacriticSensitive)\n return null;\n let locale = null;\n if (options['locale']) {\n locale = context.parse(options['locale'], 1, StringType);\n if (!locale)\n return null;\n }\n return new CollatorExpression(caseSensitive, diacriticSensitive, locale);\n }\n evaluate(ctx) {\n return new Collator(this.caseSensitive.evaluate(ctx), this.diacriticSensitive.evaluate(ctx), this.locale ? this.locale.evaluate(ctx) : null);\n }\n eachChild(fn) {\n fn(this.caseSensitive);\n fn(this.diacriticSensitive);\n if (this.locale) {\n fn(this.locale);\n }\n }\n outputDefined() {\n // Technically the set of possible outputs is the combinatoric set of Collators produced\n // by all possible outputs of locale/caseSensitive/diacriticSensitive\n // But for the primary use of Collators in comparison operators, we ignore the Collator's\n // possible outputs anyway, so we can get away with leaving this false for now.\n return false;\n }\n}\n\nconst EXTENT = 8192;\nfunction updateBBox(bbox, coord) {\n bbox[0] = Math.min(bbox[0], coord[0]);\n bbox[1] = Math.min(bbox[1], coord[1]);\n bbox[2] = Math.max(bbox[2], coord[0]);\n bbox[3] = Math.max(bbox[3], coord[1]);\n}\nfunction mercatorXfromLng(lng) {\n return (180 + lng) / 360;\n}\nfunction mercatorYfromLat(lat) {\n return (180 - (180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360)))) / 360;\n}\nfunction boxWithinBox(bbox1, bbox2) {\n if (bbox1[0] <= bbox2[0])\n return false;\n if (bbox1[2] >= bbox2[2])\n return false;\n if (bbox1[1] <= bbox2[1])\n return false;\n if (bbox1[3] >= bbox2[3])\n return false;\n return true;\n}\nfunction getTileCoordinates(p, canonical) {\n const x = mercatorXfromLng(p[0]);\n const y = mercatorYfromLat(p[1]);\n const tilesAtZoom = Math.pow(2, canonical.z);\n return [Math.round(x * tilesAtZoom * EXTENT), Math.round(y * tilesAtZoom * EXTENT)];\n}\nfunction onBoundary(p, p1, p2) {\n const x1 = p[0] - p1[0];\n const y1 = p[1] - p1[1];\n const x2 = p[0] - p2[0];\n const y2 = p[1] - p2[1];\n return (x1 * y2 - x2 * y1 === 0) && (x1 * x2 <= 0) && (y1 * y2 <= 0);\n}\nfunction rayIntersect(p, p1, p2) {\n return ((p1[1] > p[1]) !== (p2[1] > p[1])) && (p[0] < (p2[0] - p1[0]) * (p[1] - p1[1]) / (p2[1] - p1[1]) + p1[0]);\n}\n// ray casting algorithm for detecting if point is in polygon\nfunction pointWithinPolygon(point, rings) {\n let inside = false;\n for (let i = 0, len = rings.length; i < len; i++) {\n const ring = rings[i];\n for (let j = 0, len2 = ring.length; j < len2 - 1; j++) {\n if (onBoundary(point, ring[j], ring[j + 1]))\n return false;\n if (rayIntersect(point, ring[j], ring[j + 1]))\n inside = !inside;\n }\n }\n return inside;\n}\nfunction pointWithinPolygons(point, polygons) {\n for (let i = 0; i < polygons.length; i++) {\n if (pointWithinPolygon(point, polygons[i]))\n return true;\n }\n return false;\n}\nfunction perp(v1, v2) {\n return (v1[0] * v2[1] - v1[1] * v2[0]);\n}\n// check if p1 and p2 are in different sides of line segment q1->q2\nfunction twoSided(p1, p2, q1, q2) {\n // q1->p1 (x1, y1), q1->p2 (x2, y2), q1->q2 (x3, y3)\n const x1 = p1[0] - q1[0];\n const y1 = p1[1] - q1[1];\n const x2 = p2[0] - q1[0];\n const y2 = p2[1] - q1[1];\n const x3 = q2[0] - q1[0];\n const y3 = q2[1] - q1[1];\n const det1 = (x1 * y3 - x3 * y1);\n const det2 = (x2 * y3 - x3 * y2);\n if ((det1 > 0 && det2 < 0) || (det1 < 0 && det2 > 0))\n return true;\n return false;\n}\n// a, b are end points for line segment1, c and d are end points for line segment2\nfunction lineIntersectLine(a, b, c, d) {\n // check if two segments are parallel or not\n // precondition is end point a, b is inside polygon, if line a->b is\n // parallel to polygon edge c->d, then a->b won't intersect with c->d\n const vectorP = [b[0] - a[0], b[1] - a[1]];\n const vectorQ = [d[0] - c[0], d[1] - c[1]];\n if (perp(vectorQ, vectorP) === 0)\n return false;\n // If lines are intersecting with each other, the relative location should be:\n // a and b lie in different sides of segment c->d\n // c and d lie in different sides of segment a->b\n if (twoSided(a, b, c, d) && twoSided(c, d, a, b))\n return true;\n return false;\n}\nfunction lineIntersectPolygon(p1, p2, polygon) {\n for (const ring of polygon) {\n // loop through every edge of the ring\n for (let j = 0; j < ring.length - 1; ++j) {\n if (lineIntersectLine(p1, p2, ring[j], ring[j + 1])) {\n return true;\n }\n }\n }\n return false;\n}\nfunction lineStringWithinPolygon(line, polygon) {\n // First, check if geometry points of line segments are all inside polygon\n for (let i = 0; i < line.length; ++i) {\n if (!pointWithinPolygon(line[i], polygon)) {\n return false;\n }\n }\n // Second, check if there is line segment intersecting polygon edge\n for (let i = 0; i < line.length - 1; ++i) {\n if (lineIntersectPolygon(line[i], line[i + 1], polygon)) {\n return false;\n }\n }\n return true;\n}\nfunction lineStringWithinPolygons(line, polygons) {\n for (let i = 0; i < polygons.length; i++) {\n if (lineStringWithinPolygon(line, polygons[i]))\n return true;\n }\n return false;\n}\nfunction getTilePolygon(coordinates, bbox, canonical) {\n const polygon = [];\n for (let i = 0; i < coordinates.length; i++) {\n const ring = [];\n for (let j = 0; j < coordinates[i].length; j++) {\n const coord = getTileCoordinates(coordinates[i][j], canonical);\n updateBBox(bbox, coord);\n ring.push(coord);\n }\n polygon.push(ring);\n }\n return polygon;\n}\nfunction getTilePolygons(coordinates, bbox, canonical) {\n const polygons = [];\n for (let i = 0; i < coordinates.length; i++) {\n const polygon = getTilePolygon(coordinates[i], bbox, canonical);\n polygons.push(polygon);\n }\n return polygons;\n}\nfunction updatePoint(p, bbox, polyBBox, worldSize) {\n if (p[0] < polyBBox[0] || p[0] > polyBBox[2]) {\n const halfWorldSize = worldSize * 0.5;\n let shift = (p[0] - polyBBox[0] > halfWorldSize) ? -worldSize : (polyBBox[0] - p[0] > halfWorldSize) ? worldSize : 0;\n if (shift === 0) {\n shift = (p[0] - polyBBox[2] > halfWorldSize) ? -worldSize : (polyBBox[2] - p[0] > halfWorldSize) ? worldSize : 0;\n }\n p[0] += shift;\n }\n updateBBox(bbox, p);\n}\nfunction resetBBox(bbox) {\n bbox[0] = bbox[1] = Infinity;\n bbox[2] = bbox[3] = -Infinity;\n}\nfunction getTilePoints(geometry, pointBBox, polyBBox, canonical) {\n const worldSize = Math.pow(2, canonical.z) * EXTENT;\n const shifts = [canonical.x * EXTENT, canonical.y * EXTENT];\n const tilePoints = [];\n for (const points of geometry) {\n for (const point of points) {\n const p = [point.x + shifts[0], point.y + shifts[1]];\n updatePoint(p, pointBBox, polyBBox, worldSize);\n tilePoints.push(p);\n }\n }\n return tilePoints;\n}\nfunction getTileLines(geometry, lineBBox, polyBBox, canonical) {\n const worldSize = Math.pow(2, canonical.z) * EXTENT;\n const shifts = [canonical.x * EXTENT, canonical.y * EXTENT];\n const tileLines = [];\n for (const line of geometry) {\n const tileLine = [];\n for (const point of line) {\n const p = [point.x + shifts[0], point.y + shifts[1]];\n updateBBox(lineBBox, p);\n tileLine.push(p);\n }\n tileLines.push(tileLine);\n }\n if (lineBBox[2] - lineBBox[0] <= worldSize / 2) {\n resetBBox(lineBBox);\n for (const line of tileLines) {\n for (const p of line) {\n updatePoint(p, lineBBox, polyBBox, worldSize);\n }\n }\n }\n return tileLines;\n}\nfunction pointsWithinPolygons(ctx, polygonGeometry) {\n const pointBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const polyBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const canonical = ctx.canonicalID();\n if (polygonGeometry.type === 'Polygon') {\n const tilePolygon = getTilePolygon(polygonGeometry.coordinates, polyBBox, canonical);\n const tilePoints = getTilePoints(ctx.geometry(), pointBBox, polyBBox, canonical);\n if (!boxWithinBox(pointBBox, polyBBox))\n return false;\n for (const point of tilePoints) {\n if (!pointWithinPolygon(point, tilePolygon))\n return false;\n }\n }\n if (polygonGeometry.type === 'MultiPolygon') {\n const tilePolygons = getTilePolygons(polygonGeometry.coordinates, polyBBox, canonical);\n const tilePoints = getTilePoints(ctx.geometry(), pointBBox, polyBBox, canonical);\n if (!boxWithinBox(pointBBox, polyBBox))\n return false;\n for (const point of tilePoints) {\n if (!pointWithinPolygons(point, tilePolygons))\n return false;\n }\n }\n return true;\n}\nfunction linesWithinPolygons(ctx, polygonGeometry) {\n const lineBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const polyBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const canonical = ctx.canonicalID();\n if (polygonGeometry.type === 'Polygon') {\n const tilePolygon = getTilePolygon(polygonGeometry.coordinates, polyBBox, canonical);\n const tileLines = getTileLines(ctx.geometry(), lineBBox, polyBBox, canonical);\n if (!boxWithinBox(lineBBox, polyBBox))\n return false;\n for (const line of tileLines) {\n if (!lineStringWithinPolygon(line, tilePolygon))\n return false;\n }\n }\n if (polygonGeometry.type === 'MultiPolygon') {\n const tilePolygons = getTilePolygons(polygonGeometry.coordinates, polyBBox, canonical);\n const tileLines = getTileLines(ctx.geometry(), lineBBox, polyBBox, canonical);\n if (!boxWithinBox(lineBBox, polyBBox))\n return false;\n for (const line of tileLines) {\n if (!lineStringWithinPolygons(line, tilePolygons))\n return false;\n }\n }\n return true;\n}\nclass Within {\n constructor(geojson, geometries) {\n this.type = BooleanType;\n this.geojson = geojson;\n this.geometries = geometries;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error(`'within' expression requires exactly one argument, but found ${args.length - 1} instead.`);\n if (isValue(args[1])) {\n const geojson = args[1];\n if (geojson.type === 'FeatureCollection') {\n for (let i = 0; i < geojson.features.length; ++i) {\n const type = geojson.features[i].geometry.type;\n if (type === 'Polygon' || type === 'MultiPolygon') {\n return new Within(geojson, geojson.features[i].geometry);\n }\n }\n }\n else if (geojson.type === 'Feature') {\n const type = geojson.geometry.type;\n if (type === 'Polygon' || type === 'MultiPolygon') {\n return new Within(geojson, geojson.geometry);\n }\n }\n else if (geojson.type === 'Polygon' || geojson.type === 'MultiPolygon') {\n return new Within(geojson, geojson);\n }\n }\n return context.error('\\'within\\' expression requires valid geojson object that contains polygon geometry type.');\n }\n evaluate(ctx) {\n if (ctx.geometry() != null && ctx.canonicalID() != null) {\n if (ctx.geometryType() === 'Point') {\n return pointsWithinPolygons(ctx, this.geometries);\n }\n else if (ctx.geometryType() === 'LineString') {\n return linesWithinPolygons(ctx, this.geometries);\n }\n }\n return false;\n }\n eachChild() { }\n outputDefined() {\n return true;\n }\n}\n\nclass Var {\n constructor(name, boundExpression) {\n this.type = boundExpression.type;\n this.name = name;\n this.boundExpression = boundExpression;\n }\n static parse(args, context) {\n if (args.length !== 2 || typeof args[1] !== 'string')\n return context.error('\\'var\\' expression requires exactly one string literal argument.');\n const name = args[1];\n if (!context.scope.has(name)) {\n return context.error(`Unknown variable \"${name}\". Make sure \"${name}\" has been bound in an enclosing \"let\" expression before using it.`, 1);\n }\n return new Var(name, context.scope.get(name));\n }\n evaluate(ctx) {\n return this.boundExpression.evaluate(ctx);\n }\n eachChild() { }\n outputDefined() {\n return false;\n }\n}\n\nclass CompoundExpression {\n constructor(name, type, evaluate, args) {\n this.name = name;\n this.type = type;\n this._evaluate = evaluate;\n this.args = args;\n }\n evaluate(ctx) {\n return this._evaluate(ctx, this.args);\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return false;\n }\n static parse(args, context) {\n const op = args[0];\n const definition = CompoundExpression.definitions[op];\n if (!definition) {\n return context.error(`Unknown expression \"${op}\". If you wanted a literal array, use [\"literal\", [...]].`, 0);\n }\n // Now check argument types against each signature\n const type = Array.isArray(definition) ?\n definition[0] : definition.type;\n const availableOverloads = Array.isArray(definition) ?\n [[definition[1], definition[2]]] :\n definition.overloads;\n const overloads = availableOverloads.filter(([signature]) => (!Array.isArray(signature) || // varags\n signature.length === args.length - 1 // correct param count\n ));\n let signatureContext = null;\n for (const [params, evaluate] of overloads) {\n // Use a fresh context for each attempted signature so that, if\n // we eventually succeed, we haven't polluted `context.errors`.\n signatureContext = new ParsingContext(context.registry, isExpressionConstant, context.path, null, context.scope);\n // First parse all the args, potentially coercing to the\n // types expected by this overload.\n const parsedArgs = [];\n let argParseFailed = false;\n for (let i = 1; i < args.length; i++) {\n const arg = args[i];\n const expectedType = Array.isArray(params) ?\n params[i - 1] :\n params.type;\n const parsed = signatureContext.parse(arg, 1 + parsedArgs.length, expectedType);\n if (!parsed) {\n argParseFailed = true;\n break;\n }\n parsedArgs.push(parsed);\n }\n if (argParseFailed) {\n // Couldn't coerce args of this overload to expected type, move\n // on to next one.\n continue;\n }\n if (Array.isArray(params)) {\n if (params.length !== parsedArgs.length) {\n signatureContext.error(`Expected ${params.length} arguments, but found ${parsedArgs.length} instead.`);\n continue;\n }\n }\n for (let i = 0; i < parsedArgs.length; i++) {\n const expected = Array.isArray(params) ? params[i] : params.type;\n const arg = parsedArgs[i];\n signatureContext.concat(i + 1).checkSubtype(expected, arg.type);\n }\n if (signatureContext.errors.length === 0) {\n return new CompoundExpression(op, type, evaluate, parsedArgs);\n }\n }\n if (overloads.length === 1) {\n context.errors.push(...signatureContext.errors);\n }\n else {\n const expected = overloads.length ? overloads : availableOverloads;\n const signatures = expected\n .map(([params]) => stringifySignature(params))\n .join(' | ');\n const actualTypes = [];\n // For error message, re-parse arguments without trying to\n // apply any coercions\n for (let i = 1; i < args.length; i++) {\n const parsed = context.parse(args[i], 1 + actualTypes.length);\n if (!parsed)\n return null;\n actualTypes.push(toString$1(parsed.type));\n }\n context.error(`Expected arguments of type ${signatures}, but found (${actualTypes.join(', ')}) instead.`);\n }\n return null;\n }\n static register(registry, definitions) {\n CompoundExpression.definitions = definitions;\n for (const name in definitions) {\n registry[name] = CompoundExpression;\n }\n }\n}\nfunction stringifySignature(signature) {\n if (Array.isArray(signature)) {\n return `(${signature.map(toString$1).join(', ')})`;\n }\n else {\n return `(${toString$1(signature.type)}...)`;\n }\n}\nfunction isExpressionConstant(expression) {\n if (expression instanceof Var) {\n return isExpressionConstant(expression.boundExpression);\n }\n else if (expression instanceof CompoundExpression && expression.name === 'error') {\n return false;\n }\n else if (expression instanceof CollatorExpression) {\n // Although the results of a Collator expression with fixed arguments\n // generally shouldn't change between executions, we can't serialize them\n // as constant expressions because results change based on environment.\n return false;\n }\n else if (expression instanceof Within) {\n return false;\n }\n const isTypeAnnotation = expression instanceof Coercion ||\n expression instanceof Assertion;\n let childrenConstant = true;\n expression.eachChild(child => {\n // We can _almost_ assume that if `expressions` children are constant,\n // they would already have been evaluated to Literal values when they\n // were parsed. Type annotations are the exception, because they might\n // have been inferred and added after a child was parsed.\n // So we recurse into isConstant() for the children of type annotations,\n // but otherwise simply check whether they are Literals.\n if (isTypeAnnotation) {\n childrenConstant = childrenConstant && isExpressionConstant(child);\n }\n else {\n childrenConstant = childrenConstant && child instanceof Literal;\n }\n });\n if (!childrenConstant) {\n return false;\n }\n return isFeatureConstant(expression) &&\n isGlobalPropertyConstant(expression, ['zoom', 'heatmap-density', 'line-progress', 'accumulated', 'is-supported-script']);\n}\nfunction isFeatureConstant(e) {\n if (e instanceof CompoundExpression) {\n if (e.name === 'get' && e.args.length === 1) {\n return false;\n }\n else if (e.name === 'feature-state') {\n return false;\n }\n else if (e.name === 'has' && e.args.length === 1) {\n return false;\n }\n else if (e.name === 'properties' ||\n e.name === 'geometry-type' ||\n e.name === 'id') {\n return false;\n }\n else if (/^filter-/.test(e.name)) {\n return false;\n }\n }\n if (e instanceof Within) {\n return false;\n }\n let result = true;\n e.eachChild(arg => {\n if (result && !isFeatureConstant(arg)) {\n result = false;\n }\n });\n return result;\n}\nfunction isStateConstant(e) {\n if (e instanceof CompoundExpression) {\n if (e.name === 'feature-state') {\n return false;\n }\n }\n let result = true;\n e.eachChild(arg => {\n if (result && !isStateConstant(arg)) {\n result = false;\n }\n });\n return result;\n}\nfunction isGlobalPropertyConstant(e, properties) {\n if (e instanceof CompoundExpression && properties.indexOf(e.name) >= 0) {\n return false;\n }\n let result = true;\n e.eachChild((arg) => {\n if (result && !isGlobalPropertyConstant(arg, properties)) {\n result = false;\n }\n });\n return result;\n}\n\n/**\n * Returns the index of the last stop <= input, or 0 if it doesn't exist.\n * @private\n */\nfunction findStopLessThanOrEqualTo(stops, input) {\n const lastIndex = stops.length - 1;\n let lowerIndex = 0;\n let upperIndex = lastIndex;\n let currentIndex = 0;\n let currentValue, nextValue;\n while (lowerIndex <= upperIndex) {\n currentIndex = Math.floor((lowerIndex + upperIndex) / 2);\n currentValue = stops[currentIndex];\n nextValue = stops[currentIndex + 1];\n if (currentValue <= input) {\n if (currentIndex === lastIndex || input < nextValue) { // Search complete\n return currentIndex;\n }\n lowerIndex = currentIndex + 1;\n }\n else if (currentValue > input) {\n upperIndex = currentIndex - 1;\n }\n else {\n throw new RuntimeError('Input is not a number.');\n }\n }\n return 0;\n}\n\nclass Step {\n constructor(type, input, stops) {\n this.type = type;\n this.input = input;\n this.labels = [];\n this.outputs = [];\n for (const [label, expression] of stops) {\n this.labels.push(label);\n this.outputs.push(expression);\n }\n }\n static parse(args, context) {\n if (args.length - 1 < 4) {\n return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`);\n }\n if ((args.length - 1) % 2 !== 0) {\n return context.error('Expected an even number of arguments.');\n }\n const input = context.parse(args[1], 1, NumberType);\n if (!input)\n return null;\n const stops = [];\n let outputType = null;\n if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n for (let i = 1; i < args.length; i += 2) {\n const label = i === 1 ? -Infinity : args[i];\n const value = args[i + 1];\n const labelKey = i;\n const valueKey = i + 1;\n if (typeof label !== 'number') {\n return context.error('Input/output pairs for \"step\" expressions must be defined using literal numeric values (not computed expressions) for the input values.', labelKey);\n }\n if (stops.length && stops[stops.length - 1][0] >= label) {\n return context.error('Input/output pairs for \"step\" expressions must be arranged with input values in strictly ascending order.', labelKey);\n }\n const parsed = context.parse(value, valueKey, outputType);\n if (!parsed)\n return null;\n outputType = outputType || parsed.type;\n stops.push([label, parsed]);\n }\n return new Step(outputType, input, stops);\n }\n evaluate(ctx) {\n const labels = this.labels;\n const outputs = this.outputs;\n if (labels.length === 1) {\n return outputs[0].evaluate(ctx);\n }\n const value = this.input.evaluate(ctx);\n if (value <= labels[0]) {\n return outputs[0].evaluate(ctx);\n }\n const stopCount = labels.length;\n if (value >= labels[stopCount - 1]) {\n return outputs[stopCount - 1].evaluate(ctx);\n }\n const index = findStopLessThanOrEqualTo(labels, value);\n return outputs[index].evaluate(ctx);\n }\n eachChild(fn) {\n fn(this.input);\n for (const expression of this.outputs) {\n fn(expression);\n }\n }\n outputDefined() {\n return this.outputs.every(out => out.outputDefined());\n }\n}\n\n/**\n * Checks whether the specified color space is one of the supported interpolation color spaces.\n *\n * @param colorSpace Color space key to verify.\n * @returns `true` if the specified color space is one of the supported\n * interpolation color spaces, `false` otherwise\n */\nfunction isSupportedInterpolationColorSpace(colorSpace) {\n return colorSpace === 'rgb' || colorSpace === 'hcl' || colorSpace === 'lab';\n}\n/**\n * @param interpolationType Interpolation type\n * @returns interpolation fn\n * @deprecated use `interpolate[type]` instead\n */\nconst interpolateFactory = (interpolationType) => {\n switch (interpolationType) {\n case 'number': return number;\n case 'color': return color;\n case 'array': return array;\n case 'padding': return padding;\n case 'variableAnchorOffsetCollection': return variableAnchorOffsetCollection;\n }\n};\nfunction number(from, to, t) {\n return from + t * (to - from);\n}\nfunction color(from, to, t, spaceKey = 'rgb') {\n switch (spaceKey) {\n case 'rgb': {\n const [r, g, b, alpha] = array(from.rgb, to.rgb, t);\n return new Color(r, g, b, alpha, false);\n }\n case 'hcl': {\n const [hue0, chroma0, light0, alphaF] = from.hcl;\n const [hue1, chroma1, light1, alphaT] = to.hcl;\n // https://github.com/gka/chroma.js/blob/cd1b3c0926c7a85cbdc3b1453b3a94006de91a92/src/interpolator/_hsx.js\n let hue, chroma;\n if (!isNaN(hue0) && !isNaN(hue1)) {\n let dh = hue1 - hue0;\n if (hue1 > hue0 && dh > 180) {\n dh -= 360;\n }\n else if (hue1 < hue0 && hue0 - hue1 > 180) {\n dh += 360;\n }\n hue = hue0 + t * dh;\n }\n else if (!isNaN(hue0)) {\n hue = hue0;\n if (light1 === 1 || light1 === 0)\n chroma = chroma0;\n }\n else if (!isNaN(hue1)) {\n hue = hue1;\n if (light0 === 1 || light0 === 0)\n chroma = chroma1;\n }\n else {\n hue = NaN;\n }\n const [r, g, b, alpha] = hclToRgb([\n hue,\n chroma !== null && chroma !== void 0 ? chroma : number(chroma0, chroma1, t),\n number(light0, light1, t),\n number(alphaF, alphaT, t),\n ]);\n return new Color(r, g, b, alpha, false);\n }\n case 'lab': {\n const [r, g, b, alpha] = labToRgb(array(from.lab, to.lab, t));\n return new Color(r, g, b, alpha, false);\n }\n }\n}\nfunction array(from, to, t) {\n return from.map((d, i) => {\n return number(d, to[i], t);\n });\n}\nfunction padding(from, to, t) {\n return new Padding(array(from.values, to.values, t));\n}\nfunction variableAnchorOffsetCollection(from, to, t) {\n const fromValues = from.values;\n const toValues = to.values;\n if (fromValues.length !== toValues.length) {\n throw new RuntimeError(`Cannot interpolate values of different length. from: ${from.toString()}, to: ${to.toString()}`);\n }\n const output = [];\n for (let i = 0; i < fromValues.length; i += 2) {\n // Anchor entries must match\n if (fromValues[i] !== toValues[i]) {\n throw new RuntimeError(`Cannot interpolate values containing mismatched anchors. from[${i}]: ${fromValues[i]}, to[${i}]: ${toValues[i]}`);\n }\n output.push(fromValues[i]);\n // Interpolate the offset values for each anchor\n const [fx, fy] = fromValues[i + 1];\n const [tx, ty] = toValues[i + 1];\n output.push([number(fx, tx, t), number(fy, ty, t)]);\n }\n return new VariableAnchorOffsetCollection(output);\n}\nconst interpolate = {\n number,\n color,\n array,\n padding,\n variableAnchorOffsetCollection\n};\n\nclass Interpolate {\n constructor(type, operator, interpolation, input, stops) {\n this.type = type;\n this.operator = operator;\n this.interpolation = interpolation;\n this.input = input;\n this.labels = [];\n this.outputs = [];\n for (const [label, expression] of stops) {\n this.labels.push(label);\n this.outputs.push(expression);\n }\n }\n static interpolationFactor(interpolation, input, lower, upper) {\n let t = 0;\n if (interpolation.name === 'exponential') {\n t = exponentialInterpolation(input, interpolation.base, lower, upper);\n }\n else if (interpolation.name === 'linear') {\n t = exponentialInterpolation(input, 1, lower, upper);\n }\n else if (interpolation.name === 'cubic-bezier') {\n const c = interpolation.controlPoints;\n const ub = new UnitBezier(c[0], c[1], c[2], c[3]);\n t = ub.solve(exponentialInterpolation(input, 1, lower, upper));\n }\n return t;\n }\n static parse(args, context) {\n let [operator, interpolation, input, ...rest] = args;\n if (!Array.isArray(interpolation) || interpolation.length === 0) {\n return context.error('Expected an interpolation type expression.', 1);\n }\n if (interpolation[0] === 'linear') {\n interpolation = { name: 'linear' };\n }\n else if (interpolation[0] === 'exponential') {\n const base = interpolation[1];\n if (typeof base !== 'number')\n return context.error('Exponential interpolation requires a numeric base.', 1, 1);\n interpolation = {\n name: 'exponential',\n base\n };\n }\n else if (interpolation[0] === 'cubic-bezier') {\n const controlPoints = interpolation.slice(1);\n if (controlPoints.length !== 4 ||\n controlPoints.some(t => typeof t !== 'number' || t < 0 || t > 1)) {\n return context.error('Cubic bezier interpolation requires four numeric arguments with values between 0 and 1.', 1);\n }\n interpolation = {\n name: 'cubic-bezier',\n controlPoints: controlPoints\n };\n }\n else {\n return context.error(`Unknown interpolation type ${String(interpolation[0])}`, 1, 0);\n }\n if (args.length - 1 < 4) {\n return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`);\n }\n if ((args.length - 1) % 2 !== 0) {\n return context.error('Expected an even number of arguments.');\n }\n input = context.parse(input, 2, NumberType);\n if (!input)\n return null;\n const stops = [];\n let outputType = null;\n if (operator === 'interpolate-hcl' || operator === 'interpolate-lab') {\n outputType = ColorType;\n }\n else if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n for (let i = 0; i < rest.length; i += 2) {\n const label = rest[i];\n const value = rest[i + 1];\n const labelKey = i + 3;\n const valueKey = i + 4;\n if (typeof label !== 'number') {\n return context.error('Input/output pairs for \"interpolate\" expressions must be defined using literal numeric values (not computed expressions) for the input values.', labelKey);\n }\n if (stops.length && stops[stops.length - 1][0] >= label) {\n return context.error('Input/output pairs for \"interpolate\" expressions must be arranged with input values in strictly ascending order.', labelKey);\n }\n const parsed = context.parse(value, valueKey, outputType);\n if (!parsed)\n return null;\n outputType = outputType || parsed.type;\n stops.push([label, parsed]);\n }\n if (!verifyType(outputType, NumberType) &&\n !verifyType(outputType, ColorType) &&\n !verifyType(outputType, PaddingType) &&\n !verifyType(outputType, VariableAnchorOffsetCollectionType) &&\n !verifyType(outputType, array$1(NumberType))) {\n return context.error(`Type ${toString$1(outputType)} is not interpolatable.`);\n }\n return new Interpolate(outputType, operator, interpolation, input, stops);\n }\n evaluate(ctx) {\n const labels = this.labels;\n const outputs = this.outputs;\n if (labels.length === 1) {\n return outputs[0].evaluate(ctx);\n }\n const value = this.input.evaluate(ctx);\n if (value <= labels[0]) {\n return outputs[0].evaluate(ctx);\n }\n const stopCount = labels.length;\n if (value >= labels[stopCount - 1]) {\n return outputs[stopCount - 1].evaluate(ctx);\n }\n const index = findStopLessThanOrEqualTo(labels, value);\n const lower = labels[index];\n const upper = labels[index + 1];\n const t = Interpolate.interpolationFactor(this.interpolation, value, lower, upper);\n const outputLower = outputs[index].evaluate(ctx);\n const outputUpper = outputs[index + 1].evaluate(ctx);\n switch (this.operator) {\n case 'interpolate':\n return interpolate[this.type.kind](outputLower, outputUpper, t);\n case 'interpolate-hcl':\n return interpolate.color(outputLower, outputUpper, t, 'hcl');\n case 'interpolate-lab':\n return interpolate.color(outputLower, outputUpper, t, 'lab');\n }\n }\n eachChild(fn) {\n fn(this.input);\n for (const expression of this.outputs) {\n fn(expression);\n }\n }\n outputDefined() {\n return this.outputs.every(out => out.outputDefined());\n }\n}\n/**\n * Returns a ratio that can be used to interpolate between exponential function\n * stops.\n * How it works: Two consecutive stop values define a (scaled and shifted) exponential function `f(x) = a * base^x + b`, where `base` is the user-specified base,\n * and `a` and `b` are constants affording sufficient degrees of freedom to fit\n * the function to the given stops.\n *\n * Here's a bit of algebra that lets us compute `f(x)` directly from the stop\n * values without explicitly solving for `a` and `b`:\n *\n * First stop value: `f(x0) = y0 = a * base^x0 + b`\n * Second stop value: `f(x1) = y1 = a * base^x1 + b`\n * => `y1 - y0 = a(base^x1 - base^x0)`\n * => `a = (y1 - y0)/(base^x1 - base^x0)`\n *\n * Desired value: `f(x) = y = a * base^x + b`\n * => `f(x) = y0 + a * (base^x - base^x0)`\n *\n * From the above, we can replace the `a` in `a * (base^x - base^x0)` and do a\n * little algebra:\n * ```\n * a * (base^x - base^x0) = (y1 - y0)/(base^x1 - base^x0) * (base^x - base^x0)\n * = (y1 - y0) * (base^x - base^x0) / (base^x1 - base^x0)\n * ```\n *\n * If we let `(base^x - base^x0) / (base^x1 base^x0)`, then we have\n * `f(x) = y0 + (y1 - y0) * ratio`. In other words, `ratio` may be treated as\n * an interpolation factor between the two stops' output values.\n *\n * (Note: a slightly different form for `ratio`,\n * `(base^(x-x0) - 1) / (base^(x1-x0) - 1) `, is equivalent, but requires fewer\n * expensive `Math.pow()` operations.)\n *\n * @private\n*/\nfunction exponentialInterpolation(input, base, lowerValue, upperValue) {\n const difference = upperValue - lowerValue;\n const progress = input - lowerValue;\n if (difference === 0) {\n return 0;\n }\n else if (base === 1) {\n return progress / difference;\n }\n else {\n return (Math.pow(base, progress) - 1) / (Math.pow(base, difference) - 1);\n }\n}\n\nclass Coalesce {\n constructor(type, args) {\n this.type = type;\n this.args = args;\n }\n static parse(args, context) {\n if (args.length < 2) {\n return context.error('Expectected at least one argument.');\n }\n let outputType = null;\n const expectedType = context.expectedType;\n if (expectedType && expectedType.kind !== 'value') {\n outputType = expectedType;\n }\n const parsedArgs = [];\n for (const arg of args.slice(1)) {\n const parsed = context.parse(arg, 1 + parsedArgs.length, outputType, undefined, { typeAnnotation: 'omit' });\n if (!parsed)\n return null;\n outputType = outputType || parsed.type;\n parsedArgs.push(parsed);\n }\n if (!outputType)\n throw new Error('No output type');\n // Above, we parse arguments without inferred type annotation so that\n // they don't produce a runtime error for `null` input, which would\n // preempt the desired null-coalescing behavior.\n // Thus, if any of our arguments would have needed an annotation, we\n // need to wrap the enclosing coalesce expression with it instead.\n const needsAnnotation = expectedType &&\n parsedArgs.some(arg => checkSubtype(expectedType, arg.type));\n return needsAnnotation ?\n new Coalesce(ValueType, parsedArgs) :\n new Coalesce(outputType, parsedArgs);\n }\n evaluate(ctx) {\n let result = null;\n let argCount = 0;\n let requestedImageName;\n for (const arg of this.args) {\n argCount++;\n result = arg.evaluate(ctx);\n // we need to keep track of the first requested image in a coalesce statement\n // if coalesce can't find a valid image, we return the first image name so styleimagemissing can fire\n if (result && result instanceof ResolvedImage && !result.available) {\n if (!requestedImageName) {\n requestedImageName = result.name;\n }\n result = null;\n if (argCount === this.args.length) {\n result = requestedImageName;\n }\n }\n if (result !== null)\n break;\n }\n return result;\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return this.args.every(arg => arg.outputDefined());\n }\n}\n\nclass Let {\n constructor(bindings, result) {\n this.type = result.type;\n this.bindings = [].concat(bindings);\n this.result = result;\n }\n evaluate(ctx) {\n return this.result.evaluate(ctx);\n }\n eachChild(fn) {\n for (const binding of this.bindings) {\n fn(binding[1]);\n }\n fn(this.result);\n }\n static parse(args, context) {\n if (args.length < 4)\n return context.error(`Expected at least 3 arguments, but found ${args.length - 1} instead.`);\n const bindings = [];\n for (let i = 1; i < args.length - 1; i += 2) {\n const name = args[i];\n if (typeof name !== 'string') {\n return context.error(`Expected string, but found ${typeof name} instead.`, i);\n }\n if (/[^a-zA-Z0-9_]/.test(name)) {\n return context.error('Variable names must contain only alphanumeric characters or \\'_\\'.', i);\n }\n const value = context.parse(args[i + 1], i + 1);\n if (!value)\n return null;\n bindings.push([name, value]);\n }\n const result = context.parse(args[args.length - 1], args.length - 1, context.expectedType, bindings);\n if (!result)\n return null;\n return new Let(bindings, result);\n }\n outputDefined() {\n return this.result.outputDefined();\n }\n}\n\nclass At {\n constructor(type, index, input) {\n this.type = type;\n this.index = index;\n this.input = input;\n }\n static parse(args, context) {\n if (args.length !== 3)\n return context.error(`Expected 2 arguments, but found ${args.length - 1} instead.`);\n const index = context.parse(args[1], 1, NumberType);\n const input = context.parse(args[2], 2, array$1(context.expectedType || ValueType));\n if (!index || !input)\n return null;\n const t = input.type;\n return new At(t.itemType, index, input);\n }\n evaluate(ctx) {\n const index = this.index.evaluate(ctx);\n const array = this.input.evaluate(ctx);\n if (index < 0) {\n throw new RuntimeError(`Array index out of bounds: ${index} < 0.`);\n }\n if (index >= array.length) {\n throw new RuntimeError(`Array index out of bounds: ${index} > ${array.length - 1}.`);\n }\n if (index !== Math.floor(index)) {\n throw new RuntimeError(`Array index must be an integer, but found ${index} instead.`);\n }\n return array[index];\n }\n eachChild(fn) {\n fn(this.index);\n fn(this.input);\n }\n outputDefined() {\n return false;\n }\n}\n\nclass In {\n constructor(needle, haystack) {\n this.type = BooleanType;\n this.needle = needle;\n this.haystack = haystack;\n }\n static parse(args, context) {\n if (args.length !== 3) {\n return context.error(`Expected 2 arguments, but found ${args.length - 1} instead.`);\n }\n const needle = context.parse(args[1], 1, ValueType);\n const haystack = context.parse(args[2], 2, ValueType);\n if (!needle || !haystack)\n return null;\n if (!isValidType(needle.type, [BooleanType, StringType, NumberType, NullType, ValueType])) {\n return context.error(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(needle.type)} instead`);\n }\n return new In(needle, haystack);\n }\n evaluate(ctx) {\n const needle = this.needle.evaluate(ctx);\n const haystack = this.haystack.evaluate(ctx);\n if (!haystack)\n return false;\n if (!isValidNativeType(needle, ['boolean', 'string', 'number', 'null'])) {\n throw new RuntimeError(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(typeOf(needle))} instead.`);\n }\n if (!isValidNativeType(haystack, ['string', 'array'])) {\n throw new RuntimeError(`Expected second argument to be of type array or string, but found ${toString$1(typeOf(haystack))} instead.`);\n }\n return haystack.indexOf(needle) >= 0;\n }\n eachChild(fn) {\n fn(this.needle);\n fn(this.haystack);\n }\n outputDefined() {\n return true;\n }\n}\n\nclass IndexOf {\n constructor(needle, haystack, fromIndex) {\n this.type = NumberType;\n this.needle = needle;\n this.haystack = haystack;\n this.fromIndex = fromIndex;\n }\n static parse(args, context) {\n if (args.length <= 2 || args.length >= 5) {\n return context.error(`Expected 3 or 4 arguments, but found ${args.length - 1} instead.`);\n }\n const needle = context.parse(args[1], 1, ValueType);\n const haystack = context.parse(args[2], 2, ValueType);\n if (!needle || !haystack)\n return null;\n if (!isValidType(needle.type, [BooleanType, StringType, NumberType, NullType, ValueType])) {\n return context.error(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(needle.type)} instead`);\n }\n if (args.length === 4) {\n const fromIndex = context.parse(args[3], 3, NumberType);\n if (!fromIndex)\n return null;\n return new IndexOf(needle, haystack, fromIndex);\n }\n else {\n return new IndexOf(needle, haystack);\n }\n }\n evaluate(ctx) {\n const needle = this.needle.evaluate(ctx);\n const haystack = this.haystack.evaluate(ctx);\n if (!isValidNativeType(needle, ['boolean', 'string', 'number', 'null'])) {\n throw new RuntimeError(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(typeOf(needle))} instead.`);\n }\n if (!isValidNativeType(haystack, ['string', 'array'])) {\n throw new RuntimeError(`Expected second argument to be of type array or string, but found ${toString$1(typeOf(haystack))} instead.`);\n }\n if (this.fromIndex) {\n const fromIndex = this.fromIndex.evaluate(ctx);\n return haystack.indexOf(needle, fromIndex);\n }\n return haystack.indexOf(needle);\n }\n eachChild(fn) {\n fn(this.needle);\n fn(this.haystack);\n if (this.fromIndex) {\n fn(this.fromIndex);\n }\n }\n outputDefined() {\n return false;\n }\n}\n\nclass Match {\n constructor(inputType, outputType, input, cases, outputs, otherwise) {\n this.inputType = inputType;\n this.type = outputType;\n this.input = input;\n this.cases = cases;\n this.outputs = outputs;\n this.otherwise = otherwise;\n }\n static parse(args, context) {\n if (args.length < 5)\n return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`);\n if (args.length % 2 !== 1)\n return context.error('Expected an even number of arguments.');\n let inputType;\n let outputType;\n if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n const cases = {};\n const outputs = [];\n for (let i = 2; i < args.length - 1; i += 2) {\n let labels = args[i];\n const value = args[i + 1];\n if (!Array.isArray(labels)) {\n labels = [labels];\n }\n const labelContext = context.concat(i);\n if (labels.length === 0) {\n return labelContext.error('Expected at least one branch label.');\n }\n for (const label of labels) {\n if (typeof label !== 'number' && typeof label !== 'string') {\n return labelContext.error('Branch labels must be numbers or strings.');\n }\n else if (typeof label === 'number' && Math.abs(label) > Number.MAX_SAFE_INTEGER) {\n return labelContext.error(`Branch labels must be integers no larger than ${Number.MAX_SAFE_INTEGER}.`);\n }\n else if (typeof label === 'number' && Math.floor(label) !== label) {\n return labelContext.error('Numeric branch labels must be integer values.');\n }\n else if (!inputType) {\n inputType = typeOf(label);\n }\n else if (labelContext.checkSubtype(inputType, typeOf(label))) {\n return null;\n }\n if (typeof cases[String(label)] !== 'undefined') {\n return labelContext.error('Branch labels must be unique.');\n }\n cases[String(label)] = outputs.length;\n }\n const result = context.parse(value, i, outputType);\n if (!result)\n return null;\n outputType = outputType || result.type;\n outputs.push(result);\n }\n const input = context.parse(args[1], 1, ValueType);\n if (!input)\n return null;\n const otherwise = context.parse(args[args.length - 1], args.length - 1, outputType);\n if (!otherwise)\n return null;\n if (input.type.kind !== 'value' && context.concat(1).checkSubtype(inputType, input.type)) {\n return null;\n }\n return new Match(inputType, outputType, input, cases, outputs, otherwise);\n }\n evaluate(ctx) {\n const input = this.input.evaluate(ctx);\n const output = (typeOf(input) === this.inputType && this.outputs[this.cases[input]]) || this.otherwise;\n return output.evaluate(ctx);\n }\n eachChild(fn) {\n fn(this.input);\n this.outputs.forEach(fn);\n fn(this.otherwise);\n }\n outputDefined() {\n return this.outputs.every(out => out.outputDefined()) && this.otherwise.outputDefined();\n }\n}\n\nclass Case {\n constructor(type, branches, otherwise) {\n this.type = type;\n this.branches = branches;\n this.otherwise = otherwise;\n }\n static parse(args, context) {\n if (args.length < 4)\n return context.error(`Expected at least 3 arguments, but found only ${args.length - 1}.`);\n if (args.length % 2 !== 0)\n return context.error('Expected an odd number of arguments.');\n let outputType;\n if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n const branches = [];\n for (let i = 1; i < args.length - 1; i += 2) {\n const test = context.parse(args[i], i, BooleanType);\n if (!test)\n return null;\n const result = context.parse(args[i + 1], i + 1, outputType);\n if (!result)\n return null;\n branches.push([test, result]);\n outputType = outputType || result.type;\n }\n const otherwise = context.parse(args[args.length - 1], args.length - 1, outputType);\n if (!otherwise)\n return null;\n if (!outputType)\n throw new Error('Can\\'t infer output type');\n return new Case(outputType, branches, otherwise);\n }\n evaluate(ctx) {\n for (const [test, expression] of this.branches) {\n if (test.evaluate(ctx)) {\n return expression.evaluate(ctx);\n }\n }\n return this.otherwise.evaluate(ctx);\n }\n eachChild(fn) {\n for (const [test, expression] of this.branches) {\n fn(test);\n fn(expression);\n }\n fn(this.otherwise);\n }\n outputDefined() {\n return this.branches.every(([_, out]) => out.outputDefined()) && this.otherwise.outputDefined();\n }\n}\n\nclass Slice {\n constructor(type, input, beginIndex, endIndex) {\n this.type = type;\n this.input = input;\n this.beginIndex = beginIndex;\n this.endIndex = endIndex;\n }\n static parse(args, context) {\n if (args.length <= 2 || args.length >= 5) {\n return context.error(`Expected 3 or 4 arguments, but found ${args.length - 1} instead.`);\n }\n const input = context.parse(args[1], 1, ValueType);\n const beginIndex = context.parse(args[2], 2, NumberType);\n if (!input || !beginIndex)\n return null;\n if (!isValidType(input.type, [array$1(ValueType), StringType, ValueType])) {\n return context.error(`Expected first argument to be of type array or string, but found ${toString$1(input.type)} instead`);\n }\n if (args.length === 4) {\n const endIndex = context.parse(args[3], 3, NumberType);\n if (!endIndex)\n return null;\n return new Slice(input.type, input, beginIndex, endIndex);\n }\n else {\n return new Slice(input.type, input, beginIndex);\n }\n }\n evaluate(ctx) {\n const input = this.input.evaluate(ctx);\n const beginIndex = this.beginIndex.evaluate(ctx);\n if (!isValidNativeType(input, ['string', 'array'])) {\n throw new RuntimeError(`Expected first argument to be of type array or string, but found ${toString$1(typeOf(input))} instead.`);\n }\n if (this.endIndex) {\n const endIndex = this.endIndex.evaluate(ctx);\n return input.slice(beginIndex, endIndex);\n }\n return input.slice(beginIndex);\n }\n eachChild(fn) {\n fn(this.input);\n fn(this.beginIndex);\n if (this.endIndex) {\n fn(this.endIndex);\n }\n }\n outputDefined() {\n return false;\n }\n}\n\nfunction isComparableType(op, type) {\n if (op === '==' || op === '!=') {\n // equality operator\n return type.kind === 'boolean' ||\n type.kind === 'string' ||\n type.kind === 'number' ||\n type.kind === 'null' ||\n type.kind === 'value';\n }\n else {\n // ordering operator\n return type.kind === 'string' ||\n type.kind === 'number' ||\n type.kind === 'value';\n }\n}\nfunction eq(ctx, a, b) { return a === b; }\nfunction neq(ctx, a, b) { return a !== b; }\nfunction lt(ctx, a, b) { return a < b; }\nfunction gt(ctx, a, b) { return a > b; }\nfunction lteq(ctx, a, b) { return a <= b; }\nfunction gteq(ctx, a, b) { return a >= b; }\nfunction eqCollate(ctx, a, b, c) { return c.compare(a, b) === 0; }\nfunction neqCollate(ctx, a, b, c) { return !eqCollate(ctx, a, b, c); }\nfunction ltCollate(ctx, a, b, c) { return c.compare(a, b) < 0; }\nfunction gtCollate(ctx, a, b, c) { return c.compare(a, b) > 0; }\nfunction lteqCollate(ctx, a, b, c) { return c.compare(a, b) <= 0; }\nfunction gteqCollate(ctx, a, b, c) { return c.compare(a, b) >= 0; }\n/**\n * Special form for comparison operators, implementing the signatures:\n * - (T, T, ?Collator) => boolean\n * - (T, value, ?Collator) => boolean\n * - (value, T, ?Collator) => boolean\n *\n * For inequalities, T must be either value, string, or number. For ==/!=, it\n * can also be boolean or null.\n *\n * Equality semantics are equivalent to Javascript's strict equality (===/!==)\n * -- i.e., when the arguments' types don't match, == evaluates to false, != to\n * true.\n *\n * When types don't match in an ordering comparison, a runtime error is thrown.\n *\n * @private\n */\nfunction makeComparison(op, compareBasic, compareWithCollator) {\n const isOrderComparison = op !== '==' && op !== '!=';\n return class Comparison {\n constructor(lhs, rhs, collator) {\n this.type = BooleanType;\n this.lhs = lhs;\n this.rhs = rhs;\n this.collator = collator;\n this.hasUntypedArgument = lhs.type.kind === 'value' || rhs.type.kind === 'value';\n }\n static parse(args, context) {\n if (args.length !== 3 && args.length !== 4)\n return context.error('Expected two or three arguments.');\n const op = args[0];\n let lhs = context.parse(args[1], 1, ValueType);\n if (!lhs)\n return null;\n if (!isComparableType(op, lhs.type)) {\n return context.concat(1).error(`\"${op}\" comparisons are not supported for type '${toString$1(lhs.type)}'.`);\n }\n let rhs = context.parse(args[2], 2, ValueType);\n if (!rhs)\n return null;\n if (!isComparableType(op, rhs.type)) {\n return context.concat(2).error(`\"${op}\" comparisons are not supported for type '${toString$1(rhs.type)}'.`);\n }\n if (lhs.type.kind !== rhs.type.kind &&\n lhs.type.kind !== 'value' &&\n rhs.type.kind !== 'value') {\n return context.error(`Cannot compare types '${toString$1(lhs.type)}' and '${toString$1(rhs.type)}'.`);\n }\n if (isOrderComparison) {\n // typing rules specific to less/greater than operators\n if (lhs.type.kind === 'value' && rhs.type.kind !== 'value') {\n // (value, T)\n lhs = new Assertion(rhs.type, [lhs]);\n }\n else if (lhs.type.kind !== 'value' && rhs.type.kind === 'value') {\n // (T, value)\n rhs = new Assertion(lhs.type, [rhs]);\n }\n }\n let collator = null;\n if (args.length === 4) {\n if (lhs.type.kind !== 'string' &&\n rhs.type.kind !== 'string' &&\n lhs.type.kind !== 'value' &&\n rhs.type.kind !== 'value') {\n return context.error('Cannot use collator to compare non-string types.');\n }\n collator = context.parse(args[3], 3, CollatorType);\n if (!collator)\n return null;\n }\n return new Comparison(lhs, rhs, collator);\n }\n evaluate(ctx) {\n const lhs = this.lhs.evaluate(ctx);\n const rhs = this.rhs.evaluate(ctx);\n if (isOrderComparison && this.hasUntypedArgument) {\n const lt = typeOf(lhs);\n const rt = typeOf(rhs);\n // check that type is string or number, and equal\n if (lt.kind !== rt.kind || !(lt.kind === 'string' || lt.kind === 'number')) {\n throw new RuntimeError(`Expected arguments for \"${op}\" to be (string, string) or (number, number), but found (${lt.kind}, ${rt.kind}) instead.`);\n }\n }\n if (this.collator && !isOrderComparison && this.hasUntypedArgument) {\n const lt = typeOf(lhs);\n const rt = typeOf(rhs);\n if (lt.kind !== 'string' || rt.kind !== 'string') {\n return compareBasic(ctx, lhs, rhs);\n }\n }\n return this.collator ?\n compareWithCollator(ctx, lhs, rhs, this.collator.evaluate(ctx)) :\n compareBasic(ctx, lhs, rhs);\n }\n eachChild(fn) {\n fn(this.lhs);\n fn(this.rhs);\n if (this.collator) {\n fn(this.collator);\n }\n }\n outputDefined() {\n return true;\n }\n };\n}\nconst Equals = makeComparison('==', eq, eqCollate);\nconst NotEquals = makeComparison('!=', neq, neqCollate);\nconst LessThan = makeComparison('<', lt, ltCollate);\nconst GreaterThan = makeComparison('>', gt, gtCollate);\nconst LessThanOrEqual = makeComparison('<=', lteq, lteqCollate);\nconst GreaterThanOrEqual = makeComparison('>=', gteq, gteqCollate);\n\nclass NumberFormat {\n constructor(number, locale, currency, minFractionDigits, maxFractionDigits) {\n this.type = StringType;\n this.number = number;\n this.locale = locale;\n this.currency = currency;\n this.minFractionDigits = minFractionDigits;\n this.maxFractionDigits = maxFractionDigits;\n }\n static parse(args, context) {\n if (args.length !== 3)\n return context.error('Expected two arguments.');\n const number = context.parse(args[1], 1, NumberType);\n if (!number)\n return null;\n const options = args[2];\n if (typeof options !== 'object' || Array.isArray(options))\n return context.error('NumberFormat options argument must be an object.');\n let locale = null;\n if (options['locale']) {\n locale = context.parse(options['locale'], 1, StringType);\n if (!locale)\n return null;\n }\n let currency = null;\n if (options['currency']) {\n currency = context.parse(options['currency'], 1, StringType);\n if (!currency)\n return null;\n }\n let minFractionDigits = null;\n if (options['min-fraction-digits']) {\n minFractionDigits = context.parse(options['min-fraction-digits'], 1, NumberType);\n if (!minFractionDigits)\n return null;\n }\n let maxFractionDigits = null;\n if (options['max-fraction-digits']) {\n maxFractionDigits = context.parse(options['max-fraction-digits'], 1, NumberType);\n if (!maxFractionDigits)\n return null;\n }\n return new NumberFormat(number, locale, currency, minFractionDigits, maxFractionDigits);\n }\n evaluate(ctx) {\n return new Intl.NumberFormat(this.locale ? this.locale.evaluate(ctx) : [], {\n style: this.currency ? 'currency' : 'decimal',\n currency: this.currency ? this.currency.evaluate(ctx) : undefined,\n minimumFractionDigits: this.minFractionDigits ? this.minFractionDigits.evaluate(ctx) : undefined,\n maximumFractionDigits: this.maxFractionDigits ? this.maxFractionDigits.evaluate(ctx) : undefined,\n }).format(this.number.evaluate(ctx));\n }\n eachChild(fn) {\n fn(this.number);\n if (this.locale) {\n fn(this.locale);\n }\n if (this.currency) {\n fn(this.currency);\n }\n if (this.minFractionDigits) {\n fn(this.minFractionDigits);\n }\n if (this.maxFractionDigits) {\n fn(this.maxFractionDigits);\n }\n }\n outputDefined() {\n return false;\n }\n}\n\nclass FormatExpression {\n constructor(sections) {\n this.type = FormattedType;\n this.sections = sections;\n }\n static parse(args, context) {\n if (args.length < 2) {\n return context.error('Expected at least one argument.');\n }\n const firstArg = args[1];\n if (!Array.isArray(firstArg) && typeof firstArg === 'object') {\n return context.error('First argument must be an image or text section.');\n }\n const sections = [];\n let nextTokenMayBeObject = false;\n for (let i = 1; i <= args.length - 1; ++i) {\n const arg = args[i];\n if (nextTokenMayBeObject && typeof arg === 'object' && !Array.isArray(arg)) {\n nextTokenMayBeObject = false;\n let scale = null;\n if (arg['font-scale']) {\n scale = context.parse(arg['font-scale'], 1, NumberType);\n if (!scale)\n return null;\n }\n let font = null;\n if (arg['text-font']) {\n font = context.parse(arg['text-font'], 1, array$1(StringType));\n if (!font)\n return null;\n }\n let textColor = null;\n if (arg['text-color']) {\n textColor = context.parse(arg['text-color'], 1, ColorType);\n if (!textColor)\n return null;\n }\n const lastExpression = sections[sections.length - 1];\n lastExpression.scale = scale;\n lastExpression.font = font;\n lastExpression.textColor = textColor;\n }\n else {\n const content = context.parse(args[i], 1, ValueType);\n if (!content)\n return null;\n const kind = content.type.kind;\n if (kind !== 'string' && kind !== 'value' && kind !== 'null' && kind !== 'resolvedImage')\n return context.error('Formatted text type must be \\'string\\', \\'value\\', \\'image\\' or \\'null\\'.');\n nextTokenMayBeObject = true;\n sections.push({ content, scale: null, font: null, textColor: null });\n }\n }\n return new FormatExpression(sections);\n }\n evaluate(ctx) {\n const evaluateSection = section => {\n const evaluatedContent = section.content.evaluate(ctx);\n if (typeOf(evaluatedContent) === ResolvedImageType) {\n return new FormattedSection('', evaluatedContent, null, null, null);\n }\n return new FormattedSection(toString(evaluatedContent), null, section.scale ? section.scale.evaluate(ctx) : null, section.font ? section.font.evaluate(ctx).join(',') : null, section.textColor ? section.textColor.evaluate(ctx) : null);\n };\n return new Formatted(this.sections.map(evaluateSection));\n }\n eachChild(fn) {\n for (const section of this.sections) {\n fn(section.content);\n if (section.scale) {\n fn(section.scale);\n }\n if (section.font) {\n fn(section.font);\n }\n if (section.textColor) {\n fn(section.textColor);\n }\n }\n }\n outputDefined() {\n // Technically the combinatoric set of all children\n // Usually, this.text will be undefined anyway\n return false;\n }\n}\n\nclass ImageExpression {\n constructor(input) {\n this.type = ResolvedImageType;\n this.input = input;\n }\n static parse(args, context) {\n if (args.length !== 2) {\n return context.error('Expected two arguments.');\n }\n const name = context.parse(args[1], 1, StringType);\n if (!name)\n return context.error('No image name provided.');\n return new ImageExpression(name);\n }\n evaluate(ctx) {\n const evaluatedImageName = this.input.evaluate(ctx);\n const value = ResolvedImage.fromString(evaluatedImageName);\n if (value && ctx.availableImages)\n value.available = ctx.availableImages.indexOf(evaluatedImageName) > -1;\n return value;\n }\n eachChild(fn) {\n fn(this.input);\n }\n outputDefined() {\n // The output of image is determined by the list of available images in the evaluation context\n return false;\n }\n}\n\nclass Length {\n constructor(input) {\n this.type = NumberType;\n this.input = input;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error(`Expected 1 argument, but found ${args.length - 1} instead.`);\n const input = context.parse(args[1], 1);\n if (!input)\n return null;\n if (input.type.kind !== 'array' && input.type.kind !== 'string' && input.type.kind !== 'value')\n return context.error(`Expected argument of type string or array, but found ${toString$1(input.type)} instead.`);\n return new Length(input);\n }\n evaluate(ctx) {\n const input = this.input.evaluate(ctx);\n if (typeof input === 'string') {\n return input.length;\n }\n else if (Array.isArray(input)) {\n return input.length;\n }\n else {\n throw new RuntimeError(`Expected value to be of type string or array, but found ${toString$1(typeOf(input))} instead.`);\n }\n }\n eachChild(fn) {\n fn(this.input);\n }\n outputDefined() {\n return false;\n }\n}\n\nconst expressions = {\n // special forms\n '==': Equals,\n '!=': NotEquals,\n '>': GreaterThan,\n '<': LessThan,\n '>=': GreaterThanOrEqual,\n '<=': LessThanOrEqual,\n 'array': Assertion,\n 'at': At,\n 'boolean': Assertion,\n 'case': Case,\n 'coalesce': Coalesce,\n 'collator': CollatorExpression,\n 'format': FormatExpression,\n 'image': ImageExpression,\n 'in': In,\n 'index-of': IndexOf,\n 'interpolate': Interpolate,\n 'interpolate-hcl': Interpolate,\n 'interpolate-lab': Interpolate,\n 'length': Length,\n 'let': Let,\n 'literal': Literal,\n 'match': Match,\n 'number': Assertion,\n 'number-format': NumberFormat,\n 'object': Assertion,\n 'slice': Slice,\n 'step': Step,\n 'string': Assertion,\n 'to-boolean': Coercion,\n 'to-color': Coercion,\n 'to-number': Coercion,\n 'to-string': Coercion,\n 'var': Var,\n 'within': Within\n};\nfunction rgba(ctx, [r, g, b, a]) {\n r = r.evaluate(ctx);\n g = g.evaluate(ctx);\n b = b.evaluate(ctx);\n const alpha = a ? a.evaluate(ctx) : 1;\n const error = validateRGBA(r, g, b, alpha);\n if (error)\n throw new RuntimeError(error);\n return new Color(r / 255, g / 255, b / 255, alpha, false);\n}\nfunction has(key, obj) {\n return key in obj;\n}\nfunction get(key, obj) {\n const v = obj[key];\n return typeof v === 'undefined' ? null : v;\n}\nfunction binarySearch(v, a, i, j) {\n while (i <= j) {\n const m = (i + j) >> 1;\n if (a[m] === v)\n return true;\n if (a[m] > v)\n j = m - 1;\n else\n i = m + 1;\n }\n return false;\n}\nfunction varargs(type) {\n return { type };\n}\nCompoundExpression.register(expressions, {\n 'error': [\n ErrorType,\n [StringType],\n (ctx, [v]) => { throw new RuntimeError(v.evaluate(ctx)); }\n ],\n 'typeof': [\n StringType,\n [ValueType],\n (ctx, [v]) => toString$1(typeOf(v.evaluate(ctx)))\n ],\n 'to-rgba': [\n array$1(NumberType, 4),\n [ColorType],\n (ctx, [v]) => {\n const [r, g, b, a] = v.evaluate(ctx).rgb;\n return [r * 255, g * 255, b * 255, a];\n },\n ],\n 'rgb': [\n ColorType,\n [NumberType, NumberType, NumberType],\n rgba\n ],\n 'rgba': [\n ColorType,\n [NumberType, NumberType, NumberType, NumberType],\n rgba\n ],\n 'has': {\n type: BooleanType,\n overloads: [\n [\n [StringType],\n (ctx, [key]) => has(key.evaluate(ctx), ctx.properties())\n ], [\n [StringType, ObjectType],\n (ctx, [key, obj]) => has(key.evaluate(ctx), obj.evaluate(ctx))\n ]\n ]\n },\n 'get': {\n type: ValueType,\n overloads: [\n [\n [StringType],\n (ctx, [key]) => get(key.evaluate(ctx), ctx.properties())\n ], [\n [StringType, ObjectType],\n (ctx, [key, obj]) => get(key.evaluate(ctx), obj.evaluate(ctx))\n ]\n ]\n },\n 'feature-state': [\n ValueType,\n [StringType],\n (ctx, [key]) => get(key.evaluate(ctx), ctx.featureState || {})\n ],\n 'properties': [\n ObjectType,\n [],\n (ctx) => ctx.properties()\n ],\n 'geometry-type': [\n StringType,\n [],\n (ctx) => ctx.geometryType()\n ],\n 'id': [\n ValueType,\n [],\n (ctx) => ctx.id()\n ],\n 'zoom': [\n NumberType,\n [],\n (ctx) => ctx.globals.zoom\n ],\n 'heatmap-density': [\n NumberType,\n [],\n (ctx) => ctx.globals.heatmapDensity || 0\n ],\n 'line-progress': [\n NumberType,\n [],\n (ctx) => ctx.globals.lineProgress || 0\n ],\n 'accumulated': [\n ValueType,\n [],\n (ctx) => ctx.globals.accumulated === undefined ? null : ctx.globals.accumulated\n ],\n '+': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => {\n let result = 0;\n for (const arg of args) {\n result += arg.evaluate(ctx);\n }\n return result;\n }\n ],\n '*': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => {\n let result = 1;\n for (const arg of args) {\n result *= arg.evaluate(ctx);\n }\n return result;\n }\n ],\n '-': {\n type: NumberType,\n overloads: [\n [\n [NumberType, NumberType],\n (ctx, [a, b]) => a.evaluate(ctx) - b.evaluate(ctx)\n ], [\n [NumberType],\n (ctx, [a]) => -a.evaluate(ctx)\n ]\n ]\n },\n '/': [\n NumberType,\n [NumberType, NumberType],\n (ctx, [a, b]) => a.evaluate(ctx) / b.evaluate(ctx)\n ],\n '%': [\n NumberType,\n [NumberType, NumberType],\n (ctx, [a, b]) => a.evaluate(ctx) % b.evaluate(ctx)\n ],\n 'ln2': [\n NumberType,\n [],\n () => Math.LN2\n ],\n 'pi': [\n NumberType,\n [],\n () => Math.PI\n ],\n 'e': [\n NumberType,\n [],\n () => Math.E\n ],\n '^': [\n NumberType,\n [NumberType, NumberType],\n (ctx, [b, e]) => Math.pow(b.evaluate(ctx), e.evaluate(ctx))\n ],\n 'sqrt': [\n NumberType,\n [NumberType],\n (ctx, [x]) => Math.sqrt(x.evaluate(ctx))\n ],\n 'log10': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.log(n.evaluate(ctx)) / Math.LN10\n ],\n 'ln': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.log(n.evaluate(ctx))\n ],\n 'log2': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.log(n.evaluate(ctx)) / Math.LN2\n ],\n 'sin': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.sin(n.evaluate(ctx))\n ],\n 'cos': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.cos(n.evaluate(ctx))\n ],\n 'tan': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.tan(n.evaluate(ctx))\n ],\n 'asin': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.asin(n.evaluate(ctx))\n ],\n 'acos': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.acos(n.evaluate(ctx))\n ],\n 'atan': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.atan(n.evaluate(ctx))\n ],\n 'min': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => Math.min(...args.map(arg => arg.evaluate(ctx)))\n ],\n 'max': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => Math.max(...args.map(arg => arg.evaluate(ctx)))\n ],\n 'abs': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.abs(n.evaluate(ctx))\n ],\n 'round': [\n NumberType,\n [NumberType],\n (ctx, [n]) => {\n const v = n.evaluate(ctx);\n // Javascript's Math.round() rounds towards +Infinity for halfway\n // values, even when they're negative. It's more common to round\n // away from 0 (e.g., this is what python and C++ do)\n return v < 0 ? -Math.round(-v) : Math.round(v);\n }\n ],\n 'floor': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.floor(n.evaluate(ctx))\n ],\n 'ceil': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.ceil(n.evaluate(ctx))\n ],\n 'filter-==': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => ctx.properties()[k.value] === v.value\n ],\n 'filter-id-==': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => ctx.id() === v.value\n ],\n 'filter-type-==': [\n BooleanType,\n [StringType],\n (ctx, [v]) => ctx.geometryType() === v.value\n ],\n 'filter-<': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a < b;\n }\n ],\n 'filter-id-<': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a < b;\n }\n ],\n 'filter->': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a > b;\n }\n ],\n 'filter-id->': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a > b;\n }\n ],\n 'filter-<=': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a <= b;\n }\n ],\n 'filter-id-<=': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a <= b;\n }\n ],\n 'filter->=': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a >= b;\n }\n ],\n 'filter-id->=': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a >= b;\n }\n ],\n 'filter-has': [\n BooleanType,\n [ValueType],\n (ctx, [k]) => k.value in ctx.properties()\n ],\n 'filter-has-id': [\n BooleanType,\n [],\n (ctx) => (ctx.id() !== null && ctx.id() !== undefined)\n ],\n 'filter-type-in': [\n BooleanType,\n [array$1(StringType)],\n (ctx, [v]) => v.value.indexOf(ctx.geometryType()) >= 0\n ],\n 'filter-id-in': [\n BooleanType,\n [array$1(ValueType)],\n (ctx, [v]) => v.value.indexOf(ctx.id()) >= 0\n ],\n 'filter-in-small': [\n BooleanType,\n [StringType, array$1(ValueType)],\n // assumes v is an array literal\n (ctx, [k, v]) => v.value.indexOf(ctx.properties()[k.value]) >= 0\n ],\n 'filter-in-large': [\n BooleanType,\n [StringType, array$1(ValueType)],\n // assumes v is a array literal with values sorted in ascending order and of a single type\n (ctx, [k, v]) => binarySearch(ctx.properties()[k.value], v.value, 0, v.value.length - 1)\n ],\n 'all': {\n type: BooleanType,\n overloads: [\n [\n [BooleanType, BooleanType],\n (ctx, [a, b]) => a.evaluate(ctx) && b.evaluate(ctx)\n ],\n [\n varargs(BooleanType),\n (ctx, args) => {\n for (const arg of args) {\n if (!arg.evaluate(ctx))\n return false;\n }\n return true;\n }\n ]\n ]\n },\n 'any': {\n type: BooleanType,\n overloads: [\n [\n [BooleanType, BooleanType],\n (ctx, [a, b]) => a.evaluate(ctx) || b.evaluate(ctx)\n ],\n [\n varargs(BooleanType),\n (ctx, args) => {\n for (const arg of args) {\n if (arg.evaluate(ctx))\n return true;\n }\n return false;\n }\n ]\n ]\n },\n '!': [\n BooleanType,\n [BooleanType],\n (ctx, [b]) => !b.evaluate(ctx)\n ],\n 'is-supported-script': [\n BooleanType,\n [StringType],\n // At parse time this will always return true, so we need to exclude this expression with isGlobalPropertyConstant\n (ctx, [s]) => {\n const isSupportedScript = ctx.globals && ctx.globals.isSupportedScript;\n if (isSupportedScript) {\n return isSupportedScript(s.evaluate(ctx));\n }\n return true;\n }\n ],\n 'upcase': [\n StringType,\n [StringType],\n (ctx, [s]) => s.evaluate(ctx).toUpperCase()\n ],\n 'downcase': [\n StringType,\n [StringType],\n (ctx, [s]) => s.evaluate(ctx).toLowerCase()\n ],\n 'concat': [\n StringType,\n varargs(ValueType),\n (ctx, args) => args.map(arg => toString(arg.evaluate(ctx))).join('')\n ],\n 'resolved-locale': [\n StringType,\n [CollatorType],\n (ctx, [collator]) => collator.evaluate(ctx).resolvedLocale()\n ]\n});\n\nfunction success(value) {\n return { result: 'success', value };\n}\nfunction error(value) {\n return { result: 'error', value };\n}\n\nfunction supportsPropertyExpression(spec) {\n return spec['property-type'] === 'data-driven' || spec['property-type'] === 'cross-faded-data-driven';\n}\nfunction supportsZoomExpression(spec) {\n return !!spec.expression && spec.expression.parameters.indexOf('zoom') > -1;\n}\nfunction supportsInterpolation(spec) {\n return !!spec.expression && spec.expression.interpolated;\n}\n\nfunction getType(val) {\n if (val instanceof Number) {\n return 'number';\n }\n else if (val instanceof String) {\n return 'string';\n }\n else if (val instanceof Boolean) {\n return 'boolean';\n }\n else if (Array.isArray(val)) {\n return 'array';\n }\n else if (val === null) {\n return 'null';\n }\n else {\n return typeof val;\n }\n}\n\nfunction isFunction(value) {\n return typeof value === 'object' && value !== null && !Array.isArray(value);\n}\nfunction identityFunction(x) {\n return x;\n}\nfunction createFunction(parameters, propertySpec) {\n const isColor = propertySpec.type === 'color';\n const zoomAndFeatureDependent = parameters.stops && typeof parameters.stops[0][0] === 'object';\n const featureDependent = zoomAndFeatureDependent || parameters.property !== undefined;\n const zoomDependent = zoomAndFeatureDependent || !featureDependent;\n const type = parameters.type || (supportsInterpolation(propertySpec) ? 'exponential' : 'interval');\n if (isColor || propertySpec.type === 'padding') {\n const parseFn = isColor ? Color.parse : Padding.parse;\n parameters = extendBy({}, parameters);\n if (parameters.stops) {\n parameters.stops = parameters.stops.map((stop) => {\n return [stop[0], parseFn(stop[1])];\n });\n }\n if (parameters.default) {\n parameters.default = parseFn(parameters.default);\n }\n else {\n parameters.default = parseFn(propertySpec.default);\n }\n }\n if (parameters.colorSpace && !isSupportedInterpolationColorSpace(parameters.colorSpace)) {\n throw new Error(`Unknown color space: \"${parameters.colorSpace}\"`);\n }\n let innerFun;\n let hashedStops;\n let categoricalKeyType;\n if (type === 'exponential') {\n innerFun = evaluateExponentialFunction;\n }\n else if (type === 'interval') {\n innerFun = evaluateIntervalFunction;\n }\n else if (type === 'categorical') {\n innerFun = evaluateCategoricalFunction;\n // For categorical functions, generate an Object as a hashmap of the stops for fast searching\n hashedStops = Object.create(null);\n for (const stop of parameters.stops) {\n hashedStops[stop[0]] = stop[1];\n }\n // Infer key type based on first stop key-- used to encforce strict type checking later\n categoricalKeyType = typeof parameters.stops[0][0];\n }\n else if (type === 'identity') {\n innerFun = evaluateIdentityFunction;\n }\n else {\n throw new Error(`Unknown function type \"${type}\"`);\n }\n if (zoomAndFeatureDependent) {\n const featureFunctions = {};\n const zoomStops = [];\n for (let s = 0; s < parameters.stops.length; s++) {\n const stop = parameters.stops[s];\n const zoom = stop[0].zoom;\n if (featureFunctions[zoom] === undefined) {\n featureFunctions[zoom] = {\n zoom,\n type: parameters.type,\n property: parameters.property,\n default: parameters.default,\n stops: []\n };\n zoomStops.push(zoom);\n }\n featureFunctions[zoom].stops.push([stop[0].value, stop[1]]);\n }\n const featureFunctionStops = [];\n for (const z of zoomStops) {\n featureFunctionStops.push([featureFunctions[z].zoom, createFunction(featureFunctions[z], propertySpec)]);\n }\n const interpolationType = { name: 'linear' };\n return {\n kind: 'composite',\n interpolationType,\n interpolationFactor: Interpolate.interpolationFactor.bind(undefined, interpolationType),\n zoomStops: featureFunctionStops.map(s => s[0]),\n evaluate({ zoom }, properties) {\n return evaluateExponentialFunction({\n stops: featureFunctionStops,\n base: parameters.base\n }, propertySpec, zoom).evaluate(zoom, properties);\n }\n };\n }\n else if (zoomDependent) {\n const interpolationType = type === 'exponential' ?\n { name: 'exponential', base: parameters.base !== undefined ? parameters.base : 1 } : null;\n return {\n kind: 'camera',\n interpolationType,\n interpolationFactor: Interpolate.interpolationFactor.bind(undefined, interpolationType),\n zoomStops: parameters.stops.map(s => s[0]),\n evaluate: ({ zoom }) => innerFun(parameters, propertySpec, zoom, hashedStops, categoricalKeyType)\n };\n }\n else {\n return {\n kind: 'source',\n evaluate(_, feature) {\n const value = feature && feature.properties ? feature.properties[parameters.property] : undefined;\n if (value === undefined) {\n return coalesce$1(parameters.default, propertySpec.default);\n }\n return innerFun(parameters, propertySpec, value, hashedStops, categoricalKeyType);\n }\n };\n }\n}\nfunction coalesce$1(a, b, c) {\n if (a !== undefined)\n return a;\n if (b !== undefined)\n return b;\n if (c !== undefined)\n return c;\n}\nfunction evaluateCategoricalFunction(parameters, propertySpec, input, hashedStops, keyType) {\n const evaluated = typeof input === keyType ? hashedStops[input] : undefined; // Enforce strict typing on input\n return coalesce$1(evaluated, parameters.default, propertySpec.default);\n}\nfunction evaluateIntervalFunction(parameters, propertySpec, input) {\n // Edge cases\n if (getType(input) !== 'number')\n return coalesce$1(parameters.default, propertySpec.default);\n const n = parameters.stops.length;\n if (n === 1)\n return parameters.stops[0][1];\n if (input <= parameters.stops[0][0])\n return parameters.stops[0][1];\n if (input >= parameters.stops[n - 1][0])\n return parameters.stops[n - 1][1];\n const index = findStopLessThanOrEqualTo(parameters.stops.map((stop) => stop[0]), input);\n return parameters.stops[index][1];\n}\nfunction evaluateExponentialFunction(parameters, propertySpec, input) {\n const base = parameters.base !== undefined ? parameters.base : 1;\n // Edge cases\n if (getType(input) !== 'number')\n return coalesce$1(parameters.default, propertySpec.default);\n const n = parameters.stops.length;\n if (n === 1)\n return parameters.stops[0][1];\n if (input <= parameters.stops[0][0])\n return parameters.stops[0][1];\n if (input >= parameters.stops[n - 1][0])\n return parameters.stops[n - 1][1];\n const index = findStopLessThanOrEqualTo(parameters.stops.map((stop) => stop[0]), input);\n const t = interpolationFactor(input, base, parameters.stops[index][0], parameters.stops[index + 1][0]);\n const outputLower = parameters.stops[index][1];\n const outputUpper = parameters.stops[index + 1][1];\n const interp = interpolate[propertySpec.type] || identityFunction;\n if (typeof outputLower.evaluate === 'function') {\n return {\n evaluate(...args) {\n const evaluatedLower = outputLower.evaluate.apply(undefined, args);\n const evaluatedUpper = outputUpper.evaluate.apply(undefined, args);\n // Special case for fill-outline-color, which has no spec default.\n if (evaluatedLower === undefined || evaluatedUpper === undefined) {\n return undefined;\n }\n return interp(evaluatedLower, evaluatedUpper, t, parameters.colorSpace);\n }\n };\n }\n return interp(outputLower, outputUpper, t, parameters.colorSpace);\n}\nfunction evaluateIdentityFunction(parameters, propertySpec, input) {\n switch (propertySpec.type) {\n case 'color':\n input = Color.parse(input);\n break;\n case 'formatted':\n input = Formatted.fromString(input.toString());\n break;\n case 'resolvedImage':\n input = ResolvedImage.fromString(input.toString());\n break;\n case 'padding':\n input = Padding.parse(input);\n break;\n default:\n if (getType(input) !== propertySpec.type && (propertySpec.type !== 'enum' || !propertySpec.values[input])) {\n input = undefined;\n }\n }\n return coalesce$1(input, parameters.default, propertySpec.default);\n}\n/**\n * Returns a ratio that can be used to interpolate between exponential function\n * stops.\n *\n * How it works:\n * Two consecutive stop values define a (scaled and shifted) exponential\n * function `f(x) = a * base^x + b`, where `base` is the user-specified base,\n * and `a` and `b` are constants affording sufficient degrees of freedom to fit\n * the function to the given stops.\n *\n * Here's a bit of algebra that lets us compute `f(x)` directly from the stop\n * values without explicitly solving for `a` and `b`:\n *\n * First stop value: `f(x0) = y0 = a * base^x0 + b`\n * Second stop value: `f(x1) = y1 = a * base^x1 + b`\n * => `y1 - y0 = a(base^x1 - base^x0)`\n * => `a = (y1 - y0)/(base^x1 - base^x0)`\n *\n * Desired value: `f(x) = y = a * base^x + b`\n * => `f(x) = y0 + a * (base^x - base^x0)`\n *\n * From the above, we can replace the `a` in `a * (base^x - base^x0)` and do a\n * little algebra:\n * ```\n * a * (base^x - base^x0) = (y1 - y0)/(base^x1 - base^x0) * (base^x - base^x0)\n * = (y1 - y0) * (base^x - base^x0) / (base^x1 - base^x0)\n * ```\n *\n * If we let `(base^x - base^x0) / (base^x1 base^x0)`, then we have\n * `f(x) = y0 + (y1 - y0) * ratio`. In other words, `ratio` may be treated as\n * an interpolation factor between the two stops' output values.\n *\n * (Note: a slightly different form for `ratio`,\n * `(base^(x-x0) - 1) / (base^(x1-x0) - 1) `, is equivalent, but requires fewer\n * expensive `Math.pow()` operations.)\n *\n * @private\n */\nfunction interpolationFactor(input, base, lowerValue, upperValue) {\n const difference = upperValue - lowerValue;\n const progress = input - lowerValue;\n if (difference === 0) {\n return 0;\n }\n else if (base === 1) {\n return progress / difference;\n }\n else {\n return (Math.pow(base, progress) - 1) / (Math.pow(base, difference) - 1);\n }\n}\n\nclass StyleExpression {\n constructor(expression, propertySpec) {\n this.expression = expression;\n this._warningHistory = {};\n this._evaluator = new EvaluationContext();\n this._defaultValue = propertySpec ? getDefaultValue(propertySpec) : null;\n this._enumValues = propertySpec && propertySpec.type === 'enum' ? propertySpec.values : null;\n }\n evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) {\n this._evaluator.globals = globals;\n this._evaluator.feature = feature;\n this._evaluator.featureState = featureState;\n this._evaluator.canonical = canonical;\n this._evaluator.availableImages = availableImages || null;\n this._evaluator.formattedSection = formattedSection;\n return this.expression.evaluate(this._evaluator);\n }\n evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) {\n this._evaluator.globals = globals;\n this._evaluator.feature = feature || null;\n this._evaluator.featureState = featureState || null;\n this._evaluator.canonical = canonical;\n this._evaluator.availableImages = availableImages || null;\n this._evaluator.formattedSection = formattedSection || null;\n try {\n const val = this.expression.evaluate(this._evaluator);\n // eslint-disable-next-line no-self-compare\n if (val === null || val === undefined || (typeof val === 'number' && val !== val)) {\n return this._defaultValue;\n }\n if (this._enumValues && !(val in this._enumValues)) {\n throw new RuntimeError(`Expected value to be one of ${Object.keys(this._enumValues).map(v => JSON.stringify(v)).join(', ')}, but found ${JSON.stringify(val)} instead.`);\n }\n return val;\n }\n catch (e) {\n if (!this._warningHistory[e.message]) {\n this._warningHistory[e.message] = true;\n if (typeof console !== 'undefined') {\n console.warn(e.message);\n }\n }\n return this._defaultValue;\n }\n }\n}\nfunction isExpression(expression) {\n return Array.isArray(expression) && expression.length > 0 &&\n typeof expression[0] === 'string' && expression[0] in expressions;\n}\n/**\n * Parse and typecheck the given style spec JSON expression. If\n * options.defaultValue is provided, then the resulting StyleExpression's\n * `evaluate()` method will handle errors by logging a warning (once per\n * message) and returning the default value. Otherwise, it will throw\n * evaluation errors.\n *\n * @private\n */\nfunction createExpression(expression, propertySpec) {\n const parser = new ParsingContext(expressions, isExpressionConstant, [], propertySpec ? getExpectedType(propertySpec) : undefined);\n // For string-valued properties, coerce to string at the top level rather than asserting.\n const parsed = parser.parse(expression, undefined, undefined, undefined, propertySpec && propertySpec.type === 'string' ? { typeAnnotation: 'coerce' } : undefined);\n if (!parsed) {\n return error(parser.errors);\n }\n return success(new StyleExpression(parsed, propertySpec));\n}\nclass ZoomConstantExpression {\n constructor(kind, expression) {\n this.kind = kind;\n this._styleExpression = expression;\n this.isStateDependent = kind !== 'constant' && !isStateConstant(expression.expression);\n }\n evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluate(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n}\nclass ZoomDependentExpression {\n constructor(kind, expression, zoomStops, interpolationType) {\n this.kind = kind;\n this.zoomStops = zoomStops;\n this._styleExpression = expression;\n this.isStateDependent = kind !== 'camera' && !isStateConstant(expression.expression);\n this.interpolationType = interpolationType;\n }\n evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluate(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n interpolationFactor(input, lower, upper) {\n if (this.interpolationType) {\n return Interpolate.interpolationFactor(this.interpolationType, input, lower, upper);\n }\n else {\n return 0;\n }\n }\n}\nfunction isZoomExpression(expression) {\n return expression._styleExpression !== undefined;\n}\nfunction createPropertyExpression(expressionInput, propertySpec) {\n const expression = createExpression(expressionInput, propertySpec);\n if (expression.result === 'error') {\n return expression;\n }\n const parsed = expression.value.expression;\n const isFeatureConstantResult = isFeatureConstant(parsed);\n if (!isFeatureConstantResult && !supportsPropertyExpression(propertySpec)) {\n return error([new ExpressionParsingError('', 'data expressions not supported')]);\n }\n const isZoomConstant = isGlobalPropertyConstant(parsed, ['zoom']);\n if (!isZoomConstant && !supportsZoomExpression(propertySpec)) {\n return error([new ExpressionParsingError('', 'zoom expressions not supported')]);\n }\n const zoomCurve = findZoomCurve(parsed);\n if (!zoomCurve && !isZoomConstant) {\n return error([new ExpressionParsingError('', '\"zoom\" expression may only be used as input to a top-level \"step\" or \"interpolate\" expression.')]);\n }\n else if (zoomCurve instanceof ExpressionParsingError) {\n return error([zoomCurve]);\n }\n else if (zoomCurve instanceof Interpolate && !supportsInterpolation(propertySpec)) {\n return error([new ExpressionParsingError('', '\"interpolate\" expressions cannot be used with this property')]);\n }\n if (!zoomCurve) {\n return success(isFeatureConstantResult ?\n new ZoomConstantExpression('constant', expression.value) :\n new ZoomConstantExpression('source', expression.value));\n }\n const interpolationType = zoomCurve instanceof Interpolate ? zoomCurve.interpolation : undefined;\n return success(isFeatureConstantResult ?\n new ZoomDependentExpression('camera', expression.value, zoomCurve.labels, interpolationType) :\n new ZoomDependentExpression('composite', expression.value, zoomCurve.labels, interpolationType));\n}\n// serialization wrapper for old-style stop functions normalized to the\n// expression interface\nclass StylePropertyFunction {\n constructor(parameters, specification) {\n this._parameters = parameters;\n this._specification = specification;\n extendBy(this, createFunction(this._parameters, this._specification));\n }\n static deserialize(serialized) {\n return new StylePropertyFunction(serialized._parameters, serialized._specification);\n }\n static serialize(input) {\n return {\n _parameters: input._parameters,\n _specification: input._specification\n };\n }\n}\nfunction normalizePropertyExpression(value, specification) {\n if (isFunction(value)) {\n return new StylePropertyFunction(value, specification);\n }\n else if (isExpression(value)) {\n const expression = createPropertyExpression(value, specification);\n if (expression.result === 'error') {\n // this should have been caught in validation\n throw new Error(expression.value.map(err => `${err.key}: ${err.message}`).join(', '));\n }\n return expression.value;\n }\n else {\n let constant = value;\n if (specification.type === 'color' && typeof value === 'string') {\n constant = Color.parse(value);\n }\n else if (specification.type === 'padding' && (typeof value === 'number' || Array.isArray(value))) {\n constant = Padding.parse(value);\n }\n else if (specification.type === 'variableAnchorOffsetCollection' && Array.isArray(value)) {\n constant = VariableAnchorOffsetCollection.parse(value);\n }\n return {\n kind: 'constant',\n evaluate: () => constant\n };\n }\n}\n// Zoom-dependent expressions may only use [\"zoom\"] as the input to a top-level \"step\" or \"interpolate\"\n// expression (collectively referred to as a \"curve\"). The curve may be wrapped in one or more \"let\" or\n// \"coalesce\" expressions.\nfunction findZoomCurve(expression) {\n let result = null;\n if (expression instanceof Let) {\n result = findZoomCurve(expression.result);\n }\n else if (expression instanceof Coalesce) {\n for (const arg of expression.args) {\n result = findZoomCurve(arg);\n if (result) {\n break;\n }\n }\n }\n else if ((expression instanceof Step || expression instanceof Interpolate) &&\n expression.input instanceof CompoundExpression &&\n expression.input.name === 'zoom') {\n result = expression;\n }\n if (result instanceof ExpressionParsingError) {\n return result;\n }\n expression.eachChild((child) => {\n const childResult = findZoomCurve(child);\n if (childResult instanceof ExpressionParsingError) {\n result = childResult;\n }\n else if (!result && childResult) {\n result = new ExpressionParsingError('', '\"zoom\" expression may only be used as input to a top-level \"step\" or \"interpolate\" expression.');\n }\n else if (result && childResult && result !== childResult) {\n result = new ExpressionParsingError('', 'Only one zoom-based \"step\" or \"interpolate\" subexpression may be used in an expression.');\n }\n });\n return result;\n}\nfunction getExpectedType(spec) {\n const types = {\n color: ColorType,\n string: StringType,\n number: NumberType,\n enum: StringType,\n boolean: BooleanType,\n formatted: FormattedType,\n padding: PaddingType,\n resolvedImage: ResolvedImageType,\n variableAnchorOffsetCollection: VariableAnchorOffsetCollectionType\n };\n if (spec.type === 'array') {\n return array$1(types[spec.value] || ValueType, spec.length);\n }\n return types[spec.type];\n}\nfunction getDefaultValue(spec) {\n if (spec.type === 'color' && isFunction(spec.default)) {\n // Special case for heatmap-color: it uses the 'default:' to define a\n // default color ramp, but createExpression expects a simple value to fall\n // back to in case of runtime errors\n return new Color(0, 0, 0, 0);\n }\n else if (spec.type === 'color') {\n return Color.parse(spec.default) || null;\n }\n else if (spec.type === 'padding') {\n return Padding.parse(spec.default) || null;\n }\n else if (spec.type === 'variableAnchorOffsetCollection') {\n return VariableAnchorOffsetCollection.parse(spec.default) || null;\n }\n else if (spec.default === undefined) {\n return null;\n }\n else {\n return spec.default;\n }\n}\n\nfunction isExpressionFilter(filter) {\n if (filter === true || filter === false) {\n return true;\n }\n if (!Array.isArray(filter) || filter.length === 0) {\n return false;\n }\n switch (filter[0]) {\n case 'has':\n return filter.length >= 2 && filter[1] !== '$id' && filter[1] !== '$type';\n case 'in':\n return filter.length >= 3 && (typeof filter[1] !== 'string' || Array.isArray(filter[2]));\n case '!in':\n case '!has':\n case 'none':\n return false;\n case '==':\n case '!=':\n case '>':\n case '>=':\n case '<':\n case '<=':\n return filter.length !== 3 || (Array.isArray(filter[1]) || Array.isArray(filter[2]));\n case 'any':\n case 'all':\n for (const f of filter.slice(1)) {\n if (!isExpressionFilter(f) && typeof f !== 'boolean') {\n return false;\n }\n }\n return true;\n default:\n return true;\n }\n}\nconst filterSpec = {\n 'type': 'boolean',\n 'default': false,\n 'transition': false,\n 'property-type': 'data-driven',\n 'expression': {\n 'interpolated': false,\n 'parameters': ['zoom', 'feature']\n }\n};\n/**\n * Given a filter expressed as nested arrays, return a new function\n * that evaluates whether a given feature (with a .properties or .tags property)\n * passes its test.\n *\n * @private\n * @param {Array} filter MapLibre filter\n * @returns {Function} filter-evaluating function\n */\nfunction createFilter(filter) {\n if (filter === null || filter === undefined) {\n return { filter: () => true, needGeometry: false };\n }\n if (!isExpressionFilter(filter)) {\n filter = convertFilter$1(filter);\n }\n const compiled = createExpression(filter, filterSpec);\n if (compiled.result === 'error') {\n throw new Error(compiled.value.map(err => `${err.key}: ${err.message}`).join(', '));\n }\n else {\n const needGeometry = geometryNeeded(filter);\n return { filter: (globalProperties, feature, canonical) => compiled.value.evaluate(globalProperties, feature, {}, canonical),\n needGeometry };\n }\n}\n// Comparison function to sort numbers and strings\nfunction compare(a, b) {\n return a < b ? -1 : a > b ? 1 : 0;\n}\nfunction geometryNeeded(filter) {\n if (!Array.isArray(filter))\n return false;\n if (filter[0] === 'within')\n return true;\n for (let index = 1; index < filter.length; index++) {\n if (geometryNeeded(filter[index]))\n return true;\n }\n return false;\n}\nfunction convertFilter$1(filter) {\n if (!filter)\n return true;\n const op = filter[0];\n if (filter.length <= 1)\n return (op !== 'any');\n const converted = op === '==' ? convertComparisonOp$1(filter[1], filter[2], '==') :\n op === '!=' ? convertNegation(convertComparisonOp$1(filter[1], filter[2], '==')) :\n op === '<' ||\n op === '>' ||\n op === '<=' ||\n op === '>=' ? convertComparisonOp$1(filter[1], filter[2], op) :\n op === 'any' ? convertDisjunctionOp(filter.slice(1)) :\n op === 'all' ? ['all'].concat(filter.slice(1).map(convertFilter$1)) :\n op === 'none' ? ['all'].concat(filter.slice(1).map(convertFilter$1).map(convertNegation)) :\n op === 'in' ? convertInOp$1(filter[1], filter.slice(2)) :\n op === '!in' ? convertNegation(convertInOp$1(filter[1], filter.slice(2))) :\n op === 'has' ? convertHasOp$1(filter[1]) :\n op === '!has' ? convertNegation(convertHasOp$1(filter[1])) :\n op === 'within' ? filter :\n true;\n return converted;\n}\nfunction convertComparisonOp$1(property, value, op) {\n switch (property) {\n case '$type':\n return [`filter-type-${op}`, value];\n case '$id':\n return [`filter-id-${op}`, value];\n default:\n return [`filter-${op}`, property, value];\n }\n}\nfunction convertDisjunctionOp(filters) {\n return ['any'].concat(filters.map(convertFilter$1));\n}\nfunction convertInOp$1(property, values) {\n if (values.length === 0) {\n return false;\n }\n switch (property) {\n case '$type':\n return ['filter-type-in', ['literal', values]];\n case '$id':\n return ['filter-id-in', ['literal', values]];\n default:\n if (values.length > 200 && !values.some(v => typeof v !== typeof values[0])) {\n return ['filter-in-large', property, ['literal', values.sort(compare)]];\n }\n else {\n return ['filter-in-small', property, ['literal', values]];\n }\n }\n}\nfunction convertHasOp$1(property) {\n switch (property) {\n case '$type':\n return true;\n case '$id':\n return ['filter-has-id'];\n default:\n return ['filter-has', property];\n }\n}\nfunction convertNegation(filter) {\n return ['!', filter];\n}\n\n/*\n * Convert the given filter to an expression, storing the expected types for\n * any feature properties referenced in expectedTypes.\n *\n * These expected types are needed in order to construct preflight type checks\n * needed for handling 'any' filters. A preflight type check is necessary in\n * order to mimic legacy filters' semantics around expected type mismatches.\n * For example, consider the legacy filter:\n *\n * [\"any\", [\"all\", [\">\", \"y\", 0], [\">\", \"y\", 0]], [\">\", \"x\", 0]]\n *\n * Naively, we might convert this to the expression:\n *\n * [\"any\", [\"all\", [\">\", [\"get\", \"y\"], 0], [\">\", [\"get\", \"z\"], 0]], [\">\", [\"get\", \"x\"], 0]]\n *\n * But if we tried to evaluate this against, say `{x: 1, y: null, z: 0}`, the\n * [\">\", [\"get\", \"y\"], 0] would cause an evaluation error, leading to the\n * entire filter returning false. Legacy filter semantics, though, ask for\n * [\">\", \"y\", 0] to simply return `false` when `y` is of the wrong type,\n * allowing the subsequent terms of the outer \"any\" expression to be evaluated\n * (resulting, in this case, in a `true` value, because x > 0).\n *\n * We account for this by inserting a preflight type-checking expression before\n * each \"any\" term, allowing us to avoid evaluating the actual converted filter\n * if any type mismatches would cause it to produce an evalaution error:\n *\n * [\"any\",\n * [\"case\",\n * [\"all\", [\"==\", [\"typeof\", [\"get\", \"y\"]], \"number\"], [\"==\", [\"typeof\", [\"get\", \"z\"], \"number]],\n * [\"all\", [\">\", [\"get\", \"y\"], 0], [\">\", [\"get\", \"z\"], 0]],\n * false\n * ],\n * [\"case\",\n * [\"==\", [\"typeof\", [\"get\", \"x\"], \"number\"]],\n * [\">\", [\"get\", \"x\"], 0],\n * false\n * ]\n * ]\n *\n * An alternative, possibly more direct approach would be to use type checks\n * in the conversion of each comparison operator, so that the converted version\n * of each individual ==, >=, etc. would mimic the legacy filter semantics. The\n * downside of this approach is that it can lead to many more type checks than\n * would otherwise be necessary: outside the context of an \"any\" expression,\n * bailing out due to a runtime type error (expression semantics) and returning\n * false (legacy filter semantics) are equivalent: they cause the filter to\n * produce a `false` result.\n */\nfunction convertFilter(filter, expectedTypes = {}) {\n if (isExpressionFilter(filter))\n return filter;\n if (!filter)\n return true;\n const legacyFilter = filter;\n const legacyOp = legacyFilter[0];\n if (filter.length <= 1)\n return (legacyOp !== 'any');\n switch (legacyOp) {\n case '==':\n case '!=':\n case '<':\n case '>':\n case '<=':\n case '>=': {\n const [, property, value] = filter;\n return convertComparisonOp(property, value, legacyOp, expectedTypes);\n }\n case 'any': {\n const [, ...conditions] = legacyFilter;\n const children = conditions.map((f) => {\n const types = {};\n const child = convertFilter(f, types);\n const typechecks = runtimeTypeChecks(types);\n return typechecks === true ? child : ['case', typechecks, child, false];\n });\n return ['any', ...children];\n }\n case 'all': {\n const [, ...conditions] = legacyFilter;\n const children = conditions.map(f => convertFilter(f, expectedTypes));\n return children.length > 1 ? ['all', ...children] : children[0];\n }\n case 'none': {\n const [, ...conditions] = legacyFilter;\n return ['!', convertFilter(['any', ...conditions], {})];\n }\n case 'in': {\n const [, property, ...values] = legacyFilter;\n return convertInOp(property, values);\n }\n case '!in': {\n const [, property, ...values] = legacyFilter;\n return convertInOp(property, values, true);\n }\n case 'has':\n return convertHasOp(legacyFilter[1]);\n case '!has':\n return ['!', convertHasOp(legacyFilter[1])];\n default:\n return true;\n }\n}\n// Given a set of feature properties and an expected type for each one,\n// construct an boolean expression that tests whether each property has the\n// right type.\n// E.g.: for {name: 'string', population: 'number'}, return\n// [ 'all',\n// ['==', ['typeof', ['get', 'name'], 'string']],\n// ['==', ['typeof', ['get', 'population'], 'number]]\n// ]\nfunction runtimeTypeChecks(expectedTypes) {\n const conditions = [];\n for (const property in expectedTypes) {\n const get = property === '$id' ? ['id'] : ['get', property];\n conditions.push(['==', ['typeof', get], expectedTypes[property]]);\n }\n if (conditions.length === 0)\n return true;\n if (conditions.length === 1)\n return conditions[0];\n return ['all', ...conditions];\n}\nfunction convertComparisonOp(property, value, op, expectedTypes) {\n let get;\n if (property === '$type') {\n return [op, ['geometry-type'], value];\n }\n else if (property === '$id') {\n get = ['id'];\n }\n else {\n get = ['get', property];\n }\n if (expectedTypes && value !== null) {\n const type = typeof value;\n expectedTypes[property] = type;\n }\n if (op === '==' && property !== '$id' && value === null) {\n return [\n 'all',\n ['has', property],\n ['==', get, null]\n ];\n }\n else if (op === '!=' && property !== '$id' && value === null) {\n return [\n 'any',\n ['!', ['has', property]],\n ['!=', get, null]\n ];\n }\n return [op, get, value];\n}\nfunction convertInOp(property, values, negate = false) {\n if (values.length === 0)\n return negate;\n let get;\n if (property === '$type') {\n get = ['geometry-type'];\n }\n else if (property === '$id') {\n get = ['id'];\n }\n else {\n get = ['get', property];\n }\n // Determine if the list of values to be searched is homogenously typed.\n // If so (and if the type is string or number), then we can use a\n // [match, input, [...values], true, false] construction rather than a\n // bunch of `==` tests.\n let uniformTypes = true;\n const type = typeof values[0];\n for (const value of values) {\n if (typeof value !== type) {\n uniformTypes = false;\n break;\n }\n }\n if (uniformTypes && (type === 'string' || type === 'number')) {\n // Match expressions must have unique values.\n const uniqueValues = values.sort().filter((v, i) => i === 0 || values[i - 1] !== v);\n return ['match', get, uniqueValues, !negate, negate];\n }\n if (negate) {\n return ['all', ...values.map(v => ['!=', get, v])];\n }\n else {\n return ['any', ...values.map(v => ['==', get, v])];\n }\n}\nfunction convertHasOp(property) {\n if (property === '$type') {\n return true;\n }\n else if (property === '$id') {\n return ['!=', ['id'], null];\n }\n else {\n return ['has', property];\n }\n}\n\nfunction convertLiteral(value) {\n return typeof value === 'object' ? ['literal', value] : value;\n}\nfunction convertFunction(parameters, propertySpec) {\n let stops = parameters.stops;\n if (!stops) {\n // identity function\n return convertIdentityFunction(parameters, propertySpec);\n }\n const zoomAndFeatureDependent = stops && typeof stops[0][0] === 'object';\n const featureDependent = zoomAndFeatureDependent || parameters.property !== undefined;\n const zoomDependent = zoomAndFeatureDependent || !featureDependent;\n stops = stops.map((stop) => {\n if (!featureDependent && propertySpec.tokens && typeof stop[1] === 'string') {\n return [stop[0], convertTokenString(stop[1])];\n }\n return [stop[0], convertLiteral(stop[1])];\n });\n if (zoomAndFeatureDependent) {\n return convertZoomAndPropertyFunction(parameters, propertySpec, stops);\n }\n else if (zoomDependent) {\n return convertZoomFunction(parameters, propertySpec, stops);\n }\n else {\n return convertPropertyFunction(parameters, propertySpec, stops);\n }\n}\nfunction convertIdentityFunction(parameters, propertySpec) {\n const get = ['get', parameters.property];\n if (parameters.default === undefined) {\n // By default, expressions for string-valued properties get coerced. To preserve\n // legacy function semantics, insert an explicit assertion instead.\n return propertySpec.type === 'string' ? ['string', get] : get;\n }\n else if (propertySpec.type === 'enum') {\n return [\n 'match',\n get,\n Object.keys(propertySpec.values),\n get,\n parameters.default\n ];\n }\n else {\n const expression = [propertySpec.type === 'color' ? 'to-color' : propertySpec.type, get, convertLiteral(parameters.default)];\n if (propertySpec.type === 'array') {\n expression.splice(1, 0, propertySpec.value, propertySpec.length || null);\n }\n return expression;\n }\n}\nfunction getInterpolateOperator(parameters) {\n switch (parameters.colorSpace) {\n case 'hcl': return 'interpolate-hcl';\n case 'lab': return 'interpolate-lab';\n default: return 'interpolate';\n }\n}\nfunction convertZoomAndPropertyFunction(parameters, propertySpec, stops) {\n const featureFunctionParameters = {};\n const featureFunctionStops = {};\n const zoomStops = [];\n for (let s = 0; s < stops.length; s++) {\n const stop = stops[s];\n const zoom = stop[0].zoom;\n if (featureFunctionParameters[zoom] === undefined) {\n featureFunctionParameters[zoom] = {\n zoom,\n type: parameters.type,\n property: parameters.property,\n default: parameters.default,\n };\n featureFunctionStops[zoom] = [];\n zoomStops.push(zoom);\n }\n featureFunctionStops[zoom].push([stop[0].value, stop[1]]);\n }\n // the interpolation type for the zoom dimension of a zoom-and-property\n // function is determined directly from the style property specification\n // for which it's being used: linear for interpolatable properties, step\n // otherwise.\n const functionType = getFunctionType({}, propertySpec);\n if (functionType === 'exponential') {\n const expression = [getInterpolateOperator(parameters), ['linear'], ['zoom']];\n for (const z of zoomStops) {\n const output = convertPropertyFunction(featureFunctionParameters[z], propertySpec, featureFunctionStops[z]);\n appendStopPair(expression, z, output, false);\n }\n return expression;\n }\n else {\n const expression = ['step', ['zoom']];\n for (const z of zoomStops) {\n const output = convertPropertyFunction(featureFunctionParameters[z], propertySpec, featureFunctionStops[z]);\n appendStopPair(expression, z, output, true);\n }\n fixupDegenerateStepCurve(expression);\n return expression;\n }\n}\nfunction coalesce(a, b) {\n if (a !== undefined)\n return a;\n if (b !== undefined)\n return b;\n}\nfunction getFallback(parameters, propertySpec) {\n const defaultValue = convertLiteral(coalesce(parameters.default, propertySpec.default));\n /*\n * Some fields with type: resolvedImage have an undefined default.\n * Because undefined is an invalid value for resolvedImage, set fallback to\n * an empty string instead of undefined to ensure output\n * passes validation.\n */\n if (defaultValue === undefined && propertySpec.type === 'resolvedImage') {\n return '';\n }\n return defaultValue;\n}\nfunction convertPropertyFunction(parameters, propertySpec, stops) {\n const type = getFunctionType(parameters, propertySpec);\n const get = ['get', parameters.property];\n if (type === 'categorical' && typeof stops[0][0] === 'boolean') {\n const expression = ['case'];\n for (const stop of stops) {\n expression.push(['==', get, stop[0]], stop[1]);\n }\n expression.push(getFallback(parameters, propertySpec));\n return expression;\n }\n else if (type === 'categorical') {\n const expression = ['match', get];\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], false);\n }\n expression.push(getFallback(parameters, propertySpec));\n return expression;\n }\n else if (type === 'interval') {\n const expression = ['step', ['number', get]];\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], true);\n }\n fixupDegenerateStepCurve(expression);\n return parameters.default === undefined ? expression : [\n 'case',\n ['==', ['typeof', get], 'number'],\n expression,\n convertLiteral(parameters.default)\n ];\n }\n else if (type === 'exponential') {\n const base = parameters.base !== undefined ? parameters.base : 1;\n const expression = [\n getInterpolateOperator(parameters),\n base === 1 ? ['linear'] : ['exponential', base],\n ['number', get]\n ];\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], false);\n }\n return parameters.default === undefined ? expression : [\n 'case',\n ['==', ['typeof', get], 'number'],\n expression,\n convertLiteral(parameters.default)\n ];\n }\n else {\n throw new Error(`Unknown property function type ${type}`);\n }\n}\nfunction convertZoomFunction(parameters, propertySpec, stops, input = ['zoom']) {\n const type = getFunctionType(parameters, propertySpec);\n let expression;\n let isStep = false;\n if (type === 'interval') {\n expression = ['step', input];\n isStep = true;\n }\n else if (type === 'exponential') {\n const base = parameters.base !== undefined ? parameters.base : 1;\n expression = [getInterpolateOperator(parameters), base === 1 ? ['linear'] : ['exponential', base], input];\n }\n else {\n throw new Error(`Unknown zoom function type \"${type}\"`);\n }\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], isStep);\n }\n fixupDegenerateStepCurve(expression);\n return expression;\n}\nfunction fixupDegenerateStepCurve(expression) {\n // degenerate step curve (i.e. a constant function): add a noop stop\n if (expression[0] === 'step' && expression.length === 3) {\n expression.push(0);\n expression.push(expression[3]);\n }\n}\nfunction appendStopPair(curve, input, output, isStep) {\n // Skip duplicate stop values. They were not validated for functions, but they are for expressions.\n // https://github.com/mapbox/mapbox-gl-js/issues/4107\n if (curve.length > 3 && input === curve[curve.length - 2]) {\n return;\n }\n // step curves don't get the first input value, as it is redundant.\n if (!(isStep && curve.length === 2)) {\n curve.push(input);\n }\n curve.push(output);\n}\nfunction getFunctionType(parameters, propertySpec) {\n if (parameters.type) {\n return parameters.type;\n }\n else {\n return propertySpec.expression.interpolated ? 'exponential' : 'interval';\n }\n}\n// \"String with {name} token\" => [\"concat\", \"String with \", [\"get\", \"name\"], \" token\"]\nfunction convertTokenString(s) {\n const result = ['concat'];\n const re = /{([^{}]+)}/g;\n let pos = 0;\n for (let match = re.exec(s); match !== null; match = re.exec(s)) {\n const literal = s.slice(pos, re.lastIndex - match[0].length);\n pos = re.lastIndex;\n if (literal.length > 0)\n result.push(literal);\n result.push(['get', match[1]]);\n }\n if (result.length === 1) {\n return s;\n }\n if (pos < s.length) {\n result.push(s.slice(pos));\n }\n else if (result.length === 2) {\n return ['to-string', result[1]];\n }\n return result;\n}\n\nfunction getPropertyReference(propertyName) {\n for (let i = 0; i < v8Spec.layout.length; i++) {\n for (const key in v8Spec[v8Spec.layout[i]]) {\n if (key === propertyName)\n return v8Spec[v8Spec.layout[i]][key];\n }\n }\n for (let i = 0; i < v8Spec.paint.length; i++) {\n for (const key in v8Spec[v8Spec.paint[i]]) {\n if (key === propertyName)\n return v8Spec[v8Spec.paint[i]][key];\n }\n }\n return null;\n}\nfunction eachSource(style, callback) {\n for (const k in style.sources) {\n callback(style.sources[k]);\n }\n}\nfunction eachLayer(style, callback) {\n for (const layer of style.layers) {\n callback(layer);\n }\n}\nfunction eachProperty(style, options, callback) {\n function inner(layer, propertyType) {\n const properties = layer[propertyType];\n if (!properties)\n return;\n Object.keys(properties).forEach((key) => {\n callback({\n path: [layer.id, propertyType, key],\n key,\n value: properties[key],\n reference: getPropertyReference(key),\n set(x) {\n properties[key] = x;\n }\n });\n });\n }\n eachLayer(style, (layer) => {\n if (options.paint) {\n inner(layer, 'paint');\n }\n if (options.layout) {\n inner(layer, 'layout');\n }\n });\n}\n\nfunction stringify(obj) {\n const type = typeof obj;\n if (type === 'number' || type === 'boolean' || type === 'string' || obj === undefined || obj === null)\n return JSON.stringify(obj);\n if (Array.isArray(obj)) {\n let str = '[';\n for (const val of obj) {\n str += `${stringify(val)},`;\n }\n return `${str}]`;\n }\n const keys = Object.keys(obj).sort();\n let str = '{';\n for (let i = 0; i < keys.length; i++) {\n str += `${JSON.stringify(keys[i])}:${stringify(obj[keys[i]])},`;\n }\n return `${str}}`;\n}\nfunction getKey(layer) {\n let key = '';\n for (const k of refProperties) {\n key += `/${stringify(layer[k])}`;\n }\n return key;\n}\n/**\n * Given an array of layers, return an array of arrays of layers where all\n * layers in each group have identical layout-affecting properties. These\n * are the properties that were formerly used by explicit `ref` mechanism\n * for layers: 'type', 'source', 'source-layer', 'minzoom', 'maxzoom',\n * 'filter', and 'layout'.\n *\n * The input is not modified. The output layers are references to the\n * input layers.\n *\n * @private\n * @param {Array} layers\n * @param {Object} [cachedKeys] - an object to keep already calculated keys.\n * @returns {Array>}\n */\nfunction groupByLayout(layers, cachedKeys) {\n const groups = {};\n for (let i = 0; i < layers.length; i++) {\n const k = (cachedKeys && cachedKeys[layers[i].id]) || getKey(layers[i]);\n // update the cache if there is one\n if (cachedKeys)\n cachedKeys[layers[i].id] = k;\n let group = groups[k];\n if (!group) {\n group = groups[k] = [];\n }\n group.push(layers[i]);\n }\n const result = [];\n for (const k in groups) {\n result.push(groups[k]);\n }\n return result;\n}\n\nfunction emptyStyle() {\n const style = {};\n const version = v8Spec['$version'];\n for (const styleKey in v8Spec['$root']) {\n const spec = v8Spec['$root'][styleKey];\n if (spec.required) {\n let value = null;\n if (styleKey === 'version') {\n value = version;\n }\n else {\n if (spec.type === 'array') {\n value = [];\n }\n else {\n value = {};\n }\n }\n if (value != null) {\n style[styleKey] = value;\n }\n }\n }\n return style;\n}\n\nfunction validateConstants(options) {\n const key = options.key;\n const constants = options.value;\n if (constants) {\n return [new ValidationError(key, constants, 'constants have been deprecated as of v8')];\n }\n else {\n return [];\n }\n}\n\n// Turn jsonlint-lines-primitives objects into primitive objects\nfunction unbundle(value) {\n if (value instanceof Number || value instanceof String || value instanceof Boolean) {\n return value.valueOf();\n }\n else {\n return value;\n }\n}\nfunction deepUnbundle(value) {\n if (Array.isArray(value)) {\n return value.map(deepUnbundle);\n }\n else if (value instanceof Object && !(value instanceof Number || value instanceof String || value instanceof Boolean)) {\n const unbundledValue = {};\n for (const key in value) {\n unbundledValue[key] = deepUnbundle(value[key]);\n }\n return unbundledValue;\n }\n return unbundle(value);\n}\n\nfunction validateObject(options) {\n const key = options.key;\n const object = options.value;\n const elementSpecs = options.valueSpec || {};\n const elementValidators = options.objectElementValidators || {};\n const style = options.style;\n const styleSpec = options.styleSpec;\n const validateSpec = options.validateSpec;\n let errors = [];\n const type = getType(object);\n if (type !== 'object') {\n return [new ValidationError(key, object, `object expected, ${type} found`)];\n }\n for (const objectKey in object) {\n const elementSpecKey = objectKey.split('.')[0]; // treat 'paint.*' as 'paint'\n const elementSpec = elementSpecs[elementSpecKey] || elementSpecs['*'];\n let validateElement;\n if (elementValidators[elementSpecKey]) {\n validateElement = elementValidators[elementSpecKey];\n }\n else if (elementSpecs[elementSpecKey]) {\n validateElement = validateSpec;\n }\n else if (elementValidators['*']) {\n validateElement = elementValidators['*'];\n }\n else if (elementSpecs['*']) {\n validateElement = validateSpec;\n }\n else {\n errors.push(new ValidationError(key, object[objectKey], `unknown property \"${objectKey}\"`));\n continue;\n }\n errors = errors.concat(validateElement({\n key: (key ? `${key}.` : key) + objectKey,\n value: object[objectKey],\n valueSpec: elementSpec,\n style,\n styleSpec,\n object,\n objectKey,\n validateSpec,\n }, object));\n }\n for (const elementSpecKey in elementSpecs) {\n // Don't check `required` when there's a custom validator for that property.\n if (elementValidators[elementSpecKey]) {\n continue;\n }\n if (elementSpecs[elementSpecKey].required && elementSpecs[elementSpecKey]['default'] === undefined && object[elementSpecKey] === undefined) {\n errors.push(new ValidationError(key, object, `missing required property \"${elementSpecKey}\"`));\n }\n }\n return errors;\n}\n\nfunction validateArray(options) {\n const array = options.value;\n const arraySpec = options.valueSpec;\n const validateSpec = options.validateSpec;\n const style = options.style;\n const styleSpec = options.styleSpec;\n const key = options.key;\n const validateArrayElement = options.arrayElementValidator || validateSpec;\n if (getType(array) !== 'array') {\n return [new ValidationError(key, array, `array expected, ${getType(array)} found`)];\n }\n if (arraySpec.length && array.length !== arraySpec.length) {\n return [new ValidationError(key, array, `array length ${arraySpec.length} expected, length ${array.length} found`)];\n }\n if (arraySpec['min-length'] && array.length < arraySpec['min-length']) {\n return [new ValidationError(key, array, `array length at least ${arraySpec['min-length']} expected, length ${array.length} found`)];\n }\n let arrayElementSpec = {\n 'type': arraySpec.value,\n 'values': arraySpec.values\n };\n if (styleSpec.$version < 7) {\n arrayElementSpec['function'] = arraySpec.function;\n }\n if (getType(arraySpec.value) === 'object') {\n arrayElementSpec = arraySpec.value;\n }\n let errors = [];\n for (let i = 0; i < array.length; i++) {\n errors = errors.concat(validateArrayElement({\n array,\n arrayIndex: i,\n value: array[i],\n valueSpec: arrayElementSpec,\n validateSpec: options.validateSpec,\n style,\n styleSpec,\n key: `${key}[${i}]`\n }));\n }\n return errors;\n}\n\nfunction validateNumber(options) {\n const key = options.key;\n const value = options.value;\n const valueSpec = options.valueSpec;\n let type = getType(value);\n // eslint-disable-next-line no-self-compare\n if (type === 'number' && value !== value) {\n type = 'NaN';\n }\n if (type !== 'number') {\n return [new ValidationError(key, value, `number expected, ${type} found`)];\n }\n if ('minimum' in valueSpec && value < valueSpec.minimum) {\n return [new ValidationError(key, value, `${value} is less than the minimum value ${valueSpec.minimum}`)];\n }\n if ('maximum' in valueSpec && value > valueSpec.maximum) {\n return [new ValidationError(key, value, `${value} is greater than the maximum value ${valueSpec.maximum}`)];\n }\n return [];\n}\n\nfunction validateFunction(options) {\n const functionValueSpec = options.valueSpec;\n const functionType = unbundle(options.value.type);\n let stopKeyType;\n let stopDomainValues = {};\n let previousStopDomainValue;\n let previousStopDomainZoom;\n const isZoomFunction = functionType !== 'categorical' && options.value.property === undefined;\n const isPropertyFunction = !isZoomFunction;\n const isZoomAndPropertyFunction = getType(options.value.stops) === 'array' &&\n getType(options.value.stops[0]) === 'array' &&\n getType(options.value.stops[0][0]) === 'object';\n const errors = validateObject({\n key: options.key,\n value: options.value,\n valueSpec: options.styleSpec.function,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec,\n objectElementValidators: {\n stops: validateFunctionStops,\n default: validateFunctionDefault\n }\n });\n if (functionType === 'identity' && isZoomFunction) {\n errors.push(new ValidationError(options.key, options.value, 'missing required property \"property\"'));\n }\n if (functionType !== 'identity' && !options.value.stops) {\n errors.push(new ValidationError(options.key, options.value, 'missing required property \"stops\"'));\n }\n if (functionType === 'exponential' && options.valueSpec.expression && !supportsInterpolation(options.valueSpec)) {\n errors.push(new ValidationError(options.key, options.value, 'exponential functions not supported'));\n }\n if (options.styleSpec.$version >= 8) {\n if (isPropertyFunction && !supportsPropertyExpression(options.valueSpec)) {\n errors.push(new ValidationError(options.key, options.value, 'property functions not supported'));\n }\n else if (isZoomFunction && !supportsZoomExpression(options.valueSpec)) {\n errors.push(new ValidationError(options.key, options.value, 'zoom functions not supported'));\n }\n }\n if ((functionType === 'categorical' || isZoomAndPropertyFunction) && options.value.property === undefined) {\n errors.push(new ValidationError(options.key, options.value, '\"property\" property is required'));\n }\n return errors;\n function validateFunctionStops(options) {\n if (functionType === 'identity') {\n return [new ValidationError(options.key, options.value, 'identity function may not have a \"stops\" property')];\n }\n let errors = [];\n const value = options.value;\n errors = errors.concat(validateArray({\n key: options.key,\n value,\n valueSpec: options.valueSpec,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec,\n arrayElementValidator: validateFunctionStop\n }));\n if (getType(value) === 'array' && value.length === 0) {\n errors.push(new ValidationError(options.key, value, 'array must have at least one stop'));\n }\n return errors;\n }\n function validateFunctionStop(options) {\n let errors = [];\n const value = options.value;\n const key = options.key;\n if (getType(value) !== 'array') {\n return [new ValidationError(key, value, `array expected, ${getType(value)} found`)];\n }\n if (value.length !== 2) {\n return [new ValidationError(key, value, `array length 2 expected, length ${value.length} found`)];\n }\n if (isZoomAndPropertyFunction) {\n if (getType(value[0]) !== 'object') {\n return [new ValidationError(key, value, `object expected, ${getType(value[0])} found`)];\n }\n if (value[0].zoom === undefined) {\n return [new ValidationError(key, value, 'object stop key must have zoom')];\n }\n if (value[0].value === undefined) {\n return [new ValidationError(key, value, 'object stop key must have value')];\n }\n if (previousStopDomainZoom && previousStopDomainZoom > unbundle(value[0].zoom)) {\n return [new ValidationError(key, value[0].zoom, 'stop zoom values must appear in ascending order')];\n }\n if (unbundle(value[0].zoom) !== previousStopDomainZoom) {\n previousStopDomainZoom = unbundle(value[0].zoom);\n previousStopDomainValue = undefined;\n stopDomainValues = {};\n }\n errors = errors.concat(validateObject({\n key: `${key}[0]`,\n value: value[0],\n valueSpec: { zoom: {} },\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec,\n objectElementValidators: { zoom: validateNumber, value: validateStopDomainValue }\n }));\n }\n else {\n errors = errors.concat(validateStopDomainValue({\n key: `${key}[0]`,\n value: value[0],\n valueSpec: {},\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec\n }, value));\n }\n if (isExpression(deepUnbundle(value[1]))) {\n return errors.concat([new ValidationError(`${key}[1]`, value[1], 'expressions are not allowed in function stops.')]);\n }\n return errors.concat(options.validateSpec({\n key: `${key}[1]`,\n value: value[1],\n valueSpec: functionValueSpec,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec\n }));\n }\n function validateStopDomainValue(options, stop) {\n const type = getType(options.value);\n const value = unbundle(options.value);\n const reportValue = options.value !== null ? options.value : stop;\n if (!stopKeyType) {\n stopKeyType = type;\n }\n else if (type !== stopKeyType) {\n return [new ValidationError(options.key, reportValue, `${type} stop domain type must match previous stop domain type ${stopKeyType}`)];\n }\n if (type !== 'number' && type !== 'string' && type !== 'boolean') {\n return [new ValidationError(options.key, reportValue, 'stop domain value must be a number, string, or boolean')];\n }\n if (type !== 'number' && functionType !== 'categorical') {\n let message = `number expected, ${type} found`;\n if (supportsPropertyExpression(functionValueSpec) && functionType === undefined) {\n message += '\\nIf you intended to use a categorical function, specify `\"type\": \"categorical\"`.';\n }\n return [new ValidationError(options.key, reportValue, message)];\n }\n if (functionType === 'categorical' && type === 'number' && (!isFinite(value) || Math.floor(value) !== value)) {\n return [new ValidationError(options.key, reportValue, `integer expected, found ${value}`)];\n }\n if (functionType !== 'categorical' && type === 'number' && previousStopDomainValue !== undefined && value < previousStopDomainValue) {\n return [new ValidationError(options.key, reportValue, 'stop domain values must appear in ascending order')];\n }\n else {\n previousStopDomainValue = value;\n }\n if (functionType === 'categorical' && value in stopDomainValues) {\n return [new ValidationError(options.key, reportValue, 'stop domain values must be unique')];\n }\n else {\n stopDomainValues[value] = true;\n }\n return [];\n }\n function validateFunctionDefault(options) {\n return options.validateSpec({\n key: options.key,\n value: options.value,\n valueSpec: functionValueSpec,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec\n });\n }\n}\n\nfunction validateExpression(options) {\n const expression = (options.expressionContext === 'property' ? createPropertyExpression : createExpression)(deepUnbundle(options.value), options.valueSpec);\n if (expression.result === 'error') {\n return expression.value.map((error) => {\n return new ValidationError(`${options.key}${error.key}`, options.value, error.message);\n });\n }\n const expressionObj = expression.value.expression || expression.value._styleExpression.expression;\n if (options.expressionContext === 'property' && (options.propertyKey === 'text-font') &&\n !expressionObj.outputDefined()) {\n return [new ValidationError(options.key, options.value, `Invalid data expression for \"${options.propertyKey}\". Output values must be contained as literals within the expression.`)];\n }\n if (options.expressionContext === 'property' && options.propertyType === 'layout' &&\n (!isStateConstant(expressionObj))) {\n return [new ValidationError(options.key, options.value, '\"feature-state\" data expressions are not supported with layout properties.')];\n }\n if (options.expressionContext === 'filter' && !isStateConstant(expressionObj)) {\n return [new ValidationError(options.key, options.value, '\"feature-state\" data expressions are not supported with filters.')];\n }\n if (options.expressionContext && options.expressionContext.indexOf('cluster') === 0) {\n if (!isGlobalPropertyConstant(expressionObj, ['zoom', 'feature-state'])) {\n return [new ValidationError(options.key, options.value, '\"zoom\" and \"feature-state\" expressions are not supported with cluster properties.')];\n }\n if (options.expressionContext === 'cluster-initial' && !isFeatureConstant(expressionObj)) {\n return [new ValidationError(options.key, options.value, 'Feature data expressions are not supported with initial expression part of cluster properties.')];\n }\n }\n return [];\n}\n\nfunction validateBoolean(options) {\n const value = options.value;\n const key = options.key;\n const type = getType(value);\n if (type !== 'boolean') {\n return [new ValidationError(key, value, `boolean expected, ${type} found`)];\n }\n return [];\n}\n\nfunction validateColor(options) {\n const key = options.key;\n const value = options.value;\n const type = getType(value);\n if (type !== 'string') {\n return [new ValidationError(key, value, `color expected, ${type} found`)];\n }\n if (!Color.parse(String(value))) { // cast String object to string primitive\n return [new ValidationError(key, value, `color expected, \"${value}\" found`)];\n }\n return [];\n}\n\nfunction validateEnum(options) {\n const key = options.key;\n const value = options.value;\n const valueSpec = options.valueSpec;\n const errors = [];\n if (Array.isArray(valueSpec.values)) { // <=v7\n if (valueSpec.values.indexOf(unbundle(value)) === -1) {\n errors.push(new ValidationError(key, value, `expected one of [${valueSpec.values.join(', ')}], ${JSON.stringify(value)} found`));\n }\n }\n else { // >=v8\n if (Object.keys(valueSpec.values).indexOf(unbundle(value)) === -1) {\n errors.push(new ValidationError(key, value, `expected one of [${Object.keys(valueSpec.values).join(', ')}], ${JSON.stringify(value)} found`));\n }\n }\n return errors;\n}\n\nfunction validateFilter(options) {\n if (isExpressionFilter(deepUnbundle(options.value))) {\n return validateExpression(extendBy({}, options, {\n expressionContext: 'filter',\n valueSpec: { value: 'boolean' }\n }));\n }\n else {\n return validateNonExpressionFilter(options);\n }\n}\nfunction validateNonExpressionFilter(options) {\n const value = options.value;\n const key = options.key;\n if (getType(value) !== 'array') {\n return [new ValidationError(key, value, `array expected, ${getType(value)} found`)];\n }\n const styleSpec = options.styleSpec;\n let type;\n let errors = [];\n if (value.length < 1) {\n return [new ValidationError(key, value, 'filter array must have at least 1 element')];\n }\n errors = errors.concat(validateEnum({\n key: `${key}[0]`,\n value: value[0],\n valueSpec: styleSpec.filter_operator,\n style: options.style,\n styleSpec: options.styleSpec\n }));\n switch (unbundle(value[0])) {\n case '<':\n case '<=':\n case '>':\n case '>=':\n if (value.length >= 2 && unbundle(value[1]) === '$type') {\n errors.push(new ValidationError(key, value, `\"$type\" cannot be use with operator \"${value[0]}\"`));\n }\n /* falls through */\n case '==':\n case '!=':\n if (value.length !== 3) {\n errors.push(new ValidationError(key, value, `filter array for operator \"${value[0]}\" must have 3 elements`));\n }\n /* falls through */\n case 'in':\n case '!in':\n if (value.length >= 2) {\n type = getType(value[1]);\n if (type !== 'string') {\n errors.push(new ValidationError(`${key}[1]`, value[1], `string expected, ${type} found`));\n }\n }\n for (let i = 2; i < value.length; i++) {\n type = getType(value[i]);\n if (unbundle(value[1]) === '$type') {\n errors = errors.concat(validateEnum({\n key: `${key}[${i}]`,\n value: value[i],\n valueSpec: styleSpec.geometry_type,\n style: options.style,\n styleSpec: options.styleSpec\n }));\n }\n else if (type !== 'string' && type !== 'number' && type !== 'boolean') {\n errors.push(new ValidationError(`${key}[${i}]`, value[i], `string, number, or boolean expected, ${type} found`));\n }\n }\n break;\n case 'any':\n case 'all':\n case 'none':\n for (let i = 1; i < value.length; i++) {\n errors = errors.concat(validateNonExpressionFilter({\n key: `${key}[${i}]`,\n value: value[i],\n style: options.style,\n styleSpec: options.styleSpec\n }));\n }\n break;\n case 'has':\n case '!has':\n type = getType(value[1]);\n if (value.length !== 2) {\n errors.push(new ValidationError(key, value, `filter array for \"${value[0]}\" operator must have 2 elements`));\n }\n else if (type !== 'string') {\n errors.push(new ValidationError(`${key}[1]`, value[1], `string expected, ${type} found`));\n }\n break;\n case 'within':\n type = getType(value[1]);\n if (value.length !== 2) {\n errors.push(new ValidationError(key, value, `filter array for \"${value[0]}\" operator must have 2 elements`));\n }\n else if (type !== 'object') {\n errors.push(new ValidationError(`${key}[1]`, value[1], `object expected, ${type} found`));\n }\n break;\n }\n return errors;\n}\n\nfunction validateProperty(options, propertyType) {\n const key = options.key;\n const validateSpec = options.validateSpec;\n const style = options.style;\n const styleSpec = options.styleSpec;\n const value = options.value;\n const propertyKey = options.objectKey;\n const layerSpec = styleSpec[`${propertyType}_${options.layerType}`];\n if (!layerSpec)\n return [];\n const transitionMatch = propertyKey.match(/^(.*)-transition$/);\n if (propertyType === 'paint' && transitionMatch && layerSpec[transitionMatch[1]] && layerSpec[transitionMatch[1]].transition) {\n return validateSpec({\n key,\n value,\n valueSpec: styleSpec.transition,\n style,\n styleSpec\n });\n }\n const valueSpec = options.valueSpec || layerSpec[propertyKey];\n if (!valueSpec) {\n return [new ValidationError(key, value, `unknown property \"${propertyKey}\"`)];\n }\n let tokenMatch;\n if (getType(value) === 'string' && supportsPropertyExpression(valueSpec) && !valueSpec.tokens && (tokenMatch = /^{([^}]+)}$/.exec(value))) {\n return [new ValidationError(key, value, `\"${propertyKey}\" does not support interpolation syntax\\n` +\n `Use an identity property function instead: \\`{ \"type\": \"identity\", \"property\": ${JSON.stringify(tokenMatch[1])} }\\`.`)];\n }\n const errors = [];\n if (options.layerType === 'symbol') {\n if (propertyKey === 'text-field' && style && !style.glyphs) {\n errors.push(new ValidationError(key, value, 'use of \"text-field\" requires a style \"glyphs\" property'));\n }\n if (propertyKey === 'text-font' && isFunction(deepUnbundle(value)) && unbundle(value.type) === 'identity') {\n errors.push(new ValidationError(key, value, '\"text-font\" does not support identity functions'));\n }\n }\n return errors.concat(validateSpec({\n key: options.key,\n value,\n valueSpec,\n style,\n styleSpec,\n expressionContext: 'property',\n propertyType,\n propertyKey\n }));\n}\n\nfunction validatePaintProperty(options) {\n return validateProperty(options, 'paint');\n}\n\nfunction validateLayoutProperty(options) {\n return validateProperty(options, 'layout');\n}\n\nfunction validateLayer(options) {\n let errors = [];\n const layer = options.value;\n const key = options.key;\n const style = options.style;\n const styleSpec = options.styleSpec;\n if (!layer.type && !layer.ref) {\n errors.push(new ValidationError(key, layer, 'either \"type\" or \"ref\" is required'));\n }\n let type = unbundle(layer.type);\n const ref = unbundle(layer.ref);\n if (layer.id) {\n const layerId = unbundle(layer.id);\n for (let i = 0; i < options.arrayIndex; i++) {\n const otherLayer = style.layers[i];\n if (unbundle(otherLayer.id) === layerId) {\n errors.push(new ValidationError(key, layer.id, `duplicate layer id \"${layer.id}\", previously used at line ${otherLayer.id.__line__}`));\n }\n }\n }\n if ('ref' in layer) {\n ['type', 'source', 'source-layer', 'filter', 'layout'].forEach((p) => {\n if (p in layer) {\n errors.push(new ValidationError(key, layer[p], `\"${p}\" is prohibited for ref layers`));\n }\n });\n let parent;\n style.layers.forEach((layer) => {\n if (unbundle(layer.id) === ref)\n parent = layer;\n });\n if (!parent) {\n errors.push(new ValidationError(key, layer.ref, `ref layer \"${ref}\" not found`));\n }\n else if (parent.ref) {\n errors.push(new ValidationError(key, layer.ref, 'ref cannot reference another ref layer'));\n }\n else {\n type = unbundle(parent.type);\n }\n }\n else if (type !== 'background') {\n if (!layer.source) {\n errors.push(new ValidationError(key, layer, 'missing required property \"source\"'));\n }\n else {\n const source = style.sources && style.sources[layer.source];\n const sourceType = source && unbundle(source.type);\n if (!source) {\n errors.push(new ValidationError(key, layer.source, `source \"${layer.source}\" not found`));\n }\n else if (sourceType === 'vector' && type === 'raster') {\n errors.push(new ValidationError(key, layer.source, `layer \"${layer.id}\" requires a raster source`));\n }\n else if (sourceType !== 'raster-dem' && type === 'hillshade') {\n errors.push(new ValidationError(key, layer.source, `layer \"${layer.id}\" requires a raster-dem source`));\n }\n else if (sourceType === 'raster' && type !== 'raster') {\n errors.push(new ValidationError(key, layer.source, `layer \"${layer.id}\" requires a vector source`));\n }\n else if (sourceType === 'vector' && !layer['source-layer']) {\n errors.push(new ValidationError(key, layer, `layer \"${layer.id}\" must specify a \"source-layer\"`));\n }\n else if (sourceType === 'raster-dem' && type !== 'hillshade') {\n errors.push(new ValidationError(key, layer.source, 'raster-dem source can only be used with layer type \\'hillshade\\'.'));\n }\n else if (type === 'line' && layer.paint && layer.paint['line-gradient'] &&\n (sourceType !== 'geojson' || !source.lineMetrics)) {\n errors.push(new ValidationError(key, layer, `layer \"${layer.id}\" specifies a line-gradient, which requires a GeoJSON source with \\`lineMetrics\\` enabled.`));\n }\n }\n }\n errors = errors.concat(validateObject({\n key,\n value: layer,\n valueSpec: styleSpec.layer,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n objectElementValidators: {\n '*'() {\n return [];\n },\n // We don't want to enforce the spec's `\"requires\": true` for backward compatibility with refs;\n // the actual requirement is validated above. See https://github.com/mapbox/mapbox-gl-js/issues/5772.\n type() {\n return options.validateSpec({\n key: `${key}.type`,\n value: layer.type,\n valueSpec: styleSpec.layer.type,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n object: layer,\n objectKey: 'type'\n });\n },\n filter: validateFilter,\n layout(options) {\n return validateObject({\n layer,\n key: options.key,\n value: options.value,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n objectElementValidators: {\n '*'(options) {\n return validateLayoutProperty(extendBy({ layerType: type }, options));\n }\n }\n });\n },\n paint(options) {\n return validateObject({\n layer,\n key: options.key,\n value: options.value,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n objectElementValidators: {\n '*'(options) {\n return validatePaintProperty(extendBy({ layerType: type }, options));\n }\n }\n });\n }\n }\n }));\n return errors;\n}\n\nfunction validateString(options) {\n const value = options.value;\n const key = options.key;\n const type = getType(value);\n if (type !== 'string') {\n return [new ValidationError(key, value, `string expected, ${type} found`)];\n }\n return [];\n}\n\nfunction validateRasterDEMSource(options) {\n var _a;\n const sourceName = (_a = options.sourceName) !== null && _a !== void 0 ? _a : '';\n const rasterDEM = options.value;\n const styleSpec = options.styleSpec;\n const rasterDEMSpec = styleSpec.source_raster_dem;\n const style = options.style;\n let errors = [];\n const rootType = getType(rasterDEM);\n if (rasterDEM === undefined) {\n return errors;\n }\n else if (rootType !== 'object') {\n errors.push(new ValidationError('source_raster_dem', rasterDEM, `object expected, ${rootType} found`));\n return errors;\n }\n const encoding = unbundle(rasterDEM.encoding);\n const isCustomEncoding = encoding === 'custom';\n const customEncodingKeys = ['redFactor', 'greenFactor', 'blueFactor', 'baseShift'];\n const encodingName = options.value.encoding ? `\"${options.value.encoding}\"` : 'Default';\n for (const key in rasterDEM) {\n if (!isCustomEncoding && customEncodingKeys.includes(key)) {\n errors.push(new ValidationError(key, rasterDEM[key], `In \"${sourceName}\": \"${key}\" is only valid when \"encoding\" is set to \"custom\". ${encodingName} encoding found`));\n }\n else if (rasterDEMSpec[key]) {\n errors = errors.concat(options.validateSpec({\n key,\n value: rasterDEM[key],\n valueSpec: rasterDEMSpec[key],\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else {\n errors.push(new ValidationError(key, rasterDEM[key], `unknown property \"${key}\"`));\n }\n }\n return errors;\n}\n\nconst objectElementValidators = {\n promoteId: validatePromoteId\n};\nfunction validateSource(options) {\n const value = options.value;\n const key = options.key;\n const styleSpec = options.styleSpec;\n const style = options.style;\n const validateSpec = options.validateSpec;\n if (!value.type) {\n return [new ValidationError(key, value, '\"type\" is required')];\n }\n const type = unbundle(value.type);\n let errors;\n switch (type) {\n case 'vector':\n case 'raster':\n errors = validateObject({\n key,\n value,\n valueSpec: styleSpec[`source_${type.replace('-', '_')}`],\n style: options.style,\n styleSpec,\n objectElementValidators,\n validateSpec,\n });\n return errors;\n case 'raster-dem':\n errors = validateRasterDEMSource({\n sourceName: key,\n value,\n style: options.style,\n styleSpec,\n validateSpec,\n });\n return errors;\n case 'geojson':\n errors = validateObject({\n key,\n value,\n valueSpec: styleSpec.source_geojson,\n style,\n styleSpec,\n validateSpec,\n objectElementValidators\n });\n if (value.cluster) {\n for (const prop in value.clusterProperties) {\n const [operator, mapExpr] = value.clusterProperties[prop];\n const reduceExpr = typeof operator === 'string' ? [operator, ['accumulated'], ['get', prop]] : operator;\n errors.push(...validateExpression({\n key: `${key}.${prop}.map`,\n value: mapExpr,\n validateSpec,\n expressionContext: 'cluster-map'\n }));\n errors.push(...validateExpression({\n key: `${key}.${prop}.reduce`,\n value: reduceExpr,\n validateSpec,\n expressionContext: 'cluster-reduce'\n }));\n }\n }\n return errors;\n case 'video':\n return validateObject({\n key,\n value,\n valueSpec: styleSpec.source_video,\n style,\n validateSpec,\n styleSpec\n });\n case 'image':\n return validateObject({\n key,\n value,\n valueSpec: styleSpec.source_image,\n style,\n validateSpec,\n styleSpec\n });\n case 'canvas':\n return [new ValidationError(key, null, 'Please use runtime APIs to add canvas sources, rather than including them in stylesheets.', 'source.canvas')];\n default:\n return validateEnum({\n key: `${key}.type`,\n value: value.type,\n valueSpec: { values: ['vector', 'raster', 'raster-dem', 'geojson', 'video', 'image'] },\n style,\n validateSpec,\n styleSpec\n });\n }\n}\nfunction validatePromoteId({ key, value }) {\n if (getType(value) === 'string') {\n return validateString({ key, value });\n }\n else {\n const errors = [];\n for (const prop in value) {\n errors.push(...validateString({ key: `${key}.${prop}`, value: value[prop] }));\n }\n return errors;\n }\n}\n\nfunction validateLight(options) {\n const light = options.value;\n const styleSpec = options.styleSpec;\n const lightSpec = styleSpec.light;\n const style = options.style;\n let errors = [];\n const rootType = getType(light);\n if (light === undefined) {\n return errors;\n }\n else if (rootType !== 'object') {\n errors = errors.concat([new ValidationError('light', light, `object expected, ${rootType} found`)]);\n return errors;\n }\n for (const key in light) {\n const transitionMatch = key.match(/^(.*)-transition$/);\n if (transitionMatch && lightSpec[transitionMatch[1]] && lightSpec[transitionMatch[1]].transition) {\n errors = errors.concat(options.validateSpec({\n key,\n value: light[key],\n valueSpec: styleSpec.transition,\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else if (lightSpec[key]) {\n errors = errors.concat(options.validateSpec({\n key,\n value: light[key],\n valueSpec: lightSpec[key],\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else {\n errors = errors.concat([new ValidationError(key, light[key], `unknown property \"${key}\"`)]);\n }\n }\n return errors;\n}\n\nfunction validateTerrain(options) {\n const terrain = options.value;\n const styleSpec = options.styleSpec;\n const terrainSpec = styleSpec.terrain;\n const style = options.style;\n let errors = [];\n const rootType = getType(terrain);\n if (terrain === undefined) {\n return errors;\n }\n else if (rootType !== 'object') {\n errors = errors.concat([new ValidationError('terrain', terrain, `object expected, ${rootType} found`)]);\n return errors;\n }\n for (const key in terrain) {\n if (terrainSpec[key]) {\n errors = errors.concat(options.validateSpec({\n key,\n value: terrain[key],\n valueSpec: terrainSpec[key],\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else {\n errors = errors.concat([new ValidationError(key, terrain[key], `unknown property \"${key}\"`)]);\n }\n }\n return errors;\n}\n\nfunction validateFormatted(options) {\n if (validateString(options).length === 0) {\n return [];\n }\n return validateExpression(options);\n}\n\nfunction validateImage(options) {\n if (validateString(options).length === 0) {\n return [];\n }\n return validateExpression(options);\n}\n\nfunction validatePadding(options) {\n const key = options.key;\n const value = options.value;\n const type = getType(value);\n if (type === 'array') {\n if (value.length < 1 || value.length > 4) {\n return [new ValidationError(key, value, `padding requires 1 to 4 values; ${value.length} values found`)];\n }\n const arrayElementSpec = {\n type: 'number'\n };\n let errors = [];\n for (let i = 0; i < value.length; i++) {\n errors = errors.concat(options.validateSpec({\n key: `${key}[${i}]`,\n value: value[i],\n validateSpec: options.validateSpec,\n valueSpec: arrayElementSpec\n }));\n }\n return errors;\n }\n else {\n return validateNumber({\n key,\n value,\n valueSpec: {}\n });\n }\n}\n\nfunction validateVariableAnchorOffsetCollection(options) {\n const key = options.key;\n const value = options.value;\n const type = getType(value);\n const styleSpec = options.styleSpec;\n if (type !== 'array' || value.length < 1 || value.length % 2 !== 0) {\n return [new ValidationError(key, value, 'variableAnchorOffsetCollection requires a non-empty array of even length')];\n }\n let errors = [];\n for (let i = 0; i < value.length; i += 2) {\n // Elements in even positions should be values from text-anchor enum\n errors = errors.concat(validateEnum({\n key: `${key}[${i}]`,\n value: value[i],\n valueSpec: styleSpec['layout_symbol']['text-anchor']\n }));\n // Elements in odd positions should be points (2-element numeric arrays)\n errors = errors.concat(validateArray({\n key: `${key}[${i + 1}]`,\n value: value[i + 1],\n valueSpec: {\n length: 2,\n value: 'number'\n },\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec\n }));\n }\n return errors;\n}\n\nfunction validateSprite(options) {\n let errors = [];\n const sprite = options.value;\n const key = options.key;\n if (!Array.isArray(sprite)) {\n return validateString({\n key,\n value: sprite\n });\n }\n else {\n const allSpriteIds = [];\n const allSpriteURLs = [];\n for (const i in sprite) {\n if (sprite[i].id && allSpriteIds.includes(sprite[i].id))\n errors.push(new ValidationError(key, sprite, `all the sprites' ids must be unique, but ${sprite[i].id} is duplicated`));\n allSpriteIds.push(sprite[i].id);\n if (sprite[i].url && allSpriteURLs.includes(sprite[i].url))\n errors.push(new ValidationError(key, sprite, `all the sprites' URLs must be unique, but ${sprite[i].url} is duplicated`));\n allSpriteURLs.push(sprite[i].url);\n const pairSpec = {\n id: {\n type: 'string',\n required: true,\n },\n url: {\n type: 'string',\n required: true,\n }\n };\n errors = errors.concat(validateObject({\n key: `${key}[${i}]`,\n value: sprite[i],\n valueSpec: pairSpec,\n validateSpec: options.validateSpec,\n }));\n }\n return errors;\n }\n}\n\nconst VALIDATORS = {\n '*'() {\n return [];\n },\n 'array': validateArray,\n 'boolean': validateBoolean,\n 'number': validateNumber,\n 'color': validateColor,\n 'constants': validateConstants,\n 'enum': validateEnum,\n 'filter': validateFilter,\n 'function': validateFunction,\n 'layer': validateLayer,\n 'object': validateObject,\n 'source': validateSource,\n 'light': validateLight,\n 'terrain': validateTerrain,\n 'string': validateString,\n 'formatted': validateFormatted,\n 'resolvedImage': validateImage,\n 'padding': validatePadding,\n 'variableAnchorOffsetCollection': validateVariableAnchorOffsetCollection,\n 'sprite': validateSprite,\n};\n// Main recursive validation function. Tracks:\n//\n// - key: string representing location of validation in style tree. Used only\n// for more informative error reporting.\n// - value: current value from style being evaluated. May be anything from a\n// high level object that needs to be descended into deeper or a simple\n// scalar value.\n// - valueSpec: current spec being evaluated. Tracks value.\n// - styleSpec: current full spec being evaluated.\nfunction validate(options) {\n const value = options.value;\n const valueSpec = options.valueSpec;\n const styleSpec = options.styleSpec;\n options.validateSpec = validate;\n if (valueSpec.expression && isFunction(unbundle(value))) {\n return validateFunction(options);\n }\n else if (valueSpec.expression && isExpression(deepUnbundle(value))) {\n return validateExpression(options);\n }\n else if (valueSpec.type && VALIDATORS[valueSpec.type]) {\n return VALIDATORS[valueSpec.type](options);\n }\n else {\n const valid = validateObject(extendBy({}, options, {\n valueSpec: valueSpec.type ? styleSpec[valueSpec.type] : valueSpec\n }));\n return valid;\n }\n}\n\nfunction validateGlyphsUrl(options) {\n const value = options.value;\n const key = options.key;\n const errors = validateString(options);\n if (errors.length)\n return errors;\n if (value.indexOf('{fontstack}') === -1) {\n errors.push(new ValidationError(key, value, '\"glyphs\" url must include a \"{fontstack}\" token'));\n }\n if (value.indexOf('{range}') === -1) {\n errors.push(new ValidationError(key, value, '\"glyphs\" url must include a \"{range}\" token'));\n }\n return errors;\n}\n\n/**\n * Validate a MapLibre style against the style specification. This entrypoint,\n * `maplibre-gl-style-spec/lib/validate_style.min`, is designed to produce as\n * small a browserify bundle as possible by omitting unnecessary functionality\n * and legacy style specifications.\n *\n * @private\n * @param {Object} style The style to be validated.\n * @param {Object} [styleSpec] The style specification to validate against.\n * If omitted, the latest style spec is used.\n * @returns {Array}\n * @example\n * var validate = require('maplibre-gl-style-spec/lib/validate_style.min');\n * var errors = validate(style);\n */\nfunction validateStyleMin(style, styleSpec = v8Spec) {\n let errors = [];\n errors = errors.concat(validate({\n key: '',\n value: style,\n valueSpec: styleSpec.$root,\n styleSpec,\n style,\n validateSpec: validate,\n objectElementValidators: {\n glyphs: validateGlyphsUrl,\n '*'() {\n return [];\n }\n }\n }));\n if (style['constants']) {\n errors = errors.concat(validateConstants({\n key: 'constants',\n value: style['constants'],\n style,\n styleSpec,\n validateSpec: validate,\n }));\n }\n return sortErrors(errors);\n}\nvalidateStyleMin.source = wrapCleanErrors(injectValidateSpec(validateSource));\nvalidateStyleMin.sprite = wrapCleanErrors(injectValidateSpec(validateSprite));\nvalidateStyleMin.glyphs = wrapCleanErrors(injectValidateSpec(validateGlyphsUrl));\nvalidateStyleMin.light = wrapCleanErrors(injectValidateSpec(validateLight));\nvalidateStyleMin.terrain = wrapCleanErrors(injectValidateSpec(validateTerrain));\nvalidateStyleMin.layer = wrapCleanErrors(injectValidateSpec(validateLayer));\nvalidateStyleMin.filter = wrapCleanErrors(injectValidateSpec(validateFilter));\nvalidateStyleMin.paintProperty = wrapCleanErrors(injectValidateSpec(validatePaintProperty));\nvalidateStyleMin.layoutProperty = wrapCleanErrors(injectValidateSpec(validateLayoutProperty));\nfunction injectValidateSpec(validator) {\n return function (options) {\n return validator({\n ...options,\n validateSpec: validate,\n });\n };\n}\nfunction sortErrors(errors) {\n return [].concat(errors).sort((a, b) => {\n return a.line - b.line;\n });\n}\nfunction wrapCleanErrors(inner) {\n return function (...args) {\n return sortErrors(inner.apply(this, args));\n };\n}\n\nconst v8 = v8Spec;\nconst expression = {\n StyleExpression,\n StylePropertyFunction,\n ZoomConstantExpression,\n ZoomDependentExpression,\n createExpression,\n createPropertyExpression,\n isExpression,\n isExpressionFilter,\n isZoomExpression,\n normalizePropertyExpression,\n};\nconst styleFunction = {\n convertFunction,\n createFunction,\n isFunction\n};\nconst visit = { eachLayer, eachProperty, eachSource };\n\nexport { Color, ColorType, CompoundExpression, EvaluationContext, FormatExpression, Formatted, FormattedSection, FormattedType, Interpolate, Literal, NullType, Padding, ParsingError, ResolvedImage, Step, StyleExpression, StylePropertyFunction, ValidationError, VariableAnchorOffsetCollection, ZoomConstantExpression, ZoomDependentExpression, convertFilter, convertFunction, createExpression, createFunction, createPropertyExpression, derefLayers, diffStyles as diff, emptyStyle, expression, expressions, createFilter as featureFilter, styleFunction as function, groupByLayout, interpolateFactory, interpolate as interpolates, isExpression, isFunction, isZoomExpression, v8Spec as latest, normalizePropertyExpression, operations, supportsPropertyExpression, toString$1 as toString, typeOf, v8, validateStyleMin, visit };\n//# sourceMappingURL=index.mjs.map\n","import {validateStyleMin} from '@maplibre/maplibre-gl-style-spec';\nimport {ErrorEvent} from '../util/evented';\n\nimport type {Evented} from '../util/evented';\n\ntype ValidationError = {\n message: string;\n line: number;\n identifier?: string;\n};\n\nexport type Validator = (a: any) => ReadonlyArray;\n\ntype ValidateStyle = {\n source: Validator;\n sprite: Validator;\n glyphs: Validator;\n layer: Validator;\n light: Validator;\n terrain: Validator;\n filter: Validator;\n paintProperty: Validator;\n layoutProperty: Validator;\n (b: any, a?: any | null): ReadonlyArray;\n};\n\nexport const validateStyle = (validateStyleMin as unknown as ValidateStyle);\n\nexport const validateSource = validateStyle.source;\nexport const validateLight = validateStyle.light;\nexport const validateTerrain = validateStyle.terrain;\nexport const validateFilter = validateStyle.filter;\nexport const validatePaintProperty = validateStyle.paintProperty;\nexport const validateLayoutProperty = validateStyle.layoutProperty;\n\nexport function emitValidationErrors(\n emitter: Evented,\n errors?: ReadonlyArray<{\n message: string;\n identifier?: string;\n }> | null\n): boolean {\n let hasErrors = false;\n if (errors && errors.length) {\n for (const error of errors) {\n emitter.fire(new ErrorEvent(new Error(error.message)));\n hasErrors = true;\n }\n }\n return hasErrors;\n}\n","/*\nThis file was copied from https://github.com/mapbox/grid-index and was\nmigrated from JavaScript to TypeScript.\n\nCopyright (c) 2016, Mapbox\n\nPermission to use, copy, modify, and/or distribute this software for any purpose\nwith or without fee is hereby granted, provided that the above copyright notice\nand this permission notice appear in all copies.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND\nFITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS\nOF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER\nTORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF\nTHIS SOFTWARE.\n*/\n\nconst NUM_PARAMS = 3;\n\nexport type SerializedGrid = {\n buffer: ArrayBuffer;\n};\n\nexport class TransferableGridIndex {\n cells: number[][];\n arrayBuffer: ArrayBuffer;\n d: number;\n keys: number[];\n bboxes: number[];\n n: number;\n extent: number;\n padding: number;\n scale: any;\n uid: number;\n min: number;\n max: number;\n\n constructor(extent: number | ArrayBuffer, n?: number, padding?: number) {\n const cells = this.cells = [];\n\n if (extent instanceof ArrayBuffer) {\n this.arrayBuffer = extent;\n const array = new Int32Array(this.arrayBuffer);\n extent = array[0];\n n = array[1];\n padding = array[2];\n\n this.d = n + 2 * padding;\n for (let k = 0; k < this.d * this.d; k++) {\n const start = array[NUM_PARAMS + k];\n const end = array[NUM_PARAMS + k + 1];\n cells.push(start === end ? null : array.subarray(start, end));\n }\n const keysOffset = array[NUM_PARAMS + cells.length];\n const bboxesOffset = array[NUM_PARAMS + cells.length + 1];\n this.keys = array.subarray(keysOffset, bboxesOffset) as any as number[];\n this.bboxes = array.subarray(bboxesOffset) as any as number[];\n\n this.insert = this._insertReadonly;\n\n } else {\n this.d = n + 2 * padding;\n for (let i = 0; i < this.d * this.d; i++) {\n cells.push([]);\n }\n this.keys = [];\n this.bboxes = [];\n }\n\n this.n = n;\n this.extent = extent;\n this.padding = padding;\n this.scale = n / extent;\n this.uid = 0;\n\n const p = (padding / n) * extent;\n this.min = -p;\n this.max = extent + p;\n }\n\n insert(key: number, x1: number, y1: number, x2: number, y2: number) {\n this._forEachCell(x1, y1, x2, y2, this._insertCell, this.uid++, undefined, undefined);\n this.keys.push(key);\n this.bboxes.push(x1);\n this.bboxes.push(y1);\n this.bboxes.push(x2);\n this.bboxes.push(y2);\n }\n\n _insertReadonly() {\n throw new Error('Cannot insert into a GridIndex created from an ArrayBuffer.');\n }\n\n _insertCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) {\n this.cells[cellIndex].push(uid);\n }\n\n query(x1: number, y1: number, x2: number, y2: number, intersectionTest?: Function): number[] {\n const min = this.min;\n const max = this.max;\n if (x1 <= min && y1 <= min && max <= x2 && max <= y2 && !intersectionTest) {\n // We use `Array#slice` because `this.keys` may be a `Int32Array` and\n // some browsers (Safari and IE) do not support `TypedArray#slice`\n // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/TypedArray/slice#Browser_compatibility\n return Array.prototype.slice.call(this.keys);\n\n } else {\n const result = [];\n const seenUids = {};\n this._forEachCell(x1, y1, x2, y2, this._queryCell, result, seenUids, intersectionTest);\n return result;\n }\n }\n\n _queryCell(x1: number, y1: number, x2: number, y2:number, cellIndex:number, result, seenUids, intersectionTest: Function) {\n const cell = this.cells[cellIndex];\n if (cell !== null) {\n const keys = this.keys;\n const bboxes = this.bboxes;\n for (let u = 0; u < cell.length; u++) {\n const uid = cell[u];\n if (seenUids[uid] === undefined) {\n const offset = uid * 4;\n if (intersectionTest ?\n intersectionTest(bboxes[offset + 0], bboxes[offset + 1], bboxes[offset + 2], bboxes[offset + 3]) :\n ((x1 <= bboxes[offset + 2]) &&\n (y1 <= bboxes[offset + 3]) &&\n (x2 >= bboxes[offset + 0]) &&\n (y2 >= bboxes[offset + 1]))) {\n seenUids[uid] = true;\n result.push(keys[uid]);\n } else {\n seenUids[uid] = false;\n }\n }\n }\n }\n }\n\n _forEachCell(x1: number, y1: number, x2:number, y2:number, fn: Function, arg1, arg2, intersectionTest) {\n const cx1 = this._convertToCellCoord(x1);\n const cy1 = this._convertToCellCoord(y1);\n const cx2 = this._convertToCellCoord(x2);\n const cy2 = this._convertToCellCoord(y2);\n for (let x = cx1; x <= cx2; x++) {\n for (let y = cy1; y <= cy2; y++) {\n const cellIndex = this.d * y + x;\n if (intersectionTest && !intersectionTest(\n this._convertFromCellCoord(x),\n this._convertFromCellCoord(y),\n this._convertFromCellCoord(x + 1),\n this._convertFromCellCoord(y + 1))) continue;\n if (fn.call(this, x1, y1, x2, y2, cellIndex, arg1, arg2, intersectionTest)) return;\n }\n }\n }\n\n _convertFromCellCoord (x) {\n return (x - this.padding) / this.scale;\n }\n\n _convertToCellCoord(x) {\n return Math.max(0, Math.min(this.d - 1, Math.floor(x * this.scale) + this.padding));\n }\n\n toArrayBuffer(): ArrayBuffer {\n if (this.arrayBuffer) return this.arrayBuffer;\n\n const cells = this.cells;\n\n const metadataLength = NUM_PARAMS + this.cells.length + 1 + 1;\n let totalCellLength = 0;\n for (let i = 0; i < this.cells.length; i++) {\n totalCellLength += this.cells[i].length;\n }\n\n const array = new Int32Array(metadataLength + totalCellLength + this.keys.length + this.bboxes.length);\n array[0] = this.extent;\n array[1] = this.n;\n array[2] = this.padding;\n\n let offset = metadataLength;\n for (let k = 0; k < cells.length; k++) {\n const cell = cells[k];\n array[NUM_PARAMS + k] = offset;\n array.set(cell, offset);\n offset += cell.length;\n }\n\n array[NUM_PARAMS + cells.length] = offset;\n array.set(this.keys, offset);\n offset += this.keys.length;\n\n array[NUM_PARAMS + cells.length + 1] = offset;\n array.set(this.bboxes, offset);\n offset += this.bboxes.length;\n\n return array.buffer;\n }\n\n public static serialize(grid: TransferableGridIndex, transferables?: Array): SerializedGrid {\n const buffer = grid.toArrayBuffer();\n if (transferables) {\n transferables.push(buffer);\n }\n return {buffer};\n }\n\n public static deserialize(serialized: SerializedGrid): TransferableGridIndex {\n return new TransferableGridIndex(serialized.buffer);\n }\n}\n","import {TransferableGridIndex} from './transferable_grid_index';\nimport {Color, CompoundExpression, expressions, ResolvedImage, StylePropertyFunction,\n StyleExpression, ZoomDependentExpression, ZoomConstantExpression} from '@maplibre/maplibre-gl-style-spec';\n\nimport {AJAXError} from './ajax';\n\nimport type {Transferable} from '../types/transferable';\nimport {isImageBitmap} from './util';\n\ntype SerializedObject = {\n [_: string]: S;\n};\n\nexport type Serialized = null | void | boolean | number | string | Boolean | Number | String | Date | RegExp | ArrayBuffer | ArrayBufferView | ImageData | ImageBitmap | Blob | Array | SerializedObject;\n\ntype Registry = {\n [_: string]: {\n klass: {\n new (...args: any): any;\n deserialize?: (input: Serialized) => unknown;\n };\n omit: ReadonlyArray;\n shallow: ReadonlyArray;\n };\n};\n\n/**\n * Register options\n */\ntype RegisterOptions = {\n /**\n * List of properties to omit from serialization (e.g., cached/computed properties)\n */\n omit?: ReadonlyArray;\n /**\n * List of properties that should be serialized by a simple shallow copy, rather than by a recursive call to serialize().\n */\n shallow?: ReadonlyArray;\n};\n\nconst registry: Registry = {};\n\n/**\n * Register the given class as serializable.\n *\n * @param options - the registration options\n */\nexport function register(\n name: string,\n klass: {\n new (...args: any): T;\n },\n options: RegisterOptions = {}\n) {\n if (registry[name]) throw new Error(`${name} is already registered.`);\n ((Object.defineProperty as any))(klass, '_classRegistryKey', {\n value: name,\n writeable: false\n });\n registry[name] = {\n klass,\n omit: options.omit as ReadonlyArray || [],\n shallow: options.shallow as ReadonlyArray || []\n };\n}\n\nregister('Object', Object);\nregister('TransferableGridIndex', TransferableGridIndex);\n\nregister('Color', Color);\nregister('Error', Error);\nregister('AJAXError', AJAXError);\nregister('ResolvedImage', ResolvedImage);\n\nregister('StylePropertyFunction', StylePropertyFunction);\nregister('StyleExpression', StyleExpression, {omit: ['_evaluator']});\n\nregister('ZoomDependentExpression', ZoomDependentExpression);\nregister('ZoomConstantExpression', ZoomConstantExpression);\nregister('CompoundExpression', CompoundExpression, {omit: ['_evaluate']});\nfor (const name in expressions) {\n if ((expressions[name] as any)._classRegistryKey) continue;\n register(`Expression_${name}`, expressions[name]);\n}\n\nfunction isArrayBuffer(value: any): value is ArrayBuffer {\n return value && typeof ArrayBuffer !== 'undefined' &&\n (value instanceof ArrayBuffer || (value.constructor && value.constructor.name === 'ArrayBuffer'));\n}\n\n/**\n * Serialize the given object for transfer to or from a web worker.\n *\n * For non-builtin types, recursively serialize each property (possibly\n * omitting certain properties - see register()), and package the result along\n * with the constructor's `name` so that the appropriate constructor can be\n * looked up in `deserialize()`.\n *\n * If a `transferables` array is provided, add any transferable objects (i.e.,\n * any ArrayBuffers or ArrayBuffer views) to the list. (If a copy is needed,\n * this should happen in the client code, before using serialize().)\n */\nexport function serialize(input: unknown, transferables?: Array | null): Serialized {\n if (input === null ||\n input === undefined ||\n typeof input === 'boolean' ||\n typeof input === 'number' ||\n typeof input === 'string' ||\n input instanceof Boolean ||\n input instanceof Number ||\n input instanceof String ||\n input instanceof Date ||\n input instanceof RegExp ||\n input instanceof Blob) {\n return input;\n }\n\n if (isArrayBuffer(input)) {\n if (transferables) {\n transferables.push(input);\n }\n return input;\n }\n\n if (isImageBitmap(input)) {\n if (transferables) {\n transferables.push(input);\n }\n return input;\n }\n\n if (ArrayBuffer.isView(input)) {\n const view = input;\n if (transferables) {\n transferables.push(view.buffer);\n }\n return view;\n }\n\n if (input instanceof ImageData) {\n if (transferables) {\n transferables.push(input.data.buffer);\n }\n return input;\n }\n\n if (Array.isArray(input)) {\n const serialized: Array = [];\n for (const item of input) {\n serialized.push(serialize(item, transferables));\n }\n return serialized;\n }\n\n if (typeof input === 'object') {\n const klass = (input.constructor as any);\n const name = klass._classRegistryKey;\n if (!name) {\n throw new Error('can\\'t serialize object of unregistered class');\n }\n if (!registry[name]) throw new Error(`${name} is not registered.`);\n\n const properties: SerializedObject = klass.serialize ?\n // (Temporary workaround) allow a class to provide static\n // `serialize()` and `deserialize()` methods to bypass the generic\n // approach.\n // This temporary workaround lets us use the generic serialization\n // approach for objects whose members include instances of dynamic\n // StructArray types. Once we refactor StructArray to be static,\n // we can remove this complexity.\n (klass.serialize(input, transferables) as SerializedObject) : {};\n\n if (!klass.serialize) {\n for (const key in input) {\n if (!input.hasOwnProperty(key)) continue; // eslint-disable-line no-prototype-builtins\n if (registry[name].omit.indexOf(key) >= 0) continue;\n const property = input[key];\n properties[key] = registry[name].shallow.indexOf(key) >= 0 ?\n property :\n serialize(property, transferables);\n }\n if (input instanceof Error) {\n properties.message = input.message;\n }\n } else {\n if (transferables && properties === transferables[transferables.length - 1]) {\n throw new Error('statically serialized object won\\'t survive transfer of $name property');\n }\n }\n\n if (properties.$name) {\n throw new Error('$name property is reserved for worker serialization logic.');\n }\n if (name !== 'Object') {\n properties.$name = name;\n }\n\n return properties;\n }\n\n throw new Error(`can't serialize object of type ${typeof input}`);\n}\n\nexport function deserialize(input: Serialized): unknown {\n if (input === null ||\n input === undefined ||\n typeof input === 'boolean' ||\n typeof input === 'number' ||\n typeof input === 'string' ||\n input instanceof Boolean ||\n input instanceof Number ||\n input instanceof String ||\n input instanceof Date ||\n input instanceof RegExp ||\n input instanceof Blob ||\n isArrayBuffer(input) ||\n isImageBitmap(input) ||\n ArrayBuffer.isView(input) ||\n input instanceof ImageData) {\n return input;\n }\n\n if (Array.isArray(input)) {\n return input.map(deserialize);\n }\n\n if (typeof input === 'object') {\n const name = input.$name || 'Object';\n if (!registry[name]) {\n throw new Error(`can't deserialize unregistered class ${name}`);\n }\n const {klass} = registry[name];\n if (!klass) {\n throw new Error(`can't deserialize unregistered class ${name}`);\n }\n\n if (klass.deserialize) {\n return klass.deserialize(input);\n }\n\n const result = Object.create(klass.prototype);\n\n for (const key of Object.keys(input)) {\n if (key === '$name') continue;\n const value = (input as SerializedObject)[key];\n result[key] = registry[name].shallow.indexOf(key) >= 0 ? value : deserialize(value);\n }\n\n return result;\n }\n\n throw new Error(`can't deserialize object of type ${typeof input}`);\n}\n","export class ZoomHistory {\n lastZoom: number;\n lastFloorZoom: number;\n lastIntegerZoom: number;\n lastIntegerZoomTime: number;\n first: boolean;\n\n constructor() {\n this.first = true;\n }\n\n update(z: number, now: number) {\n const floorZ = Math.floor(z);\n\n if (this.first) {\n this.first = false;\n this.lastIntegerZoom = floorZ;\n this.lastIntegerZoomTime = 0;\n this.lastZoom = z;\n this.lastFloorZoom = floorZ;\n return true;\n }\n\n if (this.lastFloorZoom > floorZ) {\n this.lastIntegerZoom = floorZ + 1;\n this.lastIntegerZoomTime = now;\n } else if (this.lastFloorZoom < floorZ) {\n this.lastIntegerZoom = floorZ;\n this.lastIntegerZoomTime = now;\n }\n\n if (z !== this.lastZoom) {\n this.lastZoom = z;\n this.lastFloorZoom = floorZ;\n return true;\n }\n\n return false;\n }\n}\n","// The following table comes from .\n// Keep it synchronized with .\n\ntype UnicodeBlockLookup = {[key: string]: (char: number) => boolean};\n\nexport const unicodeBlockLookup: UnicodeBlockLookup = {\n // 'Basic Latin': (char) => char >= 0x0000 && char <= 0x007F,\n 'Latin-1 Supplement': (char) => char >= 0x0080 && char <= 0x00FF,\n // 'Latin Extended-A': (char) => char >= 0x0100 && char <= 0x017F,\n // 'Latin Extended-B': (char) => char >= 0x0180 && char <= 0x024F,\n // 'IPA Extensions': (char) => char >= 0x0250 && char <= 0x02AF,\n // 'Spacing Modifier Letters': (char) => char >= 0x02B0 && char <= 0x02FF,\n // 'Combining Diacritical Marks': (char) => char >= 0x0300 && char <= 0x036F,\n // 'Greek and Coptic': (char) => char >= 0x0370 && char <= 0x03FF,\n // 'Cyrillic': (char) => char >= 0x0400 && char <= 0x04FF,\n // 'Cyrillic Supplement': (char) => char >= 0x0500 && char <= 0x052F,\n // 'Armenian': (char) => char >= 0x0530 && char <= 0x058F,\n //'Hebrew': (char) => char >= 0x0590 && char <= 0x05FF,\n 'Arabic': (char) => char >= 0x0600 && char <= 0x06FF,\n //'Syriac': (char) => char >= 0x0700 && char <= 0x074F,\n 'Arabic Supplement': (char) => char >= 0x0750 && char <= 0x077F,\n // 'Thaana': (char) => char >= 0x0780 && char <= 0x07BF,\n // 'NKo': (char) => char >= 0x07C0 && char <= 0x07FF,\n // 'Samaritan': (char) => char >= 0x0800 && char <= 0x083F,\n // 'Mandaic': (char) => char >= 0x0840 && char <= 0x085F,\n // 'Syriac Supplement': (char) => char >= 0x0860 && char <= 0x086F,\n 'Arabic Extended-A': (char) => char >= 0x08A0 && char <= 0x08FF,\n // 'Devanagari': (char) => char >= 0x0900 && char <= 0x097F,\n // 'Bengali': (char) => char >= 0x0980 && char <= 0x09FF,\n // 'Gurmukhi': (char) => char >= 0x0A00 && char <= 0x0A7F,\n // 'Gujarati': (char) => char >= 0x0A80 && char <= 0x0AFF,\n // 'Oriya': (char) => char >= 0x0B00 && char <= 0x0B7F,\n // 'Tamil': (char) => char >= 0x0B80 && char <= 0x0BFF,\n // 'Telugu': (char) => char >= 0x0C00 && char <= 0x0C7F,\n // 'Kannada': (char) => char >= 0x0C80 && char <= 0x0CFF,\n // 'Malayalam': (char) => char >= 0x0D00 && char <= 0x0D7F,\n // 'Sinhala': (char) => char >= 0x0D80 && char <= 0x0DFF,\n // 'Thai': (char) => char >= 0x0E00 && char <= 0x0E7F,\n // 'Lao': (char) => char >= 0x0E80 && char <= 0x0EFF,\n // 'Tibetan': (char) => char >= 0x0F00 && char <= 0x0FFF,\n // 'Myanmar': (char) => char >= 0x1000 && char <= 0x109F,\n // 'Georgian': (char) => char >= 0x10A0 && char <= 0x10FF,\n 'Hangul Jamo': (char) => char >= 0x1100 && char <= 0x11FF,\n // 'Ethiopic': (char) => char >= 0x1200 && char <= 0x137F,\n // 'Ethiopic Supplement': (char) => char >= 0x1380 && char <= 0x139F,\n // 'Cherokee': (char) => char >= 0x13A0 && char <= 0x13FF,\n 'Unified Canadian Aboriginal Syllabics': (char) => char >= 0x1400 && char <= 0x167F,\n // 'Ogham': (char) => char >= 0x1680 && char <= 0x169F,\n // 'Runic': (char) => char >= 0x16A0 && char <= 0x16FF,\n // 'Tagalog': (char) => char >= 0x1700 && char <= 0x171F,\n // 'Hanunoo': (char) => char >= 0x1720 && char <= 0x173F,\n // 'Buhid': (char) => char >= 0x1740 && char <= 0x175F,\n // 'Tagbanwa': (char) => char >= 0x1760 && char <= 0x177F,\n 'Khmer': (char) => char >= 0x1780 && char <= 0x17FF,\n // 'Mongolian': (char) => char >= 0x1800 && char <= 0x18AF,\n 'Unified Canadian Aboriginal Syllabics Extended': (char) => char >= 0x18B0 && char <= 0x18FF,\n // 'Limbu': (char) => char >= 0x1900 && char <= 0x194F,\n // 'Tai Le': (char) => char >= 0x1950 && char <= 0x197F,\n // 'New Tai Lue': (char) => char >= 0x1980 && char <= 0x19DF,\n // 'Khmer Symbols': (char) => char >= 0x19E0 && char <= 0x19FF,\n // 'Buginese': (char) => char >= 0x1A00 && char <= 0x1A1F,\n // 'Tai Tham': (char) => char >= 0x1A20 && char <= 0x1AAF,\n // 'Combining Diacritical Marks Extended': (char) => char >= 0x1AB0 && char <= 0x1AFF,\n // 'Balinese': (char) => char >= 0x1B00 && char <= 0x1B7F,\n // 'Sundanese': (char) => char >= 0x1B80 && char <= 0x1BBF,\n // 'Batak': (char) => char >= 0x1BC0 && char <= 0x1BFF,\n // 'Lepcha': (char) => char >= 0x1C00 && char <= 0x1C4F,\n // 'Ol Chiki': (char) => char >= 0x1C50 && char <= 0x1C7F,\n // 'Cyrillic Extended-C': (char) => char >= 0x1C80 && char <= 0x1C8F,\n // 'Georgian Extended': (char) => char >= 0x1C90 && char <= 0x1CBF,\n // 'Sundanese Supplement': (char) => char >= 0x1CC0 && char <= 0x1CCF,\n // 'Vedic Extensions': (char) => char >= 0x1CD0 && char <= 0x1CFF,\n // 'Phonetic Extensions': (char) => char >= 0x1D00 && char <= 0x1D7F,\n // 'Phonetic Extensions Supplement': (char) => char >= 0x1D80 && char <= 0x1DBF,\n // 'Combining Diacritical Marks Supplement': (char) => char >= 0x1DC0 && char <= 0x1DFF,\n // 'Latin Extended Additional': (char) => char >= 0x1E00 && char <= 0x1EFF,\n // 'Greek Extended': (char) => char >= 0x1F00 && char <= 0x1FFF,\n 'General Punctuation': (char) => char >= 0x2000 && char <= 0x206F,\n // 'Superscripts and Subscripts': (char) => char >= 0x2070 && char <= 0x209F,\n // 'Currency Symbols': (char) => char >= 0x20A0 && char <= 0x20CF,\n // 'Combining Diacritical Marks for Symbols': (char) => char >= 0x20D0 && char <= 0x20FF,\n 'Letterlike Symbols': (char) => char >= 0x2100 && char <= 0x214F,\n 'Number Forms': (char) => char >= 0x2150 && char <= 0x218F,\n // 'Arrows': (char) => char >= 0x2190 && char <= 0x21FF,\n // 'Mathematical Operators': (char) => char >= 0x2200 && char <= 0x22FF,\n 'Miscellaneous Technical': (char) => char >= 0x2300 && char <= 0x23FF,\n 'Control Pictures': (char) => char >= 0x2400 && char <= 0x243F,\n 'Optical Character Recognition': (char) => char >= 0x2440 && char <= 0x245F,\n 'Enclosed Alphanumerics': (char) => char >= 0x2460 && char <= 0x24FF,\n // 'Box Drawing': (char) => char >= 0x2500 && char <= 0x257F,\n // 'Block Elements': (char) => char >= 0x2580 && char <= 0x259F,\n 'Geometric Shapes': (char) => char >= 0x25A0 && char <= 0x25FF,\n 'Miscellaneous Symbols': (char) => char >= 0x2600 && char <= 0x26FF,\n // 'Dingbats': (char) => char >= 0x2700 && char <= 0x27BF,\n // 'Miscellaneous Mathematical Symbols-A': (char) => char >= 0x27C0 && char <= 0x27EF,\n // 'Supplemental Arrows-A': (char) => char >= 0x27F0 && char <= 0x27FF,\n // 'Braille Patterns': (char) => char >= 0x2800 && char <= 0x28FF,\n // 'Supplemental Arrows-B': (char) => char >= 0x2900 && char <= 0x297F,\n // 'Miscellaneous Mathematical Symbols-B': (char) => char >= 0x2980 && char <= 0x29FF,\n // 'Supplemental Mathematical Operators': (char) => char >= 0x2A00 && char <= 0x2AFF,\n 'Miscellaneous Symbols and Arrows': (char) => char >= 0x2B00 && char <= 0x2BFF,\n // 'Glagolitic': (char) => char >= 0x2C00 && char <= 0x2C5F,\n // 'Latin Extended-C': (char) => char >= 0x2C60 && char <= 0x2C7F,\n // 'Coptic': (char) => char >= 0x2C80 && char <= 0x2CFF,\n // 'Georgian Supplement': (char) => char >= 0x2D00 && char <= 0x2D2F,\n // 'Tifinagh': (char) => char >= 0x2D30 && char <= 0x2D7F,\n // 'Ethiopic Extended': (char) => char >= 0x2D80 && char <= 0x2DDF,\n // 'Cyrillic Extended-A': (char) => char >= 0x2DE0 && char <= 0x2DFF,\n // 'Supplemental Punctuation': (char) => char >= 0x2E00 && char <= 0x2E7F,\n 'CJK Radicals Supplement': (char) => char >= 0x2E80 && char <= 0x2EFF,\n 'Kangxi Radicals': (char) => char >= 0x2F00 && char <= 0x2FDF,\n 'Ideographic Description Characters': (char) => char >= 0x2FF0 && char <= 0x2FFF,\n 'CJK Symbols and Punctuation': (char) => char >= 0x3000 && char <= 0x303F,\n 'Hiragana': (char) => char >= 0x3040 && char <= 0x309F,\n 'Katakana': (char) => char >= 0x30A0 && char <= 0x30FF,\n 'Bopomofo': (char) => char >= 0x3100 && char <= 0x312F,\n 'Hangul Compatibility Jamo': (char) => char >= 0x3130 && char <= 0x318F,\n 'Kanbun': (char) => char >= 0x3190 && char <= 0x319F,\n 'Bopomofo Extended': (char) => char >= 0x31A0 && char <= 0x31BF,\n 'CJK Strokes': (char) => char >= 0x31C0 && char <= 0x31EF,\n 'Katakana Phonetic Extensions': (char) => char >= 0x31F0 && char <= 0x31FF,\n 'Enclosed CJK Letters and Months': (char) => char >= 0x3200 && char <= 0x32FF,\n 'CJK Compatibility': (char) => char >= 0x3300 && char <= 0x33FF,\n 'CJK Unified Ideographs Extension A': (char) => char >= 0x3400 && char <= 0x4DBF,\n 'Yijing Hexagram Symbols': (char) => char >= 0x4DC0 && char <= 0x4DFF,\n 'CJK Unified Ideographs': (char) => char >= 0x4E00 && char <= 0x9FFF,\n 'Yi Syllables': (char) => char >= 0xA000 && char <= 0xA48F,\n 'Yi Radicals': (char) => char >= 0xA490 && char <= 0xA4CF,\n // 'Lisu': (char) => char >= 0xA4D0 && char <= 0xA4FF,\n // 'Vai': (char) => char >= 0xA500 && char <= 0xA63F,\n // 'Cyrillic Extended-B': (char) => char >= 0xA640 && char <= 0xA69F,\n // 'Bamum': (char) => char >= 0xA6A0 && char <= 0xA6FF,\n // 'Modifier Tone Letters': (char) => char >= 0xA700 && char <= 0xA71F,\n // 'Latin Extended-D': (char) => char >= 0xA720 && char <= 0xA7FF,\n // 'Syloti Nagri': (char) => char >= 0xA800 && char <= 0xA82F,\n // 'Common Indic Number Forms': (char) => char >= 0xA830 && char <= 0xA83F,\n // 'Phags-pa': (char) => char >= 0xA840 && char <= 0xA87F,\n // 'Saurashtra': (char) => char >= 0xA880 && char <= 0xA8DF,\n // 'Devanagari Extended': (char) => char >= 0xA8E0 && char <= 0xA8FF,\n // 'Kayah Li': (char) => char >= 0xA900 && char <= 0xA92F,\n // 'Rejang': (char) => char >= 0xA930 && char <= 0xA95F,\n 'Hangul Jamo Extended-A': (char) => char >= 0xA960 && char <= 0xA97F,\n // 'Javanese': (char) => char >= 0xA980 && char <= 0xA9DF,\n // 'Myanmar Extended-B': (char) => char >= 0xA9E0 && char <= 0xA9FF,\n // 'Cham': (char) => char >= 0xAA00 && char <= 0xAA5F,\n // 'Myanmar Extended-A': (char) => char >= 0xAA60 && char <= 0xAA7F,\n // 'Tai Viet': (char) => char >= 0xAA80 && char <= 0xAADF,\n // 'Meetei Mayek Extensions': (char) => char >= 0xAAE0 && char <= 0xAAFF,\n // 'Ethiopic Extended-A': (char) => char >= 0xAB00 && char <= 0xAB2F,\n // 'Latin Extended-E': (char) => char >= 0xAB30 && char <= 0xAB6F,\n // 'Cherokee Supplement': (char) => char >= 0xAB70 && char <= 0xABBF,\n // 'Meetei Mayek': (char) => char >= 0xABC0 && char <= 0xABFF,\n 'Hangul Syllables': (char) => char >= 0xAC00 && char <= 0xD7AF,\n 'Hangul Jamo Extended-B': (char) => char >= 0xD7B0 && char <= 0xD7FF,\n // 'High Surrogates': (char) => char >= 0xD800 && char <= 0xDB7F,\n // 'High Private Use Surrogates': (char) => char >= 0xDB80 && char <= 0xDBFF,\n // 'Low Surrogates': (char) => char >= 0xDC00 && char <= 0xDFFF,\n 'Private Use Area': (char) => char >= 0xE000 && char <= 0xF8FF,\n 'CJK Compatibility Ideographs': (char) => char >= 0xF900 && char <= 0xFAFF,\n // 'Alphabetic Presentation Forms': (char) => char >= 0xFB00 && char <= 0xFB4F,\n 'Arabic Presentation Forms-A': (char) => char >= 0xFB50 && char <= 0xFDFF,\n // 'Variation Selectors': (char) => char >= 0xFE00 && char <= 0xFE0F,\n 'Vertical Forms': (char) => char >= 0xFE10 && char <= 0xFE1F,\n // 'Combining Half Marks': (char) => char >= 0xFE20 && char <= 0xFE2F,\n 'CJK Compatibility Forms': (char) => char >= 0xFE30 && char <= 0xFE4F,\n 'Small Form Variants': (char) => char >= 0xFE50 && char <= 0xFE6F,\n 'Arabic Presentation Forms-B': (char) => char >= 0xFE70 && char <= 0xFEFF,\n 'Halfwidth and Fullwidth Forms': (char) => char >= 0xFF00 && char <= 0xFFEF\n // 'Specials': (char) => char >= 0xFFF0 && char <= 0xFFFF,\n // 'Linear B Syllabary': (char) => char >= 0x10000 && char <= 0x1007F,\n // 'Linear B Ideograms': (char) => char >= 0x10080 && char <= 0x100FF,\n // 'Aegean Numbers': (char) => char >= 0x10100 && char <= 0x1013F,\n // 'Ancient Greek Numbers': (char) => char >= 0x10140 && char <= 0x1018F,\n // 'Ancient Symbols': (char) => char >= 0x10190 && char <= 0x101CF,\n // 'Phaistos Disc': (char) => char >= 0x101D0 && char <= 0x101FF,\n // 'Lycian': (char) => char >= 0x10280 && char <= 0x1029F,\n // 'Carian': (char) => char >= 0x102A0 && char <= 0x102DF,\n // 'Coptic Epact Numbers': (char) => char >= 0x102E0 && char <= 0x102FF,\n // 'Old Italic': (char) => char >= 0x10300 && char <= 0x1032F,\n // 'Gothic': (char) => char >= 0x10330 && char <= 0x1034F,\n // 'Old Permic': (char) => char >= 0x10350 && char <= 0x1037F,\n // 'Ugaritic': (char) => char >= 0x10380 && char <= 0x1039F,\n // 'Old Persian': (char) => char >= 0x103A0 && char <= 0x103DF,\n // 'Deseret': (char) => char >= 0x10400 && char <= 0x1044F,\n // 'Shavian': (char) => char >= 0x10450 && char <= 0x1047F,\n // 'Osmanya': (char) => char >= 0x10480 && char <= 0x104AF,\n // 'Osage': (char) => char >= 0x104B0 && char <= 0x104FF,\n // 'Elbasan': (char) => char >= 0x10500 && char <= 0x1052F,\n // 'Caucasian Albanian': (char) => char >= 0x10530 && char <= 0x1056F,\n // 'Linear A': (char) => char >= 0x10600 && char <= 0x1077F,\n // 'Cypriot Syllabary': (char) => char >= 0x10800 && char <= 0x1083F,\n // 'Imperial Aramaic': (char) => char >= 0x10840 && char <= 0x1085F,\n // 'Palmyrene': (char) => char >= 0x10860 && char <= 0x1087F,\n // 'Nabataean': (char) => char >= 0x10880 && char <= 0x108AF,\n // 'Hatran': (char) => char >= 0x108E0 && char <= 0x108FF,\n // 'Phoenician': (char) => char >= 0x10900 && char <= 0x1091F,\n // 'Lydian': (char) => char >= 0x10920 && char <= 0x1093F,\n // 'Meroitic Hieroglyphs': (char) => char >= 0x10980 && char <= 0x1099F,\n // 'Meroitic Cursive': (char) => char >= 0x109A0 && char <= 0x109FF,\n // 'Kharoshthi': (char) => char >= 0x10A00 && char <= 0x10A5F,\n // 'Old South Arabian': (char) => char >= 0x10A60 && char <= 0x10A7F,\n // 'Old North Arabian': (char) => char >= 0x10A80 && char <= 0x10A9F,\n // 'Manichaean': (char) => char >= 0x10AC0 && char <= 0x10AFF,\n // 'Avestan': (char) => char >= 0x10B00 && char <= 0x10B3F,\n // 'Inscriptional Parthian': (char) => char >= 0x10B40 && char <= 0x10B5F,\n // 'Inscriptional Pahlavi': (char) => char >= 0x10B60 && char <= 0x10B7F,\n // 'Psalter Pahlavi': (char) => char >= 0x10B80 && char <= 0x10BAF,\n // 'Old Turkic': (char) => char >= 0x10C00 && char <= 0x10C4F,\n // 'Old Hungarian': (char) => char >= 0x10C80 && char <= 0x10CFF,\n // 'Hanifi Rohingya': (char) => char >= 0x10D00 && char <= 0x10D3F,\n // 'Rumi Numeral Symbols': (char) => char >= 0x10E60 && char <= 0x10E7F,\n // 'Old Sogdian': (char) => char >= 0x10F00 && char <= 0x10F2F,\n // 'Sogdian': (char) => char >= 0x10F30 && char <= 0x10F6F,\n // 'Elymaic': (char) => char >= 0x10FE0 && char <= 0x10FFF,\n // 'Brahmi': (char) => char >= 0x11000 && char <= 0x1107F,\n // 'Kaithi': (char) => char >= 0x11080 && char <= 0x110CF,\n // 'Sora Sompeng': (char) => char >= 0x110D0 && char <= 0x110FF,\n // 'Chakma': (char) => char >= 0x11100 && char <= 0x1114F,\n // 'Mahajani': (char) => char >= 0x11150 && char <= 0x1117F,\n // 'Sharada': (char) => char >= 0x11180 && char <= 0x111DF,\n // 'Sinhala Archaic Numbers': (char) => char >= 0x111E0 && char <= 0x111FF,\n // 'Khojki': (char) => char >= 0x11200 && char <= 0x1124F,\n // 'Multani': (char) => char >= 0x11280 && char <= 0x112AF,\n // 'Khudawadi': (char) => char >= 0x112B0 && char <= 0x112FF,\n // 'Grantha': (char) => char >= 0x11300 && char <= 0x1137F,\n // 'Newa': (char) => char >= 0x11400 && char <= 0x1147F,\n // 'Tirhuta': (char) => char >= 0x11480 && char <= 0x114DF,\n // 'Siddham': (char) => char >= 0x11580 && char <= 0x115FF,\n // 'Modi': (char) => char >= 0x11600 && char <= 0x1165F,\n // 'Mongolian Supplement': (char) => char >= 0x11660 && char <= 0x1167F,\n // 'Takri': (char) => char >= 0x11680 && char <= 0x116CF,\n // 'Ahom': (char) => char >= 0x11700 && char <= 0x1173F,\n // 'Dogra': (char) => char >= 0x11800 && char <= 0x1184F,\n // 'Warang Citi': (char) => char >= 0x118A0 && char <= 0x118FF,\n // 'Nandinagari': (char) => char >= 0x119A0 && char <= 0x119FF,\n // 'Zanabazar Square': (char) => char >= 0x11A00 && char <= 0x11A4F,\n // 'Soyombo': (char) => char >= 0x11A50 && char <= 0x11AAF,\n // 'Pau Cin Hau': (char) => char >= 0x11AC0 && char <= 0x11AFF,\n // 'Bhaiksuki': (char) => char >= 0x11C00 && char <= 0x11C6F,\n // 'Marchen': (char) => char >= 0x11C70 && char <= 0x11CBF,\n // 'Masaram Gondi': (char) => char >= 0x11D00 && char <= 0x11D5F,\n // 'Gunjala Gondi': (char) => char >= 0x11D60 && char <= 0x11DAF,\n // 'Makasar': (char) => char >= 0x11EE0 && char <= 0x11EFF,\n // 'Tamil Supplement': (char) => char >= 0x11FC0 && char <= 0x11FFF,\n // 'Cuneiform': (char) => char >= 0x12000 && char <= 0x123FF,\n // 'Cuneiform Numbers and Punctuation': (char) => char >= 0x12400 && char <= 0x1247F,\n // 'Early Dynastic Cuneiform': (char) => char >= 0x12480 && char <= 0x1254F,\n // 'Egyptian Hieroglyphs': (char) => char >= 0x13000 && char <= 0x1342F,\n // 'Egyptian Hieroglyph Format Controls': (char) => char >= 0x13430 && char <= 0x1343F,\n // 'Anatolian Hieroglyphs': (char) => char >= 0x14400 && char <= 0x1467F,\n // 'Bamum Supplement': (char) => char >= 0x16800 && char <= 0x16A3F,\n // 'Mro': (char) => char >= 0x16A40 && char <= 0x16A6F,\n // 'Bassa Vah': (char) => char >= 0x16AD0 && char <= 0x16AFF,\n // 'Pahawh Hmong': (char) => char >= 0x16B00 && char <= 0x16B8F,\n // 'Medefaidrin': (char) => char >= 0x16E40 && char <= 0x16E9F,\n // 'Miao': (char) => char >= 0x16F00 && char <= 0x16F9F,\n // 'Ideographic Symbols and Punctuation': (char) => char >= 0x16FE0 && char <= 0x16FFF,\n // 'Tangut': (char) => char >= 0x17000 && char <= 0x187FF,\n // 'Tangut Components': (char) => char >= 0x18800 && char <= 0x18AFF,\n // 'Kana Supplement': (char) => char >= 0x1B000 && char <= 0x1B0FF,\n // 'Kana Extended-A': (char) => char >= 0x1B100 && char <= 0x1B12F,\n // 'Small Kana Extension': (char) => char >= 0x1B130 && char <= 0x1B16F,\n // 'Nushu': (char) => char >= 0x1B170 && char <= 0x1B2FF,\n // 'Duployan': (char) => char >= 0x1BC00 && char <= 0x1BC9F,\n // 'Shorthand Format Controls': (char) => char >= 0x1BCA0 && char <= 0x1BCAF,\n // 'Byzantine Musical Symbols': (char) => char >= 0x1D000 && char <= 0x1D0FF,\n // 'Musical Symbols': (char) => char >= 0x1D100 && char <= 0x1D1FF,\n // 'Ancient Greek Musical Notation': (char) => char >= 0x1D200 && char <= 0x1D24F,\n // 'Mayan Numerals': (char) => char >= 0x1D2E0 && char <= 0x1D2FF,\n // 'Tai Xuan Jing Symbols': (char) => char >= 0x1D300 && char <= 0x1D35F,\n // 'Counting Rod Numerals': (char) => char >= 0x1D360 && char <= 0x1D37F,\n // 'Mathematical Alphanumeric Symbols': (char) => char >= 0x1D400 && char <= 0x1D7FF,\n // 'Sutton SignWriting': (char) => char >= 0x1D800 && char <= 0x1DAAF,\n // 'Glagolitic Supplement': (char) => char >= 0x1E000 && char <= 0x1E02F,\n // 'Nyiakeng Puachue Hmong': (char) => char >= 0x1E100 && char <= 0x1E14F,\n // 'Wancho': (char) => char >= 0x1E2C0 && char <= 0x1E2FF,\n // 'Mende Kikakui': (char) => char >= 0x1E800 && char <= 0x1E8DF,\n // 'Adlam': (char) => char >= 0x1E900 && char <= 0x1E95F,\n // 'Indic Siyaq Numbers': (char) => char >= 0x1EC70 && char <= 0x1ECBF,\n // 'Ottoman Siyaq Numbers': (char) => char >= 0x1ED00 && char <= 0x1ED4F,\n // 'Arabic Mathematical Alphabetic Symbols': (char) => char >= 0x1EE00 && char <= 0x1EEFF,\n // 'Mahjong Tiles': (char) => char >= 0x1F000 && char <= 0x1F02F,\n // 'Domino Tiles': (char) => char >= 0x1F030 && char <= 0x1F09F,\n // 'Playing Cards': (char) => char >= 0x1F0A0 && char <= 0x1F0FF,\n // 'Enclosed Alphanumeric Supplement': (char) => char >= 0x1F100 && char <= 0x1F1FF,\n // 'Enclosed Ideographic Supplement': (char) => char >= 0x1F200 && char <= 0x1F2FF,\n // 'Miscellaneous Symbols and Pictographs': (char) => char >= 0x1F300 && char <= 0x1F5FF,\n // 'Emoticons': (char) => char >= 0x1F600 && char <= 0x1F64F,\n // 'Ornamental Dingbats': (char) => char >= 0x1F650 && char <= 0x1F67F,\n // 'Transport and Map Symbols': (char) => char >= 0x1F680 && char <= 0x1F6FF,\n // 'Alchemical Symbols': (char) => char >= 0x1F700 && char <= 0x1F77F,\n // 'Geometric Shapes Extended': (char) => char >= 0x1F780 && char <= 0x1F7FF,\n // 'Supplemental Arrows-C': (char) => char >= 0x1F800 && char <= 0x1F8FF,\n // 'Supplemental Symbols and Pictographs': (char) => char >= 0x1F900 && char <= 0x1F9FF,\n // 'Chess Symbols': (char) => char >= 0x1FA00 && char <= 0x1FA6F,\n // 'Symbols and Pictographs Extended-A': (char) => char >= 0x1FA70 && char <= 0x1FAFF,\n // 'CJK Unified Ideographs Extension B': (char) => char >= 0x20000 && char <= 0x2A6DF,\n // 'CJK Unified Ideographs Extension C': (char) => char >= 0x2A700 && char <= 0x2B73F,\n // 'CJK Unified Ideographs Extension D': (char) => char >= 0x2B740 && char <= 0x2B81F,\n // 'CJK Unified Ideographs Extension E': (char) => char >= 0x2B820 && char <= 0x2CEAF,\n // 'CJK Unified Ideographs Extension F': (char) => char >= 0x2CEB0 && char <= 0x2EBEF,\n // 'CJK Compatibility Ideographs Supplement': (char) => char >= 0x2F800 && char <= 0x2FA1F,\n // 'Tags': (char) => char >= 0xE0000 && char <= 0xE007F,\n // 'Variation Selectors Supplement': (char) => char >= 0xE0100 && char <= 0xE01EF,\n // 'Supplementary Private Use Area-A': (char) => char >= 0xF0000 && char <= 0xFFFFF,\n // 'Supplementary Private Use Area-B': (char) => char >= 0x100000 && char <= 0x10FFFF,\n};\n","/* eslint-disable new-cap */\n\nimport {unicodeBlockLookup as isChar} from './is_char_in_unicode_block';\n\nexport function allowsIdeographicBreaking(chars: string) {\n for (const char of chars) {\n if (!charAllowsIdeographicBreaking(char.charCodeAt(0))) return false;\n }\n return true;\n}\n\nexport function allowsVerticalWritingMode(chars: string) {\n for (const char of chars) {\n if (charHasUprightVerticalOrientation(char.charCodeAt(0))) return true;\n }\n return false;\n}\n\nexport function allowsLetterSpacing(chars: string) {\n for (const char of chars) {\n if (!charAllowsLetterSpacing(char.charCodeAt(0))) return false;\n }\n return true;\n}\n\nexport function charAllowsLetterSpacing(char: number) {\n if (isChar['Arabic'](char)) return false;\n if (isChar['Arabic Supplement'](char)) return false;\n if (isChar['Arabic Extended-A'](char)) return false;\n if (isChar['Arabic Presentation Forms-A'](char)) return false;\n if (isChar['Arabic Presentation Forms-B'](char)) return false;\n\n return true;\n}\n\nexport function charAllowsIdeographicBreaking(char: number) {\n // Return early for characters outside all ideographic ranges.\n if (char < 0x2E80) return false;\n\n if (isChar['Bopomofo Extended'](char)) return true;\n if (isChar['Bopomofo'](char)) return true;\n if (isChar['CJK Compatibility Forms'](char)) return true;\n if (isChar['CJK Compatibility Ideographs'](char)) return true;\n if (isChar['CJK Compatibility'](char)) return true;\n if (isChar['CJK Radicals Supplement'](char)) return true;\n if (isChar['CJK Strokes'](char)) return true;\n if (isChar['CJK Symbols and Punctuation'](char)) return true;\n if (isChar['CJK Unified Ideographs Extension A'](char)) return true;\n if (isChar['CJK Unified Ideographs'](char)) return true;\n if (isChar['Enclosed CJK Letters and Months'](char)) return true;\n if (isChar['Halfwidth and Fullwidth Forms'](char)) return true;\n if (isChar['Hiragana'](char)) return true;\n if (isChar['Ideographic Description Characters'](char)) return true;\n if (isChar['Kangxi Radicals'](char)) return true;\n if (isChar['Katakana Phonetic Extensions'](char)) return true;\n if (isChar['Katakana'](char)) return true;\n if (isChar['Vertical Forms'](char)) return true;\n if (isChar['Yi Radicals'](char)) return true;\n if (isChar['Yi Syllables'](char)) return true;\n\n return false;\n}\n\n// The following logic comes from\n// .\n// Keep it synchronized with\n// .\n// The data file denotes with “U” or “Tu” any codepoint that may be drawn\n// upright in vertical text but does not distinguish between upright and\n// “neutral” characters.\n\n// Blocks in the Unicode supplementary planes are excluded from this module due\n// to .\n\n/**\n * Returns true if the given Unicode codepoint identifies a character with\n * upright orientation.\n *\n * A character has upright orientation if it is drawn upright (unrotated)\n * whether the line is oriented horizontally or vertically, even if both\n * adjacent characters can be rotated. For example, a Chinese character is\n * always drawn upright. An uprightly oriented character causes an adjacent\n * “neutral” character to be drawn upright as well.\n */\nexport function charHasUprightVerticalOrientation(char: number) {\n if (char === 0x02EA /* modifier letter yin departing tone mark */ ||\n char === 0x02EB /* modifier letter yang departing tone mark */) {\n return true;\n }\n\n // Return early for characters outside all ranges whose characters remain\n // upright in vertical writing mode.\n if (char < 0x1100) return false;\n\n if (isChar['Bopomofo Extended'](char)) return true;\n if (isChar['Bopomofo'](char)) return true;\n if (isChar['CJK Compatibility Forms'](char)) {\n if (!((char >= 0xFE49 /* dashed overline */ && char <= 0xFE4F) /* wavy low line */)) {\n return true;\n }\n }\n if (isChar['CJK Compatibility Ideographs'](char)) return true;\n if (isChar['CJK Compatibility'](char)) return true;\n if (isChar['CJK Radicals Supplement'](char)) return true;\n if (isChar['CJK Strokes'](char)) return true;\n if (isChar['CJK Symbols and Punctuation'](char)) {\n if (!((char >= 0x3008 /* left angle bracket */ && char <= 0x3011) /* right black lenticular bracket */) &&\n !((char >= 0x3014 /* left tortoise shell bracket */ && char <= 0x301F) /* low double prime quotation mark */) &&\n char !== 0x3030 /* wavy dash */) {\n return true;\n }\n }\n if (isChar['CJK Unified Ideographs Extension A'](char)) return true;\n if (isChar['CJK Unified Ideographs'](char)) return true;\n if (isChar['Enclosed CJK Letters and Months'](char)) return true;\n if (isChar['Hangul Compatibility Jamo'](char)) return true;\n if (isChar['Hangul Jamo Extended-A'](char)) return true;\n if (isChar['Hangul Jamo Extended-B'](char)) return true;\n if (isChar['Hangul Jamo'](char)) return true;\n if (isChar['Hangul Syllables'](char)) return true;\n if (isChar['Hiragana'](char)) return true;\n if (isChar['Ideographic Description Characters'](char)) return true;\n if (isChar['Kanbun'](char)) return true;\n if (isChar['Kangxi Radicals'](char)) return true;\n if (isChar['Katakana Phonetic Extensions'](char)) return true;\n if (isChar['Katakana'](char)) {\n if (char !== 0x30FC /* katakana-hiragana prolonged sound mark */) {\n return true;\n }\n }\n if (isChar['Halfwidth and Fullwidth Forms'](char)) {\n if (char !== 0xFF08 /* fullwidth left parenthesis */ &&\n char !== 0xFF09 /* fullwidth right parenthesis */ &&\n char !== 0xFF0D /* fullwidth hyphen-minus */ &&\n !((char >= 0xFF1A /* fullwidth colon */ && char <= 0xFF1E) /* fullwidth greater-than sign */) &&\n char !== 0xFF3B /* fullwidth left square bracket */ &&\n char !== 0xFF3D /* fullwidth right square bracket */ &&\n char !== 0xFF3F /* fullwidth low line */ &&\n !(char >= 0xFF5B /* fullwidth left curly bracket */ && char <= 0xFFDF) &&\n char !== 0xFFE3 /* fullwidth macron */ &&\n !(char >= 0xFFE8 /* halfwidth forms light vertical */ && char <= 0xFFEF)) {\n return true;\n }\n }\n if (isChar['Small Form Variants'](char)) {\n if (!((char >= 0xFE58 /* small em dash */ && char <= 0xFE5E) /* small right tortoise shell bracket */) &&\n !((char >= 0xFE63 /* small hyphen-minus */ && char <= 0xFE66) /* small equals sign */)) {\n return true;\n }\n }\n if (isChar['Unified Canadian Aboriginal Syllabics'](char)) return true;\n if (isChar['Unified Canadian Aboriginal Syllabics Extended'](char)) return true;\n if (isChar['Vertical Forms'](char)) return true;\n if (isChar['Yijing Hexagram Symbols'](char)) return true;\n if (isChar['Yi Syllables'](char)) return true;\n if (isChar['Yi Radicals'](char)) return true;\n\n return false;\n}\n\n/**\n * Returns true if the given Unicode codepoint identifies a character with\n * neutral orientation.\n *\n * A character has neutral orientation if it may be drawn rotated or unrotated\n * when the line is oriented vertically, depending on the orientation of the\n * adjacent characters. For example, along a verticlly oriented line, the vulgar\n * fraction ½ is drawn upright among Chinese characters but rotated among Latin\n * letters. A neutrally oriented character does not influence whether an\n * adjacent character is drawn upright or rotated.\n */\nexport function charHasNeutralVerticalOrientation(char: number) {\n if (isChar['Latin-1 Supplement'](char)) {\n if (char === 0x00A7 /* section sign */ ||\n char === 0x00A9 /* copyright sign */ ||\n char === 0x00AE /* registered sign */ ||\n char === 0x00B1 /* plus-minus sign */ ||\n char === 0x00BC /* vulgar fraction one quarter */ ||\n char === 0x00BD /* vulgar fraction one half */ ||\n char === 0x00BE /* vulgar fraction three quarters */ ||\n char === 0x00D7 /* multiplication sign */ ||\n char === 0x00F7 /* division sign */) {\n return true;\n }\n }\n if (isChar['General Punctuation'](char)) {\n if (char === 0x2016 /* double vertical line */ ||\n char === 0x2020 /* dagger */ ||\n char === 0x2021 /* double dagger */ ||\n char === 0x2030 /* per mille sign */ ||\n char === 0x2031 /* per ten thousand sign */ ||\n char === 0x203B /* reference mark */ ||\n char === 0x203C /* double exclamation mark */ ||\n char === 0x2042 /* asterism */ ||\n char === 0x2047 /* double question mark */ ||\n char === 0x2048 /* question exclamation mark */ ||\n char === 0x2049 /* exclamation question mark */ ||\n char === 0x2051 /* two asterisks aligned vertically */) {\n return true;\n }\n }\n if (isChar['Letterlike Symbols'](char)) return true;\n if (isChar['Number Forms'](char)) return true;\n if (isChar['Miscellaneous Technical'](char)) {\n if ((char >= 0x2300 /* diameter sign */ && char <= 0x2307 /* wavy line */) ||\n (char >= 0x230C /* bottom right crop */ && char <= 0x231F /* bottom right corner */) ||\n (char >= 0x2324 /* up arrowhead between two horizontal bars */ && char <= 0x2328 /* keyboard */) ||\n char === 0x232B /* erase to the left */ ||\n (char >= 0x237D /* shouldered open box */ && char <= 0x239A /* clear screen symbol */) ||\n (char >= 0x23BE /* dentistry symbol light vertical and top right */ && char <= 0x23CD /* square foot */) ||\n char === 0x23CF /* eject symbol */ ||\n (char >= 0x23D1 /* metrical breve */ && char <= 0x23DB /* fuse */) ||\n (char >= 0x23E2 /* white trapezium */ && char <= 0x23FF)) {\n return true;\n }\n }\n if (isChar['Control Pictures'](char) && char !== 0x2423 /* open box */) return true;\n if (isChar['Optical Character Recognition'](char)) return true;\n if (isChar['Enclosed Alphanumerics'](char)) return true;\n if (isChar['Geometric Shapes'](char)) return true;\n if (isChar['Miscellaneous Symbols'](char)) {\n if (!((char >= 0x261A /* black left pointing index */ && char <= 0x261F) /* white down pointing index */)) {\n return true;\n }\n }\n if (isChar['Miscellaneous Symbols and Arrows'](char)) {\n if ((char >= 0x2B12 /* square with top half black */ && char <= 0x2B2F /* white vertical ellipse */) ||\n (char >= 0x2B50 /* white medium star */ && char <= 0x2B59 /* heavy circled saltire */) ||\n (char >= 0x2BB8 /* upwards white arrow from bar with horizontal bar */ && char <= 0x2BEB)) {\n return true;\n }\n }\n if (isChar['CJK Symbols and Punctuation'](char)) return true;\n if (isChar['Katakana'](char)) return true;\n if (isChar['Private Use Area'](char)) return true;\n if (isChar['CJK Compatibility Forms'](char)) return true;\n if (isChar['Small Form Variants'](char)) return true;\n if (isChar['Halfwidth and Fullwidth Forms'](char)) return true;\n\n if (char === 0x221E /* infinity */ ||\n char === 0x2234 /* therefore */ ||\n char === 0x2235 /* because */ ||\n (char >= 0x2700 /* black safety scissors */ && char <= 0x2767 /* rotated floral heart bullet */) ||\n (char >= 0x2776 /* dingbat negative circled digit one */ && char <= 0x2793 /* dingbat negative circled sans-serif number ten */) ||\n char === 0xFFFC /* object replacement character */ ||\n char === 0xFFFD /* replacement character */) {\n return true;\n }\n\n return false;\n}\n\n/**\n * Returns true if the given Unicode codepoint identifies a character with\n * rotated orientation.\n *\n * A character has rotated orientation if it is drawn rotated when the line is\n * oriented vertically, even if both adjacent characters are upright. For\n * example, a Latin letter is drawn rotated along a vertical line. A rotated\n * character causes an adjacent “neutral” character to be drawn rotated as well.\n */\nexport function charHasRotatedVerticalOrientation(char: number) {\n return !(charHasUprightVerticalOrientation(char) ||\n charHasNeutralVerticalOrientation(char));\n}\n\nexport function charInComplexShapingScript(char: number) {\n return isChar['Arabic'](char) ||\n isChar['Arabic Supplement'](char) ||\n isChar['Arabic Extended-A'](char) ||\n isChar['Arabic Presentation Forms-A'](char) ||\n isChar['Arabic Presentation Forms-B'](char);\n}\n\nexport function charInRTLScript(char: number) {\n // Main blocks for Hebrew, Arabic, Thaana and other RTL scripts\n return (char >= 0x0590 && char <= 0x08FF) ||\n isChar['Arabic Presentation Forms-A'](char) ||\n isChar['Arabic Presentation Forms-B'](char);\n}\n\nexport function charInSupportedScript(char: number, canRenderRTL: boolean) {\n // This is a rough heuristic: whether we \"can render\" a script\n // actually depends on the properties of the font being used\n // and whether differences from the ideal rendering are considered\n // semantically significant.\n\n // Even in Latin script, we \"can't render\" combinations such as the fi\n // ligature, but we don't consider that semantically significant.\n if (!canRenderRTL && charInRTLScript(char)) {\n return false;\n }\n if ((char >= 0x0900 && char <= 0x0DFF) ||\n // Main blocks for Indic scripts and Sinhala\n (char >= 0x0F00 && char <= 0x109F) ||\n // Main blocks for Tibetan and Myanmar\n isChar['Khmer'](char)) {\n // These blocks cover common scripts that require\n // complex text shaping, based on unicode script metadata:\n // http://www.unicode.org/repos/cldr/trunk/common/properties/scriptMetadata.txt\n // where \"Web Rank <= 32\" \"Shaping Required = YES\"\n return false;\n }\n return true;\n}\n\nexport function stringContainsRTLText(chars: string): boolean {\n for (const char of chars) {\n if (charInRTLScript(char.charCodeAt(0))) {\n return true;\n }\n }\n return false;\n}\n\nexport function isStringInSupportedScript(chars: string, canRenderRTL: boolean) {\n for (const char of chars) {\n if (!charInSupportedScript(char.charCodeAt(0), canRenderRTL)) {\n return false;\n }\n }\n return true;\n}\n","import {getArrayBuffer} from '../util/ajax';\nimport {browser} from '../util/browser';\nimport {Event, Evented} from '../util/evented';\nimport {isWorker} from '../util/util';\n\nconst status = {\n unavailable: 'unavailable', // Not loaded\n deferred: 'deferred', // The plugin URL has been specified, but loading has been deferred\n loading: 'loading', // request in-flight\n loaded: 'loaded',\n error: 'error'\n};\n\nexport type PluginState = {\n pluginStatus: typeof status[keyof typeof status];\n pluginURL: string;\n};\n\n/**\n * An error callback\n */\ntype ErrorCallback = (error?: Error | null) => void;\ntype PluginStateSyncCallback = (state: PluginState) => void;\nlet _completionCallback = null;\n\n//Variables defining the current state of the plugin\nlet pluginStatus = status.unavailable;\nlet pluginURL = null;\n\nexport const triggerPluginCompletionEvent = function(error: Error | string) {\n // NetworkError's are not correctly reflected by the plugin status which prevents reloading plugin\n if (error && typeof error === 'string' && error.indexOf('NetworkError') > -1) {\n pluginStatus = status.error;\n }\n\n if (_completionCallback) {\n _completionCallback(error);\n }\n};\n\nfunction sendPluginStateToWorker() {\n evented.fire(new Event('pluginStateChange', {pluginStatus, pluginURL}));\n}\n\nexport const evented = new Evented();\n\nexport const getRTLTextPluginStatus = function () {\n return pluginStatus;\n};\n\nexport const registerForPluginStateChange = function(callback: PluginStateSyncCallback) {\n // Do an initial sync of the state\n callback({pluginStatus, pluginURL});\n // Listen for all future state changes\n evented.on('pluginStateChange', callback);\n return callback;\n};\n\nexport const clearRTLTextPlugin = function() {\n pluginStatus = status.unavailable;\n pluginURL = null;\n _completionCallback = null;\n};\n\nexport const setRTLTextPlugin = function(url: string, callback: ErrorCallback, deferred: boolean = false) {\n if (pluginStatus === status.deferred || pluginStatus === status.loading || pluginStatus === status.loaded) {\n throw new Error('setRTLTextPlugin cannot be called multiple times.');\n }\n pluginURL = browser.resolveURL(url);\n pluginStatus = status.deferred;\n _completionCallback = callback;\n sendPluginStateToWorker();\n\n //Start downloading the plugin immediately if not intending to lazy-load\n if (!deferred) {\n downloadRTLTextPlugin();\n }\n};\n\nexport const downloadRTLTextPlugin = function() {\n if (pluginStatus !== status.deferred || !pluginURL) {\n throw new Error('rtl-text-plugin cannot be downloaded unless a pluginURL is specified');\n }\n pluginStatus = status.loading;\n sendPluginStateToWorker();\n if (pluginURL) {\n getArrayBuffer({url: pluginURL}, (error) => {\n if (error) {\n triggerPluginCompletionEvent(error);\n } else {\n pluginStatus = status.loaded;\n sendPluginStateToWorker();\n }\n });\n }\n};\n\nexport const plugin: {\n applyArabicShaping: Function;\n processBidirectionalText: ((b: string, a: Array) => Array);\n processStyledBidirectionalText: ((c: string, b: Array, a: Array) => Array<[string, Array]>);\n isLoaded: () => boolean;\n isLoading: () => boolean;\n setState: (state: PluginState) => void;\n isParsed: () => boolean;\n getPluginURL: () => string;\n} = {\n applyArabicShaping: null,\n processBidirectionalText: null,\n processStyledBidirectionalText: null,\n isLoaded() {\n return pluginStatus === status.loaded || // Main Thread: loaded if the completion callback returned successfully\n plugin.applyArabicShaping != null; // Web-worker: loaded if the plugin functions have been compiled\n },\n isLoading() { // Main Thread Only: query the loading status, this function does not return the correct value in the worker context.\n return pluginStatus === status.loading;\n },\n setState(state: PluginState) { // Worker thread only: this tells the worker threads that the plugin is available on the Main thread\n if (!isWorker()) throw new Error('Cannot set the state of the rtl-text-plugin when not in the web-worker context');\n\n pluginStatus = state.pluginStatus;\n pluginURL = state.pluginURL;\n },\n isParsed(): boolean {\n if (!isWorker()) throw new Error('rtl-text-plugin is only parsed on the worker-threads');\n\n return plugin.applyArabicShaping != null &&\n plugin.processBidirectionalText != null &&\n plugin.processStyledBidirectionalText != null;\n },\n getPluginURL(): string {\n if (!isWorker()) throw new Error('rtl-text-plugin url can only be queried from the worker threads');\n return pluginURL;\n }\n};\n\nexport const lazyLoadRTLTextPlugin = function() {\n if (!plugin.isLoading() &&\n !plugin.isLoaded() &&\n getRTLTextPluginStatus() === 'deferred'\n ) {\n downloadRTLTextPlugin();\n }\n};\n","import {ZoomHistory} from './zoom_history';\nimport {isStringInSupportedScript} from '../util/script_detection';\nimport {plugin as rtlTextPlugin} from '../source/rtl_text_plugin';\n\nimport type {TransitionSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type CrossfadeParameters = {\n fromScale: number;\n toScale: number;\n t: number;\n};\n\n/**\n * @internal\n * A parameter that can be evaluated to a value\n */\nexport class EvaluationParameters {\n zoom: number;\n now: number;\n fadeDuration: number;\n zoomHistory: ZoomHistory;\n transition: TransitionSpecification;\n\n // \"options\" may also be another EvaluationParameters to copy, see CrossFadedProperty.possiblyEvaluate\n constructor(zoom: number, options?: any) {\n this.zoom = zoom;\n\n if (options) {\n this.now = options.now;\n this.fadeDuration = options.fadeDuration;\n this.zoomHistory = options.zoomHistory;\n this.transition = options.transition;\n } else {\n this.now = 0;\n this.fadeDuration = 0;\n this.zoomHistory = new ZoomHistory();\n this.transition = {};\n }\n }\n\n isSupportedScript(str: string): boolean {\n return isStringInSupportedScript(str, rtlTextPlugin.isLoaded());\n }\n\n crossFadingFactor() {\n if (this.fadeDuration === 0) {\n return 1;\n } else {\n return Math.min((this.now - this.zoomHistory.lastIntegerZoomTime) / this.fadeDuration, 1);\n }\n }\n\n getCrossfadeParameters(): CrossfadeParameters {\n const z = this.zoom;\n const fraction = z - Math.floor(z);\n const t = this.crossFadingFactor();\n\n return z > this.zoomHistory.lastIntegerZoom ?\n {fromScale: 2, toScale: 1, t: fraction + (1 - fraction) * t} :\n {fromScale: 0.5, toScale: 1, t: 1 - (1 - t) * fraction};\n }\n}\n","import {clone, extend, easeCubicInOut} from '../util/util';\nimport {interpolates, Color, StylePropertySpecification, normalizePropertyExpression,\n Feature,\n FeatureState,\n StylePropertyExpression,\n SourceExpression,\n CompositeExpression, TransitionSpecification,\n PropertyValueSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport {register} from '../util/web_worker_transfer';\nimport {EvaluationParameters} from './evaluation_parameters';\n\nimport {CanonicalTileID} from '../source/tile_id';\n\ntype TimePoint = number;\n\n/**\n * A from-to type\n */\nexport type CrossFaded = {\n to: T;\n from: T;\n};\n\n/**\n * @internal\n * Implementations of the `Property` interface:\n *\n * * Hold metadata about a property that's independent of any specific value: stuff like the type of the value,\n * the default value, etc. This comes from the style specification JSON.\n * * Define behavior that needs to be polymorphic across different properties: \"possibly evaluating\"\n * an input value (see below), and interpolating between two possibly-evaluted values.\n *\n * The type `T` is the fully-evaluated value type (e.g. `number`, `string`, `Color`).\n * The type `R` is the intermediate \"possibly evaluated\" value type. See below.\n *\n * There are two main implementations of the interface -- one for properties that allow data-driven values,\n * and one for properties that don't. There are a few \"special case\" implementations as well: one for properties\n * which cross-fade between two values rather than interpolating, one for `heatmap-color` and `line-gradient`,\n * and one for `light-position`.\n */\nexport interface Property {\n specification: StylePropertySpecification;\n possiblyEvaluate(\n value: PropertyValue,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): R;\n interpolate(a: R, b: R, t: number): R;\n}\n\n/**\n * @internal\n * `PropertyValue` represents the value part of a property key-value unit. It's used to represent both\n * paint and layout property values, and regardless of whether or not their property supports data-driven\n * expressions.\n *\n * `PropertyValue` stores the raw input value as seen in a style or a runtime styling API call, i.e. one of the\n * following:\n *\n * * A constant value of the type appropriate for the property\n * * A function which produces a value of that type (but functions are quasi-deprecated in favor of expressions)\n * * An expression which produces a value of that type\n * * \"undefined\"/\"not present\", in which case the property is assumed to take on its default value.\n *\n * In addition to storing the original input value, `PropertyValue` also stores a normalized representation,\n * effectively treating functions as if they are expressions, and constant or default values as if they are\n * (constant) expressions.\n */\nexport class PropertyValue {\n property: Property;\n value: PropertyValueSpecification | void;\n expression: StylePropertyExpression;\n\n constructor(property: Property, value: PropertyValueSpecification | void) {\n this.property = property;\n this.value = value;\n this.expression = normalizePropertyExpression(value === undefined ? property.specification.default : value, property.specification);\n }\n\n isDataDriven(): boolean {\n return this.expression.kind === 'source' || this.expression.kind === 'composite';\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): R {\n return this.property.possiblyEvaluate(this, parameters, canonical, availableImages);\n }\n}\n\nexport type TransitionParameters = {\n now: TimePoint;\n transition: TransitionSpecification;\n};\n\n/**\n * @internal\n * Paint properties are _transitionable_: they can change in a fluid manner, interpolating or cross-fading between\n * old and new value. The duration of the transition, and the delay before it begins, is configurable.\n *\n * `TransitionablePropertyValue` is a compositional class that stores both the property value and that transition\n * configuration.\n *\n * A `TransitionablePropertyValue` can calculate the next step in the evaluation chain for paint property values:\n * `TransitioningPropertyValue`.\n */\nclass TransitionablePropertyValue {\n property: Property;\n value: PropertyValue;\n transition: TransitionSpecification | void;\n\n constructor(property: Property) {\n this.property = property;\n this.value = new PropertyValue(property, undefined);\n }\n\n transitioned(parameters: TransitionParameters, prior: TransitioningPropertyValue): TransitioningPropertyValue {\n return new TransitioningPropertyValue(this.property, this.value, prior,\n extend({}, parameters.transition, this.transition), parameters.now);\n }\n\n untransitioned(): TransitioningPropertyValue {\n return new TransitioningPropertyValue(this.property, this.value, null, {}, 0);\n }\n}\n\n/**\n * @internal\n * `Transitionable` stores a map of all (property name, `TransitionablePropertyValue`) pairs for paint properties of a\n * given layer type. It can calculate the `TransitioningPropertyValue`s for all of them at once, producing a\n * `Transitioning` instance for the same set of properties.\n */\nexport class Transitionable {\n _properties: Properties;\n _values: {[K in keyof Props]: TransitionablePropertyValue};\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = (Object.create(properties.defaultTransitionablePropertyValues) as any);\n }\n\n getValue(name: S): PropertyValueSpecification | void {\n return clone(this._values[name].value.value);\n }\n\n setValue(name: S, value: PropertyValueSpecification | void) {\n if (!Object.prototype.hasOwnProperty.call(this._values, name)) {\n this._values[name] = new TransitionablePropertyValue(this._values[name].property);\n }\n // Note that we do not _remove_ an own property in the case where a value is being reset\n // to the default: the transition might still be non-default.\n this._values[name].value = new PropertyValue(this._values[name].property, value === null ? undefined : clone(value));\n }\n\n getTransition(name: S): TransitionSpecification | void {\n return clone(this._values[name].transition);\n }\n\n setTransition(name: S, value: TransitionSpecification | void) {\n if (!Object.prototype.hasOwnProperty.call(this._values, name)) {\n this._values[name] = new TransitionablePropertyValue(this._values[name].property);\n }\n this._values[name].transition = clone(value) || undefined;\n }\n\n serialize() {\n const result: any = {};\n for (const property of Object.keys(this._values)) {\n const value = this.getValue(property as keyof Props);\n if (value !== undefined) {\n result[property] = value;\n }\n\n const transition = this.getTransition(property as keyof Props);\n if (transition !== undefined) {\n result[`${property}-transition`] = transition;\n }\n }\n return result;\n }\n\n transitioned(parameters: TransitionParameters, prior: Transitioning): Transitioning {\n const result = new Transitioning(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].transitioned(parameters, prior._values[property]);\n }\n return result;\n }\n\n untransitioned(): Transitioning {\n const result = new Transitioning(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].untransitioned();\n }\n return result;\n }\n}\n\n/**\n * @internal\n * `TransitioningPropertyValue` implements the first of two intermediate steps in the evaluation chain of a paint\n * property value. In this step, transitions between old and new values are handled: as long as the transition is in\n * progress, `TransitioningPropertyValue` maintains a reference to the prior value, and interpolates between it and\n * the new value based on the current time and the configured transition duration and delay. The product is the next\n * step in the evaluation chain: the \"possibly evaluated\" result type `R`. See below for more on this concept.\n */\nclass TransitioningPropertyValue {\n property: Property;\n value: PropertyValue;\n prior: TransitioningPropertyValue;\n begin: TimePoint;\n end: TimePoint;\n\n constructor(property: Property,\n value: PropertyValue,\n prior: TransitioningPropertyValue,\n transition: TransitionSpecification,\n now: TimePoint) {\n this.property = property;\n this.value = value;\n this.begin = now + transition.delay || 0;\n this.end = this.begin + transition.duration || 0;\n if (property.specification.transition && (transition.delay || transition.duration)) {\n this.prior = prior;\n }\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical: CanonicalTileID,\n availableImages: Array\n ): R {\n const now = parameters.now || 0;\n const finalValue = this.value.possiblyEvaluate(parameters, canonical, availableImages);\n const prior = this.prior;\n if (!prior) {\n // No prior value.\n return finalValue;\n } else if (now > this.end) {\n // Transition from prior value is now complete.\n this.prior = null;\n return finalValue;\n } else if (this.value.isDataDriven()) {\n // Transitions to data-driven properties are not supported.\n // We snap immediately to the data-driven value so that, when we perform layout,\n // we see the data-driven function and can use it to populate vertex buffers.\n this.prior = null;\n return finalValue;\n } else if (now < this.begin) {\n // Transition hasn't started yet.\n return prior.possiblyEvaluate(parameters, canonical, availableImages);\n } else {\n // Interpolate between recursively-calculated prior value and final.\n const t = (now - this.begin) / (this.end - this.begin);\n return this.property.interpolate(prior.possiblyEvaluate(parameters, canonical, availableImages), finalValue, easeCubicInOut(t));\n }\n }\n}\n\n/**\n * @internal\n * `Transitioning` stores a map of all (property name, `TransitioningPropertyValue`) pairs for paint properties of a\n * given layer type. It can calculate the possibly-evaluated values for all of them at once, producing a\n * `PossiblyEvaluated` instance for the same set of properties.\n */\nexport class Transitioning {\n _properties: Properties;\n _values: {[K in keyof Props]: PossiblyEvaluatedPropertyValue};\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = (Object.create(properties.defaultTransitioningPropertyValues) as any);\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluated {\n const result = new PossiblyEvaluated(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].possiblyEvaluate(parameters, canonical, availableImages);\n }\n return result;\n }\n\n hasTransition() {\n for (const property of Object.keys(this._values)) {\n if (this._values[property].prior) {\n return true;\n }\n }\n return false;\n }\n}\n\n// ------- Layout -------\n\n/**\n * Because layout properties are not transitionable, they have a simpler representation and evaluation chain than\n * paint properties: `PropertyValue`s are possibly evaluated, producing possibly evaluated values, which are then\n * fully evaluated.\n *\n * `Layout` stores a map of all (property name, `PropertyValue`) pairs for layout properties of a\n * given layer type. It can calculate the possibly-evaluated values for all of them at once, producing a\n * `PossiblyEvaluated` instance for the same set of properties.\n */\nexport class Layout {\n _properties: Properties;\n _values: {[K in keyof Props]: PropertyValue>};\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = (Object.create(properties.defaultPropertyValues) as any);\n }\n\n hasValue(name: S) {\n return this._values[name].value !== undefined;\n }\n\n getValue(name: S) {\n return clone(this._values[name].value);\n }\n\n setValue(name: S, value: any) {\n this._values[name] = new PropertyValue(this._values[name].property, value === null ? undefined : clone(value)) as any;\n }\n\n serialize() {\n const result: any = {};\n for (const property of Object.keys(this._values)) {\n const value = this.getValue(property as keyof Props);\n if (value !== undefined) {\n result[property] = value;\n }\n }\n return result;\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluated {\n const result = new PossiblyEvaluated(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].possiblyEvaluate(parameters, canonical, availableImages);\n }\n return result;\n }\n}\n\n// ------- PossiblyEvaluated -------\n\n/**\n * \"Possibly evaluated value\" is an intermediate stage in the evaluation chain for both paint and layout property\n * values. The purpose of this stage is to optimize away unnecessary recalculations for data-driven properties. Code\n * which uses data-driven property values must assume that the value is dependent on feature data, and request that it\n * be evaluated for each feature. But when that property value is in fact a constant or camera function, the calculation\n * will not actually depend on the feature, and we can benefit from returning the prior result of having done the\n * evaluation once, ahead of time, in an intermediate step whose inputs are just the value and \"global\" parameters\n * such as current zoom level.\n *\n * `PossiblyEvaluatedValue` represents the three possible outcomes of this step: if the input value was a constant or\n * camera expression, then the \"possibly evaluated\" result is a constant value. Otherwise, the input value was either\n * a source or composite expression, and we must defer final evaluation until supplied a feature. We separate\n * the source and composite cases because they are handled differently when generating GL attributes, buffers, and\n * uniforms.\n *\n * Note that `PossiblyEvaluatedValue` (and `PossiblyEvaluatedPropertyValue`, below) are _not_ used for properties that\n * do not allow data-driven values. For such properties, we know that the \"possibly evaluated\" result is always a constant\n * scalar value. See below.\n */\ntype PossiblyEvaluatedValue = {\n kind: 'constant';\n value: T;\n} | SourceExpression | CompositeExpression;\n\n/**\n * @internal\n * `PossiblyEvaluatedPropertyValue` is used for data-driven paint and layout property values. It holds a\n * `PossiblyEvaluatedValue` and the `GlobalProperties` that were used to generate it. You're not allowed to supply\n * a different set of `GlobalProperties` when performing the final evaluation because they would be ignored in the\n * case where the input value was a constant or camera function.\n */\nexport class PossiblyEvaluatedPropertyValue {\n property: DataDrivenProperty;\n value: PossiblyEvaluatedValue;\n parameters: EvaluationParameters;\n\n constructor(property: DataDrivenProperty, value: PossiblyEvaluatedValue, parameters: EvaluationParameters) {\n this.property = property;\n this.value = value;\n this.parameters = parameters;\n }\n\n isConstant(): boolean {\n return this.value.kind === 'constant';\n }\n\n constantOr(value: T): T {\n if (this.value.kind === 'constant') {\n return this.value.value;\n } else {\n return value;\n }\n }\n\n evaluate(\n feature: Feature,\n featureState: FeatureState,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): T {\n return this.property.evaluate(this.value, this.parameters, feature, featureState, canonical, availableImages);\n }\n}\n\n/**\n * @internal\n * `PossiblyEvaluated` stores a map of all (property name, `R`) pairs for paint or layout properties of a\n * given layer type.\n */\nexport class PossiblyEvaluated {\n _properties: Properties;\n _values: PossibleEvaluatedProps;\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = Object.create(properties.defaultPossiblyEvaluatedValues);\n }\n\n get(name: S): PossibleEvaluatedProps[S] {\n return this._values[name];\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for properties that do not permit data-driven (source or composite) expressions.\n * This restriction allows us to declare statically that the result of possibly evaluating this kind of property\n * is in fact always the scalar type `T`, and can be used without further evaluating the value on a per-feature basis.\n */\nexport class DataConstantProperty implements Property {\n specification: StylePropertySpecification;\n\n constructor(specification: StylePropertySpecification) {\n this.specification = specification;\n }\n\n possiblyEvaluate(value: PropertyValue, parameters: EvaluationParameters): T {\n if (value.isDataDriven()) throw new Error('Value should not be data driven');\n return value.expression.evaluate(parameters);\n }\n\n interpolate(a: T, b: T, t: number): T {\n const interpolationType = this.specification.type as keyof typeof interpolates;\n const interpolationFn = interpolates[interpolationType] as ((from: T, to: T, t: number) => T) | undefined;\n if (interpolationFn) {\n return interpolationFn(a, b, t);\n } else {\n return a;\n }\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for properties that permit data-driven (source or composite) expressions.\n * The result of possibly evaluating this kind of property is `PossiblyEvaluatedPropertyValue`; obtaining\n * a scalar value `T` requires further evaluation on a per-feature basis.\n */\nexport class DataDrivenProperty implements Property> {\n specification: StylePropertySpecification;\n overrides: any;\n\n constructor(specification: StylePropertySpecification, overrides?: any) {\n this.specification = specification;\n this.overrides = overrides;\n }\n\n possiblyEvaluate(\n value: PropertyValue>,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluatedPropertyValue {\n if (value.expression.kind === 'constant' || value.expression.kind === 'camera') {\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: value.expression.evaluate(parameters, null, {}, canonical, availableImages)}, parameters);\n } else {\n return new PossiblyEvaluatedPropertyValue(this, value.expression, parameters);\n }\n }\n\n interpolate(\n a: PossiblyEvaluatedPropertyValue,\n b: PossiblyEvaluatedPropertyValue,\n t: number\n ): PossiblyEvaluatedPropertyValue {\n // If either possibly-evaluated value is non-constant, give up: we aren't able to interpolate data-driven values.\n if (a.value.kind !== 'constant' || b.value.kind !== 'constant') {\n return a;\n }\n\n // Special case hack solely for fill-outline-color. The undefined value is subsequently handled in\n // FillStyleLayer#recalculate, which sets fill-outline-color to the fill-color value if the former\n // is a PossiblyEvaluatedPropertyValue containing a constant undefined value. In addition to the\n // return value here, the other source of a PossiblyEvaluatedPropertyValue containing a constant\n // undefined value is the \"default value\" for fill-outline-color held in\n // `Properties#defaultPossiblyEvaluatedValues`, which serves as the prototype of\n // `PossiblyEvaluated#_values`.\n if (a.value.value === undefined || b.value.value === undefined) {\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: undefined}, a.parameters);\n }\n\n const interpolationType = this.specification.type as keyof typeof interpolates;\n const interpolationFn = interpolates[interpolationType] as ((from: T, to: T, t: number) => T) | undefined;\n if (interpolationFn) {\n const interpolatedValue = interpolationFn(a.value.value, b.value.value, t);\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: interpolatedValue}, a.parameters);\n } else {\n return a;\n }\n }\n\n evaluate(\n value: PossiblyEvaluatedValue,\n parameters: EvaluationParameters,\n feature: Feature,\n featureState: FeatureState,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): T {\n if (value.kind === 'constant') {\n return value.value;\n } else {\n return value.evaluate(parameters, feature, featureState, canonical, availableImages);\n }\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for data driven `line-pattern` which are transitioned by cross-fading\n * rather than interpolation.\n */\n\nexport class CrossFadedDataDrivenProperty extends DataDrivenProperty> {\n\n possiblyEvaluate(\n value: PropertyValue, PossiblyEvaluatedPropertyValue>>,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluatedPropertyValue> {\n if (value.value === undefined) {\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: undefined}, parameters);\n } else if (value.expression.kind === 'constant') {\n const evaluatedValue = value.expression.evaluate(parameters, null, {}, canonical, availableImages);\n const isImageExpression = value.property.specification.type as any === 'resolvedImage';\n const constantValue = isImageExpression && typeof evaluatedValue !== 'string' ? evaluatedValue.name : evaluatedValue;\n const constant = this._calculate(constantValue, constantValue, constantValue, parameters);\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: constant}, parameters);\n } else if (value.expression.kind === 'camera') {\n const cameraVal = this._calculate(\n value.expression.evaluate({zoom: parameters.zoom - 1.0}),\n value.expression.evaluate({zoom: parameters.zoom}),\n value.expression.evaluate({zoom: parameters.zoom + 1.0}),\n parameters);\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: cameraVal}, parameters);\n } else {\n // source or composite expression\n return new PossiblyEvaluatedPropertyValue(this, value.expression, parameters);\n }\n }\n\n evaluate(\n value: PossiblyEvaluatedValue>,\n globals: EvaluationParameters,\n feature: Feature,\n featureState: FeatureState,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): CrossFaded {\n if (value.kind === 'source') {\n const constant = value.evaluate(globals, feature, featureState, canonical, availableImages);\n return this._calculate(constant, constant, constant, globals);\n } else if (value.kind === 'composite') {\n return this._calculate(\n value.evaluate({zoom: Math.floor(globals.zoom) - 1.0}, feature, featureState),\n value.evaluate({zoom: Math.floor(globals.zoom)}, feature, featureState),\n value.evaluate({zoom: Math.floor(globals.zoom) + 1.0}, feature, featureState),\n globals);\n } else {\n return value.value;\n }\n }\n\n _calculate(min: T, mid: T, max: T, parameters: EvaluationParameters): CrossFaded {\n const z = parameters.zoom;\n return z > parameters.zoomHistory.lastIntegerZoom ? {from: min, to: mid} : {from: max, to: mid};\n }\n\n interpolate(a: PossiblyEvaluatedPropertyValue>): PossiblyEvaluatedPropertyValue> {\n return a;\n }\n}\n/**\n * @internal\n * An implementation of `Property` for `*-pattern` and `line-dasharray`, which are transitioned by cross-fading\n * rather than interpolation.\n */\nexport class CrossFadedProperty implements Property> {\n specification: StylePropertySpecification;\n\n constructor(specification: StylePropertySpecification) {\n this.specification = specification;\n }\n\n possiblyEvaluate(\n value: PropertyValue>,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): CrossFaded {\n if (value.value === undefined) {\n return undefined;\n } else if (value.expression.kind === 'constant') {\n const constant = value.expression.evaluate(parameters, null, {}, canonical, availableImages);\n return this._calculate(constant, constant, constant, parameters);\n } else {\n return this._calculate(\n value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom - 1.0), parameters)),\n value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom), parameters)),\n value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom + 1.0), parameters)),\n parameters);\n }\n }\n\n _calculate(min: T, mid: T, max: T, parameters: EvaluationParameters): CrossFaded {\n const z = parameters.zoom;\n return z > parameters.zoomHistory.lastIntegerZoom ? {from: min, to: mid} : {from: max, to: mid};\n }\n\n interpolate(a?: CrossFaded | null): CrossFaded {\n return a;\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for `heatmap-color` and `line-gradient`. Interpolation is a no-op, and\n * evaluation returns a boolean value in order to indicate its presence, but the real\n * evaluation happens in StyleLayer classes.\n */\n\nexport class ColorRampProperty implements Property {\n specification: StylePropertySpecification;\n\n constructor(specification: StylePropertySpecification) {\n this.specification = specification;\n }\n\n possiblyEvaluate(\n value: PropertyValue,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): boolean {\n return !!value.expression.evaluate(parameters, null, {}, canonical, availableImages);\n }\n\n interpolate(): boolean { return false; }\n}\n\n/**\n * @internal\n * `Properties` holds objects containing default values for the layout or paint property set of a given\n * layer type. These objects are immutable, and they are used as the prototypes for the `_values` members of\n * `Transitionable`, `Transitioning`, `Layout`, and `PossiblyEvaluated`. This allows these classes to avoid\n * doing work in the common case where a property has no explicit value set and should be considered to take\n * on the default value: using `for (const property of Object.keys(this._values))`, they can iterate over\n * only the _own_ properties of `_values`, skipping repeated calculation of transitions and possible/final\n * evaluations for defaults, the result of which will always be the same.\n */\nexport class Properties {\n properties: Props;\n defaultPropertyValues: {[K in keyof Props]: PropertyValue};\n defaultTransitionablePropertyValues: {[K in keyof Props]: TransitionablePropertyValue};\n defaultTransitioningPropertyValues: {[K in keyof Props]: TransitioningPropertyValue};\n defaultPossiblyEvaluatedValues: {[K in keyof Props]: PossiblyEvaluatedPropertyValue};\n overridableProperties: Array;\n\n constructor(properties: Props) {\n this.properties = properties;\n this.defaultPropertyValues = ({} as any);\n this.defaultTransitionablePropertyValues = ({} as any);\n this.defaultTransitioningPropertyValues = ({} as any);\n this.defaultPossiblyEvaluatedValues = ({} as any);\n this.overridableProperties = ([] as any);\n\n for (const property in properties) {\n const prop = properties[property] as any;\n if (prop.specification.overridable) {\n this.overridableProperties.push(property);\n }\n const defaultPropertyValue = this.defaultPropertyValues[property] =\n new PropertyValue(prop, undefined);\n const defaultTransitionablePropertyValue = this.defaultTransitionablePropertyValues[property] =\n new TransitionablePropertyValue(prop);\n this.defaultTransitioningPropertyValues[property] =\n defaultTransitionablePropertyValue.untransitioned();\n this.defaultPossiblyEvaluatedValues[property] =\n defaultPropertyValue.possiblyEvaluate({} as any);\n }\n }\n}\n\nregister('DataDrivenProperty', DataDrivenProperty);\nregister('DataConstantProperty', DataConstantProperty);\nregister('CrossFadedDataDrivenProperty', CrossFadedDataDrivenProperty);\nregister('CrossFadedProperty', CrossFadedProperty);\nregister('ColorRampProperty', ColorRampProperty);\n","import {filterObject} from '../util/util';\n\nimport {latest as styleSpec, supportsPropertyExpression} from '@maplibre/maplibre-gl-style-spec';\nimport {\n validateStyle,\n validateLayoutProperty,\n validatePaintProperty,\n emitValidationErrors\n} from './validate_style';\nimport {Evented} from '../util/evented';\nimport {Layout, Transitionable, Transitioning, Properties, PossiblyEvaluated, PossiblyEvaluatedPropertyValue} from './properties';\n\nimport type {Bucket} from '../data/bucket';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureFilter, FeatureState,\n LayerSpecification,\n FilterSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {TransitionParameters, PropertyValue} from './properties';\nimport {EvaluationParameters} from './evaluation_parameters';\nimport type {CrossfadeParameters} from './evaluation_parameters';\n\nimport type {Transform} from '../geo/transform';\nimport type {CustomLayerInterface} from './style_layer/custom_style_layer';\nimport type {Map} from '../ui/map';\nimport type {StyleSetterOptions} from './style';\nimport {mat4} from 'gl-matrix';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nconst TRANSITION_SUFFIX = '-transition';\n\n/**\n * A base class for style layers\n */\nexport abstract class StyleLayer extends Evented {\n id: string;\n metadata: unknown;\n type: LayerSpecification['type'] | CustomLayerInterface['type'];\n source: string;\n sourceLayer: string;\n minzoom: number;\n maxzoom: number;\n filter: FilterSpecification | void;\n visibility: 'visible' | 'none' | void;\n _crossfadeParameters: CrossfadeParameters;\n\n _unevaluatedLayout: Layout;\n readonly layout: unknown;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n readonly paint: unknown;\n\n _featureFilter: FeatureFilter;\n\n readonly onAdd: ((map: Map) => void);\n readonly onRemove: ((map: Map) => void);\n\n queryRadius?(bucket: Bucket): number;\n queryIntersectsFeature?(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number,\n pixelPosMatrix: mat4\n ): boolean | number;\n\n constructor(layer: LayerSpecification | CustomLayerInterface, properties: Readonly<{\n layout?: Properties;\n paint?: Properties;\n }>) {\n super();\n\n this.id = layer.id;\n this.type = layer.type;\n this._featureFilter = {filter: () => true, needGeometry: false};\n\n if (layer.type === 'custom') return;\n\n layer = (layer as any as LayerSpecification);\n\n this.metadata = layer.metadata;\n this.minzoom = layer.minzoom;\n this.maxzoom = layer.maxzoom;\n\n if (layer.type !== 'background') {\n this.source = layer.source;\n this.sourceLayer = layer['source-layer'];\n this.filter = layer.filter;\n }\n\n if (properties.layout) {\n this._unevaluatedLayout = new Layout(properties.layout);\n }\n\n if (properties.paint) {\n this._transitionablePaint = new Transitionable(properties.paint);\n\n for (const property in layer.paint) {\n this.setPaintProperty(property, layer.paint[property], {validate: false});\n }\n for (const property in layer.layout) {\n this.setLayoutProperty(property, layer.layout[property], {validate: false});\n }\n\n this._transitioningPaint = this._transitionablePaint.untransitioned();\n //$FlowFixMe\n this.paint = new PossiblyEvaluated(properties.paint);\n }\n }\n\n getCrossfadeParameters() {\n return this._crossfadeParameters;\n }\n\n getLayoutProperty(name: string) {\n if (name === 'visibility') {\n return this.visibility;\n }\n\n return this._unevaluatedLayout.getValue(name);\n }\n\n setLayoutProperty(name: string, value: any, options: StyleSetterOptions = {}) {\n if (value !== null && value !== undefined) {\n const key = `layers.${this.id}.layout.${name}`;\n if (this._validate(validateLayoutProperty, key, name, value, options)) {\n return;\n }\n }\n\n if (name === 'visibility') {\n this.visibility = value;\n return;\n }\n\n this._unevaluatedLayout.setValue(name, value);\n }\n\n getPaintProperty(name: string) {\n if (name.endsWith(TRANSITION_SUFFIX)) {\n return this._transitionablePaint.getTransition(name.slice(0, -TRANSITION_SUFFIX.length));\n } else {\n return this._transitionablePaint.getValue(name);\n }\n }\n\n setPaintProperty(name: string, value: unknown, options: StyleSetterOptions = {}) {\n if (value !== null && value !== undefined) {\n const key = `layers.${this.id}.paint.${name}`;\n if (this._validate(validatePaintProperty, key, name, value, options)) {\n return false;\n }\n }\n\n if (name.endsWith(TRANSITION_SUFFIX)) {\n this._transitionablePaint.setTransition(name.slice(0, -TRANSITION_SUFFIX.length), (value as any) || undefined);\n return false;\n } else {\n const transitionable = this._transitionablePaint._values[name];\n const isCrossFadedProperty = transitionable.property.specification['property-type'] === 'cross-faded-data-driven';\n const wasDataDriven = transitionable.value.isDataDriven();\n const oldValue = transitionable.value;\n\n this._transitionablePaint.setValue(name, value);\n this._handleSpecialPaintPropertyUpdate(name);\n\n const newValue = this._transitionablePaint._values[name].value;\n const isDataDriven = newValue.isDataDriven();\n\n // if a cross-faded value is changed, we need to make sure the new icons get added to each tile's iconAtlas\n // so a call to _updateLayer is necessary, and we return true from this function so it gets called in\n // Style#setPaintProperty\n return isDataDriven || wasDataDriven || isCrossFadedProperty || this._handleOverridablePaintPropertyUpdate(name, oldValue, newValue);\n }\n }\n\n _handleSpecialPaintPropertyUpdate(_: string) {\n // No-op; can be overridden by derived classes.\n }\n\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _handleOverridablePaintPropertyUpdate(name: string, oldValue: PropertyValue, newValue: PropertyValue): boolean {\n // No-op; can be overridden by derived classes.\n return false;\n }\n\n isHidden(zoom: number) {\n if (this.minzoom && zoom < this.minzoom) return true;\n if (this.maxzoom && zoom >= this.maxzoom) return true;\n return this.visibility === 'none';\n }\n\n updateTransitions(parameters: TransitionParameters) {\n this._transitioningPaint = this._transitionablePaint.transitioned(parameters, this._transitioningPaint);\n }\n\n hasTransition() {\n return this._transitioningPaint.hasTransition();\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n if (parameters.getCrossfadeParameters) {\n this._crossfadeParameters = parameters.getCrossfadeParameters();\n }\n\n if (this._unevaluatedLayout) {\n (this as any).layout = this._unevaluatedLayout.possiblyEvaluate(parameters, undefined, availableImages);\n }\n\n (this as any).paint = this._transitioningPaint.possiblyEvaluate(parameters, undefined, availableImages);\n }\n\n serialize(): LayerSpecification {\n const output: LayerSpecification = {\n 'id': this.id,\n 'type': this.type as LayerSpecification['type'],\n 'source': this.source,\n 'source-layer': this.sourceLayer,\n 'metadata': this.metadata,\n 'minzoom': this.minzoom,\n 'maxzoom': this.maxzoom,\n 'filter': this.filter as FilterSpecification,\n 'layout': this._unevaluatedLayout && this._unevaluatedLayout.serialize(),\n 'paint': this._transitionablePaint && this._transitionablePaint.serialize()\n };\n\n if (this.visibility) {\n output.layout = output.layout || {};\n output.layout.visibility = this.visibility;\n }\n\n return filterObject(output, (value, key) => {\n return value !== undefined &&\n !(key === 'layout' && !Object.keys(value).length) &&\n !(key === 'paint' && !Object.keys(value).length);\n });\n }\n\n _validate(validate: Function, key: string, name: string, value: unknown, options: StyleSetterOptions = {}) {\n if (options && options.validate === false) {\n return false;\n }\n return emitValidationErrors(this, validate.call(validateStyle, {\n key,\n layerType: this.type,\n objectKey: name,\n value,\n styleSpec,\n // Workaround for https://github.com/mapbox/mapbox-gl-js/issues/2407\n style: {glyphs: true, sprite: true}\n }));\n }\n\n is3D() {\n return false;\n }\n\n isTileClipped() {\n return false;\n }\n\n hasOffscreenPass() {\n return false;\n }\n\n resize() {\n // noop\n }\n\n isStateDependent() {\n for (const property in (this as any).paint._values) {\n const value = (this as any).paint.get(property);\n if (!(value instanceof PossiblyEvaluatedPropertyValue) || !supportsPropertyExpression(value.property.specification)) {\n continue;\n }\n\n if ((value.value.kind === 'source' || value.value.kind === 'composite') &&\n value.value.isStateDependent) {\n return true;\n }\n }\n return false;\n }\n}\n","// Note: all \"sizes\" are measured in bytes\n\nimport type {Transferable} from '../types/transferable';\n\n/**\n * @internal\n * A view type size\n */\nconst viewTypes = {\n 'Int8': Int8Array,\n 'Uint8': Uint8Array,\n 'Int16': Int16Array,\n 'Uint16': Uint16Array,\n 'Int32': Int32Array,\n 'Uint32': Uint32Array,\n 'Float32': Float32Array\n};\n\n/**\n * @internal\n * A view type size\n */\nexport type ViewType = keyof typeof viewTypes;\n\n/** @internal */\nclass Struct {\n _pos1: number;\n _pos2: number;\n _pos4: number;\n _pos8: number;\n readonly _structArray: StructArray;\n\n // The following properties are defined on the prototype of sub classes.\n size: number;\n\n /**\n * @param structArray - The StructArray the struct is stored in\n * @param index - The index of the struct in the StructArray.\n */\n constructor(structArray: StructArray, index: number) {\n (this as any)._structArray = structArray;\n this._pos1 = index * this.size;\n this._pos2 = this._pos1 / 2;\n this._pos4 = this._pos1 / 4;\n this._pos8 = this._pos1 / 8;\n }\n}\n\nconst DEFAULT_CAPACITY = 128;\nconst RESIZE_MULTIPLIER = 5;\n\n/**\n * @internal\n * A struct array memeber\n */\nexport type StructArrayMember = {\n name: string;\n type: ViewType;\n components: number;\n offset: number;\n};\n\nexport type StructArrayLayout = {\n members: Array;\n size: number;\n alignment: number;\n};\n\n/**\n * An array that can be desialized\n */\nexport type SerializedStructArray = {\n length: number;\n arrayBuffer: ArrayBuffer;\n};\n\n/**\n * @internal\n * `StructArray` provides an abstraction over `ArrayBuffer` and `TypedArray`\n * making it behave like an array of typed structs.\n *\n * Conceptually, a StructArray is comprised of elements, i.e., instances of its\n * associated struct type. Each particular struct type, together with an\n * alignment size, determines the memory layout of a StructArray whose elements\n * are of that type. Thus, for each such layout that we need, we have\n * a corresponding StructArrayLayout class, inheriting from StructArray and\n * implementing `emplaceBack()` and `_refreshViews()`.\n *\n * In some cases, where we need to access particular elements of a StructArray,\n * we implement a more specific subclass that inherits from one of the\n * StructArrayLayouts and adds a `get(i): T` accessor that returns a structured\n * object whose properties are proxies into the underlying memory space for the\n * i-th element. This affords the convience of working with (seemingly) plain\n * Javascript objects without the overhead of serializing/deserializing them\n * into ArrayBuffers for efficient web worker transfer.\n */\nabstract class StructArray {\n capacity: number;\n length: number;\n isTransferred: boolean;\n arrayBuffer: ArrayBuffer;\n uint8: Uint8Array;\n\n // The following properties are defined on the prototype.\n members: Array;\n bytesPerElement: number;\n abstract emplaceBack(...v: number[]);\n abstract emplace(i: number, ...v: number[]);\n\n constructor() {\n this.isTransferred = false;\n this.capacity = -1;\n this.resize(0);\n }\n\n /**\n * Serialize a StructArray instance. Serializes both the raw data and the\n * metadata needed to reconstruct the StructArray base class during\n * deserialization.\n */\n static serialize(array: StructArray, transferables?: Array): SerializedStructArray {\n\n array._trim();\n\n if (transferables) {\n array.isTransferred = true;\n transferables.push(array.arrayBuffer);\n }\n\n return {\n length: array.length,\n arrayBuffer: array.arrayBuffer,\n };\n }\n\n static deserialize(input: SerializedStructArray) {\n const structArray = Object.create(this.prototype);\n structArray.arrayBuffer = input.arrayBuffer;\n structArray.length = input.length;\n structArray.capacity = input.arrayBuffer.byteLength / structArray.bytesPerElement;\n structArray._refreshViews();\n return structArray;\n }\n\n /**\n * Resize the array to discard unused capacity.\n */\n _trim() {\n if (this.length !== this.capacity) {\n this.capacity = this.length;\n this.arrayBuffer = this.arrayBuffer.slice(0, this.length * this.bytesPerElement);\n this._refreshViews();\n }\n }\n\n /**\n * Resets the length of the array to 0 without de-allocating capcacity.\n */\n clear() {\n this.length = 0;\n }\n\n /**\n * Resize the array.\n * If `n` is greater than the current length then additional elements with undefined values are added.\n * If `n` is less than the current length then the array will be reduced to the first `n` elements.\n * @param n - The new size of the array.\n */\n resize(n: number) {\n this.reserve(n);\n this.length = n;\n }\n\n /**\n * Indicate a planned increase in size, so that any necessary allocation may\n * be done once, ahead of time.\n * @param n - The expected size of the array.\n */\n reserve(n: number) {\n if (n > this.capacity) {\n this.capacity = Math.max(n, Math.floor(this.capacity * RESIZE_MULTIPLIER), DEFAULT_CAPACITY);\n this.arrayBuffer = new ArrayBuffer(this.capacity * this.bytesPerElement);\n\n const oldUint8Array = this.uint8;\n this._refreshViews();\n if (oldUint8Array) this.uint8.set(oldUint8Array);\n }\n }\n\n /**\n * Create TypedArray views for the current ArrayBuffer.\n */\n _refreshViews() {\n throw new Error('_refreshViews() must be implemented by each concrete StructArray layout');\n }\n}\n\n/**\n * Given a list of member fields, create a full StructArrayLayout, in\n * particular calculating the correct byte offset for each field. This data\n * is used at build time to generate StructArrayLayout_*#emplaceBack() and\n * other accessors, and at runtime for binding vertex buffer attributes.\n */\nfunction createLayout(\n members: Array<{\n name: string;\n type: ViewType;\n readonly components?: number;\n }>,\n alignment: number = 1\n): StructArrayLayout {\n\n let offset = 0;\n let maxSize = 0;\n const layoutMembers = members.map((member) => {\n const typeSize = sizeOf(member.type);\n const memberOffset = offset = align(offset, Math.max(alignment, typeSize));\n const components = member.components || 1;\n\n maxSize = Math.max(maxSize, typeSize);\n offset += typeSize * components;\n\n return {\n name: member.name,\n type: member.type,\n components,\n offset: memberOffset,\n };\n });\n\n const size = align(offset, Math.max(maxSize, alignment));\n\n return {\n members: layoutMembers,\n size,\n alignment\n };\n}\n\nfunction sizeOf(type: ViewType): number {\n return viewTypes[type].BYTES_PER_ELEMENT;\n}\n\nfunction align(offset: number, size: number): number {\n return Math.ceil(offset / size) * size;\n}\n\nexport {StructArray, Struct, viewTypes, createLayout};\n","// This file is generated. Edit build/generate-struct-arrays.ts, then run `npm run codegen`.\n\nimport {Struct, StructArray} from '../util/struct_array';\nimport {register} from '../util/web_worker_transfer';\nimport Point from '@mapbox/point-geometry';\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n *\n */\nclass StructArrayLayout2i4 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1);\n }\n\n public emplace(i: number, v0: number, v1: number) {\n const o2 = i * 2;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n return i;\n }\n}\n\nStructArrayLayout2i4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout2i4', StructArrayLayout2i4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[3]\n *\n */\nclass StructArrayLayout3i6 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o2 = i * 3;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout3i6.prototype.bytesPerElement = 6;\nregister('StructArrayLayout3i6', StructArrayLayout3i6);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[4]\n *\n */\nclass StructArrayLayout4i8 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number) {\n const o2 = i * 4;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n return i;\n }\n}\n\nStructArrayLayout4i8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout4i8', StructArrayLayout4i8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Int16[4]\n *\n */\nclass StructArrayLayout2i4i12 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const o2 = i * 6;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n return i;\n }\n}\n\nStructArrayLayout2i4i12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout2i4i12', StructArrayLayout2i4i12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Uint8[4]\n *\n */\nclass StructArrayLayout2i4ub8 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const o2 = i * 4;\n const o1 = i * 8;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.uint8[o1 + 4] = v2;\n this.uint8[o1 + 5] = v3;\n this.uint8[o1 + 6] = v4;\n this.uint8[o1 + 7] = v5;\n return i;\n }\n}\n\nStructArrayLayout2i4ub8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout2i4ub8', StructArrayLayout2i4ub8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[2]\n *\n */\nclass StructArrayLayout2f8 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1);\n }\n\n public emplace(i: number, v0: number, v1: number) {\n const o4 = i * 2;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n return i;\n }\n}\n\nStructArrayLayout2f8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout2f8', StructArrayLayout2f8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[10]\n *\n */\nclass StructArrayLayout10ui20 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number) {\n const o2 = i * 10;\n this.uint16[o2 + 0] = v0;\n this.uint16[o2 + 1] = v1;\n this.uint16[o2 + 2] = v2;\n this.uint16[o2 + 3] = v3;\n this.uint16[o2 + 4] = v4;\n this.uint16[o2 + 5] = v5;\n this.uint16[o2 + 6] = v6;\n this.uint16[o2 + 7] = v7;\n this.uint16[o2 + 8] = v8;\n this.uint16[o2 + 9] = v9;\n return i;\n }\n}\n\nStructArrayLayout10ui20.prototype.bytesPerElement = 20;\nregister('StructArrayLayout10ui20', StructArrayLayout10ui20);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[4]\n * [8]: Uint16[4]\n * [16]: Int16[4]\n *\n */\nclass StructArrayLayout4i4ui4i24 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number) {\n const o2 = i * 12;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.uint16[o2 + 4] = v4;\n this.uint16[o2 + 5] = v5;\n this.uint16[o2 + 6] = v6;\n this.uint16[o2 + 7] = v7;\n this.int16[o2 + 8] = v8;\n this.int16[o2 + 9] = v9;\n this.int16[o2 + 10] = v10;\n this.int16[o2 + 11] = v11;\n return i;\n }\n}\n\nStructArrayLayout4i4ui4i24.prototype.bytesPerElement = 24;\nregister('StructArrayLayout4i4ui4i24', StructArrayLayout4i4ui4i24);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[3]\n *\n */\nclass StructArrayLayout3f12 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o4 = i * 3;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout3f12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout3f12', StructArrayLayout3f12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint32[1]\n *\n */\nclass StructArrayLayout1ul4 extends StructArray {\n uint8: Uint8Array;\n uint32: Uint32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0);\n }\n\n public emplace(i: number, v0: number) {\n const o4 = i * 1;\n this.uint32[o4 + 0] = v0;\n return i;\n }\n}\n\nStructArrayLayout1ul4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout1ul4', StructArrayLayout1ul4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[6]\n * [12]: Uint32[1]\n * [16]: Uint16[2]\n *\n */\nclass StructArrayLayout6i1ul2ui20 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint32: Uint32Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number) {\n const o2 = i * 10;\n const o4 = i * 5;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n this.uint32[o4 + 3] = v6;\n this.uint16[o2 + 8] = v7;\n this.uint16[o2 + 9] = v8;\n return i;\n }\n}\n\nStructArrayLayout6i1ul2ui20.prototype.bytesPerElement = 20;\nregister('StructArrayLayout6i1ul2ui20', StructArrayLayout6i1ul2ui20);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Int16[2]\n * [8]: Int16[2]\n *\n */\nclass StructArrayLayout2i2i2i12 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const o2 = i * 6;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n return i;\n }\n}\n\nStructArrayLayout2i2i2i12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout2i2i2i12', StructArrayLayout2i2i2i12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[2]\n * [8]: Float32[1]\n * [12]: Int16[2]\n *\n */\nclass StructArrayLayout2f1f2i16 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number) {\n const o4 = i * 4;\n const o2 = i * 8;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n this.int16[o2 + 6] = v3;\n this.int16[o2 + 7] = v4;\n return i;\n }\n}\n\nStructArrayLayout2f1f2i16.prototype.bytesPerElement = 16;\nregister('StructArrayLayout2f1f2i16', StructArrayLayout2f1f2i16);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint8[2]\n * [4]: Float32[2]\n *\n */\nclass StructArrayLayout2ub2f12 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number) {\n const o1 = i * 12;\n const o4 = i * 3;\n this.uint8[o1 + 0] = v0;\n this.uint8[o1 + 1] = v1;\n this.float32[o4 + 1] = v2;\n this.float32[o4 + 2] = v3;\n return i;\n }\n}\n\nStructArrayLayout2ub2f12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout2ub2f12', StructArrayLayout2ub2f12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[3]\n *\n */\nclass StructArrayLayout3ui6 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o2 = i * 3;\n this.uint16[o2 + 0] = v0;\n this.uint16[o2 + 1] = v1;\n this.uint16[o2 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout3ui6.prototype.bytesPerElement = 6;\nregister('StructArrayLayout3ui6', StructArrayLayout3ui6);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Uint16[2]\n * [8]: Uint32[3]\n * [20]: Uint16[3]\n * [28]: Float32[2]\n * [36]: Uint8[3]\n * [40]: Uint32[1]\n * [44]: Int16[1]\n *\n */\nclass StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint16: Uint16Array;\n uint32: Uint32Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number) {\n const o2 = i * 24;\n const o4 = i * 12;\n const o1 = i * 48;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.uint16[o2 + 2] = v2;\n this.uint16[o2 + 3] = v3;\n this.uint32[o4 + 2] = v4;\n this.uint32[o4 + 3] = v5;\n this.uint32[o4 + 4] = v6;\n this.uint16[o2 + 10] = v7;\n this.uint16[o2 + 11] = v8;\n this.uint16[o2 + 12] = v9;\n this.float32[o4 + 7] = v10;\n this.float32[o4 + 8] = v11;\n this.uint8[o1 + 36] = v12;\n this.uint8[o1 + 37] = v13;\n this.uint8[o1 + 38] = v14;\n this.uint32[o4 + 10] = v15;\n this.int16[o2 + 22] = v16;\n return i;\n }\n}\n\nStructArrayLayout2i2ui3ul3ui2f3ub1ul1i48.prototype.bytesPerElement = 48;\nregister('StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48', StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[8]\n * [16]: Uint16[15]\n * [48]: Uint32[1]\n * [52]: Float32[2]\n * [60]: Uint16[2]\n *\n */\nclass StructArrayLayout8i15ui1ul2f2ui64 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint16: Uint16Array;\n uint32: Uint32Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number, v17: number, v18: number, v19: number, v20: number, v21: number, v22: number, v23: number, v24: number, v25: number, v26: number, v27: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number, v17: number, v18: number, v19: number, v20: number, v21: number, v22: number, v23: number, v24: number, v25: number, v26: number, v27: number) {\n const o2 = i * 32;\n const o4 = i * 16;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n this.int16[o2 + 6] = v6;\n this.int16[o2 + 7] = v7;\n this.uint16[o2 + 8] = v8;\n this.uint16[o2 + 9] = v9;\n this.uint16[o2 + 10] = v10;\n this.uint16[o2 + 11] = v11;\n this.uint16[o2 + 12] = v12;\n this.uint16[o2 + 13] = v13;\n this.uint16[o2 + 14] = v14;\n this.uint16[o2 + 15] = v15;\n this.uint16[o2 + 16] = v16;\n this.uint16[o2 + 17] = v17;\n this.uint16[o2 + 18] = v18;\n this.uint16[o2 + 19] = v19;\n this.uint16[o2 + 20] = v20;\n this.uint16[o2 + 21] = v21;\n this.uint16[o2 + 22] = v22;\n this.uint32[o4 + 12] = v23;\n this.float32[o4 + 13] = v24;\n this.float32[o4 + 14] = v25;\n this.uint16[o2 + 30] = v26;\n this.uint16[o2 + 31] = v27;\n return i;\n }\n}\n\nStructArrayLayout8i15ui1ul2f2ui64.prototype.bytesPerElement = 64;\nregister('StructArrayLayout8i15ui1ul2f2ui64', StructArrayLayout8i15ui1ul2f2ui64);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[1]\n *\n */\nclass StructArrayLayout1f4 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0);\n }\n\n public emplace(i: number, v0: number) {\n const o4 = i * 1;\n this.float32[o4 + 0] = v0;\n return i;\n }\n}\n\nStructArrayLayout1f4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout1f4', StructArrayLayout1f4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[1]\n * [4]: Float32[2]\n *\n */\nclass StructArrayLayout1ui2f12 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o2 = i * 6;\n const o4 = i * 3;\n this.uint16[o2 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout1ui2f12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout1ui2f12', StructArrayLayout1ui2f12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint32[1]\n * [4]: Uint16[2]\n *\n */\nclass StructArrayLayout1ul2ui8 extends StructArray {\n uint8: Uint8Array;\n uint32: Uint32Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o4 = i * 2;\n const o2 = i * 4;\n this.uint32[o4 + 0] = v0;\n this.uint16[o2 + 2] = v1;\n this.uint16[o2 + 3] = v2;\n return i;\n }\n}\n\nStructArrayLayout1ul2ui8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout1ul2ui8', StructArrayLayout1ul2ui8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[2]\n *\n */\nclass StructArrayLayout2ui4 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1);\n }\n\n public emplace(i: number, v0: number, v1: number) {\n const o2 = i * 2;\n this.uint16[o2 + 0] = v0;\n this.uint16[o2 + 1] = v1;\n return i;\n }\n}\n\nStructArrayLayout2ui4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout2ui4', StructArrayLayout2ui4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[1]\n *\n */\nclass StructArrayLayout1ui2 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0);\n }\n\n public emplace(i: number, v0: number) {\n const o2 = i * 1;\n this.uint16[o2 + 0] = v0;\n return i;\n }\n}\n\nStructArrayLayout1ui2.prototype.bytesPerElement = 2;\nregister('StructArrayLayout1ui2', StructArrayLayout1ui2);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[4]\n *\n */\nclass StructArrayLayout4f16 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number) {\n const o4 = i * 4;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n this.float32[o4 + 3] = v3;\n return i;\n }\n}\n\nStructArrayLayout4f16.prototype.bytesPerElement = 16;\nregister('StructArrayLayout4f16', StructArrayLayout4f16);\n\n/** @internal */\nclass CollisionBoxStruct extends Struct {\n _structArray: CollisionBoxArray;\n get anchorPointX() { return this._structArray.int16[this._pos2 + 0]; }\n get anchorPointY() { return this._structArray.int16[this._pos2 + 1]; }\n get x1() { return this._structArray.int16[this._pos2 + 2]; }\n get y1() { return this._structArray.int16[this._pos2 + 3]; }\n get x2() { return this._structArray.int16[this._pos2 + 4]; }\n get y2() { return this._structArray.int16[this._pos2 + 5]; }\n get featureIndex() { return this._structArray.uint32[this._pos4 + 3]; }\n get sourceLayerIndex() { return this._structArray.uint16[this._pos2 + 8]; }\n get bucketIndex() { return this._structArray.uint16[this._pos2 + 9]; }\n get anchorPoint() { return new Point(this.anchorPointX, this.anchorPointY); }\n}\n\nCollisionBoxStruct.prototype.size = 20;\n\nexport type CollisionBox = CollisionBoxStruct;\n\n/** @internal */\nexport class CollisionBoxArray extends StructArrayLayout6i1ul2ui20 {\n /**\n * Return the CollisionBoxStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): CollisionBoxStruct {\n return new CollisionBoxStruct(this, index);\n }\n}\n\nregister('CollisionBoxArray', CollisionBoxArray);\n\n/** @internal */\nclass PlacedSymbolStruct extends Struct {\n _structArray: PlacedSymbolArray;\n get anchorX() { return this._structArray.int16[this._pos2 + 0]; }\n get anchorY() { return this._structArray.int16[this._pos2 + 1]; }\n get glyphStartIndex() { return this._structArray.uint16[this._pos2 + 2]; }\n get numGlyphs() { return this._structArray.uint16[this._pos2 + 3]; }\n get vertexStartIndex() { return this._structArray.uint32[this._pos4 + 2]; }\n get lineStartIndex() { return this._structArray.uint32[this._pos4 + 3]; }\n get lineLength() { return this._structArray.uint32[this._pos4 + 4]; }\n get segment() { return this._structArray.uint16[this._pos2 + 10]; }\n get lowerSize() { return this._structArray.uint16[this._pos2 + 11]; }\n get upperSize() { return this._structArray.uint16[this._pos2 + 12]; }\n get lineOffsetX() { return this._structArray.float32[this._pos4 + 7]; }\n get lineOffsetY() { return this._structArray.float32[this._pos4 + 8]; }\n get writingMode() { return this._structArray.uint8[this._pos1 + 36]; }\n get placedOrientation() { return this._structArray.uint8[this._pos1 + 37]; }\n set placedOrientation(x: number) { this._structArray.uint8[this._pos1 + 37] = x; }\n get hidden() { return this._structArray.uint8[this._pos1 + 38]; }\n set hidden(x: number) { this._structArray.uint8[this._pos1 + 38] = x; }\n get crossTileID() { return this._structArray.uint32[this._pos4 + 10]; }\n set crossTileID(x: number) { this._structArray.uint32[this._pos4 + 10] = x; }\n get associatedIconIndex() { return this._structArray.int16[this._pos2 + 22]; }\n}\n\nPlacedSymbolStruct.prototype.size = 48;\n\nexport type PlacedSymbol = PlacedSymbolStruct;\n\n/** @internal */\nexport class PlacedSymbolArray extends StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 {\n /**\n * Return the PlacedSymbolStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): PlacedSymbolStruct {\n return new PlacedSymbolStruct(this, index);\n }\n}\n\nregister('PlacedSymbolArray', PlacedSymbolArray);\n\n/** @internal */\nclass SymbolInstanceStruct extends Struct {\n _structArray: SymbolInstanceArray;\n get anchorX() { return this._structArray.int16[this._pos2 + 0]; }\n get anchorY() { return this._structArray.int16[this._pos2 + 1]; }\n get rightJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 2]; }\n get centerJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 3]; }\n get leftJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 4]; }\n get verticalPlacedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 5]; }\n get placedIconSymbolIndex() { return this._structArray.int16[this._pos2 + 6]; }\n get verticalPlacedIconSymbolIndex() { return this._structArray.int16[this._pos2 + 7]; }\n get key() { return this._structArray.uint16[this._pos2 + 8]; }\n get textBoxStartIndex() { return this._structArray.uint16[this._pos2 + 9]; }\n get textBoxEndIndex() { return this._structArray.uint16[this._pos2 + 10]; }\n get verticalTextBoxStartIndex() { return this._structArray.uint16[this._pos2 + 11]; }\n get verticalTextBoxEndIndex() { return this._structArray.uint16[this._pos2 + 12]; }\n get iconBoxStartIndex() { return this._structArray.uint16[this._pos2 + 13]; }\n get iconBoxEndIndex() { return this._structArray.uint16[this._pos2 + 14]; }\n get verticalIconBoxStartIndex() { return this._structArray.uint16[this._pos2 + 15]; }\n get verticalIconBoxEndIndex() { return this._structArray.uint16[this._pos2 + 16]; }\n get featureIndex() { return this._structArray.uint16[this._pos2 + 17]; }\n get numHorizontalGlyphVertices() { return this._structArray.uint16[this._pos2 + 18]; }\n get numVerticalGlyphVertices() { return this._structArray.uint16[this._pos2 + 19]; }\n get numIconVertices() { return this._structArray.uint16[this._pos2 + 20]; }\n get numVerticalIconVertices() { return this._structArray.uint16[this._pos2 + 21]; }\n get useRuntimeCollisionCircles() { return this._structArray.uint16[this._pos2 + 22]; }\n get crossTileID() { return this._structArray.uint32[this._pos4 + 12]; }\n set crossTileID(x: number) { this._structArray.uint32[this._pos4 + 12] = x; }\n get textBoxScale() { return this._structArray.float32[this._pos4 + 13]; }\n get collisionCircleDiameter() { return this._structArray.float32[this._pos4 + 14]; }\n get textAnchorOffsetStartIndex() { return this._structArray.uint16[this._pos2 + 30]; }\n get textAnchorOffsetEndIndex() { return this._structArray.uint16[this._pos2 + 31]; }\n}\n\nSymbolInstanceStruct.prototype.size = 64;\n\nexport type SymbolInstance = SymbolInstanceStruct;\n\n/** @internal */\nexport class SymbolInstanceArray extends StructArrayLayout8i15ui1ul2f2ui64 {\n /**\n * Return the SymbolInstanceStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): SymbolInstanceStruct {\n return new SymbolInstanceStruct(this, index);\n }\n}\n\nregister('SymbolInstanceArray', SymbolInstanceArray);\n\n/** @internal */\nexport class GlyphOffsetArray extends StructArrayLayout1f4 {\n getoffsetX(index: number) { return this.float32[index * 1 + 0]; }\n}\n\nregister('GlyphOffsetArray', GlyphOffsetArray);\n\n/** @internal */\nexport class SymbolLineVertexArray extends StructArrayLayout3i6 {\n getx(index: number) { return this.int16[index * 3 + 0]; }\n gety(index: number) { return this.int16[index * 3 + 1]; }\n gettileUnitDistanceFromAnchor(index: number) { return this.int16[index * 3 + 2]; }\n}\n\nregister('SymbolLineVertexArray', SymbolLineVertexArray);\n\n/** @internal */\nclass TextAnchorOffsetStruct extends Struct {\n _structArray: TextAnchorOffsetArray;\n get textAnchor() { return this._structArray.uint16[this._pos2 + 0]; }\n get textOffset0() { return this._structArray.float32[this._pos4 + 1]; }\n get textOffset1() { return this._structArray.float32[this._pos4 + 2]; }\n}\n\nTextAnchorOffsetStruct.prototype.size = 12;\n\nexport type TextAnchorOffset = TextAnchorOffsetStruct;\n\n/** @internal */\nexport class TextAnchorOffsetArray extends StructArrayLayout1ui2f12 {\n /**\n * Return the TextAnchorOffsetStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): TextAnchorOffsetStruct {\n return new TextAnchorOffsetStruct(this, index);\n }\n}\n\nregister('TextAnchorOffsetArray', TextAnchorOffsetArray);\n\n/** @internal */\nclass FeatureIndexStruct extends Struct {\n _structArray: FeatureIndexArray;\n get featureIndex() { return this._structArray.uint32[this._pos4 + 0]; }\n get sourceLayerIndex() { return this._structArray.uint16[this._pos2 + 2]; }\n get bucketIndex() { return this._structArray.uint16[this._pos2 + 3]; }\n}\n\nFeatureIndexStruct.prototype.size = 8;\n\nexport type FeatureIndex = FeatureIndexStruct;\n\n/** @internal */\nexport class FeatureIndexArray extends StructArrayLayout1ul2ui8 {\n /**\n * Return the FeatureIndexStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): FeatureIndexStruct {\n return new FeatureIndexStruct(this, index);\n }\n}\n\nregister('FeatureIndexArray', FeatureIndexArray);\n\nexport class PosArray extends StructArrayLayout2i4 {}\nexport class Pos3dArray extends StructArrayLayout3i6 {}\nexport class RasterBoundsArray extends StructArrayLayout4i8 {}\nexport class CircleLayoutArray extends StructArrayLayout2i4 {}\nexport class FillLayoutArray extends StructArrayLayout2i4 {}\nexport class FillExtrusionLayoutArray extends StructArrayLayout2i4i12 {}\nexport class HeatmapLayoutArray extends StructArrayLayout2i4 {}\nexport class LineLayoutArray extends StructArrayLayout2i4ub8 {}\nexport class LineExtLayoutArray extends StructArrayLayout2f8 {}\nexport class PatternLayoutArray extends StructArrayLayout10ui20 {}\nexport class SymbolLayoutArray extends StructArrayLayout4i4ui4i24 {}\nexport class SymbolDynamicLayoutArray extends StructArrayLayout3f12 {}\nexport class SymbolOpacityArray extends StructArrayLayout1ul4 {}\nexport class CollisionBoxLayoutArray extends StructArrayLayout2i2i2i12 {}\nexport class CollisionCircleLayoutArray extends StructArrayLayout2f1f2i16 {}\nexport class CollisionVertexArray extends StructArrayLayout2ub2f12 {}\nexport class QuadTriangleArray extends StructArrayLayout3ui6 {}\nexport class TriangleIndexArray extends StructArrayLayout3ui6 {}\nexport class LineIndexArray extends StructArrayLayout2ui4 {}\nexport class LineStripIndexArray extends StructArrayLayout1ui2 {}\nexport {\n StructArrayLayout2i4,\n StructArrayLayout3i6,\n StructArrayLayout4i8,\n StructArrayLayout2i4i12,\n StructArrayLayout2i4ub8,\n StructArrayLayout2f8,\n StructArrayLayout10ui20,\n StructArrayLayout4i4ui4i24,\n StructArrayLayout3f12,\n StructArrayLayout1ul4,\n StructArrayLayout6i1ul2ui20,\n StructArrayLayout2i2i2i12,\n StructArrayLayout2f1f2i16,\n StructArrayLayout2ub2f12,\n StructArrayLayout3ui6,\n StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48,\n StructArrayLayout8i15ui1ul2f2ui64,\n StructArrayLayout1f4,\n StructArrayLayout1ui2f12,\n StructArrayLayout1ul2ui8,\n StructArrayLayout2ui4,\n StructArrayLayout1ui2,\n StructArrayLayout4f16\n};\n","import {createLayout} from '../../util/struct_array';\n\nconst layout = createLayout([\n {name: 'a_pos', components: 2, type: 'Int16'}\n], 4);\n\nexport default layout;\nexport const {members, size, alignment} = layout;\n","import {warnOnce} from '../util/util';\n\nimport {register} from '../util/web_worker_transfer';\n\nimport type {VertexArrayObject} from '../render/vertex_array_object';\nimport type {StructArray} from '../util/struct_array';\n\n/**\n * @internal\n * A single segment of a vector\n */\nexport type Segment = {\n sortKey?: number;\n vertexOffset: number;\n primitiveOffset: number;\n vertexLength: number;\n primitiveLength: number;\n vaos: {[_: string]: VertexArrayObject};\n};\n\n/**\n * @internal\n * Used for calculations on vector segments\n */\nexport class SegmentVector {\n static MAX_VERTEX_ARRAY_LENGTH: number;\n segments: Array;\n\n constructor(segments: Array = []) {\n this.segments = segments;\n }\n\n prepareSegment(\n numVertices: number,\n layoutVertexArray: StructArray,\n indexArray: StructArray,\n sortKey?: number\n ): Segment {\n let segment: Segment = this.segments[this.segments.length - 1];\n if (numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) warnOnce(`Max vertices per segment is ${SegmentVector.MAX_VERTEX_ARRAY_LENGTH}: bucket requested ${numVertices}`);\n if (!segment || segment.vertexLength + numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH || segment.sortKey !== sortKey) {\n segment = ({\n vertexOffset: layoutVertexArray.length,\n primitiveOffset: indexArray.length,\n vertexLength: 0,\n primitiveLength: 0\n } as any);\n if (sortKey !== undefined) segment.sortKey = sortKey;\n this.segments.push(segment);\n }\n return segment;\n }\n\n get() {\n return this.segments;\n }\n\n destroy() {\n for (const segment of this.segments) {\n for (const k in segment.vaos) {\n segment.vaos[k].destroy();\n }\n }\n }\n\n static simpleSegment(\n vertexOffset: number,\n primitiveOffset: number,\n vertexLength: number,\n primitiveLength: number\n ): SegmentVector {\n return new SegmentVector([{\n vertexOffset,\n primitiveOffset,\n vertexLength,\n primitiveLength,\n vaos: {},\n sortKey: 0\n }]);\n }\n}\n\n/**\n * The maximum size of a vertex array. This limit is imposed by WebGL's 16 bit\n * addressing of vertex buffers.\n */\nSegmentVector.MAX_VERTEX_ARRAY_LENGTH = Math.pow(2, 16) - 1;\n\nregister('SegmentVector', SegmentVector);\n","import {clamp} from '../util/util';\n\n/**\n * Packs two numbers, interpreted as 8-bit unsigned integers, into a single\n * float. Unpack them in the shader using the `unpack_float()` function,\n * defined in _prelude.vertex.glsl\n */\nexport function packUint8ToFloat(a: number, b: number) {\n // coerce a and b to 8-bit ints\n a = clamp(Math.floor(a), 0, 255);\n b = clamp(Math.floor(b), 0, 255);\n return 256 * a + b;\n}\n","import {createLayout} from '../../util/struct_array';\n\nexport const patternAttributes = createLayout([\n // [tl.x, tl.y, br.x, br.y]\n {name: 'a_pattern_from', components: 4, type: 'Uint16'},\n {name: 'a_pattern_to', components: 4, type: 'Uint16'},\n {name: 'a_pixel_ratio_from', components: 1, type: 'Uint16'},\n {name: 'a_pixel_ratio_to', components: 1, type: 'Uint16'},\n]);\n","/**\n * JS Implementation of MurmurHash3 (r136) (as of May 20, 2011)\n * \n * @author Gary Court\n * @see http://github.com/garycourt/murmurhash-js\n * @author Austin Appleby\n * @see http://sites.google.com/site/murmurhash/\n * \n * @param {string} key ASCII only\n * @param {number} seed Positive integer only\n * @return {number} 32-bit positive integer hash \n */\n\nfunction murmurhash3_32_gc(key, seed) {\n\tvar remainder, bytes, h1, h1b, c1, c1b, c2, c2b, k1, i;\n\t\n\tremainder = key.length & 3; // key.length % 4\n\tbytes = key.length - remainder;\n\th1 = seed;\n\tc1 = 0xcc9e2d51;\n\tc2 = 0x1b873593;\n\ti = 0;\n\t\n\twhile (i < bytes) {\n\t \tk1 = \n\t \t ((key.charCodeAt(i) & 0xff)) |\n\t \t ((key.charCodeAt(++i) & 0xff) << 8) |\n\t \t ((key.charCodeAt(++i) & 0xff) << 16) |\n\t \t ((key.charCodeAt(++i) & 0xff) << 24);\n\t\t++i;\n\t\t\n\t\tk1 = ((((k1 & 0xffff) * c1) + ((((k1 >>> 16) * c1) & 0xffff) << 16))) & 0xffffffff;\n\t\tk1 = (k1 << 15) | (k1 >>> 17);\n\t\tk1 = ((((k1 & 0xffff) * c2) + ((((k1 >>> 16) * c2) & 0xffff) << 16))) & 0xffffffff;\n\n\t\th1 ^= k1;\n h1 = (h1 << 13) | (h1 >>> 19);\n\t\th1b = ((((h1 & 0xffff) * 5) + ((((h1 >>> 16) * 5) & 0xffff) << 16))) & 0xffffffff;\n\t\th1 = (((h1b & 0xffff) + 0x6b64) + ((((h1b >>> 16) + 0xe654) & 0xffff) << 16));\n\t}\n\t\n\tk1 = 0;\n\t\n\tswitch (remainder) {\n\t\tcase 3: k1 ^= (key.charCodeAt(i + 2) & 0xff) << 16;\n\t\tcase 2: k1 ^= (key.charCodeAt(i + 1) & 0xff) << 8;\n\t\tcase 1: k1 ^= (key.charCodeAt(i) & 0xff);\n\t\t\n\t\tk1 = (((k1 & 0xffff) * c1) + ((((k1 >>> 16) * c1) & 0xffff) << 16)) & 0xffffffff;\n\t\tk1 = (k1 << 15) | (k1 >>> 17);\n\t\tk1 = (((k1 & 0xffff) * c2) + ((((k1 >>> 16) * c2) & 0xffff) << 16)) & 0xffffffff;\n\t\th1 ^= k1;\n\t}\n\t\n\th1 ^= key.length;\n\n\th1 ^= h1 >>> 16;\n\th1 = (((h1 & 0xffff) * 0x85ebca6b) + ((((h1 >>> 16) * 0x85ebca6b) & 0xffff) << 16)) & 0xffffffff;\n\th1 ^= h1 >>> 13;\n\th1 = ((((h1 & 0xffff) * 0xc2b2ae35) + ((((h1 >>> 16) * 0xc2b2ae35) & 0xffff) << 16))) & 0xffffffff;\n\th1 ^= h1 >>> 16;\n\n\treturn h1 >>> 0;\n}\n\nif(typeof module !== \"undefined\") {\n module.exports = murmurhash3_32_gc\n}","/**\n * JS Implementation of MurmurHash2\n * \n * @author Gary Court\n * @see http://github.com/garycourt/murmurhash-js\n * @author Austin Appleby\n * @see http://sites.google.com/site/murmurhash/\n * \n * @param {string} str ASCII only\n * @param {number} seed Positive integer only\n * @return {number} 32-bit positive integer hash\n */\n\nfunction murmurhash2_32_gc(str, seed) {\n var\n l = str.length,\n h = seed ^ l,\n i = 0,\n k;\n \n while (l >= 4) {\n \tk = \n \t ((str.charCodeAt(i) & 0xff)) |\n \t ((str.charCodeAt(++i) & 0xff) << 8) |\n \t ((str.charCodeAt(++i) & 0xff) << 16) |\n \t ((str.charCodeAt(++i) & 0xff) << 24);\n \n k = (((k & 0xffff) * 0x5bd1e995) + ((((k >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n k ^= k >>> 24;\n k = (((k & 0xffff) * 0x5bd1e995) + ((((k >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n\n\th = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16)) ^ k;\n\n l -= 4;\n ++i;\n }\n \n switch (l) {\n case 3: h ^= (str.charCodeAt(i + 2) & 0xff) << 16;\n case 2: h ^= (str.charCodeAt(i + 1) & 0xff) << 8;\n case 1: h ^= (str.charCodeAt(i) & 0xff);\n h = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n }\n\n h ^= h >>> 13;\n h = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n h ^= h >>> 15;\n\n return h >>> 0;\n}\n\nif(typeof module !== undefined) {\n module.exports = murmurhash2_32_gc\n}\n","var murmur3 = require(\"./murmurhash3_gc.js\")\nvar murmur2 = require(\"./murmurhash2_gc.js\")\n\nmodule.exports = murmur3\nmodule.exports.murmur3 = murmur3\nmodule.exports.murmur2 = murmur2\n","import murmur3 from 'murmurhash-js';\nimport {register} from '../util/web_worker_transfer';\n\ntype SerializedFeaturePositionMap = {\n ids: Float64Array;\n positions: Uint32Array;\n};\n\ntype FeaturePosition = {\n index: number;\n start: number;\n end: number;\n};\n\n// A transferable data structure that maps feature ids to their indices and buffer offsets\nexport class FeaturePositionMap {\n ids: Array;\n positions: Array;\n indexed: boolean;\n\n constructor() {\n this.ids = [];\n this.positions = [];\n this.indexed = false;\n }\n\n add(id: unknown, index: number, start: number, end: number) {\n this.ids.push(getNumericId(id));\n this.positions.push(index, start, end);\n }\n\n getPositions(id: unknown): Array {\n if (!this.indexed) throw new Error('Trying to get index, but feature positions are not indexed');\n\n const intId = getNumericId(id);\n\n // binary search for the first occurrence of id in this.ids;\n // relies on ids/positions being sorted by id, which happens in serialization\n let i = 0;\n let j = this.ids.length - 1;\n while (i < j) {\n const m = (i + j) >> 1;\n if (this.ids[m] >= intId) {\n j = m;\n } else {\n i = m + 1;\n }\n }\n const positions = [];\n while (this.ids[i] === intId) {\n const index = this.positions[3 * i];\n const start = this.positions[3 * i + 1];\n const end = this.positions[3 * i + 2];\n positions.push({index, start, end});\n i++;\n }\n return positions;\n }\n\n static serialize(map: FeaturePositionMap, transferables: Array): SerializedFeaturePositionMap {\n const ids = new Float64Array(map.ids);\n const positions = new Uint32Array(map.positions);\n\n sort(ids, positions, 0, ids.length - 1);\n\n if (transferables) {\n transferables.push(ids.buffer, positions.buffer);\n }\n\n return {ids, positions};\n }\n\n static deserialize(obj: SerializedFeaturePositionMap): FeaturePositionMap {\n const map = new FeaturePositionMap();\n // after transferring, we only use these arrays statically (no pushes),\n // so TypedArray vs Array distinction that flow points out doesn't matter\n map.ids = (obj.ids as any);\n map.positions = (obj.positions as any);\n map.indexed = true;\n return map;\n }\n}\n\nfunction getNumericId(value: unknown) {\n const numValue = +value;\n if (!isNaN(numValue) && numValue <= Number.MAX_SAFE_INTEGER) {\n return numValue;\n }\n return murmur3(String(value));\n}\n\n// custom quicksort that sorts ids, indices and offsets together (by ids)\n// uses Hoare partitioning & manual tail call optimization to avoid worst case scenarios\nfunction sort(ids, positions, left, right) {\n while (left < right) {\n const pivot = ids[(left + right) >> 1];\n let i = left - 1;\n let j = right + 1;\n\n while (true) {\n do i++; while (ids[i] < pivot);\n do j--; while (ids[j] > pivot);\n if (i >= j) break;\n swap(ids, i, j);\n swap(positions, 3 * i, 3 * j);\n swap(positions, 3 * i + 1, 3 * j + 1);\n swap(positions, 3 * i + 2, 3 * j + 2);\n }\n\n if (j - left < right - j) {\n sort(ids, positions, left, j);\n left = j + 1;\n } else {\n sort(ids, positions, j + 1, right);\n right = j;\n }\n }\n}\n\nfunction swap(arr, i, j) {\n const tmp = arr[i];\n arr[i] = arr[j];\n arr[j] = tmp;\n}\n\nregister('FeaturePositionMap', FeaturePositionMap);\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {Context} from '../gl/context';\nimport {mat4, vec2, vec3, vec4} from 'gl-matrix';\n\ntype $ObjMap any> = {\n [K in keyof T]: F extends (v: T[K]) => infer R ? R : never;\n};\n\nexport type UniformValues = $ObjMap(u: Uniform) => V>;\nexport type UniformLocations = {[_: string]: WebGLUniformLocation};\n\n/**\n * @internal\n * A base uniform abstract class\n */\nabstract class Uniform {\n gl: WebGLRenderingContext|WebGL2RenderingContext;\n location: WebGLUniformLocation;\n current: T;\n\n constructor(context: Context, location: WebGLUniformLocation) {\n this.gl = context.gl;\n this.location = location;\n }\n\n abstract set(v: T): void;\n}\n\nclass Uniform1i extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = 0;\n }\n\n set(v: number): void {\n if (this.current !== v) {\n this.current = v;\n this.gl.uniform1i(this.location, v);\n }\n }\n}\n\nclass Uniform1f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = 0;\n }\n\n set(v: number): void {\n if (this.current !== v) {\n this.current = v;\n this.gl.uniform1f(this.location, v);\n }\n }\n}\n\nclass Uniform2f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = [0, 0];\n }\n\n set(v: vec2): void {\n if (v[0] !== this.current[0] || v[1] !== this.current[1]) {\n this.current = v;\n this.gl.uniform2f(this.location, v[0], v[1]);\n }\n }\n}\n\nclass Uniform3f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = [0, 0, 0];\n }\n\n set(v: vec3): void {\n if (v[0] !== this.current[0] || v[1] !== this.current[1] || v[2] !== this.current[2]) {\n this.current = v;\n this.gl.uniform3f(this.location, v[0], v[1], v[2]);\n }\n }\n}\n\nclass Uniform4f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = [0, 0, 0, 0];\n }\n\n set(v: vec4): void {\n if (v[0] !== this.current[0] || v[1] !== this.current[1] ||\n v[2] !== this.current[2] || v[3] !== this.current[3]) {\n this.current = v;\n this.gl.uniform4f(this.location, v[0], v[1], v[2], v[3]);\n }\n }\n}\n\nclass UniformColor extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = Color.transparent;\n }\n\n set(v: Color): void {\n if (v.r !== this.current.r || v.g !== this.current.g ||\n v.b !== this.current.b || v.a !== this.current.a) {\n this.current = v;\n this.gl.uniform4f(this.location, v.r, v.g, v.b, v.a);\n }\n }\n}\n\nconst emptyMat4 = new Float32Array(16) as mat4;\nclass UniformMatrix4f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = emptyMat4;\n }\n\n set(v: mat4): void {\n // The vast majority of matrix comparisons that will trip this set\n // happen at i=12 or i=0, so we check those first to avoid lots of\n // unnecessary iteration:\n if (v[12] !== this.current[12] || v[0] !== this.current[0]) {\n this.current = v;\n this.gl.uniformMatrix4fv(this.location, false, v);\n return;\n }\n for (let i = 1; i < 16; i++) {\n if (v[i] !== this.current[i]) {\n this.current = v;\n this.gl.uniformMatrix4fv(this.location, false, v);\n break;\n }\n }\n }\n}\n\nexport {\n Uniform,\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f,\n Uniform4f,\n UniformColor,\n UniformMatrix4f\n};\n\n/**\n * @internal\n * A uniform bindings\n */\nexport type UniformBindings = {[_: string]: Uniform};\n","import {packUint8ToFloat} from '../shaders/encode_attribute';\nimport {Color, supportsPropertyExpression} from '@maplibre/maplibre-gl-style-spec';\nimport {register} from '../util/web_worker_transfer';\nimport {PossiblyEvaluatedPropertyValue} from '../style/properties';\nimport {StructArrayLayout1f4, StructArrayLayout2f8, StructArrayLayout4f16, PatternLayoutArray} from './array_types.g';\nimport {clamp} from '../util/util';\nimport {patternAttributes} from './bucket/pattern_attributes';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {FeaturePositionMap} from './feature_position_map';\nimport {Uniform, Uniform1f, UniformColor, Uniform4f} from '../render/uniform_binding';\n\nimport type {UniformLocations} from '../render/uniform_binding';\n\nimport type {CanonicalTileID} from '../source/tile_id';\nimport type {Context} from '../gl/context';\nimport type {TypedStyleLayer} from '../style/style_layer/typed_style_layer';\nimport type {CrossfadeParameters} from '../style/evaluation_parameters';\nimport type {StructArray, StructArrayMember} from '../util/struct_array';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {ImagePosition} from '../render/image_atlas';\nimport type {\n Feature,\n FeatureState,\n GlobalProperties,\n SourceExpression,\n CompositeExpression,\n FormattedSection\n} from '@maplibre/maplibre-gl-style-spec';\nimport type {FeatureStates} from '../source/source_state';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nexport type BinderUniform = {\n name: string;\n property: string;\n binding: Uniform;\n};\n\nfunction packColor(color: Color): [number, number] {\n return [\n packUint8ToFloat(255 * color.r, 255 * color.g),\n packUint8ToFloat(255 * color.b, 255 * color.a)\n ];\n}\n\n/**\n * `Binder` is the interface definition for the strategies for constructing,\n * uploading, and binding paint property data as GLSL attributes. Most style-\n * spec properties have a 1:1 relationship to shader attribute/uniforms, but\n * some require multiple values per feature to be passed to the GPU, and in\n * those cases we bind multiple attributes/uniforms.\n *\n * It has three implementations, one for each of the three strategies we use:\n *\n * * For _constant_ properties -- those whose value is a constant, or the constant\n * result of evaluating a camera expression at a particular camera position -- we\n * don't need a vertex attribute buffer, and instead use a uniform.\n * * For data expressions, we use a vertex buffer with a single attribute value,\n * the evaluated result of the source function for the given feature.\n * * For composite expressions, we use a vertex buffer with two attributes: min and\n * max values covering the range of zooms at which we expect the tile to be\n * displayed. These values are calculated by evaluating the composite expression for\n * the given feature at strategically chosen zoom levels. In addition to this\n * attribute data, we also use a uniform value which the shader uses to interpolate\n * between the min and max value at the final displayed zoom level. The use of a\n * uniform allows us to cheaply update the value on every frame.\n *\n * Note that the shader source varies depending on whether we're using a uniform or\n * attribute. We dynamically compile shaders at runtime to accommodate this.\n */\ninterface AttributeBinder {\n populatePaintArray(\n length: number,\n feature: Feature,\n imagePositions: {[_: string]: ImagePosition},\n canonical?: CanonicalTileID,\n formattedSection?: FormattedSection\n ): void;\n updatePaintArray(\n start: number,\n length: number,\n feature: Feature,\n featureState: FeatureState,\n imagePositions: {[_: string]: ImagePosition}\n ): void;\n upload(a: Context): void;\n destroy(): void;\n}\n\ninterface UniformBinder {\n uniformNames: Array;\n setUniform(\n uniform: Uniform,\n globals: GlobalProperties,\n currentValue: PossiblyEvaluatedPropertyValue,\n uniformName: string\n ): void;\n getBinding(context: Context, location: WebGLUniformLocation, name: string): Partial>;\n}\n\nclass ConstantBinder implements UniformBinder {\n value: unknown;\n type: string;\n uniformNames: Array;\n\n constructor(value: unknown, names: Array, type: string) {\n this.value = value;\n this.uniformNames = names.map(name => `u_${name}`);\n this.type = type;\n }\n\n setUniform(\n uniform: Uniform,\n globals: GlobalProperties,\n currentValue: PossiblyEvaluatedPropertyValue\n ): void {\n uniform.set(currentValue.constantOr(this.value));\n }\n\n getBinding(context: Context, location: WebGLUniformLocation, _: string): Partial> {\n return (this.type === 'color') ?\n new UniformColor(context, location) :\n new Uniform1f(context, location);\n }\n}\n\nclass CrossFadedConstantBinder implements UniformBinder {\n uniformNames: Array;\n patternFrom: Array;\n patternTo: Array;\n pixelRatioFrom: number;\n pixelRatioTo: number;\n\n constructor(value: unknown, names: Array) {\n this.uniformNames = names.map(name => `u_${name}`);\n this.patternFrom = null;\n this.patternTo = null;\n this.pixelRatioFrom = 1.0;\n this.pixelRatioTo = 1.0;\n }\n\n setConstantPatternPositions(posTo: ImagePosition, posFrom: ImagePosition) {\n this.pixelRatioFrom = posFrom.pixelRatio;\n this.pixelRatioTo = posTo.pixelRatio;\n this.patternFrom = posFrom.tlbr;\n this.patternTo = posTo.tlbr;\n }\n\n setUniform(uniform: Uniform, globals: GlobalProperties, currentValue: PossiblyEvaluatedPropertyValue, uniformName: string) {\n const pos =\n uniformName === 'u_pattern_to' ? this.patternTo :\n uniformName === 'u_pattern_from' ? this.patternFrom :\n uniformName === 'u_pixel_ratio_to' ? this.pixelRatioTo :\n uniformName === 'u_pixel_ratio_from' ? this.pixelRatioFrom : null;\n if (pos) uniform.set(pos);\n }\n\n getBinding(context: Context, location: WebGLUniformLocation, name: string): Partial> {\n return name.substr(0, 9) === 'u_pattern' ?\n new Uniform4f(context, location) :\n new Uniform1f(context, location);\n }\n}\n\nclass SourceExpressionBinder implements AttributeBinder {\n expression: SourceExpression;\n type: string;\n maxValue: number;\n\n paintVertexArray: StructArray;\n paintVertexAttributes: Array;\n paintVertexBuffer: VertexBuffer;\n\n constructor(expression: SourceExpression, names: Array, type: string, PaintVertexArray: {\n new (...args: any): StructArray;\n }) {\n this.expression = expression;\n this.type = type;\n this.maxValue = 0;\n this.paintVertexAttributes = names.map((name) => ({\n name: `a_${name}`,\n type: 'Float32',\n components: type === 'color' ? 2 : 1,\n offset: 0\n }));\n this.paintVertexArray = new PaintVertexArray();\n }\n\n populatePaintArray(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) {\n const start = this.paintVertexArray.length;\n const value = this.expression.evaluate(new EvaluationParameters(0), feature, {}, canonical, [], formattedSection);\n this.paintVertexArray.resize(newLength);\n this._setPaintValue(start, newLength, value);\n }\n\n updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState) {\n const value = this.expression.evaluate({zoom: 0}, feature, featureState);\n this._setPaintValue(start, end, value);\n }\n\n _setPaintValue(start, end, value) {\n if (this.type === 'color') {\n const color = packColor(value);\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, color[0], color[1]);\n }\n } else {\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, value);\n }\n this.maxValue = Math.max(this.maxValue, Math.abs(value));\n }\n }\n\n upload(context: Context) {\n if (this.paintVertexArray && this.paintVertexArray.arrayBuffer) {\n if (this.paintVertexBuffer && this.paintVertexBuffer.buffer) {\n this.paintVertexBuffer.updateData(this.paintVertexArray);\n } else {\n this.paintVertexBuffer = context.createVertexBuffer(this.paintVertexArray, this.paintVertexAttributes, this.expression.isStateDependent);\n }\n }\n }\n\n destroy() {\n if (this.paintVertexBuffer) {\n this.paintVertexBuffer.destroy();\n }\n }\n}\n\nclass CompositeExpressionBinder implements AttributeBinder, UniformBinder {\n expression: CompositeExpression;\n uniformNames: Array;\n type: string;\n useIntegerZoom: boolean;\n zoom: number;\n maxValue: number;\n\n paintVertexArray: StructArray;\n paintVertexAttributes: Array;\n paintVertexBuffer: VertexBuffer;\n\n constructor(expression: CompositeExpression, names: Array, type: string, useIntegerZoom: boolean, zoom: number, PaintVertexArray: {\n new (...args: any): StructArray;\n }) {\n this.expression = expression;\n this.uniformNames = names.map(name => `u_${name}_t`);\n this.type = type;\n this.useIntegerZoom = useIntegerZoom;\n this.zoom = zoom;\n this.maxValue = 0;\n this.paintVertexAttributes = names.map((name) => ({\n name: `a_${name}`,\n type: 'Float32',\n components: type === 'color' ? 4 : 2,\n offset: 0\n }));\n this.paintVertexArray = new PaintVertexArray();\n }\n\n populatePaintArray(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) {\n const min = this.expression.evaluate(new EvaluationParameters(this.zoom), feature, {}, canonical, [], formattedSection);\n const max = this.expression.evaluate(new EvaluationParameters(this.zoom + 1), feature, {}, canonical, [], formattedSection);\n const start = this.paintVertexArray.length;\n this.paintVertexArray.resize(newLength);\n this._setPaintValue(start, newLength, min, max);\n }\n\n updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState) {\n const min = this.expression.evaluate({zoom: this.zoom}, feature, featureState);\n const max = this.expression.evaluate({zoom: this.zoom + 1}, feature, featureState);\n this._setPaintValue(start, end, min, max);\n }\n\n _setPaintValue(start, end, min, max) {\n if (this.type === 'color') {\n const minColor = packColor(min);\n const maxColor = packColor(max);\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, minColor[0], minColor[1], maxColor[0], maxColor[1]);\n }\n } else {\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, min, max);\n }\n this.maxValue = Math.max(this.maxValue, Math.abs(min), Math.abs(max));\n }\n }\n\n upload(context: Context) {\n if (this.paintVertexArray && this.paintVertexArray.arrayBuffer) {\n if (this.paintVertexBuffer && this.paintVertexBuffer.buffer) {\n this.paintVertexBuffer.updateData(this.paintVertexArray);\n } else {\n this.paintVertexBuffer = context.createVertexBuffer(this.paintVertexArray, this.paintVertexAttributes, this.expression.isStateDependent);\n }\n }\n }\n\n destroy() {\n if (this.paintVertexBuffer) {\n this.paintVertexBuffer.destroy();\n }\n }\n\n setUniform(uniform: Uniform, globals: GlobalProperties): void {\n const currentZoom = this.useIntegerZoom ? Math.floor(globals.zoom) : globals.zoom;\n const factor = clamp(this.expression.interpolationFactor(currentZoom, this.zoom, this.zoom + 1), 0, 1);\n uniform.set(factor);\n }\n\n getBinding(context: Context, location: WebGLUniformLocation, _: string): Uniform1f {\n return new Uniform1f(context, location);\n }\n}\n\nclass CrossFadedCompositeBinder implements AttributeBinder {\n expression: CompositeExpression;\n type: string;\n useIntegerZoom: boolean;\n zoom: number;\n layerId: string;\n\n zoomInPaintVertexArray: StructArray;\n zoomOutPaintVertexArray: StructArray;\n zoomInPaintVertexBuffer: VertexBuffer;\n zoomOutPaintVertexBuffer: VertexBuffer;\n paintVertexAttributes: Array;\n\n constructor(expression: CompositeExpression, type: string, useIntegerZoom: boolean, zoom: number, PaintVertexArray: {\n new (...args: any): StructArray;\n }, layerId: string) {\n this.expression = expression;\n this.type = type;\n this.useIntegerZoom = useIntegerZoom;\n this.zoom = zoom;\n this.layerId = layerId;\n\n this.zoomInPaintVertexArray = new PaintVertexArray();\n this.zoomOutPaintVertexArray = new PaintVertexArray();\n }\n\n populatePaintArray(length: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}) {\n const start = this.zoomInPaintVertexArray.length;\n this.zoomInPaintVertexArray.resize(length);\n this.zoomOutPaintVertexArray.resize(length);\n this._setPaintValues(start, length, feature.patterns && feature.patterns[this.layerId], imagePositions);\n }\n\n updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState, imagePositions: {[_: string]: ImagePosition}) {\n this._setPaintValues(start, end, feature.patterns && feature.patterns[this.layerId], imagePositions);\n }\n\n _setPaintValues(start, end, patterns, positions) {\n if (!positions || !patterns) return;\n\n const {min, mid, max} = patterns;\n const imageMin = positions[min];\n const imageMid = positions[mid];\n const imageMax = positions[max];\n if (!imageMin || !imageMid || !imageMax) return;\n\n // We populate two paint arrays because, for cross-faded properties, we don't know which direction\n // we're cross-fading to at layout time. In order to keep vertex attributes to a minimum and not pass\n // unnecessary vertex data to the shaders, we determine which to upload at draw time.\n for (let i = start; i < end; i++) {\n this.zoomInPaintVertexArray.emplace(i,\n imageMid.tl[0], imageMid.tl[1], imageMid.br[0], imageMid.br[1],\n imageMin.tl[0], imageMin.tl[1], imageMin.br[0], imageMin.br[1],\n imageMid.pixelRatio,\n imageMin.pixelRatio,\n );\n this.zoomOutPaintVertexArray.emplace(i,\n imageMid.tl[0], imageMid.tl[1], imageMid.br[0], imageMid.br[1],\n imageMax.tl[0], imageMax.tl[1], imageMax.br[0], imageMax.br[1],\n imageMid.pixelRatio,\n imageMax.pixelRatio,\n );\n }\n }\n\n upload(context: Context) {\n if (this.zoomInPaintVertexArray && this.zoomInPaintVertexArray.arrayBuffer && this.zoomOutPaintVertexArray && this.zoomOutPaintVertexArray.arrayBuffer) {\n this.zoomInPaintVertexBuffer = context.createVertexBuffer(this.zoomInPaintVertexArray, patternAttributes.members, this.expression.isStateDependent);\n this.zoomOutPaintVertexBuffer = context.createVertexBuffer(this.zoomOutPaintVertexArray, patternAttributes.members, this.expression.isStateDependent);\n }\n }\n\n destroy() {\n if (this.zoomOutPaintVertexBuffer) this.zoomOutPaintVertexBuffer.destroy();\n if (this.zoomInPaintVertexBuffer) this.zoomInPaintVertexBuffer.destroy();\n }\n}\n\n/**\n * @internal\n * ProgramConfiguration contains the logic for binding style layer properties and tile\n * layer feature data into GL program uniforms and vertex attributes.\n *\n * Non-data-driven property values are bound to shader uniforms. Data-driven property\n * values are bound to vertex attributes. In order to support a uniform GLSL syntax over\n * both, [Mapbox GL Shaders](https://github.com/mapbox/mapbox-gl-shaders) defines a `#pragma`\n * abstraction, which ProgramConfiguration is responsible for implementing. At runtime,\n * it examines the attributes of a particular layer, combines this with fixed knowledge\n * about how layers of the particular type are implemented, and determines which uniforms\n * and vertex attributes will be required. It can then substitute the appropriate text\n * into the shader source code, create and link a program, and bind the uniforms and\n * vertex attributes in preparation for drawing.\n *\n * When a vector tile is parsed, this same configuration information is used to\n * populate the attribute buffers needed for data-driven styling using the zoom\n * level and feature property data.\n */\nexport class ProgramConfiguration {\n binders: {[_: string]: AttributeBinder | UniformBinder};\n cacheKey: string;\n\n _buffers: Array;\n\n constructor(layer: TypedStyleLayer, zoom: number, filterProperties: (_: string) => boolean) {\n this.binders = {};\n this._buffers = [];\n\n const keys = [];\n\n for (const property in layer.paint._values) {\n if (!filterProperties(property)) continue;\n const value = (layer.paint as any).get(property);\n if (!(value instanceof PossiblyEvaluatedPropertyValue) || !supportsPropertyExpression(value.property.specification)) {\n continue;\n }\n const names = paintAttributeNames(property, layer.type);\n const expression = value.value;\n const type = value.property.specification.type;\n const useIntegerZoom = (value.property as any).useIntegerZoom;\n const propType = value.property.specification['property-type'];\n const isCrossFaded = propType === 'cross-faded' || propType === 'cross-faded-data-driven';\n\n if (expression.kind === 'constant') {\n this.binders[property] = isCrossFaded ?\n new CrossFadedConstantBinder(expression.value, names) :\n new ConstantBinder(expression.value, names, type);\n keys.push(`/u_${property}`);\n\n } else if (expression.kind === 'source' || isCrossFaded) {\n const StructArrayLayout = layoutType(property, type, 'source');\n this.binders[property] = isCrossFaded ?\n new CrossFadedCompositeBinder(expression as CompositeExpression, type, useIntegerZoom, zoom, StructArrayLayout, layer.id) :\n new SourceExpressionBinder(expression as SourceExpression, names, type, StructArrayLayout);\n keys.push(`/a_${property}`);\n\n } else {\n const StructArrayLayout = layoutType(property, type, 'composite');\n this.binders[property] = new CompositeExpressionBinder(expression, names, type, useIntegerZoom, zoom, StructArrayLayout);\n keys.push(`/z_${property}`);\n }\n }\n\n this.cacheKey = keys.sort().join('');\n }\n\n getMaxValue(property: string): number {\n const binder = this.binders[property];\n return binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder ? binder.maxValue : 0;\n }\n\n populatePaintArrays(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder)\n (binder as AttributeBinder).populatePaintArray(newLength, feature, imagePositions, canonical, formattedSection);\n }\n }\n setConstantPatternPositions(posTo: ImagePosition, posFrom: ImagePosition) {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof CrossFadedConstantBinder)\n binder.setConstantPatternPositions(posTo, posFrom);\n }\n }\n\n updatePaintArrays(\n featureStates: FeatureStates,\n featureMap: FeaturePositionMap,\n vtLayer: VectorTileLayer,\n layer: TypedStyleLayer,\n imagePositions: {[_: string]: ImagePosition}\n ): boolean {\n let dirty: boolean = false;\n for (const id in featureStates) {\n const positions = featureMap.getPositions(id);\n\n for (const pos of positions) {\n const feature = vtLayer.feature(pos.index);\n\n for (const property in this.binders) {\n const binder = this.binders[property];\n if ((binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder ||\n binder instanceof CrossFadedCompositeBinder) && (binder as any).expression.isStateDependent === true) {\n //AHM: Remove after https://github.com/mapbox/mapbox-gl-js/issues/6255\n const value = (layer.paint as any).get(property);\n (binder as any).expression = value.value;\n (binder as AttributeBinder).updatePaintArray(pos.start, pos.end, feature, featureStates[id], imagePositions);\n dirty = true;\n }\n }\n }\n }\n return dirty;\n }\n\n defines(): Array {\n const result = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder) {\n result.push(...binder.uniformNames.map(name => `#define HAS_UNIFORM_${name}`));\n }\n }\n return result;\n }\n\n getBinderAttributes(): Array {\n const result = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder) {\n for (let i = 0; i < binder.paintVertexAttributes.length; i++) {\n result.push(binder.paintVertexAttributes[i].name);\n }\n } else if (binder instanceof CrossFadedCompositeBinder) {\n for (let i = 0; i < patternAttributes.members.length; i++) {\n result.push(patternAttributes.members[i].name);\n }\n }\n }\n return result;\n }\n\n getBinderUniforms(): Array {\n const uniforms = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder || binder instanceof CompositeExpressionBinder) {\n for (const uniformName of binder.uniformNames) {\n uniforms.push(uniformName);\n }\n }\n }\n return uniforms;\n }\n\n getPaintVertexBuffers(): Array {\n return this._buffers;\n }\n\n getUniforms(context: Context, locations: UniformLocations): Array {\n const uniforms = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder || binder instanceof CompositeExpressionBinder) {\n for (const name of binder.uniformNames) {\n if (locations[name]) {\n const binding = binder.getBinding(context, locations[name], name);\n uniforms.push({name, property, binding});\n }\n }\n }\n }\n return uniforms;\n }\n\n setUniforms(\n context: Context,\n binderUniforms: Array,\n properties: any,\n globals: GlobalProperties\n ) {\n // Uniform state bindings are owned by the Program, but we set them\n // from within the ProgramConfiguraton's binder members.\n for (const {name, property, binding} of binderUniforms) {\n (this.binders[property] as any).setUniform(binding, globals, properties.get(property), name);\n }\n }\n\n updatePaintBuffers(crossfade?: CrossfadeParameters) {\n this._buffers = [];\n\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (crossfade && binder instanceof CrossFadedCompositeBinder) {\n const patternVertexBuffer = crossfade.fromScale === 2 ? binder.zoomInPaintVertexBuffer : binder.zoomOutPaintVertexBuffer;\n if (patternVertexBuffer) this._buffers.push(patternVertexBuffer);\n\n } else if ((binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder) && binder.paintVertexBuffer) {\n this._buffers.push(binder.paintVertexBuffer);\n }\n }\n }\n\n upload(context: Context) {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder)\n binder.upload(context);\n }\n this.updatePaintBuffers();\n }\n\n destroy() {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder)\n binder.destroy();\n }\n }\n}\n\nexport class ProgramConfigurationSet {\n programConfigurations: {[_: string]: ProgramConfiguration};\n needsUpload: boolean;\n _featureMap: FeaturePositionMap;\n _bufferOffset: number;\n\n constructor(layers: ReadonlyArray, zoom: number, filterProperties: (_: string) => boolean = () => true) {\n this.programConfigurations = {};\n for (const layer of layers) {\n this.programConfigurations[layer.id] = new ProgramConfiguration(layer, zoom, filterProperties);\n }\n this.needsUpload = false;\n this._featureMap = new FeaturePositionMap();\n this._bufferOffset = 0;\n }\n\n populatePaintArrays(length: number, feature: Feature, index: number, imagePositions: {[_: string]: ImagePosition}, canonical: CanonicalTileID, formattedSection?: FormattedSection) {\n for (const key in this.programConfigurations) {\n this.programConfigurations[key].populatePaintArrays(length, feature, imagePositions, canonical, formattedSection);\n }\n\n if (feature.id !== undefined) {\n this._featureMap.add(feature.id, index, this._bufferOffset, length);\n }\n this._bufferOffset = length;\n\n this.needsUpload = true;\n }\n\n updatePaintArrays(featureStates: FeatureStates, vtLayer: VectorTileLayer, layers: ReadonlyArray, imagePositions: {[_: string]: ImagePosition}) {\n for (const layer of layers) {\n this.needsUpload = this.programConfigurations[layer.id].updatePaintArrays(featureStates, this._featureMap, vtLayer, layer, imagePositions) || this.needsUpload;\n }\n }\n\n get(layerId: string) {\n return this.programConfigurations[layerId];\n }\n\n upload(context: Context) {\n if (!this.needsUpload) return;\n for (const layerId in this.programConfigurations) {\n this.programConfigurations[layerId].upload(context);\n }\n this.needsUpload = false;\n }\n\n destroy() {\n for (const layerId in this.programConfigurations) {\n this.programConfigurations[layerId].destroy();\n }\n }\n}\n\nfunction paintAttributeNames(property, type) {\n const attributeNameExceptions = {\n 'text-opacity': ['opacity'],\n 'icon-opacity': ['opacity'],\n 'text-color': ['fill_color'],\n 'icon-color': ['fill_color'],\n 'text-halo-color': ['halo_color'],\n 'icon-halo-color': ['halo_color'],\n 'text-halo-blur': ['halo_blur'],\n 'icon-halo-blur': ['halo_blur'],\n 'text-halo-width': ['halo_width'],\n 'icon-halo-width': ['halo_width'],\n 'line-gap-width': ['gapwidth'],\n 'line-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'],\n 'fill-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'],\n 'fill-extrusion-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'],\n };\n\n return attributeNameExceptions[property] || [property.replace(`${type}-`, '').replace(/-/g, '_')];\n}\n\nfunction getLayoutException(property) {\n const propertyExceptions = {\n 'line-pattern': {\n 'source': PatternLayoutArray,\n 'composite': PatternLayoutArray\n },\n 'fill-pattern': {\n 'source': PatternLayoutArray,\n 'composite': PatternLayoutArray\n },\n 'fill-extrusion-pattern': {\n 'source': PatternLayoutArray,\n 'composite': PatternLayoutArray\n }\n };\n\n return propertyExceptions[property];\n}\n\nfunction layoutType(property, type, binderType) {\n const defaultLayouts = {\n 'color': {\n 'source': StructArrayLayout2f8,\n 'composite': StructArrayLayout4f16\n },\n 'number': {\n 'source': StructArrayLayout1f4,\n 'composite': StructArrayLayout2f8\n }\n };\n\n const layoutException = getLayoutException(property);\n return layoutException && layoutException[binderType] || defaultLayouts[type][binderType];\n}\n\nregister('ConstantBinder', ConstantBinder);\nregister('CrossFadedConstantBinder', CrossFadedConstantBinder);\nregister('SourceExpressionBinder', SourceExpressionBinder);\nregister('CrossFadedCompositeBinder', CrossFadedCompositeBinder);\nregister('CompositeExpressionBinder', CompositeExpressionBinder);\nregister('ProgramConfiguration', ProgramConfiguration, {omit: ['_buffers']});\nregister('ProgramConfigurationSet', ProgramConfigurationSet);\n","/**\n * The maximum value of a coordinate in the internal tile coordinate system. Coordinates of\n * all source features normalized to this extent upon load.\n *\n * The value is a consequence of the following:\n *\n * * Vertex buffer store positions as signed 16 bit integers.\n * * One bit is lost for signedness to support tile buffers.\n * * One bit is lost because the line vertex buffer used to pack 1 bit of other data into the int.\n * * One bit is lost to support features extending past the extent on the right edge of the tile.\n * * This leaves us with 2^13 = 8192\n */\nexport const EXTENT = 8192;\n","import {warnOnce, clamp} from '../util/util';\n\nimport {EXTENT} from './extent';\n\nimport type Point from '@mapbox/point-geometry';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\n// These bounds define the minimum and maximum supported coordinate values.\n// While visible coordinates are within [0, EXTENT], tiles may theoretically\n// contain coordinates within [-Infinity, Infinity]. Our range is limited by the\n// number of bits used to represent the coordinate.\nconst BITS = 15;\nconst MAX = Math.pow(2, BITS - 1) - 1;\nconst MIN = -MAX - 1;\n\n/**\n * Loads a geometry from a VectorTileFeature and scales it to the common extent\n * used internally.\n * @param feature - the vector tile feature to load\n */\nexport function loadGeometry(feature: VectorTileFeature): Array> {\n const scale = EXTENT / feature.extent;\n const geometry = feature.loadGeometry();\n for (let r = 0; r < geometry.length; r++) {\n const ring = geometry[r];\n for (let p = 0; p < ring.length; p++) {\n const point = ring[p];\n // round here because mapbox-gl-native uses integers to represent\n // points and we need to do the same to avoid renering differences.\n const x = Math.round(point.x * scale);\n const y = Math.round(point.y * scale);\n\n point.x = clamp(x, MIN, MAX);\n point.y = clamp(y, MIN, MAX);\n\n if (x < point.x || x > point.x + 1 || y < point.y || y > point.y + 1) {\n // warn when exceeding allowed extent except for the 1-px-off case\n // https://github.com/mapbox/mapbox-gl-js/issues/8992\n warnOnce('Geometry exceeds allowed extent, reduce your vector tile buffer size');\n }\n }\n }\n return geometry;\n}\n","import {loadGeometry} from './load_geometry';\nimport type Point from '@mapbox/point-geometry';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\nimport type {Feature} from '@maplibre/maplibre-gl-style-spec';\n\ntype EvaluationFeature = Feature & { geometry: Array> };\n/**\n * Construct a new feature based on a VectorTileFeature for expression evaluation, the geometry of which\n * will be loaded based on necessity.\n * @param feature - the feature to evaluate\n * @param needGeometry - if set to true this will load the geometry\n */\nexport function toEvaluationFeature(feature: VectorTileFeature, needGeometry: boolean): EvaluationFeature {\n return {type: feature.type,\n id: feature.id,\n properties: feature.properties,\n geometry: needGeometry ? loadGeometry(feature) : []};\n}\n","import {CircleLayoutArray} from '../array_types.g';\n\nimport {members as layoutAttributes} from './circle_attributes';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray} from '../index_array_type';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EXTENT} from '../extent';\nimport {register} from '../../util/web_worker_transfer';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {CircleStyleLayer} from '../../style/style_layer/circle_style_layer';\nimport type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nfunction addCircleVertex(layoutVertexArray, x, y, extrudeX, extrudeY) {\n layoutVertexArray.emplaceBack(\n (x * 2) + ((extrudeX + 1) / 2),\n (y * 2) + ((extrudeY + 1) / 2));\n}\n\n/**\n * @internal\n * Circles are represented by two triangles.\n *\n * Each corner has a pos that is the center of the circle and an extrusion\n * vector that is where it points.\n */\nexport class CircleBucket implements Bucket {\n index: number;\n zoom: number;\n overscaling: number;\n layerIds: Array;\n layers: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n\n layoutVertexArray: CircleLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n uploaded: boolean;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n\n this.layoutVertexArray = new CircleLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.segments = new SegmentVector();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n const styleLayer = this.layers[0];\n const bucketFeatures: BucketFeature[] = [];\n let circleSortKey = null;\n let sortFeaturesByKey = false;\n\n // Heatmap layers are handled in this bucket and have no evaluated properties, so we check our access\n if (styleLayer.type === 'circle') {\n circleSortKey = (styleLayer as CircleStyleLayer).layout.get('circle-sort-key');\n sortFeaturesByKey = !circleSortKey.isConstant();\n }\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const sortKey = sortFeaturesByKey ?\n circleSortKey.evaluate(evaluationFeature, {}, canonical) :\n undefined;\n\n const bucketFeature: BucketFeature = {\n id,\n properties: feature.properties,\n type: feature.type,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n patterns: {},\n sortKey\n };\n\n bucketFeatures.push(bucketFeature);\n\n }\n\n if (sortFeaturesByKey) {\n bucketFeatures.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const bucketFeature of bucketFeatures) {\n const {geometry, index, sourceLayerIndex} = bucketFeature;\n const feature = features[index].feature;\n\n this.addFeature(bucketFeature, geometry, index, canonical);\n options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0;\n }\n\n uploadPending() {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID) {\n for (const ring of geometry) {\n for (const point of ring) {\n const x = point.x;\n const y = point.y;\n\n // Do not include points that are outside the tile boundaries.\n if (x < 0 || x >= EXTENT || y < 0 || y >= EXTENT) continue;\n\n // this geometry will be of the Point type, and we'll derive\n // two triangles from it.\n //\n // ┌─────────┐\n // │ 3 2 │\n // │ │\n // │ 0 1 │\n // └─────────┘\n\n const segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray, feature.sortKey);\n const index = segment.vertexLength;\n\n addCircleVertex(this.layoutVertexArray, x, y, -1, -1);\n addCircleVertex(this.layoutVertexArray, x, y, 1, -1);\n addCircleVertex(this.layoutVertexArray, x, y, 1, 1);\n addCircleVertex(this.layoutVertexArray, x, y, -1, 1);\n\n this.indexArray.emplaceBack(index, index + 1, index + 2);\n this.indexArray.emplaceBack(index, index + 3, index + 2);\n\n segment.vertexLength += 4;\n segment.primitiveLength += 2;\n }\n }\n\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, {}, canonical);\n }\n}\n\nregister('CircleBucket', CircleBucket, {omit: ['layers']});\n","import {isCounterClockwise} from './util';\n\nimport Point from '@mapbox/point-geometry';\n\nexport {polygonIntersectsBufferedPoint, polygonIntersectsMultiPolygon, polygonIntersectsBufferedMultiLine, polygonIntersectsPolygon, distToSegmentSquared, polygonIntersectsBox};\n\ntype Line = Array;\ntype MultiLine = Array;\ntype Ring = Array;\ntype Polygon = Array;\ntype MultiPolygon = Array;\n\nfunction polygonIntersectsPolygon(polygonA: Polygon, polygonB: Polygon) {\n for (let i = 0; i < polygonA.length; i++) {\n if (polygonContainsPoint(polygonB, polygonA[i])) return true;\n }\n\n for (let i = 0; i < polygonB.length; i++) {\n if (polygonContainsPoint(polygonA, polygonB[i])) return true;\n }\n\n if (lineIntersectsLine(polygonA, polygonB)) return true;\n\n return false;\n}\n\nfunction polygonIntersectsBufferedPoint(polygon: Polygon, point: Point, radius: number) {\n if (polygonContainsPoint(polygon, point)) return true;\n if (pointIntersectsBufferedLine(point, polygon, radius)) return true;\n return false;\n}\n\nfunction polygonIntersectsMultiPolygon(polygon: Polygon, multiPolygon: MultiPolygon) {\n\n if (polygon.length === 1) {\n return multiPolygonContainsPoint(multiPolygon, polygon[0]);\n }\n\n for (let m = 0; m < multiPolygon.length; m++) {\n const ring = multiPolygon[m];\n for (let n = 0; n < ring.length; n++) {\n if (polygonContainsPoint(polygon, ring[n])) return true;\n }\n }\n\n for (let i = 0; i < polygon.length; i++) {\n if (multiPolygonContainsPoint(multiPolygon, polygon[i])) return true;\n }\n\n for (let k = 0; k < multiPolygon.length; k++) {\n if (lineIntersectsLine(polygon, multiPolygon[k])) return true;\n }\n\n return false;\n}\n\nfunction polygonIntersectsBufferedMultiLine(polygon: Polygon, multiLine: MultiLine, radius: number) {\n for (let i = 0; i < multiLine.length; i++) {\n const line = multiLine[i];\n\n if (polygon.length >= 3) {\n for (let k = 0; k < line.length; k++) {\n if (polygonContainsPoint(polygon, line[k])) return true;\n }\n }\n\n if (lineIntersectsBufferedLine(polygon, line, radius)) return true;\n }\n return false;\n}\n\nfunction lineIntersectsBufferedLine(lineA: Line, lineB: Line, radius: number) {\n\n if (lineA.length > 1) {\n if (lineIntersectsLine(lineA, lineB)) return true;\n\n // Check whether any point in either line is within radius of the other line\n for (let j = 0; j < lineB.length; j++) {\n if (pointIntersectsBufferedLine(lineB[j], lineA, radius)) return true;\n }\n }\n\n for (let k = 0; k < lineA.length; k++) {\n if (pointIntersectsBufferedLine(lineA[k], lineB, radius)) return true;\n }\n\n return false;\n}\n\nfunction lineIntersectsLine(lineA: Line, lineB: Line) {\n if (lineA.length === 0 || lineB.length === 0) return false;\n for (let i = 0; i < lineA.length - 1; i++) {\n const a0 = lineA[i];\n const a1 = lineA[i + 1];\n for (let j = 0; j < lineB.length - 1; j++) {\n const b0 = lineB[j];\n const b1 = lineB[j + 1];\n if (lineSegmentIntersectsLineSegment(a0, a1, b0, b1)) return true;\n }\n }\n return false;\n}\n\nfunction lineSegmentIntersectsLineSegment(a0: Point, a1: Point, b0: Point, b1: Point) {\n return isCounterClockwise(a0, b0, b1) !== isCounterClockwise(a1, b0, b1) &&\n isCounterClockwise(a0, a1, b0) !== isCounterClockwise(a0, a1, b1);\n}\n\nfunction pointIntersectsBufferedLine(p: Point, line: Line, radius: number) {\n const radiusSquared = radius * radius;\n\n if (line.length === 1) return p.distSqr(line[0]) < radiusSquared;\n\n for (let i = 1; i < line.length; i++) {\n // Find line segments that have a distance <= radius^2 to p\n // In that case, we treat the line as \"containing point p\".\n const v = line[i - 1], w = line[i];\n if (distToSegmentSquared(p, v, w) < radiusSquared) return true;\n }\n return false;\n}\n\n// Code from http://stackoverflow.com/a/1501725/331379.\nfunction distToSegmentSquared(p: Point, v: Point, w: Point) {\n const l2 = v.distSqr(w);\n if (l2 === 0) return p.distSqr(v);\n const t = ((p.x - v.x) * (w.x - v.x) + (p.y - v.y) * (w.y - v.y)) / l2;\n if (t < 0) return p.distSqr(v);\n if (t > 1) return p.distSqr(w);\n return p.distSqr(w.sub(v)._mult(t)._add(v));\n}\n\n// point in polygon ray casting algorithm\nfunction multiPolygonContainsPoint(rings: Array, p: Point) {\n let c = false,\n ring, p1, p2;\n\n for (let k = 0; k < rings.length; k++) {\n ring = rings[k];\n for (let i = 0, j = ring.length - 1; i < ring.length; j = i++) {\n p1 = ring[i];\n p2 = ring[j];\n if (((p1.y > p.y) !== (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) {\n c = !c;\n }\n }\n }\n return c;\n}\n\nfunction polygonContainsPoint(ring: Ring, p: Point) {\n let c = false;\n for (let i = 0, j = ring.length - 1; i < ring.length; j = i++) {\n const p1 = ring[i];\n const p2 = ring[j];\n if (((p1.y > p.y) !== (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) {\n c = !c;\n }\n }\n return c;\n}\n\nfunction polygonIntersectsBox(ring: Ring, boxX1: number, boxY1: number, boxX2: number, boxY2: number) {\n for (const p of ring) {\n if (boxX1 <= p.x &&\n boxY1 <= p.y &&\n boxX2 >= p.x &&\n boxY2 >= p.y) return true;\n }\n\n const corners = [\n new Point(boxX1, boxY1),\n new Point(boxX1, boxY2),\n new Point(boxX2, boxY2),\n new Point(boxX2, boxY1)];\n\n if (ring.length > 2) {\n for (const corner of corners) {\n if (polygonContainsPoint(ring, corner)) return true;\n }\n }\n\n for (let i = 0; i < ring.length - 1; i++) {\n const p1 = ring[i];\n const p2 = ring[i + 1];\n if (edgeIntersectsBox(p1, p2, corners)) return true;\n }\n\n return false;\n}\n\nfunction edgeIntersectsBox(e1: Point, e2: Point, corners: Array) {\n const tl = corners[0];\n const br = corners[2];\n // the edge and box do not intersect in either the x or y dimensions\n if (((e1.x < tl.x) && (e2.x < tl.x)) ||\n ((e1.x > br.x) && (e2.x > br.x)) ||\n ((e1.y < tl.y) && (e2.y < tl.y)) ||\n ((e1.y > br.y) && (e2.y > br.y))) return false;\n\n // check if all corners of the box are on the same side of the edge\n const dir = isCounterClockwise(e1, e2, corners[0]);\n return dir !== isCounterClockwise(e1, e2, corners[1]) ||\n dir !== isCounterClockwise(e1, e2, corners[2]) ||\n dir !== isCounterClockwise(e1, e2, corners[3]);\n}\n","import Point from '@mapbox/point-geometry';\n\nimport type {PossiblyEvaluatedPropertyValue} from './properties';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {CircleBucket} from '../data/bucket/circle_bucket';\nimport type {LineBucket} from '../data/bucket/line_bucket';\n\nexport function getMaximumPaintValue(\n property: string,\n layer: StyleLayer,\n bucket: CircleBucket | LineBucket\n): number {\n const value = ((layer.paint as any).get(property) as PossiblyEvaluatedPropertyValue).value;\n if (value.kind === 'constant') {\n return value.value;\n } else {\n return bucket.programConfigurations.get(layer.id).getMaxValue(property);\n }\n}\n\nexport function translateDistance(translate: [number, number]) {\n return Math.sqrt(translate[0] * translate[0] + translate[1] * translate[1]);\n}\n\nexport function translate(queryGeometry: Array,\n translate: [number, number],\n translateAnchor: 'viewport' | 'map',\n bearing: number,\n pixelsToTileUnits: number) {\n if (!translate[0] && !translate[1]) {\n return queryGeometry;\n }\n const pt = Point.convert(translate)._mult(pixelsToTileUnits);\n\n if (translateAnchor === 'viewport') {\n pt._rotate(-bearing);\n }\n\n const translated = [];\n for (let i = 0; i < queryGeometry.length; i++) {\n const point = queryGeometry[i];\n translated.push(point.sub(pt));\n }\n return translated;\n}\n\nexport function offsetLine(rings: Array>, offset: number) {\n const newRings: Array> = [];\n for (let ringIndex = 0; ringIndex < rings.length; ringIndex++) {\n const ring = rings[ringIndex];\n const newRing: Array = [];\n for (let index = 0; index < ring.length; index++) {\n const a = ring[index - 1];\n const b = ring[index];\n const c = ring[index + 1];\n const aToB = index === 0 ? new Point(0, 0) : b.sub(a)._unit()._perp();\n const bToC = index === ring.length - 1 ? new Point(0, 0) : c.sub(b)._unit()._perp();\n const extrude = aToB._add(bToC)._unit();\n\n const cosHalfAngle = extrude.x * bToC.x + extrude.y * bToC.y;\n if (cosHalfAngle !== 0) {\n extrude._mult(1 / cosHalfAngle);\n }\n\n newRing.push(extrude._mult(offset)._add(b));\n }\n newRings.push(newRing);\n }\n return newRings;\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type CircleLayoutProps = {\n \"circle-sort-key\": DataDrivenProperty,\n};\n\nexport type CircleLayoutPropsPossiblyEvaluated = {\n \"circle-sort-key\": PossiblyEvaluatedPropertyValue,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"circle-sort-key\": new DataDrivenProperty(styleSpec[\"layout_circle\"][\"circle-sort-key\"] as any as StylePropertySpecification),\n});\n\nexport type CirclePaintProps = {\n \"circle-radius\": DataDrivenProperty,\n \"circle-color\": DataDrivenProperty,\n \"circle-blur\": DataDrivenProperty,\n \"circle-opacity\": DataDrivenProperty,\n \"circle-translate\": DataConstantProperty<[number, number]>,\n \"circle-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"circle-pitch-scale\": DataConstantProperty<\"map\" | \"viewport\">,\n \"circle-pitch-alignment\": DataConstantProperty<\"map\" | \"viewport\">,\n \"circle-stroke-width\": DataDrivenProperty,\n \"circle-stroke-color\": DataDrivenProperty,\n \"circle-stroke-opacity\": DataDrivenProperty,\n};\n\nexport type CirclePaintPropsPossiblyEvaluated = {\n \"circle-radius\": PossiblyEvaluatedPropertyValue,\n \"circle-color\": PossiblyEvaluatedPropertyValue,\n \"circle-blur\": PossiblyEvaluatedPropertyValue,\n \"circle-opacity\": PossiblyEvaluatedPropertyValue,\n \"circle-translate\": [number, number],\n \"circle-translate-anchor\": \"map\" | \"viewport\",\n \"circle-pitch-scale\": \"map\" | \"viewport\",\n \"circle-pitch-alignment\": \"map\" | \"viewport\",\n \"circle-stroke-width\": PossiblyEvaluatedPropertyValue,\n \"circle-stroke-color\": PossiblyEvaluatedPropertyValue,\n \"circle-stroke-opacity\": PossiblyEvaluatedPropertyValue,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"circle-radius\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-radius\"] as any as StylePropertySpecification),\n \"circle-color\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-color\"] as any as StylePropertySpecification),\n \"circle-blur\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-blur\"] as any as StylePropertySpecification),\n \"circle-opacity\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-opacity\"] as any as StylePropertySpecification),\n \"circle-translate\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-translate\"] as any as StylePropertySpecification),\n \"circle-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-translate-anchor\"] as any as StylePropertySpecification),\n \"circle-pitch-scale\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-pitch-scale\"] as any as StylePropertySpecification),\n \"circle-pitch-alignment\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-pitch-alignment\"] as any as StylePropertySpecification),\n \"circle-stroke-width\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-stroke-width\"] as any as StylePropertySpecification),\n \"circle-stroke-color\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-stroke-color\"] as any as StylePropertySpecification),\n \"circle-stroke-opacity\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-stroke-opacity\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","/**\n * Common utilities\n * @module glMatrix\n */\n// Configuration Constants\nexport var EPSILON = 0.000001;\nexport var ARRAY_TYPE = typeof Float32Array !== 'undefined' ? Float32Array : Array;\nexport var RANDOM = Math.random;\n/**\n * Sets the type of array used when creating new vectors and matrices\n *\n * @param {Float32ArrayConstructor | ArrayConstructor} type Array type, such as Float32Array or Array\n */\n\nexport function setMatrixArrayType(type) {\n ARRAY_TYPE = type;\n}\nvar degree = Math.PI / 180;\n/**\n * Convert Degree To Radian\n *\n * @param {Number} a Angle in Degrees\n */\n\nexport function toRadian(a) {\n return a * degree;\n}\n/**\n * Tests whether or not the arguments have approximately the same value, within an absolute\n * or relative tolerance of glMatrix.EPSILON (an absolute tolerance is used for values less\n * than or equal to 1.0, and a relative tolerance is used for larger values)\n *\n * @param {Number} a The first number to test.\n * @param {Number} b The second number to test.\n * @returns {Boolean} True if the numbers are approximately equal, false otherwise.\n */\n\nexport function equals(a, b) {\n return Math.abs(a - b) <= EPSILON * Math.max(1.0, Math.abs(a), Math.abs(b));\n}\nif (!Math.hypot) Math.hypot = function () {\n var y = 0,\n i = arguments.length;\n\n while (i--) {\n y += arguments[i] * arguments[i];\n }\n\n return Math.sqrt(y);\n};","import * as glMatrix from \"./common.js\";\n/**\n * 2x2 Matrix\n * @module mat2\n */\n\n/**\n * Creates a new identity mat2\n *\n * @returns {mat2} a new 2x2 matrix\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(4);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[1] = 0;\n out[2] = 0;\n }\n\n out[0] = 1;\n out[3] = 1;\n return out;\n}\n/**\n * Creates a new mat2 initialized with values from an existing matrix\n *\n * @param {ReadonlyMat2} a matrix to clone\n * @returns {mat2} a new 2x2 matrix\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Copy the values from one mat2 to another\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the source matrix\n * @returns {mat2} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Set a mat2 to the identity matrix\n *\n * @param {mat2} out the receiving matrix\n * @returns {mat2} out\n */\n\nexport function identity(out) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 1;\n return out;\n}\n/**\n * Create a new mat2 with the given values\n *\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m10 Component in column 1, row 0 position (index 2)\n * @param {Number} m11 Component in column 1, row 1 position (index 3)\n * @returns {mat2} out A new 2x2 matrix\n */\n\nexport function fromValues(m00, m01, m10, m11) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = m00;\n out[1] = m01;\n out[2] = m10;\n out[3] = m11;\n return out;\n}\n/**\n * Set the components of a mat2 to the given values\n *\n * @param {mat2} out the receiving matrix\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m10 Component in column 1, row 0 position (index 2)\n * @param {Number} m11 Component in column 1, row 1 position (index 3)\n * @returns {mat2} out\n */\n\nexport function set(out, m00, m01, m10, m11) {\n out[0] = m00;\n out[1] = m01;\n out[2] = m10;\n out[3] = m11;\n return out;\n}\n/**\n * Transpose the values of a mat2\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the source matrix\n * @returns {mat2} out\n */\n\nexport function transpose(out, a) {\n // If we are transposing ourselves we can skip a few steps but have to cache\n // some values\n if (out === a) {\n var a1 = a[1];\n out[1] = a[2];\n out[2] = a1;\n } else {\n out[0] = a[0];\n out[1] = a[2];\n out[2] = a[1];\n out[3] = a[3];\n }\n\n return out;\n}\n/**\n * Inverts a mat2\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the source matrix\n * @returns {mat2} out\n */\n\nexport function invert(out, a) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3]; // Calculate the determinant\n\n var det = a0 * a3 - a2 * a1;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = a3 * det;\n out[1] = -a1 * det;\n out[2] = -a2 * det;\n out[3] = a0 * det;\n return out;\n}\n/**\n * Calculates the adjugate of a mat2\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the source matrix\n * @returns {mat2} out\n */\n\nexport function adjoint(out, a) {\n // Caching this value is nessecary if out == a\n var a0 = a[0];\n out[0] = a[3];\n out[1] = -a[1];\n out[2] = -a[2];\n out[3] = a0;\n return out;\n}\n/**\n * Calculates the determinant of a mat2\n *\n * @param {ReadonlyMat2} a the source matrix\n * @returns {Number} determinant of a\n */\n\nexport function determinant(a) {\n return a[0] * a[3] - a[2] * a[1];\n}\n/**\n * Multiplies two mat2's\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the first operand\n * @param {ReadonlyMat2} b the second operand\n * @returns {mat2} out\n */\n\nexport function multiply(out, a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n out[0] = a0 * b0 + a2 * b1;\n out[1] = a1 * b0 + a3 * b1;\n out[2] = a0 * b2 + a2 * b3;\n out[3] = a1 * b2 + a3 * b3;\n return out;\n}\n/**\n * Rotates a mat2 by the given angle\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat2} out\n */\n\nexport function rotate(out, a, rad) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n out[0] = a0 * c + a2 * s;\n out[1] = a1 * c + a3 * s;\n out[2] = a0 * -s + a2 * c;\n out[3] = a1 * -s + a3 * c;\n return out;\n}\n/**\n * Scales the mat2 by the dimensions in the given vec2\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the matrix to rotate\n * @param {ReadonlyVec2} v the vec2 to scale the matrix by\n * @returns {mat2} out\n **/\n\nexport function scale(out, a, v) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var v0 = v[0],\n v1 = v[1];\n out[0] = a0 * v0;\n out[1] = a1 * v0;\n out[2] = a2 * v1;\n out[3] = a3 * v1;\n return out;\n}\n/**\n * Creates a matrix from a given angle\n * This is equivalent to (but much faster than):\n *\n * mat2.identity(dest);\n * mat2.rotate(dest, dest, rad);\n *\n * @param {mat2} out mat2 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat2} out\n */\n\nexport function fromRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n out[0] = c;\n out[1] = s;\n out[2] = -s;\n out[3] = c;\n return out;\n}\n/**\n * Creates a matrix from a vector scaling\n * This is equivalent to (but much faster than):\n *\n * mat2.identity(dest);\n * mat2.scale(dest, dest, vec);\n *\n * @param {mat2} out mat2 receiving operation result\n * @param {ReadonlyVec2} v Scaling vector\n * @returns {mat2} out\n */\n\nexport function fromScaling(out, v) {\n out[0] = v[0];\n out[1] = 0;\n out[2] = 0;\n out[3] = v[1];\n return out;\n}\n/**\n * Returns a string representation of a mat2\n *\n * @param {ReadonlyMat2} a matrix to represent as a string\n * @returns {String} string representation of the matrix\n */\n\nexport function str(a) {\n return \"mat2(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \")\";\n}\n/**\n * Returns Frobenius norm of a mat2\n *\n * @param {ReadonlyMat2} a the matrix to calculate Frobenius norm of\n * @returns {Number} Frobenius norm\n */\n\nexport function frob(a) {\n return Math.hypot(a[0], a[1], a[2], a[3]);\n}\n/**\n * Returns L, D and U matrices (Lower triangular, Diagonal and Upper triangular) by factorizing the input matrix\n * @param {ReadonlyMat2} L the lower triangular matrix\n * @param {ReadonlyMat2} D the diagonal matrix\n * @param {ReadonlyMat2} U the upper triangular matrix\n * @param {ReadonlyMat2} a the input matrix to factorize\n */\n\nexport function LDU(L, D, U, a) {\n L[2] = a[2] / a[0];\n U[0] = a[0];\n U[1] = a[1];\n U[3] = a[3] - L[2] * U[1];\n return [L, D, U];\n}\n/**\n * Adds two mat2's\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the first operand\n * @param {ReadonlyMat2} b the second operand\n * @returns {mat2} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n return out;\n}\n/**\n * Subtracts matrix b from matrix a\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the first operand\n * @param {ReadonlyMat2} b the second operand\n * @returns {mat2} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n return out;\n}\n/**\n * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyMat2} a The first matrix.\n * @param {ReadonlyMat2} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3];\n}\n/**\n * Returns whether or not the matrices have approximately the same elements in the same position.\n *\n * @param {ReadonlyMat2} a The first matrix.\n * @param {ReadonlyMat2} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3));\n}\n/**\n * Multiply each element of the matrix by a scalar.\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the matrix to scale\n * @param {Number} b amount to scale the matrix's elements by\n * @returns {mat2} out\n */\n\nexport function multiplyScalar(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n return out;\n}\n/**\n * Adds two mat2's after multiplying each element of the second operand by a scalar value.\n *\n * @param {mat2} out the receiving vector\n * @param {ReadonlyMat2} a the first operand\n * @param {ReadonlyMat2} b the second operand\n * @param {Number} scale the amount to scale b's elements by before adding\n * @returns {mat2} out\n */\n\nexport function multiplyScalarAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n return out;\n}\n/**\n * Alias for {@link mat2.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link mat2.subtract}\n * @function\n */\n\nexport var sub = subtract;","import * as glMatrix from \"./common.js\";\n/**\n * 2x3 Matrix\n * @module mat2d\n * @description\n * A mat2d contains six elements defined as:\n *
\n * [a, b,\n *  c, d,\n *  tx, ty]\n * 
\n * This is a short form for the 3x3 matrix:\n *
\n * [a, b, 0,\n *  c, d, 0,\n *  tx, ty, 1]\n * 
\n * The last column is ignored so the array is shorter and operations are faster.\n */\n\n/**\n * Creates a new identity mat2d\n *\n * @returns {mat2d} a new 2x3 matrix\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(6);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[1] = 0;\n out[2] = 0;\n out[4] = 0;\n out[5] = 0;\n }\n\n out[0] = 1;\n out[3] = 1;\n return out;\n}\n/**\n * Creates a new mat2d initialized with values from an existing matrix\n *\n * @param {ReadonlyMat2d} a matrix to clone\n * @returns {mat2d} a new 2x3 matrix\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(6);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n return out;\n}\n/**\n * Copy the values from one mat2d to another\n *\n * @param {mat2d} out the receiving matrix\n * @param {ReadonlyMat2d} a the source matrix\n * @returns {mat2d} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n return out;\n}\n/**\n * Set a mat2d to the identity matrix\n *\n * @param {mat2d} out the receiving matrix\n * @returns {mat2d} out\n */\n\nexport function identity(out) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 1;\n out[4] = 0;\n out[5] = 0;\n return out;\n}\n/**\n * Create a new mat2d with the given values\n *\n * @param {Number} a Component A (index 0)\n * @param {Number} b Component B (index 1)\n * @param {Number} c Component C (index 2)\n * @param {Number} d Component D (index 3)\n * @param {Number} tx Component TX (index 4)\n * @param {Number} ty Component TY (index 5)\n * @returns {mat2d} A new mat2d\n */\n\nexport function fromValues(a, b, c, d, tx, ty) {\n var out = new glMatrix.ARRAY_TYPE(6);\n out[0] = a;\n out[1] = b;\n out[2] = c;\n out[3] = d;\n out[4] = tx;\n out[5] = ty;\n return out;\n}\n/**\n * Set the components of a mat2d to the given values\n *\n * @param {mat2d} out the receiving matrix\n * @param {Number} a Component A (index 0)\n * @param {Number} b Component B (index 1)\n * @param {Number} c Component C (index 2)\n * @param {Number} d Component D (index 3)\n * @param {Number} tx Component TX (index 4)\n * @param {Number} ty Component TY (index 5)\n * @returns {mat2d} out\n */\n\nexport function set(out, a, b, c, d, tx, ty) {\n out[0] = a;\n out[1] = b;\n out[2] = c;\n out[3] = d;\n out[4] = tx;\n out[5] = ty;\n return out;\n}\n/**\n * Inverts a mat2d\n *\n * @param {mat2d} out the receiving matrix\n * @param {ReadonlyMat2d} a the source matrix\n * @returns {mat2d} out\n */\n\nexport function invert(out, a) {\n var aa = a[0],\n ab = a[1],\n ac = a[2],\n ad = a[3];\n var atx = a[4],\n aty = a[5];\n var det = aa * ad - ab * ac;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = ad * det;\n out[1] = -ab * det;\n out[2] = -ac * det;\n out[3] = aa * det;\n out[4] = (ac * aty - ad * atx) * det;\n out[5] = (ab * atx - aa * aty) * det;\n return out;\n}\n/**\n * Calculates the determinant of a mat2d\n *\n * @param {ReadonlyMat2d} a the source matrix\n * @returns {Number} determinant of a\n */\n\nexport function determinant(a) {\n return a[0] * a[3] - a[1] * a[2];\n}\n/**\n * Multiplies two mat2d's\n *\n * @param {mat2d} out the receiving matrix\n * @param {ReadonlyMat2d} a the first operand\n * @param {ReadonlyMat2d} b the second operand\n * @returns {mat2d} out\n */\n\nexport function multiply(out, a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3],\n a4 = a[4],\n a5 = a[5];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3],\n b4 = b[4],\n b5 = b[5];\n out[0] = a0 * b0 + a2 * b1;\n out[1] = a1 * b0 + a3 * b1;\n out[2] = a0 * b2 + a2 * b3;\n out[3] = a1 * b2 + a3 * b3;\n out[4] = a0 * b4 + a2 * b5 + a4;\n out[5] = a1 * b4 + a3 * b5 + a5;\n return out;\n}\n/**\n * Rotates a mat2d by the given angle\n *\n * @param {mat2d} out the receiving matrix\n * @param {ReadonlyMat2d} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat2d} out\n */\n\nexport function rotate(out, a, rad) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3],\n a4 = a[4],\n a5 = a[5];\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n out[0] = a0 * c + a2 * s;\n out[1] = a1 * c + a3 * s;\n out[2] = a0 * -s + a2 * c;\n out[3] = a1 * -s + a3 * c;\n out[4] = a4;\n out[5] = a5;\n return out;\n}\n/**\n * Scales the mat2d by the dimensions in the given vec2\n *\n * @param {mat2d} out the receiving matrix\n * @param {ReadonlyMat2d} a the matrix to translate\n * @param {ReadonlyVec2} v the vec2 to scale the matrix by\n * @returns {mat2d} out\n **/\n\nexport function scale(out, a, v) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3],\n a4 = a[4],\n a5 = a[5];\n var v0 = v[0],\n v1 = v[1];\n out[0] = a0 * v0;\n out[1] = a1 * v0;\n out[2] = a2 * v1;\n out[3] = a3 * v1;\n out[4] = a4;\n out[5] = a5;\n return out;\n}\n/**\n * Translates the mat2d by the dimensions in the given vec2\n *\n * @param {mat2d} out the receiving matrix\n * @param {ReadonlyMat2d} a the matrix to translate\n * @param {ReadonlyVec2} v the vec2 to translate the matrix by\n * @returns {mat2d} out\n **/\n\nexport function translate(out, a, v) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3],\n a4 = a[4],\n a5 = a[5];\n var v0 = v[0],\n v1 = v[1];\n out[0] = a0;\n out[1] = a1;\n out[2] = a2;\n out[3] = a3;\n out[4] = a0 * v0 + a2 * v1 + a4;\n out[5] = a1 * v0 + a3 * v1 + a5;\n return out;\n}\n/**\n * Creates a matrix from a given angle\n * This is equivalent to (but much faster than):\n *\n * mat2d.identity(dest);\n * mat2d.rotate(dest, dest, rad);\n *\n * @param {mat2d} out mat2d receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat2d} out\n */\n\nexport function fromRotation(out, rad) {\n var s = Math.sin(rad),\n c = Math.cos(rad);\n out[0] = c;\n out[1] = s;\n out[2] = -s;\n out[3] = c;\n out[4] = 0;\n out[5] = 0;\n return out;\n}\n/**\n * Creates a matrix from a vector scaling\n * This is equivalent to (but much faster than):\n *\n * mat2d.identity(dest);\n * mat2d.scale(dest, dest, vec);\n *\n * @param {mat2d} out mat2d receiving operation result\n * @param {ReadonlyVec2} v Scaling vector\n * @returns {mat2d} out\n */\n\nexport function fromScaling(out, v) {\n out[0] = v[0];\n out[1] = 0;\n out[2] = 0;\n out[3] = v[1];\n out[4] = 0;\n out[5] = 0;\n return out;\n}\n/**\n * Creates a matrix from a vector translation\n * This is equivalent to (but much faster than):\n *\n * mat2d.identity(dest);\n * mat2d.translate(dest, dest, vec);\n *\n * @param {mat2d} out mat2d receiving operation result\n * @param {ReadonlyVec2} v Translation vector\n * @returns {mat2d} out\n */\n\nexport function fromTranslation(out, v) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 1;\n out[4] = v[0];\n out[5] = v[1];\n return out;\n}\n/**\n * Returns a string representation of a mat2d\n *\n * @param {ReadonlyMat2d} a matrix to represent as a string\n * @returns {String} string representation of the matrix\n */\n\nexport function str(a) {\n return \"mat2d(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \", \" + a[4] + \", \" + a[5] + \")\";\n}\n/**\n * Returns Frobenius norm of a mat2d\n *\n * @param {ReadonlyMat2d} a the matrix to calculate Frobenius norm of\n * @returns {Number} Frobenius norm\n */\n\nexport function frob(a) {\n return Math.hypot(a[0], a[1], a[2], a[3], a[4], a[5], 1);\n}\n/**\n * Adds two mat2d's\n *\n * @param {mat2d} out the receiving matrix\n * @param {ReadonlyMat2d} a the first operand\n * @param {ReadonlyMat2d} b the second operand\n * @returns {mat2d} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n out[4] = a[4] + b[4];\n out[5] = a[5] + b[5];\n return out;\n}\n/**\n * Subtracts matrix b from matrix a\n *\n * @param {mat2d} out the receiving matrix\n * @param {ReadonlyMat2d} a the first operand\n * @param {ReadonlyMat2d} b the second operand\n * @returns {mat2d} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n out[4] = a[4] - b[4];\n out[5] = a[5] - b[5];\n return out;\n}\n/**\n * Multiply each element of the matrix by a scalar.\n *\n * @param {mat2d} out the receiving matrix\n * @param {ReadonlyMat2d} a the matrix to scale\n * @param {Number} b amount to scale the matrix's elements by\n * @returns {mat2d} out\n */\n\nexport function multiplyScalar(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n out[4] = a[4] * b;\n out[5] = a[5] * b;\n return out;\n}\n/**\n * Adds two mat2d's after multiplying each element of the second operand by a scalar value.\n *\n * @param {mat2d} out the receiving vector\n * @param {ReadonlyMat2d} a the first operand\n * @param {ReadonlyMat2d} b the second operand\n * @param {Number} scale the amount to scale b's elements by before adding\n * @returns {mat2d} out\n */\n\nexport function multiplyScalarAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n out[4] = a[4] + b[4] * scale;\n out[5] = a[5] + b[5] * scale;\n return out;\n}\n/**\n * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyMat2d} a The first matrix.\n * @param {ReadonlyMat2d} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5];\n}\n/**\n * Returns whether or not the matrices have approximately the same elements in the same position.\n *\n * @param {ReadonlyMat2d} a The first matrix.\n * @param {ReadonlyMat2d} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3],\n a4 = a[4],\n a5 = a[5];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3],\n b4 = b[4],\n b5 = b[5];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5));\n}\n/**\n * Alias for {@link mat2d.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link mat2d.subtract}\n * @function\n */\n\nexport var sub = subtract;","import * as glMatrix from \"./common.js\";\n/**\n * 3x3 Matrix\n * @module mat3\n */\n\n/**\n * Creates a new identity mat3\n *\n * @returns {mat3} a new 3x3 matrix\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(9);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n }\n\n out[0] = 1;\n out[4] = 1;\n out[8] = 1;\n return out;\n}\n/**\n * Copies the upper-left 3x3 values into the given mat3.\n *\n * @param {mat3} out the receiving 3x3 matrix\n * @param {ReadonlyMat4} a the source 4x4 matrix\n * @returns {mat3} out\n */\n\nexport function fromMat4(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[4];\n out[4] = a[5];\n out[5] = a[6];\n out[6] = a[8];\n out[7] = a[9];\n out[8] = a[10];\n return out;\n}\n/**\n * Creates a new mat3 initialized with values from an existing matrix\n *\n * @param {ReadonlyMat3} a matrix to clone\n * @returns {mat3} a new 3x3 matrix\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(9);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n return out;\n}\n/**\n * Copy the values from one mat3 to another\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the source matrix\n * @returns {mat3} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n return out;\n}\n/**\n * Create a new mat3 with the given values\n *\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m10 Component in column 1, row 0 position (index 3)\n * @param {Number} m11 Component in column 1, row 1 position (index 4)\n * @param {Number} m12 Component in column 1, row 2 position (index 5)\n * @param {Number} m20 Component in column 2, row 0 position (index 6)\n * @param {Number} m21 Component in column 2, row 1 position (index 7)\n * @param {Number} m22 Component in column 2, row 2 position (index 8)\n * @returns {mat3} A new mat3\n */\n\nexport function fromValues(m00, m01, m02, m10, m11, m12, m20, m21, m22) {\n var out = new glMatrix.ARRAY_TYPE(9);\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m10;\n out[4] = m11;\n out[5] = m12;\n out[6] = m20;\n out[7] = m21;\n out[8] = m22;\n return out;\n}\n/**\n * Set the components of a mat3 to the given values\n *\n * @param {mat3} out the receiving matrix\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m10 Component in column 1, row 0 position (index 3)\n * @param {Number} m11 Component in column 1, row 1 position (index 4)\n * @param {Number} m12 Component in column 1, row 2 position (index 5)\n * @param {Number} m20 Component in column 2, row 0 position (index 6)\n * @param {Number} m21 Component in column 2, row 1 position (index 7)\n * @param {Number} m22 Component in column 2, row 2 position (index 8)\n * @returns {mat3} out\n */\n\nexport function set(out, m00, m01, m02, m10, m11, m12, m20, m21, m22) {\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m10;\n out[4] = m11;\n out[5] = m12;\n out[6] = m20;\n out[7] = m21;\n out[8] = m22;\n return out;\n}\n/**\n * Set a mat3 to the identity matrix\n *\n * @param {mat3} out the receiving matrix\n * @returns {mat3} out\n */\n\nexport function identity(out) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 1;\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 1;\n return out;\n}\n/**\n * Transpose the values of a mat3\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the source matrix\n * @returns {mat3} out\n */\n\nexport function transpose(out, a) {\n // If we are transposing ourselves we can skip a few steps but have to cache some values\n if (out === a) {\n var a01 = a[1],\n a02 = a[2],\n a12 = a[5];\n out[1] = a[3];\n out[2] = a[6];\n out[3] = a01;\n out[5] = a[7];\n out[6] = a02;\n out[7] = a12;\n } else {\n out[0] = a[0];\n out[1] = a[3];\n out[2] = a[6];\n out[3] = a[1];\n out[4] = a[4];\n out[5] = a[7];\n out[6] = a[2];\n out[7] = a[5];\n out[8] = a[8];\n }\n\n return out;\n}\n/**\n * Inverts a mat3\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the source matrix\n * @returns {mat3} out\n */\n\nexport function invert(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2];\n var a10 = a[3],\n a11 = a[4],\n a12 = a[5];\n var a20 = a[6],\n a21 = a[7],\n a22 = a[8];\n var b01 = a22 * a11 - a12 * a21;\n var b11 = -a22 * a10 + a12 * a20;\n var b21 = a21 * a10 - a11 * a20; // Calculate the determinant\n\n var det = a00 * b01 + a01 * b11 + a02 * b21;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = b01 * det;\n out[1] = (-a22 * a01 + a02 * a21) * det;\n out[2] = (a12 * a01 - a02 * a11) * det;\n out[3] = b11 * det;\n out[4] = (a22 * a00 - a02 * a20) * det;\n out[5] = (-a12 * a00 + a02 * a10) * det;\n out[6] = b21 * det;\n out[7] = (-a21 * a00 + a01 * a20) * det;\n out[8] = (a11 * a00 - a01 * a10) * det;\n return out;\n}\n/**\n * Calculates the adjugate of a mat3\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the source matrix\n * @returns {mat3} out\n */\n\nexport function adjoint(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2];\n var a10 = a[3],\n a11 = a[4],\n a12 = a[5];\n var a20 = a[6],\n a21 = a[7],\n a22 = a[8];\n out[0] = a11 * a22 - a12 * a21;\n out[1] = a02 * a21 - a01 * a22;\n out[2] = a01 * a12 - a02 * a11;\n out[3] = a12 * a20 - a10 * a22;\n out[4] = a00 * a22 - a02 * a20;\n out[5] = a02 * a10 - a00 * a12;\n out[6] = a10 * a21 - a11 * a20;\n out[7] = a01 * a20 - a00 * a21;\n out[8] = a00 * a11 - a01 * a10;\n return out;\n}\n/**\n * Calculates the determinant of a mat3\n *\n * @param {ReadonlyMat3} a the source matrix\n * @returns {Number} determinant of a\n */\n\nexport function determinant(a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2];\n var a10 = a[3],\n a11 = a[4],\n a12 = a[5];\n var a20 = a[6],\n a21 = a[7],\n a22 = a[8];\n return a00 * (a22 * a11 - a12 * a21) + a01 * (-a22 * a10 + a12 * a20) + a02 * (a21 * a10 - a11 * a20);\n}\n/**\n * Multiplies two mat3's\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the first operand\n * @param {ReadonlyMat3} b the second operand\n * @returns {mat3} out\n */\n\nexport function multiply(out, a, b) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2];\n var a10 = a[3],\n a11 = a[4],\n a12 = a[5];\n var a20 = a[6],\n a21 = a[7],\n a22 = a[8];\n var b00 = b[0],\n b01 = b[1],\n b02 = b[2];\n var b10 = b[3],\n b11 = b[4],\n b12 = b[5];\n var b20 = b[6],\n b21 = b[7],\n b22 = b[8];\n out[0] = b00 * a00 + b01 * a10 + b02 * a20;\n out[1] = b00 * a01 + b01 * a11 + b02 * a21;\n out[2] = b00 * a02 + b01 * a12 + b02 * a22;\n out[3] = b10 * a00 + b11 * a10 + b12 * a20;\n out[4] = b10 * a01 + b11 * a11 + b12 * a21;\n out[5] = b10 * a02 + b11 * a12 + b12 * a22;\n out[6] = b20 * a00 + b21 * a10 + b22 * a20;\n out[7] = b20 * a01 + b21 * a11 + b22 * a21;\n out[8] = b20 * a02 + b21 * a12 + b22 * a22;\n return out;\n}\n/**\n * Translate a mat3 by the given vector\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the matrix to translate\n * @param {ReadonlyVec2} v vector to translate by\n * @returns {mat3} out\n */\n\nexport function translate(out, a, v) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a10 = a[3],\n a11 = a[4],\n a12 = a[5],\n a20 = a[6],\n a21 = a[7],\n a22 = a[8],\n x = v[0],\n y = v[1];\n out[0] = a00;\n out[1] = a01;\n out[2] = a02;\n out[3] = a10;\n out[4] = a11;\n out[5] = a12;\n out[6] = x * a00 + y * a10 + a20;\n out[7] = x * a01 + y * a11 + a21;\n out[8] = x * a02 + y * a12 + a22;\n return out;\n}\n/**\n * Rotates a mat3 by the given angle\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat3} out\n */\n\nexport function rotate(out, a, rad) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a10 = a[3],\n a11 = a[4],\n a12 = a[5],\n a20 = a[6],\n a21 = a[7],\n a22 = a[8],\n s = Math.sin(rad),\n c = Math.cos(rad);\n out[0] = c * a00 + s * a10;\n out[1] = c * a01 + s * a11;\n out[2] = c * a02 + s * a12;\n out[3] = c * a10 - s * a00;\n out[4] = c * a11 - s * a01;\n out[5] = c * a12 - s * a02;\n out[6] = a20;\n out[7] = a21;\n out[8] = a22;\n return out;\n}\n/**\n * Scales the mat3 by the dimensions in the given vec2\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the matrix to rotate\n * @param {ReadonlyVec2} v the vec2 to scale the matrix by\n * @returns {mat3} out\n **/\n\nexport function scale(out, a, v) {\n var x = v[0],\n y = v[1];\n out[0] = x * a[0];\n out[1] = x * a[1];\n out[2] = x * a[2];\n out[3] = y * a[3];\n out[4] = y * a[4];\n out[5] = y * a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n return out;\n}\n/**\n * Creates a matrix from a vector translation\n * This is equivalent to (but much faster than):\n *\n * mat3.identity(dest);\n * mat3.translate(dest, dest, vec);\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {ReadonlyVec2} v Translation vector\n * @returns {mat3} out\n */\n\nexport function fromTranslation(out, v) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 1;\n out[5] = 0;\n out[6] = v[0];\n out[7] = v[1];\n out[8] = 1;\n return out;\n}\n/**\n * Creates a matrix from a given angle\n * This is equivalent to (but much faster than):\n *\n * mat3.identity(dest);\n * mat3.rotate(dest, dest, rad);\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat3} out\n */\n\nexport function fromRotation(out, rad) {\n var s = Math.sin(rad),\n c = Math.cos(rad);\n out[0] = c;\n out[1] = s;\n out[2] = 0;\n out[3] = -s;\n out[4] = c;\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 1;\n return out;\n}\n/**\n * Creates a matrix from a vector scaling\n * This is equivalent to (but much faster than):\n *\n * mat3.identity(dest);\n * mat3.scale(dest, dest, vec);\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {ReadonlyVec2} v Scaling vector\n * @returns {mat3} out\n */\n\nexport function fromScaling(out, v) {\n out[0] = v[0];\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = v[1];\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 1;\n return out;\n}\n/**\n * Copies the values from a mat2d into a mat3\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat2d} a the matrix to copy\n * @returns {mat3} out\n **/\n\nexport function fromMat2d(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = 0;\n out[3] = a[2];\n out[4] = a[3];\n out[5] = 0;\n out[6] = a[4];\n out[7] = a[5];\n out[8] = 1;\n return out;\n}\n/**\n * Calculates a 3x3 matrix from the given quaternion\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {ReadonlyQuat} q Quaternion to create matrix from\n *\n * @returns {mat3} out\n */\n\nexport function fromQuat(out, q) {\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var yx = y * x2;\n var yy = y * y2;\n var zx = z * x2;\n var zy = z * y2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n out[0] = 1 - yy - zz;\n out[3] = yx - wz;\n out[6] = zx + wy;\n out[1] = yx + wz;\n out[4] = 1 - xx - zz;\n out[7] = zy - wx;\n out[2] = zx - wy;\n out[5] = zy + wx;\n out[8] = 1 - xx - yy;\n return out;\n}\n/**\n * Calculates a 3x3 normal matrix (transpose inverse) from the 4x4 matrix\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {ReadonlyMat4} a Mat4 to derive the normal matrix from\n *\n * @returns {mat3} out\n */\n\nexport function normalFromMat4(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n var b00 = a00 * a11 - a01 * a10;\n var b01 = a00 * a12 - a02 * a10;\n var b02 = a00 * a13 - a03 * a10;\n var b03 = a01 * a12 - a02 * a11;\n var b04 = a01 * a13 - a03 * a11;\n var b05 = a02 * a13 - a03 * a12;\n var b06 = a20 * a31 - a21 * a30;\n var b07 = a20 * a32 - a22 * a30;\n var b08 = a20 * a33 - a23 * a30;\n var b09 = a21 * a32 - a22 * a31;\n var b10 = a21 * a33 - a23 * a31;\n var b11 = a22 * a33 - a23 * a32; // Calculate the determinant\n\n var det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det;\n out[1] = (a12 * b08 - a10 * b11 - a13 * b07) * det;\n out[2] = (a10 * b10 - a11 * b08 + a13 * b06) * det;\n out[3] = (a02 * b10 - a01 * b11 - a03 * b09) * det;\n out[4] = (a00 * b11 - a02 * b08 + a03 * b07) * det;\n out[5] = (a01 * b08 - a00 * b10 - a03 * b06) * det;\n out[6] = (a31 * b05 - a32 * b04 + a33 * b03) * det;\n out[7] = (a32 * b02 - a30 * b05 - a33 * b01) * det;\n out[8] = (a30 * b04 - a31 * b02 + a33 * b00) * det;\n return out;\n}\n/**\n * Generates a 2D projection matrix with the given bounds\n *\n * @param {mat3} out mat3 frustum matrix will be written into\n * @param {number} width Width of your gl context\n * @param {number} height Height of gl context\n * @returns {mat3} out\n */\n\nexport function projection(out, width, height) {\n out[0] = 2 / width;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = -2 / height;\n out[5] = 0;\n out[6] = -1;\n out[7] = 1;\n out[8] = 1;\n return out;\n}\n/**\n * Returns a string representation of a mat3\n *\n * @param {ReadonlyMat3} a matrix to represent as a string\n * @returns {String} string representation of the matrix\n */\n\nexport function str(a) {\n return \"mat3(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \", \" + a[4] + \", \" + a[5] + \", \" + a[6] + \", \" + a[7] + \", \" + a[8] + \")\";\n}\n/**\n * Returns Frobenius norm of a mat3\n *\n * @param {ReadonlyMat3} a the matrix to calculate Frobenius norm of\n * @returns {Number} Frobenius norm\n */\n\nexport function frob(a) {\n return Math.hypot(a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8]);\n}\n/**\n * Adds two mat3's\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the first operand\n * @param {ReadonlyMat3} b the second operand\n * @returns {mat3} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n out[4] = a[4] + b[4];\n out[5] = a[5] + b[5];\n out[6] = a[6] + b[6];\n out[7] = a[7] + b[7];\n out[8] = a[8] + b[8];\n return out;\n}\n/**\n * Subtracts matrix b from matrix a\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the first operand\n * @param {ReadonlyMat3} b the second operand\n * @returns {mat3} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n out[4] = a[4] - b[4];\n out[5] = a[5] - b[5];\n out[6] = a[6] - b[6];\n out[7] = a[7] - b[7];\n out[8] = a[8] - b[8];\n return out;\n}\n/**\n * Multiply each element of the matrix by a scalar.\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the matrix to scale\n * @param {Number} b amount to scale the matrix's elements by\n * @returns {mat3} out\n */\n\nexport function multiplyScalar(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n out[4] = a[4] * b;\n out[5] = a[5] * b;\n out[6] = a[6] * b;\n out[7] = a[7] * b;\n out[8] = a[8] * b;\n return out;\n}\n/**\n * Adds two mat3's after multiplying each element of the second operand by a scalar value.\n *\n * @param {mat3} out the receiving vector\n * @param {ReadonlyMat3} a the first operand\n * @param {ReadonlyMat3} b the second operand\n * @param {Number} scale the amount to scale b's elements by before adding\n * @returns {mat3} out\n */\n\nexport function multiplyScalarAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n out[4] = a[4] + b[4] * scale;\n out[5] = a[5] + b[5] * scale;\n out[6] = a[6] + b[6] * scale;\n out[7] = a[7] + b[7] * scale;\n out[8] = a[8] + b[8] * scale;\n return out;\n}\n/**\n * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyMat3} a The first matrix.\n * @param {ReadonlyMat3} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5] && a[6] === b[6] && a[7] === b[7] && a[8] === b[8];\n}\n/**\n * Returns whether or not the matrices have approximately the same elements in the same position.\n *\n * @param {ReadonlyMat3} a The first matrix.\n * @param {ReadonlyMat3} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3],\n a4 = a[4],\n a5 = a[5],\n a6 = a[6],\n a7 = a[7],\n a8 = a[8];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3],\n b4 = b[4],\n b5 = b[5],\n b6 = b[6],\n b7 = b[7],\n b8 = b[8];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5)) && Math.abs(a6 - b6) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a6), Math.abs(b6)) && Math.abs(a7 - b7) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a7), Math.abs(b7)) && Math.abs(a8 - b8) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a8), Math.abs(b8));\n}\n/**\n * Alias for {@link mat3.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link mat3.subtract}\n * @function\n */\n\nexport var sub = subtract;","import * as glMatrix from \"./common.js\";\n/**\n * 4x4 Matrix
Format: column-major, when typed out it looks like row-major
The matrices are being post multiplied.\n * @module mat4\n */\n\n/**\n * Creates a new identity mat4\n *\n * @returns {mat4} a new 4x4 matrix\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(16);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n }\n\n out[0] = 1;\n out[5] = 1;\n out[10] = 1;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a new mat4 initialized with values from an existing matrix\n *\n * @param {ReadonlyMat4} a matrix to clone\n * @returns {mat4} a new 4x4 matrix\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(16);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n out[9] = a[9];\n out[10] = a[10];\n out[11] = a[11];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n return out;\n}\n/**\n * Copy the values from one mat4 to another\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n out[9] = a[9];\n out[10] = a[10];\n out[11] = a[11];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n return out;\n}\n/**\n * Create a new mat4 with the given values\n *\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m03 Component in column 0, row 3 position (index 3)\n * @param {Number} m10 Component in column 1, row 0 position (index 4)\n * @param {Number} m11 Component in column 1, row 1 position (index 5)\n * @param {Number} m12 Component in column 1, row 2 position (index 6)\n * @param {Number} m13 Component in column 1, row 3 position (index 7)\n * @param {Number} m20 Component in column 2, row 0 position (index 8)\n * @param {Number} m21 Component in column 2, row 1 position (index 9)\n * @param {Number} m22 Component in column 2, row 2 position (index 10)\n * @param {Number} m23 Component in column 2, row 3 position (index 11)\n * @param {Number} m30 Component in column 3, row 0 position (index 12)\n * @param {Number} m31 Component in column 3, row 1 position (index 13)\n * @param {Number} m32 Component in column 3, row 2 position (index 14)\n * @param {Number} m33 Component in column 3, row 3 position (index 15)\n * @returns {mat4} A new mat4\n */\n\nexport function fromValues(m00, m01, m02, m03, m10, m11, m12, m13, m20, m21, m22, m23, m30, m31, m32, m33) {\n var out = new glMatrix.ARRAY_TYPE(16);\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m03;\n out[4] = m10;\n out[5] = m11;\n out[6] = m12;\n out[7] = m13;\n out[8] = m20;\n out[9] = m21;\n out[10] = m22;\n out[11] = m23;\n out[12] = m30;\n out[13] = m31;\n out[14] = m32;\n out[15] = m33;\n return out;\n}\n/**\n * Set the components of a mat4 to the given values\n *\n * @param {mat4} out the receiving matrix\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m03 Component in column 0, row 3 position (index 3)\n * @param {Number} m10 Component in column 1, row 0 position (index 4)\n * @param {Number} m11 Component in column 1, row 1 position (index 5)\n * @param {Number} m12 Component in column 1, row 2 position (index 6)\n * @param {Number} m13 Component in column 1, row 3 position (index 7)\n * @param {Number} m20 Component in column 2, row 0 position (index 8)\n * @param {Number} m21 Component in column 2, row 1 position (index 9)\n * @param {Number} m22 Component in column 2, row 2 position (index 10)\n * @param {Number} m23 Component in column 2, row 3 position (index 11)\n * @param {Number} m30 Component in column 3, row 0 position (index 12)\n * @param {Number} m31 Component in column 3, row 1 position (index 13)\n * @param {Number} m32 Component in column 3, row 2 position (index 14)\n * @param {Number} m33 Component in column 3, row 3 position (index 15)\n * @returns {mat4} out\n */\n\nexport function set(out, m00, m01, m02, m03, m10, m11, m12, m13, m20, m21, m22, m23, m30, m31, m32, m33) {\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m03;\n out[4] = m10;\n out[5] = m11;\n out[6] = m12;\n out[7] = m13;\n out[8] = m20;\n out[9] = m21;\n out[10] = m22;\n out[11] = m23;\n out[12] = m30;\n out[13] = m31;\n out[14] = m32;\n out[15] = m33;\n return out;\n}\n/**\n * Set a mat4 to the identity matrix\n *\n * @param {mat4} out the receiving matrix\n * @returns {mat4} out\n */\n\nexport function identity(out) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = 1;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 1;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Transpose the values of a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function transpose(out, a) {\n // If we are transposing ourselves we can skip a few steps but have to cache some values\n if (out === a) {\n var a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a12 = a[6],\n a13 = a[7];\n var a23 = a[11];\n out[1] = a[4];\n out[2] = a[8];\n out[3] = a[12];\n out[4] = a01;\n out[6] = a[9];\n out[7] = a[13];\n out[8] = a02;\n out[9] = a12;\n out[11] = a[14];\n out[12] = a03;\n out[13] = a13;\n out[14] = a23;\n } else {\n out[0] = a[0];\n out[1] = a[4];\n out[2] = a[8];\n out[3] = a[12];\n out[4] = a[1];\n out[5] = a[5];\n out[6] = a[9];\n out[7] = a[13];\n out[8] = a[2];\n out[9] = a[6];\n out[10] = a[10];\n out[11] = a[14];\n out[12] = a[3];\n out[13] = a[7];\n out[14] = a[11];\n out[15] = a[15];\n }\n\n return out;\n}\n/**\n * Inverts a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function invert(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n var b00 = a00 * a11 - a01 * a10;\n var b01 = a00 * a12 - a02 * a10;\n var b02 = a00 * a13 - a03 * a10;\n var b03 = a01 * a12 - a02 * a11;\n var b04 = a01 * a13 - a03 * a11;\n var b05 = a02 * a13 - a03 * a12;\n var b06 = a20 * a31 - a21 * a30;\n var b07 = a20 * a32 - a22 * a30;\n var b08 = a20 * a33 - a23 * a30;\n var b09 = a21 * a32 - a22 * a31;\n var b10 = a21 * a33 - a23 * a31;\n var b11 = a22 * a33 - a23 * a32; // Calculate the determinant\n\n var det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det;\n out[1] = (a02 * b10 - a01 * b11 - a03 * b09) * det;\n out[2] = (a31 * b05 - a32 * b04 + a33 * b03) * det;\n out[3] = (a22 * b04 - a21 * b05 - a23 * b03) * det;\n out[4] = (a12 * b08 - a10 * b11 - a13 * b07) * det;\n out[5] = (a00 * b11 - a02 * b08 + a03 * b07) * det;\n out[6] = (a32 * b02 - a30 * b05 - a33 * b01) * det;\n out[7] = (a20 * b05 - a22 * b02 + a23 * b01) * det;\n out[8] = (a10 * b10 - a11 * b08 + a13 * b06) * det;\n out[9] = (a01 * b08 - a00 * b10 - a03 * b06) * det;\n out[10] = (a30 * b04 - a31 * b02 + a33 * b00) * det;\n out[11] = (a21 * b02 - a20 * b04 - a23 * b00) * det;\n out[12] = (a11 * b07 - a10 * b09 - a12 * b06) * det;\n out[13] = (a00 * b09 - a01 * b07 + a02 * b06) * det;\n out[14] = (a31 * b01 - a30 * b03 - a32 * b00) * det;\n out[15] = (a20 * b03 - a21 * b01 + a22 * b00) * det;\n return out;\n}\n/**\n * Calculates the adjugate of a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function adjoint(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n out[0] = a11 * (a22 * a33 - a23 * a32) - a21 * (a12 * a33 - a13 * a32) + a31 * (a12 * a23 - a13 * a22);\n out[1] = -(a01 * (a22 * a33 - a23 * a32) - a21 * (a02 * a33 - a03 * a32) + a31 * (a02 * a23 - a03 * a22));\n out[2] = a01 * (a12 * a33 - a13 * a32) - a11 * (a02 * a33 - a03 * a32) + a31 * (a02 * a13 - a03 * a12);\n out[3] = -(a01 * (a12 * a23 - a13 * a22) - a11 * (a02 * a23 - a03 * a22) + a21 * (a02 * a13 - a03 * a12));\n out[4] = -(a10 * (a22 * a33 - a23 * a32) - a20 * (a12 * a33 - a13 * a32) + a30 * (a12 * a23 - a13 * a22));\n out[5] = a00 * (a22 * a33 - a23 * a32) - a20 * (a02 * a33 - a03 * a32) + a30 * (a02 * a23 - a03 * a22);\n out[6] = -(a00 * (a12 * a33 - a13 * a32) - a10 * (a02 * a33 - a03 * a32) + a30 * (a02 * a13 - a03 * a12));\n out[7] = a00 * (a12 * a23 - a13 * a22) - a10 * (a02 * a23 - a03 * a22) + a20 * (a02 * a13 - a03 * a12);\n out[8] = a10 * (a21 * a33 - a23 * a31) - a20 * (a11 * a33 - a13 * a31) + a30 * (a11 * a23 - a13 * a21);\n out[9] = -(a00 * (a21 * a33 - a23 * a31) - a20 * (a01 * a33 - a03 * a31) + a30 * (a01 * a23 - a03 * a21));\n out[10] = a00 * (a11 * a33 - a13 * a31) - a10 * (a01 * a33 - a03 * a31) + a30 * (a01 * a13 - a03 * a11);\n out[11] = -(a00 * (a11 * a23 - a13 * a21) - a10 * (a01 * a23 - a03 * a21) + a20 * (a01 * a13 - a03 * a11));\n out[12] = -(a10 * (a21 * a32 - a22 * a31) - a20 * (a11 * a32 - a12 * a31) + a30 * (a11 * a22 - a12 * a21));\n out[13] = a00 * (a21 * a32 - a22 * a31) - a20 * (a01 * a32 - a02 * a31) + a30 * (a01 * a22 - a02 * a21);\n out[14] = -(a00 * (a11 * a32 - a12 * a31) - a10 * (a01 * a32 - a02 * a31) + a30 * (a01 * a12 - a02 * a11));\n out[15] = a00 * (a11 * a22 - a12 * a21) - a10 * (a01 * a22 - a02 * a21) + a20 * (a01 * a12 - a02 * a11);\n return out;\n}\n/**\n * Calculates the determinant of a mat4\n *\n * @param {ReadonlyMat4} a the source matrix\n * @returns {Number} determinant of a\n */\n\nexport function determinant(a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n var b00 = a00 * a11 - a01 * a10;\n var b01 = a00 * a12 - a02 * a10;\n var b02 = a00 * a13 - a03 * a10;\n var b03 = a01 * a12 - a02 * a11;\n var b04 = a01 * a13 - a03 * a11;\n var b05 = a02 * a13 - a03 * a12;\n var b06 = a20 * a31 - a21 * a30;\n var b07 = a20 * a32 - a22 * a30;\n var b08 = a20 * a33 - a23 * a30;\n var b09 = a21 * a32 - a22 * a31;\n var b10 = a21 * a33 - a23 * a31;\n var b11 = a22 * a33 - a23 * a32; // Calculate the determinant\n\n return b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n}\n/**\n * Multiplies two mat4s\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @returns {mat4} out\n */\n\nexport function multiply(out, a, b) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15]; // Cache only the current line of the second matrix\n\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n out[0] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[1] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[2] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[3] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n b0 = b[4];\n b1 = b[5];\n b2 = b[6];\n b3 = b[7];\n out[4] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[5] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[6] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[7] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n b0 = b[8];\n b1 = b[9];\n b2 = b[10];\n b3 = b[11];\n out[8] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[9] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[10] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[11] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n b0 = b[12];\n b1 = b[13];\n b2 = b[14];\n b3 = b[15];\n out[12] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[13] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[14] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[15] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n return out;\n}\n/**\n * Translate a mat4 by the given vector\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to translate\n * @param {ReadonlyVec3} v vector to translate by\n * @returns {mat4} out\n */\n\nexport function translate(out, a, v) {\n var x = v[0],\n y = v[1],\n z = v[2];\n var a00, a01, a02, a03;\n var a10, a11, a12, a13;\n var a20, a21, a22, a23;\n\n if (a === out) {\n out[12] = a[0] * x + a[4] * y + a[8] * z + a[12];\n out[13] = a[1] * x + a[5] * y + a[9] * z + a[13];\n out[14] = a[2] * x + a[6] * y + a[10] * z + a[14];\n out[15] = a[3] * x + a[7] * y + a[11] * z + a[15];\n } else {\n a00 = a[0];\n a01 = a[1];\n a02 = a[2];\n a03 = a[3];\n a10 = a[4];\n a11 = a[5];\n a12 = a[6];\n a13 = a[7];\n a20 = a[8];\n a21 = a[9];\n a22 = a[10];\n a23 = a[11];\n out[0] = a00;\n out[1] = a01;\n out[2] = a02;\n out[3] = a03;\n out[4] = a10;\n out[5] = a11;\n out[6] = a12;\n out[7] = a13;\n out[8] = a20;\n out[9] = a21;\n out[10] = a22;\n out[11] = a23;\n out[12] = a00 * x + a10 * y + a20 * z + a[12];\n out[13] = a01 * x + a11 * y + a21 * z + a[13];\n out[14] = a02 * x + a12 * y + a22 * z + a[14];\n out[15] = a03 * x + a13 * y + a23 * z + a[15];\n }\n\n return out;\n}\n/**\n * Scales the mat4 by the dimensions in the given vec3 not using vectorization\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to scale\n * @param {ReadonlyVec3} v the vec3 to scale the matrix by\n * @returns {mat4} out\n **/\n\nexport function scale(out, a, v) {\n var x = v[0],\n y = v[1],\n z = v[2];\n out[0] = a[0] * x;\n out[1] = a[1] * x;\n out[2] = a[2] * x;\n out[3] = a[3] * x;\n out[4] = a[4] * y;\n out[5] = a[5] * y;\n out[6] = a[6] * y;\n out[7] = a[7] * y;\n out[8] = a[8] * z;\n out[9] = a[9] * z;\n out[10] = a[10] * z;\n out[11] = a[11] * z;\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n return out;\n}\n/**\n * Rotates a mat4 by the given angle around the given axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @param {ReadonlyVec3} axis the axis to rotate around\n * @returns {mat4} out\n */\n\nexport function rotate(out, a, rad, axis) {\n var x = axis[0],\n y = axis[1],\n z = axis[2];\n var len = Math.hypot(x, y, z);\n var s, c, t;\n var a00, a01, a02, a03;\n var a10, a11, a12, a13;\n var a20, a21, a22, a23;\n var b00, b01, b02;\n var b10, b11, b12;\n var b20, b21, b22;\n\n if (len < glMatrix.EPSILON) {\n return null;\n }\n\n len = 1 / len;\n x *= len;\n y *= len;\n z *= len;\n s = Math.sin(rad);\n c = Math.cos(rad);\n t = 1 - c;\n a00 = a[0];\n a01 = a[1];\n a02 = a[2];\n a03 = a[3];\n a10 = a[4];\n a11 = a[5];\n a12 = a[6];\n a13 = a[7];\n a20 = a[8];\n a21 = a[9];\n a22 = a[10];\n a23 = a[11]; // Construct the elements of the rotation matrix\n\n b00 = x * x * t + c;\n b01 = y * x * t + z * s;\n b02 = z * x * t - y * s;\n b10 = x * y * t - z * s;\n b11 = y * y * t + c;\n b12 = z * y * t + x * s;\n b20 = x * z * t + y * s;\n b21 = y * z * t - x * s;\n b22 = z * z * t + c; // Perform rotation-specific matrix multiplication\n\n out[0] = a00 * b00 + a10 * b01 + a20 * b02;\n out[1] = a01 * b00 + a11 * b01 + a21 * b02;\n out[2] = a02 * b00 + a12 * b01 + a22 * b02;\n out[3] = a03 * b00 + a13 * b01 + a23 * b02;\n out[4] = a00 * b10 + a10 * b11 + a20 * b12;\n out[5] = a01 * b10 + a11 * b11 + a21 * b12;\n out[6] = a02 * b10 + a12 * b11 + a22 * b12;\n out[7] = a03 * b10 + a13 * b11 + a23 * b12;\n out[8] = a00 * b20 + a10 * b21 + a20 * b22;\n out[9] = a01 * b20 + a11 * b21 + a21 * b22;\n out[10] = a02 * b20 + a12 * b21 + a22 * b22;\n out[11] = a03 * b20 + a13 * b21 + a23 * b22;\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged last row\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n }\n\n return out;\n}\n/**\n * Rotates a matrix by the given angle around the X axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function rotateX(out, a, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n var a10 = a[4];\n var a11 = a[5];\n var a12 = a[6];\n var a13 = a[7];\n var a20 = a[8];\n var a21 = a[9];\n var a22 = a[10];\n var a23 = a[11];\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged rows\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n } // Perform axis-specific matrix multiplication\n\n\n out[4] = a10 * c + a20 * s;\n out[5] = a11 * c + a21 * s;\n out[6] = a12 * c + a22 * s;\n out[7] = a13 * c + a23 * s;\n out[8] = a20 * c - a10 * s;\n out[9] = a21 * c - a11 * s;\n out[10] = a22 * c - a12 * s;\n out[11] = a23 * c - a13 * s;\n return out;\n}\n/**\n * Rotates a matrix by the given angle around the Y axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function rotateY(out, a, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n var a00 = a[0];\n var a01 = a[1];\n var a02 = a[2];\n var a03 = a[3];\n var a20 = a[8];\n var a21 = a[9];\n var a22 = a[10];\n var a23 = a[11];\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged rows\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n } // Perform axis-specific matrix multiplication\n\n\n out[0] = a00 * c - a20 * s;\n out[1] = a01 * c - a21 * s;\n out[2] = a02 * c - a22 * s;\n out[3] = a03 * c - a23 * s;\n out[8] = a00 * s + a20 * c;\n out[9] = a01 * s + a21 * c;\n out[10] = a02 * s + a22 * c;\n out[11] = a03 * s + a23 * c;\n return out;\n}\n/**\n * Rotates a matrix by the given angle around the Z axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function rotateZ(out, a, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n var a00 = a[0];\n var a01 = a[1];\n var a02 = a[2];\n var a03 = a[3];\n var a10 = a[4];\n var a11 = a[5];\n var a12 = a[6];\n var a13 = a[7];\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged last row\n out[8] = a[8];\n out[9] = a[9];\n out[10] = a[10];\n out[11] = a[11];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n } // Perform axis-specific matrix multiplication\n\n\n out[0] = a00 * c + a10 * s;\n out[1] = a01 * c + a11 * s;\n out[2] = a02 * c + a12 * s;\n out[3] = a03 * c + a13 * s;\n out[4] = a10 * c - a00 * s;\n out[5] = a11 * c - a01 * s;\n out[6] = a12 * c - a02 * s;\n out[7] = a13 * c - a03 * s;\n return out;\n}\n/**\n * Creates a matrix from a vector translation\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, dest, vec);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {ReadonlyVec3} v Translation vector\n * @returns {mat4} out\n */\n\nexport function fromTranslation(out, v) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = 1;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 1;\n out[11] = 0;\n out[12] = v[0];\n out[13] = v[1];\n out[14] = v[2];\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a vector scaling\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.scale(dest, dest, vec);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {ReadonlyVec3} v Scaling vector\n * @returns {mat4} out\n */\n\nexport function fromScaling(out, v) {\n out[0] = v[0];\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = v[1];\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = v[2];\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a given angle around a given axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotate(dest, dest, rad, axis);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @param {ReadonlyVec3} axis the axis to rotate around\n * @returns {mat4} out\n */\n\nexport function fromRotation(out, rad, axis) {\n var x = axis[0],\n y = axis[1],\n z = axis[2];\n var len = Math.hypot(x, y, z);\n var s, c, t;\n\n if (len < glMatrix.EPSILON) {\n return null;\n }\n\n len = 1 / len;\n x *= len;\n y *= len;\n z *= len;\n s = Math.sin(rad);\n c = Math.cos(rad);\n t = 1 - c; // Perform rotation-specific matrix multiplication\n\n out[0] = x * x * t + c;\n out[1] = y * x * t + z * s;\n out[2] = z * x * t - y * s;\n out[3] = 0;\n out[4] = x * y * t - z * s;\n out[5] = y * y * t + c;\n out[6] = z * y * t + x * s;\n out[7] = 0;\n out[8] = x * z * t + y * s;\n out[9] = y * z * t - x * s;\n out[10] = z * z * t + c;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from the given angle around the X axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotateX(dest, dest, rad);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function fromXRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad); // Perform axis-specific matrix multiplication\n\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = c;\n out[6] = s;\n out[7] = 0;\n out[8] = 0;\n out[9] = -s;\n out[10] = c;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from the given angle around the Y axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotateY(dest, dest, rad);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function fromYRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad); // Perform axis-specific matrix multiplication\n\n out[0] = c;\n out[1] = 0;\n out[2] = -s;\n out[3] = 0;\n out[4] = 0;\n out[5] = 1;\n out[6] = 0;\n out[7] = 0;\n out[8] = s;\n out[9] = 0;\n out[10] = c;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from the given angle around the Z axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotateZ(dest, dest, rad);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function fromZRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad); // Perform axis-specific matrix multiplication\n\n out[0] = c;\n out[1] = s;\n out[2] = 0;\n out[3] = 0;\n out[4] = -s;\n out[5] = c;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 1;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a quaternion rotation and vector translation\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, vec);\n * let quatMat = mat4.create();\n * quat4.toMat4(quat, quatMat);\n * mat4.multiply(dest, quatMat);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {ReadonlyVec3} v Translation vector\n * @returns {mat4} out\n */\n\nexport function fromRotationTranslation(out, q, v) {\n // Quaternion math\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var xy = x * y2;\n var xz = x * z2;\n var yy = y * y2;\n var yz = y * z2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n out[0] = 1 - (yy + zz);\n out[1] = xy + wz;\n out[2] = xz - wy;\n out[3] = 0;\n out[4] = xy - wz;\n out[5] = 1 - (xx + zz);\n out[6] = yz + wx;\n out[7] = 0;\n out[8] = xz + wy;\n out[9] = yz - wx;\n out[10] = 1 - (xx + yy);\n out[11] = 0;\n out[12] = v[0];\n out[13] = v[1];\n out[14] = v[2];\n out[15] = 1;\n return out;\n}\n/**\n * Creates a new mat4 from a dual quat.\n *\n * @param {mat4} out Matrix\n * @param {ReadonlyQuat2} a Dual Quaternion\n * @returns {mat4} mat4 receiving operation result\n */\n\nexport function fromQuat2(out, a) {\n var translation = new glMatrix.ARRAY_TYPE(3);\n var bx = -a[0],\n by = -a[1],\n bz = -a[2],\n bw = a[3],\n ax = a[4],\n ay = a[5],\n az = a[6],\n aw = a[7];\n var magnitude = bx * bx + by * by + bz * bz + bw * bw; //Only scale if it makes sense\n\n if (magnitude > 0) {\n translation[0] = (ax * bw + aw * bx + ay * bz - az * by) * 2 / magnitude;\n translation[1] = (ay * bw + aw * by + az * bx - ax * bz) * 2 / magnitude;\n translation[2] = (az * bw + aw * bz + ax * by - ay * bx) * 2 / magnitude;\n } else {\n translation[0] = (ax * bw + aw * bx + ay * bz - az * by) * 2;\n translation[1] = (ay * bw + aw * by + az * bx - ax * bz) * 2;\n translation[2] = (az * bw + aw * bz + ax * by - ay * bx) * 2;\n }\n\n fromRotationTranslation(out, a, translation);\n return out;\n}\n/**\n * Returns the translation vector component of a transformation\n * matrix. If a matrix is built with fromRotationTranslation,\n * the returned vector will be the same as the translation vector\n * originally supplied.\n * @param {vec3} out Vector to receive translation component\n * @param {ReadonlyMat4} mat Matrix to be decomposed (input)\n * @return {vec3} out\n */\n\nexport function getTranslation(out, mat) {\n out[0] = mat[12];\n out[1] = mat[13];\n out[2] = mat[14];\n return out;\n}\n/**\n * Returns the scaling factor component of a transformation\n * matrix. If a matrix is built with fromRotationTranslationScale\n * with a normalized Quaternion paramter, the returned vector will be\n * the same as the scaling vector\n * originally supplied.\n * @param {vec3} out Vector to receive scaling factor component\n * @param {ReadonlyMat4} mat Matrix to be decomposed (input)\n * @return {vec3} out\n */\n\nexport function getScaling(out, mat) {\n var m11 = mat[0];\n var m12 = mat[1];\n var m13 = mat[2];\n var m21 = mat[4];\n var m22 = mat[5];\n var m23 = mat[6];\n var m31 = mat[8];\n var m32 = mat[9];\n var m33 = mat[10];\n out[0] = Math.hypot(m11, m12, m13);\n out[1] = Math.hypot(m21, m22, m23);\n out[2] = Math.hypot(m31, m32, m33);\n return out;\n}\n/**\n * Returns a quaternion representing the rotational component\n * of a transformation matrix. If a matrix is built with\n * fromRotationTranslation, the returned quaternion will be the\n * same as the quaternion originally supplied.\n * @param {quat} out Quaternion to receive the rotation component\n * @param {ReadonlyMat4} mat Matrix to be decomposed (input)\n * @return {quat} out\n */\n\nexport function getRotation(out, mat) {\n var scaling = new glMatrix.ARRAY_TYPE(3);\n getScaling(scaling, mat);\n var is1 = 1 / scaling[0];\n var is2 = 1 / scaling[1];\n var is3 = 1 / scaling[2];\n var sm11 = mat[0] * is1;\n var sm12 = mat[1] * is2;\n var sm13 = mat[2] * is3;\n var sm21 = mat[4] * is1;\n var sm22 = mat[5] * is2;\n var sm23 = mat[6] * is3;\n var sm31 = mat[8] * is1;\n var sm32 = mat[9] * is2;\n var sm33 = mat[10] * is3;\n var trace = sm11 + sm22 + sm33;\n var S = 0;\n\n if (trace > 0) {\n S = Math.sqrt(trace + 1.0) * 2;\n out[3] = 0.25 * S;\n out[0] = (sm23 - sm32) / S;\n out[1] = (sm31 - sm13) / S;\n out[2] = (sm12 - sm21) / S;\n } else if (sm11 > sm22 && sm11 > sm33) {\n S = Math.sqrt(1.0 + sm11 - sm22 - sm33) * 2;\n out[3] = (sm23 - sm32) / S;\n out[0] = 0.25 * S;\n out[1] = (sm12 + sm21) / S;\n out[2] = (sm31 + sm13) / S;\n } else if (sm22 > sm33) {\n S = Math.sqrt(1.0 + sm22 - sm11 - sm33) * 2;\n out[3] = (sm31 - sm13) / S;\n out[0] = (sm12 + sm21) / S;\n out[1] = 0.25 * S;\n out[2] = (sm23 + sm32) / S;\n } else {\n S = Math.sqrt(1.0 + sm33 - sm11 - sm22) * 2;\n out[3] = (sm12 - sm21) / S;\n out[0] = (sm31 + sm13) / S;\n out[1] = (sm23 + sm32) / S;\n out[2] = 0.25 * S;\n }\n\n return out;\n}\n/**\n * Creates a matrix from a quaternion rotation, vector translation and vector scale\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, vec);\n * let quatMat = mat4.create();\n * quat4.toMat4(quat, quatMat);\n * mat4.multiply(dest, quatMat);\n * mat4.scale(dest, scale)\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {ReadonlyVec3} v Translation vector\n * @param {ReadonlyVec3} s Scaling vector\n * @returns {mat4} out\n */\n\nexport function fromRotationTranslationScale(out, q, v, s) {\n // Quaternion math\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var xy = x * y2;\n var xz = x * z2;\n var yy = y * y2;\n var yz = y * z2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n var sx = s[0];\n var sy = s[1];\n var sz = s[2];\n out[0] = (1 - (yy + zz)) * sx;\n out[1] = (xy + wz) * sx;\n out[2] = (xz - wy) * sx;\n out[3] = 0;\n out[4] = (xy - wz) * sy;\n out[5] = (1 - (xx + zz)) * sy;\n out[6] = (yz + wx) * sy;\n out[7] = 0;\n out[8] = (xz + wy) * sz;\n out[9] = (yz - wx) * sz;\n out[10] = (1 - (xx + yy)) * sz;\n out[11] = 0;\n out[12] = v[0];\n out[13] = v[1];\n out[14] = v[2];\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a quaternion rotation, vector translation and vector scale, rotating and scaling around the given origin\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, vec);\n * mat4.translate(dest, origin);\n * let quatMat = mat4.create();\n * quat4.toMat4(quat, quatMat);\n * mat4.multiply(dest, quatMat);\n * mat4.scale(dest, scale)\n * mat4.translate(dest, negativeOrigin);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {ReadonlyVec3} v Translation vector\n * @param {ReadonlyVec3} s Scaling vector\n * @param {ReadonlyVec3} o The origin vector around which to scale and rotate\n * @returns {mat4} out\n */\n\nexport function fromRotationTranslationScaleOrigin(out, q, v, s, o) {\n // Quaternion math\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var xy = x * y2;\n var xz = x * z2;\n var yy = y * y2;\n var yz = y * z2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n var sx = s[0];\n var sy = s[1];\n var sz = s[2];\n var ox = o[0];\n var oy = o[1];\n var oz = o[2];\n var out0 = (1 - (yy + zz)) * sx;\n var out1 = (xy + wz) * sx;\n var out2 = (xz - wy) * sx;\n var out4 = (xy - wz) * sy;\n var out5 = (1 - (xx + zz)) * sy;\n var out6 = (yz + wx) * sy;\n var out8 = (xz + wy) * sz;\n var out9 = (yz - wx) * sz;\n var out10 = (1 - (xx + yy)) * sz;\n out[0] = out0;\n out[1] = out1;\n out[2] = out2;\n out[3] = 0;\n out[4] = out4;\n out[5] = out5;\n out[6] = out6;\n out[7] = 0;\n out[8] = out8;\n out[9] = out9;\n out[10] = out10;\n out[11] = 0;\n out[12] = v[0] + ox - (out0 * ox + out4 * oy + out8 * oz);\n out[13] = v[1] + oy - (out1 * ox + out5 * oy + out9 * oz);\n out[14] = v[2] + oz - (out2 * ox + out6 * oy + out10 * oz);\n out[15] = 1;\n return out;\n}\n/**\n * Calculates a 4x4 matrix from the given quaternion\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {ReadonlyQuat} q Quaternion to create matrix from\n *\n * @returns {mat4} out\n */\n\nexport function fromQuat(out, q) {\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var yx = y * x2;\n var yy = y * y2;\n var zx = z * x2;\n var zy = z * y2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n out[0] = 1 - yy - zz;\n out[1] = yx + wz;\n out[2] = zx - wy;\n out[3] = 0;\n out[4] = yx - wz;\n out[5] = 1 - xx - zz;\n out[6] = zy + wx;\n out[7] = 0;\n out[8] = zx + wy;\n out[9] = zy - wx;\n out[10] = 1 - xx - yy;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Generates a frustum matrix with the given bounds\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {Number} left Left bound of the frustum\n * @param {Number} right Right bound of the frustum\n * @param {Number} bottom Bottom bound of the frustum\n * @param {Number} top Top bound of the frustum\n * @param {Number} near Near bound of the frustum\n * @param {Number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function frustum(out, left, right, bottom, top, near, far) {\n var rl = 1 / (right - left);\n var tb = 1 / (top - bottom);\n var nf = 1 / (near - far);\n out[0] = near * 2 * rl;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = near * 2 * tb;\n out[6] = 0;\n out[7] = 0;\n out[8] = (right + left) * rl;\n out[9] = (top + bottom) * tb;\n out[10] = (far + near) * nf;\n out[11] = -1;\n out[12] = 0;\n out[13] = 0;\n out[14] = far * near * 2 * nf;\n out[15] = 0;\n return out;\n}\n/**\n * Generates a perspective projection matrix with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [-1, 1],\n * which matches WebGL/OpenGL's clip volume.\n * Passing null/undefined/no value for far will generate infinite projection matrix.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} fovy Vertical field of view in radians\n * @param {number} aspect Aspect ratio. typically viewport width/height\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum, can be null or Infinity\n * @returns {mat4} out\n */\n\nexport function perspectiveNO(out, fovy, aspect, near, far) {\n var f = 1.0 / Math.tan(fovy / 2),\n nf;\n out[0] = f / aspect;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = f;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[11] = -1;\n out[12] = 0;\n out[13] = 0;\n out[15] = 0;\n\n if (far != null && far !== Infinity) {\n nf = 1 / (near - far);\n out[10] = (far + near) * nf;\n out[14] = 2 * far * near * nf;\n } else {\n out[10] = -1;\n out[14] = -2 * near;\n }\n\n return out;\n}\n/**\n * Alias for {@link mat4.perspectiveNO}\n * @function\n */\n\nexport var perspective = perspectiveNO;\n/**\n * Generates a perspective projection matrix suitable for WebGPU with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [0, 1],\n * which matches WebGPU/Vulkan/DirectX/Metal's clip volume.\n * Passing null/undefined/no value for far will generate infinite projection matrix.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} fovy Vertical field of view in radians\n * @param {number} aspect Aspect ratio. typically viewport width/height\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum, can be null or Infinity\n * @returns {mat4} out\n */\n\nexport function perspectiveZO(out, fovy, aspect, near, far) {\n var f = 1.0 / Math.tan(fovy / 2),\n nf;\n out[0] = f / aspect;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = f;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[11] = -1;\n out[12] = 0;\n out[13] = 0;\n out[15] = 0;\n\n if (far != null && far !== Infinity) {\n nf = 1 / (near - far);\n out[10] = far * nf;\n out[14] = far * near * nf;\n } else {\n out[10] = -1;\n out[14] = -near;\n }\n\n return out;\n}\n/**\n * Generates a perspective projection matrix with the given field of view.\n * This is primarily useful for generating projection matrices to be used\n * with the still experiemental WebVR API.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {Object} fov Object containing the following values: upDegrees, downDegrees, leftDegrees, rightDegrees\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function perspectiveFromFieldOfView(out, fov, near, far) {\n var upTan = Math.tan(fov.upDegrees * Math.PI / 180.0);\n var downTan = Math.tan(fov.downDegrees * Math.PI / 180.0);\n var leftTan = Math.tan(fov.leftDegrees * Math.PI / 180.0);\n var rightTan = Math.tan(fov.rightDegrees * Math.PI / 180.0);\n var xScale = 2.0 / (leftTan + rightTan);\n var yScale = 2.0 / (upTan + downTan);\n out[0] = xScale;\n out[1] = 0.0;\n out[2] = 0.0;\n out[3] = 0.0;\n out[4] = 0.0;\n out[5] = yScale;\n out[6] = 0.0;\n out[7] = 0.0;\n out[8] = -((leftTan - rightTan) * xScale * 0.5);\n out[9] = (upTan - downTan) * yScale * 0.5;\n out[10] = far / (near - far);\n out[11] = -1.0;\n out[12] = 0.0;\n out[13] = 0.0;\n out[14] = far * near / (near - far);\n out[15] = 0.0;\n return out;\n}\n/**\n * Generates a orthogonal projection matrix with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [-1, 1],\n * which matches WebGL/OpenGL's clip volume.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} left Left bound of the frustum\n * @param {number} right Right bound of the frustum\n * @param {number} bottom Bottom bound of the frustum\n * @param {number} top Top bound of the frustum\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function orthoNO(out, left, right, bottom, top, near, far) {\n var lr = 1 / (left - right);\n var bt = 1 / (bottom - top);\n var nf = 1 / (near - far);\n out[0] = -2 * lr;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = -2 * bt;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 2 * nf;\n out[11] = 0;\n out[12] = (left + right) * lr;\n out[13] = (top + bottom) * bt;\n out[14] = (far + near) * nf;\n out[15] = 1;\n return out;\n}\n/**\n * Alias for {@link mat4.orthoNO}\n * @function\n */\n\nexport var ortho = orthoNO;\n/**\n * Generates a orthogonal projection matrix with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [0, 1],\n * which matches WebGPU/Vulkan/DirectX/Metal's clip volume.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} left Left bound of the frustum\n * @param {number} right Right bound of the frustum\n * @param {number} bottom Bottom bound of the frustum\n * @param {number} top Top bound of the frustum\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function orthoZO(out, left, right, bottom, top, near, far) {\n var lr = 1 / (left - right);\n var bt = 1 / (bottom - top);\n var nf = 1 / (near - far);\n out[0] = -2 * lr;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = -2 * bt;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = nf;\n out[11] = 0;\n out[12] = (left + right) * lr;\n out[13] = (top + bottom) * bt;\n out[14] = near * nf;\n out[15] = 1;\n return out;\n}\n/**\n * Generates a look-at matrix with the given eye position, focal point, and up axis.\n * If you want a matrix that actually makes an object look at another object, you should use targetTo instead.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {ReadonlyVec3} eye Position of the viewer\n * @param {ReadonlyVec3} center Point the viewer is looking at\n * @param {ReadonlyVec3} up vec3 pointing up\n * @returns {mat4} out\n */\n\nexport function lookAt(out, eye, center, up) {\n var x0, x1, x2, y0, y1, y2, z0, z1, z2, len;\n var eyex = eye[0];\n var eyey = eye[1];\n var eyez = eye[2];\n var upx = up[0];\n var upy = up[1];\n var upz = up[2];\n var centerx = center[0];\n var centery = center[1];\n var centerz = center[2];\n\n if (Math.abs(eyex - centerx) < glMatrix.EPSILON && Math.abs(eyey - centery) < glMatrix.EPSILON && Math.abs(eyez - centerz) < glMatrix.EPSILON) {\n return identity(out);\n }\n\n z0 = eyex - centerx;\n z1 = eyey - centery;\n z2 = eyez - centerz;\n len = 1 / Math.hypot(z0, z1, z2);\n z0 *= len;\n z1 *= len;\n z2 *= len;\n x0 = upy * z2 - upz * z1;\n x1 = upz * z0 - upx * z2;\n x2 = upx * z1 - upy * z0;\n len = Math.hypot(x0, x1, x2);\n\n if (!len) {\n x0 = 0;\n x1 = 0;\n x2 = 0;\n } else {\n len = 1 / len;\n x0 *= len;\n x1 *= len;\n x2 *= len;\n }\n\n y0 = z1 * x2 - z2 * x1;\n y1 = z2 * x0 - z0 * x2;\n y2 = z0 * x1 - z1 * x0;\n len = Math.hypot(y0, y1, y2);\n\n if (!len) {\n y0 = 0;\n y1 = 0;\n y2 = 0;\n } else {\n len = 1 / len;\n y0 *= len;\n y1 *= len;\n y2 *= len;\n }\n\n out[0] = x0;\n out[1] = y0;\n out[2] = z0;\n out[3] = 0;\n out[4] = x1;\n out[5] = y1;\n out[6] = z1;\n out[7] = 0;\n out[8] = x2;\n out[9] = y2;\n out[10] = z2;\n out[11] = 0;\n out[12] = -(x0 * eyex + x1 * eyey + x2 * eyez);\n out[13] = -(y0 * eyex + y1 * eyey + y2 * eyez);\n out[14] = -(z0 * eyex + z1 * eyey + z2 * eyez);\n out[15] = 1;\n return out;\n}\n/**\n * Generates a matrix that makes something look at something else.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {ReadonlyVec3} eye Position of the viewer\n * @param {ReadonlyVec3} center Point the viewer is looking at\n * @param {ReadonlyVec3} up vec3 pointing up\n * @returns {mat4} out\n */\n\nexport function targetTo(out, eye, target, up) {\n var eyex = eye[0],\n eyey = eye[1],\n eyez = eye[2],\n upx = up[0],\n upy = up[1],\n upz = up[2];\n var z0 = eyex - target[0],\n z1 = eyey - target[1],\n z2 = eyez - target[2];\n var len = z0 * z0 + z1 * z1 + z2 * z2;\n\n if (len > 0) {\n len = 1 / Math.sqrt(len);\n z0 *= len;\n z1 *= len;\n z2 *= len;\n }\n\n var x0 = upy * z2 - upz * z1,\n x1 = upz * z0 - upx * z2,\n x2 = upx * z1 - upy * z0;\n len = x0 * x0 + x1 * x1 + x2 * x2;\n\n if (len > 0) {\n len = 1 / Math.sqrt(len);\n x0 *= len;\n x1 *= len;\n x2 *= len;\n }\n\n out[0] = x0;\n out[1] = x1;\n out[2] = x2;\n out[3] = 0;\n out[4] = z1 * x2 - z2 * x1;\n out[5] = z2 * x0 - z0 * x2;\n out[6] = z0 * x1 - z1 * x0;\n out[7] = 0;\n out[8] = z0;\n out[9] = z1;\n out[10] = z2;\n out[11] = 0;\n out[12] = eyex;\n out[13] = eyey;\n out[14] = eyez;\n out[15] = 1;\n return out;\n}\n/**\n * Returns a string representation of a mat4\n *\n * @param {ReadonlyMat4} a matrix to represent as a string\n * @returns {String} string representation of the matrix\n */\n\nexport function str(a) {\n return \"mat4(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \", \" + a[4] + \", \" + a[5] + \", \" + a[6] + \", \" + a[7] + \", \" + a[8] + \", \" + a[9] + \", \" + a[10] + \", \" + a[11] + \", \" + a[12] + \", \" + a[13] + \", \" + a[14] + \", \" + a[15] + \")\";\n}\n/**\n * Returns Frobenius norm of a mat4\n *\n * @param {ReadonlyMat4} a the matrix to calculate Frobenius norm of\n * @returns {Number} Frobenius norm\n */\n\nexport function frob(a) {\n return Math.hypot(a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8], a[9], a[10], a[11], a[12], a[13], a[14], a[15]);\n}\n/**\n * Adds two mat4's\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @returns {mat4} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n out[4] = a[4] + b[4];\n out[5] = a[5] + b[5];\n out[6] = a[6] + b[6];\n out[7] = a[7] + b[7];\n out[8] = a[8] + b[8];\n out[9] = a[9] + b[9];\n out[10] = a[10] + b[10];\n out[11] = a[11] + b[11];\n out[12] = a[12] + b[12];\n out[13] = a[13] + b[13];\n out[14] = a[14] + b[14];\n out[15] = a[15] + b[15];\n return out;\n}\n/**\n * Subtracts matrix b from matrix a\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @returns {mat4} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n out[4] = a[4] - b[4];\n out[5] = a[5] - b[5];\n out[6] = a[6] - b[6];\n out[7] = a[7] - b[7];\n out[8] = a[8] - b[8];\n out[9] = a[9] - b[9];\n out[10] = a[10] - b[10];\n out[11] = a[11] - b[11];\n out[12] = a[12] - b[12];\n out[13] = a[13] - b[13];\n out[14] = a[14] - b[14];\n out[15] = a[15] - b[15];\n return out;\n}\n/**\n * Multiply each element of the matrix by a scalar.\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to scale\n * @param {Number} b amount to scale the matrix's elements by\n * @returns {mat4} out\n */\n\nexport function multiplyScalar(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n out[4] = a[4] * b;\n out[5] = a[5] * b;\n out[6] = a[6] * b;\n out[7] = a[7] * b;\n out[8] = a[8] * b;\n out[9] = a[9] * b;\n out[10] = a[10] * b;\n out[11] = a[11] * b;\n out[12] = a[12] * b;\n out[13] = a[13] * b;\n out[14] = a[14] * b;\n out[15] = a[15] * b;\n return out;\n}\n/**\n * Adds two mat4's after multiplying each element of the second operand by a scalar value.\n *\n * @param {mat4} out the receiving vector\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @param {Number} scale the amount to scale b's elements by before adding\n * @returns {mat4} out\n */\n\nexport function multiplyScalarAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n out[4] = a[4] + b[4] * scale;\n out[5] = a[5] + b[5] * scale;\n out[6] = a[6] + b[6] * scale;\n out[7] = a[7] + b[7] * scale;\n out[8] = a[8] + b[8] * scale;\n out[9] = a[9] + b[9] * scale;\n out[10] = a[10] + b[10] * scale;\n out[11] = a[11] + b[11] * scale;\n out[12] = a[12] + b[12] * scale;\n out[13] = a[13] + b[13] * scale;\n out[14] = a[14] + b[14] * scale;\n out[15] = a[15] + b[15] * scale;\n return out;\n}\n/**\n * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyMat4} a The first matrix.\n * @param {ReadonlyMat4} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5] && a[6] === b[6] && a[7] === b[7] && a[8] === b[8] && a[9] === b[9] && a[10] === b[10] && a[11] === b[11] && a[12] === b[12] && a[13] === b[13] && a[14] === b[14] && a[15] === b[15];\n}\n/**\n * Returns whether or not the matrices have approximately the same elements in the same position.\n *\n * @param {ReadonlyMat4} a The first matrix.\n * @param {ReadonlyMat4} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var a4 = a[4],\n a5 = a[5],\n a6 = a[6],\n a7 = a[7];\n var a8 = a[8],\n a9 = a[9],\n a10 = a[10],\n a11 = a[11];\n var a12 = a[12],\n a13 = a[13],\n a14 = a[14],\n a15 = a[15];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n var b4 = b[4],\n b5 = b[5],\n b6 = b[6],\n b7 = b[7];\n var b8 = b[8],\n b9 = b[9],\n b10 = b[10],\n b11 = b[11];\n var b12 = b[12],\n b13 = b[13],\n b14 = b[14],\n b15 = b[15];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5)) && Math.abs(a6 - b6) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a6), Math.abs(b6)) && Math.abs(a7 - b7) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a7), Math.abs(b7)) && Math.abs(a8 - b8) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a8), Math.abs(b8)) && Math.abs(a9 - b9) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a9), Math.abs(b9)) && Math.abs(a10 - b10) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a10), Math.abs(b10)) && Math.abs(a11 - b11) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a11), Math.abs(b11)) && Math.abs(a12 - b12) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a12), Math.abs(b12)) && Math.abs(a13 - b13) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a13), Math.abs(b13)) && Math.abs(a14 - b14) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a14), Math.abs(b14)) && Math.abs(a15 - b15) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a15), Math.abs(b15));\n}\n/**\n * Alias for {@link mat4.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link mat4.subtract}\n * @function\n */\n\nexport var sub = subtract;","import * as glMatrix from \"./common.js\";\n/**\n * 3 Dimensional Vector\n * @module vec3\n */\n\n/**\n * Creates a new, empty vec3\n *\n * @returns {vec3} a new 3D vector\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(3);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[0] = 0;\n out[1] = 0;\n out[2] = 0;\n }\n\n return out;\n}\n/**\n * Creates a new vec3 initialized with values from an existing vector\n *\n * @param {ReadonlyVec3} a vector to clone\n * @returns {vec3} a new 3D vector\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(3);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n return out;\n}\n/**\n * Calculates the length of a vec3\n *\n * @param {ReadonlyVec3} a vector to calculate length of\n * @returns {Number} length of a\n */\n\nexport function length(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n return Math.hypot(x, y, z);\n}\n/**\n * Creates a new vec3 initialized with the given values\n *\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @returns {vec3} a new 3D vector\n */\n\nexport function fromValues(x, y, z) {\n var out = new glMatrix.ARRAY_TYPE(3);\n out[0] = x;\n out[1] = y;\n out[2] = z;\n return out;\n}\n/**\n * Copy the values from one vec3 to another\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the source vector\n * @returns {vec3} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n return out;\n}\n/**\n * Set the components of a vec3 to the given values\n *\n * @param {vec3} out the receiving vector\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @returns {vec3} out\n */\n\nexport function set(out, x, y, z) {\n out[0] = x;\n out[1] = y;\n out[2] = z;\n return out;\n}\n/**\n * Adds two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n return out;\n}\n/**\n * Subtracts vector b from vector a\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n return out;\n}\n/**\n * Multiplies two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function multiply(out, a, b) {\n out[0] = a[0] * b[0];\n out[1] = a[1] * b[1];\n out[2] = a[2] * b[2];\n return out;\n}\n/**\n * Divides two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function divide(out, a, b) {\n out[0] = a[0] / b[0];\n out[1] = a[1] / b[1];\n out[2] = a[2] / b[2];\n return out;\n}\n/**\n * Math.ceil the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to ceil\n * @returns {vec3} out\n */\n\nexport function ceil(out, a) {\n out[0] = Math.ceil(a[0]);\n out[1] = Math.ceil(a[1]);\n out[2] = Math.ceil(a[2]);\n return out;\n}\n/**\n * Math.floor the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to floor\n * @returns {vec3} out\n */\n\nexport function floor(out, a) {\n out[0] = Math.floor(a[0]);\n out[1] = Math.floor(a[1]);\n out[2] = Math.floor(a[2]);\n return out;\n}\n/**\n * Returns the minimum of two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function min(out, a, b) {\n out[0] = Math.min(a[0], b[0]);\n out[1] = Math.min(a[1], b[1]);\n out[2] = Math.min(a[2], b[2]);\n return out;\n}\n/**\n * Returns the maximum of two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function max(out, a, b) {\n out[0] = Math.max(a[0], b[0]);\n out[1] = Math.max(a[1], b[1]);\n out[2] = Math.max(a[2], b[2]);\n return out;\n}\n/**\n * Math.round the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to round\n * @returns {vec3} out\n */\n\nexport function round(out, a) {\n out[0] = Math.round(a[0]);\n out[1] = Math.round(a[1]);\n out[2] = Math.round(a[2]);\n return out;\n}\n/**\n * Scales a vec3 by a scalar number\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the vector to scale\n * @param {Number} b amount to scale the vector by\n * @returns {vec3} out\n */\n\nexport function scale(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n return out;\n}\n/**\n * Adds two vec3's after scaling the second operand by a scalar value\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @param {Number} scale the amount to scale b by before adding\n * @returns {vec3} out\n */\n\nexport function scaleAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n return out;\n}\n/**\n * Calculates the euclidian distance between two vec3's\n *\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {Number} distance between a and b\n */\n\nexport function distance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n return Math.hypot(x, y, z);\n}\n/**\n * Calculates the squared euclidian distance between two vec3's\n *\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {Number} squared distance between a and b\n */\n\nexport function squaredDistance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n return x * x + y * y + z * z;\n}\n/**\n * Calculates the squared length of a vec3\n *\n * @param {ReadonlyVec3} a vector to calculate squared length of\n * @returns {Number} squared length of a\n */\n\nexport function squaredLength(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n return x * x + y * y + z * z;\n}\n/**\n * Negates the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to negate\n * @returns {vec3} out\n */\n\nexport function negate(out, a) {\n out[0] = -a[0];\n out[1] = -a[1];\n out[2] = -a[2];\n return out;\n}\n/**\n * Returns the inverse of the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to invert\n * @returns {vec3} out\n */\n\nexport function inverse(out, a) {\n out[0] = 1.0 / a[0];\n out[1] = 1.0 / a[1];\n out[2] = 1.0 / a[2];\n return out;\n}\n/**\n * Normalize a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to normalize\n * @returns {vec3} out\n */\n\nexport function normalize(out, a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var len = x * x + y * y + z * z;\n\n if (len > 0) {\n //TODO: evaluate use of glm_invsqrt here?\n len = 1 / Math.sqrt(len);\n }\n\n out[0] = a[0] * len;\n out[1] = a[1] * len;\n out[2] = a[2] * len;\n return out;\n}\n/**\n * Calculates the dot product of two vec3's\n *\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {Number} dot product of a and b\n */\n\nexport function dot(a, b) {\n return a[0] * b[0] + a[1] * b[1] + a[2] * b[2];\n}\n/**\n * Computes the cross product of two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function cross(out, a, b) {\n var ax = a[0],\n ay = a[1],\n az = a[2];\n var bx = b[0],\n by = b[1],\n bz = b[2];\n out[0] = ay * bz - az * by;\n out[1] = az * bx - ax * bz;\n out[2] = ax * by - ay * bx;\n return out;\n}\n/**\n * Performs a linear interpolation between two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec3} out\n */\n\nexport function lerp(out, a, b, t) {\n var ax = a[0];\n var ay = a[1];\n var az = a[2];\n out[0] = ax + t * (b[0] - ax);\n out[1] = ay + t * (b[1] - ay);\n out[2] = az + t * (b[2] - az);\n return out;\n}\n/**\n * Performs a hermite interpolation with two control points\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @param {ReadonlyVec3} c the third operand\n * @param {ReadonlyVec3} d the fourth operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec3} out\n */\n\nexport function hermite(out, a, b, c, d, t) {\n var factorTimes2 = t * t;\n var factor1 = factorTimes2 * (2 * t - 3) + 1;\n var factor2 = factorTimes2 * (t - 2) + t;\n var factor3 = factorTimes2 * (t - 1);\n var factor4 = factorTimes2 * (3 - 2 * t);\n out[0] = a[0] * factor1 + b[0] * factor2 + c[0] * factor3 + d[0] * factor4;\n out[1] = a[1] * factor1 + b[1] * factor2 + c[1] * factor3 + d[1] * factor4;\n out[2] = a[2] * factor1 + b[2] * factor2 + c[2] * factor3 + d[2] * factor4;\n return out;\n}\n/**\n * Performs a bezier interpolation with two control points\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @param {ReadonlyVec3} c the third operand\n * @param {ReadonlyVec3} d the fourth operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec3} out\n */\n\nexport function bezier(out, a, b, c, d, t) {\n var inverseFactor = 1 - t;\n var inverseFactorTimesTwo = inverseFactor * inverseFactor;\n var factorTimes2 = t * t;\n var factor1 = inverseFactorTimesTwo * inverseFactor;\n var factor2 = 3 * t * inverseFactorTimesTwo;\n var factor3 = 3 * factorTimes2 * inverseFactor;\n var factor4 = factorTimes2 * t;\n out[0] = a[0] * factor1 + b[0] * factor2 + c[0] * factor3 + d[0] * factor4;\n out[1] = a[1] * factor1 + b[1] * factor2 + c[1] * factor3 + d[1] * factor4;\n out[2] = a[2] * factor1 + b[2] * factor2 + c[2] * factor3 + d[2] * factor4;\n return out;\n}\n/**\n * Generates a random vector with the given scale\n *\n * @param {vec3} out the receiving vector\n * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned\n * @returns {vec3} out\n */\n\nexport function random(out, scale) {\n scale = scale || 1.0;\n var r = glMatrix.RANDOM() * 2.0 * Math.PI;\n var z = glMatrix.RANDOM() * 2.0 - 1.0;\n var zScale = Math.sqrt(1.0 - z * z) * scale;\n out[0] = Math.cos(r) * zScale;\n out[1] = Math.sin(r) * zScale;\n out[2] = z * scale;\n return out;\n}\n/**\n * Transforms the vec3 with a mat4.\n * 4th vector component is implicitly '1'\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the vector to transform\n * @param {ReadonlyMat4} m matrix to transform with\n * @returns {vec3} out\n */\n\nexport function transformMat4(out, a, m) {\n var x = a[0],\n y = a[1],\n z = a[2];\n var w = m[3] * x + m[7] * y + m[11] * z + m[15];\n w = w || 1.0;\n out[0] = (m[0] * x + m[4] * y + m[8] * z + m[12]) / w;\n out[1] = (m[1] * x + m[5] * y + m[9] * z + m[13]) / w;\n out[2] = (m[2] * x + m[6] * y + m[10] * z + m[14]) / w;\n return out;\n}\n/**\n * Transforms the vec3 with a mat3.\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the vector to transform\n * @param {ReadonlyMat3} m the 3x3 matrix to transform with\n * @returns {vec3} out\n */\n\nexport function transformMat3(out, a, m) {\n var x = a[0],\n y = a[1],\n z = a[2];\n out[0] = x * m[0] + y * m[3] + z * m[6];\n out[1] = x * m[1] + y * m[4] + z * m[7];\n out[2] = x * m[2] + y * m[5] + z * m[8];\n return out;\n}\n/**\n * Transforms the vec3 with a quat\n * Can also be used for dual quaternions. (Multiply it with the real part)\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the vector to transform\n * @param {ReadonlyQuat} q quaternion to transform with\n * @returns {vec3} out\n */\n\nexport function transformQuat(out, a, q) {\n // benchmarks: https://jsperf.com/quaternion-transform-vec3-implementations-fixed\n var qx = q[0],\n qy = q[1],\n qz = q[2],\n qw = q[3];\n var x = a[0],\n y = a[1],\n z = a[2]; // var qvec = [qx, qy, qz];\n // var uv = vec3.cross([], qvec, a);\n\n var uvx = qy * z - qz * y,\n uvy = qz * x - qx * z,\n uvz = qx * y - qy * x; // var uuv = vec3.cross([], qvec, uv);\n\n var uuvx = qy * uvz - qz * uvy,\n uuvy = qz * uvx - qx * uvz,\n uuvz = qx * uvy - qy * uvx; // vec3.scale(uv, uv, 2 * w);\n\n var w2 = qw * 2;\n uvx *= w2;\n uvy *= w2;\n uvz *= w2; // vec3.scale(uuv, uuv, 2);\n\n uuvx *= 2;\n uuvy *= 2;\n uuvz *= 2; // return vec3.add(out, a, vec3.add(out, uv, uuv));\n\n out[0] = x + uvx + uuvx;\n out[1] = y + uvy + uuvy;\n out[2] = z + uvz + uuvz;\n return out;\n}\n/**\n * Rotate a 3D vector around the x-axis\n * @param {vec3} out The receiving vec3\n * @param {ReadonlyVec3} a The vec3 point to rotate\n * @param {ReadonlyVec3} b The origin of the rotation\n * @param {Number} rad The angle of rotation in radians\n * @returns {vec3} out\n */\n\nexport function rotateX(out, a, b, rad) {\n var p = [],\n r = []; //Translate point to the origin\n\n p[0] = a[0] - b[0];\n p[1] = a[1] - b[1];\n p[2] = a[2] - b[2]; //perform rotation\n\n r[0] = p[0];\n r[1] = p[1] * Math.cos(rad) - p[2] * Math.sin(rad);\n r[2] = p[1] * Math.sin(rad) + p[2] * Math.cos(rad); //translate to correct position\n\n out[0] = r[0] + b[0];\n out[1] = r[1] + b[1];\n out[2] = r[2] + b[2];\n return out;\n}\n/**\n * Rotate a 3D vector around the y-axis\n * @param {vec3} out The receiving vec3\n * @param {ReadonlyVec3} a The vec3 point to rotate\n * @param {ReadonlyVec3} b The origin of the rotation\n * @param {Number} rad The angle of rotation in radians\n * @returns {vec3} out\n */\n\nexport function rotateY(out, a, b, rad) {\n var p = [],\n r = []; //Translate point to the origin\n\n p[0] = a[0] - b[0];\n p[1] = a[1] - b[1];\n p[2] = a[2] - b[2]; //perform rotation\n\n r[0] = p[2] * Math.sin(rad) + p[0] * Math.cos(rad);\n r[1] = p[1];\n r[2] = p[2] * Math.cos(rad) - p[0] * Math.sin(rad); //translate to correct position\n\n out[0] = r[0] + b[0];\n out[1] = r[1] + b[1];\n out[2] = r[2] + b[2];\n return out;\n}\n/**\n * Rotate a 3D vector around the z-axis\n * @param {vec3} out The receiving vec3\n * @param {ReadonlyVec3} a The vec3 point to rotate\n * @param {ReadonlyVec3} b The origin of the rotation\n * @param {Number} rad The angle of rotation in radians\n * @returns {vec3} out\n */\n\nexport function rotateZ(out, a, b, rad) {\n var p = [],\n r = []; //Translate point to the origin\n\n p[0] = a[0] - b[0];\n p[1] = a[1] - b[1];\n p[2] = a[2] - b[2]; //perform rotation\n\n r[0] = p[0] * Math.cos(rad) - p[1] * Math.sin(rad);\n r[1] = p[0] * Math.sin(rad) + p[1] * Math.cos(rad);\n r[2] = p[2]; //translate to correct position\n\n out[0] = r[0] + b[0];\n out[1] = r[1] + b[1];\n out[2] = r[2] + b[2];\n return out;\n}\n/**\n * Get the angle between two 3D vectors\n * @param {ReadonlyVec3} a The first operand\n * @param {ReadonlyVec3} b The second operand\n * @returns {Number} The angle in radians\n */\n\nexport function angle(a, b) {\n var ax = a[0],\n ay = a[1],\n az = a[2],\n bx = b[0],\n by = b[1],\n bz = b[2],\n mag1 = Math.sqrt(ax * ax + ay * ay + az * az),\n mag2 = Math.sqrt(bx * bx + by * by + bz * bz),\n mag = mag1 * mag2,\n cosine = mag && dot(a, b) / mag;\n return Math.acos(Math.min(Math.max(cosine, -1), 1));\n}\n/**\n * Set the components of a vec3 to zero\n *\n * @param {vec3} out the receiving vector\n * @returns {vec3} out\n */\n\nexport function zero(out) {\n out[0] = 0.0;\n out[1] = 0.0;\n out[2] = 0.0;\n return out;\n}\n/**\n * Returns a string representation of a vector\n *\n * @param {ReadonlyVec3} a vector to represent as a string\n * @returns {String} string representation of the vector\n */\n\nexport function str(a) {\n return \"vec3(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \")\";\n}\n/**\n * Returns whether or not the vectors have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyVec3} a The first vector.\n * @param {ReadonlyVec3} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2];\n}\n/**\n * Returns whether or not the vectors have approximately the same elements in the same position.\n *\n * @param {ReadonlyVec3} a The first vector.\n * @param {ReadonlyVec3} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2));\n}\n/**\n * Alias for {@link vec3.subtract}\n * @function\n */\n\nexport var sub = subtract;\n/**\n * Alias for {@link vec3.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link vec3.divide}\n * @function\n */\n\nexport var div = divide;\n/**\n * Alias for {@link vec3.distance}\n * @function\n */\n\nexport var dist = distance;\n/**\n * Alias for {@link vec3.squaredDistance}\n * @function\n */\n\nexport var sqrDist = squaredDistance;\n/**\n * Alias for {@link vec3.length}\n * @function\n */\n\nexport var len = length;\n/**\n * Alias for {@link vec3.squaredLength}\n * @function\n */\n\nexport var sqrLen = squaredLength;\n/**\n * Perform some operation over an array of vec3s.\n *\n * @param {Array} a the array of vectors to iterate over\n * @param {Number} stride Number of elements between the start of each vec3. If 0 assumes tightly packed\n * @param {Number} offset Number of elements to skip at the beginning of the array\n * @param {Number} count Number of vec3s to iterate over. If 0 iterates over entire array\n * @param {Function} fn Function to call for each vector in the array\n * @param {Object} [arg] additional argument to pass to fn\n * @returns {Array} a\n * @function\n */\n\nexport var forEach = function () {\n var vec = create();\n return function (a, stride, offset, count, fn, arg) {\n var i, l;\n\n if (!stride) {\n stride = 3;\n }\n\n if (!offset) {\n offset = 0;\n }\n\n if (count) {\n l = Math.min(count * stride + offset, a.length);\n } else {\n l = a.length;\n }\n\n for (i = offset; i < l; i += stride) {\n vec[0] = a[i];\n vec[1] = a[i + 1];\n vec[2] = a[i + 2];\n fn(vec, vec, arg);\n a[i] = vec[0];\n a[i + 1] = vec[1];\n a[i + 2] = vec[2];\n }\n\n return a;\n };\n}();","import * as glMatrix from \"./common.js\";\n/**\n * 4 Dimensional Vector\n * @module vec4\n */\n\n/**\n * Creates a new, empty vec4\n *\n * @returns {vec4} a new 4D vector\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(4);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[0] = 0;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n }\n\n return out;\n}\n/**\n * Creates a new vec4 initialized with values from an existing vector\n *\n * @param {ReadonlyVec4} a vector to clone\n * @returns {vec4} a new 4D vector\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Creates a new vec4 initialized with the given values\n *\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @param {Number} w W component\n * @returns {vec4} a new 4D vector\n */\n\nexport function fromValues(x, y, z, w) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = x;\n out[1] = y;\n out[2] = z;\n out[3] = w;\n return out;\n}\n/**\n * Copy the values from one vec4 to another\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the source vector\n * @returns {vec4} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Set the components of a vec4 to the given values\n *\n * @param {vec4} out the receiving vector\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @param {Number} w W component\n * @returns {vec4} out\n */\n\nexport function set(out, x, y, z, w) {\n out[0] = x;\n out[1] = y;\n out[2] = z;\n out[3] = w;\n return out;\n}\n/**\n * Adds two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n return out;\n}\n/**\n * Subtracts vector b from vector a\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n return out;\n}\n/**\n * Multiplies two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function multiply(out, a, b) {\n out[0] = a[0] * b[0];\n out[1] = a[1] * b[1];\n out[2] = a[2] * b[2];\n out[3] = a[3] * b[3];\n return out;\n}\n/**\n * Divides two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function divide(out, a, b) {\n out[0] = a[0] / b[0];\n out[1] = a[1] / b[1];\n out[2] = a[2] / b[2];\n out[3] = a[3] / b[3];\n return out;\n}\n/**\n * Math.ceil the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to ceil\n * @returns {vec4} out\n */\n\nexport function ceil(out, a) {\n out[0] = Math.ceil(a[0]);\n out[1] = Math.ceil(a[1]);\n out[2] = Math.ceil(a[2]);\n out[3] = Math.ceil(a[3]);\n return out;\n}\n/**\n * Math.floor the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to floor\n * @returns {vec4} out\n */\n\nexport function floor(out, a) {\n out[0] = Math.floor(a[0]);\n out[1] = Math.floor(a[1]);\n out[2] = Math.floor(a[2]);\n out[3] = Math.floor(a[3]);\n return out;\n}\n/**\n * Returns the minimum of two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function min(out, a, b) {\n out[0] = Math.min(a[0], b[0]);\n out[1] = Math.min(a[1], b[1]);\n out[2] = Math.min(a[2], b[2]);\n out[3] = Math.min(a[3], b[3]);\n return out;\n}\n/**\n * Returns the maximum of two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function max(out, a, b) {\n out[0] = Math.max(a[0], b[0]);\n out[1] = Math.max(a[1], b[1]);\n out[2] = Math.max(a[2], b[2]);\n out[3] = Math.max(a[3], b[3]);\n return out;\n}\n/**\n * Math.round the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to round\n * @returns {vec4} out\n */\n\nexport function round(out, a) {\n out[0] = Math.round(a[0]);\n out[1] = Math.round(a[1]);\n out[2] = Math.round(a[2]);\n out[3] = Math.round(a[3]);\n return out;\n}\n/**\n * Scales a vec4 by a scalar number\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the vector to scale\n * @param {Number} b amount to scale the vector by\n * @returns {vec4} out\n */\n\nexport function scale(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n return out;\n}\n/**\n * Adds two vec4's after scaling the second operand by a scalar value\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @param {Number} scale the amount to scale b by before adding\n * @returns {vec4} out\n */\n\nexport function scaleAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n return out;\n}\n/**\n * Calculates the euclidian distance between two vec4's\n *\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {Number} distance between a and b\n */\n\nexport function distance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n var w = b[3] - a[3];\n return Math.hypot(x, y, z, w);\n}\n/**\n * Calculates the squared euclidian distance between two vec4's\n *\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {Number} squared distance between a and b\n */\n\nexport function squaredDistance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n var w = b[3] - a[3];\n return x * x + y * y + z * z + w * w;\n}\n/**\n * Calculates the length of a vec4\n *\n * @param {ReadonlyVec4} a vector to calculate length of\n * @returns {Number} length of a\n */\n\nexport function length(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var w = a[3];\n return Math.hypot(x, y, z, w);\n}\n/**\n * Calculates the squared length of a vec4\n *\n * @param {ReadonlyVec4} a vector to calculate squared length of\n * @returns {Number} squared length of a\n */\n\nexport function squaredLength(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var w = a[3];\n return x * x + y * y + z * z + w * w;\n}\n/**\n * Negates the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to negate\n * @returns {vec4} out\n */\n\nexport function negate(out, a) {\n out[0] = -a[0];\n out[1] = -a[1];\n out[2] = -a[2];\n out[3] = -a[3];\n return out;\n}\n/**\n * Returns the inverse of the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to invert\n * @returns {vec4} out\n */\n\nexport function inverse(out, a) {\n out[0] = 1.0 / a[0];\n out[1] = 1.0 / a[1];\n out[2] = 1.0 / a[2];\n out[3] = 1.0 / a[3];\n return out;\n}\n/**\n * Normalize a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to normalize\n * @returns {vec4} out\n */\n\nexport function normalize(out, a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var w = a[3];\n var len = x * x + y * y + z * z + w * w;\n\n if (len > 0) {\n len = 1 / Math.sqrt(len);\n }\n\n out[0] = x * len;\n out[1] = y * len;\n out[2] = z * len;\n out[3] = w * len;\n return out;\n}\n/**\n * Calculates the dot product of two vec4's\n *\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {Number} dot product of a and b\n */\n\nexport function dot(a, b) {\n return a[0] * b[0] + a[1] * b[1] + a[2] * b[2] + a[3] * b[3];\n}\n/**\n * Returns the cross-product of three vectors in a 4-dimensional space\n *\n * @param {ReadonlyVec4} result the receiving vector\n * @param {ReadonlyVec4} U the first vector\n * @param {ReadonlyVec4} V the second vector\n * @param {ReadonlyVec4} W the third vector\n * @returns {vec4} result\n */\n\nexport function cross(out, u, v, w) {\n var A = v[0] * w[1] - v[1] * w[0],\n B = v[0] * w[2] - v[2] * w[0],\n C = v[0] * w[3] - v[3] * w[0],\n D = v[1] * w[2] - v[2] * w[1],\n E = v[1] * w[3] - v[3] * w[1],\n F = v[2] * w[3] - v[3] * w[2];\n var G = u[0];\n var H = u[1];\n var I = u[2];\n var J = u[3];\n out[0] = H * F - I * E + J * D;\n out[1] = -(G * F) + I * C - J * B;\n out[2] = G * E - H * C + J * A;\n out[3] = -(G * D) + H * B - I * A;\n return out;\n}\n/**\n * Performs a linear interpolation between two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec4} out\n */\n\nexport function lerp(out, a, b, t) {\n var ax = a[0];\n var ay = a[1];\n var az = a[2];\n var aw = a[3];\n out[0] = ax + t * (b[0] - ax);\n out[1] = ay + t * (b[1] - ay);\n out[2] = az + t * (b[2] - az);\n out[3] = aw + t * (b[3] - aw);\n return out;\n}\n/**\n * Generates a random vector with the given scale\n *\n * @param {vec4} out the receiving vector\n * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned\n * @returns {vec4} out\n */\n\nexport function random(out, scale) {\n scale = scale || 1.0; // Marsaglia, George. Choosing a Point from the Surface of a\n // Sphere. Ann. Math. Statist. 43 (1972), no. 2, 645--646.\n // http://projecteuclid.org/euclid.aoms/1177692644;\n\n var v1, v2, v3, v4;\n var s1, s2;\n\n do {\n v1 = glMatrix.RANDOM() * 2 - 1;\n v2 = glMatrix.RANDOM() * 2 - 1;\n s1 = v1 * v1 + v2 * v2;\n } while (s1 >= 1);\n\n do {\n v3 = glMatrix.RANDOM() * 2 - 1;\n v4 = glMatrix.RANDOM() * 2 - 1;\n s2 = v3 * v3 + v4 * v4;\n } while (s2 >= 1);\n\n var d = Math.sqrt((1 - s1) / s2);\n out[0] = scale * v1;\n out[1] = scale * v2;\n out[2] = scale * v3 * d;\n out[3] = scale * v4 * d;\n return out;\n}\n/**\n * Transforms the vec4 with a mat4.\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the vector to transform\n * @param {ReadonlyMat4} m matrix to transform with\n * @returns {vec4} out\n */\n\nexport function transformMat4(out, a, m) {\n var x = a[0],\n y = a[1],\n z = a[2],\n w = a[3];\n out[0] = m[0] * x + m[4] * y + m[8] * z + m[12] * w;\n out[1] = m[1] * x + m[5] * y + m[9] * z + m[13] * w;\n out[2] = m[2] * x + m[6] * y + m[10] * z + m[14] * w;\n out[3] = m[3] * x + m[7] * y + m[11] * z + m[15] * w;\n return out;\n}\n/**\n * Transforms the vec4 with a quat\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the vector to transform\n * @param {ReadonlyQuat} q quaternion to transform with\n * @returns {vec4} out\n */\n\nexport function transformQuat(out, a, q) {\n var x = a[0],\n y = a[1],\n z = a[2];\n var qx = q[0],\n qy = q[1],\n qz = q[2],\n qw = q[3]; // calculate quat * vec\n\n var ix = qw * x + qy * z - qz * y;\n var iy = qw * y + qz * x - qx * z;\n var iz = qw * z + qx * y - qy * x;\n var iw = -qx * x - qy * y - qz * z; // calculate result * inverse quat\n\n out[0] = ix * qw + iw * -qx + iy * -qz - iz * -qy;\n out[1] = iy * qw + iw * -qy + iz * -qx - ix * -qz;\n out[2] = iz * qw + iw * -qz + ix * -qy - iy * -qx;\n out[3] = a[3];\n return out;\n}\n/**\n * Set the components of a vec4 to zero\n *\n * @param {vec4} out the receiving vector\n * @returns {vec4} out\n */\n\nexport function zero(out) {\n out[0] = 0.0;\n out[1] = 0.0;\n out[2] = 0.0;\n out[3] = 0.0;\n return out;\n}\n/**\n * Returns a string representation of a vector\n *\n * @param {ReadonlyVec4} a vector to represent as a string\n * @returns {String} string representation of the vector\n */\n\nexport function str(a) {\n return \"vec4(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \")\";\n}\n/**\n * Returns whether or not the vectors have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyVec4} a The first vector.\n * @param {ReadonlyVec4} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3];\n}\n/**\n * Returns whether or not the vectors have approximately the same elements in the same position.\n *\n * @param {ReadonlyVec4} a The first vector.\n * @param {ReadonlyVec4} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3));\n}\n/**\n * Alias for {@link vec4.subtract}\n * @function\n */\n\nexport var sub = subtract;\n/**\n * Alias for {@link vec4.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link vec4.divide}\n * @function\n */\n\nexport var div = divide;\n/**\n * Alias for {@link vec4.distance}\n * @function\n */\n\nexport var dist = distance;\n/**\n * Alias for {@link vec4.squaredDistance}\n * @function\n */\n\nexport var sqrDist = squaredDistance;\n/**\n * Alias for {@link vec4.length}\n * @function\n */\n\nexport var len = length;\n/**\n * Alias for {@link vec4.squaredLength}\n * @function\n */\n\nexport var sqrLen = squaredLength;\n/**\n * Perform some operation over an array of vec4s.\n *\n * @param {Array} a the array of vectors to iterate over\n * @param {Number} stride Number of elements between the start of each vec4. If 0 assumes tightly packed\n * @param {Number} offset Number of elements to skip at the beginning of the array\n * @param {Number} count Number of vec4s to iterate over. If 0 iterates over entire array\n * @param {Function} fn Function to call for each vector in the array\n * @param {Object} [arg] additional argument to pass to fn\n * @returns {Array} a\n * @function\n */\n\nexport var forEach = function () {\n var vec = create();\n return function (a, stride, offset, count, fn, arg) {\n var i, l;\n\n if (!stride) {\n stride = 4;\n }\n\n if (!offset) {\n offset = 0;\n }\n\n if (count) {\n l = Math.min(count * stride + offset, a.length);\n } else {\n l = a.length;\n }\n\n for (i = offset; i < l; i += stride) {\n vec[0] = a[i];\n vec[1] = a[i + 1];\n vec[2] = a[i + 2];\n vec[3] = a[i + 3];\n fn(vec, vec, arg);\n a[i] = vec[0];\n a[i + 1] = vec[1];\n a[i + 2] = vec[2];\n a[i + 3] = vec[3];\n }\n\n return a;\n };\n}();","import * as glMatrix from \"./common.js\";\nimport * as mat3 from \"./mat3.js\";\nimport * as vec3 from \"./vec3.js\";\nimport * as vec4 from \"./vec4.js\";\n/**\n * Quaternion\n * @module quat\n */\n\n/**\n * Creates a new identity quat\n *\n * @returns {quat} a new quaternion\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(4);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[0] = 0;\n out[1] = 0;\n out[2] = 0;\n }\n\n out[3] = 1;\n return out;\n}\n/**\n * Set a quat to the identity quaternion\n *\n * @param {quat} out the receiving quaternion\n * @returns {quat} out\n */\n\nexport function identity(out) {\n out[0] = 0;\n out[1] = 0;\n out[2] = 0;\n out[3] = 1;\n return out;\n}\n/**\n * Sets a quat from the given angle and rotation axis,\n * then returns it.\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyVec3} axis the axis around which to rotate\n * @param {Number} rad the angle in radians\n * @returns {quat} out\n **/\n\nexport function setAxisAngle(out, axis, rad) {\n rad = rad * 0.5;\n var s = Math.sin(rad);\n out[0] = s * axis[0];\n out[1] = s * axis[1];\n out[2] = s * axis[2];\n out[3] = Math.cos(rad);\n return out;\n}\n/**\n * Gets the rotation axis and angle for a given\n * quaternion. If a quaternion is created with\n * setAxisAngle, this method will return the same\n * values as providied in the original parameter list\n * OR functionally equivalent values.\n * Example: The quaternion formed by axis [0, 0, 1] and\n * angle -90 is the same as the quaternion formed by\n * [0, 0, 1] and 270. This method favors the latter.\n * @param {vec3} out_axis Vector receiving the axis of rotation\n * @param {ReadonlyQuat} q Quaternion to be decomposed\n * @return {Number} Angle, in radians, of the rotation\n */\n\nexport function getAxisAngle(out_axis, q) {\n var rad = Math.acos(q[3]) * 2.0;\n var s = Math.sin(rad / 2.0);\n\n if (s > glMatrix.EPSILON) {\n out_axis[0] = q[0] / s;\n out_axis[1] = q[1] / s;\n out_axis[2] = q[2] / s;\n } else {\n // If s is zero, return any axis (no rotation - axis does not matter)\n out_axis[0] = 1;\n out_axis[1] = 0;\n out_axis[2] = 0;\n }\n\n return rad;\n}\n/**\n * Gets the angular distance between two unit quaternions\n *\n * @param {ReadonlyQuat} a Origin unit quaternion\n * @param {ReadonlyQuat} b Destination unit quaternion\n * @return {Number} Angle, in radians, between the two quaternions\n */\n\nexport function getAngle(a, b) {\n var dotproduct = dot(a, b);\n return Math.acos(2 * dotproduct * dotproduct - 1);\n}\n/**\n * Multiplies two quat's\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a the first operand\n * @param {ReadonlyQuat} b the second operand\n * @returns {quat} out\n */\n\nexport function multiply(out, a, b) {\n var ax = a[0],\n ay = a[1],\n az = a[2],\n aw = a[3];\n var bx = b[0],\n by = b[1],\n bz = b[2],\n bw = b[3];\n out[0] = ax * bw + aw * bx + ay * bz - az * by;\n out[1] = ay * bw + aw * by + az * bx - ax * bz;\n out[2] = az * bw + aw * bz + ax * by - ay * bx;\n out[3] = aw * bw - ax * bx - ay * by - az * bz;\n return out;\n}\n/**\n * Rotates a quaternion by the given angle about the X axis\n *\n * @param {quat} out quat receiving operation result\n * @param {ReadonlyQuat} a quat to rotate\n * @param {number} rad angle (in radians) to rotate\n * @returns {quat} out\n */\n\nexport function rotateX(out, a, rad) {\n rad *= 0.5;\n var ax = a[0],\n ay = a[1],\n az = a[2],\n aw = a[3];\n var bx = Math.sin(rad),\n bw = Math.cos(rad);\n out[0] = ax * bw + aw * bx;\n out[1] = ay * bw + az * bx;\n out[2] = az * bw - ay * bx;\n out[3] = aw * bw - ax * bx;\n return out;\n}\n/**\n * Rotates a quaternion by the given angle about the Y axis\n *\n * @param {quat} out quat receiving operation result\n * @param {ReadonlyQuat} a quat to rotate\n * @param {number} rad angle (in radians) to rotate\n * @returns {quat} out\n */\n\nexport function rotateY(out, a, rad) {\n rad *= 0.5;\n var ax = a[0],\n ay = a[1],\n az = a[2],\n aw = a[3];\n var by = Math.sin(rad),\n bw = Math.cos(rad);\n out[0] = ax * bw - az * by;\n out[1] = ay * bw + aw * by;\n out[2] = az * bw + ax * by;\n out[3] = aw * bw - ay * by;\n return out;\n}\n/**\n * Rotates a quaternion by the given angle about the Z axis\n *\n * @param {quat} out quat receiving operation result\n * @param {ReadonlyQuat} a quat to rotate\n * @param {number} rad angle (in radians) to rotate\n * @returns {quat} out\n */\n\nexport function rotateZ(out, a, rad) {\n rad *= 0.5;\n var ax = a[0],\n ay = a[1],\n az = a[2],\n aw = a[3];\n var bz = Math.sin(rad),\n bw = Math.cos(rad);\n out[0] = ax * bw + ay * bz;\n out[1] = ay * bw - ax * bz;\n out[2] = az * bw + aw * bz;\n out[3] = aw * bw - az * bz;\n return out;\n}\n/**\n * Calculates the W component of a quat from the X, Y, and Z components.\n * Assumes that quaternion is 1 unit in length.\n * Any existing W component will be ignored.\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a quat to calculate W component of\n * @returns {quat} out\n */\n\nexport function calculateW(out, a) {\n var x = a[0],\n y = a[1],\n z = a[2];\n out[0] = x;\n out[1] = y;\n out[2] = z;\n out[3] = Math.sqrt(Math.abs(1.0 - x * x - y * y - z * z));\n return out;\n}\n/**\n * Calculate the exponential of a unit quaternion.\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a quat to calculate the exponential of\n * @returns {quat} out\n */\n\nexport function exp(out, a) {\n var x = a[0],\n y = a[1],\n z = a[2],\n w = a[3];\n var r = Math.sqrt(x * x + y * y + z * z);\n var et = Math.exp(w);\n var s = r > 0 ? et * Math.sin(r) / r : 0;\n out[0] = x * s;\n out[1] = y * s;\n out[2] = z * s;\n out[3] = et * Math.cos(r);\n return out;\n}\n/**\n * Calculate the natural logarithm of a unit quaternion.\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a quat to calculate the exponential of\n * @returns {quat} out\n */\n\nexport function ln(out, a) {\n var x = a[0],\n y = a[1],\n z = a[2],\n w = a[3];\n var r = Math.sqrt(x * x + y * y + z * z);\n var t = r > 0 ? Math.atan2(r, w) / r : 0;\n out[0] = x * t;\n out[1] = y * t;\n out[2] = z * t;\n out[3] = 0.5 * Math.log(x * x + y * y + z * z + w * w);\n return out;\n}\n/**\n * Calculate the scalar power of a unit quaternion.\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a quat to calculate the exponential of\n * @param {Number} b amount to scale the quaternion by\n * @returns {quat} out\n */\n\nexport function pow(out, a, b) {\n ln(out, a);\n scale(out, out, b);\n exp(out, out);\n return out;\n}\n/**\n * Performs a spherical linear interpolation between two quat\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a the first operand\n * @param {ReadonlyQuat} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {quat} out\n */\n\nexport function slerp(out, a, b, t) {\n // benchmarks:\n // http://jsperf.com/quaternion-slerp-implementations\n var ax = a[0],\n ay = a[1],\n az = a[2],\n aw = a[3];\n var bx = b[0],\n by = b[1],\n bz = b[2],\n bw = b[3];\n var omega, cosom, sinom, scale0, scale1; // calc cosine\n\n cosom = ax * bx + ay * by + az * bz + aw * bw; // adjust signs (if necessary)\n\n if (cosom < 0.0) {\n cosom = -cosom;\n bx = -bx;\n by = -by;\n bz = -bz;\n bw = -bw;\n } // calculate coefficients\n\n\n if (1.0 - cosom > glMatrix.EPSILON) {\n // standard case (slerp)\n omega = Math.acos(cosom);\n sinom = Math.sin(omega);\n scale0 = Math.sin((1.0 - t) * omega) / sinom;\n scale1 = Math.sin(t * omega) / sinom;\n } else {\n // \"from\" and \"to\" quaternions are very close\n // ... so we can do a linear interpolation\n scale0 = 1.0 - t;\n scale1 = t;\n } // calculate final values\n\n\n out[0] = scale0 * ax + scale1 * bx;\n out[1] = scale0 * ay + scale1 * by;\n out[2] = scale0 * az + scale1 * bz;\n out[3] = scale0 * aw + scale1 * bw;\n return out;\n}\n/**\n * Generates a random unit quaternion\n *\n * @param {quat} out the receiving quaternion\n * @returns {quat} out\n */\n\nexport function random(out) {\n // Implementation of http://planning.cs.uiuc.edu/node198.html\n // TODO: Calling random 3 times is probably not the fastest solution\n var u1 = glMatrix.RANDOM();\n var u2 = glMatrix.RANDOM();\n var u3 = glMatrix.RANDOM();\n var sqrt1MinusU1 = Math.sqrt(1 - u1);\n var sqrtU1 = Math.sqrt(u1);\n out[0] = sqrt1MinusU1 * Math.sin(2.0 * Math.PI * u2);\n out[1] = sqrt1MinusU1 * Math.cos(2.0 * Math.PI * u2);\n out[2] = sqrtU1 * Math.sin(2.0 * Math.PI * u3);\n out[3] = sqrtU1 * Math.cos(2.0 * Math.PI * u3);\n return out;\n}\n/**\n * Calculates the inverse of a quat\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a quat to calculate inverse of\n * @returns {quat} out\n */\n\nexport function invert(out, a) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var dot = a0 * a0 + a1 * a1 + a2 * a2 + a3 * a3;\n var invDot = dot ? 1.0 / dot : 0; // TODO: Would be faster to return [0,0,0,0] immediately if dot == 0\n\n out[0] = -a0 * invDot;\n out[1] = -a1 * invDot;\n out[2] = -a2 * invDot;\n out[3] = a3 * invDot;\n return out;\n}\n/**\n * Calculates the conjugate of a quat\n * If the quaternion is normalized, this function is faster than quat.inverse and produces the same result.\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a quat to calculate conjugate of\n * @returns {quat} out\n */\n\nexport function conjugate(out, a) {\n out[0] = -a[0];\n out[1] = -a[1];\n out[2] = -a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Creates a quaternion from the given 3x3 rotation matrix.\n *\n * NOTE: The resultant quaternion is not normalized, so you should be sure\n * to renormalize the quaternion yourself where necessary.\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyMat3} m rotation matrix\n * @returns {quat} out\n * @function\n */\n\nexport function fromMat3(out, m) {\n // Algorithm in Ken Shoemake's article in 1987 SIGGRAPH course notes\n // article \"Quaternion Calculus and Fast Animation\".\n var fTrace = m[0] + m[4] + m[8];\n var fRoot;\n\n if (fTrace > 0.0) {\n // |w| > 1/2, may as well choose w > 1/2\n fRoot = Math.sqrt(fTrace + 1.0); // 2w\n\n out[3] = 0.5 * fRoot;\n fRoot = 0.5 / fRoot; // 1/(4w)\n\n out[0] = (m[5] - m[7]) * fRoot;\n out[1] = (m[6] - m[2]) * fRoot;\n out[2] = (m[1] - m[3]) * fRoot;\n } else {\n // |w| <= 1/2\n var i = 0;\n if (m[4] > m[0]) i = 1;\n if (m[8] > m[i * 3 + i]) i = 2;\n var j = (i + 1) % 3;\n var k = (i + 2) % 3;\n fRoot = Math.sqrt(m[i * 3 + i] - m[j * 3 + j] - m[k * 3 + k] + 1.0);\n out[i] = 0.5 * fRoot;\n fRoot = 0.5 / fRoot;\n out[3] = (m[j * 3 + k] - m[k * 3 + j]) * fRoot;\n out[j] = (m[j * 3 + i] + m[i * 3 + j]) * fRoot;\n out[k] = (m[k * 3 + i] + m[i * 3 + k]) * fRoot;\n }\n\n return out;\n}\n/**\n * Creates a quaternion from the given euler angle x, y, z.\n *\n * @param {quat} out the receiving quaternion\n * @param {x} Angle to rotate around X axis in degrees.\n * @param {y} Angle to rotate around Y axis in degrees.\n * @param {z} Angle to rotate around Z axis in degrees.\n * @returns {quat} out\n * @function\n */\n\nexport function fromEuler(out, x, y, z) {\n var halfToRad = 0.5 * Math.PI / 180.0;\n x *= halfToRad;\n y *= halfToRad;\n z *= halfToRad;\n var sx = Math.sin(x);\n var cx = Math.cos(x);\n var sy = Math.sin(y);\n var cy = Math.cos(y);\n var sz = Math.sin(z);\n var cz = Math.cos(z);\n out[0] = sx * cy * cz - cx * sy * sz;\n out[1] = cx * sy * cz + sx * cy * sz;\n out[2] = cx * cy * sz - sx * sy * cz;\n out[3] = cx * cy * cz + sx * sy * sz;\n return out;\n}\n/**\n * Returns a string representation of a quatenion\n *\n * @param {ReadonlyQuat} a vector to represent as a string\n * @returns {String} string representation of the vector\n */\n\nexport function str(a) {\n return \"quat(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \")\";\n}\n/**\n * Creates a new quat initialized with values from an existing quaternion\n *\n * @param {ReadonlyQuat} a quaternion to clone\n * @returns {quat} a new quaternion\n * @function\n */\n\nexport var clone = vec4.clone;\n/**\n * Creates a new quat initialized with the given values\n *\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @param {Number} w W component\n * @returns {quat} a new quaternion\n * @function\n */\n\nexport var fromValues = vec4.fromValues;\n/**\n * Copy the values from one quat to another\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a the source quaternion\n * @returns {quat} out\n * @function\n */\n\nexport var copy = vec4.copy;\n/**\n * Set the components of a quat to the given values\n *\n * @param {quat} out the receiving quaternion\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @param {Number} w W component\n * @returns {quat} out\n * @function\n */\n\nexport var set = vec4.set;\n/**\n * Adds two quat's\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a the first operand\n * @param {ReadonlyQuat} b the second operand\n * @returns {quat} out\n * @function\n */\n\nexport var add = vec4.add;\n/**\n * Alias for {@link quat.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Scales a quat by a scalar number\n *\n * @param {quat} out the receiving vector\n * @param {ReadonlyQuat} a the vector to scale\n * @param {Number} b amount to scale the vector by\n * @returns {quat} out\n * @function\n */\n\nexport var scale = vec4.scale;\n/**\n * Calculates the dot product of two quat's\n *\n * @param {ReadonlyQuat} a the first operand\n * @param {ReadonlyQuat} b the second operand\n * @returns {Number} dot product of a and b\n * @function\n */\n\nexport var dot = vec4.dot;\n/**\n * Performs a linear interpolation between two quat's\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a the first operand\n * @param {ReadonlyQuat} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {quat} out\n * @function\n */\n\nexport var lerp = vec4.lerp;\n/**\n * Calculates the length of a quat\n *\n * @param {ReadonlyQuat} a vector to calculate length of\n * @returns {Number} length of a\n */\n\nexport var length = vec4.length;\n/**\n * Alias for {@link quat.length}\n * @function\n */\n\nexport var len = length;\n/**\n * Calculates the squared length of a quat\n *\n * @param {ReadonlyQuat} a vector to calculate squared length of\n * @returns {Number} squared length of a\n * @function\n */\n\nexport var squaredLength = vec4.squaredLength;\n/**\n * Alias for {@link quat.squaredLength}\n * @function\n */\n\nexport var sqrLen = squaredLength;\n/**\n * Normalize a quat\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a quaternion to normalize\n * @returns {quat} out\n * @function\n */\n\nexport var normalize = vec4.normalize;\n/**\n * Returns whether or not the quaternions have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyQuat} a The first quaternion.\n * @param {ReadonlyQuat} b The second quaternion.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport var exactEquals = vec4.exactEquals;\n/**\n * Returns whether or not the quaternions have approximately the same elements in the same position.\n *\n * @param {ReadonlyQuat} a The first vector.\n * @param {ReadonlyQuat} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport var equals = vec4.equals;\n/**\n * Sets a quaternion to represent the shortest rotation from one\n * vector to another.\n *\n * Both vectors are assumed to be unit length.\n *\n * @param {quat} out the receiving quaternion.\n * @param {ReadonlyVec3} a the initial vector\n * @param {ReadonlyVec3} b the destination vector\n * @returns {quat} out\n */\n\nexport var rotationTo = function () {\n var tmpvec3 = vec3.create();\n var xUnitVec3 = vec3.fromValues(1, 0, 0);\n var yUnitVec3 = vec3.fromValues(0, 1, 0);\n return function (out, a, b) {\n var dot = vec3.dot(a, b);\n\n if (dot < -0.999999) {\n vec3.cross(tmpvec3, xUnitVec3, a);\n if (vec3.len(tmpvec3) < 0.000001) vec3.cross(tmpvec3, yUnitVec3, a);\n vec3.normalize(tmpvec3, tmpvec3);\n setAxisAngle(out, tmpvec3, Math.PI);\n return out;\n } else if (dot > 0.999999) {\n out[0] = 0;\n out[1] = 0;\n out[2] = 0;\n out[3] = 1;\n return out;\n } else {\n vec3.cross(tmpvec3, a, b);\n out[0] = tmpvec3[0];\n out[1] = tmpvec3[1];\n out[2] = tmpvec3[2];\n out[3] = 1 + dot;\n return normalize(out, out);\n }\n };\n}();\n/**\n * Performs a spherical linear interpolation with two control points\n *\n * @param {quat} out the receiving quaternion\n * @param {ReadonlyQuat} a the first operand\n * @param {ReadonlyQuat} b the second operand\n * @param {ReadonlyQuat} c the third operand\n * @param {ReadonlyQuat} d the fourth operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {quat} out\n */\n\nexport var sqlerp = function () {\n var temp1 = create();\n var temp2 = create();\n return function (out, a, b, c, d, t) {\n slerp(temp1, a, d, t);\n slerp(temp2, b, c, t);\n slerp(out, temp1, temp2, 2 * t * (1 - t));\n return out;\n };\n}();\n/**\n * Sets the specified quaternion with values corresponding to the given\n * axes. Each axis is a vec3 and is expected to be unit length and\n * perpendicular to all other specified axes.\n *\n * @param {ReadonlyVec3} view the vector representing the viewing direction\n * @param {ReadonlyVec3} right the vector representing the local \"right\" direction\n * @param {ReadonlyVec3} up the vector representing the local \"up\" direction\n * @returns {quat} out\n */\n\nexport var setAxes = function () {\n var matr = mat3.create();\n return function (out, view, right, up) {\n matr[0] = right[0];\n matr[3] = right[1];\n matr[6] = right[2];\n matr[1] = up[0];\n matr[4] = up[1];\n matr[7] = up[2];\n matr[2] = -view[0];\n matr[5] = -view[1];\n matr[8] = -view[2];\n return normalize(out, fromMat3(out, matr));\n };\n}();","import * as glMatrix from \"./common.js\";\nimport * as quat from \"./quat.js\";\nimport * as mat4 from \"./mat4.js\";\n/**\n * Dual Quaternion
\n * Format: [real, dual]
\n * Quaternion format: XYZW
\n * Make sure to have normalized dual quaternions, otherwise the functions may not work as intended.
\n * @module quat2\n */\n\n/**\n * Creates a new identity dual quat\n *\n * @returns {quat2} a new dual quaternion [real -> rotation, dual -> translation]\n */\n\nexport function create() {\n var dq = new glMatrix.ARRAY_TYPE(8);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n dq[0] = 0;\n dq[1] = 0;\n dq[2] = 0;\n dq[4] = 0;\n dq[5] = 0;\n dq[6] = 0;\n dq[7] = 0;\n }\n\n dq[3] = 1;\n return dq;\n}\n/**\n * Creates a new quat initialized with values from an existing quaternion\n *\n * @param {ReadonlyQuat2} a dual quaternion to clone\n * @returns {quat2} new dual quaternion\n * @function\n */\n\nexport function clone(a) {\n var dq = new glMatrix.ARRAY_TYPE(8);\n dq[0] = a[0];\n dq[1] = a[1];\n dq[2] = a[2];\n dq[3] = a[3];\n dq[4] = a[4];\n dq[5] = a[5];\n dq[6] = a[6];\n dq[7] = a[7];\n return dq;\n}\n/**\n * Creates a new dual quat initialized with the given values\n *\n * @param {Number} x1 X component\n * @param {Number} y1 Y component\n * @param {Number} z1 Z component\n * @param {Number} w1 W component\n * @param {Number} x2 X component\n * @param {Number} y2 Y component\n * @param {Number} z2 Z component\n * @param {Number} w2 W component\n * @returns {quat2} new dual quaternion\n * @function\n */\n\nexport function fromValues(x1, y1, z1, w1, x2, y2, z2, w2) {\n var dq = new glMatrix.ARRAY_TYPE(8);\n dq[0] = x1;\n dq[1] = y1;\n dq[2] = z1;\n dq[3] = w1;\n dq[4] = x2;\n dq[5] = y2;\n dq[6] = z2;\n dq[7] = w2;\n return dq;\n}\n/**\n * Creates a new dual quat from the given values (quat and translation)\n *\n * @param {Number} x1 X component\n * @param {Number} y1 Y component\n * @param {Number} z1 Z component\n * @param {Number} w1 W component\n * @param {Number} x2 X component (translation)\n * @param {Number} y2 Y component (translation)\n * @param {Number} z2 Z component (translation)\n * @returns {quat2} new dual quaternion\n * @function\n */\n\nexport function fromRotationTranslationValues(x1, y1, z1, w1, x2, y2, z2) {\n var dq = new glMatrix.ARRAY_TYPE(8);\n dq[0] = x1;\n dq[1] = y1;\n dq[2] = z1;\n dq[3] = w1;\n var ax = x2 * 0.5,\n ay = y2 * 0.5,\n az = z2 * 0.5;\n dq[4] = ax * w1 + ay * z1 - az * y1;\n dq[5] = ay * w1 + az * x1 - ax * z1;\n dq[6] = az * w1 + ax * y1 - ay * x1;\n dq[7] = -ax * x1 - ay * y1 - az * z1;\n return dq;\n}\n/**\n * Creates a dual quat from a quaternion and a translation\n *\n * @param {ReadonlyQuat2} dual quaternion receiving operation result\n * @param {ReadonlyQuat} q a normalized quaternion\n * @param {ReadonlyVec3} t tranlation vector\n * @returns {quat2} dual quaternion receiving operation result\n * @function\n */\n\nexport function fromRotationTranslation(out, q, t) {\n var ax = t[0] * 0.5,\n ay = t[1] * 0.5,\n az = t[2] * 0.5,\n bx = q[0],\n by = q[1],\n bz = q[2],\n bw = q[3];\n out[0] = bx;\n out[1] = by;\n out[2] = bz;\n out[3] = bw;\n out[4] = ax * bw + ay * bz - az * by;\n out[5] = ay * bw + az * bx - ax * bz;\n out[6] = az * bw + ax * by - ay * bx;\n out[7] = -ax * bx - ay * by - az * bz;\n return out;\n}\n/**\n * Creates a dual quat from a translation\n *\n * @param {ReadonlyQuat2} dual quaternion receiving operation result\n * @param {ReadonlyVec3} t translation vector\n * @returns {quat2} dual quaternion receiving operation result\n * @function\n */\n\nexport function fromTranslation(out, t) {\n out[0] = 0;\n out[1] = 0;\n out[2] = 0;\n out[3] = 1;\n out[4] = t[0] * 0.5;\n out[5] = t[1] * 0.5;\n out[6] = t[2] * 0.5;\n out[7] = 0;\n return out;\n}\n/**\n * Creates a dual quat from a quaternion\n *\n * @param {ReadonlyQuat2} dual quaternion receiving operation result\n * @param {ReadonlyQuat} q the quaternion\n * @returns {quat2} dual quaternion receiving operation result\n * @function\n */\n\nexport function fromRotation(out, q) {\n out[0] = q[0];\n out[1] = q[1];\n out[2] = q[2];\n out[3] = q[3];\n out[4] = 0;\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n return out;\n}\n/**\n * Creates a new dual quat from a matrix (4x4)\n *\n * @param {quat2} out the dual quaternion\n * @param {ReadonlyMat4} a the matrix\n * @returns {quat2} dual quat receiving operation result\n * @function\n */\n\nexport function fromMat4(out, a) {\n //TODO Optimize this\n var outer = quat.create();\n mat4.getRotation(outer, a);\n var t = new glMatrix.ARRAY_TYPE(3);\n mat4.getTranslation(t, a);\n fromRotationTranslation(out, outer, t);\n return out;\n}\n/**\n * Copy the values from one dual quat to another\n *\n * @param {quat2} out the receiving dual quaternion\n * @param {ReadonlyQuat2} a the source dual quaternion\n * @returns {quat2} out\n * @function\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n return out;\n}\n/**\n * Set a dual quat to the identity dual quaternion\n *\n * @param {quat2} out the receiving quaternion\n * @returns {quat2} out\n */\n\nexport function identity(out) {\n out[0] = 0;\n out[1] = 0;\n out[2] = 0;\n out[3] = 1;\n out[4] = 0;\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n return out;\n}\n/**\n * Set the components of a dual quat to the given values\n *\n * @param {quat2} out the receiving quaternion\n * @param {Number} x1 X component\n * @param {Number} y1 Y component\n * @param {Number} z1 Z component\n * @param {Number} w1 W component\n * @param {Number} x2 X component\n * @param {Number} y2 Y component\n * @param {Number} z2 Z component\n * @param {Number} w2 W component\n * @returns {quat2} out\n * @function\n */\n\nexport function set(out, x1, y1, z1, w1, x2, y2, z2, w2) {\n out[0] = x1;\n out[1] = y1;\n out[2] = z1;\n out[3] = w1;\n out[4] = x2;\n out[5] = y2;\n out[6] = z2;\n out[7] = w2;\n return out;\n}\n/**\n * Gets the real part of a dual quat\n * @param {quat} out real part\n * @param {ReadonlyQuat2} a Dual Quaternion\n * @return {quat} real part\n */\n\nexport var getReal = quat.copy;\n/**\n * Gets the dual part of a dual quat\n * @param {quat} out dual part\n * @param {ReadonlyQuat2} a Dual Quaternion\n * @return {quat} dual part\n */\n\nexport function getDual(out, a) {\n out[0] = a[4];\n out[1] = a[5];\n out[2] = a[6];\n out[3] = a[7];\n return out;\n}\n/**\n * Set the real component of a dual quat to the given quaternion\n *\n * @param {quat2} out the receiving quaternion\n * @param {ReadonlyQuat} q a quaternion representing the real part\n * @returns {quat2} out\n * @function\n */\n\nexport var setReal = quat.copy;\n/**\n * Set the dual component of a dual quat to the given quaternion\n *\n * @param {quat2} out the receiving quaternion\n * @param {ReadonlyQuat} q a quaternion representing the dual part\n * @returns {quat2} out\n * @function\n */\n\nexport function setDual(out, q) {\n out[4] = q[0];\n out[5] = q[1];\n out[6] = q[2];\n out[7] = q[3];\n return out;\n}\n/**\n * Gets the translation of a normalized dual quat\n * @param {vec3} out translation\n * @param {ReadonlyQuat2} a Dual Quaternion to be decomposed\n * @return {vec3} translation\n */\n\nexport function getTranslation(out, a) {\n var ax = a[4],\n ay = a[5],\n az = a[6],\n aw = a[7],\n bx = -a[0],\n by = -a[1],\n bz = -a[2],\n bw = a[3];\n out[0] = (ax * bw + aw * bx + ay * bz - az * by) * 2;\n out[1] = (ay * bw + aw * by + az * bx - ax * bz) * 2;\n out[2] = (az * bw + aw * bz + ax * by - ay * bx) * 2;\n return out;\n}\n/**\n * Translates a dual quat by the given vector\n *\n * @param {quat2} out the receiving dual quaternion\n * @param {ReadonlyQuat2} a the dual quaternion to translate\n * @param {ReadonlyVec3} v vector to translate by\n * @returns {quat2} out\n */\n\nexport function translate(out, a, v) {\n var ax1 = a[0],\n ay1 = a[1],\n az1 = a[2],\n aw1 = a[3],\n bx1 = v[0] * 0.5,\n by1 = v[1] * 0.5,\n bz1 = v[2] * 0.5,\n ax2 = a[4],\n ay2 = a[5],\n az2 = a[6],\n aw2 = a[7];\n out[0] = ax1;\n out[1] = ay1;\n out[2] = az1;\n out[3] = aw1;\n out[4] = aw1 * bx1 + ay1 * bz1 - az1 * by1 + ax2;\n out[5] = aw1 * by1 + az1 * bx1 - ax1 * bz1 + ay2;\n out[6] = aw1 * bz1 + ax1 * by1 - ay1 * bx1 + az2;\n out[7] = -ax1 * bx1 - ay1 * by1 - az1 * bz1 + aw2;\n return out;\n}\n/**\n * Rotates a dual quat around the X axis\n *\n * @param {quat2} out the receiving dual quaternion\n * @param {ReadonlyQuat2} a the dual quaternion to rotate\n * @param {number} rad how far should the rotation be\n * @returns {quat2} out\n */\n\nexport function rotateX(out, a, rad) {\n var bx = -a[0],\n by = -a[1],\n bz = -a[2],\n bw = a[3],\n ax = a[4],\n ay = a[5],\n az = a[6],\n aw = a[7],\n ax1 = ax * bw + aw * bx + ay * bz - az * by,\n ay1 = ay * bw + aw * by + az * bx - ax * bz,\n az1 = az * bw + aw * bz + ax * by - ay * bx,\n aw1 = aw * bw - ax * bx - ay * by - az * bz;\n quat.rotateX(out, a, rad);\n bx = out[0];\n by = out[1];\n bz = out[2];\n bw = out[3];\n out[4] = ax1 * bw + aw1 * bx + ay1 * bz - az1 * by;\n out[5] = ay1 * bw + aw1 * by + az1 * bx - ax1 * bz;\n out[6] = az1 * bw + aw1 * bz + ax1 * by - ay1 * bx;\n out[7] = aw1 * bw - ax1 * bx - ay1 * by - az1 * bz;\n return out;\n}\n/**\n * Rotates a dual quat around the Y axis\n *\n * @param {quat2} out the receiving dual quaternion\n * @param {ReadonlyQuat2} a the dual quaternion to rotate\n * @param {number} rad how far should the rotation be\n * @returns {quat2} out\n */\n\nexport function rotateY(out, a, rad) {\n var bx = -a[0],\n by = -a[1],\n bz = -a[2],\n bw = a[3],\n ax = a[4],\n ay = a[5],\n az = a[6],\n aw = a[7],\n ax1 = ax * bw + aw * bx + ay * bz - az * by,\n ay1 = ay * bw + aw * by + az * bx - ax * bz,\n az1 = az * bw + aw * bz + ax * by - ay * bx,\n aw1 = aw * bw - ax * bx - ay * by - az * bz;\n quat.rotateY(out, a, rad);\n bx = out[0];\n by = out[1];\n bz = out[2];\n bw = out[3];\n out[4] = ax1 * bw + aw1 * bx + ay1 * bz - az1 * by;\n out[5] = ay1 * bw + aw1 * by + az1 * bx - ax1 * bz;\n out[6] = az1 * bw + aw1 * bz + ax1 * by - ay1 * bx;\n out[7] = aw1 * bw - ax1 * bx - ay1 * by - az1 * bz;\n return out;\n}\n/**\n * Rotates a dual quat around the Z axis\n *\n * @param {quat2} out the receiving dual quaternion\n * @param {ReadonlyQuat2} a the dual quaternion to rotate\n * @param {number} rad how far should the rotation be\n * @returns {quat2} out\n */\n\nexport function rotateZ(out, a, rad) {\n var bx = -a[0],\n by = -a[1],\n bz = -a[2],\n bw = a[3],\n ax = a[4],\n ay = a[5],\n az = a[6],\n aw = a[7],\n ax1 = ax * bw + aw * bx + ay * bz - az * by,\n ay1 = ay * bw + aw * by + az * bx - ax * bz,\n az1 = az * bw + aw * bz + ax * by - ay * bx,\n aw1 = aw * bw - ax * bx - ay * by - az * bz;\n quat.rotateZ(out, a, rad);\n bx = out[0];\n by = out[1];\n bz = out[2];\n bw = out[3];\n out[4] = ax1 * bw + aw1 * bx + ay1 * bz - az1 * by;\n out[5] = ay1 * bw + aw1 * by + az1 * bx - ax1 * bz;\n out[6] = az1 * bw + aw1 * bz + ax1 * by - ay1 * bx;\n out[7] = aw1 * bw - ax1 * bx - ay1 * by - az1 * bz;\n return out;\n}\n/**\n * Rotates a dual quat by a given quaternion (a * q)\n *\n * @param {quat2} out the receiving dual quaternion\n * @param {ReadonlyQuat2} a the dual quaternion to rotate\n * @param {ReadonlyQuat} q quaternion to rotate by\n * @returns {quat2} out\n */\n\nexport function rotateByQuatAppend(out, a, q) {\n var qx = q[0],\n qy = q[1],\n qz = q[2],\n qw = q[3],\n ax = a[0],\n ay = a[1],\n az = a[2],\n aw = a[3];\n out[0] = ax * qw + aw * qx + ay * qz - az * qy;\n out[1] = ay * qw + aw * qy + az * qx - ax * qz;\n out[2] = az * qw + aw * qz + ax * qy - ay * qx;\n out[3] = aw * qw - ax * qx - ay * qy - az * qz;\n ax = a[4];\n ay = a[5];\n az = a[6];\n aw = a[7];\n out[4] = ax * qw + aw * qx + ay * qz - az * qy;\n out[5] = ay * qw + aw * qy + az * qx - ax * qz;\n out[6] = az * qw + aw * qz + ax * qy - ay * qx;\n out[7] = aw * qw - ax * qx - ay * qy - az * qz;\n return out;\n}\n/**\n * Rotates a dual quat by a given quaternion (q * a)\n *\n * @param {quat2} out the receiving dual quaternion\n * @param {ReadonlyQuat} q quaternion to rotate by\n * @param {ReadonlyQuat2} a the dual quaternion to rotate\n * @returns {quat2} out\n */\n\nexport function rotateByQuatPrepend(out, q, a) {\n var qx = q[0],\n qy = q[1],\n qz = q[2],\n qw = q[3],\n bx = a[0],\n by = a[1],\n bz = a[2],\n bw = a[3];\n out[0] = qx * bw + qw * bx + qy * bz - qz * by;\n out[1] = qy * bw + qw * by + qz * bx - qx * bz;\n out[2] = qz * bw + qw * bz + qx * by - qy * bx;\n out[3] = qw * bw - qx * bx - qy * by - qz * bz;\n bx = a[4];\n by = a[5];\n bz = a[6];\n bw = a[7];\n out[4] = qx * bw + qw * bx + qy * bz - qz * by;\n out[5] = qy * bw + qw * by + qz * bx - qx * bz;\n out[6] = qz * bw + qw * bz + qx * by - qy * bx;\n out[7] = qw * bw - qx * bx - qy * by - qz * bz;\n return out;\n}\n/**\n * Rotates a dual quat around a given axis. Does the normalisation automatically\n *\n * @param {quat2} out the receiving dual quaternion\n * @param {ReadonlyQuat2} a the dual quaternion to rotate\n * @param {ReadonlyVec3} axis the axis to rotate around\n * @param {Number} rad how far the rotation should be\n * @returns {quat2} out\n */\n\nexport function rotateAroundAxis(out, a, axis, rad) {\n //Special case for rad = 0\n if (Math.abs(rad) < glMatrix.EPSILON) {\n return copy(out, a);\n }\n\n var axisLength = Math.hypot(axis[0], axis[1], axis[2]);\n rad = rad * 0.5;\n var s = Math.sin(rad);\n var bx = s * axis[0] / axisLength;\n var by = s * axis[1] / axisLength;\n var bz = s * axis[2] / axisLength;\n var bw = Math.cos(rad);\n var ax1 = a[0],\n ay1 = a[1],\n az1 = a[2],\n aw1 = a[3];\n out[0] = ax1 * bw + aw1 * bx + ay1 * bz - az1 * by;\n out[1] = ay1 * bw + aw1 * by + az1 * bx - ax1 * bz;\n out[2] = az1 * bw + aw1 * bz + ax1 * by - ay1 * bx;\n out[3] = aw1 * bw - ax1 * bx - ay1 * by - az1 * bz;\n var ax = a[4],\n ay = a[5],\n az = a[6],\n aw = a[7];\n out[4] = ax * bw + aw * bx + ay * bz - az * by;\n out[5] = ay * bw + aw * by + az * bx - ax * bz;\n out[6] = az * bw + aw * bz + ax * by - ay * bx;\n out[7] = aw * bw - ax * bx - ay * by - az * bz;\n return out;\n}\n/**\n * Adds two dual quat's\n *\n * @param {quat2} out the receiving dual quaternion\n * @param {ReadonlyQuat2} a the first operand\n * @param {ReadonlyQuat2} b the second operand\n * @returns {quat2} out\n * @function\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n out[4] = a[4] + b[4];\n out[5] = a[5] + b[5];\n out[6] = a[6] + b[6];\n out[7] = a[7] + b[7];\n return out;\n}\n/**\n * Multiplies two dual quat's\n *\n * @param {quat2} out the receiving dual quaternion\n * @param {ReadonlyQuat2} a the first operand\n * @param {ReadonlyQuat2} b the second operand\n * @returns {quat2} out\n */\n\nexport function multiply(out, a, b) {\n var ax0 = a[0],\n ay0 = a[1],\n az0 = a[2],\n aw0 = a[3],\n bx1 = b[4],\n by1 = b[5],\n bz1 = b[6],\n bw1 = b[7],\n ax1 = a[4],\n ay1 = a[5],\n az1 = a[6],\n aw1 = a[7],\n bx0 = b[0],\n by0 = b[1],\n bz0 = b[2],\n bw0 = b[3];\n out[0] = ax0 * bw0 + aw0 * bx0 + ay0 * bz0 - az0 * by0;\n out[1] = ay0 * bw0 + aw0 * by0 + az0 * bx0 - ax0 * bz0;\n out[2] = az0 * bw0 + aw0 * bz0 + ax0 * by0 - ay0 * bx0;\n out[3] = aw0 * bw0 - ax0 * bx0 - ay0 * by0 - az0 * bz0;\n out[4] = ax0 * bw1 + aw0 * bx1 + ay0 * bz1 - az0 * by1 + ax1 * bw0 + aw1 * bx0 + ay1 * bz0 - az1 * by0;\n out[5] = ay0 * bw1 + aw0 * by1 + az0 * bx1 - ax0 * bz1 + ay1 * bw0 + aw1 * by0 + az1 * bx0 - ax1 * bz0;\n out[6] = az0 * bw1 + aw0 * bz1 + ax0 * by1 - ay0 * bx1 + az1 * bw0 + aw1 * bz0 + ax1 * by0 - ay1 * bx0;\n out[7] = aw0 * bw1 - ax0 * bx1 - ay0 * by1 - az0 * bz1 + aw1 * bw0 - ax1 * bx0 - ay1 * by0 - az1 * bz0;\n return out;\n}\n/**\n * Alias for {@link quat2.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Scales a dual quat by a scalar number\n *\n * @param {quat2} out the receiving dual quat\n * @param {ReadonlyQuat2} a the dual quat to scale\n * @param {Number} b amount to scale the dual quat by\n * @returns {quat2} out\n * @function\n */\n\nexport function scale(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n out[4] = a[4] * b;\n out[5] = a[5] * b;\n out[6] = a[6] * b;\n out[7] = a[7] * b;\n return out;\n}\n/**\n * Calculates the dot product of two dual quat's (The dot product of the real parts)\n *\n * @param {ReadonlyQuat2} a the first operand\n * @param {ReadonlyQuat2} b the second operand\n * @returns {Number} dot product of a and b\n * @function\n */\n\nexport var dot = quat.dot;\n/**\n * Performs a linear interpolation between two dual quats's\n * NOTE: The resulting dual quaternions won't always be normalized (The error is most noticeable when t = 0.5)\n *\n * @param {quat2} out the receiving dual quat\n * @param {ReadonlyQuat2} a the first operand\n * @param {ReadonlyQuat2} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {quat2} out\n */\n\nexport function lerp(out, a, b, t) {\n var mt = 1 - t;\n if (dot(a, b) < 0) t = -t;\n out[0] = a[0] * mt + b[0] * t;\n out[1] = a[1] * mt + b[1] * t;\n out[2] = a[2] * mt + b[2] * t;\n out[3] = a[3] * mt + b[3] * t;\n out[4] = a[4] * mt + b[4] * t;\n out[5] = a[5] * mt + b[5] * t;\n out[6] = a[6] * mt + b[6] * t;\n out[7] = a[7] * mt + b[7] * t;\n return out;\n}\n/**\n * Calculates the inverse of a dual quat. If they are normalized, conjugate is cheaper\n *\n * @param {quat2} out the receiving dual quaternion\n * @param {ReadonlyQuat2} a dual quat to calculate inverse of\n * @returns {quat2} out\n */\n\nexport function invert(out, a) {\n var sqlen = squaredLength(a);\n out[0] = -a[0] / sqlen;\n out[1] = -a[1] / sqlen;\n out[2] = -a[2] / sqlen;\n out[3] = a[3] / sqlen;\n out[4] = -a[4] / sqlen;\n out[5] = -a[5] / sqlen;\n out[6] = -a[6] / sqlen;\n out[7] = a[7] / sqlen;\n return out;\n}\n/**\n * Calculates the conjugate of a dual quat\n * If the dual quaternion is normalized, this function is faster than quat2.inverse and produces the same result.\n *\n * @param {quat2} out the receiving quaternion\n * @param {ReadonlyQuat2} a quat to calculate conjugate of\n * @returns {quat2} out\n */\n\nexport function conjugate(out, a) {\n out[0] = -a[0];\n out[1] = -a[1];\n out[2] = -a[2];\n out[3] = a[3];\n out[4] = -a[4];\n out[5] = -a[5];\n out[6] = -a[6];\n out[7] = a[7];\n return out;\n}\n/**\n * Calculates the length of a dual quat\n *\n * @param {ReadonlyQuat2} a dual quat to calculate length of\n * @returns {Number} length of a\n * @function\n */\n\nexport var length = quat.length;\n/**\n * Alias for {@link quat2.length}\n * @function\n */\n\nexport var len = length;\n/**\n * Calculates the squared length of a dual quat\n *\n * @param {ReadonlyQuat2} a dual quat to calculate squared length of\n * @returns {Number} squared length of a\n * @function\n */\n\nexport var squaredLength = quat.squaredLength;\n/**\n * Alias for {@link quat2.squaredLength}\n * @function\n */\n\nexport var sqrLen = squaredLength;\n/**\n * Normalize a dual quat\n *\n * @param {quat2} out the receiving dual quaternion\n * @param {ReadonlyQuat2} a dual quaternion to normalize\n * @returns {quat2} out\n * @function\n */\n\nexport function normalize(out, a) {\n var magnitude = squaredLength(a);\n\n if (magnitude > 0) {\n magnitude = Math.sqrt(magnitude);\n var a0 = a[0] / magnitude;\n var a1 = a[1] / magnitude;\n var a2 = a[2] / magnitude;\n var a3 = a[3] / magnitude;\n var b0 = a[4];\n var b1 = a[5];\n var b2 = a[6];\n var b3 = a[7];\n var a_dot_b = a0 * b0 + a1 * b1 + a2 * b2 + a3 * b3;\n out[0] = a0;\n out[1] = a1;\n out[2] = a2;\n out[3] = a3;\n out[4] = (b0 - a0 * a_dot_b) / magnitude;\n out[5] = (b1 - a1 * a_dot_b) / magnitude;\n out[6] = (b2 - a2 * a_dot_b) / magnitude;\n out[7] = (b3 - a3 * a_dot_b) / magnitude;\n }\n\n return out;\n}\n/**\n * Returns a string representation of a dual quatenion\n *\n * @param {ReadonlyQuat2} a dual quaternion to represent as a string\n * @returns {String} string representation of the dual quat\n */\n\nexport function str(a) {\n return \"quat2(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \", \" + a[4] + \", \" + a[5] + \", \" + a[6] + \", \" + a[7] + \")\";\n}\n/**\n * Returns whether or not the dual quaternions have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyQuat2} a the first dual quaternion.\n * @param {ReadonlyQuat2} b the second dual quaternion.\n * @returns {Boolean} true if the dual quaternions are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5] && a[6] === b[6] && a[7] === b[7];\n}\n/**\n * Returns whether or not the dual quaternions have approximately the same elements in the same position.\n *\n * @param {ReadonlyQuat2} a the first dual quat.\n * @param {ReadonlyQuat2} b the second dual quat.\n * @returns {Boolean} true if the dual quats are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3],\n a4 = a[4],\n a5 = a[5],\n a6 = a[6],\n a7 = a[7];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3],\n b4 = b[4],\n b5 = b[5],\n b6 = b[6],\n b7 = b[7];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5)) && Math.abs(a6 - b6) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a6), Math.abs(b6)) && Math.abs(a7 - b7) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a7), Math.abs(b7));\n}","import * as glMatrix from \"./common.js\";\n/**\n * 2 Dimensional Vector\n * @module vec2\n */\n\n/**\n * Creates a new, empty vec2\n *\n * @returns {vec2} a new 2D vector\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(2);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[0] = 0;\n out[1] = 0;\n }\n\n return out;\n}\n/**\n * Creates a new vec2 initialized with values from an existing vector\n *\n * @param {ReadonlyVec2} a vector to clone\n * @returns {vec2} a new 2D vector\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(2);\n out[0] = a[0];\n out[1] = a[1];\n return out;\n}\n/**\n * Creates a new vec2 initialized with the given values\n *\n * @param {Number} x X component\n * @param {Number} y Y component\n * @returns {vec2} a new 2D vector\n */\n\nexport function fromValues(x, y) {\n var out = new glMatrix.ARRAY_TYPE(2);\n out[0] = x;\n out[1] = y;\n return out;\n}\n/**\n * Copy the values from one vec2 to another\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the source vector\n * @returns {vec2} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n return out;\n}\n/**\n * Set the components of a vec2 to the given values\n *\n * @param {vec2} out the receiving vector\n * @param {Number} x X component\n * @param {Number} y Y component\n * @returns {vec2} out\n */\n\nexport function set(out, x, y) {\n out[0] = x;\n out[1] = y;\n return out;\n}\n/**\n * Adds two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n return out;\n}\n/**\n * Subtracts vector b from vector a\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n return out;\n}\n/**\n * Multiplies two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function multiply(out, a, b) {\n out[0] = a[0] * b[0];\n out[1] = a[1] * b[1];\n return out;\n}\n/**\n * Divides two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function divide(out, a, b) {\n out[0] = a[0] / b[0];\n out[1] = a[1] / b[1];\n return out;\n}\n/**\n * Math.ceil the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to ceil\n * @returns {vec2} out\n */\n\nexport function ceil(out, a) {\n out[0] = Math.ceil(a[0]);\n out[1] = Math.ceil(a[1]);\n return out;\n}\n/**\n * Math.floor the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to floor\n * @returns {vec2} out\n */\n\nexport function floor(out, a) {\n out[0] = Math.floor(a[0]);\n out[1] = Math.floor(a[1]);\n return out;\n}\n/**\n * Returns the minimum of two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function min(out, a, b) {\n out[0] = Math.min(a[0], b[0]);\n out[1] = Math.min(a[1], b[1]);\n return out;\n}\n/**\n * Returns the maximum of two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function max(out, a, b) {\n out[0] = Math.max(a[0], b[0]);\n out[1] = Math.max(a[1], b[1]);\n return out;\n}\n/**\n * Math.round the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to round\n * @returns {vec2} out\n */\n\nexport function round(out, a) {\n out[0] = Math.round(a[0]);\n out[1] = Math.round(a[1]);\n return out;\n}\n/**\n * Scales a vec2 by a scalar number\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to scale\n * @param {Number} b amount to scale the vector by\n * @returns {vec2} out\n */\n\nexport function scale(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n return out;\n}\n/**\n * Adds two vec2's after scaling the second operand by a scalar value\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @param {Number} scale the amount to scale b by before adding\n * @returns {vec2} out\n */\n\nexport function scaleAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n return out;\n}\n/**\n * Calculates the euclidian distance between two vec2's\n *\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {Number} distance between a and b\n */\n\nexport function distance(a, b) {\n var x = b[0] - a[0],\n y = b[1] - a[1];\n return Math.hypot(x, y);\n}\n/**\n * Calculates the squared euclidian distance between two vec2's\n *\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {Number} squared distance between a and b\n */\n\nexport function squaredDistance(a, b) {\n var x = b[0] - a[0],\n y = b[1] - a[1];\n return x * x + y * y;\n}\n/**\n * Calculates the length of a vec2\n *\n * @param {ReadonlyVec2} a vector to calculate length of\n * @returns {Number} length of a\n */\n\nexport function length(a) {\n var x = a[0],\n y = a[1];\n return Math.hypot(x, y);\n}\n/**\n * Calculates the squared length of a vec2\n *\n * @param {ReadonlyVec2} a vector to calculate squared length of\n * @returns {Number} squared length of a\n */\n\nexport function squaredLength(a) {\n var x = a[0],\n y = a[1];\n return x * x + y * y;\n}\n/**\n * Negates the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to negate\n * @returns {vec2} out\n */\n\nexport function negate(out, a) {\n out[0] = -a[0];\n out[1] = -a[1];\n return out;\n}\n/**\n * Returns the inverse of the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to invert\n * @returns {vec2} out\n */\n\nexport function inverse(out, a) {\n out[0] = 1.0 / a[0];\n out[1] = 1.0 / a[1];\n return out;\n}\n/**\n * Normalize a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to normalize\n * @returns {vec2} out\n */\n\nexport function normalize(out, a) {\n var x = a[0],\n y = a[1];\n var len = x * x + y * y;\n\n if (len > 0) {\n //TODO: evaluate use of glm_invsqrt here?\n len = 1 / Math.sqrt(len);\n }\n\n out[0] = a[0] * len;\n out[1] = a[1] * len;\n return out;\n}\n/**\n * Calculates the dot product of two vec2's\n *\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {Number} dot product of a and b\n */\n\nexport function dot(a, b) {\n return a[0] * b[0] + a[1] * b[1];\n}\n/**\n * Computes the cross product of two vec2's\n * Note that the cross product must by definition produce a 3D vector\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec3} out\n */\n\nexport function cross(out, a, b) {\n var z = a[0] * b[1] - a[1] * b[0];\n out[0] = out[1] = 0;\n out[2] = z;\n return out;\n}\n/**\n * Performs a linear interpolation between two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec2} out\n */\n\nexport function lerp(out, a, b, t) {\n var ax = a[0],\n ay = a[1];\n out[0] = ax + t * (b[0] - ax);\n out[1] = ay + t * (b[1] - ay);\n return out;\n}\n/**\n * Generates a random vector with the given scale\n *\n * @param {vec2} out the receiving vector\n * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned\n * @returns {vec2} out\n */\n\nexport function random(out, scale) {\n scale = scale || 1.0;\n var r = glMatrix.RANDOM() * 2.0 * Math.PI;\n out[0] = Math.cos(r) * scale;\n out[1] = Math.sin(r) * scale;\n return out;\n}\n/**\n * Transforms the vec2 with a mat2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to transform\n * @param {ReadonlyMat2} m matrix to transform with\n * @returns {vec2} out\n */\n\nexport function transformMat2(out, a, m) {\n var x = a[0],\n y = a[1];\n out[0] = m[0] * x + m[2] * y;\n out[1] = m[1] * x + m[3] * y;\n return out;\n}\n/**\n * Transforms the vec2 with a mat2d\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to transform\n * @param {ReadonlyMat2d} m matrix to transform with\n * @returns {vec2} out\n */\n\nexport function transformMat2d(out, a, m) {\n var x = a[0],\n y = a[1];\n out[0] = m[0] * x + m[2] * y + m[4];\n out[1] = m[1] * x + m[3] * y + m[5];\n return out;\n}\n/**\n * Transforms the vec2 with a mat3\n * 3rd vector component is implicitly '1'\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to transform\n * @param {ReadonlyMat3} m matrix to transform with\n * @returns {vec2} out\n */\n\nexport function transformMat3(out, a, m) {\n var x = a[0],\n y = a[1];\n out[0] = m[0] * x + m[3] * y + m[6];\n out[1] = m[1] * x + m[4] * y + m[7];\n return out;\n}\n/**\n * Transforms the vec2 with a mat4\n * 3rd vector component is implicitly '0'\n * 4th vector component is implicitly '1'\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to transform\n * @param {ReadonlyMat4} m matrix to transform with\n * @returns {vec2} out\n */\n\nexport function transformMat4(out, a, m) {\n var x = a[0];\n var y = a[1];\n out[0] = m[0] * x + m[4] * y + m[12];\n out[1] = m[1] * x + m[5] * y + m[13];\n return out;\n}\n/**\n * Rotate a 2D vector\n * @param {vec2} out The receiving vec2\n * @param {ReadonlyVec2} a The vec2 point to rotate\n * @param {ReadonlyVec2} b The origin of the rotation\n * @param {Number} rad The angle of rotation in radians\n * @returns {vec2} out\n */\n\nexport function rotate(out, a, b, rad) {\n //Translate point to the origin\n var p0 = a[0] - b[0],\n p1 = a[1] - b[1],\n sinC = Math.sin(rad),\n cosC = Math.cos(rad); //perform rotation and translate to correct position\n\n out[0] = p0 * cosC - p1 * sinC + b[0];\n out[1] = p0 * sinC + p1 * cosC + b[1];\n return out;\n}\n/**\n * Get the angle between two 2D vectors\n * @param {ReadonlyVec2} a The first operand\n * @param {ReadonlyVec2} b The second operand\n * @returns {Number} The angle in radians\n */\n\nexport function angle(a, b) {\n var x1 = a[0],\n y1 = a[1],\n x2 = b[0],\n y2 = b[1],\n // mag is the product of the magnitudes of a and b\n mag = Math.sqrt(x1 * x1 + y1 * y1) * Math.sqrt(x2 * x2 + y2 * y2),\n // mag &&.. short circuits if mag == 0\n cosine = mag && (x1 * x2 + y1 * y2) / mag; // Math.min(Math.max(cosine, -1), 1) clamps the cosine between -1 and 1\n\n return Math.acos(Math.min(Math.max(cosine, -1), 1));\n}\n/**\n * Set the components of a vec2 to zero\n *\n * @param {vec2} out the receiving vector\n * @returns {vec2} out\n */\n\nexport function zero(out) {\n out[0] = 0.0;\n out[1] = 0.0;\n return out;\n}\n/**\n * Returns a string representation of a vector\n *\n * @param {ReadonlyVec2} a vector to represent as a string\n * @returns {String} string representation of the vector\n */\n\nexport function str(a) {\n return \"vec2(\" + a[0] + \", \" + a[1] + \")\";\n}\n/**\n * Returns whether or not the vectors exactly have the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyVec2} a The first vector.\n * @param {ReadonlyVec2} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1];\n}\n/**\n * Returns whether or not the vectors have approximately the same elements in the same position.\n *\n * @param {ReadonlyVec2} a The first vector.\n * @param {ReadonlyVec2} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1];\n var b0 = b[0],\n b1 = b[1];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1));\n}\n/**\n * Alias for {@link vec2.length}\n * @function\n */\n\nexport var len = length;\n/**\n * Alias for {@link vec2.subtract}\n * @function\n */\n\nexport var sub = subtract;\n/**\n * Alias for {@link vec2.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link vec2.divide}\n * @function\n */\n\nexport var div = divide;\n/**\n * Alias for {@link vec2.distance}\n * @function\n */\n\nexport var dist = distance;\n/**\n * Alias for {@link vec2.squaredDistance}\n * @function\n */\n\nexport var sqrDist = squaredDistance;\n/**\n * Alias for {@link vec2.squaredLength}\n * @function\n */\n\nexport var sqrLen = squaredLength;\n/**\n * Perform some operation over an array of vec2s.\n *\n * @param {Array} a the array of vectors to iterate over\n * @param {Number} stride Number of elements between the start of each vec2. If 0 assumes tightly packed\n * @param {Number} offset Number of elements to skip at the beginning of the array\n * @param {Number} count Number of vec2s to iterate over. If 0 iterates over entire array\n * @param {Function} fn Function to call for each vector in the array\n * @param {Object} [arg] additional argument to pass to fn\n * @returns {Array} a\n * @function\n */\n\nexport var forEach = function () {\n var vec = create();\n return function (a, stride, offset, count, fn, arg) {\n var i, l;\n\n if (!stride) {\n stride = 2;\n }\n\n if (!offset) {\n offset = 0;\n }\n\n if (count) {\n l = Math.min(count * stride + offset, a.length);\n } else {\n l = a.length;\n }\n\n for (i = offset; i < l; i += stride) {\n vec[0] = a[i];\n vec[1] = a[i + 1];\n fn(vec, vec, arg);\n a[i] = vec[0];\n a[i + 1] = vec[1];\n }\n\n return a;\n };\n}();","import {StyleLayer} from '../style_layer';\n\nimport {CircleBucket} from '../../data/bucket/circle_bucket';\nimport {polygonIntersectsBufferedPoint} from '../../util/intersection_tests';\nimport {getMaximumPaintValue, translateDistance, translate} from '../query_utils';\nimport properties, {CircleLayoutPropsPossiblyEvaluated, CirclePaintPropsPossiblyEvaluated} from './circle_style_layer_properties.g';\nimport {Transitionable, Transitioning, Layout, PossiblyEvaluated} from '../properties';\nimport {mat4, vec4} from 'gl-matrix';\nimport Point from '@mapbox/point-geometry';\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Transform} from '../../geo/transform';\nimport type {Bucket, BucketParameters} from '../../data/bucket';\nimport type {CircleLayoutProps, CirclePaintProps} from './circle_style_layer_properties.g';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\n/**\n * A style layer that defines a circle\n */\nexport class CircleStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n createBucket(parameters: BucketParameters) {\n return new CircleBucket(parameters);\n }\n\n queryRadius(bucket: Bucket): number {\n const circleBucket: CircleBucket = (bucket as any);\n return getMaximumPaintValue('circle-radius', this, circleBucket) +\n getMaximumPaintValue('circle-stroke-width', this, circleBucket) +\n translateDistance(this.paint.get('circle-translate'));\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number,\n pixelPosMatrix: mat4\n ): boolean {\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('circle-translate'),\n this.paint.get('circle-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n const radius = this.paint.get('circle-radius').evaluate(feature, featureState);\n const stroke = this.paint.get('circle-stroke-width').evaluate(feature, featureState);\n const size = radius + stroke;\n\n // For pitch-alignment: map, compare feature geometry to query geometry in the plane of the tile\n // // Otherwise, compare geometry in the plane of the viewport\n // // A circle with fixed scaling relative to the viewport gets larger in tile space as it moves into the distance\n // // A circle with fixed scaling relative to the map gets smaller in viewport space as it moves into the distance\n const alignWithMap = this.paint.get('circle-pitch-alignment') === 'map';\n const transformedPolygon = alignWithMap ? translatedPolygon : projectQueryGeometry(translatedPolygon, pixelPosMatrix);\n const transformedSize = alignWithMap ? size * pixelsToTileUnits : size;\n\n for (const ring of geometry) {\n for (const point of ring) {\n\n const transformedPoint = alignWithMap ? point : projectPoint(point, pixelPosMatrix);\n\n let adjustedSize = transformedSize;\n const projectedCenter = vec4.transformMat4([] as any, [point.x, point.y, 0, 1], pixelPosMatrix);\n if (this.paint.get('circle-pitch-scale') === 'viewport' && this.paint.get('circle-pitch-alignment') === 'map') {\n adjustedSize *= projectedCenter[3] / transform.cameraToCenterDistance;\n } else if (this.paint.get('circle-pitch-scale') === 'map' && this.paint.get('circle-pitch-alignment') === 'viewport') {\n adjustedSize *= transform.cameraToCenterDistance / projectedCenter[3];\n }\n\n if (polygonIntersectsBufferedPoint(transformedPolygon, transformedPoint, adjustedSize)) return true;\n }\n }\n\n return false;\n }\n}\n\nfunction projectPoint(p: Point, pixelPosMatrix: mat4) {\n const point = vec4.transformMat4([] as any, [p.x, p.y, 0, 1], pixelPosMatrix);\n return new Point(point[0] / point[3], point[1] / point[3]);\n}\n\nfunction projectQueryGeometry(queryGeometry: Array, pixelPosMatrix: mat4) {\n return queryGeometry.map((p) => {\n return projectPoint(p, pixelPosMatrix);\n });\n}\n","import {CircleBucket} from './circle_bucket';\nimport {register} from '../../util/web_worker_transfer';\n\nimport type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer';\n\nexport class HeatmapBucket extends CircleBucket {\n // Needed for flow to accept omit: ['layers'] below, due to\n // https://github.com/facebook/flow/issues/4262\n layers: Array;\n}\n\nregister('HeatmapBucket', HeatmapBucket, {omit: ['layers']});\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type HeatmapPaintProps = {\n \"heatmap-radius\": DataDrivenProperty,\n \"heatmap-weight\": DataDrivenProperty,\n \"heatmap-intensity\": DataConstantProperty,\n \"heatmap-color\": ColorRampProperty,\n \"heatmap-opacity\": DataConstantProperty,\n};\n\nexport type HeatmapPaintPropsPossiblyEvaluated = {\n \"heatmap-radius\": PossiblyEvaluatedPropertyValue,\n \"heatmap-weight\": PossiblyEvaluatedPropertyValue,\n \"heatmap-intensity\": number,\n \"heatmap-color\": ColorRampProperty,\n \"heatmap-opacity\": number,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"heatmap-radius\": new DataDrivenProperty(styleSpec[\"paint_heatmap\"][\"heatmap-radius\"] as any as StylePropertySpecification),\n \"heatmap-weight\": new DataDrivenProperty(styleSpec[\"paint_heatmap\"][\"heatmap-weight\"] as any as StylePropertySpecification),\n \"heatmap-intensity\": new DataConstantProperty(styleSpec[\"paint_heatmap\"][\"heatmap-intensity\"] as any as StylePropertySpecification),\n \"heatmap-color\": new ColorRampProperty(styleSpec[\"paint_heatmap\"][\"heatmap-color\"] as any as StylePropertySpecification),\n \"heatmap-opacity\": new DataConstantProperty(styleSpec[\"paint_heatmap\"][\"heatmap-opacity\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {register} from './web_worker_transfer';\n\nexport type Size = {\n width: number;\n height: number;\n};\n\ntype Point2D = {\n x: number;\n y: number;\n};\n\nfunction createImage(image: any, {\n width,\n height\n}: Size, channels: number, data?: Uint8Array | Uint8ClampedArray) {\n if (!data) {\n data = new Uint8Array(width * height * channels);\n } else if (data instanceof Uint8ClampedArray) {\n data = new Uint8Array(data.buffer);\n } else if (data.length !== width * height * channels) {\n throw new RangeError(`mismatched image size. expected: ${data.length} but got: ${width * height * channels}`);\n }\n image.width = width;\n image.height = height;\n image.data = data;\n return image;\n}\n\nfunction resizeImage(image: any, {\n width,\n height\n}: Size, channels: number) {\n if (width === image.width && height === image.height) {\n return;\n }\n\n const newImage = createImage({}, {width, height}, channels);\n\n copyImage(image, newImage, {x: 0, y: 0}, {x: 0, y: 0}, {\n width: Math.min(image.width, width),\n height: Math.min(image.height, height)\n }, channels);\n\n image.width = width;\n image.height = height;\n image.data = newImage.data;\n}\n\nfunction copyImage(srcImg: any, dstImg: any, srcPt: Point2D, dstPt: Point2D, size: Size, channels: number) {\n if (size.width === 0 || size.height === 0) {\n return dstImg;\n }\n\n if (size.width > srcImg.width ||\n size.height > srcImg.height ||\n srcPt.x > srcImg.width - size.width ||\n srcPt.y > srcImg.height - size.height) {\n throw new RangeError('out of range source coordinates for image copy');\n }\n\n if (size.width > dstImg.width ||\n size.height > dstImg.height ||\n dstPt.x > dstImg.width - size.width ||\n dstPt.y > dstImg.height - size.height) {\n throw new RangeError('out of range destination coordinates for image copy');\n }\n\n const srcData = srcImg.data;\n const dstData = dstImg.data;\n\n if (srcData === dstData) throw new Error('srcData equals dstData, so image is already copied');\n\n for (let y = 0; y < size.height; y++) {\n const srcOffset = ((srcPt.y + y) * srcImg.width + srcPt.x) * channels;\n const dstOffset = ((dstPt.y + y) * dstImg.width + dstPt.x) * channels;\n for (let i = 0; i < size.width * channels; i++) {\n dstData[dstOffset + i] = srcData[srcOffset + i];\n }\n }\n return dstImg;\n}\n\n/**\n * @internal\n * An image with alpha color value\n */\nexport class AlphaImage {\n width: number;\n height: number;\n data: Uint8Array;\n\n constructor(size: Size, data?: Uint8Array | Uint8ClampedArray) {\n createImage(this, size, 1, data);\n }\n\n resize(size: Size) {\n resizeImage(this, size, 1);\n }\n\n clone() {\n return new AlphaImage({width: this.width, height: this.height}, new Uint8Array(this.data));\n }\n\n static copy(srcImg: AlphaImage, dstImg: AlphaImage, srcPt: Point2D, dstPt: Point2D, size: Size) {\n copyImage(srcImg, dstImg, srcPt, dstPt, size, 1);\n }\n}\n\n/**\n * An object to store image data not premultiplied, because ImageData is not premultiplied.\n * UNPACK_PREMULTIPLY_ALPHA_WEBGL must be used when uploading to a texture.\n */\nexport class RGBAImage {\n width: number;\n height: number;\n\n /**\n * data must be a Uint8Array instead of Uint8ClampedArray because texImage2D does not support Uint8ClampedArray in all browsers.\n */\n data: Uint8Array;\n\n constructor(size: Size, data?: Uint8Array | Uint8ClampedArray) {\n createImage(this, size, 4, data);\n }\n\n resize(size: Size) {\n resizeImage(this, size, 4);\n }\n\n replace(data: Uint8Array | Uint8ClampedArray, copy?: boolean) {\n if (copy) {\n this.data.set(data);\n } else if (data instanceof Uint8ClampedArray) {\n this.data = new Uint8Array(data.buffer);\n } else {\n this.data = data;\n }\n }\n\n clone() {\n return new RGBAImage({width: this.width, height: this.height}, new Uint8Array(this.data));\n }\n\n static copy(srcImg: RGBAImage | ImageData, dstImg: RGBAImage, srcPt: Point2D, dstPt: Point2D, size: Size) {\n copyImage(srcImg, dstImg, srcPt, dstPt, size, 4);\n }\n}\n\nregister('AlphaImage', AlphaImage);\nregister('RGBAImage', RGBAImage);\n","import {RGBAImage} from './image';\nimport {isPowerOfTwo} from './util';\n\nimport type {StylePropertyExpression} from '@maplibre/maplibre-gl-style-spec';\n\nexport type ColorRampParams = {\n expression: StylePropertyExpression;\n evaluationKey: string;\n resolution?: number;\n image?: RGBAImage;\n clips?: Array;\n};\n\n/**\n * Given an expression that should evaluate to a color ramp,\n * return a RGBA image representing that ramp expression.\n */\nexport function renderColorRamp(params: ColorRampParams): RGBAImage {\n const evaluationGlobals = {};\n const width = params.resolution || 256;\n const height = params.clips ? params.clips.length : 1;\n const image = params.image || new RGBAImage({width, height});\n\n if (!isPowerOfTwo(width)) throw new Error(`width is not a power of 2 - ${width}`);\n\n const renderPixel = (stride, index, progress) => {\n evaluationGlobals[params.evaluationKey] = progress;\n const pxColor = params.expression.evaluate(evaluationGlobals as any);\n // the colors are being unpremultiplied because Color uses\n // premultiplied values, and the Texture class expects unpremultiplied ones\n image.data[stride + index + 0] = Math.floor(pxColor.r * 255 / pxColor.a);\n image.data[stride + index + 1] = Math.floor(pxColor.g * 255 / pxColor.a);\n image.data[stride + index + 2] = Math.floor(pxColor.b * 255 / pxColor.a);\n image.data[stride + index + 3] = Math.floor(pxColor.a * 255);\n };\n\n if (!params.clips) {\n for (let i = 0, j = 0; i < width; i++, j += 4) {\n const progress = i / (width - 1);\n\n renderPixel(0, j, progress);\n }\n } else {\n for (let clip = 0, stride = 0; clip < height; ++clip, stride += width * 4) {\n for (let i = 0, j = 0; i < width; i++, j += 4) {\n // Remap progress between clips\n const progress = i / (width - 1);\n const {start, end} = params.clips[clip];\n const evaluationProgress = start * (1 - progress) + end * progress;\n renderPixel(stride, j, evaluationProgress);\n }\n }\n }\n\n return image;\n}\n","import {StyleLayer} from '../style_layer';\n\nimport {HeatmapBucket} from '../../data/bucket/heatmap_bucket';\nimport {RGBAImage} from '../../util/image';\nimport properties, {HeatmapPaintPropsPossiblyEvaluated} from './heatmap_style_layer_properties.g';\nimport {renderColorRamp} from '../../util/color_ramp';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {Texture} from '../../render/texture';\nimport type {Framebuffer} from '../../gl/framebuffer';\nimport type {HeatmapPaintProps} from './heatmap_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A style layer that defines a heatmap\n */\nexport class HeatmapStyleLayer extends StyleLayer {\n\n heatmapFbo: Framebuffer;\n colorRamp: RGBAImage;\n colorRampTexture: Texture;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n createBucket(options: any) {\n return new HeatmapBucket(options);\n }\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n\n // make sure color ramp texture is generated for default heatmap color too\n this._updateColorRamp();\n }\n\n _handleSpecialPaintPropertyUpdate(name: string) {\n if (name === 'heatmap-color') {\n this._updateColorRamp();\n }\n }\n\n _updateColorRamp() {\n const expression = this._transitionablePaint._values['heatmap-color'].value.expression;\n this.colorRamp = renderColorRamp({\n expression,\n evaluationKey: 'heatmapDensity',\n image: this.colorRamp\n });\n this.colorRampTexture = null;\n }\n\n resize() {\n if (this.heatmapFbo) {\n this.heatmapFbo.destroy();\n this.heatmapFbo = null;\n }\n }\n\n queryRadius(): number {\n return 0;\n }\n\n queryIntersectsFeature(): boolean {\n return false;\n }\n\n hasOffscreenPass() {\n return this.paint.get('heatmap-opacity') !== 0 && this.visibility !== 'none';\n }\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type HillshadePaintProps = {\n \"hillshade-illumination-direction\": DataConstantProperty,\n \"hillshade-illumination-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"hillshade-exaggeration\": DataConstantProperty,\n \"hillshade-shadow-color\": DataConstantProperty,\n \"hillshade-highlight-color\": DataConstantProperty,\n \"hillshade-accent-color\": DataConstantProperty,\n};\n\nexport type HillshadePaintPropsPossiblyEvaluated = {\n \"hillshade-illumination-direction\": number,\n \"hillshade-illumination-anchor\": \"map\" | \"viewport\",\n \"hillshade-exaggeration\": number,\n \"hillshade-shadow-color\": Color,\n \"hillshade-highlight-color\": Color,\n \"hillshade-accent-color\": Color,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"hillshade-illumination-direction\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-illumination-direction\"] as any as StylePropertySpecification),\n \"hillshade-illumination-anchor\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-illumination-anchor\"] as any as StylePropertySpecification),\n \"hillshade-exaggeration\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-exaggeration\"] as any as StylePropertySpecification),\n \"hillshade-shadow-color\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-shadow-color\"] as any as StylePropertySpecification),\n \"hillshade-highlight-color\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-highlight-color\"] as any as StylePropertySpecification),\n \"hillshade-accent-color\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-accent-color\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport properties, {HillshadePaintPropsPossiblyEvaluated} from './hillshade_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {HillshadePaintProps} from './hillshade_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport class HillshadeStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n hasOffscreenPass() {\n return this.paint.get('hillshade-exaggeration') !== 0 && this.visibility !== 'none';\n }\n}\n","import {createLayout} from '../../util/struct_array';\n\nconst layout = createLayout([\n {name: 'a_pos', components: 2, type: 'Int16'}\n], 4);\n\nexport default layout;\nexport const {members, size, alignment} = layout;\n","'use strict';\n\nmodule.exports = earcut;\nmodule.exports.default = earcut;\n\nfunction earcut(data, holeIndices, dim) {\n\n dim = dim || 2;\n\n var hasHoles = holeIndices && holeIndices.length,\n outerLen = hasHoles ? holeIndices[0] * dim : data.length,\n outerNode = linkedList(data, 0, outerLen, dim, true),\n triangles = [];\n\n if (!outerNode || outerNode.next === outerNode.prev) return triangles;\n\n var minX, minY, maxX, maxY, x, y, invSize;\n\n if (hasHoles) outerNode = eliminateHoles(data, holeIndices, outerNode, dim);\n\n // if the shape is not too simple, we'll use z-order curve hash later; calculate polygon bbox\n if (data.length > 80 * dim) {\n minX = maxX = data[0];\n minY = maxY = data[1];\n\n for (var i = dim; i < outerLen; i += dim) {\n x = data[i];\n y = data[i + 1];\n if (x < minX) minX = x;\n if (y < minY) minY = y;\n if (x > maxX) maxX = x;\n if (y > maxY) maxY = y;\n }\n\n // minX, minY and invSize are later used to transform coords into integers for z-order calculation\n invSize = Math.max(maxX - minX, maxY - minY);\n invSize = invSize !== 0 ? 32767 / invSize : 0;\n }\n\n earcutLinked(outerNode, triangles, dim, minX, minY, invSize, 0);\n\n return triangles;\n}\n\n// create a circular doubly linked list from polygon points in the specified winding order\nfunction linkedList(data, start, end, dim, clockwise) {\n var i, last;\n\n if (clockwise === (signedArea(data, start, end, dim) > 0)) {\n for (i = start; i < end; i += dim) last = insertNode(i, data[i], data[i + 1], last);\n } else {\n for (i = end - dim; i >= start; i -= dim) last = insertNode(i, data[i], data[i + 1], last);\n }\n\n if (last && equals(last, last.next)) {\n removeNode(last);\n last = last.next;\n }\n\n return last;\n}\n\n// eliminate colinear or duplicate points\nfunction filterPoints(start, end) {\n if (!start) return start;\n if (!end) end = start;\n\n var p = start,\n again;\n do {\n again = false;\n\n if (!p.steiner && (equals(p, p.next) || area(p.prev, p, p.next) === 0)) {\n removeNode(p);\n p = end = p.prev;\n if (p === p.next) break;\n again = true;\n\n } else {\n p = p.next;\n }\n } while (again || p !== end);\n\n return end;\n}\n\n// main ear slicing loop which triangulates a polygon (given as a linked list)\nfunction earcutLinked(ear, triangles, dim, minX, minY, invSize, pass) {\n if (!ear) return;\n\n // interlink polygon nodes in z-order\n if (!pass && invSize) indexCurve(ear, minX, minY, invSize);\n\n var stop = ear,\n prev, next;\n\n // iterate through ears, slicing them one by one\n while (ear.prev !== ear.next) {\n prev = ear.prev;\n next = ear.next;\n\n if (invSize ? isEarHashed(ear, minX, minY, invSize) : isEar(ear)) {\n // cut off the triangle\n triangles.push(prev.i / dim | 0);\n triangles.push(ear.i / dim | 0);\n triangles.push(next.i / dim | 0);\n\n removeNode(ear);\n\n // skipping the next vertex leads to less sliver triangles\n ear = next.next;\n stop = next.next;\n\n continue;\n }\n\n ear = next;\n\n // if we looped through the whole remaining polygon and can't find any more ears\n if (ear === stop) {\n // try filtering points and slicing again\n if (!pass) {\n earcutLinked(filterPoints(ear), triangles, dim, minX, minY, invSize, 1);\n\n // if this didn't work, try curing all small self-intersections locally\n } else if (pass === 1) {\n ear = cureLocalIntersections(filterPoints(ear), triangles, dim);\n earcutLinked(ear, triangles, dim, minX, minY, invSize, 2);\n\n // as a last resort, try splitting the remaining polygon into two\n } else if (pass === 2) {\n splitEarcut(ear, triangles, dim, minX, minY, invSize);\n }\n\n break;\n }\n }\n}\n\n// check whether a polygon node forms a valid ear with adjacent nodes\nfunction isEar(ear) {\n var a = ear.prev,\n b = ear,\n c = ear.next;\n\n if (area(a, b, c) >= 0) return false; // reflex, can't be an ear\n\n // now make sure we don't have other points inside the potential ear\n var ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y;\n\n // triangle bbox; min & max are calculated like this for speed\n var x0 = ax < bx ? (ax < cx ? ax : cx) : (bx < cx ? bx : cx),\n y0 = ay < by ? (ay < cy ? ay : cy) : (by < cy ? by : cy),\n x1 = ax > bx ? (ax > cx ? ax : cx) : (bx > cx ? bx : cx),\n y1 = ay > by ? (ay > cy ? ay : cy) : (by > cy ? by : cy);\n\n var p = c.next;\n while (p !== a) {\n if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) &&\n area(p.prev, p, p.next) >= 0) return false;\n p = p.next;\n }\n\n return true;\n}\n\nfunction isEarHashed(ear, minX, minY, invSize) {\n var a = ear.prev,\n b = ear,\n c = ear.next;\n\n if (area(a, b, c) >= 0) return false; // reflex, can't be an ear\n\n var ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y;\n\n // triangle bbox; min & max are calculated like this for speed\n var x0 = ax < bx ? (ax < cx ? ax : cx) : (bx < cx ? bx : cx),\n y0 = ay < by ? (ay < cy ? ay : cy) : (by < cy ? by : cy),\n x1 = ax > bx ? (ax > cx ? ax : cx) : (bx > cx ? bx : cx),\n y1 = ay > by ? (ay > cy ? ay : cy) : (by > cy ? by : cy);\n\n // z-order range for the current triangle bbox;\n var minZ = zOrder(x0, y0, minX, minY, invSize),\n maxZ = zOrder(x1, y1, minX, minY, invSize);\n\n var p = ear.prevZ,\n n = ear.nextZ;\n\n // look for points inside the triangle in both directions\n while (p && p.z >= minZ && n && n.z <= maxZ) {\n if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 && p !== a && p !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) && area(p.prev, p, p.next) >= 0) return false;\n p = p.prevZ;\n\n if (n.x >= x0 && n.x <= x1 && n.y >= y0 && n.y <= y1 && n !== a && n !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) && area(n.prev, n, n.next) >= 0) return false;\n n = n.nextZ;\n }\n\n // look for remaining points in decreasing z-order\n while (p && p.z >= minZ) {\n if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 && p !== a && p !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) && area(p.prev, p, p.next) >= 0) return false;\n p = p.prevZ;\n }\n\n // look for remaining points in increasing z-order\n while (n && n.z <= maxZ) {\n if (n.x >= x0 && n.x <= x1 && n.y >= y0 && n.y <= y1 && n !== a && n !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) && area(n.prev, n, n.next) >= 0) return false;\n n = n.nextZ;\n }\n\n return true;\n}\n\n// go through all polygon nodes and cure small local self-intersections\nfunction cureLocalIntersections(start, triangles, dim) {\n var p = start;\n do {\n var a = p.prev,\n b = p.next.next;\n\n if (!equals(a, b) && intersects(a, p, p.next, b) && locallyInside(a, b) && locallyInside(b, a)) {\n\n triangles.push(a.i / dim | 0);\n triangles.push(p.i / dim | 0);\n triangles.push(b.i / dim | 0);\n\n // remove two nodes involved\n removeNode(p);\n removeNode(p.next);\n\n p = start = b;\n }\n p = p.next;\n } while (p !== start);\n\n return filterPoints(p);\n}\n\n// try splitting polygon into two and triangulate them independently\nfunction splitEarcut(start, triangles, dim, minX, minY, invSize) {\n // look for a valid diagonal that divides the polygon into two\n var a = start;\n do {\n var b = a.next.next;\n while (b !== a.prev) {\n if (a.i !== b.i && isValidDiagonal(a, b)) {\n // split the polygon in two by the diagonal\n var c = splitPolygon(a, b);\n\n // filter colinear points around the cuts\n a = filterPoints(a, a.next);\n c = filterPoints(c, c.next);\n\n // run earcut on each half\n earcutLinked(a, triangles, dim, minX, minY, invSize, 0);\n earcutLinked(c, triangles, dim, minX, minY, invSize, 0);\n return;\n }\n b = b.next;\n }\n a = a.next;\n } while (a !== start);\n}\n\n// link every hole into the outer loop, producing a single-ring polygon without holes\nfunction eliminateHoles(data, holeIndices, outerNode, dim) {\n var queue = [],\n i, len, start, end, list;\n\n for (i = 0, len = holeIndices.length; i < len; i++) {\n start = holeIndices[i] * dim;\n end = i < len - 1 ? holeIndices[i + 1] * dim : data.length;\n list = linkedList(data, start, end, dim, false);\n if (list === list.next) list.steiner = true;\n queue.push(getLeftmost(list));\n }\n\n queue.sort(compareX);\n\n // process holes from left to right\n for (i = 0; i < queue.length; i++) {\n outerNode = eliminateHole(queue[i], outerNode);\n }\n\n return outerNode;\n}\n\nfunction compareX(a, b) {\n return a.x - b.x;\n}\n\n// find a bridge between vertices that connects hole with an outer ring and and link it\nfunction eliminateHole(hole, outerNode) {\n var bridge = findHoleBridge(hole, outerNode);\n if (!bridge) {\n return outerNode;\n }\n\n var bridgeReverse = splitPolygon(bridge, hole);\n\n // filter collinear points around the cuts\n filterPoints(bridgeReverse, bridgeReverse.next);\n return filterPoints(bridge, bridge.next);\n}\n\n// David Eberly's algorithm for finding a bridge between hole and outer polygon\nfunction findHoleBridge(hole, outerNode) {\n var p = outerNode,\n hx = hole.x,\n hy = hole.y,\n qx = -Infinity,\n m;\n\n // find a segment intersected by a ray from the hole's leftmost point to the left;\n // segment's endpoint with lesser x will be potential connection point\n do {\n if (hy <= p.y && hy >= p.next.y && p.next.y !== p.y) {\n var x = p.x + (hy - p.y) * (p.next.x - p.x) / (p.next.y - p.y);\n if (x <= hx && x > qx) {\n qx = x;\n m = p.x < p.next.x ? p : p.next;\n if (x === hx) return m; // hole touches outer segment; pick leftmost endpoint\n }\n }\n p = p.next;\n } while (p !== outerNode);\n\n if (!m) return null;\n\n // look for points inside the triangle of hole point, segment intersection and endpoint;\n // if there are no points found, we have a valid connection;\n // otherwise choose the point of the minimum angle with the ray as connection point\n\n var stop = m,\n mx = m.x,\n my = m.y,\n tanMin = Infinity,\n tan;\n\n p = m;\n\n do {\n if (hx >= p.x && p.x >= mx && hx !== p.x &&\n pointInTriangle(hy < my ? hx : qx, hy, mx, my, hy < my ? qx : hx, hy, p.x, p.y)) {\n\n tan = Math.abs(hy - p.y) / (hx - p.x); // tangential\n\n if (locallyInside(p, hole) &&\n (tan < tanMin || (tan === tanMin && (p.x > m.x || (p.x === m.x && sectorContainsSector(m, p)))))) {\n m = p;\n tanMin = tan;\n }\n }\n\n p = p.next;\n } while (p !== stop);\n\n return m;\n}\n\n// whether sector in vertex m contains sector in vertex p in the same coordinates\nfunction sectorContainsSector(m, p) {\n return area(m.prev, m, p.prev) < 0 && area(p.next, m, m.next) < 0;\n}\n\n// interlink polygon nodes in z-order\nfunction indexCurve(start, minX, minY, invSize) {\n var p = start;\n do {\n if (p.z === 0) p.z = zOrder(p.x, p.y, minX, minY, invSize);\n p.prevZ = p.prev;\n p.nextZ = p.next;\n p = p.next;\n } while (p !== start);\n\n p.prevZ.nextZ = null;\n p.prevZ = null;\n\n sortLinked(p);\n}\n\n// Simon Tatham's linked list merge sort algorithm\n// http://www.chiark.greenend.org.uk/~sgtatham/algorithms/listsort.html\nfunction sortLinked(list) {\n var i, p, q, e, tail, numMerges, pSize, qSize,\n inSize = 1;\n\n do {\n p = list;\n list = null;\n tail = null;\n numMerges = 0;\n\n while (p) {\n numMerges++;\n q = p;\n pSize = 0;\n for (i = 0; i < inSize; i++) {\n pSize++;\n q = q.nextZ;\n if (!q) break;\n }\n qSize = inSize;\n\n while (pSize > 0 || (qSize > 0 && q)) {\n\n if (pSize !== 0 && (qSize === 0 || !q || p.z <= q.z)) {\n e = p;\n p = p.nextZ;\n pSize--;\n } else {\n e = q;\n q = q.nextZ;\n qSize--;\n }\n\n if (tail) tail.nextZ = e;\n else list = e;\n\n e.prevZ = tail;\n tail = e;\n }\n\n p = q;\n }\n\n tail.nextZ = null;\n inSize *= 2;\n\n } while (numMerges > 1);\n\n return list;\n}\n\n// z-order of a point given coords and inverse of the longer side of data bbox\nfunction zOrder(x, y, minX, minY, invSize) {\n // coords are transformed into non-negative 15-bit integer range\n x = (x - minX) * invSize | 0;\n y = (y - minY) * invSize | 0;\n\n x = (x | (x << 8)) & 0x00FF00FF;\n x = (x | (x << 4)) & 0x0F0F0F0F;\n x = (x | (x << 2)) & 0x33333333;\n x = (x | (x << 1)) & 0x55555555;\n\n y = (y | (y << 8)) & 0x00FF00FF;\n y = (y | (y << 4)) & 0x0F0F0F0F;\n y = (y | (y << 2)) & 0x33333333;\n y = (y | (y << 1)) & 0x55555555;\n\n return x | (y << 1);\n}\n\n// find the leftmost node of a polygon ring\nfunction getLeftmost(start) {\n var p = start,\n leftmost = start;\n do {\n if (p.x < leftmost.x || (p.x === leftmost.x && p.y < leftmost.y)) leftmost = p;\n p = p.next;\n } while (p !== start);\n\n return leftmost;\n}\n\n// check if a point lies within a convex triangle\nfunction pointInTriangle(ax, ay, bx, by, cx, cy, px, py) {\n return (cx - px) * (ay - py) >= (ax - px) * (cy - py) &&\n (ax - px) * (by - py) >= (bx - px) * (ay - py) &&\n (bx - px) * (cy - py) >= (cx - px) * (by - py);\n}\n\n// check if a diagonal between two polygon nodes is valid (lies in polygon interior)\nfunction isValidDiagonal(a, b) {\n return a.next.i !== b.i && a.prev.i !== b.i && !intersectsPolygon(a, b) && // dones't intersect other edges\n (locallyInside(a, b) && locallyInside(b, a) && middleInside(a, b) && // locally visible\n (area(a.prev, a, b.prev) || area(a, b.prev, b)) || // does not create opposite-facing sectors\n equals(a, b) && area(a.prev, a, a.next) > 0 && area(b.prev, b, b.next) > 0); // special zero-length case\n}\n\n// signed area of a triangle\nfunction area(p, q, r) {\n return (q.y - p.y) * (r.x - q.x) - (q.x - p.x) * (r.y - q.y);\n}\n\n// check if two points are equal\nfunction equals(p1, p2) {\n return p1.x === p2.x && p1.y === p2.y;\n}\n\n// check if two segments intersect\nfunction intersects(p1, q1, p2, q2) {\n var o1 = sign(area(p1, q1, p2));\n var o2 = sign(area(p1, q1, q2));\n var o3 = sign(area(p2, q2, p1));\n var o4 = sign(area(p2, q2, q1));\n\n if (o1 !== o2 && o3 !== o4) return true; // general case\n\n if (o1 === 0 && onSegment(p1, p2, q1)) return true; // p1, q1 and p2 are collinear and p2 lies on p1q1\n if (o2 === 0 && onSegment(p1, q2, q1)) return true; // p1, q1 and q2 are collinear and q2 lies on p1q1\n if (o3 === 0 && onSegment(p2, p1, q2)) return true; // p2, q2 and p1 are collinear and p1 lies on p2q2\n if (o4 === 0 && onSegment(p2, q1, q2)) return true; // p2, q2 and q1 are collinear and q1 lies on p2q2\n\n return false;\n}\n\n// for collinear points p, q, r, check if point q lies on segment pr\nfunction onSegment(p, q, r) {\n return q.x <= Math.max(p.x, r.x) && q.x >= Math.min(p.x, r.x) && q.y <= Math.max(p.y, r.y) && q.y >= Math.min(p.y, r.y);\n}\n\nfunction sign(num) {\n return num > 0 ? 1 : num < 0 ? -1 : 0;\n}\n\n// check if a polygon diagonal intersects any polygon segments\nfunction intersectsPolygon(a, b) {\n var p = a;\n do {\n if (p.i !== a.i && p.next.i !== a.i && p.i !== b.i && p.next.i !== b.i &&\n intersects(p, p.next, a, b)) return true;\n p = p.next;\n } while (p !== a);\n\n return false;\n}\n\n// check if a polygon diagonal is locally inside the polygon\nfunction locallyInside(a, b) {\n return area(a.prev, a, a.next) < 0 ?\n area(a, b, a.next) >= 0 && area(a, a.prev, b) >= 0 :\n area(a, b, a.prev) < 0 || area(a, a.next, b) < 0;\n}\n\n// check if the middle point of a polygon diagonal is inside the polygon\nfunction middleInside(a, b) {\n var p = a,\n inside = false,\n px = (a.x + b.x) / 2,\n py = (a.y + b.y) / 2;\n do {\n if (((p.y > py) !== (p.next.y > py)) && p.next.y !== p.y &&\n (px < (p.next.x - p.x) * (py - p.y) / (p.next.y - p.y) + p.x))\n inside = !inside;\n p = p.next;\n } while (p !== a);\n\n return inside;\n}\n\n// link two polygon vertices with a bridge; if the vertices belong to the same ring, it splits polygon into two;\n// if one belongs to the outer ring and another to a hole, it merges it into a single ring\nfunction splitPolygon(a, b) {\n var a2 = new Node(a.i, a.x, a.y),\n b2 = new Node(b.i, b.x, b.y),\n an = a.next,\n bp = b.prev;\n\n a.next = b;\n b.prev = a;\n\n a2.next = an;\n an.prev = a2;\n\n b2.next = a2;\n a2.prev = b2;\n\n bp.next = b2;\n b2.prev = bp;\n\n return b2;\n}\n\n// create a node and optionally link it with previous one (in a circular doubly linked list)\nfunction insertNode(i, x, y, last) {\n var p = new Node(i, x, y);\n\n if (!last) {\n p.prev = p;\n p.next = p;\n\n } else {\n p.next = last.next;\n p.prev = last;\n last.next.prev = p;\n last.next = p;\n }\n return p;\n}\n\nfunction removeNode(p) {\n p.next.prev = p.prev;\n p.prev.next = p.next;\n\n if (p.prevZ) p.prevZ.nextZ = p.nextZ;\n if (p.nextZ) p.nextZ.prevZ = p.prevZ;\n}\n\nfunction Node(i, x, y) {\n // vertex index in coordinates array\n this.i = i;\n\n // vertex coordinates\n this.x = x;\n this.y = y;\n\n // previous and next vertex nodes in a polygon ring\n this.prev = null;\n this.next = null;\n\n // z-order curve value\n this.z = 0;\n\n // previous and next nodes in z-order\n this.prevZ = null;\n this.nextZ = null;\n\n // indicates whether this is a steiner point\n this.steiner = false;\n}\n\n// return a percentage difference between the polygon area and its triangulation area;\n// used to verify correctness of triangulation\nearcut.deviation = function (data, holeIndices, dim, triangles) {\n var hasHoles = holeIndices && holeIndices.length;\n var outerLen = hasHoles ? holeIndices[0] * dim : data.length;\n\n var polygonArea = Math.abs(signedArea(data, 0, outerLen, dim));\n if (hasHoles) {\n for (var i = 0, len = holeIndices.length; i < len; i++) {\n var start = holeIndices[i] * dim;\n var end = i < len - 1 ? holeIndices[i + 1] * dim : data.length;\n polygonArea -= Math.abs(signedArea(data, start, end, dim));\n }\n }\n\n var trianglesArea = 0;\n for (i = 0; i < triangles.length; i += 3) {\n var a = triangles[i] * dim;\n var b = triangles[i + 1] * dim;\n var c = triangles[i + 2] * dim;\n trianglesArea += Math.abs(\n (data[a] - data[c]) * (data[b + 1] - data[a + 1]) -\n (data[a] - data[b]) * (data[c + 1] - data[a + 1]));\n }\n\n return polygonArea === 0 && trianglesArea === 0 ? 0 :\n Math.abs((trianglesArea - polygonArea) / polygonArea);\n};\n\nfunction signedArea(data, start, end, dim) {\n var sum = 0;\n for (var i = start, j = end - dim; i < end; i += dim) {\n sum += (data[j] - data[i]) * (data[i + 1] + data[j + 1]);\n j = i;\n }\n return sum;\n}\n\n// turn a polygon in a multi-dimensional array form (e.g. as in GeoJSON) into a form Earcut accepts\nearcut.flatten = function (data) {\n var dim = data[0][0].length,\n result = {vertices: [], holes: [], dimensions: dim},\n holeIndex = 0;\n\n for (var i = 0; i < data.length; i++) {\n for (var j = 0; j < data[i].length; j++) {\n for (var d = 0; d < dim; d++) result.vertices.push(data[i][j][d]);\n }\n if (i > 0) {\n holeIndex += data[i - 1].length;\n result.holes.push(holeIndex);\n }\n }\n return result;\n};\n","\nexport default function quickselect(arr, k, left, right, compare) {\n quickselectStep(arr, k, left || 0, right || (arr.length - 1), compare || defaultCompare);\n}\n\nfunction quickselectStep(arr, k, left, right, compare) {\n\n while (right > left) {\n if (right - left > 600) {\n var n = right - left + 1;\n var m = k - left + 1;\n var z = Math.log(n);\n var s = 0.5 * Math.exp(2 * z / 3);\n var sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);\n var newLeft = Math.max(left, Math.floor(k - m * s / n + sd));\n var newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));\n quickselectStep(arr, k, newLeft, newRight, compare);\n }\n\n var t = arr[k];\n var i = left;\n var j = right;\n\n swap(arr, left, k);\n if (compare(arr[right], t) > 0) swap(arr, left, right);\n\n while (i < j) {\n swap(arr, i, j);\n i++;\n j--;\n while (compare(arr[i], t) < 0) i++;\n while (compare(arr[j], t) > 0) j--;\n }\n\n if (compare(arr[left], t) === 0) swap(arr, left, j);\n else {\n j++;\n swap(arr, j, right);\n }\n\n if (j <= k) left = j + 1;\n if (k <= j) right = j - 1;\n }\n}\n\nfunction swap(arr, i, j) {\n var tmp = arr[i];\n arr[i] = arr[j];\n arr[j] = tmp;\n}\n\nfunction defaultCompare(a, b) {\n return a < b ? -1 : a > b ? 1 : 0;\n}\n","import quickselect from 'quickselect';\n\nimport {calculateSignedArea} from './util';\n\nimport type Point from '@mapbox/point-geometry';\n\n// classifies an array of rings into polygons with outer rings and holes\nexport function classifyRings(rings: Array>, maxRings: number) {\n const len = rings.length;\n\n if (len <= 1) return [rings];\n\n const polygons = [];\n let polygon,\n ccw;\n\n for (let i = 0; i < len; i++) {\n const area = calculateSignedArea(rings[i]);\n if (area === 0) continue;\n\n (rings[i] as any).area = Math.abs(area);\n\n if (ccw === undefined) ccw = area < 0;\n\n if (ccw === area < 0) {\n if (polygon) polygons.push(polygon);\n polygon = [rings[i]];\n\n } else {\n (polygon as any).push(rings[i]);\n }\n }\n if (polygon) polygons.push(polygon);\n\n // Earcut performance degrades with the # of rings in a polygon. For this\n // reason, we limit strip out all but the `maxRings` largest rings.\n if (maxRings > 1) {\n for (let j = 0; j < polygons.length; j++) {\n if (polygons[j].length <= maxRings) continue;\n quickselect(polygons[j], maxRings, 1, polygons[j].length - 1, compareAreas);\n polygons[j] = polygons[j].slice(0, maxRings);\n }\n }\n\n return polygons;\n}\n\nfunction compareAreas(a, b) {\n return b.area - a.area;\n}\n","import type {FillStyleLayer} from '../../style/style_layer/fill_style_layer';\nimport type {FillExtrusionStyleLayer} from '../../style/style_layer/fill_extrusion_style_layer';\nimport type {LineStyleLayer} from '../../style/style_layer/line_style_layer';\n\nimport type {\n BucketFeature,\n PopulateParameters\n} from '../bucket';\nimport {PossiblyEvaluated} from '../../style/properties';\n\ntype PatternStyleLayers = Array | Array | Array;\n\nexport function hasPattern(type: string, layers: PatternStyleLayers, options: PopulateParameters) {\n const patterns = options.patternDependencies;\n let hasPattern = false;\n\n for (const layer of layers) {\n const patternProperty = (layer.paint as PossiblyEvaluated).get(`${type}-pattern`);\n if (!patternProperty.isConstant()) {\n hasPattern = true;\n }\n\n const constantPattern = patternProperty.constantOr(null);\n if (constantPattern) {\n hasPattern = true;\n patterns[constantPattern.to] = true;\n patterns[constantPattern.from] = true;\n }\n }\n\n return hasPattern;\n}\n\nexport function addPatternDependencies(type: string, layers: PatternStyleLayers, patternFeature: BucketFeature, zoom: number, options: PopulateParameters) {\n const patterns = options.patternDependencies;\n for (const layer of layers) {\n const patternProperty = (layer.paint as PossiblyEvaluated).get(`${type}-pattern`);\n\n const patternPropertyValue = patternProperty.value;\n if (patternPropertyValue.kind !== 'constant') {\n let min = patternPropertyValue.evaluate({zoom: zoom - 1}, patternFeature, {}, options.availableImages);\n let mid = patternPropertyValue.evaluate({zoom}, patternFeature, {}, options.availableImages);\n let max = patternPropertyValue.evaluate({zoom: zoom + 1}, patternFeature, {}, options.availableImages);\n min = min && min.name ? min.name : min;\n mid = mid && mid.name ? mid.name : mid;\n max = max && max.name ? max.name : max;\n // add to patternDependencies\n patterns[min] = true;\n patterns[mid] = true;\n patterns[max] = true;\n\n // save for layout\n patternFeature.patterns[layer.id] = {min, mid, max};\n }\n }\n return patternFeature;\n}\n","import {FillLayoutArray} from '../array_types.g';\n\nimport {members as layoutAttributes} from './fill_attributes';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {LineIndexArray, TriangleIndexArray} from '../index_array_type';\nimport earcut from 'earcut';\nimport {classifyRings} from '../../util/classify_rings';\nconst EARCUT_MAX_RINGS = 500;\nimport {register} from '../../util/web_worker_transfer';\nimport {hasPattern, addPatternDependencies} from './pattern_bucket_features';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {FillStyleLayer} from '../../style/style_layer/fill_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nexport class FillBucket implements Bucket {\n index: number;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n patternFeatures: Array;\n\n layoutVertexArray: FillLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n indexArray2: LineIndexArray;\n indexBuffer2: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n segments2: SegmentVector;\n uploaded: boolean;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n this.patternFeatures = [];\n\n this.layoutVertexArray = new FillLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.indexArray2 = new LineIndexArray();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.segments = new SegmentVector();\n this.segments2 = new SegmentVector();\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n this.hasPattern = hasPattern('fill', this.layers, options);\n const fillSortKey = this.layers[0].layout.get('fill-sort-key');\n const sortFeaturesByKey = !fillSortKey.isConstant();\n const bucketFeatures: BucketFeature[] = [];\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const sortKey = sortFeaturesByKey ?\n fillSortKey.evaluate(evaluationFeature, {}, canonical, options.availableImages) :\n undefined;\n\n const bucketFeature: BucketFeature = {\n id,\n properties: feature.properties,\n type: feature.type,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n patterns: {},\n sortKey\n };\n\n bucketFeatures.push(bucketFeature);\n }\n\n if (sortFeaturesByKey) {\n bucketFeatures.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const bucketFeature of bucketFeatures) {\n const {geometry, index, sourceLayerIndex} = bucketFeature;\n\n if (this.hasPattern) {\n const patternFeature = addPatternDependencies('fill', this.layers, bucketFeature, this.zoom, options);\n // pattern features are added only once the pattern is loaded into the image atlas\n // so are stored during populate until later updated with positions by tile worker in addFeatures\n this.patternFeatures.push(patternFeature);\n } else {\n this.addFeature(bucketFeature, geometry, index, canonical, {});\n }\n\n const feature = features[index].feature;\n options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {\n [_: string]: ImagePosition;\n }) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {\n [_: string]: ImagePosition;\n }) {\n for (const feature of this.patternFeatures) {\n this.addFeature(feature, feature.geometry, feature.index, canonical, imagePositions);\n }\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0;\n }\n\n uploadPending(): boolean {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n upload(context: Context) {\n if (!this.uploaded) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n this.indexBuffer2 = context.createIndexBuffer(this.indexArray2);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.indexBuffer2.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n this.segments2.destroy();\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {\n [_: string]: ImagePosition;\n }) {\n for (const polygon of classifyRings(geometry, EARCUT_MAX_RINGS)) {\n let numVertices = 0;\n for (const ring of polygon) {\n numVertices += ring.length;\n }\n\n const triangleSegment = this.segments.prepareSegment(numVertices, this.layoutVertexArray, this.indexArray);\n const triangleIndex = triangleSegment.vertexLength;\n\n const flattened = [];\n const holeIndices = [];\n\n for (const ring of polygon) {\n if (ring.length === 0) {\n continue;\n }\n\n if (ring !== polygon[0]) {\n holeIndices.push(flattened.length / 2);\n }\n\n const lineSegment = this.segments2.prepareSegment(ring.length, this.layoutVertexArray, this.indexArray2);\n const lineIndex = lineSegment.vertexLength;\n\n this.layoutVertexArray.emplaceBack(ring[0].x, ring[0].y);\n this.indexArray2.emplaceBack(lineIndex + ring.length - 1, lineIndex);\n flattened.push(ring[0].x);\n flattened.push(ring[0].y);\n\n for (let i = 1; i < ring.length; i++) {\n this.layoutVertexArray.emplaceBack(ring[i].x, ring[i].y);\n this.indexArray2.emplaceBack(lineIndex + i - 1, lineIndex + i);\n flattened.push(ring[i].x);\n flattened.push(ring[i].y);\n }\n\n lineSegment.vertexLength += ring.length;\n lineSegment.primitiveLength += ring.length;\n }\n\n const indices = earcut(flattened, holeIndices);\n\n for (let i = 0; i < indices.length; i += 3) {\n this.indexArray.emplaceBack(\n triangleIndex + indices[i],\n triangleIndex + indices[i + 1],\n triangleIndex + indices[i + 2]);\n }\n\n triangleSegment.vertexLength += numVertices;\n triangleSegment.primitiveLength += indices.length / 3;\n }\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical);\n }\n}\n\nregister('FillBucket', FillBucket, {omit: ['layers', 'patternFeatures']});\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type FillLayoutProps = {\n \"fill-sort-key\": DataDrivenProperty,\n};\n\nexport type FillLayoutPropsPossiblyEvaluated = {\n \"fill-sort-key\": PossiblyEvaluatedPropertyValue,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"fill-sort-key\": new DataDrivenProperty(styleSpec[\"layout_fill\"][\"fill-sort-key\"] as any as StylePropertySpecification),\n});\n\nexport type FillPaintProps = {\n \"fill-antialias\": DataConstantProperty,\n \"fill-opacity\": DataDrivenProperty,\n \"fill-color\": DataDrivenProperty,\n \"fill-outline-color\": DataDrivenProperty,\n \"fill-translate\": DataConstantProperty<[number, number]>,\n \"fill-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"fill-pattern\": CrossFadedDataDrivenProperty,\n};\n\nexport type FillPaintPropsPossiblyEvaluated = {\n \"fill-antialias\": boolean,\n \"fill-opacity\": PossiblyEvaluatedPropertyValue,\n \"fill-color\": PossiblyEvaluatedPropertyValue,\n \"fill-outline-color\": PossiblyEvaluatedPropertyValue,\n \"fill-translate\": [number, number],\n \"fill-translate-anchor\": \"map\" | \"viewport\",\n \"fill-pattern\": PossiblyEvaluatedPropertyValue>,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"fill-antialias\": new DataConstantProperty(styleSpec[\"paint_fill\"][\"fill-antialias\"] as any as StylePropertySpecification),\n \"fill-opacity\": new DataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-opacity\"] as any as StylePropertySpecification),\n \"fill-color\": new DataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-color\"] as any as StylePropertySpecification),\n \"fill-outline-color\": new DataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-outline-color\"] as any as StylePropertySpecification),\n \"fill-translate\": new DataConstantProperty(styleSpec[\"paint_fill\"][\"fill-translate\"] as any as StylePropertySpecification),\n \"fill-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_fill\"][\"fill-translate-anchor\"] as any as StylePropertySpecification),\n \"fill-pattern\": new CrossFadedDataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-pattern\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import {StyleLayer} from '../style_layer';\n\nimport {FillBucket} from '../../data/bucket/fill_bucket';\nimport {polygonIntersectsMultiPolygon} from '../../util/intersection_tests';\nimport {translateDistance, translate} from '../query_utils';\nimport properties, {FillLayoutPropsPossiblyEvaluated, FillPaintPropsPossiblyEvaluated} from './fill_style_layer_properties.g';\nimport {Transitionable, Transitioning, Layout, PossiblyEvaluated} from '../properties';\n\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {BucketParameters} from '../../data/bucket';\nimport type Point from '@mapbox/point-geometry';\nimport type {FillLayoutProps, FillPaintProps} from './fill_style_layer_properties.g';\nimport type {EvaluationParameters} from '../evaluation_parameters';\nimport type {Transform} from '../../geo/transform';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nexport class FillStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n super.recalculate(parameters, availableImages);\n\n const outlineColor = this.paint._values['fill-outline-color'];\n if (outlineColor.value.kind === 'constant' && outlineColor.value.value === undefined) {\n this.paint._values['fill-outline-color'] = this.paint._values['fill-color'];\n }\n }\n\n createBucket(parameters: BucketParameters) {\n return new FillBucket(parameters);\n }\n\n queryRadius(): number {\n return translateDistance(this.paint.get('fill-translate'));\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number\n ): boolean {\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('fill-translate'),\n this.paint.get('fill-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n return polygonIntersectsMultiPolygon(translatedPolygon, geometry);\n }\n\n isTileClipped() {\n return true;\n }\n}\n","import {createLayout} from '../../util/struct_array';\n\nconst layout = createLayout([\n {name: 'a_pos', components: 2, type: 'Int16'},\n {name: 'a_normal_ed', components: 4, type: 'Int16'},\n], 4);\n\nexport const centroidAttributes = createLayout([\n {name: 'a_centroid', components: 2, type: 'Int16'}\n], 4);\n\nexport default layout;\nexport const {members, size, alignment} = layout;\n","'use strict';\n\nvar Point = require('@mapbox/point-geometry');\n\nmodule.exports = VectorTileFeature;\n\nfunction VectorTileFeature(pbf, end, extent, keys, values) {\n // Public\n this.properties = {};\n this.extent = extent;\n this.type = 0;\n\n // Private\n this._pbf = pbf;\n this._geometry = -1;\n this._keys = keys;\n this._values = values;\n\n pbf.readFields(readFeature, this, end);\n}\n\nfunction readFeature(tag, feature, pbf) {\n if (tag == 1) feature.id = pbf.readVarint();\n else if (tag == 2) readTag(pbf, feature);\n else if (tag == 3) feature.type = pbf.readVarint();\n else if (tag == 4) feature._geometry = pbf.pos;\n}\n\nfunction readTag(pbf, feature) {\n var end = pbf.readVarint() + pbf.pos;\n\n while (pbf.pos < end) {\n var key = feature._keys[pbf.readVarint()],\n value = feature._values[pbf.readVarint()];\n feature.properties[key] = value;\n }\n}\n\nVectorTileFeature.types = ['Unknown', 'Point', 'LineString', 'Polygon'];\n\nVectorTileFeature.prototype.loadGeometry = function() {\n var pbf = this._pbf;\n pbf.pos = this._geometry;\n\n var end = pbf.readVarint() + pbf.pos,\n cmd = 1,\n length = 0,\n x = 0,\n y = 0,\n lines = [],\n line;\n\n while (pbf.pos < end) {\n if (length <= 0) {\n var cmdLen = pbf.readVarint();\n cmd = cmdLen & 0x7;\n length = cmdLen >> 3;\n }\n\n length--;\n\n if (cmd === 1 || cmd === 2) {\n x += pbf.readSVarint();\n y += pbf.readSVarint();\n\n if (cmd === 1) { // moveTo\n if (line) lines.push(line);\n line = [];\n }\n\n line.push(new Point(x, y));\n\n } else if (cmd === 7) {\n\n // Workaround for https://github.com/mapbox/mapnik-vector-tile/issues/90\n if (line) {\n line.push(line[0].clone()); // closePolygon\n }\n\n } else {\n throw new Error('unknown command ' + cmd);\n }\n }\n\n if (line) lines.push(line);\n\n return lines;\n};\n\nVectorTileFeature.prototype.bbox = function() {\n var pbf = this._pbf;\n pbf.pos = this._geometry;\n\n var end = pbf.readVarint() + pbf.pos,\n cmd = 1,\n length = 0,\n x = 0,\n y = 0,\n x1 = Infinity,\n x2 = -Infinity,\n y1 = Infinity,\n y2 = -Infinity;\n\n while (pbf.pos < end) {\n if (length <= 0) {\n var cmdLen = pbf.readVarint();\n cmd = cmdLen & 0x7;\n length = cmdLen >> 3;\n }\n\n length--;\n\n if (cmd === 1 || cmd === 2) {\n x += pbf.readSVarint();\n y += pbf.readSVarint();\n if (x < x1) x1 = x;\n if (x > x2) x2 = x;\n if (y < y1) y1 = y;\n if (y > y2) y2 = y;\n\n } else if (cmd !== 7) {\n throw new Error('unknown command ' + cmd);\n }\n }\n\n return [x1, y1, x2, y2];\n};\n\nVectorTileFeature.prototype.toGeoJSON = function(x, y, z) {\n var size = this.extent * Math.pow(2, z),\n x0 = this.extent * x,\n y0 = this.extent * y,\n coords = this.loadGeometry(),\n type = VectorTileFeature.types[this.type],\n i, j;\n\n function project(line) {\n for (var j = 0; j < line.length; j++) {\n var p = line[j], y2 = 180 - (p.y + y0) * 360 / size;\n line[j] = [\n (p.x + x0) * 360 / size - 180,\n 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90\n ];\n }\n }\n\n switch (this.type) {\n case 1:\n var points = [];\n for (i = 0; i < coords.length; i++) {\n points[i] = coords[i][0];\n }\n coords = points;\n project(coords);\n break;\n\n case 2:\n for (i = 0; i < coords.length; i++) {\n project(coords[i]);\n }\n break;\n\n case 3:\n coords = classifyRings(coords);\n for (i = 0; i < coords.length; i++) {\n for (j = 0; j < coords[i].length; j++) {\n project(coords[i][j]);\n }\n }\n break;\n }\n\n if (coords.length === 1) {\n coords = coords[0];\n } else {\n type = 'Multi' + type;\n }\n\n var result = {\n type: \"Feature\",\n geometry: {\n type: type,\n coordinates: coords\n },\n properties: this.properties\n };\n\n if ('id' in this) {\n result.id = this.id;\n }\n\n return result;\n};\n\n// classifies an array of rings into polygons with outer rings and holes\n\nfunction classifyRings(rings) {\n var len = rings.length;\n\n if (len <= 1) return [rings];\n\n var polygons = [],\n polygon,\n ccw;\n\n for (var i = 0; i < len; i++) {\n var area = signedArea(rings[i]);\n if (area === 0) continue;\n\n if (ccw === undefined) ccw = area < 0;\n\n if (ccw === area < 0) {\n if (polygon) polygons.push(polygon);\n polygon = [rings[i]];\n\n } else {\n polygon.push(rings[i]);\n }\n }\n if (polygon) polygons.push(polygon);\n\n return polygons;\n}\n\nfunction signedArea(ring) {\n var sum = 0;\n for (var i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) {\n p1 = ring[i];\n p2 = ring[j];\n sum += (p2.x - p1.x) * (p1.y + p2.y);\n }\n return sum;\n}\n","'use strict';\n\nvar VectorTileFeature = require('./vectortilefeature.js');\n\nmodule.exports = VectorTileLayer;\n\nfunction VectorTileLayer(pbf, end) {\n // Public\n this.version = 1;\n this.name = null;\n this.extent = 4096;\n this.length = 0;\n\n // Private\n this._pbf = pbf;\n this._keys = [];\n this._values = [];\n this._features = [];\n\n pbf.readFields(readLayer, this, end);\n\n this.length = this._features.length;\n}\n\nfunction readLayer(tag, layer, pbf) {\n if (tag === 15) layer.version = pbf.readVarint();\n else if (tag === 1) layer.name = pbf.readString();\n else if (tag === 5) layer.extent = pbf.readVarint();\n else if (tag === 2) layer._features.push(pbf.pos);\n else if (tag === 3) layer._keys.push(pbf.readString());\n else if (tag === 4) layer._values.push(readValueMessage(pbf));\n}\n\nfunction readValueMessage(pbf) {\n var value = null,\n end = pbf.readVarint() + pbf.pos;\n\n while (pbf.pos < end) {\n var tag = pbf.readVarint() >> 3;\n\n value = tag === 1 ? pbf.readString() :\n tag === 2 ? pbf.readFloat() :\n tag === 3 ? pbf.readDouble() :\n tag === 4 ? pbf.readVarint64() :\n tag === 5 ? pbf.readVarint() :\n tag === 6 ? pbf.readSVarint() :\n tag === 7 ? pbf.readBoolean() : null;\n }\n\n return value;\n}\n\n// return feature `i` from this layer as a `VectorTileFeature`\nVectorTileLayer.prototype.feature = function(i) {\n if (i < 0 || i >= this._features.length) throw new Error('feature index out of bounds');\n\n this._pbf.pos = this._features[i];\n\n var end = this._pbf.readVarint() + this._pbf.pos;\n return new VectorTileFeature(this._pbf, end, this.extent, this._keys, this._values);\n};\n","'use strict';\n\nvar VectorTileLayer = require('./vectortilelayer');\n\nmodule.exports = VectorTile;\n\nfunction VectorTile(pbf, end) {\n this.layers = pbf.readFields(readTile, {}, end);\n}\n\nfunction readTile(tag, layers, pbf) {\n if (tag === 3) {\n var layer = new VectorTileLayer(pbf, pbf.readVarint() + pbf.pos);\n if (layer.length) layers[layer.name] = layer;\n }\n}\n\n","module.exports.VectorTile = require('./lib/vectortile.js');\nmodule.exports.VectorTileFeature = require('./lib/vectortilefeature.js');\nmodule.exports.VectorTileLayer = require('./lib/vectortilelayer.js');\n","import {FillExtrusionLayoutArray, PosArray} from '../array_types.g';\n\nimport {members as layoutAttributes, centroidAttributes} from './fill_extrusion_attributes';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray} from '../index_array_type';\nimport {EXTENT} from '../extent';\nimport earcut from 'earcut';\nimport mvt from '@mapbox/vector-tile';\nconst vectorTileFeatureTypes = mvt.VectorTileFeature.types;\nimport {classifyRings} from '../../util/classify_rings';\nconst EARCUT_MAX_RINGS = 500;\nimport {register} from '../../util/web_worker_transfer';\nimport {hasPattern, addPatternDependencies} from './pattern_bucket_features';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\n\nimport type {FillExtrusionStyleLayer} from '../../style/style_layer/fill_extrusion_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nconst FACTOR = Math.pow(2, 13);\n\nfunction addVertex(vertexArray, x, y, nx, ny, nz, t, e) {\n vertexArray.emplaceBack(\n // a_pos\n x,\n y,\n // a_normal_ed: 3-component normal and 1-component edgedistance\n Math.floor(nx * FACTOR) * 2 + t,\n ny * FACTOR * 2,\n nz * FACTOR * 2,\n // edgedistance (used for wrapping patterns around extrusion sides)\n Math.round(e)\n );\n}\n\nexport class FillExtrusionBucket implements Bucket {\n index: number;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n\n layoutVertexArray: FillExtrusionLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n centroidVertexArray: PosArray;\n centroidVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n uploaded: boolean;\n features: Array;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n\n this.layoutVertexArray = new FillExtrusionLayoutArray();\n this.centroidVertexArray = new PosArray();\n this.indexArray = new TriangleIndexArray();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.segments = new SegmentVector();\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n this.features = [];\n this.hasPattern = hasPattern('fill-extrusion', this.layers, options);\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const bucketFeature: BucketFeature = {\n id,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n properties: feature.properties,\n type: feature.type,\n patterns: {}\n };\n\n if (this.hasPattern) {\n this.features.push(addPatternDependencies('fill-extrusion', this.layers, bucketFeature, this.zoom, options));\n } else {\n this.addFeature(bucketFeature, bucketFeature.geometry, index, canonical, {});\n }\n\n options.featureIndex.insert(feature, bucketFeature.geometry, index, sourceLayerIndex, this.index, true);\n }\n }\n\n addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n for (const feature of this.features) {\n const {geometry} = feature;\n this.addFeature(feature, geometry, feature.index, canonical, imagePositions);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0 && this.centroidVertexArray.length === 0;\n }\n\n uploadPending() {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.centroidVertexBuffer = context.createVertexBuffer(this.centroidVertexArray, centroidAttributes.members, true);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n this.centroidVertexBuffer.destroy();\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n const centroid = {x: 0, y: 0, vertexCount: 0};\n for (const polygon of classifyRings(geometry, EARCUT_MAX_RINGS)) {\n let numVertices = 0;\n for (const ring of polygon) {\n numVertices += ring.length;\n }\n let segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray);\n\n for (const ring of polygon) {\n if (ring.length === 0) {\n continue;\n }\n\n if (isEntirelyOutside(ring)) {\n continue;\n }\n\n let edgeDistance = 0;\n\n for (let p = 0; p < ring.length; p++) {\n const p1 = ring[p];\n\n if (p >= 1) {\n const p2 = ring[p - 1];\n\n if (!isBoundaryEdge(p1, p2)) {\n if (segment.vertexLength + 4 > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) {\n segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray);\n }\n\n const perp = p1.sub(p2)._perp()._unit();\n const dist = p2.dist(p1);\n if (edgeDistance + dist > 32768) edgeDistance = 0;\n\n addVertex(this.layoutVertexArray, p1.x, p1.y, perp.x, perp.y, 0, 0, edgeDistance);\n addVertex(this.layoutVertexArray, p1.x, p1.y, perp.x, perp.y, 0, 1, edgeDistance);\n centroid.x += 2 * p1.x;\n centroid.y += 2 * p1.y;\n centroid.vertexCount += 2;\n\n edgeDistance += dist;\n\n addVertex(this.layoutVertexArray, p2.x, p2.y, perp.x, perp.y, 0, 0, edgeDistance);\n addVertex(this.layoutVertexArray, p2.x, p2.y, perp.x, perp.y, 0, 1, edgeDistance);\n centroid.x += 2 * p2.x;\n centroid.y += 2 * p2.y;\n centroid.vertexCount += 2;\n\n const bottomRight = segment.vertexLength;\n\n // ┌──────┐\n // │ 0 1 │ Counter-clockwise winding order.\n // │ │ Triangle 1: 0 => 2 => 1\n // │ 2 3 │ Triangle 2: 1 => 2 => 3\n // └──────┘\n this.indexArray.emplaceBack(bottomRight, bottomRight + 2, bottomRight + 1);\n this.indexArray.emplaceBack(bottomRight + 1, bottomRight + 2, bottomRight + 3);\n\n segment.vertexLength += 4;\n segment.primitiveLength += 2;\n }\n }\n }\n\n }\n\n if (segment.vertexLength + numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) {\n segment = this.segments.prepareSegment(numVertices, this.layoutVertexArray, this.indexArray);\n }\n\n //Only triangulate and draw the area of the feature if it is a polygon\n //Other feature types (e.g. LineString) do not have area, so triangulation is pointless / undefined\n if (vectorTileFeatureTypes[feature.type] !== 'Polygon')\n continue;\n\n const flattened = [];\n const holeIndices = [];\n const triangleIndex = segment.vertexLength;\n\n for (const ring of polygon) {\n if (ring.length === 0) {\n continue;\n }\n\n if (ring !== polygon[0]) {\n holeIndices.push(flattened.length / 2);\n }\n\n for (let i = 0; i < ring.length; i++) {\n const p = ring[i];\n\n addVertex(this.layoutVertexArray, p.x, p.y, 0, 0, 1, 1, 0);\n centroid.x += p.x;\n centroid.y += p.y;\n centroid.vertexCount += 1;\n\n flattened.push(p.x);\n flattened.push(p.y);\n }\n\n }\n\n const indices = earcut(flattened, holeIndices);\n\n for (let j = 0; j < indices.length; j += 3) {\n // Counter-clockwise winding order.\n this.indexArray.emplaceBack(\n triangleIndex + indices[j],\n triangleIndex + indices[j + 2],\n triangleIndex + indices[j + 1]);\n }\n\n segment.primitiveLength += indices.length / 3;\n segment.vertexLength += numVertices;\n }\n\n // remember polygon centroid to calculate elevation in GPU\n for (let i = 0; i < centroid.vertexCount; i++) {\n this.centroidVertexArray.emplaceBack(\n Math.floor(centroid.x / centroid.vertexCount),\n Math.floor(centroid.y / centroid.vertexCount)\n );\n }\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical);\n }\n}\n\nregister('FillExtrusionBucket', FillExtrusionBucket, {omit: ['layers', 'features']});\n\nfunction isBoundaryEdge(p1, p2) {\n return (p1.x === p2.x && (p1.x < 0 || p1.x > EXTENT)) ||\n (p1.y === p2.y && (p1.y < 0 || p1.y > EXTENT));\n}\n\nfunction isEntirelyOutside(ring) {\n return ring.every(p => p.x < 0) ||\n ring.every(p => p.x > EXTENT) ||\n ring.every(p => p.y < 0) ||\n ring.every(p => p.y > EXTENT);\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type FillExtrusionPaintProps = {\n \"fill-extrusion-opacity\": DataConstantProperty,\n \"fill-extrusion-color\": DataDrivenProperty,\n \"fill-extrusion-translate\": DataConstantProperty<[number, number]>,\n \"fill-extrusion-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"fill-extrusion-pattern\": CrossFadedDataDrivenProperty,\n \"fill-extrusion-height\": DataDrivenProperty,\n \"fill-extrusion-base\": DataDrivenProperty,\n \"fill-extrusion-vertical-gradient\": DataConstantProperty,\n};\n\nexport type FillExtrusionPaintPropsPossiblyEvaluated = {\n \"fill-extrusion-opacity\": number,\n \"fill-extrusion-color\": PossiblyEvaluatedPropertyValue,\n \"fill-extrusion-translate\": [number, number],\n \"fill-extrusion-translate-anchor\": \"map\" | \"viewport\",\n \"fill-extrusion-pattern\": PossiblyEvaluatedPropertyValue>,\n \"fill-extrusion-height\": PossiblyEvaluatedPropertyValue,\n \"fill-extrusion-base\": PossiblyEvaluatedPropertyValue,\n \"fill-extrusion-vertical-gradient\": boolean,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"fill-extrusion-opacity\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-opacity\"] as any as StylePropertySpecification),\n \"fill-extrusion-color\": new DataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-color\"] as any as StylePropertySpecification),\n \"fill-extrusion-translate\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-translate\"] as any as StylePropertySpecification),\n \"fill-extrusion-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-translate-anchor\"] as any as StylePropertySpecification),\n \"fill-extrusion-pattern\": new CrossFadedDataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-pattern\"] as any as StylePropertySpecification),\n \"fill-extrusion-height\": new DataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-height\"] as any as StylePropertySpecification),\n \"fill-extrusion-base\": new DataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-base\"] as any as StylePropertySpecification),\n \"fill-extrusion-vertical-gradient\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-vertical-gradient\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport {FillExtrusionBucket} from '../../data/bucket/fill_extrusion_bucket';\nimport {polygonIntersectsPolygon, polygonIntersectsMultiPolygon} from '../../util/intersection_tests';\nimport {translateDistance, translate} from '../query_utils';\nimport properties, {FillExtrusionPaintPropsPossiblyEvaluated} from './fill_extrusion_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\nimport {mat4, vec4} from 'gl-matrix';\nimport Point from '@mapbox/point-geometry';\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {BucketParameters} from '../../data/bucket';\nimport type {FillExtrusionPaintProps} from './fill_extrusion_style_layer_properties.g';\nimport type {Transform} from '../../geo/transform';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nexport class Point3D extends Point {\n z: number;\n}\n\nexport class FillExtrusionStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n createBucket(parameters: BucketParameters) {\n return new FillExtrusionBucket(parameters);\n }\n\n queryRadius(): number {\n return translateDistance(this.paint.get('fill-extrusion-translate'));\n }\n\n is3D(): boolean {\n return true;\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number,\n pixelPosMatrix: mat4\n ): boolean | number {\n\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('fill-extrusion-translate'),\n this.paint.get('fill-extrusion-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n\n const height = this.paint.get('fill-extrusion-height').evaluate(feature, featureState);\n const base = this.paint.get('fill-extrusion-base').evaluate(feature, featureState);\n\n const projectedQueryGeometry = projectQueryGeometry(translatedPolygon, pixelPosMatrix, transform, 0);\n\n const projected = projectExtrusion(geometry, base, height, pixelPosMatrix);\n const projectedBase = projected[0];\n const projectedTop = projected[1];\n return checkIntersection(projectedBase, projectedTop, projectedQueryGeometry);\n }\n}\n\nfunction dot(a, b) {\n return a.x * b.x + a.y * b.y;\n}\n\nexport function getIntersectionDistance(projectedQueryGeometry: Array, projectedFace: Array) {\n\n if (projectedQueryGeometry.length === 1) {\n // For point queries calculate the z at which the point intersects the face\n // using barycentric coordinates.\n\n // Find the barycentric coordinates of the projected point within the first\n // triangle of the face, using only the xy plane. It doesn't matter if the\n // point is outside the first triangle because all the triangles in the face\n // are in the same plane.\n //\n // Check whether points are coincident and use other points if they are.\n let i = 0;\n const a = projectedFace[i++];\n let b;\n while (!b || a.equals(b)) {\n b = projectedFace[i++];\n if (!b) return Infinity;\n }\n\n // Loop until point `c` is not colinear with points `a` and `b`.\n for (; i < projectedFace.length; i++) {\n const c = projectedFace[i];\n\n const p = projectedQueryGeometry[0];\n\n const ab = b.sub(a);\n const ac = c.sub(a);\n const ap = p.sub(a);\n\n const dotABAB = dot(ab, ab);\n const dotABAC = dot(ab, ac);\n const dotACAC = dot(ac, ac);\n const dotAPAB = dot(ap, ab);\n const dotAPAC = dot(ap, ac);\n const denom = dotABAB * dotACAC - dotABAC * dotABAC;\n\n const v = (dotACAC * dotAPAB - dotABAC * dotAPAC) / denom;\n const w = (dotABAB * dotAPAC - dotABAC * dotAPAB) / denom;\n const u = 1 - v - w;\n\n // Use the barycentric weighting along with the original triangle z coordinates to get the point of intersection.\n const distance = a.z * u + b.z * v + c.z * w;\n\n if (isFinite(distance)) return distance;\n }\n\n return Infinity;\n\n } else {\n // The counts as closest is less clear when the query is a box. This\n // returns the distance to the nearest point on the face, whether it is\n // within the query or not. It could be more correct to return the\n // distance to the closest point within the query box but this would be\n // more complicated and expensive to calculate with little benefit.\n let closestDistance = Infinity;\n for (const p of projectedFace) {\n closestDistance = Math.min(closestDistance, p.z);\n }\n return closestDistance;\n }\n}\n\nfunction checkIntersection(projectedBase: Array>, projectedTop: Array>, projectedQueryGeometry: Array) {\n let closestDistance = Infinity;\n\n if (polygonIntersectsMultiPolygon(projectedQueryGeometry, projectedTop)) {\n closestDistance = getIntersectionDistance(projectedQueryGeometry, projectedTop[0]);\n }\n\n for (let r = 0; r < projectedTop.length; r++) {\n const ringTop = projectedTop[r];\n const ringBase = projectedBase[r];\n for (let p = 0; p < ringTop.length - 1; p++) {\n const topA = ringTop[p];\n const topB = ringTop[p + 1];\n const baseA = ringBase[p];\n const baseB = ringBase[p + 1];\n const face = [topA, topB, baseB, baseA, topA];\n if (polygonIntersectsPolygon(projectedQueryGeometry, face)) {\n closestDistance = Math.min(closestDistance, getIntersectionDistance(projectedQueryGeometry, face));\n }\n }\n }\n\n return closestDistance === Infinity ? false : closestDistance;\n}\n\n/*\n * Project the geometry using matrix `m`. This is essentially doing\n * `vec4.transformMat4([], [p.x, p.y, z, 1], m)` but the multiplication\n * is inlined so that parts of the projection that are the same across\n * different points can only be done once. This produced a measurable\n * performance improvement.\n */\nfunction projectExtrusion(geometry: Array>, zBase: number, zTop: number, m: mat4): [Array>, Array>] {\n const projectedBase = [] as Array>;\n const projectedTop = [] as Array>;\n const baseXZ = m[8] * zBase;\n const baseYZ = m[9] * zBase;\n const baseZZ = m[10] * zBase;\n const baseWZ = m[11] * zBase;\n const topXZ = m[8] * zTop;\n const topYZ = m[9] * zTop;\n const topZZ = m[10] * zTop;\n const topWZ = m[11] * zTop;\n\n for (const r of geometry) {\n const ringBase = [] as Array;\n const ringTop = [] as Array;\n for (const p of r) {\n const x = p.x;\n const y = p.y;\n\n const sX = m[0] * x + m[4] * y + m[12];\n const sY = m[1] * x + m[5] * y + m[13];\n const sZ = m[2] * x + m[6] * y + m[14];\n const sW = m[3] * x + m[7] * y + m[15];\n\n const baseX = sX + baseXZ;\n const baseY = sY + baseYZ;\n const baseZ = sZ + baseZZ;\n const baseW = sW + baseWZ;\n\n const topX = sX + topXZ;\n const topY = sY + topYZ;\n const topZ = sZ + topZZ;\n const topW = sW + topWZ;\n\n const b = new Point(baseX / baseW, baseY / baseW) as Point3D;\n b.z = baseZ / baseW;\n ringBase.push(b);\n\n const t = new Point(topX / topW, topY / topW) as Point3D;\n t.z = topZ / topW;\n ringTop.push(t);\n }\n projectedBase.push(ringBase);\n projectedTop.push(ringTop);\n }\n return [projectedBase, projectedTop];\n}\n\nfunction projectQueryGeometry(queryGeometry: Array, pixelPosMatrix: mat4, transform: Transform, z: number) {\n const projectedQueryGeometry = [];\n for (const p of queryGeometry) {\n const v = [p.x, p.y, z, 1] as vec4;\n vec4.transformMat4(v, v, pixelPosMatrix);\n projectedQueryGeometry.push(new Point(v[0] / v[3], v[1] / v[3]));\n }\n return projectedQueryGeometry;\n}\n","import {createLayout} from '../../util/struct_array';\n\nexport const lineLayoutAttributes = createLayout([\n {name: 'a_pos_normal', components: 2, type: 'Int16'},\n {name: 'a_data', components: 4, type: 'Uint8'}\n], 4);\n\nexport const {members, size, alignment} = lineLayoutAttributes;\n","import {createLayout} from '../../util/struct_array';\n\nexport const lineLayoutAttributesExt = createLayout([\n {name: 'a_uv_x', components: 1, type: 'Float32'},\n {name: 'a_split_index', components: 1, type: 'Float32'},\n]);\n\nexport const {members, size, alignment} = lineLayoutAttributesExt;\n","import {LineLayoutArray, LineExtLayoutArray} from '../array_types.g';\n\nimport {members as layoutAttributes} from './line_attributes';\nimport {members as layoutAttributesExt} from './line_attributes_ext';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray} from '../index_array_type';\nimport {EXTENT} from '../extent';\nimport mvt from '@mapbox/vector-tile';\nconst vectorTileFeatureTypes = mvt.VectorTileFeature.types;\nimport {register} from '../../util/web_worker_transfer';\nimport {hasPattern, addPatternDependencies} from './pattern_bucket_features';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {LineStyleLayer} from '../../style/style_layer/line_style_layer';\nimport type Point from '@mapbox/point-geometry';\nimport type {Segment} from '../segment';\nimport {RGBAImage} from '../../util/image';\nimport type {Context} from '../../gl/context';\nimport type {Texture} from '../../render/texture';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\n// NOTE ON EXTRUDE SCALE:\n// scale the extrusion vector so that the normal length is this value.\n// contains the \"texture\" normals (-1..1). this is distinct from the extrude\n// normals for line joins, because the x-value remains 0 for the texture\n// normal array, while the extrude normal actually moves the vertex to create\n// the acute/bevelled line join.\nconst EXTRUDE_SCALE = 63;\n\n/*\n * Sharp corners cause dashed lines to tilt because the distance along the line\n * is the same at both the inner and outer corners. To improve the appearance of\n * dashed lines we add extra points near sharp corners so that a smaller part\n * of the line is tilted.\n *\n * COS_HALF_SHARP_CORNER controls how sharp a corner has to be for us to add an\n * extra vertex. The default is 75 degrees.\n *\n * The newly created vertices are placed SHARP_CORNER_OFFSET pixels from the corner.\n */\nconst COS_HALF_SHARP_CORNER = Math.cos(75 / 2 * (Math.PI / 180));\nconst SHARP_CORNER_OFFSET = 15;\n\n// Angle per triangle for approximating round line joins.\nconst DEG_PER_TRIANGLE = 20;\n\n// The number of bits that is used to store the line distance in the buffer.\nconst LINE_DISTANCE_BUFFER_BITS = 15;\n\n// We don't have enough bits for the line distance as we'd like to have, so\n// use this value to scale the line distance (in tile units) down to a smaller\n// value. This lets us store longer distances while sacrificing precision.\nconst LINE_DISTANCE_SCALE = 1 / 2;\n\n// The maximum line distance, in tile units, that fits in the buffer.\nconst MAX_LINE_DISTANCE = Math.pow(2, LINE_DISTANCE_BUFFER_BITS - 1) / LINE_DISTANCE_SCALE;\n\ntype LineClips = {\n start: number;\n end: number;\n};\n\ntype GradientTexture = {\n texture?: Texture;\n gradient?: RGBAImage;\n version?: number;\n};\n\n/**\n * @internal\n * Line bucket class\n */\nexport class LineBucket implements Bucket {\n distance: number;\n totalDistance: number;\n maxLineLength: number;\n scaledDistance: number;\n lineClips?: LineClips;\n\n e1: number;\n e2: number;\n\n index: number;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n gradients: {[x: string]: GradientTexture};\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n patternFeatures: Array;\n lineClipsArray: Array;\n\n layoutVertexArray: LineLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n layoutVertexArray2: LineExtLayoutArray;\n layoutVertexBuffer2: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n uploaded: boolean;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n this.patternFeatures = [];\n this.lineClipsArray = [];\n this.gradients = {};\n this.layers.forEach(layer => {\n this.gradients[layer.id] = {};\n });\n\n this.layoutVertexArray = new LineLayoutArray();\n this.layoutVertexArray2 = new LineExtLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.segments = new SegmentVector();\n this.maxLineLength = 0;\n\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n this.hasPattern = hasPattern('line', this.layers, options);\n const lineSortKey = this.layers[0].layout.get('line-sort-key');\n const sortFeaturesByKey = !lineSortKey.isConstant();\n const bucketFeatures: BucketFeature[] = [];\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const sortKey = sortFeaturesByKey ?\n lineSortKey.evaluate(evaluationFeature, {}, canonical) :\n undefined;\n\n const bucketFeature: BucketFeature = {\n id,\n properties: feature.properties,\n type: feature.type,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n patterns: {},\n sortKey\n };\n\n bucketFeatures.push(bucketFeature);\n }\n\n if (sortFeaturesByKey) {\n bucketFeatures.sort((a, b) => {\n return (a.sortKey) - (b.sortKey);\n });\n }\n\n for (const bucketFeature of bucketFeatures) {\n const {geometry, index, sourceLayerIndex} = bucketFeature;\n\n if (this.hasPattern) {\n const patternBucketFeature = addPatternDependencies('line', this.layers, bucketFeature, this.zoom, options);\n // pattern features are added only once the pattern is loaded into the image atlas\n // so are stored during populate until later updated with positions by tile worker in addFeatures\n this.patternFeatures.push(patternBucketFeature);\n } else {\n this.addFeature(bucketFeature, geometry, index, canonical, {});\n }\n\n const feature = features[index].feature;\n options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n for (const feature of this.patternFeatures) {\n this.addFeature(feature, feature.geometry, feature.index, canonical, imagePositions);\n }\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0;\n }\n\n uploadPending() {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded) {\n if (this.layoutVertexArray2.length !== 0) {\n this.layoutVertexBuffer2 = context.createVertexBuffer(this.layoutVertexArray2, layoutAttributesExt);\n }\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n }\n\n lineFeatureClips(feature: BucketFeature): LineClips | undefined {\n if (!!feature.properties && Object.prototype.hasOwnProperty.call(feature.properties, 'mapbox_clip_start') && Object.prototype.hasOwnProperty.call(feature.properties, 'mapbox_clip_end')) {\n const start = +feature.properties['mapbox_clip_start'];\n const end = +feature.properties['mapbox_clip_end'];\n return {start, end};\n }\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n const layout = this.layers[0].layout;\n const join = layout.get('line-join').evaluate(feature, {});\n const cap = layout.get('line-cap');\n const miterLimit = layout.get('line-miter-limit');\n const roundLimit = layout.get('line-round-limit');\n this.lineClips = this.lineFeatureClips(feature);\n\n for (const line of geometry) {\n this.addLine(line, feature, join, cap, miterLimit, roundLimit);\n }\n\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical);\n }\n\n addLine(vertices: Array, feature: BucketFeature, join: string, cap: string, miterLimit: number, roundLimit: number) {\n this.distance = 0;\n this.scaledDistance = 0;\n this.totalDistance = 0;\n\n if (this.lineClips) {\n this.lineClipsArray.push(this.lineClips);\n // Calculate the total distance, in tile units, of this tiled line feature\n for (let i = 0; i < vertices.length - 1; i++) {\n this.totalDistance += vertices[i].dist(vertices[i + 1]);\n }\n this.updateScaledDistance();\n this.maxLineLength = Math.max(this.maxLineLength, this.totalDistance);\n }\n\n const isPolygon = vectorTileFeatureTypes[feature.type] === 'Polygon';\n\n // If the line has duplicate vertices at the ends, adjust start/length to remove them.\n let len = vertices.length;\n while (len >= 2 && vertices[len - 1].equals(vertices[len - 2])) {\n len--;\n }\n let first = 0;\n while (first < len - 1 && vertices[first].equals(vertices[first + 1])) {\n first++;\n }\n\n // Ignore invalid geometry.\n if (len < (isPolygon ? 3 : 2)) return;\n\n if (join === 'bevel') miterLimit = 1.05;\n\n const sharpCornerOffset = this.overscaling <= 16 ?\n SHARP_CORNER_OFFSET * EXTENT / (512 * this.overscaling) :\n 0;\n\n // we could be more precise, but it would only save a negligible amount of space\n const segment = this.segments.prepareSegment(len * 10, this.layoutVertexArray, this.indexArray);\n\n let currentVertex: Point;\n let prevVertex: Point;\n let nextVertex: Point;\n let prevNormal: Point;\n let nextNormal: Point;\n\n // the last two vertices added\n this.e1 = this.e2 = -1;\n\n if (isPolygon) {\n currentVertex = vertices[len - 2];\n nextNormal = vertices[first].sub(currentVertex)._unit()._perp();\n }\n\n for (let i = first; i < len; i++) {\n\n nextVertex = i === len - 1 ?\n (isPolygon ? vertices[first + 1] : undefined) : // if it's a polygon, treat the last vertex like the first\n vertices[i + 1]; // just the next vertex\n\n // if two consecutive vertices exist, skip the current one\n if (nextVertex && vertices[i].equals(nextVertex)) continue;\n\n if (nextNormal) prevNormal = nextNormal;\n if (currentVertex) prevVertex = currentVertex;\n\n currentVertex = vertices[i];\n\n // Calculate the normal towards the next vertex in this line. In case\n // there is no next vertex, pretend that the line is continuing straight,\n // meaning that we are just using the previous normal.\n nextNormal = nextVertex ? nextVertex.sub(currentVertex)._unit()._perp() : prevNormal;\n\n // If we still don't have a previous normal, this is the beginning of a\n // non-closed line, so we're doing a straight \"join\".\n prevNormal = prevNormal || nextNormal;\n\n // Determine the normal of the join extrusion. It is the angle bisector\n // of the segments between the previous line and the next line.\n // In the case of 180° angles, the prev and next normals cancel each other out:\n // prevNormal + nextNormal = (0, 0), its magnitude is 0, so the unit vector would be\n // undefined. In that case, we're keeping the joinNormal at (0, 0), so that the cosHalfAngle\n // below will also become 0 and miterLength will become Infinity.\n let joinNormal = prevNormal.add(nextNormal);\n if (joinNormal.x !== 0 || joinNormal.y !== 0) {\n joinNormal._unit();\n }\n /* joinNormal prevNormal\n * ↖ ↑\n * .________. prevVertex\n * |\n * nextNormal ← | currentVertex\n * |\n * nextVertex !\n *\n */\n\n // calculate cosines of the angle (and its half) using dot product\n const cosAngle = prevNormal.x * nextNormal.x + prevNormal.y * nextNormal.y;\n const cosHalfAngle = joinNormal.x * nextNormal.x + joinNormal.y * nextNormal.y;\n\n // Calculate the length of the miter (the ratio of the miter to the width)\n // as the inverse of cosine of the angle between next and join normals\n const miterLength = cosHalfAngle !== 0 ? 1 / cosHalfAngle : Infinity;\n\n // approximate angle from cosine\n const approxAngle = 2 * Math.sqrt(2 - 2 * cosHalfAngle);\n\n const isSharpCorner = cosHalfAngle < COS_HALF_SHARP_CORNER && prevVertex && nextVertex;\n const lineTurnsLeft = prevNormal.x * nextNormal.y - prevNormal.y * nextNormal.x > 0;\n\n if (isSharpCorner && i > first) {\n const prevSegmentLength = currentVertex.dist(prevVertex);\n if (prevSegmentLength > 2 * sharpCornerOffset) {\n const newPrevVertex = currentVertex.sub(currentVertex.sub(prevVertex)._mult(sharpCornerOffset / prevSegmentLength)._round());\n this.updateDistance(prevVertex, newPrevVertex);\n this.addCurrentVertex(newPrevVertex, prevNormal, 0, 0, segment);\n prevVertex = newPrevVertex;\n }\n }\n\n // The join if a middle vertex, otherwise the cap.\n const middleVertex = prevVertex && nextVertex;\n let currentJoin = middleVertex ? join : isPolygon ? 'butt' : cap;\n\n if (middleVertex && currentJoin === 'round') {\n if (miterLength < roundLimit) {\n currentJoin = 'miter';\n } else if (miterLength <= 2) {\n currentJoin = 'fakeround';\n }\n }\n\n if (currentJoin === 'miter' && miterLength > miterLimit) {\n currentJoin = 'bevel';\n }\n\n if (currentJoin === 'bevel') {\n // The maximum extrude length is 128 / 63 = 2 times the width of the line\n // so if miterLength >= 2 we need to draw a different type of bevel here.\n if (miterLength > 2) currentJoin = 'flipbevel';\n\n // If the miterLength is really small and the line bevel wouldn't be visible,\n // just draw a miter join to save a triangle.\n if (miterLength < miterLimit) currentJoin = 'miter';\n }\n\n // Calculate how far along the line the currentVertex is\n if (prevVertex) this.updateDistance(prevVertex, currentVertex);\n\n if (currentJoin === 'miter') {\n\n joinNormal._mult(miterLength);\n this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment);\n\n } else if (currentJoin === 'flipbevel') {\n // miter is too big, flip the direction to make a beveled join\n\n if (miterLength > 100) {\n // Almost parallel lines\n joinNormal = nextNormal.mult(-1);\n\n } else {\n const bevelLength = miterLength * prevNormal.add(nextNormal).mag() / prevNormal.sub(nextNormal).mag();\n joinNormal._perp()._mult(bevelLength * (lineTurnsLeft ? -1 : 1));\n }\n this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment);\n this.addCurrentVertex(currentVertex, joinNormal.mult(-1), 0, 0, segment);\n\n } else if (currentJoin === 'bevel' || currentJoin === 'fakeround') {\n const offset = -Math.sqrt(miterLength * miterLength - 1);\n const offsetA = lineTurnsLeft ? offset : 0;\n const offsetB = lineTurnsLeft ? 0 : offset;\n\n // Close previous segment with a bevel\n if (prevVertex) {\n this.addCurrentVertex(currentVertex, prevNormal, offsetA, offsetB, segment);\n }\n\n if (currentJoin === 'fakeround') {\n // The join angle is sharp enough that a round join would be visible.\n // Bevel joins fill the gap between segments with a single pie slice triangle.\n // Create a round join by adding multiple pie slices. The join isn't actually round, but\n // it looks like it is at the sizes we render lines at.\n\n // pick the number of triangles for approximating round join by based on the angle between normals\n const n = Math.round((approxAngle * 180 / Math.PI) / DEG_PER_TRIANGLE);\n\n for (let m = 1; m < n; m++) {\n let t = m / n;\n if (t !== 0.5) {\n // approximate spherical interpolation https://observablehq.com/@mourner/approximating-geometric-slerp\n const t2 = t - 0.5;\n const A = 1.0904 + cosAngle * (-3.2452 + cosAngle * (3.55645 - cosAngle * 1.43519));\n const B = 0.848013 + cosAngle * (-1.06021 + cosAngle * 0.215638);\n t = t + t * t2 * (t - 1) * (A * t2 * t2 + B);\n }\n const extrude = nextNormal.sub(prevNormal)._mult(t)._add(prevNormal)._unit()._mult(lineTurnsLeft ? -1 : 1);\n this.addHalfVertex(currentVertex, extrude.x, extrude.y, false, lineTurnsLeft, 0, segment);\n }\n }\n\n if (nextVertex) {\n // Start next segment\n this.addCurrentVertex(currentVertex, nextNormal, -offsetA, -offsetB, segment);\n }\n\n } else if (currentJoin === 'butt') {\n this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment); // butt cap\n\n } else if (currentJoin === 'square') {\n const offset = prevVertex ? 1 : -1; // closing or starting square cap\n this.addCurrentVertex(currentVertex, joinNormal, offset, offset, segment);\n\n } else if (currentJoin === 'round') {\n\n if (prevVertex) {\n // Close previous segment with butt\n this.addCurrentVertex(currentVertex, prevNormal, 0, 0, segment);\n\n // Add round cap or linejoin at end of segment\n this.addCurrentVertex(currentVertex, prevNormal, 1, 1, segment, true);\n }\n if (nextVertex) {\n // Add round cap before first segment\n this.addCurrentVertex(currentVertex, nextNormal, -1, -1, segment, true);\n\n // Start next segment with a butt\n this.addCurrentVertex(currentVertex, nextNormal, 0, 0, segment);\n }\n }\n\n if (isSharpCorner && i < len - 1) {\n const nextSegmentLength = currentVertex.dist(nextVertex);\n if (nextSegmentLength > 2 * sharpCornerOffset) {\n const newCurrentVertex = currentVertex.add(nextVertex.sub(currentVertex)._mult(sharpCornerOffset / nextSegmentLength)._round());\n this.updateDistance(currentVertex, newCurrentVertex);\n this.addCurrentVertex(newCurrentVertex, nextNormal, 0, 0, segment);\n currentVertex = newCurrentVertex;\n }\n }\n }\n }\n\n /**\n * Add two vertices to the buffers.\n *\n * @param p - the line vertex to add buffer vertices for\n * @param normal - vertex normal\n * @param endLeft - extrude to shift the left vertex along the line\n * @param endRight - extrude to shift the left vertex along the line\n * @param segment - the segment object to add the vertex to\n * @param round - whether this is a round cap\n */\n addCurrentVertex(p: Point, normal: Point, endLeft: number, endRight: number, segment: Segment, round: boolean = false) {\n // left and right extrude vectors, perpendicularly shifted by endLeft/endRight\n const leftX = normal.x + normal.y * endLeft;\n const leftY = normal.y - normal.x * endLeft;\n const rightX = -normal.x + normal.y * endRight;\n const rightY = -normal.y - normal.x * endRight;\n\n this.addHalfVertex(p, leftX, leftY, round, false, endLeft, segment);\n this.addHalfVertex(p, rightX, rightY, round, true, -endRight, segment);\n\n // There is a maximum \"distance along the line\" that we can store in the buffers.\n // When we get close to the distance, reset it to zero and add the vertex again with\n // a distance of zero. The max distance is determined by the number of bits we allocate\n // to `linesofar`.\n if (this.distance > MAX_LINE_DISTANCE / 2 && this.totalDistance === 0) {\n this.distance = 0;\n this.updateScaledDistance();\n this.addCurrentVertex(p, normal, endLeft, endRight, segment, round);\n }\n }\n\n addHalfVertex({x, y}: Point, extrudeX: number, extrudeY: number, round: boolean, up: boolean, dir: number, segment: Segment) {\n const totalDistance = this.lineClips ? this.scaledDistance * (MAX_LINE_DISTANCE - 1) : this.scaledDistance;\n // scale down so that we can store longer distances while sacrificing precision.\n const linesofarScaled = totalDistance * LINE_DISTANCE_SCALE;\n\n this.layoutVertexArray.emplaceBack(\n // a_pos_normal\n // Encode round/up the least significant bits\n (x << 1) + (round ? 1 : 0),\n (y << 1) + (up ? 1 : 0),\n // a_data\n // add 128 to store a byte in an unsigned byte\n Math.round(EXTRUDE_SCALE * extrudeX) + 128,\n Math.round(EXTRUDE_SCALE * extrudeY) + 128,\n // Encode the -1/0/1 direction value into the first two bits of .z of a_data.\n // Combine it with the lower 6 bits of `linesofarScaled` (shifted by 2 bits to make\n // room for the direction value). The upper 8 bits of `linesofarScaled` are placed in\n // the `w` component.\n ((dir === 0 ? 0 : (dir < 0 ? -1 : 1)) + 1) | ((linesofarScaled & 0x3F) << 2),\n linesofarScaled >> 6);\n\n // Constructs a second vertex buffer with higher precision line progress\n if (this.lineClips) {\n const progressRealigned = this.scaledDistance - this.lineClips.start;\n const endClipRealigned = this.lineClips.end - this.lineClips.start;\n const uvX = progressRealigned / endClipRealigned;\n this.layoutVertexArray2.emplaceBack(uvX, this.lineClipsArray.length);\n }\n\n const e = segment.vertexLength++;\n if (this.e1 >= 0 && this.e2 >= 0) {\n this.indexArray.emplaceBack(this.e1, this.e2, e);\n segment.primitiveLength++;\n }\n if (up) {\n this.e2 = e;\n } else {\n this.e1 = e;\n }\n }\n\n updateScaledDistance() {\n // Knowing the ratio of the full linestring covered by this tiled feature, as well\n // as the total distance (in tile units) of this tiled feature, and the distance\n // (in tile units) of the current vertex, we can determine the relative distance\n // of this vertex along the full linestring feature and scale it to [0, 2^15)\n this.scaledDistance = this.lineClips ?\n this.lineClips.start + (this.lineClips.end - this.lineClips.start) * this.distance / this.totalDistance :\n this.distance;\n }\n\n updateDistance(prev: Point, next: Point) {\n this.distance += prev.dist(next);\n this.updateScaledDistance();\n }\n}\n\nregister('LineBucket', LineBucket, {omit: ['layers', 'patternFeatures']});\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type LineLayoutProps = {\n \"line-cap\": DataConstantProperty<\"butt\" | \"round\" | \"square\">,\n \"line-join\": DataDrivenProperty<\"bevel\" | \"round\" | \"miter\">,\n \"line-miter-limit\": DataConstantProperty,\n \"line-round-limit\": DataConstantProperty,\n \"line-sort-key\": DataDrivenProperty,\n};\n\nexport type LineLayoutPropsPossiblyEvaluated = {\n \"line-cap\": \"butt\" | \"round\" | \"square\",\n \"line-join\": PossiblyEvaluatedPropertyValue<\"bevel\" | \"round\" | \"miter\">,\n \"line-miter-limit\": number,\n \"line-round-limit\": number,\n \"line-sort-key\": PossiblyEvaluatedPropertyValue,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"line-cap\": new DataConstantProperty(styleSpec[\"layout_line\"][\"line-cap\"] as any as StylePropertySpecification),\n \"line-join\": new DataDrivenProperty(styleSpec[\"layout_line\"][\"line-join\"] as any as StylePropertySpecification),\n \"line-miter-limit\": new DataConstantProperty(styleSpec[\"layout_line\"][\"line-miter-limit\"] as any as StylePropertySpecification),\n \"line-round-limit\": new DataConstantProperty(styleSpec[\"layout_line\"][\"line-round-limit\"] as any as StylePropertySpecification),\n \"line-sort-key\": new DataDrivenProperty(styleSpec[\"layout_line\"][\"line-sort-key\"] as any as StylePropertySpecification),\n});\n\nexport type LinePaintProps = {\n \"line-opacity\": DataDrivenProperty,\n \"line-color\": DataDrivenProperty,\n \"line-translate\": DataConstantProperty<[number, number]>,\n \"line-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"line-width\": DataDrivenProperty,\n \"line-gap-width\": DataDrivenProperty,\n \"line-offset\": DataDrivenProperty,\n \"line-blur\": DataDrivenProperty,\n \"line-dasharray\": CrossFadedProperty>,\n \"line-pattern\": CrossFadedDataDrivenProperty,\n \"line-gradient\": ColorRampProperty,\n};\n\nexport type LinePaintPropsPossiblyEvaluated = {\n \"line-opacity\": PossiblyEvaluatedPropertyValue,\n \"line-color\": PossiblyEvaluatedPropertyValue,\n \"line-translate\": [number, number],\n \"line-translate-anchor\": \"map\" | \"viewport\",\n \"line-width\": PossiblyEvaluatedPropertyValue,\n \"line-gap-width\": PossiblyEvaluatedPropertyValue,\n \"line-offset\": PossiblyEvaluatedPropertyValue,\n \"line-blur\": PossiblyEvaluatedPropertyValue,\n \"line-dasharray\": CrossFaded>,\n \"line-pattern\": PossiblyEvaluatedPropertyValue>,\n \"line-gradient\": ColorRampProperty,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"line-opacity\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-opacity\"] as any as StylePropertySpecification),\n \"line-color\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-color\"] as any as StylePropertySpecification),\n \"line-translate\": new DataConstantProperty(styleSpec[\"paint_line\"][\"line-translate\"] as any as StylePropertySpecification),\n \"line-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_line\"][\"line-translate-anchor\"] as any as StylePropertySpecification),\n \"line-width\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-width\"] as any as StylePropertySpecification),\n \"line-gap-width\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-gap-width\"] as any as StylePropertySpecification),\n \"line-offset\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-offset\"] as any as StylePropertySpecification),\n \"line-blur\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-blur\"] as any as StylePropertySpecification),\n \"line-dasharray\": new CrossFadedProperty(styleSpec[\"paint_line\"][\"line-dasharray\"] as any as StylePropertySpecification),\n \"line-pattern\": new CrossFadedDataDrivenProperty(styleSpec[\"paint_line\"][\"line-pattern\"] as any as StylePropertySpecification),\n \"line-gradient\": new ColorRampProperty(styleSpec[\"paint_line\"][\"line-gradient\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import Point from '@mapbox/point-geometry';\n\nimport {StyleLayer} from '../style_layer';\nimport {LineBucket} from '../../data/bucket/line_bucket';\nimport {polygonIntersectsBufferedMultiLine} from '../../util/intersection_tests';\nimport {getMaximumPaintValue, translateDistance, translate, offsetLine} from '../query_utils';\nimport properties, {LineLayoutPropsPossiblyEvaluated, LinePaintPropsPossiblyEvaluated} from './line_style_layer_properties.g';\nimport {extend} from '../../util/util';\nimport {EvaluationParameters} from '../evaluation_parameters';\nimport {Transitionable, Transitioning, Layout, PossiblyEvaluated, DataDrivenProperty} from '../properties';\n\nimport {isZoomExpression, Step} from '@maplibre/maplibre-gl-style-spec';\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Bucket, BucketParameters} from '../../data/bucket';\nimport type {LineLayoutProps, LinePaintProps} from './line_style_layer_properties.g';\nimport type {Transform} from '../../geo/transform';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nexport class LineFloorwidthProperty extends DataDrivenProperty {\n useIntegerZoom: true;\n\n possiblyEvaluate(value, parameters) {\n parameters = new EvaluationParameters(Math.floor(parameters.zoom), {\n now: parameters.now,\n fadeDuration: parameters.fadeDuration,\n zoomHistory: parameters.zoomHistory,\n transition: parameters.transition\n });\n return super.possiblyEvaluate(value, parameters);\n }\n\n evaluate(value, globals, feature, featureState) {\n globals = extend({}, globals, {zoom: Math.floor(globals.zoom)});\n return super.evaluate(value, globals, feature, featureState);\n }\n}\n\nlet lineFloorwidthProperty: LineFloorwidthProperty;\n\nexport class LineStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n gradientVersion: number;\n stepInterpolant: boolean;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n this.gradientVersion = 0;\n if (!lineFloorwidthProperty) {\n lineFloorwidthProperty =\n new LineFloorwidthProperty(properties.paint.properties['line-width'].specification);\n lineFloorwidthProperty.useIntegerZoom = true;\n }\n }\n\n _handleSpecialPaintPropertyUpdate(name: string) {\n if (name === 'line-gradient') {\n const expression = this.gradientExpression();\n if (isZoomExpression(expression)) {\n this.stepInterpolant = expression._styleExpression.expression instanceof Step;\n } else {\n this.stepInterpolant = false;\n }\n this.gradientVersion = (this.gradientVersion + 1) % Number.MAX_SAFE_INTEGER;\n }\n }\n\n gradientExpression() {\n return this._transitionablePaint._values['line-gradient'].value.expression;\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n super.recalculate(parameters, availableImages);\n (this.paint._values as any)['line-floorwidth'] =\n lineFloorwidthProperty.possiblyEvaluate(this._transitioningPaint._values['line-width'].value, parameters);\n }\n\n createBucket(parameters: BucketParameters) {\n return new LineBucket(parameters);\n }\n\n queryRadius(bucket: Bucket): number {\n const lineBucket: LineBucket = (bucket as any);\n const width = getLineWidth(\n getMaximumPaintValue('line-width', this, lineBucket),\n getMaximumPaintValue('line-gap-width', this, lineBucket));\n const offset = getMaximumPaintValue('line-offset', this, lineBucket);\n return width / 2 + Math.abs(offset) + translateDistance(this.paint.get('line-translate'));\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number\n ): boolean {\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('line-translate'),\n this.paint.get('line-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n const halfWidth = pixelsToTileUnits / 2 * getLineWidth(\n this.paint.get('line-width').evaluate(feature, featureState),\n this.paint.get('line-gap-width').evaluate(feature, featureState));\n const lineOffset = this.paint.get('line-offset').evaluate(feature, featureState);\n if (lineOffset) {\n geometry = offsetLine(geometry, lineOffset * pixelsToTileUnits);\n }\n\n return polygonIntersectsBufferedMultiLine(translatedPolygon, geometry, halfWidth);\n }\n\n isTileClipped() {\n return true;\n }\n}\n\nfunction getLineWidth(lineWidth, lineGapWidth) {\n if (lineGapWidth > 0) {\n return lineGapWidth + 2 * lineWidth;\n } else {\n return lineWidth;\n }\n}\n","import {createLayout} from '../../util/struct_array';\n\nexport const symbolLayoutAttributes = createLayout([\n {name: 'a_pos_offset', components: 4, type: 'Int16'},\n {name: 'a_data', components: 4, type: 'Uint16'},\n {name: 'a_pixeloffset', components: 4, type: 'Int16'}\n], 4);\n\nexport const dynamicLayoutAttributes = createLayout([\n {name: 'a_projected_pos', components: 3, type: 'Float32'}\n], 4);\n\nexport const placementOpacityAttributes = createLayout([\n {name: 'a_fade_opacity', components: 1, type: 'Uint32'}\n], 4);\n\nexport const collisionVertexAttributes = createLayout([\n {name: 'a_placed', components: 2, type: 'Uint8'},\n {name: 'a_shift', components: 2, type: 'Float32'}\n]);\n\nexport const collisionBox = createLayout([\n // the box is centered around the anchor point\n {type: 'Int16', name: 'anchorPointX'},\n {type: 'Int16', name: 'anchorPointY'},\n\n // distances to the edges from the anchor\n {type: 'Int16', name: 'x1'},\n {type: 'Int16', name: 'y1'},\n {type: 'Int16', name: 'x2'},\n {type: 'Int16', name: 'y2'},\n\n // the index of the feature in the original vectortile\n {type: 'Uint32', name: 'featureIndex'},\n // the source layer the feature appears in\n {type: 'Uint16', name: 'sourceLayerIndex'},\n // the bucket the feature appears in\n {type: 'Uint16', name: 'bucketIndex'},\n]);\n\nexport const collisionBoxLayout = createLayout([ // used to render collision boxes for debugging purposes\n {name: 'a_pos', components: 2, type: 'Int16'},\n {name: 'a_anchor_pos', components: 2, type: 'Int16'},\n {name: 'a_extrude', components: 2, type: 'Int16'}\n], 4);\n\nexport const collisionCircleLayout = createLayout([ // used to render collision circles for debugging purposes\n {name: 'a_pos', components: 2, type: 'Float32'},\n {name: 'a_radius', components: 1, type: 'Float32'},\n {name: 'a_flags', components: 2, type: 'Int16'}\n], 4);\n\nexport const quadTriangle = createLayout([\n {name: 'triangle', components: 3, type: 'Uint16'},\n]);\n\nexport const placement = createLayout([\n {type: 'Int16', name: 'anchorX'},\n {type: 'Int16', name: 'anchorY'},\n {type: 'Uint16', name: 'glyphStartIndex'},\n {type: 'Uint16', name: 'numGlyphs'},\n {type: 'Uint32', name: 'vertexStartIndex'},\n {type: 'Uint32', name: 'lineStartIndex'},\n {type: 'Uint32', name: 'lineLength'},\n {type: 'Uint16', name: 'segment'},\n {type: 'Uint16', name: 'lowerSize'},\n {type: 'Uint16', name: 'upperSize'},\n {type: 'Float32', name: 'lineOffsetX'},\n {type: 'Float32', name: 'lineOffsetY'},\n {type: 'Uint8', name: 'writingMode'},\n {type: 'Uint8', name: 'placedOrientation'},\n {type: 'Uint8', name: 'hidden'},\n {type: 'Uint32', name: 'crossTileID'},\n {type: 'Int16', name: 'associatedIconIndex'}\n]);\n\nexport const symbolInstance = createLayout([\n {type: 'Int16', name: 'anchorX'},\n {type: 'Int16', name: 'anchorY'},\n {type: 'Int16', name: 'rightJustifiedTextSymbolIndex'},\n {type: 'Int16', name: 'centerJustifiedTextSymbolIndex'},\n {type: 'Int16', name: 'leftJustifiedTextSymbolIndex'},\n {type: 'Int16', name: 'verticalPlacedTextSymbolIndex'},\n {type: 'Int16', name: 'placedIconSymbolIndex'},\n {type: 'Int16', name: 'verticalPlacedIconSymbolIndex'},\n {type: 'Uint16', name: 'key'},\n {type: 'Uint16', name: 'textBoxStartIndex'},\n {type: 'Uint16', name: 'textBoxEndIndex'},\n {type: 'Uint16', name: 'verticalTextBoxStartIndex'},\n {type: 'Uint16', name: 'verticalTextBoxEndIndex'},\n {type: 'Uint16', name: 'iconBoxStartIndex'},\n {type: 'Uint16', name: 'iconBoxEndIndex'},\n {type: 'Uint16', name: 'verticalIconBoxStartIndex'},\n {type: 'Uint16', name: 'verticalIconBoxEndIndex'},\n {type: 'Uint16', name: 'featureIndex'},\n {type: 'Uint16', name: 'numHorizontalGlyphVertices'},\n {type: 'Uint16', name: 'numVerticalGlyphVertices'},\n {type: 'Uint16', name: 'numIconVertices'},\n {type: 'Uint16', name: 'numVerticalIconVertices'},\n {type: 'Uint16', name: 'useRuntimeCollisionCircles'},\n {type: 'Uint32', name: 'crossTileID'},\n {type: 'Float32', name: 'textBoxScale'},\n {type: 'Float32', name: 'collisionCircleDiameter'},\n {type: 'Uint16', name: 'textAnchorOffsetStartIndex'},\n {type: 'Uint16', name: 'textAnchorOffsetEndIndex'}\n]);\n\nexport const glyphOffset = createLayout([\n {type: 'Float32', name: 'offsetX'}\n]);\n\nexport const lineVertex = createLayout([\n {type: 'Int16', name: 'x'},\n {type: 'Int16', name: 'y'},\n {type: 'Int16', name: 'tileUnitDistanceFromAnchor'}\n]);\n\nexport const textAnchorOffset = createLayout([\n {type: 'Uint16', name: 'textAnchor'},\n {type: 'Float32', components: 2, name: 'textOffset'}\n]);\n","import {plugin as rtlTextPlugin} from '../source/rtl_text_plugin';\n\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\nimport type {Feature} from '@maplibre/maplibre-gl-style-spec';\nimport {Formatted} from '@maplibre/maplibre-gl-style-spec';\n\nfunction transformTextInternal(text: string, layer: SymbolStyleLayer, feature: Feature) {\n const transform = layer.layout.get('text-transform').evaluate(feature, {});\n if (transform === 'uppercase') {\n text = text.toLocaleUpperCase();\n } else if (transform === 'lowercase') {\n text = text.toLocaleLowerCase();\n }\n\n if (rtlTextPlugin.applyArabicShaping) {\n text = rtlTextPlugin.applyArabicShaping(text);\n }\n\n return text;\n}\n\nexport function transformText(text: Formatted, layer: SymbolStyleLayer, feature: Feature): Formatted {\n text.sections.forEach(section => {\n section.text = transformTextInternal(section.text, layer, feature);\n });\n return text;\n}\n","import type {SymbolFeature} from '../data/bucket/symbol_bucket';\n\nexport function mergeLines(features: Array): Array {\n const leftIndex: {[_: string]: number} = {};\n const rightIndex: {[_: string]: number} = {};\n const mergedFeatures = [];\n let mergedIndex = 0;\n\n function add(k) {\n mergedFeatures.push(features[k]);\n mergedIndex++;\n }\n\n function mergeFromRight(leftKey: string, rightKey: string, geom) {\n const i = rightIndex[leftKey];\n delete rightIndex[leftKey];\n rightIndex[rightKey] = i;\n\n mergedFeatures[i].geometry[0].pop();\n mergedFeatures[i].geometry[0] = mergedFeatures[i].geometry[0].concat(geom[0]);\n return i;\n }\n\n function mergeFromLeft(leftKey: string, rightKey: string, geom) {\n const i = leftIndex[rightKey];\n delete leftIndex[rightKey];\n leftIndex[leftKey] = i;\n\n mergedFeatures[i].geometry[0].shift();\n mergedFeatures[i].geometry[0] = geom[0].concat(mergedFeatures[i].geometry[0]);\n return i;\n }\n\n function getKey(text, geom, onRight?) {\n const point = onRight ? geom[0][geom[0].length - 1] : geom[0][0];\n return `${text}:${point.x}:${point.y}`;\n }\n\n for (let k = 0; k < features.length; k++) {\n const feature = features[k];\n const geom = feature.geometry;\n const text = feature.text ? feature.text.toString() : null;\n\n if (!text) {\n add(k);\n continue;\n }\n\n const leftKey = getKey(text, geom),\n rightKey = getKey(text, geom, true);\n\n if ((leftKey in rightIndex) && (rightKey in leftIndex) && (rightIndex[leftKey] !== leftIndex[rightKey])) {\n // found lines with the same text adjacent to both ends of the current line, merge all three\n const j = mergeFromLeft(leftKey, rightKey, geom);\n const i = mergeFromRight(leftKey, rightKey, mergedFeatures[j].geometry);\n\n delete leftIndex[leftKey];\n delete rightIndex[rightKey];\n\n rightIndex[getKey(text, mergedFeatures[i].geometry, true)] = i;\n mergedFeatures[j].geometry = null;\n\n } else if (leftKey in rightIndex) {\n // found mergeable line adjacent to the start of the current line, merge\n mergeFromRight(leftKey, rightKey, geom);\n\n } else if (rightKey in leftIndex) {\n // found mergeable line adjacent to the end of the current line, merge\n mergeFromLeft(leftKey, rightKey, geom);\n\n } else {\n // no adjacent lines, add as a new item\n add(k);\n leftIndex[leftKey] = mergedIndex - 1;\n rightIndex[rightKey] = mergedIndex - 1;\n }\n }\n\n return mergedFeatures.filter((f) => f.geometry);\n}\n","import {charHasRotatedVerticalOrientation} from './script_detection';\n\nexport const verticalizedCharacterMap = {\n '!': '︕',\n '#': '#',\n '$': '$',\n '%': '%',\n '&': '&',\n '(': '︵',\n ')': '︶',\n '*': '*',\n '+': '+',\n ',': '︐',\n '-': '︲',\n '.': '・',\n '/': '/',\n ':': '︓',\n ';': '︔',\n '<': '︿',\n '=': '=',\n '>': '﹀',\n '?': '︖',\n '@': '@',\n '[': '﹇',\n '\\\\': '\',\n ']': '﹈',\n '^': '^',\n '_': '︳',\n '`': '`',\n '{': '︷',\n '|': '―',\n '}': '︸',\n '~': '~',\n '¢': '¢',\n '£': '£',\n '¥': '¥',\n '¦': '¦',\n '¬': '¬',\n '¯': ' ̄',\n '–': '︲',\n '—': '︱',\n '‘': '﹃',\n '’': '﹄',\n '“': '﹁',\n '”': '﹂',\n '…': '︙',\n '‧': '・',\n '₩': '₩',\n '、': '︑',\n '。': '︒',\n '〈': '︿',\n '〉': '﹀',\n '《': '︽',\n '》': '︾',\n '「': '﹁',\n '」': '﹂',\n '『': '﹃',\n '』': '﹄',\n '【': '︻',\n '】': '︼',\n '〔': '︹',\n '〕': '︺',\n '〖': '︗',\n '〗': '︘',\n '!': '︕',\n '(': '︵',\n ')': '︶',\n ',': '︐',\n '-': '︲',\n '.': '・',\n ':': '︓',\n ';': '︔',\n '<': '︿',\n '>': '﹀',\n '?': '︖',\n '[': '﹇',\n ']': '﹈',\n '_': '︳',\n '{': '︷',\n '|': '―',\n '}': '︸',\n '⦅': '︵',\n '⦆': '︶',\n '。': '︒',\n '「': '﹁',\n '」': '﹂'\n};\n\nexport function verticalizePunctuation(input: string) {\n let output = '';\n\n for (let i = 0; i < input.length; i++) {\n const nextCharCode = input.charCodeAt(i + 1) || null;\n const prevCharCode = input.charCodeAt(i - 1) || null;\n\n const canReplacePunctuation = (\n (!nextCharCode || !charHasRotatedVerticalOrientation(nextCharCode) || verticalizedCharacterMap[input[i + 1]]) &&\n (!prevCharCode || !charHasRotatedVerticalOrientation(prevCharCode) || verticalizedCharacterMap[input[i - 1]])\n );\n\n if (canReplacePunctuation && verticalizedCharacterMap[input[i]]) {\n output += verticalizedCharacterMap[input[i]];\n } else {\n output += input[i];\n }\n }\n\n return output;\n}\n\n","// ONE_EM constant used to go between \"em\" units used in style spec and \"points\" used internally for layout\n\nexport default 24;\n","/*! ieee754. BSD-3-Clause License. Feross Aboukhadijeh */\nexports.read = function (buffer, offset, isLE, mLen, nBytes) {\n var e, m\n var eLen = (nBytes * 8) - mLen - 1\n var eMax = (1 << eLen) - 1\n var eBias = eMax >> 1\n var nBits = -7\n var i = isLE ? (nBytes - 1) : 0\n var d = isLE ? -1 : 1\n var s = buffer[offset + i]\n\n i += d\n\n e = s & ((1 << (-nBits)) - 1)\n s >>= (-nBits)\n nBits += eLen\n for (; nBits > 0; e = (e * 256) + buffer[offset + i], i += d, nBits -= 8) {}\n\n m = e & ((1 << (-nBits)) - 1)\n e >>= (-nBits)\n nBits += mLen\n for (; nBits > 0; m = (m * 256) + buffer[offset + i], i += d, nBits -= 8) {}\n\n if (e === 0) {\n e = 1 - eBias\n } else if (e === eMax) {\n return m ? NaN : ((s ? -1 : 1) * Infinity)\n } else {\n m = m + Math.pow(2, mLen)\n e = e - eBias\n }\n return (s ? -1 : 1) * m * Math.pow(2, e - mLen)\n}\n\nexports.write = function (buffer, value, offset, isLE, mLen, nBytes) {\n var e, m, c\n var eLen = (nBytes * 8) - mLen - 1\n var eMax = (1 << eLen) - 1\n var eBias = eMax >> 1\n var rt = (mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0)\n var i = isLE ? 0 : (nBytes - 1)\n var d = isLE ? 1 : -1\n var s = value < 0 || (value === 0 && 1 / value < 0) ? 1 : 0\n\n value = Math.abs(value)\n\n if (isNaN(value) || value === Infinity) {\n m = isNaN(value) ? 1 : 0\n e = eMax\n } else {\n e = Math.floor(Math.log(value) / Math.LN2)\n if (value * (c = Math.pow(2, -e)) < 1) {\n e--\n c *= 2\n }\n if (e + eBias >= 1) {\n value += rt / c\n } else {\n value += rt * Math.pow(2, 1 - eBias)\n }\n if (value * c >= 2) {\n e++\n c /= 2\n }\n\n if (e + eBias >= eMax) {\n m = 0\n e = eMax\n } else if (e + eBias >= 1) {\n m = ((value * c) - 1) * Math.pow(2, mLen)\n e = e + eBias\n } else {\n m = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen)\n e = 0\n }\n }\n\n for (; mLen >= 8; buffer[offset + i] = m & 0xff, i += d, m /= 256, mLen -= 8) {}\n\n e = (e << mLen) | m\n eLen += mLen\n for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8) {}\n\n buffer[offset + i - d] |= s * 128\n}\n","'use strict';\n\nmodule.exports = Pbf;\n\nvar ieee754 = require('ieee754');\n\nfunction Pbf(buf) {\n this.buf = ArrayBuffer.isView && ArrayBuffer.isView(buf) ? buf : new Uint8Array(buf || 0);\n this.pos = 0;\n this.type = 0;\n this.length = this.buf.length;\n}\n\nPbf.Varint = 0; // varint: int32, int64, uint32, uint64, sint32, sint64, bool, enum\nPbf.Fixed64 = 1; // 64-bit: double, fixed64, sfixed64\nPbf.Bytes = 2; // length-delimited: string, bytes, embedded messages, packed repeated fields\nPbf.Fixed32 = 5; // 32-bit: float, fixed32, sfixed32\n\nvar SHIFT_LEFT_32 = (1 << 16) * (1 << 16),\n SHIFT_RIGHT_32 = 1 / SHIFT_LEFT_32;\n\n// Threshold chosen based on both benchmarking and knowledge about browser string\n// data structures (which currently switch structure types at 12 bytes or more)\nvar TEXT_DECODER_MIN_LENGTH = 12;\nvar utf8TextDecoder = typeof TextDecoder === 'undefined' ? null : new TextDecoder('utf8');\n\nPbf.prototype = {\n\n destroy: function() {\n this.buf = null;\n },\n\n // === READING =================================================================\n\n readFields: function(readField, result, end) {\n end = end || this.length;\n\n while (this.pos < end) {\n var val = this.readVarint(),\n tag = val >> 3,\n startPos = this.pos;\n\n this.type = val & 0x7;\n readField(tag, result, this);\n\n if (this.pos === startPos) this.skip(val);\n }\n return result;\n },\n\n readMessage: function(readField, result) {\n return this.readFields(readField, result, this.readVarint() + this.pos);\n },\n\n readFixed32: function() {\n var val = readUInt32(this.buf, this.pos);\n this.pos += 4;\n return val;\n },\n\n readSFixed32: function() {\n var val = readInt32(this.buf, this.pos);\n this.pos += 4;\n return val;\n },\n\n // 64-bit int handling is based on github.com/dpw/node-buffer-more-ints (MIT-licensed)\n\n readFixed64: function() {\n var val = readUInt32(this.buf, this.pos) + readUInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;\n this.pos += 8;\n return val;\n },\n\n readSFixed64: function() {\n var val = readUInt32(this.buf, this.pos) + readInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;\n this.pos += 8;\n return val;\n },\n\n readFloat: function() {\n var val = ieee754.read(this.buf, this.pos, true, 23, 4);\n this.pos += 4;\n return val;\n },\n\n readDouble: function() {\n var val = ieee754.read(this.buf, this.pos, true, 52, 8);\n this.pos += 8;\n return val;\n },\n\n readVarint: function(isSigned) {\n var buf = this.buf,\n val, b;\n\n b = buf[this.pos++]; val = b & 0x7f; if (b < 0x80) return val;\n b = buf[this.pos++]; val |= (b & 0x7f) << 7; if (b < 0x80) return val;\n b = buf[this.pos++]; val |= (b & 0x7f) << 14; if (b < 0x80) return val;\n b = buf[this.pos++]; val |= (b & 0x7f) << 21; if (b < 0x80) return val;\n b = buf[this.pos]; val |= (b & 0x0f) << 28;\n\n return readVarintRemainder(val, isSigned, this);\n },\n\n readVarint64: function() { // for compatibility with v2.0.1\n return this.readVarint(true);\n },\n\n readSVarint: function() {\n var num = this.readVarint();\n return num % 2 === 1 ? (num + 1) / -2 : num / 2; // zigzag encoding\n },\n\n readBoolean: function() {\n return Boolean(this.readVarint());\n },\n\n readString: function() {\n var end = this.readVarint() + this.pos;\n var pos = this.pos;\n this.pos = end;\n\n if (end - pos >= TEXT_DECODER_MIN_LENGTH && utf8TextDecoder) {\n // longer strings are fast with the built-in browser TextDecoder API\n return readUtf8TextDecoder(this.buf, pos, end);\n }\n // short strings are fast with our custom implementation\n return readUtf8(this.buf, pos, end);\n },\n\n readBytes: function() {\n var end = this.readVarint() + this.pos,\n buffer = this.buf.subarray(this.pos, end);\n this.pos = end;\n return buffer;\n },\n\n // verbose for performance reasons; doesn't affect gzipped size\n\n readPackedVarint: function(arr, isSigned) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readVarint(isSigned));\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readVarint(isSigned));\n return arr;\n },\n readPackedSVarint: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readSVarint());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readSVarint());\n return arr;\n },\n readPackedBoolean: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readBoolean());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readBoolean());\n return arr;\n },\n readPackedFloat: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readFloat());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readFloat());\n return arr;\n },\n readPackedDouble: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readDouble());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readDouble());\n return arr;\n },\n readPackedFixed32: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readFixed32());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readFixed32());\n return arr;\n },\n readPackedSFixed32: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed32());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readSFixed32());\n return arr;\n },\n readPackedFixed64: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readFixed64());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readFixed64());\n return arr;\n },\n readPackedSFixed64: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed64());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readSFixed64());\n return arr;\n },\n\n skip: function(val) {\n var type = val & 0x7;\n if (type === Pbf.Varint) while (this.buf[this.pos++] > 0x7f) {}\n else if (type === Pbf.Bytes) this.pos = this.readVarint() + this.pos;\n else if (type === Pbf.Fixed32) this.pos += 4;\n else if (type === Pbf.Fixed64) this.pos += 8;\n else throw new Error('Unimplemented type: ' + type);\n },\n\n // === WRITING =================================================================\n\n writeTag: function(tag, type) {\n this.writeVarint((tag << 3) | type);\n },\n\n realloc: function(min) {\n var length = this.length || 16;\n\n while (length < this.pos + min) length *= 2;\n\n if (length !== this.length) {\n var buf = new Uint8Array(length);\n buf.set(this.buf);\n this.buf = buf;\n this.length = length;\n }\n },\n\n finish: function() {\n this.length = this.pos;\n this.pos = 0;\n return this.buf.subarray(0, this.length);\n },\n\n writeFixed32: function(val) {\n this.realloc(4);\n writeInt32(this.buf, val, this.pos);\n this.pos += 4;\n },\n\n writeSFixed32: function(val) {\n this.realloc(4);\n writeInt32(this.buf, val, this.pos);\n this.pos += 4;\n },\n\n writeFixed64: function(val) {\n this.realloc(8);\n writeInt32(this.buf, val & -1, this.pos);\n writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4);\n this.pos += 8;\n },\n\n writeSFixed64: function(val) {\n this.realloc(8);\n writeInt32(this.buf, val & -1, this.pos);\n writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4);\n this.pos += 8;\n },\n\n writeVarint: function(val) {\n val = +val || 0;\n\n if (val > 0xfffffff || val < 0) {\n writeBigVarint(val, this);\n return;\n }\n\n this.realloc(4);\n\n this.buf[this.pos++] = val & 0x7f | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return;\n this.buf[this.pos++] = ((val >>>= 7) & 0x7f) | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return;\n this.buf[this.pos++] = ((val >>>= 7) & 0x7f) | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return;\n this.buf[this.pos++] = (val >>> 7) & 0x7f;\n },\n\n writeSVarint: function(val) {\n this.writeVarint(val < 0 ? -val * 2 - 1 : val * 2);\n },\n\n writeBoolean: function(val) {\n this.writeVarint(Boolean(val));\n },\n\n writeString: function(str) {\n str = String(str);\n this.realloc(str.length * 4);\n\n this.pos++; // reserve 1 byte for short string length\n\n var startPos = this.pos;\n // write the string directly to the buffer and see how much was written\n this.pos = writeUtf8(this.buf, str, this.pos);\n var len = this.pos - startPos;\n\n if (len >= 0x80) makeRoomForExtraLength(startPos, len, this);\n\n // finally, write the message length in the reserved place and restore the position\n this.pos = startPos - 1;\n this.writeVarint(len);\n this.pos += len;\n },\n\n writeFloat: function(val) {\n this.realloc(4);\n ieee754.write(this.buf, val, this.pos, true, 23, 4);\n this.pos += 4;\n },\n\n writeDouble: function(val) {\n this.realloc(8);\n ieee754.write(this.buf, val, this.pos, true, 52, 8);\n this.pos += 8;\n },\n\n writeBytes: function(buffer) {\n var len = buffer.length;\n this.writeVarint(len);\n this.realloc(len);\n for (var i = 0; i < len; i++) this.buf[this.pos++] = buffer[i];\n },\n\n writeRawMessage: function(fn, obj) {\n this.pos++; // reserve 1 byte for short message length\n\n // write the message directly to the buffer and see how much was written\n var startPos = this.pos;\n fn(obj, this);\n var len = this.pos - startPos;\n\n if (len >= 0x80) makeRoomForExtraLength(startPos, len, this);\n\n // finally, write the message length in the reserved place and restore the position\n this.pos = startPos - 1;\n this.writeVarint(len);\n this.pos += len;\n },\n\n writeMessage: function(tag, fn, obj) {\n this.writeTag(tag, Pbf.Bytes);\n this.writeRawMessage(fn, obj);\n },\n\n writePackedVarint: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedVarint, arr); },\n writePackedSVarint: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSVarint, arr); },\n writePackedBoolean: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedBoolean, arr); },\n writePackedFloat: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFloat, arr); },\n writePackedDouble: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedDouble, arr); },\n writePackedFixed32: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFixed32, arr); },\n writePackedSFixed32: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSFixed32, arr); },\n writePackedFixed64: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFixed64, arr); },\n writePackedSFixed64: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSFixed64, arr); },\n\n writeBytesField: function(tag, buffer) {\n this.writeTag(tag, Pbf.Bytes);\n this.writeBytes(buffer);\n },\n writeFixed32Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed32);\n this.writeFixed32(val);\n },\n writeSFixed32Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed32);\n this.writeSFixed32(val);\n },\n writeFixed64Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed64);\n this.writeFixed64(val);\n },\n writeSFixed64Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed64);\n this.writeSFixed64(val);\n },\n writeVarintField: function(tag, val) {\n this.writeTag(tag, Pbf.Varint);\n this.writeVarint(val);\n },\n writeSVarintField: function(tag, val) {\n this.writeTag(tag, Pbf.Varint);\n this.writeSVarint(val);\n },\n writeStringField: function(tag, str) {\n this.writeTag(tag, Pbf.Bytes);\n this.writeString(str);\n },\n writeFloatField: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed32);\n this.writeFloat(val);\n },\n writeDoubleField: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed64);\n this.writeDouble(val);\n },\n writeBooleanField: function(tag, val) {\n this.writeVarintField(tag, Boolean(val));\n }\n};\n\nfunction readVarintRemainder(l, s, p) {\n var buf = p.buf,\n h, b;\n\n b = buf[p.pos++]; h = (b & 0x70) >> 4; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 3; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 10; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 17; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 24; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x01) << 31; if (b < 0x80) return toNum(l, h, s);\n\n throw new Error('Expected varint not more than 10 bytes');\n}\n\nfunction readPackedEnd(pbf) {\n return pbf.type === Pbf.Bytes ?\n pbf.readVarint() + pbf.pos : pbf.pos + 1;\n}\n\nfunction toNum(low, high, isSigned) {\n if (isSigned) {\n return high * 0x100000000 + (low >>> 0);\n }\n\n return ((high >>> 0) * 0x100000000) + (low >>> 0);\n}\n\nfunction writeBigVarint(val, pbf) {\n var low, high;\n\n if (val >= 0) {\n low = (val % 0x100000000) | 0;\n high = (val / 0x100000000) | 0;\n } else {\n low = ~(-val % 0x100000000);\n high = ~(-val / 0x100000000);\n\n if (low ^ 0xffffffff) {\n low = (low + 1) | 0;\n } else {\n low = 0;\n high = (high + 1) | 0;\n }\n }\n\n if (val >= 0x10000000000000000 || val < -0x10000000000000000) {\n throw new Error('Given varint doesn\\'t fit into 10 bytes');\n }\n\n pbf.realloc(10);\n\n writeBigVarintLow(low, high, pbf);\n writeBigVarintHigh(high, pbf);\n}\n\nfunction writeBigVarintLow(low, high, pbf) {\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos] = low & 0x7f;\n}\n\nfunction writeBigVarintHigh(high, pbf) {\n var lsb = (high & 0x07) << 4;\n\n pbf.buf[pbf.pos++] |= lsb | ((high >>>= 3) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f;\n}\n\nfunction makeRoomForExtraLength(startPos, len, pbf) {\n var extraLen =\n len <= 0x3fff ? 1 :\n len <= 0x1fffff ? 2 :\n len <= 0xfffffff ? 3 : Math.floor(Math.log(len) / (Math.LN2 * 7));\n\n // if 1 byte isn't enough for encoding message length, shift the data to the right\n pbf.realloc(extraLen);\n for (var i = pbf.pos - 1; i >= startPos; i--) pbf.buf[i + extraLen] = pbf.buf[i];\n}\n\nfunction writePackedVarint(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeVarint(arr[i]); }\nfunction writePackedSVarint(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSVarint(arr[i]); }\nfunction writePackedFloat(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFloat(arr[i]); }\nfunction writePackedDouble(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeDouble(arr[i]); }\nfunction writePackedBoolean(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeBoolean(arr[i]); }\nfunction writePackedFixed32(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFixed32(arr[i]); }\nfunction writePackedSFixed32(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSFixed32(arr[i]); }\nfunction writePackedFixed64(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFixed64(arr[i]); }\nfunction writePackedSFixed64(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSFixed64(arr[i]); }\n\n// Buffer code below from https://github.com/feross/buffer, MIT-licensed\n\nfunction readUInt32(buf, pos) {\n return ((buf[pos]) |\n (buf[pos + 1] << 8) |\n (buf[pos + 2] << 16)) +\n (buf[pos + 3] * 0x1000000);\n}\n\nfunction writeInt32(buf, val, pos) {\n buf[pos] = val;\n buf[pos + 1] = (val >>> 8);\n buf[pos + 2] = (val >>> 16);\n buf[pos + 3] = (val >>> 24);\n}\n\nfunction readInt32(buf, pos) {\n return ((buf[pos]) |\n (buf[pos + 1] << 8) |\n (buf[pos + 2] << 16)) +\n (buf[pos + 3] << 24);\n}\n\nfunction readUtf8(buf, pos, end) {\n var str = '';\n var i = pos;\n\n while (i < end) {\n var b0 = buf[i];\n var c = null; // codepoint\n var bytesPerSequence =\n b0 > 0xEF ? 4 :\n b0 > 0xDF ? 3 :\n b0 > 0xBF ? 2 : 1;\n\n if (i + bytesPerSequence > end) break;\n\n var b1, b2, b3;\n\n if (bytesPerSequence === 1) {\n if (b0 < 0x80) {\n c = b0;\n }\n } else if (bytesPerSequence === 2) {\n b1 = buf[i + 1];\n if ((b1 & 0xC0) === 0x80) {\n c = (b0 & 0x1F) << 0x6 | (b1 & 0x3F);\n if (c <= 0x7F) {\n c = null;\n }\n }\n } else if (bytesPerSequence === 3) {\n b1 = buf[i + 1];\n b2 = buf[i + 2];\n if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80) {\n c = (b0 & 0xF) << 0xC | (b1 & 0x3F) << 0x6 | (b2 & 0x3F);\n if (c <= 0x7FF || (c >= 0xD800 && c <= 0xDFFF)) {\n c = null;\n }\n }\n } else if (bytesPerSequence === 4) {\n b1 = buf[i + 1];\n b2 = buf[i + 2];\n b3 = buf[i + 3];\n if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80 && (b3 & 0xC0) === 0x80) {\n c = (b0 & 0xF) << 0x12 | (b1 & 0x3F) << 0xC | (b2 & 0x3F) << 0x6 | (b3 & 0x3F);\n if (c <= 0xFFFF || c >= 0x110000) {\n c = null;\n }\n }\n }\n\n if (c === null) {\n c = 0xFFFD;\n bytesPerSequence = 1;\n\n } else if (c > 0xFFFF) {\n c -= 0x10000;\n str += String.fromCharCode(c >>> 10 & 0x3FF | 0xD800);\n c = 0xDC00 | c & 0x3FF;\n }\n\n str += String.fromCharCode(c);\n i += bytesPerSequence;\n }\n\n return str;\n}\n\nfunction readUtf8TextDecoder(buf, pos, end) {\n return utf8TextDecoder.decode(buf.subarray(pos, end));\n}\n\nfunction writeUtf8(buf, str, pos) {\n for (var i = 0, c, lead; i < str.length; i++) {\n c = str.charCodeAt(i); // code point\n\n if (c > 0xD7FF && c < 0xE000) {\n if (lead) {\n if (c < 0xDC00) {\n buf[pos++] = 0xEF;\n buf[pos++] = 0xBF;\n buf[pos++] = 0xBD;\n lead = c;\n continue;\n } else {\n c = lead - 0xD800 << 10 | c - 0xDC00 | 0x10000;\n lead = null;\n }\n } else {\n if (c > 0xDBFF || (i + 1 === str.length)) {\n buf[pos++] = 0xEF;\n buf[pos++] = 0xBF;\n buf[pos++] = 0xBD;\n } else {\n lead = c;\n }\n continue;\n }\n } else if (lead) {\n buf[pos++] = 0xEF;\n buf[pos++] = 0xBF;\n buf[pos++] = 0xBD;\n lead = null;\n }\n\n if (c < 0x80) {\n buf[pos++] = c;\n } else {\n if (c < 0x800) {\n buf[pos++] = c >> 0x6 | 0xC0;\n } else {\n if (c < 0x10000) {\n buf[pos++] = c >> 0xC | 0xE0;\n } else {\n buf[pos++] = c >> 0x12 | 0xF0;\n buf[pos++] = c >> 0xC & 0x3F | 0x80;\n }\n buf[pos++] = c >> 0x6 & 0x3F | 0x80;\n }\n buf[pos++] = c & 0x3F | 0x80;\n }\n }\n return pos;\n}\n","import {AlphaImage} from '../util/image';\n\nimport Protobuf from 'pbf';\nconst border = 3;\n\nimport type {StyleGlyph} from './style_glyph';\n\nfunction readFontstacks(tag: number, glyphs: Array, pbf: Protobuf) {\n if (tag === 1) {\n pbf.readMessage(readFontstack, glyphs);\n }\n}\n\nfunction readFontstack(tag: number, glyphs: Array, pbf: Protobuf) {\n if (tag === 3) {\n const {id, bitmap, width, height, left, top, advance} = pbf.readMessage(readGlyph, {});\n glyphs.push({\n id,\n bitmap: new AlphaImage({\n width: width + 2 * border,\n height: height + 2 * border\n }, bitmap),\n metrics: {width, height, left, top, advance}\n });\n }\n}\n\nfunction readGlyph(tag: number, glyph: any, pbf: Protobuf) {\n if (tag === 1) glyph.id = pbf.readVarint();\n else if (tag === 2) glyph.bitmap = pbf.readBytes();\n else if (tag === 3) glyph.width = pbf.readVarint();\n else if (tag === 4) glyph.height = pbf.readVarint();\n else if (tag === 5) glyph.left = pbf.readSVarint();\n else if (tag === 6) glyph.top = pbf.readSVarint();\n else if (tag === 7) glyph.advance = pbf.readVarint();\n}\n\nexport function parseGlyphPbf(data: ArrayBuffer | Uint8Array): Array {\n return new Protobuf(data).readFields(readFontstacks, []);\n}\n\nexport const GLYPH_PBF_BORDER = border;\n","\nexport default function potpack(boxes) {\n\n // calculate total box area and maximum box width\n let area = 0;\n let maxWidth = 0;\n\n for (const box of boxes) {\n area += box.w * box.h;\n maxWidth = Math.max(maxWidth, box.w);\n }\n\n // sort the boxes for insertion by height, descending\n boxes.sort((a, b) => b.h - a.h);\n\n // aim for a squarish resulting container,\n // slightly adjusted for sub-100% space utilization\n const startWidth = Math.max(Math.ceil(Math.sqrt(area / 0.95)), maxWidth);\n\n // start with a single empty space, unbounded at the bottom\n const spaces = [{x: 0, y: 0, w: startWidth, h: Infinity}];\n\n let width = 0;\n let height = 0;\n\n for (const box of boxes) {\n // look through spaces backwards so that we check smaller spaces first\n for (let i = spaces.length - 1; i >= 0; i--) {\n const space = spaces[i];\n\n // look for empty spaces that can accommodate the current box\n if (box.w > space.w || box.h > space.h) continue;\n\n // found the space; add the box to its top-left corner\n // |-------|-------|\n // | box | |\n // |_______| |\n // | space |\n // |_______________|\n box.x = space.x;\n box.y = space.y;\n\n height = Math.max(height, box.y + box.h);\n width = Math.max(width, box.x + box.w);\n\n if (box.w === space.w && box.h === space.h) {\n // space matches the box exactly; remove it\n const last = spaces.pop();\n if (i < spaces.length) spaces[i] = last;\n\n } else if (box.h === space.h) {\n // space matches the box height; update it accordingly\n // |-------|---------------|\n // | box | updated space |\n // |_______|_______________|\n space.x += box.w;\n space.w -= box.w;\n\n } else if (box.w === space.w) {\n // space matches the box width; update it accordingly\n // |---------------|\n // | box |\n // |_______________|\n // | updated space |\n // |_______________|\n space.y += box.h;\n space.h -= box.h;\n\n } else {\n // otherwise the box splits the space into two spaces\n // |-------|-----------|\n // | box | new space |\n // |_______|___________|\n // | updated space |\n // |___________________|\n spaces.push({\n x: space.x + box.w,\n y: space.y,\n w: space.w - box.w,\n h: box.h\n });\n space.y += box.h;\n space.h -= box.h;\n }\n break;\n }\n }\n\n return {\n w: width, // container width\n h: height, // container height\n fill: (area / (width * height)) || 0 // space utilization\n };\n}\n","/* eslint-disable key-spacing */\nimport {RGBAImage} from '../util/image';\nimport {register} from '../util/web_worker_transfer';\nimport potpack from 'potpack';\n\nimport type {StyleImage} from '../style/style_image';\nimport type {ImageManager} from './image_manager';\nimport type {Texture} from './texture';\nimport type {Rect} from './glyph_atlas';\n\nconst IMAGE_PADDING: number = 1;\nexport {IMAGE_PADDING};\n\nexport class ImagePosition {\n paddedRect: Rect;\n pixelRatio: number;\n version: number;\n stretchY: Array<[number, number]>;\n stretchX: Array<[number, number]>;\n content: [number, number, number, number];\n\n constructor(paddedRect: Rect, {\n pixelRatio,\n version,\n stretchX,\n stretchY,\n content\n }: StyleImage) {\n this.paddedRect = paddedRect;\n this.pixelRatio = pixelRatio;\n this.stretchX = stretchX;\n this.stretchY = stretchY;\n this.content = content;\n this.version = version;\n }\n\n get tl(): [number, number] {\n return [\n this.paddedRect.x + IMAGE_PADDING,\n this.paddedRect.y + IMAGE_PADDING\n ];\n }\n\n get br(): [number, number] {\n return [\n this.paddedRect.x + this.paddedRect.w - IMAGE_PADDING,\n this.paddedRect.y + this.paddedRect.h - IMAGE_PADDING\n ];\n }\n\n get tlbr(): Array {\n return this.tl.concat(this.br);\n }\n\n get displaySize(): [number, number] {\n return [\n (this.paddedRect.w - IMAGE_PADDING * 2) / this.pixelRatio,\n (this.paddedRect.h - IMAGE_PADDING * 2) / this.pixelRatio\n ];\n }\n}\n\n/**\n * @internal\n * A class holding all the images\n */\nexport class ImageAtlas {\n image: RGBAImage;\n iconPositions: {[_: string]: ImagePosition};\n patternPositions: {[_: string]: ImagePosition};\n haveRenderCallbacks: Array;\n uploaded: boolean;\n\n constructor(icons: {[_: string]: StyleImage}, patterns: {[_: string]: StyleImage}) {\n const iconPositions = {}, patternPositions = {};\n this.haveRenderCallbacks = [];\n\n const bins = [];\n\n this.addImages(icons, iconPositions, bins);\n this.addImages(patterns, patternPositions, bins);\n\n const {w, h} = potpack(bins);\n const image = new RGBAImage({width: w || 1, height: h || 1});\n\n for (const id in icons) {\n const src = icons[id];\n const bin = iconPositions[id].paddedRect;\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x: bin.x + IMAGE_PADDING, y: bin.y + IMAGE_PADDING}, src.data);\n }\n\n for (const id in patterns) {\n const src = patterns[id];\n const bin = patternPositions[id].paddedRect;\n const x = bin.x + IMAGE_PADDING,\n y = bin.y + IMAGE_PADDING,\n w = src.data.width,\n h = src.data.height;\n\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x, y}, src.data);\n // Add 1 pixel wrapped padding on each side of the image.\n RGBAImage.copy(src.data, image, {x: 0, y: h - 1}, {x, y: y - 1}, {width: w, height: 1}); // T\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x, y: y + h}, {width: w, height: 1}); // B\n RGBAImage.copy(src.data, image, {x: w - 1, y: 0}, {x: x - 1, y}, {width: 1, height: h}); // L\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x: x + w, y}, {width: 1, height: h}); // R\n }\n\n this.image = image;\n this.iconPositions = iconPositions;\n this.patternPositions = patternPositions;\n }\n\n addImages(images: {[_: string]: StyleImage}, positions: {[_: string]: ImagePosition}, bins: Array) {\n for (const id in images) {\n const src = images[id];\n const bin = {\n x: 0,\n y: 0,\n w: src.data.width + 2 * IMAGE_PADDING,\n h: src.data.height + 2 * IMAGE_PADDING,\n };\n bins.push(bin);\n positions[id] = new ImagePosition(bin, src);\n\n if (src.hasRenderCallback) {\n this.haveRenderCallbacks.push(id);\n }\n }\n }\n\n patchUpdatedImages(imageManager: ImageManager, texture: Texture) {\n imageManager.dispatchRenderCallbacks(this.haveRenderCallbacks);\n for (const name in imageManager.updatedImages) {\n this.patchUpdatedImage(this.iconPositions[name], imageManager.getImage(name), texture);\n this.patchUpdatedImage(this.patternPositions[name], imageManager.getImage(name), texture);\n }\n }\n\n patchUpdatedImage(position: ImagePosition, image: StyleImage, texture: Texture) {\n if (!position || !image) return;\n\n if (position.version === image.version) return;\n\n position.version = image.version;\n const [x, y] = position.tl;\n texture.update(image.data, undefined, {x, y});\n }\n\n}\n\nregister('ImagePosition', ImagePosition);\nregister('ImageAtlas', ImageAtlas);\n","import {\n charHasUprightVerticalOrientation,\n charAllowsIdeographicBreaking,\n charInComplexShapingScript\n} from '../util/script_detection';\nimport {verticalizePunctuation} from '../util/verticalize_punctuation';\nimport {plugin as rtlTextPlugin} from '../source/rtl_text_plugin';\nimport ONE_EM from './one_em';\nimport {warnOnce} from '../util/util';\n\nimport type {StyleGlyph, GlyphMetrics} from '../style/style_glyph';\nimport {GLYPH_PBF_BORDER} from '../style/parse_glyph_pbf';\nimport type {ImagePosition} from '../render/image_atlas';\nimport {IMAGE_PADDING} from '../render/image_atlas';\nimport type {Rect, GlyphPosition} from '../render/glyph_atlas';\nimport {Formatted, FormattedSection} from '@maplibre/maplibre-gl-style-spec';\n\nenum WritingMode {\n none = 0,\n horizontal = 1,\n vertical = 2,\n horizontalOnly = 3\n}\n\nconst SHAPING_DEFAULT_OFFSET = -17;\nexport {shapeText, shapeIcon, fitIconToText, getAnchorAlignment, WritingMode, SHAPING_DEFAULT_OFFSET};\n\n// The position of a glyph relative to the text's anchor point.\nexport type PositionedGlyph = {\n glyph: number;\n imageName: string | null;\n x: number;\n y: number;\n vertical: boolean;\n scale: number;\n fontStack: string;\n sectionIndex: number;\n metrics: GlyphMetrics;\n rect: Rect | null;\n};\n\nexport type PositionedLine = {\n positionedGlyphs: Array;\n lineOffset: number;\n};\n\n// A collection of positioned glyphs and some metadata\nexport type Shaping = {\n positionedLines: Array;\n top: number;\n bottom: number;\n left: number;\n right: number;\n writingMode: WritingMode.horizontal | WritingMode.vertical;\n text: string;\n iconsInText: boolean;\n verticalizable: boolean;\n};\n\nfunction isEmpty(positionedLines: Array) {\n for (const line of positionedLines) {\n if (line.positionedGlyphs.length !== 0) {\n return false;\n }\n }\n return true;\n}\n\nexport type SymbolAnchor = 'center' | 'left' | 'right' | 'top' | 'bottom' | 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right';\nexport type TextJustify = 'left' | 'center' | 'right';\n\n// Max number of images in label is 6401 U+E000–U+F8FF that covers\n// Basic Multilingual Plane Unicode Private Use Area (PUA).\nconst PUAbegin = 0xE000;\nconst PUAend = 0xF8FF;\n\nclass SectionOptions {\n // Text options\n scale: number;\n fontStack: string;\n // Image options\n imageName: string | null;\n\n constructor() {\n this.scale = 1.0;\n this.fontStack = '';\n this.imageName = null;\n }\n\n static forText(scale: number | null, fontStack: string) {\n const textOptions = new SectionOptions();\n textOptions.scale = scale || 1;\n textOptions.fontStack = fontStack;\n return textOptions;\n }\n\n static forImage(imageName: string) {\n const imageOptions = new SectionOptions();\n imageOptions.imageName = imageName;\n return imageOptions;\n }\n\n}\n\nclass TaggedString {\n text: string;\n sectionIndex: Array; // maps each character in 'text' to its corresponding entry in 'sections'\n sections: Array;\n imageSectionID: number | null;\n\n constructor() {\n this.text = '';\n this.sectionIndex = [];\n this.sections = [];\n this.imageSectionID = null;\n }\n\n static fromFeature(text: Formatted, defaultFontStack: string) {\n const result = new TaggedString();\n for (let i = 0; i < text.sections.length; i++) {\n const section = text.sections[i];\n if (!section.image) {\n result.addTextSection(section, defaultFontStack);\n } else {\n result.addImageSection(section);\n }\n }\n return result;\n }\n\n length(): number {\n return this.text.length;\n }\n\n getSection(index: number): SectionOptions {\n return this.sections[this.sectionIndex[index]];\n }\n\n getSectionIndex(index: number): number {\n return this.sectionIndex[index];\n }\n\n getCharCode(index: number): number {\n return this.text.charCodeAt(index);\n }\n\n verticalizePunctuation() {\n this.text = verticalizePunctuation(this.text);\n }\n\n trim() {\n let beginningWhitespace = 0;\n for (let i = 0;\n i < this.text.length && whitespace[this.text.charCodeAt(i)];\n i++) {\n beginningWhitespace++;\n }\n let trailingWhitespace = this.text.length;\n for (let i = this.text.length - 1;\n i >= 0 && i >= beginningWhitespace && whitespace[this.text.charCodeAt(i)];\n i--) {\n trailingWhitespace--;\n }\n this.text = this.text.substring(beginningWhitespace, trailingWhitespace);\n this.sectionIndex = this.sectionIndex.slice(beginningWhitespace, trailingWhitespace);\n }\n\n substring(start: number, end: number): TaggedString {\n const substring = new TaggedString();\n substring.text = this.text.substring(start, end);\n substring.sectionIndex = this.sectionIndex.slice(start, end);\n substring.sections = this.sections;\n return substring;\n }\n\n toString(): string {\n return this.text;\n }\n\n getMaxScale() {\n return this.sectionIndex.reduce((max, index) => Math.max(max, this.sections[index].scale), 0);\n }\n\n addTextSection(section: FormattedSection, defaultFontStack: string) {\n this.text += section.text;\n this.sections.push(SectionOptions.forText(section.scale, section.fontStack || defaultFontStack));\n const index = this.sections.length - 1;\n for (let i = 0; i < section.text.length; ++i) {\n this.sectionIndex.push(index);\n }\n }\n\n addImageSection(section: FormattedSection) {\n const imageName = section.image ? section.image.name : '';\n if (imageName.length === 0) {\n warnOnce('Can\\'t add FormattedSection with an empty image.');\n return;\n }\n\n const nextImageSectionCharCode = this.getNextImageSectionCharCode();\n if (!nextImageSectionCharCode) {\n warnOnce(`Reached maximum number of images ${PUAend - PUAbegin + 2}`);\n return;\n }\n\n this.text += String.fromCharCode(nextImageSectionCharCode);\n this.sections.push(SectionOptions.forImage(imageName));\n this.sectionIndex.push(this.sections.length - 1);\n }\n\n getNextImageSectionCharCode(): number | null {\n if (!this.imageSectionID) {\n this.imageSectionID = PUAbegin;\n return this.imageSectionID;\n }\n\n if (this.imageSectionID >= PUAend) return null;\n return ++this.imageSectionID;\n }\n}\n\nfunction breakLines(input: TaggedString, lineBreakPoints: Array): Array {\n const lines = [];\n const text = input.text;\n let start = 0;\n for (const lineBreak of lineBreakPoints) {\n lines.push(input.substring(start, lineBreak));\n start = lineBreak;\n }\n\n if (start < text.length) {\n lines.push(input.substring(start, text.length));\n }\n return lines;\n}\n\nfunction shapeText(\n text: Formatted,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n glyphPositions: {\n [_: string]: {\n [_: number]: GlyphPosition;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n defaultFontStack: string,\n maxWidth: number,\n lineHeight: number,\n textAnchor: SymbolAnchor,\n textJustify: TextJustify,\n spacing: number,\n translate: [number, number],\n writingMode: WritingMode.horizontal | WritingMode.vertical,\n allowVerticalPlacement: boolean,\n symbolPlacement: string,\n layoutTextSize: number,\n layoutTextSizeThisZoom: number\n): Shaping | false {\n const logicalInput = TaggedString.fromFeature(text, defaultFontStack);\n\n if (writingMode === WritingMode.vertical) {\n logicalInput.verticalizePunctuation();\n }\n\n let lines: Array;\n\n const {processBidirectionalText, processStyledBidirectionalText} = rtlTextPlugin;\n if (processBidirectionalText && logicalInput.sections.length === 1) {\n // Bidi doesn't have to be style-aware\n lines = [];\n const untaggedLines =\n processBidirectionalText(logicalInput.toString(),\n determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize));\n for (const line of untaggedLines) {\n const taggedLine = new TaggedString();\n taggedLine.text = line;\n taggedLine.sections = logicalInput.sections;\n for (let i = 0; i < line.length; i++) {\n taggedLine.sectionIndex.push(0);\n }\n lines.push(taggedLine);\n }\n } else if (processStyledBidirectionalText) {\n // Need version of mapbox-gl-rtl-text with style support for combining RTL text\n // with formatting\n lines = [];\n const processedLines =\n processStyledBidirectionalText(logicalInput.text,\n logicalInput.sectionIndex,\n determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize));\n for (const line of processedLines) {\n const taggedLine = new TaggedString();\n taggedLine.text = line[0];\n taggedLine.sectionIndex = line[1];\n taggedLine.sections = logicalInput.sections;\n lines.push(taggedLine);\n }\n } else {\n lines = breakLines(logicalInput, determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize));\n }\n\n const positionedLines = [];\n const shaping = {\n positionedLines,\n text: logicalInput.toString(),\n top: translate[1],\n bottom: translate[1],\n left: translate[0],\n right: translate[0],\n writingMode,\n iconsInText: false,\n verticalizable: false\n };\n\n shapeLines(shaping, glyphMap, glyphPositions, imagePositions, lines, lineHeight, textAnchor, textJustify, writingMode, spacing, allowVerticalPlacement, layoutTextSizeThisZoom);\n if (isEmpty(positionedLines)) return false;\n\n return shaping;\n}\n\n// using computed properties due to https://github.com/facebook/flow/issues/380\n/* eslint no-useless-computed-key: 0 */\n\nconst whitespace: {\n [_: number]: boolean;\n} = {\n [0x09]: true, // tab\n [0x0a]: true, // newline\n [0x0b]: true, // vertical tab\n [0x0c]: true, // form feed\n [0x0d]: true, // carriage return\n [0x20]: true, // space\n};\n\nconst breakable: {\n [_: number]: boolean;\n} = {\n [0x0a]: true, // newline\n [0x20]: true, // space\n [0x26]: true, // ampersand\n [0x28]: true, // left parenthesis\n [0x29]: true, // right parenthesis\n [0x2b]: true, // plus sign\n [0x2d]: true, // hyphen-minus\n [0x2f]: true, // solidus\n [0xad]: true, // soft hyphen\n [0xb7]: true, // middle dot\n [0x200b]: true, // zero-width space\n [0x2010]: true, // hyphen\n [0x2013]: true, // en dash\n [0x2027]: true // interpunct\n // Many other characters may be reasonable breakpoints\n // Consider \"neutral orientation\" characters at scriptDetection.charHasNeutralVerticalOrientation\n // See https://github.com/mapbox/mapbox-gl-js/issues/3658\n};\n\nfunction getGlyphAdvance(\n codePoint: number,\n section: SectionOptions,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n spacing: number,\n layoutTextSize: number\n): number {\n if (!section.imageName) {\n const positions = glyphMap[section.fontStack];\n const glyph = positions && positions[codePoint];\n if (!glyph) return 0;\n return glyph.metrics.advance * section.scale + spacing;\n } else {\n const imagePosition = imagePositions[section.imageName];\n if (!imagePosition) return 0;\n return imagePosition.displaySize[0] * section.scale * ONE_EM / layoutTextSize + spacing;\n }\n}\n\nfunction determineAverageLineWidth(logicalInput: TaggedString,\n spacing: number,\n maxWidth: number,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n layoutTextSize: number) {\n let totalWidth = 0;\n\n for (let index = 0; index < logicalInput.length(); index++) {\n const section = logicalInput.getSection(index);\n totalWidth += getGlyphAdvance(logicalInput.getCharCode(index), section, glyphMap, imagePositions, spacing, layoutTextSize);\n }\n\n const lineCount = Math.max(1, Math.ceil(totalWidth / maxWidth));\n return totalWidth / lineCount;\n}\n\nfunction calculateBadness(lineWidth: number,\n targetWidth: number,\n penalty: number,\n isLastBreak: boolean) {\n const raggedness = Math.pow(lineWidth - targetWidth, 2);\n if (isLastBreak) {\n // Favor finals lines shorter than average over longer than average\n if (lineWidth < targetWidth) {\n return raggedness / 2;\n } else {\n return raggedness * 2;\n }\n }\n\n return raggedness + Math.abs(penalty) * penalty;\n}\n\nfunction calculatePenalty(codePoint: number, nextCodePoint: number, penalizableIdeographicBreak: boolean) {\n let penalty = 0;\n // Force break on newline\n if (codePoint === 0x0a) {\n penalty -= 10000;\n }\n // Penalize breaks between characters that allow ideographic breaking because\n // they are less preferable than breaks at spaces (or zero width spaces).\n if (penalizableIdeographicBreak) {\n penalty += 150;\n }\n\n // Penalize open parenthesis at end of line\n if (codePoint === 0x28 || codePoint === 0xff08) {\n penalty += 50;\n }\n\n // Penalize close parenthesis at beginning of line\n if (nextCodePoint === 0x29 || nextCodePoint === 0xff09) {\n penalty += 50;\n }\n return penalty;\n}\n\ntype Break = {\n index: number;\n x: number;\n priorBreak: Break;\n badness: number;\n};\n\nfunction evaluateBreak(\n breakIndex: number,\n breakX: number,\n targetWidth: number,\n potentialBreaks: Array,\n penalty: number,\n isLastBreak: boolean\n): Break {\n // We could skip evaluating breaks where the line length (breakX - priorBreak.x) > maxWidth\n // ...but in fact we allow lines longer than maxWidth (if there's no break points)\n // ...and when targetWidth and maxWidth are close, strictly enforcing maxWidth can give\n // more lopsided results.\n\n let bestPriorBreak: Break = null;\n let bestBreakBadness = calculateBadness(breakX, targetWidth, penalty, isLastBreak);\n\n for (const potentialBreak of potentialBreaks) {\n const lineWidth = breakX - potentialBreak.x;\n const breakBadness =\n calculateBadness(lineWidth, targetWidth, penalty, isLastBreak) + potentialBreak.badness;\n if (breakBadness <= bestBreakBadness) {\n bestPriorBreak = potentialBreak;\n bestBreakBadness = breakBadness;\n }\n }\n\n return {\n index: breakIndex,\n x: breakX,\n priorBreak: bestPriorBreak,\n badness: bestBreakBadness\n };\n}\n\nfunction leastBadBreaks(lastLineBreak?: Break | null): Array {\n if (!lastLineBreak) {\n return [];\n }\n return leastBadBreaks(lastLineBreak.priorBreak).concat(lastLineBreak.index);\n}\n\nfunction determineLineBreaks(\n logicalInput: TaggedString,\n spacing: number,\n maxWidth: number,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n symbolPlacement: string,\n layoutTextSize: number\n): Array {\n if (symbolPlacement !== 'point')\n return [];\n\n if (!logicalInput)\n return [];\n\n const potentialLineBreaks = [];\n const targetWidth = determineAverageLineWidth(logicalInput, spacing, maxWidth, glyphMap, imagePositions, layoutTextSize);\n\n const hasServerSuggestedBreakpoints = logicalInput.text.indexOf('\\u200b') >= 0;\n\n let currentX = 0;\n\n for (let i = 0; i < logicalInput.length(); i++) {\n const section = logicalInput.getSection(i);\n const codePoint = logicalInput.getCharCode(i);\n if (!whitespace[codePoint]) currentX += getGlyphAdvance(codePoint, section, glyphMap, imagePositions, spacing, layoutTextSize);\n\n // Ideographic characters, spaces, and word-breaking punctuation that often appear without\n // surrounding spaces.\n if ((i < logicalInput.length() - 1)) {\n const ideographicBreak = charAllowsIdeographicBreaking(codePoint);\n if (breakable[codePoint] || ideographicBreak || section.imageName) {\n\n potentialLineBreaks.push(\n evaluateBreak(\n i + 1,\n currentX,\n targetWidth,\n potentialLineBreaks,\n calculatePenalty(codePoint, logicalInput.getCharCode(i + 1), ideographicBreak && hasServerSuggestedBreakpoints),\n false));\n }\n }\n }\n\n return leastBadBreaks(\n evaluateBreak(\n logicalInput.length(),\n currentX,\n targetWidth,\n potentialLineBreaks,\n 0,\n true));\n}\n\nfunction getAnchorAlignment(anchor: SymbolAnchor) {\n let horizontalAlign = 0.5, verticalAlign = 0.5;\n\n switch (anchor) {\n case 'right':\n case 'top-right':\n case 'bottom-right':\n horizontalAlign = 1;\n break;\n case 'left':\n case 'top-left':\n case 'bottom-left':\n horizontalAlign = 0;\n break;\n }\n\n switch (anchor) {\n case 'bottom':\n case 'bottom-right':\n case 'bottom-left':\n verticalAlign = 1;\n break;\n case 'top':\n case 'top-right':\n case 'top-left':\n verticalAlign = 0;\n break;\n }\n\n return {horizontalAlign, verticalAlign};\n}\n\nfunction shapeLines(shaping: Shaping,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n glyphPositions: {\n [_: string]: {\n [_: number]: GlyphPosition;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n lines: Array,\n lineHeight: number,\n textAnchor: SymbolAnchor,\n textJustify: TextJustify,\n writingMode: WritingMode.horizontal | WritingMode.vertical,\n spacing: number,\n allowVerticalPlacement: boolean,\n layoutTextSizeThisZoom: number) {\n\n let x = 0;\n let y = SHAPING_DEFAULT_OFFSET;\n\n let maxLineLength = 0;\n let maxLineHeight = 0;\n\n const justify =\n textJustify === 'right' ? 1 :\n textJustify === 'left' ? 0 : 0.5;\n\n let lineIndex = 0;\n for (const line of lines) {\n line.trim();\n\n const lineMaxScale = line.getMaxScale();\n const maxLineOffset = (lineMaxScale - 1) * ONE_EM;\n const positionedLine = {positionedGlyphs: [], lineOffset: 0};\n shaping.positionedLines[lineIndex] = positionedLine;\n const positionedGlyphs = positionedLine.positionedGlyphs;\n let lineOffset = 0.0;\n\n if (!line.length()) {\n y += lineHeight; // Still need a line feed after empty line\n ++lineIndex;\n continue;\n }\n\n for (let i = 0; i < line.length(); i++) {\n const section = line.getSection(i);\n const sectionIndex = line.getSectionIndex(i);\n const codePoint = line.getCharCode(i);\n let baselineOffset = 0.0;\n let metrics = null;\n let rect = null;\n let imageName = null;\n let verticalAdvance = ONE_EM;\n const vertical = !(writingMode === WritingMode.horizontal ||\n // Don't verticalize glyphs that have no upright orientation if vertical placement is disabled.\n (!allowVerticalPlacement && !charHasUprightVerticalOrientation(codePoint)) ||\n // If vertical placement is enabled, don't verticalize glyphs that\n // are from complex text layout script, or whitespaces.\n (allowVerticalPlacement && (whitespace[codePoint] || charInComplexShapingScript(codePoint))));\n\n if (!section.imageName) {\n const positions = glyphPositions[section.fontStack];\n const glyphPosition = positions && positions[codePoint];\n if (glyphPosition && glyphPosition.rect) {\n rect = glyphPosition.rect;\n metrics = glyphPosition.metrics;\n } else {\n const glyphs = glyphMap[section.fontStack];\n const glyph = glyphs && glyphs[codePoint];\n if (!glyph) continue;\n metrics = glyph.metrics;\n }\n\n // We don't know the baseline, but since we're laying out\n // at 24 points, we can calculate how much it will move when\n // we scale up or down.\n baselineOffset = (lineMaxScale - section.scale) * ONE_EM;\n } else {\n const imagePosition = imagePositions[section.imageName];\n if (!imagePosition) continue;\n imageName = section.imageName;\n shaping.iconsInText = shaping.iconsInText || true;\n rect = imagePosition.paddedRect;\n const size = imagePosition.displaySize;\n // If needed, allow to set scale factor for an image using\n // alias \"image-scale\" that could be alias for \"font-scale\"\n // when FormattedSection is an image section.\n section.scale = section.scale * ONE_EM / layoutTextSizeThisZoom;\n\n metrics = {width: size[0],\n height: size[1],\n left: IMAGE_PADDING,\n top: -GLYPH_PBF_BORDER,\n advance: vertical ? size[1] : size[0]};\n\n // Difference between one EM and an image size.\n // Aligns bottom of an image to a baseline level.\n const imageOffset = ONE_EM - size[1] * section.scale;\n baselineOffset = maxLineOffset + imageOffset;\n verticalAdvance = metrics.advance;\n\n // Difference between height of an image and one EM at max line scale.\n // Pushes current line down if an image size is over 1 EM at max line scale.\n const offset = vertical ? size[0] * section.scale - ONE_EM * lineMaxScale :\n size[1] * section.scale - ONE_EM * lineMaxScale;\n if (offset > 0 && offset > lineOffset) {\n lineOffset = offset;\n }\n }\n\n if (!vertical) {\n positionedGlyphs.push({glyph: codePoint, imageName, x, y: y + baselineOffset, vertical, scale: section.scale, fontStack: section.fontStack, sectionIndex, metrics, rect});\n x += metrics.advance * section.scale + spacing;\n } else {\n shaping.verticalizable = true;\n positionedGlyphs.push({glyph: codePoint, imageName, x, y: y + baselineOffset, vertical, scale: section.scale, fontStack: section.fontStack, sectionIndex, metrics, rect});\n x += verticalAdvance * section.scale + spacing;\n }\n }\n\n // Only justify if we placed at least one glyph\n if (positionedGlyphs.length !== 0) {\n const lineLength = x - spacing;\n maxLineLength = Math.max(lineLength, maxLineLength);\n justifyLine(positionedGlyphs, 0, positionedGlyphs.length - 1, justify, lineOffset);\n }\n\n x = 0;\n const currentLineHeight = lineHeight * lineMaxScale + lineOffset;\n positionedLine.lineOffset = Math.max(lineOffset, maxLineOffset);\n y += currentLineHeight;\n maxLineHeight = Math.max(currentLineHeight, maxLineHeight);\n ++lineIndex;\n }\n\n // Calculate the bounding box and justify / align text block.\n const height = y - SHAPING_DEFAULT_OFFSET;\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(textAnchor);\n align(shaping.positionedLines, justify, horizontalAlign, verticalAlign, maxLineLength, maxLineHeight, lineHeight, height, lines.length);\n\n shaping.top += -verticalAlign * height;\n shaping.bottom = shaping.top + height;\n shaping.left += -horizontalAlign * maxLineLength;\n shaping.right = shaping.left + maxLineLength;\n}\n\n// justify right = 1, left = 0, center = 0.5\nfunction justifyLine(positionedGlyphs: Array,\n start: number,\n end: number,\n justify: 1 | 0 | 0.5,\n lineOffset: number) {\n if (!justify && !lineOffset)\n return;\n\n const lastPositionedGlyph = positionedGlyphs[end];\n const lastAdvance = lastPositionedGlyph.metrics.advance * lastPositionedGlyph.scale;\n const lineIndent = (positionedGlyphs[end].x + lastAdvance) * justify;\n\n for (let j = start; j <= end; j++) {\n positionedGlyphs[j].x -= lineIndent;\n positionedGlyphs[j].y += lineOffset;\n }\n}\n\nfunction align(positionedLines: Array,\n justify: number,\n horizontalAlign: number,\n verticalAlign: number,\n maxLineLength: number,\n maxLineHeight: number,\n lineHeight: number,\n blockHeight: number,\n lineCount: number) {\n const shiftX = (justify - horizontalAlign) * maxLineLength;\n let shiftY = 0;\n\n if (maxLineHeight !== lineHeight) {\n shiftY = -blockHeight * verticalAlign - SHAPING_DEFAULT_OFFSET;\n } else {\n shiftY = (-verticalAlign * lineCount + 0.5) * lineHeight;\n }\n\n for (const line of positionedLines) {\n for (const positionedGlyph of line.positionedGlyphs) {\n positionedGlyph.x += shiftX;\n positionedGlyph.y += shiftY;\n }\n }\n}\n\nexport type PositionedIcon = {\n image: ImagePosition;\n top: number;\n bottom: number;\n left: number;\n right: number;\n collisionPadding?: [number, number, number, number];\n};\n\nfunction shapeIcon(\n image: ImagePosition,\n iconOffset: [number, number],\n iconAnchor: SymbolAnchor\n): PositionedIcon {\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(iconAnchor);\n const dx = iconOffset[0];\n const dy = iconOffset[1];\n const x1 = dx - image.displaySize[0] * horizontalAlign;\n const x2 = x1 + image.displaySize[0];\n const y1 = dy - image.displaySize[1] * verticalAlign;\n const y2 = y1 + image.displaySize[1];\n return {image, top: y1, bottom: y2, left: x1, right: x2};\n}\n\nfunction fitIconToText(\n shapedIcon: PositionedIcon,\n shapedText: Shaping,\n textFit: string,\n padding: [number, number, number, number],\n iconOffset: [number, number],\n fontScale: number\n): PositionedIcon {\n\n const image = shapedIcon.image;\n\n let collisionPadding;\n if (image.content) {\n const content = image.content;\n const pixelRatio = image.pixelRatio || 1;\n collisionPadding = [\n content[0] / pixelRatio,\n content[1] / pixelRatio,\n image.displaySize[0] - content[2] / pixelRatio,\n image.displaySize[1] - content[3] / pixelRatio\n ];\n }\n\n // We don't respect the icon-anchor, because icon-text-fit is set. Instead,\n // the icon will be centered on the text, then stretched in the given\n // dimensions.\n\n const textLeft = shapedText.left * fontScale;\n const textRight = shapedText.right * fontScale;\n\n let top, right, bottom, left;\n if (textFit === 'width' || textFit === 'both') {\n // Stretched horizontally to the text width\n left = iconOffset[0] + textLeft - padding[3];\n right = iconOffset[0] + textRight + padding[1];\n } else {\n // Centered on the text\n left = iconOffset[0] + (textLeft + textRight - image.displaySize[0]) / 2;\n right = left + image.displaySize[0];\n }\n\n const textTop = shapedText.top * fontScale;\n const textBottom = shapedText.bottom * fontScale;\n if (textFit === 'height' || textFit === 'both') {\n // Stretched vertically to the text height\n top = iconOffset[1] + textTop - padding[0];\n bottom = iconOffset[1] + textBottom + padding[2];\n } else {\n // Centered on the text\n top = iconOffset[1] + (textTop + textBottom - image.displaySize[1]) / 2;\n bottom = top + image.displaySize[1];\n }\n\n return {image, top, right, bottom, left, collisionPadding};\n}\n","import {Interpolate, interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {clamp} from '../util/util';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\n\nimport type {PropertyValue, PossiblyEvaluatedPropertyValue} from '../style/properties';\nimport type {InterpolationType} from '@maplibre/maplibre-gl-style-spec';\n\nconst MAX_GLYPH_ICON_SIZE = 255;\nconst SIZE_PACK_FACTOR = 128;\nconst MAX_PACKED_SIZE = MAX_GLYPH_ICON_SIZE * SIZE_PACK_FACTOR;\n\nexport {getSizeData, evaluateSizeForFeature, evaluateSizeForZoom, SIZE_PACK_FACTOR, MAX_GLYPH_ICON_SIZE, MAX_PACKED_SIZE};\n\nexport type SizeData = {\n kind: 'constant';\n layoutSize: number;\n} | {\n kind: 'source';\n} | {\n kind: 'camera';\n minZoom: number;\n maxZoom: number;\n minSize: number;\n maxSize: number;\n interpolationType: InterpolationType;\n} | {\n kind: 'composite';\n minZoom: number;\n maxZoom: number;\n interpolationType: InterpolationType;\n};\n\nexport type EvaluatedZoomSize = {uSizeT: number; uSize: number};\n\n// For {text,icon}-size, get the bucket-level data that will be needed by\n// the painter to set symbol-size-related uniforms\nfunction getSizeData(\n tileZoom: number,\n value: PropertyValue>\n): SizeData {\n const {expression} = value;\n\n if (expression.kind === 'constant') {\n const layoutSize = expression.evaluate(new EvaluationParameters(tileZoom + 1));\n return {kind: 'constant', layoutSize};\n\n } else if (expression.kind === 'source') {\n return {kind: 'source'};\n\n } else {\n const {zoomStops, interpolationType} = expression;\n\n // calculate covering zoom stops for zoom-dependent values\n let lower = 0;\n while (lower < zoomStops.length && zoomStops[lower] <= tileZoom) lower++;\n lower = Math.max(0, lower - 1);\n let upper = lower;\n while (upper < zoomStops.length && zoomStops[upper] < tileZoom + 1) upper++;\n upper = Math.min(zoomStops.length - 1, upper);\n\n const minZoom = zoomStops[lower];\n const maxZoom = zoomStops[upper];\n\n // We'd like to be able to use CameraExpression or CompositeExpression in these\n // return types rather than ExpressionSpecification, but the former are not\n // transferrable across Web Worker boundaries.\n if (expression.kind === 'composite') {\n return {kind: 'composite', minZoom, maxZoom, interpolationType};\n }\n\n // for camera functions, also save off the function values\n // evaluated at the covering zoom levels\n const minSize = expression.evaluate(new EvaluationParameters(minZoom));\n const maxSize = expression.evaluate(new EvaluationParameters(maxZoom));\n\n return {kind: 'camera', minZoom, maxZoom, minSize, maxSize, interpolationType};\n }\n}\n\nfunction evaluateSizeForFeature(sizeData: SizeData,\n {\n uSize,\n uSizeT\n }: {\n uSize: number;\n uSizeT: number;\n },\n {\n lowerSize,\n upperSize\n }: {\n lowerSize: number;\n upperSize: number;\n }): number {\n if (sizeData.kind === 'source') {\n return lowerSize / SIZE_PACK_FACTOR;\n } else if (sizeData.kind === 'composite') {\n return interpolates.number(lowerSize / SIZE_PACK_FACTOR, upperSize / SIZE_PACK_FACTOR, uSizeT);\n }\n return uSize;\n}\n\nfunction evaluateSizeForZoom(sizeData: SizeData, zoom: number): EvaluatedZoomSize {\n let uSizeT = 0;\n let uSize = 0;\n\n if (sizeData.kind === 'constant') {\n uSize = sizeData.layoutSize;\n\n } else if (sizeData.kind !== 'source') {\n const {interpolationType, minZoom, maxZoom} = sizeData;\n\n // Even though we could get the exact value of the camera function\n // at z = tr.zoom, we intentionally do not: instead, we interpolate\n // between the camera function values at a pair of zoom stops covering\n // [tileZoom, tileZoom + 1] in order to be consistent with this\n // restriction on composite functions\n const t = !interpolationType ? 0 : clamp(\n Interpolate.interpolationFactor(interpolationType, zoom, minZoom, maxZoom), 0, 1);\n\n if (sizeData.kind === 'camera') {\n uSize = interpolates.number(sizeData.minSize, sizeData.maxSize, t);\n } else {\n uSizeT = t;\n }\n }\n\n return {uSizeT, uSize};\n}\n","import {SymbolLayoutPropsPossiblyEvaluated} from './symbol_style_layer_properties.g';\nimport type {SymbolLayoutProps} from './symbol_style_layer_properties.g';\nimport {PossiblyEvaluated} from '../properties';\n\n/**\n * The overlap mode for properties like `icon-overlap`and `text-overlap`\n */\nexport type OverlapMode = 'never' | 'always' | 'cooperative';\n\nexport function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'icon-overlap', allowOverlapProp: 'icon-allow-overlap'): OverlapMode;\nexport function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'text-overlap', allowOverlapProp: 'text-allow-overlap'): OverlapMode;\nexport function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'icon-overlap' | 'text-overlap', allowOverlapProp: 'icon-allow-overlap' | 'text-allow-overlap'): OverlapMode {\n let result: OverlapMode = 'never';\n const overlap = layout.get(overlapProp);\n\n if (overlap) {\n // if -overlap is set, use it\n result = overlap;\n } else if (layout.get(allowOverlapProp)) {\n // fall back to -allow-overlap, with false='never', true='always'\n result = 'always';\n }\n\n return result;\n}\n","import {\n symbolLayoutAttributes,\n collisionVertexAttributes,\n collisionBoxLayout,\n dynamicLayoutAttributes,\n} from './symbol_attributes';\n\nimport {SymbolLayoutArray,\n SymbolDynamicLayoutArray,\n SymbolOpacityArray,\n CollisionBoxLayoutArray,\n CollisionVertexArray,\n PlacedSymbolArray,\n SymbolInstanceArray,\n GlyphOffsetArray,\n SymbolLineVertexArray,\n TextAnchorOffsetArray\n} from '../array_types.g';\n\nimport Point from '@mapbox/point-geometry';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray, LineIndexArray} from '../index_array_type';\nimport {transformText} from '../../symbol/transform_text';\nimport {mergeLines} from '../../symbol/merge_lines';\nimport {allowsVerticalWritingMode, stringContainsRTLText} from '../../util/script_detection';\nimport {WritingMode} from '../../symbol/shaping';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport mvt from '@mapbox/vector-tile';\nconst vectorTileFeatureTypes = mvt.VectorTileFeature.types;\nimport {verticalizedCharacterMap} from '../../util/verticalize_punctuation';\nimport {Anchor} from '../../symbol/anchor';\nimport {getSizeData, MAX_PACKED_SIZE} from '../../symbol/symbol_size';\n\nimport {register} from '../../util/web_worker_transfer';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\nimport {Formatted, ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport {plugin as globalRTLTextPlugin, getRTLTextPluginStatus} from '../../source/rtl_text_plugin';\nimport {mat4} from 'gl-matrix';\nimport {getOverlapMode} from '../../style/style_layer/overlap_mode';\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {CollisionBoxArray, CollisionBox, SymbolInstance} from '../array_types.g';\nimport type {StructArray, StructArrayMember, ViewType} from '../../util/struct_array';\nimport type {SymbolStyleLayer} from '../../style/style_layer/symbol_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type {SymbolQuad} from '../../symbol/quads';\nimport type {SizeData} from '../../symbol/symbol_size';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nexport type SingleCollisionBox = {\n x1: number;\n y1: number;\n x2: number;\n y2: number;\n anchorPointX: number;\n anchorPointY: number;\n};\n\nexport type CollisionArrays = {\n textBox?: SingleCollisionBox;\n verticalTextBox?: SingleCollisionBox;\n iconBox?: SingleCollisionBox;\n verticalIconBox?: SingleCollisionBox;\n textFeatureIndex?: number;\n verticalTextFeatureIndex?: number;\n iconFeatureIndex?: number;\n verticalIconFeatureIndex?: number;\n};\n\nexport type SymbolFeature = {\n sortKey: number | void;\n text: Formatted | void;\n icon: ResolvedImage;\n index: number;\n sourceLayerIndex: number;\n geometry: Array>;\n properties: any;\n type: 'Unknown' | 'Point' | 'LineString' | 'Polygon';\n id?: any;\n};\n\nexport type SortKeyRange = {\n sortKey: number;\n symbolInstanceStart: number;\n symbolInstanceEnd: number;\n};\n\n// Opacity arrays are frequently updated but don't contain a lot of information, so we pack them\n// tight. Each Uint32 is actually four duplicate Uint8s for the four corners of a glyph\n// 7 bits are for the current opacity, and the lowest bit is the target opacity\n\n// actually defined in symbol_attributes.js\n// const placementOpacityAttributes = [\n// { name: 'a_fade_opacity', components: 1, type: 'Uint32' }\n// ];\nconst shaderOpacityAttributes = [\n {name: 'a_fade_opacity', components: 1, type: 'Uint8' as ViewType, offset: 0}\n];\n\nfunction addVertex(\n array: StructArray,\n anchorX: number,\n anchorY: number,\n ox: number,\n oy: number,\n tx: number,\n ty: number,\n sizeVertex: number,\n isSDF: boolean,\n pixelOffsetX: number,\n pixelOffsetY: number,\n minFontScaleX: number,\n minFontScaleY: number\n) {\n const aSizeX = sizeVertex ? Math.min(MAX_PACKED_SIZE, Math.round(sizeVertex[0])) : 0;\n const aSizeY = sizeVertex ? Math.min(MAX_PACKED_SIZE, Math.round(sizeVertex[1])) : 0;\n array.emplaceBack(\n // a_pos_offset\n anchorX,\n anchorY,\n Math.round(ox * 32),\n Math.round(oy * 32),\n\n // a_data\n tx, // x coordinate of symbol on glyph atlas texture\n ty, // y coordinate of symbol on glyph atlas texture\n (aSizeX << 1) + (isSDF ? 1 : 0),\n aSizeY,\n pixelOffsetX * 16,\n pixelOffsetY * 16,\n minFontScaleX * 256,\n minFontScaleY * 256\n );\n}\n\nfunction addDynamicAttributes(dynamicLayoutVertexArray: StructArray, p: Point, angle: number) {\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n}\n\nfunction containsRTLText(formattedText: Formatted): boolean {\n for (const section of formattedText.sections) {\n if (stringContainsRTLText(section.text)) {\n return true;\n }\n }\n return false;\n}\n\nexport class SymbolBuffers {\n layoutVertexArray: SymbolLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n\n dynamicLayoutVertexArray: SymbolDynamicLayoutArray;\n dynamicLayoutVertexBuffer: VertexBuffer;\n\n opacityVertexArray: SymbolOpacityArray;\n opacityVertexBuffer: VertexBuffer;\n hasVisibleVertices: boolean;\n\n collisionVertexArray: CollisionVertexArray;\n collisionVertexBuffer: VertexBuffer;\n\n placedSymbolArray: PlacedSymbolArray;\n\n constructor(programConfigurations: ProgramConfigurationSet) {\n this.layoutVertexArray = new SymbolLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.programConfigurations = programConfigurations;\n this.segments = new SegmentVector();\n this.dynamicLayoutVertexArray = new SymbolDynamicLayoutArray();\n this.opacityVertexArray = new SymbolOpacityArray();\n this.hasVisibleVertices = false;\n this.placedSymbolArray = new PlacedSymbolArray();\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0 &&\n this.indexArray.length === 0 &&\n this.dynamicLayoutVertexArray.length === 0 &&\n this.opacityVertexArray.length === 0;\n }\n\n upload(context: Context, dynamicIndexBuffer: boolean, upload?: boolean, update?: boolean) {\n if (this.isEmpty()) {\n return;\n }\n\n if (upload) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, symbolLayoutAttributes.members);\n this.indexBuffer = context.createIndexBuffer(this.indexArray, dynamicIndexBuffer);\n this.dynamicLayoutVertexBuffer = context.createVertexBuffer(this.dynamicLayoutVertexArray, dynamicLayoutAttributes.members, true);\n this.opacityVertexBuffer = context.createVertexBuffer(this.opacityVertexArray, shaderOpacityAttributes, true);\n // This is a performance hack so that we can write to opacityVertexArray with uint32s\n // even though the shaders read uint8s\n this.opacityVertexBuffer.itemSize = 1;\n }\n if (upload || update) {\n this.programConfigurations.upload(context);\n }\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n this.dynamicLayoutVertexBuffer.destroy();\n this.opacityVertexBuffer.destroy();\n }\n}\n\nregister('SymbolBuffers', SymbolBuffers);\n\nclass CollisionBuffers {\n layoutVertexArray: StructArray;\n layoutAttributes: Array;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray | LineIndexArray;\n indexBuffer: IndexBuffer;\n\n segments: SegmentVector;\n\n collisionVertexArray: CollisionVertexArray;\n collisionVertexBuffer: VertexBuffer;\n\n constructor(LayoutArray: {\n new (...args: any): StructArray;\n },\n layoutAttributes: Array,\n IndexArray: {\n new (...args: any): TriangleIndexArray | LineIndexArray;\n }) {\n this.layoutVertexArray = new LayoutArray();\n this.layoutAttributes = layoutAttributes;\n this.indexArray = new IndexArray();\n this.segments = new SegmentVector();\n this.collisionVertexArray = new CollisionVertexArray();\n }\n\n upload(context: Context) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, this.layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n this.collisionVertexBuffer = context.createVertexBuffer(this.collisionVertexArray, collisionVertexAttributes.members, true);\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.segments.destroy();\n this.collisionVertexBuffer.destroy();\n }\n}\n\nregister('CollisionBuffers', CollisionBuffers);\n\n/**\n * @internal\n * Unlike other buckets, which simply implement #addFeature with type-specific\n * logic for (essentially) triangulating feature geometries, SymbolBucket\n * requires specialized behavior:\n *\n * 1. WorkerTile#parse(), the logical owner of the bucket creation process,\n * calls SymbolBucket#populate(), which resolves text and icon tokens on\n * each feature, adds each glyphs and symbols needed to the passed-in\n * collections options.glyphDependencies and options.iconDependencies, and\n * stores the feature data for use in subsequent step (this.features).\n *\n * 2. WorkerTile asynchronously requests from the main thread all of the glyphs\n * and icons needed (by this bucket and any others). When glyphs and icons\n * have been received, the WorkerTile creates a CollisionIndex and invokes:\n *\n * 3. performSymbolLayout(bucket, stacks, icons) perform texts shaping and\n * layout on a Symbol Bucket. This step populates:\n * `this.symbolInstances`: metadata on generated symbols\n * `this.collisionBoxArray`: collision data for use by foreground\n * `this.text`: SymbolBuffers for text symbols\n * `this.icons`: SymbolBuffers for icons\n * `this.iconCollisionBox`: Debug SymbolBuffers for icon collision boxes\n * `this.textCollisionBox`: Debug SymbolBuffers for text collision boxes\n * The results are sent to the foreground for rendering\n *\n * 4. performSymbolPlacement(bucket, collisionIndex) is run on the foreground,\n * and uses the CollisionIndex along with current camera settings to determine\n * which symbols can actually show on the map. Collided symbols are hidden\n * using a dynamic \"OpacityVertexArray\".\n */\nexport class SymbolBucket implements Bucket {\n static MAX_GLYPHS: number;\n static addDynamicAttributes: typeof addDynamicAttributes;\n\n collisionBoxArray: CollisionBoxArray;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n\n index: number;\n sdfIcons: boolean;\n iconsInText: boolean;\n iconsNeedLinear: boolean;\n bucketInstanceId: number;\n justReloaded: boolean;\n hasPattern: boolean;\n\n textSizeData: SizeData;\n iconSizeData: SizeData;\n\n glyphOffsetArray: GlyphOffsetArray;\n lineVertexArray: SymbolLineVertexArray;\n features: Array;\n symbolInstances: SymbolInstanceArray;\n textAnchorOffsets: TextAnchorOffsetArray;\n collisionArrays: Array;\n sortKeyRanges: Array;\n pixelRatio: number;\n tilePixelRatio: number;\n compareText: {[_: string]: Array};\n fadeStartTime: number;\n sortFeaturesByKey: boolean;\n sortFeaturesByY: boolean;\n canOverlap: boolean;\n sortedAngle: number;\n featureSortOrder: Array;\n\n collisionCircleArray: Array;\n placementInvProjMatrix: mat4;\n placementViewportMatrix: mat4;\n\n text: SymbolBuffers;\n icon: SymbolBuffers;\n textCollisionBox: CollisionBuffers;\n iconCollisionBox: CollisionBuffers;\n uploaded: boolean;\n sourceLayerIndex: number;\n sourceID: string;\n symbolInstanceIndexes: Array;\n writingModes: WritingMode[];\n allowVerticalPlacement: boolean;\n hasRTLText: boolean;\n\n constructor(options: BucketParameters) {\n this.collisionBoxArray = options.collisionBoxArray;\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.pixelRatio = options.pixelRatio;\n this.sourceLayerIndex = options.sourceLayerIndex;\n this.hasPattern = false;\n this.hasRTLText = false;\n this.sortKeyRanges = [];\n\n this.collisionCircleArray = [];\n this.placementInvProjMatrix = mat4.identity([] as any);\n this.placementViewportMatrix = mat4.identity([] as any);\n\n const layer = this.layers[0];\n const unevaluatedLayoutValues = layer._unevaluatedLayout._values;\n\n this.textSizeData = getSizeData(this.zoom, unevaluatedLayoutValues['text-size']);\n this.iconSizeData = getSizeData(this.zoom, unevaluatedLayoutValues['icon-size']);\n\n const layout = this.layers[0].layout;\n const sortKey = layout.get('symbol-sort-key');\n const zOrder = layout.get('symbol-z-order');\n this.canOverlap =\n getOverlapMode(layout, 'text-overlap', 'text-allow-overlap') !== 'never' ||\n getOverlapMode(layout, 'icon-overlap', 'icon-allow-overlap') !== 'never' ||\n layout.get('text-ignore-placement') ||\n layout.get('icon-ignore-placement');\n this.sortFeaturesByKey = zOrder !== 'viewport-y' && !sortKey.isConstant();\n const zOrderByViewportY = zOrder === 'viewport-y' || (zOrder === 'auto' && !this.sortFeaturesByKey);\n this.sortFeaturesByY = zOrderByViewportY && this.canOverlap;\n\n if (layout.get('symbol-placement') === 'point') {\n this.writingModes = layout.get('text-writing-mode').map(wm => WritingMode[wm]);\n }\n\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n\n this.sourceID = options.sourceID;\n }\n\n createArrays() {\n this.text = new SymbolBuffers(new ProgramConfigurationSet(this.layers, this.zoom, property => /^text/.test(property)));\n this.icon = new SymbolBuffers(new ProgramConfigurationSet(this.layers, this.zoom, property => /^icon/.test(property)));\n\n this.glyphOffsetArray = new GlyphOffsetArray();\n this.lineVertexArray = new SymbolLineVertexArray();\n this.symbolInstances = new SymbolInstanceArray();\n this.textAnchorOffsets = new TextAnchorOffsetArray();\n }\n\n calculateGlyphDependencies(text: string, stack: {[_: number]: boolean}, textAlongLine: boolean, allowVerticalPlacement: boolean, doesAllowVerticalWritingMode: boolean) {\n for (let i = 0; i < text.length; i++) {\n stack[text.charCodeAt(i)] = true;\n if ((textAlongLine || allowVerticalPlacement) && doesAllowVerticalWritingMode) {\n const verticalChar = verticalizedCharacterMap[text.charAt(i)];\n if (verticalChar) {\n stack[verticalChar.charCodeAt(0)] = true;\n }\n }\n }\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n const layer = this.layers[0];\n const layout = layer.layout;\n\n const textFont = layout.get('text-font');\n const textField = layout.get('text-field');\n const iconImage = layout.get('icon-image');\n const hasText =\n (textField.value.kind !== 'constant' ||\n (textField.value.value instanceof Formatted && !textField.value.value.isEmpty()) ||\n textField.value.value.toString().length > 0) &&\n (textFont.value.kind !== 'constant' || textFont.value.value.length > 0);\n // we should always resolve the icon-image value if the property was defined in the style\n // this allows us to fire the styleimagemissing event if image evaluation returns null\n // the only way to distinguish between null returned from a coalesce statement with no valid images\n // and null returned because icon-image wasn't defined is to check whether or not iconImage.parameters is an empty object\n const hasIcon = iconImage.value.kind !== 'constant' || !!iconImage.value.value || Object.keys(iconImage.parameters).length > 0;\n const symbolSortKey = layout.get('symbol-sort-key');\n\n this.features = [];\n\n if (!hasText && !hasIcon) {\n return;\n }\n\n const icons = options.iconDependencies;\n const stacks = options.glyphDependencies;\n const availableImages = options.availableImages;\n const globalProperties = new EvaluationParameters(this.zoom);\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n\n const needGeometry = layer._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n if (!layer._featureFilter.filter(globalProperties, evaluationFeature, canonical)) {\n continue;\n }\n\n if (!needGeometry) evaluationFeature.geometry = loadGeometry(feature);\n\n let text: Formatted | void;\n if (hasText) {\n // Expression evaluation will automatically coerce to Formatted\n // but plain string token evaluation skips that pathway so do the\n // conversion here.\n const resolvedTokens = layer.getValueAndResolveTokens('text-field', evaluationFeature, canonical, availableImages);\n const formattedText = Formatted.factory(resolvedTokens);\n if (containsRTLText(formattedText)) {\n this.hasRTLText = true;\n }\n if (\n !this.hasRTLText || // non-rtl text so can proceed safely\n getRTLTextPluginStatus() === 'unavailable' || // We don't intend to lazy-load the rtl text plugin, so proceed with incorrect shaping\n this.hasRTLText && globalRTLTextPlugin.isParsed() // Use the rtlText plugin to shape text\n ) {\n text = transformText(formattedText, layer, evaluationFeature);\n }\n }\n\n let icon: ResolvedImage;\n if (hasIcon) {\n // Expression evaluation will automatically coerce to Image\n // but plain string token evaluation skips that pathway so do the\n // conversion here.\n const resolvedTokens = layer.getValueAndResolveTokens('icon-image', evaluationFeature, canonical, availableImages);\n if (resolvedTokens instanceof ResolvedImage) {\n icon = resolvedTokens;\n } else {\n icon = ResolvedImage.fromString(resolvedTokens);\n }\n }\n\n if (!text && !icon) {\n continue;\n }\n const sortKey = this.sortFeaturesByKey ?\n symbolSortKey.evaluate(evaluationFeature, {}, canonical) :\n undefined;\n\n const symbolFeature: SymbolFeature = {\n id,\n text,\n icon,\n index,\n sourceLayerIndex,\n geometry: evaluationFeature.geometry,\n properties: feature.properties,\n type: vectorTileFeatureTypes[feature.type],\n sortKey\n };\n this.features.push(symbolFeature);\n\n if (icon) {\n icons[icon.name] = true;\n }\n\n if (text) {\n const fontStack = textFont.evaluate(evaluationFeature, {}, canonical).join(',');\n const textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point';\n this.allowVerticalPlacement = this.writingModes && this.writingModes.indexOf(WritingMode.vertical) >= 0;\n for (const section of text.sections) {\n if (!section.image) {\n const doesAllowVerticalWritingMode = allowsVerticalWritingMode(text.toString());\n const sectionFont = section.fontStack || fontStack;\n const sectionStack = stacks[sectionFont] = stacks[sectionFont] || {};\n this.calculateGlyphDependencies(section.text, sectionStack, textAlongLine, this.allowVerticalPlacement, doesAllowVerticalWritingMode);\n } else {\n // Add section image to the list of dependencies.\n icons[section.image.name] = true;\n }\n }\n }\n }\n\n if (layout.get('symbol-placement') === 'line') {\n // Merge adjacent lines with the same text to improve labelling.\n // It's better to place labels on one long line than on many short segments.\n this.features = mergeLines(this.features);\n }\n\n if (this.sortFeaturesByKey) {\n this.features.sort((a, b) => {\n // a.sortKey is always a number when sortFeaturesByKey is true\n return (a.sortKey as number) - (b.sortKey as number);\n });\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.text.programConfigurations.updatePaintArrays(states, vtLayer, this.layers, imagePositions);\n this.icon.programConfigurations.updatePaintArrays(states, vtLayer, this.layers, imagePositions);\n }\n\n isEmpty() {\n // When the bucket encounters only rtl-text but the plugin isnt loaded, no symbol instances will be created.\n // In order for the bucket to be serialized, and not discarded as an empty bucket both checks are necessary.\n return this.symbolInstances.length === 0 && !this.hasRTLText;\n }\n\n uploadPending() {\n return !this.uploaded || this.text.programConfigurations.needsUpload || this.icon.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded && this.hasDebugData()) {\n this.textCollisionBox.upload(context);\n this.iconCollisionBox.upload(context);\n }\n this.text.upload(context, this.sortFeaturesByY, !this.uploaded, this.text.programConfigurations.needsUpload);\n this.icon.upload(context, this.sortFeaturesByY, !this.uploaded, this.icon.programConfigurations.needsUpload);\n this.uploaded = true;\n }\n\n destroyDebugData() {\n this.textCollisionBox.destroy();\n this.iconCollisionBox.destroy();\n }\n\n destroy() {\n this.text.destroy();\n this.icon.destroy();\n\n if (this.hasDebugData()) {\n this.destroyDebugData();\n }\n }\n\n addToLineVertexArray(anchor: Anchor, line: any) {\n const lineStartIndex = this.lineVertexArray.length;\n if (anchor.segment !== undefined) {\n let sumForwardLength = anchor.dist(line[anchor.segment + 1]);\n let sumBackwardLength = anchor.dist(line[anchor.segment]);\n const vertices = {};\n for (let i = anchor.segment + 1; i < line.length; i++) {\n vertices[i] = {x: line[i].x, y: line[i].y, tileUnitDistanceFromAnchor: sumForwardLength};\n if (i < line.length - 1) {\n sumForwardLength += line[i + 1].dist(line[i]);\n }\n }\n for (let i = anchor.segment || 0; i >= 0; i--) {\n vertices[i] = {x: line[i].x, y: line[i].y, tileUnitDistanceFromAnchor: sumBackwardLength};\n if (i > 0) {\n sumBackwardLength += line[i - 1].dist(line[i]);\n }\n }\n for (let i = 0; i < line.length; i++) {\n const vertex = vertices[i];\n this.lineVertexArray.emplaceBack(vertex.x, vertex.y, vertex.tileUnitDistanceFromAnchor);\n }\n }\n return {\n lineStartIndex,\n lineLength: this.lineVertexArray.length - lineStartIndex\n };\n }\n\n addSymbols(arrays: SymbolBuffers,\n quads: Array,\n sizeVertex: any,\n lineOffset: [number, number],\n alongLine: boolean,\n feature: SymbolFeature,\n writingMode: WritingMode,\n labelAnchor: Anchor,\n lineStartIndex: number,\n lineLength: number,\n associatedIconIndex: number,\n canonical: CanonicalTileID) {\n const indexArray = arrays.indexArray;\n const layoutVertexArray = arrays.layoutVertexArray;\n\n const segment = arrays.segments.prepareSegment(4 * quads.length, layoutVertexArray, indexArray, this.canOverlap ? feature.sortKey as number : undefined);\n const glyphOffsetArrayStart = this.glyphOffsetArray.length;\n const vertexStartIndex = segment.vertexLength;\n\n const angle = (this.allowVerticalPlacement && writingMode === WritingMode.vertical) ? Math.PI / 2 : 0;\n\n const sections = feature.text && feature.text.sections;\n\n for (let i = 0; i < quads.length; i++) {\n const {tl, tr, bl, br, tex, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY, glyphOffset, isSDF, sectionIndex} = quads[i];\n const index = segment.vertexLength;\n\n const y = glyphOffset[1];\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, tl.x, y + tl.y, tex.x, tex.y, sizeVertex, isSDF, pixelOffsetTL.x, pixelOffsetTL.y, minFontScaleX, minFontScaleY);\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, tr.x, y + tr.y, tex.x + tex.w, tex.y, sizeVertex, isSDF, pixelOffsetBR.x, pixelOffsetTL.y, minFontScaleX, minFontScaleY);\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, bl.x, y + bl.y, tex.x, tex.y + tex.h, sizeVertex, isSDF, pixelOffsetTL.x, pixelOffsetBR.y, minFontScaleX, minFontScaleY);\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, br.x, y + br.y, tex.x + tex.w, tex.y + tex.h, sizeVertex, isSDF, pixelOffsetBR.x, pixelOffsetBR.y, minFontScaleX, minFontScaleY);\n\n addDynamicAttributes(arrays.dynamicLayoutVertexArray, labelAnchor, angle);\n\n indexArray.emplaceBack(index, index + 1, index + 2);\n indexArray.emplaceBack(index + 1, index + 2, index + 3);\n\n segment.vertexLength += 4;\n segment.primitiveLength += 2;\n\n this.glyphOffsetArray.emplaceBack(glyphOffset[0]);\n\n if (i === quads.length - 1 || sectionIndex !== quads[i + 1].sectionIndex) {\n arrays.programConfigurations.populatePaintArrays(layoutVertexArray.length, feature, feature.index, {}, canonical, sections && sections[sectionIndex]);\n }\n }\n\n arrays.placedSymbolArray.emplaceBack(\n labelAnchor.x, labelAnchor.y,\n glyphOffsetArrayStart,\n this.glyphOffsetArray.length - glyphOffsetArrayStart,\n vertexStartIndex,\n lineStartIndex,\n lineLength,\n labelAnchor.segment,\n sizeVertex ? sizeVertex[0] : 0,\n sizeVertex ? sizeVertex[1] : 0,\n lineOffset[0], lineOffset[1],\n writingMode,\n // placedOrientation is null initially; will be updated to horizontal(1)/vertical(2) if placed\n 0,\n false as unknown as number,\n // The crossTileID is only filled/used on the foreground for dynamic text anchors\n 0,\n associatedIconIndex\n );\n }\n\n _addCollisionDebugVertex(layoutVertexArray: StructArray, collisionVertexArray: StructArray, point: Point, anchorX: number, anchorY: number, extrude: Point) {\n collisionVertexArray.emplaceBack(0, 0);\n return layoutVertexArray.emplaceBack(\n // pos\n point.x,\n point.y,\n // a_anchor_pos\n anchorX,\n anchorY,\n // extrude\n Math.round(extrude.x),\n Math.round(extrude.y));\n }\n\n addCollisionDebugVertices(x1: number, y1: number, x2: number, y2: number, arrays: CollisionBuffers, boxAnchorPoint: Point, symbolInstance: SymbolInstance) {\n const segment = arrays.segments.prepareSegment(4, arrays.layoutVertexArray, arrays.indexArray);\n const index = segment.vertexLength;\n\n const layoutVertexArray = arrays.layoutVertexArray;\n const collisionVertexArray = arrays.collisionVertexArray;\n\n const anchorX = symbolInstance.anchorX;\n const anchorY = symbolInstance.anchorY;\n\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x1, y1));\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x2, y1));\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x2, y2));\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x1, y2));\n\n segment.vertexLength += 4;\n\n const indexArray = arrays.indexArray as LineIndexArray;\n indexArray.emplaceBack(index, index + 1);\n indexArray.emplaceBack(index + 1, index + 2);\n indexArray.emplaceBack(index + 2, index + 3);\n indexArray.emplaceBack(index + 3, index);\n\n segment.primitiveLength += 4;\n }\n\n addDebugCollisionBoxes(startIndex: number, endIndex: number, symbolInstance: SymbolInstance, isText: boolean) {\n for (let b = startIndex; b < endIndex; b++) {\n const box: CollisionBox = this.collisionBoxArray.get(b);\n const x1 = box.x1;\n const y1 = box.y1;\n const x2 = box.x2;\n const y2 = box.y2;\n\n this.addCollisionDebugVertices(x1, y1, x2, y2,\n isText ? this.textCollisionBox : this.iconCollisionBox,\n box.anchorPoint, symbolInstance);\n }\n }\n\n generateCollisionDebugBuffers() {\n if (this.hasDebugData()) {\n this.destroyDebugData();\n }\n\n this.textCollisionBox = new CollisionBuffers(CollisionBoxLayoutArray, collisionBoxLayout.members, LineIndexArray);\n this.iconCollisionBox = new CollisionBuffers(CollisionBoxLayoutArray, collisionBoxLayout.members, LineIndexArray);\n\n for (let i = 0; i < this.symbolInstances.length; i++) {\n const symbolInstance = this.symbolInstances.get(i);\n this.addDebugCollisionBoxes(symbolInstance.textBoxStartIndex, symbolInstance.textBoxEndIndex, symbolInstance, true);\n this.addDebugCollisionBoxes(symbolInstance.verticalTextBoxStartIndex, symbolInstance.verticalTextBoxEndIndex, symbolInstance, true);\n this.addDebugCollisionBoxes(symbolInstance.iconBoxStartIndex, symbolInstance.iconBoxEndIndex, symbolInstance, false);\n this.addDebugCollisionBoxes(symbolInstance.verticalIconBoxStartIndex, symbolInstance.verticalIconBoxEndIndex, symbolInstance, false);\n }\n }\n\n // These flat arrays are meant to be quicker to iterate over than the source\n // CollisionBoxArray\n _deserializeCollisionBoxesForSymbol(\n collisionBoxArray: CollisionBoxArray,\n textStartIndex: number,\n textEndIndex: number,\n verticalTextStartIndex: number,\n verticalTextEndIndex: number,\n iconStartIndex: number,\n iconEndIndex: number,\n verticalIconStartIndex: number,\n verticalIconEndIndex: number\n ): CollisionArrays {\n\n const collisionArrays = {} as CollisionArrays;\n for (let k = textStartIndex; k < textEndIndex; k++) {\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.textBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.textFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n for (let k = verticalTextStartIndex; k < verticalTextEndIndex; k++) {\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.verticalTextBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.verticalTextFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n for (let k = iconStartIndex; k < iconEndIndex; k++) {\n // An icon can only have one box now, so this indexing is a bit vestigial...\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.iconBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.iconFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n for (let k = verticalIconStartIndex; k < verticalIconEndIndex; k++) {\n // An icon can only have one box now, so this indexing is a bit vestigial...\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.verticalIconBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.verticalIconFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n return collisionArrays;\n }\n\n deserializeCollisionBoxes(collisionBoxArray: CollisionBoxArray) {\n this.collisionArrays = [];\n for (let i = 0; i < this.symbolInstances.length; i++) {\n const symbolInstance = this.symbolInstances.get(i);\n this.collisionArrays.push(this._deserializeCollisionBoxesForSymbol(\n collisionBoxArray,\n symbolInstance.textBoxStartIndex,\n symbolInstance.textBoxEndIndex,\n symbolInstance.verticalTextBoxStartIndex,\n symbolInstance.verticalTextBoxEndIndex,\n symbolInstance.iconBoxStartIndex,\n symbolInstance.iconBoxEndIndex,\n symbolInstance.verticalIconBoxStartIndex,\n symbolInstance.verticalIconBoxEndIndex\n ));\n }\n }\n\n hasTextData() {\n return this.text.segments.get().length > 0;\n }\n\n hasIconData() {\n return this.icon.segments.get().length > 0;\n }\n\n hasDebugData() {\n return this.textCollisionBox && this.iconCollisionBox;\n }\n\n hasTextCollisionBoxData() {\n return this.hasDebugData() && this.textCollisionBox.segments.get().length > 0;\n }\n\n hasIconCollisionBoxData() {\n return this.hasDebugData() && this.iconCollisionBox.segments.get().length > 0;\n }\n\n addIndicesForPlacedSymbol(iconOrText: SymbolBuffers, placedSymbolIndex: number) {\n const placedSymbol = iconOrText.placedSymbolArray.get(placedSymbolIndex);\n\n const endIndex = placedSymbol.vertexStartIndex + placedSymbol.numGlyphs * 4;\n for (let vertexIndex = placedSymbol.vertexStartIndex; vertexIndex < endIndex; vertexIndex += 4) {\n iconOrText.indexArray.emplaceBack(vertexIndex, vertexIndex + 1, vertexIndex + 2);\n iconOrText.indexArray.emplaceBack(vertexIndex + 1, vertexIndex + 2, vertexIndex + 3);\n }\n }\n\n getSortedSymbolIndexes(angle: number) {\n if (this.sortedAngle === angle && this.symbolInstanceIndexes !== undefined) {\n return this.symbolInstanceIndexes;\n }\n const sin = Math.sin(angle);\n const cos = Math.cos(angle);\n const rotatedYs = [];\n const featureIndexes = [];\n const result = [];\n\n for (let i = 0; i < this.symbolInstances.length; ++i) {\n result.push(i);\n const symbolInstance = this.symbolInstances.get(i);\n rotatedYs.push(Math.round(sin * symbolInstance.anchorX + cos * symbolInstance.anchorY) | 0);\n featureIndexes.push(symbolInstance.featureIndex);\n }\n\n result.sort((aIndex, bIndex) => {\n return (rotatedYs[aIndex] - rotatedYs[bIndex]) ||\n (featureIndexes[bIndex] - featureIndexes[aIndex]);\n });\n\n return result;\n }\n\n addToSortKeyRanges(symbolInstanceIndex: number, sortKey: number) {\n const last = this.sortKeyRanges[this.sortKeyRanges.length - 1];\n if (last && last.sortKey === sortKey) {\n last.symbolInstanceEnd = symbolInstanceIndex + 1;\n } else {\n this.sortKeyRanges.push({\n sortKey,\n symbolInstanceStart: symbolInstanceIndex,\n symbolInstanceEnd: symbolInstanceIndex + 1\n });\n }\n }\n\n sortFeatures(angle: number) {\n if (!this.sortFeaturesByY) return;\n if (this.sortedAngle === angle) return;\n\n // The current approach to sorting doesn't sort across segments so don't try.\n // Sorting within segments separately seemed not to be worth the complexity.\n if (this.text.segments.get().length > 1 || this.icon.segments.get().length > 1) return;\n\n // If the symbols are allowed to overlap sort them by their vertical screen position.\n // The index array buffer is rewritten to reference the (unchanged) vertices in the\n // sorted order.\n\n // To avoid sorting the actual symbolInstance array we sort an array of indexes.\n this.symbolInstanceIndexes = this.getSortedSymbolIndexes(angle);\n this.sortedAngle = angle;\n\n this.text.indexArray.clear();\n this.icon.indexArray.clear();\n\n this.featureSortOrder = [];\n\n for (const i of this.symbolInstanceIndexes) {\n const symbolInstance = this.symbolInstances.get(i);\n this.featureSortOrder.push(symbolInstance.featureIndex);\n\n [\n symbolInstance.rightJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.leftJustifiedTextSymbolIndex\n ].forEach((index, i, array) => {\n // Only add a given index the first time it shows up,\n // to avoid duplicate opacity entries when multiple justifications\n // share the same glyphs.\n if (index >= 0 && array.indexOf(index) === i) {\n this.addIndicesForPlacedSymbol(this.text, index);\n }\n });\n\n if (symbolInstance.verticalPlacedTextSymbolIndex >= 0) {\n this.addIndicesForPlacedSymbol(this.text, symbolInstance.verticalPlacedTextSymbolIndex);\n }\n\n if (symbolInstance.placedIconSymbolIndex >= 0) {\n this.addIndicesForPlacedSymbol(this.icon, symbolInstance.placedIconSymbolIndex);\n }\n\n if (symbolInstance.verticalPlacedIconSymbolIndex >= 0) {\n this.addIndicesForPlacedSymbol(this.icon, symbolInstance.verticalPlacedIconSymbolIndex);\n }\n }\n\n if (this.text.indexBuffer) this.text.indexBuffer.updateData(this.text.indexArray);\n if (this.icon.indexBuffer) this.icon.indexBuffer.updateData(this.icon.indexArray);\n }\n}\n\nregister('SymbolBucket', SymbolBucket, {\n omit: ['layers', 'collisionBoxArray', 'features', 'compareText']\n});\n\n// this constant is based on the size of StructArray indexes used in a symbol\n// bucket--namely, glyphOffsetArrayStart\n// eg the max valid UInt16 is 65,535\n// See https://github.com/mapbox/mapbox-gl-js/issues/2907 for motivation\n// lineStartIndex and textBoxStartIndex could potentially be concerns\n// but we expect there to be many fewer boxes/lines than glyphs\nSymbolBucket.MAX_GLYPHS = 65535;\n\nSymbolBucket.addDynamicAttributes = addDynamicAttributes;\n\nexport {addDynamicAttributes};\n","/**\n * Replace tokens in a string template with values in an object\n *\n * @param properties - a key/value relationship between tokens and replacements\n * @param text - the template string\n * @returns the template with tokens replaced\n */\nexport function resolveTokens(\n properties: {\n readonly [x: string]: unknown;\n } | null,\n text: string\n): string {\n return text.replace(/{([^{}]+)}/g, (match, key: string) => {\n return properties && key in properties ? String(properties[key]) : '';\n });\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n ColorType\n } from '@maplibre/maplibre-gl-style-spec';\n \nexport type SymbolLayoutProps = {\n \"symbol-placement\": DataConstantProperty<\"point\" | \"line\" | \"line-center\">,\n \"symbol-spacing\": DataConstantProperty,\n \"symbol-avoid-edges\": DataConstantProperty,\n \"symbol-sort-key\": DataDrivenProperty,\n \"symbol-z-order\": DataConstantProperty<\"auto\" | \"viewport-y\" | \"source\">,\n \"icon-allow-overlap\": DataConstantProperty,\n \"icon-overlap\": DataConstantProperty<\"never\" | \"always\" | \"cooperative\">,\n \"icon-ignore-placement\": DataConstantProperty,\n \"icon-optional\": DataConstantProperty,\n \"icon-rotation-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"auto\">,\n \"icon-size\": DataDrivenProperty,\n \"icon-text-fit\": DataConstantProperty<\"none\" | \"width\" | \"height\" | \"both\">,\n \"icon-text-fit-padding\": DataConstantProperty<[number, number, number, number]>,\n \"icon-image\": DataDrivenProperty,\n \"icon-rotate\": DataDrivenProperty,\n \"icon-padding\": DataDrivenProperty,\n \"icon-keep-upright\": DataConstantProperty,\n \"icon-offset\": DataDrivenProperty<[number, number]>,\n \"icon-anchor\": DataDrivenProperty<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"icon-pitch-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"auto\">,\n \"text-pitch-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"auto\">,\n \"text-rotation-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"viewport-glyph\" | \"auto\">,\n \"text-field\": DataDrivenProperty,\n \"text-font\": DataDrivenProperty>,\n \"text-size\": DataDrivenProperty,\n \"text-max-width\": DataDrivenProperty,\n \"text-line-height\": DataConstantProperty,\n \"text-letter-spacing\": DataDrivenProperty,\n \"text-justify\": DataDrivenProperty<\"auto\" | \"left\" | \"center\" | \"right\">,\n \"text-radial-offset\": DataDrivenProperty,\n \"text-variable-anchor\": DataConstantProperty>,\n \"text-variable-anchor-offset\": DataDrivenProperty,\n \"text-anchor\": DataDrivenProperty<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"text-max-angle\": DataConstantProperty,\n \"text-writing-mode\": DataConstantProperty>,\n \"text-rotate\": DataDrivenProperty,\n \"text-padding\": DataConstantProperty,\n \"text-keep-upright\": DataConstantProperty,\n \"text-transform\": DataDrivenProperty<\"none\" | \"uppercase\" | \"lowercase\">,\n \"text-offset\": DataDrivenProperty<[number, number]>,\n \"text-allow-overlap\": DataConstantProperty,\n \"text-overlap\": DataConstantProperty<\"never\" | \"always\" | \"cooperative\">,\n \"text-ignore-placement\": DataConstantProperty,\n \"text-optional\": DataConstantProperty,\n};\n\nexport type SymbolLayoutPropsPossiblyEvaluated = {\n \"symbol-placement\": \"point\" | \"line\" | \"line-center\",\n \"symbol-spacing\": number,\n \"symbol-avoid-edges\": boolean,\n \"symbol-sort-key\": PossiblyEvaluatedPropertyValue,\n \"symbol-z-order\": \"auto\" | \"viewport-y\" | \"source\",\n \"icon-allow-overlap\": boolean,\n \"icon-overlap\": \"never\" | \"always\" | \"cooperative\",\n \"icon-ignore-placement\": boolean,\n \"icon-optional\": boolean,\n \"icon-rotation-alignment\": \"map\" | \"viewport\" | \"auto\",\n \"icon-size\": PossiblyEvaluatedPropertyValue,\n \"icon-text-fit\": \"none\" | \"width\" | \"height\" | \"both\",\n \"icon-text-fit-padding\": [number, number, number, number],\n \"icon-image\": PossiblyEvaluatedPropertyValue,\n \"icon-rotate\": PossiblyEvaluatedPropertyValue,\n \"icon-padding\": PossiblyEvaluatedPropertyValue,\n \"icon-keep-upright\": boolean,\n \"icon-offset\": PossiblyEvaluatedPropertyValue<[number, number]>,\n \"icon-anchor\": PossiblyEvaluatedPropertyValue<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"icon-pitch-alignment\": \"map\" | \"viewport\" | \"auto\",\n \"text-pitch-alignment\": \"map\" | \"viewport\" | \"auto\",\n \"text-rotation-alignment\": \"map\" | \"viewport\" | \"viewport-glyph\" | \"auto\",\n \"text-field\": PossiblyEvaluatedPropertyValue,\n \"text-font\": PossiblyEvaluatedPropertyValue>,\n \"text-size\": PossiblyEvaluatedPropertyValue,\n \"text-max-width\": PossiblyEvaluatedPropertyValue,\n \"text-line-height\": number,\n \"text-letter-spacing\": PossiblyEvaluatedPropertyValue,\n \"text-justify\": PossiblyEvaluatedPropertyValue<\"auto\" | \"left\" | \"center\" | \"right\">,\n \"text-radial-offset\": PossiblyEvaluatedPropertyValue,\n \"text-variable-anchor\": Array<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"text-variable-anchor-offset\": PossiblyEvaluatedPropertyValue,\n \"text-anchor\": PossiblyEvaluatedPropertyValue<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"text-max-angle\": number,\n \"text-writing-mode\": Array<\"horizontal\" | \"vertical\">,\n \"text-rotate\": PossiblyEvaluatedPropertyValue,\n \"text-padding\": number,\n \"text-keep-upright\": boolean,\n \"text-transform\": PossiblyEvaluatedPropertyValue<\"none\" | \"uppercase\" | \"lowercase\">,\n \"text-offset\": PossiblyEvaluatedPropertyValue<[number, number]>,\n \"text-allow-overlap\": boolean,\n \"text-overlap\": \"never\" | \"always\" | \"cooperative\",\n \"text-ignore-placement\": boolean,\n \"text-optional\": boolean,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"symbol-placement\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-placement\"] as any as StylePropertySpecification),\n \"symbol-spacing\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-spacing\"] as any as StylePropertySpecification),\n \"symbol-avoid-edges\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-avoid-edges\"] as any as StylePropertySpecification),\n \"symbol-sort-key\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"symbol-sort-key\"] as any as StylePropertySpecification),\n \"symbol-z-order\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-z-order\"] as any as StylePropertySpecification),\n \"icon-allow-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-allow-overlap\"] as any as StylePropertySpecification),\n \"icon-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-overlap\"] as any as StylePropertySpecification),\n \"icon-ignore-placement\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-ignore-placement\"] as any as StylePropertySpecification),\n \"icon-optional\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-optional\"] as any as StylePropertySpecification),\n \"icon-rotation-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-rotation-alignment\"] as any as StylePropertySpecification),\n \"icon-size\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-size\"] as any as StylePropertySpecification),\n \"icon-text-fit\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-text-fit\"] as any as StylePropertySpecification),\n \"icon-text-fit-padding\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-text-fit-padding\"] as any as StylePropertySpecification),\n \"icon-image\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-image\"] as any as StylePropertySpecification),\n \"icon-rotate\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-rotate\"] as any as StylePropertySpecification),\n \"icon-padding\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-padding\"] as any as StylePropertySpecification),\n \"icon-keep-upright\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-keep-upright\"] as any as StylePropertySpecification),\n \"icon-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-offset\"] as any as StylePropertySpecification),\n \"icon-anchor\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-anchor\"] as any as StylePropertySpecification),\n \"icon-pitch-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-pitch-alignment\"] as any as StylePropertySpecification),\n \"text-pitch-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-pitch-alignment\"] as any as StylePropertySpecification),\n \"text-rotation-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-rotation-alignment\"] as any as StylePropertySpecification),\n \"text-field\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-field\"] as any as StylePropertySpecification),\n \"text-font\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-font\"] as any as StylePropertySpecification),\n \"text-size\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-size\"] as any as StylePropertySpecification),\n \"text-max-width\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-max-width\"] as any as StylePropertySpecification),\n \"text-line-height\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-line-height\"] as any as StylePropertySpecification),\n \"text-letter-spacing\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-letter-spacing\"] as any as StylePropertySpecification),\n \"text-justify\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-justify\"] as any as StylePropertySpecification),\n \"text-radial-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-radial-offset\"] as any as StylePropertySpecification),\n \"text-variable-anchor\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-variable-anchor\"] as any as StylePropertySpecification),\n \"text-variable-anchor-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-variable-anchor-offset\"] as any as StylePropertySpecification),\n \"text-anchor\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-anchor\"] as any as StylePropertySpecification),\n \"text-max-angle\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-max-angle\"] as any as StylePropertySpecification),\n \"text-writing-mode\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-writing-mode\"] as any as StylePropertySpecification),\n \"text-rotate\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-rotate\"] as any as StylePropertySpecification),\n \"text-padding\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-padding\"] as any as StylePropertySpecification),\n \"text-keep-upright\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-keep-upright\"] as any as StylePropertySpecification),\n \"text-transform\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-transform\"] as any as StylePropertySpecification),\n \"text-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-offset\"] as any as StylePropertySpecification),\n \"text-allow-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-allow-overlap\"] as any as StylePropertySpecification),\n \"text-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-overlap\"] as any as StylePropertySpecification),\n \"text-ignore-placement\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-ignore-placement\"] as any as StylePropertySpecification),\n \"text-optional\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-optional\"] as any as StylePropertySpecification),\n});\n\nexport type SymbolPaintProps = {\n \"icon-opacity\": DataDrivenProperty,\n \"icon-color\": DataDrivenProperty,\n \"icon-halo-color\": DataDrivenProperty,\n \"icon-halo-width\": DataDrivenProperty,\n \"icon-halo-blur\": DataDrivenProperty,\n \"icon-translate\": DataConstantProperty<[number, number]>,\n \"icon-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"text-opacity\": DataDrivenProperty,\n \"text-color\": DataDrivenProperty,\n \"text-halo-color\": DataDrivenProperty,\n \"text-halo-width\": DataDrivenProperty,\n \"text-halo-blur\": DataDrivenProperty,\n \"text-translate\": DataConstantProperty<[number, number]>,\n \"text-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n};\n\nexport type SymbolPaintPropsPossiblyEvaluated = {\n \"icon-opacity\": PossiblyEvaluatedPropertyValue,\n \"icon-color\": PossiblyEvaluatedPropertyValue,\n \"icon-halo-color\": PossiblyEvaluatedPropertyValue,\n \"icon-halo-width\": PossiblyEvaluatedPropertyValue,\n \"icon-halo-blur\": PossiblyEvaluatedPropertyValue,\n \"icon-translate\": [number, number],\n \"icon-translate-anchor\": \"map\" | \"viewport\",\n \"text-opacity\": PossiblyEvaluatedPropertyValue,\n \"text-color\": PossiblyEvaluatedPropertyValue,\n \"text-halo-color\": PossiblyEvaluatedPropertyValue,\n \"text-halo-width\": PossiblyEvaluatedPropertyValue,\n \"text-halo-blur\": PossiblyEvaluatedPropertyValue,\n \"text-translate\": [number, number],\n \"text-translate-anchor\": \"map\" | \"viewport\",\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"icon-opacity\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-opacity\"] as any as StylePropertySpecification),\n \"icon-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-color\"] as any as StylePropertySpecification),\n \"icon-halo-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-halo-color\"] as any as StylePropertySpecification),\n \"icon-halo-width\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-halo-width\"] as any as StylePropertySpecification),\n \"icon-halo-blur\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-halo-blur\"] as any as StylePropertySpecification),\n \"icon-translate\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"icon-translate\"] as any as StylePropertySpecification),\n \"icon-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"icon-translate-anchor\"] as any as StylePropertySpecification),\n \"text-opacity\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-opacity\"] as any as StylePropertySpecification),\n \"text-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-color\"] as any as StylePropertySpecification, { runtimeType: ColorType, getOverride: (o) => o.textColor, hasOverride: (o) => !!o.textColor }),\n \"text-halo-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-halo-color\"] as any as StylePropertySpecification),\n \"text-halo-width\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-halo-width\"] as any as StylePropertySpecification),\n \"text-halo-blur\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-halo-blur\"] as any as StylePropertySpecification),\n \"text-translate\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"text-translate\"] as any as StylePropertySpecification),\n \"text-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"text-translate-anchor\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import type {Expression, EvaluationContext, Type, ZoomConstantExpression} from '@maplibre/maplibre-gl-style-spec';\nimport {NullType} from '@maplibre/maplibre-gl-style-spec';\nimport {PossiblyEvaluatedPropertyValue} from './properties';\nimport {register} from '../util/web_worker_transfer';\n\n// This is an internal expression class. It is only used in GL JS and\n// has GL JS dependencies which can break the standalone style-spec module\nexport class FormatSectionOverride implements Expression {\n type: Type;\n defaultValue: PossiblyEvaluatedPropertyValue;\n\n constructor(defaultValue: PossiblyEvaluatedPropertyValue) {\n if (defaultValue.property.overrides === undefined) throw new Error('overrides must be provided to instantiate FormatSectionOverride class');\n this.type = defaultValue.property.overrides ? defaultValue.property.overrides.runtimeType : NullType;\n this.defaultValue = defaultValue;\n }\n\n evaluate(ctx: EvaluationContext) {\n if (ctx.formattedSection) {\n const overrides = this.defaultValue.property.overrides;\n if (overrides && overrides.hasOverride(ctx.formattedSection)) {\n return overrides.getOverride(ctx.formattedSection);\n }\n }\n\n if (ctx.feature && ctx.featureState) {\n return this.defaultValue.evaluate(ctx.feature, ctx.featureState);\n }\n\n return this.defaultValue.property.specification.default;\n }\n\n eachChild(fn: (_: Expression) => void) {\n if (!this.defaultValue.isConstant()) {\n const expr: ZoomConstantExpression<'source'> = (this.defaultValue.value as any);\n fn(expr._styleExpression.expression);\n }\n }\n\n // Cannot be statically evaluated, as the output depends on the evaluation context.\n outputDefined() {\n return false;\n }\n\n serialize() {\n return null;\n }\n}\n\nregister('FormatSectionOverride', FormatSectionOverride, {omit: ['defaultValue']});\n","import {StyleLayer} from '../style_layer';\n\nimport {SymbolBucket, SymbolFeature} from '../../data/bucket/symbol_bucket';\nimport {resolveTokens} from '../../util/resolve_tokens';\nimport properties, {SymbolLayoutPropsPossiblyEvaluated, SymbolPaintPropsPossiblyEvaluated} from './symbol_style_layer_properties.g';\n\nimport {\n Transitionable,\n Transitioning,\n Layout,\n PossiblyEvaluated,\n PossiblyEvaluatedPropertyValue,\n PropertyValue\n} from '../properties';\n\nimport {\n isExpression,\n StyleExpression,\n ZoomConstantExpression,\n ZoomDependentExpression,\n FormattedType,\n typeOf,\n Formatted,\n FormatExpression,\n Literal} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {BucketParameters} from '../../data/bucket';\nimport type {SymbolLayoutProps, SymbolPaintProps} from './symbol_style_layer_properties.g';\nimport type {EvaluationParameters} from '../evaluation_parameters';\nimport type {Expression, Feature, SourceExpression, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport {FormatSectionOverride} from '../format_section_override';\n\nexport class SymbolStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n super.recalculate(parameters, availableImages);\n\n if (this.layout.get('icon-rotation-alignment') === 'auto') {\n if (this.layout.get('symbol-placement') !== 'point') {\n this.layout._values['icon-rotation-alignment'] = 'map';\n } else {\n this.layout._values['icon-rotation-alignment'] = 'viewport';\n }\n }\n\n if (this.layout.get('text-rotation-alignment') === 'auto') {\n if (this.layout.get('symbol-placement') !== 'point') {\n this.layout._values['text-rotation-alignment'] = 'map';\n } else {\n this.layout._values['text-rotation-alignment'] = 'viewport';\n }\n }\n\n // If unspecified, `*-pitch-alignment` inherits `*-rotation-alignment`\n if (this.layout.get('text-pitch-alignment') === 'auto') {\n this.layout._values['text-pitch-alignment'] = this.layout.get('text-rotation-alignment') === 'map' ? 'map' : 'viewport';\n }\n if (this.layout.get('icon-pitch-alignment') === 'auto') {\n this.layout._values['icon-pitch-alignment'] = this.layout.get('icon-rotation-alignment');\n }\n\n if (this.layout.get('symbol-placement') === 'point') {\n const writingModes = this.layout.get('text-writing-mode');\n if (writingModes) {\n // remove duplicates, preserving order\n const deduped = [];\n for (const m of writingModes) {\n if (deduped.indexOf(m) < 0) deduped.push(m);\n }\n this.layout._values['text-writing-mode'] = deduped;\n } else {\n this.layout._values['text-writing-mode'] = ['horizontal'];\n }\n }\n\n this._setPaintOverrides();\n }\n\n getValueAndResolveTokens(name: any, feature: Feature, canonical: CanonicalTileID, availableImages: Array) {\n const value = this.layout.get(name).evaluate(feature, {}, canonical, availableImages);\n const unevaluated = this._unevaluatedLayout._values[name];\n if (!unevaluated.isDataDriven() && !isExpression(unevaluated.value) && value) {\n return resolveTokens(feature.properties, value);\n }\n\n return value;\n }\n\n createBucket(parameters: BucketParameters) {\n return new SymbolBucket(parameters);\n }\n\n queryRadius(): number {\n return 0;\n }\n\n queryIntersectsFeature(): boolean {\n throw new Error('Should take a different path in FeatureIndex');\n }\n\n _setPaintOverrides() {\n for (const overridable of properties.paint.overridableProperties) {\n if (!SymbolStyleLayer.hasPaintOverride(this.layout, overridable)) {\n continue;\n }\n const overriden = this.paint.get(overridable as keyof SymbolPaintPropsPossiblyEvaluated) as PossiblyEvaluatedPropertyValue;\n const override = new FormatSectionOverride(overriden);\n const styleExpression = new StyleExpression(override, overriden.property.specification);\n let expression = null;\n if (overriden.value.kind === 'constant' || overriden.value.kind === 'source') {\n expression = new ZoomConstantExpression('source', styleExpression) as SourceExpression;\n } else {\n expression = new ZoomDependentExpression('composite',\n styleExpression,\n overriden.value.zoomStops);\n }\n this.paint._values[overridable] = new PossiblyEvaluatedPropertyValue(overriden.property,\n expression,\n overriden.parameters);\n }\n }\n\n _handleOverridablePaintPropertyUpdate(name: string, oldValue: PropertyValue, newValue: PropertyValue): boolean {\n if (!this.layout || oldValue.isDataDriven() || newValue.isDataDriven()) {\n return false;\n }\n return SymbolStyleLayer.hasPaintOverride(this.layout, name);\n }\n\n static hasPaintOverride(layout: PossiblyEvaluated, propertyName: string): boolean {\n const textField = layout.get('text-field');\n const property = properties.paint.properties[propertyName];\n let hasOverrides = false;\n\n const checkSections = (sections) => {\n for (const section of sections) {\n if (property.overrides && property.overrides.hasOverride(section)) {\n hasOverrides = true;\n return;\n }\n }\n };\n\n if (textField.value.kind === 'constant' && textField.value.value instanceof Formatted) {\n checkSections(textField.value.value.sections);\n } else if (textField.value.kind === 'source') {\n\n const checkExpression = (expression: Expression) => {\n if (hasOverrides) return;\n\n if (expression instanceof Literal && typeOf(expression.value) === FormattedType) {\n const formatted: Formatted = (expression.value as any);\n checkSections(formatted.sections);\n } else if (expression instanceof FormatExpression) {\n checkSections(expression.sections);\n } else {\n expression.eachChild(checkExpression);\n }\n };\n\n const expr: ZoomConstantExpression<'source'> = (textField.value as any);\n if (expr._styleExpression) {\n checkExpression(expr._styleExpression.expression);\n }\n }\n\n return hasOverrides;\n }\n}\n\nexport type SymbolPadding = [number, number, number, number];\n\nexport function getIconPadding(layout: PossiblyEvaluated, feature: SymbolFeature, canonical: CanonicalTileID, pixelRatio = 1): SymbolPadding {\n // Support text-padding in addition to icon-padding? Unclear how to apply asymmetric text-padding to the radius for collision circles.\n const result = layout.get('icon-padding').evaluate(feature, {}, canonical);\n const values = result && result.values;\n\n return [\n values[0] * pixelRatio,\n values[1] * pixelRatio,\n values[2] * pixelRatio,\n values[3] * pixelRatio,\n ];\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type BackgroundPaintProps = {\n \"background-color\": DataConstantProperty,\n \"background-pattern\": CrossFadedProperty,\n \"background-opacity\": DataConstantProperty,\n};\n\nexport type BackgroundPaintPropsPossiblyEvaluated = {\n \"background-color\": Color,\n \"background-pattern\": CrossFaded,\n \"background-opacity\": number,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"background-color\": new DataConstantProperty(styleSpec[\"paint_background\"][\"background-color\"] as any as StylePropertySpecification),\n \"background-pattern\": new CrossFadedProperty(styleSpec[\"paint_background\"][\"background-pattern\"] as any as StylePropertySpecification),\n \"background-opacity\": new DataConstantProperty(styleSpec[\"paint_background\"][\"background-opacity\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport properties, {BackgroundPaintPropsPossiblyEvaluated} from './background_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {BackgroundPaintProps} from './background_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport class BackgroundStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type RasterPaintProps = {\n \"raster-opacity\": DataConstantProperty,\n \"raster-hue-rotate\": DataConstantProperty,\n \"raster-brightness-min\": DataConstantProperty,\n \"raster-brightness-max\": DataConstantProperty,\n \"raster-saturation\": DataConstantProperty,\n \"raster-contrast\": DataConstantProperty,\n \"raster-resampling\": DataConstantProperty<\"linear\" | \"nearest\">,\n \"raster-fade-duration\": DataConstantProperty,\n};\n\nexport type RasterPaintPropsPossiblyEvaluated = {\n \"raster-opacity\": number,\n \"raster-hue-rotate\": number,\n \"raster-brightness-min\": number,\n \"raster-brightness-max\": number,\n \"raster-saturation\": number,\n \"raster-contrast\": number,\n \"raster-resampling\": \"linear\" | \"nearest\",\n \"raster-fade-duration\": number,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"raster-opacity\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-opacity\"] as any as StylePropertySpecification),\n \"raster-hue-rotate\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-hue-rotate\"] as any as StylePropertySpecification),\n \"raster-brightness-min\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-brightness-min\"] as any as StylePropertySpecification),\n \"raster-brightness-max\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-brightness-max\"] as any as StylePropertySpecification),\n \"raster-saturation\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-saturation\"] as any as StylePropertySpecification),\n \"raster-contrast\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-contrast\"] as any as StylePropertySpecification),\n \"raster-resampling\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-resampling\"] as any as StylePropertySpecification),\n \"raster-fade-duration\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-fade-duration\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport properties, {RasterPaintPropsPossiblyEvaluated} from './raster_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {RasterPaintProps} from './raster_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport class RasterStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n}\n","import {StyleLayer} from '../style_layer';\nimport type {Map} from '../../ui/map';\nimport {mat4} from 'gl-matrix';\nimport {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * @param gl - The map's gl context.\n * @param matrix - The map's camera matrix. It projects spherical mercator\n * coordinates to gl coordinates. The spherical mercator coordinate `[0, 0]` represents the\n * top left corner of the mercator world and `[1, 1]` represents the bottom right corner. When\n * the `renderingMode` is `\"3d\"`, the z coordinate is conformal. A box with identical x, y, and z\n * lengths in mercator units would be rendered as a cube. {@link MercatorCoordinate.fromLngLat}\n * can be used to project a `LngLat` to a mercator coordinate.\n */\ntype CustomRenderMethod = (gl: WebGLRenderingContext|WebGL2RenderingContext, matrix: mat4) => void;\n\n/**\n * Interface for custom style layers. This is a specification for\n * implementers to model: it is not an exported method or class.\n *\n * Custom layers allow a user to render directly into the map's GL context using the map's camera.\n * These layers can be added between any regular layers using {@link Map#addLayer}.\n *\n * Custom layers must have a unique `id` and must have the `type` of `\"custom\"`.\n * They must implement `render` and may implement `prerender`, `onAdd` and `onRemove`.\n * They can trigger rendering using {@link Map#triggerRepaint}\n * and they should appropriately handle {@link MapContextEvent} with `webglcontextlost` and `webglcontextrestored`.\n *\n * The `renderingMode` property controls whether the layer is treated as a `\"2d\"` or `\"3d\"` map layer. Use:\n * - `\"renderingMode\": \"3d\"` to use the depth buffer and share it with other layers\n * - `\"renderingMode\": \"2d\"` to add a layer with no depth. If you need to use the depth buffer for a `\"2d\"` layer you must use an offscreen\n * framebuffer and {@link CustomLayerInterface#prerender}\n *\n * @example\n * Custom layer implemented as ES6 class\n * ```ts\n * class NullIslandLayer {\n * constructor() {\n * this.id = 'null-island';\n * this.type = 'custom';\n * this.renderingMode = '2d';\n * }\n *\n * onAdd(map, gl) {\n * const vertexSource = `\n * uniform mat4 u_matrix;\n * void main() {\n * gl_Position = u_matrix * vec4(0.5, 0.5, 0.0, 1.0);\n * gl_PointSize = 20.0;\n * }`;\n *\n * const fragmentSource = `\n * void main() {\n * fragColor = vec4(1.0, 0.0, 0.0, 1.0);\n * }`;\n *\n * const vertexShader = gl.createShader(gl.VERTEX_SHADER);\n * gl.shaderSource(vertexShader, vertexSource);\n * gl.compileShader(vertexShader);\n * const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);\n * gl.shaderSource(fragmentShader, fragmentSource);\n * gl.compileShader(fragmentShader);\n *\n * this.program = gl.createProgram();\n * gl.attachShader(this.program, vertexShader);\n * gl.attachShader(this.program, fragmentShader);\n * gl.linkProgram(this.program);\n * }\n *\n * render(gl, matrix) {\n * gl.useProgram(this.program);\n * gl.uniformMatrix4fv(gl.getUniformLocation(this.program, \"u_matrix\"), false, matrix);\n * gl.drawArrays(gl.POINTS, 0, 1);\n * }\n * }\n *\n * map.on('load', function() {\n * map.addLayer(new NullIslandLayer());\n * });\n * ```\n */\nexport interface CustomLayerInterface {\n /**\n * A unique layer id.\n */\n id: string;\n /**\n * The layer's type. Must be `\"custom\"`.\n */\n type: 'custom';\n /**\n * Either `\"2d\"` or `\"3d\"`. Defaults to `\"2d\"`.\n */\n renderingMode?: '2d' | '3d';\n /**\n * Called during a render frame allowing the layer to draw into the GL context.\n *\n * The layer can assume blending and depth state is set to allow the layer to properly\n * blend and clip other layers. The layer cannot make any other assumptions about the\n * current GL state.\n *\n * If the layer needs to render to a texture, it should implement the `prerender` method\n * to do this and only use the `render` method for drawing directly into the main framebuffer.\n *\n * The blend function is set to `gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA)`. This expects\n * colors to be provided in premultiplied alpha form where the `r`, `g` and `b` values are already\n * multiplied by the `a` value. If you are unable to provide colors in premultiplied form you\n * may want to change the blend function to\n * `gl.blendFuncSeparate(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA, gl.ONE, gl.ONE_MINUS_SRC_ALPHA)`.\n */\n render: CustomRenderMethod;\n /**\n * Optional method called during a render frame to allow a layer to prepare resources or render into a texture.\n *\n * The layer cannot make any assumptions about the current GL state and must bind a framebuffer before rendering.\n */\n prerender?: CustomRenderMethod;\n /**\n * Optional method called when the layer has been added to the Map with {@link Map#addLayer}. This\n * gives the layer a chance to initialize gl resources and register event listeners.\n *\n * @param map - The Map this custom layer was just added to.\n * @param gl - The gl context for the map.\n */\n onAdd?(map: Map, gl: WebGLRenderingContext | WebGL2RenderingContext): void;\n /**\n * Optional method called when the layer has been removed from the Map with {@link Map#removeLayer}. This\n * gives the layer a chance to clean up gl resources and event listeners.\n *\n * @param map - The Map this custom layer was just added to.\n * @param gl - The gl context for the map.\n */\n onRemove?(map: Map, gl: WebGLRenderingContext | WebGL2RenderingContext): void;\n}\n\nexport function validateCustomStyleLayer(layerObject: CustomLayerInterface) {\n const errors = [];\n const id = layerObject.id;\n\n if (id === undefined) {\n errors.push({\n message: `layers.${id}: missing required property \"id\"`\n });\n }\n\n if (layerObject.render === undefined) {\n errors.push({\n message: `layers.${id}: missing required method \"render\"`\n });\n }\n\n if (layerObject.renderingMode &&\n layerObject.renderingMode !== '2d' &&\n layerObject.renderingMode !== '3d') {\n errors.push({\n message: `layers.${id}: property \"renderingMode\" must be either \"2d\" or \"3d\"`\n });\n }\n\n return errors;\n}\n\nexport class CustomStyleLayer extends StyleLayer {\n\n implementation: CustomLayerInterface;\n\n constructor(implementation: CustomLayerInterface) {\n super(implementation, {});\n this.implementation = implementation;\n }\n\n is3D() {\n return this.implementation.renderingMode === '3d';\n }\n\n hasOffscreenPass() {\n return this.implementation.prerender !== undefined;\n }\n\n recalculate() {}\n updateTransitions() {}\n hasTransition() { return false; }\n\n serialize(): LayerSpecification {\n throw new Error('Custom layers cannot be serialized');\n }\n\n onAdd = (map: Map) => {\n if (this.implementation.onAdd) {\n this.implementation.onAdd(map, map.painter.context.gl);\n }\n };\n\n onRemove = (map: Map) => {\n if (this.implementation.onRemove) {\n this.implementation.onRemove(map, map.painter.context.gl);\n }\n };\n}\n","import {CircleStyleLayer} from './style_layer/circle_style_layer';\nimport {HeatmapStyleLayer} from './style_layer/heatmap_style_layer';\nimport {HillshadeStyleLayer} from './style_layer/hillshade_style_layer';\nimport {FillStyleLayer} from './style_layer/fill_style_layer';\nimport {FillExtrusionStyleLayer} from './style_layer/fill_extrusion_style_layer';\nimport {LineStyleLayer} from './style_layer/line_style_layer';\nimport {SymbolStyleLayer} from './style_layer/symbol_style_layer';\nimport {BackgroundStyleLayer} from './style_layer/background_style_layer';\nimport {RasterStyleLayer} from './style_layer/raster_style_layer';\nimport {CustomStyleLayer, type CustomLayerInterface} from './style_layer/custom_style_layer';\n\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport function createStyleLayer(layer: LayerSpecification | CustomLayerInterface) {\n if (layer.type === 'custom') {\n return new CustomStyleLayer(layer);\n }\n switch (layer.type) {\n case 'background':\n return new BackgroundStyleLayer(layer);\n case 'circle':\n return new CircleStyleLayer(layer);\n case 'fill':\n return new FillStyleLayer(layer);\n case 'fill-extrusion':\n return new FillExtrusionStyleLayer(layer);\n case 'heatmap':\n return new HeatmapStyleLayer(layer);\n case 'hillshade':\n return new HillshadeStyleLayer(layer);\n case 'line':\n return new LineStyleLayer(layer);\n case 'raster':\n return new RasterStyleLayer(layer);\n case 'symbol':\n return new SymbolStyleLayer(layer);\n }\n}\n\n","/**\n * Invokes the wrapped function in a non-blocking way when trigger() is called. Invocation requests\n * are ignored until the function was actually invoked.\n */\nexport class ThrottledInvoker {\n _channel: MessageChannel;\n _triggered: boolean;\n _callback: Function;\n\n constructor(callback: Function) {\n this._callback = callback;\n this._triggered = false;\n if (typeof MessageChannel !== 'undefined') {\n this._channel = new MessageChannel();\n this._channel.port2.onmessage = () => {\n this._triggered = false;\n this._callback();\n };\n }\n }\n\n trigger() {\n if (!this._triggered) {\n this._triggered = true;\n if (this._channel) {\n this._channel.port1.postMessage(true);\n } else {\n setTimeout(() => {\n this._triggered = false;\n this._callback();\n }, 0);\n }\n }\n }\n\n remove() {\n delete this._channel;\n this._callback = () => {};\n }\n}\n","import {isWorker} from './util';\nimport {serialize, deserialize, Serialized} from './web_worker_transfer';\nimport {ThrottledInvoker} from './throttled_invoker';\n\nimport type {Transferable} from '../types/transferable';\nimport type {Cancelable} from '../types/cancelable';\nimport type {WorkerSource} from '../source/worker_source';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {Callback} from '../types/callback';\nimport type {StyleGlyph} from '../style/style_glyph';\n\nexport interface ActorTarget {\n addEventListener: typeof window.addEventListener;\n removeEventListener: typeof window.removeEventListener;\n postMessage: typeof window.postMessage;\n terminate?: () => void;\n}\n\nexport interface WorkerSourceProvider {\n getWorkerSource(mapId: string | number, sourceType: string, sourceName: string): WorkerSource;\n}\n\nexport interface GlyphsProvider {\n getGlyphs(mapId: string, params: {\n stacks: {[_: string]: Array};\n source: string;\n tileID: OverscaledTileID;\n type: string;\n },\n callback: Callback<{[_: string]: {[_: number]: StyleGlyph}}>\n );\n}\n\nexport type MessageType = '' | '' |\n'geojson.getClusterExpansionZoom' | 'geojson.getClusterChildren' | 'geojson.getClusterLeaves' | 'geojson.loadData' |\n'removeSource' | 'loadWorkerSource' | 'loadDEMTile' | 'removeDEMTile' |\n'removeTile' | 'reloadTile' | 'abortTile' | 'loadTile' | 'getTile' |\n'getGlyphs' | 'getImages' | 'setImages' |\n'syncRTLPluginState' | 'setReferrer' | 'setLayers' | 'updateLayers';\n\nexport type MessageData = {\n id: string;\n type: MessageType;\n data?: Serialized;\n targetMapId?: string | number | null;\n mustQueue?: boolean;\n error?: Serialized | null;\n hasCallback?: boolean;\n sourceMapId: string | number | null;\n}\n\nexport type Message = {\n data: MessageData;\n}\n\n/**\n * An implementation of the [Actor design pattern](http://en.wikipedia.org/wiki/Actor_model)\n * that maintains the relationship between asynchronous tasks and the objects\n * that spin them off - in this case, tasks like parsing parts of styles,\n * owned by the styles\n */\nexport class Actor {\n target: ActorTarget;\n parent: WorkerSourceProvider | GlyphsProvider;\n mapId: string | number | null;\n callbacks: { [x: number]: Function};\n name: string;\n tasks: { [x: number]: MessageData };\n taskQueue: Array;\n cancelCallbacks: { [x: number]: () => void };\n invoker: ThrottledInvoker;\n globalScope: ActorTarget;\n\n /**\n * @param target - The target\n * @param parent - The parent\n * @param mapId - A unique identifier for the Map instance using this Actor.\n */\n constructor(target: ActorTarget, parent: WorkerSourceProvider | GlyphsProvider, mapId?: string | number) {\n this.target = target;\n this.parent = parent;\n this.mapId = mapId;\n this.callbacks = {};\n this.tasks = {};\n this.taskQueue = [];\n this.cancelCallbacks = {};\n this.invoker = new ThrottledInvoker(this.process);\n this.target.addEventListener('message', this.receive, false);\n this.globalScope = isWorker() ? target : window;\n }\n\n /**\n * Sends a message from a main-thread map to a Worker or from a Worker back to\n * a main-thread map instance.\n *\n * @param type - The name of the target method to invoke or '[source-type].[source-name].name' for a method on a WorkerSource.\n * @param targetMapId - A particular mapId to which to send this message.\n */\n send(\n type: MessageType,\n data: unknown,\n callback?: Function | null,\n targetMapId?: string | null,\n mustQueue: boolean = false\n ): Cancelable {\n // We're using a string ID instead of numbers because they are being used as object keys\n // anyway, and thus stringified implicitly. We use random IDs because an actor may receive\n // message from multiple other actors which could run in different execution context. A\n // linearly increasing ID could produce collisions.\n const id = Math.round((Math.random() * 1e18)).toString(36).substring(0, 10);\n if (callback) {\n this.callbacks[id] = callback;\n }\n const buffers: Array = [];\n const message: MessageData = {\n id,\n type,\n hasCallback: !!callback,\n targetMapId,\n mustQueue,\n sourceMapId: this.mapId,\n data: serialize(data, buffers)\n };\n\n this.target.postMessage(message, {transfer: buffers});\n return {\n cancel: () => {\n if (callback) {\n // Set the callback to null so that it never fires after the request is aborted.\n delete this.callbacks[id];\n }\n const cancelMessage: MessageData = {\n id,\n type: '',\n targetMapId,\n sourceMapId: this.mapId\n };\n this.target.postMessage(cancelMessage);\n }\n };\n }\n\n receive = (message: Message) => {\n const data = message.data;\n const id = data.id;\n\n if (!id) {\n return;\n }\n\n if (data.targetMapId && this.mapId !== data.targetMapId) {\n return;\n }\n\n if (data.type === '') {\n // Remove the original request from the queue. This is only possible if it\n // hasn't been kicked off yet. The id will remain in the queue, but because\n // there is no associated task, it will be dropped once it's time to execute it.\n delete this.tasks[id];\n const cancel = this.cancelCallbacks[id];\n delete this.cancelCallbacks[id];\n if (cancel) {\n cancel();\n }\n } else {\n if (isWorker() || data.mustQueue) {\n // In workers, store the tasks that we need to process before actually processing them. This\n // is necessary because we want to keep receiving messages, and in particular,\n // messages. Some tasks may take a while in the worker thread, so before\n // executing the next task in our queue, postMessage preempts this and \n // messages can be processed. We're using a MessageChannel object to get throttle the\n // process() flow to one at a time.\n this.tasks[id] = data;\n this.taskQueue.push(id);\n this.invoker.trigger();\n } else {\n // In the main thread, process messages immediately so that other work does not slip in\n // between getting partial data back from workers.\n this.processTask(id, data);\n }\n }\n };\n\n process = () => {\n if (!this.taskQueue.length) {\n return;\n }\n const id = this.taskQueue.shift();\n const task = this.tasks[id];\n delete this.tasks[id];\n // Schedule another process call if we know there's more to process _before_ invoking the\n // current task. This is necessary so that processing continues even if the current task\n // doesn't execute successfully.\n if (this.taskQueue.length) {\n this.invoker.trigger();\n }\n if (!task) {\n // If the task ID doesn't have associated task data anymore, it was canceled.\n return;\n }\n\n this.processTask(id, task);\n };\n\n processTask(id: string, task: MessageData) {\n if (task.type === '') {\n // The done() function in the counterpart has been called, and we are now\n // firing the callback in the originating actor, if there is one.\n const callback = this.callbacks[id];\n delete this.callbacks[id];\n if (callback) {\n // If we get a response, but don't have a callback, the request was canceled.\n if (task.error) {\n callback(deserialize(task.error));\n } else {\n callback(null, deserialize(task.data));\n }\n }\n } else {\n let completed = false;\n const buffers: Array = [];\n const done = task.hasCallback ? (err: Error, data?: any) => {\n completed = true;\n delete this.cancelCallbacks[id];\n const responseMessage: MessageData = {\n id,\n type: '',\n sourceMapId: this.mapId,\n error: err ? serialize(err) : null,\n data: serialize(data, buffers)\n };\n this.target.postMessage(responseMessage, {transfer: buffers});\n } : (_) => {\n completed = true;\n };\n\n let callback: Cancelable = null;\n const params = deserialize(task.data);\n if (this.parent[task.type]) {\n // task.type == 'loadTile', 'removeTile', etc.\n callback = this.parent[task.type](task.sourceMapId, params, done);\n } else if ('getWorkerSource' in this.parent) {\n // task.type == sourcetype.method\n const keys = task.type.split('.');\n const scope = this.parent.getWorkerSource(task.sourceMapId, keys[0], (params as any).source);\n callback = scope[keys[1]](params, done);\n } else {\n // No function was found.\n done(new Error(`Could not find function ${task.type}`));\n }\n\n if (!completed && callback && callback.cancel) {\n // Allows canceling the task as long as it hasn't been completed yet.\n this.cancelCallbacks[id] = callback.cancel;\n }\n }\n }\n\n remove() {\n this.invoker.remove();\n this.target.removeEventListener('message', this.receive, false);\n }\n}\n","import {wrap} from '../util/util';\n\n/*\n* Approximate radius of the earth in meters.\n* Uses the WGS-84 approximation. The radius at the equator is ~6378137 and at the poles is ~6356752. https://en.wikipedia.org/wiki/World_Geodetic_System#WGS84\n* 6371008.8 is one published \"average radius\" see https://en.wikipedia.org/wiki/Earth_radius#Mean_radius, or ftp://athena.fsv.cvut.cz/ZFG/grs80-Moritz.pdf p.4\n*/\nexport const earthRadius = 6371008.8;\n\n/**\n * A {@link LngLat} object, an array of two numbers representing longitude and latitude,\n * or an object with `lng` and `lat` or `lon` and `lat` properties.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let v1 = new maplibregl.LngLat(-122.420679, 37.772537);\n * let v2 = [-122.420679, 37.772537];\n * let v3 = {lon: -122.420679, lat: 37.772537};\n * ```\n */\nexport type LngLatLike = LngLat | {\n lng: number;\n lat: number;\n} | {\n lon: number;\n lat: number;\n} | [number, number];\n\n/**\n * A `LngLat` object represents a given longitude and latitude coordinate, measured in degrees.\n * These coordinates are based on the [WGS84 (EPSG:4326) standard](https://en.wikipedia.org/wiki/World_Geodetic_System#WGS84).\n *\n * MapLibre GL JS uses longitude, latitude coordinate order (as opposed to latitude, longitude) to match the\n * [GeoJSON specification](https://tools.ietf.org/html/rfc7946).\n *\n * Note that any MapLibre GL JS method that accepts a `LngLat` object as an argument or option\n * can also accept an `Array` of two numbers and will perform an implicit conversion.\n * This flexible type is documented as {@link LngLatLike}.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(-123.9749, 40.7736);\n * ll.lng; // = -123.9749\n * ```\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Create a timeline animation](https://maplibre.org/maplibre-gl-js/docs/examples/timeline-animation/)\n */\nexport class LngLat {\n lng: number;\n lat: number;\n\n /**\n * @param lng - Longitude, measured in degrees.\n * @param lat - Latitude, measured in degrees.\n */\n constructor(lng: number, lat: number) {\n if (isNaN(lng) || isNaN(lat)) {\n throw new Error(`Invalid LngLat object: (${lng}, ${lat})`);\n }\n this.lng = +lng;\n this.lat = +lat;\n if (this.lat > 90 || this.lat < -90) {\n throw new Error('Invalid LngLat latitude value: must be between -90 and 90');\n }\n }\n\n /**\n * Returns a new `LngLat` object whose longitude is wrapped to the range (-180, 180).\n *\n * @returns The wrapped `LngLat` object.\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(286.0251, 40.7736);\n * let wrapped = ll.wrap();\n * wrapped.lng; // = -73.9749\n * ```\n */\n wrap() {\n return new LngLat(wrap(this.lng, -180, 180), this.lat);\n }\n\n /**\n * Returns the coordinates represented as an array of two numbers.\n *\n * @returns The coordinates represented as an array of longitude and latitude.\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(-73.9749, 40.7736);\n * ll.toArray(); // = [-73.9749, 40.7736]\n * ```\n */\n toArray(): [number, number] {\n return [this.lng, this.lat];\n }\n\n /**\n * Returns the coordinates represent as a string.\n *\n * @returns The coordinates represented as a string of the format `'LngLat(lng, lat)'`.\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(-73.9749, 40.7736);\n * ll.toString(); // = \"LngLat(-73.9749, 40.7736)\"\n * ```\n */\n toString(): string {\n return `LngLat(${this.lng}, ${this.lat})`;\n }\n\n /**\n * Returns the approximate distance between a pair of coordinates in meters\n * Uses the Haversine Formula (from R.W. Sinnott, \"Virtues of the Haversine\", Sky and Telescope, vol. 68, no. 2, 1984, p. 159)\n *\n * @param lngLat - coordinates to compute the distance to\n * @returns Distance in meters between the two coordinates.\n * @example\n * ```ts\n * let new_york = new maplibregl.LngLat(-74.0060, 40.7128);\n * let los_angeles = new maplibregl.LngLat(-118.2437, 34.0522);\n * new_york.distanceTo(los_angeles); // = 3935751.690893987, \"true distance\" using a non-spherical approximation is ~3966km\n * ```\n */\n distanceTo(lngLat: LngLat): number {\n const rad = Math.PI / 180;\n const lat1 = this.lat * rad;\n const lat2 = lngLat.lat * rad;\n const a = Math.sin(lat1) * Math.sin(lat2) + Math.cos(lat1) * Math.cos(lat2) * Math.cos((lngLat.lng - this.lng) * rad);\n\n const maxMeters = earthRadius * Math.acos(Math.min(a, 1));\n return maxMeters;\n }\n\n /**\n * Converts an array of two numbers or an object with `lng` and `lat` or `lon` and `lat` properties\n * to a `LngLat` object.\n *\n * If a `LngLat` object is passed in, the function returns it unchanged.\n *\n * @param input - An array of two numbers or object to convert, or a `LngLat` object to return.\n * @returns A new `LngLat` object, if a conversion occurred, or the original `LngLat` object.\n * @example\n * ```ts\n * let arr = [-73.9749, 40.7736];\n * let ll = maplibregl.LngLat.convert(arr);\n * ll; // = LngLat {lng: -73.9749, lat: 40.7736}\n * ```\n */\n static convert(input: LngLatLike): LngLat {\n if (input instanceof LngLat) {\n return input;\n }\n if (Array.isArray(input) && (input.length === 2 || input.length === 3)) {\n return new LngLat(Number(input[0]), Number(input[1]));\n }\n if (!Array.isArray(input) && typeof input === 'object' && input !== null) {\n return new LngLat(\n // flow can't refine this to have one of lng or lat, so we have to cast to any\n Number('lng' in input ? (input as any).lng : (input as any).lon),\n Number(input.lat)\n );\n }\n throw new Error('`LngLatLike` argument must be specified as a LngLat instance, an object {lng: , lat: }, an object {lon: , lat: }, or an array of [, ]');\n }\n}\n","import {LngLat, earthRadius} from '../geo/lng_lat';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport {IMercatorCoordinate} from '@maplibre/maplibre-gl-style-spec';\n\n/*\n * The average circumference of the world in meters.\n */\nconst earthCircumfrence = 2 * Math.PI * earthRadius; // meters\n\n/*\n * The circumference at a line of latitude in meters.\n */\nfunction circumferenceAtLatitude(latitude: number) {\n return earthCircumfrence * Math.cos(latitude * Math.PI / 180);\n}\n\nexport function mercatorXfromLng(lng: number) {\n return (180 + lng) / 360;\n}\n\nexport function mercatorYfromLat(lat: number) {\n return (180 - (180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360)))) / 360;\n}\n\nexport function mercatorZfromAltitude(altitude: number, lat: number) {\n return altitude / circumferenceAtLatitude(lat);\n}\n\nexport function lngFromMercatorX(x: number) {\n return x * 360 - 180;\n}\n\nexport function latFromMercatorY(y: number) {\n const y2 = 180 - y * 360;\n return 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90;\n}\n\nexport function altitudeFromMercatorZ(z: number, y: number) {\n return z * circumferenceAtLatitude(latFromMercatorY(y));\n}\n\n/**\n * Determine the Mercator scale factor for a given latitude, see\n * https://en.wikipedia.org/wiki/Mercator_projection#Scale_factor\n *\n * At the equator the scale factor will be 1, which increases at higher latitudes.\n *\n * @param lat - Latitude\n * @returns scale factor\n */\nexport function mercatorScale(lat: number) {\n return 1 / Math.cos(lat * Math.PI / 180);\n}\n\n/**\n * A `MercatorCoordinate` object represents a projected three dimensional position.\n *\n * `MercatorCoordinate` uses the web mercator projection ([EPSG:3857](https://epsg.io/3857)) with slightly different units:\n * - the size of 1 unit is the width of the projected world instead of the \"mercator meter\"\n * - the origin of the coordinate space is at the north-west corner instead of the middle\n *\n * For example, `MercatorCoordinate(0, 0, 0)` is the north-west corner of the mercator world and\n * `MercatorCoordinate(1, 1, 0)` is the south-east corner. If you are familiar with\n * [vector tiles](https://github.com/mapbox/vector-tile-spec) it may be helpful to think\n * of the coordinate space as the `0/0/0` tile with an extent of `1`.\n *\n * The `z` dimension of `MercatorCoordinate` is conformal. A cube in the mercator coordinate space would be rendered as a cube.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let nullIsland = new maplibregl.MercatorCoordinate(0.5, 0.5, 0);\n * ```\n * @see [Add a custom style layer](https://maplibre.org/maplibre-gl-js/docs/examples/custom-style-layer/)\n */\nexport class MercatorCoordinate implements IMercatorCoordinate {\n x: number;\n y: number;\n z: number;\n\n /**\n * @param x - The x component of the position.\n * @param y - The y component of the position.\n * @param z - The z component of the position.\n */\n constructor(x: number, y: number, z: number = 0) {\n this.x = +x;\n this.y = +y;\n this.z = +z;\n }\n\n /**\n * Project a `LngLat` to a `MercatorCoordinate`.\n *\n * @param lngLatLike - The location to project.\n * @param altitude - The altitude in meters of the position.\n * @returns The projected mercator coordinate.\n * @example\n * ```ts\n * let coord = maplibregl.MercatorCoordinate.fromLngLat({ lng: 0, lat: 0}, 0);\n * coord; // MercatorCoordinate(0.5, 0.5, 0)\n * ```\n */\n static fromLngLat(lngLatLike: LngLatLike, altitude: number = 0): MercatorCoordinate {\n const lngLat = LngLat.convert(lngLatLike);\n\n return new MercatorCoordinate(\n mercatorXfromLng(lngLat.lng),\n mercatorYfromLat(lngLat.lat),\n mercatorZfromAltitude(altitude, lngLat.lat));\n }\n\n /**\n * Returns the `LngLat` for the coordinate.\n *\n * @returns The `LngLat` object.\n * @example\n * ```ts\n * let coord = new maplibregl.MercatorCoordinate(0.5, 0.5, 0);\n * let lngLat = coord.toLngLat(); // LngLat(0, 0)\n * ```\n */\n toLngLat() {\n return new LngLat(\n lngFromMercatorX(this.x),\n latFromMercatorY(this.y));\n }\n\n /**\n * Returns the altitude in meters of the coordinate.\n *\n * @returns The altitude in meters.\n * @example\n * ```ts\n * let coord = new maplibregl.MercatorCoordinate(0, 0, 0.02);\n * coord.toAltitude(); // 6914.281956295339\n * ```\n */\n toAltitude(): number {\n return altitudeFromMercatorZ(this.z, this.y);\n }\n\n /**\n * Returns the distance of 1 meter in `MercatorCoordinate` units at this latitude.\n *\n * For coordinates in real world units using meters, this naturally provides the scale\n * to transform into `MercatorCoordinate`s.\n *\n * @returns Distance of 1 meter in `MercatorCoordinate` units.\n */\n meterInMercatorCoordinateUnits(): number {\n // 1 meter / circumference at equator in meters * Mercator projection scale factor at this latitude\n return 1 / earthCircumfrence * mercatorScale(latFromMercatorY(this.y));\n }\n}\n","export { getURL, getTileBBox, getMercCoords };\n\n\n/**\n * getURL\n *\n * @param {String} baseUrl Base url of the WMS server\n * @param {String} layer Layer name\n * @param {Number} x Tile coordinate x\n * @param {Number} y Tile coordinate y\n * @param {Number} z Tile zoom\n * @param {Object} [options]\n * @param {String} [options.format='image/png']\n * @param {String} [options.service='WMS']\n * @param {String} [options.version='1.1.1']\n * @param {String} [options.request='GetMap']\n * @param {String} [options.srs='EPSG:3857']\n * @param {Number} [options.width='256']\n * @param {Number} [options.height='256']\n * @returns {String} url\n * @example\n * var baseUrl = 'http://geodata.state.nj.us/imagerywms/Natural2015';\n * var layer = 'Natural2015';\n * var url = whoots.getURL(baseUrl, layer, 154308, 197167, 19);\n */\nfunction getURL(baseUrl, layer, x, y, z, options) {\n options = options || {};\n\n var url = baseUrl + '?' + [\n 'bbox=' + getTileBBox(x, y, z),\n 'format=' + (options.format || 'image/png'),\n 'service=' + (options.service || 'WMS'),\n 'version=' + (options.version || '1.1.1'),\n 'request=' + (options.request || 'GetMap'),\n 'srs=' + (options.srs || 'EPSG:3857'),\n 'width=' + (options.width || 256),\n 'height=' + (options.height || 256),\n 'layers=' + layer\n ].join('&');\n\n return url;\n}\n\n\n/**\n * getTileBBox\n *\n * @param {Number} x Tile coordinate x\n * @param {Number} y Tile coordinate y\n * @param {Number} z Tile zoom\n * @returns {String} String of the bounding box\n */\nfunction getTileBBox(x, y, z) {\n // for Google/OSM tile scheme we need to alter the y\n y = (Math.pow(2, z) - y - 1);\n\n var min = getMercCoords(x * 256, y * 256, z),\n max = getMercCoords((x + 1) * 256, (y + 1) * 256, z);\n\n return min[0] + ',' + min[1] + ',' + max[0] + ',' + max[1];\n}\n\n\n/**\n * getMercCoords\n *\n * @param {Number} x Pixel coordinate x\n * @param {Number} y Pixel coordinate y\n * @param {Number} z Tile zoom\n * @returns {Array} [x, y]\n */\nfunction getMercCoords(x, y, z) {\n var resolution = (2 * Math.PI * 6378137 / 256) / Math.pow(2, z),\n merc_x = (x * resolution - 2 * Math.PI * 6378137 / 2.0),\n merc_y = (y * resolution - 2 * Math.PI * 6378137 / 2.0);\n\n return [merc_x, merc_y];\n}\n","import {getTileBBox} from '@mapbox/whoots-js';\nimport {EXTENT} from '../data/extent';\nimport Point from '@mapbox/point-geometry';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\nimport {register} from '../util/web_worker_transfer';\nimport {mat4} from 'gl-matrix';\nimport {ICanonicalTileID, IMercatorCoordinate} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A canonical way to define a tile ID\n */\nexport class CanonicalTileID implements ICanonicalTileID {\n z: number;\n x: number;\n y: number;\n key: string;\n\n constructor(z: number, x: number, y: number) {\n\n if (z < 0 || z > 25 || y < 0 || y >= Math.pow(2, z) || x < 0 || x >= Math.pow(2, z)) {\n throw new Error(`x=${x}, y=${y}, z=${z} outside of bounds. 0<=x<${Math.pow(2, z)}, 0<=y<${Math.pow(2, z)} 0<=z<=25 `);\n }\n\n this.z = z;\n this.x = x;\n this.y = y;\n this.key = calculateKey(0, z, z, x, y);\n }\n\n equals(id: ICanonicalTileID) {\n return this.z === id.z && this.x === id.x && this.y === id.y;\n }\n\n // given a list of urls, choose a url template and return a tile URL\n url(urls: Array, pixelRatio: number, scheme?: string | null) {\n const bbox = getTileBBox(this.x, this.y, this.z);\n const quadkey = getQuadkey(this.z, this.x, this.y);\n\n return urls[(this.x + this.y) % urls.length]\n .replace(/{prefix}/g, (this.x % 16).toString(16) + (this.y % 16).toString(16))\n .replace(/{z}/g, String(this.z))\n .replace(/{x}/g, String(this.x))\n .replace(/{y}/g, String(scheme === 'tms' ? (Math.pow(2, this.z) - this.y - 1) : this.y))\n .replace(/{ratio}/g, pixelRatio > 1 ? '@2x' : '')\n .replace(/{quadkey}/g, quadkey)\n .replace(/{bbox-epsg-3857}/g, bbox);\n }\n\n isChildOf(parent: ICanonicalTileID) {\n const dz = this.z - parent.z;\n return dz > 0 && parent.x === (this.x >> dz) && parent.y === (this.y >> dz);\n }\n\n getTilePoint(coord: IMercatorCoordinate) {\n const tilesAtZoom = Math.pow(2, this.z);\n return new Point(\n (coord.x * tilesAtZoom - this.x) * EXTENT,\n (coord.y * tilesAtZoom - this.y) * EXTENT);\n }\n\n toString() {\n return `${this.z}/${this.x}/${this.y}`;\n }\n}\n\n/**\n * @internal\n * An unwrapped tile identifier\n */\nexport class UnwrappedTileID {\n wrap: number;\n canonical: CanonicalTileID;\n key: string;\n\n constructor(wrap: number, canonical: CanonicalTileID) {\n this.wrap = wrap;\n this.canonical = canonical;\n this.key = calculateKey(wrap, canonical.z, canonical.z, canonical.x, canonical.y);\n }\n}\n\n/**\n * An overscaled tile identifier\n */\nexport class OverscaledTileID {\n overscaledZ: number;\n wrap: number;\n canonical: CanonicalTileID;\n key: string;\n posMatrix: mat4;\n\n constructor(overscaledZ: number, wrap: number, z: number, x: number, y: number) {\n if (overscaledZ < z) throw new Error(`overscaledZ should be >= z; overscaledZ = ${overscaledZ}; z = ${z}`);\n this.overscaledZ = overscaledZ;\n this.wrap = wrap;\n this.canonical = new CanonicalTileID(z, +x, +y);\n this.key = calculateKey(wrap, overscaledZ, z, x, y);\n }\n\n clone() {\n return new OverscaledTileID(this.overscaledZ, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y);\n }\n\n equals(id: OverscaledTileID) {\n return this.overscaledZ === id.overscaledZ && this.wrap === id.wrap && this.canonical.equals(id.canonical);\n }\n\n scaledTo(targetZ: number) {\n if (targetZ > this.overscaledZ) throw new Error(`targetZ > this.overscaledZ; targetZ = ${targetZ}; overscaledZ = ${this.overscaledZ}`);\n const zDifference = this.canonical.z - targetZ;\n if (targetZ > this.canonical.z) {\n return new OverscaledTileID(targetZ, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y);\n } else {\n return new OverscaledTileID(targetZ, this.wrap, targetZ, this.canonical.x >> zDifference, this.canonical.y >> zDifference);\n }\n }\n\n /*\n * calculateScaledKey is an optimization:\n * when withWrap == true, implements the same as this.scaledTo(z).key,\n * when withWrap == false, implements the same as this.scaledTo(z).wrapped().key.\n */\n calculateScaledKey(targetZ: number, withWrap: boolean): string {\n if (targetZ > this.overscaledZ) throw new Error(`targetZ > this.overscaledZ; targetZ = ${targetZ}; overscaledZ = ${this.overscaledZ}`);\n const zDifference = this.canonical.z - targetZ;\n if (targetZ > this.canonical.z) {\n return calculateKey(this.wrap * +withWrap, targetZ, this.canonical.z, this.canonical.x, this.canonical.y);\n } else {\n return calculateKey(this.wrap * +withWrap, targetZ, targetZ, this.canonical.x >> zDifference, this.canonical.y >> zDifference);\n }\n }\n\n isChildOf(parent: OverscaledTileID) {\n if (parent.wrap !== this.wrap) {\n // We can't be a child if we're in a different world copy\n return false;\n }\n const zDifference = this.canonical.z - parent.canonical.z;\n // We're first testing for z == 0, to avoid a 32 bit shift, which is undefined.\n return parent.overscaledZ === 0 || (\n parent.overscaledZ < this.overscaledZ &&\n parent.canonical.x === (this.canonical.x >> zDifference) &&\n parent.canonical.y === (this.canonical.y >> zDifference));\n }\n\n children(sourceMaxZoom: number) {\n if (this.overscaledZ >= sourceMaxZoom) {\n // return a single tile coord representing a an overscaled tile\n return [new OverscaledTileID(this.overscaledZ + 1, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y)];\n }\n\n const z = this.canonical.z + 1;\n const x = this.canonical.x * 2;\n const y = this.canonical.y * 2;\n return [\n new OverscaledTileID(z, this.wrap, z, x, y),\n new OverscaledTileID(z, this.wrap, z, x + 1, y),\n new OverscaledTileID(z, this.wrap, z, x, y + 1),\n new OverscaledTileID(z, this.wrap, z, x + 1, y + 1)\n ];\n }\n\n isLessThan(rhs: OverscaledTileID) {\n if (this.wrap < rhs.wrap) return true;\n if (this.wrap > rhs.wrap) return false;\n\n if (this.overscaledZ < rhs.overscaledZ) return true;\n if (this.overscaledZ > rhs.overscaledZ) return false;\n\n if (this.canonical.x < rhs.canonical.x) return true;\n if (this.canonical.x > rhs.canonical.x) return false;\n\n if (this.canonical.y < rhs.canonical.y) return true;\n return false;\n }\n\n wrapped() {\n return new OverscaledTileID(this.overscaledZ, 0, this.canonical.z, this.canonical.x, this.canonical.y);\n }\n\n unwrapTo(wrap: number) {\n return new OverscaledTileID(this.overscaledZ, wrap, this.canonical.z, this.canonical.x, this.canonical.y);\n }\n\n overscaleFactor() {\n return Math.pow(2, this.overscaledZ - this.canonical.z);\n }\n\n toUnwrapped() {\n return new UnwrappedTileID(this.wrap, this.canonical);\n }\n\n toString() {\n return `${this.overscaledZ}/${this.canonical.x}/${this.canonical.y}`;\n }\n\n getTilePoint(coord: MercatorCoordinate) {\n return this.canonical.getTilePoint(new MercatorCoordinate(coord.x - this.wrap, coord.y));\n }\n}\n\nfunction calculateKey(wrap: number, overscaledZ: number, z: number, x: number, y: number): string {\n wrap *= 2;\n if (wrap < 0) wrap = wrap * -1 - 1;\n const dim = 1 << z;\n return (dim * dim * wrap + dim * y + x).toString(36) + z.toString(36) + overscaledZ.toString(36);\n}\n\nfunction getQuadkey(z, x, y) {\n let quadkey = '', mask;\n for (let i = z; i > 0; i--) {\n mask = 1 << (i - 1);\n quadkey += ((x & mask ? 1 : 0) + (y & mask ? 2 : 0));\n }\n return quadkey;\n}\n\nregister('CanonicalTileID', CanonicalTileID);\nregister('OverscaledTileID', OverscaledTileID, {omit: ['posMatrix']});\n","import {RGBAImage} from '../util/image';\n\nimport {warnOnce} from '../util/util';\nimport {register} from '../util/web_worker_transfer';\n\n// DEMData is a data structure for decoding, backfilling, and storing elevation data for processing in the hillshade shaders\n// data can be populated either from a pngraw image tile or from serliazed data sent back from a worker. When data is initially\n// loaded from a image tile, we decode the pixel values using the appropriate decoding formula, but we store the\n// elevation data as an Int32 value. we add 65536 (2^16) to eliminate negative values and enable the use of\n// integer overflow when creating the texture used in the hillshadePrepare step.\n\n// DEMData also handles the backfilling of data from a tile's neighboring tiles. This is necessary because we use a pixel's 8\n// surrounding pixel values to compute the slope at that pixel, and we cannot accurately calculate the slope at pixels on a\n// tile's edge without backfilling from neighboring tiles.\n\nexport type DEMEncoding = 'mapbox' | 'terrarium' | 'custom'\n\nexport class DEMData {\n uid: string;\n data: Uint32Array;\n stride: number;\n dim: number;\n min: number;\n max: number;\n redFactor: number;\n greenFactor: number;\n blueFactor: number;\n baseShift: number;\n\n // RGBAImage data has uniform 1px padding on all sides: square tile edge size defines stride\n // and dim is calculated as stride - 2.\n constructor(uid: string, data: RGBAImage, encoding: DEMEncoding, redFactor = 1.0, greenFactor = 1.0, blueFactor = 1.0, baseShift = 0.0) {\n this.uid = uid;\n if (data.height !== data.width) throw new RangeError('DEM tiles must be square');\n if (encoding && !['mapbox', 'terrarium', 'custom'].includes(encoding)) {\n warnOnce(`\"${encoding}\" is not a valid encoding type. Valid types include \"mapbox\", \"terrarium\" and \"custom\".`);\n return;\n }\n this.stride = data.height;\n const dim = this.dim = data.height - 2;\n this.data = new Uint32Array(data.data.buffer);\n switch (encoding) {\n case 'terrarium':\n // unpacking formula for mapzen terrarium:\n // https://aws.amazon.com/public-datasets/terrain/\n this.redFactor = 256.0;\n this.greenFactor = 1.0;\n this.blueFactor = 1.0 / 256.0;\n this.baseShift = 32768.0;\n break;\n case 'custom':\n this.redFactor = redFactor;\n this.greenFactor = greenFactor;\n this.blueFactor = blueFactor;\n this.baseShift = baseShift;\n break;\n case 'mapbox':\n default:\n // unpacking formula for mapbox.terrain-rgb:\n // https://www.mapbox.com/help/access-elevation-data/#mapbox-terrain-rgb\n this.redFactor = 6553.6;\n this.greenFactor = 25.6;\n this.blueFactor = 0.1;\n this.baseShift = 10000.0;\n break;\n }\n\n // in order to avoid flashing seams between tiles, here we are initially populating a 1px border of pixels around the image\n // with the data of the nearest pixel from the image. this data is eventually replaced when the tile's neighboring\n // tiles are loaded and the accurate data can be backfilled using DEMData#backfillBorder\n for (let x = 0; x < dim; x++) {\n // left vertical border\n this.data[this._idx(-1, x)] = this.data[this._idx(0, x)];\n // right vertical border\n this.data[this._idx(dim, x)] = this.data[this._idx(dim - 1, x)];\n // left horizontal border\n this.data[this._idx(x, -1)] = this.data[this._idx(x, 0)];\n // right horizontal border\n this.data[this._idx(x, dim)] = this.data[this._idx(x, dim - 1)];\n }\n // corners\n this.data[this._idx(-1, -1)] = this.data[this._idx(0, 0)];\n this.data[this._idx(dim, -1)] = this.data[this._idx(dim - 1, 0)];\n this.data[this._idx(-1, dim)] = this.data[this._idx(0, dim - 1)];\n this.data[this._idx(dim, dim)] = this.data[this._idx(dim - 1, dim - 1)];\n\n // calculate min/max values\n this.min = Number.MAX_SAFE_INTEGER;\n this.max = Number.MIN_SAFE_INTEGER;\n for (let x = 0; x < dim; x++) {\n for (let y = 0; y < dim; y++) {\n const ele = this.get(x, y);\n if (ele > this.max) this.max = ele;\n if (ele < this.min) this.min = ele;\n }\n }\n }\n\n get(x: number, y: number) {\n const pixels = new Uint8Array(this.data.buffer);\n const index = this._idx(x, y) * 4;\n return this.unpack(pixels[index], pixels[index + 1], pixels[index + 2]);\n }\n\n getUnpackVector() {\n return [this.redFactor, this.greenFactor, this.blueFactor, this.baseShift];\n }\n\n _idx(x: number, y: number) {\n if (x < -1 || x >= this.dim + 1 || y < -1 || y >= this.dim + 1) throw new RangeError('out of range source coordinates for DEM data');\n return (y + 1) * this.stride + (x + 1);\n }\n\n unpack(r: number, g: number, b: number) {\n return (r * this.redFactor + g * this.greenFactor + b * this.blueFactor - this.baseShift);\n }\n\n getPixels() {\n return new RGBAImage({width: this.stride, height: this.stride}, new Uint8Array(this.data.buffer));\n }\n\n backfillBorder(borderTile: DEMData, dx: number, dy: number) {\n if (this.dim !== borderTile.dim) throw new Error('dem dimension mismatch');\n\n let xMin = dx * this.dim,\n xMax = dx * this.dim + this.dim,\n yMin = dy * this.dim,\n yMax = dy * this.dim + this.dim;\n\n switch (dx) {\n case -1:\n xMin = xMax - 1;\n break;\n case 1:\n xMax = xMin + 1;\n break;\n }\n\n switch (dy) {\n case -1:\n yMin = yMax - 1;\n break;\n case 1:\n yMax = yMin + 1;\n break;\n }\n\n const ox = -dx * this.dim;\n const oy = -dy * this.dim;\n for (let y = yMin; y < yMax; y++) {\n for (let x = xMin; x < xMax; x++) {\n this.data[this._idx(x, y)] = borderTile.data[this._idx(x + ox, y + oy)];\n }\n }\n }\n}\n\nregister('DEMData', DEMData);\n","export class DictionaryCoder {\n _stringToNumber: {[_: string]: number};\n _numberToString: Array;\n\n constructor(strings: Array) {\n this._stringToNumber = {};\n this._numberToString = [];\n for (let i = 0; i < strings.length; i++) {\n const string = strings[i];\n this._stringToNumber[string] = i;\n this._numberToString[i] = string;\n }\n }\n\n encode(string: string) {\n return this._stringToNumber[string];\n }\n\n decode(n: number) {\n if (n >= this._numberToString.length) throw new Error(`Out of bounds. Index requested n=${n} can't be >= this._numberToString.length ${this._numberToString.length}`);\n return this._numberToString[n];\n }\n}\n","import type {VectorTileFeature} from '@mapbox/vector-tile';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A helper for type to omit a property from a type\n */\ntype DistributiveKeys = T extends T ? keyof T : never;\n/**\n * A helper for type to omit a property from a type\n */\ntype DistributiveOmit> = T extends unknown\n ? Omit\n : never;\n\n/**\n * An extended geojson feature used by the events to return data to the listener\n */\nexport type MapGeoJSONFeature = GeoJSONFeature & {\n layer: DistributiveOmit & {source: string};\n source: string;\n sourceLayer?: string;\n state: { [key: string]: any };\n}\n\n/**\n * A geojson feature\n */\nexport class GeoJSONFeature {\n type: 'Feature';\n _geometry: GeoJSON.Geometry;\n properties: { [name: string]: any };\n id: number | string | undefined;\n\n _vectorTileFeature: VectorTileFeature;\n\n constructor(vectorTileFeature: VectorTileFeature, z: number, x: number, y: number, id: string | number | undefined) {\n this.type = 'Feature';\n\n this._vectorTileFeature = vectorTileFeature;\n (vectorTileFeature as any)._z = z;\n (vectorTileFeature as any)._x = x;\n (vectorTileFeature as any)._y = y;\n\n this.properties = vectorTileFeature.properties;\n this.id = id;\n }\n\n get geometry(): GeoJSON.Geometry {\n if (this._geometry === undefined) {\n this._geometry = this._vectorTileFeature.toGeoJSON(\n (this._vectorTileFeature as any)._x,\n (this._vectorTileFeature as any)._y,\n (this._vectorTileFeature as any)._z).geometry;\n }\n return this._geometry;\n }\n\n set geometry(g: GeoJSON.Geometry) {\n this._geometry = g;\n }\n\n toJSON() {\n const json: any = {\n geometry: this.geometry\n };\n for (const i in this) {\n if (i === '_geometry' || i === '_vectorTileFeature') continue;\n json[i] = (this)[i];\n }\n return json;\n }\n}\n","import Point from '@mapbox/point-geometry';\nimport {loadGeometry} from './load_geometry';\nimport {toEvaluationFeature} from './evaluation_feature';\nimport {EXTENT} from './extent';\nimport {featureFilter} from '@maplibre/maplibre-gl-style-spec';\nimport {TransferableGridIndex} from '../util/transferable_grid_index';\nimport {DictionaryCoder} from '../util/dictionary_coder';\nimport vt from '@mapbox/vector-tile';\nimport Protobuf from 'pbf';\nimport {GeoJSONFeature} from '../util/vectortile_to_geojson';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\nimport {arraysIntersect, mapObject, extend} from '../util/util';\nimport {OverscaledTileID} from '../source/tile_id';\nimport {register} from '../util/web_worker_transfer';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {SourceFeatureState} from '../source/source_state';\nimport {polygonIntersectsBox} from '../util/intersection_tests';\nimport {PossiblyEvaluated} from '../style/properties';\nimport {FeatureIndexArray} from './array_types.g';\nimport {mat4} from 'gl-matrix';\n\nimport type {StyleLayer} from '../style/style_layer';\nimport type {FeatureFilter, FeatureState, FilterSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Transform} from '../geo/transform';\nimport type {VectorTileFeature, VectorTileLayer} from '@mapbox/vector-tile';\n\ntype QueryParameters = {\n scale: number;\n pixelPosMatrix: mat4;\n transform: Transform;\n tileSize: number;\n queryGeometry: Array;\n cameraQueryGeometry: Array;\n queryPadding: number;\n params: {\n filter: FilterSpecification;\n layers: Array;\n availableImages: Array;\n };\n};\n\n/**\n * @internal\n * An in memory index class to allow fast interaction with features\n */\nexport class FeatureIndex {\n tileID: OverscaledTileID;\n x: number;\n y: number;\n z: number;\n grid: TransferableGridIndex;\n grid3D: TransferableGridIndex;\n featureIndexArray: FeatureIndexArray;\n promoteId?: PromoteIdSpecification;\n\n rawTileData: ArrayBuffer;\n bucketLayerIDs: Array>;\n\n vtLayers: {[_: string]: VectorTileLayer};\n sourceLayerCoder: DictionaryCoder;\n\n constructor(tileID: OverscaledTileID, promoteId?: PromoteIdSpecification | null) {\n this.tileID = tileID;\n this.x = tileID.canonical.x;\n this.y = tileID.canonical.y;\n this.z = tileID.canonical.z;\n this.grid = new TransferableGridIndex(EXTENT, 16, 0);\n this.grid3D = new TransferableGridIndex(EXTENT, 16, 0);\n this.featureIndexArray = new FeatureIndexArray();\n this.promoteId = promoteId;\n }\n\n insert(feature: VectorTileFeature, geometry: Array>, featureIndex: number, sourceLayerIndex: number, bucketIndex: number, is3D?: boolean) {\n const key = this.featureIndexArray.length;\n this.featureIndexArray.emplaceBack(featureIndex, sourceLayerIndex, bucketIndex);\n\n const grid = is3D ? this.grid3D : this.grid;\n\n for (let r = 0; r < geometry.length; r++) {\n const ring = geometry[r];\n\n const bbox = [Infinity, Infinity, -Infinity, -Infinity];\n for (let i = 0; i < ring.length; i++) {\n const p = ring[i];\n bbox[0] = Math.min(bbox[0], p.x);\n bbox[1] = Math.min(bbox[1], p.y);\n bbox[2] = Math.max(bbox[2], p.x);\n bbox[3] = Math.max(bbox[3], p.y);\n }\n\n if (bbox[0] < EXTENT &&\n bbox[1] < EXTENT &&\n bbox[2] >= 0 &&\n bbox[3] >= 0) {\n grid.insert(key, bbox[0], bbox[1], bbox[2], bbox[3]);\n }\n }\n }\n\n loadVTLayers(): {[_: string]: VectorTileLayer} {\n if (!this.vtLayers) {\n this.vtLayers = new vt.VectorTile(new Protobuf(this.rawTileData)).layers;\n this.sourceLayerCoder = new DictionaryCoder(this.vtLayers ? Object.keys(this.vtLayers).sort() : ['_geojsonTileLayer']);\n }\n return this.vtLayers;\n }\n\n // Finds non-symbol features in this tile at a particular position.\n query(\n args: QueryParameters,\n styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n sourceFeatureState: SourceFeatureState\n ): {[_: string]: Array<{featureIndex: number; feature: GeoJSONFeature}>} {\n this.loadVTLayers();\n\n const params = args.params || {} as { filter: any; layers: string[]; availableImages: string[] },\n pixelsToTileUnits = EXTENT / args.tileSize / args.scale,\n filter = featureFilter(params.filter);\n\n const queryGeometry = args.queryGeometry;\n const queryPadding = args.queryPadding * pixelsToTileUnits;\n\n const bounds = getBounds(queryGeometry);\n const matching = this.grid.query(bounds.minX - queryPadding, bounds.minY - queryPadding, bounds.maxX + queryPadding, bounds.maxY + queryPadding);\n\n const cameraBounds = getBounds(args.cameraQueryGeometry);\n const matching3D = this.grid3D.query(\n cameraBounds.minX - queryPadding, cameraBounds.minY - queryPadding, cameraBounds.maxX + queryPadding, cameraBounds.maxY + queryPadding,\n (bx1, by1, bx2, by2) => {\n return polygonIntersectsBox(args.cameraQueryGeometry, bx1 - queryPadding, by1 - queryPadding, bx2 + queryPadding, by2 + queryPadding);\n });\n\n for (const key of matching3D) {\n matching.push(key);\n }\n\n matching.sort(topDownFeatureComparator);\n\n const result = {};\n let previousIndex;\n for (let k = 0; k < matching.length; k++) {\n const index = matching[k];\n\n // don't check the same feature more than once\n if (index === previousIndex) continue;\n previousIndex = index;\n\n const match = this.featureIndexArray.get(index);\n let featureGeometry = null;\n this.loadMatchingFeature(\n result,\n match.bucketIndex,\n match.sourceLayerIndex,\n match.featureIndex,\n filter,\n params.layers,\n params.availableImages,\n styleLayers,\n serializedLayers,\n sourceFeatureState,\n (feature: VectorTileFeature, styleLayer: StyleLayer, featureState: FeatureState) => {\n if (!featureGeometry) {\n featureGeometry = loadGeometry(feature);\n }\n\n return styleLayer.queryIntersectsFeature(queryGeometry, feature, featureState, featureGeometry, this.z, args.transform, pixelsToTileUnits, args.pixelPosMatrix);\n }\n );\n }\n\n return result;\n }\n\n loadMatchingFeature(\n result: {\n [_: string]: Array<{\n featureIndex: number;\n feature: GeoJSONFeature;\n intersectionZ?: boolean | number;\n }>;\n },\n bucketIndex: number,\n sourceLayerIndex: number,\n featureIndex: number,\n filter: FeatureFilter,\n filterLayerIDs: Array,\n availableImages: Array,\n styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n sourceFeatureState?: SourceFeatureState,\n intersectionTest?: (\n feature: VectorTileFeature,\n styleLayer: StyleLayer,\n featureState: any,\n id: string | number | void\n ) => boolean | number) {\n\n const layerIDs = this.bucketLayerIDs[bucketIndex];\n if (filterLayerIDs && !arraysIntersect(filterLayerIDs, layerIDs))\n return;\n\n const sourceLayerName = this.sourceLayerCoder.decode(sourceLayerIndex);\n const sourceLayer = this.vtLayers[sourceLayerName];\n const feature = sourceLayer.feature(featureIndex);\n\n if (filter.needGeometry) {\n const evaluationFeature = toEvaluationFeature(feature, true);\n if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), evaluationFeature, this.tileID.canonical)) {\n return;\n }\n } else if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), feature)) {\n return;\n }\n\n const id = this.getId(feature, sourceLayerName);\n\n for (let l = 0; l < layerIDs.length; l++) {\n const layerID = layerIDs[l];\n\n if (filterLayerIDs && filterLayerIDs.indexOf(layerID) < 0) {\n continue;\n }\n\n const styleLayer = styleLayers[layerID];\n\n if (!styleLayer) continue;\n\n let featureState = {};\n if (id && sourceFeatureState) {\n // `feature-state` expression evaluation requires feature state to be available\n featureState = sourceFeatureState.getState(styleLayer.sourceLayer || '_geojsonTileLayer', id);\n }\n\n const serializedLayer = extend({}, serializedLayers[layerID]);\n\n serializedLayer.paint = evaluateProperties(serializedLayer.paint, styleLayer.paint, feature, featureState, availableImages);\n serializedLayer.layout = evaluateProperties(serializedLayer.layout, styleLayer.layout, feature, featureState, availableImages);\n\n const intersectionZ = !intersectionTest || intersectionTest(feature, styleLayer, featureState);\n if (!intersectionZ) {\n // Only applied for non-symbol features\n continue;\n }\n\n const geojsonFeature = new GeoJSONFeature(feature, this.z, this.x, this.y, id) as MapGeoJSONFeature;\n geojsonFeature.layer = serializedLayer;\n let layerResult = result[layerID];\n if (layerResult === undefined) {\n layerResult = result[layerID] = [];\n }\n layerResult.push({featureIndex, feature: geojsonFeature, intersectionZ});\n }\n }\n\n // Given a set of symbol indexes that have already been looked up,\n // return a matching set of GeoJSONFeatures\n lookupSymbolFeatures(symbolFeatureIndexes: Array,\n serializedLayers: {[_: string]: StyleLayer},\n bucketIndex: number,\n sourceLayerIndex: number,\n filterSpec: FilterSpecification,\n filterLayerIDs: Array,\n availableImages: Array,\n styleLayers: {[_: string]: StyleLayer}) {\n const result = {};\n this.loadVTLayers();\n\n const filter = featureFilter(filterSpec);\n\n for (const symbolFeatureIndex of symbolFeatureIndexes) {\n this.loadMatchingFeature(\n result,\n bucketIndex,\n sourceLayerIndex,\n symbolFeatureIndex,\n filter,\n filterLayerIDs,\n availableImages,\n styleLayers,\n serializedLayers\n );\n\n }\n return result;\n }\n\n hasLayer(id: string) {\n for (const layerIDs of this.bucketLayerIDs) {\n for (const layerID of layerIDs) {\n if (id === layerID) return true;\n }\n }\n\n return false;\n }\n\n getId(feature: VectorTileFeature, sourceLayerId: string): string | number {\n let id: string | number = feature.id;\n if (this.promoteId) {\n const propName = typeof this.promoteId === 'string' ? this.promoteId : this.promoteId[sourceLayerId];\n id = feature.properties[propName] as string | number;\n if (typeof id === 'boolean') id = Number(id);\n }\n return id;\n }\n}\n\nregister(\n 'FeatureIndex',\n FeatureIndex,\n {omit: ['rawTileData', 'sourceLayerCoder']}\n);\n\nfunction evaluateProperties(serializedProperties, styleLayerProperties, feature, featureState, availableImages) {\n return mapObject(serializedProperties, (property, key) => {\n const prop = styleLayerProperties instanceof PossiblyEvaluated ? styleLayerProperties.get(key) : null;\n return prop && prop.evaluate ? prop.evaluate(feature, featureState, availableImages) : prop;\n });\n}\n\nfunction getBounds(geometry: Array) {\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n for (const p of geometry) {\n minX = Math.min(minX, p.x);\n minY = Math.min(minY, p.y);\n maxX = Math.max(maxX, p.x);\n maxY = Math.max(maxY, p.y);\n }\n return {minX, minY, maxX, maxY};\n}\n\nfunction topDownFeatureComparator(a, b) {\n return b - a;\n}\n","import Point from '@mapbox/point-geometry';\n\n/**\n * Returns the part of a multiline that intersects with the provided rectangular box.\n *\n * @param lines - the lines to check\n * @param x1 - the left edge of the box\n * @param y1 - the top edge of the box\n * @param x2 - the right edge of the box\n * @param y2 - the bottom edge of the box\n * @returns lines\n */\nexport function clipLine(lines: Array>, x1: number, y1: number, x2: number, y2: number): Array> {\n const clippedLines = [];\n\n for (let l = 0; l < lines.length; l++) {\n const line = lines[l];\n let clippedLine;\n\n for (let i = 0; i < line.length - 1; i++) {\n let p0 = line[i];\n let p1 = line[i + 1];\n\n if (p0.x < x1 && p1.x < x1) {\n continue;\n } else if (p0.x < x1) {\n p0 = new Point(x1, p0.y + (p1.y - p0.y) * ((x1 - p0.x) / (p1.x - p0.x)))._round();\n } else if (p1.x < x1) {\n p1 = new Point(x1, p0.y + (p1.y - p0.y) * ((x1 - p0.x) / (p1.x - p0.x)))._round();\n }\n\n if (p0.y < y1 && p1.y < y1) {\n continue;\n } else if (p0.y < y1) {\n p0 = new Point(p0.x + (p1.x - p0.x) * ((y1 - p0.y) / (p1.y - p0.y)), y1)._round();\n } else if (p1.y < y1) {\n p1 = new Point(p0.x + (p1.x - p0.x) * ((y1 - p0.y) / (p1.y - p0.y)), y1)._round();\n }\n\n if (p0.x >= x2 && p1.x >= x2) {\n continue;\n } else if (p0.x >= x2) {\n p0 = new Point(x2, p0.y + (p1.y - p0.y) * ((x2 - p0.x) / (p1.x - p0.x)))._round();\n } else if (p1.x >= x2) {\n p1 = new Point(x2, p0.y + (p1.y - p0.y) * ((x2 - p0.x) / (p1.x - p0.x)))._round();\n }\n\n if (p0.y >= y2 && p1.y >= y2) {\n continue;\n } else if (p0.y >= y2) {\n p0 = new Point(p0.x + (p1.x - p0.x) * ((y2 - p0.y) / (p1.y - p0.y)), y2)._round();\n } else if (p1.y >= y2) {\n p1 = new Point(p0.x + (p1.x - p0.x) * ((y2 - p0.y) / (p1.y - p0.y)), y2)._round();\n }\n\n if (!clippedLine || !p0.equals(clippedLine[clippedLine.length - 1])) {\n clippedLine = [p0];\n clippedLines.push(clippedLine);\n }\n\n clippedLine.push(p1);\n }\n }\n\n return clippedLines;\n}\n","import Point from '@mapbox/point-geometry';\n\nimport {register} from '../util/web_worker_transfer';\n\nexport class Anchor extends Point {\n angle: any;\n segment?: number;\n\n constructor(x: number, y: number, angle: number, segment?: number) {\n super(x, y);\n this.angle = angle;\n if (segment !== undefined) {\n this.segment = segment;\n }\n }\n\n clone() {\n return new Anchor(this.x, this.y, this.angle, this.segment);\n }\n}\n\nregister('Anchor', Anchor);\n","import type Point from '@mapbox/point-geometry';\nimport type {Anchor} from './anchor';\n\n/**\n * Labels placed around really sharp angles aren't readable. Check if any\n * part of the potential label has a combined angle that is too big.\n *\n * @param line - The line to check\n * @param anchor - The point on the line around which the label is anchored.\n * @param labelLength - The length of the label in geometry units.\n * @param windowSize - The check fails if the combined angles within a part of the line that is `windowSize` long is too big.\n * @param maxAngle - The maximum combined angle that any window along the label is allowed to have.\n *\n * @returns whether the label should be placed\n */\nexport function checkMaxAngle(line: Array, anchor: Anchor, labelLength: number, windowSize: number, maxAngle: number) {\n\n // horizontal labels and labels with length 0 always pass\n if (anchor.segment === undefined || labelLength === 0) return true;\n\n let p = anchor;\n let index = anchor.segment + 1;\n let anchorDistance = 0;\n\n // move backwards along the line to the first segment the label appears on\n while (anchorDistance > -labelLength / 2) {\n index--;\n\n // there isn't enough room for the label after the beginning of the line\n if (index < 0) return false;\n\n anchorDistance -= line[index].dist(p);\n p = line[index];\n }\n\n anchorDistance += line[index].dist(line[index + 1]);\n index++;\n\n // store recent corners and their total angle difference\n const recentCorners = [];\n let recentAngleDelta = 0;\n\n // move forwards by the length of the label and check angles along the way\n while (anchorDistance < labelLength / 2) {\n const prev = line[index - 1];\n const current = line[index];\n const next = line[index + 1];\n\n // there isn't enough room for the label before the end of the line\n if (!next) return false;\n\n let angleDelta = prev.angleTo(current) - current.angleTo(next);\n // restrict angle to -pi..pi range\n angleDelta = Math.abs(((angleDelta + 3 * Math.PI) % (Math.PI * 2)) - Math.PI);\n\n recentCorners.push({\n distance: anchorDistance,\n angleDelta\n });\n recentAngleDelta += angleDelta;\n\n // remove corners that are far enough away from the list of recent anchors\n while (anchorDistance - recentCorners[0].distance > windowSize) {\n recentAngleDelta -= recentCorners.shift().angleDelta;\n }\n\n // the sum of angles within the window area exceeds the maximum allowed value. check fails.\n if (recentAngleDelta > maxAngle) return false;\n\n index++;\n anchorDistance += current.dist(next);\n }\n\n // no part of the line had an angle greater than the maximum allowed. check passes.\n return true;\n}\n","import {interpolates} from '@maplibre/maplibre-gl-style-spec';\n\nimport {Anchor} from '../symbol/anchor';\nimport {checkMaxAngle} from './check_max_angle';\n\nimport type Point from '@mapbox/point-geometry';\nimport type {Shaping, PositionedIcon} from './shaping';\n\nexport {getAnchors, getCenterAnchor};\n\nfunction getLineLength(line: Array): number {\n let lineLength = 0;\n for (let k = 0; k < line.length - 1; k++) {\n lineLength += line[k].dist(line[k + 1]);\n }\n return lineLength;\n}\n\nfunction getAngleWindowSize(\n shapedText: Shaping,\n glyphSize: number,\n boxScale: number\n): number {\n return shapedText ?\n 3 / 5 * glyphSize * boxScale :\n 0;\n}\n\nfunction getShapedLabelLength(shapedText?: Shaping | null, shapedIcon?: PositionedIcon | null): number {\n return Math.max(\n shapedText ? shapedText.right - shapedText.left : 0,\n shapedIcon ? shapedIcon.right - shapedIcon.left : 0);\n}\n\nfunction getCenterAnchor(line: Array,\n maxAngle: number,\n shapedText: Shaping,\n shapedIcon: PositionedIcon,\n glyphSize: number,\n boxScale: number) {\n const angleWindowSize = getAngleWindowSize(shapedText, glyphSize, boxScale);\n const labelLength = getShapedLabelLength(shapedText, shapedIcon) * boxScale;\n\n let prevDistance = 0;\n const centerDistance = getLineLength(line) / 2;\n\n for (let i = 0; i < line.length - 1; i++) {\n\n const a = line[i],\n b = line[i + 1];\n\n const segmentDistance = a.dist(b);\n\n if (prevDistance + segmentDistance > centerDistance) {\n // The center is on this segment\n const t = (centerDistance - prevDistance) / segmentDistance,\n x = interpolates.number(a.x, b.x, t),\n y = interpolates.number(a.y, b.y, t);\n\n const anchor = new Anchor(x, y, b.angleTo(a), i);\n anchor._round();\n if (!angleWindowSize || checkMaxAngle(line, anchor, labelLength, angleWindowSize, maxAngle)) {\n return anchor;\n } else {\n return;\n }\n }\n\n prevDistance += segmentDistance;\n }\n}\n\nfunction getAnchors(line: Array,\n spacing: number,\n maxAngle: number,\n shapedText: Shaping,\n shapedIcon: PositionedIcon,\n glyphSize: number,\n boxScale: number,\n overscaling: number,\n tileExtent: number) {\n\n // Resample a line to get anchor points for labels and check that each\n // potential label passes text-max-angle check and has enough froom to fit\n // on the line.\n\n const angleWindowSize = getAngleWindowSize(shapedText, glyphSize, boxScale);\n const shapedLabelLength = getShapedLabelLength(shapedText, shapedIcon);\n const labelLength = shapedLabelLength * boxScale;\n\n // Is the line continued from outside the tile boundary?\n const isLineContinued = line[0].x === 0 || line[0].x === tileExtent || line[0].y === 0 || line[0].y === tileExtent;\n\n // Is the label long, relative to the spacing?\n // If so, adjust the spacing so there is always a minimum space of `spacing / 4` between label edges.\n if (spacing - labelLength < spacing / 4) {\n spacing = labelLength + spacing / 4;\n }\n\n // Offset the first anchor by:\n // Either half the label length plus a fixed extra offset if the line is not continued\n // Or half the spacing if the line is continued.\n\n // For non-continued lines, add a bit of fixed extra offset to avoid collisions at T intersections.\n const fixedExtraOffset = glyphSize * 2;\n\n const offset = !isLineContinued ?\n ((shapedLabelLength / 2 + fixedExtraOffset) * boxScale * overscaling) % spacing :\n (spacing / 2 * overscaling) % spacing;\n\n return resample(line, offset, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, false, tileExtent);\n}\n\nfunction resample(line, offset, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, placeAtMiddle, tileExtent) {\n\n const halfLabelLength = labelLength / 2;\n const lineLength = getLineLength(line);\n\n let distance = 0,\n markedDistance = offset - spacing;\n\n let anchors = [];\n\n for (let i = 0; i < line.length - 1; i++) {\n\n const a = line[i],\n b = line[i + 1];\n\n const segmentDist = a.dist(b),\n angle = b.angleTo(a);\n\n while (markedDistance + spacing < distance + segmentDist) {\n markedDistance += spacing;\n\n const t = (markedDistance - distance) / segmentDist,\n x = interpolates.number(a.x, b.x, t),\n y = interpolates.number(a.y, b.y, t);\n\n // Check that the point is within the tile boundaries and that\n // the label would fit before the beginning and end of the line\n // if placed at this point.\n if (x >= 0 && x < tileExtent && y >= 0 && y < tileExtent &&\n markedDistance - halfLabelLength >= 0 &&\n markedDistance + halfLabelLength <= lineLength) {\n const anchor = new Anchor(x, y, angle, i);\n anchor._round();\n\n if (!angleWindowSize || checkMaxAngle(line, anchor, labelLength, angleWindowSize, maxAngle)) {\n anchors.push(anchor);\n }\n }\n }\n\n distance += segmentDist;\n }\n\n if (!placeAtMiddle && !anchors.length && !isLineContinued) {\n // The first attempt at finding anchors at which labels can be placed failed.\n // Try again, but this time just try placing one anchor at the middle of the line.\n // This has the most effect for short lines in overscaled tiles, since the\n // initial offset used in overscaled tiles is calculated to align labels with positions in\n // parent tiles instead of placing the label as close to the beginning as possible.\n anchors = resample(line, distance / 2, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, true, tileExtent);\n }\n\n return anchors;\n}\n","import Point from '@mapbox/point-geometry';\n\nimport {GLYPH_PBF_BORDER} from '../style/parse_glyph_pbf';\n\nimport type {Anchor} from './anchor';\nimport type {PositionedIcon, Shaping} from './shaping';\nimport {SHAPING_DEFAULT_OFFSET} from './shaping';\nimport {IMAGE_PADDING} from '../render/image_atlas';\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\nimport type {Feature} from '@maplibre/maplibre-gl-style-spec';\nimport type {StyleImage} from '../style/style_image';\nimport ONE_EM from './one_em';\nimport {Rect} from '../render/glyph_atlas';\n\n/**\n * A textured quad for rendering a single icon or glyph.\n *\n * The zoom range the glyph can be shown is defined by minScale and maxScale.\n *\n * @param tl - The offset of the top left corner from the anchor.\n * @param tr - The offset of the top right corner from the anchor.\n * @param bl - The offset of the bottom left corner from the anchor.\n * @param br - The offset of the bottom right corner from the anchor.\n * @param tex - The texture coordinates.\n */\nexport type SymbolQuad = {\n tl: Point;\n tr: Point;\n bl: Point;\n br: Point;\n tex: {\n x: number;\n y: number;\n w: number;\n h: number;\n };\n pixelOffsetTL: Point;\n pixelOffsetBR: Point;\n writingMode: any | void;\n glyphOffset: [number, number];\n sectionIndex: number;\n isSDF: boolean;\n minFontScaleX: number;\n minFontScaleY: number;\n};\n\n// If you have a 10px icon that isn't perfectly aligned to the pixel grid it will cover 11 actual\n// pixels. The quad needs to be padded to account for this, otherwise they'll look slightly clipped\n// on one edge in some cases.\nconst border = IMAGE_PADDING;\n\n/**\n * Create the quads used for rendering an icon.\n */\nexport function getIconQuads(\n shapedIcon: PositionedIcon,\n iconRotate: number,\n isSDFIcon: boolean,\n hasIconTextFit: boolean\n): Array {\n const quads = [];\n\n const image = shapedIcon.image;\n const pixelRatio = image.pixelRatio;\n const imageWidth = image.paddedRect.w - 2 * border;\n const imageHeight = image.paddedRect.h - 2 * border;\n\n const iconWidth = shapedIcon.right - shapedIcon.left;\n const iconHeight = shapedIcon.bottom - shapedIcon.top;\n\n const stretchX = image.stretchX || [[0, imageWidth]];\n const stretchY = image.stretchY || [[0, imageHeight]];\n\n const reduceRanges = (sum, range) => sum + range[1] - range[0];\n const stretchWidth = stretchX.reduce(reduceRanges, 0);\n const stretchHeight = stretchY.reduce(reduceRanges, 0);\n const fixedWidth = imageWidth - stretchWidth;\n const fixedHeight = imageHeight - stretchHeight;\n\n let stretchOffsetX = 0;\n let stretchContentWidth = stretchWidth;\n let stretchOffsetY = 0;\n let stretchContentHeight = stretchHeight;\n let fixedOffsetX = 0;\n let fixedContentWidth = fixedWidth;\n let fixedOffsetY = 0;\n let fixedContentHeight = fixedHeight;\n\n if (image.content && hasIconTextFit) {\n const content = image.content;\n stretchOffsetX = sumWithinRange(stretchX, 0, content[0]);\n stretchOffsetY = sumWithinRange(stretchY, 0, content[1]);\n stretchContentWidth = sumWithinRange(stretchX, content[0], content[2]);\n stretchContentHeight = sumWithinRange(stretchY, content[1], content[3]);\n fixedOffsetX = content[0] - stretchOffsetX;\n fixedOffsetY = content[1] - stretchOffsetY;\n fixedContentWidth = content[2] - content[0] - stretchContentWidth;\n fixedContentHeight = content[3] - content[1] - stretchContentHeight;\n }\n\n const makeBox = (left, top, right, bottom) => {\n\n const leftEm = getEmOffset(left.stretch - stretchOffsetX, stretchContentWidth, iconWidth, shapedIcon.left);\n const leftPx = getPxOffset(left.fixed - fixedOffsetX, fixedContentWidth, left.stretch, stretchWidth);\n\n const topEm = getEmOffset(top.stretch - stretchOffsetY, stretchContentHeight, iconHeight, shapedIcon.top);\n const topPx = getPxOffset(top.fixed - fixedOffsetY, fixedContentHeight, top.stretch, stretchHeight);\n\n const rightEm = getEmOffset(right.stretch - stretchOffsetX, stretchContentWidth, iconWidth, shapedIcon.left);\n const rightPx = getPxOffset(right.fixed - fixedOffsetX, fixedContentWidth, right.stretch, stretchWidth);\n\n const bottomEm = getEmOffset(bottom.stretch - stretchOffsetY, stretchContentHeight, iconHeight, shapedIcon.top);\n const bottomPx = getPxOffset(bottom.fixed - fixedOffsetY, fixedContentHeight, bottom.stretch, stretchHeight);\n\n const tl = new Point(leftEm, topEm);\n const tr = new Point(rightEm, topEm);\n const br = new Point(rightEm, bottomEm);\n const bl = new Point(leftEm, bottomEm);\n const pixelOffsetTL = new Point(leftPx / pixelRatio, topPx / pixelRatio);\n const pixelOffsetBR = new Point(rightPx / pixelRatio, bottomPx / pixelRatio);\n\n const angle = iconRotate * Math.PI / 180;\n\n if (angle) {\n const sin = Math.sin(angle),\n cos = Math.cos(angle),\n matrix = [cos, -sin, sin, cos];\n\n tl._matMult(matrix);\n tr._matMult(matrix);\n bl._matMult(matrix);\n br._matMult(matrix);\n }\n\n const x1 = left.stretch + left.fixed;\n const x2 = right.stretch + right.fixed;\n const y1 = top.stretch + top.fixed;\n const y2 = bottom.stretch + bottom.fixed;\n\n const subRect = {\n x: image.paddedRect.x + border + x1,\n y: image.paddedRect.y + border + y1,\n w: x2 - x1,\n h: y2 - y1\n };\n\n const minFontScaleX = fixedContentWidth / pixelRatio / iconWidth;\n const minFontScaleY = fixedContentHeight / pixelRatio / iconHeight;\n\n // Icon quad is padded, so texture coordinates also need to be padded.\n return {tl, tr, bl, br, tex: subRect, writingMode: undefined, glyphOffset: [0, 0], sectionIndex: 0, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY, isSDF: isSDFIcon};\n };\n\n if (!hasIconTextFit || (!image.stretchX && !image.stretchY)) {\n quads.push(makeBox(\n {fixed: 0, stretch: -1},\n {fixed: 0, stretch: -1},\n {fixed: 0, stretch: imageWidth + 1},\n {fixed: 0, stretch: imageHeight + 1}));\n } else {\n const xCuts = stretchZonesToCuts(stretchX, fixedWidth, stretchWidth);\n const yCuts = stretchZonesToCuts(stretchY, fixedHeight, stretchHeight);\n\n for (let xi = 0; xi < xCuts.length - 1; xi++) {\n const x1 = xCuts[xi];\n const x2 = xCuts[xi + 1];\n for (let yi = 0; yi < yCuts.length - 1; yi++) {\n const y1 = yCuts[yi];\n const y2 = yCuts[yi + 1];\n quads.push(makeBox(x1, y1, x2, y2));\n }\n }\n }\n\n return quads;\n}\n\nfunction sumWithinRange(ranges, min, max) {\n let sum = 0;\n for (const range of ranges) {\n sum += Math.max(min, Math.min(max, range[1])) - Math.max(min, Math.min(max, range[0]));\n }\n return sum;\n}\n\nfunction stretchZonesToCuts(stretchZones, fixedSize, stretchSize) {\n const cuts = [{fixed: -border, stretch: 0}];\n\n for (const [c1, c2] of stretchZones) {\n const last = cuts[cuts.length - 1];\n cuts.push({\n fixed: c1 - last.stretch,\n stretch: last.stretch\n });\n cuts.push({\n fixed: c1 - last.stretch,\n stretch: last.stretch + (c2 - c1)\n });\n }\n cuts.push({\n fixed: fixedSize + border,\n stretch: stretchSize\n });\n return cuts;\n}\n\nfunction getEmOffset(stretchOffset, stretchSize, iconSize, iconOffset) {\n return stretchOffset / stretchSize * iconSize + iconOffset;\n}\n\nfunction getPxOffset(fixedOffset, fixedSize, stretchOffset, stretchSize) {\n return fixedOffset - fixedSize * stretchOffset / stretchSize;\n}\n\n/**\n * Create the quads used for rendering a text label.\n */\nexport function getGlyphQuads(\n anchor: Anchor,\n shaping: Shaping,\n textOffset: [number, number],\n layer: SymbolStyleLayer,\n alongLine: boolean,\n feature: Feature,\n imageMap: {[_: string]: StyleImage},\n allowVerticalPlacement: boolean\n): Array {\n\n const textRotate = layer.layout.get('text-rotate').evaluate(feature, {}) * Math.PI / 180;\n const quads = [];\n\n for (const line of shaping.positionedLines) {\n for (const positionedGlyph of line.positionedGlyphs) {\n if (!positionedGlyph.rect) continue;\n const textureRect: Rect = positionedGlyph.rect || {} as Rect;\n\n // The rects have an additional buffer that is not included in their size.\n const glyphPadding = 1.0;\n let rectBuffer = GLYPH_PBF_BORDER + glyphPadding;\n let isSDF = true;\n let pixelRatio = 1.0;\n let lineOffset = 0.0;\n\n const rotateVerticalGlyph = (alongLine || allowVerticalPlacement) && positionedGlyph.vertical;\n const halfAdvance = positionedGlyph.metrics.advance * positionedGlyph.scale / 2;\n\n // Align images and scaled glyphs in the middle of a vertical line.\n if (allowVerticalPlacement && shaping.verticalizable) {\n const scaledGlyphOffset = (positionedGlyph.scale - 1) * ONE_EM;\n const imageOffset = (ONE_EM - positionedGlyph.metrics.width * positionedGlyph.scale) / 2;\n lineOffset = line.lineOffset / 2 - (positionedGlyph.imageName ? -imageOffset : scaledGlyphOffset);\n }\n\n if (positionedGlyph.imageName) {\n const image = imageMap[positionedGlyph.imageName];\n isSDF = image.sdf;\n pixelRatio = image.pixelRatio;\n rectBuffer = IMAGE_PADDING / pixelRatio;\n }\n\n const glyphOffset = alongLine ?\n [positionedGlyph.x + halfAdvance, positionedGlyph.y] :\n [0, 0];\n\n let builtInOffset: [number, number] = alongLine ?\n [0, 0] :\n [positionedGlyph.x + halfAdvance + textOffset[0], positionedGlyph.y + textOffset[1] - lineOffset];\n\n let verticalizedLabelOffset = [0, 0] as [number, number];\n if (rotateVerticalGlyph) {\n // Vertical POI labels that are rotated 90deg CW and whose glyphs must preserve upright orientation\n // need to be rotated 90deg CCW. After a quad is rotated, it is translated to the original built-in offset.\n verticalizedLabelOffset = builtInOffset;\n builtInOffset = [0, 0];\n }\n\n const textureScale = positionedGlyph.metrics.isDoubleResolution ? 2 : 1;\n\n const x1 = (positionedGlyph.metrics.left - rectBuffer) * positionedGlyph.scale - halfAdvance + builtInOffset[0];\n const y1 = (-positionedGlyph.metrics.top - rectBuffer) * positionedGlyph.scale + builtInOffset[1];\n const x2 = x1 + textureRect.w / textureScale * positionedGlyph.scale / pixelRatio;\n const y2 = y1 + textureRect.h / textureScale * positionedGlyph.scale / pixelRatio;\n\n const tl = new Point(x1, y1);\n const tr = new Point(x2, y1);\n const bl = new Point(x1, y2);\n const br = new Point(x2, y2);\n\n if (rotateVerticalGlyph) {\n // Vertical-supporting glyphs are laid out in 24x24 point boxes (1 square em)\n // In horizontal orientation, the y values for glyphs are below the midline\n // and we use a \"yOffset\" of -17 to pull them up to the middle.\n // By rotating counter-clockwise around the point at the center of the left\n // edge of a 24x24 layout box centered below the midline, we align the center\n // of the glyphs with the horizontal midline, so the yOffset is no longer\n // necessary, but we also pull the glyph to the left along the x axis.\n // The y coordinate includes baseline yOffset, thus needs to be accounted\n // for when glyph is rotated and translated.\n const center = new Point(-halfAdvance, halfAdvance - SHAPING_DEFAULT_OFFSET);\n const verticalRotation = -Math.PI / 2;\n\n // xHalfWidthOffsetCorrection is a difference between full-width and half-width\n // advance, should be 0 for full-width glyphs and will pull up half-width glyphs.\n const xHalfWidthOffsetCorrection = ONE_EM / 2 - halfAdvance;\n const yImageOffsetCorrection = positionedGlyph.imageName ? xHalfWidthOffsetCorrection : 0.0;\n const halfWidthOffsetCorrection = new Point(5 - SHAPING_DEFAULT_OFFSET - xHalfWidthOffsetCorrection, -yImageOffsetCorrection);\n const verticalOffsetCorrection = new Point(...verticalizedLabelOffset);\n tl._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection);\n tr._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection);\n bl._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection);\n br._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection);\n }\n\n if (textRotate) {\n const sin = Math.sin(textRotate),\n cos = Math.cos(textRotate),\n matrix = [cos, -sin, sin, cos];\n\n tl._matMult(matrix);\n tr._matMult(matrix);\n bl._matMult(matrix);\n br._matMult(matrix);\n }\n\n const pixelOffsetTL = new Point(0, 0);\n const pixelOffsetBR = new Point(0, 0);\n const minFontScaleX = 0;\n const minFontScaleY = 0;\n quads.push({tl, tr, bl, br, tex: textureRect, writingMode: shaping.writingMode, glyphOffset, sectionIndex: positionedGlyph.sectionIndex, isSDF, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY});\n }\n }\n\n return quads;\n}\n","import type {CollisionBoxArray} from '../data/array_types.g';\nimport Point from '@mapbox/point-geometry';\nimport type {Anchor} from './anchor';\nimport {SymbolPadding} from '../style/style_layer/symbol_style_layer';\n\n/**\n * A CollisionFeature represents the area of the tile covered by a single label.\n * It is used with CollisionIndex to check if the label overlaps with any\n * previous labels. A CollisionFeature is mostly just a set of CollisionBox\n * objects.\n */\nexport class CollisionFeature {\n boxStartIndex: number;\n boxEndIndex: number;\n circleDiameter: number;\n\n /**\n * Create a CollisionFeature, adding its collision box data to the given collisionBoxArray in the process.\n * For line aligned labels a collision circle diameter is computed instead.\n *\n * @param anchor - The point along the line around which the label is anchored.\n * @param shaped - The text or icon shaping results.\n * @param boxScale - A magic number used to convert from glyph metrics units to geometry units.\n * @param padding - The amount of padding to add around the label edges.\n * @param alignLine - Whether the label is aligned with the line or the viewport.\n */\n constructor(collisionBoxArray: CollisionBoxArray,\n anchor: Anchor,\n featureIndex: number,\n sourceLayerIndex: number,\n bucketIndex: number,\n shaped: any,\n boxScale: number,\n padding: SymbolPadding,\n alignLine: boolean,\n rotate: number) {\n\n this.boxStartIndex = collisionBoxArray.length;\n\n if (alignLine) {\n // Compute height of the shape in glyph metrics and apply collision padding.\n // Note that the pixel based 'text-padding' is applied at runtime\n let top = shaped.top;\n let bottom = shaped.bottom;\n const collisionPadding = shaped.collisionPadding;\n\n if (collisionPadding) {\n top -= collisionPadding[1];\n bottom += collisionPadding[3];\n }\n\n let height = bottom - top;\n\n if (height > 0) {\n // set minimum box height to avoid very many small labels\n height = Math.max(10, height);\n this.circleDiameter = height;\n }\n } else {\n // margin is in CSS order: [top, right, bottom, left]\n let y1 = shaped.top * boxScale - padding[0];\n let y2 = shaped.bottom * boxScale + padding[2];\n let x1 = shaped.left * boxScale - padding[3];\n let x2 = shaped.right * boxScale + padding[1];\n\n const collisionPadding = shaped.collisionPadding;\n if (collisionPadding) {\n x1 -= collisionPadding[0] * boxScale;\n y1 -= collisionPadding[1] * boxScale;\n x2 += collisionPadding[2] * boxScale;\n y2 += collisionPadding[3] * boxScale;\n }\n\n if (rotate) {\n // Account for *-rotate in point collision boxes\n // See https://github.com/mapbox/mapbox-gl-js/issues/6075\n // Doesn't account for icon-text-fit\n\n const tl = new Point(x1, y1);\n const tr = new Point(x2, y1);\n const bl = new Point(x1, y2);\n const br = new Point(x2, y2);\n\n const rotateRadians = rotate * Math.PI / 180;\n\n tl._rotate(rotateRadians);\n tr._rotate(rotateRadians);\n bl._rotate(rotateRadians);\n br._rotate(rotateRadians);\n\n // Collision features require an \"on-axis\" geometry,\n // so take the envelope of the rotated geometry\n // (may be quite large for wide labels rotated 45 degrees)\n x1 = Math.min(tl.x, tr.x, bl.x, br.x);\n x2 = Math.max(tl.x, tr.x, bl.x, br.x);\n y1 = Math.min(tl.y, tr.y, bl.y, br.y);\n y2 = Math.max(tl.y, tr.y, bl.y, br.y);\n }\n collisionBoxArray.emplaceBack(anchor.x, anchor.y, x1, y1, x2, y2, featureIndex, sourceLayerIndex, bucketIndex);\n }\n\n this.boxEndIndex = collisionBoxArray.length;\n }\n}\n","\nexport default class TinyQueue {\n constructor(data = [], compare = defaultCompare) {\n this.data = data;\n this.length = this.data.length;\n this.compare = compare;\n\n if (this.length > 0) {\n for (let i = (this.length >> 1) - 1; i >= 0; i--) this._down(i);\n }\n }\n\n push(item) {\n this.data.push(item);\n this.length++;\n this._up(this.length - 1);\n }\n\n pop() {\n if (this.length === 0) return undefined;\n\n const top = this.data[0];\n const bottom = this.data.pop();\n this.length--;\n\n if (this.length > 0) {\n this.data[0] = bottom;\n this._down(0);\n }\n\n return top;\n }\n\n peek() {\n return this.data[0];\n }\n\n _up(pos) {\n const {data, compare} = this;\n const item = data[pos];\n\n while (pos > 0) {\n const parent = (pos - 1) >> 1;\n const current = data[parent];\n if (compare(item, current) >= 0) break;\n data[pos] = current;\n pos = parent;\n }\n\n data[pos] = item;\n }\n\n _down(pos) {\n const {data, compare} = this;\n const halfLength = this.length >> 1;\n const item = data[pos];\n\n while (pos < halfLength) {\n let left = (pos << 1) + 1;\n let best = data[left];\n const right = left + 1;\n\n if (right < this.length && compare(data[right], best) < 0) {\n left = right;\n best = data[right];\n }\n if (compare(best, item) >= 0) break;\n\n data[pos] = best;\n pos = left;\n }\n\n data[pos] = item;\n }\n}\n\nfunction defaultCompare(a, b) {\n return a < b ? -1 : a > b ? 1 : 0;\n}\n","import Queue from 'tinyqueue';\n\nimport Point from '@mapbox/point-geometry';\nimport {distToSegmentSquared} from './intersection_tests';\n\n/**\n * Finds an approximation of a polygon's Pole Of Inaccessibiliy https://en.wikipedia.org/wiki/Pole_of_inaccessibility\n * This is a copy of http://github.com/mapbox/polylabel adapted to use Points\n *\n * @param polygonRings - first item in array is the outer ring followed optionally by the list of holes, should be an element of the result of util/classify_rings\n * @param precision - Specified in input coordinate units. If 0 returns after first run, if `> 0` repeatedly narrows the search space until the radius of the area searched for the best pole is less than precision\n * @param debug - Print some statistics to the console during execution\n * @returns Pole of Inaccessibiliy.\n */\nexport function findPoleOfInaccessibility(\n polygonRings: Array>,\n precision: number = 1,\n debug: boolean = false\n): Point {\n // find the bounding box of the outer ring\n let minX = Infinity, minY = Infinity, maxX = -Infinity, maxY = -Infinity;\n const outerRing = polygonRings[0];\n for (let i = 0; i < outerRing.length; i++) {\n const p = outerRing[i];\n if (!i || p.x < minX) minX = p.x;\n if (!i || p.y < minY) minY = p.y;\n if (!i || p.x > maxX) maxX = p.x;\n if (!i || p.y > maxY) maxY = p.y;\n }\n\n const width = maxX - minX;\n const height = maxY - minY;\n const cellSize = Math.min(width, height);\n let h = cellSize / 2;\n\n // a priority queue of cells in order of their \"potential\" (max distance to polygon)\n const cellQueue = new Queue([], compareMax);\n\n if (cellSize === 0) return new Point(minX, minY);\n\n // cover polygon with initial cells\n for (let x = minX; x < maxX; x += cellSize) {\n for (let y = minY; y < maxY; y += cellSize) {\n cellQueue.push(new Cell(x + h, y + h, h, polygonRings));\n }\n }\n\n // take centroid as the first best guess\n let bestCell = getCentroidCell(polygonRings);\n let numProbes = cellQueue.length;\n\n while (cellQueue.length) {\n // pick the most promising cell from the queue\n const cell = cellQueue.pop();\n\n // update the best cell if we found a better one\n if (cell.d > bestCell.d || !bestCell.d) {\n bestCell = cell;\n if (debug) console.log('found best %d after %d probes', Math.round(1e4 * cell.d) / 1e4, numProbes);\n }\n\n // do not drill down further if there's no chance of a better solution\n if (cell.max - bestCell.d <= precision) continue;\n\n // split the cell into four cells\n h = cell.h / 2;\n cellQueue.push(new Cell(cell.p.x - h, cell.p.y - h, h, polygonRings));\n cellQueue.push(new Cell(cell.p.x + h, cell.p.y - h, h, polygonRings));\n cellQueue.push(new Cell(cell.p.x - h, cell.p.y + h, h, polygonRings));\n cellQueue.push(new Cell(cell.p.x + h, cell.p.y + h, h, polygonRings));\n numProbes += 4;\n }\n\n if (debug) {\n console.log(`num probes: ${numProbes}`);\n console.log(`best distance: ${bestCell.d}`);\n }\n\n return bestCell.p;\n}\n\nfunction compareMax(a, b) {\n return b.max - a.max;\n}\n\nfunction Cell(x, y, h, polygon) {\n this.p = new Point(x, y);\n this.h = h; // half the cell size\n this.d = pointToPolygonDist(this.p, polygon); // distance from cell center to polygon\n this.max = this.d + this.h * Math.SQRT2; // max distance to polygon within a cell\n}\n\n// signed distance from point to polygon outline (negative if point is outside)\nfunction pointToPolygonDist(p, polygon) {\n let inside = false;\n let minDistSq = Infinity;\n\n for (let k = 0; k < polygon.length; k++) {\n const ring = polygon[k];\n\n for (let i = 0, len = ring.length, j = len - 1; i < len; j = i++) {\n const a = ring[i];\n const b = ring[j];\n\n if ((a.y > p.y !== b.y > p.y) &&\n (p.x < (b.x - a.x) * (p.y - a.y) / (b.y - a.y) + a.x)) inside = !inside;\n\n minDistSq = Math.min(minDistSq, distToSegmentSquared(p, a, b));\n }\n }\n\n return (inside ? 1 : -1) * Math.sqrt(minDistSq);\n}\n\n// get polygon centroid\nfunction getCentroidCell(polygon) {\n let area = 0;\n let x = 0;\n let y = 0;\n const points = polygon[0];\n for (let i = 0, len = points.length, j = len - 1; i < len; j = i++) {\n const a = points[i];\n const b = points[j];\n const f = a.x * b.y - b.x * a.y;\n x += (a.x + b.x) * f;\n y += (a.y + b.y) * f;\n area += f * 3;\n }\n return new Cell(x / area, y / area, 0, polygon);\n}\n","import {VariableAnchorOffsetCollection, VariableAnchorOffsetCollectionSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport {SymbolFeature} from '../../data/bucket/symbol_bucket';\nimport {CanonicalTileID} from '../../source/tile_id';\nimport ONE_EM from '../../symbol/one_em';\nimport {SymbolStyleLayer} from './symbol_style_layer';\n\nexport enum TextAnchorEnum {\n 'center' = 1,\n 'left' = 2,\n 'right' = 3,\n 'top' = 4,\n 'bottom' = 5,\n 'top-left' = 6,\n 'top-right' = 7,\n 'bottom-left' = 8,\n 'bottom-right' = 9\n}\n\nexport type TextAnchor = keyof typeof TextAnchorEnum;\n\n// The radial offset is to the edge of the text box\n// In the horizontal direction, the edge of the text box is where glyphs start\n// But in the vertical direction, the glyphs appear to \"start\" at the baseline\n// We don't actually load baseline data, but we assume an offset of ONE_EM - 17\n// (see \"yOffset\" in shaping.js)\nconst baselineOffset = 7;\nexport const INVALID_TEXT_OFFSET = Number.POSITIVE_INFINITY;\n\nexport function evaluateVariableOffset(anchor: TextAnchor, offset: [number, number]): [number, number] {\n\n function fromRadialOffset(anchor: TextAnchor, radialOffset: number): [number, number] {\n let x = 0, y = 0;\n if (radialOffset < 0) radialOffset = 0; // Ignore negative offset.\n // solve for r where r^2 + r^2 = radialOffset^2\n const hypotenuse = radialOffset / Math.SQRT2;\n switch (anchor) {\n case 'top-right':\n case 'top-left':\n y = hypotenuse - baselineOffset;\n break;\n case 'bottom-right':\n case 'bottom-left':\n y = -hypotenuse + baselineOffset;\n break;\n case 'bottom':\n y = -radialOffset + baselineOffset;\n break;\n case 'top':\n y = radialOffset - baselineOffset;\n break;\n }\n\n switch (anchor) {\n case 'top-right':\n case 'bottom-right':\n x = -hypotenuse;\n break;\n case 'top-left':\n case 'bottom-left':\n x = hypotenuse;\n break;\n case 'left':\n x = radialOffset;\n break;\n case 'right':\n x = -radialOffset;\n break;\n }\n\n return [x, y];\n }\n\n function fromTextOffset(anchor: TextAnchor, offsetX: number, offsetY: number): [number, number] {\n let x = 0, y = 0;\n // Use absolute offset values.\n offsetX = Math.abs(offsetX);\n offsetY = Math.abs(offsetY);\n\n switch (anchor) {\n case 'top-right':\n case 'top-left':\n case 'top':\n y = offsetY - baselineOffset;\n break;\n case 'bottom-right':\n case 'bottom-left':\n case 'bottom':\n y = -offsetY + baselineOffset;\n break;\n }\n\n switch (anchor) {\n case 'top-right':\n case 'bottom-right':\n case 'right':\n x = -offsetX;\n break;\n case 'top-left':\n case 'bottom-left':\n case 'left':\n x = offsetX;\n break;\n }\n\n return [x, y];\n }\n\n return (offset[1] !== INVALID_TEXT_OFFSET) ? fromTextOffset(anchor, offset[0], offset[1]) : fromRadialOffset(anchor, offset[0]);\n}\n\n// Helper to support both text-variable-anchor and text-variable-anchor-offset. Offset values converted from EMs to PXs\nexport function getTextVariableAnchorOffset(layer: SymbolStyleLayer, feature: SymbolFeature, canonical: CanonicalTileID): VariableAnchorOffsetCollection | null {\n const layout = layer.layout;\n // If style specifies text-variable-anchor-offset, just return it\n const variableAnchorOffset = layout.get('text-variable-anchor-offset')?.evaluate(feature, {}, canonical);\n\n if (variableAnchorOffset) {\n const sourceValues = variableAnchorOffset.values;\n const destValues: VariableAnchorOffsetCollectionSpecification = [];\n\n // Convert offsets from EM to PX, and apply baseline shift\n for (let i = 0; i < sourceValues.length; i += 2) {\n const anchor = destValues[i] = sourceValues[i] as TextAnchor;\n const offset = (sourceValues[i + 1] as [number, number]).map(t => t * ONE_EM) as [number, number];\n\n if (anchor.startsWith('top')) {\n offset[1] -= baselineOffset;\n } else if (anchor.startsWith('bottom')) {\n offset[1] += baselineOffset;\n }\n\n destValues[i + 1] = offset;\n }\n\n return new VariableAnchorOffsetCollection(destValues);\n }\n\n // If style specifies text-variable-anchor, convert to the new format\n const variableAnchor = layout.get('text-variable-anchor');\n\n if (variableAnchor) {\n let textOffset: [number, number];\n const unevaluatedLayout = layer._unevaluatedLayout;\n\n // The style spec says don't use `text-offset` and `text-radial-offset` together\n // but doesn't actually specify what happens if you use both. We go with the radial offset.\n if (unevaluatedLayout.getValue('text-radial-offset') !== undefined) {\n textOffset = [layout.get('text-radial-offset').evaluate(feature, {}, canonical) * ONE_EM, INVALID_TEXT_OFFSET];\n } else {\n textOffset = layout.get('text-offset').evaluate(feature, {}, canonical).map(t => t * ONE_EM) as [number, number];\n }\n\n const anchorOffsets: VariableAnchorOffsetCollectionSpecification = [];\n\n for (const anchor of variableAnchor) {\n anchorOffsets.push(anchor, evaluateVariableOffset(anchor, textOffset));\n }\n\n return new VariableAnchorOffsetCollection(anchorOffsets);\n }\n\n return null;\n}\n","import {Anchor} from './anchor';\n\nimport {getAnchors, getCenterAnchor} from './get_anchors';\nimport {clipLine} from './clip_line';\nimport {shapeText, shapeIcon, WritingMode, fitIconToText} from './shaping';\nimport {getGlyphQuads, getIconQuads} from './quads';\nimport {CollisionFeature} from './collision_feature';\nimport {warnOnce} from '../util/util';\nimport {\n allowsVerticalWritingMode,\n allowsLetterSpacing\n} from '../util/script_detection';\nimport {findPoleOfInaccessibility} from '../util/find_pole_of_inaccessibility';\nimport {classifyRings} from '../util/classify_rings';\nimport {EXTENT} from '../data/extent';\nimport {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {SIZE_PACK_FACTOR, MAX_PACKED_SIZE, MAX_GLYPH_ICON_SIZE} from './symbol_size';\nimport ONE_EM from './one_em';\nimport type {CanonicalTileID} from '../source/tile_id';\nimport type {Shaping, PositionedIcon, TextJustify} from './shaping';\nimport type {CollisionBoxArray, TextAnchorOffsetArray} from '../data/array_types.g';\nimport type {SymbolFeature} from '../data/bucket/symbol_bucket';\nimport type {StyleImage} from '../style/style_image';\nimport type {StyleGlyph} from '../style/style_glyph';\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\nimport type {ImagePosition} from '../render/image_atlas';\nimport type {GlyphPosition} from '../render/glyph_atlas';\nimport type {PossiblyEvaluatedPropertyValue} from '../style/properties';\n\nimport Point from '@mapbox/point-geometry';\nimport murmur3 from 'murmurhash-js';\nimport {getIconPadding, SymbolPadding} from '../style/style_layer/symbol_style_layer';\nimport {VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {getTextVariableAnchorOffset, evaluateVariableOffset, INVALID_TEXT_OFFSET, TextAnchor, TextAnchorEnum} from '../style/style_layer/variable_text_anchor';\n\n// The symbol layout process needs `text-size` evaluated at up to five different zoom levels, and\n// `icon-size` at up to three:\n//\n// 1. `text-size` at the zoom level of the bucket. Used to calculate a per-feature size for source `text-size`\n// expressions, and to calculate the box dimensions for icon-text-fit.\n// 2. `icon-size` at the zoom level of the bucket. Used to calculate a per-feature size for source `icon-size`\n// expressions.\n// 3. `text-size` and `icon-size` at the zoom level of the bucket, plus one. Used to calculate collision boxes.\n// 4. `text-size` at zoom level 18. Used for something line-symbol-placement-related.\n// 5. For composite `*-size` expressions: two zoom levels of curve stops that \"cover\" the zoom level of the\n// bucket. These go into a vertex buffer and are used by the shader to interpolate the size at render time.\n//\n// (1) and (2) are stored in `bucket.layers[0].layout`. The remainder are below.\n//\ntype Sizes = {\n layoutTextSize: PossiblyEvaluatedPropertyValue; // (3),\n layoutIconSize: PossiblyEvaluatedPropertyValue; // (3),\n textMaxSize: PossiblyEvaluatedPropertyValue; // (4),\n compositeTextSizes: [PossiblyEvaluatedPropertyValue, PossiblyEvaluatedPropertyValue]; // (5),\n compositeIconSizes: [PossiblyEvaluatedPropertyValue, PossiblyEvaluatedPropertyValue]; // (5)\n};\n\ntype ShapedTextOrientations = {\n vertical: Shaping | false;\n horizontal: Record;\n};\n\nexport function performSymbolLayout(args: {\n bucket: SymbolBucket;\n glyphMap: {\n [_: string]: {\n [x: number]: StyleGlyph;\n };\n };\n glyphPositions: {\n [_: string]: {\n [x: number]: GlyphPosition;\n };\n };\n imageMap: {[_: string]: StyleImage};\n imagePositions: {[_: string]: ImagePosition};\n showCollisionBoxes: boolean;\n canonical: CanonicalTileID;\n}) {\n args.bucket.createArrays();\n\n const tileSize = 512 * args.bucket.overscaling;\n args.bucket.tilePixelRatio = EXTENT / tileSize;\n args.bucket.compareText = {};\n args.bucket.iconsNeedLinear = false;\n\n const layer = args.bucket.layers[0];\n const layout = layer.layout;\n const unevaluatedLayoutValues = layer._unevaluatedLayout._values;\n\n const sizes: Sizes = {\n // Filled in below, if *SizeData.kind is 'composite'\n // compositeIconSizes: undefined,\n // compositeTextSizes: undefined,\n layoutIconSize: unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(args.bucket.zoom + 1), args.canonical),\n layoutTextSize: unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(args.bucket.zoom + 1), args.canonical),\n textMaxSize: unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(18))\n } as Sizes;\n\n if (args.bucket.textSizeData.kind === 'composite') {\n const {minZoom, maxZoom} = args.bucket.textSizeData;\n sizes.compositeTextSizes = [\n unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(minZoom), args.canonical),\n unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), args.canonical)\n ];\n }\n\n if (args.bucket.iconSizeData.kind === 'composite') {\n const {minZoom, maxZoom} = args.bucket.iconSizeData;\n sizes.compositeIconSizes = [\n unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(minZoom), args.canonical),\n unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), args.canonical)\n ];\n }\n\n const lineHeight = layout.get('text-line-height') * ONE_EM;\n const textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point';\n const keepUpright = layout.get('text-keep-upright');\n const textSize = layout.get('text-size');\n\n for (const feature of args.bucket.features) {\n const fontstack = layout.get('text-font').evaluate(feature, {}, args.canonical).join(',');\n const layoutTextSizeThisZoom = textSize.evaluate(feature, {}, args.canonical);\n const layoutTextSize = sizes.layoutTextSize.evaluate(feature, {}, args.canonical);\n const layoutIconSize = sizes.layoutIconSize.evaluate(feature, {}, args.canonical);\n\n const shapedTextOrientations: ShapedTextOrientations = {\n horizontal: {} as Record,\n vertical: undefined\n };\n const text = feature.text;\n let textOffset: [number, number] = [0, 0];\n if (text) {\n const unformattedText = text.toString();\n const spacing = layout.get('text-letter-spacing').evaluate(feature, {}, args.canonical) * ONE_EM;\n const spacingIfAllowed = allowsLetterSpacing(unformattedText) ? spacing : 0;\n\n const textAnchor = layout.get('text-anchor').evaluate(feature, {}, args.canonical);\n const variableAnchorOffset = getTextVariableAnchorOffset(layer, feature, args.canonical);\n\n if (!variableAnchorOffset) {\n const radialOffset = layout.get('text-radial-offset').evaluate(feature, {}, args.canonical);\n // Layers with variable anchors use the `text-radial-offset` property and the [x, y] offset vector\n // is calculated at placement time instead of layout time\n if (radialOffset) {\n // The style spec says don't use `text-offset` and `text-radial-offset` together\n // but doesn't actually specify what happens if you use both. We go with the radial offset.\n textOffset = evaluateVariableOffset(textAnchor, [radialOffset * ONE_EM, INVALID_TEXT_OFFSET]);\n } else {\n textOffset = (layout.get('text-offset').evaluate(feature, {}, args.canonical).map(t => t * ONE_EM) as [number, number]);\n }\n }\n\n let textJustify = textAlongLine ?\n 'center' :\n layout.get('text-justify').evaluate(feature, {}, args.canonical);\n\n const symbolPlacement = layout.get('symbol-placement');\n const maxWidth = symbolPlacement === 'point' ?\n layout.get('text-max-width').evaluate(feature, {}, args.canonical) * ONE_EM :\n 0;\n\n const addVerticalShapingForPointLabelIfNeeded = () => {\n if (args.bucket.allowVerticalPlacement && allowsVerticalWritingMode(unformattedText)) {\n // Vertical POI label placement is meant to be used for scripts that support vertical\n // writing mode, thus, default left justification is used. If Latin\n // scripts would need to be supported, this should take into account other justifications.\n shapedTextOrientations.vertical = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor,\n 'left', spacingIfAllowed, textOffset, WritingMode.vertical, true, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n }\n };\n\n // If this layer uses text-variable-anchor, generate shapings for all justification possibilities.\n if (!textAlongLine && variableAnchorOffset) {\n const justifications = new Set();\n\n if (textJustify === 'auto') {\n for (let i = 0; i < variableAnchorOffset.values.length; i += 2) {\n justifications.add(getAnchorJustification(variableAnchorOffset.values[i] as TextAnchor));\n }\n } else {\n justifications.add(textJustify);\n }\n\n let singleLine = false;\n for (const justification of justifications) {\n if (shapedTextOrientations.horizontal[justification]) continue;\n if (singleLine) {\n // If the shaping for the first justification was only a single line, we\n // can re-use it for the other justifications\n shapedTextOrientations.horizontal[justification] = shapedTextOrientations.horizontal[0];\n } else {\n // If using text-variable-anchor for the layer, we use a center anchor for all shapings and apply\n // the offsets for the anchor in the placement step.\n const shaping = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, 'center',\n justification, spacingIfAllowed, textOffset, WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n if (shaping) {\n shapedTextOrientations.horizontal[justification] = shaping;\n singleLine = shaping.positionedLines.length === 1;\n }\n }\n }\n\n addVerticalShapingForPointLabelIfNeeded();\n } else {\n if (textJustify === 'auto') {\n textJustify = getAnchorJustification(textAnchor);\n }\n\n // Horizontal point or line label.\n const shaping = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify, spacingIfAllowed,\n textOffset, WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n if (shaping) shapedTextOrientations.horizontal[textJustify] = shaping;\n\n // Vertical point label (if allowVerticalPlacement is enabled).\n addVerticalShapingForPointLabelIfNeeded();\n\n // Verticalized line label.\n if (allowsVerticalWritingMode(unformattedText) && textAlongLine && keepUpright) {\n shapedTextOrientations.vertical = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify,\n spacingIfAllowed, textOffset, WritingMode.vertical, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n }\n }\n }\n\n let shapedIcon;\n let isSDFIcon = false;\n if (feature.icon && feature.icon.name) {\n const image = args.imageMap[feature.icon.name];\n if (image) {\n shapedIcon = shapeIcon(\n args.imagePositions[feature.icon.name],\n layout.get('icon-offset').evaluate(feature, {}, args.canonical),\n layout.get('icon-anchor').evaluate(feature, {}, args.canonical));\n // null/undefined SDF property treated same as default (false)\n isSDFIcon = !!image.sdf;\n if (args.bucket.sdfIcons === undefined) {\n args.bucket.sdfIcons = isSDFIcon;\n } else if (args.bucket.sdfIcons !== isSDFIcon) {\n warnOnce('Style sheet warning: Cannot mix SDF and non-SDF icons in one buffer');\n }\n if (image.pixelRatio !== args.bucket.pixelRatio) {\n args.bucket.iconsNeedLinear = true;\n } else if (layout.get('icon-rotate').constantOr(1) !== 0) {\n args.bucket.iconsNeedLinear = true;\n }\n }\n }\n\n const shapedText = getDefaultHorizontalShaping(shapedTextOrientations.horizontal) || shapedTextOrientations.vertical;\n args.bucket.iconsInText = shapedText ? shapedText.iconsInText : false;\n if (shapedText || shapedIcon) {\n addFeature(args.bucket, feature, shapedTextOrientations, shapedIcon, args.imageMap, sizes, layoutTextSize, layoutIconSize, textOffset, isSDFIcon, args.canonical);\n }\n }\n\n if (args.showCollisionBoxes) {\n args.bucket.generateCollisionDebugBuffers();\n }\n}\n\n// Choose the justification that matches the direction of the TextAnchor\nexport function getAnchorJustification(anchor: TextAnchor): TextJustify {\n switch (anchor) {\n case 'right':\n case 'top-right':\n case 'bottom-right':\n return 'right';\n case 'left':\n case 'top-left':\n case 'bottom-left':\n return 'left';\n }\n return 'center';\n}\n\n/**\n * Given a feature and its shaped text and icon data, add a 'symbol\n * instance' for each _possible_ placement of the symbol feature.\n * (At render timePlaceSymbols#place() selects which of these instances to\n * show or hide based on collisions with symbols in other layers.)\n */\nfunction addFeature(bucket: SymbolBucket,\n feature: SymbolFeature,\n shapedTextOrientations: ShapedTextOrientations,\n shapedIcon: PositionedIcon,\n imageMap: {[_: string]: StyleImage},\n sizes: Sizes,\n layoutTextSize: number,\n layoutIconSize: number,\n textOffset: [number, number],\n isSDFIcon: boolean, canonical: CanonicalTileID) {\n // To reduce the number of labels that jump around when zooming we need\n // to use a text-size value that is the same for all zoom levels.\n // bucket calculates text-size at a high zoom level so that all tiles can\n // use the same value when calculating anchor positions.\n let textMaxSize = sizes.textMaxSize.evaluate(feature, {});\n if (textMaxSize === undefined) {\n textMaxSize = layoutTextSize;\n }\n const layout = bucket.layers[0].layout;\n const iconOffset = layout.get('icon-offset').evaluate(feature, {}, canonical);\n const defaultHorizontalShaping = getDefaultHorizontalShaping(shapedTextOrientations.horizontal);\n const glyphSize = 24,\n fontScale = layoutTextSize / glyphSize,\n textBoxScale = bucket.tilePixelRatio * fontScale,\n textMaxBoxScale = bucket.tilePixelRatio * textMaxSize / glyphSize,\n iconBoxScale = bucket.tilePixelRatio * layoutIconSize,\n symbolMinDistance = bucket.tilePixelRatio * layout.get('symbol-spacing'),\n textPadding = layout.get('text-padding') * bucket.tilePixelRatio,\n iconPadding = getIconPadding(layout, feature, canonical, bucket.tilePixelRatio),\n textMaxAngle = layout.get('text-max-angle') / 180 * Math.PI,\n textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point',\n iconAlongLine = layout.get('icon-rotation-alignment') === 'map' && layout.get('symbol-placement') !== 'point',\n symbolPlacement = layout.get('symbol-placement'),\n textRepeatDistance = symbolMinDistance / 2;\n\n const iconTextFit = layout.get('icon-text-fit');\n let verticallyShapedIcon;\n // Adjust shaped icon size when icon-text-fit is used.\n if (shapedIcon && iconTextFit !== 'none') {\n if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) {\n verticallyShapedIcon = fitIconToText(shapedIcon, shapedTextOrientations.vertical, iconTextFit,\n layout.get('icon-text-fit-padding'), iconOffset, fontScale);\n }\n if (defaultHorizontalShaping) {\n shapedIcon = fitIconToText(shapedIcon, defaultHorizontalShaping, iconTextFit,\n layout.get('icon-text-fit-padding'), iconOffset, fontScale);\n }\n }\n\n const addSymbolAtAnchor = (line, anchor) => {\n if (anchor.x < 0 || anchor.x >= EXTENT || anchor.y < 0 || anchor.y >= EXTENT) {\n // Symbol layers are drawn across tile boundaries, We filter out symbols\n // outside our tile boundaries (which may be included in vector tile buffers)\n // to prevent double-drawing symbols.\n return;\n }\n\n addSymbol(bucket, anchor, line, shapedTextOrientations, shapedIcon, imageMap, verticallyShapedIcon, bucket.layers[0],\n bucket.collisionBoxArray, feature.index, feature.sourceLayerIndex, bucket.index,\n textBoxScale, [textPadding, textPadding, textPadding, textPadding], textAlongLine, textOffset,\n iconBoxScale, iconPadding, iconAlongLine, iconOffset,\n feature, sizes, isSDFIcon, canonical, layoutTextSize);\n };\n\n if (symbolPlacement === 'line') {\n for (const line of clipLine(feature.geometry, 0, 0, EXTENT, EXTENT)) {\n const anchors = getAnchors(\n line,\n symbolMinDistance,\n textMaxAngle,\n shapedTextOrientations.vertical || defaultHorizontalShaping,\n shapedIcon,\n glyphSize,\n textMaxBoxScale,\n bucket.overscaling,\n EXTENT\n );\n for (const anchor of anchors) {\n const shapedText = defaultHorizontalShaping;\n if (!shapedText || !anchorIsTooClose(bucket, shapedText.text, textRepeatDistance, anchor)) {\n addSymbolAtAnchor(line, anchor);\n }\n }\n }\n } else if (symbolPlacement === 'line-center') {\n // No clipping, multiple lines per feature are allowed\n // \"lines\" with only one point are ignored as in clipLines\n for (const line of feature.geometry) {\n if (line.length > 1) {\n const anchor = getCenterAnchor(\n line,\n textMaxAngle,\n shapedTextOrientations.vertical || defaultHorizontalShaping,\n shapedIcon,\n glyphSize,\n textMaxBoxScale);\n if (anchor) {\n addSymbolAtAnchor(line, anchor);\n }\n }\n }\n } else if (feature.type === 'Polygon') {\n for (const polygon of classifyRings(feature.geometry, 0)) {\n // 16 here represents 2 pixels\n const poi = findPoleOfInaccessibility(polygon, 16);\n addSymbolAtAnchor(polygon[0], new Anchor(poi.x, poi.y, 0));\n }\n } else if (feature.type === 'LineString') {\n // https://github.com/mapbox/mapbox-gl-js/issues/3808\n for (const line of feature.geometry) {\n addSymbolAtAnchor(line, new Anchor(line[0].x, line[0].y, 0));\n }\n } else if (feature.type === 'Point') {\n for (const points of feature.geometry) {\n for (const point of points) {\n addSymbolAtAnchor([point], new Anchor(point.x, point.y, 0));\n }\n }\n }\n}\n\nfunction addTextVariableAnchorOffsets(textAnchorOffsets: TextAnchorOffsetArray, variableAnchorOffset: VariableAnchorOffsetCollection): [number, number] {\n const startIndex = textAnchorOffsets.length;\n const values = variableAnchorOffset?.values;\n\n if (values?.length > 0) {\n for (let i = 0; i < values.length; i += 2) {\n const anchor = TextAnchorEnum[values[i] as TextAnchor];\n const offset = values[i + 1] as [number, number];\n\n textAnchorOffsets.emplaceBack(anchor, offset[0], offset[1]);\n }\n }\n\n return [startIndex, textAnchorOffsets.length];\n}\n\nfunction addTextVertices(bucket: SymbolBucket,\n anchor: Point,\n shapedText: Shaping,\n imageMap: {[_: string]: StyleImage},\n layer: SymbolStyleLayer,\n textAlongLine: boolean,\n feature: SymbolFeature,\n textOffset: [number, number],\n lineArray: {\n lineStartIndex: number;\n lineLength: number;\n },\n writingMode: WritingMode,\n placementTypes: Array<'vertical' | 'center' | 'left' | 'right'>,\n placedTextSymbolIndices: {[_: string]: number},\n placedIconIndex: number,\n sizes: Sizes,\n canonical: CanonicalTileID) {\n const glyphQuads = getGlyphQuads(anchor, shapedText, textOffset,\n layer, textAlongLine, feature, imageMap, bucket.allowVerticalPlacement);\n\n const sizeData = bucket.textSizeData;\n let textSizeData = null;\n\n if (sizeData.kind === 'source') {\n textSizeData = [\n SIZE_PACK_FACTOR * layer.layout.get('text-size').evaluate(feature, {})\n ];\n if (textSizeData[0] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"text-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"text-size\".`);\n }\n } else if (sizeData.kind === 'composite') {\n textSizeData = [\n SIZE_PACK_FACTOR * sizes.compositeTextSizes[0].evaluate(feature, {}, canonical),\n SIZE_PACK_FACTOR * sizes.compositeTextSizes[1].evaluate(feature, {}, canonical)\n ];\n if (textSizeData[0] > MAX_PACKED_SIZE || textSizeData[1] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"text-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"text-size\".`);\n }\n }\n\n bucket.addSymbols(\n bucket.text,\n glyphQuads,\n textSizeData,\n textOffset,\n textAlongLine,\n feature,\n writingMode,\n anchor,\n lineArray.lineStartIndex,\n lineArray.lineLength,\n placedIconIndex,\n canonical);\n\n // The placedSymbolArray is used at render time in drawTileSymbols\n // These indices allow access to the array at collision detection time\n for (const placementType of placementTypes) {\n placedTextSymbolIndices[placementType] = bucket.text.placedSymbolArray.length - 1;\n }\n\n return glyphQuads.length * 4;\n}\n\nfunction getDefaultHorizontalShaping(\n horizontalShaping: Record\n): Shaping | null {\n // We don't care which shaping we get because this is used for collision purposes\n // and all the justifications have the same collision box\n for (const justification in horizontalShaping) {\n return horizontalShaping[justification];\n }\n return null;\n}\n\n/**\n * Add a single label & icon placement.\n */\nfunction addSymbol(bucket: SymbolBucket,\n anchor: Anchor,\n line: Array,\n shapedTextOrientations: ShapedTextOrientations,\n shapedIcon: PositionedIcon | void,\n imageMap: {[_: string]: StyleImage},\n verticallyShapedIcon: PositionedIcon | void,\n layer: SymbolStyleLayer,\n collisionBoxArray: CollisionBoxArray,\n featureIndex: number,\n sourceLayerIndex: number,\n bucketIndex: number,\n textBoxScale: number,\n textPadding: SymbolPadding,\n textAlongLine: boolean,\n textOffset: [number, number],\n iconBoxScale: number,\n iconPadding: SymbolPadding,\n iconAlongLine: boolean,\n iconOffset: [number, number],\n feature: SymbolFeature,\n sizes: Sizes,\n isSDFIcon: boolean,\n canonical: CanonicalTileID,\n layoutTextSize: number) {\n const lineArray = bucket.addToLineVertexArray(anchor, line);\n\n let textCollisionFeature, iconCollisionFeature, verticalTextCollisionFeature, verticalIconCollisionFeature;\n\n let numIconVertices = 0;\n let numVerticalIconVertices = 0;\n let numHorizontalGlyphVertices = 0;\n let numVerticalGlyphVertices = 0;\n let placedIconSymbolIndex = -1;\n let verticalPlacedIconSymbolIndex = -1;\n const placedTextSymbolIndices: {[k: string]: number} = {};\n let key = murmur3('');\n\n if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) {\n const textRotation = layer.layout.get('text-rotate').evaluate(feature, {}, canonical);\n const verticalTextRotation = textRotation + 90.0;\n const verticalShaping = shapedTextOrientations.vertical;\n verticalTextCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticalShaping, textBoxScale, textPadding, textAlongLine, verticalTextRotation);\n\n if (verticallyShapedIcon) {\n verticalIconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticallyShapedIcon, iconBoxScale, iconPadding, textAlongLine, verticalTextRotation);\n }\n }\n\n //Place icon first, so text can have a reference to its index in the placed symbol array.\n //Text symbols can lazily shift at render-time because of variable anchor placement.\n //If the style specifies an `icon-text-fit` then the icon would have to shift along with it.\n // For more info check `updateVariableAnchors` in `draw_symbol.js` .\n if (shapedIcon) {\n const iconRotate = layer.layout.get('icon-rotate').evaluate(feature, {});\n const hasIconTextFit = layer.layout.get('icon-text-fit') !== 'none';\n const iconQuads = getIconQuads(shapedIcon, iconRotate, isSDFIcon, hasIconTextFit);\n const verticalIconQuads = verticallyShapedIcon ? getIconQuads(verticallyShapedIcon, iconRotate, isSDFIcon, hasIconTextFit) : undefined;\n iconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shapedIcon, iconBoxScale, iconPadding, /*align boxes to line*/false, iconRotate);\n\n numIconVertices = iconQuads.length * 4;\n\n const sizeData = bucket.iconSizeData;\n let iconSizeData = null;\n\n if (sizeData.kind === 'source') {\n iconSizeData = [\n SIZE_PACK_FACTOR * layer.layout.get('icon-size').evaluate(feature, {})\n ];\n if (iconSizeData[0] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"icon-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"icon-size\".`);\n }\n } else if (sizeData.kind === 'composite') {\n iconSizeData = [\n SIZE_PACK_FACTOR * sizes.compositeIconSizes[0].evaluate(feature, {}, canonical),\n SIZE_PACK_FACTOR * sizes.compositeIconSizes[1].evaluate(feature, {}, canonical)\n ];\n if (iconSizeData[0] > MAX_PACKED_SIZE || iconSizeData[1] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"icon-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"icon-size\".`);\n }\n }\n\n bucket.addSymbols(\n bucket.icon,\n iconQuads,\n iconSizeData,\n iconOffset,\n iconAlongLine,\n feature,\n WritingMode.none,\n anchor,\n lineArray.lineStartIndex,\n lineArray.lineLength,\n // The icon itself does not have an associated symbol since the text isnt placed yet\n -1, canonical);\n\n placedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1;\n\n if (verticalIconQuads) {\n numVerticalIconVertices = verticalIconQuads.length * 4;\n\n bucket.addSymbols(\n bucket.icon,\n verticalIconQuads,\n iconSizeData,\n iconOffset,\n iconAlongLine,\n feature,\n WritingMode.vertical,\n anchor,\n lineArray.lineStartIndex,\n lineArray.lineLength,\n // The icon itself does not have an associated symbol since the text isnt placed yet\n -1, canonical);\n\n verticalPlacedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1;\n }\n }\n\n const justifications = Object.keys(shapedTextOrientations.horizontal) as TextJustify[];\n for (const justification of justifications) {\n const shaping = shapedTextOrientations.horizontal[justification];\n\n if (!textCollisionFeature) {\n key = murmur3(shaping.text);\n const textRotate = layer.layout.get('text-rotate').evaluate(feature, {}, canonical);\n // As a collision approximation, we can use either the vertical or any of the horizontal versions of the feature\n // We're counting on all versions having similar dimensions\n textCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shaping, textBoxScale, textPadding, textAlongLine, textRotate);\n }\n\n const singleLine = shaping.positionedLines.length === 1;\n numHorizontalGlyphVertices += addTextVertices(\n bucket, anchor, shaping, imageMap, layer, textAlongLine, feature, textOffset, lineArray,\n shapedTextOrientations.vertical ? WritingMode.horizontal : WritingMode.horizontalOnly,\n singleLine ? justifications : [justification],\n placedTextSymbolIndices, placedIconSymbolIndex, sizes, canonical);\n\n if (singleLine) {\n break;\n }\n }\n\n if (shapedTextOrientations.vertical) {\n numVerticalGlyphVertices += addTextVertices(\n bucket, anchor, shapedTextOrientations.vertical, imageMap, layer, textAlongLine, feature,\n textOffset, lineArray, WritingMode.vertical, ['vertical'], placedTextSymbolIndices, verticalPlacedIconSymbolIndex, sizes, canonical);\n }\n\n const textBoxStartIndex = textCollisionFeature ? textCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const textBoxEndIndex = textCollisionFeature ? textCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n const verticalTextBoxStartIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const verticalTextBoxEndIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n const iconBoxStartIndex = iconCollisionFeature ? iconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const iconBoxEndIndex = iconCollisionFeature ? iconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n const verticalIconBoxStartIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const verticalIconBoxEndIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n // Check if runtime collision circles should be used for any of the collision features.\n // It is enough to choose the tallest feature shape as circles are always placed on a line.\n // All measurements are in glyph metrics and later converted into pixels using proper font size \"layoutTextSize\"\n let collisionCircleDiameter = -1;\n\n const getCollisionCircleHeight = (feature: CollisionFeature, prevHeight: number): number => {\n if (feature && feature.circleDiameter)\n return Math.max(feature.circleDiameter, prevHeight);\n return prevHeight;\n };\n\n collisionCircleDiameter = getCollisionCircleHeight(textCollisionFeature, collisionCircleDiameter);\n collisionCircleDiameter = getCollisionCircleHeight(verticalTextCollisionFeature, collisionCircleDiameter);\n collisionCircleDiameter = getCollisionCircleHeight(iconCollisionFeature, collisionCircleDiameter);\n collisionCircleDiameter = getCollisionCircleHeight(verticalIconCollisionFeature, collisionCircleDiameter);\n const useRuntimeCollisionCircles = (collisionCircleDiameter > -1) ? 1 : 0;\n\n // Convert circle collision height into pixels\n if (useRuntimeCollisionCircles)\n collisionCircleDiameter *= layoutTextSize / ONE_EM;\n\n if (bucket.glyphOffsetArray.length >= SymbolBucket.MAX_GLYPHS) warnOnce(\n 'Too many glyphs being rendered in a tile. See https://github.com/mapbox/mapbox-gl-js/issues/2907'\n );\n\n if (feature.sortKey !== undefined) {\n bucket.addToSortKeyRanges(bucket.symbolInstances.length, feature.sortKey as number);\n }\n\n const variableAnchorOffset = getTextVariableAnchorOffset(layer, feature, canonical);\n const [textAnchorOffsetStartIndex, textAnchorOffsetEndIndex] = addTextVariableAnchorOffsets(bucket.textAnchorOffsets, variableAnchorOffset);\n\n bucket.symbolInstances.emplaceBack(\n anchor.x,\n anchor.y,\n placedTextSymbolIndices.right >= 0 ? placedTextSymbolIndices.right : -1,\n placedTextSymbolIndices.center >= 0 ? placedTextSymbolIndices.center : -1,\n placedTextSymbolIndices.left >= 0 ? placedTextSymbolIndices.left : -1,\n placedTextSymbolIndices.vertical || -1,\n placedIconSymbolIndex,\n verticalPlacedIconSymbolIndex,\n key,\n textBoxStartIndex,\n textBoxEndIndex,\n verticalTextBoxStartIndex,\n verticalTextBoxEndIndex,\n iconBoxStartIndex,\n iconBoxEndIndex,\n verticalIconBoxStartIndex,\n verticalIconBoxEndIndex,\n featureIndex,\n numHorizontalGlyphVertices,\n numVerticalGlyphVertices,\n numIconVertices,\n numVerticalIconVertices,\n useRuntimeCollisionCircles,\n 0,\n textBoxScale,\n collisionCircleDiameter,\n textAnchorOffsetStartIndex,\n textAnchorOffsetEndIndex);\n}\n\nfunction anchorIsTooClose(bucket: SymbolBucket, text: string, repeatDistance: number, anchor: Point) {\n const compareText = bucket.compareText;\n if (!(text in compareText)) {\n compareText[text] = [];\n } else {\n const otherAnchors = compareText[text];\n for (let k = otherAnchors.length - 1; k >= 0; k--) {\n if (anchor.dist(otherAnchors[k]) < repeatDistance) {\n // If it's within repeatDistance of one anchor, stop looking\n return true;\n }\n }\n }\n // If anchor is not within repeatDistance of any other anchor, add to array\n compareText[text].push(anchor);\n return false;\n}\n","\nconst ARRAY_TYPES = [\n Int8Array, Uint8Array, Uint8ClampedArray, Int16Array, Uint16Array,\n Int32Array, Uint32Array, Float32Array, Float64Array\n];\n\n/** @typedef {Int8ArrayConstructor | Uint8ArrayConstructor | Uint8ClampedArrayConstructor | Int16ArrayConstructor | Uint16ArrayConstructor | Int32ArrayConstructor | Uint32ArrayConstructor | Float32ArrayConstructor | Float64ArrayConstructor} TypedArrayConstructor */\n\nconst VERSION = 1; // serialized format version\nconst HEADER_SIZE = 8;\n\nexport default class KDBush {\n\n /**\n * Creates an index from raw `ArrayBuffer` data.\n * @param {ArrayBuffer} data\n */\n static from(data) {\n if (!(data instanceof ArrayBuffer)) {\n throw new Error('Data must be an instance of ArrayBuffer.');\n }\n const [magic, versionAndType] = new Uint8Array(data, 0, 2);\n if (magic !== 0xdb) {\n throw new Error('Data does not appear to be in a KDBush format.');\n }\n const version = versionAndType >> 4;\n if (version !== VERSION) {\n throw new Error(`Got v${version} data when expected v${VERSION}.`);\n }\n const ArrayType = ARRAY_TYPES[versionAndType & 0x0f];\n if (!ArrayType) {\n throw new Error('Unrecognized array type.');\n }\n const [nodeSize] = new Uint16Array(data, 2, 1);\n const [numItems] = new Uint32Array(data, 4, 1);\n\n return new KDBush(numItems, nodeSize, ArrayType, data);\n }\n\n /**\n * Creates an index that will hold a given number of items.\n * @param {number} numItems\n * @param {number} [nodeSize=64] Size of the KD-tree node (64 by default).\n * @param {TypedArrayConstructor} [ArrayType=Float64Array] The array type used for coordinates storage (`Float64Array` by default).\n * @param {ArrayBuffer} [data] (For internal use only)\n */\n constructor(numItems, nodeSize = 64, ArrayType = Float64Array, data) {\n if (isNaN(numItems) || numItems < 0) throw new Error(`Unpexpected numItems value: ${numItems}.`);\n\n this.numItems = +numItems;\n this.nodeSize = Math.min(Math.max(+nodeSize, 2), 65535);\n this.ArrayType = ArrayType;\n this.IndexArrayType = numItems < 65536 ? Uint16Array : Uint32Array;\n\n const arrayTypeIndex = ARRAY_TYPES.indexOf(this.ArrayType);\n const coordsByteSize = numItems * 2 * this.ArrayType.BYTES_PER_ELEMENT;\n const idsByteSize = numItems * this.IndexArrayType.BYTES_PER_ELEMENT;\n const padCoords = (8 - idsByteSize % 8) % 8;\n\n if (arrayTypeIndex < 0) {\n throw new Error(`Unexpected typed array class: ${ArrayType}.`);\n }\n\n if (data && (data instanceof ArrayBuffer)) { // reconstruct an index from a buffer\n this.data = data;\n this.ids = new this.IndexArrayType(this.data, HEADER_SIZE, numItems);\n this.coords = new this.ArrayType(this.data, HEADER_SIZE + idsByteSize + padCoords, numItems * 2);\n this._pos = numItems * 2;\n this._finished = true;\n } else { // initialize a new index\n this.data = new ArrayBuffer(HEADER_SIZE + coordsByteSize + idsByteSize + padCoords);\n this.ids = new this.IndexArrayType(this.data, HEADER_SIZE, numItems);\n this.coords = new this.ArrayType(this.data, HEADER_SIZE + idsByteSize + padCoords, numItems * 2);\n this._pos = 0;\n this._finished = false;\n\n // set header\n new Uint8Array(this.data, 0, 2).set([0xdb, (VERSION << 4) + arrayTypeIndex]);\n new Uint16Array(this.data, 2, 1)[0] = nodeSize;\n new Uint32Array(this.data, 4, 1)[0] = numItems;\n }\n }\n\n /**\n * Add a point to the index.\n * @param {number} x\n * @param {number} y\n * @returns {number} An incremental index associated with the added item (starting from `0`).\n */\n add(x, y) {\n const index = this._pos >> 1;\n this.ids[index] = index;\n this.coords[this._pos++] = x;\n this.coords[this._pos++] = y;\n return index;\n }\n\n /**\n * Perform indexing of the added points.\n */\n finish() {\n const numAdded = this._pos >> 1;\n if (numAdded !== this.numItems) {\n throw new Error(`Added ${numAdded} items when expected ${this.numItems}.`);\n }\n // kd-sort both arrays for efficient search\n sort(this.ids, this.coords, this.nodeSize, 0, this.numItems - 1, 0);\n\n this._finished = true;\n return this;\n }\n\n /**\n * Search the index for items within a given bounding box.\n * @param {number} minX\n * @param {number} minY\n * @param {number} maxX\n * @param {number} maxY\n * @returns {number[]} An array of indices correponding to the found items.\n */\n range(minX, minY, maxX, maxY) {\n if (!this._finished) throw new Error('Data not yet indexed - call index.finish().');\n\n const {ids, coords, nodeSize} = this;\n const stack = [0, ids.length - 1, 0];\n const result = [];\n\n // recursively search for items in range in the kd-sorted arrays\n while (stack.length) {\n const axis = stack.pop() || 0;\n const right = stack.pop() || 0;\n const left = stack.pop() || 0;\n\n // if we reached \"tree node\", search linearly\n if (right - left <= nodeSize) {\n for (let i = left; i <= right; i++) {\n const x = coords[2 * i];\n const y = coords[2 * i + 1];\n if (x >= minX && x <= maxX && y >= minY && y <= maxY) result.push(ids[i]);\n }\n continue;\n }\n\n // otherwise find the middle index\n const m = (left + right) >> 1;\n\n // include the middle item if it's in range\n const x = coords[2 * m];\n const y = coords[2 * m + 1];\n if (x >= minX && x <= maxX && y >= minY && y <= maxY) result.push(ids[m]);\n\n // queue search in halves that intersect the query\n if (axis === 0 ? minX <= x : minY <= y) {\n stack.push(left);\n stack.push(m - 1);\n stack.push(1 - axis);\n }\n if (axis === 0 ? maxX >= x : maxY >= y) {\n stack.push(m + 1);\n stack.push(right);\n stack.push(1 - axis);\n }\n }\n\n return result;\n }\n\n /**\n * Search the index for items within a given radius.\n * @param {number} qx\n * @param {number} qy\n * @param {number} r Query radius.\n * @returns {number[]} An array of indices correponding to the found items.\n */\n within(qx, qy, r) {\n if (!this._finished) throw new Error('Data not yet indexed - call index.finish().');\n\n const {ids, coords, nodeSize} = this;\n const stack = [0, ids.length - 1, 0];\n const result = [];\n const r2 = r * r;\n\n // recursively search for items within radius in the kd-sorted arrays\n while (stack.length) {\n const axis = stack.pop() || 0;\n const right = stack.pop() || 0;\n const left = stack.pop() || 0;\n\n // if we reached \"tree node\", search linearly\n if (right - left <= nodeSize) {\n for (let i = left; i <= right; i++) {\n if (sqDist(coords[2 * i], coords[2 * i + 1], qx, qy) <= r2) result.push(ids[i]);\n }\n continue;\n }\n\n // otherwise find the middle index\n const m = (left + right) >> 1;\n\n // include the middle item if it's in range\n const x = coords[2 * m];\n const y = coords[2 * m + 1];\n if (sqDist(x, y, qx, qy) <= r2) result.push(ids[m]);\n\n // queue search in halves that intersect the query\n if (axis === 0 ? qx - r <= x : qy - r <= y) {\n stack.push(left);\n stack.push(m - 1);\n stack.push(1 - axis);\n }\n if (axis === 0 ? qx + r >= x : qy + r >= y) {\n stack.push(m + 1);\n stack.push(right);\n stack.push(1 - axis);\n }\n }\n\n return result;\n }\n}\n\n/**\n * @param {Uint16Array | Uint32Array} ids\n * @param {InstanceType} coords\n * @param {number} nodeSize\n * @param {number} left\n * @param {number} right\n * @param {number} axis\n */\nfunction sort(ids, coords, nodeSize, left, right, axis) {\n if (right - left <= nodeSize) return;\n\n const m = (left + right) >> 1; // middle index\n\n // sort ids and coords around the middle index so that the halves lie\n // either left/right or top/bottom correspondingly (taking turns)\n select(ids, coords, m, left, right, axis);\n\n // recursively kd-sort first half and second half on the opposite axis\n sort(ids, coords, nodeSize, left, m - 1, 1 - axis);\n sort(ids, coords, nodeSize, m + 1, right, 1 - axis);\n}\n\n/**\n * Custom Floyd-Rivest selection algorithm: sort ids and coords so that\n * [left..k-1] items are smaller than k-th item (on either x or y axis)\n * @param {Uint16Array | Uint32Array} ids\n * @param {InstanceType} coords\n * @param {number} k\n * @param {number} left\n * @param {number} right\n * @param {number} axis\n */\nfunction select(ids, coords, k, left, right, axis) {\n\n while (right > left) {\n if (right - left > 600) {\n const n = right - left + 1;\n const m = k - left + 1;\n const z = Math.log(n);\n const s = 0.5 * Math.exp(2 * z / 3);\n const sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);\n const newLeft = Math.max(left, Math.floor(k - m * s / n + sd));\n const newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));\n select(ids, coords, k, newLeft, newRight, axis);\n }\n\n const t = coords[2 * k + axis];\n let i = left;\n let j = right;\n\n swapItem(ids, coords, left, k);\n if (coords[2 * right + axis] > t) swapItem(ids, coords, left, right);\n\n while (i < j) {\n swapItem(ids, coords, i, j);\n i++;\n j--;\n while (coords[2 * i + axis] < t) i++;\n while (coords[2 * j + axis] > t) j--;\n }\n\n if (coords[2 * left + axis] === t) swapItem(ids, coords, left, j);\n else {\n j++;\n swapItem(ids, coords, j, right);\n }\n\n if (j <= k) left = j + 1;\n if (k <= j) right = j - 1;\n }\n}\n\n/**\n * @param {Uint16Array | Uint32Array} ids\n * @param {InstanceType} coords\n * @param {number} i\n * @param {number} j\n */\nfunction swapItem(ids, coords, i, j) {\n swap(ids, i, j);\n swap(coords, 2 * i, 2 * j);\n swap(coords, 2 * i + 1, 2 * j + 1);\n}\n\n/**\n * @param {InstanceType} arr\n * @param {number} i\n * @param {number} j\n */\nfunction swap(arr, i, j) {\n const tmp = arr[i];\n arr[i] = arr[j];\n arr[j] = tmp;\n}\n\n/**\n * @param {number} ax\n * @param {number} ay\n * @param {number} bx\n * @param {number} by\n */\nfunction sqDist(ax, ay, bx, by) {\n const dx = ax - bx;\n const dy = ay - by;\n return dx * dx + dy * dy;\n}\n","import type {RequestParameters} from '../util/ajax';\n\nexport type PerformanceMetrics = {\n loadTime: number;\n fullLoadTime: number;\n fps: number;\n percentDroppedFrames: number;\n totalFrames: number;\n};\n\nexport enum PerformanceMarkers {\n create = 'create',\n load = 'load',\n fullLoad = 'fullLoad'\n}\n\nlet lastFrameTime = null;\nlet frameTimes = [];\n\nconst minFramerateTarget = 60;\nconst frameTimeTarget = 1000 / minFramerateTarget;\n\nconst loadTimeKey = 'loadTime';\nconst fullLoadTimeKey = 'fullLoadTime';\n\nexport const PerformanceUtils = {\n mark(marker: PerformanceMarkers) {\n performance.mark(marker);\n },\n frame(timestamp: number) {\n const currTimestamp = timestamp;\n if (lastFrameTime != null) {\n const frameTime = currTimestamp - lastFrameTime;\n frameTimes.push(frameTime);\n }\n lastFrameTime = currTimestamp;\n },\n clearMetrics() {\n lastFrameTime = null;\n frameTimes = [];\n performance.clearMeasures(loadTimeKey);\n performance.clearMeasures(fullLoadTimeKey);\n\n for (const marker in PerformanceMarkers) {\n performance.clearMarks(PerformanceMarkers[marker]);\n }\n },\n\n getPerformanceMetrics(): PerformanceMetrics {\n performance.measure(loadTimeKey, PerformanceMarkers.create, PerformanceMarkers.load);\n performance.measure(fullLoadTimeKey, PerformanceMarkers.create, PerformanceMarkers.fullLoad);\n const loadTime = performance.getEntriesByName(loadTimeKey)[0].duration;\n const fullLoadTime = performance.getEntriesByName(fullLoadTimeKey)[0].duration;\n const totalFrames = frameTimes.length;\n\n const avgFrameTime = frameTimes.reduce((prev, curr) => prev + curr, 0) / totalFrames / 1000;\n const fps = 1 / avgFrameTime;\n\n // count frames that missed our framerate target\n const droppedFrames = frameTimes\n .filter((frameTime) => frameTime > frameTimeTarget)\n .reduce((acc, curr) => {\n return acc + (curr - frameTimeTarget) / frameTimeTarget;\n }, 0);\n const percentDroppedFrames = (droppedFrames / (totalFrames + droppedFrames)) * 100;\n\n return {\n loadTime,\n fullLoadTime,\n fps,\n percentDroppedFrames,\n totalFrames\n };\n }\n};\n\n/**\n * @internal\n * Safe wrapper for the performance resource timing API in web workers with graceful degradation\n */\nexport class RequestPerformance {\n _marks: {\n start: string;\n end: string;\n measure: string;\n };\n\n constructor (request: RequestParameters) {\n this._marks = {\n start: [request.url, 'start'].join('#'),\n end: [request.url, 'end'].join('#'),\n measure: request.url.toString()\n };\n\n performance.mark(this._marks.start);\n }\n\n finish() {\n performance.mark(this._marks.end);\n let resourceTimingData = performance.getEntriesByName(this._marks.measure);\n\n // fallback if web worker implementation of perf.getEntriesByName returns empty\n if (resourceTimingData.length === 0) {\n performance.measure(this._marks.measure, this._marks.start, this._marks.end);\n resourceTimingData = performance.getEntriesByName(this._marks.measure);\n\n // cleanup\n performance.clearMarks(this._marks.start);\n performance.clearMarks(this._marks.end);\n performance.clearMeasures(this._marks.measure);\n }\n\n return resourceTimingData;\n }\n}\n\nexport default performance;\n","import {StyleLayer} from './style_layer';\nimport {createStyleLayer} from './create_style_layer';\n\nimport {featureFilter, groupByLayout} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {TypedStyleLayer} from './style_layer/typed_style_layer';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type LayerConfigs = {[_: string]: LayerSpecification};\nexport type Family = Array;\n\nexport class StyleLayerIndex {\n familiesBySource: {\n [source: string]: {\n [sourceLayer: string]: Array>;\n };\n };\n keyCache: {[source: string]: string};\n\n _layerConfigs: LayerConfigs;\n _layers: {[_: string]: StyleLayer};\n\n constructor(layerConfigs?: Array | null) {\n this.keyCache = {};\n if (layerConfigs) {\n this.replace(layerConfigs);\n }\n }\n\n replace(layerConfigs: Array) {\n this._layerConfigs = {};\n this._layers = {};\n this.update(layerConfigs, []);\n }\n\n update(layerConfigs: Array, removedIds: Array) {\n for (const layerConfig of layerConfigs) {\n this._layerConfigs[layerConfig.id] = layerConfig;\n\n const layer = this._layers[layerConfig.id] = createStyleLayer(layerConfig);\n layer._featureFilter = featureFilter(layer.filter);\n if (this.keyCache[layerConfig.id])\n delete this.keyCache[layerConfig.id];\n }\n for (const id of removedIds) {\n delete this.keyCache[id];\n delete this._layerConfigs[id];\n delete this._layers[id];\n }\n\n this.familiesBySource = {};\n\n const groups = groupByLayout(Object.values(this._layerConfigs), this.keyCache);\n\n for (const layerConfigs of groups) {\n const layers = layerConfigs.map((layerConfig) => this._layers[layerConfig.id]);\n\n const layer = layers[0];\n if (layer.visibility === 'none') {\n continue;\n }\n\n const sourceId = layer.source || '';\n let sourceGroup = this.familiesBySource[sourceId];\n if (!sourceGroup) {\n sourceGroup = this.familiesBySource[sourceId] = {};\n }\n\n const sourceLayerId = layer.sourceLayer || '_geojsonTileLayer';\n let sourceLayerFamilies = sourceGroup[sourceLayerId];\n if (!sourceLayerFamilies) {\n sourceLayerFamilies = sourceGroup[sourceLayerId] = [];\n }\n\n sourceLayerFamilies.push(layers);\n }\n }\n}\n","import {AlphaImage} from '../util/image';\nimport {register} from '../util/web_worker_transfer';\nimport potpack from 'potpack';\n\nimport type {GlyphMetrics, StyleGlyph} from '../style/style_glyph';\n\nconst padding = 1;\n\n/**\n * A rectangle type with postion, width and height.\n */\nexport type Rect = {\n x: number;\n y: number;\n w: number;\n h: number;\n};\n\n/**\n * The glyph's position\n */\nexport type GlyphPosition = {\n rect: Rect;\n metrics: GlyphMetrics;\n};\n\n/**\n * The glyphs' positions\n */\nexport type GlyphPositions = {\n [_: string]: {\n [_: number]: GlyphPosition;\n };\n};\n\nexport class GlyphAtlas {\n image: AlphaImage;\n positions: GlyphPositions;\n\n constructor(stacks: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n }) {\n const positions = {};\n const bins = [];\n\n for (const stack in stacks) {\n const glyphs = stacks[stack];\n const stackPositions = positions[stack] = {};\n\n for (const id in glyphs) {\n const src = glyphs[+id];\n if (!src || src.bitmap.width === 0 || src.bitmap.height === 0) continue;\n\n const bin = {\n x: 0,\n y: 0,\n w: src.bitmap.width + 2 * padding,\n h: src.bitmap.height + 2 * padding\n };\n bins.push(bin);\n stackPositions[id] = {rect: bin, metrics: src.metrics};\n }\n }\n\n const {w, h} = potpack(bins);\n const image = new AlphaImage({width: w || 1, height: h || 1});\n\n for (const stack in stacks) {\n const glyphs = stacks[stack];\n\n for (const id in glyphs) {\n const src = glyphs[+id];\n if (!src || src.bitmap.width === 0 || src.bitmap.height === 0) continue;\n const bin = positions[stack][id].rect;\n AlphaImage.copy(src.bitmap, image, {x: 0, y: 0}, {x: bin.x + padding, y: bin.y + padding}, src.bitmap);\n }\n }\n\n this.image = image;\n this.positions = positions;\n }\n}\n\nregister('GlyphAtlas', GlyphAtlas);\n","import {FeatureIndex} from '../data/feature_index';\nimport {performSymbolLayout} from '../symbol/symbol_layout';\nimport {CollisionBoxArray} from '../data/array_types.g';\nimport {DictionaryCoder} from '../util/dictionary_coder';\nimport {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {LineBucket} from '../data/bucket/line_bucket';\nimport {FillBucket} from '../data/bucket/fill_bucket';\nimport {FillExtrusionBucket} from '../data/bucket/fill_extrusion_bucket';\nimport {warnOnce, mapObject} from '../util/util';\nimport {ImageAtlas} from '../render/image_atlas';\nimport {GlyphAtlas} from '../render/glyph_atlas';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {OverscaledTileID} from './tile_id';\n\nimport type {Bucket} from '../data/bucket';\nimport type {Actor} from '../util/actor';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {StyleLayerIndex} from '../style/style_layer_index';\nimport type {StyleImage} from '../style/style_image';\nimport type {StyleGlyph} from '../style/style_glyph';\nimport type {\n WorkerTileParameters,\n WorkerTileCallback,\n} from '../source/worker_source';\nimport type {PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {VectorTile} from '@mapbox/vector-tile';\nimport {Cancelable} from '../types/cancelable';\n\nexport class WorkerTile {\n tileID: OverscaledTileID;\n uid: string;\n zoom: number;\n pixelRatio: number;\n tileSize: number;\n source: string;\n promoteId: PromoteIdSpecification;\n overscaling: number;\n showCollisionBoxes: boolean;\n collectResourceTiming: boolean;\n returnDependencies: boolean;\n\n status: 'parsing' | 'done';\n data: VectorTile;\n collisionBoxArray: CollisionBoxArray;\n\n abort: (() => void);\n vectorTile: VectorTile;\n inFlightDependencies: Cancelable[];\n dependencySentinel: number;\n\n constructor(params: WorkerTileParameters) {\n this.tileID = new OverscaledTileID(params.tileID.overscaledZ, params.tileID.wrap, params.tileID.canonical.z, params.tileID.canonical.x, params.tileID.canonical.y);\n this.uid = params.uid;\n this.zoom = params.zoom;\n this.pixelRatio = params.pixelRatio;\n this.tileSize = params.tileSize;\n this.source = params.source;\n this.overscaling = this.tileID.overscaleFactor();\n this.showCollisionBoxes = params.showCollisionBoxes;\n this.collectResourceTiming = !!params.collectResourceTiming;\n this.returnDependencies = !!params.returnDependencies;\n this.promoteId = params.promoteId;\n this.inFlightDependencies = [];\n this.dependencySentinel = -1;\n }\n\n parse(data: VectorTile, layerIndex: StyleLayerIndex, availableImages: Array, actor: Actor, callback: WorkerTileCallback) {\n this.status = 'parsing';\n this.data = data;\n\n this.collisionBoxArray = new CollisionBoxArray();\n const sourceLayerCoder = new DictionaryCoder(Object.keys(data.layers).sort());\n\n const featureIndex = new FeatureIndex(this.tileID, this.promoteId);\n featureIndex.bucketLayerIDs = [];\n\n const buckets: {[_: string]: Bucket} = {};\n\n const options = {\n featureIndex,\n iconDependencies: {},\n patternDependencies: {},\n glyphDependencies: {},\n availableImages\n };\n\n const layerFamilies = layerIndex.familiesBySource[this.source];\n for (const sourceLayerId in layerFamilies) {\n const sourceLayer = data.layers[sourceLayerId];\n if (!sourceLayer) {\n continue;\n }\n\n if (sourceLayer.version === 1) {\n warnOnce(`Vector tile source \"${this.source}\" layer \"${sourceLayerId}\" ` +\n 'does not use vector tile spec v2 and therefore may have some rendering errors.');\n }\n\n const sourceLayerIndex = sourceLayerCoder.encode(sourceLayerId);\n const features = [];\n for (let index = 0; index < sourceLayer.length; index++) {\n const feature = sourceLayer.feature(index);\n const id = featureIndex.getId(feature, sourceLayerId);\n features.push({feature, id, index, sourceLayerIndex});\n }\n\n for (const family of layerFamilies[sourceLayerId]) {\n const layer = family[0];\n\n if (layer.source !== this.source) {\n warnOnce(`layer.source = ${layer.source} does not equal this.source = ${this.source}`);\n }\n if (layer.minzoom && this.zoom < Math.floor(layer.minzoom)) continue;\n if (layer.maxzoom && this.zoom >= layer.maxzoom) continue;\n if (layer.visibility === 'none') continue;\n\n recalculateLayers(family, this.zoom, availableImages);\n\n const bucket = buckets[layer.id] = layer.createBucket({\n index: featureIndex.bucketLayerIDs.length,\n layers: family,\n zoom: this.zoom,\n pixelRatio: this.pixelRatio,\n overscaling: this.overscaling,\n collisionBoxArray: this.collisionBoxArray,\n sourceLayerIndex,\n sourceID: this.source\n });\n\n bucket.populate(features, options, this.tileID.canonical);\n featureIndex.bucketLayerIDs.push(family.map((l) => l.id));\n }\n }\n\n let error: Error;\n let glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n };\n let iconMap: {[_: string]: StyleImage};\n let patternMap: {[_: string]: StyleImage};\n\n const stacks = mapObject(options.glyphDependencies, (glyphs) => Object.keys(glyphs).map(Number));\n\n this.inFlightDependencies.forEach((request) => request?.cancel());\n this.inFlightDependencies = [];\n\n // cancelling seems to be not sufficient, we seems to still manage to get a callback hit, so use a sentinel to drop stale results\n const dependencySentinel = ++this.dependencySentinel;\n if (Object.keys(stacks).length) {\n this.inFlightDependencies.push(actor.send('getGlyphs', {uid: this.uid, stacks, source: this.source, tileID: this.tileID, type: 'glyphs'}, (err, result) => {\n if (dependencySentinel !== this.dependencySentinel) {\n return;\n }\n if (!error) {\n error = err;\n glyphMap = result;\n maybePrepare.call(this);\n }\n }));\n } else {\n glyphMap = {};\n }\n\n const icons = Object.keys(options.iconDependencies);\n if (icons.length) {\n this.inFlightDependencies.push(actor.send('getImages', {icons, source: this.source, tileID: this.tileID, type: 'icons'}, (err, result) => {\n if (dependencySentinel !== this.dependencySentinel) {\n return;\n }\n if (!error) {\n error = err;\n iconMap = result;\n maybePrepare.call(this);\n }\n }));\n } else {\n iconMap = {};\n }\n\n const patterns = Object.keys(options.patternDependencies);\n if (patterns.length) {\n this.inFlightDependencies.push(actor.send('getImages', {icons: patterns, source: this.source, tileID: this.tileID, type: 'patterns'}, (err, result) => {\n if (dependencySentinel !== this.dependencySentinel) {\n return;\n }\n if (!error) {\n error = err;\n patternMap = result;\n maybePrepare.call(this);\n }\n }));\n } else {\n patternMap = {};\n }\n\n maybePrepare.call(this);\n\n function maybePrepare() {\n if (error) {\n return callback(error);\n } else if (glyphMap && iconMap && patternMap) {\n const glyphAtlas = new GlyphAtlas(glyphMap);\n const imageAtlas = new ImageAtlas(iconMap, patternMap);\n\n for (const key in buckets) {\n const bucket = buckets[key];\n if (bucket instanceof SymbolBucket) {\n recalculateLayers(bucket.layers, this.zoom, availableImages);\n performSymbolLayout({\n bucket,\n glyphMap,\n glyphPositions: glyphAtlas.positions,\n imageMap: iconMap,\n imagePositions: imageAtlas.iconPositions,\n showCollisionBoxes: this.showCollisionBoxes,\n canonical: this.tileID.canonical\n });\n } else if (bucket.hasPattern &&\n (bucket instanceof LineBucket ||\n bucket instanceof FillBucket ||\n bucket instanceof FillExtrusionBucket)) {\n recalculateLayers(bucket.layers, this.zoom, availableImages);\n bucket.addFeatures(options, this.tileID.canonical, imageAtlas.patternPositions);\n }\n }\n\n this.status = 'done';\n callback(null, {\n buckets: Object.values(buckets).filter(b => !b.isEmpty()),\n featureIndex,\n collisionBoxArray: this.collisionBoxArray,\n glyphAtlasImage: glyphAtlas.image,\n imageAtlas,\n // Only used for benchmarking:\n glyphMap: this.returnDependencies ? glyphMap : null,\n iconMap: this.returnDependencies ? iconMap : null,\n glyphPositions: this.returnDependencies ? glyphAtlas.positions : null\n });\n }\n }\n }\n}\n\nfunction recalculateLayers(layers: ReadonlyArray, zoom: number, availableImages: Array) {\n // Layers are shared and may have been used by a WorkerTile with a different zoom.\n const parameters = new EvaluationParameters(zoom);\n for (const layer of layers) {\n layer.recalculate(parameters, availableImages);\n }\n}\n","import {ExpiryData, getArrayBuffer} from '../util/ajax';\n\nimport vt from '@mapbox/vector-tile';\nimport Protobuf from 'pbf';\nimport {WorkerTile} from './worker_tile';\nimport {extend} from '../util/util';\nimport {RequestPerformance} from '../util/performance';\n\nimport type {\n WorkerSource,\n WorkerTileParameters,\n WorkerTileCallback,\n TileParameters\n} from '../source/worker_source';\n\nimport type {Actor} from '../util/actor';\nimport type {StyleLayerIndex} from '../style/style_layer_index';\nimport type {Callback} from '../types/callback';\nimport type {VectorTile} from '@mapbox/vector-tile';\n\nexport type LoadVectorTileResult = {\n vectorTile: VectorTile;\n rawData: ArrayBuffer;\n resourceTiming?: Array;\n} & ExpiryData;\n\ntype FetchingState = {\n rawTileData: ArrayBuffer;\n cacheControl: ExpiryData;\n resourceTiming: any;\n}\n\n/**\n * The callback when finished loading vector data\n */\nexport type LoadVectorDataCallback = Callback;\n\nexport type AbortVectorData = () => void;\nexport type LoadVectorData = (params: WorkerTileParameters, callback: LoadVectorDataCallback) => AbortVectorData | void;\n\n/**\n * Loads a vector tile\n */\nfunction loadVectorTile(params: WorkerTileParameters, callback: LoadVectorDataCallback) {\n const request = getArrayBuffer(params.request, (err?: Error | null, data?: ArrayBuffer | null, cacheControl?: string | null, expires?: string | null) => {\n if (err) {\n callback(err);\n } else if (data) {\n try {\n const vectorTile = new vt.VectorTile(new Protobuf(data));\n callback(null, {\n vectorTile,\n rawData: data,\n cacheControl,\n expires\n });\n } catch (ex) {\n const bytes = new Uint8Array(data);\n const isGzipped = bytes[0] === 0x1f && bytes[1] === 0x8b;\n let errorMessage = `Unable to parse the tile at ${params.request.url}, `;\n if (isGzipped) {\n errorMessage += 'please make sure the data is not gzipped and that you have configured the relevant header in the server';\n } else {\n errorMessage += `got error: ${ex.messge}`;\n }\n callback(new Error(errorMessage));\n }\n }\n });\n return () => {\n request.cancel();\n callback();\n };\n}\n\n/**\n * The {@link WorkerSource} implementation that supports {@link VectorTileSource}.\n * This class is designed to be easily reused to support custom source types\n * for data formats that can be parsed/converted into an in-memory VectorTile\n * representation. To do so, create it with\n * `new VectorTileWorkerSource(actor, styleLayers, customLoadVectorDataFunction)`.\n */\nexport class VectorTileWorkerSource implements WorkerSource {\n actor: Actor;\n layerIndex: StyleLayerIndex;\n availableImages: Array;\n loadVectorData: LoadVectorData;\n fetching: {[_: string]: FetchingState };\n loading: {[_: string]: WorkerTile};\n loaded: {[_: string]: WorkerTile};\n\n /**\n * @param loadVectorData - Optional method for custom loading of a VectorTile\n * object based on parameters passed from the main-thread Source. See\n * {@link VectorTileWorkerSource#loadTile}. The default implementation simply\n * loads the pbf at `params.url`.\n */\n constructor(actor: Actor, layerIndex: StyleLayerIndex, availableImages: Array, loadVectorData?: LoadVectorData | null) {\n this.actor = actor;\n this.layerIndex = layerIndex;\n this.availableImages = availableImages;\n this.loadVectorData = loadVectorData || loadVectorTile;\n this.fetching = {};\n this.loading = {};\n this.loaded = {};\n }\n\n /**\n * Implements {@link WorkerSource#loadTile}. Delegates to\n * {@link VectorTileWorkerSource#loadVectorData} (which by default expects\n * a `params.url` property) for fetching and producing a VectorTile object.\n */\n loadTile(params: WorkerTileParameters, callback: WorkerTileCallback) {\n const uid = params.uid;\n\n if (!this.loading)\n this.loading = {};\n\n const perf = (params && params.request && params.request.collectResourceTiming) ?\n new RequestPerformance(params.request) : false;\n\n const workerTile = this.loading[uid] = new WorkerTile(params);\n workerTile.abort = this.loadVectorData(params, (err, response) => {\n delete this.loading[uid];\n\n if (err || !response) {\n workerTile.status = 'done';\n this.loaded[uid] = workerTile;\n return callback(err);\n }\n\n const rawTileData = response.rawData;\n const cacheControl = {} as ExpiryData;\n if (response.expires) cacheControl.expires = response.expires;\n if (response.cacheControl) cacheControl.cacheControl = response.cacheControl;\n\n const resourceTiming = {} as {resourceTiming: any};\n if (perf) {\n const resourceTimingData = perf.finish();\n // it's necessary to eval the result of getEntriesByName() here via parse/stringify\n // late evaluation in the main thread causes TypeError: illegal invocation\n if (resourceTimingData)\n resourceTiming.resourceTiming = JSON.parse(JSON.stringify(resourceTimingData));\n }\n\n workerTile.vectorTile = response.vectorTile;\n workerTile.parse(response.vectorTile, this.layerIndex, this.availableImages, this.actor, (err, result) => {\n delete this.fetching[uid];\n if (err || !result) return callback(err);\n\n // Transferring a copy of rawTileData because the worker needs to retain its copy.\n callback(null, extend({rawTileData: rawTileData.slice(0)}, result, cacheControl, resourceTiming));\n });\n\n this.loaded = this.loaded || {};\n this.loaded[uid] = workerTile;\n // keep the original fetching state so that reload tile can pick it up if the original parse is cancelled by reloads' parse\n this.fetching[uid] = {rawTileData, cacheControl, resourceTiming};\n }) as AbortVectorData;\n }\n\n /**\n * Implements {@link WorkerSource#reloadTile}.\n */\n reloadTile(params: WorkerTileParameters, callback: WorkerTileCallback) {\n const loaded = this.loaded;\n const uid = params.uid;\n if (loaded && loaded[uid]) {\n const workerTile = loaded[uid];\n workerTile.showCollisionBoxes = params.showCollisionBoxes;\n if (workerTile.status === 'parsing') {\n workerTile.parse(workerTile.vectorTile, this.layerIndex, this.availableImages, this.actor, (err, result) => {\n if (err || !result) return callback(err, result);\n\n // if we have cancelled the original parse, make sure to pass the rawTileData from the original fetch\n let parseResult;\n if (this.fetching[uid]) {\n const {rawTileData, cacheControl, resourceTiming} = this.fetching[uid];\n delete this.fetching[uid];\n parseResult = extend({rawTileData: rawTileData.slice(0)}, result, cacheControl, resourceTiming);\n } else {\n parseResult = result;\n }\n\n callback(null, parseResult);\n });\n } else if (workerTile.status === 'done') {\n // if there was no vector tile data on the initial load, don't try and re-parse tile\n if (workerTile.vectorTile) {\n workerTile.parse(workerTile.vectorTile, this.layerIndex, this.availableImages, this.actor, callback);\n } else {\n callback();\n }\n }\n }\n }\n\n /**\n * Implements {@link WorkerSource#abortTile}.\n *\n * @param params - The tile parameters\n * @param callback - The callback\n */\n abortTile(params: TileParameters, callback: WorkerTileCallback) {\n const loading = this.loading,\n uid = params.uid;\n if (loading && loading[uid] && loading[uid].abort) {\n loading[uid].abort();\n delete loading[uid];\n }\n callback();\n }\n\n /**\n * Implements {@link WorkerSource#removeTile}.\n *\n * @param params - The tile parameters\n * @param callback - The callback\n */\n removeTile(params: TileParameters, callback: WorkerTileCallback) {\n const loaded = this.loaded,\n uid = params.uid;\n if (loaded && loaded[uid]) {\n delete loaded[uid];\n }\n callback();\n }\n}\n","import {DEMData} from '../data/dem_data';\nimport {RGBAImage} from '../util/image';\nimport type {Actor} from '../util/actor';\nimport type {\n WorkerDEMTileParameters,\n WorkerDEMTileCallback,\n TileParameters\n} from './worker_source';\nimport {getImageData, isImageBitmap} from '../util/util';\n\nexport class RasterDEMTileWorkerSource {\n actor: Actor;\n loaded: {[_: string]: DEMData};\n\n constructor() {\n this.loaded = {};\n }\n\n async loadTile(params: WorkerDEMTileParameters, callback: WorkerDEMTileCallback) {\n const {uid, encoding, rawImageData, redFactor, greenFactor, blueFactor, baseShift} = params;\n const width = rawImageData.width + 2;\n const height = rawImageData.height + 2;\n const imagePixels: RGBAImage = isImageBitmap(rawImageData) ?\n new RGBAImage({width, height}, await getImageData(rawImageData, -1, -1, width, height)) :\n rawImageData;\n const dem = new DEMData(uid, imagePixels, encoding, redFactor, greenFactor, blueFactor, baseShift);\n this.loaded = this.loaded || {};\n this.loaded[uid] = dem;\n callback(null, dem);\n }\n\n removeTile(params: TileParameters) {\n const loaded = this.loaded,\n uid = params.uid;\n if (loaded && loaded[uid]) {\n delete loaded[uid];\n }\n }\n}\n","\nmodule.exports = rewind;\n\nfunction rewind(gj, outer) {\n var type = gj && gj.type, i;\n\n if (type === 'FeatureCollection') {\n for (i = 0; i < gj.features.length; i++) rewind(gj.features[i], outer);\n\n } else if (type === 'GeometryCollection') {\n for (i = 0; i < gj.geometries.length; i++) rewind(gj.geometries[i], outer);\n\n } else if (type === 'Feature') {\n rewind(gj.geometry, outer);\n\n } else if (type === 'Polygon') {\n rewindRings(gj.coordinates, outer);\n\n } else if (type === 'MultiPolygon') {\n for (i = 0; i < gj.coordinates.length; i++) rewindRings(gj.coordinates[i], outer);\n }\n\n return gj;\n}\n\nfunction rewindRings(rings, outer) {\n if (rings.length === 0) return;\n\n rewindRing(rings[0], outer);\n for (var i = 1; i < rings.length; i++) {\n rewindRing(rings[i], !outer);\n }\n}\n\nfunction rewindRing(ring, dir) {\n var area = 0, err = 0;\n for (var i = 0, len = ring.length, j = len - 1; i < len; j = i++) {\n var k = (ring[i][0] - ring[j][0]) * (ring[j][1] + ring[i][1]);\n var m = area + k;\n err += Math.abs(area) >= Math.abs(k) ? area - m + k : k - m + area;\n area = m;\n }\n if (area + err >= 0 !== !!dir) ring.reverse();\n}\n","import Point from '@mapbox/point-geometry';\n\nimport mvt from '@mapbox/vector-tile';\nimport type {VectorTileFeature, VectorTileLayer, VectorTile} from '@mapbox/vector-tile';\nconst toGeoJSON = mvt.VectorTileFeature.prototype.toGeoJSON;\nimport {EXTENT} from '../data/extent';\nimport type {TileFeature, AnyProps} from 'supercluster';\nimport type {Feature as GeoJSONVTFeature} from 'geojson-vt';\n\nexport type Feature = TileFeature | GeoJSONVTFeature;\n\nclass FeatureWrapper implements VectorTileFeature {\n _feature: Feature;\n\n extent: number;\n type: Feature['type'];\n id: number;\n properties: {[_: string]: string | number | boolean};\n\n constructor(feature: Feature) {\n this._feature = feature;\n\n this.extent = EXTENT;\n this.type = feature.type;\n this.properties = feature.tags;\n\n // If the feature has a top-level `id` property, copy it over, but only\n // if it can be coerced to an integer, because this wrapper is used for\n // serializing geojson feature data into vector tile PBF data, and the\n // vector tile spec only supports integer values for feature ids --\n // allowing non-integer values here results in a non-compliant PBF\n // that causes an exception when it is parsed with vector-tile-js\n if ('id' in feature && !isNaN(feature.id as any)) {\n this.id = parseInt(feature.id, 10);\n }\n }\n\n loadGeometry() {\n if (this._feature.type === 1) {\n const geometry = [];\n for (const point of this._feature.geometry) {\n geometry.push([new Point(point[0], point[1])]);\n }\n return geometry;\n } else {\n const geometry = [];\n for (const ring of this._feature.geometry) {\n const newRing = [];\n for (const point of ring) {\n newRing.push(new Point(point[0], point[1]));\n }\n geometry.push(newRing);\n }\n return geometry;\n }\n }\n\n toGeoJSON(x: number, y: number, z: number) {\n return toGeoJSON.call(this, x, y, z);\n }\n}\n\nexport class GeoJSONWrapper implements VectorTile, VectorTileLayer {\n layers: {[_: string]: VectorTileLayer};\n name: string;\n extent: number;\n length: number;\n _features: Array;\n\n constructor(features: Array) {\n this.layers = {'_geojsonTileLayer': this};\n this.name = '_geojsonTileLayer';\n this.extent = EXTENT;\n this.length = features.length;\n this._features = features;\n }\n\n feature(i: number): VectorTileFeature {\n return new FeatureWrapper(this._features[i]);\n }\n}\n","'use strict'\n\nvar Point = require('@mapbox/point-geometry')\nvar VectorTileFeature = require('@mapbox/vector-tile').VectorTileFeature\n\nmodule.exports = GeoJSONWrapper\n\n// conform to vectortile api\nfunction GeoJSONWrapper (features, options) {\n this.options = options || {}\n this.features = features\n this.length = features.length\n}\n\nGeoJSONWrapper.prototype.feature = function (i) {\n return new FeatureWrapper(this.features[i], this.options.extent)\n}\n\nfunction FeatureWrapper (feature, extent) {\n this.id = typeof feature.id === 'number' ? feature.id : undefined\n this.type = feature.type\n this.rawGeometry = feature.type === 1 ? [feature.geometry] : feature.geometry\n this.properties = feature.tags\n this.extent = extent || 4096\n}\n\nFeatureWrapper.prototype.loadGeometry = function () {\n var rings = this.rawGeometry\n this.geometry = []\n\n for (var i = 0; i < rings.length; i++) {\n var ring = rings[i]\n var newRing = []\n for (var j = 0; j < ring.length; j++) {\n newRing.push(new Point(ring[j][0], ring[j][1]))\n }\n this.geometry.push(newRing)\n }\n return this.geometry\n}\n\nFeatureWrapper.prototype.bbox = function () {\n if (!this.geometry) this.loadGeometry()\n\n var rings = this.geometry\n var x1 = Infinity\n var x2 = -Infinity\n var y1 = Infinity\n var y2 = -Infinity\n\n for (var i = 0; i < rings.length; i++) {\n var ring = rings[i]\n\n for (var j = 0; j < ring.length; j++) {\n var coord = ring[j]\n\n x1 = Math.min(x1, coord.x)\n x2 = Math.max(x2, coord.x)\n y1 = Math.min(y1, coord.y)\n y2 = Math.max(y2, coord.y)\n }\n }\n\n return [x1, y1, x2, y2]\n}\n\nFeatureWrapper.prototype.toGeoJSON = VectorTileFeature.prototype.toGeoJSON\n","var Pbf = require('pbf')\nvar GeoJSONWrapper = require('./lib/geojson_wrapper')\n\nmodule.exports = fromVectorTileJs\nmodule.exports.fromVectorTileJs = fromVectorTileJs\nmodule.exports.fromGeojsonVt = fromGeojsonVt\nmodule.exports.GeoJSONWrapper = GeoJSONWrapper\n\n/**\n * Serialize a vector-tile-js-created tile to pbf\n *\n * @param {Object} tile\n * @return {Buffer} uncompressed, pbf-serialized tile data\n */\nfunction fromVectorTileJs (tile) {\n var out = new Pbf()\n writeTile(tile, out)\n return out.finish()\n}\n\n/**\n * Serialized a geojson-vt-created tile to pbf.\n *\n * @param {Object} layers - An object mapping layer names to geojson-vt-created vector tile objects\n * @param {Object} [options] - An object specifying the vector-tile specification version and extent that were used to create `layers`.\n * @param {Number} [options.version=1] - Version of vector-tile spec used\n * @param {Number} [options.extent=4096] - Extent of the vector tile\n * @return {Buffer} uncompressed, pbf-serialized tile data\n */\nfunction fromGeojsonVt (layers, options) {\n options = options || {}\n var l = {}\n for (var k in layers) {\n l[k] = new GeoJSONWrapper(layers[k].features, options)\n l[k].name = k\n l[k].version = options.version\n l[k].extent = options.extent\n }\n return fromVectorTileJs({ layers: l })\n}\n\nfunction writeTile (tile, pbf) {\n for (var key in tile.layers) {\n pbf.writeMessage(3, writeLayer, tile.layers[key])\n }\n}\n\nfunction writeLayer (layer, pbf) {\n pbf.writeVarintField(15, layer.version || 1)\n pbf.writeStringField(1, layer.name || '')\n pbf.writeVarintField(5, layer.extent || 4096)\n\n var i\n var context = {\n keys: [],\n values: [],\n keycache: {},\n valuecache: {}\n }\n\n for (i = 0; i < layer.length; i++) {\n context.feature = layer.feature(i)\n pbf.writeMessage(2, writeFeature, context)\n }\n\n var keys = context.keys\n for (i = 0; i < keys.length; i++) {\n pbf.writeStringField(3, keys[i])\n }\n\n var values = context.values\n for (i = 0; i < values.length; i++) {\n pbf.writeMessage(4, writeValue, values[i])\n }\n}\n\nfunction writeFeature (context, pbf) {\n var feature = context.feature\n\n if (feature.id !== undefined) {\n pbf.writeVarintField(1, feature.id)\n }\n\n pbf.writeMessage(2, writeProperties, context)\n pbf.writeVarintField(3, feature.type)\n pbf.writeMessage(4, writeGeometry, feature)\n}\n\nfunction writeProperties (context, pbf) {\n var feature = context.feature\n var keys = context.keys\n var values = context.values\n var keycache = context.keycache\n var valuecache = context.valuecache\n\n for (var key in feature.properties) {\n var value = feature.properties[key]\n\n var keyIndex = keycache[key]\n if (value === null) continue // don't encode null value properties\n\n if (typeof keyIndex === 'undefined') {\n keys.push(key)\n keyIndex = keys.length - 1\n keycache[key] = keyIndex\n }\n pbf.writeVarint(keyIndex)\n\n var type = typeof value\n if (type !== 'string' && type !== 'boolean' && type !== 'number') {\n value = JSON.stringify(value)\n }\n var valueKey = type + ':' + value\n var valueIndex = valuecache[valueKey]\n if (typeof valueIndex === 'undefined') {\n values.push(value)\n valueIndex = values.length - 1\n valuecache[valueKey] = valueIndex\n }\n pbf.writeVarint(valueIndex)\n }\n}\n\nfunction command (cmd, length) {\n return (length << 3) + (cmd & 0x7)\n}\n\nfunction zigzag (num) {\n return (num << 1) ^ (num >> 31)\n}\n\nfunction writeGeometry (feature, pbf) {\n var geometry = feature.loadGeometry()\n var type = feature.type\n var x = 0\n var y = 0\n var rings = geometry.length\n for (var r = 0; r < rings; r++) {\n var ring = geometry[r]\n var count = 1\n if (type === 1) {\n count = ring.length\n }\n pbf.writeVarint(command(1, count)) // moveto\n // do not write polygon closing path as lineto\n var lineCount = type === 3 ? ring.length - 1 : ring.length\n for (var i = 0; i < lineCount; i++) {\n if (i === 1 && type !== 1) {\n pbf.writeVarint(command(2, lineCount - 1)) // lineto\n }\n var dx = ring[i].x - x\n var dy = ring[i].y - y\n pbf.writeVarint(zigzag(dx))\n pbf.writeVarint(zigzag(dy))\n x += dx\n y += dy\n }\n if (type === 3) {\n pbf.writeVarint(command(7, 1)) // closepath\n }\n }\n}\n\nfunction writeValue (value, pbf) {\n var type = typeof value\n if (type === 'string') {\n pbf.writeStringField(1, value)\n } else if (type === 'boolean') {\n pbf.writeBooleanField(7, value)\n } else if (type === 'number') {\n if (value % 1 !== 0) {\n pbf.writeDoubleField(3, value)\n } else if (value < 0) {\n pbf.writeSVarintField(6, value)\n } else {\n pbf.writeVarintField(5, value)\n }\n }\n}\n","\nimport KDBush from 'kdbush';\n\nconst defaultOptions = {\n minZoom: 0, // min zoom to generate clusters on\n maxZoom: 16, // max zoom level to cluster the points on\n minPoints: 2, // minimum points to form a cluster\n radius: 40, // cluster radius in pixels\n extent: 512, // tile extent (radius is calculated relative to it)\n nodeSize: 64, // size of the KD-tree leaf node, affects performance\n log: false, // whether to log timing info\n\n // whether to generate numeric ids for input features (in vector tiles)\n generateId: false,\n\n // a reduce function for calculating custom cluster properties\n reduce: null, // (accumulated, props) => { accumulated.sum += props.sum; }\n\n // properties to use for individual points when running the reducer\n map: props => props // props => ({sum: props.my_value})\n};\n\nconst fround = Math.fround || (tmp => ((x) => { tmp[0] = +x; return tmp[0]; }))(new Float32Array(1));\n\nconst OFFSET_ZOOM = 2;\nconst OFFSET_ID = 3;\nconst OFFSET_PARENT = 4;\nconst OFFSET_NUM = 5;\nconst OFFSET_PROP = 6;\n\nexport default class Supercluster {\n constructor(options) {\n this.options = Object.assign(Object.create(defaultOptions), options);\n this.trees = new Array(this.options.maxZoom + 1);\n this.stride = this.options.reduce ? 7 : 6;\n this.clusterProps = [];\n }\n\n load(points) {\n const {log, minZoom, maxZoom} = this.options;\n\n if (log) console.time('total time');\n\n const timerId = `prepare ${ points.length } points`;\n if (log) console.time(timerId);\n\n this.points = points;\n\n // generate a cluster object for each point and index input points into a KD-tree\n const data = [];\n\n for (let i = 0; i < points.length; i++) {\n const p = points[i];\n if (!p.geometry) continue;\n\n const [lng, lat] = p.geometry.coordinates;\n const x = fround(lngX(lng));\n const y = fround(latY(lat));\n // store internal point/cluster data in flat numeric arrays for performance\n data.push(\n x, y, // projected point coordinates\n Infinity, // the last zoom the point was processed at\n i, // index of the source feature in the original input array\n -1, // parent cluster id\n 1 // number of points in a cluster\n );\n if (this.options.reduce) data.push(0); // noop\n }\n let tree = this.trees[maxZoom + 1] = this._createTree(data);\n\n if (log) console.timeEnd(timerId);\n\n // cluster points on max zoom, then cluster the results on previous zoom, etc.;\n // results in a cluster hierarchy across zoom levels\n for (let z = maxZoom; z >= minZoom; z--) {\n const now = +Date.now();\n\n // create a new set of clusters for the zoom and index them with a KD-tree\n tree = this.trees[z] = this._createTree(this._cluster(tree, z));\n\n if (log) console.log('z%d: %d clusters in %dms', z, tree.numItems, +Date.now() - now);\n }\n\n if (log) console.timeEnd('total time');\n\n return this;\n }\n\n getClusters(bbox, zoom) {\n let minLng = ((bbox[0] + 180) % 360 + 360) % 360 - 180;\n const minLat = Math.max(-90, Math.min(90, bbox[1]));\n let maxLng = bbox[2] === 180 ? 180 : ((bbox[2] + 180) % 360 + 360) % 360 - 180;\n const maxLat = Math.max(-90, Math.min(90, bbox[3]));\n\n if (bbox[2] - bbox[0] >= 360) {\n minLng = -180;\n maxLng = 180;\n } else if (minLng > maxLng) {\n const easternHem = this.getClusters([minLng, minLat, 180, maxLat], zoom);\n const westernHem = this.getClusters([-180, minLat, maxLng, maxLat], zoom);\n return easternHem.concat(westernHem);\n }\n\n const tree = this.trees[this._limitZoom(zoom)];\n const ids = tree.range(lngX(minLng), latY(maxLat), lngX(maxLng), latY(minLat));\n const data = tree.data;\n const clusters = [];\n for (const id of ids) {\n const k = this.stride * id;\n clusters.push(data[k + OFFSET_NUM] > 1 ? getClusterJSON(data, k, this.clusterProps) : this.points[data[k + OFFSET_ID]]);\n }\n return clusters;\n }\n\n getChildren(clusterId) {\n const originId = this._getOriginId(clusterId);\n const originZoom = this._getOriginZoom(clusterId);\n const errorMsg = 'No cluster with the specified id.';\n\n const tree = this.trees[originZoom];\n if (!tree) throw new Error(errorMsg);\n\n const data = tree.data;\n if (originId * this.stride >= data.length) throw new Error(errorMsg);\n\n const r = this.options.radius / (this.options.extent * Math.pow(2, originZoom - 1));\n const x = data[originId * this.stride];\n const y = data[originId * this.stride + 1];\n const ids = tree.within(x, y, r);\n const children = [];\n for (const id of ids) {\n const k = id * this.stride;\n if (data[k + OFFSET_PARENT] === clusterId) {\n children.push(data[k + OFFSET_NUM] > 1 ? getClusterJSON(data, k, this.clusterProps) : this.points[data[k + OFFSET_ID]]);\n }\n }\n\n if (children.length === 0) throw new Error(errorMsg);\n\n return children;\n }\n\n getLeaves(clusterId, limit, offset) {\n limit = limit || 10;\n offset = offset || 0;\n\n const leaves = [];\n this._appendLeaves(leaves, clusterId, limit, offset, 0);\n\n return leaves;\n }\n\n getTile(z, x, y) {\n const tree = this.trees[this._limitZoom(z)];\n const z2 = Math.pow(2, z);\n const {extent, radius} = this.options;\n const p = radius / extent;\n const top = (y - p) / z2;\n const bottom = (y + 1 + p) / z2;\n\n const tile = {\n features: []\n };\n\n this._addTileFeatures(\n tree.range((x - p) / z2, top, (x + 1 + p) / z2, bottom),\n tree.data, x, y, z2, tile);\n\n if (x === 0) {\n this._addTileFeatures(\n tree.range(1 - p / z2, top, 1, bottom),\n tree.data, z2, y, z2, tile);\n }\n if (x === z2 - 1) {\n this._addTileFeatures(\n tree.range(0, top, p / z2, bottom),\n tree.data, -1, y, z2, tile);\n }\n\n return tile.features.length ? tile : null;\n }\n\n getClusterExpansionZoom(clusterId) {\n let expansionZoom = this._getOriginZoom(clusterId) - 1;\n while (expansionZoom <= this.options.maxZoom) {\n const children = this.getChildren(clusterId);\n expansionZoom++;\n if (children.length !== 1) break;\n clusterId = children[0].properties.cluster_id;\n }\n return expansionZoom;\n }\n\n _appendLeaves(result, clusterId, limit, offset, skipped) {\n const children = this.getChildren(clusterId);\n\n for (const child of children) {\n const props = child.properties;\n\n if (props && props.cluster) {\n if (skipped + props.point_count <= offset) {\n // skip the whole cluster\n skipped += props.point_count;\n } else {\n // enter the cluster\n skipped = this._appendLeaves(result, props.cluster_id, limit, offset, skipped);\n // exit the cluster\n }\n } else if (skipped < offset) {\n // skip a single point\n skipped++;\n } else {\n // add a single point\n result.push(child);\n }\n if (result.length === limit) break;\n }\n\n return skipped;\n }\n\n _createTree(data) {\n const tree = new KDBush(data.length / this.stride | 0, this.options.nodeSize, Float32Array);\n for (let i = 0; i < data.length; i += this.stride) tree.add(data[i], data[i + 1]);\n tree.finish();\n tree.data = data;\n return tree;\n }\n\n _addTileFeatures(ids, data, x, y, z2, tile) {\n for (const i of ids) {\n const k = i * this.stride;\n const isCluster = data[k + OFFSET_NUM] > 1;\n\n let tags, px, py;\n if (isCluster) {\n tags = getClusterProperties(data, k, this.clusterProps);\n px = data[k];\n py = data[k + 1];\n } else {\n const p = this.points[data[k + OFFSET_ID]];\n tags = p.properties;\n const [lng, lat] = p.geometry.coordinates;\n px = lngX(lng);\n py = latY(lat);\n }\n\n const f = {\n type: 1,\n geometry: [[\n Math.round(this.options.extent * (px * z2 - x)),\n Math.round(this.options.extent * (py * z2 - y))\n ]],\n tags\n };\n\n // assign id\n let id;\n if (isCluster || this.options.generateId) {\n // optionally generate id for points\n id = data[k + OFFSET_ID];\n } else {\n // keep id if already assigned\n id = this.points[data[k + OFFSET_ID]].id;\n }\n\n if (id !== undefined) f.id = id;\n\n tile.features.push(f);\n }\n }\n\n _limitZoom(z) {\n return Math.max(this.options.minZoom, Math.min(Math.floor(+z), this.options.maxZoom + 1));\n }\n\n _cluster(tree, zoom) {\n const {radius, extent, reduce, minPoints} = this.options;\n const r = radius / (extent * Math.pow(2, zoom));\n const data = tree.data;\n const nextData = [];\n const stride = this.stride;\n\n // loop through each point\n for (let i = 0; i < data.length; i += stride) {\n // if we've already visited the point at this zoom level, skip it\n if (data[i + OFFSET_ZOOM] <= zoom) continue;\n data[i + OFFSET_ZOOM] = zoom;\n\n // find all nearby points\n const x = data[i];\n const y = data[i + 1];\n const neighborIds = tree.within(data[i], data[i + 1], r);\n\n const numPointsOrigin = data[i + OFFSET_NUM];\n let numPoints = numPointsOrigin;\n\n // count the number of points in a potential cluster\n for (const neighborId of neighborIds) {\n const k = neighborId * stride;\n // filter out neighbors that are already processed\n if (data[k + OFFSET_ZOOM] > zoom) numPoints += data[k + OFFSET_NUM];\n }\n\n // if there were neighbors to merge, and there are enough points to form a cluster\n if (numPoints > numPointsOrigin && numPoints >= minPoints) {\n let wx = x * numPointsOrigin;\n let wy = y * numPointsOrigin;\n\n let clusterProperties;\n let clusterPropIndex = -1;\n\n // encode both zoom and point index on which the cluster originated -- offset by total length of features\n const id = ((i / stride | 0) << 5) + (zoom + 1) + this.points.length;\n\n for (const neighborId of neighborIds) {\n const k = neighborId * stride;\n\n if (data[k + OFFSET_ZOOM] <= zoom) continue;\n data[k + OFFSET_ZOOM] = zoom; // save the zoom (so it doesn't get processed twice)\n\n const numPoints2 = data[k + OFFSET_NUM];\n wx += data[k] * numPoints2; // accumulate coordinates for calculating weighted center\n wy += data[k + 1] * numPoints2;\n\n data[k + OFFSET_PARENT] = id;\n\n if (reduce) {\n if (!clusterProperties) {\n clusterProperties = this._map(data, i, true);\n clusterPropIndex = this.clusterProps.length;\n this.clusterProps.push(clusterProperties);\n }\n reduce(clusterProperties, this._map(data, k));\n }\n }\n\n data[i + OFFSET_PARENT] = id;\n nextData.push(wx / numPoints, wy / numPoints, Infinity, id, -1, numPoints);\n if (reduce) nextData.push(clusterPropIndex);\n\n } else { // left points as unclustered\n for (let j = 0; j < stride; j++) nextData.push(data[i + j]);\n\n if (numPoints > 1) {\n for (const neighborId of neighborIds) {\n const k = neighborId * stride;\n if (data[k + OFFSET_ZOOM] <= zoom) continue;\n data[k + OFFSET_ZOOM] = zoom;\n for (let j = 0; j < stride; j++) nextData.push(data[k + j]);\n }\n }\n }\n }\n\n return nextData;\n }\n\n // get index of the point from which the cluster originated\n _getOriginId(clusterId) {\n return (clusterId - this.points.length) >> 5;\n }\n\n // get zoom of the point from which the cluster originated\n _getOriginZoom(clusterId) {\n return (clusterId - this.points.length) % 32;\n }\n\n _map(data, i, clone) {\n if (data[i + OFFSET_NUM] > 1) {\n const props = this.clusterProps[data[i + OFFSET_PROP]];\n return clone ? Object.assign({}, props) : props;\n }\n const original = this.points[data[i + OFFSET_ID]].properties;\n const result = this.options.map(original);\n return clone && result === original ? Object.assign({}, result) : result;\n }\n}\n\nfunction getClusterJSON(data, i, clusterProps) {\n return {\n type: 'Feature',\n id: data[i + OFFSET_ID],\n properties: getClusterProperties(data, i, clusterProps),\n geometry: {\n type: 'Point',\n coordinates: [xLng(data[i]), yLat(data[i + 1])]\n }\n };\n}\n\nfunction getClusterProperties(data, i, clusterProps) {\n const count = data[i + OFFSET_NUM];\n const abbrev =\n count >= 10000 ? `${Math.round(count / 1000) }k` :\n count >= 1000 ? `${Math.round(count / 100) / 10 }k` : count;\n const propIndex = data[i + OFFSET_PROP];\n const properties = propIndex === -1 ? {} : Object.assign({}, clusterProps[propIndex]);\n return Object.assign(properties, {\n cluster: true,\n cluster_id: data[i + OFFSET_ID],\n point_count: count,\n point_count_abbreviated: abbrev\n });\n}\n\n// longitude/latitude to spherical mercator in [0..1] range\nfunction lngX(lng) {\n return lng / 360 + 0.5;\n}\nfunction latY(lat) {\n const sin = Math.sin(lat * Math.PI / 180);\n const y = (0.5 - 0.25 * Math.log((1 + sin) / (1 - sin)) / Math.PI);\n return y < 0 ? 0 : y > 1 ? 1 : y;\n}\n\n// spherical mercator to longitude/latitude\nfunction xLng(x) {\n return (x - 0.5) * 360;\n}\nfunction yLat(y) {\n const y2 = (180 - y * 360) * Math.PI / 180;\n return 360 * Math.atan(Math.exp(y2)) / Math.PI - 90;\n}\n","\n// calculate simplification data using optimized Douglas-Peucker algorithm\n\nexport default function simplify(coords, first, last, sqTolerance) {\n var maxSqDist = sqTolerance;\n var mid = (last - first) >> 1;\n var minPosToMid = last - first;\n var index;\n\n var ax = coords[first];\n var ay = coords[first + 1];\n var bx = coords[last];\n var by = coords[last + 1];\n\n for (var i = first + 3; i < last; i += 3) {\n var d = getSqSegDist(coords[i], coords[i + 1], ax, ay, bx, by);\n\n if (d > maxSqDist) {\n index = i;\n maxSqDist = d;\n\n } else if (d === maxSqDist) {\n // a workaround to ensure we choose a pivot close to the middle of the list,\n // reducing recursion depth, for certain degenerate inputs\n // https://github.com/mapbox/geojson-vt/issues/104\n var posToMid = Math.abs(i - mid);\n if (posToMid < minPosToMid) {\n index = i;\n minPosToMid = posToMid;\n }\n }\n }\n\n if (maxSqDist > sqTolerance) {\n if (index - first > 3) simplify(coords, first, index, sqTolerance);\n coords[index + 2] = maxSqDist;\n if (last - index > 3) simplify(coords, index, last, sqTolerance);\n }\n}\n\n// square distance from a point to a segment\nfunction getSqSegDist(px, py, x, y, bx, by) {\n\n var dx = bx - x;\n var dy = by - y;\n\n if (dx !== 0 || dy !== 0) {\n\n var t = ((px - x) * dx + (py - y) * dy) / (dx * dx + dy * dy);\n\n if (t > 1) {\n x = bx;\n y = by;\n\n } else if (t > 0) {\n x += dx * t;\n y += dy * t;\n }\n }\n\n dx = px - x;\n dy = py - y;\n\n return dx * dx + dy * dy;\n}\n","\nexport default function createFeature(id, type, geom, tags) {\n var feature = {\n id: typeof id === 'undefined' ? null : id,\n type: type,\n geometry: geom,\n tags: tags,\n minX: Infinity,\n minY: Infinity,\n maxX: -Infinity,\n maxY: -Infinity\n };\n calcBBox(feature);\n return feature;\n}\n\nfunction calcBBox(feature) {\n var geom = feature.geometry;\n var type = feature.type;\n\n if (type === 'Point' || type === 'MultiPoint' || type === 'LineString') {\n calcLineBBox(feature, geom);\n\n } else if (type === 'Polygon' || type === 'MultiLineString') {\n for (var i = 0; i < geom.length; i++) {\n calcLineBBox(feature, geom[i]);\n }\n\n } else if (type === 'MultiPolygon') {\n for (i = 0; i < geom.length; i++) {\n for (var j = 0; j < geom[i].length; j++) {\n calcLineBBox(feature, geom[i][j]);\n }\n }\n }\n}\n\nfunction calcLineBBox(feature, geom) {\n for (var i = 0; i < geom.length; i += 3) {\n feature.minX = Math.min(feature.minX, geom[i]);\n feature.minY = Math.min(feature.minY, geom[i + 1]);\n feature.maxX = Math.max(feature.maxX, geom[i]);\n feature.maxY = Math.max(feature.maxY, geom[i + 1]);\n }\n}\n","\nimport simplify from './simplify';\nimport createFeature from './feature';\n\n// converts GeoJSON feature into an intermediate projected JSON vector format with simplification data\n\nexport default function convert(data, options) {\n var features = [];\n if (data.type === 'FeatureCollection') {\n for (var i = 0; i < data.features.length; i++) {\n convertFeature(features, data.features[i], options, i);\n }\n\n } else if (data.type === 'Feature') {\n convertFeature(features, data, options);\n\n } else {\n // single geometry or a geometry collection\n convertFeature(features, {geometry: data}, options);\n }\n\n return features;\n}\n\nfunction convertFeature(features, geojson, options, index) {\n if (!geojson.geometry) return;\n\n var coords = geojson.geometry.coordinates;\n var type = geojson.geometry.type;\n var tolerance = Math.pow(options.tolerance / ((1 << options.maxZoom) * options.extent), 2);\n var geometry = [];\n var id = geojson.id;\n if (options.promoteId) {\n id = geojson.properties[options.promoteId];\n } else if (options.generateId) {\n id = index || 0;\n }\n if (type === 'Point') {\n convertPoint(coords, geometry);\n\n } else if (type === 'MultiPoint') {\n for (var i = 0; i < coords.length; i++) {\n convertPoint(coords[i], geometry);\n }\n\n } else if (type === 'LineString') {\n convertLine(coords, geometry, tolerance, false);\n\n } else if (type === 'MultiLineString') {\n if (options.lineMetrics) {\n // explode into linestrings to be able to track metrics\n for (i = 0; i < coords.length; i++) {\n geometry = [];\n convertLine(coords[i], geometry, tolerance, false);\n features.push(createFeature(id, 'LineString', geometry, geojson.properties));\n }\n return;\n } else {\n convertLines(coords, geometry, tolerance, false);\n }\n\n } else if (type === 'Polygon') {\n convertLines(coords, geometry, tolerance, true);\n\n } else if (type === 'MultiPolygon') {\n for (i = 0; i < coords.length; i++) {\n var polygon = [];\n convertLines(coords[i], polygon, tolerance, true);\n geometry.push(polygon);\n }\n } else if (type === 'GeometryCollection') {\n for (i = 0; i < geojson.geometry.geometries.length; i++) {\n convertFeature(features, {\n id: id,\n geometry: geojson.geometry.geometries[i],\n properties: geojson.properties\n }, options, index);\n }\n return;\n } else {\n throw new Error('Input data is not a valid GeoJSON object.');\n }\n\n features.push(createFeature(id, type, geometry, geojson.properties));\n}\n\nfunction convertPoint(coords, out) {\n out.push(projectX(coords[0]));\n out.push(projectY(coords[1]));\n out.push(0);\n}\n\nfunction convertLine(ring, out, tolerance, isPolygon) {\n var x0, y0;\n var size = 0;\n\n for (var j = 0; j < ring.length; j++) {\n var x = projectX(ring[j][0]);\n var y = projectY(ring[j][1]);\n\n out.push(x);\n out.push(y);\n out.push(0);\n\n if (j > 0) {\n if (isPolygon) {\n size += (x0 * y - x * y0) / 2; // area\n } else {\n size += Math.sqrt(Math.pow(x - x0, 2) + Math.pow(y - y0, 2)); // length\n }\n }\n x0 = x;\n y0 = y;\n }\n\n var last = out.length - 3;\n out[2] = 1;\n simplify(out, 0, last, tolerance);\n out[last + 2] = 1;\n\n out.size = Math.abs(size);\n out.start = 0;\n out.end = out.size;\n}\n\nfunction convertLines(rings, out, tolerance, isPolygon) {\n for (var i = 0; i < rings.length; i++) {\n var geom = [];\n convertLine(rings[i], geom, tolerance, isPolygon);\n out.push(geom);\n }\n}\n\nfunction projectX(x) {\n return x / 360 + 0.5;\n}\n\nfunction projectY(y) {\n var sin = Math.sin(y * Math.PI / 180);\n var y2 = 0.5 - 0.25 * Math.log((1 + sin) / (1 - sin)) / Math.PI;\n return y2 < 0 ? 0 : y2 > 1 ? 1 : y2;\n}\n","\nimport createFeature from './feature';\n\n/* clip features between two axis-parallel lines:\n * | |\n * ___|___ | /\n * / | \\____|____/\n * | |\n */\n\nexport default function clip(features, scale, k1, k2, axis, minAll, maxAll, options) {\n\n k1 /= scale;\n k2 /= scale;\n\n if (minAll >= k1 && maxAll < k2) return features; // trivial accept\n else if (maxAll < k1 || minAll >= k2) return null; // trivial reject\n\n var clipped = [];\n\n for (var i = 0; i < features.length; i++) {\n\n var feature = features[i];\n var geometry = feature.geometry;\n var type = feature.type;\n\n var min = axis === 0 ? feature.minX : feature.minY;\n var max = axis === 0 ? feature.maxX : feature.maxY;\n\n if (min >= k1 && max < k2) { // trivial accept\n clipped.push(feature);\n continue;\n } else if (max < k1 || min >= k2) { // trivial reject\n continue;\n }\n\n var newGeometry = [];\n\n if (type === 'Point' || type === 'MultiPoint') {\n clipPoints(geometry, newGeometry, k1, k2, axis);\n\n } else if (type === 'LineString') {\n clipLine(geometry, newGeometry, k1, k2, axis, false, options.lineMetrics);\n\n } else if (type === 'MultiLineString') {\n clipLines(geometry, newGeometry, k1, k2, axis, false);\n\n } else if (type === 'Polygon') {\n clipLines(geometry, newGeometry, k1, k2, axis, true);\n\n } else if (type === 'MultiPolygon') {\n for (var j = 0; j < geometry.length; j++) {\n var polygon = [];\n clipLines(geometry[j], polygon, k1, k2, axis, true);\n if (polygon.length) {\n newGeometry.push(polygon);\n }\n }\n }\n\n if (newGeometry.length) {\n if (options.lineMetrics && type === 'LineString') {\n for (j = 0; j < newGeometry.length; j++) {\n clipped.push(createFeature(feature.id, type, newGeometry[j], feature.tags));\n }\n continue;\n }\n\n if (type === 'LineString' || type === 'MultiLineString') {\n if (newGeometry.length === 1) {\n type = 'LineString';\n newGeometry = newGeometry[0];\n } else {\n type = 'MultiLineString';\n }\n }\n if (type === 'Point' || type === 'MultiPoint') {\n type = newGeometry.length === 3 ? 'Point' : 'MultiPoint';\n }\n\n clipped.push(createFeature(feature.id, type, newGeometry, feature.tags));\n }\n }\n\n return clipped.length ? clipped : null;\n}\n\nfunction clipPoints(geom, newGeom, k1, k2, axis) {\n for (var i = 0; i < geom.length; i += 3) {\n var a = geom[i + axis];\n\n if (a >= k1 && a <= k2) {\n newGeom.push(geom[i]);\n newGeom.push(geom[i + 1]);\n newGeom.push(geom[i + 2]);\n }\n }\n}\n\nfunction clipLine(geom, newGeom, k1, k2, axis, isPolygon, trackMetrics) {\n\n var slice = newSlice(geom);\n var intersect = axis === 0 ? intersectX : intersectY;\n var len = geom.start;\n var segLen, t;\n\n for (var i = 0; i < geom.length - 3; i += 3) {\n var ax = geom[i];\n var ay = geom[i + 1];\n var az = geom[i + 2];\n var bx = geom[i + 3];\n var by = geom[i + 4];\n var a = axis === 0 ? ax : ay;\n var b = axis === 0 ? bx : by;\n var exited = false;\n\n if (trackMetrics) segLen = Math.sqrt(Math.pow(ax - bx, 2) + Math.pow(ay - by, 2));\n\n if (a < k1) {\n // ---|--> | (line enters the clip region from the left)\n if (b > k1) {\n t = intersect(slice, ax, ay, bx, by, k1);\n if (trackMetrics) slice.start = len + segLen * t;\n }\n } else if (a > k2) {\n // | <--|--- (line enters the clip region from the right)\n if (b < k2) {\n t = intersect(slice, ax, ay, bx, by, k2);\n if (trackMetrics) slice.start = len + segLen * t;\n }\n } else {\n addPoint(slice, ax, ay, az);\n }\n if (b < k1 && a >= k1) {\n // <--|--- | or <--|-----|--- (line exits the clip region on the left)\n t = intersect(slice, ax, ay, bx, by, k1);\n exited = true;\n }\n if (b > k2 && a <= k2) {\n // | ---|--> or ---|-----|--> (line exits the clip region on the right)\n t = intersect(slice, ax, ay, bx, by, k2);\n exited = true;\n }\n\n if (!isPolygon && exited) {\n if (trackMetrics) slice.end = len + segLen * t;\n newGeom.push(slice);\n slice = newSlice(geom);\n }\n\n if (trackMetrics) len += segLen;\n }\n\n // add the last point\n var last = geom.length - 3;\n ax = geom[last];\n ay = geom[last + 1];\n az = geom[last + 2];\n a = axis === 0 ? ax : ay;\n if (a >= k1 && a <= k2) addPoint(slice, ax, ay, az);\n\n // close the polygon if its endpoints are not the same after clipping\n last = slice.length - 3;\n if (isPolygon && last >= 3 && (slice[last] !== slice[0] || slice[last + 1] !== slice[1])) {\n addPoint(slice, slice[0], slice[1], slice[2]);\n }\n\n // add the final slice\n if (slice.length) {\n newGeom.push(slice);\n }\n}\n\nfunction newSlice(line) {\n var slice = [];\n slice.size = line.size;\n slice.start = line.start;\n slice.end = line.end;\n return slice;\n}\n\nfunction clipLines(geom, newGeom, k1, k2, axis, isPolygon) {\n for (var i = 0; i < geom.length; i++) {\n clipLine(geom[i], newGeom, k1, k2, axis, isPolygon, false);\n }\n}\n\nfunction addPoint(out, x, y, z) {\n out.push(x);\n out.push(y);\n out.push(z);\n}\n\nfunction intersectX(out, ax, ay, bx, by, x) {\n var t = (x - ax) / (bx - ax);\n out.push(x);\n out.push(ay + (by - ay) * t);\n out.push(1);\n return t;\n}\n\nfunction intersectY(out, ax, ay, bx, by, y) {\n var t = (y - ay) / (by - ay);\n out.push(ax + (bx - ax) * t);\n out.push(y);\n out.push(1);\n return t;\n}\n","\nimport clip from './clip';\nimport createFeature from './feature';\n\nexport default function wrap(features, options) {\n var buffer = options.buffer / options.extent;\n var merged = features;\n var left = clip(features, 1, -1 - buffer, buffer, 0, -1, 2, options); // left world copy\n var right = clip(features, 1, 1 - buffer, 2 + buffer, 0, -1, 2, options); // right world copy\n\n if (left || right) {\n merged = clip(features, 1, -buffer, 1 + buffer, 0, -1, 2, options) || []; // center world copy\n\n if (left) merged = shiftFeatureCoords(left, 1).concat(merged); // merge left into center\n if (right) merged = merged.concat(shiftFeatureCoords(right, -1)); // merge right into center\n }\n\n return merged;\n}\n\nfunction shiftFeatureCoords(features, offset) {\n var newFeatures = [];\n\n for (var i = 0; i < features.length; i++) {\n var feature = features[i],\n type = feature.type;\n\n var newGeometry;\n\n if (type === 'Point' || type === 'MultiPoint' || type === 'LineString') {\n newGeometry = shiftCoords(feature.geometry, offset);\n\n } else if (type === 'MultiLineString' || type === 'Polygon') {\n newGeometry = [];\n for (var j = 0; j < feature.geometry.length; j++) {\n newGeometry.push(shiftCoords(feature.geometry[j], offset));\n }\n } else if (type === 'MultiPolygon') {\n newGeometry = [];\n for (j = 0; j < feature.geometry.length; j++) {\n var newPolygon = [];\n for (var k = 0; k < feature.geometry[j].length; k++) {\n newPolygon.push(shiftCoords(feature.geometry[j][k], offset));\n }\n newGeometry.push(newPolygon);\n }\n }\n\n newFeatures.push(createFeature(feature.id, type, newGeometry, feature.tags));\n }\n\n return newFeatures;\n}\n\nfunction shiftCoords(points, offset) {\n var newPoints = [];\n newPoints.size = points.size;\n\n if (points.start !== undefined) {\n newPoints.start = points.start;\n newPoints.end = points.end;\n }\n\n for (var i = 0; i < points.length; i += 3) {\n newPoints.push(points[i] + offset, points[i + 1], points[i + 2]);\n }\n return newPoints;\n}\n","\n// Transforms the coordinates of each feature in the given tile from\n// mercator-projected space into (extent x extent) tile space.\nexport default function transformTile(tile, extent) {\n if (tile.transformed) return tile;\n\n var z2 = 1 << tile.z,\n tx = tile.x,\n ty = tile.y,\n i, j, k;\n\n for (i = 0; i < tile.features.length; i++) {\n var feature = tile.features[i],\n geom = feature.geometry,\n type = feature.type;\n\n feature.geometry = [];\n\n if (type === 1) {\n for (j = 0; j < geom.length; j += 2) {\n feature.geometry.push(transformPoint(geom[j], geom[j + 1], extent, z2, tx, ty));\n }\n } else {\n for (j = 0; j < geom.length; j++) {\n var ring = [];\n for (k = 0; k < geom[j].length; k += 2) {\n ring.push(transformPoint(geom[j][k], geom[j][k + 1], extent, z2, tx, ty));\n }\n feature.geometry.push(ring);\n }\n }\n }\n\n tile.transformed = true;\n\n return tile;\n}\n\nfunction transformPoint(x, y, extent, z2, tx, ty) {\n return [\n Math.round(extent * (x * z2 - tx)),\n Math.round(extent * (y * z2 - ty))];\n}\n","\nexport default function createTile(features, z, tx, ty, options) {\n var tolerance = z === options.maxZoom ? 0 : options.tolerance / ((1 << z) * options.extent);\n var tile = {\n features: [],\n numPoints: 0,\n numSimplified: 0,\n numFeatures: 0,\n source: null,\n x: tx,\n y: ty,\n z: z,\n transformed: false,\n minX: 2,\n minY: 1,\n maxX: -1,\n maxY: 0\n };\n for (var i = 0; i < features.length; i++) {\n tile.numFeatures++;\n addFeature(tile, features[i], tolerance, options);\n\n var minX = features[i].minX;\n var minY = features[i].minY;\n var maxX = features[i].maxX;\n var maxY = features[i].maxY;\n\n if (minX < tile.minX) tile.minX = minX;\n if (minY < tile.minY) tile.minY = minY;\n if (maxX > tile.maxX) tile.maxX = maxX;\n if (maxY > tile.maxY) tile.maxY = maxY;\n }\n return tile;\n}\n\nfunction addFeature(tile, feature, tolerance, options) {\n\n var geom = feature.geometry,\n type = feature.type,\n simplified = [];\n\n if (type === 'Point' || type === 'MultiPoint') {\n for (var i = 0; i < geom.length; i += 3) {\n simplified.push(geom[i]);\n simplified.push(geom[i + 1]);\n tile.numPoints++;\n tile.numSimplified++;\n }\n\n } else if (type === 'LineString') {\n addLine(simplified, geom, tile, tolerance, false, false);\n\n } else if (type === 'MultiLineString' || type === 'Polygon') {\n for (i = 0; i < geom.length; i++) {\n addLine(simplified, geom[i], tile, tolerance, type === 'Polygon', i === 0);\n }\n\n } else if (type === 'MultiPolygon') {\n\n for (var k = 0; k < geom.length; k++) {\n var polygon = geom[k];\n for (i = 0; i < polygon.length; i++) {\n addLine(simplified, polygon[i], tile, tolerance, true, i === 0);\n }\n }\n }\n\n if (simplified.length) {\n var tags = feature.tags || null;\n if (type === 'LineString' && options.lineMetrics) {\n tags = {};\n for (var key in feature.tags) tags[key] = feature.tags[key];\n tags['mapbox_clip_start'] = geom.start / geom.size;\n tags['mapbox_clip_end'] = geom.end / geom.size;\n }\n var tileFeature = {\n geometry: simplified,\n type: type === 'Polygon' || type === 'MultiPolygon' ? 3 :\n type === 'LineString' || type === 'MultiLineString' ? 2 : 1,\n tags: tags\n };\n if (feature.id !== null) {\n tileFeature.id = feature.id;\n }\n tile.features.push(tileFeature);\n }\n}\n\nfunction addLine(result, geom, tile, tolerance, isPolygon, isOuter) {\n var sqTolerance = tolerance * tolerance;\n\n if (tolerance > 0 && (geom.size < (isPolygon ? sqTolerance : tolerance))) {\n tile.numPoints += geom.length / 3;\n return;\n }\n\n var ring = [];\n\n for (var i = 0; i < geom.length; i += 3) {\n if (tolerance === 0 || geom[i + 2] > sqTolerance) {\n tile.numSimplified++;\n ring.push(geom[i]);\n ring.push(geom[i + 1]);\n }\n tile.numPoints++;\n }\n\n if (isPolygon) rewind(ring, isOuter);\n\n result.push(ring);\n}\n\nfunction rewind(ring, clockwise) {\n var area = 0;\n for (var i = 0, len = ring.length, j = len - 2; i < len; j = i, i += 2) {\n area += (ring[i] - ring[j]) * (ring[i + 1] + ring[j + 1]);\n }\n if (area > 0 === clockwise) {\n for (i = 0, len = ring.length; i < len / 2; i += 2) {\n var x = ring[i];\n var y = ring[i + 1];\n ring[i] = ring[len - 2 - i];\n ring[i + 1] = ring[len - 1 - i];\n ring[len - 2 - i] = x;\n ring[len - 1 - i] = y;\n }\n }\n}\n","\nimport convert from './convert'; // GeoJSON conversion and preprocessing\nimport clip from './clip'; // stripe clipping algorithm\nimport wrap from './wrap'; // date line processing\nimport transform from './transform'; // coordinate transformation\nimport createTile from './tile'; // final simplified tile generation\n\nexport default function geojsonvt(data, options) {\n return new GeoJSONVT(data, options);\n}\n\nfunction GeoJSONVT(data, options) {\n options = this.options = extend(Object.create(this.options), options);\n\n var debug = options.debug;\n\n if (debug) console.time('preprocess data');\n\n if (options.maxZoom < 0 || options.maxZoom > 24) throw new Error('maxZoom should be in the 0-24 range');\n if (options.promoteId && options.generateId) throw new Error('promoteId and generateId cannot be used together.');\n\n var features = convert(data, options);\n\n this.tiles = {};\n this.tileCoords = [];\n\n if (debug) {\n console.timeEnd('preprocess data');\n console.log('index: maxZoom: %d, maxPoints: %d', options.indexMaxZoom, options.indexMaxPoints);\n console.time('generate tiles');\n this.stats = {};\n this.total = 0;\n }\n\n features = wrap(features, options);\n\n // start slicing from the top tile down\n if (features.length) this.splitTile(features, 0, 0, 0);\n\n if (debug) {\n if (features.length) console.log('features: %d, points: %d', this.tiles[0].numFeatures, this.tiles[0].numPoints);\n console.timeEnd('generate tiles');\n console.log('tiles generated:', this.total, JSON.stringify(this.stats));\n }\n}\n\nGeoJSONVT.prototype.options = {\n maxZoom: 14, // max zoom to preserve detail on\n indexMaxZoom: 5, // max zoom in the tile index\n indexMaxPoints: 100000, // max number of points per tile in the tile index\n tolerance: 3, // simplification tolerance (higher means simpler)\n extent: 4096, // tile extent\n buffer: 64, // tile buffer on each side\n lineMetrics: false, // whether to calculate line metrics\n promoteId: null, // name of a feature property to be promoted to feature.id\n generateId: false, // whether to generate feature ids. Cannot be used with promoteId\n debug: 0 // logging level (0, 1 or 2)\n};\n\nGeoJSONVT.prototype.splitTile = function (features, z, x, y, cz, cx, cy) {\n\n var stack = [features, z, x, y],\n options = this.options,\n debug = options.debug;\n\n // avoid recursion by using a processing queue\n while (stack.length) {\n y = stack.pop();\n x = stack.pop();\n z = stack.pop();\n features = stack.pop();\n\n var z2 = 1 << z,\n id = toID(z, x, y),\n tile = this.tiles[id];\n\n if (!tile) {\n if (debug > 1) console.time('creation');\n\n tile = this.tiles[id] = createTile(features, z, x, y, options);\n this.tileCoords.push({z: z, x: x, y: y});\n\n if (debug) {\n if (debug > 1) {\n console.log('tile z%d-%d-%d (features: %d, points: %d, simplified: %d)',\n z, x, y, tile.numFeatures, tile.numPoints, tile.numSimplified);\n console.timeEnd('creation');\n }\n var key = 'z' + z;\n this.stats[key] = (this.stats[key] || 0) + 1;\n this.total++;\n }\n }\n\n // save reference to original geometry in tile so that we can drill down later if we stop now\n tile.source = features;\n\n // if it's the first-pass tiling\n if (!cz) {\n // stop tiling if we reached max zoom, or if the tile is too simple\n if (z === options.indexMaxZoom || tile.numPoints <= options.indexMaxPoints) continue;\n\n // if a drilldown to a specific tile\n } else {\n // stop tiling if we reached base zoom or our target tile zoom\n if (z === options.maxZoom || z === cz) continue;\n\n // stop tiling if it's not an ancestor of the target tile\n var m = 1 << (cz - z);\n if (x !== Math.floor(cx / m) || y !== Math.floor(cy / m)) continue;\n }\n\n // if we slice further down, no need to keep source geometry\n tile.source = null;\n\n if (features.length === 0) continue;\n\n if (debug > 1) console.time('clipping');\n\n // values we'll use for clipping\n var k1 = 0.5 * options.buffer / options.extent,\n k2 = 0.5 - k1,\n k3 = 0.5 + k1,\n k4 = 1 + k1,\n tl, bl, tr, br, left, right;\n\n tl = bl = tr = br = null;\n\n left = clip(features, z2, x - k1, x + k3, 0, tile.minX, tile.maxX, options);\n right = clip(features, z2, x + k2, x + k4, 0, tile.minX, tile.maxX, options);\n features = null;\n\n if (left) {\n tl = clip(left, z2, y - k1, y + k3, 1, tile.minY, tile.maxY, options);\n bl = clip(left, z2, y + k2, y + k4, 1, tile.minY, tile.maxY, options);\n left = null;\n }\n\n if (right) {\n tr = clip(right, z2, y - k1, y + k3, 1, tile.minY, tile.maxY, options);\n br = clip(right, z2, y + k2, y + k4, 1, tile.minY, tile.maxY, options);\n right = null;\n }\n\n if (debug > 1) console.timeEnd('clipping');\n\n stack.push(tl || [], z + 1, x * 2, y * 2);\n stack.push(bl || [], z + 1, x * 2, y * 2 + 1);\n stack.push(tr || [], z + 1, x * 2 + 1, y * 2);\n stack.push(br || [], z + 1, x * 2 + 1, y * 2 + 1);\n }\n};\n\nGeoJSONVT.prototype.getTile = function (z, x, y) {\n var options = this.options,\n extent = options.extent,\n debug = options.debug;\n\n if (z < 0 || z > 24) return null;\n\n var z2 = 1 << z;\n x = ((x % z2) + z2) % z2; // wrap tile x coordinate\n\n var id = toID(z, x, y);\n if (this.tiles[id]) return transform(this.tiles[id], extent);\n\n if (debug > 1) console.log('drilling down to z%d-%d-%d', z, x, y);\n\n var z0 = z,\n x0 = x,\n y0 = y,\n parent;\n\n while (!parent && z0 > 0) {\n z0--;\n x0 = Math.floor(x0 / 2);\n y0 = Math.floor(y0 / 2);\n parent = this.tiles[toID(z0, x0, y0)];\n }\n\n if (!parent || !parent.source) return null;\n\n // if we found a parent tile containing the original geometry, we can drill down from it\n if (debug > 1) console.log('found parent tile z%d-%d-%d', z0, x0, y0);\n\n if (debug > 1) console.time('drilling down');\n this.splitTile(parent.source, z0, x0, y0, z, x, y);\n if (debug > 1) console.timeEnd('drilling down');\n\n return this.tiles[id] ? transform(this.tiles[id], extent) : null;\n};\n\nfunction toID(z, x, y) {\n return (((1 << z) * y + x) * 32) + z;\n}\n\nfunction extend(dest, src) {\n for (var i in src) dest[i] = src[i];\n return dest;\n}\n","/**\n * A way to indentify a feature, either by string or by number\n */\nexport type GeoJSONFeatureId = number | string;\n\n/**\n * The geojson source diff object\n */\nexport type GeoJSONSourceDiff = {\n /**\n * When set to `true` it will remove all features\n */\n removeAll?: boolean;\n /**\n * An array of features IDs to remove\n */\n remove?: Array;\n /**\n * An array of features to add\n */\n add?: Array;\n /**\n * An array of update objects\n */\n update?: Array;\n}\n\n/**\n * A geojson feature diff object\n */\nexport type GeoJSONFeatureDiff = {\n /**\n * The feature ID\n */\n id: GeoJSONFeatureId;\n /**\n * If it's a new geometry, place it here\n */\n newGeometry?: GeoJSON.Geometry;\n /**\n * Setting to `true` will remove all preperties\n */\n removeAllProperties?: boolean;\n /**\n * The properties keys to remove\n */\n removeProperties?: Array;\n /**\n * The properties to add or update along side their values\n */\n addOrUpdateProperties?: Array<{key: string; value: any}>;\n}\n\nexport type UpdateableGeoJSON = GeoJSON.Feature | GeoJSON.FeatureCollection | undefined;\n\nfunction getFeatureId(feature: GeoJSON.Feature, promoteId?: string): GeoJSONFeatureId | undefined {\n return promoteId ? feature.properties[promoteId] : feature.id;\n}\n\nexport function isUpdateableGeoJSON(data: GeoJSON.GeoJSON | undefined, promoteId?: string): data is UpdateableGeoJSON {\n // null can be updated\n if (data == null) {\n return true;\n }\n\n // a single feature with an id can be updated, need to explicitly check against null because 0 is a valid feature id that is falsy\n if (data.type === 'Feature') {\n return getFeatureId(data, promoteId) != null;\n }\n\n // a feature collection can be updated if every feature has an id, and the ids are all unique\n // this prevents us from silently dropping features if ids get reused\n if (data.type === 'FeatureCollection') {\n const seenIds = new Set();\n for (const feature of data.features) {\n const id = getFeatureId(feature, promoteId);\n if (id == null) {\n return false;\n }\n\n if (seenIds.has(id)) {\n return false;\n }\n\n seenIds.add(id);\n }\n\n return true;\n }\n\n return false;\n}\n\nexport function toUpdateable(data: UpdateableGeoJSON, promoteId?: string) {\n const result = new Map();\n if (data == null) {\n // empty result\n } else if (data.type === 'Feature') {\n result.set(getFeatureId(data, promoteId)!, data);\n } else {\n for (const feature of data.features) {\n result.set(getFeatureId(feature, promoteId)!, feature);\n }\n }\n\n return result;\n}\n\n// mutates updateable\nexport function applySourceDiff(updateable: Map, diff: GeoJSONSourceDiff, promoteId?: string): void {\n if (diff.removeAll) {\n updateable.clear();\n }\n\n if (diff.remove) {\n for (const id of diff.remove) {\n updateable.delete(id);\n }\n }\n\n if (diff.add) {\n for (const feature of diff.add) {\n const id = getFeatureId(feature, promoteId);\n\n if (id != null) {\n updateable.set(id, feature);\n }\n }\n }\n\n if (diff.update) {\n for (const update of diff.update) {\n let feature = updateable.get(update.id);\n\n if (feature == null) {\n continue;\n }\n\n // be careful to clone the feature and/or properties objects to avoid mutating our input\n const cloneFeature = update.newGeometry || update.removeAllProperties;\n // note: removeAllProperties gives us a new properties object, so we can skip the clone step\n const cloneProperties = !update.removeAllProperties && (update.removeProperties?.length > 0 || update.addOrUpdateProperties?.length > 0);\n if (cloneFeature || cloneProperties) {\n feature = {...feature};\n updateable.set(update.id, feature);\n if (cloneProperties) {\n feature.properties = {...feature.properties};\n }\n }\n\n if (update.newGeometry) {\n feature.geometry = update.newGeometry;\n }\n\n if (update.removeAllProperties) {\n feature.properties = {};\n } else if (update.removeProperties?.length > 0) {\n for (const prop of update.removeProperties) {\n if (Object.prototype.hasOwnProperty.call(feature.properties, prop)) {\n delete feature.properties[prop];\n }\n }\n }\n\n if (update.addOrUpdateProperties?.length > 0) {\n for (const {key, value} of update.addOrUpdateProperties) {\n feature.properties[key] = value;\n }\n }\n }\n }\n}\n","import {getJSON} from '../util/ajax';\n\nimport {RequestPerformance} from '../util/performance';\nimport rewind from '@mapbox/geojson-rewind';\nimport {GeoJSONWrapper} from './geojson_wrapper';\nimport vtpbf from 'vt-pbf';\nimport Supercluster, {type Options as SuperclusterOptions, type ClusterProperties} from 'supercluster';\nimport geojsonvt, {type Options as GeoJSONVTOptions} from 'geojson-vt';\nimport {VectorTileWorkerSource} from './vector_tile_worker_source';\nimport {createExpression} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {\n WorkerTileParameters,\n WorkerTileCallback,\n} from '../source/worker_source';\n\nimport type {Actor} from '../util/actor';\nimport type {StyleLayerIndex} from '../style/style_layer_index';\n\nimport type {LoadVectorDataCallback} from './vector_tile_worker_source';\nimport type {RequestParameters, ResponseCallback} from '../util/ajax';\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\nimport {isUpdateableGeoJSON, type GeoJSONSourceDiff, applySourceDiff, toUpdateable, GeoJSONFeatureId} from './geojson_source_diff';\n\nexport type LoadGeoJSONParameters = {\n request?: RequestParameters;\n /**\n * Literal GeoJSON data. Must be provided if `request.url` is not.\n */\n data?: string;\n dataDiff?: GeoJSONSourceDiff;\n source: string;\n cluster: boolean;\n superclusterOptions?: SuperclusterOptions;\n geojsonVtOptions?: GeoJSONVTOptions;\n clusterProperties?: ClusterProperties;\n filter?: Array;\n promoteId?: string;\n};\n\nexport type LoadGeoJSON = (params: LoadGeoJSONParameters, callback: ResponseCallback) => Cancelable;\n\ntype GeoJSONIndex = ReturnType | Supercluster;\n\n/**\n * The {@link WorkerSource} implementation that supports {@link GeoJSONSource}.\n * This class is designed to be easily reused to support custom source types\n * for data formats that can be parsed/converted into an in-memory GeoJSON\n * representation. To do so, create it with\n * `new GeoJSONWorkerSource(actor, layerIndex, customLoadGeoJSONFunction)`.\n * For a full example, see [mapbox-gl-topojson](https://github.com/developmentseed/mapbox-gl-topojson).\n */\nexport class GeoJSONWorkerSource extends VectorTileWorkerSource {\n _pendingCallback: Callback<{\n resourceTiming?: {[_: string]: Array};\n abandoned?: boolean;\n }>;\n _pendingRequest: Cancelable;\n _geoJSONIndex: GeoJSONIndex;\n _dataUpdateable = new Map();\n\n /**\n * @param loadGeoJSON - Optional method for custom loading/parsing of\n * GeoJSON based on parameters passed from the main-thread Source.\n * See {@link GeoJSONWorkerSource#loadGeoJSON}.\n */\n constructor(actor: Actor, layerIndex: StyleLayerIndex, availableImages: Array, loadGeoJSON?: LoadGeoJSON | null) {\n super(actor, layerIndex, availableImages);\n this.loadVectorData = this.loadGeoJSONTile;\n if (loadGeoJSON) {\n this.loadGeoJSON = loadGeoJSON;\n }\n }\n\n loadGeoJSONTile(params: WorkerTileParameters, callback: LoadVectorDataCallback): (() => void) | void {\n const canonical = params.tileID.canonical;\n\n if (!this._geoJSONIndex) {\n return callback(null, null); // we couldn't load the file\n }\n\n const geoJSONTile = this._geoJSONIndex.getTile(canonical.z, canonical.x, canonical.y);\n if (!geoJSONTile) {\n return callback(null, null); // nothing in the given tile\n }\n\n const geojsonWrapper = new GeoJSONWrapper(geoJSONTile.features);\n // Encode the geojson-vt tile into binary vector tile form. This\n // is a convenience that allows `FeatureIndex` to operate the same way\n // across `VectorTileSource` and `GeoJSONSource` data.\n let pbf = vtpbf(geojsonWrapper);\n if (pbf.byteOffset !== 0 || pbf.byteLength !== pbf.buffer.byteLength) {\n // Compatibility with node Buffer (https://github.com/mapbox/pbf/issues/35)\n pbf = new Uint8Array(pbf);\n }\n\n callback(null, {\n vectorTile: geojsonWrapper,\n rawData: pbf.buffer\n });\n }\n\n /**\n * Fetches (if appropriate), parses, and index geojson data into tiles. This\n * preparatory method must be called before {@link GeoJSONWorkerSource#loadTile}\n * can correctly serve up tiles.\n *\n * Defers to {@link GeoJSONWorkerSource#loadGeoJSON} for the fetching/parsing,\n * expecting `callback(error, data)` to be called with either an error or a\n * parsed GeoJSON object.\n *\n * When a `loadData` request comes in while a previous one is being processed,\n * the previous one is aborted.\n *\n * @param params - the parameters\n * @param callback - the callback for completion or error\n */\n loadData(params: LoadGeoJSONParameters, callback: Callback<{\n resourceTiming?: {[_: string]: Array};\n abandoned?: boolean;\n }>) {\n this._pendingRequest?.cancel();\n if (this._pendingCallback) {\n // Tell the foreground the previous call has been abandoned\n this._pendingCallback(null, {abandoned: true});\n }\n\n const perf = (params && params.request && params.request.collectResourceTiming) ?\n new RequestPerformance(params.request) : false;\n\n this._pendingCallback = callback;\n this._pendingRequest = this.loadGeoJSON(params, (err?: Error | null, data?: any | null) => {\n delete this._pendingCallback;\n delete this._pendingRequest;\n\n if (err || !data) {\n return callback(err);\n } else if (typeof data !== 'object') {\n return callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`));\n } else {\n rewind(data, true);\n\n try {\n if (params.filter) {\n const compiled = createExpression(params.filter, {type: 'boolean', 'property-type': 'data-driven', overridable: false, transition: false} as any);\n if (compiled.result === 'error')\n throw new Error(compiled.value.map(err => `${err.key}: ${err.message}`).join(', '));\n\n const features = data.features.filter(feature => compiled.value.evaluate({zoom: 0}, feature));\n data = {type: 'FeatureCollection', features};\n }\n\n this._geoJSONIndex = params.cluster ?\n new Supercluster(getSuperclusterOptions(params)).load(data.features) :\n geojsonvt(data, params.geojsonVtOptions);\n } catch (err) {\n return callback(err);\n }\n\n this.loaded = {};\n\n const result = {} as { resourceTiming: any };\n if (perf) {\n const resourceTimingData = perf.finish();\n // it's necessary to eval the result of getEntriesByName() here via parse/stringify\n // late evaluation in the main thread causes TypeError: illegal invocation\n if (resourceTimingData) {\n result.resourceTiming = {};\n result.resourceTiming[params.source] = JSON.parse(JSON.stringify(resourceTimingData));\n }\n }\n callback(null, result);\n }\n });\n }\n\n /**\n * Implements {@link WorkerSource#reloadTile}.\n *\n * If the tile is loaded, uses the implementation in VectorTileWorkerSource.\n * Otherwise, such as after a setData() call, we load the tile fresh.\n *\n * @param params - the parameters\n * @param callback - the callback for completion or error\n */\n reloadTile(params: WorkerTileParameters, callback: WorkerTileCallback) {\n const loaded = this.loaded,\n uid = params.uid;\n\n if (loaded && loaded[uid]) {\n return super.reloadTile(params, callback);\n } else {\n return this.loadTile(params, callback);\n }\n }\n\n /**\n * Fetch and parse GeoJSON according to the given params. Calls `callback`\n * with `(err, data)`, where `data` is a parsed GeoJSON object.\n *\n * GeoJSON is loaded and parsed from `params.url` if it exists, or else\n * expected as a literal (string or object) `params.data`.\n *\n * @param params - the parameters\n * @param callback - the callback for completion or error\n * @returns A Cancelable object.\n */\n loadGeoJSON = (params: LoadGeoJSONParameters, callback: ResponseCallback): Cancelable => {\n const {promoteId} = params;\n // Because of same origin issues, urls must either include an explicit\n // origin or absolute path.\n // ie: /foo/bar.json or http://example.com/bar.json\n // but not ../foo/bar.json\n if (params.request) {\n return getJSON(params.request, (\n error?: Error,\n data?: any,\n cacheControl?: string,\n expires?: string\n ) => {\n this._dataUpdateable = isUpdateableGeoJSON(data, promoteId) ? toUpdateable(data, promoteId) : undefined;\n callback(error, data, cacheControl, expires);\n });\n } else if (typeof params.data === 'string') {\n try {\n const parsed = JSON.parse(params.data);\n this._dataUpdateable = isUpdateableGeoJSON(parsed, promoteId) ? toUpdateable(parsed, promoteId) : undefined;\n callback(null, parsed);\n } catch (e) {\n callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`));\n }\n } else if (params.dataDiff) {\n if (this._dataUpdateable) {\n applySourceDiff(this._dataUpdateable, params.dataDiff, promoteId);\n callback(null, {type: 'FeatureCollection', features: Array.from(this._dataUpdateable.values())});\n } else {\n callback(new Error(`Cannot update existing geojson data in ${params.source}`));\n }\n } else {\n callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`));\n }\n\n return {cancel: () => {}};\n };\n\n removeSource(params: {\n source: string;\n }, callback: WorkerTileCallback) {\n if (this._pendingCallback) {\n // Don't leak callbacks\n this._pendingCallback(null, {abandoned: true});\n }\n callback();\n }\n\n getClusterExpansionZoom(params: {\n clusterId: number;\n }, callback: Callback) {\n try {\n callback(null, (this._geoJSONIndex as Supercluster).getClusterExpansionZoom(params.clusterId));\n } catch (e) {\n callback(e);\n }\n }\n\n getClusterChildren(params: {\n clusterId: number;\n }, callback: Callback>) {\n try {\n callback(null, (this._geoJSONIndex as Supercluster).getChildren(params.clusterId));\n } catch (e) {\n callback(e);\n }\n }\n\n getClusterLeaves(params: {\n clusterId: number;\n limit: number;\n offset: number;\n }, callback: Callback>) {\n try {\n callback(null, (this._geoJSONIndex as Supercluster).getLeaves(params.clusterId, params.limit, params.offset));\n } catch (e) {\n callback(e);\n }\n }\n}\n\nfunction getSuperclusterOptions({superclusterOptions, clusterProperties}: LoadGeoJSONParameters) {\n if (!clusterProperties || !superclusterOptions) return superclusterOptions;\n\n const mapExpressions = {};\n const reduceExpressions = {};\n const globals = {accumulated: null, zoom: 0};\n const feature = {properties: null};\n const propertyNames = Object.keys(clusterProperties);\n\n for (const key of propertyNames) {\n const [operator, mapExpression] = clusterProperties[key];\n\n const mapExpressionParsed = createExpression(mapExpression);\n const reduceExpressionParsed = createExpression(\n typeof operator === 'string' ? [operator, ['accumulated'], ['get', key]] : operator);\n\n mapExpressions[key] = mapExpressionParsed.value;\n reduceExpressions[key] = reduceExpressionParsed.value;\n }\n\n superclusterOptions.map = (pointProperties) => {\n feature.properties = pointProperties;\n const properties = {};\n for (const key of propertyNames) {\n properties[key] = mapExpressions[key].evaluate(globals, feature);\n }\n return properties;\n };\n superclusterOptions.reduce = (accumulated, clusterProperties) => {\n feature.properties = clusterProperties;\n for (const key of propertyNames) {\n globals.accumulated = accumulated[key];\n accumulated[key] = reduceExpressions[key].evaluate(globals, feature);\n }\n };\n\n return superclusterOptions;\n}\n","import {Actor, ActorTarget} from '../util/actor';\nimport {StyleLayerIndex} from '../style/style_layer_index';\nimport {VectorTileWorkerSource} from './vector_tile_worker_source';\nimport {RasterDEMTileWorkerSource} from './raster_dem_tile_worker_source';\nimport {GeoJSONWorkerSource} from './geojson_worker_source';\nimport {plugin as globalRTLTextPlugin} from './rtl_text_plugin';\nimport {isWorker} from '../util/util';\n\nimport type {\n WorkerSource,\n WorkerTileParameters,\n WorkerDEMTileParameters,\n WorkerTileCallback,\n WorkerDEMTileCallback,\n TileParameters\n} from '../source/worker_source';\n\nimport type {WorkerGlobalScopeInterface} from '../util/web_worker';\nimport type {Callback} from '../types/callback';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {PluginState} from './rtl_text_plugin';\n\n/**\n * The Worker class responsidble for background thread related execution\n */\nexport default class Worker {\n self: WorkerGlobalScopeInterface & ActorTarget;\n actor: Actor;\n layerIndexes: {[_: string]: StyleLayerIndex};\n availableImages: {[_: string]: Array};\n workerSourceTypes: {\n [_: string]: {\n new (...args: any): WorkerSource;\n };\n };\n workerSources: {\n [_: string]: {\n [_: string]: {\n [_: string]: WorkerSource;\n };\n };\n };\n demWorkerSources: {\n [_: string]: {\n [_: string]: RasterDEMTileWorkerSource;\n };\n };\n referrer: string;\n\n constructor(self: WorkerGlobalScopeInterface & ActorTarget) {\n this.self = self;\n this.actor = new Actor(self, this);\n\n this.layerIndexes = {};\n this.availableImages = {};\n\n this.workerSourceTypes = {\n vector: VectorTileWorkerSource,\n geojson: GeoJSONWorkerSource\n };\n\n // [mapId][sourceType][sourceName] => worker source instance\n this.workerSources = {};\n this.demWorkerSources = {};\n\n this.self.registerWorkerSource = (name: string, WorkerSource: {\n new (...args: any): WorkerSource;\n }) => {\n if (this.workerSourceTypes[name]) {\n throw new Error(`Worker source with name \"${name}\" already registered.`);\n }\n this.workerSourceTypes[name] = WorkerSource;\n };\n\n // This is invoked by the RTL text plugin when the download via the `importScripts` call has finished, and the code has been parsed.\n this.self.registerRTLTextPlugin = (rtlTextPlugin: {\n applyArabicShaping: Function;\n processBidirectionalText: ((b: string, a: Array) => Array);\n processStyledBidirectionalText?: ((c: string, b: Array, a: Array) => Array<[string, Array]>);\n }) => {\n if (globalRTLTextPlugin.isParsed()) {\n throw new Error('RTL text plugin already registered.');\n }\n globalRTLTextPlugin['applyArabicShaping'] = rtlTextPlugin.applyArabicShaping;\n globalRTLTextPlugin['processBidirectionalText'] = rtlTextPlugin.processBidirectionalText;\n globalRTLTextPlugin['processStyledBidirectionalText'] = rtlTextPlugin.processStyledBidirectionalText;\n };\n }\n\n setReferrer(mapID: string, referrer: string) {\n this.referrer = referrer;\n }\n\n setImages(mapId: string, images: Array, callback: WorkerTileCallback) {\n this.availableImages[mapId] = images;\n for (const workerSource in this.workerSources[mapId]) {\n const ws = this.workerSources[mapId][workerSource];\n for (const source in ws) {\n ws[source].availableImages = images;\n }\n }\n callback();\n }\n\n setLayers(mapId: string, layers: Array, callback: WorkerTileCallback) {\n this.getLayerIndex(mapId).replace(layers);\n callback();\n }\n\n updateLayers(mapId: string, params: {\n layers: Array;\n removedIds: Array;\n }, callback: WorkerTileCallback) {\n this.getLayerIndex(mapId).update(params.layers, params.removedIds);\n callback();\n }\n\n loadTile(mapId: string, params: WorkerTileParameters & {\n type: string;\n }, callback: WorkerTileCallback) {\n this.getWorkerSource(mapId, params.type, params.source).loadTile(params, callback);\n }\n\n loadDEMTile(mapId: string, params: WorkerDEMTileParameters, callback: WorkerDEMTileCallback) {\n this.getDEMWorkerSource(mapId, params.source).loadTile(params, callback);\n }\n\n reloadTile(mapId: string, params: WorkerTileParameters & {\n type: string;\n }, callback: WorkerTileCallback) {\n this.getWorkerSource(mapId, params.type, params.source).reloadTile(params, callback);\n }\n\n abortTile(mapId: string, params: TileParameters & {\n type: string;\n }, callback: WorkerTileCallback) {\n this.getWorkerSource(mapId, params.type, params.source).abortTile(params, callback);\n }\n\n removeTile(mapId: string, params: TileParameters & {\n type: string;\n }, callback: WorkerTileCallback) {\n this.getWorkerSource(mapId, params.type, params.source).removeTile(params, callback);\n }\n\n removeDEMTile(mapId: string, params: TileParameters) {\n this.getDEMWorkerSource(mapId, params.source).removeTile(params);\n }\n\n removeSource(mapId: string, params: {\n source: string;\n } & {\n type: string;\n }, callback: WorkerTileCallback) {\n\n if (!this.workerSources[mapId] ||\n !this.workerSources[mapId][params.type] ||\n !this.workerSources[mapId][params.type][params.source]) {\n return;\n }\n\n const worker = this.workerSources[mapId][params.type][params.source];\n delete this.workerSources[mapId][params.type][params.source];\n\n if (worker.removeSource !== undefined) {\n worker.removeSource(params, callback);\n } else {\n callback();\n }\n }\n\n /**\n * Load a {@link WorkerSource} script at params.url. The script is run\n * (using importScripts) with `registerWorkerSource` in scope, which is a\n * function taking `(name, workerSourceObject)`.\n */\n loadWorkerSource(map: string, params: {\n url: string;\n }, callback: Callback) {\n try {\n this.self.importScripts(params.url);\n callback();\n } catch (e) {\n callback(e.toString());\n }\n }\n\n syncRTLPluginState(map: string, state: PluginState, callback: Callback) {\n try {\n globalRTLTextPlugin.setState(state);\n const pluginURL = globalRTLTextPlugin.getPluginURL();\n if (\n globalRTLTextPlugin.isLoaded() &&\n !globalRTLTextPlugin.isParsed() &&\n pluginURL != null // Not possible when `isLoaded` is true, but keeps flow happy\n ) {\n this.self.importScripts(pluginURL);\n const complete = globalRTLTextPlugin.isParsed();\n const error = complete ? undefined : new Error(`RTL Text Plugin failed to import scripts from ${pluginURL}`);\n callback(error, complete);\n }\n } catch (e) {\n callback(e.toString());\n }\n }\n\n getAvailableImages(mapId: string) {\n let availableImages = this.availableImages[mapId];\n\n if (!availableImages) {\n availableImages = [];\n }\n\n return availableImages;\n }\n\n getLayerIndex(mapId: string) {\n let layerIndexes = this.layerIndexes[mapId];\n if (!layerIndexes) {\n layerIndexes = this.layerIndexes[mapId] = new StyleLayerIndex();\n }\n return layerIndexes;\n }\n\n getWorkerSource(mapId: string, sourceType: string, sourceName: string): WorkerSource {\n if (!this.workerSources[mapId])\n this.workerSources[mapId] = {};\n if (!this.workerSources[mapId][sourceType])\n this.workerSources[mapId][sourceType] = {};\n\n if (!this.workerSources[mapId][sourceType][sourceName]) {\n // use a wrapped actor so that we can attach a target mapId param\n // to any messages invoked by the WorkerSource\n const actor = {\n send: (type, data, callback) => {\n this.actor.send(type, data, callback, mapId);\n }\n };\n this.workerSources[mapId][sourceType][sourceName] = new (this.workerSourceTypes[sourceType] as any)((actor as any), this.getLayerIndex(mapId), this.getAvailableImages(mapId));\n }\n\n return this.workerSources[mapId][sourceType][sourceName];\n }\n\n getDEMWorkerSource(mapId: string, source: string) {\n if (!this.demWorkerSources[mapId])\n this.demWorkerSources[mapId] = {};\n\n if (!this.demWorkerSources[mapId][source]) {\n this.demWorkerSources[mapId][source] = new RasterDEMTileWorkerSource();\n }\n\n return this.demWorkerSources[mapId][source];\n }\n}\n\nif (isWorker()) {\n (self as any).worker = new Worker(self as any);\n}\n","import Point from '@mapbox/point-geometry';\n\nexport class DOM {\n private static readonly docStyle = typeof window !== 'undefined' && window.document && window.document.documentElement.style;\n\n private static userSelect: string;\n\n private static selectProp = DOM.testProp(['userSelect', 'MozUserSelect', 'WebkitUserSelect', 'msUserSelect']);\n\n private static transformProp = DOM.testProp(['transform', 'WebkitTransform']);\n\n private static testProp(props: string[]): string {\n if (!DOM.docStyle) return props[0];\n for (let i = 0; i < props.length; i++) {\n if (props[i] in DOM.docStyle) {\n return props[i];\n }\n }\n return props[0];\n }\n\n public static create(tagName: K, className?: string, container?: HTMLElement): HTMLElementTagNameMap[K] {\n const el = window.document.createElement(tagName);\n if (className !== undefined) el.className = className;\n if (container) container.appendChild(el);\n return el;\n }\n\n public static createNS(namespaceURI: string, tagName: string) {\n const el = window.document.createElementNS(namespaceURI, tagName);\n return el;\n }\n\n public static disableDrag() {\n if (DOM.docStyle && DOM.selectProp) {\n DOM.userSelect = DOM.docStyle[DOM.selectProp];\n DOM.docStyle[DOM.selectProp] = 'none';\n }\n }\n\n public static enableDrag() {\n if (DOM.docStyle && DOM.selectProp) {\n DOM.docStyle[DOM.selectProp] = DOM.userSelect;\n }\n }\n\n public static setTransform(el: HTMLElement, value: string) {\n el.style[DOM.transformProp] = value;\n }\n\n public static addEventListener(target: HTMLElement | Window | Document, type: string, callback: EventListenerOrEventListenerObject, options: {\n passive?: boolean;\n capture?: boolean;\n } = {}) {\n if ('passive' in options) {\n target.addEventListener(type, callback, options);\n } else {\n target.addEventListener(type, callback, options.capture);\n }\n }\n\n public static removeEventListener(target: HTMLElement | Window | Document, type: string, callback: EventListenerOrEventListenerObject, options: {\n passive?: boolean;\n capture?: boolean;\n } = {}) {\n if ('passive' in options) {\n target.removeEventListener(type, callback, options);\n } else {\n target.removeEventListener(type, callback, options.capture);\n }\n }\n\n // Suppress the next click, but only if it's immediate.\n private static suppressClickInternal(e) {\n e.preventDefault();\n e.stopPropagation();\n window.removeEventListener('click', DOM.suppressClickInternal, true);\n }\n\n public static suppressClick() {\n window.addEventListener('click', DOM.suppressClickInternal, true);\n window.setTimeout(() => {\n window.removeEventListener('click', DOM.suppressClickInternal, true);\n }, 0);\n }\n\n public static mousePos(el: HTMLElement, e: MouseEvent | Touch) {\n const rect = el.getBoundingClientRect();\n return new Point(\n e.clientX - rect.left - el.clientLeft,\n e.clientY - rect.top - el.clientTop\n );\n }\n\n public static touchPos(el: HTMLElement, touches: TouchList) {\n const rect = el.getBoundingClientRect();\n const points: Point[] = [];\n for (let i = 0; i < touches.length; i++) {\n points.push(new Point(\n touches[i].clientX - rect.left - el.clientLeft,\n touches[i].clientY - rect.top - el.clientTop\n ));\n }\n return points;\n }\n\n public static mouseButton(e: MouseEvent) {\n return e.button;\n }\n\n public static remove(node: HTMLElement) {\n if (node.parentNode) {\n node.parentNode.removeChild(node);\n }\n }\n}\n","export const webpSupported = {\n supported: false,\n testSupport\n};\n\nlet glForTesting: WebGLRenderingContext|WebGL2RenderingContext;\nlet webpCheckComplete = false;\nlet webpImgTest;\nlet webpImgTestOnloadComplete = false;\n\nif (typeof document !== 'undefined') {\n webpImgTest = document.createElement('img');\n webpImgTest.onload = function() {\n if (glForTesting) testWebpTextureUpload(glForTesting);\n glForTesting = null;\n webpImgTestOnloadComplete = true;\n };\n webpImgTest.onerror = function() {\n webpCheckComplete = true;\n glForTesting = null;\n };\n webpImgTest.src = '';\n}\n\nfunction testSupport(gl: WebGLRenderingContext | WebGL2RenderingContext) {\n if (webpCheckComplete || !webpImgTest) return;\n\n // HTMLImageElement.complete is set when an image is done loading it's source\n // regardless of whether the load was successful or not.\n // It's possible for an error to set HTMLImageElement.complete to true which would trigger\n // testWebpTextureUpload and mistakenly set exported.supported to true in browsers which don't support webp\n // To avoid this, we set a flag in the image's onload handler and only call testWebpTextureUpload\n // after a successful image load event.\n if (webpImgTestOnloadComplete) {\n testWebpTextureUpload(gl);\n } else {\n glForTesting = gl;\n\n }\n}\n\nfunction testWebpTextureUpload(gl: WebGLRenderingContext|WebGL2RenderingContext) {\n // Edge 18 supports WebP but not uploading a WebP image to a gl texture\n // Test support for this before allowing WebP images.\n // https://github.com/mapbox/mapbox-gl-js/issues/7671\n const texture = gl.createTexture();\n gl.bindTexture(gl.TEXTURE_2D, texture);\n\n try {\n gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, webpImgTest);\n\n // The error does not get triggered in Edge if the context is lost\n if (gl.isContextLost()) return;\n\n webpSupported.supported = true;\n } catch (e) {\n // Catch \"Unspecified Error.\" in Edge 18.\n }\n\n gl.deleteTexture(texture);\n\n webpCheckComplete = true;\n}\n","import type {Cancelable} from '../types/cancelable';\nimport {RequestParameters, ExpiryData, makeRequest, sameOrigin, getProtocolAction} from './ajax';\nimport type {Callback} from '../types/callback';\n\nimport {arrayBufferToImageBitmap, arrayBufferToImage, extend, isWorker, isImageBitmap} from './util';\nimport {webpSupported} from './webp_supported';\nimport {config} from './config';\n\n/**\n * The callback that is being called after an image was fetched\n */\nexport type GetImageCallback = (error?: Error | null, image?: HTMLImageElement | ImageBitmap | null, expiry?: ExpiryData | null) => void;\n\ntype ImageQueueThrottleControlCallback = () => boolean;\n\nexport type ImageRequestQueueItem = Cancelable & {\n requestParameters: RequestParameters;\n supportImageRefresh: boolean;\n callback: GetImageCallback;\n cancelled: boolean;\n completed: boolean;\n innerRequest?: Cancelable;\n}\n\ntype ImageQueueThrottleCallbackDictionary = {\n [Key: number]: ImageQueueThrottleControlCallback;\n}\n\ntype HTMLImageElementWithPriority = HTMLImageElement &\n{\n // fetchPriority is experimental property supported on Chromium browsers from Version 102\n // By default images are downloaded with priority low, whereas fetch request downloads with priority high\n // https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/fetchPriority\n fetchPriority?: 'auto' | 'high' | 'low';\n};\n\n/**\n * By default, the image queue is self driven, meaning as soon as one requested item is processed,\n * it will move on to next one as quickly as it can while limiting\n * the number of concurrent requests to MAX_PARALLEL_IMAGE_REQUESTS. The default behavior\n * ensures that static views of the map can be rendered with minimal delay.\n *\n * However, the default behavior can prevent dynamic views of the map from rendering\n * smoothly in that many requests can finish in one render frame, putting too much pressure on GPU.\n *\n * When the view of the map is moving dynamically, smoother frame rates can be achieved\n * by throttling the number of items processed by the queue per frame. This can be\n * accomplished by using {@link addThrottleControl} to allow the caller to\n * use a lambda function to determine when the queue should be throttled (e.g. when isMoving())\n * and manually calling {@link processQueue} in the render loop.\n */\nexport namespace ImageRequest {\n let imageRequestQueue : ImageRequestQueueItem[];\n let currentParallelImageRequests:number;\n\n let throttleControlCallbackHandleCounter: number;\n let throttleControlCallbacks: ImageQueueThrottleCallbackDictionary;\n\n /**\n * Reset the image request queue, removing all pending requests.\n */\n export const resetRequestQueue = (): void => {\n imageRequestQueue = [];\n currentParallelImageRequests = 0;\n throttleControlCallbackHandleCounter = 0;\n throttleControlCallbacks = {};\n };\n\n /**\n * Install a callback to control when image queue throttling is desired.\n * (e.g. when the map view is moving)\n * @param callback - The callback function to install\n * @returns handle that identifies the installed callback.\n */\n export const addThrottleControl = (callback: ImageQueueThrottleControlCallback): number => {\n const handle = throttleControlCallbackHandleCounter++;\n throttleControlCallbacks[handle] = callback;\n return handle;\n };\n\n /**\n * Remove a previously installed callback by passing in the handle returned\n * by {@link addThrottleControl}.\n * @param callbackHandle - The handle for the callback to remove.\n */\n export const removeThrottleControl = (callbackHandle: number): void => {\n delete throttleControlCallbacks[callbackHandle];\n // Try updating the queue\n processQueue();\n };\n\n /**\n * Check to see if any of the installed callbacks are requesting the queue\n * to be throttled.\n * @returns `true` if any callback is causing the queue to be throttled.\n */\n const isThrottled = (): boolean => {\n const allControlKeys = Object.keys(throttleControlCallbacks);\n let throttleingRequested = false;\n if (allControlKeys.length > 0) {\n for (const key of allControlKeys) {\n throttleingRequested = throttleControlCallbacks[key]();\n if (throttleingRequested) {\n break;\n }\n }\n }\n return throttleingRequested;\n };\n\n /**\n * Request to load an image.\n * @param requestParameters - Request parameters.\n * @param callback - Callback to issue when the request completes.\n * @param supportImageRefresh - `true`, if the image request need to support refresh based on cache headers.\n * @returns Cancelable request.\n */\n export const getImage = (\n requestParameters: RequestParameters,\n callback: GetImageCallback,\n supportImageRefresh: boolean = true\n ): ImageRequestQueueItem => {\n if (webpSupported.supported) {\n if (!requestParameters.headers) {\n requestParameters.headers = {};\n }\n requestParameters.headers.accept = 'image/webp,*/*';\n }\n\n const request:ImageRequestQueueItem = {\n requestParameters,\n supportImageRefresh,\n callback,\n cancelled: false,\n completed: false,\n cancel: () => {\n if (!request.completed && !request.cancelled) {\n request.cancelled = true;\n\n // Only reduce currentParallelImageRequests, if the image request was issued.\n if (request.innerRequest) {\n request.innerRequest.cancel();\n currentParallelImageRequests--;\n }\n\n // in the case of cancelling, it WILL move on\n processQueue();\n }\n }\n };\n\n imageRequestQueue.push(request);\n processQueue();\n return request;\n };\n\n const arrayBufferToCanvasImageSource = (data: ArrayBuffer, callback: Callback) => {\n const imageBitmapSupported = typeof createImageBitmap === 'function';\n if (imageBitmapSupported) {\n arrayBufferToImageBitmap(data, callback);\n } else {\n arrayBufferToImage(data, callback);\n }\n };\n\n const doImageRequest = (itemInQueue: ImageRequestQueueItem): Cancelable => {\n const {requestParameters, supportImageRefresh, callback} = itemInQueue;\n extend(requestParameters, {type: 'image'});\n\n // - If refreshExpiredTiles is false, then we can use HTMLImageElement to download raster images.\n // - Fetch/XHR (via MakeRequest API) will be used to download images for following scenarios:\n // 1. Style image sprite will had a issue with HTMLImageElement as described\n // here: https://github.com/mapbox/mapbox-gl-js/issues/1470\n // 2. If refreshExpiredTiles is true (default), then in order to read the image cache header,\n // fetch/XHR request will be required\n // - For any special case handling like use of AddProtocol, worker initiated request or additional headers\n // let makeRequest handle it.\n // - HtmlImageElement request automatically adds accept header for all the browser supported images\n const canUseHTMLImageElement = supportImageRefresh === false &&\n !isWorker() &&\n !getProtocolAction(requestParameters.url) &&\n (!requestParameters.headers ||\n Object.keys(requestParameters.headers).reduce((acc, item) => acc && item === 'accept', true));\n\n const action = canUseHTMLImageElement ? getImageUsingHtmlImage : makeRequest;\n return action(\n requestParameters,\n (err?: Error | null,\n data?: HTMLImageElement | ImageBitmap | ArrayBuffer | null,\n cacheControl?: string | null,\n expires?: string | null) => {\n onImageResponse(itemInQueue, callback, err, data, cacheControl, expires);\n });\n };\n\n const onImageResponse = (\n itemInQueue: ImageRequestQueueItem,\n callback:GetImageCallback,\n err?: Error | null,\n data?: HTMLImageElement | ImageBitmap | ArrayBuffer | null,\n cacheControl?: string | null,\n expires?: string | null): void => {\n if (err) {\n callback(err);\n } else if (data instanceof HTMLImageElement || isImageBitmap(data)) {\n // User using addProtocol can directly return HTMLImageElement/ImageBitmap type\n // If HtmlImageElement is used to get image then response type will be HTMLImageElement\n callback(null, data);\n } else if (data) {\n const decoratedCallback = (imgErr?: Error | null, imgResult?: CanvasImageSource | null) => {\n if (imgErr != null) {\n callback(imgErr);\n } else if (imgResult != null) {\n callback(null, imgResult as (HTMLImageElement | ImageBitmap), {cacheControl, expires});\n }\n };\n arrayBufferToCanvasImageSource(data, decoratedCallback);\n }\n if (!itemInQueue.cancelled) {\n itemInQueue.completed = true;\n currentParallelImageRequests--;\n\n processQueue();\n }\n };\n\n /**\n * Process some number of items in the image request queue.\n */\n const processQueue = (): void => {\n\n const maxImageRequests = isThrottled() ?\n config.MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME :\n config.MAX_PARALLEL_IMAGE_REQUESTS;\n\n // limit concurrent image loads to help with raster sources performance on big screens\n for (let numImageRequests = currentParallelImageRequests;\n numImageRequests < maxImageRequests && imageRequestQueue.length > 0;\n numImageRequests++) {\n\n const topItemInQueue: ImageRequestQueueItem = imageRequestQueue.shift();\n if (topItemInQueue.cancelled) {\n numImageRequests--;\n continue;\n }\n\n const innerRequest = doImageRequest(topItemInQueue);\n\n currentParallelImageRequests++;\n\n topItemInQueue.innerRequest = innerRequest;\n }\n };\n\n const getImageUsingHtmlImage = (requestParameters: RequestParameters, callback: GetImageCallback): Cancelable => {\n const image = new Image() as HTMLImageElementWithPriority;\n const url = requestParameters.url;\n let requestCancelled = false;\n const credentials = requestParameters.credentials;\n if (credentials && credentials === 'include') {\n image.crossOrigin = 'use-credentials';\n } else if ((credentials && credentials === 'same-origin') || !sameOrigin(url)) {\n image.crossOrigin = 'anonymous';\n }\n\n image.fetchPriority = 'high';\n image.onload = () => {\n callback(null, image);\n image.onerror = image.onload = null;\n };\n image.onerror = () => {\n if (!requestCancelled) {\n callback(new Error('Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.'));\n }\n image.onerror = image.onload = null;\n };\n image.src = url;\n return {\n cancel: () => {\n requestCancelled = true;\n // Set src to '' to actually cancel the request\n image.src = '';\n }\n };\n };\n}\n\nImageRequest.resetRequestQueue();\n","import type {RequestParameters} from './ajax';\n\n/**\n * A type of MapLibre resource.\n */\nexport const enum ResourceType {\n Glyphs = 'Glyphs',\n Image = 'Image',\n Source = 'Source',\n SpriteImage = 'SpriteImage',\n SpriteJSON = 'SpriteJSON',\n Style = 'Style',\n Tile = 'Tile',\n Unknown = 'Unknown',\n}\n\n/**\n * This function is used to tranform a request.\n * It is used just before executing the relevant request.\n */\nexport type RequestTransformFunction = (url: string, resourceType?: ResourceType) => RequestParameters | undefined;\n\ntype UrlObject = {\n protocol: string;\n authority: string;\n path: string;\n params: Array;\n};\n\nexport class RequestManager {\n _transformRequestFn: RequestTransformFunction;\n\n constructor(transformRequestFn?: RequestTransformFunction) {\n this._transformRequestFn = transformRequestFn;\n }\n\n transformRequest(url: string, type: ResourceType) {\n if (this._transformRequestFn) {\n return this._transformRequestFn(url, type) || {url};\n }\n\n return {url};\n }\n\n normalizeSpriteURL(url: string, format: string, extension: string): string {\n const urlObject = parseUrl(url);\n urlObject.path += `${format}${extension}`;\n return formatUrl(urlObject);\n }\n\n setTransformRequest(transformRequest: RequestTransformFunction) {\n this._transformRequestFn = transformRequest;\n }\n}\n\nconst urlRe = /^(\\w+):\\/\\/([^/?]*)(\\/[^?]+)?\\??(.+)?/;\n\nfunction parseUrl(url: string): UrlObject {\n const parts = url.match(urlRe);\n if (!parts) {\n throw new Error(`Unable to parse URL \"${url}\"`);\n }\n return {\n protocol: parts[1],\n authority: parts[2],\n path: parts[3] || '/',\n params: parts[4] ? parts[4].split('&') : []\n };\n}\n\nfunction formatUrl(obj: UrlObject): string {\n const params = obj.params.length ? `?${obj.params.join('&')}` : '';\n return `${obj.protocol}://${obj.authority}${obj.path}${params}`;\n}\n","import {SpriteSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * Takes a SpriteSpecification value and returns it in its array form. If `undefined` is passed as an input value, an\n * empty array is returned.\n * duplicated entries with identical id/url will be removed in returned array\n * @param sprite - optional sprite to coerce\n * @returns an empty array in case `undefined` is passed; id-url pairs otherwise\n */\nexport function coerceSpriteToArray(sprite?: SpriteSpecification): {id: string; url: string}[] {\n const resultArray: {id: string; url: string}[] = [];\n\n if (typeof sprite === 'string') {\n resultArray.push({id: 'default', url: sprite});\n } else if (sprite && sprite.length > 0) {\n const dedupArray: string[] = [];\n for (const {id, url} of sprite) {\n const key = `${id}${url}`;\n if (dedupArray.indexOf(key) === -1) {\n dedupArray.push(key);\n resultArray.push({id, url});\n }\n }\n }\n\n return resultArray;\n\n}\n","import {getJSON} from '../util/ajax';\nimport {ImageRequest} from '../util/image_request';\nimport {ResourceType} from '../util/request_manager';\n\nimport {browser} from '../util/browser';\nimport {coerceSpriteToArray} from '../util/style';\n\nimport type {SpriteSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {StyleImage} from './style_image';\nimport type {RequestManager} from '../util/request_manager';\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\n\nexport function loadSprite(\n originalSprite: SpriteSpecification,\n requestManager: RequestManager,\n pixelRatio: number,\n callback: Callback<{[spriteName: string]: {[id: string]: StyleImage}}>\n): Cancelable {\n const spriteArray = coerceSpriteToArray(originalSprite);\n const spriteArrayLength = spriteArray.length;\n const format = pixelRatio > 1 ? '@2x' : '';\n\n const combinedRequestsMap: {[requestKey: string]: Cancelable} = {};\n const jsonsMap: {[id: string]: any} = {};\n const imagesMap: {[id: string]: (HTMLImageElement | ImageBitmap)} = {};\n\n for (const {id, url} of spriteArray) {\n const jsonRequestParameters = requestManager.transformRequest(requestManager.normalizeSpriteURL(url, format, '.json'), ResourceType.SpriteJSON);\n const jsonRequestKey = `${id}_${jsonRequestParameters.url}`; // use id_url as requestMap key to make sure it is unique\n combinedRequestsMap[jsonRequestKey] = getJSON(jsonRequestParameters, (err?: Error | null, data?: any | null) => {\n delete combinedRequestsMap[jsonRequestKey];\n jsonsMap[id] = data;\n doOnceCompleted(callback, jsonsMap, imagesMap, err, spriteArrayLength);\n });\n\n const imageRequestParameters = requestManager.transformRequest(requestManager.normalizeSpriteURL(url, format, '.png'), ResourceType.SpriteImage);\n const imageRequestKey = `${id}_${imageRequestParameters.url}`; // use id_url as requestMap key to make sure it is unique\n combinedRequestsMap[imageRequestKey] = ImageRequest.getImage(imageRequestParameters, (err, img) => {\n delete combinedRequestsMap[imageRequestKey];\n imagesMap[id] = img;\n doOnceCompleted(callback, jsonsMap, imagesMap, err, spriteArrayLength);\n });\n }\n\n return {\n cancel() {\n for (const requst of Object.values(combinedRequestsMap)) {\n requst.cancel();\n }\n }\n };\n}\n\n/**\n * @param callbackFunc - the callback function (both erro and success)\n * @param jsonsMap - JSON data map\n * @param imagesMap - image data map\n * @param err - error object\n * @param expectedResultCounter - number of expected JSON or Image results when everything is finished, respectively.\n */\nfunction doOnceCompleted(\n callbackFunc:Callback<{[spriteName: string]: {[id: string]: StyleImage}}>,\n jsonsMap:{[id: string]: any},\n imagesMap:{[id: string]: (HTMLImageElement | ImageBitmap)},\n err: Error,\n expectedResultCounter: number): void {\n\n if (err) {\n callbackFunc(err);\n return;\n }\n\n if (expectedResultCounter !== Object.values(jsonsMap).length || expectedResultCounter !== Object.values(imagesMap).length) {\n // not done yet, nothing to do\n return;\n }\n\n const result = {} as {[spriteName: string]: {[id: string]: StyleImage}};\n for (const spriteName in jsonsMap) {\n result[spriteName] = {};\n\n const context = browser.getImageCanvasContext(imagesMap[spriteName]);\n const json = jsonsMap[spriteName];\n\n for (const id in json) {\n const {width, height, x, y, sdf, pixelRatio, stretchX, stretchY, content} = json[id];\n const spriteData = {width, height, x, y, context};\n result[spriteName][id] = {data: null, pixelRatio, sdf, stretchX, stretchY, content, spriteData};\n }\n }\n\n callbackFunc(null, result);\n}\n","import type {Context} from '../gl/context';\nimport type {RGBAImage, AlphaImage} from '../util/image';\nimport {isImageBitmap} from '../util/util';\n\nexport type TextureFormat = WebGLRenderingContextBase['RGBA'] | WebGLRenderingContextBase['ALPHA'];\nexport type TextureFilter = WebGLRenderingContextBase['LINEAR'] | WebGLRenderingContextBase['LINEAR_MIPMAP_NEAREST'] | WebGLRenderingContextBase['NEAREST'];\nexport type TextureWrap = WebGLRenderingContextBase['REPEAT'] | WebGLRenderingContextBase['CLAMP_TO_EDGE'] | WebGLRenderingContextBase['MIRRORED_REPEAT'];\n\ntype EmptyImage = {\n width: number;\n height: number;\n data: null;\n};\n\ntype DataTextureImage = RGBAImage | AlphaImage | EmptyImage;\nexport type TextureImage = TexImageSource | DataTextureImage;\n\n/**\n * @internal\n * A `Texture` GL related object\n */\nexport class Texture {\n context: Context;\n size: [number, number];\n texture: WebGLTexture;\n format: TextureFormat;\n filter: TextureFilter;\n wrap: TextureWrap;\n useMipmap: boolean;\n\n constructor(context: Context, image: TextureImage, format: TextureFormat, options?: {\n premultiply?: boolean;\n useMipmap?: boolean;\n } | null) {\n this.context = context;\n this.format = format;\n this.texture = context.gl.createTexture();\n this.update(image, options);\n }\n\n update(image: TextureImage, options?: {\n premultiply?: boolean;\n useMipmap?: boolean;\n } | null, position?: {\n x: number;\n y: number;\n }) {\n const {width, height} = image as {width: number; height: number};\n const resize = (!this.size || this.size[0] !== width || this.size[1] !== height) && !position;\n const {context} = this;\n const {gl} = context;\n\n this.useMipmap = Boolean(options && options.useMipmap);\n gl.bindTexture(gl.TEXTURE_2D, this.texture);\n\n context.pixelStoreUnpackFlipY.set(false);\n context.pixelStoreUnpack.set(1);\n context.pixelStoreUnpackPremultiplyAlpha.set(this.format === gl.RGBA && (!options || options.premultiply !== false));\n\n if (resize) {\n this.size = [width, height];\n\n if (image instanceof HTMLImageElement || image instanceof HTMLCanvasElement || image instanceof HTMLVideoElement || image instanceof ImageData || isImageBitmap(image)) {\n gl.texImage2D(gl.TEXTURE_2D, 0, this.format, this.format, gl.UNSIGNED_BYTE, image);\n } else {\n gl.texImage2D(gl.TEXTURE_2D, 0, this.format, width, height, 0, this.format, gl.UNSIGNED_BYTE, (image as DataTextureImage).data);\n }\n\n } else {\n const {x, y} = position || {x: 0, y: 0};\n if (image instanceof HTMLImageElement || image instanceof HTMLCanvasElement || image instanceof HTMLVideoElement || image instanceof ImageData || isImageBitmap(image)) {\n gl.texSubImage2D(gl.TEXTURE_2D, 0, x, y, gl.RGBA, gl.UNSIGNED_BYTE, image);\n } else {\n gl.texSubImage2D(gl.TEXTURE_2D, 0, x, y, width, height, gl.RGBA, gl.UNSIGNED_BYTE, (image as DataTextureImage).data);\n }\n }\n\n if (this.useMipmap && this.isSizePowerOfTwo()) {\n gl.generateMipmap(gl.TEXTURE_2D);\n }\n }\n\n bind(filter: TextureFilter, wrap: TextureWrap, minFilter?: TextureFilter | null) {\n const {context} = this;\n const {gl} = context;\n gl.bindTexture(gl.TEXTURE_2D, this.texture);\n\n if (minFilter === gl.LINEAR_MIPMAP_NEAREST && !this.isSizePowerOfTwo()) {\n minFilter = gl.LINEAR;\n }\n\n if (filter !== this.filter) {\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, filter);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, minFilter || filter);\n this.filter = filter;\n }\n\n if (wrap !== this.wrap) {\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, wrap);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, wrap);\n this.wrap = wrap;\n }\n }\n\n isSizePowerOfTwo() {\n return this.size[0] === this.size[1] && (Math.log(this.size[0]) / Math.LN2) % 1 === 0;\n }\n\n destroy() {\n const {gl} = this.context;\n gl.deleteTexture(this.texture);\n this.texture = null;\n }\n}\n","import {RGBAImage} from '../util/image';\n\nimport type {Map} from '../ui/map';\n\n/**\n * The sprite data\n */\nexport type SpriteOnDemandStyleImage = {\n width: number;\n height: number;\n x: number;\n y: number;\n context: CanvasRenderingContext2D;\n};\n\n/**\n * The style's image metadata\n */\nexport type StyleImageData = {\n data: RGBAImage;\n version?: number;\n hasRenderCallback?: boolean;\n userImage?: StyleImageInterface;\n spriteData?: SpriteOnDemandStyleImage;\n};\n\n/**\n * The style's image metadata\n */\nexport type StyleImageMetadata = {\n /**\n * The ratio of pixels in the image to physical pixels on the screen\n */\n pixelRatio: number;\n /**\n * Whether the image should be interpreted as an SDF image\n */\n sdf: boolean;\n /**\n * If `icon-text-fit` is used in a layer with this image, this option defines the part(s) of the image that can be stretched horizontally.\n */\n stretchX?: Array<[number, number]>;\n /**\n * If `icon-text-fit` is used in a layer with this image, this option defines the part(s) of the image that can be stretched vertically.\n */\n stretchY?: Array<[number, number]>;\n /**\n * If `icon-text-fit` is used in a layer with this image, this option defines the part of the image that can be covered by the content in `text-field`.\n */\n content?: [number, number, number, number];\n};\n\n/**\n * the style's image, including data and metedata\n */\nexport type StyleImage = StyleImageData & StyleImageMetadata;\n\n/**\n * Interface for dynamically generated style images. This is a specification for\n * implementers to model: it is not an exported method or class.\n *\n * Images implementing this interface can be redrawn for every frame. They can be used to animate\n * icons and patterns or make them respond to user input. Style images can implement a\n * {@link StyleImageInterface#render} method. The method is called every frame and\n * can be used to update the image.\n *\n * @see [Add an animated icon to the map.](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-animated/)\n *\n * @example\n * ```ts\n * let flashingSquare = {\n * width: 64,\n * height: 64,\n * data: new Uint8Array(64 * 64 * 4),\n *\n * onAdd: function(map) {\n * this.map = map;\n * },\n *\n * render: function() {\n * // keep repainting while the icon is on the map\n * this.map.triggerRepaint();\n *\n * // alternate between black and white based on the time\n * let value = Math.round(Date.now() / 1000) % 2 === 0 ? 255 : 0;\n *\n * // check if image needs to be changed\n * if (value !== this.previousValue) {\n * this.previousValue = value;\n *\n * let bytesPerPixel = 4;\n * for (let x = 0; x < this.width; x++) {\n * for (let y = 0; y < this.height; y++) {\n * let offset = (y * this.width + x) * bytesPerPixel;\n * this.data[offset + 0] = value;\n * this.data[offset + 1] = value;\n * this.data[offset + 2] = value;\n * this.data[offset + 3] = 255;\n * }\n * }\n *\n * // return true to indicate that the image changed\n * return true;\n * }\n * }\n * }\n *\n * map.addImage('flashing_square', flashingSquare);\n * ```\n */\n\nexport interface StyleImageInterface {\n width: number;\n height: number;\n data: Uint8Array | Uint8ClampedArray;\n /**\n * This method is called once before every frame where the icon will be used.\n * The method can optionally update the image's `data` member with a new image.\n *\n * If the method updates the image it must return `true` to commit the change.\n * If the method returns `false` or nothing the image is assumed to not have changed.\n *\n * If updates are infrequent it maybe easier to use {@link Map#updateImage} to update\n * the image instead of implementing this method.\n *\n * @returns `true` if this method updated the image. `false` if the image was not changed.\n */\n render?: () => boolean;\n /**\n * Optional method called when the layer has been added to the Map with {@link Map#addImage}.\n *\n * @param map - The Map this custom layer was just added to.\n */\n onAdd?: (map: Map, id: string) => void;\n /**\n * Optional method called when the icon is removed from the map with {@link Map#removeImage}.\n * This gives the image a chance to clean up resources and event listeners.\n */\n onRemove?: () => void;\n}\n\nexport function renderStyleImage(image: StyleImage) {\n const {userImage} = image;\n if (userImage && userImage.render) {\n const updated = userImage.render();\n if (updated) {\n image.data.replace(new Uint8Array(userImage.data.buffer));\n return true;\n }\n }\n return false;\n}\n","/* eslint-disable key-spacing */\nimport potpack from 'potpack';\n\nimport {Event, ErrorEvent, Evented} from '../util/evented';\nimport {RGBAImage} from '../util/image';\nimport {ImagePosition} from './image_atlas';\nimport {Texture} from './texture';\nimport {renderStyleImage} from '../style/style_image';\nimport {warnOnce} from '../util/util';\n\nimport type {StyleImage} from '../style/style_image';\nimport type {Context} from '../gl/context';\nimport type {PotpackBox} from 'potpack';\nimport type {Callback} from '../types/callback';\n\ntype Pattern = {\n bin: PotpackBox;\n position: ImagePosition;\n};\n\n// When copied into the atlas texture, image data is padded by one pixel on each side. Icon\n// images are padded with fully transparent pixels, while pattern images are padded with a\n// copy of the image data wrapped from the opposite side. In both cases, this ensures the\n// correct behavior of GL_LINEAR texture sampling mode.\nconst padding = 1;\n\n/*\n ImageManager does three things:\n\n 1. Tracks requests for icon images from tile workers and sends responses when the requests are fulfilled.\n 2. Builds a texture atlas for pattern images.\n 3. Rerenders renderable images once per frame\n\n These are disparate responsibilities and should eventually be handled by different classes. When we implement\n data-driven support for `*-pattern`, we'll likely use per-bucket pattern atlases, and that would be a good time\n to refactor this.\n*/\nexport class ImageManager extends Evented {\n images: {[_: string]: StyleImage};\n updatedImages: {[_: string]: boolean};\n callbackDispatchedThisFrame: {[_: string]: boolean};\n loaded: boolean;\n requestors: Array<{\n ids: Array;\n callback: Callback<{[_: string]: StyleImage}>;\n }>;\n\n patterns: {[_: string]: Pattern};\n atlasImage: RGBAImage;\n atlasTexture: Texture;\n dirty: boolean;\n\n constructor() {\n super();\n this.images = {};\n this.updatedImages = {};\n this.callbackDispatchedThisFrame = {};\n this.loaded = false;\n this.requestors = [];\n\n this.patterns = {};\n this.atlasImage = new RGBAImage({width: 1, height: 1});\n this.dirty = true;\n }\n\n isLoaded() {\n return this.loaded;\n }\n\n setLoaded(loaded: boolean) {\n if (this.loaded === loaded) {\n return;\n }\n\n this.loaded = loaded;\n\n if (loaded) {\n for (const {ids, callback} of this.requestors) {\n this._notify(ids, callback);\n }\n this.requestors = [];\n }\n }\n\n getImage(id: string): StyleImage {\n const image = this.images[id];\n\n // Extract sprite image data on demand\n if (image && !image.data && image.spriteData) {\n const spriteData = image.spriteData;\n image.data = new RGBAImage({\n width: spriteData.width,\n height: spriteData.height\n }, spriteData.context.getImageData(\n spriteData.x,\n spriteData.y,\n spriteData.width,\n spriteData.height).data);\n image.spriteData = null;\n }\n\n return image;\n }\n\n addImage(id: string, image: StyleImage) {\n if (this.images[id]) throw new Error(`Image id ${id} already exist, use updateImage instead`);\n if (this._validate(id, image)) {\n this.images[id] = image;\n }\n }\n\n _validate(id: string, image: StyleImage) {\n let valid = true;\n const data = image.data || image.spriteData;\n if (!this._validateStretch(image.stretchX, data && data.width)) {\n this.fire(new ErrorEvent(new Error(`Image \"${id}\" has invalid \"stretchX\" value`)));\n valid = false;\n }\n if (!this._validateStretch(image.stretchY, data && data.height)) {\n this.fire(new ErrorEvent(new Error(`Image \"${id}\" has invalid \"stretchY\" value`)));\n valid = false;\n }\n if (!this._validateContent(image.content, image)) {\n this.fire(new ErrorEvent(new Error(`Image \"${id}\" has invalid \"content\" value`)));\n valid = false;\n }\n return valid;\n }\n\n _validateStretch(stretch: Array<[number, number]>, size: number) {\n if (!stretch) return true;\n let last = 0;\n for (const part of stretch) {\n if (part[0] < last || part[1] < part[0] || size < part[1]) return false;\n last = part[1];\n }\n return true;\n }\n\n _validateContent(content: [number, number, number, number], image: StyleImage) {\n if (!content) return true;\n if (content.length !== 4) return false;\n const spriteData = image.spriteData;\n const width = (spriteData && spriteData.width) || image.data.width;\n const height = (spriteData && spriteData.height) || image.data.height;\n if (content[0] < 0 || width < content[0]) return false;\n if (content[1] < 0 || height < content[1]) return false;\n if (content[2] < 0 || width < content[2]) return false;\n if (content[3] < 0 || height < content[3]) return false;\n if (content[2] < content[0]) return false;\n if (content[3] < content[1]) return false;\n return true;\n }\n\n updateImage(id: string, image: StyleImage, validate = true) {\n const oldImage = this.getImage(id);\n if (validate && (oldImage.data.width !== image.data.width || oldImage.data.height !== image.data.height)) {\n throw new Error(`size mismatch between old image (${oldImage.data.width}x${oldImage.data.height}) and new image (${image.data.width}x${image.data.height}).`);\n }\n image.version = oldImage.version + 1;\n this.images[id] = image;\n this.updatedImages[id] = true;\n }\n\n removeImage(id: string) {\n const image = this.images[id];\n delete this.images[id];\n delete this.patterns[id];\n\n if (image.userImage && image.userImage.onRemove) {\n image.userImage.onRemove();\n }\n }\n\n listImages(): Array {\n return Object.keys(this.images);\n }\n\n getImages(ids: Array, callback: Callback<{[_: string]: StyleImage}>) {\n // If the sprite has been loaded, or if all the icon dependencies are already present\n // (i.e. if they've been added via runtime styling), then notify the requestor immediately.\n // Otherwise, delay notification until the sprite is loaded. At that point, if any of the\n // dependencies are still unavailable, we'll just assume they are permanently missing.\n let hasAllDependencies = true;\n if (!this.isLoaded()) {\n for (const id of ids) {\n if (!this.images[id]) {\n hasAllDependencies = false;\n }\n }\n }\n if (this.isLoaded() || hasAllDependencies) {\n this._notify(ids, callback);\n } else {\n this.requestors.push({ids, callback});\n }\n }\n\n _notify(ids: Array, callback: Callback<{[_: string]: StyleImage}>) {\n const response = {};\n\n for (const id of ids) {\n let image = this.getImage(id);\n\n if (!image) {\n this.fire(new Event('styleimagemissing', {id}));\n //Try to acquire image again in case styleimagemissing has populated it\n image = this.getImage(id);\n }\n\n if (image) {\n // Clone the image so that our own copy of its ArrayBuffer doesn't get transferred.\n response[id] = {\n data: image.data.clone(),\n pixelRatio: image.pixelRatio,\n sdf: image.sdf,\n version: image.version,\n stretchX: image.stretchX,\n stretchY: image.stretchY,\n content: image.content,\n hasRenderCallback: Boolean(image.userImage && image.userImage.render)\n };\n } else {\n warnOnce(`Image \"${id}\" could not be loaded. Please make sure you have added the image with map.addImage() or a \"sprite\" property in your style. You can provide missing images by listening for the \"styleimagemissing\" map event.`);\n }\n }\n\n callback(null, response);\n }\n\n // Pattern stuff\n\n getPixelSize() {\n const {width, height} = this.atlasImage;\n return {width, height};\n }\n\n getPattern(id: string): ImagePosition {\n const pattern = this.patterns[id];\n\n const image = this.getImage(id);\n if (!image) {\n return null;\n }\n\n if (pattern && pattern.position.version === image.version) {\n return pattern.position;\n }\n\n if (!pattern) {\n const w = image.data.width + padding * 2;\n const h = image.data.height + padding * 2;\n const bin = {w, h, x: 0, y: 0};\n const position = new ImagePosition(bin, image);\n this.patterns[id] = {bin, position};\n } else {\n pattern.position.version = image.version;\n }\n\n this._updatePatternAtlas();\n\n return this.patterns[id].position;\n }\n\n bind(context: Context) {\n const gl = context.gl;\n if (!this.atlasTexture) {\n this.atlasTexture = new Texture(context, this.atlasImage, gl.RGBA);\n } else if (this.dirty) {\n this.atlasTexture.update(this.atlasImage);\n this.dirty = false;\n }\n\n this.atlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n }\n\n _updatePatternAtlas() {\n const bins = [];\n for (const id in this.patterns) {\n bins.push(this.patterns[id].bin);\n }\n\n const {w, h} = potpack(bins);\n\n const dst = this.atlasImage;\n dst.resize({width: w || 1, height: h || 1});\n\n for (const id in this.patterns) {\n const {bin} = this.patterns[id];\n const x = bin.x + padding;\n const y = bin.y + padding;\n const src = this.getImage(id).data;\n const w = src.width;\n const h = src.height;\n\n RGBAImage.copy(src, dst, {x: 0, y: 0}, {x, y}, {width: w, height: h});\n\n // Add 1 pixel wrapped padding on each side of the image.\n RGBAImage.copy(src, dst, {x: 0, y: h - 1}, {x, y: y - 1}, {width: w, height: 1}); // T\n RGBAImage.copy(src, dst, {x: 0, y: 0}, {x, y: y + h}, {width: w, height: 1}); // B\n RGBAImage.copy(src, dst, {x: w - 1, y: 0}, {x: x - 1, y}, {width: 1, height: h}); // L\n RGBAImage.copy(src, dst, {x: 0, y: 0}, {x: x + w, y}, {width: 1, height: h}); // R\n }\n\n this.dirty = true;\n }\n\n beginFrame() {\n this.callbackDispatchedThisFrame = {};\n }\n\n dispatchRenderCallbacks(ids: Array) {\n for (const id of ids) {\n\n // the callback for the image was already dispatched for a different frame\n if (this.callbackDispatchedThisFrame[id]) continue;\n this.callbackDispatchedThisFrame[id] = true;\n\n const image = this.getImage(id);\n if (!image) warnOnce(`Image with ID: \"${id}\" was not found`);\n\n const updated = renderStyleImage(image);\n if (updated) {\n this.updateImage(id, image);\n }\n }\n }\n}\n","import {getArrayBuffer} from '../util/ajax';\nimport {ResourceType} from '../util/request_manager';\n\nimport {parseGlyphPbf} from './parse_glyph_pbf';\n\nimport type {StyleGlyph} from './style_glyph';\nimport type {RequestManager} from '../util/request_manager';\nimport type {Callback} from '../types/callback';\n\nexport function loadGlyphRange(fontstack: string,\n range: number,\n urlTemplate: string,\n requestManager: RequestManager,\n callback: Callback<{\n [_: number]: StyleGlyph | null;\n }>) {\n const begin = range * 256;\n const end = begin + 255;\n\n const request = requestManager.transformRequest(\n urlTemplate.replace('{fontstack}', fontstack).replace('{range}', `${begin}-${end}`),\n ResourceType.Glyphs\n );\n\n getArrayBuffer(request, (err?: Error | null, data?: ArrayBuffer | null) => {\n if (err) {\n callback(err);\n } else if (data) {\n const glyphs = {};\n\n for (const glyph of parseGlyphPbf(data)) {\n glyphs[glyph.id] = glyph;\n }\n\n callback(null, glyphs);\n }\n });\n}\n","const INF = 1e20;\n\nexport default class TinySDF {\n constructor({\n fontSize = 24,\n buffer = 3,\n radius = 8,\n cutoff = 0.25,\n fontFamily = 'sans-serif',\n fontWeight = 'normal',\n fontStyle = 'normal'\n } = {}) {\n this.buffer = buffer;\n this.cutoff = cutoff;\n this.radius = radius;\n\n // make the canvas size big enough to both have the specified buffer around the glyph\n // for \"halo\", and account for some glyphs possibly being larger than their font size\n const size = this.size = fontSize + buffer * 4;\n\n const canvas = this._createCanvas(size);\n const ctx = this.ctx = canvas.getContext('2d', {willReadFrequently: true});\n ctx.font = `${fontStyle} ${fontWeight} ${fontSize}px ${fontFamily}`;\n\n ctx.textBaseline = 'alphabetic';\n ctx.textAlign = 'left'; // Necessary so that RTL text doesn't have different alignment\n ctx.fillStyle = 'black';\n\n // temporary arrays for the distance transform\n this.gridOuter = new Float64Array(size * size);\n this.gridInner = new Float64Array(size * size);\n this.f = new Float64Array(size);\n this.z = new Float64Array(size + 1);\n this.v = new Uint16Array(size);\n }\n\n _createCanvas(size) {\n const canvas = document.createElement('canvas');\n canvas.width = canvas.height = size;\n return canvas;\n }\n\n draw(char) {\n const {\n width: glyphAdvance,\n actualBoundingBoxAscent,\n actualBoundingBoxDescent,\n actualBoundingBoxLeft,\n actualBoundingBoxRight\n } = this.ctx.measureText(char);\n\n // The integer/pixel part of the top alignment is encoded in metrics.glyphTop\n // The remainder is implicitly encoded in the rasterization\n const glyphTop = Math.ceil(actualBoundingBoxAscent);\n const glyphLeft = 0;\n\n // If the glyph overflows the canvas size, it will be clipped at the bottom/right\n const glyphWidth = Math.max(0, Math.min(this.size - this.buffer, Math.ceil(actualBoundingBoxRight - actualBoundingBoxLeft)));\n const glyphHeight = Math.min(this.size - this.buffer, glyphTop + Math.ceil(actualBoundingBoxDescent));\n\n const width = glyphWidth + 2 * this.buffer;\n const height = glyphHeight + 2 * this.buffer;\n\n const len = Math.max(width * height, 0);\n const data = new Uint8ClampedArray(len);\n const glyph = {data, width, height, glyphWidth, glyphHeight, glyphTop, glyphLeft, glyphAdvance};\n if (glyphWidth === 0 || glyphHeight === 0) return glyph;\n\n const {ctx, buffer, gridInner, gridOuter} = this;\n ctx.clearRect(buffer, buffer, glyphWidth, glyphHeight);\n ctx.fillText(char, buffer, buffer + glyphTop);\n const imgData = ctx.getImageData(buffer, buffer, glyphWidth, glyphHeight);\n\n // Initialize grids outside the glyph range to alpha 0\n gridOuter.fill(INF, 0, len);\n gridInner.fill(0, 0, len);\n\n for (let y = 0; y < glyphHeight; y++) {\n for (let x = 0; x < glyphWidth; x++) {\n const a = imgData.data[4 * (y * glyphWidth + x) + 3] / 255; // alpha value\n if (a === 0) continue; // empty pixels\n\n const j = (y + buffer) * width + x + buffer;\n\n if (a === 1) { // fully drawn pixels\n gridOuter[j] = 0;\n gridInner[j] = INF;\n\n } else { // aliased pixels\n const d = 0.5 - a;\n gridOuter[j] = d > 0 ? d * d : 0;\n gridInner[j] = d < 0 ? d * d : 0;\n }\n }\n }\n\n edt(gridOuter, 0, 0, width, height, width, this.f, this.v, this.z);\n edt(gridInner, buffer, buffer, glyphWidth, glyphHeight, width, this.f, this.v, this.z);\n\n for (let i = 0; i < len; i++) {\n const d = Math.sqrt(gridOuter[i]) - Math.sqrt(gridInner[i]);\n data[i] = Math.round(255 - 255 * (d / this.radius + this.cutoff));\n }\n\n return glyph;\n }\n}\n\n// 2D Euclidean squared distance transform by Felzenszwalb & Huttenlocher https://cs.brown.edu/~pff/papers/dt-final.pdf\nfunction edt(data, x0, y0, width, height, gridSize, f, v, z) {\n for (let x = x0; x < x0 + width; x++) edt1d(data, y0 * gridSize + x, gridSize, height, f, v, z);\n for (let y = y0; y < y0 + height; y++) edt1d(data, y * gridSize + x0, 1, width, f, v, z);\n}\n\n// 1D squared distance transform\nfunction edt1d(grid, offset, stride, length, f, v, z) {\n v[0] = 0;\n z[0] = -INF;\n z[1] = INF;\n f[0] = grid[offset];\n\n for (let q = 1, k = 0, s = 0; q < length; q++) {\n f[q] = grid[offset + q * stride];\n const q2 = q * q;\n do {\n const r = v[k];\n s = (f[q] - f[r] + q2 - r * r) / (q - r) / 2;\n } while (s <= z[k] && --k > -1);\n\n k++;\n v[k] = q;\n z[k] = s;\n z[k + 1] = INF;\n }\n\n for (let q = 0, k = 0; q < length; q++) {\n while (z[k + 1] < q) k++;\n const r = v[k];\n const qr = q - r;\n grid[offset + q * stride] = f[r] + qr * qr;\n }\n}\n","import {loadGlyphRange} from '../style/load_glyph_range';\n\nimport TinySDF from '@mapbox/tiny-sdf';\nimport {unicodeBlockLookup} from '../util/is_char_in_unicode_block';\nimport {asyncAll} from '../util/util';\nimport {AlphaImage} from '../util/image';\n\nimport type {StyleGlyph} from '../style/style_glyph';\nimport type {RequestManager} from '../util/request_manager';\nimport type {Callback} from '../types/callback';\n\ntype Entry = {\n // null means we've requested the range, but the glyph wasn't included in the result.\n glyphs: {\n [id: number]: StyleGlyph | null;\n };\n requests: {\n [range: number]: Array>;\n };\n ranges: {\n [range: number]: boolean | null;\n };\n tinySDF?: TinySDF;\n};\n\nexport class GlyphManager {\n requestManager: RequestManager;\n localIdeographFontFamily: string;\n entries: {\n [_: string]: Entry;\n };\n url: string;\n\n // exposed as statics to enable stubbing in unit tests\n static loadGlyphRange = loadGlyphRange;\n static TinySDF = TinySDF;\n\n constructor(requestManager: RequestManager, localIdeographFontFamily?: string | null) {\n this.requestManager = requestManager;\n this.localIdeographFontFamily = localIdeographFontFamily;\n this.entries = {};\n }\n\n setURL(url?: string | null) {\n this.url = url;\n }\n\n getGlyphs(glyphs: {\n [stack: string]: Array;\n }, callback: Callback<{\n [stack: string]: {\n [id: number]: StyleGlyph;\n };\n }>) {\n const all = [];\n\n for (const stack in glyphs) {\n for (const id of glyphs[stack]) {\n all.push({stack, id});\n }\n }\n\n asyncAll(all, ({stack, id}, callback: Callback<{\n stack: string;\n id: number;\n glyph: StyleGlyph;\n }>) => {\n let entry = this.entries[stack];\n if (!entry) {\n entry = this.entries[stack] = {\n glyphs: {},\n requests: {},\n ranges: {}\n };\n }\n\n let glyph = entry.glyphs[id];\n if (glyph !== undefined) {\n callback(null, {stack, id, glyph});\n return;\n }\n\n glyph = this._tinySDF(entry, stack, id);\n if (glyph) {\n entry.glyphs[id] = glyph;\n callback(null, {stack, id, glyph});\n return;\n }\n\n const range = Math.floor(id / 256);\n if (range * 256 > 65535) {\n callback(new Error('glyphs > 65535 not supported'));\n return;\n }\n\n if (entry.ranges[range]) {\n callback(null, {stack, id, glyph});\n return;\n }\n\n if (!this.url) {\n callback(new Error('glyphsUrl is not set'));\n return;\n }\n\n let requests = entry.requests[range];\n if (!requests) {\n requests = entry.requests[range] = [];\n GlyphManager.loadGlyphRange(stack, range, this.url, this.requestManager,\n (err, response?: {\n [_: number]: StyleGlyph | null;\n } | null) => {\n if (response) {\n for (const id in response) {\n if (!this._doesCharSupportLocalGlyph(+id)) {\n entry.glyphs[+id] = response[+id];\n }\n }\n entry.ranges[range] = true;\n }\n for (const cb of requests) {\n cb(err, response);\n }\n delete entry.requests[range];\n });\n }\n\n requests.push((err, result?: {\n [_: number]: StyleGlyph | null;\n } | null) => {\n if (err) {\n callback(err);\n } else if (result) {\n callback(null, {stack, id, glyph: result[id] || null});\n }\n });\n }, (err, glyphs?: Array<{\n stack: string;\n id: number;\n glyph: StyleGlyph;\n }> | null) => {\n if (err) {\n callback(err);\n } else if (glyphs) {\n const result = {};\n\n for (const {stack, id, glyph} of glyphs) {\n // Clone the glyph so that our own copy of its ArrayBuffer doesn't get transferred.\n (result[stack] || (result[stack] = {}))[id] = glyph && {\n id: glyph.id,\n bitmap: glyph.bitmap.clone(),\n metrics: glyph.metrics\n };\n }\n\n callback(null, result);\n }\n });\n }\n\n _doesCharSupportLocalGlyph(id: number): boolean {\n /* eslint-disable new-cap */\n return !!this.localIdeographFontFamily &&\n (unicodeBlockLookup['CJK Unified Ideographs'](id) ||\n unicodeBlockLookup['Hangul Syllables'](id) ||\n unicodeBlockLookup['Hiragana'](id) ||\n unicodeBlockLookup['Katakana'](id));\n /* eslint-enable new-cap */\n }\n\n _tinySDF(entry: Entry, stack: string, id: number): StyleGlyph {\n const fontFamily = this.localIdeographFontFamily;\n if (!fontFamily) {\n return;\n }\n\n if (!this._doesCharSupportLocalGlyph(id)) {\n return;\n }\n\n // Client-generated glyphs are rendered at 2x texture scale,\n // because CJK glyphs are more detailed than others.\n const textureScale = 2;\n\n let tinySDF = entry.tinySDF;\n if (!tinySDF) {\n let fontWeight = '400';\n if (/bold/i.test(stack)) {\n fontWeight = '900';\n } else if (/medium/i.test(stack)) {\n fontWeight = '500';\n } else if (/light/i.test(stack)) {\n fontWeight = '200';\n }\n tinySDF = entry.tinySDF = new GlyphManager.TinySDF({\n fontSize: 24 * textureScale,\n buffer: 3 * textureScale,\n radius: 8 * textureScale,\n cutoff: 0.25,\n fontFamily,\n fontWeight\n });\n }\n\n const char = tinySDF.draw(String.fromCharCode(id));\n\n /**\n * TinySDF's \"top\" is the distance from the alphabetic baseline to the top of the glyph.\n * Server-generated fonts specify \"top\" relative to an origin above the em box (the origin\n * comes from FreeType, but I'm unclear on exactly how it's derived)\n * ref: https://github.com/mapbox/sdf-glyph-foundry\n *\n * Server fonts don't yet include baseline information, so we can't line up exactly with them\n * (and they don't line up with each other)\n * ref: https://github.com/mapbox/node-fontnik/pull/160\n *\n * To approximately align TinySDF glyphs with server-provided glyphs, we use this baseline adjustment\n * factor calibrated to be in between DIN Pro and Arial Unicode (but closer to Arial Unicode)\n */\n const topAdjustment = 27.5;\n\n const leftAdjustment = 0.5;\n\n return {\n id,\n bitmap: new AlphaImage({width: char.width || 30 * textureScale, height: char.height || 30 * textureScale}, char.data),\n metrics: {\n width: char.glyphWidth / textureScale || 24,\n height: char.glyphHeight / textureScale || 24,\n left: (char.glyphLeft / textureScale + leftAdjustment) || 0,\n top: char.glyphTop / textureScale - topAdjustment || -8,\n advance: char.glyphAdvance / textureScale || 24,\n isDoubleResolution: true\n }\n };\n }\n}\n","import {interpolates, Color, latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {extend, sphericalToCartesian} from '../util/util';\nimport {Evented} from '../util/evented';\nimport {\n validateStyle,\n validateLight,\n emitValidationErrors\n} from './validate_style';\n\nimport type {StylePropertySpecification, LightSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {EvaluationParameters} from './evaluation_parameters';\nimport type {StyleSetterOptions} from '../style/style';\nimport {Properties, Transitionable, Transitioning, PossiblyEvaluated, DataConstantProperty} from './properties';\n\nimport type {\n Property,\n PropertyValue,\n TransitionParameters\n} from './properties';\n\ntype LightPosition = {\n x: number;\n y: number;\n z: number;\n};\n\nclass LightPositionProperty implements Property<[number, number, number], LightPosition> {\n specification: StylePropertySpecification;\n\n constructor() {\n this.specification = styleSpec.light.position as StylePropertySpecification;\n }\n\n possiblyEvaluate(\n value: PropertyValue<[number, number, number], LightPosition>,\n parameters: EvaluationParameters\n ): LightPosition {\n return sphericalToCartesian(value.expression.evaluate(parameters));\n }\n\n interpolate(a: LightPosition, b: LightPosition, t: number): LightPosition {\n return {\n x: interpolates.number(a.x, b.x, t),\n y: interpolates.number(a.y, b.y, t),\n z: interpolates.number(a.z, b.z, t),\n };\n }\n}\n\ntype Props = {\n 'anchor': DataConstantProperty<'map' | 'viewport'>;\n 'position': LightPositionProperty;\n 'color': DataConstantProperty;\n 'intensity': DataConstantProperty;\n};\n\ntype PropsPossiblyEvaluated = {\n 'anchor': 'map' | 'viewport';\n 'position': LightPosition;\n 'color': Color;\n 'intensity': number;\n};\n\nconst TRANSITION_SUFFIX = '-transition';\n\nlet lightProperties: Properties;\n\n/*\n * Represents the light used to light extruded features.\n */\nexport class Light extends Evented {\n _transitionable: Transitionable;\n _transitioning: Transitioning;\n properties: PossiblyEvaluated;\n\n constructor(lightOptions?: LightSpecification) {\n super();\n lightProperties = lightProperties || new Properties({\n 'anchor': new DataConstantProperty(styleSpec.light.anchor as StylePropertySpecification),\n 'position': new LightPositionProperty(),\n 'color': new DataConstantProperty(styleSpec.light.color as StylePropertySpecification),\n 'intensity': new DataConstantProperty(styleSpec.light.intensity as StylePropertySpecification),\n });\n this._transitionable = new Transitionable(lightProperties);\n this.setLight(lightOptions);\n this._transitioning = this._transitionable.untransitioned();\n }\n\n getLight(): LightSpecification {\n return this._transitionable.serialize();\n }\n\n setLight(light?: LightSpecification, options: StyleSetterOptions = {}) {\n if (this._validate(validateLight, light, options)) {\n return;\n }\n\n for (const name in light) {\n const value = light[name];\n if (name.endsWith(TRANSITION_SUFFIX)) {\n this._transitionable.setTransition(name.slice(0, -TRANSITION_SUFFIX.length) as keyof Props, value);\n } else {\n this._transitionable.setValue(name as keyof Props, value);\n }\n }\n }\n\n updateTransitions(parameters: TransitionParameters) {\n this._transitioning = this._transitionable.transitioned(parameters, this._transitioning);\n }\n\n hasTransition() {\n return this._transitioning.hasTransition();\n }\n\n recalculate(parameters: EvaluationParameters) {\n this.properties = this._transitioning.possiblyEvaluate(parameters);\n }\n\n _validate(validate: Function, value: unknown, options?: {\n validate?: boolean;\n }) {\n if (options && options.validate === false) {\n return false;\n }\n\n return emitValidationErrors(this, validate.call(validateStyle, extend({\n value,\n // Workaround for https://github.com/mapbox/mapbox-gl-js/issues/2407\n style: {glyphs: true, sprite: true},\n styleSpec\n })));\n }\n}\n","import {warnOnce} from '../util/util';\n\nimport type {Context} from '../gl/context';\n\n/**\n * A dash entry\n */\ntype DashEntry = {\n y: number;\n height: number;\n width: number;\n}\n\n/**\n * @internal\n * A LineAtlas lets us reuse rendered dashed lines\n * by writing many of them to a texture and then fetching their positions\n * using {@link LineAtlas#getDash}.\n *\n * @param width - the width\n * @param height - the height\n */\nexport class LineAtlas {\n width: number;\n height: number;\n nextRow: number;\n bytes: number;\n data: Uint8Array;\n dashEntry: {[_: string]: DashEntry};\n dirty: boolean;\n texture: WebGLTexture;\n\n constructor(width: number, height: number) {\n this.width = width;\n this.height = height;\n this.nextRow = 0;\n\n this.data = new Uint8Array(this.width * this.height);\n\n this.dashEntry = {};\n }\n\n /**\n * Get or create a dash line pattern.\n *\n * @param dasharray - the key (represented by numbers) to get the dash texture\n * @param round - whether to add circle caps in between dash segments\n * @returns position of dash texture in {@link DashEntry}\n */\n getDash(dasharray: Array, round: boolean) {\n const key = dasharray.join(',') + String(round);\n\n if (!this.dashEntry[key]) {\n this.dashEntry[key] = this.addDash(dasharray, round);\n }\n return this.dashEntry[key];\n }\n\n getDashRanges(dasharray: Array, lineAtlasWidth: number, stretch: number) {\n // If dasharray has an odd length, both the first and last parts\n // are dashes and should be joined seamlessly.\n const oddDashArray = dasharray.length % 2 === 1;\n\n const ranges = [];\n\n let left = oddDashArray ? -dasharray[dasharray.length - 1] * stretch : 0;\n let right = dasharray[0] * stretch;\n let isDash = true;\n\n ranges.push({left, right, isDash, zeroLength: dasharray[0] === 0});\n\n let currentDashLength = dasharray[0];\n for (let i = 1; i < dasharray.length; i++) {\n isDash = !isDash;\n\n const dashLength = dasharray[i];\n left = currentDashLength * stretch;\n currentDashLength += dashLength;\n right = currentDashLength * stretch;\n\n ranges.push({left, right, isDash, zeroLength: dashLength === 0});\n }\n\n return ranges;\n }\n\n addRoundDash(ranges: any, stretch: number, n: number) {\n const halfStretch = stretch / 2;\n\n for (let y = -n; y <= n; y++) {\n const row = this.nextRow + n + y;\n const index = this.width * row;\n let currIndex = 0;\n let range = ranges[currIndex];\n\n for (let x = 0; x < this.width; x++) {\n if (x / range.right > 1) { range = ranges[++currIndex]; }\n\n const distLeft = Math.abs(x - range.left);\n const distRight = Math.abs(x - range.right);\n const minDist = Math.min(distLeft, distRight);\n let signedDistance;\n\n const distMiddle = y / n * (halfStretch + 1);\n if (range.isDash) {\n const distEdge = halfStretch - Math.abs(distMiddle);\n signedDistance = Math.sqrt(minDist * minDist + distEdge * distEdge);\n } else {\n signedDistance = halfStretch - Math.sqrt(minDist * minDist + distMiddle * distMiddle);\n }\n\n this.data[index + x] = Math.max(0, Math.min(255, signedDistance + 128));\n }\n }\n }\n\n addRegularDash(ranges: any) {\n\n // Collapse any zero-length range\n // Collapse neighbouring same-type parts into a single part\n for (let i = ranges.length - 1; i >= 0; --i) {\n const part = ranges[i];\n const next = ranges[i + 1];\n if (part.zeroLength) {\n ranges.splice(i, 1);\n } else if (next && next.isDash === part.isDash) {\n next.left = part.left;\n ranges.splice(i, 1);\n }\n }\n\n // Combine the first and last parts if possible\n const first = ranges[0];\n const last = ranges[ranges.length - 1];\n if (first.isDash === last.isDash) {\n first.left = last.left - this.width;\n last.right = first.right + this.width;\n }\n\n const index = this.width * this.nextRow;\n let currIndex = 0;\n let range = ranges[currIndex];\n\n for (let x = 0; x < this.width; x++) {\n if (x / range.right > 1) {\n range = ranges[++currIndex];\n }\n\n const distLeft = Math.abs(x - range.left);\n const distRight = Math.abs(x - range.right);\n\n const minDist = Math.min(distLeft, distRight);\n const signedDistance = range.isDash ? minDist : -minDist;\n\n this.data[index + x] = Math.max(0, Math.min(255, signedDistance + 128));\n }\n }\n\n addDash(dasharray: Array, round: boolean): DashEntry {\n const n = round ? 7 : 0;\n const height = 2 * n + 1;\n\n if (this.nextRow + height > this.height) {\n warnOnce('LineAtlas out of space');\n return null;\n }\n\n let length = 0;\n for (let i = 0; i < dasharray.length; i++) { length += dasharray[i]; }\n\n if (length !== 0) {\n const stretch = this.width / length;\n const ranges = this.getDashRanges(dasharray, this.width, stretch);\n\n if (round) {\n this.addRoundDash(ranges, stretch, n);\n } else {\n this.addRegularDash(ranges);\n }\n }\n\n const dashEntry = {\n y: (this.nextRow + n + 0.5) / this.height,\n height: 2 * n / this.height,\n width: length\n };\n\n this.nextRow += height;\n this.dirty = true;\n\n return dashEntry;\n }\n\n bind(context: Context) {\n const gl = context.gl;\n if (!this.texture) {\n this.texture = gl.createTexture();\n gl.bindTexture(gl.TEXTURE_2D, this.texture);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.REPEAT);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.REPEAT);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n gl.texImage2D(gl.TEXTURE_2D, 0, gl.ALPHA, this.width, this.height, 0, gl.ALPHA, gl.UNSIGNED_BYTE, this.data);\n\n } else {\n gl.bindTexture(gl.TEXTURE_2D, this.texture);\n\n if (this.dirty) {\n this.dirty = false;\n gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, this.width, this.height, gl.ALPHA, gl.UNSIGNED_BYTE, this.data);\n }\n }\n }\n}\n","import {asyncAll} from './util';\nimport {Actor, GlyphsProvider, MessageType} from './actor';\n\nimport type {WorkerPool} from './worker_pool';\nimport type {WorkerSource} from '../source/worker_source'; /* eslint-disable-line */ // this is used for the docs' import\n/**\n * Responsible for sending messages from a {@link Source} to an associated\n * {@link WorkerSource}.\n */\nexport class Dispatcher {\n workerPool: WorkerPool;\n actors: Array;\n currentActor: number;\n id: string | number;\n\n constructor(workerPool: WorkerPool, parent: GlyphsProvider, mapId: string | number) {\n this.workerPool = workerPool;\n this.actors = [];\n this.currentActor = 0;\n this.id = mapId;\n const workers = this.workerPool.acquire(mapId);\n for (let i = 0; i < workers.length; i++) {\n const worker = workers[i];\n const actor = new Actor(worker, parent, mapId);\n actor.name = `Worker ${i}`;\n this.actors.push(actor);\n }\n if (!this.actors.length) throw new Error('No actors found');\n }\n\n /**\n * Broadcast a message to all Workers.\n */\n broadcast(type: MessageType, data: unknown, cb?: (...args: any[]) => any) {\n cb = cb || function () {};\n asyncAll(this.actors, (actor, done) => {\n actor.send(type, data, done);\n }, cb);\n }\n\n /**\n * Acquires an actor to dispatch messages to. The actors are distributed in round-robin fashion.\n * @returns An actor object backed by a web worker for processing messages.\n */\n getActor(): Actor {\n this.currentActor = (this.currentActor + 1) % this.actors.length;\n return this.actors[this.currentActor];\n }\n\n remove(mapRemoved: boolean = true) {\n this.actors.forEach((actor) => { actor.remove(); });\n this.actors = [];\n if (mapRemoved) this.workerPool.release(this.id);\n }\n}\n","import {pick, extend} from '../util/util';\n\nimport {getJSON} from '../util/ajax';\nimport {ResourceType} from '../util/request_manager';\nimport {browser} from '../util/browser';\n\nimport type {RequestManager} from '../util/request_manager';\nimport type {Callback} from '../types/callback';\nimport type {TileJSON} from '../types/tilejson';\nimport type {Cancelable} from '../types/cancelable';\nimport type {RasterDEMSourceSpecification, RasterSourceSpecification, VectorSourceSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport function loadTileJson(\n options: RasterSourceSpecification | RasterDEMSourceSpecification | VectorSourceSpecification,\n requestManager: RequestManager,\n callback: Callback\n): Cancelable {\n const loaded = function(err: Error, tileJSON: any) {\n if (err) {\n return callback(err);\n } else if (tileJSON) {\n const result: any = pick(\n // explicit source options take precedence over TileJSON\n extend(tileJSON, options),\n ['tiles', 'minzoom', 'maxzoom', 'attribution', 'bounds', 'scheme', 'tileSize', 'encoding']\n );\n\n if (tileJSON.vector_layers) {\n result.vectorLayers = tileJSON.vector_layers;\n result.vectorLayerIds = result.vectorLayers.map((layer) => { return layer.id; });\n }\n\n callback(null, result);\n }\n };\n\n if (options.url) {\n return getJSON(requestManager.transformRequest(options.url, ResourceType.Source), loaded);\n } else {\n return browser.frame(() => loaded(null, options));\n }\n}\n","import {LngLat} from './lng_lat';\nimport type {LngLatLike} from './lng_lat';\n\n/**\n * A {@link LngLatBounds} object, an array of {@link LngLatLike} objects in [sw, ne] order,\n * or an array of numbers in [west, south, east, north] order.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let v1 = new maplibregl.LngLatBounds(\n * new maplibregl.LngLat(-73.9876, 40.7661),\n * new maplibregl.LngLat(-73.9397, 40.8002)\n * );\n * let v2 = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002])\n * let v3 = [[-73.9876, 40.7661], [-73.9397, 40.8002]];\n * ```\n */\nexport type LngLatBoundsLike = LngLatBounds | [LngLatLike, LngLatLike] | [number, number, number, number];\n\n/**\n * A `LngLatBounds` object represents a geographical bounding box,\n * defined by its southwest and northeast points in longitude and latitude.\n *\n * If no arguments are provided to the constructor, a `null` bounding box is created.\n *\n * Note that any Mapbox GL method that accepts a `LngLatBounds` object as an argument or option\n * can also accept an `Array` of two {@link LngLatLike} constructs and will perform an implicit conversion.\n * This flexible type is documented as {@link LngLatBoundsLike}.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let sw = new maplibregl.LngLat(-73.9876, 40.7661);\n * let ne = new maplibregl.LngLat(-73.9397, 40.8002);\n * let llb = new maplibregl.LngLatBounds(sw, ne);\n * ```\n */\nexport class LngLatBounds {\n _ne: LngLat;\n _sw: LngLat;\n\n /**\n * @param sw - The southwest corner of the bounding box.\n * OR array of 4 numbers in the order of west, south, east, north\n * OR array of 2 LngLatLike: [sw,ne]\n * @param ne - The northeast corner of the bounding box.\n * @example\n * ```ts\n * let sw = new maplibregl.LngLat(-73.9876, 40.7661);\n * let ne = new maplibregl.LngLat(-73.9397, 40.8002);\n * let llb = new maplibregl.LngLatBounds(sw, ne);\n * ```\n * OR\n * ```ts\n * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661, -73.9397, 40.8002]);\n * ```\n * OR\n * ```ts\n * let llb = new maplibregl.LngLatBounds([sw, ne]);\n * ```\n */\n constructor(sw?: LngLatLike | [number, number, number, number] | [LngLatLike, LngLatLike], ne?: LngLatLike) {\n if (!sw) {\n // noop\n } else if (ne) {\n this.setSouthWest(sw).setNorthEast(ne);\n } else if (Array.isArray(sw)) {\n if (sw.length === 4) {\n // 4 element array: west, south, east, north\n this.setSouthWest([sw[0], sw[1]]).setNorthEast([sw[2], sw[3]]);\n } else {\n this.setSouthWest(sw[0] as LngLatLike).setNorthEast(sw[1] as LngLatLike);\n }\n }\n }\n\n /**\n * Set the northeast corner of the bounding box\n *\n * @param ne - a {@link LngLatLike} object describing the northeast corner of the bounding box.\n * @returns `this`\n */\n setNorthEast(ne: LngLatLike): this {\n this._ne = ne instanceof LngLat ? new LngLat(ne.lng, ne.lat) : LngLat.convert(ne);\n return this;\n }\n\n /**\n * Set the southwest corner of the bounding box\n *\n * @param sw - a {@link LngLatLike} object describing the southwest corner of the bounding box.\n * @returns `this`\n */\n setSouthWest(sw: LngLatLike): this {\n this._sw = sw instanceof LngLat ? new LngLat(sw.lng, sw.lat) : LngLat.convert(sw);\n return this;\n }\n\n /**\n * Extend the bounds to include a given LngLatLike or LngLatBoundsLike.\n *\n * @param obj - object to extend to\n * @returns `this`\n */\n extend(obj: LngLatLike | LngLatBoundsLike): this {\n const sw = this._sw,\n ne = this._ne;\n let sw2, ne2;\n\n if (obj instanceof LngLat) {\n sw2 = obj;\n ne2 = obj;\n\n } else if (obj instanceof LngLatBounds) {\n sw2 = obj._sw;\n ne2 = obj._ne;\n\n if (!sw2 || !ne2) return this;\n\n } else {\n if (Array.isArray(obj)) {\n if (obj.length === 4 || (obj as any[]).every(Array.isArray)) {\n const lngLatBoundsObj = (obj as any as LngLatBoundsLike);\n return this.extend(LngLatBounds.convert(lngLatBoundsObj));\n } else {\n const lngLatObj = (obj as any as LngLatLike);\n return this.extend(LngLat.convert(lngLatObj));\n }\n\n } else if (obj && ('lng' in obj || 'lon' in obj) && 'lat' in obj) {\n return this.extend(LngLat.convert(obj));\n }\n\n return this;\n }\n\n if (!sw && !ne) {\n this._sw = new LngLat(sw2.lng, sw2.lat);\n this._ne = new LngLat(ne2.lng, ne2.lat);\n\n } else {\n sw.lng = Math.min(sw2.lng, sw.lng);\n sw.lat = Math.min(sw2.lat, sw.lat);\n ne.lng = Math.max(ne2.lng, ne.lng);\n ne.lat = Math.max(ne2.lat, ne.lat);\n }\n\n return this;\n }\n\n /**\n * Returns the geographical coordinate equidistant from the bounding box's corners.\n *\n * @returns The bounding box's center.\n * @example\n * ```ts\n * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]);\n * llb.getCenter(); // = LngLat {lng: -73.96365, lat: 40.78315}\n * ```\n */\n getCenter(): LngLat {\n return new LngLat((this._sw.lng + this._ne.lng) / 2, (this._sw.lat + this._ne.lat) / 2);\n }\n\n /**\n * Returns the southwest corner of the bounding box.\n *\n * @returns The southwest corner of the bounding box.\n */\n getSouthWest(): LngLat { return this._sw; }\n\n /**\n * Returns the northeast corner of the bounding box.\n *\n * @returns The northeast corner of the bounding box.\n */\n getNorthEast(): LngLat { return this._ne; }\n\n /**\n * Returns the northwest corner of the bounding box.\n *\n * @returns The northwest corner of the bounding box.\n */\n getNorthWest(): LngLat { return new LngLat(this.getWest(), this.getNorth()); }\n\n /**\n * Returns the southeast corner of the bounding box.\n *\n * @returns The southeast corner of the bounding box.\n */\n getSouthEast(): LngLat { return new LngLat(this.getEast(), this.getSouth()); }\n\n /**\n * Returns the west edge of the bounding box.\n *\n * @returns The west edge of the bounding box.\n */\n getWest(): number { return this._sw.lng; }\n\n /**\n * Returns the south edge of the bounding box.\n *\n * @returns The south edge of the bounding box.\n */\n getSouth(): number { return this._sw.lat; }\n\n /**\n * Returns the east edge of the bounding box.\n *\n * @returns The east edge of the bounding box.\n */\n getEast(): number { return this._ne.lng; }\n\n /**\n * Returns the north edge of the bounding box.\n *\n * @returns The north edge of the bounding box.\n */\n getNorth(): number { return this._ne.lat; }\n\n /**\n * Returns the bounding box represented as an array.\n *\n * @returns The bounding box represented as an array, consisting of the\n * southwest and northeast coordinates of the bounding represented as arrays of numbers.\n * @example\n * ```ts\n * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]);\n * llb.toArray(); // = [[-73.9876, 40.7661], [-73.9397, 40.8002]]\n * ```\n */\n toArray() {\n return [this._sw.toArray(), this._ne.toArray()];\n }\n\n /**\n * Return the bounding box represented as a string.\n *\n * @returns The bounding box represents as a string of the format\n * `'LngLatBounds(LngLat(lng, lat), LngLat(lng, lat))'`.\n * @example\n * ```ts\n * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]);\n * llb.toString(); // = \"LngLatBounds(LngLat(-73.9876, 40.7661), LngLat(-73.9397, 40.8002))\"\n * ```\n */\n toString() {\n return `LngLatBounds(${this._sw.toString()}, ${this._ne.toString()})`;\n }\n\n /**\n * Check if the bounding box is an empty/`null`-type box.\n *\n * @returns True if bounds have been defined, otherwise false.\n */\n isEmpty() {\n return !(this._sw && this._ne);\n }\n\n /**\n * Check if the point is within the bounding box.\n *\n * @param lnglat - geographic point to check against.\n * @returns `true` if the point is within the bounding box.\n * @example\n * ```ts\n * let llb = new maplibregl.LngLatBounds(\n * new maplibregl.LngLat(-73.9876, 40.7661),\n * new maplibregl.LngLat(-73.9397, 40.8002)\n * );\n *\n * let ll = new maplibregl.LngLat(-73.9567, 40.7789);\n *\n * console.log(llb.contains(ll)); // = true\n * ```\n */\n contains(lnglat: LngLatLike) {\n const {lng, lat} = LngLat.convert(lnglat);\n\n const containsLatitude = this._sw.lat <= lat && lat <= this._ne.lat;\n let containsLongitude = this._sw.lng <= lng && lng <= this._ne.lng;\n if (this._sw.lng > this._ne.lng) { // wrapped coordinates\n containsLongitude = this._sw.lng >= lng && lng >= this._ne.lng;\n }\n\n return containsLatitude && containsLongitude;\n }\n\n /**\n * Converts an array to a `LngLatBounds` object.\n *\n * If a `LngLatBounds` object is passed in, the function returns it unchanged.\n *\n * Internally, the function calls `LngLat#convert` to convert arrays to `LngLat` values.\n *\n * @param input - An array of two coordinates to convert, or a `LngLatBounds` object to return.\n * @returns A new `LngLatBounds` object, if a conversion occurred, or the original `LngLatBounds` object.\n * @example\n * ```ts\n * let arr = [[-73.9876, 40.7661], [-73.9397, 40.8002]];\n * let llb = maplibregl.LngLatBounds.convert(arr); // = LngLatBounds {_sw: LngLat {lng: -73.9876, lat: 40.7661}, _ne: LngLat {lng: -73.9397, lat: 40.8002}}\n * ```\n */\n static convert(input: LngLatBoundsLike | null): LngLatBounds {\n if (input instanceof LngLatBounds) return input;\n if (!input) return input as null;\n return new LngLatBounds(input);\n }\n\n /**\n * Returns a `LngLatBounds` from the coordinates extended by a given `radius`. The returned `LngLatBounds` completely contains the `radius`.\n *\n * @param center - center coordinates of the new bounds.\n * @param radius - Distance in meters from the coordinates to extend the bounds.\n * @returns A new `LngLatBounds` object representing the coordinates extended by the `radius`.\n * @example\n * ```ts\n * let center = new maplibregl.LngLat(-73.9749, 40.7736);\n * maplibregl.LngLatBounds.fromLngLat(100).toArray(); // = [[-73.97501862141328, 40.77351016847229], [-73.97478137858673, 40.77368983152771]]\n * ```\n */\n static fromLngLat(center: LngLat, radius:number = 0): LngLatBounds {\n const earthCircumferenceInMetersAtEquator = 40075017;\n const latAccuracy = 360 * radius / earthCircumferenceInMetersAtEquator,\n lngAccuracy = latAccuracy / Math.cos((Math.PI / 180) * center.lat);\n\n return new LngLatBounds(new LngLat(center.lng - lngAccuracy, center.lat - latAccuracy),\n new LngLat(center.lng + lngAccuracy, center.lat + latAccuracy));\n }\n}\n","import {LngLatBounds, LngLatBoundsLike} from '../geo/lng_lat_bounds';\nimport {mercatorXfromLng, mercatorYfromLat} from '../geo/mercator_coordinate';\n\nimport type {CanonicalTileID} from './tile_id';\n\nexport class TileBounds {\n bounds: LngLatBounds;\n minzoom: number;\n maxzoom: number;\n\n constructor(bounds: [number, number, number, number], minzoom?: number | null, maxzoom?: number | null) {\n this.bounds = LngLatBounds.convert(this.validateBounds(bounds));\n this.minzoom = minzoom || 0;\n this.maxzoom = maxzoom || 24;\n }\n\n validateBounds(bounds: [number, number, number, number]): LngLatBoundsLike {\n // make sure the bounds property contains valid longitude and latitudes\n if (!Array.isArray(bounds) || bounds.length !== 4) return [-180, -90, 180, 90];\n return [Math.max(-180, bounds[0]), Math.max(-90, bounds[1]), Math.min(180, bounds[2]), Math.min(90, bounds[3])];\n }\n\n contains(tileID: CanonicalTileID) {\n const worldSize = Math.pow(2, tileID.z);\n const level = {\n minX: Math.floor(mercatorXfromLng(this.bounds.getWest()) * worldSize),\n minY: Math.floor(mercatorYfromLat(this.bounds.getNorth()) * worldSize),\n maxX: Math.ceil(mercatorXfromLng(this.bounds.getEast()) * worldSize),\n maxY: Math.ceil(mercatorYfromLat(this.bounds.getSouth()) * worldSize)\n };\n const hit = tileID.x >= level.minX && tileID.x < level.maxX && tileID.y >= level.minY && tileID.y < level.maxY;\n return hit;\n }\n}\n","import {Event, ErrorEvent, Evented} from '../util/evented';\n\nimport {extend, pick} from '../util/util';\nimport {loadTileJson} from './load_tilejson';\nimport {TileBounds} from './tile_bounds';\nimport {ResourceType} from '../util/request_manager';\n\nimport type {Source} from './source';\nimport type {OverscaledTileID} from './tile_id';\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\nimport type {VectorSourceSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type VectorTileSourceOptions = VectorSourceSpecification & {\n collectResourceTiming?: boolean;\n}\n\n/**\n * A source containing vector tiles in [Mapbox Vector Tile format](https://docs.mapbox.com/vector-tiles/reference/).\n * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * map.addSource('some id', {\n * type: 'vector',\n * url: 'https://demotiles.maplibre.org/tiles/tiles.json'\n * });\n * ```\n *\n * @example\n * ```ts\n * map.addSource('some id', {\n * type: 'vector',\n * tiles: ['https://d25uarhxywzl1j.cloudfront.net/v0.1/{z}/{x}/{y}.mvt'],\n * minzoom: 6,\n * maxzoom: 14\n * });\n * ```\n *\n * @example\n * ```ts\n * map.getSource('some id').setUrl(\"https://demotiles.maplibre.org/tiles/tiles.json\");\n * ```\n *\n * @example\n * ```ts\n * map.getSource('some id').setTiles(['https://d25uarhxywzl1j.cloudfront.net/v0.1/{z}/{x}/{y}.mvt']);\n * ```\n * @see [Add a vector tile source](https://maplibre.org/maplibre-gl-js/docs/examples/vector-source/)\n */\nexport class VectorTileSource extends Evented implements Source {\n type: 'vector';\n id: string;\n minzoom: number;\n maxzoom: number;\n url: string;\n scheme: string;\n tileSize: number;\n promoteId: PromoteIdSpecification;\n\n _options: VectorSourceSpecification;\n _collectResourceTiming: boolean;\n dispatcher: Dispatcher;\n map: Map;\n bounds: [number, number, number, number];\n tiles: Array;\n tileBounds: TileBounds;\n reparseOverscaled: boolean;\n isTileClipped: boolean;\n _tileJSONRequest: Cancelable;\n _loaded: boolean;\n\n constructor(id: string, options: VectorTileSourceOptions, dispatcher: Dispatcher, eventedParent: Evented) {\n super();\n this.id = id;\n this.dispatcher = dispatcher;\n\n this.type = 'vector';\n this.minzoom = 0;\n this.maxzoom = 22;\n this.scheme = 'xyz';\n this.tileSize = 512;\n this.reparseOverscaled = true;\n this.isTileClipped = true;\n this._loaded = false;\n\n extend(this, pick(options, ['url', 'scheme', 'tileSize', 'promoteId']));\n this._options = extend({type: 'vector'}, options);\n\n this._collectResourceTiming = options.collectResourceTiming;\n\n if (this.tileSize !== 512) {\n throw new Error('vector tile sources must have a tileSize of 512');\n }\n\n this.setEventedParent(eventedParent);\n }\n\n load = () => {\n this._loaded = false;\n this.fire(new Event('dataloading', {dataType: 'source'}));\n this._tileJSONRequest = loadTileJson(this._options, this.map._requestManager, (err, tileJSON) => {\n this._tileJSONRequest = null;\n this._loaded = true;\n this.map.style.sourceCaches[this.id].clearTiles();\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (tileJSON) {\n extend(this, tileJSON);\n if (tileJSON.bounds) this.tileBounds = new TileBounds(tileJSON.bounds, this.minzoom, this.maxzoom);\n\n // `content` is included here to prevent a race condition where `Style#_updateSources` is called\n // before the TileJSON arrives. this makes sure the tiles needed are loaded once TileJSON arrives\n // ref: https://github.com/mapbox/mapbox-gl-js/pull/4347#discussion_r104418088\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'}));\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'}));\n }\n });\n };\n\n loaded(): boolean {\n return this._loaded;\n }\n\n hasTile(tileID: OverscaledTileID) {\n return !this.tileBounds || this.tileBounds.contains(tileID.canonical);\n }\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n }\n\n setSourceProperty(callback: Function) {\n if (this._tileJSONRequest) {\n this._tileJSONRequest.cancel();\n }\n\n callback();\n\n this.load();\n }\n\n /**\n * Sets the source `tiles` property and re-renders the map.\n *\n * @param tiles - An array of one or more tile source URLs, as in the TileJSON spec.\n * @returns `this`\n */\n setTiles(tiles: Array): this {\n this.setSourceProperty(() => {\n this._options.tiles = tiles;\n });\n\n return this;\n }\n\n /**\n * Sets the source `url` property and re-renders the map.\n *\n * @param url - A URL to a TileJSON resource. Supported protocols are `http:` and `https:`.\n * @returns `this`\n */\n setUrl(url: string): this {\n this.setSourceProperty(() => {\n this.url = url;\n this._options.url = url;\n });\n\n return this;\n }\n\n onRemove() {\n if (this._tileJSONRequest) {\n this._tileJSONRequest.cancel();\n this._tileJSONRequest = null;\n }\n }\n\n serialize = (): VectorSourceSpecification => {\n return extend({}, this._options);\n };\n\n loadTile(tile: Tile, callback: Callback) {\n const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme);\n const params = {\n request: this.map._requestManager.transformRequest(url, ResourceType.Tile),\n uid: tile.uid,\n tileID: tile.tileID,\n zoom: tile.tileID.overscaledZ,\n tileSize: this.tileSize * tile.tileID.overscaleFactor(),\n type: this.type,\n source: this.id,\n pixelRatio: this.map.getPixelRatio(),\n showCollisionBoxes: this.map.showCollisionBoxes,\n promoteId: this.promoteId\n };\n params.request.collectResourceTiming = this._collectResourceTiming;\n\n if (!tile.actor || tile.state === 'expired') {\n tile.actor = this.dispatcher.getActor();\n tile.request = tile.actor.send('loadTile', params, done.bind(this));\n } else if (tile.state === 'loading') {\n // schedule tile reloading after it has been loaded\n tile.reloadCallback = callback;\n } else {\n tile.request = tile.actor.send('reloadTile', params, done.bind(this));\n }\n\n function done(err, data) {\n delete tile.request;\n\n if (tile.aborted)\n return callback(null);\n\n if (err && err.status !== 404) {\n return callback(err);\n }\n\n if (data && data.resourceTiming)\n tile.resourceTiming = data.resourceTiming;\n\n if (this.map._refreshExpiredTiles && data) tile.setExpiryData(data);\n tile.loadVectorData(data, this.map.painter);\n\n callback(null);\n\n if (tile.reloadCallback) {\n this.loadTile(tile, tile.reloadCallback);\n tile.reloadCallback = null;\n }\n }\n }\n\n abortTile(tile: Tile) {\n if (tile.request) {\n tile.request.cancel();\n delete tile.request;\n }\n if (tile.actor) {\n tile.actor.send('abortTile', {uid: tile.uid, type: this.type, source: this.id}, undefined);\n }\n }\n\n unloadTile(tile: Tile) {\n tile.unloadVectorData();\n if (tile.actor) {\n tile.actor.send('removeTile', {uid: tile.uid, type: this.type, source: this.id}, undefined);\n }\n }\n\n hasTransition() {\n return false;\n }\n}\n","import {extend, pick} from '../util/util';\n\nimport {ImageRequest} from '../util/image_request';\n\nimport {ResourceType} from '../util/request_manager';\nimport {Event, ErrorEvent, Evented} from '../util/evented';\nimport {loadTileJson} from './load_tilejson';\nimport {TileBounds} from './tile_bounds';\nimport {Texture} from '../render/texture';\n\nimport type {Source} from './source';\nimport type {OverscaledTileID} from './tile_id';\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\nimport type {\n RasterSourceSpecification,\n RasterDEMSourceSpecification\n} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A source containing raster tiles (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * map.addSource('raster-source', {\n * 'type': 'raster',\n * 'tiles': ['https://tiles.stadiamaps.com/tiles/stamen_watercolor/{z}/{x}/{y}.jpg'],\n * 'tileSize': 256,\n * });\n * ```\n *\n * @example\n * ```ts\n * map.addSource('wms-test-source', {\n * 'type': 'raster',\n * // use the tiles option to specify a WMS tile source URL\n * 'tiles': [\n * 'https://img.nj.gov/imagerywms/Natural2015?bbox={bbox-epsg-3857}&format=image/png&service=WMS&version=1.1.1&request=GetMap&srs=EPSG:3857&transparent=true&width=256&height=256&layers=Natural2015'\n * ],\n * 'tileSize': 256\n * });\n * ```\n * @see [Add a raster tile source](https://maplibre.org/maplibre-gl-js/docs/examples/map-tiles/)\n * @see [Add a WMS source](https://maplibre.org/maplibre-gl-js/docs/examples/wms/)\n * @see [Display a satellite map](https://maplibre.org/maplibre-gl-js/docs/examples/satellite-map/)\n */\nexport class RasterTileSource extends Evented implements Source {\n type: 'raster' | 'raster-dem';\n id: string;\n minzoom: number;\n maxzoom: number;\n url: string;\n scheme: string;\n tileSize: number;\n\n bounds: [number, number, number, number];\n tileBounds: TileBounds;\n roundZoom: boolean;\n dispatcher: Dispatcher;\n map: Map;\n tiles: Array;\n\n _loaded: boolean;\n _options: RasterSourceSpecification | RasterDEMSourceSpecification;\n _tileJSONRequest: Cancelable;\n\n constructor(id: string, options: RasterSourceSpecification | RasterDEMSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super();\n this.id = id;\n this.dispatcher = dispatcher;\n this.setEventedParent(eventedParent);\n\n this.type = 'raster';\n this.minzoom = 0;\n this.maxzoom = 22;\n this.roundZoom = true;\n this.scheme = 'xyz';\n this.tileSize = 512;\n this._loaded = false;\n\n this._options = extend({type: 'raster'}, options);\n extend(this, pick(options, ['url', 'scheme', 'tileSize']));\n }\n\n load() {\n this._loaded = false;\n this.fire(new Event('dataloading', {dataType: 'source'}));\n this._tileJSONRequest = loadTileJson(this._options, this.map._requestManager, (err, tileJSON) => {\n this._tileJSONRequest = null;\n this._loaded = true;\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (tileJSON) {\n extend(this, tileJSON);\n if (tileJSON.bounds) this.tileBounds = new TileBounds(tileJSON.bounds, this.minzoom, this.maxzoom);\n\n // `content` is included here to prevent a race condition where `Style#_updateSources` is called\n // before the TileJSON arrives. this makes sure the tiles needed are loaded once TileJSON arrives\n // ref: https://github.com/mapbox/mapbox-gl-js/pull/4347#discussion_r104418088\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'}));\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'}));\n }\n });\n }\n\n loaded(): boolean {\n return this._loaded;\n }\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n }\n\n onRemove() {\n if (this._tileJSONRequest) {\n this._tileJSONRequest.cancel();\n this._tileJSONRequest = null;\n }\n }\n\n setSourceProperty(callback: Function) {\n if (this._tileJSONRequest) {\n this._tileJSONRequest.cancel();\n }\n\n callback();\n\n this.load();\n }\n\n /**\n * Sets the source `tiles` property and re-renders the map.\n *\n * @param tiles - An array of one or more tile source URLs, as in the raster tiles spec (See the [Style Specification](https://maplibre.org/maplibre-style-spec/)\n * @returns `this`\n */\n setTiles(tiles: Array): this {\n this.setSourceProperty(() => {\n this._options.tiles = tiles;\n });\n\n return this;\n }\n\n serialize() {\n return extend({}, this._options);\n }\n\n hasTile(tileID: OverscaledTileID) {\n return !this.tileBounds || this.tileBounds.contains(tileID.canonical);\n }\n\n loadTile(tile: Tile, callback: Callback) {\n const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme);\n tile.request = ImageRequest.getImage(this.map._requestManager.transformRequest(url, ResourceType.Tile), (err, img, expiry) => {\n delete tile.request;\n\n if (tile.aborted) {\n tile.state = 'unloaded';\n callback(null);\n } else if (err) {\n tile.state = 'errored';\n callback(err);\n } else if (img) {\n if (this.map._refreshExpiredTiles && expiry) tile.setExpiryData(expiry);\n\n const context = this.map.painter.context;\n const gl = context.gl;\n tile.texture = this.map.painter.getTileTexture(img.width);\n if (tile.texture) {\n tile.texture.update(img, {useMipmap: true});\n } else {\n tile.texture = new Texture(context, img, gl.RGBA, {useMipmap: true});\n tile.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST);\n\n if (context.extTextureFilterAnisotropic) {\n gl.texParameterf(gl.TEXTURE_2D, context.extTextureFilterAnisotropic.TEXTURE_MAX_ANISOTROPY_EXT, context.extTextureFilterAnisotropicMax);\n }\n }\n\n tile.state = 'loaded';\n\n callback(null);\n }\n }, this.map._refreshExpiredTiles);\n }\n\n abortTile(tile: Tile, callback: Callback) {\n if (tile.request) {\n tile.request.cancel();\n delete tile.request;\n }\n callback();\n }\n\n unloadTile(tile: Tile, callback: Callback) {\n if (tile.texture) this.map.painter.saveTileTexture(tile.texture);\n callback();\n }\n\n hasTransition() {\n return false;\n }\n}\n","import {ImageRequest} from '../util/image_request';\nimport {ResourceType} from '../util/request_manager';\nimport {extend, isImageBitmap, readImageUsingVideoFrame} from '../util/util';\nimport {Evented} from '../util/evented';\nimport {browser} from '../util/browser';\nimport {offscreenCanvasSupported} from '../util/offscreen_canvas_supported';\nimport {OverscaledTileID} from './tile_id';\nimport {RasterTileSource} from './raster_tile_source';\n// ensure DEMData is registered for worker transfer on main thread:\nimport '../data/dem_data';\nimport type {DEMEncoding} from '../data/dem_data';\n\nimport type {Source} from './source';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {RasterDEMSourceSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {ExpiryData} from '../util/ajax';\nimport {isOffscreenCanvasDistorted} from '../util/offscreen_canvas_distorted';\nimport {RGBAImage} from '../util/image';\n\n/**\n * A source containing raster DEM tiles (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.)\n * This source can be used to show hillshading and 3D terrain\n *\n * @group Sources\n *\n * @example\n * ```ts\n * map.addSource('raster-dem-source', {\n * type: 'raster-dem',\n * url: 'https://demotiles.maplibre.org/terrain-tiles/tiles.json',\n * tileSize: 256\n * });\n * ```\n * @see [3D Terrain](https://maplibre.org/maplibre-gl-js/docs/examples/3d-terrain/)\n */\nexport class RasterDEMTileSource extends RasterTileSource implements Source {\n encoding: DEMEncoding;\n redFactor?: number;\n greenFactor?: number;\n blueFactor?: number;\n baseShift?: number;\n\n constructor(id: string, options: RasterDEMSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super(id, options, dispatcher, eventedParent);\n this.type = 'raster-dem';\n this.maxzoom = 22;\n this._options = extend({type: 'raster-dem'}, options);\n this.encoding = options.encoding || 'mapbox';\n this.redFactor = options.redFactor;\n this.greenFactor = options.greenFactor;\n this.blueFactor = options.blueFactor;\n this.baseShift = options.baseShift;\n }\n\n loadTile(tile: Tile, callback: Callback) {\n const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme);\n const request = this.map._requestManager.transformRequest(url, ResourceType.Tile);\n tile.neighboringTiles = this._getNeighboringTiles(tile.tileID);\n tile.request = ImageRequest.getImage(request, async (err: Error, img: (HTMLImageElement | ImageBitmap), expiry: ExpiryData) => {\n delete tile.request;\n if (tile.aborted) {\n tile.state = 'unloaded';\n callback(null);\n } else if (err) {\n tile.state = 'errored';\n callback(err);\n } else if (img) {\n if (this.map._refreshExpiredTiles) tile.setExpiryData(expiry);\n const transfer = isImageBitmap(img) && offscreenCanvasSupported();\n const rawImageData = transfer ? img : await readImageNow(img);\n const params = {\n uid: tile.uid,\n coord: tile.tileID,\n source: this.id,\n rawImageData,\n encoding: this.encoding,\n redFactor: this.redFactor,\n greenFactor: this.greenFactor,\n blueFactor: this.blueFactor,\n baseShift: this.baseShift\n };\n\n if (!tile.actor || tile.state === 'expired') {\n tile.actor = this.dispatcher.getActor();\n tile.actor.send('loadDEMTile', params, done);\n }\n }\n }, this.map._refreshExpiredTiles);\n\n async function readImageNow(img: ImageBitmap | HTMLImageElement): Promise {\n if (typeof VideoFrame !== 'undefined' && isOffscreenCanvasDistorted()) {\n const width = img.width + 2;\n const height = img.height + 2;\n try {\n return new RGBAImage({width, height}, await readImageUsingVideoFrame(img, -1, -1, width, height));\n } catch (e) {\n // fall-back to browser canvas decoding\n }\n }\n return browser.getImageData(img, 1);\n }\n\n function done(err, data) {\n if (err) {\n tile.state = 'errored';\n callback(err);\n }\n\n if (data) {\n tile.dem = data;\n tile.needsHillshadePrepare = true;\n tile.needsTerrainPrepare = true;\n tile.state = 'loaded';\n callback(null);\n }\n }\n }\n\n _getNeighboringTiles(tileID: OverscaledTileID) {\n const canonical = tileID.canonical;\n const dim = Math.pow(2, canonical.z);\n\n const px = (canonical.x - 1 + dim) % dim;\n const pxw = canonical.x === 0 ? tileID.wrap - 1 : tileID.wrap;\n const nx = (canonical.x + 1 + dim) % dim;\n const nxw = canonical.x + 1 === dim ? tileID.wrap + 1 : tileID.wrap;\n\n const neighboringTiles = {};\n // add adjacent tiles\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y).key] = {backfilled: false};\n\n // Add upper neighboringTiles\n if (canonical.y > 0) {\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y - 1).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, tileID.wrap, canonical.z, canonical.x, canonical.y - 1).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y - 1).key] = {backfilled: false};\n }\n // Add lower neighboringTiles\n if (canonical.y + 1 < dim) {\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y + 1).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, tileID.wrap, canonical.z, canonical.x, canonical.y + 1).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y + 1).key] = {backfilled: false};\n }\n\n return neighboringTiles;\n }\n\n unloadTile(tile: Tile) {\n if (tile.demTexture) this.map.painter.saveTileTexture(tile.demTexture);\n if (tile.fbo) {\n tile.fbo.destroy();\n delete tile.fbo;\n }\n if (tile.dem) delete tile.dem;\n delete tile.neighboringTiles;\n\n tile.state = 'unloaded';\n if (tile.actor) {\n tile.actor.send('removeDEMTile', {uid: tile.uid, source: this.id});\n }\n }\n}\n","import {Event, ErrorEvent, Evented} from '../util/evented';\n\nimport {extend} from '../util/util';\nimport {EXTENT} from '../data/extent';\nimport {ResourceType} from '../util/request_manager';\nimport {browser} from '../util/browser';\n\nimport type {Source} from './source';\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Actor} from '../util/actor';\nimport type {Callback} from '../types/callback';\nimport type {GeoJSONSourceSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {GeoJSONSourceDiff} from './geojson_source_diff';\nimport type {Options, ClusterProperties} from 'supercluster';\n\nexport type GeoJSONSourceOptions = GeoJSONSourceSpecification & {\n workerOptions?: WorkerOptions;\n collectResourceTiming?: boolean;\n}\n\nexport type GeoJsonSourceOptions = {\n data?: GeoJSON.GeoJSON | string | undefined;\n cluster?: boolean;\n clusterMaxZoom?: number;\n clusterRadius?: number;\n clusterMinPoints?: number;\n generateId?: boolean;\n}\nexport type WorkerOptions = {\n source?: string;\n cluster?: boolean;\n geojsonVtOptions?: {\n buffer?: number;\n tolerance?: number;\n extent?: number;\n maxZoom?: number;\n linemetrics?: boolean;\n generateId?: boolean;\n };\n superclusterOptions?: Options;\n clusterProperties?: ClusterProperties;\n fliter?: any;\n promoteId?: any;\n collectResourceTiming?: boolean;\n}\n\n/**\n * The cluster options to set\n */\nexport type SetClusterOptions = {\n /**\n * Whether or not to cluster\n */\n cluster?: boolean;\n /**\n * The cluster's max zoom\n */\n clusterMaxZoom?: number;\n /**\n * The cluster's radius\n */\n clusterRadius?: number;\n}\n\n/**\n * A source containing GeoJSON.\n * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-geojson) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * map.addSource('some id', {\n * type: 'geojson',\n * data: 'https://d2ad6b4ur7yvpq.cloudfront.net/naturalearth-3.3.0/ne_10m_ports.geojson'\n * });\n * ```\n *\n * @example\n * ```ts\n * map.addSource('some id', {\n * type: 'geojson',\n * data: {\n * \"type\": \"FeatureCollection\",\n * \"features\": [{\n * \"type\": \"Feature\",\n * \"properties\": {},\n * \"geometry\": {\n * \"type\": \"Point\",\n * \"coordinates\": [\n * -76.53063297271729,\n * 39.18174077994108\n * ]\n * }\n * }]\n * }\n * });\n * ```\n *\n * @example\n * ```ts\n * map.getSource('some id').setData({\n * \"type\": \"FeatureCollection\",\n * \"features\": [{\n * \"type\": \"Feature\",\n * \"properties\": { \"name\": \"Null Island\" },\n * \"geometry\": {\n * \"type\": \"Point\",\n * \"coordinates\": [ 0, 0 ]\n * }\n * }]\n * });\n * ```\n * @see [Draw GeoJSON points](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-markers/)\n * @see [Add a GeoJSON line](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-line/)\n * @see [Create a heatmap from points](https://maplibre.org/maplibre-gl-js/docs/examples/heatmap/)\n * @see [Create and style clusters](https://maplibre.org/maplibre-gl-js/docs/examples/cluster/)\n */\nexport class GeoJSONSource extends Evented implements Source {\n type: 'geojson';\n id: string;\n minzoom: number;\n maxzoom: number;\n tileSize: number;\n attribution: string;\n promoteId: PromoteIdSpecification;\n\n isTileClipped: boolean;\n reparseOverscaled: boolean;\n _data: GeoJSON.GeoJSON | string | undefined;\n _options: GeoJsonSourceOptions;\n workerOptions: WorkerOptions;\n map: Map;\n actor: Actor;\n _pendingLoads: number;\n _collectResourceTiming: boolean;\n _removed: boolean;\n\n /** @internal */\n constructor(id: string, options: GeoJSONSourceOptions, dispatcher: Dispatcher, eventedParent: Evented) {\n super();\n\n this.id = id;\n\n // `type` is a property rather than a constant to make it easy for 3rd\n // parties to use GeoJSONSource to build their own source types.\n this.type = 'geojson';\n\n this.minzoom = 0;\n this.maxzoom = 18;\n this.tileSize = 512;\n this.isTileClipped = true;\n this.reparseOverscaled = true;\n this._removed = false;\n this._pendingLoads = 0;\n\n this.actor = dispatcher.getActor();\n this.setEventedParent(eventedParent);\n\n this._data = (options.data as any);\n this._options = extend({}, options);\n\n this._collectResourceTiming = options.collectResourceTiming;\n\n if (options.maxzoom !== undefined) this.maxzoom = options.maxzoom;\n if (options.type) this.type = options.type;\n if (options.attribution) this.attribution = options.attribution;\n this.promoteId = options.promoteId;\n\n const scale = EXTENT / this.tileSize;\n\n // sent to the worker, along with `url: ...` or `data: literal geojson`,\n // so that it can load/parse/index the geojson data\n // extending with `options.workerOptions` helps to make it easy for\n // third-party sources to hack/reuse GeoJSONSource.\n this.workerOptions = extend({\n source: this.id,\n cluster: options.cluster || false,\n geojsonVtOptions: {\n buffer: (options.buffer !== undefined ? options.buffer : 128) * scale,\n tolerance: (options.tolerance !== undefined ? options.tolerance : 0.375) * scale,\n extent: EXTENT,\n maxZoom: this.maxzoom,\n lineMetrics: options.lineMetrics || false,\n generateId: options.generateId || false\n },\n superclusterOptions: {\n maxZoom: options.clusterMaxZoom !== undefined ? options.clusterMaxZoom : this.maxzoom - 1,\n minPoints: Math.max(2, options.clusterMinPoints || 2),\n extent: EXTENT,\n radius: (options.clusterRadius || 50) * scale,\n log: false,\n generateId: options.generateId || false\n },\n clusterProperties: options.clusterProperties,\n filter: options.filter\n }, options.workerOptions);\n\n // send the promoteId to the worker to have more flexible updates, but only if it is a string\n if (typeof this.promoteId === 'string') {\n this.workerOptions.promoteId = this.promoteId;\n }\n }\n\n load = () => {\n this._updateWorkerData();\n };\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n }\n\n /**\n * Sets the GeoJSON data and re-renders the map.\n *\n * @param data - A GeoJSON data object or a URL to one. The latter is preferable in the case of large GeoJSON files.\n * @returns `this`\n */\n setData(data: GeoJSON.GeoJSON | string): this {\n this._data = data;\n this._updateWorkerData();\n\n return this;\n }\n\n /**\n * Updates the source's GeoJSON, and re-renders the map.\n *\n * For sources with lots of features, this method can be used to make updates more quickly.\n *\n * This approach requires unique IDs for every feature in the source. The IDs can either be specified on the feature,\n * or by using the promoteId option to specify which property should be used as the ID.\n *\n * It is an error to call updateData on a source that did not have unique IDs for each of its features already.\n *\n * Updates are applied on a best-effort basis, updating an ID that does not exist will not result in an error.\n *\n * @param diff - The changes that need to be applied.\n * @returns `this`\n */\n updateData(diff: GeoJSONSourceDiff): this {\n this._updateWorkerData(diff);\n\n return this;\n }\n\n /**\n * To disable/enable clustering on the source options\n * @param options - The options to set\n * @returns `this`\n * @example\n * ```ts\n * map.getSource('some id').setClusterOptions({cluster: false});\n * map.getSource('some id').setClusterOptions({cluster: false, clusterRadius: 50, clusterMaxZoom: 14});\n * ```\n */\n setClusterOptions(options: SetClusterOptions): this {\n this.workerOptions.cluster = options.cluster;\n if (options) {\n if (options.clusterRadius !== undefined) this.workerOptions.superclusterOptions.radius = options.clusterRadius;\n if (options.clusterMaxZoom !== undefined) this.workerOptions.superclusterOptions.maxZoom = options.clusterMaxZoom;\n }\n this._updateWorkerData();\n return this;\n }\n\n /**\n * For clustered sources, fetches the zoom at which the given cluster expands.\n *\n * @param clusterId - The value of the cluster's `cluster_id` property.\n * @param callback - A callback to be called when the zoom value is retrieved (`(error, zoom) => { ... }`).\n * @returns `this`\n */\n getClusterExpansionZoom(clusterId: number, callback: Callback): this {\n this.actor.send('geojson.getClusterExpansionZoom', {clusterId, source: this.id}, callback);\n return this;\n }\n\n /**\n * For clustered sources, fetches the children of the given cluster on the next zoom level (as an array of GeoJSON features).\n *\n * @param clusterId - The value of the cluster's `cluster_id` property.\n * @param callback - A callback to be called when the features are retrieved (`(error, features) => { ... }`).\n * @returns `this`\n */\n getClusterChildren(clusterId: number, callback: Callback>): this {\n this.actor.send('geojson.getClusterChildren', {clusterId, source: this.id}, callback);\n return this;\n }\n\n /**\n * For clustered sources, fetches the original points that belong to the cluster (as an array of GeoJSON features).\n *\n * @param clusterId - The value of the cluster's `cluster_id` property.\n * @param limit - The maximum number of features to return.\n * @param offset - The number of features to skip (e.g. for pagination).\n * @param callback - A callback to be called when the features are retrieved (`(error, features) => { ... }`).\n * @returns `this`\n * @example\n * Retrieve cluster leaves on click\n * ```ts\n * map.on('click', 'clusters', function(e) {\n * let features = map.queryRenderedFeatures(e.point, {\n * layers: ['clusters']\n * });\n *\n * let clusterId = features[0].properties.cluster_id;\n * let pointCount = features[0].properties.point_count;\n * let clusterSource = map.getSource('clusters');\n *\n * clusterSource.getClusterLeaves(clusterId, pointCount, 0, function(error, features) {\n * // Print cluster leaves in the console\n * console.log('Cluster leaves:', error, features);\n * })\n * });\n * ```\n */\n getClusterLeaves(clusterId: number, limit: number, offset: number, callback: Callback>): this {\n this.actor.send('geojson.getClusterLeaves', {\n source: this.id,\n clusterId,\n limit,\n offset\n }, callback);\n return this;\n }\n\n /**\n * Responsible for invoking WorkerSource's geojson.loadData target, which\n * handles loading the geojson data and preparing to serve it up as tiles,\n * using geojson-vt or supercluster as appropriate.\n * @param diff - the diff object\n */\n _updateWorkerData(diff?: GeoJSONSourceDiff) {\n const options = extend({}, this.workerOptions);\n if (diff) {\n options.dataDiff = diff;\n } else if (typeof this._data === 'string') {\n options.request = this.map._requestManager.transformRequest(browser.resolveURL(this._data as string), ResourceType.Source);\n options.request.collectResourceTiming = this._collectResourceTiming;\n } else {\n options.data = JSON.stringify(this._data);\n }\n\n this._pendingLoads++;\n this.fire(new Event('dataloading', {dataType: 'source'}));\n\n // target {this.type}.loadData rather than literally geojson.loadData,\n // so that other geojson-like source types can easily reuse this\n // implementation\n this.actor.send(`${this.type}.loadData`, options, (err, result) => {\n this._pendingLoads--;\n\n if (this._removed || (result && result.abandoned)) {\n this.fire(new Event('dataabort', {dataType: 'source'}));\n return;\n }\n\n let resourceTiming = null;\n if (result && result.resourceTiming && result.resourceTiming[this.id])\n resourceTiming = result.resourceTiming[this.id].slice(0);\n\n if (err) {\n this.fire(new ErrorEvent(err));\n return;\n }\n\n const data: any = {dataType: 'source'};\n if (this._collectResourceTiming && resourceTiming && resourceTiming.length > 0)\n extend(data, {resourceTiming});\n\n // although GeoJSON sources contain no metadata, we fire this event to let the SourceCache\n // know its ok to start requesting tiles.\n this.fire(new Event('data', {...data, sourceDataType: 'metadata'}));\n this.fire(new Event('data', {...data, sourceDataType: 'content'}));\n });\n }\n\n loaded(): boolean {\n return this._pendingLoads === 0;\n }\n\n loadTile(tile: Tile, callback: Callback) {\n const message = !tile.actor ? 'loadTile' : 'reloadTile';\n tile.actor = this.actor;\n const params = {\n type: this.type,\n uid: tile.uid,\n tileID: tile.tileID,\n zoom: tile.tileID.overscaledZ,\n maxZoom: this.maxzoom,\n tileSize: this.tileSize,\n source: this.id,\n pixelRatio: this.map.getPixelRatio(),\n showCollisionBoxes: this.map.showCollisionBoxes,\n promoteId: this.promoteId\n };\n\n tile.request = this.actor.send(message, params, (err, data) => {\n delete tile.request;\n tile.unloadVectorData();\n\n if (tile.aborted) {\n return callback(null);\n }\n\n if (err) {\n return callback(err);\n }\n\n tile.loadVectorData(data, this.map.painter, message === 'reloadTile');\n\n return callback(null);\n });\n }\n\n abortTile(tile: Tile) {\n if (tile.request) {\n tile.request.cancel();\n delete tile.request;\n }\n tile.aborted = true;\n }\n\n unloadTile(tile: Tile) {\n tile.unloadVectorData();\n this.actor.send('removeTile', {uid: tile.uid, type: this.type, source: this.id});\n }\n\n onRemove() {\n this._removed = true;\n this.actor.send('removeSource', {type: this.type, source: this.id});\n }\n\n serialize = (): GeoJSONSourceSpecification => {\n return extend({}, this._options, {\n type: this.type,\n data: this._data\n });\n };\n\n hasTransition() {\n return false;\n }\n}\n","import {createLayout} from '../util/struct_array';\n\nexport default createLayout([\n {name: 'a_pos', type: 'Int16', components: 2},\n {name: 'a_texture_pos', type: 'Int16', components: 2}\n]);\n","import {CanonicalTileID} from './tile_id';\nimport {Event, ErrorEvent, Evented} from '../util/evented';\nimport {ImageRequest} from '../util/image_request';\nimport {ResourceType} from '../util/request_manager';\nimport {EXTENT} from '../data/extent';\nimport {RasterBoundsArray} from '../data/array_types.g';\nimport rasterBoundsAttributes from '../data/raster_bounds_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {Texture} from '../render/texture';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\n\nimport type {Source} from './source';\nimport type {CanvasSourceSpecification} from './canvas_source';\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {\n ImageSourceSpecification,\n VideoSourceSpecification\n} from '@maplibre/maplibre-gl-style-spec';\nimport {Cancelable} from '../types/cancelable';\n\n/**\n * Four geographical coordinates,\n * represented as arrays of longitude and latitude numbers, which define the corners of the image.\n * The coordinates start at the top left corner of the image and proceed in clockwise order.\n * They do not have to represent a rectangle.\n */\nexport type Coordinates = [[number, number], [number, number], [number, number], [number, number]];\n\n/**\n * The options object for the {@link ImageSource#updateImage} method\n */\nexport type UpdateImageOptions = {\n /**\n * Required image URL.\n */\n url: string;\n /**\n * The image coordinates\n */\n coordinates?: Coordinates;\n}\n\n/**\n * A data source containing an image.\n * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-image) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * // add to map\n * map.addSource('some id', {\n * type: 'image',\n * url: 'https://www.maplibre.org/images/foo.png',\n * coordinates: [\n * [-76.54, 39.18],\n * [-76.52, 39.18],\n * [-76.52, 39.17],\n * [-76.54, 39.17]\n * ]\n * });\n *\n * // update coordinates\n * let mySource = map.getSource('some id');\n * mySource.setCoordinates([\n * [-76.54335737228394, 39.18579907229748],\n * [-76.52803659439087, 39.1838364847587],\n * [-76.5295386314392, 39.17683392507606],\n * [-76.54520273208618, 39.17876344106642]\n * ]);\n *\n * // update url and coordinates simultaneously\n * mySource.updateImage({\n * url: 'https://www.maplibre.org/images/bar.png',\n * coordinates: [\n * [-76.54335737228394, 39.18579907229748],\n * [-76.52803659439087, 39.1838364847587],\n * [-76.5295386314392, 39.17683392507606],\n * [-76.54520273208618, 39.17876344106642]\n * ]\n * })\n *\n * map.removeSource('some id'); // remove\n * ```\n */\nexport class ImageSource extends Evented implements Source {\n type: string;\n id: string;\n minzoom: number;\n maxzoom: number;\n tileSize: number;\n url: string;\n\n coordinates: Coordinates;\n tiles: {[_: string]: Tile};\n options: any;\n dispatcher: Dispatcher;\n map: Map;\n texture: Texture | null;\n image: HTMLImageElement | ImageBitmap;\n tileID: CanonicalTileID;\n _boundsArray: RasterBoundsArray;\n boundsBuffer: VertexBuffer;\n boundsSegments: SegmentVector;\n _loaded: boolean;\n _request: Cancelable;\n\n /** @internal */\n constructor(id: string, options: ImageSourceSpecification | VideoSourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super();\n this.id = id;\n this.dispatcher = dispatcher;\n this.coordinates = options.coordinates;\n\n this.type = 'image';\n this.minzoom = 0;\n this.maxzoom = 22;\n this.tileSize = 512;\n this.tiles = {};\n this._loaded = false;\n\n this.setEventedParent(eventedParent);\n\n this.options = options;\n }\n\n load = (newCoordinates?: Coordinates, successCallback?: () => void) => {\n this._loaded = false;\n this.fire(new Event('dataloading', {dataType: 'source'}));\n\n this.url = this.options.url;\n\n this._request = ImageRequest.getImage(this.map._requestManager.transformRequest(this.url, ResourceType.Image), (err, image) => {\n this._request = null;\n this._loaded = true;\n\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (image) {\n this.image = image;\n if (newCoordinates) {\n this.coordinates = newCoordinates;\n }\n if (successCallback) {\n successCallback();\n }\n this._finishLoading();\n }\n });\n };\n\n loaded(): boolean {\n return this._loaded;\n }\n\n /**\n * Updates the image URL and, optionally, the coordinates. To avoid having the image flash after changing,\n * set the `raster-fade-duration` paint property on the raster layer to 0.\n *\n * @param options - The options object.\n * @returns `this`\n */\n updateImage(options: UpdateImageOptions): this {\n if (!options.url) {\n return this;\n }\n\n if (this._request) {\n this._request.cancel();\n this._request = null;\n }\n\n this.options.url = options.url;\n this.load(options.coordinates, () => { this.texture = null; });\n return this;\n }\n\n _finishLoading() {\n if (this.map) {\n this.setCoordinates(this.coordinates);\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'}));\n }\n }\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n }\n\n onRemove() {\n if (this._request) {\n this._request.cancel();\n this._request = null;\n }\n }\n\n /**\n * Sets the image's coordinates and re-renders the map.\n *\n * @param coordinates - Four geographical coordinates,\n * represented as arrays of longitude and latitude numbers, which define the corners of the image.\n * The coordinates start at the top left corner of the image and proceed in clockwise order.\n * They do not have to represent a rectangle.\n * @returns `this`\n */\n setCoordinates(coordinates: Coordinates): this {\n this.coordinates = coordinates;\n\n // Calculate which mercator tile is suitable for rendering the video in\n // and create a buffer with the corner coordinates. These coordinates\n // may be outside the tile, because raster tiles aren't clipped when rendering.\n\n // transform the geo coordinates into (zoom 0) tile space coordinates\n const cornerCoords = coordinates.map(MercatorCoordinate.fromLngLat);\n\n // Compute the coordinates of the tile we'll use to hold this image's\n // render data\n this.tileID = getCoordinatesCenterTileID(cornerCoords);\n\n // Constrain min/max zoom to our tile's zoom level in order to force\n // SourceCache to request this tile (no matter what the map's zoom\n // level)\n this.minzoom = this.maxzoom = this.tileID.z;\n\n // Transform the corner coordinates into the coordinate space of our\n // tile.\n const tileCoords = cornerCoords.map((coord) => this.tileID.getTilePoint(coord)._round());\n\n this._boundsArray = new RasterBoundsArray();\n this._boundsArray.emplaceBack(tileCoords[0].x, tileCoords[0].y, 0, 0);\n this._boundsArray.emplaceBack(tileCoords[1].x, tileCoords[1].y, EXTENT, 0);\n this._boundsArray.emplaceBack(tileCoords[3].x, tileCoords[3].y, 0, EXTENT);\n this._boundsArray.emplaceBack(tileCoords[2].x, tileCoords[2].y, EXTENT, EXTENT);\n\n if (this.boundsBuffer) {\n this.boundsBuffer.destroy();\n delete this.boundsBuffer;\n }\n\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'}));\n return this;\n }\n\n prepare = () => {\n if (Object.keys(this.tiles).length === 0 || !this.image) {\n return;\n }\n\n const context = this.map.painter.context;\n const gl = context.gl;\n\n if (!this.boundsBuffer) {\n this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members);\n }\n\n if (!this.boundsSegments) {\n this.boundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n }\n\n if (!this.texture) {\n this.texture = new Texture(context, this.image, gl.RGBA);\n this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n }\n\n let newTilesLoaded = false;\n for (const w in this.tiles) {\n const tile = this.tiles[w];\n if (tile.state !== 'loaded') {\n tile.state = 'loaded';\n tile.texture = this.texture;\n newTilesLoaded = true;\n }\n }\n\n if (newTilesLoaded) {\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'idle', sourceId: this.id}));\n }\n };\n\n loadTile(tile: Tile, callback: Callback) {\n // We have a single tile -- whose coordinates are this.tileID -- that\n // covers the image we want to render. If that's the one being\n // requested, set it up with the image; otherwise, mark the tile as\n // `errored` to indicate that we have no data for it.\n // If the world wraps, we may have multiple \"wrapped\" copies of the\n // single tile.\n if (this.tileID && this.tileID.equals(tile.tileID.canonical)) {\n this.tiles[String(tile.tileID.wrap)] = tile;\n tile.buckets = {};\n callback(null);\n } else {\n tile.state = 'errored';\n callback(null);\n }\n }\n\n serialize = (): ImageSourceSpecification | VideoSourceSpecification | CanvasSourceSpecification => {\n return {\n type: 'image',\n url: this.options.url,\n coordinates: this.coordinates\n };\n };\n\n hasTransition() {\n return false;\n }\n}\n\n/**\n * Given a list of coordinates, get their center as a coordinate.\n *\n * @returns centerpoint\n * @internal\n */\nexport function getCoordinatesCenterTileID(coords: Array) {\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n\n for (const coord of coords) {\n minX = Math.min(minX, coord.x);\n minY = Math.min(minY, coord.y);\n maxX = Math.max(maxX, coord.x);\n maxY = Math.max(maxY, coord.y);\n }\n\n const dx = maxX - minX;\n const dy = maxY - minY;\n const dMax = Math.max(dx, dy);\n const zoom = Math.max(0, Math.floor(-Math.log(dMax) / Math.LN2));\n const tilesAtZoom = Math.pow(2, zoom);\n\n return new CanonicalTileID(\n zoom,\n Math.floor((minX + maxX) / 2 * tilesAtZoom),\n Math.floor((minY + maxY) / 2 * tilesAtZoom));\n}\n","import {getVideo} from '../util/ajax';\nimport {ResourceType} from '../util/request_manager';\n\nimport {ImageSource} from './image_source';\nimport rasterBoundsAttributes from '../data/raster_bounds_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {Texture} from '../render/texture';\nimport {Event, ErrorEvent} from '../util/evented';\nimport {ValidationError} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Evented} from '../util/evented';\nimport type {VideoSourceSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A data source containing video.\n * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-video) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * // add to map\n * map.addSource('some id', {\n * type: 'video',\n * url: [\n * 'https://www.mapbox.com/blog/assets/baltimore-smoke.mp4',\n * 'https://www.mapbox.com/blog/assets/baltimore-smoke.webm'\n * ],\n * coordinates: [\n * [-76.54, 39.18],\n * [-76.52, 39.18],\n * [-76.52, 39.17],\n * [-76.54, 39.17]\n * ]\n * });\n *\n * // update\n * let mySource = map.getSource('some id');\n * mySource.setCoordinates([\n * [-76.54335737228394, 39.18579907229748],\n * [-76.52803659439087, 39.1838364847587],\n * [-76.5295386314392, 39.17683392507606],\n * [-76.54520273208618, 39.17876344106642]\n * ]);\n *\n * map.removeSource('some id'); // remove\n * ```\n * @see [Add a video](https://maplibre.org/maplibre-gl-js/docs/examples/video-on-a-map/)\n *\n * Note that when rendered as a raster layer, the layer's `raster-fade-duration` property will cause the video to fade in.\n * This happens when playback is started, paused and resumed, or when the video's coordinates are updated. To avoid this behavior,\n * set the layer's `raster-fade-duration` property to `0`.\n */\nexport class VideoSource extends ImageSource {\n options: VideoSourceSpecification;\n urls: Array;\n video: HTMLVideoElement;\n roundZoom: boolean;\n\n constructor(id: string, options: VideoSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super(id, options, dispatcher, eventedParent);\n this.roundZoom = true;\n this.type = 'video';\n this.options = options;\n }\n\n load = () => {\n this._loaded = false;\n const options = this.options;\n\n this.urls = [];\n for (const url of options.urls) {\n this.urls.push(this.map._requestManager.transformRequest(url, ResourceType.Source).url);\n }\n\n getVideo(this.urls, (err, video) => {\n this._loaded = true;\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (video) {\n this.video = video;\n this.video.loop = true;\n\n // Start repainting when video starts playing. hasTransition() will then return\n // true to trigger additional frames as long as the videos continues playing.\n this.video.addEventListener('playing', () => {\n this.map.triggerRepaint();\n });\n\n if (this.map) {\n this.video.play();\n }\n\n this._finishLoading();\n }\n });\n };\n\n /**\n * Pauses the video.\n */\n pause() {\n if (this.video) {\n this.video.pause();\n }\n }\n\n /**\n * Plays the video.\n */\n play() {\n if (this.video) {\n this.video.play();\n }\n }\n\n /**\n * Sets playback to a timestamp, in seconds.\n */\n seek(seconds: number) {\n if (this.video) {\n const seekableRange = this.video.seekable;\n if (seconds < seekableRange.start(0) || seconds > seekableRange.end(0)) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${this.id}`, null, `Playback for this video can be set only between the ${seekableRange.start(0)} and ${seekableRange.end(0)}-second mark.`)));\n } else this.video.currentTime = seconds;\n }\n }\n\n /**\n * Returns the HTML `video` element.\n *\n * @returns The HTML `video` element.\n */\n getVideo(): HTMLVideoElement {\n return this.video;\n }\n\n onAdd(map: Map) {\n if (this.map) return;\n this.map = map;\n this.load();\n if (this.video) {\n this.video.play();\n this.setCoordinates(this.coordinates);\n }\n }\n\n /**\n * Sets the video's coordinates and re-renders the map.\n *\n * @returns `this`\n */\n prepare = (): this => {\n if (Object.keys(this.tiles).length === 0 || this.video.readyState < 2) {\n return; // not enough data for current position\n }\n\n const context = this.map.painter.context;\n const gl = context.gl;\n\n if (!this.boundsBuffer) {\n this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members);\n }\n\n if (!this.boundsSegments) {\n this.boundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n }\n\n if (!this.texture) {\n this.texture = new Texture(context, this.video, gl.RGBA);\n this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n } else if (!this.video.paused) {\n this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, gl.RGBA, gl.UNSIGNED_BYTE, this.video);\n }\n\n let newTilesLoaded = false;\n for (const w in this.tiles) {\n const tile = this.tiles[w];\n if (tile.state !== 'loaded') {\n tile.state = 'loaded';\n tile.texture = this.texture;\n newTilesLoaded = true;\n }\n }\n\n if (newTilesLoaded) {\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'idle', sourceId: this.id}));\n }\n };\n\n serialize = (): VideoSourceSpecification => {\n return {\n type: 'video',\n urls: this.urls,\n coordinates: this.coordinates\n };\n };\n\n hasTransition() {\n return this.video && !this.video.paused;\n }\n}\n","import {ImageSource} from './image_source';\n\nimport rasterBoundsAttributes from '../data/raster_bounds_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {Texture} from '../render/texture';\nimport {Event, ErrorEvent} from '../util/evented';\nimport {ValidationError} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Evented} from '../util/evented';\n\n/**\n * Options to add a canvas source type to the map.\n */\nexport type CanvasSourceSpecification = {\n /**\n * Source type. Must be `\"canvas\"`.\n */\n type: 'canvas';\n /**\n * Four geographical coordinates denoting where to place the corners of the canvas, specified in `[longitude, latitude]` pairs.\n */\n coordinates: [[number, number], [number, number], [number, number], [number, number]];\n /**\n * Whether the canvas source is animated. If the canvas is static (i.e. pixels do not need to be re-read on every frame), `animate` should be set to `false` to improve performance.\n * @defaultValue true\n */\n animate?: boolean;\n /**\n * Canvas source from which to read pixels. Can be a string representing the ID of the canvas element, or the `HTMLCanvasElement` itself.\n */\n canvas?: string | HTMLCanvasElement;\n};\n\n/**\n * A data source containing the contents of an HTML canvas. See {@link CanvasSourceSpecification} for detailed documentation of options.\n *\n * @group Sources\n *\n * @example\n * ```ts\n * // add to map\n * map.addSource('some id', {\n * type: 'canvas',\n * canvas: 'idOfMyHTMLCanvas',\n * animate: true,\n * coordinates: [\n * [-76.54, 39.18],\n * [-76.52, 39.18],\n * [-76.52, 39.17],\n * [-76.54, 39.17]\n * ]\n * });\n *\n * // update\n * let mySource = map.getSource('some id');\n * mySource.setCoordinates([\n * [-76.54335737228394, 39.18579907229748],\n * [-76.52803659439087, 39.1838364847587],\n * [-76.5295386314392, 39.17683392507606],\n * [-76.54520273208618, 39.17876344106642]\n * ]);\n *\n * map.removeSource('some id'); // remove\n * ```\n */\nexport class CanvasSource extends ImageSource {\n options: CanvasSourceSpecification;\n animate: boolean;\n canvas: HTMLCanvasElement;\n width: number;\n height: number;\n /**\n * Enables animation. The image will be copied from the canvas to the map on each frame.\n */\n play: () => void;\n /**\n * Disables animation. The map will display a static copy of the canvas image.\n */\n pause: () => void;\n _playing: boolean;\n\n /** @internal */\n constructor(id: string, options: CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super(id, options, dispatcher, eventedParent);\n\n // We build in some validation here, since canvas sources aren't included in the style spec:\n if (!options.coordinates) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, 'missing required property \"coordinates\"')));\n } else if (!Array.isArray(options.coordinates) || options.coordinates.length !== 4 ||\n options.coordinates.some(c => !Array.isArray(c) || c.length !== 2 || c.some(l => typeof l !== 'number'))) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, '\"coordinates\" property must be an array of 4 longitude/latitude array pairs')));\n }\n\n if (options.animate && typeof options.animate !== 'boolean') {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, 'optional \"animate\" property must be a boolean value')));\n }\n\n if (!options.canvas) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, 'missing required property \"canvas\"')));\n } else if (typeof options.canvas !== 'string' && !(options.canvas instanceof HTMLCanvasElement)) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, '\"canvas\" must be either a string representing the ID of the canvas element from which to read, or an HTMLCanvasElement instance')));\n }\n\n this.options = options;\n this.animate = options.animate !== undefined ? options.animate : true;\n }\n\n load = () => {\n this._loaded = true;\n if (!this.canvas) {\n this.canvas = (this.options.canvas instanceof HTMLCanvasElement) ?\n this.options.canvas :\n document.getElementById(this.options.canvas) as HTMLCanvasElement;\n // cast to HTMLCanvasElement in else of ternary\n // should we do a safety check and throw if it's not actually HTMLCanvasElement?\n }\n this.width = this.canvas.width;\n this.height = this.canvas.height;\n\n if (this._hasInvalidDimensions()) {\n this.fire(new ErrorEvent(new Error('Canvas dimensions cannot be less than or equal to zero.')));\n return;\n }\n\n this.play = function() {\n this._playing = true;\n this.map.triggerRepaint();\n };\n\n this.pause = function() {\n if (this._playing) {\n this.prepare();\n this._playing = false;\n }\n };\n\n this._finishLoading();\n };\n\n /**\n * Returns the HTML `canvas` element.\n *\n * @returns The HTML `canvas` element.\n */\n getCanvas(): HTMLCanvasElement {\n return this.canvas;\n }\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n if (this.canvas) {\n if (this.animate) this.play();\n }\n }\n\n onRemove() {\n this.pause();\n }\n\n prepare = () => {\n let resize = false;\n if (this.canvas.width !== this.width) {\n this.width = this.canvas.width;\n resize = true;\n }\n if (this.canvas.height !== this.height) {\n this.height = this.canvas.height;\n resize = true;\n }\n\n if (this._hasInvalidDimensions()) return;\n\n if (Object.keys(this.tiles).length === 0) return; // not enough data for current position\n\n const context = this.map.painter.context;\n const gl = context.gl;\n\n if (!this.boundsBuffer) {\n this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members);\n }\n\n if (!this.boundsSegments) {\n this.boundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n }\n\n if (!this.texture) {\n this.texture = new Texture(context, this.canvas, gl.RGBA, {premultiply: true});\n } else if (resize || this._playing) {\n this.texture.update(this.canvas, {premultiply: true});\n }\n\n let newTilesLoaded = false;\n for (const w in this.tiles) {\n const tile = this.tiles[w];\n if (tile.state !== 'loaded') {\n tile.state = 'loaded';\n tile.texture = this.texture;\n newTilesLoaded = true;\n }\n }\n\n if (newTilesLoaded) {\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'idle', sourceId: this.id}));\n }\n };\n\n serialize = (): CanvasSourceSpecification => {\n return {\n type: 'canvas',\n coordinates: this.coordinates\n };\n };\n\n hasTransition() {\n return this._playing;\n }\n\n _hasInvalidDimensions() {\n for (const x of [this.canvas.width, this.canvas.height]) {\n if (isNaN(x) || x <= 0) return true;\n }\n return false;\n }\n}\n","import {VectorTileSource} from '../source/vector_tile_source';\nimport {RasterTileSource} from '../source/raster_tile_source';\nimport {RasterDEMTileSource} from '../source/raster_dem_tile_source';\nimport {GeoJSONSource} from '../source/geojson_source';\nimport {VideoSource} from '../source/video_source';\nimport {ImageSource} from '../source/image_source';\nimport {CanvasSource} from '../source/canvas_source';\n\nimport type {SourceSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Event, Evented} from '../util/evented';\nimport type {Map} from '../ui/map';\nimport type {Tile} from './tile';\nimport type {OverscaledTileID, CanonicalTileID} from './tile_id';\nimport type {Callback} from '../types/callback';\nimport type {CanvasSourceSpecification} from '../source/canvas_source';\n\nconst registeredSources = {} as {[key:string]: SourceClass};\n\n/**\n * The `Source` interface must be implemented by each source type, including \"core\" types (`vector`, `raster`,\n * `video`, etc.) and all custom, third-party types.\n *\n * @event `data` - Fired with `{dataType: 'source', sourceDataType: 'metadata'}` to indicate that any necessary metadata\n * has been loaded so that it's okay to call `loadTile`; and with `{dataType: 'source', sourceDataType: 'content'}`\n * to indicate that the source data has changed, so that any current caches should be flushed.\n *\n * @group Sources\n */\nexport interface Source {\n readonly type: string;\n /**\n * The id for the source. Must not be used by any existing source.\n */\n id: string;\n minzoom: number;\n maxzoom: number;\n tileSize: number;\n attribution?: string;\n /**\n * `true` if zoom levels are rounded to the nearest integer in the source data, `false` if they are floor-ed to the nearest integer.\n */\n roundZoom?: boolean;\n /**\n * `false` if tiles can be drawn outside their boundaries, `true` if they cannot.\n */\n isTileClipped?: boolean;\n tileID?: CanonicalTileID;\n /**\n * `true` if tiles should be sent back to the worker for each overzoomed zoom level, `false` if not.\n */\n reparseOverscaled?: boolean;\n vectorLayerIds?: Array;\n hasTransition(): boolean;\n loaded(): boolean;\n fire(event: Event): unknown;\n readonly onAdd?: (map: Map) => void;\n readonly onRemove?: (map: Map) => void;\n loadTile(tile: Tile, callback: Callback): void;\n readonly hasTile?: (tileID: OverscaledTileID) => boolean;\n readonly abortTile?: (tile: Tile, callback: Callback) => void;\n readonly unloadTile?: (tile: Tile, callback: Callback) => void;\n /**\n * @returns A plain (stringifiable) JS object representing the current state of the source.\n * Creating a source using the returned object as the `options` should result in a Source that is\n * equivalent to this one.\n */\n serialize(): any;\n readonly prepare?: () => void;\n}\n\n/**\n * A supporting type to the source definition\n */\ntype SourceStatics = {\n /*\n * An optional URL to a script which, when run by a Worker, registers a {@link WorkerSource}\n * implementation for this Source type by calling `self.registerWorkerSource(workerSource: WorkerSource)`.\n */\n workerSourceURL?: URL;\n};\n\n/**\n * A general definition of a {@link Source} class for factory usage\n */\nexport type SourceClass = {\n new (id: string, specification: SourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented): Source;\n} & SourceStatics;\n\n/**\n * Creates a tiled data source instance given an options object.\n *\n * @param id - The id for the source. Must not be used by any existing source.\n * @param specification - Source options, specific to the source type (except for `options.type`, which is always required).\n * @param source - A source definition object compliant with\n * [`maplibre-gl-style-spec`](https://maplibre.org/maplibre-style-spec/#sources) or, for a third-party source type,\n * with that type's requirements.\n * @param dispatcher - A {@link Dispatcher} instance, which can be used to send messages to the workers.\n * @returns a newly created source\n */\nexport const create = (id: string, specification: SourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented): Source => {\n\n const Class = getSourceType(specification.type);\n const source = new Class(id, specification, dispatcher, eventedParent);\n\n if (source.id !== id) {\n throw new Error(`Expected Source id to be ${id} instead of ${source.id}`);\n }\n\n return source;\n};\n\nexport const getSourceType = (name: string): SourceClass => {\n switch (name) {\n case 'geojson':\n return GeoJSONSource;\n case 'image':\n return ImageSource;\n case 'raster':\n return RasterTileSource;\n case 'raster-dem':\n return RasterDEMTileSource;\n case 'vector':\n return VectorTileSource;\n case 'video':\n return VideoSource;\n case 'canvas':\n return CanvasSource;\n }\n return registeredSources[name];\n};\n\nexport const setSourceType = (name: string, type: SourceClass) => {\n registeredSources[name] = type;\n};\n\nexport interface Actor {\n send(type: string, data: any, callback: Callback): void;\n}\n","import type {SourceCache} from './source_cache';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {CollisionIndex} from '../symbol/collision_index';\nimport type {Transform} from '../geo/transform';\nimport type {RetainedQueryData} from '../symbol/placement';\nimport type {FilterSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\nimport type Point from '@mapbox/point-geometry';\nimport {mat4} from 'gl-matrix';\n\n/**\n * Options to pass to query the map for the rendered features\n */\nexport type QueryRenderedFeaturesOptions = {\n /**\n * An array of [style layer IDs](https://maplibre.org/maplibre-style-spec/#layer-id) for the query to inspect.\n * Only features within these layers will be returned. If this parameter is undefined, all layers will be checked.\n */\n layers?: Array;\n /**\n * A [filter](https://maplibre.org/maplibre-style-spec/layers/#filter) to limit query results.\n */\n filter?: FilterSpecification;\n /**\n * An array of string representing the available images\n */\n availableImages?: Array;\n /**\n * Whether to check if the [options.filter] conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function.\n */\n validate?: boolean;\n};\n\n/**\n * The options object related to the {@link Map#queryRenderedFeatures} method\n */\nexport type QuerySourceFeatureOptions = {\n /**\n * The name of the source layer to query. *For vector tile sources, this parameter is required.* For GeoJSON sources, it is ignored.\n */\n sourceLayer?: string;\n /**\n * A [filter](https://maplibre.org/maplibre-style-spec/layers/#filter)\n * to limit query results.\n */\n filter?: FilterSpecification;\n /**\n * Whether to check if the [parameters.filter] conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function.\n * @defaultValue true\n */\n validate?: boolean;\n}\n\n/*\n * Returns a matrix that can be used to convert from tile coordinates to viewport pixel coordinates.\n */\nfunction getPixelPosMatrix(transform, tileID) {\n const t = mat4.create();\n mat4.translate(t, t, [1, 1, 0]);\n mat4.scale(t, t, [transform.width * 0.5, transform.height * 0.5, 1]);\n return mat4.multiply(t, t, transform.calculatePosMatrix(tileID.toUnwrapped()));\n}\n\nfunction queryIncludes3DLayer(layers: Array, styleLayers: {[_: string]: StyleLayer}, sourceID: string) {\n if (layers) {\n for (const layerID of layers) {\n const layer = styleLayers[layerID];\n if (layer && layer.source === sourceID && layer.type === 'fill-extrusion') {\n return true;\n }\n }\n } else {\n for (const key in styleLayers) {\n const layer = styleLayers[key];\n if (layer.source === sourceID && layer.type === 'fill-extrusion') {\n return true;\n }\n }\n }\n return false;\n}\n\nexport function queryRenderedFeatures(\n sourceCache: SourceCache,\n styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n queryGeometry: Array,\n params: QueryRenderedFeaturesOptions,\n transform: Transform\n): { [key: string]: Array<{featureIndex: number; feature: MapGeoJSONFeature}> } {\n\n const has3DLayer = queryIncludes3DLayer(params && params.layers, styleLayers, sourceCache.id);\n const maxPitchScaleFactor = transform.maxPitchScaleFactor();\n const tilesIn = sourceCache.tilesIn(queryGeometry, maxPitchScaleFactor, has3DLayer);\n\n tilesIn.sort(sortTilesIn);\n const renderedFeatureLayers = [];\n for (const tileIn of tilesIn) {\n renderedFeatureLayers.push({\n wrappedTileID: tileIn.tileID.wrapped().key,\n queryResults: tileIn.tile.queryRenderedFeatures(\n styleLayers,\n serializedLayers,\n sourceCache._state,\n tileIn.queryGeometry,\n tileIn.cameraQueryGeometry,\n tileIn.scale,\n params,\n transform,\n maxPitchScaleFactor,\n getPixelPosMatrix(sourceCache.transform, tileIn.tileID))\n });\n }\n\n const result = mergeRenderedFeatureLayers(renderedFeatureLayers);\n\n // Merge state from SourceCache into the results\n for (const layerID in result) {\n result[layerID].forEach((featureWrapper) => {\n const feature = featureWrapper.feature as MapGeoJSONFeature;\n const state = sourceCache.getFeatureState(feature.layer['source-layer'], feature.id);\n feature.source = feature.layer.source;\n if (feature.layer['source-layer']) {\n feature.sourceLayer = feature.layer['source-layer'];\n }\n feature.state = state;\n });\n }\n return result;\n}\n\nexport function queryRenderedSymbols(styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: StyleLayer},\n sourceCaches: {[_: string]: SourceCache},\n queryGeometry: Array,\n params: QueryRenderedFeaturesOptions,\n collisionIndex: CollisionIndex,\n retainedQueryData: {\n [_: number]: RetainedQueryData;\n }) {\n const result = {};\n const renderedSymbols = collisionIndex.queryRenderedSymbols(queryGeometry);\n const bucketQueryData = [];\n for (const bucketInstanceId of Object.keys(renderedSymbols).map(Number)) {\n bucketQueryData.push(retainedQueryData[bucketInstanceId]);\n }\n bucketQueryData.sort(sortTilesIn);\n\n for (const queryData of bucketQueryData) {\n const bucketSymbols = queryData.featureIndex.lookupSymbolFeatures(\n renderedSymbols[queryData.bucketInstanceId],\n serializedLayers,\n queryData.bucketIndex,\n queryData.sourceLayerIndex,\n params.filter,\n params.layers,\n params.availableImages,\n styleLayers);\n\n for (const layerID in bucketSymbols) {\n const resultFeatures = result[layerID] = result[layerID] || [];\n const layerSymbols = bucketSymbols[layerID];\n layerSymbols.sort((a, b) => {\n // Match topDownFeatureComparator from FeatureIndex, but using\n // most recent sorting of features from bucket.sortFeatures\n const featureSortOrder = queryData.featureSortOrder;\n if (featureSortOrder) {\n // queryRenderedSymbols documentation says we'll return features in\n // \"top-to-bottom\" rendering order (aka last-to-first).\n // Actually there can be multiple symbol instances per feature, so\n // we sort each feature based on the first matching symbol instance.\n const sortedA = featureSortOrder.indexOf(a.featureIndex);\n const sortedB = featureSortOrder.indexOf(b.featureIndex);\n return sortedB - sortedA;\n } else {\n // Bucket hasn't been re-sorted based on angle, so use the\n // reverse of the order the features appeared in the data.\n return b.featureIndex - a.featureIndex;\n }\n });\n for (const symbolFeature of layerSymbols) {\n resultFeatures.push(symbolFeature);\n }\n }\n }\n\n // Merge state from SourceCache into the results\n for (const layerName in result) {\n result[layerName].forEach((featureWrapper) => {\n const feature = featureWrapper.feature;\n const layer = styleLayers[layerName];\n const sourceCache = sourceCaches[layer.source];\n const state = sourceCache.getFeatureState(feature.layer['source-layer'], feature.id);\n feature.source = feature.layer.source;\n if (feature.layer['source-layer']) {\n feature.sourceLayer = feature.layer['source-layer'];\n }\n feature.state = state;\n });\n }\n return result;\n}\n\nexport function querySourceFeatures(sourceCache: SourceCache, params: QuerySourceFeatureOptions) {\n const tiles = sourceCache.getRenderableIds().map((id) => {\n return sourceCache.getTileByID(id);\n });\n\n const result = [];\n\n const dataTiles = {};\n for (let i = 0; i < tiles.length; i++) {\n const tile = tiles[i];\n const dataID = tile.tileID.canonical.key;\n if (!dataTiles[dataID]) {\n dataTiles[dataID] = true;\n tile.querySourceFeatures(result, params);\n }\n }\n\n return result;\n}\n\nfunction sortTilesIn(a, b) {\n const idA = a.tileID;\n const idB = b.tileID;\n return (idA.overscaledZ - idB.overscaledZ) || (idA.canonical.y - idB.canonical.y) || (idA.wrap - idB.wrap) || (idA.canonical.x - idB.canonical.x);\n}\n\nfunction mergeRenderedFeatureLayers(tiles) {\n // Merge results from all tiles, but if two tiles share the same\n // wrapped ID, don't duplicate features between the two tiles\n const result = {};\n const wrappedIDLayerMap = {};\n for (const tile of tiles) {\n const queryResults = tile.queryResults;\n const wrappedID = tile.wrappedTileID;\n const wrappedIDLayers = wrappedIDLayerMap[wrappedID] = wrappedIDLayerMap[wrappedID] || {};\n for (const layerID in queryResults) {\n const tileFeatures = queryResults[layerID];\n const wrappedIDFeatures = wrappedIDLayers[layerID] = wrappedIDLayers[layerID] || {};\n const resultFeatures = result[layerID] = result[layerID] || [];\n for (const tileFeature of tileFeatures) {\n if (!wrappedIDFeatures[tileFeature.featureIndex]) {\n wrappedIDFeatures[tileFeature.featureIndex] = true;\n resultFeatures.push(tileFeature);\n }\n }\n }\n }\n return result;\n}\n","import type {CollisionBoxArray} from './array_types.g';\nimport type {Style} from '../style/style';\nimport type {TypedStyleLayer} from '../style/style_layer/typed_style_layer';\nimport type {FeatureIndex} from './feature_index';\nimport type {Context} from '../gl/context';\nimport type {FeatureStates} from '../source/source_state';\nimport type {ImagePosition} from '../render/image_atlas';\nimport type {CanonicalTileID} from '../source/tile_id';\nimport type {VectorTileFeature, VectorTileLayer} from '@mapbox/vector-tile';\nimport Point from '@mapbox/point-geometry';\n\nexport type BucketParameters = {\n index: number;\n layers: Array;\n zoom: number;\n pixelRatio: number;\n overscaling: number;\n collisionBoxArray: CollisionBoxArray;\n sourceLayerIndex: number;\n sourceID: string;\n};\n\nexport type PopulateParameters = {\n featureIndex: FeatureIndex;\n iconDependencies: {};\n patternDependencies: {};\n glyphDependencies: {};\n availableImages: Array;\n};\n\nexport type IndexedFeature = {\n feature: VectorTileFeature;\n id: number | string;\n index: number;\n sourceLayerIndex: number;\n};\n\nexport type BucketFeature = {\n index: number;\n sourceLayerIndex: number;\n geometry: Array>;\n properties: any;\n type: 0 | 1 | 2 | 3;\n id?: any;\n readonly patterns: {\n [_: string]: {\n 'min': string;\n 'mid': string;\n 'max': string;\n };\n };\n sortKey?: number;\n};\n\n/**\n * The `Bucket` interface is the single point of knowledge about turning vector\n * tiles into WebGL buffers.\n *\n * `Bucket` is an abstract interface. An implementation exists for each style layer type.\n * Create a bucket via the `StyleLayer#createBucket` method.\n *\n * The concrete bucket types, using layout options from the style layer,\n * transform feature geometries into vertex and index data for use by the\n * vertex shader. They also (via `ProgramConfiguration`) use feature\n * properties and the zoom level to populate the attributes needed for\n * data-driven styling.\n *\n * Buckets are designed to be built on a worker thread and then serialized and\n * transferred back to the main thread for rendering. On the worker side, a\n * bucket's vertex, index, and attribute data is stored in `bucket.arrays: ArrayGroup`.\n * When a bucket's data is serialized and sent back to the main thread,\n * is gets deserialized (using `new Bucket(serializedBucketData)`, with\n * the array data now stored in `bucket.buffers: BufferGroup`. BufferGroups\n * hold the same data as ArrayGroups, but are tuned for consumption by WebGL.\n */\nexport interface Bucket {\n layerIds: Array;\n hasPattern: boolean;\n readonly layers: Array;\n readonly stateDependentLayers: Array;\n readonly stateDependentLayerIds: Array;\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID): void;\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}): void;\n isEmpty(): boolean;\n upload(context: Context): void;\n uploadPending(): boolean;\n /**\n * Release the WebGL resources associated with the buffers. Note that because\n * buckets are shared between layers having the same layout properties, they\n * must be destroyed in groups (all buckets for a tile, or all symbol buckets).\n */\n destroy(): void;\n}\n\nexport function deserialize(input: Array, style: Style): {[_: string]: Bucket} {\n const output = {};\n\n // Guard against the case where the map's style has been set to null while\n // this bucket has been parsing.\n if (!style) return output;\n\n for (const bucket of input) {\n const layers = bucket.layerIds\n .map((id) => style.getLayer(id))\n .filter(Boolean);\n\n if (layers.length === 0) {\n continue;\n }\n\n // look up StyleLayer objects from layer ids (since we don't\n // want to waste time serializing/copying them from the worker)\n (bucket as any).layers = layers;\n if (bucket.stateDependentLayerIds) {\n (bucket as any).stateDependentLayers = bucket.stateDependentLayerIds.map((lId) => layers.filter((l) => l.id === lId)[0]);\n }\n for (const layer of layers) {\n output[layer.id] = bucket;\n }\n }\n\n return output;\n}\n","import {uniqueId, parseCacheControl} from '../util/util';\nimport {deserialize as deserializeBucket} from '../data/bucket';\nimport '../data/feature_index';\nimport type {FeatureIndex} from '../data/feature_index';\nimport {GeoJSONFeature} from '../util/vectortile_to_geojson';\nimport {featureFilter} from '@maplibre/maplibre-gl-style-spec';\nimport {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {CollisionBoxArray} from '../data/array_types.g';\nimport {Texture} from '../render/texture';\nimport {browser} from '../util/browser';\nimport {toEvaluationFeature} from '../data/evaluation_feature';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {SourceFeatureState} from '../source/source_state';\nimport {lazyLoadRTLTextPlugin} from './rtl_text_plugin';\n\nconst CLOCK_SKEW_RETRY_TIMEOUT = 30000;\n\nimport type {Bucket} from '../data/bucket';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {WorkerTileResult} from './worker_source';\nimport type {Actor} from '../util/actor';\nimport type {DEMData} from '../data/dem_data';\nimport type {AlphaImage} from '../util/image';\nimport type {ImageAtlas} from '../render/image_atlas';\nimport type {ImageManager} from '../render/image_manager';\nimport type {Context} from '../gl/context';\nimport type {OverscaledTileID} from './tile_id';\nimport type {Framebuffer} from '../gl/framebuffer';\nimport type {Transform} from '../geo/transform';\nimport type {LayerFeatureStates} from './source_state';\nimport type {Cancelable} from '../types/cancelable';\nimport type {FilterSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type Point from '@mapbox/point-geometry';\nimport {mat4} from 'gl-matrix';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\nimport {ExpiryData} from '../util/ajax';\n\n/**\n * The tile's state, can be:\n * - `loading` Tile data is in the process of loading.\n * - `loaded` Tile data has been loaded. Tile can be rendered.\n * - `reloading` Tile data has been loaded and is being updated. Tile can be rendered.\n * - `unloaded` Tile data has been deleted.\n * - `errored` Tile data was not loaded because of an error.\n * - `expired` Tile data was previously loaded, but has expired per its HTTP headers and is in the process of refreshing.\n */\nexport type TileState = 'loading' | 'loaded' | 'reloading' | 'unloaded' | 'errored' | 'expired';\n\n/**\n * @internal\n * A tile object is the combination of a Coordinate, which defines\n * its place, as well as a unique ID and data tracking for its content\n */\nexport class Tile {\n tileID: OverscaledTileID;\n uid: number;\n uses: number;\n tileSize: number;\n buckets: {[_: string]: Bucket};\n latestFeatureIndex: FeatureIndex;\n latestRawTileData: ArrayBuffer;\n imageAtlas: ImageAtlas;\n imageAtlasTexture: Texture;\n glyphAtlasImage: AlphaImage;\n glyphAtlasTexture: Texture;\n expirationTime: any;\n expiredRequestCount: number;\n state: TileState;\n timeAdded: number = 0;\n fadeEndTime: number = 0;\n collisionBoxArray: CollisionBoxArray;\n redoWhenDone: boolean;\n showCollisionBoxes: boolean;\n placementSource: any;\n actor: Actor;\n vtLayers: {[_: string]: VectorTileLayer};\n\n neighboringTiles: any;\n dem: DEMData;\n demMatrix: mat4;\n aborted: boolean;\n needsHillshadePrepare: boolean;\n needsTerrainPrepare: boolean;\n request: Cancelable;\n texture: any;\n fbo: Framebuffer;\n demTexture: Texture;\n refreshedUponExpiration: boolean;\n reloadCallback: any;\n resourceTiming: Array;\n queryPadding: number;\n\n symbolFadeHoldUntil: number;\n hasSymbolBuckets: boolean;\n hasRTLText: boolean;\n dependencies: any;\n rtt: Array<{id: number; stamp: number}>;\n rttCoords: {[_:string]: string};\n\n /**\n * @param tileID - the tile ID\n * @param size - The tile size\n */\n constructor(tileID: OverscaledTileID, size: number) {\n this.tileID = tileID;\n this.uid = uniqueId();\n this.uses = 0;\n this.tileSize = size;\n this.buckets = {};\n this.expirationTime = null;\n this.queryPadding = 0;\n this.hasSymbolBuckets = false;\n this.hasRTLText = false;\n this.dependencies = {};\n this.rtt = [];\n this.rttCoords = {};\n\n // Counts the number of times a response was already expired when\n // received. We're using this to add a delay when making a new request\n // so we don't have to keep retrying immediately in case of a server\n // serving expired tiles.\n this.expiredRequestCount = 0;\n\n this.state = 'loading';\n }\n\n registerFadeDuration(duration: number) {\n const fadeEndTime = duration + this.timeAdded;\n\n if (fadeEndTime < this.fadeEndTime) {\n return;\n }\n\n this.fadeEndTime = fadeEndTime;\n }\n\n wasRequested() {\n return this.state === 'errored' || this.state === 'loaded' || this.state === 'reloading';\n }\n\n clearTextures(painter: any) {\n if (this.demTexture) painter.saveTileTexture(this.demTexture);\n this.demTexture = null;\n }\n\n /**\n * Given a data object with a 'buffers' property, load it into\n * this tile's elementGroups and buffers properties and set loaded\n * to true. If the data is null, like in the case of an empty\n * GeoJSON tile, no-op but still set loaded to true.\n * @param data - The data from the worker\n * @param painter - the painter\n * @param justReloaded - `true` to just reload\n */\n loadVectorData(data: WorkerTileResult, painter: any, justReloaded?: boolean | null) {\n if (this.hasData()) {\n this.unloadVectorData();\n }\n\n this.state = 'loaded';\n\n // empty GeoJSON tile\n if (!data) {\n this.collisionBoxArray = new CollisionBoxArray();\n return;\n }\n\n if (data.featureIndex) {\n this.latestFeatureIndex = data.featureIndex;\n if (data.rawTileData) {\n // Only vector tiles have rawTileData, and they won't update it for\n // 'reloadTile'\n this.latestRawTileData = data.rawTileData;\n this.latestFeatureIndex.rawTileData = data.rawTileData;\n } else if (this.latestRawTileData) {\n // If rawTileData hasn't updated, hold onto a pointer to the last\n // one we received\n this.latestFeatureIndex.rawTileData = this.latestRawTileData;\n }\n }\n this.collisionBoxArray = data.collisionBoxArray;\n this.buckets = deserializeBucket(data.buckets, painter.style);\n\n this.hasSymbolBuckets = false;\n for (const id in this.buckets) {\n const bucket = this.buckets[id];\n if (bucket instanceof SymbolBucket) {\n this.hasSymbolBuckets = true;\n if (justReloaded) {\n bucket.justReloaded = true;\n } else {\n break;\n }\n }\n }\n\n this.hasRTLText = false;\n if (this.hasSymbolBuckets) {\n for (const id in this.buckets) {\n const bucket = this.buckets[id];\n if (bucket instanceof SymbolBucket) {\n if (bucket.hasRTLText) {\n this.hasRTLText = true;\n lazyLoadRTLTextPlugin();\n break;\n }\n }\n }\n }\n\n this.queryPadding = 0;\n for (const id in this.buckets) {\n const bucket = this.buckets[id];\n this.queryPadding = Math.max(this.queryPadding, painter.style.getLayer(id).queryRadius(bucket));\n }\n\n if (data.imageAtlas) {\n this.imageAtlas = data.imageAtlas;\n }\n if (data.glyphAtlasImage) {\n this.glyphAtlasImage = data.glyphAtlasImage;\n }\n }\n\n /**\n * Release any data or WebGL resources referenced by this tile.\n */\n unloadVectorData() {\n for (const id in this.buckets) {\n this.buckets[id].destroy();\n }\n this.buckets = {};\n\n if (this.imageAtlasTexture) {\n this.imageAtlasTexture.destroy();\n }\n\n if (this.imageAtlas) {\n this.imageAtlas = null;\n }\n\n if (this.glyphAtlasTexture) {\n this.glyphAtlasTexture.destroy();\n }\n\n this.latestFeatureIndex = null;\n this.state = 'unloaded';\n }\n\n getBucket(layer: StyleLayer) {\n return this.buckets[layer.id];\n }\n\n upload(context: Context) {\n for (const id in this.buckets) {\n const bucket = this.buckets[id];\n if (bucket.uploadPending()) {\n bucket.upload(context);\n }\n }\n\n const gl = context.gl;\n if (this.imageAtlas && !this.imageAtlas.uploaded) {\n this.imageAtlasTexture = new Texture(context, this.imageAtlas.image, gl.RGBA);\n this.imageAtlas.uploaded = true;\n }\n\n if (this.glyphAtlasImage) {\n this.glyphAtlasTexture = new Texture(context, this.glyphAtlasImage, gl.ALPHA);\n this.glyphAtlasImage = null;\n }\n }\n\n prepare(imageManager: ImageManager) {\n if (this.imageAtlas) {\n this.imageAtlas.patchUpdatedImages(imageManager, this.imageAtlasTexture);\n }\n }\n\n // Queries non-symbol features rendered for this tile.\n // Symbol features are queried globally\n queryRenderedFeatures(\n layers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n sourceFeatureState: SourceFeatureState,\n queryGeometry: Array,\n cameraQueryGeometry: Array,\n scale: number,\n params: {\n filter: FilterSpecification;\n layers: Array;\n availableImages: Array;\n },\n transform: Transform,\n maxPitchScaleFactor: number,\n pixelPosMatrix: mat4\n ): {[_: string]: Array<{featureIndex: number; feature: GeoJSONFeature}>} {\n if (!this.latestFeatureIndex || !this.latestFeatureIndex.rawTileData)\n return {};\n\n return this.latestFeatureIndex.query({\n queryGeometry,\n cameraQueryGeometry,\n scale,\n tileSize: this.tileSize,\n pixelPosMatrix,\n transform,\n params,\n queryPadding: this.queryPadding * maxPitchScaleFactor\n }, layers, serializedLayers, sourceFeatureState);\n }\n\n querySourceFeatures(result: Array, params?: {\n sourceLayer?: string;\n filter?: FilterSpecification;\n validate?: boolean;\n }) {\n const featureIndex = this.latestFeatureIndex;\n if (!featureIndex || !featureIndex.rawTileData) return;\n\n const vtLayers = featureIndex.loadVTLayers();\n\n const sourceLayer = params && params.sourceLayer ? params.sourceLayer : '';\n const layer = vtLayers._geojsonTileLayer || vtLayers[sourceLayer];\n\n if (!layer) return;\n\n const filter = featureFilter(params && params.filter);\n const {z, x, y} = this.tileID.canonical;\n const coord = {z, x, y};\n\n for (let i = 0; i < layer.length; i++) {\n const feature = layer.feature(i);\n if (filter.needGeometry) {\n const evaluationFeature = toEvaluationFeature(feature, true);\n if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), evaluationFeature, this.tileID.canonical)) continue;\n } else if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), feature)) {\n continue;\n }\n const id = featureIndex.getId(feature, sourceLayer);\n const geojsonFeature = new GeoJSONFeature(feature, z, x, y, id);\n (geojsonFeature as any).tile = coord;\n result.push(geojsonFeature);\n }\n }\n\n hasData() {\n return this.state === 'loaded' || this.state === 'reloading' || this.state === 'expired';\n }\n\n patternsLoaded() {\n return this.imageAtlas && !!Object.keys(this.imageAtlas.patternPositions).length;\n }\n\n setExpiryData(data: ExpiryData) {\n const prior = this.expirationTime;\n\n if (data.cacheControl) {\n const parsedCC = parseCacheControl(data.cacheControl);\n if (parsedCC['max-age']) this.expirationTime = Date.now() + parsedCC['max-age'] * 1000;\n } else if (data.expires) {\n this.expirationTime = new Date(data.expires).getTime();\n }\n\n if (this.expirationTime) {\n const now = Date.now();\n let isExpired = false;\n\n if (this.expirationTime > now) {\n isExpired = false;\n } else if (!prior) {\n isExpired = true;\n } else if (this.expirationTime < prior) {\n // Expiring date is going backwards:\n // fall back to exponential backoff\n isExpired = true;\n\n } else {\n const delta = this.expirationTime - prior;\n\n if (!delta) {\n // Server is serving the same expired resource over and over: fall\n // back to exponential backoff.\n isExpired = true;\n\n } else {\n // Assume that either the client or the server clock is wrong and\n // try to interpolate a valid expiration date (from the client POV)\n // observing a minimum timeout.\n this.expirationTime = now + Math.max(delta, CLOCK_SKEW_RETRY_TIMEOUT);\n\n }\n }\n\n if (isExpired) {\n this.expiredRequestCount++;\n this.state = 'expired';\n } else {\n this.expiredRequestCount = 0;\n }\n }\n }\n\n getExpiryTimeout() {\n if (this.expirationTime) {\n if (this.expiredRequestCount) {\n return 1000 * (1 << Math.min(this.expiredRequestCount - 1, 31));\n } else {\n // Max value for `setTimeout` implementations is a 32 bit integer; cap this accordingly\n return Math.min(this.expirationTime - new Date().getTime(), Math.pow(2, 31) - 1);\n }\n }\n }\n\n setFeatureState(states: LayerFeatureStates, painter: any) {\n if (!this.latestFeatureIndex ||\n !this.latestFeatureIndex.rawTileData ||\n Object.keys(states).length === 0) {\n return;\n }\n\n const vtLayers = this.latestFeatureIndex.loadVTLayers();\n\n for (const id in this.buckets) {\n if (!painter.style.hasLayer(id)) continue;\n\n const bucket = this.buckets[id];\n // Buckets are grouped by common source-layer\n const sourceLayerId = bucket.layers[0]['sourceLayer'] || '_geojsonTileLayer';\n const sourceLayer = vtLayers[sourceLayerId];\n const sourceLayerStates = states[sourceLayerId];\n if (!sourceLayer || !sourceLayerStates || Object.keys(sourceLayerStates).length === 0) continue;\n\n bucket.update(sourceLayerStates, sourceLayer, this.imageAtlas && this.imageAtlas.patternPositions || {});\n const layer = painter && painter.style && painter.style.getLayer(id);\n if (layer) {\n this.queryPadding = Math.max(this.queryPadding, layer.queryRadius(bucket));\n }\n }\n }\n\n holdingForFade(): boolean {\n return this.symbolFadeHoldUntil !== undefined;\n }\n\n symbolFadeFinished(): boolean {\n return !this.symbolFadeHoldUntil || this.symbolFadeHoldUntil < browser.now();\n }\n\n clearFadeHold() {\n this.symbolFadeHoldUntil = undefined;\n }\n\n setHoldDuration(duration: number) {\n this.symbolFadeHoldUntil = browser.now() + duration;\n }\n\n setDependencies(namespace: string, dependencies: Array) {\n const index = {};\n for (const dep of dependencies) {\n index[dep] = true;\n }\n this.dependencies[namespace] = index;\n }\n\n hasDependency(namespaces: Array, keys: Array) {\n for (const namespace of namespaces) {\n const dependencies = this.dependencies[namespace];\n if (dependencies) {\n for (const key of keys) {\n if (dependencies[key]) {\n return true;\n }\n }\n }\n }\n return false;\n }\n}\n","import {OverscaledTileID} from './tile_id';\nimport type {Tile} from './tile';\n\n/**\n * @internal\n * A [least-recently-used cache](http://en.wikipedia.org/wiki/Cache_algorithms)\n * with hash lookup made possible by keeping a list of keys in parallel to\n * an array of dictionary of values\n */\nexport class TileCache {\n max: number;\n data: {\n [key: string]: Array<{\n value: Tile;\n timeout: ReturnType;\n }>;\n };\n order: Array;\n onRemove: (element: Tile) => void;\n /**\n * @param max - number of permitted values\n * @param onRemove - callback called with items when they expire\n */\n constructor(max: number, onRemove: (element: Tile) => void) {\n this.max = max;\n this.onRemove = onRemove;\n this.reset();\n }\n\n /**\n * Clear the cache\n *\n * @returns this cache\n */\n reset() {\n for (const key in this.data) {\n for (const removedData of this.data[key]) {\n if (removedData.timeout) clearTimeout(removedData.timeout);\n this.onRemove(removedData.value);\n }\n }\n\n this.data = {};\n this.order = [];\n\n return this;\n }\n\n /**\n * Add a key, value combination to the cache, trimming its size if this pushes\n * it over max length.\n *\n * @param tileID - lookup key for the item\n * @param data - tile data\n *\n * @returns this cache\n */\n add(tileID: OverscaledTileID, data: Tile, expiryTimeout: number | void) {\n const key = tileID.wrapped().key;\n if (this.data[key] === undefined) {\n this.data[key] = [];\n }\n\n const dataWrapper = {\n value: data,\n timeout: undefined\n };\n\n if (expiryTimeout !== undefined) {\n dataWrapper.timeout = setTimeout(() => {\n this.remove(tileID, dataWrapper);\n }, expiryTimeout as number);\n }\n\n this.data[key].push(dataWrapper);\n this.order.push(key);\n\n if (this.order.length > this.max) {\n const removedData = this._getAndRemoveByKey(this.order[0]);\n if (removedData) this.onRemove(removedData);\n }\n\n return this;\n }\n\n /**\n * Determine whether the value attached to `key` is present\n *\n * @param tileID - the key to be looked-up\n * @returns whether the cache has this value\n */\n has(tileID: OverscaledTileID): boolean {\n return tileID.wrapped().key in this.data;\n }\n\n /**\n * Get the value attached to a specific key and remove data from cache.\n * If the key is not found, returns `null`\n *\n * @param tileID - the key to look up\n * @returns the tile data, or null if it isn't found\n */\n getAndRemove(tileID: OverscaledTileID): Tile {\n if (!this.has(tileID)) { return null; }\n return this._getAndRemoveByKey(tileID.wrapped().key);\n }\n\n /*\n * Get and remove the value with the specified key.\n */\n _getAndRemoveByKey(key: string): Tile {\n const data = this.data[key].shift();\n if (data.timeout) clearTimeout(data.timeout);\n\n if (this.data[key].length === 0) {\n delete this.data[key];\n }\n this.order.splice(this.order.indexOf(key), 1);\n\n return data.value;\n }\n\n /*\n * Get the value with the specified (wrapped tile) key.\n */\n getByKey(key: string): Tile {\n const data = this.data[key];\n return data ? data[0].value : null;\n }\n\n /**\n * Get the value attached to a specific key without removing data\n * from the cache. If the key is not found, returns `null`\n *\n * @param tileID - the key to look up\n * @returns the tile data, or null if it isn't found\n */\n get(tileID: OverscaledTileID): Tile {\n if (!this.has(tileID)) { return null; }\n\n const data = this.data[tileID.wrapped().key][0];\n return data.value;\n }\n\n /**\n * Remove a key/value combination from the cache.\n *\n * @param tileID - the key for the pair to delete\n * @param value - If a value is provided, remove that exact version of the value.\n * @returns this cache\n */\n remove(tileID: OverscaledTileID, value?: {\n value: Tile;\n timeout: ReturnType;\n }) {\n if (!this.has(tileID)) { return this; }\n const key = tileID.wrapped().key;\n\n const dataIndex = value === undefined ? 0 : this.data[key].indexOf(value);\n const data = this.data[key][dataIndex];\n this.data[key].splice(dataIndex, 1);\n if (data.timeout) clearTimeout(data.timeout);\n if (this.data[key].length === 0) {\n delete this.data[key];\n }\n this.onRemove(data.value);\n this.order.splice(this.order.indexOf(key), 1);\n\n return this;\n }\n\n /**\n * Change the max size of the cache.\n *\n * @param max - the max size of the cache\n * @returns this cache\n */\n setMaxSize(max: number): TileCache {\n this.max = max;\n\n while (this.order.length > this.max) {\n const removedData = this._getAndRemoveByKey(this.order[0]);\n if (removedData) this.onRemove(removedData);\n }\n\n return this;\n }\n\n /**\n * Remove entries that do not pass a filter function. Used for removing\n * stale tiles from the cache.\n *\n * @param filterFn - Determines whether the tile is filtered. If the supplied function returns false, the tile will be filtered out.\n */\n filter(filterFn: (tile: Tile) => boolean) {\n const removed = [];\n for (const key in this.data) {\n for (const entry of this.data[key]) {\n if (!filterFn(entry.value)) {\n removed.push(entry);\n }\n }\n }\n for (const r of removed) {\n this.remove(r.value.tileID, r);\n }\n }\n}\n","import {extend} from '../util/util';\nimport {Tile} from './tile';\nimport type {FeatureState} from '@maplibre/maplibre-gl-style-spec';\n\nexport type FeatureStates = {[featureId: string]: FeatureState};\nexport type LayerFeatureStates = {[layer: string]: FeatureStates};\n\n/**\n * @internal\n * SourceFeatureState manages the state and pending changes\n * to features in a source, separated by source layer.\n * stateChanges and deletedStates batch all changes to the tile (updates and removes, respectively)\n * between coalesce() events. addFeatureState() and removeFeatureState() also update their counterpart's\n * list of changes, such that coalesce() can apply the proper state changes while agnostic to the order of operations.\n * In deletedStates, all null's denote complete removal of state at that scope\n*/\nexport class SourceFeatureState {\n state: LayerFeatureStates;\n stateChanges: LayerFeatureStates;\n deletedStates: {};\n\n constructor() {\n this.state = {};\n this.stateChanges = {};\n this.deletedStates = {};\n }\n\n updateState(sourceLayer: string, featureId: number | string, newState: any) {\n const feature = String(featureId);\n this.stateChanges[sourceLayer] = this.stateChanges[sourceLayer] || {};\n this.stateChanges[sourceLayer][feature] = this.stateChanges[sourceLayer][feature] || {};\n extend(this.stateChanges[sourceLayer][feature], newState);\n\n if (this.deletedStates[sourceLayer] === null) {\n this.deletedStates[sourceLayer] = {};\n for (const ft in this.state[sourceLayer]) {\n if (ft !== feature) this.deletedStates[sourceLayer][ft] = null;\n }\n } else {\n const featureDeletionQueued = this.deletedStates[sourceLayer] && this.deletedStates[sourceLayer][feature] === null;\n if (featureDeletionQueued) {\n this.deletedStates[sourceLayer][feature] = {};\n for (const prop in this.state[sourceLayer][feature]) {\n if (!newState[prop]) this.deletedStates[sourceLayer][feature][prop] = null;\n }\n } else {\n for (const key in newState) {\n const deletionInQueue = this.deletedStates[sourceLayer] && this.deletedStates[sourceLayer][feature] && this.deletedStates[sourceLayer][feature][key] === null;\n if (deletionInQueue) delete this.deletedStates[sourceLayer][feature][key];\n }\n }\n }\n }\n\n removeFeatureState(sourceLayer: string, featureId?: number | string, key?: string) {\n const sourceLayerDeleted = this.deletedStates[sourceLayer] === null;\n if (sourceLayerDeleted) return;\n\n const feature = String(featureId);\n\n this.deletedStates[sourceLayer] = this.deletedStates[sourceLayer] || {};\n\n if (key && featureId !== undefined) {\n if (this.deletedStates[sourceLayer][feature] !== null) {\n this.deletedStates[sourceLayer][feature] = this.deletedStates[sourceLayer][feature] || {};\n this.deletedStates[sourceLayer][feature][key] = null;\n }\n } else if (featureId !== undefined) {\n const updateInQueue = this.stateChanges[sourceLayer] && this.stateChanges[sourceLayer][feature];\n if (updateInQueue) {\n this.deletedStates[sourceLayer][feature] = {};\n for (key in this.stateChanges[sourceLayer][feature]) this.deletedStates[sourceLayer][feature][key] = null;\n\n } else {\n this.deletedStates[sourceLayer][feature] = null;\n }\n } else {\n this.deletedStates[sourceLayer] = null;\n }\n\n }\n\n getState(sourceLayer: string, featureId: number | string) {\n const feature = String(featureId);\n const base = this.state[sourceLayer] || {};\n const changes = this.stateChanges[sourceLayer] || {};\n\n const reconciledState = extend({}, base[feature], changes[feature]);\n\n //return empty object if the whole source layer is awaiting deletion\n if (this.deletedStates[sourceLayer] === null) return {};\n else if (this.deletedStates[sourceLayer]) {\n const featureDeletions = this.deletedStates[sourceLayer][featureId];\n if (featureDeletions === null) return {};\n for (const prop in featureDeletions) delete reconciledState[prop];\n }\n return reconciledState;\n }\n\n initializeTileState(tile: Tile, painter: any) {\n tile.setFeatureState(this.state, painter);\n }\n\n coalesceChanges(tiles: {\n [_ in any]: Tile;\n }, painter: any) {\n //track changes with full state objects, but only for features that got modified\n const featuresChanged: LayerFeatureStates = {};\n\n for (const sourceLayer in this.stateChanges) {\n this.state[sourceLayer] = this.state[sourceLayer] || {};\n const layerStates = {};\n for (const feature in this.stateChanges[sourceLayer]) {\n if (!this.state[sourceLayer][feature]) this.state[sourceLayer][feature] = {};\n extend(this.state[sourceLayer][feature], this.stateChanges[sourceLayer][feature]);\n layerStates[feature] = this.state[sourceLayer][feature];\n }\n featuresChanged[sourceLayer] = layerStates;\n }\n\n for (const sourceLayer in this.deletedStates) {\n this.state[sourceLayer] = this.state[sourceLayer] || {};\n const layerStates = {};\n\n if (this.deletedStates[sourceLayer] === null) {\n for (const ft in this.state[sourceLayer]) {\n layerStates[ft] = {};\n this.state[sourceLayer][ft] = {};\n }\n } else {\n for (const feature in this.deletedStates[sourceLayer]) {\n const deleteWholeFeatureState = this.deletedStates[sourceLayer][feature] === null;\n if (deleteWholeFeatureState) this.state[sourceLayer][feature] = {};\n else {\n for (const key of Object.keys(this.deletedStates[sourceLayer][feature])) {\n delete this.state[sourceLayer][feature][key];\n }\n }\n layerStates[feature] = this.state[sourceLayer][feature];\n }\n }\n\n featuresChanged[sourceLayer] = featuresChanged[sourceLayer] || {};\n extend(featuresChanged[sourceLayer], layerStates);\n }\n\n this.stateChanges = {};\n this.deletedStates = {};\n\n if (Object.keys(featuresChanged).length === 0) return;\n\n for (const id in tiles) {\n const tile = tiles[id];\n tile.setFeatureState(featuresChanged, painter);\n }\n }\n}\n","import {create as createSource} from './source';\n\nimport {Tile} from './tile';\nimport {Event, ErrorEvent, Evented} from '../util/evented';\nimport {TileCache} from './tile_cache';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\nimport {keysDifference} from '../util/util';\nimport {EXTENT} from '../data/extent';\nimport {Context} from '../gl/context';\nimport Point from '@mapbox/point-geometry';\nimport {browser} from '../util/browser';\nimport {OverscaledTileID} from './tile_id';\nimport {SourceFeatureState} from './source_state';\n\nimport type {Source} from './source';\nimport type {Map} from '../ui/map';\nimport type {Style} from '../style/style';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Transform} from '../geo/transform';\nimport type {TileState} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {SourceSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {MapSourceDataEvent} from '../ui/events';\nimport {Terrain} from '../render/terrain';\nimport {config} from '../util/config';\n\n/**\n * @internal\n * `SourceCache` is responsible for\n *\n * - creating an instance of `Source`\n * - forwarding events from `Source`\n * - caching tiles loaded from an instance of `Source`\n * - loading the tiles needed to render a given viewport\n * - unloading the cached tiles not needed to render a given viewport\n */\nexport class SourceCache extends Evented {\n id: string;\n dispatcher: Dispatcher;\n map: Map;\n style: Style;\n\n _source: Source;\n _sourceLoaded: boolean;\n _sourceErrored: boolean;\n _tiles: {[_: string]: Tile};\n _prevLng: number;\n _cache: TileCache;\n _timers: {\n [_ in any]: ReturnType;\n };\n _cacheTimers: {\n [_ in any]: ReturnType;\n };\n _maxTileCacheSize: number;\n _maxTileCacheZoomLevels: number;\n _paused: boolean;\n _shouldReloadOnResume: boolean;\n _coveredTiles: {[_: string]: boolean};\n transform: Transform;\n terrain: Terrain;\n used: boolean;\n usedForTerrain: boolean;\n tileSize: number;\n _state: SourceFeatureState;\n _loadedParentTiles: {[_: string]: Tile};\n _didEmitContent: boolean;\n _updated: boolean;\n\n static maxUnderzooming: number;\n static maxOverzooming: number;\n\n constructor(id: string, options: SourceSpecification, dispatcher: Dispatcher) {\n super();\n this.id = id;\n this.dispatcher = dispatcher;\n\n this.on('data', (e: MapSourceDataEvent) => {\n // this._sourceLoaded signifies that the TileJSON is loaded if applicable.\n // if the source type does not come with a TileJSON, the flag signifies the\n // source data has loaded (i.e geojson has been tiled on the worker and is ready)\n if (e.dataType === 'source' && e.sourceDataType === 'metadata') this._sourceLoaded = true;\n\n // for sources with mutable data, this event fires when the underlying data\n // to a source is changed. (i.e. GeoJSONSource#setData and ImageSource#serCoordinates)\n if (this._sourceLoaded && !this._paused && e.dataType === 'source' && e.sourceDataType === 'content') {\n this.reload();\n if (this.transform) {\n this.update(this.transform, this.terrain);\n }\n\n this._didEmitContent = true;\n }\n });\n\n this.on('dataloading', () => {\n this._sourceErrored = false;\n });\n\n this.on('error', () => {\n // Only set _sourceErrored if the source does not have pending loads.\n this._sourceErrored = this._source.loaded();\n });\n\n this._source = createSource(id, options, dispatcher, this);\n\n this._tiles = {};\n this._cache = new TileCache(0, this._unloadTile.bind(this));\n this._timers = {};\n this._cacheTimers = {};\n this._maxTileCacheSize = null;\n this._maxTileCacheZoomLevels = null;\n this._loadedParentTiles = {};\n\n this._coveredTiles = {};\n this._state = new SourceFeatureState();\n this._didEmitContent = false;\n this._updated = false;\n }\n\n onAdd(map: Map) {\n this.map = map;\n this._maxTileCacheSize = map ? map._maxTileCacheSize : null;\n this._maxTileCacheZoomLevels = map ? map._maxTileCacheZoomLevels : null;\n if (this._source && this._source.onAdd) {\n this._source.onAdd(map);\n }\n }\n\n onRemove(map: Map) {\n this.clearTiles();\n if (this._source && this._source.onRemove) {\n this._source.onRemove(map);\n }\n }\n\n /**\n * Return true if no tile data is pending, tiles will not change unless\n * an additional API call is received.\n */\n loaded(): boolean {\n if (this._sourceErrored) { return true; }\n if (!this._sourceLoaded) { return false; }\n if (!this._source.loaded()) { return false; }\n if ((this.used !== undefined || this.usedForTerrain !== undefined) && !this.used && !this.usedForTerrain) { return true; }\n // do not consider as loaded if the update hasn't been called yet (we do not know if we will have any tiles to fetch)\n if (!this._updated) { return false; }\n\n for (const t in this._tiles) {\n const tile = this._tiles[t];\n if (tile.state !== 'loaded' && tile.state !== 'errored')\n return false;\n }\n return true;\n }\n\n getSource(): Source {\n return this._source;\n }\n\n pause() {\n this._paused = true;\n }\n\n resume() {\n if (!this._paused) return;\n const shouldReload = this._shouldReloadOnResume;\n this._paused = false;\n this._shouldReloadOnResume = false;\n if (shouldReload) this.reload();\n if (this.transform) this.update(this.transform, this.terrain);\n }\n\n _loadTile(tile: Tile, callback: Callback) {\n return this._source.loadTile(tile, callback);\n }\n\n _unloadTile(tile: Tile) {\n if (this._source.unloadTile)\n return this._source.unloadTile(tile, () => {});\n }\n\n _abortTile(tile: Tile) {\n if (this._source.abortTile)\n this._source.abortTile(tile, () => {});\n\n this._source.fire(new Event('dataabort', {tile, coord: tile.tileID, dataType: 'source'}));\n }\n\n serialize() {\n return this._source.serialize();\n }\n\n prepare(context: Context) {\n if (this._source.prepare) {\n this._source.prepare();\n }\n\n this._state.coalesceChanges(this._tiles, this.map ? this.map.painter : null);\n for (const i in this._tiles) {\n const tile = this._tiles[i];\n tile.upload(context);\n tile.prepare(this.map.style.imageManager);\n }\n }\n\n /**\n * Return all tile ids ordered with z-order, and cast to numbers\n */\n getIds(): Array {\n return (Object.values(this._tiles) as any).map((tile: Tile) => tile.tileID).sort(compareTileId).map(id => id.key);\n }\n\n getRenderableIds(symbolLayer?: boolean): Array {\n const renderables: Array = [];\n for (const id in this._tiles) {\n if (this._isIdRenderable(id, symbolLayer)) renderables.push(this._tiles[id]);\n }\n if (symbolLayer) {\n return renderables.sort((a_: Tile, b_: Tile) => {\n const a = a_.tileID;\n const b = b_.tileID;\n const rotatedA = (new Point(a.canonical.x, a.canonical.y))._rotate(this.transform.angle);\n const rotatedB = (new Point(b.canonical.x, b.canonical.y))._rotate(this.transform.angle);\n return a.overscaledZ - b.overscaledZ || rotatedB.y - rotatedA.y || rotatedB.x - rotatedA.x;\n }).map(tile => tile.tileID.key);\n }\n return renderables.map(tile => tile.tileID).sort(compareTileId).map(id => id.key);\n }\n\n hasRenderableParent(tileID: OverscaledTileID) {\n const parentTile = this.findLoadedParent(tileID, 0);\n if (parentTile) {\n return this._isIdRenderable(parentTile.tileID.key);\n }\n return false;\n }\n\n _isIdRenderable(id: string, symbolLayer?: boolean) {\n return this._tiles[id] && this._tiles[id].hasData() &&\n !this._coveredTiles[id] && (symbolLayer || !this._tiles[id].holdingForFade());\n }\n\n reload() {\n if (this._paused) {\n this._shouldReloadOnResume = true;\n return;\n }\n\n this._cache.reset();\n\n for (const i in this._tiles) {\n if (this._tiles[i].state !== 'errored') this._reloadTile(i, 'reloading');\n }\n }\n\n _reloadTile(id: string, state: TileState) {\n const tile = this._tiles[id];\n\n // this potentially does not address all underlying\n // issues https://github.com/mapbox/mapbox-gl-js/issues/4252\n // - hard to tell without repro steps\n if (!tile) return;\n\n // The difference between \"loading\" tiles and \"reloading\" or \"expired\"\n // tiles is that \"reloading\"/\"expired\" tiles are \"renderable\".\n // Therefore, a \"loading\" tile cannot become a \"reloading\" tile without\n // first becoming a \"loaded\" tile.\n if (tile.state !== 'loading') {\n tile.state = state;\n }\n\n this._loadTile(tile, this._tileLoaded.bind(this, tile, id, state));\n }\n\n _tileLoaded(tile: Tile, id: string, previousState: TileState, err?: Error | null) {\n if (err) {\n tile.state = 'errored';\n if ((err as any).status !== 404) this._source.fire(new ErrorEvent(err, {tile}));\n // continue to try loading parent/children tiles if a tile doesn't exist (404)\n else this.update(this.transform, this.terrain);\n return;\n }\n\n tile.timeAdded = browser.now();\n if (previousState === 'expired') tile.refreshedUponExpiration = true;\n this._setTileReloadTimer(id, tile);\n if (this.getSource().type === 'raster-dem' && tile.dem) this._backfillDEM(tile);\n this._state.initializeTileState(tile, this.map ? this.map.painter : null);\n\n if (!tile.aborted) {\n this._source.fire(new Event('data', {dataType: 'source', tile, coord: tile.tileID}));\n }\n }\n\n /**\n * For raster terrain source, backfill DEM to eliminate visible tile boundaries\n */\n _backfillDEM(tile: Tile) {\n const renderables = this.getRenderableIds();\n for (let i = 0; i < renderables.length; i++) {\n const borderId = renderables[i];\n if (tile.neighboringTiles && tile.neighboringTiles[borderId]) {\n const borderTile = this.getTileByID(borderId);\n fillBorder(tile, borderTile);\n fillBorder(borderTile, tile);\n }\n }\n\n function fillBorder(tile, borderTile) {\n tile.needsHillshadePrepare = true;\n tile.needsTerrainPrepare = true;\n let dx = borderTile.tileID.canonical.x - tile.tileID.canonical.x;\n const dy = borderTile.tileID.canonical.y - tile.tileID.canonical.y;\n const dim = Math.pow(2, tile.tileID.canonical.z);\n const borderId = borderTile.tileID.key;\n if (dx === 0 && dy === 0) return;\n\n if (Math.abs(dy) > 1) {\n return;\n }\n if (Math.abs(dx) > 1) {\n // Adjust the delta coordinate for world wraparound.\n if (Math.abs(dx + dim) === 1) {\n dx += dim;\n } else if (Math.abs(dx - dim) === 1) {\n dx -= dim;\n }\n }\n if (!borderTile.dem || !tile.dem) return;\n tile.dem.backfillBorder(borderTile.dem, dx, dy);\n if (tile.neighboringTiles && tile.neighboringTiles[borderId])\n tile.neighboringTiles[borderId].backfilled = true;\n }\n }\n /**\n * Get a specific tile by TileID\n */\n getTile(tileID: OverscaledTileID): Tile {\n return this.getTileByID(tileID.key);\n }\n\n /**\n * Get a specific tile by id\n */\n getTileByID(id: string): Tile {\n return this._tiles[id];\n }\n\n /**\n * For a given set of tiles, retain children that are loaded and have a zoom\n * between `zoom` (exclusive) and `maxCoveringZoom` (inclusive)\n */\n _retainLoadedChildren(\n idealTiles: {\n [_ in any]: OverscaledTileID;\n },\n zoom: number,\n maxCoveringZoom: number,\n retain: {\n [_ in any]: OverscaledTileID;\n }\n ) {\n for (const id in this._tiles) {\n let tile = this._tiles[id];\n\n // only consider renderable tiles up to maxCoveringZoom\n if (retain[id] ||\n !tile.hasData() ||\n tile.tileID.overscaledZ <= zoom ||\n tile.tileID.overscaledZ > maxCoveringZoom\n ) continue;\n\n // loop through parents and retain the topmost loaded one if found\n let topmostLoadedID = tile.tileID;\n while (tile && tile.tileID.overscaledZ > zoom + 1) {\n const parentID = tile.tileID.scaledTo(tile.tileID.overscaledZ - 1);\n\n tile = this._tiles[parentID.key];\n\n if (tile && tile.hasData()) {\n topmostLoadedID = parentID;\n }\n }\n\n // loop through ancestors of the topmost loaded child to see if there's one that needed it\n let tileID = topmostLoadedID;\n while (tileID.overscaledZ > zoom) {\n tileID = tileID.scaledTo(tileID.overscaledZ - 1);\n\n if (idealTiles[tileID.key]) {\n // found a parent that needed a loaded child; retain that child\n retain[topmostLoadedID.key] = topmostLoadedID;\n break;\n }\n }\n }\n }\n\n /**\n * Find a loaded parent of the given tile (up to minCoveringZoom)\n */\n findLoadedParent(tileID: OverscaledTileID, minCoveringZoom: number): Tile {\n if (tileID.key in this._loadedParentTiles) {\n const parent = this._loadedParentTiles[tileID.key];\n if (parent && parent.tileID.overscaledZ >= minCoveringZoom) {\n return parent;\n } else {\n return null;\n }\n }\n for (let z = tileID.overscaledZ - 1; z >= minCoveringZoom; z--) {\n const parentTileID = tileID.scaledTo(z);\n const tile = this._getLoadedTile(parentTileID);\n if (tile) {\n return tile;\n }\n }\n }\n\n _getLoadedTile(tileID: OverscaledTileID): Tile {\n const tile = this._tiles[tileID.key];\n if (tile && tile.hasData()) {\n return tile;\n }\n // TileCache ignores wrap in lookup.\n const cachedTile = this._cache.getByKey(tileID.wrapped().key);\n return cachedTile;\n }\n\n /**\n * Resizes the tile cache based on the current viewport's size\n * or the maxTileCacheSize option passed during map creation\n *\n * Larger viewports use more tiles and need larger caches. Larger viewports\n * are more likely to be found on devices with more memory and on pages where\n * the map is more important.\n */\n updateCacheSize(transform: Transform) {\n const widthInTiles = Math.ceil(transform.width / this._source.tileSize) + 1;\n const heightInTiles = Math.ceil(transform.height / this._source.tileSize) + 1;\n const approxTilesInView = widthInTiles * heightInTiles;\n const commonZoomRange = this._maxTileCacheZoomLevels === null ?\n config.MAX_TILE_CACHE_ZOOM_LEVELS : this._maxTileCacheZoomLevels;\n const viewDependentMaxSize = Math.floor(approxTilesInView * commonZoomRange);\n const maxSize = typeof this._maxTileCacheSize === 'number' ?\n Math.min(this._maxTileCacheSize, viewDependentMaxSize) : viewDependentMaxSize;\n\n this._cache.setMaxSize(maxSize);\n }\n\n handleWrapJump(lng: number) {\n // On top of the regular z/x/y values, TileIDs have a `wrap` value that specify\n // which cppy of the world the tile belongs to. For example, at `lng: 10` you\n // might render z/x/y/0 while at `lng: 370` you would render z/x/y/1.\n //\n // When lng values get wrapped (going from `lng: 370` to `long: 10`) you expect\n // to see the same thing on the screen (370 degrees and 10 degrees is the same\n // place in the world) but all the TileIDs will have different wrap values.\n //\n // In order to make this transition seamless, we calculate the rounded difference of\n // \"worlds\" between the last frame and the current frame. If the map panned by\n // a world, then we can assign all the tiles new TileIDs with updated wrap values.\n // For example, assign z/x/y/1 a new id: z/x/y/0. It is the same tile, just rendered\n // in a different position.\n //\n // This enables us to reuse the tiles at more ideal locations and prevent flickering.\n const prevLng = this._prevLng === undefined ? lng : this._prevLng;\n const lngDifference = lng - prevLng;\n const worldDifference = lngDifference / 360;\n const wrapDelta = Math.round(worldDifference);\n this._prevLng = lng;\n\n if (wrapDelta) {\n const tiles: {[_: string]: Tile} = {};\n for (const key in this._tiles) {\n const tile = this._tiles[key];\n tile.tileID = tile.tileID.unwrapTo(tile.tileID.wrap + wrapDelta);\n tiles[tile.tileID.key] = tile;\n }\n this._tiles = tiles;\n\n // Reset tile reload timers\n for (const id in this._timers) {\n clearTimeout(this._timers[id]);\n delete this._timers[id];\n }\n for (const id in this._tiles) {\n const tile = this._tiles[id];\n this._setTileReloadTimer(id, tile);\n }\n }\n }\n\n /**\n * Removes tiles that are outside the viewport and adds new tiles that\n * are inside the viewport.\n */\n update(transform: Transform, terrain?: Terrain) {\n this.transform = transform;\n this.terrain = terrain;\n if (!this._sourceLoaded || this._paused) { return; }\n\n this.updateCacheSize(transform);\n this.handleWrapJump(this.transform.center.lng);\n\n // Covered is a list of retained tiles who's areas are fully covered by other,\n // better, retained tiles. They are not drawn separately.\n this._coveredTiles = {};\n\n let idealTileIDs;\n if (!this.used && !this.usedForTerrain) {\n idealTileIDs = [];\n } else if (this._source.tileID) {\n idealTileIDs = transform.getVisibleUnwrappedCoordinates(this._source.tileID)\n .map((unwrapped) => new OverscaledTileID(unwrapped.canonical.z, unwrapped.wrap, unwrapped.canonical.z, unwrapped.canonical.x, unwrapped.canonical.y));\n } else {\n idealTileIDs = transform.coveringTiles({\n tileSize: this.usedForTerrain ? this.tileSize : this._source.tileSize,\n minzoom: this._source.minzoom,\n maxzoom: this._source.maxzoom,\n roundZoom: this.usedForTerrain ? false : this._source.roundZoom,\n reparseOverscaled: this._source.reparseOverscaled,\n terrain\n });\n\n if (this._source.hasTile) {\n idealTileIDs = idealTileIDs.filter((coord) => (this._source.hasTile as any)(coord));\n }\n }\n\n // Determine the overzooming/underzooming amounts.\n const zoom = transform.coveringZoomLevel(this._source);\n const minCoveringZoom = Math.max(zoom - SourceCache.maxOverzooming, this._source.minzoom);\n const maxCoveringZoom = Math.max(zoom + SourceCache.maxUnderzooming, this._source.minzoom);\n\n // When sourcecache is used for terrain also load parent tiles to avoid flickering when zooming out\n if (this.usedForTerrain) {\n const parents = {};\n for (const tileID of idealTileIDs) {\n if (tileID.canonical.z > this._source.minzoom) {\n const parent = tileID.scaledTo(tileID.canonical.z - 1);\n parents[parent.key] = parent;\n // load very low zoom to calculate tile visibility in transform.coveringTiles and high zoomlevels correct\n const parent2 = tileID.scaledTo(Math.max(this._source.minzoom, Math.min(tileID.canonical.z, 5)));\n parents[parent2.key] = parent2;\n }\n }\n idealTileIDs = idealTileIDs.concat(Object.values(parents));\n }\n\n const noPendingDataEmissions = idealTileIDs.length === 0 && !this._updated && this._didEmitContent;\n this._updated = true;\n // if we won't have any tiles to fetch and content is already emitted\n // there will be no more data emissions, so we need to emit the event with isSourceLoaded = true\n if (noPendingDataEmissions) {\n this.fire(new Event('data', {sourceDataType: 'idle', dataType: 'source', sourceId: this.id}));\n }\n\n // Retain is a list of tiles that we shouldn't delete, even if they are not\n // the most ideal tile for the current viewport. This may include tiles like\n // parent or child tiles that are *already* loaded.\n const retain = this._updateRetainedTiles(idealTileIDs, zoom);\n\n if (isRasterType(this._source.type)) {\n const parentsForFading: {[_: string]: OverscaledTileID} = {};\n const fadingTiles = {};\n const ids = Object.keys(retain);\n const now = browser.now();\n for (const id of ids) {\n const tileID = retain[id];\n\n const tile = this._tiles[id];\n\n // when fadeEndTime is 0, the tile is created but registerFadeDuration\n // has not been called, therefore must be kept in fadingTiles dictionary\n // for next round of rendering\n if (!tile || (tile.fadeEndTime !== 0 && tile.fadeEndTime <= now)) {\n continue;\n }\n\n // if the tile is loaded but still fading in, find parents to cross-fade with it\n const parentTile = this.findLoadedParent(tileID, minCoveringZoom);\n if (parentTile) {\n this._addTile(parentTile.tileID);\n parentsForFading[parentTile.tileID.key] = parentTile.tileID;\n }\n\n fadingTiles[id] = tileID;\n }\n\n // for tiles that are still fading in, also find children to cross-fade with\n this._retainLoadedChildren(fadingTiles, zoom, maxCoveringZoom, retain);\n\n for (const id in parentsForFading) {\n if (!retain[id]) {\n // If a tile is only needed for fading, mark it as covered so that it isn't rendered on it's own.\n this._coveredTiles[id] = true;\n retain[id] = parentsForFading[id];\n }\n }\n\n // disable fading logic in terrain3D mode to avoid rendering two tiles on the same place\n if (terrain) {\n const idealRasterTileIDs: {[_: string]: OverscaledTileID} = {};\n const missingTileIDs: {[_: string]: OverscaledTileID} = {};\n for (const tileID of idealTileIDs) {\n if (this._tiles[tileID.key].hasData())\n idealRasterTileIDs[tileID.key] = tileID;\n else\n missingTileIDs[tileID.key] = tileID;\n }\n // search for a complete set of children for each missing tile\n for (const key in missingTileIDs) {\n const children = missingTileIDs[key].children(this._source.maxzoom);\n if (this._tiles[children[0].key] && this._tiles[children[1].key] && this._tiles[children[2].key] && this._tiles[children[3].key]) {\n idealRasterTileIDs[children[0].key] = retain[children[0].key] = children[0];\n idealRasterTileIDs[children[1].key] = retain[children[1].key] = children[1];\n idealRasterTileIDs[children[2].key] = retain[children[2].key] = children[2];\n idealRasterTileIDs[children[3].key] = retain[children[3].key] = children[3];\n delete missingTileIDs[key];\n }\n }\n // search for parent for each missing tile\n for (const key in missingTileIDs) {\n const parent = this.findLoadedParent(missingTileIDs[key], this._source.minzoom);\n if (parent) {\n idealRasterTileIDs[parent.tileID.key] = retain[parent.tileID.key] = parent.tileID;\n // remove idealTiles which would be rendered twice\n for (const key in idealRasterTileIDs) {\n if (idealRasterTileIDs[key].isChildOf(parent.tileID)) delete idealRasterTileIDs[key];\n }\n }\n }\n // cover all tiles which are not needed\n for (const key in this._tiles) {\n if (!idealRasterTileIDs[key]) this._coveredTiles[key] = true;\n }\n }\n }\n\n for (const retainedId in retain) {\n // Make sure retained tiles always clear any existing fade holds\n // so that if they're removed again their fade timer starts fresh.\n this._tiles[retainedId].clearFadeHold();\n }\n\n // Remove the tiles we don't need anymore.\n const remove = keysDifference(this._tiles, retain);\n for (const tileID of remove) {\n const tile = this._tiles[tileID];\n if (tile.hasSymbolBuckets && !tile.holdingForFade()) {\n tile.setHoldDuration(this.map._fadeDuration);\n } else if (!tile.hasSymbolBuckets || tile.symbolFadeFinished()) {\n this._removeTile(tileID);\n }\n }\n\n // Construct a cache of loaded parents\n this._updateLoadedParentTileCache();\n }\n\n releaseSymbolFadeTiles() {\n for (const id in this._tiles) {\n if (this._tiles[id].holdingForFade()) {\n this._removeTile(id);\n }\n }\n }\n\n _updateRetainedTiles(idealTileIDs: Array, zoom: number): {[_: string]: OverscaledTileID} {\n const retain: {[_: string]: OverscaledTileID} = {};\n const checked: {[_: string]: boolean} = {};\n const minCoveringZoom = Math.max(zoom - SourceCache.maxOverzooming, this._source.minzoom);\n const maxCoveringZoom = Math.max(zoom + SourceCache.maxUnderzooming, this._source.minzoom);\n\n const missingTiles = {};\n for (const tileID of idealTileIDs) {\n const tile = this._addTile(tileID);\n\n // retain the tile even if it's not loaded because it's an ideal tile.\n retain[tileID.key] = tileID;\n\n if (tile.hasData()) continue;\n\n if (zoom < this._source.maxzoom) {\n // save missing tiles that potentially have loaded children\n missingTiles[tileID.key] = tileID;\n }\n }\n\n // retain any loaded children of ideal tiles up to maxCoveringZoom\n this._retainLoadedChildren(missingTiles, zoom, maxCoveringZoom, retain);\n\n for (const tileID of idealTileIDs) {\n let tile = this._tiles[tileID.key];\n\n if (tile.hasData()) continue;\n\n // The tile we require is not yet loaded or does not exist;\n // Attempt to find children that fully cover it.\n\n if (zoom + 1 > this._source.maxzoom) {\n // We're looking for an overzoomed child tile.\n const childCoord = tileID.children(this._source.maxzoom)[0];\n const childTile = this.getTile(childCoord);\n if (!!childTile && childTile.hasData()) {\n retain[childCoord.key] = childCoord;\n continue; // tile is covered by overzoomed child\n }\n } else {\n // check if all 4 immediate children are loaded (i.e. the missing ideal tile is covered)\n const children = tileID.children(this._source.maxzoom);\n\n if (retain[children[0].key] &&\n retain[children[1].key] &&\n retain[children[2].key] &&\n retain[children[3].key]) continue; // tile is covered by children\n }\n\n // We couldn't find child tiles that entirely cover the ideal tile; look for parents now.\n\n // As we ascend up the tile pyramid of the ideal tile, we check whether the parent\n // tile has been previously requested (and errored because we only loop over tiles with no data)\n // in order to determine if we need to request its parent.\n let parentWasRequested = tile.wasRequested();\n\n for (let overscaledZ = tileID.overscaledZ - 1; overscaledZ >= minCoveringZoom; --overscaledZ) {\n const parentId = tileID.scaledTo(overscaledZ);\n\n // Break parent tile ascent if this route has been previously checked by another child.\n if (checked[parentId.key]) break;\n checked[parentId.key] = true;\n\n tile = this.getTile(parentId);\n if (!tile && parentWasRequested) {\n tile = this._addTile(parentId);\n }\n if (tile) {\n const hasData = tile.hasData();\n if (parentWasRequested || hasData) {\n retain[parentId.key] = parentId;\n }\n // Save the current values, since they're the parent of the next iteration\n // of the parent tile ascent loop.\n parentWasRequested = tile.wasRequested();\n if (hasData) break;\n }\n }\n }\n\n return retain;\n }\n\n _updateLoadedParentTileCache() {\n this._loadedParentTiles = {};\n\n for (const tileKey in this._tiles) {\n const path = [];\n let parentTile: Tile;\n let currentId = this._tiles[tileKey].tileID;\n\n // Find the closest loaded ancestor by traversing the tile tree towards the root and\n // caching results along the way\n while (currentId.overscaledZ > 0) {\n\n // Do we have a cached result from previous traversals?\n if (currentId.key in this._loadedParentTiles) {\n parentTile = this._loadedParentTiles[currentId.key];\n break;\n }\n\n path.push(currentId.key);\n\n // Is the parent loaded?\n const parentId = currentId.scaledTo(currentId.overscaledZ - 1);\n parentTile = this._getLoadedTile(parentId);\n if (parentTile) {\n break;\n }\n\n currentId = parentId;\n }\n\n // Cache the result of this traversal to all newly visited tiles\n for (const key of path) {\n this._loadedParentTiles[key] = parentTile;\n }\n }\n }\n\n /**\n * Add a tile, given its coordinate, to the pyramid.\n */\n _addTile(tileID: OverscaledTileID): Tile {\n let tile = this._tiles[tileID.key];\n if (tile)\n return tile;\n\n tile = this._cache.getAndRemove(tileID);\n if (tile) {\n this._setTileReloadTimer(tileID.key, tile);\n // set the tileID because the cached tile could have had a different wrap value\n tile.tileID = tileID;\n this._state.initializeTileState(tile, this.map ? this.map.painter : null);\n if (this._cacheTimers[tileID.key]) {\n clearTimeout(this._cacheTimers[tileID.key]);\n delete this._cacheTimers[tileID.key];\n this._setTileReloadTimer(tileID.key, tile);\n }\n }\n\n const cached = tile;\n\n if (!tile) {\n tile = new Tile(tileID, this._source.tileSize * tileID.overscaleFactor());\n this._loadTile(tile, this._tileLoaded.bind(this, tile, tileID.key, tile.state));\n }\n\n tile.uses++;\n this._tiles[tileID.key] = tile;\n if (!cached) {\n this._source.fire(new Event('dataloading', {tile, coord: tile.tileID, dataType: 'source'}));\n }\n\n return tile;\n }\n\n _setTileReloadTimer(id: string, tile: Tile) {\n if (id in this._timers) {\n clearTimeout(this._timers[id]);\n delete this._timers[id];\n }\n\n const expiryTimeout = tile.getExpiryTimeout();\n if (expiryTimeout) {\n this._timers[id] = setTimeout(() => {\n this._reloadTile(id, 'expired');\n delete this._timers[id];\n }, expiryTimeout);\n }\n }\n\n /**\n * Remove a tile, given its id, from the pyramid\n */\n _removeTile(id: string) {\n const tile = this._tiles[id];\n if (!tile)\n return;\n\n tile.uses--;\n delete this._tiles[id];\n if (this._timers[id]) {\n clearTimeout(this._timers[id]);\n delete this._timers[id];\n }\n\n if (tile.uses > 0)\n return;\n\n if (tile.hasData() && tile.state !== 'reloading') {\n this._cache.add(tile.tileID, tile, tile.getExpiryTimeout());\n } else {\n tile.aborted = true;\n this._abortTile(tile);\n this._unloadTile(tile);\n }\n }\n\n /**\n * Remove all tiles from this pyramid\n */\n clearTiles() {\n this._shouldReloadOnResume = false;\n this._paused = false;\n\n for (const id in this._tiles)\n this._removeTile(id);\n\n this._cache.reset();\n }\n\n /**\n * Search through our current tiles and attempt to find the tiles that\n * cover the given bounds.\n * @param pointQueryGeometry - coordinates of the corners of bounding rectangle\n * @returns result items have `{tile, minX, maxX, minY, maxY}`, where min/max bounding values are the given bounds transformed in into the coordinate space of this tile.\n */\n tilesIn(pointQueryGeometry: Array, maxPitchScaleFactor: number, has3DLayer: boolean): any[] {\n\n const tileResults = [];\n\n const transform = this.transform;\n if (!transform) return tileResults;\n\n const cameraPointQueryGeometry = has3DLayer ?\n transform.getCameraQueryGeometry(pointQueryGeometry) :\n pointQueryGeometry;\n\n const queryGeometry = pointQueryGeometry.map((p: Point) => transform.pointCoordinate(p, this.terrain));\n const cameraQueryGeometry = cameraPointQueryGeometry.map((p: Point) => transform.pointCoordinate(p, this.terrain));\n\n const ids = this.getIds();\n\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n\n for (const p of cameraQueryGeometry) {\n minX = Math.min(minX, p.x);\n minY = Math.min(minY, p.y);\n maxX = Math.max(maxX, p.x);\n maxY = Math.max(maxY, p.y);\n }\n\n for (let i = 0; i < ids.length; i++) {\n const tile = this._tiles[ids[i]];\n if (tile.holdingForFade()) {\n // Tiles held for fading are covered by tiles that are closer to ideal\n continue;\n }\n const tileID = tile.tileID;\n const scale = Math.pow(2, transform.zoom - tile.tileID.overscaledZ);\n const queryPadding = maxPitchScaleFactor * tile.queryPadding * EXTENT / tile.tileSize / scale;\n\n const tileSpaceBounds = [\n tileID.getTilePoint(new MercatorCoordinate(minX, minY)),\n tileID.getTilePoint(new MercatorCoordinate(maxX, maxY))\n ];\n\n if (tileSpaceBounds[0].x - queryPadding < EXTENT && tileSpaceBounds[0].y - queryPadding < EXTENT &&\n tileSpaceBounds[1].x + queryPadding >= 0 && tileSpaceBounds[1].y + queryPadding >= 0) {\n\n const tileSpaceQueryGeometry: Array = queryGeometry.map((c) => tileID.getTilePoint(c));\n const tileSpaceCameraQueryGeometry = cameraQueryGeometry.map((c) => tileID.getTilePoint(c));\n\n tileResults.push({\n tile,\n tileID,\n queryGeometry: tileSpaceQueryGeometry,\n cameraQueryGeometry: tileSpaceCameraQueryGeometry,\n scale\n });\n }\n }\n\n return tileResults;\n }\n\n getVisibleCoordinates(symbolLayer?: boolean): Array {\n const coords = this.getRenderableIds(symbolLayer).map((id) => this._tiles[id].tileID);\n for (const coord of coords) {\n coord.posMatrix = this.transform.calculatePosMatrix(coord.toUnwrapped());\n }\n return coords;\n }\n\n hasTransition() {\n if (this._source.hasTransition()) {\n return true;\n }\n\n if (isRasterType(this._source.type)) {\n const now = browser.now();\n for (const id in this._tiles) {\n const tile = this._tiles[id];\n if (tile.fadeEndTime >= now) {\n return true;\n }\n }\n }\n\n return false;\n }\n\n /**\n * Set the value of a particular state for a feature\n */\n setFeatureState(sourceLayer: string, featureId: number | string, state: any) {\n sourceLayer = sourceLayer || '_geojsonTileLayer';\n this._state.updateState(sourceLayer, featureId, state);\n }\n\n /**\n * Resets the value of a particular state key for a feature\n */\n removeFeatureState(sourceLayer?: string, featureId?: number | string, key?: string) {\n sourceLayer = sourceLayer || '_geojsonTileLayer';\n this._state.removeFeatureState(sourceLayer, featureId, key);\n }\n\n /**\n * Get the entire state object for a feature\n */\n getFeatureState(sourceLayer: string, featureId: number | string) {\n sourceLayer = sourceLayer || '_geojsonTileLayer';\n return this._state.getState(sourceLayer, featureId);\n }\n\n /**\n * Sets the set of keys that the tile depends on. This allows tiles to\n * be reloaded when their dependencies change.\n */\n setDependencies(tileKey: string, namespace: string, dependencies: Array) {\n const tile = this._tiles[tileKey];\n if (tile) {\n tile.setDependencies(namespace, dependencies);\n }\n }\n\n /**\n * Reloads all tiles that depend on the given keys.\n */\n reloadTilesForDependencies(namespaces: Array, keys: Array) {\n for (const id in this._tiles) {\n const tile = this._tiles[id];\n if (tile.hasDependency(namespaces, keys)) {\n this._reloadTile(id, 'reloading');\n }\n }\n this._cache.filter(tile => !tile.hasDependency(namespaces, keys));\n }\n}\n\nSourceCache.maxOverzooming = 10;\nSourceCache.maxUnderzooming = 3;\n\nfunction compareTileId(a: OverscaledTileID, b: OverscaledTileID): number {\n // Different copies of the world are sorted based on their distance to the center.\n // Wrap values are converted to unsigned distances by reserving odd number for copies\n // with negative wrap and even numbers for copies with positive wrap.\n const aWrap = Math.abs(a.wrap * 2) - +(a.wrap < 0);\n const bWrap = Math.abs(b.wrap * 2) - +(b.wrap < 0);\n return a.overscaledZ - b.overscaledZ || bWrap - aWrap || b.canonical.y - a.canonical.y || b.canonical.x - a.canonical.x;\n}\n\nfunction isRasterType(type) {\n return type === 'raster' || type === 'image' || type === 'video';\n}\n","import {config} from './config';\n\nimport type {WorkerSource} from '../source/worker_source';\n\nexport interface WorkerGlobalScopeInterface {\n importScripts(...urls: Array): void;\n registerWorkerSource: (\n b: string,\n a: {\n new(...args: any): WorkerSource;\n }\n ) => void;\n registerRTLTextPlugin: (_: any) => void;\n}\n\nexport function workerFactory() {\n return new Worker(config.WORKER_URL);\n}\n","import {workerFactory} from './web_worker';\nimport {browser} from './browser';\nimport {isSafari} from './util';\nimport {ActorTarget} from './actor';\n\nexport const PRELOAD_POOL_ID = 'mapboxgl_preloaded_worker_pool';\n\n/**\n * Constructs a worker pool.\n */\nexport class WorkerPool {\n static workerCount: number;\n\n active: {\n [_ in number | string]: boolean;\n };\n workers: Array;\n\n constructor() {\n this.active = {};\n }\n\n acquire(mapId: number | string): Array {\n if (!this.workers) {\n // Lazily look up the value of mapboxgl.workerCount so that\n // client code has had a chance to set it.\n this.workers = [];\n while (this.workers.length < WorkerPool.workerCount) {\n this.workers.push(workerFactory());\n }\n }\n\n this.active[mapId] = true;\n return this.workers.slice();\n }\n\n release(mapId: number | string) {\n delete this.active[mapId];\n if (this.numActive() === 0) {\n this.workers.forEach((w) => {\n w.terminate();\n });\n this.workers = null;\n }\n }\n\n isPreloaded(): boolean {\n return !!this.active[PRELOAD_POOL_ID];\n }\n\n numActive(): number {\n return Object.keys(this.active).length;\n }\n}\n\n// Based on results from A/B testing: https://github.com/maplibre/maplibre-gl-js/pull/2354\nconst availableLogicalProcessors = Math.floor(browser.hardwareConcurrency / 2);\nWorkerPool.workerCount = isSafari(globalThis) ? Math.max(Math.min(availableLogicalProcessors, 3), 1) : 1;\n","import {WorkerPool, PRELOAD_POOL_ID} from './worker_pool';\n\nlet globalWorkerPool;\n\n/**\n * Creates (if necessary) and returns the single, global WorkerPool instance\n * to be shared across each Map\n */\nexport function getGlobalWorkerPool() {\n if (!globalWorkerPool) {\n globalWorkerPool = new WorkerPool();\n }\n return globalWorkerPool;\n}\n\nexport function prewarm() {\n const workerPool = getGlobalWorkerPool();\n workerPool.acquire(PRELOAD_POOL_ID);\n}\n\nexport function clearPrewarmedResources() {\n const pool = globalWorkerPool;\n if (pool) {\n // Remove the pool only if all maps that referenced the preloaded global worker pool have been removed.\n if (pool.isPreloaded() && pool.numActive() === 1) {\n pool.release(PRELOAD_POOL_ID);\n globalWorkerPool = null;\n } else {\n console.warn('Could not clear WebWorkers since there are active Map instances that still reference it. The pre-warmed WebWorker pool can only be cleared when all map instances have been removed with map.remove()');\n }\n }\n}\n","import {clamp} from '../util/util';\nimport Point from '@mapbox/point-geometry';\n\nexport class PathInterpolator {\n points: Array;\n length: number;\n paddedLength: number;\n padding: number;\n _distances: Array;\n\n constructor(points_?: Array | null, padding_?: number | null) {\n this.reset(points_, padding_);\n }\n\n reset(points_?: Array | null, padding_?: number | null) {\n this.points = points_ || [];\n\n // Compute cumulative distance from first point to every other point in the segment.\n // Last entry in the array is total length of the path\n this._distances = [0.0];\n\n for (let i = 1; i < this.points.length; i++) {\n this._distances[i] = this._distances[i - 1] + this.points[i].dist(this.points[i - 1]);\n }\n\n this.length = this._distances[this._distances.length - 1];\n this.padding = Math.min(padding_ || 0, this.length * 0.5);\n this.paddedLength = this.length - this.padding * 2.0;\n }\n\n lerp(t: number): Point {\n if (this.points.length === 1) {\n return this.points[0];\n }\n\n t = clamp(t, 0, 1);\n\n // Find the correct segment [p0, p1] where p0 <= x < p1\n let currentIndex = 1;\n let distOfCurrentIdx = this._distances[currentIndex];\n const distToTarget = t * this.paddedLength + this.padding;\n\n while (distOfCurrentIdx < distToTarget && currentIndex < this._distances.length) {\n distOfCurrentIdx = this._distances[++currentIndex];\n }\n\n // Interpolate between the two points of the segment\n const idxOfPrevPoint = currentIndex - 1;\n const distOfPrevIdx = this._distances[idxOfPrevPoint];\n const segmentLength = distOfCurrentIdx - distOfPrevIdx;\n const segmentT = segmentLength > 0 ? (distToTarget - distOfPrevIdx) / segmentLength : 0;\n\n return this.points[idxOfPrevPoint].mult(1.0 - segmentT).add(this.points[currentIndex].mult(segmentT));\n }\n}\n","import type {OverlapMode} from '../style/style_layer/overlap_mode';\n\ntype QueryArgs = {\n hitTest: boolean;\n overlapMode?: OverlapMode;\n circle?: {\n x: number;\n y: number;\n radius: number;\n };\n seenUids: {\n box: {\n [_: number]: boolean;\n };\n circle: {\n [_: number]: boolean;\n };\n };\n};\n\ntype QueryResult = {\n key: T;\n x1: number;\n y1: number;\n x2: number;\n y2: number;\n};\n\n/**\n * A key for the grid\n */\nexport type GridKey = {\n overlapMode?: OverlapMode;\n}\n\nfunction overlapAllowed(overlapA: OverlapMode, overlapB: OverlapMode): boolean {\n let allowed = true;\n\n if (overlapA === 'always') {\n // symbol A using 'always' overlap - allowed to overlap anything.\n } else if (overlapA === 'never' || overlapB === 'never') {\n // symbol A using 'never' overlap - can't overlap anything\n // symbol A using 'cooperative' overlap - can overlap 'always' or 'cooperative' symbol; can't overlap 'never'\n allowed = false;\n }\n\n return allowed;\n}\n\n/**\n * @internal\n * GridIndex is a data structure for testing the intersection of\n * circles and rectangles in a 2d plane.\n * It is optimized for rapid insertion and querying.\n * GridIndex splits the plane into a set of \"cells\" and keeps track\n * of which geometries intersect with each cell. At query time,\n * full geometry comparisons are only done for items that share\n * at least one cell. As long as the geometries are relatively\n * uniformly distributed across the plane, this greatly reduces\n * the number of comparisons necessary.\n */\nexport class GridIndex {\n circleKeys: Array;\n boxKeys: Array;\n boxCells: Array>;\n circleCells: Array>;\n bboxes: Array;\n circles: Array;\n xCellCount: number;\n yCellCount: number;\n width: number;\n height: number;\n xScale: number;\n yScale: number;\n boxUid: number;\n circleUid: number;\n\n constructor (width: number, height: number, cellSize: number) {\n const boxCells = this.boxCells = [];\n const circleCells = this.circleCells = [];\n\n // More cells -> fewer geometries to check per cell, but items tend\n // to be split across more cells.\n // Sweet spot allows most small items to fit in one cell\n this.xCellCount = Math.ceil(width / cellSize);\n this.yCellCount = Math.ceil(height / cellSize);\n\n for (let i = 0; i < this.xCellCount * this.yCellCount; i++) {\n boxCells.push([]);\n circleCells.push([]);\n }\n this.circleKeys = [];\n this.boxKeys = [];\n this.bboxes = [];\n this.circles = [];\n\n this.width = width;\n this.height = height;\n this.xScale = this.xCellCount / width;\n this.yScale = this.yCellCount / height;\n this.boxUid = 0;\n this.circleUid = 0;\n }\n\n keysLength() {\n return this.boxKeys.length + this.circleKeys.length;\n }\n\n insert(key: T, x1: number, y1: number, x2: number, y2: number) {\n this._forEachCell(x1, y1, x2, y2, this._insertBoxCell, this.boxUid++);\n this.boxKeys.push(key);\n this.bboxes.push(x1);\n this.bboxes.push(y1);\n this.bboxes.push(x2);\n this.bboxes.push(y2);\n }\n\n insertCircle(key: T, x: number, y: number, radius: number) {\n // Insert circle into grid for all cells in the circumscribing square\n // It's more than necessary (by a factor of 4/PI), but fast to insert\n this._forEachCell(x - radius, y - radius, x + radius, y + radius, this._insertCircleCell, this.circleUid++);\n this.circleKeys.push(key);\n this.circles.push(x);\n this.circles.push(y);\n this.circles.push(radius);\n }\n\n private _insertBoxCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) {\n this.boxCells[cellIndex].push(uid);\n }\n\n private _insertCircleCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) {\n this.circleCells[cellIndex].push(uid);\n }\n\n private _query(x1: number, y1: number, x2: number, y2: number, hitTest: boolean, overlapMode: OverlapMode, predicate?: (key: T) => boolean): Array> {\n if (x2 < 0 || x1 > this.width || y2 < 0 || y1 > this.height) {\n return [];\n }\n const result: Array> = [];\n if (x1 <= 0 && y1 <= 0 && this.width <= x2 && this.height <= y2) {\n if (hitTest) {\n // Covers the entire grid, so collides with everything\n return [{\n key: null,\n x1,\n y1,\n x2,\n y2\n }];\n }\n for (let boxUid = 0; boxUid < this.boxKeys.length; boxUid++) {\n result.push({\n key: this.boxKeys[boxUid],\n x1: this.bboxes[boxUid * 4],\n y1: this.bboxes[boxUid * 4 + 1],\n x2: this.bboxes[boxUid * 4 + 2],\n y2: this.bboxes[boxUid * 4 + 3]\n });\n }\n for (let circleUid = 0; circleUid < this.circleKeys.length; circleUid++) {\n const x = this.circles[circleUid * 3];\n const y = this.circles[circleUid * 3 + 1];\n const radius = this.circles[circleUid * 3 + 2];\n result.push({\n key: this.circleKeys[circleUid],\n x1: x - radius,\n y1: y - radius,\n x2: x + radius,\n y2: y + radius\n });\n }\n } else {\n const queryArgs: QueryArgs = {\n hitTest,\n overlapMode,\n seenUids: {box: {}, circle: {}}\n };\n this._forEachCell(x1, y1, x2, y2, this._queryCell, result, queryArgs, predicate);\n }\n\n return result;\n }\n\n query(x1: number, y1: number, x2: number, y2: number): Array> {\n return this._query(x1, y1, x2, y2, false, null);\n }\n\n hitTest(x1: number, y1: number, x2: number, y2: number, overlapMode: OverlapMode, predicate?: (key: T) => boolean): boolean {\n return this._query(x1, y1, x2, y2, true, overlapMode, predicate).length > 0;\n }\n\n hitTestCircle(x: number, y: number, radius: number, overlapMode: OverlapMode, predicate?: (key: T) => boolean): boolean {\n // Insert circle into grid for all cells in the circumscribing square\n // It's more than necessary (by a factor of 4/PI), but fast to insert\n const x1 = x - radius;\n const x2 = x + radius;\n const y1 = y - radius;\n const y2 = y + radius;\n if (x2 < 0 || x1 > this.width || y2 < 0 || y1 > this.height) {\n return false;\n }\n\n // Box query early exits if the bounding box is larger than the grid, but we don't do\n // the equivalent calculation for circle queries because early exit is less likely\n // and the calculation is more expensive\n const result: boolean[] = [];\n const queryArgs: QueryArgs = {\n hitTest: true,\n overlapMode,\n circle: {x, y, radius},\n seenUids: {box: {}, circle: {}}\n };\n this._forEachCell(x1, y1, x2, y2, this._queryCellCircle, result, queryArgs, predicate);\n return result.length > 0;\n }\n\n private _queryCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, result: Array>, queryArgs: QueryArgs, predicate?: (key: T) => boolean): boolean {\n const {seenUids, hitTest, overlapMode} = queryArgs;\n const boxCell = this.boxCells[cellIndex];\n\n if (boxCell !== null) {\n const bboxes = this.bboxes;\n for (const boxUid of boxCell) {\n if (!seenUids.box[boxUid]) {\n seenUids.box[boxUid] = true;\n const offset = boxUid * 4;\n const key = this.boxKeys[boxUid];\n\n if ((x1 <= bboxes[offset + 2]) &&\n (y1 <= bboxes[offset + 3]) &&\n (x2 >= bboxes[offset + 0]) &&\n (y2 >= bboxes[offset + 1]) &&\n (!predicate || predicate(key))) {\n if (!hitTest || !overlapAllowed(overlapMode, key.overlapMode)) {\n result.push({\n key,\n x1: bboxes[offset],\n y1: bboxes[offset + 1],\n x2: bboxes[offset + 2],\n y2: bboxes[offset + 3]\n });\n if (hitTest) {\n // true return value stops the query after first match\n return true;\n }\n }\n }\n }\n }\n }\n const circleCell = this.circleCells[cellIndex];\n if (circleCell !== null) {\n const circles = this.circles;\n for (const circleUid of circleCell) {\n if (!seenUids.circle[circleUid]) {\n seenUids.circle[circleUid] = true;\n const offset = circleUid * 3;\n const key = this.circleKeys[circleUid];\n\n if (this._circleAndRectCollide(\n circles[offset],\n circles[offset + 1],\n circles[offset + 2],\n x1,\n y1,\n x2,\n y2) &&\n (!predicate || predicate(key))) {\n if (!hitTest || !overlapAllowed(overlapMode, key.overlapMode)) {\n const x = circles[offset];\n const y = circles[offset + 1];\n const radius = circles[offset + 2];\n result.push({\n key,\n x1: x - radius,\n y1: y - radius,\n x2: x + radius,\n y2: y + radius\n });\n if (hitTest) {\n // true return value stops the query after first match\n return true;\n }\n }\n }\n }\n }\n }\n\n // false return to continue query\n return false;\n }\n\n private _queryCellCircle(x1: number, y1: number, x2: number, y2: number, cellIndex: number, result: Array, queryArgs: QueryArgs, predicate?: (key: T) => boolean): boolean {\n const {circle, seenUids, overlapMode} = queryArgs;\n const boxCell = this.boxCells[cellIndex];\n\n if (boxCell !== null) {\n const bboxes = this.bboxes;\n for (const boxUid of boxCell) {\n if (!seenUids.box[boxUid]) {\n seenUids.box[boxUid] = true;\n const offset = boxUid * 4;\n const key = this.boxKeys[boxUid];\n if (this._circleAndRectCollide(\n circle.x,\n circle.y,\n circle.radius,\n bboxes[offset + 0],\n bboxes[offset + 1],\n bboxes[offset + 2],\n bboxes[offset + 3]) &&\n (!predicate || predicate(key)) &&\n !overlapAllowed(overlapMode, key.overlapMode)) {\n result.push(true);\n return true;\n }\n }\n }\n }\n\n const circleCell = this.circleCells[cellIndex];\n if (circleCell !== null) {\n const circles = this.circles;\n for (const circleUid of circleCell) {\n if (!seenUids.circle[circleUid]) {\n seenUids.circle[circleUid] = true;\n const offset = circleUid * 3;\n const key = this.circleKeys[circleUid];\n if (this._circlesCollide(\n circles[offset],\n circles[offset + 1],\n circles[offset + 2],\n circle.x,\n circle.y,\n circle.radius) &&\n (!predicate || predicate(key)) &&\n !overlapAllowed(overlapMode, key.overlapMode)) {\n result.push(true);\n return true;\n }\n }\n }\n }\n }\n\n private _forEachCell(\n x1: number,\n y1: number,\n x2: number,\n y2: number,\n fn: (x1: number, y1: number, x2: number, y2: number, cellIndex: number, arg1: TArg, arg2?: QueryArgs, predicate?: (key: T) => boolean) => boolean | void,\n arg1: TArg,\n arg2?: QueryArgs,\n predicate?: (key: T) => boolean) {\n const cx1 = this._convertToXCellCoord(x1);\n const cy1 = this._convertToYCellCoord(y1);\n const cx2 = this._convertToXCellCoord(x2);\n const cy2 = this._convertToYCellCoord(y2);\n\n for (let x = cx1; x <= cx2; x++) {\n for (let y = cy1; y <= cy2; y++) {\n const cellIndex = this.xCellCount * y + x;\n if (fn.call(this, x1, y1, x2, y2, cellIndex, arg1, arg2, predicate)) return;\n }\n }\n }\n\n private _convertToXCellCoord(x: number) {\n return Math.max(0, Math.min(this.xCellCount - 1, Math.floor(x * this.xScale)));\n }\n\n private _convertToYCellCoord(y: number) {\n return Math.max(0, Math.min(this.yCellCount - 1, Math.floor(y * this.yScale)));\n }\n\n private _circlesCollide(x1: number, y1: number, r1: number, x2: number, y2: number, r2: number): boolean {\n const dx = x2 - x1;\n const dy = y2 - y1;\n const bothRadii = r1 + r2;\n return (bothRadii * bothRadii) > (dx * dx + dy * dy);\n }\n\n private _circleAndRectCollide(\n circleX: number,\n circleY: number,\n radius: number,\n x1: number,\n y1: number,\n x2: number,\n y2: number\n ): boolean {\n const halfRectWidth = (x2 - x1) / 2;\n const distX = Math.abs(circleX - (x1 + halfRectWidth));\n if (distX > (halfRectWidth + radius)) {\n return false;\n }\n\n const halfRectHeight = (y2 - y1) / 2;\n const distY = Math.abs(circleY - (y1 + halfRectHeight));\n if (distY > (halfRectHeight + radius)) {\n return false;\n }\n\n if (distX <= halfRectWidth || distY <= halfRectHeight) {\n return true;\n }\n\n const dx = distX - halfRectWidth;\n const dy = distY - halfRectHeight;\n return (dx * dx + dy * dy <= (radius * radius));\n }\n}\n","import Point from '@mapbox/point-geometry';\n\nimport {mat4, vec4} from 'gl-matrix';\nimport * as symbolSize from './symbol_size';\nimport {addDynamicAttributes} from '../data/bucket/symbol_bucket';\n\nimport type {Painter} from '../render/painter';\nimport type {Transform} from '../geo/transform';\nimport type {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport type {\n GlyphOffsetArray,\n SymbolLineVertexArray,\n SymbolDynamicLayoutArray\n} from '../data/array_types.g';\nimport {WritingMode} from '../symbol/shaping';\nimport {findLineIntersection} from '../util/util';\n\nexport {updateLineLabels, hideGlyphs, getLabelPlaneMatrix, getGlCoordMatrix, project, getPerspectiveRatio, placeFirstAndLastGlyph, placeGlyphAlongLine, xyTransformMat4, projectVertexToViewport, findOffsetIntersectionPoint, transformToOffsetNormal};\n\n/*\n * # Overview of coordinate spaces\n *\n * ## Tile coordinate spaces\n * Each label has an anchor. Some labels have corresponding line geometries.\n * The points for both anchors and lines are stored in tile units. Each tile has it's own\n * coordinate space going from (0, 0) at the top left to (EXTENT, EXTENT) at the bottom right.\n *\n * ## GL coordinate space\n * At the end of everything, the vertex shader needs to produce a position in GL coordinate space,\n * which is (-1, 1) at the top left and (1, -1) in the bottom right.\n *\n * ## Map pixel coordinate spaces\n * Each tile has a pixel coordinate space. It's just the tile units scaled so that one unit is\n * whatever counts as 1 pixel at the current zoom.\n * This space is used for pitch-alignment=map, rotation-alignment=map\n *\n * ## Rotated map pixel coordinate spaces\n * Like the above, but rotated so axis of the space are aligned with the viewport instead of the tile.\n * This space is used for pitch-alignment=map, rotation-alignment=viewport\n *\n * ## Viewport pixel coordinate space\n * (0, 0) is at the top left of the canvas and (pixelWidth, pixelHeight) is at the bottom right corner\n * of the canvas. This space is used for pitch-alignment=viewport\n *\n *\n * # Vertex projection\n * It goes roughly like this:\n * 1. project the anchor and line from tile units into the correct label coordinate space\n * - map pixel space pitch-alignment=map rotation-alignment=map\n * - rotated map pixel space pitch-alignment=map rotation-alignment=viewport\n * - viewport pixel space pitch-alignment=viewport rotation-alignment=*\n * 2. if the label follows a line, find the point along the line that is the correct distance from the anchor.\n * 3. add the glyph's corner offset to the point from step 3\n * 4. convert from the label coordinate space to gl coordinates\n *\n * For horizontal labels we want to do step 1 in the shader for performance reasons (no cpu work).\n * This is what `u_label_plane_matrix` is used for.\n * For labels aligned with lines we have to steps 1 and 2 on the cpu since we need access to the line geometry.\n * This is what `updateLineLabels(...)` does.\n * Since the conversion is handled on the cpu we just set `u_label_plane_matrix` to an identity matrix.\n *\n * Steps 3 and 4 are done in the shaders for all labels.\n */\n\n/*\n * Returns a matrix for converting from tile units to the correct label coordinate space.\n */\nfunction getLabelPlaneMatrix(posMatrix: mat4,\n pitchWithMap: boolean,\n rotateWithMap: boolean,\n transform: Transform,\n pixelsToTileUnits: number) {\n const m = mat4.create();\n if (pitchWithMap) {\n mat4.scale(m, m, [1 / pixelsToTileUnits, 1 / pixelsToTileUnits, 1]);\n if (!rotateWithMap) {\n mat4.rotateZ(m, m, transform.angle);\n }\n } else {\n mat4.multiply(m, transform.labelPlaneMatrix, posMatrix);\n }\n return m;\n}\n\n/*\n * Returns a matrix for converting from the correct label coordinate space to gl coords.\n */\nfunction getGlCoordMatrix(posMatrix: mat4,\n pitchWithMap: boolean,\n rotateWithMap: boolean,\n transform: Transform,\n pixelsToTileUnits: number) {\n if (pitchWithMap) {\n const m = mat4.clone(posMatrix);\n mat4.scale(m, m, [pixelsToTileUnits, pixelsToTileUnits, 1]);\n if (!rotateWithMap) {\n mat4.rotateZ(m, m, -transform.angle);\n }\n return m;\n } else {\n return transform.glCoordMatrix;\n }\n}\n\nfunction project(point: Point, matrix: mat4, getElevation?: (x: number, y: number) => number) {\n let pos;\n if (getElevation) { // slow because of handle z-index\n pos = [point.x, point.y, getElevation(point.x, point.y), 1] as vec4;\n vec4.transformMat4(pos, pos, matrix);\n } else { // fast because of ignore z-index\n pos = [point.x, point.y, 0, 1] as vec4;\n xyTransformMat4(pos, pos, matrix);\n }\n const w = pos[3];\n return {\n point: new Point(pos[0] / w, pos[1] / w),\n signedDistanceFromCamera: w\n };\n}\n\nfunction getPerspectiveRatio(cameraToCenterDistance: number, signedDistanceFromCamera: number): number {\n return 0.5 + 0.5 * (cameraToCenterDistance / signedDistanceFromCamera);\n}\n\nfunction isVisible(anchorPos: vec4,\n clippingBuffer: [number, number]) {\n const x = anchorPos[0] / anchorPos[3];\n const y = anchorPos[1] / anchorPos[3];\n const inPaddedViewport = (\n x >= -clippingBuffer[0] &&\n x <= clippingBuffer[0] &&\n y >= -clippingBuffer[1] &&\n y <= clippingBuffer[1]);\n return inPaddedViewport;\n}\n\n/*\n * Update the `dynamicLayoutVertexBuffer` for the buffer with the correct glyph positions for the current map view.\n * This is only run on labels that are aligned with lines. Horizontal labels are handled entirely in the shader.\n */\nfunction updateLineLabels(bucket: SymbolBucket,\n posMatrix: mat4,\n painter: Painter,\n isText: boolean,\n labelPlaneMatrix: mat4,\n glCoordMatrix: mat4,\n pitchWithMap: boolean,\n keepUpright: boolean,\n rotateToLine: boolean,\n getElevation: (x: number, y: number) => number) {\n\n const sizeData = isText ? bucket.textSizeData : bucket.iconSizeData;\n const partiallyEvaluatedSize = symbolSize.evaluateSizeForZoom(sizeData, painter.transform.zoom);\n\n const clippingBuffer: [number, number] = [256 / painter.width * 2 + 1, 256 / painter.height * 2 + 1];\n\n const dynamicLayoutVertexArray = isText ?\n bucket.text.dynamicLayoutVertexArray :\n bucket.icon.dynamicLayoutVertexArray;\n dynamicLayoutVertexArray.clear();\n\n const lineVertexArray = bucket.lineVertexArray;\n const placedSymbols = isText ? bucket.text.placedSymbolArray : bucket.icon.placedSymbolArray;\n\n const aspectRatio = painter.transform.width / painter.transform.height;\n\n let useVertical = false;\n\n for (let s = 0; s < placedSymbols.length; s++) {\n const symbol = placedSymbols.get(s);\n\n // Don't do calculations for vertical glyphs unless the previous symbol was horizontal\n // and we determined that vertical glyphs were necessary.\n // Also don't do calculations for symbols that are collided and fully faded out\n if (symbol.hidden || symbol.writingMode === WritingMode.vertical && !useVertical) {\n hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray);\n continue;\n }\n // Awkward... but we're counting on the paired \"vertical\" symbol coming immediately after its horizontal counterpart\n useVertical = false;\n\n let anchorPos;\n if (getElevation) { // slow because of handle z-index\n anchorPos = [symbol.anchorX, symbol.anchorY, getElevation(symbol.anchorX, symbol.anchorY), 1] as vec4;\n vec4.transformMat4(anchorPos, anchorPos, posMatrix);\n } else { // fast because of ignore z-index\n anchorPos = [symbol.anchorX, symbol.anchorY, 0, 1] as vec4;\n xyTransformMat4(anchorPos, anchorPos, posMatrix);\n }\n\n // Don't bother calculating the correct point for invisible labels.\n if (!isVisible(anchorPos, clippingBuffer)) {\n hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray);\n continue;\n }\n\n const cameraToAnchorDistance = anchorPos[3];\n const perspectiveRatio = getPerspectiveRatio(painter.transform.cameraToCenterDistance, cameraToAnchorDistance);\n\n const fontSize = symbolSize.evaluateSizeForFeature(sizeData, partiallyEvaluatedSize, symbol);\n const pitchScaledFontSize = pitchWithMap ? fontSize / perspectiveRatio : fontSize * perspectiveRatio;\n\n const tileAnchorPoint = new Point(symbol.anchorX, symbol.anchorY);\n const anchorPoint = project(tileAnchorPoint, labelPlaneMatrix, getElevation).point;\n const projectionCache = {projections: {}, offsets: {}};\n\n const placeUnflipped: any = placeGlyphsAlongLine(symbol, pitchScaledFontSize, false /*unflipped*/, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix,\n bucket.glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation);\n\n useVertical = placeUnflipped.useVertical;\n\n if (placeUnflipped.notEnoughRoom || useVertical ||\n (placeUnflipped.needsFlipping &&\n (placeGlyphsAlongLine(symbol, pitchScaledFontSize, true /*flipped*/, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix,\n bucket.glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation) as any).notEnoughRoom)) {\n hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray);\n }\n }\n\n if (isText) {\n bucket.text.dynamicLayoutVertexBuffer.updateData(dynamicLayoutVertexArray);\n } else {\n bucket.icon.dynamicLayoutVertexBuffer.updateData(dynamicLayoutVertexArray);\n }\n}\n\ntype FirstAndLastGlyphPlacement = {\n first: PlacedGlyph;\n last: PlacedGlyph;\n} | null;\n\n/*\n * Place the first and last glyph of a line label, projected to the label plane.\n * This function is called both during collision detection (to determine the label's size)\n * and during line label rendering (to make sure the label fits on the line geometry with\n * the current camera position, which may differ from the position used during collision detection).\n *\n * Calling this function has the effect of populating the \"projectionCache\" with all projected\n * vertex locations the label will need, making future calls to placeGlyphAlongLine (for all the\n * intermediate glyphs) much cheaper.\n *\n * Returns null if the label can't fit on the geometry\n */\nfunction placeFirstAndLastGlyph(fontScale: number, glyphOffsetArray: GlyphOffsetArray, lineOffsetX: number, lineOffsetY: number, flip: boolean, anchorPoint: Point, tileAnchorPoint: Point, symbol: any, lineVertexArray: SymbolLineVertexArray, labelPlaneMatrix: mat4, projectionCache: ProjectionCache, rotateToLine: boolean, getElevation: (x: number, y: number) => number): FirstAndLastGlyphPlacement {\n const glyphEndIndex = symbol.glyphStartIndex + symbol.numGlyphs;\n const lineStartIndex = symbol.lineStartIndex;\n const lineEndIndex = symbol.lineStartIndex + symbol.lineLength;\n\n const firstGlyphOffset = glyphOffsetArray.getoffsetX(symbol.glyphStartIndex);\n const lastGlyphOffset = glyphOffsetArray.getoffsetX(glyphEndIndex - 1);\n\n const firstPlacedGlyph = placeGlyphAlongLine(fontScale * firstGlyphOffset, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment,\n lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation);\n if (!firstPlacedGlyph)\n return null;\n\n const lastPlacedGlyph = placeGlyphAlongLine(fontScale * lastGlyphOffset, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment,\n lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation);\n if (!lastPlacedGlyph)\n return null;\n\n return {first: firstPlacedGlyph, last: lastPlacedGlyph};\n}\n\nfunction requiresOrientationChange(writingMode, firstPoint, lastPoint, aspectRatio) {\n if (writingMode === WritingMode.horizontal) {\n // On top of choosing whether to flip, choose whether to render this version of the glyphs or the alternate\n // vertical glyphs. We can't just filter out vertical glyphs in the horizontal range because the horizontal\n // and vertical versions can have slightly different projections which could lead to angles where both or\n // neither showed.\n const rise = Math.abs(lastPoint.y - firstPoint.y);\n const run = Math.abs(lastPoint.x - firstPoint.x) * aspectRatio;\n if (rise > run) {\n return {useVertical: true};\n }\n }\n\n if (writingMode === WritingMode.vertical ? firstPoint.y < lastPoint.y : firstPoint.x > lastPoint.x) {\n // Includes \"horizontalOnly\" case for labels without vertical glyphs\n return {needsFlipping: true};\n }\n\n return null;\n}\n\n/*\n* Place first and last glyph along the line projected to label plane, and if they fit\n* iterate through all the intermediate glyphs, calculating their label plane positions\n* from the projected line.\n*\n* Finally, add resulting glyph position calculations to dynamicLayoutVertexArray for\n* upload to the GPU\n*/\nfunction placeGlyphsAlongLine(symbol, fontSize, flip, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix, glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation) {\n const fontScale = fontSize / 24;\n const lineOffsetX = symbol.lineOffsetX * fontScale;\n const lineOffsetY = symbol.lineOffsetY * fontScale;\n\n let placedGlyphs;\n if (symbol.numGlyphs > 1) {\n const glyphEndIndex = symbol.glyphStartIndex + symbol.numGlyphs;\n const lineStartIndex = symbol.lineStartIndex;\n const lineEndIndex = symbol.lineStartIndex + symbol.lineLength;\n\n // Place the first and the last glyph in the label first, so we can figure out\n // the overall orientation of the label and determine whether it needs to be flipped in keepUpright mode\n const firstAndLastGlyph = placeFirstAndLastGlyph(fontScale, glyphOffsetArray, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation);\n if (!firstAndLastGlyph) {\n return {notEnoughRoom: true};\n }\n const firstPoint = project(firstAndLastGlyph.first.point, glCoordMatrix, getElevation).point;\n const lastPoint = project(firstAndLastGlyph.last.point, glCoordMatrix, getElevation).point;\n\n if (keepUpright && !flip) {\n const orientationChange = requiresOrientationChange(symbol.writingMode, firstPoint, lastPoint, aspectRatio);\n if (orientationChange) {\n return orientationChange;\n }\n }\n\n placedGlyphs = [firstAndLastGlyph.first];\n for (let glyphIndex = symbol.glyphStartIndex + 1; glyphIndex < glyphEndIndex - 1; glyphIndex++) {\n // Since first and last glyph fit on the line, we're sure that the rest of the glyphs can be placed\n placedGlyphs.push(placeGlyphAlongLine(fontScale * glyphOffsetArray.getoffsetX(glyphIndex), lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment,\n lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation));\n }\n placedGlyphs.push(firstAndLastGlyph.last);\n } else {\n // Only a single glyph to place\n // So, determine whether to flip based on projected angle of the line segment it's on\n if (keepUpright && !flip) {\n const a = project(tileAnchorPoint, posMatrix, getElevation).point;\n const tileVertexIndex = (symbol.lineStartIndex + symbol.segment + 1);\n const tileSegmentEnd = new Point(lineVertexArray.getx(tileVertexIndex), lineVertexArray.gety(tileVertexIndex));\n const projectedVertex = project(tileSegmentEnd, posMatrix, getElevation);\n // We know the anchor will be in the viewport, but the end of the line segment may be\n // behind the plane of the camera, in which case we can use a point at any arbitrary (closer)\n // point on the segment.\n const b = (projectedVertex.signedDistanceFromCamera > 0) ?\n projectedVertex.point :\n projectTruncatedLineSegment(tileAnchorPoint, tileSegmentEnd, a, 1, posMatrix, getElevation);\n\n const orientationChange = requiresOrientationChange(symbol.writingMode, a, b, aspectRatio);\n if (orientationChange) {\n return orientationChange;\n }\n }\n const singleGlyph = placeGlyphAlongLine(fontScale * glyphOffsetArray.getoffsetX(symbol.glyphStartIndex), lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment,\n symbol.lineStartIndex, symbol.lineStartIndex + symbol.lineLength, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation);\n if (!singleGlyph)\n return {notEnoughRoom: true};\n\n placedGlyphs = [singleGlyph];\n }\n\n for (const glyph of placedGlyphs) {\n addDynamicAttributes(dynamicLayoutVertexArray, glyph.point, glyph.angle);\n }\n return {};\n}\n\nfunction projectTruncatedLineSegment(previousTilePoint: Point, currentTilePoint: Point, previousProjectedPoint: Point, minimumLength: number, projectionMatrix: mat4, getElevation: (x: number, y: number) => number) {\n // We are assuming \"previousTilePoint\" won't project to a point within one unit of the camera plane\n // If it did, that would mean our label extended all the way out from within the viewport to a (very distant)\n // point near the plane of the camera. We wouldn't be able to render the label anyway once it crossed the\n // plane of the camera.\n const projectedUnitVertex = project(previousTilePoint.add(previousTilePoint.sub(currentTilePoint)._unit()), projectionMatrix, getElevation).point;\n const projectedUnitSegment = previousProjectedPoint.sub(projectedUnitVertex);\n\n return previousProjectedPoint.add(projectedUnitSegment._mult(minimumLength / projectedUnitSegment.mag()));\n}\n\ntype IndexToPointCache = { [lineIndex: number]: Point };\n\n/**\n * We calculate label-plane projected points for line vertices as we place glyphs along the line\n * Since we will use the same vertices for potentially many glyphs, cache the results for this bucket\n * over the course of the render. Each vertex location also potentially has one offset equivalent\n * for us to hold onto. The vertex indices are per-symbol-bucket.\n */\ntype ProjectionCache = {\n /**\n * tile-unit vertices projected into label-plane units\n */\n projections: IndexToPointCache;\n /**\n * label-plane vertices which have been shifted to follow an offset line\n */\n offsets: IndexToPointCache;\n};\n\n/**\n * Arguments necessary to project a vertex to the label plane\n */\ntype ProjectionArgs = {\n /**\n * Used to cache results, save cost if projecting the same vertex multiple times\n */\n projectionCache: ProjectionCache;\n /**\n * The array of tile-unit vertices transferred from worker\n */\n lineVertexArray: SymbolLineVertexArray;\n /**\n * Label plane projection matrix\n */\n labelPlaneMatrix: mat4;\n /**\n * Function to get elevation at a point\n * @param x - the x coordinate\n * @param y - the y coordinate\n */\n getElevation: (x: number, y: number) => number;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n tileAnchorPoint: Point;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n distanceFromAnchor: number;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n previousVertex: Point;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n direction: number;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n absOffsetX: number;\n};\n\n/**\n * Transform a vertex from tile coordinates to label plane coordinates\n * @param index - index of vertex to project\n * @param projectionArgs - necessary data to project a vertex\n * @returns the vertex projected to the label plane\n */\nfunction projectVertexToViewport(index: number, projectionArgs: ProjectionArgs): Point {\n const {projectionCache, lineVertexArray, labelPlaneMatrix, tileAnchorPoint, distanceFromAnchor, getElevation, previousVertex, direction, absOffsetX} = projectionArgs;\n if (projectionCache.projections[index]) {\n return projectionCache.projections[index];\n }\n const currentVertex = new Point(lineVertexArray.getx(index), lineVertexArray.gety(index));\n const projection = project(currentVertex, labelPlaneMatrix, getElevation);\n if (projection.signedDistanceFromCamera > 0) {\n projectionCache.projections[index] = projection.point;\n return projection.point;\n }\n\n // The vertex is behind the plane of the camera, so we can't project it\n // Instead, we'll create a vertex along the line that's far enough to include the glyph\n const previousLineVertexIndex = index - direction;\n const previousTilePoint = distanceFromAnchor === 0 ?\n tileAnchorPoint :\n new Point(lineVertexArray.getx(previousLineVertexIndex), lineVertexArray.gety(previousLineVertexIndex));\n // Don't cache because the new vertex might not be far enough out for future glyphs on the same segment\n return projectTruncatedLineSegment(previousTilePoint, currentVertex, previousVertex, absOffsetX - distanceFromAnchor + 1, labelPlaneMatrix, getElevation);\n}\n\n/**\n * Calculate the normal vector for a line segment\n * @param segmentVector - will be mutated as a tiny optimization\n * @param offset - magnitude of resulting vector\n * @param direction - direction of line traversal\n * @returns a normal vector from the segment, with magnitude equal to offset amount\n */\nfunction transformToOffsetNormal(segmentVector: Point, offset: number, direction: number): Point {\n return segmentVector._unit()._perp()._mult(offset * direction);\n}\n\n/**\n * Construct offset line segments for the current segment and the next segment, then extend/shrink\n * the segments until they intersect. If the segments are parallel, then they will touch with no modification.\n *\n * @param index - Index of the current vertex\n * @param prevToCurrentOffsetNormal - Normal vector of the line segment from the previous vertex to the current vertex\n * @param currentVertex - Current (non-offset) vertex projected to the label plane\n * @param lineStartIndex - Beginning index for the line this label is on\n * @param lineEndIndex - End index for the line this label is on\n * @param offsetPreviousVertex - The previous vertex projected to the label plane, and then offset along the previous segments normal\n * @param lineOffsetY - Magnitude of the offset\n * @param projectionArgs - Necessary data for tile-to-label-plane projection\n * @returns The point at which the current and next line segments intersect, once offset and extended/shrunk to their meeting point\n */\nfunction findOffsetIntersectionPoint(index: number, prevToCurrentOffsetNormal: Point, currentVertex: Point, lineStartIndex: number, lineEndIndex: number, offsetPreviousVertex: Point, lineOffsetY: number, projectionArgs: ProjectionArgs) {\n const {projectionCache, direction} = projectionArgs;\n if (projectionCache.offsets[index]) {\n return projectionCache.offsets[index];\n }\n\n const offsetCurrentVertex = currentVertex.add(prevToCurrentOffsetNormal);\n\n if (index + direction < lineStartIndex || index + direction >= lineEndIndex) {\n // This is the end of the line, no intersection to calculate\n projectionCache.offsets[index] = offsetCurrentVertex;\n return offsetCurrentVertex;\n }\n // Offset the vertices for the next segment\n const nextVertex = projectVertexToViewport(index + direction, projectionArgs);\n const currentToNextOffsetNormal = transformToOffsetNormal(nextVertex.sub(currentVertex), lineOffsetY, direction);\n const offsetNextSegmentBegin = currentVertex.add(currentToNextOffsetNormal);\n const offsetNextSegmentEnd = nextVertex.add(currentToNextOffsetNormal);\n\n // find the intersection of these two lines\n // if the lines are parallel, offsetCurrent/offsetNextBegin will touch\n projectionCache.offsets[index] = findLineIntersection(offsetPreviousVertex, offsetCurrentVertex, offsetNextSegmentBegin, offsetNextSegmentEnd) || offsetCurrentVertex;\n\n return projectionCache.offsets[index];\n}\n\n/**\n * Placed Glyph type\n */\ntype PlacedGlyph = {\n /**\n * The point at which the glyph should be placed, in label plane coordinates\n */\n point: Point;\n /**\n * The angle at which the glyph should be placed\n */\n angle: number;\n /**\n * The label-plane path used to reach this glyph: used only for collision detection\n */\n path: Array;\n};\n\n/*\n * Place a single glyph along its line, projected into the label plane, by iterating outward\n * from the anchor point until the distance traversed in the label plane equals the glyph's\n * offsetX. Returns null if the glyph can't fit on the line geometry.\n */\nfunction placeGlyphAlongLine(\n offsetX: number,\n lineOffsetX: number,\n lineOffsetY: number,\n flip: boolean,\n anchorPoint: Point,\n tileAnchorPoint: Point,\n anchorSegment: number,\n lineStartIndex: number,\n lineEndIndex: number,\n lineVertexArray: SymbolLineVertexArray,\n labelPlaneMatrix: mat4,\n projectionCache: ProjectionCache,\n rotateToLine: boolean,\n getElevation: (x: number, y: number) => number): PlacedGlyph | null {\n\n const combinedOffsetX = flip ?\n offsetX - lineOffsetX :\n offsetX + lineOffsetX;\n\n let direction = combinedOffsetX > 0 ? 1 : -1;\n\n let angle = 0;\n if (flip) {\n // The label needs to be flipped to keep text upright.\n // Iterate in the reverse direction.\n direction *= -1;\n angle = Math.PI;\n }\n\n if (direction < 0) angle += Math.PI;\n\n let currentIndex = direction > 0 ?\n lineStartIndex + anchorSegment :\n lineStartIndex + anchorSegment + 1;\n\n let currentVertex = anchorPoint;\n let previousVertex = anchorPoint;\n\n // offsetPrev and intersectionPoint are analogous to previousVertex and currentVertex\n // but if there's a line offset they are calculated in parallel as projection happens\n let offsetIntersectionPoint: Point;\n let offsetPreviousVertex: Point;\n\n let distanceFromAnchor = 0;\n let currentSegmentDistance = 0;\n const absOffsetX = Math.abs(combinedOffsetX);\n const pathVertices: Array = [];\n\n let currentLineSegment: Point;\n while (distanceFromAnchor + currentSegmentDistance <= absOffsetX) {\n currentIndex += direction;\n\n // offset does not fit on the projected line\n if (currentIndex < lineStartIndex || currentIndex >= lineEndIndex)\n return null;\n\n // accumulate values from last iteration\n distanceFromAnchor += currentSegmentDistance;\n previousVertex = currentVertex;\n offsetPreviousVertex = offsetIntersectionPoint;\n\n const projectionArgs: ProjectionArgs = {\n projectionCache,\n lineVertexArray,\n labelPlaneMatrix,\n tileAnchorPoint,\n distanceFromAnchor,\n getElevation,\n previousVertex,\n direction,\n absOffsetX\n };\n\n // find next vertex in viewport space\n currentVertex = projectVertexToViewport(currentIndex, projectionArgs);\n if (lineOffsetY === 0) {\n // Store vertices for collision detection and update current segment geometry\n pathVertices.push(previousVertex);\n currentLineSegment = currentVertex.sub(previousVertex);\n } else {\n // Calculate the offset for this section\n let prevToCurrentOffsetNormal;\n const prevToCurrent = currentVertex.sub(previousVertex);\n if (prevToCurrent.mag() === 0) {\n // We are starting with our anchor point directly on the vertex, so look one vertex ahead\n // to calculate a normal\n const nextVertex = projectVertexToViewport(currentIndex + direction, projectionArgs);\n prevToCurrentOffsetNormal = transformToOffsetNormal(nextVertex.sub(currentVertex), lineOffsetY, direction);\n } else {\n prevToCurrentOffsetNormal = transformToOffsetNormal(prevToCurrent, lineOffsetY, direction);\n }\n // Initialize offsetPrev on our first iteration, after that it will be pre-calculated\n if (!offsetPreviousVertex)\n offsetPreviousVertex = previousVertex.add(prevToCurrentOffsetNormal);\n\n offsetIntersectionPoint = findOffsetIntersectionPoint(currentIndex, prevToCurrentOffsetNormal, currentVertex, lineStartIndex, lineEndIndex, offsetPreviousVertex, lineOffsetY, projectionArgs);\n\n pathVertices.push(offsetPreviousVertex);\n currentLineSegment = offsetIntersectionPoint.sub(offsetPreviousVertex);\n }\n currentSegmentDistance = currentLineSegment.mag();\n }\n\n // The point is on the current segment. Interpolate to find it.\n const segmentInterpolationT = (absOffsetX - distanceFromAnchor) / currentSegmentDistance;\n const p = currentLineSegment._mult(segmentInterpolationT)._add(offsetPreviousVertex || previousVertex);\n\n const segmentAngle = angle + Math.atan2(currentVertex.y - previousVertex.y, currentVertex.x - previousVertex.x);\n\n pathVertices.push(p);\n\n return {\n point: p,\n angle: rotateToLine ? segmentAngle : 0.0,\n path: pathVertices\n };\n}\n\nconst hiddenGlyphAttributes = new Float32Array([-Infinity, -Infinity, 0, -Infinity, -Infinity, 0, -Infinity, -Infinity, 0, -Infinity, -Infinity, 0]);\n\n// Hide them by moving them offscreen. We still need to add them to the buffer\n// because the dynamic buffer is paired with a static buffer that doesn't get updated.\nfunction hideGlyphs(num: number, dynamicLayoutVertexArray: SymbolDynamicLayoutArray) {\n for (let i = 0; i < num; i++) {\n const offset = dynamicLayoutVertexArray.length;\n dynamicLayoutVertexArray.resize(offset + 4);\n // Since all hidden glyphs have the same attributes, we can build up the array faster with a single call to Float32Array.set\n // for each set of four vertices, instead of calling addDynamicAttributes for each vertex.\n dynamicLayoutVertexArray.float32.set(hiddenGlyphAttributes, offset * 3);\n }\n}\n\n// For line label layout, we're not using z output and our w input is always 1\n// This custom matrix transformation ignores those components to make projection faster\nfunction xyTransformMat4(out: vec4, a: vec4, m: mat4) {\n const x = a[0], y = a[1];\n out[0] = m[0] * x + m[4] * y + m[12];\n out[1] = m[1] * x + m[5] * y + m[13];\n out[3] = m[3] * x + m[7] * y + m[15];\n return out;\n}\n","import Point from '@mapbox/point-geometry';\nimport {clipLine} from './clip_line';\nimport {PathInterpolator} from './path_interpolator';\n\nimport * as intersectionTests from '../util/intersection_tests';\nimport {GridIndex} from './grid_index';\nimport {mat4, vec4} from 'gl-matrix';\nimport ONE_EM from '../symbol/one_em';\n\nimport * as projection from '../symbol/projection';\n\nimport type {Transform} from '../geo/transform';\nimport type {SingleCollisionBox} from '../data/bucket/symbol_bucket';\nimport type {\n GlyphOffsetArray,\n SymbolLineVertexArray\n} from '../data/array_types.g';\nimport type {OverlapMode} from '../style/style_layer/overlap_mode';\n\n// When a symbol crosses the edge that causes it to be included in\n// collision detection, it will cause changes in the symbols around\n// it. This constant specifies how many pixels to pad the edge of\n// the viewport for collision detection so that the bulk of the changes\n// occur offscreen. Making this constant greater increases label\n// stability, but it's expensive.\nconst viewportPadding = 100;\n\nexport type FeatureKey = {\n bucketInstanceId: number;\n featureIndex: number;\n collisionGroupID: number;\n overlapMode: OverlapMode;\n};\n\n/**\n * @internal\n * A collision index used to prevent symbols from overlapping. It keep tracks of\n * where previous symbols have been placed and is used to check if a new\n * symbol overlaps with any previously added symbols.\n *\n * There are two steps to insertion: first placeCollisionBox/Circles checks if\n * there's room for a symbol, then insertCollisionBox/Circles actually puts the\n * symbol in the index. The two step process allows paired symbols to be inserted\n * together even if they overlap.\n */\nexport class CollisionIndex {\n grid: GridIndex;\n ignoredGrid: GridIndex;\n transform: Transform;\n pitchfactor: number;\n screenRightBoundary: number;\n screenBottomBoundary: number;\n gridRightBoundary: number;\n gridBottomBoundary: number;\n\n // With perspectiveRatio the fontsize is calculated for tilted maps (near = bigger, far = smaller).\n // The cutoff defines a threshold to no longer render labels near the horizon.\n perspectiveRatioCutoff: number;\n\n constructor(\n transform: Transform,\n grid = new GridIndex(transform.width + 2 * viewportPadding, transform.height + 2 * viewportPadding, 25),\n ignoredGrid = new GridIndex(transform.width + 2 * viewportPadding, transform.height + 2 * viewportPadding, 25)\n ) {\n this.transform = transform;\n\n this.grid = grid;\n this.ignoredGrid = ignoredGrid;\n this.pitchfactor = Math.cos(transform._pitch) * transform.cameraToCenterDistance;\n\n this.screenRightBoundary = transform.width + viewportPadding;\n this.screenBottomBoundary = transform.height + viewportPadding;\n this.gridRightBoundary = transform.width + 2 * viewportPadding;\n this.gridBottomBoundary = transform.height + 2 * viewportPadding;\n\n this.perspectiveRatioCutoff = 0.6;\n }\n\n placeCollisionBox(\n collisionBox: SingleCollisionBox,\n overlapMode: OverlapMode,\n textPixelRatio: number,\n posMatrix: mat4,\n collisionGroupPredicate?: (key: FeatureKey) => boolean,\n getElevation?: (x: number, y: number) => number\n ): {\n box: Array;\n offscreen: boolean;\n } {\n const projectedPoint = this.projectAndGetPerspectiveRatio(posMatrix, collisionBox.anchorPointX, collisionBox.anchorPointY, getElevation);\n const tileToViewport = textPixelRatio * projectedPoint.perspectiveRatio;\n const tlX = collisionBox.x1 * tileToViewport + projectedPoint.point.x;\n const tlY = collisionBox.y1 * tileToViewport + projectedPoint.point.y;\n const brX = collisionBox.x2 * tileToViewport + projectedPoint.point.x;\n const brY = collisionBox.y2 * tileToViewport + projectedPoint.point.y;\n\n if (!this.isInsideGrid(tlX, tlY, brX, brY) ||\n (overlapMode !== 'always' && this.grid.hitTest(tlX, tlY, brX, brY, overlapMode, collisionGroupPredicate)) ||\n projectedPoint.perspectiveRatio < this.perspectiveRatioCutoff) {\n return {\n box: [],\n offscreen: false\n };\n }\n\n return {\n box: [tlX, tlY, brX, brY],\n offscreen: this.isOffscreen(tlX, tlY, brX, brY)\n };\n }\n\n placeCollisionCircles(\n overlapMode: OverlapMode,\n symbol: any,\n lineVertexArray: SymbolLineVertexArray,\n glyphOffsetArray: GlyphOffsetArray,\n fontSize: number,\n posMatrix: mat4,\n labelPlaneMatrix: mat4,\n labelToScreenMatrix: mat4,\n showCollisionCircles: boolean,\n pitchWithMap: boolean,\n collisionGroupPredicate: (key: FeatureKey) => boolean,\n circlePixelDiameter: number,\n textPixelPadding: number,\n getElevation: (x: number, y: number) => number\n ): {\n circles: Array;\n offscreen: boolean;\n collisionDetected: boolean;\n } {\n const placedCollisionCircles = [];\n\n const tileUnitAnchorPoint = new Point(symbol.anchorX, symbol.anchorY);\n const screenAnchorPoint = projection.project(tileUnitAnchorPoint, posMatrix, getElevation);\n const perspectiveRatio = projection.getPerspectiveRatio(this.transform.cameraToCenterDistance, screenAnchorPoint.signedDistanceFromCamera);\n const labelPlaneFontSize = pitchWithMap ? fontSize / perspectiveRatio : fontSize * perspectiveRatio;\n const labelPlaneFontScale = labelPlaneFontSize / ONE_EM;\n\n const labelPlaneAnchorPoint = projection.project(tileUnitAnchorPoint, labelPlaneMatrix, getElevation).point;\n\n const projectionCache = {projections: {}, offsets: {}};\n const lineOffsetX = symbol.lineOffsetX * labelPlaneFontScale;\n const lineOffsetY = symbol.lineOffsetY * labelPlaneFontScale;\n\n const firstAndLastGlyph = projection.placeFirstAndLastGlyph(\n labelPlaneFontScale,\n glyphOffsetArray,\n lineOffsetX,\n lineOffsetY,\n /*flip*/ false,\n labelPlaneAnchorPoint,\n tileUnitAnchorPoint,\n symbol,\n lineVertexArray,\n labelPlaneMatrix,\n projectionCache,\n false,\n getElevation);\n\n let collisionDetected = false;\n let inGrid = false;\n let entirelyOffscreen = true;\n\n if (firstAndLastGlyph) {\n const radius = circlePixelDiameter * 0.5 * perspectiveRatio + textPixelPadding;\n const screenPlaneMin = new Point(-viewportPadding, -viewportPadding);\n const screenPlaneMax = new Point(this.screenRightBoundary, this.screenBottomBoundary);\n const interpolator = new PathInterpolator();\n\n // Construct a projected path from projected line vertices. Anchor points are ignored and removed\n const first = firstAndLastGlyph.first;\n const last = firstAndLastGlyph.last;\n\n let projectedPath = [];\n for (let i = first.path.length - 1; i >= 1; i--) {\n projectedPath.push(first.path[i]);\n }\n for (let i = 1; i < last.path.length; i++) {\n projectedPath.push(last.path[i]);\n }\n\n // Tolerate a slightly longer distance than one diameter between two adjacent circles\n const circleDist = radius * 2.5;\n\n // The path might need to be converted into screen space if a pitched map is used as the label space\n if (labelToScreenMatrix) {\n const screenSpacePath = projectedPath.map(p => projection.project(p, labelToScreenMatrix, getElevation));\n\n // Do not try to place collision circles if even of the points is behind the camera.\n // This is a plausible scenario with big camera pitch angles\n if (screenSpacePath.some(point => point.signedDistanceFromCamera <= 0)) {\n projectedPath = [];\n } else {\n projectedPath = screenSpacePath.map(p => p.point);\n }\n }\n\n let segments = [];\n\n if (projectedPath.length > 0) {\n // Quickly check if the path is fully inside or outside of the padded collision region.\n // For overlapping paths we'll only create collision circles for the visible segments\n const minPoint = projectedPath[0].clone();\n const maxPoint = projectedPath[0].clone();\n\n for (let i = 1; i < projectedPath.length; i++) {\n minPoint.x = Math.min(minPoint.x, projectedPath[i].x);\n minPoint.y = Math.min(minPoint.y, projectedPath[i].y);\n maxPoint.x = Math.max(maxPoint.x, projectedPath[i].x);\n maxPoint.y = Math.max(maxPoint.y, projectedPath[i].y);\n }\n\n if (minPoint.x >= screenPlaneMin.x && maxPoint.x <= screenPlaneMax.x &&\n minPoint.y >= screenPlaneMin.y && maxPoint.y <= screenPlaneMax.y) {\n // Quad fully visible\n segments = [projectedPath];\n } else if (maxPoint.x < screenPlaneMin.x || minPoint.x > screenPlaneMax.x ||\n maxPoint.y < screenPlaneMin.y || minPoint.y > screenPlaneMax.y) {\n // Not visible\n segments = [];\n } else {\n segments = clipLine([projectedPath], screenPlaneMin.x, screenPlaneMin.y, screenPlaneMax.x, screenPlaneMax.y);\n }\n }\n\n for (const seg of segments) {\n // interpolate positions for collision circles. Add a small padding to both ends of the segment\n interpolator.reset(seg, radius * 0.25);\n\n let numCircles = 0;\n\n if (interpolator.length <= 0.5 * radius) {\n numCircles = 1;\n } else {\n numCircles = Math.ceil(interpolator.paddedLength / circleDist) + 1;\n }\n\n for (let i = 0; i < numCircles; i++) {\n const t = i / Math.max(numCircles - 1, 1);\n const circlePosition = interpolator.lerp(t);\n\n // add viewport padding to the position and perform initial collision check\n const centerX = circlePosition.x + viewportPadding;\n const centerY = circlePosition.y + viewportPadding;\n\n placedCollisionCircles.push(centerX, centerY, radius, 0);\n\n const x1 = centerX - radius;\n const y1 = centerY - radius;\n const x2 = centerX + radius;\n const y2 = centerY + radius;\n\n entirelyOffscreen = entirelyOffscreen && this.isOffscreen(x1, y1, x2, y2);\n inGrid = inGrid || this.isInsideGrid(x1, y1, x2, y2);\n\n if (overlapMode !== 'always' && this.grid.hitTestCircle(centerX, centerY, radius, overlapMode, collisionGroupPredicate)) {\n // Don't early exit if we're showing the debug circles because we still want to calculate\n // which circles are in use\n collisionDetected = true;\n if (!showCollisionCircles) {\n return {\n circles: [],\n offscreen: false,\n collisionDetected\n };\n }\n }\n }\n }\n }\n\n return {\n circles: ((!showCollisionCircles && collisionDetected) || !inGrid || perspectiveRatio < this.perspectiveRatioCutoff) ? [] : placedCollisionCircles,\n offscreen: entirelyOffscreen,\n collisionDetected\n };\n }\n\n /**\n * Because the geometries in the CollisionIndex are an approximation of the shape of\n * symbols on the map, we use the CollisionIndex to look up the symbol part of\n * `queryRenderedFeatures`.\n */\n queryRenderedSymbols(viewportQueryGeometry: Array) {\n if (viewportQueryGeometry.length === 0 || (this.grid.keysLength() === 0 && this.ignoredGrid.keysLength() === 0)) {\n return {};\n }\n\n const query = [];\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n for (const point of viewportQueryGeometry) {\n const gridPoint = new Point(point.x + viewportPadding, point.y + viewportPadding);\n minX = Math.min(minX, gridPoint.x);\n minY = Math.min(minY, gridPoint.y);\n maxX = Math.max(maxX, gridPoint.x);\n maxY = Math.max(maxY, gridPoint.y);\n query.push(gridPoint);\n }\n\n const features = this.grid.query(minX, minY, maxX, maxY)\n .concat(this.ignoredGrid.query(minX, minY, maxX, maxY));\n\n const seenFeatures = {};\n const result = {};\n\n for (const feature of features) {\n const featureKey = feature.key;\n // Skip already seen features.\n if (seenFeatures[featureKey.bucketInstanceId] === undefined) {\n seenFeatures[featureKey.bucketInstanceId] = {};\n }\n if (seenFeatures[featureKey.bucketInstanceId][featureKey.featureIndex]) {\n continue;\n }\n\n // Check if query intersects with the feature box\n // \"Collision Circles\" for line labels are treated as boxes here\n // Since there's no actual collision taking place, the circle vs. square\n // distinction doesn't matter as much, and box geometry is easier\n // to work with.\n const bbox = [\n new Point(feature.x1, feature.y1),\n new Point(feature.x2, feature.y1),\n new Point(feature.x2, feature.y2),\n new Point(feature.x1, feature.y2)\n ];\n if (!intersectionTests.polygonIntersectsPolygon(query, bbox)) {\n continue;\n }\n\n seenFeatures[featureKey.bucketInstanceId][featureKey.featureIndex] = true;\n if (result[featureKey.bucketInstanceId] === undefined) {\n result[featureKey.bucketInstanceId] = [];\n }\n result[featureKey.bucketInstanceId].push(featureKey.featureIndex);\n }\n\n return result;\n }\n\n insertCollisionBox(collisionBox: Array, overlapMode: OverlapMode, ignorePlacement: boolean, bucketInstanceId: number, featureIndex: number, collisionGroupID: number) {\n const grid = ignorePlacement ? this.ignoredGrid : this.grid;\n\n const key = {bucketInstanceId, featureIndex, collisionGroupID, overlapMode};\n grid.insert(key, collisionBox[0], collisionBox[1], collisionBox[2], collisionBox[3]);\n }\n\n insertCollisionCircles(collisionCircles: Array, overlapMode: OverlapMode, ignorePlacement: boolean, bucketInstanceId: number, featureIndex: number, collisionGroupID: number) {\n const grid = ignorePlacement ? this.ignoredGrid : this.grid;\n\n const key = {bucketInstanceId, featureIndex, collisionGroupID, overlapMode};\n for (let k = 0; k < collisionCircles.length; k += 4) {\n grid.insertCircle(key, collisionCircles[k], collisionCircles[k + 1], collisionCircles[k + 2]);\n }\n }\n\n projectAndGetPerspectiveRatio(posMatrix: mat4, x: number, y: number, getElevation?: (x: number, y: number) => number) {\n let p;\n if (getElevation) { // slow because of handle z-index\n p = [x, y, getElevation(x, y), 1] as vec4;\n vec4.transformMat4(p, p, posMatrix);\n } else { // fast because of ignore z-index\n p = [x, y, 0, 1] as vec4;\n projection.xyTransformMat4(p, p, posMatrix);\n }\n const a = new Point(\n (((p[0] / p[3] + 1) / 2) * this.transform.width) + viewportPadding,\n (((-p[1] / p[3] + 1) / 2) * this.transform.height) + viewportPadding\n );\n return {\n point: a,\n // See perspective ratio comment in symbol_sdf.vertex\n // We're doing collision detection in viewport space so we need\n // to scale down boxes in the distance\n perspectiveRatio: 0.5 + 0.5 * (this.transform.cameraToCenterDistance / p[3])\n };\n }\n\n isOffscreen(x1: number, y1: number, x2: number, y2: number) {\n return x2 < viewportPadding || x1 >= this.screenRightBoundary || y2 < viewportPadding || y1 > this.screenBottomBoundary;\n }\n\n isInsideGrid(x1: number, y1: number, x2: number, y2: number) {\n return x2 >= 0 && x1 < this.gridRightBoundary && y2 >= 0 && y1 < this.gridBottomBoundary;\n }\n\n /*\n * Returns a matrix for transforming collision shapes to viewport coordinate space.\n * Use this function to render e.g. collision circles on the screen.\n * example transformation: clipPos = glCoordMatrix * viewportMatrix * circle_pos\n */\n getViewportMatrix() {\n const m = mat4.identity([] as any);\n mat4.translate(m, m, [-viewportPadding, -viewportPadding, 0.0]);\n return m;\n }\n}\n","import {EXTENT} from '../data/extent';\n\nimport type {OverscaledTileID} from './tile_id';\n\n/**\n * Converts a pixel value at a the given zoom level to tile units.\n *\n * The shaders mostly calculate everything in tile units so style\n * properties need to be converted from pixels to tile units using this.\n *\n * For example, a translation by 30 pixels at zoom 6.5 will be a\n * translation by pixelsToTileUnits(30, 6.5) tile units.\n *\n * @returns value in tile units\n */\nexport function pixelsToTileUnits(\n tile: {\n tileID: OverscaledTileID;\n tileSize: number;\n },\n pixelValue: number,\n z: number\n): number {\n return pixelValue * (EXTENT / (tile.tileSize * Math.pow(2, z - tile.tileID.overscaledZ)));\n}\n","import {CollisionIndex} from './collision_index';\nimport type {FeatureKey} from './collision_index';\nimport {EXTENT} from '../data/extent';\nimport * as symbolSize from './symbol_size';\nimport * as projection from './projection';\nimport {getAnchorJustification} from './symbol_layout';\nimport {getAnchorAlignment, WritingMode} from './shaping';\nimport {mat4} from 'gl-matrix';\nimport {pixelsToTileUnits} from '../source/pixels_to_tile_units';\nimport Point from '@mapbox/point-geometry';\nimport type {Transform} from '../geo/transform';\nimport type {StyleLayer} from '../style/style_layer';\nimport {PossiblyEvaluated} from '../style/properties';\nimport type {SymbolLayoutProps, SymbolLayoutPropsPossiblyEvaluated} from '../style/style_layer/symbol_style_layer_properties.g';\nimport {getOverlapMode, OverlapMode} from '../style/style_layer/overlap_mode';\n\nimport type {Tile} from '../source/tile';\nimport {SymbolBucket, CollisionArrays, SingleCollisionBox} from '../data/bucket/symbol_bucket';\n\nimport type {CollisionBoxArray, CollisionVertexArray, SymbolInstance, TextAnchorOffset} from '../data/array_types.g';\nimport type {FeatureIndex} from '../data/feature_index';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport {Terrain} from '../render/terrain';\nimport {warnOnce} from '../util/util';\nimport {TextAnchor, TextAnchorEnum} from '../style/style_layer/variable_text_anchor';\n\nclass OpacityState {\n opacity: number;\n placed: boolean;\n constructor(prevState: OpacityState, increment: number, placed: boolean, skipFade?: boolean | null) {\n if (prevState) {\n this.opacity = Math.max(0, Math.min(1, prevState.opacity + (prevState.placed ? increment : -increment)));\n } else {\n this.opacity = (skipFade && placed) ? 1 : 0;\n }\n this.placed = placed;\n }\n isHidden() {\n return this.opacity === 0 && !this.placed;\n }\n}\n\nclass JointOpacityState {\n text: OpacityState;\n icon: OpacityState;\n constructor(prevState: JointOpacityState, increment: number, placedText: boolean, placedIcon: boolean, skipFade?: boolean | null) {\n this.text = new OpacityState(prevState ? prevState.text : null, increment, placedText, skipFade);\n this.icon = new OpacityState(prevState ? prevState.icon : null, increment, placedIcon, skipFade);\n }\n isHidden() {\n return this.text.isHidden() && this.icon.isHidden();\n }\n}\n\nclass JointPlacement {\n text: boolean;\n icon: boolean;\n // skipFade = outside viewport, but within CollisionIndex::viewportPadding px of the edge\n // Because these symbols aren't onscreen yet, we can skip the \"fade in\" animation,\n // and if a subsequent viewport change brings them into view, they'll be fully\n // visible right away.\n skipFade: boolean;\n constructor(text: boolean, icon: boolean, skipFade: boolean) {\n this.text = text;\n this.icon = icon;\n this.skipFade = skipFade;\n }\n}\n\nclass CollisionCircleArray {\n // Stores collision circles and placement matrices of a bucket for debug rendering.\n invProjMatrix: mat4;\n viewportMatrix: mat4;\n circles: Array;\n\n constructor() {\n this.invProjMatrix = mat4.create();\n this.viewportMatrix = mat4.create();\n this.circles = [];\n }\n}\n\nexport class RetainedQueryData {\n bucketInstanceId: number;\n featureIndex: FeatureIndex;\n sourceLayerIndex: number;\n bucketIndex: number;\n tileID: OverscaledTileID;\n featureSortOrder: Array;\n constructor(bucketInstanceId: number,\n featureIndex: FeatureIndex,\n sourceLayerIndex: number,\n bucketIndex: number,\n tileID: OverscaledTileID) {\n this.bucketInstanceId = bucketInstanceId;\n this.featureIndex = featureIndex;\n this.sourceLayerIndex = sourceLayerIndex;\n this.bucketIndex = bucketIndex;\n this.tileID = tileID;\n }\n}\n\ntype CollisionGroup = {\n ID: number;\n predicate?: (key: FeatureKey) => boolean;\n};\n\nclass CollisionGroups {\n collisionGroups: {[groupName: string]: CollisionGroup};\n maxGroupID: number;\n crossSourceCollisions: boolean;\n\n constructor(crossSourceCollisions: boolean) {\n this.crossSourceCollisions = crossSourceCollisions;\n this.maxGroupID = 0;\n this.collisionGroups = {};\n }\n\n get(sourceID: string) {\n // The predicate/groupID mechanism allows for arbitrary grouping,\n // but the current interface defines one source == one group when\n // crossSourceCollisions == true.\n if (!this.crossSourceCollisions) {\n if (!this.collisionGroups[sourceID]) {\n const nextGroupID = ++this.maxGroupID;\n this.collisionGroups[sourceID] = {\n ID: nextGroupID,\n predicate: (key) => {\n return key.collisionGroupID === nextGroupID;\n }\n };\n }\n return this.collisionGroups[sourceID];\n } else {\n return {ID: 0, predicate: null};\n }\n }\n}\n\nfunction calculateVariableLayoutShift(\n anchor: TextAnchor,\n width: number,\n height: number,\n textOffset: [number, number],\n textBoxScale: number\n): Point {\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(anchor);\n const shiftX = -(horizontalAlign - 0.5) * width;\n const shiftY = -(verticalAlign - 0.5) * height;\n return new Point(\n shiftX + textOffset[0] * textBoxScale,\n shiftY + textOffset[1] * textBoxScale\n );\n}\n\nfunction shiftVariableCollisionBox(collisionBox: SingleCollisionBox,\n shiftX: number, shiftY: number,\n rotateWithMap: boolean, pitchWithMap: boolean,\n angle: number) {\n const {x1, x2, y1, y2, anchorPointX, anchorPointY} = collisionBox;\n const rotatedOffset = new Point(shiftX, shiftY);\n if (rotateWithMap) {\n rotatedOffset._rotate(pitchWithMap ? angle : -angle);\n }\n return {\n x1: x1 + rotatedOffset.x,\n y1: y1 + rotatedOffset.y,\n x2: x2 + rotatedOffset.x,\n y2: y2 + rotatedOffset.y,\n // symbol anchor point stays the same regardless of text-anchor\n anchorPointX,\n anchorPointY\n };\n}\n\nexport type VariableOffset = {\n textOffset: [number, number];\n width: number;\n height: number;\n anchor: TextAnchor;\n textBoxScale: number;\n prevAnchor?: TextAnchor;\n};\n\ntype TileLayerParameters = {\n bucket: SymbolBucket;\n layout: PossiblyEvaluated;\n posMatrix: mat4;\n textLabelPlaneMatrix: mat4;\n labelToScreenMatrix: mat4;\n scale: number;\n textPixelRatio: number;\n holdingForFade: boolean;\n collisionBoxArray: CollisionBoxArray;\n partiallyEvaluatedTextSize: {\n uSize: number;\n uSizeT: number;\n };\n collisionGroup: CollisionGroup;\n};\n\nexport type BucketPart = {\n sortKey?: number | void;\n symbolInstanceStart: number;\n symbolInstanceEnd: number;\n parameters: TileLayerParameters;\n};\n\nexport type CrossTileID = string | number;\n\nexport class Placement {\n transform: Transform;\n terrain: Terrain;\n collisionIndex: CollisionIndex;\n placements: {\n [_ in CrossTileID]: JointPlacement;\n };\n opacities: {\n [_ in CrossTileID]: JointOpacityState;\n };\n variableOffsets: {\n [_ in CrossTileID]: VariableOffset;\n };\n placedOrientations: {\n [_ in CrossTileID]: number;\n };\n commitTime: number;\n prevZoomAdjustment: number;\n lastPlacementChangeTime: number;\n stale: boolean;\n fadeDuration: number;\n retainedQueryData: {\n [_: number]: RetainedQueryData;\n };\n collisionGroups: CollisionGroups;\n prevPlacement: Placement;\n zoomAtLastRecencyCheck: number;\n collisionCircleArrays: {\n [k in any]: CollisionCircleArray;\n };\n\n constructor(transform: Transform, terrain: Terrain, fadeDuration: number, crossSourceCollisions: boolean, prevPlacement?: Placement) {\n this.transform = transform.clone();\n this.terrain = terrain;\n this.collisionIndex = new CollisionIndex(this.transform);\n this.placements = {};\n this.opacities = {};\n this.variableOffsets = {};\n this.stale = false;\n this.commitTime = 0;\n this.fadeDuration = fadeDuration;\n this.retainedQueryData = {};\n this.collisionGroups = new CollisionGroups(crossSourceCollisions);\n this.collisionCircleArrays = {};\n\n this.prevPlacement = prevPlacement;\n if (prevPlacement) {\n prevPlacement.prevPlacement = undefined; // Only hold on to one placement back\n }\n\n this.placedOrientations = {};\n }\n\n getBucketParts(results: Array, styleLayer: StyleLayer, tile: Tile, sortAcrossTiles: boolean) {\n const symbolBucket = (tile.getBucket(styleLayer) as SymbolBucket);\n const bucketFeatureIndex = tile.latestFeatureIndex;\n if (!symbolBucket || !bucketFeatureIndex || styleLayer.id !== symbolBucket.layerIds[0])\n return;\n\n const collisionBoxArray = tile.collisionBoxArray;\n\n const layout = symbolBucket.layers[0].layout;\n\n const scale = Math.pow(2, this.transform.zoom - tile.tileID.overscaledZ);\n const textPixelRatio = tile.tileSize / EXTENT;\n\n const posMatrix = this.transform.calculatePosMatrix(tile.tileID.toUnwrapped());\n\n const pitchWithMap = layout.get('text-pitch-alignment') === 'map';\n const rotateWithMap = layout.get('text-rotation-alignment') === 'map';\n const pixelsToTiles = pixelsToTileUnits(tile, 1, this.transform.zoom);\n\n const textLabelPlaneMatrix = projection.getLabelPlaneMatrix(posMatrix,\n pitchWithMap,\n rotateWithMap,\n this.transform,\n pixelsToTiles);\n\n let labelToScreenMatrix = null;\n\n if (pitchWithMap) {\n const glMatrix = projection.getGlCoordMatrix(\n posMatrix,\n pitchWithMap,\n rotateWithMap,\n this.transform,\n pixelsToTiles);\n\n labelToScreenMatrix = mat4.multiply([] as any, this.transform.labelPlaneMatrix, glMatrix);\n }\n\n // As long as this placement lives, we have to hold onto this bucket's\n // matching FeatureIndex/data for querying purposes\n this.retainedQueryData[symbolBucket.bucketInstanceId] = new RetainedQueryData(\n symbolBucket.bucketInstanceId,\n bucketFeatureIndex,\n symbolBucket.sourceLayerIndex,\n symbolBucket.index,\n tile.tileID\n );\n\n const parameters = {\n bucket: symbolBucket,\n layout,\n posMatrix,\n textLabelPlaneMatrix,\n labelToScreenMatrix,\n scale,\n textPixelRatio,\n holdingForFade: tile.holdingForFade(),\n collisionBoxArray,\n partiallyEvaluatedTextSize: symbolSize.evaluateSizeForZoom(symbolBucket.textSizeData, this.transform.zoom),\n collisionGroup: this.collisionGroups.get(symbolBucket.sourceID)\n };\n\n if (sortAcrossTiles) {\n for (const range of symbolBucket.sortKeyRanges) {\n const {sortKey, symbolInstanceStart, symbolInstanceEnd} = range;\n results.push({sortKey, symbolInstanceStart, symbolInstanceEnd, parameters});\n }\n } else {\n results.push({\n symbolInstanceStart: 0,\n symbolInstanceEnd: symbolBucket.symbolInstances.length,\n parameters\n });\n }\n }\n\n attemptAnchorPlacement(\n textAnchorOffset: TextAnchorOffset,\n textBox: SingleCollisionBox,\n width: number,\n height: number,\n textBoxScale: number,\n rotateWithMap: boolean,\n pitchWithMap: boolean,\n textPixelRatio: number,\n posMatrix: mat4,\n collisionGroup: CollisionGroup,\n textOverlapMode: OverlapMode,\n symbolInstance: SymbolInstance,\n bucket: SymbolBucket,\n orientation: number,\n iconBox?: SingleCollisionBox | null,\n getElevation?: (x: number, y: number) => number\n ): {\n shift: Point;\n placedGlyphBoxes: {\n box: Array;\n offscreen: boolean;\n };\n } {\n\n const anchor = TextAnchorEnum[textAnchorOffset.textAnchor] as TextAnchor;\n const textOffset = [textAnchorOffset.textOffset0, textAnchorOffset.textOffset1] as [number, number];\n const shift = calculateVariableLayoutShift(anchor, width, height, textOffset, textBoxScale);\n\n const placedGlyphBoxes = this.collisionIndex.placeCollisionBox(\n shiftVariableCollisionBox(\n textBox, shift.x, shift.y,\n rotateWithMap, pitchWithMap, this.transform.angle),\n textOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation);\n\n if (iconBox) {\n const placedIconBoxes = this.collisionIndex.placeCollisionBox(\n shiftVariableCollisionBox(\n iconBox, shift.x, shift.y,\n rotateWithMap, pitchWithMap, this.transform.angle),\n textOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation);\n if (placedIconBoxes.box.length === 0) return;\n }\n\n if (placedGlyphBoxes.box.length > 0) {\n let prevAnchor;\n // If this label was placed in the previous placement, record the anchor position\n // to allow us to animate the transition\n if (this.prevPlacement &&\n this.prevPlacement.variableOffsets[symbolInstance.crossTileID] &&\n this.prevPlacement.placements[symbolInstance.crossTileID] &&\n this.prevPlacement.placements[symbolInstance.crossTileID].text) {\n prevAnchor = this.prevPlacement.variableOffsets[symbolInstance.crossTileID].anchor;\n }\n if (symbolInstance.crossTileID === 0) throw new Error('symbolInstance.crossTileID can\\'t be 0');\n this.variableOffsets[symbolInstance.crossTileID] = {\n textOffset,\n width,\n height,\n anchor,\n textBoxScale,\n prevAnchor\n };\n this.markUsedJustification(bucket, anchor, symbolInstance, orientation);\n\n if (bucket.allowVerticalPlacement) {\n this.markUsedOrientation(bucket, orientation, symbolInstance);\n this.placedOrientations[symbolInstance.crossTileID] = orientation;\n }\n\n return {shift, placedGlyphBoxes};\n }\n }\n\n placeLayerBucketPart(bucketPart: BucketPart, seenCrossTileIDs: {\n [k in string | number]: boolean;\n }, showCollisionBoxes: boolean) {\n\n const {\n bucket,\n layout,\n posMatrix,\n textLabelPlaneMatrix,\n labelToScreenMatrix,\n textPixelRatio,\n holdingForFade,\n collisionBoxArray,\n partiallyEvaluatedTextSize,\n collisionGroup\n } = bucketPart.parameters;\n\n const textOptional = layout.get('text-optional');\n const iconOptional = layout.get('icon-optional');\n const textOverlapMode = getOverlapMode(layout, 'text-overlap', 'text-allow-overlap');\n const textAlwaysOverlap = textOverlapMode === 'always';\n const iconOverlapMode = getOverlapMode(layout, 'icon-overlap', 'icon-allow-overlap');\n const iconAlwaysOverlap = iconOverlapMode === 'always';\n const rotateWithMap = layout.get('text-rotation-alignment') === 'map';\n const pitchWithMap = layout.get('text-pitch-alignment') === 'map';\n const hasIconTextFit = layout.get('icon-text-fit') !== 'none';\n const zOrderByViewportY = layout.get('symbol-z-order') === 'viewport-y';\n\n // This logic is similar to the \"defaultOpacityState\" logic below in updateBucketOpacities\n // If we know a symbol is always supposed to show, force it to be marked visible even if\n // it wasn't placed into the collision index (because some or all of it was outside the range\n // of the collision grid).\n // There is a subtle edge case here we're accepting:\n // Symbol A has text-allow-overlap: true, icon-allow-overlap: true, icon-optional: false\n // A's icon is outside the grid, so doesn't get placed\n // A's text would be inside grid, but doesn't get placed because of icon-optional: false\n // We still show A because of the allow-overlap settings.\n // Symbol B has allow-overlap: false, and gets placed where A's text would be\n // On panning in, there is a short period when Symbol B and Symbol A will overlap\n // This is the reverse of our normal policy of \"fade in on pan\", but should look like any other\n // collision and hopefully not be too noticeable.\n // See https://github.com/mapbox/mapbox-gl-js/issues/7172\n const alwaysShowText = textAlwaysOverlap && (iconAlwaysOverlap || !bucket.hasIconData() || iconOptional);\n const alwaysShowIcon = iconAlwaysOverlap && (textAlwaysOverlap || !bucket.hasTextData() || textOptional);\n\n if (!bucket.collisionArrays && collisionBoxArray) {\n bucket.deserializeCollisionBoxes(collisionBoxArray);\n }\n\n const tileID = this.retainedQueryData[bucket.bucketInstanceId].tileID;\n const getElevation = this.terrain ? (x: number, y: number) => this.terrain.getElevation(tileID, x, y) : null;\n\n const placeSymbol = (symbolInstance: SymbolInstance, collisionArrays: CollisionArrays) => {\n if (seenCrossTileIDs[symbolInstance.crossTileID]) return;\n if (holdingForFade) {\n // Mark all symbols from this tile as \"not placed\", but don't add to seenCrossTileIDs, because we don't\n // know yet if we have a duplicate in a parent tile that _should_ be placed.\n this.placements[symbolInstance.crossTileID] = new JointPlacement(false, false, false);\n return;\n }\n\n let placeText = false;\n let placeIcon = false;\n let offscreen = true;\n let shift = null;\n\n let placed = {box: null, offscreen: null};\n let placedVerticalText = {box: null, offscreen: null};\n\n let placedGlyphBoxes = null;\n let placedGlyphCircles = null;\n let placedIconBoxes = null;\n let textFeatureIndex = 0;\n let verticalTextFeatureIndex = 0;\n let iconFeatureIndex = 0;\n\n if (collisionArrays.textFeatureIndex) {\n textFeatureIndex = collisionArrays.textFeatureIndex;\n } else if (symbolInstance.useRuntimeCollisionCircles) {\n textFeatureIndex = symbolInstance.featureIndex;\n }\n if (collisionArrays.verticalTextFeatureIndex) {\n verticalTextFeatureIndex = collisionArrays.verticalTextFeatureIndex;\n }\n\n const textBox = collisionArrays.textBox;\n if (textBox) {\n\n const updatePreviousOrientationIfNotPlaced = (isPlaced) => {\n let previousOrientation = WritingMode.horizontal;\n if (bucket.allowVerticalPlacement && !isPlaced && this.prevPlacement) {\n const prevPlacedOrientation = this.prevPlacement.placedOrientations[symbolInstance.crossTileID];\n if (prevPlacedOrientation) {\n this.placedOrientations[symbolInstance.crossTileID] = prevPlacedOrientation;\n previousOrientation = prevPlacedOrientation;\n this.markUsedOrientation(bucket, previousOrientation, symbolInstance);\n }\n }\n return previousOrientation;\n };\n\n const placeTextForPlacementModes = (placeHorizontalFn, placeVerticalFn) => {\n if (bucket.allowVerticalPlacement && symbolInstance.numVerticalGlyphVertices > 0 && collisionArrays.verticalTextBox) {\n for (const placementMode of bucket.writingModes) {\n if (placementMode === WritingMode.vertical) {\n placed = placeVerticalFn();\n placedVerticalText = placed;\n } else {\n placed = placeHorizontalFn();\n }\n if (placed && placed.box && placed.box.length) break;\n }\n } else {\n placed = placeHorizontalFn();\n }\n };\n\n const textAnchorOffsetStart = symbolInstance.textAnchorOffsetStartIndex;\n const textAnchorOffsetEnd = symbolInstance.textAnchorOffsetEndIndex;\n\n // If start+end indices match, text-variable-anchor is not in play.\n if (textAnchorOffsetEnd === textAnchorOffsetStart) {\n const placeBox = (collisionTextBox, orientation) => {\n const placedFeature = this.collisionIndex.placeCollisionBox(\n collisionTextBox,\n textOverlapMode,\n textPixelRatio,\n posMatrix,\n collisionGroup.predicate,\n getElevation\n );\n if (placedFeature && placedFeature.box && placedFeature.box.length) {\n this.markUsedOrientation(bucket, orientation, symbolInstance);\n this.placedOrientations[symbolInstance.crossTileID] = orientation;\n }\n return placedFeature;\n };\n\n const placeHorizontal = () => {\n return placeBox(textBox, WritingMode.horizontal);\n };\n\n const placeVertical = () => {\n const verticalTextBox = collisionArrays.verticalTextBox;\n if (bucket.allowVerticalPlacement && symbolInstance.numVerticalGlyphVertices > 0 && verticalTextBox) {\n return placeBox(verticalTextBox, WritingMode.vertical);\n }\n return {box: null, offscreen: null};\n };\n\n placeTextForPlacementModes(placeHorizontal, placeVertical);\n updatePreviousOrientationIfNotPlaced(placed && placed.box && placed.box.length);\n\n } else {\n // If this symbol was in the last placement, prefer placement using same anchor, if it's still available\n let prevAnchor = TextAnchorEnum[this.prevPlacement?.variableOffsets[symbolInstance.crossTileID]?.anchor];\n\n const placeBoxForVariableAnchors = (collisionTextBox, collisionIconBox, orientation) => {\n const width = collisionTextBox.x2 - collisionTextBox.x1;\n const height = collisionTextBox.y2 - collisionTextBox.y1;\n const textBoxScale = symbolInstance.textBoxScale;\n const variableIconBox = hasIconTextFit && (iconOverlapMode === 'never') ? collisionIconBox : null;\n\n let placedBox: {\n box: Array;\n offscreen: boolean;\n } = {box: [], offscreen: false};\n let placementPasses = (textOverlapMode === 'never') ? 1 : 2;\n let overlapMode: OverlapMode = 'never';\n\n if (prevAnchor) {\n placementPasses++;\n }\n\n for (let pass = 0; pass < placementPasses; pass++) {\n for (let i = textAnchorOffsetStart; i < textAnchorOffsetEnd; i++) {\n const textAnchorOffset = bucket.textAnchorOffsets.get(i);\n\n if (prevAnchor && textAnchorOffset.textAnchor !== prevAnchor) {\n continue;\n }\n\n const result = this.attemptAnchorPlacement(\n textAnchorOffset, collisionTextBox, width, height,\n textBoxScale, rotateWithMap, pitchWithMap, textPixelRatio, posMatrix,\n collisionGroup, overlapMode, symbolInstance, bucket, orientation, variableIconBox, getElevation);\n\n if (result) {\n placedBox = result.placedGlyphBoxes;\n if (placedBox && placedBox.box && placedBox.box.length) {\n placeText = true;\n shift = result.shift;\n return placedBox;\n }\n }\n }\n\n if (prevAnchor) {\n prevAnchor = null;\n } else {\n overlapMode = textOverlapMode;\n }\n }\n\n return placedBox;\n };\n\n const placeHorizontal = () => {\n return placeBoxForVariableAnchors(textBox, collisionArrays.iconBox, WritingMode.horizontal);\n };\n\n const placeVertical = () => {\n const verticalTextBox = collisionArrays.verticalTextBox;\n const wasPlaced = placed && placed.box && placed.box.length;\n if (bucket.allowVerticalPlacement && !wasPlaced && symbolInstance.numVerticalGlyphVertices > 0 && verticalTextBox) {\n return placeBoxForVariableAnchors(verticalTextBox, collisionArrays.verticalIconBox, WritingMode.vertical);\n }\n return {box: null, offscreen: null};\n };\n\n placeTextForPlacementModes(placeHorizontal, placeVertical);\n\n if (placed) {\n placeText = placed.box;\n offscreen = placed.offscreen;\n }\n\n const prevOrientation = updatePreviousOrientationIfNotPlaced(placed && placed.box);\n\n // If we didn't get placed, we still need to copy our position from the last placement for\n // fade animations\n if (!placeText && this.prevPlacement) {\n const prevOffset = this.prevPlacement.variableOffsets[symbolInstance.crossTileID];\n if (prevOffset) {\n this.variableOffsets[symbolInstance.crossTileID] = prevOffset;\n this.markUsedJustification(bucket, prevOffset.anchor, symbolInstance, prevOrientation);\n }\n }\n\n }\n }\n\n placedGlyphBoxes = placed;\n placeText = placedGlyphBoxes && placedGlyphBoxes.box && placedGlyphBoxes.box.length > 0;\n\n offscreen = placedGlyphBoxes && placedGlyphBoxes.offscreen;\n\n if (symbolInstance.useRuntimeCollisionCircles) {\n const placedSymbol = bucket.text.placedSymbolArray.get(symbolInstance.centerJustifiedTextSymbolIndex);\n const fontSize = symbolSize.evaluateSizeForFeature(bucket.textSizeData, partiallyEvaluatedTextSize, placedSymbol);\n\n const textPixelPadding = layout.get('text-padding');\n const circlePixelDiameter = symbolInstance.collisionCircleDiameter;\n\n placedGlyphCircles = this.collisionIndex.placeCollisionCircles(\n textOverlapMode,\n placedSymbol,\n bucket.lineVertexArray,\n bucket.glyphOffsetArray,\n fontSize,\n posMatrix,\n textLabelPlaneMatrix,\n labelToScreenMatrix,\n showCollisionBoxes,\n pitchWithMap,\n collisionGroup.predicate,\n circlePixelDiameter,\n textPixelPadding,\n getElevation\n );\n\n if (placedGlyphCircles.circles.length && placedGlyphCircles.collisionDetected && !showCollisionBoxes) {\n warnOnce('Collisions detected, but collision boxes are not shown');\n }\n\n // If text-overlap is set to 'always', force \"placedCircles\" to true\n // In theory there should always be at least one circle placed\n // in this case, but for now quirks in text-anchor\n // and text-offset may prevent that from being true.\n placeText = textAlwaysOverlap || (placedGlyphCircles.circles.length > 0 && !placedGlyphCircles.collisionDetected);\n offscreen = offscreen && placedGlyphCircles.offscreen;\n }\n\n if (collisionArrays.iconFeatureIndex) {\n iconFeatureIndex = collisionArrays.iconFeatureIndex;\n }\n\n if (collisionArrays.iconBox) {\n const placeIconFeature = iconBox => {\n const shiftedIconBox = hasIconTextFit && shift ?\n shiftVariableCollisionBox(\n iconBox, shift.x, shift.y,\n rotateWithMap, pitchWithMap, this.transform.angle) :\n iconBox;\n return this.collisionIndex.placeCollisionBox(shiftedIconBox,\n iconOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation);\n };\n\n if (placedVerticalText && placedVerticalText.box && placedVerticalText.box.length && collisionArrays.verticalIconBox) {\n placedIconBoxes = placeIconFeature(collisionArrays.verticalIconBox);\n placeIcon = placedIconBoxes.box.length > 0;\n } else {\n placedIconBoxes = placeIconFeature(collisionArrays.iconBox);\n placeIcon = placedIconBoxes.box.length > 0;\n }\n offscreen = offscreen && placedIconBoxes.offscreen;\n }\n\n const iconWithoutText = textOptional ||\n (symbolInstance.numHorizontalGlyphVertices === 0 && symbolInstance.numVerticalGlyphVertices === 0);\n const textWithoutIcon = iconOptional || symbolInstance.numIconVertices === 0;\n\n // Combine the scales for icons and text.\n if (!iconWithoutText && !textWithoutIcon) {\n placeIcon = placeText = placeIcon && placeText;\n } else if (!textWithoutIcon) {\n placeText = placeIcon && placeText;\n } else if (!iconWithoutText) {\n placeIcon = placeIcon && placeText;\n }\n\n if (placeText && placedGlyphBoxes && placedGlyphBoxes.box) {\n if (placedVerticalText && placedVerticalText.box && verticalTextFeatureIndex) {\n this.collisionIndex.insertCollisionBox(\n placedGlyphBoxes.box,\n textOverlapMode,\n layout.get('text-ignore-placement'),\n bucket.bucketInstanceId,\n verticalTextFeatureIndex,\n collisionGroup.ID);\n } else {\n this.collisionIndex.insertCollisionBox(\n placedGlyphBoxes.box,\n textOverlapMode,\n layout.get('text-ignore-placement'),\n bucket.bucketInstanceId,\n textFeatureIndex,\n collisionGroup.ID);\n }\n\n }\n if (placeIcon && placedIconBoxes) {\n this.collisionIndex.insertCollisionBox(\n placedIconBoxes.box,\n iconOverlapMode,\n layout.get('icon-ignore-placement'),\n bucket.bucketInstanceId,\n iconFeatureIndex,\n collisionGroup.ID);\n }\n if (placedGlyphCircles) {\n if (placeText) {\n this.collisionIndex.insertCollisionCircles(\n placedGlyphCircles.circles,\n textOverlapMode,\n layout.get('text-ignore-placement'),\n bucket.bucketInstanceId,\n textFeatureIndex,\n collisionGroup.ID);\n }\n\n if (showCollisionBoxes) {\n const id = bucket.bucketInstanceId;\n let circleArray = this.collisionCircleArrays[id];\n\n // Group collision circles together by bucket. Circles can't be pushed forward for rendering yet as the symbol placement\n // for a bucket is not guaranteed to be complete before the commit-function has been called\n if (circleArray === undefined)\n circleArray = this.collisionCircleArrays[id] = new CollisionCircleArray();\n\n for (let i = 0; i < placedGlyphCircles.circles.length; i += 4) {\n circleArray.circles.push(placedGlyphCircles.circles[i + 0]); // x\n circleArray.circles.push(placedGlyphCircles.circles[i + 1]); // y\n circleArray.circles.push(placedGlyphCircles.circles[i + 2]); // radius\n circleArray.circles.push(placedGlyphCircles.collisionDetected ? 1 : 0); // collisionDetected-flag\n }\n }\n }\n\n if (symbolInstance.crossTileID === 0) throw new Error('symbolInstance.crossTileID can\\'t be 0');\n if (bucket.bucketInstanceId === 0) throw new Error('bucket.bucketInstanceId can\\'t be 0');\n\n this.placements[symbolInstance.crossTileID] = new JointPlacement(placeText || alwaysShowText, placeIcon || alwaysShowIcon, offscreen || bucket.justReloaded);\n seenCrossTileIDs[symbolInstance.crossTileID] = true;\n };\n\n if (zOrderByViewportY) {\n if (bucketPart.symbolInstanceStart !== 0) throw new Error('bucket.bucketInstanceId should be 0');\n const symbolIndexes = bucket.getSortedSymbolIndexes(this.transform.angle);\n for (let i = symbolIndexes.length - 1; i >= 0; --i) {\n const symbolIndex = symbolIndexes[i];\n placeSymbol(bucket.symbolInstances.get(symbolIndex), bucket.collisionArrays[symbolIndex]);\n }\n } else {\n for (let i = bucketPart.symbolInstanceStart; i < bucketPart.symbolInstanceEnd; i++) {\n placeSymbol(bucket.symbolInstances.get(i), bucket.collisionArrays[i]);\n }\n }\n\n if (showCollisionBoxes && bucket.bucketInstanceId in this.collisionCircleArrays) {\n const circleArray = this.collisionCircleArrays[bucket.bucketInstanceId];\n\n // Store viewport and inverse projection matrices per bucket\n mat4.invert(circleArray.invProjMatrix, posMatrix);\n circleArray.viewportMatrix = this.collisionIndex.getViewportMatrix();\n }\n\n bucket.justReloaded = false;\n }\n\n markUsedJustification(bucket: SymbolBucket, placedAnchor: TextAnchor, symbolInstance: SymbolInstance, orientation: number) {\n const justifications = {\n 'left': symbolInstance.leftJustifiedTextSymbolIndex,\n 'center': symbolInstance.centerJustifiedTextSymbolIndex,\n 'right': symbolInstance.rightJustifiedTextSymbolIndex\n };\n\n let autoIndex;\n if (orientation === WritingMode.vertical) {\n autoIndex = symbolInstance.verticalPlacedTextSymbolIndex;\n } else {\n autoIndex = justifications[getAnchorJustification(placedAnchor)];\n }\n\n const indexes = [\n symbolInstance.leftJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.rightJustifiedTextSymbolIndex,\n symbolInstance.verticalPlacedTextSymbolIndex\n ];\n\n for (const index of indexes) {\n if (index >= 0) {\n if (autoIndex >= 0 && index !== autoIndex) {\n // There are multiple justifications and this one isn't it: shift offscreen\n bucket.text.placedSymbolArray.get(index).crossTileID = 0;\n } else {\n // Either this is the chosen justification or the justification is hardwired: use this one\n bucket.text.placedSymbolArray.get(index).crossTileID = symbolInstance.crossTileID;\n }\n }\n }\n }\n\n markUsedOrientation(bucket: SymbolBucket, orientation: number, symbolInstance: SymbolInstance) {\n const horizontal = (orientation === WritingMode.horizontal || orientation === WritingMode.horizontalOnly) ? orientation : 0;\n const vertical = orientation === WritingMode.vertical ? orientation : 0;\n\n const horizontalIndexes = [\n symbolInstance.leftJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.rightJustifiedTextSymbolIndex\n ];\n\n for (const index of horizontalIndexes) {\n bucket.text.placedSymbolArray.get(index).placedOrientation = horizontal;\n }\n\n if (symbolInstance.verticalPlacedTextSymbolIndex) {\n bucket.text.placedSymbolArray.get(symbolInstance.verticalPlacedTextSymbolIndex).placedOrientation = vertical;\n }\n }\n\n commit(now: number): void {\n this.commitTime = now;\n this.zoomAtLastRecencyCheck = this.transform.zoom;\n\n const prevPlacement = this.prevPlacement;\n let placementChanged = false;\n\n this.prevZoomAdjustment = prevPlacement ? prevPlacement.zoomAdjustment(this.transform.zoom) : 0;\n const increment = prevPlacement ? prevPlacement.symbolFadeChange(now) : 1;\n\n const prevOpacities = prevPlacement ? prevPlacement.opacities : {};\n const prevOffsets = prevPlacement ? prevPlacement.variableOffsets : {};\n const prevOrientations = prevPlacement ? prevPlacement.placedOrientations : {};\n\n // add the opacities from the current placement, and copy their current values from the previous placement\n for (const crossTileID in this.placements) {\n const jointPlacement = this.placements[crossTileID];\n const prevOpacity = prevOpacities[crossTileID];\n if (prevOpacity) {\n this.opacities[crossTileID] = new JointOpacityState(prevOpacity, increment, jointPlacement.text, jointPlacement.icon);\n placementChanged = placementChanged ||\n jointPlacement.text !== prevOpacity.text.placed ||\n jointPlacement.icon !== prevOpacity.icon.placed;\n } else {\n this.opacities[crossTileID] = new JointOpacityState(null, increment, jointPlacement.text, jointPlacement.icon, jointPlacement.skipFade);\n placementChanged = placementChanged || jointPlacement.text || jointPlacement.icon;\n }\n }\n\n // copy and update values from the previous placement that aren't in the current placement but haven't finished fading\n for (const crossTileID in prevOpacities) {\n const prevOpacity = prevOpacities[crossTileID];\n if (!this.opacities[crossTileID]) {\n const jointOpacity = new JointOpacityState(prevOpacity, increment, false, false);\n if (!jointOpacity.isHidden()) {\n this.opacities[crossTileID] = jointOpacity;\n placementChanged = placementChanged || prevOpacity.text.placed || prevOpacity.icon.placed;\n }\n }\n }\n for (const crossTileID in prevOffsets) {\n if (!this.variableOffsets[crossTileID] && this.opacities[crossTileID] && !this.opacities[crossTileID].isHidden()) {\n this.variableOffsets[crossTileID] = prevOffsets[crossTileID];\n }\n }\n\n for (const crossTileID in prevOrientations) {\n if (!this.placedOrientations[crossTileID] && this.opacities[crossTileID] && !this.opacities[crossTileID].isHidden()) {\n this.placedOrientations[crossTileID] = prevOrientations[crossTileID];\n }\n }\n\n // this.lastPlacementChangeTime is the time of the last commit() that\n // resulted in a placement change -- in other words, the start time of\n // the last symbol fade animation\n if (prevPlacement && prevPlacement.lastPlacementChangeTime === undefined) {\n throw new Error('Last placement time for previous placement is not defined');\n }\n if (placementChanged) {\n this.lastPlacementChangeTime = now;\n } else if (typeof this.lastPlacementChangeTime !== 'number') {\n this.lastPlacementChangeTime = prevPlacement ? prevPlacement.lastPlacementChangeTime : now;\n }\n }\n\n updateLayerOpacities(styleLayer: StyleLayer, tiles: Array) {\n const seenCrossTileIDs = {};\n for (const tile of tiles) {\n const symbolBucket = tile.getBucket(styleLayer) as SymbolBucket;\n if (symbolBucket && tile.latestFeatureIndex && styleLayer.id === symbolBucket.layerIds[0]) {\n this.updateBucketOpacities(symbolBucket, seenCrossTileIDs, tile.collisionBoxArray);\n }\n }\n }\n\n updateBucketOpacities(bucket: SymbolBucket, seenCrossTileIDs: {\n [k in string | number]: boolean;\n }, collisionBoxArray?: CollisionBoxArray | null) {\n if (bucket.hasTextData()) {\n bucket.text.opacityVertexArray.clear();\n bucket.text.hasVisibleVertices = false;\n }\n if (bucket.hasIconData()) {\n bucket.icon.opacityVertexArray.clear();\n bucket.icon.hasVisibleVertices = false;\n }\n if (bucket.hasIconCollisionBoxData()) bucket.iconCollisionBox.collisionVertexArray.clear();\n if (bucket.hasTextCollisionBoxData()) bucket.textCollisionBox.collisionVertexArray.clear();\n\n const layer = bucket.layers[0];\n const layout = layer.layout;\n const duplicateOpacityState = new JointOpacityState(null, 0, false, false, true);\n const textAllowOverlap = layout.get('text-allow-overlap');\n const iconAllowOverlap = layout.get('icon-allow-overlap');\n const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset');\n const rotateWithMap = layout.get('text-rotation-alignment') === 'map';\n const pitchWithMap = layout.get('text-pitch-alignment') === 'map';\n const hasIconTextFit = layout.get('icon-text-fit') !== 'none';\n // If allow-overlap is true, we can show symbols before placement runs on them\n // But we have to wait for placement if we potentially depend on a paired icon/text\n // with allow-overlap: false.\n // See https://github.com/mapbox/mapbox-gl-js/issues/7032\n const defaultOpacityState = new JointOpacityState(null, 0,\n textAllowOverlap && (iconAllowOverlap || !bucket.hasIconData() || layout.get('icon-optional')),\n iconAllowOverlap && (textAllowOverlap || !bucket.hasTextData() || layout.get('text-optional')),\n true);\n\n if (!bucket.collisionArrays && collisionBoxArray && ((bucket.hasIconCollisionBoxData() || bucket.hasTextCollisionBoxData()))) {\n bucket.deserializeCollisionBoxes(collisionBoxArray);\n }\n\n const addOpacities = (iconOrText, numVertices: number, opacity: number) => {\n for (let i = 0; i < numVertices / 4; i++) {\n iconOrText.opacityVertexArray.emplaceBack(opacity);\n }\n iconOrText.hasVisibleVertices = iconOrText.hasVisibleVertices || (opacity !== PACKED_HIDDEN_OPACITY);\n };\n\n for (let s = 0; s < bucket.symbolInstances.length; s++) {\n const symbolInstance = bucket.symbolInstances.get(s);\n const {\n numHorizontalGlyphVertices,\n numVerticalGlyphVertices,\n crossTileID\n } = symbolInstance;\n\n const isDuplicate = seenCrossTileIDs[crossTileID];\n\n let opacityState = this.opacities[crossTileID];\n if (isDuplicate) {\n opacityState = duplicateOpacityState;\n } else if (!opacityState) {\n opacityState = defaultOpacityState;\n // store the state so that future placements use it as a starting point\n this.opacities[crossTileID] = opacityState;\n }\n\n seenCrossTileIDs[crossTileID] = true;\n\n const hasText = numHorizontalGlyphVertices > 0 || numVerticalGlyphVertices > 0;\n const hasIcon = symbolInstance.numIconVertices > 0;\n\n const placedOrientation = this.placedOrientations[symbolInstance.crossTileID];\n const horizontalHidden = placedOrientation === WritingMode.vertical;\n const verticalHidden = placedOrientation === WritingMode.horizontal || placedOrientation === WritingMode.horizontalOnly;\n\n if (hasText) {\n const packedOpacity = packOpacity(opacityState.text);\n // Vertical text fades in/out on collision the same way as corresponding\n // horizontal text. Switch between vertical/horizontal should be instantaneous\n const horizontalOpacity = horizontalHidden ? PACKED_HIDDEN_OPACITY : packedOpacity;\n addOpacities(bucket.text, numHorizontalGlyphVertices, horizontalOpacity);\n const verticalOpacity = verticalHidden ? PACKED_HIDDEN_OPACITY : packedOpacity;\n addOpacities(bucket.text, numVerticalGlyphVertices, verticalOpacity);\n\n // If this label is completely faded, mark it so that we don't have to calculate\n // its position at render time. If this layer has variable placement, shift the various\n // symbol instances appropriately so that symbols from buckets that have yet to be placed\n // offset appropriately.\n const symbolHidden = opacityState.text.isHidden();\n [\n symbolInstance.rightJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.leftJustifiedTextSymbolIndex\n ].forEach(index => {\n if (index >= 0) {\n bucket.text.placedSymbolArray.get(index).hidden = symbolHidden || horizontalHidden ? 1 : 0;\n }\n });\n\n if (symbolInstance.verticalPlacedTextSymbolIndex >= 0) {\n bucket.text.placedSymbolArray.get(symbolInstance.verticalPlacedTextSymbolIndex).hidden = symbolHidden || verticalHidden ? 1 : 0;\n }\n\n const prevOffset = this.variableOffsets[symbolInstance.crossTileID];\n if (prevOffset) {\n this.markUsedJustification(bucket, prevOffset.anchor, symbolInstance, placedOrientation);\n }\n\n const prevOrientation = this.placedOrientations[symbolInstance.crossTileID];\n if (prevOrientation) {\n this.markUsedJustification(bucket, 'left', symbolInstance, prevOrientation);\n this.markUsedOrientation(bucket, prevOrientation, symbolInstance);\n }\n }\n\n if (hasIcon) {\n const packedOpacity = packOpacity(opacityState.icon);\n\n const useHorizontal = !(hasIconTextFit && symbolInstance.verticalPlacedIconSymbolIndex && horizontalHidden);\n\n if (symbolInstance.placedIconSymbolIndex >= 0) {\n const horizontalOpacity = useHorizontal ? packedOpacity : PACKED_HIDDEN_OPACITY;\n addOpacities(bucket.icon, symbolInstance.numIconVertices, horizontalOpacity);\n bucket.icon.placedSymbolArray.get(symbolInstance.placedIconSymbolIndex).hidden =\n (opacityState.icon.isHidden() as any);\n }\n\n if (symbolInstance.verticalPlacedIconSymbolIndex >= 0) {\n const verticalOpacity = !useHorizontal ? packedOpacity : PACKED_HIDDEN_OPACITY;\n addOpacities(bucket.icon, symbolInstance.numVerticalIconVertices, verticalOpacity);\n bucket.icon.placedSymbolArray.get(symbolInstance.verticalPlacedIconSymbolIndex).hidden =\n (opacityState.icon.isHidden() as any);\n }\n }\n\n if (bucket.hasIconCollisionBoxData() || bucket.hasTextCollisionBoxData()) {\n const collisionArrays = bucket.collisionArrays[s];\n if (collisionArrays) {\n let shift = new Point(0, 0);\n if (collisionArrays.textBox || collisionArrays.verticalTextBox) {\n let used = true;\n if (hasVariablePlacement) {\n const variableOffset = this.variableOffsets[crossTileID];\n if (variableOffset) {\n // This will show either the currently placed position or the last\n // successfully placed position (so you can visualize what collision\n // just made the symbol disappear, and the most likely place for the\n // symbol to come back)\n shift = calculateVariableLayoutShift(variableOffset.anchor,\n variableOffset.width,\n variableOffset.height,\n variableOffset.textOffset,\n variableOffset.textBoxScale);\n if (rotateWithMap) {\n shift._rotate(pitchWithMap ? this.transform.angle : -this.transform.angle);\n }\n } else {\n // No offset -> this symbol hasn't been placed since coming on-screen\n // No single box is particularly meaningful and all of them would be too noisy\n // Use the center box just to show something's there, but mark it \"not used\"\n used = false;\n }\n }\n\n if (collisionArrays.textBox) {\n updateCollisionVertices(bucket.textCollisionBox.collisionVertexArray, opacityState.text.placed, !used || horizontalHidden, shift.x, shift.y);\n }\n if (collisionArrays.verticalTextBox) {\n updateCollisionVertices(bucket.textCollisionBox.collisionVertexArray, opacityState.text.placed, !used || verticalHidden, shift.x, shift.y);\n }\n }\n\n const verticalIconUsed = Boolean(!verticalHidden && collisionArrays.verticalIconBox);\n\n if (collisionArrays.iconBox) {\n updateCollisionVertices(bucket.iconCollisionBox.collisionVertexArray, opacityState.icon.placed, verticalIconUsed,\n hasIconTextFit ? shift.x : 0,\n hasIconTextFit ? shift.y : 0);\n }\n\n if (collisionArrays.verticalIconBox) {\n updateCollisionVertices(bucket.iconCollisionBox.collisionVertexArray, opacityState.icon.placed, !verticalIconUsed,\n hasIconTextFit ? shift.x : 0,\n hasIconTextFit ? shift.y : 0);\n }\n }\n }\n }\n\n bucket.sortFeatures(this.transform.angle);\n if (this.retainedQueryData[bucket.bucketInstanceId]) {\n this.retainedQueryData[bucket.bucketInstanceId].featureSortOrder = bucket.featureSortOrder;\n }\n\n if (bucket.hasTextData() && bucket.text.opacityVertexBuffer) {\n bucket.text.opacityVertexBuffer.updateData(bucket.text.opacityVertexArray);\n }\n if (bucket.hasIconData() && bucket.icon.opacityVertexBuffer) {\n bucket.icon.opacityVertexBuffer.updateData(bucket.icon.opacityVertexArray);\n }\n if (bucket.hasIconCollisionBoxData() && bucket.iconCollisionBox.collisionVertexBuffer) {\n bucket.iconCollisionBox.collisionVertexBuffer.updateData(bucket.iconCollisionBox.collisionVertexArray);\n }\n if (bucket.hasTextCollisionBoxData() && bucket.textCollisionBox.collisionVertexBuffer) {\n bucket.textCollisionBox.collisionVertexBuffer.updateData(bucket.textCollisionBox.collisionVertexArray);\n }\n\n if (bucket.text.opacityVertexArray.length !== bucket.text.layoutVertexArray.length / 4) throw new Error(`bucket.text.opacityVertexArray.length (= ${bucket.text.opacityVertexArray.length}) !== bucket.text.layoutVertexArray.length (= ${bucket.text.layoutVertexArray.length}) / 4`);\n if (bucket.icon.opacityVertexArray.length !== bucket.icon.layoutVertexArray.length / 4) throw new Error(`bucket.icon.opacityVertexArray.length (= ${bucket.icon.opacityVertexArray.length}) !== bucket.icon.layoutVertexArray.length (= ${bucket.icon.layoutVertexArray.length}) / 4`);\n\n // Push generated collision circles to the bucket for debug rendering\n if (bucket.bucketInstanceId in this.collisionCircleArrays) {\n const instance = this.collisionCircleArrays[bucket.bucketInstanceId];\n\n bucket.placementInvProjMatrix = instance.invProjMatrix;\n bucket.placementViewportMatrix = instance.viewportMatrix;\n bucket.collisionCircleArray = instance.circles;\n\n delete this.collisionCircleArrays[bucket.bucketInstanceId];\n }\n }\n\n symbolFadeChange(now: number) {\n return this.fadeDuration === 0 ?\n 1 :\n ((now - this.commitTime) / this.fadeDuration + this.prevZoomAdjustment);\n }\n\n zoomAdjustment(zoom: number) {\n // When zooming out quickly, labels can overlap each other. This\n // adjustment is used to reduce the interval between placement calculations\n // and to reduce the fade duration when zooming out quickly. Discovering the\n // collisions more quickly and fading them more quickly reduces the unwanted effect.\n return Math.max(0, (this.transform.zoom - zoom) / 1.5);\n }\n\n hasTransitions(now: number) {\n return this.stale ||\n now - this.lastPlacementChangeTime < this.fadeDuration;\n }\n\n stillRecent(now: number, zoom: number) {\n // The adjustment makes placement more frequent when zooming.\n // This condition applies the adjustment only after the map has\n // stopped zooming. This avoids adding extra jank while zooming.\n const durationAdjustment = this.zoomAtLastRecencyCheck === zoom ?\n (1 - this.zoomAdjustment(zoom)) :\n 1;\n this.zoomAtLastRecencyCheck = zoom;\n\n return this.commitTime + this.fadeDuration * durationAdjustment > now;\n }\n\n setStale() {\n this.stale = true;\n }\n}\n\nfunction updateCollisionVertices(collisionVertexArray: CollisionVertexArray, placed: boolean, notUsed: boolean | number, shiftX?: number, shiftY?: number) {\n collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0);\n collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0);\n collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0);\n collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0);\n}\n\n// All four vertices for a glyph will have the same opacity state\n// So we pack the opacity into a uint8, and then repeat it four times\n// to make a single uint32 that we can upload for each glyph in the\n// label.\nconst shift25 = Math.pow(2, 25);\nconst shift24 = Math.pow(2, 24);\nconst shift17 = Math.pow(2, 17);\nconst shift16 = Math.pow(2, 16);\nconst shift9 = Math.pow(2, 9);\nconst shift8 = Math.pow(2, 8);\nconst shift1 = Math.pow(2, 1);\nfunction packOpacity(opacityState: OpacityState): number {\n if (opacityState.opacity === 0 && !opacityState.placed) {\n return 0;\n } else if (opacityState.opacity === 1 && opacityState.placed) {\n return 4294967295;\n }\n const targetBit = opacityState.placed ? 1 : 0;\n const opacityBits = Math.floor(opacityState.opacity * 127);\n return opacityBits * shift25 + targetBit * shift24 +\n opacityBits * shift17 + targetBit * shift16 +\n opacityBits * shift9 + targetBit * shift8 +\n opacityBits * shift1 + targetBit;\n}\n\nconst PACKED_HIDDEN_OPACITY = 0;\n","import {browser} from '../util/browser';\n\nimport {Placement} from '../symbol/placement';\n\nimport type {Transform} from '../geo/transform';\nimport type {StyleLayer} from './style_layer';\nimport type {SymbolStyleLayer} from './style_layer/symbol_style_layer';\nimport type {Tile} from '../source/tile';\nimport type {BucketPart} from '../symbol/placement';\nimport {Terrain} from '../render/terrain';\n\nclass LayerPlacement {\n _sortAcrossTiles: boolean;\n _currentTileIndex: number;\n _currentPartIndex: number;\n _seenCrossTileIDs: {\n [k in string | number]: boolean;\n };\n _bucketParts: Array;\n\n constructor(styleLayer: SymbolStyleLayer) {\n this._sortAcrossTiles = styleLayer.layout.get('symbol-z-order') !== 'viewport-y' &&\n !styleLayer.layout.get('symbol-sort-key').isConstant();\n\n this._currentTileIndex = 0;\n this._currentPartIndex = 0;\n this._seenCrossTileIDs = {};\n this._bucketParts = [];\n }\n\n continuePlacement(tiles: Array, placement: Placement, showCollisionBoxes: boolean, styleLayer: StyleLayer, shouldPausePlacement: () => boolean) {\n\n const bucketParts = this._bucketParts;\n\n while (this._currentTileIndex < tiles.length) {\n const tile = tiles[this._currentTileIndex];\n placement.getBucketParts(bucketParts, styleLayer, tile, this._sortAcrossTiles);\n\n this._currentTileIndex++;\n if (shouldPausePlacement()) {\n return true;\n }\n }\n\n if (this._sortAcrossTiles) {\n this._sortAcrossTiles = false;\n bucketParts.sort((a, b) => (a.sortKey as any as number) - (b.sortKey as any as number));\n }\n\n while (this._currentPartIndex < bucketParts.length) {\n const bucketPart = bucketParts[this._currentPartIndex];\n placement.placeLayerBucketPart(bucketPart, this._seenCrossTileIDs, showCollisionBoxes);\n\n this._currentPartIndex++;\n if (shouldPausePlacement()) {\n return true;\n }\n }\n return false;\n }\n}\n\nexport class PauseablePlacement {\n placement: Placement;\n _done: boolean;\n _currentPlacementIndex: number;\n _forceFullPlacement: boolean;\n _showCollisionBoxes: boolean;\n _inProgressLayer: LayerPlacement;\n\n constructor(\n transform: Transform,\n terrain: Terrain,\n order: Array,\n forceFullPlacement: boolean,\n showCollisionBoxes: boolean,\n fadeDuration: number,\n crossSourceCollisions: boolean,\n prevPlacement?: Placement\n ) {\n this.placement = new Placement(transform, terrain, fadeDuration, crossSourceCollisions, prevPlacement);\n this._currentPlacementIndex = order.length - 1;\n this._forceFullPlacement = forceFullPlacement;\n this._showCollisionBoxes = showCollisionBoxes;\n this._done = false;\n }\n\n isDone() {\n return this._done;\n }\n\n continuePlacement(\n order: Array,\n layers: {[_: string]: StyleLayer},\n layerTiles: {[_: string]: Array}\n ) {\n const startTime = browser.now();\n\n const shouldPausePlacement = () => {\n return this._forceFullPlacement ? false : (browser.now() - startTime) > 2;\n };\n\n while (this._currentPlacementIndex >= 0) {\n const layerId = order[this._currentPlacementIndex];\n const layer = layers[layerId];\n const placementZoom = this.placement.collisionIndex.transform.zoom;\n if (layer.type === 'symbol' &&\n (!layer.minzoom || layer.minzoom <= placementZoom) &&\n (!layer.maxzoom || layer.maxzoom > placementZoom)) {\n\n if (!this._inProgressLayer) {\n this._inProgressLayer = new LayerPlacement(layer as any as SymbolStyleLayer);\n }\n\n const pausePlacement = this._inProgressLayer.continuePlacement(layerTiles[layer.source], this.placement, this._showCollisionBoxes, layer, shouldPausePlacement);\n\n if (pausePlacement) {\n // We didn't finish placing all layers within 2ms,\n // but we can keep rendering with a partial placement\n // We'll resume here on the next frame\n return;\n }\n\n delete this._inProgressLayer;\n }\n\n this._currentPlacementIndex--;\n }\n\n this._done = true;\n }\n\n commit(now: number) {\n this.placement.commit(now);\n return this.placement;\n }\n}\n","import KDBush from 'kdbush';\nimport {EXTENT} from '../data/extent';\n\nimport {SymbolInstanceArray} from '../data/array_types.g';\n\nimport type {SymbolInstance} from '../data/array_types.g';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {Tile} from '../source/tile';\n\n/*\n The CrossTileSymbolIndex generally works on the assumption that\n a conceptual \"unique symbol\" can be identified by the text of\n the label combined with the anchor point. The goal is to assign\n these conceptual \"unique symbols\" a shared crossTileID that can be\n used by Placement to keep fading opacity states consistent and to\n deduplicate labels.\n\n The CrossTileSymbolIndex indexes all the current symbol instances and\n their crossTileIDs. When a symbol bucket gets added or updated, the\n index assigns a crossTileID to each of it's symbol instances by either\n matching it with an existing id or assigning a new one.\n*/\n\n// Round anchor positions to roughly 4 pixel grid\nconst roundingFactor = 512 / EXTENT / 2;\n\nexport const KDBUSH_THRESHHOLD = 128;\n\ninterface SymbolsByKeyEntry {\n index?: KDBush;\n positions?: {x: number; y: number}[];\n crossTileIDs: number[];\n}\n\nclass TileLayerIndex {\n _symbolsByKey: Record = {};\n\n constructor(public tileID: OverscaledTileID, symbolInstances: SymbolInstanceArray, public bucketInstanceId: number) {\n // group the symbolInstances by key\n const symbolInstancesByKey = new Map();\n for (let i = 0; i < symbolInstances.length; i++) {\n const symbolInstance = symbolInstances.get(i);\n const key = symbolInstance.key;\n const instances = symbolInstancesByKey.get(key);\n if (instances) {\n // This tile may have multiple symbol instances with the same key\n // Store each one along with its coordinates\n instances.push(symbolInstance);\n } else {\n symbolInstancesByKey.set(key, [symbolInstance]);\n }\n }\n\n // index the SymbolInstances in this each bucket\n for (const [key, symbols] of symbolInstancesByKey) {\n const positions = symbols.map(symbolInstance => ({x: Math.floor(symbolInstance.anchorX * roundingFactor), y: Math.floor(symbolInstance.anchorY * roundingFactor)}));\n const crossTileIDs = symbols.map(v => v.crossTileID);\n const entry: SymbolsByKeyEntry = {positions, crossTileIDs};\n\n // once we get too many symbols for a given key, it becomes much faster to index it before queries\n if (entry.positions.length > KDBUSH_THRESHHOLD) {\n\n const index = new KDBush(entry.positions.length, 16, Uint16Array);\n for (const {x, y} of entry.positions) index.add(x, y);\n index.finish();\n\n // clear all references to the original positions data\n delete entry.positions;\n entry.index = index;\n }\n\n this._symbolsByKey[key] = entry;\n }\n }\n\n // Converts the coordinates of the input symbol instance into coordinates that be can compared\n // against other symbols in this index. Coordinates are:\n // (1) local-tile-based (so after correction we get x,y values relative to our local anchorX/Y)\n // (2) converted to the z-scale of this TileLayerIndex\n // (3) down-sampled by \"roundingFactor\" from tile coordinate precision in order to be\n // more tolerant of small differences between tiles.\n getScaledCoordinates(symbolInstance: SymbolInstance, childTileID: OverscaledTileID): {x: number; y: number} {\n const {x: localX, y: localY, z: localZ} = this.tileID.canonical;\n const {x, y, z} = childTileID.canonical;\n\n const zDifference = z - localZ;\n const scale = roundingFactor / Math.pow(2, zDifference);\n const xWorld = (x * EXTENT + symbolInstance.anchorX) * scale;\n const yWorld = (y * EXTENT + symbolInstance.anchorY) * scale;\n const xOffset = localX * EXTENT * roundingFactor;\n const yOffset = localY * EXTENT * roundingFactor;\n const result = {\n x: Math.floor(xWorld - xOffset),\n y: Math.floor(yWorld - yOffset)\n };\n\n return result;\n }\n\n findMatches(symbolInstances: SymbolInstanceArray, newTileID: OverscaledTileID, zoomCrossTileIDs: {\n [crossTileID: number]: boolean;\n }) {\n const tolerance = this.tileID.canonical.z < newTileID.canonical.z ? 1 : Math.pow(2, this.tileID.canonical.z - newTileID.canonical.z);\n\n for (let i = 0; i < symbolInstances.length; i++) {\n const symbolInstance = symbolInstances.get(i);\n if (symbolInstance.crossTileID) {\n // already has a match, skip\n continue;\n }\n\n const entry = this._symbolsByKey[symbolInstance.key];\n if (!entry) {\n // No symbol with this key in this bucket\n continue;\n }\n\n const scaledSymbolCoord = this.getScaledCoordinates(symbolInstance, newTileID);\n\n if (entry.index) {\n // Return any symbol with the same keys whose coordinates are within 1\n // grid unit. (with a 4px grid, this covers a 12px by 12px area)\n const indexes = entry.index.range(\n scaledSymbolCoord.x - tolerance,\n scaledSymbolCoord.y - tolerance,\n scaledSymbolCoord.x + tolerance,\n scaledSymbolCoord.y + tolerance).sort();\n\n for (const i of indexes) {\n const crossTileID = entry.crossTileIDs[i];\n\n if (!zoomCrossTileIDs[crossTileID]) {\n // Once we've marked ourselves duplicate against this parent symbol,\n // don't let any other symbols at the same zoom level duplicate against\n // the same parent (see issue #5993)\n zoomCrossTileIDs[crossTileID] = true;\n symbolInstance.crossTileID = crossTileID;\n break;\n }\n }\n } else if (entry.positions) {\n for (let i = 0; i < entry.positions.length; i++) {\n const thisTileSymbol = entry.positions[i];\n const crossTileID = entry.crossTileIDs[i];\n\n // Return any symbol with the same keys whose coordinates are within 1\n // grid unit. (with a 4px grid, this covers a 12px by 12px area)\n if (Math.abs(thisTileSymbol.x - scaledSymbolCoord.x) <= tolerance &&\n Math.abs(thisTileSymbol.y - scaledSymbolCoord.y) <= tolerance &&\n !zoomCrossTileIDs[crossTileID]) {\n // Once we've marked ourselves duplicate against this parent symbol,\n // don't let any other symbols at the same zoom level duplicate against\n // the same parent (see issue #5993)\n zoomCrossTileIDs[crossTileID] = true;\n symbolInstance.crossTileID = crossTileID;\n break;\n }\n }\n }\n }\n }\n\n getCrossTileIDsLists() {\n return Object.values(this._symbolsByKey).map(({crossTileIDs}) => crossTileIDs);\n }\n}\n\nclass CrossTileIDs {\n maxCrossTileID: number;\n constructor() {\n this.maxCrossTileID = 0;\n }\n generate() {\n return ++this.maxCrossTileID;\n }\n}\n\nclass CrossTileSymbolLayerIndex {\n indexes: {\n [zoom in string | number]: {\n [tileId in string | number]: TileLayerIndex;\n };\n };\n usedCrossTileIDs: {\n [zoom in string | number]: {\n [crossTileID: number]: boolean;\n };\n };\n lng: number;\n\n constructor() {\n this.indexes = {};\n this.usedCrossTileIDs = {};\n this.lng = 0;\n }\n\n /*\n * Sometimes when a user pans across the antimeridian the longitude value gets wrapped.\n * To prevent labels from flashing out and in we adjust the tileID values in the indexes\n * so that they match the new wrapped version of the map.\n */\n handleWrapJump(lng: number) {\n const wrapDelta = Math.round((lng - this.lng) / 360);\n if (wrapDelta !== 0) {\n for (const zoom in this.indexes) {\n const zoomIndexes = this.indexes[zoom];\n const newZoomIndex = {};\n for (const key in zoomIndexes) {\n // change the tileID's wrap and add it to a new index\n const index = zoomIndexes[key];\n index.tileID = index.tileID.unwrapTo(index.tileID.wrap + wrapDelta);\n newZoomIndex[index.tileID.key] = index;\n }\n this.indexes[zoom] = newZoomIndex;\n }\n }\n this.lng = lng;\n }\n\n addBucket(tileID: OverscaledTileID, bucket: SymbolBucket, crossTileIDs: CrossTileIDs) {\n if (this.indexes[tileID.overscaledZ] &&\n this.indexes[tileID.overscaledZ][tileID.key]) {\n if (this.indexes[tileID.overscaledZ][tileID.key].bucketInstanceId ===\n bucket.bucketInstanceId) {\n return false;\n } else {\n // We're replacing this bucket with an updated version\n // Remove the old bucket's \"used crossTileIDs\" now so that\n // the new bucket can claim them.\n // The old index entries themselves stick around until\n // 'removeStaleBuckets' is called.\n this.removeBucketCrossTileIDs(tileID.overscaledZ,\n this.indexes[tileID.overscaledZ][tileID.key]);\n }\n }\n\n for (let i = 0; i < bucket.symbolInstances.length; i++) {\n const symbolInstance = bucket.symbolInstances.get(i);\n symbolInstance.crossTileID = 0;\n }\n\n if (!this.usedCrossTileIDs[tileID.overscaledZ]) {\n this.usedCrossTileIDs[tileID.overscaledZ] = {};\n }\n const zoomCrossTileIDs = this.usedCrossTileIDs[tileID.overscaledZ];\n\n for (const zoom in this.indexes) {\n const zoomIndexes = this.indexes[zoom];\n if (Number(zoom) > tileID.overscaledZ) {\n for (const id in zoomIndexes) {\n const childIndex = zoomIndexes[id];\n if (childIndex.tileID.isChildOf(tileID)) {\n childIndex.findMatches(bucket.symbolInstances, tileID, zoomCrossTileIDs);\n }\n }\n } else {\n const parentCoord = tileID.scaledTo(Number(zoom));\n const parentIndex = zoomIndexes[parentCoord.key];\n if (parentIndex) {\n parentIndex.findMatches(bucket.symbolInstances, tileID, zoomCrossTileIDs);\n }\n }\n }\n\n for (let i = 0; i < bucket.symbolInstances.length; i++) {\n const symbolInstance = bucket.symbolInstances.get(i);\n if (!symbolInstance.crossTileID) {\n // symbol did not match any known symbol, assign a new id\n symbolInstance.crossTileID = crossTileIDs.generate();\n zoomCrossTileIDs[symbolInstance.crossTileID] = true;\n }\n }\n\n if (this.indexes[tileID.overscaledZ] === undefined) {\n this.indexes[tileID.overscaledZ] = {};\n }\n this.indexes[tileID.overscaledZ][tileID.key] = new TileLayerIndex(tileID, bucket.symbolInstances, bucket.bucketInstanceId);\n\n return true;\n }\n\n removeBucketCrossTileIDs(zoom: string | number, removedBucket: TileLayerIndex) {\n for (const crossTileIDs of removedBucket.getCrossTileIDsLists()) {\n for (const crossTileID of crossTileIDs) {\n delete this.usedCrossTileIDs[zoom][crossTileID];\n }\n }\n }\n\n removeStaleBuckets(currentIDs: {\n [k in string | number]: boolean;\n }) {\n let tilesChanged = false;\n for (const z in this.indexes) {\n const zoomIndexes = this.indexes[z];\n for (const tileKey in zoomIndexes) {\n if (!currentIDs[zoomIndexes[tileKey].bucketInstanceId]) {\n this.removeBucketCrossTileIDs(z, zoomIndexes[tileKey]);\n delete zoomIndexes[tileKey];\n tilesChanged = true;\n }\n }\n }\n return tilesChanged;\n }\n}\n\nexport class CrossTileSymbolIndex {\n layerIndexes: {[layerId: string]: CrossTileSymbolLayerIndex};\n crossTileIDs: CrossTileIDs;\n maxBucketInstanceId: number;\n bucketsInCurrentPlacement: {[_: number]: boolean};\n\n constructor() {\n this.layerIndexes = {};\n this.crossTileIDs = new CrossTileIDs();\n this.maxBucketInstanceId = 0;\n this.bucketsInCurrentPlacement = {};\n }\n\n addLayer(styleLayer: StyleLayer, tiles: Array, lng: number) {\n let layerIndex = this.layerIndexes[styleLayer.id];\n if (layerIndex === undefined) {\n layerIndex = this.layerIndexes[styleLayer.id] = new CrossTileSymbolLayerIndex();\n }\n\n let symbolBucketsChanged = false;\n const currentBucketIDs = {};\n\n layerIndex.handleWrapJump(lng);\n\n for (const tile of tiles) {\n const symbolBucket = (tile.getBucket(styleLayer) as any as SymbolBucket);\n if (!symbolBucket || styleLayer.id !== symbolBucket.layerIds[0])\n continue;\n\n if (!symbolBucket.bucketInstanceId) {\n symbolBucket.bucketInstanceId = ++this.maxBucketInstanceId;\n }\n\n if (layerIndex.addBucket(tile.tileID, symbolBucket, this.crossTileIDs)) {\n symbolBucketsChanged = true;\n }\n currentBucketIDs[symbolBucket.bucketInstanceId] = true;\n }\n\n if (layerIndex.removeStaleBuckets(currentBucketIDs)) {\n symbolBucketsChanged = true;\n }\n\n return symbolBucketsChanged;\n }\n\n pruneUnusedLayers(usedLayers: Array) {\n const usedLayerMap = {};\n usedLayers.forEach((usedLayer) => {\n usedLayerMap[usedLayer] = true;\n });\n for (const layerId in this.layerIndexes) {\n if (!usedLayerMap[layerId]) {\n delete this.layerIndexes[layerId];\n }\n }\n }\n}\n","import {Event, ErrorEvent, Evented} from '../util/evented';\nimport {StyleLayer} from './style_layer';\nimport {createStyleLayer} from './create_style_layer';\nimport {loadSprite} from './load_sprite';\nimport {ImageManager} from '../render/image_manager';\nimport {GlyphManager} from '../render/glyph_manager';\nimport {Light} from './light';\nimport {LineAtlas} from '../render/line_atlas';\nimport {pick, clone, extend, deepEqual, filterObject, mapObject} from '../util/util';\nimport {coerceSpriteToArray} from '../util/style';\nimport {getJSON, getReferrer, makeRequest} from '../util/ajax';\nimport {ResourceType} from '../util/request_manager';\nimport {browser} from '../util/browser';\nimport {Dispatcher} from '../util/dispatcher';\nimport {validateStyle, emitValidationErrors as _emitValidationErrors} from './validate_style';\nimport {getSourceType, setSourceType, Source} from '../source/source';\nimport type {SourceClass} from '../source/source';\nimport {QueryRenderedFeaturesOptions, QuerySourceFeatureOptions, queryRenderedFeatures, queryRenderedSymbols, querySourceFeatures} from '../source/query_features';\nimport {SourceCache} from '../source/source_cache';\nimport {GeoJSONSource} from '../source/geojson_source';\nimport {latest as styleSpec, derefLayers as deref, emptyStyle, diff as diffStyles, operations as diffOperations} from '@maplibre/maplibre-gl-style-spec';\nimport {getGlobalWorkerPool} from '../util/global_worker_pool';\nimport {\n registerForPluginStateChange,\n evented as rtlTextPluginEvented,\n triggerPluginCompletionEvent\n} from '../source/rtl_text_plugin';\nimport {PauseablePlacement} from './pauseable_placement';\nimport {ZoomHistory} from './zoom_history';\nimport {CrossTileSymbolIndex} from '../symbol/cross_tile_symbol_index';\nimport {validateCustomStyleLayer} from './style_layer/custom_style_layer';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\n\n// We're skipping validation errors with the `source.canvas` identifier in order\n// to continue to allow canvas sources to be added at runtime/updated in\n// smart setStyle (see https://github.com/mapbox/mapbox-gl-js/pull/6424):\nconst emitValidationErrors = (evented: Evented, errors?: ReadonlyArray<{\n message: string;\n identifier?: string;\n}> | null) =>\n _emitValidationErrors(evented, errors && errors.filter(error => error.identifier !== 'source.canvas'));\n\nimport type {Map} from '../ui/map';\nimport type {Transform} from '../geo/transform';\nimport type {StyleImage} from './style_image';\nimport type {StyleGlyph} from './style_glyph';\nimport type {Callback} from '../types/callback';\nimport type {EvaluationParameters} from './evaluation_parameters';\nimport type {Placement} from '../symbol/placement';\nimport type {Cancelable} from '../types/cancelable';\nimport type {RequestParameters, ResponseCallback} from '../util/ajax';\nimport type {\n LayerSpecification,\n FilterSpecification,\n StyleSpecification,\n LightSpecification,\n SourceSpecification,\n SpriteSpecification,\n} from '@maplibre/maplibre-gl-style-spec';\nimport type {CustomLayerInterface} from './style_layer/custom_style_layer';\nimport type {Validator} from './validate_style';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nconst supportedDiffOperations = pick(diffOperations, [\n 'addLayer',\n 'removeLayer',\n 'setPaintProperty',\n 'setLayoutProperty',\n 'setFilter',\n 'addSource',\n 'removeSource',\n 'setLayerZoomRange',\n 'setLight',\n 'setTransition',\n 'setGeoJSONSourceData',\n 'setGlyphs',\n 'setSprite',\n]);\n\nconst ignoredDiffOperations = pick(diffOperations, [\n 'setCenter',\n 'setZoom',\n 'setBearing',\n 'setPitch'\n]);\n\nconst empty = emptyStyle() as StyleSpecification;\n/**\n * A feature identifier that is bound to a source\n */\nexport type FeatureIdentifier = {\n /**\n * Unique id of the feature.\n */\n id?: string | number | undefined;\n /**\n * The id of the vector or GeoJSON source for the feature.\n */\n source: string;\n /**\n * *For vector tile sources, `sourceLayer` is required.*\n */\n sourceLayer?: string | undefined;\n};\n\n/**\n * The options object related to the {@link Map}'s style related methods\n */\nexport type StyleOptions = {\n /**\n * If false, style validation will be skipped. Useful in production environment.\n */\n validate?: boolean;\n /**\n * Defines a CSS\n * font-family for locally overriding generation of glyphs in the 'CJK Unified Ideographs', 'Hiragana', 'Katakana' and 'Hangul Syllables' ranges.\n * In these ranges, font settings from the map's style will be ignored, except for font-weight keywords (light/regular/medium/bold).\n * Set to `false`, to enable font settings from the map's style for these glyph ranges.\n * Forces a full update.\n */\n localIdeographFontFamily?: string;\n};\n\n/**\n * Supporting type to add validation to another style related type\n */\nexport type StyleSetterOptions = {\n /**\n * Whether to check if the filter conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function.\n */\n validate?: boolean;\n};\n\n/**\n * Part of {@link Map#setStyle} options, transformStyle is a convenience function that allows to modify a style after it is fetched but before it is committed to the map state\n * this function exposes previous and next styles, it can be commonly used to support a range of functionalities like:\n * when previous style carries certain 'state' that needs to be carried over to a new style gracefully\n * when a desired style is a certain combination of previous and incoming style\n * when an incoming style requires modification based on external state\n *\n * @param previousStyle - The current style.\n * @param nextStyle - The next style.\n * @returns resulting style that will to be applied to the map\n *\n * @example\n * ```ts\n * map.setStyle('https://demotiles.maplibre.org/style.json', {\n * transformStyle: (previousStyle, nextStyle) => ({\n * ...nextStyle,\n * sources: {\n * ...nextStyle.sources,\n * // copy a source from previous style\n * 'osm': previousStyle.sources.osm\n * },\n * layers: [\n * // background layer\n * nextStyle.layers[0],\n * // copy a layer from previous style\n * previousStyle.layers[0],\n * // other layers from the next style\n * ...nextStyle.layers.slice(1).map(layer => {\n * // hide the layers we don't need from demotiles style\n * if (layer.id.startsWith('geolines')) {\n * layer.layout = {...layer.layout || {}, visibility: 'none'};\n * // filter out US polygons\n * } else if (layer.id.startsWith('coastline') || layer.id.startsWith('countries')) {\n * layer.filter = ['!=', ['get', 'ADM0_A3'], 'USA'];\n * }\n * return layer;\n * })\n * ]\n * })\n * });\n * ```\n */\nexport type TransformStyleFunction = (previous: StyleSpecification | undefined, next: StyleSpecification) => StyleSpecification;\n\n/**\n * The options object related to the {@link Map}'s style related methods\n */\nexport type StyleSwapOptions = {\n /**\n * If false, force a 'full' update, removing the current style\n * and building the given one instead of attempting a diff-based update.\n */\n diff?: boolean;\n /**\n * TransformStyleFunction is a convenience function\n * that allows to modify a style after it is fetched but before it is committed to the map state. Refer to {@link TransformStyleFunction}.\n */\n transformStyle?: TransformStyleFunction;\n}\n\n/**\n * Specifies a layer to be added to a {@link Style}. In addition to a standard {@link LayerSpecification}\n * or a {@link CustomLayerInterface}, a {@link LayerSpecification} with an embedded {@link SourceSpecification} can also be provided.\n */\nexport type AddLayerObject = LayerSpecification | (Omit & {source: SourceSpecification}) | CustomLayerInterface;\n\n/**\n * The Style base class\n */\nexport class Style extends Evented {\n map: Map;\n stylesheet: StyleSpecification;\n dispatcher: Dispatcher;\n imageManager: ImageManager;\n glyphManager: GlyphManager;\n lineAtlas: LineAtlas;\n light: Light;\n\n _request: Cancelable;\n _spriteRequest: Cancelable;\n _layers: {[_: string]: StyleLayer};\n _serializedLayers: {[_: string]: LayerSpecification};\n _order: Array;\n sourceCaches: {[_: string]: SourceCache};\n zoomHistory: ZoomHistory;\n _loaded: boolean;\n _rtlTextPluginCallback: (a: any) => any;\n _changed: boolean;\n _updatedSources: {[_: string]: 'clear' | 'reload'};\n _updatedLayers: {[_: string]: true};\n _removedLayers: {[_: string]: StyleLayer};\n _changedImages: {[_: string]: true};\n _glyphsDidChange: boolean;\n _updatedPaintProps: {[layer: string]: true};\n _layerOrderChanged: boolean;\n // image ids of images loaded from style's sprite\n _spritesImagesIds: {[spriteId: string]: string[]};\n // image ids of all images loaded (sprite + user)\n _availableImages: Array;\n\n crossTileSymbolIndex: CrossTileSymbolIndex;\n pauseablePlacement: PauseablePlacement;\n placement: Placement;\n z: number;\n\n static registerForPluginStateChange: typeof registerForPluginStateChange;\n\n constructor(map: Map, options: StyleOptions = {}) {\n super();\n\n this.map = map;\n this.dispatcher = new Dispatcher(getGlobalWorkerPool(), this, map._getMapId());\n this.imageManager = new ImageManager();\n this.imageManager.setEventedParent(this);\n this.glyphManager = new GlyphManager(map._requestManager, options.localIdeographFontFamily);\n this.lineAtlas = new LineAtlas(256, 512);\n this.crossTileSymbolIndex = new CrossTileSymbolIndex();\n\n this._spritesImagesIds = {};\n this._layers = {};\n\n this._order = [];\n this.sourceCaches = {};\n this.zoomHistory = new ZoomHistory();\n this._loaded = false;\n this._availableImages = [];\n\n this._resetUpdates();\n\n this.dispatcher.broadcast('setReferrer', getReferrer());\n\n const self = this;\n this._rtlTextPluginCallback = Style.registerForPluginStateChange((event) => {\n const state = {\n pluginStatus: event.pluginStatus,\n pluginURL: event.pluginURL\n };\n self.dispatcher.broadcast('syncRTLPluginState', state, (err, results) => {\n triggerPluginCompletionEvent(err);\n if (results) {\n const allComplete = results.every((elem) => elem);\n if (allComplete) {\n for (const id in self.sourceCaches) {\n const sourceType = self.sourceCaches[id].getSource().type;\n if (sourceType === 'vector' || sourceType === 'geojson') {\n // Non-vector sources don't have any symbols buckets to reload when the RTL text plugin loads\n // They also load more quickly, so they're more likely to have already displaying tiles\n // that would be unnecessarily booted by the plugin load event\n self.sourceCaches[id].reload(); // Should be a no-op if the plugin loads before any tiles load\n }\n }\n }\n }\n\n });\n });\n\n this.on('data', (event) => {\n if (event.dataType !== 'source' || event.sourceDataType !== 'metadata') {\n return;\n }\n\n const sourceCache = this.sourceCaches[event.sourceId];\n if (!sourceCache) {\n return;\n }\n\n const source = sourceCache.getSource();\n if (!source || !source.vectorLayerIds) {\n return;\n }\n\n for (const layerId in this._layers) {\n const layer = this._layers[layerId];\n if (layer.source === source.id) {\n this._validateLayer(layer);\n }\n }\n });\n }\n\n loadURL(url: string, options: StyleSwapOptions & StyleSetterOptions = {}, previousStyle?: StyleSpecification) {\n this.fire(new Event('dataloading', {dataType: 'style'}));\n\n options.validate = typeof options.validate === 'boolean' ?\n options.validate : true;\n\n const request = this.map._requestManager.transformRequest(url, ResourceType.Style);\n this._request = getJSON(request, (error?: Error | null, json?: any | null) => {\n this._request = null;\n if (error) {\n this.fire(new ErrorEvent(error));\n } else if (json) {\n this._load(json, options, previousStyle);\n }\n });\n }\n\n loadJSON(json: StyleSpecification, options: StyleSetterOptions & StyleSwapOptions = {}, previousStyle?: StyleSpecification) {\n this.fire(new Event('dataloading', {dataType: 'style'}));\n\n this._request = browser.frame(() => {\n this._request = null;\n options.validate = options.validate !== false;\n this._load(json, options, previousStyle);\n });\n }\n\n loadEmpty() {\n this.fire(new Event('dataloading', {dataType: 'style'}));\n this._load(empty, {validate: false});\n }\n\n _load(json: StyleSpecification, options: StyleSwapOptions & StyleSetterOptions, previousStyle?: StyleSpecification) {\n const nextState = options.transformStyle ? options.transformStyle(previousStyle, json) : json;\n if (options.validate && emitValidationErrors(this, validateStyle(nextState))) {\n return;\n }\n\n this._loaded = true;\n this.stylesheet = nextState;\n\n for (const id in nextState.sources) {\n this.addSource(id, nextState.sources[id], {validate: false});\n }\n\n if (nextState.sprite) {\n this._loadSprite(nextState.sprite);\n } else {\n this.imageManager.setLoaded(true);\n }\n\n this.glyphManager.setURL(nextState.glyphs);\n this._createLayers();\n\n this.light = new Light(this.stylesheet.light);\n\n this.map.setTerrain(this.stylesheet.terrain ?? null);\n\n this.fire(new Event('data', {dataType: 'style'}));\n this.fire(new Event('style.load'));\n }\n\n private _createLayers() {\n const dereferencedLayers = deref(this.stylesheet.layers);\n\n // Broadcast layers to workers first, so that expensive style processing (createStyleLayer)\n // can happen in parallel on both main and worker threads.\n this.dispatcher.broadcast('setLayers', dereferencedLayers);\n\n this._order = dereferencedLayers.map((layer) => layer.id);\n this._layers = {};\n\n // reset serialization field, to be populated only when needed\n this._serializedLayers = null;\n for (const layer of dereferencedLayers) {\n const styledLayer = createStyleLayer(layer);\n styledLayer.setEventedParent(this, {layer: {id: layer.id}});\n this._layers[layer.id] = styledLayer;\n }\n }\n\n _loadSprite(sprite: SpriteSpecification, isUpdate: boolean = false, completion: (err: Error) => void = undefined) {\n this.imageManager.setLoaded(false);\n\n this._spriteRequest = loadSprite(sprite, this.map._requestManager, this.map.getPixelRatio(), (err, images) => {\n this._spriteRequest = null;\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (images) {\n for (const spriteId in images) {\n this._spritesImagesIds[spriteId] = [];\n\n // remove old sprite's loaded images (for the same sprite id) that are not in new sprite\n const imagesToRemove = this._spritesImagesIds[spriteId] ? this._spritesImagesIds[spriteId].filter(id => !(id in images)) : [];\n for (const id of imagesToRemove) {\n this.imageManager.removeImage(id);\n this._changedImages[id] = true;\n }\n\n for (const id in images[spriteId]) {\n // don't prefix images of the \"default\" sprite\n const imageId = spriteId === 'default' ? id : `${spriteId}:${id}`;\n // save all the sprite's images' ids to be able to delete them in `removeSprite`\n this._spritesImagesIds[spriteId].push(imageId);\n if (imageId in this.imageManager.images) {\n this.imageManager.updateImage(imageId, images[spriteId][id], false);\n } else {\n this.imageManager.addImage(imageId, images[spriteId][id]);\n }\n\n if (isUpdate) {\n this._changedImages[imageId] = true;\n }\n }\n }\n }\n\n this.imageManager.setLoaded(true);\n this._availableImages = this.imageManager.listImages();\n\n if (isUpdate) {\n this._changed = true;\n }\n\n this.dispatcher.broadcast('setImages', this._availableImages);\n this.fire(new Event('data', {dataType: 'style'}));\n\n if (completion) {\n completion(err);\n }\n });\n }\n\n _unloadSprite() {\n for (const id of Object.values(this._spritesImagesIds).flat()) {\n this.imageManager.removeImage(id);\n this._changedImages[id] = true;\n }\n\n this._spritesImagesIds = {};\n this._availableImages = this.imageManager.listImages();\n this._changed = true;\n this.dispatcher.broadcast('setImages', this._availableImages);\n this.fire(new Event('data', {dataType: 'style'}));\n }\n\n _validateLayer(layer: StyleLayer) {\n const sourceCache = this.sourceCaches[layer.source];\n if (!sourceCache) {\n return;\n }\n\n const sourceLayer = layer.sourceLayer;\n if (!sourceLayer) {\n return;\n }\n\n const source = sourceCache.getSource();\n if (source.type === 'geojson' || (source.vectorLayerIds && source.vectorLayerIds.indexOf(sourceLayer) === -1)) {\n this.fire(new ErrorEvent(new Error(\n `Source layer \"${sourceLayer}\" ` +\n `does not exist on source \"${source.id}\" ` +\n `as specified by style layer \"${layer.id}\".`\n )));\n }\n }\n\n loaded() {\n if (!this._loaded)\n return false;\n\n if (Object.keys(this._updatedSources).length)\n return false;\n\n for (const id in this.sourceCaches)\n if (!this.sourceCaches[id].loaded())\n return false;\n\n if (!this.imageManager.isLoaded())\n return false;\n\n return true;\n }\n\n /**\n * take an array of string IDs, and based on this._layers, generate an array of LayerSpecification\n * @param ids - an array of string IDs, for which serialized layers will be generated. If omitted, all serialized layers will be returned\n * @returns generated result\n */\n private _serializeByIds(ids?: Array): Array {\n\n const serializedLayersDictionary = this._serializedAllLayers();\n if (!ids || ids.length === 0) {\n return Object.values(serializedLayersDictionary);\n }\n\n const serializedLayers = [];\n for (const id of ids) {\n // this check will skip all custom layers\n if (serializedLayersDictionary[id]) {\n serializedLayers.push(serializedLayersDictionary[id]);\n }\n }\n\n return serializedLayers;\n }\n\n /**\n * Lazy initialization of this._serializedLayers dictionary and return it\n * @returns this._serializedLayers dictionary\n */\n private _serializedAllLayers(): {[_: string]: LayerSpecification} {\n let serializedLayers = this._serializedLayers;\n if (serializedLayers) {\n return serializedLayers;\n }\n\n serializedLayers = this._serializedLayers = {};\n const allLayerIds: string [] = Object.keys(this._layers);\n for (const layerId of allLayerIds) {\n const layer = this._layers[layerId];\n if (layer.type !== 'custom') {\n serializedLayers[layerId] = layer.serialize();\n }\n }\n\n return serializedLayers;\n }\n\n hasTransitions() {\n if (this.light && this.light.hasTransition()) {\n return true;\n }\n\n for (const id in this.sourceCaches) {\n if (this.sourceCaches[id].hasTransition()) {\n return true;\n }\n }\n\n for (const id in this._layers) {\n if (this._layers[id].hasTransition()) {\n return true;\n }\n }\n\n return false;\n }\n\n _checkLoaded() {\n if (!this._loaded) {\n throw new Error('Style is not done loading.');\n }\n }\n\n /**\n * @internal\n * Apply queued style updates in a batch and recalculate zoom-dependent paint properties.\n */\n update(parameters: EvaluationParameters) {\n if (!this._loaded) {\n return;\n }\n\n const changed = this._changed;\n if (this._changed) {\n const updatedIds = Object.keys(this._updatedLayers);\n const removedIds = Object.keys(this._removedLayers);\n\n if (updatedIds.length || removedIds.length) {\n this._updateWorkerLayers(updatedIds, removedIds);\n }\n for (const id in this._updatedSources) {\n const action = this._updatedSources[id];\n\n if (action === 'reload') {\n this._reloadSource(id);\n } else if (action === 'clear') {\n this._clearSource(id);\n } else {\n throw new Error(`Invalid action ${action}`);\n }\n }\n\n this._updateTilesForChangedImages();\n this._updateTilesForChangedGlyphs();\n\n for (const id in this._updatedPaintProps) {\n this._layers[id].updateTransitions(parameters);\n }\n\n this.light.updateTransitions(parameters);\n\n this._resetUpdates();\n }\n\n const sourcesUsedBefore = {};\n\n for (const sourceId in this.sourceCaches) {\n const sourceCache = this.sourceCaches[sourceId];\n sourcesUsedBefore[sourceId] = sourceCache.used;\n sourceCache.used = false;\n }\n\n for (const layerId of this._order) {\n const layer = this._layers[layerId];\n\n layer.recalculate(parameters, this._availableImages);\n if (!layer.isHidden(parameters.zoom) && layer.source) {\n this.sourceCaches[layer.source].used = true;\n }\n }\n\n for (const sourceId in sourcesUsedBefore) {\n const sourceCache = this.sourceCaches[sourceId];\n if (sourcesUsedBefore[sourceId] !== sourceCache.used) {\n sourceCache.fire(new Event('data', {sourceDataType: 'visibility', dataType: 'source', sourceId}));\n }\n }\n\n this.light.recalculate(parameters);\n this.z = parameters.zoom;\n\n if (changed) {\n this.fire(new Event('data', {dataType: 'style'}));\n }\n\n }\n\n /*\n * Apply any queued image changes.\n */\n _updateTilesForChangedImages() {\n const changedImages = Object.keys(this._changedImages);\n if (changedImages.length) {\n for (const name in this.sourceCaches) {\n this.sourceCaches[name].reloadTilesForDependencies(['icons', 'patterns'], changedImages);\n }\n this._changedImages = {};\n }\n }\n\n _updateTilesForChangedGlyphs() {\n if (this._glyphsDidChange) {\n for (const name in this.sourceCaches) {\n this.sourceCaches[name].reloadTilesForDependencies(['glyphs'], ['']);\n }\n this._glyphsDidChange = false;\n }\n }\n\n _updateWorkerLayers(updatedIds: Array, removedIds: Array) {\n this.dispatcher.broadcast('updateLayers', {\n layers: this._serializeByIds(updatedIds),\n removedIds\n });\n }\n\n _resetUpdates() {\n this._changed = false;\n\n this._updatedLayers = {};\n this._removedLayers = {};\n\n this._updatedSources = {};\n this._updatedPaintProps = {};\n\n this._changedImages = {};\n this._glyphsDidChange = false;\n }\n\n /**\n * Update this style's state to match the given style JSON, performing only\n * the necessary mutations.\n *\n * May throw an Error ('Unimplemented: METHOD') if the mapbox-gl-style-spec\n * diff algorithm produces an operation that is not supported.\n *\n * @returns true if any changes were made; false otherwise\n */\n setState(nextState: StyleSpecification, options: StyleSwapOptions = {}) {\n this._checkLoaded();\n\n const serializedStyle = this.serialize();\n nextState = options.transformStyle ? options.transformStyle(serializedStyle, nextState) : nextState;\n if (emitValidationErrors(this, validateStyle(nextState))) return false;\n\n nextState = clone(nextState);\n nextState.layers = deref(nextState.layers);\n\n const changes = diffStyles(serializedStyle, nextState)\n .filter(op => !(op.command in ignoredDiffOperations));\n\n if (changes.length === 0) {\n return false;\n }\n\n const unimplementedOps = changes.filter(op => !(op.command in supportedDiffOperations));\n if (unimplementedOps.length > 0) {\n throw new Error(`Unimplemented: ${unimplementedOps.map(op => op.command).join(', ')}.`);\n }\n\n for (const op of changes) {\n if (op.command === 'setTransition') {\n // `transition` is always read directly off of\n // `this.stylesheet`, which we update below\n continue;\n }\n (this as any)[op.command].apply(this, op.args);\n }\n\n this.stylesheet = nextState;\n\n // reset serialization field, to be populated only when needed\n this._serializedLayers = null;\n\n return true;\n }\n\n addImage(id: string, image: StyleImage) {\n if (this.getImage(id)) {\n return this.fire(new ErrorEvent(new Error(`An image named \"${id}\" already exists.`)));\n }\n this.imageManager.addImage(id, image);\n this._afterImageUpdated(id);\n }\n\n updateImage(id: string, image: StyleImage) {\n this.imageManager.updateImage(id, image);\n }\n\n getImage(id: string): StyleImage {\n return this.imageManager.getImage(id);\n }\n\n removeImage(id: string) {\n if (!this.getImage(id)) {\n return this.fire(new ErrorEvent(new Error(`An image named \"${id}\" does not exist.`)));\n }\n this.imageManager.removeImage(id);\n this._afterImageUpdated(id);\n }\n\n _afterImageUpdated(id: string) {\n this._availableImages = this.imageManager.listImages();\n this._changedImages[id] = true;\n this._changed = true;\n this.dispatcher.broadcast('setImages', this._availableImages);\n this.fire(new Event('data', {dataType: 'style'}));\n }\n\n listImages() {\n this._checkLoaded();\n\n return this.imageManager.listImages();\n }\n\n addSource(id: string, source: SourceSpecification, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n if (this.sourceCaches[id] !== undefined) {\n throw new Error(`Source \"${id}\" already exists.`);\n }\n\n if (!source.type) {\n throw new Error(`The type property must be defined, but only the following properties were given: ${Object.keys(source).join(', ')}.`);\n }\n\n const builtIns = ['vector', 'raster', 'geojson', 'video', 'image'];\n const shouldValidate = builtIns.indexOf(source.type) >= 0;\n if (shouldValidate && this._validate(validateStyle.source, `sources.${id}`, source, null, options)) return;\n\n if (this.map && this.map._collectResourceTiming) (source as any).collectResourceTiming = true;\n const sourceCache = this.sourceCaches[id] = new SourceCache(id, source, this.dispatcher);\n sourceCache.style = this;\n sourceCache.setEventedParent(this, () => ({\n isSourceLoaded: sourceCache.loaded(),\n source: sourceCache.serialize(),\n sourceId: id\n }));\n\n sourceCache.onAdd(this.map);\n this._changed = true;\n }\n\n /**\n * Remove a source from this stylesheet, given its id.\n * @param id - id of the source to remove\n * @throws if no source is found with the given ID\n * @returns `this`.\n */\n removeSource(id: string): this {\n this._checkLoaded();\n\n if (this.sourceCaches[id] === undefined) {\n throw new Error('There is no source with this ID');\n }\n for (const layerId in this._layers) {\n if (this._layers[layerId].source === id) {\n return this.fire(new ErrorEvent(new Error(`Source \"${id}\" cannot be removed while layer \"${layerId}\" is using it.`)));\n }\n }\n\n const sourceCache = this.sourceCaches[id];\n delete this.sourceCaches[id];\n delete this._updatedSources[id];\n sourceCache.fire(new Event('data', {sourceDataType: 'metadata', dataType: 'source', sourceId: id}));\n sourceCache.setEventedParent(null);\n sourceCache.onRemove(this.map);\n this._changed = true;\n }\n\n /**\n * Set the data of a GeoJSON source, given its id.\n * @param id - id of the source\n * @param data - GeoJSON source\n */\n setGeoJSONSourceData(id: string, data: GeoJSON.GeoJSON | string) {\n this._checkLoaded();\n\n if (this.sourceCaches[id] === undefined) throw new Error(`There is no source with this ID=${id}`);\n const geojsonSource: GeoJSONSource = (this.sourceCaches[id].getSource() as any);\n if (geojsonSource.type !== 'geojson') throw new Error(`geojsonSource.type is ${geojsonSource.type}, which is !== 'geojson`);\n\n geojsonSource.setData(data);\n this._changed = true;\n }\n\n /**\n * Get a source by ID.\n * @param id - ID of the desired source\n * @returns source\n */\n getSource(id: string): Source | undefined {\n return this.sourceCaches[id] && this.sourceCaches[id].getSource();\n }\n\n /**\n * Add a layer to the map style. The layer will be inserted before the layer with\n * ID `before`, or appended if `before` is omitted.\n * @param layerObject - The style layer to add.\n * @param before - ID of an existing layer to insert before\n * @param options - Style setter options.\n * @returns `this`.\n */\n addLayer(layerObject: AddLayerObject, before?: string, options: StyleSetterOptions = {}): this {\n this._checkLoaded();\n\n const id = layerObject.id;\n\n if (this.getLayer(id)) {\n this.fire(new ErrorEvent(new Error(`Layer \"${id}\" already exists on this map.`)));\n return;\n }\n\n let layer: ReturnType;\n if (layerObject.type === 'custom') {\n\n if (emitValidationErrors(this, validateCustomStyleLayer(layerObject))) return;\n\n layer = createStyleLayer(layerObject);\n\n } else {\n if ('source' in layerObject && typeof layerObject.source === 'object') {\n this.addSource(id, layerObject.source);\n layerObject = clone(layerObject);\n layerObject = extend(layerObject, {source: id});\n }\n\n // this layer is not in the style.layers array, so we pass an impossible array index\n if (this._validate(validateStyle.layer,\n `layers.${id}`, layerObject, {arrayIndex: -1}, options)) return;\n\n layer = createStyleLayer(layerObject as LayerSpecification | CustomLayerInterface);\n this._validateLayer(layer);\n\n layer.setEventedParent(this, {layer: {id}});\n }\n\n const index = before ? this._order.indexOf(before) : this._order.length;\n if (before && index === -1) {\n this.fire(new ErrorEvent(new Error(`Cannot add layer \"${id}\" before non-existing layer \"${before}\".`)));\n return;\n }\n\n this._order.splice(index, 0, id);\n this._layerOrderChanged = true;\n\n this._layers[id] = layer;\n\n if (this._removedLayers[id] && layer.source && layer.type !== 'custom') {\n // If, in the current batch, we have already removed this layer\n // and we are now re-adding it with a different `type`, then we\n // need to clear (rather than just reload) the underlying source's\n // tiles. Otherwise, tiles marked 'reloading' will have buckets /\n // buffers that are set up for the _previous_ version of this\n // layer, causing, e.g.:\n // https://github.com/mapbox/mapbox-gl-js/issues/3633\n const removed = this._removedLayers[id];\n delete this._removedLayers[id];\n if (removed.type !== layer.type) {\n this._updatedSources[layer.source] = 'clear';\n } else {\n this._updatedSources[layer.source] = 'reload';\n this.sourceCaches[layer.source].pause();\n }\n }\n this._updateLayer(layer);\n\n if (layer.onAdd) {\n layer.onAdd(this.map);\n }\n }\n\n /**\n * Moves a layer to a different z-position. The layer will be inserted before the layer with\n * ID `before`, or appended if `before` is omitted.\n * @param id - ID of the layer to move\n * @param before - ID of an existing layer to insert before\n */\n moveLayer(id: string, before?: string) {\n this._checkLoaded();\n this._changed = true;\n\n const layer = this._layers[id];\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`The layer '${id}' does not exist in the map's style and cannot be moved.`)));\n return;\n }\n\n if (id === before) {\n return;\n }\n\n const index = this._order.indexOf(id);\n this._order.splice(index, 1);\n\n const newIndex = before ? this._order.indexOf(before) : this._order.length;\n if (before && newIndex === -1) {\n this.fire(new ErrorEvent(new Error(`Cannot move layer \"${id}\" before non-existing layer \"${before}\".`)));\n return;\n }\n this._order.splice(newIndex, 0, id);\n\n this._layerOrderChanged = true;\n }\n\n /**\n * Remove the layer with the given id from the style.\n *\n * If no such layer exists, an `error` event is fired.\n *\n * @param id - id of the layer to remove\n * @event `error` - Fired if the layer does not exist\n */\n removeLayer(id: string) {\n this._checkLoaded();\n\n const layer = this._layers[id];\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot remove non-existing layer \"${id}\".`)));\n return;\n }\n\n layer.setEventedParent(null);\n\n const index = this._order.indexOf(id);\n this._order.splice(index, 1);\n\n this._layerOrderChanged = true;\n this._changed = true;\n this._removedLayers[id] = layer;\n delete this._layers[id];\n\n if (this._serializedLayers) {\n delete this._serializedLayers[id];\n }\n delete this._updatedLayers[id];\n delete this._updatedPaintProps[id];\n\n if (layer.onRemove) {\n layer.onRemove(this.map);\n }\n }\n\n /**\n * Return the style layer object with the given `id`.\n *\n * @param id - id of the desired layer\n * @returns a layer, if one with the given `id` exists\n */\n getLayer(id: string): StyleLayer | undefined {\n return this._layers[id];\n }\n\n /**\n * Return the ids of all layers currently in the style, including custom layers, in order.\n *\n * @returns ids of layers, in order\n */\n getLayersOrder(): string[] {\n return [...this._order];\n }\n\n /**\n * Checks if a specific layer is present within the style.\n *\n * @param id - the id of the desired layer\n * @returns a boolean specifying if the given layer is present\n */\n hasLayer(id: string): boolean {\n return id in this._layers;\n }\n\n setLayerZoomRange(layerId: string, minzoom?: number | null, maxzoom?: number | null) {\n this._checkLoaded();\n\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot set the zoom range of non-existing layer \"${layerId}\".`)));\n return;\n }\n\n if (layer.minzoom === minzoom && layer.maxzoom === maxzoom) return;\n\n if (minzoom != null) {\n layer.minzoom = minzoom;\n }\n if (maxzoom != null) {\n layer.maxzoom = maxzoom;\n }\n this._updateLayer(layer);\n }\n\n setFilter(layerId: string, filter?: FilterSpecification | null, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot filter non-existing layer \"${layerId}\".`)));\n return;\n }\n\n if (deepEqual(layer.filter, filter)) {\n return;\n }\n\n if (filter === null || filter === undefined) {\n layer.filter = undefined;\n this._updateLayer(layer);\n return;\n }\n\n if (this._validate(validateStyle.filter, `layers.${layer.id}.filter`, filter, null, options)) {\n return;\n }\n\n layer.filter = clone(filter);\n this._updateLayer(layer);\n }\n\n /**\n * Get a layer's filter object\n * @param layer - the layer to inspect\n * @returns the layer's filter, if any\n */\n getFilter(layer: string): FilterSpecification | void {\n return clone(this.getLayer(layer).filter);\n }\n\n setLayoutProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot style non-existing layer \"${layerId}\".`)));\n return;\n }\n\n if (deepEqual(layer.getLayoutProperty(name), value)) return;\n\n layer.setLayoutProperty(name, value, options);\n this._updateLayer(layer);\n }\n\n /**\n * Get a layout property's value from a given layer\n * @param layerId - the layer to inspect\n * @param name - the name of the layout property\n * @returns the property value\n */\n getLayoutProperty(layerId: string, name: string) {\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot get style of non-existing layer \"${layerId}\".`)));\n return;\n }\n\n return layer.getLayoutProperty(name);\n }\n\n setPaintProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot style non-existing layer \"${layerId}\".`)));\n return;\n }\n\n if (deepEqual(layer.getPaintProperty(name), value)) return;\n\n const requiresRelayout = layer.setPaintProperty(name, value, options);\n if (requiresRelayout) {\n this._updateLayer(layer);\n }\n\n this._changed = true;\n this._updatedPaintProps[layerId] = true;\n }\n\n getPaintProperty(layer: string, name: string) {\n return this.getLayer(layer).getPaintProperty(name);\n }\n\n setFeatureState(target: FeatureIdentifier, state: any) {\n this._checkLoaded();\n const sourceId = target.source;\n const sourceLayer = target.sourceLayer;\n const sourceCache = this.sourceCaches[sourceId];\n\n if (sourceCache === undefined) {\n this.fire(new ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`)));\n return;\n }\n const sourceType = sourceCache.getSource().type;\n if (sourceType === 'geojson' && sourceLayer) {\n this.fire(new ErrorEvent(new Error('GeoJSON sources cannot have a sourceLayer parameter.')));\n return;\n }\n if (sourceType === 'vector' && !sourceLayer) {\n this.fire(new ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.')));\n return;\n }\n if (target.id === undefined) {\n this.fire(new ErrorEvent(new Error('The feature id parameter must be provided.')));\n }\n\n sourceCache.setFeatureState(sourceLayer, target.id, state);\n }\n\n removeFeatureState(target: FeatureIdentifier, key?: string) {\n this._checkLoaded();\n const sourceId = target.source;\n const sourceCache = this.sourceCaches[sourceId];\n\n if (sourceCache === undefined) {\n this.fire(new ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`)));\n return;\n }\n\n const sourceType = sourceCache.getSource().type;\n const sourceLayer = sourceType === 'vector' ? target.sourceLayer : undefined;\n\n if (sourceType === 'vector' && !sourceLayer) {\n this.fire(new ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.')));\n return;\n }\n\n if (key && (typeof target.id !== 'string' && typeof target.id !== 'number')) {\n this.fire(new ErrorEvent(new Error('A feature id is required to remove its specific state property.')));\n return;\n }\n\n sourceCache.removeFeatureState(sourceLayer, target.id, key);\n }\n\n getFeatureState(target: FeatureIdentifier) {\n this._checkLoaded();\n const sourceId = target.source;\n const sourceLayer = target.sourceLayer;\n const sourceCache = this.sourceCaches[sourceId];\n\n if (sourceCache === undefined) {\n this.fire(new ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`)));\n return;\n }\n const sourceType = sourceCache.getSource().type;\n if (sourceType === 'vector' && !sourceLayer) {\n this.fire(new ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.')));\n return;\n }\n if (target.id === undefined) {\n this.fire(new ErrorEvent(new Error('The feature id parameter must be provided.')));\n }\n\n return sourceCache.getFeatureState(sourceLayer, target.id);\n }\n\n getTransition() {\n return extend({duration: 300, delay: 0}, this.stylesheet && this.stylesheet.transition);\n }\n\n serialize(): StyleSpecification {\n // We return undefined before we're loaded, following the pattern of Map.getStyle() before\n // the Style object is initialized.\n // Internally, Style._validate() calls Style.serialize() but callers are responsible for\n // calling Style._checkLoaded() first if their validation requires the style to be loaded.\n if (!this._loaded) return;\n\n const sources = mapObject(this.sourceCaches, (source) => source.serialize());\n const layers = this._serializeByIds(this._order);\n const terrain = this.map.getTerrain() || undefined;\n const myStyleSheet = this.stylesheet;\n\n return filterObject({\n version: myStyleSheet.version,\n name: myStyleSheet.name,\n metadata: myStyleSheet.metadata,\n light: myStyleSheet.light,\n center: myStyleSheet.center,\n zoom: myStyleSheet.zoom,\n bearing: myStyleSheet.bearing,\n pitch: myStyleSheet.pitch,\n sprite: myStyleSheet.sprite,\n glyphs: myStyleSheet.glyphs,\n transition: myStyleSheet.transition,\n sources,\n layers,\n terrain\n },\n (value) => { return value !== undefined; });\n }\n\n _updateLayer(layer: StyleLayer) {\n this._updatedLayers[layer.id] = true;\n if (layer.source && !this._updatedSources[layer.source] &&\n //Skip for raster layers (https://github.com/mapbox/mapbox-gl-js/issues/7865)\n this.sourceCaches[layer.source].getSource().type !== 'raster') {\n this._updatedSources[layer.source] = 'reload';\n this.sourceCaches[layer.source].pause();\n }\n\n // upon updating, serilized layer dictionary should be reset.\n // When needed, it will be populated with the correct copy again.\n this._serializedLayers = null;\n this._changed = true;\n }\n\n _flattenAndSortRenderedFeatures(sourceResults: Array<{ [key: string]: Array<{featureIndex: number; feature: MapGeoJSONFeature}> }>) {\n // Feature order is complicated.\n // The order between features in two 2D layers is always determined by layer order.\n // The order between features in two 3D layers is always determined by depth.\n // The order between a feature in a 2D layer and a 3D layer is tricky:\n // Most often layer order determines the feature order in this case. If\n // a line layer is above a extrusion layer the line feature will be rendered\n // above the extrusion. If the line layer is below the extrusion layer,\n // it will be rendered below it.\n //\n // There is a weird case though.\n // You have layers in this order: extrusion_layer_a, line_layer, extrusion_layer_b\n // Each layer has a feature that overlaps the other features.\n // The feature in extrusion_layer_a is closer than the feature in extrusion_layer_b so it is rendered above.\n // The feature in line_layer is rendered above extrusion_layer_a.\n // This means that that the line_layer feature is above the extrusion_layer_b feature despite\n // it being in an earlier layer.\n\n const isLayer3D = layerId => this._layers[layerId].type === 'fill-extrusion';\n\n const layerIndex = {};\n const features3D = [];\n for (let l = this._order.length - 1; l >= 0; l--) {\n const layerId = this._order[l];\n if (isLayer3D(layerId)) {\n layerIndex[layerId] = l;\n for (const sourceResult of sourceResults) {\n const layerFeatures = sourceResult[layerId];\n if (layerFeatures) {\n for (const featureWrapper of layerFeatures) {\n features3D.push(featureWrapper);\n }\n }\n }\n }\n }\n\n features3D.sort((a, b) => {\n return b.intersectionZ - a.intersectionZ;\n });\n\n const features = [];\n for (let l = this._order.length - 1; l >= 0; l--) {\n const layerId = this._order[l];\n\n if (isLayer3D(layerId)) {\n // add all 3D features that are in or above the current layer\n for (let i = features3D.length - 1; i >= 0; i--) {\n const topmost3D = features3D[i].feature;\n if (layerIndex[topmost3D.layer.id] < l) break;\n features.push(topmost3D);\n features3D.pop();\n }\n } else {\n for (const sourceResult of sourceResults) {\n const layerFeatures = sourceResult[layerId];\n if (layerFeatures) {\n for (const featureWrapper of layerFeatures) {\n features.push(featureWrapper.feature);\n }\n }\n }\n }\n }\n\n return features;\n }\n\n queryRenderedFeatures(queryGeometry: any, params: QueryRenderedFeaturesOptions, transform: Transform) {\n if (params && params.filter) {\n this._validate(validateStyle.filter, 'queryRenderedFeatures.filter', params.filter, null, params);\n }\n\n const includedSources = {};\n if (params && params.layers) {\n if (!Array.isArray(params.layers)) {\n this.fire(new ErrorEvent(new Error('parameters.layers must be an Array.')));\n return [];\n }\n for (const layerId of params.layers) {\n const layer = this._layers[layerId];\n if (!layer) {\n // this layer is not in the style.layers array\n this.fire(new ErrorEvent(new Error(`The layer '${layerId}' does not exist in the map's style and cannot be queried for features.`)));\n return [];\n }\n includedSources[layer.source] = true;\n }\n }\n\n const sourceResults = [];\n\n params.availableImages = this._availableImages;\n\n // LayerSpecification is serialized StyleLayer, and this casting is safe.\n const serializedLayers = this._serializedAllLayers() as {[_: string]: StyleLayer};\n\n for (const id in this.sourceCaches) {\n if (params.layers && !includedSources[id]) continue;\n sourceResults.push(\n queryRenderedFeatures(\n this.sourceCaches[id],\n this._layers,\n serializedLayers,\n queryGeometry,\n params,\n transform)\n );\n }\n\n if (this.placement) {\n // If a placement has run, query against its CollisionIndex\n // for symbol results, and treat it as an extra source to merge\n sourceResults.push(\n queryRenderedSymbols(\n this._layers,\n serializedLayers,\n this.sourceCaches,\n queryGeometry,\n params,\n this.placement.collisionIndex,\n this.placement.retainedQueryData)\n );\n }\n\n return this._flattenAndSortRenderedFeatures(sourceResults);\n }\n\n querySourceFeatures(\n sourceID: string,\n params?: QuerySourceFeatureOptions\n ) {\n if (params && params.filter) {\n this._validate(validateStyle.filter, 'querySourceFeatures.filter', params.filter, null, params);\n }\n const sourceCache = this.sourceCaches[sourceID];\n return sourceCache ? querySourceFeatures(sourceCache, params) : [];\n }\n\n addSourceType(name: string, SourceType: SourceClass, callback: Callback) {\n if (getSourceType(name)) {\n return callback(new Error(`A source type called \"${name}\" already exists.`));\n }\n\n setSourceType(name, SourceType);\n\n if (!SourceType.workerSourceURL) {\n return callback(null, null);\n }\n\n this.dispatcher.broadcast('loadWorkerSource', {\n name,\n url: SourceType.workerSourceURL\n }, callback);\n }\n\n getLight() {\n return this.light.getLight();\n }\n\n setLight(lightOptions: LightSpecification, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n const light = this.light.getLight();\n let _update = false;\n for (const key in lightOptions) {\n if (!deepEqual(lightOptions[key], light[key])) {\n _update = true;\n break;\n }\n }\n if (!_update) return;\n\n const parameters = {\n now: browser.now(),\n transition: extend({\n duration: 300,\n delay: 0\n }, this.stylesheet.transition)\n };\n\n this.light.setLight(lightOptions, options);\n this.light.updateTransitions(parameters);\n }\n\n _validate(validate: Validator, key: string, value: any, props: any, options: {\n validate?: boolean;\n } = {}) {\n if (options && options.validate === false) {\n return false;\n }\n return emitValidationErrors(this, validate.call(validateStyle, extend({\n key,\n style: this.serialize(),\n value,\n styleSpec\n }, props)));\n }\n\n _remove(mapRemoved: boolean = true) {\n if (this._request) {\n this._request.cancel();\n this._request = null;\n }\n if (this._spriteRequest) {\n this._spriteRequest.cancel();\n this._spriteRequest = null;\n }\n rtlTextPluginEvented.off('pluginStateChange', this._rtlTextPluginCallback);\n for (const layerId in this._layers) {\n const layer: StyleLayer = this._layers[layerId];\n layer.setEventedParent(null);\n }\n for (const id in this.sourceCaches) {\n const sourceCache = this.sourceCaches[id];\n sourceCache.setEventedParent(null);\n sourceCache.onRemove(this.map);\n }\n this.imageManager.setEventedParent(null);\n this.setEventedParent(null);\n this.dispatcher.remove(mapRemoved);\n }\n\n _clearSource(id: string) {\n this.sourceCaches[id].clearTiles();\n }\n\n _reloadSource(id: string) {\n this.sourceCaches[id].resume();\n this.sourceCaches[id].reload();\n }\n\n _updateSources(transform: Transform) {\n for (const id in this.sourceCaches) {\n this.sourceCaches[id].update(transform, this.map.terrain);\n }\n }\n\n _generateCollisionBoxes() {\n for (const id in this.sourceCaches) {\n this._reloadSource(id);\n }\n }\n\n _updatePlacement(transform: Transform, showCollisionBoxes: boolean, fadeDuration: number, crossSourceCollisions: boolean, forceFullPlacement: boolean = false) {\n let symbolBucketsChanged = false;\n let placementCommitted = false;\n\n const layerTiles = {};\n\n for (const layerID of this._order) {\n const styleLayer = this._layers[layerID];\n if (styleLayer.type !== 'symbol') continue;\n\n if (!layerTiles[styleLayer.source]) {\n const sourceCache = this.sourceCaches[styleLayer.source];\n layerTiles[styleLayer.source] = sourceCache.getRenderableIds(true)\n .map((id) => sourceCache.getTileByID(id))\n .sort((a, b) => (b.tileID.overscaledZ - a.tileID.overscaledZ) || (a.tileID.isLessThan(b.tileID) ? -1 : 1));\n }\n\n const layerBucketsChanged = this.crossTileSymbolIndex.addLayer(styleLayer, layerTiles[styleLayer.source], transform.center.lng);\n symbolBucketsChanged = symbolBucketsChanged || layerBucketsChanged;\n }\n this.crossTileSymbolIndex.pruneUnusedLayers(this._order);\n\n // Anything that changes our \"in progress\" layer and tile indices requires us\n // to start over. When we start over, we do a full placement instead of incremental\n // to prevent starvation.\n // We need to restart placement to keep layer indices in sync.\n // Also force full placement when fadeDuration === 0 to ensure that newly loaded\n // tiles will fully display symbols in their first frame\n forceFullPlacement = forceFullPlacement || this._layerOrderChanged || fadeDuration === 0;\n\n if (forceFullPlacement || !this.pauseablePlacement || (this.pauseablePlacement.isDone() && !this.placement.stillRecent(browser.now(), transform.zoom))) {\n this.pauseablePlacement = new PauseablePlacement(transform, this.map.terrain, this._order, forceFullPlacement, showCollisionBoxes, fadeDuration, crossSourceCollisions, this.placement);\n this._layerOrderChanged = false;\n }\n\n if (this.pauseablePlacement.isDone()) {\n // the last placement finished running, but the next one hasn’t\n // started yet because of the `stillRecent` check immediately\n // above, so mark it stale to ensure that we request another\n // render frame\n this.placement.setStale();\n } else {\n this.pauseablePlacement.continuePlacement(this._order, this._layers, layerTiles);\n\n if (this.pauseablePlacement.isDone()) {\n this.placement = this.pauseablePlacement.commit(browser.now());\n placementCommitted = true;\n }\n\n if (symbolBucketsChanged) {\n // since the placement gets split over multiple frames it is possible\n // these buckets were processed before they were changed and so the\n // placement is already stale while it is in progress\n this.pauseablePlacement.placement.setStale();\n }\n }\n\n if (placementCommitted || symbolBucketsChanged) {\n for (const layerID of this._order) {\n const styleLayer = this._layers[layerID];\n if (styleLayer.type !== 'symbol') continue;\n this.placement.updateLayerOpacities(styleLayer, layerTiles[styleLayer.source]);\n }\n }\n\n // needsRender is false when we have just finished a placement that didn't change the visibility of any symbols\n const needsRerender = !this.pauseablePlacement.isDone() || this.placement.hasTransitions(browser.now());\n return needsRerender;\n }\n\n _releaseSymbolFadeTiles() {\n for (const id in this.sourceCaches) {\n this.sourceCaches[id].releaseSymbolFadeTiles();\n }\n }\n\n // Callbacks from web workers\n\n getImages(\n mapId: string,\n params: {\n icons: Array;\n source: string;\n tileID: OverscaledTileID;\n type: string;\n },\n callback: Callback<{[_: string]: StyleImage}>\n ) {\n this.imageManager.getImages(params.icons, callback);\n\n // Apply queued image changes before setting the tile's dependencies so that the tile\n // is not reloaded unnecessarily. Without this forced update the reload could happen in cases\n // like this one:\n // - icons contains \"my-image\"\n // - imageManager.getImages(...) triggers `onstyleimagemissing`\n // - the user adds \"my-image\" within the callback\n // - addImage adds \"my-image\" to this._changedImages\n // - the next frame triggers a reload of this tile even though it already has the latest version\n this._updateTilesForChangedImages();\n\n const sourceCache = this.sourceCaches[params.source];\n if (sourceCache) {\n sourceCache.setDependencies(params.tileID.key, params.type, params.icons);\n }\n }\n\n getGlyphs(\n mapId: string,\n params: {\n stacks: {[_: string]: Array};\n source: string;\n tileID: OverscaledTileID;\n type: string;\n },\n callback: Callback<{[_: string]: {[_: number]: StyleGlyph}}>\n ) {\n this.glyphManager.getGlyphs(params.stacks, callback);\n const sourceCache = this.sourceCaches[params.source];\n if (sourceCache) {\n // we are not setting stacks as dependencies since for now\n // we just need to know which tiles have glyph dependencies\n sourceCache.setDependencies(params.tileID.key, params.type, ['']);\n }\n }\n\n getResource(mapId: string, params: RequestParameters, callback: ResponseCallback): Cancelable {\n return makeRequest(params, callback);\n }\n\n getGlyphsUrl() {\n return this.stylesheet.glyphs || null;\n }\n\n setGlyphs(glyphsUrl: string | null, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n if (glyphsUrl && this._validate(validateStyle.glyphs, 'glyphs', glyphsUrl, null, options)) {\n return;\n }\n\n this._glyphsDidChange = true;\n this.stylesheet.glyphs = glyphsUrl;\n this.glyphManager.entries = {};\n this.glyphManager.setURL(glyphsUrl);\n }\n\n /**\n * Add a sprite.\n *\n * @param id - The id of the desired sprite\n * @param url - The url to load the desired sprite from\n * @param options - The style setter options\n * @param completion - The completion handler\n */\n addSprite(id: string, url: string, options: StyleSetterOptions = {}, completion?: (err: Error) => void) {\n this._checkLoaded();\n\n const spriteToAdd = [{id, url}];\n const updatedSprite = [\n ...coerceSpriteToArray(this.stylesheet.sprite),\n ...spriteToAdd\n ];\n\n if (this._validate(validateStyle.sprite, 'sprite', updatedSprite, null, options)) return;\n\n this.stylesheet.sprite = updatedSprite;\n this._loadSprite(spriteToAdd, true, completion);\n }\n\n /**\n * Remove a sprite by its id. When the last sprite is removed, the whole `this.stylesheet.sprite` object becomes\n * `undefined`. This falsy `undefined` value later prevents attempts to load the sprite when it's absent.\n *\n * @param id - the id of the sprite to remove\n */\n removeSprite(id: string) {\n this._checkLoaded();\n\n const internalSpriteRepresentation = coerceSpriteToArray(this.stylesheet.sprite);\n\n if (!internalSpriteRepresentation.find(sprite => sprite.id === id)) {\n this.fire(new ErrorEvent(new Error(`Sprite \"${id}\" doesn't exists on this map.`)));\n return;\n }\n\n if (this._spritesImagesIds[id]) {\n for (const imageId of this._spritesImagesIds[id]) {\n this.imageManager.removeImage(imageId);\n this._changedImages[imageId] = true;\n }\n }\n\n internalSpriteRepresentation.splice(internalSpriteRepresentation.findIndex(sprite => sprite.id === id), 1);\n this.stylesheet.sprite = internalSpriteRepresentation.length > 0 ? internalSpriteRepresentation : undefined;\n\n delete this._spritesImagesIds[id];\n this._availableImages = this.imageManager.listImages();\n this._changed = true;\n this.dispatcher.broadcast('setImages', this._availableImages);\n this.fire(new Event('data', {dataType: 'style'}));\n }\n\n /**\n * Get the current sprite value.\n *\n * @returns empty array when no sprite is set; id-url pairs otherwise\n */\n getSprite() {\n return coerceSpriteToArray(this.stylesheet.sprite);\n }\n\n /**\n * Set a new value for the style's sprite.\n *\n * @param sprite - new sprite value\n * @param options - style setter options\n * @param completion - the completion handler\n */\n setSprite(sprite: SpriteSpecification, options: StyleSetterOptions = {}, completion?: (err: Error) => void) {\n this._checkLoaded();\n\n if (sprite && this._validate(validateStyle.sprite, 'sprite', sprite, null, options)) {\n return;\n }\n\n this.stylesheet.sprite = sprite;\n\n if (sprite) {\n this._loadSprite(sprite, true, completion);\n } else {\n this._unloadSprite();\n if (completion) {\n completion(null);\n }\n }\n }\n}\n\nStyle.registerForPluginStateChange = registerForPluginStateChange;\n","import {createLayout} from '../util/struct_array';\n\nexport default createLayout([\n {name: 'a_pos', type: 'Int16', components: 2}\n]);\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision mediump float;\\n#else\\n#if !defined(lowp)\\n#define lowp\\n#endif\\n#if !defined(mediump)\\n#define mediump\\n#endif\\n#if !defined(highp)\\n#define highp\\n#endif\\n#endif\\n';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision highp float;\\n#else\\n#if !defined(lowp)\\n#define lowp\\n#endif\\n#if !defined(mediump)\\n#define mediump\\n#endif\\n#if !defined(highp)\\n#define highp\\n#endif\\n#endif\\nvec2 unpack_float(const float packedValue) {int packedIntValue=int(packedValue);int v0=packedIntValue/256;return vec2(v0,packedIntValue-v0*256);}vec2 unpack_opacity(const float packedOpacity) {int intOpacity=int(packedOpacity)/2;return vec2(float(intOpacity)/127.0,mod(packedOpacity,2.0));}vec4 decode_color(const vec2 encodedColor) {return vec4(unpack_float(encodedColor[0])/255.0,unpack_float(encodedColor[1])/255.0\\n);}float unpack_mix_vec2(const vec2 packedValue,const float t) {return mix(packedValue[0],packedValue[1],t);}vec4 unpack_mix_color(const vec4 packedColors,const float t) {vec4 minColor=decode_color(vec2(packedColors[0],packedColors[1]));vec4 maxColor=decode_color(vec2(packedColors[2],packedColors[3]));return mix(minColor,maxColor,t);}vec2 get_pattern_pos(const vec2 pixel_coord_upper,const vec2 pixel_coord_lower,const vec2 pattern_size,const float tile_units_to_pixels,const vec2 pos) {vec2 offset=mod(mod(mod(pixel_coord_upper,pattern_size)*256.0,pattern_size)*256.0+pixel_coord_lower,pattern_size);return (tile_units_to_pixels*pos+offset)/pattern_size;}\\n#ifdef TERRAIN3D\\nuniform sampler2D u_terrain;uniform float u_terrain_dim;uniform mat4 u_terrain_matrix;uniform vec4 u_terrain_unpack;uniform float u_terrain_exaggeration;uniform highp sampler2D u_depth;\\n#endif\\nconst highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitShifts=vec4(1.)/bitSh;highp float unpack(highp vec4 color) {return dot(color,bitShifts);}highp float depthOpacity(vec3 frag) {\\n#ifdef TERRAIN3D\\nhighp float d=unpack(texture2D(u_depth,frag.xy*0.5+0.5))+0.0001-frag.z;return 1.0-max(0.0,min(1.0,-d*500.0));\\n#else\\nreturn 1.0;\\n#endif\\n}float calculate_visibility(vec4 pos) {\\n#ifdef TERRAIN3D\\nvec3 frag=pos.xyz/pos.w;highp float d=depthOpacity(frag);if (d > 0.95) return 1.0;return (d+depthOpacity(frag+vec3(0.0,0.01,0.0)))/2.0;\\n#else\\nreturn 1.0;\\n#endif\\n}float ele(vec2 pos) {\\n#ifdef TERRAIN3D\\nvec4 rgb=(texture2D(u_terrain,pos)*255.0)*u_terrain_unpack;return rgb.r+rgb.g+rgb.b-u_terrain_unpack.a;\\n#else\\nreturn 0.0;\\n#endif\\n}float get_elevation(vec2 pos) {\\n#ifdef TERRAIN3D\\nvec2 coord=(u_terrain_matrix*vec4(pos,0.0,1.0)).xy*u_terrain_dim+1.0;vec2 f=fract(coord);vec2 c=(floor(coord)+0.5)/(u_terrain_dim+2.0);float d=1.0/(u_terrain_dim+2.0);float tl=ele(c);float tr=ele(c+vec2(d,0.0));float bl=ele(c+vec2(0.0,d));float br=ele(c+vec2(d,d));float elevation=mix(mix(tl,tr,f.x),mix(bl,br,f.x),f.y);return elevation*u_terrain_exaggeration;\\n#else\\nreturn 0.0;\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform vec4 u_color;uniform float u_opacity;void main() {gl_FragColor=u_color*u_opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform vec2 u_pattern_tl_a;uniform vec2 u_pattern_br_a;uniform vec2 u_pattern_tl_b;uniform vec2 u_pattern_br_b;uniform vec2 u_texsize;uniform float u_mix;uniform float u_opacity;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(u_pattern_tl_a/u_texsize,u_pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(u_pattern_tl_b/u_texsize,u_pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_mix)*u_opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_pattern_size_a;uniform vec2 u_pattern_size_b;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_scale_a;uniform float u_scale_b;uniform float u_tile_units_to_pixels;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_a*u_pattern_size_a,u_tile_units_to_pixels,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_b*u_pattern_size_b,u_tile_units_to_pixels,a_pos);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying vec3 v_data;varying float v_visibility;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define mediump float radius\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define highp vec4 stroke_color\\n#pragma mapbox: define mediump float stroke_width\\n#pragma mapbox: define lowp float stroke_opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize mediump float radius\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize highp vec4 stroke_color\\n#pragma mapbox: initialize mediump float stroke_width\\n#pragma mapbox: initialize lowp float stroke_opacity\\nvec2 extrude=v_data.xy;float extrude_length=length(extrude);lowp float antialiasblur=v_data.z;float antialiased_blur=-max(blur,antialiasblur);float opacity_t=smoothstep(0.0,antialiased_blur,extrude_length-1.0);float color_t=stroke_width < 0.01 ? 0.0 : smoothstep(antialiased_blur,0.0,extrude_length-radius/(radius+stroke_width));gl_FragColor=v_visibility*opacity_t*mix(color*opacity,stroke_color*stroke_opacity,color_t);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform bool u_scale_with_map;uniform bool u_pitch_with_map;uniform vec2 u_extrude_scale;uniform lowp float u_device_pixel_ratio;uniform highp float u_camera_to_center_distance;attribute vec2 a_pos;varying vec3 v_data;varying float v_visibility;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define mediump float radius\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define highp vec4 stroke_color\\n#pragma mapbox: define mediump float stroke_width\\n#pragma mapbox: define lowp float stroke_opacity\\nvoid main(void) {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize mediump float radius\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize highp vec4 stroke_color\\n#pragma mapbox: initialize mediump float stroke_width\\n#pragma mapbox: initialize lowp float stroke_opacity\\nvec2 extrude=vec2(mod(a_pos,2.0)*2.0-1.0);vec2 circle_center=floor(a_pos*0.5);float ele=get_elevation(circle_center);v_visibility=calculate_visibility(u_matrix*vec4(circle_center,ele,1.0));if (u_pitch_with_map) {vec2 corner_position=circle_center;if (u_scale_with_map) {corner_position+=extrude*(radius+stroke_width)*u_extrude_scale;} else {vec4 projected_center=u_matrix*vec4(circle_center,0,1);corner_position+=extrude*(radius+stroke_width)*u_extrude_scale*(projected_center.w/u_camera_to_center_distance);}gl_Position=u_matrix*vec4(corner_position,ele,1);} else {gl_Position=u_matrix*vec4(circle_center,ele,1);if (u_scale_with_map) {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*u_camera_to_center_distance;} else {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*gl_Position.w;}}lowp float antialiasblur=1.0/u_device_pixel_ratio/(radius+stroke_width);v_data=vec3(extrude.x,extrude.y,antialiasblur);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'void main() {gl_FragColor=vec4(1.0);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform highp float u_intensity;varying vec2 v_extrude;\\n#pragma mapbox: define highp float weight\\n#define GAUSS_COEF 0.3989422804014327\\nvoid main() {\\n#pragma mapbox: initialize highp float weight\\nfloat d=-0.5*3.0*3.0*dot(v_extrude,v_extrude);float val=weight*u_intensity*GAUSS_COEF*exp(d);gl_FragColor=vec4(val,1.0,1.0,1.0);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform float u_extrude_scale;uniform float u_opacity;uniform float u_intensity;attribute vec2 a_pos;varying vec2 v_extrude;\\n#pragma mapbox: define highp float weight\\n#pragma mapbox: define mediump float radius\\nconst highp float ZERO=1.0/255.0/16.0;\\n#define GAUSS_COEF 0.3989422804014327\\nvoid main(void) {\\n#pragma mapbox: initialize highp float weight\\n#pragma mapbox: initialize mediump float radius\\nvec2 unscaled_extrude=vec2(mod(a_pos,2.0)*2.0-1.0);float S=sqrt(-2.0*log(ZERO/weight/u_intensity/GAUSS_COEF))/3.0;v_extrude=S*unscaled_extrude;vec2 extrude=v_extrude*radius*u_extrude_scale;vec4 pos=vec4(floor(a_pos*0.5)+extrude,0,1);gl_Position=u_matrix*pos;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform sampler2D u_image;uniform sampler2D u_color_ramp;uniform float u_opacity;varying vec2 v_pos;void main() {float t=texture2D(u_image,v_pos).r;vec4 color=texture2D(u_color_ramp,vec2(t,0.5));gl_FragColor=color*u_opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(0.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_world;attribute vec2 a_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos*u_world,0,1);v_pos.x=a_pos.x;v_pos.y=1.0-a_pos.y;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying float v_placed;varying float v_notUsed;void main() {float alpha=0.5;gl_FragColor=vec4(1.0,0.0,0.0,1.0)*alpha;if (v_placed > 0.5) {gl_FragColor=vec4(0.0,0.0,1.0,0.5)*alpha;}if (v_notUsed > 0.5) {gl_FragColor*=.1;}}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;attribute vec2 a_anchor_pos;attribute vec2 a_extrude;attribute vec2 a_placed;attribute vec2 a_shift;uniform mat4 u_matrix;uniform vec2 u_extrude_scale;uniform float u_camera_to_center_distance;varying float v_placed;varying float v_notUsed;void main() {vec4 projectedPoint=u_matrix*vec4(a_anchor_pos,0,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);gl_Position=u_matrix*vec4(a_pos,get_elevation(a_pos),1.0);gl_Position.xy+=(a_extrude+a_shift)*u_extrude_scale*gl_Position.w*collision_perspective_ratio;v_placed=a_placed.x;v_notUsed=a_placed.y;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;void main() {float alpha=0.5*min(v_perspective_ratio,1.0);float stroke_radius=0.9*max(v_perspective_ratio,1.0);float distance_to_center=length(v_extrude);float distance_to_edge=abs(distance_to_center-v_radius);float opacity_t=smoothstep(-stroke_radius,0.0,-distance_to_edge);vec4 color=mix(vec4(0.0,0.0,1.0,0.5),vec4(1.0,0.0,0.0,1.0),v_collision);gl_FragColor=color*alpha*opacity_t;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;attribute float a_radius;attribute vec2 a_flags;uniform mat4 u_matrix;uniform mat4 u_inv_matrix;uniform vec2 u_viewport_size;uniform float u_camera_to_center_distance;varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;vec3 toTilePosition(vec2 screenPos) {vec4 rayStart=u_inv_matrix*vec4(screenPos,-1.0,1.0);vec4 rayEnd =u_inv_matrix*vec4(screenPos, 1.0,1.0);rayStart.xyz/=rayStart.w;rayEnd.xyz /=rayEnd.w;highp float t=(0.0-rayStart.z)/(rayEnd.z-rayStart.z);return mix(rayStart.xyz,rayEnd.xyz,t);}void main() {vec2 quadCenterPos=a_pos;float radius=a_radius;float collision=a_flags.x;float vertexIdx=a_flags.y;vec2 quadVertexOffset=vec2(mix(-1.0,1.0,float(vertexIdx >=2.0)),mix(-1.0,1.0,float(vertexIdx >=1.0 && vertexIdx <=2.0)));vec2 quadVertexExtent=quadVertexOffset*radius;vec3 tilePos=toTilePosition(quadCenterPos);vec4 clipPos=u_matrix*vec4(tilePos,1.0);highp float camera_to_anchor_distance=clipPos.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);float padding_factor=1.2;v_radius=radius;v_extrude=quadVertexExtent*padding_factor;v_perspective_ratio=collision_perspective_ratio;v_collision=collision;gl_Position=vec4(clipPos.xyz/clipPos.w,1.0)+vec4(quadVertexExtent*padding_factor/u_viewport_size*2.0,0.0,0.0);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform highp vec4 u_color;uniform sampler2D u_overlay;varying vec2 v_uv;void main() {vec4 overlay_color=texture2D(u_overlay,v_uv);gl_FragColor=mix(u_color,overlay_color,overlay_color.a);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;varying vec2 v_uv;uniform mat4 u_matrix;uniform float u_overlay_scale;void main() {v_uv=a_pos/8192.0;gl_Position=u_matrix*vec4(a_pos*u_overlay_scale,get_elevation(a_pos),1);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float opacity\\ngl_FragColor=color*opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;uniform mat4 u_matrix;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float opacity\\ngl_Position=u_matrix*vec4(a_pos,0,1);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying vec2 v_pos;\\n#pragma mapbox: define highp vec4 outline_color\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 outline_color\\n#pragma mapbox: initialize lowp float opacity\\nfloat dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=outline_color*(alpha*opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;uniform mat4 u_matrix;uniform vec2 u_world;varying vec2 v_pos;\\n#pragma mapbox: define highp vec4 outline_color\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 outline_color\\n#pragma mapbox: initialize lowp float opacity\\ngl_Position=u_matrix*vec4(a_pos,0,1);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform vec2 u_texsize;uniform sampler2D u_image;uniform float u_fade;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);float dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=mix(color1,color2,u_fade)*alpha*opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_world;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;gl_Position=u_matrix*vec4(a_pos,0,1);vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,a_pos);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision highp float;\\n#endif\\nuniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_fade)*opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileZoomRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileZoomRatio,a_pos);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying vec4 v_color;void main() {gl_FragColor=v_color;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;uniform float u_vertical_gradient;uniform lowp float u_opacity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\\n#ifdef TERRAIN3D\\nattribute vec2 a_centroid;\\n#endif\\nvarying vec4 v_color;\\n#pragma mapbox: define highp float base\\n#pragma mapbox: define highp float height\\n#pragma mapbox: define highp vec4 color\\nvoid main() {\\n#pragma mapbox: initialize highp float base\\n#pragma mapbox: initialize highp float height\\n#pragma mapbox: initialize highp vec4 color\\nvec3 normal=a_normal_ed.xyz;\\n#ifdef TERRAIN3D\\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\\n#else\\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\\n#endif\\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);gl_Position=u_matrix*vec4(a_pos,t > 0.0 ? height : base,1);float colorvalue=color.r*0.2126+color.g*0.7152+color.b*0.0722;v_color=vec4(0.0,0.0,0.0,1.0);vec4 ambientlight=vec4(0.03,0.03,0.03,1.0);color+=ambientlight;float directional=clamp(dot(normal/16384.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((1.0-colorvalue+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_color.r+=clamp(color.r*directional*u_lightcolor.r,mix(0.0,0.3,1.0-u_lightcolor.r),1.0);v_color.g+=clamp(color.g*directional*u_lightcolor.g,mix(0.0,0.3,1.0-u_lightcolor.g),1.0);v_color.b+=clamp(color.b*directional*u_lightcolor.b,mix(0.0,0.3,1.0-u_lightcolor.b),1.0);v_color*=u_opacity;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\\n#pragma mapbox: define lowp float base\\n#pragma mapbox: define lowp float height\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float base\\n#pragma mapbox: initialize lowp float height\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);vec4 mixedColor=mix(color1,color2,u_fade);gl_FragColor=mixedColor*v_lighting;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_height_factor;uniform vec3 u_scale;uniform float u_vertical_gradient;uniform lowp float u_opacity;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\\n#ifdef TERRAIN3D\\nattribute vec2 a_centroid;\\n#endif\\nvarying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\\n#pragma mapbox: define lowp float base\\n#pragma mapbox: define lowp float height\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float base\\n#pragma mapbox: initialize lowp float height\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec3 normal=a_normal_ed.xyz;float edgedistance=a_normal_ed.w;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;\\n#ifdef TERRAIN3D\\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\\n#else\\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\\n#endif\\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);float z=t > 0.0 ? height : base;gl_Position=u_matrix*vec4(a_pos,z,1);vec2 pos=normal.x==1.0 && normal.y==0.0 && normal.z==16384.0\\n? a_pos\\n: vec2(edgedistance,z*u_height_factor);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,pos);v_lighting=vec4(0.0,0.0,0.0,1.0);float directional=clamp(dot(normal/16383.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((0.5+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_lighting.rgb+=clamp(directional*u_lightcolor,mix(vec3(0.0),vec3(0.3),1.0-u_lightcolor),vec3(1.0));v_lighting*=u_opacity;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision highp float;\\n#endif\\nuniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_dimension;uniform float u_zoom;uniform vec4 u_unpack;float getElevation(vec2 coord,float bias) {vec4 data=texture2D(u_image,coord)*255.0;data.a=-1.0;return dot(data,u_unpack)/4.0;}void main() {vec2 epsilon=1.0/u_dimension;float a=getElevation(v_pos+vec2(-epsilon.x,-epsilon.y),0.0);float b=getElevation(v_pos+vec2(0,-epsilon.y),0.0);float c=getElevation(v_pos+vec2(epsilon.x,-epsilon.y),0.0);float d=getElevation(v_pos+vec2(-epsilon.x,0),0.0);float e=getElevation(v_pos,0.0);float f=getElevation(v_pos+vec2(epsilon.x,0),0.0);float g=getElevation(v_pos+vec2(-epsilon.x,epsilon.y),0.0);float h=getElevation(v_pos+vec2(0,epsilon.y),0.0);float i=getElevation(v_pos+vec2(epsilon.x,epsilon.y),0.0);float exaggerationFactor=u_zoom < 2.0 ? 0.4 : u_zoom < 4.5 ? 0.35 : 0.3;float exaggeration=u_zoom < 15.0 ? (u_zoom-15.0)*exaggerationFactor : 0.0;vec2 deriv=vec2((c+f+f+i)-(a+d+d+g),(g+h+h+i)-(a+b+b+c))/pow(2.0,exaggeration+(19.2562-u_zoom));gl_FragColor=clamp(vec4(deriv.x/2.0+0.5,deriv.y/2.0+0.5,1.0,1.0),0.0,1.0);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_dimension;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);highp vec2 epsilon=1.0/u_dimension;float scale=(u_dimension.x-2.0)/u_dimension.x;v_pos=(a_texture_pos/8192.0)*scale+epsilon;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_latrange;uniform vec2 u_light;uniform vec4 u_shadow;uniform vec4 u_highlight;uniform vec4 u_accent;\\n#define PI 3.141592653589793\\nvoid main() {vec4 pixel=texture2D(u_image,v_pos);vec2 deriv=((pixel.rg*2.0)-1.0);float scaleFactor=cos(radians((u_latrange[0]-u_latrange[1])*(1.0-v_pos.y)+u_latrange[1]));float slope=atan(1.25*length(deriv)/scaleFactor);float aspect=deriv.x !=0.0 ? atan(deriv.y,-deriv.x) : PI/2.0*(deriv.y > 0.0 ? 1.0 :-1.0);float intensity=u_light.x;float azimuth=u_light.y+PI;float base=1.875-intensity*1.75;float maxValue=0.5*PI;float scaledSlope=intensity !=0.5 ? ((pow(base,slope)-1.0)/(pow(base,maxValue)-1.0))*maxValue : slope;float accent=cos(scaledSlope);vec4 accent_color=(1.0-accent)*u_accent*clamp(intensity*2.0,0.0,1.0);float shade=abs(mod((aspect+azimuth)/PI+0.5,2.0)-1.0);vec4 shade_color=mix(u_shadow,u_highlight,shade)*sin(scaledSlope)*clamp(intensity*2.0,0.0,1.0);gl_FragColor=accent_color*(1.0-shade_color.a)+shade_color;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos=a_texture_pos/8192.0;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform lowp float u_device_pixel_ratio;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);gl_FragColor=color*(alpha*opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '\\n#define scale 0.015873016\\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform vec2 u_units_to_pixels;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp float v_linesofar;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define mediump float gapwidth\\n#pragma mapbox: define lowp float offset\\n#pragma mapbox: define mediump float width\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump float gapwidth\\n#pragma mapbox: initialize lowp float offset\\n#pragma mapbox: initialize mediump float width\\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;v_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*2.0;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\\n#ifdef TERRAIN3D\\nv_gamma_scale=1.0;\\n#else\\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\\n#endif\\nv_width2=vec2(outset,inset);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;varying highp vec2 v_uv;\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);vec4 color=texture2D(u_image,v_uv);gl_FragColor=color*(alpha*opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '\\n#define scale 0.015873016\\nattribute vec2 a_pos_normal;attribute vec4 a_data;attribute float a_uv_x;attribute float a_split_index;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_units_to_pixels;uniform float u_image_height;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp vec2 v_uv;\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define mediump float gapwidth\\n#pragma mapbox: define lowp float offset\\n#pragma mapbox: define mediump float width\\nvoid main() {\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump float gapwidth\\n#pragma mapbox: initialize lowp float offset\\n#pragma mapbox: initialize mediump float width\\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;highp float texel_height=1.0/u_image_height;highp float half_texel_height=0.5*texel_height;v_uv=vec2(a_uv_x,a_split_index*texel_height-half_texel_height);vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\\n#ifdef TERRAIN3D\\nv_gamma_scale=1.0;\\n#else\\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\\n#endif\\nv_width2=vec2(outset,inset);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision highp float;\\n#endif\\nuniform lowp float u_device_pixel_ratio;uniform vec2 u_texsize;uniform float u_fade;uniform mediump vec3 u_scale;uniform sampler2D u_image;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;vec2 pattern_size_a=vec2(display_size_a.x*fromScale/tileZoomRatio,display_size_a.y);vec2 pattern_size_b=vec2(display_size_b.x*toScale/tileZoomRatio,display_size_b.y);float aspect_a=display_size_a.y/v_width;float aspect_b=display_size_b.y/v_width;float dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float x_a=mod(v_linesofar/pattern_size_a.x*aspect_a,1.0);float x_b=mod(v_linesofar/pattern_size_b.x*aspect_b,1.0);float y=0.5*v_normal.y+0.5;vec2 texel_size=1.0/u_texsize;vec2 pos_a=mix(pattern_tl_a*texel_size-texel_size,pattern_br_a*texel_size+texel_size,vec2(x_a,y));vec2 pos_b=mix(pattern_tl_b*texel_size-texel_size,pattern_br_b*texel_size+texel_size,vec2(x_b,y));vec4 color=mix(texture2D(u_image,pos_a),texture2D(u_image,pos_b),u_fade);gl_FragColor=color*alpha*opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '\\n#define scale 0.015873016\\n#define LINE_DISTANCE_SCALE 2.0\\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform vec2 u_units_to_pixels;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float offset\\n#pragma mapbox: define mediump float gapwidth\\n#pragma mapbox: define mediump float width\\n#pragma mapbox: define lowp float floorwidth\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float offset\\n#pragma mapbox: initialize mediump float gapwidth\\n#pragma mapbox: initialize mediump float width\\n#pragma mapbox: initialize lowp float floorwidth\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\\n#ifdef TERRAIN3D\\nv_gamma_scale=1.0;\\n#else\\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\\n#endif\\nv_linesofar=a_linesofar;v_width2=vec2(outset,inset);v_width=floorwidth;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;uniform float u_sdfgamma;uniform float u_mix;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define mediump float width\\n#pragma mapbox: define lowp float floorwidth\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump float width\\n#pragma mapbox: initialize lowp float floorwidth\\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float sdfdist_a=texture2D(u_image,v_tex_a).a;float sdfdist_b=texture2D(u_image,v_tex_b).a;float sdfdist=mix(sdfdist_a,sdfdist_b,u_mix);alpha*=smoothstep(0.5-u_sdfgamma/floorwidth,0.5+u_sdfgamma/floorwidth,sdfdist);gl_FragColor=color*(alpha*opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '\\n#define scale 0.015873016\\n#define LINE_DISTANCE_SCALE 2.0\\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_patternscale_a;uniform float u_tex_y_a;uniform vec2 u_patternscale_b;uniform float u_tex_y_b;uniform vec2 u_units_to_pixels;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define mediump float gapwidth\\n#pragma mapbox: define lowp float offset\\n#pragma mapbox: define mediump float width\\n#pragma mapbox: define lowp float floorwidth\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump float gapwidth\\n#pragma mapbox: initialize lowp float offset\\n#pragma mapbox: initialize mediump float width\\n#pragma mapbox: initialize lowp float floorwidth\\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\\n#ifdef TERRAIN3D\\nv_gamma_scale=1.0;\\n#else\\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\\n#endif\\nv_tex_a=vec2(a_linesofar*u_patternscale_a.x/floorwidth,normal.y*u_patternscale_a.y+u_tex_y_a);v_tex_b=vec2(a_linesofar*u_patternscale_b.x/floorwidth,normal.y*u_patternscale_b.y+u_tex_y_b);v_width2=vec2(outset,inset);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform float u_fade_t;uniform float u_opacity;uniform sampler2D u_image0;uniform sampler2D u_image1;varying vec2 v_pos0;varying vec2 v_pos1;uniform float u_brightness_low;uniform float u_brightness_high;uniform float u_saturation_factor;uniform float u_contrast_factor;uniform vec3 u_spin_weights;void main() {vec4 color0=texture2D(u_image0,v_pos0);vec4 color1=texture2D(u_image1,v_pos1);if (color0.a > 0.0) {color0.rgb=color0.rgb/color0.a;}if (color1.a > 0.0) {color1.rgb=color1.rgb/color1.a;}vec4 color=mix(color0,color1,u_fade_t);color.a*=u_opacity;vec3 rgb=color.rgb;rgb=vec3(dot(rgb,u_spin_weights.xyz),dot(rgb,u_spin_weights.zxy),dot(rgb,u_spin_weights.yzx));float average=(color.r+color.g+color.b)/3.0;rgb+=(average-rgb)*u_saturation_factor;rgb=(rgb-0.5)*u_contrast_factor+0.5;vec3 u_high_vec=vec3(u_brightness_low,u_brightness_low,u_brightness_low);vec3 u_low_vec=vec3(u_brightness_high,u_brightness_high,u_brightness_high);gl_FragColor=vec4(mix(u_high_vec,u_low_vec,rgb)*color.a,color.a);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_tl_parent;uniform float u_scale_parent;uniform float u_buffer_scale;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos0;varying vec2 v_pos1;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos0=(((a_texture_pos/8192.0)-0.5)/u_buffer_scale )+0.5;v_pos1=(v_pos0*u_scale_parent)+u_tl_parent;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform sampler2D u_texture;varying vec2 v_tex;varying float v_fade_opacity;\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\nlowp float alpha=opacity*v_fade_opacity;gl_FragColor=texture2D(u_texture,v_tex)*alpha;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform highp float u_camera_to_center_distance;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform float u_fade_change;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform vec2 u_texsize;varying vec2 v_tex;varying float v_fade_opacity;\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;vec2 a_minFontScale=a_pixeloffset.zw/256.0;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\\ncamera_to_anchor_distance/u_camera_to_center_distance :\\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*max(a_minFontScale,fontScale)+a_pxoffset/16.0),z,1.0);v_tex=a_tex/u_texsize;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float visibility=calculate_visibility(projectedPoint);v_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#define SDF_PX 8.0\\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;uniform bool u_is_text;varying vec2 v_data0;varying vec3 v_data1;\\n#pragma mapbox: define highp vec4 fill_color\\n#pragma mapbox: define highp vec4 halo_color\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float halo_width\\n#pragma mapbox: define lowp float halo_blur\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 fill_color\\n#pragma mapbox: initialize highp vec4 halo_color\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float halo_width\\n#pragma mapbox: initialize lowp float halo_blur\\nfloat EDGE_GAMMA=0.105/u_device_pixel_ratio;vec2 tex=v_data0.xy;float gamma_scale=v_data1.x;float size=v_data1.y;float fade_opacity=v_data1[2];float fontScale=u_is_text ? size/24.0 : size;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float inner_edge=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);inner_edge=inner_edge+gamma*gamma_scale;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(inner_edge-gamma_scaled,inner_edge+gamma_scaled,dist);if (u_is_halo) {lowp float halo_edge=(6.0-halo_width/fontScale)/SDF_PX;alpha=min(smoothstep(halo_edge-gamma_scaled,halo_edge+gamma_scaled,dist),1.0-alpha);}gl_FragColor=color*(alpha*opacity*fade_opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;varying vec2 v_data0;varying vec3 v_data1;\\n#pragma mapbox: define highp vec4 fill_color\\n#pragma mapbox: define highp vec4 halo_color\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float halo_width\\n#pragma mapbox: define lowp float halo_blur\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 fill_color\\n#pragma mapbox: initialize highp vec4 halo_color\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float halo_width\\n#pragma mapbox: initialize lowp float halo_blur\\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\\ncamera_to_anchor_distance/u_camera_to_center_distance :\\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale+a_pxoffset),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0=a_tex/u_texsize;v_data1=vec3(gamma_scale,size,interpolated_fade_opacity);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#define SDF_PX 8.0\\n#define SDF 1.0\\n#define ICON 0.0\\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform sampler2D u_texture_icon;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;varying vec4 v_data0;varying vec4 v_data1;\\n#pragma mapbox: define highp vec4 fill_color\\n#pragma mapbox: define highp vec4 halo_color\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float halo_width\\n#pragma mapbox: define lowp float halo_blur\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 fill_color\\n#pragma mapbox: initialize highp vec4 halo_color\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float halo_width\\n#pragma mapbox: initialize lowp float halo_blur\\nfloat fade_opacity=v_data1[2];if (v_data1.w==ICON) {vec2 tex_icon=v_data0.zw;lowp float alpha=opacity*fade_opacity;gl_FragColor=texture2D(u_texture_icon,tex_icon)*alpha;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\nreturn;}vec2 tex=v_data0.xy;float EDGE_GAMMA=0.105/u_device_pixel_ratio;float gamma_scale=v_data1.x;float size=v_data1.y;float fontScale=size/24.0;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float buff=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);buff=(6.0-halo_width/fontScale)/SDF_PX;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(buff-gamma_scaled,buff+gamma_scaled,dist);gl_FragColor=color*(alpha*opacity*fade_opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;uniform vec2 u_texsize_icon;varying vec4 v_data0;varying vec4 v_data1;\\n#pragma mapbox: define highp vec4 fill_color\\n#pragma mapbox: define highp vec4 halo_color\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float halo_width\\n#pragma mapbox: define lowp float halo_blur\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 fill_color\\n#pragma mapbox: initialize highp vec4 halo_color\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float halo_width\\n#pragma mapbox: initialize lowp float halo_blur\\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);float is_sdf=a_size[0]-2.0*a_size_min;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\\ncamera_to_anchor_distance/u_camera_to_center_distance :\\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=size/24.0;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0.xy=a_tex/u_texsize;v_data0.zw=a_tex/u_texsize_icon;v_data1=vec4(gamma_scale,size,interpolated_fade_opacity,is_sdf);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying float v_depth;const highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitMsk=vec4(0.,vec3(1./256.0));highp vec4 pack(highp float value) {highp vec4 comp=fract(value*bitSh);comp-=comp.xxyz*bitMsk;return comp;}void main() {gl_FragColor=pack(v_depth);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'precision mediump float;uniform sampler2D u_texture;uniform float u_terrain_coords_id;varying vec2 v_texture_pos;void main() {vec4 rgba=texture2D(u_texture,v_texture_pos);gl_FragColor=vec4(rgba.r,rgba.g,rgba.b,u_terrain_coords_id);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform sampler2D u_texture;varying vec2 v_texture_pos;void main() {gl_FragColor=texture2D(u_texture,v_texture_pos);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec3 a_pos3d;uniform mat4 u_matrix;uniform float u_ele_delta;varying vec2 v_texture_pos;varying float v_depth;void main() {float extent=8192.0;float ele_delta=a_pos3d.z==1.0 ? u_ele_delta : 0.0;v_texture_pos=a_pos3d.xy/extent;gl_Position=u_matrix*vec4(a_pos3d.xy,get_elevation(a_pos3d.xy)-ele_delta,1.0);v_depth=gl_Position.z/gl_Position.w;}';\n","\n// Disable Flow annotations here because Flow doesn't support importing GLSL files\n\nimport preludeFrag from './_prelude.fragment.glsl.g';\nimport preludeVert from './_prelude.vertex.glsl.g';\nimport backgroundFrag from './background.fragment.glsl.g';\nimport backgroundVert from './background.vertex.glsl.g';\nimport backgroundPatternFrag from './background_pattern.fragment.glsl.g';\nimport backgroundPatternVert from './background_pattern.vertex.glsl.g';\nimport circleFrag from './circle.fragment.glsl.g';\nimport circleVert from './circle.vertex.glsl.g';\nimport clippingMaskFrag from './clipping_mask.fragment.glsl.g';\nimport clippingMaskVert from './clipping_mask.vertex.glsl.g';\nimport heatmapFrag from './heatmap.fragment.glsl.g';\nimport heatmapVert from './heatmap.vertex.glsl.g';\nimport heatmapTextureFrag from './heatmap_texture.fragment.glsl.g';\nimport heatmapTextureVert from './heatmap_texture.vertex.glsl.g';\nimport collisionBoxFrag from './collision_box.fragment.glsl.g';\nimport collisionBoxVert from './collision_box.vertex.glsl.g';\nimport collisionCircleFrag from './collision_circle.fragment.glsl.g';\nimport collisionCircleVert from './collision_circle.vertex.glsl.g';\nimport debugFrag from './debug.fragment.glsl.g';\nimport debugVert from './debug.vertex.glsl.g';\nimport fillFrag from './fill.fragment.glsl.g';\nimport fillVert from './fill.vertex.glsl.g';\nimport fillOutlineFrag from './fill_outline.fragment.glsl.g';\nimport fillOutlineVert from './fill_outline.vertex.glsl.g';\nimport fillOutlinePatternFrag from './fill_outline_pattern.fragment.glsl.g';\nimport fillOutlinePatternVert from './fill_outline_pattern.vertex.glsl.g';\nimport fillPatternFrag from './fill_pattern.fragment.glsl.g';\nimport fillPatternVert from './fill_pattern.vertex.glsl.g';\nimport fillExtrusionFrag from './fill_extrusion.fragment.glsl.g';\nimport fillExtrusionVert from './fill_extrusion.vertex.glsl.g';\nimport fillExtrusionPatternFrag from './fill_extrusion_pattern.fragment.glsl.g';\nimport fillExtrusionPatternVert from './fill_extrusion_pattern.vertex.glsl.g';\nimport hillshadePrepareFrag from './hillshade_prepare.fragment.glsl.g';\nimport hillshadePrepareVert from './hillshade_prepare.vertex.glsl.g';\nimport hillshadeFrag from './hillshade.fragment.glsl.g';\nimport hillshadeVert from './hillshade.vertex.glsl.g';\nimport lineFrag from './line.fragment.glsl.g';\nimport lineVert from './line.vertex.glsl.g';\nimport lineGradientFrag from './line_gradient.fragment.glsl.g';\nimport lineGradientVert from './line_gradient.vertex.glsl.g';\nimport linePatternFrag from './line_pattern.fragment.glsl.g';\nimport linePatternVert from './line_pattern.vertex.glsl.g';\nimport lineSDFFrag from './line_sdf.fragment.glsl.g';\nimport lineSDFVert from './line_sdf.vertex.glsl.g';\nimport rasterFrag from './raster.fragment.glsl.g';\nimport rasterVert from './raster.vertex.glsl.g';\nimport symbolIconFrag from './symbol_icon.fragment.glsl.g';\nimport symbolIconVert from './symbol_icon.vertex.glsl.g';\nimport symbolSDFFrag from './symbol_sdf.fragment.glsl.g';\nimport symbolSDFVert from './symbol_sdf.vertex.glsl.g';\nimport symbolTextAndIconFrag from './symbol_text_and_icon.fragment.glsl.g';\nimport symbolTextAndIconVert from './symbol_text_and_icon.vertex.glsl.g';\nimport terrainDepthFrag from './terrain_depth.fragment.glsl.g';\nimport terrainCoordsFrag from './terrain_coords.fragment.glsl.g';\nimport terrainFrag from './terrain.fragment.glsl.g';\nimport terrainVert from './terrain.vertex.glsl.g';\n\nexport const shaders = {\n prelude: compile(preludeFrag, preludeVert),\n background: compile(backgroundFrag, backgroundVert),\n backgroundPattern: compile(backgroundPatternFrag, backgroundPatternVert),\n circle: compile(circleFrag, circleVert),\n clippingMask: compile(clippingMaskFrag, clippingMaskVert),\n heatmap: compile(heatmapFrag, heatmapVert),\n heatmapTexture: compile(heatmapTextureFrag, heatmapTextureVert),\n collisionBox: compile(collisionBoxFrag, collisionBoxVert),\n collisionCircle: compile(collisionCircleFrag, collisionCircleVert),\n debug: compile(debugFrag, debugVert),\n fill: compile(fillFrag, fillVert),\n fillOutline: compile(fillOutlineFrag, fillOutlineVert),\n fillOutlinePattern: compile(fillOutlinePatternFrag, fillOutlinePatternVert),\n fillPattern: compile(fillPatternFrag, fillPatternVert),\n fillExtrusion: compile(fillExtrusionFrag, fillExtrusionVert),\n fillExtrusionPattern: compile(fillExtrusionPatternFrag, fillExtrusionPatternVert),\n hillshadePrepare: compile(hillshadePrepareFrag, hillshadePrepareVert),\n hillshade: compile(hillshadeFrag, hillshadeVert),\n line: compile(lineFrag, lineVert),\n lineGradient: compile(lineGradientFrag, lineGradientVert),\n linePattern: compile(linePatternFrag, linePatternVert),\n lineSDF: compile(lineSDFFrag, lineSDFVert),\n raster: compile(rasterFrag, rasterVert),\n symbolIcon: compile(symbolIconFrag, symbolIconVert),\n symbolSDF: compile(symbolSDFFrag, symbolSDFVert),\n symbolTextAndIcon: compile(symbolTextAndIconFrag, symbolTextAndIconVert),\n terrain: compile(terrainFrag, terrainVert),\n terrainDepth: compile(terrainDepthFrag, terrainVert),\n terrainCoords: compile(terrainCoordsFrag, terrainVert)\n};\n\n// Expand #pragmas to #ifdefs.\n\nfunction compile(fragmentSource, vertexSource) {\n const re = /#pragma mapbox: ([\\w]+) ([\\w]+) ([\\w]+) ([\\w]+)/g;\n\n const staticAttributes = vertexSource.match(/attribute ([\\w]+) ([\\w]+)/g);\n const fragmentUniforms = fragmentSource.match(/uniform ([\\w]+) ([\\w]+)([\\s]*)([\\w]*)/g);\n const vertexUniforms = vertexSource.match(/uniform ([\\w]+) ([\\w]+)([\\s]*)([\\w]*)/g);\n const staticUniforms = vertexUniforms ? vertexUniforms.concat(fragmentUniforms) : fragmentUniforms;\n\n const fragmentPragmas = {};\n\n fragmentSource = fragmentSource.replace(re, (match, operation, precision, type, name) => {\n fragmentPragmas[name] = true;\n if (operation === 'define') {\n return `\n#ifndef HAS_UNIFORM_u_${name}\nvarying ${precision} ${type} ${name};\n#else\nuniform ${precision} ${type} u_${name};\n#endif\n`;\n } else /* if (operation === 'initialize') */ {\n return `\n#ifdef HAS_UNIFORM_u_${name}\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n }\n });\n\n vertexSource = vertexSource.replace(re, (match, operation, precision, type, name) => {\n const attrType = type === 'float' ? 'vec2' : 'vec4';\n const unpackType = name.match(/color/) ? 'color' : attrType;\n\n if (fragmentPragmas[name]) {\n if (operation === 'define') {\n return `\n#ifndef HAS_UNIFORM_u_${name}\nuniform lowp float u_${name}_t;\nattribute ${precision} ${attrType} a_${name};\nvarying ${precision} ${type} ${name};\n#else\nuniform ${precision} ${type} u_${name};\n#endif\n`;\n } else /* if (operation === 'initialize') */ {\n if (unpackType === 'vec4') {\n // vec4 attributes are only used for cross-faded properties, and are not packed\n return `\n#ifndef HAS_UNIFORM_u_${name}\n ${name} = a_${name};\n#else\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n } else {\n return `\n#ifndef HAS_UNIFORM_u_${name}\n ${name} = unpack_mix_${unpackType}(a_${name}, u_${name}_t);\n#else\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n }\n }\n } else {\n if (operation === 'define') {\n return `\n#ifndef HAS_UNIFORM_u_${name}\nuniform lowp float u_${name}_t;\nattribute ${precision} ${attrType} a_${name};\n#else\nuniform ${precision} ${type} u_${name};\n#endif\n`;\n } else /* if (operation === 'initialize') */ {\n if (unpackType === 'vec4') {\n // vec4 attributes are only used for cross-faded properties, and are not packed\n return `\n#ifndef HAS_UNIFORM_u_${name}\n ${precision} ${type} ${name} = a_${name};\n#else\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n } else /* */ {\n return `\n#ifndef HAS_UNIFORM_u_${name}\n ${precision} ${type} ${name} = unpack_mix_${unpackType}(a_${name}, u_${name}_t);\n#else\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n }\n }\n }\n });\n\n return {fragmentSource, vertexSource, staticAttributes, staticUniforms};\n}\n","\nimport type {Program} from './program';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {IndexBuffer} from '../gl/index_buffer';\nimport type {Context} from '../gl/context';\n\n/**\n * @internal\n * A vertex array object used to pass data to the webgl code\n */\nexport class VertexArrayObject {\n context: Context;\n boundProgram: Program;\n boundLayoutVertexBuffer: VertexBuffer;\n boundPaintVertexBuffers: Array;\n boundIndexBuffer: IndexBuffer;\n boundVertexOffset: number;\n boundDynamicVertexBuffer: VertexBuffer;\n boundDynamicVertexBuffer2: VertexBuffer;\n boundDynamicVertexBuffer3: VertexBuffer;\n vao: any;\n\n constructor() {\n this.boundProgram = null;\n this.boundLayoutVertexBuffer = null;\n this.boundPaintVertexBuffers = [];\n this.boundIndexBuffer = null;\n this.boundVertexOffset = null;\n this.boundDynamicVertexBuffer = null;\n this.vao = null;\n }\n\n bind(context: Context,\n program: Program,\n layoutVertexBuffer: VertexBuffer,\n paintVertexBuffers: Array,\n indexBuffer?: IndexBuffer | null,\n vertexOffset?: number | null,\n dynamicVertexBuffer?: VertexBuffer | null,\n dynamicVertexBuffer2?: VertexBuffer | null,\n dynamicVertexBuffer3?: VertexBuffer | null) {\n\n this.context = context;\n\n let paintBuffersDiffer = this.boundPaintVertexBuffers.length !== paintVertexBuffers.length;\n for (let i = 0; !paintBuffersDiffer && i < paintVertexBuffers.length; i++) {\n if (this.boundPaintVertexBuffers[i] !== paintVertexBuffers[i]) {\n paintBuffersDiffer = true;\n }\n }\n\n const isFreshBindRequired = (\n !this.vao ||\n this.boundProgram !== program ||\n this.boundLayoutVertexBuffer !== layoutVertexBuffer ||\n paintBuffersDiffer ||\n this.boundIndexBuffer !== indexBuffer ||\n this.boundVertexOffset !== vertexOffset ||\n this.boundDynamicVertexBuffer !== dynamicVertexBuffer ||\n this.boundDynamicVertexBuffer2 !== dynamicVertexBuffer2 ||\n this.boundDynamicVertexBuffer3 !== dynamicVertexBuffer3\n );\n\n if (isFreshBindRequired) {\n this.freshBind(program, layoutVertexBuffer, paintVertexBuffers, indexBuffer, vertexOffset, dynamicVertexBuffer, dynamicVertexBuffer2, dynamicVertexBuffer3);\n } else {\n context.bindVertexArray.set(this.vao);\n\n if (dynamicVertexBuffer) {\n // The buffer may have been updated. Rebind to upload data.\n dynamicVertexBuffer.bind();\n }\n\n if (indexBuffer && indexBuffer.dynamicDraw) {\n indexBuffer.bind();\n }\n\n if (dynamicVertexBuffer2) {\n dynamicVertexBuffer2.bind();\n }\n\n if (dynamicVertexBuffer3) {\n dynamicVertexBuffer3.bind();\n }\n }\n }\n\n freshBind(program: Program,\n layoutVertexBuffer: VertexBuffer,\n paintVertexBuffers: Array,\n indexBuffer?: IndexBuffer | null,\n vertexOffset?: number | null,\n dynamicVertexBuffer?: VertexBuffer | null,\n dynamicVertexBuffer2?: VertexBuffer | null,\n dynamicVertexBuffer3?: VertexBuffer | null) {\n\n const numNextAttributes = program.numAttributes;\n\n const context = this.context;\n const gl = context.gl;\n\n if (this.vao) this.destroy();\n this.vao = context.createVertexArray();\n context.bindVertexArray.set(this.vao);\n\n // store the arguments so that we can verify them when the vao is bound again\n this.boundProgram = program;\n this.boundLayoutVertexBuffer = layoutVertexBuffer;\n this.boundPaintVertexBuffers = paintVertexBuffers;\n this.boundIndexBuffer = indexBuffer;\n this.boundVertexOffset = vertexOffset;\n this.boundDynamicVertexBuffer = dynamicVertexBuffer;\n this.boundDynamicVertexBuffer2 = dynamicVertexBuffer2;\n this.boundDynamicVertexBuffer3 = dynamicVertexBuffer3;\n\n layoutVertexBuffer.enableAttributes(gl, program);\n for (const vertexBuffer of paintVertexBuffers) {\n vertexBuffer.enableAttributes(gl, program);\n }\n\n if (dynamicVertexBuffer) {\n dynamicVertexBuffer.enableAttributes(gl, program);\n }\n if (dynamicVertexBuffer2) {\n dynamicVertexBuffer2.enableAttributes(gl, program);\n }\n if (dynamicVertexBuffer3) {\n dynamicVertexBuffer3.enableAttributes(gl, program);\n }\n\n layoutVertexBuffer.bind();\n layoutVertexBuffer.setVertexAttribPointers(gl, program, vertexOffset);\n for (const vertexBuffer of paintVertexBuffers) {\n vertexBuffer.bind();\n vertexBuffer.setVertexAttribPointers(gl, program, vertexOffset);\n }\n\n if (dynamicVertexBuffer) {\n dynamicVertexBuffer.bind();\n dynamicVertexBuffer.setVertexAttribPointers(gl, program, vertexOffset);\n }\n if (indexBuffer) {\n indexBuffer.bind();\n }\n if (dynamicVertexBuffer2) {\n dynamicVertexBuffer2.bind();\n dynamicVertexBuffer2.setVertexAttribPointers(gl, program, vertexOffset);\n }\n if (dynamicVertexBuffer3) {\n dynamicVertexBuffer3.bind();\n dynamicVertexBuffer3.setVertexAttribPointers(gl, program, vertexOffset);\n }\n\n context.currentNumAttributes = numNextAttributes;\n }\n\n destroy() {\n if (this.vao) {\n this.context.deleteVertexArray(this.vao);\n this.vao = null;\n }\n }\n}\n","import {\n Uniform1i,\n Uniform1f,\n Uniform4f,\n UniformMatrix4f\n} from '../uniform_binding';\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../../render/uniform_binding';\nimport {mat4} from 'gl-matrix';\n\nexport type TerrainPreludeUniformsType = {\n 'u_depth': Uniform1i;\n 'u_terrain': Uniform1i;\n 'u_terrain_dim': Uniform1f;\n 'u_terrain_matrix': UniformMatrix4f;\n 'u_terrain_unpack': Uniform4f;\n 'u_terrain_exaggeration': Uniform1f;\n};\n\nexport type TerrainUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_texture': Uniform1i;\n 'u_ele_delta': Uniform1f;\n};\n\nexport type TerrainDepthUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_ele_delta': Uniform1f;\n};\n\nexport type TerrainCoordsUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_texture': Uniform1i;\n 'u_terrain_coords_id': Uniform1f;\n 'u_ele_delta': Uniform1f;\n};\n\nconst terrainPreludeUniforms = (context: Context, locations: UniformLocations): TerrainPreludeUniformsType => ({\n 'u_depth': new Uniform1i(context, locations.u_depth),\n 'u_terrain': new Uniform1i(context, locations.u_terrain),\n 'u_terrain_dim': new Uniform1f(context, locations.u_terrain_dim),\n 'u_terrain_matrix': new UniformMatrix4f(context, locations.u_terrain_matrix),\n 'u_terrain_unpack': new Uniform4f(context, locations.u_terrain_unpack),\n 'u_terrain_exaggeration': new Uniform1f(context, locations.u_terrain_exaggeration)\n});\n\nconst terrainUniforms = (context: Context, locations: UniformLocations): TerrainUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_texture': new Uniform1i(context, locations.u_texture),\n 'u_ele_delta': new Uniform1f(context, locations.u_ele_delta)\n});\n\nconst terrainDepthUniforms = (context: Context, locations: UniformLocations): TerrainDepthUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_ele_delta': new Uniform1f(context, locations.u_ele_delta)\n});\n\nconst terrainCoordsUniforms = (context: Context, locations: UniformLocations): TerrainCoordsUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_texture': new Uniform1i(context, locations.u_texture),\n 'u_terrain_coords_id': new Uniform1f(context, locations.u_terrain_coords_id),\n 'u_ele_delta': new Uniform1f(context, locations.u_ele_delta)\n});\n\nconst terrainUniformValues = (\n matrix: mat4,\n eleDelta: number\n): UniformValues => ({\n 'u_matrix': matrix,\n 'u_texture': 0,\n 'u_ele_delta': eleDelta\n});\n\nconst terrainDepthUniformValues = (\n matrix: mat4,\n eleDelta: number\n): UniformValues => ({\n 'u_matrix': matrix,\n 'u_ele_delta': eleDelta\n});\n\nconst terrainCoordsUniformValues = (\n matrix: mat4,\n coordsId: number,\n eleDelta: number\n): UniformValues => ({\n 'u_matrix': matrix,\n 'u_terrain_coords_id': coordsId / 255,\n 'u_texture': 0,\n 'u_ele_delta': eleDelta\n});\n\nexport {terrainUniforms, terrainDepthUniforms, terrainCoordsUniforms, terrainPreludeUniforms, terrainUniformValues, terrainDepthUniformValues, terrainCoordsUniformValues};\n","import {shaders} from '../shaders/shaders';\nimport {ProgramConfiguration} from '../data/program_configuration';\nimport {VertexArrayObject} from './vertex_array_object';\nimport {Context} from '../gl/context';\n\nimport type {SegmentVector} from '../data/segment';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {IndexBuffer} from '../gl/index_buffer';\nimport type {DepthMode} from '../gl/depth_mode';\nimport type {StencilMode} from '../gl/stencil_mode';\nimport type {ColorMode} from '../gl/color_mode';\nimport type {CullFaceMode} from '../gl/cull_face_mode';\nimport type {UniformBindings, UniformValues, UniformLocations} from './uniform_binding';\nimport type {BinderUniform} from '../data/program_configuration';\nimport {terrainPreludeUniforms, TerrainPreludeUniformsType} from './program/terrain_program';\nimport type {TerrainData} from '../render/terrain';\nimport {Terrain} from '../render/terrain';\n\nexport type DrawMode = WebGLRenderingContextBase['LINES'] | WebGLRenderingContextBase['TRIANGLES'] | WebGL2RenderingContext['LINE_STRIP'];\n\nfunction getTokenizedAttributesAndUniforms(array: Array): Array {\n const result = [];\n\n for (let i = 0; i < array.length; i++) {\n if (array[i] === null) continue;\n const token = array[i].split(' ');\n result.push(token.pop());\n }\n return result;\n}\n\n/**\n * @internal\n * A webgl program to execute in the GPU space\n */\nexport class Program {\n program: WebGLProgram;\n attributes: {[_: string]: number};\n numAttributes: number;\n fixedUniforms: Us;\n terrainUniforms: TerrainPreludeUniformsType;\n binderUniforms: Array;\n failedToCreate: boolean;\n\n constructor(context: Context,\n source: {\n fragmentSource: string;\n vertexSource: string;\n staticAttributes: Array;\n staticUniforms: Array;\n },\n configuration: ProgramConfiguration,\n fixedUniforms: (b: Context, a: UniformLocations) => Us,\n showOverdrawInspector: boolean,\n terrain: Terrain) {\n\n const gl = context.gl;\n this.program = gl.createProgram();\n\n const staticAttrInfo = getTokenizedAttributesAndUniforms(source.staticAttributes);\n const dynamicAttrInfo = configuration ? configuration.getBinderAttributes() : [];\n const allAttrInfo = staticAttrInfo.concat(dynamicAttrInfo);\n\n const preludeUniformsInfo = shaders.prelude.staticUniforms ? getTokenizedAttributesAndUniforms(shaders.prelude.staticUniforms) : [];\n const staticUniformsInfo = source.staticUniforms ? getTokenizedAttributesAndUniforms(source.staticUniforms) : [];\n const dynamicUniformsInfo = configuration ? configuration.getBinderUniforms() : [];\n // remove duplicate uniforms\n const uniformList = preludeUniformsInfo.concat(staticUniformsInfo).concat(dynamicUniformsInfo);\n const allUniformsInfo = [];\n for (const uniform of uniformList) {\n if (allUniformsInfo.indexOf(uniform) < 0) allUniformsInfo.push(uniform);\n }\n\n const defines = configuration ? configuration.defines() : [];\n if (showOverdrawInspector) {\n defines.push('#define OVERDRAW_INSPECTOR;');\n }\n if (terrain) {\n defines.push('#define TERRAIN3D;');\n }\n\n const fragmentSource = defines.concat(shaders.prelude.fragmentSource, source.fragmentSource).join('\\n');\n const vertexSource = defines.concat(shaders.prelude.vertexSource, source.vertexSource).join('\\n');\n\n const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);\n if (gl.isContextLost()) {\n this.failedToCreate = true;\n return;\n }\n gl.shaderSource(fragmentShader, fragmentSource);\n gl.compileShader(fragmentShader);\n\n if (!gl.getShaderParameter(fragmentShader, gl.COMPILE_STATUS)) {\n throw new Error(`Could not compile fragment shader: ${gl.getShaderInfoLog(fragmentShader)}`);\n }\n\n gl.attachShader(this.program, fragmentShader);\n\n const vertexShader = gl.createShader(gl.VERTEX_SHADER);\n if (gl.isContextLost()) {\n this.failedToCreate = true;\n return;\n }\n gl.shaderSource(vertexShader, vertexSource);\n gl.compileShader(vertexShader);\n\n if (!gl.getShaderParameter(vertexShader, gl.COMPILE_STATUS)) {\n throw new Error(`Could not compile vertex shader: ${gl.getShaderInfoLog(vertexShader)}`);\n }\n\n gl.attachShader(this.program, vertexShader);\n\n this.attributes = {};\n const uniformLocations = {};\n\n this.numAttributes = allAttrInfo.length;\n\n for (let i = 0; i < this.numAttributes; i++) {\n if (allAttrInfo[i]) {\n gl.bindAttribLocation(this.program, i, allAttrInfo[i]);\n this.attributes[allAttrInfo[i]] = i;\n }\n }\n\n gl.linkProgram(this.program);\n\n if (!gl.getProgramParameter(this.program, gl.LINK_STATUS)) {\n throw new Error(`Program failed to link: ${gl.getProgramInfoLog(this.program)}`);\n }\n\n gl.deleteShader(vertexShader);\n gl.deleteShader(fragmentShader);\n\n for (let it = 0; it < allUniformsInfo.length; it++) {\n const uniform = allUniformsInfo[it];\n if (uniform && !uniformLocations[uniform]) {\n const uniformLocation = gl.getUniformLocation(this.program, uniform);\n if (uniformLocation) {\n uniformLocations[uniform] = uniformLocation;\n }\n }\n }\n\n this.fixedUniforms = fixedUniforms(context, uniformLocations);\n this.terrainUniforms = terrainPreludeUniforms(context, uniformLocations);\n this.binderUniforms = configuration ? configuration.getUniforms(context, uniformLocations) : [];\n }\n\n draw(context: Context,\n drawMode: DrawMode,\n depthMode: Readonly,\n stencilMode: Readonly,\n colorMode: Readonly,\n cullFaceMode: Readonly,\n uniformValues: UniformValues,\n terrain: TerrainData,\n layerID: string,\n layoutVertexBuffer: VertexBuffer,\n indexBuffer: IndexBuffer,\n segments: SegmentVector,\n currentProperties?: any,\n zoom?: number | null,\n configuration?: ProgramConfiguration | null,\n dynamicLayoutBuffer?: VertexBuffer | null,\n dynamicLayoutBuffer2?: VertexBuffer | null,\n dynamicLayoutBuffer3?: VertexBuffer | null) {\n\n const gl = context.gl;\n\n if (this.failedToCreate) return;\n\n context.program.set(this.program);\n context.setDepthMode(depthMode);\n context.setStencilMode(stencilMode);\n context.setColorMode(colorMode);\n context.setCullFace(cullFaceMode);\n\n // set variables used by the 3d functions defined in _prelude.vertex.glsl\n if (terrain) {\n context.activeTexture.set(gl.TEXTURE2);\n gl.bindTexture(gl.TEXTURE_2D, terrain.depthTexture);\n context.activeTexture.set(gl.TEXTURE3);\n gl.bindTexture(gl.TEXTURE_2D, terrain.texture);\n for (const name in this.terrainUniforms) {\n this.terrainUniforms[name].set(terrain[name]);\n }\n }\n\n for (const name in this.fixedUniforms) {\n this.fixedUniforms[name].set(uniformValues[name]);\n }\n\n if (configuration) {\n configuration.setUniforms(context, this.binderUniforms, currentProperties, {zoom: (zoom as any)});\n }\n\n let primitiveSize = 0;\n switch (drawMode) {\n case gl.LINES:\n primitiveSize = 2;\n break;\n case gl.TRIANGLES:\n primitiveSize = 3;\n break;\n case gl.LINE_STRIP:\n primitiveSize = 1;\n break;\n }\n\n for (const segment of segments.get()) {\n const vaos = segment.vaos || (segment.vaos = {});\n const vao: VertexArrayObject = vaos[layerID] || (vaos[layerID] = new VertexArrayObject());\n\n vao.bind(\n context,\n this,\n layoutVertexBuffer,\n configuration ? configuration.getPaintVertexBuffers() : [],\n indexBuffer,\n segment.vertexOffset,\n dynamicLayoutBuffer,\n dynamicLayoutBuffer2,\n dynamicLayoutBuffer3\n );\n\n gl.drawElements(\n drawMode,\n segment.primitiveLength * primitiveSize,\n gl.UNSIGNED_SHORT,\n segment.primitiveOffset * primitiveSize * 2);\n }\n }\n}\n","import {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f\n} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\n\nimport type {Painter} from '../painter';\nimport type {OverscaledTileID} from '../../source/tile_id';\nimport type {CrossFaded} from '../../style/properties';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport type {UniformValues} from '../uniform_binding';\nimport type {Tile} from '../../source/tile';\nimport type {ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\n\ntype BackgroundPatternUniformsType = {\n 'u_image': Uniform1i;\n 'u_pattern_tl_a': Uniform2f;\n 'u_pattern_br_a': Uniform2f;\n 'u_pattern_tl_b': Uniform2f;\n 'u_pattern_br_b': Uniform2f;\n 'u_texsize': Uniform2f;\n 'u_mix': Uniform1f;\n 'u_pattern_size_a': Uniform2f;\n 'u_pattern_size_b': Uniform2f;\n 'u_scale_a': Uniform1f;\n 'u_scale_b': Uniform1f;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_tile_units_to_pixels': Uniform1f;\n};\n\nexport type PatternUniformsType = {\n // pattern uniforms:\n 'u_image': Uniform1i;\n 'u_texsize': Uniform2f;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n};\n\nfunction patternUniformValues(crossfade: CrossfadeParameters, painter: Painter, tile: Tile): UniformValues {\n\n const tileRatio = 1 / pixelsToTileUnits(tile, 1, painter.transform.tileZoom);\n\n const numTiles = Math.pow(2, tile.tileID.overscaledZ);\n const tileSizeAtNearestZoom = tile.tileSize * Math.pow(2, painter.transform.tileZoom) / numTiles;\n\n const pixelX = tileSizeAtNearestZoom * (tile.tileID.canonical.x + tile.tileID.wrap * numTiles);\n const pixelY = tileSizeAtNearestZoom * tile.tileID.canonical.y;\n\n return {\n 'u_image': 0,\n 'u_texsize': tile.imageAtlasTexture.size,\n 'u_scale': [tileRatio, crossfade.fromScale, crossfade.toScale],\n 'u_fade': crossfade.t,\n // split the pixel coord into two pairs of 16 bit numbers. The glsl spec only guarantees 16 bits of precision.\n 'u_pixel_coord_upper': [pixelX >> 16, pixelY >> 16],\n 'u_pixel_coord_lower': [pixelX & 0xFFFF, pixelY & 0xFFFF]\n };\n}\n\nfunction bgPatternUniformValues(\n image: CrossFaded,\n crossfade: CrossfadeParameters,\n painter: Painter,\n tile: {\n tileID: OverscaledTileID;\n tileSize: number;\n }\n): UniformValues {\n const imagePosA = painter.imageManager.getPattern(image.from.toString());\n const imagePosB = painter.imageManager.getPattern(image.to.toString());\n const {width, height} = painter.imageManager.getPixelSize();\n\n const numTiles = Math.pow(2, tile.tileID.overscaledZ);\n const tileSizeAtNearestZoom = tile.tileSize * Math.pow(2, painter.transform.tileZoom) / numTiles;\n\n const pixelX = tileSizeAtNearestZoom * (tile.tileID.canonical.x + tile.tileID.wrap * numTiles);\n const pixelY = tileSizeAtNearestZoom * tile.tileID.canonical.y;\n\n return {\n 'u_image': 0,\n 'u_pattern_tl_a': (imagePosA as any).tl,\n 'u_pattern_br_a': (imagePosA as any).br,\n 'u_pattern_tl_b': (imagePosB as any).tl,\n 'u_pattern_br_b': (imagePosB as any).br,\n 'u_texsize': [width, height],\n 'u_mix': crossfade.t,\n 'u_pattern_size_a': (imagePosA as any).displaySize,\n 'u_pattern_size_b': (imagePosB as any).displaySize,\n 'u_scale_a': crossfade.fromScale,\n 'u_scale_b': crossfade.toScale,\n 'u_tile_units_to_pixels': 1 / pixelsToTileUnits(tile, 1, painter.transform.tileZoom),\n // split the pixel coord into two pairs of 16 bit numbers. The glsl spec only guarantees 16 bits of precision.\n 'u_pixel_coord_upper': [pixelX >> 16, pixelY >> 16],\n 'u_pixel_coord_lower': [pixelX & 0xFFFF, pixelY & 0xFFFF]\n };\n}\nexport {bgPatternUniformValues, patternUniformValues};\n","import {patternUniformValues} from './pattern';\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f,\n UniformMatrix4f\n} from '../uniform_binding';\n\nimport {mat3, mat4, vec3} from 'gl-matrix';\nimport {extend} from '../../util/util';\n\nimport type {Context} from '../../gl/context';\nimport type {Painter} from '../painter';\nimport type {OverscaledTileID} from '../../source/tile_id';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport type {Tile} from '../../source/tile';\n\nexport type FillExtrusionUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_lightpos': Uniform3f;\n 'u_lightintensity': Uniform1f;\n 'u_lightcolor': Uniform3f;\n 'u_vertical_gradient': Uniform1f;\n 'u_opacity': Uniform1f;\n};\n\nexport type FillExtrusionPatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_lightpos': Uniform3f;\n 'u_lightintensity': Uniform1f;\n 'u_lightcolor': Uniform3f;\n 'u_height_factor': Uniform1f;\n 'u_vertical_gradient': Uniform1f;\n // pattern uniforms:\n 'u_texsize': Uniform2f;\n 'u_image': Uniform1i;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n 'u_opacity': Uniform1f;\n};\n\nconst fillExtrusionUniforms = (context: Context, locations: UniformLocations): FillExtrusionUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_lightpos': new Uniform3f(context, locations.u_lightpos),\n 'u_lightintensity': new Uniform1f(context, locations.u_lightintensity),\n 'u_lightcolor': new Uniform3f(context, locations.u_lightcolor),\n 'u_vertical_gradient': new Uniform1f(context, locations.u_vertical_gradient),\n 'u_opacity': new Uniform1f(context, locations.u_opacity)\n});\n\nconst fillExtrusionPatternUniforms = (context: Context, locations: UniformLocations): FillExtrusionPatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_lightpos': new Uniform3f(context, locations.u_lightpos),\n 'u_lightintensity': new Uniform1f(context, locations.u_lightintensity),\n 'u_lightcolor': new Uniform3f(context, locations.u_lightcolor),\n 'u_vertical_gradient': new Uniform1f(context, locations.u_vertical_gradient),\n 'u_height_factor': new Uniform1f(context, locations.u_height_factor),\n // pattern uniforms\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper),\n 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower),\n 'u_scale': new Uniform3f(context, locations.u_scale),\n 'u_fade': new Uniform1f(context, locations.u_fade),\n 'u_opacity': new Uniform1f(context, locations.u_opacity)\n});\n\nconst fillExtrusionUniformValues = (\n matrix: mat4,\n painter: Painter,\n shouldUseVerticalGradient: boolean,\n opacity: number\n): UniformValues => {\n const light = painter.style.light;\n const _lp = light.properties.get('position');\n const lightPos = [_lp.x, _lp.y, _lp.z] as vec3;\n const lightMat = mat3.create();\n if (light.properties.get('anchor') === 'viewport') {\n mat3.fromRotation(lightMat, -painter.transform.angle);\n }\n vec3.transformMat3(lightPos, lightPos, lightMat);\n\n const lightColor = light.properties.get('color');\n\n return {\n 'u_matrix': matrix,\n 'u_lightpos': lightPos,\n 'u_lightintensity': light.properties.get('intensity'),\n 'u_lightcolor': [lightColor.r, lightColor.g, lightColor.b],\n 'u_vertical_gradient': +shouldUseVerticalGradient,\n 'u_opacity': opacity\n };\n};\n\nconst fillExtrusionPatternUniformValues = (\n matrix: mat4,\n painter: Painter,\n shouldUseVerticalGradient: boolean,\n opacity: number,\n coord: OverscaledTileID,\n crossfade: CrossfadeParameters,\n tile: Tile\n): UniformValues => {\n return extend(fillExtrusionUniformValues(matrix, painter, shouldUseVerticalGradient, opacity),\n patternUniformValues(crossfade, painter, tile),\n {\n 'u_height_factor': -Math.pow(2, coord.overscaledZ) / tile.tileSize / 8\n });\n};\n\nexport {\n fillExtrusionUniforms,\n fillExtrusionPatternUniforms,\n fillExtrusionUniformValues,\n fillExtrusionPatternUniformValues\n};\n","import {patternUniformValues} from './pattern';\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f,\n UniformMatrix4f\n} from '../uniform_binding';\nimport {extend} from '../../util/util';\n\nimport type {Painter} from '../painter';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Context} from '../../gl/context';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport type {Tile} from '../../source/tile';\nimport {mat4} from 'gl-matrix';\n\nexport type FillUniformsType = {\n 'u_matrix': UniformMatrix4f;\n};\n\nexport type FillOutlineUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_world': Uniform2f;\n};\n\nexport type FillPatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n // pattern uniforms:\n 'u_texsize': Uniform2f;\n 'u_image': Uniform1i;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n};\n\nexport type FillOutlinePatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_world': Uniform2f;\n // pattern uniforms:\n 'u_texsize': Uniform2f;\n 'u_image': Uniform1i;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n};\n\nconst fillUniforms = (context: Context, locations: UniformLocations): FillUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix)\n});\n\nconst fillPatternUniforms = (context: Context, locations: UniformLocations): FillPatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper),\n 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower),\n 'u_scale': new Uniform3f(context, locations.u_scale),\n 'u_fade': new Uniform1f(context, locations.u_fade)\n});\n\nconst fillOutlineUniforms = (context: Context, locations: UniformLocations): FillOutlineUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_world': new Uniform2f(context, locations.u_world)\n});\n\nconst fillOutlinePatternUniforms = (context: Context, locations: UniformLocations): FillOutlinePatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_world': new Uniform2f(context, locations.u_world),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper),\n 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower),\n 'u_scale': new Uniform3f(context, locations.u_scale),\n 'u_fade': new Uniform1f(context, locations.u_fade)\n});\n\nconst fillUniformValues = (matrix: mat4): UniformValues => ({\n 'u_matrix': matrix\n});\n\nconst fillPatternUniformValues = (\n matrix: mat4,\n painter: Painter,\n crossfade: CrossfadeParameters,\n tile: Tile\n): UniformValues => extend(\n fillUniformValues(matrix),\n patternUniformValues(crossfade, painter, tile)\n);\n\nconst fillOutlineUniformValues = (matrix: mat4, drawingBufferSize: [number, number]): UniformValues => ({\n 'u_matrix': matrix,\n 'u_world': drawingBufferSize\n});\n\nconst fillOutlinePatternUniformValues = (\n matrix: mat4,\n painter: Painter,\n crossfade: CrossfadeParameters,\n tile: Tile,\n drawingBufferSize: [number, number]\n): UniformValues => extend(\n fillPatternUniformValues(matrix, painter, crossfade, tile),\n {\n 'u_world': drawingBufferSize\n }\n);\n\nexport {\n fillUniforms,\n fillPatternUniforms,\n fillOutlineUniforms,\n fillOutlinePatternUniforms,\n fillUniformValues,\n fillPatternUniformValues,\n fillOutlineUniformValues,\n fillOutlinePatternUniformValues\n};\n","import {Uniform1i, Uniform1f, Uniform2f, UniformMatrix4f} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {OverscaledTileID} from '../../source/tile_id';\nimport type {Tile} from '../../source/tile';\nimport type {CircleStyleLayer} from '../../style/style_layer/circle_style_layer';\nimport type {Painter} from '../painter';\n\nexport type CircleUniformsType = {\n 'u_camera_to_center_distance': Uniform1f;\n 'u_scale_with_map': Uniform1i;\n 'u_pitch_with_map': Uniform1i;\n 'u_extrude_scale': Uniform2f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n};\n\nconst circleUniforms = (context: Context, locations: UniformLocations): CircleUniformsType => ({\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_scale_with_map': new Uniform1i(context, locations.u_scale_with_map),\n 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map),\n 'u_extrude_scale': new Uniform2f(context, locations.u_extrude_scale),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix)\n});\n\nconst circleUniformValues = (\n painter: Painter,\n coord: OverscaledTileID,\n tile: Tile,\n layer: CircleStyleLayer\n): UniformValues => {\n const transform = painter.transform;\n\n let pitchWithMap: boolean, extrudeScale: [number, number];\n if (layer.paint.get('circle-pitch-alignment') === 'map') {\n const pixelRatio = pixelsToTileUnits(tile, 1, transform.zoom);\n pitchWithMap = true;\n extrudeScale = [pixelRatio, pixelRatio];\n } else {\n pitchWithMap = false;\n extrudeScale = transform.pixelsToGLUnits;\n }\n\n return {\n 'u_camera_to_center_distance': transform.cameraToCenterDistance,\n 'u_scale_with_map': +(layer.paint.get('circle-pitch-scale') === 'map'),\n 'u_matrix': painter.translatePosMatrix(\n coord.posMatrix,\n tile,\n layer.paint.get('circle-translate'),\n layer.paint.get('circle-translate-anchor')),\n 'u_pitch_with_map': +(pitchWithMap),\n 'u_device_pixel_ratio': painter.pixelRatio,\n 'u_extrude_scale': extrudeScale\n };\n};\n\nexport {circleUniforms, circleUniformValues};\n","import {Uniform1f, Uniform2f, UniformMatrix4f} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Transform} from '../../geo/transform';\nimport type {Tile} from '../../source/tile';\nimport {mat4} from 'gl-matrix';\n\nexport type CollisionUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_pixels_to_tile_units': Uniform1f;\n 'u_extrude_scale': Uniform2f;\n 'u_overscale_factor': Uniform1f;\n};\n\nexport type CollisionCircleUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_inv_matrix': UniformMatrix4f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_viewport_size': Uniform2f;\n};\n\nconst collisionUniforms = (context: Context, locations: UniformLocations): CollisionUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_pixels_to_tile_units': new Uniform1f(context, locations.u_pixels_to_tile_units),\n 'u_extrude_scale': new Uniform2f(context, locations.u_extrude_scale),\n 'u_overscale_factor': new Uniform1f(context, locations.u_overscale_factor)\n});\n\nconst collisionCircleUniforms = (context: Context, locations: UniformLocations): CollisionCircleUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_inv_matrix': new UniformMatrix4f(context, locations.u_inv_matrix),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_viewport_size': new Uniform2f(context, locations.u_viewport_size)\n});\n\nconst collisionUniformValues = (matrix: mat4, transform: Transform, tile: Tile): UniformValues => {\n const pixelRatio = pixelsToTileUnits(tile, 1, transform.zoom);\n const scale = Math.pow(2, transform.zoom - tile.tileID.overscaledZ);\n const overscaleFactor = tile.tileID.overscaleFactor();\n return {\n 'u_matrix': matrix,\n 'u_camera_to_center_distance': transform.cameraToCenterDistance,\n 'u_pixels_to_tile_units': pixelRatio,\n 'u_extrude_scale': [transform.pixelsToGLUnits[0] / (pixelRatio * scale),\n transform.pixelsToGLUnits[1] / (pixelRatio * scale)],\n 'u_overscale_factor': overscaleFactor\n };\n};\n\nconst collisionCircleUniformValues = (matrix: mat4, invMatrix: mat4, transform: Transform): UniformValues => {\n return {\n 'u_matrix': matrix,\n 'u_inv_matrix': invMatrix,\n 'u_camera_to_center_distance': transform.cameraToCenterDistance,\n 'u_viewport_size': [transform.width, transform.height]\n };\n};\n\nexport {collisionUniforms, collisionUniformValues, collisionCircleUniforms, collisionCircleUniformValues};\n","import {UniformColor, UniformMatrix4f, Uniform1i, Uniform1f} from '../uniform_binding';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {mat4} from 'gl-matrix';\n\nexport type DebugUniformsType = {\n 'u_color': UniformColor;\n 'u_matrix': UniformMatrix4f;\n 'u_overlay': Uniform1i;\n 'u_overlay_scale': Uniform1f;\n};\n\nconst debugUniforms = (context: Context, locations: UniformLocations): DebugUniformsType => ({\n 'u_color': new UniformColor(context, locations.u_color),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_overlay': new Uniform1i(context, locations.u_overlay),\n 'u_overlay_scale': new Uniform1f(context, locations.u_overlay_scale)\n});\n\nconst debugUniformValues = (matrix: mat4, color: Color, scaleRatio: number = 1): UniformValues => ({\n 'u_matrix': matrix,\n 'u_color': color,\n 'u_overlay': 0,\n 'u_overlay_scale': scaleRatio\n});\n\nexport {debugUniforms, debugUniformValues};\n","import {UniformMatrix4f} from '../uniform_binding';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport {mat4} from 'gl-matrix';\n\nexport type ClippingMaskUniformsType = {\n 'u_matrix': UniformMatrix4f;\n};\n\nconst clippingMaskUniforms = (context: Context, locations: UniformLocations): ClippingMaskUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix)\n});\n\nconst clippingMaskUniformValues = (matrix: mat4): UniformValues => ({\n 'u_matrix': matrix\n});\n\nexport {clippingMaskUniforms, clippingMaskUniformValues};\n","import {mat4} from 'gl-matrix';\n\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n UniformMatrix4f\n} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\n\nimport type {Context} from '../../gl/context';\nimport type {Tile} from '../../source/tile';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Painter} from '../painter';\nimport type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer';\n\nexport type HeatmapUniformsType = {\n 'u_extrude_scale': Uniform1f;\n 'u_intensity': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n};\n\nexport type HeatmapTextureUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_world': Uniform2f;\n 'u_image': Uniform1i;\n 'u_color_ramp': Uniform1i;\n 'u_opacity': Uniform1f;\n};\n\nconst heatmapUniforms = (context: Context, locations: UniformLocations): HeatmapUniformsType => ({\n 'u_extrude_scale': new Uniform1f(context, locations.u_extrude_scale),\n 'u_intensity': new Uniform1f(context, locations.u_intensity),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix)\n});\n\nconst heatmapTextureUniforms = (context: Context, locations: UniformLocations): HeatmapTextureUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_world': new Uniform2f(context, locations.u_world),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_color_ramp': new Uniform1i(context, locations.u_color_ramp),\n 'u_opacity': new Uniform1f(context, locations.u_opacity)\n});\n\nconst heatmapUniformValues = (matrix: mat4, tile: Tile, zoom: number, intensity: number): UniformValues => ({\n 'u_matrix': matrix,\n 'u_extrude_scale': pixelsToTileUnits(tile, 1, zoom),\n 'u_intensity': intensity\n});\n\nconst heatmapTextureUniformValues = (\n painter: Painter,\n layer: HeatmapStyleLayer,\n textureUnit: number,\n colorRampUnit: number\n): UniformValues => {\n const matrix = mat4.create();\n mat4.ortho(matrix, 0, painter.width, painter.height, 0, 0, 1);\n\n const gl = painter.context.gl;\n\n return {\n 'u_matrix': matrix,\n 'u_world': [gl.drawingBufferWidth, gl.drawingBufferHeight],\n 'u_image': textureUnit,\n 'u_color_ramp': colorRampUnit,\n 'u_opacity': layer.paint.get('heatmap-opacity')\n };\n};\n\nexport {\n heatmapUniforms,\n heatmapTextureUniforms,\n heatmapUniformValues,\n heatmapTextureUniformValues\n};\n","import {mat4} from 'gl-matrix';\n\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n UniformColor,\n UniformMatrix4f,\n Uniform4f\n} from '../uniform_binding';\nimport {EXTENT} from '../../data/extent';\nimport {MercatorCoordinate} from '../../geo/mercator_coordinate';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Tile} from '../../source/tile';\nimport type {Painter} from '../painter';\nimport type {HillshadeStyleLayer} from '../../style/style_layer/hillshade_style_layer';\nimport type {DEMData} from '../../data/dem_data';\nimport type {OverscaledTileID} from '../../source/tile_id';\n\nexport type HillshadeUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_image': Uniform1i;\n 'u_latrange': Uniform2f;\n 'u_light': Uniform2f;\n 'u_shadow': UniformColor;\n 'u_highlight': UniformColor;\n 'u_accent': UniformColor;\n};\n\nexport type HillshadePrepareUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_image': Uniform1i;\n 'u_dimension': Uniform2f;\n 'u_zoom': Uniform1f;\n 'u_unpack': Uniform4f;\n};\n\nconst hillshadeUniforms = (context: Context, locations: UniformLocations): HillshadeUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_latrange': new Uniform2f(context, locations.u_latrange),\n 'u_light': new Uniform2f(context, locations.u_light),\n 'u_shadow': new UniformColor(context, locations.u_shadow),\n 'u_highlight': new UniformColor(context, locations.u_highlight),\n 'u_accent': new UniformColor(context, locations.u_accent)\n});\n\nconst hillshadePrepareUniforms = (context: Context, locations: UniformLocations): HillshadePrepareUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_dimension': new Uniform2f(context, locations.u_dimension),\n 'u_zoom': new Uniform1f(context, locations.u_zoom),\n 'u_unpack': new Uniform4f(context, locations.u_unpack)\n});\n\nconst hillshadeUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: HillshadeStyleLayer,\n coord: OverscaledTileID\n): UniformValues => {\n const shadow = layer.paint.get('hillshade-shadow-color');\n const highlight = layer.paint.get('hillshade-highlight-color');\n const accent = layer.paint.get('hillshade-accent-color');\n\n let azimuthal = layer.paint.get('hillshade-illumination-direction') * (Math.PI / 180);\n // modify azimuthal angle by map rotation if light is anchored at the viewport\n if (layer.paint.get('hillshade-illumination-anchor') === 'viewport') {\n azimuthal -= painter.transform.angle;\n }\n const align = !painter.options.moving;\n return {\n 'u_matrix': coord ? coord.posMatrix : painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped(), align),\n 'u_image': 0,\n 'u_latrange': getTileLatRange(painter, tile.tileID),\n 'u_light': [layer.paint.get('hillshade-exaggeration'), azimuthal],\n 'u_shadow': shadow,\n 'u_highlight': highlight,\n 'u_accent': accent\n };\n};\n\nconst hillshadeUniformPrepareValues = (tileID: OverscaledTileID, dem: DEMData): UniformValues => {\n\n const stride = dem.stride;\n const matrix = mat4.create();\n // Flip rendering at y axis.\n mat4.ortho(matrix, 0, EXTENT, -EXTENT, 0, 0, 1);\n mat4.translate(matrix, matrix, [0, -EXTENT, 0]);\n\n return {\n 'u_matrix': matrix,\n 'u_image': 1,\n 'u_dimension': [stride, stride],\n 'u_zoom': tileID.overscaledZ,\n 'u_unpack': dem.getUnpackVector()\n };\n};\n\nfunction getTileLatRange(painter: Painter, tileID: OverscaledTileID) {\n // for scaling the magnitude of a points slope by its latitude\n const tilesAtZoom = Math.pow(2, tileID.canonical.z);\n const y = tileID.canonical.y;\n return [\n new MercatorCoordinate(0, y / tilesAtZoom).toLngLat().lat,\n new MercatorCoordinate(0, (y + 1) / tilesAtZoom).toLngLat().lat];\n}\n\nexport {\n hillshadeUniforms,\n hillshadePrepareUniforms,\n hillshadeUniformValues,\n hillshadeUniformPrepareValues\n};\n","import {Uniform1i, Uniform1f, Uniform2f, Uniform3f, UniformMatrix4f} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\nimport {extend} from '../../util/util';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Transform} from '../../geo/transform';\nimport type {Tile} from '../../source/tile';\nimport type {CrossFaded} from '../../style/properties';\nimport type {LineStyleLayer} from '../../style/style_layer/line_style_layer';\nimport type {Painter} from '../painter';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport {OverscaledTileID} from '../../source/tile_id';\n\nexport type LineUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_ratio': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_units_to_pixels': Uniform2f;\n};\n\nexport type LineGradientUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_ratio': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_units_to_pixels': Uniform2f;\n 'u_image': Uniform1i;\n 'u_image_height': Uniform1f;\n};\n\nexport type LinePatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_texsize': Uniform2f;\n 'u_ratio': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_units_to_pixels': Uniform2f;\n 'u_image': Uniform1i;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n};\n\nexport type LineSDFUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_ratio': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_units_to_pixels': Uniform2f;\n 'u_patternscale_a': Uniform2f;\n 'u_patternscale_b': Uniform2f;\n 'u_sdfgamma': Uniform1f;\n 'u_image': Uniform1i;\n 'u_tex_y_a': Uniform1f;\n 'u_tex_y_b': Uniform1f;\n 'u_mix': Uniform1f;\n};\n\nconst lineUniforms = (context: Context, locations: UniformLocations): LineUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_ratio': new Uniform1f(context, locations.u_ratio),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels)\n});\n\nconst lineGradientUniforms = (context: Context, locations: UniformLocations): LineGradientUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_ratio': new Uniform1f(context, locations.u_ratio),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_image_height': new Uniform1f(context, locations.u_image_height)\n});\n\nconst linePatternUniforms = (context: Context, locations: UniformLocations): LinePatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_ratio': new Uniform1f(context, locations.u_ratio),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels),\n 'u_scale': new Uniform3f(context, locations.u_scale),\n 'u_fade': new Uniform1f(context, locations.u_fade)\n});\n\nconst lineSDFUniforms = (context: Context, locations: UniformLocations): LineSDFUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_ratio': new Uniform1f(context, locations.u_ratio),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels),\n 'u_patternscale_a': new Uniform2f(context, locations.u_patternscale_a),\n 'u_patternscale_b': new Uniform2f(context, locations.u_patternscale_b),\n 'u_sdfgamma': new Uniform1f(context, locations.u_sdfgamma),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_tex_y_a': new Uniform1f(context, locations.u_tex_y_a),\n 'u_tex_y_b': new Uniform1f(context, locations.u_tex_y_b),\n 'u_mix': new Uniform1f(context, locations.u_mix)\n});\n\nconst lineUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: LineStyleLayer,\n coord: OverscaledTileID\n): UniformValues => {\n const transform = painter.transform;\n\n return {\n 'u_matrix': calculateMatrix(painter, tile, layer, coord),\n 'u_ratio': 1 / pixelsToTileUnits(tile, 1, transform.zoom),\n 'u_device_pixel_ratio': painter.pixelRatio,\n 'u_units_to_pixels': [\n 1 / transform.pixelsToGLUnits[0],\n 1 / transform.pixelsToGLUnits[1]\n ]\n };\n};\n\nconst lineGradientUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: LineStyleLayer,\n imageHeight: number,\n coord: OverscaledTileID\n): UniformValues => {\n return extend(lineUniformValues(painter, tile, layer, coord), {\n 'u_image': 0,\n 'u_image_height': imageHeight,\n });\n};\n\nconst linePatternUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: LineStyleLayer,\n crossfade: CrossfadeParameters,\n coord: OverscaledTileID\n): UniformValues => {\n const transform = painter.transform;\n const tileZoomRatio = calculateTileRatio(tile, transform);\n return {\n 'u_matrix': calculateMatrix(painter, tile, layer, coord),\n 'u_texsize': tile.imageAtlasTexture.size,\n // camera zoom ratio\n 'u_ratio': 1 / pixelsToTileUnits(tile, 1, transform.zoom),\n 'u_device_pixel_ratio': painter.pixelRatio,\n 'u_image': 0,\n 'u_scale': [tileZoomRatio, crossfade.fromScale, crossfade.toScale],\n 'u_fade': crossfade.t,\n 'u_units_to_pixels': [\n 1 / transform.pixelsToGLUnits[0],\n 1 / transform.pixelsToGLUnits[1]\n ]\n };\n};\n\nconst lineSDFUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: LineStyleLayer,\n dasharray: CrossFaded>,\n crossfade: CrossfadeParameters,\n coord: OverscaledTileID\n): UniformValues => {\n const transform = painter.transform;\n const lineAtlas = painter.lineAtlas;\n const tileRatio = calculateTileRatio(tile, transform);\n\n const round = layer.layout.get('line-cap') === 'round';\n\n const posA = lineAtlas.getDash(dasharray.from, round);\n const posB = lineAtlas.getDash(dasharray.to, round);\n\n const widthA = posA.width * crossfade.fromScale;\n const widthB = posB.width * crossfade.toScale;\n\n return extend(lineUniformValues(painter, tile, layer, coord), {\n 'u_patternscale_a': [tileRatio / widthA, -posA.height / 2],\n 'u_patternscale_b': [tileRatio / widthB, -posB.height / 2],\n 'u_sdfgamma': lineAtlas.width / (Math.min(widthA, widthB) * 256 * painter.pixelRatio) / 2,\n 'u_image': 0,\n 'u_tex_y_a': posA.y,\n 'u_tex_y_b': posB.y,\n 'u_mix': crossfade.t\n });\n};\n\nfunction calculateTileRatio(tile: Tile, transform: Transform) {\n return 1 / pixelsToTileUnits(tile, 1, transform.tileZoom);\n}\n\nfunction calculateMatrix(painter: Painter, tile: Tile, layer: LineStyleLayer, coord: OverscaledTileID) {\n return painter.translatePosMatrix(\n coord ? coord.posMatrix : tile.tileID.posMatrix,\n tile,\n layer.paint.get('line-translate'),\n layer.paint.get('line-translate-anchor')\n );\n}\n\nexport {\n lineUniforms,\n lineGradientUniforms,\n linePatternUniforms,\n lineSDFUniforms,\n lineUniformValues,\n lineGradientUniformValues,\n linePatternUniformValues,\n lineSDFUniformValues\n};\n","import {Uniform1i, Uniform1f, Uniform2f, Uniform3f, UniformMatrix4f} from '../uniform_binding';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {RasterStyleLayer} from '../../style/style_layer/raster_style_layer';\nimport {mat4} from 'gl-matrix';\n\nexport type RasterUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_tl_parent': Uniform2f;\n 'u_scale_parent': Uniform1f;\n 'u_buffer_scale': Uniform1f;\n 'u_fade_t': Uniform1f;\n 'u_opacity': Uniform1f;\n 'u_image0': Uniform1i;\n 'u_image1': Uniform1i;\n 'u_brightness_low': Uniform1f;\n 'u_brightness_high': Uniform1f;\n 'u_saturation_factor': Uniform1f;\n 'u_contrast_factor': Uniform1f;\n 'u_spin_weights': Uniform3f;\n};\n\nconst rasterUniforms = (context: Context, locations: UniformLocations): RasterUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_tl_parent': new Uniform2f(context, locations.u_tl_parent),\n 'u_scale_parent': new Uniform1f(context, locations.u_scale_parent),\n 'u_buffer_scale': new Uniform1f(context, locations.u_buffer_scale),\n 'u_fade_t': new Uniform1f(context, locations.u_fade_t),\n 'u_opacity': new Uniform1f(context, locations.u_opacity),\n 'u_image0': new Uniform1i(context, locations.u_image0),\n 'u_image1': new Uniform1i(context, locations.u_image1),\n 'u_brightness_low': new Uniform1f(context, locations.u_brightness_low),\n 'u_brightness_high': new Uniform1f(context, locations.u_brightness_high),\n 'u_saturation_factor': new Uniform1f(context, locations.u_saturation_factor),\n 'u_contrast_factor': new Uniform1f(context, locations.u_contrast_factor),\n 'u_spin_weights': new Uniform3f(context, locations.u_spin_weights)\n});\n\nconst rasterUniformValues = (\n matrix: mat4,\n parentTL: [number, number],\n parentScaleBy: number,\n fade: {\n mix: number;\n opacity: number;\n },\n layer: RasterStyleLayer\n): UniformValues => ({\n 'u_matrix': matrix,\n 'u_tl_parent': parentTL,\n 'u_scale_parent': parentScaleBy,\n 'u_buffer_scale': 1,\n 'u_fade_t': fade.mix,\n 'u_opacity': fade.opacity * layer.paint.get('raster-opacity'),\n 'u_image0': 0,\n 'u_image1': 1,\n 'u_brightness_low': layer.paint.get('raster-brightness-min'),\n 'u_brightness_high': layer.paint.get('raster-brightness-max'),\n 'u_saturation_factor': saturationFactor(layer.paint.get('raster-saturation')),\n 'u_contrast_factor': contrastFactor(layer.paint.get('raster-contrast')),\n 'u_spin_weights': spinWeights(layer.paint.get('raster-hue-rotate'))\n});\n\nfunction spinWeights(angle) {\n angle *= Math.PI / 180;\n const s = Math.sin(angle);\n const c = Math.cos(angle);\n return [\n (2 * c + 1) / 3,\n (-Math.sqrt(3) * s - c + 1) / 3,\n (Math.sqrt(3) * s - c + 1) / 3\n ];\n}\n\nfunction contrastFactor(contrast) {\n return contrast > 0 ?\n 1 / (1 - contrast) :\n 1 + contrast;\n}\n\nfunction saturationFactor(saturation) {\n return saturation > 0 ?\n 1 - 1 / (1.001 - saturation) :\n -saturation;\n}\n\nexport {rasterUniforms, rasterUniformValues};\n","import {Uniform1i, Uniform1f, Uniform2f, UniformMatrix4f} from '../uniform_binding';\nimport {extend} from '../../util/util';\n\nimport type {Context} from '../../gl/context';\nimport type {Painter} from '../painter';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport {mat4} from 'gl-matrix';\n\nexport type SymbolIconUniformsType = {\n 'u_is_size_zoom_constant': Uniform1i;\n 'u_is_size_feature_constant': Uniform1i;\n 'u_size_t': Uniform1f;\n 'u_size': Uniform1f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_pitch': Uniform1f;\n 'u_rotate_symbol': Uniform1i;\n 'u_aspect_ratio': Uniform1f;\n 'u_fade_change': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n 'u_label_plane_matrix': UniformMatrix4f;\n 'u_coord_matrix': UniformMatrix4f;\n 'u_is_text': Uniform1i;\n 'u_pitch_with_map': Uniform1i;\n 'u_texsize': Uniform2f;\n 'u_texture': Uniform1i;\n};\n\nexport type SymbolSDFUniformsType = {\n 'u_is_size_zoom_constant': Uniform1i;\n 'u_is_size_feature_constant': Uniform1i;\n 'u_size_t': Uniform1f;\n 'u_size': Uniform1f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_pitch': Uniform1f;\n 'u_rotate_symbol': Uniform1i;\n 'u_aspect_ratio': Uniform1f;\n 'u_fade_change': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n 'u_label_plane_matrix': UniformMatrix4f;\n 'u_coord_matrix': UniformMatrix4f;\n 'u_is_text': Uniform1i;\n 'u_pitch_with_map': Uniform1i;\n 'u_texsize': Uniform2f;\n 'u_texture': Uniform1i;\n 'u_gamma_scale': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_is_halo': Uniform1i;\n};\n\nexport type symbolTextAndIconUniformsType = {\n 'u_is_size_zoom_constant': Uniform1i;\n 'u_is_size_feature_constant': Uniform1i;\n 'u_size_t': Uniform1f;\n 'u_size': Uniform1f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_pitch': Uniform1f;\n 'u_rotate_symbol': Uniform1i;\n 'u_aspect_ratio': Uniform1f;\n 'u_fade_change': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n 'u_label_plane_matrix': UniformMatrix4f;\n 'u_coord_matrix': UniformMatrix4f;\n 'u_is_text': Uniform1i;\n 'u_pitch_with_map': Uniform1i;\n 'u_texsize': Uniform2f;\n 'u_texsize_icon': Uniform2f;\n 'u_texture': Uniform1i;\n 'u_texture_icon': Uniform1i;\n 'u_gamma_scale': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_is_halo': Uniform1i;\n};\n\nconst symbolIconUniforms = (context: Context, locations: UniformLocations): SymbolIconUniformsType => ({\n 'u_is_size_zoom_constant': new Uniform1i(context, locations.u_is_size_zoom_constant),\n 'u_is_size_feature_constant': new Uniform1i(context, locations.u_is_size_feature_constant),\n 'u_size_t': new Uniform1f(context, locations.u_size_t),\n 'u_size': new Uniform1f(context, locations.u_size),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_pitch': new Uniform1f(context, locations.u_pitch),\n 'u_rotate_symbol': new Uniform1i(context, locations.u_rotate_symbol),\n 'u_aspect_ratio': new Uniform1f(context, locations.u_aspect_ratio),\n 'u_fade_change': new Uniform1f(context, locations.u_fade_change),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_label_plane_matrix': new UniformMatrix4f(context, locations.u_label_plane_matrix),\n 'u_coord_matrix': new UniformMatrix4f(context, locations.u_coord_matrix),\n 'u_is_text': new Uniform1i(context, locations.u_is_text),\n 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_texture': new Uniform1i(context, locations.u_texture)\n});\n\nconst symbolSDFUniforms = (context: Context, locations: UniformLocations): SymbolSDFUniformsType => ({\n 'u_is_size_zoom_constant': new Uniform1i(context, locations.u_is_size_zoom_constant),\n 'u_is_size_feature_constant': new Uniform1i(context, locations.u_is_size_feature_constant),\n 'u_size_t': new Uniform1f(context, locations.u_size_t),\n 'u_size': new Uniform1f(context, locations.u_size),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_pitch': new Uniform1f(context, locations.u_pitch),\n 'u_rotate_symbol': new Uniform1i(context, locations.u_rotate_symbol),\n 'u_aspect_ratio': new Uniform1f(context, locations.u_aspect_ratio),\n 'u_fade_change': new Uniform1f(context, locations.u_fade_change),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_label_plane_matrix': new UniformMatrix4f(context, locations.u_label_plane_matrix),\n 'u_coord_matrix': new UniformMatrix4f(context, locations.u_coord_matrix),\n 'u_is_text': new Uniform1i(context, locations.u_is_text),\n 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_texture': new Uniform1i(context, locations.u_texture),\n 'u_gamma_scale': new Uniform1f(context, locations.u_gamma_scale),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_is_halo': new Uniform1i(context, locations.u_is_halo)\n});\n\nconst symbolTextAndIconUniforms = (context: Context, locations: UniformLocations): symbolTextAndIconUniformsType => ({\n 'u_is_size_zoom_constant': new Uniform1i(context, locations.u_is_size_zoom_constant),\n 'u_is_size_feature_constant': new Uniform1i(context, locations.u_is_size_feature_constant),\n 'u_size_t': new Uniform1f(context, locations.u_size_t),\n 'u_size': new Uniform1f(context, locations.u_size),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_pitch': new Uniform1f(context, locations.u_pitch),\n 'u_rotate_symbol': new Uniform1i(context, locations.u_rotate_symbol),\n 'u_aspect_ratio': new Uniform1f(context, locations.u_aspect_ratio),\n 'u_fade_change': new Uniform1f(context, locations.u_fade_change),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_label_plane_matrix': new UniformMatrix4f(context, locations.u_label_plane_matrix),\n 'u_coord_matrix': new UniformMatrix4f(context, locations.u_coord_matrix),\n 'u_is_text': new Uniform1i(context, locations.u_is_text),\n 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_texsize_icon': new Uniform2f(context, locations.u_texsize_icon),\n 'u_texture': new Uniform1i(context, locations.u_texture),\n 'u_texture_icon': new Uniform1i(context, locations.u_texture_icon),\n 'u_gamma_scale': new Uniform1f(context, locations.u_gamma_scale),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_is_halo': new Uniform1i(context, locations.u_is_halo)\n});\n\nconst symbolIconUniformValues = (\n functionType: string,\n size: {\n uSizeT: number;\n uSize: number;\n },\n rotateInShader: boolean,\n pitchWithMap: boolean,\n painter: Painter,\n matrix: mat4,\n labelPlaneMatrix: mat4,\n glCoordMatrix: mat4,\n isText: boolean,\n texSize: [number, number]\n): UniformValues => {\n const transform = painter.transform;\n\n return {\n 'u_is_size_zoom_constant': +(functionType === 'constant' || functionType === 'source'),\n 'u_is_size_feature_constant': +(functionType === 'constant' || functionType === 'camera'),\n 'u_size_t': size ? size.uSizeT : 0,\n 'u_size': size ? size.uSize : 0,\n 'u_camera_to_center_distance': transform.cameraToCenterDistance,\n 'u_pitch': transform.pitch / 360 * 2 * Math.PI,\n 'u_rotate_symbol': +rotateInShader,\n 'u_aspect_ratio': transform.width / transform.height,\n 'u_fade_change': painter.options.fadeDuration ? painter.symbolFadeChange : 1,\n 'u_matrix': matrix,\n 'u_label_plane_matrix': labelPlaneMatrix,\n 'u_coord_matrix': glCoordMatrix,\n 'u_is_text': +isText,\n 'u_pitch_with_map': +pitchWithMap,\n 'u_texsize': texSize,\n 'u_texture': 0\n };\n};\n\nconst symbolSDFUniformValues = (\n functionType: string,\n size: {\n uSizeT: number;\n uSize: number;\n },\n rotateInShader: boolean,\n pitchWithMap: boolean,\n painter: Painter,\n matrix: mat4,\n labelPlaneMatrix: mat4,\n glCoordMatrix: mat4,\n isText: boolean,\n texSize: [number, number],\n isHalo: boolean\n): UniformValues => {\n const transform = painter.transform;\n\n return extend(symbolIconUniformValues(functionType, size,\n rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix,\n glCoordMatrix, isText, texSize), {\n 'u_gamma_scale': (pitchWithMap ? Math.cos(transform._pitch) * transform.cameraToCenterDistance : 1),\n 'u_device_pixel_ratio': painter.pixelRatio,\n 'u_is_halo': +isHalo\n });\n};\n\nconst symbolTextAndIconUniformValues = (\n functionType: string,\n size: {\n uSizeT: number;\n uSize: number;\n },\n rotateInShader: boolean,\n pitchWithMap: boolean,\n painter: Painter,\n matrix: mat4,\n labelPlaneMatrix: mat4,\n glCoordMatrix: mat4,\n texSizeSDF: [number, number],\n texSizeIcon: [number, number]\n): UniformValues => {\n return extend(symbolSDFUniformValues(functionType, size,\n rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix,\n glCoordMatrix, true, texSizeSDF, true), {\n 'u_texsize_icon': texSizeIcon,\n 'u_texture_icon': 1\n });\n};\n\nexport {symbolIconUniforms, symbolSDFUniforms, symbolIconUniformValues, symbolSDFUniformValues, symbolTextAndIconUniformValues, symbolTextAndIconUniforms};\n","import {bgPatternUniformValues} from './pattern';\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n UniformColor,\n UniformMatrix4f\n} from '../uniform_binding';\nimport {extend} from '../../util/util';\n\nimport type {Painter} from '../painter';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Context} from '../../gl/context';\nimport type {Color, ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport type {CrossFaded} from '../../style/properties';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport type {OverscaledTileID} from '../../source/tile_id';\nimport {mat4} from 'gl-matrix';\n\nexport type BackgroundUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_opacity': Uniform1f;\n 'u_color': UniformColor;\n};\n\nexport type BackgroundPatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_opacity': Uniform1f;\n // pattern uniforms:\n 'u_image': Uniform1i;\n 'u_pattern_tl_a': Uniform2f;\n 'u_pattern_br_a': Uniform2f;\n 'u_pattern_tl_b': Uniform2f;\n 'u_pattern_br_b': Uniform2f;\n 'u_texsize': Uniform2f;\n 'u_mix': Uniform1f;\n 'u_pattern_size_a': Uniform2f;\n 'u_pattern_size_b': Uniform2f;\n 'u_scale_a': Uniform1f;\n 'u_scale_b': Uniform1f;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_tile_units_to_pixels': Uniform1f;\n};\n\nconst backgroundUniforms = (context: Context, locations: UniformLocations): BackgroundUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_opacity': new Uniform1f(context, locations.u_opacity),\n 'u_color': new UniformColor(context, locations.u_color)\n});\n\nconst backgroundPatternUniforms = (context: Context, locations: UniformLocations): BackgroundPatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_opacity': new Uniform1f(context, locations.u_opacity),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_pattern_tl_a': new Uniform2f(context, locations.u_pattern_tl_a),\n 'u_pattern_br_a': new Uniform2f(context, locations.u_pattern_br_a),\n 'u_pattern_tl_b': new Uniform2f(context, locations.u_pattern_tl_b),\n 'u_pattern_br_b': new Uniform2f(context, locations.u_pattern_br_b),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_mix': new Uniform1f(context, locations.u_mix),\n 'u_pattern_size_a': new Uniform2f(context, locations.u_pattern_size_a),\n 'u_pattern_size_b': new Uniform2f(context, locations.u_pattern_size_b),\n 'u_scale_a': new Uniform1f(context, locations.u_scale_a),\n 'u_scale_b': new Uniform1f(context, locations.u_scale_b),\n 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper),\n 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower),\n 'u_tile_units_to_pixels': new Uniform1f(context, locations.u_tile_units_to_pixels)\n});\n\nconst backgroundUniformValues = (matrix: mat4, opacity: number, color: Color): UniformValues => ({\n 'u_matrix': matrix,\n 'u_opacity': opacity,\n 'u_color': color\n});\n\nconst backgroundPatternUniformValues = (\n matrix: mat4,\n opacity: number,\n painter: Painter,\n image: CrossFaded,\n tile: {\n tileID: OverscaledTileID;\n tileSize: number;\n },\n crossfade: CrossfadeParameters\n): UniformValues => extend(\n bgPatternUniformValues(image, crossfade, painter, tile),\n {\n 'u_matrix': matrix,\n 'u_opacity': opacity\n }\n);\n\nexport {\n backgroundUniforms,\n backgroundPatternUniforms,\n backgroundUniformValues,\n backgroundPatternUniformValues\n};\n","import {fillExtrusionUniforms, fillExtrusionPatternUniforms} from './fill_extrusion_program';\nimport {fillUniforms, fillPatternUniforms, fillOutlineUniforms, fillOutlinePatternUniforms} from './fill_program';\nimport {circleUniforms} from './circle_program';\nimport {collisionUniforms, collisionCircleUniforms} from './collision_program';\nimport {debugUniforms} from './debug_program';\nimport {clippingMaskUniforms} from './clipping_mask_program';\nimport {heatmapUniforms, heatmapTextureUniforms} from './heatmap_program';\nimport {hillshadeUniforms, hillshadePrepareUniforms} from './hillshade_program';\nimport {lineUniforms, lineGradientUniforms, linePatternUniforms, lineSDFUniforms} from './line_program';\nimport {rasterUniforms} from './raster_program';\nimport {symbolIconUniforms, symbolSDFUniforms, symbolTextAndIconUniforms} from './symbol_program';\nimport {backgroundUniforms, backgroundPatternUniforms} from './background_program';\nimport {terrainUniforms, terrainDepthUniforms, terrainCoordsUniforms} from './terrain_program';\n\nexport const programUniforms = {\n fillExtrusion: fillExtrusionUniforms,\n fillExtrusionPattern: fillExtrusionPatternUniforms,\n fill: fillUniforms,\n fillPattern: fillPatternUniforms,\n fillOutline: fillOutlineUniforms,\n fillOutlinePattern: fillOutlinePatternUniforms,\n circle: circleUniforms,\n collisionBox: collisionUniforms,\n collisionCircle: collisionCircleUniforms,\n debug: debugUniforms,\n clippingMask: clippingMaskUniforms,\n heatmap: heatmapUniforms,\n heatmapTexture: heatmapTextureUniforms,\n hillshade: hillshadeUniforms,\n hillshadePrepare: hillshadePrepareUniforms,\n line: lineUniforms,\n lineGradient: lineGradientUniforms,\n linePattern: linePatternUniforms,\n lineSDF: lineSDFUniforms,\n raster: rasterUniforms,\n symbolIcon: symbolIconUniforms,\n symbolSDF: symbolSDFUniforms,\n symbolTextAndIcon: symbolTextAndIconUniforms,\n background: backgroundUniforms,\n backgroundPattern: backgroundPatternUniforms,\n terrain: terrainUniforms,\n terrainDepth: terrainDepthUniforms,\n terrainCoords: terrainCoordsUniforms\n};\n","\nimport type {StructArray} from '../util/struct_array';\nimport type {TriangleIndexArray, LineIndexArray, LineStripIndexArray} from '../data/index_array_type';\nimport type {Context} from '../gl/context';\n\n/**\n * @internal\n * an index buffer class\n */\nexport class IndexBuffer {\n context: Context;\n buffer: WebGLBuffer;\n dynamicDraw: boolean;\n\n constructor(context: Context, array: TriangleIndexArray | LineIndexArray | LineStripIndexArray, dynamicDraw?: boolean) {\n this.context = context;\n const gl = context.gl;\n this.buffer = gl.createBuffer();\n this.dynamicDraw = Boolean(dynamicDraw);\n\n // The bound index buffer is part of vertex array object state. We don't want to\n // modify whatever VAO happens to be currently bound, so make sure the default\n // vertex array provided by the context is bound instead.\n this.context.unbindVAO();\n\n context.bindElementBuffer.set(this.buffer);\n gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, array.arrayBuffer, this.dynamicDraw ? gl.DYNAMIC_DRAW : gl.STATIC_DRAW);\n\n if (!this.dynamicDraw) {\n delete array.arrayBuffer;\n }\n }\n\n bind() {\n this.context.bindElementBuffer.set(this.buffer);\n }\n\n updateData(array: StructArray) {\n const gl = this.context.gl;\n if (!this.dynamicDraw) throw new Error('Attempted to update data while not in dynamic mode.');\n // The right VAO will get this buffer re-bound later in VertexArrayObject#bind\n // See https://github.com/mapbox/mapbox-gl-js/issues/5620\n this.context.unbindVAO();\n this.bind();\n gl.bufferSubData(gl.ELEMENT_ARRAY_BUFFER, 0, array.arrayBuffer);\n }\n\n destroy() {\n const gl = this.context.gl;\n if (this.buffer) {\n gl.deleteBuffer(this.buffer);\n delete this.buffer;\n }\n }\n}\n","\nimport type {\n StructArray,\n StructArrayMember\n} from '../util/struct_array';\n\nimport type {Program} from '../render/program';\nimport type {Context} from '../gl/context';\n\n/**\n * An Enum for AttributeType\n */\nconst AttributeType = {\n Int8: 'BYTE',\n Uint8: 'UNSIGNED_BYTE',\n Int16: 'SHORT',\n Uint16: 'UNSIGNED_SHORT',\n Int32: 'INT',\n Uint32: 'UNSIGNED_INT',\n Float32: 'FLOAT'\n};\n\n/**\n * @internal\n * The `VertexBuffer` class turns a `StructArray` into a WebGL buffer. Each member of the StructArray's\n * Struct type is converted to a WebGL attribute.\n */\nexport class VertexBuffer {\n length: number;\n attributes: ReadonlyArray;\n itemSize: number;\n dynamicDraw: boolean;\n context: Context;\n buffer: WebGLBuffer;\n\n /**\n * @param dynamicDraw - Whether this buffer will be repeatedly updated.\n */\n constructor(context: Context, array: StructArray, attributes: ReadonlyArray, dynamicDraw?: boolean) {\n this.length = array.length;\n this.attributes = attributes;\n this.itemSize = array.bytesPerElement;\n this.dynamicDraw = dynamicDraw;\n\n this.context = context;\n const gl = context.gl;\n this.buffer = gl.createBuffer();\n context.bindVertexBuffer.set(this.buffer);\n gl.bufferData(gl.ARRAY_BUFFER, array.arrayBuffer, this.dynamicDraw ? gl.DYNAMIC_DRAW : gl.STATIC_DRAW);\n\n if (!this.dynamicDraw) {\n delete array.arrayBuffer;\n }\n }\n\n bind() {\n this.context.bindVertexBuffer.set(this.buffer);\n }\n\n updateData(array: StructArray) {\n if (array.length !== this.length) throw new Error(`Length of new data is ${array.length}, which doesn't match current length of ${this.length}`);\n const gl = this.context.gl;\n this.bind();\n gl.bufferSubData(gl.ARRAY_BUFFER, 0, array.arrayBuffer);\n }\n\n enableAttributes(gl: WebGLRenderingContext|WebGL2RenderingContext, program: Program) {\n for (let j = 0; j < this.attributes.length; j++) {\n const member = this.attributes[j];\n const attribIndex: number | void = program.attributes[member.name];\n if (attribIndex !== undefined) {\n gl.enableVertexAttribArray(attribIndex);\n }\n }\n }\n\n /**\n * Set the attribute pointers in a WebGL context\n * @param gl - The WebGL context\n * @param program - The active WebGL program\n * @param vertexOffset - Index of the starting vertex of the segment\n */\n setVertexAttribPointers(gl: WebGLRenderingContext|WebGL2RenderingContext, program: Program, vertexOffset?: number | null) {\n for (let j = 0; j < this.attributes.length; j++) {\n const member = this.attributes[j];\n const attribIndex: number | void = program.attributes[member.name];\n\n if (attribIndex !== undefined) {\n gl.vertexAttribPointer(\n attribIndex,\n member.components,\n (gl as any)[AttributeType[member.type]],\n false,\n this.itemSize,\n member.offset + (this.itemSize * (vertexOffset || 0))\n );\n }\n }\n }\n\n /**\n * Destroy the GL buffer bound to the given WebGL context\n */\n destroy() {\n const gl = this.context.gl;\n if (this.buffer) {\n gl.deleteBuffer(this.buffer);\n delete this.buffer;\n }\n }\n}\n","const cache = new WeakMap();\nexport function isWebGL2(\n gl: WebGLRenderingContext | WebGL2RenderingContext\n): gl is WebGL2RenderingContext {\n if (cache.has(gl)) {\n return cache.get(gl);\n } else {\n const value = gl.getParameter(gl.VERSION)?.startsWith('WebGL 2.0');\n cache.set(gl, value);\n return value;\n }\n}\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {isWebGL2} from './webgl2';\n\nimport type {Context} from './context';\nimport type {\n BlendFuncType,\n BlendEquationType,\n ColorMaskType,\n DepthRangeType,\n DepthMaskType,\n StencilFuncType,\n StencilOpType,\n DepthFuncType,\n TextureUnitType,\n ViewportType,\n CullFaceModeType,\n FrontFaceType,\n} from './types';\n\nexport interface IValue {\n current: T;\n default: T;\n dirty: boolean;\n get(): T;\n setDefault(): void;\n set(value: T): void;\n}\n\nclass BaseValue implements IValue {\n gl: WebGLRenderingContext|WebGL2RenderingContext;\n current: T;\n default: T;\n dirty: boolean;\n\n constructor(context: Context) {\n this.gl = context.gl;\n this.default = this.getDefault();\n this.current = this.default;\n this.dirty = false;\n }\n\n get(): T {\n return this.current;\n }\n set(value: T) { // eslint-disable-line\n // overridden in child classes;\n }\n\n getDefault(): T {\n return this.default; // overriden in child classes\n }\n setDefault() {\n this.set(this.default);\n }\n}\n\nexport class ClearColor extends BaseValue {\n getDefault(): Color {\n return Color.transparent;\n }\n set(v: Color) {\n const c = this.current;\n if (v.r === c.r && v.g === c.g && v.b === c.b && v.a === c.a && !this.dirty) return;\n this.gl.clearColor(v.r, v.g, v.b, v.a);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ClearDepth extends BaseValue {\n getDefault(): number {\n return 1;\n }\n set(v: number) {\n if (v === this.current && !this.dirty) return;\n this.gl.clearDepth(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ClearStencil extends BaseValue {\n getDefault(): number {\n return 0;\n }\n set(v: number) {\n if (v === this.current && !this.dirty) return;\n this.gl.clearStencil(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ColorMask extends BaseValue {\n getDefault(): ColorMaskType {\n return [true, true, true, true];\n }\n set(v: ColorMaskType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && v[3] === c[3] && !this.dirty) return;\n this.gl.colorMask(v[0], v[1], v[2], v[3]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthMask extends BaseValue {\n getDefault(): DepthMaskType {\n return true;\n }\n set(v: DepthMaskType): void {\n if (v === this.current && !this.dirty) return;\n this.gl.depthMask(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class StencilMask extends BaseValue {\n getDefault(): number {\n return 0xFF;\n }\n set(v: number): void {\n if (v === this.current && !this.dirty) return;\n this.gl.stencilMask(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class StencilFunc extends BaseValue {\n getDefault(): StencilFuncType {\n return {\n func: this.gl.ALWAYS,\n ref: 0,\n mask: 0xFF\n };\n }\n set(v: StencilFuncType): void {\n const c = this.current;\n if (v.func === c.func && v.ref === c.ref && v.mask === c.mask && !this.dirty) return;\n this.gl.stencilFunc(v.func, v.ref, v.mask);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class StencilOp extends BaseValue {\n getDefault(): StencilOpType {\n const gl = this.gl;\n return [gl.KEEP, gl.KEEP, gl.KEEP];\n }\n set(v: StencilOpType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && !this.dirty) return;\n this.gl.stencilOp(v[0], v[1], v[2]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class StencilTest extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n if (v) {\n gl.enable(gl.STENCIL_TEST);\n } else {\n gl.disable(gl.STENCIL_TEST);\n }\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthRange extends BaseValue {\n getDefault(): DepthRangeType {\n return [0, 1];\n }\n set(v: DepthRangeType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && !this.dirty) return;\n this.gl.depthRange(v[0], v[1]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthTest extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n if (v) {\n gl.enable(gl.DEPTH_TEST);\n } else {\n gl.disable(gl.DEPTH_TEST);\n }\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthFunc extends BaseValue {\n getDefault(): DepthFuncType {\n return this.gl.LESS;\n }\n set(v: DepthFuncType) {\n if (v === this.current && !this.dirty) return;\n this.gl.depthFunc(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class Blend extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n if (v) {\n gl.enable(gl.BLEND);\n } else {\n gl.disable(gl.BLEND);\n }\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BlendFunc extends BaseValue {\n getDefault(): BlendFuncType {\n const gl = this.gl;\n return [gl.ONE, gl.ZERO];\n }\n set(v: BlendFuncType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && !this.dirty) return;\n this.gl.blendFunc(v[0], v[1]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BlendColor extends BaseValue {\n getDefault(): Color {\n return Color.transparent;\n }\n set(v: Color) {\n const c = this.current;\n if (v.r === c.r && v.g === c.g && v.b === c.b && v.a === c.a && !this.dirty) return;\n this.gl.blendColor(v.r, v.g, v.b, v.a);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BlendEquation extends BaseValue {\n getDefault(): BlendEquationType {\n return this.gl.FUNC_ADD;\n }\n set(v: BlendEquationType) {\n if (v === this.current && !this.dirty) return;\n this.gl.blendEquation(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class CullFace extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n if (v) {\n gl.enable(gl.CULL_FACE);\n } else {\n gl.disable(gl.CULL_FACE);\n }\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class CullFaceSide extends BaseValue {\n getDefault(): CullFaceModeType {\n return this.gl.BACK;\n }\n set(v: CullFaceModeType) {\n if (v === this.current && !this.dirty) return;\n this.gl.cullFace(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class FrontFace extends BaseValue {\n getDefault(): FrontFaceType {\n return this.gl.CCW;\n }\n set(v: FrontFaceType) {\n if (v === this.current && !this.dirty) return;\n this.gl.frontFace(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ProgramValue extends BaseValue {\n getDefault(): WebGLProgram {\n return null;\n }\n set(v?: WebGLProgram | null) {\n if (v === this.current && !this.dirty) return;\n this.gl.useProgram(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ActiveTextureUnit extends BaseValue {\n getDefault(): TextureUnitType {\n return this.gl.TEXTURE0;\n }\n set(v: TextureUnitType) {\n if (v === this.current && !this.dirty) return;\n this.gl.activeTexture(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class Viewport extends BaseValue {\n getDefault(): ViewportType {\n const gl = this.gl;\n return [0, 0, gl.drawingBufferWidth, gl.drawingBufferHeight];\n }\n set(v: ViewportType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && v[3] === c[3] && !this.dirty) return;\n this.gl.viewport(v[0], v[1], v[2], v[3]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindFramebuffer extends BaseValue {\n getDefault(): WebGLFramebuffer {\n return null;\n }\n set(v?: WebGLFramebuffer | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.bindFramebuffer(gl.FRAMEBUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindRenderbuffer extends BaseValue {\n getDefault(): WebGLRenderbuffer {\n return null;\n }\n set(v?: WebGLRenderbuffer | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.bindRenderbuffer(gl.RENDERBUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindTexture extends BaseValue {\n getDefault(): WebGLTexture {\n return null;\n }\n set(v?: WebGLTexture | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.bindTexture(gl.TEXTURE_2D, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindVertexBuffer extends BaseValue {\n getDefault(): WebGLBuffer {\n return null;\n }\n set(v?: WebGLBuffer | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.bindBuffer(gl.ARRAY_BUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindElementBuffer extends BaseValue {\n getDefault(): WebGLBuffer {\n return null;\n }\n set(v?: WebGLBuffer | null) {\n // Always rebind\n const gl = this.gl;\n gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindVertexArray extends BaseValue {\n getDefault(): WebGLVertexArrayObject | null {\n return null;\n }\n set(v: WebGLVertexArrayObject | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n\n if (isWebGL2(gl)) {\n gl.bindVertexArray(v);\n } else {\n gl.getExtension('OES_vertex_array_object')?.bindVertexArrayOES(v);\n }\n\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class PixelStoreUnpack extends BaseValue {\n getDefault(): number {\n return 4;\n }\n set(v: number) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.pixelStorei(gl.UNPACK_ALIGNMENT, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class PixelStoreUnpackPremultiplyAlpha extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean): void {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, ((v as any)));\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class PixelStoreUnpackFlipY extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean): void {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, ((v as any)));\n this.current = v;\n this.dirty = false;\n }\n}\n\nclass FramebufferAttachment extends BaseValue {\n parent: WebGLFramebuffer;\n context: Context;\n\n constructor(context: Context, parent: WebGLFramebuffer) {\n super(context);\n this.context = context;\n this.parent = parent;\n }\n getDefault() {\n return null;\n }\n}\n\nexport class ColorAttachment extends FramebufferAttachment {\n setDirty() {\n this.dirty = true;\n }\n set(v?: WebGLTexture | null): void {\n if (v === this.current && !this.dirty) return;\n this.context.bindFramebuffer.set(this.parent);\n // note: it's possible to attach a renderbuffer to the color\n // attachment point, but thus far MBGL only uses textures for color\n const gl = this.gl;\n gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, v, 0);\n\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthAttachment extends FramebufferAttachment {\n set(v?: WebGLRenderbuffer | null): void {\n if (v === this.current && !this.dirty) return;\n this.context.bindFramebuffer.set(this.parent);\n // note: it's possible to attach a texture to the depth attachment\n // point, but thus far MBGL only uses renderbuffers for depth\n const gl = this.gl;\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_ATTACHMENT, gl.RENDERBUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthStencilAttachment extends FramebufferAttachment {\n set(v?: WebGLRenderbuffer | null): void {\n if (v === this.current && !this.dirty) return;\n this.context.bindFramebuffer.set(this.parent);\n // note: it's possible to attach a texture to the depth attachment\n // point, but thus far MBGL only uses renderbuffers for depth\n const gl = this.gl;\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n","import {ColorAttachment, DepthAttachment, DepthStencilAttachment} from './value';\n\nimport type {Context} from './context';\n\n/**\n * @internal\n * A framebuffer holder object\n */\nexport class Framebuffer {\n context: Context;\n width: number;\n height: number;\n framebuffer: WebGLFramebuffer;\n colorAttachment: ColorAttachment;\n depthAttachment: DepthAttachment;\n\n constructor(context: Context, width: number, height: number, hasDepth: boolean, hasStencil: boolean) {\n this.context = context;\n this.width = width;\n this.height = height;\n const gl = context.gl;\n const fbo = this.framebuffer = gl.createFramebuffer();\n\n this.colorAttachment = new ColorAttachment(context, fbo);\n if (hasDepth) {\n this.depthAttachment = hasStencil ? new DepthStencilAttachment(context, fbo) : new DepthAttachment(context, fbo);\n } else if (hasStencil) {\n throw new Error('Stencil cannot be setted without depth');\n }\n if (gl.checkFramebufferStatus(gl.FRAMEBUFFER) !== gl.FRAMEBUFFER_COMPLETE) {\n throw new Error('Framebuffer is not complete');\n }\n }\n\n destroy() {\n const gl = this.context.gl;\n\n const texture = this.colorAttachment.get();\n if (texture) gl.deleteTexture(texture);\n\n if (this.depthAttachment) {\n const renderbuffer = this.depthAttachment.get();\n if (renderbuffer) gl.deleteRenderbuffer(renderbuffer);\n }\n\n gl.deleteFramebuffer(this.framebuffer);\n }\n}\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {BlendFuncType, ColorMaskType} from './types';\n\nconst ZERO = 0x0000;\nconst ONE = 0x0001;\nconst ONE_MINUS_SRC_ALPHA = 0x0303;\n\nexport class ColorMode {\n blendFunction: BlendFuncType;\n blendColor: Color;\n mask: ColorMaskType;\n\n constructor(blendFunction: BlendFuncType, blendColor: Color, mask: ColorMaskType) {\n this.blendFunction = blendFunction;\n this.blendColor = blendColor;\n this.mask = mask;\n }\n\n static Replace: BlendFuncType;\n\n static disabled: Readonly;\n static unblended: Readonly;\n static alphaBlended: Readonly;\n}\n\nColorMode.Replace = [ONE, ZERO];\n\nColorMode.disabled = new ColorMode(ColorMode.Replace, Color.transparent, [false, false, false, false]);\nColorMode.unblended = new ColorMode(ColorMode.Replace, Color.transparent, [true, true, true, true]);\nColorMode.alphaBlended = new ColorMode([ONE, ONE_MINUS_SRC_ALPHA], Color.transparent, [true, true, true, true]);\n","import {IndexBuffer} from './index_buffer';\n\nimport {VertexBuffer} from './vertex_buffer';\nimport {Framebuffer} from './framebuffer';\nimport {DepthMode} from './depth_mode';\nimport {StencilMode} from './stencil_mode';\nimport {ColorMode} from './color_mode';\nimport {CullFaceMode} from './cull_face_mode';\nimport {deepEqual} from '../util/util';\nimport {ClearColor, ClearDepth, ClearStencil, ColorMask, DepthMask, StencilMask, StencilFunc, StencilOp, StencilTest, DepthRange, DepthTest, DepthFunc, Blend, BlendFunc, BlendColor, BlendEquation, CullFace, CullFaceSide, FrontFace, ProgramValue, ActiveTextureUnit, Viewport, BindFramebuffer, BindRenderbuffer, BindTexture, BindVertexBuffer, BindElementBuffer, BindVertexArray, PixelStoreUnpack, PixelStoreUnpackPremultiplyAlpha, PixelStoreUnpackFlipY} from './value';\n\nimport type {TriangleIndexArray, LineIndexArray, LineStripIndexArray} from '../data/index_array_type';\nimport type {\n StructArray,\n StructArrayMember\n} from '../util/struct_array';\nimport type {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {isWebGL2} from './webgl2';\n\ntype ClearArgs = {\n color?: Color;\n depth?: number;\n stencil?: number;\n};\n\n/**\n * @internal\n * A webgl wrapper class to allow injection, mocking and abstaction\n */\nexport class Context {\n gl: WebGLRenderingContext | WebGL2RenderingContext;\n\n currentNumAttributes: number;\n maxTextureSize: number;\n\n clearColor: ClearColor;\n clearDepth: ClearDepth;\n clearStencil: ClearStencil;\n colorMask: ColorMask;\n depthMask: DepthMask;\n stencilMask: StencilMask;\n stencilFunc: StencilFunc;\n stencilOp: StencilOp;\n stencilTest: StencilTest;\n depthRange: DepthRange;\n depthTest: DepthTest;\n depthFunc: DepthFunc;\n blend: Blend;\n blendFunc: BlendFunc;\n blendColor: BlendColor;\n blendEquation: BlendEquation;\n cullFace: CullFace;\n cullFaceSide: CullFaceSide;\n frontFace: FrontFace;\n program: ProgramValue;\n activeTexture: ActiveTextureUnit;\n viewport: Viewport;\n bindFramebuffer: BindFramebuffer;\n bindRenderbuffer: BindRenderbuffer;\n bindTexture: BindTexture;\n bindVertexBuffer: BindVertexBuffer;\n bindElementBuffer: BindElementBuffer;\n bindVertexArray: BindVertexArray;\n pixelStoreUnpack: PixelStoreUnpack;\n pixelStoreUnpackPremultiplyAlpha: PixelStoreUnpackPremultiplyAlpha;\n pixelStoreUnpackFlipY: PixelStoreUnpackFlipY;\n\n // eslint-disable-next-line camelcase\n extTextureFilterAnisotropic: EXT_texture_filter_anisotropic | null;\n extTextureFilterAnisotropicMax?: GLfloat;\n HALF_FLOAT?: GLenum;\n RGBA16F?: GLenum;\n RGB16F?: GLenum;\n\n constructor(gl: WebGLRenderingContext | WebGL2RenderingContext) {\n this.gl = gl;\n this.clearColor = new ClearColor(this);\n this.clearDepth = new ClearDepth(this);\n this.clearStencil = new ClearStencil(this);\n this.colorMask = new ColorMask(this);\n this.depthMask = new DepthMask(this);\n this.stencilMask = new StencilMask(this);\n this.stencilFunc = new StencilFunc(this);\n this.stencilOp = new StencilOp(this);\n this.stencilTest = new StencilTest(this);\n this.depthRange = new DepthRange(this);\n this.depthTest = new DepthTest(this);\n this.depthFunc = new DepthFunc(this);\n this.blend = new Blend(this);\n this.blendFunc = new BlendFunc(this);\n this.blendColor = new BlendColor(this);\n this.blendEquation = new BlendEquation(this);\n this.cullFace = new CullFace(this);\n this.cullFaceSide = new CullFaceSide(this);\n this.frontFace = new FrontFace(this);\n this.program = new ProgramValue(this);\n this.activeTexture = new ActiveTextureUnit(this);\n this.viewport = new Viewport(this);\n this.bindFramebuffer = new BindFramebuffer(this);\n this.bindRenderbuffer = new BindRenderbuffer(this);\n this.bindTexture = new BindTexture(this);\n this.bindVertexBuffer = new BindVertexBuffer(this);\n this.bindElementBuffer = new BindElementBuffer(this);\n this.bindVertexArray = new BindVertexArray(this);\n this.pixelStoreUnpack = new PixelStoreUnpack(this);\n this.pixelStoreUnpackPremultiplyAlpha = new PixelStoreUnpackPremultiplyAlpha(this);\n this.pixelStoreUnpackFlipY = new PixelStoreUnpackFlipY(this);\n\n this.extTextureFilterAnisotropic = (\n gl.getExtension('EXT_texture_filter_anisotropic') ||\n gl.getExtension('MOZ_EXT_texture_filter_anisotropic') ||\n gl.getExtension('WEBKIT_EXT_texture_filter_anisotropic')\n );\n\n if (this.extTextureFilterAnisotropic) {\n this.extTextureFilterAnisotropicMax = gl.getParameter(this.extTextureFilterAnisotropic.MAX_TEXTURE_MAX_ANISOTROPY_EXT);\n }\n\n this.maxTextureSize = gl.getParameter(gl.MAX_TEXTURE_SIZE);\n\n if (isWebGL2(gl)) {\n this.HALF_FLOAT = gl.HALF_FLOAT;\n const extColorBufferHalfFloat = gl.getExtension('EXT_color_buffer_half_float');\n this.RGBA16F = gl.RGBA16F ?? extColorBufferHalfFloat?.RGBA16F_EXT;\n this.RGB16F = gl.RGB16F ?? extColorBufferHalfFloat?.RGB16F_EXT;\n gl.getExtension('EXT_color_buffer_float');\n } else {\n gl.getExtension('EXT_color_buffer_half_float');\n gl.getExtension('OES_texture_half_float_linear');\n const extTextureHalfFloat = gl.getExtension('OES_texture_half_float');\n this.HALF_FLOAT = extTextureHalfFloat?.HALF_FLOAT_OES;\n }\n }\n\n setDefault() {\n this.unbindVAO();\n\n this.clearColor.setDefault();\n this.clearDepth.setDefault();\n this.clearStencil.setDefault();\n this.colorMask.setDefault();\n this.depthMask.setDefault();\n this.stencilMask.setDefault();\n this.stencilFunc.setDefault();\n this.stencilOp.setDefault();\n this.stencilTest.setDefault();\n this.depthRange.setDefault();\n this.depthTest.setDefault();\n this.depthFunc.setDefault();\n this.blend.setDefault();\n this.blendFunc.setDefault();\n this.blendColor.setDefault();\n this.blendEquation.setDefault();\n this.cullFace.setDefault();\n this.cullFaceSide.setDefault();\n this.frontFace.setDefault();\n this.program.setDefault();\n this.activeTexture.setDefault();\n this.bindFramebuffer.setDefault();\n this.pixelStoreUnpack.setDefault();\n this.pixelStoreUnpackPremultiplyAlpha.setDefault();\n this.pixelStoreUnpackFlipY.setDefault();\n }\n\n setDirty() {\n this.clearColor.dirty = true;\n this.clearDepth.dirty = true;\n this.clearStencil.dirty = true;\n this.colorMask.dirty = true;\n this.depthMask.dirty = true;\n this.stencilMask.dirty = true;\n this.stencilFunc.dirty = true;\n this.stencilOp.dirty = true;\n this.stencilTest.dirty = true;\n this.depthRange.dirty = true;\n this.depthTest.dirty = true;\n this.depthFunc.dirty = true;\n this.blend.dirty = true;\n this.blendFunc.dirty = true;\n this.blendColor.dirty = true;\n this.blendEquation.dirty = true;\n this.cullFace.dirty = true;\n this.cullFaceSide.dirty = true;\n this.frontFace.dirty = true;\n this.program.dirty = true;\n this.activeTexture.dirty = true;\n this.viewport.dirty = true;\n this.bindFramebuffer.dirty = true;\n this.bindRenderbuffer.dirty = true;\n this.bindTexture.dirty = true;\n this.bindVertexBuffer.dirty = true;\n this.bindElementBuffer.dirty = true;\n this.bindVertexArray.dirty = true;\n this.pixelStoreUnpack.dirty = true;\n this.pixelStoreUnpackPremultiplyAlpha.dirty = true;\n this.pixelStoreUnpackFlipY.dirty = true;\n }\n\n createIndexBuffer(array: TriangleIndexArray | LineIndexArray | LineStripIndexArray, dynamicDraw?: boolean) {\n return new IndexBuffer(this, array, dynamicDraw);\n }\n\n createVertexBuffer(array: StructArray, attributes: ReadonlyArray, dynamicDraw?: boolean) {\n return new VertexBuffer(this, array, attributes, dynamicDraw);\n }\n\n createRenderbuffer(storageFormat: number, width: number, height: number) {\n const gl = this.gl;\n\n const rbo = gl.createRenderbuffer();\n this.bindRenderbuffer.set(rbo);\n gl.renderbufferStorage(gl.RENDERBUFFER, storageFormat, width, height);\n this.bindRenderbuffer.set(null);\n\n return rbo;\n }\n\n createFramebuffer(width: number, height: number, hasDepth: boolean, hasStencil: boolean) {\n return new Framebuffer(this, width, height, hasDepth, hasStencil);\n }\n\n clear({\n color,\n depth,\n stencil\n }: ClearArgs) {\n const gl = this.gl;\n let mask = 0;\n\n if (color) {\n mask |= gl.COLOR_BUFFER_BIT;\n this.clearColor.set(color);\n this.colorMask.set([true, true, true, true]);\n }\n\n if (typeof depth !== 'undefined') {\n mask |= gl.DEPTH_BUFFER_BIT;\n\n // Workaround for platforms where clearDepth doesn't seem to work\n // without resetting the depthRange. See https://github.com/mapbox/mapbox-gl-js/issues/3437\n this.depthRange.set([0, 1]);\n\n this.clearDepth.set(depth);\n this.depthMask.set(true);\n }\n\n if (typeof stencil !== 'undefined') {\n mask |= gl.STENCIL_BUFFER_BIT;\n this.clearStencil.set(stencil);\n this.stencilMask.set(0xFF);\n }\n\n gl.clear(mask);\n }\n\n setCullFace(cullFaceMode: Readonly) {\n if (cullFaceMode.enable === false) {\n this.cullFace.set(false);\n } else {\n this.cullFace.set(true);\n this.cullFaceSide.set(cullFaceMode.mode);\n this.frontFace.set(cullFaceMode.frontFace);\n }\n }\n\n setDepthMode(depthMode: Readonly) {\n if (depthMode.func === this.gl.ALWAYS && !depthMode.mask) {\n this.depthTest.set(false);\n } else {\n this.depthTest.set(true);\n this.depthFunc.set(depthMode.func);\n this.depthMask.set(depthMode.mask);\n this.depthRange.set(depthMode.range);\n }\n }\n\n setStencilMode(stencilMode: Readonly) {\n if (stencilMode.test.func === this.gl.ALWAYS && !stencilMode.mask) {\n this.stencilTest.set(false);\n } else {\n this.stencilTest.set(true);\n this.stencilMask.set(stencilMode.mask);\n this.stencilOp.set([stencilMode.fail, stencilMode.depthFail, stencilMode.pass]);\n this.stencilFunc.set({\n func: stencilMode.test.func,\n ref: stencilMode.ref,\n mask: stencilMode.test.mask\n });\n }\n }\n\n setColorMode(colorMode: Readonly) {\n if (deepEqual(colorMode.blendFunction, ColorMode.Replace)) {\n this.blend.set(false);\n } else {\n this.blend.set(true);\n this.blendFunc.set(colorMode.blendFunction);\n this.blendColor.set(colorMode.blendColor);\n }\n\n this.colorMask.set(colorMode.mask);\n }\n\n createVertexArray(): WebGLVertexArrayObject | undefined {\n if (isWebGL2(this.gl))\n return this.gl.createVertexArray();\n return this.gl.getExtension('OES_vertex_array_object')?.createVertexArrayOES();\n }\n\n deleteVertexArray(x: WebGLVertexArrayObject | undefined) {\n if (isWebGL2(this.gl))\n return this.gl.deleteVertexArray(x);\n return this.gl.getExtension('OES_vertex_array_object')?.deleteVertexArrayOES(x);\n }\n\n unbindVAO() {\n // Unbinding the VAO prevents other things (custom layers, new buffer creation) from\n // unintentionally changing the state of the last VAO used.\n this.bindVertexArray.set(null);\n }\n}\n","import type {DepthFuncType, DepthMaskType, DepthRangeType} from './types';\n\nconst ALWAYS = 0x0207;\n\nexport class DepthMode {\n func: DepthFuncType;\n mask: DepthMaskType;\n range: DepthRangeType;\n\n // DepthMask enums\n static ReadOnly: boolean;\n static ReadWrite: boolean;\n\n constructor(depthFunc: DepthFuncType, depthMask: DepthMaskType, depthRange: DepthRangeType) {\n this.func = depthFunc;\n this.mask = depthMask;\n this.range = depthRange;\n }\n\n static disabled: Readonly;\n}\n\nDepthMode.ReadOnly = false;\nDepthMode.ReadWrite = true;\n\nDepthMode.disabled = new DepthMode(ALWAYS, DepthMode.ReadOnly, [0, 1]);\n","import type {StencilOpConstant, StencilTestGL} from './types';\n\nconst ALWAYS = 0x0207;\nconst KEEP = 0x1E00;\n\nexport class StencilMode {\n test: StencilTestGL;\n ref: number;\n mask: number;\n fail: StencilOpConstant;\n depthFail: StencilOpConstant;\n pass: StencilOpConstant;\n\n constructor(test: StencilTestGL, ref: number, mask: number, fail: StencilOpConstant,\n depthFail: StencilOpConstant, pass: StencilOpConstant) {\n this.test = test;\n this.ref = ref;\n this.mask = mask;\n this.fail = fail;\n this.depthFail = depthFail;\n this.pass = pass;\n }\n\n static disabled: Readonly;\n}\n\nStencilMode.disabled = new StencilMode({func: ALWAYS, mask: 0}, 0, 0, KEEP, KEEP, KEEP);\n","import type {CullFaceModeType, FrontFaceType} from './types';\n\nconst BACK = 0x0405;\nconst CCW = 0x0901;\n\nexport class CullFaceMode {\n enable: boolean;\n mode: CullFaceModeType;\n frontFace: FrontFaceType;\n\n constructor(enable: boolean, mode: CullFaceModeType, frontFace: FrontFaceType) {\n this.enable = enable;\n this.mode = mode;\n this.frontFace = frontFace;\n }\n\n static disabled: Readonly;\n static backCCW: Readonly;\n}\n\nCullFaceMode.disabled = new CullFaceMode(false, BACK, CCW);\nCullFaceMode.backCCW = new CullFaceMode(true, BACK, CCW);\n","import type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {collisionUniformValues, collisionCircleUniformValues} from './program/collision_program';\n\nimport {QuadTriangleArray, CollisionCircleLayoutArray} from '../data/array_types.g';\nimport {collisionCircleLayout} from '../data/bucket/symbol_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {mat4} from 'gl-matrix';\nimport {VertexBuffer} from '../gl/vertex_buffer';\nimport {IndexBuffer} from '../gl/index_buffer';\n\ntype TileBatch = {\n circleArray: Array;\n circleOffset: number;\n transform: mat4;\n invTransform: mat4;\n coord: OverscaledTileID;\n};\n\nlet quadTriangles: QuadTriangleArray;\n\nexport function drawCollisionDebug(painter: Painter, sourceCache: SourceCache, layer: StyleLayer, coords: Array, translate: [number, number], translateAnchor: 'map' | 'viewport', isText: boolean) {\n const context = painter.context;\n const gl = context.gl;\n const program = painter.useProgram('collisionBox');\n const tileBatches: Array = [];\n let circleCount = 0;\n let circleOffset = 0;\n\n for (let i = 0; i < coords.length; i++) {\n const coord = coords[i];\n const tile = sourceCache.getTile(coord);\n const bucket: SymbolBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n let posMatrix = coord.posMatrix;\n if (translate[0] !== 0 || translate[1] !== 0) {\n posMatrix = painter.translatePosMatrix(coord.posMatrix, tile, translate, translateAnchor);\n }\n const buffers = isText ? bucket.textCollisionBox : bucket.iconCollisionBox;\n // Get collision circle data of this bucket\n const circleArray: Array = bucket.collisionCircleArray;\n if (circleArray.length > 0) {\n // We need to know the projection matrix that was used for projecting collision circles to the screen.\n // This might vary between buckets as the symbol placement is a continuous process. This matrix is\n // required for transforming points from previous screen space to the current one\n const invTransform = mat4.create();\n const transform = posMatrix;\n\n mat4.mul(invTransform, bucket.placementInvProjMatrix, painter.transform.glCoordMatrix);\n mat4.mul(invTransform, invTransform, bucket.placementViewportMatrix);\n\n tileBatches.push({\n circleArray,\n circleOffset,\n transform,\n invTransform,\n coord\n });\n\n circleCount += circleArray.length / 4; // 4 values per circle\n circleOffset = circleCount;\n }\n if (!buffers) continue;\n program.draw(context, gl.LINES,\n DepthMode.disabled, StencilMode.disabled,\n painter.colorModeForRenderPass(),\n CullFaceMode.disabled,\n collisionUniformValues(\n posMatrix,\n painter.transform,\n tile),\n painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord),\n layer.id, buffers.layoutVertexBuffer, buffers.indexBuffer,\n buffers.segments, null, painter.transform.zoom, null, null,\n buffers.collisionVertexBuffer);\n }\n\n if (!isText || !tileBatches.length) {\n return;\n }\n\n // Render collision circles\n const circleProgram = painter.useProgram('collisionCircle');\n\n // Construct vertex data\n const vertexData = new CollisionCircleLayoutArray();\n vertexData.resize(circleCount * 4);\n vertexData._trim();\n\n let vertexOffset = 0;\n\n for (const batch of tileBatches) {\n for (let i = 0; i < batch.circleArray.length / 4; i++) {\n const circleIdx = i * 4;\n const x = batch.circleArray[circleIdx + 0];\n const y = batch.circleArray[circleIdx + 1];\n const radius = batch.circleArray[circleIdx + 2];\n const collision = batch.circleArray[circleIdx + 3];\n\n // 4 floats per vertex, 4 vertices per quad\n vertexData.emplace(vertexOffset++, x, y, radius, collision, 0);\n vertexData.emplace(vertexOffset++, x, y, radius, collision, 1);\n vertexData.emplace(vertexOffset++, x, y, radius, collision, 2);\n vertexData.emplace(vertexOffset++, x, y, radius, collision, 3);\n }\n }\n if (!quadTriangles || quadTriangles.length < circleCount * 2) {\n quadTriangles = createQuadTriangles(circleCount);\n }\n\n const indexBuffer: IndexBuffer = context.createIndexBuffer(quadTriangles, true);\n const vertexBuffer: VertexBuffer = context.createVertexBuffer(vertexData, collisionCircleLayout.members, true);\n\n // Render batches\n for (const batch of tileBatches) {\n const uniforms = collisionCircleUniformValues(\n batch.transform,\n batch.invTransform,\n painter.transform\n );\n\n circleProgram.draw(\n context,\n gl.TRIANGLES,\n DepthMode.disabled,\n StencilMode.disabled,\n painter.colorModeForRenderPass(),\n CullFaceMode.disabled,\n uniforms,\n painter.style.map.terrain && painter.style.map.terrain.getTerrainData(batch.coord),\n layer.id,\n vertexBuffer,\n indexBuffer,\n SegmentVector.simpleSegment(0, batch.circleOffset * 2, batch.circleArray.length, batch.circleArray.length / 2),\n null,\n painter.transform.zoom,\n null,\n null,\n null);\n }\n\n vertexBuffer.destroy();\n indexBuffer.destroy();\n}\n\nfunction createQuadTriangles(quadCount: number): QuadTriangleArray {\n const triCount = quadCount * 2;\n const array = new QuadTriangleArray();\n\n array.resize(triCount);\n array._trim();\n\n // Two triangles and 4 vertices per quad.\n for (let i = 0; i < triCount; i++) {\n const idx = i * 6;\n\n array.uint16[idx + 0] = i * 4 + 0;\n array.uint16[idx + 1] = i * 4 + 1;\n array.uint16[idx + 2] = i * 4 + 2;\n array.uint16[idx + 3] = i * 4 + 2;\n array.uint16[idx + 4] = i * 4 + 3;\n array.uint16[idx + 5] = i * 4 + 0;\n }\n\n return array;\n}\n","import Point from '@mapbox/point-geometry';\nimport {drawCollisionDebug} from './draw_collision_debug';\n\nimport {SegmentVector} from '../data/segment';\nimport {pixelsToTileUnits} from '../source/pixels_to_tile_units';\nimport * as symbolProjection from '../symbol/projection';\nimport {EvaluatedZoomSize, evaluateSizeForFeature, evaluateSizeForZoom} from '../symbol/symbol_size';\nimport {mat4} from 'gl-matrix';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {addDynamicAttributes} from '../data/bucket/symbol_bucket';\n\nimport {getAnchorAlignment, WritingMode} from '../symbol/shaping';\nimport ONE_EM from '../symbol/one_em';\n\nimport {\n SymbolIconUniformsType,\n symbolIconUniformValues,\n symbolSDFUniformValues,\n symbolTextAndIconUniformValues\n} from './program/symbol_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\n\nimport type {Texture} from '../render/texture';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {UniformValues} from './uniform_binding';\nimport type {SymbolSDFUniformsType} from '../render/program/symbol_program';\nimport type {CrossTileID, VariableOffset} from '../symbol/placement';\nimport type {SymbolBucket, SymbolBuffers} from '../data/bucket/symbol_bucket';\nimport type {TerrainData} from '../render/terrain';\nimport type {SymbolLayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Transform} from '../geo/transform';\nimport type {ColorMode} from '../gl/color_mode';\nimport type {Program} from './program';\nimport type {TextAnchor} from '../style/style_layer/variable_text_anchor';\n\ntype SymbolTileRenderState = {\n segments: SegmentVector;\n sortKey: number;\n terrainData: TerrainData;\n state: {\n program: Program;\n buffers: SymbolBuffers;\n uniformValues: UniformValues;\n atlasTexture: Texture;\n atlasTextureIcon: Texture | null;\n atlasInterpolation: GLenum;\n atlasInterpolationIcon: GLenum;\n isSDF: boolean;\n hasHalo: boolean;\n };\n};\n\nconst identityMat4 = mat4.identity(new Float32Array(16));\n\nexport function drawSymbols(painter: Painter, sourceCache: SourceCache, layer: SymbolStyleLayer, coords: Array, variableOffsets: {\n [_ in CrossTileID]: VariableOffset;\n}) {\n if (painter.renderPass !== 'translucent') return;\n\n // Disable the stencil test so that labels aren't clipped to tile boundaries.\n const stencilMode = StencilMode.disabled;\n const colorMode = painter.colorModeForRenderPass();\n const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset');\n\n //Compute variable-offsets before painting since icons and text data positioning\n //depend on each other in this case.\n if (hasVariablePlacement) {\n updateVariableAnchors(coords, painter, layer, sourceCache,\n layer.layout.get('text-rotation-alignment'),\n layer.layout.get('text-pitch-alignment'),\n variableOffsets\n );\n }\n\n if (layer.paint.get('icon-opacity').constantOr(1) !== 0) {\n drawLayerSymbols(painter, sourceCache, layer, coords, false,\n layer.paint.get('icon-translate'),\n layer.paint.get('icon-translate-anchor'),\n layer.layout.get('icon-rotation-alignment'),\n layer.layout.get('icon-pitch-alignment'),\n layer.layout.get('icon-keep-upright'),\n stencilMode, colorMode\n );\n }\n\n if (layer.paint.get('text-opacity').constantOr(1) !== 0) {\n drawLayerSymbols(painter, sourceCache, layer, coords, true,\n layer.paint.get('text-translate'),\n layer.paint.get('text-translate-anchor'),\n layer.layout.get('text-rotation-alignment'),\n layer.layout.get('text-pitch-alignment'),\n layer.layout.get('text-keep-upright'),\n stencilMode, colorMode\n );\n }\n\n if (sourceCache.map.showCollisionBoxes) {\n drawCollisionDebug(painter, sourceCache, layer, coords, layer.paint.get('text-translate'),\n layer.paint.get('text-translate-anchor'), true);\n drawCollisionDebug(painter, sourceCache, layer, coords, layer.paint.get('icon-translate'),\n layer.paint.get('icon-translate-anchor'), false);\n }\n}\n\nfunction calculateVariableRenderShift(\n anchor: TextAnchor,\n width: number,\n height: number,\n textOffset: [number, number],\n textBoxScale: number,\n renderTextSize: number): Point {\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(anchor);\n const shiftX = -(horizontalAlign - 0.5) * width;\n const shiftY = -(verticalAlign - 0.5) * height;\n return new Point(\n (shiftX / textBoxScale + textOffset[0]) * renderTextSize,\n (shiftY / textBoxScale + textOffset[1]) * renderTextSize\n );\n}\n\nfunction updateVariableAnchors(coords: Array,\n painter: Painter,\n layer:SymbolStyleLayer, sourceCache: SourceCache,\n rotationAlignment: SymbolLayerSpecification['layout']['text-rotation-alignment'],\n pitchAlignment: SymbolLayerSpecification['layout']['text-pitch-alignment'],\n variableOffsets: {[_ in CrossTileID]: VariableOffset}) {\n const tr = painter.transform;\n const rotateWithMap = rotationAlignment === 'map';\n const pitchWithMap = pitchAlignment === 'map';\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n const bucket = tile.getBucket(layer) as SymbolBucket;\n if (!bucket || !bucket.text || !bucket.text.segments.get().length) continue;\n\n const sizeData = bucket.textSizeData;\n const size = evaluateSizeForZoom(sizeData, tr.zoom);\n\n const pixelToTileScale = pixelsToTileUnits(tile, 1, painter.transform.zoom);\n const labelPlaneMatrix = symbolProjection.getLabelPlaneMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, pixelToTileScale);\n const updateTextFitIcon = layer.layout.get('icon-text-fit') !== 'none' && bucket.hasIconData();\n\n if (size) {\n const tileScale = Math.pow(2, tr.zoom - tile.tileID.overscaledZ);\n const getElevation = painter.style.map.terrain ? (x: number, y: number) => painter.style.map.terrain.getElevation(coord, x, y) : null;\n updateVariableAnchorsForBucket(bucket, rotateWithMap, pitchWithMap, variableOffsets,\n tr, labelPlaneMatrix, coord.posMatrix, tileScale, size, updateTextFitIcon, getElevation);\n }\n }\n}\n\nfunction updateVariableAnchorsForBucket(\n bucket: SymbolBucket,\n rotateWithMap: boolean,\n pitchWithMap: boolean,\n variableOffsets: {[_ in CrossTileID]: VariableOffset},\n transform: Transform,\n labelPlaneMatrix: mat4,\n posMatrix: mat4,\n tileScale: number,\n size: EvaluatedZoomSize,\n updateTextFitIcon: boolean,\n getElevation: (x: number, y: number) => number) {\n const placedSymbols = bucket.text.placedSymbolArray;\n const dynamicTextLayoutVertexArray = bucket.text.dynamicLayoutVertexArray;\n const dynamicIconLayoutVertexArray = bucket.icon.dynamicLayoutVertexArray;\n const placedTextShifts = {};\n\n dynamicTextLayoutVertexArray.clear();\n for (let s = 0; s < placedSymbols.length; s++) {\n const symbol = placedSymbols.get(s);\n const skipOrientation = bucket.allowVerticalPlacement && !symbol.placedOrientation;\n const variableOffset = (!symbol.hidden && symbol.crossTileID && !skipOrientation) ? variableOffsets[symbol.crossTileID] : null;\n\n if (!variableOffset) {\n // These symbols are from a justification that is not being used, or a label that wasn't placed\n // so we don't need to do the extra math to figure out what incremental shift to apply.\n symbolProjection.hideGlyphs(symbol.numGlyphs, dynamicTextLayoutVertexArray);\n } else {\n const tileAnchor = new Point(symbol.anchorX, symbol.anchorY);\n const projectedAnchor = symbolProjection.project(tileAnchor, pitchWithMap ? posMatrix : labelPlaneMatrix, getElevation);\n const perspectiveRatio = symbolProjection.getPerspectiveRatio(transform.cameraToCenterDistance, projectedAnchor.signedDistanceFromCamera);\n let renderTextSize = evaluateSizeForFeature(bucket.textSizeData, size, symbol) * perspectiveRatio / ONE_EM;\n if (pitchWithMap) {\n // Go from size in pixels to equivalent size in tile units\n renderTextSize *= bucket.tilePixelRatio / tileScale;\n }\n\n const {width, height, anchor, textOffset, textBoxScale} = variableOffset;\n\n const shift = calculateVariableRenderShift(\n anchor, width, height, textOffset, textBoxScale, renderTextSize);\n\n // Usual case is that we take the projected anchor and add the pixel-based shift\n // calculated above. In the (somewhat weird) case of pitch-aligned text, we add an equivalent\n // tile-unit based shift to the anchor before projecting to the label plane.\n const shiftedAnchor = pitchWithMap ?\n symbolProjection.project(tileAnchor.add(shift), labelPlaneMatrix, getElevation).point :\n projectedAnchor.point.add(rotateWithMap ?\n shift.rotate(-transform.angle) :\n shift);\n\n const angle = (bucket.allowVerticalPlacement && symbol.placedOrientation === WritingMode.vertical) ? Math.PI / 2 : 0;\n for (let g = 0; g < symbol.numGlyphs; g++) {\n addDynamicAttributes(dynamicTextLayoutVertexArray, shiftedAnchor, angle);\n }\n //Only offset horizontal text icons\n if (updateTextFitIcon && symbol.associatedIconIndex >= 0) {\n placedTextShifts[symbol.associatedIconIndex] = {shiftedAnchor, angle};\n }\n }\n }\n\n if (updateTextFitIcon) {\n dynamicIconLayoutVertexArray.clear();\n const placedIcons = bucket.icon.placedSymbolArray;\n for (let i = 0; i < placedIcons.length; i++) {\n const placedIcon = placedIcons.get(i);\n if (placedIcon.hidden) {\n symbolProjection.hideGlyphs(placedIcon.numGlyphs, dynamicIconLayoutVertexArray);\n } else {\n const shift = placedTextShifts[i];\n if (!shift) {\n symbolProjection.hideGlyphs(placedIcon.numGlyphs, dynamicIconLayoutVertexArray);\n } else {\n for (let g = 0; g < placedIcon.numGlyphs; g++) {\n addDynamicAttributes(dynamicIconLayoutVertexArray, shift.shiftedAnchor, shift.angle);\n }\n }\n }\n }\n bucket.icon.dynamicLayoutVertexBuffer.updateData(dynamicIconLayoutVertexArray);\n }\n bucket.text.dynamicLayoutVertexBuffer.updateData(dynamicTextLayoutVertexArray);\n}\n\nfunction getSymbolProgramName(isSDF: boolean, isText: boolean, bucket: SymbolBucket) {\n if (bucket.iconsInText && isText) {\n return 'symbolTextAndIcon';\n } else if (isSDF) {\n return 'symbolSDF';\n } else {\n return 'symbolIcon';\n }\n}\n\nfunction drawLayerSymbols(\n painter: Painter,\n sourceCache: SourceCache,\n layer: SymbolStyleLayer,\n coords: Array,\n isText: boolean,\n translate: [number, number],\n translateAnchor: 'map' | 'viewport',\n rotationAlignment: SymbolLayerSpecification['layout']['text-rotation-alignment'],\n pitchAlignment: SymbolLayerSpecification['layout']['text-pitch-alignment'],\n keepUpright: boolean,\n stencilMode: StencilMode,\n colorMode: Readonly) {\n\n const context = painter.context;\n const gl = context.gl;\n const tr = painter.transform;\n\n const rotateWithMap = rotationAlignment === 'map';\n const pitchWithMap = pitchAlignment === 'map';\n const alongLine = rotationAlignment !== 'viewport' && layer.layout.get('symbol-placement') !== 'point';\n // Line label rotation happens in `updateLineLabels`\n // Pitched point labels are automatically rotated by the labelPlaneMatrix projection\n // Unpitched point labels need to have their rotation applied after projection\n const rotateInShader = rotateWithMap && !pitchWithMap && !alongLine;\n\n const hasSortKey = !layer.layout.get('symbol-sort-key').isConstant();\n let sortFeaturesByKey = false;\n\n const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n\n const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset');\n\n const tileRenderState: Array = [];\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n const bucket = tile.getBucket(layer) as SymbolBucket;\n if (!bucket) continue;\n const buffers = isText ? bucket.text : bucket.icon;\n\n if (!buffers || !buffers.segments.get().length || !buffers.hasVisibleVertices) continue;\n const programConfiguration = buffers.programConfigurations.get(layer.id);\n\n const isSDF = isText || bucket.sdfIcons;\n\n const sizeData = isText ? bucket.textSizeData : bucket.iconSizeData;\n const transformed = pitchWithMap || tr.pitch !== 0;\n\n const program = painter.useProgram(getSymbolProgramName(isSDF, isText, bucket), programConfiguration);\n const size = evaluateSizeForZoom(sizeData, tr.zoom);\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n let texSize: [number, number];\n let texSizeIcon: [number, number] = [0, 0];\n let atlasTexture: Texture;\n let atlasInterpolation: GLenum;\n let atlasTextureIcon = null;\n let atlasInterpolationIcon: GLenum;\n if (isText) {\n atlasTexture = tile.glyphAtlasTexture;\n atlasInterpolation = gl.LINEAR;\n texSize = tile.glyphAtlasTexture.size;\n if (bucket.iconsInText) {\n texSizeIcon = tile.imageAtlasTexture.size;\n atlasTextureIcon = tile.imageAtlasTexture;\n const zoomDependentSize = sizeData.kind === 'composite' || sizeData.kind === 'camera';\n atlasInterpolationIcon = transformed || painter.options.rotating || painter.options.zooming || zoomDependentSize ? gl.LINEAR : gl.NEAREST;\n }\n } else {\n const iconScaled = layer.layout.get('icon-size').constantOr(0) !== 1 || bucket.iconsNeedLinear;\n atlasTexture = tile.imageAtlasTexture;\n atlasInterpolation = isSDF || painter.options.rotating || painter.options.zooming || iconScaled || transformed ?\n gl.LINEAR :\n gl.NEAREST;\n texSize = tile.imageAtlasTexture.size;\n }\n\n const s = pixelsToTileUnits(tile, 1, painter.transform.zoom);\n const labelPlaneMatrix = symbolProjection.getLabelPlaneMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, s);\n const glCoordMatrix = symbolProjection.getGlCoordMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, s);\n\n const hasVariableAnchors = hasVariablePlacement && bucket.hasTextData();\n const updateTextFitIcon = layer.layout.get('icon-text-fit') !== 'none' &&\n hasVariableAnchors &&\n bucket.hasIconData();\n\n if (alongLine) {\n const getElevation = painter.style.map.terrain ? (x: number, y: number) => painter.style.map.terrain.getElevation(coord, x, y) : null;\n const rotateToLine = layer.layout.get('text-rotation-alignment') === 'map';\n symbolProjection.updateLineLabels(bucket, coord.posMatrix, painter, isText, labelPlaneMatrix, glCoordMatrix, pitchWithMap, keepUpright, rotateToLine, getElevation);\n }\n\n const matrix = painter.translatePosMatrix(coord.posMatrix, tile, translate, translateAnchor),\n uLabelPlaneMatrix = (alongLine || (isText && hasVariablePlacement) || updateTextFitIcon) ? identityMat4 : labelPlaneMatrix,\n uglCoordMatrix = painter.translatePosMatrix(glCoordMatrix, tile, translate, translateAnchor, true);\n\n const hasHalo = isSDF && layer.paint.get(isText ? 'text-halo-width' : 'icon-halo-width').constantOr(1) !== 0;\n\n let uniformValues: UniformValues;\n if (isSDF) {\n if (!bucket.iconsInText) {\n uniformValues = symbolSDFUniformValues(sizeData.kind,\n size, rotateInShader, pitchWithMap, painter, matrix,\n uLabelPlaneMatrix, uglCoordMatrix, isText, texSize, true);\n } else {\n uniformValues = symbolTextAndIconUniformValues(sizeData.kind,\n size, rotateInShader, pitchWithMap, painter, matrix,\n uLabelPlaneMatrix, uglCoordMatrix, texSize, texSizeIcon);\n }\n } else {\n uniformValues = symbolIconUniformValues(sizeData.kind,\n size, rotateInShader, pitchWithMap, painter, matrix,\n uLabelPlaneMatrix, uglCoordMatrix, isText, texSize);\n }\n\n const state = {\n program,\n buffers,\n uniformValues,\n atlasTexture,\n atlasTextureIcon,\n atlasInterpolation,\n atlasInterpolationIcon,\n isSDF,\n hasHalo\n };\n\n if (hasSortKey && bucket.canOverlap) {\n sortFeaturesByKey = true;\n const oldSegments = buffers.segments.get();\n for (const segment of oldSegments) {\n tileRenderState.push({\n segments: new SegmentVector([segment]),\n sortKey: segment.sortKey,\n state,\n terrainData\n });\n }\n } else {\n tileRenderState.push({\n segments: buffers.segments,\n sortKey: 0,\n state,\n terrainData\n });\n }\n }\n\n if (sortFeaturesByKey) {\n tileRenderState.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const segmentState of tileRenderState) {\n const state = segmentState.state;\n\n context.activeTexture.set(gl.TEXTURE0);\n // @ts-ignore\n state.atlasTexture.bind(state.atlasInterpolation, gl.CLAMP_TO_EDGE);\n if (state.atlasTextureIcon) {\n context.activeTexture.set(gl.TEXTURE1);\n if (state.atlasTextureIcon) {\n // @ts-ignore\n state.atlasTextureIcon.bind(state.atlasInterpolationIcon, gl.CLAMP_TO_EDGE);\n }\n }\n\n if (state.isSDF) {\n const uniformValues = state.uniformValues;\n if (state.hasHalo) {\n uniformValues['u_is_halo'] = 1;\n drawSymbolElements(state.buffers, segmentState.segments, layer, painter, state.program, depthMode, stencilMode, colorMode, uniformValues, segmentState.terrainData);\n }\n uniformValues['u_is_halo'] = 0;\n }\n drawSymbolElements(state.buffers, segmentState.segments, layer, painter, state.program, depthMode, stencilMode, colorMode, state.uniformValues, segmentState.terrainData);\n }\n}\n\nfunction drawSymbolElements(\n buffers: SymbolBuffers,\n segments: SegmentVector,\n layer: SymbolStyleLayer,\n painter: Painter,\n program: Program,\n depthMode: Readonly,\n stencilMode: StencilMode,\n colorMode: Readonly,\n uniformValues: UniformValues,\n terrainData: TerrainData) {\n const context = painter.context;\n const gl = context.gl;\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id, buffers.layoutVertexBuffer,\n buffers.indexBuffer, segments, layer.paint,\n painter.transform.zoom, buffers.programConfigurations.get(layer.id),\n buffers.dynamicLayoutVertexBuffer, buffers.opacityVertexBuffer);\n}\n","import {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Program} from './program';\nimport {circleUniformValues} from './program/circle_program';\nimport {SegmentVector} from '../data/segment';\nimport {OverscaledTileID} from '../source/tile_id';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {CircleStyleLayer} from '../style/style_layer/circle_style_layer';\nimport type {CircleBucket} from '../data/bucket/circle_bucket';\nimport type {ProgramConfiguration} from '../data/program_configuration';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {IndexBuffer} from '../gl/index_buffer';\nimport type {UniformValues} from './uniform_binding';\nimport type {CircleUniformsType} from './program/circle_program';\nimport type {TerrainData} from '../render/terrain';\n\ntype TileRenderState = {\n programConfiguration: ProgramConfiguration;\n program: Program;\n layoutVertexBuffer: VertexBuffer;\n indexBuffer: IndexBuffer;\n uniformValues: UniformValues;\n terrainData: TerrainData;\n};\n\ntype SegmentsTileRenderState = {\n segments: SegmentVector;\n sortKey: number;\n state: TileRenderState;\n};\n\nexport function drawCircles(painter: Painter, sourceCache: SourceCache, layer: CircleStyleLayer, coords: Array) {\n if (painter.renderPass !== 'translucent') return;\n\n const opacity = layer.paint.get('circle-opacity');\n const strokeWidth = layer.paint.get('circle-stroke-width');\n const strokeOpacity = layer.paint.get('circle-stroke-opacity');\n const sortFeaturesByKey = !layer.layout.get('circle-sort-key').isConstant();\n\n if (opacity.constantOr(1) === 0 && (strokeWidth.constantOr(1) === 0 || strokeOpacity.constantOr(1) === 0)) {\n return;\n }\n\n const context = painter.context;\n const gl = context.gl;\n\n const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n // Turn off stencil testing to allow circles to be drawn across boundaries,\n // so that large circles are not clipped to tiles\n const stencilMode = StencilMode.disabled;\n const colorMode = painter.colorModeForRenderPass();\n\n const segmentsRenderStates: Array = [];\n\n for (let i = 0; i < coords.length; i++) {\n const coord = coords[i];\n\n const tile = sourceCache.getTile(coord);\n const bucket: CircleBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const program = painter.useProgram('circle', programConfiguration);\n const layoutVertexBuffer = bucket.layoutVertexBuffer;\n const indexBuffer = bucket.indexBuffer;\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n const uniformValues = circleUniformValues(painter, coord, tile, layer);\n\n const state: TileRenderState = {\n programConfiguration,\n program,\n layoutVertexBuffer,\n indexBuffer,\n uniformValues,\n terrainData\n };\n\n if (sortFeaturesByKey) {\n const oldSegments = bucket.segments.get();\n for (const segment of oldSegments) {\n segmentsRenderStates.push({\n segments: new SegmentVector([segment]),\n sortKey: (segment.sortKey as any as number),\n state\n });\n }\n } else {\n segmentsRenderStates.push({\n segments: bucket.segments,\n sortKey: 0,\n state\n });\n }\n\n }\n\n if (sortFeaturesByKey) {\n segmentsRenderStates.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const segmentsState of segmentsRenderStates) {\n const {programConfiguration, program, layoutVertexBuffer, indexBuffer, uniformValues, terrainData} = segmentsState.state;\n const segments = segmentsState.segments;\n\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id,\n layoutVertexBuffer, indexBuffer, segments,\n layer.paint, painter.transform.zoom, programConfiguration);\n }\n}\n","import {Texture} from './texture';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Context} from '../gl/context';\nimport {Framebuffer} from '../gl/framebuffer';\nimport {\n heatmapUniformValues,\n heatmapTextureUniformValues\n} from './program/heatmap_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {HeatmapStyleLayer} from '../style/style_layer/heatmap_style_layer';\nimport type {HeatmapBucket} from '../data/bucket/heatmap_bucket';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nexport function drawHeatmap(painter: Painter, sourceCache: SourceCache, layer: HeatmapStyleLayer, coords: Array) {\n if (layer.paint.get('heatmap-opacity') === 0) {\n return;\n }\n\n if (painter.renderPass === 'offscreen') {\n const context = painter.context;\n const gl = context.gl;\n\n // Allow kernels to be drawn across boundaries, so that\n // large kernels are not clipped to tiles\n const stencilMode = StencilMode.disabled;\n // Turn on additive blending for kernels, which is a key aspect of kernel density estimation formula\n const colorMode = new ColorMode([gl.ONE, gl.ONE], Color.transparent, [true, true, true, true]);\n\n bindFramebuffer(context, painter, layer);\n\n context.clear({color: Color.transparent});\n\n for (let i = 0; i < coords.length; i++) {\n const coord = coords[i];\n\n // Skip tiles that have uncovered parents to avoid flickering; we don't need\n // to use complex tile masking here because the change between zoom levels is subtle,\n // so it's fine to simply render the parent until all its 4 children are loaded\n if (sourceCache.hasRenderableParent(coord)) continue;\n\n const tile = sourceCache.getTile(coord);\n const bucket: HeatmapBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const program = painter.useProgram('heatmap', programConfiguration);\n const {zoom} = painter.transform;\n\n program.draw(context, gl.TRIANGLES, DepthMode.disabled, stencilMode, colorMode, CullFaceMode.disabled,\n heatmapUniformValues(coord.posMatrix, tile, zoom, layer.paint.get('heatmap-intensity')), null,\n layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer,\n bucket.segments, layer.paint, painter.transform.zoom,\n programConfiguration);\n }\n\n context.viewport.set([0, 0, painter.width, painter.height]);\n\n } else if (painter.renderPass === 'translucent') {\n painter.context.setColorMode(painter.colorModeForRenderPass());\n renderTextureToMap(painter, layer);\n }\n}\n\nfunction bindFramebuffer(context: Context, painter: Painter, layer: HeatmapStyleLayer) {\n const gl = context.gl;\n context.activeTexture.set(gl.TEXTURE1);\n\n // Use a 4x downscaled screen texture for better performance\n context.viewport.set([0, 0, painter.width / 4, painter.height / 4]);\n\n let fbo = layer.heatmapFbo;\n\n if (!fbo) {\n const texture = gl.createTexture();\n gl.bindTexture(gl.TEXTURE_2D, texture);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n\n fbo = layer.heatmapFbo = context.createFramebuffer(painter.width / 4, painter.height / 4, false, false);\n\n bindTextureToFramebuffer(context, painter, texture, fbo);\n\n } else {\n gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get());\n context.bindFramebuffer.set(fbo.framebuffer);\n }\n}\n\nfunction bindTextureToFramebuffer(context: Context, painter: Painter, texture: WebGLTexture, fbo: Framebuffer) {\n const gl = context.gl;\n // Use the higher precision half-float texture where available (producing much smoother looking heatmaps);\n // Otherwise, fall back to a low precision texture\n\n const numType = context.HALF_FLOAT ?? gl.UNSIGNED_BYTE;\n const internalFormat = context.RGBA16F ?? gl.RGBA;\n\n gl.texImage2D(gl.TEXTURE_2D, 0, internalFormat, painter.width / 4, painter.height / 4, 0, gl.RGBA, numType, null);\n fbo.colorAttachment.set(texture);\n}\n\nfunction renderTextureToMap(painter: Painter, layer: HeatmapStyleLayer) {\n const context = painter.context;\n const gl = context.gl;\n\n // Here we bind two different textures from which we'll sample in drawing\n // heatmaps: the kernel texture, prepared in the offscreen pass, and a\n // color ramp texture.\n const fbo = layer.heatmapFbo;\n if (!fbo) return;\n context.activeTexture.set(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get());\n\n context.activeTexture.set(gl.TEXTURE1);\n let colorRampTexture = layer.colorRampTexture;\n if (!colorRampTexture) {\n colorRampTexture = layer.colorRampTexture = new Texture(context, layer.colorRamp, gl.RGBA);\n }\n colorRampTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n\n painter.useProgram('heatmapTexture').draw(context, gl.TRIANGLES,\n DepthMode.disabled, StencilMode.disabled, painter.colorModeForRenderPass(), CullFaceMode.disabled,\n heatmapTextureUniformValues(painter, layer, 0, 1), null,\n layer.id, painter.viewportBuffer, painter.quadTriangleIndexBuffer,\n painter.viewportSegments, layer.paint, painter.transform.zoom);\n}\n","import {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Texture} from './texture';\nimport {\n lineUniformValues,\n linePatternUniformValues,\n lineSDFUniformValues,\n lineGradientUniformValues\n} from './program/line_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {LineStyleLayer} from '../style/style_layer/line_style_layer';\nimport type {LineBucket} from '../data/bucket/line_bucket';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport {clamp, nextPowerOfTwo} from '../util/util';\nimport {renderColorRamp} from '../util/color_ramp';\nimport {EXTENT} from '../data/extent';\n\nexport function drawLine(painter: Painter, sourceCache: SourceCache, layer: LineStyleLayer, coords: Array) {\n if (painter.renderPass !== 'translucent') return;\n\n const opacity = layer.paint.get('line-opacity');\n const width = layer.paint.get('line-width');\n if (opacity.constantOr(1) === 0 || width.constantOr(1) === 0) return;\n\n const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n const colorMode = painter.colorModeForRenderPass();\n\n const dasharray = layer.paint.get('line-dasharray');\n const patternProperty = layer.paint.get('line-pattern');\n const image = patternProperty.constantOr(1 as any);\n\n const gradient = layer.paint.get('line-gradient');\n const crossfade = layer.getCrossfadeParameters();\n\n const programId =\n image ? 'linePattern' :\n dasharray ? 'lineSDF' :\n gradient ? 'lineGradient' : 'line';\n\n const context = painter.context;\n const gl = context.gl;\n\n let firstTile = true;\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n\n if (image && !tile.patternsLoaded()) continue;\n\n const bucket: LineBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const prevProgram = painter.context.program.get();\n const program = painter.useProgram(programId, programConfiguration);\n const programChanged = firstTile || program.program !== prevProgram;\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n const constantPattern = patternProperty.constantOr(null);\n if (constantPattern && tile.imageAtlas) {\n const atlas = tile.imageAtlas;\n const posTo = atlas.patternPositions[constantPattern.to.toString()];\n const posFrom = atlas.patternPositions[constantPattern.from.toString()];\n if (posTo && posFrom) programConfiguration.setConstantPatternPositions(posTo, posFrom);\n }\n\n const terrainCoord = terrainData ? coord : null;\n const uniformValues = image ? linePatternUniformValues(painter, tile, layer, crossfade, terrainCoord) :\n dasharray ? lineSDFUniformValues(painter, tile, layer, dasharray, crossfade, terrainCoord) :\n gradient ? lineGradientUniformValues(painter, tile, layer, bucket.lineClipsArray.length, terrainCoord) :\n lineUniformValues(painter, tile, layer, terrainCoord);\n\n if (image) {\n context.activeTexture.set(gl.TEXTURE0);\n tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n programConfiguration.updatePaintBuffers(crossfade);\n } else if (dasharray && (programChanged || painter.lineAtlas.dirty)) {\n context.activeTexture.set(gl.TEXTURE0);\n painter.lineAtlas.bind(context);\n } else if (gradient) {\n const layerGradient = bucket.gradients[layer.id];\n let gradientTexture = layerGradient.texture;\n if (layer.gradientVersion !== layerGradient.version) {\n let textureResolution = 256;\n if (layer.stepInterpolant) {\n const sourceMaxZoom = sourceCache.getSource().maxzoom;\n const potentialOverzoom = coord.canonical.z === sourceMaxZoom ?\n Math.ceil(1 << (painter.transform.maxZoom - coord.canonical.z)) : 1;\n const lineLength = bucket.maxLineLength / EXTENT;\n // Logical pixel tile size is 512px, and 1024px right before current zoom + 1\n const maxTilePixelSize = 1024;\n // Maximum possible texture coverage heuristic, bound by hardware max texture size\n const maxTextureCoverage = lineLength * maxTilePixelSize * potentialOverzoom;\n textureResolution = clamp(nextPowerOfTwo(maxTextureCoverage), 256, context.maxTextureSize);\n }\n layerGradient.gradient = renderColorRamp({\n expression: layer.gradientExpression(),\n evaluationKey: 'lineProgress',\n resolution: textureResolution,\n image: layerGradient.gradient || undefined,\n clips: bucket.lineClipsArray\n });\n if (layerGradient.texture) {\n layerGradient.texture.update(layerGradient.gradient);\n } else {\n layerGradient.texture = new Texture(context, layerGradient.gradient, gl.RGBA);\n }\n layerGradient.version = layer.gradientVersion;\n gradientTexture = layerGradient.texture;\n }\n context.activeTexture.set(gl.TEXTURE0);\n gradientTexture.bind(layer.stepInterpolant ? gl.NEAREST : gl.LINEAR, gl.CLAMP_TO_EDGE);\n }\n\n program.draw(context, gl.TRIANGLES, depthMode,\n painter.stencilModeForClipping(coord), colorMode, CullFaceMode.disabled, uniformValues, terrainData,\n layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer, bucket.segments,\n layer.paint, painter.transform.zoom, programConfiguration, bucket.layoutVertexBuffer2);\n\n firstTile = false;\n // once refactored so that bound texture state is managed, we'll also be able to remove this firstTile/programChanged logic\n }\n}\n","import type {CrossFaded} from '../style/properties';\nimport type {ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport type {Tile} from '../source/tile';\nimport type {ProgramConfiguration} from '../data/program_configuration';\nimport type {FillExtrusionStyleLayer} from '../style/style_layer/fill_extrusion_style_layer';\nimport type {FillStyleLayer} from '../style/style_layer/fill_style_layer';\n\n/**\n * A simple helper shared by draw_fill and draw_fill_extrusions to find the correct pattern positions AND update program.\n * For transtionable properties, especially 'fill-pattern' and 'fill-extrusion-pattern', while rendering certain frames\n * tile.imageAtlas has been updated by worker to hold the new pattern only, but rendering code is still looking for the previous image.\n * The mismatch was causing setConstantPatternPositions method not being called and pixelRatio was always the\n * default of 1, instead of actual values set by original map.addImage.\n *\n * @param programConfiguration - to be used to set pattern position and device pixel ratio.\n * @param propertyName - 'fill-pattern' or 'fill-extrusion-pattern' property key\n * @param constantPattern - either 'fill-pattern' or 'fill-extrusion-pattern' property value\n * @param tile - current tile being drawn\n * @param layer - current layer being rendered\n */\nexport function updatePatternPositionsInProgram(\n programConfiguration: ProgramConfiguration,\n propertyName: 'fill-pattern' | 'fill-extrusion-pattern',\n constantPattern: CrossFaded,\n tile: Tile,\n layer: FillStyleLayer | FillExtrusionStyleLayer): void {\n\n if (!constantPattern || !tile || !tile.imageAtlas) {\n return;\n }\n\n const patternPositions = tile.imageAtlas.patternPositions;\n let posTo = patternPositions[constantPattern.to.toString()];\n let posFrom = patternPositions[constantPattern.from.toString()];\n\n // https://github.com/maplibre/maplibre-gl-js/issues/3377\n if (!posTo && posFrom) posTo = posFrom;\n if (!posFrom && posTo) posFrom = posTo;\n\n // try again in case patternPositions has been updated by worker\n if (!posTo || !posFrom) {\n const transitioned = layer.getPaintProperty(propertyName) as string;\n posTo = patternPositions[transitioned];\n posFrom = patternPositions[transitioned];\n }\n\n if (posTo && posFrom) {\n programConfiguration.setConstantPatternPositions(posTo, posFrom);\n }\n}\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {\n fillUniformValues,\n fillPatternUniformValues,\n fillOutlineUniformValues,\n fillOutlinePatternUniformValues\n} from './program/fill_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {FillStyleLayer} from '../style/style_layer/fill_style_layer';\nimport type {FillBucket} from '../data/bucket/fill_bucket';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport {updatePatternPositionsInProgram} from './update_pattern_positions_in_program';\n\nexport function drawFill(painter: Painter, sourceCache: SourceCache, layer: FillStyleLayer, coords: Array) {\n const color = layer.paint.get('fill-color');\n const opacity = layer.paint.get('fill-opacity');\n\n if (opacity.constantOr(1) === 0) {\n return;\n }\n\n const colorMode = painter.colorModeForRenderPass();\n\n const pattern = layer.paint.get('fill-pattern');\n const pass = painter.opaquePassEnabledForLayer() &&\n (!pattern.constantOr(1 as any) &&\n color.constantOr(Color.transparent).a === 1 &&\n opacity.constantOr(0) === 1) ? 'opaque' : 'translucent';\n\n // Draw fill\n if (painter.renderPass === pass) {\n const depthMode = painter.depthModeForSublayer(\n 1, painter.renderPass === 'opaque' ? DepthMode.ReadWrite : DepthMode.ReadOnly);\n drawFillTiles(painter, sourceCache, layer, coords, depthMode, colorMode, false);\n }\n\n // Draw stroke\n if (painter.renderPass === 'translucent' && layer.paint.get('fill-antialias')) {\n\n // If we defined a different color for the fill outline, we are\n // going to ignore the bits in 0x07 and just care about the global\n // clipping mask.\n // Otherwise, we only want to drawFill the antialiased parts that are\n // *outside* the current shape. This is important in case the fill\n // or stroke color is translucent. If we wouldn't clip to outside\n // the current shape, some pixels from the outline stroke overlapped\n // the (non-antialiased) fill.\n const depthMode = painter.depthModeForSublayer(\n layer.getPaintProperty('fill-outline-color') ? 2 : 0, DepthMode.ReadOnly);\n drawFillTiles(painter, sourceCache, layer, coords, depthMode, colorMode, true);\n }\n}\n\nfunction drawFillTiles(\n painter: Painter,\n sourceCache: SourceCache,\n layer: FillStyleLayer,\n coords: Array,\n depthMode: Readonly,\n colorMode: Readonly,\n isOutline: boolean) {\n const gl = painter.context.gl;\n const fillPropertyName = 'fill-pattern';\n const patternProperty = layer.paint.get(fillPropertyName);\n const image = patternProperty && patternProperty.constantOr(1 as any);\n const crossfade = layer.getCrossfadeParameters();\n let drawMode, programName, uniformValues, indexBuffer, segments;\n\n if (!isOutline) {\n programName = image ? 'fillPattern' : 'fill';\n drawMode = gl.TRIANGLES;\n } else {\n programName = image && !layer.getPaintProperty('fill-outline-color') ? 'fillOutlinePattern' : 'fillOutline';\n drawMode = gl.LINES;\n }\n\n const constantPattern = patternProperty.constantOr(null);\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n if (image && !tile.patternsLoaded()) continue;\n\n const bucket: FillBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const program = painter.useProgram(programName, programConfiguration);\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n if (image) {\n painter.context.activeTexture.set(gl.TEXTURE0);\n tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n programConfiguration.updatePaintBuffers(crossfade);\n }\n\n updatePatternPositionsInProgram(programConfiguration, fillPropertyName, constantPattern, tile, layer);\n\n const terrainCoord = terrainData ? coord : null;\n const posMatrix = terrainCoord ? terrainCoord.posMatrix : coord.posMatrix;\n const tileMatrix = painter.translatePosMatrix(posMatrix, tile,\n layer.paint.get('fill-translate'), layer.paint.get('fill-translate-anchor'));\n\n if (!isOutline) {\n indexBuffer = bucket.indexBuffer;\n segments = bucket.segments;\n uniformValues = image ?\n fillPatternUniformValues(tileMatrix, painter, crossfade, tile) :\n fillUniformValues(tileMatrix);\n } else {\n indexBuffer = bucket.indexBuffer2;\n segments = bucket.segments2;\n const drawingBufferSize = [gl.drawingBufferWidth, gl.drawingBufferHeight] as [number, number];\n uniformValues = (programName === 'fillOutlinePattern' && image) ?\n fillOutlinePatternUniformValues(tileMatrix, painter, crossfade, tile, drawingBufferSize) :\n fillOutlineUniformValues(tileMatrix, drawingBufferSize);\n }\n\n program.draw(painter.context, drawMode, depthMode,\n painter.stencilModeForClipping(coord), colorMode, CullFaceMode.disabled, uniformValues, terrainData,\n layer.id, bucket.layoutVertexBuffer, indexBuffer, segments,\n layer.paint, painter.transform.zoom, programConfiguration);\n }\n}\n","import {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {\n fillExtrusionUniformValues,\n fillExtrusionPatternUniformValues,\n} from './program/fill_extrusion_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {FillExtrusionStyleLayer} from '../style/style_layer/fill_extrusion_style_layer';\nimport type {FillExtrusionBucket} from '../data/bucket/fill_extrusion_bucket';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nimport {updatePatternPositionsInProgram} from './update_pattern_positions_in_program';\n\nexport function drawFillExtrusion(painter: Painter, source: SourceCache, layer: FillExtrusionStyleLayer, coords: Array) {\n const opacity = layer.paint.get('fill-extrusion-opacity');\n if (opacity === 0) {\n return;\n }\n\n if (painter.renderPass === 'translucent') {\n const depthMode = new DepthMode(painter.context.gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D);\n\n if (opacity === 1 && !layer.paint.get('fill-extrusion-pattern').constantOr(1 as any)) {\n const colorMode = painter.colorModeForRenderPass();\n drawExtrusionTiles(painter, source, layer, coords, depthMode, StencilMode.disabled, colorMode);\n\n } else {\n // Draw transparent buildings in two passes so that only the closest surface is drawn.\n // First draw all the extrusions into only the depth buffer. No colors are drawn.\n drawExtrusionTiles(painter, source, layer, coords, depthMode,\n StencilMode.disabled,\n ColorMode.disabled);\n\n // Then draw all the extrusions a second type, only coloring fragments if they have the\n // same depth value as the closest fragment in the previous pass. Use the stencil buffer\n // to prevent the second draw in cases where we have coincident polygons.\n drawExtrusionTiles(painter, source, layer, coords, depthMode,\n painter.stencilModeFor3D(),\n painter.colorModeForRenderPass());\n }\n }\n}\n\nfunction drawExtrusionTiles(\n painter: Painter,\n source: SourceCache,\n layer: FillExtrusionStyleLayer,\n coords: OverscaledTileID[],\n depthMode: DepthMode,\n stencilMode: Readonly,\n colorMode: Readonly) {\n const context = painter.context;\n const gl = context.gl;\n const fillPropertyName = 'fill-extrusion-pattern';\n const patternProperty = layer.paint.get(fillPropertyName);\n const image = patternProperty.constantOr(1 as any);\n const crossfade = layer.getCrossfadeParameters();\n const opacity = layer.paint.get('fill-extrusion-opacity');\n const constantPattern = patternProperty.constantOr(null);\n for (const coord of coords) {\n const tile = source.getTile(coord);\n const bucket: FillExtrusionBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const program = painter.useProgram(image ? 'fillExtrusionPattern' : 'fillExtrusion', programConfiguration);\n\n if (image) {\n painter.context.activeTexture.set(gl.TEXTURE0);\n tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n programConfiguration.updatePaintBuffers(crossfade);\n }\n\n updatePatternPositionsInProgram(programConfiguration, fillPropertyName, constantPattern, tile, layer);\n\n const matrix = painter.translatePosMatrix(\n coord.posMatrix,\n tile,\n layer.paint.get('fill-extrusion-translate'),\n layer.paint.get('fill-extrusion-translate-anchor'));\n\n const shouldUseVerticalGradient = layer.paint.get('fill-extrusion-vertical-gradient');\n const uniformValues = image ?\n fillExtrusionPatternUniformValues(matrix, painter, shouldUseVerticalGradient, opacity, coord, crossfade, tile) :\n fillExtrusionUniformValues(matrix, painter, shouldUseVerticalGradient, opacity);\n\n program.draw(context, context.gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.backCCW,\n uniformValues, terrainData, layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer,\n bucket.segments, layer.paint, painter.transform.zoom,\n programConfiguration, painter.style.map.terrain && bucket.centroidVertexBuffer);\n }\n}\n","import {Texture} from './texture';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {Tile} from '../source/tile';\nimport {\n hillshadeUniformValues,\n hillshadeUniformPrepareValues\n} from './program/hillshade_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {HillshadeStyleLayer} from '../style/style_layer/hillshade_style_layer';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nexport function drawHillshade(painter: Painter, sourceCache: SourceCache, layer: HillshadeStyleLayer, tileIDs: Array) {\n if (painter.renderPass !== 'offscreen' && painter.renderPass !== 'translucent') return;\n\n const context = painter.context;\n\n const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n const colorMode = painter.colorModeForRenderPass();\n\n const [stencilModes, coords] = painter.renderPass === 'translucent' ?\n painter.stencilConfigForOverlap(tileIDs) : [{}, tileIDs];\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n if (typeof tile.needsHillshadePrepare !== 'undefined' && tile.needsHillshadePrepare && painter.renderPass === 'offscreen') {\n prepareHillshade(painter, tile, layer, depthMode, StencilMode.disabled, colorMode);\n } else if (painter.renderPass === 'translucent') {\n renderHillshade(painter, coord, tile, layer, depthMode, stencilModes[coord.overscaledZ], colorMode);\n }\n }\n\n context.viewport.set([0, 0, painter.width, painter.height]);\n}\n\nfunction renderHillshade(\n painter: Painter,\n coord: OverscaledTileID,\n tile: Tile,\n layer: HillshadeStyleLayer,\n depthMode: Readonly,\n stencilMode: Readonly,\n colorMode: Readonly) {\n const context = painter.context;\n const gl = context.gl;\n const fbo = tile.fbo;\n if (!fbo) return;\n\n const program = painter.useProgram('hillshade');\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n context.activeTexture.set(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get());\n\n const terrainCoord = terrainData ? coord : null;\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n hillshadeUniformValues(painter, tile, layer, terrainCoord), terrainData, layer.id, painter.rasterBoundsBuffer,\n painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments);\n\n}\n\n// hillshade rendering is done in two steps. the prepare step first calculates the slope of the terrain in the x and y\n// directions for each pixel, and saves those values to a framebuffer texture in the r and g channels.\nfunction prepareHillshade(\n painter: Painter,\n tile: Tile,\n layer: HillshadeStyleLayer,\n depthMode: Readonly,\n stencilMode: Readonly,\n colorMode: Readonly) {\n const context = painter.context;\n const gl = context.gl;\n const dem = tile.dem;\n if (dem && dem.data) {\n const tileSize = dem.dim;\n const textureStride = dem.stride;\n\n const pixelData = dem.getPixels();\n context.activeTexture.set(gl.TEXTURE1);\n\n context.pixelStoreUnpackPremultiplyAlpha.set(false);\n tile.demTexture = tile.demTexture || painter.getTileTexture(textureStride);\n if (tile.demTexture) {\n const demTexture = tile.demTexture;\n demTexture.update(pixelData, {premultiply: false});\n demTexture.bind(gl.NEAREST, gl.CLAMP_TO_EDGE);\n } else {\n tile.demTexture = new Texture(context, pixelData, gl.RGBA, {premultiply: false});\n tile.demTexture.bind(gl.NEAREST, gl.CLAMP_TO_EDGE);\n }\n\n context.activeTexture.set(gl.TEXTURE0);\n\n let fbo = tile.fbo;\n\n if (!fbo) {\n const renderTexture = new Texture(context, {width: tileSize, height: tileSize, data: null}, gl.RGBA);\n renderTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n\n fbo = tile.fbo = context.createFramebuffer(tileSize, tileSize, true, false);\n fbo.colorAttachment.set(renderTexture.texture);\n }\n\n context.bindFramebuffer.set(fbo.framebuffer);\n context.viewport.set([0, 0, tileSize, tileSize]);\n\n painter.useProgram('hillshadePrepare').draw(context, gl.TRIANGLES,\n depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n hillshadeUniformPrepareValues(tile.tileID, dem),\n null, layer.id, painter.rasterBoundsBuffer,\n painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments);\n\n tile.needsHillshadePrepare = false;\n }\n}\n","import {clamp} from '../util/util';\n\nimport {ImageSource} from '../source/image_source';\nimport {browser} from '../util/browser';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {rasterUniformValues} from './program/raster_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {RasterStyleLayer} from '../style/style_layer/raster_style_layer';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nexport function drawRaster(painter: Painter, sourceCache: SourceCache, layer: RasterStyleLayer, tileIDs: Array) {\n if (painter.renderPass !== 'translucent') return;\n if (layer.paint.get('raster-opacity') === 0) return;\n if (!tileIDs.length) return;\n\n const context = painter.context;\n const gl = context.gl;\n const source = sourceCache.getSource();\n const program = painter.useProgram('raster');\n\n const colorMode = painter.colorModeForRenderPass();\n\n const [stencilModes, coords] = source instanceof ImageSource ? [{}, tileIDs] :\n painter.stencilConfigForOverlap(tileIDs);\n\n const minTileZ = coords[coords.length - 1].overscaledZ;\n\n const align = !painter.options.moving;\n for (const coord of coords) {\n // Set the lower zoom level to sublayer 0, and higher zoom levels to higher sublayers\n // Use gl.LESS to prevent double drawing in areas where tiles overlap.\n const depthMode = painter.depthModeForSublayer(coord.overscaledZ - minTileZ,\n layer.paint.get('raster-opacity') === 1 ? DepthMode.ReadWrite : DepthMode.ReadOnly, gl.LESS);\n\n const tile = sourceCache.getTile(coord);\n\n tile.registerFadeDuration(layer.paint.get('raster-fade-duration'));\n\n const parentTile = sourceCache.findLoadedParent(coord, 0),\n fade = getFadeValues(tile, parentTile, sourceCache, layer, painter.transform, painter.style.map.terrain);\n\n let parentScaleBy, parentTL;\n\n const textureFilter = layer.paint.get('raster-resampling') === 'nearest' ? gl.NEAREST : gl.LINEAR;\n\n context.activeTexture.set(gl.TEXTURE0);\n tile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST);\n\n context.activeTexture.set(gl.TEXTURE1);\n\n if (parentTile) {\n parentTile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST);\n parentScaleBy = Math.pow(2, parentTile.tileID.overscaledZ - tile.tileID.overscaledZ);\n parentTL = [tile.tileID.canonical.x * parentScaleBy % 1, tile.tileID.canonical.y * parentScaleBy % 1];\n\n } else {\n tile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST);\n }\n\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n const terrainCoord = terrainData ? coord : null;\n const posMatrix = terrainCoord ? terrainCoord.posMatrix : painter.transform.calculatePosMatrix(coord.toUnwrapped(), align);\n const uniformValues = rasterUniformValues(posMatrix, parentTL || [0, 0], parentScaleBy || 1, fade, layer);\n\n if (source instanceof ImageSource) {\n program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id, source.boundsBuffer,\n painter.quadTriangleIndexBuffer, source.boundsSegments);\n } else {\n program.draw(context, gl.TRIANGLES, depthMode, stencilModes[coord.overscaledZ], colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id, painter.rasterBoundsBuffer,\n painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments);\n }\n }\n}\n\nfunction getFadeValues(tile, parentTile, sourceCache, layer, transform, terrain) {\n const fadeDuration = layer.paint.get('raster-fade-duration');\n\n if (!terrain && fadeDuration > 0) {\n const now = browser.now();\n const sinceTile = (now - tile.timeAdded) / fadeDuration;\n const sinceParent = parentTile ? (now - parentTile.timeAdded) / fadeDuration : -1;\n\n const source = sourceCache.getSource();\n const idealZ = transform.coveringZoomLevel({\n tileSize: source.tileSize,\n roundZoom: source.roundZoom\n });\n\n // if no parent or parent is older, fade in; if parent is younger, fade out\n const fadeIn = !parentTile || Math.abs(parentTile.tileID.overscaledZ - idealZ) > Math.abs(tile.tileID.overscaledZ - idealZ);\n\n const childOpacity = (fadeIn && tile.refreshedUponExpiration) ? 1 : clamp(fadeIn ? sinceTile : 1 - sinceParent, 0, 1);\n\n // we don't crossfade tiles that were just refreshed upon expiring:\n // once they're old enough to pass the crossfading threshold\n // (fadeDuration), unset the `refreshedUponExpiration` flag so we don't\n // incorrectly fail to crossfade them when zooming\n if (tile.refreshedUponExpiration && sinceTile >= 1) tile.refreshedUponExpiration = false;\n\n if (parentTile) {\n return {\n opacity: 1,\n mix: 1 - childOpacity\n };\n } else {\n return {\n opacity: childOpacity,\n mix: 0\n };\n }\n } else {\n return {\n opacity: 1,\n mix: 0\n };\n }\n}\n","import {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {\n backgroundUniformValues,\n backgroundPatternUniformValues\n} from './program/background_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {BackgroundStyleLayer} from '../style/style_layer/background_style_layer';\nimport {OverscaledTileID} from '../source/tile_id';\n\nexport function drawBackground(painter: Painter, sourceCache: SourceCache, layer: BackgroundStyleLayer, coords?: Array) {\n const color = layer.paint.get('background-color');\n const opacity = layer.paint.get('background-opacity');\n\n if (opacity === 0) return;\n\n const context = painter.context;\n const gl = context.gl;\n const transform = painter.transform;\n const tileSize = transform.tileSize;\n const image = layer.paint.get('background-pattern');\n if (painter.isPatternMissing(image)) return;\n\n const pass = (!image && color.a === 1 && opacity === 1 && painter.opaquePassEnabledForLayer()) ? 'opaque' : 'translucent';\n if (painter.renderPass !== pass) return;\n\n const stencilMode = StencilMode.disabled;\n const depthMode = painter.depthModeForSublayer(0, pass === 'opaque' ? DepthMode.ReadWrite : DepthMode.ReadOnly);\n const colorMode = painter.colorModeForRenderPass();\n const program = painter.useProgram(image ? 'backgroundPattern' : 'background');\n const tileIDs = coords ? coords : transform.coveringTiles({tileSize, terrain: painter.style.map.terrain});\n\n if (image) {\n context.activeTexture.set(gl.TEXTURE0);\n painter.imageManager.bind(painter.context);\n }\n\n const crossfade = layer.getCrossfadeParameters();\n for (const tileID of tileIDs) {\n const matrix = coords ? tileID.posMatrix : painter.transform.calculatePosMatrix(tileID.toUnwrapped());\n const uniformValues = image ?\n backgroundPatternUniformValues(matrix, opacity, painter, image, {tileID, tileSize}, crossfade) :\n backgroundUniformValues(matrix, opacity, color);\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(tileID);\n\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id, painter.tileExtentBuffer,\n painter.quadTriangleIndexBuffer, painter.tileExtentSegments);\n }\n}\n","import {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {debugUniformValues} from './program/debug_program';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {ColorMode} from '../gl/color_mode';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport {Style} from '../style/style';\n\nconst topColor = new Color(1, 0, 0, 1);\nconst btmColor = new Color(0, 1, 0, 1);\nconst leftColor = new Color(0, 0, 1, 1);\nconst rightColor = new Color(1, 0, 1, 1);\nconst centerColor = new Color(0, 1, 1, 1);\n\nexport function drawDebugPadding(painter: Painter) {\n const padding = painter.transform.padding;\n const lineWidth = 3;\n // Top\n drawHorizontalLine(painter, painter.transform.height - (padding.top || 0), lineWidth, topColor);\n // Bottom\n drawHorizontalLine(painter, padding.bottom || 0, lineWidth, btmColor);\n // Left\n drawVerticalLine(painter, padding.left || 0, lineWidth, leftColor);\n // Right\n drawVerticalLine(painter, painter.transform.width - (padding.right || 0), lineWidth, rightColor);\n // Center\n const center = painter.transform.centerPoint;\n drawCrosshair(painter, center.x, painter.transform.height - center.y, centerColor);\n}\n\nfunction drawCrosshair(painter: Painter, x: number, y: number, color: Color) {\n const size = 20;\n const lineWidth = 2;\n //Vertical line\n drawDebugSSRect(painter, x - lineWidth / 2, y - size / 2, lineWidth, size, color);\n //Horizontal line\n drawDebugSSRect(painter, x - size / 2, y - lineWidth / 2, size, lineWidth, color);\n}\n\nfunction drawHorizontalLine(painter: Painter, y: number, lineWidth: number, color: Color) {\n drawDebugSSRect(painter, 0, y + lineWidth / 2, painter.transform.width, lineWidth, color);\n}\n\nfunction drawVerticalLine(painter: Painter, x: number, lineWidth: number, color: Color) {\n drawDebugSSRect(painter, x - lineWidth / 2, 0, lineWidth, painter.transform.height, color);\n}\n\nfunction drawDebugSSRect(painter: Painter, x: number, y: number, width: number, height: number, color: Color) {\n const context = painter.context;\n const gl = context.gl;\n\n gl.enable(gl.SCISSOR_TEST);\n gl.scissor(x * painter.pixelRatio, y * painter.pixelRatio, width * painter.pixelRatio, height * painter.pixelRatio);\n context.clear({color});\n gl.disable(gl.SCISSOR_TEST);\n}\n\nexport function drawDebug(painter: Painter, sourceCache: SourceCache, coords: Array) {\n for (let i = 0; i < coords.length; i++) {\n drawDebugTile(painter, sourceCache, coords[i]);\n }\n}\n\nfunction drawDebugTile(painter: Painter, sourceCache: SourceCache, coord: OverscaledTileID) {\n const context = painter.context;\n const gl = context.gl;\n\n const posMatrix = coord.posMatrix;\n const program = painter.useProgram('debug');\n\n const depthMode = DepthMode.disabled;\n const stencilMode = StencilMode.disabled;\n const colorMode = painter.colorModeForRenderPass();\n const id = '$debug';\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n context.activeTexture.set(gl.TEXTURE0);\n\n const tileRawData = sourceCache.getTileByID(coord.key).latestRawTileData;\n const tileByteLength = (tileRawData && tileRawData.byteLength) || 0;\n const tileSizeKb = Math.floor(tileByteLength / 1024);\n const tileSize = sourceCache.getTile(coord).tileSize;\n const scaleRatio = (512 / Math.min(tileSize, 512) * (coord.overscaledZ / painter.transform.zoom)) * 0.5;\n let tileIdText = coord.canonical.toString();\n if (coord.overscaledZ !== coord.canonical.z) {\n tileIdText += ` => ${coord.overscaledZ}`;\n }\n const tileLabel = `${tileIdText} ${tileSizeKb}kB`;\n drawTextToOverlay(painter, tileLabel);\n\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, ColorMode.alphaBlended, CullFaceMode.disabled,\n debugUniformValues(posMatrix, Color.transparent, scaleRatio), null, id,\n painter.debugBuffer, painter.quadTriangleIndexBuffer, painter.debugSegments);\n program.draw(context, gl.LINE_STRIP, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n debugUniformValues(posMatrix, Color.red), terrainData, id,\n painter.debugBuffer, painter.tileBorderIndexBuffer, painter.debugSegments);\n}\n\nfunction drawTextToOverlay(painter: Painter, text: string) {\n painter.initDebugOverlayCanvas();\n const canvas = painter.debugOverlayCanvas;\n const gl = painter.context.gl;\n const ctx2d = painter.debugOverlayCanvas.getContext('2d');\n ctx2d.clearRect(0, 0, canvas.width, canvas.height);\n\n ctx2d.shadowColor = 'white';\n ctx2d.shadowBlur = 2;\n ctx2d.lineWidth = 1.5;\n ctx2d.strokeStyle = 'white';\n ctx2d.textBaseline = 'top';\n ctx2d.font = `bold ${36}px Open Sans, sans-serif`;\n ctx2d.fillText(text, 5, 5);\n ctx2d.strokeText(text, 5, 5);\n\n painter.debugOverlayTexture.update(canvas);\n painter.debugOverlayTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n}\n\nexport function selectDebugSource(style: Style, zoom: number): SourceCache | null {\n // Use vector source with highest maxzoom\n // Else use source with highest maxzoom of any type\n let selectedSource: SourceCache = null;\n const layers = Object.values(style._layers);\n const sources = layers.flatMap((layer) => {\n if (layer.source && !layer.isHidden(zoom)) {\n const sourceCache = style.sourceCaches[layer.source];\n return [sourceCache];\n } else {\n return [];\n }\n });\n const vectorSources = sources.filter((source) => source.getSource().type === 'vector');\n const otherSources = sources.filter((source) => source.getSource().type !== 'vector');\n const considerSource = (source: SourceCache) => {\n if (!selectedSource || (selectedSource.getSource().maxzoom < source.getSource().maxzoom)) {\n selectedSource = source;\n }\n };\n vectorSources.forEach((source) => considerSource(source));\n if (!selectedSource) {\n otherSources.forEach((source) => considerSource(source));\n }\n return selectedSource;\n}\n","import {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {CustomStyleLayer} from '../style/style_layer/custom_style_layer';\n\nexport function drawCustom(painter: Painter, sourceCache: SourceCache, layer: CustomStyleLayer) {\n\n const context = painter.context;\n const implementation = layer.implementation;\n\n if (painter.renderPass === 'offscreen') {\n\n const prerender = implementation.prerender;\n if (prerender) {\n painter.setCustomLayerDefaults();\n context.setColorMode(painter.colorModeForRenderPass());\n\n prerender.call(implementation, context.gl, painter.transform.customLayerMatrix());\n\n context.setDirty();\n painter.setBaseState();\n }\n\n } else if (painter.renderPass === 'translucent') {\n\n painter.setCustomLayerDefaults();\n\n context.setColorMode(painter.colorModeForRenderPass());\n context.setStencilMode(StencilMode.disabled);\n\n const depthMode = implementation.renderingMode === '3d' ?\n new DepthMode(painter.context.gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D) :\n painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n\n context.setDepthMode(depthMode);\n\n implementation.render(context.gl, painter.transform.customLayerMatrix());\n\n context.setDirty();\n painter.setBaseState();\n context.bindFramebuffer.set(null);\n }\n}\n","import {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {terrainUniformValues, terrainDepthUniformValues, terrainCoordsUniformValues} from './program/terrain_program';\nimport type {Painter} from './painter';\nimport type {Tile} from '../source/tile';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {ColorMode} from '../gl/color_mode';\nimport {Terrain} from './terrain';\n\n/**\n * Redraw the Depth Framebuffer\n * @param painter - the painter\n * @param terrain - the terrain\n */\nfunction drawDepth(painter: Painter, terrain: Terrain) {\n const context = painter.context;\n const gl = context.gl;\n const colorMode = ColorMode.unblended;\n const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, [0, 1]);\n const mesh = terrain.getTerrainMesh();\n const tiles = terrain.sourceCache.getRenderableTiles();\n const program = painter.useProgram('terrainDepth');\n context.bindFramebuffer.set(terrain.getFramebuffer('depth').framebuffer);\n context.viewport.set([0, 0, painter.width / devicePixelRatio, painter.height / devicePixelRatio]);\n context.clear({color: Color.transparent, depth: 1});\n for (const tile of tiles) {\n const terrainData = terrain.getTerrainData(tile.tileID);\n const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped());\n const uniformValues = terrainDepthUniformValues(posMatrix, terrain.getMeshFrameDelta(painter.transform.zoom));\n program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments);\n }\n context.bindFramebuffer.set(null);\n context.viewport.set([0, 0, painter.width, painter.height]);\n}\n\n/**\n * Redraw the Coords Framebuffers\n * @param painter - the painter\n * @param terrain - the terrain\n */\nfunction drawCoords(painter: Painter, terrain: Terrain) {\n const context = painter.context;\n const gl = context.gl;\n const colorMode = ColorMode.unblended;\n const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, [0, 1]);\n const mesh = terrain.getTerrainMesh();\n const coords = terrain.getCoordsTexture();\n const tiles = terrain.sourceCache.getRenderableTiles();\n\n // draw tile-coords into framebuffer\n const program = painter.useProgram('terrainCoords');\n context.bindFramebuffer.set(terrain.getFramebuffer('coords').framebuffer);\n context.viewport.set([0, 0, painter.width / devicePixelRatio, painter.height / devicePixelRatio]);\n context.clear({color: Color.transparent, depth: 1});\n terrain.coordsIndex = [];\n for (const tile of tiles) {\n const terrainData = terrain.getTerrainData(tile.tileID);\n context.activeTexture.set(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, coords.texture);\n const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped());\n const uniformValues = terrainCoordsUniformValues(posMatrix, 255 - terrain.coordsIndex.length, terrain.getMeshFrameDelta(painter.transform.zoom));\n program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments);\n terrain.coordsIndex.push(tile.tileID.key);\n }\n context.bindFramebuffer.set(null);\n context.viewport.set([0, 0, painter.width, painter.height]);\n}\n\nfunction drawTerrain(painter: Painter, terrain: Terrain, tiles: Array) {\n const context = painter.context;\n const gl = context.gl;\n const colorMode = painter.colorModeForRenderPass();\n const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D);\n const program = painter.useProgram('terrain');\n const mesh = terrain.getTerrainMesh();\n\n context.bindFramebuffer.set(null);\n context.viewport.set([0, 0, painter.width, painter.height]);\n\n for (const tile of tiles) {\n const texture = painter.renderToTexture.getTexture(tile);\n const terrainData = terrain.getTerrainData(tile.tileID);\n context.activeTexture.set(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, texture.texture);\n const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped());\n const uniformValues = terrainUniformValues(posMatrix, terrain.getMeshFrameDelta(painter.transform.zoom));\n program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments);\n }\n\n}\n\nexport {\n drawTerrain,\n drawDepth,\n drawCoords\n};\n","import {browser} from '../util/browser';\nimport {mat4, vec3} from 'gl-matrix';\nimport {SourceCache} from '../source/source_cache';\nimport {EXTENT} from '../data/extent';\nimport {pixelsToTileUnits} from '../source/pixels_to_tile_units';\nimport {SegmentVector} from '../data/segment';\nimport {RasterBoundsArray, PosArray, TriangleIndexArray, LineStripIndexArray} from '../data/array_types.g';\nimport rasterBoundsAttributes from '../data/raster_bounds_attributes';\nimport posAttributes from '../data/pos_attributes';\nimport {ProgramConfiguration} from '../data/program_configuration';\nimport {CrossTileSymbolIndex} from '../symbol/cross_tile_symbol_index';\nimport {shaders} from '../shaders/shaders';\nimport {Program} from './program';\nimport {programUniforms} from './program/program_uniforms';\nimport {Context} from '../gl/context';\nimport {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Texture} from './texture';\nimport {clippingMaskUniformValues} from './program/clipping_mask_program';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {drawSymbols} from './draw_symbol';\nimport {drawCircles} from './draw_circle';\nimport {drawHeatmap} from './draw_heatmap';\nimport {drawLine} from './draw_line';\nimport {drawFill} from './draw_fill';\nimport {drawFillExtrusion} from './draw_fill_extrusion';\nimport {drawHillshade} from './draw_hillshade';\nimport {drawRaster} from './draw_raster';\nimport {drawBackground} from './draw_background';\nimport {drawDebug, drawDebugPadding, selectDebugSource} from './draw_debug';\nimport {drawCustom} from './draw_custom';\nimport {drawDepth, drawCoords} from './draw_terrain';\nimport {OverscaledTileID} from '../source/tile_id';\n\nimport type {Transform} from '../geo/transform';\nimport type {Tile} from '../source/tile';\nimport type {Style} from '../style/style';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {CrossFaded} from '../style/properties';\nimport type {LineAtlas} from './line_atlas';\nimport type {ImageManager} from './image_manager';\nimport type {GlyphManager} from './glyph_manager';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {IndexBuffer} from '../gl/index_buffer';\nimport type {DepthRangeType, DepthMaskType, DepthFuncType} from '../gl/types';\nimport type {ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport {RenderToTexture} from './render_to_texture';\n\nexport type RenderPass = 'offscreen' | 'opaque' | 'translucent';\n\ntype PainterOptions = {\n showOverdrawInspector: boolean;\n showTileBoundaries: boolean;\n showPadding: boolean;\n rotating: boolean;\n zooming: boolean;\n moving: boolean;\n fadeDuration: number;\n};\n\n/**\n * @internal\n * Initialize a new painter object.\n */\nexport class Painter {\n context: Context;\n transform: Transform;\n renderToTexture: RenderToTexture;\n _tileTextures: {\n [_: number]: Array;\n };\n numSublayers: number;\n depthEpsilon: number;\n emptyProgramConfiguration: ProgramConfiguration;\n width: number;\n height: number;\n pixelRatio: number;\n tileExtentBuffer: VertexBuffer;\n tileExtentSegments: SegmentVector;\n debugBuffer: VertexBuffer;\n debugSegments: SegmentVector;\n rasterBoundsBuffer: VertexBuffer;\n rasterBoundsSegments: SegmentVector;\n viewportBuffer: VertexBuffer;\n viewportSegments: SegmentVector;\n quadTriangleIndexBuffer: IndexBuffer;\n tileBorderIndexBuffer: IndexBuffer;\n _tileClippingMaskIDs: {[_: string]: number};\n stencilClearMode: StencilMode;\n style: Style;\n options: PainterOptions;\n lineAtlas: LineAtlas;\n imageManager: ImageManager;\n glyphManager: GlyphManager;\n depthRangeFor3D: DepthRangeType;\n opaquePassCutoff: number;\n renderPass: RenderPass;\n currentLayer: number;\n currentStencilSource: string;\n nextStencilID: number;\n id: string;\n _showOverdrawInspector: boolean;\n cache: {[_: string]: Program};\n crossTileSymbolIndex: CrossTileSymbolIndex;\n symbolFadeChange: number;\n debugOverlayTexture: Texture;\n debugOverlayCanvas: HTMLCanvasElement;\n // this object stores the current camera-matrix and the last render time\n // of the terrain-facilitators. e.g. depth & coords framebuffers\n // every time the camera-matrix changes the terrain-facilitators will be redrawn.\n terrainFacilitator: {dirty: boolean; matrix: mat4; renderTime: number};\n\n constructor(gl: WebGLRenderingContext | WebGL2RenderingContext, transform: Transform) {\n this.context = new Context(gl);\n this.transform = transform;\n this._tileTextures = {};\n this.terrainFacilitator = {dirty: true, matrix: mat4.create(), renderTime: 0};\n\n this.setup();\n\n // Within each layer there are multiple distinct z-planes that can be drawn to.\n // This is implemented using the WebGL depth buffer.\n this.numSublayers = SourceCache.maxUnderzooming + SourceCache.maxOverzooming + 1;\n this.depthEpsilon = 1 / Math.pow(2, 16);\n\n this.crossTileSymbolIndex = new CrossTileSymbolIndex();\n }\n\n /*\n * Update the GL viewport, projection matrix, and transforms to compensate\n * for a new width and height value.\n */\n resize(width: number, height: number, pixelRatio: number) {\n this.width = Math.floor(width * pixelRatio);\n this.height = Math.floor(height * pixelRatio);\n this.pixelRatio = pixelRatio;\n this.context.viewport.set([0, 0, this.width, this.height]);\n\n if (this.style) {\n for (const layerId of this.style._order) {\n this.style._layers[layerId].resize();\n }\n }\n }\n\n setup() {\n const context = this.context;\n\n const tileExtentArray = new PosArray();\n tileExtentArray.emplaceBack(0, 0);\n tileExtentArray.emplaceBack(EXTENT, 0);\n tileExtentArray.emplaceBack(0, EXTENT);\n tileExtentArray.emplaceBack(EXTENT, EXTENT);\n this.tileExtentBuffer = context.createVertexBuffer(tileExtentArray, posAttributes.members);\n this.tileExtentSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n\n const debugArray = new PosArray();\n debugArray.emplaceBack(0, 0);\n debugArray.emplaceBack(EXTENT, 0);\n debugArray.emplaceBack(0, EXTENT);\n debugArray.emplaceBack(EXTENT, EXTENT);\n this.debugBuffer = context.createVertexBuffer(debugArray, posAttributes.members);\n this.debugSegments = SegmentVector.simpleSegment(0, 0, 4, 5);\n\n const rasterBoundsArray = new RasterBoundsArray();\n rasterBoundsArray.emplaceBack(0, 0, 0, 0);\n rasterBoundsArray.emplaceBack(EXTENT, 0, EXTENT, 0);\n rasterBoundsArray.emplaceBack(0, EXTENT, 0, EXTENT);\n rasterBoundsArray.emplaceBack(EXTENT, EXTENT, EXTENT, EXTENT);\n this.rasterBoundsBuffer = context.createVertexBuffer(rasterBoundsArray, rasterBoundsAttributes.members);\n this.rasterBoundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n\n const viewportArray = new PosArray();\n viewportArray.emplaceBack(0, 0);\n viewportArray.emplaceBack(1, 0);\n viewportArray.emplaceBack(0, 1);\n viewportArray.emplaceBack(1, 1);\n this.viewportBuffer = context.createVertexBuffer(viewportArray, posAttributes.members);\n this.viewportSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n\n const tileLineStripIndices = new LineStripIndexArray();\n tileLineStripIndices.emplaceBack(0);\n tileLineStripIndices.emplaceBack(1);\n tileLineStripIndices.emplaceBack(3);\n tileLineStripIndices.emplaceBack(2);\n tileLineStripIndices.emplaceBack(0);\n this.tileBorderIndexBuffer = context.createIndexBuffer(tileLineStripIndices);\n\n const quadTriangleIndices = new TriangleIndexArray();\n quadTriangleIndices.emplaceBack(0, 1, 2);\n quadTriangleIndices.emplaceBack(2, 1, 3);\n this.quadTriangleIndexBuffer = context.createIndexBuffer(quadTriangleIndices);\n\n const gl = this.context.gl;\n this.stencilClearMode = new StencilMode({func: gl.ALWAYS, mask: 0}, 0x0, 0xFF, gl.ZERO, gl.ZERO, gl.ZERO);\n }\n\n /*\n * Reset the drawing canvas by clearing the stencil buffer so that we can draw\n * new tiles at the same location, while retaining previously drawn pixels.\n */\n clearStencil() {\n const context = this.context;\n const gl = context.gl;\n\n this.nextStencilID = 1;\n this.currentStencilSource = undefined;\n\n // As a temporary workaround for https://github.com/mapbox/mapbox-gl-js/issues/5490,\n // pending an upstream fix, we draw a fullscreen stencil=0 clipping mask here,\n // effectively clearing the stencil buffer: once an upstream patch lands, remove\n // this function in favor of context.clear({ stencil: 0x0 })\n\n const matrix = mat4.create();\n mat4.ortho(matrix, 0, this.width, this.height, 0, 0, 1);\n mat4.scale(matrix, matrix, [gl.drawingBufferWidth, gl.drawingBufferHeight, 0]);\n\n this.useProgram('clippingMask').draw(context, gl.TRIANGLES,\n DepthMode.disabled, this.stencilClearMode, ColorMode.disabled, CullFaceMode.disabled,\n clippingMaskUniformValues(matrix), null,\n '$clipping', this.viewportBuffer,\n this.quadTriangleIndexBuffer, this.viewportSegments);\n }\n\n _renderTileClippingMasks(layer: StyleLayer, tileIDs: Array) {\n if (this.currentStencilSource === layer.source || !layer.isTileClipped() || !tileIDs || !tileIDs.length) return;\n\n this.currentStencilSource = layer.source;\n\n const context = this.context;\n const gl = context.gl;\n\n if (this.nextStencilID + tileIDs.length > 256) {\n // we'll run out of fresh IDs so we need to clear and start from scratch\n this.clearStencil();\n }\n\n context.setColorMode(ColorMode.disabled);\n context.setDepthMode(DepthMode.disabled);\n\n const program = this.useProgram('clippingMask');\n\n this._tileClippingMaskIDs = {};\n\n for (const tileID of tileIDs) {\n const id = this._tileClippingMaskIDs[tileID.key] = this.nextStencilID++;\n const terrainData = this.style.map.terrain && this.style.map.terrain.getTerrainData(tileID);\n\n program.draw(context, gl.TRIANGLES, DepthMode.disabled,\n // Tests will always pass, and ref value will be written to stencil buffer.\n new StencilMode({func: gl.ALWAYS, mask: 0}, id, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE),\n ColorMode.disabled, CullFaceMode.disabled, clippingMaskUniformValues(tileID.posMatrix),\n terrainData, '$clipping', this.tileExtentBuffer,\n this.quadTriangleIndexBuffer, this.tileExtentSegments);\n }\n }\n\n stencilModeFor3D(): StencilMode {\n this.currentStencilSource = undefined;\n\n if (this.nextStencilID + 1 > 256) {\n this.clearStencil();\n }\n\n const id = this.nextStencilID++;\n const gl = this.context.gl;\n return new StencilMode({func: gl.NOTEQUAL, mask: 0xFF}, id, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE);\n }\n\n stencilModeForClipping(tileID: OverscaledTileID): StencilMode {\n const gl = this.context.gl;\n return new StencilMode({func: gl.EQUAL, mask: 0xFF}, this._tileClippingMaskIDs[tileID.key], 0x00, gl.KEEP, gl.KEEP, gl.REPLACE);\n }\n\n /*\n * Sort coordinates by Z as drawing tiles is done in Z-descending order.\n * All children with the same Z write the same stencil value. Children\n * stencil values are greater than parent's. This is used only for raster\n * and raster-dem tiles, which are already clipped to tile boundaries, to\n * mask area of tile overlapped by children tiles.\n * Stencil ref values continue range used in _tileClippingMaskIDs.\n *\n * Returns [StencilMode for tile overscaleZ map, sortedCoords].\n */\n stencilConfigForOverlap(tileIDs: Array): [{\n [_: number]: Readonly;\n }, Array] {\n const gl = this.context.gl;\n const coords = tileIDs.sort((a, b) => b.overscaledZ - a.overscaledZ);\n const minTileZ = coords[coords.length - 1].overscaledZ;\n const stencilValues = coords[0].overscaledZ - minTileZ + 1;\n if (stencilValues > 1) {\n this.currentStencilSource = undefined;\n if (this.nextStencilID + stencilValues > 256) {\n this.clearStencil();\n }\n const zToStencilMode = {};\n for (let i = 0; i < stencilValues; i++) {\n zToStencilMode[i + minTileZ] = new StencilMode({func: gl.GEQUAL, mask: 0xFF}, i + this.nextStencilID, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE);\n }\n this.nextStencilID += stencilValues;\n return [zToStencilMode, coords];\n }\n return [{[minTileZ]: StencilMode.disabled}, coords];\n }\n\n colorModeForRenderPass(): Readonly {\n const gl = this.context.gl;\n if (this._showOverdrawInspector) {\n const numOverdrawSteps = 8;\n const a = 1 / numOverdrawSteps;\n\n return new ColorMode([gl.CONSTANT_COLOR, gl.ONE], new Color(a, a, a, 0), [true, true, true, true]);\n } else if (this.renderPass === 'opaque') {\n return ColorMode.unblended;\n } else {\n return ColorMode.alphaBlended;\n }\n }\n\n depthModeForSublayer(n: number, mask: DepthMaskType, func?: DepthFuncType | null): Readonly {\n if (!this.opaquePassEnabledForLayer()) return DepthMode.disabled;\n const depth = 1 - ((1 + this.currentLayer) * this.numSublayers + n) * this.depthEpsilon;\n return new DepthMode(func || this.context.gl.LEQUAL, mask, [depth, depth]);\n }\n\n /*\n * The opaque pass and 3D layers both use the depth buffer.\n * Layers drawn above 3D layers need to be drawn using the\n * painter's algorithm so that they appear above 3D features.\n * This returns true for layers that can be drawn using the\n * opaque pass.\n */\n opaquePassEnabledForLayer() {\n return this.currentLayer < this.opaquePassCutoff;\n }\n\n render(style: Style, options: PainterOptions) {\n this.style = style;\n this.options = options;\n\n this.lineAtlas = style.lineAtlas;\n this.imageManager = style.imageManager;\n this.glyphManager = style.glyphManager;\n\n this.symbolFadeChange = style.placement.symbolFadeChange(browser.now());\n\n this.imageManager.beginFrame();\n\n const layerIds = this.style._order;\n const sourceCaches = this.style.sourceCaches;\n\n const coordsAscending: {[_: string]: Array} = {};\n const coordsDescending: {[_: string]: Array} = {};\n const coordsDescendingSymbol: {[_: string]: Array} = {};\n\n for (const id in sourceCaches) {\n const sourceCache = sourceCaches[id];\n if (sourceCache.used) {\n sourceCache.prepare(this.context);\n }\n\n coordsAscending[id] = sourceCache.getVisibleCoordinates();\n coordsDescending[id] = coordsAscending[id].slice().reverse();\n coordsDescendingSymbol[id] = sourceCache.getVisibleCoordinates(true).reverse();\n }\n\n this.opaquePassCutoff = Infinity;\n for (let i = 0; i < layerIds.length; i++) {\n const layerId = layerIds[i];\n if (this.style._layers[layerId].is3D()) {\n this.opaquePassCutoff = i;\n break;\n }\n }\n\n if (this.renderToTexture) {\n this.renderToTexture.prepareForRender(this.style, this.transform.zoom);\n // this is disabled, because render-to-texture is rendering all layers from bottom to top.\n this.opaquePassCutoff = 0;\n\n // update coords/depth-framebuffer on camera movement, or tile reloading\n const newTiles = this.style.map.terrain.sourceCache.tilesAfterTime(this.terrainFacilitator.renderTime);\n if (this.terrainFacilitator.dirty || !mat4.equals(this.terrainFacilitator.matrix, this.transform.projMatrix) || newTiles.length) {\n mat4.copy(this.terrainFacilitator.matrix, this.transform.projMatrix);\n this.terrainFacilitator.renderTime = Date.now();\n this.terrainFacilitator.dirty = false;\n drawDepth(this, this.style.map.terrain);\n drawCoords(this, this.style.map.terrain);\n }\n }\n\n // Offscreen pass ===============================================\n // We first do all rendering that requires rendering to a separate\n // framebuffer, and then save those for rendering back to the map\n // later: in doing this we avoid doing expensive framebuffer restores.\n this.renderPass = 'offscreen';\n\n for (const layerId of layerIds) {\n const layer = this.style._layers[layerId];\n if (!layer.hasOffscreenPass() || layer.isHidden(this.transform.zoom)) continue;\n\n const coords = coordsDescending[layer.source];\n if (layer.type !== 'custom' && !coords.length) continue;\n\n this.renderLayer(this, sourceCaches[layer.source], layer, coords);\n }\n\n // Rebind the main framebuffer now that all offscreen layers have been rendered:\n this.context.bindFramebuffer.set(null);\n\n // Clear buffers in preparation for drawing to the main framebuffer\n this.context.clear({color: options.showOverdrawInspector ? Color.black : Color.transparent, depth: 1});\n this.clearStencil();\n\n this._showOverdrawInspector = options.showOverdrawInspector;\n this.depthRangeFor3D = [0, 1 - ((style._order.length + 2) * this.numSublayers * this.depthEpsilon)];\n\n // Opaque pass ===============================================\n // Draw opaque layers top-to-bottom first.\n if (!this.renderToTexture) {\n this.renderPass = 'opaque';\n\n for (this.currentLayer = layerIds.length - 1; this.currentLayer >= 0; this.currentLayer--) {\n const layer = this.style._layers[layerIds[this.currentLayer]];\n const sourceCache = sourceCaches[layer.source];\n const coords = coordsAscending[layer.source];\n\n this._renderTileClippingMasks(layer, coords);\n this.renderLayer(this, sourceCache, layer, coords);\n }\n }\n\n // Translucent pass ===============================================\n // Draw all other layers bottom-to-top.\n this.renderPass = 'translucent';\n\n for (this.currentLayer = 0; this.currentLayer < layerIds.length; this.currentLayer++) {\n const layer = this.style._layers[layerIds[this.currentLayer]];\n const sourceCache = sourceCaches[layer.source];\n\n if (this.renderToTexture && this.renderToTexture.renderLayer(layer)) continue;\n\n // For symbol layers in the translucent pass, we add extra tiles to the renderable set\n // for cross-tile symbol fading. Symbol layers don't use tile clipping, so no need to render\n // separate clipping masks\n const coords = (layer.type === 'symbol' ? coordsDescendingSymbol : coordsDescending)[layer.source];\n\n this._renderTileClippingMasks(layer, coordsAscending[layer.source]);\n this.renderLayer(this, sourceCache, layer, coords);\n }\n\n if (this.options.showTileBoundaries) {\n const selectedSource = selectDebugSource(this.style, this.transform.zoom);\n if (selectedSource) {\n drawDebug(this, selectedSource, selectedSource.getVisibleCoordinates());\n }\n }\n\n if (this.options.showPadding) {\n drawDebugPadding(this);\n }\n\n // Set defaults for most GL values so that anyone using the state after the render\n // encounters more expected values.\n this.context.setDefault();\n }\n\n renderLayer(painter: Painter, sourceCache: SourceCache, layer: StyleLayer, coords: Array) {\n if (layer.isHidden(this.transform.zoom)) return;\n if (layer.type !== 'background' && layer.type !== 'custom' && !(coords || []).length) return;\n this.id = layer.id;\n\n switch (layer.type) {\n case 'symbol':\n drawSymbols(painter, sourceCache, layer as any, coords, this.style.placement.variableOffsets);\n break;\n case 'circle':\n drawCircles(painter, sourceCache, layer as any, coords);\n break;\n case 'heatmap':\n drawHeatmap(painter, sourceCache, layer as any, coords);\n break;\n case 'line':\n drawLine(painter, sourceCache, layer as any, coords);\n break;\n case 'fill':\n drawFill(painter, sourceCache, layer as any, coords);\n break;\n case 'fill-extrusion':\n drawFillExtrusion(painter, sourceCache, layer as any, coords);\n break;\n case 'hillshade':\n drawHillshade(painter, sourceCache, layer as any, coords);\n break;\n case 'raster':\n drawRaster(painter, sourceCache, layer as any, coords);\n break;\n case 'background':\n drawBackground(painter, sourceCache, layer as any, coords);\n break;\n case 'custom':\n drawCustom(painter, sourceCache, layer as any);\n break;\n }\n }\n\n /**\n * Transform a matrix to incorporate the *-translate and *-translate-anchor properties into it.\n * @param inViewportPixelUnitsUnits - True when the units accepted by the matrix are in viewport pixels instead of tile units.\n * @returns matrix\n */\n translatePosMatrix(matrix: mat4, tile: Tile, translate: [number, number], translateAnchor: 'map' | 'viewport', inViewportPixelUnitsUnits?: boolean): mat4 {\n if (!translate[0] && !translate[1]) return matrix;\n\n const angle = inViewportPixelUnitsUnits ?\n (translateAnchor === 'map' ? this.transform.angle : 0) :\n (translateAnchor === 'viewport' ? -this.transform.angle : 0);\n\n if (angle) {\n const sinA = Math.sin(angle);\n const cosA = Math.cos(angle);\n translate = [\n translate[0] * cosA - translate[1] * sinA,\n translate[0] * sinA + translate[1] * cosA\n ];\n }\n\n const translation = [\n inViewportPixelUnitsUnits ? translate[0] : pixelsToTileUnits(tile, translate[0], this.transform.zoom),\n inViewportPixelUnitsUnits ? translate[1] : pixelsToTileUnits(tile, translate[1], this.transform.zoom),\n 0\n ] as vec3;\n\n const translatedMatrix = new Float32Array(16);\n mat4.translate(translatedMatrix, matrix, translation);\n return translatedMatrix;\n }\n\n saveTileTexture(texture: Texture) {\n const textures = this._tileTextures[texture.size[0]];\n if (!textures) {\n this._tileTextures[texture.size[0]] = [texture];\n } else {\n textures.push(texture);\n }\n }\n\n getTileTexture(size: number) {\n const textures = this._tileTextures[size];\n return textures && textures.length > 0 ? textures.pop() : null;\n }\n\n /**\n * Checks whether a pattern image is needed, and if it is, whether it is not loaded.\n *\n * @returns true if a needed image is missing and rendering needs to be skipped.\n */\n isPatternMissing(image?: CrossFaded | null): boolean {\n if (!image) return false;\n if (!image.from || !image.to) return true;\n const imagePosA = this.imageManager.getPattern(image.from.toString());\n const imagePosB = this.imageManager.getPattern(image.to.toString());\n return !imagePosA || !imagePosB;\n }\n\n useProgram(name: string, programConfiguration?: ProgramConfiguration | null): Program {\n this.cache = this.cache || {};\n const key = name +\n (programConfiguration ? programConfiguration.cacheKey : '') +\n (this._showOverdrawInspector ? '/overdraw' : '') +\n (this.style.map.terrain ? '/terrain' : '');\n if (!this.cache[key]) {\n this.cache[key] = new Program(\n this.context,\n shaders[name],\n programConfiguration,\n programUniforms[name],\n this._showOverdrawInspector,\n this.style.map.terrain\n );\n }\n return this.cache[key];\n }\n\n /*\n * Reset some GL state to default values to avoid hard-to-debug bugs\n * in custom layers.\n */\n setCustomLayerDefaults() {\n // Prevent custom layers from unintentionally modify the last VAO used.\n // All other state is state is restored on it's own, but for VAOs it's\n // simpler to unbind so that we don't have to track the state of VAOs.\n this.context.unbindVAO();\n\n // The default values for this state is meaningful and often expected.\n // Leaving this state dirty could cause a lot of confusion for users.\n this.context.cullFace.setDefault();\n this.context.activeTexture.setDefault();\n this.context.pixelStoreUnpack.setDefault();\n this.context.pixelStoreUnpackPremultiplyAlpha.setDefault();\n this.context.pixelStoreUnpackFlipY.setDefault();\n }\n\n /*\n * Set GL state that is shared by all layers.\n */\n setBaseState() {\n const gl = this.context.gl;\n this.context.cullFace.set(false);\n this.context.viewport.set([0, 0, this.width, this.height]);\n this.context.blendEquation.set(gl.FUNC_ADD);\n }\n\n initDebugOverlayCanvas() {\n if (this.debugOverlayCanvas == null) {\n this.debugOverlayCanvas = document.createElement('canvas');\n this.debugOverlayCanvas.width = 512;\n this.debugOverlayCanvas.height = 512;\n const gl = this.context.gl;\n this.debugOverlayTexture = new Texture(this.context, this.debugOverlayCanvas, gl.RGBA);\n }\n }\n\n destroy() {\n if (this.debugOverlayTexture) {\n this.debugOverlayTexture.destroy();\n }\n }\n\n /*\n * Return true if drawing buffer size is != from requested size.\n * That means that we've reached GL limits somehow.\n * Note: drawing buffer size changes only when canvas size changes\n */\n overLimit() {\n const {drawingBufferWidth, drawingBufferHeight} = this.context.gl;\n return this.width !== drawingBufferWidth || this.height !== drawingBufferHeight;\n }\n}\n","import {mat4, vec3, vec4} from 'gl-matrix';\n\nclass Frustum {\n\n constructor(public points: vec4[], public planes: vec4[]) { }\n\n public static fromInvProjectionMatrix(invProj: mat4, worldSize: number, zoom: number): Frustum {\n const clipSpaceCorners = [\n [-1, 1, -1, 1],\n [1, 1, -1, 1],\n [1, -1, -1, 1],\n [-1, -1, -1, 1],\n [-1, 1, 1, 1],\n [1, 1, 1, 1],\n [1, -1, 1, 1],\n [-1, -1, 1, 1]\n ];\n\n const scale = Math.pow(2, zoom);\n\n // Transform frustum corner points from clip space to tile space, Z to meters\n const frustumCoords = clipSpaceCorners.map(v => {\n v = vec4.transformMat4([] as any, v as any, invProj) as any;\n const s = 1.0 / v[3] / worldSize * scale;\n return vec4.mul(v as any, v as any, [s, s, 1.0 / v[3], s] as vec4);\n });\n\n const frustumPlanePointIndices = [\n [0, 1, 2], // near\n [6, 5, 4], // far\n [0, 3, 7], // left\n [2, 1, 5], // right\n [3, 2, 6], // bottom\n [0, 4, 5] // top\n ];\n\n const frustumPlanes = frustumPlanePointIndices.map((p: number[]) => {\n const a = vec3.sub([] as any, frustumCoords[p[0]] as vec3, frustumCoords[p[1]] as vec3);\n const b = vec3.sub([] as any, frustumCoords[p[2]] as vec3, frustumCoords[p[1]] as vec3);\n const n = vec3.normalize([] as any, vec3.cross([] as any, a, b)) as any;\n const d = -vec3.dot(n, frustumCoords[p[1]] as vec3);\n return n.concat(d);\n });\n\n return new Frustum(frustumCoords, frustumPlanes);\n }\n}\n\nclass Aabb {\n min: vec3;\n max: vec3;\n center: vec3;\n\n constructor(min_: vec3, max_: vec3) {\n this.min = min_;\n this.max = max_;\n this.center = vec3.scale([] as any, vec3.add([] as any, this.min, this.max), 0.5);\n }\n\n quadrant(index: number): Aabb {\n const split = [(index % 2) === 0, index < 2];\n const qMin = vec3.clone(this.min);\n const qMax = vec3.clone(this.max);\n for (let axis = 0; axis < split.length; axis++) {\n qMin[axis] = split[axis] ? this.min[axis] : this.center[axis];\n qMax[axis] = split[axis] ? this.center[axis] : this.max[axis];\n }\n // Elevation is always constant, hence quadrant.max.z = this.max.z\n qMax[2] = this.max[2];\n return new Aabb(qMin, qMax);\n }\n\n distanceX(point: Array): number {\n const pointOnAabb = Math.max(Math.min(this.max[0], point[0]), this.min[0]);\n return pointOnAabb - point[0];\n }\n\n distanceY(point: Array): number {\n const pointOnAabb = Math.max(Math.min(this.max[1], point[1]), this.min[1]);\n return pointOnAabb - point[1];\n }\n\n // Performs a frustum-aabb intersection test. Returns 0 if there's no intersection,\n // 1 if shapes are intersecting and 2 if the aabb if fully inside the frustum.\n intersects(frustum: Frustum): number {\n // Execute separating axis test between two convex objects to find intersections\n // Each frustum plane together with 3 major axes define the separating axes\n\n const aabbPoints = [\n [this.min[0], this.min[1], this.min[2], 1],\n [this.max[0], this.min[1], this.min[2], 1],\n [this.max[0], this.max[1], this.min[2], 1],\n [this.min[0], this.max[1], this.min[2], 1],\n [this.min[0], this.min[1], this.max[2], 1],\n [this.max[0], this.min[1], this.max[2], 1],\n [this.max[0], this.max[1], this.max[2], 1],\n [this.min[0], this.max[1], this.max[2], 1]\n ];\n\n let fullyInside = true;\n\n for (let p = 0; p < frustum.planes.length; p++) {\n const plane = frustum.planes[p];\n let pointsInside = 0;\n\n for (let i = 0; i < aabbPoints.length; i++) {\n if (vec4.dot(plane, aabbPoints[i] as any) >= 0) {\n pointsInside++;\n }\n }\n\n if (pointsInside === 0)\n return 0;\n\n if (pointsInside !== aabbPoints.length)\n fullyInside = false;\n }\n\n if (fullyInside)\n return 2;\n\n for (let axis = 0; axis < 3; axis++) {\n let projMin = Number.MAX_VALUE;\n let projMax = -Number.MAX_VALUE;\n\n for (let p = 0; p < frustum.points.length; p++) {\n const projectedPoint = frustum.points[p][axis] - this.min[axis];\n\n projMin = Math.min(projMin, projectedPoint);\n projMax = Math.max(projMax, projectedPoint);\n }\n\n if (projMax < 0 || projMin > this.max[axis] - this.min[axis])\n return 0;\n }\n\n return 1;\n }\n}\nexport {\n Aabb,\n Frustum\n};\n","import {interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport Point from '@mapbox/point-geometry';\nimport {clamp} from '../util/util';\n\n/**\n * An `EdgeInset` object represents screen space padding applied to the edges of the viewport.\n * This shifts the apprent center or the vanishing point of the map. This is useful for adding floating UI elements\n * on top of the map and having the vanishing point shift as UI elements resize.\n *\n * @group Geography and Geometry\n */\nexport class EdgeInsets {\n /**\n * @defaultValue 0\n */\n top: number;\n /**\n * @defaultValue 0\n */\n bottom: number;\n /**\n * @defaultValue 0\n */\n left: number;\n /**\n * @defaultValue 0\n */\n right: number;\n\n constructor(top: number = 0, bottom: number = 0, left: number = 0, right: number = 0) {\n if (isNaN(top) || top < 0 ||\n isNaN(bottom) || bottom < 0 ||\n isNaN(left) || left < 0 ||\n isNaN(right) || right < 0\n ) {\n throw new Error('Invalid value for edge-insets, top, bottom, left and right must all be numbers');\n }\n\n this.top = top;\n this.bottom = bottom;\n this.left = left;\n this.right = right;\n }\n\n /**\n * Interpolates the inset in-place.\n * This maintains the current inset value for any inset not present in `target`.\n * @param start - interpolation start\n * @param target - interpolation target\n * @param t - interpolation step/weight\n * @returns the insets\n */\n interpolate(start: PaddingOptions | EdgeInsets, target: PaddingOptions, t: number): EdgeInsets {\n if (target.top != null && start.top != null) this.top = interpolates.number(start.top, target.top, t);\n if (target.bottom != null && start.bottom != null) this.bottom = interpolates.number(start.bottom, target.bottom, t);\n if (target.left != null && start.left != null) this.left = interpolates.number(start.left, target.left, t);\n if (target.right != null && start.right != null) this.right = interpolates.number(start.right, target.right, t);\n\n return this;\n }\n\n /**\n * Utility method that computes the new apprent center or vanishing point after applying insets.\n * This is in pixels and with the top left being (0.0) and +y being downwards.\n *\n * @param width - the width\n * @param height - the height\n * @returns the point\n */\n getCenter(width: number, height: number): Point {\n // Clamp insets so they never overflow width/height and always calculate a valid center\n const x = clamp((this.left + width - this.right) / 2, 0, width);\n const y = clamp((this.top + height - this.bottom) / 2, 0, height);\n\n return new Point(x, y);\n }\n\n equals(other: PaddingOptions): boolean {\n return this.top === other.top &&\n this.bottom === other.bottom &&\n this.left === other.left &&\n this.right === other.right;\n }\n\n clone(): EdgeInsets {\n return new EdgeInsets(this.top, this.bottom, this.left, this.right);\n }\n\n /**\n * Returns the current state as json, useful when you want to have a\n * read-only representation of the inset.\n *\n * @returns state as json\n */\n toJSON(): PaddingOptions {\n return {\n top: this.top,\n bottom: this.bottom,\n left: this.left,\n right: this.right\n };\n }\n}\n\n/**\n * Options for setting padding on calls to methods such as {@link Map#fitBounds}, {@link Map#fitScreenCoordinates}, and {@link Map#setPadding}. Adjust these options to set the amount of padding in pixels added to the edges of the canvas. Set a uniform padding on all edges or individual values for each edge. All properties of this object must be\n * non-negative integers.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let bbox = [[-79, 43], [-73, 45]];\n * map.fitBounds(bbox, {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n *\n * @example\n * ```ts\n * let bbox = [[-79, 43], [-73, 45]];\n * map.fitBounds(bbox, {\n * padding: 20\n * });\n * ```\n * @see [Fit to the bounds of a LineString](https://maplibre.org/maplibre-gl-js/docs/examples/zoomto-linestring/)\n * @see [Fit a map to a bounding box](https://maplibre.org/maplibre-gl-js/docs/examples/fitbounds/)\n */\nexport type PaddingOptions = {\n /**\n * Padding in pixels from the top of the map canvas.\n */\n top: number;\n /**\n * Padding in pixels from the bottom of the map canvas.\n */\n bottom: number;\n /**\n * Padding in pixels from the left of the map canvas.\n */\n right: number;\n /**\n * Padding in pixels from the right of the map canvas.\n */\n left: number;\n};\n","import {LngLat} from './lng_lat';\nimport {LngLatBounds} from './lng_lat_bounds';\nimport {MercatorCoordinate, mercatorXfromLng, mercatorYfromLat, mercatorZfromAltitude} from './mercator_coordinate';\nimport Point from '@mapbox/point-geometry';\nimport {wrap, clamp} from '../util/util';\nimport {interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {EXTENT} from '../data/extent';\nimport {vec3, vec4, mat4, mat2, vec2} from 'gl-matrix';\nimport {Aabb, Frustum} from '../util/primitives';\nimport {EdgeInsets} from './edge_insets';\n\nimport {UnwrappedTileID, OverscaledTileID, CanonicalTileID} from '../source/tile_id';\nimport type {PaddingOptions} from './edge_insets';\nimport {Terrain} from '../render/terrain';\n\n/**\n * @internal\n * A single transform, generally used for a single tile to be\n * scaled, rotated, and zoomed.\n */\nexport class Transform {\n tileSize: number;\n tileZoom: number;\n lngRange: [number, number];\n latRange: [number, number];\n maxValidLatitude: number;\n scale: number;\n width: number;\n height: number;\n angle: number;\n rotationMatrix: mat2;\n pixelsToGLUnits: [number, number];\n cameraToCenterDistance: number;\n mercatorMatrix: mat4;\n projMatrix: mat4;\n invProjMatrix: mat4;\n alignedProjMatrix: mat4;\n pixelMatrix: mat4;\n pixelMatrix3D: mat4;\n pixelMatrixInverse: mat4;\n glCoordMatrix: mat4;\n labelPlaneMatrix: mat4;\n _fov: number;\n _pitch: number;\n _zoom: number;\n _unmodified: boolean;\n _renderWorldCopies: boolean;\n _minZoom: number;\n _maxZoom: number;\n _minPitch: number;\n _maxPitch: number;\n _center: LngLat;\n _elevation: number;\n _pixelPerMeter: number;\n _edgeInsets: EdgeInsets;\n _constraining: boolean;\n _posMatrixCache: {[_: string]: mat4};\n _alignedPosMatrixCache: {[_: string]: mat4};\n _minEleveationForCurrentTile: number;\n\n constructor(minZoom?: number, maxZoom?: number, minPitch?: number, maxPitch?: number, renderWorldCopies?: boolean) {\n this.tileSize = 512; // constant\n this.maxValidLatitude = 85.051129; // constant\n\n this._renderWorldCopies = renderWorldCopies === undefined ? true : !!renderWorldCopies;\n this._minZoom = minZoom || 0;\n this._maxZoom = maxZoom || 22;\n\n this._minPitch = (minPitch === undefined || minPitch === null) ? 0 : minPitch;\n this._maxPitch = (maxPitch === undefined || maxPitch === null) ? 60 : maxPitch;\n\n this.setMaxBounds();\n\n this.width = 0;\n this.height = 0;\n this._center = new LngLat(0, 0);\n this._elevation = 0;\n this.zoom = 0;\n this.angle = 0;\n this._fov = 0.6435011087932844;\n this._pitch = 0;\n this._unmodified = true;\n this._edgeInsets = new EdgeInsets();\n this._posMatrixCache = {};\n this._alignedPosMatrixCache = {};\n this._minEleveationForCurrentTile = 0;\n }\n\n clone(): Transform {\n const clone = new Transform(this._minZoom, this._maxZoom, this._minPitch, this.maxPitch, this._renderWorldCopies);\n clone.apply(this);\n return clone;\n }\n\n apply(that: Transform) {\n this.tileSize = that.tileSize;\n this.latRange = that.latRange;\n this.width = that.width;\n this.height = that.height;\n this._center = that._center;\n this._elevation = that._elevation;\n this._minEleveationForCurrentTile = that._minEleveationForCurrentTile;\n this.zoom = that.zoom;\n this.angle = that.angle;\n this._fov = that._fov;\n this._pitch = that._pitch;\n this._unmodified = that._unmodified;\n this._edgeInsets = that._edgeInsets.clone();\n this._calcMatrices();\n }\n\n get minZoom(): number { return this._minZoom; }\n set minZoom(zoom: number) {\n if (this._minZoom === zoom) return;\n this._minZoom = zoom;\n this.zoom = Math.max(this.zoom, zoom);\n }\n\n get maxZoom(): number { return this._maxZoom; }\n set maxZoom(zoom: number) {\n if (this._maxZoom === zoom) return;\n this._maxZoom = zoom;\n this.zoom = Math.min(this.zoom, zoom);\n }\n\n get minPitch(): number { return this._minPitch; }\n set minPitch(pitch: number) {\n if (this._minPitch === pitch) return;\n this._minPitch = pitch;\n this.pitch = Math.max(this.pitch, pitch);\n }\n\n get maxPitch(): number { return this._maxPitch; }\n set maxPitch(pitch: number) {\n if (this._maxPitch === pitch) return;\n this._maxPitch = pitch;\n this.pitch = Math.min(this.pitch, pitch);\n }\n\n get renderWorldCopies(): boolean { return this._renderWorldCopies; }\n set renderWorldCopies(renderWorldCopies: boolean) {\n if (renderWorldCopies === undefined) {\n renderWorldCopies = true;\n } else if (renderWorldCopies === null) {\n renderWorldCopies = false;\n }\n\n this._renderWorldCopies = renderWorldCopies;\n }\n\n get worldSize(): number {\n return this.tileSize * this.scale;\n }\n\n get centerOffset(): Point {\n return this.centerPoint._sub(this.size._div(2));\n }\n\n get size(): Point {\n return new Point(this.width, this.height);\n }\n\n get bearing(): number {\n return -this.angle / Math.PI * 180;\n }\n set bearing(bearing: number) {\n const b = -wrap(bearing, -180, 180) * Math.PI / 180;\n if (this.angle === b) return;\n this._unmodified = false;\n this.angle = b;\n this._calcMatrices();\n\n // 2x2 matrix for rotating points\n this.rotationMatrix = mat2.create();\n mat2.rotate(this.rotationMatrix, this.rotationMatrix, this.angle);\n }\n\n get pitch(): number {\n return this._pitch / Math.PI * 180;\n }\n set pitch(pitch: number) {\n const p = clamp(pitch, this.minPitch, this.maxPitch) / 180 * Math.PI;\n if (this._pitch === p) return;\n this._unmodified = false;\n this._pitch = p;\n this._calcMatrices();\n }\n\n get fov(): number {\n return this._fov / Math.PI * 180;\n }\n set fov(fov: number) {\n fov = Math.max(0.01, Math.min(60, fov));\n if (this._fov === fov) return;\n this._unmodified = false;\n this._fov = fov / 180 * Math.PI;\n this._calcMatrices();\n }\n\n get zoom(): number { return this._zoom; }\n set zoom(zoom: number) {\n const constrainedZoom = Math.min(Math.max(zoom, this.minZoom), this.maxZoom);\n if (this._zoom === constrainedZoom) return;\n this._unmodified = false;\n this._zoom = constrainedZoom;\n this.tileZoom = Math.max(0, Math.floor(constrainedZoom));\n this.scale = this.zoomScale(constrainedZoom);\n this._constrain();\n this._calcMatrices();\n }\n\n get center(): LngLat { return this._center; }\n set center(center: LngLat) {\n if (center.lat === this._center.lat && center.lng === this._center.lng) return;\n this._unmodified = false;\n this._center = center;\n this._constrain();\n this._calcMatrices();\n }\n\n get elevation(): number { return this._elevation; }\n set elevation(elevation: number) {\n if (elevation === this._elevation) return;\n this._elevation = elevation;\n this._constrain();\n this._calcMatrices();\n }\n\n get padding(): PaddingOptions { return this._edgeInsets.toJSON(); }\n set padding(padding: PaddingOptions) {\n if (this._edgeInsets.equals(padding)) return;\n this._unmodified = false;\n //Update edge-insets inplace\n this._edgeInsets.interpolate(this._edgeInsets, padding, 1);\n this._calcMatrices();\n }\n\n /**\n * The center of the screen in pixels with the top-left corner being (0,0)\n * and +y axis pointing downwards. This accounts for padding.\n */\n get centerPoint(): Point {\n return this._edgeInsets.getCenter(this.width, this.height);\n }\n\n /**\n * Returns if the padding params match\n *\n * @param padding - the padding to check against\n * @returns true if they are equal, false otherwise\n */\n isPaddingEqual(padding: PaddingOptions): boolean {\n return this._edgeInsets.equals(padding);\n }\n\n /**\n * Helper method to update edge-insets in place\n *\n * @param start - the starting padding\n * @param target - the target padding\n * @param t - the step/weight\n */\n interpolatePadding(start: PaddingOptions, target: PaddingOptions, t: number) {\n this._unmodified = false;\n this._edgeInsets.interpolate(start, target, t);\n this._constrain();\n this._calcMatrices();\n }\n\n /**\n * Return a zoom level that will cover all tiles the transform\n * @param options - the options\n * @returns zoom level An integer zoom level at which all tiles will be visible.\n */\n coveringZoomLevel(options: {\n /**\n * Target zoom level. If true, the value will be rounded to the closest integer. Otherwise the value will be floored.\n */\n roundZoom?: boolean;\n /**\n * Tile size, expressed in screen pixels.\n */\n tileSize: number;\n }): number {\n const z = (options.roundZoom ? Math.round : Math.floor)(\n this.zoom + this.scaleZoom(this.tileSize / options.tileSize)\n );\n // At negative zoom levels load tiles from z0 because negative tile zoom levels don't exist.\n return Math.max(0, z);\n }\n\n /**\n * Return any \"wrapped\" copies of a given tile coordinate that are visible\n * in the current view.\n */\n getVisibleUnwrappedCoordinates(tileID: CanonicalTileID) {\n const result = [new UnwrappedTileID(0, tileID)];\n if (this._renderWorldCopies) {\n const utl = this.pointCoordinate(new Point(0, 0));\n const utr = this.pointCoordinate(new Point(this.width, 0));\n const ubl = this.pointCoordinate(new Point(this.width, this.height));\n const ubr = this.pointCoordinate(new Point(0, this.height));\n const w0 = Math.floor(Math.min(utl.x, utr.x, ubl.x, ubr.x));\n const w1 = Math.floor(Math.max(utl.x, utr.x, ubl.x, ubr.x));\n\n // Add an extra copy of the world on each side to properly render ImageSources and CanvasSources.\n // Both sources draw outside the tile boundaries of the tile that \"contains them\" so we need\n // to add extra copies on both sides in case offscreen tiles need to draw into on-screen ones.\n const extraWorldCopy = 1;\n\n for (let w = w0 - extraWorldCopy; w <= w1 + extraWorldCopy; w++) {\n if (w === 0) continue;\n result.push(new UnwrappedTileID(w, tileID));\n }\n }\n return result;\n }\n\n /**\n * Return all coordinates that could cover this transform for a covering\n * zoom level.\n * @param options - the options\n * @returns OverscaledTileIDs\n */\n coveringTiles(\n options: {\n tileSize: number;\n minzoom?: number;\n maxzoom?: number;\n roundZoom?: boolean;\n reparseOverscaled?: boolean;\n renderWorldCopies?: boolean;\n terrain?: Terrain;\n }\n ): Array {\n let z = this.coveringZoomLevel(options);\n const actualZ = z;\n\n if (options.minzoom !== undefined && z < options.minzoom) return [];\n if (options.maxzoom !== undefined && z > options.maxzoom) z = options.maxzoom;\n\n const cameraCoord = this.pointCoordinate(this.getCameraPoint());\n const centerCoord = MercatorCoordinate.fromLngLat(this.center);\n const numTiles = Math.pow(2, z);\n const cameraPoint = [numTiles * cameraCoord.x, numTiles * cameraCoord.y, 0];\n const centerPoint = [numTiles * centerCoord.x, numTiles * centerCoord.y, 0];\n const cameraFrustum = Frustum.fromInvProjectionMatrix(this.invProjMatrix, this.worldSize, z);\n\n // No change of LOD behavior for pitch lower than 60 and when there is no top padding: return only tile ids from the requested zoom level\n let minZoom = options.minzoom || 0;\n // Use 0.1 as an epsilon to avoid for explicit == 0.0 floating point checks\n if (!options.terrain && this.pitch <= 60.0 && this._edgeInsets.top < 0.1)\n minZoom = z;\n\n // There should always be a certain number of maximum zoom level tiles surrounding the center location in 2D or in front of the camera in 3D\n const radiusOfMaxLvlLodInTiles = options.terrain ? 2 / Math.min(this.tileSize, options.tileSize) * this.tileSize : 3;\n\n const newRootTile = (wrap: number): any => {\n return {\n aabb: new Aabb([wrap * numTiles, 0, 0], [(wrap + 1) * numTiles, numTiles, 0]),\n zoom: 0,\n x: 0,\n y: 0,\n wrap,\n fullyVisible: false\n };\n };\n\n // Do a depth-first traversal to find visible tiles and proper levels of detail\n const stack = [];\n const result = [];\n const maxZoom = z;\n const overscaledZ = options.reparseOverscaled ? actualZ : z;\n\n if (this._renderWorldCopies) {\n // Render copy of the globe thrice on both sides\n for (let i = 1; i <= 3; i++) {\n stack.push(newRootTile(-i));\n stack.push(newRootTile(i));\n }\n }\n\n stack.push(newRootTile(0));\n\n while (stack.length > 0) {\n const it = stack.pop();\n const x = it.x;\n const y = it.y;\n let fullyVisible = it.fullyVisible;\n\n // Visibility of a tile is not required if any of its ancestor if fully inside the frustum\n if (!fullyVisible) {\n const intersectResult = it.aabb.intersects(cameraFrustum);\n\n if (intersectResult === 0)\n continue;\n\n fullyVisible = intersectResult === 2;\n }\n\n const refPoint = options.terrain ? cameraPoint : centerPoint;\n const distanceX = it.aabb.distanceX(refPoint);\n const distanceY = it.aabb.distanceY(refPoint);\n const longestDim = Math.max(Math.abs(distanceX), Math.abs(distanceY));\n\n // We're using distance based heuristics to determine if a tile should be split into quadrants or not.\n // radiusOfMaxLvlLodInTiles defines that there's always a certain number of maxLevel tiles next to the map center.\n // Using the fact that a parent node in quadtree is twice the size of its children (per dimension)\n // we can define distance thresholds for each relative level:\n // f(k) = offset + 2 + 4 + 8 + 16 + ... + 2^k. This is the same as \"offset+2^(k+1)-2\"\n const distToSplit = radiusOfMaxLvlLodInTiles + (1 << (maxZoom - it.zoom)) - 2;\n\n // Have we reached the target depth or is the tile too far away to be any split further?\n if (it.zoom === maxZoom || (longestDim > distToSplit && it.zoom >= minZoom)) {\n const dz = maxZoom - it.zoom, dx = cameraPoint[0] - 0.5 - (x << dz), dy = cameraPoint[1] - 0.5 - (y << dz);\n result.push({\n tileID: new OverscaledTileID(it.zoom === maxZoom ? overscaledZ : it.zoom, it.wrap, it.zoom, x, y),\n distanceSq: vec2.sqrLen([centerPoint[0] - 0.5 - x, centerPoint[1] - 0.5 - y]),\n // this variable is currently not used, but may be important to reduce the amount of loaded tiles\n tileDistanceToCamera: Math.sqrt(dx * dx + dy * dy)\n });\n continue;\n }\n\n for (let i = 0; i < 4; i++) {\n const childX = (x << 1) + (i % 2);\n const childY = (y << 1) + (i >> 1);\n const childZ = it.zoom + 1;\n let quadrant = it.aabb.quadrant(i);\n if (options.terrain) {\n const tileID = new OverscaledTileID(childZ, it.wrap, childZ, childX, childY);\n const minMax = options.terrain.getMinMaxElevation(tileID);\n const minElevation = minMax.minElevation ?? this.elevation;\n const maxElevation = minMax.maxElevation ?? this.elevation;\n quadrant = new Aabb(\n [quadrant.min[0], quadrant.min[1], minElevation] as vec3,\n [quadrant.max[0], quadrant.max[1], maxElevation] as vec3\n );\n }\n stack.push({aabb: quadrant, zoom: childZ, x: childX, y: childY, wrap: it.wrap, fullyVisible});\n }\n }\n\n return result.sort((a, b) => a.distanceSq - b.distanceSq).map(a => a.tileID);\n }\n\n resize(width: number, height: number) {\n this.width = width;\n this.height = height;\n\n this.pixelsToGLUnits = [2 / width, -2 / height];\n this._constrain();\n this._calcMatrices();\n }\n\n get unmodified(): boolean { return this._unmodified; }\n\n zoomScale(zoom: number) { return Math.pow(2, zoom); }\n scaleZoom(scale: number) { return Math.log(scale) / Math.LN2; }\n\n project(lnglat: LngLat) {\n const lat = clamp(lnglat.lat, -this.maxValidLatitude, this.maxValidLatitude);\n return new Point(\n mercatorXfromLng(lnglat.lng) * this.worldSize,\n mercatorYfromLat(lat) * this.worldSize);\n }\n\n unproject(point: Point): LngLat {\n return new MercatorCoordinate(point.x / this.worldSize, point.y / this.worldSize).toLngLat();\n }\n\n get point(): Point { return this.project(this.center); }\n\n /**\n * get the camera position in LngLat and altitudes in meter\n * @returns An object with lngLat & altitude.\n */\n getCameraPosition(): {\n lngLat: LngLat;\n altitude: number;\n } {\n const lngLat = this.pointLocation(this.getCameraPoint());\n const altitude = Math.cos(this._pitch) * this.cameraToCenterDistance / this._pixelPerMeter;\n return {lngLat, altitude: altitude + this.elevation};\n }\n\n /**\n * This method works in combination with freezeElevation activated.\n * freezeElevtion is enabled during map-panning because during this the camera should sit in constant height.\n * After panning finished, call this method to recalculate the zoomlevel for the current camera-height in current terrain.\n * @param terrain - the terrain\n */\n recalculateZoom(terrain: Terrain) {\n // find position the camera is looking on\n const center = this.pointLocation(this.centerPoint, terrain);\n const elevation = terrain.getElevationForLngLatZoom(center, this.tileZoom);\n const deltaElevation = this.elevation - elevation;\n if (!deltaElevation) return;\n\n // calculate mercator distance between camera & target\n const cameraPosition = this.getCameraPosition();\n const camera = MercatorCoordinate.fromLngLat(cameraPosition.lngLat, cameraPosition.altitude);\n const target = MercatorCoordinate.fromLngLat(center, elevation);\n const dx = camera.x - target.x, dy = camera.y - target.y, dz = camera.z - target.z;\n const distance = Math.sqrt(dx * dx + dy * dy + dz * dz);\n\n // from this distance we calculate the new zoomlevel\n const zoom = this.scaleZoom(this.cameraToCenterDistance / distance / this.tileSize);\n\n // update matrices\n this._elevation = elevation;\n this._center = center;\n this.zoom = zoom;\n }\n\n setLocationAtPoint(lnglat: LngLat, point: Point) {\n const a = this.pointCoordinate(point);\n const b = this.pointCoordinate(this.centerPoint);\n const loc = this.locationCoordinate(lnglat);\n const newCenter = new MercatorCoordinate(\n loc.x - (a.x - b.x),\n loc.y - (a.y - b.y));\n this.center = this.coordinateLocation(newCenter);\n if (this._renderWorldCopies) {\n this.center = this.center.wrap();\n }\n }\n\n /**\n * Given a location, return the screen point that corresponds to it\n * @param lnglat - location\n * @param terrain - optional terrain\n * @returns screen point\n */\n locationPoint(lnglat: LngLat, terrain?: Terrain): Point {\n return terrain ?\n this.coordinatePoint(this.locationCoordinate(lnglat), terrain.getElevationForLngLatZoom(lnglat, this.tileZoom), this.pixelMatrix3D) :\n this.coordinatePoint(this.locationCoordinate(lnglat));\n }\n\n /**\n * Given a point on screen, return its lnglat\n * @param p - screen point\n * @param terrain - optional terrain\n * @returns lnglat location\n */\n pointLocation(p: Point, terrain?: Terrain): LngLat {\n return this.coordinateLocation(this.pointCoordinate(p, terrain));\n }\n\n /**\n * Given a geographical lnglat, return an unrounded\n * coordinate that represents it at this transform's zoom level.\n * @param lnglat - the location\n * @returns The mercator coordinate\n */\n locationCoordinate(lnglat: LngLat): MercatorCoordinate {\n return MercatorCoordinate.fromLngLat(lnglat);\n }\n\n /**\n * Given a Coordinate, return its geographical position.\n * @param coord - mercator coordivates\n * @returns lng and lat\n */\n coordinateLocation(coord: MercatorCoordinate): LngLat {\n return coord && coord.toLngLat();\n }\n\n /**\n * Given a Point, return its mercator coordinate.\n * @param p - the point\n * @param terrain - optional terrain\n * @returns lnglat\n */\n pointCoordinate(p: Point, terrain?: Terrain): MercatorCoordinate {\n // get point-coordinate from terrain coordinates framebuffer\n if (terrain) {\n const coordinate = terrain.pointCoordinate(p);\n if (coordinate != null) {\n return coordinate;\n }\n }\n\n // calculate point-coordinate on flat earth\n const targetZ = 0;\n // since we don't know the correct projected z value for the point,\n // unproject two points to get a line and then find the point on that\n // line with z=0\n\n const coord0 = [p.x, p.y, 0, 1] as any;\n const coord1 = [p.x, p.y, 1, 1] as any;\n\n vec4.transformMat4(coord0, coord0, this.pixelMatrixInverse);\n vec4.transformMat4(coord1, coord1, this.pixelMatrixInverse);\n\n const w0 = coord0[3];\n const w1 = coord1[3];\n const x0 = coord0[0] / w0;\n const x1 = coord1[0] / w1;\n const y0 = coord0[1] / w0;\n const y1 = coord1[1] / w1;\n const z0 = coord0[2] / w0;\n const z1 = coord1[2] / w1;\n\n const t = z0 === z1 ? 0 : (targetZ - z0) / (z1 - z0);\n\n return new MercatorCoordinate(\n interpolates.number(x0, x1, t) / this.worldSize,\n interpolates.number(y0, y1, t) / this.worldSize);\n }\n\n /**\n * Given a coordinate, return the screen point that corresponds to it\n * @param coord - the coordinates\n * @param elevation - the elevation\n * @param pixelMatrix - the pixel matrix\n * @returns screen point\n */\n coordinatePoint(coord: MercatorCoordinate, elevation: number = 0, pixelMatrix = this.pixelMatrix): Point {\n const p = [coord.x * this.worldSize, coord.y * this.worldSize, elevation, 1] as any;\n vec4.transformMat4(p, p, pixelMatrix);\n return new Point(p[0] / p[3], p[1] / p[3]);\n }\n\n /**\n * Returns the map's geographical bounds. When the bearing or pitch is non-zero, the visible region is not\n * an axis-aligned rectangle, and the result is the smallest bounds that encompasses the visible region.\n * @returns Returns a {@link LngLatBounds} object describing the map's geographical bounds.\n */\n getBounds(): LngLatBounds {\n const top = Math.max(0, this.height / 2 - this.getHorizon());\n return new LngLatBounds()\n .extend(this.pointLocation(new Point(0, top)))\n .extend(this.pointLocation(new Point(this.width, top)))\n .extend(this.pointLocation(new Point(this.width, this.height)))\n .extend(this.pointLocation(new Point(0, this.height)));\n }\n\n /**\n * Returns the maximum geographical bounds the map is constrained to, or `null` if none set.\n * @returns max bounds\n */\n getMaxBounds(): LngLatBounds | null {\n if (!this.latRange || this.latRange.length !== 2 ||\n !this.lngRange || this.lngRange.length !== 2) return null;\n\n return new LngLatBounds([this.lngRange[0], this.latRange[0]], [this.lngRange[1], this.latRange[1]]);\n }\n\n /**\n * Calculate pixel height of the visible horizon in relation to map-center (e.g. height/2),\n * multiplied by a static factor to simulate the earth-radius.\n * The calculated value is the horizontal line from the camera-height to sea-level.\n * @returns Horizon above center in pixels.\n */\n getHorizon(): number {\n return Math.tan(Math.PI / 2 - this._pitch) * this.cameraToCenterDistance * 0.85;\n }\n\n /**\n * Sets or clears the map's geographical constraints.\n * @param bounds - A {@link LngLatBounds} object describing the new geographic boundaries of the map.\n */\n setMaxBounds(bounds?: LngLatBounds | null) {\n if (bounds) {\n this.lngRange = [bounds.getWest(), bounds.getEast()];\n this.latRange = [bounds.getSouth(), bounds.getNorth()];\n this._constrain();\n } else {\n this.lngRange = null;\n this.latRange = [-this.maxValidLatitude, this.maxValidLatitude];\n }\n }\n\n /**\n * Calculate the posMatrix that, given a tile coordinate, would be used to display the tile on a map.\n * @param unwrappedTileID - the tile ID\n */\n calculatePosMatrix(unwrappedTileID: UnwrappedTileID, aligned: boolean = false): mat4 {\n const posMatrixKey = unwrappedTileID.key;\n const cache = aligned ? this._alignedPosMatrixCache : this._posMatrixCache;\n if (cache[posMatrixKey]) {\n return cache[posMatrixKey];\n }\n\n const canonical = unwrappedTileID.canonical;\n const scale = this.worldSize / this.zoomScale(canonical.z);\n const unwrappedX = canonical.x + Math.pow(2, canonical.z) * unwrappedTileID.wrap;\n\n const posMatrix = mat4.identity(new Float64Array(16) as any);\n mat4.translate(posMatrix, posMatrix, [unwrappedX * scale, canonical.y * scale, 0]);\n mat4.scale(posMatrix, posMatrix, [scale / EXTENT, scale / EXTENT, 1]);\n mat4.multiply(posMatrix, aligned ? this.alignedProjMatrix : this.projMatrix, posMatrix);\n\n cache[posMatrixKey] = new Float32Array(posMatrix);\n return cache[posMatrixKey];\n }\n\n customLayerMatrix(): mat4 {\n return this.mercatorMatrix.slice() as any;\n }\n\n _constrain() {\n if (!this.center || !this.width || !this.height || this._constraining) return;\n\n this._constraining = true;\n\n let minY = -90;\n let maxY = 90;\n let minX = -180;\n let maxX = 180;\n let sy, sx, x2, y2;\n const size = this.size,\n unmodified = this._unmodified;\n\n if (this.latRange) {\n const latRange = this.latRange;\n minY = mercatorYfromLat(latRange[1]) * this.worldSize;\n maxY = mercatorYfromLat(latRange[0]) * this.worldSize;\n sy = maxY - minY < size.y ? size.y / (maxY - minY) : 0;\n }\n\n if (this.lngRange) {\n const lngRange = this.lngRange;\n\n minX = wrap(\n mercatorXfromLng(lngRange[0]) * this.worldSize,\n 0,\n this.worldSize\n );\n maxX = wrap(\n mercatorXfromLng(lngRange[1]) * this.worldSize,\n 0,\n this.worldSize\n );\n\n if (maxX < minX) maxX += this.worldSize;\n\n sx = maxX - minX < size.x ? size.x / (maxX - minX) : 0;\n }\n\n const point = this.point;\n\n // how much the map should scale to fit the screen into given latitude/longitude ranges\n const s = Math.max(sx || 0, sy || 0);\n\n if (s) {\n this.center = this.unproject(new Point(\n sx ? (maxX + minX) / 2 : point.x,\n sy ? (maxY + minY) / 2 : point.y));\n this.zoom += this.scaleZoom(s);\n this._unmodified = unmodified;\n this._constraining = false;\n return;\n }\n\n if (this.latRange) {\n const y = point.y,\n h2 = size.y / 2;\n\n if (y - h2 < minY) y2 = minY + h2;\n if (y + h2 > maxY) y2 = maxY - h2;\n }\n\n if (this.lngRange) {\n const centerX = (minX + maxX) / 2;\n const x = wrap(point.x, centerX - this.worldSize / 2, centerX + this.worldSize / 2);\n const w2 = size.x / 2;\n\n if (x - w2 < minX) x2 = minX + w2;\n if (x + w2 > maxX) x2 = maxX - w2;\n }\n\n // pan the map if the screen goes off the range\n if (x2 !== undefined || y2 !== undefined) {\n this.center = this.unproject(new Point(\n x2 !== undefined ? x2 : point.x,\n y2 !== undefined ? y2 : point.y)).wrap();\n }\n\n this._unmodified = unmodified;\n this._constraining = false;\n }\n\n _calcMatrices() {\n if (!this.height) return;\n\n const halfFov = this._fov / 2;\n const offset = this.centerOffset;\n const x = this.point.x, y = this.point.y;\n this.cameraToCenterDistance = 0.5 / Math.tan(halfFov) * this.height;\n this._pixelPerMeter = mercatorZfromAltitude(1, this.center.lat) * this.worldSize;\n\n let m = mat4.identity(new Float64Array(16) as any);\n mat4.scale(m, m, [this.width / 2, -this.height / 2, 1]);\n mat4.translate(m, m, [1, -1, 0]);\n this.labelPlaneMatrix = m;\n\n m = mat4.identity(new Float64Array(16) as any);\n mat4.scale(m, m, [1, -1, 1]);\n mat4.translate(m, m, [-1, -1, 0]);\n mat4.scale(m, m, [2 / this.width, 2 / this.height, 1]);\n this.glCoordMatrix = m;\n\n // Calculate the camera to sea-level distance in pixel in respect of terrain\n const cameraToSeaLevelDistance = this.cameraToCenterDistance + this._elevation * this._pixelPerMeter / Math.cos(this._pitch);\n // In case of negative minimum elevation (e.g. the dead see, under the sea maps) use a lower plane for calculation\n const minElevation = Math.min(this.elevation, this._minEleveationForCurrentTile);\n const cameraToLowestPointDistance = cameraToSeaLevelDistance - minElevation * this._pixelPerMeter / Math.cos(this._pitch);\n const lowestPlane = minElevation < 0 ? cameraToLowestPointDistance : cameraToSeaLevelDistance;\n\n // Find the distance from the center point [width/2 + offset.x, height/2 + offset.y] to the\n // center top point [width/2 + offset.x, 0] in Z units, using the law of sines.\n // 1 Z unit is equivalent to 1 horizontal px at the center of the map\n // (the distance between[width/2, height/2] and [width/2 + 1, height/2])\n const groundAngle = Math.PI / 2 + this._pitch;\n const fovAboveCenter = this._fov * (0.5 + offset.y / this.height);\n const topHalfSurfaceDistance = Math.sin(fovAboveCenter) * lowestPlane / Math.sin(clamp(Math.PI - groundAngle - fovAboveCenter, 0.01, Math.PI - 0.01));\n\n // Find the distance from the center point to the horizon\n const horizon = this.getHorizon();\n const horizonAngle = Math.atan(horizon / this.cameraToCenterDistance);\n const fovCenterToHorizon = 2 * horizonAngle * (0.5 + offset.y / (horizon * 2));\n const topHalfSurfaceDistanceHorizon = Math.sin(fovCenterToHorizon) * lowestPlane / Math.sin(clamp(Math.PI - groundAngle - fovCenterToHorizon, 0.01, Math.PI - 0.01));\n\n // Calculate z distance of the farthest fragment that should be rendered.\n // Add a bit extra to avoid precision problems when a fragment's distance is exactly `furthestDistance`\n const topHalfMinDistance = Math.min(topHalfSurfaceDistance, topHalfSurfaceDistanceHorizon);\n const farZ = (Math.cos(Math.PI / 2 - this._pitch) * topHalfMinDistance + lowestPlane) * 1.01;\n\n // The larger the value of nearZ is\n // - the more depth precision is available for features (good)\n // - clipping starts appearing sooner when the camera is close to 3d features (bad)\n //\n // Smaller values worked well for mapbox-gl-js but deckgl was encountering precision issues\n // when rendering it's layers using custom layers. This value was experimentally chosen and\n // seems to solve z-fighting issues in deckgl while not clipping buildings too close to the camera.\n const nearZ = this.height / 50;\n\n // matrix for conversion from location to GL coordinates (-1 .. 1)\n m = new Float64Array(16) as any;\n mat4.perspective(m, this._fov, this.width / this.height, nearZ, farZ);\n\n // Apply center of perspective offset\n m[8] = -offset.x * 2 / this.width;\n m[9] = offset.y * 2 / this.height;\n\n mat4.scale(m, m, [1, -1, 1]);\n mat4.translate(m, m, [0, 0, -this.cameraToCenterDistance]);\n mat4.rotateX(m, m, this._pitch);\n mat4.rotateZ(m, m, this.angle);\n mat4.translate(m, m, [-x, -y, 0]);\n\n // The mercatorMatrix can be used to transform points from mercator coordinates\n // ([0, 0] nw, [1, 1] se) to GL coordinates.\n this.mercatorMatrix = mat4.scale([] as any, m, [this.worldSize, this.worldSize, this.worldSize]);\n\n // scale vertically to meters per pixel (inverse of ground resolution):\n mat4.scale(m, m, [1, 1, this._pixelPerMeter]);\n\n // matrix for conversion from location to screen coordinates in 2D\n this.pixelMatrix = mat4.multiply(new Float64Array(16) as any, this.labelPlaneMatrix, m);\n\n // matrix for conversion from location to GL coordinates (-1 .. 1)\n mat4.translate(m, m, [0, 0, -this.elevation]); // elevate camera over terrain\n this.projMatrix = m;\n this.invProjMatrix = mat4.invert([] as any, m);\n\n // matrix for conversion from location to screen coordinates in 2D\n this.pixelMatrix3D = mat4.multiply(new Float64Array(16) as any, this.labelPlaneMatrix, m);\n\n // Make a second projection matrix that is aligned to a pixel grid for rendering raster tiles.\n // We're rounding the (floating point) x/y values to achieve to avoid rendering raster images to fractional\n // coordinates. Additionally, we adjust by half a pixel in either direction in case that viewport dimension\n // is an odd integer to preserve rendering to the pixel grid. We're rotating this shift based on the angle\n // of the transformation so that 0°, 90°, 180°, and 270° rasters are crisp, and adjust the shift so that\n // it is always <= 0.5 pixels.\n const xShift = (this.width % 2) / 2, yShift = (this.height % 2) / 2,\n angleCos = Math.cos(this.angle), angleSin = Math.sin(this.angle),\n dx = x - Math.round(x) + angleCos * xShift + angleSin * yShift,\n dy = y - Math.round(y) + angleCos * yShift + angleSin * xShift;\n const alignedM = new Float64Array(m) as any as mat4;\n mat4.translate(alignedM, alignedM, [dx > 0.5 ? dx - 1 : dx, dy > 0.5 ? dy - 1 : dy, 0]);\n this.alignedProjMatrix = alignedM;\n\n // inverse matrix for conversion from screen coordinaes to location\n m = mat4.invert(new Float64Array(16) as any, this.pixelMatrix);\n if (!m) throw new Error('failed to invert matrix');\n this.pixelMatrixInverse = m;\n\n this._posMatrixCache = {};\n this._alignedPosMatrixCache = {};\n }\n\n maxPitchScaleFactor() {\n // calcMatrices hasn't run yet\n if (!this.pixelMatrixInverse) return 1;\n\n const coord = this.pointCoordinate(new Point(0, 0));\n const p = [coord.x * this.worldSize, coord.y * this.worldSize, 0, 1] as vec4;\n const topPoint = vec4.transformMat4(p, p, this.pixelMatrix);\n return topPoint[3] / this.cameraToCenterDistance;\n }\n\n /**\n * The camera looks at the map from a 3D (lng, lat, altitude) location. Let's use `cameraLocation`\n * as the name for the location under the camera and on the surface of the earth (lng, lat, 0).\n * `cameraPoint` is the projected position of the `cameraLocation`.\n *\n * This point is useful to us because only fill-extrusions that are between `cameraPoint` and\n * the query point on the surface of the earth can extend and intersect the query.\n *\n * When the map is not pitched the `cameraPoint` is equivalent to the center of the map because\n * the camera is right above the center of the map.\n */\n getCameraPoint() {\n const pitch = this._pitch;\n const yOffset = Math.tan(pitch) * (this.cameraToCenterDistance || 1);\n return this.centerPoint.add(new Point(0, yOffset));\n }\n\n /**\n * When the map is pitched, some of the 3D features that intersect a query will not intersect\n * the query at the surface of the earth. Instead the feature may be closer and only intersect\n * the query because it extrudes into the air.\n * @param queryGeometry - For point queries, the line from the query point to the \"camera point\",\n * for other geometries, the envelope of the query geometry and the \"camera point\"\n * @returns a geometry that includes all of the original query as well as all possible ares of the\n * screen where the *base* of a visible extrusion could be.\n *\n */\n getCameraQueryGeometry(queryGeometry: Array): Array {\n const c = this.getCameraPoint();\n\n if (queryGeometry.length === 1) {\n return [queryGeometry[0], c];\n } else {\n let minX = c.x;\n let minY = c.y;\n let maxX = c.x;\n let maxY = c.y;\n for (const p of queryGeometry) {\n minX = Math.min(minX, p.x);\n minY = Math.min(minY, p.y);\n maxX = Math.max(maxX, p.x);\n maxY = Math.max(maxY, p.y);\n }\n return [\n new Point(minX, minY),\n new Point(maxX, minY),\n new Point(maxX, maxY),\n new Point(minX, maxY),\n new Point(minX, minY)\n ];\n }\n }\n}\n","/**\n * Throttle the given function to run at most every `period` milliseconds.\n */\nexport function throttle void>(fn: T, time: number): (...args: Parameters) => ReturnType {\n let pending = false;\n let timerId: ReturnType = null;\n let lastCallContext = null;\n let lastCallArgs: Parameters;\n\n const later = () => {\n timerId = null;\n if (pending) {\n fn.apply(lastCallContext, lastCallArgs);\n timerId = setTimeout(later, time);\n pending = false;\n }\n };\n\n return (...args: Parameters) => {\n pending = true;\n lastCallContext = this;\n lastCallArgs = args;\n if (!timerId) {\n later();\n }\n return timerId;\n };\n}\n","import {throttle} from '../util/throttle';\n\nimport type {Map} from './map';\n\n/**\n * Adds the map's position to its page's location hash.\n * Passed as an option to the map object.\n *\n * @group Markers and Controls\n */\nexport class Hash {\n _map: Map;\n _hashName: string;\n\n constructor(hashName?: string | null) {\n this._hashName = hashName && encodeURIComponent(hashName);\n }\n\n /**\n * Map element to listen for coordinate changes\n *\n * @param map - The map object\n * @returns `this`\n */\n addTo(map: Map) {\n this._map = map;\n addEventListener('hashchange', this._onHashChange, false);\n this._map.on('moveend', this._updateHash);\n return this;\n }\n\n /**\n * Removes hash\n *\n * @returns `this`\n */\n remove() {\n removeEventListener('hashchange', this._onHashChange, false);\n this._map.off('moveend', this._updateHash);\n clearTimeout(this._updateHash());\n\n delete this._map;\n return this;\n }\n\n getHashString(mapFeedback?: boolean) {\n const center = this._map.getCenter(),\n zoom = Math.round(this._map.getZoom() * 100) / 100,\n // derived from equation: 512px * 2^z / 360 / 10^d < 0.5px\n precision = Math.ceil((zoom * Math.LN2 + Math.log(512 / 360 / 0.5)) / Math.LN10),\n m = Math.pow(10, precision),\n lng = Math.round(center.lng * m) / m,\n lat = Math.round(center.lat * m) / m,\n bearing = this._map.getBearing(),\n pitch = this._map.getPitch();\n let hash = '';\n if (mapFeedback) {\n // new map feedback site has some constraints that don't allow\n // us to use the same hash format as we do for the Map hash option.\n hash += `/${lng}/${lat}/${zoom}`;\n } else {\n hash += `${zoom}/${lat}/${lng}`;\n }\n\n if (bearing || pitch) hash += (`/${Math.round(bearing * 10) / 10}`);\n if (pitch) hash += (`/${Math.round(pitch)}`);\n\n if (this._hashName) {\n const hashName = this._hashName;\n let found = false;\n const parts = window.location.hash.slice(1).split('&').map(part => {\n const key = part.split('=')[0];\n if (key === hashName) {\n found = true;\n return `${key}=${hash}`;\n }\n return part;\n }).filter(a => a);\n if (!found) {\n parts.push(`${hashName}=${hash}`);\n }\n return `#${parts.join('&')}`;\n }\n\n return `#${hash}`;\n }\n\n _getCurrentHash = () => {\n // Get the current hash from location, stripped from its number sign\n const hash = window.location.hash.replace('#', '');\n if (this._hashName) {\n // Split the parameter-styled hash into parts and find the value we need\n let keyval;\n hash.split('&').map(\n part => part.split('=')\n ).forEach(part => {\n if (part[0] === this._hashName) {\n keyval = part;\n }\n });\n return (keyval ? keyval[1] || '' : '').split('/');\n }\n return hash.split('/');\n };\n\n _onHashChange = () => {\n const loc = this._getCurrentHash();\n if (loc.length >= 3 && !loc.some(v => isNaN(v))) {\n const bearing = this._map.dragRotate.isEnabled() && this._map.touchZoomRotate.isEnabled() ? +(loc[3] || 0) : this._map.getBearing();\n this._map.jumpTo({\n center: [+loc[2], +loc[1]],\n zoom: +loc[0],\n bearing,\n pitch: +(loc[4] || 0)\n });\n return true;\n }\n return false;\n };\n\n _updateHashUnthrottled = () => {\n // Replace if already present, else append the updated hash string\n const location = window.location.href.replace(/(#.+)?$/, this.getHashString());\n try {\n window.history.replaceState(window.history.state, null, location);\n } catch (SecurityError) {\n // IE11 does not allow this if the page is within an iframe created\n // with iframe.contentWindow.document.write(...).\n // https://github.com/mapbox/mapbox-gl-js/issues/7410\n }\n };\n\n /**\n * Mobile Safari doesn't allow updating the hash more than 100 times per 30 seconds.\n */\n _updateHash: () => ReturnType = throttle(this._updateHashUnthrottled, 30 * 1000 / 100);\n}\n","import {browser} from '../util/browser';\nimport type {Map} from './map';\nimport {bezier, clamp, extend} from '../util/util';\nimport Point from '@mapbox/point-geometry';\nimport type {DragPanOptions} from './handler/shim/drag_pan';\n\nconst defaultInertiaOptions = {\n linearity: 0.3,\n easing: bezier(0, 0, 0.3, 1),\n};\n\nconst defaultPanInertiaOptions = extend({\n deceleration: 2500,\n maxSpeed: 1400\n}, defaultInertiaOptions);\n\nconst defaultZoomInertiaOptions = extend({\n deceleration: 20,\n maxSpeed: 1400\n}, defaultInertiaOptions);\n\nconst defaultBearingInertiaOptions = extend({\n deceleration: 1000,\n maxSpeed: 360\n}, defaultInertiaOptions);\n\nconst defaultPitchInertiaOptions = extend({\n deceleration: 1000,\n maxSpeed: 90\n}, defaultInertiaOptions);\n\nexport type InertiaOptions = {\n linearity: number;\n easing: (t: number) => number;\n deceleration: number;\n maxSpeed: number;\n};\n\nexport type InputEvent = MouseEvent | TouchEvent | KeyboardEvent | WheelEvent;\n\nexport class HandlerInertia {\n _map: Map;\n _inertiaBuffer: Array<{\n time: number;\n settings: any;\n }>;\n\n constructor(map: Map) {\n this._map = map;\n this.clear();\n }\n\n clear() {\n this._inertiaBuffer = [];\n }\n\n record(settings: any) {\n this._drainInertiaBuffer();\n this._inertiaBuffer.push({time: browser.now(), settings});\n }\n\n _drainInertiaBuffer() {\n const inertia = this._inertiaBuffer,\n now = browser.now(),\n cutoff = 160; //msec\n\n while (inertia.length > 0 && now - inertia[0].time > cutoff)\n inertia.shift();\n }\n\n _onMoveEnd(panInertiaOptions?: DragPanOptions | boolean) {\n this._drainInertiaBuffer();\n if (this._inertiaBuffer.length < 2) {\n return;\n }\n\n const deltas = {\n zoom: 0,\n bearing: 0,\n pitch: 0,\n pan: new Point(0, 0),\n pinchAround: undefined,\n around: undefined\n };\n\n for (const {settings} of this._inertiaBuffer) {\n deltas.zoom += settings.zoomDelta || 0;\n deltas.bearing += settings.bearingDelta || 0;\n deltas.pitch += settings.pitchDelta || 0;\n if (settings.panDelta) deltas.pan._add(settings.panDelta);\n if (settings.around) deltas.around = settings.around;\n if (settings.pinchAround) deltas.pinchAround = settings.pinchAround;\n }\n\n const lastEntry = this._inertiaBuffer[this._inertiaBuffer.length - 1];\n const duration = (lastEntry.time - this._inertiaBuffer[0].time);\n\n const easeOptions = {} as any;\n\n if (deltas.pan.mag()) {\n const result = calculateEasing(deltas.pan.mag(), duration, extend({}, defaultPanInertiaOptions, panInertiaOptions || {}));\n easeOptions.offset = deltas.pan.mult(result.amount / deltas.pan.mag());\n easeOptions.center = this._map.transform.center;\n extendDuration(easeOptions, result);\n }\n\n if (deltas.zoom) {\n const result = calculateEasing(deltas.zoom, duration, defaultZoomInertiaOptions);\n easeOptions.zoom = this._map.transform.zoom + result.amount;\n extendDuration(easeOptions, result);\n }\n\n if (deltas.bearing) {\n const result = calculateEasing(deltas.bearing, duration, defaultBearingInertiaOptions);\n easeOptions.bearing = this._map.transform.bearing + clamp(result.amount, -179, 179);\n extendDuration(easeOptions, result);\n }\n\n if (deltas.pitch) {\n const result = calculateEasing(deltas.pitch, duration, defaultPitchInertiaOptions);\n easeOptions.pitch = this._map.transform.pitch + result.amount;\n extendDuration(easeOptions, result);\n }\n\n if (easeOptions.zoom || easeOptions.bearing) {\n const last = deltas.pinchAround === undefined ? deltas.around : deltas.pinchAround;\n easeOptions.around = last ? this._map.unproject(last) : this._map.getCenter();\n }\n\n this.clear();\n return extend(easeOptions, {\n noMoveStart: true\n });\n\n }\n}\n\n// Unfortunately zoom, bearing, etc can't have different durations and easings so\n// we need to choose one. We use the longest duration and it's corresponding easing.\nfunction extendDuration(easeOptions, result) {\n if (!easeOptions.duration || easeOptions.duration < result.duration) {\n easeOptions.duration = result.duration;\n easeOptions.easing = result.easing;\n }\n}\n\nfunction calculateEasing(amount, inertiaDuration: number, inertiaOptions) {\n const {maxSpeed, linearity, deceleration} = inertiaOptions;\n const speed = clamp(\n amount * linearity / (inertiaDuration / 1000),\n -maxSpeed,\n maxSpeed);\n const duration = Math.abs(speed) / (deceleration * linearity);\n return {\n easing: inertiaOptions.easing,\n duration: duration * 1000,\n amount: speed * (duration / 2)\n };\n}\n","import {Event} from '../util/evented';\n\nimport {DOM} from '../util/dom';\nimport Point from '@mapbox/point-geometry';\nimport {extend} from '../util/util';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\n\nimport type {Map} from './map';\nimport type {LngLat} from '../geo/lng_lat';\nimport {SourceSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * An event from the mouse relevant to a specific layer.\n *\n * @group Event Related\n */\nexport type MapLayerMouseEvent = MapMouseEvent & { features?: MapGeoJSONFeature[] };\n\n/**\n * An event from a touch device relevat to a specific layer.\n *\n * @group Event Related\n */\nexport type MapLayerTouchEvent = MapTouchEvent & { features?: MapGeoJSONFeature[] };\n\n/**\n * The source event data type\n */\nexport type MapSourceDataType = 'content' | 'metadata' | 'visibility' | 'idle';\n\n/**\n * `MapLayerEventType` - a mapping between the event name and the event.\n * **Note:** These events are compatible with the optional `layerId` parameter.\n * If `layerId` is included as the second argument in {@link Map#on}, the event listener will fire only when the\n * event action contains a visible portion of the specified layer.\n * The following example can be used for all the events.\n *\n * @group Event Related\n * @example\n * ```ts\n * // Initialize the map\n * let map = new maplibregl.Map({ // map options });\n * // Set an event listener for a specific layer\n * map.on('the-event-name', 'poi-label', function(e) {\n * console.log('An event has occurred on a visible portion of the poi-label layer');\n * });\n * ```\n */\nexport type MapLayerEventType = {\n /**\n * Fired when a pointing device (usually a mouse) is pressed and released contains a visible portion of the specified layer.\n *\n * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/)\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n */\n click: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed and released twice contains a visible portion of the specified layer.\n *\n * **Note:** Under normal conditions, this event will be preceded by two `click` events.\n */\n dblclick: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed while inside a visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n mousedown: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is released while inside a visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n mouseup: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is moved while the cursor is inside a visible portion of the specified layer.\n * As you move the cursor across the layer, the event will fire every time the cursor changes position within that layer.\n *\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on over](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n */\n mousemove: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) enters a visible portion of a specified layer from\n * outside that layer or outside the map canvas.\n *\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n */\n mouseenter: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) leaves a visible portion of a specified layer, or leaves\n * the map canvas.\n *\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n */\n mouseleave: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is moved inside a visible portion of the specified layer.\n *\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n */\n mouseover: MapLayerMouseEvent;\n /**\n * Fired when a point device (usually a mouse) leaves the visible portion of the specified layer.\n */\n mouseout: MapLayerMouseEvent;\n /**\n * Fired when the right button of the mouse is clicked or the context menu key is pressed within visible portion of the specified layer.\n */\n contextmenu: MapLayerMouseEvent;\n /**\n * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchstart: MapLayerTouchEvent;\n /**\n * Fired when a [`touchend`](https://developer.mozilla.org/en-US/docs/Web/Events/touchend) event occurs within the visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchend: MapLayerTouchEvent;\n /**\n * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchcancel: MapLayerTouchEvent;\n};\n\n/**\n * `MapEventType` - a mapping between the event name and the event value.\n * These events are used with the {@link Map#on} method.\n * When using a `layerId` with {@link Map#on} method, please refer to {@link MapLayerEventType}.\n * The following example can be used for all the events.\n *\n * @group Event Related\n * @example\n * ```ts\n * // Initialize the map\n * let map = new maplibregl.Map({ // map options });\n * // Set an event listener\n * map.on('the-event-name', () => {\n * console.log('An event has occurred!');\n * });\n * ```\n */\nexport type MapEventType = {\n /**\n * Fired when an error occurs. This is GL JS's primary error reporting\n * mechanism. We use an event instead of `throw` to better accommodate\n * asynchronous operations. If no listeners are bound to the `error` event, the\n * error will be printed to the console.\n */\n error: ErrorEvent;\n /**\n * @event `load` Fired immediately after all necessary resources have been downloaded\n * and the first visually complete rendering of the map has occurred.\n *\n * @see [Draw GeoJSON points](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-markers/)\n * @see [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/)\n * @see [Animate a point](https://maplibre.org/maplibre-gl-js/docs/examples/animate-point-along-line/)\n */\n load: MapLibreEvent;\n /**\n * Fired after the last frame rendered before the map enters an\n * \"idle\" state:\n *\n * - No camera transitions are in progress\n * - All currently requested tiles have loaded\n * - All fade/transition animations have completed\n */\n idle: MapLibreEvent;\n /**\n * Fired immediately after the map has been removed with {@link Map#remove}.\n */\n remove: MapLibreEvent;\n /**\n * Fired whenever the map is drawn to the screen, as the result of\n *\n * - a change to the map's position, zoom, pitch, or bearing\n * - a change to the map's style\n * - a change to a GeoJSON source\n * - the loading of a vector tile, GeoJSON file, glyph, or sprite\n */\n render: MapLibreEvent;\n /**\n * Fired immediately after the map has been resized.\n */\n resize: MapLibreEvent;\n /**\n * Fired when the WebGL context is lost.\n */\n webglcontextlost: MapContextEvent;\n /**\n * Fired when the WebGL context is restored.\n */\n webglcontextrestored: MapContextEvent;\n /**\n * Fired when any map data (style, source, tile, etc) begins loading or\n * changing asynchronously. All `dataloading` events are followed by a `data`,\n * `dataabort` or `error` event.\n */\n dataloading: MapDataEvent;\n /**\n * Fired when any map data loads or changes. See {@link MapDataEvent} for more information.\n * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/)\n */\n data: MapDataEvent;\n tiledataloading: MapDataEvent;\n /**\n * Fired when one of the map's sources begins loading or changing asynchronously.\n * All `sourcedataloading` events are followed by a `sourcedata`, `sourcedataabort` or `error` event.\n */\n sourcedataloading: MapSourceDataEvent;\n /**\n * Fired when the map's style begins loading or changing asynchronously.\n * All `styledataloading` events are followed by a `styledata`\n * or `error` event.\n */\n styledataloading: MapStyleDataEvent;\n /**\n * Fired when one of the map's sources loads or changes, including if a tile belonging\n * to a source loads or changes.\n */\n sourcedata: MapSourceDataEvent;\n /**\n * Fired when the map's style loads or changes.\n */\n styledata: MapStyleDataEvent;\n /**\n * Fired when an icon or pattern needed by the style is missing. The missing image can\n * be added with {@link Map#addImage} within this event listener callback to prevent the image from\n * being skipped. This event can be used to dynamically generate icons and patterns.\n * @see [Generate and add a missing icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-missing-generated/)\n */\n styleimagemissing: MapStyleImageMissingEvent;\n /**\n * Fired when a request for one of the map's sources' tiles or data is aborted.\n */\n dataabort: MapDataEvent;\n /**\n * Fired when a request for one of the map's sources' data is aborted.\n */\n sourcedataabort: MapSourceDataEvent;\n /**\n * Fired when the user cancels a \"box zoom\" interaction, or when the bounding box does not meet the minimum size threshold.\n * See {@link BoxZoomHandler}.\n */\n boxzoomcancel: MapLibreZoomEvent;\n /**\n * Fired when a \"box zoom\" interaction starts. See {@link BoxZoomHandler}.\n */\n boxzoomstart: MapLibreZoomEvent;\n /**\n * Fired when a \"box zoom\" interaction ends. See {@link BoxZoomHandler}.\n */\n boxzoomend: MapLibreZoomEvent;\n /**\n * Fired when a [`touchcancel`](https://developer.mozilla.org/en-US/docs/Web/Events/touchcancel) event occurs within the map.\n */\n touchcancel: MapTouchEvent;\n /**\n * Fired when a [`touchmove`](https://developer.mozilla.org/en-US/docs/Web/Events/touchmove) event occurs within the map.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchmove: MapTouchEvent;\n /**\n * Fired when a [`touchend`](https://developer.mozilla.org/en-US/docs/Web/Events/touchend) event occurs within the map.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchend: MapTouchEvent;\n /**\n * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the map.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchstart: MapTouchEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed and released at the same point on the map.\n *\n * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/)\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n */\n click: MapMouseEvent;\n /**\n * Fired when the right button of the mouse is clicked or the context menu key is pressed within the map.\n */\n contextmenu: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed and released twice at the same point on the map in rapid succession.\n *\n * **Note:** Under normal conditions, this event will be preceded by two `click` events.\n */\n dblclick: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is moved while the cursor is inside the map.\n * As you move the cursor across the map, the event will fire every time the cursor changes position within the map.\n *\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on over](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n */\n mousemove: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is released within the map.\n *\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n mouseup: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed within the map.\n *\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n mousedown: MapMouseEvent;\n /**\n * Fired when a point device (usually a mouse) leaves the map's canvas.\n */\n mouseout: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is moved within the map.\n * As you move the cursor across a web page containing a map,\n * the event will fire each time it enters the map or any child elements.\n *\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n */\n mouseover: MapMouseEvent;\n /**\n * Fired just before the map begins a transition from one\n * view to another, as the result of either user interaction or methods such as {@link Map#jumpTo}.\n *\n */\n movestart: MapLibreEvent;\n /**\n * Fired repeatedly during an animated transition from one view to\n * another, as the result of either user interaction or methods such as {@link Map#flyTo}.\n *\n * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/)\n */\n move: MapLibreEvent;\n /**\n * Fired just after the map completes a transition from one\n * view to another, as the result of either user interaction or methods such as {@link Map#jumpTo}.\n *\n * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/)\n */\n moveend: MapLibreEvent;\n /**\n * Fired just before the map begins a transition from one zoom level to another,\n * as the result of either user interaction or methods such as {@link Map#flyTo}.\n */\n zoomstart: MapLibreEvent;\n /**\n * Fired repeatedly during an animated transition from one zoom level to another,\n * as the result of either user interaction or methods such as {@link Map#flyTo}.\n */\n zoom: MapLibreEvent;\n /**\n * Fired just after the map completes a transition from one zoom level to another,\n * as the result of either user interaction or methods such as {@link Map#flyTo}.\n */\n zoomend: MapLibreEvent;\n /**\n * Fired when a \"drag to rotate\" interaction starts. See {@link DragRotateHandler}.\n */\n rotatestart: MapLibreEvent;\n /**\n * Fired repeatedly during a \"drag to rotate\" interaction. See {@link DragRotateHandler}.\n */\n rotate: MapLibreEvent;\n /**\n * Fired when a \"drag to rotate\" interaction ends. See {@link DragRotateHandler}.\n */\n rotateend: MapLibreEvent;\n /**\n * Fired when a \"drag to pan\" interaction starts. See {@link DragPanHandler}.\n */\n dragstart: MapLibreEvent;\n /**\n * Fired repeatedly during a \"drag to pan\" interaction. See {@link DragPanHandler}.\n */\n drag: MapLibreEvent;\n /**\n * Fired when a \"drag to pan\" interaction ends. See {@link DragPanHandler}.\n * @see [Create a draggable marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/)\n */\n dragend: MapLibreEvent;\n /**\n * Fired whenever the map's pitch (tilt) begins a change as\n * the result of either user interaction or methods such as {@link Map#flyTo} .\n */\n pitchstart: MapLibreEvent;\n /**\n * Fired repeatedly during the map's pitch (tilt) animation between\n * one state and another as the result of either user interaction\n * or methods such as {@link Map#flyTo}.\n */\n pitch: MapLibreEvent;\n /**\n * Fired immediately after the map's pitch (tilt) finishes changing as\n * the result of either user interaction or methods such as {@link Map#flyTo}.\n */\n pitchend: MapLibreEvent;\n /**\n * Fired when a [`wheel`](https://developer.mozilla.org/en-US/docs/Web/Events/wheel) event occurs within the map.\n */\n wheel: MapWheelEvent;\n /**\n * Fired when terrain is changed\n */\n terrain: MapTerrainEvent;\n};\n\n/**\n * The base event for MapLibre\n *\n * @group Event Related\n */\nexport type MapLibreEvent = {\n type: keyof MapEventType | keyof MapLayerEventType;\n target: Map;\n originalEvent: TOrig;\n}\n\n/**\n * The style data event\n *\n * @group Event Related\n */\nexport type MapStyleDataEvent = MapLibreEvent & {\n dataType: 'style';\n}\n\n/**\n * The source data event interface\n *\n * @group Event Related\n */\nexport type MapSourceDataEvent = MapLibreEvent & {\n dataType: 'source';\n /**\n * True if the event has a `dataType` of `source` and the source has no outstanding network requests.\n */\n isSourceLoaded: boolean;\n /**\n * The [style spec representation of the source](https://maplibre.org/maplibre-style-spec/#sources) if the event has a `dataType` of `source`.\n */\n source: SourceSpecification;\n sourceId: string;\n sourceDataType: MapSourceDataType;\n /**\n * The tile being loaded or changed, if the event has a `dataType` of `source` and\n * the event is related to loading of a tile.\n */\n tile: any;\n}\n/**\n * `MapMouseEvent` is the event type for mouse-related map events.\n * @example\n * ```ts\n * // The `click` event is an example of a `MapMouseEvent`.\n * // Set up an event listener on the map.\n * map.on('click', function(e) {\n * // The event object (e) contains information like the\n * // coordinates of the point on the map that was clicked.\n * console.log('A click event has occurred at ' + e.lngLat);\n * });\n * ```\n */\nexport class MapMouseEvent extends Event implements MapLibreEvent {\n /**\n * The event type\n */\n type: 'mousedown' | 'mouseup' | 'click' | 'dblclick' | 'mousemove' | 'mouseover' | 'mouseenter' | 'mouseleave' | 'mouseout' | 'contextmenu';\n\n /**\n * The `Map` object that fired the event.\n */\n target: Map;\n\n /**\n * The DOM event which caused the map event.\n */\n originalEvent: MouseEvent;\n\n /**\n * The pixel coordinates of the mouse cursor, relative to the map and measured from the top left corner.\n */\n point: Point;\n\n /**\n * The geographic location on the map of the mouse cursor.\n */\n lngLat: LngLat;\n\n /**\n * Prevents subsequent default processing of the event by the map.\n *\n * Calling this method will prevent the following default map behaviors:\n *\n * * On `mousedown` events, the behavior of {@link DragPanHandler}\n * * On `mousedown` events, the behavior of {@link DragRotateHandler}\n * * On `mousedown` events, the behavior of {@link BoxZoomHandler}\n * * On `dblclick` events, the behavior of {@link DoubleClickZoomHandler}\n *\n */\n preventDefault() {\n this._defaultPrevented = true;\n }\n\n /**\n * `true` if `preventDefault` has been called.\n */\n get defaultPrevented(): boolean {\n return this._defaultPrevented;\n }\n\n _defaultPrevented: boolean;\n\n constructor(type: string, map: Map, originalEvent: MouseEvent, data: any = {}) {\n const point = DOM.mousePos(map.getCanvasContainer(), originalEvent);\n const lngLat = map.unproject(point);\n super(type, extend({point, lngLat, originalEvent}, data));\n this._defaultPrevented = false;\n this.target = map;\n }\n}\n\n/**\n * `MapTouchEvent` is the event type for touch-related map events.\n *\n * @group Event Related\n */\nexport class MapTouchEvent extends Event implements MapLibreEvent {\n /**\n * The event type.\n */\n type: 'touchstart' | 'touchmove' | 'touchend' | 'touchcancel';\n\n /**\n * The `Map` object that fired the event.\n */\n target: Map;\n\n /**\n * The DOM event which caused the map event.\n */\n originalEvent: TouchEvent;\n\n /**\n * The geographic location on the map of the center of the touch event points.\n */\n lngLat: LngLat;\n\n /**\n * The pixel coordinates of the center of the touch event points, relative to the map and measured from the top left\n * corner.\n */\n point: Point;\n\n /**\n * The array of pixel coordinates corresponding to a\n * [touch event's `touches`](https://developer.mozilla.org/en-US/docs/Web/API/TouchEvent/touches) property.\n */\n points: Array;\n\n /**\n * The geographical locations on the map corresponding to a\n * [touch event's `touches`](https://developer.mozilla.org/en-US/docs/Web/API/TouchEvent/touches) property.\n */\n lngLats: Array;\n\n /**\n * Prevents subsequent default processing of the event by the map.\n *\n * Calling this method will prevent the following default map behaviors:\n *\n * * On `touchstart` events, the behavior of {@link DragPanHandler}\n * * On `touchstart` events, the behavior of {@link TwoFingersTouchZoomRotateHandler}\n *\n */\n preventDefault() {\n this._defaultPrevented = true;\n }\n\n /**\n * `true` if `preventDefault` has been called.\n */\n get defaultPrevented(): boolean {\n return this._defaultPrevented;\n }\n\n _defaultPrevented: boolean;\n\n constructor(type: string, map: Map, originalEvent: TouchEvent) {\n const touches = type === 'touchend' ? originalEvent.changedTouches : originalEvent.touches;\n const points = DOM.touchPos(map.getCanvasContainer(), touches);\n const lngLats = points.map((t) => map.unproject(t));\n const point = points.reduce((prev, curr, i, arr) => {\n return prev.add(curr.div(arr.length));\n }, new Point(0, 0));\n const lngLat = map.unproject(point);\n super(type, {points, point, lngLats, lngLat, originalEvent});\n this._defaultPrevented = false;\n }\n}\n\n/**\n * `MapWheelEvent` is the event type for the `wheel` map event.\n *\n * @group Event Related\n */\nexport class MapWheelEvent extends Event {\n /**\n * The event type.\n */\n type: 'wheel';\n\n /**\n * The `Map` object that fired the event.\n */\n target: Map;\n\n /**\n * The DOM event which caused the map event.\n */\n originalEvent: WheelEvent;\n\n /**\n * Prevents subsequent default processing of the event by the map.\n *\n * Calling this method will prevent the behavior of {@link ScrollZoomHandler}.\n */\n preventDefault() {\n this._defaultPrevented = true;\n }\n\n /**\n * `true` if `preventDefault` has been called.\n */\n get defaultPrevented(): boolean {\n return this._defaultPrevented;\n }\n\n _defaultPrevented: boolean;\n\n /** */\n constructor(type: string, map: Map, originalEvent: WheelEvent) {\n super(type, {originalEvent});\n this._defaultPrevented = false;\n }\n}\n\n/**\n * A `MapLibreZoomEvent` is the event type for the boxzoom-related map events emitted by the {@link BoxZoomHandler}.\n *\n * @group Event Related\n */\nexport type MapLibreZoomEvent = {\n /**\n * The type of boxzoom event. One of `boxzoomstart`, `boxzoomend` or `boxzoomcancel`\n */\n type: 'boxzoomstart' | 'boxzoomend' | 'boxzoomcancel';\n /**\n * The `Map` instance that triggered the event\n */\n target: Map;\n /**\n * The DOM event that triggered the boxzoom event. Can be a `MouseEvent` or `KeyboardEvent`\n */\n originalEvent: MouseEvent;\n};\n\n/**\n * A `MapDataEvent` object is emitted with the `data`\n * and `dataloading` events. Possible values for\n * `dataType`s are:\n *\n * - `'source'`: The non-tile data associated with any source\n * - `'style'`: The [style](https://maplibre.org/maplibre-style-spec/) used by the map\n *\n * Possible values for `sourceDataType`s are:\n *\n * - `'metadata'`: indicates that any necessary source metadata has been loaded (such as TileJSON) and it is ok to start loading tiles\n * - `'content'`: indicates the source data has changed (such as when source.setData() has been called on GeoJSONSource)\n * - `'visibility'`: send when the source becomes used when at least one of its layers becomes visible in style sense (inside the layer's zoom range and with layout.visibility set to 'visible')\n * - `'idle'`: indicates that no new source data has been fetched (but the source has done loading)\n *\n * @group Event Related\n *\n * @example\n * ```ts\n * // The sourcedata event is an example of MapDataEvent.\n * // Set up an event listener on the map.\n * map.on('sourcedata', function(e) {\n * if (e.isSourceLoaded) {\n * // Do something when the source has finished loading\n * }\n * });\n * ```\n */\nexport type MapDataEvent = {\n /**\n * The event type.\n */\n type: string;\n /**\n * The type of data that has changed. One of `'source'`, `'style'`.\n */\n dataType: string;\n /**\n * Included if the event has a `dataType` of `source` and the event signals that internal data has been received or changed. Possible values are `metadata`, `content`, `visibility` and `idle`.\n */\n sourceDataType: MapSourceDataType;\n};\n\n/**\n * The terrain event\n *\n * @group Event Related\n */\nexport type MapTerrainEvent = {\n type: 'terrain';\n};\n\n/**\n * An event related to the web gl context\n *\n * @group Event Related\n */\nexport type MapContextEvent = {\n type: 'webglcontextlost' | 'webglcontextrestored';\n originalEvent: WebGLContextEvent;\n};\n\n/**\n * The style image missing event\n *\n * @group Event Related\n *\n * @see [Generate and add a missing icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-missing-generated/)\n */\nexport type MapStyleImageMissingEvent = MapLibreEvent & {\n type: 'styleimagemissing';\n id: string;\n}\n","import {MapMouseEvent, MapTouchEvent, MapWheelEvent} from '../events';\nimport {Handler} from '../handler_manager';\nimport type {Map} from '../map';\nimport type Point from '@mapbox/point-geometry';\n\nexport class MapEventHandler implements Handler {\n\n _mousedownPos: Point;\n _clickTolerance: number;\n _map: Map;\n\n constructor(map: Map, options: {\n clickTolerance: number;\n }) {\n this._map = map;\n this._clickTolerance = options.clickTolerance;\n }\n\n reset() {\n delete this._mousedownPos;\n }\n\n wheel(e: WheelEvent) {\n // If mapEvent.preventDefault() is called by the user, prevent handlers such as:\n // - ScrollZoom\n return this._firePreventable(new MapWheelEvent(e.type, this._map, e));\n }\n\n mousedown(e: MouseEvent, point: Point) {\n this._mousedownPos = point;\n // If mapEvent.preventDefault() is called by the user, prevent handlers such as:\n // - MousePan\n // - MouseRotate\n // - MousePitch\n // - DblclickHandler\n return this._firePreventable(new MapMouseEvent(e.type, this._map, e));\n }\n\n mouseup(e: MouseEvent) {\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n click(e: MouseEvent, point: Point) {\n if (this._mousedownPos && this._mousedownPos.dist(point) >= this._clickTolerance) return;\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n dblclick(e: MouseEvent) {\n // If mapEvent.preventDefault() is called by the user, prevent handlers such as:\n // - DblClickZoom\n return this._firePreventable(new MapMouseEvent(e.type, this._map, e));\n }\n\n mouseover(e: MouseEvent) {\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n mouseout(e: MouseEvent) {\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n touchstart(e: TouchEvent) {\n // If mapEvent.preventDefault() is called by the user, prevent handlers such as:\n // - TouchPan\n // - TouchZoom\n // - TouchRotate\n // - TouchPitch\n // - TapZoom\n // - SwipeZoom\n return this._firePreventable(new MapTouchEvent(e.type, this._map, e));\n }\n\n touchmove(e: TouchEvent) {\n this._map.fire(new MapTouchEvent(e.type, this._map, e));\n }\n\n touchend(e: TouchEvent) {\n this._map.fire(new MapTouchEvent(e.type, this._map, e));\n }\n\n touchcancel(e: TouchEvent) {\n this._map.fire(new MapTouchEvent(e.type, this._map, e));\n }\n\n _firePreventable(mapEvent: MapMouseEvent | MapTouchEvent | MapWheelEvent) {\n this._map.fire(mapEvent);\n if (mapEvent.defaultPrevented) {\n // returning an object marks the handler as active and resets other handlers\n return {};\n }\n }\n\n isEnabled() {\n return true;\n }\n\n isActive() {\n return false;\n }\n enable() {}\n disable() {}\n}\n\nexport class BlockableMapEventHandler {\n _map: Map;\n _delayContextMenu: boolean;\n _ignoreContextMenu: boolean;\n _contextMenuEvent: MouseEvent;\n\n constructor(map: Map) {\n this._map = map;\n }\n\n reset() {\n this._delayContextMenu = false;\n this._ignoreContextMenu = true;\n delete this._contextMenuEvent;\n }\n\n mousemove(e: MouseEvent) {\n // mousemove map events should not be fired when interaction handlers (pan, rotate, etc) are active\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n mousedown() {\n this._delayContextMenu = true;\n this._ignoreContextMenu = false;\n }\n\n mouseup() {\n this._delayContextMenu = false;\n if (this._contextMenuEvent) {\n this._map.fire(new MapMouseEvent('contextmenu', this._map, this._contextMenuEvent));\n delete this._contextMenuEvent;\n }\n }\n contextmenu(e: MouseEvent) {\n if (this._delayContextMenu) {\n // Mac: contextmenu fired on mousedown; we save it until mouseup for consistency's sake\n this._contextMenuEvent = e;\n } else if (!this._ignoreContextMenu) {\n // Windows: contextmenu fired on mouseup, so fire event now\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n // prevent browser context menu when necessary\n if (this._map.listens('contextmenu')) {\n e.preventDefault();\n }\n }\n\n isEnabled() {\n return true;\n }\n\n isActive() {\n return false;\n }\n enable() {}\n disable() {}\n}\n","import type {Map} from '../map';\nimport type {PointLike} from '../camera';\nimport type {Transform} from '../../geo/transform';\nimport Point from '@mapbox/point-geometry';\nimport {LngLat} from '../../geo/lng_lat';\n\n/**\n * @internal\n * Shared utilities for the Handler classes to access the correct camera state.\n * If Camera.transformCameraUpdate is specified, the \"desired state\" of camera may differ from the state used for rendering.\n * The handlers need the \"desired state\" to track accumulated changes.\n */\nexport class TransformProvider {\n _map: Map;\n\n constructor(map: Map) {\n this._map = map;\n }\n\n get transform(): Transform {\n return this._map._requestedCameraState || this._map.transform;\n }\n\n get center() {\n return {lng: this.transform.center.lng, lat: this.transform.center.lat};\n }\n\n get zoom() {\n return this.transform.zoom;\n }\n\n get pitch() {\n return this.transform.pitch;\n }\n\n get bearing() {\n return this.transform.bearing;\n }\n\n unproject(point: PointLike): LngLat {\n return this.transform.pointLocation(Point.convert(point), this._map.terrain);\n }\n}\n","import {DOM} from '../../util/dom';\n\nimport {Event} from '../../util/evented';\nimport {TransformProvider} from './transform-provider';\n\nimport type {Map} from '../map';\nimport type Point from '@mapbox/point-geometry';\nimport {Handler} from '../handler_manager';\n\n/**\n * The `BoxZoomHandler` allows the user to zoom the map to fit within a bounding box.\n * The bounding box is defined by clicking and holding `shift` while dragging the cursor.\n *\n * @group Handlers\n */\nexport class BoxZoomHandler implements Handler {\n _map: Map;\n _tr: TransformProvider;\n _el: HTMLElement;\n _container: HTMLElement;\n _enabled: boolean;\n _active: boolean;\n _startPos: Point;\n _lastPos: Point;\n _box: HTMLElement;\n _clickTolerance: number;\n\n /** @internal */\n constructor(map: Map, options: {\n clickTolerance: number;\n }) {\n this._map = map;\n this._tr = new TransformProvider(map);\n this._el = map.getCanvasContainer();\n this._container = map.getContainer();\n this._clickTolerance = options.clickTolerance || 1;\n }\n\n /**\n * Returns a Boolean indicating whether the \"box zoom\" interaction is enabled.\n *\n * @returns `true` if the \"box zoom\" interaction is enabled.\n */\n isEnabled() {\n return !!this._enabled;\n }\n\n /**\n * Returns a Boolean indicating whether the \"box zoom\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"box zoom\" interaction is active.\n */\n isActive() {\n return !!this._active;\n }\n\n /**\n * Enables the \"box zoom\" interaction.\n *\n * @example\n * ```ts\n * map.boxZoom.enable();\n * ```\n */\n enable() {\n if (this.isEnabled()) return;\n this._enabled = true;\n }\n\n /**\n * Disables the \"box zoom\" interaction.\n *\n * @example\n * ```ts\n * map.boxZoom.disable();\n * ```\n */\n disable() {\n if (!this.isEnabled()) return;\n this._enabled = false;\n }\n\n mousedown(e: MouseEvent, point: Point) {\n if (!this.isEnabled()) return;\n if (!(e.shiftKey && e.button === 0)) return;\n\n DOM.disableDrag();\n this._startPos = this._lastPos = point;\n this._active = true;\n }\n\n mousemoveWindow(e: MouseEvent, point: Point) {\n if (!this._active) return;\n\n const pos = point;\n\n if (this._lastPos.equals(pos) || (!this._box && pos.dist(this._startPos) < this._clickTolerance)) {\n return;\n }\n\n const p0 = this._startPos;\n this._lastPos = pos;\n\n if (!this._box) {\n this._box = DOM.create('div', 'maplibregl-boxzoom', this._container);\n this._container.classList.add('maplibregl-crosshair');\n this._fireEvent('boxzoomstart', e);\n }\n\n const minX = Math.min(p0.x, pos.x),\n maxX = Math.max(p0.x, pos.x),\n minY = Math.min(p0.y, pos.y),\n maxY = Math.max(p0.y, pos.y);\n\n DOM.setTransform(this._box, `translate(${minX}px,${minY}px)`);\n\n this._box.style.width = `${maxX - minX}px`;\n this._box.style.height = `${maxY - minY}px`;\n }\n\n mouseupWindow(e: MouseEvent, point: Point) {\n if (!this._active) return;\n\n if (e.button !== 0) return;\n\n const p0 = this._startPos,\n p1 = point;\n\n this.reset();\n\n DOM.suppressClick();\n\n if (p0.x === p1.x && p0.y === p1.y) {\n this._fireEvent('boxzoomcancel', e);\n } else {\n this._map.fire(new Event('boxzoomend', {originalEvent: e}));\n return {\n cameraAnimation: map => map.fitScreenCoordinates(p0, p1, this._tr.bearing, {linear: true})\n };\n }\n }\n\n keydown(e: KeyboardEvent) {\n if (!this._active) return;\n\n if (e.keyCode === 27) {\n this.reset();\n this._fireEvent('boxzoomcancel', e);\n }\n }\n\n reset() {\n this._active = false;\n\n this._container.classList.remove('maplibregl-crosshair');\n\n if (this._box) {\n DOM.remove(this._box);\n this._box = null;\n }\n\n DOM.enableDrag();\n\n delete this._startPos;\n delete this._lastPos;\n }\n\n _fireEvent(type: string, e: any) {\n return this._map.fire(new Event(type, {originalEvent: e}));\n }\n}\n","import Point from '@mapbox/point-geometry';\n\nexport function indexTouches(touches: Array, points: Array) {\n if (touches.length !== points.length) throw new Error(`The number of touches and points are not equal - touches ${touches.length}, points ${points.length}`);\n const obj = {};\n for (let i = 0; i < touches.length; i++) {\n obj[touches[i].identifier] = points[i];\n }\n return obj;\n}\n","import Point from '@mapbox/point-geometry';\nimport {indexTouches} from './handler_util';\n\nfunction getCentroid(points: Array) {\n const sum = new Point(0, 0);\n for (const point of points) {\n sum._add(point);\n }\n return sum.div(points.length);\n}\n\nexport const MAX_TAP_INTERVAL = 500;\nconst MAX_TOUCH_TIME = 500;\nexport const MAX_DIST = 30;\n\nexport class SingleTapRecognizer {\n\n numTouches: number;\n centroid: Point;\n startTime: number;\n aborted: boolean;\n touches: {\n [k in number | string]: Point;\n };\n\n constructor(options: {\n numTouches: number;\n }) {\n this.reset();\n this.numTouches = options.numTouches;\n }\n\n reset() {\n delete this.centroid;\n delete this.startTime;\n delete this.touches;\n this.aborted = false;\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n\n if (this.centroid || mapTouches.length > this.numTouches) {\n this.aborted = true;\n }\n if (this.aborted) {\n return;\n }\n\n if (this.startTime === undefined) {\n this.startTime = e.timeStamp;\n }\n\n if (mapTouches.length === this.numTouches) {\n this.centroid = getCentroid(points);\n this.touches = indexTouches(mapTouches, points);\n }\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n if (this.aborted || !this.centroid) return;\n\n const newTouches = indexTouches(mapTouches, points);\n for (const id in this.touches) {\n const prevPos = this.touches[id];\n const pos = newTouches[id];\n if (!pos || pos.dist(prevPos) > MAX_DIST) {\n this.aborted = true;\n }\n }\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this.centroid || e.timeStamp - this.startTime > MAX_TOUCH_TIME) {\n this.aborted = true;\n }\n\n if (mapTouches.length === 0) {\n const centroid = !this.aborted && this.centroid;\n this.reset();\n if (centroid) return centroid;\n }\n }\n\n}\n\nexport class TapRecognizer {\n\n singleTap: SingleTapRecognizer;\n numTaps: number;\n lastTime: number;\n lastTap: Point;\n count: number;\n\n constructor(options: {\n numTaps: number;\n numTouches: number;\n }) {\n this.singleTap = new SingleTapRecognizer(options);\n this.numTaps = options.numTaps;\n this.reset();\n }\n\n reset() {\n this.lastTime = Infinity;\n delete this.lastTap;\n this.count = 0;\n this.singleTap.reset();\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n this.singleTap.touchstart(e, points, mapTouches);\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n this.singleTap.touchmove(e, points, mapTouches);\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n const tap = this.singleTap.touchend(e, points, mapTouches);\n if (tap) {\n const soonEnough = e.timeStamp - this.lastTime < MAX_TAP_INTERVAL;\n const closeEnough = !this.lastTap || this.lastTap.dist(tap) < MAX_DIST;\n\n if (!soonEnough || !closeEnough) {\n this.reset();\n }\n\n this.count++;\n this.lastTime = e.timeStamp;\n this.lastTap = tap;\n\n if (this.count === this.numTaps) {\n this.reset();\n return tap;\n }\n }\n }\n}\n","import {TapRecognizer} from './tap_recognizer';\nimport type Point from '@mapbox/point-geometry';\nimport type {Map} from '../map';\nimport {TransformProvider} from './transform-provider';\nimport {Handler} from '../handler_manager';\n\nexport class TapZoomHandler implements Handler {\n _tr: TransformProvider;\n _enabled: boolean;\n _active: boolean;\n _zoomIn: TapRecognizer;\n _zoomOut: TapRecognizer;\n\n constructor(map: Map) {\n this._tr = new TransformProvider(map);\n this._zoomIn = new TapRecognizer({\n numTouches: 1,\n numTaps: 2\n });\n\n this._zoomOut = new TapRecognizer({\n numTouches: 2,\n numTaps: 1\n });\n\n this.reset();\n }\n\n reset() {\n this._active = false;\n this._zoomIn.reset();\n this._zoomOut.reset();\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n this._zoomIn.touchstart(e, points, mapTouches);\n this._zoomOut.touchstart(e, points, mapTouches);\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n this._zoomIn.touchmove(e, points, mapTouches);\n this._zoomOut.touchmove(e, points, mapTouches);\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n const zoomInPoint = this._zoomIn.touchend(e, points, mapTouches);\n const zoomOutPoint = this._zoomOut.touchend(e, points, mapTouches);\n const tr = this._tr;\n\n if (zoomInPoint) {\n this._active = true;\n e.preventDefault();\n setTimeout(() => this.reset(), 0);\n return {\n cameraAnimation: (map: Map) => map.easeTo({\n duration: 300,\n zoom: tr.zoom + 1,\n around: tr.unproject(zoomInPoint)\n }, {originalEvent: e})\n };\n } else if (zoomOutPoint) {\n this._active = true;\n e.preventDefault();\n setTimeout(() => this.reset(), 0);\n return {\n cameraAnimation: (map: Map) => map.easeTo({\n duration: 300,\n zoom: tr.zoom - 1,\n around: tr.unproject(zoomOutPoint)\n }, {originalEvent: e})\n };\n }\n }\n\n touchcancel() {\n this.reset();\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n}\n","import {DOM} from '../../util/dom';\nimport type Point from '@mapbox/point-geometry';\nimport {DragMoveStateManager} from './drag_move_state_manager';\nimport {Handler} from '../handler_manager';\n\ninterface DragMovementResult {\n bearingDelta?: number;\n pitchDelta?: number;\n around?: Point;\n panDelta?: Point;\n}\n\nexport interface DragPanResult extends DragMovementResult {\n around: Point;\n panDelta: Point;\n}\n\nexport interface DragRotateResult extends DragMovementResult {\n bearingDelta: number;\n}\n\nexport interface DragPitchResult extends DragMovementResult {\n pitchDelta: number;\n}\n\ntype DragMoveFunction = (lastPoint: Point, point: Point) => T;\n\nexport interface DragMoveHandler extends Handler {\n dragStart: (e: E, point: Point) => void;\n dragMove: (e: E, point: Point) => T | void;\n dragEnd: (e: E) => void;\n getClickTolerance: () => number;\n}\n\nexport type DragMoveHandlerOptions = {\n /**\n * If the movement is shorter than this value, consider it a click.\n */\n clickTolerance: number;\n /**\n * The move function to run on a valid movement.\n */\n move: DragMoveFunction;\n /**\n * A class used to manage the state of the drag event - start, checking valid moves, end. See the class documentation for more details.\n */\n moveStateManager: DragMoveStateManager;\n /**\n * A method used to assign the dragStart, dragMove, and dragEnd methods to the relevant event handlers, as well as assigning the contextmenu handler\n * @param handler - the handler\n */\n assignEvents: (handler: DragMoveHandler) => void;\n /**\n * Should the move start on the \"start\" event, or should it start on the first valid move.\n */\n activateOnStart?: boolean;\n /**\n * If true, handler will be enabled during construction\n */\n enable?: boolean;\n}\n\n/**\n * A generic class to create handlers for drag events, from both mouse and touch events.\n */\nexport class DragHandler implements DragMoveHandler {\n // Event handlers that may be assigned by the implementations of this class\n contextmenu?: Handler['contextmenu'];\n mousedown?: Handler['mousedown'];\n mousemoveWindow?: Handler['mousemoveWindow'];\n mouseup?: Handler['mouseup'];\n touchstart?: Handler['touchstart'];\n touchmoveWindow?: Handler['touchmoveWindow'];\n touchend?: Handler['touchend'];\n\n _clickTolerance: number;\n _moveFunction: DragMoveFunction;\n _activateOnStart: boolean;\n _active: boolean;\n _enabled: boolean;\n _moved: boolean;\n _lastPoint: Point | null;\n _moveStateManager: DragMoveStateManager;\n\n constructor(options: DragMoveHandlerOptions) {\n this._enabled = !!options.enable;\n this._moveStateManager = options.moveStateManager;\n this._clickTolerance = options.clickTolerance || 1;\n this._moveFunction = options.move;\n this._activateOnStart = !!options.activateOnStart;\n\n options.assignEvents(this);\n\n this.reset();\n }\n\n reset(e?: E) {\n this._active = false;\n this._moved = false;\n delete this._lastPoint;\n this._moveStateManager.endMove(e);\n }\n\n _move(...params: Parameters>) {\n const move = this._moveFunction(...params);\n if (move.bearingDelta || move.pitchDelta || move.around || move.panDelta) {\n this._active = true;\n return move;\n }\n }\n\n dragStart(e: E, point: Point);\n dragStart(e: E, point: Point[]);\n dragStart(e: E, point: Point | Point[]) {\n if (!this.isEnabled() || this._lastPoint) return;\n\n if (!this._moveStateManager.isValidStartEvent(e)) return;\n this._moveStateManager.startMove(e);\n\n this._lastPoint = point['length'] ? point[0] : point;\n\n if (this._activateOnStart && this._lastPoint) this._active = true;\n }\n\n dragMove(e: E, point: Point);\n dragMove(e: E, point: Point[]);\n dragMove(e: E, point: Point | Point[]) {\n if (!this.isEnabled()) return;\n const lastPoint = this._lastPoint;\n if (!lastPoint) return;\n e.preventDefault();\n\n if (!this._moveStateManager.isValidMoveEvent(e)) {\n this.reset(e);\n return;\n }\n\n const movePoint = point['length'] ? point[0] : point;\n\n if (!this._moved && movePoint.dist(lastPoint) < this._clickTolerance) return;\n this._moved = true;\n this._lastPoint = movePoint;\n\n return this._move(lastPoint, movePoint);\n }\n\n dragEnd(e: E) {\n if (!this.isEnabled() || !this._lastPoint) return;\n if (!this._moveStateManager.isValidEndEvent(e)) return;\n if (this._moved) DOM.suppressClick();\n this.reset(e);\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n\n getClickTolerance() {\n return this._clickTolerance;\n }\n}\n","import {DOM} from '../../util/dom';\n\nconst LEFT_BUTTON = 0;\nconst RIGHT_BUTTON = 2;\n\n// the values for each button in MouseEvent.buttons\nconst BUTTONS_FLAGS = {\n [LEFT_BUTTON]: 1,\n [RIGHT_BUTTON]: 2\n};\n\nfunction buttonNoLongerPressed(e: MouseEvent, button: number) {\n const flag = BUTTONS_FLAGS[button];\n return e.buttons === undefined || (e.buttons & flag) !== flag;\n}\n\n/*\n * Drag events are initiated by specific interaction which needs to be tracked until it ends.\n * This requires some state management:\n * 1. registering the initiating event,\n * 2. tracking that it was not canceled / not confusing it with another event firing.\n * 3. recognizing the ending event and cleaning up any internal state\n *\n * Concretely, we implement two state managers:\n * 1. MouseMoveStateManager\n * Receives a functions that is used to recognize mouse events that should be registered as the\n * relevant drag interactions - i.e. dragging with the right mouse button, or while CTRL is pressed.\n * 2. OneFingerTouchMoveStateManager\n * Checks if a drag event is using one finger, and continuously tracking that this is the same event\n * (i.e. to make sure not additional finger has started interacting with the screen before raising\n * the first finger).\n */\nexport interface DragMoveStateManager {\n startMove: (e: E) => void;\n endMove: (e?: E) => void;\n isValidStartEvent: (e: E) => boolean;\n isValidMoveEvent: (e: E) => boolean;\n isValidEndEvent: (e?: E) => boolean;\n}\n\nexport class MouseMoveStateManager implements DragMoveStateManager {\n _eventButton: number | undefined;\n _correctEvent: (e: MouseEvent) => boolean;\n\n constructor(options: {\n checkCorrectEvent: (e: MouseEvent) => boolean;\n }) {\n this._correctEvent = options.checkCorrectEvent;\n }\n\n startMove(e: MouseEvent) {\n const eventButton = DOM.mouseButton(e);\n this._eventButton = eventButton;\n }\n\n endMove(_e?: MouseEvent) {\n delete this._eventButton;\n }\n\n isValidStartEvent(e: MouseEvent) {\n return this._correctEvent(e);\n }\n\n isValidMoveEvent(e: MouseEvent) {\n // Some browsers don't fire a `mouseup` when the mouseup occurs outside\n // the window or iframe:\n // https://github.com/mapbox/mapbox-gl-js/issues/4622\n //\n // If the button is no longer pressed during this `mousemove` it may have\n // been released outside of the window or iframe.\n return !buttonNoLongerPressed(e, this._eventButton);\n }\n\n isValidEndEvent(e: MouseEvent) {\n const eventButton = DOM.mouseButton(e);\n return eventButton === this._eventButton;\n }\n}\n\nexport class OneFingerTouchMoveStateManager implements DragMoveStateManager {\n _firstTouch: number | undefined;\n\n constructor() {\n this._firstTouch = undefined;\n }\n\n _isOneFingerTouch(e: TouchEvent) {\n return e.targetTouches.length === 1;\n }\n\n _isSameTouchEvent(e: TouchEvent) {\n return e.targetTouches[0].identifier === this._firstTouch;\n }\n\n startMove(e: TouchEvent) {\n const firstTouch = e.targetTouches[0].identifier;\n this._firstTouch = firstTouch;\n }\n\n endMove(_e?: TouchEvent) {\n delete this._firstTouch;\n }\n\n isValidStartEvent(e: TouchEvent) {\n return this._isOneFingerTouch(e);\n }\n\n isValidMoveEvent(e: TouchEvent) {\n return this._isOneFingerTouch(e) && this._isSameTouchEvent(e);\n }\n\n isValidEndEvent(e: TouchEvent) {\n return this._isOneFingerTouch(e) && this._isSameTouchEvent(e);\n }\n}\n","import type Point from '@mapbox/point-geometry';\n\nimport {DOM} from '../../util/dom';\nimport {DragMoveHandler, DragPanResult, DragRotateResult, DragPitchResult, DragHandler} from './drag_handler';\nimport {MouseMoveStateManager} from './drag_move_state_manager';\n\nexport interface MousePanHandler extends DragMoveHandler {}\nexport interface MouseRotateHandler extends DragMoveHandler {}\nexport interface MousePitchHandler extends DragMoveHandler {}\n\nconst LEFT_BUTTON = 0;\nconst RIGHT_BUTTON = 2;\n\nconst assignEvents = (handler: DragHandler) => {\n handler.mousedown = handler.dragStart;\n handler.mousemoveWindow = handler.dragMove;\n handler.mouseup = handler.dragEnd;\n handler.contextmenu = function(e: MouseEvent) {\n e.preventDefault();\n };\n};\n\nexport const generateMousePanHandler = ({enable, clickTolerance,}: {\n clickTolerance: number;\n enable?: boolean;\n}): MousePanHandler => {\n const mouseMoveStateManager = new MouseMoveStateManager({\n checkCorrectEvent: (e: MouseEvent) => DOM.mouseButton(e) === LEFT_BUTTON && !e.ctrlKey,\n });\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({around: point, panDelta: point.sub(lastPoint)}),\n activateOnStart: true,\n moveStateManager: mouseMoveStateManager,\n enable,\n assignEvents,\n });\n};\n\nexport const generateMouseRotationHandler = ({enable, clickTolerance, bearingDegreesPerPixelMoved = 0.8}: {\n clickTolerance: number;\n bearingDegreesPerPixelMoved?: number;\n enable?: boolean;\n}): MouseRotateHandler => {\n const mouseMoveStateManager = new MouseMoveStateManager({\n checkCorrectEvent: (e: MouseEvent): boolean =>\n (DOM.mouseButton(e) === LEFT_BUTTON && e.ctrlKey) ||\n (DOM.mouseButton(e) === RIGHT_BUTTON),\n });\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({bearingDelta: (point.x - lastPoint.x) * bearingDegreesPerPixelMoved}),\n // prevent browser context menu when necessary; we don't allow it with rotation\n // because we can't discern rotation gesture start from contextmenu on Mac\n moveStateManager: mouseMoveStateManager,\n enable,\n assignEvents,\n });\n};\n\nexport const generateMousePitchHandler = ({enable, clickTolerance, pitchDegreesPerPixelMoved = -0.5}: {\n clickTolerance: number;\n pitchDegreesPerPixelMoved?: number;\n enable?: boolean;\n}): MousePitchHandler => {\n const mouseMoveStateManager = new MouseMoveStateManager({\n checkCorrectEvent: (e: MouseEvent): boolean =>\n (DOM.mouseButton(e) === LEFT_BUTTON && e.ctrlKey) ||\n (DOM.mouseButton(e) === RIGHT_BUTTON),\n });\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({pitchDelta: (point.y - lastPoint.y) * pitchDegreesPerPixelMoved}),\n // prevent browser context menu when necessary; we don't allow it with rotation\n // because we can't discern rotation gesture start from contextmenu on Mac\n moveStateManager: mouseMoveStateManager,\n enable,\n assignEvents,\n });\n};\n","import Point from '@mapbox/point-geometry';\nimport {indexTouches} from './handler_util';\nimport type {Map} from '../map';\nimport {GestureOptions} from '../map';\nimport {Handler} from '../handler_manager';\n\nexport class TouchPanHandler implements Handler {\n\n _enabled: boolean;\n _active: boolean;\n _touches: {\n [k in string | number]: Point;\n };\n _minTouches: number;\n _clickTolerance: number;\n _sum: Point;\n _map: Map;\n _cancelCooperativeMessage: boolean;\n\n constructor(options: {\n clickTolerance: number;\n cooperativeGestures: boolean | GestureOptions;\n }, map: Map) {\n this._minTouches = options.cooperativeGestures ? 2 : 1;\n this._clickTolerance = options.clickTolerance || 1;\n this._map = map;\n this.reset();\n }\n\n reset() {\n this._active = false;\n this._touches = {};\n this._sum = new Point(0, 0);\n\n // Put a delay on the cooperative gesture message so it's less twitchy\n setTimeout(() => {\n this._cancelCooperativeMessage = false;\n }, 200);\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n return this._calculateTransform(e, points, mapTouches);\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n if (this._map._cooperativeGestures) {\n if (this._minTouches === 2 && mapTouches.length < 2 && !this._cancelCooperativeMessage) {\n // If coop gesture enabled, show panning info to user\n this._map._onCooperativeGesture(e, false, mapTouches.length);\n } else if (!this._cancelCooperativeMessage) {\n // If user is successfully navigating, we don't need this warning until the touch resets\n this._cancelCooperativeMessage = true;\n }\n }\n if (!this._active || mapTouches.length < this._minTouches) return;\n e.preventDefault();\n return this._calculateTransform(e, points, mapTouches);\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n this._calculateTransform(e, points, mapTouches);\n\n if (this._active && mapTouches.length < this._minTouches) {\n this.reset();\n }\n }\n\n touchcancel() {\n this.reset();\n }\n\n _calculateTransform(e: TouchEvent, points: Array, mapTouches: Array) {\n if (mapTouches.length > 0) this._active = true;\n\n const touches = indexTouches(mapTouches, points);\n\n const touchPointSum = new Point(0, 0);\n const touchDeltaSum = new Point(0, 0);\n let touchDeltaCount = 0;\n\n for (const identifier in touches) {\n const point = touches[identifier];\n const prevPoint = this._touches[identifier];\n if (prevPoint) {\n touchPointSum._add(point);\n touchDeltaSum._add(point.sub(prevPoint));\n touchDeltaCount++;\n touches[identifier] = point;\n }\n }\n\n this._touches = touches;\n\n if (touchDeltaCount < this._minTouches || !touchDeltaSum.mag()) return;\n\n const panDelta = touchDeltaSum.div(touchDeltaCount);\n this._sum._add(panDelta);\n if (this._sum.mag() < this._clickTolerance) return;\n\n const around = touchPointSum.div(touchDeltaCount);\n\n return {\n around,\n panDelta\n };\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n}\n","import Point from '@mapbox/point-geometry';\nimport {DOM} from '../../util/dom';\nimport type {Map} from '../map';\nimport {Handler} from '../handler_manager';\n\n/**\n * An options object sent to the enable function of some of the handlers\n */\nexport type AroundCenterOptions = {\n /**\n * If \"center\" is passed, map will zoom around the center of map\n */\n around: 'center';\n}\n\n/**\n * The `TwoFingersTouchHandler`s allows the user to zoom, pitch and rotate the map using two fingers\n *\n * @group Handlers\n */\nabstract class TwoFingersTouchHandler implements Handler {\n\n _enabled: boolean;\n _active: boolean;\n _firstTwoTouches: [number, number];\n _vector: Point;\n _startVector: Point;\n _aroundCenter: boolean;\n\n /** @internal */\n constructor() {\n this.reset();\n }\n\n reset() {\n this._active = false;\n delete this._firstTwoTouches;\n }\n\n abstract _start(points: [Point, Point]);\n abstract _move(points: [Point, Point], pinchAround: Point, e: TouchEvent);\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n //log('touchstart', points, e.target.innerHTML, e.targetTouches.length ? e.targetTouches[0].target.innerHTML: undefined);\n if (this._firstTwoTouches || mapTouches.length < 2) return;\n\n this._firstTwoTouches = [\n mapTouches[0].identifier,\n mapTouches[1].identifier\n ];\n\n // implemented by child classes\n this._start([points[0], points[1]]);\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this._firstTwoTouches) return;\n\n e.preventDefault();\n\n const [idA, idB] = this._firstTwoTouches;\n const a = getTouchById(mapTouches, points, idA);\n const b = getTouchById(mapTouches, points, idB);\n if (!a || !b) return;\n const pinchAround = this._aroundCenter ? null : a.add(b).div(2);\n\n // implemented by child classes\n return this._move([a, b], pinchAround, e);\n\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this._firstTwoTouches) return;\n\n const [idA, idB] = this._firstTwoTouches;\n const a = getTouchById(mapTouches, points, idA);\n const b = getTouchById(mapTouches, points, idB);\n if (a && b) return;\n\n if (this._active) DOM.suppressClick();\n\n this.reset();\n }\n\n touchcancel() {\n this.reset();\n }\n\n /**\n * Enables the \"drag to pitch\" interaction.\n *\n * @example\n * ```ts\n * map.touchPitch.enable();\n * ```\n */\n enable(options?: AroundCenterOptions | boolean | null) {\n this._enabled = true;\n this._aroundCenter = !!options && (options as AroundCenterOptions).around === 'center';\n }\n\n /**\n * Disables the \"drag to pitch\" interaction.\n *\n * @example\n * ```ts\n * map.touchPitch.disable();\n * ```\n */\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to pitch\" interaction is enabled.\n *\n * @returns `true` if the \"drag to pitch\" interaction is enabled.\n */\n isEnabled() {\n return this._enabled;\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to pitch\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"drag to pitch\" interaction is active.\n */\n isActive() {\n return this._active;\n }\n}\n\nfunction getTouchById(mapTouches: Array, points: Array, identifier: number) {\n for (let i = 0; i < mapTouches.length; i++) {\n if (mapTouches[i].identifier === identifier) return points[i];\n }\n}\n\n/* ZOOM */\n\nconst ZOOM_THRESHOLD = 0.1;\n\nfunction getZoomDelta(distance, lastDistance) {\n return Math.log(distance / lastDistance) / Math.LN2;\n}\n\n/**\n * The `TwoFingersTouchHandler`s allows the user to zoom the map two fingers\n *\n * @group Handlers\n */\nexport class TwoFingersTouchZoomHandler extends TwoFingersTouchHandler {\n\n _distance: number;\n _startDistance: number;\n\n reset() {\n super.reset();\n delete this._distance;\n delete this._startDistance;\n }\n\n _start(points: [Point, Point]) {\n this._startDistance = this._distance = points[0].dist(points[1]);\n }\n\n _move(points: [Point, Point], pinchAround: Point) {\n const lastDistance = this._distance;\n this._distance = points[0].dist(points[1]);\n if (!this._active && Math.abs(getZoomDelta(this._distance, this._startDistance)) < ZOOM_THRESHOLD) return;\n this._active = true;\n return {\n zoomDelta: getZoomDelta(this._distance, lastDistance),\n pinchAround\n };\n }\n}\n\n/* ROTATE */\n\nconst ROTATION_THRESHOLD = 25; // pixels along circumference of touch circle\n\nfunction getBearingDelta(a, b) {\n return a.angleWith(b) * 180 / Math.PI;\n}\n\n/**\n * The `TwoFingersTouchHandler`s allows the user to rotate the map two fingers\n *\n * @group Handlers\n */\nexport class TwoFingersTouchRotateHandler extends TwoFingersTouchHandler {\n _minDiameter: number;\n\n reset() {\n super.reset();\n delete this._minDiameter;\n delete this._startVector;\n delete this._vector;\n }\n\n _start(points: [Point, Point]) {\n this._startVector = this._vector = points[0].sub(points[1]);\n this._minDiameter = points[0].dist(points[1]);\n }\n\n _move(points: [Point, Point], pinchAround: Point) {\n const lastVector = this._vector;\n this._vector = points[0].sub(points[1]);\n\n if (!this._active && this._isBelowThreshold(this._vector)) return;\n this._active = true;\n\n return {\n bearingDelta: getBearingDelta(this._vector, lastVector),\n pinchAround\n };\n }\n\n _isBelowThreshold(vector: Point) {\n /*\n * The threshold before a rotation actually happens is configured in\n * pixels along the circumference of the circle formed by the two fingers.\n * This makes the threshold in degrees larger when the fingers are close\n * together and smaller when the fingers are far apart.\n *\n * Use the smallest diameter from the whole gesture to reduce sensitivity\n * when pinching in and out.\n */\n\n this._minDiameter = Math.min(this._minDiameter, vector.mag());\n const circumference = Math.PI * this._minDiameter;\n const threshold = ROTATION_THRESHOLD / circumference * 360;\n\n const bearingDeltaSinceStart = getBearingDelta(vector, this._startVector);\n return Math.abs(bearingDeltaSinceStart) < threshold;\n }\n}\n\n/* PITCH */\n\nfunction isVertical(vector) {\n return Math.abs(vector.y) > Math.abs(vector.x);\n}\n\nconst ALLOWED_SINGLE_TOUCH_TIME = 100;\n\n/**\n * The `TwoFingersTouchPitchHandler` allows the user to pitch the map by dragging up and down with two fingers.\n *\n * @group Handlers\n */\nexport class TwoFingersTouchPitchHandler extends TwoFingersTouchHandler {\n\n _valid: boolean | void;\n _firstMove: number;\n _lastPoints: [Point, Point];\n _map: Map;\n _currentTouchCount: number;\n\n constructor(map: Map) {\n super();\n this._map = map;\n }\n\n reset() {\n super.reset();\n this._valid = undefined;\n delete this._firstMove;\n delete this._lastPoints;\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n super.touchstart(e, points, mapTouches);\n this._currentTouchCount = mapTouches.length;\n }\n\n _start(points: [Point, Point]) {\n this._lastPoints = points;\n if (isVertical(points[0].sub(points[1]))) {\n // fingers are more horizontal than vertical\n this._valid = false;\n\n }\n }\n\n _move(points: [Point, Point], center: Point, e: TouchEvent) {\n // If cooperative gestures is enabled, we need a 3-finger minimum for this gesture to register\n if (this._map._cooperativeGestures && this._currentTouchCount < 3) {\n return;\n }\n\n const vectorA = points[0].sub(this._lastPoints[0]);\n const vectorB = points[1].sub(this._lastPoints[1]);\n\n this._valid = this.gestureBeginsVertically(vectorA, vectorB, e.timeStamp);\n if (!this._valid) return;\n\n this._lastPoints = points;\n this._active = true;\n const yDeltaAverage = (vectorA.y + vectorB.y) / 2;\n const degreesPerPixelMoved = -0.5;\n return {\n pitchDelta: yDeltaAverage * degreesPerPixelMoved\n };\n }\n\n gestureBeginsVertically(vectorA: Point, vectorB: Point, timeStamp: number) {\n if (this._valid !== undefined) return this._valid;\n\n const threshold = 2;\n const movedA = vectorA.mag() >= threshold;\n const movedB = vectorB.mag() >= threshold;\n\n // neither finger has moved a meaningful amount, wait\n if (!movedA && !movedB) return;\n\n // One finger has moved and the other has not.\n // If enough time has passed, decide it is not a pitch.\n if (!movedA || !movedB) {\n if (this._firstMove === undefined) {\n this._firstMove = timeStamp;\n }\n\n if (timeStamp - this._firstMove < ALLOWED_SINGLE_TOUCH_TIME) {\n // still waiting for a movement from the second finger\n return undefined;\n } else {\n return false;\n }\n }\n\n const isSameDirection = vectorA.y > 0 === vectorB.y > 0;\n return isVertical(vectorA) && isVertical(vectorB) && isSameDirection;\n }\n}\n","import {Handler} from '../handler_manager';\nimport type {Map} from '../map';\nimport {TransformProvider} from './transform-provider';\n\nconst defaultOptions = {\n panStep: 100,\n bearingStep: 15,\n pitchStep: 10\n};\n\n/**\n * The `KeyboardHandler` allows the user to zoom, rotate, and pan the map using\n * the following keyboard shortcuts:\n *\n * - `=` / `+`: Increase the zoom level by 1.\n * - `Shift-=` / `Shift-+`: Increase the zoom level by 2.\n * - `-`: Decrease the zoom level by 1.\n * - `Shift--`: Decrease the zoom level by 2.\n * - Arrow keys: Pan by 100 pixels.\n * - `Shift+⇢`: Increase the rotation by 15 degrees.\n * - `Shift+⇠`: Decrease the rotation by 15 degrees.\n * - `Shift+⇡`: Increase the pitch by 10 degrees.\n * - `Shift+⇣`: Decrease the pitch by 10 degrees.\n *\n * @group Handlers\n */\nexport class KeyboardHandler implements Handler {\n _tr: TransformProvider;\n _enabled: boolean;\n _active: boolean;\n _panStep: number;\n _bearingStep: number;\n _pitchStep: number;\n _rotationDisabled: boolean;\n\n /** @internal */\n constructor(map: Map) {\n this._tr = new TransformProvider(map);\n const stepOptions = defaultOptions;\n this._panStep = stepOptions.panStep;\n this._bearingStep = stepOptions.bearingStep;\n this._pitchStep = stepOptions.pitchStep;\n this._rotationDisabled = false;\n }\n\n reset() {\n this._active = false;\n }\n\n keydown(e: KeyboardEvent) {\n if (e.altKey || e.ctrlKey || e.metaKey) return;\n\n let zoomDir = 0;\n let bearingDir = 0;\n let pitchDir = 0;\n let xDir = 0;\n let yDir = 0;\n\n switch (e.keyCode) {\n case 61:\n case 107:\n case 171:\n case 187:\n zoomDir = 1;\n break;\n\n case 189:\n case 109:\n case 173:\n zoomDir = -1;\n break;\n\n case 37:\n if (e.shiftKey) {\n bearingDir = -1;\n } else {\n e.preventDefault();\n xDir = -1;\n }\n break;\n\n case 39:\n if (e.shiftKey) {\n bearingDir = 1;\n } else {\n e.preventDefault();\n xDir = 1;\n }\n break;\n\n case 38:\n if (e.shiftKey) {\n pitchDir = 1;\n } else {\n e.preventDefault();\n yDir = -1;\n }\n break;\n\n case 40:\n if (e.shiftKey) {\n pitchDir = -1;\n } else {\n e.preventDefault();\n yDir = 1;\n }\n break;\n\n default:\n return;\n }\n\n if (this._rotationDisabled) {\n bearingDir = 0;\n pitchDir = 0;\n }\n\n return {\n cameraAnimation: (map: Map) => {\n const tr = this._tr;\n map.easeTo({\n duration: 300,\n easeId: 'keyboardHandler',\n easing: easeOut,\n\n zoom: zoomDir ? Math.round(tr.zoom) + zoomDir * (e.shiftKey ? 2 : 1) : tr.zoom,\n bearing: tr.bearing + bearingDir * this._bearingStep,\n pitch: tr.pitch + pitchDir * this._pitchStep,\n offset: [-xDir * this._panStep, -yDir * this._panStep],\n center: tr.center\n }, {originalEvent: e});\n }\n };\n }\n\n /**\n * Enables the \"keyboard rotate and zoom\" interaction.\n *\n * @example\n * ```ts\n * map.keyboard.enable();\n * ```\n */\n enable() {\n this._enabled = true;\n }\n\n /**\n * Disables the \"keyboard rotate and zoom\" interaction.\n *\n * @example\n * ```ts\n * map.keyboard.disable();\n * ```\n */\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n /**\n * Returns a Boolean indicating whether the \"keyboard rotate and zoom\"\n * interaction is enabled.\n *\n * @returns `true` if the \"keyboard rotate and zoom\"\n * interaction is enabled.\n */\n isEnabled() {\n return this._enabled;\n }\n\n /**\n * Returns true if the handler is enabled and has detected the start of a\n * zoom/rotate gesture.\n *\n * @returns `true` if the handler is enabled and has detected the\n * start of a zoom/rotate gesture.\n */\n isActive() {\n return this._active;\n }\n\n /**\n * Disables the \"keyboard pan/rotate\" interaction, leaving the\n * \"keyboard zoom\" interaction enabled.\n *\n * @example\n * ```ts\n * map.keyboard.disableRotation();\n * ```\n */\n disableRotation() {\n this._rotationDisabled = true;\n }\n\n /**\n * Enables the \"keyboard pan/rotate\" interaction.\n *\n * @example\n * ```ts\n * map.keyboard.enable();\n * map.keyboard.enableRotation();\n * ```\n */\n enableRotation() {\n this._rotationDisabled = false;\n }\n}\n\nfunction easeOut(t: number) {\n return t * (2 - t);\n}\n","import {DOM} from '../../util/dom';\n\nimport {defaultEasing, bezier} from '../../util/util';\nimport {browser} from '../../util/browser';\nimport {interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {LngLat} from '../../geo/lng_lat';\nimport {TransformProvider} from './transform-provider';\n\nimport type {Map} from '../map';\nimport type Point from '@mapbox/point-geometry';\nimport type {AroundCenterOptions} from './two_fingers_touch';\nimport {Handler} from '../handler_manager';\n\n// deltaY value for mouse scroll wheel identification\nconst wheelZoomDelta = 4.000244140625;\n\n// These magic numbers control the rate of zoom. Trackpad events fire at a greater\n// frequency than mouse scroll wheel, so reduce the zoom rate per wheel tick\nconst defaultZoomRate = 1 / 100;\nconst wheelZoomRate = 1 / 450;\n\n// upper bound on how much we scale the map in any single render frame; this\n// is used to limit zoom rate in the case of very fast scrolling\nconst maxScalePerFrame = 2;\n\n/**\n * The `ScrollZoomHandler` allows the user to zoom the map by scrolling.\n *\n * @group Handlers\n */\nexport class ScrollZoomHandler implements Handler {\n _map: Map;\n _tr: TransformProvider;\n _el: HTMLElement;\n _enabled: boolean;\n _active: boolean;\n _zooming: boolean;\n _aroundCenter: boolean;\n _around: LngLat;\n _aroundPoint: Point;\n _type: 'wheel' | 'trackpad' | null;\n _lastValue: number;\n _timeout: ReturnType; // used for delayed-handling of a single wheel movement\n _finishTimeout: ReturnType; // used to delay final '{move,zoom}end' events\n\n _lastWheelEvent: any;\n _lastWheelEventTime: number;\n\n _startZoom: number;\n _targetZoom: number;\n _delta: number;\n _easing: ((a: number) => number);\n _prevEase: {\n start: number;\n duration: number;\n easing: (_: number) => number;\n };\n\n _frameId: boolean;\n _triggerRenderFrame: () => void;\n\n _defaultZoomRate: number;\n _wheelZoomRate: number;\n\n /** @internal */\n constructor(map: Map, triggerRenderFrame: () => void) {\n this._map = map;\n this._tr = new TransformProvider(map);\n this._el = map.getCanvasContainer();\n this._triggerRenderFrame = triggerRenderFrame;\n\n this._delta = 0;\n\n this._defaultZoomRate = defaultZoomRate;\n this._wheelZoomRate = wheelZoomRate;\n }\n\n /**\n * Set the zoom rate of a trackpad\n * @param zoomRate - 1/100 The rate used to scale trackpad movement to a zoom value.\n * @example\n * Speed up trackpad zoom\n * ```ts\n * map.scrollZoom.setZoomRate(1/25);\n * ```\n */\n setZoomRate(zoomRate: number) {\n this._defaultZoomRate = zoomRate;\n }\n\n /**\n * Set the zoom rate of a mouse wheel\n * @param wheelZoomRate - 1/450 The rate used to scale mouse wheel movement to a zoom value.\n * @example\n * Slow down zoom of mouse wheel\n * ```ts\n * map.scrollZoom.setWheelZoomRate(1/600);\n * ```\n */\n setWheelZoomRate(wheelZoomRate: number) {\n this._wheelZoomRate = wheelZoomRate;\n }\n\n /**\n * Returns a Boolean indicating whether the \"scroll to zoom\" interaction is enabled.\n * @returns `true` if the \"scroll to zoom\" interaction is enabled.\n */\n isEnabled() {\n return !!this._enabled;\n }\n\n /*\n * Active state is turned on and off with every scroll wheel event and is set back to false before the map\n * render is called, so _active is not a good candidate for determining if a scroll zoom animation is in\n * progress.\n */\n isActive() {\n return !!this._active || this._finishTimeout !== undefined;\n }\n\n isZooming() {\n return !!this._zooming;\n }\n\n /**\n * Enables the \"scroll to zoom\" interaction.\n *\n * @param options - Options object.\n * @example\n * ```ts\n * map.scrollZoom.enable();\n * map.scrollZoom.enable({ around: 'center' })\n * ```\n */\n enable(options?: AroundCenterOptions | boolean) {\n if (this.isEnabled()) return;\n this._enabled = true;\n this._aroundCenter = !!options && (options as AroundCenterOptions).around === 'center';\n }\n\n /**\n * Disables the \"scroll to zoom\" interaction.\n *\n * @example\n * ```ts\n * map.scrollZoom.disable();\n * ```\n */\n disable() {\n if (!this.isEnabled()) return;\n this._enabled = false;\n }\n\n wheel(e: WheelEvent) {\n if (!this.isEnabled()) return;\n if (this._map._cooperativeGestures) {\n if (e[this._map._metaKey]) {\n e.preventDefault();\n } else {\n return;\n }\n }\n let value = e.deltaMode === WheelEvent.DOM_DELTA_LINE ? e.deltaY * 40 : e.deltaY;\n const now = browser.now(),\n timeDelta = now - (this._lastWheelEventTime || 0);\n\n this._lastWheelEventTime = now;\n\n if (value !== 0 && (value % wheelZoomDelta) === 0) {\n // This one is definitely a mouse wheel event.\n this._type = 'wheel';\n\n } else if (value !== 0 && Math.abs(value) < 4) {\n // This one is definitely a trackpad event because it is so small.\n this._type = 'trackpad';\n\n } else if (timeDelta > 400) {\n // This is likely a new scroll action.\n this._type = null;\n this._lastValue = value;\n\n // Start a timeout in case this was a singular event, and dely it by up to 40ms.\n this._timeout = setTimeout(this._onTimeout, 40, e);\n\n } else if (!this._type) {\n // This is a repeating event, but we don't know the type of event just yet.\n // If the delta per time is small, we assume it's a fast trackpad; otherwise we switch into wheel mode.\n this._type = (Math.abs(timeDelta * value) < 200) ? 'trackpad' : 'wheel';\n\n // Make sure our delayed event isn't fired again, because we accumulate\n // the previous event (which was less than 40ms ago) into this event.\n if (this._timeout) {\n clearTimeout(this._timeout);\n this._timeout = null;\n value += this._lastValue;\n }\n }\n\n // Slow down zoom if shift key is held for more precise zooming\n if (e.shiftKey && value) value = value / 4;\n\n // Only fire the callback if we actually know what type of scrolling device the user uses.\n if (this._type) {\n this._lastWheelEvent = e;\n this._delta -= value;\n if (!this._active) {\n this._start(e);\n }\n }\n\n e.preventDefault();\n }\n\n _onTimeout = (initialEvent: MouseEvent) => {\n this._type = 'wheel';\n this._delta -= this._lastValue;\n if (!this._active) {\n this._start(initialEvent);\n }\n };\n\n _start(e: MouseEvent) {\n if (!this._delta) return;\n\n if (this._frameId) {\n this._frameId = null;\n }\n\n this._active = true;\n if (!this.isZooming()) {\n this._zooming = true;\n }\n\n if (this._finishTimeout) {\n clearTimeout(this._finishTimeout);\n delete this._finishTimeout;\n }\n\n const pos = DOM.mousePos(this._el, e);\n const tr = this._tr;\n\n this._around = LngLat.convert(this._aroundCenter ? tr.center : tr.unproject(pos));\n this._aroundPoint = tr.transform.locationPoint(this._around);\n if (!this._frameId) {\n this._frameId = true;\n this._triggerRenderFrame();\n }\n }\n\n renderFrame() {\n if (!this._frameId) return;\n this._frameId = null;\n\n if (!this.isActive()) return;\n const tr = this._tr.transform;\n\n // if we've had scroll events since the last render frame, consume the\n // accumulated delta, and update the target zoom level accordingly\n if (this._delta !== 0) {\n // For trackpad events and single mouse wheel ticks, use the default zoom rate\n const zoomRate = (this._type === 'wheel' && Math.abs(this._delta) > wheelZoomDelta) ? this._wheelZoomRate : this._defaultZoomRate;\n // Scale by sigmoid of scroll wheel delta.\n let scale = maxScalePerFrame / (1 + Math.exp(-Math.abs(this._delta * zoomRate)));\n\n if (this._delta < 0 && scale !== 0) {\n scale = 1 / scale;\n }\n\n const fromScale = typeof this._targetZoom === 'number' ? tr.zoomScale(this._targetZoom) : tr.scale;\n this._targetZoom = Math.min(tr.maxZoom, Math.max(tr.minZoom, tr.scaleZoom(fromScale * scale)));\n\n // if this is a mouse wheel, refresh the starting zoom and easing\n // function we're using to smooth out the zooming between wheel\n // events\n if (this._type === 'wheel') {\n this._startZoom = tr.zoom;\n this._easing = this._smoothOutEasing(200);\n }\n\n this._delta = 0;\n }\n\n const targetZoom = typeof this._targetZoom === 'number' ?\n this._targetZoom : tr.zoom;\n const startZoom = this._startZoom;\n const easing = this._easing;\n\n let finished = false;\n let zoom;\n if (this._type === 'wheel' && startZoom && easing) {\n\n const t = Math.min((browser.now() - this._lastWheelEventTime) / 200, 1);\n const k = easing(t);\n zoom = interpolates.number(startZoom, targetZoom, k);\n if (t < 1) {\n if (!this._frameId) {\n this._frameId = true;\n }\n } else {\n finished = true;\n }\n } else {\n zoom = targetZoom;\n finished = true;\n }\n\n this._active = true;\n\n if (finished) {\n this._active = false;\n this._finishTimeout = setTimeout(() => {\n this._zooming = false;\n this._triggerRenderFrame();\n delete this._targetZoom;\n delete this._finishTimeout;\n }, 200);\n }\n\n return {\n noInertia: true,\n needsRenderFrame: !finished,\n zoomDelta: zoom - tr.zoom,\n around: this._aroundPoint,\n originalEvent: this._lastWheelEvent\n };\n }\n\n _smoothOutEasing(duration: number) {\n let easing = defaultEasing;\n\n if (this._prevEase) {\n const currentEase = this._prevEase;\n const t = (browser.now() - currentEase.start) / currentEase.duration;\n const speed = currentEase.easing(t + 0.01) - currentEase.easing(t);\n\n // Quick hack to make new bezier that is continuous with last\n const x = 0.27 / Math.sqrt(speed * speed + 0.0001) * 0.01;\n const y = Math.sqrt(0.27 * 0.27 - x * x);\n\n easing = bezier(x, y, 0.25, 1);\n }\n\n this._prevEase = {\n start: browser.now(),\n duration,\n easing\n };\n\n return easing;\n }\n\n reset() {\n this._active = false;\n this._zooming = false;\n delete this._targetZoom;\n if (this._finishTimeout) {\n clearTimeout(this._finishTimeout);\n delete this._finishTimeout;\n }\n }\n}\n","import type {ClickZoomHandler} from '../click_zoom';\nimport type {TapZoomHandler} from './../tap_zoom';\n\n/**\n * The `DoubleClickZoomHandler` allows the user to zoom the map at a point by\n * double clicking or double tapping.\n *\n * @group Handlers\n */\nexport class DoubleClickZoomHandler {\n\n _clickZoom: ClickZoomHandler;\n _tapZoom: TapZoomHandler;\n\n /** @internal */\n constructor(clickZoom: ClickZoomHandler, TapZoom: TapZoomHandler) {\n this._clickZoom = clickZoom;\n this._tapZoom = TapZoom;\n }\n\n /**\n * Enables the \"double click to zoom\" interaction.\n *\n * @example\n * ```ts\n * map.doubleClickZoom.enable();\n * ```\n */\n enable() {\n this._clickZoom.enable();\n this._tapZoom.enable();\n }\n\n /**\n * Disables the \"double click to zoom\" interaction.\n *\n * @example\n * ```ts\n * map.doubleClickZoom.disable();\n * ```\n */\n disable() {\n this._clickZoom.disable();\n this._tapZoom.disable();\n }\n\n /**\n * Returns a Boolean indicating whether the \"double click to zoom\" interaction is enabled.\n *\n * @returns `true` if the \"double click to zoom\" interaction is enabled.\n */\n isEnabled() {\n return this._clickZoom.isEnabled() && this._tapZoom.isEnabled();\n }\n\n /**\n * Returns a Boolean indicating whether the \"double click to zoom\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"double click to zoom\" interaction is active.\n */\n isActive() {\n return this._clickZoom.isActive() || this._tapZoom.isActive();\n }\n}\n","import type Point from '@mapbox/point-geometry';\nimport type {Map} from '../map';\nimport {TransformProvider} from './transform-provider';\nimport {Handler} from '../handler_manager';\n\n/**\n * The `ClickZoomHandler` allows the user to zoom the map at a point by double clicking\n * It is used by other handlers\n */\nexport class ClickZoomHandler implements Handler {\n\n _tr: TransformProvider;\n _enabled: boolean;\n _active: boolean;\n\n /** @internal */\n constructor(map: Map) {\n this._tr = new TransformProvider(map);\n this.reset();\n }\n\n reset() {\n this._active = false;\n }\n\n dblclick(e: MouseEvent, point: Point) {\n e.preventDefault();\n return {\n cameraAnimation: (map: Map) => {\n map.easeTo({\n duration: 300,\n zoom: this._tr.zoom + (e.shiftKey ? -1 : 1),\n around: this._tr.unproject(point)\n }, {originalEvent: e});\n }\n };\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n}\n","import {Handler} from '../handler_manager';\nimport {TapRecognizer, MAX_TAP_INTERVAL, MAX_DIST} from './tap_recognizer';\nimport type Point from '@mapbox/point-geometry';\n\nexport class TapDragZoomHandler implements Handler {\n\n _enabled: boolean;\n _active: boolean;\n _swipePoint: Point;\n _swipeTouch: number;\n _tapTime: number;\n _tapPoint: Point;\n _tap: TapRecognizer;\n\n constructor() {\n\n this._tap = new TapRecognizer({\n numTouches: 1,\n numTaps: 1\n });\n\n this.reset();\n }\n\n reset() {\n this._active = false;\n delete this._swipePoint;\n delete this._swipeTouch;\n delete this._tapTime;\n delete this._tapPoint;\n this._tap.reset();\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n if (this._swipePoint) return;\n\n if (!this._tapTime) {\n this._tap.touchstart(e, points, mapTouches);\n } else {\n const swipePoint = points[0];\n\n const soonEnough = e.timeStamp - this._tapTime < MAX_TAP_INTERVAL;\n const closeEnough = this._tapPoint.dist(swipePoint) < MAX_DIST;\n\n if (!soonEnough || !closeEnough) {\n this.reset();\n } else if (mapTouches.length > 0) {\n this._swipePoint = swipePoint;\n this._swipeTouch = mapTouches[0].identifier;\n }\n }\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this._tapTime) {\n this._tap.touchmove(e, points, mapTouches);\n } else if (this._swipePoint) {\n if (mapTouches[0].identifier !== this._swipeTouch) {\n return;\n }\n\n const newSwipePoint = points[0];\n const dist = newSwipePoint.y - this._swipePoint.y;\n this._swipePoint = newSwipePoint;\n\n e.preventDefault();\n this._active = true;\n\n return {\n zoomDelta: dist / 128\n };\n }\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this._tapTime) {\n const point = this._tap.touchend(e, points, mapTouches);\n if (point) {\n this._tapTime = e.timeStamp;\n this._tapPoint = point;\n }\n } else if (this._swipePoint) {\n if (mapTouches.length === 0) {\n this.reset();\n }\n }\n }\n\n touchcancel() {\n this.reset();\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n}\n","import type {MousePanHandler} from '../mouse';\nimport type {TouchPanHandler} from './../touch_pan';\n\n/**\n * A {@link DragPanHandler} options object\n */\nexport type DragPanOptions = {\n /**\n * factor used to scale the drag velocity\n * @defaultValue 0\n */\n linearity?: number;\n /**\n * easing function applled to `map.panTo` when applying the drag.\n * @param t - the easing function\n * @defaultValue bezier(0, 0, 0.3, 1)\n */\n easing?: (t: number) => number;\n /**\n * the maximum value of the drag velocity.\n * @defaultValue 1400\n */\n deceleration?: number;\n /**\n * the rate at which the speed reduces after the pan ends.\n * @defaultValue 2500\n */\n maxSpeed?: number;\n};\n\n/**\n * The `DragPanHandler` allows the user to pan the map by clicking and dragging\n * the cursor.\n *\n * @group Handlers\n */\nexport class DragPanHandler {\n\n _el: HTMLElement;\n _mousePan: MousePanHandler;\n _touchPan: TouchPanHandler;\n _inertiaOptions: DragPanOptions | boolean;\n\n /** @internal */\n constructor(el: HTMLElement, mousePan: MousePanHandler, touchPan: TouchPanHandler) {\n this._el = el;\n this._mousePan = mousePan;\n this._touchPan = touchPan;\n }\n\n /**\n * Enables the \"drag to pan\" interaction.\n *\n * @param options - Options object\n * @example\n * ```ts\n * map.dragPan.enable();\n * map.dragPan.enable({\n * linearity: 0.3,\n * easing: bezier(0, 0, 0.3, 1),\n * maxSpeed: 1400,\n * deceleration: 2500,\n * });\n * ```\n */\n enable(options?: DragPanOptions | boolean) {\n this._inertiaOptions = options || {};\n this._mousePan.enable();\n this._touchPan.enable();\n this._el.classList.add('maplibregl-touch-drag-pan');\n }\n\n /**\n * Disables the \"drag to pan\" interaction.\n *\n * @example\n * ```ts\n * map.dragPan.disable();\n * ```\n */\n disable() {\n this._mousePan.disable();\n this._touchPan.disable();\n this._el.classList.remove('maplibregl-touch-drag-pan');\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to pan\" interaction is enabled.\n *\n * @returns `true` if the \"drag to pan\" interaction is enabled.\n */\n isEnabled() {\n return this._mousePan.isEnabled() && this._touchPan.isEnabled();\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to pan\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"drag to pan\" interaction is active.\n */\n isActive() {\n return this._mousePan.isActive() || this._touchPan.isActive();\n }\n}\n","import type {MousePitchHandler, MouseRotateHandler} from '../mouse';\n\nexport type DragRotateHandlerOptions = {\n /**\n * Control the map pitch in addition to the bearing\n * @defaultValue true\n */\n pitchWithRotate: boolean;\n}\n\n/**\n * The `DragRotateHandler` allows the user to rotate the map by clicking and\n * dragging the cursor while holding the right mouse button or `ctrl` key.\n *\n * @group Handlers\n */\nexport class DragRotateHandler {\n\n _mouseRotate: MouseRotateHandler;\n _mousePitch: MousePitchHandler;\n _pitchWithRotate: boolean;\n\n /** @internal */\n constructor(options: DragRotateHandlerOptions, mouseRotate: MouseRotateHandler, mousePitch: MousePitchHandler) {\n this._pitchWithRotate = options.pitchWithRotate;\n this._mouseRotate = mouseRotate;\n this._mousePitch = mousePitch;\n }\n\n /**\n * Enables the \"drag to rotate\" interaction.\n *\n * @example\n * ```ts\n * map.dragRotate.enable();\n * ```\n */\n enable() {\n this._mouseRotate.enable();\n if (this._pitchWithRotate) this._mousePitch.enable();\n }\n\n /**\n * Disables the \"drag to rotate\" interaction.\n *\n * @example\n * ```ts\n * map.dragRotate.disable();\n * ```\n */\n disable() {\n this._mouseRotate.disable();\n this._mousePitch.disable();\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to rotate\" interaction is enabled.\n *\n * @returns `true` if the \"drag to rotate\" interaction is enabled.\n */\n isEnabled() {\n return this._mouseRotate.isEnabled() && (!this._pitchWithRotate || this._mousePitch.isEnabled());\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to rotate\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"drag to rotate\" interaction is active.\n */\n isActive() {\n return this._mouseRotate.isActive() || this._mousePitch.isActive();\n }\n}\n","import type {TwoFingersTouchZoomHandler, TwoFingersTouchRotateHandler, AroundCenterOptions} from '../two_fingers_touch';\nimport type {TapDragZoomHandler} from '../tap_drag_zoom';\n\n/**\n * The `TwoFingersTouchZoomRotateHandler` allows the user to zoom and rotate the map by\n * pinching on a touchscreen.\n *\n * They can zoom with one finger by double tapping and dragging. On the second tap,\n * hold the finger down and drag up or down to zoom in or out.\n *\n * @group Handlers\n */\nexport class TwoFingersTouchZoomRotateHandler {\n\n _el: HTMLElement;\n _touchZoom: TwoFingersTouchZoomHandler;\n _touchRotate: TwoFingersTouchRotateHandler;\n _tapDragZoom: TapDragZoomHandler;\n _rotationDisabled: boolean;\n _enabled: boolean;\n\n /** @internal */\n constructor(el: HTMLElement, touchZoom: TwoFingersTouchZoomHandler, touchRotate: TwoFingersTouchRotateHandler, tapDragZoom: TapDragZoomHandler) {\n this._el = el;\n this._touchZoom = touchZoom;\n this._touchRotate = touchRotate;\n this._tapDragZoom = tapDragZoom;\n this._rotationDisabled = false;\n this._enabled = true;\n }\n\n /**\n * Enables the \"pinch to rotate and zoom\" interaction.\n *\n * @param options - Options object.\n *\n * @example\n * ```ts\n * map.touchZoomRotate.enable();\n * map.touchZoomRotate.enable({ around: 'center' });\n * ```\n */\n enable(options?: AroundCenterOptions | boolean | null) {\n this._touchZoom.enable(options);\n if (!this._rotationDisabled) this._touchRotate.enable(options);\n this._tapDragZoom.enable();\n this._el.classList.add('maplibregl-touch-zoom-rotate');\n }\n\n /**\n * Disables the \"pinch to rotate and zoom\" interaction.\n *\n * @example\n * ```ts\n * map.touchZoomRotate.disable();\n * ```\n */\n disable() {\n this._touchZoom.disable();\n this._touchRotate.disable();\n this._tapDragZoom.disable();\n this._el.classList.remove('maplibregl-touch-zoom-rotate');\n }\n\n /**\n * Returns a Boolean indicating whether the \"pinch to rotate and zoom\" interaction is enabled.\n *\n * @returns `true` if the \"pinch to rotate and zoom\" interaction is enabled.\n */\n isEnabled() {\n return this._touchZoom.isEnabled() &&\n (this._rotationDisabled || this._touchRotate.isEnabled()) &&\n this._tapDragZoom.isEnabled();\n }\n\n /**\n * Returns true if the handler is enabled and has detected the start of a zoom/rotate gesture.\n *\n * @returns `true` if the handler is active, `false` otherwise\n */\n isActive() {\n return this._touchZoom.isActive() || this._touchRotate.isActive() || this._tapDragZoom.isActive();\n }\n\n /**\n * Disables the \"pinch to rotate\" interaction, leaving the \"pinch to zoom\"\n * interaction enabled.\n *\n * @example\n * ```ts\n * map.touchZoomRotate.disableRotation();\n * ```\n */\n disableRotation() {\n this._rotationDisabled = true;\n this._touchRotate.disable();\n }\n\n /**\n * Enables the \"pinch to rotate\" interaction.\n *\n * @example\n * ```ts\n * map.touchZoomRotate.enable();\n * map.touchZoomRotate.enableRotation();\n * ```\n */\n enableRotation() {\n this._rotationDisabled = false;\n if (this._touchZoom.isEnabled()) this._touchRotate.enable();\n }\n}\n","import {Event} from '../util/evented';\nimport {DOM} from '../util/dom';\nimport {Map, CompleteMapOptions} from './map';\nimport {HandlerInertia} from './handler_inertia';\nimport {MapEventHandler, BlockableMapEventHandler} from './handler/map_event';\nimport {BoxZoomHandler} from './handler/box_zoom';\nimport {TapZoomHandler} from './handler/tap_zoom';\nimport {generateMouseRotationHandler, generateMousePitchHandler, generateMousePanHandler} from './handler/mouse';\nimport {TouchPanHandler} from './handler/touch_pan';\nimport {TwoFingersTouchZoomHandler, TwoFingersTouchRotateHandler, TwoFingersTouchPitchHandler} from './handler/two_fingers_touch';\nimport {KeyboardHandler} from './handler/keyboard';\nimport {ScrollZoomHandler} from './handler/scroll_zoom';\nimport {DoubleClickZoomHandler} from './handler/shim/dblclick_zoom';\nimport {ClickZoomHandler} from './handler/click_zoom';\nimport {TapDragZoomHandler} from './handler/tap_drag_zoom';\nimport {DragPanHandler} from './handler/shim/drag_pan';\nimport {DragRotateHandler} from './handler/shim/drag_rotate';\nimport {TwoFingersTouchZoomRotateHandler} from './handler/shim/two_fingers_touch';\nimport {extend} from '../util/util';\nimport {browser} from '../util/browser';\nimport Point from '@mapbox/point-geometry';\n\nexport type InputEvent = MouseEvent | TouchEvent | KeyboardEvent | WheelEvent;\n\nconst isMoving = p => p.zoom || p.drag || p.pitch || p.rotate;\n\nclass RenderFrameEvent extends Event {\n type: 'renderFrame';\n timeStamp: number;\n}\n\n/**\n * Handlers interpret dom events and return camera changes that should be\n * applied to the map (`HandlerResult`s). The camera changes are all deltas.\n * The handler itself should have no knowledge of the map's current state.\n * This makes it easier to merge multiple results and keeps handlers simpler.\n * For example, if there is a mousedown and mousemove, the mousePan handler\n * would return a `panDelta` on the mousemove.\n */\nexport interface Handler {\n enable(): void;\n disable(): void;\n isEnabled(): boolean;\n isActive(): boolean;\n /**\n * `reset` can be called by the manager at any time and must reset everything to it's original state\n */\n reset(): void;\n // Handlers can optionally implement these methods.\n // They are called with dom events whenever those dom evens are received.\n readonly touchstart?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly touchmove?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly touchmoveWindow?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly touchend?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly touchcancel?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly mousedown?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly mousemove?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly mousemoveWindow?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly mouseup?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly mouseupWindow?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly dblclick?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly contextmenu?: (e: MouseEvent) => HandlerResult | void;\n readonly wheel?: (e: WheelEvent, point: Point) => HandlerResult | void;\n readonly keydown?: (e: KeyboardEvent) => HandlerResult | void;\n readonly keyup?: (e: KeyboardEvent) => HandlerResult | void;\n /**\n * `renderFrame` is the only non-dom event. It is called during render\n * frames and can be used to smooth camera changes (see scroll handler).\n */\n readonly renderFrame?: () => HandlerResult | void;\n}\n\n/**\n * All handler methods that are called with events can optionally return a `HandlerResult`.\n */\nexport type HandlerResult = {\n panDelta?: Point;\n zoomDelta?: number;\n bearingDelta?: number;\n pitchDelta?: number;\n /**\n * the point to not move when changing the camera\n */\n around?: Point | null;\n /**\n * same as above, except for pinch actions, which are given higher priority\n */\n pinchAround?: Point | null;\n /**\n * A method that can fire a one-off easing by directly changing the map's camera.\n */\n cameraAnimation?: (map: Map) => any;\n /**\n * The last three properties are needed by only one handler: scrollzoom.\n * The DOM event to be used as the `originalEvent` on any camera change events.\n */\n originalEvent?: Event;\n /**\n * Makes the manager trigger a frame, allowing the handler to return multiple results over time (see scrollzoom).\n */\n needsRenderFrame?: boolean;\n /**\n * The camera changes won't get recorded for inertial zooming.\n */\n noInertia?: boolean;\n};\n\nexport type EventInProgress = {\n handlerName: string;\n originalEvent: Event;\n}\n\nexport type EventsInProgress = {\n zoom?: EventInProgress;\n pitch?: EventInProgress;\n rotate?: EventInProgress;\n drag?: EventInProgress;\n}\n\nfunction hasChange(result: HandlerResult) {\n return (result.panDelta && result.panDelta.mag()) || result.zoomDelta || result.bearingDelta || result.pitchDelta;\n}\n\nexport class HandlerManager {\n _map: Map;\n _el: HTMLElement;\n _handlers: Array<{\n handlerName: string;\n handler: Handler;\n allowed: Array;\n }>;\n _eventsInProgress: EventsInProgress;\n _frameId: number;\n _inertia: HandlerInertia;\n _bearingSnap: number;\n _handlersById: {[x: string]: Handler};\n _updatingCamera: boolean;\n _changes: Array<[HandlerResult, EventsInProgress, {[handlerName: string]: Event}]>;\n _terrainMovement: boolean;\n _zoom: {handlerName: string};\n _previousActiveHandlers: {[x: string]: Handler};\n _listeners: Array<[Window | Document | HTMLElement, string, {\n passive?: boolean;\n capture?: boolean;\n } | undefined]>;\n\n constructor(map: Map, options: CompleteMapOptions) {\n this._map = map;\n this._el = this._map.getCanvasContainer();\n this._handlers = [];\n this._handlersById = {};\n this._changes = [];\n\n this._inertia = new HandlerInertia(map);\n this._bearingSnap = options.bearingSnap;\n this._previousActiveHandlers = {};\n\n // Track whether map is currently moving, to compute start/move/end events\n this._eventsInProgress = {};\n\n this._addDefaultHandlers(options);\n\n const el = this._el;\n\n this._listeners = [\n // This needs to be `passive: true` so that a double tap fires two\n // pairs of touchstart/end events in iOS Safari 13. If this is set to\n // `passive: false` then the second pair of events is only fired if\n // preventDefault() is called on the first touchstart. Calling preventDefault()\n // undesirably prevents click events.\n [el, 'touchstart', {passive: true}],\n // This needs to be `passive: false` so that scrolls and pinches can be\n // prevented in browsers that don't support `touch-actions: none`, for example iOS Safari 12.\n [el, 'touchmove', {passive: false}],\n [el, 'touchend', undefined],\n [el, 'touchcancel', undefined],\n\n [el, 'mousedown', undefined],\n [el, 'mousemove', undefined],\n [el, 'mouseup', undefined],\n\n // Bind window-level event listeners for move and up/end events. In the absence of\n // the pointer capture API, which is not supported by all necessary platforms,\n // window-level event listeners give us the best shot at capturing events that\n // fall outside the map canvas element. Use `{capture: true}` for the move event\n // to prevent map move events from being fired during a drag.\n [document, 'mousemove', {capture: true}],\n [document, 'mouseup', undefined],\n\n [el, 'mouseover', undefined],\n [el, 'mouseout', undefined],\n [el, 'dblclick', undefined],\n [el, 'click', undefined],\n\n [el, 'keydown', {capture: false}],\n [el, 'keyup', undefined],\n\n [el, 'wheel', {passive: false}],\n [el, 'contextmenu', undefined],\n\n [window, 'blur', undefined]\n ];\n\n for (const [target, type, listenerOptions] of this._listeners) {\n DOM.addEventListener(target, type, target === document ? this.handleWindowEvent : this.handleEvent, listenerOptions);\n }\n }\n\n destroy() {\n for (const [target, type, listenerOptions] of this._listeners) {\n DOM.removeEventListener(target, type, target === document ? this.handleWindowEvent : this.handleEvent, listenerOptions);\n }\n }\n\n _addDefaultHandlers(options: CompleteMapOptions) {\n const map = this._map;\n const el = map.getCanvasContainer();\n this._add('mapEvent', new MapEventHandler(map, options));\n\n const boxZoom = map.boxZoom = new BoxZoomHandler(map, options);\n this._add('boxZoom', boxZoom);\n if (options.interactive && options.boxZoom) {\n boxZoom.enable();\n }\n\n const tapZoom = new TapZoomHandler(map);\n const clickZoom = new ClickZoomHandler(map);\n map.doubleClickZoom = new DoubleClickZoomHandler(clickZoom, tapZoom);\n this._add('tapZoom', tapZoom);\n this._add('clickZoom', clickZoom);\n if (options.interactive && options.doubleClickZoom) {\n map.doubleClickZoom.enable();\n }\n\n const tapDragZoom = new TapDragZoomHandler();\n this._add('tapDragZoom', tapDragZoom);\n\n const touchPitch = map.touchPitch = new TwoFingersTouchPitchHandler(map);\n this._add('touchPitch', touchPitch);\n if (options.interactive && options.touchPitch) {\n map.touchPitch.enable(options.touchPitch);\n }\n\n const mouseRotate = generateMouseRotationHandler(options);\n const mousePitch = generateMousePitchHandler(options);\n map.dragRotate = new DragRotateHandler(options, mouseRotate, mousePitch);\n this._add('mouseRotate', mouseRotate, ['mousePitch']);\n this._add('mousePitch', mousePitch, ['mouseRotate']);\n if (options.interactive && options.dragRotate) {\n map.dragRotate.enable();\n }\n\n const mousePan = generateMousePanHandler(options);\n const touchPan = new TouchPanHandler(options, map);\n map.dragPan = new DragPanHandler(el, mousePan, touchPan);\n this._add('mousePan', mousePan);\n this._add('touchPan', touchPan, ['touchZoom', 'touchRotate']);\n if (options.interactive && options.dragPan) {\n map.dragPan.enable(options.dragPan);\n }\n\n const touchRotate = new TwoFingersTouchRotateHandler();\n const touchZoom = new TwoFingersTouchZoomHandler();\n map.touchZoomRotate = new TwoFingersTouchZoomRotateHandler(el, touchZoom, touchRotate, tapDragZoom);\n this._add('touchRotate', touchRotate, ['touchPan', 'touchZoom']);\n this._add('touchZoom', touchZoom, ['touchPan', 'touchRotate']);\n if (options.interactive && options.touchZoomRotate) {\n map.touchZoomRotate.enable(options.touchZoomRotate);\n }\n\n const scrollZoom = map.scrollZoom = new ScrollZoomHandler(map, () => this._triggerRenderFrame());\n this._add('scrollZoom', scrollZoom, ['mousePan']);\n if (options.interactive && options.scrollZoom) {\n map.scrollZoom.enable(options.scrollZoom);\n }\n\n const keyboard = map.keyboard = new KeyboardHandler(map);\n this._add('keyboard', keyboard);\n if (options.interactive && options.keyboard) {\n map.keyboard.enable();\n }\n\n this._add('blockableMapEvent', new BlockableMapEventHandler(map));\n }\n\n _add(handlerName: string, handler: Handler, allowed?: Array) {\n this._handlers.push({handlerName, handler, allowed});\n this._handlersById[handlerName] = handler;\n }\n\n stop(allowEndAnimation: boolean) {\n // do nothing if this method was triggered by a gesture update\n if (this._updatingCamera) return;\n\n for (const {handler} of this._handlers) {\n handler.reset();\n }\n this._inertia.clear();\n this._fireEvents({}, {}, allowEndAnimation);\n this._changes = [];\n }\n\n isActive() {\n for (const {handler} of this._handlers) {\n if (handler.isActive()) return true;\n }\n return false;\n }\n\n isZooming() {\n return !!this._eventsInProgress.zoom || this._map.scrollZoom.isZooming();\n }\n isRotating() {\n return !!this._eventsInProgress.rotate;\n }\n\n isMoving() {\n return Boolean(isMoving(this._eventsInProgress)) || this.isZooming();\n }\n\n _blockedByActive(activeHandlers: {[x: string]: Handler}, allowed: Array, myName: string) {\n for (const name in activeHandlers) {\n if (name === myName) continue;\n if (!allowed || allowed.indexOf(name) < 0) {\n return true;\n }\n }\n return false;\n }\n\n handleWindowEvent = (e: { type: 'mousemove' | 'mouseup' | 'touchmove'}) => {\n this.handleEvent(e, `${e.type}Window`);\n };\n\n _getMapTouches(touches: TouchList) {\n const mapTouches = [];\n for (const t of touches) {\n const target = (t.target as any as Node);\n if (this._el.contains(target)) {\n mapTouches.push(t);\n }\n }\n return mapTouches as any as TouchList;\n }\n\n handleEvent = (e: Event, eventName?: keyof Handler) => {\n\n if (e.type === 'blur') {\n this.stop(true);\n return;\n }\n\n this._updatingCamera = true;\n\n const inputEvent = e.type === 'renderFrame' ? undefined : e as InputEvent;\n\n /*\n * We don't call e.preventDefault() for any events by default.\n * Handlers are responsible for calling it where necessary.\n */\n\n const mergedHandlerResult: HandlerResult = {needsRenderFrame: false};\n const eventsInProgress: EventsInProgress = {};\n const activeHandlers = {};\n const eventTouches = (e as TouchEvent).touches;\n\n const mapTouches = eventTouches ? this._getMapTouches(eventTouches) : undefined;\n const points = mapTouches ? DOM.touchPos(this._el, mapTouches) : DOM.mousePos(this._el, ((e as MouseEvent)));\n\n for (const {handlerName, handler, allowed} of this._handlers) {\n if (!handler.isEnabled()) continue;\n\n let data: HandlerResult;\n if (this._blockedByActive(activeHandlers, allowed, handlerName)) {\n handler.reset();\n\n } else {\n if (handler[eventName || e.type]) {\n data = handler[eventName || e.type](e, points, mapTouches);\n this.mergeHandlerResult(mergedHandlerResult, eventsInProgress, data, handlerName, inputEvent);\n if (data && data.needsRenderFrame) {\n this._triggerRenderFrame();\n }\n }\n }\n\n if (data || handler.isActive()) {\n activeHandlers[handlerName] = handler;\n }\n }\n\n const deactivatedHandlers: {[handlerName: string]: Event} = {};\n for (const name in this._previousActiveHandlers) {\n if (!activeHandlers[name]) {\n deactivatedHandlers[name] = inputEvent;\n }\n }\n this._previousActiveHandlers = activeHandlers;\n\n if (Object.keys(deactivatedHandlers).length || hasChange(mergedHandlerResult)) {\n this._changes.push([mergedHandlerResult, eventsInProgress, deactivatedHandlers]);\n this._triggerRenderFrame();\n }\n\n if (Object.keys(activeHandlers).length || hasChange(mergedHandlerResult)) {\n this._map._stop(true);\n }\n\n this._updatingCamera = false;\n\n const {cameraAnimation} = mergedHandlerResult;\n if (cameraAnimation) {\n this._inertia.clear();\n this._fireEvents({}, {}, true);\n this._changes = [];\n cameraAnimation(this._map);\n }\n };\n\n mergeHandlerResult(mergedHandlerResult: HandlerResult,\n eventsInProgress: EventsInProgress,\n handlerResult: HandlerResult,\n name: string,\n e?: InputEvent) {\n if (!handlerResult) return;\n\n extend(mergedHandlerResult, handlerResult);\n\n const eventData = {handlerName: name, originalEvent: handlerResult.originalEvent || e};\n\n // track which handler changed which camera property\n if (handlerResult.zoomDelta !== undefined) {\n eventsInProgress.zoom = eventData;\n }\n if (handlerResult.panDelta !== undefined) {\n eventsInProgress.drag = eventData;\n }\n if (handlerResult.pitchDelta !== undefined) {\n eventsInProgress.pitch = eventData;\n }\n if (handlerResult.bearingDelta !== undefined) {\n eventsInProgress.rotate = eventData;\n }\n\n }\n\n _applyChanges() {\n const combined: HandlerResult = {};\n const combinedEventsInProgress: EventsInProgress = {};\n const combinedDeactivatedHandlers = {};\n\n for (const [change, eventsInProgress, deactivatedHandlers] of this._changes) {\n\n if (change.panDelta) combined.panDelta = (combined.panDelta || new Point(0, 0))._add(change.panDelta);\n if (change.zoomDelta) combined.zoomDelta = (combined.zoomDelta || 0) + change.zoomDelta;\n if (change.bearingDelta) combined.bearingDelta = (combined.bearingDelta || 0) + change.bearingDelta;\n if (change.pitchDelta) combined.pitchDelta = (combined.pitchDelta || 0) + change.pitchDelta;\n if (change.around !== undefined) combined.around = change.around;\n if (change.pinchAround !== undefined) combined.pinchAround = change.pinchAround;\n if (change.noInertia) combined.noInertia = change.noInertia;\n\n extend(combinedEventsInProgress, eventsInProgress);\n extend(combinedDeactivatedHandlers, deactivatedHandlers);\n }\n\n this._updateMapTransform(combined, combinedEventsInProgress, combinedDeactivatedHandlers);\n this._changes = [];\n }\n\n _updateMapTransform(combinedResult: HandlerResult,\n combinedEventsInProgress: EventsInProgress,\n deactivatedHandlers: {[handlerName: string]: Event}) {\n const map = this._map;\n const tr = map._getTransformForUpdate();\n const terrain = map.terrain;\n\n if (!hasChange(combinedResult) && !(terrain && this._terrainMovement)) {\n return this._fireEvents(combinedEventsInProgress, deactivatedHandlers, true);\n }\n\n let {panDelta, zoomDelta, bearingDelta, pitchDelta, around, pinchAround} = combinedResult;\n\n if (pinchAround !== undefined) {\n around = pinchAround;\n }\n\n // stop any ongoing camera animations (easeTo, flyTo)\n map._stop(true);\n\n around = around || map.transform.centerPoint;\n const loc = tr.pointLocation(panDelta ? around.sub(panDelta) : around);\n if (bearingDelta) tr.bearing += bearingDelta;\n if (pitchDelta) tr.pitch += pitchDelta;\n if (zoomDelta) tr.zoom += zoomDelta;\n\n if (!terrain) {\n tr.setLocationAtPoint(loc, around);\n } else {\n // when 3d-terrain is enabled act a little different:\n // - dragging do not drag the picked point itself, instead it drags the map by pixel-delta.\n // With this approach it is no longer possible to pick a point from somewhere near\n // the horizon to the center in one move.\n // So this logic avoids the problem, that in such cases you easily loose orientation.\n if (!this._terrainMovement &&\n (combinedEventsInProgress.drag || combinedEventsInProgress.zoom)) {\n // When starting to drag or move, flag it and register moveend to clear flagging\n this._terrainMovement = true;\n this._map._elevationFreeze = true;\n tr.setLocationAtPoint(loc, around);\n this._map.once('moveend', () => {\n this._map._elevationFreeze = false;\n this._terrainMovement = false;\n tr.recalculateZoom(map.terrain);\n });\n } else if (combinedEventsInProgress.drag && this._terrainMovement) {\n // drag map\n tr.center = tr.pointLocation(tr.centerPoint.sub(panDelta));\n } else {\n tr.setLocationAtPoint(loc, around);\n }\n }\n\n map._applyUpdatedTransform(tr);\n\n this._map._update();\n if (!combinedResult.noInertia) this._inertia.record(combinedResult);\n this._fireEvents(combinedEventsInProgress, deactivatedHandlers, true);\n\n }\n\n _fireEvents(newEventsInProgress: EventsInProgress, deactivatedHandlers: {[handlerName: string]: Event}, allowEndAnimation: boolean) {\n\n const wasMoving = isMoving(this._eventsInProgress);\n const nowMoving = isMoving(newEventsInProgress);\n\n const startEvents = {};\n\n for (const eventName in newEventsInProgress) {\n const {originalEvent} = newEventsInProgress[eventName];\n if (!this._eventsInProgress[eventName]) {\n startEvents[`${eventName}start`] = originalEvent;\n }\n this._eventsInProgress[eventName] = newEventsInProgress[eventName];\n }\n\n // fire start events only after this._eventsInProgress has been updated\n if (!wasMoving && nowMoving) {\n this._fireEvent('movestart', nowMoving.originalEvent);\n }\n\n for (const name in startEvents) {\n this._fireEvent(name, startEvents[name]);\n }\n\n if (nowMoving) {\n this._fireEvent('move', nowMoving.originalEvent);\n }\n\n for (const eventName in newEventsInProgress) {\n const {originalEvent} = newEventsInProgress[eventName];\n this._fireEvent(eventName, originalEvent);\n }\n\n const endEvents = {};\n\n let originalEndEvent;\n for (const eventName in this._eventsInProgress) {\n const {handlerName, originalEvent} = this._eventsInProgress[eventName];\n if (!this._handlersById[handlerName].isActive()) {\n delete this._eventsInProgress[eventName];\n originalEndEvent = deactivatedHandlers[handlerName] || originalEvent;\n endEvents[`${eventName}end`] = originalEndEvent;\n }\n }\n\n for (const name in endEvents) {\n this._fireEvent(name, endEvents[name]);\n }\n\n const stillMoving = isMoving(this._eventsInProgress);\n if (allowEndAnimation && (wasMoving || nowMoving) && !stillMoving) {\n this._updatingCamera = true;\n const inertialEase = this._inertia._onMoveEnd(this._map.dragPan._inertiaOptions);\n\n const shouldSnapToNorth = bearing => bearing !== 0 && -this._bearingSnap < bearing && bearing < this._bearingSnap;\n\n if (inertialEase && (inertialEase.essential || !browser.prefersReducedMotion)) {\n if (shouldSnapToNorth(inertialEase.bearing || this._map.getBearing())) {\n inertialEase.bearing = 0;\n }\n inertialEase.freezeElevation = true;\n this._map.easeTo(inertialEase, {originalEvent: originalEndEvent});\n } else {\n this._map.fire(new Event('moveend', {originalEvent: originalEndEvent}));\n if (shouldSnapToNorth(this._map.getBearing())) {\n this._map.resetNorth();\n }\n }\n this._updatingCamera = false;\n }\n\n }\n\n _fireEvent(type: string, e?: Event) {\n this._map.fire(new Event(type, e ? {originalEvent: e} : {}));\n }\n\n _requestFrame() {\n this._map.triggerRepaint();\n return this._map._renderTaskQueue.add(timeStamp => {\n delete this._frameId;\n this.handleEvent(new RenderFrameEvent('renderFrame', {timeStamp}));\n this._applyChanges();\n });\n }\n\n _triggerRenderFrame() {\n if (this._frameId === undefined) {\n this._frameId = this._requestFrame();\n }\n }\n}\n","import {extend, warnOnce, clamp, wrap, defaultEasing, pick} from '../util/util';\nimport {interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {browser} from '../util/browser';\nimport {LngLat} from '../geo/lng_lat';\nimport {LngLatBounds} from '../geo/lng_lat_bounds';\nimport Point from '@mapbox/point-geometry';\nimport {Event, Evented} from '../util/evented';\nimport {Terrain} from '../render/terrain';\n\nimport type {Transform} from '../geo/transform';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport type {LngLatBoundsLike} from '../geo/lng_lat_bounds';\nimport type {TaskID} from '../util/task_queue';\nimport type {PaddingOptions} from '../geo/edge_insets';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\n\n/**\n * A [Point](https://github.com/mapbox/point-geometry) or an array of two numbers representing `x` and `y` screen coordinates in pixels.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let p1 = new Point(-77, 38); // a PointLike which is a Point\n * let p2 = [-77, 38]; // a PointLike which is an array of two numbers\n * ```\n */\nexport type PointLike = Point | [number, number];\n\n/**\n * A helper to allow require of at least one propery\n */\nexport type RequireAtLeastOne = { [K in keyof T]-?: Required> & Partial>>; }[keyof T];\n\n/**\n * Options common to {@link Map#jumpTo}, {@link Map#easeTo}, and {@link Map#flyTo}, controlling the desired location,\n * zoom, bearing, and pitch of the camera. All properties are optional, and when a property is omitted, the current\n * camera value for that property will remain unchanged.\n *\n * @example\n * Set the map's initial perspective with CameraOptions\n * ```ts\n * let map = new maplibregl.Map({\n * container: 'map',\n * style: 'https://demotiles.maplibre.org/style.json',\n * center: [-73.5804, 45.53483],\n * pitch: 60,\n * bearing: -60,\n * zoom: 10\n * });\n * ```\n * @see [Set pitch and bearing](https://maplibre.org/maplibre-gl-js/docs/examples/set-perspective/)\n * @see [Jump to a series of locations](https://maplibre.org/maplibre-gl-js/docs/examples/jump-to/)\n * @see [Fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto/)\n * @see [Display buildings in 3D](https://maplibre.org/maplibre-gl-js/docs/examples/3d-buildings/)\n */\nexport type CameraOptions = CenterZoomBearing & {\n /**\n * The desired pitch in degrees. The pitch is the angle towards the horizon\n * measured in degrees with a range between 0 and 60 degrees. For example, pitch: 0 provides the appearance\n * of looking straight down at the map, while pitch: 60 tilts the user's perspective towards the horizon.\n * Increasing the pitch value is often used to display 3D objects.\n */\n pitch?: number;\n /**\n * If `zoom` is specified, `around` determines the point around which the zoom is centered.\n */\n around?: LngLatLike;\n};\n\n/**\n * Holds center, zoom and bearing properties\n */\nexport type CenterZoomBearing = {\n /**\n * The desired center.\n */\n center?: LngLatLike;\n /**\n * The desired zoom level.\n */\n zoom?: number;\n /**\n * The desired bearing in degrees. The bearing is the compass direction that\n * is \"up\". For example, `bearing: 90` orients the map so that east is up.\n */\n bearing?: number;\n}\n\n/**\n * The options object related to the {@link Map#jumpTo} method\n */\nexport type JumpToOptions = CameraOptions & {\n /**\n * Dimensions in pixels applied on each side of the viewport for shifting the vanishing point.\n */\n padding?: PaddingOptions;\n}\n\n/**\n * A options object for the {@link Map#cameraForBounds} method\n */\nexport type CameraForBoundsOptions = CameraOptions & {\n /**\n * The amount of padding in pixels to add to the given bounds.\n */\n padding?: number | RequireAtLeastOne;\n /**\n * The center of the given bounds relative to the map's center, measured in pixels.\n * @defaultValue [0, 0]\n */\n offset?: PointLike;\n /**\n * The maximum zoom level to allow when the camera would transition to the specified bounds.\n */\n maxZoom?: number;\n}\n\n/**\n * The {@link Map#flyTo} options object\n */\nexport type FlyToOptions = AnimationOptions & CameraOptions & {\n /**\n * The zooming \"curve\" that will occur along the\n * flight path. A high value maximizes zooming for an exaggerated animation, while a low\n * value minimizes zooming for an effect closer to {@link Map#easeTo}. 1.42 is the average\n * value selected by participants in the user study discussed in\n * [van Wijk (2003)](https://www.win.tue.nl/~vanwijk/zoompan.pdf). A value of\n * `Math.pow(6, 0.25)` would be equivalent to the root mean squared average velocity. A\n * value of 1 would produce a circular motion.\n * @defaultValue 1.42\n */\n curve?: number;\n /**\n * The zero-based zoom level at the peak of the flight path. If\n * `options.curve` is specified, this option is ignored.\n */\n minZoom?: number;\n /**\n * The average speed of the animation defined in relation to\n * `options.curve`. A speed of 1.2 means that the map appears to move along the flight path\n * by 1.2 times `options.curve` screenfuls every second. A _screenful_ is the map's visible span.\n * It does not correspond to a fixed physical distance, but varies by zoom level.\n * @defaultValue 1.2\n */\n speed?: number;\n /**\n * The average speed of the animation measured in screenfuls\n * per second, assuming a linear timing curve. If `options.speed` is specified, this option is ignored.\n */\n screenSpeed?: number;\n /**\n * The animation's maximum duration, measured in milliseconds.\n * If duration exceeds maximum duration, it resets to 0.\n */\n maxDuration?: number;\n /**\n * The amount of padding in pixels to add to the given bounds.\n */\n padding?: number | RequireAtLeastOne;\n}\n\nexport type EaseToOptions = AnimationOptions & CameraOptions & {\n delayEndEvents?: number;\n padding?: number | RequireAtLeastOne;\n}\n\n/**\n * Options for {@link Map#fitBounds} method\n */\nexport type FitBoundsOptions = FlyToOptions & {\n /**\n * If `true`, the map transitions using {@link Map#easeTo}. If `false`, the map transitions using {@link Map#flyTo}.\n * See those functions and {@link AnimationOptions} for information about options available.\n * @defaultValue false\n */\n linear?: boolean;\n /**\n * The center of the given bounds relative to the map's center, measured in pixels.\n * @defaultValue [0, 0]\n */\n offset?: PointLike;\n /**\n * The maximum zoom level to allow when the map view transitions to the specified bounds.\n */\n maxZoom?: number;\n}\n\n/**\n * Options common to map movement methods that involve animation, such as {@link Map#panBy} and\n * {@link Map#easeTo}, controlling the duration and easing function of the animation. All properties\n * are optional.\n *\n */\nexport type AnimationOptions = {\n /**\n * The animation's duration, measured in milliseconds.\n */\n duration?: number;\n /**\n * A function taking a time in the range 0..1 and returning a number where 0 is\n * the initial state and 1 is the final state.\n */\n easing?: (_: number) => number;\n /**\n * of the target center relative to real map container center at the end of animation.\n */\n offset?: PointLike;\n /**\n * If `false`, no animation will occur.\n */\n animate?: boolean;\n /**\n * If `true`, then the animation is considered essential and will not be affected by\n * [`prefers-reduced-motion`](https://developer.mozilla.org/en-US/docs/Web/CSS/\\@media/prefers-reduced-motion).\n */\n essential?: boolean;\n /**\n * Default false. Needed in 3D maps to let the camera stay in a constant\n * height based on sea-level. After the animation finished the zoom-level will be recalculated in respect of\n * the distance from the camera to the center-coordinate-altitude.\n */\n freezeElevation?: boolean;\n};\n\n/**\n * A callback hook that allows manipulating the camera and being notified about camera updates before they happen\n */\nexport type CameraUpdateTransformFunction = (next: {\n center: LngLat;\n zoom: number;\n pitch: number;\n bearing: number;\n elevation: number;\n}) => {\n center?: LngLat;\n zoom?: number;\n pitch?: number;\n bearing?: number;\n elevation?: number;\n};\n\nexport abstract class Camera extends Evented {\n transform: Transform;\n terrain: Terrain;\n\n _moving: boolean;\n _zooming: boolean;\n _rotating: boolean;\n _pitching: boolean;\n _padding: boolean;\n\n _bearingSnap: number;\n _easeStart: number;\n _easeOptions: {\n duration?: number;\n easing?: (_: number) => number;\n };\n _easeId: string | void;\n\n _onEaseFrame: (_: number) => void;\n _onEaseEnd: (easeId?: string) => void;\n _easeFrameId: TaskID;\n\n /**\n * @internal\n * holds the geographical coordinate of the target\n */\n _elevationCenter: LngLat;\n /**\n * @internal\n * holds the targ altitude value, = center elevation of the target.\n * This value may changes during flight, because new terrain-tiles loads during flight.\n */\n _elevationTarget: number;\n /**\n * @internal\n * holds the start altitude value, = center elevation before animation begins\n * this value will recalculated during flight in respect of changing _elevationTarget values,\n * so the linear interpolation between start and target keeps smooth and without jumps.\n */\n _elevationStart: number;\n /**\n * @internal\n * Saves the current state of the elevation freeze - this is used during map movement to prevent \"rocky\" camera movement.\n */\n _elevationFreeze: boolean;\n /**\n * @internal\n * Used to track accumulated changes during continuous interaction\n */\n _requestedCameraState?: Transform;\n /**\n * A callback used to defer camera updates or apply arbitrary constraints.\n * If specified, this Camera instance can be used as a stateless component in React etc.\n */\n transformCameraUpdate: CameraUpdateTransformFunction | null;\n\n abstract _requestRenderFrame(a: () => void): TaskID;\n abstract _cancelRenderFrame(_: TaskID): void;\n\n constructor(transform: Transform, options: {\n bearingSnap: number;\n }) {\n super();\n this._moving = false;\n this._zooming = false;\n this.transform = transform;\n this._bearingSnap = options.bearingSnap;\n\n this.on('moveend', () => {\n delete this._requestedCameraState;\n });\n }\n\n /**\n * Returns the map's geographical centerpoint.\n *\n * @returns The map's geographical centerpoint.\n * @example\n * Return a LngLat object such as `{lng: 0, lat: 0}`\n * ```ts\n * let center = map.getCenter();\n * // access longitude and latitude values directly\n * let {lng, lat} = map.getCenter();\n * ```\n */\n getCenter(): LngLat { return new LngLat(this.transform.center.lng, this.transform.center.lat); }\n\n /**\n * Sets the map's geographical centerpoint. Equivalent to `jumpTo({center: center})`.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param center - The centerpoint to set.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * map.setCenter([-74, 38]);\n * ```\n */\n setCenter(center: LngLatLike, eventData?: any) {\n return this.jumpTo({center}, eventData);\n }\n\n /**\n * Pans the map by the specified offset.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param offset - `x` and `y` coordinates by which to pan the map.\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/)\n */\n panBy(offset: PointLike, options?: AnimationOptions, eventData?: any): this {\n offset = Point.convert(offset).mult(-1);\n return this.panTo(this.transform.center, extend({offset}, options), eventData);\n }\n\n /**\n * Pans the map to the specified location with an animated transition.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param lnglat - The location to pan the map to.\n * @param options - Options describing the destination and animation of the transition.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * map.panTo([-74, 38]);\n * // Specify that the panTo animation should last 5000 milliseconds.\n * map.panTo([-74, 38], {duration: 5000});\n * ```\n * @see [Update a feature in realtime](https://maplibre.org/maplibre-gl-js/docs/examples/live-update-feature/)\n */\n panTo(lnglat: LngLatLike, options?: AnimationOptions, eventData?: any): this {\n return this.easeTo(extend({\n center: lnglat\n }, options), eventData);\n }\n\n /**\n * Returns the map's current zoom level.\n *\n * @returns The map's current zoom level.\n * @example\n * ```ts\n * map.getZoom();\n * ```\n */\n getZoom(): number { return this.transform.zoom; }\n\n /**\n * Sets the map's zoom level. Equivalent to `jumpTo({zoom: zoom})`.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`.\n *\n * @param zoom - The zoom level to set (0-20).\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Zoom to the zoom level 5 without an animated transition\n * ```ts\n * map.setZoom(5);\n * ```\n */\n setZoom(zoom: number, eventData?: any): this {\n this.jumpTo({zoom}, eventData);\n return this;\n }\n\n /**\n * Zooms the map to the specified zoom level, with an animated transition.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`.\n *\n * @param zoom - The zoom level to transition to.\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * // Zoom to the zoom level 5 without an animated transition\n * map.zoomTo(5);\n * // Zoom to the zoom level 8 with an animated transition\n * map.zoomTo(8, {\n * duration: 2000,\n * offset: [100, 50]\n * });\n * ```\n */\n zoomTo(zoom: number, options?: AnimationOptions | null, eventData?: any): this {\n return this.easeTo(extend({\n zoom\n }, options), eventData);\n }\n\n /**\n * Increases the map's zoom level by 1.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Zoom the map in one level with a custom animation duration\n * ```ts\n * map.zoomIn({duration: 1000});\n * ```\n */\n zoomIn(options?: AnimationOptions, eventData?: any): this {\n this.zoomTo(this.getZoom() + 1, options, eventData);\n return this;\n }\n\n /**\n * Decreases the map's zoom level by 1.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Zoom the map out one level with a custom animation offset\n * ```ts\n * map.zoomOut({offset: [80, 60]});\n * ```\n */\n zoomOut(options?: AnimationOptions, eventData?: any): this {\n this.zoomTo(this.getZoom() - 1, options, eventData);\n return this;\n }\n\n /**\n * Returns the map's current bearing. The bearing is the compass direction that is \"up\"; for example, a bearing\n * of 90° orients the map so that east is up.\n *\n * @returns The map's current bearing.\n * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/)\n */\n getBearing(): number { return this.transform.bearing; }\n\n /**\n * Sets the map's bearing (rotation). The bearing is the compass direction that is \"up\"; for example, a bearing\n * of 90° orients the map so that east is up.\n *\n * Equivalent to `jumpTo({bearing: bearing})`.\n *\n * Triggers the following events: `movestart`, `moveend`, and `rotate`.\n *\n * @param bearing - The desired bearing.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Rotate the map to 90 degrees\n * ```ts\n * map.setBearing(90);\n * ```\n */\n setBearing(bearing: number, eventData?: any): this {\n this.jumpTo({bearing}, eventData);\n return this;\n }\n\n /**\n * Returns the current padding applied around the map viewport.\n *\n * @returns The current padding around the map viewport.\n */\n getPadding(): PaddingOptions { return this.transform.padding; }\n\n /**\n * Sets the padding in pixels around the viewport.\n *\n * Equivalent to `jumpTo({padding: padding})`.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param padding - The desired padding.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Sets a left padding of 300px, and a top padding of 50px\n * ```ts\n * map.setPadding({ left: 300, top: 50 });\n * ```\n */\n setPadding(padding: PaddingOptions, eventData?: any): this {\n this.jumpTo({padding}, eventData);\n return this;\n }\n\n /**\n * Rotates the map to the specified bearing, with an animated transition. The bearing is the compass direction\n * that is \"up\"; for example, a bearing of 90° orients the map so that east is up.\n *\n * Triggers the following events: `movestart`, `moveend`, and `rotate`.\n *\n * @param bearing - The desired bearing.\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n rotateTo(bearing: number, options?: AnimationOptions, eventData?: any): this {\n return this.easeTo(extend({\n bearing\n }, options), eventData);\n }\n\n /**\n * Rotates the map so that north is up (0° bearing), with an animated transition.\n *\n * Triggers the following events: `movestart`, `moveend`, and `rotate`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n resetNorth(options?: AnimationOptions, eventData?: any): this {\n this.rotateTo(0, extend({duration: 1000}, options), eventData);\n return this;\n }\n\n /**\n * Rotates and pitches the map so that north is up (0° bearing) and pitch is 0°, with an animated transition.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `pitchstart`, `pitch`, `pitchend`, and `rotate`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n resetNorthPitch(options?: AnimationOptions, eventData?: any): this {\n this.easeTo(extend({\n bearing: 0,\n pitch: 0,\n duration: 1000\n }, options), eventData);\n return this;\n }\n\n /**\n * Snaps the map so that north is up (0° bearing), if the current bearing is close enough to it (i.e. within the\n * `bearingSnap` threshold).\n *\n * Triggers the following events: `movestart`, `moveend`, and `rotate`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n snapToNorth(options?: AnimationOptions, eventData?: any): this {\n if (Math.abs(this.getBearing()) < this._bearingSnap) {\n return this.resetNorth(options, eventData);\n }\n return this;\n }\n\n /**\n * Returns the map's current pitch (tilt).\n *\n * @returns The map's current pitch, measured in degrees away from the plane of the screen.\n */\n getPitch(): number { return this.transform.pitch; }\n\n /**\n * Sets the map's pitch (tilt). Equivalent to `jumpTo({pitch: pitch})`.\n *\n * Triggers the following events: `movestart`, `moveend`, `pitchstart`, and `pitchend`.\n *\n * @param pitch - The pitch to set, measured in degrees away from the plane of the screen (0-60).\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n setPitch(pitch: number, eventData?: any): this {\n this.jumpTo({pitch}, eventData);\n return this;\n }\n\n /**\n * @param bounds - Calculate the center for these bounds in the viewport and use\n * the highest zoom level up to and including `Map#getMaxZoom()` that fits\n * in the viewport. LngLatBounds represent a box that is always axis-aligned with bearing 0.\n * @param options - Options object\n * @returns If map is able to fit to provided bounds, returns `center`, `zoom`, and `bearing`.\n * If map is unable to fit, method will warn and return undefined.\n * @example\n * ```ts\n * let bbox = [[-79, 43], [-73, 45]];\n * let newCameraTransform = map.cameraForBounds(bbox, {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n */\n cameraForBounds(bounds: LngLatBoundsLike, options?: CameraForBoundsOptions): CenterZoomBearing {\n bounds = LngLatBounds.convert(bounds);\n const bearing = options && options.bearing || 0;\n return this._cameraForBoxAndBearing(bounds.getNorthWest(), bounds.getSouthEast(), bearing, options);\n }\n\n /**\n * @internal\n * Calculate the center of these two points in the viewport and use\n * the highest zoom level up to and including `Map#getMaxZoom()` that fits\n * the points in the viewport at the specified bearing.\n * @param p0 - First point\n * @param p1 - Second point\n * @param bearing - Desired map bearing at end of animation, in degrees\n * @param options - the camera options\n * @returns If map is able to fit to provided bounds, returns `center`, `zoom`, and `bearing`.\n * If map is unable to fit, method will warn and return undefined.\n * @example\n * ```ts\n * let p0 = [-79, 43];\n * let p1 = [-73, 45];\n * let bearing = 90;\n * let newCameraTransform = map._cameraForBoxAndBearing(p0, p1, bearing, {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n */\n _cameraForBoxAndBearing(p0: LngLatLike, p1: LngLatLike, bearing: number, options?: CameraForBoundsOptions): CenterZoomBearing {\n const defaultPadding = {\n top: 0,\n bottom: 0,\n right: 0,\n left: 0\n };\n options = extend({\n padding: defaultPadding,\n offset: [0, 0],\n maxZoom: this.transform.maxZoom\n }, options);\n\n if (typeof options.padding === 'number') {\n const p = options.padding;\n options.padding = {\n top: p,\n bottom: p,\n right: p,\n left: p\n };\n }\n\n options.padding = extend(defaultPadding, options.padding) as PaddingOptions;\n const tr = this.transform;\n const edgePadding = tr.padding;\n\n // We want to calculate the upper right and lower left of the box defined by p0 and p1\n // in a coordinate system rotate to match the destination bearing.\n const p0world = tr.project(LngLat.convert(p0));\n const p1world = tr.project(LngLat.convert(p1));\n const p0rotated = p0world.rotate(-bearing * Math.PI / 180);\n const p1rotated = p1world.rotate(-bearing * Math.PI / 180);\n\n const upperRight = new Point(Math.max(p0rotated.x, p1rotated.x), Math.max(p0rotated.y, p1rotated.y));\n const lowerLeft = new Point(Math.min(p0rotated.x, p1rotated.x), Math.min(p0rotated.y, p1rotated.y));\n\n // Calculate zoom: consider the original bbox and padding.\n const size = upperRight.sub(lowerLeft);\n const scaleX = (tr.width - (edgePadding.left + edgePadding.right + options.padding.left + options.padding.right)) / size.x;\n const scaleY = (tr.height - (edgePadding.top + edgePadding.bottom + options.padding.top + options.padding.bottom)) / size.y;\n\n if (scaleY < 0 || scaleX < 0) {\n warnOnce(\n 'Map cannot fit within canvas with the given bounds, padding, and/or offset.'\n );\n return undefined;\n }\n\n const zoom = Math.min(tr.scaleZoom(tr.scale * Math.min(scaleX, scaleY)), options.maxZoom);\n\n // Calculate center: apply the zoom, the configured offset, as well as offset that exists as a result of padding.\n const offset = Point.convert(options.offset);\n const paddingOffsetX = (options.padding.left - options.padding.right) / 2;\n const paddingOffsetY = (options.padding.top - options.padding.bottom) / 2;\n const paddingOffset = new Point(paddingOffsetX, paddingOffsetY);\n const rotatedPaddingOffset = paddingOffset.rotate(bearing * Math.PI / 180);\n const offsetAtInitialZoom = offset.add(rotatedPaddingOffset);\n const offsetAtFinalZoom = offsetAtInitialZoom.mult(tr.scale / tr.zoomScale(zoom));\n\n const center = tr.unproject(p0world.add(p1world).div(2).sub(offsetAtFinalZoom));\n\n return {\n center,\n zoom,\n bearing\n };\n }\n\n /**\n * Pans and zooms the map to contain its visible area within the specified geographical bounds.\n * This function will also reset the map's bearing to 0 if bearing is nonzero.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param bounds - Center these bounds in the viewport and use the highest\n * zoom level up to and including `Map#getMaxZoom()` that fits them in the viewport.\n * @param options- Options supports all properties from {@link AnimationOptions} and {@link CameraOptions} in addition to the fields below.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * let bbox = [[-79, 43], [-73, 45]];\n * map.fitBounds(bbox, {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n * @see [Fit a map to a bounding box](https://maplibre.org/maplibre-gl-js/docs/examples/fitbounds/)\n */\n fitBounds(bounds: LngLatBoundsLike, options?: FitBoundsOptions, eventData?: any): this {\n return this._fitInternal(\n this.cameraForBounds(bounds, options),\n options,\n eventData);\n }\n\n /**\n * Pans, rotates and zooms the map to to fit the box made by points p0 and p1\n * once the map is rotated to the specified bearing. To zoom without rotating,\n * pass in the current map bearing.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend` and `rotate`.\n *\n * @param p0 - First point on screen, in pixel coordinates\n * @param p1 - Second point on screen, in pixel coordinates\n * @param bearing - Desired map bearing at end of animation, in degrees\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * let p0 = [220, 400];\n * let p1 = [500, 900];\n * map.fitScreenCoordinates(p0, p1, map.getBearing(), {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n * @see Used by {@link BoxZoomHandler}\n */\n fitScreenCoordinates(p0: PointLike, p1: PointLike, bearing: number, options?: FitBoundsOptions, eventData?: any): this {\n return this._fitInternal(\n this._cameraForBoxAndBearing(\n this.transform.pointLocation(Point.convert(p0)),\n this.transform.pointLocation(Point.convert(p1)),\n bearing,\n options),\n options,\n eventData);\n }\n\n _fitInternal(calculatedOptions?: CenterZoomBearing, options?: FitBoundsOptions, eventData?: any): this {\n // cameraForBounds warns + returns undefined if unable to fit:\n if (!calculatedOptions) return this;\n\n options = extend(calculatedOptions, options);\n // Explicitly remove the padding field because, calculatedOptions already accounts for padding by setting zoom and center accordingly.\n delete options.padding;\n\n return options.linear ?\n this.easeTo(options, eventData) :\n this.flyTo(options, eventData);\n }\n\n /**\n * Changes any combination of center, zoom, bearing, and pitch, without\n * an animated transition. The map will retain its current values for any\n * details not specified in `options`.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`,\n * `pitch`, `pitchend`, and `rotate`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * // jump to coordinates at current zoom\n * map.jumpTo({center: [0, 0]});\n * // jump with zoom, pitch, and bearing options\n * map.jumpTo({\n * center: [0, 0],\n * zoom: 8,\n * pitch: 45,\n * bearing: 90\n * });\n * ```\n * @see [Jump to a series of locations](https://maplibre.org/maplibre-gl-js/docs/examples/jump-to/)\n * @see [Update a feature in realtime](https://maplibre.org/maplibre-gl-js/docs/examples/live-update-feature/)\n */\n jumpTo(options: JumpToOptions, eventData?: any): this {\n this.stop();\n\n const tr = this._getTransformForUpdate();\n let zoomChanged = false,\n bearingChanged = false,\n pitchChanged = false;\n\n if ('zoom' in options && tr.zoom !== +options.zoom) {\n zoomChanged = true;\n tr.zoom = +options.zoom;\n }\n\n if (options.center !== undefined) {\n tr.center = LngLat.convert(options.center);\n }\n\n if ('bearing' in options && tr.bearing !== +options.bearing) {\n bearingChanged = true;\n tr.bearing = +options.bearing;\n }\n\n if ('pitch' in options && tr.pitch !== +options.pitch) {\n pitchChanged = true;\n tr.pitch = +options.pitch;\n }\n\n if (options.padding != null && !tr.isPaddingEqual(options.padding)) {\n tr.padding = options.padding;\n }\n this._applyUpdatedTransform(tr);\n\n this.fire(new Event('movestart', eventData))\n .fire(new Event('move', eventData));\n\n if (zoomChanged) {\n this.fire(new Event('zoomstart', eventData))\n .fire(new Event('zoom', eventData))\n .fire(new Event('zoomend', eventData));\n }\n\n if (bearingChanged) {\n this.fire(new Event('rotatestart', eventData))\n .fire(new Event('rotate', eventData))\n .fire(new Event('rotateend', eventData));\n }\n\n if (pitchChanged) {\n this.fire(new Event('pitchstart', eventData))\n .fire(new Event('pitch', eventData))\n .fire(new Event('pitchend', eventData));\n }\n\n return this.fire(new Event('moveend', eventData));\n }\n\n /**\n * Calculates pitch, zoom and bearing for looking at `newCenter` with the camera position being `newCenter`\n * and returns them as {@link CameraOptions}.\n * @param from - The camera to look from\n * @param altitudeFrom - The altitude of the camera to look from\n * @param to - The center to look at\n * @param altitudeTo - Optional altitude of the center to look at. If none given the ground height will be used.\n * @returns the calculated camera options\n */\n calculateCameraOptionsFromTo(from: LngLat, altitudeFrom: number, to: LngLat, altitudeTo: number = 0): CameraOptions {\n const fromMerc = MercatorCoordinate.fromLngLat(from, altitudeFrom);\n const toMerc = MercatorCoordinate.fromLngLat(to, altitudeTo);\n const dx = toMerc.x - fromMerc.x;\n const dy = toMerc.y - fromMerc.y;\n const dz = toMerc.z - fromMerc.z;\n\n const distance3D = Math.hypot(dx, dy, dz);\n if (distance3D === 0) throw new Error('Can\\'t calculate camera options with same From and To');\n\n const groundDistance = Math.hypot(dx, dy);\n\n const zoom = this.transform.scaleZoom(this.transform.cameraToCenterDistance / distance3D / this.transform.tileSize);\n const bearing = (Math.atan2(dx, -dy) * 180) / Math.PI;\n let pitch = (Math.acos(groundDistance / distance3D) * 180) / Math.PI;\n pitch = dz < 0 ? 90 - pitch : 90 + pitch;\n\n return {\n center: toMerc.toLngLat(),\n zoom,\n pitch,\n bearing\n };\n }\n\n /**\n * Changes any combination of `center`, `zoom`, `bearing`, `pitch`, and `padding` with an animated transition\n * between old and new values. The map will retain its current values for any\n * details not specified in `options`.\n *\n * Note: The transition will happen instantly if the user has enabled\n * the `reduced motion` accessibility feature enabled in their operating system,\n * unless `options` includes `essential: true`.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`,\n * `pitch`, `pitchend`, and `rotate`.\n *\n * @param options - Options describing the destination and animation of the transition.\n * Accepts {@link CameraOptions} and {@link AnimationOptions}.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/)\n */\n easeTo(options: EaseToOptions & {\n easeId?: string;\n noMoveStart?: boolean;\n }, eventData?: any): this {\n this._stop(false, options.easeId);\n\n options = extend({\n offset: [0, 0],\n duration: 500,\n easing: defaultEasing\n }, options);\n\n if (options.animate === false || (!options.essential && browser.prefersReducedMotion)) options.duration = 0;\n\n const tr = this._getTransformForUpdate(),\n startZoom = this.getZoom(),\n startBearing = this.getBearing(),\n startPitch = this.getPitch(),\n startPadding = this.getPadding(),\n\n zoom = 'zoom' in options ? +options.zoom : startZoom,\n bearing = 'bearing' in options ? this._normalizeBearing(options.bearing, startBearing) : startBearing,\n pitch = 'pitch' in options ? +options.pitch : startPitch,\n padding = 'padding' in options ? options.padding : tr.padding;\n\n const offsetAsPoint = Point.convert(options.offset);\n let pointAtOffset = tr.centerPoint.add(offsetAsPoint);\n const locationAtOffset = tr.pointLocation(pointAtOffset);\n const center = LngLat.convert(options.center || locationAtOffset);\n this._normalizeCenter(center);\n\n const from = tr.project(locationAtOffset);\n const delta = tr.project(center).sub(from);\n const finalScale = tr.zoomScale(zoom - startZoom);\n\n let around, aroundPoint;\n\n if (options.around) {\n around = LngLat.convert(options.around);\n aroundPoint = tr.locationPoint(around);\n }\n\n const currently = {\n moving: this._moving,\n zooming: this._zooming,\n rotating: this._rotating,\n pitching: this._pitching\n };\n\n this._zooming = this._zooming || (zoom !== startZoom);\n this._rotating = this._rotating || (startBearing !== bearing);\n this._pitching = this._pitching || (pitch !== startPitch);\n this._padding = !tr.isPaddingEqual(padding as PaddingOptions);\n\n this._easeId = options.easeId;\n this._prepareEase(eventData, options.noMoveStart, currently);\n if (this.terrain) this._prepareElevation(center);\n\n this._ease((k) => {\n if (this._zooming) {\n tr.zoom = interpolates.number(startZoom, zoom, k);\n }\n if (this._rotating) {\n tr.bearing = interpolates.number(startBearing, bearing, k);\n }\n if (this._pitching) {\n tr.pitch = interpolates.number(startPitch, pitch, k);\n }\n if (this._padding) {\n tr.interpolatePadding(startPadding, padding as PaddingOptions, k);\n // When padding is being applied, Transform#centerPoint is changing continuously,\n // thus we need to recalculate offsetPoint every frame\n pointAtOffset = tr.centerPoint.add(offsetAsPoint);\n }\n\n if (this.terrain && !options.freezeElevation) this._updateElevation(k);\n\n if (around) {\n tr.setLocationAtPoint(around, aroundPoint);\n } else {\n const scale = tr.zoomScale(tr.zoom - startZoom);\n const base = zoom > startZoom ?\n Math.min(2, finalScale) :\n Math.max(0.5, finalScale);\n const speedup = Math.pow(base, 1 - k);\n const newCenter = tr.unproject(from.add(delta.mult(k * speedup)).mult(scale));\n tr.setLocationAtPoint(tr.renderWorldCopies ? newCenter.wrap() : newCenter, pointAtOffset);\n }\n\n this._applyUpdatedTransform(tr);\n\n this._fireMoveEvents(eventData);\n\n }, (interruptingEaseId?: string) => {\n if (this.terrain) this._finalizeElevation();\n this._afterEase(eventData, interruptingEaseId);\n }, options as any);\n\n return this;\n }\n\n _prepareEase(eventData: any, noMoveStart: boolean, currently: any = {}) {\n this._moving = true;\n if (!noMoveStart && !currently.moving) {\n this.fire(new Event('movestart', eventData));\n }\n if (this._zooming && !currently.zooming) {\n this.fire(new Event('zoomstart', eventData));\n }\n if (this._rotating && !currently.rotating) {\n this.fire(new Event('rotatestart', eventData));\n }\n if (this._pitching && !currently.pitching) {\n this.fire(new Event('pitchstart', eventData));\n }\n }\n\n _prepareElevation(center: LngLat) {\n this._elevationCenter = center;\n this._elevationStart = this.transform.elevation;\n this._elevationTarget = this.terrain.getElevationForLngLatZoom(center, this.transform.tileZoom);\n this._elevationFreeze = true;\n }\n\n _updateElevation(k: number) {\n this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this._elevationCenter, this.transform.tileZoom);\n const elevation = this.terrain.getElevationForLngLatZoom(this._elevationCenter, this.transform.tileZoom);\n // target terrain updated during flight, slowly move camera to new height\n if (k < 1 && elevation !== this._elevationTarget) {\n const pitch1 = this._elevationTarget - this._elevationStart;\n const pitch2 = (elevation - (pitch1 * k + this._elevationStart)) / (1 - k);\n this._elevationStart += k * (pitch1 - pitch2);\n this._elevationTarget = elevation;\n }\n this.transform.elevation = interpolates.number(this._elevationStart, this._elevationTarget, k);\n }\n\n _finalizeElevation() {\n this._elevationFreeze = false;\n this.transform.recalculateZoom(this.terrain);\n }\n\n /**\n * @internal\n * Called when the camera is about to be manipulated.\n * If `transformCameraUpdate` is specified, a copy of the current transform is created to track the accumulated changes.\n * This underlying transform represents the \"desired state\" proposed by input handlers / animations / UI controls.\n * It may differ from the state used for rendering (`this.transform`).\n * @returns Transform to apply changes to\n */\n _getTransformForUpdate(): Transform {\n if (!this.transformCameraUpdate) return this.transform;\n\n if (!this._requestedCameraState) {\n this._requestedCameraState = this.transform.clone();\n }\n return this._requestedCameraState;\n }\n\n /**\n * @internal\n * Called after the camera is done being manipulated.\n * @param tr - the requested camera end state\n * Call `transformCameraUpdate` if present, and then apply the \"approved\" changes.\n */\n _applyUpdatedTransform(tr: Transform) {\n if (!this.transformCameraUpdate) return;\n\n const nextTransform = tr.clone();\n const {\n center,\n zoom,\n pitch,\n bearing,\n elevation\n } = this.transformCameraUpdate(nextTransform);\n if (center) nextTransform.center = center;\n if (zoom !== undefined) nextTransform.zoom = zoom;\n if (pitch !== undefined) nextTransform.pitch = pitch;\n if (bearing !== undefined) nextTransform.bearing = bearing;\n if (elevation !== undefined) nextTransform.elevation = elevation;\n this.transform.apply(nextTransform);\n }\n\n _fireMoveEvents(eventData?: any) {\n this.fire(new Event('move', eventData));\n if (this._zooming) {\n this.fire(new Event('zoom', eventData));\n }\n if (this._rotating) {\n this.fire(new Event('rotate', eventData));\n }\n if (this._pitching) {\n this.fire(new Event('pitch', eventData));\n }\n }\n\n _afterEase(eventData?: any, easeId?: string) {\n // if this easing is being stopped to start another easing with\n // the same id then don't fire any events to avoid extra start/stop events\n if (this._easeId && easeId && this._easeId === easeId) {\n return;\n }\n delete this._easeId;\n\n const wasZooming = this._zooming;\n const wasRotating = this._rotating;\n const wasPitching = this._pitching;\n this._moving = false;\n this._zooming = false;\n this._rotating = false;\n this._pitching = false;\n this._padding = false;\n\n if (wasZooming) {\n this.fire(new Event('zoomend', eventData));\n }\n if (wasRotating) {\n this.fire(new Event('rotateend', eventData));\n }\n if (wasPitching) {\n this.fire(new Event('pitchend', eventData));\n }\n this.fire(new Event('moveend', eventData));\n }\n\n /**\n * Changes any combination of center, zoom, bearing, and pitch, animating the transition along a curve that\n * evokes flight. The animation seamlessly incorporates zooming and panning to help\n * the user maintain her bearings even after traversing a great distance.\n *\n * Note: The animation will be skipped, and this will behave equivalently to `jumpTo`\n * if the user has the `reduced motion` accessibility feature enabled in their operating system,\n * unless 'options' includes `essential: true`.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`,\n * `pitch`, `pitchend`, and `rotate`.\n *\n * @param options - Options describing the destination and animation of the transition.\n * Accepts {@link CameraOptions}, {@link AnimationOptions},\n * and the following additional options.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * // fly with default options to null island\n * map.flyTo({center: [0, 0], zoom: 9});\n * // using flyTo options\n * map.flyTo({\n * center: [0, 0],\n * zoom: 9,\n * speed: 0.2,\n * curve: 1,\n * easing(t) {\n * return t;\n * }\n * });\n * ```\n * @see [Fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto/)\n * @see [Slowly fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto-options/)\n * @see [Fly to a location based on scroll position](https://maplibre.org/maplibre-gl-js/docs/examples/scroll-fly-to/)\n */\n flyTo(options: FlyToOptions, eventData?: any): this {\n // Fall through to jumpTo if user has set prefers-reduced-motion\n if (!options.essential && browser.prefersReducedMotion) {\n const coercedOptions = pick(options, ['center', 'zoom', 'bearing', 'pitch', 'around']) as CameraOptions;\n return this.jumpTo(coercedOptions, eventData);\n }\n\n // This method implements an “optimal path” animation, as detailed in:\n //\n // Van Wijk, Jarke J.; Nuij, Wim A. A. “Smooth and efficient zooming and panning.” INFOVIS\n // ’03. pp. 15–22. .\n //\n // Where applicable, local variable documentation begins with the associated variable or\n // function in van Wijk (2003).\n\n this.stop();\n\n options = extend({\n offset: [0, 0],\n speed: 1.2,\n curve: 1.42,\n easing: defaultEasing\n }, options);\n\n const tr = this._getTransformForUpdate(),\n startZoom = this.getZoom(),\n startBearing = this.getBearing(),\n startPitch = this.getPitch(),\n startPadding = this.getPadding();\n\n const zoom = 'zoom' in options ? clamp(+options.zoom, tr.minZoom, tr.maxZoom) : startZoom;\n const bearing = 'bearing' in options ? this._normalizeBearing(options.bearing, startBearing) : startBearing;\n const pitch = 'pitch' in options ? +options.pitch : startPitch;\n const padding = 'padding' in options ? options.padding : tr.padding;\n\n const scale = tr.zoomScale(zoom - startZoom);\n const offsetAsPoint = Point.convert(options.offset);\n let pointAtOffset = tr.centerPoint.add(offsetAsPoint);\n const locationAtOffset = tr.pointLocation(pointAtOffset);\n const center = LngLat.convert(options.center || locationAtOffset);\n this._normalizeCenter(center);\n\n const from = tr.project(locationAtOffset);\n const delta = tr.project(center).sub(from);\n\n let rho = options.curve;\n\n // w₀: Initial visible span, measured in pixels at the initial scale.\n const w0 = Math.max(tr.width, tr.height),\n // w₁: Final visible span, measured in pixels with respect to the initial scale.\n w1 = w0 / scale,\n // Length of the flight path as projected onto the ground plane, measured in pixels from\n // the world image origin at the initial scale.\n u1 = delta.mag();\n\n if ('minZoom' in options) {\n const minZoom = clamp(Math.min(options.minZoom, startZoom, zoom), tr.minZoom, tr.maxZoom);\n // wm: Maximum visible span, measured in pixels with respect to the initial\n // scale.\n const wMax = w0 / tr.zoomScale(minZoom - startZoom);\n rho = Math.sqrt(wMax / u1 * 2);\n }\n\n // ρ²\n const rho2 = rho * rho;\n\n /**\n * rᵢ: Returns the zoom-out factor at one end of the animation.\n *\n * @param descent - `true` for the descent, `false` for the ascent\n */\n function zoomOutFactor(descent: boolean) {\n const b = (w1 * w1 - w0 * w0 + (descent ? -1 : 1) * rho2 * rho2 * u1 * u1) / (2 * (descent ? w1 : w0) * rho2 * u1);\n return Math.log(Math.sqrt(b * b + 1) - b);\n }\n\n function sinh(n) { return (Math.exp(n) - Math.exp(-n)) / 2; }\n function cosh(n) { return (Math.exp(n) + Math.exp(-n)) / 2; }\n function tanh(n) { return sinh(n) / cosh(n); }\n\n // r₀: Zoom-out factor during ascent.\n const r0 = zoomOutFactor(false);\n\n // w(s): Returns the visible span on the ground, measured in pixels with respect to the\n // initial scale. Assumes an angular field of view of 2 arctan ½ ≈ 53°.\n let w: (_: number) => number = function (s) {\n return (cosh(r0) / cosh(r0 + rho * s));\n };\n\n // u(s): Returns the distance along the flight path as projected onto the ground plane,\n // measured in pixels from the world image origin at the initial scale.\n let u: (_: number) => number = function (s) {\n return w0 * ((cosh(r0) * tanh(r0 + rho * s) - sinh(r0)) / rho2) / u1;\n };\n\n // S: Total length of the flight path, measured in ρ-screenfuls.\n let S = (zoomOutFactor(true) - r0) / rho;\n\n // When u₀ = u₁, the optimal path doesn’t require both ascent and descent.\n if (Math.abs(u1) < 0.000001 || !isFinite(S)) {\n // Perform a more or less instantaneous transition if the path is too short.\n if (Math.abs(w0 - w1) < 0.000001) return this.easeTo(options, eventData);\n\n const k = w1 < w0 ? -1 : 1;\n S = Math.abs(Math.log(w1 / w0)) / rho;\n\n u = function() { return 0; };\n w = function(s) { return Math.exp(k * rho * s); };\n }\n\n if ('duration' in options) {\n options.duration = +options.duration;\n } else {\n const V = 'screenSpeed' in options ? +options.screenSpeed / rho : +options.speed;\n options.duration = 1000 * S / V;\n }\n\n if (options.maxDuration && options.duration > options.maxDuration) {\n options.duration = 0;\n }\n\n this._zooming = true;\n this._rotating = (startBearing !== bearing);\n this._pitching = (pitch !== startPitch);\n this._padding = !tr.isPaddingEqual(padding as PaddingOptions);\n\n this._prepareEase(eventData, false);\n if (this.terrain) this._prepareElevation(center);\n\n this._ease((k) => {\n // s: The distance traveled along the flight path, measured in ρ-screenfuls.\n const s = k * S;\n const scale = 1 / w(s);\n tr.zoom = k === 1 ? zoom : startZoom + tr.scaleZoom(scale);\n\n if (this._rotating) {\n tr.bearing = interpolates.number(startBearing, bearing, k);\n }\n if (this._pitching) {\n tr.pitch = interpolates.number(startPitch, pitch, k);\n }\n if (this._padding) {\n tr.interpolatePadding(startPadding, padding as PaddingOptions, k);\n // When padding is being applied, Transform#centerPoint is changing continuously,\n // thus we need to recalculate offsetPoint every frame\n pointAtOffset = tr.centerPoint.add(offsetAsPoint);\n }\n\n if (this.terrain && !options.freezeElevation) this._updateElevation(k);\n\n const newCenter = k === 1 ? center : tr.unproject(from.add(delta.mult(u(s))).mult(scale));\n tr.setLocationAtPoint(tr.renderWorldCopies ? newCenter.wrap() : newCenter, pointAtOffset);\n\n this._applyUpdatedTransform(tr);\n\n this._fireMoveEvents(eventData);\n\n }, () => {\n if (this.terrain) this._finalizeElevation();\n this._afterEase(eventData);\n }, options);\n\n return this;\n }\n\n isEasing() {\n return !!this._easeFrameId;\n }\n\n /**\n * Stops any animated transition underway.\n *\n * @returns `this`\n */\n stop(): this {\n return this._stop();\n }\n\n _stop(allowGestures?: boolean, easeId?: string): this {\n if (this._easeFrameId) {\n this._cancelRenderFrame(this._easeFrameId);\n delete this._easeFrameId;\n delete this._onEaseFrame;\n }\n\n if (this._onEaseEnd) {\n // The _onEaseEnd function might emit events which trigger new\n // animation, which sets a new _onEaseEnd. Ensure we don't delete\n // it unintentionally.\n const onEaseEnd = this._onEaseEnd;\n delete this._onEaseEnd;\n onEaseEnd.call(this, easeId);\n }\n if (!allowGestures) {\n const handlers = (this as any).handlers;\n if (handlers) handlers.stop(false);\n }\n return this;\n }\n\n _ease(frame: (_: number) => void,\n finish: () => void,\n options: {\n animate?: boolean;\n duration?: number;\n easing?: (_: number) => number;\n }) {\n if (options.animate === false || options.duration === 0) {\n frame(1);\n finish();\n } else {\n this._easeStart = browser.now();\n this._easeOptions = options;\n this._onEaseFrame = frame;\n this._onEaseEnd = finish;\n this._easeFrameId = this._requestRenderFrame(this._renderFrameCallback);\n }\n }\n\n // Callback for map._requestRenderFrame\n _renderFrameCallback = () => {\n const t = Math.min((browser.now() - this._easeStart) / this._easeOptions.duration, 1);\n this._onEaseFrame(this._easeOptions.easing(t));\n\n // if _stop is called during _onEaseFrame from _fireMoveEvents we should avoid a new _requestRenderFrame, checking it by ensuring _easeFrameId was not deleted\n if (t < 1 && this._easeFrameId) {\n this._easeFrameId = this._requestRenderFrame(this._renderFrameCallback);\n } else {\n this.stop();\n }\n };\n\n // convert bearing so that it's numerically close to the current one so that it interpolates properly\n _normalizeBearing(bearing: number, currentBearing: number) {\n bearing = wrap(bearing, -180, 180);\n const diff = Math.abs(bearing - currentBearing);\n if (Math.abs(bearing - 360 - currentBearing) < diff) bearing -= 360;\n if (Math.abs(bearing + 360 - currentBearing) < diff) bearing += 360;\n return bearing;\n }\n\n // If a path crossing the antimeridian would be shorter, extend the final coordinate so that\n // interpolating between the two endpoints will cross it.\n _normalizeCenter(center: LngLat) {\n const tr = this.transform;\n if (!tr.renderWorldCopies || tr.lngRange) return;\n\n const delta = center.lng - tr.center.lng;\n center.lng +=\n delta > 180 ? -360 :\n delta < -180 ? 360 : 0;\n }\n\n /**\n * Query the current elevation of location. It return null if terrain is not enabled. the elevation is in meters relative to mean sea-level\n * @param lngLatLike - [x,y] or LngLat coordinates of the location\n * @returns elevation in meters\n */\n queryTerrainElevation(lngLatLike: LngLatLike): number | null {\n if (!this.terrain) {\n return null;\n }\n const elevation = this.terrain.getElevationForLngLatZoom(LngLat.convert(lngLatLike), this.transform.tileZoom);\n /**\n * Different zoomlevels with different terrain-tiles the elevation-values are not the same.\n * map.transform.elevation variable with the center-altitude.\n * In maplibre the proj-matrix is translated by this value in negative z-direction.\n * So we need to add this value to the elevation to get the correct value.\n */\n return elevation - this.transform.elevation;\n }\n}\n","import {DOM} from '../../util/dom';\n\nimport type {Map} from '../map';\nimport type {ControlPosition, IControl} from './control';\nimport type {MapDataEvent} from '../events';\nimport type {StyleSpecification} from '@maplibre/maplibre-gl-style-spec';\n/**\n * The {@link AttributionControl} options\n */\ntype AttributionOptions = {\n /**\n * If `true`, the attribution control will always collapse when moving the map. If `false`,\n * force the expanded attribution control. The default is a responsive attribution that collapses when the user moves the map on maps less than 640 pixels wide.\n * **Attribution should not be collapsed if it can comfortably fit on the map. `compact` should only be used to modify default attribution when map size makes it impossible to fit default attribution and when the automatic compact resizing for default settings are not sufficient.**\n */\n compact?: boolean;\n /**\n * Attributions to show in addition to any other attributions.\n */\n customAttribution?: string | Array;\n};\n\n/**\n * An `AttributionControl` control presents the map's attribution information. By default, the attribution control is expanded (regardless of map width).\n * @group Markers and Controls\n * @example\n * ```ts\n * let map = new maplibregl.Map({attributionControl: false})\n * .addControl(new maplibregl.AttributionControl({\n * compact: true\n * }));\n * ```\n */\nexport class AttributionControl implements IControl {\n options: AttributionOptions;\n _map: Map;\n _compact: boolean;\n _container: HTMLElement;\n _innerContainer: HTMLElement;\n _compactButton: HTMLElement;\n _editLink: HTMLAnchorElement;\n _attribHTML: string;\n styleId: string;\n styleOwner: string;\n\n /**\n * @param options - the attribution options\n */\n constructor(options: AttributionOptions = {}) {\n this.options = options;\n }\n\n getDefaultPosition(): ControlPosition {\n return 'bottom-right';\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._compact = this.options && this.options.compact;\n this._container = DOM.create('details', 'maplibregl-ctrl maplibregl-ctrl-attrib');\n this._compactButton = DOM.create('summary', 'maplibregl-ctrl-attrib-button', this._container);\n this._compactButton.addEventListener('click', this._toggleAttribution);\n this._setElementTitle(this._compactButton, 'ToggleAttribution');\n this._innerContainer = DOM.create('div', 'maplibregl-ctrl-attrib-inner', this._container);\n\n this._updateAttributions();\n this._updateCompact();\n\n this._map.on('styledata', this._updateData);\n this._map.on('sourcedata', this._updateData);\n this._map.on('terrain', this._updateData);\n this._map.on('resize', this._updateCompact);\n this._map.on('drag', this._updateCompactMinimize);\n\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._container);\n\n this._map.off('styledata', this._updateData);\n this._map.off('sourcedata', this._updateData);\n this._map.off('terrain', this._updateData);\n this._map.off('resize', this._updateCompact);\n this._map.off('drag', this._updateCompactMinimize);\n\n this._map = undefined;\n this._compact = undefined;\n this._attribHTML = undefined;\n }\n\n _setElementTitle(element: HTMLElement, title: string) {\n const str = this._map._getUIString(`AttributionControl.${title}`);\n element.title = str;\n element.setAttribute('aria-label', str);\n }\n\n _toggleAttribution = () => {\n if (this._container.classList.contains('maplibregl-compact')) {\n if (this._container.classList.contains('maplibregl-compact-show')) {\n this._container.setAttribute('open', '');\n this._container.classList.remove('maplibregl-compact-show');\n } else {\n this._container.classList.add('maplibregl-compact-show');\n this._container.removeAttribute('open');\n }\n }\n };\n\n _updateData = (e: MapDataEvent) => {\n if (e && (e.sourceDataType === 'metadata' || e.sourceDataType === 'visibility' || e.dataType === 'style' || e.type === 'terrain')) {\n this._updateAttributions();\n }\n };\n\n _updateAttributions() {\n if (!this._map.style) return;\n let attributions: Array = [];\n if (this.options.customAttribution) {\n if (Array.isArray(this.options.customAttribution)) {\n attributions = attributions.concat(\n this.options.customAttribution.map(attribution => {\n if (typeof attribution !== 'string') return '';\n return attribution;\n })\n );\n } else if (typeof this.options.customAttribution === 'string') {\n attributions.push(this.options.customAttribution);\n }\n }\n\n if (this._map.style.stylesheet) {\n const stylesheet = this._map.style.stylesheet as StyleSpecification & { owner: string; id: string };\n this.styleOwner = stylesheet.owner;\n this.styleId = stylesheet.id;\n }\n\n const sourceCaches = this._map.style.sourceCaches;\n for (const id in sourceCaches) {\n const sourceCache = sourceCaches[id];\n if (sourceCache.used || sourceCache.usedForTerrain) {\n const source = sourceCache.getSource();\n if (source.attribution && attributions.indexOf(source.attribution) < 0) {\n attributions.push(source.attribution);\n }\n }\n }\n\n // remove any entries that are whitespace\n attributions = attributions.filter(e => String(e).trim());\n\n // remove any entries that are substrings of another entry.\n // first sort by length so that substrings come first\n attributions.sort((a, b) => a.length - b.length);\n attributions = attributions.filter((attrib, i) => {\n for (let j = i + 1; j < attributions.length; j++) {\n if (attributions[j].indexOf(attrib) >= 0) { return false; }\n }\n return true;\n });\n\n // check if attribution string is different to minimize DOM changes\n const attribHTML = attributions.join(' | ');\n if (attribHTML === this._attribHTML) return;\n\n this._attribHTML = attribHTML;\n\n if (attributions.length) {\n this._innerContainer.innerHTML = attribHTML;\n this._container.classList.remove('maplibregl-attrib-empty');\n } else {\n this._container.classList.add('maplibregl-attrib-empty');\n }\n this._updateCompact();\n // remove old DOM node from _editLink\n this._editLink = null;\n }\n\n _updateCompact = () => {\n if (this._map.getCanvasContainer().offsetWidth <= 640 || this._compact) {\n if (this._compact === false) {\n this._container.setAttribute('open', '');\n } else if (!this._container.classList.contains('maplibregl-compact') && !this._container.classList.contains('maplibregl-attrib-empty')) {\n this._container.setAttribute('open', '');\n this._container.classList.add('maplibregl-compact', 'maplibregl-compact-show');\n }\n } else {\n this._container.setAttribute('open', '');\n if (this._container.classList.contains('maplibregl-compact')) {\n this._container.classList.remove('maplibregl-compact', 'maplibregl-compact-show');\n }\n }\n };\n\n _updateCompactMinimize = () => {\n if (this._container.classList.contains('maplibregl-compact')) {\n if (this._container.classList.contains('maplibregl-compact-show')) {\n this._container.classList.remove('maplibregl-compact-show');\n }\n }\n };\n}\n","import {DOM} from '../../util/dom';\n\nimport type {Map} from '../map';\nimport type {ControlPosition, IControl} from './control';\n\n/**\n * The {@link LogoControl} options object\n */\ntype LogoOptions = {\n /**\n * If `true`, force a compact logo.\n * If `false`, force the full logo. The default is a responsive logo that collapses when the map is less than 640 pixels wide.\n */\n compact?: boolean;\n};\n\n/**\n * A `LogoControl` is a control that adds the watermark.\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * map.addControl(new maplibregl.LogoControl({compact: false}));\n * ```\n **/\nexport class LogoControl implements IControl {\n options: LogoOptions;\n _map: Map;\n _compact: boolean;\n _container: HTMLElement;\n\n constructor(options: LogoOptions = {}) {\n this.options = options;\n }\n\n getDefaultPosition(): ControlPosition {\n return 'bottom-left';\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._compact = this.options && this.options.compact;\n this._container = DOM.create('div', 'maplibregl-ctrl');\n const anchor = DOM.create('a', 'maplibregl-ctrl-logo');\n anchor.target = '_blank';\n anchor.rel = 'noopener nofollow';\n anchor.href = 'https://maplibre.org/';\n anchor.setAttribute('aria-label', this._map._getUIString('LogoControl.Title'));\n anchor.setAttribute('rel', 'noopener nofollow');\n this._container.appendChild(anchor);\n this._container.style.display = 'block';\n\n this._map.on('resize', this._updateCompact);\n this._updateCompact();\n\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._container);\n this._map.off('resize', this._updateCompact);\n this._map = undefined;\n this._compact = undefined;\n }\n\n _updateCompact = () => {\n const containerChildren = this._container.children;\n if (containerChildren.length) {\n const anchor = containerChildren[0];\n if (this._map.getCanvasContainer().offsetWidth <= 640 || this._compact) {\n if (this._compact !== false) {\n anchor.classList.add('maplibregl-compact');\n }\n } else {\n anchor.classList.remove('maplibregl-compact');\n }\n }\n };\n\n}\n","export type TaskID = number; // can't mark opaque due to https://github.com/flowtype/flow-remove-types/pull/61\n\ntype Task = {\n callback: (timeStamp: number) => void;\n id: TaskID;\n cancelled: boolean;\n};\n\nexport class TaskQueue {\n _queue: Array;\n _id: TaskID;\n _cleared: boolean;\n _currentlyRunning: Array | false;\n\n constructor() {\n this._queue = [];\n this._id = 0;\n this._cleared = false;\n this._currentlyRunning = false;\n }\n\n add(callback: (timeStamp: number) => void): TaskID {\n const id = ++this._id;\n const queue = this._queue;\n queue.push({callback, id, cancelled: false});\n return id;\n }\n\n remove(id: TaskID) {\n const running = this._currentlyRunning;\n const queue = running ? this._queue.concat(running) : this._queue;\n for (const task of queue) {\n if (task.id === id) {\n task.cancelled = true;\n return;\n }\n }\n }\n\n run(timeStamp: number = 0) {\n if (this._currentlyRunning) throw new Error('Attempting to run(), but is already running.');\n const queue = this._currentlyRunning = this._queue;\n\n // Tasks queued by callbacks in the current queue should be executed\n // on the next run, not the current run.\n this._queue = [];\n\n for (const task of queue) {\n if (task.cancelled) continue;\n task.callback(timeStamp);\n if (this._cleared) break;\n }\n\n this._cleared = false;\n this._currentlyRunning = false;\n }\n\n clear() {\n if (this._currentlyRunning) {\n this._cleared = true;\n }\n this._queue = [];\n }\n}\n","export const defaultLocale = {\n 'AttributionControl.ToggleAttribution': 'Toggle attribution',\n 'AttributionControl.MapFeedback': 'Map feedback',\n 'FullscreenControl.Enter': 'Enter fullscreen',\n 'FullscreenControl.Exit': 'Exit fullscreen',\n 'GeolocateControl.FindMyLocation': 'Find my location',\n 'GeolocateControl.LocationNotAvailable': 'Location not available',\n 'LogoControl.Title': 'Mapbox logo',\n 'NavigationControl.ResetBearing': 'Reset bearing to north',\n 'NavigationControl.ZoomIn': 'Zoom in',\n 'NavigationControl.ZoomOut': 'Zoom out',\n 'ScaleControl.Feet': 'ft',\n 'ScaleControl.Meters': 'm',\n 'ScaleControl.Kilometers': 'km',\n 'ScaleControl.Miles': 'mi',\n 'ScaleControl.NauticalMiles': 'nm',\n 'TerrainControl.enableTerrain': 'Enable terrain',\n 'TerrainControl.disableTerrain': 'Disable terrain'\n};\n","import {createLayout} from '../util/struct_array';\n\nexport default createLayout([\n {name: 'a_pos3d', type: 'Int16', components: 3}\n]);\n","import {OverscaledTileID} from './tile_id';\nimport {Tile} from './tile';\nimport {EXTENT} from '../data/extent';\nimport {mat4} from 'gl-matrix';\nimport {Evented} from '../util/evented';\nimport type {Transform} from '../geo/transform';\nimport type {SourceCache} from '../source/source_cache';\nimport {Terrain} from '../render/terrain';\n\n/**\n * @internal\n * This class is a helper for the Terrain-class, it:\n * - loads raster-dem tiles\n * - manages all renderToTexture tiles.\n * - caches previous rendered tiles.\n * - finds all necessary renderToTexture tiles for a OverscaledTileID area\n * - finds the corresponding raster-dem tile for OverscaledTileID\n */\nexport class TerrainSourceCache extends Evented {\n /**\n * source-cache for the raster-dem source.\n */\n sourceCache: SourceCache;\n /**\n * stores all render-to-texture tiles.\n */\n _tiles: {[_: string]: Tile};\n /**\n * contains a list of tileID-keys for the current scene. (only for performance)\n */\n _renderableTilesKeys: Array;\n /**\n * raster-dem-tile for a TileID cache.\n */\n _sourceTileCache: {[_: string]: string};\n /**\n * minimum zoomlevel to render the terrain.\n */\n minzoom: number;\n /**\n * maximum zoomlevel to render the terrain.\n */\n maxzoom: number;\n /**\n * render-to-texture tileSize in scene.\n */\n tileSize: number;\n /**\n * raster-dem tiles will load for performance the actualZoom - deltaZoom zoom-level.\n */\n deltaZoom: number;\n\n constructor(sourceCache: SourceCache) {\n super();\n this.sourceCache = sourceCache;\n this._tiles = {};\n this._renderableTilesKeys = [];\n this._sourceTileCache = {};\n this.minzoom = 0;\n this.maxzoom = 22;\n this.tileSize = 512;\n this.deltaZoom = 1;\n sourceCache.usedForTerrain = true;\n sourceCache.tileSize = this.tileSize * 2 ** this.deltaZoom;\n }\n\n destruct() {\n this.sourceCache.usedForTerrain = false;\n this.sourceCache.tileSize = null;\n }\n\n /**\n * Load Terrain Tiles, create internal render-to-texture tiles, free GPU memory.\n * @param transform - the operation to do\n * @param terrain - the terrain\n */\n update(transform: Transform, terrain: Terrain): void {\n // load raster-dem tiles for the current scene.\n this.sourceCache.update(transform, terrain);\n // create internal render-to-texture tiles for the current scene.\n this._renderableTilesKeys = [];\n const keys = {};\n for (const tileID of transform.coveringTiles({\n tileSize: this.tileSize,\n minzoom: this.minzoom,\n maxzoom: this.maxzoom,\n reparseOverscaled: false,\n terrain\n })) {\n keys[tileID.key] = true;\n this._renderableTilesKeys.push(tileID.key);\n if (!this._tiles[tileID.key]) {\n tileID.posMatrix = new Float64Array(16) as any;\n mat4.ortho(tileID.posMatrix, 0, EXTENT, 0, EXTENT, 0, 1);\n this._tiles[tileID.key] = new Tile(tileID, this.tileSize);\n }\n }\n // free unused tiles\n for (const key in this._tiles) {\n if (!keys[key]) delete this._tiles[key];\n }\n }\n\n /**\n * Free render to texture cache\n * @param tileID - optional, free only corresponding to tileID.\n */\n freeRtt(tileID?: OverscaledTileID) {\n for (const key in this._tiles) {\n const tile = this._tiles[key];\n if (!tileID || tile.tileID.equals(tileID) || tile.tileID.isChildOf(tileID) || tileID.isChildOf(tile.tileID))\n tile.rtt = [];\n }\n }\n\n /**\n * get a list of tiles, which are loaded and should be rendered in the current scene\n * @returns the renderable tiles\n */\n getRenderableTiles(): Array {\n return this._renderableTilesKeys.map(key => this.getTileByID(key));\n }\n\n /**\n * get terrain tile by the TileID key\n * @param id - the tile id\n * @returns the tile\n */\n getTileByID(id: string): Tile {\n return this._tiles[id];\n }\n\n /**\n * Searches for the corresponding current renderable terrain-tiles\n * @param tileID - the tile to look for\n * @returns the tiles that were found\n */\n getTerrainCoords(tileID: OverscaledTileID): Record {\n const coords = {};\n for (const key of this._renderableTilesKeys) {\n const _tileID = this._tiles[key].tileID;\n if (_tileID.canonical.equals(tileID.canonical)) {\n const coord = tileID.clone();\n coord.posMatrix = new Float64Array(16) as any;\n mat4.ortho(coord.posMatrix, 0, EXTENT, 0, EXTENT, 0, 1);\n coords[key] = coord;\n } else if (_tileID.canonical.isChildOf(tileID.canonical)) {\n const coord = tileID.clone();\n coord.posMatrix = new Float64Array(16) as any;\n const dz = _tileID.canonical.z - tileID.canonical.z;\n const dx = _tileID.canonical.x - (_tileID.canonical.x >> dz << dz);\n const dy = _tileID.canonical.y - (_tileID.canonical.y >> dz << dz);\n const size = EXTENT >> dz;\n mat4.ortho(coord.posMatrix, 0, size, 0, size, 0, 1);\n mat4.translate(coord.posMatrix, coord.posMatrix, [-dx * size, -dy * size, 0]);\n coords[key] = coord;\n } else if (tileID.canonical.isChildOf(_tileID.canonical)) {\n const coord = tileID.clone();\n coord.posMatrix = new Float64Array(16) as any;\n const dz = tileID.canonical.z - _tileID.canonical.z;\n const dx = tileID.canonical.x - (tileID.canonical.x >> dz << dz);\n const dy = tileID.canonical.y - (tileID.canonical.y >> dz << dz);\n const size = EXTENT >> dz;\n mat4.ortho(coord.posMatrix, 0, EXTENT, 0, EXTENT, 0, 1);\n mat4.translate(coord.posMatrix, coord.posMatrix, [dx * size, dy * size, 0]);\n mat4.scale(coord.posMatrix, coord.posMatrix, [1 / (2 ** dz), 1 / (2 ** dz), 0]);\n coords[key] = coord;\n }\n }\n return coords;\n }\n\n /**\n * find the covering raster-dem tile\n * @param tileID - the tile to look for\n * @param searchForDEM - Optinal parameter to search for (parent) souretiles with loaded dem.\n * @returns the tile\n */\n getSourceTile(tileID: OverscaledTileID, searchForDEM?: boolean): Tile {\n const source = this.sourceCache._source;\n let z = tileID.overscaledZ - this.deltaZoom;\n if (z > source.maxzoom) z = source.maxzoom;\n if (z < source.minzoom) return null;\n // cache for tileID to terrain-tileID\n if (!this._sourceTileCache[tileID.key])\n this._sourceTileCache[tileID.key] = tileID.scaledTo(z).key;\n let tile = this.sourceCache.getTileByID(this._sourceTileCache[tileID.key]);\n // during tile-loading phase look if parent tiles (with loaded dem) are available.\n if (!(tile && tile.dem) && searchForDEM)\n while (z >= source.minzoom && !(tile && tile.dem))\n tile = this.sourceCache.getTileByID(tileID.scaledTo(z--).key);\n return tile;\n }\n\n /**\n * get a list of tiles, loaded after a spezific time. This is used to update depth & coords framebuffers.\n * @param time - the time\n * @returns the relevant tiles\n */\n tilesAfterTime(time = Date.now()): Array {\n return Object.values(this._tiles).filter(t => t.timeAdded >= time);\n }\n}\n","\nimport {Tile} from '../source/tile';\nimport {mat4, vec2} from 'gl-matrix';\nimport {OverscaledTileID} from '../source/tile_id';\nimport {RGBAImage} from '../util/image';\nimport {warnOnce} from '../util/util';\nimport {Pos3dArray, TriangleIndexArray} from '../data/array_types.g';\nimport pos3dAttributes from '../data/pos3d_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {VertexBuffer} from '../gl/vertex_buffer';\nimport {IndexBuffer} from '../gl/index_buffer';\nimport {Painter} from './painter';\nimport {Texture} from '../render/texture';\nimport type {Framebuffer} from '../gl/framebuffer';\nimport Point from '@mapbox/point-geometry';\nimport {MercatorCoordinate, lngFromMercatorX, mercatorXfromLng} from '../geo/mercator_coordinate';\nimport {TerrainSourceCache} from '../source/terrain_source_cache';\nimport {SourceCache} from '../source/source_cache';\nimport {EXTENT} from '../data/extent';\nimport type {TerrainSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport {LngLat, earthRadius} from '../geo/lng_lat';\n\n/**\n * @internal\n * A terrain GPU related object\n */\nexport type TerrainData = {\n 'u_depth': number;\n 'u_terrain': number;\n 'u_terrain_dim': number;\n 'u_terrain_matrix': mat4;\n 'u_terrain_unpack': number[];\n 'u_terrain_exaggeration': number;\n texture: WebGLTexture;\n depthTexture: WebGLTexture;\n tile: Tile;\n}\n\n/**\n * @internal\n * A terrain mesh object\n */\nexport type TerrainMesh = {\n indexBuffer: IndexBuffer;\n vertexBuffer: VertexBuffer;\n segments: SegmentVector;\n}\n\n/**\n * @internal\n * This is the main class which handles most of the 3D Terrain logic. It has the following topics:\n * 1) loads raster-dem tiles via the internal sourceCache this.sourceCache\n * 2) creates a depth-framebuffer, which is used to calculate the visibility of coordinates\n * 3) creates a coords-framebuffer, which is used the get to tile-coordinate for a screen-pixel\n * 4) stores all render-to-texture tiles in the this.sourceCache._tiles\n * 5) calculates the elevation for a specific tile-coordinate\n * 6) creates a terrain-mesh\n *\n * A note about the GPU resource-usage:\n * Framebuffers:\n * - one for the depth & coords framebuffer with the size of the map-div.\n * - one for rendering a tile to texture with the size of tileSize (= 512x512).\n * Textures:\n * - one texture for an empty raster-dem tile with size 1x1\n * - one texture for an empty depth-buffer, when terrain is disabled with size 1x1\n * - one texture for an each loaded raster-dem with size of the source.tileSize\n * - one texture for the coords-framebuffer with the size of the map-div.\n * - one texture for the depth-framebuffer with the size of the map-div.\n * - one texture for the encoded tile-coords with the size 2*tileSize (=1024x1024)\n * - finally for each render-to-texture tile (= this._tiles) a set of textures\n * for each render stack (The stack-concept is documented in painter.ts).\n * Normally there exists 1-3 Textures per tile, depending on the stylesheet.\n * Each Textures has the size 2*tileSize (= 1024x1024). Also there exists a\n * cache of the last 150 newest rendered tiles.\n *\n */\nexport class Terrain {\n /**\n * The style this terrain crresponds to\n */\n painter: Painter;\n /**\n * the sourcecache this terrain is based on\n */\n sourceCache: TerrainSourceCache;\n /**\n * the TerrainSpecification object passed to this instance\n */\n options: TerrainSpecification;\n /**\n * define the meshSize per tile.\n */\n meshSize: number;\n /**\n * multiplicator for the elevation. Used to make terrain more \"extreme\".\n */\n exaggeration: number;\n /**\n * to not see pixels in the render-to-texture tiles it is good to render them bigger\n * this number is the multiplicator (must be a power of 2) for the current tileSize.\n * So to get good results with not too much memory footprint a value of 2 should be fine.\n */\n qualityFactor: number;\n /**\n * holds the framebuffer object in size of the screen to render the coords & depth into a texture.\n */\n _fbo: Framebuffer;\n _fboCoordsTexture: Texture;\n _fboDepthTexture: Texture;\n _emptyDepthTexture: Texture;\n /**\n * GL Objects for the terrain-mesh\n * The mesh is a regular mesh, which has the advantage that it can be reused for all tiles.\n */\n _mesh: TerrainMesh;\n /**\n * coords index contains a list of tileID.keys. This index is used to identify\n * the tile via the alpha-cannel in the coords-texture.\n * As the alpha-channel has 1 Byte a max of 255 tiles can rendered without an error.\n */\n coordsIndex: Array;\n /**\n * tile-coords encoded in the rgb channel, _coordsIndex is in the alpha-channel.\n */\n _coordsTexture: Texture;\n /**\n * accuracy of the coords. 2 * tileSize should be enoughth.\n */\n _coordsTextureSize: number;\n /**\n * variables for an empty dem texture, which is used while the raster-dem tile is loading.\n */\n _emptyDemUnpack: number[];\n _emptyDemTexture: Texture;\n _emptyDemMatrix: mat4;\n /**\n * as of overzooming of raster-dem tiles in high zoomlevels, this cache contains\n * matrices to transform from vector-tile coords to raster-dem-tile coords.\n */\n _demMatrixCache: {[_: string]: { matrix: mat4; coord: OverscaledTileID }};\n\n constructor(painter: Painter, sourceCache: SourceCache, options: TerrainSpecification) {\n this.painter = painter;\n this.sourceCache = new TerrainSourceCache(sourceCache);\n this.options = options;\n this.exaggeration = typeof options.exaggeration === 'number' ? options.exaggeration : 1.0;\n this.qualityFactor = 2;\n this.meshSize = 128;\n this._demMatrixCache = {};\n this.coordsIndex = [];\n this._coordsTextureSize = 1024;\n }\n\n /**\n * get the elevation-value from original dem-data for a given tile-coordinate\n * @param tileID - the tile to get elevation for\n * @param x - between 0 .. EXTENT\n * @param y - between 0 .. EXTENT\n * @param extent - optional, default 8192\n * @returns the elevation\n */\n getDEMElevation(tileID: OverscaledTileID, x: number, y: number, extent: number = EXTENT): number {\n if (!(x >= 0 && x < extent && y >= 0 && y < extent)) return 0;\n const terrain = this.getTerrainData(tileID);\n const dem = terrain.tile?.dem;\n if (!dem)\n return 0;\n\n const pos = vec2.transformMat4([] as any, [x / extent * EXTENT, y / extent * EXTENT], terrain.u_terrain_matrix);\n const coord = [pos[0] * dem.dim, pos[1] * dem.dim];\n\n // bilinear interpolation\n const cx = Math.floor(coord[0]),\n cy = Math.floor(coord[1]),\n tx = coord[0] - cx,\n ty = coord[1] - cy;\n return (\n dem.get(cx, cy) * (1 - tx) * (1 - ty) +\n dem.get(cx + 1, cy) * (tx) * (1 - ty) +\n dem.get(cx, cy + 1) * (1 - tx) * (ty) +\n dem.get(cx + 1, cy + 1) * (tx) * (ty)\n );\n }\n\n /**\n * Get the elevation for given {@link LngLat} in respect of exaggeration.\n * @param lnglat - the location\n * @param zoom - the zoom\n * @returns the elevation\n */\n getElevationForLngLatZoom(lnglat: LngLat, zoom: number) {\n const {tileID, mercatorX, mercatorY} = this._getOverscaledTileIDFromLngLatZoom(lnglat, zoom);\n return this.getElevation(tileID, mercatorX % EXTENT, mercatorY % EXTENT, EXTENT);\n }\n\n /**\n * Get the elevation for given coordinate in respect of exaggeration.\n * @param tileID - the tile id\n * @param x - between 0 .. EXTENT\n * @param y - between 0 .. EXTENT\n * @param extent - optional, default 8192\n * @returns the elevation\n */\n getElevation(tileID: OverscaledTileID, x: number, y: number, extent: number = EXTENT): number {\n return this.getDEMElevation(tileID, x, y, extent) * this.exaggeration;\n }\n\n /**\n * returns a Terrain Object for a tile. Unless the tile corresponds to data (e.g. tile is loading), return a flat dem object\n * @param tileID - the tile to get the terrain for\n * @returns the terrain data to use in the program\n */\n getTerrainData(tileID: OverscaledTileID): TerrainData {\n // create empty DEM Objects, which will used while raster-dem tiles are loading.\n // creates an empty depth-buffer texture which is needed, during the initialization process of the 3d mesh..\n if (!this._emptyDemTexture) {\n const context = this.painter.context;\n const image = new RGBAImage({width: 1, height: 1}, new Uint8Array(1 * 4));\n this._emptyDepthTexture = new Texture(context, image, context.gl.RGBA, {premultiply: false});\n this._emptyDemUnpack = [0, 0, 0, 0];\n this._emptyDemTexture = new Texture(context, new RGBAImage({width: 1, height: 1}), context.gl.RGBA, {premultiply: false});\n this._emptyDemTexture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE);\n this._emptyDemMatrix = mat4.identity([] as any);\n }\n // find covering dem tile and prepare demTexture\n const sourceTile = this.sourceCache.getSourceTile(tileID, true);\n if (sourceTile && sourceTile.dem && (!sourceTile.demTexture || sourceTile.needsTerrainPrepare)) {\n const context = this.painter.context;\n sourceTile.demTexture = this.painter.getTileTexture(sourceTile.dem.stride);\n if (sourceTile.demTexture) sourceTile.demTexture.update(sourceTile.dem.getPixels(), {premultiply: false});\n else sourceTile.demTexture = new Texture(context, sourceTile.dem.getPixels(), context.gl.RGBA, {premultiply: false});\n sourceTile.demTexture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE);\n sourceTile.needsTerrainPrepare = false;\n }\n // create matrix for lookup in dem data\n const matrixKey = sourceTile && (sourceTile + sourceTile.tileID.key) + tileID.key;\n if (matrixKey && !this._demMatrixCache[matrixKey]) {\n const maxzoom = this.sourceCache.sourceCache._source.maxzoom;\n let dz = tileID.canonical.z - sourceTile.tileID.canonical.z;\n if (tileID.overscaledZ > tileID.canonical.z) {\n if (tileID.canonical.z >= maxzoom) dz = tileID.canonical.z - maxzoom;\n else warnOnce('cannot calculate elevation if elevation maxzoom > source.maxzoom');\n }\n const dx = tileID.canonical.x - (tileID.canonical.x >> dz << dz);\n const dy = tileID.canonical.y - (tileID.canonical.y >> dz << dz);\n const demMatrix = mat4.fromScaling(new Float64Array(16) as any, [1 / (EXTENT << dz), 1 / (EXTENT << dz), 0]);\n mat4.translate(demMatrix, demMatrix, [dx * EXTENT, dy * EXTENT, 0]);\n this._demMatrixCache[tileID.key] = {matrix: demMatrix, coord: tileID};\n }\n // return uniform values & textures\n return {\n 'u_depth': 2,\n 'u_terrain': 3,\n 'u_terrain_dim': sourceTile && sourceTile.dem && sourceTile.dem.dim || 1,\n 'u_terrain_matrix': matrixKey ? this._demMatrixCache[tileID.key].matrix : this._emptyDemMatrix,\n 'u_terrain_unpack': sourceTile && sourceTile.dem && sourceTile.dem.getUnpackVector() || this._emptyDemUnpack,\n 'u_terrain_exaggeration': this.exaggeration,\n texture: (sourceTile && sourceTile.demTexture || this._emptyDemTexture).texture,\n depthTexture: (this._fboDepthTexture || this._emptyDepthTexture).texture,\n tile: sourceTile\n };\n }\n\n /**\n * get a framebuffer as big as the map-div, which will be used to render depth & coords into a texture\n * @param texture - the texture\n * @returns the frame buffer\n */\n getFramebuffer(texture: string): Framebuffer {\n const painter = this.painter;\n const width = painter.width / devicePixelRatio;\n const height = painter.height / devicePixelRatio;\n if (this._fbo && (this._fbo.width !== width || this._fbo.height !== height)) {\n this._fbo.destroy();\n this._fboCoordsTexture.destroy();\n this._fboDepthTexture.destroy();\n delete this._fbo;\n delete this._fboDepthTexture;\n delete this._fboCoordsTexture;\n }\n if (!this._fboCoordsTexture) {\n this._fboCoordsTexture = new Texture(painter.context, {width, height, data: null}, painter.context.gl.RGBA, {premultiply: false});\n this._fboCoordsTexture.bind(painter.context.gl.NEAREST, painter.context.gl.CLAMP_TO_EDGE);\n }\n if (!this._fboDepthTexture) {\n this._fboDepthTexture = new Texture(painter.context, {width, height, data: null}, painter.context.gl.RGBA, {premultiply: false});\n this._fboDepthTexture.bind(painter.context.gl.NEAREST, painter.context.gl.CLAMP_TO_EDGE);\n }\n if (!this._fbo) {\n this._fbo = painter.context.createFramebuffer(width, height, true, false);\n this._fbo.depthAttachment.set(painter.context.createRenderbuffer(painter.context.gl.DEPTH_COMPONENT16, width, height));\n }\n this._fbo.colorAttachment.set(texture === 'coords' ? this._fboCoordsTexture.texture : this._fboDepthTexture.texture);\n return this._fbo;\n }\n\n /**\n * create coords texture, needed to grab coordinates from canvas\n * encode coords coordinate into 4 bytes:\n * - 8 lower bits for x\n * - 8 lower bits for y\n * - 4 higher bits for x\n * - 4 higher bits for y\n * - 8 bits for coordsIndex (1 .. 255) (= number of terraintile), is later setted in draw_terrain uniform value\n * @returns the texture\n */\n getCoordsTexture(): Texture {\n const context = this.painter.context;\n if (this._coordsTexture) return this._coordsTexture;\n const data = new Uint8Array(this._coordsTextureSize * this._coordsTextureSize * 4);\n for (let y = 0, i = 0; y < this._coordsTextureSize; y++) for (let x = 0; x < this._coordsTextureSize; x++, i += 4) {\n data[i + 0] = x & 255;\n data[i + 1] = y & 255;\n data[i + 2] = ((x >> 8) << 4) | (y >> 8);\n data[i + 3] = 0;\n }\n const image = new RGBAImage({width: this._coordsTextureSize, height: this._coordsTextureSize}, new Uint8Array(data.buffer));\n const texture = new Texture(context, image, context.gl.RGBA, {premultiply: false});\n texture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE);\n this._coordsTexture = texture;\n return texture;\n }\n\n /**\n * Reads a pixel from the coords-framebuffer and translate this to mercator.\n * @param p - Screen-Coordinate\n * @returns mercator coordinate for a screen pixel\n */\n pointCoordinate(p: Point): MercatorCoordinate {\n const rgba = new Uint8Array(4);\n const context = this.painter.context, gl = context.gl;\n // grab coordinate pixel from coordinates framebuffer\n context.bindFramebuffer.set(this.getFramebuffer('coords').framebuffer);\n gl.readPixels(p.x, this.painter.height / devicePixelRatio - p.y - 1, 1, 1, gl.RGBA, gl.UNSIGNED_BYTE, rgba);\n context.bindFramebuffer.set(null);\n // decode coordinates (encoding see getCoordsTexture)\n const x = rgba[0] + ((rgba[2] >> 4) << 8);\n const y = rgba[1] + ((rgba[2] & 15) << 8);\n const tileID = this.coordsIndex[255 - rgba[3]];\n const tile = tileID && this.sourceCache.getTileByID(tileID);\n if (!tile) return null;\n const coordsSize = this._coordsTextureSize;\n const worldSize = (1 << tile.tileID.canonical.z) * coordsSize;\n const mercatorX = (tile.tileID.canonical.x * coordsSize + x) / worldSize;\n return new MercatorCoordinate(\n this._allowMercatorOverflow(p, mercatorX),\n (tile.tileID.canonical.y * coordsSize + y) / worldSize,\n this.getElevation(tile.tileID, x, y, coordsSize)\n );\n }\n\n /**\n * create a regular mesh which will be used by all terrain-tiles\n * @returns the created regular mesh\n */\n getTerrainMesh(): TerrainMesh {\n if (this._mesh) return this._mesh;\n const context = this.painter.context;\n const vertexArray = new Pos3dArray();\n const indexArray = new TriangleIndexArray();\n const meshSize = this.meshSize;\n const delta = EXTENT / meshSize;\n const meshSize2 = meshSize * meshSize;\n for (let y = 0; y <= meshSize; y++) for (let x = 0; x <= meshSize; x++)\n vertexArray.emplaceBack(x * delta, y * delta, 0);\n for (let y = 0; y < meshSize2; y += meshSize + 1) for (let x = 0; x < meshSize; x++) {\n indexArray.emplaceBack(x + y, meshSize + x + y + 1, meshSize + x + y + 2);\n indexArray.emplaceBack(x + y, meshSize + x + y + 2, x + y + 1);\n }\n // add an extra frame around the mesh to avoid stiching on tile boundaries with different zoomlevels\n // first code-block is for top-bottom frame and second for left-right frame\n const offsetTop = vertexArray.length, offsetBottom = offsetTop + (meshSize + 1) * 2;\n for (const y of [0, 1]) for (let x = 0; x <= meshSize; x++) for (const z of [0, 1])\n vertexArray.emplaceBack(x * delta, y * EXTENT, z);\n for (let x = 0; x < meshSize * 2; x += 2) {\n indexArray.emplaceBack(offsetBottom + x, offsetBottom + x + 1, offsetBottom + x + 3);\n indexArray.emplaceBack(offsetBottom + x, offsetBottom + x + 3, offsetBottom + x + 2);\n indexArray.emplaceBack(offsetTop + x, offsetTop + x + 3, offsetTop + x + 1);\n indexArray.emplaceBack(offsetTop + x, offsetTop + x + 2, offsetTop + x + 3);\n }\n const offsetLeft = vertexArray.length, offsetRight = offsetLeft + (meshSize + 1) * 2;\n for (const x of [0, 1]) for (let y = 0; y <= meshSize; y++) for (const z of [0, 1])\n vertexArray.emplaceBack(x * EXTENT, y * delta, z);\n for (let y = 0; y < meshSize * 2; y += 2) {\n indexArray.emplaceBack(offsetLeft + y, offsetLeft + y + 1, offsetLeft + y + 3);\n indexArray.emplaceBack(offsetLeft + y, offsetLeft + y + 3, offsetLeft + y + 2);\n indexArray.emplaceBack(offsetRight + y, offsetRight + y + 3, offsetRight + y + 1);\n indexArray.emplaceBack(offsetRight + y, offsetRight + y + 2, offsetRight + y + 3);\n }\n this._mesh = {\n indexBuffer: context.createIndexBuffer(indexArray),\n vertexBuffer: context.createVertexBuffer(vertexArray, pos3dAttributes.members),\n segments: SegmentVector.simpleSegment(0, 0, vertexArray.length, indexArray.length)\n };\n return this._mesh;\n }\n\n /**\n * Calculates a height of the frame around the terrain-mesh to avoid stiching between\n * tile boundaries in different zoomlevels.\n * @param zoom - current zoomlevel\n * @returns the elevation delta in meters\n */\n getMeshFrameDelta(zoom: number): number {\n // divide by 5 is evaluated by trial & error to get a frame in the right height\n return 2 * Math.PI * earthRadius / Math.pow(2, zoom) / 5;\n }\n\n getMinTileElevationForLngLatZoom(lnglat: LngLat, zoom: number) {\n const {tileID} = this._getOverscaledTileIDFromLngLatZoom(lnglat, zoom);\n return this.getMinMaxElevation(tileID).minElevation ?? 0;\n }\n\n /**\n * Get the minimum and maximum elevation contained in a tile. This includes any\n * exaggeration included in the terrain.\n *\n * @param tileID - ID of the tile to be used as a source for the min/max elevation\n * @returns the minimum and maximum elevation found in the tile, including the terrain's\n * exaggeration\n */\n getMinMaxElevation(tileID: OverscaledTileID): {minElevation: number | null; maxElevation: number | null} {\n const tile = this.getTerrainData(tileID).tile;\n const minMax = {minElevation: null, maxElevation: null};\n if (tile && tile.dem) {\n minMax.minElevation = tile.dem.min * this.exaggeration;\n minMax.maxElevation = tile.dem.max * this.exaggeration;\n }\n return minMax;\n }\n\n _getOverscaledTileIDFromLngLatZoom(lnglat: LngLat, zoom: number): { tileID: OverscaledTileID; mercatorX: number; mercatorY: number} {\n const mercatorCoordinate = MercatorCoordinate.fromLngLat(lnglat.wrap());\n const worldSize = (1 << zoom) * EXTENT;\n const mercatorX = mercatorCoordinate.x * worldSize;\n const mercatorY = mercatorCoordinate.y * worldSize;\n const tileX = Math.floor(mercatorX / EXTENT), tileY = Math.floor(mercatorY / EXTENT);\n const tileID = new OverscaledTileID(zoom, 0, zoom, tileX, tileY);\n return {\n tileID,\n mercatorX,\n mercatorY\n };\n }\n\n _allowMercatorOverflow(p: Point, mercatorX: number): number {\n const inLeftHalf = p.x < (this.painter.width / 2);\n let lng = lngFromMercatorX(mercatorX);\n const centerLng = this.painter.transform.center.lng;\n if (\n (inLeftHalf && Math.sign(lng) > 0 && Math.sign(centerLng) < 0) ||\n (!inLeftHalf && Math.sign(lng) < 0 && Math.sign(centerLng) > 0)\n ) {\n lng = 360 * Math.sign(centerLng) + lng;\n return mercatorXfromLng(lng);\n }\n return mercatorX;\n }\n}\n","import {Texture} from '../render/texture';\nimport {Context} from './context';\nimport {Framebuffer} from './framebuffer';\n\nexport type PoolObject = {\n id: number;\n fbo: Framebuffer;\n texture: Texture;\n stamp: number;\n inUse: boolean;\n};\n/**\n * @internal\n * `RenderPool` is a resource pool for textures and framebuffers\n */\nexport class RenderPool {\n private _objects: Array;\n /**\n * An index array of recently used pool objects.\n * Items that are used recently are last in the array\n */\n private _recentlyUsed: Array;\n private _stamp: number;\n\n constructor(\n private readonly _context: Context,\n private readonly _size: number,\n private readonly _tileSize: number) {\n this._objects = [];\n this._recentlyUsed = [];\n this._stamp = 0;\n }\n\n public destruct() {\n for (const obj of this._objects) {\n obj.texture.destroy();\n obj.fbo.destroy();\n }\n }\n\n private _createObject(id: number): PoolObject {\n const fbo = this._context.createFramebuffer(this._tileSize, this._tileSize, true, true);\n const texture = new Texture(this._context, {width: this._tileSize, height: this._tileSize, data: null}, this._context.gl.RGBA);\n texture.bind(this._context.gl.LINEAR, this._context.gl.CLAMP_TO_EDGE);\n fbo.depthAttachment.set(this._context.createRenderbuffer(this._context.gl.DEPTH_STENCIL, this._tileSize, this._tileSize));\n fbo.colorAttachment.set(texture.texture);\n return {id, fbo, texture, stamp: -1, inUse: false};\n }\n\n public getObjectForId(id: number): PoolObject {\n return this._objects[id];\n }\n\n public useObject(obj: PoolObject) {\n obj.inUse = true;\n this._recentlyUsed = this._recentlyUsed.filter(id => obj.id !== id);\n this._recentlyUsed.push(obj.id);\n }\n\n public stampObject(obj: PoolObject) {\n obj.stamp = ++this._stamp;\n }\n\n public getOrCreateFreeObject(): PoolObject {\n // check for free existing object\n for (const id of this._recentlyUsed) {\n if (!this._objects[id].inUse)\n return this._objects[id];\n }\n if (this._objects.length >= this._size)\n throw new Error('No free RenderPool available, call freeAllObjects() required!');\n // create new object\n const obj = this._createObject(this._objects.length);\n this._objects.push(obj);\n return obj;\n }\n\n public freeObject(obj: PoolObject) {\n obj.inUse = false;\n }\n\n public freeAllObjects() {\n for (const obj of this._objects)\n this.freeObject(obj);\n }\n\n public isFull(): boolean {\n if (this._objects.length < this._size) {\n return false;\n }\n return this._objects.some(o => !o.inUse) === false;\n }\n}\n","import {Painter} from './painter';\nimport {Tile} from '../source/tile';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {OverscaledTileID} from '../source/tile_id';\nimport {drawTerrain} from './draw_terrain';\nimport {Style} from '../style/style';\nimport {Terrain} from './terrain';\nimport {RenderPool} from '../gl/render_pool';\nimport {Texture} from './texture';\nimport type {StyleLayer} from '../style/style_layer';\n\n/**\n * lookup table which layers should rendered to texture\n */\nconst LAYERS: { [keyof in StyleLayer['type']]?: boolean } = {\n background: true,\n fill: true,\n line: true,\n raster: true,\n hillshade: true\n};\n\n/**\n * @internal\n * A helper class to help define what should be rendered to texture and how\n */\nexport class RenderToTexture {\n painter: Painter;\n terrain: Terrain;\n pool: RenderPool;\n /**\n * coordsDescendingInv contains a list of all tiles which should be rendered for one render-to-texture tile\n * e.g. render 4 raster-tiles with size 256px to the 512px render-to-texture tile\n */\n _coordsDescendingInv: {[_: string]: {[_:string]: Array}};\n /**\n * create a string representation of all to tiles rendered to render-to-texture tiles\n * this string representation is used to check if tile should be re-rendered.\n */\n _coordsDescendingInvStr: {[_: string]: {[_:string]: string}};\n /**\n * store for render-stacks\n * a render stack is a set of layers which should be rendered into one texture\n * every stylesheet can have multiple stacks. A new stack is created if layers which should\n * not rendered to texture sit inbetween layers which should rendered to texture. e.g. hillshading or symbols\n */\n _stacks: Array>;\n /**\n * remember the previous processed layer to check if a new stack is needed\n */\n _prevType: string;\n /**\n * a list of tiles that can potentially rendered\n */\n _renderableTiles: Array;\n /**\n * a list of tiles that should be rendered to screen in the next render-call\n */\n _rttTiles: Array;\n /**\n * a list of all layer-ids which should be rendered\n */\n _renderableLayerIds: Array;\n\n constructor(painter: Painter, terrain: Terrain) {\n this.painter = painter;\n this.terrain = terrain;\n this.pool = new RenderPool(painter.context, 30, terrain.sourceCache.tileSize * terrain.qualityFactor);\n }\n\n destruct() {\n this.pool.destruct();\n }\n\n getTexture(tile: Tile): Texture {\n return this.pool.getObjectForId(tile.rtt[this._stacks.length - 1].id).texture;\n }\n\n prepareForRender(style: Style, zoom: number) {\n this._stacks = [];\n this._prevType = null;\n this._rttTiles = [];\n this._renderableTiles = this.terrain.sourceCache.getRenderableTiles();\n this._renderableLayerIds = style._order.filter(id => !style._layers[id].isHidden(zoom));\n\n this._coordsDescendingInv = {};\n for (const id in style.sourceCaches) {\n this._coordsDescendingInv[id] = {};\n const tileIDs = style.sourceCaches[id].getVisibleCoordinates();\n for (const tileID of tileIDs) {\n const keys = this.terrain.sourceCache.getTerrainCoords(tileID);\n for (const key in keys) {\n if (!this._coordsDescendingInv[id][key]) this._coordsDescendingInv[id][key] = [];\n this._coordsDescendingInv[id][key].push(keys[key]);\n }\n }\n }\n\n this._coordsDescendingInvStr = {};\n for (const id of style._order) {\n const layer = style._layers[id], source = layer.source;\n if (LAYERS[layer.type]) {\n if (!this._coordsDescendingInvStr[source]) {\n this._coordsDescendingInvStr[source] = {};\n for (const key in this._coordsDescendingInv[source])\n this._coordsDescendingInvStr[source][key] = this._coordsDescendingInv[source][key].map(c => c.key).sort().join();\n }\n }\n }\n\n // check tiles to render\n for (const tile of this._renderableTiles) {\n for (const source in this._coordsDescendingInvStr) {\n // rerender if there are more coords to render than in the last rendering\n const coords = this._coordsDescendingInvStr[source][tile.tileID.key];\n if (coords && coords !== tile.rttCoords[source]) tile.rtt = [];\n }\n }\n }\n\n /**\n * due that switching textures is relatively slow, the render\n * layer-by-layer context is not practicable. To bypass this problem\n * this lines of code stack all layers and later render all at once.\n * Because of the stylesheet possibility to mixing render-to-texture layers\n * and 'live'-layers (f.e. symbols) it is necessary to create more stacks. For example\n * a symbol-layer is in between of fill-layers.\n * @param layer - the layer to render\n * @returns if true layer is rendered to texture, otherwise false\n */\n renderLayer(layer: StyleLayer): boolean {\n if (layer.isHidden(this.painter.transform.zoom)) return false;\n\n const type = layer.type;\n const painter = this.painter;\n const isLastLayer = this._renderableLayerIds[this._renderableLayerIds.length - 1] === layer.id;\n\n // remember background, fill, line & raster layer to render into a stack\n if (LAYERS[type]) {\n // create a new stack if previous layer was not rendered to texture (f.e. symbols)\n if (!this._prevType || !LAYERS[this._prevType]) this._stacks.push([]);\n // push current render-to-texture layer to render-stack\n this._prevType = type;\n this._stacks[this._stacks.length - 1].push(layer.id);\n // rendering is done later, all in once\n if (!isLastLayer) return true;\n }\n\n // in case a stack is finished render all collected stack-layers into a texture\n if (LAYERS[this._prevType] || (LAYERS[type] && isLastLayer)) {\n this._prevType = type;\n const stack = this._stacks.length - 1, layers = this._stacks[stack] || [];\n for (const tile of this._renderableTiles) {\n // if render pool is full draw current tiles to screen and free pool\n if (this.pool.isFull()) {\n drawTerrain(this.painter, this.terrain, this._rttTiles);\n this._rttTiles = [];\n this.pool.freeAllObjects();\n }\n this._rttTiles.push(tile);\n // check for cached PoolObject\n if (tile.rtt[stack]) {\n const obj = this.pool.getObjectForId(tile.rtt[stack].id);\n if (obj.stamp === tile.rtt[stack].stamp) {\n this.pool.useObject(obj);\n continue;\n }\n }\n // get free PoolObject\n const obj = this.pool.getOrCreateFreeObject();\n this.pool.useObject(obj);\n this.pool.stampObject(obj);\n tile.rtt[stack] = {id: obj.id, stamp: obj.stamp};\n // prepare PoolObject for rendering\n painter.context.bindFramebuffer.set(obj.fbo.framebuffer);\n painter.context.clear({color: Color.transparent, stencil: 0});\n painter.currentStencilSource = undefined;\n for (let l = 0; l < layers.length; l++) {\n const layer = painter.style._layers[layers[l]];\n const coords = layer.source ? this._coordsDescendingInv[layer.source][tile.tileID.key] : [tile.tileID];\n painter.context.viewport.set([0, 0, obj.fbo.width, obj.fbo.height]);\n painter._renderTileClippingMasks(layer, coords);\n painter.renderLayer(painter, painter.style.sourceCaches[layer.source], layer, coords);\n if (layer.source) tile.rttCoords[layer.source] = this._coordsDescendingInvStr[layer.source][tile.tileID.key];\n }\n }\n drawTerrain(this.painter, this.terrain, this._rttTiles);\n this._rttTiles = [];\n this.pool.freeAllObjects();\n\n return LAYERS[type];\n }\n\n return false;\n }\n\n}\n","import {extend, warnOnce, uniqueId, isImageBitmap} from '../util/util';\nimport {browser} from '../util/browser';\nimport {DOM} from '../util/dom';\nimport packageJSON from '../../package.json' assert {type: 'json'};\n\nimport {getJSON} from '../util/ajax';\nimport {ImageRequest} from '../util/image_request';\nimport type {GetImageCallback} from '../util/image_request';\n\nimport {RequestManager, ResourceType} from '../util/request_manager';\nimport {Style, StyleSwapOptions} from '../style/style';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {Painter} from '../render/painter';\nimport {Transform} from '../geo/transform';\nimport {Hash} from './hash';\nimport {HandlerManager} from './handler_manager';\nimport {Camera, CameraOptions, CameraUpdateTransformFunction, FitBoundsOptions} from './camera';\nimport {LngLat} from '../geo/lng_lat';\nimport {LngLatBounds} from '../geo/lng_lat_bounds';\nimport Point from '@mapbox/point-geometry';\nimport {AttributionControl} from './control/attribution_control';\nimport {LogoControl} from './control/logo_control';\n\nimport {RGBAImage} from '../util/image';\nimport {Event, ErrorEvent, Listener} from '../util/evented';\nimport {MapEventType, MapLayerEventType, MapMouseEvent, MapSourceDataEvent, MapStyleDataEvent} from './events';\nimport {TaskQueue} from '../util/task_queue';\nimport {throttle} from '../util/throttle';\nimport {webpSupported} from '../util/webp_supported';\nimport {PerformanceMarkers, PerformanceUtils} from '../util/performance';\nimport {Source, SourceClass} from '../source/source';\nimport {StyleLayer} from '../style/style_layer';\n\nimport type {RequestTransformFunction} from '../util/request_manager';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport type {LngLatBoundsLike} from '../geo/lng_lat_bounds';\nimport type {AddLayerObject, FeatureIdentifier, StyleOptions, StyleSetterOptions} from '../style/style';\nimport type {MapDataEvent} from './events';\nimport type {StyleImage, StyleImageInterface, StyleImageMetadata} from '../style/style_image';\nimport type {PointLike} from './camera';\nimport type {ScrollZoomHandler} from './handler/scroll_zoom';\nimport type {BoxZoomHandler} from './handler/box_zoom';\nimport type {AroundCenterOptions, TwoFingersTouchPitchHandler} from './handler/two_fingers_touch';\nimport type {DragRotateHandler} from './handler/shim/drag_rotate';\nimport {DragPanHandler, DragPanOptions} from './handler/shim/drag_pan';\n\nimport type {KeyboardHandler} from './handler/keyboard';\nimport type {DoubleClickZoomHandler} from './handler/shim/dblclick_zoom';\nimport type {TwoFingersTouchZoomRotateHandler} from './handler/shim/two_fingers_touch';\nimport {defaultLocale} from './default_locale';\nimport type {TaskID} from '../util/task_queue';\nimport type {Cancelable} from '../types/cancelable';\nimport type {\n FilterSpecification,\n StyleSpecification,\n LightSpecification,\n SourceSpecification,\n TerrainSpecification\n} from '@maplibre/maplibre-gl-style-spec';\n\nimport {Callback} from '../types/callback';\nimport type {ControlPosition, IControl} from './control/control';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\nimport {Terrain} from '../render/terrain';\nimport {RenderToTexture} from '../render/render_to_texture';\nimport {config} from '../util/config';\nimport type {QueryRenderedFeaturesOptions, QuerySourceFeatureOptions} from '../source/query_features';\n\nconst version = packageJSON.version;\n\n/**\n * The {@link Map} options object.\n */\nexport type MapOptions = {\n /**\n * If `true`, the map's position (zoom, center latitude, center longitude, bearing, and pitch) will be synced with the hash fragment of the page's URL.\n * For example, `http://path/to/my/page.html#2.59/39.26/53.07/-24.1/60`.\n * An additional string may optionally be provided to indicate a parameter-styled hash,\n * e.g. http://path/to/my/page.html#map=2.59/39.26/53.07/-24.1/60&foo=bar, where foo\n * is a custom parameter and bar is an arbitrary hash distinct from the map hash.\n * @defaultValue false\n */\n hash?: boolean | string;\n /**\n * If `false`, no mouse, touch, or keyboard listeners will be attached to the map, so it will not respond to interaction.\n * @defaultValue true\n */\n interactive?: boolean;\n /**\n * The HTML element in which MapLibre GL JS will render the map, or the element's string `id`. The specified element must have no children.\n */\n container: HTMLElement | string;\n /**\n * The threshold, measured in degrees, that determines when the map's\n * bearing will snap to north. For example, with a `bearingSnap` of 7, if the user rotates\n * the map within 7 degrees of north, the map will automatically snap to exact north.\n * @defaultValue 7\n */\n bearingSnap?: number;\n /**\n * If `true`, an {@link AttributionControl} will be added to the map.\n * @defaultValue true\n */\n attributionControl?: boolean;\n /**\n * Attribution text to show in an {@link AttributionControl}. Only applicable if `options.attributionControl` is `true`.\n */\n customAttribution?: string | Array;\n /**\n * If `true`, the MapLibre logo will be shown.\n * @defaultValue false\n */\n maplibreLogo?: boolean;\n /**\n * A string representing the position of the MapLibre wordmark on the map. Valid options are `top-left`,`top-right`, `bottom-left`, or `bottom-right`.\n * @defaultValue 'bottom-left'\n */\n logoPosition?: ControlPosition;\n /**\n * If `true`, map creation will fail if the performance of MapLibre GL JS would be dramatically worse than expected\n * (i.e. a software renderer would be used).\n * @defaultValue false\n */\n failIfMajorPerformanceCaveat?: boolean;\n /**\n * If `true`, the map's canvas can be exported to a PNG using `map.getCanvas().toDataURL()`. This is `false` by default as a performance optimization.\n * @defaultValue false\n */\n preserveDrawingBuffer?: boolean;\n /**\n * If `true`, the gl context will be created with MSAA antialiasing, which can be useful for antialiasing custom layers. This is `false` by default as a performance optimization.\n */\n antialias?: boolean;\n /**\n * If `false`, the map won't attempt to re-request tiles once they expire per their HTTP `cacheControl`/`expires` headers.\n * @defaultValue true\n */\n refreshExpiredTiles?: boolean;\n /**\n * If set, the map will be constrained to the given bounds.\n */\n maxBounds?: LngLatBoundsLike;\n /**\n * If `true`, the \"scroll to zoom\" interaction is enabled. {@link AroundCenterOptions} are passed as options to {@link ScrollZoomHandler#enable}.\n * @defaultValue true\n */\n scrollZoom?: boolean | AroundCenterOptions;\n /**\n * The minimum zoom level of the map (0-24).\n * @defaultValue 0\n */\n minZoom?: number | null;\n /**\n * The maximum zoom level of the map (0-24).\n * @defaultValue 22\n */\n maxZoom?: number | null;\n /**\n * The minimum pitch of the map (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * @defaultValue 0\n */\n minPitch?: number | null;\n /**\n * The maximum pitch of the map (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * @defaultValue 60\n */\n maxPitch?: number | null;\n /**\n * If `true`, the \"box zoom\" interaction is enabled (see {@link BoxZoomHandler}).\n * @defaultValue true\n */\n boxZoom?: boolean;\n /**\n * If `true`, the \"drag to rotate\" interaction is enabled (see {@link DragRotateHandler}).\n * @defaultValue true\n */\n dragRotate?: boolean;\n /**\n * If `true`, the \"drag to pan\" interaction is enabled. An `Object` value is passed as options to {@link DragPanHandler#enable}.\n * @defaultValue true\n */\n dragPan?: boolean | DragPanOptions;\n /**\n * If `true`, keyboard shortcuts are enabled (see {@link KeyboardHandler}).\n * @defaultValue true\n */\n keyboard?: boolean;\n /**\n * If `true`, the \"double click to zoom\" interaction is enabled (see {@link DoubleClickZoomHandler}).\n * @defaultValue true\n */\n doubleClickZoom?: boolean;\n /**\n * If `true`, the \"pinch to rotate and zoom\" interaction is enabled. An `Object` value is passed as options to {@link TwoFingersTouchZoomRotateHandler#enable}.\n * @defaultValue true\n */\n touchZoomRotate?: boolean | AroundCenterOptions;\n /**\n * If `true`, the \"drag to pitch\" interaction is enabled. An `Object` value is passed as options to {@link TwoFingersTouchPitchHandler#enable}.\n * @defaultValue true\n */\n touchPitch?: boolean | AroundCenterOptions;\n /**\n * If `true` or set to an options object, the map is only accessible on desktop while holding Command/Ctrl and only accessible on mobile with two fingers. Interacting with the map using normal gestures will trigger an informational screen. With this option enabled, \"drag to pitch\" requires a three-finger gesture. Cooperative gestures are disabled when a map enters fullscreen using {@link FullscreenControl}.\n * @defaultValue undefined\n */\n cooperativeGestures?: boolean | GestureOptions;\n /**\n * If `true`, the map will automatically resize when the browser window resizes.\n * @defaultValue true\n */\n trackResize?: boolean;\n /**\n * The initial geographical centerpoint of the map. If `center` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `[0, 0]` Note: MapLibre GL JS uses longitude, latitude coordinate order (as opposed to latitude, longitude) to match GeoJSON.\n * @defaultValue [0, 0]\n */\n center?: LngLatLike;\n /**\n * The initial zoom level of the map. If `zoom` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`.\n * @defaultValue 0\n */\n zoom?: number;\n /**\n * The initial bearing (rotation) of the map, measured in degrees counter-clockwise from north. If `bearing` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`.\n * @defaultValue 0\n */\n bearing?: number;\n /**\n * The initial pitch (tilt) of the map, measured in degrees away from the plane of the screen (0-85). If `pitch` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`. Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * @defaultValue 0\n */\n pitch?: number;\n /**\n * If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`:\n * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire\n * container, there will be blank space beyond 180 and -180 degrees longitude.\n * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the\n * map and the other on the left edge of the map) at every zoom level.\n * @defaultValue true\n */\n renderWorldCopies?: boolean;\n /**\n * The maximum number of tiles stored in the tile cache for a given source. If omitted, the cache will be dynamically sized based on the current viewport which can be set using `maxTileCacheZoomLevels` constructor options.\n * @defaultValue null\n */\n maxTileCacheSize?: number;\n /**\n * The maximum number of zoom levels for which to store tiles for a given source. Tile cache dynamic size is calculated by multiplying `maxTileCacheZoomLevels` with the approximate number of tiles in the viewport for a given source.\n * @defaultValue 5\n */\n maxTileCacheZoomLevels?: number;\n /**\n * A callback run before the Map makes a request for an external URL. The callback can be used to modify the url, set headers, or set the credentials property for cross-origin requests.\n * Expected to return an object with a `url` property and optionally `headers` and `credentials` properties.\n */\n transformRequest?: RequestTransformFunction;\n /**\n * A callback run before the map's camera is moved due to user input or animation. The callback can be used to modify the new center, zoom, pitch and bearing.\n * Expected to return an object containing center, zoom, pitch or bearing values to overwrite.\n */\n transformCameraUpdate?: CameraUpdateTransformFunction;\n /**\n * A patch to apply to the default localization table for UI strings, e.g. control tooltips. The `locale` object maps namespaced UI string IDs to translated strings in the target language; see `src/ui/default_locale.js` for an example with all supported string IDs. The object may specify all UI strings (thereby adding support for a new translation) or only a subset of strings (thereby patching the default translation table).\n * @defaultValue null\n */\n locale?: any;\n /**\n * Controls the duration of the fade-in/fade-out animation for label collisions after initial map load, in milliseconds. This setting affects all symbol layers. This setting does not affect the duration of runtime styling transitions or raster tile cross-fading.\n * @defaultValue 300\n */\n fadeDuration?: number;\n /**\n * If `true`, symbols from multiple sources can collide with each other during collision detection. If `false`, collision detection is run separately for the symbols in each source.\n * @defaultValue true\n */\n crossSourceCollisions?: boolean;\n /**\n * If `true`, Resource Timing API information will be collected for requests made by GeoJSON and Vector Tile web workers (this information is normally inaccessible from the main Javascript thread). Information will be returned in a `resourceTiming` property of relevant `data` events.\n * @defaultValue false\n */\n collectResourceTiming?: boolean;\n /**\n * The max number of pixels a user can shift the mouse pointer during a click for it to be considered a valid click (as opposed to a mouse drag).\n * @defaultValue true\n */\n clickTolerance?: number;\n /**\n * The initial bounds of the map. If `bounds` is specified, it overrides `center` and `zoom` constructor options.\n */\n bounds?: LngLatBoundsLike;\n /**\n * A {@link FitBoundsOptions} options object to use _only_ when fitting the initial `bounds` provided above.\n */\n fitBoundsOptions?: FitBoundsOptions;\n /**\n * Defines a CSS\n * font-family for locally overriding generation of glyphs in the 'CJK Unified Ideographs', 'Hiragana', 'Katakana' and 'Hangul Syllables' ranges.\n * In these ranges, font settings from the map's style will be ignored, except for font-weight keywords (light/regular/medium/bold).\n * Set to `false`, to enable font settings from the map's style for these glyph ranges.\n * The purpose of this option is to avoid bandwidth-intensive glyph server requests. (See [Use locally generated ideographs](https://maplibre.org/maplibre-gl-js/docs/examples/local-ideographs).)\n * @defaultValue 'sans-serif'\n */\n localIdeographFontFamily?: string;\n /**\n * The map's MapLibre style. This must be a JSON object conforming to\n * the schema described in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/),\n * or a URL to such JSON.\n */\n style: StyleSpecification | string;\n /**\n * If `false`, the map's pitch (tilt) control with \"drag to rotate\" interaction will be disabled.\n * @defaultValue true\n */\n pitchWithRotate?: boolean;\n /**\n * The pixel ratio. The canvas' `width` attribute will be `container.clientWidth * pixelRatio` and its `height` attribute will be `container.clientHeight * pixelRatio`. Defaults to `devicePixelRatio` if not specified.\n */\n pixelRatio?: number;\n /**\n * If false, style validation will be skipped. Useful in production environment.\n * @defaultValue true\n */\n validateStyle?: boolean;\n /**\n * The canvas' `width` and `height` max size. The values are passed as an array where the first element is max width and the second element is max height.\n * You shouldn't set this above WebGl `MAX_TEXTURE_SIZE`. Defaults to [4096, 4096].\n */\n maxCanvasSize?: [number, number];\n};\n\n/**\n * An options object for the gesture settings\n * @example\n * ```ts\n * let options = {\n * windowsHelpText: \"Use Ctrl + scroll to zoom the map\",\n * macHelpText: \"Use ⌘ + scroll to zoom the map\",\n * mobileHelpText: \"Use two fingers to move the map\",\n * }\n * ```\n */\nexport type GestureOptions = {\n windowsHelpText?: string;\n macHelpText?: string;\n mobileHelpText?: string;\n};\n\nexport type AddImageOptions = {\n\n}\n\n// See article here: https://medium.com/terria/typescript-transforming-optional-properties-to-required-properties-that-may-be-undefined-7482cb4e1585\ntype Complete = {\n [P in keyof Required]: Pick extends Required> ? T[P] : (T[P] | undefined);\n}\n\n// This type is used inside map since all properties are assigned a default value.\nexport type CompleteMapOptions = Complete;\n\nconst defaultMinZoom = -2;\nconst defaultMaxZoom = 22;\n\n// the default values, but also the valid range\nconst defaultMinPitch = 0;\nconst defaultMaxPitch = 60;\n\n// use this variable to check maxPitch for validity\nconst maxPitchThreshold = 85;\n\nconst defaultOptions = {\n center: [0, 0],\n zoom: 0,\n bearing: 0,\n pitch: 0,\n\n minZoom: defaultMinZoom,\n maxZoom: defaultMaxZoom,\n\n minPitch: defaultMinPitch,\n maxPitch: defaultMaxPitch,\n\n interactive: true,\n scrollZoom: true,\n boxZoom: true,\n dragRotate: true,\n dragPan: true,\n keyboard: true,\n doubleClickZoom: true,\n touchZoomRotate: true,\n touchPitch: true,\n cooperativeGestures: undefined,\n\n bearingSnap: 7,\n clickTolerance: 3,\n pitchWithRotate: true,\n\n hash: false,\n attributionControl: true,\n maplibreLogo: false,\n\n failIfMajorPerformanceCaveat: false,\n preserveDrawingBuffer: false,\n trackResize: true,\n renderWorldCopies: true,\n refreshExpiredTiles: true,\n maxTileCacheSize: null,\n maxTileCacheZoomLevels: config.MAX_TILE_CACHE_ZOOM_LEVELS,\n localIdeographFontFamily: 'sans-serif',\n transformRequest: null,\n transformCameraUpdate: null,\n fadeDuration: 300,\n crossSourceCollisions: true,\n validateStyle: true,\n /**Because GL MAX_TEXTURE_SIZE is usually at least 4096px. */\n maxCanvasSize: [4096, 4096]\n} as CompleteMapOptions;\n\n/**\n * The `Map` object represents the map on your page. It exposes methods\n * and properties that enable you to programmatically change the map,\n * and fires events as users interact with it.\n *\n * You create a `Map` by specifying a `container` and other options, see {@link MapOptions} for the full list.\n * Then MapLibre GL JS initializes the map on the page and returns your `Map` object.\n *\n * @group Main\n *\n * @example\n * ```ts\n * let map = new maplibregl.Map({\n * container: 'map',\n * center: [-122.420679, 37.772537],\n * zoom: 13,\n * style: style_object,\n * hash: true,\n * transformRequest: (url, resourceType)=> {\n * if(resourceType === 'Source' && url.startsWith('http://myHost')) {\n * return {\n * url: url.replace('http', 'https'),\n * headers: { 'my-custom-header': true},\n * credentials: 'include' // Include cookies for cross-origin requests\n * }\n * }\n * }\n * });\n * ```\n * @see [Display a map](https://maplibre.org/maplibre-gl-js/docs/examples/simple-map/)\n */\nexport class Map extends Camera {\n style: Style;\n painter: Painter;\n handlers: HandlerManager;\n\n _container: HTMLElement;\n _canvasContainer: HTMLElement;\n _controlContainer: HTMLElement;\n _controlPositions: {[_: string]: HTMLElement};\n _interactive: boolean;\n _cooperativeGestures: boolean | GestureOptions;\n _cooperativeGesturesScreen: HTMLElement;\n _metaKey: keyof MouseEvent;\n _showTileBoundaries: boolean;\n _showCollisionBoxes: boolean;\n _showPadding: boolean;\n _showOverdrawInspector: boolean;\n _repaint: boolean;\n _vertices: boolean;\n _canvas: HTMLCanvasElement;\n _maxTileCacheSize: number;\n _maxTileCacheZoomLevels: number;\n _frame: Cancelable;\n _styleDirty: boolean;\n _sourcesDirty: boolean;\n _placementDirty: boolean;\n\n _loaded: boolean;\n _idleTriggered: boolean;\n // accounts for placement finishing as well\n _fullyLoaded: boolean;\n _trackResize: boolean;\n _resizeObserver: ResizeObserver;\n _preserveDrawingBuffer: boolean;\n _failIfMajorPerformanceCaveat: boolean;\n _antialias: boolean;\n _refreshExpiredTiles: boolean;\n _hash: Hash;\n _delegatedListeners: any;\n _fadeDuration: number;\n _crossSourceCollisions: boolean;\n _crossFadingFactor: number;\n _collectResourceTiming: boolean;\n _renderTaskQueue: TaskQueue;\n _controls: Array;\n _mapId: number;\n _localIdeographFontFamily: string;\n _validateStyle: boolean;\n _requestManager: RequestManager;\n _locale: any;\n _removed: boolean;\n _clickTolerance: number;\n _overridePixelRatio: number | null;\n _maxCanvasSize: [number, number];\n _terrainDataCallback: (e: MapStyleDataEvent | MapSourceDataEvent) => void;\n\n /**\n * @internal\n * image queue throttling handle. To be used later when clean up\n */\n _imageQueueHandle: number;\n\n /**\n * The map's {@link ScrollZoomHandler}, which implements zooming in and out with a scroll wheel or trackpad.\n * Find more details and examples using `scrollZoom` in the {@link ScrollZoomHandler} section.\n */\n scrollZoom: ScrollZoomHandler;\n\n /**\n * The map's {@link BoxZoomHandler}, which implements zooming using a drag gesture with the Shift key pressed.\n * Find more details and examples using `boxZoom` in the {@link BoxZoomHandler} section.\n */\n boxZoom: BoxZoomHandler;\n\n /**\n * The map's {@link DragRotateHandler}, which implements rotating the map while dragging with the right\n * mouse button or with the Control key pressed. Find more details and examples using `dragRotate`\n * in the {@link DragRotateHandler} section.\n */\n dragRotate: DragRotateHandler;\n\n /**\n * The map's {@link DragPanHandler}, which implements dragging the map with a mouse or touch gesture.\n * Find more details and examples using `dragPan` in the {@link DragPanHandler} section.\n */\n dragPan: DragPanHandler;\n\n /**\n * The map's {@link KeyboardHandler}, which allows the user to zoom, rotate, and pan the map using keyboard\n * shortcuts. Find more details and examples using `keyboard` in the {@link KeyboardHandler} section.\n */\n keyboard: KeyboardHandler;\n\n /**\n * The map's {@link DoubleClickZoomHandler}, which allows the user to zoom by double clicking.\n * Find more details and examples using `doubleClickZoom` in the {@link DoubleClickZoomHandler} section.\n */\n doubleClickZoom: DoubleClickZoomHandler;\n\n /**\n * The map's {@link TwoFingersTouchZoomRotateHandler}, which allows the user to zoom or rotate the map with touch gestures.\n * Find more details and examples using `touchZoomRotate` in the {@link TwoFingersTouchZoomRotateHandler} section.\n */\n touchZoomRotate: TwoFingersTouchZoomRotateHandler;\n\n /**\n * The map's {@link TwoFingersTouchPitchHandler}, which allows the user to pitch the map with touch gestures.\n * Find more details and examples using `touchPitch` in the {@link TwoFingersTouchPitchHandler} section.\n */\n touchPitch: TwoFingersTouchPitchHandler;\n\n constructor(options: MapOptions) {\n PerformanceUtils.mark(PerformanceMarkers.create);\n\n options = extend({}, defaultOptions, options);\n\n if (options.minZoom != null && options.maxZoom != null && options.minZoom > options.maxZoom) {\n throw new Error('maxZoom must be greater than or equal to minZoom');\n }\n\n if (options.minPitch != null && options.maxPitch != null && options.minPitch > options.maxPitch) {\n throw new Error('maxPitch must be greater than or equal to minPitch');\n }\n\n if (options.minPitch != null && options.minPitch < defaultMinPitch) {\n throw new Error(`minPitch must be greater than or equal to ${defaultMinPitch}`);\n }\n\n if (options.maxPitch != null && options.maxPitch > maxPitchThreshold) {\n throw new Error(`maxPitch must be less than or equal to ${maxPitchThreshold}`);\n }\n\n const transform = new Transform(options.minZoom, options.maxZoom, options.minPitch, options.maxPitch, options.renderWorldCopies);\n super(transform, {bearingSnap: options.bearingSnap});\n\n this._interactive = options.interactive;\n this._cooperativeGestures = options.cooperativeGestures;\n this._metaKey = navigator.platform.indexOf('Mac') === 0 ? 'metaKey' : 'ctrlKey';\n this._maxTileCacheSize = options.maxTileCacheSize;\n this._maxTileCacheZoomLevels = options.maxTileCacheZoomLevels;\n this._failIfMajorPerformanceCaveat = options.failIfMajorPerformanceCaveat;\n this._preserveDrawingBuffer = options.preserveDrawingBuffer;\n this._antialias = options.antialias;\n this._trackResize = options.trackResize;\n this._bearingSnap = options.bearingSnap;\n this._refreshExpiredTiles = options.refreshExpiredTiles;\n this._fadeDuration = options.fadeDuration;\n this._crossSourceCollisions = options.crossSourceCollisions;\n this._crossFadingFactor = 1;\n this._collectResourceTiming = options.collectResourceTiming;\n this._renderTaskQueue = new TaskQueue();\n this._controls = [];\n this._mapId = uniqueId();\n this._locale = extend({}, defaultLocale, options.locale);\n this._clickTolerance = options.clickTolerance;\n this._overridePixelRatio = options.pixelRatio;\n this._maxCanvasSize = options.maxCanvasSize;\n this.transformCameraUpdate = options.transformCameraUpdate;\n\n this._imageQueueHandle = ImageRequest.addThrottleControl(() => this.isMoving());\n\n this._requestManager = new RequestManager(options.transformRequest);\n\n if (typeof options.container === 'string') {\n this._container = document.getElementById(options.container);\n if (!this._container) {\n throw new Error(`Container '${options.container}' not found.`);\n }\n } else if (options.container instanceof HTMLElement) {\n this._container = options.container;\n } else {\n throw new Error('Invalid type: \\'container\\' must be a String or HTMLElement.');\n }\n\n if (options.maxBounds) {\n this.setMaxBounds(options.maxBounds);\n }\n\n this._setupContainer();\n this._setupPainter();\n\n this.on('move', () => this._update(false));\n this.on('moveend', () => this._update(false));\n this.on('zoom', () => this._update(true));\n this.on('terrain', () => {\n this.painter.terrainFacilitator.dirty = true;\n this._update(true);\n });\n this.once('idle', () => { this._idleTriggered = true; });\n\n if (typeof window !== 'undefined') {\n addEventListener('online', this._onWindowOnline, false);\n let initialResizeEventCaptured = false;\n const throttledResizeCallback = throttle((entries: ResizeObserverEntry[]) => {\n if (this._trackResize && !this._removed) {\n this.resize(entries)._update();\n }\n }, 50);\n this._resizeObserver = new ResizeObserver((entries) => {\n if (!initialResizeEventCaptured) {\n initialResizeEventCaptured = true;\n return;\n }\n throttledResizeCallback(entries);\n });\n this._resizeObserver.observe(this._container);\n }\n\n this.handlers = new HandlerManager(this, options as CompleteMapOptions);\n\n if (this._cooperativeGestures) {\n this._setupCooperativeGestures();\n }\n\n const hashName = (typeof options.hash === 'string' && options.hash) || undefined;\n this._hash = options.hash && (new Hash(hashName)).addTo(this);\n // don't set position from options if set through hash\n if (!this._hash || !this._hash._onHashChange()) {\n this.jumpTo({\n center: options.center,\n zoom: options.zoom,\n bearing: options.bearing,\n pitch: options.pitch\n });\n\n if (options.bounds) {\n this.resize();\n this.fitBounds(options.bounds, extend({}, options.fitBoundsOptions, {duration: 0}));\n }\n }\n\n this.resize();\n\n this._localIdeographFontFamily = options.localIdeographFontFamily;\n this._validateStyle = options.validateStyle;\n\n if (options.style) this.setStyle(options.style, {localIdeographFontFamily: options.localIdeographFontFamily});\n\n if (options.attributionControl)\n this.addControl(new AttributionControl({customAttribution: options.customAttribution}));\n\n if (options.maplibreLogo)\n this.addControl(new LogoControl(), options.logoPosition);\n\n this.on('style.load', () => {\n if (this.transform.unmodified) {\n this.jumpTo(this.style.stylesheet as any);\n }\n });\n this.on('data', (event: MapDataEvent) => {\n this._update(event.dataType === 'style');\n this.fire(new Event(`${event.dataType}data`, event));\n });\n this.on('dataloading', (event: MapDataEvent) => {\n this.fire(new Event(`${event.dataType}dataloading`, event));\n });\n this.on('dataabort', (event: MapDataEvent) => {\n this.fire(new Event('sourcedataabort', event));\n });\n }\n\n /**\n * @internal\n * Returns a unique number for this map instance which is used for the MapLoadEvent\n * to make sure we only fire one event per instantiated map object.\n * @returns the uniq map ID\n */\n _getMapId() {\n return this._mapId;\n }\n\n /**\n * Adds an {@link IControl} to the map, calling `control.onAdd(this)`.\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param control - The {@link IControl} to add.\n * @param position - position on the map to which the control will be added.\n * Valid values are `'top-left'`, `'top-right'`, `'bottom-left'`, and `'bottom-right'`. Defaults to `'top-right'`.\n * @returns `this`\n * @example\n * Add zoom and rotation controls to the map.\n * ```ts\n * map.addControl(new maplibregl.NavigationControl());\n * ```\n * @see [Display map navigation controls](https://maplibre.org/maplibre-gl-js/docs/examples/navigation/)\n */\n addControl(control: IControl, position?: ControlPosition): Map {\n if (position === undefined) {\n if (control.getDefaultPosition) {\n position = control.getDefaultPosition();\n } else {\n position = 'top-right';\n }\n }\n if (!control || !control.onAdd) {\n return this.fire(new ErrorEvent(new Error(\n 'Invalid argument to map.addControl(). Argument must be a control with onAdd and onRemove methods.')));\n }\n const controlElement = control.onAdd(this);\n this._controls.push(control);\n\n const positionContainer = this._controlPositions[position];\n if (position.indexOf('bottom') !== -1) {\n positionContainer.insertBefore(controlElement, positionContainer.firstChild);\n } else {\n positionContainer.appendChild(controlElement);\n }\n return this;\n }\n\n /**\n * Removes the control from the map.\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param control - The {@link IControl} to remove.\n * @returns `this`\n * @example\n * ```ts\n * // Define a new navigation control.\n * let navigation = new maplibregl.NavigationControl();\n * // Add zoom and rotation controls to the map.\n * map.addControl(navigation);\n * // Remove zoom and rotation controls from the map.\n * map.removeControl(navigation);\n * ```\n */\n removeControl(control: IControl): Map {\n if (!control || !control.onRemove) {\n return this.fire(new ErrorEvent(new Error(\n 'Invalid argument to map.removeControl(). Argument must be a control with onAdd and onRemove methods.')));\n }\n const ci = this._controls.indexOf(control);\n if (ci > -1) this._controls.splice(ci, 1);\n control.onRemove(this);\n return this;\n }\n\n /**\n * Checks if a control exists on the map.\n *\n * @param control - The {@link IControl} to check.\n * @returns true if map contains control.\n * @example\n * ```ts\n * // Define a new navigation control.\n * let navigation = new maplibregl.NavigationControl();\n * // Add zoom and rotation controls to the map.\n * map.addControl(navigation);\n * // Check that the navigation control exists on the map.\n * map.hasControl(navigation);\n * ```\n */\n hasControl(control: IControl): boolean {\n return this._controls.indexOf(control) > -1;\n }\n\n calculateCameraOptionsFromTo(from: LngLat, altitudeFrom: number, to: LngLat, altitudeTo?: number): CameraOptions {\n if (altitudeTo == null && this.terrain) {\n altitudeTo = this.terrain.getElevationForLngLatZoom(to, this.transform.tileZoom);\n }\n return super.calculateCameraOptionsFromTo(from, altitudeFrom, to, altitudeTo);\n }\n\n /**\n * Resizes the map according to the dimensions of its\n * `container` element.\n *\n * Checks if the map container size changed and updates the map if it has changed.\n * This method must be called after the map's `container` is resized programmatically\n * or when the map is shown after being initially hidden with CSS.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, and `resize`.\n *\n * @param eventData - Additional properties to be passed to `movestart`, `move`, `resize`, and `moveend`\n * events that get triggered as a result of resize. This can be useful for differentiating the\n * source of an event (for example, user-initiated or programmatically-triggered events).\n * @returns `this`\n * @example\n * Resize the map when the map container is shown after being initially hidden with CSS.\n * ```ts\n * let mapDiv = document.getElementById('map');\n * if (mapDiv.style.visibility === true) map.resize();\n * ```\n */\n resize(eventData?: any): Map {\n const dimensions = this._containerDimensions();\n const width = dimensions[0];\n const height = dimensions[1];\n\n const clampedPixelRatio = this._getClampedPixelRatio(width, height);\n this._resizeCanvas(width, height, clampedPixelRatio);\n this.painter.resize(width, height, clampedPixelRatio);\n\n // check if we've reached GL limits, in that case further clamps pixelRatio\n if (this.painter.overLimit()) {\n const gl = this.painter.context.gl;\n // store updated _maxCanvasSize value\n this._maxCanvasSize = [gl.drawingBufferWidth, gl.drawingBufferHeight];\n const clampedPixelRatio = this._getClampedPixelRatio(width, height);\n this._resizeCanvas(width, height, clampedPixelRatio);\n this.painter.resize(width, height, clampedPixelRatio);\n }\n\n this.transform.resize(width, height);\n this._requestedCameraState?.resize(width, height);\n\n const fireMoving = !this._moving;\n if (fireMoving) {\n this.stop();\n this.fire(new Event('movestart', eventData))\n .fire(new Event('move', eventData));\n }\n\n this.fire(new Event('resize', eventData));\n\n if (fireMoving) this.fire(new Event('moveend', eventData));\n\n return this;\n }\n\n /**\n * @internal\n * Return the map's pixel ratio eventually scaled down to respect maxCanvasSize.\n * Internally you should use this and not getPixelRatio().\n */\n _getClampedPixelRatio(width: number, height: number): number {\n const {0: maxCanvasWidth, 1: maxCanvasHeight} = this._maxCanvasSize;\n const pixelRatio = this.getPixelRatio();\n\n const canvasWidth = width * pixelRatio;\n const canvasHeight = height * pixelRatio;\n\n const widthScaleFactor = canvasWidth > maxCanvasWidth ? (maxCanvasWidth / canvasWidth) : 1;\n const heightScaleFactor = canvasHeight > maxCanvasHeight ? (maxCanvasHeight / canvasHeight) : 1;\n\n return Math.min(widthScaleFactor, heightScaleFactor) * pixelRatio;\n }\n\n /**\n * Returns the map's pixel ratio.\n * Note that the pixel ratio actually applied may be lower to respect maxCanvasSize.\n * @returns The pixel ratio.\n */\n getPixelRatio(): number {\n return this._overridePixelRatio ?? devicePixelRatio;\n }\n\n /**\n * Sets the map's pixel ratio. This allows to override `devicePixelRatio`.\n * After this call, the canvas' `width` attribute will be `container.clientWidth * pixelRatio`\n * and its height attribute will be `container.clientHeight * pixelRatio`.\n * Set this to null to disable `devicePixelRatio` override.\n * Note that the pixel ratio actually applied may be lower to respect maxCanvasSize.\n * @param pixelRatio - The pixel ratio.\n */\n setPixelRatio(pixelRatio: number) {\n this._overridePixelRatio = pixelRatio;\n this.resize();\n }\n\n /**\n * Returns the map's geographical bounds. When the bearing or pitch is non-zero, the visible region is not\n * an axis-aligned rectangle, and the result is the smallest bounds that encompasses the visible region.\n * @returns The geographical bounds of the map as {@link LngLatBounds}.\n * @example\n * ```ts\n * let bounds = map.getBounds();\n * ```\n */\n getBounds(): LngLatBounds {\n return this.transform.getBounds();\n }\n\n /**\n * Returns the maximum geographical bounds the map is constrained to, or `null` if none set.\n * @returns The map object.\n * @example\n * ```ts\n * let maxBounds = map.getMaxBounds();\n * ```\n */\n getMaxBounds(): LngLatBounds | null {\n return this.transform.getMaxBounds();\n }\n\n /**\n * Sets or clears the map's geographical bounds.\n *\n * Pan and zoom operations are constrained within these bounds.\n * If a pan or zoom is performed that would\n * display regions outside these bounds, the map will\n * instead display a position and zoom level\n * as close as possible to the operation's request while still\n * remaining within the bounds.\n *\n * @param bounds - The maximum bounds to set. If `null` or `undefined` is provided, the function removes the map's maximum bounds.\n * @returns `this`\n * @example\n * Define bounds that conform to the `LngLatBoundsLike` object as set the max bounds.\n * ```ts\n * let bounds = [\n * [-74.04728, 40.68392], // [west, south]\n * [-73.91058, 40.87764] // [east, north]\n * ];\n * map.setMaxBounds(bounds);\n * ```\n */\n setMaxBounds(bounds?: LngLatBoundsLike | null): Map {\n this.transform.setMaxBounds(LngLatBounds.convert(bounds));\n return this._update();\n }\n\n /**\n * Sets or clears the map's minimum zoom level.\n * If the map's current zoom level is lower than the new minimum,\n * the map will zoom to the new minimum.\n *\n * It is not always possible to zoom out and reach the set `minZoom`.\n * Other factors such as map height may restrict zooming. For example,\n * if the map is 512px tall it will not be possible to zoom below zoom 0\n * no matter what the `minZoom` is set to.\n *\n * A {@link ErrorEvent} event will be fired if minZoom is out of bounds.\n *\n * @param minZoom - The minimum zoom level to set (-2 - 24).\n * If `null` or `undefined` is provided, the function removes the current minimum zoom (i.e. sets it to -2).\n * @returns `this`\n * @example\n * ```ts\n * map.setMinZoom(12.25);\n * ```\n */\n setMinZoom(minZoom?: number | null): Map {\n\n minZoom = minZoom === null || minZoom === undefined ? defaultMinZoom : minZoom;\n\n if (minZoom >= defaultMinZoom && minZoom <= this.transform.maxZoom) {\n this.transform.minZoom = minZoom;\n this._update();\n\n if (this.getZoom() < minZoom) this.setZoom(minZoom);\n\n return this;\n\n } else throw new Error(`minZoom must be between ${defaultMinZoom} and the current maxZoom, inclusive`);\n }\n\n /**\n * Returns the map's minimum allowable zoom level.\n *\n * @returns minZoom\n * @example\n * ```ts\n * let minZoom = map.getMinZoom();\n * ```\n */\n getMinZoom(): number { return this.transform.minZoom; }\n\n /**\n * Sets or clears the map's maximum zoom level.\n * If the map's current zoom level is higher than the new maximum,\n * the map will zoom to the new maximum.\n *\n * A {@link ErrorEvent} event will be fired if minZoom is out of bounds.\n *\n * @param maxZoom - The maximum zoom level to set.\n * If `null` or `undefined` is provided, the function removes the current maximum zoom (sets it to 22).\n * @returns `this`\n * @example\n * ```ts\n * map.setMaxZoom(18.75);\n * ```\n */\n setMaxZoom(maxZoom?: number | null): Map {\n\n maxZoom = maxZoom === null || maxZoom === undefined ? defaultMaxZoom : maxZoom;\n\n if (maxZoom >= this.transform.minZoom) {\n this.transform.maxZoom = maxZoom;\n this._update();\n\n if (this.getZoom() > maxZoom) this.setZoom(maxZoom);\n\n return this;\n\n } else throw new Error('maxZoom must be greater than the current minZoom');\n }\n\n /**\n * Returns the map's maximum allowable zoom level.\n *\n * @returns The maxZoom\n * @example\n * ```ts\n * let maxZoom = map.getMaxZoom();\n * ```\n */\n getMaxZoom(): number { return this.transform.maxZoom; }\n\n /**\n * Sets or clears the map's minimum pitch.\n * If the map's current pitch is lower than the new minimum,\n * the map will pitch to the new minimum.\n *\n * A {@link ErrorEvent} event will be fired if minPitch is out of bounds.\n *\n * @param minPitch - The minimum pitch to set (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * If `null` or `undefined` is provided, the function removes the current minimum pitch (i.e. sets it to 0).\n * @returns `this`\n */\n setMinPitch(minPitch?: number | null): Map {\n\n minPitch = minPitch === null || minPitch === undefined ? defaultMinPitch : minPitch;\n\n if (minPitch < defaultMinPitch) {\n throw new Error(`minPitch must be greater than or equal to ${defaultMinPitch}`);\n }\n\n if (minPitch >= defaultMinPitch && minPitch <= this.transform.maxPitch) {\n this.transform.minPitch = minPitch;\n this._update();\n\n if (this.getPitch() < minPitch) this.setPitch(minPitch);\n\n return this;\n\n } else throw new Error(`minPitch must be between ${defaultMinPitch} and the current maxPitch, inclusive`);\n }\n\n /**\n * Returns the map's minimum allowable pitch.\n *\n * @returns The minPitch\n */\n getMinPitch(): number { return this.transform.minPitch; }\n\n /**\n * Sets or clears the map's maximum pitch.\n * If the map's current pitch is higher than the new maximum,\n * the map will pitch to the new maximum.\n *\n * A {@link ErrorEvent} event will be fired if maxPitch is out of bounds.\n *\n * @param maxPitch - The maximum pitch to set (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * If `null` or `undefined` is provided, the function removes the current maximum pitch (sets it to 60).\n * @returns `this`\n */\n setMaxPitch(maxPitch?: number | null): Map {\n\n maxPitch = maxPitch === null || maxPitch === undefined ? defaultMaxPitch : maxPitch;\n\n if (maxPitch > maxPitchThreshold) {\n throw new Error(`maxPitch must be less than or equal to ${maxPitchThreshold}`);\n }\n\n if (maxPitch >= this.transform.minPitch) {\n this.transform.maxPitch = maxPitch;\n this._update();\n\n if (this.getPitch() > maxPitch) this.setPitch(maxPitch);\n\n return this;\n\n } else throw new Error('maxPitch must be greater than the current minPitch');\n }\n\n /**\n * Returns the map's maximum allowable pitch.\n *\n * @returns The maxPitch\n */\n getMaxPitch(): number { return this.transform.maxPitch; }\n\n /**\n * Returns the state of `renderWorldCopies`. If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`:\n * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire\n * container, there will be blank space beyond 180 and -180 degrees longitude.\n * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the\n * map and the other on the left edge of the map) at every zoom level.\n * @returns The renderWorldCopies\n * @example\n * ```ts\n * let worldCopiesRendered = map.getRenderWorldCopies();\n * ```\n * @see [Render world copies](https://maplibre.org/maplibre-gl-js/docs/examples/render-world-copies/)\n */\n getRenderWorldCopies(): boolean { return this.transform.renderWorldCopies; }\n\n /**\n * Sets the state of `renderWorldCopies`.\n *\n * @param renderWorldCopies - If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`:\n * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire\n * container, there will be blank space beyond 180 and -180 degrees longitude.\n * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the\n * map and the other on the left edge of the map) at every zoom level.\n *\n * `undefined` is treated as `true`, `null` is treated as `false`.\n * @returns `this`\n * @example\n * ```ts\n * map.setRenderWorldCopies(true);\n * ```\n * @see [Render world copies](https://maplibre.org/maplibre-gl-js/docs/examples/render-world-copies/)\n */\n setRenderWorldCopies(renderWorldCopies?: boolean | null): Map {\n this.transform.renderWorldCopies = renderWorldCopies;\n return this._update();\n }\n\n /**\n * Gets the map's cooperativeGestures option\n *\n * @returns The gestureOptions\n */\n getCooperativeGestures(): boolean | GestureOptions {\n return this._cooperativeGestures;\n }\n\n /**\n * Sets or clears the map's cooperativeGestures option\n *\n * @param gestureOptions - If `true` or set to an options object, map is only accessible on desktop while holding Command/Ctrl and only accessible on mobile with two fingers. Interacting with the map using normal gestures will trigger an informational screen. With this option enabled, \"drag to pitch\" requires a three-finger gesture.\n * @returns `this`\n */\n setCooperativeGestures(gestureOptions?: GestureOptions | boolean | null): Map {\n this._cooperativeGestures = gestureOptions;\n if (this._cooperativeGestures) {\n this._setupCooperativeGestures();\n } else {\n this._destroyCooperativeGestures();\n }\n\n return this;\n }\n\n /**\n * Returns a [Point](https://github.com/mapbox/point-geometry) representing pixel coordinates, relative to the map's `container`,\n * that correspond to the specified geographical location.\n *\n * @param lnglat - The geographical location to project.\n * @returns The [Point](https://github.com/mapbox/point-geometry) corresponding to `lnglat`, relative to the map's `container`.\n * @example\n * ```ts\n * let coordinate = [-122.420679, 37.772537];\n * let point = map.project(coordinate);\n * ```\n */\n project(lnglat: LngLatLike): Point {\n return this.transform.locationPoint(LngLat.convert(lnglat), this.style && this.terrain);\n }\n\n /**\n * Returns a {@link LngLat} representing geographical coordinates that correspond\n * to the specified pixel coordinates.\n *\n * @param point - The pixel coordinates to unproject.\n * @returns The {@link LngLat} corresponding to `point`.\n * @example\n * ```ts\n * map.on('click', function(e) {\n * // When the map is clicked, get the geographic coordinate.\n * let coordinate = map.unproject(e.point);\n * });\n * ```\n */\n unproject(point: PointLike): LngLat {\n return this.transform.pointLocation(Point.convert(point), this.terrain);\n }\n\n /**\n * Returns true if the map is panning, zooming, rotating, or pitching due to a camera animation or user gesture.\n * @returns true if the map is moving.\n * @example\n * ```ts\n * let isMoving = map.isMoving();\n * ```\n */\n isMoving(): boolean {\n return this._moving || this.handlers?.isMoving();\n }\n\n /**\n * Returns true if the map is zooming due to a camera animation or user gesture.\n * @returns true if the map is zooming.\n * @example\n * ```ts\n * let isZooming = map.isZooming();\n * ```\n */\n isZooming(): boolean {\n return this._zooming || this.handlers?.isZooming();\n }\n\n /**\n * Returns true if the map is rotating due to a camera animation or user gesture.\n * @returns true if the map is rotating.\n * @example\n * ```ts\n * map.isRotating();\n * ```\n */\n isRotating(): boolean {\n return this._rotating || this.handlers?.isRotating();\n }\n\n _createDelegatedListener(type: keyof MapEventType | string, layerId: string, listener: Listener): {\n layer: string;\n listener: Listener;\n delegates: {[type in keyof MapEventType]?: (e: any) => void};\n } {\n if (type === 'mouseenter' || type === 'mouseover') {\n let mousein = false;\n const mousemove = (e) => {\n const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, {layers: [layerId]}) : [];\n if (!features.length) {\n mousein = false;\n } else if (!mousein) {\n mousein = true;\n listener.call(this, new MapMouseEvent(type, this, e.originalEvent, {features}));\n }\n };\n const mouseout = () => {\n mousein = false;\n };\n return {layer: layerId, listener, delegates: {mousemove, mouseout}};\n } else if (type === 'mouseleave' || type === 'mouseout') {\n let mousein = false;\n const mousemove = (e) => {\n const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, {layers: [layerId]}) : [];\n if (features.length) {\n mousein = true;\n } else if (mousein) {\n mousein = false;\n listener.call(this, new MapMouseEvent(type, this, e.originalEvent));\n }\n };\n const mouseout = (e) => {\n if (mousein) {\n mousein = false;\n listener.call(this, new MapMouseEvent(type, this, e.originalEvent));\n }\n };\n return {layer: layerId, listener, delegates: {mousemove, mouseout}};\n } else {\n const delegate = (e) => {\n const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, {layers: [layerId]}) : [];\n if (features.length) {\n // Here we need to mutate the original event, so that preventDefault works as expected.\n e.features = features;\n listener.call(this, e);\n delete e.features;\n }\n };\n return {layer: layerId, listener, delegates: {[type]: delegate}};\n }\n }\n\n /**\n * @event\n * Adds a listener for events of a specified type, optionally limited to features in a specified style layer.\n * See {@link MapEventType} and {@link MapLayerEventType} for a full list of events and their description.\n *\n * | Event | Compatible with `layerId` |\n * |------------------------|---------------------------|\n * | `mousedown` | yes |\n * | `mouseup` | yes |\n * | `mouseover` | yes |\n * | `mouseout` | yes |\n * | `mousemove` | yes |\n * | `mouseenter` | yes (required) |\n * | `mouseleave` | yes (required) |\n * | `click` | yes |\n * | `dblclick` | yes |\n * | `contextmenu` | yes |\n * | `touchstart` | yes |\n * | `touchend` | yes |\n * | `touchcancel` | yes |\n * | `wheel` | |\n * | `resize` | |\n * | `remove` | |\n * | `touchmove` | |\n * | `movestart` | |\n * | `move` | |\n * | `moveend` | |\n * | `dragstart` | |\n * | `drag` | |\n * | `dragend` | |\n * | `zoomstart` | |\n * | `zoom` | |\n * | `zoomend` | |\n * | `rotatestart` | |\n * | `rotate` | |\n * | `rotateend` | |\n * | `pitchstart` | |\n * | `pitch` | |\n * | `pitchend` | |\n * | `boxzoomstart` | |\n * | `boxzoomend` | |\n * | `boxzoomcancel` | |\n * | `webglcontextlost` | |\n * | `webglcontextrestored` | |\n * | `load` | |\n * | `render` | |\n * | `idle` | |\n * | `error` | |\n * | `data` | |\n * | `styledata` | |\n * | `sourcedata` | |\n * | `dataloading` | |\n * | `styledataloading` | |\n * | `sourcedataloading` | |\n * | `styleimagemissing` | |\n * | `dataabort` | |\n * | `sourcedataabort` | |\n *\n * @param type - The event type to listen for. Events compatible with the optional `layerId` parameter are triggered\n * when the cursor enters a visible portion of the specified layer from outside that layer or outside the map canvas.\n * @param layer - The ID of a style layer or a listener if no ID is provided. Event will only be triggered if its location\n * is within a visible feature in this layer. The event will have a `features` property containing\n * an array of the matching features. If `layer` is not supplied, the event will not have a `features` property.\n * Please note that many event types are not compatible with the optional `layer` parameter.\n * @param listener - The function to be called when the event is fired.\n * @returns `this`\n * @example\n * ```ts\n * // Set an event listener that will fire\n * // when the map has finished loading\n * map.on('load', function() {\n * // Once the map has finished loading,\n * // add a new layer\n * map.addLayer({\n * id: 'points-of-interest',\n * source: {\n * type: 'vector',\n * url: 'https://maplibre.org/maplibre-style-spec/'\n * },\n * 'source-layer': 'poi_label',\n * type: 'circle',\n * paint: {\n * // MapLibre Style Specification paint properties\n * },\n * layout: {\n * // MapLibre Style Specification layout properties\n * }\n * });\n * });\n * ```\n * @example\n * ```ts\n * // Set an event listener that will fire\n * // when a feature on the countries layer of the map is clicked\n * map.on('click', 'countries', (e) => {\n * new maplibregl.Popup()\n * .setLngLat(e.lngLat)\n * .setHTML(`Country name: ${e.features[0].properties.name}`)\n * .addTo(map);\n * });\n * ```\n * @see [Display popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n * @see [Create a hover effect](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Create a draggable marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n on(\n type: T,\n layer: string,\n listener: (ev: MapLayerEventType[T] & Object) => void,\n ): Map;\n /**\n * Overload of the `on` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The listener callback.\n * @returns `this`\n */\n on(type: T, listener: (ev: MapEventType[T] & Object) => void): this;\n /**\n * Overload of the `on` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The listener callback.\n * @returns `this`\n */\n on(type: keyof MapEventType | string, listener: Listener): this;\n on(type: keyof MapEventType | string, layerIdOrListener: string | Listener, listener?: Listener): this {\n if (listener === undefined) {\n return super.on(type, layerIdOrListener as Listener);\n }\n\n const delegatedListener = this._createDelegatedListener(type, layerIdOrListener as string, listener);\n\n this._delegatedListeners = this._delegatedListeners || {};\n this._delegatedListeners[type] = this._delegatedListeners[type] || [];\n this._delegatedListeners[type].push(delegatedListener);\n\n for (const event in delegatedListener.delegates) {\n this.on(event, delegatedListener.delegates[event]);\n }\n\n return this;\n }\n\n /**\n * Adds a listener that will be called only once to a specified event type, optionally limited to features in a specified style layer.\n *\n * @event\n * @param type - The event type to listen for; one of `'mousedown'`, `'mouseup'`, `'click'`, `'dblclick'`,\n * `'mousemove'`, `'mouseenter'`, `'mouseleave'`, `'mouseover'`, `'mouseout'`, `'contextmenu'`, `'touchstart'`,\n * `'touchend'`, or `'touchcancel'`. `mouseenter` and `mouseover` events are triggered when the cursor enters\n * a visible portion of the specified layer from outside that layer or outside the map canvas. `mouseleave`\n * and `mouseout` events are triggered when the cursor leaves a visible portion of the specified layer, or leaves\n * the map canvas.\n * @param layer - The ID of a style layer or a listener if no ID is provided. Only events whose location is within a visible\n * feature in this layer will trigger the listener. The event will have a `features` property containing\n * an array of the matching features.\n * @param listener - The function to be called when the event is fired.\n * @returns `this` if listener is provided, promise otherwise to allow easier usage of async/await\n */\n once(\n type: T,\n layer: string,\n listener?: (ev: MapLayerEventType[T] & Object) => void,\n ): this | Promise;\n /**\n * Overload of the `once` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The listener callback.\n * @returns `this`\n */\n once(type: T, listener?: (ev: MapEventType[T] & Object) => void): this | Promise;\n /**\n * Overload of the `once` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The listener callback.\n * @returns `this`\n */\n once(type: keyof MapEventType | string, listener?: Listener): this | Promise;\n once(type: keyof MapEventType | string, layerIdOrListener: string | Listener, listener?: Listener): this | Promise {\n\n if (listener === undefined) {\n return super.once(type, layerIdOrListener as Listener);\n }\n\n const delegatedListener = this._createDelegatedListener(type, layerIdOrListener as string, listener);\n\n for (const event in delegatedListener.delegates) {\n this.once(event, delegatedListener.delegates[event]);\n }\n\n return this;\n }\n\n /**\n * Removes an event listener for events previously added with `Map#on`.\n *\n * @event\n * @param type - The event type previously used to install the listener.\n * @param layer - The layer ID or listener previously used to install the listener.\n * @param listener - The function previously installed as a listener.\n * @returns `this`\n */\n off(\n type: T,\n layer: string,\n listener: (ev: MapLayerEventType[T] & Object) => void,\n ): this;\n /**\n * Overload of the `off` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The function previously installed as a listener.\n * @returns `this`\n */\n off(type: T, listener: (ev: MapEventType[T] & Object) => void): this;\n /**\n * Overload of the `off` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The function previously installed as a listener.\n * @returns `this`\n */\n off(type: keyof MapEventType | string, listener: Listener): this;\n off(type: keyof MapEventType | string, layerIdOrListener: string | Listener, listener?: Listener): this {\n if (listener === undefined) {\n return super.off(type, layerIdOrListener as Listener);\n }\n\n const removeDelegatedListener = (delegatedListeners) => {\n const listeners = delegatedListeners[type];\n for (let i = 0; i < listeners.length; i++) {\n const delegatedListener = listeners[i];\n if (delegatedListener.layer === layerIdOrListener && delegatedListener.listener === listener) {\n for (const event in delegatedListener.delegates) {\n this.off(((event as any)), delegatedListener.delegates[event]);\n }\n listeners.splice(i, 1);\n return this;\n }\n }\n };\n\n if (this._delegatedListeners && this._delegatedListeners[type]) {\n removeDelegatedListener(this._delegatedListeners);\n }\n\n return this;\n }\n\n /**\n * Returns an array of MapGeoJSONFeature objects\n * representing visible features that satisfy the query parameters.\n *\n * @param geometryOrOptions - (optional) The geometry of the query region:\n * either a single point or southwest and northeast points describing a bounding box.\n * Omitting this parameter (i.e. calling {@link Map#queryRenderedFeatures} with zero arguments,\n * or with only a `options` argument) is equivalent to passing a bounding box encompassing the entire\n * map viewport.\n * The geometryOrOptions can receive a {@link QueryRenderedFeaturesOptions} only to support a situation where the function receives only one parameter which is the options parameter.\n * @param options - (optional) Options object.\n *\n * @returns An array of MapGeoJSONFeature objects.\n *\n * The `properties` value of each returned feature object contains the properties of its source feature. For GeoJSON sources, only\n * string and numeric property values are supported (i.e. `null`, `Array`, and `Object` values are not supported).\n *\n * Each feature includes top-level `layer`, `source`, and `sourceLayer` properties. The `layer` property is an object\n * representing the style layer to which the feature belongs. Layout and paint properties in this object contain values\n * which are fully evaluated for the given zoom level and feature.\n *\n * Only features that are currently rendered are included. Some features will **not** be included, like:\n *\n * - Features from layers whose `visibility` property is `\"none\"`.\n * - Features from layers whose zoom range excludes the current zoom level.\n * - Symbol features that have been hidden due to text or icon collision.\n *\n * Features from all other layers are included, including features that may have no visible\n * contribution to the rendered result; for example, because the layer's opacity or color alpha component is set to\n * 0.\n *\n * The topmost rendered feature appears first in the returned array, and subsequent features are sorted by\n * descending z-order. Features that are rendered multiple times (due to wrapping across the antemeridian at low\n * zoom levels) are returned only once (though subject to the following caveat).\n *\n * Because features come from tiled vector data or GeoJSON data that is converted to tiles internally, feature\n * geometries may be split or duplicated across tile boundaries and, as a result, features may appear multiple\n * times in query results. For example, suppose there is a highway running through the bounding rectangle of a query.\n * The results of the query will be those parts of the highway that lie within the map tiles covering the bounding\n * rectangle, even if the highway extends into other tiles, and the portion of the highway within each map tile\n * will be returned as a separate feature. Similarly, a point feature near a tile boundary may appear in multiple\n * tiles due to tile buffering.\n *\n * @example\n * Find all features at a point\n * ```ts\n * let features = map.queryRenderedFeatures(\n * [20, 35],\n * { layers: ['my-layer-name'] }\n * );\n * ```\n *\n * @example\n * Find all features within a static bounding box\n * ```ts\n * let features = map.queryRenderedFeatures(\n * [[10, 20], [30, 50]],\n * { layers: ['my-layer-name'] }\n * );\n * ```\n *\n * @example\n * Find all features within a bounding box around a point\n * ```ts\n * let width = 10;\n * let height = 20;\n * let features = map.queryRenderedFeatures([\n * [point.x - width / 2, point.y - height / 2],\n * [point.x + width / 2, point.y + height / 2]\n * ], { layers: ['my-layer-name'] });\n * ```\n *\n * @example\n * Query all rendered features from a single layer\n * ```ts\n * let features = map.queryRenderedFeatures({ layers: ['my-layer-name'] });\n * ```\n * @see [Get features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/queryrenderedfeatures/)\n */\n queryRenderedFeatures(geometryOrOptions?: PointLike | [PointLike, PointLike] | QueryRenderedFeaturesOptions, options?: QueryRenderedFeaturesOptions): MapGeoJSONFeature[] {\n if (!this.style) {\n return [];\n }\n let queryGeometry;\n const isGeometry = geometryOrOptions instanceof Point || Array.isArray(geometryOrOptions);\n const geometry = isGeometry ? geometryOrOptions : [[0, 0], [this.transform.width, this.transform.height]];\n options = options || (isGeometry ? {} : geometryOrOptions) || {};\n\n if (geometry instanceof Point || typeof geometry[0] === 'number') {\n queryGeometry = [Point.convert(geometry as PointLike)];\n } else {\n const tl = Point.convert(geometry[0] as PointLike);\n const br = Point.convert(geometry[1] as PointLike);\n queryGeometry = [tl, new Point(br.x, tl.y), br, new Point(tl.x, br.y), tl];\n }\n\n return this.style.queryRenderedFeatures(queryGeometry, options, this.transform);\n }\n\n /**\n * Returns an array of MapGeoJSONFeature objects\n * representing features within the specified vector tile or GeoJSON source that satisfy the query parameters.\n *\n * @param sourceId - The ID of the vector tile or GeoJSON source to query.\n * @param parameters - The options object.\n * @returns An array of MapGeoJSONFeature objects.\n *\n * In contrast to {@link Map#queryRenderedFeatures}, this function returns all features matching the query parameters,\n * whether or not they are rendered by the current style (i.e. visible). The domain of the query includes all currently-loaded\n * vector tiles and GeoJSON source tiles: this function does not check tiles outside the currently\n * visible viewport.\n *\n * Because features come from tiled vector data or GeoJSON data that is converted to tiles internally, feature\n * geometries may be split or duplicated across tile boundaries and, as a result, features may appear multiple\n * times in query results. For example, suppose there is a highway running through the bounding rectangle of a query.\n * The results of the query will be those parts of the highway that lie within the map tiles covering the bounding\n * rectangle, even if the highway extends into other tiles, and the portion of the highway within each map tile\n * will be returned as a separate feature. Similarly, a point feature near a tile boundary may appear in multiple\n * tiles due to tile buffering.\n *\n * @example\n * Find all features in one source layer in a vector source\n * ```ts\n * let features = map.querySourceFeatures('your-source-id', {\n * sourceLayer: 'your-source-layer'\n * });\n * ```\n *\n */\n querySourceFeatures(sourceId: string, parameters?: QuerySourceFeatureOptions | null): MapGeoJSONFeature[] {\n return this.style.querySourceFeatures(sourceId, parameters);\n }\n\n /**\n * Updates the map's MapLibre style object with a new value.\n *\n * If a style is already set when this is used and options.diff is set to true, the map renderer will attempt to compare the given style\n * against the map's current state and perform only the changes necessary to make the map style match the desired state. Changes in sprites\n * (images used for icons and patterns) and glyphs (fonts for label text) **cannot** be diffed. If the sprites or fonts used in the current\n * style and the given style are different in any way, the map renderer will force a full update, removing the current style and building\n * the given one from scratch.\n *\n *\n * @param style - A JSON object conforming to the schema described in the\n * [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/), or a URL to such JSON.\n * @param options - The options object.\n * @returns `this`\n *\n * @example\n * ```ts\n * map.setStyle(\"https://demotiles.maplibre.org/style.json\");\n *\n * map.setStyle('https://demotiles.maplibre.org/style.json', {\n * transformStyle: (previousStyle, nextStyle) => ({\n * ...nextStyle,\n * sources: {\n * ...nextStyle.sources,\n * // copy a source from previous style\n * 'osm': previousStyle.sources.osm\n * },\n * layers: [\n * // background layer\n * nextStyle.layers[0],\n * // copy a layer from previous style\n * previousStyle.layers[0],\n * // other layers from the next style\n * ...nextStyle.layers.slice(1).map(layer => {\n * // hide the layers we don't need from demotiles style\n * if (layer.id.startsWith('geolines')) {\n * layer.layout = {...layer.layout || {}, visibility: 'none'};\n * // filter out US polygons\n * } else if (layer.id.startsWith('coastline') || layer.id.startsWith('countries')) {\n * layer.filter = ['!=', ['get', 'ADM0_A3'], 'USA'];\n * }\n * return layer;\n * })\n * ]\n * })\n * });\n * ```\n */\n setStyle(style: StyleSpecification | string | null, options?: StyleSwapOptions & StyleOptions): this {\n options = extend({},\n {\n localIdeographFontFamily: this._localIdeographFontFamily,\n validate: this._validateStyle\n }, options);\n\n if ((options.diff !== false && options.localIdeographFontFamily === this._localIdeographFontFamily) && this.style && style) {\n this._diffStyle(style, options);\n return this;\n } else {\n this._localIdeographFontFamily = options.localIdeographFontFamily;\n return this._updateStyle(style, options);\n }\n }\n\n /**\n * Updates the requestManager's transform request with a new function\n *\n * @param transformRequest - A callback run before the Map makes a request for an external URL. The callback can be used to modify the url, set headers, or set the credentials property for cross-origin requests.\n * Expected to return an object with a `url` property and optionally `headers` and `credentials` properties\n *\n * @returns `this`\n *\n * @example\n * ```ts\n * map.setTransformRequest((url: string, resourceType: string) => {});\n * ```\n */\n setTransformRequest(transformRequest: RequestTransformFunction): this {\n this._requestManager.setTransformRequest(transformRequest);\n return this;\n }\n\n _getUIString(key: string) {\n const str = this._locale[key];\n if (str == null) {\n throw new Error(`Missing UI string '${key}'`);\n }\n\n return str;\n }\n\n _updateStyle(style: StyleSpecification | string | null, options?: StyleSwapOptions & StyleOptions) {\n // transformStyle relies on having previous style serialized, if it is not loaded yet, delay _updateStyle until previous style is loaded\n if (options.transformStyle && this.style && !this.style._loaded) {\n this.style.once('style.load', () => this._updateStyle(style, options));\n return;\n }\n\n const previousStyle = this.style && options.transformStyle ? this.style.serialize() : undefined;\n if (this.style) {\n this.style.setEventedParent(null);\n\n // Only release workers when map is getting disposed\n this.style._remove(!style);\n }\n\n if (!style) {\n delete this.style;\n return this;\n } else {\n this.style = new Style(this, options || {});\n }\n\n this.style.setEventedParent(this, {style: this.style});\n\n if (typeof style === 'string') {\n this.style.loadURL(style, options, previousStyle);\n } else {\n this.style.loadJSON(style, options, previousStyle);\n }\n\n return this;\n }\n\n _lazyInitEmptyStyle() {\n if (!this.style) {\n this.style = new Style(this, {});\n this.style.setEventedParent(this, {style: this.style});\n this.style.loadEmpty();\n }\n }\n\n _diffStyle(style: StyleSpecification | string, options?: StyleSwapOptions & StyleOptions) {\n if (typeof style === 'string') {\n const url = style;\n const request = this._requestManager.transformRequest(url, ResourceType.Style);\n getJSON(request, (error?: Error | null, json?: any | null) => {\n if (error) {\n this.fire(new ErrorEvent(error));\n } else if (json) {\n this._updateDiff(json, options);\n }\n });\n } else if (typeof style === 'object') {\n this._updateDiff(style, options);\n }\n }\n\n _updateDiff(style: StyleSpecification, options?: StyleSwapOptions & StyleOptions) {\n try {\n if (this.style.setState(style, options)) {\n this._update(true);\n }\n } catch (e) {\n warnOnce(\n `Unable to perform style diff: ${e.message || e.error || e}. Rebuilding the style from scratch.`\n );\n this._updateStyle(style, options);\n }\n }\n\n /**\n * Returns the map's MapLibre style object, a JSON object which can be used to recreate the map's style.\n *\n * @returns The map's style JSON object.\n *\n * @example\n * ```ts\n * let styleJson = map.getStyle();\n * ```\n *\n */\n getStyle(): StyleSpecification {\n if (this.style) {\n return this.style.serialize();\n }\n }\n\n /**\n * Returns a Boolean indicating whether the map's style is fully loaded.\n *\n * @returns A Boolean indicating whether the style is fully loaded.\n *\n * @example\n * ```ts\n * let styleLoadStatus = map.isStyleLoaded();\n * ```\n */\n isStyleLoaded(): boolean | void {\n if (!this.style) return warnOnce('There is no style added to the map.');\n return this.style.loaded();\n }\n\n /**\n * Adds a source to the map's style.\n *\n * Events triggered:\n *\n * Triggers the `source.add` event.\n *\n * @param id - The ID of the source to add. Must not conflict with existing sources.\n * @param source - The source object, conforming to the\n * MapLibre Style Specification's [source definition](https://maplibre.org/maplibre-style-spec/sources) or\n * {@link CanvasSourceSpecification}.\n * @returns `this`\n * @example\n * ```ts\n * map.addSource('my-data', {\n * type: 'vector',\n * url: 'https://demotiles.maplibre.org/tiles/tiles.json'\n * });\n * ```\n * @example\n * ```ts\n * map.addSource('my-data', {\n * \"type\": \"geojson\",\n * \"data\": {\n * \"type\": \"Feature\",\n * \"geometry\": {\n * \"type\": \"Point\",\n * \"coordinates\": [-77.0323, 38.9131]\n * },\n * \"properties\": {\n * \"title\": \"Mapbox DC\",\n * \"marker-symbol\": \"monument\"\n * }\n * }\n * });\n * ```\n * @see GeoJSON source: [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/)\n */\n addSource(id: string, source: SourceSpecification): this {\n this._lazyInitEmptyStyle();\n this.style.addSource(id, source);\n return this._update(true);\n }\n\n /**\n * Returns a Boolean indicating whether the source is loaded. Returns `true` if the source with\n * the given ID in the map's style has no outstanding network requests, otherwise `false`.\n *\n * A {@link ErrorEvent} event will be fired if there is no source wit the specified ID.\n *\n * @param id - The ID of the source to be checked.\n * @returns A Boolean indicating whether the source is loaded.\n * @example\n * ```ts\n * let sourceLoaded = map.isSourceLoaded('bathymetry-data');\n * ```\n */\n isSourceLoaded(id: string): boolean {\n const source = this.style && this.style.sourceCaches[id];\n if (source === undefined) {\n this.fire(new ErrorEvent(new Error(`There is no source with ID '${id}'`)));\n return;\n }\n return source.loaded();\n }\n\n /**\n * Loads a 3D terrain mesh, based on a \"raster-dem\" source.\n *\n * Triggers the `terrain` event.\n *\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setTerrain({ source: 'terrain' });\n * ```\n */\n setTerrain(options: TerrainSpecification | null): this {\n this.style._checkLoaded();\n\n // clear event handlers\n if (this._terrainDataCallback) this.style.off('data', this._terrainDataCallback);\n\n if (!options) {\n // remove terrain\n if (this.terrain) this.terrain.sourceCache.destruct();\n this.terrain = null;\n if (this.painter.renderToTexture) this.painter.renderToTexture.destruct();\n this.painter.renderToTexture = null;\n this.transform._minEleveationForCurrentTile = 0;\n this.transform.elevation = 0;\n } else {\n // add terrain\n const sourceCache = this.style.sourceCaches[options.source];\n if (!sourceCache) throw new Error(`cannot load terrain, because there exists no source with ID: ${options.source}`);\n // Warn once if user is using the same source for hillshade and terrain\n for (const index in this.style._layers) {\n const thisLayer = this.style._layers[index];\n if (thisLayer.type === 'hillshade' && thisLayer.source === options.source) {\n warnOnce('You are using the same source for a hillshade layer and for 3D terrain. Please consider using two separate sources to improve rendering quality.');\n }\n }\n this.terrain = new Terrain(this.painter, sourceCache, options);\n this.painter.renderToTexture = new RenderToTexture(this.painter, this.terrain);\n this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n this._terrainDataCallback = e => {\n if (e.dataType === 'style') {\n this.terrain.sourceCache.freeRtt();\n } else if (e.dataType === 'source' && e.tile) {\n if (e.sourceId === options.source && !this._elevationFreeze) {\n this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n }\n this.terrain.sourceCache.freeRtt(e.tile.tileID);\n }\n };\n this.style.on('data', this._terrainDataCallback);\n }\n\n this.fire(new Event('terrain', {terrain: options}));\n return this;\n }\n\n /**\n * Get the terrain-options if terrain is loaded\n * @returns the TerrainSpecification passed to setTerrain\n * @example\n * ```ts\n * map.getTerrain(); // { source: 'terrain' };\n * ```\n */\n getTerrain(): TerrainSpecification | null {\n return this.terrain?.options ?? null;\n }\n\n /**\n * Returns a Boolean indicating whether all tiles in the viewport from all sources on\n * the style are loaded.\n *\n * @returns A Boolean indicating whether all tiles are loaded.\n * @example\n * ```ts\n * let tilesLoaded = map.areTilesLoaded();\n * ```\n */\n areTilesLoaded(): boolean {\n const sources = this.style && this.style.sourceCaches;\n for (const id in sources) {\n const source = sources[id];\n const tiles = source._tiles;\n for (const t in tiles) {\n const tile = tiles[t];\n if (!(tile.state === 'loaded' || tile.state === 'errored')) return false;\n }\n }\n return true;\n }\n\n /**\n * Adds a [custom source type](#Custom Sources), making it available for use with\n * {@link Map#addSource}.\n * @param name - The name of the source type; source definition objects use this name in the `{type: ...}` field.\n * @param SourceType - A {@link Source} constructor.\n * @param callback - Called when the source type is ready or with an error argument if there is an error.\n */\n addSourceType(name: string, SourceType: SourceClass, callback: Callback) {\n this._lazyInitEmptyStyle();\n return this.style.addSourceType(name, SourceType, callback);\n }\n\n /**\n * Removes a source from the map's style.\n *\n * @param id - The ID of the source to remove.\n * @returns `this`\n * @example\n * ```ts\n * map.removeSource('bathymetry-data');\n * ```\n */\n removeSource(id: string): Map {\n this.style.removeSource(id);\n return this._update(true);\n }\n\n /**\n * Returns the source with the specified ID in the map's style.\n *\n * This method is often used to update a source using the instance members for the relevant\n * source type as defined in [Sources](#sources).\n * For example, setting the `data` for a GeoJSON source or updating the `url` and `coordinates`\n * of an image source.\n *\n * @param id - The ID of the source to get.\n * @returns The style source with the specified ID or `undefined` if the ID\n * corresponds to no existing sources.\n * The shape of the object varies by source type.\n * A list of options for each source type is available on the MapLibre Style Specification's\n * [Sources](https://maplibre.org/maplibre-style-spec/sources/) page.\n * @example\n * ```ts\n * let sourceObject = map.getSource('points');\n * ```\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n * @see [Animate a point](https://maplibre.org/maplibre-gl-js/docs/examples/animate-point-along-line/)\n * @see [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/)\n */\n getSource(id: string): Source | undefined {\n return this.style.getSource(id);\n }\n\n /**\n * Add an image to the style. This image can be displayed on the map like any other icon in the style's\n * sprite using the image's ID with\n * [`icon-image`](https://maplibre.org/maplibre-style-spec/layers/#layout-symbol-icon-image),\n * [`background-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-background-background-pattern),\n * [`fill-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-fill-fill-pattern),\n * or [`line-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-line-line-pattern).\n *\n * A {@link ErrorEvent} event will be fired if the image parameter is invalid or there is not enough space in the sprite to add this image.\n *\n * @param id - The ID of the image.\n * @param image - The image as an `HTMLImageElement`, `ImageData`, `ImageBitmap` or object with `width`, `height`, and `data`\n * properties with the same format as `ImageData`.\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * // If the style's sprite does not already contain an image with ID 'cat',\n * // add the image 'cat-icon.png' to the style's sprite with the ID 'cat'.\n * map.loadImage('https://upload.wikimedia.org/wikipedia/commons/thumb/6/60/Cat_silhouette.svg/400px-Cat_silhouette.svg.png', function(error, image) {\n * if (error) throw error;\n * if (!map.hasImage('cat')) map.addImage('cat', image);\n * });\n *\n * // Add a stretchable image that can be used with `icon-text-fit`\n * // In this example, the image is 600px wide by 400px high.\n * map.loadImage('https://upload.wikimedia.org/wikipedia/commons/8/89/Black_and_White_Boxed_%28bordered%29.png', function(error, image) {\n * if (error) throw error;\n * if (!map.hasImage('border-image')) {\n * map.addImage('border-image', image, {\n * content: [16, 16, 300, 384], // place text over left half of image, avoiding the 16px border\n * stretchX: [[16, 584]], // stretch everything horizontally except the 16px border\n * stretchY: [[16, 384]], // stretch everything vertically except the 16px border\n * });\n * }\n * });\n * ```\n * @see Use `HTMLImageElement`: [Add an icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image/)\n * @see Use `ImageData`: [Add a generated icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-generated/)\n */\n addImage(id: string,\n image: HTMLImageElement | ImageBitmap | ImageData | {\n width: number;\n height: number;\n data: Uint8Array | Uint8ClampedArray;\n } | StyleImageInterface,\n options: Partial = {}): this {\n const {\n pixelRatio = 1,\n sdf = false,\n stretchX,\n stretchY,\n content\n } = options;\n this._lazyInitEmptyStyle();\n const version = 0;\n\n if (image instanceof HTMLImageElement || isImageBitmap(image)) {\n const {width, height, data} = browser.getImageData(image);\n this.style.addImage(id, {data: new RGBAImage({width, height}, data), pixelRatio, stretchX, stretchY, content, sdf, version});\n } else if (image.width === undefined || image.height === undefined) {\n return this.fire(new ErrorEvent(new Error(\n 'Invalid arguments to map.addImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, ' +\n 'or object with `width`, `height`, and `data` properties with the same format as `ImageData`')));\n } else {\n const {width, height, data} = image as ImageData;\n const userImage = (image as any as StyleImageInterface);\n\n this.style.addImage(id, {\n data: new RGBAImage({width, height}, new Uint8Array(data)),\n pixelRatio,\n stretchX,\n stretchY,\n content,\n sdf,\n version,\n userImage\n });\n\n if (userImage.onAdd) {\n userImage.onAdd(this, id);\n }\n return this;\n }\n }\n\n /**\n * Update an existing image in a style. This image can be displayed on the map like any other icon in the style's\n * sprite using the image's ID with\n * [`icon-image`](https://maplibre.org/maplibre-style-spec/layers/#layout-symbol-icon-image),\n * [`background-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-background-background-pattern),\n * [`fill-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-fill-fill-pattern),\n * or [`line-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-line-line-pattern).\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param id - The ID of the image.\n * @param image - The image as an `HTMLImageElement`, `ImageData`, `ImageBitmap` or object with `width`, `height`, and `data`\n * properties with the same format as `ImageData`.\n * @returns `this`\n * @example\n * ```ts\n * // If an image with the ID 'cat' already exists in the style's sprite,\n * // replace that image with a new image, 'other-cat-icon.png'.\n * if (map.hasImage('cat')) map.updateImage('cat', './other-cat-icon.png');\n * ```\n */\n updateImage(id: string,\n image: HTMLImageElement | ImageBitmap | ImageData | {\n width: number;\n height: number;\n data: Uint8Array | Uint8ClampedArray;\n } | StyleImageInterface): this {\n\n const existingImage = this.style.getImage(id);\n if (!existingImage) {\n return this.fire(new ErrorEvent(new Error(\n 'The map has no image with that id. If you are adding a new image use `map.addImage(...)` instead.')));\n }\n const imageData = (image instanceof HTMLImageElement || isImageBitmap(image)) ?\n browser.getImageData(image) :\n image;\n const {width, height, data} = imageData;\n\n if (width === undefined || height === undefined) {\n return this.fire(new ErrorEvent(new Error(\n 'Invalid arguments to map.updateImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, ' +\n 'or object with `width`, `height`, and `data` properties with the same format as `ImageData`')));\n }\n\n if (width !== existingImage.data.width || height !== existingImage.data.height) {\n return this.fire(new ErrorEvent(new Error(\n 'The width and height of the updated image must be that same as the previous version of the image')));\n }\n\n const copy = !(image instanceof HTMLImageElement || isImageBitmap(image));\n existingImage.data.replace(data, copy);\n\n this.style.updateImage(id, existingImage);\n return this;\n }\n\n /**\n * Returns an image, specified by ID, currently available in the map.\n * This includes both images from the style's original sprite\n * and any images that have been added at runtime using {@link Map#addImage}.\n *\n * @param id - The ID of the image.\n * @returns An image in the map with the specified ID.\n *\n * @example\n * ```ts\n * let coffeeShopIcon = map.getImage(\"coffee_cup\");\n * ```\n */\n getImage(id: string): StyleImage {\n return this.style.getImage(id);\n }\n\n /**\n * Check whether or not an image with a specific ID exists in the style. This checks both images\n * in the style's original sprite and any images\n * that have been added at runtime using {@link Map#addImage}.\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param id - The ID of the image.\n *\n * @returns A Boolean indicating whether the image exists.\n * @example\n * Check if an image with the ID 'cat' exists in the style's sprite.\n * ```ts\n * let catIconExists = map.hasImage('cat');\n * ```\n */\n hasImage(id: string): boolean {\n if (!id) {\n this.fire(new ErrorEvent(new Error('Missing required image id')));\n return false;\n }\n\n return !!this.style.getImage(id);\n }\n\n /**\n * Remove an image from a style. This can be an image from the style's original\n * sprite or any images\n * that have been added at runtime using {@link Map#addImage}.\n *\n * @param id - The ID of the image.\n *\n * @example\n * ```ts\n * // If an image with the ID 'cat' exists in\n * // the style's sprite, remove it.\n * if (map.hasImage('cat')) map.removeImage('cat');\n * ```\n */\n removeImage(id: string) {\n this.style.removeImage(id);\n }\n\n /**\n * Load an image from an external URL to be used with {@link Map#addImage}. External\n * domains must support [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/Access_control_CORS).\n *\n * @param url - The URL of the image file. Image file must be in png, webp, or jpg format.\n * @param callback - Expecting `callback(error, data)`. Called when the image has loaded or with an error argument if there is an error.\n *\n * @example\n * Load an image from an external URL.\n * ```ts\n * map.loadImage('http://placekitten.com/50/50', function(error, image) {\n * if (error) throw error;\n * // Add the loaded image to the style's sprite with the ID 'kitten'.\n * map.addImage('kitten', image);\n * });\n * ```\n * @see [Add an icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image/)\n */\n loadImage(url: string, callback: GetImageCallback) {\n ImageRequest.getImage(this._requestManager.transformRequest(url, ResourceType.Image), callback);\n }\n\n /**\n * Returns an Array of strings containing the IDs of all images currently available in the map.\n * This includes both images from the style's original sprite\n * and any images that have been added at runtime using {@link Map#addImage}.\n *\n * @returns An Array of strings containing the names of all sprites/images currently available in the map.\n *\n * @example\n * ```ts\n * let allImages = map.listImages();\n * ```\n */\n listImages(): Array {\n return this.style.listImages();\n }\n\n /**\n * Adds a [MapLibre style layer](https://maplibre.org/maplibre-style-spec/layers)\n * to the map's style.\n *\n * A layer defines how data from a specified source will be styled. Read more about layer types\n * and available paint and layout properties in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/layers).\n *\n * @param layer - The layer to add,\n * conforming to either the MapLibre Style Specification's [layer definition](https://maplibre.org/maplibre-style-spec/layers) or,\n * less commonly, the {@link CustomLayerInterface} specification. Can also be a layer definition with an embedded source definition.\n * The MapLibre Style Specification's layer definition is appropriate for most layers.\n *\n * @param beforeId - The ID of an existing layer to insert the new layer before,\n * resulting in the new layer appearing visually beneath the existing layer.\n * If this argument is not specified, the layer will be appended to the end of the layers array\n * and appear visually above all other layers.\n *\n * @returns `this`\n *\n * @example\n * Add a circle layer with a vector source\n * ```ts\n * map.addLayer({\n * id: 'points-of-interest',\n * source: {\n * type: 'vector',\n * url: 'https://demotiles.maplibre.org/tiles/tiles.json'\n * },\n * 'source-layer': 'poi_label',\n * type: 'circle',\n * paint: {\n * // MapLibre Style Specification paint properties\n * },\n * layout: {\n * // MapLibre Style Specification layout properties\n * }\n * });\n * ```\n *\n * @example\n * Define a source before using it to create a new layer\n * ```ts\n * map.addSource('state-data', {\n * type: 'geojson',\n * data: 'path/to/data.geojson'\n * });\n *\n * map.addLayer({\n * id: 'states',\n * // References the GeoJSON source defined above\n * // and does not require a `source-layer`\n * source: 'state-data',\n * type: 'symbol',\n * layout: {\n * // Set the label content to the\n * // feature's `name` property\n * text-field: ['get', 'name']\n * }\n * });\n * ```\n *\n * @example\n * Add a new symbol layer before an existing layer\n * ```ts\n * map.addLayer({\n * id: 'states',\n * // References a source that's already been defined\n * source: 'state-data',\n * type: 'symbol',\n * layout: {\n * // Set the label content to the\n * // feature's `name` property\n * text-field: ['get', 'name']\n * }\n * // Add the layer before the existing `cities` layer\n * }, 'cities');\n * ```\n * @see [Create and style clusters](https://maplibre.org/maplibre-gl-js/docs/examples/cluster/)\n * @see [Add a vector tile source](https://maplibre.org/maplibre-gl-js/docs/examples/vector-source/)\n * @see [Add a WMS source](https://maplibre.org/maplibre-gl-js/docs/examples/wms/)\n */\n addLayer(layer: AddLayerObject, beforeId?: string) {\n this._lazyInitEmptyStyle();\n this.style.addLayer(layer, beforeId);\n return this._update(true);\n }\n\n /**\n * Moves a layer to a different z-position.\n *\n * @param id - The ID of the layer to move.\n * @param beforeId - The ID of an existing layer to insert the new layer before. When viewing the map, the `id` layer will appear beneath the `beforeId` layer. If `beforeId` is omitted, the layer will be appended to the end of the layers array and appear above all other layers on the map.\n * @returns `this`\n *\n * @example\n * Move a layer with ID 'polygon' before the layer with ID 'country-label'. The `polygon` layer will appear beneath the `country-label` layer on the map.\n * ```ts\n * map.moveLayer('polygon', 'country-label');\n * ```\n */\n moveLayer(id: string, beforeId?: string): this {\n this.style.moveLayer(id, beforeId);\n return this._update(true);\n }\n\n /**\n * Removes the layer with the given ID from the map's style.\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param id - The ID of the layer to remove\n * @returns `this`\n *\n * @example\n * If a layer with ID 'state-data' exists, remove it.\n * ```ts\n * if (map.getLayer('state-data')) map.removeLayer('state-data');\n * ```\n */\n removeLayer(id: string): this {\n this.style.removeLayer(id);\n return this._update(true);\n }\n\n /**\n * Returns the layer with the specified ID in the map's style.\n *\n * @param id - The ID of the layer to get.\n * @returns The layer with the specified ID, or `undefined`\n * if the ID corresponds to no existing layers.\n *\n * @example\n * ```ts\n * let stateDataLayer = map.getLayer('state-data');\n * ```\n * @see [Filter symbols by toggling a list](https://maplibre.org/maplibre-gl-js/docs/examples/filter-markers/)\n * @see [Filter symbols by text input](https://maplibre.org/maplibre-gl-js/docs/examples/filter-markers-by-input/)\n */\n getLayer(id: string): StyleLayer | undefined {\n return this.style.getLayer(id);\n }\n\n /**\n * Return the ids of all layers currently in the style, including custom layers, in order.\n *\n * @returns ids of layers, in order\n *\n * @example\n * ```ts\n * const orderedLayerIds = map.getLayersOrder();\n * ```\n */\n getLayersOrder(): string[] {\n return this.style.getLayersOrder();\n }\n\n /**\n * Sets the zoom extent for the specified style layer. The zoom extent includes the\n * [minimum zoom level](https://maplibre.org/maplibre-style-spec/layers/#minzoom)\n * and [maximum zoom level](https://maplibre.org/maplibre-style-spec/layers/#maxzoom))\n * at which the layer will be rendered.\n *\n * Note: For style layers using vector sources, style layers cannot be rendered at zoom levels lower than the\n * minimum zoom level of the _source layer_ because the data does not exist at those zoom levels. If the minimum\n * zoom level of the source layer is higher than the minimum zoom level defined in the style layer, the style\n * layer will not be rendered at all zoom levels in the zoom range.\n *\n * @param layerId - The ID of the layer to which the zoom extent will be applied.\n * @param minzoom - The minimum zoom to set (0-24).\n * @param maxzoom - The maximum zoom to set (0-24).\n * @returns `this`\n *\n * @example\n * ```ts\n * map.setLayerZoomRange('my-layer', 2, 5);\n * ```\n */\n setLayerZoomRange(layerId: string, minzoom: number, maxzoom: number): this {\n this.style.setLayerZoomRange(layerId, minzoom, maxzoom);\n return this._update(true);\n }\n\n /**\n * Sets the filter for the specified style layer.\n *\n * Filters control which features a style layer renders from its source.\n * Any feature for which the filter expression evaluates to `true` will be\n * rendered on the map. Those that are false will be hidden.\n *\n * Use `setFilter` to show a subset of your source data.\n *\n * To clear the filter, pass `null` or `undefined` as the second parameter.\n *\n * @param layerId - The ID of the layer to which the filter will be applied.\n * @param filter - The filter, conforming to the MapLibre Style Specification's\n * [filter definition](https://maplibre.org/maplibre-style-spec/layers/#filter). If `null` or `undefined` is provided, the function removes any existing filter from the layer.\n * @param options - Options object.\n * @returns `this`\n *\n * @example\n * Display only features with the 'name' property 'USA'\n * ```ts\n * map.setFilter('my-layer', ['==', ['get', 'name'], 'USA']);\n * ```\n * @example\n * Display only features with five or more 'available-spots'\n * ```ts\n * map.setFilter('bike-docks', ['>=', ['get', 'available-spots'], 5]);\n * ```\n * @example\n * Remove the filter for the 'bike-docks' style layer\n * ```ts\n * map.setFilter('bike-docks', null);\n * ```\n * @see [Create a timeline animation](https://maplibre.org/maplibre-gl-js/docs/examples/timeline-animation/)\n */\n setFilter(layerId: string, filter?: FilterSpecification | null, options: StyleSetterOptions = {}) {\n this.style.setFilter(layerId, filter, options);\n return this._update(true);\n }\n\n /**\n * Returns the filter applied to the specified style layer.\n *\n * @param layerId - The ID of the style layer whose filter to get.\n * @returns The layer's filter.\n */\n getFilter(layerId: string): FilterSpecification | void {\n return this.style.getFilter(layerId);\n }\n\n /**\n * Sets the value of a paint property in the specified style layer.\n *\n * @param layerId - The ID of the layer to set the paint property in.\n * @param name - The name of the paint property to set.\n * @param value - The value of the paint property to set.\n * Must be of a type appropriate for the property, as defined in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/).\n * Pass `null` to unset the existing value.\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setPaintProperty('my-layer', 'fill-color', '#faafee');\n * ```\n * @see [Change a layer's color with buttons](https://maplibre.org/maplibre-gl-js/docs/examples/color-switcher/)\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n setPaintProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}): this {\n this.style.setPaintProperty(layerId, name, value, options);\n return this._update(true);\n }\n\n /**\n * Returns the value of a paint property in the specified style layer.\n *\n * @param layerId - The ID of the layer to get the paint property from.\n * @param name - The name of a paint property to get.\n * @returns The value of the specified paint property.\n */\n getPaintProperty(layerId: string, name: string) {\n return this.style.getPaintProperty(layerId, name);\n }\n\n /**\n * Sets the value of a layout property in the specified style layer.\n *\n * @param layerId - The ID of the layer to set the layout property in.\n * @param name - The name of the layout property to set.\n * @param value - The value of the layout property. Must be of a type appropriate for the property, as defined in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/).\n * @param options - The options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setLayoutProperty('my-layer', 'visibility', 'none');\n * ```\n */\n setLayoutProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}): this {\n this.style.setLayoutProperty(layerId, name, value, options);\n return this._update(true);\n }\n\n /**\n * Returns the value of a layout property in the specified style layer.\n *\n * @param layerId - The ID of the layer to get the layout property from.\n * @param name - The name of the layout property to get.\n * @returns The value of the specified layout property.\n */\n getLayoutProperty(layerId: string, name: string) {\n return this.style.getLayoutProperty(layerId, name);\n }\n\n /**\n * Sets the value of the style's glyphs property.\n *\n * @param glyphsUrl - Glyph URL to set. Must conform to the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/glyphs/).\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setGlyphs('https://demotiles.maplibre.org/font/{fontstack}/{range}.pbf');\n * ```\n */\n setGlyphs(glyphsUrl: string | null, options: StyleSetterOptions = {}): this {\n this._lazyInitEmptyStyle();\n this.style.setGlyphs(glyphsUrl, options);\n return this._update(true);\n }\n\n /**\n * Returns the value of the style's glyphs URL\n *\n * @returns glyphs Style's glyphs url\n */\n getGlyphs(): string | null {\n return this.style.getGlyphsUrl();\n }\n\n /**\n * Adds a sprite to the map's style. Fires the `style` event.\n *\n * @param id - The ID of the sprite to add. Must not conflict with existing sprites.\n * @param url - The URL to load the sprite from\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.addSprite('sprite-two', 'http://example.com/sprite-two');\n * ```\n */\n addSprite(id: string, url: string, options: StyleSetterOptions = {}): this {\n this._lazyInitEmptyStyle();\n this.style.addSprite(id, url, options, (err) => {\n if (!err) {\n this._update(true);\n }\n });\n return this;\n }\n\n /**\n * Removes the sprite from the map's style. Fires the `style` event.\n *\n * @param id - The ID of the sprite to remove. If the sprite is declared as a single URL, the ID must be \"default\".\n * @returns `this`\n * @example\n * ```ts\n * map.removeSprite('sprite-two');\n * map.removeSprite('default');\n * ```\n */\n removeSprite(id: string) {\n this._lazyInitEmptyStyle();\n this.style.removeSprite(id);\n return this._update(true);\n }\n\n /**\n * Returns the as-is value of the style's sprite.\n *\n * @returns style's sprite list of id-url pairs\n */\n getSprite(): {id: string; url: string}[] {\n return this.style.getSprite();\n }\n\n /**\n * Sets the value of the style's sprite property.\n *\n * @param spriteUrl - Sprite URL to set.\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setSprite('YOUR_SPRITE_URL');\n * ```\n */\n setSprite(spriteUrl: string | null, options: StyleSetterOptions = {}) {\n this._lazyInitEmptyStyle();\n this.style.setSprite(spriteUrl, options, (err) => {\n if (!err) {\n this._update(true);\n }\n });\n return this;\n }\n\n /**\n * Sets the any combination of light values.\n *\n * @param light - Light properties to set. Must conform to the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/light).\n * @param options - Options object.\n * @returns `this`\n *\n * @example\n * ```ts\n * let layerVisibility = map.getLayoutProperty('my-layer', 'visibility');\n * ```\n */\n setLight(light: LightSpecification, options: StyleSetterOptions = {}) {\n this._lazyInitEmptyStyle();\n this.style.setLight(light, options);\n return this._update(true);\n }\n\n /**\n * Returns the value of the light object.\n *\n * @returns light Light properties of the style.\n */\n getLight(): LightSpecification {\n return this.style.getLight();\n }\n\n /**\n * Sets the `state` of a feature.\n * A feature's `state` is a set of user-defined key-value pairs that are assigned to a feature at runtime.\n * When using this method, the `state` object is merged with any existing key-value pairs in the feature's state.\n * Features are identified by their `feature.id` attribute, which can be any number or string.\n *\n * This method can only be used with sources that have a `feature.id` attribute. The `feature.id` attribute can be defined in three ways:\n * - For vector or GeoJSON sources, including an `id` attribute in the original data file.\n * - For vector or GeoJSON sources, using the [`promoteId`](https://maplibre.org/maplibre-style-spec/sources/#vector-promoteId) option at the time the source is defined.\n * - For GeoJSON sources, using the [`generateId`](https://maplibre.org/maplibre-style-spec/sources/#geojson-generateId) option to auto-assign an `id` based on the feature's index in the source data. If you change feature data using `map.getSource('some id').setData(..)`, you may need to re-apply state taking into account updated `id` values.\n *\n * _Note: You can use the [`feature-state` expression](https://maplibre.org/maplibre-style-spec/expressions/#feature-state) to access the values in a feature's state object for the purposes of styling._\n *\n * @param feature - Feature identifier. Feature objects returned from\n * {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers.\n * @param state - A set of key-value pairs. The values should be valid JSON types.\n * @returns `this`\n *\n * @example\n * ```ts\n * // When the mouse moves over the `my-layer` layer, update\n * // the feature state for the feature under the mouse\n * map.on('mousemove', 'my-layer', function(e) {\n * if (e.features.length > 0) {\n * map.setFeatureState({\n * source: 'my-source',\n * sourceLayer: 'my-source-layer',\n * id: e.features[0].id,\n * }, {\n * hover: true\n * });\n * }\n * });\n * ```\n * @see [Create a hover effect](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n */\n setFeatureState(feature: FeatureIdentifier, state: any): this {\n this.style.setFeatureState(feature, state);\n return this._update();\n }\n\n /**\n * Removes the `state` of a feature, setting it back to the default behavior.\n * If only a `target.source` is specified, it will remove the state for all features from that source.\n * If `target.id` is also specified, it will remove all keys for that feature's state.\n * If `key` is also specified, it removes only that key from that feature's state.\n * Features are identified by their `feature.id` attribute, which can be any number or string.\n *\n * @param target - Identifier of where to remove state. It can be a source, a feature, or a specific key of feature.\n * Feature objects returned from {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers.\n * @param key - (optional) The key in the feature state to reset.\n * @returns `this`\n * @example\n * Reset the entire state object for all features in the `my-source` source\n * ```ts\n * map.removeFeatureState({\n * source: 'my-source'\n * });\n * ```\n *\n * @example\n * When the mouse leaves the `my-layer` layer,\n * reset the entire state object for the\n * feature under the mouse\n * ```ts\n * map.on('mouseleave', 'my-layer', function(e) {\n * map.removeFeatureState({\n * source: 'my-source',\n * sourceLayer: 'my-source-layer',\n * id: e.features[0].id\n * });\n * });\n * ```\n *\n * @example\n * When the mouse leaves the `my-layer` layer,\n * reset only the `hover` key-value pair in the\n * state for the feature under the mouse\n * ```ts\n * map.on('mouseleave', 'my-layer', function(e) {\n * map.removeFeatureState({\n * source: 'my-source',\n * sourceLayer: 'my-source-layer',\n * id: e.features[0].id\n * }, 'hover');\n * });\n * ```\n */\n removeFeatureState(target: FeatureIdentifier, key?: string): this {\n this.style.removeFeatureState(target, key);\n return this._update();\n }\n\n /**\n * Gets the `state` of a feature.\n * A feature's `state` is a set of user-defined key-value pairs that are assigned to a feature at runtime.\n * Features are identified by their `feature.id` attribute, which can be any number or string.\n *\n * _Note: To access the values in a feature's state object for the purposes of styling the feature, use the [`feature-state` expression](https://maplibre.org/maplibre-style-spec/expressions/#feature-state)._\n *\n * @param feature - Feature identifier. Feature objects returned from\n * {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers.\n * @returns The state of the feature: a set of key-value pairs that was assigned to the feature at runtime.\n *\n * @example\n * When the mouse moves over the `my-layer` layer,\n * get the feature state for the feature under the mouse\n * ```ts\n * map.on('mousemove', 'my-layer', function(e) {\n * if (e.features.length > 0) {\n * map.getFeatureState({\n * source: 'my-source',\n * sourceLayer: 'my-source-layer',\n * id: e.features[0].id\n * });\n * }\n * });\n * ```\n */\n getFeatureState(feature: FeatureIdentifier): any {\n return this.style.getFeatureState(feature);\n }\n\n /**\n * Returns the map's containing HTML element.\n *\n * @returns The map's container.\n */\n getContainer(): HTMLElement {\n return this._container;\n }\n\n /**\n * Returns the HTML element containing the map's `` element.\n *\n * If you want to add non-GL overlays to the map, you should append them to this element.\n *\n * This is the element to which event bindings for map interactivity (such as panning and zooming) are\n * attached. It will receive bubbled events from child elements such as the ``, but not from\n * map controls.\n *\n * @returns The container of the map's ``.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n getCanvasContainer(): HTMLElement {\n return this._canvasContainer;\n }\n\n /**\n * Returns the map's `` element.\n *\n * @returns The map's `` element.\n * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n */\n getCanvas(): HTMLCanvasElement {\n return this._canvas;\n }\n\n _containerDimensions() {\n let width = 0;\n let height = 0;\n\n if (this._container) {\n width = this._container.clientWidth || 400;\n height = this._container.clientHeight || 300;\n }\n\n return [width, height];\n }\n\n _setupContainer() {\n const container = this._container;\n container.classList.add('maplibregl-map');\n\n const canvasContainer = this._canvasContainer = DOM.create('div', 'maplibregl-canvas-container', container);\n if (this._interactive) {\n canvasContainer.classList.add('maplibregl-interactive');\n }\n\n this._canvas = DOM.create('canvas', 'maplibregl-canvas', canvasContainer);\n this._canvas.addEventListener('webglcontextlost', this._contextLost, false);\n this._canvas.addEventListener('webglcontextrestored', this._contextRestored, false);\n this._canvas.setAttribute('tabindex', '0');\n this._canvas.setAttribute('aria-label', 'Map');\n this._canvas.setAttribute('role', 'region');\n\n const dimensions = this._containerDimensions();\n const clampedPixelRatio = this._getClampedPixelRatio(dimensions[0], dimensions[1]);\n this._resizeCanvas(dimensions[0], dimensions[1], clampedPixelRatio);\n\n const controlContainer = this._controlContainer = DOM.create('div', 'maplibregl-control-container', container);\n const positions = this._controlPositions = {};\n ['top-left', 'top-right', 'bottom-left', 'bottom-right'].forEach((positionName) => {\n positions[positionName] = DOM.create('div', `maplibregl-ctrl-${positionName} `, controlContainer);\n });\n\n this._container.addEventListener('scroll', this._onMapScroll, false);\n }\n\n _cooperativeGesturesOnWheel = (event: WheelEvent) => {\n this._onCooperativeGesture(event, event[this._metaKey], 1);\n };\n\n _setupCooperativeGestures() {\n const container = this._container;\n this._cooperativeGesturesScreen = DOM.create('div', 'maplibregl-cooperative-gesture-screen', container);\n let desktopMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.windowsHelpText ? this._cooperativeGestures.windowsHelpText : 'Use Ctrl + scroll to zoom the map';\n if (navigator.platform.indexOf('Mac') === 0) {\n desktopMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.macHelpText ? this._cooperativeGestures.macHelpText : 'Use ⌘ + scroll to zoom the map';\n }\n const mobileMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.mobileHelpText ? this._cooperativeGestures.mobileHelpText : 'Use two fingers to move the map';\n this._cooperativeGesturesScreen.innerHTML = `\n
${desktopMessage}
\n
${mobileMessage}
\n `;\n\n // Remove cooperative gesture screen from the accessibility tree since screenreaders cannot interact with the map using gestures\n this._cooperativeGesturesScreen.setAttribute('aria-hidden', 'true');\n\n // Add event to canvas container since gesture container is pointer-events: none\n this._canvasContainer.addEventListener('wheel', this._cooperativeGesturesOnWheel, false);\n\n // Add a cooperative gestures class (enable touch-action: pan-x pan-y;)\n this._canvasContainer.classList.add('maplibregl-cooperative-gestures');\n }\n\n _destroyCooperativeGestures() {\n DOM.remove(this._cooperativeGesturesScreen);\n this._canvasContainer.removeEventListener('wheel', this._cooperativeGesturesOnWheel, false);\n this._canvasContainer.classList.remove('maplibregl-cooperative-gestures');\n }\n\n _resizeCanvas(width: number, height: number, pixelRatio: number) {\n // Request the required canvas size taking the pixelratio into account.\n this._canvas.width = Math.floor(pixelRatio * width);\n this._canvas.height = Math.floor(pixelRatio * height);\n\n // Maintain the same canvas size, potentially downscaling it for HiDPI displays\n this._canvas.style.width = `${width}px`;\n this._canvas.style.height = `${height}px`;\n }\n\n _setupPainter() {\n\n const attributes = {\n alpha: true,\n stencil: true,\n depth: true,\n failIfMajorPerformanceCaveat: this._failIfMajorPerformanceCaveat,\n preserveDrawingBuffer: this._preserveDrawingBuffer,\n antialias: this._antialias || false\n };\n\n let webglcontextcreationerrorDetailObject: any = null;\n this._canvas.addEventListener('webglcontextcreationerror', (args: WebGLContextEvent) => {\n webglcontextcreationerrorDetailObject = {requestedAttributes: attributes};\n if (args) {\n webglcontextcreationerrorDetailObject.statusMessage = args.statusMessage;\n webglcontextcreationerrorDetailObject.type = args.type;\n }\n }, {once: true});\n\n const gl =\n this._canvas.getContext('webgl2', attributes) as WebGL2RenderingContext ||\n this._canvas.getContext('webgl', attributes) as WebGLRenderingContext;\n\n if (!gl) {\n const msg = 'Failed to initialize WebGL';\n if (webglcontextcreationerrorDetailObject) {\n webglcontextcreationerrorDetailObject.message = msg;\n throw new Error(JSON.stringify(webglcontextcreationerrorDetailObject));\n } else {\n throw new Error(msg);\n }\n }\n\n this.painter = new Painter(gl, this.transform);\n\n webpSupported.testSupport(gl);\n }\n\n _contextLost = (event: any) => {\n event.preventDefault();\n if (this._frame) {\n this._frame.cancel();\n this._frame = null;\n }\n this.fire(new Event('webglcontextlost', {originalEvent: event}));\n };\n\n _contextRestored = (event: any) => {\n this._setupPainter();\n this.resize();\n this._update();\n this.fire(new Event('webglcontextrestored', {originalEvent: event}));\n };\n\n _onMapScroll = (event: any) => {\n if (event.target !== this._container) return;\n\n // Revert any scroll which would move the canvas outside of the view\n this._container.scrollTop = 0;\n this._container.scrollLeft = 0;\n return false;\n };\n\n _onCooperativeGesture(event: any, metaPress, touches) {\n if (!metaPress && touches < 2) {\n // Alert user how to scroll/pan\n this._cooperativeGesturesScreen.classList.add('maplibregl-show');\n setTimeout(() => {\n this._cooperativeGesturesScreen.classList.remove('maplibregl-show');\n }, 100);\n }\n return false;\n }\n\n /**\n * Returns a Boolean indicating whether the map is fully loaded.\n *\n * Returns `false` if the style is not yet fully loaded,\n * or if there has been a change to the sources or style that\n * has not yet fully loaded.\n *\n * @returns A Boolean indicating whether the map is fully loaded.\n */\n loaded(): boolean {\n return !this._styleDirty && !this._sourcesDirty && !!this.style && this.style.loaded();\n }\n\n /**\n * @internal\n * Update this map's style and sources, and re-render the map.\n *\n * @param updateStyle - mark the map's style for reprocessing as\n * well as its sources\n * @returns `this`\n */\n _update(updateStyle?: boolean) {\n if (!this.style || !this.style._loaded) return this;\n\n this._styleDirty = this._styleDirty || updateStyle;\n this._sourcesDirty = true;\n this.triggerRepaint();\n\n return this;\n }\n\n /**\n * @internal\n * Request that the given callback be executed during the next render\n * frame. Schedule a render frame if one is not already scheduled.\n *\n * @returns An id that can be used to cancel the callback\n */\n _requestRenderFrame(callback: () => void): TaskID {\n this._update();\n return this._renderTaskQueue.add(callback);\n }\n\n _cancelRenderFrame(id: TaskID) {\n this._renderTaskQueue.remove(id);\n }\n\n /**\n * @internal\n * Call when a (re-)render of the map is required:\n * - The style has changed (`setPaintProperty()`, etc.)\n * - Source data has changed (e.g. tiles have finished loading)\n * - The map has is moving (or just finished moving)\n * - A transition is in progress\n *\n * @param paintStartTimeStamp - The time when the animation frame began executing.\n *\n * @returns `this`\n */\n _render(paintStartTimeStamp: number) {\n const fadeDuration = this._idleTriggered ? this._fadeDuration : 0;\n\n // A custom layer may have used the context asynchronously. Mark the state as dirty.\n this.painter.context.setDirty();\n this.painter.setBaseState();\n\n this._renderTaskQueue.run(paintStartTimeStamp);\n // A task queue callback may have fired a user event which may have removed the map\n if (this._removed) return;\n\n let crossFading = false;\n\n // If the style has changed, the map is being zoomed, or a transition or fade is in progress:\n // - Apply style changes (in a batch)\n // - Recalculate paint properties.\n if (this.style && this._styleDirty) {\n this._styleDirty = false;\n\n const zoom = this.transform.zoom;\n const now = browser.now();\n this.style.zoomHistory.update(zoom, now);\n\n const parameters = new EvaluationParameters(zoom, {\n now,\n fadeDuration,\n zoomHistory: this.style.zoomHistory,\n transition: this.style.getTransition()\n });\n\n const factor = parameters.crossFadingFactor();\n if (factor !== 1 || factor !== this._crossFadingFactor) {\n crossFading = true;\n this._crossFadingFactor = factor;\n }\n\n this.style.update(parameters);\n }\n\n // If we are in _render for any reason other than an in-progress paint\n // transition, update source caches to check for and load any tiles we\n // need for the current transform\n if (this.style && this._sourcesDirty) {\n this._sourcesDirty = false;\n this.style._updateSources(this.transform);\n }\n\n // update terrain stuff\n if (this.terrain) {\n this.terrain.sourceCache.update(this.transform, this.terrain);\n this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n if (!this._elevationFreeze) {\n this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n }\n } else {\n this.transform._minEleveationForCurrentTile = 0;\n this.transform.elevation = 0;\n }\n\n this._placementDirty = this.style && this.style._updatePlacement(this.painter.transform, this.showCollisionBoxes, fadeDuration, this._crossSourceCollisions);\n\n // Actually draw\n this.painter.render(this.style, {\n showTileBoundaries: this.showTileBoundaries,\n showOverdrawInspector: this._showOverdrawInspector,\n rotating: this.isRotating(),\n zooming: this.isZooming(),\n moving: this.isMoving(),\n fadeDuration,\n showPadding: this.showPadding,\n });\n\n this.fire(new Event('render'));\n\n if (this.loaded() && !this._loaded) {\n this._loaded = true;\n PerformanceUtils.mark(PerformanceMarkers.load);\n this.fire(new Event('load'));\n }\n\n if (this.style && (this.style.hasTransitions() || crossFading)) {\n this._styleDirty = true;\n }\n\n if (this.style && !this._placementDirty) {\n // Since no fade operations are in progress, we can release\n // all tiles held for fading. If we didn't do this, the tiles\n // would just sit in the SourceCaches until the next render\n this.style._releaseSymbolFadeTiles();\n }\n\n // Schedule another render frame if it's needed.\n //\n // Even though `_styleDirty` and `_sourcesDirty` are reset in this\n // method, synchronous events fired during Style#update or\n // Style#_updateSources could have caused them to be set again.\n const somethingDirty = this._sourcesDirty || this._styleDirty || this._placementDirty;\n if (somethingDirty || this._repaint) {\n this.triggerRepaint();\n } else if (!this.isMoving() && this.loaded()) {\n this.fire(new Event('idle'));\n }\n\n if (this._loaded && !this._fullyLoaded && !somethingDirty) {\n this._fullyLoaded = true;\n PerformanceUtils.mark(PerformanceMarkers.fullLoad);\n }\n\n return this;\n }\n\n /**\n * Force a synchronous redraw of the map.\n * @returns `this`\n * @example\n * ```ts\n * map.redraw();\n * ```\n */\n redraw(): this {\n if (this.style) {\n // cancel the scheduled update\n if (this._frame) {\n this._frame.cancel();\n this._frame = null;\n }\n this._render(0);\n }\n return this;\n }\n\n /**\n * Clean up and release all internal resources associated with this map.\n *\n * This includes DOM elements, event bindings, web workers, and WebGL resources.\n *\n * Use this method when you are done using the map and wish to ensure that it no\n * longer consumes browser resources. Afterwards, you must not call any other\n * methods on the map.\n */\n remove() {\n if (this._hash) this._hash.remove();\n\n for (const control of this._controls) control.onRemove(this);\n this._controls = [];\n\n if (this._frame) {\n this._frame.cancel();\n this._frame = null;\n }\n this._renderTaskQueue.clear();\n this.painter.destroy();\n this.handlers.destroy();\n delete this.handlers;\n this.setStyle(null);\n if (typeof window !== 'undefined') {\n removeEventListener('online', this._onWindowOnline, false);\n }\n\n ImageRequest.removeThrottleControl(this._imageQueueHandle);\n\n this._resizeObserver?.disconnect();\n const extension = this.painter.context.gl.getExtension('WEBGL_lose_context');\n if (extension) extension.loseContext();\n this._canvas.removeEventListener('webglcontextrestored', this._contextRestored, false);\n this._canvas.removeEventListener('webglcontextlost', this._contextLost, false);\n DOM.remove(this._canvasContainer);\n DOM.remove(this._controlContainer);\n if (this._cooperativeGestures) {\n this._destroyCooperativeGestures();\n }\n this._container.classList.remove('maplibregl-map');\n\n PerformanceUtils.clearMetrics();\n\n this._removed = true;\n this.fire(new Event('remove'));\n }\n\n /**\n * Trigger the rendering of a single frame. Use this method with custom layers to\n * repaint the map when the layer changes. Calling this multiple times before the\n * next frame is rendered will still result in only a single frame being rendered.\n * @example\n * ```ts\n * map.triggerRepaint();\n * ```\n * @see [Add a 3D model](https://maplibre.org/maplibre-gl-js/docs/examples/add-3d-model/)\n * @see [Add an animated icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-animated/)\n */\n triggerRepaint() {\n if (this.style && !this._frame) {\n this._frame = browser.frame((paintStartTimeStamp: number) => {\n PerformanceUtils.frame(paintStartTimeStamp);\n this._frame = null;\n this._render(paintStartTimeStamp);\n });\n }\n }\n\n _onWindowOnline = () => {\n this._update();\n };\n\n /**\n * Gets and sets a Boolean indicating whether the map will render an outline\n * around each tile and the tile ID. These tile boundaries are useful for\n * debugging.\n *\n * The uncompressed file size of the first vector source is drawn in the top left\n * corner of each tile, next to the tile ID.\n *\n * @example\n * ```ts\n * map.showTileBoundaries = true;\n * ```\n */\n get showTileBoundaries(): boolean { return !!this._showTileBoundaries; }\n set showTileBoundaries(value: boolean) {\n if (this._showTileBoundaries === value) return;\n this._showTileBoundaries = value;\n this._update();\n }\n\n /**\n * Gets and sets a Boolean indicating whether the map will visualize\n * the padding offsets.\n */\n get showPadding(): boolean { return !!this._showPadding; }\n set showPadding(value: boolean) {\n if (this._showPadding === value) return;\n this._showPadding = value;\n this._update();\n }\n\n /**\n * Gets and sets a Boolean indicating whether the map will render boxes\n * around all symbols in the data source, revealing which symbols\n * were rendered or which were hidden due to collisions.\n * This information is useful for debugging.\n */\n get showCollisionBoxes(): boolean { return !!this._showCollisionBoxes; }\n set showCollisionBoxes(value: boolean) {\n if (this._showCollisionBoxes === value) return;\n this._showCollisionBoxes = value;\n if (value) {\n // When we turn collision boxes on we have to generate them for existing tiles\n // When we turn them off, there's no cost to leaving existing boxes in place\n this.style._generateCollisionBoxes();\n } else {\n // Otherwise, call an update to remove collision boxes\n this._update();\n }\n }\n\n /**\n * Gets and sets a Boolean indicating whether the map should color-code\n * each fragment to show how many times it has been shaded.\n * White fragments have been shaded 8 or more times.\n * Black fragments have been shaded 0 times.\n * This information is useful for debugging.\n */\n get showOverdrawInspector(): boolean { return !!this._showOverdrawInspector; }\n set showOverdrawInspector(value: boolean) {\n if (this._showOverdrawInspector === value) return;\n this._showOverdrawInspector = value;\n this._update();\n }\n\n /**\n * Gets and sets a Boolean indicating whether the map will\n * continuously repaint. This information is useful for analyzing performance.\n */\n get repaint(): boolean { return !!this._repaint; }\n set repaint(value: boolean) {\n if (this._repaint !== value) {\n this._repaint = value;\n this.triggerRepaint();\n }\n }\n // show vertices\n get vertices(): boolean { return !!this._vertices; }\n set vertices(value: boolean) { this._vertices = value; this._update(); }\n\n /**\n * Returns the package version of the library\n * @returns Package version of the library\n */\n get version(): string {\n return version;\n }\n\n /**\n * Returns the elevation for the point where the camera is looking.\n * This value corresponds to:\n * \"meters above sea level\" * \"exaggeration\"\n * @returns The elevation.\n */\n getCameraTargetElevation(): number {\n return this.transform.elevation;\n }\n}\n","import type Point from '@mapbox/point-geometry';\n\nimport {DragMoveHandler, DragRotateResult, DragPitchResult, DragHandler} from './drag_handler';\nimport {OneFingerTouchMoveStateManager} from './drag_move_state_manager';\n\nexport interface OneFingerTouchRotateHandler extends DragMoveHandler {}\nexport interface OneFingerTouchPitchHandler extends DragMoveHandler {}\n\nconst assignEvents = (handler: DragHandler) => {\n handler.touchstart = handler.dragStart;\n handler.touchmoveWindow = handler.dragMove;\n handler.touchend = handler.dragEnd;\n};\n\nexport const generateOneFingerTouchRotationHandler = ({enable, clickTolerance, bearingDegreesPerPixelMoved = 0.8}: {\n clickTolerance: number;\n bearingDegreesPerPixelMoved?: number;\n enable?: boolean;\n}): OneFingerTouchRotateHandler => {\n const touchMoveStateManager = new OneFingerTouchMoveStateManager();\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({bearingDelta: (point.x - lastPoint.x) * bearingDegreesPerPixelMoved}),\n moveStateManager: touchMoveStateManager,\n enable,\n assignEvents,\n });\n};\n\nexport const generateOneFingerTouchPitchHandler = ({enable, clickTolerance, pitchDegreesPerPixelMoved = -0.5}: {\n clickTolerance: number;\n pitchDegreesPerPixelMoved?: number;\n enable?: boolean;\n}): OneFingerTouchPitchHandler => {\n const touchMoveStateManager = new OneFingerTouchMoveStateManager();\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({pitchDelta: (point.y - lastPoint.y) * pitchDegreesPerPixelMoved}),\n moveStateManager: touchMoveStateManager,\n enable,\n assignEvents,\n });\n};\n","import Point from '@mapbox/point-geometry';\n\nimport {DOM} from '../../util/dom';\nimport {extend} from '../../util/util';\nimport {generateMousePitchHandler, generateMouseRotationHandler, MousePitchHandler, MouseRotateHandler} from '../handler/mouse';\nimport {generateOneFingerTouchPitchHandler, generateOneFingerTouchRotationHandler, OneFingerTouchPitchHandler, OneFingerTouchRotateHandler} from '../handler/one_finger_touch_drag';\n\nimport type {Map} from '../map';\nimport type {IControl} from './control';\n\n/**\n * The {@link NavigationControl} options object\n */\ntype NavigationOptions = {\n /**\n * If `true` the compass button is included.\n */\n showCompass?: boolean;\n /**\n * If `true` the zoom-in and zoom-out buttons are included.\n */\n showZoom?: boolean;\n /**\n * If `true` the pitch is visualized by rotating X-axis of compass.\n */\n visualizePitch?: boolean;\n};\n\nconst defaultOptions: NavigationOptions = {\n showCompass: true,\n showZoom: true,\n visualizePitch: false\n};\n\n/**\n * A `NavigationControl` control contains zoom buttons and a compass.\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * let nav = new maplibregl.NavigationControl();\n * map.addControl(nav, 'top-left');\n * ```\n * @see [Display map navigation controls](https://maplibre.org/maplibre-gl-js/docs/examples/navigation/)\n */\nexport class NavigationControl implements IControl {\n _map: Map;\n options: NavigationOptions;\n _container: HTMLElement;\n _zoomInButton: HTMLButtonElement;\n _zoomOutButton: HTMLButtonElement;\n _compass: HTMLButtonElement;\n _compassIcon: HTMLElement;\n _handler: MouseRotateWrapper;\n\n /**\n * @param options - the control's options\n */\n constructor(options?: NavigationOptions) {\n this.options = extend({}, defaultOptions, options);\n\n this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group');\n this._container.addEventListener('contextmenu', (e) => e.preventDefault());\n\n if (this.options.showZoom) {\n this._zoomInButton = this._createButton('maplibregl-ctrl-zoom-in', (e) => this._map.zoomIn({}, {originalEvent: e}));\n DOM.create('span', 'maplibregl-ctrl-icon', this._zoomInButton).setAttribute('aria-hidden', 'true');\n this._zoomOutButton = this._createButton('maplibregl-ctrl-zoom-out', (e) => this._map.zoomOut({}, {originalEvent: e}));\n DOM.create('span', 'maplibregl-ctrl-icon', this._zoomOutButton).setAttribute('aria-hidden', 'true');\n }\n if (this.options.showCompass) {\n this._compass = this._createButton('maplibregl-ctrl-compass', (e) => {\n if (this.options.visualizePitch) {\n this._map.resetNorthPitch({}, {originalEvent: e});\n } else {\n this._map.resetNorth({}, {originalEvent: e});\n }\n });\n this._compassIcon = DOM.create('span', 'maplibregl-ctrl-icon', this._compass);\n this._compassIcon.setAttribute('aria-hidden', 'true');\n }\n }\n\n _updateZoomButtons = () => {\n const zoom = this._map.getZoom();\n const isMax = zoom === this._map.getMaxZoom();\n const isMin = zoom === this._map.getMinZoom();\n this._zoomInButton.disabled = isMax;\n this._zoomOutButton.disabled = isMin;\n this._zoomInButton.setAttribute('aria-disabled', isMax.toString());\n this._zoomOutButton.setAttribute('aria-disabled', isMin.toString());\n };\n\n _rotateCompassArrow = () => {\n const rotate = this.options.visualizePitch ?\n `scale(${1 / Math.pow(Math.cos(this._map.transform.pitch * (Math.PI / 180)), 0.5)}) rotateX(${this._map.transform.pitch}deg) rotateZ(${this._map.transform.angle * (180 / Math.PI)}deg)` :\n `rotate(${this._map.transform.angle * (180 / Math.PI)}deg)`;\n\n this._compassIcon.style.transform = rotate;\n };\n\n onAdd(map: Map) {\n this._map = map;\n if (this.options.showZoom) {\n this._setButtonTitle(this._zoomInButton, 'ZoomIn');\n this._setButtonTitle(this._zoomOutButton, 'ZoomOut');\n this._map.on('zoom', this._updateZoomButtons);\n this._updateZoomButtons();\n }\n if (this.options.showCompass) {\n this._setButtonTitle(this._compass, 'ResetBearing');\n if (this.options.visualizePitch) {\n this._map.on('pitch', this._rotateCompassArrow);\n }\n this._map.on('rotate', this._rotateCompassArrow);\n this._rotateCompassArrow();\n this._handler = new MouseRotateWrapper(this._map, this._compass, this.options.visualizePitch);\n }\n return this._container;\n }\n\n onRemove() {\n DOM.remove(this._container);\n if (this.options.showZoom) {\n this._map.off('zoom', this._updateZoomButtons);\n }\n if (this.options.showCompass) {\n if (this.options.visualizePitch) {\n this._map.off('pitch', this._rotateCompassArrow);\n }\n this._map.off('rotate', this._rotateCompassArrow);\n this._handler.off();\n delete this._handler;\n }\n\n delete this._map;\n }\n\n _createButton(className: string, fn: (e?: any) => unknown) {\n const a = DOM.create('button', className, this._container) as HTMLButtonElement;\n a.type = 'button';\n a.addEventListener('click', fn);\n return a;\n }\n\n _setButtonTitle = (button: HTMLButtonElement, title: string) => {\n const str = this._map._getUIString(`NavigationControl.${title}`);\n button.title = str;\n button.setAttribute('aria-label', str);\n };\n}\n\nclass MouseRotateWrapper {\n\n map: Map;\n _clickTolerance: number;\n element: HTMLElement;\n // Rotation and pitch handlers are separated due to different _clickTolerance values\n mouseRotate: MouseRotateHandler;\n touchRotate: OneFingerTouchRotateHandler;\n mousePitch: MousePitchHandler;\n touchPitch: OneFingerTouchPitchHandler;\n _startPos: Point;\n _lastPos: Point;\n\n constructor(map: Map, element: HTMLElement, pitch: boolean = false) {\n this._clickTolerance = 10;\n const mapRotateTolerance = map.dragRotate._mouseRotate.getClickTolerance();\n const mapPitchTolerance = map.dragRotate._mousePitch.getClickTolerance();\n this.element = element;\n this.mouseRotate = generateMouseRotationHandler({clickTolerance: mapRotateTolerance, enable: true});\n this.touchRotate = generateOneFingerTouchRotationHandler({clickTolerance: mapRotateTolerance, enable: true});\n this.map = map;\n if (pitch) {\n this.mousePitch = generateMousePitchHandler({clickTolerance: mapPitchTolerance, enable: true});\n this.touchPitch = generateOneFingerTouchPitchHandler({clickTolerance: mapPitchTolerance, enable: true});\n }\n\n DOM.addEventListener(element, 'mousedown', this.mousedown);\n DOM.addEventListener(element, 'touchstart', this.touchstart, {passive: false});\n DOM.addEventListener(element, 'touchcancel', this.reset);\n }\n\n startMouse(e: MouseEvent, point: Point) {\n this.mouseRotate.dragStart(e, point);\n if (this.mousePitch) this.mousePitch.dragStart(e, point);\n DOM.disableDrag();\n }\n\n startTouch(e: TouchEvent, point: Point) {\n this.touchRotate.dragStart(e, point);\n if (this.touchPitch) this.touchPitch.dragStart(e, point);\n DOM.disableDrag();\n }\n\n moveMouse(e: MouseEvent, point: Point) {\n const map = this.map;\n const {bearingDelta} = this.mouseRotate.dragMove(e, point) || {};\n if (bearingDelta) map.setBearing(map.getBearing() + bearingDelta);\n if (this.mousePitch) {\n const {pitchDelta} = this.mousePitch.dragMove(e, point) || {};\n if (pitchDelta) map.setPitch(map.getPitch() + pitchDelta);\n }\n }\n\n moveTouch(e: TouchEvent, point: Point) {\n const map = this.map;\n const {bearingDelta} = this.touchRotate.dragMove(e, point) || {};\n if (bearingDelta) map.setBearing(map.getBearing() + bearingDelta);\n if (this.touchPitch) {\n const {pitchDelta} = this.touchPitch.dragMove(e, point) || {};\n if (pitchDelta) map.setPitch(map.getPitch() + pitchDelta);\n }\n }\n\n off() {\n const element = this.element;\n DOM.removeEventListener(element, 'mousedown', this.mousedown);\n DOM.removeEventListener(element, 'touchstart', this.touchstart, {passive: false});\n DOM.removeEventListener(window, 'touchmove', this.touchmove, {passive: false});\n DOM.removeEventListener(window, 'touchend', this.touchend);\n DOM.removeEventListener(element, 'touchcancel', this.reset);\n this.offTemp();\n }\n\n offTemp() {\n DOM.enableDrag();\n DOM.removeEventListener(window, 'mousemove', this.mousemove);\n DOM.removeEventListener(window, 'mouseup', this.mouseup);\n DOM.removeEventListener(window, 'touchmove', this.touchmove, {passive: false});\n DOM.removeEventListener(window, 'touchend', this.touchend);\n }\n\n mousedown = (e: MouseEvent) => {\n this.startMouse(extend({}, e, {ctrlKey: true, preventDefault: () => e.preventDefault()}), DOM.mousePos(this.element, e));\n DOM.addEventListener(window, 'mousemove', this.mousemove);\n DOM.addEventListener(window, 'mouseup', this.mouseup);\n };\n\n mousemove = (e: MouseEvent) => {\n this.moveMouse(e, DOM.mousePos(this.element, e));\n };\n\n mouseup = (e: MouseEvent) => {\n this.mouseRotate.dragEnd(e);\n if (this.mousePitch) this.mousePitch.dragEnd(e);\n this.offTemp();\n };\n\n touchstart = (e: TouchEvent) => {\n if (e.targetTouches.length !== 1) {\n this.reset();\n } else {\n this._startPos = this._lastPos = DOM.touchPos(this.element, e.targetTouches)[0];\n this.startTouch(e, this._startPos);\n DOM.addEventListener(window, 'touchmove', this.touchmove, {passive: false});\n DOM.addEventListener(window, 'touchend', this.touchend);\n }\n };\n\n touchmove = (e: TouchEvent) => {\n if (e.targetTouches.length !== 1) {\n this.reset();\n } else {\n this._lastPos = DOM.touchPos(this.element, e.targetTouches)[0];\n this.moveTouch(e, this._lastPos);\n }\n };\n\n touchend = (e: TouchEvent) => {\n if (e.targetTouches.length === 0 &&\n this._startPos &&\n this._lastPos &&\n this._startPos.dist(this._lastPos) < this._clickTolerance) {\n this.element.click();\n }\n delete this._startPos;\n delete this._lastPos;\n this.offTemp();\n };\n\n reset = () => {\n this.mouseRotate.reset();\n if (this.mousePitch) this.mousePitch.reset();\n this.touchRotate.reset();\n if (this.touchPitch) this.touchPitch.reset();\n delete this._startPos;\n delete this._lastPos;\n this.offTemp();\n };\n}\n","let supportsGeolocation;\n\nexport function checkGeolocationSupport(callback: (supported: boolean) => void, forceRecalculation = false): void {\n if (supportsGeolocation !== undefined && !forceRecalculation) {\n callback(supportsGeolocation);\n } else if (window.navigator.permissions !== undefined) {\n // navigator.permissions has incomplete browser support\n // http://caniuse.com/#feat=permissions-api\n // Test for the case where a browser disables Geolocation because of an\n // insecure origin\n window.navigator.permissions.query({name: 'geolocation'}).then((p) => {\n supportsGeolocation = p.state !== 'denied';\n callback(supportsGeolocation);\n }).catch(() => {\n // Fix for iOS16 which rejects query but still supports geolocation\n supportsGeolocation = !!window.navigator.geolocation;\n callback(supportsGeolocation);\n });\n\n } else {\n supportsGeolocation = !!window.navigator.geolocation;\n callback(supportsGeolocation);\n }\n}\n","import {LngLat} from '../geo/lng_lat';\n\nimport type Point from '@mapbox/point-geometry';\nimport type {Transform} from '../geo/transform';\n\n/**\n * Given a LngLat, prior projected position, and a transform, return a new LngLat shifted\n * n × 360° east or west for some n ≥ 0 such that:\n *\n * * the projected location of the result is on screen, if possible, and secondarily:\n * * the difference between the projected location of the result and the prior position\n * is minimized.\n *\n * The object is to preserve perceived object constancy for Popups and Markers as much as\n * possible; they should avoid shifting large distances across the screen, even when the\n * map center changes by ±360° due to automatic wrapping, and when about to go off screen,\n * should wrap just enough to avoid doing so.\n */\nexport function smartWrap(lngLat: LngLat, priorPos: Point, transform: Transform): LngLat {\n lngLat = new LngLat(lngLat.lng, lngLat.lat);\n\n // First, try shifting one world in either direction, and see if either is closer to the\n // prior position. This preserves object constancy when the map center is auto-wrapped\n // during animations.\n if (priorPos) {\n const left = new LngLat(lngLat.lng - 360, lngLat.lat);\n const right = new LngLat(lngLat.lng + 360, lngLat.lat);\n const delta = transform.locationPoint(lngLat).distSqr(priorPos);\n if (transform.locationPoint(left).distSqr(priorPos) < delta) {\n lngLat = left;\n } else if (transform.locationPoint(right).distSqr(priorPos) < delta) {\n lngLat = right;\n }\n }\n\n // Second, wrap toward the center until the new position is on screen, or we can't get\n // any closer.\n while (Math.abs(lngLat.lng - transform.center.lng) > 180) {\n const pos = transform.locationPoint(lngLat);\n if (pos.x >= 0 && pos.y >= 0 && pos.x <= transform.width && pos.y <= transform.height) {\n break;\n }\n if (lngLat.lng > transform.center.lng) {\n lngLat.lng -= 360;\n } else {\n lngLat.lng += 360;\n }\n }\n\n return lngLat;\n}\n","/**\n * Where to position the anchor.\n * Used by a popup and a marker.\n */\nexport type PositionAnchor = 'center' | 'top' | 'bottom' | 'left' | 'right' | 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right';\n\nexport const anchorTranslate: {\n [_ in PositionAnchor]: string;\n} = {\n 'center': 'translate(-50%,-50%)',\n 'top': 'translate(-50%,0)',\n 'top-left': 'translate(0,0)',\n 'top-right': 'translate(-100%,0)',\n 'bottom': 'translate(-50%,-100%)',\n 'bottom-left': 'translate(0,-100%)',\n 'bottom-right': 'translate(-100%,-100%)',\n 'left': 'translate(0,-50%)',\n 'right': 'translate(-100%,-50%)'\n};\n\nexport function applyAnchorClass(element: HTMLElement, anchor: PositionAnchor, prefix: string) {\n const classList = element.classList;\n for (const key in anchorTranslate) {\n classList.remove(`maplibregl-${prefix}-anchor-${key}`);\n }\n classList.add(`maplibregl-${prefix}-anchor-${anchor}`);\n}\n","import {DOM} from '../util/dom';\nimport {LngLat} from '../geo/lng_lat';\nimport Point from '@mapbox/point-geometry';\nimport {smartWrap} from '../util/smart_wrap';\nimport {anchorTranslate, applyAnchorClass} from './anchor';\nimport type {PositionAnchor} from './anchor';\nimport {Event, Evented} from '../util/evented';\nimport type {Map} from './map';\nimport {Popup, Offset} from './popup';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport type {MapMouseEvent, MapTouchEvent} from './events';\nimport type {PointLike} from './camera';\n\n/**\n * Alignment options of rotation and pitch\n */\ntype Alignment = 'map' | 'viewport' | 'auto';\n\n/**\n * The {@link Marker} options object\n */\ntype MarkerOptions = {\n /**\n * DOM element to use as a marker. The default is a light blue, droplet-shaped SVG marker.\n */\n element?: HTMLElement;\n /**\n * Space-separated CSS class names to add to marker element.\n */\n className?: string;\n /**\n * The offset in pixels as a {@link PointLike} object to apply relative to the element's center. Negatives indicate left and up.\n */\n offset?: PointLike;\n /**\n * A string indicating the part of the Marker that should be positioned closest to the coordinate set via {@link Marker#setLngLat}.\n * Options are `'center'`, `'top'`, `'bottom'`, `'left'`, `'right'`, `'top-left'`, `'top-right'`, `'bottom-left'`, and `'bottom-right'`.\n * @defaultValue 'center'\n * */\n anchor?: PositionAnchor;\n /**\n * The color to use for the default marker if options.element is not provided. The default is light blue.\n * @defaultValue '#3FB1CE'\n */\n color?: string;\n /**\n * The scale to use for the default marker if options.element is not provided. The default scale corresponds to a height of `41px` and a width of `27px`.\n * @defaultValue 1\n */\n scale?: number;\n /**\n * A boolean indicating whether or not a marker is able to be dragged to a new position on the map.\n * @defaultValue false\n */\n draggable?: boolean;\n /**\n * The max number of pixels a user can shift the mouse pointer during a click on the marker for it to be considered a valid click (as opposed to a marker drag). The default is to inherit map's clickTolerance.\n * @defaultValue 0\n */\n clickTolerance?: number;\n /**\n * The rotation angle of the marker in degrees, relative to its respective `rotationAlignment` setting. A positive value will rotate the marker clockwise.\n * @defaultValue 0\n */\n rotation?: number;\n /**\n * `map` aligns the `Marker`'s rotation relative to the map, maintaining a bearing as the map rotates. `viewport` aligns the `Marker`'s rotation relative to the viewport, agnostic to map rotations. `auto` is equivalent to `viewport`.\n * @defaultValue 'auto'\n */\n rotationAlignment?: Alignment;\n /**\n * `map` aligns the `Marker` to the plane of the map. `viewport` aligns the `Marker` to the plane of the viewport. `auto` automatically matches the value of `rotationAlignment`.\n * @defaultValue 'auto'\n */\n pitchAlignment?: Alignment;\n};\n\n/**\n * Creates a marker component\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([30.5, 50.5])\n * .addTo(map);\n * ```\n *\n * @example\n * Set options\n * ```ts\n * let marker = new maplibregl.Marker({\n * color: \"#FFFFFF\",\n * draggable: true\n * }).setLngLat([30.5, 50.5])\n * .addTo(map);\n * ```\n * @see [Add custom icons with Markers](https://maplibre.org/maplibre-gl-js/docs/examples/custom-marker-icons/)\n * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/)\n *\n * ### Events\n *\n * @event `dragstart` Fired when dragging starts, `marker` object that is being dragged\n *\n * @event `drag` Fired while dragging. `marker` object that is being dragged\n *\n * @event `dragend` Fired when the marker is finished being dragged, `marker` object that was dragged\n */\nexport class Marker extends Evented {\n _map: Map;\n _anchor: PositionAnchor;\n _offset: Point;\n _element: HTMLElement;\n _popup: Popup;\n _lngLat: LngLat;\n _pos: Point;\n _color: string;\n _scale: number;\n _defaultMarker: boolean;\n _draggable: boolean;\n _clickTolerance: number;\n _isDragging: boolean;\n _state: 'inactive' | 'pending' | 'active'; // used for handling drag events\n _positionDelta: Point;\n _pointerdownPos: Point;\n _rotation: number;\n _pitchAlignment: Alignment;\n _rotationAlignment: Alignment;\n _originalTabIndex: string; // original tabindex of _element\n _opacityTimeout: ReturnType;\n\n /**\n * @param options - the options\n */\n constructor(options?: MarkerOptions) {\n super();\n\n this._anchor = options && options.anchor || 'center';\n this._color = options && options.color || '#3FB1CE';\n this._scale = options && options.scale || 1;\n this._draggable = options && options.draggable || false;\n this._clickTolerance = options && options.clickTolerance || 0;\n this._isDragging = false;\n this._state = 'inactive';\n this._rotation = options && options.rotation || 0;\n this._rotationAlignment = options && options.rotationAlignment || 'auto';\n this._pitchAlignment = options && options.pitchAlignment && options.pitchAlignment !== 'auto' ? options.pitchAlignment : this._rotationAlignment;\n\n if (!options || !options.element) {\n this._defaultMarker = true;\n this._element = DOM.create('div');\n this._element.setAttribute('aria-label', 'Map marker');\n\n // create default map marker SVG\n const svg = DOM.createNS('http://www.w3.org/2000/svg', 'svg');\n const defaultHeight = 41;\n const defaultWidth = 27;\n svg.setAttributeNS(null, 'display', 'block');\n svg.setAttributeNS(null, 'height', `${defaultHeight}px`);\n svg.setAttributeNS(null, 'width', `${defaultWidth}px`);\n svg.setAttributeNS(null, 'viewBox', `0 0 ${defaultWidth} ${defaultHeight}`);\n\n const markerLarge = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n markerLarge.setAttributeNS(null, 'stroke', 'none');\n markerLarge.setAttributeNS(null, 'stroke-width', '1');\n markerLarge.setAttributeNS(null, 'fill', 'none');\n markerLarge.setAttributeNS(null, 'fill-rule', 'evenodd');\n\n const page1 = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n page1.setAttributeNS(null, 'fill-rule', 'nonzero');\n\n const shadow = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n shadow.setAttributeNS(null, 'transform', 'translate(3.0, 29.0)');\n shadow.setAttributeNS(null, 'fill', '#000000');\n\n const ellipses = [\n {'rx': '10.5', 'ry': '5.25002273'},\n {'rx': '10.5', 'ry': '5.25002273'},\n {'rx': '9.5', 'ry': '4.77275007'},\n {'rx': '8.5', 'ry': '4.29549936'},\n {'rx': '7.5', 'ry': '3.81822308'},\n {'rx': '6.5', 'ry': '3.34094679'},\n {'rx': '5.5', 'ry': '2.86367051'},\n {'rx': '4.5', 'ry': '2.38636864'}\n ];\n\n for (const data of ellipses) {\n const ellipse = DOM.createNS('http://www.w3.org/2000/svg', 'ellipse');\n ellipse.setAttributeNS(null, 'opacity', '0.04');\n ellipse.setAttributeNS(null, 'cx', '10.5');\n ellipse.setAttributeNS(null, 'cy', '5.80029008');\n ellipse.setAttributeNS(null, 'rx', data['rx']);\n ellipse.setAttributeNS(null, 'ry', data['ry']);\n shadow.appendChild(ellipse);\n }\n\n const background = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n background.setAttributeNS(null, 'fill', this._color);\n\n const bgPath = DOM.createNS('http://www.w3.org/2000/svg', 'path');\n bgPath.setAttributeNS(null, 'd', 'M27,13.5 C27,19.074644 20.250001,27.000002 14.75,34.500002 C14.016665,35.500004 12.983335,35.500004 12.25,34.500002 C6.7499993,27.000002 0,19.222562 0,13.5 C0,6.0441559 6.0441559,0 13.5,0 C20.955844,0 27,6.0441559 27,13.5 Z');\n\n background.appendChild(bgPath);\n\n const border = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n border.setAttributeNS(null, 'opacity', '0.25');\n border.setAttributeNS(null, 'fill', '#000000');\n\n const borderPath = DOM.createNS('http://www.w3.org/2000/svg', 'path');\n borderPath.setAttributeNS(null, 'd', 'M13.5,0 C6.0441559,0 0,6.0441559 0,13.5 C0,19.222562 6.7499993,27 12.25,34.5 C13,35.522727 14.016664,35.500004 14.75,34.5 C20.250001,27 27,19.074644 27,13.5 C27,6.0441559 20.955844,0 13.5,0 Z M13.5,1 C20.415404,1 26,6.584596 26,13.5 C26,15.898657 24.495584,19.181431 22.220703,22.738281 C19.945823,26.295132 16.705119,30.142167 13.943359,33.908203 C13.743445,34.180814 13.612715,34.322738 13.5,34.441406 C13.387285,34.322738 13.256555,34.180814 13.056641,33.908203 C10.284481,30.127985 7.4148684,26.314159 5.015625,22.773438 C2.6163816,19.232715 1,15.953538 1,13.5 C1,6.584596 6.584596,1 13.5,1 Z');\n\n border.appendChild(borderPath);\n\n const maki = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n maki.setAttributeNS(null, 'transform', 'translate(6.0, 7.0)');\n maki.setAttributeNS(null, 'fill', '#FFFFFF');\n\n const circleContainer = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n circleContainer.setAttributeNS(null, 'transform', 'translate(8.0, 8.0)');\n\n const circle1 = DOM.createNS('http://www.w3.org/2000/svg', 'circle');\n circle1.setAttributeNS(null, 'fill', '#000000');\n circle1.setAttributeNS(null, 'opacity', '0.25');\n circle1.setAttributeNS(null, 'cx', '5.5');\n circle1.setAttributeNS(null, 'cy', '5.5');\n circle1.setAttributeNS(null, 'r', '5.4999962');\n\n const circle2 = DOM.createNS('http://www.w3.org/2000/svg', 'circle');\n circle2.setAttributeNS(null, 'fill', '#FFFFFF');\n circle2.setAttributeNS(null, 'cx', '5.5');\n circle2.setAttributeNS(null, 'cy', '5.5');\n circle2.setAttributeNS(null, 'r', '5.4999962');\n\n circleContainer.appendChild(circle1);\n circleContainer.appendChild(circle2);\n\n page1.appendChild(shadow);\n page1.appendChild(background);\n page1.appendChild(border);\n page1.appendChild(maki);\n page1.appendChild(circleContainer);\n\n svg.appendChild(page1);\n\n svg.setAttributeNS(null, 'height', `${defaultHeight * this._scale}px`);\n svg.setAttributeNS(null, 'width', `${defaultWidth * this._scale}px`);\n\n this._element.appendChild(svg);\n\n // if no element and no offset option given apply an offset for the default marker\n // the -14 as the y value of the default marker offset was determined as follows\n //\n // the marker tip is at the center of the shadow ellipse from the default svg\n // the y value of the center of the shadow ellipse relative to the svg top left is \"shadow transform translate-y (29.0) + ellipse cy (5.80029008)\"\n // offset to the svg center \"height (41 / 2)\" gives (29.0 + 5.80029008) - (41 / 2) and rounded for an integer pixel offset gives 14\n // negative is used to move the marker up from the center so the tip is at the Marker lngLat\n this._offset = Point.convert(options && options.offset || [0, -14]);\n } else {\n this._element = options.element;\n this._offset = Point.convert(options && options.offset || [0, 0]);\n }\n\n this._element.classList.add('maplibregl-marker');\n this._element.addEventListener('dragstart', (e: DragEvent) => {\n e.preventDefault();\n });\n this._element.addEventListener('mousedown', (e: MouseEvent) => {\n // prevent focusing on click\n e.preventDefault();\n });\n applyAnchorClass(this._element, this._anchor, 'marker');\n\n if (options && options.className) {\n for (const name of options.className.split(' ')) {\n this._element.classList.add(name);\n }\n }\n\n this._popup = null;\n }\n\n /**\n * Attaches the `Marker` to a `Map` object.\n * @param map - The MapLibre GL JS map to add the marker to.\n * @returns `this`\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([30.5, 50.5])\n * .addTo(map); // add the marker to the map\n * ```\n */\n addTo(map: Map): this {\n this.remove();\n this._map = map;\n map.getCanvasContainer().appendChild(this._element);\n map.on('move', this._update);\n map.on('moveend', this._update);\n map.on('terrain', this._update);\n\n this.setDraggable(this._draggable);\n this._update();\n\n // If we attached the `click` listener to the marker element, the popup\n // would close once the event propagated to `map` due to the\n // `Popup#_onClickClose` listener.\n this._map.on('click', this._onMapClick);\n\n return this;\n }\n\n /**\n * Removes the marker from a map\n * @example\n * ```ts\n * let marker = new maplibregl.Marker().addTo(map);\n * marker.remove();\n * ```\n * @returns `this`\n */\n remove(): this {\n if (this._opacityTimeout) {\n clearTimeout(this._opacityTimeout);\n delete this._opacityTimeout;\n }\n if (this._map) {\n this._map.off('click', this._onMapClick);\n this._map.off('move', this._update);\n this._map.off('moveend', this._update);\n this._map.off('mousedown', this._addDragHandler);\n this._map.off('touchstart', this._addDragHandler);\n this._map.off('mouseup', this._onUp);\n this._map.off('touchend', this._onUp);\n this._map.off('mousemove', this._onMove);\n this._map.off('touchmove', this._onMove);\n delete this._map;\n }\n DOM.remove(this._element);\n if (this._popup) this._popup.remove();\n return this;\n }\n\n /**\n * Get the marker's geographical location.\n *\n * The longitude of the result may differ by a multiple of 360 degrees from the longitude previously\n * set by `setLngLat` because `Marker` wraps the anchor longitude across copies of the world to keep\n * the marker on screen.\n *\n * @returns A {@link LngLat} describing the marker's location.\n * @example\n * ```ts\n * // Store the marker's longitude and latitude coordinates in a variable\n * let lngLat = marker.getLngLat();\n * // Print the marker's longitude and latitude values in the console\n * console.log('Longitude: ' + lngLat.lng + ', Latitude: ' + lngLat.lat )\n * ```\n * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/)\n */\n getLngLat(): LngLat {\n return this._lngLat;\n }\n\n /**\n * Set the marker's geographical position and move it.\n * @param lnglat - A {@link LngLat} describing where the marker should be located.\n * @returns `this`\n * @example\n * Create a new marker, set the longitude and latitude, and add it to the map\n * ```ts\n * new maplibregl.Marker()\n * .setLngLat([-65.017, -16.457])\n * .addTo(map);\n * ```\n * @see [Add custom icons with Markers](https://maplibre.org/maplibre-gl-js/docs/examples/custom-marker-icons/)\n * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/)\n */\n setLngLat(lnglat: LngLatLike): this {\n this._lngLat = LngLat.convert(lnglat);\n this._pos = null;\n if (this._popup) this._popup.setLngLat(this._lngLat);\n this._update();\n return this;\n }\n\n /**\n * Returns the `Marker`'s HTML element.\n * @returns element\n */\n getElement(): HTMLElement {\n return this._element;\n }\n\n /**\n * Binds a {@link Popup} to the {@link Marker}.\n * @param popup - An instance of the {@link Popup} class. If undefined or null, any popup\n * set on this {@link Marker} instance is unset.\n * @returns `this`\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([0, 0])\n * .setPopup(new maplibregl.Popup().setHTML(\"

Hello World!

\")) // add popup\n * .addTo(map);\n * ```\n * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/)\n */\n setPopup(popup?: Popup | null): this {\n if (this._popup) {\n this._popup.remove();\n this._popup = null;\n this._element.removeEventListener('keypress', this._onKeyPress);\n\n if (!this._originalTabIndex) {\n this._element.removeAttribute('tabindex');\n }\n }\n\n if (popup) {\n if (!('offset' in popup.options)) {\n const markerHeight = 41 - (5.8 / 2);\n const markerRadius = 13.5;\n const linearOffset = Math.abs(markerRadius) / Math.SQRT2;\n popup.options.offset = this._defaultMarker ? {\n 'top': [0, 0],\n 'top-left': [0, 0],\n 'top-right': [0, 0],\n 'bottom': [0, -markerHeight],\n 'bottom-left': [linearOffset, (markerHeight - markerRadius + linearOffset) * -1],\n 'bottom-right': [-linearOffset, (markerHeight - markerRadius + linearOffset) * -1],\n 'left': [markerRadius, (markerHeight - markerRadius) * -1],\n 'right': [-markerRadius, (markerHeight - markerRadius) * -1]\n } as Offset : this._offset;\n }\n this._popup = popup;\n if (this._lngLat) this._popup.setLngLat(this._lngLat);\n\n this._originalTabIndex = this._element.getAttribute('tabindex');\n if (!this._originalTabIndex) {\n this._element.setAttribute('tabindex', '0');\n }\n this._element.addEventListener('keypress', this._onKeyPress);\n }\n\n return this;\n }\n\n _onKeyPress = (e: KeyboardEvent) => {\n const code = e.code;\n const legacyCode = e.charCode || e.keyCode;\n\n if (\n (code === 'Space') || (code === 'Enter') ||\n (legacyCode === 32) || (legacyCode === 13) // space or enter\n ) {\n this.togglePopup();\n }\n };\n\n _onMapClick = (e: MapMouseEvent) => {\n const targetElement = e.originalEvent.target;\n const element = this._element;\n\n if (this._popup && (targetElement === element || element.contains(targetElement as any))) {\n this.togglePopup();\n }\n };\n\n /**\n * Returns the {@link Popup} instance that is bound to the {@link Marker}.\n * @returns popup\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([0, 0])\n * .setPopup(new maplibregl.Popup().setHTML(\"

Hello World!

\"))\n * .addTo(map);\n *\n * console.log(marker.getPopup()); // return the popup instance\n * ```\n */\n getPopup(): Popup {\n return this._popup;\n }\n\n /**\n * Opens or closes the {@link Popup} instance that is bound to the {@link Marker}, depending on the current state of the {@link Popup}.\n * @returns `this`\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([0, 0])\n * .setPopup(new maplibregl.Popup().setHTML(\"

Hello World!

\"))\n * .addTo(map);\n *\n * marker.togglePopup(); // toggle popup open or closed\n * ```\n */\n togglePopup(): this {\n const popup = this._popup;\n\n if (!popup) return this;\n else if (popup.isOpen()) popup.remove();\n else popup.addTo(this._map);\n return this;\n }\n\n _update = (e?: { type: 'move' | 'moveend' | 'terrain' | 'render' }) => {\n if (!this._map) return;\n\n const isFullyLoaded = this._map.loaded() && !this._map.isMoving();\n if (e?.type === 'terrain' || (e?.type === 'render' && !isFullyLoaded)) {\n this._map.once('render', this._update);\n }\n\n if (this._map.transform.renderWorldCopies) {\n this._lngLat = smartWrap(this._lngLat, this._pos, this._map.transform);\n }\n\n this._pos = this._map.project(this._lngLat)._add(this._offset);\n\n let rotation = '';\n if (this._rotationAlignment === 'viewport' || this._rotationAlignment === 'auto') {\n rotation = `rotateZ(${this._rotation}deg)`;\n } else if (this._rotationAlignment === 'map') {\n rotation = `rotateZ(${this._rotation - this._map.getBearing()}deg)`;\n }\n\n let pitch = '';\n if (this._pitchAlignment === 'viewport' || this._pitchAlignment === 'auto') {\n pitch = 'rotateX(0deg)';\n } else if (this._pitchAlignment === 'map') {\n pitch = `rotateX(${this._map.getPitch()}deg)`;\n }\n\n // because rounding the coordinates at every `move` event causes stuttered zooming\n // we only round them when _update is called with `moveend` or when its called with\n // no arguments (when the Marker is initialized or Marker#setLngLat is invoked).\n if (!e || e.type === 'moveend') {\n this._pos = this._pos.round();\n }\n\n DOM.setTransform(this._element, `${anchorTranslate[this._anchor]} translate(${this._pos.x}px, ${this._pos.y}px) ${pitch} ${rotation}`);\n\n // in case of 3D, ask the terrain coords-framebuffer for this pos and check if the marker is visible\n // call this logic in setTimeout with a timeout of 100ms to save performance in map-movement\n if (this._map.terrain && !this._opacityTimeout) this._opacityTimeout = setTimeout(() => {\n const lnglat = this._map.unproject(this._pos);\n const metresPerPixel = 40075016.686 * Math.abs(Math.cos(this._lngLat.lat * Math.PI / 180)) / Math.pow(2, this._map.transform.tileZoom + 8);\n this._element.style.opacity = lnglat.distanceTo(this._lngLat) > metresPerPixel * 20 ? '0.2' : '1.0';\n this._opacityTimeout = null;\n }, 100);\n };\n\n /**\n * Get the marker's offset.\n * @returns The marker's screen coordinates in pixels.\n */\n getOffset(): Point {\n return this._offset;\n }\n\n /**\n * Sets the offset of the marker\n * @param offset - The offset in pixels as a {@link PointLike} object to apply relative to the element's center. Negatives indicate left and up.\n * @returns `this`\n */\n setOffset(offset: PointLike): this {\n this._offset = Point.convert(offset);\n this._update();\n return this;\n }\n\n /**\n * Adds a CSS class to the marker element.\n *\n * @param className - on-empty string with CSS class name to add to marker element\n *\n * @example\n * ```\n * let marker = new maplibregl.Marker()\n * marker.addClassName('some-class')\n * ```\n */\n addClassName(className: string) {\n this._element.classList.add(className);\n }\n\n /**\n * Removes a CSS class from the marker element.\n *\n * @param className - Non-empty string with CSS class name to remove from marker element\n *\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * marker.removeClassName('some-class')\n * ```\n */\n removeClassName(className: string) {\n this._element.classList.remove(className);\n }\n\n /**\n * Add or remove the given CSS class on the marker element, depending on whether the element currently has that class.\n *\n * @param className - Non-empty string with CSS class name to add/remove\n *\n * @returns if the class was removed return false, if class was added, then return true\n *\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * marker.toggleClassName('toggleClass')\n * ```\n */\n toggleClassName(className: string): boolean {\n return this._element.classList.toggle(className);\n }\n\n _onMove = (e: MapMouseEvent | MapTouchEvent) => {\n if (!this._isDragging) {\n const clickTolerance = this._clickTolerance || this._map._clickTolerance;\n this._isDragging = e.point.dist(this._pointerdownPos) >= clickTolerance;\n }\n if (!this._isDragging) return;\n\n this._pos = e.point.sub(this._positionDelta);\n this._lngLat = this._map.unproject(this._pos);\n this.setLngLat(this._lngLat);\n // suppress click event so that popups don't toggle on drag\n this._element.style.pointerEvents = 'none';\n\n // make sure dragstart only fires on the first move event after mousedown.\n // this can't be on mousedown because that event doesn't necessarily\n // imply that a drag is about to happen.\n if (this._state === 'pending') {\n this._state = 'active';\n this.fire(new Event('dragstart'));\n }\n this.fire(new Event('drag'));\n };\n\n _onUp = () => {\n // revert to normal pointer event handling\n this._element.style.pointerEvents = 'auto';\n this._positionDelta = null;\n this._pointerdownPos = null;\n this._isDragging = false;\n this._map.off('mousemove', this._onMove);\n this._map.off('touchmove', this._onMove);\n\n // only fire dragend if it was preceded by at least one drag event\n if (this._state === 'active') {\n this.fire(new Event('dragend'));\n }\n\n this._state = 'inactive';\n };\n\n _addDragHandler = (e: MapMouseEvent | MapTouchEvent) => {\n if (this._element.contains(e.originalEvent.target as any)) {\n e.preventDefault();\n\n // We need to calculate the pixel distance between the click point\n // and the marker position, with the offset accounted for. Then we\n // can subtract this distance from the mousemove event's position\n // to calculate the new marker position.\n // If we don't do this, the marker 'jumps' to the click position\n // creating a jarring UX effect.\n this._positionDelta = e.point.sub(this._pos).add(this._offset);\n\n this._pointerdownPos = e.point;\n\n this._state = 'pending';\n this._map.on('mousemove', this._onMove);\n this._map.on('touchmove', this._onMove);\n this._map.once('mouseup', this._onUp);\n this._map.once('touchend', this._onUp);\n }\n };\n\n /**\n * Sets the `draggable` property and functionality of the marker\n * @param shouldBeDraggable - Turns drag functionality on/off\n * @returns `this`\n */\n setDraggable(shouldBeDraggable?: boolean): this {\n this._draggable = !!shouldBeDraggable; // convert possible undefined value to false\n\n // handle case where map may not exist yet\n // e.g. when setDraggable is called before addTo\n if (this._map) {\n if (shouldBeDraggable) {\n this._map.on('mousedown', this._addDragHandler);\n this._map.on('touchstart', this._addDragHandler);\n } else {\n this._map.off('mousedown', this._addDragHandler);\n this._map.off('touchstart', this._addDragHandler);\n }\n }\n\n return this;\n }\n\n /**\n * Returns true if the marker can be dragged\n * @returns True if the marker is draggable.\n */\n isDraggable(): boolean {\n return this._draggable;\n }\n\n /**\n * Sets the `rotation` property of the marker.\n * @param rotation - The rotation angle of the marker (clockwise, in degrees), relative to its respective {@link Marker#setRotationAlignment} setting.\n * @returns `this`\n */\n setRotation(rotation?: number): this {\n this._rotation = rotation || 0;\n this._update();\n return this;\n }\n\n /**\n * Returns the current rotation angle of the marker (in degrees).\n * @returns The current rotation angle of the marker.\n */\n getRotation(): number {\n return this._rotation;\n }\n\n /**\n * Sets the `rotationAlignment` property of the marker.\n * @param alignment - Sets the `rotationAlignment` property of the marker. defaults to 'auto'\n * @returns `this`\n */\n setRotationAlignment(alignment?: Alignment): this {\n this._rotationAlignment = alignment || 'auto';\n this._update();\n return this;\n }\n\n /**\n * Returns the current `rotationAlignment` property of the marker.\n * @returns The current rotational alignment of the marker.\n */\n getRotationAlignment(): Alignment {\n return this._rotationAlignment;\n }\n\n /**\n * Sets the `pitchAlignment` property of the marker.\n * @param alignment - Sets the `pitchAlignment` property of the marker. If alignment is 'auto', it will automatically match `rotationAlignment`.\n * @returns `this`\n */\n setPitchAlignment(alignment?: Alignment): this {\n this._pitchAlignment = alignment && alignment !== 'auto' ? alignment : this._rotationAlignment;\n this._update();\n return this;\n }\n\n /**\n * Returns the current `pitchAlignment` property of the marker.\n * @returns The current pitch alignment of the marker in degrees.\n */\n getPitchAlignment(): Alignment {\n return this._pitchAlignment;\n }\n}\n","import {Event, Evented} from '../../util/evented';\nimport {DOM} from '../../util/dom';\nimport {extend, warnOnce} from '../../util/util';\nimport {checkGeolocationSupport} from '../../util/geolocation_support';\nimport {LngLat} from '../../geo/lng_lat';\nimport {Marker} from '../marker';\n\nimport type {Map} from '../map';\nimport type {FitBoundsOptions} from '../camera';\nimport type {IControl} from './control';\nimport {LngLatBounds} from '../../geo/lng_lat_bounds';\n\n/**\n * The {@link GeolocateControl} options\n */\ntype GeolocateOptions = {\n /**\n * A Geolocation API [PositionOptions](https://developer.mozilla.org/en-US/docs/Web/API/PositionOptions) object.\n * @defaultValue `{enableHighAccuracy: false, timeout: 6000}`\n */\n positionOptions?: PositionOptions;\n /**\n * A options object to use when the map is panned and zoomed to the user's location. The default is to use a `maxZoom` of 15 to limit how far the map will zoom in for very accurate locations.\n */\n fitBoundsOptions?: FitBoundsOptions;\n /**\n * If `true` the Geolocate Control becomes a toggle button and when active the map will receive updates to the user's location as it changes.\n * @defaultValue false\n */\n trackUserLocation?: boolean;\n /**\n * By default, if showUserLocation is `true`, a transparent circle will be drawn around the user location indicating the accuracy (95% confidence level) of the user's location. Set to `false` to disable. Always disabled when showUserLocation is `false`.\n * @defaultValue true\n */\n showAccuracyCircle?: boolean;\n /**\n * By default a dot will be shown on the map at the user's location. Set to `false` to disable.\n * @defaultValue true\n */\n showUserLocation?: boolean;\n};\n\nconst defaultOptions: GeolocateOptions = {\n positionOptions: {\n enableHighAccuracy: false,\n maximumAge: 0,\n timeout: 6000 /* 6 sec */\n },\n fitBoundsOptions: {\n maxZoom: 15\n },\n trackUserLocation: false,\n showAccuracyCircle: true,\n showUserLocation: true\n};\n\nlet numberOfWatches = 0;\nlet noTimeout = false;\n\n/**\n * A `GeolocateControl` control provides a button that uses the browser's geolocation\n * API to locate the user on the map.\n *\n * Not all browsers support geolocation,\n * and some users may disable the feature. Geolocation support for modern\n * browsers including Chrome requires sites to be served over HTTPS. If\n * geolocation support is not available, the GeolocateControl will show\n * as disabled.\n *\n * The zoom level applied will depend on the accuracy of the geolocation provided by the device.\n *\n * The GeolocateControl has two modes. If `trackUserLocation` is `false` (default) the control acts as a button, which when pressed will set the map's camera to target the user location. If the user moves, the map won't update. This is most suited for the desktop. If `trackUserLocation` is `true` the control acts as a toggle button that when active the user's location is actively monitored for changes. In this mode the GeolocateControl has three interaction states:\n * * active - the map's camera automatically updates as the user's location changes, keeping the location dot in the center. Initial state and upon clicking the `GeolocateControl` button.\n * * passive - the user's location dot automatically updates, but the map's camera does not. Occurs upon the user initiating a map movement.\n * * disabled - occurs if Geolocation is not available, disabled or denied.\n *\n * These interaction states can't be controlled programmatically, rather they are set based on user interactions.\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * map.addControl(new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * }));\n * ```\n * @see [Locate the user](https://maplibre.org/maplibre-gl-js/docs/examples/locate-user/)\n *\n * ### Events\n *\n * @event `trackuserlocationend` - Fired when the Geolocate Control changes to the background state, which happens when a user changes the camera during an active position lock. This only applies when trackUserLocation is true. In the background state, the dot on the map will update with location updates but the camera will not.\n *\n * @event `trackuserlocationstart` - Fired when the Geolocate Control changes to the active lock state, which happens either upon first obtaining a successful Geolocation API position for the user (a geolocate event will follow), or the user clicks the geolocate button when in the background state which uses the last known position to recenter the map and enter active lock state (no geolocate event will follow unless the users's location changes).\n *\n * @event `geolocate` - Fired on each Geolocation API position update which returned as success.\n * `data` - The returned [Position](https://developer.mozilla.org/en-US/docs/Web/API/Position) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition).\n *\n * @event `error` - Fired on each Geolocation API position update which returned as an error.\n * `data` - The returned [PositionError](https://developer.mozilla.org/en-US/docs/Web/API/PositionError) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition).\n *\n * @event `outofmaxbounds` Fired on each Geolocation API position update which returned as success but user position is out of map maxBounds.\n * `data` - The returned [Position](https://developer.mozilla.org/en-US/docs/Web/API/Position) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition).\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when a trackuserlocationend event occurs.\n * geolocate.on('trackuserlocationend', function() {\n * console.log('A trackuserlocationend event has occurred.')\n * });\n * ```\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when a trackuserlocationstart event occurs.\n * geolocate.on('trackuserlocationstart', function() {\n * console.log('A trackuserlocationstart event has occurred.')\n * });\n * ```\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when a geolocate event occurs.\n * geolocate.on('geolocate', function() {\n * console.log('A geolocate event has occurred.')\n * });\n * ```\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when an error event occurs.\n * geolocate.on('error', function() {\n * console.log('An error event has occurred.')\n * });\n * ```\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when an outofmaxbounds event occurs.\n * geolocate.on('outofmaxbounds', function() {\n * console.log('An outofmaxbounds event has occurred.')\n * });\n * ```\n */\nexport class GeolocateControl extends Evented implements IControl {\n _map: Map;\n options: GeolocateOptions;\n _container: HTMLElement;\n _dotElement: HTMLElement;\n _circleElement: HTMLElement;\n _geolocateButton: HTMLButtonElement;\n _geolocationWatchID: number;\n _timeoutId: ReturnType;\n /* Geolocate Control Watch States\n * This is the private state of the control.\n *\n * OFF\n * off/inactive\n * WAITING_ACTIVE\n * Geolocate Control was clicked but still waiting for Geolocation API response with user location\n * ACTIVE_LOCK\n * Showing the user location as a dot AND tracking the camera to be fixed to their location. If their location changes the map moves to follow.\n * ACTIVE_ERROR\n * There was en error from the Geolocation API while trying to show and track the user location.\n * BACKGROUND\n * Showing the user location as a dot but the camera doesn't follow their location as it changes.\n * BACKGROUND_ERROR\n * There was an error from the Geolocation API while trying to show (but not track) the user location.\n */\n _watchState: 'OFF' | 'ACTIVE_LOCK' | 'WAITING_ACTIVE' | 'ACTIVE_ERROR' | 'BACKGROUND' | 'BACKGROUND_ERROR';\n _lastKnownPosition: any;\n _userLocationDotMarker: Marker;\n _accuracyCircleMarker: Marker;\n _accuracy: number;\n _setup: boolean; // set to true once the control has been setup\n\n constructor(options: GeolocateOptions) {\n super();\n this.options = extend({}, defaultOptions, options);\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group');\n checkGeolocationSupport(this._setupUI);\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n // clear the geolocation watch if exists\n if (this._geolocationWatchID !== undefined) {\n window.navigator.geolocation.clearWatch(this._geolocationWatchID);\n this._geolocationWatchID = undefined;\n }\n\n // clear the markers from the map\n if (this.options.showUserLocation && this._userLocationDotMarker) {\n this._userLocationDotMarker.remove();\n }\n if (this.options.showAccuracyCircle && this._accuracyCircleMarker) {\n this._accuracyCircleMarker.remove();\n }\n\n DOM.remove(this._container);\n this._map.off('zoom', this._onZoom);\n this._map = undefined;\n numberOfWatches = 0;\n noTimeout = false;\n }\n\n /**\n * Check if the Geolocation API Position is outside the map's maxbounds.\n *\n * @param position - the Geolocation API Position\n * @returns `true` if position is outside the map's maxbounds, otherwise returns `false`.\n */\n _isOutOfMapMaxBounds(position: GeolocationPosition) {\n const bounds = this._map.getMaxBounds();\n const coordinates = position.coords;\n\n return bounds && (\n coordinates.longitude < bounds.getWest() ||\n coordinates.longitude > bounds.getEast() ||\n coordinates.latitude < bounds.getSouth() ||\n coordinates.latitude > bounds.getNorth()\n );\n }\n\n _setErrorState() {\n switch (this._watchState) {\n case 'WAITING_ACTIVE':\n this._watchState = 'ACTIVE_ERROR';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active-error');\n break;\n case 'ACTIVE_LOCK':\n this._watchState = 'ACTIVE_ERROR';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active-error');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting');\n // turn marker grey\n break;\n case 'BACKGROUND':\n this._watchState = 'BACKGROUND_ERROR';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background-error');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting');\n // turn marker grey\n break;\n case 'ACTIVE_ERROR':\n break;\n default:\n throw new Error(`Unexpected watchState ${this._watchState}`);\n }\n }\n\n /**\n * When the Geolocation API returns a new location, update the GeolocateControl.\n *\n * @param position - the Geolocation API Position\n */\n _onSuccess = (position: GeolocationPosition) => {\n if (!this._map) {\n // control has since been removed\n return;\n }\n\n if (this._isOutOfMapMaxBounds(position)) {\n this._setErrorState();\n\n this.fire(new Event('outofmaxbounds', position));\n this._updateMarker();\n this._finish();\n\n return;\n }\n\n if (this.options.trackUserLocation) {\n // keep a record of the position so that if the state is BACKGROUND and the user\n // clicks the button, we can move to ACTIVE_LOCK immediately without waiting for\n // watchPosition to trigger _onSuccess\n this._lastKnownPosition = position;\n\n switch (this._watchState) {\n case 'WAITING_ACTIVE':\n case 'ACTIVE_LOCK':\n case 'ACTIVE_ERROR':\n this._watchState = 'ACTIVE_LOCK';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active');\n break;\n case 'BACKGROUND':\n case 'BACKGROUND_ERROR':\n this._watchState = 'BACKGROUND';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background');\n break;\n default:\n throw new Error(`Unexpected watchState ${this._watchState}`);\n }\n }\n\n // if showUserLocation and the watch state isn't off then update the marker location\n if (this.options.showUserLocation && this._watchState !== 'OFF') {\n this._updateMarker(position);\n }\n\n // if in normal mode (not watch mode), or if in watch mode and the state is active watch\n // then update the camera\n if (!this.options.trackUserLocation || this._watchState === 'ACTIVE_LOCK') {\n this._updateCamera(position);\n }\n\n if (this.options.showUserLocation) {\n this._dotElement.classList.remove('maplibregl-user-location-dot-stale');\n }\n\n this.fire(new Event('geolocate', position));\n this._finish();\n };\n\n /**\n * Update the camera location to center on the current position\n *\n * @param position - the Geolocation API Position\n */\n _updateCamera = (position: GeolocationPosition) => {\n const center = new LngLat(position.coords.longitude, position.coords.latitude);\n const radius = position.coords.accuracy;\n const bearing = this._map.getBearing();\n const options = extend({bearing}, this.options.fitBoundsOptions);\n const newBounds = LngLatBounds.fromLngLat(center, radius);\n\n this._map.fitBounds(newBounds, options, {\n geolocateSource: true // tag this camera change so it won't cause the control to change to background state\n });\n };\n\n /**\n * Update the user location dot Marker to the current position\n *\n * @param position - the Geolocation API Position\n */\n _updateMarker = (position?: GeolocationPosition | null) => {\n if (position) {\n const center = new LngLat(position.coords.longitude, position.coords.latitude);\n this._accuracyCircleMarker.setLngLat(center).addTo(this._map);\n this._userLocationDotMarker.setLngLat(center).addTo(this._map);\n this._accuracy = position.coords.accuracy;\n if (this.options.showUserLocation && this.options.showAccuracyCircle) {\n this._updateCircleRadius();\n }\n } else {\n this._userLocationDotMarker.remove();\n this._accuracyCircleMarker.remove();\n }\n };\n\n _updateCircleRadius() {\n const bounds = this._map.getBounds();\n const southEastPoint = bounds.getSouthEast();\n const northEastPoint = bounds.getNorthEast();\n const mapHeightInMeters = southEastPoint.distanceTo(northEastPoint);\n const mapHeightInPixels = this._map._container.clientHeight;\n const circleDiameter = Math.ceil(2 * (this._accuracy / (mapHeightInMeters / mapHeightInPixels)));\n this._circleElement.style.width = `${circleDiameter}px`;\n this._circleElement.style.height = `${circleDiameter}px`;\n }\n\n _onZoom = () => {\n if (this.options.showUserLocation && this.options.showAccuracyCircle) {\n this._updateCircleRadius();\n }\n };\n\n _onError = (error: GeolocationPositionError) => {\n if (!this._map) {\n // control has since been removed\n return;\n }\n\n if (this.options.trackUserLocation) {\n if (error.code === 1) {\n // PERMISSION_DENIED\n this._watchState = 'OFF';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error');\n this._geolocateButton.disabled = true;\n const title = this._map._getUIString('GeolocateControl.LocationNotAvailable');\n this._geolocateButton.title = title;\n this._geolocateButton.setAttribute('aria-label', title);\n\n if (this._geolocationWatchID !== undefined) {\n this._clearWatch();\n }\n } else if (error.code === 3 && noTimeout) {\n // this represents a forced error state\n // this was triggered to force immediate geolocation when a watch is already present\n // see https://github.com/mapbox/mapbox-gl-js/issues/8214\n // and https://w3c.github.io/geolocation-api/#example-5-forcing-the-user-agent-to-return-a-fresh-cached-position\n return;\n } else {\n this._setErrorState();\n }\n }\n\n if (this._watchState !== 'OFF' && this.options.showUserLocation) {\n this._dotElement.classList.add('maplibregl-user-location-dot-stale');\n }\n\n this.fire(new Event('error', error));\n\n this._finish();\n };\n\n _finish = () => {\n if (this._timeoutId) { clearTimeout(this._timeoutId); }\n this._timeoutId = undefined;\n };\n\n _setupUI = (supported: boolean) => {\n // this method is called asynchronously during onAdd\n // the control could have been removed before reaching here\n if (!this._map) {\n return;\n }\n\n this._container.addEventListener('contextmenu', (e: MouseEvent) => e.preventDefault());\n this._geolocateButton = DOM.create('button', 'maplibregl-ctrl-geolocate', this._container);\n DOM.create('span', 'maplibregl-ctrl-icon', this._geolocateButton).setAttribute('aria-hidden', 'true');\n this._geolocateButton.type = 'button';\n\n if (supported === false) {\n warnOnce('Geolocation support is not available so the GeolocateControl will be disabled.');\n const title = this._map._getUIString('GeolocateControl.LocationNotAvailable');\n this._geolocateButton.disabled = true;\n this._geolocateButton.title = title;\n this._geolocateButton.setAttribute('aria-label', title);\n } else {\n const title = this._map._getUIString('GeolocateControl.FindMyLocation');\n this._geolocateButton.title = title;\n this._geolocateButton.setAttribute('aria-label', title);\n }\n\n if (this.options.trackUserLocation) {\n this._geolocateButton.setAttribute('aria-pressed', 'false');\n this._watchState = 'OFF';\n }\n\n // when showUserLocation is enabled, keep the Geolocate button disabled until the device location marker is setup on the map\n if (this.options.showUserLocation) {\n this._dotElement = DOM.create('div', 'maplibregl-user-location-dot');\n\n this._userLocationDotMarker = new Marker({element: this._dotElement});\n\n this._circleElement = DOM.create('div', 'maplibregl-user-location-accuracy-circle');\n this._accuracyCircleMarker = new Marker({element: this._circleElement, pitchAlignment: 'map'});\n\n if (this.options.trackUserLocation) this._watchState = 'OFF';\n\n this._map.on('zoom', this._onZoom);\n }\n\n this._geolocateButton.addEventListener('click',\n this.trigger.bind(this));\n\n this._setup = true;\n\n // when the camera is changed (and it's not as a result of the Geolocation Control) change\n // the watch mode to background watch, so that the marker is updated but not the camera.\n if (this.options.trackUserLocation) {\n this._map.on('movestart', (event: any) => {\n const fromResize = event.originalEvent && event.originalEvent.type === 'resize';\n if (!event.geolocateSource && this._watchState === 'ACTIVE_LOCK' && !fromResize) {\n this._watchState = 'BACKGROUND';\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n\n this.fire(new Event('trackuserlocationend'));\n }\n });\n }\n };\n\n /**\n * Programmatically request and move the map to the user's location.\n *\n * @returns `false` if called before control was added to a map, otherwise returns `true`.\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * map.on('load', function() {\n * geolocate.trigger();\n * });\n * ```\n */\n trigger(): boolean {\n if (!this._setup) {\n warnOnce('Geolocate control triggered before added to a map');\n return false;\n }\n if (this.options.trackUserLocation) {\n // update watchState and do any outgoing state cleanup\n switch (this._watchState) {\n case 'OFF':\n // turn on the Geolocate Control\n this._watchState = 'WAITING_ACTIVE';\n\n this.fire(new Event('trackuserlocationstart'));\n break;\n case 'WAITING_ACTIVE':\n case 'ACTIVE_LOCK':\n case 'ACTIVE_ERROR':\n case 'BACKGROUND_ERROR':\n // turn off the Geolocate Control\n numberOfWatches--;\n noTimeout = false;\n this._watchState = 'OFF';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error');\n\n this.fire(new Event('trackuserlocationend'));\n break;\n case 'BACKGROUND':\n this._watchState = 'ACTIVE_LOCK';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background');\n // set camera to last known location\n if (this._lastKnownPosition) this._updateCamera(this._lastKnownPosition);\n\n this.fire(new Event('trackuserlocationstart'));\n break;\n default:\n throw new Error(`Unexpected watchState ${this._watchState}`);\n }\n\n // incoming state setup\n switch (this._watchState) {\n case 'WAITING_ACTIVE':\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active');\n break;\n case 'ACTIVE_LOCK':\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active');\n break;\n case 'OFF':\n break;\n default:\n throw new Error(`Unexpected watchState ${this._watchState}`);\n }\n\n // manage geolocation.watchPosition / geolocation.clearWatch\n if (this._watchState === 'OFF' && this._geolocationWatchID !== undefined) {\n // clear watchPosition as we've changed to an OFF state\n this._clearWatch();\n } else if (this._geolocationWatchID === undefined) {\n // enable watchPosition since watchState is not OFF and there is no watchPosition already running\n\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.setAttribute('aria-pressed', 'true');\n\n numberOfWatches++;\n let positionOptions;\n if (numberOfWatches > 1) {\n positionOptions = {maximumAge: 600000, timeout: 0};\n noTimeout = true;\n } else {\n positionOptions = this.options.positionOptions;\n noTimeout = false;\n }\n\n this._geolocationWatchID = window.navigator.geolocation.watchPosition(\n this._onSuccess, this._onError, positionOptions);\n }\n } else {\n window.navigator.geolocation.getCurrentPosition(\n this._onSuccess, this._onError, this.options.positionOptions);\n\n // This timeout ensures that we still call finish() even if\n // the user declines to share their location in Firefox\n this._timeoutId = setTimeout(this._finish, 10000 /* 10sec */);\n }\n\n return true;\n }\n\n _clearWatch() {\n window.navigator.geolocation.clearWatch(this._geolocationWatchID);\n\n this._geolocationWatchID = undefined;\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.setAttribute('aria-pressed', 'false');\n\n if (this.options.showUserLocation) {\n this._updateMarker(null);\n }\n }\n}\n\n","import {DOM} from '../../util/dom';\nimport {extend} from '../../util/util';\n\nimport type {Map} from '../map';\nimport type {ControlPosition, IControl} from './control';\n\n/**\n * The unit type for length to use for the {@link ScaleControl}\n */\nexport type Unit = 'imperial' | 'metric' | 'nautical';\n\n/**\n * The {@link ScaleControl} options object\n */\ntype ScaleOptions = {\n /**\n * The maximum length of the scale control in pixels.\n * @defaultValue 100\n */\n maxWidth?: number;\n /**\n * Unit of the distance (`'imperial'`, `'metric'` or `'nautical'`).\n * @defaultValue 'metric'\n */\n unit?: Unit;\n};\n\nconst defaultOptions: ScaleOptions = {\n maxWidth: 100,\n unit: 'metric'\n};\n\n/**\n * A `ScaleControl` control displays the ratio of a distance on the map to the corresponding distance on the ground.\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * let scale = new maplibregl.ScaleControl({\n * maxWidth: 80,\n * unit: 'imperial'\n * });\n * map.addControl(scale);\n *\n * scale.setUnit('metric');\n * ```\n */\nexport class ScaleControl implements IControl {\n _map: Map;\n _container: HTMLElement;\n options: ScaleOptions;\n\n constructor(options: ScaleOptions) {\n this.options = extend({}, defaultOptions, options);\n }\n\n getDefaultPosition(): ControlPosition {\n return 'bottom-left';\n }\n\n _onMove = () => {\n updateScale(this._map, this._container, this.options);\n };\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-scale', map.getContainer());\n\n this._map.on('move', this._onMove);\n this._onMove();\n\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._container);\n this._map.off('move', this._onMove);\n this._map = undefined;\n }\n\n /**\n * Set the scale's unit of the distance\n *\n * @param unit - Unit of the distance (`'imperial'`, `'metric'` or `'nautical'`).\n */\n setUnit = (unit: Unit) => {\n this.options.unit = unit;\n updateScale(this._map, this._container, this.options);\n };\n}\n\nfunction updateScale(map, container, options) {\n // A horizontal scale is imagined to be present at center of the map\n // container with maximum length (Default) as 100px.\n // Using spherical law of cosines approximation, the real distance is\n // found between the two coordinates.\n const maxWidth = options && options.maxWidth || 100;\n\n const y = map._container.clientHeight / 2;\n const left = map.unproject([0, y]);\n const right = map.unproject([maxWidth, y]);\n const maxMeters = left.distanceTo(right);\n // The real distance corresponding to 100px scale length is rounded off to\n // near pretty number and the scale length for the same is found out.\n // Default unit of the scale is based on User's locale.\n if (options && options.unit === 'imperial') {\n const maxFeet = 3.2808 * maxMeters;\n if (maxFeet > 5280) {\n const maxMiles = maxFeet / 5280;\n setScale(container, maxWidth, maxMiles, map._getUIString('ScaleControl.Miles'));\n } else {\n setScale(container, maxWidth, maxFeet, map._getUIString('ScaleControl.Feet'));\n }\n } else if (options && options.unit === 'nautical') {\n const maxNauticals = maxMeters / 1852;\n setScale(container, maxWidth, maxNauticals, map._getUIString('ScaleControl.NauticalMiles'));\n } else if (maxMeters >= 1000) {\n setScale(container, maxWidth, maxMeters / 1000, map._getUIString('ScaleControl.Kilometers'));\n } else {\n setScale(container, maxWidth, maxMeters, map._getUIString('ScaleControl.Meters'));\n }\n}\n\nfunction setScale(container, maxWidth, maxDistance, unit) {\n const distance = getRoundNum(maxDistance);\n const ratio = distance / maxDistance;\n container.style.width = `${maxWidth * ratio}px`;\n container.innerHTML = `${distance} ${unit}`;\n}\n\nfunction getDecimalRoundNum(d) {\n const multiplier = Math.pow(10, Math.ceil(-Math.log(d) / Math.LN10));\n return Math.round(d * multiplier) / multiplier;\n}\n\nfunction getRoundNum(num) {\n const pow10 = Math.pow(10, (`${Math.floor(num)}`).length - 1);\n let d = num / pow10;\n\n d = d >= 10 ? 10 :\n d >= 5 ? 5 :\n d >= 3 ? 3 :\n d >= 2 ? 2 :\n d >= 1 ? 1 : getDecimalRoundNum(d);\n\n return pow10 * d;\n}\n","import {DOM} from '../../util/dom';\n\nimport {warnOnce} from '../../util/util';\n\nimport {Event, Evented} from '../../util/evented';\nimport type {Map, GestureOptions} from '../map';\nimport type {IControl} from './control';\n\n/**\n * The {@link FullscreenControl} options\n */\ntype FullscreenOptions = {\n /**\n * `container` is the [compatible DOM element](https://developer.mozilla.org/en-US/docs/Web/API/Element/requestFullScreen#Compatible_elements) which should be made full screen. By default, the map container element will be made full screen.\n */\n container?: HTMLElement;\n};\n\n/**\n * A `FullscreenControl` control contains a button for toggling the map in and out of fullscreen mode.\n * When [requestFullscreen](https://developer.mozilla.org/en-US/docs/Web/API/Element/requestFullscreen) is not supported, fullscreen is handled via CSS properties.\n * The map's `cooperativeGestures` option is temporarily disabled while the map\n * is in fullscreen mode, and is restored when the map exist fullscreen mode.\n *\n * @group Markers and Controls\n * @param options - the full screen control options\n *\n * @example\n * ```ts\n * map.addControl(new maplibregl.FullscreenControl({container: document.querySelector('body')}));\n * ```\n * @see [View a fullscreen map](https://maplibre.org/maplibre-gl-js/docs/examples/fullscreen/)\n *\n * ### Events\n *\n * @event `fullscreenstart` - Fired when fullscreen mode has started\n *\n * @event `fullscreenend` - Fired when fullscreen mode has ended\n */\nexport class FullscreenControl extends Evented implements IControl {\n _map: Map;\n _controlContainer: HTMLElement;\n _fullscreen: boolean;\n _fullscreenchange: string;\n _fullscreenButton: HTMLButtonElement;\n _container: HTMLElement;\n _prevCooperativeGestures: boolean | GestureOptions;\n\n constructor(options: FullscreenOptions = {}) {\n super();\n this._fullscreen = false;\n\n if (options && options.container) {\n if (options.container instanceof HTMLElement) {\n this._container = options.container;\n } else {\n warnOnce('Full screen control \\'container\\' must be a DOM element.');\n }\n }\n\n if ('onfullscreenchange' in document) {\n this._fullscreenchange = 'fullscreenchange';\n } else if ('onmozfullscreenchange' in document) {\n this._fullscreenchange = 'mozfullscreenchange';\n } else if ('onwebkitfullscreenchange' in document) {\n this._fullscreenchange = 'webkitfullscreenchange';\n } else if ('onmsfullscreenchange' in document) {\n this._fullscreenchange = 'MSFullscreenChange';\n }\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map): HTMLElement {\n this._map = map;\n if (!this._container) this._container = this._map.getContainer();\n this._controlContainer = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group');\n this._setupUI();\n return this._controlContainer;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._controlContainer);\n this._map = null;\n window.document.removeEventListener(this._fullscreenchange, this._onFullscreenChange);\n }\n\n _setupUI() {\n const button = this._fullscreenButton = DOM.create('button', (('maplibregl-ctrl-fullscreen')), this._controlContainer);\n DOM.create('span', 'maplibregl-ctrl-icon', button).setAttribute('aria-hidden', 'true');\n button.type = 'button';\n this._updateTitle();\n this._fullscreenButton.addEventListener('click', this._onClickFullscreen);\n window.document.addEventListener(this._fullscreenchange, this._onFullscreenChange);\n }\n\n _updateTitle() {\n const title = this._getTitle();\n this._fullscreenButton.setAttribute('aria-label', title);\n this._fullscreenButton.title = title;\n }\n\n _getTitle() {\n return this._map._getUIString(this._isFullscreen() ? 'FullscreenControl.Exit' : 'FullscreenControl.Enter');\n }\n\n _isFullscreen() {\n return this._fullscreen;\n }\n\n _onFullscreenChange = () => {\n const fullscreenElement =\n window.document.fullscreenElement ||\n (window.document as any).mozFullScreenElement ||\n (window.document as any).webkitFullscreenElement ||\n (window.document as any).msFullscreenElement;\n\n if ((fullscreenElement === this._container) !== this._fullscreen) {\n this._handleFullscreenChange();\n }\n };\n\n _handleFullscreenChange() {\n this._fullscreen = !this._fullscreen;\n this._fullscreenButton.classList.toggle('maplibregl-ctrl-shrink');\n this._fullscreenButton.classList.toggle('maplibregl-ctrl-fullscreen');\n this._updateTitle();\n\n if (this._fullscreen) {\n this.fire(new Event('fullscreenstart'));\n if (this._map._cooperativeGestures) {\n this._prevCooperativeGestures = this._map._cooperativeGestures;\n this._map.setCooperativeGestures();\n }\n } else {\n this.fire(new Event('fullscreenend'));\n if (this._prevCooperativeGestures) {\n this._map.setCooperativeGestures(this._prevCooperativeGestures);\n delete this._prevCooperativeGestures;\n }\n }\n }\n\n _onClickFullscreen = () => {\n if (this._isFullscreen()) {\n this._exitFullscreen();\n } else {\n this._requestFullscreen();\n }\n };\n\n _exitFullscreen() {\n if (window.document.exitFullscreen) {\n (window.document as any).exitFullscreen();\n } else if ((window.document as any).mozCancelFullScreen) {\n (window.document as any).mozCancelFullScreen();\n } else if ((window.document as any).msExitFullscreen) {\n (window.document as any).msExitFullscreen();\n } else if ((window.document as any).webkitCancelFullScreen) {\n (window.document as any).webkitCancelFullScreen();\n } else {\n this._togglePseudoFullScreen();\n }\n }\n\n _requestFullscreen() {\n if (this._container.requestFullscreen) {\n this._container.requestFullscreen();\n } else if ((this._container as any).mozRequestFullScreen) {\n (this._container as any).mozRequestFullScreen();\n } else if ((this._container as any).msRequestFullscreen) {\n (this._container as any).msRequestFullscreen();\n } else if ((this._container as any).webkitRequestFullscreen) {\n (this._container as any).webkitRequestFullscreen();\n } else {\n this._togglePseudoFullScreen();\n }\n }\n\n _togglePseudoFullScreen() {\n this._container.classList.toggle('maplibregl-pseudo-fullscreen');\n this._handleFullscreenChange();\n this._map.resize();\n }\n}\n","import {DOM} from '../../util/dom';\n\nimport type {Map} from '../map';\nimport type {IControl} from './control';\nimport type {TerrainSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A `TerrainControl` control contains a button for turning the terrain on and off.\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * let map = new maplibregl.Map({TerrainControl: false})\n * .addControl(new maplibregl.TerrainControl({\n * source: \"terrain\"\n * }));\n * ```\n */\nexport class TerrainControl implements IControl {\n options: TerrainSpecification;\n _map: Map;\n _container: HTMLElement;\n _terrainButton: HTMLButtonElement;\n\n constructor(options: TerrainSpecification) {\n this.options = options;\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group');\n this._terrainButton = DOM.create('button', 'maplibregl-ctrl-terrain', this._container);\n DOM.create('span', 'maplibregl-ctrl-icon', this._terrainButton).setAttribute('aria-hidden', 'true');\n this._terrainButton.type = 'button';\n this._terrainButton.addEventListener('click', this._toggleTerrain);\n\n this._updateTerrainIcon();\n this._map.on('terrain', this._updateTerrainIcon);\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._container);\n this._map.off('terrain', this._updateTerrainIcon);\n this._map = undefined;\n }\n\n _toggleTerrain = () => {\n if (this._map.getTerrain()) {\n this._map.setTerrain(null);\n } else {\n this._map.setTerrain(this.options);\n }\n this._updateTerrainIcon();\n };\n\n _updateTerrainIcon = () => {\n this._terrainButton.classList.remove('maplibregl-ctrl-terrain');\n this._terrainButton.classList.remove('maplibregl-ctrl-terrain-enabled');\n if (this._map.terrain) {\n this._terrainButton.classList.add('maplibregl-ctrl-terrain-enabled');\n this._terrainButton.title = this._map._getUIString('TerrainControl.disableTerrain');\n } else {\n this._terrainButton.classList.add('maplibregl-ctrl-terrain');\n this._terrainButton.title = this._map._getUIString('TerrainControl.enableTerrain');\n }\n };\n}\n","import {extend} from '../util/util';\nimport {Event, Evented} from '../util/evented';\nimport {MapMouseEvent} from '../ui/events';\nimport {DOM} from '../util/dom';\nimport {LngLat} from '../geo/lng_lat';\nimport Point from '@mapbox/point-geometry';\nimport {smartWrap} from '../util/smart_wrap';\nimport {anchorTranslate, applyAnchorClass} from './anchor';\n\nimport type {PositionAnchor} from './anchor';\nimport type {Map} from './map';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport type {PointLike} from './camera';\n\nconst defaultOptions = {\n closeButton: true,\n closeOnClick: true,\n focusAfterOpen: true,\n className: '',\n maxWidth: '240px'\n};\n\n/**\n * A pixel offset specified as:\n * - a single number specifying a distance from the location\n * - a {@link PointLike} specifying a constant offset\n * - an object of {@link Point}s specifying an offset for each anchor position\n * Negative offsets indicate left and up.\n */\nexport type Offset = number | PointLike | {\n [_ in PositionAnchor]: PointLike;\n};\n\nexport type PopupOptions = {\n /**\n * If `true`, a close button will appear in the top right corner of the popup.\n * @defaultValue true\n */\n closeButton?: boolean;\n /**\n * If `true`, the popup will closed when the map is clicked.\n * @defaultValue true\n */\n closeOnClick?: boolean;\n /**\n * If `true`, the popup will closed when the map moves.\n * @defaultValue false\n */\n closeOnMove?: boolean;\n /**\n * If `true`, the popup will try to focus the first focusable element inside the popup.\n * @defaultValue true\n */\n focusAfterOpen?: boolean;\n /**\n * A string indicating the part of the Popup that should\n * be positioned closest to the coordinate set via {@link Popup#setLngLat}.\n * Options are `'center'`, `'top'`, `'bottom'`, `'left'`, `'right'`, `'top-left'`,\n * `'top-right'`, `'bottom-left'`, and `'bottom-right'`. If unset the anchor will be\n * dynamically set to ensure the popup falls within the map container with a preference\n * for `'bottom'`.\n */\n anchor?: PositionAnchor;\n /**\n * A pixel offset applied to the popup's location\n */\n offset?: Offset;\n /**\n * Space-separated CSS class names to add to popup container\n */\n className?: string;\n /**\n * A string that sets the CSS property of the popup's maximum width, eg `'300px'`.\n * To ensure the popup resizes to fit its content, set this property to `'none'`.\n * Available values can be found here: https://developer.mozilla.org/en-US/docs/Web/CSS/max-width\n * @defaultValue '240px'\n */\n maxWidth?: string;\n};\n\nconst focusQuerySelector = [\n 'a[href]',\n '[tabindex]:not([tabindex=\\'-1\\'])',\n '[contenteditable]:not([contenteditable=\\'false\\'])',\n 'button:not([disabled])',\n 'input:not([disabled])',\n 'select:not([disabled])',\n 'textarea:not([disabled])',\n].join(', ');\n\n/**\n * A popup component.\n *\n * @group Markers and Controls\n *\n *\n * @example\n * Create a popup\n * ```ts\n * let popup = new maplibregl.Popup();\n * // Set an event listener that will fire\n * // any time the popup is opened\n * popup.on('open', function(){\n * console.log('popup was opened');\n * });\n * ```\n *\n * @example\n * Create a popup\n * ```ts\n * let popup = new maplibregl.Popup();\n * // Set an event listener that will fire\n * // any time the popup is closed\n * popup.on('close', function(){\n * console.log('popup was closed');\n * });\n * ```\n *\n * @example\n * ```ts\n * let markerHeight = 50, markerRadius = 10, linearOffset = 25;\n * let popupOffsets = {\n * 'top': [0, 0],\n * 'top-left': [0,0],\n * 'top-right': [0,0],\n * 'bottom': [0, -markerHeight],\n * 'bottom-left': [linearOffset, (markerHeight - markerRadius + linearOffset) * -1],\n * 'bottom-right': [-linearOffset, (markerHeight - markerRadius + linearOffset) * -1],\n * 'left': [markerRadius, (markerHeight - markerRadius) * -1],\n * 'right': [-markerRadius, (markerHeight - markerRadius) * -1]\n * };\n * let popup = new maplibregl.Popup({offset: popupOffsets, className: 'my-class'})\n * .setLngLat(e.lngLat)\n * .setHTML(\"

Hello World!

\")\n * .setMaxWidth(\"300px\")\n * .addTo(map);\n * ```\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/)\n *\n * ### Events\n *\n * @event `open` Fired when the popup is opened manually or programmatically. `popup` object that was opened\n *\n * @event `close` Fired when the popup is closed manually or programmatically. `popup` object that was closed\n */\nexport class Popup extends Evented {\n _map: Map;\n options: PopupOptions;\n _content: HTMLElement;\n _container: HTMLElement;\n _closeButton: HTMLButtonElement;\n _tip: HTMLElement;\n _lngLat: LngLat;\n _trackPointer: boolean;\n _pos: Point;\n\n constructor(options?: PopupOptions) {\n super();\n this.options = extend(Object.create(defaultOptions), options);\n }\n\n /**\n * Adds the popup to a map.\n *\n * @param map - The MapLibre GL JS map to add the popup to.\n * @returns `this`\n * @example\n * ```ts\n * new maplibregl.Popup()\n * .setLngLat([0, 0])\n * .setHTML(\"

Null Island

\")\n * .addTo(map);\n * ```\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n * @see [Show polygon information on click](https://maplibre.org/maplibre-gl-js/docs/examples/polygon-popup-on-click/)\n */\n addTo(map: Map): this {\n if (this._map) this.remove();\n\n this._map = map;\n if (this.options.closeOnClick) {\n this._map.on('click', this._onClose);\n }\n\n if (this.options.closeOnMove) {\n this._map.on('move', this._onClose);\n }\n\n this._map.on('remove', this.remove);\n this._update();\n this._focusFirstElement();\n\n if (this._trackPointer) {\n this._map.on('mousemove', this._onMouseMove);\n this._map.on('mouseup', this._onMouseUp);\n if (this._container) {\n this._container.classList.add('maplibregl-popup-track-pointer');\n }\n this._map._canvasContainer.classList.add('maplibregl-track-pointer');\n } else {\n this._map.on('move', this._update);\n }\n\n this.fire(new Event('open'));\n\n return this;\n }\n\n /**\n * @returns `true` if the popup is open, `false` if it is closed.\n */\n isOpen() {\n return !!this._map;\n }\n\n /**\n * Removes the popup from the map it has been added to.\n *\n * @example\n * ```ts\n * let popup = new maplibregl.Popup().addTo(map);\n * popup.remove();\n * ```\n * @returns `this`\n */\n remove = (): this => {\n if (this._content) {\n DOM.remove(this._content);\n }\n\n if (this._container) {\n DOM.remove(this._container);\n delete this._container;\n }\n\n if (this._map) {\n this._map.off('move', this._update);\n this._map.off('move', this._onClose);\n this._map.off('click', this._onClose);\n this._map.off('remove', this.remove);\n this._map.off('mousemove', this._onMouseMove);\n this._map.off('mouseup', this._onMouseUp);\n this._map.off('drag', this._onDrag);\n delete this._map;\n }\n\n this.fire(new Event('close'));\n\n return this;\n };\n\n /**\n * Returns the geographical location of the popup's anchor.\n *\n * The longitude of the result may differ by a multiple of 360 degrees from the longitude previously\n * set by `setLngLat` because `Popup` wraps the anchor longitude across copies of the world to keep\n * the popup on screen.\n *\n * @returns The geographical location of the popup's anchor.\n */\n getLngLat(): LngLat {\n return this._lngLat;\n }\n\n /**\n * Sets the geographical location of the popup's anchor, and moves the popup to it. Replaces trackPointer() behavior.\n *\n * @param lnglat - The geographical location to set as the popup's anchor.\n * @returns `this`\n */\n setLngLat(lnglat: LngLatLike): this {\n this._lngLat = LngLat.convert(lnglat);\n this._pos = null;\n\n this._trackPointer = false;\n\n this._update();\n\n if (this._map) {\n this._map.on('move', this._update);\n this._map.off('mousemove', this._onMouseMove);\n if (this._container) {\n this._container.classList.remove('maplibregl-popup-track-pointer');\n }\n this._map._canvasContainer.classList.remove('maplibregl-track-pointer');\n }\n\n return this;\n }\n\n /**\n * Tracks the popup anchor to the cursor position on screens with a pointer device (it will be hidden on touchscreens). Replaces the `setLngLat` behavior.\n * For most use cases, set `closeOnClick` and `closeButton` to `false`.\n * @example\n * ```ts\n * let popup = new maplibregl.Popup({ closeOnClick: false, closeButton: false })\n * .setHTML(\"

Hello World!

\")\n * .trackPointer()\n * .addTo(map);\n * ```\n * @returns `this`\n */\n trackPointer(): this {\n this._trackPointer = true;\n this._pos = null;\n this._update();\n if (this._map) {\n this._map.off('move', this._update);\n this._map.on('mousemove', this._onMouseMove);\n this._map.on('drag', this._onDrag);\n if (this._container) {\n this._container.classList.add('maplibregl-popup-track-pointer');\n }\n this._map._canvasContainer.classList.add('maplibregl-track-pointer');\n }\n\n return this;\n\n }\n\n /**\n * Returns the `Popup`'s HTML element.\n * @example\n * Change the `Popup` element's font size\n * ```ts\n * let popup = new maplibregl.Popup()\n * .setLngLat([-96, 37.8])\n * .setHTML(\"

Hello World!

\")\n * .addTo(map);\n * let popupElem = popup.getElement();\n * popupElem.style.fontSize = \"25px\";\n * ```\n * @returns element\n */\n getElement(): HTMLElement {\n return this._container;\n }\n\n /**\n * Sets the popup's content to a string of text.\n *\n * This function creates a [Text](https://developer.mozilla.org/en-US/docs/Web/API/Text) node in the DOM,\n * so it cannot insert raw HTML. Use this method for security against XSS\n * if the popup content is user-provided.\n *\n * @param text - Textual content for the popup.\n * @returns `this`\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * .setLngLat(e.lngLat)\n * .setText('Hello, world!')\n * .addTo(map);\n * ```\n */\n setText(text: string): this {\n return this.setDOMContent(document.createTextNode(text));\n }\n\n /**\n * Sets the popup's content to the HTML provided as a string.\n *\n * This method does not perform HTML filtering or sanitization, and must be\n * used only with trusted content. Consider {@link Popup#setText} if\n * the content is an untrusted text string.\n *\n * @param html - A string representing HTML content for the popup.\n * @returns `this`\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * .setLngLat(e.lngLat)\n * .setHTML(\"

Hello World!

\")\n * .addTo(map);\n * ```\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/)\n */\n setHTML(html: string): this {\n const frag = document.createDocumentFragment();\n const temp = document.createElement('body');\n let child;\n temp.innerHTML = html;\n while (true) {\n child = temp.firstChild;\n if (!child) break;\n frag.appendChild(child);\n }\n\n return this.setDOMContent(frag);\n }\n\n /**\n * Returns the popup's maximum width.\n *\n * @returns The maximum width of the popup.\n */\n getMaxWidth(): string {\n return this._container?.style.maxWidth;\n }\n\n /**\n * Sets the popup's maximum width. This is setting the CSS property `max-width`.\n * Available values can be found here: https://developer.mozilla.org/en-US/docs/Web/CSS/max-width\n *\n * @param maxWidth - A string representing the value for the maximum width.\n * @returns `this`\n */\n setMaxWidth(maxWidth: string): this {\n this.options.maxWidth = maxWidth;\n this._update();\n return this;\n }\n\n /**\n * Sets the popup's content to the element provided as a DOM node.\n *\n * @param htmlNode - A DOM node to be used as content for the popup.\n * @returns `this`\n * @example\n * Create an element with the popup content\n * ```ts\n * let div = document.createElement('div');\n * div.innerHTML = 'Hello, world!';\n * let popup = new maplibregl.Popup()\n * .setLngLat(e.lngLat)\n * .setDOMContent(div)\n * .addTo(map);\n * ```\n */\n setDOMContent(htmlNode: Node): this {\n if (this._content) {\n // Clear out children first.\n while (this._content.hasChildNodes()) {\n if (this._content.firstChild) {\n this._content.removeChild(this._content.firstChild);\n }\n }\n } else {\n this._content = DOM.create('div', 'maplibregl-popup-content', this._container);\n }\n\n // The close button should be the last tabbable element inside the popup for a good keyboard UX.\n this._content.appendChild(htmlNode);\n this._createCloseButton();\n this._update();\n this._focusFirstElement();\n return this;\n }\n\n /**\n * Adds a CSS class to the popup container element.\n *\n * @param className - Non-empty string with CSS class name to add to popup container\n *\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * popup.addClassName('some-class')\n * ```\n */\n addClassName(className: string) {\n if (this._container) {\n this._container.classList.add(className);\n }\n }\n\n /**\n * Removes a CSS class from the popup container element.\n *\n * @param className - Non-empty string with CSS class name to remove from popup container\n *\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * popup.removeClassName('some-class')\n * ```\n */\n removeClassName(className: string) {\n if (this._container) {\n this._container.classList.remove(className);\n }\n }\n\n /**\n * Sets the popup's offset.\n *\n * @param offset - Sets the popup's offset.\n * @returns `this`\n */\n setOffset (offset?: Offset): this {\n this.options.offset = offset;\n this._update();\n return this;\n }\n\n /**\n * Add or remove the given CSS class on the popup container, depending on whether the container currently has that class.\n *\n * @param className - Non-empty string with CSS class name to add/remove\n *\n * @returns if the class was removed return false, if class was added, then return true, undefined if there is no container\n *\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * popup.toggleClassName('toggleClass')\n * ```\n */\n toggleClassName(className: string): boolean | undefined {\n if (this._container) {\n return this._container.classList.toggle(className);\n }\n }\n\n _createCloseButton() {\n if (this.options.closeButton) {\n this._closeButton = DOM.create('button', 'maplibregl-popup-close-button', this._content);\n this._closeButton.type = 'button';\n this._closeButton.setAttribute('aria-label', 'Close popup');\n this._closeButton.innerHTML = '×';\n this._closeButton.addEventListener('click', this._onClose);\n }\n }\n\n _onMouseUp = (event: MapMouseEvent) => {\n this._update(event.point);\n };\n\n _onMouseMove = (event: MapMouseEvent) => {\n this._update(event.point);\n };\n\n _onDrag = (event: MapMouseEvent) => {\n this._update(event.point);\n };\n\n _update = (cursor?: Point) => {\n const hasPosition = this._lngLat || this._trackPointer;\n\n if (!this._map || !hasPosition || !this._content) { return; }\n\n if (!this._container) {\n this._container = DOM.create('div', 'maplibregl-popup', this._map.getContainer());\n this._tip = DOM.create('div', 'maplibregl-popup-tip', this._container);\n this._container.appendChild(this._content);\n if (this.options.className) {\n for (const name of this.options.className.split(' ')) {\n this._container.classList.add(name);\n }\n }\n\n if (this._trackPointer) {\n this._container.classList.add('maplibregl-popup-track-pointer');\n }\n }\n\n if (this.options.maxWidth && this._container.style.maxWidth !== this.options.maxWidth) {\n this._container.style.maxWidth = this.options.maxWidth;\n }\n\n if (this._map.transform.renderWorldCopies && !this._trackPointer) {\n this._lngLat = smartWrap(this._lngLat, this._pos, this._map.transform);\n }\n\n if (this._trackPointer && !cursor) return;\n\n const pos = this._pos = this._trackPointer && cursor ? cursor : this._map.project(this._lngLat);\n\n let anchor = this.options.anchor;\n const offset = normalizeOffset(this.options.offset);\n\n if (!anchor) {\n const width = this._container.offsetWidth;\n const height = this._container.offsetHeight;\n let anchorComponents;\n\n if (pos.y + offset.bottom.y < height) {\n anchorComponents = ['top'];\n } else if (pos.y > this._map.transform.height - height) {\n anchorComponents = ['bottom'];\n } else {\n anchorComponents = [];\n }\n\n if (pos.x < width / 2) {\n anchorComponents.push('left');\n } else if (pos.x > this._map.transform.width - width / 2) {\n anchorComponents.push('right');\n }\n\n if (anchorComponents.length === 0) {\n anchor = 'bottom';\n } else {\n anchor = (anchorComponents.join('-') as any);\n }\n }\n\n const offsetedPos = pos.add(offset[anchor]).round();\n DOM.setTransform(this._container, `${anchorTranslate[anchor]} translate(${offsetedPos.x}px,${offsetedPos.y}px)`);\n applyAnchorClass(this._container, anchor, 'popup');\n };\n\n _focusFirstElement() {\n if (!this.options.focusAfterOpen || !this._container) return;\n\n const firstFocusable = this._container.querySelector(focusQuerySelector) as HTMLElement;\n\n if (firstFocusable) firstFocusable.focus();\n }\n\n _onClose = () => {\n this.remove();\n };\n}\n\nfunction normalizeOffset(offset?: Offset | null) {\n if (!offset) {\n return normalizeOffset(new Point(0, 0));\n\n } else if (typeof offset === 'number') {\n // input specifies a radius from which to calculate offsets at all positions\n const cornerOffset = Math.round(Math.abs(offset) / Math.SQRT2);\n return {\n 'center': new Point(0, 0),\n 'top': new Point(0, offset),\n 'top-left': new Point(cornerOffset, cornerOffset),\n 'top-right': new Point(-cornerOffset, cornerOffset),\n 'bottom': new Point(0, -offset),\n 'bottom-left': new Point(cornerOffset, -cornerOffset),\n 'bottom-right': new Point(-cornerOffset, -cornerOffset),\n 'left': new Point(offset, 0),\n 'right': new Point(-offset, 0)\n };\n\n } else if (offset instanceof Point || Array.isArray(offset)) {\n // input specifies a single offset to be applied to all positions\n const convertedOffset = Point.convert(offset);\n return {\n 'center': convertedOffset,\n 'top': convertedOffset,\n 'top-left': convertedOffset,\n 'top-right': convertedOffset,\n 'bottom': convertedOffset,\n 'bottom-left': convertedOffset,\n 'bottom-right': convertedOffset,\n 'left': convertedOffset,\n 'right': convertedOffset\n };\n\n } else {\n // input specifies an offset per position\n return {\n 'center': Point.convert(offset['center'] || [0, 0]),\n 'top': Point.convert(offset['top'] || [0, 0]),\n 'top-left': Point.convert(offset['top-left'] || [0, 0]),\n 'top-right': Point.convert(offset['top-right'] || [0, 0]),\n 'bottom': Point.convert(offset['bottom'] || [0, 0]),\n 'bottom-left': Point.convert(offset['bottom-left'] || [0, 0]),\n 'bottom-right': Point.convert(offset['bottom-right'] || [0, 0]),\n 'left': Point.convert(offset['left'] || [0, 0]),\n 'right': Point.convert(offset['right'] || [0, 0])\n };\n }\n}\n","import {extend} from './util';\n\n/**\n * This is a private namespace for utility functions that will get automatically stripped\n * out in production builds.\n */\nexport const Debug = {\n extend(dest: any, ...sources: Array): any {\n return extend(dest, ...sources);\n },\n\n run(fn: () => any) {\n fn();\n },\n\n logToElement(message: string, overwrite: boolean = false, id: string = 'log') {\n const el = window.document.getElementById(id);\n if (el) {\n if (overwrite) el.innerHTML = '';\n el.innerHTML += `
${message}`;\n }\n\n }\n};\n","import packageJSON from '../package.json' assert {type: 'json'};\nimport {Map} from './ui/map';\nimport {NavigationControl} from './ui/control/navigation_control';\nimport {GeolocateControl} from './ui/control/geolocate_control';\nimport {AttributionControl} from './ui/control/attribution_control';\nimport {LogoControl} from './ui/control/logo_control';\nimport {ScaleControl} from './ui/control/scale_control';\nimport {FullscreenControl} from './ui/control/fullscreen_control';\nimport {TerrainControl} from './ui/control/terrain_control';\nimport {Popup} from './ui/popup';\nimport {Marker} from './ui/marker';\nimport {Style} from './style/style';\nimport {LngLat} from './geo/lng_lat';\nimport {LngLatBounds} from './geo/lng_lat_bounds';\nimport Point from '@mapbox/point-geometry';\nimport {MercatorCoordinate} from './geo/mercator_coordinate';\nimport {Evented} from './util/evented';\nimport {config} from './util/config';\nimport {Debug} from './util/debug';\nimport {isSafari} from './util/util';\nimport {setRTLTextPlugin, getRTLTextPluginStatus} from './source/rtl_text_plugin';\nimport {WorkerPool} from './util/worker_pool';\nimport {prewarm, clearPrewarmedResources} from './util/global_worker_pool';\nimport {PerformanceUtils} from './util/performance';\nimport {AJAXError} from './util/ajax';\nimport type {RequestParameters, ResponseCallback} from './util/ajax';\nimport type {Cancelable} from './types/cancelable';\nimport {GeoJSONSource} from './source/geojson_source';\nimport {CanvasSource} from './source/canvas_source';\nimport {ImageSource} from './source/image_source';\nimport {RasterDEMTileSource} from './source/raster_dem_tile_source';\nimport {RasterTileSource} from './source/raster_tile_source';\nimport {VectorTileSource} from './source/vector_tile_source';\nimport {VideoSource} from './source/video_source';\n\nconst version = packageJSON.version;\n\nexport type * from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * `maplibregl` is the global object that allows configurations that are not specific to a map instance\n *\n * @group Main\n */\nclass MapLibreGL {\n static Map = Map;\n static NavigationControl = NavigationControl;\n static GeolocateControl = GeolocateControl;\n static AttributionControl = AttributionControl;\n static LogoControl = LogoControl;\n static ScaleControl = ScaleControl;\n static FullscreenControl = FullscreenControl;\n static TerrainControl = TerrainControl;\n static Popup = Popup;\n static Marker = Marker;\n static Style = Style;\n static LngLat = LngLat;\n static LngLatBounds = LngLatBounds;\n static Point = Point;\n static MercatorCoordinate = MercatorCoordinate;\n static Evented = Evented;\n static AJAXError = AJAXError;\n static config = config;\n static CanvasSource = CanvasSource;\n static GeoJSONSource = GeoJSONSource;\n static ImageSource = ImageSource;\n static RasterDEMTileSource = RasterDEMTileSource;\n static RasterTileSource = RasterTileSource;\n static VectorTileSource = VectorTileSource;\n static VideoSource = VideoSource;\n /**\n * Sets the map's [RTL text plugin](https://www.mapbox.com/mapbox-gl-js/plugins/#mapbox-gl-rtl-text).\n * Necessary for supporting the Arabic and Hebrew languages, which are written right-to-left.\n *\n * @param pluginURL - URL pointing to the Mapbox RTL text plugin source.\n * @param callback - Called with an error argument if there is an error.\n * @param lazy - If set to `true`, mapboxgl will defer loading the plugin until rtl text is encountered,\n * rtl text will then be rendered only after the plugin finishes loading.\n * @example\n * ```ts\n * maplibregl.setRTLTextPlugin('https://unpkg.com/@mapbox/mapbox-gl-rtl-text@0.2.3/mapbox-gl-rtl-text.js');\n * ```\n * @see [Add support for right-to-left scripts](https://maplibre.org/maplibre-gl-js/docs/examples/mapbox-gl-rtl-text/)\n */\n static setRTLTextPlugin = setRTLTextPlugin;\n /**\n * Gets the map's [RTL text plugin](https://www.mapbox.com/mapbox-gl-js/plugins/#mapbox-gl-rtl-text) status.\n * The status can be `unavailable` (i.e. not requested or removed), `loading`, `loaded` or `error`.\n * If the status is `loaded` and the plugin is requested again, an error will be thrown.\n *\n * @example\n * ```ts\n * const pluginStatus = maplibregl.getRTLTextPluginStatus();\n * ```\n */\n static getRTLTextPluginStatus = getRTLTextPluginStatus;\n /**\n * Initializes resources like WebWorkers that can be shared across maps to lower load\n * times in some situations. `maplibregl.workerUrl` and `maplibregl.workerCount`, if being\n * used, must be set before `prewarm()` is called to have an effect.\n *\n * By default, the lifecycle of these resources is managed automatically, and they are\n * lazily initialized when a Map is first created. By invoking `prewarm()`, these\n * resources will be created ahead of time, and will not be cleared when the last Map\n * is removed from the page. This allows them to be re-used by new Map instances that\n * are created later. They can be manually cleared by calling\n * `maplibregl.clearPrewarmedResources()`. This is only necessary if your web page remains\n * active but stops using maps altogether.\n *\n * This is primarily useful when using GL-JS maps in a single page app, wherein a user\n * would navigate between various views that can cause Map instances to constantly be\n * created and destroyed.\n *\n * @example\n * ```ts\n * maplibregl.prewarm()\n * ```\n */\n static prewarm = prewarm;\n /**\n * Clears up resources that have previously been created by `maplibregl.prewarm()`.\n * Note that this is typically not necessary. You should only call this function\n * if you expect the user of your app to not return to a Map view at any point\n * in your application.\n *\n * @example\n * ```ts\n * maplibregl.clearPrewarmedResources()\n * ```\n */\n static clearPrewarmedResources = clearPrewarmedResources;\n /**\n * Returns the package version of the library\n * @returns Package version of the library\n */\n static get version(): string {\n return version;\n }\n\n /**\n * Gets and sets the number of web workers instantiated on a page with GL JS maps.\n * By default, workerCount is 1 except for Safari browser where it is set to half the number of CPU cores (capped at 3).\n * Make sure to set this property before creating any map instances for it to have effect.\n *\n * @returns Number of workers currently configured.\n * @example\n * ```ts\n * maplibregl.workerCount = 2;\n * ```\n */\n static get workerCount(): number {\n return WorkerPool.workerCount;\n }\n\n static set workerCount(count: number) {\n WorkerPool.workerCount = count;\n }\n /**\n * Gets and sets the maximum number of images (raster tiles, sprites, icons) to load in parallel,\n * which affects performance in raster-heavy maps. 16 by default.\n *\n * @returns Number of parallel requests currently configured.\n * @example\n * ```ts\n * maplibregl.maxParallelImageRequests = 10;\n * ```\n */\n static get maxParallelImageRequests(): number {\n return config.MAX_PARALLEL_IMAGE_REQUESTS;\n }\n\n static set maxParallelImageRequests(numRequests: number) {\n config.MAX_PARALLEL_IMAGE_REQUESTS = numRequests;\n }\n\n static get workerUrl(): string {\n return config.WORKER_URL;\n }\n\n static set workerUrl(value: string) {\n config.WORKER_URL = value;\n }\n\n /**\n * Sets a custom load tile function that will be called when using a source that starts with a custom url schema.\n * The example below will be triggered for custom:// urls defined in the sources list in the style definitions.\n * The function passed will receive the request parameters and should call the callback with the resulting request,\n * for example a pbf vector tile, non-compressed, represented as ArrayBuffer.\n *\n * @param customProtocol - the protocol to hook, for example 'custom'\n * @param loadFn - the function to use when trying to fetch a tile specified by the customProtocol\n * @example\n * This will fetch a file using the fetch API (this is obviously a non interesting example...)\n * ```ts\n * maplibregl.addProtocol('custom', (params, callback) => {\n fetch(`https://${params.url.split(\"://\")[1]}`)\n .then(t => {\n if (t.status == 200) {\n t.arrayBuffer().then(arr => {\n callback(null, arr, null, null);\n });\n } else {\n callback(new Error(`Tile fetch error: ${t.statusText}`));\n }\n })\n .catch(e => {\n callback(new Error(e));\n });\n return { cancel: () => { } };\n });\n * // the following is an example of a way to return an error when trying to load a tile\n * maplibregl.addProtocol('custom2', (params, callback) => {\n * callback(new Error('someErrorMessage'));\n * return { cancel: () => { } };\n * });\n * ```\n */\n static addProtocol(customProtocol: string, loadFn: (requestParameters: RequestParameters, callback: ResponseCallback) => Cancelable) {\n config.REGISTERED_PROTOCOLS[customProtocol] = loadFn;\n }\n\n /**\n * Removes a previously added protocol\n *\n * @param customProtocol - the custom protocol to remove registration for\n * @example\n * ```ts\n * maplibregl.removeProtocol('custom');\n * ```\n */\n static removeProtocol(customProtocol: string) {\n delete config.REGISTERED_PROTOCOLS[customProtocol];\n }\n}\n\n//This gets automatically stripped out in production builds.\nDebug.extend(MapLibreGL, {isSafari, getPerformanceMetrics: PerformanceUtils.getPerformanceMetrics});\n\nexport default MapLibreGL;\n","//\n// Our custom intro provides a specialized \"define()\" function, called by the\n// AMD modules below, that sets up the worker blob URL and then executes the\n// main module, storing its exported value as 'maplibregl'\n\n// The three \"chunks\" imported here are produced by a first Rollup pass,\n// which outputs them as AMD modules.\n\n// Shared dependencies, i.e.:\n/*\ndefine(['exports'], function (exports) {\n // Code for all common dependencies\n // Each module's exports are attached attached to 'exports' (with\n // names rewritten to avoid collisions, etc.)\n})\n*/\nimport '../../staging/maplibregl/shared';\n\n// Worker and its unique dependencies, i.e.:\n/*\ndefine(['./shared.js'], function (__shared__js) {\n // Code for worker script and its unique dependencies.\n // Expects the output of 'shared' module to be passed in as an argument,\n // since all references to common deps look like, e.g.,\n // __shared__js.shapeText().\n});\n*/\n// When this wrapper function is passed to our custom define() above,\n// it gets stringified, together with the shared wrapper (using\n// Function.toString()), and the resulting string of code is made into a\n// Blob URL that gets used by the main module to create the web workers.\nimport '../../staging/maplibregl/worker';\n\n// Main module and its unique dependencies\n/*\ndefine(['./shared.js'], function (__shared__js) {\n // Code for main GL JS module and its unique dependencies.\n // Expects the output of 'shared' module to be passed in as an argument,\n // since all references to common deps look like, e.g.,\n // __shared__js.shapeText().\n //\n // Returns the actual maplibregl (i.e. src/index.js)\n});\n*/\nimport '../../staging/maplibregl/index';\n\nexport default maplibregl;\n"],"names":["Point","bezier","UnitBezier","clamp","deepEqual","clone","layout","paint","EXTENT","mercatorXfromLng","mercatorYfromLat","validateFilter","validatePaintProperty","validateLayoutProperty","validateSource","validateLight","validateTerrain","isChar","rtlTextPlugin","interpolates","styleSpec","align","members","size","alignment","require$$0","require$$1","murmurhashJsModule","sort","murmur3","swap","layoutAttributes","translate","getLayout","getPaint","equals","create","glMatrix.ARRAY_TYPE","copy","identity","fromValues","set","transpose","invert","adjoint","determinant","multiply","rotate","scale","fromRotation","fromScaling","str","frob","add","subtract","exactEquals","glMatrix.EPSILON","multiplyScalar","multiplyScalarAndAdd","mul","sub","fromTranslation","fromMat4","fromQuat","rotateX","rotateY","rotateZ","fromRotationTranslation","getTranslation","length","divide","ceil","floor","min","max","round","scaleAndAdd","distance","squaredDistance","squaredLength","negate","inverse","normalize","dot","cross","lerp","random","glMatrix.RANDOM","transformMat4","transformMat3","transformQuat","angle","zero","div","dist","sqrDist","len","sqrLen","forEach","conjugate","vec4.clone","vec4.fromValues","vec4.copy","vec4.set","vec4.add","vec4.scale","vec4.dot","vec4.lerp","vec4.length","vec4.squaredLength","vec4.normalize","vec4.exactEquals","vec4.equals","vec3.create","vec3.fromValues","vec3.dot","vec3.cross","vec3.len","vec3.normalize","mat3.create","quat.create","mat4.getRotation","mat4.getTranslation","quat.copy","quat.rotateX","quat.rotateY","quat.rotateZ","quat.dot","quat.length","quat.squaredLength","properties","projectQueryGeometry","vec4.transformMat4","earcutModule","signedArea","defaultCompare","classifyRings","EARCUT_MAX_RINGS","earcut","VectorTileFeature","VectorTileLayer","VectorTile","require$$2","vectorTileFeatureTypes","mvt","addVertex","layoutAttributesExt","ieee754","border","WritingMode","mat4.identity","globalRTLTextPlugin","vt","featureFilter","Queue","TextAnchorEnum","PerformanceMarkers","createStyleLayer","groupByLayout","potpack","AlphaImage","register","OverscaledTileID","CollisionBoxArray","DictionaryCoder","FeatureIndex","warnOnce","mapObject","ImageAtlas","SymbolBucket","performSymbolLayout","LineBucket","FillBucket","FillExtrusionBucket","EvaluationParameters","getArrayBuffer","Protobuf","RequestPerformance","extend","isImageBitmap","RGBAImage","getImageData","DEMData","rewind","FeatureWrapper","GeoJSONWrapper","vtPbfModule","KDBush","transform","getJSON","createExpression","Actor","isWorker","arrayBufferToImageBitmap","arrayBufferToImage","getProtocolAction","makeRequest","config","sameOrigin","browser","Evented","ErrorEvent","Event","ImagePosition","parseGlyphPbf","asyncAll","unicodeBlockLookup","sphericalToCartesian","Properties","DataConstantProperty","Transitionable","emitValidationErrors","validateStyle","pick","LngLat","__awaiter","offscreenCanvasSupported","isOffscreenCanvasDistorted","readImageUsingVideoFrame","createLayout","SegmentVector","MercatorCoordinate","RasterBoundsArray","CanonicalTileID","getVideo","ValidationError","mat4.create","mat4.translate","mat4.scale","mat4.multiply","uniqueId","deserializeBucket","lazyLoadRTLTextPlugin","toEvaluationFeature","GeoJSONFeature","parseCacheControl","createSource","keysDifference","isSafari","mat4.rotateZ","mat4.clone","symbolSize.evaluateSizeForZoom","symbolSize.evaluateSizeForFeature","addDynamicAttributes","findLineIntersection","projection.project","projection.getPerspectiveRatio","ONE_EM","projection.placeFirstAndLastGlyph","clipLine","intersectionTests.polygonIntersectsPolygon","projection.xyTransformMat4","getAnchorAlignment","projection.getLabelPlaneMatrix","projection.getGlCoordMatrix","getOverlapMode","mat4.invert","getAnchorJustification","_emitValidationErrors","diffOperations","emptyStyle","ZoomHistory","getReferrer","triggerPluginCompletionEvent","deref","diffStyles","validateCustomStyleLayer","filterObject","rtlTextPluginEvented","registerForPluginStateChange","Uniform1i","Uniform1f","UniformMatrix4f","Uniform4f","Uniform3f","Uniform2f","mat3.fromRotation","vec3.transformMat3","UniformColor","mat4.ortho","Color","ALWAYS","mat4.mul","CollisionCircleLayoutArray","collisionCircleLayout","QuadTriangleArray","evaluateSizeForZoom","symbolProjection.getLabelPlaneMatrix","symbolProjection.hideGlyphs","symbolProjection.project","symbolProjection.getPerspectiveRatio","evaluateSizeForFeature","symbolProjection.getGlCoordMatrix","symbolProjection.updateLineLabels","nextPowerOfTwo","renderColorRamp","PosArray","LineStripIndexArray","TriangleIndexArray","mat4.equals","mat4.copy","vec4.mul","vec3.sub","vec3.scale","vec3.add","vec3.clone","wrap","mat2.create","mat2.rotate","UnwrappedTileID","vec2.sqrLen","mercatorZfromAltitude","mat4.perspective","mat4.rotateX","LEFT_BUTTON","RIGHT_BUTTON","assignEvents","defaultOptions","defaultEasing","vec2.transformMat4","mat4.fromScaling","Pos3dArray","earthRadius","lngFromMercatorX","version","PerformanceUtils","Map","AJAXError","setRTLTextPlugin","getRTLTextPluginStatus"],"mappings":"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C,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACtD,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACtD,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACtD,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACtD,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACtD,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,IAAI,GAAG,CAAC;AACtD,EAAE,OAAO,GAAG,CAAC;AACb,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS,OAAO,CAAC,GAAG,EAAE,CAAC,EAAE;AAChC,EAAE,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACjB,EAAE,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACjB,EAAE,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC;AACjB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AAClB,EAAE,IAAI,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC;AACjB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC;AACjB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC;AACjB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AAClB,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC;AACzG,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC;AAC5G,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC;AACzG,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC;AAC5G,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC;AAC5G,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC;AACzG,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC;AAC5G,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC;AACzG,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC;AACzG,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC;AAC5G,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC;AAC1G,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC;AAC7G,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC;AAC7G,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC;AAC1G,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC,CAAC;AAC7G,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC,CAAC;AAC1G,EAAE,OAAO,GAAG,CAAC;AACb,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS,WAAW,CAAC,CAAC,EAAE;AAC/B,EAAE,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACjB,EAAE,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACjB,EAAE,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC;AACjB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AAClB,EAAE,IAAI,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC;AACjB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC;AACjB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC;AACjB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AAClB,EAAE,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAClC,EAAE,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAClC,EAAE,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAClC,EAAE,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAClC,EAAE,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAClC,EAAE,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAClC,EAAE,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAClC,EAAE,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAClC,EAAE,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAClC,EAAE,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAClC,EAAE,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAClC,EAAE,IAAI,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAClC;AACA,EAAE,OAAO,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAC/E,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAASG,UAAQ,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,EAAE;AACpC,EAAE,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACjB,EAAE,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACjB,EAAE,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC;AACjB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AAClB,EAAE,IAAI,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC;AACjB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC;AACjB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC;AACjB,MAAM,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AAClB;AACA,EAAE,IAAI,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC;AACf,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC;AACf,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC;AACf,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AAChB,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACrD,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACZ,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACZ,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACZ,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACZ,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACrD,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACZ,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACZ,EAAE,EAAE,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AACb,EAAE,EAAE,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AACb,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACrD,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACtD,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACtD,EAAE,EAAE,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AACb,EAAE,EAAE,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AACb,EAAE,EAAE,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AACb,EAAE,EAAE,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AACb,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACtD,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACtD,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACtD,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC;AACtD,EAAE,OAAO,GAAG,CAAC;AACb,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAASd,WAAS,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,EAAE;AACrC,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;AACd,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;AACd,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACf,EAAE,IAAI,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC;AACzB,EAAE,IAAI,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC;AACzB,EAAE,IAAI,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC;AACzB;AACA,EAAE,IAAI,CAAC,KAAK,GAAG,EAAE;AACjB,IAAI,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AACrD,IAAI,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AACrD,IAAI,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AACtD,IAAI,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AACtD,GAAG,MAAM;AACT,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACf,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACf,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACf,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACf,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACf,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACf,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACf,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACf,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACf,IAAI,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACf,IAAI,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AAChB,IAAI,GAAG,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;AAChB,IAAI,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC;AACjB,IAAI,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC;AACjB,IAAI,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC;AACjB,IAAI,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC;AACjB,IAAI,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC;AACjB,IAAI,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC;AACjB,IAAI,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC;AACjB,IAAI,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC;AACjB,IAAI,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC;AACjB,IAAI,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC;AACjB,IAAI,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC;AAClB,IAAI,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC;AAClB,IAAI,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,C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AAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;AAC1B,EAAE,GAAG,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;AACtB,EAAE,GAAG,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;AAC1B,EAAE,GAAG,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;AAC1B,EAAE,GAAG,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;AAC1B,EAAE,GAAG,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;AACtB;AACA,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAC7C,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAC7C,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAC7C,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAC7C,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAC7C,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAC7C,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAC7C,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AAC7C,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GA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,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC;AAC/B,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC;AAC/B,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC;AAC/B,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC;AAC/B,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC;AAC/B,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC;AAC/B,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC;AAC/B,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC;AAC/B,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC;AAC/B,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC;AAC/B,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC;AAClC,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC;AAClC,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,GAAG,CA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B,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,EAAE,GAAG,EAAE,CAAC,IAAIA,OAAgB,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,EAAE,GAAG,EAAE,CAAC,IAAIA,OAAgB,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,EAAE,GAAG,EAAE,CAAC,IAAIA,OAAgB,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,EAAE,GAAG,EAAE,CAAC,IAAIA,OAAgB,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,CAAC,IAAIA,OAAgB,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,CAAC,IAAIA,OAAgB,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,CAAC,IAAIA,OAAgB,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC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CA;AACA;AACA;AACO,IAAI/D,MAAI,GAAGgE,MAAS,CAAC;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,IAAI7D,KAAG,GAAG8D,KAAQ,CAAC;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,IAAIlD,KAAG,GAAGmD,KAAQ,CAAC;AAC1B;AACA;AACA;AACA;AACA;AACO,IAAI7C,KAAG,GAAGb,UAAQ,CAAC;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,IAAIE,OAAK,GAAGyD,OAAU,CAAC;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,IAAItB,KAAG,GAAGuB,KAAQ,CAAC;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,IAAIrB,MAAI,GAAGsB,MAAS,CAAC;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACO,IAAItC,QAAM,GAAGuC,QAAW,CAAC;AAChC;AACA;AACA;AACA;AACA;AACO,IAAIZ,KAAG,GAAG3B,QAAM,CAAC;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,IAAIU,eAAa,GAAG8B,eAAkB,CAAC;AAC9C;AACA;AACA;AACA;AACA;AACO,IAAIZ,QAAM,GAAGlB,eAAa,CAAC;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,IAAIG,WAAS,GAAG4B,WAAc,CAAC;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,IAAIvD,aAAW,GAAGwD,aAAgB,CAAC;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,IAAI5E,QAAM,GAAG6E,QAAW,CAAC;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,IAAI,UAAU,GAAG,YAAY;AACpC,EAAE,IAAI,OAAO,GAAGC,QAAW,EAAE,CAAC;AAC9B,EAAE,IAAI,SAAS,GAAGC,YAAe,CAAC,CA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EAAE,GAAG,CAAC,CAAC,CAAC,CAAC;AACf,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC;AACf,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AAChB,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC;AACjD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC;AACjD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC;AACjD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC;AACjD,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACZ,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACZ,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACZ,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACZ,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC;AACjD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC;AACjD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC;AACjD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GA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AAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;AAChB,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACjB,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AACzD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AACzD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AACzD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AACzD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AACzG,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AACzG,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;AACzG,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,GAAG,G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G;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAAG;AACR,IAAA,GAAG,EAAE,GAA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","x_google_ignoreList":[0,1,2,10,27,28,29,40,41,42,43,44,45,46,47,48,49,59,60,67,68,69,70,84,85,87,107,119,123,130,132,133,134,135,136,137,138,139,140,141,142,156]} \ No newline at end of file diff --git a/web/libraries/maplibre-gl/dist/maplibre-gl.css b/web/libraries/maplibre-gl/dist/maplibre-gl.css new file mode 100644 index 00000000..317ea49a --- /dev/null +++ b/web/libraries/maplibre-gl/dist/maplibre-gl.css @@ -0,0 +1 @@ +.maplibregl-map{-webkit-tap-highlight-color:rgb(0 0 0/0);font:12px/20px Helvetica Neue,Arial,Helvetica,sans-serif;overflow:hidden;position:relative}.maplibregl-canvas{left:0;position:absolute;top:0}.maplibregl-map:fullscreen{height:100%;width:100%}.maplibregl-ctrl-group button.maplibregl-ctrl-compass{touch-action:none}.maplibregl-canvas-container.maplibregl-interactive,.maplibregl-ctrl-group button.maplibregl-ctrl-compass{cursor:grab;-moz-user-select:none;-webkit-user-select:none;-ms-user-select:none;user-select:none}.maplibregl-canvas-container.maplibregl-interactive.maplibregl-track-pointer{cursor:pointer}.maplibregl-canvas-container.maplibregl-interactive:active,.maplibregl-ctrl-group button.maplibregl-ctrl-compass:active{cursor:grabbing}.maplibregl-canvas-container.maplibregl-touch-zoom-rotate,.maplibregl-canvas-container.maplibregl-touch-zoom-rotate .maplibregl-canvas{touch-action:pan-x pan-y}.maplibregl-canvas-container.maplibregl-touch-drag-pan,.maplibregl-canvas-container.maplibregl-touch-drag-pan .maplibregl-canvas{touch-action:pinch-zoom}.maplibregl-canvas-container.maplibregl-touch-zoom-rotate.maplibregl-touch-drag-pan,.maplibregl-canvas-container.maplibregl-touch-zoom-rotate.maplibregl-touch-drag-pan .maplibregl-canvas{touch-action:none}.maplibregl-canvas-container.maplibregl-touch-drag-pan.maplibregl-cooperative-gestures,.maplibregl-canvas-container.maplibregl-touch-drag-pan.maplibregl-cooperative-gestures .maplibregl-canvas{touch-action:pan-x pan-y}.maplibregl-ctrl-bottom-left,.maplibregl-ctrl-bottom-right,.maplibregl-ctrl-top-left,.maplibregl-ctrl-top-right{pointer-events:none;position:absolute;z-index:2}.maplibregl-ctrl-top-left{left:0;top:0}.maplibregl-ctrl-top-right{right:0;top:0}.maplibregl-ctrl-bottom-left{bottom:0;left:0}.maplibregl-ctrl-bottom-right{bottom:0;right:0}.maplibregl-ctrl{clear:both;pointer-events:auto;transform:translate(0)}.maplibregl-ctrl-top-left .maplibregl-ctrl{float:left;margin:10px 0 0 10px}.maplibregl-ctrl-top-right .maplibregl-ctrl{float:right;margin:10px 10px 0 0}.maplibregl-ctrl-bottom-left .maplibregl-ctrl{float:left;margin:0 0 10px 10px}.maplibregl-ctrl-bottom-right .maplibregl-ctrl{float:right;margin:0 10px 10px 0}.maplibregl-ctrl-group{background:#fff;border-radius:4px}.maplibregl-ctrl-group:not(:empty){box-shadow:0 0 0 2px rgba(0,0,0,.1)}@media (-ms-high-contrast:active){.maplibregl-ctrl-group:not(:empty){box-shadow:0 0 0 2px ButtonText}}.maplibregl-ctrl-group button{background-color:transparent;border:0;box-sizing:border-box;cursor:pointer;display:block;height:29px;outline:none;padding:0;width:29px}.maplibregl-ctrl-group button+button{border-top:1px solid #ddd}.maplibregl-ctrl button .maplibregl-ctrl-icon{background-position:50%;background-repeat:no-repeat;display:block;height:100%;width:100%}@media (-ms-high-contrast:active){.maplibregl-ctrl-icon{background-color:transparent}.maplibregl-ctrl-group button+button{border-top:1px solid ButtonText}}.maplibregl-ctrl button::-moz-focus-inner{border:0;padding:0}.maplibregl-ctrl-attrib-button:focus,.maplibregl-ctrl-group button:focus{box-shadow:0 0 2px 2px #0096ff}.maplibregl-ctrl button:disabled{cursor:not-allowed}.maplibregl-ctrl button:disabled .maplibregl-ctrl-icon{opacity:.25}.maplibregl-ctrl button:not(:disabled):hover{background-color:rgb(0 0 0/5%)}.maplibregl-ctrl-group button:focus:focus-visible{box-shadow:0 0 2px 2px #0096ff}.maplibregl-ctrl-group button:focus:not(:focus-visible){box-shadow:none}.maplibregl-ctrl-group button:focus:first-child{border-radius:4px 4px 0 0}.maplibregl-ctrl-group button:focus:last-child{border-radius:0 0 4px 4px}.maplibregl-ctrl-group button:focus:only-child{border-radius:inherit}.maplibregl-ctrl button.maplibregl-ctrl-zoom-out .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23333' viewBox='0 0 29 29'%3E%3Cpath d='M10 13c-.75 0-1.5.75-1.5 1.5S9.25 16 10 16h9c.75 0 1.5-.75 1.5-1.5S19.75 13 19 13h-9z'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-zoom-in .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23333' viewBox='0 0 29 29'%3E%3Cpath d='M14.5 8.5c-.75 0-1.5.75-1.5 1.5v3h-3c-.75 0-1.5.75-1.5 1.5S9.25 16 10 16h3v3c0 .75.75 1.5 1.5 1.5S16 19.75 16 19v-3h3c.75 0 1.5-.75 1.5-1.5S19.75 13 19 13h-3v-3c0-.75-.75-1.5-1.5-1.5z'/%3E%3C/svg%3E")}@media (-ms-high-contrast:active){.maplibregl-ctrl button.maplibregl-ctrl-zoom-out .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23fff' viewBox='0 0 29 29'%3E%3Cpath d='M10 13c-.75 0-1.5.75-1.5 1.5S9.25 16 10 16h9c.75 0 1.5-.75 1.5-1.5S19.75 13 19 13h-9z'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-zoom-in .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23fff' viewBox='0 0 29 29'%3E%3Cpath d='M14.5 8.5c-.75 0-1.5.75-1.5 1.5v3h-3c-.75 0-1.5.75-1.5 1.5S9.25 16 10 16h3v3c0 .75.75 1.5 1.5 1.5S16 19.75 16 19v-3h3c.75 0 1.5-.75 1.5-1.5S19.75 13 19 13h-3v-3c0-.75-.75-1.5-1.5-1.5z'/%3E%3C/svg%3E")}}@media (-ms-high-contrast:black-on-white){.maplibregl-ctrl button.maplibregl-ctrl-zoom-out .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' viewBox='0 0 29 29'%3E%3Cpath d='M10 13c-.75 0-1.5.75-1.5 1.5S9.25 16 10 16h9c.75 0 1.5-.75 1.5-1.5S19.75 13 19 13h-9z'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-zoom-in .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' viewBox='0 0 29 29'%3E%3Cpath d='M14.5 8.5c-.75 0-1.5.75-1.5 1.5v3h-3c-.75 0-1.5.75-1.5 1.5S9.25 16 10 16h3v3c0 .75.75 1.5 1.5 1.5S16 19.75 16 19v-3h3c.75 0 1.5-.75 1.5-1.5S19.75 13 19 13h-3v-3c0-.75-.75-1.5-1.5-1.5z'/%3E%3C/svg%3E")}}.maplibregl-ctrl button.maplibregl-ctrl-fullscreen .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23333' viewBox='0 0 29 29'%3E%3Cpath d='M24 16v5.5c0 1.75-.75 2.5-2.5 2.5H16v-1l3-1.5-4-5.5 1-1 5.5 4 1.5-3h1zM6 16l1.5 3 5.5-4 1 1-4 5.5 3 1.5v1H7.5C5.75 24 5 23.25 5 21.5V16h1zm7-11v1l-3 1.5 4 5.5-1 1-5.5-4L6 13H5V7.5C5 5.75 5.75 5 7.5 5H13zm11 2.5c0-1.75-.75-2.5-2.5-2.5H16v1l3 1.5-4 5.5 1 1 5.5-4 1.5 3h1V7.5z'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-shrink .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' viewBox='0 0 29 29'%3E%3Cpath d='M18.5 16c-1.75 0-2.5.75-2.5 2.5V24h1l1.5-3 5.5 4 1-1-4-5.5 3-1.5v-1h-5.5zM13 18.5c0-1.75-.75-2.5-2.5-2.5H5v1l3 1.5L4 24l1 1 5.5-4 1.5 3h1v-5.5zm3-8c0 1.75.75 2.5 2.5 2.5H24v-1l-3-1.5L25 5l-1-1-5.5 4L17 5h-1v5.5zM10.5 13c1.75 0 2.5-.75 2.5-2.5V5h-1l-1.5 3L5 4 4 5l4 5.5L5 12v1h5.5z'/%3E%3C/svg%3E")}@media (-ms-high-contrast:active){.maplibregl-ctrl button.maplibregl-ctrl-fullscreen .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23fff' viewBox='0 0 29 29'%3E%3Cpath d='M24 16v5.5c0 1.75-.75 2.5-2.5 2.5H16v-1l3-1.5-4-5.5 1-1 5.5 4 1.5-3h1zM6 16l1.5 3 5.5-4 1 1-4 5.5 3 1.5v1H7.5C5.75 24 5 23.25 5 21.5V16h1zm7-11v1l-3 1.5 4 5.5-1 1-5.5-4L6 13H5V7.5C5 5.75 5.75 5 7.5 5H13zm11 2.5c0-1.75-.75-2.5-2.5-2.5H16v1l3 1.5-4 5.5 1 1 5.5-4 1.5 3h1V7.5z'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-shrink .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23fff' viewBox='0 0 29 29'%3E%3Cpath d='M18.5 16c-1.75 0-2.5.75-2.5 2.5V24h1l1.5-3 5.5 4 1-1-4-5.5 3-1.5v-1h-5.5zM13 18.5c0-1.75-.75-2.5-2.5-2.5H5v1l3 1.5L4 24l1 1 5.5-4 1.5 3h1v-5.5zm3-8c0 1.75.75 2.5 2.5 2.5H24v-1l-3-1.5L25 5l-1-1-5.5 4L17 5h-1v5.5zM10.5 13c1.75 0 2.5-.75 2.5-2.5V5h-1l-1.5 3L5 4 4 5l4 5.5L5 12v1h5.5z'/%3E%3C/svg%3E")}}@media (-ms-high-contrast:black-on-white){.maplibregl-ctrl button.maplibregl-ctrl-fullscreen .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' viewBox='0 0 29 29'%3E%3Cpath d='M24 16v5.5c0 1.75-.75 2.5-2.5 2.5H16v-1l3-1.5-4-5.5 1-1 5.5 4 1.5-3h1zM6 16l1.5 3 5.5-4 1 1-4 5.5 3 1.5v1H7.5C5.75 24 5 23.25 5 21.5V16h1zm7-11v1l-3 1.5 4 5.5-1 1-5.5-4L6 13H5V7.5C5 5.75 5.75 5 7.5 5H13zm11 2.5c0-1.75-.75-2.5-2.5-2.5H16v1l3 1.5-4 5.5 1 1 5.5-4 1.5 3h1V7.5z'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-shrink .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' viewBox='0 0 29 29'%3E%3Cpath d='M18.5 16c-1.75 0-2.5.75-2.5 2.5V24h1l1.5-3 5.5 4 1-1-4-5.5 3-1.5v-1h-5.5zM13 18.5c0-1.75-.75-2.5-2.5-2.5H5v1l3 1.5L4 24l1 1 5.5-4 1.5 3h1v-5.5zm3-8c0 1.75.75 2.5 2.5 2.5H24v-1l-3-1.5L25 5l-1-1-5.5 4L17 5h-1v5.5zM10.5 13c1.75 0 2.5-.75 2.5-2.5V5h-1l-1.5 3L5 4 4 5l4 5.5L5 12v1h5.5z'/%3E%3C/svg%3E")}}.maplibregl-ctrl button.maplibregl-ctrl-compass .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23333' viewBox='0 0 29 29'%3E%3Cpath d='m10.5 14 4-8 4 8h-8z'/%3E%3Cpath fill='%23ccc' d='m10.5 16 4 8 4-8h-8z'/%3E%3C/svg%3E")}@media (-ms-high-contrast:active){.maplibregl-ctrl button.maplibregl-ctrl-compass .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23fff' viewBox='0 0 29 29'%3E%3Cpath d='m10.5 14 4-8 4 8h-8z'/%3E%3Cpath fill='%23999' d='m10.5 16 4 8 4-8h-8z'/%3E%3C/svg%3E")}}@media (-ms-high-contrast:black-on-white){.maplibregl-ctrl button.maplibregl-ctrl-compass .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' viewBox='0 0 29 29'%3E%3Cpath d='m10.5 14 4-8 4 8h-8z'/%3E%3Cpath fill='%23ccc' d='m10.5 16 4 8 4-8h-8z'/%3E%3C/svg%3E")}}.maplibregl-ctrl button.maplibregl-ctrl-terrain .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='22' height='22' fill='%23333' viewBox='0 0 22 22'%3E%3Cpath d='m1.754 13.406 4.453-4.851 3.09 3.09 3.281 3.277.969-.969-3.309-3.312 3.844-4.121 6.148 6.886h1.082v-.855l-7.207-8.07-4.84 5.187L6.169 6.57l-5.48 5.965v.871ZM.688 16.844h20.625v1.375H.688Zm0 0'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-terrain-enabled .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='22' height='22' fill='%2333b5e5' viewBox='0 0 22 22'%3E%3Cpath d='m1.754 13.406 4.453-4.851 3.09 3.09 3.281 3.277.969-.969-3.309-3.312 3.844-4.121 6.148 6.886h1.082v-.855l-7.207-8.07-4.84 5.187L6.169 6.57l-5.48 5.965v.871ZM.688 16.844h20.625v1.375H.688Zm0 0'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23333' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3Ccircle cx='10' cy='10' r='2'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate:disabled .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23aaa' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3Ccircle cx='10' cy='10' r='2'/%3E%3Cpath fill='red' d='m14 5 1 1-9 9-1-1 9-9z'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-active .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%2333b5e5' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3Ccircle cx='10' cy='10' r='2'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-active-error .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23e58978' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3Ccircle cx='10' cy='10' r='2'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-background .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%2333b5e5' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-background-error .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23e54e33' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-waiting .maplibregl-ctrl-icon{animation:maplibregl-spin 2s linear infinite}@media (-ms-high-contrast:active){.maplibregl-ctrl button.maplibregl-ctrl-geolocate .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23fff' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3Ccircle cx='10' cy='10' r='2'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate:disabled .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23999' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3Ccircle cx='10' cy='10' r='2'/%3E%3Cpath fill='red' d='m14 5 1 1-9 9-1-1 9-9z'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-active .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%2333b5e5' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3Ccircle cx='10' cy='10' r='2'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-active-error .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23e58978' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3Ccircle cx='10' cy='10' r='2'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-background .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%2333b5e5' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-background-error .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23e54e33' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3C/svg%3E")}}@media (-ms-high-contrast:black-on-white){.maplibregl-ctrl button.maplibregl-ctrl-geolocate .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3Ccircle cx='10' cy='10' r='2'/%3E%3C/svg%3E")}.maplibregl-ctrl button.maplibregl-ctrl-geolocate:disabled .maplibregl-ctrl-icon{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='29' height='29' fill='%23666' viewBox='0 0 20 20'%3E%3Cpath d='M10 4C9 4 9 5 9 5v.1A5 5 0 0 0 5.1 9H5s-1 0-1 1 1 1 1 1h.1A5 5 0 0 0 9 14.9v.1s0 1 1 1 1-1 1-1v-.1a5 5 0 0 0 3.9-3.9h.1s1 0 1-1-1-1-1-1h-.1A5 5 0 0 0 11 5.1V5s0-1-1-1zm0 2.5a3.5 3.5 0 1 1 0 7 3.5 3.5 0 1 1 0-7z'/%3E%3Ccircle cx='10' cy='10' r='2'/%3E%3Cpath fill='red' d='m14 5 1 1-9 9-1-1 9-9z'/%3E%3C/svg%3E")}}@keyframes maplibregl-spin{0%{transform:rotate(0deg)}to{transform:rotate(1turn)}}a.maplibregl-ctrl-logo{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='88' height='23' fill='none'%3E%3Cpath fill='%23000' fill-opacity='.4' fill-rule='evenodd' d='M17.408 16.796h-1.827l2.501-12.095h.198l3.324 6.533.988 2.19.988-2.19 3.258-6.533h.181l2.6 12.095h-1.81l-1.218-5.644-.362-1.71-.658 1.71-2.929 5.644h-.098l-2.914-5.644-.757-1.71-.345 1.71zm1.958-3.42-.726 3.663a1.255 1.255 0 0 1-1.232 1.011h-1.827a1.255 1.255 0 0 1-1.229-1.509l2.501-12.095a1.255 1.255 0 0 1 1.23-1.001h.197a1.255 1.255 0 0 1 1.12.685l3.19 6.273 3.125-6.263a1.255 1.255 0 0 1 1.123-.695h.181a1.255 1.255 0 0 1 1.227.991l1.443 6.71a5.11 5.11 0 0 1 .314-.787l.009-.016a4.623 4.623 0 0 1 1.777-1.887c.782-.46 1.668-.667 2.611-.667a4.548 4.548 0 0 1 1.7.32l.306.134c.21-.16.474-.256.759-.256h1.694a1.255 1.255 0 0 1 1.212.925 1.255 1.255 0 0 1 1.212-.925h1.711c.284 0 .545.094.755.252.613-.3 1.312-.45 2.075-.45 1.356 0 2.557.445 3.482 1.4.314.319.566.676.763 1.064V4.701a1.255 1.255 0 0 1 1.255-1.255h1.86A1.255 1.255 0 0 1 54.44 4.7v9.194h2.217c.19 0 .37.043.532.118v-4.77c0-.356.147-.678.385-.906a2.416 2.416 0 0 1-.682-1.71c0-.665.267-1.253.735-1.7a2.448 2.448 0 0 1 1.722-.674 2.43 2.43 0 0 1 1.705.675c.211.2.381.43.504.683V4.7a1.255 1.255 0 0 1 1.255-1.255h1.744A1.255 1.255 0 0 1 65.812 4.7v3.335a4.76 4.76 0 0 1 1.526-.246c.938 0 1.817.214 2.59.69a4.47 4.47 0 0 1 1.67 1.743v-.98a1.255 1.255 0 0 1 1.256-1.256h1.777c.233 0 .451.064.639.174a3.407 3.407 0 0 1 1.567-.372c.346 0 .861.02 1.285.232a1.255 1.255 0 0 1 .689 1.004 4.73 4.73 0 0 1 .853-.588c.795-.44 1.675-.647 2.61-.647 1.385 0 2.65.39 3.525 1.396.836.938 1.168 2.173 1.168 3.528 0 .343-.02.694-.056 1.051a1.255 1.255 0 0 1-.947 1.09l.408.952a1.255 1.255 0 0 1-.477 1.552c-.418.268-.92.463-1.458.612-.613.171-1.304.244-2.049.244-1.06 0-2.043-.207-2.886-.698l-.015-.008c-.798-.48-1.419-1.135-1.818-1.963l-.004-.008a5.815 5.815 0 0 1-.548-2.512c0-.286.017-.567.053-.843a1.255 1.255 0 0 1-.333-.086l-.166-.004c-.223 0-.426.062-.643.228-.03.024-.142.139-.142.59v3.883a1.255 1.255 0 0 1-1.256 1.256h-1.777a1.255 1.255 0 0 1-1.256-1.256V15.69l-.032.057a4.778 4.778 0 0 1-1.86 1.833 5.04 5.04 0 0 1-2.484.634 4.47 4.47 0 0 1-1.935-.424 1.252 1.252 0 0 1-.764.258h-1.71a1.255 1.255 0 0 1-1.256-1.255V7.687a2.402 2.402 0 0 1-.428.625c.253.23.412.561.412.93v7.553a1.255 1.255 0 0 1-1.256 1.255h-1.843a1.25 1.25 0 0 1-.894-.373c-.228.23-.544.373-.894.373H51.32a1.255 1.255 0 0 1-1.256-1.255v-1.251l-.061.117a4.703 4.703 0 0 1-1.782 1.884 4.767 4.767 0 0 1-2.485.67 5.6 5.6 0 0 1-1.485-.188l.009 2.764a1.255 1.255 0 0 1-1.255 1.259h-1.729a1.255 1.255 0 0 1-1.255-1.255v-3.537a1.255 1.255 0 0 1-1.167.793h-1.679a1.25 1.25 0 0 1-.77-.263 4.47 4.47 0 0 1-1.945.429c-.885 0-1.724-.21-2.495-.632l-.017-.01a4.983 4.983 0 0 1-1.081-.836 1.255 1.255 0 0 1-1.254 1.312h-1.81a1.255 1.255 0 0 1-1.228-.99l-.782-3.625-2.044 3.939a1.255 1.255 0 0 1-1.115.676h-.098a1.255 1.255 0 0 1-1.116-.68l-2.061-3.994zM35.92 16.63l.207-.114.223-.15c.329-.237.574-.499.735-.785l.061-.118.033 1.332h1.678V9.242h-1.694l-.033 1.267c-.088-.22-.264-.438-.526-.658l-.032-.028a3.16 3.16 0 0 0-.668-.428l-.27-.12a3.293 3.293 0 0 0-1.235-.23c-.757 0-1.415.163-1.974.493a3.36 3.36 0 0 0-1.3 1.382c-.297.593-.444 1.284-.444 2.074 0 .8.17 1.503.51 2.107a3.795 3.795 0 0 0 1.382 1.381 3.883 3.883 0 0 0 1.893.477c.53 0 1.015-.11 1.455-.33zm-2.789-5.38c-.384.45-.575 1.038-.575 1.762 0 .735.186 1.332.559 1.794.384.45.933.675 1.645.675a2.25 2.25 0 0 0 .934-.19 2.17 2.17 0 0 0 .468-.29l.178-.161a2.163 2.163 0 0 0 .397-.561c.163-.333.244-.717.244-1.15v-.115c0-.472-.098-.894-.296-1.267l-.043-.077a2.211 2.211 0 0 0-.633-.709l-.13-.086-.047-.028a2.099 2.099 0 0 0-1.073-.285c-.702 0-1.244.231-1.629.692zm2.316 2.706c.163-.17.28-.407.28-.83v-.114c0-.292-.06-.508-.15-.68a.958.958 0 0 0-.353-.389.851.851 0 0 0-.464-.127c-.4 0-.56.114-.664.239l-.01.012c-.148.174-.275.45-.275.945 0 .506.122.801.27.99.097.11.266.224.68.224.303 0 .504-.09.687-.269zm7.545 1.705a2.626 2.626 0 0 0 .331.423c.213.22.464.402.755.548l.173.074c.433.17.93.255 1.49.255.68 0 1.295-.165 1.844-.493a3.447 3.447 0 0 0 1.316-1.4c.329-.603.493-1.299.493-2.089 0-1.273-.33-2.243-.988-2.913-.658-.68-1.52-1.02-2.584-1.02-.598 0-1.124.115-1.575.347a2.807 2.807 0 0 0-.415.262l-.199.166a3.35 3.35 0 0 0-.64.82V9.242h-1.712v11.553h1.729l-.017-5.134zm.53-1.138c.137.193.297.36.48.5l.155.11.053.034c.34.197.713.297 1.119.297.714 0 1.262-.225 1.645-.675.385-.46.576-1.048.576-1.762 0-.746-.192-1.338-.576-1.777-.372-.45-.92-.675-1.645-.675-.29 0-.569.053-.835.16a2.366 2.366 0 0 0-.284.136 1.99 1.99 0 0 0-.363.254 2.237 2.237 0 0 0-.46.569l-.082.162a2.56 2.56 0 0 0-.213 1.072v.115c0 .471.098.894.296 1.267l.135.211zm.964-.818a1.11 1.11 0 0 0 .367.385.937.937 0 0 0 .476.118c.423 0 .59-.117.687-.23.159-.194.28-.478.28-.95 0-.53-.133-.8-.266-.952l-.021-.025c-.078-.094-.231-.221-.68-.221a.995.995 0 0 0-.503.135l-.012.007a.859.859 0 0 0-.335.343c-.073.133-.132.324-.132.614v.115a1.43 1.43 0 0 0 .14.66zm15.7-6.222c.232-.23.346-.516.346-.856a1.053 1.053 0 0 0-.345-.79 1.175 1.175 0 0 0-.84-.329c-.34 0-.625.11-.855.33a1.053 1.053 0 0 0-.346.79c0 .34.115.625.346.855.23.23.516.346.856.346.34 0 .62-.115.839-.346zm4.337 9.314.033-1.332c.128.269.324.518.59.747l.098.081a3.727 3.727 0 0 0 .316.224l.223.122a3.21 3.21 0 0 0 1.44.322 3.785 3.785 0 0 0 1.875-.477 3.52 3.52 0 0 0 1.382-1.366c.352-.593.526-1.29.526-2.09 0-.79-.147-1.48-.444-2.073a3.235 3.235 0 0 0-1.283-1.399c-.549-.34-1.195-.51-1.942-.51a3.476 3.476 0 0 0-1.527.344l-.086.043-.165.09a3.412 3.412 0 0 0-.33.214c-.288.21-.507.446-.656.707a1.893 1.893 0 0 0-.099.198l.082-1.283V4.701h-1.744v12.095zm.473-2.509a2.482 2.482 0 0 0 .566.7c.078.065.159.125.245.18l.144.08a2.105 2.105 0 0 0 .975.232c.713 0 1.262-.225 1.645-.675.384-.46.576-1.053.576-1.778 0-.734-.192-1.327-.576-1.777-.373-.46-.921-.692-1.645-.692a2.18 2.18 0 0 0-1.015.235c-.147.075-.285.17-.415.282l-.15.142a2.086 2.086 0 0 0-.42.594c-.149.32-.223.685-.223 1.1v.115c0 .47.097.89.293 1.26zm2.616-.293c.157-.191.28-.479.28-.967 0-.51-.13-.79-.276-.961l-.021-.026c-.082-.1-.232-.225-.67-.225a.868.868 0 0 0-.681.279l-.012.011c-.154.155-.274.38-.274.807v.115c0 .285.057.499.144.669a1.13 1.13 0 0 0 .367.405c.137.082.28.123.455.123.423 0 .59-.118.686-.23zm8.266-3.013c.23-.087.472-.134.724-.14l.069-.002c.329 0 .542.033.642.099l.247-1.794c-.13-.066-.37-.099-.717-.099a2.3 2.3 0 0 0-.545.063 2.086 2.086 0 0 0-.411.148 2.18 2.18 0 0 0-.4.249 2.482 2.482 0 0 0-.485.499 2.659 2.659 0 0 0-.32.581l-.05.137v-1.48h-1.778v7.553h1.777v-3.884c0-.364.053-.678.159-.943a1.49 1.49 0 0 1 .466-.636 2.52 2.52 0 0 1 .399-.253 2.19 2.19 0 0 1 .224-.099zm9.784 2.656.05-.922c0-1.162-.285-2.062-.856-2.698-.559-.647-1.42-.97-2.584-.97-.746 0-1.415.163-2.007.493a3.462 3.462 0 0 0-1.4 1.382c-.329.604-.493 1.306-.493 2.106 0 .714.143 1.371.428 1.975.285.593.73 1.07 1.332 1.432.604.351 1.355.526 2.255.526.649 0 1.204-.062 1.668-.185l.044-.012.135-.04c.409-.122.736-.263.984-.421l-.542-1.267c-.2.108-.415.199-.642.274l-.297.087c-.34.088-.773.131-1.3.131-.636 0-1.135-.147-1.497-.444a1.573 1.573 0 0 1-.192-.193c-.244-.294-.415-.705-.512-1.234l-.004-.021h5.43zm-5.427-1.256-.003.022h3.752v-.138c-.007-.485-.104-.857-.288-1.118a1.056 1.056 0 0 0-.156-.176c-.307-.285-.746-.428-1.316-.428-.657 0-1.155.202-1.494.604-.253.3-.417.712-.494 1.234zm-27.053 2.77V4.7h-1.86v12.095h5.333V15.15zm7.103-5.908v7.553h-1.843V9.242h1.843z'/%3E%3Cpath fill='%23fff' d='m19.63 11.151-.757-1.71-.345 1.71-1.12 5.644h-1.827L18.083 4.7h.197l3.325 6.533.988 2.19.988-2.19L26.839 4.7h.181l2.6 12.095h-1.81l-1.218-5.644-.362-1.71-.658 1.71-2.93 5.644h-.098l-2.913-5.644zm14.836 5.81c-.68 0-1.311-.16-1.893-.478a3.795 3.795 0 0 1-1.381-1.382c-.34-.604-.51-1.306-.51-2.106 0-.79.147-1.482.444-2.074a3.364 3.364 0 0 1 1.3-1.382c.559-.33 1.217-.494 1.974-.494a3.293 3.293 0 0 1 1.234.231 3.341 3.341 0 0 1 .97.575c.264.22.44.439.527.659l.033-1.267h1.694v7.553H37.18l-.033-1.332c-.186.395-.526.746-1.02 1.053a3.167 3.167 0 0 1-1.662.444zm.296-1.482c.626 0 1.152-.214 1.58-.642.428-.44.642-1.01.642-1.711v-.115c0-.472-.098-.894-.296-1.267a2.211 2.211 0 0 0-.807-.872 2.098 2.098 0 0 0-1.119-.313c-.702 0-1.245.231-1.629.692-.384.45-.575 1.037-.575 1.76 0 .736.186 1.333.559 1.795.384.45.933.675 1.645.675zm6.521-6.237h1.711v1.4c.604-1.065 1.547-1.597 2.83-1.597 1.064 0 1.926.34 2.584 1.02.659.67.988 1.641.988 2.914 0 .79-.164 1.487-.493 2.09a3.456 3.456 0 0 1-1.316 1.399 3.51 3.51 0 0 1-1.844.493c-.636 0-1.19-.11-1.662-.329a2.665 2.665 0 0 1-1.086-.97l.017 5.134h-1.728V9.242zm4.048 6.22c.714 0 1.262-.224 1.645-.674.385-.46.576-1.048.576-1.762 0-.746-.192-1.338-.576-1.777-.372-.45-.92-.675-1.645-.675-.395 0-.768.098-1.12.296-.34.187-.613.46-.822.823-.197.351-.296.763-.296 1.234v.115c0 .472.098.894.296 1.267.209.362.483.647.823.855.34.197.713.297 1.119.297z'/%3E%3Cpath fill='%23e1e3e9' d='M51.325 4.7h1.86v10.45h3.473v1.646h-5.333zm7.12 4.542h1.843v7.553h-1.843zm.905-1.415a1.159 1.159 0 0 1-.856-.346 1.165 1.165 0 0 1-.346-.856 1.053 1.053 0 0 1 .346-.79c.23-.219.516-.329.856-.329.329 0 .609.11.839.33a1.053 1.053 0 0 1 .345.79 1.159 1.159 0 0 1-.345.855c-.22.23-.5.346-.84.346zm7.875 9.133a3.167 3.167 0 0 1-1.662-.444c-.482-.307-.817-.658-1.004-1.053l-.033 1.332h-1.71V4.701h1.743v4.657l-.082 1.283c.186-.438.548-.812 1.086-1.119a3.486 3.486 0 0 1 1.778-.477c.746 0 1.393.17 1.942.51a3.235 3.235 0 0 1 1.283 1.4c.297.592.444 1.282.444 2.072 0 .8-.175 1.498-.526 2.09a3.52 3.52 0 0 1-1.382 1.366 3.785 3.785 0 0 1-1.876.477zm-.296-1.481c.713 0 1.26-.225 1.645-.675.384-.46.577-1.053.577-1.778 0-.734-.193-1.327-.577-1.776-.373-.46-.921-.692-1.645-.692a2.115 2.115 0 0 0-1.58.659c-.428.428-.642.992-.642 1.694v.115c0 .473.098.895.296 1.267a2.385 2.385 0 0 0 .807.872 2.1 2.1 0 0 0 1.119.313zm5.927-6.237h1.777v1.481c.176-.505.46-.91.856-1.217a2.14 2.14 0 0 1 1.349-.46c.351 0 .593.032.724.098l-.247 1.794c-.099-.066-.313-.099-.642-.099-.516 0-.988.164-1.416.494-.417.329-.626.855-.626 1.58v3.883h-1.777V9.242zm9.534 7.718c-.9 0-1.651-.175-2.255-.526-.603-.362-1.047-.84-1.332-1.432a4.567 4.567 0 0 1-.428-1.975c0-.8.164-1.502.493-2.106a3.462 3.462 0 0 1 1.4-1.382c.592-.33 1.262-.494 2.007-.494 1.163 0 2.024.324 2.584.97.57.637.856 1.537.856 2.7 0 .296-.017.603-.05.92h-5.43c.12.67.356 1.153.708 1.45.362.296.86.443 1.497.443.526 0 .96-.044 1.3-.131a4.123 4.123 0 0 0 .938-.362l.542 1.267c-.274.175-.647.329-1.119.46-.472.132-1.042.197-1.711.197zm1.596-4.558c.01-.68-.137-1.158-.444-1.432-.307-.285-.746-.428-1.316-.428-1.152 0-1.815.62-1.991 1.86h3.752z'/%3E%3Cg fill-rule='evenodd' stroke-width='1.036'%3E%3Cpath fill='%23000' fill-opacity='.4' d='m8.166 16.146-.002.002a1.54 1.54 0 0 1-2.009 0l-.002-.002-.043-.034-.002-.002-.199-.162H4.377a.657.657 0 0 0-.659.659v1.84a.657.657 0 0 0 .659.659h5.565a.657.657 0 0 0 .659-.659v-1.84a.657.657 0 0 0-.659-.659H8.411l-.202.164zm-1.121-.905a.29.29 0 0 0 .113.023.286.286 0 0 0 .189-.07l.077-.063c.634-.508 4.672-3.743 4.672-7.575 0-2.55-2.215-4.625-4.938-4.625S2.221 5.006 2.221 7.556c0 3.225 2.86 6.027 4.144 7.137h.004l.04.038.484.4.077.063a.628.628 0 0 0 .074.047zm-2.52-.548a16.898 16.898 0 0 1-1.183-1.315C2.187 11.942.967 9.897.967 7.555c0-3.319 2.855-5.88 6.192-5.88 3.338 0 6.193 2.561 6.193 5.881 0 2.34-1.22 4.387-2.376 5.822a16.898 16.898 0 0 1-1.182 1.315h.15a1.912 1.912 0 0 1 1.914 1.914v1.84a1.912 1.912 0 0 1-1.914 1.914H4.377a1.912 1.912 0 0 1-1.914-1.914v-1.84a1.912 1.912 0 0 1 1.914-1.914zm3.82-6.935c0 .692-.55 1.222-1.187 1.222s-1.185-.529-1.185-1.222.548-1.222 1.185-1.222c.638 0 1.186.529 1.186 1.222zm-1.186 2.477c1.348 0 2.442-1.11 2.442-2.478S8.507 5.28 7.159 5.28 4.72 6.39 4.72 7.758s1.092 2.477 2.44 2.477zm2.048 7.71H5.114v-.838h4.093z'/%3E%3Cpath fill='%23e1e3e9' d='M2.222 7.555c0-2.55 2.214-4.625 4.937-4.625 2.723 0 4.938 2.075 4.938 4.625 0 3.832-4.038 7.068-4.672 7.575l-.077.063a.286.286 0 0 1-.189.07.286.286 0 0 1-.188-.07l-.077-.063c-.634-.507-4.672-3.743-4.672-7.575zm4.937 2.68c1.348 0 2.442-1.11 2.442-2.478S8.507 5.28 7.159 5.28 4.72 6.39 4.72 7.758s1.092 2.477 2.44 2.477z'/%3E%3Cpath fill='%23fff' d='M4.377 15.948a.657.657 0 0 0-.659.659v1.84a.657.657 0 0 0 .659.659h5.565a.657.657 0 0 0 .659-.659v-1.84a.657.657 0 0 0-.659-.659zm4.83 1.16H5.114v.838h4.093z'/%3E%3C/g%3E%3C/svg%3E");background-repeat:no-repeat;cursor:pointer;display:block;height:23px;margin:0 0 -4px -4px;overflow:hidden;width:88px}a.maplibregl-ctrl-logo.maplibregl-compact{width:14px}@media (-ms-high-contrast:active){a.maplibregl-ctrl-logo{background-color:transparent;background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='88' height='23' fill='none'%3E%3Cpath fill='%23000' fill-opacity='.4' fill-rule='evenodd' d='M17.408 16.796h-1.827l2.501-12.095h.198l3.324 6.533.988 2.19.988-2.19 3.258-6.533h.181l2.6 12.095h-1.81l-1.218-5.644-.362-1.71-.658 1.71-2.929 5.644h-.098l-2.914-5.644-.757-1.71-.345 1.71zm1.958-3.42-.726 3.663a1.255 1.255 0 0 1-1.232 1.011h-1.827a1.255 1.255 0 0 1-1.229-1.509l2.501-12.095a1.255 1.255 0 0 1 1.23-1.001h.197a1.255 1.255 0 0 1 1.12.685l3.19 6.273 3.125-6.263a1.255 1.255 0 0 1 1.123-.695h.181a1.255 1.255 0 0 1 1.227.991l1.443 6.71a5.11 5.11 0 0 1 .314-.787l.009-.016a4.623 4.623 0 0 1 1.777-1.887c.782-.46 1.668-.667 2.611-.667a4.548 4.548 0 0 1 1.7.32l.306.134c.21-.16.474-.256.759-.256h1.694a1.255 1.255 0 0 1 1.212.925 1.255 1.255 0 0 1 1.212-.925h1.711c.284 0 .545.094.755.252.613-.3 1.312-.45 2.075-.45 1.356 0 2.557.445 3.482 1.4.314.319.566.676.763 1.064V4.701a1.255 1.255 0 0 1 1.255-1.255h1.86A1.255 1.255 0 0 1 54.44 4.7v9.194h2.217c.19 0 .37.043.532.118v-4.77c0-.356.147-.678.385-.906a2.416 2.416 0 0 1-.682-1.71c0-.665.267-1.253.735-1.7a2.448 2.448 0 0 1 1.722-.674 2.43 2.43 0 0 1 1.705.675c.211.2.381.43.504.683V4.7a1.255 1.255 0 0 1 1.255-1.255h1.744A1.255 1.255 0 0 1 65.812 4.7v3.335a4.76 4.76 0 0 1 1.526-.246c.938 0 1.817.214 2.59.69a4.47 4.47 0 0 1 1.67 1.743v-.98a1.255 1.255 0 0 1 1.256-1.256h1.777c.233 0 .451.064.639.174a3.407 3.407 0 0 1 1.567-.372c.346 0 .861.02 1.285.232a1.255 1.255 0 0 1 .689 1.004 4.73 4.73 0 0 1 .853-.588c.795-.44 1.675-.647 2.61-.647 1.385 0 2.65.39 3.525 1.396.836.938 1.168 2.173 1.168 3.528 0 .343-.02.694-.056 1.051a1.255 1.255 0 0 1-.947 1.09l.408.952a1.255 1.255 0 0 1-.477 1.552c-.418.268-.92.463-1.458.612-.613.171-1.304.244-2.049.244-1.06 0-2.043-.207-2.886-.698l-.015-.008c-.798-.48-1.419-1.135-1.818-1.963l-.004-.008a5.815 5.815 0 0 1-.548-2.512c0-.286.017-.567.053-.843a1.255 1.255 0 0 1-.333-.086l-.166-.004c-.223 0-.426.062-.643.228-.03.024-.142.139-.142.59v3.883a1.255 1.255 0 0 1-1.256 1.256h-1.777a1.255 1.255 0 0 1-1.256-1.256V15.69l-.032.057a4.778 4.778 0 0 1-1.86 1.833 5.04 5.04 0 0 1-2.484.634 4.47 4.47 0 0 1-1.935-.424 1.252 1.252 0 0 1-.764.258h-1.71a1.255 1.255 0 0 1-1.256-1.255V7.687a2.402 2.402 0 0 1-.428.625c.253.23.412.561.412.93v7.553a1.255 1.255 0 0 1-1.256 1.255h-1.843a1.25 1.25 0 0 1-.894-.373c-.228.23-.544.373-.894.373H51.32a1.255 1.255 0 0 1-1.256-1.255v-1.251l-.061.117a4.703 4.703 0 0 1-1.782 1.884 4.767 4.767 0 0 1-2.485.67 5.6 5.6 0 0 1-1.485-.188l.009 2.764a1.255 1.255 0 0 1-1.255 1.259h-1.729a1.255 1.255 0 0 1-1.255-1.255v-3.537a1.255 1.255 0 0 1-1.167.793h-1.679a1.25 1.25 0 0 1-.77-.263 4.47 4.47 0 0 1-1.945.429c-.885 0-1.724-.21-2.495-.632l-.017-.01a4.983 4.983 0 0 1-1.081-.836 1.255 1.255 0 0 1-1.254 1.312h-1.81a1.255 1.255 0 0 1-1.228-.99l-.782-3.625-2.044 3.939a1.255 1.255 0 0 1-1.115.676h-.098a1.255 1.255 0 0 1-1.116-.68l-2.061-3.994zM35.92 16.63l.207-.114.223-.15c.329-.237.574-.499.735-.785l.061-.118.033 1.332h1.678V9.242h-1.694l-.033 1.267c-.088-.22-.264-.438-.526-.658l-.032-.028a3.16 3.16 0 0 0-.668-.428l-.27-.12a3.293 3.293 0 0 0-1.235-.23c-.757 0-1.415.163-1.974.493a3.36 3.36 0 0 0-1.3 1.382c-.297.593-.444 1.284-.444 2.074 0 .8.17 1.503.51 2.107a3.795 3.795 0 0 0 1.382 1.381 3.883 3.883 0 0 0 1.893.477c.53 0 1.015-.11 1.455-.33zm-2.789-5.38c-.384.45-.575 1.038-.575 1.762 0 .735.186 1.332.559 1.794.384.45.933.675 1.645.675a2.25 2.25 0 0 0 .934-.19 2.17 2.17 0 0 0 .468-.29l.178-.161a2.163 2.163 0 0 0 .397-.561c.163-.333.244-.717.244-1.15v-.115c0-.472-.098-.894-.296-1.267l-.043-.077a2.211 2.211 0 0 0-.633-.709l-.13-.086-.047-.028a2.099 2.099 0 0 0-1.073-.285c-.702 0-1.244.231-1.629.692zm2.316 2.706c.163-.17.28-.407.28-.83v-.114c0-.292-.06-.508-.15-.68a.958.958 0 0 0-.353-.389.851.851 0 0 0-.464-.127c-.4 0-.56.114-.664.239l-.01.012c-.148.174-.275.45-.275.945 0 .506.122.801.27.99.097.11.266.224.68.224.303 0 .504-.09.687-.269zm7.545 1.705a2.626 2.626 0 0 0 .331.423c.213.22.464.402.755.548l.173.074c.433.17.93.255 1.49.255.68 0 1.295-.165 1.844-.493a3.447 3.447 0 0 0 1.316-1.4c.329-.603.493-1.299.493-2.089 0-1.273-.33-2.243-.988-2.913-.658-.68-1.52-1.02-2.584-1.02-.598 0-1.124.115-1.575.347a2.807 2.807 0 0 0-.415.262l-.199.166a3.35 3.35 0 0 0-.64.82V9.242h-1.712v11.553h1.729l-.017-5.134zm.53-1.138c.137.193.297.36.48.5l.155.11.053.034c.34.197.713.297 1.119.297.714 0 1.262-.225 1.645-.675.385-.46.576-1.048.576-1.762 0-.746-.192-1.338-.576-1.777-.372-.45-.92-.675-1.645-.675-.29 0-.569.053-.835.16a2.366 2.366 0 0 0-.284.136 1.99 1.99 0 0 0-.363.254 2.237 2.237 0 0 0-.46.569l-.082.162a2.56 2.56 0 0 0-.213 1.072v.115c0 .471.098.894.296 1.267l.135.211zm.964-.818a1.11 1.11 0 0 0 .367.385.937.937 0 0 0 .476.118c.423 0 .59-.117.687-.23.159-.194.28-.478.28-.95 0-.53-.133-.8-.266-.952l-.021-.025c-.078-.094-.231-.221-.68-.221a.995.995 0 0 0-.503.135l-.012.007a.859.859 0 0 0-.335.343c-.073.133-.132.324-.132.614v.115a1.43 1.43 0 0 0 .14.66zm15.7-6.222c.232-.23.346-.516.346-.856a1.053 1.053 0 0 0-.345-.79 1.175 1.175 0 0 0-.84-.329c-.34 0-.625.11-.855.33a1.053 1.053 0 0 0-.346.79c0 .34.115.625.346.855.23.23.516.346.856.346.34 0 .62-.115.839-.346zm4.337 9.314.033-1.332c.128.269.324.518.59.747l.098.081a3.727 3.727 0 0 0 .316.224l.223.122a3.21 3.21 0 0 0 1.44.322 3.785 3.785 0 0 0 1.875-.477 3.52 3.52 0 0 0 1.382-1.366c.352-.593.526-1.29.526-2.09 0-.79-.147-1.48-.444-2.073a3.235 3.235 0 0 0-1.283-1.399c-.549-.34-1.195-.51-1.942-.51a3.476 3.476 0 0 0-1.527.344l-.086.043-.165.09a3.412 3.412 0 0 0-.33.214c-.288.21-.507.446-.656.707a1.893 1.893 0 0 0-.099.198l.082-1.283V4.701h-1.744v12.095zm.473-2.509a2.482 2.482 0 0 0 .566.7c.078.065.159.125.245.18l.144.08a2.105 2.105 0 0 0 .975.232c.713 0 1.262-.225 1.645-.675.384-.46.576-1.053.576-1.778 0-.734-.192-1.327-.576-1.777-.373-.46-.921-.692-1.645-.692a2.18 2.18 0 0 0-1.015.235c-.147.075-.285.17-.415.282l-.15.142a2.086 2.086 0 0 0-.42.594c-.149.32-.223.685-.223 1.1v.115c0 .47.097.89.293 1.26zm2.616-.293c.157-.191.28-.479.28-.967 0-.51-.13-.79-.276-.961l-.021-.026c-.082-.1-.232-.225-.67-.225a.868.868 0 0 0-.681.279l-.012.011c-.154.155-.274.38-.274.807v.115c0 .285.057.499.144.669a1.13 1.13 0 0 0 .367.405c.137.082.28.123.455.123.423 0 .59-.118.686-.23zm8.266-3.013c.23-.087.472-.134.724-.14l.069-.002c.329 0 .542.033.642.099l.247-1.794c-.13-.066-.37-.099-.717-.099a2.3 2.3 0 0 0-.545.063 2.086 2.086 0 0 0-.411.148 2.18 2.18 0 0 0-.4.249 2.482 2.482 0 0 0-.485.499 2.659 2.659 0 0 0-.32.581l-.05.137v-1.48h-1.778v7.553h1.777v-3.884c0-.364.053-.678.159-.943a1.49 1.49 0 0 1 .466-.636 2.52 2.52 0 0 1 .399-.253 2.19 2.19 0 0 1 .224-.099zm9.784 2.656.05-.922c0-1.162-.285-2.062-.856-2.698-.559-.647-1.42-.97-2.584-.97-.746 0-1.415.163-2.007.493a3.462 3.462 0 0 0-1.4 1.382c-.329.604-.493 1.306-.493 2.106 0 .714.143 1.371.428 1.975.285.593.73 1.07 1.332 1.432.604.351 1.355.526 2.255.526.649 0 1.204-.062 1.668-.185l.044-.012.135-.04c.409-.122.736-.263.984-.421l-.542-1.267c-.2.108-.415.199-.642.274l-.297.087c-.34.088-.773.131-1.3.131-.636 0-1.135-.147-1.497-.444a1.573 1.573 0 0 1-.192-.193c-.244-.294-.415-.705-.512-1.234l-.004-.021h5.43zm-5.427-1.256-.003.022h3.752v-.138c-.007-.485-.104-.857-.288-1.118a1.056 1.056 0 0 0-.156-.176c-.307-.285-.746-.428-1.316-.428-.657 0-1.155.202-1.494.604-.253.3-.417.712-.494 1.234zm-27.053 2.77V4.7h-1.86v12.095h5.333V15.15zm7.103-5.908v7.553h-1.843V9.242h1.843z'/%3E%3Cpath fill='%23fff' d='m19.63 11.151-.757-1.71-.345 1.71-1.12 5.644h-1.827L18.083 4.7h.197l3.325 6.533.988 2.19.988-2.19L26.839 4.7h.181l2.6 12.095h-1.81l-1.218-5.644-.362-1.71-.658 1.71-2.93 5.644h-.098l-2.913-5.644zm14.836 5.81c-.68 0-1.311-.16-1.893-.478a3.795 3.795 0 0 1-1.381-1.382c-.34-.604-.51-1.306-.51-2.106 0-.79.147-1.482.444-2.074a3.364 3.364 0 0 1 1.3-1.382c.559-.33 1.217-.494 1.974-.494a3.293 3.293 0 0 1 1.234.231 3.341 3.341 0 0 1 .97.575c.264.22.44.439.527.659l.033-1.267h1.694v7.553H37.18l-.033-1.332c-.186.395-.526.746-1.02 1.053a3.167 3.167 0 0 1-1.662.444zm.296-1.482c.626 0 1.152-.214 1.58-.642.428-.44.642-1.01.642-1.711v-.115c0-.472-.098-.894-.296-1.267a2.211 2.211 0 0 0-.807-.872 2.098 2.098 0 0 0-1.119-.313c-.702 0-1.245.231-1.629.692-.384.45-.575 1.037-.575 1.76 0 .736.186 1.333.559 1.795.384.45.933.675 1.645.675zm6.521-6.237h1.711v1.4c.604-1.065 1.547-1.597 2.83-1.597 1.064 0 1.926.34 2.584 1.02.659.67.988 1.641.988 2.914 0 .79-.164 1.487-.493 2.09a3.456 3.456 0 0 1-1.316 1.399 3.51 3.51 0 0 1-1.844.493c-.636 0-1.19-.11-1.662-.329a2.665 2.665 0 0 1-1.086-.97l.017 5.134h-1.728V9.242zm4.048 6.22c.714 0 1.262-.224 1.645-.674.385-.46.576-1.048.576-1.762 0-.746-.192-1.338-.576-1.777-.372-.45-.92-.675-1.645-.675-.395 0-.768.098-1.12.296-.34.187-.613.46-.822.823-.197.351-.296.763-.296 1.234v.115c0 .472.098.894.296 1.267.209.362.483.647.823.855.34.197.713.297 1.119.297z'/%3E%3Cpath fill='%23e1e3e9' d='M51.325 4.7h1.86v10.45h3.473v1.646h-5.333zm7.12 4.542h1.843v7.553h-1.843zm.905-1.415a1.159 1.159 0 0 1-.856-.346 1.165 1.165 0 0 1-.346-.856 1.053 1.053 0 0 1 .346-.79c.23-.219.516-.329.856-.329.329 0 .609.11.839.33a1.053 1.053 0 0 1 .345.79 1.159 1.159 0 0 1-.345.855c-.22.23-.5.346-.84.346zm7.875 9.133a3.167 3.167 0 0 1-1.662-.444c-.482-.307-.817-.658-1.004-1.053l-.033 1.332h-1.71V4.701h1.743v4.657l-.082 1.283c.186-.438.548-.812 1.086-1.119a3.486 3.486 0 0 1 1.778-.477c.746 0 1.393.17 1.942.51a3.235 3.235 0 0 1 1.283 1.4c.297.592.444 1.282.444 2.072 0 .8-.175 1.498-.526 2.09a3.52 3.52 0 0 1-1.382 1.366 3.785 3.785 0 0 1-1.876.477zm-.296-1.481c.713 0 1.26-.225 1.645-.675.384-.46.577-1.053.577-1.778 0-.734-.193-1.327-.577-1.776-.373-.46-.921-.692-1.645-.692a2.115 2.115 0 0 0-1.58.659c-.428.428-.642.992-.642 1.694v.115c0 .473.098.895.296 1.267a2.385 2.385 0 0 0 .807.872 2.1 2.1 0 0 0 1.119.313zm5.927-6.237h1.777v1.481c.176-.505.46-.91.856-1.217a2.14 2.14 0 0 1 1.349-.46c.351 0 .593.032.724.098l-.247 1.794c-.099-.066-.313-.099-.642-.099-.516 0-.988.164-1.416.494-.417.329-.626.855-.626 1.58v3.883h-1.777V9.242zm9.534 7.718c-.9 0-1.651-.175-2.255-.526-.603-.362-1.047-.84-1.332-1.432a4.567 4.567 0 0 1-.428-1.975c0-.8.164-1.502.493-2.106a3.462 3.462 0 0 1 1.4-1.382c.592-.33 1.262-.494 2.007-.494 1.163 0 2.024.324 2.584.97.57.637.856 1.537.856 2.7 0 .296-.017.603-.05.92h-5.43c.12.67.356 1.153.708 1.45.362.296.86.443 1.497.443.526 0 .96-.044 1.3-.131a4.123 4.123 0 0 0 .938-.362l.542 1.267c-.274.175-.647.329-1.119.46-.472.132-1.042.197-1.711.197zm1.596-4.558c.01-.68-.137-1.158-.444-1.432-.307-.285-.746-.428-1.316-.428-1.152 0-1.815.62-1.991 1.86h3.752z'/%3E%3Cg fill-rule='evenodd' stroke-width='1.036'%3E%3Cpath fill='%23000' fill-opacity='.4' d='m8.166 16.146-.002.002a1.54 1.54 0 0 1-2.009 0l-.002-.002-.043-.034-.002-.002-.199-.162H4.377a.657.657 0 0 0-.659.659v1.84a.657.657 0 0 0 .659.659h5.565a.657.657 0 0 0 .659-.659v-1.84a.657.657 0 0 0-.659-.659H8.411l-.202.164zm-1.121-.905a.29.29 0 0 0 .113.023.286.286 0 0 0 .189-.07l.077-.063c.634-.508 4.672-3.743 4.672-7.575 0-2.55-2.215-4.625-4.938-4.625S2.221 5.006 2.221 7.556c0 3.225 2.86 6.027 4.144 7.137h.004l.04.038.484.4.077.063a.628.628 0 0 0 .074.047zm-2.52-.548a16.898 16.898 0 0 1-1.183-1.315C2.187 11.942.967 9.897.967 7.555c0-3.319 2.855-5.88 6.192-5.88 3.338 0 6.193 2.561 6.193 5.881 0 2.34-1.22 4.387-2.376 5.822a16.898 16.898 0 0 1-1.182 1.315h.15a1.912 1.912 0 0 1 1.914 1.914v1.84a1.912 1.912 0 0 1-1.914 1.914H4.377a1.912 1.912 0 0 1-1.914-1.914v-1.84a1.912 1.912 0 0 1 1.914-1.914zm3.82-6.935c0 .692-.55 1.222-1.187 1.222s-1.185-.529-1.185-1.222.548-1.222 1.185-1.222c.638 0 1.186.529 1.186 1.222zm-1.186 2.477c1.348 0 2.442-1.11 2.442-2.478S8.507 5.28 7.159 5.28 4.72 6.39 4.72 7.758s1.092 2.477 2.44 2.477zm2.048 7.71H5.114v-.838h4.093z'/%3E%3Cpath fill='%23e1e3e9' d='M2.222 7.555c0-2.55 2.214-4.625 4.937-4.625 2.723 0 4.938 2.075 4.938 4.625 0 3.832-4.038 7.068-4.672 7.575l-.077.063a.286.286 0 0 1-.189.07.286.286 0 0 1-.188-.07l-.077-.063c-.634-.507-4.672-3.743-4.672-7.575zm4.937 2.68c1.348 0 2.442-1.11 2.442-2.478S8.507 5.28 7.159 5.28 4.72 6.39 4.72 7.758s1.092 2.477 2.44 2.477z'/%3E%3Cpath fill='%23fff' d='M4.377 15.948a.657.657 0 0 0-.659.659v1.84a.657.657 0 0 0 .659.659h5.565a.657.657 0 0 0 .659-.659v-1.84a.657.657 0 0 0-.659-.659zm4.83 1.16H5.114v.838h4.093z'/%3E%3C/g%3E%3C/svg%3E")}}@media (-ms-high-contrast:black-on-white){a.maplibregl-ctrl-logo{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='88' height='23' fill='none'%3E%3Cpath fill='%23000' fill-opacity='.4' fill-rule='evenodd' d='M17.408 16.796h-1.827l2.501-12.095h.198l3.324 6.533.988 2.19.988-2.19 3.258-6.533h.181l2.6 12.095h-1.81l-1.218-5.644-.362-1.71-.658 1.71-2.929 5.644h-.098l-2.914-5.644-.757-1.71-.345 1.71zm1.958-3.42-.726 3.663a1.255 1.255 0 0 1-1.232 1.011h-1.827a1.255 1.255 0 0 1-1.229-1.509l2.501-12.095a1.255 1.255 0 0 1 1.23-1.001h.197a1.255 1.255 0 0 1 1.12.685l3.19 6.273 3.125-6.263a1.255 1.255 0 0 1 1.123-.695h.181a1.255 1.255 0 0 1 1.227.991l1.443 6.71a5.11 5.11 0 0 1 .314-.787l.009-.016a4.623 4.623 0 0 1 1.777-1.887c.782-.46 1.668-.667 2.611-.667a4.548 4.548 0 0 1 1.7.32l.306.134c.21-.16.474-.256.759-.256h1.694a1.255 1.255 0 0 1 1.212.925 1.255 1.255 0 0 1 1.212-.925h1.711c.284 0 .545.094.755.252.613-.3 1.312-.45 2.075-.45 1.356 0 2.557.445 3.482 1.4.314.319.566.676.763 1.064V4.701a1.255 1.255 0 0 1 1.255-1.255h1.86A1.255 1.255 0 0 1 54.44 4.7v9.194h2.217c.19 0 .37.043.532.118v-4.77c0-.356.147-.678.385-.906a2.416 2.416 0 0 1-.682-1.71c0-.665.267-1.253.735-1.7a2.448 2.448 0 0 1 1.722-.674 2.43 2.43 0 0 1 1.705.675c.211.2.381.43.504.683V4.7a1.255 1.255 0 0 1 1.255-1.255h1.744A1.255 1.255 0 0 1 65.812 4.7v3.335a4.76 4.76 0 0 1 1.526-.246c.938 0 1.817.214 2.59.69a4.47 4.47 0 0 1 1.67 1.743v-.98a1.255 1.255 0 0 1 1.256-1.256h1.777c.233 0 .451.064.639.174a3.407 3.407 0 0 1 1.567-.372c.346 0 .861.02 1.285.232a1.255 1.255 0 0 1 .689 1.004 4.73 4.73 0 0 1 .853-.588c.795-.44 1.675-.647 2.61-.647 1.385 0 2.65.39 3.525 1.396.836.938 1.168 2.173 1.168 3.528 0 .343-.02.694-.056 1.051a1.255 1.255 0 0 1-.947 1.09l.408.952a1.255 1.255 0 0 1-.477 1.552c-.418.268-.92.463-1.458.612-.613.171-1.304.244-2.049.244-1.06 0-2.043-.207-2.886-.698l-.015-.008c-.798-.48-1.419-1.135-1.818-1.963l-.004-.008a5.815 5.815 0 0 1-.548-2.512c0-.286.017-.567.053-.843a1.255 1.255 0 0 1-.333-.086l-.166-.004c-.223 0-.426.062-.643.228-.03.024-.142.139-.142.59v3.883a1.255 1.255 0 0 1-1.256 1.256h-1.777a1.255 1.255 0 0 1-1.256-1.256V15.69l-.032.057a4.778 4.778 0 0 1-1.86 1.833 5.04 5.04 0 0 1-2.484.634 4.47 4.47 0 0 1-1.935-.424 1.252 1.252 0 0 1-.764.258h-1.71a1.255 1.255 0 0 1-1.256-1.255V7.687a2.402 2.402 0 0 1-.428.625c.253.23.412.561.412.93v7.553a1.255 1.255 0 0 1-1.256 1.255h-1.843a1.25 1.25 0 0 1-.894-.373c-.228.23-.544.373-.894.373H51.32a1.255 1.255 0 0 1-1.256-1.255v-1.251l-.061.117a4.703 4.703 0 0 1-1.782 1.884 4.767 4.767 0 0 1-2.485.67 5.6 5.6 0 0 1-1.485-.188l.009 2.764a1.255 1.255 0 0 1-1.255 1.259h-1.729a1.255 1.255 0 0 1-1.255-1.255v-3.537a1.255 1.255 0 0 1-1.167.793h-1.679a1.25 1.25 0 0 1-.77-.263 4.47 4.47 0 0 1-1.945.429c-.885 0-1.724-.21-2.495-.632l-.017-.01a4.983 4.983 0 0 1-1.081-.836 1.255 1.255 0 0 1-1.254 1.312h-1.81a1.255 1.255 0 0 1-1.228-.99l-.782-3.625-2.044 3.939a1.255 1.255 0 0 1-1.115.676h-.098a1.255 1.255 0 0 1-1.116-.68l-2.061-3.994zM35.92 16.63l.207-.114.223-.15c.329-.237.574-.499.735-.785l.061-.118.033 1.332h1.678V9.242h-1.694l-.033 1.267c-.088-.22-.264-.438-.526-.658l-.032-.028a3.16 3.16 0 0 0-.668-.428l-.27-.12a3.293 3.293 0 0 0-1.235-.23c-.757 0-1.415.163-1.974.493a3.36 3.36 0 0 0-1.3 1.382c-.297.593-.444 1.284-.444 2.074 0 .8.17 1.503.51 2.107a3.795 3.795 0 0 0 1.382 1.381 3.883 3.883 0 0 0 1.893.477c.53 0 1.015-.11 1.455-.33zm-2.789-5.38c-.384.45-.575 1.038-.575 1.762 0 .735.186 1.332.559 1.794.384.45.933.675 1.645.675a2.25 2.25 0 0 0 .934-.19 2.17 2.17 0 0 0 .468-.29l.178-.161a2.163 2.163 0 0 0 .397-.561c.163-.333.244-.717.244-1.15v-.115c0-.472-.098-.894-.296-1.267l-.043-.077a2.211 2.211 0 0 0-.633-.709l-.13-.086-.047-.028a2.099 2.099 0 0 0-1.073-.285c-.702 0-1.244.231-1.629.692zm2.316 2.706c.163-.17.28-.407.28-.83v-.114c0-.292-.06-.508-.15-.68a.958.958 0 0 0-.353-.389.851.851 0 0 0-.464-.127c-.4 0-.56.114-.664.239l-.01.012c-.148.174-.275.45-.275.945 0 .506.122.801.27.99.097.11.266.224.68.224.303 0 .504-.09.687-.269zm7.545 1.705a2.626 2.626 0 0 0 .331.423c.213.22.464.402.755.548l.173.074c.433.17.93.255 1.49.255.68 0 1.295-.165 1.844-.493a3.447 3.447 0 0 0 1.316-1.4c.329-.603.493-1.299.493-2.089 0-1.273-.33-2.243-.988-2.913-.658-.68-1.52-1.02-2.584-1.02-.598 0-1.124.115-1.575.347a2.807 2.807 0 0 0-.415.262l-.199.166a3.35 3.35 0 0 0-.64.82V9.242h-1.712v11.553h1.729l-.017-5.134zm.53-1.138c.137.193.297.36.48.5l.155.11.053.034c.34.197.713.297 1.119.297.714 0 1.262-.225 1.645-.675.385-.46.576-1.048.576-1.762 0-.746-.192-1.338-.576-1.777-.372-.45-.92-.675-1.645-.675-.29 0-.569.053-.835.16a2.366 2.366 0 0 0-.284.136 1.99 1.99 0 0 0-.363.254 2.237 2.237 0 0 0-.46.569l-.082.162a2.56 2.56 0 0 0-.213 1.072v.115c0 .471.098.894.296 1.267l.135.211zm.964-.818a1.11 1.11 0 0 0 .367.385.937.937 0 0 0 .476.118c.423 0 .59-.117.687-.23.159-.194.28-.478.28-.95 0-.53-.133-.8-.266-.952l-.021-.025c-.078-.094-.231-.221-.68-.221a.995.995 0 0 0-.503.135l-.012.007a.859.859 0 0 0-.335.343c-.073.133-.132.324-.132.614v.115a1.43 1.43 0 0 0 .14.66zm15.7-6.222c.232-.23.346-.516.346-.856a1.053 1.053 0 0 0-.345-.79 1.175 1.175 0 0 0-.84-.329c-.34 0-.625.11-.855.33a1.053 1.053 0 0 0-.346.79c0 .34.115.625.346.855.23.23.516.346.856.346.34 0 .62-.115.839-.346zm4.337 9.314.033-1.332c.128.269.324.518.59.747l.098.081a3.727 3.727 0 0 0 .316.224l.223.122a3.21 3.21 0 0 0 1.44.322 3.785 3.785 0 0 0 1.875-.477 3.52 3.52 0 0 0 1.382-1.366c.352-.593.526-1.29.526-2.09 0-.79-.147-1.48-.444-2.073a3.235 3.235 0 0 0-1.283-1.399c-.549-.34-1.195-.51-1.942-.51a3.476 3.476 0 0 0-1.527.344l-.086.043-.165.09a3.412 3.412 0 0 0-.33.214c-.288.21-.507.446-.656.707a1.893 1.893 0 0 0-.099.198l.082-1.283V4.701h-1.744v12.095zm.473-2.509a2.482 2.482 0 0 0 .566.7c.078.065.159.125.245.18l.144.08a2.105 2.105 0 0 0 .975.232c.713 0 1.262-.225 1.645-.675.384-.46.576-1.053.576-1.778 0-.734-.192-1.327-.576-1.777-.373-.46-.921-.692-1.645-.692a2.18 2.18 0 0 0-1.015.235c-.147.075-.285.17-.415.282l-.15.142a2.086 2.086 0 0 0-.42.594c-.149.32-.223.685-.223 1.1v.115c0 .47.097.89.293 1.26zm2.616-.293c.157-.191.28-.479.28-.967 0-.51-.13-.79-.276-.961l-.021-.026c-.082-.1-.232-.225-.67-.225a.868.868 0 0 0-.681.279l-.012.011c-.154.155-.274.38-.274.807v.115c0 .285.057.499.144.669a1.13 1.13 0 0 0 .367.405c.137.082.28.123.455.123.423 0 .59-.118.686-.23zm8.266-3.013c.23-.087.472-.134.724-.14l.069-.002c.329 0 .542.033.642.099l.247-1.794c-.13-.066-.37-.099-.717-.099a2.3 2.3 0 0 0-.545.063 2.086 2.086 0 0 0-.411.148 2.18 2.18 0 0 0-.4.249 2.482 2.482 0 0 0-.485.499 2.659 2.659 0 0 0-.32.581l-.05.137v-1.48h-1.778v7.553h1.777v-3.884c0-.364.053-.678.159-.943a1.49 1.49 0 0 1 .466-.636 2.52 2.52 0 0 1 .399-.253 2.19 2.19 0 0 1 .224-.099zm9.784 2.656.05-.922c0-1.162-.285-2.062-.856-2.698-.559-.647-1.42-.97-2.584-.97-.746 0-1.415.163-2.007.493a3.462 3.462 0 0 0-1.4 1.382c-.329.604-.493 1.306-.493 2.106 0 .714.143 1.371.428 1.975.285.593.73 1.07 1.332 1.432.604.351 1.355.526 2.255.526.649 0 1.204-.062 1.668-.185l.044-.012.135-.04c.409-.122.736-.263.984-.421l-.542-1.267c-.2.108-.415.199-.642.274l-.297.087c-.34.088-.773.131-1.3.131-.636 0-1.135-.147-1.497-.444a1.573 1.573 0 0 1-.192-.193c-.244-.294-.415-.705-.512-1.234l-.004-.021h5.43zm-5.427-1.256-.003.022h3.752v-.138c-.007-.485-.104-.857-.288-1.118a1.056 1.056 0 0 0-.156-.176c-.307-.285-.746-.428-1.316-.428-.657 0-1.155.202-1.494.604-.253.3-.417.712-.494 1.234zm-27.053 2.77V4.7h-1.86v12.095h5.333V15.15zm7.103-5.908v7.553h-1.843V9.242h1.843z'/%3E%3Cpath fill='%23fff' d='m19.63 11.151-.757-1.71-.345 1.71-1.12 5.644h-1.827L18.083 4.7h.197l3.325 6.533.988 2.19.988-2.19L26.839 4.7h.181l2.6 12.095h-1.81l-1.218-5.644-.362-1.71-.658 1.71-2.93 5.644h-.098l-2.913-5.644zm14.836 5.81c-.68 0-1.311-.16-1.893-.478a3.795 3.795 0 0 1-1.381-1.382c-.34-.604-.51-1.306-.51-2.106 0-.79.147-1.482.444-2.074a3.364 3.364 0 0 1 1.3-1.382c.559-.33 1.217-.494 1.974-.494a3.293 3.293 0 0 1 1.234.231 3.341 3.341 0 0 1 .97.575c.264.22.44.439.527.659l.033-1.267h1.694v7.553H37.18l-.033-1.332c-.186.395-.526.746-1.02 1.053a3.167 3.167 0 0 1-1.662.444zm.296-1.482c.626 0 1.152-.214 1.58-.642.428-.44.642-1.01.642-1.711v-.115c0-.472-.098-.894-.296-1.267a2.211 2.211 0 0 0-.807-.872 2.098 2.098 0 0 0-1.119-.313c-.702 0-1.245.231-1.629.692-.384.45-.575 1.037-.575 1.76 0 .736.186 1.333.559 1.795.384.45.933.675 1.645.675zm6.521-6.237h1.711v1.4c.604-1.065 1.547-1.597 2.83-1.597 1.064 0 1.926.34 2.584 1.02.659.67.988 1.641.988 2.914 0 .79-.164 1.487-.493 2.09a3.456 3.456 0 0 1-1.316 1.399 3.51 3.51 0 0 1-1.844.493c-.636 0-1.19-.11-1.662-.329a2.665 2.665 0 0 1-1.086-.97l.017 5.134h-1.728V9.242zm4.048 6.22c.714 0 1.262-.224 1.645-.674.385-.46.576-1.048.576-1.762 0-.746-.192-1.338-.576-1.777-.372-.45-.92-.675-1.645-.675-.395 0-.768.098-1.12.296-.34.187-.613.46-.822.823-.197.351-.296.763-.296 1.234v.115c0 .472.098.894.296 1.267.209.362.483.647.823.855.34.197.713.297 1.119.297z'/%3E%3Cpath fill='%23e1e3e9' d='M51.325 4.7h1.86v10.45h3.473v1.646h-5.333zm7.12 4.542h1.843v7.553h-1.843zm.905-1.415a1.159 1.159 0 0 1-.856-.346 1.165 1.165 0 0 1-.346-.856 1.053 1.053 0 0 1 .346-.79c.23-.219.516-.329.856-.329.329 0 .609.11.839.33a1.053 1.053 0 0 1 .345.79 1.159 1.159 0 0 1-.345.855c-.22.23-.5.346-.84.346zm7.875 9.133a3.167 3.167 0 0 1-1.662-.444c-.482-.307-.817-.658-1.004-1.053l-.033 1.332h-1.71V4.701h1.743v4.657l-.082 1.283c.186-.438.548-.812 1.086-1.119a3.486 3.486 0 0 1 1.778-.477c.746 0 1.393.17 1.942.51a3.235 3.235 0 0 1 1.283 1.4c.297.592.444 1.282.444 2.072 0 .8-.175 1.498-.526 2.09a3.52 3.52 0 0 1-1.382 1.366 3.785 3.785 0 0 1-1.876.477zm-.296-1.481c.713 0 1.26-.225 1.645-.675.384-.46.577-1.053.577-1.778 0-.734-.193-1.327-.577-1.776-.373-.46-.921-.692-1.645-.692a2.115 2.115 0 0 0-1.58.659c-.428.428-.642.992-.642 1.694v.115c0 .473.098.895.296 1.267a2.385 2.385 0 0 0 .807.872 2.1 2.1 0 0 0 1.119.313zm5.927-6.237h1.777v1.481c.176-.505.46-.91.856-1.217a2.14 2.14 0 0 1 1.349-.46c.351 0 .593.032.724.098l-.247 1.794c-.099-.066-.313-.099-.642-.099-.516 0-.988.164-1.416.494-.417.329-.626.855-.626 1.58v3.883h-1.777V9.242zm9.534 7.718c-.9 0-1.651-.175-2.255-.526-.603-.362-1.047-.84-1.332-1.432a4.567 4.567 0 0 1-.428-1.975c0-.8.164-1.502.493-2.106a3.462 3.462 0 0 1 1.4-1.382c.592-.33 1.262-.494 2.007-.494 1.163 0 2.024.324 2.584.97.57.637.856 1.537.856 2.7 0 .296-.017.603-.05.92h-5.43c.12.67.356 1.153.708 1.45.362.296.86.443 1.497.443.526 0 .96-.044 1.3-.131a4.123 4.123 0 0 0 .938-.362l.542 1.267c-.274.175-.647.329-1.119.46-.472.132-1.042.197-1.711.197zm1.596-4.558c.01-.68-.137-1.158-.444-1.432-.307-.285-.746-.428-1.316-.428-1.152 0-1.815.62-1.991 1.86h3.752z'/%3E%3Cg fill-rule='evenodd' stroke-width='1.036'%3E%3Cpath fill='%23000' fill-opacity='.4' d='m8.166 16.146-.002.002a1.54 1.54 0 0 1-2.009 0l-.002-.002-.043-.034-.002-.002-.199-.162H4.377a.657.657 0 0 0-.659.659v1.84a.657.657 0 0 0 .659.659h5.565a.657.657 0 0 0 .659-.659v-1.84a.657.657 0 0 0-.659-.659H8.411l-.202.164zm-1.121-.905a.29.29 0 0 0 .113.023.286.286 0 0 0 .189-.07l.077-.063c.634-.508 4.672-3.743 4.672-7.575 0-2.55-2.215-4.625-4.938-4.625S2.221 5.006 2.221 7.556c0 3.225 2.86 6.027 4.144 7.137h.004l.04.038.484.4.077.063a.628.628 0 0 0 .074.047zm-2.52-.548a16.898 16.898 0 0 1-1.183-1.315C2.187 11.942.967 9.897.967 7.555c0-3.319 2.855-5.88 6.192-5.88 3.338 0 6.193 2.561 6.193 5.881 0 2.34-1.22 4.387-2.376 5.822a16.898 16.898 0 0 1-1.182 1.315h.15a1.912 1.912 0 0 1 1.914 1.914v1.84a1.912 1.912 0 0 1-1.914 1.914H4.377a1.912 1.912 0 0 1-1.914-1.914v-1.84a1.912 1.912 0 0 1 1.914-1.914zm3.82-6.935c0 .692-.55 1.222-1.187 1.222s-1.185-.529-1.185-1.222.548-1.222 1.185-1.222c.638 0 1.186.529 1.186 1.222zm-1.186 2.477c1.348 0 2.442-1.11 2.442-2.478S8.507 5.28 7.159 5.28 4.72 6.39 4.72 7.758s1.092 2.477 2.44 2.477zm2.048 7.71H5.114v-.838h4.093z'/%3E%3Cpath fill='%23e1e3e9' d='M2.222 7.555c0-2.55 2.214-4.625 4.937-4.625 2.723 0 4.938 2.075 4.938 4.625 0 3.832-4.038 7.068-4.672 7.575l-.077.063a.286.286 0 0 1-.189.07.286.286 0 0 1-.188-.07l-.077-.063c-.634-.507-4.672-3.743-4.672-7.575zm4.937 2.68c1.348 0 2.442-1.11 2.442-2.478S8.507 5.28 7.159 5.28 4.72 6.39 4.72 7.758s1.092 2.477 2.44 2.477z'/%3E%3Cpath fill='%23fff' d='M4.377 15.948a.657.657 0 0 0-.659.659v1.84a.657.657 0 0 0 .659.659h5.565a.657.657 0 0 0 .659-.659v-1.84a.657.657 0 0 0-.659-.659zm4.83 1.16H5.114v.838h4.093z'/%3E%3C/g%3E%3C/svg%3E")}}.maplibregl-ctrl.maplibregl-ctrl-attrib{background-color:hsla(0,0%,100%,.5);margin:0;padding:0 5px}@media screen{.maplibregl-ctrl-attrib.maplibregl-compact{background-color:#fff;border-radius:12px;box-sizing:content-box;margin:10px;min-height:20px;padding:2px 24px 2px 0;position:relative}.maplibregl-ctrl-attrib.maplibregl-compact-show{padding:2px 28px 2px 8px;visibility:visible}.maplibregl-ctrl-bottom-left>.maplibregl-ctrl-attrib.maplibregl-compact-show,.maplibregl-ctrl-top-left>.maplibregl-ctrl-attrib.maplibregl-compact-show{border-radius:12px;padding:2px 8px 2px 28px}.maplibregl-ctrl-attrib.maplibregl-compact .maplibregl-ctrl-attrib-inner{display:none}.maplibregl-ctrl-attrib-button{background-color:hsla(0,0%,100%,.5);background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' fill-rule='evenodd' viewBox='0 0 20 20'%3E%3Cpath d='M4 10a6 6 0 1 0 12 0 6 6 0 1 0-12 0m5-3a1 1 0 1 0 2 0 1 1 0 1 0-2 0m0 3a1 1 0 1 1 2 0v3a1 1 0 1 1-2 0'/%3E%3C/svg%3E");border:0;border-radius:12px;box-sizing:border-box;cursor:pointer;display:none;height:24px;outline:none;position:absolute;right:0;top:0;width:24px}.maplibregl-ctrl-attrib summary.maplibregl-ctrl-attrib-button{appearance:none;list-style:none}.maplibregl-ctrl-attrib summary.maplibregl-ctrl-attrib-button::-webkit-details-marker{display:none}.maplibregl-ctrl-bottom-left .maplibregl-ctrl-attrib-button,.maplibregl-ctrl-top-left .maplibregl-ctrl-attrib-button{left:0}.maplibregl-ctrl-attrib.maplibregl-compact .maplibregl-ctrl-attrib-button,.maplibregl-ctrl-attrib.maplibregl-compact-show .maplibregl-ctrl-attrib-inner{display:block}.maplibregl-ctrl-attrib.maplibregl-compact-show .maplibregl-ctrl-attrib-button{background-color:rgb(0 0 0/5%)}.maplibregl-ctrl-bottom-right>.maplibregl-ctrl-attrib.maplibregl-compact:after{bottom:0;right:0}.maplibregl-ctrl-top-right>.maplibregl-ctrl-attrib.maplibregl-compact:after{right:0;top:0}.maplibregl-ctrl-top-left>.maplibregl-ctrl-attrib.maplibregl-compact:after{left:0;top:0}.maplibregl-ctrl-bottom-left>.maplibregl-ctrl-attrib.maplibregl-compact:after{bottom:0;left:0}}@media screen and (-ms-high-contrast:active){.maplibregl-ctrl-attrib.maplibregl-compact:after{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' fill='%23fff' fill-rule='evenodd' viewBox='0 0 20 20'%3E%3Cpath d='M4 10a6 6 0 1 0 12 0 6 6 0 1 0-12 0m5-3a1 1 0 1 0 2 0 1 1 0 1 0-2 0m0 3a1 1 0 1 1 2 0v3a1 1 0 1 1-2 0'/%3E%3C/svg%3E")}}@media screen and (-ms-high-contrast:black-on-white){.maplibregl-ctrl-attrib.maplibregl-compact:after{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' fill-rule='evenodd' viewBox='0 0 20 20'%3E%3Cpath d='M4 10a6 6 0 1 0 12 0 6 6 0 1 0-12 0m5-3a1 1 0 1 0 2 0 1 1 0 1 0-2 0m0 3a1 1 0 1 1 2 0v3a1 1 0 1 1-2 0'/%3E%3C/svg%3E")}}.maplibregl-ctrl-attrib a{color:rgba(0,0,0,.75);text-decoration:none}.maplibregl-ctrl-attrib a:hover{color:inherit;text-decoration:underline}.maplibregl-attrib-empty{display:none}.maplibregl-ctrl-scale{background-color:hsla(0,0%,100%,.75);border:2px solid #333;border-top:#333;box-sizing:border-box;color:#333;font-size:10px;padding:0 5px}.maplibregl-popup{display:flex;left:0;pointer-events:none;position:absolute;top:0;will-change:transform}.maplibregl-popup-anchor-top,.maplibregl-popup-anchor-top-left,.maplibregl-popup-anchor-top-right{flex-direction:column}.maplibregl-popup-anchor-bottom,.maplibregl-popup-anchor-bottom-left,.maplibregl-popup-anchor-bottom-right{flex-direction:column-reverse}.maplibregl-popup-anchor-left{flex-direction:row}.maplibregl-popup-anchor-right{flex-direction:row-reverse}.maplibregl-popup-tip{border:10px solid transparent;height:0;width:0;z-index:1}.maplibregl-popup-anchor-top .maplibregl-popup-tip{align-self:center;border-bottom-color:#fff;border-top:none}.maplibregl-popup-anchor-top-left .maplibregl-popup-tip{align-self:flex-start;border-bottom-color:#fff;border-left:none;border-top:none}.maplibregl-popup-anchor-top-right .maplibregl-popup-tip{align-self:flex-end;border-bottom-color:#fff;border-right:none;border-top:none}.maplibregl-popup-anchor-bottom .maplibregl-popup-tip{align-self:center;border-bottom:none;border-top-color:#fff}.maplibregl-popup-anchor-bottom-left .maplibregl-popup-tip{align-self:flex-start;border-bottom:none;border-left:none;border-top-color:#fff}.maplibregl-popup-anchor-bottom-right .maplibregl-popup-tip{align-self:flex-end;border-bottom:none;border-right:none;border-top-color:#fff}.maplibregl-popup-anchor-left .maplibregl-popup-tip{align-self:center;border-left:none;border-right-color:#fff}.maplibregl-popup-anchor-right .maplibregl-popup-tip{align-self:center;border-left-color:#fff;border-right:none}.maplibregl-popup-close-button{background-color:transparent;border:0;border-radius:0 3px 0 0;cursor:pointer;position:absolute;right:0;top:0}.maplibregl-popup-close-button:hover{background-color:rgb(0 0 0/5%)}.maplibregl-popup-content{background:#fff;border-radius:3px;box-shadow:0 1px 2px rgba(0,0,0,.1);padding:15px 10px;pointer-events:auto;position:relative}.maplibregl-popup-anchor-top-left .maplibregl-popup-content{border-top-left-radius:0}.maplibregl-popup-anchor-top-right .maplibregl-popup-content{border-top-right-radius:0}.maplibregl-popup-anchor-bottom-left .maplibregl-popup-content{border-bottom-left-radius:0}.maplibregl-popup-anchor-bottom-right .maplibregl-popup-content{border-bottom-right-radius:0}.maplibregl-popup-track-pointer{display:none}.maplibregl-popup-track-pointer *{pointer-events:none;user-select:none}.maplibregl-map:hover .maplibregl-popup-track-pointer{display:flex}.maplibregl-map:active .maplibregl-popup-track-pointer{display:none}.maplibregl-marker{left:0;position:absolute;top:0;will-change:transform}.maplibregl-user-location-dot,.maplibregl-user-location-dot:before{background-color:#1da1f2;border-radius:50%;height:15px;width:15px}.maplibregl-user-location-dot:before{animation:maplibregl-user-location-dot-pulse 2s infinite;content:"";position:absolute}.maplibregl-user-location-dot:after{border:2px solid #fff;border-radius:50%;box-shadow:0 0 3px rgba(0,0,0,.35);box-sizing:border-box;content:"";height:19px;left:-2px;position:absolute;top:-2px;width:19px}@keyframes maplibregl-user-location-dot-pulse{0%{opacity:1;transform:scale(1)}70%{opacity:0;transform:scale(3)}to{opacity:0;transform:scale(1)}}.maplibregl-user-location-dot-stale{background-color:#aaa}.maplibregl-user-location-dot-stale:after{display:none}.maplibregl-user-location-accuracy-circle{background-color:#1da1f233;border-radius:100%;height:1px;width:1px}.maplibregl-crosshair,.maplibregl-crosshair .maplibregl-interactive,.maplibregl-crosshair .maplibregl-interactive:active{cursor:crosshair}.maplibregl-boxzoom{background:#fff;border:2px dotted #202020;height:0;left:0;opacity:.5;position:absolute;top:0;width:0}.maplibregl-cooperative-gesture-screen{align-items:center;background:rgba(0,0,0,.4);color:#fff;display:flex;font-size:1.4em;inset:0;justify-content:center;line-height:1.2;opacity:0;padding:1rem;pointer-events:none;position:absolute;transition:opacity 1s ease 1s;z-index:99999}.maplibregl-cooperative-gesture-screen.maplibregl-show{opacity:1;transition:opacity .05s}.maplibregl-cooperative-gesture-screen .maplibregl-mobile-message{display:none}@media (hover:none),(width <= 480px){.maplibregl-cooperative-gesture-screen .maplibregl-desktop-message{display:none}.maplibregl-cooperative-gesture-screen .maplibregl-mobile-message{display:block}}.maplibregl-pseudo-fullscreen{height:100%!important;left:0!important;position:fixed!important;top:0!important;width:100%!important;z-index:99999} \ No newline at end of file diff --git a/web/libraries/maplibre-gl/dist/maplibre-gl.d.ts b/web/libraries/maplibre-gl/dist/maplibre-gl.d.ts new file mode 100644 index 00000000..7bb2791b --- /dev/null +++ b/web/libraries/maplibre-gl/dist/maplibre-gl.d.ts @@ -0,0 +1,11683 @@ +// Generated by dts-bundle-generator v8.1.2 + +import Point from '@mapbox/point-geometry'; +import TinySDF from '@mapbox/tiny-sdf'; +import { VectorTileFeature, VectorTileLayer } from '@mapbox/vector-tile'; +import { Color, CompositeExpression, Feature, FeatureFilter, FeatureState, FilterSpecification, Formatted, FormattedSection, GeoJSONSourceSpecification, GlobalProperties, ICanonicalTileID, IMercatorCoordinate, ImageSourceSpecification, InterpolationType, LayerSpecification, LightSpecification, Padding, PromoteIdSpecification, PropertyValueSpecification, RasterDEMSourceSpecification, RasterSourceSpecification, ResolvedImage, SourceExpression, SourceSpecification, SpriteSpecification, StylePropertyExpression, StylePropertySpecification, StyleSpecification, TerrainSpecification, TransitionSpecification, VariableAnchorOffsetCollection, VectorSourceSpecification, VideoSourceSpecification } from '@maplibre/maplibre-gl-style-spec'; +import { mat2, mat4, vec4 } from 'gl-matrix'; +import KDBush from 'kdbush'; +import { PotpackBox } from 'potpack'; +import { ClusterProperties, Options } from 'supercluster'; + +/** + * A request that can be cancelled + */ +export type Cancelable = { + cancel: () => void; +}; +/** + * A callback definition. + * + * @example + * ```ts + * asyncFunction((error, result) => { + * if (error) { + * // handle error + * } else if (result) { + * // handle success + * } + * }); + * ``` + */ +export type Callback = (error?: Error | null, result?: T | null) => void; +/** + * A `RequestParameters` object to be returned from Map.options.transformRequest callbacks. + * @example + * ```ts + * // use transformRequest to modify requests that begin with `http://myHost` + * transformRequest: function(url, resourceType) { + * if (resourceType === 'Source' && url.indexOf('http://myHost') > -1) { + * return { + * url: url.replace('http', 'https'), + * headers: { 'my-custom-header': true }, + * credentials: 'include' // Include cookies for cross-origin requests + * } + * } + * } + * ``` + */ +export type RequestParameters = { + /** + * The URL to be requested. + */ + url: string; + /** + * The headers to be sent with the request. + */ + headers?: any; + /** + * Request method `'GET' | 'POST' | 'PUT'`. + */ + method?: "GET" | "POST" | "PUT"; + /** + * Request body. + */ + body?: string; + /** + * Response body type to be returned `'string' | 'json' | 'arrayBuffer'`. + */ + type?: "string" | "json" | "arrayBuffer" | "image"; + /** + * `'same-origin'|'include'` Use 'include' to send cookies with cross-origin requests. + */ + credentials?: "same-origin" | "include"; + /** + * If `true`, Resource Timing API information will be collected for these transformed requests and returned in a resourceTiming property of relevant data events. + */ + collectResourceTiming?: boolean; + /** + * Parameters supported only by browser fetch API. Property of the Request interface contains the cache mode of the request. It controls how the request will interact with the browser's HTTP cache. (https://developer.mozilla.org/en-US/docs/Web/API/Request/cache) + */ + cache?: RequestCache; +}; +/** + * The response callback used in various places + */ +export type ResponseCallback = (error?: Error | null, data?: T | null, cacheControl?: string | null, expires?: string | null) => void; +export declare class AJAXError extends Error { + /** + * The response's HTTP status code. + */ + status: number; + /** + * The response's HTTP status text. + */ + statusText: string; + /** + * The request's URL. + */ + url: string; + /** + * The response's body. + */ + body: Blob; + /** + * @param status - The response's HTTP status code. + * @param statusText - The response's HTTP status text. + * @param url - The request's URL. + * @param body - The response's body. + */ + constructor(status: number, statusText: string, url: string, body: Blob); +} +/** + * A type used to store the tile's expiration date and cache control definition + */ +export type ExpiryData = { + cacheControl?: string | null; + expires?: Date | string | null; +}; +/** + * The callback that is being called after an image was fetched + */ +export type GetImageCallback = (error?: Error | null, image?: HTMLImageElement | ImageBitmap | null, expiry?: ExpiryData | null) => void; +/** + * A type of MapLibre resource. + */ +export declare const enum ResourceType { + Glyphs = "Glyphs", + Image = "Image", + Source = "Source", + SpriteImage = "SpriteImage", + SpriteJSON = "SpriteJSON", + Style = "Style", + Tile = "Tile", + Unknown = "Unknown" +} +/** + * This function is used to tranform a request. + * It is used just before executing the relevant request. + */ +export type RequestTransformFunction = (url: string, resourceType?: ResourceType) => RequestParameters | undefined; +export declare class RequestManager { + _transformRequestFn: RequestTransformFunction; + constructor(transformRequestFn?: RequestTransformFunction); + transformRequest(url: string, type: ResourceType): RequestParameters; + normalizeSpriteURL(url: string, format: string, extension: string): string; + setTransformRequest(transformRequest: RequestTransformFunction): void; +} +/** + * A listener method used as a callback to events + */ +export type Listener = (a: any) => any; +export type Listeners = { + [_: string]: Array; +}; +export declare class Event { + readonly type: string; + constructor(type: string, data?: any); +} +export declare class Evented { + _listeners: Listeners; + _oneTimeListeners: Listeners; + _eventedParent: Evented; + _eventedParentData: any | (() => any); + /** + * Adds a listener to a specified event type. + * + * @param type - The event type to add a listen for. + * @param listener - The function to be called when the event is fired. + * The listener function is called with the data object passed to `fire`, + * extended with `target` and `type` properties. + * @returns `this` + */ + on(type: string, listener: Listener): this; + /** + * Removes a previously registered event listener. + * + * @param type - The event type to remove listeners for. + * @param listener - The listener function to remove. + * @returns `this` + */ + off(type: string, listener: Listener): this; + /** + * Adds a listener that will be called only once to a specified event type. + * + * The listener will be called first time the event fires after the listener is registered. + * + * @param type - The event type to listen for. + * @param listener - The function to be called when the event is fired the first time. + * @returns `this` or a promise if a listener is not provided + */ + once(type: string, listener?: Listener): this | Promise; + fire(event: Event | string, properties?: any): this; + /** + * Returns a true if this instance of Evented or any forwardeed instances of Evented have a listener for the specified type. + * + * @param type - The event type + * @returns `true` if there is at least one registered listener for specified event type, `false` otherwise + */ + listens(type: string): boolean; + /** + * Bubble all events fired by this instance of Evented to this parent instance of Evented. + * @returns `this` + */ + setEventedParent(parent?: Evented | null, data?: any | (() => any)): this; +} +export declare class ZoomHistory { + lastZoom: number; + lastFloorZoom: number; + lastIntegerZoom: number; + lastIntegerZoomTime: number; + first: boolean; + constructor(); + update(z: number, now: number): boolean; +} +export type CrossfadeParameters = { + fromScale: number; + toScale: number; + t: number; +}; +export declare class EvaluationParameters { + zoom: number; + now: number; + fadeDuration: number; + zoomHistory: ZoomHistory; + transition: TransitionSpecification; + constructor(zoom: number, options?: any); + isSupportedScript(str: string): boolean; + crossFadingFactor(): number; + getCrossfadeParameters(): CrossfadeParameters; +} +/** + * A {@link LngLat} object, an array of two numbers representing longitude and latitude, + * or an object with `lng` and `lat` or `lon` and `lat` properties. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let v1 = new maplibregl.LngLat(-122.420679, 37.772537); + * let v2 = [-122.420679, 37.772537]; + * let v3 = {lon: -122.420679, lat: 37.772537}; + * ``` + */ +export type LngLatLike = LngLat | { + lng: number; + lat: number; +} | { + lon: number; + lat: number; +} | [ + number, + number +]; +export declare class LngLat { + lng: number; + lat: number; + /** + * @param lng - Longitude, measured in degrees. + * @param lat - Latitude, measured in degrees. + */ + constructor(lng: number, lat: number); + /** + * Returns a new `LngLat` object whose longitude is wrapped to the range (-180, 180). + * + * @returns The wrapped `LngLat` object. + * @example + * ```ts + * let ll = new maplibregl.LngLat(286.0251, 40.7736); + * let wrapped = ll.wrap(); + * wrapped.lng; // = -73.9749 + * ``` + */ + wrap(): LngLat; + /** + * Returns the coordinates represented as an array of two numbers. + * + * @returns The coordinates represented as an array of longitude and latitude. + * @example + * ```ts + * let ll = new maplibregl.LngLat(-73.9749, 40.7736); + * ll.toArray(); // = [-73.9749, 40.7736] + * ``` + */ + toArray(): [ + number, + number + ]; + /** + * Returns the coordinates represent as a string. + * + * @returns The coordinates represented as a string of the format `'LngLat(lng, lat)'`. + * @example + * ```ts + * let ll = new maplibregl.LngLat(-73.9749, 40.7736); + * ll.toString(); // = "LngLat(-73.9749, 40.7736)" + * ``` + */ + toString(): string; + /** + * Returns the approximate distance between a pair of coordinates in meters + * Uses the Haversine Formula (from R.W. Sinnott, "Virtues of the Haversine", Sky and Telescope, vol. 68, no. 2, 1984, p. 159) + * + * @param lngLat - coordinates to compute the distance to + * @returns Distance in meters between the two coordinates. + * @example + * ```ts + * let new_york = new maplibregl.LngLat(-74.0060, 40.7128); + * let los_angeles = new maplibregl.LngLat(-118.2437, 34.0522); + * new_york.distanceTo(los_angeles); // = 3935751.690893987, "true distance" using a non-spherical approximation is ~3966km + * ``` + */ + distanceTo(lngLat: LngLat): number; + /** + * Converts an array of two numbers or an object with `lng` and `lat` or `lon` and `lat` properties + * to a `LngLat` object. + * + * If a `LngLat` object is passed in, the function returns it unchanged. + * + * @param input - An array of two numbers or object to convert, or a `LngLat` object to return. + * @returns A new `LngLat` object, if a conversion occurred, or the original `LngLat` object. + * @example + * ```ts + * let arr = [-73.9749, 40.7736]; + * let ll = maplibregl.LngLat.convert(arr); + * ll; // = LngLat {lng: -73.9749, lat: 40.7736} + * ``` + */ + static convert(input: LngLatLike): LngLat; +} +export declare class MercatorCoordinate implements IMercatorCoordinate { + x: number; + y: number; + z: number; + /** + * @param x - The x component of the position. + * @param y - The y component of the position. + * @param z - The z component of the position. + */ + constructor(x: number, y: number, z?: number); + /** + * Project a `LngLat` to a `MercatorCoordinate`. + * + * @param lngLatLike - The location to project. + * @param altitude - The altitude in meters of the position. + * @returns The projected mercator coordinate. + * @example + * ```ts + * let coord = maplibregl.MercatorCoordinate.fromLngLat({ lng: 0, lat: 0}, 0); + * coord; // MercatorCoordinate(0.5, 0.5, 0) + * ``` + */ + static fromLngLat(lngLatLike: LngLatLike, altitude?: number): MercatorCoordinate; + /** + * Returns the `LngLat` for the coordinate. + * + * @returns The `LngLat` object. + * @example + * ```ts + * let coord = new maplibregl.MercatorCoordinate(0.5, 0.5, 0); + * let lngLat = coord.toLngLat(); // LngLat(0, 0) + * ``` + */ + toLngLat(): LngLat; + /** + * Returns the altitude in meters of the coordinate. + * + * @returns The altitude in meters. + * @example + * ```ts + * let coord = new maplibregl.MercatorCoordinate(0, 0, 0.02); + * coord.toAltitude(); // 6914.281956295339 + * ``` + */ + toAltitude(): number; + /** + * Returns the distance of 1 meter in `MercatorCoordinate` units at this latitude. + * + * For coordinates in real world units using meters, this naturally provides the scale + * to transform into `MercatorCoordinate`s. + * + * @returns Distance of 1 meter in `MercatorCoordinate` units. + */ + meterInMercatorCoordinateUnits(): number; +} +export declare class CanonicalTileID implements ICanonicalTileID { + z: number; + x: number; + y: number; + key: string; + constructor(z: number, x: number, y: number); + equals(id: ICanonicalTileID): boolean; + url(urls: Array, pixelRatio: number, scheme?: string | null): string; + isChildOf(parent: ICanonicalTileID): boolean; + getTilePoint(coord: IMercatorCoordinate): Point; + toString(): string; +} +export declare class UnwrappedTileID { + wrap: number; + canonical: CanonicalTileID; + key: string; + constructor(wrap: number, canonical: CanonicalTileID); +} +export declare class OverscaledTileID { + overscaledZ: number; + wrap: number; + canonical: CanonicalTileID; + key: string; + posMatrix: mat4; + constructor(overscaledZ: number, wrap: number, z: number, x: number, y: number); + clone(): OverscaledTileID; + equals(id: OverscaledTileID): boolean; + scaledTo(targetZ: number): OverscaledTileID; + calculateScaledKey(targetZ: number, withWrap: boolean): string; + isChildOf(parent: OverscaledTileID): boolean; + children(sourceMaxZoom: number): OverscaledTileID[]; + isLessThan(rhs: OverscaledTileID): boolean; + wrapped(): OverscaledTileID; + unwrapTo(wrap: number): OverscaledTileID; + overscaleFactor(): number; + toUnwrapped(): UnwrappedTileID; + toString(): string; + getTilePoint(coord: MercatorCoordinate): Point; +} +export type TimePoint = number; +/** + * A from-to type + */ +export type CrossFaded = { + to: T; + from: T; +}; +/** + * @internal + * Implementations of the `Property` interface: + * + * * Hold metadata about a property that's independent of any specific value: stuff like the type of the value, + * the default value, etc. This comes from the style specification JSON. + * * Define behavior that needs to be polymorphic across different properties: "possibly evaluating" + * an input value (see below), and interpolating between two possibly-evaluted values. + * + * The type `T` is the fully-evaluated value type (e.g. `number`, `string`, `Color`). + * The type `R` is the intermediate "possibly evaluated" value type. See below. + * + * There are two main implementations of the interface -- one for properties that allow data-driven values, + * and one for properties that don't. There are a few "special case" implementations as well: one for properties + * which cross-fade between two values rather than interpolating, one for `heatmap-color` and `line-gradient`, + * and one for `light-position`. + */ +export interface Property { + specification: StylePropertySpecification; + possiblyEvaluate(value: PropertyValue, parameters: EvaluationParameters, canonical?: CanonicalTileID, availableImages?: Array): R; + interpolate(a: R, b: R, t: number): R; +} +export declare class PropertyValue { + property: Property; + value: PropertyValueSpecification | void; + expression: StylePropertyExpression; + constructor(property: Property, value: PropertyValueSpecification | void); + isDataDriven(): boolean; + possiblyEvaluate(parameters: EvaluationParameters, canonical?: CanonicalTileID, availableImages?: Array): R; +} +export type TransitionParameters = { + now: TimePoint; + transition: TransitionSpecification; +}; +export declare class TransitionablePropertyValue { + property: Property; + value: PropertyValue; + transition: TransitionSpecification | void; + constructor(property: Property); + transitioned(parameters: TransitionParameters, prior: TransitioningPropertyValue): TransitioningPropertyValue; + untransitioned(): TransitioningPropertyValue; +} +export declare class Transitionable { + _properties: Properties; + _values: { + [K in keyof Props]: TransitionablePropertyValue; + }; + constructor(properties: Properties); + getValue(name: S): PropertyValueSpecification | void; + setValue(name: S, value: PropertyValueSpecification | void): void; + getTransition(name: S): TransitionSpecification | void; + setTransition(name: S, value: TransitionSpecification | void): void; + serialize(): any; + transitioned(parameters: TransitionParameters, prior: Transitioning): Transitioning; + untransitioned(): Transitioning; +} +export declare class TransitioningPropertyValue { + property: Property; + value: PropertyValue; + prior: TransitioningPropertyValue; + begin: TimePoint; + end: TimePoint; + constructor(property: Property, value: PropertyValue, prior: TransitioningPropertyValue, transition: TransitionSpecification, now: TimePoint); + possiblyEvaluate(parameters: EvaluationParameters, canonical: CanonicalTileID, availableImages: Array): R; +} +export declare class Transitioning { + _properties: Properties; + _values: { + [K in keyof Props]: PossiblyEvaluatedPropertyValue; + }; + constructor(properties: Properties); + possiblyEvaluate(parameters: EvaluationParameters, canonical?: CanonicalTileID, availableImages?: Array): PossiblyEvaluated; + hasTransition(): boolean; +} +export declare class Layout { + _properties: Properties; + _values: { + [K in keyof Props]: PropertyValue>; + }; + constructor(properties: Properties); + hasValue(name: S): boolean; + getValue(name: S): any; + setValue(name: S, value: any): void; + serialize(): any; + possiblyEvaluate(parameters: EvaluationParameters, canonical?: CanonicalTileID, availableImages?: Array): PossiblyEvaluated; +} +/** + * "Possibly evaluated value" is an intermediate stage in the evaluation chain for both paint and layout property + * values. The purpose of this stage is to optimize away unnecessary recalculations for data-driven properties. Code + * which uses data-driven property values must assume that the value is dependent on feature data, and request that it + * be evaluated for each feature. But when that property value is in fact a constant or camera function, the calculation + * will not actually depend on the feature, and we can benefit from returning the prior result of having done the + * evaluation once, ahead of time, in an intermediate step whose inputs are just the value and "global" parameters + * such as current zoom level. + * + * `PossiblyEvaluatedValue` represents the three possible outcomes of this step: if the input value was a constant or + * camera expression, then the "possibly evaluated" result is a constant value. Otherwise, the input value was either + * a source or composite expression, and we must defer final evaluation until supplied a feature. We separate + * the source and composite cases because they are handled differently when generating GL attributes, buffers, and + * uniforms. + * + * Note that `PossiblyEvaluatedValue` (and `PossiblyEvaluatedPropertyValue`, below) are _not_ used for properties that + * do not allow data-driven values. For such properties, we know that the "possibly evaluated" result is always a constant + * scalar value. See below. + */ +export type PossiblyEvaluatedValue = { + kind: "constant"; + value: T; +} | SourceExpression | CompositeExpression; +export declare class PossiblyEvaluatedPropertyValue { + property: DataDrivenProperty; + value: PossiblyEvaluatedValue; + parameters: EvaluationParameters; + constructor(property: DataDrivenProperty, value: PossiblyEvaluatedValue, parameters: EvaluationParameters); + isConstant(): boolean; + constantOr(value: T): T; + evaluate(feature: Feature, featureState: FeatureState, canonical?: CanonicalTileID, availableImages?: Array): T; +} +export declare class PossiblyEvaluated { + _properties: Properties; + _values: PossibleEvaluatedProps; + constructor(properties: Properties); + get(name: S): PossibleEvaluatedProps[S]; +} +export declare class DataConstantProperty implements Property { + specification: StylePropertySpecification; + constructor(specification: StylePropertySpecification); + possiblyEvaluate(value: PropertyValue, parameters: EvaluationParameters): T; + interpolate(a: T, b: T, t: number): T; +} +export declare class DataDrivenProperty implements Property> { + specification: StylePropertySpecification; + overrides: any; + constructor(specification: StylePropertySpecification, overrides?: any); + possiblyEvaluate(value: PropertyValue>, parameters: EvaluationParameters, canonical?: CanonicalTileID, availableImages?: Array): PossiblyEvaluatedPropertyValue; + interpolate(a: PossiblyEvaluatedPropertyValue, b: PossiblyEvaluatedPropertyValue, t: number): PossiblyEvaluatedPropertyValue; + evaluate(value: PossiblyEvaluatedValue, parameters: EvaluationParameters, feature: Feature, featureState: FeatureState, canonical?: CanonicalTileID, availableImages?: Array): T; +} +export declare class CrossFadedDataDrivenProperty extends DataDrivenProperty> { + possiblyEvaluate(value: PropertyValue, PossiblyEvaluatedPropertyValue>>, parameters: EvaluationParameters, canonical?: CanonicalTileID, availableImages?: Array): PossiblyEvaluatedPropertyValue>; + evaluate(value: PossiblyEvaluatedValue>, globals: EvaluationParameters, feature: Feature, featureState: FeatureState, canonical?: CanonicalTileID, availableImages?: Array): CrossFaded; + _calculate(min: T, mid: T, max: T, parameters: EvaluationParameters): CrossFaded; + interpolate(a: PossiblyEvaluatedPropertyValue>): PossiblyEvaluatedPropertyValue>; +} +export declare class CrossFadedProperty implements Property> { + specification: StylePropertySpecification; + constructor(specification: StylePropertySpecification); + possiblyEvaluate(value: PropertyValue>, parameters: EvaluationParameters, canonical?: CanonicalTileID, availableImages?: Array): CrossFaded; + _calculate(min: T, mid: T, max: T, parameters: EvaluationParameters): CrossFaded; + interpolate(a?: CrossFaded | null): CrossFaded; +} +export declare class ColorRampProperty implements Property { + specification: StylePropertySpecification; + constructor(specification: StylePropertySpecification); + possiblyEvaluate(value: PropertyValue, parameters: EvaluationParameters, canonical?: CanonicalTileID, availableImages?: Array): boolean; + interpolate(): boolean; +} +export declare class Properties { + properties: Props; + defaultPropertyValues: { + [K in keyof Props]: PropertyValue; + }; + defaultTransitionablePropertyValues: { + [K in keyof Props]: TransitionablePropertyValue; + }; + defaultTransitioningPropertyValues: { + [K in keyof Props]: TransitioningPropertyValue; + }; + defaultPossiblyEvaluatedValues: { + [K in keyof Props]: PossiblyEvaluatedPropertyValue; + }; + overridableProperties: Array; + constructor(properties: Props); +} +/** + * An object with is transferable between main and worker thread + */ +export type Transferable = ArrayBuffer | MessagePort | ImageBitmap; +declare const viewTypes: { + Int8: Int8ArrayConstructor; + Uint8: Uint8ArrayConstructor; + Int16: Int16ArrayConstructor; + Uint16: Uint16ArrayConstructor; + Int32: Int32ArrayConstructor; + Uint32: Uint32ArrayConstructor; + Float32: Float32ArrayConstructor; +}; +/** + * @internal + * A view type size + */ +export type ViewType = keyof typeof viewTypes; +export declare class Struct { + _pos1: number; + _pos2: number; + _pos4: number; + _pos8: number; + readonly _structArray: StructArray; + size: number; + /** + * @param structArray - The StructArray the struct is stored in + * @param index - The index of the struct in the StructArray. + */ + constructor(structArray: StructArray, index: number); +} +/** + * @internal + * A struct array memeber + */ +export type StructArrayMember = { + name: string; + type: ViewType; + components: number; + offset: number; +}; +/** + * An array that can be desialized + */ +export type SerializedStructArray = { + length: number; + arrayBuffer: ArrayBuffer; +}; +declare abstract class StructArray { + capacity: number; + length: number; + isTransferred: boolean; + arrayBuffer: ArrayBuffer; + uint8: Uint8Array; + members: Array; + bytesPerElement: number; + abstract emplaceBack(...v: number[]): any; + abstract emplace(i: number, ...v: number[]): any; + constructor(); + /** + * Serialize a StructArray instance. Serializes both the raw data and the + * metadata needed to reconstruct the StructArray base class during + * deserialization. + */ + static serialize(array: StructArray, transferables?: Array): SerializedStructArray; + static deserialize(input: SerializedStructArray): any; + /** + * Resize the array to discard unused capacity. + */ + _trim(): void; + /** + * Resets the length of the array to 0 without de-allocating capcacity. + */ + clear(): void; + /** + * Resize the array. + * If `n` is greater than the current length then additional elements with undefined values are added. + * If `n` is less than the current length then the array will be reduced to the first `n` elements. + * @param n - The new size of the array. + */ + resize(n: number): void; + /** + * Indicate a planned increase in size, so that any necessary allocation may + * be done once, ahead of time. + * @param n - The expected size of the array. + */ + reserve(n: number): void; + /** + * Create TypedArray views for the current ArrayBuffer. + */ + _refreshViews(): void; +} +export declare class StructArrayLayout2i4 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number): number; + emplace(i: number, v0: number, v1: number): number; +} +export declare class StructArrayLayout3i6 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number): number; + emplace(i: number, v0: number, v1: number, v2: number): number; +} +export declare class StructArrayLayout4i8 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number, v3: number): number; + emplace(i: number, v0: number, v1: number, v2: number, v3: number): number; +} +export declare class StructArrayLayout2i4i12 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number): number; + emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number): number; +} +export declare class StructArrayLayout2i4ub8 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number): number; + emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number): number; +} +export declare class StructArrayLayout2f8 extends StructArray { + uint8: Uint8Array; + float32: Float32Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number): number; + emplace(i: number, v0: number, v1: number): number; +} +export declare class StructArrayLayout4i4ui4i24 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + uint16: Uint16Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number): number; + emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number): number; +} +export declare class StructArrayLayout3f12 extends StructArray { + uint8: Uint8Array; + float32: Float32Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number): number; + emplace(i: number, v0: number, v1: number, v2: number): number; +} +export declare class StructArrayLayout1ul4 extends StructArray { + uint8: Uint8Array; + uint32: Uint32Array; + _refreshViews(): void; + emplaceBack(v0: number): number; + emplace(i: number, v0: number): number; +} +export declare class StructArrayLayout6i1ul2ui20 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + uint32: Uint32Array; + uint16: Uint16Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number): number; + emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number): number; +} +export declare class StructArrayLayout2ub2f12 extends StructArray { + uint8: Uint8Array; + float32: Float32Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number, v3: number): number; + emplace(i: number, v0: number, v1: number, v2: number, v3: number): number; +} +export declare class StructArrayLayout3ui6 extends StructArray { + uint8: Uint8Array; + uint16: Uint16Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number): number; + emplace(i: number, v0: number, v1: number, v2: number): number; +} +export declare class StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + uint16: Uint16Array; + uint32: Uint32Array; + float32: Float32Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number): number; + emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number): number; +} +export declare class StructArrayLayout8i15ui1ul2f2ui64 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + uint16: Uint16Array; + uint32: Uint32Array; + float32: Float32Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number, v17: number, v18: number, v19: number, v20: number, v21: number, v22: number, v23: number, v24: number, v25: number, v26: number, v27: number): number; + emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number, v17: number, v18: number, v19: number, v20: number, v21: number, v22: number, v23: number, v24: number, v25: number, v26: number, v27: number): number; +} +export declare class StructArrayLayout1f4 extends StructArray { + uint8: Uint8Array; + float32: Float32Array; + _refreshViews(): void; + emplaceBack(v0: number): number; + emplace(i: number, v0: number): number; +} +export declare class StructArrayLayout1ui2f12 extends StructArray { + uint8: Uint8Array; + uint16: Uint16Array; + float32: Float32Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number): number; + emplace(i: number, v0: number, v1: number, v2: number): number; +} +export declare class StructArrayLayout1ul2ui8 extends StructArray { + uint8: Uint8Array; + uint32: Uint32Array; + uint16: Uint16Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number, v2: number): number; + emplace(i: number, v0: number, v1: number, v2: number): number; +} +export declare class StructArrayLayout2ui4 extends StructArray { + uint8: Uint8Array; + uint16: Uint16Array; + _refreshViews(): void; + emplaceBack(v0: number, v1: number): number; + emplace(i: number, v0: number, v1: number): number; +} +export declare class StructArrayLayout1ui2 extends StructArray { + uint8: Uint8Array; + uint16: Uint16Array; + _refreshViews(): void; + emplaceBack(v0: number): number; + emplace(i: number, v0: number): number; +} +export declare class CollisionBoxStruct extends Struct { + _structArray: CollisionBoxArray; + get anchorPointX(): number; + get anchorPointY(): number; + get x1(): number; + get y1(): number; + get x2(): number; + get y2(): number; + get featureIndex(): number; + get sourceLayerIndex(): number; + get bucketIndex(): number; + get anchorPoint(): Point; +} +export declare class CollisionBoxArray extends StructArrayLayout6i1ul2ui20 { + /** + * Return the CollisionBoxStruct at the given location in the array. + * @param index The index of the element. + */ + get(index: number): CollisionBoxStruct; +} +export declare class PlacedSymbolStruct extends Struct { + _structArray: PlacedSymbolArray; + get anchorX(): number; + get anchorY(): number; + get glyphStartIndex(): number; + get numGlyphs(): number; + get vertexStartIndex(): number; + get lineStartIndex(): number; + get lineLength(): number; + get segment(): number; + get lowerSize(): number; + get upperSize(): number; + get lineOffsetX(): number; + get lineOffsetY(): number; + get writingMode(): number; + get placedOrientation(): number; + set placedOrientation(x: number); + get hidden(): number; + set hidden(x: number); + get crossTileID(): number; + set crossTileID(x: number); + get associatedIconIndex(): number; +} +export declare class PlacedSymbolArray extends StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 { + /** + * Return the PlacedSymbolStruct at the given location in the array. + * @param index The index of the element. + */ + get(index: number): PlacedSymbolStruct; +} +export declare class SymbolInstanceStruct extends Struct { + _structArray: SymbolInstanceArray; + get anchorX(): number; + get anchorY(): number; + get rightJustifiedTextSymbolIndex(): number; + get centerJustifiedTextSymbolIndex(): number; + get leftJustifiedTextSymbolIndex(): number; + get verticalPlacedTextSymbolIndex(): number; + get placedIconSymbolIndex(): number; + get verticalPlacedIconSymbolIndex(): number; + get key(): number; + get textBoxStartIndex(): number; + get textBoxEndIndex(): number; + get verticalTextBoxStartIndex(): number; + get verticalTextBoxEndIndex(): number; + get iconBoxStartIndex(): number; + get iconBoxEndIndex(): number; + get verticalIconBoxStartIndex(): number; + get verticalIconBoxEndIndex(): number; + get featureIndex(): number; + get numHorizontalGlyphVertices(): number; + get numVerticalGlyphVertices(): number; + get numIconVertices(): number; + get numVerticalIconVertices(): number; + get useRuntimeCollisionCircles(): number; + get crossTileID(): number; + set crossTileID(x: number); + get textBoxScale(): number; + get collisionCircleDiameter(): number; + get textAnchorOffsetStartIndex(): number; + get textAnchorOffsetEndIndex(): number; +} +export type SymbolInstance = SymbolInstanceStruct; +export declare class SymbolInstanceArray extends StructArrayLayout8i15ui1ul2f2ui64 { + /** + * Return the SymbolInstanceStruct at the given location in the array. + * @param index The index of the element. + */ + get(index: number): SymbolInstanceStruct; +} +export declare class GlyphOffsetArray extends StructArrayLayout1f4 { + getoffsetX(index: number): number; +} +export declare class SymbolLineVertexArray extends StructArrayLayout3i6 { + getx(index: number): number; + gety(index: number): number; + gettileUnitDistanceFromAnchor(index: number): number; +} +export declare class TextAnchorOffsetStruct extends Struct { + _structArray: TextAnchorOffsetArray; + get textAnchor(): number; + get textOffset0(): number; + get textOffset1(): number; +} +export type TextAnchorOffset = TextAnchorOffsetStruct; +export declare class TextAnchorOffsetArray extends StructArrayLayout1ui2f12 { + /** + * Return the TextAnchorOffsetStruct at the given location in the array. + * @param index The index of the element. + */ + get(index: number): TextAnchorOffsetStruct; +} +export declare class FeatureIndexStruct extends Struct { + _structArray: FeatureIndexArray; + get featureIndex(): number; + get sourceLayerIndex(): number; + get bucketIndex(): number; +} +export declare class FeatureIndexArray extends StructArrayLayout1ul2ui8 { + /** + * Return the FeatureIndexStruct at the given location in the array. + * @param index The index of the element. + */ + get(index: number): FeatureIndexStruct; +} +export declare class PosArray extends StructArrayLayout2i4 { +} +export declare class RasterBoundsArray extends StructArrayLayout4i8 { +} +export declare class CircleLayoutArray extends StructArrayLayout2i4 { +} +export declare class FillLayoutArray extends StructArrayLayout2i4 { +} +export declare class FillExtrusionLayoutArray extends StructArrayLayout2i4i12 { +} +export declare class LineLayoutArray extends StructArrayLayout2i4ub8 { +} +export declare class LineExtLayoutArray extends StructArrayLayout2f8 { +} +export declare class SymbolLayoutArray extends StructArrayLayout4i4ui4i24 { +} +export declare class SymbolDynamicLayoutArray extends StructArrayLayout3f12 { +} +export declare class SymbolOpacityArray extends StructArrayLayout1ul4 { +} +export declare class CollisionVertexArray extends StructArrayLayout2ub2f12 { +} +export declare class TriangleIndexArray extends StructArrayLayout3ui6 { +} +export declare class LineIndexArray extends StructArrayLayout2ui4 { +} +export declare class LineStripIndexArray extends StructArrayLayout1ui2 { +} +export type SerializedFeaturePositionMap = { + ids: Float64Array; + positions: Uint32Array; +}; +export type FeaturePosition = { + index: number; + start: number; + end: number; +}; +export declare class FeaturePositionMap { + ids: Array; + positions: Array; + indexed: boolean; + constructor(); + add(id: unknown, index: number, start: number, end: number): void; + getPositions(id: unknown): Array; + static serialize(map: FeaturePositionMap, transferables: Array): SerializedFeaturePositionMap; + static deserialize(obj: SerializedFeaturePositionMap): FeaturePositionMap; +} +export declare class IndexBuffer { + context: Context; + buffer: WebGLBuffer; + dynamicDraw: boolean; + constructor(context: Context, array: TriangleIndexArray | LineIndexArray | LineStripIndexArray, dynamicDraw?: boolean); + bind(): void; + updateData(array: StructArray): void; + destroy(): void; +} +export declare class VertexBuffer { + length: number; + attributes: ReadonlyArray; + itemSize: number; + dynamicDraw: boolean; + context: Context; + buffer: WebGLBuffer; + /** + * @param dynamicDraw - Whether this buffer will be repeatedly updated. + */ + constructor(context: Context, array: StructArray, attributes: ReadonlyArray, dynamicDraw?: boolean); + bind(): void; + updateData(array: StructArray): void; + enableAttributes(gl: WebGLRenderingContext | WebGL2RenderingContext, program: Program): void; + /** + * Set the attribute pointers in a WebGL context + * @param gl - The WebGL context + * @param program - The active WebGL program + * @param vertexOffset - Index of the starting vertex of the segment + */ + setVertexAttribPointers(gl: WebGLRenderingContext | WebGL2RenderingContext, program: Program, vertexOffset?: number | null): void; + /** + * Destroy the GL buffer bound to the given WebGL context + */ + destroy(): void; +} +export type BlendFuncConstant = WebGLRenderingContextBase["ZERO"] | WebGLRenderingContextBase["ONE"] | WebGLRenderingContextBase["SRC_COLOR"] | WebGLRenderingContextBase["ONE_MINUS_SRC_COLOR"] | WebGLRenderingContextBase["DST_COLOR"] | WebGLRenderingContextBase["ONE_MINUS_DST_COLOR"] | WebGLRenderingContextBase["SRC_ALPHA"] | WebGLRenderingContextBase["ONE_MINUS_SRC_ALPHA"] | WebGLRenderingContextBase["DST_ALPHA"] | WebGLRenderingContextBase["ONE_MINUS_DST_ALPHA"] | WebGLRenderingContextBase["CONSTANT_COLOR"] | WebGLRenderingContextBase["ONE_MINUS_CONSTANT_COLOR"] | WebGLRenderingContextBase["CONSTANT_ALPHA"] | WebGLRenderingContextBase["ONE_MINUS_CONSTANT_ALPHA"] | WebGLRenderingContextBase["BLEND_COLOR"]; +export type BlendFuncType = [ + BlendFuncConstant, + BlendFuncConstant +]; +export type BlendEquationType = WebGLRenderingContextBase["FUNC_ADD"] | WebGLRenderingContextBase["FUNC_SUBTRACT"] | WebGLRenderingContextBase["FUNC_REVERSE_SUBTRACT"]; +export type ColorMaskType = [ + boolean, + boolean, + boolean, + boolean +]; +export type CompareFuncType = WebGLRenderingContextBase["NEVER"] | WebGLRenderingContextBase["LESS"] | WebGLRenderingContextBase["EQUAL"] | WebGLRenderingContextBase["LEQUAL"] | WebGLRenderingContextBase["GREATER"] | WebGLRenderingContextBase["NOTEQUAL"] | WebGLRenderingContextBase["GEQUAL"] | WebGLRenderingContextBase["ALWAYS"]; +export type DepthMaskType = boolean; +export type DepthRangeType = [ + number, + number +]; +export type DepthFuncType = CompareFuncType; +export type StencilFuncType = { + func: CompareFuncType; + ref: number; + mask: number; +}; +export type StencilOpConstant = WebGLRenderingContextBase["KEEP"] | WebGLRenderingContextBase["ZERO"] | WebGLRenderingContextBase["REPLACE"] | WebGLRenderingContextBase["INCR"] | WebGLRenderingContextBase["INCR_WRAP"] | WebGLRenderingContextBase["DECR"] | WebGLRenderingContextBase["DECR_WRAP"] | WebGLRenderingContextBase["INVERT"]; +export type StencilOpType = [ + StencilOpConstant, + StencilOpConstant, + StencilOpConstant +]; +export type TextureUnitType = number; +export type ViewportType = [ + number, + number, + number, + number +]; +export type StencilTestGL = { + func: WebGLRenderingContextBase["NEVER"]; + mask: 0; +} | { + func: WebGLRenderingContextBase["LESS"]; + mask: number; +} | { + func: WebGLRenderingContextBase["EQUAL"]; + mask: number; +} | { + func: WebGLRenderingContextBase["LEQUAL"]; + mask: number; +} | { + func: WebGLRenderingContextBase["GREATER"]; + mask: number; +} | { + func: WebGLRenderingContextBase["NOTEQUAL"]; + mask: number; +} | { + func: WebGLRenderingContextBase["GEQUAL"]; + mask: number; +} | { + func: WebGLRenderingContextBase["ALWAYS"]; + mask: 0; +}; +export type CullFaceModeType = WebGLRenderingContextBase["FRONT"] | WebGLRenderingContextBase["BACK"] | WebGLRenderingContextBase["FRONT_AND_BACK"]; +export type FrontFaceType = WebGLRenderingContextBase["CW"] | WebGLRenderingContextBase["CCW"]; +export interface IValue { + current: T; + default: T; + dirty: boolean; + get(): T; + setDefault(): void; + set(value: T): void; +} +export declare class BaseValue implements IValue { + gl: WebGLRenderingContext | WebGL2RenderingContext; + current: T; + default: T; + dirty: boolean; + constructor(context: Context); + get(): T; + set(value: T): void; + getDefault(): T; + setDefault(): void; +} +export declare class ClearColor extends BaseValue { + getDefault(): Color; + set(v: Color): void; +} +export declare class ClearDepth extends BaseValue { + getDefault(): number; + set(v: number): void; +} +export declare class ClearStencil extends BaseValue { + getDefault(): number; + set(v: number): void; +} +export declare class ColorMask extends BaseValue { + getDefault(): ColorMaskType; + set(v: ColorMaskType): void; +} +export declare class DepthMask extends BaseValue { + getDefault(): DepthMaskType; + set(v: DepthMaskType): void; +} +export declare class StencilMask extends BaseValue { + getDefault(): number; + set(v: number): void; +} +export declare class StencilFunc extends BaseValue { + getDefault(): StencilFuncType; + set(v: StencilFuncType): void; +} +export declare class StencilOp extends BaseValue { + getDefault(): StencilOpType; + set(v: StencilOpType): void; +} +export declare class StencilTest extends BaseValue { + getDefault(): boolean; + set(v: boolean): void; +} +export declare class DepthRange extends BaseValue { + getDefault(): DepthRangeType; + set(v: DepthRangeType): void; +} +export declare class DepthTest extends BaseValue { + getDefault(): boolean; + set(v: boolean): void; +} +export declare class DepthFunc extends BaseValue { + getDefault(): DepthFuncType; + set(v: DepthFuncType): void; +} +export declare class Blend extends BaseValue { + getDefault(): boolean; + set(v: boolean): void; +} +export declare class BlendFunc extends BaseValue { + getDefault(): BlendFuncType; + set(v: BlendFuncType): void; +} +export declare class BlendColor extends BaseValue { + getDefault(): Color; + set(v: Color): void; +} +export declare class BlendEquation extends BaseValue { + getDefault(): BlendEquationType; + set(v: BlendEquationType): void; +} +export declare class CullFace extends BaseValue { + getDefault(): boolean; + set(v: boolean): void; +} +export declare class CullFaceSide extends BaseValue { + getDefault(): CullFaceModeType; + set(v: CullFaceModeType): void; +} +export declare class FrontFace extends BaseValue { + getDefault(): FrontFaceType; + set(v: FrontFaceType): void; +} +export declare class ProgramValue extends BaseValue { + getDefault(): WebGLProgram; + set(v?: WebGLProgram | null): void; +} +export declare class ActiveTextureUnit extends BaseValue { + getDefault(): TextureUnitType; + set(v: TextureUnitType): void; +} +export declare class Viewport extends BaseValue { + getDefault(): ViewportType; + set(v: ViewportType): void; +} +export declare class BindFramebuffer extends BaseValue { + getDefault(): WebGLFramebuffer; + set(v?: WebGLFramebuffer | null): void; +} +export declare class BindRenderbuffer extends BaseValue { + getDefault(): WebGLRenderbuffer; + set(v?: WebGLRenderbuffer | null): void; +} +export declare class BindTexture extends BaseValue { + getDefault(): WebGLTexture; + set(v?: WebGLTexture | null): void; +} +export declare class BindVertexBuffer extends BaseValue { + getDefault(): WebGLBuffer; + set(v?: WebGLBuffer | null): void; +} +export declare class BindElementBuffer extends BaseValue { + getDefault(): WebGLBuffer; + set(v?: WebGLBuffer | null): void; +} +export declare class BindVertexArray extends BaseValue { + getDefault(): WebGLVertexArrayObject | null; + set(v: WebGLVertexArrayObject | null): void; +} +export declare class PixelStoreUnpack extends BaseValue { + getDefault(): number; + set(v: number): void; +} +export declare class PixelStoreUnpackPremultiplyAlpha extends BaseValue { + getDefault(): boolean; + set(v: boolean): void; +} +export declare class PixelStoreUnpackFlipY extends BaseValue { + getDefault(): boolean; + set(v: boolean): void; +} +export declare class FramebufferAttachment extends BaseValue { + parent: WebGLFramebuffer; + context: Context; + constructor(context: Context, parent: WebGLFramebuffer); + getDefault(): any; +} +export declare class ColorAttachment extends FramebufferAttachment { + setDirty(): void; + set(v?: WebGLTexture | null): void; +} +export declare class DepthAttachment extends FramebufferAttachment { + set(v?: WebGLRenderbuffer | null): void; +} +export declare class Framebuffer { + context: Context; + width: number; + height: number; + framebuffer: WebGLFramebuffer; + colorAttachment: ColorAttachment; + depthAttachment: DepthAttachment; + constructor(context: Context, width: number, height: number, hasDepth: boolean, hasStencil: boolean); + destroy(): void; +} +export declare class DepthMode { + func: DepthFuncType; + mask: DepthMaskType; + range: DepthRangeType; + static ReadOnly: boolean; + static ReadWrite: boolean; + constructor(depthFunc: DepthFuncType, depthMask: DepthMaskType, depthRange: DepthRangeType); + static disabled: Readonly; +} +export declare class StencilMode { + test: StencilTestGL; + ref: number; + mask: number; + fail: StencilOpConstant; + depthFail: StencilOpConstant; + pass: StencilOpConstant; + constructor(test: StencilTestGL, ref: number, mask: number, fail: StencilOpConstant, depthFail: StencilOpConstant, pass: StencilOpConstant); + static disabled: Readonly; +} +export declare class ColorMode { + blendFunction: BlendFuncType; + blendColor: Color; + mask: ColorMaskType; + constructor(blendFunction: BlendFuncType, blendColor: Color, mask: ColorMaskType); + static Replace: BlendFuncType; + static disabled: Readonly; + static unblended: Readonly; + static alphaBlended: Readonly; +} +export declare class CullFaceMode { + enable: boolean; + mode: CullFaceModeType; + frontFace: FrontFaceType; + constructor(enable: boolean, mode: CullFaceModeType, frontFace: FrontFaceType); + static disabled: Readonly; + static backCCW: Readonly; +} +export type ClearArgs = { + color?: Color; + depth?: number; + stencil?: number; +}; +export declare class Context { + gl: WebGLRenderingContext | WebGL2RenderingContext; + currentNumAttributes: number; + maxTextureSize: number; + clearColor: ClearColor; + clearDepth: ClearDepth; + clearStencil: ClearStencil; + colorMask: ColorMask; + depthMask: DepthMask; + stencilMask: StencilMask; + stencilFunc: StencilFunc; + stencilOp: StencilOp; + stencilTest: StencilTest; + depthRange: DepthRange; + depthTest: DepthTest; + depthFunc: DepthFunc; + blend: Blend; + blendFunc: BlendFunc; + blendColor: BlendColor; + blendEquation: BlendEquation; + cullFace: CullFace; + cullFaceSide: CullFaceSide; + frontFace: FrontFace; + program: ProgramValue; + activeTexture: ActiveTextureUnit; + viewport: Viewport; + bindFramebuffer: BindFramebuffer; + bindRenderbuffer: BindRenderbuffer; + bindTexture: BindTexture; + bindVertexBuffer: BindVertexBuffer; + bindElementBuffer: BindElementBuffer; + bindVertexArray: BindVertexArray; + pixelStoreUnpack: PixelStoreUnpack; + pixelStoreUnpackPremultiplyAlpha: PixelStoreUnpackPremultiplyAlpha; + pixelStoreUnpackFlipY: PixelStoreUnpackFlipY; + extTextureFilterAnisotropic: EXT_texture_filter_anisotropic | null; + extTextureFilterAnisotropicMax?: GLfloat; + HALF_FLOAT?: GLenum; + RGBA16F?: GLenum; + RGB16F?: GLenum; + constructor(gl: WebGLRenderingContext | WebGL2RenderingContext); + setDefault(): void; + setDirty(): void; + createIndexBuffer(array: TriangleIndexArray | LineIndexArray | LineStripIndexArray, dynamicDraw?: boolean): IndexBuffer; + createVertexBuffer(array: StructArray, attributes: ReadonlyArray, dynamicDraw?: boolean): VertexBuffer; + createRenderbuffer(storageFormat: number, width: number, height: number): WebGLRenderbuffer; + createFramebuffer(width: number, height: number, hasDepth: boolean, hasStencil: boolean): Framebuffer; + clear({ color, depth, stencil }: ClearArgs): void; + setCullFace(cullFaceMode: Readonly): void; + setDepthMode(depthMode: Readonly): void; + setStencilMode(stencilMode: Readonly): void; + setColorMode(colorMode: Readonly): void; + createVertexArray(): WebGLVertexArrayObject | undefined; + deleteVertexArray(x: WebGLVertexArrayObject | undefined): void; + unbindVAO(): void; +} +export type $ObjMap any> = { + [K in keyof T]: F extends (v: T[K]) => infer R ? R : never; +}; +export type UniformValues = $ObjMap(u: Uniform) => V>; +export type UniformLocations = { + [_: string]: WebGLUniformLocation; +}; +declare abstract class Uniform { + gl: WebGLRenderingContext | WebGL2RenderingContext; + location: WebGLUniformLocation; + current: T; + constructor(context: Context, location: WebGLUniformLocation); + abstract set(v: T): void; +} +export declare class Uniform1i extends Uniform { + constructor(context: Context, location: WebGLUniformLocation); + set(v: number): void; +} +export declare class Uniform1f extends Uniform { + constructor(context: Context, location: WebGLUniformLocation); + set(v: number): void; +} +export declare class Uniform4f extends Uniform { + constructor(context: Context, location: WebGLUniformLocation); + set(v: vec4): void; +} +export declare class UniformMatrix4f extends Uniform { + constructor(context: Context, location: WebGLUniformLocation); + set(v: mat4): void; +} +/** + * @internal + * A uniform bindings + */ +export type UniformBindings = { + [_: string]: Uniform; +}; +export type Size = { + width: number; + height: number; +}; +export type Point2D = { + x: number; + y: number; +}; +export declare class AlphaImage { + width: number; + height: number; + data: Uint8Array; + constructor(size: Size, data?: Uint8Array | Uint8ClampedArray); + resize(size: Size): void; + clone(): AlphaImage; + static copy(srcImg: AlphaImage, dstImg: AlphaImage, srcPt: Point2D, dstPt: Point2D, size: Size): void; +} +export declare class RGBAImage { + width: number; + height: number; + /** + * data must be a Uint8Array instead of Uint8ClampedArray because texImage2D does not support Uint8ClampedArray in all browsers. + */ + data: Uint8Array; + constructor(size: Size, data?: Uint8Array | Uint8ClampedArray); + resize(size: Size): void; + replace(data: Uint8Array | Uint8ClampedArray, copy?: boolean): void; + clone(): RGBAImage; + static copy(srcImg: RGBAImage | ImageData, dstImg: RGBAImage, srcPt: Point2D, dstPt: Point2D, size: Size): void; +} +/** + * The sprite data + */ +export type SpriteOnDemandStyleImage = { + width: number; + height: number; + x: number; + y: number; + context: CanvasRenderingContext2D; +}; +/** + * The style's image metadata + */ +export type StyleImageData = { + data: RGBAImage; + version?: number; + hasRenderCallback?: boolean; + userImage?: StyleImageInterface; + spriteData?: SpriteOnDemandStyleImage; +}; +/** + * The style's image metadata + */ +export type StyleImageMetadata = { + /** + * The ratio of pixels in the image to physical pixels on the screen + */ + pixelRatio: number; + /** + * Whether the image should be interpreted as an SDF image + */ + sdf: boolean; + /** + * If `icon-text-fit` is used in a layer with this image, this option defines the part(s) of the image that can be stretched horizontally. + */ + stretchX?: Array<[ + number, + number + ]>; + /** + * If `icon-text-fit` is used in a layer with this image, this option defines the part(s) of the image that can be stretched vertically. + */ + stretchY?: Array<[ + number, + number + ]>; + /** + * If `icon-text-fit` is used in a layer with this image, this option defines the part of the image that can be covered by the content in `text-field`. + */ + content?: [ + number, + number, + number, + number + ]; +}; +/** + * the style's image, including data and metedata + */ +export type StyleImage = StyleImageData & StyleImageMetadata; +/** + * Interface for dynamically generated style images. This is a specification for + * implementers to model: it is not an exported method or class. + * + * Images implementing this interface can be redrawn for every frame. They can be used to animate + * icons and patterns or make them respond to user input. Style images can implement a + * {@link StyleImageInterface#render} method. The method is called every frame and + * can be used to update the image. + * + * @see [Add an animated icon to the map.](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-animated/) + * + * @example + * ```ts + * let flashingSquare = { + * width: 64, + * height: 64, + * data: new Uint8Array(64 * 64 * 4), + * + * onAdd: function(map) { + * this.map = map; + * }, + * + * render: function() { + * // keep repainting while the icon is on the map + * this.map.triggerRepaint(); + * + * // alternate between black and white based on the time + * let value = Math.round(Date.now() / 1000) % 2 === 0 ? 255 : 0; + * + * // check if image needs to be changed + * if (value !== this.previousValue) { + * this.previousValue = value; + * + * let bytesPerPixel = 4; + * for (let x = 0; x < this.width; x++) { + * for (let y = 0; y < this.height; y++) { + * let offset = (y * this.width + x) * bytesPerPixel; + * this.data[offset + 0] = value; + * this.data[offset + 1] = value; + * this.data[offset + 2] = value; + * this.data[offset + 3] = 255; + * } + * } + * + * // return true to indicate that the image changed + * return true; + * } + * } + * } + * + * map.addImage('flashing_square', flashingSquare); + * ``` + */ +export interface StyleImageInterface { + width: number; + height: number; + data: Uint8Array | Uint8ClampedArray; + /** + * This method is called once before every frame where the icon will be used. + * The method can optionally update the image's `data` member with a new image. + * + * If the method updates the image it must return `true` to commit the change. + * If the method returns `false` or nothing the image is assumed to not have changed. + * + * If updates are infrequent it maybe easier to use {@link Map#updateImage} to update + * the image instead of implementing this method. + * + * @returns `true` if this method updated the image. `false` if the image was not changed. + */ + render?: () => boolean; + /** + * Optional method called when the layer has been added to the Map with {@link Map#addImage}. + * + * @param map - The Map this custom layer was just added to. + */ + onAdd?: (map: Map, id: string) => void; + /** + * Optional method called when the icon is removed from the map with {@link Map#removeImage}. + * This gives the image a chance to clean up resources and event listeners. + */ + onRemove?: () => void; +} +export type TextureFormat = WebGLRenderingContextBase["RGBA"] | WebGLRenderingContextBase["ALPHA"]; +export type TextureFilter = WebGLRenderingContextBase["LINEAR"] | WebGLRenderingContextBase["LINEAR_MIPMAP_NEAREST"] | WebGLRenderingContextBase["NEAREST"]; +export type TextureWrap = WebGLRenderingContextBase["REPEAT"] | WebGLRenderingContextBase["CLAMP_TO_EDGE"] | WebGLRenderingContextBase["MIRRORED_REPEAT"]; +export type EmptyImage = { + width: number; + height: number; + data: null; +}; +export type DataTextureImage = RGBAImage | AlphaImage | EmptyImage; +export type TextureImage = TexImageSource | DataTextureImage; +export declare class Texture { + context: Context; + size: [ + number, + number + ]; + texture: WebGLTexture; + format: TextureFormat; + filter: TextureFilter; + wrap: TextureWrap; + useMipmap: boolean; + constructor(context: Context, image: TextureImage, format: TextureFormat, options?: { + premultiply?: boolean; + useMipmap?: boolean; + } | null); + update(image: TextureImage, options?: { + premultiply?: boolean; + useMipmap?: boolean; + } | null, position?: { + x: number; + y: number; + }): void; + bind(filter: TextureFilter, wrap: TextureWrap, minFilter?: TextureFilter | null): void; + isSizePowerOfTwo(): boolean; + destroy(): void; +} +export type Pattern = { + bin: PotpackBox; + position: ImagePosition; +}; +export declare class ImageManager extends Evented { + images: { + [_: string]: StyleImage; + }; + updatedImages: { + [_: string]: boolean; + }; + callbackDispatchedThisFrame: { + [_: string]: boolean; + }; + loaded: boolean; + requestors: Array<{ + ids: Array; + callback: Callback<{ + [_: string]: StyleImage; + }>; + }>; + patterns: { + [_: string]: Pattern; + }; + atlasImage: RGBAImage; + atlasTexture: Texture; + dirty: boolean; + constructor(); + isLoaded(): boolean; + setLoaded(loaded: boolean): void; + getImage(id: string): StyleImage; + addImage(id: string, image: StyleImage): void; + _validate(id: string, image: StyleImage): boolean; + _validateStretch(stretch: Array<[ + number, + number + ]>, size: number): boolean; + _validateContent(content: [ + number, + number, + number, + number + ], image: StyleImage): boolean; + updateImage(id: string, image: StyleImage, validate?: boolean): void; + removeImage(id: string): void; + listImages(): Array; + getImages(ids: Array, callback: Callback<{ + [_: string]: StyleImage; + }>): void; + _notify(ids: Array, callback: Callback<{ + [_: string]: StyleImage; + }>): void; + getPixelSize(): { + width: number; + height: number; + }; + getPattern(id: string): ImagePosition; + bind(context: Context): void; + _updatePatternAtlas(): void; + beginFrame(): void; + dispatchRenderCallbacks(ids: Array): void; +} +export type GlyphMetrics = { + width: number; + height: number; + left: number; + top: number; + advance: number; + /** + * isDoubleResolution = true for 48px textures + */ + isDoubleResolution?: boolean; +}; +/** + * @internal + * A style glyph type + */ +export type StyleGlyph = { + id: number; + bitmap: AlphaImage; + metrics: GlyphMetrics; +}; +/** + * A rectangle type with postion, width and height. + */ +export type Rect = { + x: number; + y: number; + w: number; + h: number; +}; +/** + * The glyph's position + */ +export type GlyphPosition = { + rect: Rect; + metrics: GlyphMetrics; +}; +/** + * The glyphs' positions + */ +export type GlyphPositions = { + [_: string]: { + [_: number]: GlyphPosition; + }; +}; +export declare class ImagePosition { + paddedRect: Rect; + pixelRatio: number; + version: number; + stretchY: Array<[ + number, + number + ]>; + stretchX: Array<[ + number, + number + ]>; + content: [ + number, + number, + number, + number + ]; + constructor(paddedRect: Rect, { pixelRatio, version, stretchX, stretchY, content }: StyleImage); + get tl(): [ + number, + number + ]; + get br(): [ + number, + number + ]; + get tlbr(): Array; + get displaySize(): [ + number, + number + ]; +} +export declare class ImageAtlas { + image: RGBAImage; + iconPositions: { + [_: string]: ImagePosition; + }; + patternPositions: { + [_: string]: ImagePosition; + }; + haveRenderCallbacks: Array; + uploaded: boolean; + constructor(icons: { + [_: string]: StyleImage; + }, patterns: { + [_: string]: StyleImage; + }); + addImages(images: { + [_: string]: StyleImage; + }, positions: { + [_: string]: ImagePosition; + }, bins: Array): void; + patchUpdatedImages(imageManager: ImageManager, texture: Texture): void; + patchUpdatedImage(position: ImagePosition, image: StyleImage, texture: Texture): void; +} +export type SerializedGrid = { + buffer: ArrayBuffer; +}; +export declare class TransferableGridIndex { + cells: number[][]; + arrayBuffer: ArrayBuffer; + d: number; + keys: number[]; + bboxes: number[]; + n: number; + extent: number; + padding: number; + scale: any; + uid: number; + min: number; + max: number; + constructor(extent: number | ArrayBuffer, n?: number, padding?: number); + insert(key: number, x1: number, y1: number, x2: number, y2: number): void; + _insertReadonly(): void; + _insertCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number): void; + query(x1: number, y1: number, x2: number, y2: number, intersectionTest?: Function): number[]; + _queryCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, result: any, seenUids: any, intersectionTest: Function): void; + _forEachCell(x1: number, y1: number, x2: number, y2: number, fn: Function, arg1: any, arg2: any, intersectionTest: any): void; + _convertFromCellCoord(x: any): number; + _convertToCellCoord(x: any): number; + toArrayBuffer(): ArrayBuffer; + static serialize(grid: TransferableGridIndex, transferables?: Array): SerializedGrid; + static deserialize(serialized: SerializedGrid): TransferableGridIndex; +} +export declare class DictionaryCoder { + _stringToNumber: { + [_: string]: number; + }; + _numberToString: Array; + constructor(strings: Array); + encode(string: string): number; + decode(n: number): string; +} +/** + * A helper for type to omit a property from a type + */ +export type DistributiveKeys = T extends T ? keyof T : never; +/** + * A helper for type to omit a property from a type + */ +export type DistributiveOmit> = T extends unknown ? Omit : never; +/** + * An extended geojson feature used by the events to return data to the listener + */ +export type MapGeoJSONFeature = GeoJSONFeature & { + layer: DistributiveOmit & { + source: string; + }; + source: string; + sourceLayer?: string; + state: { + [key: string]: any; + }; +}; +export declare class GeoJSONFeature { + type: "Feature"; + _geometry: GeoJSON.Geometry; + properties: { + [name: string]: any; + }; + id: number | string | undefined; + _vectorTileFeature: VectorTileFeature; + constructor(vectorTileFeature: VectorTileFeature, z: number, x: number, y: number, id: string | number | undefined); + get geometry(): GeoJSON.Geometry; + set geometry(g: GeoJSON.Geometry); + toJSON(): any; +} +/** + * A {@link LngLatBounds} object, an array of {@link LngLatLike} objects in [sw, ne] order, + * or an array of numbers in [west, south, east, north] order. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let v1 = new maplibregl.LngLatBounds( + * new maplibregl.LngLat(-73.9876, 40.7661), + * new maplibregl.LngLat(-73.9397, 40.8002) + * ); + * let v2 = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]) + * let v3 = [[-73.9876, 40.7661], [-73.9397, 40.8002]]; + * ``` + */ +export type LngLatBoundsLike = LngLatBounds | [ + LngLatLike, + LngLatLike +] | [ + number, + number, + number, + number +]; +export declare class LngLatBounds { + _ne: LngLat; + _sw: LngLat; + /** + * @param sw - The southwest corner of the bounding box. + * OR array of 4 numbers in the order of west, south, east, north + * OR array of 2 LngLatLike: [sw,ne] + * @param ne - The northeast corner of the bounding box. + * @example + * ```ts + * let sw = new maplibregl.LngLat(-73.9876, 40.7661); + * let ne = new maplibregl.LngLat(-73.9397, 40.8002); + * let llb = new maplibregl.LngLatBounds(sw, ne); + * ``` + * OR + * ```ts + * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661, -73.9397, 40.8002]); + * ``` + * OR + * ```ts + * let llb = new maplibregl.LngLatBounds([sw, ne]); + * ``` + */ + constructor(sw?: LngLatLike | [ + number, + number, + number, + number + ] | [ + LngLatLike, + LngLatLike + ], ne?: LngLatLike); + /** + * Set the northeast corner of the bounding box + * + * @param ne - a {@link LngLatLike} object describing the northeast corner of the bounding box. + * @returns `this` + */ + setNorthEast(ne: LngLatLike): this; + /** + * Set the southwest corner of the bounding box + * + * @param sw - a {@link LngLatLike} object describing the southwest corner of the bounding box. + * @returns `this` + */ + setSouthWest(sw: LngLatLike): this; + /** + * Extend the bounds to include a given LngLatLike or LngLatBoundsLike. + * + * @param obj - object to extend to + * @returns `this` + */ + extend(obj: LngLatLike | LngLatBoundsLike): this; + /** + * Returns the geographical coordinate equidistant from the bounding box's corners. + * + * @returns The bounding box's center. + * @example + * ```ts + * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]); + * llb.getCenter(); // = LngLat {lng: -73.96365, lat: 40.78315} + * ``` + */ + getCenter(): LngLat; + /** + * Returns the southwest corner of the bounding box. + * + * @returns The southwest corner of the bounding box. + */ + getSouthWest(): LngLat; + /** + * Returns the northeast corner of the bounding box. + * + * @returns The northeast corner of the bounding box. + */ + getNorthEast(): LngLat; + /** + * Returns the northwest corner of the bounding box. + * + * @returns The northwest corner of the bounding box. + */ + getNorthWest(): LngLat; + /** + * Returns the southeast corner of the bounding box. + * + * @returns The southeast corner of the bounding box. + */ + getSouthEast(): LngLat; + /** + * Returns the west edge of the bounding box. + * + * @returns The west edge of the bounding box. + */ + getWest(): number; + /** + * Returns the south edge of the bounding box. + * + * @returns The south edge of the bounding box. + */ + getSouth(): number; + /** + * Returns the east edge of the bounding box. + * + * @returns The east edge of the bounding box. + */ + getEast(): number; + /** + * Returns the north edge of the bounding box. + * + * @returns The north edge of the bounding box. + */ + getNorth(): number; + /** + * Returns the bounding box represented as an array. + * + * @returns The bounding box represented as an array, consisting of the + * southwest and northeast coordinates of the bounding represented as arrays of numbers. + * @example + * ```ts + * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]); + * llb.toArray(); // = [[-73.9876, 40.7661], [-73.9397, 40.8002]] + * ``` + */ + toArray(): [ + number, + number + ][]; + /** + * Return the bounding box represented as a string. + * + * @returns The bounding box represents as a string of the format + * `'LngLatBounds(LngLat(lng, lat), LngLat(lng, lat))'`. + * @example + * ```ts + * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]); + * llb.toString(); // = "LngLatBounds(LngLat(-73.9876, 40.7661), LngLat(-73.9397, 40.8002))" + * ``` + */ + toString(): string; + /** + * Check if the bounding box is an empty/`null`-type box. + * + * @returns True if bounds have been defined, otherwise false. + */ + isEmpty(): boolean; + /** + * Check if the point is within the bounding box. + * + * @param lnglat - geographic point to check against. + * @returns `true` if the point is within the bounding box. + * @example + * ```ts + * let llb = new maplibregl.LngLatBounds( + * new maplibregl.LngLat(-73.9876, 40.7661), + * new maplibregl.LngLat(-73.9397, 40.8002) + * ); + * + * let ll = new maplibregl.LngLat(-73.9567, 40.7789); + * + * console.log(llb.contains(ll)); // = true + * ``` + */ + contains(lnglat: LngLatLike): boolean; + /** + * Converts an array to a `LngLatBounds` object. + * + * If a `LngLatBounds` object is passed in, the function returns it unchanged. + * + * Internally, the function calls `LngLat#convert` to convert arrays to `LngLat` values. + * + * @param input - An array of two coordinates to convert, or a `LngLatBounds` object to return. + * @returns A new `LngLatBounds` object, if a conversion occurred, or the original `LngLatBounds` object. + * @example + * ```ts + * let arr = [[-73.9876, 40.7661], [-73.9397, 40.8002]]; + * let llb = maplibregl.LngLatBounds.convert(arr); // = LngLatBounds {_sw: LngLat {lng: -73.9876, lat: 40.7661}, _ne: LngLat {lng: -73.9397, lat: 40.8002}} + * ``` + */ + static convert(input: LngLatBoundsLike | null): LngLatBounds; + /** + * Returns a `LngLatBounds` from the coordinates extended by a given `radius`. The returned `LngLatBounds` completely contains the `radius`. + * + * @param center - center coordinates of the new bounds. + * @param radius - Distance in meters from the coordinates to extend the bounds. + * @returns A new `LngLatBounds` object representing the coordinates extended by the `radius`. + * @example + * ```ts + * let center = new maplibregl.LngLat(-73.9749, 40.7736); + * maplibregl.LngLatBounds.fromLngLat(100).toArray(); // = [[-73.97501862141328, 40.77351016847229], [-73.97478137858673, 40.77368983152771]] + * ``` + */ + static fromLngLat(center: LngLat, radius?: number): LngLatBounds; +} +export declare class EdgeInsets { + /** + * @defaultValue 0 + */ + top: number; + /** + * @defaultValue 0 + */ + bottom: number; + /** + * @defaultValue 0 + */ + left: number; + /** + * @defaultValue 0 + */ + right: number; + constructor(top?: number, bottom?: number, left?: number, right?: number); + /** + * Interpolates the inset in-place. + * This maintains the current inset value for any inset not present in `target`. + * @param start - interpolation start + * @param target - interpolation target + * @param t - interpolation step/weight + * @returns the insets + */ + interpolate(start: PaddingOptions | EdgeInsets, target: PaddingOptions, t: number): EdgeInsets; + /** + * Utility method that computes the new apprent center or vanishing point after applying insets. + * This is in pixels and with the top left being (0.0) and +y being downwards. + * + * @param width - the width + * @param height - the height + * @returns the point + */ + getCenter(width: number, height: number): Point; + equals(other: PaddingOptions): boolean; + clone(): EdgeInsets; + /** + * Returns the current state as json, useful when you want to have a + * read-only representation of the inset. + * + * @returns state as json + */ + toJSON(): PaddingOptions; +} +/** + * Options for setting padding on calls to methods such as {@link Map#fitBounds}, {@link Map#fitScreenCoordinates}, and {@link Map#setPadding}. Adjust these options to set the amount of padding in pixels added to the edges of the canvas. Set a uniform padding on all edges or individual values for each edge. All properties of this object must be + * non-negative integers. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let bbox = [[-79, 43], [-73, 45]]; + * map.fitBounds(bbox, { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + * + * @example + * ```ts + * let bbox = [[-79, 43], [-73, 45]]; + * map.fitBounds(bbox, { + * padding: 20 + * }); + * ``` + * @see [Fit to the bounds of a LineString](https://maplibre.org/maplibre-gl-js/docs/examples/zoomto-linestring/) + * @see [Fit a map to a bounding box](https://maplibre.org/maplibre-gl-js/docs/examples/fitbounds/) + */ +export type PaddingOptions = { + /** + * Padding in pixels from the top of the map canvas. + */ + top: number; + /** + * Padding in pixels from the bottom of the map canvas. + */ + bottom: number; + /** + * Padding in pixels from the left of the map canvas. + */ + right: number; + /** + * Padding in pixels from the right of the map canvas. + */ + left: number; +}; +export declare class TileCache { + max: number; + data: { + [key: string]: Array<{ + value: Tile; + timeout: ReturnType; + }>; + }; + order: Array; + onRemove: (element: Tile) => void; + /** + * @param max - number of permitted values + * @param onRemove - callback called with items when they expire + */ + constructor(max: number, onRemove: (element: Tile) => void); + /** + * Clear the cache + * + * @returns this cache + */ + reset(): this; + /** + * Add a key, value combination to the cache, trimming its size if this pushes + * it over max length. + * + * @param tileID - lookup key for the item + * @param data - tile data + * + * @returns this cache + */ + add(tileID: OverscaledTileID, data: Tile, expiryTimeout: number | void): this; + /** + * Determine whether the value attached to `key` is present + * + * @param tileID - the key to be looked-up + * @returns whether the cache has this value + */ + has(tileID: OverscaledTileID): boolean; + /** + * Get the value attached to a specific key and remove data from cache. + * If the key is not found, returns `null` + * + * @param tileID - the key to look up + * @returns the tile data, or null if it isn't found + */ + getAndRemove(tileID: OverscaledTileID): Tile; + _getAndRemoveByKey(key: string): Tile; + getByKey(key: string): Tile; + /** + * Get the value attached to a specific key without removing data + * from the cache. If the key is not found, returns `null` + * + * @param tileID - the key to look up + * @returns the tile data, or null if it isn't found + */ + get(tileID: OverscaledTileID): Tile; + /** + * Remove a key/value combination from the cache. + * + * @param tileID - the key for the pair to delete + * @param value - If a value is provided, remove that exact version of the value. + * @returns this cache + */ + remove(tileID: OverscaledTileID, value?: { + value: Tile; + timeout: ReturnType; + }): this; + /** + * Change the max size of the cache. + * + * @param max - the max size of the cache + * @returns this cache + */ + setMaxSize(max: number): TileCache; + /** + * Remove entries that do not pass a filter function. Used for removing + * stale tiles from the cache. + * + * @param filterFn - Determines whether the tile is filtered. If the supplied function returns false, the tile will be filtered out. + */ + filter(filterFn: (tile: Tile) => boolean): void; +} +export type SerializedObject = { + [_: string]: S; +}; +export type Serialized = null | void | boolean | number | string | Boolean | Number | String | Date | RegExp | ArrayBuffer | ArrayBufferView | ImageData | ImageBitmap | Blob | Array | SerializedObject; +export declare class ThrottledInvoker { + _channel: MessageChannel; + _triggered: boolean; + _callback: Function; + constructor(callback: Function); + trigger(): void; + remove(): void; +} +export type DEMEncoding = "mapbox" | "terrarium" | "custom"; +export declare class DEMData { + uid: string; + data: Uint32Array; + stride: number; + dim: number; + min: number; + max: number; + redFactor: number; + greenFactor: number; + blueFactor: number; + baseShift: number; + constructor(uid: string, data: RGBAImage, encoding: DEMEncoding, redFactor?: number, greenFactor?: number, blueFactor?: number, baseShift?: number); + get(x: number, y: number): number; + getUnpackVector(): number[]; + _idx(x: number, y: number): number; + unpack(r: number, g: number, b: number): number; + getPixels(): RGBAImage; + backfillBorder(borderTile: DEMData, dx: number, dy: number): void; +} +export type TileParameters = { + source: string; + uid: string; +}; +export type WorkerTileParameters = TileParameters & { + tileID: OverscaledTileID; + request: RequestParameters; + zoom: number; + maxZoom: number; + tileSize: number; + promoteId: PromoteIdSpecification; + pixelRatio: number; + showCollisionBoxes: boolean; + collectResourceTiming?: boolean; + returnDependencies?: boolean; +}; +/** + * @internal + * The worker tile's result type + */ +export type WorkerTileResult = { + buckets: Array; + imageAtlas: ImageAtlas; + glyphAtlasImage: AlphaImage; + featureIndex: FeatureIndex; + collisionBoxArray: CollisionBoxArray; + rawTileData?: ArrayBuffer; + resourceTiming?: Array; + glyphMap?: { + [_: string]: { + [_: number]: StyleGlyph; + }; + } | null; + iconMap?: { + [_: string]: StyleImage; + } | null; + glyphPositions?: GlyphPositions | null; +}; +export type WorkerTileCallback = (error?: Error | null, result?: WorkerTileResult | null) => void; +/** + * May be implemented by custom source types to provide code that can be run on + * the WebWorkers. In addition to providing a custom + * {@link WorkerSource#loadTile}, any other methods attached to a `WorkerSource` + * implementation may also be targeted by the {@link Source} via + * `dispatcher.getActor().send('source-type.methodname', params, callback)`. + * + * @see {@link Map#addSourceType} + */ +export interface WorkerSource { + availableImages: Array; + /** + * Loads a tile from the given params and parse it into buckets ready to send + * back to the main thread for rendering. Should call the callback with: + * `{ buckets, featureIndex, collisionIndex, rawTileData}`. + */ + loadTile(params: WorkerTileParameters, callback: WorkerTileCallback): void; + /** + * Re-parses a tile that has already been loaded. Yields the same data as + * {@link WorkerSource#loadTile}. + */ + reloadTile(params: WorkerTileParameters, callback: WorkerTileCallback): void; + /** + * Aborts loading a tile that is in progress. + */ + abortTile(params: TileParameters, callback: WorkerTileCallback): void; + /** + * Removes this tile from any local caches. + */ + removeTile(params: TileParameters, callback: WorkerTileCallback): void; + /** + * Tells the WorkerSource to abort in-progress tasks and release resources. + * The foreground Source is responsible for ensuring that 'removeSource' is + * the last message sent to the WorkerSource. + */ + removeSource?: (params: { + source: string; + }, callback: WorkerTileCallback) => void; +} +export interface ActorTarget { + addEventListener: typeof window.addEventListener; + removeEventListener: typeof window.removeEventListener; + postMessage: typeof window.postMessage; + terminate?: () => void; +} +export interface WorkerSourceProvider { + getWorkerSource(mapId: string | number, sourceType: string, sourceName: string): WorkerSource; +} +export interface GlyphsProvider { + getGlyphs(mapId: string, params: { + stacks: { + [_: string]: Array; + }; + source: string; + tileID: OverscaledTileID; + type: string; + }, callback: Callback<{ + [_: string]: { + [_: number]: StyleGlyph; + }; + }>): any; +} +export type MessageType = "" | "" | "geojson.getClusterExpansionZoom" | "geojson.getClusterChildren" | "geojson.getClusterLeaves" | "geojson.loadData" | "removeSource" | "loadWorkerSource" | "loadDEMTile" | "removeDEMTile" | "removeTile" | "reloadTile" | "abortTile" | "loadTile" | "getTile" | "getGlyphs" | "getImages" | "setImages" | "syncRTLPluginState" | "setReferrer" | "setLayers" | "updateLayers"; +export type MessageData = { + id: string; + type: MessageType; + data?: Serialized; + targetMapId?: string | number | null; + mustQueue?: boolean; + error?: Serialized | null; + hasCallback?: boolean; + sourceMapId: string | number | null; +}; +export type Message = { + data: MessageData; +}; +export declare class Actor { + target: ActorTarget; + parent: WorkerSourceProvider | GlyphsProvider; + mapId: string | number | null; + callbacks: { + [x: number]: Function; + }; + name: string; + tasks: { + [x: number]: MessageData; + }; + taskQueue: Array; + cancelCallbacks: { + [x: number]: () => void; + }; + invoker: ThrottledInvoker; + globalScope: ActorTarget; + /** + * @param target - The target + * @param parent - The parent + * @param mapId - A unique identifier for the Map instance using this Actor. + */ + constructor(target: ActorTarget, parent: WorkerSourceProvider | GlyphsProvider, mapId?: string | number); + /** + * Sends a message from a main-thread map to a Worker or from a Worker back to + * a main-thread map instance. + * + * @param type - The name of the target method to invoke or '[source-type].[source-name].name' for a method on a WorkerSource. + * @param targetMapId - A particular mapId to which to send this message. + */ + send(type: MessageType, data: unknown, callback?: Function | null, targetMapId?: string | null, mustQueue?: boolean): Cancelable; + receive: (message: Message) => void; + process: () => void; + processTask(id: string, task: MessageData): void; + remove(): void; +} +export declare class WorkerPool { + static workerCount: number; + active: { + [_ in number | string]: boolean; + }; + workers: Array; + constructor(); + acquire(mapId: number | string): Array; + release(mapId: number | string): void; + isPreloaded(): boolean; + numActive(): number; +} +export declare class Dispatcher { + workerPool: WorkerPool; + actors: Array; + currentActor: number; + id: string | number; + constructor(workerPool: WorkerPool, parent: GlyphsProvider, mapId: string | number); + /** + * Broadcast a message to all Workers. + */ + broadcast(type: MessageType, data: unknown, cb?: (...args: any[]) => any): void; + /** + * Acquires an actor to dispatch messages to. The actors are distributed in round-robin fashion. + * @returns An actor object backed by a web worker for processing messages. + */ + getActor(): Actor; + remove(mapRemoved?: boolean): void; +} +/** + * Four geographical coordinates, + * represented as arrays of longitude and latitude numbers, which define the corners of the image. + * The coordinates start at the top left corner of the image and proceed in clockwise order. + * They do not have to represent a rectangle. + */ +export type Coordinates = [ + [ + number, + number + ], + [ + number, + number + ], + [ + number, + number + ], + [ + number, + number + ] +]; +/** + * The options object for the {@link ImageSource#updateImage} method + */ +export type UpdateImageOptions = { + /** + * Required image URL. + */ + url: string; + /** + * The image coordinates + */ + coordinates?: Coordinates; +}; +export declare class ImageSource extends Evented implements Source { + type: string; + id: string; + minzoom: number; + maxzoom: number; + tileSize: number; + url: string; + coordinates: Coordinates; + tiles: { + [_: string]: Tile; + }; + options: any; + dispatcher: Dispatcher; + map: Map; + texture: Texture | null; + image: HTMLImageElement | ImageBitmap; + tileID: CanonicalTileID; + _boundsArray: RasterBoundsArray; + boundsBuffer: VertexBuffer; + boundsSegments: SegmentVector; + _loaded: boolean; + _request: Cancelable; + /** @internal */ + constructor(id: string, options: ImageSourceSpecification | VideoSourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented); + load: (newCoordinates?: Coordinates, successCallback?: () => void) => void; + loaded(): boolean; + /** + * Updates the image URL and, optionally, the coordinates. To avoid having the image flash after changing, + * set the `raster-fade-duration` paint property on the raster layer to 0. + * + * @param options - The options object. + * @returns `this` + */ + updateImage(options: UpdateImageOptions): this; + _finishLoading(): void; + onAdd(map: Map): void; + onRemove(): void; + /** + * Sets the image's coordinates and re-renders the map. + * + * @param coordinates - Four geographical coordinates, + * represented as arrays of longitude and latitude numbers, which define the corners of the image. + * The coordinates start at the top left corner of the image and proceed in clockwise order. + * They do not have to represent a rectangle. + * @returns `this` + */ + setCoordinates(coordinates: Coordinates): this; + prepare: () => void; + loadTile(tile: Tile, callback: Callback): void; + serialize: () => ImageSourceSpecification | VideoSourceSpecification | CanvasSourceSpecification; + hasTransition(): boolean; +} +/** + * Options to add a canvas source type to the map. + */ +export type CanvasSourceSpecification = { + /** + * Source type. Must be `"canvas"`. + */ + type: "canvas"; + /** + * Four geographical coordinates denoting where to place the corners of the canvas, specified in `[longitude, latitude]` pairs. + */ + coordinates: [ + [ + number, + number + ], + [ + number, + number + ], + [ + number, + number + ], + [ + number, + number + ] + ]; + /** + * Whether the canvas source is animated. If the canvas is static (i.e. pixels do not need to be re-read on every frame), `animate` should be set to `false` to improve performance. + * @defaultValue true + */ + animate?: boolean; + /** + * Canvas source from which to read pixels. Can be a string representing the ID of the canvas element, or the `HTMLCanvasElement` itself. + */ + canvas?: string | HTMLCanvasElement; +}; +export declare class CanvasSource extends ImageSource { + options: CanvasSourceSpecification; + animate: boolean; + canvas: HTMLCanvasElement; + width: number; + height: number; + /** + * Enables animation. The image will be copied from the canvas to the map on each frame. + */ + play: () => void; + /** + * Disables animation. The map will display a static copy of the canvas image. + */ + pause: () => void; + _playing: boolean; + /** @internal */ + constructor(id: string, options: CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented); + load: () => void; + /** + * Returns the HTML `canvas` element. + * + * @returns The HTML `canvas` element. + */ + getCanvas(): HTMLCanvasElement; + onAdd(map: Map): void; + onRemove(): void; + prepare: () => void; + serialize: () => CanvasSourceSpecification; + hasTransition(): boolean; + _hasInvalidDimensions(): boolean; +} +/** + * The `Source` interface must be implemented by each source type, including "core" types (`vector`, `raster`, + * `video`, etc.) and all custom, third-party types. + * + * @event `data` - Fired with `{dataType: 'source', sourceDataType: 'metadata'}` to indicate that any necessary metadata + * has been loaded so that it's okay to call `loadTile`; and with `{dataType: 'source', sourceDataType: 'content'}` + * to indicate that the source data has changed, so that any current caches should be flushed. + * + * @group Sources + */ +export interface Source { + readonly type: string; + /** + * The id for the source. Must not be used by any existing source. + */ + id: string; + minzoom: number; + maxzoom: number; + tileSize: number; + attribution?: string; + /** + * `true` if zoom levels are rounded to the nearest integer in the source data, `false` if they are floor-ed to the nearest integer. + */ + roundZoom?: boolean; + /** + * `false` if tiles can be drawn outside their boundaries, `true` if they cannot. + */ + isTileClipped?: boolean; + tileID?: CanonicalTileID; + /** + * `true` if tiles should be sent back to the worker for each overzoomed zoom level, `false` if not. + */ + reparseOverscaled?: boolean; + vectorLayerIds?: Array; + hasTransition(): boolean; + loaded(): boolean; + fire(event: Event): unknown; + readonly onAdd?: (map: Map) => void; + readonly onRemove?: (map: Map) => void; + loadTile(tile: Tile, callback: Callback): void; + readonly hasTile?: (tileID: OverscaledTileID) => boolean; + readonly abortTile?: (tile: Tile, callback: Callback) => void; + readonly unloadTile?: (tile: Tile, callback: Callback) => void; + /** + * @returns A plain (stringifiable) JS object representing the current state of the source. + * Creating a source using the returned object as the `options` should result in a Source that is + * equivalent to this one. + */ + serialize(): any; + readonly prepare?: () => void; +} +/** + * A supporting type to the source definition + */ +export type SourceStatics = { + workerSourceURL?: URL; +}; +/** + * A general definition of a {@link Source} class for factory usage + */ +export type SourceClass = { + new (id: string, specification: SourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented): Source; +} & SourceStatics; +export declare class SourceCache extends Evented { + id: string; + dispatcher: Dispatcher; + map: Map; + style: Style; + _source: Source; + _sourceLoaded: boolean; + _sourceErrored: boolean; + _tiles: { + [_: string]: Tile; + }; + _prevLng: number; + _cache: TileCache; + _timers: { + [_ in any]: ReturnType; + }; + _cacheTimers: { + [_ in any]: ReturnType; + }; + _maxTileCacheSize: number; + _maxTileCacheZoomLevels: number; + _paused: boolean; + _shouldReloadOnResume: boolean; + _coveredTiles: { + [_: string]: boolean; + }; + transform: Transform; + terrain: Terrain; + used: boolean; + usedForTerrain: boolean; + tileSize: number; + _state: SourceFeatureState; + _loadedParentTiles: { + [_: string]: Tile; + }; + _didEmitContent: boolean; + _updated: boolean; + static maxUnderzooming: number; + static maxOverzooming: number; + constructor(id: string, options: SourceSpecification, dispatcher: Dispatcher); + onAdd(map: Map): void; + onRemove(map: Map): void; + /** + * Return true if no tile data is pending, tiles will not change unless + * an additional API call is received. + */ + loaded(): boolean; + getSource(): Source; + pause(): void; + resume(): void; + _loadTile(tile: Tile, callback: Callback): void; + _unloadTile(tile: Tile): void; + _abortTile(tile: Tile): void; + serialize(): any; + prepare(context: Context): void; + /** + * Return all tile ids ordered with z-order, and cast to numbers + */ + getIds(): Array; + getRenderableIds(symbolLayer?: boolean): Array; + hasRenderableParent(tileID: OverscaledTileID): boolean; + _isIdRenderable(id: string, symbolLayer?: boolean): boolean; + reload(): void; + _reloadTile(id: string, state: TileState): void; + _tileLoaded(tile: Tile, id: string, previousState: TileState, err?: Error | null): void; + /** + * For raster terrain source, backfill DEM to eliminate visible tile boundaries + */ + _backfillDEM(tile: Tile): void; + /** + * Get a specific tile by TileID + */ + getTile(tileID: OverscaledTileID): Tile; + /** + * Get a specific tile by id + */ + getTileByID(id: string): Tile; + /** + * For a given set of tiles, retain children that are loaded and have a zoom + * between `zoom` (exclusive) and `maxCoveringZoom` (inclusive) + */ + _retainLoadedChildren(idealTiles: { + [_ in any]: OverscaledTileID; + }, zoom: number, maxCoveringZoom: number, retain: { + [_ in any]: OverscaledTileID; + }): void; + /** + * Find a loaded parent of the given tile (up to minCoveringZoom) + */ + findLoadedParent(tileID: OverscaledTileID, minCoveringZoom: number): Tile; + _getLoadedTile(tileID: OverscaledTileID): Tile; + /** + * Resizes the tile cache based on the current viewport's size + * or the maxTileCacheSize option passed during map creation + * + * Larger viewports use more tiles and need larger caches. Larger viewports + * are more likely to be found on devices with more memory and on pages where + * the map is more important. + */ + updateCacheSize(transform: Transform): void; + handleWrapJump(lng: number): void; + /** + * Removes tiles that are outside the viewport and adds new tiles that + * are inside the viewport. + */ + update(transform: Transform, terrain?: Terrain): void; + releaseSymbolFadeTiles(): void; + _updateRetainedTiles(idealTileIDs: Array, zoom: number): { + [_: string]: OverscaledTileID; + }; + _updateLoadedParentTileCache(): void; + /** + * Add a tile, given its coordinate, to the pyramid. + */ + _addTile(tileID: OverscaledTileID): Tile; + _setTileReloadTimer(id: string, tile: Tile): void; + /** + * Remove a tile, given its id, from the pyramid + */ + _removeTile(id: string): void; + /** + * Remove all tiles from this pyramid + */ + clearTiles(): void; + /** + * Search through our current tiles and attempt to find the tiles that + * cover the given bounds. + * @param pointQueryGeometry - coordinates of the corners of bounding rectangle + * @returns result items have `{tile, minX, maxX, minY, maxY}`, where min/max bounding values are the given bounds transformed in into the coordinate space of this tile. + */ + tilesIn(pointQueryGeometry: Array, maxPitchScaleFactor: number, has3DLayer: boolean): any[]; + getVisibleCoordinates(symbolLayer?: boolean): Array; + hasTransition(): boolean; + /** + * Set the value of a particular state for a feature + */ + setFeatureState(sourceLayer: string, featureId: number | string, state: any): void; + /** + * Resets the value of a particular state key for a feature + */ + removeFeatureState(sourceLayer?: string, featureId?: number | string, key?: string): void; + /** + * Get the entire state object for a feature + */ + getFeatureState(sourceLayer: string, featureId: number | string): any; + /** + * Sets the set of keys that the tile depends on. This allows tiles to + * be reloaded when their dependencies change. + */ + setDependencies(tileKey: string, namespace: string, dependencies: Array): void; + /** + * Reloads all tiles that depend on the given keys. + */ + reloadTilesForDependencies(namespaces: Array, keys: Array): void; +} +declare enum WritingMode { + none = 0, + horizontal = 1, + vertical = 2, + horizontalOnly = 3 +} +export declare class Anchor extends Point { + angle: any; + segment?: number; + constructor(x: number, y: number, angle: number, segment?: number); + clone(): Anchor; +} +export type SymbolLayoutProps = { + "symbol-placement": DataConstantProperty<"point" | "line" | "line-center">; + "symbol-spacing": DataConstantProperty; + "symbol-avoid-edges": DataConstantProperty; + "symbol-sort-key": DataDrivenProperty; + "symbol-z-order": DataConstantProperty<"auto" | "viewport-y" | "source">; + "icon-allow-overlap": DataConstantProperty; + "icon-overlap": DataConstantProperty<"never" | "always" | "cooperative">; + "icon-ignore-placement": DataConstantProperty; + "icon-optional": DataConstantProperty; + "icon-rotation-alignment": DataConstantProperty<"map" | "viewport" | "auto">; + "icon-size": DataDrivenProperty; + "icon-text-fit": DataConstantProperty<"none" | "width" | "height" | "both">; + "icon-text-fit-padding": DataConstantProperty<[ + number, + number, + number, + number + ]>; + "icon-image": DataDrivenProperty; + "icon-rotate": DataDrivenProperty; + "icon-padding": DataDrivenProperty; + "icon-keep-upright": DataConstantProperty; + "icon-offset": DataDrivenProperty<[ + number, + number + ]>; + "icon-anchor": DataDrivenProperty<"center" | "left" | "right" | "top" | "bottom" | "top-left" | "top-right" | "bottom-left" | "bottom-right">; + "icon-pitch-alignment": DataConstantProperty<"map" | "viewport" | "auto">; + "text-pitch-alignment": DataConstantProperty<"map" | "viewport" | "auto">; + "text-rotation-alignment": DataConstantProperty<"map" | "viewport" | "viewport-glyph" | "auto">; + "text-field": DataDrivenProperty; + "text-font": DataDrivenProperty>; + "text-size": DataDrivenProperty; + "text-max-width": DataDrivenProperty; + "text-line-height": DataConstantProperty; + "text-letter-spacing": DataDrivenProperty; + "text-justify": DataDrivenProperty<"auto" | "left" | "center" | "right">; + "text-radial-offset": DataDrivenProperty; + "text-variable-anchor": DataConstantProperty>; + "text-variable-anchor-offset": DataDrivenProperty; + "text-anchor": DataDrivenProperty<"center" | "left" | "right" | "top" | "bottom" | "top-left" | "top-right" | "bottom-left" | "bottom-right">; + "text-max-angle": DataConstantProperty; + "text-writing-mode": DataConstantProperty>; + "text-rotate": DataDrivenProperty; + "text-padding": DataConstantProperty; + "text-keep-upright": DataConstantProperty; + "text-transform": DataDrivenProperty<"none" | "uppercase" | "lowercase">; + "text-offset": DataDrivenProperty<[ + number, + number + ]>; + "text-allow-overlap": DataConstantProperty; + "text-overlap": DataConstantProperty<"never" | "always" | "cooperative">; + "text-ignore-placement": DataConstantProperty; + "text-optional": DataConstantProperty; +}; +export type SymbolLayoutPropsPossiblyEvaluated = { + "symbol-placement": "point" | "line" | "line-center"; + "symbol-spacing": number; + "symbol-avoid-edges": boolean; + "symbol-sort-key": PossiblyEvaluatedPropertyValue; + "symbol-z-order": "auto" | "viewport-y" | "source"; + "icon-allow-overlap": boolean; + "icon-overlap": "never" | "always" | "cooperative"; + "icon-ignore-placement": boolean; + "icon-optional": boolean; + "icon-rotation-alignment": "map" | "viewport" | "auto"; + "icon-size": PossiblyEvaluatedPropertyValue; + "icon-text-fit": "none" | "width" | "height" | "both"; + "icon-text-fit-padding": [ + number, + number, + number, + number + ]; + "icon-image": PossiblyEvaluatedPropertyValue; + "icon-rotate": PossiblyEvaluatedPropertyValue; + "icon-padding": PossiblyEvaluatedPropertyValue; + "icon-keep-upright": boolean; + "icon-offset": PossiblyEvaluatedPropertyValue<[ + number, + number + ]>; + "icon-anchor": PossiblyEvaluatedPropertyValue<"center" | "left" | "right" | "top" | "bottom" | "top-left" | "top-right" | "bottom-left" | "bottom-right">; + "icon-pitch-alignment": "map" | "viewport" | "auto"; + "text-pitch-alignment": "map" | "viewport" | "auto"; + "text-rotation-alignment": "map" | "viewport" | "viewport-glyph" | "auto"; + "text-field": PossiblyEvaluatedPropertyValue; + "text-font": PossiblyEvaluatedPropertyValue>; + "text-size": PossiblyEvaluatedPropertyValue; + "text-max-width": PossiblyEvaluatedPropertyValue; + "text-line-height": number; + "text-letter-spacing": PossiblyEvaluatedPropertyValue; + "text-justify": PossiblyEvaluatedPropertyValue<"auto" | "left" | "center" | "right">; + "text-radial-offset": PossiblyEvaluatedPropertyValue; + "text-variable-anchor": Array<"center" | "left" | "right" | "top" | "bottom" | "top-left" | "top-right" | "bottom-left" | "bottom-right">; + "text-variable-anchor-offset": PossiblyEvaluatedPropertyValue; + "text-anchor": PossiblyEvaluatedPropertyValue<"center" | "left" | "right" | "top" | "bottom" | "top-left" | "top-right" | "bottom-left" | "bottom-right">; + "text-max-angle": number; + "text-writing-mode": Array<"horizontal" | "vertical">; + "text-rotate": PossiblyEvaluatedPropertyValue; + "text-padding": number; + "text-keep-upright": boolean; + "text-transform": PossiblyEvaluatedPropertyValue<"none" | "uppercase" | "lowercase">; + "text-offset": PossiblyEvaluatedPropertyValue<[ + number, + number + ]>; + "text-allow-overlap": boolean; + "text-overlap": "never" | "always" | "cooperative"; + "text-ignore-placement": boolean; + "text-optional": boolean; +}; +export type SymbolPaintProps = { + "icon-opacity": DataDrivenProperty; + "icon-color": DataDrivenProperty; + "icon-halo-color": DataDrivenProperty; + "icon-halo-width": DataDrivenProperty; + "icon-halo-blur": DataDrivenProperty; + "icon-translate": DataConstantProperty<[ + number, + number + ]>; + "icon-translate-anchor": DataConstantProperty<"map" | "viewport">; + "text-opacity": DataDrivenProperty; + "text-color": DataDrivenProperty; + "text-halo-color": DataDrivenProperty; + "text-halo-width": DataDrivenProperty; + "text-halo-blur": DataDrivenProperty; + "text-translate": DataConstantProperty<[ + number, + number + ]>; + "text-translate-anchor": DataConstantProperty<"map" | "viewport">; +}; +export type SymbolPaintPropsPossiblyEvaluated = { + "icon-opacity": PossiblyEvaluatedPropertyValue; + "icon-color": PossiblyEvaluatedPropertyValue; + "icon-halo-color": PossiblyEvaluatedPropertyValue; + "icon-halo-width": PossiblyEvaluatedPropertyValue; + "icon-halo-blur": PossiblyEvaluatedPropertyValue; + "icon-translate": [ + number, + number + ]; + "icon-translate-anchor": "map" | "viewport"; + "text-opacity": PossiblyEvaluatedPropertyValue; + "text-color": PossiblyEvaluatedPropertyValue; + "text-halo-color": PossiblyEvaluatedPropertyValue; + "text-halo-width": PossiblyEvaluatedPropertyValue; + "text-halo-blur": PossiblyEvaluatedPropertyValue; + "text-translate": [ + number, + number + ]; + "text-translate-anchor": "map" | "viewport"; +}; +export declare class SymbolStyleLayer extends StyleLayer { + _unevaluatedLayout: Layout; + layout: PossiblyEvaluated; + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + constructor(layer: LayerSpecification); + recalculate(parameters: EvaluationParameters, availableImages: Array): void; + getValueAndResolveTokens(name: any, feature: Feature, canonical: CanonicalTileID, availableImages: Array): any; + createBucket(parameters: BucketParameters): SymbolBucket; + queryRadius(): number; + queryIntersectsFeature(): boolean; + _setPaintOverrides(): void; + _handleOverridablePaintPropertyUpdate(name: string, oldValue: PropertyValue, newValue: PropertyValue): boolean; + static hasPaintOverride(layout: PossiblyEvaluated, propertyName: string): boolean; +} +/** + * A textured quad for rendering a single icon or glyph. + * + * The zoom range the glyph can be shown is defined by minScale and maxScale. + * + * @param tl - The offset of the top left corner from the anchor. + * @param tr - The offset of the top right corner from the anchor. + * @param bl - The offset of the bottom left corner from the anchor. + * @param br - The offset of the bottom right corner from the anchor. + * @param tex - The texture coordinates. + */ +export type SymbolQuad = { + tl: Point; + tr: Point; + bl: Point; + br: Point; + tex: { + x: number; + y: number; + w: number; + h: number; + }; + pixelOffsetTL: Point; + pixelOffsetBR: Point; + writingMode: any | void; + glyphOffset: [ + number, + number + ]; + sectionIndex: number; + isSDF: boolean; + minFontScaleX: number; + minFontScaleY: number; +}; +export type SizeData = { + kind: "constant"; + layoutSize: number; +} | { + kind: "source"; +} | { + kind: "camera"; + minZoom: number; + maxZoom: number; + minSize: number; + maxSize: number; + interpolationType: InterpolationType; +} | { + kind: "composite"; + minZoom: number; + maxZoom: number; + interpolationType: InterpolationType; +}; +export type SingleCollisionBox = { + x1: number; + y1: number; + x2: number; + y2: number; + anchorPointX: number; + anchorPointY: number; +}; +export type CollisionArrays = { + textBox?: SingleCollisionBox; + verticalTextBox?: SingleCollisionBox; + iconBox?: SingleCollisionBox; + verticalIconBox?: SingleCollisionBox; + textFeatureIndex?: number; + verticalTextFeatureIndex?: number; + iconFeatureIndex?: number; + verticalIconFeatureIndex?: number; +}; +export type SymbolFeature = { + sortKey: number | void; + text: Formatted | void; + icon: ResolvedImage; + index: number; + sourceLayerIndex: number; + geometry: Array>; + properties: any; + type: "Unknown" | "Point" | "LineString" | "Polygon"; + id?: any; +}; +export type SortKeyRange = { + sortKey: number; + symbolInstanceStart: number; + symbolInstanceEnd: number; +}; +declare function addDynamicAttributes(dynamicLayoutVertexArray: StructArray, p: Point, angle: number): void; +export declare class SymbolBuffers { + layoutVertexArray: SymbolLayoutArray; + layoutVertexBuffer: VertexBuffer; + indexArray: TriangleIndexArray; + indexBuffer: IndexBuffer; + programConfigurations: ProgramConfigurationSet; + segments: SegmentVector; + dynamicLayoutVertexArray: SymbolDynamicLayoutArray; + dynamicLayoutVertexBuffer: VertexBuffer; + opacityVertexArray: SymbolOpacityArray; + opacityVertexBuffer: VertexBuffer; + hasVisibleVertices: boolean; + collisionVertexArray: CollisionVertexArray; + collisionVertexBuffer: VertexBuffer; + placedSymbolArray: PlacedSymbolArray; + constructor(programConfigurations: ProgramConfigurationSet); + isEmpty(): boolean; + upload(context: Context, dynamicIndexBuffer: boolean, upload?: boolean, update?: boolean): void; + destroy(): void; +} +export declare class CollisionBuffers { + layoutVertexArray: StructArray; + layoutAttributes: Array; + layoutVertexBuffer: VertexBuffer; + indexArray: TriangleIndexArray | LineIndexArray; + indexBuffer: IndexBuffer; + segments: SegmentVector; + collisionVertexArray: CollisionVertexArray; + collisionVertexBuffer: VertexBuffer; + constructor(LayoutArray: { + new (...args: any): StructArray; + }, layoutAttributes: Array, IndexArray: { + new (...args: any): TriangleIndexArray | LineIndexArray; + }); + upload(context: Context): void; + destroy(): void; +} +export declare class SymbolBucket implements Bucket { + static MAX_GLYPHS: number; + static addDynamicAttributes: typeof addDynamicAttributes; + collisionBoxArray: CollisionBoxArray; + zoom: number; + overscaling: number; + layers: Array; + layerIds: Array; + stateDependentLayers: Array; + stateDependentLayerIds: Array; + index: number; + sdfIcons: boolean; + iconsInText: boolean; + iconsNeedLinear: boolean; + bucketInstanceId: number; + justReloaded: boolean; + hasPattern: boolean; + textSizeData: SizeData; + iconSizeData: SizeData; + glyphOffsetArray: GlyphOffsetArray; + lineVertexArray: SymbolLineVertexArray; + features: Array; + symbolInstances: SymbolInstanceArray; + textAnchorOffsets: TextAnchorOffsetArray; + collisionArrays: Array; + sortKeyRanges: Array; + pixelRatio: number; + tilePixelRatio: number; + compareText: { + [_: string]: Array; + }; + fadeStartTime: number; + sortFeaturesByKey: boolean; + sortFeaturesByY: boolean; + canOverlap: boolean; + sortedAngle: number; + featureSortOrder: Array; + collisionCircleArray: Array; + placementInvProjMatrix: mat4; + placementViewportMatrix: mat4; + text: SymbolBuffers; + icon: SymbolBuffers; + textCollisionBox: CollisionBuffers; + iconCollisionBox: CollisionBuffers; + uploaded: boolean; + sourceLayerIndex: number; + sourceID: string; + symbolInstanceIndexes: Array; + writingModes: WritingMode[]; + allowVerticalPlacement: boolean; + hasRTLText: boolean; + constructor(options: BucketParameters); + createArrays(): void; + calculateGlyphDependencies(text: string, stack: { + [_: number]: boolean; + }, textAlongLine: boolean, allowVerticalPlacement: boolean, doesAllowVerticalWritingMode: boolean): void; + populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID): void; + update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: { + [_: string]: ImagePosition; + }): void; + isEmpty(): boolean; + uploadPending(): boolean; + upload(context: Context): void; + destroyDebugData(): void; + destroy(): void; + addToLineVertexArray(anchor: Anchor, line: any): { + lineStartIndex: number; + lineLength: number; + }; + addSymbols(arrays: SymbolBuffers, quads: Array, sizeVertex: any, lineOffset: [ + number, + number + ], alongLine: boolean, feature: SymbolFeature, writingMode: WritingMode, labelAnchor: Anchor, lineStartIndex: number, lineLength: number, associatedIconIndex: number, canonical: CanonicalTileID): void; + _addCollisionDebugVertex(layoutVertexArray: StructArray, collisionVertexArray: StructArray, point: Point, anchorX: number, anchorY: number, extrude: Point): any; + addCollisionDebugVertices(x1: number, y1: number, x2: number, y2: number, arrays: CollisionBuffers, boxAnchorPoint: Point, symbolInstance: SymbolInstance): void; + addDebugCollisionBoxes(startIndex: number, endIndex: number, symbolInstance: SymbolInstance, isText: boolean): void; + generateCollisionDebugBuffers(): void; + _deserializeCollisionBoxesForSymbol(collisionBoxArray: CollisionBoxArray, textStartIndex: number, textEndIndex: number, verticalTextStartIndex: number, verticalTextEndIndex: number, iconStartIndex: number, iconEndIndex: number, verticalIconStartIndex: number, verticalIconEndIndex: number): CollisionArrays; + deserializeCollisionBoxes(collisionBoxArray: CollisionBoxArray): void; + hasTextData(): boolean; + hasIconData(): boolean; + hasDebugData(): CollisionBuffers; + hasTextCollisionBoxData(): boolean; + hasIconCollisionBoxData(): boolean; + addIndicesForPlacedSymbol(iconOrText: SymbolBuffers, placedSymbolIndex: number): void; + getSortedSymbolIndexes(angle: number): any[]; + addToSortKeyRanges(symbolInstanceIndex: number, sortKey: number): void; + sortFeatures(angle: number): void; +} +export interface SymbolsByKeyEntry { + index?: KDBush; + positions?: { + x: number; + y: number; + }[]; + crossTileIDs: number[]; +} +export declare class TileLayerIndex { + tileID: OverscaledTileID; + bucketInstanceId: number; + _symbolsByKey: Record; + constructor(tileID: OverscaledTileID, symbolInstances: SymbolInstanceArray, bucketInstanceId: number); + getScaledCoordinates(symbolInstance: SymbolInstance, childTileID: OverscaledTileID): { + x: number; + y: number; + }; + findMatches(symbolInstances: SymbolInstanceArray, newTileID: OverscaledTileID, zoomCrossTileIDs: { + [crossTileID: number]: boolean; + }): void; + getCrossTileIDsLists(): number[][]; +} +export declare class CrossTileIDs { + maxCrossTileID: number; + constructor(); + generate(): number; +} +export declare class CrossTileSymbolLayerIndex { + indexes: { + [zoom in string | number]: { + [tileId in string | number]: TileLayerIndex; + }; + }; + usedCrossTileIDs: { + [zoom in string | number]: { + [crossTileID: number]: boolean; + }; + }; + lng: number; + constructor(); + handleWrapJump(lng: number): void; + addBucket(tileID: OverscaledTileID, bucket: SymbolBucket, crossTileIDs: CrossTileIDs): boolean; + removeBucketCrossTileIDs(zoom: string | number, removedBucket: TileLayerIndex): void; + removeStaleBuckets(currentIDs: { + [k in string | number]: boolean; + }): boolean; +} +export declare class CrossTileSymbolIndex { + layerIndexes: { + [layerId: string]: CrossTileSymbolLayerIndex; + }; + crossTileIDs: CrossTileIDs; + maxBucketInstanceId: number; + bucketsInCurrentPlacement: { + [_: number]: boolean; + }; + constructor(); + addLayer(styleLayer: StyleLayer, tiles: Array, lng: number): boolean; + pruneUnusedLayers(usedLayers: Array): void; +} +/** + * A dash entry + */ +export type DashEntry = { + y: number; + height: number; + width: number; +}; +export declare class LineAtlas { + width: number; + height: number; + nextRow: number; + bytes: number; + data: Uint8Array; + dashEntry: { + [_: string]: DashEntry; + }; + dirty: boolean; + texture: WebGLTexture; + constructor(width: number, height: number); + /** + * Get or create a dash line pattern. + * + * @param dasharray - the key (represented by numbers) to get the dash texture + * @param round - whether to add circle caps in between dash segments + * @returns position of dash texture in {@link DashEntry} + */ + getDash(dasharray: Array, round: boolean): DashEntry; + getDashRanges(dasharray: Array, lineAtlasWidth: number, stretch: number): any[]; + addRoundDash(ranges: any, stretch: number, n: number): void; + addRegularDash(ranges: any): void; + addDash(dasharray: Array, round: boolean): DashEntry; + bind(context: Context): void; +} +declare function loadGlyphRange(fontstack: string, range: number, urlTemplate: string, requestManager: RequestManager, callback: Callback<{ + [_: number]: StyleGlyph | null; +}>): void; +export type Entry = { + glyphs: { + [id: number]: StyleGlyph | null; + }; + requests: { + [range: number]: Array>; + }; + ranges: { + [range: number]: boolean | null; + }; + tinySDF?: TinySDF; +}; +export declare class GlyphManager { + requestManager: RequestManager; + localIdeographFontFamily: string; + entries: { + [_: string]: Entry; + }; + url: string; + static loadGlyphRange: typeof loadGlyphRange; + static TinySDF: typeof TinySDF; + constructor(requestManager: RequestManager, localIdeographFontFamily?: string | null); + setURL(url?: string | null): void; + getGlyphs(glyphs: { + [stack: string]: Array; + }, callback: Callback<{ + [stack: string]: { + [id: number]: StyleGlyph; + }; + }>): void; + _doesCharSupportLocalGlyph(id: number): boolean; + _tinySDF(entry: Entry, stack: string, id: number): StyleGlyph; +} +export type PoolObject = { + id: number; + fbo: Framebuffer; + texture: Texture; + stamp: number; + inUse: boolean; +}; +export declare class RenderPool { + private readonly _context; + private readonly _size; + private readonly _tileSize; + private _objects; + /** + * An index array of recently used pool objects. + * Items that are used recently are last in the array + */ + private _recentlyUsed; + private _stamp; + constructor(_context: Context, _size: number, _tileSize: number); + destruct(): void; + private _createObject; + getObjectForId(id: number): PoolObject; + useObject(obj: PoolObject): void; + stampObject(obj: PoolObject): void; + getOrCreateFreeObject(): PoolObject; + freeObject(obj: PoolObject): void; + freeAllObjects(): void; + isFull(): boolean; +} +export declare class RenderToTexture { + painter: Painter; + terrain: Terrain; + pool: RenderPool; + /** + * coordsDescendingInv contains a list of all tiles which should be rendered for one render-to-texture tile + * e.g. render 4 raster-tiles with size 256px to the 512px render-to-texture tile + */ + _coordsDescendingInv: { + [_: string]: { + [_: string]: Array; + }; + }; + /** + * create a string representation of all to tiles rendered to render-to-texture tiles + * this string representation is used to check if tile should be re-rendered. + */ + _coordsDescendingInvStr: { + [_: string]: { + [_: string]: string; + }; + }; + /** + * store for render-stacks + * a render stack is a set of layers which should be rendered into one texture + * every stylesheet can have multiple stacks. A new stack is created if layers which should + * not rendered to texture sit inbetween layers which should rendered to texture. e.g. hillshading or symbols + */ + _stacks: Array>; + /** + * remember the previous processed layer to check if a new stack is needed + */ + _prevType: string; + /** + * a list of tiles that can potentially rendered + */ + _renderableTiles: Array; + /** + * a list of tiles that should be rendered to screen in the next render-call + */ + _rttTiles: Array; + /** + * a list of all layer-ids which should be rendered + */ + _renderableLayerIds: Array; + constructor(painter: Painter, terrain: Terrain); + destruct(): void; + getTexture(tile: Tile): Texture; + prepareForRender(style: Style, zoom: number): void; + /** + * due that switching textures is relatively slow, the render + * layer-by-layer context is not practicable. To bypass this problem + * this lines of code stack all layers and later render all at once. + * Because of the stylesheet possibility to mixing render-to-texture layers + * and 'live'-layers (f.e. symbols) it is necessary to create more stacks. For example + * a symbol-layer is in between of fill-layers. + * @param layer - the layer to render + * @returns if true layer is rendered to texture, otherwise false + */ + renderLayer(layer: StyleLayer): boolean; +} +export type RenderPass = "offscreen" | "opaque" | "translucent"; +export type PainterOptions = { + showOverdrawInspector: boolean; + showTileBoundaries: boolean; + showPadding: boolean; + rotating: boolean; + zooming: boolean; + moving: boolean; + fadeDuration: number; +}; +export declare class Painter { + context: Context; + transform: Transform; + renderToTexture: RenderToTexture; + _tileTextures: { + [_: number]: Array; + }; + numSublayers: number; + depthEpsilon: number; + emptyProgramConfiguration: ProgramConfiguration; + width: number; + height: number; + pixelRatio: number; + tileExtentBuffer: VertexBuffer; + tileExtentSegments: SegmentVector; + debugBuffer: VertexBuffer; + debugSegments: SegmentVector; + rasterBoundsBuffer: VertexBuffer; + rasterBoundsSegments: SegmentVector; + viewportBuffer: VertexBuffer; + viewportSegments: SegmentVector; + quadTriangleIndexBuffer: IndexBuffer; + tileBorderIndexBuffer: IndexBuffer; + _tileClippingMaskIDs: { + [_: string]: number; + }; + stencilClearMode: StencilMode; + style: Style; + options: PainterOptions; + lineAtlas: LineAtlas; + imageManager: ImageManager; + glyphManager: GlyphManager; + depthRangeFor3D: DepthRangeType; + opaquePassCutoff: number; + renderPass: RenderPass; + currentLayer: number; + currentStencilSource: string; + nextStencilID: number; + id: string; + _showOverdrawInspector: boolean; + cache: { + [_: string]: Program; + }; + crossTileSymbolIndex: CrossTileSymbolIndex; + symbolFadeChange: number; + debugOverlayTexture: Texture; + debugOverlayCanvas: HTMLCanvasElement; + terrainFacilitator: { + dirty: boolean; + matrix: mat4; + renderTime: number; + }; + constructor(gl: WebGLRenderingContext | WebGL2RenderingContext, transform: Transform); + resize(width: number, height: number, pixelRatio: number): void; + setup(): void; + clearStencil(): void; + _renderTileClippingMasks(layer: StyleLayer, tileIDs: Array): void; + stencilModeFor3D(): StencilMode; + stencilModeForClipping(tileID: OverscaledTileID): StencilMode; + stencilConfigForOverlap(tileIDs: Array): [ + { + [_: number]: Readonly; + }, + Array + ]; + colorModeForRenderPass(): Readonly; + depthModeForSublayer(n: number, mask: DepthMaskType, func?: DepthFuncType | null): Readonly; + opaquePassEnabledForLayer(): boolean; + render(style: Style, options: PainterOptions): void; + renderLayer(painter: Painter, sourceCache: SourceCache, layer: StyleLayer, coords: Array): void; + /** + * Transform a matrix to incorporate the *-translate and *-translate-anchor properties into it. + * @param inViewportPixelUnitsUnits - True when the units accepted by the matrix are in viewport pixels instead of tile units. + * @returns matrix + */ + translatePosMatrix(matrix: mat4, tile: Tile, translate: [ + number, + number + ], translateAnchor: "map" | "viewport", inViewportPixelUnitsUnits?: boolean): mat4; + saveTileTexture(texture: Texture): void; + getTileTexture(size: number): Texture; + /** + * Checks whether a pattern image is needed, and if it is, whether it is not loaded. + * + * @returns true if a needed image is missing and rendering needs to be skipped. + */ + isPatternMissing(image?: CrossFaded | null): boolean; + useProgram(name: string, programConfiguration?: ProgramConfiguration | null): Program; + setCustomLayerDefaults(): void; + setBaseState(): void; + initDebugOverlayCanvas(): void; + destroy(): void; + overLimit(): boolean; +} +export declare class TerrainSourceCache extends Evented { + /** + * source-cache for the raster-dem source. + */ + sourceCache: SourceCache; + /** + * stores all render-to-texture tiles. + */ + _tiles: { + [_: string]: Tile; + }; + /** + * contains a list of tileID-keys for the current scene. (only for performance) + */ + _renderableTilesKeys: Array; + /** + * raster-dem-tile for a TileID cache. + */ + _sourceTileCache: { + [_: string]: string; + }; + /** + * minimum zoomlevel to render the terrain. + */ + minzoom: number; + /** + * maximum zoomlevel to render the terrain. + */ + maxzoom: number; + /** + * render-to-texture tileSize in scene. + */ + tileSize: number; + /** + * raster-dem tiles will load for performance the actualZoom - deltaZoom zoom-level. + */ + deltaZoom: number; + constructor(sourceCache: SourceCache); + destruct(): void; + /** + * Load Terrain Tiles, create internal render-to-texture tiles, free GPU memory. + * @param transform - the operation to do + * @param terrain - the terrain + */ + update(transform: Transform, terrain: Terrain): void; + /** + * Free render to texture cache + * @param tileID - optional, free only corresponding to tileID. + */ + freeRtt(tileID?: OverscaledTileID): void; + /** + * get a list of tiles, which are loaded and should be rendered in the current scene + * @returns the renderable tiles + */ + getRenderableTiles(): Array; + /** + * get terrain tile by the TileID key + * @param id - the tile id + * @returns the tile + */ + getTileByID(id: string): Tile; + /** + * Searches for the corresponding current renderable terrain-tiles + * @param tileID - the tile to look for + * @returns the tiles that were found + */ + getTerrainCoords(tileID: OverscaledTileID): Record; + /** + * find the covering raster-dem tile + * @param tileID - the tile to look for + * @param searchForDEM - Optinal parameter to search for (parent) souretiles with loaded dem. + * @returns the tile + */ + getSourceTile(tileID: OverscaledTileID, searchForDEM?: boolean): Tile; + /** + * get a list of tiles, loaded after a spezific time. This is used to update depth & coords framebuffers. + * @param time - the time + * @returns the relevant tiles + */ + tilesAfterTime(time?: number): Array; +} +/** + * @internal + * A terrain GPU related object + */ +export type TerrainData = { + "u_depth": number; + "u_terrain": number; + "u_terrain_dim": number; + "u_terrain_matrix": mat4; + "u_terrain_unpack": number[]; + "u_terrain_exaggeration": number; + texture: WebGLTexture; + depthTexture: WebGLTexture; + tile: Tile; +}; +/** + * @internal + * A terrain mesh object + */ +export type TerrainMesh = { + indexBuffer: IndexBuffer; + vertexBuffer: VertexBuffer; + segments: SegmentVector; +}; +export declare class Terrain { + /** + * The style this terrain crresponds to + */ + painter: Painter; + /** + * the sourcecache this terrain is based on + */ + sourceCache: TerrainSourceCache; + /** + * the TerrainSpecification object passed to this instance + */ + options: TerrainSpecification; + /** + * define the meshSize per tile. + */ + meshSize: number; + /** + * multiplicator for the elevation. Used to make terrain more "extreme". + */ + exaggeration: number; + /** + * to not see pixels in the render-to-texture tiles it is good to render them bigger + * this number is the multiplicator (must be a power of 2) for the current tileSize. + * So to get good results with not too much memory footprint a value of 2 should be fine. + */ + qualityFactor: number; + /** + * holds the framebuffer object in size of the screen to render the coords & depth into a texture. + */ + _fbo: Framebuffer; + _fboCoordsTexture: Texture; + _fboDepthTexture: Texture; + _emptyDepthTexture: Texture; + /** + * GL Objects for the terrain-mesh + * The mesh is a regular mesh, which has the advantage that it can be reused for all tiles. + */ + _mesh: TerrainMesh; + /** + * coords index contains a list of tileID.keys. This index is used to identify + * the tile via the alpha-cannel in the coords-texture. + * As the alpha-channel has 1 Byte a max of 255 tiles can rendered without an error. + */ + coordsIndex: Array; + /** + * tile-coords encoded in the rgb channel, _coordsIndex is in the alpha-channel. + */ + _coordsTexture: Texture; + /** + * accuracy of the coords. 2 * tileSize should be enoughth. + */ + _coordsTextureSize: number; + /** + * variables for an empty dem texture, which is used while the raster-dem tile is loading. + */ + _emptyDemUnpack: number[]; + _emptyDemTexture: Texture; + _emptyDemMatrix: mat4; + /** + * as of overzooming of raster-dem tiles in high zoomlevels, this cache contains + * matrices to transform from vector-tile coords to raster-dem-tile coords. + */ + _demMatrixCache: { + [_: string]: { + matrix: mat4; + coord: OverscaledTileID; + }; + }; + constructor(painter: Painter, sourceCache: SourceCache, options: TerrainSpecification); + /** + * get the elevation-value from original dem-data for a given tile-coordinate + * @param tileID - the tile to get elevation for + * @param x - between 0 .. EXTENT + * @param y - between 0 .. EXTENT + * @param extent - optional, default 8192 + * @returns the elevation + */ + getDEMElevation(tileID: OverscaledTileID, x: number, y: number, extent?: number): number; + /** + * Get the elevation for given {@link LngLat} in respect of exaggeration. + * @param lnglat - the location + * @param zoom - the zoom + * @returns the elevation + */ + getElevationForLngLatZoom(lnglat: LngLat, zoom: number): number; + /** + * Get the elevation for given coordinate in respect of exaggeration. + * @param tileID - the tile id + * @param x - between 0 .. EXTENT + * @param y - between 0 .. EXTENT + * @param extent - optional, default 8192 + * @returns the elevation + */ + getElevation(tileID: OverscaledTileID, x: number, y: number, extent?: number): number; + /** + * returns a Terrain Object for a tile. Unless the tile corresponds to data (e.g. tile is loading), return a flat dem object + * @param tileID - the tile to get the terrain for + * @returns the terrain data to use in the program + */ + getTerrainData(tileID: OverscaledTileID): TerrainData; + /** + * get a framebuffer as big as the map-div, which will be used to render depth & coords into a texture + * @param texture - the texture + * @returns the frame buffer + */ + getFramebuffer(texture: string): Framebuffer; + /** + * create coords texture, needed to grab coordinates from canvas + * encode coords coordinate into 4 bytes: + * - 8 lower bits for x + * - 8 lower bits for y + * - 4 higher bits for x + * - 4 higher bits for y + * - 8 bits for coordsIndex (1 .. 255) (= number of terraintile), is later setted in draw_terrain uniform value + * @returns the texture + */ + getCoordsTexture(): Texture; + /** + * Reads a pixel from the coords-framebuffer and translate this to mercator. + * @param p - Screen-Coordinate + * @returns mercator coordinate for a screen pixel + */ + pointCoordinate(p: Point): MercatorCoordinate; + /** + * create a regular mesh which will be used by all terrain-tiles + * @returns the created regular mesh + */ + getTerrainMesh(): TerrainMesh; + /** + * Calculates a height of the frame around the terrain-mesh to avoid stiching between + * tile boundaries in different zoomlevels. + * @param zoom - current zoomlevel + * @returns the elevation delta in meters + */ + getMeshFrameDelta(zoom: number): number; + getMinTileElevationForLngLatZoom(lnglat: LngLat, zoom: number): number; + /** + * Get the minimum and maximum elevation contained in a tile. This includes any + * exaggeration included in the terrain. + * + * @param tileID - ID of the tile to be used as a source for the min/max elevation + * @returns the minimum and maximum elevation found in the tile, including the terrain's + * exaggeration + */ + getMinMaxElevation(tileID: OverscaledTileID): { + minElevation: number | null; + maxElevation: number | null; + }; + _getOverscaledTileIDFromLngLatZoom(lnglat: LngLat, zoom: number): { + tileID: OverscaledTileID; + mercatorX: number; + mercatorY: number; + }; + _allowMercatorOverflow(p: Point, mercatorX: number): number; +} +export declare class Transform { + tileSize: number; + tileZoom: number; + lngRange: [ + number, + number + ]; + latRange: [ + number, + number + ]; + maxValidLatitude: number; + scale: number; + width: number; + height: number; + angle: number; + rotationMatrix: mat2; + pixelsToGLUnits: [ + number, + number + ]; + cameraToCenterDistance: number; + mercatorMatrix: mat4; + projMatrix: mat4; + invProjMatrix: mat4; + alignedProjMatrix: mat4; + pixelMatrix: mat4; + pixelMatrix3D: mat4; + pixelMatrixInverse: mat4; + glCoordMatrix: mat4; + labelPlaneMatrix: mat4; + _fov: number; + _pitch: number; + _zoom: number; + _unmodified: boolean; + _renderWorldCopies: boolean; + _minZoom: number; + _maxZoom: number; + _minPitch: number; + _maxPitch: number; + _center: LngLat; + _elevation: number; + _pixelPerMeter: number; + _edgeInsets: EdgeInsets; + _constraining: boolean; + _posMatrixCache: { + [_: string]: mat4; + }; + _alignedPosMatrixCache: { + [_: string]: mat4; + }; + _minEleveationForCurrentTile: number; + constructor(minZoom?: number, maxZoom?: number, minPitch?: number, maxPitch?: number, renderWorldCopies?: boolean); + clone(): Transform; + apply(that: Transform): void; + get minZoom(): number; + set minZoom(zoom: number); + get maxZoom(): number; + set maxZoom(zoom: number); + get minPitch(): number; + set minPitch(pitch: number); + get maxPitch(): number; + set maxPitch(pitch: number); + get renderWorldCopies(): boolean; + set renderWorldCopies(renderWorldCopies: boolean); + get worldSize(): number; + get centerOffset(): Point; + get size(): Point; + get bearing(): number; + set bearing(bearing: number); + get pitch(): number; + set pitch(pitch: number); + get fov(): number; + set fov(fov: number); + get zoom(): number; + set zoom(zoom: number); + get center(): LngLat; + set center(center: LngLat); + get elevation(): number; + set elevation(elevation: number); + get padding(): PaddingOptions; + set padding(padding: PaddingOptions); + /** + * The center of the screen in pixels with the top-left corner being (0,0) + * and +y axis pointing downwards. This accounts for padding. + */ + get centerPoint(): Point; + /** + * Returns if the padding params match + * + * @param padding - the padding to check against + * @returns true if they are equal, false otherwise + */ + isPaddingEqual(padding: PaddingOptions): boolean; + /** + * Helper method to update edge-insets in place + * + * @param start - the starting padding + * @param target - the target padding + * @param t - the step/weight + */ + interpolatePadding(start: PaddingOptions, target: PaddingOptions, t: number): void; + /** + * Return a zoom level that will cover all tiles the transform + * @param options - the options + * @returns zoom level An integer zoom level at which all tiles will be visible. + */ + coveringZoomLevel(options: { + /** + * Target zoom level. If true, the value will be rounded to the closest integer. Otherwise the value will be floored. + */ + roundZoom?: boolean; + /** + * Tile size, expressed in screen pixels. + */ + tileSize: number; + }): number; + /** + * Return any "wrapped" copies of a given tile coordinate that are visible + * in the current view. + */ + getVisibleUnwrappedCoordinates(tileID: CanonicalTileID): UnwrappedTileID[]; + /** + * Return all coordinates that could cover this transform for a covering + * zoom level. + * @param options - the options + * @returns OverscaledTileIDs + */ + coveringTiles(options: { + tileSize: number; + minzoom?: number; + maxzoom?: number; + roundZoom?: boolean; + reparseOverscaled?: boolean; + renderWorldCopies?: boolean; + terrain?: Terrain; + }): Array; + resize(width: number, height: number): void; + get unmodified(): boolean; + zoomScale(zoom: number): number; + scaleZoom(scale: number): number; + project(lnglat: LngLat): Point; + unproject(point: Point): LngLat; + get point(): Point; + /** + * get the camera position in LngLat and altitudes in meter + * @returns An object with lngLat & altitude. + */ + getCameraPosition(): { + lngLat: LngLat; + altitude: number; + }; + /** + * This method works in combination with freezeElevation activated. + * freezeElevtion is enabled during map-panning because during this the camera should sit in constant height. + * After panning finished, call this method to recalculate the zoomlevel for the current camera-height in current terrain. + * @param terrain - the terrain + */ + recalculateZoom(terrain: Terrain): void; + setLocationAtPoint(lnglat: LngLat, point: Point): void; + /** + * Given a location, return the screen point that corresponds to it + * @param lnglat - location + * @param terrain - optional terrain + * @returns screen point + */ + locationPoint(lnglat: LngLat, terrain?: Terrain): Point; + /** + * Given a point on screen, return its lnglat + * @param p - screen point + * @param terrain - optional terrain + * @returns lnglat location + */ + pointLocation(p: Point, terrain?: Terrain): LngLat; + /** + * Given a geographical lnglat, return an unrounded + * coordinate that represents it at this transform's zoom level. + * @param lnglat - the location + * @returns The mercator coordinate + */ + locationCoordinate(lnglat: LngLat): MercatorCoordinate; + /** + * Given a Coordinate, return its geographical position. + * @param coord - mercator coordivates + * @returns lng and lat + */ + coordinateLocation(coord: MercatorCoordinate): LngLat; + /** + * Given a Point, return its mercator coordinate. + * @param p - the point + * @param terrain - optional terrain + * @returns lnglat + */ + pointCoordinate(p: Point, terrain?: Terrain): MercatorCoordinate; + /** + * Given a coordinate, return the screen point that corresponds to it + * @param coord - the coordinates + * @param elevation - the elevation + * @param pixelMatrix - the pixel matrix + * @returns screen point + */ + coordinatePoint(coord: MercatorCoordinate, elevation?: number, pixelMatrix?: mat4): Point; + /** + * Returns the map's geographical bounds. When the bearing or pitch is non-zero, the visible region is not + * an axis-aligned rectangle, and the result is the smallest bounds that encompasses the visible region. + * @returns Returns a {@link LngLatBounds} object describing the map's geographical bounds. + */ + getBounds(): LngLatBounds; + /** + * Returns the maximum geographical bounds the map is constrained to, or `null` if none set. + * @returns max bounds + */ + getMaxBounds(): LngLatBounds | null; + /** + * Calculate pixel height of the visible horizon in relation to map-center (e.g. height/2), + * multiplied by a static factor to simulate the earth-radius. + * The calculated value is the horizontal line from the camera-height to sea-level. + * @returns Horizon above center in pixels. + */ + getHorizon(): number; + /** + * Sets or clears the map's geographical constraints. + * @param bounds - A {@link LngLatBounds} object describing the new geographic boundaries of the map. + */ + setMaxBounds(bounds?: LngLatBounds | null): void; + /** + * Calculate the posMatrix that, given a tile coordinate, would be used to display the tile on a map. + * @param unwrappedTileID - the tile ID + */ + calculatePosMatrix(unwrappedTileID: UnwrappedTileID, aligned?: boolean): mat4; + customLayerMatrix(): mat4; + _constrain(): void; + _calcMatrices(): void; + maxPitchScaleFactor(): number; + /** + * The camera looks at the map from a 3D (lng, lat, altitude) location. Let's use `cameraLocation` + * as the name for the location under the camera and on the surface of the earth (lng, lat, 0). + * `cameraPoint` is the projected position of the `cameraLocation`. + * + * This point is useful to us because only fill-extrusions that are between `cameraPoint` and + * the query point on the surface of the earth can extend and intersect the query. + * + * When the map is not pitched the `cameraPoint` is equivalent to the center of the map because + * the camera is right above the center of the map. + */ + getCameraPoint(): Point; + /** + * When the map is pitched, some of the 3D features that intersect a query will not intersect + * the query at the surface of the earth. Instead the feature may be closer and only intersect + * the query because it extrudes into the air. + * @param queryGeometry - For point queries, the line from the query point to the "camera point", + * for other geometries, the envelope of the query geometry and the "camera point" + * @returns a geometry that includes all of the original query as well as all possible ares of the + * screen where the *base* of a visible extrusion could be. + * + */ + getCameraQueryGeometry(queryGeometry: Array): Array; +} +export type QueryParameters = { + scale: number; + pixelPosMatrix: mat4; + transform: Transform; + tileSize: number; + queryGeometry: Array; + cameraQueryGeometry: Array; + queryPadding: number; + params: { + filter: FilterSpecification; + layers: Array; + availableImages: Array; + }; +}; +export declare class FeatureIndex { + tileID: OverscaledTileID; + x: number; + y: number; + z: number; + grid: TransferableGridIndex; + grid3D: TransferableGridIndex; + featureIndexArray: FeatureIndexArray; + promoteId?: PromoteIdSpecification; + rawTileData: ArrayBuffer; + bucketLayerIDs: Array>; + vtLayers: { + [_: string]: VectorTileLayer; + }; + sourceLayerCoder: DictionaryCoder; + constructor(tileID: OverscaledTileID, promoteId?: PromoteIdSpecification | null); + insert(feature: VectorTileFeature, geometry: Array>, featureIndex: number, sourceLayerIndex: number, bucketIndex: number, is3D?: boolean): void; + loadVTLayers(): { + [_: string]: VectorTileLayer; + }; + query(args: QueryParameters, styleLayers: { + [_: string]: StyleLayer; + }, serializedLayers: { + [_: string]: any; + }, sourceFeatureState: SourceFeatureState): { + [_: string]: Array<{ + featureIndex: number; + feature: GeoJSONFeature; + }>; + }; + loadMatchingFeature(result: { + [_: string]: Array<{ + featureIndex: number; + feature: GeoJSONFeature; + intersectionZ?: boolean | number; + }>; + }, bucketIndex: number, sourceLayerIndex: number, featureIndex: number, filter: FeatureFilter, filterLayerIDs: Array, availableImages: Array, styleLayers: { + [_: string]: StyleLayer; + }, serializedLayers: { + [_: string]: any; + }, sourceFeatureState?: SourceFeatureState, intersectionTest?: (feature: VectorTileFeature, styleLayer: StyleLayer, featureState: any, id: string | number | void) => boolean | number): void; + lookupSymbolFeatures(symbolFeatureIndexes: Array, serializedLayers: { + [_: string]: StyleLayer; + }, bucketIndex: number, sourceLayerIndex: number, filterSpec: FilterSpecification, filterLayerIDs: Array, availableImages: Array, styleLayers: { + [_: string]: StyleLayer; + }): {}; + hasLayer(id: string): boolean; + getId(feature: VectorTileFeature, sourceLayerId: string): string | number; +} +/** + * The tile's state, can be: + * - `loading` Tile data is in the process of loading. + * - `loaded` Tile data has been loaded. Tile can be rendered. + * - `reloading` Tile data has been loaded and is being updated. Tile can be rendered. + * - `unloaded` Tile data has been deleted. + * - `errored` Tile data was not loaded because of an error. + * - `expired` Tile data was previously loaded, but has expired per its HTTP headers and is in the process of refreshing. + */ +export type TileState = "loading" | "loaded" | "reloading" | "unloaded" | "errored" | "expired"; +export declare class Tile { + tileID: OverscaledTileID; + uid: number; + uses: number; + tileSize: number; + buckets: { + [_: string]: Bucket; + }; + latestFeatureIndex: FeatureIndex; + latestRawTileData: ArrayBuffer; + imageAtlas: ImageAtlas; + imageAtlasTexture: Texture; + glyphAtlasImage: AlphaImage; + glyphAtlasTexture: Texture; + expirationTime: any; + expiredRequestCount: number; + state: TileState; + timeAdded: number; + fadeEndTime: number; + collisionBoxArray: CollisionBoxArray; + redoWhenDone: boolean; + showCollisionBoxes: boolean; + placementSource: any; + actor: Actor; + vtLayers: { + [_: string]: VectorTileLayer; + }; + neighboringTiles: any; + dem: DEMData; + demMatrix: mat4; + aborted: boolean; + needsHillshadePrepare: boolean; + needsTerrainPrepare: boolean; + request: Cancelable; + texture: any; + fbo: Framebuffer; + demTexture: Texture; + refreshedUponExpiration: boolean; + reloadCallback: any; + resourceTiming: Array; + queryPadding: number; + symbolFadeHoldUntil: number; + hasSymbolBuckets: boolean; + hasRTLText: boolean; + dependencies: any; + rtt: Array<{ + id: number; + stamp: number; + }>; + rttCoords: { + [_: string]: string; + }; + /** + * @param tileID - the tile ID + * @param size - The tile size + */ + constructor(tileID: OverscaledTileID, size: number); + registerFadeDuration(duration: number): void; + wasRequested(): boolean; + clearTextures(painter: any): void; + /** + * Given a data object with a 'buffers' property, load it into + * this tile's elementGroups and buffers properties and set loaded + * to true. If the data is null, like in the case of an empty + * GeoJSON tile, no-op but still set loaded to true. + * @param data - The data from the worker + * @param painter - the painter + * @param justReloaded - `true` to just reload + */ + loadVectorData(data: WorkerTileResult, painter: any, justReloaded?: boolean | null): void; + /** + * Release any data or WebGL resources referenced by this tile. + */ + unloadVectorData(): void; + getBucket(layer: StyleLayer): Bucket; + upload(context: Context): void; + prepare(imageManager: ImageManager): void; + queryRenderedFeatures(layers: { + [_: string]: StyleLayer; + }, serializedLayers: { + [_: string]: any; + }, sourceFeatureState: SourceFeatureState, queryGeometry: Array, cameraQueryGeometry: Array, scale: number, params: { + filter: FilterSpecification; + layers: Array; + availableImages: Array; + }, transform: Transform, maxPitchScaleFactor: number, pixelPosMatrix: mat4): { + [_: string]: Array<{ + featureIndex: number; + feature: GeoJSONFeature; + }>; + }; + querySourceFeatures(result: Array, params?: { + sourceLayer?: string; + filter?: FilterSpecification; + validate?: boolean; + }): void; + hasData(): boolean; + patternsLoaded(): boolean; + setExpiryData(data: ExpiryData): void; + getExpiryTimeout(): number; + setFeatureState(states: LayerFeatureStates, painter: any): void; + holdingForFade(): boolean; + symbolFadeFinished(): boolean; + clearFadeHold(): void; + setHoldDuration(duration: number): void; + setDependencies(namespace: string, dependencies: Array): void; + hasDependency(namespaces: Array, keys: Array): boolean; +} +export type FeatureStates = { + [featureId: string]: FeatureState; +}; +export type LayerFeatureStates = { + [layer: string]: FeatureStates; +}; +export declare class SourceFeatureState { + state: LayerFeatureStates; + stateChanges: LayerFeatureStates; + deletedStates: {}; + constructor(); + updateState(sourceLayer: string, featureId: number | string, newState: any): void; + removeFeatureState(sourceLayer: string, featureId?: number | string, key?: string): void; + getState(sourceLayer: string, featureId: number | string): any; + initializeTileState(tile: Tile, painter: any): void; + coalesceChanges(tiles: { + [_ in any]: Tile; + }, painter: any): void; +} +export type BinderUniform = { + name: string; + property: string; + binding: Uniform; +}; +/** + * `Binder` is the interface definition for the strategies for constructing, + * uploading, and binding paint property data as GLSL attributes. Most style- + * spec properties have a 1:1 relationship to shader attribute/uniforms, but + * some require multiple values per feature to be passed to the GPU, and in + * those cases we bind multiple attributes/uniforms. + * + * It has three implementations, one for each of the three strategies we use: + * + * * For _constant_ properties -- those whose value is a constant, or the constant + * result of evaluating a camera expression at a particular camera position -- we + * don't need a vertex attribute buffer, and instead use a uniform. + * * For data expressions, we use a vertex buffer with a single attribute value, + * the evaluated result of the source function for the given feature. + * * For composite expressions, we use a vertex buffer with two attributes: min and + * max values covering the range of zooms at which we expect the tile to be + * displayed. These values are calculated by evaluating the composite expression for + * the given feature at strategically chosen zoom levels. In addition to this + * attribute data, we also use a uniform value which the shader uses to interpolate + * between the min and max value at the final displayed zoom level. The use of a + * uniform allows us to cheaply update the value on every frame. + * + * Note that the shader source varies depending on whether we're using a uniform or + * attribute. We dynamically compile shaders at runtime to accommodate this. + */ +export interface AttributeBinder { + populatePaintArray(length: number, feature: Feature, imagePositions: { + [_: string]: ImagePosition; + }, canonical?: CanonicalTileID, formattedSection?: FormattedSection): void; + updatePaintArray(start: number, length: number, feature: Feature, featureState: FeatureState, imagePositions: { + [_: string]: ImagePosition; + }): void; + upload(a: Context): void; + destroy(): void; +} +export interface UniformBinder { + uniformNames: Array; + setUniform(uniform: Uniform, globals: GlobalProperties, currentValue: PossiblyEvaluatedPropertyValue, uniformName: string): void; + getBinding(context: Context, location: WebGLUniformLocation, name: string): Partial>; +} +export declare class ProgramConfiguration { + binders: { + [_: string]: AttributeBinder | UniformBinder; + }; + cacheKey: string; + _buffers: Array; + constructor(layer: TypedStyleLayer, zoom: number, filterProperties: (_: string) => boolean); + getMaxValue(property: string): number; + populatePaintArrays(newLength: number, feature: Feature, imagePositions: { + [_: string]: ImagePosition; + }, canonical?: CanonicalTileID, formattedSection?: FormattedSection): void; + setConstantPatternPositions(posTo: ImagePosition, posFrom: ImagePosition): void; + updatePaintArrays(featureStates: FeatureStates, featureMap: FeaturePositionMap, vtLayer: VectorTileLayer, layer: TypedStyleLayer, imagePositions: { + [_: string]: ImagePosition; + }): boolean; + defines(): Array; + getBinderAttributes(): Array; + getBinderUniforms(): Array; + getPaintVertexBuffers(): Array; + getUniforms(context: Context, locations: UniformLocations): Array; + setUniforms(context: Context, binderUniforms: Array, properties: any, globals: GlobalProperties): void; + updatePaintBuffers(crossfade?: CrossfadeParameters): void; + upload(context: Context): void; + destroy(): void; +} +export declare class ProgramConfigurationSet { + programConfigurations: { + [_: string]: ProgramConfiguration; + }; + needsUpload: boolean; + _featureMap: FeaturePositionMap; + _bufferOffset: number; + constructor(layers: ReadonlyArray, zoom: number, filterProperties?: (_: string) => boolean); + populatePaintArrays(length: number, feature: Feature, index: number, imagePositions: { + [_: string]: ImagePosition; + }, canonical: CanonicalTileID, formattedSection?: FormattedSection): void; + updatePaintArrays(featureStates: FeatureStates, vtLayer: VectorTileLayer, layers: ReadonlyArray, imagePositions: { + [_: string]: ImagePosition; + }): void; + get(layerId: string): ProgramConfiguration; + upload(context: Context): void; + destroy(): void; +} +export type TerrainPreludeUniformsType = { + "u_depth": Uniform1i; + "u_terrain": Uniform1i; + "u_terrain_dim": Uniform1f; + "u_terrain_matrix": UniformMatrix4f; + "u_terrain_unpack": Uniform4f; + "u_terrain_exaggeration": Uniform1f; +}; +export type DrawMode = WebGLRenderingContextBase["LINES"] | WebGLRenderingContextBase["TRIANGLES"] | WebGL2RenderingContext["LINE_STRIP"]; +export declare class Program { + program: WebGLProgram; + attributes: { + [_: string]: number; + }; + numAttributes: number; + fixedUniforms: Us; + terrainUniforms: TerrainPreludeUniformsType; + binderUniforms: Array; + failedToCreate: boolean; + constructor(context: Context, source: { + fragmentSource: string; + vertexSource: string; + staticAttributes: Array; + staticUniforms: Array; + }, configuration: ProgramConfiguration, fixedUniforms: (b: Context, a: UniformLocations) => Us, showOverdrawInspector: boolean, terrain: Terrain); + draw(context: Context, drawMode: DrawMode, depthMode: Readonly, stencilMode: Readonly, colorMode: Readonly, cullFaceMode: Readonly, uniformValues: UniformValues, terrain: TerrainData, layerID: string, layoutVertexBuffer: VertexBuffer, indexBuffer: IndexBuffer, segments: SegmentVector, currentProperties?: any, zoom?: number | null, configuration?: ProgramConfiguration | null, dynamicLayoutBuffer?: VertexBuffer | null, dynamicLayoutBuffer2?: VertexBuffer | null, dynamicLayoutBuffer3?: VertexBuffer | null): void; +} +export declare class VertexArrayObject { + context: Context; + boundProgram: Program; + boundLayoutVertexBuffer: VertexBuffer; + boundPaintVertexBuffers: Array; + boundIndexBuffer: IndexBuffer; + boundVertexOffset: number; + boundDynamicVertexBuffer: VertexBuffer; + boundDynamicVertexBuffer2: VertexBuffer; + boundDynamicVertexBuffer3: VertexBuffer; + vao: any; + constructor(); + bind(context: Context, program: Program, layoutVertexBuffer: VertexBuffer, paintVertexBuffers: Array, indexBuffer?: IndexBuffer | null, vertexOffset?: number | null, dynamicVertexBuffer?: VertexBuffer | null, dynamicVertexBuffer2?: VertexBuffer | null, dynamicVertexBuffer3?: VertexBuffer | null): void; + freshBind(program: Program, layoutVertexBuffer: VertexBuffer, paintVertexBuffers: Array, indexBuffer?: IndexBuffer | null, vertexOffset?: number | null, dynamicVertexBuffer?: VertexBuffer | null, dynamicVertexBuffer2?: VertexBuffer | null, dynamicVertexBuffer3?: VertexBuffer | null): void; + destroy(): void; +} +/** + * @internal + * A single segment of a vector + */ +export type Segment = { + sortKey?: number; + vertexOffset: number; + primitiveOffset: number; + vertexLength: number; + primitiveLength: number; + vaos: { + [_: string]: VertexArrayObject; + }; +}; +export declare class SegmentVector { + static MAX_VERTEX_ARRAY_LENGTH: number; + segments: Array; + constructor(segments?: Array); + prepareSegment(numVertices: number, layoutVertexArray: StructArray, indexArray: StructArray, sortKey?: number): Segment; + get(): Segment[]; + destroy(): void; + static simpleSegment(vertexOffset: number, primitiveOffset: number, vertexLength: number, primitiveLength: number): SegmentVector; +} +export declare class HeatmapBucket extends CircleBucket { + layers: Array; +} +export type HeatmapPaintProps = { + "heatmap-radius": DataDrivenProperty; + "heatmap-weight": DataDrivenProperty; + "heatmap-intensity": DataConstantProperty; + "heatmap-color": ColorRampProperty; + "heatmap-opacity": DataConstantProperty; +}; +export type HeatmapPaintPropsPossiblyEvaluated = { + "heatmap-radius": PossiblyEvaluatedPropertyValue; + "heatmap-weight": PossiblyEvaluatedPropertyValue; + "heatmap-intensity": number; + "heatmap-color": ColorRampProperty; + "heatmap-opacity": number; +}; +export declare class HeatmapStyleLayer extends StyleLayer { + heatmapFbo: Framebuffer; + colorRamp: RGBAImage; + colorRampTexture: Texture; + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + createBucket(options: any): HeatmapBucket; + constructor(layer: LayerSpecification); + _handleSpecialPaintPropertyUpdate(name: string): void; + _updateColorRamp(): void; + resize(): void; + queryRadius(): number; + queryIntersectsFeature(): boolean; + hasOffscreenPass(): boolean; +} +export declare class CircleBucket implements Bucket { + index: number; + zoom: number; + overscaling: number; + layerIds: Array; + layers: Array; + stateDependentLayers: Array; + stateDependentLayerIds: Array; + layoutVertexArray: CircleLayoutArray; + layoutVertexBuffer: VertexBuffer; + indexArray: TriangleIndexArray; + indexBuffer: IndexBuffer; + hasPattern: boolean; + programConfigurations: ProgramConfigurationSet; + segments: SegmentVector; + uploaded: boolean; + constructor(options: BucketParameters); + populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID): void; + update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: { + [_: string]: ImagePosition; + }): void; + isEmpty(): boolean; + uploadPending(): boolean; + upload(context: Context): void; + destroy(): void; + addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID): void; +} +export type CircleLayoutProps = { + "circle-sort-key": DataDrivenProperty; +}; +export type CircleLayoutPropsPossiblyEvaluated = { + "circle-sort-key": PossiblyEvaluatedPropertyValue; +}; +export type CirclePaintProps = { + "circle-radius": DataDrivenProperty; + "circle-color": DataDrivenProperty; + "circle-blur": DataDrivenProperty; + "circle-opacity": DataDrivenProperty; + "circle-translate": DataConstantProperty<[ + number, + number + ]>; + "circle-translate-anchor": DataConstantProperty<"map" | "viewport">; + "circle-pitch-scale": DataConstantProperty<"map" | "viewport">; + "circle-pitch-alignment": DataConstantProperty<"map" | "viewport">; + "circle-stroke-width": DataDrivenProperty; + "circle-stroke-color": DataDrivenProperty; + "circle-stroke-opacity": DataDrivenProperty; +}; +export type CirclePaintPropsPossiblyEvaluated = { + "circle-radius": PossiblyEvaluatedPropertyValue; + "circle-color": PossiblyEvaluatedPropertyValue; + "circle-blur": PossiblyEvaluatedPropertyValue; + "circle-opacity": PossiblyEvaluatedPropertyValue; + "circle-translate": [ + number, + number + ]; + "circle-translate-anchor": "map" | "viewport"; + "circle-pitch-scale": "map" | "viewport"; + "circle-pitch-alignment": "map" | "viewport"; + "circle-stroke-width": PossiblyEvaluatedPropertyValue; + "circle-stroke-color": PossiblyEvaluatedPropertyValue; + "circle-stroke-opacity": PossiblyEvaluatedPropertyValue; +}; +export declare class CircleStyleLayer extends StyleLayer { + _unevaluatedLayout: Layout; + layout: PossiblyEvaluated; + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + constructor(layer: LayerSpecification); + createBucket(parameters: BucketParameters): CircleBucket; + queryRadius(bucket: Bucket): number; + queryIntersectsFeature(queryGeometry: Array, feature: VectorTileFeature, featureState: FeatureState, geometry: Array>, zoom: number, transform: Transform, pixelsToTileUnits: number, pixelPosMatrix: mat4): boolean; +} +export declare class FillBucket implements Bucket { + index: number; + zoom: number; + overscaling: number; + layers: Array; + layerIds: Array; + stateDependentLayers: Array; + stateDependentLayerIds: Array; + patternFeatures: Array; + layoutVertexArray: FillLayoutArray; + layoutVertexBuffer: VertexBuffer; + indexArray: TriangleIndexArray; + indexBuffer: IndexBuffer; + indexArray2: LineIndexArray; + indexBuffer2: IndexBuffer; + hasPattern: boolean; + programConfigurations: ProgramConfigurationSet; + segments: SegmentVector; + segments2: SegmentVector; + uploaded: boolean; + constructor(options: BucketParameters); + populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID): void; + update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: { + [_: string]: ImagePosition; + }): void; + addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: { + [_: string]: ImagePosition; + }): void; + isEmpty(): boolean; + uploadPending(): boolean; + upload(context: Context): void; + destroy(): void; + addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: { + [_: string]: ImagePosition; + }): void; +} +export type FillLayoutProps = { + "fill-sort-key": DataDrivenProperty; +}; +export type FillLayoutPropsPossiblyEvaluated = { + "fill-sort-key": PossiblyEvaluatedPropertyValue; +}; +export type FillPaintProps = { + "fill-antialias": DataConstantProperty; + "fill-opacity": DataDrivenProperty; + "fill-color": DataDrivenProperty; + "fill-outline-color": DataDrivenProperty; + "fill-translate": DataConstantProperty<[ + number, + number + ]>; + "fill-translate-anchor": DataConstantProperty<"map" | "viewport">; + "fill-pattern": CrossFadedDataDrivenProperty; +}; +export type FillPaintPropsPossiblyEvaluated = { + "fill-antialias": boolean; + "fill-opacity": PossiblyEvaluatedPropertyValue; + "fill-color": PossiblyEvaluatedPropertyValue; + "fill-outline-color": PossiblyEvaluatedPropertyValue; + "fill-translate": [ + number, + number + ]; + "fill-translate-anchor": "map" | "viewport"; + "fill-pattern": PossiblyEvaluatedPropertyValue>; +}; +export declare class FillStyleLayer extends StyleLayer { + _unevaluatedLayout: Layout; + layout: PossiblyEvaluated; + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + constructor(layer: LayerSpecification); + recalculate(parameters: EvaluationParameters, availableImages: Array): void; + createBucket(parameters: BucketParameters): FillBucket; + queryRadius(): number; + queryIntersectsFeature(queryGeometry: Array, feature: VectorTileFeature, featureState: FeatureState, geometry: Array>, zoom: number, transform: Transform, pixelsToTileUnits: number): boolean; + isTileClipped(): boolean; +} +export declare class FillExtrusionBucket implements Bucket { + index: number; + zoom: number; + overscaling: number; + layers: Array; + layerIds: Array; + stateDependentLayers: Array; + stateDependentLayerIds: Array; + layoutVertexArray: FillExtrusionLayoutArray; + layoutVertexBuffer: VertexBuffer; + centroidVertexArray: PosArray; + centroidVertexBuffer: VertexBuffer; + indexArray: TriangleIndexArray; + indexBuffer: IndexBuffer; + hasPattern: boolean; + programConfigurations: ProgramConfigurationSet; + segments: SegmentVector; + uploaded: boolean; + features: Array; + constructor(options: BucketParameters); + populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID): void; + addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: { + [_: string]: ImagePosition; + }): void; + update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: { + [_: string]: ImagePosition; + }): void; + isEmpty(): boolean; + uploadPending(): boolean; + upload(context: Context): void; + destroy(): void; + addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: { + [_: string]: ImagePosition; + }): void; +} +export type FillExtrusionPaintProps = { + "fill-extrusion-opacity": DataConstantProperty; + "fill-extrusion-color": DataDrivenProperty; + "fill-extrusion-translate": DataConstantProperty<[ + number, + number + ]>; + "fill-extrusion-translate-anchor": DataConstantProperty<"map" | "viewport">; + "fill-extrusion-pattern": CrossFadedDataDrivenProperty; + "fill-extrusion-height": DataDrivenProperty; + "fill-extrusion-base": DataDrivenProperty; + "fill-extrusion-vertical-gradient": DataConstantProperty; +}; +export type FillExtrusionPaintPropsPossiblyEvaluated = { + "fill-extrusion-opacity": number; + "fill-extrusion-color": PossiblyEvaluatedPropertyValue; + "fill-extrusion-translate": [ + number, + number + ]; + "fill-extrusion-translate-anchor": "map" | "viewport"; + "fill-extrusion-pattern": PossiblyEvaluatedPropertyValue>; + "fill-extrusion-height": PossiblyEvaluatedPropertyValue; + "fill-extrusion-base": PossiblyEvaluatedPropertyValue; + "fill-extrusion-vertical-gradient": boolean; +}; +export declare class FillExtrusionStyleLayer extends StyleLayer { + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + constructor(layer: LayerSpecification); + createBucket(parameters: BucketParameters): FillExtrusionBucket; + queryRadius(): number; + is3D(): boolean; + queryIntersectsFeature(queryGeometry: Array, feature: VectorTileFeature, featureState: FeatureState, geometry: Array>, zoom: number, transform: Transform, pixelsToTileUnits: number, pixelPosMatrix: mat4): boolean | number; +} +export type HillshadePaintProps = { + "hillshade-illumination-direction": DataConstantProperty; + "hillshade-illumination-anchor": DataConstantProperty<"map" | "viewport">; + "hillshade-exaggeration": DataConstantProperty; + "hillshade-shadow-color": DataConstantProperty; + "hillshade-highlight-color": DataConstantProperty; + "hillshade-accent-color": DataConstantProperty; +}; +export type HillshadePaintPropsPossiblyEvaluated = { + "hillshade-illumination-direction": number; + "hillshade-illumination-anchor": "map" | "viewport"; + "hillshade-exaggeration": number; + "hillshade-shadow-color": Color; + "hillshade-highlight-color": Color; + "hillshade-accent-color": Color; +}; +export declare class HillshadeStyleLayer extends StyleLayer { + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + constructor(layer: LayerSpecification); + hasOffscreenPass(): boolean; +} +export type LineClips = { + start: number; + end: number; +}; +export type GradientTexture = { + texture?: Texture; + gradient?: RGBAImage; + version?: number; +}; +export declare class LineBucket implements Bucket { + distance: number; + totalDistance: number; + maxLineLength: number; + scaledDistance: number; + lineClips?: LineClips; + e1: number; + e2: number; + index: number; + zoom: number; + overscaling: number; + layers: Array; + layerIds: Array; + gradients: { + [x: string]: GradientTexture; + }; + stateDependentLayers: Array; + stateDependentLayerIds: Array; + patternFeatures: Array; + lineClipsArray: Array; + layoutVertexArray: LineLayoutArray; + layoutVertexBuffer: VertexBuffer; + layoutVertexArray2: LineExtLayoutArray; + layoutVertexBuffer2: VertexBuffer; + indexArray: TriangleIndexArray; + indexBuffer: IndexBuffer; + hasPattern: boolean; + programConfigurations: ProgramConfigurationSet; + segments: SegmentVector; + uploaded: boolean; + constructor(options: BucketParameters); + populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID): void; + update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: { + [_: string]: ImagePosition; + }): void; + addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: { + [_: string]: ImagePosition; + }): void; + isEmpty(): boolean; + uploadPending(): boolean; + upload(context: Context): void; + destroy(): void; + lineFeatureClips(feature: BucketFeature): LineClips | undefined; + addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: { + [_: string]: ImagePosition; + }): void; + addLine(vertices: Array, feature: BucketFeature, join: string, cap: string, miterLimit: number, roundLimit: number): void; + /** + * Add two vertices to the buffers. + * + * @param p - the line vertex to add buffer vertices for + * @param normal - vertex normal + * @param endLeft - extrude to shift the left vertex along the line + * @param endRight - extrude to shift the left vertex along the line + * @param segment - the segment object to add the vertex to + * @param round - whether this is a round cap + */ + addCurrentVertex(p: Point, normal: Point, endLeft: number, endRight: number, segment: Segment, round?: boolean): void; + addHalfVertex({ x, y }: Point, extrudeX: number, extrudeY: number, round: boolean, up: boolean, dir: number, segment: Segment): void; + updateScaledDistance(): void; + updateDistance(prev: Point, next: Point): void; +} +export type LineLayoutProps = { + "line-cap": DataConstantProperty<"butt" | "round" | "square">; + "line-join": DataDrivenProperty<"bevel" | "round" | "miter">; + "line-miter-limit": DataConstantProperty; + "line-round-limit": DataConstantProperty; + "line-sort-key": DataDrivenProperty; +}; +export type LineLayoutPropsPossiblyEvaluated = { + "line-cap": "butt" | "round" | "square"; + "line-join": PossiblyEvaluatedPropertyValue<"bevel" | "round" | "miter">; + "line-miter-limit": number; + "line-round-limit": number; + "line-sort-key": PossiblyEvaluatedPropertyValue; +}; +export type LinePaintProps = { + "line-opacity": DataDrivenProperty; + "line-color": DataDrivenProperty; + "line-translate": DataConstantProperty<[ + number, + number + ]>; + "line-translate-anchor": DataConstantProperty<"map" | "viewport">; + "line-width": DataDrivenProperty; + "line-gap-width": DataDrivenProperty; + "line-offset": DataDrivenProperty; + "line-blur": DataDrivenProperty; + "line-dasharray": CrossFadedProperty>; + "line-pattern": CrossFadedDataDrivenProperty; + "line-gradient": ColorRampProperty; +}; +export type LinePaintPropsPossiblyEvaluated = { + "line-opacity": PossiblyEvaluatedPropertyValue; + "line-color": PossiblyEvaluatedPropertyValue; + "line-translate": [ + number, + number + ]; + "line-translate-anchor": "map" | "viewport"; + "line-width": PossiblyEvaluatedPropertyValue; + "line-gap-width": PossiblyEvaluatedPropertyValue; + "line-offset": PossiblyEvaluatedPropertyValue; + "line-blur": PossiblyEvaluatedPropertyValue; + "line-dasharray": CrossFaded>; + "line-pattern": PossiblyEvaluatedPropertyValue>; + "line-gradient": ColorRampProperty; +}; +export declare class LineStyleLayer extends StyleLayer { + _unevaluatedLayout: Layout; + layout: PossiblyEvaluated; + gradientVersion: number; + stepInterpolant: boolean; + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + constructor(layer: LayerSpecification); + _handleSpecialPaintPropertyUpdate(name: string): void; + gradientExpression(): import("@maplibre/maplibre-gl-style-spec").StylePropertyExpression; + recalculate(parameters: EvaluationParameters, availableImages: Array): void; + createBucket(parameters: BucketParameters): LineBucket; + queryRadius(bucket: Bucket): number; + queryIntersectsFeature(queryGeometry: Array, feature: VectorTileFeature, featureState: FeatureState, geometry: Array>, zoom: number, transform: Transform, pixelsToTileUnits: number): boolean; + isTileClipped(): boolean; +} +export type TypedStyleLayer = CircleStyleLayer | FillStyleLayer | FillExtrusionStyleLayer | HeatmapStyleLayer | HillshadeStyleLayer | LineStyleLayer | SymbolStyleLayer; +export type BucketParameters = { + index: number; + layers: Array; + zoom: number; + pixelRatio: number; + overscaling: number; + collisionBoxArray: CollisionBoxArray; + sourceLayerIndex: number; + sourceID: string; +}; +export type PopulateParameters = { + featureIndex: FeatureIndex; + iconDependencies: {}; + patternDependencies: {}; + glyphDependencies: {}; + availableImages: Array; +}; +export type IndexedFeature = { + feature: VectorTileFeature; + id: number | string; + index: number; + sourceLayerIndex: number; +}; +export type BucketFeature = { + index: number; + sourceLayerIndex: number; + geometry: Array>; + properties: any; + type: 0 | 1 | 2 | 3; + id?: any; + readonly patterns: { + [_: string]: { + "min": string; + "mid": string; + "max": string; + }; + }; + sortKey?: number; +}; +/** + * The `Bucket` interface is the single point of knowledge about turning vector + * tiles into WebGL buffers. + * + * `Bucket` is an abstract interface. An implementation exists for each style layer type. + * Create a bucket via the `StyleLayer#createBucket` method. + * + * The concrete bucket types, using layout options from the style layer, + * transform feature geometries into vertex and index data for use by the + * vertex shader. They also (via `ProgramConfiguration`) use feature + * properties and the zoom level to populate the attributes needed for + * data-driven styling. + * + * Buckets are designed to be built on a worker thread and then serialized and + * transferred back to the main thread for rendering. On the worker side, a + * bucket's vertex, index, and attribute data is stored in `bucket.arrays: ArrayGroup`. + * When a bucket's data is serialized and sent back to the main thread, + * is gets deserialized (using `new Bucket(serializedBucketData)`, with + * the array data now stored in `bucket.buffers: BufferGroup`. BufferGroups + * hold the same data as ArrayGroups, but are tuned for consumption by WebGL. + */ +export interface Bucket { + layerIds: Array; + hasPattern: boolean; + readonly layers: Array; + readonly stateDependentLayers: Array; + readonly stateDependentLayerIds: Array; + populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID): void; + update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: { + [_: string]: ImagePosition; + }): void; + isEmpty(): boolean; + upload(context: Context): void; + uploadPending(): boolean; + /** + * Release the WebGL resources associated with the buffers. Note that because + * buckets are shared between layers having the same layout properties, they + * must be destroyed in groups (all buckets for a tile, or all symbol buckets). + */ + destroy(): void; +} +/** + * @param gl - The map's gl context. + * @param matrix - The map's camera matrix. It projects spherical mercator + * coordinates to gl coordinates. The spherical mercator coordinate `[0, 0]` represents the + * top left corner of the mercator world and `[1, 1]` represents the bottom right corner. When + * the `renderingMode` is `"3d"`, the z coordinate is conformal. A box with identical x, y, and z + * lengths in mercator units would be rendered as a cube. {@link MercatorCoordinate.fromLngLat} + * can be used to project a `LngLat` to a mercator coordinate. + */ +export type CustomRenderMethod = (gl: WebGLRenderingContext | WebGL2RenderingContext, matrix: mat4) => void; +/** + * Interface for custom style layers. This is a specification for + * implementers to model: it is not an exported method or class. + * + * Custom layers allow a user to render directly into the map's GL context using the map's camera. + * These layers can be added between any regular layers using {@link Map#addLayer}. + * + * Custom layers must have a unique `id` and must have the `type` of `"custom"`. + * They must implement `render` and may implement `prerender`, `onAdd` and `onRemove`. + * They can trigger rendering using {@link Map#triggerRepaint} + * and they should appropriately handle {@link MapContextEvent} with `webglcontextlost` and `webglcontextrestored`. + * + * The `renderingMode` property controls whether the layer is treated as a `"2d"` or `"3d"` map layer. Use: + * - `"renderingMode": "3d"` to use the depth buffer and share it with other layers + * - `"renderingMode": "2d"` to add a layer with no depth. If you need to use the depth buffer for a `"2d"` layer you must use an offscreen + * framebuffer and {@link CustomLayerInterface#prerender} + * + * @example + * Custom layer implemented as ES6 class + * ```ts + * class NullIslandLayer { + * constructor() { + * this.id = 'null-island'; + * this.type = 'custom'; + * this.renderingMode = '2d'; + * } + * + * onAdd(map, gl) { + * const vertexSource = ` + * uniform mat4 u_matrix; + * void main() { + * gl_Position = u_matrix * vec4(0.5, 0.5, 0.0, 1.0); + * gl_PointSize = 20.0; + * }`; + * + * const fragmentSource = ` + * void main() { + * fragColor = vec4(1.0, 0.0, 0.0, 1.0); + * }`; + * + * const vertexShader = gl.createShader(gl.VERTEX_SHADER); + * gl.shaderSource(vertexShader, vertexSource); + * gl.compileShader(vertexShader); + * const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER); + * gl.shaderSource(fragmentShader, fragmentSource); + * gl.compileShader(fragmentShader); + * + * this.program = gl.createProgram(); + * gl.attachShader(this.program, vertexShader); + * gl.attachShader(this.program, fragmentShader); + * gl.linkProgram(this.program); + * } + * + * render(gl, matrix) { + * gl.useProgram(this.program); + * gl.uniformMatrix4fv(gl.getUniformLocation(this.program, "u_matrix"), false, matrix); + * gl.drawArrays(gl.POINTS, 0, 1); + * } + * } + * + * map.on('load', function() { + * map.addLayer(new NullIslandLayer()); + * }); + * ``` + */ +export interface CustomLayerInterface { + /** + * A unique layer id. + */ + id: string; + /** + * The layer's type. Must be `"custom"`. + */ + type: "custom"; + /** + * Either `"2d"` or `"3d"`. Defaults to `"2d"`. + */ + renderingMode?: "2d" | "3d"; + /** + * Called during a render frame allowing the layer to draw into the GL context. + * + * The layer can assume blending and depth state is set to allow the layer to properly + * blend and clip other layers. The layer cannot make any other assumptions about the + * current GL state. + * + * If the layer needs to render to a texture, it should implement the `prerender` method + * to do this and only use the `render` method for drawing directly into the main framebuffer. + * + * The blend function is set to `gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA)`. This expects + * colors to be provided in premultiplied alpha form where the `r`, `g` and `b` values are already + * multiplied by the `a` value. If you are unable to provide colors in premultiplied form you + * may want to change the blend function to + * `gl.blendFuncSeparate(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA, gl.ONE, gl.ONE_MINUS_SRC_ALPHA)`. + */ + render: CustomRenderMethod; + /** + * Optional method called during a render frame to allow a layer to prepare resources or render into a texture. + * + * The layer cannot make any assumptions about the current GL state and must bind a framebuffer before rendering. + */ + prerender?: CustomRenderMethod; + /** + * Optional method called when the layer has been added to the Map with {@link Map#addLayer}. This + * gives the layer a chance to initialize gl resources and register event listeners. + * + * @param map - The Map this custom layer was just added to. + * @param gl - The gl context for the map. + */ + onAdd?(map: Map, gl: WebGLRenderingContext | WebGL2RenderingContext): void; + /** + * Optional method called when the layer has been removed from the Map with {@link Map#removeLayer}. This + * gives the layer a chance to clean up gl resources and event listeners. + * + * @param map - The Map this custom layer was just added to. + * @param gl - The gl context for the map. + */ + onRemove?(map: Map, gl: WebGLRenderingContext | WebGL2RenderingContext): void; +} +declare abstract class StyleLayer extends Evented { + id: string; + metadata: unknown; + type: LayerSpecification["type"] | CustomLayerInterface["type"]; + source: string; + sourceLayer: string; + minzoom: number; + maxzoom: number; + filter: FilterSpecification | void; + visibility: "visible" | "none" | void; + _crossfadeParameters: CrossfadeParameters; + _unevaluatedLayout: Layout; + readonly layout: unknown; + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + readonly paint: unknown; + _featureFilter: FeatureFilter; + readonly onAdd: ((map: Map) => void); + readonly onRemove: ((map: Map) => void); + queryRadius?(bucket: Bucket): number; + queryIntersectsFeature?(queryGeometry: Array, feature: VectorTileFeature, featureState: FeatureState, geometry: Array>, zoom: number, transform: Transform, pixelsToTileUnits: number, pixelPosMatrix: mat4): boolean | number; + constructor(layer: LayerSpecification | CustomLayerInterface, properties: Readonly<{ + layout?: Properties; + paint?: Properties; + }>); + getCrossfadeParameters(): CrossfadeParameters; + getLayoutProperty(name: string): any; + setLayoutProperty(name: string, value: any, options?: StyleSetterOptions): void; + getPaintProperty(name: string): unknown; + setPaintProperty(name: string, value: unknown, options?: StyleSetterOptions): boolean; + _handleSpecialPaintPropertyUpdate(_: string): void; + _handleOverridablePaintPropertyUpdate(name: string, oldValue: PropertyValue, newValue: PropertyValue): boolean; + isHidden(zoom: number): boolean; + updateTransitions(parameters: TransitionParameters): void; + hasTransition(): boolean; + recalculate(parameters: EvaluationParameters, availableImages: Array): void; + serialize(): LayerSpecification; + _validate(validate: Function, key: string, name: string, value: unknown, options?: StyleSetterOptions): boolean; + is3D(): boolean; + isTileClipped(): boolean; + hasOffscreenPass(): boolean; + resize(): void; + isStateDependent(): boolean; +} +export type LightPosition = { + x: number; + y: number; + z: number; +}; +export declare class LightPositionProperty implements Property<[ + number, + number, + number +], LightPosition> { + specification: StylePropertySpecification; + constructor(); + possiblyEvaluate(value: PropertyValue<[ + number, + number, + number + ], LightPosition>, parameters: EvaluationParameters): LightPosition; + interpolate(a: LightPosition, b: LightPosition, t: number): LightPosition; +} +export type Props = { + "anchor": DataConstantProperty<"map" | "viewport">; + "position": LightPositionProperty; + "color": DataConstantProperty; + "intensity": DataConstantProperty; +}; +export type PropsPossiblyEvaluated = { + "anchor": "map" | "viewport"; + "position": LightPosition; + "color": Color; + "intensity": number; +}; +export declare class Light extends Evented { + _transitionable: Transitionable; + _transitioning: Transitioning; + properties: PossiblyEvaluated; + constructor(lightOptions?: LightSpecification); + getLight(): LightSpecification; + setLight(light?: LightSpecification, options?: StyleSetterOptions): void; + updateTransitions(parameters: TransitionParameters): void; + hasTransition(): boolean; + recalculate(parameters: EvaluationParameters): void; + _validate(validate: Function, value: unknown, options?: { + validate?: boolean; + }): boolean; +} +/** + * The overlap mode for properties like `icon-overlap`and `text-overlap` + */ +export type OverlapMode = "never" | "always" | "cooperative"; +export type QueryResult = { + key: T; + x1: number; + y1: number; + x2: number; + y2: number; +}; +/** + * A key for the grid + */ +export type GridKey = { + overlapMode?: OverlapMode; +}; +export declare class GridIndex { + circleKeys: Array; + boxKeys: Array; + boxCells: Array>; + circleCells: Array>; + bboxes: Array; + circles: Array; + xCellCount: number; + yCellCount: number; + width: number; + height: number; + xScale: number; + yScale: number; + boxUid: number; + circleUid: number; + constructor(width: number, height: number, cellSize: number); + keysLength(): number; + insert(key: T, x1: number, y1: number, x2: number, y2: number): void; + insertCircle(key: T, x: number, y: number, radius: number): void; + private _insertBoxCell; + private _insertCircleCell; + private _query; + query(x1: number, y1: number, x2: number, y2: number): Array>; + hitTest(x1: number, y1: number, x2: number, y2: number, overlapMode: OverlapMode, predicate?: (key: T) => boolean): boolean; + hitTestCircle(x: number, y: number, radius: number, overlapMode: OverlapMode, predicate?: (key: T) => boolean): boolean; + private _queryCell; + private _queryCellCircle; + private _forEachCell; + private _convertToXCellCoord; + private _convertToYCellCoord; + private _circlesCollide; + private _circleAndRectCollide; +} +export type FeatureKey = { + bucketInstanceId: number; + featureIndex: number; + collisionGroupID: number; + overlapMode: OverlapMode; +}; +export declare class CollisionIndex { + grid: GridIndex; + ignoredGrid: GridIndex; + transform: Transform; + pitchfactor: number; + screenRightBoundary: number; + screenBottomBoundary: number; + gridRightBoundary: number; + gridBottomBoundary: number; + perspectiveRatioCutoff: number; + constructor(transform: Transform, grid?: GridIndex, ignoredGrid?: GridIndex); + placeCollisionBox(collisionBox: SingleCollisionBox, overlapMode: OverlapMode, textPixelRatio: number, posMatrix: mat4, collisionGroupPredicate?: (key: FeatureKey) => boolean, getElevation?: (x: number, y: number) => number): { + box: Array; + offscreen: boolean; + }; + placeCollisionCircles(overlapMode: OverlapMode, symbol: any, lineVertexArray: SymbolLineVertexArray, glyphOffsetArray: GlyphOffsetArray, fontSize: number, posMatrix: mat4, labelPlaneMatrix: mat4, labelToScreenMatrix: mat4, showCollisionCircles: boolean, pitchWithMap: boolean, collisionGroupPredicate: (key: FeatureKey) => boolean, circlePixelDiameter: number, textPixelPadding: number, getElevation: (x: number, y: number) => number): { + circles: Array; + offscreen: boolean; + collisionDetected: boolean; + }; + /** + * Because the geometries in the CollisionIndex are an approximation of the shape of + * symbols on the map, we use the CollisionIndex to look up the symbol part of + * `queryRenderedFeatures`. + */ + queryRenderedSymbols(viewportQueryGeometry: Array): {}; + insertCollisionBox(collisionBox: Array, overlapMode: OverlapMode, ignorePlacement: boolean, bucketInstanceId: number, featureIndex: number, collisionGroupID: number): void; + insertCollisionCircles(collisionCircles: Array, overlapMode: OverlapMode, ignorePlacement: boolean, bucketInstanceId: number, featureIndex: number, collisionGroupID: number): void; + projectAndGetPerspectiveRatio(posMatrix: mat4, x: number, y: number, getElevation?: (x: number, y: number) => number): { + point: Point; + perspectiveRatio: number; + }; + isOffscreen(x1: number, y1: number, x2: number, y2: number): boolean; + isInsideGrid(x1: number, y1: number, x2: number, y2: number): boolean; + getViewportMatrix(): mat4; +} +declare enum TextAnchorEnum { + "center" = 1, + "left" = 2, + "right" = 3, + "top" = 4, + "bottom" = 5, + "top-left" = 6, + "top-right" = 7, + "bottom-left" = 8, + "bottom-right" = 9 +} +export type TextAnchor = keyof typeof TextAnchorEnum; +export declare class OpacityState { + opacity: number; + placed: boolean; + constructor(prevState: OpacityState, increment: number, placed: boolean, skipFade?: boolean | null); + isHidden(): boolean; +} +export declare class JointOpacityState { + text: OpacityState; + icon: OpacityState; + constructor(prevState: JointOpacityState, increment: number, placedText: boolean, placedIcon: boolean, skipFade?: boolean | null); + isHidden(): boolean; +} +export declare class JointPlacement { + text: boolean; + icon: boolean; + skipFade: boolean; + constructor(text: boolean, icon: boolean, skipFade: boolean); +} +export declare class CollisionCircleArray { + invProjMatrix: mat4; + viewportMatrix: mat4; + circles: Array; + constructor(); +} +export declare class RetainedQueryData { + bucketInstanceId: number; + featureIndex: FeatureIndex; + sourceLayerIndex: number; + bucketIndex: number; + tileID: OverscaledTileID; + featureSortOrder: Array; + constructor(bucketInstanceId: number, featureIndex: FeatureIndex, sourceLayerIndex: number, bucketIndex: number, tileID: OverscaledTileID); +} +export type CollisionGroup = { + ID: number; + predicate?: (key: FeatureKey) => boolean; +}; +export declare class CollisionGroups { + collisionGroups: { + [groupName: string]: CollisionGroup; + }; + maxGroupID: number; + crossSourceCollisions: boolean; + constructor(crossSourceCollisions: boolean); + get(sourceID: string): CollisionGroup; +} +export type VariableOffset = { + textOffset: [ + number, + number + ]; + width: number; + height: number; + anchor: TextAnchor; + textBoxScale: number; + prevAnchor?: TextAnchor; +}; +export type TileLayerParameters = { + bucket: SymbolBucket; + layout: PossiblyEvaluated; + posMatrix: mat4; + textLabelPlaneMatrix: mat4; + labelToScreenMatrix: mat4; + scale: number; + textPixelRatio: number; + holdingForFade: boolean; + collisionBoxArray: CollisionBoxArray; + partiallyEvaluatedTextSize: { + uSize: number; + uSizeT: number; + }; + collisionGroup: CollisionGroup; +}; +export type BucketPart = { + sortKey?: number | void; + symbolInstanceStart: number; + symbolInstanceEnd: number; + parameters: TileLayerParameters; +}; +export type CrossTileID = string | number; +export declare class Placement { + transform: Transform; + terrain: Terrain; + collisionIndex: CollisionIndex; + placements: { + [_ in CrossTileID]: JointPlacement; + }; + opacities: { + [_ in CrossTileID]: JointOpacityState; + }; + variableOffsets: { + [_ in CrossTileID]: VariableOffset; + }; + placedOrientations: { + [_ in CrossTileID]: number; + }; + commitTime: number; + prevZoomAdjustment: number; + lastPlacementChangeTime: number; + stale: boolean; + fadeDuration: number; + retainedQueryData: { + [_: number]: RetainedQueryData; + }; + collisionGroups: CollisionGroups; + prevPlacement: Placement; + zoomAtLastRecencyCheck: number; + collisionCircleArrays: { + [k in any]: CollisionCircleArray; + }; + constructor(transform: Transform, terrain: Terrain, fadeDuration: number, crossSourceCollisions: boolean, prevPlacement?: Placement); + getBucketParts(results: Array, styleLayer: StyleLayer, tile: Tile, sortAcrossTiles: boolean): void; + attemptAnchorPlacement(textAnchorOffset: TextAnchorOffset, textBox: SingleCollisionBox, width: number, height: number, textBoxScale: number, rotateWithMap: boolean, pitchWithMap: boolean, textPixelRatio: number, posMatrix: mat4, collisionGroup: CollisionGroup, textOverlapMode: OverlapMode, symbolInstance: SymbolInstance, bucket: SymbolBucket, orientation: number, iconBox?: SingleCollisionBox | null, getElevation?: (x: number, y: number) => number): { + shift: Point; + placedGlyphBoxes: { + box: Array; + offscreen: boolean; + }; + }; + placeLayerBucketPart(bucketPart: BucketPart, seenCrossTileIDs: { + [k in string | number]: boolean; + }, showCollisionBoxes: boolean): void; + markUsedJustification(bucket: SymbolBucket, placedAnchor: TextAnchor, symbolInstance: SymbolInstance, orientation: number): void; + markUsedOrientation(bucket: SymbolBucket, orientation: number, symbolInstance: SymbolInstance): void; + commit(now: number): void; + updateLayerOpacities(styleLayer: StyleLayer, tiles: Array): void; + updateBucketOpacities(bucket: SymbolBucket, seenCrossTileIDs: { + [k in string | number]: boolean; + }, collisionBoxArray?: CollisionBoxArray | null): void; + symbolFadeChange(now: number): number; + zoomAdjustment(zoom: number): number; + hasTransitions(now: number): boolean; + stillRecent(now: number, zoom: number): boolean; + setStale(): void; +} +/** + * Options to pass to query the map for the rendered features + */ +export type QueryRenderedFeaturesOptions = { + /** + * An array of [style layer IDs](https://maplibre.org/maplibre-style-spec/#layer-id) for the query to inspect. + * Only features within these layers will be returned. If this parameter is undefined, all layers will be checked. + */ + layers?: Array; + /** + * A [filter](https://maplibre.org/maplibre-style-spec/layers/#filter) to limit query results. + */ + filter?: FilterSpecification; + /** + * An array of string representing the available images + */ + availableImages?: Array; + /** + * Whether to check if the [options.filter] conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function. + */ + validate?: boolean; +}; +/** + * The options object related to the {@link Map#queryRenderedFeatures} method + */ +export type QuerySourceFeatureOptions = { + /** + * The name of the source layer to query. *For vector tile sources, this parameter is required.* For GeoJSON sources, it is ignored. + */ + sourceLayer?: string; + /** + * A [filter](https://maplibre.org/maplibre-style-spec/layers/#filter) + * to limit query results. + */ + filter?: FilterSpecification; + /** + * Whether to check if the [parameters.filter] conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function. + * @defaultValue true + */ + validate?: boolean; +}; +declare const status: { + unavailable: string; + deferred: string; + loading: string; + loaded: string; + error: string; +}; +export type PluginState = { + pluginStatus: typeof status[keyof typeof status]; + pluginURL: string; +}; +export type PluginStateSyncCallback = (state: PluginState) => void; +declare const registerForPluginStateChange: (callback: PluginStateSyncCallback) => PluginStateSyncCallback; +export declare class LayerPlacement { + _sortAcrossTiles: boolean; + _currentTileIndex: number; + _currentPartIndex: number; + _seenCrossTileIDs: { + [k in string | number]: boolean; + }; + _bucketParts: Array; + constructor(styleLayer: SymbolStyleLayer); + continuePlacement(tiles: Array, placement: Placement, showCollisionBoxes: boolean, styleLayer: StyleLayer, shouldPausePlacement: () => boolean): boolean; +} +export declare class PauseablePlacement { + placement: Placement; + _done: boolean; + _currentPlacementIndex: number; + _forceFullPlacement: boolean; + _showCollisionBoxes: boolean; + _inProgressLayer: LayerPlacement; + constructor(transform: Transform, terrain: Terrain, order: Array, forceFullPlacement: boolean, showCollisionBoxes: boolean, fadeDuration: number, crossSourceCollisions: boolean, prevPlacement?: Placement); + isDone(): boolean; + continuePlacement(order: Array, layers: { + [_: string]: StyleLayer; + }, layerTiles: { + [_: string]: Array; + }): void; + commit(now: number): Placement; +} +export type ValidationError = { + message: string; + line: number; + identifier?: string; +}; +export type Validator = (a: any) => ReadonlyArray; +/** + * A feature identifier that is bound to a source + */ +export type FeatureIdentifier = { + /** + * Unique id of the feature. + */ + id?: string | number | undefined; + /** + * The id of the vector or GeoJSON source for the feature. + */ + source: string; + /** + * *For vector tile sources, `sourceLayer` is required.* + */ + sourceLayer?: string | undefined; +}; +/** + * The options object related to the {@link Map}'s style related methods + */ +export type StyleOptions = { + /** + * If false, style validation will be skipped. Useful in production environment. + */ + validate?: boolean; + /** + * Defines a CSS + * font-family for locally overriding generation of glyphs in the 'CJK Unified Ideographs', 'Hiragana', 'Katakana' and 'Hangul Syllables' ranges. + * In these ranges, font settings from the map's style will be ignored, except for font-weight keywords (light/regular/medium/bold). + * Set to `false`, to enable font settings from the map's style for these glyph ranges. + * Forces a full update. + */ + localIdeographFontFamily?: string; +}; +/** + * Supporting type to add validation to another style related type + */ +export type StyleSetterOptions = { + /** + * Whether to check if the filter conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function. + */ + validate?: boolean; +}; +/** + * Part of {@link Map#setStyle} options, transformStyle is a convenience function that allows to modify a style after it is fetched but before it is committed to the map state + * this function exposes previous and next styles, it can be commonly used to support a range of functionalities like: + * when previous style carries certain 'state' that needs to be carried over to a new style gracefully + * when a desired style is a certain combination of previous and incoming style + * when an incoming style requires modification based on external state + * + * @param previousStyle - The current style. + * @param nextStyle - The next style. + * @returns resulting style that will to be applied to the map + * + * @example + * ```ts + * map.setStyle('https://demotiles.maplibre.org/style.json', { + * transformStyle: (previousStyle, nextStyle) => ({ + * ...nextStyle, + * sources: { + * ...nextStyle.sources, + * // copy a source from previous style + * 'osm': previousStyle.sources.osm + * }, + * layers: [ + * // background layer + * nextStyle.layers[0], + * // copy a layer from previous style + * previousStyle.layers[0], + * // other layers from the next style + * ...nextStyle.layers.slice(1).map(layer => { + * // hide the layers we don't need from demotiles style + * if (layer.id.startsWith('geolines')) { + * layer.layout = {...layer.layout || {}, visibility: 'none'}; + * // filter out US polygons + * } else if (layer.id.startsWith('coastline') || layer.id.startsWith('countries')) { + * layer.filter = ['!=', ['get', 'ADM0_A3'], 'USA']; + * } + * return layer; + * }) + * ] + * }) + * }); + * ``` + */ +export type TransformStyleFunction = (previous: StyleSpecification | undefined, next: StyleSpecification) => StyleSpecification; +/** + * The options object related to the {@link Map}'s style related methods + */ +export type StyleSwapOptions = { + /** + * If false, force a 'full' update, removing the current style + * and building the given one instead of attempting a diff-based update. + */ + diff?: boolean; + /** + * TransformStyleFunction is a convenience function + * that allows to modify a style after it is fetched but before it is committed to the map state. Refer to {@link TransformStyleFunction}. + */ + transformStyle?: TransformStyleFunction; +}; +/** + * Specifies a layer to be added to a {@link Style}. In addition to a standard {@link LayerSpecification} + * or a {@link CustomLayerInterface}, a {@link LayerSpecification} with an embedded {@link SourceSpecification} can also be provided. + */ +export type AddLayerObject = LayerSpecification | (Omit & { + source: SourceSpecification; +}) | CustomLayerInterface; +export declare class Style extends Evented { + map: Map; + stylesheet: StyleSpecification; + dispatcher: Dispatcher; + imageManager: ImageManager; + glyphManager: GlyphManager; + lineAtlas: LineAtlas; + light: Light; + _request: Cancelable; + _spriteRequest: Cancelable; + _layers: { + [_: string]: StyleLayer; + }; + _serializedLayers: { + [_: string]: LayerSpecification; + }; + _order: Array; + sourceCaches: { + [_: string]: SourceCache; + }; + zoomHistory: ZoomHistory; + _loaded: boolean; + _rtlTextPluginCallback: (a: any) => any; + _changed: boolean; + _updatedSources: { + [_: string]: "clear" | "reload"; + }; + _updatedLayers: { + [_: string]: true; + }; + _removedLayers: { + [_: string]: StyleLayer; + }; + _changedImages: { + [_: string]: true; + }; + _glyphsDidChange: boolean; + _updatedPaintProps: { + [layer: string]: true; + }; + _layerOrderChanged: boolean; + _spritesImagesIds: { + [spriteId: string]: string[]; + }; + _availableImages: Array; + crossTileSymbolIndex: CrossTileSymbolIndex; + pauseablePlacement: PauseablePlacement; + placement: Placement; + z: number; + static registerForPluginStateChange: typeof registerForPluginStateChange; + constructor(map: Map, options?: StyleOptions); + loadURL(url: string, options?: StyleSwapOptions & StyleSetterOptions, previousStyle?: StyleSpecification): void; + loadJSON(json: StyleSpecification, options?: StyleSetterOptions & StyleSwapOptions, previousStyle?: StyleSpecification): void; + loadEmpty(): void; + _load(json: StyleSpecification, options: StyleSwapOptions & StyleSetterOptions, previousStyle?: StyleSpecification): void; + private _createLayers; + _loadSprite(sprite: SpriteSpecification, isUpdate?: boolean, completion?: (err: Error) => void): void; + _unloadSprite(): void; + _validateLayer(layer: StyleLayer): void; + loaded(): boolean; + /** + * take an array of string IDs, and based on this._layers, generate an array of LayerSpecification + * @param ids - an array of string IDs, for which serialized layers will be generated. If omitted, all serialized layers will be returned + * @returns generated result + */ + private _serializeByIds; + /** + * Lazy initialization of this._serializedLayers dictionary and return it + * @returns this._serializedLayers dictionary + */ + private _serializedAllLayers; + hasTransitions(): boolean; + _checkLoaded(): void; + /** + * @internal + * Apply queued style updates in a batch and recalculate zoom-dependent paint properties. + */ + update(parameters: EvaluationParameters): void; + _updateTilesForChangedImages(): void; + _updateTilesForChangedGlyphs(): void; + _updateWorkerLayers(updatedIds: Array, removedIds: Array): void; + _resetUpdates(): void; + /** + * Update this style's state to match the given style JSON, performing only + * the necessary mutations. + * + * May throw an Error ('Unimplemented: METHOD') if the mapbox-gl-style-spec + * diff algorithm produces an operation that is not supported. + * + * @returns true if any changes were made; false otherwise + */ + setState(nextState: StyleSpecification, options?: StyleSwapOptions): boolean; + addImage(id: string, image: StyleImage): this; + updateImage(id: string, image: StyleImage): void; + getImage(id: string): StyleImage; + removeImage(id: string): this; + _afterImageUpdated(id: string): void; + listImages(): string[]; + addSource(id: string, source: SourceSpecification, options?: StyleSetterOptions): void; + /** + * Remove a source from this stylesheet, given its id. + * @param id - id of the source to remove + * @throws if no source is found with the given ID + * @returns `this`. + */ + removeSource(id: string): this; + /** + * Set the data of a GeoJSON source, given its id. + * @param id - id of the source + * @param data - GeoJSON source + */ + setGeoJSONSourceData(id: string, data: GeoJSON.GeoJSON | string): void; + /** + * Get a source by ID. + * @param id - ID of the desired source + * @returns source + */ + getSource(id: string): Source | undefined; + /** + * Add a layer to the map style. The layer will be inserted before the layer with + * ID `before`, or appended if `before` is omitted. + * @param layerObject - The style layer to add. + * @param before - ID of an existing layer to insert before + * @param options - Style setter options. + * @returns `this`. + */ + addLayer(layerObject: AddLayerObject, before?: string, options?: StyleSetterOptions): this; + /** + * Moves a layer to a different z-position. The layer will be inserted before the layer with + * ID `before`, or appended if `before` is omitted. + * @param id - ID of the layer to move + * @param before - ID of an existing layer to insert before + */ + moveLayer(id: string, before?: string): void; + /** + * Remove the layer with the given id from the style. + * + * If no such layer exists, an `error` event is fired. + * + * @param id - id of the layer to remove + * @event `error` - Fired if the layer does not exist + */ + removeLayer(id: string): void; + /** + * Return the style layer object with the given `id`. + * + * @param id - id of the desired layer + * @returns a layer, if one with the given `id` exists + */ + getLayer(id: string): StyleLayer | undefined; + /** + * Return the ids of all layers currently in the style, including custom layers, in order. + * + * @returns ids of layers, in order + */ + getLayersOrder(): string[]; + /** + * Checks if a specific layer is present within the style. + * + * @param id - the id of the desired layer + * @returns a boolean specifying if the given layer is present + */ + hasLayer(id: string): boolean; + setLayerZoomRange(layerId: string, minzoom?: number | null, maxzoom?: number | null): void; + setFilter(layerId: string, filter?: FilterSpecification | null, options?: StyleSetterOptions): void; + /** + * Get a layer's filter object + * @param layer - the layer to inspect + * @returns the layer's filter, if any + */ + getFilter(layer: string): FilterSpecification | void; + setLayoutProperty(layerId: string, name: string, value: any, options?: StyleSetterOptions): void; + /** + * Get a layout property's value from a given layer + * @param layerId - the layer to inspect + * @param name - the name of the layout property + * @returns the property value + */ + getLayoutProperty(layerId: string, name: string): any; + setPaintProperty(layerId: string, name: string, value: any, options?: StyleSetterOptions): void; + getPaintProperty(layer: string, name: string): unknown; + setFeatureState(target: FeatureIdentifier, state: any): void; + removeFeatureState(target: FeatureIdentifier, key?: string): void; + getFeatureState(target: FeatureIdentifier): any; + getTransition(): any; + serialize(): StyleSpecification; + _updateLayer(layer: StyleLayer): void; + _flattenAndSortRenderedFeatures(sourceResults: Array<{ + [key: string]: Array<{ + featureIndex: number; + feature: MapGeoJSONFeature; + }>; + }>): any[]; + queryRenderedFeatures(queryGeometry: any, params: QueryRenderedFeaturesOptions, transform: Transform): any[]; + querySourceFeatures(sourceID: string, params?: QuerySourceFeatureOptions): any[]; + addSourceType(name: string, SourceType: SourceClass, callback: Callback): void; + getLight(): LightSpecification; + setLight(lightOptions: LightSpecification, options?: StyleSetterOptions): void; + _validate(validate: Validator, key: string, value: any, props: any, options?: { + validate?: boolean; + }): boolean; + _remove(mapRemoved?: boolean): void; + _clearSource(id: string): void; + _reloadSource(id: string): void; + _updateSources(transform: Transform): void; + _generateCollisionBoxes(): void; + _updatePlacement(transform: Transform, showCollisionBoxes: boolean, fadeDuration: number, crossSourceCollisions: boolean, forceFullPlacement?: boolean): boolean; + _releaseSymbolFadeTiles(): void; + getImages(mapId: string, params: { + icons: Array; + source: string; + tileID: OverscaledTileID; + type: string; + }, callback: Callback<{ + [_: string]: StyleImage; + }>): void; + getGlyphs(mapId: string, params: { + stacks: { + [_: string]: Array; + }; + source: string; + tileID: OverscaledTileID; + type: string; + }, callback: Callback<{ + [_: string]: { + [_: number]: StyleGlyph; + }; + }>): void; + getResource(mapId: string, params: RequestParameters, callback: ResponseCallback): Cancelable; + getGlyphsUrl(): string; + setGlyphs(glyphsUrl: string | null, options?: StyleSetterOptions): void; + /** + * Add a sprite. + * + * @param id - The id of the desired sprite + * @param url - The url to load the desired sprite from + * @param options - The style setter options + * @param completion - The completion handler + */ + addSprite(id: string, url: string, options?: StyleSetterOptions, completion?: (err: Error) => void): void; + /** + * Remove a sprite by its id. When the last sprite is removed, the whole `this.stylesheet.sprite` object becomes + * `undefined`. This falsy `undefined` value later prevents attempts to load the sprite when it's absent. + * + * @param id - the id of the sprite to remove + */ + removeSprite(id: string): void; + /** + * Get the current sprite value. + * + * @returns empty array when no sprite is set; id-url pairs otherwise + */ + getSprite(): { + id: string; + url: string; + }[]; + /** + * Set a new value for the style's sprite. + * + * @param sprite - new sprite value + * @param options - style setter options + * @param completion - the completion handler + */ + setSprite(sprite: SpriteSpecification, options?: StyleSetterOptions, completion?: (err: Error) => void): void; +} +export declare class Hash { + _map: Map; + _hashName: string; + constructor(hashName?: string | null); + /** + * Map element to listen for coordinate changes + * + * @param map - The map object + * @returns `this` + */ + addTo(map: Map): this; + /** + * Removes hash + * + * @returns `this` + */ + remove(): this; + getHashString(mapFeedback?: boolean): string; + _getCurrentHash: () => any; + _onHashChange: () => boolean; + _updateHashUnthrottled: () => void; + /** + * Mobile Safari doesn't allow updating the hash more than 100 times per 30 seconds. + */ + _updateHash: () => ReturnType; +} +export interface DragMovementResult { + bearingDelta?: number; + pitchDelta?: number; + around?: Point; + panDelta?: Point; +} +export interface DragPanResult extends DragMovementResult { + around: Point; + panDelta: Point; +} +export interface DragRotateResult extends DragMovementResult { + bearingDelta: number; +} +export interface DragPitchResult extends DragMovementResult { + pitchDelta: number; +} +export interface DragMoveHandler extends Handler { + dragStart: (e: E, point: Point) => void; + dragMove: (e: E, point: Point) => T | void; + dragEnd: (e: E) => void; + getClickTolerance: () => number; +} +export interface MousePanHandler extends DragMoveHandler { +} +export interface MouseRotateHandler extends DragMoveHandler { +} +export interface MousePitchHandler extends DragMoveHandler { +} +export declare class TouchPanHandler implements Handler { + _enabled: boolean; + _active: boolean; + _touches: { + [k in string | number]: Point; + }; + _minTouches: number; + _clickTolerance: number; + _sum: Point; + _map: Map; + _cancelCooperativeMessage: boolean; + constructor(options: { + clickTolerance: number; + cooperativeGestures: boolean | GestureOptions; + }, map: Map); + reset(): void; + touchstart(e: TouchEvent, points: Array, mapTouches: Array): { + around: Point; + panDelta: Point; + }; + touchmove(e: TouchEvent, points: Array, mapTouches: Array): { + around: Point; + panDelta: Point; + }; + touchend(e: TouchEvent, points: Array, mapTouches: Array): void; + touchcancel(): void; + _calculateTransform(e: TouchEvent, points: Array, mapTouches: Array): { + around: Point; + panDelta: Point; + }; + enable(): void; + disable(): void; + isEnabled(): boolean; + isActive(): boolean; +} +/** + * A {@link DragPanHandler} options object + */ +export type DragPanOptions = { + /** + * factor used to scale the drag velocity + * @defaultValue 0 + */ + linearity?: number; + /** + * easing function applled to `map.panTo` when applying the drag. + * @param t - the easing function + * @defaultValue bezier(0, 0, 0.3, 1) + */ + easing?: (t: number) => number; + /** + * the maximum value of the drag velocity. + * @defaultValue 1400 + */ + deceleration?: number; + /** + * the rate at which the speed reduces after the pan ends. + * @defaultValue 2500 + */ + maxSpeed?: number; +}; +export declare class DragPanHandler { + _el: HTMLElement; + _mousePan: MousePanHandler; + _touchPan: TouchPanHandler; + _inertiaOptions: DragPanOptions | boolean; + /** @internal */ + constructor(el: HTMLElement, mousePan: MousePanHandler, touchPan: TouchPanHandler); + /** + * Enables the "drag to pan" interaction. + * + * @param options - Options object + * @example + * ```ts + * map.dragPan.enable(); + * map.dragPan.enable({ + * linearity: 0.3, + * easing: bezier(0, 0, 0.3, 1), + * maxSpeed: 1400, + * deceleration: 2500, + * }); + * ``` + */ + enable(options?: DragPanOptions | boolean): void; + /** + * Disables the "drag to pan" interaction. + * + * @example + * ```ts + * map.dragPan.disable(); + * ``` + */ + disable(): void; + /** + * Returns a Boolean indicating whether the "drag to pan" interaction is enabled. + * + * @returns `true` if the "drag to pan" interaction is enabled. + */ + isEnabled(): boolean; + /** + * Returns a Boolean indicating whether the "drag to pan" interaction is active, i.e. currently being used. + * + * @returns `true` if the "drag to pan" interaction is active. + */ + isActive(): boolean; +} +export declare class HandlerInertia { + _map: Map; + _inertiaBuffer: Array<{ + time: number; + settings: any; + }>; + constructor(map: Map); + clear(): void; + record(settings: any): void; + _drainInertiaBuffer(): void; + _onMoveEnd(panInertiaOptions?: DragPanOptions | boolean): any; +} +export type InputEvent = MouseEvent | TouchEvent | KeyboardEvent | WheelEvent; +/** + * Handlers interpret dom events and return camera changes that should be + * applied to the map (`HandlerResult`s). The camera changes are all deltas. + * The handler itself should have no knowledge of the map's current state. + * This makes it easier to merge multiple results and keeps handlers simpler. + * For example, if there is a mousedown and mousemove, the mousePan handler + * would return a `panDelta` on the mousemove. + */ +export interface Handler { + enable(): void; + disable(): void; + isEnabled(): boolean; + isActive(): boolean; + /** + * `reset` can be called by the manager at any time and must reset everything to it's original state + */ + reset(): void; + readonly touchstart?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void; + readonly touchmove?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void; + readonly touchmoveWindow?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void; + readonly touchend?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void; + readonly touchcancel?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void; + readonly mousedown?: (e: MouseEvent, point: Point) => HandlerResult | void; + readonly mousemove?: (e: MouseEvent, point: Point) => HandlerResult | void; + readonly mousemoveWindow?: (e: MouseEvent, point: Point) => HandlerResult | void; + readonly mouseup?: (e: MouseEvent, point: Point) => HandlerResult | void; + readonly mouseupWindow?: (e: MouseEvent, point: Point) => HandlerResult | void; + readonly dblclick?: (e: MouseEvent, point: Point) => HandlerResult | void; + readonly contextmenu?: (e: MouseEvent) => HandlerResult | void; + readonly wheel?: (e: WheelEvent, point: Point) => HandlerResult | void; + readonly keydown?: (e: KeyboardEvent) => HandlerResult | void; + readonly keyup?: (e: KeyboardEvent) => HandlerResult | void; + /** + * `renderFrame` is the only non-dom event. It is called during render + * frames and can be used to smooth camera changes (see scroll handler). + */ + readonly renderFrame?: () => HandlerResult | void; +} +/** + * All handler methods that are called with events can optionally return a `HandlerResult`. + */ +export type HandlerResult = { + panDelta?: Point; + zoomDelta?: number; + bearingDelta?: number; + pitchDelta?: number; + /** + * the point to not move when changing the camera + */ + around?: Point | null; + /** + * same as above, except for pinch actions, which are given higher priority + */ + pinchAround?: Point | null; + /** + * A method that can fire a one-off easing by directly changing the map's camera. + */ + cameraAnimation?: (map: Map) => any; + /** + * The last three properties are needed by only one handler: scrollzoom. + * The DOM event to be used as the `originalEvent` on any camera change events. + */ + originalEvent?: Event; + /** + * Makes the manager trigger a frame, allowing the handler to return multiple results over time (see scrollzoom). + */ + needsRenderFrame?: boolean; + /** + * The camera changes won't get recorded for inertial zooming. + */ + noInertia?: boolean; +}; +export type EventInProgress = { + handlerName: string; + originalEvent: Event; +}; +export type EventsInProgress = { + zoom?: EventInProgress; + pitch?: EventInProgress; + rotate?: EventInProgress; + drag?: EventInProgress; +}; +export declare class HandlerManager { + _map: Map; + _el: HTMLElement; + _handlers: Array<{ + handlerName: string; + handler: Handler; + allowed: Array; + }>; + _eventsInProgress: EventsInProgress; + _frameId: number; + _inertia: HandlerInertia; + _bearingSnap: number; + _handlersById: { + [x: string]: Handler; + }; + _updatingCamera: boolean; + _changes: Array<[ + HandlerResult, + EventsInProgress, + { + [handlerName: string]: Event; + } + ]>; + _terrainMovement: boolean; + _zoom: { + handlerName: string; + }; + _previousActiveHandlers: { + [x: string]: Handler; + }; + _listeners: Array<[ + Window | Document | HTMLElement, + string, + { + passive?: boolean; + capture?: boolean; + } | undefined + ]>; + constructor(map: Map, options: CompleteMapOptions); + destroy(): void; + _addDefaultHandlers(options: CompleteMapOptions): void; + _add(handlerName: string, handler: Handler, allowed?: Array): void; + stop(allowEndAnimation: boolean): void; + isActive(): boolean; + isZooming(): boolean; + isRotating(): boolean; + isMoving(): boolean; + _blockedByActive(activeHandlers: { + [x: string]: Handler; + }, allowed: Array, myName: string): boolean; + handleWindowEvent: (e: { + type: "mousemove" | "mouseup" | "touchmove"; + }) => void; + _getMapTouches(touches: TouchList): TouchList; + handleEvent: (e: Event, eventName?: keyof Handler) => void; + mergeHandlerResult(mergedHandlerResult: HandlerResult, eventsInProgress: EventsInProgress, handlerResult: HandlerResult, name: string, e?: InputEvent): void; + _applyChanges(): void; + _updateMapTransform(combinedResult: HandlerResult, combinedEventsInProgress: EventsInProgress, deactivatedHandlers: { + [handlerName: string]: Event; + }): void; + _fireEvents(newEventsInProgress: EventsInProgress, deactivatedHandlers: { + [handlerName: string]: Event; + }, allowEndAnimation: boolean): void; + _fireEvent(type: string, e?: Event): void; + _requestFrame(): number; + _triggerRenderFrame(): void; +} +export type TaskID = number; +export type Task = { + callback: (timeStamp: number) => void; + id: TaskID; + cancelled: boolean; +}; +export declare class TaskQueue { + _queue: Array; + _id: TaskID; + _cleared: boolean; + _currentlyRunning: Array | false; + constructor(); + add(callback: (timeStamp: number) => void): TaskID; + remove(id: TaskID): void; + run(timeStamp?: number): void; + clear(): void; +} +/** + * A [Point](https://github.com/mapbox/point-geometry) or an array of two numbers representing `x` and `y` screen coordinates in pixels. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let p1 = new Point(-77, 38); // a PointLike which is a Point + * let p2 = [-77, 38]; // a PointLike which is an array of two numbers + * ``` + */ +export type PointLike = Point | [ + number, + number +]; +/** + * A helper to allow require of at least one propery + */ +export type RequireAtLeastOne = { + [K in keyof T]-?: Required> & Partial>>; +}[keyof T]; +/** + * Options common to {@link Map#jumpTo}, {@link Map#easeTo}, and {@link Map#flyTo}, controlling the desired location, + * zoom, bearing, and pitch of the camera. All properties are optional, and when a property is omitted, the current + * camera value for that property will remain unchanged. + * + * @example + * Set the map's initial perspective with CameraOptions + * ```ts + * let map = new maplibregl.Map({ + * container: 'map', + * style: 'https://demotiles.maplibre.org/style.json', + * center: [-73.5804, 45.53483], + * pitch: 60, + * bearing: -60, + * zoom: 10 + * }); + * ``` + * @see [Set pitch and bearing](https://maplibre.org/maplibre-gl-js/docs/examples/set-perspective/) + * @see [Jump to a series of locations](https://maplibre.org/maplibre-gl-js/docs/examples/jump-to/) + * @see [Fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto/) + * @see [Display buildings in 3D](https://maplibre.org/maplibre-gl-js/docs/examples/3d-buildings/) + */ +export type CameraOptions = CenterZoomBearing & { + /** + * The desired pitch in degrees. The pitch is the angle towards the horizon + * measured in degrees with a range between 0 and 60 degrees. For example, pitch: 0 provides the appearance + * of looking straight down at the map, while pitch: 60 tilts the user's perspective towards the horizon. + * Increasing the pitch value is often used to display 3D objects. + */ + pitch?: number; + /** + * If `zoom` is specified, `around` determines the point around which the zoom is centered. + */ + around?: LngLatLike; +}; +/** + * Holds center, zoom and bearing properties + */ +export type CenterZoomBearing = { + /** + * The desired center. + */ + center?: LngLatLike; + /** + * The desired zoom level. + */ + zoom?: number; + /** + * The desired bearing in degrees. The bearing is the compass direction that + * is "up". For example, `bearing: 90` orients the map so that east is up. + */ + bearing?: number; +}; +/** + * The options object related to the {@link Map#jumpTo} method + */ +export type JumpToOptions = CameraOptions & { + /** + * Dimensions in pixels applied on each side of the viewport for shifting the vanishing point. + */ + padding?: PaddingOptions; +}; +/** + * A options object for the {@link Map#cameraForBounds} method + */ +export type CameraForBoundsOptions = CameraOptions & { + /** + * The amount of padding in pixels to add to the given bounds. + */ + padding?: number | RequireAtLeastOne; + /** + * The center of the given bounds relative to the map's center, measured in pixels. + * @defaultValue [0, 0] + */ + offset?: PointLike; + /** + * The maximum zoom level to allow when the camera would transition to the specified bounds. + */ + maxZoom?: number; +}; +/** + * The {@link Map#flyTo} options object + */ +export type FlyToOptions = AnimationOptions & CameraOptions & { + /** + * The zooming "curve" that will occur along the + * flight path. A high value maximizes zooming for an exaggerated animation, while a low + * value minimizes zooming for an effect closer to {@link Map#easeTo}. 1.42 is the average + * value selected by participants in the user study discussed in + * [van Wijk (2003)](https://www.win.tue.nl/~vanwijk/zoompan.pdf). A value of + * `Math.pow(6, 0.25)` would be equivalent to the root mean squared average velocity. A + * value of 1 would produce a circular motion. + * @defaultValue 1.42 + */ + curve?: number; + /** + * The zero-based zoom level at the peak of the flight path. If + * `options.curve` is specified, this option is ignored. + */ + minZoom?: number; + /** + * The average speed of the animation defined in relation to + * `options.curve`. A speed of 1.2 means that the map appears to move along the flight path + * by 1.2 times `options.curve` screenfuls every second. A _screenful_ is the map's visible span. + * It does not correspond to a fixed physical distance, but varies by zoom level. + * @defaultValue 1.2 + */ + speed?: number; + /** + * The average speed of the animation measured in screenfuls + * per second, assuming a linear timing curve. If `options.speed` is specified, this option is ignored. + */ + screenSpeed?: number; + /** + * The animation's maximum duration, measured in milliseconds. + * If duration exceeds maximum duration, it resets to 0. + */ + maxDuration?: number; + /** + * The amount of padding in pixels to add to the given bounds. + */ + padding?: number | RequireAtLeastOne; +}; +export type EaseToOptions = AnimationOptions & CameraOptions & { + delayEndEvents?: number; + padding?: number | RequireAtLeastOne; +}; +/** + * Options for {@link Map#fitBounds} method + */ +export type FitBoundsOptions = FlyToOptions & { + /** + * If `true`, the map transitions using {@link Map#easeTo}. If `false`, the map transitions using {@link Map#flyTo}. + * See those functions and {@link AnimationOptions} for information about options available. + * @defaultValue false + */ + linear?: boolean; + /** + * The center of the given bounds relative to the map's center, measured in pixels. + * @defaultValue [0, 0] + */ + offset?: PointLike; + /** + * The maximum zoom level to allow when the map view transitions to the specified bounds. + */ + maxZoom?: number; +}; +/** + * Options common to map movement methods that involve animation, such as {@link Map#panBy} and + * {@link Map#easeTo}, controlling the duration and easing function of the animation. All properties + * are optional. + * + */ +export type AnimationOptions = { + /** + * The animation's duration, measured in milliseconds. + */ + duration?: number; + /** + * A function taking a time in the range 0..1 and returning a number where 0 is + * the initial state and 1 is the final state. + */ + easing?: (_: number) => number; + /** + * of the target center relative to real map container center at the end of animation. + */ + offset?: PointLike; + /** + * If `false`, no animation will occur. + */ + animate?: boolean; + /** + * If `true`, then the animation is considered essential and will not be affected by + * [`prefers-reduced-motion`](https://developer.mozilla.org/en-US/docs/Web/CSS/\@media/prefers-reduced-motion). + */ + essential?: boolean; + /** + * Default false. Needed in 3D maps to let the camera stay in a constant + * height based on sea-level. After the animation finished the zoom-level will be recalculated in respect of + * the distance from the camera to the center-coordinate-altitude. + */ + freezeElevation?: boolean; +}; +/** + * A callback hook that allows manipulating the camera and being notified about camera updates before they happen + */ +export type CameraUpdateTransformFunction = (next: { + center: LngLat; + zoom: number; + pitch: number; + bearing: number; + elevation: number; +}) => { + center?: LngLat; + zoom?: number; + pitch?: number; + bearing?: number; + elevation?: number; +}; +declare abstract class Camera extends Evented { + transform: Transform; + terrain: Terrain; + _moving: boolean; + _zooming: boolean; + _rotating: boolean; + _pitching: boolean; + _padding: boolean; + _bearingSnap: number; + _easeStart: number; + _easeOptions: { + duration?: number; + easing?: (_: number) => number; + }; + _easeId: string | void; + _onEaseFrame: (_: number) => void; + _onEaseEnd: (easeId?: string) => void; + _easeFrameId: TaskID; + /** + * @internal + * holds the geographical coordinate of the target + */ + _elevationCenter: LngLat; + /** + * @internal + * holds the targ altitude value, = center elevation of the target. + * This value may changes during flight, because new terrain-tiles loads during flight. + */ + _elevationTarget: number; + /** + * @internal + * holds the start altitude value, = center elevation before animation begins + * this value will recalculated during flight in respect of changing _elevationTarget values, + * so the linear interpolation between start and target keeps smooth and without jumps. + */ + _elevationStart: number; + /** + * @internal + * Saves the current state of the elevation freeze - this is used during map movement to prevent "rocky" camera movement. + */ + _elevationFreeze: boolean; + /** + * @internal + * Used to track accumulated changes during continuous interaction + */ + _requestedCameraState?: Transform; + /** + * A callback used to defer camera updates or apply arbitrary constraints. + * If specified, this Camera instance can be used as a stateless component in React etc. + */ + transformCameraUpdate: CameraUpdateTransformFunction | null; + abstract _requestRenderFrame(a: () => void): TaskID; + abstract _cancelRenderFrame(_: TaskID): void; + constructor(transform: Transform, options: { + bearingSnap: number; + }); + /** + * Returns the map's geographical centerpoint. + * + * @returns The map's geographical centerpoint. + * @example + * Return a LngLat object such as `{lng: 0, lat: 0}` + * ```ts + * let center = map.getCenter(); + * // access longitude and latitude values directly + * let {lng, lat} = map.getCenter(); + * ``` + */ + getCenter(): LngLat; + /** + * Sets the map's geographical centerpoint. Equivalent to `jumpTo({center: center})`. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param center - The centerpoint to set. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * map.setCenter([-74, 38]); + * ``` + */ + setCenter(center: LngLatLike, eventData?: any): this; + /** + * Pans the map by the specified offset. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param offset - `x` and `y` coordinates by which to pan the map. + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/) + */ + panBy(offset: PointLike, options?: AnimationOptions, eventData?: any): this; + /** + * Pans the map to the specified location with an animated transition. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param lnglat - The location to pan the map to. + * @param options - Options describing the destination and animation of the transition. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * map.panTo([-74, 38]); + * // Specify that the panTo animation should last 5000 milliseconds. + * map.panTo([-74, 38], {duration: 5000}); + * ``` + * @see [Update a feature in realtime](https://maplibre.org/maplibre-gl-js/docs/examples/live-update-feature/) + */ + panTo(lnglat: LngLatLike, options?: AnimationOptions, eventData?: any): this; + /** + * Returns the map's current zoom level. + * + * @returns The map's current zoom level. + * @example + * ```ts + * map.getZoom(); + * ``` + */ + getZoom(): number; + /** + * Sets the map's zoom level. Equivalent to `jumpTo({zoom: zoom})`. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`. + * + * @param zoom - The zoom level to set (0-20). + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Zoom to the zoom level 5 without an animated transition + * ```ts + * map.setZoom(5); + * ``` + */ + setZoom(zoom: number, eventData?: any): this; + /** + * Zooms the map to the specified zoom level, with an animated transition. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`. + * + * @param zoom - The zoom level to transition to. + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * // Zoom to the zoom level 5 without an animated transition + * map.zoomTo(5); + * // Zoom to the zoom level 8 with an animated transition + * map.zoomTo(8, { + * duration: 2000, + * offset: [100, 50] + * }); + * ``` + */ + zoomTo(zoom: number, options?: AnimationOptions | null, eventData?: any): this; + /** + * Increases the map's zoom level by 1. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Zoom the map in one level with a custom animation duration + * ```ts + * map.zoomIn({duration: 1000}); + * ``` + */ + zoomIn(options?: AnimationOptions, eventData?: any): this; + /** + * Decreases the map's zoom level by 1. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Zoom the map out one level with a custom animation offset + * ```ts + * map.zoomOut({offset: [80, 60]}); + * ``` + */ + zoomOut(options?: AnimationOptions, eventData?: any): this; + /** + * Returns the map's current bearing. The bearing is the compass direction that is "up"; for example, a bearing + * of 90° orients the map so that east is up. + * + * @returns The map's current bearing. + * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/) + */ + getBearing(): number; + /** + * Sets the map's bearing (rotation). The bearing is the compass direction that is "up"; for example, a bearing + * of 90° orients the map so that east is up. + * + * Equivalent to `jumpTo({bearing: bearing})`. + * + * Triggers the following events: `movestart`, `moveend`, and `rotate`. + * + * @param bearing - The desired bearing. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Rotate the map to 90 degrees + * ```ts + * map.setBearing(90); + * ``` + */ + setBearing(bearing: number, eventData?: any): this; + /** + * Returns the current padding applied around the map viewport. + * + * @returns The current padding around the map viewport. + */ + getPadding(): PaddingOptions; + /** + * Sets the padding in pixels around the viewport. + * + * Equivalent to `jumpTo({padding: padding})`. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param padding - The desired padding. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Sets a left padding of 300px, and a top padding of 50px + * ```ts + * map.setPadding({ left: 300, top: 50 }); + * ``` + */ + setPadding(padding: PaddingOptions, eventData?: any): this; + /** + * Rotates the map to the specified bearing, with an animated transition. The bearing is the compass direction + * that is "up"; for example, a bearing of 90° orients the map so that east is up. + * + * Triggers the following events: `movestart`, `moveend`, and `rotate`. + * + * @param bearing - The desired bearing. + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + rotateTo(bearing: number, options?: AnimationOptions, eventData?: any): this; + /** + * Rotates the map so that north is up (0° bearing), with an animated transition. + * + * Triggers the following events: `movestart`, `moveend`, and `rotate`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + resetNorth(options?: AnimationOptions, eventData?: any): this; + /** + * Rotates and pitches the map so that north is up (0° bearing) and pitch is 0°, with an animated transition. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `pitchstart`, `pitch`, `pitchend`, and `rotate`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + resetNorthPitch(options?: AnimationOptions, eventData?: any): this; + /** + * Snaps the map so that north is up (0° bearing), if the current bearing is close enough to it (i.e. within the + * `bearingSnap` threshold). + * + * Triggers the following events: `movestart`, `moveend`, and `rotate`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + snapToNorth(options?: AnimationOptions, eventData?: any): this; + /** + * Returns the map's current pitch (tilt). + * + * @returns The map's current pitch, measured in degrees away from the plane of the screen. + */ + getPitch(): number; + /** + * Sets the map's pitch (tilt). Equivalent to `jumpTo({pitch: pitch})`. + * + * Triggers the following events: `movestart`, `moveend`, `pitchstart`, and `pitchend`. + * + * @param pitch - The pitch to set, measured in degrees away from the plane of the screen (0-60). + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + setPitch(pitch: number, eventData?: any): this; + /** + * @param bounds - Calculate the center for these bounds in the viewport and use + * the highest zoom level up to and including `Map#getMaxZoom()` that fits + * in the viewport. LngLatBounds represent a box that is always axis-aligned with bearing 0. + * @param options - Options object + * @returns If map is able to fit to provided bounds, returns `center`, `zoom`, and `bearing`. + * If map is unable to fit, method will warn and return undefined. + * @example + * ```ts + * let bbox = [[-79, 43], [-73, 45]]; + * let newCameraTransform = map.cameraForBounds(bbox, { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + */ + cameraForBounds(bounds: LngLatBoundsLike, options?: CameraForBoundsOptions): CenterZoomBearing; + /** + * @internal + * Calculate the center of these two points in the viewport and use + * the highest zoom level up to and including `Map#getMaxZoom()` that fits + * the points in the viewport at the specified bearing. + * @param p0 - First point + * @param p1 - Second point + * @param bearing - Desired map bearing at end of animation, in degrees + * @param options - the camera options + * @returns If map is able to fit to provided bounds, returns `center`, `zoom`, and `bearing`. + * If map is unable to fit, method will warn and return undefined. + * @example + * ```ts + * let p0 = [-79, 43]; + * let p1 = [-73, 45]; + * let bearing = 90; + * let newCameraTransform = map._cameraForBoxAndBearing(p0, p1, bearing, { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + */ + _cameraForBoxAndBearing(p0: LngLatLike, p1: LngLatLike, bearing: number, options?: CameraForBoundsOptions): CenterZoomBearing; + /** + * Pans and zooms the map to contain its visible area within the specified geographical bounds. + * This function will also reset the map's bearing to 0 if bearing is nonzero. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param bounds - Center these bounds in the viewport and use the highest + * zoom level up to and including `Map#getMaxZoom()` that fits them in the viewport. + * @param options- Options supports all properties from {@link AnimationOptions} and {@link CameraOptions} in addition to the fields below. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * let bbox = [[-79, 43], [-73, 45]]; + * map.fitBounds(bbox, { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + * @see [Fit a map to a bounding box](https://maplibre.org/maplibre-gl-js/docs/examples/fitbounds/) + */ + fitBounds(bounds: LngLatBoundsLike, options?: FitBoundsOptions, eventData?: any): this; + /** + * Pans, rotates and zooms the map to to fit the box made by points p0 and p1 + * once the map is rotated to the specified bearing. To zoom without rotating, + * pass in the current map bearing. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend` and `rotate`. + * + * @param p0 - First point on screen, in pixel coordinates + * @param p1 - Second point on screen, in pixel coordinates + * @param bearing - Desired map bearing at end of animation, in degrees + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * let p0 = [220, 400]; + * let p1 = [500, 900]; + * map.fitScreenCoordinates(p0, p1, map.getBearing(), { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + * @see Used by {@link BoxZoomHandler} + */ + fitScreenCoordinates(p0: PointLike, p1: PointLike, bearing: number, options?: FitBoundsOptions, eventData?: any): this; + _fitInternal(calculatedOptions?: CenterZoomBearing, options?: FitBoundsOptions, eventData?: any): this; + /** + * Changes any combination of center, zoom, bearing, and pitch, without + * an animated transition. The map will retain its current values for any + * details not specified in `options`. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`, + * `pitch`, `pitchend`, and `rotate`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * // jump to coordinates at current zoom + * map.jumpTo({center: [0, 0]}); + * // jump with zoom, pitch, and bearing options + * map.jumpTo({ + * center: [0, 0], + * zoom: 8, + * pitch: 45, + * bearing: 90 + * }); + * ``` + * @see [Jump to a series of locations](https://maplibre.org/maplibre-gl-js/docs/examples/jump-to/) + * @see [Update a feature in realtime](https://maplibre.org/maplibre-gl-js/docs/examples/live-update-feature/) + */ + jumpTo(options: JumpToOptions, eventData?: any): this; + /** + * Calculates pitch, zoom and bearing for looking at `newCenter` with the camera position being `newCenter` + * and returns them as {@link CameraOptions}. + * @param from - The camera to look from + * @param altitudeFrom - The altitude of the camera to look from + * @param to - The center to look at + * @param altitudeTo - Optional altitude of the center to look at. If none given the ground height will be used. + * @returns the calculated camera options + */ + calculateCameraOptionsFromTo(from: LngLat, altitudeFrom: number, to: LngLat, altitudeTo?: number): CameraOptions; + /** + * Changes any combination of `center`, `zoom`, `bearing`, `pitch`, and `padding` with an animated transition + * between old and new values. The map will retain its current values for any + * details not specified in `options`. + * + * Note: The transition will happen instantly if the user has enabled + * the `reduced motion` accessibility feature enabled in their operating system, + * unless `options` includes `essential: true`. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`, + * `pitch`, `pitchend`, and `rotate`. + * + * @param options - Options describing the destination and animation of the transition. + * Accepts {@link CameraOptions} and {@link AnimationOptions}. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/) + */ + easeTo(options: EaseToOptions & { + easeId?: string; + noMoveStart?: boolean; + }, eventData?: any): this; + _prepareEase(eventData: any, noMoveStart: boolean, currently?: any): void; + _prepareElevation(center: LngLat): void; + _updateElevation(k: number): void; + _finalizeElevation(): void; + /** + * @internal + * Called when the camera is about to be manipulated. + * If `transformCameraUpdate` is specified, a copy of the current transform is created to track the accumulated changes. + * This underlying transform represents the "desired state" proposed by input handlers / animations / UI controls. + * It may differ from the state used for rendering (`this.transform`). + * @returns Transform to apply changes to + */ + _getTransformForUpdate(): Transform; + /** + * @internal + * Called after the camera is done being manipulated. + * @param tr - the requested camera end state + * Call `transformCameraUpdate` if present, and then apply the "approved" changes. + */ + _applyUpdatedTransform(tr: Transform): void; + _fireMoveEvents(eventData?: any): void; + _afterEase(eventData?: any, easeId?: string): void; + /** + * Changes any combination of center, zoom, bearing, and pitch, animating the transition along a curve that + * evokes flight. The animation seamlessly incorporates zooming and panning to help + * the user maintain her bearings even after traversing a great distance. + * + * Note: The animation will be skipped, and this will behave equivalently to `jumpTo` + * if the user has the `reduced motion` accessibility feature enabled in their operating system, + * unless 'options' includes `essential: true`. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`, + * `pitch`, `pitchend`, and `rotate`. + * + * @param options - Options describing the destination and animation of the transition. + * Accepts {@link CameraOptions}, {@link AnimationOptions}, + * and the following additional options. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * // fly with default options to null island + * map.flyTo({center: [0, 0], zoom: 9}); + * // using flyTo options + * map.flyTo({ + * center: [0, 0], + * zoom: 9, + * speed: 0.2, + * curve: 1, + * easing(t) { + * return t; + * } + * }); + * ``` + * @see [Fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto/) + * @see [Slowly fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto-options/) + * @see [Fly to a location based on scroll position](https://maplibre.org/maplibre-gl-js/docs/examples/scroll-fly-to/) + */ + flyTo(options: FlyToOptions, eventData?: any): this; + isEasing(): boolean; + /** + * Stops any animated transition underway. + * + * @returns `this` + */ + stop(): this; + _stop(allowGestures?: boolean, easeId?: string): this; + _ease(frame: (_: number) => void, finish: () => void, options: { + animate?: boolean; + duration?: number; + easing?: (_: number) => number; + }): void; + _renderFrameCallback: () => void; + _normalizeBearing(bearing: number, currentBearing: number): number; + _normalizeCenter(center: LngLat): void; + /** + * Query the current elevation of location. It return null if terrain is not enabled. the elevation is in meters relative to mean sea-level + * @param lngLatLike - [x,y] or LngLat coordinates of the location + * @returns elevation in meters + */ + queryTerrainElevation(lngLatLike: LngLatLike): number | null; +} +/** + * An event from the mouse relevant to a specific layer. + * + * @group Event Related + */ +export type MapLayerMouseEvent = MapMouseEvent & { + features?: MapGeoJSONFeature[]; +}; +/** + * An event from a touch device relevat to a specific layer. + * + * @group Event Related + */ +export type MapLayerTouchEvent = MapTouchEvent & { + features?: MapGeoJSONFeature[]; +}; +/** + * The source event data type + */ +export type MapSourceDataType = "content" | "metadata" | "visibility" | "idle"; +/** + * `MapLayerEventType` - a mapping between the event name and the event. + * **Note:** These events are compatible with the optional `layerId` parameter. + * If `layerId` is included as the second argument in {@link Map#on}, the event listener will fire only when the + * event action contains a visible portion of the specified layer. + * The following example can be used for all the events. + * + * @group Event Related + * @example + * ```ts + * // Initialize the map + * let map = new maplibregl.Map({ // map options }); + * // Set an event listener for a specific layer + * map.on('the-event-name', 'poi-label', function(e) { + * console.log('An event has occurred on a visible portion of the poi-label layer'); + * }); + * ``` + */ +export type MapLayerEventType = { + /** + * Fired when a pointing device (usually a mouse) is pressed and released contains a visible portion of the specified layer. + * + * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/) + * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/) + */ + click: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is pressed and released twice contains a visible portion of the specified layer. + * + * **Note:** Under normal conditions, this event will be preceded by two `click` events. + */ + dblclick: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is pressed while inside a visible portion of the specified layer. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + mousedown: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is released while inside a visible portion of the specified layer. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + mouseup: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is moved while the cursor is inside a visible portion of the specified layer. + * As you move the cursor across the layer, the event will fire every time the cursor changes position within that layer. + * + * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/) + * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + * @see [Display a popup on over](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + */ + mousemove: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) enters a visible portion of a specified layer from + * outside that layer or outside the map canvas. + * + * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/) + * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + */ + mouseenter: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) leaves a visible portion of a specified layer, or leaves + * the map canvas. + * + * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + */ + mouseleave: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is moved inside a visible portion of the specified layer. + * + * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/) + * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + */ + mouseover: MapLayerMouseEvent; + /** + * Fired when a point device (usually a mouse) leaves the visible portion of the specified layer. + */ + mouseout: MapLayerMouseEvent; + /** + * Fired when the right button of the mouse is clicked or the context menu key is pressed within visible portion of the specified layer. + */ + contextmenu: MapLayerMouseEvent; + /** + * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the visible portion of the specified layer. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + touchstart: MapLayerTouchEvent; + /** + * Fired when a [`touchend`](https://developer.mozilla.org/en-US/docs/Web/Events/touchend) event occurs within the visible portion of the specified layer. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + touchend: MapLayerTouchEvent; + /** + * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the visible portion of the specified layer. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + touchcancel: MapLayerTouchEvent; +}; +/** + * `MapEventType` - a mapping between the event name and the event value. + * These events are used with the {@link Map#on} method. + * When using a `layerId` with {@link Map#on} method, please refer to {@link MapLayerEventType}. + * The following example can be used for all the events. + * + * @group Event Related + * @example + * ```ts + * // Initialize the map + * let map = new maplibregl.Map({ // map options }); + * // Set an event listener + * map.on('the-event-name', () => { + * console.log('An event has occurred!'); + * }); + * ``` + */ +export type MapEventType = { + /** + * Fired when an error occurs. This is GL JS's primary error reporting + * mechanism. We use an event instead of `throw` to better accommodate + * asynchronous operations. If no listeners are bound to the `error` event, the + * error will be printed to the console. + */ + error: ErrorEvent; + /** + * @event `load` Fired immediately after all necessary resources have been downloaded + * and the first visually complete rendering of the map has occurred. + * + * @see [Draw GeoJSON points](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-markers/) + * @see [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/) + * @see [Animate a point](https://maplibre.org/maplibre-gl-js/docs/examples/animate-point-along-line/) + */ + load: MapLibreEvent; + /** + * Fired after the last frame rendered before the map enters an + * "idle" state: + * + * - No camera transitions are in progress + * - All currently requested tiles have loaded + * - All fade/transition animations have completed + */ + idle: MapLibreEvent; + /** + * Fired immediately after the map has been removed with {@link Map#remove}. + */ + remove: MapLibreEvent; + /** + * Fired whenever the map is drawn to the screen, as the result of + * + * - a change to the map's position, zoom, pitch, or bearing + * - a change to the map's style + * - a change to a GeoJSON source + * - the loading of a vector tile, GeoJSON file, glyph, or sprite + */ + render: MapLibreEvent; + /** + * Fired immediately after the map has been resized. + */ + resize: MapLibreEvent; + /** + * Fired when the WebGL context is lost. + */ + webglcontextlost: MapContextEvent; + /** + * Fired when the WebGL context is restored. + */ + webglcontextrestored: MapContextEvent; + /** + * Fired when any map data (style, source, tile, etc) begins loading or + * changing asynchronously. All `dataloading` events are followed by a `data`, + * `dataabort` or `error` event. + */ + dataloading: MapDataEvent; + /** + * Fired when any map data loads or changes. See {@link MapDataEvent} for more information. + * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/) + */ + data: MapDataEvent; + tiledataloading: MapDataEvent; + /** + * Fired when one of the map's sources begins loading or changing asynchronously. + * All `sourcedataloading` events are followed by a `sourcedata`, `sourcedataabort` or `error` event. + */ + sourcedataloading: MapSourceDataEvent; + /** + * Fired when the map's style begins loading or changing asynchronously. + * All `styledataloading` events are followed by a `styledata` + * or `error` event. + */ + styledataloading: MapStyleDataEvent; + /** + * Fired when one of the map's sources loads or changes, including if a tile belonging + * to a source loads or changes. + */ + sourcedata: MapSourceDataEvent; + /** + * Fired when the map's style loads or changes. + */ + styledata: MapStyleDataEvent; + /** + * Fired when an icon or pattern needed by the style is missing. The missing image can + * be added with {@link Map#addImage} within this event listener callback to prevent the image from + * being skipped. This event can be used to dynamically generate icons and patterns. + * @see [Generate and add a missing icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-missing-generated/) + */ + styleimagemissing: MapStyleImageMissingEvent; + /** + * Fired when a request for one of the map's sources' tiles or data is aborted. + */ + dataabort: MapDataEvent; + /** + * Fired when a request for one of the map's sources' data is aborted. + */ + sourcedataabort: MapSourceDataEvent; + /** + * Fired when the user cancels a "box zoom" interaction, or when the bounding box does not meet the minimum size threshold. + * See {@link BoxZoomHandler}. + */ + boxzoomcancel: MapLibreZoomEvent; + /** + * Fired when a "box zoom" interaction starts. See {@link BoxZoomHandler}. + */ + boxzoomstart: MapLibreZoomEvent; + /** + * Fired when a "box zoom" interaction ends. See {@link BoxZoomHandler}. + */ + boxzoomend: MapLibreZoomEvent; + /** + * Fired when a [`touchcancel`](https://developer.mozilla.org/en-US/docs/Web/Events/touchcancel) event occurs within the map. + */ + touchcancel: MapTouchEvent; + /** + * Fired when a [`touchmove`](https://developer.mozilla.org/en-US/docs/Web/Events/touchmove) event occurs within the map. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + touchmove: MapTouchEvent; + /** + * Fired when a [`touchend`](https://developer.mozilla.org/en-US/docs/Web/Events/touchend) event occurs within the map. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + touchend: MapTouchEvent; + /** + * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the map. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + touchstart: MapTouchEvent; + /** + * Fired when a pointing device (usually a mouse) is pressed and released at the same point on the map. + * + * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/) + * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/) + */ + click: MapMouseEvent; + /** + * Fired when the right button of the mouse is clicked or the context menu key is pressed within the map. + */ + contextmenu: MapMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is pressed and released twice at the same point on the map in rapid succession. + * + * **Note:** Under normal conditions, this event will be preceded by two `click` events. + */ + dblclick: MapMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is moved while the cursor is inside the map. + * As you move the cursor across the map, the event will fire every time the cursor changes position within the map. + * + * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/) + * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + * @see [Display a popup on over](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + */ + mousemove: MapMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is released within the map. + * + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + mouseup: MapMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is pressed within the map. + * + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + mousedown: MapMouseEvent; + /** + * Fired when a point device (usually a mouse) leaves the map's canvas. + */ + mouseout: MapMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is moved within the map. + * As you move the cursor across a web page containing a map, + * the event will fire each time it enters the map or any child elements. + * + * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/) + * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + */ + mouseover: MapMouseEvent; + /** + * Fired just before the map begins a transition from one + * view to another, as the result of either user interaction or methods such as {@link Map#jumpTo}. + * + */ + movestart: MapLibreEvent; + /** + * Fired repeatedly during an animated transition from one view to + * another, as the result of either user interaction or methods such as {@link Map#flyTo}. + * + * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/) + */ + move: MapLibreEvent; + /** + * Fired just after the map completes a transition from one + * view to another, as the result of either user interaction or methods such as {@link Map#jumpTo}. + * + * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/) + */ + moveend: MapLibreEvent; + /** + * Fired just before the map begins a transition from one zoom level to another, + * as the result of either user interaction or methods such as {@link Map#flyTo}. + */ + zoomstart: MapLibreEvent; + /** + * Fired repeatedly during an animated transition from one zoom level to another, + * as the result of either user interaction or methods such as {@link Map#flyTo}. + */ + zoom: MapLibreEvent; + /** + * Fired just after the map completes a transition from one zoom level to another, + * as the result of either user interaction or methods such as {@link Map#flyTo}. + */ + zoomend: MapLibreEvent; + /** + * Fired when a "drag to rotate" interaction starts. See {@link DragRotateHandler}. + */ + rotatestart: MapLibreEvent; + /** + * Fired repeatedly during a "drag to rotate" interaction. See {@link DragRotateHandler}. + */ + rotate: MapLibreEvent; + /** + * Fired when a "drag to rotate" interaction ends. See {@link DragRotateHandler}. + */ + rotateend: MapLibreEvent; + /** + * Fired when a "drag to pan" interaction starts. See {@link DragPanHandler}. + */ + dragstart: MapLibreEvent; + /** + * Fired repeatedly during a "drag to pan" interaction. See {@link DragPanHandler}. + */ + drag: MapLibreEvent; + /** + * Fired when a "drag to pan" interaction ends. See {@link DragPanHandler}. + * @see [Create a draggable marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/) + */ + dragend: MapLibreEvent; + /** + * Fired whenever the map's pitch (tilt) begins a change as + * the result of either user interaction or methods such as {@link Map#flyTo} . + */ + pitchstart: MapLibreEvent; + /** + * Fired repeatedly during the map's pitch (tilt) animation between + * one state and another as the result of either user interaction + * or methods such as {@link Map#flyTo}. + */ + pitch: MapLibreEvent; + /** + * Fired immediately after the map's pitch (tilt) finishes changing as + * the result of either user interaction or methods such as {@link Map#flyTo}. + */ + pitchend: MapLibreEvent; + /** + * Fired when a [`wheel`](https://developer.mozilla.org/en-US/docs/Web/Events/wheel) event occurs within the map. + */ + wheel: MapWheelEvent; + /** + * Fired when terrain is changed + */ + terrain: MapTerrainEvent; +}; +/** + * The base event for MapLibre + * + * @group Event Related + */ +export type MapLibreEvent = { + type: keyof MapEventType | keyof MapLayerEventType; + target: Map; + originalEvent: TOrig; +}; +/** + * The style data event + * + * @group Event Related + */ +export type MapStyleDataEvent = MapLibreEvent & { + dataType: "style"; +}; +/** + * The source data event interface + * + * @group Event Related + */ +export type MapSourceDataEvent = MapLibreEvent & { + dataType: "source"; + /** + * True if the event has a `dataType` of `source` and the source has no outstanding network requests. + */ + isSourceLoaded: boolean; + /** + * The [style spec representation of the source](https://maplibre.org/maplibre-style-spec/#sources) if the event has a `dataType` of `source`. + */ + source: SourceSpecification; + sourceId: string; + sourceDataType: MapSourceDataType; + /** + * The tile being loaded or changed, if the event has a `dataType` of `source` and + * the event is related to loading of a tile. + */ + tile: any; +}; +export declare class MapMouseEvent extends Event implements MapLibreEvent { + /** + * The event type + */ + type: "mousedown" | "mouseup" | "click" | "dblclick" | "mousemove" | "mouseover" | "mouseenter" | "mouseleave" | "mouseout" | "contextmenu"; + /** + * The `Map` object that fired the event. + */ + target: Map; + /** + * The DOM event which caused the map event. + */ + originalEvent: MouseEvent; + /** + * The pixel coordinates of the mouse cursor, relative to the map and measured from the top left corner. + */ + point: Point; + /** + * The geographic location on the map of the mouse cursor. + */ + lngLat: LngLat; + /** + * Prevents subsequent default processing of the event by the map. + * + * Calling this method will prevent the following default map behaviors: + * + * * On `mousedown` events, the behavior of {@link DragPanHandler} + * * On `mousedown` events, the behavior of {@link DragRotateHandler} + * * On `mousedown` events, the behavior of {@link BoxZoomHandler} + * * On `dblclick` events, the behavior of {@link DoubleClickZoomHandler} + * + */ + preventDefault(): void; + /** + * `true` if `preventDefault` has been called. + */ + get defaultPrevented(): boolean; + _defaultPrevented: boolean; + constructor(type: string, map: Map, originalEvent: MouseEvent, data?: any); +} +export declare class MapTouchEvent extends Event implements MapLibreEvent { + /** + * The event type. + */ + type: "touchstart" | "touchmove" | "touchend" | "touchcancel"; + /** + * The `Map` object that fired the event. + */ + target: Map; + /** + * The DOM event which caused the map event. + */ + originalEvent: TouchEvent; + /** + * The geographic location on the map of the center of the touch event points. + */ + lngLat: LngLat; + /** + * The pixel coordinates of the center of the touch event points, relative to the map and measured from the top left + * corner. + */ + point: Point; + /** + * The array of pixel coordinates corresponding to a + * [touch event's `touches`](https://developer.mozilla.org/en-US/docs/Web/API/TouchEvent/touches) property. + */ + points: Array; + /** + * The geographical locations on the map corresponding to a + * [touch event's `touches`](https://developer.mozilla.org/en-US/docs/Web/API/TouchEvent/touches) property. + */ + lngLats: Array; + /** + * Prevents subsequent default processing of the event by the map. + * + * Calling this method will prevent the following default map behaviors: + * + * * On `touchstart` events, the behavior of {@link DragPanHandler} + * * On `touchstart` events, the behavior of {@link TwoFingersTouchZoomRotateHandler} + * + */ + preventDefault(): void; + /** + * `true` if `preventDefault` has been called. + */ + get defaultPrevented(): boolean; + _defaultPrevented: boolean; + constructor(type: string, map: Map, originalEvent: TouchEvent); +} +export declare class MapWheelEvent extends Event { + /** + * The event type. + */ + type: "wheel"; + /** + * The `Map` object that fired the event. + */ + target: Map; + /** + * The DOM event which caused the map event. + */ + originalEvent: WheelEvent; + /** + * Prevents subsequent default processing of the event by the map. + * + * Calling this method will prevent the behavior of {@link ScrollZoomHandler}. + */ + preventDefault(): void; + /** + * `true` if `preventDefault` has been called. + */ + get defaultPrevented(): boolean; + _defaultPrevented: boolean; + /** */ + constructor(type: string, map: Map, originalEvent: WheelEvent); +} +/** + * A `MapLibreZoomEvent` is the event type for the boxzoom-related map events emitted by the {@link BoxZoomHandler}. + * + * @group Event Related + */ +export type MapLibreZoomEvent = { + /** + * The type of boxzoom event. One of `boxzoomstart`, `boxzoomend` or `boxzoomcancel` + */ + type: "boxzoomstart" | "boxzoomend" | "boxzoomcancel"; + /** + * The `Map` instance that triggered the event + */ + target: Map; + /** + * The DOM event that triggered the boxzoom event. Can be a `MouseEvent` or `KeyboardEvent` + */ + originalEvent: MouseEvent; +}; +/** + * A `MapDataEvent` object is emitted with the `data` + * and `dataloading` events. Possible values for + * `dataType`s are: + * + * - `'source'`: The non-tile data associated with any source + * - `'style'`: The [style](https://maplibre.org/maplibre-style-spec/) used by the map + * + * Possible values for `sourceDataType`s are: + * + * - `'metadata'`: indicates that any necessary source metadata has been loaded (such as TileJSON) and it is ok to start loading tiles + * - `'content'`: indicates the source data has changed (such as when source.setData() has been called on GeoJSONSource) + * - `'visibility'`: send when the source becomes used when at least one of its layers becomes visible in style sense (inside the layer's zoom range and with layout.visibility set to 'visible') + * - `'idle'`: indicates that no new source data has been fetched (but the source has done loading) + * + * @group Event Related + * + * @example + * ```ts + * // The sourcedata event is an example of MapDataEvent. + * // Set up an event listener on the map. + * map.on('sourcedata', function(e) { + * if (e.isSourceLoaded) { + * // Do something when the source has finished loading + * } + * }); + * ``` + */ +export type MapDataEvent = { + /** + * The event type. + */ + type: string; + /** + * The type of data that has changed. One of `'source'`, `'style'`. + */ + dataType: string; + /** + * Included if the event has a `dataType` of `source` and the event signals that internal data has been received or changed. Possible values are `metadata`, `content`, `visibility` and `idle`. + */ + sourceDataType: MapSourceDataType; +}; +/** + * The terrain event + * + * @group Event Related + */ +export type MapTerrainEvent = { + type: "terrain"; +}; +/** + * An event related to the web gl context + * + * @group Event Related + */ +export type MapContextEvent = { + type: "webglcontextlost" | "webglcontextrestored"; + originalEvent: WebGLContextEvent; +}; +/** + * The style image missing event + * + * @group Event Related + * + * @see [Generate and add a missing icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-missing-generated/) + */ +export type MapStyleImageMissingEvent = MapLibreEvent & { + type: "styleimagemissing"; + id: string; +}; +export declare class TransformProvider { + _map: Map; + constructor(map: Map); + get transform(): Transform; + get center(): { + lng: number; + lat: number; + }; + get zoom(): number; + get pitch(): number; + get bearing(): number; + unproject(point: PointLike): LngLat; +} +/** + * An options object sent to the enable function of some of the handlers + */ +export type AroundCenterOptions = { + /** + * If "center" is passed, map will zoom around the center of map + */ + around: "center"; +}; +declare abstract class TwoFingersTouchHandler implements Handler { + _enabled: boolean; + _active: boolean; + _firstTwoTouches: [ + number, + number + ]; + _vector: Point; + _startVector: Point; + _aroundCenter: boolean; + /** @internal */ + constructor(); + reset(): void; + abstract _start(points: [ + Point, + Point + ]): any; + abstract _move(points: [ + Point, + Point + ], pinchAround: Point, e: TouchEvent): any; + touchstart(e: TouchEvent, points: Array, mapTouches: Array): void; + touchmove(e: TouchEvent, points: Array, mapTouches: Array): any; + touchend(e: TouchEvent, points: Array, mapTouches: Array): void; + touchcancel(): void; + /** + * Enables the "drag to pitch" interaction. + * + * @example + * ```ts + * map.touchPitch.enable(); + * ``` + */ + enable(options?: AroundCenterOptions | boolean | null): void; + /** + * Disables the "drag to pitch" interaction. + * + * @example + * ```ts + * map.touchPitch.disable(); + * ``` + */ + disable(): void; + /** + * Returns a Boolean indicating whether the "drag to pitch" interaction is enabled. + * + * @returns `true` if the "drag to pitch" interaction is enabled. + */ + isEnabled(): boolean; + /** + * Returns a Boolean indicating whether the "drag to pitch" interaction is active, i.e. currently being used. + * + * @returns `true` if the "drag to pitch" interaction is active. + */ + isActive(): boolean; +} +export declare class TwoFingersTouchZoomHandler extends TwoFingersTouchHandler { + _distance: number; + _startDistance: number; + reset(): void; + _start(points: [ + Point, + Point + ]): void; + _move(points: [ + Point, + Point + ], pinchAround: Point): { + zoomDelta: number; + pinchAround: Point; + }; +} +export declare class TwoFingersTouchRotateHandler extends TwoFingersTouchHandler { + _minDiameter: number; + reset(): void; + _start(points: [ + Point, + Point + ]): void; + _move(points: [ + Point, + Point + ], pinchAround: Point): { + bearingDelta: number; + pinchAround: Point; + }; + _isBelowThreshold(vector: Point): boolean; +} +export declare class TwoFingersTouchPitchHandler extends TwoFingersTouchHandler { + _valid: boolean | void; + _firstMove: number; + _lastPoints: [ + Point, + Point + ]; + _map: Map; + _currentTouchCount: number; + constructor(map: Map); + reset(): void; + touchstart(e: TouchEvent, points: Array, mapTouches: Array): void; + _start(points: [ + Point, + Point + ]): void; + _move(points: [ + Point, + Point + ], center: Point, e: TouchEvent): { + pitchDelta: number; + }; + gestureBeginsVertically(vectorA: Point, vectorB: Point, timeStamp: number): boolean | void; +} +export declare class ScrollZoomHandler implements Handler { + _map: Map; + _tr: TransformProvider; + _el: HTMLElement; + _enabled: boolean; + _active: boolean; + _zooming: boolean; + _aroundCenter: boolean; + _around: LngLat; + _aroundPoint: Point; + _type: "wheel" | "trackpad" | null; + _lastValue: number; + _timeout: ReturnType; + _finishTimeout: ReturnType; + _lastWheelEvent: any; + _lastWheelEventTime: number; + _startZoom: number; + _targetZoom: number; + _delta: number; + _easing: ((a: number) => number); + _prevEase: { + start: number; + duration: number; + easing: (_: number) => number; + }; + _frameId: boolean; + _triggerRenderFrame: () => void; + _defaultZoomRate: number; + _wheelZoomRate: number; + /** @internal */ + constructor(map: Map, triggerRenderFrame: () => void); + /** + * Set the zoom rate of a trackpad + * @param zoomRate - 1/100 The rate used to scale trackpad movement to a zoom value. + * @example + * Speed up trackpad zoom + * ```ts + * map.scrollZoom.setZoomRate(1/25); + * ``` + */ + setZoomRate(zoomRate: number): void; + /** + * Set the zoom rate of a mouse wheel + * @param wheelZoomRate - 1/450 The rate used to scale mouse wheel movement to a zoom value. + * @example + * Slow down zoom of mouse wheel + * ```ts + * map.scrollZoom.setWheelZoomRate(1/600); + * ``` + */ + setWheelZoomRate(wheelZoomRate: number): void; + /** + * Returns a Boolean indicating whether the "scroll to zoom" interaction is enabled. + * @returns `true` if the "scroll to zoom" interaction is enabled. + */ + isEnabled(): boolean; + isActive(): boolean; + isZooming(): boolean; + /** + * Enables the "scroll to zoom" interaction. + * + * @param options - Options object. + * @example + * ```ts + * map.scrollZoom.enable(); + * map.scrollZoom.enable({ around: 'center' }) + * ``` + */ + enable(options?: AroundCenterOptions | boolean): void; + /** + * Disables the "scroll to zoom" interaction. + * + * @example + * ```ts + * map.scrollZoom.disable(); + * ``` + */ + disable(): void; + wheel(e: WheelEvent): void; + _onTimeout: (initialEvent: MouseEvent) => void; + _start(e: MouseEvent): void; + renderFrame(): { + noInertia: boolean; + needsRenderFrame: boolean; + zoomDelta: number; + around: Point; + originalEvent: any; + }; + _smoothOutEasing(duration: number): (t: number) => number; + reset(): void; +} +export declare class BoxZoomHandler implements Handler { + _map: Map; + _tr: TransformProvider; + _el: HTMLElement; + _container: HTMLElement; + _enabled: boolean; + _active: boolean; + _startPos: Point; + _lastPos: Point; + _box: HTMLElement; + _clickTolerance: number; + /** @internal */ + constructor(map: Map, options: { + clickTolerance: number; + }); + /** + * Returns a Boolean indicating whether the "box zoom" interaction is enabled. + * + * @returns `true` if the "box zoom" interaction is enabled. + */ + isEnabled(): boolean; + /** + * Returns a Boolean indicating whether the "box zoom" interaction is active, i.e. currently being used. + * + * @returns `true` if the "box zoom" interaction is active. + */ + isActive(): boolean; + /** + * Enables the "box zoom" interaction. + * + * @example + * ```ts + * map.boxZoom.enable(); + * ``` + */ + enable(): void; + /** + * Disables the "box zoom" interaction. + * + * @example + * ```ts + * map.boxZoom.disable(); + * ``` + */ + disable(): void; + mousedown(e: MouseEvent, point: Point): void; + mousemoveWindow(e: MouseEvent, point: Point): void; + mouseupWindow(e: MouseEvent, point: Point): { + cameraAnimation: (map: any) => any; + }; + keydown(e: KeyboardEvent): void; + reset(): void; + _fireEvent(type: string, e: any): Map; +} +export type DragRotateHandlerOptions = { + /** + * Control the map pitch in addition to the bearing + * @defaultValue true + */ + pitchWithRotate: boolean; +}; +export declare class DragRotateHandler { + _mouseRotate: MouseRotateHandler; + _mousePitch: MousePitchHandler; + _pitchWithRotate: boolean; + /** @internal */ + constructor(options: DragRotateHandlerOptions, mouseRotate: MouseRotateHandler, mousePitch: MousePitchHandler); + /** + * Enables the "drag to rotate" interaction. + * + * @example + * ```ts + * map.dragRotate.enable(); + * ``` + */ + enable(): void; + /** + * Disables the "drag to rotate" interaction. + * + * @example + * ```ts + * map.dragRotate.disable(); + * ``` + */ + disable(): void; + /** + * Returns a Boolean indicating whether the "drag to rotate" interaction is enabled. + * + * @returns `true` if the "drag to rotate" interaction is enabled. + */ + isEnabled(): boolean; + /** + * Returns a Boolean indicating whether the "drag to rotate" interaction is active, i.e. currently being used. + * + * @returns `true` if the "drag to rotate" interaction is active. + */ + isActive(): boolean; +} +export declare class KeyboardHandler implements Handler { + _tr: TransformProvider; + _enabled: boolean; + _active: boolean; + _panStep: number; + _bearingStep: number; + _pitchStep: number; + _rotationDisabled: boolean; + /** @internal */ + constructor(map: Map); + reset(): void; + keydown(e: KeyboardEvent): { + cameraAnimation: (map: Map) => void; + }; + /** + * Enables the "keyboard rotate and zoom" interaction. + * + * @example + * ```ts + * map.keyboard.enable(); + * ``` + */ + enable(): void; + /** + * Disables the "keyboard rotate and zoom" interaction. + * + * @example + * ```ts + * map.keyboard.disable(); + * ``` + */ + disable(): void; + /** + * Returns a Boolean indicating whether the "keyboard rotate and zoom" + * interaction is enabled. + * + * @returns `true` if the "keyboard rotate and zoom" + * interaction is enabled. + */ + isEnabled(): boolean; + /** + * Returns true if the handler is enabled and has detected the start of a + * zoom/rotate gesture. + * + * @returns `true` if the handler is enabled and has detected the + * start of a zoom/rotate gesture. + */ + isActive(): boolean; + /** + * Disables the "keyboard pan/rotate" interaction, leaving the + * "keyboard zoom" interaction enabled. + * + * @example + * ```ts + * map.keyboard.disableRotation(); + * ``` + */ + disableRotation(): void; + /** + * Enables the "keyboard pan/rotate" interaction. + * + * @example + * ```ts + * map.keyboard.enable(); + * map.keyboard.enableRotation(); + * ``` + */ + enableRotation(): void; +} +export declare class ClickZoomHandler implements Handler { + _tr: TransformProvider; + _enabled: boolean; + _active: boolean; + /** @internal */ + constructor(map: Map); + reset(): void; + dblclick(e: MouseEvent, point: Point): { + cameraAnimation: (map: Map) => void; + }; + enable(): void; + disable(): void; + isEnabled(): boolean; + isActive(): boolean; +} +export declare class SingleTapRecognizer { + numTouches: number; + centroid: Point; + startTime: number; + aborted: boolean; + touches: { + [k in number | string]: Point; + }; + constructor(options: { + numTouches: number; + }); + reset(): void; + touchstart(e: TouchEvent, points: Array, mapTouches: Array): void; + touchmove(e: TouchEvent, points: Array, mapTouches: Array): void; + touchend(e: TouchEvent, points: Array, mapTouches: Array): Point; +} +export declare class TapRecognizer { + singleTap: SingleTapRecognizer; + numTaps: number; + lastTime: number; + lastTap: Point; + count: number; + constructor(options: { + numTaps: number; + numTouches: number; + }); + reset(): void; + touchstart(e: TouchEvent, points: Array, mapTouches: Array): void; + touchmove(e: TouchEvent, points: Array, mapTouches: Array): void; + touchend(e: TouchEvent, points: Array, mapTouches: Array): Point; +} +export declare class TapZoomHandler implements Handler { + _tr: TransformProvider; + _enabled: boolean; + _active: boolean; + _zoomIn: TapRecognizer; + _zoomOut: TapRecognizer; + constructor(map: Map); + reset(): void; + touchstart(e: TouchEvent, points: Array, mapTouches: Array): void; + touchmove(e: TouchEvent, points: Array, mapTouches: Array): void; + touchend(e: TouchEvent, points: Array, mapTouches: Array): { + cameraAnimation: (map: Map) => Map; + }; + touchcancel(): void; + enable(): void; + disable(): void; + isEnabled(): boolean; + isActive(): boolean; +} +export declare class DoubleClickZoomHandler { + _clickZoom: ClickZoomHandler; + _tapZoom: TapZoomHandler; + /** @internal */ + constructor(clickZoom: ClickZoomHandler, TapZoom: TapZoomHandler); + /** + * Enables the "double click to zoom" interaction. + * + * @example + * ```ts + * map.doubleClickZoom.enable(); + * ``` + */ + enable(): void; + /** + * Disables the "double click to zoom" interaction. + * + * @example + * ```ts + * map.doubleClickZoom.disable(); + * ``` + */ + disable(): void; + /** + * Returns a Boolean indicating whether the "double click to zoom" interaction is enabled. + * + * @returns `true` if the "double click to zoom" interaction is enabled. + */ + isEnabled(): boolean; + /** + * Returns a Boolean indicating whether the "double click to zoom" interaction is active, i.e. currently being used. + * + * @returns `true` if the "double click to zoom" interaction is active. + */ + isActive(): boolean; +} +export declare class TapDragZoomHandler implements Handler { + _enabled: boolean; + _active: boolean; + _swipePoint: Point; + _swipeTouch: number; + _tapTime: number; + _tapPoint: Point; + _tap: TapRecognizer; + constructor(); + reset(): void; + touchstart(e: TouchEvent, points: Array, mapTouches: Array): void; + touchmove(e: TouchEvent, points: Array, mapTouches: Array): { + zoomDelta: number; + }; + touchend(e: TouchEvent, points: Array, mapTouches: Array): void; + touchcancel(): void; + enable(): void; + disable(): void; + isEnabled(): boolean; + isActive(): boolean; +} +export declare class TwoFingersTouchZoomRotateHandler { + _el: HTMLElement; + _touchZoom: TwoFingersTouchZoomHandler; + _touchRotate: TwoFingersTouchRotateHandler; + _tapDragZoom: TapDragZoomHandler; + _rotationDisabled: boolean; + _enabled: boolean; + /** @internal */ + constructor(el: HTMLElement, touchZoom: TwoFingersTouchZoomHandler, touchRotate: TwoFingersTouchRotateHandler, tapDragZoom: TapDragZoomHandler); + /** + * Enables the "pinch to rotate and zoom" interaction. + * + * @param options - Options object. + * + * @example + * ```ts + * map.touchZoomRotate.enable(); + * map.touchZoomRotate.enable({ around: 'center' }); + * ``` + */ + enable(options?: AroundCenterOptions | boolean | null): void; + /** + * Disables the "pinch to rotate and zoom" interaction. + * + * @example + * ```ts + * map.touchZoomRotate.disable(); + * ``` + */ + disable(): void; + /** + * Returns a Boolean indicating whether the "pinch to rotate and zoom" interaction is enabled. + * + * @returns `true` if the "pinch to rotate and zoom" interaction is enabled. + */ + isEnabled(): boolean; + /** + * Returns true if the handler is enabled and has detected the start of a zoom/rotate gesture. + * + * @returns `true` if the handler is active, `false` otherwise + */ + isActive(): boolean; + /** + * Disables the "pinch to rotate" interaction, leaving the "pinch to zoom" + * interaction enabled. + * + * @example + * ```ts + * map.touchZoomRotate.disableRotation(); + * ``` + */ + disableRotation(): void; + /** + * Enables the "pinch to rotate" interaction. + * + * @example + * ```ts + * map.touchZoomRotate.enable(); + * map.touchZoomRotate.enableRotation(); + * ``` + */ + enableRotation(): void; +} +/** + * A position defintion for the control to be placed, can be in one of the corners of the map. + * When two or more controls are places in the same location they are stacked toward the center of the map. + */ +export type ControlPosition = "top-left" | "top-right" | "bottom-left" | "bottom-right"; +/** + * Interface for interactive controls added to the map. This is a + * specification for implementers to model: it is not + * an exported method or class. + * + * Controls must implement `onAdd` and `onRemove`, and must own an + * element, which is often a `div` element. To use MapLibre GL JS's + * default control styling, add the `maplibregl-ctrl` class to your control's + * node. + * + * @example + * Control implemented as ES6 class + * ```ts + * class HelloWorldControl { + * onAdd(map) { + * this._map = map; + * this._container = document.createElement('div'); + * this._container.className = 'maplibregl-ctrl'; + * this._container.textContent = 'Hello, world'; + * return this._container; + * } + * + * onRemove() { + * this._container.parentNode.removeChild(this._container); + * this._map = undefined; + * } + * } + * + * // Control implemented as ES5 prototypical class + * function HelloWorldControl() { } + * + * HelloWorldControl.prototype.onAdd = function(map) { + * this._map = map; + * this._container = document.createElement('div'); + * this._container.className = 'maplibregl-ctrl'; + * this._container.textContent = 'Hello, world'; + * return this._container; + * }; + * + * HelloWorldControl.prototype.onRemove = function () { + * this._container.parentNode.removeChild(this._container); + * this._map = undefined; + * }; + * ``` + */ +export interface IControl { + /** + * Register a control on the map and give it a chance to register event listeners + * and resources. This method is called by {@link Map#addControl} + * internally. + * + * @param map - the Map this control will be added to + * @returns The control's container element. This should + * be created by the control and returned by onAdd without being attached + * to the DOM: the map will insert the control's element into the DOM + * as necessary. + */ + onAdd(map: Map): HTMLElement; + /** + * Unregister a control on the map and give it a chance to detach event listeners + * and resources. This method is called by {@link Map#removeControl} + * internally. + * + * @param map - the Map this control will be removed from + */ + onRemove(map: Map): void; + /** + * Optionally provide a default position for this control. If this method + * is implemented and {@link Map#addControl} is called without the `position` + * parameter, the value returned by getDefaultPosition will be used as the + * control's position. + * + * @returns a control position, one of the values valid in addControl. + */ + readonly getDefaultPosition?: () => ControlPosition; +} +/** + * The {@link Map} options object. + */ +export type MapOptions = { + /** + * If `true`, the map's position (zoom, center latitude, center longitude, bearing, and pitch) will be synced with the hash fragment of the page's URL. + * For example, `http://path/to/my/page.html#2.59/39.26/53.07/-24.1/60`. + * An additional string may optionally be provided to indicate a parameter-styled hash, + * e.g. http://path/to/my/page.html#map=2.59/39.26/53.07/-24.1/60&foo=bar, where foo + * is a custom parameter and bar is an arbitrary hash distinct from the map hash. + * @defaultValue false + */ + hash?: boolean | string; + /** + * If `false`, no mouse, touch, or keyboard listeners will be attached to the map, so it will not respond to interaction. + * @defaultValue true + */ + interactive?: boolean; + /** + * The HTML element in which MapLibre GL JS will render the map, or the element's string `id`. The specified element must have no children. + */ + container: HTMLElement | string; + /** + * The threshold, measured in degrees, that determines when the map's + * bearing will snap to north. For example, with a `bearingSnap` of 7, if the user rotates + * the map within 7 degrees of north, the map will automatically snap to exact north. + * @defaultValue 7 + */ + bearingSnap?: number; + /** + * If `true`, an {@link AttributionControl} will be added to the map. + * @defaultValue true + */ + attributionControl?: boolean; + /** + * Attribution text to show in an {@link AttributionControl}. Only applicable if `options.attributionControl` is `true`. + */ + customAttribution?: string | Array; + /** + * If `true`, the MapLibre logo will be shown. + * @defaultValue false + */ + maplibreLogo?: boolean; + /** + * A string representing the position of the MapLibre wordmark on the map. Valid options are `top-left`,`top-right`, `bottom-left`, or `bottom-right`. + * @defaultValue 'bottom-left' + */ + logoPosition?: ControlPosition; + /** + * If `true`, map creation will fail if the performance of MapLibre GL JS would be dramatically worse than expected + * (i.e. a software renderer would be used). + * @defaultValue false + */ + failIfMajorPerformanceCaveat?: boolean; + /** + * If `true`, the map's canvas can be exported to a PNG using `map.getCanvas().toDataURL()`. This is `false` by default as a performance optimization. + * @defaultValue false + */ + preserveDrawingBuffer?: boolean; + /** + * If `true`, the gl context will be created with MSAA antialiasing, which can be useful for antialiasing custom layers. This is `false` by default as a performance optimization. + */ + antialias?: boolean; + /** + * If `false`, the map won't attempt to re-request tiles once they expire per their HTTP `cacheControl`/`expires` headers. + * @defaultValue true + */ + refreshExpiredTiles?: boolean; + /** + * If set, the map will be constrained to the given bounds. + */ + maxBounds?: LngLatBoundsLike; + /** + * If `true`, the "scroll to zoom" interaction is enabled. {@link AroundCenterOptions} are passed as options to {@link ScrollZoomHandler#enable}. + * @defaultValue true + */ + scrollZoom?: boolean | AroundCenterOptions; + /** + * The minimum zoom level of the map (0-24). + * @defaultValue 0 + */ + minZoom?: number | null; + /** + * The maximum zoom level of the map (0-24). + * @defaultValue 22 + */ + maxZoom?: number | null; + /** + * The minimum pitch of the map (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project. + * @defaultValue 0 + */ + minPitch?: number | null; + /** + * The maximum pitch of the map (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project. + * @defaultValue 60 + */ + maxPitch?: number | null; + /** + * If `true`, the "box zoom" interaction is enabled (see {@link BoxZoomHandler}). + * @defaultValue true + */ + boxZoom?: boolean; + /** + * If `true`, the "drag to rotate" interaction is enabled (see {@link DragRotateHandler}). + * @defaultValue true + */ + dragRotate?: boolean; + /** + * If `true`, the "drag to pan" interaction is enabled. An `Object` value is passed as options to {@link DragPanHandler#enable}. + * @defaultValue true + */ + dragPan?: boolean | DragPanOptions; + /** + * If `true`, keyboard shortcuts are enabled (see {@link KeyboardHandler}). + * @defaultValue true + */ + keyboard?: boolean; + /** + * If `true`, the "double click to zoom" interaction is enabled (see {@link DoubleClickZoomHandler}). + * @defaultValue true + */ + doubleClickZoom?: boolean; + /** + * If `true`, the "pinch to rotate and zoom" interaction is enabled. An `Object` value is passed as options to {@link TwoFingersTouchZoomRotateHandler#enable}. + * @defaultValue true + */ + touchZoomRotate?: boolean | AroundCenterOptions; + /** + * If `true`, the "drag to pitch" interaction is enabled. An `Object` value is passed as options to {@link TwoFingersTouchPitchHandler#enable}. + * @defaultValue true + */ + touchPitch?: boolean | AroundCenterOptions; + /** + * If `true` or set to an options object, the map is only accessible on desktop while holding Command/Ctrl and only accessible on mobile with two fingers. Interacting with the map using normal gestures will trigger an informational screen. With this option enabled, "drag to pitch" requires a three-finger gesture. Cooperative gestures are disabled when a map enters fullscreen using {@link FullscreenControl}. + * @defaultValue undefined + */ + cooperativeGestures?: boolean | GestureOptions; + /** + * If `true`, the map will automatically resize when the browser window resizes. + * @defaultValue true + */ + trackResize?: boolean; + /** + * The initial geographical centerpoint of the map. If `center` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `[0, 0]` Note: MapLibre GL JS uses longitude, latitude coordinate order (as opposed to latitude, longitude) to match GeoJSON. + * @defaultValue [0, 0] + */ + center?: LngLatLike; + /** + * The initial zoom level of the map. If `zoom` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`. + * @defaultValue 0 + */ + zoom?: number; + /** + * The initial bearing (rotation) of the map, measured in degrees counter-clockwise from north. If `bearing` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`. + * @defaultValue 0 + */ + bearing?: number; + /** + * The initial pitch (tilt) of the map, measured in degrees away from the plane of the screen (0-85). If `pitch` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`. Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project. + * @defaultValue 0 + */ + pitch?: number; + /** + * If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`: + * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire + * container, there will be blank space beyond 180 and -180 degrees longitude. + * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the + * map and the other on the left edge of the map) at every zoom level. + * @defaultValue true + */ + renderWorldCopies?: boolean; + /** + * The maximum number of tiles stored in the tile cache for a given source. If omitted, the cache will be dynamically sized based on the current viewport which can be set using `maxTileCacheZoomLevels` constructor options. + * @defaultValue null + */ + maxTileCacheSize?: number; + /** + * The maximum number of zoom levels for which to store tiles for a given source. Tile cache dynamic size is calculated by multiplying `maxTileCacheZoomLevels` with the approximate number of tiles in the viewport for a given source. + * @defaultValue 5 + */ + maxTileCacheZoomLevels?: number; + /** + * A callback run before the Map makes a request for an external URL. The callback can be used to modify the url, set headers, or set the credentials property for cross-origin requests. + * Expected to return an object with a `url` property and optionally `headers` and `credentials` properties. + */ + transformRequest?: RequestTransformFunction; + /** + * A callback run before the map's camera is moved due to user input or animation. The callback can be used to modify the new center, zoom, pitch and bearing. + * Expected to return an object containing center, zoom, pitch or bearing values to overwrite. + */ + transformCameraUpdate?: CameraUpdateTransformFunction; + /** + * A patch to apply to the default localization table for UI strings, e.g. control tooltips. The `locale` object maps namespaced UI string IDs to translated strings in the target language; see `src/ui/default_locale.js` for an example with all supported string IDs. The object may specify all UI strings (thereby adding support for a new translation) or only a subset of strings (thereby patching the default translation table). + * @defaultValue null + */ + locale?: any; + /** + * Controls the duration of the fade-in/fade-out animation for label collisions after initial map load, in milliseconds. This setting affects all symbol layers. This setting does not affect the duration of runtime styling transitions or raster tile cross-fading. + * @defaultValue 300 + */ + fadeDuration?: number; + /** + * If `true`, symbols from multiple sources can collide with each other during collision detection. If `false`, collision detection is run separately for the symbols in each source. + * @defaultValue true + */ + crossSourceCollisions?: boolean; + /** + * If `true`, Resource Timing API information will be collected for requests made by GeoJSON and Vector Tile web workers (this information is normally inaccessible from the main Javascript thread). Information will be returned in a `resourceTiming` property of relevant `data` events. + * @defaultValue false + */ + collectResourceTiming?: boolean; + /** + * The max number of pixels a user can shift the mouse pointer during a click for it to be considered a valid click (as opposed to a mouse drag). + * @defaultValue true + */ + clickTolerance?: number; + /** + * The initial bounds of the map. If `bounds` is specified, it overrides `center` and `zoom` constructor options. + */ + bounds?: LngLatBoundsLike; + /** + * A {@link FitBoundsOptions} options object to use _only_ when fitting the initial `bounds` provided above. + */ + fitBoundsOptions?: FitBoundsOptions; + /** + * Defines a CSS + * font-family for locally overriding generation of glyphs in the 'CJK Unified Ideographs', 'Hiragana', 'Katakana' and 'Hangul Syllables' ranges. + * In these ranges, font settings from the map's style will be ignored, except for font-weight keywords (light/regular/medium/bold). + * Set to `false`, to enable font settings from the map's style for these glyph ranges. + * The purpose of this option is to avoid bandwidth-intensive glyph server requests. (See [Use locally generated ideographs](https://maplibre.org/maplibre-gl-js/docs/examples/local-ideographs).) + * @defaultValue 'sans-serif' + */ + localIdeographFontFamily?: string; + /** + * The map's MapLibre style. This must be a JSON object conforming to + * the schema described in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/), + * or a URL to such JSON. + */ + style: StyleSpecification | string; + /** + * If `false`, the map's pitch (tilt) control with "drag to rotate" interaction will be disabled. + * @defaultValue true + */ + pitchWithRotate?: boolean; + /** + * The pixel ratio. The canvas' `width` attribute will be `container.clientWidth * pixelRatio` and its `height` attribute will be `container.clientHeight * pixelRatio`. Defaults to `devicePixelRatio` if not specified. + */ + pixelRatio?: number; + /** + * If false, style validation will be skipped. Useful in production environment. + * @defaultValue true + */ + validateStyle?: boolean; + /** + * The canvas' `width` and `height` max size. The values are passed as an array where the first element is max width and the second element is max height. + * You shouldn't set this above WebGl `MAX_TEXTURE_SIZE`. Defaults to [4096, 4096]. + */ + maxCanvasSize?: [ + number, + number + ]; +}; +/** + * An options object for the gesture settings + * @example + * ```ts + * let options = { + * windowsHelpText: "Use Ctrl + scroll to zoom the map", + * macHelpText: "Use ⌘ + scroll to zoom the map", + * mobileHelpText: "Use two fingers to move the map", + * } + * ``` + */ +export type GestureOptions = { + windowsHelpText?: string; + macHelpText?: string; + mobileHelpText?: string; +}; +export type Complete = { + [P in keyof Required]: Pick extends Required> ? T[P] : (T[P] | undefined); +}; +export type CompleteMapOptions = Complete; +export declare class Map extends Camera { + style: Style; + painter: Painter; + handlers: HandlerManager; + _container: HTMLElement; + _canvasContainer: HTMLElement; + _controlContainer: HTMLElement; + _controlPositions: { + [_: string]: HTMLElement; + }; + _interactive: boolean; + _cooperativeGestures: boolean | GestureOptions; + _cooperativeGesturesScreen: HTMLElement; + _metaKey: keyof MouseEvent; + _showTileBoundaries: boolean; + _showCollisionBoxes: boolean; + _showPadding: boolean; + _showOverdrawInspector: boolean; + _repaint: boolean; + _vertices: boolean; + _canvas: HTMLCanvasElement; + _maxTileCacheSize: number; + _maxTileCacheZoomLevels: number; + _frame: Cancelable; + _styleDirty: boolean; + _sourcesDirty: boolean; + _placementDirty: boolean; + _loaded: boolean; + _idleTriggered: boolean; + _fullyLoaded: boolean; + _trackResize: boolean; + _resizeObserver: ResizeObserver; + _preserveDrawingBuffer: boolean; + _failIfMajorPerformanceCaveat: boolean; + _antialias: boolean; + _refreshExpiredTiles: boolean; + _hash: Hash; + _delegatedListeners: any; + _fadeDuration: number; + _crossSourceCollisions: boolean; + _crossFadingFactor: number; + _collectResourceTiming: boolean; + _renderTaskQueue: TaskQueue; + _controls: Array; + _mapId: number; + _localIdeographFontFamily: string; + _validateStyle: boolean; + _requestManager: RequestManager; + _locale: any; + _removed: boolean; + _clickTolerance: number; + _overridePixelRatio: number | null; + _maxCanvasSize: [ + number, + number + ]; + _terrainDataCallback: (e: MapStyleDataEvent | MapSourceDataEvent) => void; + /** + * @internal + * image queue throttling handle. To be used later when clean up + */ + _imageQueueHandle: number; + /** + * The map's {@link ScrollZoomHandler}, which implements zooming in and out with a scroll wheel or trackpad. + * Find more details and examples using `scrollZoom` in the {@link ScrollZoomHandler} section. + */ + scrollZoom: ScrollZoomHandler; + /** + * The map's {@link BoxZoomHandler}, which implements zooming using a drag gesture with the Shift key pressed. + * Find more details and examples using `boxZoom` in the {@link BoxZoomHandler} section. + */ + boxZoom: BoxZoomHandler; + /** + * The map's {@link DragRotateHandler}, which implements rotating the map while dragging with the right + * mouse button or with the Control key pressed. Find more details and examples using `dragRotate` + * in the {@link DragRotateHandler} section. + */ + dragRotate: DragRotateHandler; + /** + * The map's {@link DragPanHandler}, which implements dragging the map with a mouse or touch gesture. + * Find more details and examples using `dragPan` in the {@link DragPanHandler} section. + */ + dragPan: DragPanHandler; + /** + * The map's {@link KeyboardHandler}, which allows the user to zoom, rotate, and pan the map using keyboard + * shortcuts. Find more details and examples using `keyboard` in the {@link KeyboardHandler} section. + */ + keyboard: KeyboardHandler; + /** + * The map's {@link DoubleClickZoomHandler}, which allows the user to zoom by double clicking. + * Find more details and examples using `doubleClickZoom` in the {@link DoubleClickZoomHandler} section. + */ + doubleClickZoom: DoubleClickZoomHandler; + /** + * The map's {@link TwoFingersTouchZoomRotateHandler}, which allows the user to zoom or rotate the map with touch gestures. + * Find more details and examples using `touchZoomRotate` in the {@link TwoFingersTouchZoomRotateHandler} section. + */ + touchZoomRotate: TwoFingersTouchZoomRotateHandler; + /** + * The map's {@link TwoFingersTouchPitchHandler}, which allows the user to pitch the map with touch gestures. + * Find more details and examples using `touchPitch` in the {@link TwoFingersTouchPitchHandler} section. + */ + touchPitch: TwoFingersTouchPitchHandler; + constructor(options: MapOptions); + /** + * @internal + * Returns a unique number for this map instance which is used for the MapLoadEvent + * to make sure we only fire one event per instantiated map object. + * @returns the uniq map ID + */ + _getMapId(): number; + /** + * Adds an {@link IControl} to the map, calling `control.onAdd(this)`. + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param control - The {@link IControl} to add. + * @param position - position on the map to which the control will be added. + * Valid values are `'top-left'`, `'top-right'`, `'bottom-left'`, and `'bottom-right'`. Defaults to `'top-right'`. + * @returns `this` + * @example + * Add zoom and rotation controls to the map. + * ```ts + * map.addControl(new maplibregl.NavigationControl()); + * ``` + * @see [Display map navigation controls](https://maplibre.org/maplibre-gl-js/docs/examples/navigation/) + */ + addControl(control: IControl, position?: ControlPosition): Map; + /** + * Removes the control from the map. + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param control - The {@link IControl} to remove. + * @returns `this` + * @example + * ```ts + * // Define a new navigation control. + * let navigation = new maplibregl.NavigationControl(); + * // Add zoom and rotation controls to the map. + * map.addControl(navigation); + * // Remove zoom and rotation controls from the map. + * map.removeControl(navigation); + * ``` + */ + removeControl(control: IControl): Map; + /** + * Checks if a control exists on the map. + * + * @param control - The {@link IControl} to check. + * @returns true if map contains control. + * @example + * ```ts + * // Define a new navigation control. + * let navigation = new maplibregl.NavigationControl(); + * // Add zoom and rotation controls to the map. + * map.addControl(navigation); + * // Check that the navigation control exists on the map. + * map.hasControl(navigation); + * ``` + */ + hasControl(control: IControl): boolean; + calculateCameraOptionsFromTo(from: LngLat, altitudeFrom: number, to: LngLat, altitudeTo?: number): CameraOptions; + /** + * Resizes the map according to the dimensions of its + * `container` element. + * + * Checks if the map container size changed and updates the map if it has changed. + * This method must be called after the map's `container` is resized programmatically + * or when the map is shown after being initially hidden with CSS. + * + * Triggers the following events: `movestart`, `move`, `moveend`, and `resize`. + * + * @param eventData - Additional properties to be passed to `movestart`, `move`, `resize`, and `moveend` + * events that get triggered as a result of resize. This can be useful for differentiating the + * source of an event (for example, user-initiated or programmatically-triggered events). + * @returns `this` + * @example + * Resize the map when the map container is shown after being initially hidden with CSS. + * ```ts + * let mapDiv = document.getElementById('map'); + * if (mapDiv.style.visibility === true) map.resize(); + * ``` + */ + resize(eventData?: any): Map; + /** + * @internal + * Return the map's pixel ratio eventually scaled down to respect maxCanvasSize. + * Internally you should use this and not getPixelRatio(). + */ + _getClampedPixelRatio(width: number, height: number): number; + /** + * Returns the map's pixel ratio. + * Note that the pixel ratio actually applied may be lower to respect maxCanvasSize. + * @returns The pixel ratio. + */ + getPixelRatio(): number; + /** + * Sets the map's pixel ratio. This allows to override `devicePixelRatio`. + * After this call, the canvas' `width` attribute will be `container.clientWidth * pixelRatio` + * and its height attribute will be `container.clientHeight * pixelRatio`. + * Set this to null to disable `devicePixelRatio` override. + * Note that the pixel ratio actually applied may be lower to respect maxCanvasSize. + * @param pixelRatio - The pixel ratio. + */ + setPixelRatio(pixelRatio: number): void; + /** + * Returns the map's geographical bounds. When the bearing or pitch is non-zero, the visible region is not + * an axis-aligned rectangle, and the result is the smallest bounds that encompasses the visible region. + * @returns The geographical bounds of the map as {@link LngLatBounds}. + * @example + * ```ts + * let bounds = map.getBounds(); + * ``` + */ + getBounds(): LngLatBounds; + /** + * Returns the maximum geographical bounds the map is constrained to, or `null` if none set. + * @returns The map object. + * @example + * ```ts + * let maxBounds = map.getMaxBounds(); + * ``` + */ + getMaxBounds(): LngLatBounds | null; + /** + * Sets or clears the map's geographical bounds. + * + * Pan and zoom operations are constrained within these bounds. + * If a pan or zoom is performed that would + * display regions outside these bounds, the map will + * instead display a position and zoom level + * as close as possible to the operation's request while still + * remaining within the bounds. + * + * @param bounds - The maximum bounds to set. If `null` or `undefined` is provided, the function removes the map's maximum bounds. + * @returns `this` + * @example + * Define bounds that conform to the `LngLatBoundsLike` object as set the max bounds. + * ```ts + * let bounds = [ + * [-74.04728, 40.68392], // [west, south] + * [-73.91058, 40.87764] // [east, north] + * ]; + * map.setMaxBounds(bounds); + * ``` + */ + setMaxBounds(bounds?: LngLatBoundsLike | null): Map; + /** + * Sets or clears the map's minimum zoom level. + * If the map's current zoom level is lower than the new minimum, + * the map will zoom to the new minimum. + * + * It is not always possible to zoom out and reach the set `minZoom`. + * Other factors such as map height may restrict zooming. For example, + * if the map is 512px tall it will not be possible to zoom below zoom 0 + * no matter what the `minZoom` is set to. + * + * A {@link ErrorEvent} event will be fired if minZoom is out of bounds. + * + * @param minZoom - The minimum zoom level to set (-2 - 24). + * If `null` or `undefined` is provided, the function removes the current minimum zoom (i.e. sets it to -2). + * @returns `this` + * @example + * ```ts + * map.setMinZoom(12.25); + * ``` + */ + setMinZoom(minZoom?: number | null): Map; + /** + * Returns the map's minimum allowable zoom level. + * + * @returns minZoom + * @example + * ```ts + * let minZoom = map.getMinZoom(); + * ``` + */ + getMinZoom(): number; + /** + * Sets or clears the map's maximum zoom level. + * If the map's current zoom level is higher than the new maximum, + * the map will zoom to the new maximum. + * + * A {@link ErrorEvent} event will be fired if minZoom is out of bounds. + * + * @param maxZoom - The maximum zoom level to set. + * If `null` or `undefined` is provided, the function removes the current maximum zoom (sets it to 22). + * @returns `this` + * @example + * ```ts + * map.setMaxZoom(18.75); + * ``` + */ + setMaxZoom(maxZoom?: number | null): Map; + /** + * Returns the map's maximum allowable zoom level. + * + * @returns The maxZoom + * @example + * ```ts + * let maxZoom = map.getMaxZoom(); + * ``` + */ + getMaxZoom(): number; + /** + * Sets or clears the map's minimum pitch. + * If the map's current pitch is lower than the new minimum, + * the map will pitch to the new minimum. + * + * A {@link ErrorEvent} event will be fired if minPitch is out of bounds. + * + * @param minPitch - The minimum pitch to set (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project. + * If `null` or `undefined` is provided, the function removes the current minimum pitch (i.e. sets it to 0). + * @returns `this` + */ + setMinPitch(minPitch?: number | null): Map; + /** + * Returns the map's minimum allowable pitch. + * + * @returns The minPitch + */ + getMinPitch(): number; + /** + * Sets or clears the map's maximum pitch. + * If the map's current pitch is higher than the new maximum, + * the map will pitch to the new maximum. + * + * A {@link ErrorEvent} event will be fired if maxPitch is out of bounds. + * + * @param maxPitch - The maximum pitch to set (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project. + * If `null` or `undefined` is provided, the function removes the current maximum pitch (sets it to 60). + * @returns `this` + */ + setMaxPitch(maxPitch?: number | null): Map; + /** + * Returns the map's maximum allowable pitch. + * + * @returns The maxPitch + */ + getMaxPitch(): number; + /** + * Returns the state of `renderWorldCopies`. If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`: + * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire + * container, there will be blank space beyond 180 and -180 degrees longitude. + * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the + * map and the other on the left edge of the map) at every zoom level. + * @returns The renderWorldCopies + * @example + * ```ts + * let worldCopiesRendered = map.getRenderWorldCopies(); + * ``` + * @see [Render world copies](https://maplibre.org/maplibre-gl-js/docs/examples/render-world-copies/) + */ + getRenderWorldCopies(): boolean; + /** + * Sets the state of `renderWorldCopies`. + * + * @param renderWorldCopies - If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`: + * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire + * container, there will be blank space beyond 180 and -180 degrees longitude. + * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the + * map and the other on the left edge of the map) at every zoom level. + * + * `undefined` is treated as `true`, `null` is treated as `false`. + * @returns `this` + * @example + * ```ts + * map.setRenderWorldCopies(true); + * ``` + * @see [Render world copies](https://maplibre.org/maplibre-gl-js/docs/examples/render-world-copies/) + */ + setRenderWorldCopies(renderWorldCopies?: boolean | null): Map; + /** + * Gets the map's cooperativeGestures option + * + * @returns The gestureOptions + */ + getCooperativeGestures(): boolean | GestureOptions; + /** + * Sets or clears the map's cooperativeGestures option + * + * @param gestureOptions - If `true` or set to an options object, map is only accessible on desktop while holding Command/Ctrl and only accessible on mobile with two fingers. Interacting with the map using normal gestures will trigger an informational screen. With this option enabled, "drag to pitch" requires a three-finger gesture. + * @returns `this` + */ + setCooperativeGestures(gestureOptions?: GestureOptions | boolean | null): Map; + /** + * Returns a [Point](https://github.com/mapbox/point-geometry) representing pixel coordinates, relative to the map's `container`, + * that correspond to the specified geographical location. + * + * @param lnglat - The geographical location to project. + * @returns The [Point](https://github.com/mapbox/point-geometry) corresponding to `lnglat`, relative to the map's `container`. + * @example + * ```ts + * let coordinate = [-122.420679, 37.772537]; + * let point = map.project(coordinate); + * ``` + */ + project(lnglat: LngLatLike): Point; + /** + * Returns a {@link LngLat} representing geographical coordinates that correspond + * to the specified pixel coordinates. + * + * @param point - The pixel coordinates to unproject. + * @returns The {@link LngLat} corresponding to `point`. + * @example + * ```ts + * map.on('click', function(e) { + * // When the map is clicked, get the geographic coordinate. + * let coordinate = map.unproject(e.point); + * }); + * ``` + */ + unproject(point: PointLike): LngLat; + /** + * Returns true if the map is panning, zooming, rotating, or pitching due to a camera animation or user gesture. + * @returns true if the map is moving. + * @example + * ```ts + * let isMoving = map.isMoving(); + * ``` + */ + isMoving(): boolean; + /** + * Returns true if the map is zooming due to a camera animation or user gesture. + * @returns true if the map is zooming. + * @example + * ```ts + * let isZooming = map.isZooming(); + * ``` + */ + isZooming(): boolean; + /** + * Returns true if the map is rotating due to a camera animation or user gesture. + * @returns true if the map is rotating. + * @example + * ```ts + * map.isRotating(); + * ``` + */ + isRotating(): boolean; + _createDelegatedListener(type: keyof MapEventType | string, layerId: string, listener: Listener): { + layer: string; + listener: Listener; + delegates: { + [type in keyof MapEventType]?: (e: any) => void; + }; + }; + /** + * @event + * Adds a listener for events of a specified type, optionally limited to features in a specified style layer. + * See {@link MapEventType} and {@link MapLayerEventType} for a full list of events and their description. + * + * | Event | Compatible with `layerId` | + * |------------------------|---------------------------| + * | `mousedown` | yes | + * | `mouseup` | yes | + * | `mouseover` | yes | + * | `mouseout` | yes | + * | `mousemove` | yes | + * | `mouseenter` | yes (required) | + * | `mouseleave` | yes (required) | + * | `click` | yes | + * | `dblclick` | yes | + * | `contextmenu` | yes | + * | `touchstart` | yes | + * | `touchend` | yes | + * | `touchcancel` | yes | + * | `wheel` | | + * | `resize` | | + * | `remove` | | + * | `touchmove` | | + * | `movestart` | | + * | `move` | | + * | `moveend` | | + * | `dragstart` | | + * | `drag` | | + * | `dragend` | | + * | `zoomstart` | | + * | `zoom` | | + * | `zoomend` | | + * | `rotatestart` | | + * | `rotate` | | + * | `rotateend` | | + * | `pitchstart` | | + * | `pitch` | | + * | `pitchend` | | + * | `boxzoomstart` | | + * | `boxzoomend` | | + * | `boxzoomcancel` | | + * | `webglcontextlost` | | + * | `webglcontextrestored` | | + * | `load` | | + * | `render` | | + * | `idle` | | + * | `error` | | + * | `data` | | + * | `styledata` | | + * | `sourcedata` | | + * | `dataloading` | | + * | `styledataloading` | | + * | `sourcedataloading` | | + * | `styleimagemissing` | | + * | `dataabort` | | + * | `sourcedataabort` | | + * + * @param type - The event type to listen for. Events compatible with the optional `layerId` parameter are triggered + * when the cursor enters a visible portion of the specified layer from outside that layer or outside the map canvas. + * @param layer - The ID of a style layer or a listener if no ID is provided. Event will only be triggered if its location + * is within a visible feature in this layer. The event will have a `features` property containing + * an array of the matching features. If `layer` is not supplied, the event will not have a `features` property. + * Please note that many event types are not compatible with the optional `layer` parameter. + * @param listener - The function to be called when the event is fired. + * @returns `this` + * @example + * ```ts + * // Set an event listener that will fire + * // when the map has finished loading + * map.on('load', function() { + * // Once the map has finished loading, + * // add a new layer + * map.addLayer({ + * id: 'points-of-interest', + * source: { + * type: 'vector', + * url: 'https://maplibre.org/maplibre-style-spec/' + * }, + * 'source-layer': 'poi_label', + * type: 'circle', + * paint: { + * // MapLibre Style Specification paint properties + * }, + * layout: { + * // MapLibre Style Specification layout properties + * } + * }); + * }); + * ``` + * @example + * ```ts + * // Set an event listener that will fire + * // when a feature on the countries layer of the map is clicked + * map.on('click', 'countries', (e) => { + * new maplibregl.Popup() + * .setLngLat(e.lngLat) + * .setHTML(`Country name: ${e.features[0].properties.name}`) + * .addTo(map); + * }); + * ``` + * @see [Display popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/) + * @see [Create a hover effect](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + * @see [Create a draggable marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + on(type: T, layer: string, listener: (ev: MapLayerEventType[T] & Object) => void): Map; + /** + * Overload of the `on` method that allows to listen to events without specifying a layer. + * @event + * @param type - The type of the event. + * @param listener - The listener callback. + * @returns `this` + */ + on(type: T, listener: (ev: MapEventType[T] & Object) => void): this; + /** + * Overload of the `on` method that allows to listen to events without specifying a layer. + * @event + * @param type - The type of the event. + * @param listener - The listener callback. + * @returns `this` + */ + on(type: keyof MapEventType | string, listener: Listener): this; + /** + * Adds a listener that will be called only once to a specified event type, optionally limited to features in a specified style layer. + * + * @event + * @param type - The event type to listen for; one of `'mousedown'`, `'mouseup'`, `'click'`, `'dblclick'`, + * `'mousemove'`, `'mouseenter'`, `'mouseleave'`, `'mouseover'`, `'mouseout'`, `'contextmenu'`, `'touchstart'`, + * `'touchend'`, or `'touchcancel'`. `mouseenter` and `mouseover` events are triggered when the cursor enters + * a visible portion of the specified layer from outside that layer or outside the map canvas. `mouseleave` + * and `mouseout` events are triggered when the cursor leaves a visible portion of the specified layer, or leaves + * the map canvas. + * @param layer - The ID of a style layer or a listener if no ID is provided. Only events whose location is within a visible + * feature in this layer will trigger the listener. The event will have a `features` property containing + * an array of the matching features. + * @param listener - The function to be called when the event is fired. + * @returns `this` if listener is provided, promise otherwise to allow easier usage of async/await + */ + once(type: T, layer: string, listener?: (ev: MapLayerEventType[T] & Object) => void): this | Promise; + /** + * Overload of the `once` method that allows to listen to events without specifying a layer. + * @event + * @param type - The type of the event. + * @param listener - The listener callback. + * @returns `this` + */ + once(type: T, listener?: (ev: MapEventType[T] & Object) => void): this | Promise; + /** + * Overload of the `once` method that allows to listen to events without specifying a layer. + * @event + * @param type - The type of the event. + * @param listener - The listener callback. + * @returns `this` + */ + once(type: keyof MapEventType | string, listener?: Listener): this | Promise; + /** + * Removes an event listener for events previously added with `Map#on`. + * + * @event + * @param type - The event type previously used to install the listener. + * @param layer - The layer ID or listener previously used to install the listener. + * @param listener - The function previously installed as a listener. + * @returns `this` + */ + off(type: T, layer: string, listener: (ev: MapLayerEventType[T] & Object) => void): this; + /** + * Overload of the `off` method that allows to listen to events without specifying a layer. + * @event + * @param type - The type of the event. + * @param listener - The function previously installed as a listener. + * @returns `this` + */ + off(type: T, listener: (ev: MapEventType[T] & Object) => void): this; + /** + * Overload of the `off` method that allows to listen to events without specifying a layer. + * @event + * @param type - The type of the event. + * @param listener - The function previously installed as a listener. + * @returns `this` + */ + off(type: keyof MapEventType | string, listener: Listener): this; + /** + * Returns an array of MapGeoJSONFeature objects + * representing visible features that satisfy the query parameters. + * + * @param geometryOrOptions - (optional) The geometry of the query region: + * either a single point or southwest and northeast points describing a bounding box. + * Omitting this parameter (i.e. calling {@link Map#queryRenderedFeatures} with zero arguments, + * or with only a `options` argument) is equivalent to passing a bounding box encompassing the entire + * map viewport. + * The geometryOrOptions can receive a {@link QueryRenderedFeaturesOptions} only to support a situation where the function receives only one parameter which is the options parameter. + * @param options - (optional) Options object. + * + * @returns An array of MapGeoJSONFeature objects. + * + * The `properties` value of each returned feature object contains the properties of its source feature. For GeoJSON sources, only + * string and numeric property values are supported (i.e. `null`, `Array`, and `Object` values are not supported). + * + * Each feature includes top-level `layer`, `source`, and `sourceLayer` properties. The `layer` property is an object + * representing the style layer to which the feature belongs. Layout and paint properties in this object contain values + * which are fully evaluated for the given zoom level and feature. + * + * Only features that are currently rendered are included. Some features will **not** be included, like: + * + * - Features from layers whose `visibility` property is `"none"`. + * - Features from layers whose zoom range excludes the current zoom level. + * - Symbol features that have been hidden due to text or icon collision. + * + * Features from all other layers are included, including features that may have no visible + * contribution to the rendered result; for example, because the layer's opacity or color alpha component is set to + * 0. + * + * The topmost rendered feature appears first in the returned array, and subsequent features are sorted by + * descending z-order. Features that are rendered multiple times (due to wrapping across the antemeridian at low + * zoom levels) are returned only once (though subject to the following caveat). + * + * Because features come from tiled vector data or GeoJSON data that is converted to tiles internally, feature + * geometries may be split or duplicated across tile boundaries and, as a result, features may appear multiple + * times in query results. For example, suppose there is a highway running through the bounding rectangle of a query. + * The results of the query will be those parts of the highway that lie within the map tiles covering the bounding + * rectangle, even if the highway extends into other tiles, and the portion of the highway within each map tile + * will be returned as a separate feature. Similarly, a point feature near a tile boundary may appear in multiple + * tiles due to tile buffering. + * + * @example + * Find all features at a point + * ```ts + * let features = map.queryRenderedFeatures( + * [20, 35], + * { layers: ['my-layer-name'] } + * ); + * ``` + * + * @example + * Find all features within a static bounding box + * ```ts + * let features = map.queryRenderedFeatures( + * [[10, 20], [30, 50]], + * { layers: ['my-layer-name'] } + * ); + * ``` + * + * @example + * Find all features within a bounding box around a point + * ```ts + * let width = 10; + * let height = 20; + * let features = map.queryRenderedFeatures([ + * [point.x - width / 2, point.y - height / 2], + * [point.x + width / 2, point.y + height / 2] + * ], { layers: ['my-layer-name'] }); + * ``` + * + * @example + * Query all rendered features from a single layer + * ```ts + * let features = map.queryRenderedFeatures({ layers: ['my-layer-name'] }); + * ``` + * @see [Get features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/queryrenderedfeatures/) + */ + queryRenderedFeatures(geometryOrOptions?: PointLike | [ + PointLike, + PointLike + ] | QueryRenderedFeaturesOptions, options?: QueryRenderedFeaturesOptions): MapGeoJSONFeature[]; + /** + * Returns an array of MapGeoJSONFeature objects + * representing features within the specified vector tile or GeoJSON source that satisfy the query parameters. + * + * @param sourceId - The ID of the vector tile or GeoJSON source to query. + * @param parameters - The options object. + * @returns An array of MapGeoJSONFeature objects. + * + * In contrast to {@link Map#queryRenderedFeatures}, this function returns all features matching the query parameters, + * whether or not they are rendered by the current style (i.e. visible). The domain of the query includes all currently-loaded + * vector tiles and GeoJSON source tiles: this function does not check tiles outside the currently + * visible viewport. + * + * Because features come from tiled vector data or GeoJSON data that is converted to tiles internally, feature + * geometries may be split or duplicated across tile boundaries and, as a result, features may appear multiple + * times in query results. For example, suppose there is a highway running through the bounding rectangle of a query. + * The results of the query will be those parts of the highway that lie within the map tiles covering the bounding + * rectangle, even if the highway extends into other tiles, and the portion of the highway within each map tile + * will be returned as a separate feature. Similarly, a point feature near a tile boundary may appear in multiple + * tiles due to tile buffering. + * + * @example + * Find all features in one source layer in a vector source + * ```ts + * let features = map.querySourceFeatures('your-source-id', { + * sourceLayer: 'your-source-layer' + * }); + * ``` + * + */ + querySourceFeatures(sourceId: string, parameters?: QuerySourceFeatureOptions | null): MapGeoJSONFeature[]; + /** + * Updates the map's MapLibre style object with a new value. + * + * If a style is already set when this is used and options.diff is set to true, the map renderer will attempt to compare the given style + * against the map's current state and perform only the changes necessary to make the map style match the desired state. Changes in sprites + * (images used for icons and patterns) and glyphs (fonts for label text) **cannot** be diffed. If the sprites or fonts used in the current + * style and the given style are different in any way, the map renderer will force a full update, removing the current style and building + * the given one from scratch. + * + * + * @param style - A JSON object conforming to the schema described in the + * [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/), or a URL to such JSON. + * @param options - The options object. + * @returns `this` + * + * @example + * ```ts + * map.setStyle("https://demotiles.maplibre.org/style.json"); + * + * map.setStyle('https://demotiles.maplibre.org/style.json', { + * transformStyle: (previousStyle, nextStyle) => ({ + * ...nextStyle, + * sources: { + * ...nextStyle.sources, + * // copy a source from previous style + * 'osm': previousStyle.sources.osm + * }, + * layers: [ + * // background layer + * nextStyle.layers[0], + * // copy a layer from previous style + * previousStyle.layers[0], + * // other layers from the next style + * ...nextStyle.layers.slice(1).map(layer => { + * // hide the layers we don't need from demotiles style + * if (layer.id.startsWith('geolines')) { + * layer.layout = {...layer.layout || {}, visibility: 'none'}; + * // filter out US polygons + * } else if (layer.id.startsWith('coastline') || layer.id.startsWith('countries')) { + * layer.filter = ['!=', ['get', 'ADM0_A3'], 'USA']; + * } + * return layer; + * }) + * ] + * }) + * }); + * ``` + */ + setStyle(style: StyleSpecification | string | null, options?: StyleSwapOptions & StyleOptions): this; + /** + * Updates the requestManager's transform request with a new function + * + * @param transformRequest - A callback run before the Map makes a request for an external URL. The callback can be used to modify the url, set headers, or set the credentials property for cross-origin requests. + * Expected to return an object with a `url` property and optionally `headers` and `credentials` properties + * + * @returns `this` + * + * @example + * ```ts + * map.setTransformRequest((url: string, resourceType: string) => {}); + * ``` + */ + setTransformRequest(transformRequest: RequestTransformFunction): this; + _getUIString(key: string): any; + _updateStyle(style: StyleSpecification | string | null, options?: StyleSwapOptions & StyleOptions): this; + _lazyInitEmptyStyle(): void; + _diffStyle(style: StyleSpecification | string, options?: StyleSwapOptions & StyleOptions): void; + _updateDiff(style: StyleSpecification, options?: StyleSwapOptions & StyleOptions): void; + /** + * Returns the map's MapLibre style object, a JSON object which can be used to recreate the map's style. + * + * @returns The map's style JSON object. + * + * @example + * ```ts + * let styleJson = map.getStyle(); + * ``` + * + */ + getStyle(): StyleSpecification; + /** + * Returns a Boolean indicating whether the map's style is fully loaded. + * + * @returns A Boolean indicating whether the style is fully loaded. + * + * @example + * ```ts + * let styleLoadStatus = map.isStyleLoaded(); + * ``` + */ + isStyleLoaded(): boolean | void; + /** + * Adds a source to the map's style. + * + * Events triggered: + * + * Triggers the `source.add` event. + * + * @param id - The ID of the source to add. Must not conflict with existing sources. + * @param source - The source object, conforming to the + * MapLibre Style Specification's [source definition](https://maplibre.org/maplibre-style-spec/sources) or + * {@link CanvasSourceSpecification}. + * @returns `this` + * @example + * ```ts + * map.addSource('my-data', { + * type: 'vector', + * url: 'https://demotiles.maplibre.org/tiles/tiles.json' + * }); + * ``` + * @example + * ```ts + * map.addSource('my-data', { + * "type": "geojson", + * "data": { + * "type": "Feature", + * "geometry": { + * "type": "Point", + * "coordinates": [-77.0323, 38.9131] + * }, + * "properties": { + * "title": "Mapbox DC", + * "marker-symbol": "monument" + * } + * } + * }); + * ``` + * @see GeoJSON source: [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/) + */ + addSource(id: string, source: SourceSpecification): this; + /** + * Returns a Boolean indicating whether the source is loaded. Returns `true` if the source with + * the given ID in the map's style has no outstanding network requests, otherwise `false`. + * + * A {@link ErrorEvent} event will be fired if there is no source wit the specified ID. + * + * @param id - The ID of the source to be checked. + * @returns A Boolean indicating whether the source is loaded. + * @example + * ```ts + * let sourceLoaded = map.isSourceLoaded('bathymetry-data'); + * ``` + */ + isSourceLoaded(id: string): boolean; + /** + * Loads a 3D terrain mesh, based on a "raster-dem" source. + * + * Triggers the `terrain` event. + * + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.setTerrain({ source: 'terrain' }); + * ``` + */ + setTerrain(options: TerrainSpecification | null): this; + /** + * Get the terrain-options if terrain is loaded + * @returns the TerrainSpecification passed to setTerrain + * @example + * ```ts + * map.getTerrain(); // { source: 'terrain' }; + * ``` + */ + getTerrain(): TerrainSpecification | null; + /** + * Returns a Boolean indicating whether all tiles in the viewport from all sources on + * the style are loaded. + * + * @returns A Boolean indicating whether all tiles are loaded. + * @example + * ```ts + * let tilesLoaded = map.areTilesLoaded(); + * ``` + */ + areTilesLoaded(): boolean; + /** + * Adds a [custom source type](#Custom Sources), making it available for use with + * {@link Map#addSource}. + * @param name - The name of the source type; source definition objects use this name in the `{type: ...}` field. + * @param SourceType - A {@link Source} constructor. + * @param callback - Called when the source type is ready or with an error argument if there is an error. + */ + addSourceType(name: string, SourceType: SourceClass, callback: Callback): void; + /** + * Removes a source from the map's style. + * + * @param id - The ID of the source to remove. + * @returns `this` + * @example + * ```ts + * map.removeSource('bathymetry-data'); + * ``` + */ + removeSource(id: string): Map; + /** + * Returns the source with the specified ID in the map's style. + * + * This method is often used to update a source using the instance members for the relevant + * source type as defined in [Sources](#sources). + * For example, setting the `data` for a GeoJSON source or updating the `url` and `coordinates` + * of an image source. + * + * @param id - The ID of the source to get. + * @returns The style source with the specified ID or `undefined` if the ID + * corresponds to no existing sources. + * The shape of the object varies by source type. + * A list of options for each source type is available on the MapLibre Style Specification's + * [Sources](https://maplibre.org/maplibre-style-spec/sources/) page. + * @example + * ```ts + * let sourceObject = map.getSource('points'); + * ``` + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + * @see [Animate a point](https://maplibre.org/maplibre-gl-js/docs/examples/animate-point-along-line/) + * @see [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/) + */ + getSource(id: string): Source | undefined; + /** + * Add an image to the style. This image can be displayed on the map like any other icon in the style's + * sprite using the image's ID with + * [`icon-image`](https://maplibre.org/maplibre-style-spec/layers/#layout-symbol-icon-image), + * [`background-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-background-background-pattern), + * [`fill-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-fill-fill-pattern), + * or [`line-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-line-line-pattern). + * + * A {@link ErrorEvent} event will be fired if the image parameter is invalid or there is not enough space in the sprite to add this image. + * + * @param id - The ID of the image. + * @param image - The image as an `HTMLImageElement`, `ImageData`, `ImageBitmap` or object with `width`, `height`, and `data` + * properties with the same format as `ImageData`. + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * // If the style's sprite does not already contain an image with ID 'cat', + * // add the image 'cat-icon.png' to the style's sprite with the ID 'cat'. + * map.loadImage('https://upload.wikimedia.org/wikipedia/commons/thumb/6/60/Cat_silhouette.svg/400px-Cat_silhouette.svg.png', function(error, image) { + * if (error) throw error; + * if (!map.hasImage('cat')) map.addImage('cat', image); + * }); + * + * // Add a stretchable image that can be used with `icon-text-fit` + * // In this example, the image is 600px wide by 400px high. + * map.loadImage('https://upload.wikimedia.org/wikipedia/commons/8/89/Black_and_White_Boxed_%28bordered%29.png', function(error, image) { + * if (error) throw error; + * if (!map.hasImage('border-image')) { + * map.addImage('border-image', image, { + * content: [16, 16, 300, 384], // place text over left half of image, avoiding the 16px border + * stretchX: [[16, 584]], // stretch everything horizontally except the 16px border + * stretchY: [[16, 384]], // stretch everything vertically except the 16px border + * }); + * } + * }); + * ``` + * @see Use `HTMLImageElement`: [Add an icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image/) + * @see Use `ImageData`: [Add a generated icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-generated/) + */ + addImage(id: string, image: HTMLImageElement | ImageBitmap | ImageData | { + width: number; + height: number; + data: Uint8Array | Uint8ClampedArray; + } | StyleImageInterface, options?: Partial): this; + /** + * Update an existing image in a style. This image can be displayed on the map like any other icon in the style's + * sprite using the image's ID with + * [`icon-image`](https://maplibre.org/maplibre-style-spec/layers/#layout-symbol-icon-image), + * [`background-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-background-background-pattern), + * [`fill-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-fill-fill-pattern), + * or [`line-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-line-line-pattern). + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param id - The ID of the image. + * @param image - The image as an `HTMLImageElement`, `ImageData`, `ImageBitmap` or object with `width`, `height`, and `data` + * properties with the same format as `ImageData`. + * @returns `this` + * @example + * ```ts + * // If an image with the ID 'cat' already exists in the style's sprite, + * // replace that image with a new image, 'other-cat-icon.png'. + * if (map.hasImage('cat')) map.updateImage('cat', './other-cat-icon.png'); + * ``` + */ + updateImage(id: string, image: HTMLImageElement | ImageBitmap | ImageData | { + width: number; + height: number; + data: Uint8Array | Uint8ClampedArray; + } | StyleImageInterface): this; + /** + * Returns an image, specified by ID, currently available in the map. + * This includes both images from the style's original sprite + * and any images that have been added at runtime using {@link Map#addImage}. + * + * @param id - The ID of the image. + * @returns An image in the map with the specified ID. + * + * @example + * ```ts + * let coffeeShopIcon = map.getImage("coffee_cup"); + * ``` + */ + getImage(id: string): StyleImage; + /** + * Check whether or not an image with a specific ID exists in the style. This checks both images + * in the style's original sprite and any images + * that have been added at runtime using {@link Map#addImage}. + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param id - The ID of the image. + * + * @returns A Boolean indicating whether the image exists. + * @example + * Check if an image with the ID 'cat' exists in the style's sprite. + * ```ts + * let catIconExists = map.hasImage('cat'); + * ``` + */ + hasImage(id: string): boolean; + /** + * Remove an image from a style. This can be an image from the style's original + * sprite or any images + * that have been added at runtime using {@link Map#addImage}. + * + * @param id - The ID of the image. + * + * @example + * ```ts + * // If an image with the ID 'cat' exists in + * // the style's sprite, remove it. + * if (map.hasImage('cat')) map.removeImage('cat'); + * ``` + */ + removeImage(id: string): void; + /** + * Load an image from an external URL to be used with {@link Map#addImage}. External + * domains must support [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/Access_control_CORS). + * + * @param url - The URL of the image file. Image file must be in png, webp, or jpg format. + * @param callback - Expecting `callback(error, data)`. Called when the image has loaded or with an error argument if there is an error. + * + * @example + * Load an image from an external URL. + * ```ts + * map.loadImage('http://placekitten.com/50/50', function(error, image) { + * if (error) throw error; + * // Add the loaded image to the style's sprite with the ID 'kitten'. + * map.addImage('kitten', image); + * }); + * ``` + * @see [Add an icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image/) + */ + loadImage(url: string, callback: GetImageCallback): void; + /** + * Returns an Array of strings containing the IDs of all images currently available in the map. + * This includes both images from the style's original sprite + * and any images that have been added at runtime using {@link Map#addImage}. + * + * @returns An Array of strings containing the names of all sprites/images currently available in the map. + * + * @example + * ```ts + * let allImages = map.listImages(); + * ``` + */ + listImages(): Array; + /** + * Adds a [MapLibre style layer](https://maplibre.org/maplibre-style-spec/layers) + * to the map's style. + * + * A layer defines how data from a specified source will be styled. Read more about layer types + * and available paint and layout properties in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/layers). + * + * @param layer - The layer to add, + * conforming to either the MapLibre Style Specification's [layer definition](https://maplibre.org/maplibre-style-spec/layers) or, + * less commonly, the {@link CustomLayerInterface} specification. Can also be a layer definition with an embedded source definition. + * The MapLibre Style Specification's layer definition is appropriate for most layers. + * + * @param beforeId - The ID of an existing layer to insert the new layer before, + * resulting in the new layer appearing visually beneath the existing layer. + * If this argument is not specified, the layer will be appended to the end of the layers array + * and appear visually above all other layers. + * + * @returns `this` + * + * @example + * Add a circle layer with a vector source + * ```ts + * map.addLayer({ + * id: 'points-of-interest', + * source: { + * type: 'vector', + * url: 'https://demotiles.maplibre.org/tiles/tiles.json' + * }, + * 'source-layer': 'poi_label', + * type: 'circle', + * paint: { + * // MapLibre Style Specification paint properties + * }, + * layout: { + * // MapLibre Style Specification layout properties + * } + * }); + * ``` + * + * @example + * Define a source before using it to create a new layer + * ```ts + * map.addSource('state-data', { + * type: 'geojson', + * data: 'path/to/data.geojson' + * }); + * + * map.addLayer({ + * id: 'states', + * // References the GeoJSON source defined above + * // and does not require a `source-layer` + * source: 'state-data', + * type: 'symbol', + * layout: { + * // Set the label content to the + * // feature's `name` property + * text-field: ['get', 'name'] + * } + * }); + * ``` + * + * @example + * Add a new symbol layer before an existing layer + * ```ts + * map.addLayer({ + * id: 'states', + * // References a source that's already been defined + * source: 'state-data', + * type: 'symbol', + * layout: { + * // Set the label content to the + * // feature's `name` property + * text-field: ['get', 'name'] + * } + * // Add the layer before the existing `cities` layer + * }, 'cities'); + * ``` + * @see [Create and style clusters](https://maplibre.org/maplibre-gl-js/docs/examples/cluster/) + * @see [Add a vector tile source](https://maplibre.org/maplibre-gl-js/docs/examples/vector-source/) + * @see [Add a WMS source](https://maplibre.org/maplibre-gl-js/docs/examples/wms/) + */ + addLayer(layer: AddLayerObject, beforeId?: string): this; + /** + * Moves a layer to a different z-position. + * + * @param id - The ID of the layer to move. + * @param beforeId - The ID of an existing layer to insert the new layer before. When viewing the map, the `id` layer will appear beneath the `beforeId` layer. If `beforeId` is omitted, the layer will be appended to the end of the layers array and appear above all other layers on the map. + * @returns `this` + * + * @example + * Move a layer with ID 'polygon' before the layer with ID 'country-label'. The `polygon` layer will appear beneath the `country-label` layer on the map. + * ```ts + * map.moveLayer('polygon', 'country-label'); + * ``` + */ + moveLayer(id: string, beforeId?: string): this; + /** + * Removes the layer with the given ID from the map's style. + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param id - The ID of the layer to remove + * @returns `this` + * + * @example + * If a layer with ID 'state-data' exists, remove it. + * ```ts + * if (map.getLayer('state-data')) map.removeLayer('state-data'); + * ``` + */ + removeLayer(id: string): this; + /** + * Returns the layer with the specified ID in the map's style. + * + * @param id - The ID of the layer to get. + * @returns The layer with the specified ID, or `undefined` + * if the ID corresponds to no existing layers. + * + * @example + * ```ts + * let stateDataLayer = map.getLayer('state-data'); + * ``` + * @see [Filter symbols by toggling a list](https://maplibre.org/maplibre-gl-js/docs/examples/filter-markers/) + * @see [Filter symbols by text input](https://maplibre.org/maplibre-gl-js/docs/examples/filter-markers-by-input/) + */ + getLayer(id: string): StyleLayer | undefined; + /** + * Return the ids of all layers currently in the style, including custom layers, in order. + * + * @returns ids of layers, in order + * + * @example + * ```ts + * const orderedLayerIds = map.getLayersOrder(); + * ``` + */ + getLayersOrder(): string[]; + /** + * Sets the zoom extent for the specified style layer. The zoom extent includes the + * [minimum zoom level](https://maplibre.org/maplibre-style-spec/layers/#minzoom) + * and [maximum zoom level](https://maplibre.org/maplibre-style-spec/layers/#maxzoom)) + * at which the layer will be rendered. + * + * Note: For style layers using vector sources, style layers cannot be rendered at zoom levels lower than the + * minimum zoom level of the _source layer_ because the data does not exist at those zoom levels. If the minimum + * zoom level of the source layer is higher than the minimum zoom level defined in the style layer, the style + * layer will not be rendered at all zoom levels in the zoom range. + * + * @param layerId - The ID of the layer to which the zoom extent will be applied. + * @param minzoom - The minimum zoom to set (0-24). + * @param maxzoom - The maximum zoom to set (0-24). + * @returns `this` + * + * @example + * ```ts + * map.setLayerZoomRange('my-layer', 2, 5); + * ``` + */ + setLayerZoomRange(layerId: string, minzoom: number, maxzoom: number): this; + /** + * Sets the filter for the specified style layer. + * + * Filters control which features a style layer renders from its source. + * Any feature for which the filter expression evaluates to `true` will be + * rendered on the map. Those that are false will be hidden. + * + * Use `setFilter` to show a subset of your source data. + * + * To clear the filter, pass `null` or `undefined` as the second parameter. + * + * @param layerId - The ID of the layer to which the filter will be applied. + * @param filter - The filter, conforming to the MapLibre Style Specification's + * [filter definition](https://maplibre.org/maplibre-style-spec/layers/#filter). If `null` or `undefined` is provided, the function removes any existing filter from the layer. + * @param options - Options object. + * @returns `this` + * + * @example + * Display only features with the 'name' property 'USA' + * ```ts + * map.setFilter('my-layer', ['==', ['get', 'name'], 'USA']); + * ``` + * @example + * Display only features with five or more 'available-spots' + * ```ts + * map.setFilter('bike-docks', ['>=', ['get', 'available-spots'], 5]); + * ``` + * @example + * Remove the filter for the 'bike-docks' style layer + * ```ts + * map.setFilter('bike-docks', null); + * ``` + * @see [Create a timeline animation](https://maplibre.org/maplibre-gl-js/docs/examples/timeline-animation/) + */ + setFilter(layerId: string, filter?: FilterSpecification | null, options?: StyleSetterOptions): this; + /** + * Returns the filter applied to the specified style layer. + * + * @param layerId - The ID of the style layer whose filter to get. + * @returns The layer's filter. + */ + getFilter(layerId: string): FilterSpecification | void; + /** + * Sets the value of a paint property in the specified style layer. + * + * @param layerId - The ID of the layer to set the paint property in. + * @param name - The name of the paint property to set. + * @param value - The value of the paint property to set. + * Must be of a type appropriate for the property, as defined in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/). + * Pass `null` to unset the existing value. + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.setPaintProperty('my-layer', 'fill-color', '#faafee'); + * ``` + * @see [Change a layer's color with buttons](https://maplibre.org/maplibre-gl-js/docs/examples/color-switcher/) + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + setPaintProperty(layerId: string, name: string, value: any, options?: StyleSetterOptions): this; + /** + * Returns the value of a paint property in the specified style layer. + * + * @param layerId - The ID of the layer to get the paint property from. + * @param name - The name of a paint property to get. + * @returns The value of the specified paint property. + */ + getPaintProperty(layerId: string, name: string): unknown; + /** + * Sets the value of a layout property in the specified style layer. + * + * @param layerId - The ID of the layer to set the layout property in. + * @param name - The name of the layout property to set. + * @param value - The value of the layout property. Must be of a type appropriate for the property, as defined in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/). + * @param options - The options object. + * @returns `this` + * @example + * ```ts + * map.setLayoutProperty('my-layer', 'visibility', 'none'); + * ``` + */ + setLayoutProperty(layerId: string, name: string, value: any, options?: StyleSetterOptions): this; + /** + * Returns the value of a layout property in the specified style layer. + * + * @param layerId - The ID of the layer to get the layout property from. + * @param name - The name of the layout property to get. + * @returns The value of the specified layout property. + */ + getLayoutProperty(layerId: string, name: string): any; + /** + * Sets the value of the style's glyphs property. + * + * @param glyphsUrl - Glyph URL to set. Must conform to the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/glyphs/). + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.setGlyphs('https://demotiles.maplibre.org/font/{fontstack}/{range}.pbf'); + * ``` + */ + setGlyphs(glyphsUrl: string | null, options?: StyleSetterOptions): this; + /** + * Returns the value of the style's glyphs URL + * + * @returns glyphs Style's glyphs url + */ + getGlyphs(): string | null; + /** + * Adds a sprite to the map's style. Fires the `style` event. + * + * @param id - The ID of the sprite to add. Must not conflict with existing sprites. + * @param url - The URL to load the sprite from + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.addSprite('sprite-two', 'http://example.com/sprite-two'); + * ``` + */ + addSprite(id: string, url: string, options?: StyleSetterOptions): this; + /** + * Removes the sprite from the map's style. Fires the `style` event. + * + * @param id - The ID of the sprite to remove. If the sprite is declared as a single URL, the ID must be "default". + * @returns `this` + * @example + * ```ts + * map.removeSprite('sprite-two'); + * map.removeSprite('default'); + * ``` + */ + removeSprite(id: string): this; + /** + * Returns the as-is value of the style's sprite. + * + * @returns style's sprite list of id-url pairs + */ + getSprite(): { + id: string; + url: string; + }[]; + /** + * Sets the value of the style's sprite property. + * + * @param spriteUrl - Sprite URL to set. + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.setSprite('YOUR_SPRITE_URL'); + * ``` + */ + setSprite(spriteUrl: string | null, options?: StyleSetterOptions): this; + /** + * Sets the any combination of light values. + * + * @param light - Light properties to set. Must conform to the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/light). + * @param options - Options object. + * @returns `this` + * + * @example + * ```ts + * let layerVisibility = map.getLayoutProperty('my-layer', 'visibility'); + * ``` + */ + setLight(light: LightSpecification, options?: StyleSetterOptions): this; + /** + * Returns the value of the light object. + * + * @returns light Light properties of the style. + */ + getLight(): LightSpecification; + /** + * Sets the `state` of a feature. + * A feature's `state` is a set of user-defined key-value pairs that are assigned to a feature at runtime. + * When using this method, the `state` object is merged with any existing key-value pairs in the feature's state. + * Features are identified by their `feature.id` attribute, which can be any number or string. + * + * This method can only be used with sources that have a `feature.id` attribute. The `feature.id` attribute can be defined in three ways: + * - For vector or GeoJSON sources, including an `id` attribute in the original data file. + * - For vector or GeoJSON sources, using the [`promoteId`](https://maplibre.org/maplibre-style-spec/sources/#vector-promoteId) option at the time the source is defined. + * - For GeoJSON sources, using the [`generateId`](https://maplibre.org/maplibre-style-spec/sources/#geojson-generateId) option to auto-assign an `id` based on the feature's index in the source data. If you change feature data using `map.getSource('some id').setData(..)`, you may need to re-apply state taking into account updated `id` values. + * + * _Note: You can use the [`feature-state` expression](https://maplibre.org/maplibre-style-spec/expressions/#feature-state) to access the values in a feature's state object for the purposes of styling._ + * + * @param feature - Feature identifier. Feature objects returned from + * {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers. + * @param state - A set of key-value pairs. The values should be valid JSON types. + * @returns `this` + * + * @example + * ```ts + * // When the mouse moves over the `my-layer` layer, update + * // the feature state for the feature under the mouse + * map.on('mousemove', 'my-layer', function(e) { + * if (e.features.length > 0) { + * map.setFeatureState({ + * source: 'my-source', + * sourceLayer: 'my-source-layer', + * id: e.features[0].id, + * }, { + * hover: true + * }); + * } + * }); + * ``` + * @see [Create a hover effect](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + */ + setFeatureState(feature: FeatureIdentifier, state: any): this; + /** + * Removes the `state` of a feature, setting it back to the default behavior. + * If only a `target.source` is specified, it will remove the state for all features from that source. + * If `target.id` is also specified, it will remove all keys for that feature's state. + * If `key` is also specified, it removes only that key from that feature's state. + * Features are identified by their `feature.id` attribute, which can be any number or string. + * + * @param target - Identifier of where to remove state. It can be a source, a feature, or a specific key of feature. + * Feature objects returned from {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers. + * @param key - (optional) The key in the feature state to reset. + * @returns `this` + * @example + * Reset the entire state object for all features in the `my-source` source + * ```ts + * map.removeFeatureState({ + * source: 'my-source' + * }); + * ``` + * + * @example + * When the mouse leaves the `my-layer` layer, + * reset the entire state object for the + * feature under the mouse + * ```ts + * map.on('mouseleave', 'my-layer', function(e) { + * map.removeFeatureState({ + * source: 'my-source', + * sourceLayer: 'my-source-layer', + * id: e.features[0].id + * }); + * }); + * ``` + * + * @example + * When the mouse leaves the `my-layer` layer, + * reset only the `hover` key-value pair in the + * state for the feature under the mouse + * ```ts + * map.on('mouseleave', 'my-layer', function(e) { + * map.removeFeatureState({ + * source: 'my-source', + * sourceLayer: 'my-source-layer', + * id: e.features[0].id + * }, 'hover'); + * }); + * ``` + */ + removeFeatureState(target: FeatureIdentifier, key?: string): this; + /** + * Gets the `state` of a feature. + * A feature's `state` is a set of user-defined key-value pairs that are assigned to a feature at runtime. + * Features are identified by their `feature.id` attribute, which can be any number or string. + * + * _Note: To access the values in a feature's state object for the purposes of styling the feature, use the [`feature-state` expression](https://maplibre.org/maplibre-style-spec/expressions/#feature-state)._ + * + * @param feature - Feature identifier. Feature objects returned from + * {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers. + * @returns The state of the feature: a set of key-value pairs that was assigned to the feature at runtime. + * + * @example + * When the mouse moves over the `my-layer` layer, + * get the feature state for the feature under the mouse + * ```ts + * map.on('mousemove', 'my-layer', function(e) { + * if (e.features.length > 0) { + * map.getFeatureState({ + * source: 'my-source', + * sourceLayer: 'my-source-layer', + * id: e.features[0].id + * }); + * } + * }); + * ``` + */ + getFeatureState(feature: FeatureIdentifier): any; + /** + * Returns the map's containing HTML element. + * + * @returns The map's container. + */ + getContainer(): HTMLElement; + /** + * Returns the HTML element containing the map's `` element. + * + * If you want to add non-GL overlays to the map, you should append them to this element. + * + * This is the element to which event bindings for map interactivity (such as panning and zooming) are + * attached. It will receive bubbled events from child elements such as the ``, but not from + * map controls. + * + * @returns The container of the map's ``. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + getCanvasContainer(): HTMLElement; + /** + * Returns the map's `` element. + * + * @returns The map's `` element. + * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/) + */ + getCanvas(): HTMLCanvasElement; + _containerDimensions(): number[]; + _setupContainer(): void; + _cooperativeGesturesOnWheel: (event: WheelEvent) => void; + _setupCooperativeGestures(): void; + _destroyCooperativeGestures(): void; + _resizeCanvas(width: number, height: number, pixelRatio: number): void; + _setupPainter(): void; + _contextLost: (event: any) => void; + _contextRestored: (event: any) => void; + _onMapScroll: (event: any) => boolean; + _onCooperativeGesture(event: any, metaPress: any, touches: any): boolean; + /** + * Returns a Boolean indicating whether the map is fully loaded. + * + * Returns `false` if the style is not yet fully loaded, + * or if there has been a change to the sources or style that + * has not yet fully loaded. + * + * @returns A Boolean indicating whether the map is fully loaded. + */ + loaded(): boolean; + /** + * @internal + * Update this map's style and sources, and re-render the map. + * + * @param updateStyle - mark the map's style for reprocessing as + * well as its sources + * @returns `this` + */ + _update(updateStyle?: boolean): this; + /** + * @internal + * Request that the given callback be executed during the next render + * frame. Schedule a render frame if one is not already scheduled. + * + * @returns An id that can be used to cancel the callback + */ + _requestRenderFrame(callback: () => void): TaskID; + _cancelRenderFrame(id: TaskID): void; + /** + * @internal + * Call when a (re-)render of the map is required: + * - The style has changed (`setPaintProperty()`, etc.) + * - Source data has changed (e.g. tiles have finished loading) + * - The map has is moving (or just finished moving) + * - A transition is in progress + * + * @param paintStartTimeStamp - The time when the animation frame began executing. + * + * @returns `this` + */ + _render(paintStartTimeStamp: number): this; + /** + * Force a synchronous redraw of the map. + * @returns `this` + * @example + * ```ts + * map.redraw(); + * ``` + */ + redraw(): this; + /** + * Clean up and release all internal resources associated with this map. + * + * This includes DOM elements, event bindings, web workers, and WebGL resources. + * + * Use this method when you are done using the map and wish to ensure that it no + * longer consumes browser resources. Afterwards, you must not call any other + * methods on the map. + */ + remove(): void; + /** + * Trigger the rendering of a single frame. Use this method with custom layers to + * repaint the map when the layer changes. Calling this multiple times before the + * next frame is rendered will still result in only a single frame being rendered. + * @example + * ```ts + * map.triggerRepaint(); + * ``` + * @see [Add a 3D model](https://maplibre.org/maplibre-gl-js/docs/examples/add-3d-model/) + * @see [Add an animated icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-animated/) + */ + triggerRepaint(): void; + _onWindowOnline: () => void; + /** + * Gets and sets a Boolean indicating whether the map will render an outline + * around each tile and the tile ID. These tile boundaries are useful for + * debugging. + * + * The uncompressed file size of the first vector source is drawn in the top left + * corner of each tile, next to the tile ID. + * + * @example + * ```ts + * map.showTileBoundaries = true; + * ``` + */ + get showTileBoundaries(): boolean; + set showTileBoundaries(value: boolean); + /** + * Gets and sets a Boolean indicating whether the map will visualize + * the padding offsets. + */ + get showPadding(): boolean; + set showPadding(value: boolean); + /** + * Gets and sets a Boolean indicating whether the map will render boxes + * around all symbols in the data source, revealing which symbols + * were rendered or which were hidden due to collisions. + * This information is useful for debugging. + */ + get showCollisionBoxes(): boolean; + set showCollisionBoxes(value: boolean); + /** + * Gets and sets a Boolean indicating whether the map should color-code + * each fragment to show how many times it has been shaded. + * White fragments have been shaded 8 or more times. + * Black fragments have been shaded 0 times. + * This information is useful for debugging. + */ + get showOverdrawInspector(): boolean; + set showOverdrawInspector(value: boolean); + /** + * Gets and sets a Boolean indicating whether the map will + * continuously repaint. This information is useful for analyzing performance. + */ + get repaint(): boolean; + set repaint(value: boolean); + get vertices(): boolean; + set vertices(value: boolean); + /** + * Returns the package version of the library + * @returns Package version of the library + */ + get version(): string; + /** + * Returns the elevation for the point where the camera is looking. + * This value corresponds to: + * "meters above sea level" * "exaggeration" + * @returns The elevation. + */ + getCameraTargetElevation(): number; +} +export interface OneFingerTouchRotateHandler extends DragMoveHandler { +} +export interface OneFingerTouchPitchHandler extends DragMoveHandler { +} +/** + * The {@link NavigationControl} options object + */ +export type NavigationOptions = { + /** + * If `true` the compass button is included. + */ + showCompass?: boolean; + /** + * If `true` the zoom-in and zoom-out buttons are included. + */ + showZoom?: boolean; + /** + * If `true` the pitch is visualized by rotating X-axis of compass. + */ + visualizePitch?: boolean; +}; +export declare class NavigationControl implements IControl { + _map: Map; + options: NavigationOptions; + _container: HTMLElement; + _zoomInButton: HTMLButtonElement; + _zoomOutButton: HTMLButtonElement; + _compass: HTMLButtonElement; + _compassIcon: HTMLElement; + _handler: MouseRotateWrapper; + /** + * @param options - the control's options + */ + constructor(options?: NavigationOptions); + _updateZoomButtons: () => void; + _rotateCompassArrow: () => void; + onAdd(map: Map): HTMLElement; + onRemove(): void; + _createButton(className: string, fn: (e?: any) => unknown): HTMLButtonElement; + _setButtonTitle: (button: HTMLButtonElement, title: string) => void; +} +export declare class MouseRotateWrapper { + map: Map; + _clickTolerance: number; + element: HTMLElement; + mouseRotate: MouseRotateHandler; + touchRotate: OneFingerTouchRotateHandler; + mousePitch: MousePitchHandler; + touchPitch: OneFingerTouchPitchHandler; + _startPos: Point; + _lastPos: Point; + constructor(map: Map, element: HTMLElement, pitch?: boolean); + startMouse(e: MouseEvent, point: Point): void; + startTouch(e: TouchEvent, point: Point): void; + moveMouse(e: MouseEvent, point: Point): void; + moveTouch(e: TouchEvent, point: Point): void; + off(): void; + offTemp(): void; + mousedown: (e: MouseEvent) => void; + mousemove: (e: MouseEvent) => void; + mouseup: (e: MouseEvent) => void; + touchstart: (e: TouchEvent) => void; + touchmove: (e: TouchEvent) => void; + touchend: (e: TouchEvent) => void; + reset: () => void; +} +/** + * Where to position the anchor. + * Used by a popup and a marker. + */ +export type PositionAnchor = "center" | "top" | "bottom" | "left" | "right" | "top-left" | "top-right" | "bottom-left" | "bottom-right"; +/** + * A pixel offset specified as: + * - a single number specifying a distance from the location + * - a {@link PointLike} specifying a constant offset + * - an object of {@link Point}s specifying an offset for each anchor position + * Negative offsets indicate left and up. + */ +export type Offset = number | PointLike | { + [_ in PositionAnchor]: PointLike; +}; +export type PopupOptions = { + /** + * If `true`, a close button will appear in the top right corner of the popup. + * @defaultValue true + */ + closeButton?: boolean; + /** + * If `true`, the popup will closed when the map is clicked. + * @defaultValue true + */ + closeOnClick?: boolean; + /** + * If `true`, the popup will closed when the map moves. + * @defaultValue false + */ + closeOnMove?: boolean; + /** + * If `true`, the popup will try to focus the first focusable element inside the popup. + * @defaultValue true + */ + focusAfterOpen?: boolean; + /** + * A string indicating the part of the Popup that should + * be positioned closest to the coordinate set via {@link Popup#setLngLat}. + * Options are `'center'`, `'top'`, `'bottom'`, `'left'`, `'right'`, `'top-left'`, + * `'top-right'`, `'bottom-left'`, and `'bottom-right'`. If unset the anchor will be + * dynamically set to ensure the popup falls within the map container with a preference + * for `'bottom'`. + */ + anchor?: PositionAnchor; + /** + * A pixel offset applied to the popup's location + */ + offset?: Offset; + /** + * Space-separated CSS class names to add to popup container + */ + className?: string; + /** + * A string that sets the CSS property of the popup's maximum width, eg `'300px'`. + * To ensure the popup resizes to fit its content, set this property to `'none'`. + * Available values can be found here: https://developer.mozilla.org/en-US/docs/Web/CSS/max-width + * @defaultValue '240px' + */ + maxWidth?: string; +}; +export declare class Popup extends Evented { + _map: Map; + options: PopupOptions; + _content: HTMLElement; + _container: HTMLElement; + _closeButton: HTMLButtonElement; + _tip: HTMLElement; + _lngLat: LngLat; + _trackPointer: boolean; + _pos: Point; + constructor(options?: PopupOptions); + /** + * Adds the popup to a map. + * + * @param map - The MapLibre GL JS map to add the popup to. + * @returns `this` + * @example + * ```ts + * new maplibregl.Popup() + * .setLngLat([0, 0]) + * .setHTML("

Null Island

") + * .addTo(map); + * ``` + * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + * @see [Show polygon information on click](https://maplibre.org/maplibre-gl-js/docs/examples/polygon-popup-on-click/) + */ + addTo(map: Map): this; + /** + * @returns `true` if the popup is open, `false` if it is closed. + */ + isOpen(): boolean; + /** + * Removes the popup from the map it has been added to. + * + * @example + * ```ts + * let popup = new maplibregl.Popup().addTo(map); + * popup.remove(); + * ``` + * @returns `this` + */ + remove: () => this; + /** + * Returns the geographical location of the popup's anchor. + * + * The longitude of the result may differ by a multiple of 360 degrees from the longitude previously + * set by `setLngLat` because `Popup` wraps the anchor longitude across copies of the world to keep + * the popup on screen. + * + * @returns The geographical location of the popup's anchor. + */ + getLngLat(): LngLat; + /** + * Sets the geographical location of the popup's anchor, and moves the popup to it. Replaces trackPointer() behavior. + * + * @param lnglat - The geographical location to set as the popup's anchor. + * @returns `this` + */ + setLngLat(lnglat: LngLatLike): this; + /** + * Tracks the popup anchor to the cursor position on screens with a pointer device (it will be hidden on touchscreens). Replaces the `setLngLat` behavior. + * For most use cases, set `closeOnClick` and `closeButton` to `false`. + * @example + * ```ts + * let popup = new maplibregl.Popup({ closeOnClick: false, closeButton: false }) + * .setHTML("

Hello World!

") + * .trackPointer() + * .addTo(map); + * ``` + * @returns `this` + */ + trackPointer(): this; + /** + * Returns the `Popup`'s HTML element. + * @example + * Change the `Popup` element's font size + * ```ts + * let popup = new maplibregl.Popup() + * .setLngLat([-96, 37.8]) + * .setHTML("

Hello World!

") + * .addTo(map); + * let popupElem = popup.getElement(); + * popupElem.style.fontSize = "25px"; + * ``` + * @returns element + */ + getElement(): HTMLElement; + /** + * Sets the popup's content to a string of text. + * + * This function creates a [Text](https://developer.mozilla.org/en-US/docs/Web/API/Text) node in the DOM, + * so it cannot insert raw HTML. Use this method for security against XSS + * if the popup content is user-provided. + * + * @param text - Textual content for the popup. + * @returns `this` + * @example + * ```ts + * let popup = new maplibregl.Popup() + * .setLngLat(e.lngLat) + * .setText('Hello, world!') + * .addTo(map); + * ``` + */ + setText(text: string): this; + /** + * Sets the popup's content to the HTML provided as a string. + * + * This method does not perform HTML filtering or sanitization, and must be + * used only with trusted content. Consider {@link Popup#setText} if + * the content is an untrusted text string. + * + * @param html - A string representing HTML content for the popup. + * @returns `this` + * @example + * ```ts + * let popup = new maplibregl.Popup() + * .setLngLat(e.lngLat) + * .setHTML("

Hello World!

") + * .addTo(map); + * ``` + * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/) + */ + setHTML(html: string): this; + /** + * Returns the popup's maximum width. + * + * @returns The maximum width of the popup. + */ + getMaxWidth(): string; + /** + * Sets the popup's maximum width. This is setting the CSS property `max-width`. + * Available values can be found here: https://developer.mozilla.org/en-US/docs/Web/CSS/max-width + * + * @param maxWidth - A string representing the value for the maximum width. + * @returns `this` + */ + setMaxWidth(maxWidth: string): this; + /** + * Sets the popup's content to the element provided as a DOM node. + * + * @param htmlNode - A DOM node to be used as content for the popup. + * @returns `this` + * @example + * Create an element with the popup content + * ```ts + * let div = document.createElement('div'); + * div.innerHTML = 'Hello, world!'; + * let popup = new maplibregl.Popup() + * .setLngLat(e.lngLat) + * .setDOMContent(div) + * .addTo(map); + * ``` + */ + setDOMContent(htmlNode: Node): this; + /** + * Adds a CSS class to the popup container element. + * + * @param className - Non-empty string with CSS class name to add to popup container + * + * @example + * ```ts + * let popup = new maplibregl.Popup() + * popup.addClassName('some-class') + * ``` + */ + addClassName(className: string): void; + /** + * Removes a CSS class from the popup container element. + * + * @param className - Non-empty string with CSS class name to remove from popup container + * + * @example + * ```ts + * let popup = new maplibregl.Popup() + * popup.removeClassName('some-class') + * ``` + */ + removeClassName(className: string): void; + /** + * Sets the popup's offset. + * + * @param offset - Sets the popup's offset. + * @returns `this` + */ + setOffset(offset?: Offset): this; + /** + * Add or remove the given CSS class on the popup container, depending on whether the container currently has that class. + * + * @param className - Non-empty string with CSS class name to add/remove + * + * @returns if the class was removed return false, if class was added, then return true, undefined if there is no container + * + * @example + * ```ts + * let popup = new maplibregl.Popup() + * popup.toggleClassName('toggleClass') + * ``` + */ + toggleClassName(className: string): boolean | undefined; + _createCloseButton(): void; + _onMouseUp: (event: MapMouseEvent) => void; + _onMouseMove: (event: MapMouseEvent) => void; + _onDrag: (event: MapMouseEvent) => void; + _update: (cursor?: Point) => void; + _focusFirstElement(): void; + _onClose: () => void; +} +/** + * Alignment options of rotation and pitch + */ +export type Alignment = "map" | "viewport" | "auto"; +/** + * The {@link Marker} options object + */ +export type MarkerOptions = { + /** + * DOM element to use as a marker. The default is a light blue, droplet-shaped SVG marker. + */ + element?: HTMLElement; + /** + * Space-separated CSS class names to add to marker element. + */ + className?: string; + /** + * The offset in pixels as a {@link PointLike} object to apply relative to the element's center. Negatives indicate left and up. + */ + offset?: PointLike; + /** + * A string indicating the part of the Marker that should be positioned closest to the coordinate set via {@link Marker#setLngLat}. + * Options are `'center'`, `'top'`, `'bottom'`, `'left'`, `'right'`, `'top-left'`, `'top-right'`, `'bottom-left'`, and `'bottom-right'`. + * @defaultValue 'center' + * */ + anchor?: PositionAnchor; + /** + * The color to use for the default marker if options.element is not provided. The default is light blue. + * @defaultValue '#3FB1CE' + */ + color?: string; + /** + * The scale to use for the default marker if options.element is not provided. The default scale corresponds to a height of `41px` and a width of `27px`. + * @defaultValue 1 + */ + scale?: number; + /** + * A boolean indicating whether or not a marker is able to be dragged to a new position on the map. + * @defaultValue false + */ + draggable?: boolean; + /** + * The max number of pixels a user can shift the mouse pointer during a click on the marker for it to be considered a valid click (as opposed to a marker drag). The default is to inherit map's clickTolerance. + * @defaultValue 0 + */ + clickTolerance?: number; + /** + * The rotation angle of the marker in degrees, relative to its respective `rotationAlignment` setting. A positive value will rotate the marker clockwise. + * @defaultValue 0 + */ + rotation?: number; + /** + * `map` aligns the `Marker`'s rotation relative to the map, maintaining a bearing as the map rotates. `viewport` aligns the `Marker`'s rotation relative to the viewport, agnostic to map rotations. `auto` is equivalent to `viewport`. + * @defaultValue 'auto' + */ + rotationAlignment?: Alignment; + /** + * `map` aligns the `Marker` to the plane of the map. `viewport` aligns the `Marker` to the plane of the viewport. `auto` automatically matches the value of `rotationAlignment`. + * @defaultValue 'auto' + */ + pitchAlignment?: Alignment; +}; +export declare class Marker extends Evented { + _map: Map; + _anchor: PositionAnchor; + _offset: Point; + _element: HTMLElement; + _popup: Popup; + _lngLat: LngLat; + _pos: Point; + _color: string; + _scale: number; + _defaultMarker: boolean; + _draggable: boolean; + _clickTolerance: number; + _isDragging: boolean; + _state: "inactive" | "pending" | "active"; + _positionDelta: Point; + _pointerdownPos: Point; + _rotation: number; + _pitchAlignment: Alignment; + _rotationAlignment: Alignment; + _originalTabIndex: string; + _opacityTimeout: ReturnType; + /** + * @param options - the options + */ + constructor(options?: MarkerOptions); + /** + * Attaches the `Marker` to a `Map` object. + * @param map - The MapLibre GL JS map to add the marker to. + * @returns `this` + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([30.5, 50.5]) + * .addTo(map); // add the marker to the map + * ``` + */ + addTo(map: Map): this; + /** + * Removes the marker from a map + * @example + * ```ts + * let marker = new maplibregl.Marker().addTo(map); + * marker.remove(); + * ``` + * @returns `this` + */ + remove(): this; + /** + * Get the marker's geographical location. + * + * The longitude of the result may differ by a multiple of 360 degrees from the longitude previously + * set by `setLngLat` because `Marker` wraps the anchor longitude across copies of the world to keep + * the marker on screen. + * + * @returns A {@link LngLat} describing the marker's location. + * @example + * ```ts + * // Store the marker's longitude and latitude coordinates in a variable + * let lngLat = marker.getLngLat(); + * // Print the marker's longitude and latitude values in the console + * console.log('Longitude: ' + lngLat.lng + ', Latitude: ' + lngLat.lat ) + * ``` + * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/) + */ + getLngLat(): LngLat; + /** + * Set the marker's geographical position and move it. + * @param lnglat - A {@link LngLat} describing where the marker should be located. + * @returns `this` + * @example + * Create a new marker, set the longitude and latitude, and add it to the map + * ```ts + * new maplibregl.Marker() + * .setLngLat([-65.017, -16.457]) + * .addTo(map); + * ``` + * @see [Add custom icons with Markers](https://maplibre.org/maplibre-gl-js/docs/examples/custom-marker-icons/) + * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/) + */ + setLngLat(lnglat: LngLatLike): this; + /** + * Returns the `Marker`'s HTML element. + * @returns element + */ + getElement(): HTMLElement; + /** + * Binds a {@link Popup} to the {@link Marker}. + * @param popup - An instance of the {@link Popup} class. If undefined or null, any popup + * set on this {@link Marker} instance is unset. + * @returns `this` + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([0, 0]) + * .setPopup(new maplibregl.Popup().setHTML("

Hello World!

")) // add popup + * .addTo(map); + * ``` + * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/) + */ + setPopup(popup?: Popup | null): this; + _onKeyPress: (e: KeyboardEvent) => void; + _onMapClick: (e: MapMouseEvent) => void; + /** + * Returns the {@link Popup} instance that is bound to the {@link Marker}. + * @returns popup + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([0, 0]) + * .setPopup(new maplibregl.Popup().setHTML("

Hello World!

")) + * .addTo(map); + * + * console.log(marker.getPopup()); // return the popup instance + * ``` + */ + getPopup(): Popup; + /** + * Opens or closes the {@link Popup} instance that is bound to the {@link Marker}, depending on the current state of the {@link Popup}. + * @returns `this` + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([0, 0]) + * .setPopup(new maplibregl.Popup().setHTML("

Hello World!

")) + * .addTo(map); + * + * marker.togglePopup(); // toggle popup open or closed + * ``` + */ + togglePopup(): this; + _update: (e?: { + type: "move" | "moveend" | "terrain" | "render"; + }) => void; + /** + * Get the marker's offset. + * @returns The marker's screen coordinates in pixels. + */ + getOffset(): Point; + /** + * Sets the offset of the marker + * @param offset - The offset in pixels as a {@link PointLike} object to apply relative to the element's center. Negatives indicate left and up. + * @returns `this` + */ + setOffset(offset: PointLike): this; + /** + * Adds a CSS class to the marker element. + * + * @param className - on-empty string with CSS class name to add to marker element + * + * @example + * ``` + * let marker = new maplibregl.Marker() + * marker.addClassName('some-class') + * ``` + */ + addClassName(className: string): void; + /** + * Removes a CSS class from the marker element. + * + * @param className - Non-empty string with CSS class name to remove from marker element + * + * @example + * ```ts + * let marker = new maplibregl.Marker() + * marker.removeClassName('some-class') + * ``` + */ + removeClassName(className: string): void; + /** + * Add or remove the given CSS class on the marker element, depending on whether the element currently has that class. + * + * @param className - Non-empty string with CSS class name to add/remove + * + * @returns if the class was removed return false, if class was added, then return true + * + * @example + * ```ts + * let marker = new maplibregl.Marker() + * marker.toggleClassName('toggleClass') + * ``` + */ + toggleClassName(className: string): boolean; + _onMove: (e: MapMouseEvent | MapTouchEvent) => void; + _onUp: () => void; + _addDragHandler: (e: MapMouseEvent | MapTouchEvent) => void; + /** + * Sets the `draggable` property and functionality of the marker + * @param shouldBeDraggable - Turns drag functionality on/off + * @returns `this` + */ + setDraggable(shouldBeDraggable?: boolean): this; + /** + * Returns true if the marker can be dragged + * @returns True if the marker is draggable. + */ + isDraggable(): boolean; + /** + * Sets the `rotation` property of the marker. + * @param rotation - The rotation angle of the marker (clockwise, in degrees), relative to its respective {@link Marker#setRotationAlignment} setting. + * @returns `this` + */ + setRotation(rotation?: number): this; + /** + * Returns the current rotation angle of the marker (in degrees). + * @returns The current rotation angle of the marker. + */ + getRotation(): number; + /** + * Sets the `rotationAlignment` property of the marker. + * @param alignment - Sets the `rotationAlignment` property of the marker. defaults to 'auto' + * @returns `this` + */ + setRotationAlignment(alignment?: Alignment): this; + /** + * Returns the current `rotationAlignment` property of the marker. + * @returns The current rotational alignment of the marker. + */ + getRotationAlignment(): Alignment; + /** + * Sets the `pitchAlignment` property of the marker. + * @param alignment - Sets the `pitchAlignment` property of the marker. If alignment is 'auto', it will automatically match `rotationAlignment`. + * @returns `this` + */ + setPitchAlignment(alignment?: Alignment): this; + /** + * Returns the current `pitchAlignment` property of the marker. + * @returns The current pitch alignment of the marker in degrees. + */ + getPitchAlignment(): Alignment; +} +/** + * The {@link GeolocateControl} options + */ +export type GeolocateOptions = { + /** + * A Geolocation API [PositionOptions](https://developer.mozilla.org/en-US/docs/Web/API/PositionOptions) object. + * @defaultValue `{enableHighAccuracy: false, timeout: 6000}` + */ + positionOptions?: PositionOptions; + /** + * A options object to use when the map is panned and zoomed to the user's location. The default is to use a `maxZoom` of 15 to limit how far the map will zoom in for very accurate locations. + */ + fitBoundsOptions?: FitBoundsOptions; + /** + * If `true` the Geolocate Control becomes a toggle button and when active the map will receive updates to the user's location as it changes. + * @defaultValue false + */ + trackUserLocation?: boolean; + /** + * By default, if showUserLocation is `true`, a transparent circle will be drawn around the user location indicating the accuracy (95% confidence level) of the user's location. Set to `false` to disable. Always disabled when showUserLocation is `false`. + * @defaultValue true + */ + showAccuracyCircle?: boolean; + /** + * By default a dot will be shown on the map at the user's location. Set to `false` to disable. + * @defaultValue true + */ + showUserLocation?: boolean; +}; +export declare class GeolocateControl extends Evented implements IControl { + _map: Map; + options: GeolocateOptions; + _container: HTMLElement; + _dotElement: HTMLElement; + _circleElement: HTMLElement; + _geolocateButton: HTMLButtonElement; + _geolocationWatchID: number; + _timeoutId: ReturnType; + _watchState: "OFF" | "ACTIVE_LOCK" | "WAITING_ACTIVE" | "ACTIVE_ERROR" | "BACKGROUND" | "BACKGROUND_ERROR"; + _lastKnownPosition: any; + _userLocationDotMarker: Marker; + _accuracyCircleMarker: Marker; + _accuracy: number; + _setup: boolean; + constructor(options: GeolocateOptions); + /** {@inheritDoc IControl.onAdd} */ + onAdd(map: Map): HTMLElement; + /** {@inheritDoc IControl.onRemove} */ + onRemove(): void; + /** + * Check if the Geolocation API Position is outside the map's maxbounds. + * + * @param position - the Geolocation API Position + * @returns `true` if position is outside the map's maxbounds, otherwise returns `false`. + */ + _isOutOfMapMaxBounds(position: GeolocationPosition): boolean; + _setErrorState(): void; + /** + * When the Geolocation API returns a new location, update the GeolocateControl. + * + * @param position - the Geolocation API Position + */ + _onSuccess: (position: GeolocationPosition) => void; + /** + * Update the camera location to center on the current position + * + * @param position - the Geolocation API Position + */ + _updateCamera: (position: GeolocationPosition) => void; + /** + * Update the user location dot Marker to the current position + * + * @param position - the Geolocation API Position + */ + _updateMarker: (position?: GeolocationPosition | null) => void; + _updateCircleRadius(): void; + _onZoom: () => void; + _onError: (error: GeolocationPositionError) => void; + _finish: () => void; + _setupUI: (supported: boolean) => void; + /** + * Programmatically request and move the map to the user's location. + * + * @returns `false` if called before control was added to a map, otherwise returns `true`. + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * map.on('load', function() { + * geolocate.trigger(); + * }); + * ``` + */ + trigger(): boolean; + _clearWatch(): void; +} +/** + * The {@link AttributionControl} options + */ +export type AttributionOptions = { + /** + * If `true`, the attribution control will always collapse when moving the map. If `false`, + * force the expanded attribution control. The default is a responsive attribution that collapses when the user moves the map on maps less than 640 pixels wide. + * **Attribution should not be collapsed if it can comfortably fit on the map. `compact` should only be used to modify default attribution when map size makes it impossible to fit default attribution and when the automatic compact resizing for default settings are not sufficient.** + */ + compact?: boolean; + /** + * Attributions to show in addition to any other attributions. + */ + customAttribution?: string | Array; +}; +export declare class AttributionControl implements IControl { + options: AttributionOptions; + _map: Map; + _compact: boolean; + _container: HTMLElement; + _innerContainer: HTMLElement; + _compactButton: HTMLElement; + _editLink: HTMLAnchorElement; + _attribHTML: string; + styleId: string; + styleOwner: string; + /** + * @param options - the attribution options + */ + constructor(options?: AttributionOptions); + getDefaultPosition(): ControlPosition; + /** {@inheritDoc IControl.onAdd} */ + onAdd(map: Map): HTMLElement; + /** {@inheritDoc IControl.onRemove} */ + onRemove(): void; + _setElementTitle(element: HTMLElement, title: string): void; + _toggleAttribution: () => void; + _updateData: (e: MapDataEvent) => void; + _updateAttributions(): void; + _updateCompact: () => void; + _updateCompactMinimize: () => void; +} +/** + * The {@link LogoControl} options object + */ +export type LogoOptions = { + /** + * If `true`, force a compact logo. + * If `false`, force the full logo. The default is a responsive logo that collapses when the map is less than 640 pixels wide. + */ + compact?: boolean; +}; +export declare class LogoControl implements IControl { + options: LogoOptions; + _map: Map; + _compact: boolean; + _container: HTMLElement; + constructor(options?: LogoOptions); + getDefaultPosition(): ControlPosition; + /** {@inheritDoc IControl.onAdd} */ + onAdd(map: Map): HTMLElement; + /** {@inheritDoc IControl.onRemove} */ + onRemove(): void; + _updateCompact: () => void; +} +/** + * The unit type for length to use for the {@link ScaleControl} + */ +export type Unit = "imperial" | "metric" | "nautical"; +/** + * The {@link ScaleControl} options object + */ +export type ScaleOptions = { + /** + * The maximum length of the scale control in pixels. + * @defaultValue 100 + */ + maxWidth?: number; + /** + * Unit of the distance (`'imperial'`, `'metric'` or `'nautical'`). + * @defaultValue 'metric' + */ + unit?: Unit; +}; +export declare class ScaleControl implements IControl { + _map: Map; + _container: HTMLElement; + options: ScaleOptions; + constructor(options: ScaleOptions); + getDefaultPosition(): ControlPosition; + _onMove: () => void; + /** {@inheritDoc IControl.onAdd} */ + onAdd(map: Map): HTMLElement; + /** {@inheritDoc IControl.onRemove} */ + onRemove(): void; + /** + * Set the scale's unit of the distance + * + * @param unit - Unit of the distance (`'imperial'`, `'metric'` or `'nautical'`). + */ + setUnit: (unit: Unit) => void; +} +/** + * The {@link FullscreenControl} options + */ +export type FullscreenOptions = { + /** + * `container` is the [compatible DOM element](https://developer.mozilla.org/en-US/docs/Web/API/Element/requestFullScreen#Compatible_elements) which should be made full screen. By default, the map container element will be made full screen. + */ + container?: HTMLElement; +}; +export declare class FullscreenControl extends Evented implements IControl { + _map: Map; + _controlContainer: HTMLElement; + _fullscreen: boolean; + _fullscreenchange: string; + _fullscreenButton: HTMLButtonElement; + _container: HTMLElement; + _prevCooperativeGestures: boolean | GestureOptions; + constructor(options?: FullscreenOptions); + /** {@inheritDoc IControl.onAdd} */ + onAdd(map: Map): HTMLElement; + /** {@inheritDoc IControl.onRemove} */ + onRemove(): void; + _setupUI(): void; + _updateTitle(): void; + _getTitle(): any; + _isFullscreen(): boolean; + _onFullscreenChange: () => void; + _handleFullscreenChange(): void; + _onClickFullscreen: () => void; + _exitFullscreen(): void; + _requestFullscreen(): void; + _togglePseudoFullScreen(): void; +} +export declare class TerrainControl implements IControl { + options: TerrainSpecification; + _map: Map; + _container: HTMLElement; + _terrainButton: HTMLButtonElement; + constructor(options: TerrainSpecification); + /** {@inheritDoc IControl.onAdd} */ + onAdd(map: Map): HTMLElement; + /** {@inheritDoc IControl.onRemove} */ + onRemove(): void; + _toggleTerrain: () => void; + _updateTerrainIcon: () => void; +} +declare function prewarm(): void; +declare function clearPrewarmedResources(): void; +/** + * A way to indentify a feature, either by string or by number + */ +export type GeoJSONFeatureId = number | string; +/** + * The geojson source diff object + */ +export type GeoJSONSourceDiff = { + /** + * When set to `true` it will remove all features + */ + removeAll?: boolean; + /** + * An array of features IDs to remove + */ + remove?: Array; + /** + * An array of features to add + */ + add?: Array; + /** + * An array of update objects + */ + update?: Array; +}; +/** + * A geojson feature diff object + */ +export type GeoJSONFeatureDiff = { + /** + * The feature ID + */ + id: GeoJSONFeatureId; + /** + * If it's a new geometry, place it here + */ + newGeometry?: GeoJSON.Geometry; + /** + * Setting to `true` will remove all preperties + */ + removeAllProperties?: boolean; + /** + * The properties keys to remove + */ + removeProperties?: Array; + /** + * The properties to add or update along side their values + */ + addOrUpdateProperties?: Array<{ + key: string; + value: any; + }>; +}; +export type GeoJSONSourceOptions = GeoJSONSourceSpecification & { + workerOptions?: WorkerOptions; + collectResourceTiming?: boolean; +}; +export type GeoJsonSourceOptions = { + data?: GeoJSON.GeoJSON | string | undefined; + cluster?: boolean; + clusterMaxZoom?: number; + clusterRadius?: number; + clusterMinPoints?: number; + generateId?: boolean; +}; +export type WorkerOptions = { + source?: string; + cluster?: boolean; + geojsonVtOptions?: { + buffer?: number; + tolerance?: number; + extent?: number; + maxZoom?: number; + linemetrics?: boolean; + generateId?: boolean; + }; + superclusterOptions?: Options; + clusterProperties?: ClusterProperties; + fliter?: any; + promoteId?: any; + collectResourceTiming?: boolean; +}; +/** + * The cluster options to set + */ +export type SetClusterOptions = { + /** + * Whether or not to cluster + */ + cluster?: boolean; + /** + * The cluster's max zoom + */ + clusterMaxZoom?: number; + /** + * The cluster's radius + */ + clusterRadius?: number; +}; +export declare class GeoJSONSource extends Evented implements Source { + type: "geojson"; + id: string; + minzoom: number; + maxzoom: number; + tileSize: number; + attribution: string; + promoteId: PromoteIdSpecification; + isTileClipped: boolean; + reparseOverscaled: boolean; + _data: GeoJSON.GeoJSON | string | undefined; + _options: GeoJsonSourceOptions; + workerOptions: WorkerOptions; + map: Map; + actor: Actor; + _pendingLoads: number; + _collectResourceTiming: boolean; + _removed: boolean; + /** @internal */ + constructor(id: string, options: GeoJSONSourceOptions, dispatcher: Dispatcher, eventedParent: Evented); + load: () => void; + onAdd(map: Map): void; + /** + * Sets the GeoJSON data and re-renders the map. + * + * @param data - A GeoJSON data object or a URL to one. The latter is preferable in the case of large GeoJSON files. + * @returns `this` + */ + setData(data: GeoJSON.GeoJSON | string): this; + /** + * Updates the source's GeoJSON, and re-renders the map. + * + * For sources with lots of features, this method can be used to make updates more quickly. + * + * This approach requires unique IDs for every feature in the source. The IDs can either be specified on the feature, + * or by using the promoteId option to specify which property should be used as the ID. + * + * It is an error to call updateData on a source that did not have unique IDs for each of its features already. + * + * Updates are applied on a best-effort basis, updating an ID that does not exist will not result in an error. + * + * @param diff - The changes that need to be applied. + * @returns `this` + */ + updateData(diff: GeoJSONSourceDiff): this; + /** + * To disable/enable clustering on the source options + * @param options - The options to set + * @returns `this` + * @example + * ```ts + * map.getSource('some id').setClusterOptions({cluster: false}); + * map.getSource('some id').setClusterOptions({cluster: false, clusterRadius: 50, clusterMaxZoom: 14}); + * ``` + */ + setClusterOptions(options: SetClusterOptions): this; + /** + * For clustered sources, fetches the zoom at which the given cluster expands. + * + * @param clusterId - The value of the cluster's `cluster_id` property. + * @param callback - A callback to be called when the zoom value is retrieved (`(error, zoom) => { ... }`). + * @returns `this` + */ + getClusterExpansionZoom(clusterId: number, callback: Callback): this; + /** + * For clustered sources, fetches the children of the given cluster on the next zoom level (as an array of GeoJSON features). + * + * @param clusterId - The value of the cluster's `cluster_id` property. + * @param callback - A callback to be called when the features are retrieved (`(error, features) => { ... }`). + * @returns `this` + */ + getClusterChildren(clusterId: number, callback: Callback>): this; + /** + * For clustered sources, fetches the original points that belong to the cluster (as an array of GeoJSON features). + * + * @param clusterId - The value of the cluster's `cluster_id` property. + * @param limit - The maximum number of features to return. + * @param offset - The number of features to skip (e.g. for pagination). + * @param callback - A callback to be called when the features are retrieved (`(error, features) => { ... }`). + * @returns `this` + * @example + * Retrieve cluster leaves on click + * ```ts + * map.on('click', 'clusters', function(e) { + * let features = map.queryRenderedFeatures(e.point, { + * layers: ['clusters'] + * }); + * + * let clusterId = features[0].properties.cluster_id; + * let pointCount = features[0].properties.point_count; + * let clusterSource = map.getSource('clusters'); + * + * clusterSource.getClusterLeaves(clusterId, pointCount, 0, function(error, features) { + * // Print cluster leaves in the console + * console.log('Cluster leaves:', error, features); + * }) + * }); + * ``` + */ + getClusterLeaves(clusterId: number, limit: number, offset: number, callback: Callback>): this; + /** + * Responsible for invoking WorkerSource's geojson.loadData target, which + * handles loading the geojson data and preparing to serve it up as tiles, + * using geojson-vt or supercluster as appropriate. + * @param diff - the diff object + */ + _updateWorkerData(diff?: GeoJSONSourceDiff): void; + loaded(): boolean; + loadTile(tile: Tile, callback: Callback): void; + abortTile(tile: Tile): void; + unloadTile(tile: Tile): void; + onRemove(): void; + serialize: () => GeoJSONSourceSpecification; + hasTransition(): boolean; +} +export declare class TileBounds { + bounds: LngLatBounds; + minzoom: number; + maxzoom: number; + constructor(bounds: [ + number, + number, + number, + number + ], minzoom?: number | null, maxzoom?: number | null); + validateBounds(bounds: [ + number, + number, + number, + number + ]): LngLatBoundsLike; + contains(tileID: CanonicalTileID): boolean; +} +export declare class RasterTileSource extends Evented implements Source { + type: "raster" | "raster-dem"; + id: string; + minzoom: number; + maxzoom: number; + url: string; + scheme: string; + tileSize: number; + bounds: [ + number, + number, + number, + number + ]; + tileBounds: TileBounds; + roundZoom: boolean; + dispatcher: Dispatcher; + map: Map; + tiles: Array; + _loaded: boolean; + _options: RasterSourceSpecification | RasterDEMSourceSpecification; + _tileJSONRequest: Cancelable; + constructor(id: string, options: RasterSourceSpecification | RasterDEMSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented); + load(): void; + loaded(): boolean; + onAdd(map: Map): void; + onRemove(): void; + setSourceProperty(callback: Function): void; + /** + * Sets the source `tiles` property and re-renders the map. + * + * @param tiles - An array of one or more tile source URLs, as in the raster tiles spec (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) + * @returns `this` + */ + setTiles(tiles: Array): this; + serialize(): any; + hasTile(tileID: OverscaledTileID): boolean; + loadTile(tile: Tile, callback: Callback): void; + abortTile(tile: Tile, callback: Callback): void; + unloadTile(tile: Tile, callback: Callback): void; + hasTransition(): boolean; +} +export declare class RasterDEMTileSource extends RasterTileSource implements Source { + encoding: DEMEncoding; + redFactor?: number; + greenFactor?: number; + blueFactor?: number; + baseShift?: number; + constructor(id: string, options: RasterDEMSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented); + loadTile(tile: Tile, callback: Callback): void; + _getNeighboringTiles(tileID: OverscaledTileID): {}; + unloadTile(tile: Tile): void; +} +export type VectorTileSourceOptions = VectorSourceSpecification & { + collectResourceTiming?: boolean; +}; +export declare class VectorTileSource extends Evented implements Source { + type: "vector"; + id: string; + minzoom: number; + maxzoom: number; + url: string; + scheme: string; + tileSize: number; + promoteId: PromoteIdSpecification; + _options: VectorSourceSpecification; + _collectResourceTiming: boolean; + dispatcher: Dispatcher; + map: Map; + bounds: [ + number, + number, + number, + number + ]; + tiles: Array; + tileBounds: TileBounds; + reparseOverscaled: boolean; + isTileClipped: boolean; + _tileJSONRequest: Cancelable; + _loaded: boolean; + constructor(id: string, options: VectorTileSourceOptions, dispatcher: Dispatcher, eventedParent: Evented); + load: () => void; + loaded(): boolean; + hasTile(tileID: OverscaledTileID): boolean; + onAdd(map: Map): void; + setSourceProperty(callback: Function): void; + /** + * Sets the source `tiles` property and re-renders the map. + * + * @param tiles - An array of one or more tile source URLs, as in the TileJSON spec. + * @returns `this` + */ + setTiles(tiles: Array): this; + /** + * Sets the source `url` property and re-renders the map. + * + * @param url - A URL to a TileJSON resource. Supported protocols are `http:` and `https:`. + * @returns `this` + */ + setUrl(url: string): this; + onRemove(): void; + serialize: () => VectorSourceSpecification; + loadTile(tile: Tile, callback: Callback): void; + abortTile(tile: Tile): void; + unloadTile(tile: Tile): void; + hasTransition(): boolean; +} +export declare class VideoSource extends ImageSource { + options: VideoSourceSpecification; + urls: Array; + video: HTMLVideoElement; + roundZoom: boolean; + constructor(id: string, options: VideoSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented); + load: () => void; + /** + * Pauses the video. + */ + pause(): void; + /** + * Plays the video. + */ + play(): void; + /** + * Sets playback to a timestamp, in seconds. + */ + seek(seconds: number): void; + /** + * Returns the HTML `video` element. + * + * @returns The HTML `video` element. + */ + getVideo(): HTMLVideoElement; + onAdd(map: Map): void; + /** + * Sets the video's coordinates and re-renders the map. + * + * @returns `this` + */ + prepare: () => this; + serialize: () => VideoSourceSpecification; + hasTransition(): boolean; +} +export declare class MapLibreGL { + static Map: typeof Map; + static NavigationControl: typeof NavigationControl; + static GeolocateControl: typeof GeolocateControl; + static AttributionControl: typeof AttributionControl; + static LogoControl: typeof LogoControl; + static ScaleControl: typeof ScaleControl; + static FullscreenControl: typeof FullscreenControl; + static TerrainControl: typeof TerrainControl; + static Popup: typeof Popup; + static Marker: typeof Marker; + static Style: typeof Style; + static LngLat: typeof LngLat; + static LngLatBounds: typeof LngLatBounds; + static Point: typeof Point; + static MercatorCoordinate: typeof MercatorCoordinate; + static Evented: typeof Evented; + static AJAXError: typeof AJAXError; + static config: { + MAX_PARALLEL_IMAGE_REQUESTS: number; + MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME: number; + MAX_TILE_CACHE_ZOOM_LEVELS: number; + REGISTERED_PROTOCOLS: { + [x: string]: (requestParameters: RequestParameters, callback: ResponseCallback) => Cancelable; + }; + WORKER_URL: string; + }; + static CanvasSource: typeof CanvasSource; + static GeoJSONSource: typeof GeoJSONSource; + static ImageSource: typeof ImageSource; + static RasterDEMTileSource: typeof RasterDEMTileSource; + static RasterTileSource: typeof RasterTileSource; + static VectorTileSource: typeof VectorTileSource; + static VideoSource: typeof VideoSource; + /** + * Sets the map's [RTL text plugin](https://www.mapbox.com/mapbox-gl-js/plugins/#mapbox-gl-rtl-text). + * Necessary for supporting the Arabic and Hebrew languages, which are written right-to-left. + * + * @param pluginURL - URL pointing to the Mapbox RTL text plugin source. + * @param callback - Called with an error argument if there is an error. + * @param lazy - If set to `true`, mapboxgl will defer loading the plugin until rtl text is encountered, + * rtl text will then be rendered only after the plugin finishes loading. + * @example + * ```ts + * maplibregl.setRTLTextPlugin('https://unpkg.com/@mapbox/mapbox-gl-rtl-text@0.2.3/mapbox-gl-rtl-text.js'); + * ``` + * @see [Add support for right-to-left scripts](https://maplibre.org/maplibre-gl-js/docs/examples/mapbox-gl-rtl-text/) + */ + static setRTLTextPlugin: (url: string, callback: (error?: Error) => void, deferred?: boolean) => void; + /** + * Gets the map's [RTL text plugin](https://www.mapbox.com/mapbox-gl-js/plugins/#mapbox-gl-rtl-text) status. + * The status can be `unavailable` (i.e. not requested or removed), `loading`, `loaded` or `error`. + * If the status is `loaded` and the plugin is requested again, an error will be thrown. + * + * @example + * ```ts + * const pluginStatus = maplibregl.getRTLTextPluginStatus(); + * ``` + */ + static getRTLTextPluginStatus: () => string; + /** + * Initializes resources like WebWorkers that can be shared across maps to lower load + * times in some situations. `maplibregl.workerUrl` and `maplibregl.workerCount`, if being + * used, must be set before `prewarm()` is called to have an effect. + * + * By default, the lifecycle of these resources is managed automatically, and they are + * lazily initialized when a Map is first created. By invoking `prewarm()`, these + * resources will be created ahead of time, and will not be cleared when the last Map + * is removed from the page. This allows them to be re-used by new Map instances that + * are created later. They can be manually cleared by calling + * `maplibregl.clearPrewarmedResources()`. This is only necessary if your web page remains + * active but stops using maps altogether. + * + * This is primarily useful when using GL-JS maps in a single page app, wherein a user + * would navigate between various views that can cause Map instances to constantly be + * created and destroyed. + * + * @example + * ```ts + * maplibregl.prewarm() + * ``` + */ + static prewarm: typeof prewarm; + /** + * Clears up resources that have previously been created by `maplibregl.prewarm()`. + * Note that this is typically not necessary. You should only call this function + * if you expect the user of your app to not return to a Map view at any point + * in your application. + * + * @example + * ```ts + * maplibregl.clearPrewarmedResources() + * ``` + */ + static clearPrewarmedResources: typeof clearPrewarmedResources; + /** + * Returns the package version of the library + * @returns Package version of the library + */ + static get version(): string; + /** + * Gets and sets the number of web workers instantiated on a page with GL JS maps. + * By default, workerCount is 1 except for Safari browser where it is set to half the number of CPU cores (capped at 3). + * Make sure to set this property before creating any map instances for it to have effect. + * + * @returns Number of workers currently configured. + * @example + * ```ts + * maplibregl.workerCount = 2; + * ``` + */ + static get workerCount(): number; + static set workerCount(count: number); + /** + * Gets and sets the maximum number of images (raster tiles, sprites, icons) to load in parallel, + * which affects performance in raster-heavy maps. 16 by default. + * + * @returns Number of parallel requests currently configured. + * @example + * ```ts + * maplibregl.maxParallelImageRequests = 10; + * ``` + */ + static get maxParallelImageRequests(): number; + static set maxParallelImageRequests(numRequests: number); + static get workerUrl(): string; + static set workerUrl(value: string); + /** + * Sets a custom load tile function that will be called when using a source that starts with a custom url schema. + * The example below will be triggered for custom:// urls defined in the sources list in the style definitions. + * The function passed will receive the request parameters and should call the callback with the resulting request, + * for example a pbf vector tile, non-compressed, represented as ArrayBuffer. + * + * @param customProtocol - the protocol to hook, for example 'custom' + * @param loadFn - the function to use when trying to fetch a tile specified by the customProtocol + * @example + * This will fetch a file using the fetch API (this is obviously a non interesting example...) + * ```ts + * maplibregl.addProtocol('custom', (params, callback) => { + fetch(`https://${params.url.split("://")[1]}`) + .then(t => { + if (t.status == 200) { + t.arrayBuffer().then(arr => { + callback(null, arr, null, null); + }); + } else { + callback(new Error(`Tile fetch error: ${t.statusText}`)); + } + }) + .catch(e => { + callback(new Error(e)); + }); + return { cancel: () => { } }; + }); + * // the following is an example of a way to return an error when trying to load a tile + * maplibregl.addProtocol('custom2', (params, callback) => { + * callback(new Error('someErrorMessage')); + * return { cancel: () => { } }; + * }); + * ``` + */ + static addProtocol(customProtocol: string, loadFn: (requestParameters: RequestParameters, callback: ResponseCallback) => Cancelable): void; + /** + * Removes a previously added protocol + * + * @param customProtocol - the custom protocol to remove registration for + * @example + * ```ts + * maplibregl.removeProtocol('custom'); + * ``` + */ + static removeProtocol(customProtocol: string): void; +} +export type * from "@maplibre/maplibre-gl-style-spec"; + +export { + MapLibreGL as default, +}; + +export as namespace maplibregl; + +export {}; diff --git a/web/libraries/maplibre-gl/dist/maplibre-gl.js b/web/libraries/maplibre-gl/dist/maplibre-gl.js new file mode 100644 index 00000000..1ff44afe --- /dev/null +++ b/web/libraries/maplibre-gl/dist/maplibre-gl.js @@ -0,0 +1,48 @@ +/* MapLibre GL JS is licensed under the 3-Clause BSD License. Full text of license: https://github.com/maplibre/maplibre-gl-js/blob/v3.6.2/LICENSE.txt */ +(function (global, factory) { +typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() : +typeof define === 'function' && define.amd ? define(factory) : +(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.maplibregl = factory()); +})(this, (function () { 'use strict'; + +/* eslint-disable */ + +var shared, worker, maplibregl; +// define gets called three times: one for each chunk. we rely on the order +// they're imported to know which is which +function define(_, chunk) { + if (!shared) { + shared = chunk; + } else if (!worker) { + worker = chunk; + } else { + var workerBundleString = 'var sharedChunk = {}; (' + shared + ')(sharedChunk); (' + worker + ')(sharedChunk);' + + var sharedChunk = {}; + shared(sharedChunk); + maplibregl = chunk(sharedChunk); + if (typeof window !== 'undefined') { + maplibregl.workerUrl = window.URL.createObjectURL(new Blob([workerBundleString], { type: 'text/javascript' })); + } + } +} + + +define(["exports"],(function(t){"use strict";function e(t,e,r,n){return new(r||(r=Promise))((function(i,a){function s(t){try{l(n.next(t));}catch(t){a(t);}}function o(t){try{l(n.throw(t));}catch(t){a(t);}}function l(t){var e;t.done?i(t.value):(e=t.value,e instanceof r?e:new r((function(t){t(e);}))).then(s,o);}l((n=n.apply(t,e||[])).next());}))}function r(t){return t&&t.__esModule&&Object.prototype.hasOwnProperty.call(t,"default")?t.default:t}"function"==typeof SuppressedError&&SuppressedError;var n=i;function i(t,e){this.x=t,this.y=e;}i.prototype={clone:function(){return new i(this.x,this.y)},add:function(t){return this.clone()._add(t)},sub:function(t){return this.clone()._sub(t)},multByPoint:function(t){return this.clone()._multByPoint(t)},divByPoint:function(t){return this.clone()._divByPoint(t)},mult:function(t){return this.clone()._mult(t)},div:function(t){return this.clone()._div(t)},rotate:function(t){return this.clone()._rotate(t)},rotateAround:function(t,e){return this.clone()._rotateAround(t,e)},matMult:function(t){return this.clone()._matMult(t)},unit:function(){return this.clone()._unit()},perp:function(){return this.clone()._perp()},round:function(){return this.clone()._round()},mag:function(){return Math.sqrt(this.x*this.x+this.y*this.y)},equals:function(t){return this.x===t.x&&this.y===t.y},dist:function(t){return Math.sqrt(this.distSqr(t))},distSqr:function(t){var e=t.x-this.x,r=t.y-this.y;return e*e+r*r},angle:function(){return Math.atan2(this.y,this.x)},angleTo:function(t){return Math.atan2(this.y-t.y,this.x-t.x)},angleWith:function(t){return this.angleWithSep(t.x,t.y)},angleWithSep:function(t,e){return Math.atan2(this.x*e-this.y*t,this.x*t+this.y*e)},_matMult:function(t){var e=t[2]*this.x+t[3]*this.y;return this.x=t[0]*this.x+t[1]*this.y,this.y=e,this},_add:function(t){return this.x+=t.x,this.y+=t.y,this},_sub:function(t){return this.x-=t.x,this.y-=t.y,this},_mult:function(t){return this.x*=t,this.y*=t,this},_div:function(t){return this.x/=t,this.y/=t,this},_multByPoint:function(t){return this.x*=t.x,this.y*=t.y,this},_divByPoint:function(t){return this.x/=t.x,this.y/=t.y,this},_unit:function(){return this._div(this.mag()),this},_perp:function(){var t=this.y;return this.y=this.x,this.x=-t,this},_rotate:function(t){var e=Math.cos(t),r=Math.sin(t),n=r*this.x+e*this.y;return this.x=e*this.x-r*this.y,this.y=n,this},_rotateAround:function(t,e){var r=Math.cos(t),n=Math.sin(t),i=e.y+n*(this.x-e.x)+r*(this.y-e.y);return this.x=e.x+r*(this.x-e.x)-n*(this.y-e.y),this.y=i,this},_round:function(){return this.x=Math.round(this.x),this.y=Math.round(this.y),this}},i.convert=function(t){return t instanceof i?t:Array.isArray(t)?new i(t[0],t[1]):t};var a=r(n),s=o;function o(t,e,r,n){this.cx=3*t,this.bx=3*(r-t)-this.cx,this.ax=1-this.cx-this.bx,this.cy=3*e,this.by=3*(n-e)-this.cy,this.ay=1-this.cy-this.by,this.p1x=t,this.p1y=e,this.p2x=r,this.p2y=n;}o.prototype={sampleCurveX:function(t){return ((this.ax*t+this.bx)*t+this.cx)*t},sampleCurveY:function(t){return ((this.ay*t+this.by)*t+this.cy)*t},sampleCurveDerivativeX:function(t){return (3*this.ax*t+2*this.bx)*t+this.cx},solveCurveX:function(t,e){if(void 0===e&&(e=1e-6),t<0)return 0;if(t>1)return 1;for(var r=t,n=0;n<8;n++){var i=this.sampleCurveX(r)-t;if(Math.abs(i)i?s=r:o=r,r=.5*(o-s)+s;return r},solve:function(t,e){return this.sampleCurveY(this.solveCurveX(t,e))}};var l=r(s);let u,c;function h(){return null==u&&(u="undefined"!=typeof OffscreenCanvas&&new OffscreenCanvas(1,1).getContext("2d")&&"function"==typeof createImageBitmap),u}function p(){if(null==c&&(c=!1,h())){const t=5,e=new OffscreenCanvas(t,t).getContext("2d",{willReadFrequently:!0});if(e){for(let r=0;r(e.y-t.y)*(r.x-t.x)}function k(t){let e=0;for(let r,n,i=0,a=t.length,s=a-1;icancelAnimationFrame(e)}},getImageData(t,e=0){return this.getImageCanvasContext(t).getImageData(-e,-e,t.width+2*e,t.height+2*e)},getImageCanvasContext(t){const e=window.document.createElement("canvas"),r=e.getContext("2d",{willReadFrequently:!0});if(!r)throw new Error("failed to create canvas 2d context");return e.width=t.width,e.height=t.height,r.drawImage(t,0,0,t.width,t.height),r},resolveURL:t=>(E||(E=document.createElement("a")),E.href=t,E.href),hardwareConcurrency:"undefined"!=typeof navigator&&navigator.hardwareConcurrency||4,get prefersReducedMotion(){return !!matchMedia&&(null==F&&(F=matchMedia("(prefers-reduced-motion: reduce)")),F.matches)}},$={MAX_PARALLEL_IMAGE_REQUESTS:16,MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME:8,MAX_TILE_CACHE_ZOOM_LEVELS:5,REGISTERED_PROTOCOLS:{},WORKER_URL:""};class L extends Error{constructor(t,e,r,n){super(`AJAXError: ${e} (${t}): ${r}`),this.status=t,this.statusText=e,this.url=r,this.body=n;}}const D=I()?()=>self.worker&&self.worker.referrer:()=>("blob:"===window.location.protocol?window.parent:window).location.href,O=t=>$.REGISTERED_PROTOCOLS[t.substring(0,t.indexOf("://"))];function U(t,e){const r=new AbortController,n=new Request(t.url,{method:t.method||"GET",body:t.body,credentials:t.credentials,headers:t.headers,cache:t.cache,referrer:D(),signal:r.signal});let i=!1,a=!1;"json"===t.type&&n.headers.set("Accept","application/json");return a||fetch(n).then((r=>r.ok?(r=>{("arrayBuffer"===t.type||"image"===t.type?r.arrayBuffer():"json"===t.type?r.json():r.text()).then((t=>{a||(i=!0,e(null,t,r.headers.get("Cache-Control"),r.headers.get("Expires")));})).catch((t=>{a||e(new Error(t.message));}));})(r):r.blob().then((n=>e(new L(r.status,r.statusText,t.url,n)))))).catch((t=>{20!==t.code&&e(new Error(t.message));})),{cancel:()=>{a=!0,i||r.abort();}}}const R=function(t,e){if(/:\/\//.test(t.url)&&!/^https?:|^file:/.test(t.url)){if(I()&&self.worker&&self.worker.actor)return self.worker.actor.send("getResource",t,e);if(!I())return (O(t.url)||U)(t,e)}if(!(/^file:/.test(r=t.url)||/^file:/.test(D())&&!/^\w+:/.test(r))){if(fetch&&Request&&AbortController&&Object.prototype.hasOwnProperty.call(Request.prototype,"signal"))return U(t,e);if(I()&&self.worker&&self.worker.actor)return self.worker.actor.send("getResource",t,e,void 0,!0)}var r;return function(t,e){const r=new XMLHttpRequest;r.open(t.method||"GET",t.url,!0),"arrayBuffer"!==t.type&&"image"!==t.type||(r.responseType="arraybuffer");for(const e in t.headers)r.setRequestHeader(e,t.headers[e]);return "json"===t.type&&(r.responseType="text",r.setRequestHeader("Accept","application/json")),r.withCredentials="include"===t.credentials,r.onerror=()=>{e(new Error(r.statusText));},r.onload=()=>{if((r.status>=200&&r.status<300||0===r.status)&&null!==r.response){let n=r.response;if("json"===t.type)try{n=JSON.parse(r.response);}catch(t){return e(t)}e(null,n,r.getResponseHeader("Cache-Control"),r.getResponseHeader("Expires"));}else {const n=new Blob([r.response],{type:r.getResponseHeader("Content-Type")});e(new L(r.status,r.statusText,t.url,n));}},r.send(t.body),{cancel:()=>r.abort()}}(t,e)},q=function(t,e){return R(g(t,{type:"arrayBuffer"}),e)};function j(t){if(!t||t.indexOf("://")<=0||0===t.indexOf("data:image/")||0===t.indexOf("blob:"))return !0;const e=new URL(t),r=window.location;return e.protocol===r.protocol&&e.host===r.host}function N(t,e,r){r[t]&&-1!==r[t].indexOf(e)||(r[t]=r[t]||[],r[t].push(e));}function Z(t,e,r){if(r&&r[t]){const n=r[t].indexOf(e);-1!==n&&r[t].splice(n,1);}}class K{constructor(t,e={}){g(this,e),this.type=t;}}class G extends K{constructor(t,e={}){super("error",g({error:t},e));}}class J{on(t,e){return this._listeners=this._listeners||{},N(t,e,this._listeners),this}off(t,e){return Z(t,e,this._listeners),Z(t,e,this._oneTimeListeners),this}once(t,e){return e?(this._oneTimeListeners=this._oneTimeListeners||{},N(t,e,this._oneTimeListeners),this):new Promise((e=>this.once(t,e)))}fire(t,e){"string"==typeof t&&(t=new K(t,e||{}));const r=t.type;if(this.listens(r)){t.target=this;const e=this._listeners&&this._listeners[r]?this._listeners[r].slice():[];for(const r of e)r.call(this,t);const n=this._oneTimeListeners&&this._oneTimeListeners[r]?this._oneTimeListeners[r].slice():[];for(const e of n)Z(r,e,this._oneTimeListeners),e.call(this,t);const i=this._eventedParent;i&&(g(t,"function"==typeof this._eventedParentData?this._eventedParentData():this._eventedParentData),i.fire(t));}else t instanceof G&&console.error(t.error);return this}listens(t){return this._listeners&&this._listeners[t]&&this._listeners[t].length>0||this._oneTimeListeners&&this._oneTimeListeners[t]&&this._oneTimeListeners[t].length>0||this._eventedParent&&this._eventedParent.listens(t)}setEventedParent(t,e){return this._eventedParent=t,this._eventedParentData=e,this}}var X={$version:8,$root:{version:{required:!0,type:"enum",values:[8]},name:{type:"string"},metadata:{type:"*"},center:{type:"array",value:"number"},zoom:{type:"number"},bearing:{type:"number",default:0,period:360,units:"degrees"},pitch:{type:"number",default:0,units:"degrees"},light:{type:"light"},terrain:{type:"terrain"},sources:{required:!0,type:"sources"},sprite:{type:"sprite"},glyphs:{type:"string"},transition:{type:"transition"},layers:{required:!0,type:"array",value:"layer"}},sources:{"*":{type:"source"}},source:["source_vector","source_raster","source_raster_dem","source_geojson","source_video","source_image"],source_vector:{type:{required:!0,type:"enum",values:{vector:{}}},url:{type:"string"},tiles:{type:"array",value:"string"},bounds:{type:"array",value:"number",length:4,default:[-180,-85.051129,180,85.051129]},scheme:{type:"enum",values:{xyz:{},tms:{}},default:"xyz"},minzoom:{type:"number",default:0},maxzoom:{type:"number",default:22},attribution:{type:"string"},promoteId:{type:"promoteId"},volatile:{type:"boolean",default:!1},"*":{type:"*"}},source_raster:{type:{required:!0,type:"enum",values:{raster:{}}},url:{type:"string"},tiles:{type:"array",value:"string"},bounds:{type:"array",value:"number",length:4,default:[-180,-85.051129,180,85.051129]},minzoom:{type:"number",default:0},maxzoom:{type:"number",default:22},tileSize:{type:"number",default:512,units:"pixels"},scheme:{type:"enum",values:{xyz:{},tms:{}},default:"xyz"},attribution:{type:"string"},volatile:{type:"boolean",default:!1},"*":{type:"*"}},source_raster_dem:{type:{required:!0,type:"enum",values:{"raster-dem":{}}},url:{type:"string"},tiles:{type:"array",value:"string"},bounds:{type:"array",value:"number",length:4,default:[-180,-85.051129,180,85.051129]},minzoom:{type:"number",default:0},maxzoom:{type:"number",default:22},tileSize:{type:"number",default:512,units:"pixels"},attribution:{type:"string"},encoding:{type:"enum",values:{terrarium:{},mapbox:{},custom:{}},default:"mapbox"},redFactor:{type:"number",default:1},blueFactor:{type:"number",default:1},greenFactor:{type:"number",default:1},baseShift:{type:"number",default:0},volatile:{type:"boolean",default:!1},"*":{type:"*"}},source_geojson:{type:{required:!0,type:"enum",values:{geojson:{}}},data:{required:!0,type:"*"},maxzoom:{type:"number",default:18},attribution:{type:"string"},buffer:{type:"number",default:128,maximum:512,minimum:0},filter:{type:"*"},tolerance:{type:"number",default:.375},cluster:{type:"boolean",default:!1},clusterRadius:{type:"number",default:50,minimum:0},clusterMaxZoom:{type:"number"},clusterMinPoints:{type:"number"},clusterProperties:{type:"*"},lineMetrics:{type:"boolean",default:!1},generateId:{type:"boolean",default:!1},promoteId:{type:"promoteId"}},source_video:{type:{required:!0,type:"enum",values:{video:{}}},urls:{required:!0,type:"array",value:"string"},coordinates:{required:!0,type:"array",length:4,value:{type:"array",length:2,value:"number"}}},source_image:{type:{required:!0,type:"enum",values:{image:{}}},url:{required:!0,type:"string"},coordinates:{required:!0,type:"array",length:4,value:{type:"array",length:2,value:"number"}}},layer:{id:{type:"string",required:!0},type:{type:"enum",values:{fill:{},line:{},symbol:{},circle:{},heatmap:{},"fill-extrusion":{},raster:{},hillshade:{},background:{}},required:!0},metadata:{type:"*"},source:{type:"string"},"source-layer":{type:"string"},minzoom:{type:"number",minimum:0,maximum:24},maxzoom:{type:"number",minimum:0,maximum:24},filter:{type:"filter"},layout:{type:"layout"},paint:{type:"paint"}},layout:["layout_fill","layout_line","layout_circle","layout_heatmap","layout_fill-extrusion","layout_symbol","layout_raster","layout_hillshade","layout_background"],layout_background:{visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_fill:{"fill-sort-key":{type:"number",expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_circle:{"circle-sort-key":{type:"number",expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_heatmap:{visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},"layout_fill-extrusion":{visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_line:{"line-cap":{type:"enum",values:{butt:{},round:{},square:{}},default:"butt",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"line-join":{type:"enum",values:{bevel:{},round:{},miter:{}},default:"miter",expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"line-miter-limit":{type:"number",default:2,requires:[{"line-join":"miter"}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"line-round-limit":{type:"number",default:1.05,requires:[{"line-join":"round"}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"line-sort-key":{type:"number",expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_symbol:{"symbol-placement":{type:"enum",values:{point:{},line:{},"line-center":{}},default:"point",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"symbol-spacing":{type:"number",default:250,minimum:1,units:"pixels",requires:[{"symbol-placement":"line"}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"symbol-avoid-edges":{type:"boolean",default:!1,expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"symbol-sort-key":{type:"number",expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"symbol-z-order":{type:"enum",values:{auto:{},"viewport-y":{},source:{}},default:"auto",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-allow-overlap":{type:"boolean",default:!1,requires:["icon-image",{"!":"icon-overlap"}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-overlap":{type:"enum",values:{never:{},always:{},cooperative:{}},requires:["icon-image"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-ignore-placement":{type:"boolean",default:!1,requires:["icon-image"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-optional":{type:"boolean",default:!1,requires:["icon-image","text-field"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-rotation-alignment":{type:"enum",values:{map:{},viewport:{},auto:{}},default:"auto",requires:["icon-image"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-size":{type:"number",default:1,minimum:0,units:"factor of the original icon size",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"icon-text-fit":{type:"enum",values:{none:{},width:{},height:{},both:{}},default:"none",requires:["icon-image","text-field"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-text-fit-padding":{type:"array",value:"number",length:4,default:[0,0,0,0],units:"pixels",requires:["icon-image","text-field",{"icon-text-fit":["both","width","height"]}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"icon-image":{type:"resolvedImage",tokens:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"icon-rotate":{type:"number",default:0,period:360,units:"degrees",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"icon-padding":{type:"padding",default:[2],units:"pixels",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"icon-keep-upright":{type:"boolean",default:!1,requires:["icon-image",{"icon-rotation-alignment":"map"},{"symbol-placement":["line","line-center"]}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"icon-offset":{type:"array",value:"number",length:2,default:[0,0],requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"icon-anchor":{type:"enum",values:{center:{},left:{},right:{},top:{},bottom:{},"top-left":{},"top-right":{},"bottom-left":{},"bottom-right":{}},default:"center",requires:["icon-image"],expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"icon-pitch-alignment":{type:"enum",values:{map:{},viewport:{},auto:{}},default:"auto",requires:["icon-image"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-pitch-alignment":{type:"enum",values:{map:{},viewport:{},auto:{}},default:"auto",requires:["text-field"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-rotation-alignment":{type:"enum",values:{map:{},viewport:{},"viewport-glyph":{},auto:{}},default:"auto",requires:["text-field"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-field":{type:"formatted",default:"",tokens:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-font":{type:"array",value:"string",default:["Open Sans Regular","Arial Unicode MS Regular"],requires:["text-field"],expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-size":{type:"number",default:16,minimum:0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-max-width":{type:"number",default:10,minimum:0,units:"ems",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-line-height":{type:"number",default:1.2,units:"ems",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"text-letter-spacing":{type:"number",default:0,units:"ems",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-justify":{type:"enum",values:{auto:{},left:{},center:{},right:{}},default:"center",requires:["text-field"],expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-radial-offset":{type:"number",units:"ems",default:0,requires:["text-field"],"property-type":"data-driven",expression:{interpolated:!0,parameters:["zoom","feature"]}},"text-variable-anchor":{type:"array",value:"enum",values:{center:{},left:{},right:{},top:{},bottom:{},"top-left":{},"top-right":{},"bottom-left":{},"bottom-right":{}},requires:["text-field",{"symbol-placement":["point"]}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-variable-anchor-offset":{type:"variableAnchorOffsetCollection",requires:["text-field",{"symbol-placement":["point"]}],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-anchor":{type:"enum",values:{center:{},left:{},right:{},top:{},bottom:{},"top-left":{},"top-right":{},"bottom-left":{},"bottom-right":{}},default:"center",requires:["text-field",{"!":"text-variable-anchor"}],expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-max-angle":{type:"number",default:45,units:"degrees",requires:["text-field",{"symbol-placement":["line","line-center"]}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"text-writing-mode":{type:"array",value:"enum",values:{horizontal:{},vertical:{}},requires:["text-field",{"symbol-placement":["point"]}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-rotate":{type:"number",default:0,period:360,units:"degrees",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-padding":{type:"number",default:2,minimum:0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"text-keep-upright":{type:"boolean",default:!0,requires:["text-field",{"text-rotation-alignment":"map"},{"symbol-placement":["line","line-center"]}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-transform":{type:"enum",values:{none:{},uppercase:{},lowercase:{}},default:"none",requires:["text-field"],expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-offset":{type:"array",value:"number",units:"ems",length:2,default:[0,0],requires:["text-field",{"!":"text-radial-offset"}],expression:{interpolated:!0,parameters:["zoom","feature"]},"property-type":"data-driven"},"text-allow-overlap":{type:"boolean",default:!1,requires:["text-field",{"!":"text-overlap"}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-overlap":{type:"enum",values:{never:{},always:{},cooperative:{}},requires:["text-field"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-ignore-placement":{type:"boolean",default:!1,requires:["text-field"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-optional":{type:"boolean",default:!1,requires:["text-field","icon-image"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_raster:{visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},layout_hillshade:{visibility:{type:"enum",values:{visible:{},none:{}},default:"visible","property-type":"constant"}},filter:{type:"array",value:"*"},filter_operator:{type:"enum",values:{"==":{},"!=":{},">":{},">=":{},"<":{},"<=":{},in:{},"!in":{},all:{},any:{},none:{},has:{},"!has":{},within:{}}},geometry_type:{type:"enum",values:{Point:{},LineString:{},Polygon:{}}},function:{expression:{type:"expression"},stops:{type:"array",value:"function_stop"},base:{type:"number",default:1,minimum:0},property:{type:"string",default:"$zoom"},type:{type:"enum",values:{identity:{},exponential:{},interval:{},categorical:{}},default:"exponential"},colorSpace:{type:"enum",values:{rgb:{},lab:{},hcl:{}},default:"rgb"},default:{type:"*",required:!1}},function_stop:{type:"array",minimum:0,maximum:24,value:["number","color"],length:2},expression:{type:"array",value:"*",minimum:1},light:{anchor:{type:"enum",default:"viewport",values:{map:{},viewport:{}},"property-type":"data-constant",transition:!1,expression:{interpolated:!1,parameters:["zoom"]}},position:{type:"array",default:[1.15,210,30],length:3,value:"number","property-type":"data-constant",transition:!0,expression:{interpolated:!0,parameters:["zoom"]}},color:{type:"color","property-type":"data-constant",default:"#ffffff",expression:{interpolated:!0,parameters:["zoom"]},transition:!0},intensity:{type:"number","property-type":"data-constant",default:.5,minimum:0,maximum:1,expression:{interpolated:!0,parameters:["zoom"]},transition:!0}},terrain:{source:{type:"string",required:!0},exaggeration:{type:"number",minimum:0,default:1}},paint:["paint_fill","paint_line","paint_circle","paint_heatmap","paint_fill-extrusion","paint_symbol","paint_raster","paint_hillshade","paint_background"],paint_fill:{"fill-antialias":{type:"boolean",default:!0,expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"fill-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-color":{type:"color",default:"#000000",transition:!0,requires:[{"!":"fill-pattern"}],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-outline-color":{type:"color",transition:!0,requires:[{"!":"fill-pattern"},{"fill-antialias":!0}],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"fill-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["fill-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"fill-pattern":{type:"resolvedImage",transition:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"cross-faded-data-driven"}},"paint_fill-extrusion":{"fill-extrusion-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"fill-extrusion-color":{type:"color",default:"#000000",transition:!0,requires:[{"!":"fill-extrusion-pattern"}],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-extrusion-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"fill-extrusion-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["fill-extrusion-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"fill-extrusion-pattern":{type:"resolvedImage",transition:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"cross-faded-data-driven"},"fill-extrusion-height":{type:"number",default:0,minimum:0,units:"meters",transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-extrusion-base":{type:"number",default:0,minimum:0,units:"meters",transition:!0,requires:["fill-extrusion-height"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"fill-extrusion-vertical-gradient":{type:"boolean",default:!0,transition:!1,expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"}},paint_line:{"line-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-color":{type:"color",default:"#000000",transition:!0,requires:[{"!":"line-pattern"}],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"line-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["line-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"line-width":{type:"number",default:1,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-gap-width":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-offset":{type:"number",default:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-blur":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"line-dasharray":{type:"array",value:"number",minimum:0,transition:!0,units:"line widths",requires:[{"!":"line-pattern"}],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"cross-faded"},"line-pattern":{type:"resolvedImage",transition:!0,expression:{interpolated:!1,parameters:["zoom","feature"]},"property-type":"cross-faded-data-driven"},"line-gradient":{type:"color",transition:!1,requires:[{"!":"line-dasharray"},{"!":"line-pattern"},{source:"geojson",has:{lineMetrics:!0}}],expression:{interpolated:!0,parameters:["line-progress"]},"property-type":"color-ramp"}},paint_circle:{"circle-radius":{type:"number",default:5,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-color":{type:"color",default:"#000000",transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-blur":{type:"number",default:0,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"circle-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["circle-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"circle-pitch-scale":{type:"enum",values:{map:{},viewport:{}},default:"map",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"circle-pitch-alignment":{type:"enum",values:{map:{},viewport:{}},default:"viewport",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"circle-stroke-width":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-stroke-color":{type:"color",default:"#000000",transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"circle-stroke-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"}},paint_heatmap:{"heatmap-radius":{type:"number",default:30,minimum:1,transition:!0,units:"pixels",expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"heatmap-weight":{type:"number",default:1,minimum:0,transition:!1,expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"heatmap-intensity":{type:"number",default:1,minimum:0,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"heatmap-color":{type:"color",default:["interpolate",["linear"],["heatmap-density"],0,"rgba(0, 0, 255, 0)",.1,"royalblue",.3,"cyan",.5,"lime",.7,"yellow",1,"red"],transition:!1,expression:{interpolated:!0,parameters:["heatmap-density"]},"property-type":"color-ramp"},"heatmap-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"}},paint_symbol:{"icon-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-color":{type:"color",default:"#000000",transition:!0,requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-halo-color":{type:"color",default:"rgba(0, 0, 0, 0)",transition:!0,requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-halo-width":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-halo-blur":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"icon-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",requires:["icon-image"],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"icon-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["icon-image","icon-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"text-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-color":{type:"color",default:"#000000",transition:!0,overridable:!0,requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-halo-color":{type:"color",default:"rgba(0, 0, 0, 0)",transition:!0,requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-halo-width":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-halo-blur":{type:"number",default:0,minimum:0,transition:!0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom","feature","feature-state"]},"property-type":"data-driven"},"text-translate":{type:"array",value:"number",length:2,default:[0,0],transition:!0,units:"pixels",requires:["text-field"],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"text-translate-anchor":{type:"enum",values:{map:{},viewport:{}},default:"map",requires:["text-field","text-translate"],expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"}},paint_raster:{"raster-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-hue-rotate":{type:"number",default:0,period:360,transition:!0,units:"degrees",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-brightness-min":{type:"number",default:0,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-brightness-max":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-saturation":{type:"number",default:0,minimum:-1,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-contrast":{type:"number",default:0,minimum:-1,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"raster-resampling":{type:"enum",values:{linear:{},nearest:{}},default:"linear",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"raster-fade-duration":{type:"number",default:300,minimum:0,transition:!1,units:"milliseconds",expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"}},paint_hillshade:{"hillshade-illumination-direction":{type:"number",default:335,minimum:0,maximum:359,transition:!1,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-illumination-anchor":{type:"enum",values:{map:{},viewport:{}},default:"viewport",expression:{interpolated:!1,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-exaggeration":{type:"number",default:.5,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-shadow-color":{type:"color",default:"#000000",transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-highlight-color":{type:"color",default:"#FFFFFF",transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"hillshade-accent-color":{type:"color",default:"#000000",transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"}},paint_background:{"background-color":{type:"color",default:"#000000",transition:!0,requires:[{"!":"background-pattern"}],expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"},"background-pattern":{type:"resolvedImage",transition:!0,expression:{interpolated:!1,parameters:["zoom"]},"property-type":"cross-faded"},"background-opacity":{type:"number",default:1,minimum:0,maximum:1,transition:!0,expression:{interpolated:!0,parameters:["zoom"]},"property-type":"data-constant"}},transition:{duration:{type:"number",default:300,minimum:0,units:"milliseconds"},delay:{type:"number",default:0,minimum:0,units:"milliseconds"}},"property-type":{"data-driven":{type:"property-type"},"cross-faded":{type:"property-type"},"cross-faded-data-driven":{type:"property-type"},"color-ramp":{type:"property-type"},"data-constant":{type:"property-type"},constant:{type:"property-type"}},promoteId:{"*":{type:"string"}}};const Y=["type","source","source-layer","minzoom","maxzoom","filter","layout"];function H(t,e){const r={};for(const e in t)"ref"!==e&&(r[e]=t[e]);return Y.forEach((t=>{t in e&&(r[t]=e[t]);})),r}function W(t,e){if(Array.isArray(t)){if(!Array.isArray(e)||t.length!==e.length)return !1;for(let r=0;r`:"value"===t.itemType.kind?"array":`array<${e}>`}return t.kind}const kt=[ht,pt,ft,dt,yt,vt,mt,At(gt),bt,wt,_t];function It(t,e){if("error"===e.kind)return null;if("array"===t.kind){if("array"===e.kind&&(0===e.N&&"value"===e.itemType.kind||!It(t.itemType,e.itemType))&&("number"!=typeof t.N||t.N===e.N))return null}else {if(t.kind===e.kind)return null;if("value"===t.kind)for(const t of kt)if(!It(t,e))return null}return `Expected ${St(t)} but found ${St(e)} instead.`}function zt(t,e){return e.some((e=>e.kind===t.kind))}function Mt(t,e){return e.some((e=>"null"===e?null===t:"array"===e?Array.isArray(t):"object"===e?t&&!Array.isArray(t)&&"object"==typeof t:e===typeof t))}function Ct(t,e){return "array"===t.kind&&"array"===e.kind?t.itemType.kind===e.itemType.kind&&"number"==typeof t.N:t.kind===e.kind}const Bt=.96422,Pt=.82521,Vt=4/29,Et=6/29,Ft=3*Et*Et,Tt=Et*Et*Et,$t=Math.PI/180,Lt=180/Math.PI;function Dt(t){return (t%=360)<0&&(t+=360),t}function Ot([t,e,r,n]){let i,a;const s=Rt((.2225045*(t=Ut(t))+.7168786*(e=Ut(e))+.0606169*(r=Ut(r)))/1);t===e&&e===r?i=a=s:(i=Rt((.4360747*t+.3850649*e+.1430804*r)/Bt),a=Rt((.0139322*t+.0971045*e+.7141733*r)/Pt));const o=116*s-16;return [o<0?0:o,500*(i-s),200*(s-a),n]}function Ut(t){return t<=.04045?t/12.92:Math.pow((t+.055)/1.055,2.4)}function Rt(t){return t>Tt?Math.pow(t,1/3):t/Ft+Vt}function qt([t,e,r,n]){let i=(t+16)/116,a=isNaN(e)?i:i+e/500,s=isNaN(r)?i:i-r/200;return i=1*Nt(i),a=Bt*Nt(a),s=Pt*Nt(s),[jt(3.1338561*a-1.6168667*i-.4906146*s),jt(-.9787684*a+1.9161415*i+.033454*s),jt(.0719453*a-.2289914*i+1.4052427*s),n]}function jt(t){return (t=t<=.00304?12.92*t:1.055*Math.pow(t,1/2.4)-.055)<0?0:t>1?1:t}function Nt(t){return t>Et?t*t*t:Ft*(t-Vt)}function Zt(t){return parseInt(t.padEnd(2,t),16)/255}function Kt(t,e){return Gt(e?t/100:t,0,1)}function Gt(t,e,r){return Math.min(Math.max(e,t),r)}function Jt(t){return !t.some(Number.isNaN)}const Xt={aliceblue:[240,248,255],antiquewhite:[250,235,215],aqua:[0,255,255],aquamarine:[127,255,212],azure:[240,255,255],beige:[245,245,220],bisque:[255,228,196],black:[0,0,0],blanchedalmond:[255,235,205],blue:[0,0,255],blueviolet:[138,43,226],brown:[165,42,42],burlywood:[222,184,135],cadetblue:[95,158,160],chartreuse:[127,255,0],chocolate:[210,105,30],coral:[255,127,80],cornflowerblue:[100,149,237],cornsilk:[255,248,220],crimson:[220,20,60],cyan:[0,255,255],darkblue:[0,0,139],darkcyan:[0,139,139],darkgoldenrod:[184,134,11],darkgray:[169,169,169],darkgreen:[0,100,0],darkgrey:[169,169,169],darkkhaki:[189,183,107],darkmagenta:[139,0,139],darkolivegreen:[85,107,47],darkorange:[255,140,0],darkorchid:[153,50,204],darkred:[139,0,0],darksalmon:[233,150,122],darkseagreen:[143,188,143],darkslateblue:[72,61,139],darkslategray:[47,79,79],darkslategrey:[47,79,79],darkturquoise:[0,206,209],darkviolet:[148,0,211],deeppink:[255,20,147],deepskyblue:[0,191,255],dimgray:[105,105,105],dimgrey:[105,105,105],dodgerblue:[30,144,255],firebrick:[178,34,34],floralwhite:[255,250,240],forestgreen:[34,139,34],fuchsia:[255,0,255],gainsboro:[220,220,220],ghostwhite:[248,248,255],gold:[255,215,0],goldenrod:[218,165,32],gray:[128,128,128],green:[0,128,0],greenyellow:[173,255,47],grey:[128,128,128],honeydew:[240,255,240],hotpink:[255,105,180],indianred:[205,92,92],indigo:[75,0,130],ivory:[255,255,240],khaki:[240,230,140],lavender:[230,230,250],lavenderblush:[255,240,245],lawngreen:[124,252,0],lemonchiffon:[255,250,205],lightblue:[173,216,230],lightcoral:[240,128,128],lightcyan:[224,255,255],lightgoldenrodyellow:[250,250,210],lightgray:[211,211,211],lightgreen:[144,238,144],lightgrey:[211,211,211],lightpink:[255,182,193],lightsalmon:[255,160,122],lightseagreen:[32,178,170],lightskyblue:[135,206,250],lightslategray:[119,136,153],lightslategrey:[119,136,153],lightsteelblue:[176,196,222],lightyellow:[255,255,224],lime:[0,255,0],limegreen:[50,205,50],linen:[250,240,230],magenta:[255,0,255],maroon:[128,0,0],mediumaquamarine:[102,205,170],mediumblue:[0,0,205],mediumorchid:[186,85,211],mediumpurple:[147,112,219],mediumseagreen:[60,179,113],mediumslateblue:[123,104,238],mediumspringgreen:[0,250,154],mediumturquoise:[72,209,204],mediumvioletred:[199,21,133],midnightblue:[25,25,112],mintcream:[245,255,250],mistyrose:[255,228,225],moccasin:[255,228,181],navajowhite:[255,222,173],navy:[0,0,128],oldlace:[253,245,230],olive:[128,128,0],olivedrab:[107,142,35],orange:[255,165,0],orangered:[255,69,0],orchid:[218,112,214],palegoldenrod:[238,232,170],palegreen:[152,251,152],paleturquoise:[175,238,238],palevioletred:[219,112,147],papayawhip:[255,239,213],peachpuff:[255,218,185],peru:[205,133,63],pink:[255,192,203],plum:[221,160,221],powderblue:[176,224,230],purple:[128,0,128],rebeccapurple:[102,51,153],red:[255,0,0],rosybrown:[188,143,143],royalblue:[65,105,225],saddlebrown:[139,69,19],salmon:[250,128,114],sandybrown:[244,164,96],seagreen:[46,139,87],seashell:[255,245,238],sienna:[160,82,45],silver:[192,192,192],skyblue:[135,206,235],slateblue:[106,90,205],slategray:[112,128,144],slategrey:[112,128,144],snow:[255,250,250],springgreen:[0,255,127],steelblue:[70,130,180],tan:[210,180,140],teal:[0,128,128],thistle:[216,191,216],tomato:[255,99,71],turquoise:[64,224,208],violet:[238,130,238],wheat:[245,222,179],white:[255,255,255],whitesmoke:[245,245,245],yellow:[255,255,0],yellowgreen:[154,205,50]};class Yt{constructor(t,e,r,n=1,i=!0){this.r=t,this.g=e,this.b=r,this.a=n,i||(this.r*=n,this.g*=n,this.b*=n,n||this.overwriteGetter("rgb",[t,e,r,n]));}static parse(t){if(t instanceof Yt)return t;if("string"!=typeof t)return;const e=function(t){if("transparent"===(t=t.toLowerCase().trim()))return [0,0,0,0];const e=Xt[t];if(e){const[t,r,n]=e;return [t/255,r/255,n/255,1]}if(t.startsWith("#")&&/^#(?:[0-9a-f]{3,4}|[0-9a-f]{6}|[0-9a-f]{8})$/.test(t)){const e=t.length<6?1:2;let r=1;return [Zt(t.slice(r,r+=e)),Zt(t.slice(r,r+=e)),Zt(t.slice(r,r+=e)),Zt(t.slice(r,r+e)||"ff")]}if(t.startsWith("rgb")){const e=t.match(/^rgba?\(\s*([\de.+-]+)(%)?(?:\s+|\s*(,)\s*)([\de.+-]+)(%)?(?:\s+|\s*(,)\s*)([\de.+-]+)(%)?(?:\s*([,\/])\s*([\de.+-]+)(%)?)?\s*\)$/);if(e){const[t,r,n,i,a,s,o,l,u,c,h,p]=e,f=[i||" ",o||" ",c].join("");if(" "===f||" /"===f||",,"===f||",,,"===f){const t=[n,s,u].join(""),e="%%%"===t?100:""===t?255:0;if(e){const t=[Gt(+r/e,0,1),Gt(+a/e,0,1),Gt(+l/e,0,1),h?Kt(+h,p):1];if(Jt(t))return t}}return}}const r=t.match(/^hsla?\(\s*([\de.+-]+)(?:deg)?(?:\s+|\s*(,)\s*)([\de.+-]+)%(?:\s+|\s*(,)\s*)([\de.+-]+)%(?:\s*([,\/])\s*([\de.+-]+)(%)?)?\s*\)$/);if(r){const[t,e,n,i,a,s,o,l,u]=r,c=[n||" ",a||" ",o].join("");if(" "===c||" /"===c||",,"===c||",,,"===c){const t=[+e,Gt(+i,0,100),Gt(+s,0,100),l?Kt(+l,u):1];if(Jt(t))return function([t,e,r,n]){function i(n){const i=(n+t/30)%12,a=e*Math.min(r,1-r);return r-a*Math.max(-1,Math.min(i-3,9-i,1))}return t=Dt(t),e/=100,r/=100,[i(0),i(8),i(4),n]}(t)}}}(t);return e?new Yt(...e,!1):void 0}get rgb(){const{r:t,g:e,b:r,a:n}=this,i=n||1/0;return this.overwriteGetter("rgb",[t/i,e/i,r/i,n])}get hcl(){return this.overwriteGetter("hcl",function(t){const[e,r,n,i]=Ot(t),a=Math.sqrt(r*r+n*n);return [Math.round(1e4*a)?Dt(Math.atan2(n,r)*Lt):NaN,a,e,i]}(this.rgb))}get lab(){return this.overwriteGetter("lab",Ot(this.rgb))}overwriteGetter(t,e){return Object.defineProperty(this,t,{value:e}),e}toString(){const[t,e,r,n]=this.rgb;return `rgba(${[t,e,r].map((t=>Math.round(255*t))).join(",")},${n})`}}Yt.black=new Yt(0,0,0,1),Yt.white=new Yt(1,1,1,1),Yt.transparent=new Yt(0,0,0,0),Yt.red=new Yt(1,0,0,1);class Ht{constructor(t,e,r){this.sensitivity=t?e?"variant":"case":e?"accent":"base",this.locale=r,this.collator=new Intl.Collator(this.locale?this.locale:[],{sensitivity:this.sensitivity,usage:"search"});}compare(t,e){return this.collator.compare(t,e)}resolvedLocale(){return new Intl.Collator(this.locale?this.locale:[]).resolvedOptions().locale}}class Wt{constructor(t,e,r,n,i){this.text=t,this.image=e,this.scale=r,this.fontStack=n,this.textColor=i;}}class Qt{constructor(t){this.sections=t;}static fromString(t){return new Qt([new Wt(t,null,null,null,null)])}isEmpty(){return 0===this.sections.length||!this.sections.some((t=>0!==t.text.length||t.image&&0!==t.image.name.length))}static factory(t){return t instanceof Qt?t:Qt.fromString(t)}toString(){return 0===this.sections.length?"":this.sections.map((t=>t.text)).join("")}}class te{constructor(t){this.values=t.slice();}static parse(t){if(t instanceof te)return t;if("number"==typeof t)return new te([t,t,t,t]);if(Array.isArray(t)&&!(t.length<1||t.length>4)){for(const e of t)if("number"!=typeof e)return;switch(t.length){case 1:t=[t[0],t[0],t[0],t[0]];break;case 2:t=[t[0],t[1],t[0],t[1]];break;case 3:t=[t[0],t[1],t[2],t[1]];}return new te(t)}}toString(){return JSON.stringify(this.values)}}const ee=new Set(["center","left","right","top","bottom","top-left","top-right","bottom-left","bottom-right"]);class re{constructor(t){this.values=t.slice();}static parse(t){if(t instanceof re)return t;if(Array.isArray(t)&&!(t.length<1)&&t.length%2==0){for(let e=0;e=0&&t<=255&&"number"==typeof e&&e>=0&&e<=255&&"number"==typeof r&&r>=0&&r<=255?void 0===n||"number"==typeof n&&n>=0&&n<=1?null:`Invalid rgba value [${[t,e,r,n].join(", ")}]: 'a' must be between 0 and 1.`:`Invalid rgba value [${("number"==typeof n?[t,e,r,n]:[t,e,r]).join(", ")}]: 'r', 'g', and 'b' must be between 0 and 255.`}function ae(t){if(null===t||"string"==typeof t||"boolean"==typeof t||"number"==typeof t||t instanceof Yt||t instanceof Ht||t instanceof Qt||t instanceof te||t instanceof re||t instanceof ne)return !0;if(Array.isArray(t)){for(const e of t)if(!ae(e))return !1;return !0}if("object"==typeof t){for(const e in t)if(!ae(t[e]))return !1;return !0}return !1}function se(t){if(null===t)return ht;if("string"==typeof t)return ft;if("boolean"==typeof t)return dt;if("number"==typeof t)return pt;if(t instanceof Yt)return yt;if(t instanceof Ht)return xt;if(t instanceof Qt)return vt;if(t instanceof te)return bt;if(t instanceof re)return _t;if(t instanceof ne)return wt;if(Array.isArray(t)){const e=t.length;let r;for(const e of t){const t=se(e);if(r){if(r===t)continue;r=gt;break}r=t;}return At(r||gt,e)}return mt}function oe(t){const e=typeof t;return null===t?"":"string"===e||"number"===e||"boolean"===e?String(t):t instanceof Yt||t instanceof Qt||t instanceof te||t instanceof re||t instanceof ne?t.toString():JSON.stringify(t)}class le{constructor(t,e){this.type=t,this.value=e;}static parse(t,e){if(2!==t.length)return e.error(`'literal' expression requires exactly one argument, but found ${t.length-1} instead.`);if(!ae(t[1]))return e.error("invalid value");const r=t[1];let n=se(r);const i=e.expectedType;return "array"!==n.kind||0!==n.N||!i||"array"!==i.kind||"number"==typeof i.N&&0!==i.N||(n=i),new le(n,r)}evaluate(){return this.value}eachChild(){}outputDefined(){return !0}}class ue{constructor(t){this.name="ExpressionEvaluationError",this.message=t;}toJSON(){return this.message}}const ce={string:ft,number:pt,boolean:dt,object:mt};class he{constructor(t,e){this.type=t,this.args=e;}static parse(t,e){if(t.length<2)return e.error("Expected at least one argument.");let r,n=1;const i=t[0];if("array"===i){let i,a;if(t.length>2){const r=t[1];if("string"!=typeof r||!(r in ce)||"object"===r)return e.error('The item type argument of "array" must be one of string, number, boolean',1);i=ce[r],n++;}else i=gt;if(t.length>3){if(null!==t[2]&&("number"!=typeof t[2]||t[2]<0||t[2]!==Math.floor(t[2])))return e.error('The length argument to "array" must be a positive integer literal',2);a=t[2],n++;}r=At(i,a);}else {if(!ce[i])throw new Error(`Types doesn't contain name = ${i}`);r=ce[i];}const a=[];for(;nt.outputDefined()))}}const pe={"to-boolean":dt,"to-color":yt,"to-number":pt,"to-string":ft};class fe{constructor(t,e){this.type=t,this.args=e;}static parse(t,e){if(t.length<2)return e.error("Expected at least one argument.");const r=t[0];if(!pe[r])throw new Error(`Can't parse ${r} as it is not part of the known types`);if(("to-boolean"===r||"to-string"===r)&&2!==t.length)return e.error("Expected one argument.");const n=pe[r],i=[];for(let r=1;r4?`Invalid rbga value ${JSON.stringify(e)}: expected an array containing either three or four numeric values.`:ie(e[0],e[1],e[2],e[3]),!r))return new Yt(e[0]/255,e[1]/255,e[2]/255,e[3])}throw new ue(r||`Could not parse color from value '${"string"==typeof e?e:JSON.stringify(e)}'`)}case"padding":{let e;for(const r of this.args){e=r.evaluate(t);const n=te.parse(e);if(n)return n}throw new ue(`Could not parse padding from value '${"string"==typeof e?e:JSON.stringify(e)}'`)}case"variableAnchorOffsetCollection":{let e;for(const r of this.args){e=r.evaluate(t);const n=re.parse(e);if(n)return n}throw new ue(`Could not parse variableAnchorOffsetCollection from value '${"string"==typeof e?e:JSON.stringify(e)}'`)}case"number":{let e=null;for(const r of this.args){if(e=r.evaluate(t),null===e)return 0;const n=Number(e);if(!isNaN(n))return n}throw new ue(`Could not convert ${JSON.stringify(e)} to number.`)}case"formatted":return Qt.fromString(oe(this.args[0].evaluate(t)));case"resolvedImage":return ne.fromString(oe(this.args[0].evaluate(t)));default:return oe(this.args[0].evaluate(t))}}eachChild(t){this.args.forEach(t);}outputDefined(){return this.args.every((t=>t.outputDefined()))}}const de=["Unknown","Point","LineString","Polygon"];class ye{constructor(){this.globals=null,this.feature=null,this.featureState=null,this.formattedSection=null,this._parseColorCache={},this.availableImages=null,this.canonical=null;}id(){return this.feature&&"id"in this.feature?this.feature.id:null}geometryType(){return this.feature?"number"==typeof this.feature.type?de[this.feature.type]:this.feature.type:null}geometry(){return this.feature&&"geometry"in this.feature?this.feature.geometry:null}canonicalID(){return this.canonical}properties(){return this.feature&&this.feature.properties||{}}parseColor(t){let e=this._parseColorCache[t];return e||(e=this._parseColorCache[t]=Yt.parse(t)),e}}class me{constructor(t,e,r=[],n,i=new ct,a=[]){this.registry=t,this.path=r,this.key=r.map((t=>`[${t}]`)).join(""),this.scope=i,this.errors=a,this.expectedType=n,this._isConstant=e;}parse(t,e,r,n,i={}){return e?this.concat(e,r,n)._parse(t,i):this._parse(t,i)}_parse(t,e){function r(t,e,r){return "assert"===r?new he(e,[t]):"coerce"===r?new fe(e,[t]):t}if(null!==t&&"string"!=typeof t&&"boolean"!=typeof t&&"number"!=typeof t||(t=["literal",t]),Array.isArray(t)){if(0===t.length)return this.error('Expected an array with at least one element. If you wanted a literal array, use ["literal", []].');const n=t[0];if("string"!=typeof n)return this.error(`Expression name must be a string, but found ${typeof n} instead. If you wanted a literal array, use ["literal", [...]].`,0),null;const i=this.registry[n];if(i){let n=i.parse(t,this);if(!n)return null;if(this.expectedType){const t=this.expectedType,i=n.type;if("string"!==t.kind&&"number"!==t.kind&&"boolean"!==t.kind&&"object"!==t.kind&&"array"!==t.kind||"value"!==i.kind)if("color"!==t.kind&&"formatted"!==t.kind&&"resolvedImage"!==t.kind||"value"!==i.kind&&"string"!==i.kind)if("padding"!==t.kind||"value"!==i.kind&&"number"!==i.kind&&"array"!==i.kind)if("variableAnchorOffsetCollection"!==t.kind||"value"!==i.kind&&"array"!==i.kind){if(this.checkSubtype(t,i))return null}else n=r(n,t,e.typeAnnotation||"coerce");else n=r(n,t,e.typeAnnotation||"coerce");else n=r(n,t,e.typeAnnotation||"coerce");else n=r(n,t,e.typeAnnotation||"assert");}if(!(n instanceof le)&&"resolvedImage"!==n.type.kind&&this._isConstant(n)){const t=new ye;try{n=new le(n.type,n.evaluate(t));}catch(t){return this.error(t.message),null}}return n}return this.error(`Unknown expression "${n}". If you wanted a literal array, use ["literal", [...]].`,0)}return this.error(void 0===t?"'undefined' value invalid. Use null instead.":"object"==typeof t?'Bare objects invalid. Use ["literal", {...}] instead.':`Expected an array, but found ${typeof t} instead.`)}concat(t,e,r){const n="number"==typeof t?this.path.concat(t):this.path,i=r?this.scope.concat(r):this.scope;return new me(this.registry,this._isConstant,n,e||null,i,this.errors)}error(t,...e){const r=`${this.key}${e.map((t=>`[${t}]`)).join("")}`;this.errors.push(new ut(r,t));}checkSubtype(t,e){const r=It(t,e);return r&&this.error(r),r}}class ge{constructor(t,e,r){this.type=xt,this.locale=r,this.caseSensitive=t,this.diacriticSensitive=e;}static parse(t,e){if(2!==t.length)return e.error("Expected one argument.");const r=t[1];if("object"!=typeof r||Array.isArray(r))return e.error("Collator options argument must be an object.");const n=e.parse(void 0!==r["case-sensitive"]&&r["case-sensitive"],1,dt);if(!n)return null;const i=e.parse(void 0!==r["diacritic-sensitive"]&&r["diacritic-sensitive"],1,dt);if(!i)return null;let a=null;return r.locale&&(a=e.parse(r.locale,1,ft),!a)?null:new ge(n,i,a)}evaluate(t){return new Ht(this.caseSensitive.evaluate(t),this.diacriticSensitive.evaluate(t),this.locale?this.locale.evaluate(t):null)}eachChild(t){t(this.caseSensitive),t(this.diacriticSensitive),this.locale&&t(this.locale);}outputDefined(){return !1}}const xe=8192;function ve(t,e){t[0]=Math.min(t[0],e[0]),t[1]=Math.min(t[1],e[1]),t[2]=Math.max(t[2],e[0]),t[3]=Math.max(t[3],e[1]);}function be(t,e){return !(t[0]<=e[0]||t[2]>=e[2]||t[1]<=e[1]||t[3]>=e[3])}function we(t,e){const r=(180+t[0])/360,n=(180-180/Math.PI*Math.log(Math.tan(Math.PI/4+t[1]*Math.PI/360)))/360,i=Math.pow(2,e.z);return [Math.round(r*i*xe),Math.round(n*i*xe)]}function _e(t,e,r){const n=t[0]-e[0],i=t[1]-e[1],a=t[0]-r[0],s=t[1]-r[1];return n*s-a*i==0&&n*a<=0&&i*s<=0}function Ae(t,e){let r=!1;for(let s=0,o=e.length;s(n=t)[1]!=(a=o[e+1])[1]>n[1]&&n[0]<(a[0]-i[0])*(n[1]-i[1])/(a[1]-i[1])+i[0]&&(r=!r);}}var n,i,a;return r}function Se(t,e){for(let r=0;r0&&o<0||s<0&&o>0}function Ie(t,e,r){for(const u of r)for(let r=0;rr[2]){const e=.5*n;let i=t[0]-r[0]>e?-n:r[0]-t[0]>e?n:0;0===i&&(i=t[0]-r[2]>e?-n:r[2]-t[0]>e?n:0),t[0]+=i;}ve(e,t);}function Ve(t,e,r,n){const i=Math.pow(2,n.z)*xe,a=[n.x*xe,n.y*xe],s=[];for(const n of t)for(const t of n){const n=[t.x+a[0],t.y+a[1]];Pe(n,e,r,i),s.push(n);}return s}function Ee(t,e,r,n){const i=Math.pow(2,n.z)*xe,a=[n.x*xe,n.y*xe],s=[];for(const r of t){const t=[];for(const n of r){const r=[n.x+a[0],n.y+a[1]];ve(e,r),t.push(r);}s.push(t);}if(e[2]-e[0]<=i/2){(o=e)[0]=o[1]=1/0,o[2]=o[3]=-1/0;for(const t of s)for(const n of t)Pe(n,e,r,i);}var o;return s}class Fe{constructor(t,e){this.type=dt,this.geojson=t,this.geometries=e;}static parse(t,e){if(2!==t.length)return e.error(`'within' expression requires exactly one argument, but found ${t.length-1} instead.`);if(ae(t[1])){const e=t[1];if("FeatureCollection"===e.type)for(let t=0;t!Array.isArray(e)||e.length===t.length-1));let o=null;for(const[n,a]of s){o=new me(e.registry,Le,e.path,null,e.scope);const s=[];let l=!1;for(let e=1;e{return e=t,Array.isArray(e)?`(${e.map(St).join(", ")})`:`(${St(e.type)}...)`;var e;})).join(" | "),n=[];for(let r=1;r{r=e?r&&Le(t):r&&t instanceof le;})),!!r&&De(t)&&Ue(t,["zoom","heatmap-density","line-progress","accumulated","is-supported-script"])}function De(t){if(t instanceof $e){if("get"===t.name&&1===t.args.length)return !1;if("feature-state"===t.name)return !1;if("has"===t.name&&1===t.args.length)return !1;if("properties"===t.name||"geometry-type"===t.name||"id"===t.name)return !1;if(/^filter-/.test(t.name))return !1}if(t instanceof Fe)return !1;let e=!0;return t.eachChild((t=>{e&&!De(t)&&(e=!1);})),e}function Oe(t){if(t instanceof $e&&"feature-state"===t.name)return !1;let e=!0;return t.eachChild((t=>{e&&!Oe(t)&&(e=!1);})),e}function Ue(t,e){if(t instanceof $e&&e.indexOf(t.name)>=0)return !1;let r=!0;return t.eachChild((t=>{r&&!Ue(t,e)&&(r=!1);})),r}function Re(t,e){const r=t.length-1;let n,i,a=0,s=r,o=0;for(;a<=s;)if(o=Math.floor((a+s)/2),n=t[o],i=t[o+1],n<=e){if(o===r||ee))throw new ue("Input is not a number.");s=o-1;}return 0}class qe{constructor(t,e,r){this.type=t,this.input=e,this.labels=[],this.outputs=[];for(const[t,e]of r)this.labels.push(t),this.outputs.push(e);}static parse(t,e){if(t.length-1<4)return e.error(`Expected at least 4 arguments, but found only ${t.length-1}.`);if((t.length-1)%2!=0)return e.error("Expected an even number of arguments.");const r=e.parse(t[1],1,pt);if(!r)return null;const n=[];let i=null;e.expectedType&&"value"!==e.expectedType.kind&&(i=e.expectedType);for(let r=1;r=a)return e.error('Input/output pairs for "step" expressions must be arranged with input values in strictly ascending order.',o);const u=e.parse(s,l,i);if(!u)return null;i=i||u.type,n.push([a,u]);}return new qe(i,r,n)}evaluate(t){const e=this.labels,r=this.outputs;if(1===e.length)return r[0].evaluate(t);const n=this.input.evaluate(t);if(n<=e[0])return r[0].evaluate(t);const i=e.length;return n>=e[i-1]?r[i-1].evaluate(t):r[Re(e,n)].evaluate(t)}eachChild(t){t(this.input);for(const e of this.outputs)t(e);}outputDefined(){return this.outputs.every((t=>t.outputDefined()))}}function je(t,e,r){return t+r*(e-t)}function Ne(t,e,r){return t.map(((t,n)=>je(t,e[n],r)))}const Ze={number:je,color:function(t,e,r,n="rgb"){switch(n){case"rgb":{const[n,i,a,s]=Ne(t.rgb,e.rgb,r);return new Yt(n,i,a,s,!1)}case"hcl":{const[n,i,a,s]=t.hcl,[o,l,u,c]=e.hcl;let h,p;if(isNaN(n)||isNaN(o))isNaN(n)?isNaN(o)?h=NaN:(h=o,1!==a&&0!==a||(p=l)):(h=n,1!==u&&0!==u||(p=i));else {let t=o-n;o>n&&t>180?t-=360:o180&&(t+=360),h=n+r*t;}const[f,d,y,m]=function([t,e,r,n]){return t=isNaN(t)?0:t*$t,qt([r,Math.cos(t)*e,Math.sin(t)*e,n])}([h,null!=p?p:je(i,l,r),je(a,u,r),je(s,c,r)]);return new Yt(f,d,y,m,!1)}case"lab":{const[n,i,a,s]=qt(Ne(t.lab,e.lab,r));return new Yt(n,i,a,s,!1)}}},array:Ne,padding:function(t,e,r){return new te(Ne(t.values,e.values,r))},variableAnchorOffsetCollection:function(t,e,r){const n=t.values,i=e.values;if(n.length!==i.length)throw new ue(`Cannot interpolate values of different length. from: ${t.toString()}, to: ${e.toString()}`);const a=[];for(let t=0;t"number"!=typeof t||t<0||t>1)))return e.error("Cubic bezier interpolation requires four numeric arguments with values between 0 and 1.",1);n={name:"cubic-bezier",controlPoints:t};}}if(t.length-1<4)return e.error(`Expected at least 4 arguments, but found only ${t.length-1}.`);if((t.length-1)%2!=0)return e.error("Expected an even number of arguments.");if(i=e.parse(i,2,pt),!i)return null;const s=[];let o=null;"interpolate-hcl"===r||"interpolate-lab"===r?o=yt:e.expectedType&&"value"!==e.expectedType.kind&&(o=e.expectedType);for(let t=0;t=r)return e.error('Input/output pairs for "interpolate" expressions must be arranged with input values in strictly ascending order.',i);const u=e.parse(n,l,o);if(!u)return null;o=o||u.type,s.push([r,u]);}return Ct(o,pt)||Ct(o,yt)||Ct(o,bt)||Ct(o,_t)||Ct(o,At(pt))?new Ke(o,r,n,i,s):e.error(`Type ${St(o)} is not interpolatable.`)}evaluate(t){const e=this.labels,r=this.outputs;if(1===e.length)return r[0].evaluate(t);const n=this.input.evaluate(t);if(n<=e[0])return r[0].evaluate(t);const i=e.length;if(n>=e[i-1])return r[i-1].evaluate(t);const a=Re(e,n),s=Ke.interpolationFactor(this.interpolation,n,e[a],e[a+1]),o=r[a].evaluate(t),l=r[a+1].evaluate(t);switch(this.operator){case"interpolate":return Ze[this.type.kind](o,l,s);case"interpolate-hcl":return Ze.color(o,l,s,"hcl");case"interpolate-lab":return Ze.color(o,l,s,"lab")}}eachChild(t){t(this.input);for(const e of this.outputs)t(e);}outputDefined(){return this.outputs.every((t=>t.outputDefined()))}}function Ge(t,e,r,n){const i=n-r,a=t-r;return 0===i?0:1===e?a/i:(Math.pow(e,a)-1)/(Math.pow(e,i)-1)}class Je{constructor(t,e){this.type=t,this.args=e;}static parse(t,e){if(t.length<2)return e.error("Expectected at least one argument.");let r=null;const n=e.expectedType;n&&"value"!==n.kind&&(r=n);const i=[];for(const n of t.slice(1)){const t=e.parse(n,1+i.length,r,void 0,{typeAnnotation:"omit"});if(!t)return null;r=r||t.type,i.push(t);}if(!r)throw new Error("No output type");const a=n&&i.some((t=>It(n,t.type)));return new Je(a?gt:r,i)}evaluate(t){let e,r=null,n=0;for(const i of this.args)if(n++,r=i.evaluate(t),r&&r instanceof ne&&!r.available&&(e||(e=r.name),r=null,n===this.args.length&&(r=e)),null!==r)break;return r}eachChild(t){this.args.forEach(t);}outputDefined(){return this.args.every((t=>t.outputDefined()))}}class Xe{constructor(t,e){this.type=e.type,this.bindings=[].concat(t),this.result=e;}evaluate(t){return this.result.evaluate(t)}eachChild(t){for(const e of this.bindings)t(e[1]);t(this.result);}static parse(t,e){if(t.length<4)return e.error(`Expected at least 3 arguments, but found ${t.length-1} instead.`);const r=[];for(let n=1;n=r.length)throw new ue(`Array index out of bounds: ${e} > ${r.length-1}.`);if(e!==Math.floor(e))throw new ue(`Array index must be an integer, but found ${e} instead.`);return r[e]}eachChild(t){t(this.index),t(this.input);}outputDefined(){return !1}}class He{constructor(t,e){this.type=dt,this.needle=t,this.haystack=e;}static parse(t,e){if(3!==t.length)return e.error(`Expected 2 arguments, but found ${t.length-1} instead.`);const r=e.parse(t[1],1,gt),n=e.parse(t[2],2,gt);return r&&n?zt(r.type,[dt,ft,pt,ht,gt])?new He(r,n):e.error(`Expected first argument to be of type boolean, string, number or null, but found ${St(r.type)} instead`):null}evaluate(t){const e=this.needle.evaluate(t),r=this.haystack.evaluate(t);if(!r)return !1;if(!Mt(e,["boolean","string","number","null"]))throw new ue(`Expected first argument to be of type boolean, string, number or null, but found ${St(se(e))} instead.`);if(!Mt(r,["string","array"]))throw new ue(`Expected second argument to be of type array or string, but found ${St(se(r))} instead.`);return r.indexOf(e)>=0}eachChild(t){t(this.needle),t(this.haystack);}outputDefined(){return !0}}class We{constructor(t,e,r){this.type=pt,this.needle=t,this.haystack=e,this.fromIndex=r;}static parse(t,e){if(t.length<=2||t.length>=5)return e.error(`Expected 3 or 4 arguments, but found ${t.length-1} instead.`);const r=e.parse(t[1],1,gt),n=e.parse(t[2],2,gt);if(!r||!n)return null;if(!zt(r.type,[dt,ft,pt,ht,gt]))return e.error(`Expected first argument to be of type boolean, string, number or null, but found ${St(r.type)} instead`);if(4===t.length){const i=e.parse(t[3],3,pt);return i?new We(r,n,i):null}return new We(r,n)}evaluate(t){const e=this.needle.evaluate(t),r=this.haystack.evaluate(t);if(!Mt(e,["boolean","string","number","null"]))throw new ue(`Expected first argument to be of type boolean, string, number or null, but found ${St(se(e))} instead.`);if(!Mt(r,["string","array"]))throw new ue(`Expected second argument to be of type array or string, but found ${St(se(r))} instead.`);if(this.fromIndex){const n=this.fromIndex.evaluate(t);return r.indexOf(e,n)}return r.indexOf(e)}eachChild(t){t(this.needle),t(this.haystack),this.fromIndex&&t(this.fromIndex);}outputDefined(){return !1}}class Qe{constructor(t,e,r,n,i,a){this.inputType=t,this.type=e,this.input=r,this.cases=n,this.outputs=i,this.otherwise=a;}static parse(t,e){if(t.length<5)return e.error(`Expected at least 4 arguments, but found only ${t.length-1}.`);if(t.length%2!=1)return e.error("Expected an even number of arguments.");let r,n;e.expectedType&&"value"!==e.expectedType.kind&&(n=e.expectedType);const i={},a=[];for(let s=2;sNumber.MAX_SAFE_INTEGER)return u.error(`Branch labels must be integers no larger than ${Number.MAX_SAFE_INTEGER}.`);if("number"==typeof t&&Math.floor(t)!==t)return u.error("Numeric branch labels must be integer values.");if(r){if(u.checkSubtype(r,se(t)))return null}else r=se(t);if(void 0!==i[String(t)])return u.error("Branch labels must be unique.");i[String(t)]=a.length;}const c=e.parse(l,s,n);if(!c)return null;n=n||c.type,a.push(c);}const s=e.parse(t[1],1,gt);if(!s)return null;const o=e.parse(t[t.length-1],t.length-1,n);return o?"value"!==s.type.kind&&e.concat(1).checkSubtype(r,s.type)?null:new Qe(r,n,s,i,a,o):null}evaluate(t){const e=this.input.evaluate(t);return (se(e)===this.inputType&&this.outputs[this.cases[e]]||this.otherwise).evaluate(t)}eachChild(t){t(this.input),this.outputs.forEach(t),t(this.otherwise);}outputDefined(){return this.outputs.every((t=>t.outputDefined()))&&this.otherwise.outputDefined()}}class tr{constructor(t,e,r){this.type=t,this.branches=e,this.otherwise=r;}static parse(t,e){if(t.length<4)return e.error(`Expected at least 3 arguments, but found only ${t.length-1}.`);if(t.length%2!=0)return e.error("Expected an odd number of arguments.");let r;e.expectedType&&"value"!==e.expectedType.kind&&(r=e.expectedType);const n=[];for(let i=1;ie.outputDefined()))&&this.otherwise.outputDefined()}}class er{constructor(t,e,r,n){this.type=t,this.input=e,this.beginIndex=r,this.endIndex=n;}static parse(t,e){if(t.length<=2||t.length>=5)return e.error(`Expected 3 or 4 arguments, but found ${t.length-1} instead.`);const r=e.parse(t[1],1,gt),n=e.parse(t[2],2,pt);if(!r||!n)return null;if(!zt(r.type,[At(gt),ft,gt]))return e.error(`Expected first argument to be of type array or string, but found ${St(r.type)} instead`);if(4===t.length){const i=e.parse(t[3],3,pt);return i?new er(r.type,r,n,i):null}return new er(r.type,r,n)}evaluate(t){const e=this.input.evaluate(t),r=this.beginIndex.evaluate(t);if(!Mt(e,["string","array"]))throw new ue(`Expected first argument to be of type array or string, but found ${St(se(e))} instead.`);if(this.endIndex){const n=this.endIndex.evaluate(t);return e.slice(r,n)}return e.slice(r)}eachChild(t){t(this.input),t(this.beginIndex),this.endIndex&&t(this.endIndex);}outputDefined(){return !1}}function rr(t,e){return "=="===t||"!="===t?"boolean"===e.kind||"string"===e.kind||"number"===e.kind||"null"===e.kind||"value"===e.kind:"string"===e.kind||"number"===e.kind||"value"===e.kind}function nr(t,e,r,n){return 0===n.compare(e,r)}function ir(t,e,r){const n="=="!==t&&"!="!==t;return class i{constructor(t,e,r){this.type=dt,this.lhs=t,this.rhs=e,this.collator=r,this.hasUntypedArgument="value"===t.type.kind||"value"===e.type.kind;}static parse(t,e){if(3!==t.length&&4!==t.length)return e.error("Expected two or three arguments.");const r=t[0];let a=e.parse(t[1],1,gt);if(!a)return null;if(!rr(r,a.type))return e.concat(1).error(`"${r}" comparisons are not supported for type '${St(a.type)}'.`);let s=e.parse(t[2],2,gt);if(!s)return null;if(!rr(r,s.type))return e.concat(2).error(`"${r}" comparisons are not supported for type '${St(s.type)}'.`);if(a.type.kind!==s.type.kind&&"value"!==a.type.kind&&"value"!==s.type.kind)return e.error(`Cannot compare types '${St(a.type)}' and '${St(s.type)}'.`);n&&("value"===a.type.kind&&"value"!==s.type.kind?a=new he(s.type,[a]):"value"!==a.type.kind&&"value"===s.type.kind&&(s=new he(a.type,[s])));let o=null;if(4===t.length){if("string"!==a.type.kind&&"string"!==s.type.kind&&"value"!==a.type.kind&&"value"!==s.type.kind)return e.error("Cannot use collator to compare non-string types.");if(o=e.parse(t[3],3,xt),!o)return null}return new i(a,s,o)}evaluate(i){const a=this.lhs.evaluate(i),s=this.rhs.evaluate(i);if(n&&this.hasUntypedArgument){const e=se(a),r=se(s);if(e.kind!==r.kind||"string"!==e.kind&&"number"!==e.kind)throw new ue(`Expected arguments for "${t}" to be (string, string) or (number, number), but found (${e.kind}, ${r.kind}) instead.`)}if(this.collator&&!n&&this.hasUntypedArgument){const t=se(a),r=se(s);if("string"!==t.kind||"string"!==r.kind)return e(i,a,s)}return this.collator?r(i,a,s,this.collator.evaluate(i)):e(i,a,s)}eachChild(t){t(this.lhs),t(this.rhs),this.collator&&t(this.collator);}outputDefined(){return !0}}}const ar=ir("==",(function(t,e,r){return e===r}),nr),sr=ir("!=",(function(t,e,r){return e!==r}),(function(t,e,r,n){return !nr(0,e,r,n)})),or=ir("<",(function(t,e,r){return e",(function(t,e,r){return e>r}),(function(t,e,r,n){return n.compare(e,r)>0})),ur=ir("<=",(function(t,e,r){return e<=r}),(function(t,e,r,n){return n.compare(e,r)<=0})),cr=ir(">=",(function(t,e,r){return e>=r}),(function(t,e,r,n){return n.compare(e,r)>=0}));class hr{constructor(t,e,r,n,i){this.type=ft,this.number=t,this.locale=e,this.currency=r,this.minFractionDigits=n,this.maxFractionDigits=i;}static parse(t,e){if(3!==t.length)return e.error("Expected two arguments.");const r=e.parse(t[1],1,pt);if(!r)return null;const n=t[2];if("object"!=typeof n||Array.isArray(n))return e.error("NumberFormat options argument must be an object.");let i=null;if(n.locale&&(i=e.parse(n.locale,1,ft),!i))return null;let a=null;if(n.currency&&(a=e.parse(n.currency,1,ft),!a))return null;let s=null;if(n["min-fraction-digits"]&&(s=e.parse(n["min-fraction-digits"],1,pt),!s))return null;let o=null;return n["max-fraction-digits"]&&(o=e.parse(n["max-fraction-digits"],1,pt),!o)?null:new hr(r,i,a,s,o)}evaluate(t){return new Intl.NumberFormat(this.locale?this.locale.evaluate(t):[],{style:this.currency?"currency":"decimal",currency:this.currency?this.currency.evaluate(t):void 0,minimumFractionDigits:this.minFractionDigits?this.minFractionDigits.evaluate(t):void 0,maximumFractionDigits:this.maxFractionDigits?this.maxFractionDigits.evaluate(t):void 0}).format(this.number.evaluate(t))}eachChild(t){t(this.number),this.locale&&t(this.locale),this.currency&&t(this.currency),this.minFractionDigits&&t(this.minFractionDigits),this.maxFractionDigits&&t(this.maxFractionDigits);}outputDefined(){return !1}}class pr{constructor(t){this.type=vt,this.sections=t;}static parse(t,e){if(t.length<2)return e.error("Expected at least one argument.");const r=t[1];if(!Array.isArray(r)&&"object"==typeof r)return e.error("First argument must be an image or text section.");const n=[];let i=!1;for(let r=1;r<=t.length-1;++r){const a=t[r];if(i&&"object"==typeof a&&!Array.isArray(a)){i=!1;let t=null;if(a["font-scale"]&&(t=e.parse(a["font-scale"],1,pt),!t))return null;let r=null;if(a["text-font"]&&(r=e.parse(a["text-font"],1,At(ft)),!r))return null;let s=null;if(a["text-color"]&&(s=e.parse(a["text-color"],1,yt),!s))return null;const o=n[n.length-1];o.scale=t,o.font=r,o.textColor=s;}else {const a=e.parse(t[r],1,gt);if(!a)return null;const s=a.type.kind;if("string"!==s&&"value"!==s&&"null"!==s&&"resolvedImage"!==s)return e.error("Formatted text type must be 'string', 'value', 'image' or 'null'.");i=!0,n.push({content:a,scale:null,font:null,textColor:null});}}return new pr(n)}evaluate(t){return new Qt(this.sections.map((e=>{const r=e.content.evaluate(t);return se(r)===wt?new Wt("",r,null,null,null):new Wt(oe(r),null,e.scale?e.scale.evaluate(t):null,e.font?e.font.evaluate(t).join(","):null,e.textColor?e.textColor.evaluate(t):null)})))}eachChild(t){for(const e of this.sections)t(e.content),e.scale&&t(e.scale),e.font&&t(e.font),e.textColor&&t(e.textColor);}outputDefined(){return !1}}class fr{constructor(t){this.type=wt,this.input=t;}static parse(t,e){if(2!==t.length)return e.error("Expected two arguments.");const r=e.parse(t[1],1,ft);return r?new fr(r):e.error("No image name provided.")}evaluate(t){const e=this.input.evaluate(t),r=ne.fromString(e);return r&&t.availableImages&&(r.available=t.availableImages.indexOf(e)>-1),r}eachChild(t){t(this.input);}outputDefined(){return !1}}class dr{constructor(t){this.type=pt,this.input=t;}static parse(t,e){if(2!==t.length)return e.error(`Expected 1 argument, but found ${t.length-1} instead.`);const r=e.parse(t[1],1);return r?"array"!==r.type.kind&&"string"!==r.type.kind&&"value"!==r.type.kind?e.error(`Expected argument of type string or array, but found ${St(r.type)} instead.`):new dr(r):null}evaluate(t){const e=this.input.evaluate(t);if("string"==typeof e)return e.length;if(Array.isArray(e))return e.length;throw new ue(`Expected value to be of type string or array, but found ${St(se(e))} instead.`)}eachChild(t){t(this.input);}outputDefined(){return !1}}const yr={"==":ar,"!=":sr,">":lr,"<":or,">=":cr,"<=":ur,array:he,at:Ye,boolean:he,case:tr,coalesce:Je,collator:ge,format:pr,image:fr,in:He,"index-of":We,interpolate:Ke,"interpolate-hcl":Ke,"interpolate-lab":Ke,length:dr,let:Xe,literal:le,match:Qe,number:he,"number-format":hr,object:he,slice:er,step:qe,string:he,"to-boolean":fe,"to-color":fe,"to-number":fe,"to-string":fe,var:Te,within:Fe};function mr(t,[e,r,n,i]){e=e.evaluate(t),r=r.evaluate(t),n=n.evaluate(t);const a=i?i.evaluate(t):1,s=ie(e,r,n,a);if(s)throw new ue(s);return new Yt(e/255,r/255,n/255,a,!1)}function gr(t,e){return t in e}function xr(t,e){const r=e[t];return void 0===r?null:r}function vr(t){return {type:t}}function br(t){return {result:"success",value:t}}function wr(t){return {result:"error",value:t}}function _r(t){return "data-driven"===t["property-type"]||"cross-faded-data-driven"===t["property-type"]}function Ar(t){return !!t.expression&&t.expression.parameters.indexOf("zoom")>-1}function Sr(t){return !!t.expression&&t.expression.interpolated}function kr(t){return t instanceof Number?"number":t instanceof String?"string":t instanceof Boolean?"boolean":Array.isArray(t)?"array":null===t?"null":typeof t}function Ir(t){return "object"==typeof t&&null!==t&&!Array.isArray(t)}function zr(t){return t}function Mr(t,e){const r="color"===e.type,n=t.stops&&"object"==typeof t.stops[0][0],i=n||!(n||void 0!==t.property),a=t.type||(Sr(e)?"exponential":"interval");if(r||"padding"===e.type){const n=r?Yt.parse:te.parse;(t=lt({},t)).stops&&(t.stops=t.stops.map((t=>[t[0],n(t[1])]))),t.default=n(t.default?t.default:e.default);}if(t.colorSpace&&"rgb"!==(s=t.colorSpace)&&"hcl"!==s&&"lab"!==s)throw new Error(`Unknown color space: "${t.colorSpace}"`);var s;let o,l,u;if("exponential"===a)o=Vr;else if("interval"===a)o=Pr;else if("categorical"===a){o=Br,l=Object.create(null);for(const e of t.stops)l[e[0]]=e[1];u=typeof t.stops[0][0];}else {if("identity"!==a)throw new Error(`Unknown function type "${a}"`);o=Er;}if(n){const r={},n=[];for(let e=0;et[0])),evaluate:({zoom:r},n)=>Vr({stops:i,base:t.base},e,r).evaluate(r,n)}}if(i){const r="exponential"===a?{name:"exponential",base:void 0!==t.base?t.base:1}:null;return {kind:"camera",interpolationType:r,interpolationFactor:Ke.interpolationFactor.bind(void 0,r),zoomStops:t.stops.map((t=>t[0])),evaluate:({zoom:r})=>o(t,e,r,l,u)}}return {kind:"source",evaluate(r,n){const i=n&&n.properties?n.properties[t.property]:void 0;return void 0===i?Cr(t.default,e.default):o(t,e,i,l,u)}}}function Cr(t,e,r){return void 0!==t?t:void 0!==e?e:void 0!==r?r:void 0}function Br(t,e,r,n,i){return Cr(typeof r===i?n[r]:void 0,t.default,e.default)}function Pr(t,e,r){if("number"!==kr(r))return Cr(t.default,e.default);const n=t.stops.length;if(1===n)return t.stops[0][1];if(r<=t.stops[0][0])return t.stops[0][1];if(r>=t.stops[n-1][0])return t.stops[n-1][1];const i=Re(t.stops.map((t=>t[0])),r);return t.stops[i][1]}function Vr(t,e,r){const n=void 0!==t.base?t.base:1;if("number"!==kr(r))return Cr(t.default,e.default);const i=t.stops.length;if(1===i)return t.stops[0][1];if(r<=t.stops[0][0])return t.stops[0][1];if(r>=t.stops[i-1][0])return t.stops[i-1][1];const a=Re(t.stops.map((t=>t[0])),r),s=function(t,e,r,n){const i=n-r,a=t-r;return 0===i?0:1===e?a/i:(Math.pow(e,a)-1)/(Math.pow(e,i)-1)}(r,n,t.stops[a][0],t.stops[a+1][0]),o=t.stops[a][1],l=t.stops[a+1][1],u=Ze[e.type]||zr;return "function"==typeof o.evaluate?{evaluate(...e){const r=o.evaluate.apply(void 0,e),n=l.evaluate.apply(void 0,e);if(void 0!==r&&void 0!==n)return u(r,n,s,t.colorSpace)}}:u(o,l,s,t.colorSpace)}function Er(t,e,r){switch(e.type){case"color":r=Yt.parse(r);break;case"formatted":r=Qt.fromString(r.toString());break;case"resolvedImage":r=ne.fromString(r.toString());break;case"padding":r=te.parse(r);break;default:kr(r)===e.type||"enum"===e.type&&e.values[r]||(r=void 0);}return Cr(r,t.default,e.default)}$e.register(yr,{error:[{kind:"error"},[ft],(t,[e])=>{throw new ue(e.evaluate(t))}],typeof:[ft,[gt],(t,[e])=>St(se(e.evaluate(t)))],"to-rgba":[At(pt,4),[yt],(t,[e])=>{const[r,n,i,a]=e.evaluate(t).rgb;return [255*r,255*n,255*i,a]}],rgb:[yt,[pt,pt,pt],mr],rgba:[yt,[pt,pt,pt,pt],mr],has:{type:dt,overloads:[[[ft],(t,[e])=>gr(e.evaluate(t),t.properties())],[[ft,mt],(t,[e,r])=>gr(e.evaluate(t),r.evaluate(t))]]},get:{type:gt,overloads:[[[ft],(t,[e])=>xr(e.evaluate(t),t.properties())],[[ft,mt],(t,[e,r])=>xr(e.evaluate(t),r.evaluate(t))]]},"feature-state":[gt,[ft],(t,[e])=>xr(e.evaluate(t),t.featureState||{})],properties:[mt,[],t=>t.properties()],"geometry-type":[ft,[],t=>t.geometryType()],id:[gt,[],t=>t.id()],zoom:[pt,[],t=>t.globals.zoom],"heatmap-density":[pt,[],t=>t.globals.heatmapDensity||0],"line-progress":[pt,[],t=>t.globals.lineProgress||0],accumulated:[gt,[],t=>void 0===t.globals.accumulated?null:t.globals.accumulated],"+":[pt,vr(pt),(t,e)=>{let r=0;for(const n of e)r+=n.evaluate(t);return r}],"*":[pt,vr(pt),(t,e)=>{let r=1;for(const n of e)r*=n.evaluate(t);return r}],"-":{type:pt,overloads:[[[pt,pt],(t,[e,r])=>e.evaluate(t)-r.evaluate(t)],[[pt],(t,[e])=>-e.evaluate(t)]]},"/":[pt,[pt,pt],(t,[e,r])=>e.evaluate(t)/r.evaluate(t)],"%":[pt,[pt,pt],(t,[e,r])=>e.evaluate(t)%r.evaluate(t)],ln2:[pt,[],()=>Math.LN2],pi:[pt,[],()=>Math.PI],e:[pt,[],()=>Math.E],"^":[pt,[pt,pt],(t,[e,r])=>Math.pow(e.evaluate(t),r.evaluate(t))],sqrt:[pt,[pt],(t,[e])=>Math.sqrt(e.evaluate(t))],log10:[pt,[pt],(t,[e])=>Math.log(e.evaluate(t))/Math.LN10],ln:[pt,[pt],(t,[e])=>Math.log(e.evaluate(t))],log2:[pt,[pt],(t,[e])=>Math.log(e.evaluate(t))/Math.LN2],sin:[pt,[pt],(t,[e])=>Math.sin(e.evaluate(t))],cos:[pt,[pt],(t,[e])=>Math.cos(e.evaluate(t))],tan:[pt,[pt],(t,[e])=>Math.tan(e.evaluate(t))],asin:[pt,[pt],(t,[e])=>Math.asin(e.evaluate(t))],acos:[pt,[pt],(t,[e])=>Math.acos(e.evaluate(t))],atan:[pt,[pt],(t,[e])=>Math.atan(e.evaluate(t))],min:[pt,vr(pt),(t,e)=>Math.min(...e.map((e=>e.evaluate(t))))],max:[pt,vr(pt),(t,e)=>Math.max(...e.map((e=>e.evaluate(t))))],abs:[pt,[pt],(t,[e])=>Math.abs(e.evaluate(t))],round:[pt,[pt],(t,[e])=>{const r=e.evaluate(t);return r<0?-Math.round(-r):Math.round(r)}],floor:[pt,[pt],(t,[e])=>Math.floor(e.evaluate(t))],ceil:[pt,[pt],(t,[e])=>Math.ceil(e.evaluate(t))],"filter-==":[dt,[ft,gt],(t,[e,r])=>t.properties()[e.value]===r.value],"filter-id-==":[dt,[gt],(t,[e])=>t.id()===e.value],"filter-type-==":[dt,[ft],(t,[e])=>t.geometryType()===e.value],"filter-<":[dt,[ft,gt],(t,[e,r])=>{const n=t.properties()[e.value],i=r.value;return typeof n==typeof i&&n{const r=t.id(),n=e.value;return typeof r==typeof n&&r":[dt,[ft,gt],(t,[e,r])=>{const n=t.properties()[e.value],i=r.value;return typeof n==typeof i&&n>i}],"filter-id->":[dt,[gt],(t,[e])=>{const r=t.id(),n=e.value;return typeof r==typeof n&&r>n}],"filter-<=":[dt,[ft,gt],(t,[e,r])=>{const n=t.properties()[e.value],i=r.value;return typeof n==typeof i&&n<=i}],"filter-id-<=":[dt,[gt],(t,[e])=>{const r=t.id(),n=e.value;return typeof r==typeof n&&r<=n}],"filter->=":[dt,[ft,gt],(t,[e,r])=>{const n=t.properties()[e.value],i=r.value;return typeof n==typeof i&&n>=i}],"filter-id->=":[dt,[gt],(t,[e])=>{const r=t.id(),n=e.value;return typeof r==typeof n&&r>=n}],"filter-has":[dt,[gt],(t,[e])=>e.value in t.properties()],"filter-has-id":[dt,[],t=>null!==t.id()&&void 0!==t.id()],"filter-type-in":[dt,[At(ft)],(t,[e])=>e.value.indexOf(t.geometryType())>=0],"filter-id-in":[dt,[At(gt)],(t,[e])=>e.value.indexOf(t.id())>=0],"filter-in-small":[dt,[ft,At(gt)],(t,[e,r])=>r.value.indexOf(t.properties()[e.value])>=0],"filter-in-large":[dt,[ft,At(gt)],(t,[e,r])=>function(t,e,r,n){for(;r<=n;){const i=r+n>>1;if(e[i]===t)return !0;e[i]>t?n=i-1:r=i+1;}return !1}(t.properties()[e.value],r.value,0,r.value.length-1)],all:{type:dt,overloads:[[[dt,dt],(t,[e,r])=>e.evaluate(t)&&r.evaluate(t)],[vr(dt),(t,e)=>{for(const r of e)if(!r.evaluate(t))return !1;return !0}]]},any:{type:dt,overloads:[[[dt,dt],(t,[e,r])=>e.evaluate(t)||r.evaluate(t)],[vr(dt),(t,e)=>{for(const r of e)if(r.evaluate(t))return !0;return !1}]]},"!":[dt,[dt],(t,[e])=>!e.evaluate(t)],"is-supported-script":[dt,[ft],(t,[e])=>{const r=t.globals&&t.globals.isSupportedScript;return !r||r(e.evaluate(t))}],upcase:[ft,[ft],(t,[e])=>e.evaluate(t).toUpperCase()],downcase:[ft,[ft],(t,[e])=>e.evaluate(t).toLowerCase()],concat:[ft,vr(gt),(t,e)=>e.map((e=>oe(e.evaluate(t)))).join("")],"resolved-locale":[ft,[xt],(t,[e])=>e.evaluate(t).resolvedLocale()]});class Fr{constructor(t,e){var r;this.expression=t,this._warningHistory={},this._evaluator=new ye,this._defaultValue=e?"color"===(r=e).type&&Ir(r.default)?new Yt(0,0,0,0):"color"===r.type?Yt.parse(r.default)||null:"padding"===r.type?te.parse(r.default)||null:"variableAnchorOffsetCollection"===r.type?re.parse(r.default)||null:void 0===r.default?null:r.default:null,this._enumValues=e&&"enum"===e.type?e.values:null;}evaluateWithoutErrorHandling(t,e,r,n,i,a){return this._evaluator.globals=t,this._evaluator.feature=e,this._evaluator.featureState=r,this._evaluator.canonical=n,this._evaluator.availableImages=i||null,this._evaluator.formattedSection=a,this.expression.evaluate(this._evaluator)}evaluate(t,e,r,n,i,a){this._evaluator.globals=t,this._evaluator.feature=e||null,this._evaluator.featureState=r||null,this._evaluator.canonical=n,this._evaluator.availableImages=i||null,this._evaluator.formattedSection=a||null;try{const t=this.expression.evaluate(this._evaluator);if(null==t||"number"==typeof t&&t!=t)return this._defaultValue;if(this._enumValues&&!(t in this._enumValues))throw new ue(`Expected value to be one of ${Object.keys(this._enumValues).map((t=>JSON.stringify(t))).join(", ")}, but found ${JSON.stringify(t)} instead.`);return t}catch(t){return this._warningHistory[t.message]||(this._warningHistory[t.message]=!0,"undefined"!=typeof console&&console.warn(t.message)),this._defaultValue}}}function Tr(t){return Array.isArray(t)&&t.length>0&&"string"==typeof t[0]&&t[0]in yr}function $r(t,e){const r=new me(yr,Le,[],e?function(t){const e={color:yt,string:ft,number:pt,enum:ft,boolean:dt,formatted:vt,padding:bt,resolvedImage:wt,variableAnchorOffsetCollection:_t};return "array"===t.type?At(e[t.value]||gt,t.length):e[t.type]}(e):void 0),n=r.parse(t,void 0,void 0,void 0,e&&"string"===e.type?{typeAnnotation:"coerce"}:void 0);return n?br(new Fr(n,e)):wr(r.errors)}class Lr{constructor(t,e){this.kind=t,this._styleExpression=e,this.isStateDependent="constant"!==t&&!Oe(e.expression);}evaluateWithoutErrorHandling(t,e,r,n,i,a){return this._styleExpression.evaluateWithoutErrorHandling(t,e,r,n,i,a)}evaluate(t,e,r,n,i,a){return this._styleExpression.evaluate(t,e,r,n,i,a)}}class Dr{constructor(t,e,r,n){this.kind=t,this.zoomStops=r,this._styleExpression=e,this.isStateDependent="camera"!==t&&!Oe(e.expression),this.interpolationType=n;}evaluateWithoutErrorHandling(t,e,r,n,i,a){return this._styleExpression.evaluateWithoutErrorHandling(t,e,r,n,i,a)}evaluate(t,e,r,n,i,a){return this._styleExpression.evaluate(t,e,r,n,i,a)}interpolationFactor(t,e,r){return this.interpolationType?Ke.interpolationFactor(this.interpolationType,t,e,r):0}}function Or(t,e){const r=$r(t,e);if("error"===r.result)return r;const n=r.value.expression,i=De(n);if(!i&&!_r(e))return wr([new ut("","data expressions not supported")]);const a=Ue(n,["zoom"]);if(!a&&!Ar(e))return wr([new ut("","zoom expressions not supported")]);const s=Rr(n);return s||a?s instanceof ut?wr([s]):s instanceof Ke&&!Sr(e)?wr([new ut("",'"interpolate" expressions cannot be used with this property')]):br(s?new Dr(i?"camera":"composite",r.value,s.labels,s instanceof Ke?s.interpolation:void 0):new Lr(i?"constant":"source",r.value)):wr([new ut("",'"zoom" expression may only be used as input to a top-level "step" or "interpolate" expression.')])}class Ur{constructor(t,e){this._parameters=t,this._specification=e,lt(this,Mr(this._parameters,this._specification));}static deserialize(t){return new Ur(t._parameters,t._specification)}static serialize(t){return {_parameters:t._parameters,_specification:t._specification}}}function Rr(t){let e=null;if(t instanceof Xe)e=Rr(t.result);else if(t instanceof Je){for(const r of t.args)if(e=Rr(r),e)break}else (t instanceof qe||t instanceof Ke)&&t.input instanceof $e&&"zoom"===t.input.name&&(e=t);return e instanceof ut||t.eachChild((t=>{const r=Rr(t);r instanceof ut?e=r:!e&&r?e=new ut("",'"zoom" expression may only be used as input to a top-level "step" or "interpolate" expression.'):e&&r&&e!==r&&(e=new ut("",'Only one zoom-based "step" or "interpolate" subexpression may be used in an expression.'));})),e}function qr(t){if(!0===t||!1===t)return !0;if(!Array.isArray(t)||0===t.length)return !1;switch(t[0]){case"has":return t.length>=2&&"$id"!==t[1]&&"$type"!==t[1];case"in":return t.length>=3&&("string"!=typeof t[1]||Array.isArray(t[2]));case"!in":case"!has":case"none":return !1;case"==":case"!=":case">":case">=":case"<":case"<=":return 3!==t.length||Array.isArray(t[1])||Array.isArray(t[2]);case"any":case"all":for(const e of t.slice(1))if(!qr(e)&&"boolean"!=typeof e)return !1;return !0;default:return !0}}const jr={type:"boolean",default:!1,transition:!1,"property-type":"data-driven",expression:{interpolated:!1,parameters:["zoom","feature"]}};function Nr(t){if(null==t)return {filter:()=>!0,needGeometry:!1};qr(t)||(t=Gr(t));const e=$r(t,jr);if("error"===e.result)throw new Error(e.value.map((t=>`${t.key}: ${t.message}`)).join(", "));return {filter:(t,r,n)=>e.value.evaluate(t,r,{},n),needGeometry:Kr(t)}}function Zr(t,e){return te?1:0}function Kr(t){if(!Array.isArray(t))return !1;if("within"===t[0])return !0;for(let e=1;e"===e||"<="===e||">="===e?Jr(t[1],t[2],e):"any"===e?(r=t.slice(1),["any"].concat(r.map(Gr))):"all"===e?["all"].concat(t.slice(1).map(Gr)):"none"===e?["all"].concat(t.slice(1).map(Gr).map(Hr)):"in"===e?Xr(t[1],t.slice(2)):"!in"===e?Hr(Xr(t[1],t.slice(2))):"has"===e?Yr(t[1]):"!has"===e?Hr(Yr(t[1])):"within"!==e||t;var r;}function Jr(t,e,r){switch(t){case"$type":return [`filter-type-${r}`,e];case"$id":return [`filter-id-${r}`,e];default:return [`filter-${r}`,t,e]}}function Xr(t,e){if(0===e.length)return !1;switch(t){case"$type":return ["filter-type-in",["literal",e]];case"$id":return ["filter-id-in",["literal",e]];default:return e.length>200&&!e.some((t=>typeof t!=typeof e[0]))?["filter-in-large",t,["literal",e.sort(Zr)]]:["filter-in-small",t,["literal",e]]}}function Yr(t){switch(t){case"$type":return !0;case"$id":return ["filter-has-id"];default:return ["filter-has",t]}}function Hr(t){return ["!",t]}function Wr(t){const e=typeof t;if("number"===e||"boolean"===e||"string"===e||null==t)return JSON.stringify(t);if(Array.isArray(t)){let e="[";for(const r of t)e+=`${Wr(r)},`;return `${e}]`}const r=Object.keys(t).sort();let n="{";for(let e=0;en.maximum?[new ot(e,r,`${r} is greater than the maximum value ${n.maximum}`)]:[]}function on(t){const e=t.valueSpec,r=en(t.value.type);let n,i,a,s={};const o="categorical"!==r&&void 0===t.value.property,l=!o,u="array"===kr(t.value.stops)&&"array"===kr(t.value.stops[0])&&"object"===kr(t.value.stops[0][0]),c=nn({key:t.key,value:t.value,valueSpec:t.styleSpec.function,validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec,objectElementValidators:{stops:function(t){if("identity"===r)return [new ot(t.key,t.value,'identity function may not have a "stops" property')];let e=[];const n=t.value;return e=e.concat(an({key:t.key,value:n,valueSpec:t.valueSpec,validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec,arrayElementValidator:h})),"array"===kr(n)&&0===n.length&&e.push(new ot(t.key,n,"array must have at least one stop")),e},default:function(t){return t.validateSpec({key:t.key,value:t.value,valueSpec:e,validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec})}}});return "identity"===r&&o&&c.push(new ot(t.key,t.value,'missing required property "property"')),"identity"===r||t.value.stops||c.push(new ot(t.key,t.value,'missing required property "stops"')),"exponential"===r&&t.valueSpec.expression&&!Sr(t.valueSpec)&&c.push(new ot(t.key,t.value,"exponential functions not supported")),t.styleSpec.$version>=8&&(l&&!_r(t.valueSpec)?c.push(new ot(t.key,t.value,"property functions not supported")):o&&!Ar(t.valueSpec)&&c.push(new ot(t.key,t.value,"zoom functions not supported"))),"categorical"!==r&&!u||void 0!==t.value.property||c.push(new ot(t.key,t.value,'"property" property is required')),c;function h(t){let r=[];const n=t.value,o=t.key;if("array"!==kr(n))return [new ot(o,n,`array expected, ${kr(n)} found`)];if(2!==n.length)return [new ot(o,n,`array length 2 expected, length ${n.length} found`)];if(u){if("object"!==kr(n[0]))return [new ot(o,n,`object expected, ${kr(n[0])} found`)];if(void 0===n[0].zoom)return [new ot(o,n,"object stop key must have zoom")];if(void 0===n[0].value)return [new ot(o,n,"object stop key must have value")];if(a&&a>en(n[0].zoom))return [new ot(o,n[0].zoom,"stop zoom values must appear in ascending order")];en(n[0].zoom)!==a&&(a=en(n[0].zoom),i=void 0,s={}),r=r.concat(nn({key:`${o}[0]`,value:n[0],valueSpec:{zoom:{}},validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec,objectElementValidators:{zoom:sn,value:p}}));}else r=r.concat(p({key:`${o}[0]`,value:n[0],valueSpec:{},validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec},n));return Tr(rn(n[1]))?r.concat([new ot(`${o}[1]`,n[1],"expressions are not allowed in function stops.")]):r.concat(t.validateSpec({key:`${o}[1]`,value:n[1],valueSpec:e,validateSpec:t.validateSpec,style:t.style,styleSpec:t.styleSpec}))}function p(t,a){const o=kr(t.value),l=en(t.value),u=null!==t.value?t.value:a;if(n){if(o!==n)return [new ot(t.key,u,`${o} stop domain type must match previous stop domain type ${n}`)]}else n=o;if("number"!==o&&"string"!==o&&"boolean"!==o)return [new ot(t.key,u,"stop domain value must be a number, string, or boolean")];if("number"!==o&&"categorical"!==r){let n=`number expected, ${o} found`;return _r(e)&&void 0===r&&(n+='\nIf you intended to use a categorical function, specify `"type": "categorical"`.'),[new ot(t.key,u,n)]}return "categorical"!==r||"number"!==o||isFinite(l)&&Math.floor(l)===l?"categorical"!==r&&"number"===o&&void 0!==i&&lnew ot(`${t.key}${e.key}`,t.value,e.message)));const r=e.value.expression||e.value._styleExpression.expression;if("property"===t.expressionContext&&"text-font"===t.propertyKey&&!r.outputDefined())return [new ot(t.key,t.value,`Invalid data expression for "${t.propertyKey}". Output values must be contained as literals within the expression.`)];if("property"===t.expressionContext&&"layout"===t.propertyType&&!Oe(r))return [new ot(t.key,t.value,'"feature-state" data expressions are not supported with layout properties.')];if("filter"===t.expressionContext&&!Oe(r))return [new ot(t.key,t.value,'"feature-state" data expressions are not supported with filters.')];if(t.expressionContext&&0===t.expressionContext.indexOf("cluster")){if(!Ue(r,["zoom","feature-state"]))return [new ot(t.key,t.value,'"zoom" and "feature-state" expressions are not supported with cluster properties.')];if("cluster-initial"===t.expressionContext&&!De(r))return [new ot(t.key,t.value,"Feature data expressions are not supported with initial expression part of cluster properties.")]}return []}function un(t){const e=t.key,r=t.value,n=t.valueSpec,i=[];return Array.isArray(n.values)?-1===n.values.indexOf(en(r))&&i.push(new ot(e,r,`expected one of [${n.values.join(", ")}], ${JSON.stringify(r)} found`)):-1===Object.keys(n.values).indexOf(en(r))&&i.push(new ot(e,r,`expected one of [${Object.keys(n.values).join(", ")}], ${JSON.stringify(r)} found`)),i}function cn(t){return qr(rn(t.value))?ln(lt({},t,{expressionContext:"filter",valueSpec:{value:"boolean"}})):hn(t)}function hn(t){const e=t.value,r=t.key;if("array"!==kr(e))return [new ot(r,e,`array expected, ${kr(e)} found`)];const n=t.styleSpec;let i,a=[];if(e.length<1)return [new ot(r,e,"filter array must have at least 1 element")];switch(a=a.concat(un({key:`${r}[0]`,value:e[0],valueSpec:n.filter_operator,style:t.style,styleSpec:t.styleSpec})),en(e[0])){case"<":case"<=":case">":case">=":e.length>=2&&"$type"===en(e[1])&&a.push(new ot(r,e,`"$type" cannot be use with operator "${e[0]}"`));case"==":case"!=":3!==e.length&&a.push(new ot(r,e,`filter array for operator "${e[0]}" must have 3 elements`));case"in":case"!in":e.length>=2&&(i=kr(e[1]),"string"!==i&&a.push(new ot(`${r}[1]`,e[1],`string expected, ${i} found`)));for(let s=2;s{t in r&&e.push(new ot(n,r[t],`"${t}" is prohibited for ref layers`));})),i.layers.forEach((e=>{en(e.id)===o&&(t=e);})),t?t.ref?e.push(new ot(n,r.ref,"ref cannot reference another ref layer")):s=en(t.type):e.push(new ot(n,r.ref,`ref layer "${o}" not found`));}else if("background"!==s)if(r.source){const t=i.sources&&i.sources[r.source],a=t&&en(t.type);t?"vector"===a&&"raster"===s?e.push(new ot(n,r.source,`layer "${r.id}" requires a raster source`)):"raster-dem"!==a&&"hillshade"===s?e.push(new ot(n,r.source,`layer "${r.id}" requires a raster-dem source`)):"raster"===a&&"raster"!==s?e.push(new ot(n,r.source,`layer "${r.id}" requires a vector source`)):"vector"!==a||r["source-layer"]?"raster-dem"===a&&"hillshade"!==s?e.push(new ot(n,r.source,"raster-dem source can only be used with layer type 'hillshade'.")):"line"!==s||!r.paint||!r.paint["line-gradient"]||"geojson"===a&&t.lineMetrics||e.push(new ot(n,r,`layer "${r.id}" specifies a line-gradient, which requires a GeoJSON source with \`lineMetrics\` enabled.`)):e.push(new ot(n,r,`layer "${r.id}" must specify a "source-layer"`)):e.push(new ot(n,r.source,`source "${r.source}" not found`));}else e.push(new ot(n,r,'missing required property "source"'));return e=e.concat(nn({key:n,value:r,valueSpec:a.layer,style:t.style,styleSpec:t.styleSpec,validateSpec:t.validateSpec,objectElementValidators:{"*":()=>[],type:()=>t.validateSpec({key:`${n}.type`,value:r.type,valueSpec:a.layer.type,style:t.style,styleSpec:t.styleSpec,validateSpec:t.validateSpec,object:r,objectKey:"type"}),filter:cn,layout:t=>nn({layer:r,key:t.key,value:t.value,style:t.style,styleSpec:t.styleSpec,validateSpec:t.validateSpec,objectElementValidators:{"*":t=>dn(lt({layerType:s},t))}}),paint:t=>nn({layer:r,key:t.key,value:t.value,style:t.style,styleSpec:t.styleSpec,validateSpec:t.validateSpec,objectElementValidators:{"*":t=>fn(lt({layerType:s},t))}})}})),e}function mn(t){const e=t.value,r=t.key,n=kr(e);return "string"!==n?[new ot(r,e,`string expected, ${n} found`)]:[]}const gn={promoteId:function({key:t,value:e}){if("string"===kr(e))return mn({key:t,value:e});{const r=[];for(const n in e)r.push(...mn({key:`${t}.${n}`,value:e[n]}));return r}}};function xn(t){const e=t.value,r=t.key,n=t.styleSpec,i=t.style,a=t.validateSpec;if(!e.type)return [new ot(r,e,'"type" is required')];const s=en(e.type);let o;switch(s){case"vector":case"raster":return o=nn({key:r,value:e,valueSpec:n[`source_${s.replace("-","_")}`],style:t.style,styleSpec:n,objectElementValidators:gn,validateSpec:a}),o;case"raster-dem":return o=function(t){var e;const r=null!==(e=t.sourceName)&&void 0!==e?e:"",n=t.value,i=t.styleSpec,a=i.source_raster_dem,s=t.style;let o=[];const l=kr(n);if(void 0===n)return o;if("object"!==l)return o.push(new ot("source_raster_dem",n,`object expected, ${l} found`)),o;const u="custom"===en(n.encoding),c=["redFactor","greenFactor","blueFactor","baseShift"],h=t.value.encoding?`"${t.value.encoding}"`:"Default";for(const e in n)!u&&c.includes(e)?o.push(new ot(e,n[e],`In "${r}": "${e}" is only valid when "encoding" is set to "custom". ${h} encoding found`)):a[e]?o=o.concat(t.validateSpec({key:e,value:n[e],valueSpec:a[e],validateSpec:t.validateSpec,style:s,styleSpec:i})):o.push(new ot(e,n[e],`unknown property "${e}"`));return o}({sourceName:r,value:e,style:t.style,styleSpec:n,validateSpec:a}),o;case"geojson":if(o=nn({key:r,value:e,valueSpec:n.source_geojson,style:i,styleSpec:n,validateSpec:a,objectElementValidators:gn}),e.cluster)for(const t in e.clusterProperties){const[n,i]=e.clusterProperties[t],s="string"==typeof n?[n,["accumulated"],["get",t]]:n;o.push(...ln({key:`${r}.${t}.map`,value:i,validateSpec:a,expressionContext:"cluster-map"})),o.push(...ln({key:`${r}.${t}.reduce`,value:s,validateSpec:a,expressionContext:"cluster-reduce"}));}return o;case"video":return nn({key:r,value:e,valueSpec:n.source_video,style:i,validateSpec:a,styleSpec:n});case"image":return nn({key:r,value:e,valueSpec:n.source_image,style:i,validateSpec:a,styleSpec:n});case"canvas":return [new ot(r,null,"Please use runtime APIs to add canvas sources, rather than including them in stylesheets.","source.canvas")];default:return un({key:`${r}.type`,value:e.type,valueSpec:{values:["vector","raster","raster-dem","geojson","video","image"]},style:i,validateSpec:a,styleSpec:n})}}function vn(t){const e=t.value,r=t.styleSpec,n=r.light,i=t.style;let a=[];const s=kr(e);if(void 0===e)return a;if("object"!==s)return a=a.concat([new ot("light",e,`object expected, ${s} found`)]),a;for(const s in e){const o=s.match(/^(.*)-transition$/);a=a.concat(o&&n[o[1]]&&n[o[1]].transition?t.validateSpec({key:s,value:e[s],valueSpec:r.transition,validateSpec:t.validateSpec,style:i,styleSpec:r}):n[s]?t.validateSpec({key:s,value:e[s],valueSpec:n[s],validateSpec:t.validateSpec,style:i,styleSpec:r}):[new ot(s,e[s],`unknown property "${s}"`)]);}return a}function bn(t){const e=t.value,r=t.styleSpec,n=r.terrain,i=t.style;let a=[];const s=kr(e);if(void 0===e)return a;if("object"!==s)return a=a.concat([new ot("terrain",e,`object expected, ${s} found`)]),a;for(const s in e)a=a.concat(n[s]?t.validateSpec({key:s,value:e[s],valueSpec:n[s],validateSpec:t.validateSpec,style:i,styleSpec:r}):[new ot(s,e[s],`unknown property "${s}"`)]);return a}function wn(t){let e=[];const r=t.value,n=t.key;if(Array.isArray(r)){const i=[],a=[];for(const s in r)r[s].id&&i.includes(r[s].id)&&e.push(new ot(n,r,`all the sprites' ids must be unique, but ${r[s].id} is duplicated`)),i.push(r[s].id),r[s].url&&a.includes(r[s].url)&&e.push(new ot(n,r,`all the sprites' URLs must be unique, but ${r[s].url} is duplicated`)),a.push(r[s].url),e=e.concat(nn({key:`${n}[${s}]`,value:r[s],valueSpec:{id:{type:"string",required:!0},url:{type:"string",required:!0}},validateSpec:t.validateSpec}));return e}return mn({key:n,value:r})}const _n={"*":()=>[],array:an,boolean:function(t){const e=t.value,r=t.key,n=kr(e);return "boolean"!==n?[new ot(r,e,`boolean expected, ${n} found`)]:[]},number:sn,color:function(t){const e=t.key,r=t.value,n=kr(r);return "string"!==n?[new ot(e,r,`color expected, ${n} found`)]:Yt.parse(String(r))?[]:[new ot(e,r,`color expected, "${r}" found`)]},constants:tn,enum:un,filter:cn,function:on,layer:yn,object:nn,source:xn,light:vn,terrain:bn,string:mn,formatted:function(t){return 0===mn(t).length?[]:ln(t)},resolvedImage:function(t){return 0===mn(t).length?[]:ln(t)},padding:function(t){const e=t.key,r=t.value;if("array"===kr(r)){if(r.length<1||r.length>4)return [new ot(e,r,`padding requires 1 to 4 values; ${r.length} values found`)];const n={type:"number"};let i=[];for(let a=0;a[]}})),t.constants&&(r=r.concat(tn({key:"constants",value:t.constants,style:t,styleSpec:e,validateSpec:An}))),zn(r)}function In(t){return function(e){return t({...e,validateSpec:An})}}function zn(t){return [].concat(t).sort(((t,e)=>t.line-e.line))}function Mn(t){return function(...e){return zn(t.apply(this,e))}}kn.source=Mn(In(xn)),kn.sprite=Mn(In(wn)),kn.glyphs=Mn(In(Sn)),kn.light=Mn(In(vn)),kn.terrain=Mn(In(bn)),kn.layer=Mn(In(yn)),kn.filter=Mn(In(cn)),kn.paintProperty=Mn(In(fn)),kn.layoutProperty=Mn(In(dn));const Cn=kn,Bn=Cn.light,Pn=Cn.paintProperty,Vn=Cn.layoutProperty;function En(t,e){let r=!1;if(e&&e.length)for(const n of e)t.fire(new G(new Error(n.message))),r=!0;return r}class Fn{constructor(t,e,r){const n=this.cells=[];if(t instanceof ArrayBuffer){this.arrayBuffer=t;const i=new Int32Array(this.arrayBuffer);t=i[0],this.d=(e=i[1])+2*(r=i[2]);for(let t=0;t=u[l+0]&&n>=u[l+1])?(s[h]=!0,a.push(i[h])):s[h]=!1;}}}}_forEachCell(t,e,r,n,i,a,s,o){const l=this._convertToCellCoord(t),u=this._convertToCellCoord(e),c=this._convertToCellCoord(r),h=this._convertToCellCoord(n);for(let p=l;p<=c;p++)for(let l=u;l<=h;l++){const u=this.d*l+p;if((!o||o(this._convertFromCellCoord(p),this._convertFromCellCoord(l),this._convertFromCellCoord(p+1),this._convertFromCellCoord(l+1)))&&i.call(this,t,e,r,n,u,a,s,o))return}}_convertFromCellCoord(t){return (t-this.padding)/this.scale}_convertToCellCoord(t){return Math.max(0,Math.min(this.d-1,Math.floor(t*this.scale)+this.padding))}toArrayBuffer(){if(this.arrayBuffer)return this.arrayBuffer;const t=this.cells,e=3+this.cells.length+1+1;let r=0;for(let t=0;t=0)continue;const a=t[r];i[r]=Tn[n].shallow.indexOf(r)>=0?a:Dn(a,e);}t instanceof Error&&(i.message=t.message);}if(i.$name)throw new Error("$name property is reserved for worker serialization logic.");return "Object"!==n&&(i.$name=n),i}throw new Error("can't serialize object of type "+typeof t)}function On(t){if(null==t||"boolean"==typeof t||"number"==typeof t||"string"==typeof t||t instanceof Boolean||t instanceof Number||t instanceof String||t instanceof Date||t instanceof RegExp||t instanceof Blob||Ln(t)||M(t)||ArrayBuffer.isView(t)||t instanceof ImageData)return t;if(Array.isArray(t))return t.map(On);if("object"==typeof t){const e=t.$name||"Object";if(!Tn[e])throw new Error(`can't deserialize unregistered class ${e}`);const{klass:r}=Tn[e];if(!r)throw new Error(`can't deserialize unregistered class ${e}`);if(r.deserialize)return r.deserialize(t);const n=Object.create(r.prototype);for(const r of Object.keys(t)){if("$name"===r)continue;const i=t[r];n[r]=Tn[e].shallow.indexOf(r)>=0?i:On(i);}return n}throw new Error("can't deserialize object of type "+typeof t)}class Un{constructor(){this.first=!0;}update(t,e){const r=Math.floor(t);return this.first?(this.first=!1,this.lastIntegerZoom=r,this.lastIntegerZoomTime=0,this.lastZoom=t,this.lastFloorZoom=r,!0):(this.lastFloorZoom>r?(this.lastIntegerZoom=r+1,this.lastIntegerZoomTime=e):this.lastFloorZoomt>=128&&t<=255,Arabic:t=>t>=1536&&t<=1791,"Arabic Supplement":t=>t>=1872&&t<=1919,"Arabic Extended-A":t=>t>=2208&&t<=2303,"Hangul Jamo":t=>t>=4352&&t<=4607,"Unified Canadian Aboriginal Syllabics":t=>t>=5120&&t<=5759,Khmer:t=>t>=6016&&t<=6143,"Unified Canadian Aboriginal Syllabics Extended":t=>t>=6320&&t<=6399,"General Punctuation":t=>t>=8192&&t<=8303,"Letterlike Symbols":t=>t>=8448&&t<=8527,"Number Forms":t=>t>=8528&&t<=8591,"Miscellaneous Technical":t=>t>=8960&&t<=9215,"Control Pictures":t=>t>=9216&&t<=9279,"Optical Character Recognition":t=>t>=9280&&t<=9311,"Enclosed Alphanumerics":t=>t>=9312&&t<=9471,"Geometric Shapes":t=>t>=9632&&t<=9727,"Miscellaneous Symbols":t=>t>=9728&&t<=9983,"Miscellaneous Symbols and Arrows":t=>t>=11008&&t<=11263,"CJK Radicals Supplement":t=>t>=11904&&t<=12031,"Kangxi Radicals":t=>t>=12032&&t<=12255,"Ideographic Description Characters":t=>t>=12272&&t<=12287,"CJK Symbols and Punctuation":t=>t>=12288&&t<=12351,Hiragana:t=>t>=12352&&t<=12447,Katakana:t=>t>=12448&&t<=12543,Bopomofo:t=>t>=12544&&t<=12591,"Hangul Compatibility Jamo":t=>t>=12592&&t<=12687,Kanbun:t=>t>=12688&&t<=12703,"Bopomofo Extended":t=>t>=12704&&t<=12735,"CJK Strokes":t=>t>=12736&&t<=12783,"Katakana Phonetic Extensions":t=>t>=12784&&t<=12799,"Enclosed CJK Letters and Months":t=>t>=12800&&t<=13055,"CJK Compatibility":t=>t>=13056&&t<=13311,"CJK Unified Ideographs Extension A":t=>t>=13312&&t<=19903,"Yijing Hexagram Symbols":t=>t>=19904&&t<=19967,"CJK Unified Ideographs":t=>t>=19968&&t<=40959,"Yi Syllables":t=>t>=40960&&t<=42127,"Yi Radicals":t=>t>=42128&&t<=42191,"Hangul Jamo Extended-A":t=>t>=43360&&t<=43391,"Hangul Syllables":t=>t>=44032&&t<=55215,"Hangul Jamo Extended-B":t=>t>=55216&&t<=55295,"Private Use Area":t=>t>=57344&&t<=63743,"CJK Compatibility Ideographs":t=>t>=63744&&t<=64255,"Arabic Presentation Forms-A":t=>t>=64336&&t<=65023,"Vertical Forms":t=>t>=65040&&t<=65055,"CJK Compatibility Forms":t=>t>=65072&&t<=65103,"Small Form Variants":t=>t>=65104&&t<=65135,"Arabic Presentation Forms-B":t=>t>=65136&&t<=65279,"Halfwidth and Fullwidth Forms":t=>t>=65280&&t<=65519};function qn(t){for(const e of t)if(Zn(e.charCodeAt(0)))return !0;return !1}function jn(t){for(const e of t)if(!Nn(e.charCodeAt(0)))return !1;return !0}function Nn(t){return !(Rn.Arabic(t)||Rn["Arabic Supplement"](t)||Rn["Arabic Extended-A"](t)||Rn["Arabic Presentation Forms-A"](t)||Rn["Arabic Presentation Forms-B"](t))}function Zn(t){return !(746!==t&&747!==t&&(t<4352||!(Rn["Bopomofo Extended"](t)||Rn.Bopomofo(t)||Rn["CJK Compatibility Forms"](t)&&!(t>=65097&&t<=65103)||Rn["CJK Compatibility Ideographs"](t)||Rn["CJK Compatibility"](t)||Rn["CJK Radicals Supplement"](t)||Rn["CJK Strokes"](t)||!(!Rn["CJK Symbols and Punctuation"](t)||t>=12296&&t<=12305||t>=12308&&t<=12319||12336===t)||Rn["CJK Unified Ideographs Extension A"](t)||Rn["CJK Unified Ideographs"](t)||Rn["Enclosed CJK Letters and Months"](t)||Rn["Hangul Compatibility Jamo"](t)||Rn["Hangul Jamo Extended-A"](t)||Rn["Hangul Jamo Extended-B"](t)||Rn["Hangul Jamo"](t)||Rn["Hangul Syllables"](t)||Rn.Hiragana(t)||Rn["Ideographic Description Characters"](t)||Rn.Kanbun(t)||Rn["Kangxi Radicals"](t)||Rn["Katakana Phonetic Extensions"](t)||Rn.Katakana(t)&&12540!==t||!(!Rn["Halfwidth and Fullwidth Forms"](t)||65288===t||65289===t||65293===t||t>=65306&&t<=65310||65339===t||65341===t||65343===t||t>=65371&&t<=65503||65507===t||t>=65512&&t<=65519)||!(!Rn["Small Form Variants"](t)||t>=65112&&t<=65118||t>=65123&&t<=65126)||Rn["Unified Canadian Aboriginal Syllabics"](t)||Rn["Unified Canadian Aboriginal Syllabics Extended"](t)||Rn["Vertical Forms"](t)||Rn["Yijing Hexagram Symbols"](t)||Rn["Yi Syllables"](t)||Rn["Yi Radicals"](t))))}function Kn(t){return !(Zn(t)||function(t){return !!(Rn["Latin-1 Supplement"](t)&&(167===t||169===t||174===t||177===t||188===t||189===t||190===t||215===t||247===t)||Rn["General Punctuation"](t)&&(8214===t||8224===t||8225===t||8240===t||8241===t||8251===t||8252===t||8258===t||8263===t||8264===t||8265===t||8273===t)||Rn["Letterlike Symbols"](t)||Rn["Number Forms"](t)||Rn["Miscellaneous Technical"](t)&&(t>=8960&&t<=8967||t>=8972&&t<=8991||t>=8996&&t<=9e3||9003===t||t>=9085&&t<=9114||t>=9150&&t<=9165||9167===t||t>=9169&&t<=9179||t>=9186&&t<=9215)||Rn["Control Pictures"](t)&&9251!==t||Rn["Optical Character Recognition"](t)||Rn["Enclosed Alphanumerics"](t)||Rn["Geometric Shapes"](t)||Rn["Miscellaneous Symbols"](t)&&!(t>=9754&&t<=9759)||Rn["Miscellaneous Symbols and Arrows"](t)&&(t>=11026&&t<=11055||t>=11088&&t<=11097||t>=11192&&t<=11243)||Rn["CJK Symbols and Punctuation"](t)||Rn.Katakana(t)||Rn["Private Use Area"](t)||Rn["CJK Compatibility Forms"](t)||Rn["Small Form Variants"](t)||Rn["Halfwidth and Fullwidth Forms"](t)||8734===t||8756===t||8757===t||t>=9984&&t<=10087||t>=10102&&t<=10131||65532===t||65533===t)}(t))}function Gn(t){return t>=1424&&t<=2303||Rn["Arabic Presentation Forms-A"](t)||Rn["Arabic Presentation Forms-B"](t)}function Jn(t,e){return !(!e&&Gn(t)||t>=2304&&t<=3583||t>=3840&&t<=4255||Rn.Khmer(t))}function Xn(t){for(const e of t)if(Gn(e.charCodeAt(0)))return !0;return !1}const Yn="deferred",Hn="loading",Wn="loaded";let Qn=null,ti="unavailable",ei=null;const ri=function(t){t&&"string"==typeof t&&t.indexOf("NetworkError")>-1&&(ti="error"),Qn&&Qn(t);};function ni(){ii.fire(new K("pluginStateChange",{pluginStatus:ti,pluginURL:ei}));}const ii=new J,ai=function(){return ti},si=function(){if(ti!==Yn||!ei)throw new Error("rtl-text-plugin cannot be downloaded unless a pluginURL is specified");ti=Hn,ni(),ei&&q({url:ei},(t=>{t?ri(t):(ti=Wn,ni());}));},oi={applyArabicShaping:null,processBidirectionalText:null,processStyledBidirectionalText:null,isLoaded:()=>ti===Wn||null!=oi.applyArabicShaping,isLoading:()=>ti===Hn,setState(t){if(!I())throw new Error("Cannot set the state of the rtl-text-plugin when not in the web-worker context");ti=t.pluginStatus,ei=t.pluginURL;},isParsed(){if(!I())throw new Error("rtl-text-plugin is only parsed on the worker-threads");return null!=oi.applyArabicShaping&&null!=oi.processBidirectionalText&&null!=oi.processStyledBidirectionalText},getPluginURL(){if(!I())throw new Error("rtl-text-plugin url can only be queried from the worker threads");return ei}};class li{constructor(t,e){this.zoom=t,e?(this.now=e.now,this.fadeDuration=e.fadeDuration,this.zoomHistory=e.zoomHistory,this.transition=e.transition):(this.now=0,this.fadeDuration=0,this.zoomHistory=new Un,this.transition={});}isSupportedScript(t){return function(t,e){for(const r of t)if(!Jn(r.charCodeAt(0),e))return !1;return !0}(t,oi.isLoaded())}crossFadingFactor(){return 0===this.fadeDuration?1:Math.min((this.now-this.zoomHistory.lastIntegerZoomTime)/this.fadeDuration,1)}getCrossfadeParameters(){const t=this.zoom,e=t-Math.floor(t),r=this.crossFadingFactor();return t>this.zoomHistory.lastIntegerZoom?{fromScale:2,toScale:1,t:e+(1-e)*r}:{fromScale:.5,toScale:1,t:1-(1-r)*e}}}class ui{constructor(t,e){this.property=t,this.value=e,this.expression=function(t,e){if(Ir(t))return new Ur(t,e);if(Tr(t)){const r=Or(t,e);if("error"===r.result)throw new Error(r.value.map((t=>`${t.key}: ${t.message}`)).join(", "));return r.value}{let r=t;return "color"===e.type&&"string"==typeof t?r=Yt.parse(t):"padding"!==e.type||"number"!=typeof t&&!Array.isArray(t)?"variableAnchorOffsetCollection"===e.type&&Array.isArray(t)&&(r=re.parse(t)):r=te.parse(t),{kind:"constant",evaluate:()=>r}}}(void 0===e?t.specification.default:e,t.specification);}isDataDriven(){return "source"===this.expression.kind||"composite"===this.expression.kind}possiblyEvaluate(t,e,r){return this.property.possiblyEvaluate(this,t,e,r)}}class ci{constructor(t){this.property=t,this.value=new ui(t,void 0);}transitioned(t,e){return new pi(this.property,this.value,e,g({},t.transition,this.transition),t.now)}untransitioned(){return new pi(this.property,this.value,null,{},0)}}class hi{constructor(t){this._properties=t,this._values=Object.create(t.defaultTransitionablePropertyValues);}getValue(t){return w(this._values[t].value.value)}setValue(t,e){Object.prototype.hasOwnProperty.call(this._values,t)||(this._values[t]=new ci(this._values[t].property)),this._values[t].value=new ui(this._values[t].property,null===e?void 0:w(e));}getTransition(t){return w(this._values[t].transition)}setTransition(t,e){Object.prototype.hasOwnProperty.call(this._values,t)||(this._values[t]=new ci(this._values[t].property)),this._values[t].transition=w(e)||void 0;}serialize(){const t={};for(const e of Object.keys(this._values)){const r=this.getValue(e);void 0!==r&&(t[e]=r);const n=this.getTransition(e);void 0!==n&&(t[`${e}-transition`]=n);}return t}transitioned(t,e){const r=new fi(this._properties);for(const n of Object.keys(this._values))r._values[n]=this._values[n].transitioned(t,e._values[n]);return r}untransitioned(){const t=new fi(this._properties);for(const e of Object.keys(this._values))t._values[e]=this._values[e].untransitioned();return t}}class pi{constructor(t,e,r,n,i){this.property=t,this.value=e,this.begin=i+n.delay||0,this.end=this.begin+n.duration||0,t.specification.transition&&(n.delay||n.duration)&&(this.prior=r);}possiblyEvaluate(t,e,r){const n=t.now||0,i=this.value.possiblyEvaluate(t,e,r),a=this.prior;if(a){if(n>this.end)return this.prior=null,i;if(this.value.isDataDriven())return this.prior=null,i;if(n=1)return 1;const e=t*t,r=e*t;return 4*(t<.5?r:3*(t-e)+r-.75)}(s))}}return i}}class fi{constructor(t){this._properties=t,this._values=Object.create(t.defaultTransitioningPropertyValues);}possiblyEvaluate(t,e,r){const n=new mi(this._properties);for(const i of Object.keys(this._values))n._values[i]=this._values[i].possiblyEvaluate(t,e,r);return n}hasTransition(){for(const t of Object.keys(this._values))if(this._values[t].prior)return !0;return !1}}class di{constructor(t){this._properties=t,this._values=Object.create(t.defaultPropertyValues);}hasValue(t){return void 0!==this._values[t].value}getValue(t){return w(this._values[t].value)}setValue(t,e){this._values[t]=new ui(this._values[t].property,null===e?void 0:w(e));}serialize(){const t={};for(const e of Object.keys(this._values)){const r=this.getValue(e);void 0!==r&&(t[e]=r);}return t}possiblyEvaluate(t,e,r){const n=new mi(this._properties);for(const i of Object.keys(this._values))n._values[i]=this._values[i].possiblyEvaluate(t,e,r);return n}}class yi{constructor(t,e,r){this.property=t,this.value=e,this.parameters=r;}isConstant(){return "constant"===this.value.kind}constantOr(t){return "constant"===this.value.kind?this.value.value:t}evaluate(t,e,r,n){return this.property.evaluate(this.value,this.parameters,t,e,r,n)}}class mi{constructor(t){this._properties=t,this._values=Object.create(t.defaultPossiblyEvaluatedValues);}get(t){return this._values[t]}}class gi{constructor(t){this.specification=t;}possiblyEvaluate(t,e){if(t.isDataDriven())throw new Error("Value should not be data driven");return t.expression.evaluate(e)}interpolate(t,e,r){const n=Ze[this.specification.type];return n?n(t,e,r):t}}class xi{constructor(t,e){this.specification=t,this.overrides=e;}possiblyEvaluate(t,e,r,n){return new yi(this,"constant"===t.expression.kind||"camera"===t.expression.kind?{kind:"constant",value:t.expression.evaluate(e,null,{},r,n)}:t.expression,e)}interpolate(t,e,r){if("constant"!==t.value.kind||"constant"!==e.value.kind)return t;if(void 0===t.value.value||void 0===e.value.value)return new yi(this,{kind:"constant",value:void 0},t.parameters);const n=Ze[this.specification.type];if(n){const i=n(t.value.value,e.value.value,r);return new yi(this,{kind:"constant",value:i},t.parameters)}return t}evaluate(t,e,r,n,i,a){return "constant"===t.kind?t.value:t.evaluate(e,r,n,i,a)}}class vi extends xi{possiblyEvaluate(t,e,r,n){if(void 0===t.value)return new yi(this,{kind:"constant",value:void 0},e);if("constant"===t.expression.kind){const i=t.expression.evaluate(e,null,{},r,n),a="resolvedImage"===t.property.specification.type&&"string"!=typeof i?i.name:i,s=this._calculate(a,a,a,e);return new yi(this,{kind:"constant",value:s},e)}if("camera"===t.expression.kind){const r=this._calculate(t.expression.evaluate({zoom:e.zoom-1}),t.expression.evaluate({zoom:e.zoom}),t.expression.evaluate({zoom:e.zoom+1}),e);return new yi(this,{kind:"constant",value:r},e)}return new yi(this,t.expression,e)}evaluate(t,e,r,n,i,a){if("source"===t.kind){const s=t.evaluate(e,r,n,i,a);return this._calculate(s,s,s,e)}return "composite"===t.kind?this._calculate(t.evaluate({zoom:Math.floor(e.zoom)-1},r,n),t.evaluate({zoom:Math.floor(e.zoom)},r,n),t.evaluate({zoom:Math.floor(e.zoom)+1},r,n),e):t.value}_calculate(t,e,r,n){return n.zoom>n.zoomHistory.lastIntegerZoom?{from:t,to:e}:{from:r,to:e}}interpolate(t){return t}}class bi{constructor(t){this.specification=t;}possiblyEvaluate(t,e,r,n){if(void 0!==t.value){if("constant"===t.expression.kind){const i=t.expression.evaluate(e,null,{},r,n);return this._calculate(i,i,i,e)}return this._calculate(t.expression.evaluate(new li(Math.floor(e.zoom-1),e)),t.expression.evaluate(new li(Math.floor(e.zoom),e)),t.expression.evaluate(new li(Math.floor(e.zoom+1),e)),e)}}_calculate(t,e,r,n){return n.zoom>n.zoomHistory.lastIntegerZoom?{from:t,to:e}:{from:r,to:e}}interpolate(t){return t}}class wi{constructor(t){this.specification=t;}possiblyEvaluate(t,e,r,n){return !!t.expression.evaluate(e,null,{},r,n)}interpolate(){return !1}}class _i{constructor(t){this.properties=t,this.defaultPropertyValues={},this.defaultTransitionablePropertyValues={},this.defaultTransitioningPropertyValues={},this.defaultPossiblyEvaluatedValues={},this.overridableProperties=[];for(const e in t){const r=t[e];r.specification.overridable&&this.overridableProperties.push(e);const n=this.defaultPropertyValues[e]=new ui(r,void 0),i=this.defaultTransitionablePropertyValues[e]=new ci(r);this.defaultTransitioningPropertyValues[e]=i.untransitioned(),this.defaultPossiblyEvaluatedValues[e]=n.possiblyEvaluate({});}}}$n("DataDrivenProperty",xi),$n("DataConstantProperty",gi),$n("CrossFadedDataDrivenProperty",vi),$n("CrossFadedProperty",bi),$n("ColorRampProperty",wi);const Ai="-transition";class Si extends J{constructor(t,e){if(super(),this.id=t.id,this.type=t.type,this._featureFilter={filter:()=>!0,needGeometry:!1},"custom"!==t.type&&(this.metadata=t.metadata,this.minzoom=t.minzoom,this.maxzoom=t.maxzoom,"background"!==t.type&&(this.source=t.source,this.sourceLayer=t["source-layer"],this.filter=t.filter),e.layout&&(this._unevaluatedLayout=new di(e.layout)),e.paint)){this._transitionablePaint=new hi(e.paint);for(const e in t.paint)this.setPaintProperty(e,t.paint[e],{validate:!1});for(const e in t.layout)this.setLayoutProperty(e,t.layout[e],{validate:!1});this._transitioningPaint=this._transitionablePaint.untransitioned(),this.paint=new mi(e.paint);}}getCrossfadeParameters(){return this._crossfadeParameters}getLayoutProperty(t){return "visibility"===t?this.visibility:this._unevaluatedLayout.getValue(t)}setLayoutProperty(t,e,r={}){null!=e&&this._validate(Vn,`layers.${this.id}.layout.${t}`,t,e,r)||("visibility"!==t?this._unevaluatedLayout.setValue(t,e):this.visibility=e);}getPaintProperty(t){return t.endsWith(Ai)?this._transitionablePaint.getTransition(t.slice(0,-11)):this._transitionablePaint.getValue(t)}setPaintProperty(t,e,r={}){if(null!=e&&this._validate(Pn,`layers.${this.id}.paint.${t}`,t,e,r))return !1;if(t.endsWith(Ai))return this._transitionablePaint.setTransition(t.slice(0,-11),e||void 0),!1;{const r=this._transitionablePaint._values[t],n="cross-faded-data-driven"===r.property.specification["property-type"],i=r.value.isDataDriven(),a=r.value;this._transitionablePaint.setValue(t,e),this._handleSpecialPaintPropertyUpdate(t);const s=this._transitionablePaint._values[t].value;return s.isDataDriven()||i||n||this._handleOverridablePaintPropertyUpdate(t,a,s)}}_handleSpecialPaintPropertyUpdate(t){}_handleOverridablePaintPropertyUpdate(t,e,r){return !1}isHidden(t){return !!(this.minzoom&&t=this.maxzoom)||"none"===this.visibility}updateTransitions(t){this._transitioningPaint=this._transitionablePaint.transitioned(t,this._transitioningPaint);}hasTransition(){return this._transitioningPaint.hasTransition()}recalculate(t,e){t.getCrossfadeParameters&&(this._crossfadeParameters=t.getCrossfadeParameters()),this._unevaluatedLayout&&(this.layout=this._unevaluatedLayout.possiblyEvaluate(t,void 0,e)),this.paint=this._transitioningPaint.possiblyEvaluate(t,void 0,e);}serialize(){const t={id:this.id,type:this.type,source:this.source,"source-layer":this.sourceLayer,metadata:this.metadata,minzoom:this.minzoom,maxzoom:this.maxzoom,filter:this.filter,layout:this._unevaluatedLayout&&this._unevaluatedLayout.serialize(),paint:this._transitionablePaint&&this._transitionablePaint.serialize()};return this.visibility&&(t.layout=t.layout||{},t.layout.visibility=this.visibility),b(t,((t,e)=>!(void 0===t||"layout"===e&&!Object.keys(t).length||"paint"===e&&!Object.keys(t).length)))}_validate(t,e,r,n,i={}){return (!i||!1!==i.validate)&&En(this,t.call(Cn,{key:e,layerType:this.type,objectKey:r,value:n,styleSpec:X,style:{glyphs:!0,sprite:!0}}))}is3D(){return !1}isTileClipped(){return !1}hasOffscreenPass(){return !1}resize(){}isStateDependent(){for(const t in this.paint._values){const e=this.paint.get(t);if(e instanceof yi&&_r(e.property.specification)&&("source"===e.value.kind||"composite"===e.value.kind)&&e.value.isStateDependent)return !0}return !1}}const ki={Int8:Int8Array,Uint8:Uint8Array,Int16:Int16Array,Uint16:Uint16Array,Int32:Int32Array,Uint32:Uint32Array,Float32:Float32Array};class Ii{constructor(t,e){this._structArray=t,this._pos1=e*this.size,this._pos2=this._pos1/2,this._pos4=this._pos1/4,this._pos8=this._pos1/8;}}class zi{constructor(){this.isTransferred=!1,this.capacity=-1,this.resize(0);}static serialize(t,e){return t._trim(),e&&(t.isTransferred=!0,e.push(t.arrayBuffer)),{length:t.length,arrayBuffer:t.arrayBuffer}}static deserialize(t){const e=Object.create(this.prototype);return e.arrayBuffer=t.arrayBuffer,e.length=t.length,e.capacity=t.arrayBuffer.byteLength/e.bytesPerElement,e._refreshViews(),e}_trim(){this.length!==this.capacity&&(this.capacity=this.length,this.arrayBuffer=this.arrayBuffer.slice(0,this.length*this.bytesPerElement),this._refreshViews());}clear(){this.length=0;}resize(t){this.reserve(t),this.length=t;}reserve(t){if(t>this.capacity){this.capacity=Math.max(t,Math.floor(5*this.capacity),128),this.arrayBuffer=new ArrayBuffer(this.capacity*this.bytesPerElement);const e=this.uint8;this._refreshViews(),e&&this.uint8.set(e);}}_refreshViews(){throw new Error("_refreshViews() must be implemented by each concrete StructArray layout")}}function Mi(t,e=1){let r=0,n=0;return {members:t.map((t=>{const i=ki[t.type].BYTES_PER_ELEMENT,a=r=Ci(r,Math.max(e,i)),s=t.components||1;return n=Math.max(n,i),r+=i*s,{name:t.name,type:t.type,components:s,offset:a}})),size:Ci(r,Math.max(n,e)),alignment:e}}function Ci(t,e){return Math.ceil(t/e)*e}class Bi extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer);}emplaceBack(t,e){const r=this.length;return this.resize(r+1),this.emplace(r,t,e)}emplace(t,e,r){const n=2*t;return this.int16[n+0]=e,this.int16[n+1]=r,t}}Bi.prototype.bytesPerElement=4,$n("StructArrayLayout2i4",Bi);class Pi extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer);}emplaceBack(t,e,r){const n=this.length;return this.resize(n+1),this.emplace(n,t,e,r)}emplace(t,e,r,n){const i=3*t;return this.int16[i+0]=e,this.int16[i+1]=r,this.int16[i+2]=n,t}}Pi.prototype.bytesPerElement=6,$n("StructArrayLayout3i6",Pi);class Vi extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer);}emplaceBack(t,e,r,n){const i=this.length;return this.resize(i+1),this.emplace(i,t,e,r,n)}emplace(t,e,r,n,i){const a=4*t;return this.int16[a+0]=e,this.int16[a+1]=r,this.int16[a+2]=n,this.int16[a+3]=i,t}}Vi.prototype.bytesPerElement=8,$n("StructArrayLayout4i8",Vi);class Ei extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer);}emplaceBack(t,e,r,n,i,a){const s=this.length;return this.resize(s+1),this.emplace(s,t,e,r,n,i,a)}emplace(t,e,r,n,i,a,s){const o=6*t;return this.int16[o+0]=e,this.int16[o+1]=r,this.int16[o+2]=n,this.int16[o+3]=i,this.int16[o+4]=a,this.int16[o+5]=s,t}}Ei.prototype.bytesPerElement=12,$n("StructArrayLayout2i4i12",Ei);class Fi extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer);}emplaceBack(t,e,r,n,i,a){const s=this.length;return this.resize(s+1),this.emplace(s,t,e,r,n,i,a)}emplace(t,e,r,n,i,a,s){const o=4*t,l=8*t;return this.int16[o+0]=e,this.int16[o+1]=r,this.uint8[l+4]=n,this.uint8[l+5]=i,this.uint8[l+6]=a,this.uint8[l+7]=s,t}}Fi.prototype.bytesPerElement=8,$n("StructArrayLayout2i4ub8",Fi);class Ti extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer);}emplaceBack(t,e){const r=this.length;return this.resize(r+1),this.emplace(r,t,e)}emplace(t,e,r){const n=2*t;return this.float32[n+0]=e,this.float32[n+1]=r,t}}Ti.prototype.bytesPerElement=8,$n("StructArrayLayout2f8",Ti);class $i extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer);}emplaceBack(t,e,r,n,i,a,s,o,l,u){const c=this.length;return this.resize(c+1),this.emplace(c,t,e,r,n,i,a,s,o,l,u)}emplace(t,e,r,n,i,a,s,o,l,u,c){const h=10*t;return this.uint16[h+0]=e,this.uint16[h+1]=r,this.uint16[h+2]=n,this.uint16[h+3]=i,this.uint16[h+4]=a,this.uint16[h+5]=s,this.uint16[h+6]=o,this.uint16[h+7]=l,this.uint16[h+8]=u,this.uint16[h+9]=c,t}}$i.prototype.bytesPerElement=20,$n("StructArrayLayout10ui20",$i);class Li extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer);}emplaceBack(t,e,r,n,i,a,s,o,l,u,c,h){const p=this.length;return this.resize(p+1),this.emplace(p,t,e,r,n,i,a,s,o,l,u,c,h)}emplace(t,e,r,n,i,a,s,o,l,u,c,h,p){const f=12*t;return this.int16[f+0]=e,this.int16[f+1]=r,this.int16[f+2]=n,this.int16[f+3]=i,this.uint16[f+4]=a,this.uint16[f+5]=s,this.uint16[f+6]=o,this.uint16[f+7]=l,this.int16[f+8]=u,this.int16[f+9]=c,this.int16[f+10]=h,this.int16[f+11]=p,t}}Li.prototype.bytesPerElement=24,$n("StructArrayLayout4i4ui4i24",Li);class Di extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer);}emplaceBack(t,e,r){const n=this.length;return this.resize(n+1),this.emplace(n,t,e,r)}emplace(t,e,r,n){const i=3*t;return this.float32[i+0]=e,this.float32[i+1]=r,this.float32[i+2]=n,t}}Di.prototype.bytesPerElement=12,$n("StructArrayLayout3f12",Di);class Oi extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer);}emplaceBack(t){const e=this.length;return this.resize(e+1),this.emplace(e,t)}emplace(t,e){return this.uint32[1*t+0]=e,t}}Oi.prototype.bytesPerElement=4,$n("StructArrayLayout1ul4",Oi);class Ui extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer);}emplaceBack(t,e,r,n,i,a,s,o,l){const u=this.length;return this.resize(u+1),this.emplace(u,t,e,r,n,i,a,s,o,l)}emplace(t,e,r,n,i,a,s,o,l,u){const c=10*t,h=5*t;return this.int16[c+0]=e,this.int16[c+1]=r,this.int16[c+2]=n,this.int16[c+3]=i,this.int16[c+4]=a,this.int16[c+5]=s,this.uint32[h+3]=o,this.uint16[c+8]=l,this.uint16[c+9]=u,t}}Ui.prototype.bytesPerElement=20,$n("StructArrayLayout6i1ul2ui20",Ui);class Ri extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer);}emplaceBack(t,e,r,n,i,a){const s=this.length;return this.resize(s+1),this.emplace(s,t,e,r,n,i,a)}emplace(t,e,r,n,i,a,s){const o=6*t;return this.int16[o+0]=e,this.int16[o+1]=r,this.int16[o+2]=n,this.int16[o+3]=i,this.int16[o+4]=a,this.int16[o+5]=s,t}}Ri.prototype.bytesPerElement=12,$n("StructArrayLayout2i2i2i12",Ri);class qi extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer);}emplaceBack(t,e,r,n,i){const a=this.length;return this.resize(a+1),this.emplace(a,t,e,r,n,i)}emplace(t,e,r,n,i,a){const s=4*t,o=8*t;return this.float32[s+0]=e,this.float32[s+1]=r,this.float32[s+2]=n,this.int16[o+6]=i,this.int16[o+7]=a,t}}qi.prototype.bytesPerElement=16,$n("StructArrayLayout2f1f2i16",qi);class ji extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer);}emplaceBack(t,e,r,n){const i=this.length;return this.resize(i+1),this.emplace(i,t,e,r,n)}emplace(t,e,r,n,i){const a=12*t,s=3*t;return this.uint8[a+0]=e,this.uint8[a+1]=r,this.float32[s+1]=n,this.float32[s+2]=i,t}}ji.prototype.bytesPerElement=12,$n("StructArrayLayout2ub2f12",ji);class Ni extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer);}emplaceBack(t,e,r){const n=this.length;return this.resize(n+1),this.emplace(n,t,e,r)}emplace(t,e,r,n){const i=3*t;return this.uint16[i+0]=e,this.uint16[i+1]=r,this.uint16[i+2]=n,t}}Ni.prototype.bytesPerElement=6,$n("StructArrayLayout3ui6",Ni);class Zi extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer);}emplaceBack(t,e,r,n,i,a,s,o,l,u,c,h,p,f,d,y,m){const g=this.length;return this.resize(g+1),this.emplace(g,t,e,r,n,i,a,s,o,l,u,c,h,p,f,d,y,m)}emplace(t,e,r,n,i,a,s,o,l,u,c,h,p,f,d,y,m,g){const x=24*t,v=12*t,b=48*t;return this.int16[x+0]=e,this.int16[x+1]=r,this.uint16[x+2]=n,this.uint16[x+3]=i,this.uint32[v+2]=a,this.uint32[v+3]=s,this.uint32[v+4]=o,this.uint16[x+10]=l,this.uint16[x+11]=u,this.uint16[x+12]=c,this.float32[v+7]=h,this.float32[v+8]=p,this.uint8[b+36]=f,this.uint8[b+37]=d,this.uint8[b+38]=y,this.uint32[v+10]=m,this.int16[x+22]=g,t}}Zi.prototype.bytesPerElement=48,$n("StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48",Zi);class Ki extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.int16=new Int16Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer);}emplaceBack(t,e,r,n,i,a,s,o,l,u,c,h,p,f,d,y,m,g,x,v,b,w,_,A,S,k,I,z){const M=this.length;return this.resize(M+1),this.emplace(M,t,e,r,n,i,a,s,o,l,u,c,h,p,f,d,y,m,g,x,v,b,w,_,A,S,k,I,z)}emplace(t,e,r,n,i,a,s,o,l,u,c,h,p,f,d,y,m,g,x,v,b,w,_,A,S,k,I,z,M){const C=32*t,B=16*t;return this.int16[C+0]=e,this.int16[C+1]=r,this.int16[C+2]=n,this.int16[C+3]=i,this.int16[C+4]=a,this.int16[C+5]=s,this.int16[C+6]=o,this.int16[C+7]=l,this.uint16[C+8]=u,this.uint16[C+9]=c,this.uint16[C+10]=h,this.uint16[C+11]=p,this.uint16[C+12]=f,this.uint16[C+13]=d,this.uint16[C+14]=y,this.uint16[C+15]=m,this.uint16[C+16]=g,this.uint16[C+17]=x,this.uint16[C+18]=v,this.uint16[C+19]=b,this.uint16[C+20]=w,this.uint16[C+21]=_,this.uint16[C+22]=A,this.uint32[B+12]=S,this.float32[B+13]=k,this.float32[B+14]=I,this.uint16[C+30]=z,this.uint16[C+31]=M,t}}Ki.prototype.bytesPerElement=64,$n("StructArrayLayout8i15ui1ul2f2ui64",Ki);class Gi extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer);}emplaceBack(t){const e=this.length;return this.resize(e+1),this.emplace(e,t)}emplace(t,e){return this.float32[1*t+0]=e,t}}Gi.prototype.bytesPerElement=4,$n("StructArrayLayout1f4",Gi);class Ji extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer);}emplaceBack(t,e,r){const n=this.length;return this.resize(n+1),this.emplace(n,t,e,r)}emplace(t,e,r,n){const i=3*t;return this.uint16[6*t+0]=e,this.float32[i+1]=r,this.float32[i+2]=n,t}}Ji.prototype.bytesPerElement=12,$n("StructArrayLayout1ui2f12",Ji);class Xi extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint32=new Uint32Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer);}emplaceBack(t,e,r){const n=this.length;return this.resize(n+1),this.emplace(n,t,e,r)}emplace(t,e,r,n){const i=4*t;return this.uint32[2*t+0]=e,this.uint16[i+2]=r,this.uint16[i+3]=n,t}}Xi.prototype.bytesPerElement=8,$n("StructArrayLayout1ul2ui8",Xi);class Yi extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer);}emplaceBack(t,e){const r=this.length;return this.resize(r+1),this.emplace(r,t,e)}emplace(t,e,r){const n=2*t;return this.uint16[n+0]=e,this.uint16[n+1]=r,t}}Yi.prototype.bytesPerElement=4,$n("StructArrayLayout2ui4",Yi);class Hi extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.uint16=new Uint16Array(this.arrayBuffer);}emplaceBack(t){const e=this.length;return this.resize(e+1),this.emplace(e,t)}emplace(t,e){return this.uint16[1*t+0]=e,t}}Hi.prototype.bytesPerElement=2,$n("StructArrayLayout1ui2",Hi);class Wi extends zi{_refreshViews(){this.uint8=new Uint8Array(this.arrayBuffer),this.float32=new Float32Array(this.arrayBuffer);}emplaceBack(t,e,r,n){const i=this.length;return this.resize(i+1),this.emplace(i,t,e,r,n)}emplace(t,e,r,n,i){const a=4*t;return this.float32[a+0]=e,this.float32[a+1]=r,this.float32[a+2]=n,this.float32[a+3]=i,t}}Wi.prototype.bytesPerElement=16,$n("StructArrayLayout4f16",Wi);class Qi extends Ii{get anchorPointX(){return this._structArray.int16[this._pos2+0]}get anchorPointY(){return this._structArray.int16[this._pos2+1]}get x1(){return this._structArray.int16[this._pos2+2]}get y1(){return this._structArray.int16[this._pos2+3]}get x2(){return this._structArray.int16[this._pos2+4]}get y2(){return this._structArray.int16[this._pos2+5]}get featureIndex(){return this._structArray.uint32[this._pos4+3]}get sourceLayerIndex(){return this._structArray.uint16[this._pos2+8]}get bucketIndex(){return this._structArray.uint16[this._pos2+9]}get anchorPoint(){return new a(this.anchorPointX,this.anchorPointY)}}Qi.prototype.size=20;class ta extends Ui{get(t){return new Qi(this,t)}}$n("CollisionBoxArray",ta);class ea extends Ii{get anchorX(){return this._structArray.int16[this._pos2+0]}get anchorY(){return this._structArray.int16[this._pos2+1]}get glyphStartIndex(){return this._structArray.uint16[this._pos2+2]}get numGlyphs(){return this._structArray.uint16[this._pos2+3]}get vertexStartIndex(){return this._structArray.uint32[this._pos4+2]}get lineStartIndex(){return this._structArray.uint32[this._pos4+3]}get lineLength(){return this._structArray.uint32[this._pos4+4]}get segment(){return this._structArray.uint16[this._pos2+10]}get lowerSize(){return this._structArray.uint16[this._pos2+11]}get upperSize(){return this._structArray.uint16[this._pos2+12]}get lineOffsetX(){return this._structArray.float32[this._pos4+7]}get lineOffsetY(){return this._structArray.float32[this._pos4+8]}get writingMode(){return this._structArray.uint8[this._pos1+36]}get placedOrientation(){return this._structArray.uint8[this._pos1+37]}set placedOrientation(t){this._structArray.uint8[this._pos1+37]=t;}get hidden(){return this._structArray.uint8[this._pos1+38]}set hidden(t){this._structArray.uint8[this._pos1+38]=t;}get crossTileID(){return this._structArray.uint32[this._pos4+10]}set crossTileID(t){this._structArray.uint32[this._pos4+10]=t;}get associatedIconIndex(){return this._structArray.int16[this._pos2+22]}}ea.prototype.size=48;class ra extends Zi{get(t){return new ea(this,t)}}$n("PlacedSymbolArray",ra);class na extends Ii{get anchorX(){return this._structArray.int16[this._pos2+0]}get anchorY(){return this._structArray.int16[this._pos2+1]}get rightJustifiedTextSymbolIndex(){return this._structArray.int16[this._pos2+2]}get centerJustifiedTextSymbolIndex(){return this._structArray.int16[this._pos2+3]}get leftJustifiedTextSymbolIndex(){return this._structArray.int16[this._pos2+4]}get verticalPlacedTextSymbolIndex(){return this._structArray.int16[this._pos2+5]}get placedIconSymbolIndex(){return this._structArray.int16[this._pos2+6]}get verticalPlacedIconSymbolIndex(){return this._structArray.int16[this._pos2+7]}get key(){return this._structArray.uint16[this._pos2+8]}get textBoxStartIndex(){return this._structArray.uint16[this._pos2+9]}get textBoxEndIndex(){return this._structArray.uint16[this._pos2+10]}get verticalTextBoxStartIndex(){return this._structArray.uint16[this._pos2+11]}get verticalTextBoxEndIndex(){return this._structArray.uint16[this._pos2+12]}get iconBoxStartIndex(){return this._structArray.uint16[this._pos2+13]}get iconBoxEndIndex(){return this._structArray.uint16[this._pos2+14]}get verticalIconBoxStartIndex(){return this._structArray.uint16[this._pos2+15]}get verticalIconBoxEndIndex(){return this._structArray.uint16[this._pos2+16]}get featureIndex(){return this._structArray.uint16[this._pos2+17]}get numHorizontalGlyphVertices(){return this._structArray.uint16[this._pos2+18]}get numVerticalGlyphVertices(){return this._structArray.uint16[this._pos2+19]}get numIconVertices(){return this._structArray.uint16[this._pos2+20]}get numVerticalIconVertices(){return this._structArray.uint16[this._pos2+21]}get useRuntimeCollisionCircles(){return this._structArray.uint16[this._pos2+22]}get crossTileID(){return this._structArray.uint32[this._pos4+12]}set crossTileID(t){this._structArray.uint32[this._pos4+12]=t;}get textBoxScale(){return this._structArray.float32[this._pos4+13]}get collisionCircleDiameter(){return this._structArray.float32[this._pos4+14]}get textAnchorOffsetStartIndex(){return this._structArray.uint16[this._pos2+30]}get textAnchorOffsetEndIndex(){return this._structArray.uint16[this._pos2+31]}}na.prototype.size=64;class ia extends Ki{get(t){return new na(this,t)}}$n("SymbolInstanceArray",ia);class aa extends Gi{getoffsetX(t){return this.float32[1*t+0]}}$n("GlyphOffsetArray",aa);class sa extends Pi{getx(t){return this.int16[3*t+0]}gety(t){return this.int16[3*t+1]}gettileUnitDistanceFromAnchor(t){return this.int16[3*t+2]}}$n("SymbolLineVertexArray",sa);class oa extends Ii{get textAnchor(){return this._structArray.uint16[this._pos2+0]}get textOffset0(){return this._structArray.float32[this._pos4+1]}get textOffset1(){return this._structArray.float32[this._pos4+2]}}oa.prototype.size=12;class la extends Ji{get(t){return new oa(this,t)}}$n("TextAnchorOffsetArray",la);class ua extends Ii{get featureIndex(){return this._structArray.uint32[this._pos4+0]}get sourceLayerIndex(){return this._structArray.uint16[this._pos2+2]}get bucketIndex(){return this._structArray.uint16[this._pos2+3]}}ua.prototype.size=8;class ca extends Xi{get(t){return new ua(this,t)}}$n("FeatureIndexArray",ca);class ha extends Bi{}class pa extends Bi{}class fa extends Bi{}class da extends Ei{}class ya extends Fi{}class ma extends Ti{}class ga extends $i{}class xa extends Li{}class va extends Di{}class ba extends Oi{}class wa extends Ri{}class _a extends ji{}class Aa extends Ni{}class Sa extends Yi{}const ka=Mi([{name:"a_pos",components:2,type:"Int16"}],4),{members:Ia}=ka;class za{constructor(t=[]){this.segments=t;}prepareSegment(t,e,r,n){let i=this.segments[this.segments.length-1];return t>za.MAX_VERTEX_ARRAY_LENGTH&&A(`Max vertices per segment is ${za.MAX_VERTEX_ARRAY_LENGTH}: bucket requested ${t}`),(!i||i.vertexLength+t>za.MAX_VERTEX_ARRAY_LENGTH||i.sortKey!==n)&&(i={vertexOffset:e.length,primitiveOffset:r.length,vertexLength:0,primitiveLength:0},void 0!==n&&(i.sortKey=n),this.segments.push(i)),i}get(){return this.segments}destroy(){for(const t of this.segments)for(const e in t.vaos)t.vaos[e].destroy();}static simpleSegment(t,e,r,n){return new za([{vertexOffset:t,primitiveOffset:e,vertexLength:r,primitiveLength:n,vaos:{},sortKey:0}])}}function Ma(t,e){return 256*(t=y(Math.floor(t),0,255))+y(Math.floor(e),0,255)}za.MAX_VERTEX_ARRAY_LENGTH=Math.pow(2,16)-1,$n("SegmentVector",za);const Ca=Mi([{name:"a_pattern_from",components:4,type:"Uint16"},{name:"a_pattern_to",components:4,type:"Uint16"},{name:"a_pixel_ratio_from",components:1,type:"Uint16"},{name:"a_pixel_ratio_to",components:1,type:"Uint16"}]);var Ba={exports:{}},Pa={exports:{}};Pa.exports=function(t,e){var r,n,i,a,s,o,l,u;for(n=t.length-(r=3&t.length),i=e,s=3432918353,o=461845907,u=0;u>>16)*s&65535)<<16)&4294967295)<<15|l>>>17))*o+(((l>>>16)*o&65535)<<16)&4294967295)<<13|i>>>19))+((5*(i>>>16)&65535)<<16)&4294967295))+((58964+(a>>>16)&65535)<<16);switch(l=0,r){case 3:l^=(255&t.charCodeAt(u+2))<<16;case 2:l^=(255&t.charCodeAt(u+1))<<8;case 1:i^=l=(65535&(l=(l=(65535&(l^=255&t.charCodeAt(u)))*s+(((l>>>16)*s&65535)<<16)&4294967295)<<15|l>>>17))*o+(((l>>>16)*o&65535)<<16)&4294967295;}return i^=t.length,i=2246822507*(65535&(i^=i>>>16))+((2246822507*(i>>>16)&65535)<<16)&4294967295,i=3266489909*(65535&(i^=i>>>13))+((3266489909*(i>>>16)&65535)<<16)&4294967295,(i^=i>>>16)>>>0};var Va=Pa.exports,Ea={exports:{}};Ea.exports=function(t,e){for(var r,n=t.length,i=e^n,a=0;n>=4;)r=1540483477*(65535&(r=255&t.charCodeAt(a)|(255&t.charCodeAt(++a))<<8|(255&t.charCodeAt(++a))<<16|(255&t.charCodeAt(++a))<<24))+((1540483477*(r>>>16)&65535)<<16),i=1540483477*(65535&i)+((1540483477*(i>>>16)&65535)<<16)^(r=1540483477*(65535&(r^=r>>>24))+((1540483477*(r>>>16)&65535)<<16)),n-=4,++a;switch(n){case 3:i^=(255&t.charCodeAt(a+2))<<16;case 2:i^=(255&t.charCodeAt(a+1))<<8;case 1:i=1540483477*(65535&(i^=255&t.charCodeAt(a)))+((1540483477*(i>>>16)&65535)<<16);}return i=1540483477*(65535&(i^=i>>>13))+((1540483477*(i>>>16)&65535)<<16),(i^=i>>>15)>>>0};var Fa=Va,Ta=Ea.exports;Ba.exports=Fa,Ba.exports.murmur3=Fa,Ba.exports.murmur2=Ta;var $a=r(Ba.exports);class La{constructor(){this.ids=[],this.positions=[],this.indexed=!1;}add(t,e,r,n){this.ids.push(Da(t)),this.positions.push(e,r,n);}getPositions(t){if(!this.indexed)throw new Error("Trying to get index, but feature positions are not indexed");const e=Da(t);let r=0,n=this.ids.length-1;for(;r>1;this.ids[t]>=e?n=t:r=t+1;}const i=[];for(;this.ids[r]===e;)i.push({index:this.positions[3*r],start:this.positions[3*r+1],end:this.positions[3*r+2]}),r++;return i}static serialize(t,e){const r=new Float64Array(t.ids),n=new Uint32Array(t.positions);return Oa(r,n,0,r.length-1),e&&e.push(r.buffer,n.buffer),{ids:r,positions:n}}static deserialize(t){const e=new La;return e.ids=t.ids,e.positions=t.positions,e.indexed=!0,e}}function Da(t){const e=+t;return !isNaN(e)&&e<=Number.MAX_SAFE_INTEGER?e:$a(String(t))}function Oa(t,e,r,n){for(;r>1];let a=r-1,s=n+1;for(;;){do{a++;}while(t[a]i);if(a>=s)break;Ua(t,a,s),Ua(e,3*a,3*s),Ua(e,3*a+1,3*s+1),Ua(e,3*a+2,3*s+2);}s-r`u_${t}`)),this.type=r;}setUniform(t,e,r){t.set(r.constantOr(this.value));}getBinding(t,e,r){return "color"===this.type?new Na(t,e):new qa(t,e)}}class Ja{constructor(t,e){this.uniformNames=e.map((t=>`u_${t}`)),this.patternFrom=null,this.patternTo=null,this.pixelRatioFrom=1,this.pixelRatioTo=1;}setConstantPatternPositions(t,e){this.pixelRatioFrom=e.pixelRatio,this.pixelRatioTo=t.pixelRatio,this.patternFrom=e.tlbr,this.patternTo=t.tlbr;}setUniform(t,e,r,n){const i="u_pattern_to"===n?this.patternTo:"u_pattern_from"===n?this.patternFrom:"u_pixel_ratio_to"===n?this.pixelRatioTo:"u_pixel_ratio_from"===n?this.pixelRatioFrom:null;i&&t.set(i);}getBinding(t,e,r){return "u_pattern"===r.substr(0,9)?new ja(t,e):new qa(t,e)}}class Xa{constructor(t,e,r,n){this.expression=t,this.type=r,this.maxValue=0,this.paintVertexAttributes=e.map((t=>({name:`a_${t}`,type:"Float32",components:"color"===r?2:1,offset:0}))),this.paintVertexArray=new n;}populatePaintArray(t,e,r,n,i){const a=this.paintVertexArray.length,s=this.expression.evaluate(new li(0),e,{},n,[],i);this.paintVertexArray.resize(t),this._setPaintValue(a,t,s);}updatePaintArray(t,e,r,n){const i=this.expression.evaluate({zoom:0},r,n);this._setPaintValue(t,e,i);}_setPaintValue(t,e,r){if("color"===this.type){const n=Ka(r);for(let r=t;r`u_${t}_t`)),this.type=r,this.useIntegerZoom=n,this.zoom=i,this.maxValue=0,this.paintVertexAttributes=e.map((t=>({name:`a_${t}`,type:"Float32",components:"color"===r?4:2,offset:0}))),this.paintVertexArray=new a;}populatePaintArray(t,e,r,n,i){const a=this.expression.evaluate(new li(this.zoom),e,{},n,[],i),s=this.expression.evaluate(new li(this.zoom+1),e,{},n,[],i),o=this.paintVertexArray.length;this.paintVertexArray.resize(t),this._setPaintValue(o,t,a,s);}updatePaintArray(t,e,r,n){const i=this.expression.evaluate({zoom:this.zoom},r,n),a=this.expression.evaluate({zoom:this.zoom+1},r,n);this._setPaintValue(t,e,i,a);}_setPaintValue(t,e,r,n){if("color"===this.type){const i=Ka(r),a=Ka(n);for(let r=t;r`#define HAS_UNIFORM_${t}`)));}return t}getBinderAttributes(){const t=[];for(const e in this.binders){const r=this.binders[e];if(r instanceof Xa||r instanceof Ya)for(let e=0;e!0)){this.programConfigurations={};for(const n of t)this.programConfigurations[n.id]=new Wa(n,e,r);this.needsUpload=!1,this._featureMap=new La,this._bufferOffset=0;}populatePaintArrays(t,e,r,n,i,a){for(const r in this.programConfigurations)this.programConfigurations[r].populatePaintArrays(t,e,n,i,a);void 0!==e.id&&this._featureMap.add(e.id,r,this._bufferOffset,t),this._bufferOffset=t,this.needsUpload=!0;}updatePaintArrays(t,e,r,n){for(const i of r)this.needsUpload=this.programConfigurations[i.id].updatePaintArrays(t,this._featureMap,e,i,n)||this.needsUpload;}get(t){return this.programConfigurations[t]}upload(t){if(this.needsUpload){for(const e in this.programConfigurations)this.programConfigurations[e].upload(t);this.needsUpload=!1;}}destroy(){for(const t in this.programConfigurations)this.programConfigurations[t].destroy();}}function ts(t,e){return {"text-opacity":["opacity"],"icon-opacity":["opacity"],"text-color":["fill_color"],"icon-color":["fill_color"],"text-halo-color":["halo_color"],"icon-halo-color":["halo_color"],"text-halo-blur":["halo_blur"],"icon-halo-blur":["halo_blur"],"text-halo-width":["halo_width"],"icon-halo-width":["halo_width"],"line-gap-width":["gapwidth"],"line-pattern":["pattern_to","pattern_from","pixel_ratio_to","pixel_ratio_from"],"fill-pattern":["pattern_to","pattern_from","pixel_ratio_to","pixel_ratio_from"],"fill-extrusion-pattern":["pattern_to","pattern_from","pixel_ratio_to","pixel_ratio_from"]}[t]||[t.replace(`${e}-`,"").replace(/-/g,"_")]}function es(t,e,r){const n={color:{source:Ti,composite:Wi},number:{source:Gi,composite:Ti}},i=function(t){return {"line-pattern":{source:ga,composite:ga},"fill-pattern":{source:ga,composite:ga},"fill-extrusion-pattern":{source:ga,composite:ga}}[t]}(t);return i&&i[r]||n[e][r]}$n("ConstantBinder",Ga),$n("CrossFadedConstantBinder",Ja),$n("SourceExpressionBinder",Xa),$n("CrossFadedCompositeBinder",Ha),$n("CompositeExpressionBinder",Ya),$n("ProgramConfiguration",Wa,{omit:["_buffers"]}),$n("ProgramConfigurationSet",Qa);const rs=8192,ns=Math.pow(2,14)-1,is=-ns-1;function as(t){const e=rs/t.extent,r=t.loadGeometry();for(let t=0;tr.x+1||ar.y+1)&&A("Geometry exceeds allowed extent, reduce your vector tile buffer size");}}return r}function ss(t,e){return {type:t.type,id:t.id,properties:t.properties,geometry:e?as(t):[]}}function os(t,e,r,n,i){t.emplaceBack(2*e+(n+1)/2,2*r+(i+1)/2);}class ls{constructor(t){this.zoom=t.zoom,this.overscaling=t.overscaling,this.layers=t.layers,this.layerIds=this.layers.map((t=>t.id)),this.index=t.index,this.hasPattern=!1,this.layoutVertexArray=new pa,this.indexArray=new Aa,this.segments=new za,this.programConfigurations=new Qa(t.layers,t.zoom),this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id));}populate(t,e,r){const n=this.layers[0],i=[];let a=null,s=!1;"circle"===n.type&&(a=n.layout.get("circle-sort-key"),s=!a.isConstant());for(const{feature:e,id:n,index:o,sourceLayerIndex:l}of t){const t=this.layers[0]._featureFilter.needGeometry,u=ss(e,t);if(!this.layers[0]._featureFilter.filter(new li(this.zoom),u,r))continue;const c=s?a.evaluate(u,{},r):void 0,h={id:n,properties:e.properties,type:e.type,sourceLayerIndex:l,index:o,geometry:t?u.geometry:as(e),patterns:{},sortKey:c};i.push(h);}s&&i.sort(((t,e)=>t.sortKey-e.sortKey));for(const n of i){const{geometry:i,index:a,sourceLayerIndex:s}=n,o=t[a].feature;this.addFeature(n,i,a,r),e.featureIndex.insert(o,i,a,s,this.index);}}update(t,e,r){this.stateDependentLayers.length&&this.programConfigurations.updatePaintArrays(t,e,this.stateDependentLayers,r);}isEmpty(){return 0===this.layoutVertexArray.length}uploadPending(){return !this.uploaded||this.programConfigurations.needsUpload}upload(t){this.uploaded||(this.layoutVertexBuffer=t.createVertexBuffer(this.layoutVertexArray,Ia),this.indexBuffer=t.createIndexBuffer(this.indexArray)),this.programConfigurations.upload(t),this.uploaded=!0;}destroy(){this.layoutVertexBuffer&&(this.layoutVertexBuffer.destroy(),this.indexBuffer.destroy(),this.programConfigurations.destroy(),this.segments.destroy());}addFeature(t,e,r,n){for(const r of e)for(const e of r){const r=e.x,n=e.y;if(r<0||r>=rs||n<0||n>=rs)continue;const i=this.segments.prepareSegment(4,this.layoutVertexArray,this.indexArray,t.sortKey),a=i.vertexLength;os(this.layoutVertexArray,r,n,-1,-1),os(this.layoutVertexArray,r,n,1,-1),os(this.layoutVertexArray,r,n,1,1),os(this.layoutVertexArray,r,n,-1,1),this.indexArray.emplaceBack(a,a+1,a+2),this.indexArray.emplaceBack(a,a+3,a+2),i.vertexLength+=4,i.primitiveLength+=2;}this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length,t,r,{},n);}}function us(t,e){for(let r=0;r1){if(fs(t,e))return !0;for(let n=0;n1?r:r.sub(e)._mult(i)._add(e))}function gs(t,e){let r,n,i,a=!1;for(let s=0;se.y!=i.y>e.y&&e.x<(i.x-n.x)*(e.y-n.y)/(i.y-n.y)+n.x&&(a=!a);}return a}function xs(t,e){let r=!1;for(let n=0,i=t.length-1;ne.y!=s.y>e.y&&e.x<(s.x-a.x)*(e.y-a.y)/(s.y-a.y)+a.x&&(r=!r);}return r}function vs(t,e,r){const n=r[0],i=r[2];if(t.xi.x&&e.x>i.x||t.yi.y&&e.y>i.y)return !1;const a=S(t,e,r[0]);return a!==S(t,e,r[1])||a!==S(t,e,r[2])||a!==S(t,e,r[3])}function bs(t,e,r){const n=e.paint.get(t).value;return "constant"===n.kind?n.value:r.programConfigurations.get(e.id).getMaxValue(t)}function ws(t){return Math.sqrt(t[0]*t[0]+t[1]*t[1])}function _s(t,e,r,n,i){if(!e[0]&&!e[1])return t;const s=a.convert(e)._mult(i);"viewport"===r&&s._rotate(-n);const o=[];for(let e=0;eFs(t,e)))}(l,o),p=c?u*s:u;for(const t of n)for(const e of t){const t=c?e:Fs(e,o);let r=p;const n=Vs([],[e.x,e.y,0,1],o);if("viewport"===this.paint.get("circle-pitch-scale")&&"map"===this.paint.get("circle-pitch-alignment")?r*=n[3]/a.cameraToCenterDistance:"map"===this.paint.get("circle-pitch-scale")&&"viewport"===this.paint.get("circle-pitch-alignment")&&(r*=a.cameraToCenterDistance/n[3]),cs(h,t,r))return !0}return !1}}function Fs(t,e){const r=Vs([],[t.x,t.y,0,1],e);return new a(r[0]/r[3],r[1]/r[3])}class Ts extends ls{}let $s;$n("HeatmapBucket",Ts,{omit:["layers"]});var Ls={get paint(){return $s=$s||new _i({"heatmap-radius":new xi(X.paint_heatmap["heatmap-radius"]),"heatmap-weight":new xi(X.paint_heatmap["heatmap-weight"]),"heatmap-intensity":new gi(X.paint_heatmap["heatmap-intensity"]),"heatmap-color":new wi(X.paint_heatmap["heatmap-color"]),"heatmap-opacity":new gi(X.paint_heatmap["heatmap-opacity"])})}};function Ds(t,{width:e,height:r},n,i){if(i){if(i instanceof Uint8ClampedArray)i=new Uint8Array(i.buffer);else if(i.length!==e*r*n)throw new RangeError(`mismatched image size. expected: ${i.length} but got: ${e*r*n}`)}else i=new Uint8Array(e*r*n);return t.width=e,t.height=r,t.data=i,t}function Os(t,{width:e,height:r},n){if(e===t.width&&r===t.height)return;const i=Ds({},{width:e,height:r},n);Us(t,i,{x:0,y:0},{x:0,y:0},{width:Math.min(t.width,e),height:Math.min(t.height,r)},n),t.width=e,t.height=r,t.data=i.data;}function Us(t,e,r,n,i,a){if(0===i.width||0===i.height)return e;if(i.width>t.width||i.height>t.height||r.x>t.width-i.width||r.y>t.height-i.height)throw new RangeError("out of range source coordinates for image copy");if(i.width>e.width||i.height>e.height||n.x>e.width-i.width||n.y>e.height-i.height)throw new RangeError("out of range destination coordinates for image copy");const s=t.data,o=e.data;if(s===o)throw new Error("srcData equals dstData, so image is already copied");for(let l=0;l{e[t.evaluationKey]=a;const s=t.expression.evaluate(e);i.data[r+n+0]=Math.floor(255*s.r/s.a),i.data[r+n+1]=Math.floor(255*s.g/s.a),i.data[r+n+2]=Math.floor(255*s.b/s.a),i.data[r+n+3]=Math.floor(255*s.a);};if(t.clips)for(let e=0,i=0;e80*r){n=a=t[0],i=s=t[1];for(var d=r;da&&(a=o),l>s&&(s=l);u=0!==(u=Math.max(a-n,s-i))?32767/u:0;}return to(p,f,r,n,i,u,0),f}function Ws(t,e,r,n,i){var a,s;if(i===Ao(t,e,r,n)>0)for(a=e;a=e;a-=n)s=bo(a,t[a],t[a+1],s);return s&&fo(s,s.next)&&(wo(s),s=s.next),s}function Qs(t,e){if(!t)return t;e||(e=t);var r,n=t;do{if(r=!1,n.steiner||!fo(n,n.next)&&0!==po(n.prev,n,n.next))n=n.next;else {if(wo(n),(n=e=n.prev)===n.next)break;r=!0;}}while(r||n!==e);return e}function to(t,e,r,n,i,a,s){if(t){!s&&a&&function(t,e,r,n){var i=t;do{0===i.z&&(i.z=lo(i.x,i.y,e,r,n)),i.prevZ=i.prev,i.nextZ=i.next,i=i.next;}while(i!==t);i.prevZ.nextZ=null,i.prevZ=null,function(t){var e,r,n,i,a,s,o,l,u=1;do{for(r=t,t=null,a=null,s=0;r;){for(s++,n=r,o=0,e=0;e0||l>0&&n;)0!==o&&(0===l||!n||r.z<=n.z)?(i=r,r=r.nextZ,o--):(i=n,n=n.nextZ,l--),a?a.nextZ=i:t=i,i.prevZ=a,a=i;r=n;}a.nextZ=null,u*=2;}while(s>1)}(i);}(t,n,i,a);for(var o,l,u=t;t.prev!==t.next;)if(o=t.prev,l=t.next,a?ro(t,n,i,a):eo(t))e.push(o.i/r|0),e.push(t.i/r|0),e.push(l.i/r|0),wo(t),t=l.next,u=l.next;else if((t=l)===u){s?1===s?to(t=no(Qs(t),e,r),e,r,n,i,a,2):2===s&&io(t,e,r,n,i,a):to(Qs(t),e,r,n,i,a,1);break}}}function eo(t){var e=t.prev,r=t,n=t.next;if(po(e,r,n)>=0)return !1;for(var i=e.x,a=r.x,s=n.x,o=e.y,l=r.y,u=n.y,c=ia?i>s?i:s:a>s?a:s,f=o>l?o>u?o:u:l>u?l:u,d=n.next;d!==e;){if(d.x>=c&&d.x<=p&&d.y>=h&&d.y<=f&&co(i,o,a,l,s,u,d.x,d.y)&&po(d.prev,d,d.next)>=0)return !1;d=d.next;}return !0}function ro(t,e,r,n){var i=t.prev,a=t,s=t.next;if(po(i,a,s)>=0)return !1;for(var o=i.x,l=a.x,u=s.x,c=i.y,h=a.y,p=s.y,f=ol?o>u?o:u:l>u?l:u,m=c>h?c>p?c:p:h>p?h:p,g=lo(f,d,e,r,n),x=lo(y,m,e,r,n),v=t.prevZ,b=t.nextZ;v&&v.z>=g&&b&&b.z<=x;){if(v.x>=f&&v.x<=y&&v.y>=d&&v.y<=m&&v!==i&&v!==s&&co(o,c,l,h,u,p,v.x,v.y)&&po(v.prev,v,v.next)>=0)return !1;if(v=v.prevZ,b.x>=f&&b.x<=y&&b.y>=d&&b.y<=m&&b!==i&&b!==s&&co(o,c,l,h,u,p,b.x,b.y)&&po(b.prev,b,b.next)>=0)return !1;b=b.nextZ;}for(;v&&v.z>=g;){if(v.x>=f&&v.x<=y&&v.y>=d&&v.y<=m&&v!==i&&v!==s&&co(o,c,l,h,u,p,v.x,v.y)&&po(v.prev,v,v.next)>=0)return !1;v=v.prevZ;}for(;b&&b.z<=x;){if(b.x>=f&&b.x<=y&&b.y>=d&&b.y<=m&&b!==i&&b!==s&&co(o,c,l,h,u,p,b.x,b.y)&&po(b.prev,b,b.next)>=0)return !1;b=b.nextZ;}return !0}function no(t,e,r){var n=t;do{var i=n.prev,a=n.next.next;!fo(i,a)&&yo(i,n,n.next,a)&&xo(i,a)&&xo(a,i)&&(e.push(i.i/r|0),e.push(n.i/r|0),e.push(a.i/r|0),wo(n),wo(n.next),n=t=a),n=n.next;}while(n!==t);return Qs(n)}function io(t,e,r,n,i,a){var s=t;do{for(var o=s.next.next;o!==s.prev;){if(s.i!==o.i&&ho(s,o)){var l=vo(s,o);return s=Qs(s,s.next),l=Qs(l,l.next),to(s,e,r,n,i,a,0),void to(l,e,r,n,i,a,0)}o=o.next;}s=s.next;}while(s!==t)}function ao(t,e){return t.x-e.x}function so(t,e){var r=function(t,e){var r,n=e,i=t.x,a=t.y,s=-1/0;do{if(a<=n.y&&a>=n.next.y&&n.next.y!==n.y){var o=n.x+(a-n.y)*(n.next.x-n.x)/(n.next.y-n.y);if(o<=i&&o>s&&(s=o,r=n.x=n.x&&n.x>=c&&i!==n.x&&co(ar.x||n.x===r.x&&oo(r,n)))&&(r=n,p=l)),n=n.next;}while(n!==u);return r}(t,e);if(!r)return e;var n=vo(r,t);return Qs(n,n.next),Qs(r,r.next)}function oo(t,e){return po(t.prev,t,e.prev)<0&&po(e.next,t,t.next)<0}function lo(t,e,r,n,i){return (t=1431655765&((t=858993459&((t=252645135&((t=16711935&((t=(t-r)*i|0)|t<<8))|t<<4))|t<<2))|t<<1))|(e=1431655765&((e=858993459&((e=252645135&((e=16711935&((e=(e-n)*i|0)|e<<8))|e<<4))|e<<2))|e<<1))<<1}function uo(t){var e=t,r=t;do{(e.x=(t-s)*(a-o)&&(t-s)*(n-o)>=(r-s)*(e-o)&&(r-s)*(a-o)>=(i-s)*(n-o)}function ho(t,e){return t.next.i!==e.i&&t.prev.i!==e.i&&!function(t,e){var r=t;do{if(r.i!==t.i&&r.next.i!==t.i&&r.i!==e.i&&r.next.i!==e.i&&yo(r,r.next,t,e))return !0;r=r.next;}while(r!==t);return !1}(t,e)&&(xo(t,e)&&xo(e,t)&&function(t,e){var r=t,n=!1,i=(t.x+e.x)/2,a=(t.y+e.y)/2;do{r.y>a!=r.next.y>a&&r.next.y!==r.y&&i<(r.next.x-r.x)*(a-r.y)/(r.next.y-r.y)+r.x&&(n=!n),r=r.next;}while(r!==t);return n}(t,e)&&(po(t.prev,t,e.prev)||po(t,e.prev,e))||fo(t,e)&&po(t.prev,t,t.next)>0&&po(e.prev,e,e.next)>0)}function po(t,e,r){return (e.y-t.y)*(r.x-e.x)-(e.x-t.x)*(r.y-e.y)}function fo(t,e){return t.x===e.x&&t.y===e.y}function yo(t,e,r,n){var i=go(po(t,e,r)),a=go(po(t,e,n)),s=go(po(r,n,t)),o=go(po(r,n,e));return i!==a&&s!==o||!(0!==i||!mo(t,r,e))||!(0!==a||!mo(t,n,e))||!(0!==s||!mo(r,t,n))||!(0!==o||!mo(r,e,n))}function mo(t,e,r){return e.x<=Math.max(t.x,r.x)&&e.x>=Math.min(t.x,r.x)&&e.y<=Math.max(t.y,r.y)&&e.y>=Math.min(t.y,r.y)}function go(t){return t>0?1:t<0?-1:0}function xo(t,e){return po(t.prev,t,t.next)<0?po(t,e,t.next)>=0&&po(t,t.prev,e)>=0:po(t,e,t.prev)<0||po(t,t.next,e)<0}function vo(t,e){var r=new _o(t.i,t.x,t.y),n=new _o(e.i,e.x,e.y),i=t.next,a=e.prev;return t.next=e,e.prev=t,r.next=i,i.prev=r,n.next=r,r.prev=n,a.next=n,n.prev=a,n}function bo(t,e,r,n){var i=new _o(t,e,r);return n?(i.next=n.next,i.prev=n,n.next.prev=i,n.next=i):(i.prev=i,i.next=i),i}function wo(t){t.next.prev=t.prev,t.prev.next=t.next,t.prevZ&&(t.prevZ.nextZ=t.nextZ),t.nextZ&&(t.nextZ.prevZ=t.prevZ);}function _o(t,e,r){this.i=t,this.x=e,this.y=r,this.prev=null,this.next=null,this.z=0,this.prevZ=null,this.nextZ=null,this.steiner=!1;}function Ao(t,e,r,n){for(var i=0,a=e,s=r-n;a0&&r.holes.push(n+=t[i-1].length);}return r};var So=r(Ys.exports);function ko(t,e,r,n,i){Io(t,e,r||0,n||t.length-1,i||Mo);}function Io(t,e,r,n,i){for(;n>r;){if(n-r>600){var a=n-r+1,s=e-r+1,o=Math.log(a),l=.5*Math.exp(2*o/3),u=.5*Math.sqrt(o*l*(a-l)/a)*(s-a/2<0?-1:1);Io(t,e,Math.max(r,Math.floor(e-s*l/a+u)),Math.min(n,Math.floor(e+(a-s)*l/a+u)),i);}var c=t[e],h=r,p=n;for(zo(t,r,e),i(t[n],c)>0&&zo(t,r,n);h0;)p--;}0===i(t[r],c)?zo(t,r,p):zo(t,++p,n),p<=e&&(r=p+1),e<=p&&(n=p-1);}}function zo(t,e,r){var n=t[e];t[e]=t[r],t[r]=n;}function Mo(t,e){return te?1:0}function Co(t,e){const r=t.length;if(r<=1)return [t];const n=[];let i,a;for(let e=0;e1)for(let t=0;tt.id)),this.index=t.index,this.hasPattern=!1,this.patternFeatures=[],this.layoutVertexArray=new fa,this.indexArray=new Aa,this.indexArray2=new Sa,this.programConfigurations=new Qa(t.layers,t.zoom),this.segments=new za,this.segments2=new za,this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id));}populate(t,e,r){this.hasPattern=Po("fill",this.layers,e);const n=this.layers[0].layout.get("fill-sort-key"),i=!n.isConstant(),a=[];for(const{feature:s,id:o,index:l,sourceLayerIndex:u}of t){const t=this.layers[0]._featureFilter.needGeometry,c=ss(s,t);if(!this.layers[0]._featureFilter.filter(new li(this.zoom),c,r))continue;const h=i?n.evaluate(c,{},r,e.availableImages):void 0,p={id:o,properties:s.properties,type:s.type,sourceLayerIndex:u,index:l,geometry:t?c.geometry:as(s),patterns:{},sortKey:h};a.push(p);}i&&a.sort(((t,e)=>t.sortKey-e.sortKey));for(const n of a){const{geometry:i,index:a,sourceLayerIndex:s}=n;if(this.hasPattern){const t=Vo("fill",this.layers,n,this.zoom,e);this.patternFeatures.push(t);}else this.addFeature(n,i,a,r,{});e.featureIndex.insert(t[a].feature,i,a,s,this.index);}}update(t,e,r){this.stateDependentLayers.length&&this.programConfigurations.updatePaintArrays(t,e,this.stateDependentLayers,r);}addFeatures(t,e,r){for(const t of this.patternFeatures)this.addFeature(t,t.geometry,t.index,e,r);}isEmpty(){return 0===this.layoutVertexArray.length}uploadPending(){return !this.uploaded||this.programConfigurations.needsUpload}upload(t){this.uploaded||(this.layoutVertexBuffer=t.createVertexBuffer(this.layoutVertexArray,Xs),this.indexBuffer=t.createIndexBuffer(this.indexArray),this.indexBuffer2=t.createIndexBuffer(this.indexArray2)),this.programConfigurations.upload(t),this.uploaded=!0;}destroy(){this.layoutVertexBuffer&&(this.layoutVertexBuffer.destroy(),this.indexBuffer.destroy(),this.indexBuffer2.destroy(),this.programConfigurations.destroy(),this.segments.destroy(),this.segments2.destroy());}addFeature(t,e,r,n,i){for(const t of Co(e,500)){let e=0;for(const r of t)e+=r.length;const r=this.segments.prepareSegment(e,this.layoutVertexArray,this.indexArray),n=r.vertexLength,i=[],a=[];for(const e of t){if(0===e.length)continue;e!==t[0]&&a.push(i.length/2);const r=this.segments2.prepareSegment(e.length,this.layoutVertexArray,this.indexArray2),n=r.vertexLength;this.layoutVertexArray.emplaceBack(e[0].x,e[0].y),this.indexArray2.emplaceBack(n+e.length-1,n),i.push(e[0].x),i.push(e[0].y);for(let t=1;t>3;}if(i--,1===n||2===n)a+=t.readSVarint(),s+=t.readSVarint(),1===n&&(e&&o.push(e),e=[]),e.push(new qo(a,s));else {if(7!==n)throw new Error("unknown command "+n);e&&e.push(e[0].clone());}}return e&&o.push(e),o},No.prototype.bbox=function(){var t=this._pbf;t.pos=this._geometry;for(var e=t.readVarint()+t.pos,r=1,n=0,i=0,a=0,s=1/0,o=-1/0,l=1/0,u=-1/0;t.pos>3;}if(n--,1===r||2===r)(i+=t.readSVarint())o&&(o=i),(a+=t.readSVarint())u&&(u=a);else if(7!==r)throw new Error("unknown command "+r)}return [s,l,o,u]},No.prototype.toGeoJSON=function(t,e,r){var n,i,a=this.extent*Math.pow(2,r),s=this.extent*t,o=this.extent*e,l=this.loadGeometry(),u=No.types[this.type];function c(t){for(var e=0;e>3;e=1===n?t.readString():2===n?t.readFloat():3===n?t.readDouble():4===n?t.readVarint64():5===n?t.readVarint():6===n?t.readSVarint():7===n?t.readBoolean():null;}return e}(r));}Xo.prototype.feature=function(t){if(t<0||t>=this._features.length)throw new Error("feature index out of bounds");this._pbf.pos=this._features[t];var e=this._pbf.readVarint()+this._pbf.pos;return new Go(this._pbf,e,this.extent,this._keys,this._values)};var Ho=Jo;function Wo(t,e,r){if(3===t){var n=new Ho(r,r.readVarint()+r.pos);n.length&&(e[n.name]=n);}}Ro.VectorTile=function(t,e){this.layers=t.readFields(Wo,{},e);},Ro.VectorTileFeature=jo,Ro.VectorTileLayer=Jo;const Qo=Ro.VectorTileFeature.types,tl=Math.pow(2,13);function el(t,e,r,n,i,a,s,o){t.emplaceBack(e,r,2*Math.floor(n*tl)+s,i*tl*2,a*tl*2,Math.round(o));}class rl{constructor(t){this.zoom=t.zoom,this.overscaling=t.overscaling,this.layers=t.layers,this.layerIds=this.layers.map((t=>t.id)),this.index=t.index,this.hasPattern=!1,this.layoutVertexArray=new da,this.centroidVertexArray=new ha,this.indexArray=new Aa,this.programConfigurations=new Qa(t.layers,t.zoom),this.segments=new za,this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id));}populate(t,e,r){this.features=[],this.hasPattern=Po("fill-extrusion",this.layers,e);for(const{feature:n,id:i,index:a,sourceLayerIndex:s}of t){const t=this.layers[0]._featureFilter.needGeometry,o=ss(n,t);if(!this.layers[0]._featureFilter.filter(new li(this.zoom),o,r))continue;const l={id:i,sourceLayerIndex:s,index:a,geometry:t?o.geometry:as(n),properties:n.properties,type:n.type,patterns:{}};this.hasPattern?this.features.push(Vo("fill-extrusion",this.layers,l,this.zoom,e)):this.addFeature(l,l.geometry,a,r,{}),e.featureIndex.insert(n,l.geometry,a,s,this.index,!0);}}addFeatures(t,e,r){for(const t of this.features){const{geometry:n}=t;this.addFeature(t,n,t.index,e,r);}}update(t,e,r){this.stateDependentLayers.length&&this.programConfigurations.updatePaintArrays(t,e,this.stateDependentLayers,r);}isEmpty(){return 0===this.layoutVertexArray.length&&0===this.centroidVertexArray.length}uploadPending(){return !this.uploaded||this.programConfigurations.needsUpload}upload(t){this.uploaded||(this.layoutVertexBuffer=t.createVertexBuffer(this.layoutVertexArray,Uo),this.centroidVertexBuffer=t.createVertexBuffer(this.centroidVertexArray,Oo.members,!0),this.indexBuffer=t.createIndexBuffer(this.indexArray)),this.programConfigurations.upload(t),this.uploaded=!0;}destroy(){this.layoutVertexBuffer&&(this.layoutVertexBuffer.destroy(),this.indexBuffer.destroy(),this.programConfigurations.destroy(),this.segments.destroy(),this.centroidVertexBuffer.destroy());}addFeature(t,e,r,n,i){const a={x:0,y:0,vertexCount:0};for(const r of Co(e,500)){let e=0;for(const t of r)e+=t.length;let n=this.segments.prepareSegment(4,this.layoutVertexArray,this.indexArray);for(const t of r){if(0===t.length)continue;if(il(t))continue;let e=0;for(let r=0;r=1){const s=t[r-1];if(!nl(i,s)){n.vertexLength+4>za.MAX_VERTEX_ARRAY_LENGTH&&(n=this.segments.prepareSegment(4,this.layoutVertexArray,this.indexArray));const t=i.sub(s)._perp()._unit(),r=s.dist(i);e+r>32768&&(e=0),el(this.layoutVertexArray,i.x,i.y,t.x,t.y,0,0,e),el(this.layoutVertexArray,i.x,i.y,t.x,t.y,0,1,e),a.x+=2*i.x,a.y+=2*i.y,a.vertexCount+=2,e+=r,el(this.layoutVertexArray,s.x,s.y,t.x,t.y,0,0,e),el(this.layoutVertexArray,s.x,s.y,t.x,t.y,0,1,e),a.x+=2*s.x,a.y+=2*s.y,a.vertexCount+=2;const o=n.vertexLength;this.indexArray.emplaceBack(o,o+2,o+1),this.indexArray.emplaceBack(o+1,o+2,o+3),n.vertexLength+=4,n.primitiveLength+=2;}}}}if(n.vertexLength+e>za.MAX_VERTEX_ARRAY_LENGTH&&(n=this.segments.prepareSegment(e,this.layoutVertexArray,this.indexArray)),"Polygon"!==Qo[t.type])continue;const i=[],s=[],o=n.vertexLength;for(const t of r)if(0!==t.length){t!==r[0]&&s.push(i.length/2);for(let e=0;ers)||t.y===e.y&&(t.y<0||t.y>rs)}function il(t){return t.every((t=>t.x<0))||t.every((t=>t.x>rs))||t.every((t=>t.y<0))||t.every((t=>t.y>rs))}let al;$n("FillExtrusionBucket",rl,{omit:["layers","features"]});var sl={get paint(){return al=al||new _i({"fill-extrusion-opacity":new gi(X["paint_fill-extrusion"]["fill-extrusion-opacity"]),"fill-extrusion-color":new xi(X["paint_fill-extrusion"]["fill-extrusion-color"]),"fill-extrusion-translate":new gi(X["paint_fill-extrusion"]["fill-extrusion-translate"]),"fill-extrusion-translate-anchor":new gi(X["paint_fill-extrusion"]["fill-extrusion-translate-anchor"]),"fill-extrusion-pattern":new vi(X["paint_fill-extrusion"]["fill-extrusion-pattern"]),"fill-extrusion-height":new xi(X["paint_fill-extrusion"]["fill-extrusion-height"]),"fill-extrusion-base":new xi(X["paint_fill-extrusion"]["fill-extrusion-base"]),"fill-extrusion-vertical-gradient":new gi(X["paint_fill-extrusion"]["fill-extrusion-vertical-gradient"])})}};class ol extends Si{constructor(t){super(t,sl);}createBucket(t){return new rl(t)}queryRadius(){return ws(this.paint.get("fill-extrusion-translate"))}is3D(){return !0}queryIntersectsFeature(t,e,r,n,i,s,o,l){const u=_s(t,this.paint.get("fill-extrusion-translate"),this.paint.get("fill-extrusion-translate-anchor"),s.angle,o),c=this.paint.get("fill-extrusion-height").evaluate(e,r),h=this.paint.get("fill-extrusion-base").evaluate(e,r),p=function(t,e,r,n){const i=[];for(const r of t){const t=[r.x,r.y,0,1];Vs(t,t,e),i.push(new a(t[0]/t[3],t[1]/t[3]));}return i}(u,l),f=function(t,e,r,n){const i=[],s=[],o=n[8]*e,l=n[9]*e,u=n[10]*e,c=n[11]*e,h=n[8]*r,p=n[9]*r,f=n[10]*r,d=n[11]*r;for(const e of t){const t=[],r=[];for(const i of e){const e=i.x,s=i.y,y=n[0]*e+n[4]*s+n[12],m=n[1]*e+n[5]*s+n[13],g=n[2]*e+n[6]*s+n[14],x=n[3]*e+n[7]*s+n[15],v=g+u,b=x+c,w=y+h,_=m+p,A=g+f,S=x+d,k=new a((y+o)/b,(m+l)/b);k.z=v/b,t.push(k);const I=new a(w/S,_/S);I.z=A/S,r.push(I);}i.push(t),s.push(r);}return [i,s]}(n,h,c,l);return function(t,e,r){let n=1/0;hs(r,e)&&(n=ul(r,e[0]));for(let i=0;it.id)),this.index=t.index,this.hasPattern=!1,this.patternFeatures=[],this.lineClipsArray=[],this.gradients={},this.layers.forEach((t=>{this.gradients[t.id]={};})),this.layoutVertexArray=new ya,this.layoutVertexArray2=new ma,this.indexArray=new Aa,this.programConfigurations=new Qa(t.layers,t.zoom),this.segments=new za,this.maxLineLength=0,this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id));}populate(t,e,r){this.hasPattern=Po("line",this.layers,e);const n=this.layers[0].layout.get("line-sort-key"),i=!n.isConstant(),a=[];for(const{feature:e,id:s,index:o,sourceLayerIndex:l}of t){const t=this.layers[0]._featureFilter.needGeometry,u=ss(e,t);if(!this.layers[0]._featureFilter.filter(new li(this.zoom),u,r))continue;const c=i?n.evaluate(u,{},r):void 0,h={id:s,properties:e.properties,type:e.type,sourceLayerIndex:l,index:o,geometry:t?u.geometry:as(e),patterns:{},sortKey:c};a.push(h);}i&&a.sort(((t,e)=>t.sortKey-e.sortKey));for(const n of a){const{geometry:i,index:a,sourceLayerIndex:s}=n;if(this.hasPattern){const t=Vo("line",this.layers,n,this.zoom,e);this.patternFeatures.push(t);}else this.addFeature(n,i,a,r,{});e.featureIndex.insert(t[a].feature,i,a,s,this.index);}}update(t,e,r){this.stateDependentLayers.length&&this.programConfigurations.updatePaintArrays(t,e,this.stateDependentLayers,r);}addFeatures(t,e,r){for(const t of this.patternFeatures)this.addFeature(t,t.geometry,t.index,e,r);}isEmpty(){return 0===this.layoutVertexArray.length}uploadPending(){return !this.uploaded||this.programConfigurations.needsUpload}upload(t){this.uploaded||(0!==this.layoutVertexArray2.length&&(this.layoutVertexBuffer2=t.createVertexBuffer(this.layoutVertexArray2,fl)),this.layoutVertexBuffer=t.createVertexBuffer(this.layoutVertexArray,hl),this.indexBuffer=t.createIndexBuffer(this.indexArray)),this.programConfigurations.upload(t),this.uploaded=!0;}destroy(){this.layoutVertexBuffer&&(this.layoutVertexBuffer.destroy(),this.indexBuffer.destroy(),this.programConfigurations.destroy(),this.segments.destroy());}lineFeatureClips(t){if(t.properties&&Object.prototype.hasOwnProperty.call(t.properties,"mapbox_clip_start")&&Object.prototype.hasOwnProperty.call(t.properties,"mapbox_clip_end"))return {start:+t.properties.mapbox_clip_start,end:+t.properties.mapbox_clip_end}}addFeature(t,e,r,n,i){const a=this.layers[0].layout,s=a.get("line-join").evaluate(t,{}),o=a.get("line-cap"),l=a.get("line-miter-limit"),u=a.get("line-round-limit");this.lineClips=this.lineFeatureClips(t);for(const r of e)this.addLine(r,t,s,o,l,u);this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length,t,r,i,n);}addLine(t,e,r,n,i,a){if(this.distance=0,this.scaledDistance=0,this.totalDistance=0,this.lineClips){this.lineClipsArray.push(this.lineClips);for(let e=0;e=2&&t[o-1].equals(t[o-2]);)o--;let l=0;for(;l0;if(w&&e>l){const t=h.dist(p);if(t>2*u){const e=h.sub(h.sub(p)._mult(u/t)._round());this.updateDistance(p,e),this.addCurrentVertex(e,d,0,0,c),p=e;}}const A=p&&f;let S=A?r:s?"butt":n;if(A&&"round"===S&&(vi&&(S="bevel"),"bevel"===S&&(v>2&&(S="flipbevel"),v100)m=y.mult(-1);else {const t=v*d.add(y).mag()/d.sub(y).mag();m._perp()._mult(t*(_?-1:1));}this.addCurrentVertex(h,m,0,0,c),this.addCurrentVertex(h,m.mult(-1),0,0,c);}else if("bevel"===S||"fakeround"===S){const t=-Math.sqrt(v*v-1),e=_?t:0,r=_?0:t;if(p&&this.addCurrentVertex(h,d,e,r,c),"fakeround"===S){const t=Math.round(180*b/Math.PI/20);for(let e=1;e2*u){const e=h.add(f.sub(h)._mult(u/t)._round());this.updateDistance(h,e),this.addCurrentVertex(e,y,0,0,c),h=e;}}}}addCurrentVertex(t,e,r,n,i,a=!1){const s=e.y*n-e.x,o=-e.y-e.x*n;this.addHalfVertex(t,e.x+e.y*r,e.y-e.x*r,a,!1,r,i),this.addHalfVertex(t,s,o,a,!0,-n,i),this.distance>ml/2&&0===this.totalDistance&&(this.distance=0,this.updateScaledDistance(),this.addCurrentVertex(t,e,r,n,i,a));}addHalfVertex({x:t,y:e},r,n,i,a,s,o){const l=.5*(this.lineClips?this.scaledDistance*(ml-1):this.scaledDistance);this.layoutVertexArray.emplaceBack((t<<1)+(i?1:0),(e<<1)+(a?1:0),Math.round(63*r)+128,Math.round(63*n)+128,1+(0===s?0:s<0?-1:1)|(63&l)<<2,l>>6),this.lineClips&&this.layoutVertexArray2.emplaceBack((this.scaledDistance-this.lineClips.start)/(this.lineClips.end-this.lineClips.start),this.lineClipsArray.length);const u=o.vertexLength++;this.e1>=0&&this.e2>=0&&(this.indexArray.emplaceBack(this.e1,this.e2,u),o.primitiveLength++),a?this.e2=u:this.e1=u;}updateScaledDistance(){this.scaledDistance=this.lineClips?this.lineClips.start+(this.lineClips.end-this.lineClips.start)*this.distance/this.totalDistance:this.distance;}updateDistance(t,e){this.distance+=t.dist(e),this.updateScaledDistance();}}let xl,vl;$n("LineBucket",gl,{omit:["layers","patternFeatures"]});var bl={get paint(){return vl=vl||new _i({"line-opacity":new xi(X.paint_line["line-opacity"]),"line-color":new xi(X.paint_line["line-color"]),"line-translate":new gi(X.paint_line["line-translate"]),"line-translate-anchor":new gi(X.paint_line["line-translate-anchor"]),"line-width":new xi(X.paint_line["line-width"]),"line-gap-width":new xi(X.paint_line["line-gap-width"]),"line-offset":new xi(X.paint_line["line-offset"]),"line-blur":new xi(X.paint_line["line-blur"]),"line-dasharray":new bi(X.paint_line["line-dasharray"]),"line-pattern":new vi(X.paint_line["line-pattern"]),"line-gradient":new wi(X.paint_line["line-gradient"])})},get layout(){return xl=xl||new _i({"line-cap":new gi(X.layout_line["line-cap"]),"line-join":new xi(X.layout_line["line-join"]),"line-miter-limit":new gi(X.layout_line["line-miter-limit"]),"line-round-limit":new gi(X.layout_line["line-round-limit"]),"line-sort-key":new xi(X.layout_line["line-sort-key"])})}};class wl extends xi{possiblyEvaluate(t,e){return e=new li(Math.floor(e.zoom),{now:e.now,fadeDuration:e.fadeDuration,zoomHistory:e.zoomHistory,transition:e.transition}),super.possiblyEvaluate(t,e)}evaluate(t,e,r,n){return e=g({},e,{zoom:Math.floor(e.zoom)}),super.evaluate(t,e,r,n)}}let _l;class Al extends Si{constructor(t){super(t,bl),this.gradientVersion=0,_l||(_l=new wl(bl.paint.properties["line-width"].specification),_l.useIntegerZoom=!0);}_handleSpecialPaintPropertyUpdate(t){if("line-gradient"===t){const t=this.gradientExpression();this.stepInterpolant=!!function(t){return void 0!==t._styleExpression}(t)&&t._styleExpression.expression instanceof qe,this.gradientVersion=(this.gradientVersion+1)%Number.MAX_SAFE_INTEGER;}}gradientExpression(){return this._transitionablePaint._values["line-gradient"].value.expression}recalculate(t,e){super.recalculate(t,e),this.paint._values["line-floorwidth"]=_l.possiblyEvaluate(this._transitioningPaint._values["line-width"].value,t);}createBucket(t){return new gl(t)}queryRadius(t){const e=t,r=Sl(bs("line-width",this,e),bs("line-gap-width",this,e)),n=bs("line-offset",this,e);return r/2+Math.abs(n)+ws(this.paint.get("line-translate"))}queryIntersectsFeature(t,e,r,n,i,s,o){const l=_s(t,this.paint.get("line-translate"),this.paint.get("line-translate-anchor"),s.angle,o),u=o/2*Sl(this.paint.get("line-width").evaluate(e,r),this.paint.get("line-gap-width").evaluate(e,r)),c=this.paint.get("line-offset").evaluate(e,r);return c&&(n=function(t,e){const r=[];for(let n=0;n=3)for(let e=0;e0?e+2*t:t}const kl=Mi([{name:"a_pos_offset",components:4,type:"Int16"},{name:"a_data",components:4,type:"Uint16"},{name:"a_pixeloffset",components:4,type:"Int16"}],4),Il=Mi([{name:"a_projected_pos",components:3,type:"Float32"}],4);Mi([{name:"a_fade_opacity",components:1,type:"Uint32"}],4);const zl=Mi([{name:"a_placed",components:2,type:"Uint8"},{name:"a_shift",components:2,type:"Float32"}]);Mi([{type:"Int16",name:"anchorPointX"},{type:"Int16",name:"anchorPointY"},{type:"Int16",name:"x1"},{type:"Int16",name:"y1"},{type:"Int16",name:"x2"},{type:"Int16",name:"y2"},{type:"Uint32",name:"featureIndex"},{type:"Uint16",name:"sourceLayerIndex"},{type:"Uint16",name:"bucketIndex"}]);const Ml=Mi([{name:"a_pos",components:2,type:"Int16"},{name:"a_anchor_pos",components:2,type:"Int16"},{name:"a_extrude",components:2,type:"Int16"}],4),Cl=Mi([{name:"a_pos",components:2,type:"Float32"},{name:"a_radius",components:1,type:"Float32"},{name:"a_flags",components:2,type:"Int16"}],4);function Bl(t,e,r){return t.sections.forEach((t=>{t.text=function(t,e,r){const n=e.layout.get("text-transform").evaluate(r,{});return "uppercase"===n?t=t.toLocaleUpperCase():"lowercase"===n&&(t=t.toLocaleLowerCase()),oi.applyArabicShaping&&(t=oi.applyArabicShaping(t)),t}(t.text,e,r);})),t}Mi([{name:"triangle",components:3,type:"Uint16"}]),Mi([{type:"Int16",name:"anchorX"},{type:"Int16",name:"anchorY"},{type:"Uint16",name:"glyphStartIndex"},{type:"Uint16",name:"numGlyphs"},{type:"Uint32",name:"vertexStartIndex"},{type:"Uint32",name:"lineStartIndex"},{type:"Uint32",name:"lineLength"},{type:"Uint16",name:"segment"},{type:"Uint16",name:"lowerSize"},{type:"Uint16",name:"upperSize"},{type:"Float32",name:"lineOffsetX"},{type:"Float32",name:"lineOffsetY"},{type:"Uint8",name:"writingMode"},{type:"Uint8",name:"placedOrientation"},{type:"Uint8",name:"hidden"},{type:"Uint32",name:"crossTileID"},{type:"Int16",name:"associatedIconIndex"}]),Mi([{type:"Int16",name:"anchorX"},{type:"Int16",name:"anchorY"},{type:"Int16",name:"rightJustifiedTextSymbolIndex"},{type:"Int16",name:"centerJustifiedTextSymbolIndex"},{type:"Int16",name:"leftJustifiedTextSymbolIndex"},{type:"Int16",name:"verticalPlacedTextSymbolIndex"},{type:"Int16",name:"placedIconSymbolIndex"},{type:"Int16",name:"verticalPlacedIconSymbolIndex"},{type:"Uint16",name:"key"},{type:"Uint16",name:"textBoxStartIndex"},{type:"Uint16",name:"textBoxEndIndex"},{type:"Uint16",name:"verticalTextBoxStartIndex"},{type:"Uint16",name:"verticalTextBoxEndIndex"},{type:"Uint16",name:"iconBoxStartIndex"},{type:"Uint16",name:"iconBoxEndIndex"},{type:"Uint16",name:"verticalIconBoxStartIndex"},{type:"Uint16",name:"verticalIconBoxEndIndex"},{type:"Uint16",name:"featureIndex"},{type:"Uint16",name:"numHorizontalGlyphVertices"},{type:"Uint16",name:"numVerticalGlyphVertices"},{type:"Uint16",name:"numIconVertices"},{type:"Uint16",name:"numVerticalIconVertices"},{type:"Uint16",name:"useRuntimeCollisionCircles"},{type:"Uint32",name:"crossTileID"},{type:"Float32",name:"textBoxScale"},{type:"Float32",name:"collisionCircleDiameter"},{type:"Uint16",name:"textAnchorOffsetStartIndex"},{type:"Uint16",name:"textAnchorOffsetEndIndex"}]),Mi([{type:"Float32",name:"offsetX"}]),Mi([{type:"Int16",name:"x"},{type:"Int16",name:"y"},{type:"Int16",name:"tileUnitDistanceFromAnchor"}]),Mi([{type:"Uint16",name:"textAnchor"},{type:"Float32",components:2,name:"textOffset"}]);const Pl={"!":"︕","#":"#",$:"$","%":"%","&":"&","(":"︵",")":"︶","*":"*","+":"+",",":"︐","-":"︲",".":"・","/":"/",":":"︓",";":"︔","<":"︿","=":"=",">":"﹀","?":"︖","@":"@","[":"﹇","\\":"\","]":"﹈","^":"^",_:"︳","`":"`","{":"︷","|":"―","}":"︸","~":"~","¢":"¢","£":"£","¥":"¥","¦":"¦","¬":"¬","¯":" ̄","–":"︲","—":"︱","‘":"﹃","’":"﹄","“":"﹁","”":"﹂","…":"︙","‧":"・","₩":"₩","、":"︑","。":"︒","〈":"︿","〉":"﹀","《":"︽","》":"︾","「":"﹁","」":"﹂","『":"﹃","』":"﹄","【":"︻","】":"︼","〔":"︹","〕":"︺","〖":"︗","〗":"︘","!":"︕","(":"︵",")":"︶",",":"︐","-":"︲",".":"・",":":"︓",";":"︔","<":"︿",">":"﹀","?":"︖","[":"﹇","]":"﹈","_":"︳","{":"︷","|":"―","}":"︸","⦅":"︵","⦆":"︶","。":"︒","「":"﹁","」":"﹂"};var Vl=24,El=$l,Fl=function(t,e,r,n,i){var a,s,o=8*i-n-1,l=(1<>1,c=-7,h=r?i-1:0,p=r?-1:1,f=t[e+h];for(h+=p,a=f&(1<<-c)-1,f>>=-c,c+=o;c>0;a=256*a+t[e+h],h+=p,c-=8);for(s=a&(1<<-c)-1,a>>=-c,c+=n;c>0;s=256*s+t[e+h],h+=p,c-=8);if(0===a)a=1-u;else {if(a===l)return s?NaN:1/0*(f?-1:1);s+=Math.pow(2,n),a-=u;}return (f?-1:1)*s*Math.pow(2,a-n)},Tl=function(t,e,r,n,i,a){var s,o,l,u=8*a-i-1,c=(1<>1,p=23===i?Math.pow(2,-24)-Math.pow(2,-77):0,f=n?0:a-1,d=n?1:-1,y=e<0||0===e&&1/e<0?1:0;for(e=Math.abs(e),isNaN(e)||e===1/0?(o=isNaN(e)?1:0,s=c):(s=Math.floor(Math.log(e)/Math.LN2),e*(l=Math.pow(2,-s))<1&&(s--,l*=2),(e+=s+h>=1?p/l:p*Math.pow(2,1-h))*l>=2&&(s++,l/=2),s+h>=c?(o=0,s=c):s+h>=1?(o=(e*l-1)*Math.pow(2,i),s+=h):(o=e*Math.pow(2,h-1)*Math.pow(2,i),s=0));i>=8;t[r+f]=255&o,f+=d,o/=256,i-=8);for(s=s<0;t[r+f]=255&s,f+=d,s/=256,u-=8);t[r+f-d]|=128*y;};function $l(t){this.buf=ArrayBuffer.isView&&ArrayBuffer.isView(t)?t:new Uint8Array(t||0),this.pos=0,this.type=0,this.length=this.buf.length;}$l.Varint=0,$l.Fixed64=1,$l.Bytes=2,$l.Fixed32=5;var Ll=4294967296,Dl=1/Ll,Ol="undefined"==typeof TextDecoder?null:new TextDecoder("utf8");function Ul(t){return t.type===$l.Bytes?t.readVarint()+t.pos:t.pos+1}function Rl(t,e,r){return r?4294967296*e+(t>>>0):4294967296*(e>>>0)+(t>>>0)}function ql(t,e,r){var n=e<=16383?1:e<=2097151?2:e<=268435455?3:Math.floor(Math.log(e)/(7*Math.LN2));r.realloc(n);for(var i=r.pos-1;i>=t;i--)r.buf[i+n]=r.buf[i];}function jl(t,e){for(var r=0;r>>8,t[r+2]=e>>>16,t[r+3]=e>>>24;}function tu(t,e){return (t[e]|t[e+1]<<8|t[e+2]<<16)+(t[e+3]<<24)}$l.prototype={destroy:function(){this.buf=null;},readFields:function(t,e,r){for(r=r||this.length;this.pos>3,a=this.pos;this.type=7&n,t(i,e,this),this.pos===a&&this.skip(n);}return e},readMessage:function(t,e){return this.readFields(t,e,this.readVarint()+this.pos)},readFixed32:function(){var t=Wl(this.buf,this.pos);return this.pos+=4,t},readSFixed32:function(){var t=tu(this.buf,this.pos);return this.pos+=4,t},readFixed64:function(){var t=Wl(this.buf,this.pos)+Wl(this.buf,this.pos+4)*Ll;return this.pos+=8,t},readSFixed64:function(){var t=Wl(this.buf,this.pos)+tu(this.buf,this.pos+4)*Ll;return this.pos+=8,t},readFloat:function(){var t=Fl(this.buf,this.pos,!0,23,4);return this.pos+=4,t},readDouble:function(){var t=Fl(this.buf,this.pos,!0,52,8);return this.pos+=8,t},readVarint:function(t){var e,r,n=this.buf;return e=127&(r=n[this.pos++]),r<128?e:(e|=(127&(r=n[this.pos++]))<<7,r<128?e:(e|=(127&(r=n[this.pos++]))<<14,r<128?e:(e|=(127&(r=n[this.pos++]))<<21,r<128?e:function(t,e,r){var n,i,a=r.buf;if(n=(112&(i=a[r.pos++]))>>4,i<128)return Rl(t,n,e);if(n|=(127&(i=a[r.pos++]))<<3,i<128)return Rl(t,n,e);if(n|=(127&(i=a[r.pos++]))<<10,i<128)return Rl(t,n,e);if(n|=(127&(i=a[r.pos++]))<<17,i<128)return Rl(t,n,e);if(n|=(127&(i=a[r.pos++]))<<24,i<128)return Rl(t,n,e);if(n|=(1&(i=a[r.pos++]))<<31,i<128)return Rl(t,n,e);throw new Error("Expected varint not more than 10 bytes")}(e|=(15&(r=n[this.pos]))<<28,t,this))))},readVarint64:function(){return this.readVarint(!0)},readSVarint:function(){var t=this.readVarint();return t%2==1?(t+1)/-2:t/2},readBoolean:function(){return Boolean(this.readVarint())},readString:function(){var t=this.readVarint()+this.pos,e=this.pos;return this.pos=t,t-e>=12&&Ol?function(t,e,r){return Ol.decode(t.subarray(e,r))}(this.buf,e,t):function(t,e,r){for(var n="",i=e;i239?4:l>223?3:l>191?2:1;if(i+c>r)break;1===c?l<128&&(u=l):2===c?128==(192&(a=t[i+1]))&&(u=(31&l)<<6|63&a)<=127&&(u=null):3===c?(s=t[i+2],128==(192&(a=t[i+1]))&&128==(192&s)&&((u=(15&l)<<12|(63&a)<<6|63&s)<=2047||u>=55296&&u<=57343)&&(u=null)):4===c&&(s=t[i+2],o=t[i+3],128==(192&(a=t[i+1]))&&128==(192&s)&&128==(192&o)&&((u=(15&l)<<18|(63&a)<<12|(63&s)<<6|63&o)<=65535||u>=1114112)&&(u=null)),null===u?(u=65533,c=1):u>65535&&(u-=65536,n+=String.fromCharCode(u>>>10&1023|55296),u=56320|1023&u),n+=String.fromCharCode(u),i+=c;}return n}(this.buf,e,t)},readBytes:function(){var t=this.readVarint()+this.pos,e=this.buf.subarray(this.pos,t);return this.pos=t,e},readPackedVarint:function(t,e){if(this.type!==$l.Bytes)return t.push(this.readVarint(e));var r=Ul(this);for(t=t||[];this.pos127;);else if(e===$l.Bytes)this.pos=this.readVarint()+this.pos;else if(e===$l.Fixed32)this.pos+=4;else {if(e!==$l.Fixed64)throw new Error("Unimplemented type: "+e);this.pos+=8;}},writeTag:function(t,e){this.writeVarint(t<<3|e);},realloc:function(t){for(var e=this.length||16;e268435455||t<0?function(t,e){var r,n;if(t>=0?(r=t%4294967296|0,n=t/4294967296|0):(n=~(-t/4294967296),4294967295^(r=~(-t%4294967296))?r=r+1|0:(r=0,n=n+1|0)),t>=0x10000000000000000||t<-0x10000000000000000)throw new Error("Given varint doesn't fit into 10 bytes");e.realloc(10),function(t,e,r){r.buf[r.pos++]=127&t|128,t>>>=7,r.buf[r.pos++]=127&t|128,t>>>=7,r.buf[r.pos++]=127&t|128,t>>>=7,r.buf[r.pos++]=127&t|128,r.buf[r.pos]=127&(t>>>=7);}(r,0,e),function(t,e){var r=(7&t)<<4;e.buf[e.pos++]|=r|((t>>>=3)?128:0),t&&(e.buf[e.pos++]=127&t|((t>>>=7)?128:0),t&&(e.buf[e.pos++]=127&t|((t>>>=7)?128:0),t&&(e.buf[e.pos++]=127&t|((t>>>=7)?128:0),t&&(e.buf[e.pos++]=127&t|((t>>>=7)?128:0),t&&(e.buf[e.pos++]=127&t)))));}(n,e);}(t,this):(this.realloc(4),this.buf[this.pos++]=127&t|(t>127?128:0),t<=127||(this.buf[this.pos++]=127&(t>>>=7)|(t>127?128:0),t<=127||(this.buf[this.pos++]=127&(t>>>=7)|(t>127?128:0),t<=127||(this.buf[this.pos++]=t>>>7&127))));},writeSVarint:function(t){this.writeVarint(t<0?2*-t-1:2*t);},writeBoolean:function(t){this.writeVarint(Boolean(t));},writeString:function(t){t=String(t),this.realloc(4*t.length),this.pos++;var e=this.pos;this.pos=function(t,e,r){for(var n,i,a=0;a55295&&n<57344){if(!i){n>56319||a+1===e.length?(t[r++]=239,t[r++]=191,t[r++]=189):i=n;continue}if(n<56320){t[r++]=239,t[r++]=191,t[r++]=189,i=n;continue}n=i-55296<<10|n-56320|65536,i=null;}else i&&(t[r++]=239,t[r++]=191,t[r++]=189,i=null);n<128?t[r++]=n:(n<2048?t[r++]=n>>6|192:(n<65536?t[r++]=n>>12|224:(t[r++]=n>>18|240,t[r++]=n>>12&63|128),t[r++]=n>>6&63|128),t[r++]=63&n|128);}return r}(this.buf,t,this.pos);var r=this.pos-e;r>=128&&ql(e,r,this),this.pos=e-1,this.writeVarint(r),this.pos+=r;},writeFloat:function(t){this.realloc(4),Tl(this.buf,t,this.pos,!0,23,4),this.pos+=4;},writeDouble:function(t){this.realloc(8),Tl(this.buf,t,this.pos,!0,52,8),this.pos+=8;},writeBytes:function(t){var e=t.length;this.writeVarint(e),this.realloc(e);for(var r=0;r=128&&ql(r,n,this),this.pos=r-1,this.writeVarint(n),this.pos+=n;},writeMessage:function(t,e,r){this.writeTag(t,$l.Bytes),this.writeRawMessage(e,r);},writePackedVarint:function(t,e){e.length&&this.writeMessage(t,jl,e);},writePackedSVarint:function(t,e){e.length&&this.writeMessage(t,Nl,e);},writePackedBoolean:function(t,e){e.length&&this.writeMessage(t,Gl,e);},writePackedFloat:function(t,e){e.length&&this.writeMessage(t,Zl,e);},writePackedDouble:function(t,e){e.length&&this.writeMessage(t,Kl,e);},writePackedFixed32:function(t,e){e.length&&this.writeMessage(t,Jl,e);},writePackedSFixed32:function(t,e){e.length&&this.writeMessage(t,Xl,e);},writePackedFixed64:function(t,e){e.length&&this.writeMessage(t,Yl,e);},writePackedSFixed64:function(t,e){e.length&&this.writeMessage(t,Hl,e);},writeBytesField:function(t,e){this.writeTag(t,$l.Bytes),this.writeBytes(e);},writeFixed32Field:function(t,e){this.writeTag(t,$l.Fixed32),this.writeFixed32(e);},writeSFixed32Field:function(t,e){this.writeTag(t,$l.Fixed32),this.writeSFixed32(e);},writeFixed64Field:function(t,e){this.writeTag(t,$l.Fixed64),this.writeFixed64(e);},writeSFixed64Field:function(t,e){this.writeTag(t,$l.Fixed64),this.writeSFixed64(e);},writeVarintField:function(t,e){this.writeTag(t,$l.Varint),this.writeVarint(e);},writeSVarintField:function(t,e){this.writeTag(t,$l.Varint),this.writeSVarint(e);},writeStringField:function(t,e){this.writeTag(t,$l.Bytes),this.writeString(e);},writeFloatField:function(t,e){this.writeTag(t,$l.Fixed32),this.writeFloat(e);},writeDoubleField:function(t,e){this.writeTag(t,$l.Fixed64),this.writeDouble(e);},writeBooleanField:function(t,e){this.writeVarintField(t,Boolean(e));}};var eu=r(El);const ru=3;function nu(t,e,r){1===t&&r.readMessage(iu,e);}function iu(t,e,r){if(3===t){const{id:t,bitmap:n,width:i,height:a,left:s,top:o,advance:l}=r.readMessage(au,{});e.push({id:t,bitmap:new Rs({width:i+2*ru,height:a+2*ru},n),metrics:{width:i,height:a,left:s,top:o,advance:l}});}}function au(t,e,r){1===t?e.id=r.readVarint():2===t?e.bitmap=r.readBytes():3===t?e.width=r.readVarint():4===t?e.height=r.readVarint():5===t?e.left=r.readSVarint():6===t?e.top=r.readSVarint():7===t&&(e.advance=r.readVarint());}const su=ru;function ou(t){let e=0,r=0;for(const n of t)e+=n.w*n.h,r=Math.max(r,n.w);t.sort(((t,e)=>e.h-t.h));const n=[{x:0,y:0,w:Math.max(Math.ceil(Math.sqrt(e/.95)),r),h:1/0}];let i=0,a=0;for(const e of t)for(let t=n.length-1;t>=0;t--){const r=n[t];if(!(e.w>r.w||e.h>r.h)){if(e.x=r.x,e.y=r.y,a=Math.max(a,e.y+e.h),i=Math.max(i,e.x+e.w),e.w===r.w&&e.h===r.h){const e=n.pop();t=0&&r>=t&&mu[this.text.charCodeAt(r)];r--)e--;this.text=this.text.substring(t,e),this.sectionIndex=this.sectionIndex.slice(t,e);}substring(t,e){const r=new du;return r.text=this.text.substring(t,e),r.sectionIndex=this.sectionIndex.slice(t,e),r.sections=this.sections,r}toString(){return this.text}getMaxScale(){return this.sectionIndex.reduce(((t,e)=>Math.max(t,this.sections[e].scale)),0)}addTextSection(t,e){this.text+=t.text,this.sections.push(fu.forText(t.scale,t.fontStack||e));const r=this.sections.length-1;for(let e=0;e=63743?null:++this.imageSectionID:(this.imageSectionID=57344,this.imageSectionID)}}function yu(e,r,n,i,a,s,o,l,u,c,h,p,f,d,y,m){const g=du.fromFeature(e,a);let x;p===t.ai.vertical&&g.verticalizePunctuation();const{processBidirectionalText:v,processStyledBidirectionalText:b}=oi;if(v&&1===g.sections.length){x=[];const t=v(g.toString(),Au(g,c,s,r,i,d,y));for(const e of t){const t=new du;t.text=e,t.sections=g.sections;for(let r=0;r0&&n>_&&(_=n);}else {const t=n[y.fontStack],e=t&&t[g];if(e&&e.rect)A=e.rect,b=e.metrics;else {const t=r[y.fontStack],e=t&&t[g];if(!e)continue;b=e.metrics;}x=(a-y.scale)*Vl;}I?(e.verticalizable=!0,w.push({glyph:g,imageName:S,x:f,y:d+x,vertical:I,scale:y.scale,fontStack:y.fontStack,sectionIndex:m,metrics:b,rect:A}),f+=k*y.scale+c):(w.push({glyph:g,imageName:S,x:f,y:d+x,vertical:I,scale:y.scale,fontStack:y.fontStack,sectionIndex:m,metrics:b,rect:A}),f+=b.advance*y.scale+c);}0!==w.length&&(y=Math.max(f-c,y),ku(w,0,w.length-1,g,_)),f=0;const A=s*a+_;b.lineOffset=Math.max(_,l),d+=A,m=Math.max(A,m),++x;}var v;const b=d-pu,{horizontalAlign:w,verticalAlign:_}=Su(o);((function(t,e,r,n,i,a,s,o,l){const u=(e-r)*i;let c=0;c=a!==s?-o*n-pu:(-n*l+.5)*s;for(const e of t)for(const t of e.positionedGlyphs)t.x+=u,t.y+=c;}))(e.positionedLines,g,w,_,y,m,s,b,a.length),e.top+=-_*b,e.bottom=e.top+b,e.left+=-w*y,e.right=e.left+y;}(_,r,n,i,x,o,l,u,p,c,f,m),!function(t){for(const e of t)if(0!==e.positionedGlyphs.length)return !1;return !0}(w)&&_}const mu={9:!0,10:!0,11:!0,12:!0,13:!0,32:!0},gu={10:!0,32:!0,38:!0,40:!0,41:!0,43:!0,45:!0,47:!0,173:!0,183:!0,8203:!0,8208:!0,8211:!0,8231:!0};function xu(t,e,r,n,i,a){if(e.imageName){const t=n[e.imageName];return t?t.displaySize[0]*e.scale*Vl/a+i:0}{const n=r[e.fontStack],a=n&&n[t];return a?a.metrics.advance*e.scale+i:0}}function vu(t,e,r,n){const i=Math.pow(t-e,2);return n?t=0;let c=0;for(let r=0;rt.id)),this.index=e.index,this.pixelRatio=e.pixelRatio,this.sourceLayerIndex=e.sourceLayerIndex,this.hasPattern=!1,this.hasRTLText=!1,this.sortKeyRanges=[],this.collisionCircleArray=[],this.placementInvProjMatrix=Ms([]),this.placementViewportMatrix=Ms([]);const r=this.layers[0]._unevaluatedLayout._values;this.textSizeData=Pu(this.zoom,r["text-size"]),this.iconSizeData=Pu(this.zoom,r["icon-size"]);const n=this.layers[0].layout,i=n.get("symbol-sort-key"),a=n.get("symbol-z-order");this.canOverlap="never"!==Vu(n,"text-overlap","text-allow-overlap")||"never"!==Vu(n,"icon-overlap","icon-allow-overlap")||n.get("text-ignore-placement")||n.get("icon-ignore-placement"),this.sortFeaturesByKey="viewport-y"!==a&&!i.isConstant(),this.sortFeaturesByY=("viewport-y"===a||"auto"===a&&!this.sortFeaturesByKey)&&this.canOverlap,"point"===n.get("symbol-placement")&&(this.writingModes=n.get("text-writing-mode").map((e=>t.ai[e]))),this.stateDependentLayerIds=this.layers.filter((t=>t.isStateDependent())).map((t=>t.id)),this.sourceID=e.sourceID;}createArrays(){this.text=new Du(new Qa(this.layers,this.zoom,(t=>/^text/.test(t)))),this.icon=new Du(new Qa(this.layers,this.zoom,(t=>/^icon/.test(t)))),this.glyphOffsetArray=new aa,this.lineVertexArray=new sa,this.symbolInstances=new ia,this.textAnchorOffsets=new la;}calculateGlyphDependencies(t,e,r,n,i){for(let a=0;a0)&&("constant"!==s.value.kind||s.value.value.length>0),c="constant"!==l.value.kind||!!l.value.value||Object.keys(l.parameters).length>0,h=a.get("symbol-sort-key");if(this.features=[],!u&&!c)return;const p=r.iconDependencies,f=r.glyphDependencies,d=r.availableImages,y=new li(this.zoom);for(const{feature:r,id:o,index:l,sourceLayerIndex:m}of e){const e=i._featureFilter.needGeometry,g=ss(r,e);if(!i._featureFilter.filter(y,g,n))continue;let x,v;if(e||(g.geometry=as(r)),u){const t=i.getValueAndResolveTokens("text-field",g,n,d),e=Qt.factory(t);Lu(e)&&(this.hasRTLText=!0),(!this.hasRTLText||"unavailable"===ai()||this.hasRTLText&&oi.isParsed())&&(x=Bl(e,i,g));}if(c){const t=i.getValueAndResolveTokens("icon-image",g,n,d);v=t instanceof ne?t:ne.fromString(t);}if(!x&&!v)continue;const b=this.sortFeaturesByKey?h.evaluate(g,{},n):void 0;if(this.features.push({id:o,text:x,icon:v,index:l,sourceLayerIndex:m,geometry:g.geometry,properties:r.properties,type:Eu[r.type],sortKey:b}),v&&(p[v.name]=!0),x){const e=s.evaluate(g,{},n).join(","),r="viewport"!==a.get("text-rotation-alignment")&&"point"!==a.get("symbol-placement");this.allowVerticalPlacement=this.writingModes&&this.writingModes.indexOf(t.ai.vertical)>=0;for(const t of x.sections)if(t.image)p[t.image.name]=!0;else {const n=qn(x.toString()),i=t.fontStack||e,a=f[i]=f[i]||{};this.calculateGlyphDependencies(t.text,a,r,this.allowVerticalPlacement,n);}}}"line"===a.get("symbol-placement")&&(this.features=function(t){const e={},r={},n=[];let i=0;function a(e){n.push(t[e]),i++;}function s(t,e,i){const a=r[t];return delete r[t],r[e]=a,n[a].geometry[0].pop(),n[a].geometry[0]=n[a].geometry[0].concat(i[0]),a}function o(t,r,i){const a=e[r];return delete e[r],e[t]=a,n[a].geometry[0].shift(),n[a].geometry[0]=i[0].concat(n[a].geometry[0]),a}function l(t,e,r){const n=r?e[0][e[0].length-1]:e[0][0];return `${t}:${n.x}:${n.y}`}for(let u=0;ut.geometry))}(this.features)),this.sortFeaturesByKey&&this.features.sort(((t,e)=>t.sortKey-e.sortKey));}update(t,e,r){this.stateDependentLayers.length&&(this.text.programConfigurations.updatePaintArrays(t,e,this.layers,r),this.icon.programConfigurations.updatePaintArrays(t,e,this.layers,r));}isEmpty(){return 0===this.symbolInstances.length&&!this.hasRTLText}uploadPending(){return !this.uploaded||this.text.programConfigurations.needsUpload||this.icon.programConfigurations.needsUpload}upload(t){!this.uploaded&&this.hasDebugData()&&(this.textCollisionBox.upload(t),this.iconCollisionBox.upload(t)),this.text.upload(t,this.sortFeaturesByY,!this.uploaded,this.text.programConfigurations.needsUpload),this.icon.upload(t,this.sortFeaturesByY,!this.uploaded,this.icon.programConfigurations.needsUpload),this.uploaded=!0;}destroyDebugData(){this.textCollisionBox.destroy(),this.iconCollisionBox.destroy();}destroy(){this.text.destroy(),this.icon.destroy(),this.hasDebugData()&&this.destroyDebugData();}addToLineVertexArray(t,e){const r=this.lineVertexArray.length;if(void 0!==t.segment){let r=t.dist(e[t.segment+1]),n=t.dist(e[t.segment]);const i={};for(let n=t.segment+1;n=0;r--)i[r]={x:e[r].x,y:e[r].y,tileUnitDistanceFromAnchor:n},r>0&&(n+=e[r-1].dist(e[r]));for(let t=0;t0}hasIconData(){return this.icon.segments.get().length>0}hasDebugData(){return this.textCollisionBox&&this.iconCollisionBox}hasTextCollisionBoxData(){return this.hasDebugData()&&this.textCollisionBox.segments.get().length>0}hasIconCollisionBoxData(){return this.hasDebugData()&&this.iconCollisionBox.segments.get().length>0}addIndicesForPlacedSymbol(t,e){const r=t.placedSymbolArray.get(e),n=r.vertexStartIndex+4*r.numGlyphs;for(let e=r.vertexStartIndex;en[t]-n[e]||i[e]-i[t])),a}addToSortKeyRanges(t,e){const r=this.sortKeyRanges[this.sortKeyRanges.length-1];r&&r.sortKey===e?r.symbolInstanceEnd=t+1:this.sortKeyRanges.push({sortKey:e,symbolInstanceStart:t,symbolInstanceEnd:t+1});}sortFeatures(t){if(this.sortFeaturesByY&&this.sortedAngle!==t&&!(this.text.segments.get().length>1||this.icon.segments.get().length>1)){this.symbolInstanceIndexes=this.getSortedSymbolIndexes(t),this.sortedAngle=t,this.text.indexArray.clear(),this.icon.indexArray.clear(),this.featureSortOrder=[];for(const t of this.symbolInstanceIndexes){const e=this.symbolInstances.get(t);this.featureSortOrder.push(e.featureIndex),[e.rightJustifiedTextSymbolIndex,e.centerJustifiedTextSymbolIndex,e.leftJustifiedTextSymbolIndex].forEach(((t,e,r)=>{t>=0&&r.indexOf(t)===e&&this.addIndicesForPlacedSymbol(this.text,t);})),e.verticalPlacedTextSymbolIndex>=0&&this.addIndicesForPlacedSymbol(this.text,e.verticalPlacedTextSymbolIndex),e.placedIconSymbolIndex>=0&&this.addIndicesForPlacedSymbol(this.icon,e.placedIconSymbolIndex),e.verticalPlacedIconSymbolIndex>=0&&this.addIndicesForPlacedSymbol(this.icon,e.verticalPlacedIconSymbolIndex);}this.text.indexBuffer&&this.text.indexBuffer.updateData(this.text.indexArray),this.icon.indexBuffer&&this.icon.indexBuffer.updateData(this.icon.indexArray);}}}let Ru,qu;$n("SymbolBucket",Uu,{omit:["layers","collisionBoxArray","features","compareText"]}),Uu.MAX_GLYPHS=65535,Uu.addDynamicAttributes=$u;var ju={get paint(){return qu=qu||new _i({"icon-opacity":new xi(X.paint_symbol["icon-opacity"]),"icon-color":new xi(X.paint_symbol["icon-color"]),"icon-halo-color":new xi(X.paint_symbol["icon-halo-color"]),"icon-halo-width":new xi(X.paint_symbol["icon-halo-width"]),"icon-halo-blur":new xi(X.paint_symbol["icon-halo-blur"]),"icon-translate":new gi(X.paint_symbol["icon-translate"]),"icon-translate-anchor":new gi(X.paint_symbol["icon-translate-anchor"]),"text-opacity":new xi(X.paint_symbol["text-opacity"]),"text-color":new xi(X.paint_symbol["text-color"],{runtimeType:yt,getOverride:t=>t.textColor,hasOverride:t=>!!t.textColor}),"text-halo-color":new xi(X.paint_symbol["text-halo-color"]),"text-halo-width":new xi(X.paint_symbol["text-halo-width"]),"text-halo-blur":new xi(X.paint_symbol["text-halo-blur"]),"text-translate":new gi(X.paint_symbol["text-translate"]),"text-translate-anchor":new gi(X.paint_symbol["text-translate-anchor"])})},get layout(){return Ru=Ru||new _i({"symbol-placement":new gi(X.layout_symbol["symbol-placement"]),"symbol-spacing":new gi(X.layout_symbol["symbol-spacing"]),"symbol-avoid-edges":new gi(X.layout_symbol["symbol-avoid-edges"]),"symbol-sort-key":new xi(X.layout_symbol["symbol-sort-key"]),"symbol-z-order":new gi(X.layout_symbol["symbol-z-order"]),"icon-allow-overlap":new gi(X.layout_symbol["icon-allow-overlap"]),"icon-overlap":new gi(X.layout_symbol["icon-overlap"]),"icon-ignore-placement":new gi(X.layout_symbol["icon-ignore-placement"]),"icon-optional":new gi(X.layout_symbol["icon-optional"]),"icon-rotation-alignment":new gi(X.layout_symbol["icon-rotation-alignment"]),"icon-size":new xi(X.layout_symbol["icon-size"]),"icon-text-fit":new gi(X.layout_symbol["icon-text-fit"]),"icon-text-fit-padding":new gi(X.layout_symbol["icon-text-fit-padding"]),"icon-image":new xi(X.layout_symbol["icon-image"]),"icon-rotate":new xi(X.layout_symbol["icon-rotate"]),"icon-padding":new xi(X.layout_symbol["icon-padding"]),"icon-keep-upright":new gi(X.layout_symbol["icon-keep-upright"]),"icon-offset":new xi(X.layout_symbol["icon-offset"]),"icon-anchor":new xi(X.layout_symbol["icon-anchor"]),"icon-pitch-alignment":new gi(X.layout_symbol["icon-pitch-alignment"]),"text-pitch-alignment":new gi(X.layout_symbol["text-pitch-alignment"]),"text-rotation-alignment":new gi(X.layout_symbol["text-rotation-alignment"]),"text-field":new xi(X.layout_symbol["text-field"]),"text-font":new xi(X.layout_symbol["text-font"]),"text-size":new xi(X.layout_symbol["text-size"]),"text-max-width":new xi(X.layout_symbol["text-max-width"]),"text-line-height":new gi(X.layout_symbol["text-line-height"]),"text-letter-spacing":new xi(X.layout_symbol["text-letter-spacing"]),"text-justify":new xi(X.layout_symbol["text-justify"]),"text-radial-offset":new xi(X.layout_symbol["text-radial-offset"]),"text-variable-anchor":new gi(X.layout_symbol["text-variable-anchor"]),"text-variable-anchor-offset":new xi(X.layout_symbol["text-variable-anchor-offset"]),"text-anchor":new xi(X.layout_symbol["text-anchor"]),"text-max-angle":new gi(X.layout_symbol["text-max-angle"]),"text-writing-mode":new gi(X.layout_symbol["text-writing-mode"]),"text-rotate":new xi(X.layout_symbol["text-rotate"]),"text-padding":new gi(X.layout_symbol["text-padding"]),"text-keep-upright":new gi(X.layout_symbol["text-keep-upright"]),"text-transform":new xi(X.layout_symbol["text-transform"]),"text-offset":new xi(X.layout_symbol["text-offset"]),"text-allow-overlap":new gi(X.layout_symbol["text-allow-overlap"]),"text-overlap":new gi(X.layout_symbol["text-overlap"]),"text-ignore-placement":new gi(X.layout_symbol["text-ignore-placement"]),"text-optional":new gi(X.layout_symbol["text-optional"])})}};class Nu{constructor(t){if(void 0===t.property.overrides)throw new Error("overrides must be provided to instantiate FormatSectionOverride class");this.type=t.property.overrides?t.property.overrides.runtimeType:ht,this.defaultValue=t;}evaluate(t){if(t.formattedSection){const e=this.defaultValue.property.overrides;if(e&&e.hasOverride(t.formattedSection))return e.getOverride(t.formattedSection)}return t.feature&&t.featureState?this.defaultValue.evaluate(t.feature,t.featureState):this.defaultValue.property.specification.default}eachChild(t){this.defaultValue.isConstant()||t(this.defaultValue.value._styleExpression.expression);}outputDefined(){return !1}serialize(){return null}}$n("FormatSectionOverride",Nu,{omit:["defaultValue"]});class Zu extends Si{constructor(t){super(t,ju);}recalculate(t,e){if(super.recalculate(t,e),"auto"===this.layout.get("icon-rotation-alignment")&&(this.layout._values["icon-rotation-alignment"]="point"!==this.layout.get("symbol-placement")?"map":"viewport"),"auto"===this.layout.get("text-rotation-alignment")&&(this.layout._values["text-rotation-alignment"]="point"!==this.layout.get("symbol-placement")?"map":"viewport"),"auto"===this.layout.get("text-pitch-alignment")&&(this.layout._values["text-pitch-alignment"]="map"===this.layout.get("text-rotation-alignment")?"map":"viewport"),"auto"===this.layout.get("icon-pitch-alignment")&&(this.layout._values["icon-pitch-alignment"]=this.layout.get("icon-rotation-alignment")),"point"===this.layout.get("symbol-placement")){const t=this.layout.get("text-writing-mode");if(t){const e=[];for(const r of t)e.indexOf(r)<0&&e.push(r);this.layout._values["text-writing-mode"]=e;}else this.layout._values["text-writing-mode"]=["horizontal"];}this._setPaintOverrides();}getValueAndResolveTokens(t,e,r,n){const i=this.layout.get(t).evaluate(e,{},r,n),a=this._unevaluatedLayout._values[t];return a.isDataDriven()||Tr(a.value)||!i?i:function(t,e){return e.replace(/{([^{}]+)}/g,((e,r)=>t&&r in t?String(t[r]):""))}(e.properties,i)}createBucket(t){return new Uu(t)}queryRadius(){return 0}queryIntersectsFeature(){throw new Error("Should take a different path in FeatureIndex")}_setPaintOverrides(){for(const t of ju.paint.overridableProperties){if(!Zu.hasPaintOverride(this.layout,t))continue;const e=this.paint.get(t),r=new Nu(e),n=new Fr(r,e.property.specification);let i=null;i="constant"===e.value.kind||"source"===e.value.kind?new Lr("source",n):new Dr("composite",n,e.value.zoomStops),this.paint._values[t]=new yi(e.property,i,e.parameters);}}_handleOverridablePaintPropertyUpdate(t,e,r){return !(!this.layout||e.isDataDriven()||r.isDataDriven())&&Zu.hasPaintOverride(this.layout,t)}static hasPaintOverride(t,e){const r=t.get("text-field"),n=ju.paint.properties[e];let i=!1;const a=t=>{for(const e of t)if(n.overrides&&n.overrides.hasOverride(e))return void(i=!0)};if("constant"===r.value.kind&&r.value.value instanceof Qt)a(r.value.value.sections);else if("source"===r.value.kind){const t=e=>{i||(e instanceof le&&se(e.value)===vt?a(e.value.sections):e instanceof pr?a(e.sections):e.eachChild(t));},e=r.value;e._styleExpression&&t(e._styleExpression.expression);}return i}}let Ku;var Gu={get paint(){return Ku=Ku||new _i({"background-color":new gi(X.paint_background["background-color"]),"background-pattern":new bi(X.paint_background["background-pattern"]),"background-opacity":new gi(X.paint_background["background-opacity"])})}};class Ju extends Si{constructor(t){super(t,Gu);}}let Xu;var Yu={get paint(){return Xu=Xu||new _i({"raster-opacity":new gi(X.paint_raster["raster-opacity"]),"raster-hue-rotate":new gi(X.paint_raster["raster-hue-rotate"]),"raster-brightness-min":new gi(X.paint_raster["raster-brightness-min"]),"raster-brightness-max":new gi(X.paint_raster["raster-brightness-max"]),"raster-saturation":new gi(X.paint_raster["raster-saturation"]),"raster-contrast":new gi(X.paint_raster["raster-contrast"]),"raster-resampling":new gi(X.paint_raster["raster-resampling"]),"raster-fade-duration":new gi(X.paint_raster["raster-fade-duration"])})}};class Hu extends Si{constructor(t){super(t,Yu);}}class Wu extends Si{constructor(t){super(t,{}),this.onAdd=t=>{this.implementation.onAdd&&this.implementation.onAdd(t,t.painter.context.gl);},this.onRemove=t=>{this.implementation.onRemove&&this.implementation.onRemove(t,t.painter.context.gl);},this.implementation=t;}is3D(){return "3d"===this.implementation.renderingMode}hasOffscreenPass(){return void 0!==this.implementation.prerender}recalculate(){}updateTransitions(){}hasTransition(){return !1}serialize(){throw new Error("Custom layers cannot be serialized")}}class Qu{constructor(t){this._callback=t,this._triggered=!1,"undefined"!=typeof MessageChannel&&(this._channel=new MessageChannel,this._channel.port2.onmessage=()=>{this._triggered=!1,this._callback();});}trigger(){this._triggered||(this._triggered=!0,this._channel?this._channel.port1.postMessage(!0):setTimeout((()=>{this._triggered=!1,this._callback();}),0));}remove(){delete this._channel,this._callback=()=>{};}}const tc=6371008.8;class ec{constructor(t,e){if(isNaN(t)||isNaN(e))throw new Error(`Invalid LngLat object: (${t}, ${e})`);if(this.lng=+t,this.lat=+e,this.lat>90||this.lat<-90)throw new Error("Invalid LngLat latitude value: must be between -90 and 90")}wrap(){return new ec(m(this.lng,-180,180),this.lat)}toArray(){return [this.lng,this.lat]}toString(){return `LngLat(${this.lng}, ${this.lat})`}distanceTo(t){const e=Math.PI/180,r=this.lat*e,n=t.lat*e,i=Math.sin(r)*Math.sin(n)+Math.cos(r)*Math.cos(n)*Math.cos((t.lng-this.lng)*e);return tc*Math.acos(Math.min(i,1))}static convert(t){if(t instanceof ec)return t;if(Array.isArray(t)&&(2===t.length||3===t.length))return new ec(Number(t[0]),Number(t[1]));if(!Array.isArray(t)&&"object"==typeof t&&null!==t)return new ec(Number("lng"in t?t.lng:t.lon),Number(t.lat));throw new Error("`LngLatLike` argument must be specified as a LngLat instance, an object {lng: , lat: }, an object {lon: , lat: }, or an array of [, ]")}}const rc=2*Math.PI*tc;function nc(t){return rc*Math.cos(t*Math.PI/180)}function ic(t){return (180+t)/360}function ac(t){return (180-180/Math.PI*Math.log(Math.tan(Math.PI/4+t*Math.PI/360)))/360}function sc(t,e){return t/nc(e)}function oc(t){return 360*t-180}function lc(t){return 360/Math.PI*Math.atan(Math.exp((180-360*t)*Math.PI/180))-90}class uc{constructor(t,e,r=0){this.x=+t,this.y=+e,this.z=+r;}static fromLngLat(t,e=0){const r=ec.convert(t);return new uc(ic(r.lng),ac(r.lat),sc(e,r.lat))}toLngLat(){return new ec(oc(this.x),lc(this.y))}toAltitude(){return this.z*nc(lc(this.y))}meterInMercatorCoordinateUnits(){return 1/rc*(t=lc(this.y),1/Math.cos(t*Math.PI/180));var t;}}function cc(t,e,r){var n=2*Math.PI*6378137/256/Math.pow(2,r);return [t*n-2*Math.PI*6378137/2,e*n-2*Math.PI*6378137/2]}class hc{constructor(t,e,r){if(t<0||t>25||r<0||r>=Math.pow(2,t)||e<0||e>=Math.pow(2,t))throw new Error(`x=${e}, y=${r}, z=${t} outside of bounds. 0<=x<${Math.pow(2,t)}, 0<=y<${Math.pow(2,t)} 0<=z<=25 `);this.z=t,this.x=e,this.y=r,this.key=dc(0,t,t,e,r);}equals(t){return this.z===t.z&&this.x===t.x&&this.y===t.y}url(t,e,r){const n=(a=this.y,s=this.z,o=cc(256*(i=this.x),256*(a=Math.pow(2,s)-a-1),s),l=cc(256*(i+1),256*(a+1),s),o[0]+","+o[1]+","+l[0]+","+l[1]);var i,a,s,o,l;const u=function(t,e,r){let n,i="";for(let a=t;a>0;a--)n=1<1?"@2x":"").replace(/{quadkey}/g,u).replace(/{bbox-epsg-3857}/g,n)}isChildOf(t){const e=this.z-t.z;return e>0&&t.x===this.x>>e&&t.y===this.y>>e}getTilePoint(t){const e=Math.pow(2,this.z);return new a((t.x*e-this.x)*rs,(t.y*e-this.y)*rs)}toString(){return `${this.z}/${this.x}/${this.y}`}}class pc{constructor(t,e){this.wrap=t,this.canonical=e,this.key=dc(t,e.z,e.z,e.x,e.y);}}class fc{constructor(t,e,r,n,i){if(t= z; overscaledZ = ${t}; z = ${r}`);this.overscaledZ=t,this.wrap=e,this.canonical=new hc(r,+n,+i),this.key=dc(e,t,r,n,i);}clone(){return new fc(this.overscaledZ,this.wrap,this.canonical.z,this.canonical.x,this.canonical.y)}equals(t){return this.overscaledZ===t.overscaledZ&&this.wrap===t.wrap&&this.canonical.equals(t.canonical)}scaledTo(t){if(t>this.overscaledZ)throw new Error(`targetZ > this.overscaledZ; targetZ = ${t}; overscaledZ = ${this.overscaledZ}`);const e=this.canonical.z-t;return t>this.canonical.z?new fc(t,this.wrap,this.canonical.z,this.canonical.x,this.canonical.y):new fc(t,this.wrap,t,this.canonical.x>>e,this.canonical.y>>e)}calculateScaledKey(t,e){if(t>this.overscaledZ)throw new Error(`targetZ > this.overscaledZ; targetZ = ${t}; overscaledZ = ${this.overscaledZ}`);const r=this.canonical.z-t;return t>this.canonical.z?dc(this.wrap*+e,t,this.canonical.z,this.canonical.x,this.canonical.y):dc(this.wrap*+e,t,t,this.canonical.x>>r,this.canonical.y>>r)}isChildOf(t){if(t.wrap!==this.wrap)return !1;const e=this.canonical.z-t.canonical.z;return 0===t.overscaledZ||t.overscaledZ>e&&t.canonical.y===this.canonical.y>>e}children(t){if(this.overscaledZ>=t)return [new fc(this.overscaledZ+1,this.wrap,this.canonical.z,this.canonical.x,this.canonical.y)];const e=this.canonical.z+1,r=2*this.canonical.x,n=2*this.canonical.y;return [new fc(e,this.wrap,e,r,n),new fc(e,this.wrap,e,r+1,n),new fc(e,this.wrap,e,r,n+1),new fc(e,this.wrap,e,r+1,n+1)]}isLessThan(t){return this.wrapt.wrap)&&(this.overscaledZt.overscaledZ)&&(this.canonical.xt.canonical.x)&&this.canonical.ythis.max&&(this.max=r),r=this.dim+1||e<-1||e>=this.dim+1)throw new RangeError("out of range source coordinates for DEM data");return (e+1)*this.stride+(t+1)}unpack(t,e,r){return t*this.redFactor+e*this.greenFactor+r*this.blueFactor-this.baseShift}getPixels(){return new qs({width:this.stride,height:this.stride},new Uint8Array(this.data.buffer))}backfillBorder(t,e,r){if(this.dim!==t.dim)throw new Error("dem dimension mismatch");let n=e*this.dim,i=e*this.dim+this.dim,a=r*this.dim,s=r*this.dim+this.dim;switch(e){case-1:n=i-1;break;case 1:i=n+1;}switch(r){case-1:a=s-1;break;case 1:s=a+1;}const o=-e*this.dim,l=-r*this.dim;for(let e=a;e=this._numberToString.length)throw new Error(`Out of bounds. Index requested n=${t} can't be >= this._numberToString.length ${this._numberToString.length}`);return this._numberToString[t]}}class gc{constructor(t,e,r,n,i){this.type="Feature",this._vectorTileFeature=t,t._z=e,t._x=r,t._y=n,this.properties=t.properties,this.id=i;}get geometry(){return void 0===this._geometry&&(this._geometry=this._vectorTileFeature.toGeoJSON(this._vectorTileFeature._x,this._vectorTileFeature._y,this._vectorTileFeature._z).geometry),this._geometry}set geometry(t){this._geometry=t;}toJSON(){const t={geometry:this.geometry};for(const e in this)"_geometry"!==e&&"_vectorTileFeature"!==e&&(t[e]=this[e]);return t}}class xc{constructor(t,e){this.tileID=t,this.x=t.canonical.x,this.y=t.canonical.y,this.z=t.canonical.z,this.grid=new Fn(rs,16,0),this.grid3D=new Fn(rs,16,0),this.featureIndexArray=new ca,this.promoteId=e;}insert(t,e,r,n,i,a){const s=this.featureIndexArray.length;this.featureIndexArray.emplaceBack(r,n,i);const o=a?this.grid3D:this.grid;for(let t=0;t=0&&n[3]>=0&&o.insert(s,n[0],n[1],n[2],n[3]);}}loadVTLayers(){return this.vtLayers||(this.vtLayers=new Ro.VectorTile(new eu(this.rawTileData)).layers,this.sourceLayerCoder=new mc(this.vtLayers?Object.keys(this.vtLayers).sort():["_geojsonTileLayer"])),this.vtLayers}query(t,e,r,n){this.loadVTLayers();const i=t.params||{},s=rs/t.tileSize/t.scale,o=Nr(i.filter),l=t.queryGeometry,u=t.queryPadding*s,c=bc(l),h=this.grid.query(c.minX-u,c.minY-u,c.maxX+u,c.maxY+u),p=bc(t.cameraQueryGeometry),f=this.grid3D.query(p.minX-u,p.minY-u,p.maxX+u,p.maxY+u,((e,r,n,i)=>function(t,e,r,n,i){for(const a of t)if(e<=a.x&&r<=a.y&&n>=a.x&&i>=a.y)return !0;const s=[new a(e,r),new a(e,i),new a(n,i),new a(n,r)];if(t.length>2)for(const e of s)if(xs(t,e))return !0;for(let e=0;e(p||(p=as(e)),r.queryIntersectsFeature(l,e,n,p,this.z,t.transform,s,t.pixelPosMatrix))));}return d}loadMatchingFeature(t,e,r,n,i,a,s,o,l,u,c){const h=this.bucketLayerIDs[e];if(a&&!function(t,e){for(let r=0;r=0)return !0;return !1}(a,h))return;const p=this.sourceLayerCoder.decode(r),f=this.vtLayers[p].feature(n);if(i.needGeometry){const t=ss(f,!0);if(!i.filter(new li(this.tileID.overscaledZ),t,this.tileID.canonical))return}else if(!i.filter(new li(this.tileID.overscaledZ),f))return;const d=this.getId(f,p);for(let e=0;e{const s=e instanceof mi?e.get(a):null;return s&&s.evaluate?s.evaluate(r,n,i):s}))}function bc(t){let e=1/0,r=1/0,n=-1/0,i=-1/0;for(const a of t)e=Math.min(e,a.x),r=Math.min(r,a.y),n=Math.max(n,a.x),i=Math.max(i,a.y);return {minX:e,minY:r,maxX:n,maxY:i}}function wc(t,e){return e-t}function _c(t,e,r,n,i){const s=[];for(let o=0;o=n&&c.x>=n||(o.x>=n?o=new a(n,o.y+(n-o.x)/(c.x-o.x)*(c.y-o.y))._round():c.x>=n&&(c=new a(n,o.y+(n-o.x)/(c.x-o.x)*(c.y-o.y))._round()),o.y>=i&&c.y>=i||(o.y>=i?o=new a(o.x+(i-o.y)/(c.y-o.y)*(c.x-o.x),i)._round():c.y>=i&&(c=new a(o.x+(i-o.y)/(c.y-o.y)*(c.x-o.x),i)._round()),u&&o.equals(u[u.length-1])||(u=[o],s.push(u)),u.push(c)))));}}return s}$n("FeatureIndex",xc,{omit:["rawTileData","sourceLayerCoder"]});class Ac extends a{constructor(t,e,r,n){super(t,e),this.angle=r,void 0!==n&&(this.segment=n);}clone(){return new Ac(this.x,this.y,this.angle,this.segment)}}function Sc(t,e,r,n,i){if(void 0===e.segment||0===r)return !0;let a=e,s=e.segment+1,o=0;for(;o>-r/2;){if(s--,s<0)return !1;o-=t[s].dist(a),a=t[s];}o+=t[s].dist(t[s+1]),s++;const l=[];let u=0;for(;on;)u-=l.shift().angleDelta;if(u>i)return !1;s++,o+=e.dist(r);}return !0}function kc(t){let e=0;for(let r=0;ru){const c=(u-l)/a,h=Ze.number(n.x,i.x,c),p=Ze.number(n.y,i.y,c),f=new Ac(h,p,i.angleTo(n),r);return f._round(),!s||Sc(t,f,o,s,e)?f:void 0}l+=a;}}function Cc(t,e,r,n,i,a,s,o,l){const u=Ic(n,a,s),c=zc(n,i),h=c*s,p=0===t[0].x||t[0].x===l||0===t[0].y||t[0].y===l;return e-h=0&&g=0&&x=0&&p+u<=c){const r=new Ac(g,x,y,e);r._round(),n&&!Sc(t,r,a,n,i)||f.push(r);}}h+=d;}return o||f.length||s||(f=Bc(t,h/2,r,n,i,a,s,!0,l)),f}$n("Anchor",Ac);const Pc=lu;function Vc(t,e,r,n){const i=[],s=t.image,o=s.pixelRatio,l=s.paddedRect.w-2*Pc,u=s.paddedRect.h-2*Pc,c=t.right-t.left,h=t.bottom-t.top,p=s.stretchX||[[0,l]],f=s.stretchY||[[0,u]],d=(t,e)=>t+e[1]-e[0],y=p.reduce(d,0),m=f.reduce(d,0),g=l-y,x=u-m;let v=0,b=y,w=0,_=m,A=0,S=g,k=0,I=x;if(s.content&&n){const t=s.content;v=Ec(p,0,t[0]),w=Ec(f,0,t[1]),b=Ec(p,t[0],t[2]),_=Ec(f,t[1],t[3]),A=t[0]-v,k=t[1]-w,S=t[2]-t[0]-b,I=t[3]-t[1]-_;}const z=(n,i,l,u)=>{const p=Tc(n.stretch-v,b,c,t.left),f=$c(n.fixed-A,S,n.stretch,y),d=Tc(i.stretch-w,_,h,t.top),g=$c(i.fixed-k,I,i.stretch,m),x=Tc(l.stretch-v,b,c,t.left),z=$c(l.fixed-A,S,l.stretch,y),M=Tc(u.stretch-w,_,h,t.top),C=$c(u.fixed-k,I,u.stretch,m),B=new a(p,d),P=new a(x,d),V=new a(x,M),E=new a(p,M),F=new a(f/o,g/o),T=new a(z/o,C/o),$=e*Math.PI/180;if($){const t=Math.sin($),e=Math.cos($),r=[e,-t,t,e];B._matMult(r),P._matMult(r),E._matMult(r),V._matMult(r);}const L=n.stretch+n.fixed,D=i.stretch+i.fixed;return {tl:B,tr:P,bl:E,br:V,tex:{x:s.paddedRect.x+Pc+L,y:s.paddedRect.y+Pc+D,w:l.stretch+l.fixed-L,h:u.stretch+u.fixed-D},writingMode:void 0,glyphOffset:[0,0],sectionIndex:0,pixelOffsetTL:F,pixelOffsetBR:T,minFontScaleX:S/o/c,minFontScaleY:I/o/h,isSDF:r}};if(n&&(s.stretchX||s.stretchY)){const t=Fc(p,g,y),e=Fc(f,x,m);for(let r=0;r0&&(n=Math.max(10,n),this.circleDiameter=n);}else {let u=s.top*o-l[0],h=s.bottom*o+l[2],p=s.left*o-l[3],f=s.right*o+l[1];const d=s.collisionPadding;if(d&&(p-=d[0]*o,u-=d[1]*o,f+=d[2]*o,h+=d[3]*o),c){const t=new a(p,u),e=new a(f,u),r=new a(p,h),n=new a(f,h),i=c*Math.PI/180;t._rotate(i),e._rotate(i),r._rotate(i),n._rotate(i),p=Math.min(t.x,e.x,r.x,n.x),f=Math.max(t.x,e.x,r.x,n.x),u=Math.min(t.y,e.y,r.y,n.y),h=Math.max(t.y,e.y,r.y,n.y);}t.emplaceBack(e.x,e.y,p,u,f,h,r,n,i);}this.boxEndIndex=t.length;}}class Dc{constructor(t=[],e=Oc){if(this.data=t,this.length=this.data.length,this.compare=e,this.length>0)for(let t=(this.length>>1)-1;t>=0;t--)this._down(t);}push(t){this.data.push(t),this.length++,this._up(this.length-1);}pop(){if(0===this.length)return;const t=this.data[0],e=this.data.pop();return this.length--,this.length>0&&(this.data[0]=e,this._down(0)),t}peek(){return this.data[0]}_up(t){const{data:e,compare:r}=this,n=e[t];for(;t>0;){const i=t-1>>1,a=e[i];if(r(n,a)>=0)break;e[t]=a,t=i;}e[t]=n;}_down(t){const{data:e,compare:r}=this,n=this.length>>1,i=e[t];for(;t=0)break;e[t]=a,t=n;}e[t]=i;}}function Oc(t,e){return te?1:0}function Uc(t,e=1,r=!1){let n=1/0,i=1/0,s=-1/0,o=-1/0;const l=t[0];for(let t=0;ts)&&(s=e.x),(!t||e.y>o)&&(o=e.y);}const u=Math.min(s-n,o-i);let c=u/2;const h=new Dc([],Rc);if(0===u)return new a(n,i);for(let e=n;ep.d||!p.d)&&(p=n,r&&console.log("found best %d after %d probes",Math.round(1e4*n.d)/1e4,f)),n.max-p.d<=e||(c=n.h/2,h.push(new qc(n.p.x-c,n.p.y-c,c,t)),h.push(new qc(n.p.x+c,n.p.y-c,c,t)),h.push(new qc(n.p.x-c,n.p.y+c,c,t)),h.push(new qc(n.p.x+c,n.p.y+c,c,t)),f+=4);}return r&&(console.log(`num probes: ${f}`),console.log(`best distance: ${p.d}`)),p.p}function Rc(t,e){return e.max-t.max}function qc(t,e,r,n){this.p=new a(t,e),this.h=r,this.d=function(t,e){let r=!1,n=1/0;for(let i=0;it.y!=o.y>t.y&&t.x<(o.x-i.x)*(t.y-i.y)/(o.y-i.y)+i.x&&(r=!r),n=Math.min(n,ms(t,i,o));}}return (r?1:-1)*Math.sqrt(n)}(this.p,n),this.max=this.d+this.h*Math.SQRT2;}var jc;t.aq=void 0,(jc=t.aq||(t.aq={}))[jc.center=1]="center",jc[jc.left=2]="left",jc[jc.right=3]="right",jc[jc.top=4]="top",jc[jc.bottom=5]="bottom",jc[jc["top-left"]=6]="top-left",jc[jc["top-right"]=7]="top-right",jc[jc["bottom-left"]=8]="bottom-left",jc[jc["bottom-right"]=9]="bottom-right";const Nc=7,Zc=Number.POSITIVE_INFINITY;function Kc(t,e){return e[1]!==Zc?function(t,e,r){let n=0,i=0;switch(e=Math.abs(e),r=Math.abs(r),t){case"top-right":case"top-left":case"top":i=r-Nc;break;case"bottom-right":case"bottom-left":case"bottom":i=-r+Nc;}switch(t){case"top-right":case"bottom-right":case"right":n=-e;break;case"top-left":case"bottom-left":case"left":n=e;}return [n,i]}(t,e[0],e[1]):function(t,e){let r=0,n=0;e<0&&(e=0);const i=e/Math.SQRT2;switch(t){case"top-right":case"top-left":n=i-Nc;break;case"bottom-right":case"bottom-left":n=-i+Nc;break;case"bottom":n=-e+Nc;break;case"top":n=e-Nc;}switch(t){case"top-right":case"bottom-right":r=-i;break;case"top-left":case"bottom-left":r=i;break;case"left":r=e;break;case"right":r=-e;}return [r,n]}(t,e[0])}function Gc(t,e,r){var n;const i=t.layout,a=null===(n=i.get("text-variable-anchor-offset"))||void 0===n?void 0:n.evaluate(e,{},r);if(a){const t=a.values,e=[];for(let r=0;rt*Vl));n.startsWith("top")?i[1]-=Nc:n.startsWith("bottom")&&(i[1]+=Nc),e[r+1]=i;}return new re(e)}const s=i.get("text-variable-anchor");if(s){let n;n=void 0!==t._unevaluatedLayout.getValue("text-radial-offset")?[i.get("text-radial-offset").evaluate(e,{},r)*Vl,Zc]:i.get("text-offset").evaluate(e,{},r).map((t=>t*Vl));const a=[];for(const t of s)a.push(t,Kc(t,n));return new re(a)}return null}function Jc(t){switch(t){case"right":case"top-right":case"bottom-right":return "right";case"left":case"top-left":case"bottom-left":return "left"}return "center"}function Xc(e,r,n,i,a,s,o,l,u,c,h){let p=s.textMaxSize.evaluate(r,{});void 0===p&&(p=o);const f=e.layers[0].layout,d=f.get("icon-offset").evaluate(r,{},h),y=Hc(n.horizontal),m=o/24,g=e.tilePixelRatio*m,x=e.tilePixelRatio*p/24,v=e.tilePixelRatio*l,b=e.tilePixelRatio*f.get("symbol-spacing"),w=f.get("text-padding")*e.tilePixelRatio,_=function(t,e,r,n=1){const i=t.get("icon-padding").evaluate(e,{},r),a=i&&i.values;return [a[0]*n,a[1]*n,a[2]*n,a[3]*n]}(f,r,h,e.tilePixelRatio),S=f.get("text-max-angle")/180*Math.PI,k="viewport"!==f.get("text-rotation-alignment")&&"point"!==f.get("symbol-placement"),I="map"===f.get("icon-rotation-alignment")&&"point"!==f.get("symbol-placement"),z=f.get("symbol-placement"),M=b/2,C=f.get("icon-text-fit");let B;i&&"none"!==C&&(e.allowVerticalPlacement&&n.vertical&&(B=zu(i,n.vertical,C,f.get("icon-text-fit-padding"),d,m)),y&&(i=zu(i,y,C,f.get("icon-text-fit-padding"),d,m)));const P=(l,p)=>{p.x<0||p.x>=rs||p.y<0||p.y>=rs||function(e,r,n,i,a,s,o,l,u,c,h,p,f,d,y,m,g,x,v,b,w,_,S,k,I){const z=e.addToLineVertexArray(r,n);let M,C,B,P,V=0,E=0,F=0,T=0,$=-1,L=-1;const D={};let O=$a("");if(e.allowVerticalPlacement&&i.vertical){const t=l.layout.get("text-rotate").evaluate(w,{},k)+90;B=new Lc(u,r,c,h,p,i.vertical,f,d,y,t),o&&(P=new Lc(u,r,c,h,p,o,g,x,y,t));}if(a){const n=l.layout.get("icon-rotate").evaluate(w,{}),i="none"!==l.layout.get("icon-text-fit"),s=Vc(a,n,S,i),f=o?Vc(o,n,S,i):void 0;C=new Lc(u,r,c,h,p,a,g,x,!1,n),V=4*s.length;const d=e.iconSizeData;let y=null;"source"===d.kind?(y=[Cu*l.layout.get("icon-size").evaluate(w,{})],y[0]>Bu&&A(`${e.layerIds[0]}: Value for "icon-size" is >= ${Mu}. Reduce your "icon-size".`)):"composite"===d.kind&&(y=[Cu*_.compositeIconSizes[0].evaluate(w,{},k),Cu*_.compositeIconSizes[1].evaluate(w,{},k)],(y[0]>Bu||y[1]>Bu)&&A(`${e.layerIds[0]}: Value for "icon-size" is >= ${Mu}. Reduce your "icon-size".`)),e.addSymbols(e.icon,s,y,b,v,w,t.ai.none,r,z.lineStartIndex,z.lineLength,-1,k),$=e.icon.placedSymbolArray.length-1,f&&(E=4*f.length,e.addSymbols(e.icon,f,y,b,v,w,t.ai.vertical,r,z.lineStartIndex,z.lineLength,-1,k),L=e.icon.placedSymbolArray.length-1);}const U=Object.keys(i.horizontal);for(const n of U){const a=i.horizontal[n];if(!M){O=$a(a.text);const t=l.layout.get("text-rotate").evaluate(w,{},k);M=new Lc(u,r,c,h,p,a,f,d,y,t);}const o=1===a.positionedLines.length;if(F+=Yc(e,r,a,s,l,y,w,m,z,i.vertical?t.ai.horizontal:t.ai.horizontalOnly,o?U:[n],D,$,_,k),o)break}i.vertical&&(T+=Yc(e,r,i.vertical,s,l,y,w,m,z,t.ai.vertical,["vertical"],D,L,_,k));const R=M?M.boxStartIndex:e.collisionBoxArray.length,q=M?M.boxEndIndex:e.collisionBoxArray.length,j=B?B.boxStartIndex:e.collisionBoxArray.length,N=B?B.boxEndIndex:e.collisionBoxArray.length,Z=C?C.boxStartIndex:e.collisionBoxArray.length,K=C?C.boxEndIndex:e.collisionBoxArray.length,G=P?P.boxStartIndex:e.collisionBoxArray.length,J=P?P.boxEndIndex:e.collisionBoxArray.length;let X=-1;const Y=(t,e)=>t&&t.circleDiameter?Math.max(t.circleDiameter,e):e;X=Y(M,X),X=Y(B,X),X=Y(C,X),X=Y(P,X);const H=X>-1?1:0;H&&(X*=I/Vl),e.glyphOffsetArray.length>=Uu.MAX_GLYPHS&&A("Too many glyphs being rendered in a tile. See https://github.com/mapbox/mapbox-gl-js/issues/2907"),void 0!==w.sortKey&&e.addToSortKeyRanges(e.symbolInstances.length,w.sortKey);const W=Gc(l,w,k),[Q,tt]=function(e,r){const n=e.length,i=null==r?void 0:r.values;if((null==i?void 0:i.length)>0)for(let r=0;r=0?D.right:-1,D.center>=0?D.center:-1,D.left>=0?D.left:-1,D.vertical||-1,$,L,O,R,q,j,N,Z,K,G,J,c,F,T,V,E,H,0,f,X,Q,tt);}(e,p,l,n,i,a,B,e.layers[0],e.collisionBoxArray,r.index,r.sourceLayerIndex,e.index,g,[w,w,w,w],k,u,v,_,I,d,r,s,c,h,o);};if("line"===z)for(const t of _c(r.geometry,0,0,rs,rs)){const r=Cc(t,b,S,n.vertical||y,i,24,x,e.overscaling,rs);for(const n of r)y&&Wc(e,y.text,M,n)||P(t,n);}else if("line-center"===z){for(const t of r.geometry)if(t.length>1){const e=Mc(t,S,n.vertical||y,i,24,x);e&&P(t,e);}}else if("Polygon"===r.type)for(const t of Co(r.geometry,0)){const e=Uc(t,16);P(t[0],new Ac(e.x,e.y,0));}else if("LineString"===r.type)for(const t of r.geometry)P(t,new Ac(t[0].x,t[0].y,0));else if("Point"===r.type)for(const t of r.geometry)for(const e of t)P([e],new Ac(e.x,e.y,0));}function Yc(t,e,r,n,i,s,o,l,u,c,h,p,f,d,y){const m=function(t,e,r,n,i,s,o,l){const u=n.layout.get("text-rotate").evaluate(s,{})*Math.PI/180,c=[];for(const t of e.positionedLines)for(const n of t.positionedGlyphs){if(!n.rect)continue;const s=n.rect||{};let h=su+1,p=!0,f=1,d=0;const y=(i||l)&&n.vertical,m=n.metrics.advance*n.scale/2;if(l&&e.verticalizable&&(d=t.lineOffset/2-(n.imageName?-(Vl-n.metrics.width*n.scale)/2:(n.scale-1)*Vl)),n.imageName){const t=o[n.imageName];p=t.sdf,f=t.pixelRatio,h=lu/f;}const g=i?[n.x+m,n.y]:[0,0];let x=i?[0,0]:[n.x+m+r[0],n.y+r[1]-d],v=[0,0];y&&(v=x,x=[0,0]);const b=n.metrics.isDoubleResolution?2:1,w=(n.metrics.left-h)*n.scale-m+x[0],_=(-n.metrics.top-h)*n.scale+x[1],A=w+s.w/b*n.scale/f,S=_+s.h/b*n.scale/f,k=new a(w,_),I=new a(A,_),z=new a(w,S),M=new a(A,S);if(y){const t=new a(-m,m-pu),e=-Math.PI/2,r=Vl/2-m,i=new a(5-pu-r,-(n.imageName?r:0)),s=new a(...v);k._rotateAround(e,t)._add(i)._add(s),I._rotateAround(e,t)._add(i)._add(s),z._rotateAround(e,t)._add(i)._add(s),M._rotateAround(e,t)._add(i)._add(s);}if(u){const t=Math.sin(u),e=Math.cos(u),r=[e,-t,t,e];k._matMult(r),I._matMult(r),z._matMult(r),M._matMult(r);}const C=new a(0,0),B=new a(0,0);c.push({tl:k,tr:I,bl:z,br:M,tex:s,writingMode:e.writingMode,glyphOffset:g,sectionIndex:n.sectionIndex,isSDF:p,pixelOffsetTL:C,pixelOffsetBR:B,minFontScaleX:0,minFontScaleY:0});}return c}(0,r,l,i,s,o,n,t.allowVerticalPlacement),g=t.textSizeData;let x=null;"source"===g.kind?(x=[Cu*i.layout.get("text-size").evaluate(o,{})],x[0]>Bu&&A(`${t.layerIds[0]}: Value for "text-size" is >= ${Mu}. Reduce your "text-size".`)):"composite"===g.kind&&(x=[Cu*d.compositeTextSizes[0].evaluate(o,{},y),Cu*d.compositeTextSizes[1].evaluate(o,{},y)],(x[0]>Bu||x[1]>Bu)&&A(`${t.layerIds[0]}: Value for "text-size" is >= ${Mu}. Reduce your "text-size".`)),t.addSymbols(t.text,m,x,l,s,o,c,e,u.lineStartIndex,u.lineLength,f,y);for(const e of h)p[e]=t.text.placedSymbolArray.length-1;return 4*m.length}function Hc(t){for(const e in t)return t[e];return null}function Wc(t,e,r,n){const i=t.compareText;if(e in i){const t=i[e];for(let e=t.length-1;e>=0;e--)if(n.dist(t[e])>4;if(1!==n)throw new Error(`Got v${n} data when expected v1.`);const i=Qc[15&r];if(!i)throw new Error("Unrecognized array type.");const[a]=new Uint16Array(t,2,1),[s]=new Uint32Array(t,4,1);return new th(s,a,i,t)}constructor(t,e=64,r=Float64Array,n){if(isNaN(t)||t<0)throw new Error(`Unpexpected numItems value: ${t}.`);this.numItems=+t,this.nodeSize=Math.min(Math.max(+e,2),65535),this.ArrayType=r,this.IndexArrayType=t<65536?Uint16Array:Uint32Array;const i=Qc.indexOf(this.ArrayType),a=2*t*this.ArrayType.BYTES_PER_ELEMENT,s=t*this.IndexArrayType.BYTES_PER_ELEMENT,o=(8-s%8)%8;if(i<0)throw new Error(`Unexpected typed array class: ${r}.`);n&&n instanceof ArrayBuffer?(this.data=n,this.ids=new this.IndexArrayType(this.data,8,t),this.coords=new this.ArrayType(this.data,8+s+o,2*t),this._pos=2*t,this._finished=!0):(this.data=new ArrayBuffer(8+a+s+o),this.ids=new this.IndexArrayType(this.data,8,t),this.coords=new this.ArrayType(this.data,8+s+o,2*t),this._pos=0,this._finished=!1,new Uint8Array(this.data,0,2).set([219,16+i]),new Uint16Array(this.data,2,1)[0]=e,new Uint32Array(this.data,4,1)[0]=t);}add(t,e){const r=this._pos>>1;return this.ids[r]=r,this.coords[this._pos++]=t,this.coords[this._pos++]=e,r}finish(){const t=this._pos>>1;if(t!==this.numItems)throw new Error(`Added ${t} items when expected ${this.numItems}.`);return eh(this.ids,this.coords,this.nodeSize,0,this.numItems-1,0),this._finished=!0,this}range(t,e,r,n){if(!this._finished)throw new Error("Data not yet indexed - call index.finish().");const{ids:i,coords:a,nodeSize:s}=this,o=[0,i.length-1,0],l=[];for(;o.length;){const u=o.pop()||0,c=o.pop()||0,h=o.pop()||0;if(c-h<=s){for(let s=h;s<=c;s++){const o=a[2*s],u=a[2*s+1];o>=t&&o<=r&&u>=e&&u<=n&&l.push(i[s]);}continue}const p=h+c>>1,f=a[2*p],d=a[2*p+1];f>=t&&f<=r&&d>=e&&d<=n&&l.push(i[p]),(0===u?t<=f:e<=d)&&(o.push(h),o.push(p-1),o.push(1-u)),(0===u?r>=f:n>=d)&&(o.push(p+1),o.push(c),o.push(1-u));}return l}within(t,e,r){if(!this._finished)throw new Error("Data not yet indexed - call index.finish().");const{ids:n,coords:i,nodeSize:a}=this,s=[0,n.length-1,0],o=[],l=r*r;for(;s.length;){const u=s.pop()||0,c=s.pop()||0,h=s.pop()||0;if(c-h<=a){for(let r=h;r<=c;r++)ah(i[2*r],i[2*r+1],t,e)<=l&&o.push(n[r]);continue}const p=h+c>>1,f=i[2*p],d=i[2*p+1];ah(f,d,t,e)<=l&&o.push(n[p]),(0===u?t-r<=f:e-r<=d)&&(s.push(h),s.push(p-1),s.push(1-u)),(0===u?t+r>=f:e+r>=d)&&(s.push(p+1),s.push(c),s.push(1-u));}return o}}function eh(t,e,r,n,i,a){if(i-n<=r)return;const s=n+i>>1;rh(t,e,s,n,i,a),eh(t,e,r,n,s-1,1-a),eh(t,e,r,s+1,i,1-a);}function rh(t,e,r,n,i,a){for(;i>n;){if(i-n>600){const s=i-n+1,o=r-n+1,l=Math.log(s),u=.5*Math.exp(2*l/3),c=.5*Math.sqrt(l*u*(s-u)/s)*(o-s/2<0?-1:1);rh(t,e,r,Math.max(n,Math.floor(r-o*u/s+c)),Math.min(i,Math.floor(r+(s-o)*u/s+c)),a);}const s=e[2*r+a];let o=n,l=i;for(nh(t,e,n,r),e[2*i+a]>s&&nh(t,e,n,i);os;)l--;}e[2*n+a]===s?nh(t,e,n,l):(l++,nh(t,e,l,i)),l<=r&&(n=l+1),r<=l&&(i=l-1);}}function nh(t,e,r,n){ih(t,r,n),ih(e,2*r,2*n),ih(e,2*r+1,2*n+1);}function ih(t,e,r){const n=t[e];t[e]=t[r],t[r]=n;}function ah(t,e,r,n){const i=t-r,a=e-n;return i*i+a*a}var sh;t.bh=void 0,(sh=t.bh||(t.bh={})).create="create",sh.load="load",sh.fullLoad="fullLoad";let oh=null,lh=[];const uh=1e3/60,ch="loadTime",hh="fullLoadTime",ph={mark(t){performance.mark(t);},frame(t){const e=t;null!=oh&&lh.push(e-oh),oh=e;},clearMetrics(){oh=null,lh=[],performance.clearMeasures(ch),performance.clearMeasures(hh);for(const e in t.bh)performance.clearMarks(t.bh[e]);},getPerformanceMetrics(){performance.measure(ch,t.bh.create,t.bh.load),performance.measure(hh,t.bh.create,t.bh.fullLoad);const e=performance.getEntriesByName(ch)[0].duration,r=performance.getEntriesByName(hh)[0].duration,n=lh.length,i=1/(lh.reduce(((t,e)=>t+e),0)/n/1e3),a=lh.filter((t=>t>uh)).reduce(((t,e)=>t+(e-uh)/uh),0);return {loadTime:e,fullLoadTime:r,fps:i,percentDroppedFrames:a/(n+a)*100,totalFrames:n}}};t.$=function(t,e,r){var n,i,a,s,o,l,u,c,h,p,f,d,y=r[0],m=r[1],g=r[2];return e===t?(t[12]=e[0]*y+e[4]*m+e[8]*g+e[12],t[13]=e[1]*y+e[5]*m+e[9]*g+e[13],t[14]=e[2]*y+e[6]*m+e[10]*g+e[14],t[15]=e[3]*y+e[7]*m+e[11]*g+e[15]):(i=e[1],a=e[2],s=e[3],o=e[4],l=e[5],u=e[6],c=e[7],h=e[8],p=e[9],f=e[10],d=e[11],t[0]=n=e[0],t[1]=i,t[2]=a,t[3]=s,t[4]=o,t[5]=l,t[6]=u,t[7]=c,t[8]=h,t[9]=p,t[10]=f,t[11]=d,t[12]=n*y+o*m+h*g+e[12],t[13]=i*y+l*m+p*g+e[13],t[14]=a*y+u*m+f*g+e[14],t[15]=s*y+c*m+d*g+e[15]),t},t.A=zs,t.B=Ze,t.C=class{constructor(t,e,r){this.receive=t=>{const e=t.data,r=e.id;if(r&&(!e.targetMapId||this.mapId===e.targetMapId))if(""===e.type){delete this.tasks[r];const t=this.cancelCallbacks[r];delete this.cancelCallbacks[r],t&&t();}else I()||e.mustQueue?(this.tasks[r]=e,this.taskQueue.push(r),this.invoker.trigger()):this.processTask(r,e);},this.process=()=>{if(!this.taskQueue.length)return;const t=this.taskQueue.shift(),e=this.tasks[t];delete this.tasks[t],this.taskQueue.length&&this.invoker.trigger(),e&&this.processTask(t,e);},this.target=t,this.parent=e,this.mapId=r,this.callbacks={},this.tasks={},this.taskQueue=[],this.cancelCallbacks={},this.invoker=new Qu(this.process),this.target.addEventListener("message",this.receive,!1),this.globalScope=I()?t:window;}send(t,e,r,n,i=!1){const a=Math.round(1e18*Math.random()).toString(36).substring(0,10);r&&(this.callbacks[a]=r);const s=[],o={id:a,type:t,hasCallback:!!r,targetMapId:n,mustQueue:i,sourceMapId:this.mapId,data:Dn(e,s)};return this.target.postMessage(o,{transfer:s}),{cancel:()=>{r&&delete this.callbacks[a],this.target.postMessage({id:a,type:"",targetMapId:n,sourceMapId:this.mapId});}}}processTask(t,e){if(""===e.type){const r=this.callbacks[t];delete this.callbacks[t],r&&(e.error?r(On(e.error)):r(null,On(e.data)));}else {let r=!1;const n=[],i=e.hasCallback?(e,i)=>{r=!0,delete this.cancelCallbacks[t];const a={id:t,type:"",sourceMapId:this.mapId,error:e?Dn(e):null,data:Dn(i,n)};this.target.postMessage(a,{transfer:n});}:t=>{r=!0;};let a=null;const s=On(e.data);if(this.parent[e.type])a=this.parent[e.type](e.sourceMapId,s,i);else if("getWorkerSource"in this.parent){const t=e.type.split(".");a=this.parent.getWorkerSource(e.sourceMapId,t[0],s.source)[t[1]](s,i);}else i(new Error(`Could not find function ${e.type}`));!r&&a&&a.cancel&&(this.cancelCallbacks[t]=a.cancel);}}remove(){this.invoker.remove(),this.target.removeEventListener("message",this.receive,!1);}},t.D=gi,t.E=J,t.F=function(t,e){const r={};for(let n=0;n{}}},t.Y=ot,t.Z=function(){var t=new zs(16);return zs!=Float32Array&&(t[1]=0,t[2]=0,t[3]=0,t[4]=0,t[6]=0,t[7]=0,t[8]=0,t[9]=0,t[11]=0,t[12]=0,t[13]=0,t[14]=0),t[0]=1,t[5]=1,t[10]=1,t[15]=1,t},t._=e,t.a=M,t.a$=class extends Hi{},t.a0=function(t,e,r){var n=r[0],i=r[1],a=r[2];return t[0]=e[0]*n,t[1]=e[1]*n,t[2]=e[2]*n,t[3]=e[3]*n,t[4]=e[4]*i,t[5]=e[5]*i,t[6]=e[6]*i,t[7]=e[7]*i,t[8]=e[8]*a,t[9]=e[9]*a,t[10]=e[10]*a,t[11]=e[11]*a,t[12]=e[12],t[13]=e[13],t[14]=e[14],t[15]=e[15],t},t.a1=Cs,t.a2=function(){return x++},t.a3=ta,t.a4=Uu,t.a5=function(){oi.isLoading()||oi.isLoaded()||"deferred"!==ai()||si();},t.a6=Nr,t.a7=ss,t.a8=li,t.a9=gc,t.aA=ri,t.aB=function(t){t=t.slice();const e=Object.create(null);for(let r=0;r{n[t.source]?r.push({command:Q.removeLayer,args:[t.id]}):a.push(t);})),r=r.concat(i),function(t,e,r){e=e||[];const n=(t=t||[]).map(at),i=e.map(at),a=t.reduce(st,{}),s=e.reduce(st,{}),o=n.slice(),l=Object.create(null);let u,c,h,p,f,d,y;for(u=0,c=0;u@\,;\:\\"\/\[\]\?\=\{\}\x7F]+)(?:\=(?:([^\x00-\x20\(\)<>@\,;\:\\"\/\[\]\?\=\{\}\x7F]+)|(?:\"((?:[^"\\]|\\.)*)\")))?/g,((t,r,n,i)=>{const a=n||i;return e[r]=!a||a.toLowerCase(),""})),e["max-age"]){const t=parseInt(e["max-age"],10);isNaN(t)?delete e["max-age"]:e["max-age"]=t;}return e},t.ab=function(t,e){const r=[];for(const n in t)n in e||r.push(n);return r},t.ac=function(t){if(null==z){const e=t.navigator?t.navigator.userAgent:null;z=!!t.safari||!(!e||!(/\b(iPad|iPhone|iPod)\b/.test(e)||e.match("Safari")&&!e.match("Chrome")));}return z},t.ad=y,t.ae=function(t,e,r){var n=Math.sin(r),i=Math.cos(r),a=e[0],s=e[1],o=e[2],l=e[3],u=e[4],c=e[5],h=e[6],p=e[7];return e!==t&&(t[8]=e[8],t[9]=e[9],t[10]=e[10],t[11]=e[11],t[12]=e[12],t[13]=e[13],t[14]=e[14],t[15]=e[15]),t[0]=a*i+u*n,t[1]=s*i+c*n,t[2]=o*i+h*n,t[3]=l*i+p*n,t[4]=u*i-a*n,t[5]=c*i-s*n,t[6]=h*i-o*n,t[7]=p*i-l*n,t},t.af=function(t){var e=new zs(16);return e[0]=t[0],e[1]=t[1],e[2]=t[2],e[3]=t[3],e[4]=t[4],e[5]=t[5],e[6]=t[6],e[7]=t[7],e[8]=t[8],e[9]=t[9],e[10]=t[10],e[11]=t[11],e[12]=t[12],e[13]=t[13],e[14]=t[14],e[15]=t[15],e},t.ag=Vs,t.ah=function(t,e){let r=0,n=0;if("constant"===t.kind)n=t.layoutSize;else if("source"!==t.kind){const{interpolationType:i,minZoom:a,maxZoom:s}=t,o=i?y(Ke.interpolationFactor(i,e,a,s),0,1):0;"camera"===t.kind?n=Ze.number(t.minSize,t.maxSize,o):r=o;}return {uSizeT:r,uSize:n}},t.aj=function(t,{uSize:e,uSizeT:r},{lowerSize:n,upperSize:i}){return "source"===t.kind?n/Cu:"composite"===t.kind?Ze.number(n/Cu,i/Cu,r):e},t.ak=$u,t.al=function(t,e,r,n){const i=e.y-t.y,s=e.x-t.x,o=n.y-r.y,l=n.x-r.x,u=o*s-l*i;if(0===u)return null;const c=(l*(t.y-r.y)-o*(t.x-r.x))/u;return new a(t.x+c*s,t.y+c*i)},t.am=_c,t.an=us,t.ao=Ms,t.ap=Vl,t.ar=Vu,t.as=function(t,e){var r=e[0],n=e[1],i=e[2],a=e[3],s=e[4],o=e[5],l=e[6],u=e[7],c=e[8],h=e[9],p=e[10],f=e[11],d=e[12],y=e[13],m=e[14],g=e[15],x=r*o-n*s,v=r*l-i*s,b=r*u-a*s,w=n*l-i*o,_=n*u-a*o,A=i*u-a*l,S=c*y-h*d,k=c*m-p*d,I=c*g-f*d,z=h*m-p*y,M=h*g-f*y,C=p*g-f*m,B=x*C-v*M+b*z+w*I-_*k+A*S;return B?(t[0]=(o*C-l*M+u*z)*(B=1/B),t[1]=(i*M-n*C-a*z)*B,t[2]=(y*A-m*_+g*w)*B,t[3]=(p*_-h*A-f*w)*B,t[4]=(l*I-s*C-u*k)*B,t[5]=(r*C-i*I+a*k)*B,t[6]=(m*b-d*A-g*v)*B,t[7]=(c*A-p*b+f*v)*B,t[8]=(s*M-o*I+u*S)*B,t[9]=(n*I-r*M-a*S)*B,t[10]=(d*_-y*b+g*x)*B,t[11]=(h*b-c*_-f*x)*B,t[12]=(o*k-s*z-l*S)*B,t[13]=(r*z-n*k+i*S)*B,t[14]=(y*v-d*w-m*x)*B,t[15]=(c*w-h*v+p*x)*B,t):null},t.at=Jc,t.au=Su,t.av=th,t.aw=function(){const t={},e=X.$version;for(const r in X.$root){const n=X.$root[r];if(n.required){let i=null;i="version"===r?e:"array"===n.type?[]:{},null!=i&&(t[r]=i);}}return t},t.ax=Q,t.ay=Un,t.az=D,t.b=function(t,e){const r=new Blob([new Uint8Array(t)],{type:"image/png"});createImageBitmap(r).then((t=>{e(null,t);})).catch((t=>{e(new Error(`Could not load image because of ${t.message}. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.`));}));},t.b0=Aa,t.b1=function(t,e){var r=t[0],n=t[1],i=t[2],a=t[3],s=t[4],o=t[5],l=t[6],u=t[7],c=t[8],h=t[9],p=t[10],f=t[11],d=t[12],y=t[13],m=t[14],g=t[15],x=e[0],v=e[1],b=e[2],w=e[3],_=e[4],A=e[5],S=e[6],k=e[7],I=e[8],z=e[9],M=e[10],C=e[11],B=e[12],P=e[13],V=e[14],E=e[15];return Math.abs(r-x)<=Is*Math.max(1,Math.abs(r),Math.abs(x))&&Math.abs(n-v)<=Is*Math.max(1,Math.abs(n),Math.abs(v))&&Math.abs(i-b)<=Is*Math.max(1,Math.abs(i),Math.abs(b))&&Math.abs(a-w)<=Is*Math.max(1,Math.abs(a),Math.abs(w))&&Math.abs(s-_)<=Is*Math.max(1,Math.abs(s),Math.abs(_))&&Math.abs(o-A)<=Is*Math.max(1,Math.abs(o),Math.abs(A))&&Math.abs(l-S)<=Is*Math.max(1,Math.abs(l),Math.abs(S))&&Math.abs(u-k)<=Is*Math.max(1,Math.abs(u),Math.abs(k))&&Math.abs(c-I)<=Is*Math.max(1,Math.abs(c),Math.abs(I))&&Math.abs(h-z)<=Is*Math.max(1,Math.abs(h),Math.abs(z))&&Math.abs(p-M)<=Is*Math.max(1,Math.abs(p),Math.abs(M))&&Math.abs(f-C)<=Is*Math.max(1,Math.abs(f),Math.abs(C))&&Math.abs(d-B)<=Is*Math.max(1,Math.abs(d),Math.abs(B))&&Math.abs(y-P)<=Is*Math.max(1,Math.abs(y),Math.abs(P))&&Math.abs(m-V)<=Is*Math.max(1,Math.abs(m),Math.abs(V))&&Math.abs(g-E)<=Is*Math.max(1,Math.abs(g),Math.abs(E))},t.b2=function(t,e){return t[0]=e[0],t[1]=e[1],t[2]=e[2],t[3]=e[3],t[4]=e[4],t[5]=e[5],t[6]=e[6],t[7]=e[7],t[8]=e[8],t[9]=e[9],t[10]=e[10],t[11]=e[11],t[12]=e[12],t[13]=e[13],t[14]=e[14],t[15]=e[15],t},t.b3=function(t,e,r){return t[0]=e[0]*r[0],t[1]=e[1]*r[1],t[2]=e[2]*r[2],t[3]=e[3]*r[3],t},t.b4=function(t,e){return t[0]*e[0]+t[1]*e[1]+t[2]*e[2]+t[3]*e[3]},t.b5=m,t.b6=pc,t.b7=sc,t.b8=function(t,e,r,n,i){var a,s=1/Math.tan(e/2);return t[0]=s/r,t[1]=0,t[2]=0,t[3]=0,t[4]=0,t[5]=s,t[6]=0,t[7]=0,t[8]=0,t[9]=0,t[11]=-1,t[12]=0,t[13]=0,t[15]=0,null!=i&&i!==1/0?(t[10]=(i+n)*(a=1/(n-i)),t[14]=2*i*n*a):(t[10]=-1,t[14]=-2*n),t},t.b9=function(t,e,r){var n=Math.sin(r),i=Math.cos(r),a=e[4],s=e[5],o=e[6],l=e[7],u=e[8],c=e[9],h=e[10],p=e[11];return e!==t&&(t[0]=e[0],t[1]=e[1],t[2]=e[2],t[3]=e[3],t[12]=e[12],t[13]=e[13],t[14]=e[14],t[15]=e[15]),t[4]=a*i+u*n,t[5]=s*i+c*n,t[6]=o*i+h*n,t[7]=l*i+p*n,t[8]=u*i-a*n,t[9]=c*i-s*n,t[10]=h*i-o*n,t[11]=p*i-l*n,t},t.bA=n,t.bB=El,t.bC=$r,t.bD=oi,t.ba=f,t.bb=d,t.bc=function(t,e){return t[0]=e[0],t[1]=0,t[2]=0,t[3]=0,t[4]=0,t[5]=e[1],t[6]=0,t[7]=0,t[8]=0,t[9]=0,t[10]=e[2],t[11]=0,t[12]=0,t[13]=0,t[14]=0,t[15]=1,t},t.bd=class extends Pi{},t.be=tc,t.bf=oc,t.bg=ph,t.bi=L,t.bj=function(t,e,r=!1){if(ti===Yn||ti===Hn||ti===Wn)throw new Error("setRTLTextPlugin cannot be called multiple times.");ei=T.resolveURL(t),ti=Yn,Qn=e,ni(),r||si();},t.bk=ai,t.bl=function(t,e){const r={};for(let n=0;nt*Vl));}let v=o?"center":n.get("text-justify").evaluate(i,{},e.canonical);const b=n.get("symbol-placement"),w="point"===b?n.get("text-max-width").evaluate(i,{},e.canonical)*Vl:0,_=()=>{e.bucket.allowVerticalPlacement&&qn(a)&&(d.vertical=yu(y,e.glyphMap,e.glyphPositions,e.imagePositions,c,w,s,m,"left",f,g,t.ai.vertical,!0,b,p,h));};if(!o&&x){const r=new Set;if("auto"===v)for(let t=0;t{e(null,r),URL.revokeObjectURL(r.src),r.onload=null,window.requestAnimationFrame((()=>{r.src=C;}));},r.onerror=()=>e(new Error("Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported."));const n=new Blob([new Uint8Array(t)],{type:"image/png"});r.src=t.byteLength?URL.createObjectURL(n):C;},t.e=g,t.f=function(t,e){return R(g(t,{type:"json"}),e)},t.g=O,t.h=T,t.i=I,t.j=G,t.k=K,t.l=q,t.m=R,t.n=function(t){return new eu(t).readFields(nu,[])},t.o=function(t,e,r){if(!t.length)return r(null,[]);let n=t.length;const i=new Array(t.length);let a=null;t.forEach(((t,s)=>{e(t,((t,e)=>{t&&(a=t),i[s]=e,0==--n&&r(a,i);}));}));},t.p=ou,t.q=Rs,t.r=_i,t.s=j,t.t=Bn,t.u=Rn,t.v=X,t.w=A,t.x=En,t.y=Cn,t.z=function([t,e,r]){return e+=90,e*=Math.PI/180,r*=Math.PI/180,{x:t*Math.cos(e)*Math.sin(r),y:t*Math.sin(e)*Math.sin(r),z:t*Math.cos(r)}};})); + +define(["./shared"],(function(e){"use strict";class t{constructor(e){this.keyCache={},e&&this.replace(e);}replace(e){this._layerConfigs={},this._layers={},this.update(e,[]);}update(t,i){for(const i of t){this._layerConfigs[i.id]=i;const t=this._layers[i.id]=e.aC(i);t._featureFilter=e.a6(t.filter),this.keyCache[i.id]&&delete this.keyCache[i.id];}for(const e of i)delete this.keyCache[e],delete this._layerConfigs[e],delete this._layers[e];this.familiesBySource={};const o=e.bl(Object.values(this._layerConfigs),this.keyCache);for(const e of o){const t=e.map((e=>this._layers[e.id])),i=t[0];if("none"===i.visibility)continue;const o=i.source||"";let r=this.familiesBySource[o];r||(r=this.familiesBySource[o]={});const s=i.sourceLayer||"_geojsonTileLayer";let n=r[s];n||(n=r[s]=[]),n.push(t);}}}class i{constructor(t){const i={},o=[];for(const e in t){const r=t[e],s=i[e]={};for(const e in r){const t=r[+e];if(!t||0===t.bitmap.width||0===t.bitmap.height)continue;const i={x:0,y:0,w:t.bitmap.width+2,h:t.bitmap.height+2};o.push(i),s[e]={rect:i,metrics:t.metrics};}}const{w:r,h:s}=e.p(o),n=new e.q({width:r||1,height:s||1});for(const o in t){const r=t[o];for(const t in r){const s=r[+t];if(!s||0===s.bitmap.width||0===s.bitmap.height)continue;const a=i[o][t].rect;e.q.copy(s.bitmap,n,{x:0,y:0},{x:a.x+1,y:a.y+1},s.bitmap);}}this.image=n,this.positions=i;}}e.bm("GlyphAtlas",i);class o{constructor(t){this.tileID=new e.O(t.tileID.overscaledZ,t.tileID.wrap,t.tileID.canonical.z,t.tileID.canonical.x,t.tileID.canonical.y),this.uid=t.uid,this.zoom=t.zoom,this.pixelRatio=t.pixelRatio,this.tileSize=t.tileSize,this.source=t.source,this.overscaling=this.tileID.overscaleFactor(),this.showCollisionBoxes=t.showCollisionBoxes,this.collectResourceTiming=!!t.collectResourceTiming,this.returnDependencies=!!t.returnDependencies,this.promoteId=t.promoteId,this.inFlightDependencies=[],this.dependencySentinel=-1;}parse(t,o,s,n,a){this.status="parsing",this.data=t,this.collisionBoxArray=new e.a3;const l=new e.bn(Object.keys(t.layers).sort()),c=new e.bo(this.tileID,this.promoteId);c.bucketLayerIDs=[];const h={},u={featureIndex:c,iconDependencies:{},patternDependencies:{},glyphDependencies:{},availableImages:s},d=o.familiesBySource[this.source];for(const i in d){const o=t.layers[i];if(!o)continue;1===o.version&&e.w(`Vector tile source "${this.source}" layer "${i}" does not use vector tile spec v2 and therefore may have some rendering errors.`);const n=l.encode(i),a=[];for(let e=0;e=i.maxzoom||"none"!==i.visibility&&(r(t,this.zoom,s),(h[i.id]=i.createBucket({index:c.bucketLayerIDs.length,layers:t,zoom:this.zoom,pixelRatio:this.pixelRatio,overscaling:this.overscaling,collisionBoxArray:this.collisionBoxArray,sourceLayerIndex:n,sourceID:this.source})).populate(a,u,this.tileID.canonical),c.bucketLayerIDs.push(t.map((e=>e.id))));}}let p,f,g,m;const y=e.aH(u.glyphDependencies,(e=>Object.keys(e).map(Number)));this.inFlightDependencies.forEach((e=>null==e?void 0:e.cancel())),this.inFlightDependencies=[];const v=++this.dependencySentinel;Object.keys(y).length?this.inFlightDependencies.push(n.send("getGlyphs",{uid:this.uid,stacks:y,source:this.source,tileID:this.tileID,type:"glyphs"},((e,t)=>{v===this.dependencySentinel&&(p||(p=e,f=t,b.call(this)));}))):f={};const w=Object.keys(u.iconDependencies);w.length?this.inFlightDependencies.push(n.send("getImages",{icons:w,source:this.source,tileID:this.tileID,type:"icons"},((e,t)=>{v===this.dependencySentinel&&(p||(p=e,g=t,b.call(this)));}))):g={};const x=Object.keys(u.patternDependencies);function b(){if(p)return a(p);if(f&&g&&m){const t=new i(f),o=new e.bp(g,m);for(const i in h){const n=h[i];n instanceof e.a4?(r(n.layers,this.zoom,s),e.bq({bucket:n,glyphMap:f,glyphPositions:t.positions,imageMap:g,imagePositions:o.iconPositions,showCollisionBoxes:this.showCollisionBoxes,canonical:this.tileID.canonical})):n.hasPattern&&(n instanceof e.br||n instanceof e.bs||n instanceof e.bt)&&(r(n.layers,this.zoom,s),n.addFeatures(u,this.tileID.canonical,o.patternPositions));}this.status="done",a(null,{buckets:Object.values(h).filter((e=>!e.isEmpty())),featureIndex:c,collisionBoxArray:this.collisionBoxArray,glyphAtlasImage:t.image,imageAtlas:o,glyphMap:this.returnDependencies?f:null,iconMap:this.returnDependencies?g:null,glyphPositions:this.returnDependencies?t.positions:null});}}x.length?this.inFlightDependencies.push(n.send("getImages",{icons:x,source:this.source,tileID:this.tileID,type:"patterns"},((e,t)=>{v===this.dependencySentinel&&(p||(p=e,m=t,b.call(this)));}))):m={},b.call(this);}}function r(t,i,o){const r=new e.a8(i);for(const e of t)e.recalculate(r,o);}function s(t,i){const o=e.l(t.request,((o,r,s,n)=>{if(o)i(o);else if(r)try{const t=new e.bw.VectorTile(new e.bv(r));i(null,{vectorTile:t,rawData:r,cacheControl:s,expires:n});}catch(e){const o=new Uint8Array(r);let s=`Unable to parse the tile at ${t.request.url}, `;s+=31===o[0]&&139===o[1]?"please make sure the data is not gzipped and that you have configured the relevant header in the server":`got error: ${e.messge}`,i(new Error(s));}}));return ()=>{o.cancel(),i();}}class n{constructor(e,t,i,o){this.actor=e,this.layerIndex=t,this.availableImages=i,this.loadVectorData=o||s,this.fetching={},this.loading={},this.loaded={};}loadTile(t,i){const r=t.uid;this.loading||(this.loading={});const s=!!(t&&t.request&&t.request.collectResourceTiming)&&new e.bu(t.request),n=this.loading[r]=new o(t);n.abort=this.loadVectorData(t,((t,o)=>{if(delete this.loading[r],t||!o)return n.status="done",this.loaded[r]=n,i(t);const a=o.rawData,l={};o.expires&&(l.expires=o.expires),o.cacheControl&&(l.cacheControl=o.cacheControl);const c={};if(s){const e=s.finish();e&&(c.resourceTiming=JSON.parse(JSON.stringify(e)));}n.vectorTile=o.vectorTile,n.parse(o.vectorTile,this.layerIndex,this.availableImages,this.actor,((t,o)=>{if(delete this.fetching[r],t||!o)return i(t);i(null,e.e({rawTileData:a.slice(0)},o,l,c));})),this.loaded=this.loaded||{},this.loaded[r]=n,this.fetching[r]={rawTileData:a,cacheControl:l,resourceTiming:c};}));}reloadTile(t,i){const o=this.loaded,r=t.uid;if(o&&o[r]){const s=o[r];s.showCollisionBoxes=t.showCollisionBoxes,"parsing"===s.status?s.parse(s.vectorTile,this.layerIndex,this.availableImages,this.actor,((t,o)=>{if(t||!o)return i(t,o);let s;if(this.fetching[r]){const{rawTileData:t,cacheControl:i,resourceTiming:n}=this.fetching[r];delete this.fetching[r],s=e.e({rawTileData:t.slice(0)},o,i,n);}else s=o;i(null,s);})):"done"===s.status&&(s.vectorTile?s.parse(s.vectorTile,this.layerIndex,this.availableImages,this.actor,i):i());}}abortTile(e,t){const i=this.loading,o=e.uid;i&&i[o]&&i[o].abort&&(i[o].abort(),delete i[o]),t();}removeTile(e,t){const i=this.loaded,o=e.uid;i&&i[o]&&delete i[o],t();}}class a{constructor(){this.loaded={};}loadTile(t,i){return e._(this,void 0,void 0,(function*(){const{uid:o,encoding:r,rawImageData:s,redFactor:n,greenFactor:a,blueFactor:l,baseShift:c}=t,h=s.width+2,u=s.height+2,d=e.a(s)?new e.R({width:h,height:u},yield e.bx(s,-1,-1,h,u)):s,p=new e.by(o,d,r,n,a,l,c);this.loaded=this.loaded||{},this.loaded[o]=p,i(null,p);}))}removeTile(e){const t=this.loaded,i=e.uid;t&&t[i]&&delete t[i];}}function l(e,t){if(0!==e.length){c(e[0],t);for(var i=1;i=Math.abs(a)?i-l+a:a-l+i,i=l;}i+o>=0!=!!t&&e.reverse();}var h=e.bz((function e(t,i){var o,r=t&&t.type;if("FeatureCollection"===r)for(o=0;o>31}function T(e,t){for(var i=e.loadGeometry(),o=e.type,r=0,s=0,n=i.length,a=0;ae},O=Math.fround||(C=new Float32Array(1),e=>(C[0]=+e,C[0]));var C;const L=3,F=5,z=6;class N{constructor(e){this.options=Object.assign(Object.create(D),e),this.trees=new Array(this.options.maxZoom+1),this.stride=this.options.reduce?7:6,this.clusterProps=[];}load(e){const{log:t,minZoom:i,maxZoom:o}=this.options;t&&console.time("total time");const r=`prepare ${e.length} points`;t&&console.time(r),this.points=e;const s=[];for(let t=0;t=i;e--){const i=+Date.now();n=this.trees[e]=this._createTree(this._cluster(n,e)),t&&console.log("z%d: %d clusters in %dms",e,n.numItems,+Date.now()-i);}return t&&console.timeEnd("total time"),this}getClusters(e,t){let i=((e[0]+180)%360+360)%360-180;const o=Math.max(-90,Math.min(90,e[1]));let r=180===e[2]?180:((e[2]+180)%360+360)%360-180;const s=Math.max(-90,Math.min(90,e[3]));if(e[2]-e[0]>=360)i=-180,r=180;else if(i>r){const e=this.getClusters([i,o,180,s],t),n=this.getClusters([-180,o,r,s],t);return e.concat(n)}const n=this.trees[this._limitZoom(t)],a=n.range(Z(i),G(s),Z(r),G(o)),l=n.data,c=[];for(const e of a){const t=this.stride*e;c.push(l[t+F]>1?E(l,t,this.clusterProps):this.points[l[t+L]]);}return c}getChildren(e){const t=this._getOriginId(e),i=this._getOriginZoom(e),o="No cluster with the specified id.",r=this.trees[i];if(!r)throw new Error(o);const s=r.data;if(t*this.stride>=s.length)throw new Error(o);const n=this.options.radius/(this.options.extent*Math.pow(2,i-1)),a=r.within(s[t*this.stride],s[t*this.stride+1],n),l=[];for(const t of a){const i=t*this.stride;s[i+4]===e&&l.push(s[i+F]>1?E(s,i,this.clusterProps):this.points[s[i+L]]);}if(0===l.length)throw new Error(o);return l}getLeaves(e,t,i){const o=[];return this._appendLeaves(o,e,t=t||10,i=i||0,0),o}getTile(e,t,i){const o=this.trees[this._limitZoom(e)],r=Math.pow(2,e),{extent:s,radius:n}=this.options,a=n/s,l=(i-a)/r,c=(i+1+a)/r,h={features:[]};return this._addTileFeatures(o.range((t-a)/r,l,(t+1+a)/r,c),o.data,t,i,r,h),0===t&&this._addTileFeatures(o.range(1-a/r,l,1,c),o.data,r,i,r,h),t===r-1&&this._addTileFeatures(o.range(0,l,a/r,c),o.data,-1,i,r,h),h.features.length?h:null}getClusterExpansionZoom(e){let t=this._getOriginZoom(e)-1;for(;t<=this.options.maxZoom;){const i=this.getChildren(e);if(t++,1!==i.length)break;e=i[0].properties.cluster_id;}return t}_appendLeaves(e,t,i,o,r){const s=this.getChildren(t);for(const t of s){const s=t.properties;if(s&&s.cluster?r+s.point_count<=o?r+=s.point_count:r=this._appendLeaves(e,s.cluster_id,i,o,r):r1;let l,c,h;if(a)l=j(t,e,this.clusterProps),c=t[e],h=t[e+1];else {const i=this.points[t[e+L]];l=i.properties;const[o,r]=i.geometry.coordinates;c=Z(o),h=G(r);}const u={type:1,geometry:[[Math.round(this.options.extent*(c*r-i)),Math.round(this.options.extent*(h*r-o))]],tags:l};let d;d=a||this.options.generateId?t[e+L]:this.points[t[e+L]].id,void 0!==d&&(u.id=d),s.features.push(u);}}_limitZoom(e){return Math.max(this.options.minZoom,Math.min(Math.floor(+e),this.options.maxZoom+1))}_cluster(e,t){const{radius:i,extent:o,reduce:r,minPoints:s}=this.options,n=i/(o*Math.pow(2,t)),a=e.data,l=[],c=this.stride;for(let i=0;it&&(p+=a[i+F]);}if(p>d&&p>=s){let e,s=o*d,n=h*d,f=-1;const g=((i/c|0)<<5)+(t+1)+this.points.length;for(const o of u){const l=o*c;if(a[l+2]<=t)continue;a[l+2]=t;const h=a[l+F];s+=a[l]*h,n+=a[l+1]*h,a[l+4]=g,r&&(e||(e=this._map(a,i,!0),f=this.clusterProps.length,this.clusterProps.push(e)),r(e,this._map(a,l)));}a[i+4]=g,l.push(s/p,n/p,1/0,g,-1,p),r&&l.push(f);}else {for(let e=0;e1)for(const e of u){const i=e*c;if(!(a[i+2]<=t)){a[i+2]=t;for(let e=0;e>5}_getOriginZoom(e){return (e-this.points.length)%32}_map(e,t,i){if(e[t+F]>1){const o=this.clusterProps[e[t+z]];return i?Object.assign({},o):o}const o=this.points[e[t+L]].properties,r=this.options.map(o);return i&&r===o?Object.assign({},r):r}}function E(e,t,i){return {type:"Feature",id:e[t+L],properties:j(e,t,i),geometry:{type:"Point",coordinates:[(o=e[t],360*(o-.5)),J(e[t+1])]}};var o;}function j(e,t,i){const o=e[t+F],r=o>=1e4?`${Math.round(o/1e3)}k`:o>=1e3?Math.round(o/100)/10+"k":o,s=e[t+z],n=-1===s?{}:Object.assign({},i[s]);return Object.assign(n,{cluster:!0,cluster_id:e[t+L],point_count:o,point_count_abbreviated:r})}function Z(e){return e/360+.5}function G(e){const t=Math.sin(e*Math.PI/180),i=.5-.25*Math.log((1+t)/(1-t))/Math.PI;return i<0?0:i>1?1:i}function J(e){const t=(180-360*e)*Math.PI/180;return 360*Math.atan(Math.exp(t))/Math.PI-90}function Y(e,t,i,o){for(var r,s=o,n=i-t>>1,a=i-t,l=e[t],c=e[t+1],h=e[i],u=e[i+1],d=t+3;ds)r=d,s=p;else if(p===s){var f=Math.abs(d-n);fo&&(r-t>3&&Y(e,t,r,o),e[r+2]=s,i-r>3&&Y(e,r,i,o));}function A(e,t,i,o,r,s){var n=r-i,a=s-o;if(0!==n||0!==a){var l=((e-i)*n+(t-o)*a)/(n*n+a*a);l>1?(i=r,o=s):l>0&&(i+=n*l,o+=a*l);}return (n=e-i)*n+(a=t-o)*a}function V(e,t,i,o){var r={id:void 0===e?null:e,type:t,geometry:i,tags:o,minX:1/0,minY:1/0,maxX:-1/0,maxY:-1/0};return function(e){var t=e.geometry,i=e.type;if("Point"===i||"MultiPoint"===i||"LineString"===i)B(e,t);else if("Polygon"===i||"MultiLineString"===i)for(var o=0;o0&&(n+=o?(r*c-l*s)/2:Math.sqrt(Math.pow(l-r,2)+Math.pow(c-s,2))),r=l,s=c;}var h=t.length-3;t[2]=1,Y(t,0,h,i),t[h+2]=1,t.size=Math.abs(n),t.start=0,t.end=t.size;}function W(e,t,i,o){for(var r=0;r1?1:i}function H(e,t,i,o,r,s,n,a){if(o/=t,s>=(i/=t)&&n=o)return null;for(var l=[],c=0;c=i&&f=o)){var g=[];if("Point"===d||"MultiPoint"===d)K(u,g,i,o,r);else if("LineString"===d)Q(u,g,i,o,r,!1,a.lineMetrics);else if("MultiLineString"===d)te(u,g,i,o,r,!1);else if("Polygon"===d)te(u,g,i,o,r,!0);else if("MultiPolygon"===d)for(var m=0;m=i&&n<=o&&(t.push(e[s]),t.push(e[s+1]),t.push(e[s+2]));}}function Q(e,t,i,o,r,s,n){for(var a,l,c=ee(e),h=0===r?oe:re,u=e.start,d=0;di&&(l=h(c,p,f,m,y,i),n&&(c.start=u+a*l)):v>o?w=i&&(l=h(c,p,f,m,y,i),x=!0),w>o&&v<=o&&(l=h(c,p,f,m,y,o),x=!0),!s&&x&&(n&&(c.end=u+a*l),t.push(c),c=ee(e)),n&&(u+=a);}var b=e.length-3;p=e[b],f=e[b+1],g=e[b+2],(v=0===r?p:f)>=i&&v<=o&&ie(c,p,f,g),b=c.length-3,s&&b>=3&&(c[b]!==c[0]||c[b+1]!==c[1])&&ie(c,c[0],c[1],c[2]),c.length&&t.push(c);}function ee(e){var t=[];return t.size=e.size,t.start=e.start,t.end=e.end,t}function te(e,t,i,o,r,s){for(var n=0;nn.maxX&&(n.maxX=h),u>n.maxY&&(n.maxY=u);}return n}function he(e,t,i,o){var r=t.geometry,s=t.type,n=[];if("Point"===s||"MultiPoint"===s)for(var a=0;a0&&t.size<(r?n:o))i.numPoints+=t.length/3;else {for(var a=[],l=0;ln)&&(i.numSimplified++,a.push(t[l]),a.push(t[l+1])),i.numPoints++;r&&function(e,t){for(var i=0,o=0,r=e.length,s=r-2;o0===t)for(o=0,r=e.length;o24)throw new Error("maxZoom should be in the 0-24 range");if(t.promoteId&&t.generateId)throw new Error("promoteId and generateId cannot be used together.");var o=function(e,t){var i=[];if("FeatureCollection"===e.type)for(var o=0;o1&&console.time("creation"),d=this.tiles[u]=ce(e,t,i,o,l),this.tileCoords.push({z:t,x:i,y:o}),c)){c>1&&(console.log("tile z%d-%d-%d (features: %d, points: %d, simplified: %d)",t,i,o,d.numFeatures,d.numPoints,d.numSimplified),console.timeEnd("creation"));var p="z"+t;this.stats[p]=(this.stats[p]||0)+1,this.total++;}if(d.source=e,r){if(t===l.maxZoom||t===r)continue;var f=1<1&&console.time("clipping");var g,m,y,v,w,x,b=.5*l.buffer/l.extent,S=.5-b,M=.5+b,I=1+b;g=m=y=v=null,w=H(e,h,i-b,i+M,0,d.minX,d.maxX,l),x=H(e,h,i+S,i+I,0,d.minX,d.maxX,l),e=null,w&&(g=H(w,h,o-b,o+M,1,d.minY,d.maxY,l),m=H(w,h,o+S,o+I,1,d.minY,d.maxY,l),w=null),x&&(y=H(x,h,o-b,o+M,1,d.minY,d.maxY,l),v=H(x,h,o+S,o+I,1,d.minY,d.maxY,l),x=null),c>1&&console.timeEnd("clipping"),a.push(g||[],t+1,2*i,2*o),a.push(m||[],t+1,2*i,2*o+1),a.push(y||[],t+1,2*i+1,2*o),a.push(v||[],t+1,2*i+1,2*o+1);}}},de.prototype.getTile=function(e,t,i){var o=this.options,r=o.extent,s=o.debug;if(e<0||e>24)return null;var n=1<1&&console.log("drilling down to z%d-%d-%d",e,t,i);for(var l,c=e,h=t,u=i;!l&&c>0;)c--,h=Math.floor(h/2),u=Math.floor(u/2),l=this.tiles[pe(c,h,u)];return l&&l.source?(s>1&&console.log("found parent tile z%d-%d-%d",c,h,u),s>1&&console.time("drilling down"),this.splitTile(l.source,c,h,u,e,t,i),s>1&&console.timeEnd("drilling down"),this.tiles[a]?ae(this.tiles[a],r):null):null};class ye extends n{constructor(t,i,o,r){super(t,i,o),this._dataUpdateable=new Map,this.loadGeoJSON=(t,i)=>{const{promoteId:o}=t;if(t.request)return e.f(t.request,((e,t,r,s)=>{this._dataUpdateable=ge(t,o)?me(t,o):void 0,i(e,t,r,s);}));if("string"==typeof t.data)try{const e=JSON.parse(t.data);this._dataUpdateable=ge(e,o)?me(e,o):void 0,i(null,e);}catch(e){i(new Error(`Input data given to '${t.source}' is not a valid GeoJSON object.`));}else t.dataDiff?this._dataUpdateable?(function(e,t,i){var o,r,s,n;if(t.removeAll&&e.clear(),t.remove)for(const i of t.remove)e.delete(i);if(t.add)for(const o of t.add){const t=fe(o,i);null!=t&&e.set(t,o);}if(t.update)for(const i of t.update){let t=e.get(i.id);if(null==t)continue;const a=!i.removeAllProperties&&((null===(o=i.removeProperties)||void 0===o?void 0:o.length)>0||(null===(r=i.addOrUpdateProperties)||void 0===r?void 0:r.length)>0);if((i.newGeometry||i.removeAllProperties||a)&&(t=Object.assign({},t),e.set(i.id,t),a&&(t.properties=Object.assign({},t.properties))),i.newGeometry&&(t.geometry=i.newGeometry),i.removeAllProperties)t.properties={};else if((null===(s=i.removeProperties)||void 0===s?void 0:s.length)>0)for(const e of i.removeProperties)Object.prototype.hasOwnProperty.call(t.properties,e)&&delete t.properties[e];if((null===(n=i.addOrUpdateProperties)||void 0===n?void 0:n.length)>0)for(const{key:e,value:o}of i.addOrUpdateProperties)t.properties[e]=o;}}(this._dataUpdateable,t.dataDiff,o),i(null,{type:"FeatureCollection",features:Array.from(this._dataUpdateable.values())})):i(new Error(`Cannot update existing geojson data in ${t.source}`)):i(new Error(`Input data given to '${t.source}' is not a valid GeoJSON object.`));return {cancel:()=>{}}},this.loadVectorData=this.loadGeoJSONTile,r&&(this.loadGeoJSON=r);}loadGeoJSONTile(t,i){const o=t.tileID.canonical;if(!this._geoJSONIndex)return i(null,null);const r=this._geoJSONIndex.getTile(o.z,o.x,o.y);if(!r)return i(null,null);const s=new class{constructor(t){this.layers={_geojsonTileLayer:this},this.name="_geojsonTileLayer",this.extent=e.N,this.length=t.length,this._features=t;}feature(t){return new class{constructor(t){this._feature=t,this.extent=e.N,this.type=t.type,this.properties=t.tags,"id"in t&&!isNaN(t.id)&&(this.id=parseInt(t.id,10));}loadGeometry(){if(1===this._feature.type){const t=[];for(const i of this._feature.geometry)t.push([new e.P(i[0],i[1])]);return t}{const t=[];for(const i of this._feature.geometry){const o=[];for(const t of i)o.push(new e.P(t[0],t[1]));t.push(o);}return t}}toGeoJSON(e,t,i){return u.call(this,e,t,i)}}(this._features[t])}}(r.features);let n=k(s);0===n.byteOffset&&n.byteLength===n.buffer.byteLength||(n=new Uint8Array(n)),i(null,{vectorTile:s,rawData:n.buffer});}loadData(t,i){var o;null===(o=this._pendingRequest)||void 0===o||o.cancel(),this._pendingCallback&&this._pendingCallback(null,{abandoned:!0});const r=!!(t&&t.request&&t.request.collectResourceTiming)&&new e.bu(t.request);this._pendingCallback=i,this._pendingRequest=this.loadGeoJSON(t,((o,s)=>{if(delete this._pendingCallback,delete this._pendingRequest,o||!s)return i(o);if("object"!=typeof s)return i(new Error(`Input data given to '${t.source}' is not a valid GeoJSON object.`));{h(s,!0);try{if(t.filter){const i=e.bC(t.filter,{type:"boolean","property-type":"data-driven",overridable:!1,transition:!1});if("error"===i.result)throw new Error(i.value.map((e=>`${e.key}: ${e.message}`)).join(", "));const o=s.features.filter((e=>i.value.evaluate({zoom:0},e)));s={type:"FeatureCollection",features:o};}this._geoJSONIndex=t.cluster?new N(function({superclusterOptions:t,clusterProperties:i}){if(!i||!t)return t;const o={},r={},s={accumulated:null,zoom:0},n={properties:null},a=Object.keys(i);for(const t of a){const[s,n]=i[t],a=e.bC(n),l=e.bC("string"==typeof s?[s,["accumulated"],["get",t]]:s);o[t]=a.value,r[t]=l.value;}return t.map=e=>{n.properties=e;const t={};for(const e of a)t[e]=o[e].evaluate(s,n);return t},t.reduce=(e,t)=>{n.properties=t;for(const t of a)s.accumulated=e[t],e[t]=r[t].evaluate(s,n);},t}(t)).load(s.features):function(e,t){return new de(e,t)}(s,t.geojsonVtOptions);}catch(o){return i(o)}this.loaded={};const n={};if(r){const e=r.finish();e&&(n.resourceTiming={},n.resourceTiming[t.source]=JSON.parse(JSON.stringify(e)));}i(null,n);}}));}reloadTile(e,t){const i=this.loaded;return i&&i[e.uid]?super.reloadTile(e,t):this.loadTile(e,t)}removeSource(e,t){this._pendingCallback&&this._pendingCallback(null,{abandoned:!0}),t();}getClusterExpansionZoom(e,t){try{t(null,this._geoJSONIndex.getClusterExpansionZoom(e.clusterId));}catch(e){t(e);}}getClusterChildren(e,t){try{t(null,this._geoJSONIndex.getChildren(e.clusterId));}catch(e){t(e);}}getClusterLeaves(e,t){try{t(null,this._geoJSONIndex.getLeaves(e.clusterId,e.limit,e.offset));}catch(e){t(e);}}}class ve{constructor(t){this.self=t,this.actor=new e.C(t,this),this.layerIndexes={},this.availableImages={},this.workerSourceTypes={vector:n,geojson:ye},this.workerSources={},this.demWorkerSources={},this.self.registerWorkerSource=(e,t)=>{if(this.workerSourceTypes[e])throw new Error(`Worker source with name "${e}" already registered.`);this.workerSourceTypes[e]=t;},this.self.registerRTLTextPlugin=t=>{if(e.bD.isParsed())throw new Error("RTL text plugin already registered.");e.bD.applyArabicShaping=t.applyArabicShaping,e.bD.processBidirectionalText=t.processBidirectionalText,e.bD.processStyledBidirectionalText=t.processStyledBidirectionalText;};}setReferrer(e,t){this.referrer=t;}setImages(e,t,i){this.availableImages[e]=t;for(const i in this.workerSources[e]){const o=this.workerSources[e][i];for(const e in o)o[e].availableImages=t;}i();}setLayers(e,t,i){this.getLayerIndex(e).replace(t),i();}updateLayers(e,t,i){this.getLayerIndex(e).update(t.layers,t.removedIds),i();}loadTile(e,t,i){this.getWorkerSource(e,t.type,t.source).loadTile(t,i);}loadDEMTile(e,t,i){this.getDEMWorkerSource(e,t.source).loadTile(t,i);}reloadTile(e,t,i){this.getWorkerSource(e,t.type,t.source).reloadTile(t,i);}abortTile(e,t,i){this.getWorkerSource(e,t.type,t.source).abortTile(t,i);}removeTile(e,t,i){this.getWorkerSource(e,t.type,t.source).removeTile(t,i);}removeDEMTile(e,t){this.getDEMWorkerSource(e,t.source).removeTile(t);}removeSource(e,t,i){if(!this.workerSources[e]||!this.workerSources[e][t.type]||!this.workerSources[e][t.type][t.source])return;const o=this.workerSources[e][t.type][t.source];delete this.workerSources[e][t.type][t.source],void 0!==o.removeSource?o.removeSource(t,i):i();}loadWorkerSource(e,t,i){try{this.self.importScripts(t.url),i();}catch(e){i(e.toString());}}syncRTLPluginState(t,i,o){try{e.bD.setState(i);const t=e.bD.getPluginURL();if(e.bD.isLoaded()&&!e.bD.isParsed()&&null!=t){this.self.importScripts(t);const i=e.bD.isParsed();o(i?void 0:new Error(`RTL Text Plugin failed to import scripts from ${t}`),i);}}catch(e){o(e.toString());}}getAvailableImages(e){let t=this.availableImages[e];return t||(t=[]),t}getLayerIndex(e){let i=this.layerIndexes[e];return i||(i=this.layerIndexes[e]=new t),i}getWorkerSource(e,t,i){return this.workerSources[e]||(this.workerSources[e]={}),this.workerSources[e][t]||(this.workerSources[e][t]={}),this.workerSources[e][t][i]||(this.workerSources[e][t][i]=new this.workerSourceTypes[t]({send:(t,i,o)=>{this.actor.send(t,i,o,e);}},this.getLayerIndex(e),this.getAvailableImages(e))),this.workerSources[e][t][i]}getDEMWorkerSource(e,t){return this.demWorkerSources[e]||(this.demWorkerSources[e]={}),this.demWorkerSources[e][t]||(this.demWorkerSources[e][t]=new a),this.demWorkerSources[e][t]}}return e.i()&&(self.worker=new ve(self)),ve})); + +define(["./shared"],(function(t){"use strict";var e="3.6.2";class i{static testProp(t){if(!i.docStyle)return t[0];for(let e=0;e{window.removeEventListener("click",i.suppressClickInternal,!0);}),0);}static mousePos(e,i){const s=e.getBoundingClientRect();return new t.P(i.clientX-s.left-e.clientLeft,i.clientY-s.top-e.clientTop)}static touchPos(e,i){const s=e.getBoundingClientRect(),a=[];for(let o=0;o{i=[],a=0,o=0,r={};},e.addThrottleControl=t=>{const e=o++;return r[e]=t,e},e.removeThrottleControl=t=>{delete r[t],h();},e.getImage=(t,e,o=!0)=>{s.supported&&(t.headers||(t.headers={}),t.headers.accept="image/webp,*/*");const r={requestParameters:t,supportImageRefresh:o,callback:e,cancelled:!1,completed:!1,cancel:()=>{r.completed||r.cancelled||(r.cancelled=!0,r.innerRequest&&(r.innerRequest.cancel(),a--),h());}};return i.push(r),h(),r};const n=e=>{const{requestParameters:i,supportImageRefresh:s,callback:a}=e;return t.e(i,{type:"image"}),(!1!==s||t.i()||t.g(i.url)||i.headers&&!Object.keys(i.headers).reduce(((t,e)=>t&&"accept"===e),!0)?t.m:c)(i,((t,i,s,o)=>{l(e,a,t,i,s,o);}))},l=(e,i,s,o,r,n)=>{s?i(s):o instanceof HTMLImageElement||t.a(o)?i(null,o):o&&((e,i)=>{"function"==typeof createImageBitmap?t.b(e,i):t.d(e,i);})(o,((t,e)=>{null!=t?i(t):null!=e&&i(null,e,{cacheControl:r,expires:n});})),e.cancelled||(e.completed=!0,a--,h());},h=()=>{const e=(()=>{const t=Object.keys(r);let e=!1;if(t.length>0)for(const i of t)if(e=r[i](),e)break;return e})()?t.c.MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME:t.c.MAX_PARALLEL_IMAGE_REQUESTS;for(let t=a;t0;t++){const e=i.shift();if(e.cancelled){t--;continue}const s=n(e);a++,e.innerRequest=s;}},c=(e,i)=>{const s=new Image,a=e.url;let o=!1;const r=e.credentials;return r&&"include"===r?s.crossOrigin="use-credentials":(r&&"same-origin"===r||!t.s(a))&&(s.crossOrigin="anonymous"),s.fetchPriority="high",s.onload=()=>{i(null,s),s.onerror=s.onload=null;},s.onerror=()=>{o||i(new Error("Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.")),s.onerror=s.onload=null;},s.src=a,{cancel:()=>{o=!0,s.src="";}}};}(h||(h={})),h.resetRequestQueue(),function(t){t.Glyphs="Glyphs",t.Image="Image",t.Source="Source",t.SpriteImage="SpriteImage",t.SpriteJSON="SpriteJSON",t.Style="Style",t.Tile="Tile",t.Unknown="Unknown";}(c||(c={}));class u{constructor(t){this._transformRequestFn=t;}transformRequest(t,e){return this._transformRequestFn&&this._transformRequestFn(t,e)||{url:t}}normalizeSpriteURL(t,e,i){const s=function(t){const e=t.match(d);if(!e)throw new Error(`Unable to parse URL "${t}"`);return {protocol:e[1],authority:e[2],path:e[3]||"/",params:e[4]?e[4].split("&"):[]}}(t);return s.path+=`${e}${i}`,function(t){const e=t.params.length?`?${t.params.join("&")}`:"";return `${t.protocol}://${t.authority}${t.path}${e}`}(s)}setTransformRequest(t){this._transformRequestFn=t;}}const d=/^(\w+):\/\/([^/?]*)(\/[^?]+)?\??(.+)?/;function _(e){var i=new t.A(3);return i[0]=e[0],i[1]=e[1],i[2]=e[2],i}var p,m=function(t,e,i){return t[0]=e[0]-i[0],t[1]=e[1]-i[1],t[2]=e[2]-i[2],t};p=new t.A(3),t.A!=Float32Array&&(p[0]=0,p[1]=0,p[2]=0);var f=function(t){var e=t[0],i=t[1];return e*e+i*i};function g(t){const e=[];if("string"==typeof t)e.push({id:"default",url:t});else if(t&&t.length>0){const i=[];for(const{id:s,url:a}of t){const t=`${s}${a}`;-1===i.indexOf(t)&&(i.push(t),e.push({id:s,url:a}));}}return e}function v(e,i,s,a,o){if(a)return void e(a);if(o!==Object.values(i).length||o!==Object.values(s).length)return;const r={};for(const e in i){r[e]={};const a=t.h.getImageCanvasContext(s[e]),o=i[e];for(const t in o){const{width:i,height:s,x:n,y:l,sdf:h,pixelRatio:c,stretchX:u,stretchY:d,content:_}=o[t];r[e][t]={data:null,pixelRatio:c,sdf:h,stretchX:u,stretchY:d,content:_,spriteData:{width:i,height:s,x:n,y:l,context:a}};}}e(null,r);}!function(){var e=new t.A(2);t.A!=Float32Array&&(e[0]=0,e[1]=0);}();class x{constructor(t,e,i,s){this.context=t,this.format=i,this.texture=t.gl.createTexture(),this.update(e,s);}update(e,i,s){const{width:a,height:o}=e,r=!(this.size&&this.size[0]===a&&this.size[1]===o||s),{context:n}=this,{gl:l}=n;if(this.useMipmap=Boolean(i&&i.useMipmap),l.bindTexture(l.TEXTURE_2D,this.texture),n.pixelStoreUnpackFlipY.set(!1),n.pixelStoreUnpack.set(1),n.pixelStoreUnpackPremultiplyAlpha.set(this.format===l.RGBA&&(!i||!1!==i.premultiply)),r)this.size=[a,o],e instanceof HTMLImageElement||e instanceof HTMLCanvasElement||e instanceof HTMLVideoElement||e instanceof ImageData||t.a(e)?l.texImage2D(l.TEXTURE_2D,0,this.format,this.format,l.UNSIGNED_BYTE,e):l.texImage2D(l.TEXTURE_2D,0,this.format,a,o,0,this.format,l.UNSIGNED_BYTE,e.data);else {const{x:i,y:r}=s||{x:0,y:0};e instanceof HTMLImageElement||e instanceof HTMLCanvasElement||e instanceof HTMLVideoElement||e instanceof ImageData||t.a(e)?l.texSubImage2D(l.TEXTURE_2D,0,i,r,l.RGBA,l.UNSIGNED_BYTE,e):l.texSubImage2D(l.TEXTURE_2D,0,i,r,a,o,l.RGBA,l.UNSIGNED_BYTE,e.data);}this.useMipmap&&this.isSizePowerOfTwo()&&l.generateMipmap(l.TEXTURE_2D);}bind(t,e,i){const{context:s}=this,{gl:a}=s;a.bindTexture(a.TEXTURE_2D,this.texture),i!==a.LINEAR_MIPMAP_NEAREST||this.isSizePowerOfTwo()||(i=a.LINEAR),t!==this.filter&&(a.texParameteri(a.TEXTURE_2D,a.TEXTURE_MAG_FILTER,t),a.texParameteri(a.TEXTURE_2D,a.TEXTURE_MIN_FILTER,i||t),this.filter=t),e!==this.wrap&&(a.texParameteri(a.TEXTURE_2D,a.TEXTURE_WRAP_S,e),a.texParameteri(a.TEXTURE_2D,a.TEXTURE_WRAP_T,e),this.wrap=e);}isSizePowerOfTwo(){return this.size[0]===this.size[1]&&Math.log(this.size[0])/Math.LN2%1==0}destroy(){const{gl:t}=this.context;t.deleteTexture(this.texture),this.texture=null;}}function y(t){const{userImage:e}=t;return !!(e&&e.render&&e.render())&&(t.data.replace(new Uint8Array(e.data.buffer)),!0)}class b extends t.E{constructor(){super(),this.images={},this.updatedImages={},this.callbackDispatchedThisFrame={},this.loaded=!1,this.requestors=[],this.patterns={},this.atlasImage=new t.R({width:1,height:1}),this.dirty=!0;}isLoaded(){return this.loaded}setLoaded(t){if(this.loaded!==t&&(this.loaded=t,t)){for(const{ids:t,callback:e}of this.requestors)this._notify(t,e);this.requestors=[];}}getImage(e){const i=this.images[e];if(i&&!i.data&&i.spriteData){const e=i.spriteData;i.data=new t.R({width:e.width,height:e.height},e.context.getImageData(e.x,e.y,e.width,e.height).data),i.spriteData=null;}return i}addImage(t,e){if(this.images[t])throw new Error(`Image id ${t} already exist, use updateImage instead`);this._validate(t,e)&&(this.images[t]=e);}_validate(e,i){let s=!0;const a=i.data||i.spriteData;return this._validateStretch(i.stretchX,a&&a.width)||(this.fire(new t.j(new Error(`Image "${e}" has invalid "stretchX" value`))),s=!1),this._validateStretch(i.stretchY,a&&a.height)||(this.fire(new t.j(new Error(`Image "${e}" has invalid "stretchY" value`))),s=!1),this._validateContent(i.content,i)||(this.fire(new t.j(new Error(`Image "${e}" has invalid "content" value`))),s=!1),s}_validateStretch(t,e){if(!t)return !0;let i=0;for(const s of t){if(s[0]-1);l++,o[l]=n,r[l]=h,r[l+1]=w;}for(let n=0,l=0;n{let s=this.entries[t];s||(s=this.entries[t]={glyphs:{},requests:{},ranges:{}});let a=s.glyphs[e];if(void 0!==a)return void i(null,{stack:t,id:e,glyph:a});if(a=this._tinySDF(s,t,e),a)return s.glyphs[e]=a,void i(null,{stack:t,id:e,glyph:a});const o=Math.floor(e/256);if(256*o>65535)return void i(new Error("glyphs > 65535 not supported"));if(s.ranges[o])return void i(null,{stack:t,id:e,glyph:a});if(!this.url)return void i(new Error("glyphsUrl is not set"));let r=s.requests[o];r||(r=s.requests[o]=[],E.loadGlyphRange(t,o,this.url,this.requestManager,((t,e)=>{if(e){for(const t in e)this._doesCharSupportLocalGlyph(+t)||(s.glyphs[+t]=e[+t]);s.ranges[o]=!0;}for(const i of r)i(t,e);delete s.requests[o];}))),r.push(((s,a)=>{s?i(s):a&&i(null,{stack:t,id:e,glyph:a[e]||null});}));}),((t,e)=>{if(t)i(t);else if(e){const t={};for(const{stack:i,id:s,glyph:a}of e)(t[i]||(t[i]={}))[s]=a&&{id:a.id,bitmap:a.bitmap.clone(),metrics:a.metrics};i(null,t);}}));}_doesCharSupportLocalGlyph(e){return !!this.localIdeographFontFamily&&(t.u["CJK Unified Ideographs"](e)||t.u["Hangul Syllables"](e)||t.u.Hiragana(e)||t.u.Katakana(e))}_tinySDF(e,i,s){const a=this.localIdeographFontFamily;if(!a)return;if(!this._doesCharSupportLocalGlyph(s))return;let o=e.tinySDF;if(!o){let t="400";/bold/i.test(i)?t="900":/medium/i.test(i)?t="500":/light/i.test(i)&&(t="200"),o=e.tinySDF=new E.TinySDF({fontSize:48,buffer:6,radius:16,cutoff:.25,fontFamily:a,fontWeight:t});}const r=o.draw(String.fromCharCode(s));return {id:s,bitmap:new t.q({width:r.width||60,height:r.height||60},r.data),metrics:{width:r.glyphWidth/2||24,height:r.glyphHeight/2||24,left:r.glyphLeft/2+.5||0,top:r.glyphTop/2-27.5||-8,advance:r.glyphAdvance/2||24,isDoubleResolution:!0}}}}E.loadGlyphRange=function(e,i,s,a,o){const r=256*i,n=r+255,l=a.transformRequest(s.replace("{fontstack}",e).replace("{range}",`${r}-${n}`),c.Glyphs);t.l(l,((e,i)=>{if(e)o(e);else if(i){const e={};for(const s of t.n(i))e[s.id]=s;o(null,e);}}));},E.TinySDF=class{constructor({fontSize:t=24,buffer:e=3,radius:i=8,cutoff:s=.25,fontFamily:a="sans-serif",fontWeight:o="normal",fontStyle:r="normal"}={}){this.buffer=e,this.cutoff=s,this.radius=i;const n=this.size=t+4*e,l=this._createCanvas(n),h=this.ctx=l.getContext("2d",{willReadFrequently:!0});h.font=`${r} ${o} ${t}px ${a}`,h.textBaseline="alphabetic",h.textAlign="left",h.fillStyle="black",this.gridOuter=new Float64Array(n*n),this.gridInner=new Float64Array(n*n),this.f=new Float64Array(n),this.z=new Float64Array(n+1),this.v=new Uint16Array(n);}_createCanvas(t){const e=document.createElement("canvas");return e.width=e.height=t,e}draw(t){const{width:e,actualBoundingBoxAscent:i,actualBoundingBoxDescent:s,actualBoundingBoxLeft:a,actualBoundingBoxRight:o}=this.ctx.measureText(t),r=Math.ceil(i),n=Math.max(0,Math.min(this.size-this.buffer,Math.ceil(o-a))),l=Math.min(this.size-this.buffer,r+Math.ceil(s)),h=n+2*this.buffer,c=l+2*this.buffer,u=Math.max(h*c,0),d=new Uint8ClampedArray(u),_={data:d,width:h,height:c,glyphWidth:n,glyphHeight:l,glyphTop:r,glyphLeft:0,glyphAdvance:e};if(0===n||0===l)return _;const{ctx:p,buffer:m,gridInner:f,gridOuter:g}=this;p.clearRect(m,m,n,l),p.fillText(t,m,m+r);const v=p.getImageData(m,m,n,l);g.fill(w,0,u),f.fill(0,0,u);for(let t=0;t0?t*t:0,f[s]=t<0?t*t:0;}}T(g,0,0,h,c,h,this.f,this.v,this.z),T(f,m,m,n,l,h,this.f,this.v,this.z);for(let t=0;t1&&(r=t[++o]);const l=Math.abs(n-r.left),h=Math.abs(n-r.right),c=Math.min(l,h);let u;const d=e/i*(s+1);if(r.isDash){const t=s-Math.abs(d);u=Math.sqrt(c*c+t*t);}else u=s-Math.sqrt(c*c+d*d);this.data[a+n]=Math.max(0,Math.min(255,u+128));}}}addRegularDash(t){for(let e=t.length-1;e>=0;--e){const i=t[e],s=t[e+1];i.zeroLength?t.splice(e,1):s&&s.isDash===i.isDash&&(s.left=i.left,t.splice(e,1));}const e=t[0],i=t[t.length-1];e.isDash===i.isDash&&(e.left=i.left-this.width,i.right=e.right+this.width);const s=this.width*this.nextRow;let a=0,o=t[a];for(let e=0;e1&&(o=t[++a]);const i=Math.abs(e-o.left),r=Math.abs(e-o.right),n=Math.min(i,r);this.data[s+e]=Math.max(0,Math.min(255,(o.isDash?n:-n)+128));}}addDash(e,i){const s=i?7:0,a=2*s+1;if(this.nextRow+a>this.height)return t.w("LineAtlas out of space"),null;let o=0;for(let t=0;t{t.send(e,i,s);}),s=s||function(){});}getActor(){return this.currentActor=(this.currentActor+1)%this.actors.length,this.actors[this.currentActor]}remove(t=!0){this.actors.forEach((t=>{t.remove();})),this.actors=[],t&&this.workerPool.release(this.id);}}function z(e,i,s){const a=function(i,a){if(i)return s(i);if(a){const i=t.F(t.e(a,e),["tiles","minzoom","maxzoom","attribution","bounds","scheme","tileSize","encoding"]);a.vector_layers&&(i.vectorLayers=a.vector_layers,i.vectorLayerIds=i.vectorLayers.map((t=>t.id))),s(null,i);}};return e.url?t.f(i.transformRequest(e.url,c.Source),a):t.h.frame((()=>a(null,e)))}class L{constructor(t,e){t&&(e?this.setSouthWest(t).setNorthEast(e):Array.isArray(t)&&(4===t.length?this.setSouthWest([t[0],t[1]]).setNorthEast([t[2],t[3]]):this.setSouthWest(t[0]).setNorthEast(t[1])));}setNorthEast(e){return this._ne=e instanceof t.L?new t.L(e.lng,e.lat):t.L.convert(e),this}setSouthWest(e){return this._sw=e instanceof t.L?new t.L(e.lng,e.lat):t.L.convert(e),this}extend(e){const i=this._sw,s=this._ne;let a,o;if(e instanceof t.L)a=e,o=e;else {if(!(e instanceof L))return Array.isArray(e)?4===e.length||e.every(Array.isArray)?this.extend(L.convert(e)):this.extend(t.L.convert(e)):e&&("lng"in e||"lon"in e)&&"lat"in e?this.extend(t.L.convert(e)):this;if(a=e._sw,o=e._ne,!a||!o)return this}return i||s?(i.lng=Math.min(a.lng,i.lng),i.lat=Math.min(a.lat,i.lat),s.lng=Math.max(o.lng,s.lng),s.lat=Math.max(o.lat,s.lat)):(this._sw=new t.L(a.lng,a.lat),this._ne=new t.L(o.lng,o.lat)),this}getCenter(){return new t.L((this._sw.lng+this._ne.lng)/2,(this._sw.lat+this._ne.lat)/2)}getSouthWest(){return this._sw}getNorthEast(){return this._ne}getNorthWest(){return new t.L(this.getWest(),this.getNorth())}getSouthEast(){return new t.L(this.getEast(),this.getSouth())}getWest(){return this._sw.lng}getSouth(){return this._sw.lat}getEast(){return this._ne.lng}getNorth(){return this._ne.lat}toArray(){return [this._sw.toArray(),this._ne.toArray()]}toString(){return `LngLatBounds(${this._sw.toString()}, ${this._ne.toString()})`}isEmpty(){return !(this._sw&&this._ne)}contains(e){const{lng:i,lat:s}=t.L.convert(e);let a=this._sw.lng<=i&&i<=this._ne.lng;return this._sw.lng>this._ne.lng&&(a=this._sw.lng>=i&&i>=this._ne.lng),this._sw.lat<=s&&s<=this._ne.lat&&a}static convert(t){return t instanceof L?t:t?new L(t):t}static fromLngLat(e,i=0){const s=360*i/40075017,a=s/Math.cos(Math.PI/180*e.lat);return new L(new t.L(e.lng-a,e.lat-s),new t.L(e.lng+a,e.lat+s))}}class A{constructor(t,e,i){this.bounds=L.convert(this.validateBounds(t)),this.minzoom=e||0,this.maxzoom=i||24;}validateBounds(t){return Array.isArray(t)&&4===t.length?[Math.max(-180,t[0]),Math.max(-90,t[1]),Math.min(180,t[2]),Math.min(90,t[3])]:[-180,-90,180,90]}contains(e){const i=Math.pow(2,e.z),s=Math.floor(t.G(this.bounds.getWest())*i),a=Math.floor(t.H(this.bounds.getNorth())*i),o=Math.ceil(t.G(this.bounds.getEast())*i),r=Math.ceil(t.H(this.bounds.getSouth())*i);return e.x>=s&&e.x=a&&e.y{this._loaded=!1,this.fire(new t.k("dataloading",{dataType:"source"})),this._tileJSONRequest=z(this._options,this.map._requestManager,((e,i)=>{this._tileJSONRequest=null,this._loaded=!0,this.map.style.sourceCaches[this.id].clearTiles(),e?this.fire(new t.j(e)):i&&(t.e(this,i),i.bounds&&(this.tileBounds=new A(i.bounds,this.minzoom,this.maxzoom)),this.fire(new t.k("data",{dataType:"source",sourceDataType:"metadata"})),this.fire(new t.k("data",{dataType:"source",sourceDataType:"content"})));}));},this.serialize=()=>t.e({},this._options),this.id=e,this.dispatcher=s,this.type="vector",this.minzoom=0,this.maxzoom=22,this.scheme="xyz",this.tileSize=512,this.reparseOverscaled=!0,this.isTileClipped=!0,this._loaded=!1,t.e(this,t.F(i,["url","scheme","tileSize","promoteId"])),this._options=t.e({type:"vector"},i),this._collectResourceTiming=i.collectResourceTiming,512!==this.tileSize)throw new Error("vector tile sources must have a tileSize of 512");this.setEventedParent(a);}loaded(){return this._loaded}hasTile(t){return !this.tileBounds||this.tileBounds.contains(t.canonical)}onAdd(t){this.map=t,this.load();}setSourceProperty(t){this._tileJSONRequest&&this._tileJSONRequest.cancel(),t(),this.load();}setTiles(t){return this.setSourceProperty((()=>{this._options.tiles=t;})),this}setUrl(t){return this.setSourceProperty((()=>{this.url=t,this._options.url=t;})),this}onRemove(){this._tileJSONRequest&&(this._tileJSONRequest.cancel(),this._tileJSONRequest=null);}loadTile(t,e){const i=t.tileID.canonical.url(this.tiles,this.map.getPixelRatio(),this.scheme),s={request:this.map._requestManager.transformRequest(i,c.Tile),uid:t.uid,tileID:t.tileID,zoom:t.tileID.overscaledZ,tileSize:this.tileSize*t.tileID.overscaleFactor(),type:this.type,source:this.id,pixelRatio:this.map.getPixelRatio(),showCollisionBoxes:this.map.showCollisionBoxes,promoteId:this.promoteId};function a(i,s){return delete t.request,t.aborted?e(null):i&&404!==i.status?e(i):(s&&s.resourceTiming&&(t.resourceTiming=s.resourceTiming),this.map._refreshExpiredTiles&&s&&t.setExpiryData(s),t.loadVectorData(s,this.map.painter),e(null),void(t.reloadCallback&&(this.loadTile(t,t.reloadCallback),t.reloadCallback=null)))}s.request.collectResourceTiming=this._collectResourceTiming,t.actor&&"expired"!==t.state?"loading"===t.state?t.reloadCallback=e:t.request=t.actor.send("reloadTile",s,a.bind(this)):(t.actor=this.dispatcher.getActor(),t.request=t.actor.send("loadTile",s,a.bind(this)));}abortTile(t){t.request&&(t.request.cancel(),delete t.request),t.actor&&t.actor.send("abortTile",{uid:t.uid,type:this.type,source:this.id},void 0);}unloadTile(t){t.unloadVectorData(),t.actor&&t.actor.send("removeTile",{uid:t.uid,type:this.type,source:this.id},void 0);}hasTransition(){return !1}}class k extends t.E{constructor(e,i,s,a){super(),this.id=e,this.dispatcher=s,this.setEventedParent(a),this.type="raster",this.minzoom=0,this.maxzoom=22,this.roundZoom=!0,this.scheme="xyz",this.tileSize=512,this._loaded=!1,this._options=t.e({type:"raster"},i),t.e(this,t.F(i,["url","scheme","tileSize"]));}load(){this._loaded=!1,this.fire(new t.k("dataloading",{dataType:"source"})),this._tileJSONRequest=z(this._options,this.map._requestManager,((e,i)=>{this._tileJSONRequest=null,this._loaded=!0,e?this.fire(new t.j(e)):i&&(t.e(this,i),i.bounds&&(this.tileBounds=new A(i.bounds,this.minzoom,this.maxzoom)),this.fire(new t.k("data",{dataType:"source",sourceDataType:"metadata"})),this.fire(new t.k("data",{dataType:"source",sourceDataType:"content"})));}));}loaded(){return this._loaded}onAdd(t){this.map=t,this.load();}onRemove(){this._tileJSONRequest&&(this._tileJSONRequest.cancel(),this._tileJSONRequest=null);}setSourceProperty(t){this._tileJSONRequest&&this._tileJSONRequest.cancel(),t(),this.load();}setTiles(t){return this.setSourceProperty((()=>{this._options.tiles=t;})),this}serialize(){return t.e({},this._options)}hasTile(t){return !this.tileBounds||this.tileBounds.contains(t.canonical)}loadTile(t,e){const i=t.tileID.canonical.url(this.tiles,this.map.getPixelRatio(),this.scheme);t.request=h.getImage(this.map._requestManager.transformRequest(i,c.Tile),((i,s,a)=>{if(delete t.request,t.aborted)t.state="unloaded",e(null);else if(i)t.state="errored",e(i);else if(s){this.map._refreshExpiredTiles&&a&&t.setExpiryData(a);const i=this.map.painter.context,o=i.gl;t.texture=this.map.painter.getTileTexture(s.width),t.texture?t.texture.update(s,{useMipmap:!0}):(t.texture=new x(i,s,o.RGBA,{useMipmap:!0}),t.texture.bind(o.LINEAR,o.CLAMP_TO_EDGE,o.LINEAR_MIPMAP_NEAREST),i.extTextureFilterAnisotropic&&o.texParameterf(o.TEXTURE_2D,i.extTextureFilterAnisotropic.TEXTURE_MAX_ANISOTROPY_EXT,i.extTextureFilterAnisotropicMax)),t.state="loaded",e(null);}}),this.map._refreshExpiredTiles);}abortTile(t,e){t.request&&(t.request.cancel(),delete t.request),e();}unloadTile(t,e){t.texture&&this.map.painter.saveTileTexture(t.texture),e();}hasTransition(){return !1}}class F extends k{constructor(e,i,s,a){super(e,i,s,a),this.type="raster-dem",this.maxzoom=22,this._options=t.e({type:"raster-dem"},i),this.encoding=i.encoding||"mapbox",this.redFactor=i.redFactor,this.greenFactor=i.greenFactor,this.blueFactor=i.blueFactor,this.baseShift=i.baseShift;}loadTile(e,i){const s=e.tileID.canonical.url(this.tiles,this.map.getPixelRatio(),this.scheme),a=this.map._requestManager.transformRequest(s,c.Tile);function o(t,s){t&&(e.state="errored",i(t)),s&&(e.dem=s,e.needsHillshadePrepare=!0,e.needsTerrainPrepare=!0,e.state="loaded",i(null));}e.neighboringTiles=this._getNeighboringTiles(e.tileID),e.request=h.getImage(a,((s,a,r)=>t._(this,void 0,void 0,(function*(){if(delete e.request,e.aborted)e.state="unloaded",i(null);else if(s)e.state="errored",i(s);else if(a){this.map._refreshExpiredTiles&&e.setExpiryData(r);const i=t.a(a)&&t.J()?a:yield function(e){return t._(this,void 0,void 0,(function*(){if("undefined"!=typeof VideoFrame&&t.K()){const i=e.width+2,s=e.height+2;try{return new t.R({width:i,height:s},yield t.M(e,-1,-1,i,s))}catch(t){}}return t.h.getImageData(e,1)}))}(a),s={uid:e.uid,coord:e.tileID,source:this.id,rawImageData:i,encoding:this.encoding,redFactor:this.redFactor,greenFactor:this.greenFactor,blueFactor:this.blueFactor,baseShift:this.baseShift};e.actor&&"expired"!==e.state||(e.actor=this.dispatcher.getActor(),e.actor.send("loadDEMTile",s,o));}}))),this.map._refreshExpiredTiles);}_getNeighboringTiles(e){const i=e.canonical,s=Math.pow(2,i.z),a=(i.x-1+s)%s,o=0===i.x?e.wrap-1:e.wrap,r=(i.x+1+s)%s,n=i.x+1===s?e.wrap+1:e.wrap,l={};return l[new t.O(e.overscaledZ,o,i.z,a,i.y).key]={backfilled:!1},l[new t.O(e.overscaledZ,n,i.z,r,i.y).key]={backfilled:!1},i.y>0&&(l[new t.O(e.overscaledZ,o,i.z,a,i.y-1).key]={backfilled:!1},l[new t.O(e.overscaledZ,e.wrap,i.z,i.x,i.y-1).key]={backfilled:!1},l[new t.O(e.overscaledZ,n,i.z,r,i.y-1).key]={backfilled:!1}),i.y+1{this._updateWorkerData();},this.serialize=()=>t.e({},this._options,{type:this.type,data:this._data}),this.id=e,this.type="geojson",this.minzoom=0,this.maxzoom=18,this.tileSize=512,this.isTileClipped=!0,this.reparseOverscaled=!0,this._removed=!1,this._pendingLoads=0,this.actor=s.getActor(),this.setEventedParent(a),this._data=i.data,this._options=t.e({},i),this._collectResourceTiming=i.collectResourceTiming,void 0!==i.maxzoom&&(this.maxzoom=i.maxzoom),i.type&&(this.type=i.type),i.attribution&&(this.attribution=i.attribution),this.promoteId=i.promoteId;const o=t.N/this.tileSize;this.workerOptions=t.e({source:this.id,cluster:i.cluster||!1,geojsonVtOptions:{buffer:(void 0!==i.buffer?i.buffer:128)*o,tolerance:(void 0!==i.tolerance?i.tolerance:.375)*o,extent:t.N,maxZoom:this.maxzoom,lineMetrics:i.lineMetrics||!1,generateId:i.generateId||!1},superclusterOptions:{maxZoom:void 0!==i.clusterMaxZoom?i.clusterMaxZoom:this.maxzoom-1,minPoints:Math.max(2,i.clusterMinPoints||2),extent:t.N,radius:(i.clusterRadius||50)*o,log:!1,generateId:i.generateId||!1},clusterProperties:i.clusterProperties,filter:i.filter},i.workerOptions),"string"==typeof this.promoteId&&(this.workerOptions.promoteId=this.promoteId);}onAdd(t){this.map=t,this.load();}setData(t){return this._data=t,this._updateWorkerData(),this}updateData(t){return this._updateWorkerData(t),this}setClusterOptions(t){return this.workerOptions.cluster=t.cluster,t&&(void 0!==t.clusterRadius&&(this.workerOptions.superclusterOptions.radius=t.clusterRadius),void 0!==t.clusterMaxZoom&&(this.workerOptions.superclusterOptions.maxZoom=t.clusterMaxZoom)),this._updateWorkerData(),this}getClusterExpansionZoom(t,e){return this.actor.send("geojson.getClusterExpansionZoom",{clusterId:t,source:this.id},e),this}getClusterChildren(t,e){return this.actor.send("geojson.getClusterChildren",{clusterId:t,source:this.id},e),this}getClusterLeaves(t,e,i,s){return this.actor.send("geojson.getClusterLeaves",{source:this.id,clusterId:t,limit:e,offset:i},s),this}_updateWorkerData(e){const i=t.e({},this.workerOptions);e?i.dataDiff=e:"string"==typeof this._data?(i.request=this.map._requestManager.transformRequest(t.h.resolveURL(this._data),c.Source),i.request.collectResourceTiming=this._collectResourceTiming):i.data=JSON.stringify(this._data),this._pendingLoads++,this.fire(new t.k("dataloading",{dataType:"source"})),this.actor.send(`${this.type}.loadData`,i,((e,i)=>{if(this._pendingLoads--,this._removed||i&&i.abandoned)return void this.fire(new t.k("dataabort",{dataType:"source"}));let s=null;if(i&&i.resourceTiming&&i.resourceTiming[this.id]&&(s=i.resourceTiming[this.id].slice(0)),e)return void this.fire(new t.j(e));const a={dataType:"source"};this._collectResourceTiming&&s&&s.length>0&&t.e(a,{resourceTiming:s}),this.fire(new t.k("data",Object.assign(Object.assign({},a),{sourceDataType:"metadata"}))),this.fire(new t.k("data",Object.assign(Object.assign({},a),{sourceDataType:"content"})));}));}loaded(){return 0===this._pendingLoads}loadTile(t,e){const i=t.actor?"reloadTile":"loadTile";t.actor=this.actor;const s={type:this.type,uid:t.uid,tileID:t.tileID,zoom:t.tileID.overscaledZ,maxZoom:this.maxzoom,tileSize:this.tileSize,source:this.id,pixelRatio:this.map.getPixelRatio(),showCollisionBoxes:this.map.showCollisionBoxes,promoteId:this.promoteId};t.request=this.actor.send(i,s,((s,a)=>(delete t.request,t.unloadVectorData(),t.aborted?e(null):s?e(s):(t.loadVectorData(a,this.map.painter,"reloadTile"===i),e(null)))));}abortTile(t){t.request&&(t.request.cancel(),delete t.request),t.aborted=!0;}unloadTile(t){t.unloadVectorData(),this.actor.send("removeTile",{uid:t.uid,type:this.type,source:this.id});}onRemove(){this._removed=!0,this.actor.send("removeSource",{type:this.type,source:this.id});}hasTransition(){return !1}}var O=t.Q([{name:"a_pos",type:"Int16",components:2},{name:"a_texture_pos",type:"Int16",components:2}]);class N extends t.E{constructor(e,i,s,a){super(),this.load=(e,i)=>{this._loaded=!1,this.fire(new t.k("dataloading",{dataType:"source"})),this.url=this.options.url,this._request=h.getImage(this.map._requestManager.transformRequest(this.url,c.Image),((s,a)=>{this._request=null,this._loaded=!0,s?this.fire(new t.j(s)):a&&(this.image=a,e&&(this.coordinates=e),i&&i(),this._finishLoading());}));},this.prepare=()=>{if(0===Object.keys(this.tiles).length||!this.image)return;const e=this.map.painter.context,i=e.gl;this.boundsBuffer||(this.boundsBuffer=e.createVertexBuffer(this._boundsArray,O.members)),this.boundsSegments||(this.boundsSegments=t.S.simpleSegment(0,0,4,2)),this.texture||(this.texture=new x(e,this.image,i.RGBA),this.texture.bind(i.LINEAR,i.CLAMP_TO_EDGE));let s=!1;for(const t in this.tiles){const e=this.tiles[t];"loaded"!==e.state&&(e.state="loaded",e.texture=this.texture,s=!0);}s&&this.fire(new t.k("data",{dataType:"source",sourceDataType:"idle",sourceId:this.id}));},this.serialize=()=>({type:"image",url:this.options.url,coordinates:this.coordinates}),this.id=e,this.dispatcher=s,this.coordinates=i.coordinates,this.type="image",this.minzoom=0,this.maxzoom=22,this.tileSize=512,this.tiles={},this._loaded=!1,this.setEventedParent(a),this.options=i;}loaded(){return this._loaded}updateImage(t){return t.url?(this._request&&(this._request.cancel(),this._request=null),this.options.url=t.url,this.load(t.coordinates,(()=>{this.texture=null;})),this):this}_finishLoading(){this.map&&(this.setCoordinates(this.coordinates),this.fire(new t.k("data",{dataType:"source",sourceDataType:"metadata"})));}onAdd(t){this.map=t,this.load();}onRemove(){this._request&&(this._request.cancel(),this._request=null);}setCoordinates(e){this.coordinates=e;const i=e.map(t.U.fromLngLat);this.tileID=function(e){let i=1/0,s=1/0,a=-1/0,o=-1/0;for(const t of e)i=Math.min(i,t.x),s=Math.min(s,t.y),a=Math.max(a,t.x),o=Math.max(o,t.y);const r=Math.max(a-i,o-s),n=Math.max(0,Math.floor(-Math.log(r)/Math.LN2)),l=Math.pow(2,n);return new t.W(n,Math.floor((i+a)/2*l),Math.floor((s+o)/2*l))}(i),this.minzoom=this.maxzoom=this.tileID.z;const s=i.map((t=>this.tileID.getTilePoint(t)._round()));return this._boundsArray=new t.V,this._boundsArray.emplaceBack(s[0].x,s[0].y,0,0),this._boundsArray.emplaceBack(s[1].x,s[1].y,t.N,0),this._boundsArray.emplaceBack(s[3].x,s[3].y,0,t.N),this._boundsArray.emplaceBack(s[2].x,s[2].y,t.N,t.N),this.boundsBuffer&&(this.boundsBuffer.destroy(),delete this.boundsBuffer),this.fire(new t.k("data",{dataType:"source",sourceDataType:"content"})),this}loadTile(t,e){this.tileID&&this.tileID.equals(t.tileID.canonical)?(this.tiles[String(t.tileID.wrap)]=t,t.buckets={},e(null)):(t.state="errored",e(null));}hasTransition(){return !1}}class U extends N{constructor(e,i,s,a){super(e,i,s,a),this.load=()=>{this._loaded=!1;const e=this.options;this.urls=[];for(const t of e.urls)this.urls.push(this.map._requestManager.transformRequest(t,c.Source).url);t.X(this.urls,((e,i)=>{this._loaded=!0,e?this.fire(new t.j(e)):i&&(this.video=i,this.video.loop=!0,this.video.addEventListener("playing",(()=>{this.map.triggerRepaint();})),this.map&&this.video.play(),this._finishLoading());}));},this.prepare=()=>{if(0===Object.keys(this.tiles).length||this.video.readyState<2)return;const e=this.map.painter.context,i=e.gl;this.boundsBuffer||(this.boundsBuffer=e.createVertexBuffer(this._boundsArray,O.members)),this.boundsSegments||(this.boundsSegments=t.S.simpleSegment(0,0,4,2)),this.texture?this.video.paused||(this.texture.bind(i.LINEAR,i.CLAMP_TO_EDGE),i.texSubImage2D(i.TEXTURE_2D,0,0,0,i.RGBA,i.UNSIGNED_BYTE,this.video)):(this.texture=new x(e,this.video,i.RGBA),this.texture.bind(i.LINEAR,i.CLAMP_TO_EDGE));let s=!1;for(const t in this.tiles){const e=this.tiles[t];"loaded"!==e.state&&(e.state="loaded",e.texture=this.texture,s=!0);}s&&this.fire(new t.k("data",{dataType:"source",sourceDataType:"idle",sourceId:this.id}));},this.serialize=()=>({type:"video",urls:this.urls,coordinates:this.coordinates}),this.roundZoom=!0,this.type="video",this.options=i;}pause(){this.video&&this.video.pause();}play(){this.video&&this.video.play();}seek(e){if(this.video){const i=this.video.seekable;ei.end(0)?this.fire(new t.j(new t.Y(`sources.${this.id}`,null,`Playback for this video can be set only between the ${i.start(0)} and ${i.end(0)}-second mark.`))):this.video.currentTime=e;}}getVideo(){return this.video}onAdd(t){this.map||(this.map=t,this.load(),this.video&&(this.video.play(),this.setCoordinates(this.coordinates)));}hasTransition(){return this.video&&!this.video.paused}}class Z extends N{constructor(e,i,s,a){super(e,i,s,a),this.load=()=>{this._loaded=!0,this.canvas||(this.canvas=this.options.canvas instanceof HTMLCanvasElement?this.options.canvas:document.getElementById(this.options.canvas)),this.width=this.canvas.width,this.height=this.canvas.height,this._hasInvalidDimensions()?this.fire(new t.j(new Error("Canvas dimensions cannot be less than or equal to zero."))):(this.play=function(){this._playing=!0,this.map.triggerRepaint();},this.pause=function(){this._playing&&(this.prepare(),this._playing=!1);},this._finishLoading());},this.prepare=()=>{let e=!1;if(this.canvas.width!==this.width&&(this.width=this.canvas.width,e=!0),this.canvas.height!==this.height&&(this.height=this.canvas.height,e=!0),this._hasInvalidDimensions())return;if(0===Object.keys(this.tiles).length)return;const i=this.map.painter.context,s=i.gl;this.boundsBuffer||(this.boundsBuffer=i.createVertexBuffer(this._boundsArray,O.members)),this.boundsSegments||(this.boundsSegments=t.S.simpleSegment(0,0,4,2)),this.texture?(e||this._playing)&&this.texture.update(this.canvas,{premultiply:!0}):this.texture=new x(i,this.canvas,s.RGBA,{premultiply:!0});let a=!1;for(const t in this.tiles){const e=this.tiles[t];"loaded"!==e.state&&(e.state="loaded",e.texture=this.texture,a=!0);}a&&this.fire(new t.k("data",{dataType:"source",sourceDataType:"idle",sourceId:this.id}));},this.serialize=()=>({type:"canvas",coordinates:this.coordinates}),i.coordinates?Array.isArray(i.coordinates)&&4===i.coordinates.length&&!i.coordinates.some((t=>!Array.isArray(t)||2!==t.length||t.some((t=>"number"!=typeof t))))||this.fire(new t.j(new t.Y(`sources.${e}`,null,'"coordinates" property must be an array of 4 longitude/latitude array pairs'))):this.fire(new t.j(new t.Y(`sources.${e}`,null,'missing required property "coordinates"'))),i.animate&&"boolean"!=typeof i.animate&&this.fire(new t.j(new t.Y(`sources.${e}`,null,'optional "animate" property must be a boolean value'))),i.canvas?"string"==typeof i.canvas||i.canvas instanceof HTMLCanvasElement||this.fire(new t.j(new t.Y(`sources.${e}`,null,'"canvas" must be either a string representing the ID of the canvas element from which to read, or an HTMLCanvasElement instance'))):this.fire(new t.j(new t.Y(`sources.${e}`,null,'missing required property "canvas"'))),this.options=i,this.animate=void 0===i.animate||i.animate;}getCanvas(){return this.canvas}onAdd(t){this.map=t,this.load(),this.canvas&&this.animate&&this.play();}onRemove(){this.pause();}hasTransition(){return this._playing}_hasInvalidDimensions(){for(const t of [this.canvas.width,this.canvas.height])if(isNaN(t)||t<=0)return !0;return !1}}const G={},j=t=>{switch(t){case"geojson":return B;case"image":return N;case"raster":return k;case"raster-dem":return F;case"vector":return R;case"video":return U;case"canvas":return Z}return G[t]};function V(e,i){const s=t.Z();return t.$(s,s,[1,1,0]),t.a0(s,s,[.5*e.width,.5*e.height,1]),t.a1(s,s,e.calculatePosMatrix(i.toUnwrapped()))}function q(t,e,i,s,a,o){const r=function(t,e,i){if(t)for(const s of t){const t=e[s];if(t&&t.source===i&&"fill-extrusion"===t.type)return !0}else for(const t in e){const s=e[t];if(s.source===i&&"fill-extrusion"===s.type)return !0}return !1}(a&&a.layers,e,t.id),n=o.maxPitchScaleFactor(),l=t.tilesIn(s,n,r);l.sort($);const h=[];for(const s of l)h.push({wrappedTileID:s.tileID.wrapped().key,queryResults:s.tile.queryRenderedFeatures(e,i,t._state,s.queryGeometry,s.cameraQueryGeometry,s.scale,a,o,n,V(t.transform,s.tileID))});const c=function(t){const e={},i={};for(const s of t){const t=s.queryResults,a=s.wrappedTileID,o=i[a]=i[a]||{};for(const i in t){const s=t[i],a=o[i]=o[i]||{},r=e[i]=e[i]||[];for(const t of s)a[t.featureIndex]||(a[t.featureIndex]=!0,r.push(t));}}return e}(h);for(const e in c)c[e].forEach((e=>{const i=e.feature,s=t.getFeatureState(i.layer["source-layer"],i.id);i.source=i.layer.source,i.layer["source-layer"]&&(i.sourceLayer=i.layer["source-layer"]),i.state=s;}));return c}function $(t,e){const i=t.tileID,s=e.tileID;return i.overscaledZ-s.overscaledZ||i.canonical.y-s.canonical.y||i.wrap-s.wrap||i.canonical.x-s.canonical.x}class W{constructor(e,i){this.timeAdded=0,this.fadeEndTime=0,this.tileID=e,this.uid=t.a2(),this.uses=0,this.tileSize=i,this.buckets={},this.expirationTime=null,this.queryPadding=0,this.hasSymbolBuckets=!1,this.hasRTLText=!1,this.dependencies={},this.rtt=[],this.rttCoords={},this.expiredRequestCount=0,this.state="loading";}registerFadeDuration(t){const e=t+this.timeAdded;ee.getLayer(t))).filter(Boolean);if(0!==t.length){s.layers=t,s.stateDependentLayerIds&&(s.stateDependentLayers=s.stateDependentLayerIds.map((e=>t.filter((t=>t.id===e))[0])));for(const e of t)i[e.id]=s;}}return i}(e.buckets,i.style),this.hasSymbolBuckets=!1;for(const e in this.buckets){const i=this.buckets[e];if(i instanceof t.a4){if(this.hasSymbolBuckets=!0,!s)break;i.justReloaded=!0;}}if(this.hasRTLText=!1,this.hasSymbolBuckets)for(const e in this.buckets){const i=this.buckets[e];if(i instanceof t.a4&&i.hasRTLText){this.hasRTLText=!0,t.a5();break}}this.queryPadding=0;for(const t in this.buckets){const e=this.buckets[t];this.queryPadding=Math.max(this.queryPadding,i.style.getLayer(t).queryRadius(e));}e.imageAtlas&&(this.imageAtlas=e.imageAtlas),e.glyphAtlasImage&&(this.glyphAtlasImage=e.glyphAtlasImage);}else this.collisionBoxArray=new t.a3;}unloadVectorData(){for(const t in this.buckets)this.buckets[t].destroy();this.buckets={},this.imageAtlasTexture&&this.imageAtlasTexture.destroy(),this.imageAtlas&&(this.imageAtlas=null),this.glyphAtlasTexture&&this.glyphAtlasTexture.destroy(),this.latestFeatureIndex=null,this.state="unloaded";}getBucket(t){return this.buckets[t.id]}upload(t){for(const e in this.buckets){const i=this.buckets[e];i.uploadPending()&&i.upload(t);}const e=t.gl;this.imageAtlas&&!this.imageAtlas.uploaded&&(this.imageAtlasTexture=new x(t,this.imageAtlas.image,e.RGBA),this.imageAtlas.uploaded=!0),this.glyphAtlasImage&&(this.glyphAtlasTexture=new x(t,this.glyphAtlasImage,e.ALPHA),this.glyphAtlasImage=null);}prepare(t){this.imageAtlas&&this.imageAtlas.patchUpdatedImages(t,this.imageAtlasTexture);}queryRenderedFeatures(t,e,i,s,a,o,r,n,l,h){return this.latestFeatureIndex&&this.latestFeatureIndex.rawTileData?this.latestFeatureIndex.query({queryGeometry:s,cameraQueryGeometry:a,scale:o,tileSize:this.tileSize,pixelPosMatrix:h,transform:n,params:r,queryPadding:this.queryPadding*l},t,e,i):{}}querySourceFeatures(e,i){const s=this.latestFeatureIndex;if(!s||!s.rawTileData)return;const a=s.loadVTLayers(),o=i&&i.sourceLayer?i.sourceLayer:"",r=a._geojsonTileLayer||a[o];if(!r)return;const n=t.a6(i&&i.filter),{z:l,x:h,y:c}=this.tileID.canonical,u={z:l,x:h,y:c};for(let i=0;it)e=!1;else if(i)if(this.expirationTime{this.remove(t,a);}),i)),this.data[s].push(a),this.order.push(s),this.order.length>this.max){const t=this._getAndRemoveByKey(this.order[0]);t&&this.onRemove(t);}return this}has(t){return t.wrapped().key in this.data}getAndRemove(t){return this.has(t)?this._getAndRemoveByKey(t.wrapped().key):null}_getAndRemoveByKey(t){const e=this.data[t].shift();return e.timeout&&clearTimeout(e.timeout),0===this.data[t].length&&delete this.data[t],this.order.splice(this.order.indexOf(t),1),e.value}getByKey(t){const e=this.data[t];return e?e[0].value:null}get(t){return this.has(t)?this.data[t.wrapped().key][0].value:null}remove(t,e){if(!this.has(t))return this;const i=t.wrapped().key,s=void 0===e?0:this.data[i].indexOf(e),a=this.data[i][s];return this.data[i].splice(s,1),a.timeout&&clearTimeout(a.timeout),0===this.data[i].length&&delete this.data[i],this.onRemove(a.value),this.order.splice(this.order.indexOf(i),1),this}setMaxSize(t){for(this.max=t;this.order.length>this.max;){const t=this._getAndRemoveByKey(this.order[0]);t&&this.onRemove(t);}return this}filter(t){const e=[];for(const i in this.data)for(const s of this.data[i])t(s.value)||e.push(s);for(const t of e)this.remove(t.value.tileID,t);}}class X{constructor(){this.state={},this.stateChanges={},this.deletedStates={};}updateState(e,i,s){const a=String(i);if(this.stateChanges[e]=this.stateChanges[e]||{},this.stateChanges[e][a]=this.stateChanges[e][a]||{},t.e(this.stateChanges[e][a],s),null===this.deletedStates[e]){this.deletedStates[e]={};for(const t in this.state[e])t!==a&&(this.deletedStates[e][t]=null);}else if(this.deletedStates[e]&&null===this.deletedStates[e][a]){this.deletedStates[e][a]={};for(const t in this.state[e][a])s[t]||(this.deletedStates[e][a][t]=null);}else for(const t in s)this.deletedStates[e]&&this.deletedStates[e][a]&&null===this.deletedStates[e][a][t]&&delete this.deletedStates[e][a][t];}removeFeatureState(t,e,i){if(null===this.deletedStates[t])return;const s=String(e);if(this.deletedStates[t]=this.deletedStates[t]||{},i&&void 0!==e)null!==this.deletedStates[t][s]&&(this.deletedStates[t][s]=this.deletedStates[t][s]||{},this.deletedStates[t][s][i]=null);else if(void 0!==e)if(this.stateChanges[t]&&this.stateChanges[t][s])for(i in this.deletedStates[t][s]={},this.stateChanges[t][s])this.deletedStates[t][s][i]=null;else this.deletedStates[t][s]=null;else this.deletedStates[t]=null;}getState(e,i){const s=String(i),a=t.e({},(this.state[e]||{})[s],(this.stateChanges[e]||{})[s]);if(null===this.deletedStates[e])return {};if(this.deletedStates[e]){const t=this.deletedStates[e][i];if(null===t)return {};for(const e in t)delete a[e];}return a}initializeTileState(t,e){t.setFeatureState(this.state,e);}coalesceChanges(e,i){const s={};for(const e in this.stateChanges){this.state[e]=this.state[e]||{};const i={};for(const s in this.stateChanges[e])this.state[e][s]||(this.state[e][s]={}),t.e(this.state[e][s],this.stateChanges[e][s]),i[s]=this.state[e][s];s[e]=i;}for(const e in this.deletedStates){this.state[e]=this.state[e]||{};const i={};if(null===this.deletedStates[e])for(const t in this.state[e])i[t]={},this.state[e][t]={};else for(const t in this.deletedStates[e]){if(null===this.deletedStates[e][t])this.state[e][t]={};else for(const i of Object.keys(this.deletedStates[e][t]))delete this.state[e][t][i];i[t]=this.state[e][t];}s[e]=s[e]||{},t.e(s[e],i);}if(this.stateChanges={},this.deletedStates={},0!==Object.keys(s).length)for(const t in e)e[t].setFeatureState(s,i);}}class K extends t.E{constructor(t,e,i){super(),this.id=t,this.dispatcher=i,this.on("data",(t=>{"source"===t.dataType&&"metadata"===t.sourceDataType&&(this._sourceLoaded=!0),this._sourceLoaded&&!this._paused&&"source"===t.dataType&&"content"===t.sourceDataType&&(this.reload(),this.transform&&this.update(this.transform,this.terrain),this._didEmitContent=!0);})),this.on("dataloading",(()=>{this._sourceErrored=!1;})),this.on("error",(()=>{this._sourceErrored=this._source.loaded();})),this._source=((t,e,i,s)=>{const a=new(j(e.type))(t,e,i,s);if(a.id!==t)throw new Error(`Expected Source id to be ${t} instead of ${a.id}`);return a})(t,e,i,this),this._tiles={},this._cache=new H(0,this._unloadTile.bind(this)),this._timers={},this._cacheTimers={},this._maxTileCacheSize=null,this._maxTileCacheZoomLevels=null,this._loadedParentTiles={},this._coveredTiles={},this._state=new X,this._didEmitContent=!1,this._updated=!1;}onAdd(t){this.map=t,this._maxTileCacheSize=t?t._maxTileCacheSize:null,this._maxTileCacheZoomLevels=t?t._maxTileCacheZoomLevels:null,this._source&&this._source.onAdd&&this._source.onAdd(t);}onRemove(t){this.clearTiles(),this._source&&this._source.onRemove&&this._source.onRemove(t);}loaded(){if(this._sourceErrored)return !0;if(!this._sourceLoaded)return !1;if(!this._source.loaded())return !1;if(!(void 0===this.used&&void 0===this.usedForTerrain||this.used||this.usedForTerrain))return !0;if(!this._updated)return !1;for(const t in this._tiles){const e=this._tiles[t];if("loaded"!==e.state&&"errored"!==e.state)return !1}return !0}getSource(){return this._source}pause(){this._paused=!0;}resume(){if(!this._paused)return;const t=this._shouldReloadOnResume;this._paused=!1,this._shouldReloadOnResume=!1,t&&this.reload(),this.transform&&this.update(this.transform,this.terrain);}_loadTile(t,e){return this._source.loadTile(t,e)}_unloadTile(t){if(this._source.unloadTile)return this._source.unloadTile(t,(()=>{}))}_abortTile(e){this._source.abortTile&&this._source.abortTile(e,(()=>{})),this._source.fire(new t.k("dataabort",{tile:e,coord:e.tileID,dataType:"source"}));}serialize(){return this._source.serialize()}prepare(t){this._source.prepare&&this._source.prepare(),this._state.coalesceChanges(this._tiles,this.map?this.map.painter:null);for(const e in this._tiles){const i=this._tiles[e];i.upload(t),i.prepare(this.map.style.imageManager);}}getIds(){return Object.values(this._tiles).map((t=>t.tileID)).sort(Q).map((t=>t.key))}getRenderableIds(e){const i=[];for(const t in this._tiles)this._isIdRenderable(t,e)&&i.push(this._tiles[t]);return e?i.sort(((e,i)=>{const s=e.tileID,a=i.tileID,o=new t.P(s.canonical.x,s.canonical.y)._rotate(this.transform.angle),r=new t.P(a.canonical.x,a.canonical.y)._rotate(this.transform.angle);return s.overscaledZ-a.overscaledZ||r.y-o.y||r.x-o.x})).map((t=>t.tileID.key)):i.map((t=>t.tileID)).sort(Q).map((t=>t.key))}hasRenderableParent(t){const e=this.findLoadedParent(t,0);return !!e&&this._isIdRenderable(e.tileID.key)}_isIdRenderable(t,e){return this._tiles[t]&&this._tiles[t].hasData()&&!this._coveredTiles[t]&&(e||!this._tiles[t].holdingForFade())}reload(){if(this._paused)this._shouldReloadOnResume=!0;else {this._cache.reset();for(const t in this._tiles)"errored"!==this._tiles[t].state&&this._reloadTile(t,"reloading");}}_reloadTile(t,e){const i=this._tiles[t];i&&("loading"!==i.state&&(i.state=e),this._loadTile(i,this._tileLoaded.bind(this,i,t,e)));}_tileLoaded(e,i,s,a){if(a)return e.state="errored",void(404!==a.status?this._source.fire(new t.j(a,{tile:e})):this.update(this.transform,this.terrain));e.timeAdded=t.h.now(),"expired"===s&&(e.refreshedUponExpiration=!0),this._setTileReloadTimer(i,e),"raster-dem"===this.getSource().type&&e.dem&&this._backfillDEM(e),this._state.initializeTileState(e,this.map?this.map.painter:null),e.aborted||this._source.fire(new t.k("data",{dataType:"source",tile:e,coord:e.tileID}));}_backfillDEM(t){const e=this.getRenderableIds();for(let s=0;s1||(Math.abs(i)>1&&(1===Math.abs(i+a)?i+=a:1===Math.abs(i-a)&&(i-=a)),e.dem&&t.dem&&(t.dem.backfillBorder(e.dem,i,s),t.neighboringTiles&&t.neighboringTiles[o]&&(t.neighboringTiles[o].backfilled=!0)));}}getTile(t){return this.getTileByID(t.key)}getTileByID(t){return this._tiles[t]}_retainLoadedChildren(t,e,i,s){for(const a in this._tiles){let o=this._tiles[a];if(s[a]||!o.hasData()||o.tileID.overscaledZ<=e||o.tileID.overscaledZ>i)continue;let r=o.tileID;for(;o&&o.tileID.overscaledZ>e+1;){const t=o.tileID.scaledTo(o.tileID.overscaledZ-1);o=this._tiles[t.key],o&&o.hasData()&&(r=t);}let n=r;for(;n.overscaledZ>e;)if(n=n.scaledTo(n.overscaledZ-1),t[n.key]){s[r.key]=r;break}}}findLoadedParent(t,e){if(t.key in this._loadedParentTiles){const i=this._loadedParentTiles[t.key];return i&&i.tileID.overscaledZ>=e?i:null}for(let i=t.overscaledZ-1;i>=e;i--){const e=t.scaledTo(i),s=this._getLoadedTile(e);if(s)return s}}_getLoadedTile(t){const e=this._tiles[t.key];return e&&e.hasData()?e:this._cache.getByKey(t.wrapped().key)}updateCacheSize(e){const i=Math.ceil(e.width/this._source.tileSize)+1,s=Math.ceil(e.height/this._source.tileSize)+1,a=Math.floor(i*s*(null===this._maxTileCacheZoomLevels?t.c.MAX_TILE_CACHE_ZOOM_LEVELS:this._maxTileCacheZoomLevels)),o="number"==typeof this._maxTileCacheSize?Math.min(this._maxTileCacheSize,a):a;this._cache.setMaxSize(o);}handleWrapJump(t){const e=Math.round((t-(void 0===this._prevLng?t:this._prevLng))/360);if(this._prevLng=t,e){const t={};for(const i in this._tiles){const s=this._tiles[i];s.tileID=s.tileID.unwrapTo(s.tileID.wrap+e),t[s.tileID.key]=s;}this._tiles=t;for(const t in this._timers)clearTimeout(this._timers[t]),delete this._timers[t];for(const t in this._tiles)this._setTileReloadTimer(t,this._tiles[t]);}}update(e,i){if(this.transform=e,this.terrain=i,!this._sourceLoaded||this._paused)return;let s;this.updateCacheSize(e),this.handleWrapJump(this.transform.center.lng),this._coveredTiles={},this.used||this.usedForTerrain?this._source.tileID?s=e.getVisibleUnwrappedCoordinates(this._source.tileID).map((e=>new t.O(e.canonical.z,e.wrap,e.canonical.z,e.canonical.x,e.canonical.y))):(s=e.coveringTiles({tileSize:this.usedForTerrain?this.tileSize:this._source.tileSize,minzoom:this._source.minzoom,maxzoom:this._source.maxzoom,roundZoom:!this.usedForTerrain&&this._source.roundZoom,reparseOverscaled:this._source.reparseOverscaled,terrain:i}),this._source.hasTile&&(s=s.filter((t=>this._source.hasTile(t))))):s=[];const a=e.coveringZoomLevel(this._source),o=Math.max(a-K.maxOverzooming,this._source.minzoom),r=Math.max(a+K.maxUnderzooming,this._source.minzoom);if(this.usedForTerrain){const t={};for(const e of s)if(e.canonical.z>this._source.minzoom){const i=e.scaledTo(e.canonical.z-1);t[i.key]=i;const s=e.scaledTo(Math.max(this._source.minzoom,Math.min(e.canonical.z,5)));t[s.key]=s;}s=s.concat(Object.values(t));}const n=0===s.length&&!this._updated&&this._didEmitContent;this._updated=!0,n&&this.fire(new t.k("data",{sourceDataType:"idle",dataType:"source",sourceId:this.id}));const l=this._updateRetainedTiles(s,a);if(Y(this._source.type)){const e={},n={},h=Object.keys(l),c=t.h.now();for(const t of h){const i=l[t],s=this._tiles[t];if(!s||0!==s.fadeEndTime&&s.fadeEndTime<=c)continue;const a=this.findLoadedParent(i,o);a&&(this._addTile(a.tileID),e[a.tileID.key]=a.tileID),n[t]=i;}this._retainLoadedChildren(n,a,r,l);for(const t in e)l[t]||(this._coveredTiles[t]=!0,l[t]=e[t]);if(i){const t={},e={};for(const i of s)this._tiles[i.key].hasData()?t[i.key]=i:e[i.key]=i;for(const i in e){const s=e[i].children(this._source.maxzoom);this._tiles[s[0].key]&&this._tiles[s[1].key]&&this._tiles[s[2].key]&&this._tiles[s[3].key]&&(t[s[0].key]=l[s[0].key]=s[0],t[s[1].key]=l[s[1].key]=s[1],t[s[2].key]=l[s[2].key]=s[2],t[s[3].key]=l[s[3].key]=s[3],delete e[i]);}for(const i in e){const s=this.findLoadedParent(e[i],this._source.minzoom);if(s){t[s.tileID.key]=l[s.tileID.key]=s.tileID;for(const e in t)t[e].isChildOf(s.tileID)&&delete t[e];}}for(const e in this._tiles)t[e]||(this._coveredTiles[e]=!0);}}for(const t in l)this._tiles[t].clearFadeHold();const h=t.ab(this._tiles,l);for(const t of h){const e=this._tiles[t];e.hasSymbolBuckets&&!e.holdingForFade()?e.setHoldDuration(this.map._fadeDuration):e.hasSymbolBuckets&&!e.symbolFadeFinished()||this._removeTile(t);}this._updateLoadedParentTileCache();}releaseSymbolFadeTiles(){for(const t in this._tiles)this._tiles[t].holdingForFade()&&this._removeTile(t);}_updateRetainedTiles(t,e){const i={},s={},a=Math.max(e-K.maxOverzooming,this._source.minzoom),o=Math.max(e+K.maxUnderzooming,this._source.minzoom),r={};for(const s of t){const t=this._addTile(s);i[s.key]=s,t.hasData()||ethis._source.maxzoom){const t=o.children(this._source.maxzoom)[0],e=this.getTile(t);if(e&&e.hasData()){i[t.key]=t;continue}}else {const t=o.children(this._source.maxzoom);if(i[t[0].key]&&i[t[1].key]&&i[t[2].key]&&i[t[3].key])continue}let r=t.wasRequested();for(let e=o.overscaledZ-1;e>=a;--e){const a=o.scaledTo(e);if(s[a.key])break;if(s[a.key]=!0,t=this.getTile(a),!t&&r&&(t=this._addTile(a)),t){const e=t.hasData();if((r||e)&&(i[a.key]=a),r=t.wasRequested(),e)break}}}return i}_updateLoadedParentTileCache(){this._loadedParentTiles={};for(const t in this._tiles){const e=[];let i,s=this._tiles[t].tileID;for(;s.overscaledZ>0;){if(s.key in this._loadedParentTiles){i=this._loadedParentTiles[s.key];break}e.push(s.key);const t=s.scaledTo(s.overscaledZ-1);if(i=this._getLoadedTile(t),i)break;s=t;}for(const t of e)this._loadedParentTiles[t]=i;}}_addTile(e){let i=this._tiles[e.key];if(i)return i;i=this._cache.getAndRemove(e),i&&(this._setTileReloadTimer(e.key,i),i.tileID=e,this._state.initializeTileState(i,this.map?this.map.painter:null),this._cacheTimers[e.key]&&(clearTimeout(this._cacheTimers[e.key]),delete this._cacheTimers[e.key],this._setTileReloadTimer(e.key,i)));const s=i;return i||(i=new W(e,this._source.tileSize*e.overscaleFactor()),this._loadTile(i,this._tileLoaded.bind(this,i,e.key,i.state))),i.uses++,this._tiles[e.key]=i,s||this._source.fire(new t.k("dataloading",{tile:i,coord:i.tileID,dataType:"source"})),i}_setTileReloadTimer(t,e){t in this._timers&&(clearTimeout(this._timers[t]),delete this._timers[t]);const i=e.getExpiryTimeout();i&&(this._timers[t]=setTimeout((()=>{this._reloadTile(t,"expired"),delete this._timers[t];}),i));}_removeTile(t){const e=this._tiles[t];e&&(e.uses--,delete this._tiles[t],this._timers[t]&&(clearTimeout(this._timers[t]),delete this._timers[t]),e.uses>0||(e.hasData()&&"reloading"!==e.state?this._cache.add(e.tileID,e,e.getExpiryTimeout()):(e.aborted=!0,this._abortTile(e),this._unloadTile(e))));}clearTiles(){this._shouldReloadOnResume=!1,this._paused=!1;for(const t in this._tiles)this._removeTile(t);this._cache.reset();}tilesIn(e,i,s){const a=[],o=this.transform;if(!o)return a;const r=s?o.getCameraQueryGeometry(e):e,n=e.map((t=>o.pointCoordinate(t,this.terrain))),l=r.map((t=>o.pointCoordinate(t,this.terrain))),h=this.getIds();let c=1/0,u=1/0,d=-1/0,_=-1/0;for(const t of l)c=Math.min(c,t.x),u=Math.min(u,t.y),d=Math.max(d,t.x),_=Math.max(_,t.y);for(let e=0;e=0&&f[1].y+m>=0){const t=n.map((t=>r.getTilePoint(t))),e=l.map((t=>r.getTilePoint(t)));a.push({tile:s,tileID:r,queryGeometry:t,cameraQueryGeometry:e,scale:p});}}return a}getVisibleCoordinates(t){const e=this.getRenderableIds(t).map((t=>this._tiles[t].tileID));for(const t of e)t.posMatrix=this.transform.calculatePosMatrix(t.toUnwrapped());return e}hasTransition(){if(this._source.hasTransition())return !0;if(Y(this._source.type)){const e=t.h.now();for(const t in this._tiles)if(this._tiles[t].fadeEndTime>=e)return !0}return !1}setFeatureState(t,e,i){this._state.updateState(t=t||"_geojsonTileLayer",e,i);}removeFeatureState(t,e,i){this._state.removeFeatureState(t=t||"_geojsonTileLayer",e,i);}getFeatureState(t,e){return this._state.getState(t=t||"_geojsonTileLayer",e)}setDependencies(t,e,i){const s=this._tiles[t];s&&s.setDependencies(e,i);}reloadTilesForDependencies(t,e){for(const i in this._tiles)this._tiles[i].hasDependency(t,e)&&this._reloadTile(i,"reloading");this._cache.filter((i=>!i.hasDependency(t,e)));}}function Q(t,e){const i=Math.abs(2*t.wrap)-+(t.wrap<0),s=Math.abs(2*e.wrap)-+(e.wrap<0);return t.overscaledZ-e.overscaledZ||s-i||e.canonical.y-t.canonical.y||e.canonical.x-t.canonical.x}function Y(t){return "raster"===t||"image"===t||"video"===t}K.maxOverzooming=10,K.maxUnderzooming=3;const J="mapboxgl_preloaded_worker_pool";class tt{constructor(){this.active={};}acquire(e){if(!this.workers)for(this.workers=[];this.workers.length{t.terminate();})),this.workers=null);}isPreloaded(){return !!this.active[J]}numActive(){return Object.keys(this.active).length}}const et=Math.floor(t.h.hardwareConcurrency/2);let it;function st(){return it||(it=new tt),it}tt.workerCount=t.ac(globalThis)?Math.max(Math.min(et,3),1):1;class at{constructor(t,e){this.reset(t,e);}reset(t,e){this.points=t||[],this._distances=[0];for(let t=1;t0?(a-r)/n:0;return this.points[o].mult(1-l).add(this.points[i].mult(l))}}function ot(t,e){let i=!0;return "always"===t||"never"!==t&&"never"!==e||(i=!1),i}class rt{constructor(t,e,i){const s=this.boxCells=[],a=this.circleCells=[];this.xCellCount=Math.ceil(t/i),this.yCellCount=Math.ceil(e/i);for(let t=0;tthis.width||s<0||e>this.height)return [];const n=[];if(t<=0&&e<=0&&this.width<=i&&this.height<=s){if(a)return [{key:null,x1:t,y1:e,x2:i,y2:s}];for(let t=0;t0}hitTestCircle(t,e,i,s,a){const o=t-i,r=t+i,n=e-i,l=e+i;if(r<0||o>this.width||l<0||n>this.height)return !1;const h=[];return this._forEachCell(o,n,r,l,this._queryCellCircle,h,{hitTest:!0,overlapMode:s,circle:{x:t,y:e,radius:i},seenUids:{box:{},circle:{}}},a),h.length>0}_queryCell(t,e,i,s,a,o,r,n){const{seenUids:l,hitTest:h,overlapMode:c}=r,u=this.boxCells[a];if(null!==u){const a=this.bboxes;for(const r of u)if(!l.box[r]){l.box[r]=!0;const u=4*r,d=this.boxKeys[r];if(t<=a[u+2]&&e<=a[u+3]&&i>=a[u+0]&&s>=a[u+1]&&(!n||n(d))&&(!h||!ot(c,d.overlapMode))&&(o.push({key:d,x1:a[u],y1:a[u+1],x2:a[u+2],y2:a[u+3]}),h))return !0}}const d=this.circleCells[a];if(null!==d){const a=this.circles;for(const r of d)if(!l.circle[r]){l.circle[r]=!0;const u=3*r,d=this.circleKeys[r];if(this._circleAndRectCollide(a[u],a[u+1],a[u+2],t,e,i,s)&&(!n||n(d))&&(!h||!ot(c,d.overlapMode))){const t=a[u],e=a[u+1],i=a[u+2];if(o.push({key:d,x1:t-i,y1:e-i,x2:t+i,y2:e+i}),h)return !0}}}return !1}_queryCellCircle(t,e,i,s,a,o,r,n){const{circle:l,seenUids:h,overlapMode:c}=r,u=this.boxCells[a];if(null!==u){const t=this.bboxes;for(const e of u)if(!h.box[e]){h.box[e]=!0;const i=4*e,s=this.boxKeys[e];if(this._circleAndRectCollide(l.x,l.y,l.radius,t[i+0],t[i+1],t[i+2],t[i+3])&&(!n||n(s))&&!ot(c,s.overlapMode))return o.push(!0),!0}}const d=this.circleCells[a];if(null!==d){const t=this.circles;for(const e of d)if(!h.circle[e]){h.circle[e]=!0;const i=3*e,s=this.circleKeys[e];if(this._circlesCollide(t[i],t[i+1],t[i+2],l.x,l.y,l.radius)&&(!n||n(s))&&!ot(c,s.overlapMode))return o.push(!0),!0}}}_forEachCell(t,e,i,s,a,o,r,n){const l=this._convertToXCellCoord(t),h=this._convertToYCellCoord(e),c=this._convertToXCellCoord(i),u=this._convertToYCellCoord(s);for(let d=l;d<=c;d++)for(let l=h;l<=u;l++)if(a.call(this,t,e,i,s,this.xCellCount*l+d,o,r,n))return}_convertToXCellCoord(t){return Math.max(0,Math.min(this.xCellCount-1,Math.floor(t*this.xScale)))}_convertToYCellCoord(t){return Math.max(0,Math.min(this.yCellCount-1,Math.floor(t*this.yScale)))}_circlesCollide(t,e,i,s,a,o){const r=s-t,n=a-e,l=i+o;return l*l>r*r+n*n}_circleAndRectCollide(t,e,i,s,a,o,r){const n=(o-s)/2,l=Math.abs(t-(s+n));if(l>n+i)return !1;const h=(r-a)/2,c=Math.abs(e-(a+h));if(c>h+i)return !1;if(l<=n||c<=h)return !0;const u=l-n,d=c-h;return u*u+d*d<=i*i}}function nt(e,i,s,a,o){const r=t.Z();return i?(t.a0(r,r,[1/o,1/o,1]),s||t.ae(r,r,a.angle)):t.a1(r,a.labelPlaneMatrix,e),r}function lt(e,i,s,a,o){if(i){const i=t.af(e);return t.a0(i,i,[o,o,1]),s||t.ae(i,i,-a.angle),i}return a.glCoordMatrix}function ht(e,i,s){let a;s?(a=[e.x,e.y,s(e.x,e.y),1],t.ag(a,a,i)):(a=[e.x,e.y,0,1],Tt(a,a,i));const o=a[3];return {point:new t.P(a[0]/o,a[1]/o),signedDistanceFromCamera:o}}function ct(t,e){return .5+t/e*.5}function ut(t,e){const i=t[0]/t[3],s=t[1]/t[3];return i>=-e[0]&&i<=e[0]&&s>=-e[1]&&s<=e[1]}function dt(e,i,s,a,o,r,n,l,h,c){const u=a?e.textSizeData:e.iconSizeData,d=t.ah(u,s.transform.zoom),_=[256/s.width*2+1,256/s.height*2+1],p=a?e.text.dynamicLayoutVertexArray:e.icon.dynamicLayoutVertexArray;p.clear();const m=e.lineVertexArray,f=a?e.text.placedSymbolArray:e.icon.placedSymbolArray,g=s.transform.width/s.transform.height;let v=!1;for(let a=0;aMath.abs(s.x-i.x)*a?{useVertical:!0}:(e===t.ai.vertical?i.ys.x)?{needsFlipping:!0}:null}function mt(e,i,s,a,o,r,n,l,h,c,u,d,_,p,m,f){const g=i/24,v=e.lineOffsetX*g,x=e.lineOffsetY*g;let y;if(e.numGlyphs>1){const t=e.glyphStartIndex+e.numGlyphs,i=e.lineStartIndex,o=e.lineStartIndex+e.lineLength,c=_t(g,l,v,x,s,u,d,e,h,r,_,m,f);if(!c)return {notEnoughRoom:!0};const b=ht(c.first.point,n,f).point,w=ht(c.last.point,n,f).point;if(a&&!s){const t=pt(e.writingMode,b,w,p);if(t)return t}y=[c.first];for(let a=e.glyphStartIndex+1;a0?r.point:ft(d,a,i,1,o,f),l=pt(e.writingMode,i,n,p);if(l)return l}const i=yt(g*l.getoffsetX(e.glyphStartIndex),v,x,s,u,d,e.segment,e.lineStartIndex,e.lineStartIndex+e.lineLength,h,r,_,m,f);if(!i)return {notEnoughRoom:!0};y=[i];}for(const e of y)t.ak(c,e.point,e.angle);return {}}function ft(t,e,i,s,a,o){const r=ht(t.add(t.sub(e)._unit()),a,o).point,n=i.sub(r);return i.add(n._mult(s/n.mag()))}function gt(e,i){const{projectionCache:s,lineVertexArray:a,labelPlaneMatrix:o,tileAnchorPoint:r,distanceFromAnchor:n,getElevation:l,previousVertex:h,direction:c,absOffsetX:u}=i;if(s.projections[e])return s.projections[e];const d=new t.P(a.getx(e),a.gety(e)),_=ht(d,o,l);if(_.signedDistanceFromCamera>0)return s.projections[e]=_.point,_.point;const p=e-c;return ft(0===n?r:new t.P(a.getx(p),a.gety(p)),d,h,u-n+1,o,l)}function vt(t,e,i){return t._unit()._perp()._mult(e*i)}function xt(e,i,s,a,o,r,n,l){const{projectionCache:h,direction:c}=l;if(h.offsets[e])return h.offsets[e];const u=s.add(i);if(e+c=o)return h.offsets[e]=u,u;const d=gt(e+c,l),_=vt(d.sub(s),n,c),p=s.add(_),m=d.add(_);return h.offsets[e]=t.al(r,u,p,m)||u,h.offsets[e]}function yt(t,e,i,s,a,o,r,n,l,h,c,u,d,_){const p=s?t-e:t+e;let m=p>0?1:-1,f=0;s&&(m*=-1,f=Math.PI),m<0&&(f+=Math.PI);let g,v,x=m>0?n+r:n+r+1,y=a,b=a,w=0,T=0;const I=Math.abs(p),E=[];let S;for(;w+T<=I;){if(x+=m,x=l)return null;w+=T,b=y,v=g;const t={projectionCache:u,lineVertexArray:h,labelPlaneMatrix:c,tileAnchorPoint:o,distanceFromAnchor:w,getElevation:_,previousVertex:b,direction:m,absOffsetX:I};if(y=gt(x,t),0===i)E.push(b),S=y.sub(b);else {let e;const s=y.sub(b);e=0===s.mag()?vt(gt(x+m,t).sub(y),i,m):vt(s,i,m),v||(v=b.add(e)),g=xt(x,e,y,n,l,v,i,t),E.push(v),S=g.sub(v);}T=S.mag();}const C=S._mult((I-w)/T)._add(v||b),P=f+Math.atan2(y.y-b.y,y.x-b.x);return E.push(C),{point:C,angle:d?P:0,path:E}}const bt=new Float32Array([-1/0,-1/0,0,-1/0,-1/0,0,-1/0,-1/0,0,-1/0,-1/0,0]);function wt(t,e){for(let i=0;i=1;t--)c.push(r.path[t]);for(let t=1;tht(t,l,p)));c=t.some((t=>t.signedDistanceFromCamera<=0))?[]:t.map((t=>t.point));}let g=[];if(c.length>0){const e=c[0].clone(),i=c[0].clone();for(let t=1;t=s.x&&i.x<=a.x&&e.y>=s.y&&i.y<=a.y?[c]:i.xa.x||i.ya.y?[]:t.am([c],s.x,s.y,a.x,a.y);}for(const t of g){o.reset(t,.25*i);let s=0;s=o.length<=.5*i?1:Math.ceil(o.paddedLength/f)+1;for(let t=0;t=this.screenRightBoundary||sthis.screenBottomBoundary}isInsideGrid(t,e,i,s){return i>=0&&t=0&&et.collisionGroupID===e};}return this.collisionGroups[t]}}function At(e,i,s,a,o){const{horizontalAlign:r,verticalAlign:n}=t.au(e);return new t.P(-(r-.5)*i+a[0]*o,-(n-.5)*s+a[1]*o)}function Rt(e,i,s,a,o,r){const{x1:n,x2:l,y1:h,y2:c,anchorPointX:u,anchorPointY:d}=e,_=new t.P(i,s);return a&&_._rotate(o?r:-r),{x1:n+_.x,y1:h+_.y,x2:l+_.x,y2:c+_.y,anchorPointX:u,anchorPointY:d}}class kt{constructor(t,e,i,s,a){this.transform=t.clone(),this.terrain=e,this.collisionIndex=new Et(this.transform),this.placements={},this.opacities={},this.variableOffsets={},this.stale=!1,this.commitTime=0,this.fadeDuration=i,this.retainedQueryData={},this.collisionGroups=new Lt(s),this.collisionCircleArrays={},this.prevPlacement=a,a&&(a.prevPlacement=void 0),this.placedOrientations={};}getBucketParts(e,i,s,a){const o=s.getBucket(i),r=s.latestFeatureIndex;if(!o||!r||i.id!==o.layerIds[0])return;const n=s.collisionBoxArray,l=o.layers[0].layout,h=Math.pow(2,this.transform.zoom-s.tileID.overscaledZ),c=s.tileSize/t.N,u=this.transform.calculatePosMatrix(s.tileID.toUnwrapped()),d="map"===l.get("text-pitch-alignment"),_="map"===l.get("text-rotation-alignment"),p=St(s,1,this.transform.zoom),m=nt(u,d,_,this.transform,p);let f=null;if(d){const e=lt(u,d,_,this.transform,p);f=t.a1([],this.transform.labelPlaneMatrix,e);}this.retainedQueryData[o.bucketInstanceId]=new zt(o.bucketInstanceId,r,o.sourceLayerIndex,o.index,s.tileID);const g={bucket:o,layout:l,posMatrix:u,textLabelPlaneMatrix:m,labelToScreenMatrix:f,scale:h,textPixelRatio:c,holdingForFade:s.holdingForFade(),collisionBoxArray:n,partiallyEvaluatedTextSize:t.ah(o.textSizeData,this.transform.zoom),collisionGroup:this.collisionGroups.get(o.sourceID)};if(a)for(const t of o.sortKeyRanges){const{sortKey:i,symbolInstanceStart:s,symbolInstanceEnd:a}=t;e.push({sortKey:i,symbolInstanceStart:s,symbolInstanceEnd:a,parameters:g});}else e.push({symbolInstanceStart:0,symbolInstanceEnd:o.symbolInstances.length,parameters:g});}attemptAnchorPlacement(e,i,s,a,o,r,n,l,h,c,u,d,_,p,m,f){const g=t.aq[e.textAnchor],v=[e.textOffset0,e.textOffset1],x=At(g,s,a,v,o),y=this.collisionIndex.placeCollisionBox(Rt(i,x.x,x.y,r,n,this.transform.angle),u,l,h,c.predicate,f);if((!m||0!==this.collisionIndex.placeCollisionBox(Rt(m,x.x,x.y,r,n,this.transform.angle),u,l,h,c.predicate,f).box.length)&&y.box.length>0){let t;if(this.prevPlacement&&this.prevPlacement.variableOffsets[d.crossTileID]&&this.prevPlacement.placements[d.crossTileID]&&this.prevPlacement.placements[d.crossTileID].text&&(t=this.prevPlacement.variableOffsets[d.crossTileID].anchor),0===d.crossTileID)throw new Error("symbolInstance.crossTileID can't be 0");return this.variableOffsets[d.crossTileID]={textOffset:v,width:s,height:a,anchor:g,textBoxScale:o,prevAnchor:t},this.markUsedJustification(_,g,d,p),_.allowVerticalPlacement&&(this.markUsedOrientation(_,p,d),this.placedOrientations[d.crossTileID]=p),{shift:x,placedGlyphBoxes:y}}}placeLayerBucketPart(e,i,s){const{bucket:a,layout:o,posMatrix:r,textLabelPlaneMatrix:n,labelToScreenMatrix:l,textPixelRatio:h,holdingForFade:c,collisionBoxArray:u,partiallyEvaluatedTextSize:d,collisionGroup:_}=e.parameters,p=o.get("text-optional"),m=o.get("icon-optional"),f=t.ar(o,"text-overlap","text-allow-overlap"),g="always"===f,v=t.ar(o,"icon-overlap","icon-allow-overlap"),x="always"===v,y="map"===o.get("text-rotation-alignment"),b="map"===o.get("text-pitch-alignment"),w="none"!==o.get("icon-text-fit"),T="viewport-y"===o.get("symbol-z-order"),I=g&&(x||!a.hasIconData()||m),E=x&&(g||!a.hasTextData()||p);!a.collisionArrays&&u&&a.deserializeCollisionBoxes(u);const S=this.retainedQueryData[a.bucketInstanceId].tileID,C=this.terrain?(t,e)=>this.terrain.getElevation(S,t,e):null,P=(e,u)=>{var x,T;if(i[e.crossTileID])return;if(c)return void(this.placements[e.crossTileID]=new Dt(!1,!1,!1));let S=!1,P=!1,D=!0,M=null,z={box:null,offscreen:null},L={box:null,offscreen:null},A=null,R=null,k=null,F=0,B=0,O=0;u.textFeatureIndex?F=u.textFeatureIndex:e.useRuntimeCollisionCircles&&(F=e.featureIndex),u.verticalTextFeatureIndex&&(B=u.verticalTextFeatureIndex);const N=u.textBox;if(N){const i=i=>{let s=t.ai.horizontal;if(a.allowVerticalPlacement&&!i&&this.prevPlacement){const t=this.prevPlacement.placedOrientations[e.crossTileID];t&&(this.placedOrientations[e.crossTileID]=t,s=t,this.markUsedOrientation(a,s,e));}return s},s=(i,s)=>{if(a.allowVerticalPlacement&&e.numVerticalGlyphVertices>0&&u.verticalTextBox){for(const e of a.writingModes)if(e===t.ai.vertical?(z=s(),L=z):z=i(),z&&z.box&&z.box.length)break}else z=i();},o=e.textAnchorOffsetStartIndex,n=e.textAnchorOffsetEndIndex;if(n===o){const o=(t,i)=>{const s=this.collisionIndex.placeCollisionBox(t,f,h,r,_.predicate,C);return s&&s.box&&s.box.length&&(this.markUsedOrientation(a,i,e),this.placedOrientations[e.crossTileID]=i),s};s((()=>o(N,t.ai.horizontal)),(()=>{const i=u.verticalTextBox;return a.allowVerticalPlacement&&e.numVerticalGlyphVertices>0&&i?o(i,t.ai.vertical):{box:null,offscreen:null}})),i(z&&z.box&&z.box.length);}else {let l=t.aq[null===(T=null===(x=this.prevPlacement)||void 0===x?void 0:x.variableOffsets[e.crossTileID])||void 0===T?void 0:T.anchor];const c=(t,i,s)=>{const c=t.x2-t.x1,u=t.y2-t.y1,d=e.textBoxScale,p=w&&"never"===v?i:null;let m={box:[],offscreen:!1},g="never"===f?1:2,x="never";l&&g++;for(let i=0;ic(N,u.iconBox,t.ai.horizontal)),(()=>{const i=u.verticalTextBox;return a.allowVerticalPlacement&&!(z&&z.box&&z.box.length)&&e.numVerticalGlyphVertices>0&&i?c(i,u.verticalIconBox,t.ai.vertical):{box:null,offscreen:null}})),z&&(S=z.box,D=z.offscreen);const d=i(z&&z.box);if(!S&&this.prevPlacement){const t=this.prevPlacement.variableOffsets[e.crossTileID];t&&(this.variableOffsets[e.crossTileID]=t,this.markUsedJustification(a,t.anchor,e,d));}}}if(A=z,S=A&&A.box&&A.box.length>0,D=A&&A.offscreen,e.useRuntimeCollisionCircles){const i=a.text.placedSymbolArray.get(e.centerJustifiedTextSymbolIndex),h=t.aj(a.textSizeData,d,i),c=o.get("text-padding");R=this.collisionIndex.placeCollisionCircles(f,i,a.lineVertexArray,a.glyphOffsetArray,h,r,n,l,s,b,_.predicate,e.collisionCircleDiameter,c,C),R.circles.length&&R.collisionDetected&&!s&&t.w("Collisions detected, but collision boxes are not shown"),S=g||R.circles.length>0&&!R.collisionDetected,D=D&&R.offscreen;}if(u.iconFeatureIndex&&(O=u.iconFeatureIndex),u.iconBox){const t=t=>{const e=w&&M?Rt(t,M.x,M.y,y,b,this.transform.angle):t;return this.collisionIndex.placeCollisionBox(e,v,h,r,_.predicate,C)};L&&L.box&&L.box.length&&u.verticalIconBox?(k=t(u.verticalIconBox),P=k.box.length>0):(k=t(u.iconBox),P=k.box.length>0),D=D&&k.offscreen;}const U=p||0===e.numHorizontalGlyphVertices&&0===e.numVerticalGlyphVertices,Z=m||0===e.numIconVertices;if(U||Z?Z?U||(P=P&&S):S=P&&S:P=S=P&&S,S&&A&&A.box&&this.collisionIndex.insertCollisionBox(A.box,f,o.get("text-ignore-placement"),a.bucketInstanceId,L&&L.box&&B?B:F,_.ID),P&&k&&this.collisionIndex.insertCollisionBox(k.box,v,o.get("icon-ignore-placement"),a.bucketInstanceId,O,_.ID),R&&(S&&this.collisionIndex.insertCollisionCircles(R.circles,f,o.get("text-ignore-placement"),a.bucketInstanceId,F,_.ID),s)){const t=a.bucketInstanceId;let e=this.collisionCircleArrays[t];void 0===e&&(e=this.collisionCircleArrays[t]=new Mt);for(let t=0;t=0;--e){const i=t[e];P(a.symbolInstances.get(i),a.collisionArrays[i]);}}else for(let t=e.symbolInstanceStart;t=0&&(e.text.placedSymbolArray.get(t).crossTileID=o>=0&&t!==o?0:s.crossTileID);}markUsedOrientation(e,i,s){const a=i===t.ai.horizontal||i===t.ai.horizontalOnly?i:0,o=i===t.ai.vertical?i:0,r=[s.leftJustifiedTextSymbolIndex,s.centerJustifiedTextSymbolIndex,s.rightJustifiedTextSymbolIndex];for(const t of r)e.text.placedSymbolArray.get(t).placedOrientation=a;s.verticalPlacedTextSymbolIndex&&(e.text.placedSymbolArray.get(s.verticalPlacedTextSymbolIndex).placedOrientation=o);}commit(t){this.commitTime=t,this.zoomAtLastRecencyCheck=this.transform.zoom;const e=this.prevPlacement;let i=!1;this.prevZoomAdjustment=e?e.zoomAdjustment(this.transform.zoom):0;const s=e?e.symbolFadeChange(t):1,a=e?e.opacities:{},o=e?e.variableOffsets:{},r=e?e.placedOrientations:{};for(const t in this.placements){const e=this.placements[t],o=a[t];o?(this.opacities[t]=new Pt(o,s,e.text,e.icon),i=i||e.text!==o.text.placed||e.icon!==o.icon.placed):(this.opacities[t]=new Pt(null,s,e.text,e.icon,e.skipFade),i=i||e.text||e.icon);}for(const t in a){const e=a[t];if(!this.opacities[t]){const a=new Pt(e,s,!1,!1);a.isHidden()||(this.opacities[t]=a,i=i||e.text.placed||e.icon.placed);}}for(const t in o)this.variableOffsets[t]||!this.opacities[t]||this.opacities[t].isHidden()||(this.variableOffsets[t]=o[t]);for(const t in r)this.placedOrientations[t]||!this.opacities[t]||this.opacities[t].isHidden()||(this.placedOrientations[t]=r[t]);if(e&&void 0===e.lastPlacementChangeTime)throw new Error("Last placement time for previous placement is not defined");i?this.lastPlacementChangeTime=t:"number"!=typeof this.lastPlacementChangeTime&&(this.lastPlacementChangeTime=e?e.lastPlacementChangeTime:t);}updateLayerOpacities(t,e){const i={};for(const s of e){const e=s.getBucket(t);e&&s.latestFeatureIndex&&t.id===e.layerIds[0]&&this.updateBucketOpacities(e,i,s.collisionBoxArray);}}updateBucketOpacities(e,i,s){e.hasTextData()&&(e.text.opacityVertexArray.clear(),e.text.hasVisibleVertices=!1),e.hasIconData()&&(e.icon.opacityVertexArray.clear(),e.icon.hasVisibleVertices=!1),e.hasIconCollisionBoxData()&&e.iconCollisionBox.collisionVertexArray.clear(),e.hasTextCollisionBoxData()&&e.textCollisionBox.collisionVertexArray.clear();const a=e.layers[0],o=a.layout,r=new Pt(null,0,!1,!1,!0),n=o.get("text-allow-overlap"),l=o.get("icon-allow-overlap"),h=a._unevaluatedLayout.hasValue("text-variable-anchor")||a._unevaluatedLayout.hasValue("text-variable-anchor-offset"),c="map"===o.get("text-rotation-alignment"),u="map"===o.get("text-pitch-alignment"),d="none"!==o.get("icon-text-fit"),_=new Pt(null,0,n&&(l||!e.hasIconData()||o.get("icon-optional")),l&&(n||!e.hasTextData()||o.get("text-optional")),!0);!e.collisionArrays&&s&&(e.hasIconCollisionBoxData()||e.hasTextCollisionBoxData())&&e.deserializeCollisionBoxes(s);const p=(t,e,i)=>{for(let s=0;s0,g=this.placedOrientations[a.crossTileID],v=g===t.ai.vertical,x=g===t.ai.horizontal||g===t.ai.horizontalOnly;if(o>0||n>0){const t=Vt(m.text);p(e.text,o,v?qt:t),p(e.text,n,x?qt:t);const i=m.text.isHidden();[a.rightJustifiedTextSymbolIndex,a.centerJustifiedTextSymbolIndex,a.leftJustifiedTextSymbolIndex].forEach((t=>{t>=0&&(e.text.placedSymbolArray.get(t).hidden=i||v?1:0);})),a.verticalPlacedTextSymbolIndex>=0&&(e.text.placedSymbolArray.get(a.verticalPlacedTextSymbolIndex).hidden=i||x?1:0);const s=this.variableOffsets[a.crossTileID];s&&this.markUsedJustification(e,s.anchor,a,g);const r=this.placedOrientations[a.crossTileID];r&&(this.markUsedJustification(e,"left",a,r),this.markUsedOrientation(e,r,a));}if(f){const t=Vt(m.icon),i=!(d&&a.verticalPlacedIconSymbolIndex&&v);a.placedIconSymbolIndex>=0&&(p(e.icon,a.numIconVertices,i?t:qt),e.icon.placedSymbolArray.get(a.placedIconSymbolIndex).hidden=m.icon.isHidden()),a.verticalPlacedIconSymbolIndex>=0&&(p(e.icon,a.numVerticalIconVertices,i?qt:t),e.icon.placedSymbolArray.get(a.verticalPlacedIconSymbolIndex).hidden=m.icon.isHidden());}if(e.hasIconCollisionBoxData()||e.hasTextCollisionBoxData()){const i=e.collisionArrays[s];if(i){let s=new t.P(0,0);if(i.textBox||i.verticalTextBox){let t=!0;if(h){const e=this.variableOffsets[l];e?(s=At(e.anchor,e.width,e.height,e.textOffset,e.textBoxScale),c&&s._rotate(u?this.transform.angle:-this.transform.angle)):t=!1;}i.textBox&&Ft(e.textCollisionBox.collisionVertexArray,m.text.placed,!t||v,s.x,s.y),i.verticalTextBox&&Ft(e.textCollisionBox.collisionVertexArray,m.text.placed,!t||x,s.x,s.y);}const a=Boolean(!x&&i.verticalIconBox);i.iconBox&&Ft(e.iconCollisionBox.collisionVertexArray,m.icon.placed,a,d?s.x:0,d?s.y:0),i.verticalIconBox&&Ft(e.iconCollisionBox.collisionVertexArray,m.icon.placed,!a,d?s.x:0,d?s.y:0);}}}if(e.sortFeatures(this.transform.angle),this.retainedQueryData[e.bucketInstanceId]&&(this.retainedQueryData[e.bucketInstanceId].featureSortOrder=e.featureSortOrder),e.hasTextData()&&e.text.opacityVertexBuffer&&e.text.opacityVertexBuffer.updateData(e.text.opacityVertexArray),e.hasIconData()&&e.icon.opacityVertexBuffer&&e.icon.opacityVertexBuffer.updateData(e.icon.opacityVertexArray),e.hasIconCollisionBoxData()&&e.iconCollisionBox.collisionVertexBuffer&&e.iconCollisionBox.collisionVertexBuffer.updateData(e.iconCollisionBox.collisionVertexArray),e.hasTextCollisionBoxData()&&e.textCollisionBox.collisionVertexBuffer&&e.textCollisionBox.collisionVertexBuffer.updateData(e.textCollisionBox.collisionVertexArray),e.text.opacityVertexArray.length!==e.text.layoutVertexArray.length/4)throw new Error(`bucket.text.opacityVertexArray.length (= ${e.text.opacityVertexArray.length}) !== bucket.text.layoutVertexArray.length (= ${e.text.layoutVertexArray.length}) / 4`);if(e.icon.opacityVertexArray.length!==e.icon.layoutVertexArray.length/4)throw new Error(`bucket.icon.opacityVertexArray.length (= ${e.icon.opacityVertexArray.length}) !== bucket.icon.layoutVertexArray.length (= ${e.icon.layoutVertexArray.length}) / 4`);if(e.bucketInstanceId in this.collisionCircleArrays){const t=this.collisionCircleArrays[e.bucketInstanceId];e.placementInvProjMatrix=t.invProjMatrix,e.placementViewportMatrix=t.viewportMatrix,e.collisionCircleArray=t.circles,delete this.collisionCircleArrays[e.bucketInstanceId];}}symbolFadeChange(t){return 0===this.fadeDuration?1:(t-this.commitTime)/this.fadeDuration+this.prevZoomAdjustment}zoomAdjustment(t){return Math.max(0,(this.transform.zoom-t)/1.5)}hasTransitions(t){return this.stale||t-this.lastPlacementChangeTimet}setStale(){this.stale=!0;}}function Ft(t,e,i,s,a){t.emplaceBack(e?1:0,i?1:0,s||0,a||0),t.emplaceBack(e?1:0,i?1:0,s||0,a||0),t.emplaceBack(e?1:0,i?1:0,s||0,a||0),t.emplaceBack(e?1:0,i?1:0,s||0,a||0);}const Bt=Math.pow(2,25),Ot=Math.pow(2,24),Nt=Math.pow(2,17),Ut=Math.pow(2,16),Zt=Math.pow(2,9),Gt=Math.pow(2,8),jt=Math.pow(2,1);function Vt(t){if(0===t.opacity&&!t.placed)return 0;if(1===t.opacity&&t.placed)return 4294967295;const e=t.placed?1:0,i=Math.floor(127*t.opacity);return i*Bt+e*Ot+i*Nt+e*Ut+i*Zt+e*Gt+i*jt+e}const qt=0;class $t{constructor(t){this._sortAcrossTiles="viewport-y"!==t.layout.get("symbol-z-order")&&!t.layout.get("symbol-sort-key").isConstant(),this._currentTileIndex=0,this._currentPartIndex=0,this._seenCrossTileIDs={},this._bucketParts=[];}continuePlacement(t,e,i,s,a){const o=this._bucketParts;for(;this._currentTileIndext.sortKey-e.sortKey)));this._currentPartIndex!this._forceFullPlacement&&t.h.now()-a>2;for(;this._currentPlacementIndex>=0;){const t=i[e[this._currentPlacementIndex]],a=this.placement.collisionIndex.transform.zoom;if("symbol"===t.type&&(!t.minzoom||t.minzoom<=a)&&(!t.maxzoom||t.maxzoom>a)){if(this._inProgressLayer||(this._inProgressLayer=new $t(t)),this._inProgressLayer.continuePlacement(s[t.source],this.placement,this._showCollisionBoxes,t,o))return;delete this._inProgressLayer;}this._currentPlacementIndex--;}this._done=!0;}commit(t){return this.placement.commit(t),this.placement}}const Ht=512/t.N/2;class Xt{constructor(e,i,s){this.tileID=e,this.bucketInstanceId=s,this._symbolsByKey={};const a=new Map;for(let t=0;t({x:Math.floor(t.anchorX*Ht),y:Math.floor(t.anchorY*Ht)}))),crossTileIDs:i.map((t=>t.crossTileID))};if(s.positions.length>128){const e=new t.av(s.positions.length,16,Uint16Array);for(const{x:t,y:i}of s.positions)e.add(t,i);e.finish(),delete s.positions,s.index=e;}this._symbolsByKey[e]=s;}}getScaledCoordinates(e,i){const{x:s,y:a,z:o}=this.tileID.canonical,{x:r,y:n,z:l}=i.canonical,h=Ht/Math.pow(2,l-o),c=(n*t.N+e.anchorY)*h,u=a*t.N*Ht;return {x:Math.floor((r*t.N+e.anchorX)*h-s*t.N*Ht),y:Math.floor(c-u)}}findMatches(t,e,i){const s=this.tileID.canonical.zt))}}class Kt{constructor(){this.maxCrossTileID=0;}generate(){return ++this.maxCrossTileID}}class Qt{constructor(){this.indexes={},this.usedCrossTileIDs={},this.lng=0;}handleWrapJump(t){const e=Math.round((t-this.lng)/360);if(0!==e)for(const t in this.indexes){const i=this.indexes[t],s={};for(const t in i){const a=i[t];a.tileID=a.tileID.unwrapTo(a.tileID.wrap+e),s[a.tileID.key]=a;}this.indexes[t]=s;}this.lng=t;}addBucket(t,e,i){if(this.indexes[t.overscaledZ]&&this.indexes[t.overscaledZ][t.key]){if(this.indexes[t.overscaledZ][t.key].bucketInstanceId===e.bucketInstanceId)return !1;this.removeBucketCrossTileIDs(t.overscaledZ,this.indexes[t.overscaledZ][t.key]);}for(let t=0;tt.overscaledZ)for(const i in a){const o=a[i];o.tileID.isChildOf(t)&&o.findMatches(e.symbolInstances,t,s);}else {const o=a[t.scaledTo(Number(i)).key];o&&o.findMatches(e.symbolInstances,t,s);}}for(let t=0;t{e[t]=!0;}));for(const t in this.layerIndexes)e[t]||delete this.layerIndexes[t];}}const Jt=(e,i)=>t.x(e,i&&i.filter((t=>"source.canvas"!==t.identifier))),te=t.F(t.ax,["addLayer","removeLayer","setPaintProperty","setLayoutProperty","setFilter","addSource","removeSource","setLayerZoomRange","setLight","setTransition","setGeoJSONSourceData","setGlyphs","setSprite"]),ee=t.F(t.ax,["setCenter","setZoom","setBearing","setPitch"]),ie=t.aw();class se extends t.E{constructor(e,i={}){super(),this.map=e,this.dispatcher=new M(st(),this,e._getMapId()),this.imageManager=new b,this.imageManager.setEventedParent(this),this.glyphManager=new E(e._requestManager,i.localIdeographFontFamily),this.lineAtlas=new D(256,512),this.crossTileSymbolIndex=new Yt,this._spritesImagesIds={},this._layers={},this._order=[],this.sourceCaches={},this.zoomHistory=new t.ay,this._loaded=!1,this._availableImages=[],this._resetUpdates(),this.dispatcher.broadcast("setReferrer",t.az());const s=this;this._rtlTextPluginCallback=se.registerForPluginStateChange((e=>{s.dispatcher.broadcast("syncRTLPluginState",{pluginStatus:e.pluginStatus,pluginURL:e.pluginURL},((e,i)=>{if(t.aA(e),i&&i.every((t=>t)))for(const t in s.sourceCaches){const e=s.sourceCaches[t].getSource().type;"vector"!==e&&"geojson"!==e||s.sourceCaches[t].reload();}}));})),this.on("data",(t=>{if("source"!==t.dataType||"metadata"!==t.sourceDataType)return;const e=this.sourceCaches[t.sourceId];if(!e)return;const i=e.getSource();if(i&&i.vectorLayerIds)for(const t in this._layers){const e=this._layers[t];e.source===i.id&&this._validateLayer(e);}}));}loadURL(e,i={},s){this.fire(new t.k("dataloading",{dataType:"style"})),i.validate="boolean"!=typeof i.validate||i.validate;const a=this.map._requestManager.transformRequest(e,c.Style);this._request=t.f(a,((e,a)=>{this._request=null,e?this.fire(new t.j(e)):a&&this._load(a,i,s);}));}loadJSON(e,i={},s){this.fire(new t.k("dataloading",{dataType:"style"})),this._request=t.h.frame((()=>{this._request=null,i.validate=!1!==i.validate,this._load(e,i,s);}));}loadEmpty(){this.fire(new t.k("dataloading",{dataType:"style"})),this._load(ie,{validate:!1});}_load(e,i,s){var a;const o=i.transformStyle?i.transformStyle(s,e):e;if(!i.validate||!Jt(this,t.y(o))){this._loaded=!0,this.stylesheet=o;for(const t in o.sources)this.addSource(t,o.sources[t],{validate:!1});o.sprite?this._loadSprite(o.sprite):this.imageManager.setLoaded(!0),this.glyphManager.setURL(o.glyphs),this._createLayers(),this.light=new P(this.stylesheet.light),this.map.setTerrain(null!==(a=this.stylesheet.terrain)&&void 0!==a?a:null),this.fire(new t.k("data",{dataType:"style"})),this.fire(new t.k("style.load"));}}_createLayers(){const e=t.aB(this.stylesheet.layers);this.dispatcher.broadcast("setLayers",e),this._order=e.map((t=>t.id)),this._layers={},this._serializedLayers=null;for(const i of e){const e=t.aC(i);e.setEventedParent(this,{layer:{id:i.id}}),this._layers[i.id]=e;}}_loadSprite(e,i=!1,s=void 0){this.imageManager.setLoaded(!1),this._spriteRequest=function(e,i,s,a){const o=g(e),r=o.length,n=s>1?"@2x":"",l={},u={},d={};for(const{id:e,url:s}of o){const o=i.transformRequest(i.normalizeSpriteURL(s,n,".json"),c.SpriteJSON),_=`${e}_${o.url}`;l[_]=t.f(o,((t,i)=>{delete l[_],u[e]=i,v(a,u,d,t,r);}));const p=i.transformRequest(i.normalizeSpriteURL(s,n,".png"),c.SpriteImage),m=`${e}_${p.url}`;l[m]=h.getImage(p,((t,i)=>{delete l[m],d[e]=i,v(a,u,d,t,r);}));}return {cancel(){for(const t of Object.values(l))t.cancel();}}}(e,this.map._requestManager,this.map.getPixelRatio(),((e,a)=>{if(this._spriteRequest=null,e)this.fire(new t.j(e));else if(a)for(const t in a){this._spritesImagesIds[t]=[];const e=this._spritesImagesIds[t]?this._spritesImagesIds[t].filter((t=>!(t in a))):[];for(const t of e)this.imageManager.removeImage(t),this._changedImages[t]=!0;for(const e in a[t]){const s="default"===t?e:`${t}:${e}`;this._spritesImagesIds[t].push(s),s in this.imageManager.images?this.imageManager.updateImage(s,a[t][e],!1):this.imageManager.addImage(s,a[t][e]),i&&(this._changedImages[s]=!0);}}this.imageManager.setLoaded(!0),this._availableImages=this.imageManager.listImages(),i&&(this._changed=!0),this.dispatcher.broadcast("setImages",this._availableImages),this.fire(new t.k("data",{dataType:"style"})),s&&s(e);}));}_unloadSprite(){for(const t of Object.values(this._spritesImagesIds).flat())this.imageManager.removeImage(t),this._changedImages[t]=!0;this._spritesImagesIds={},this._availableImages=this.imageManager.listImages(),this._changed=!0,this.dispatcher.broadcast("setImages",this._availableImages),this.fire(new t.k("data",{dataType:"style"}));}_validateLayer(e){const i=this.sourceCaches[e.source];if(!i)return;const s=e.sourceLayer;if(!s)return;const a=i.getSource();("geojson"===a.type||a.vectorLayerIds&&-1===a.vectorLayerIds.indexOf(s))&&this.fire(new t.j(new Error(`Source layer "${s}" does not exist on source "${a.id}" as specified by style layer "${e.id}".`)));}loaded(){if(!this._loaded)return !1;if(Object.keys(this._updatedSources).length)return !1;for(const t in this.sourceCaches)if(!this.sourceCaches[t].loaded())return !1;return !!this.imageManager.isLoaded()}_serializeByIds(t){const e=this._serializedAllLayers();if(!t||0===t.length)return Object.values(e);const i=[];for(const s of t)e[s]&&i.push(e[s]);return i}_serializedAllLayers(){let t=this._serializedLayers;if(t)return t;t=this._serializedLayers={};const e=Object.keys(this._layers);for(const i of e){const e=this._layers[i];"custom"!==e.type&&(t[i]=e.serialize());}return t}hasTransitions(){if(this.light&&this.light.hasTransition())return !0;for(const t in this.sourceCaches)if(this.sourceCaches[t].hasTransition())return !0;for(const t in this._layers)if(this._layers[t].hasTransition())return !0;return !1}_checkLoaded(){if(!this._loaded)throw new Error("Style is not done loading.")}update(e){if(!this._loaded)return;const i=this._changed;if(this._changed){const t=Object.keys(this._updatedLayers),i=Object.keys(this._removedLayers);(t.length||i.length)&&this._updateWorkerLayers(t,i);for(const t in this._updatedSources){const e=this._updatedSources[t];if("reload"===e)this._reloadSource(t);else {if("clear"!==e)throw new Error(`Invalid action ${e}`);this._clearSource(t);}}this._updateTilesForChangedImages(),this._updateTilesForChangedGlyphs();for(const t in this._updatedPaintProps)this._layers[t].updateTransitions(e);this.light.updateTransitions(e),this._resetUpdates();}const s={};for(const t in this.sourceCaches){const e=this.sourceCaches[t];s[t]=e.used,e.used=!1;}for(const t of this._order){const i=this._layers[t];i.recalculate(e,this._availableImages),!i.isHidden(e.zoom)&&i.source&&(this.sourceCaches[i.source].used=!0);}for(const e in s){const i=this.sourceCaches[e];s[e]!==i.used&&i.fire(new t.k("data",{sourceDataType:"visibility",dataType:"source",sourceId:e}));}this.light.recalculate(e),this.z=e.zoom,i&&this.fire(new t.k("data",{dataType:"style"}));}_updateTilesForChangedImages(){const t=Object.keys(this._changedImages);if(t.length){for(const e in this.sourceCaches)this.sourceCaches[e].reloadTilesForDependencies(["icons","patterns"],t);this._changedImages={};}}_updateTilesForChangedGlyphs(){if(this._glyphsDidChange){for(const t in this.sourceCaches)this.sourceCaches[t].reloadTilesForDependencies(["glyphs"],[""]);this._glyphsDidChange=!1;}}_updateWorkerLayers(t,e){this.dispatcher.broadcast("updateLayers",{layers:this._serializeByIds(t),removedIds:e});}_resetUpdates(){this._changed=!1,this._updatedLayers={},this._removedLayers={},this._updatedSources={},this._updatedPaintProps={},this._changedImages={},this._glyphsDidChange=!1;}setState(e,i={}){this._checkLoaded();const s=this.serialize();if(e=i.transformStyle?i.transformStyle(s,e):e,Jt(this,t.y(e)))return !1;(e=t.aD(e)).layers=t.aB(e.layers);const a=t.aE(s,e).filter((t=>!(t.command in ee)));if(0===a.length)return !1;const o=a.filter((t=>!(t.command in te)));if(o.length>0)throw new Error(`Unimplemented: ${o.map((t=>t.command)).join(", ")}.`);for(const t of a)"setTransition"!==t.command&&this[t.command].apply(this,t.args);return this.stylesheet=e,this._serializedLayers=null,!0}addImage(e,i){if(this.getImage(e))return this.fire(new t.j(new Error(`An image named "${e}" already exists.`)));this.imageManager.addImage(e,i),this._afterImageUpdated(e);}updateImage(t,e){this.imageManager.updateImage(t,e);}getImage(t){return this.imageManager.getImage(t)}removeImage(e){if(!this.getImage(e))return this.fire(new t.j(new Error(`An image named "${e}" does not exist.`)));this.imageManager.removeImage(e),this._afterImageUpdated(e);}_afterImageUpdated(e){this._availableImages=this.imageManager.listImages(),this._changedImages[e]=!0,this._changed=!0,this.dispatcher.broadcast("setImages",this._availableImages),this.fire(new t.k("data",{dataType:"style"}));}listImages(){return this._checkLoaded(),this.imageManager.listImages()}addSource(e,i,s={}){if(this._checkLoaded(),void 0!==this.sourceCaches[e])throw new Error(`Source "${e}" already exists.`);if(!i.type)throw new Error(`The type property must be defined, but only the following properties were given: ${Object.keys(i).join(", ")}.`);if(["vector","raster","geojson","video","image"].indexOf(i.type)>=0&&this._validate(t.y.source,`sources.${e}`,i,null,s))return;this.map&&this.map._collectResourceTiming&&(i.collectResourceTiming=!0);const a=this.sourceCaches[e]=new K(e,i,this.dispatcher);a.style=this,a.setEventedParent(this,(()=>({isSourceLoaded:a.loaded(),source:a.serialize(),sourceId:e}))),a.onAdd(this.map),this._changed=!0;}removeSource(e){if(this._checkLoaded(),void 0===this.sourceCaches[e])throw new Error("There is no source with this ID");for(const i in this._layers)if(this._layers[i].source===e)return this.fire(new t.j(new Error(`Source "${e}" cannot be removed while layer "${i}" is using it.`)));const i=this.sourceCaches[e];delete this.sourceCaches[e],delete this._updatedSources[e],i.fire(new t.k("data",{sourceDataType:"metadata",dataType:"source",sourceId:e})),i.setEventedParent(null),i.onRemove(this.map),this._changed=!0;}setGeoJSONSourceData(t,e){if(this._checkLoaded(),void 0===this.sourceCaches[t])throw new Error(`There is no source with this ID=${t}`);const i=this.sourceCaches[t].getSource();if("geojson"!==i.type)throw new Error(`geojsonSource.type is ${i.type}, which is !== 'geojson`);i.setData(e),this._changed=!0;}getSource(t){return this.sourceCaches[t]&&this.sourceCaches[t].getSource()}addLayer(e,i,s={}){this._checkLoaded();const a=e.id;if(this.getLayer(a))return void this.fire(new t.j(new Error(`Layer "${a}" already exists on this map.`)));let o;if("custom"===e.type){if(Jt(this,t.aF(e)))return;o=t.aC(e);}else {if("source"in e&&"object"==typeof e.source&&(this.addSource(a,e.source),e=t.aD(e),e=t.e(e,{source:a})),this._validate(t.y.layer,`layers.${a}`,e,{arrayIndex:-1},s))return;o=t.aC(e),this._validateLayer(o),o.setEventedParent(this,{layer:{id:a}});}const r=i?this._order.indexOf(i):this._order.length;if(i&&-1===r)this.fire(new t.j(new Error(`Cannot add layer "${a}" before non-existing layer "${i}".`)));else {if(this._order.splice(r,0,a),this._layerOrderChanged=!0,this._layers[a]=o,this._removedLayers[a]&&o.source&&"custom"!==o.type){const t=this._removedLayers[a];delete this._removedLayers[a],t.type!==o.type?this._updatedSources[o.source]="clear":(this._updatedSources[o.source]="reload",this.sourceCaches[o.source].pause());}this._updateLayer(o),o.onAdd&&o.onAdd(this.map);}}moveLayer(e,i){if(this._checkLoaded(),this._changed=!0,!this._layers[e])return void this.fire(new t.j(new Error(`The layer '${e}' does not exist in the map's style and cannot be moved.`)));if(e===i)return;const s=this._order.indexOf(e);this._order.splice(s,1);const a=i?this._order.indexOf(i):this._order.length;i&&-1===a?this.fire(new t.j(new Error(`Cannot move layer "${e}" before non-existing layer "${i}".`))):(this._order.splice(a,0,e),this._layerOrderChanged=!0);}removeLayer(e){this._checkLoaded();const i=this._layers[e];if(!i)return void this.fire(new t.j(new Error(`Cannot remove non-existing layer "${e}".`)));i.setEventedParent(null);const s=this._order.indexOf(e);this._order.splice(s,1),this._layerOrderChanged=!0,this._changed=!0,this._removedLayers[e]=i,delete this._layers[e],this._serializedLayers&&delete this._serializedLayers[e],delete this._updatedLayers[e],delete this._updatedPaintProps[e],i.onRemove&&i.onRemove(this.map);}getLayer(t){return this._layers[t]}getLayersOrder(){return [...this._order]}hasLayer(t){return t in this._layers}setLayerZoomRange(e,i,s){this._checkLoaded();const a=this.getLayer(e);a?a.minzoom===i&&a.maxzoom===s||(null!=i&&(a.minzoom=i),null!=s&&(a.maxzoom=s),this._updateLayer(a)):this.fire(new t.j(new Error(`Cannot set the zoom range of non-existing layer "${e}".`)));}setFilter(e,i,s={}){this._checkLoaded();const a=this.getLayer(e);if(a){if(!t.aG(a.filter,i))return null==i?(a.filter=void 0,void this._updateLayer(a)):void(this._validate(t.y.filter,`layers.${a.id}.filter`,i,null,s)||(a.filter=t.aD(i),this._updateLayer(a)))}else this.fire(new t.j(new Error(`Cannot filter non-existing layer "${e}".`)));}getFilter(e){return t.aD(this.getLayer(e).filter)}setLayoutProperty(e,i,s,a={}){this._checkLoaded();const o=this.getLayer(e);o?t.aG(o.getLayoutProperty(i),s)||(o.setLayoutProperty(i,s,a),this._updateLayer(o)):this.fire(new t.j(new Error(`Cannot style non-existing layer "${e}".`)));}getLayoutProperty(e,i){const s=this.getLayer(e);if(s)return s.getLayoutProperty(i);this.fire(new t.j(new Error(`Cannot get style of non-existing layer "${e}".`)));}setPaintProperty(e,i,s,a={}){this._checkLoaded();const o=this.getLayer(e);o?t.aG(o.getPaintProperty(i),s)||(o.setPaintProperty(i,s,a)&&this._updateLayer(o),this._changed=!0,this._updatedPaintProps[e]=!0):this.fire(new t.j(new Error(`Cannot style non-existing layer "${e}".`)));}getPaintProperty(t,e){return this.getLayer(t).getPaintProperty(e)}setFeatureState(e,i){this._checkLoaded();const s=e.source,a=e.sourceLayer,o=this.sourceCaches[s];if(void 0===o)return void this.fire(new t.j(new Error(`The source '${s}' does not exist in the map's style.`)));const r=o.getSource().type;"geojson"===r&&a?this.fire(new t.j(new Error("GeoJSON sources cannot have a sourceLayer parameter."))):"vector"!==r||a?(void 0===e.id&&this.fire(new t.j(new Error("The feature id parameter must be provided."))),o.setFeatureState(a,e.id,i)):this.fire(new t.j(new Error("The sourceLayer parameter must be provided for vector source types.")));}removeFeatureState(e,i){this._checkLoaded();const s=e.source,a=this.sourceCaches[s];if(void 0===a)return void this.fire(new t.j(new Error(`The source '${s}' does not exist in the map's style.`)));const o=a.getSource().type,r="vector"===o?e.sourceLayer:void 0;"vector"!==o||r?i&&"string"!=typeof e.id&&"number"!=typeof e.id?this.fire(new t.j(new Error("A feature id is required to remove its specific state property."))):a.removeFeatureState(r,e.id,i):this.fire(new t.j(new Error("The sourceLayer parameter must be provided for vector source types.")));}getFeatureState(e){this._checkLoaded();const i=e.source,s=e.sourceLayer,a=this.sourceCaches[i];if(void 0!==a)return "vector"!==a.getSource().type||s?(void 0===e.id&&this.fire(new t.j(new Error("The feature id parameter must be provided."))),a.getFeatureState(s,e.id)):void this.fire(new t.j(new Error("The sourceLayer parameter must be provided for vector source types.")));this.fire(new t.j(new Error(`The source '${i}' does not exist in the map's style.`)));}getTransition(){return t.e({duration:300,delay:0},this.stylesheet&&this.stylesheet.transition)}serialize(){if(!this._loaded)return;const e=t.aH(this.sourceCaches,(t=>t.serialize())),i=this._serializeByIds(this._order),s=this.map.getTerrain()||void 0,a=this.stylesheet;return t.aI({version:a.version,name:a.name,metadata:a.metadata,light:a.light,center:a.center,zoom:a.zoom,bearing:a.bearing,pitch:a.pitch,sprite:a.sprite,glyphs:a.glyphs,transition:a.transition,sources:e,layers:i,terrain:s},(t=>void 0!==t))}_updateLayer(t){this._updatedLayers[t.id]=!0,t.source&&!this._updatedSources[t.source]&&"raster"!==this.sourceCaches[t.source].getSource().type&&(this._updatedSources[t.source]="reload",this.sourceCaches[t.source].pause()),this._serializedLayers=null,this._changed=!0;}_flattenAndSortRenderedFeatures(t){const e=t=>"fill-extrusion"===this._layers[t].type,i={},s=[];for(let a=this._order.length-1;a>=0;a--){const o=this._order[a];if(e(o)){i[o]=a;for(const e of t){const t=e[o];if(t)for(const e of t)s.push(e);}}}s.sort(((t,e)=>e.intersectionZ-t.intersectionZ));const a=[];for(let o=this._order.length-1;o>=0;o--){const r=this._order[o];if(e(r))for(let t=s.length-1;t>=0;t--){const e=s[t].feature;if(i[e.layer.id]{const s=i.featureSortOrder;if(s){const i=s.indexOf(t.featureIndex);return s.indexOf(e.featureIndex)-i}return e.featureIndex-t.featureIndex}));for(const t of a)e.push(t);}}for(const e in n)n[e].forEach((s=>{const a=s.feature,o=i[t[e].source].getFeatureState(a.layer["source-layer"],a.id);a.source=a.layer.source,a.layer["source-layer"]&&(a.sourceLayer=a.layer["source-layer"]),a.state=o;}));return n}(this._layers,r,this.sourceCaches,e,i,this.placement.collisionIndex,this.placement.retainedQueryData)),this._flattenAndSortRenderedFeatures(o)}querySourceFeatures(e,i){i&&i.filter&&this._validate(t.y.filter,"querySourceFeatures.filter",i.filter,null,i);const s=this.sourceCaches[e];return s?function(t,e){const i=t.getRenderableIds().map((e=>t.getTileByID(e))),s=[],a={};for(let t=0;t{G[t]=e;})(t,e),e.workerSourceURL?void this.dispatcher.broadcast("loadWorkerSource",{name:t,url:e.workerSourceURL},i):i(null,null))}getLight(){return this.light.getLight()}setLight(e,i={}){this._checkLoaded();const s=this.light.getLight();let a=!1;for(const i in e)if(!t.aG(e[i],s[i])){a=!0;break}if(!a)return;const o={now:t.h.now(),transition:t.e({duration:300,delay:0},this.stylesheet.transition)};this.light.setLight(e,i),this.light.updateTransitions(o);}_validate(e,i,s,a,o={}){return (!o||!1!==o.validate)&&Jt(this,e.call(t.y,t.e({key:i,style:this.serialize(),value:s,styleSpec:t.v},a)))}_remove(e=!0){this._request&&(this._request.cancel(),this._request=null),this._spriteRequest&&(this._spriteRequest.cancel(),this._spriteRequest=null),t.aJ.off("pluginStateChange",this._rtlTextPluginCallback);for(const t in this._layers)this._layers[t].setEventedParent(null);for(const t in this.sourceCaches){const e=this.sourceCaches[t];e.setEventedParent(null),e.onRemove(this.map);}this.imageManager.setEventedParent(null),this.setEventedParent(null),this.dispatcher.remove(e);}_clearSource(t){this.sourceCaches[t].clearTiles();}_reloadSource(t){this.sourceCaches[t].resume(),this.sourceCaches[t].reload();}_updateSources(t){for(const e in this.sourceCaches)this.sourceCaches[e].update(t,this.map.terrain);}_generateCollisionBoxes(){for(const t in this.sourceCaches)this._reloadSource(t);}_updatePlacement(e,i,s,a,o=!1){let r=!1,n=!1;const l={};for(const t of this._order){const i=this._layers[t];if("symbol"!==i.type)continue;if(!l[i.source]){const t=this.sourceCaches[i.source];l[i.source]=t.getRenderableIds(!0).map((e=>t.getTileByID(e))).sort(((t,e)=>e.tileID.overscaledZ-t.tileID.overscaledZ||(t.tileID.isLessThan(e.tileID)?-1:1)));}const s=this.crossTileSymbolIndex.addLayer(i,l[i.source],e.center.lng);r=r||s;}if(this.crossTileSymbolIndex.pruneUnusedLayers(this._order),((o=o||this._layerOrderChanged||0===s)||!this.pauseablePlacement||this.pauseablePlacement.isDone()&&!this.placement.stillRecent(t.h.now(),e.zoom))&&(this.pauseablePlacement=new Wt(e,this.map.terrain,this._order,o,i,s,a,this.placement),this._layerOrderChanged=!1),this.pauseablePlacement.isDone()?this.placement.setStale():(this.pauseablePlacement.continuePlacement(this._order,this._layers,l),this.pauseablePlacement.isDone()&&(this.placement=this.pauseablePlacement.commit(t.h.now()),n=!0),r&&this.pauseablePlacement.placement.setStale()),n||r)for(const t of this._order){const e=this._layers[t];"symbol"===e.type&&this.placement.updateLayerOpacities(e,l[e.source]);}return !this.pauseablePlacement.isDone()||this.placement.hasTransitions(t.h.now())}_releaseSymbolFadeTiles(){for(const t in this.sourceCaches)this.sourceCaches[t].releaseSymbolFadeTiles();}getImages(t,e,i){this.imageManager.getImages(e.icons,i),this._updateTilesForChangedImages();const s=this.sourceCaches[e.source];s&&s.setDependencies(e.tileID.key,e.type,e.icons);}getGlyphs(t,e,i){this.glyphManager.getGlyphs(e.stacks,i);const s=this.sourceCaches[e.source];s&&s.setDependencies(e.tileID.key,e.type,[""]);}getResource(e,i,s){return t.m(i,s)}getGlyphsUrl(){return this.stylesheet.glyphs||null}setGlyphs(e,i={}){this._checkLoaded(),e&&this._validate(t.y.glyphs,"glyphs",e,null,i)||(this._glyphsDidChange=!0,this.stylesheet.glyphs=e,this.glyphManager.entries={},this.glyphManager.setURL(e));}addSprite(e,i,s={},a){this._checkLoaded();const o=[{id:e,url:i}],r=[...g(this.stylesheet.sprite),...o];this._validate(t.y.sprite,"sprite",r,null,s)||(this.stylesheet.sprite=r,this._loadSprite(o,!0,a));}removeSprite(e){this._checkLoaded();const i=g(this.stylesheet.sprite);if(i.find((t=>t.id===e))){if(this._spritesImagesIds[e])for(const t of this._spritesImagesIds[e])this.imageManager.removeImage(t),this._changedImages[t]=!0;i.splice(i.findIndex((t=>t.id===e)),1),this.stylesheet.sprite=i.length>0?i:void 0,delete this._spritesImagesIds[e],this._availableImages=this.imageManager.listImages(),this._changed=!0,this.dispatcher.broadcast("setImages",this._availableImages),this.fire(new t.k("data",{dataType:"style"}));}else this.fire(new t.j(new Error(`Sprite "${e}" doesn't exists on this map.`)));}getSprite(){return g(this.stylesheet.sprite)}setSprite(e,i={},s){this._checkLoaded(),e&&this._validate(t.y.sprite,"sprite",e,null,i)||(this.stylesheet.sprite=e,e?this._loadSprite(e,!0,s):(this._unloadSprite(),s&&s(null)));}}se.registerForPluginStateChange=t.aK;var ae=t.Q([{name:"a_pos",type:"Int16",components:2}]),oe="attribute vec3 a_pos3d;uniform mat4 u_matrix;uniform float u_ele_delta;varying vec2 v_texture_pos;varying float v_depth;void main() {float extent=8192.0;float ele_delta=a_pos3d.z==1.0 ? u_ele_delta : 0.0;v_texture_pos=a_pos3d.xy/extent;gl_Position=u_matrix*vec4(a_pos3d.xy,get_elevation(a_pos3d.xy)-ele_delta,1.0);v_depth=gl_Position.z/gl_Position.w;}";const re={prelude:ne("#ifdef GL_ES\nprecision mediump float;\n#else\n#if !defined(lowp)\n#define lowp\n#endif\n#if !defined(mediump)\n#define mediump\n#endif\n#if !defined(highp)\n#define highp\n#endif\n#endif\n","#ifdef GL_ES\nprecision highp float;\n#else\n#if !defined(lowp)\n#define lowp\n#endif\n#if !defined(mediump)\n#define mediump\n#endif\n#if !defined(highp)\n#define highp\n#endif\n#endif\nvec2 unpack_float(const float packedValue) {int packedIntValue=int(packedValue);int v0=packedIntValue/256;return vec2(v0,packedIntValue-v0*256);}vec2 unpack_opacity(const float packedOpacity) {int intOpacity=int(packedOpacity)/2;return vec2(float(intOpacity)/127.0,mod(packedOpacity,2.0));}vec4 decode_color(const vec2 encodedColor) {return vec4(unpack_float(encodedColor[0])/255.0,unpack_float(encodedColor[1])/255.0\n);}float unpack_mix_vec2(const vec2 packedValue,const float t) {return mix(packedValue[0],packedValue[1],t);}vec4 unpack_mix_color(const vec4 packedColors,const float t) {vec4 minColor=decode_color(vec2(packedColors[0],packedColors[1]));vec4 maxColor=decode_color(vec2(packedColors[2],packedColors[3]));return mix(minColor,maxColor,t);}vec2 get_pattern_pos(const vec2 pixel_coord_upper,const vec2 pixel_coord_lower,const vec2 pattern_size,const float tile_units_to_pixels,const vec2 pos) {vec2 offset=mod(mod(mod(pixel_coord_upper,pattern_size)*256.0,pattern_size)*256.0+pixel_coord_lower,pattern_size);return (tile_units_to_pixels*pos+offset)/pattern_size;}\n#ifdef TERRAIN3D\nuniform sampler2D u_terrain;uniform float u_terrain_dim;uniform mat4 u_terrain_matrix;uniform vec4 u_terrain_unpack;uniform float u_terrain_exaggeration;uniform highp sampler2D u_depth;\n#endif\nconst highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitShifts=vec4(1.)/bitSh;highp float unpack(highp vec4 color) {return dot(color,bitShifts);}highp float depthOpacity(vec3 frag) {\n#ifdef TERRAIN3D\nhighp float d=unpack(texture2D(u_depth,frag.xy*0.5+0.5))+0.0001-frag.z;return 1.0-max(0.0,min(1.0,-d*500.0));\n#else\nreturn 1.0;\n#endif\n}float calculate_visibility(vec4 pos) {\n#ifdef TERRAIN3D\nvec3 frag=pos.xyz/pos.w;highp float d=depthOpacity(frag);if (d > 0.95) return 1.0;return (d+depthOpacity(frag+vec3(0.0,0.01,0.0)))/2.0;\n#else\nreturn 1.0;\n#endif\n}float ele(vec2 pos) {\n#ifdef TERRAIN3D\nvec4 rgb=(texture2D(u_terrain,pos)*255.0)*u_terrain_unpack;return rgb.r+rgb.g+rgb.b-u_terrain_unpack.a;\n#else\nreturn 0.0;\n#endif\n}float get_elevation(vec2 pos) {\n#ifdef TERRAIN3D\nvec2 coord=(u_terrain_matrix*vec4(pos,0.0,1.0)).xy*u_terrain_dim+1.0;vec2 f=fract(coord);vec2 c=(floor(coord)+0.5)/(u_terrain_dim+2.0);float d=1.0/(u_terrain_dim+2.0);float tl=ele(c);float tr=ele(c+vec2(d,0.0));float bl=ele(c+vec2(0.0,d));float br=ele(c+vec2(d,d));float elevation=mix(mix(tl,tr,f.x),mix(bl,br,f.x),f.y);return elevation*u_terrain_exaggeration;\n#else\nreturn 0.0;\n#endif\n}"),background:ne("uniform vec4 u_color;uniform float u_opacity;void main() {gl_FragColor=u_color*u_opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}"),backgroundPattern:ne("uniform vec2 u_pattern_tl_a;uniform vec2 u_pattern_br_a;uniform vec2 u_pattern_tl_b;uniform vec2 u_pattern_br_b;uniform vec2 u_texsize;uniform float u_mix;uniform float u_opacity;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(u_pattern_tl_a/u_texsize,u_pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(u_pattern_tl_b/u_texsize,u_pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_mix)*u_opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_pattern_size_a;uniform vec2 u_pattern_size_b;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_scale_a;uniform float u_scale_b;uniform float u_tile_units_to_pixels;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_a*u_pattern_size_a,u_tile_units_to_pixels,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_b*u_pattern_size_b,u_tile_units_to_pixels,a_pos);}"),circle:ne("varying vec3 v_data;varying float v_visibility;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define mediump float radius\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define highp vec4 stroke_color\n#pragma mapbox: define mediump float stroke_width\n#pragma mapbox: define lowp float stroke_opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize mediump float radius\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize highp vec4 stroke_color\n#pragma mapbox: initialize mediump float stroke_width\n#pragma mapbox: initialize lowp float stroke_opacity\nvec2 extrude=v_data.xy;float extrude_length=length(extrude);lowp float antialiasblur=v_data.z;float antialiased_blur=-max(blur,antialiasblur);float opacity_t=smoothstep(0.0,antialiased_blur,extrude_length-1.0);float color_t=stroke_width < 0.01 ? 0.0 : smoothstep(antialiased_blur,0.0,extrude_length-radius/(radius+stroke_width));gl_FragColor=v_visibility*opacity_t*mix(color*opacity,stroke_color*stroke_opacity,color_t);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform bool u_scale_with_map;uniform bool u_pitch_with_map;uniform vec2 u_extrude_scale;uniform lowp float u_device_pixel_ratio;uniform highp float u_camera_to_center_distance;attribute vec2 a_pos;varying vec3 v_data;varying float v_visibility;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define mediump float radius\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define highp vec4 stroke_color\n#pragma mapbox: define mediump float stroke_width\n#pragma mapbox: define lowp float stroke_opacity\nvoid main(void) {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize mediump float radius\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize highp vec4 stroke_color\n#pragma mapbox: initialize mediump float stroke_width\n#pragma mapbox: initialize lowp float stroke_opacity\nvec2 extrude=vec2(mod(a_pos,2.0)*2.0-1.0);vec2 circle_center=floor(a_pos*0.5);float ele=get_elevation(circle_center);v_visibility=calculate_visibility(u_matrix*vec4(circle_center,ele,1.0));if (u_pitch_with_map) {vec2 corner_position=circle_center;if (u_scale_with_map) {corner_position+=extrude*(radius+stroke_width)*u_extrude_scale;} else {vec4 projected_center=u_matrix*vec4(circle_center,0,1);corner_position+=extrude*(radius+stroke_width)*u_extrude_scale*(projected_center.w/u_camera_to_center_distance);}gl_Position=u_matrix*vec4(corner_position,ele,1);} else {gl_Position=u_matrix*vec4(circle_center,ele,1);if (u_scale_with_map) {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*u_camera_to_center_distance;} else {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*gl_Position.w;}}lowp float antialiasblur=1.0/u_device_pixel_ratio/(radius+stroke_width);v_data=vec3(extrude.x,extrude.y,antialiasblur);}"),clippingMask:ne("void main() {gl_FragColor=vec4(1.0);}","attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}"),heatmap:ne("uniform highp float u_intensity;varying vec2 v_extrude;\n#pragma mapbox: define highp float weight\n#define GAUSS_COEF 0.3989422804014327\nvoid main() {\n#pragma mapbox: initialize highp float weight\nfloat d=-0.5*3.0*3.0*dot(v_extrude,v_extrude);float val=weight*u_intensity*GAUSS_COEF*exp(d);gl_FragColor=vec4(val,1.0,1.0,1.0);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform float u_extrude_scale;uniform float u_opacity;uniform float u_intensity;attribute vec2 a_pos;varying vec2 v_extrude;\n#pragma mapbox: define highp float weight\n#pragma mapbox: define mediump float radius\nconst highp float ZERO=1.0/255.0/16.0;\n#define GAUSS_COEF 0.3989422804014327\nvoid main(void) {\n#pragma mapbox: initialize highp float weight\n#pragma mapbox: initialize mediump float radius\nvec2 unscaled_extrude=vec2(mod(a_pos,2.0)*2.0-1.0);float S=sqrt(-2.0*log(ZERO/weight/u_intensity/GAUSS_COEF))/3.0;v_extrude=S*unscaled_extrude;vec2 extrude=v_extrude*radius*u_extrude_scale;vec4 pos=vec4(floor(a_pos*0.5)+extrude,0,1);gl_Position=u_matrix*pos;}"),heatmapTexture:ne("uniform sampler2D u_image;uniform sampler2D u_color_ramp;uniform float u_opacity;varying vec2 v_pos;void main() {float t=texture2D(u_image,v_pos).r;vec4 color=texture2D(u_color_ramp,vec2(t,0.5));gl_FragColor=color*u_opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(0.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_world;attribute vec2 a_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos*u_world,0,1);v_pos.x=a_pos.x;v_pos.y=1.0-a_pos.y;}"),collisionBox:ne("varying float v_placed;varying float v_notUsed;void main() {float alpha=0.5;gl_FragColor=vec4(1.0,0.0,0.0,1.0)*alpha;if (v_placed > 0.5) {gl_FragColor=vec4(0.0,0.0,1.0,0.5)*alpha;}if (v_notUsed > 0.5) {gl_FragColor*=.1;}}","attribute vec2 a_pos;attribute vec2 a_anchor_pos;attribute vec2 a_extrude;attribute vec2 a_placed;attribute vec2 a_shift;uniform mat4 u_matrix;uniform vec2 u_extrude_scale;uniform float u_camera_to_center_distance;varying float v_placed;varying float v_notUsed;void main() {vec4 projectedPoint=u_matrix*vec4(a_anchor_pos,0,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);gl_Position=u_matrix*vec4(a_pos,get_elevation(a_pos),1.0);gl_Position.xy+=(a_extrude+a_shift)*u_extrude_scale*gl_Position.w*collision_perspective_ratio;v_placed=a_placed.x;v_notUsed=a_placed.y;}"),collisionCircle:ne("varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;void main() {float alpha=0.5*min(v_perspective_ratio,1.0);float stroke_radius=0.9*max(v_perspective_ratio,1.0);float distance_to_center=length(v_extrude);float distance_to_edge=abs(distance_to_center-v_radius);float opacity_t=smoothstep(-stroke_radius,0.0,-distance_to_edge);vec4 color=mix(vec4(0.0,0.0,1.0,0.5),vec4(1.0,0.0,0.0,1.0),v_collision);gl_FragColor=color*alpha*opacity_t;}","attribute vec2 a_pos;attribute float a_radius;attribute vec2 a_flags;uniform mat4 u_matrix;uniform mat4 u_inv_matrix;uniform vec2 u_viewport_size;uniform float u_camera_to_center_distance;varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;vec3 toTilePosition(vec2 screenPos) {vec4 rayStart=u_inv_matrix*vec4(screenPos,-1.0,1.0);vec4 rayEnd =u_inv_matrix*vec4(screenPos, 1.0,1.0);rayStart.xyz/=rayStart.w;rayEnd.xyz /=rayEnd.w;highp float t=(0.0-rayStart.z)/(rayEnd.z-rayStart.z);return mix(rayStart.xyz,rayEnd.xyz,t);}void main() {vec2 quadCenterPos=a_pos;float radius=a_radius;float collision=a_flags.x;float vertexIdx=a_flags.y;vec2 quadVertexOffset=vec2(mix(-1.0,1.0,float(vertexIdx >=2.0)),mix(-1.0,1.0,float(vertexIdx >=1.0 && vertexIdx <=2.0)));vec2 quadVertexExtent=quadVertexOffset*radius;vec3 tilePos=toTilePosition(quadCenterPos);vec4 clipPos=u_matrix*vec4(tilePos,1.0);highp float camera_to_anchor_distance=clipPos.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);float padding_factor=1.2;v_radius=radius;v_extrude=quadVertexExtent*padding_factor;v_perspective_ratio=collision_perspective_ratio;v_collision=collision;gl_Position=vec4(clipPos.xyz/clipPos.w,1.0)+vec4(quadVertexExtent*padding_factor/u_viewport_size*2.0,0.0,0.0);}"),debug:ne("uniform highp vec4 u_color;uniform sampler2D u_overlay;varying vec2 v_uv;void main() {vec4 overlay_color=texture2D(u_overlay,v_uv);gl_FragColor=mix(u_color,overlay_color,overlay_color.a);}","attribute vec2 a_pos;varying vec2 v_uv;uniform mat4 u_matrix;uniform float u_overlay_scale;void main() {v_uv=a_pos/8192.0;gl_Position=u_matrix*vec4(a_pos*u_overlay_scale,get_elevation(a_pos),1);}"),fill:ne("#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float opacity\ngl_FragColor=color*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","attribute vec2 a_pos;uniform mat4 u_matrix;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float opacity\ngl_Position=u_matrix*vec4(a_pos,0,1);}"),fillOutline:ne("varying vec2 v_pos;\n#pragma mapbox: define highp vec4 outline_color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 outline_color\n#pragma mapbox: initialize lowp float opacity\nfloat dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=outline_color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","attribute vec2 a_pos;uniform mat4 u_matrix;uniform vec2 u_world;varying vec2 v_pos;\n#pragma mapbox: define highp vec4 outline_color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 outline_color\n#pragma mapbox: initialize lowp float opacity\ngl_Position=u_matrix*vec4(a_pos,0,1);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}"),fillOutlinePattern:ne("uniform vec2 u_texsize;uniform sampler2D u_image;uniform float u_fade;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);float dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=mix(color1,color2,u_fade)*alpha*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_world;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;gl_Position=u_matrix*vec4(a_pos,0,1);vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,a_pos);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}"),fillPattern:ne("#ifdef GL_ES\nprecision highp float;\n#endif\nuniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_fade)*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileZoomRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileZoomRatio,a_pos);}"),fillExtrusion:ne("varying vec4 v_color;void main() {gl_FragColor=v_color;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;uniform float u_vertical_gradient;uniform lowp float u_opacity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\n#ifdef TERRAIN3D\nattribute vec2 a_centroid;\n#endif\nvarying vec4 v_color;\n#pragma mapbox: define highp float base\n#pragma mapbox: define highp float height\n#pragma mapbox: define highp vec4 color\nvoid main() {\n#pragma mapbox: initialize highp float base\n#pragma mapbox: initialize highp float height\n#pragma mapbox: initialize highp vec4 color\nvec3 normal=a_normal_ed.xyz;\n#ifdef TERRAIN3D\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\n#else\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\n#endif\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);gl_Position=u_matrix*vec4(a_pos,t > 0.0 ? height : base,1);float colorvalue=color.r*0.2126+color.g*0.7152+color.b*0.0722;v_color=vec4(0.0,0.0,0.0,1.0);vec4 ambientlight=vec4(0.03,0.03,0.03,1.0);color+=ambientlight;float directional=clamp(dot(normal/16384.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((1.0-colorvalue+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_color.r+=clamp(color.r*directional*u_lightcolor.r,mix(0.0,0.3,1.0-u_lightcolor.r),1.0);v_color.g+=clamp(color.g*directional*u_lightcolor.g,mix(0.0,0.3,1.0-u_lightcolor.g),1.0);v_color.b+=clamp(color.b*directional*u_lightcolor.b,mix(0.0,0.3,1.0-u_lightcolor.b),1.0);v_color*=u_opacity;}"),fillExtrusionPattern:ne("uniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\n#pragma mapbox: define lowp float base\n#pragma mapbox: define lowp float height\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float base\n#pragma mapbox: initialize lowp float height\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);vec4 mixedColor=mix(color1,color2,u_fade);gl_FragColor=mixedColor*v_lighting;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_height_factor;uniform vec3 u_scale;uniform float u_vertical_gradient;uniform lowp float u_opacity;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\n#ifdef TERRAIN3D\nattribute vec2 a_centroid;\n#endif\nvarying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\n#pragma mapbox: define lowp float base\n#pragma mapbox: define lowp float height\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float base\n#pragma mapbox: initialize lowp float height\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec3 normal=a_normal_ed.xyz;float edgedistance=a_normal_ed.w;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;\n#ifdef TERRAIN3D\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\n#else\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\n#endif\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);float z=t > 0.0 ? height : base;gl_Position=u_matrix*vec4(a_pos,z,1);vec2 pos=normal.x==1.0 && normal.y==0.0 && normal.z==16384.0\n? a_pos\n: vec2(edgedistance,z*u_height_factor);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,pos);v_lighting=vec4(0.0,0.0,0.0,1.0);float directional=clamp(dot(normal/16383.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((0.5+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_lighting.rgb+=clamp(directional*u_lightcolor,mix(vec3(0.0),vec3(0.3),1.0-u_lightcolor),vec3(1.0));v_lighting*=u_opacity;}"),hillshadePrepare:ne("#ifdef GL_ES\nprecision highp float;\n#endif\nuniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_dimension;uniform float u_zoom;uniform vec4 u_unpack;float getElevation(vec2 coord,float bias) {vec4 data=texture2D(u_image,coord)*255.0;data.a=-1.0;return dot(data,u_unpack)/4.0;}void main() {vec2 epsilon=1.0/u_dimension;float a=getElevation(v_pos+vec2(-epsilon.x,-epsilon.y),0.0);float b=getElevation(v_pos+vec2(0,-epsilon.y),0.0);float c=getElevation(v_pos+vec2(epsilon.x,-epsilon.y),0.0);float d=getElevation(v_pos+vec2(-epsilon.x,0),0.0);float e=getElevation(v_pos,0.0);float f=getElevation(v_pos+vec2(epsilon.x,0),0.0);float g=getElevation(v_pos+vec2(-epsilon.x,epsilon.y),0.0);float h=getElevation(v_pos+vec2(0,epsilon.y),0.0);float i=getElevation(v_pos+vec2(epsilon.x,epsilon.y),0.0);float exaggerationFactor=u_zoom < 2.0 ? 0.4 : u_zoom < 4.5 ? 0.35 : 0.3;float exaggeration=u_zoom < 15.0 ? (u_zoom-15.0)*exaggerationFactor : 0.0;vec2 deriv=vec2((c+f+f+i)-(a+d+d+g),(g+h+h+i)-(a+b+b+c))/pow(2.0,exaggeration+(19.2562-u_zoom));gl_FragColor=clamp(vec4(deriv.x/2.0+0.5,deriv.y/2.0+0.5,1.0,1.0),0.0,1.0);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_dimension;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);highp vec2 epsilon=1.0/u_dimension;float scale=(u_dimension.x-2.0)/u_dimension.x;v_pos=(a_texture_pos/8192.0)*scale+epsilon;}"),hillshade:ne("uniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_latrange;uniform vec2 u_light;uniform vec4 u_shadow;uniform vec4 u_highlight;uniform vec4 u_accent;\n#define PI 3.141592653589793\nvoid main() {vec4 pixel=texture2D(u_image,v_pos);vec2 deriv=((pixel.rg*2.0)-1.0);float scaleFactor=cos(radians((u_latrange[0]-u_latrange[1])*(1.0-v_pos.y)+u_latrange[1]));float slope=atan(1.25*length(deriv)/scaleFactor);float aspect=deriv.x !=0.0 ? atan(deriv.y,-deriv.x) : PI/2.0*(deriv.y > 0.0 ? 1.0 :-1.0);float intensity=u_light.x;float azimuth=u_light.y+PI;float base=1.875-intensity*1.75;float maxValue=0.5*PI;float scaledSlope=intensity !=0.5 ? ((pow(base,slope)-1.0)/(pow(base,maxValue)-1.0))*maxValue : slope;float accent=cos(scaledSlope);vec4 accent_color=(1.0-accent)*u_accent*clamp(intensity*2.0,0.0,1.0);float shade=abs(mod((aspect+azimuth)/PI+0.5,2.0)-1.0);vec4 shade_color=mix(u_shadow,u_highlight,shade)*sin(scaledSlope)*clamp(intensity*2.0,0.0,1.0);gl_FragColor=accent_color*(1.0-shade_color.a)+shade_color;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos=a_texture_pos/8192.0;}"),line:ne("uniform lowp float u_device_pixel_ratio;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);gl_FragColor=color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","\n#define scale 0.015873016\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform vec2 u_units_to_pixels;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp float v_linesofar;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float width\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float width\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;v_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*2.0;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_width2=vec2(outset,inset);}"),lineGradient:ne("uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;varying highp vec2 v_uv;\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);vec4 color=texture2D(u_image,v_uv);gl_FragColor=color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","\n#define scale 0.015873016\nattribute vec2 a_pos_normal;attribute vec4 a_data;attribute float a_uv_x;attribute float a_split_index;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_units_to_pixels;uniform float u_image_height;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp vec2 v_uv;\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float width\nvoid main() {\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float width\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;highp float texel_height=1.0/u_image_height;highp float half_texel_height=0.5*texel_height;v_uv=vec2(a_uv_x,a_split_index*texel_height-half_texel_height);vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_width2=vec2(outset,inset);}"),linePattern:ne("#ifdef GL_ES\nprecision highp float;\n#endif\nuniform lowp float u_device_pixel_ratio;uniform vec2 u_texsize;uniform float u_fade;uniform mediump vec3 u_scale;uniform sampler2D u_image;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;vec2 pattern_size_a=vec2(display_size_a.x*fromScale/tileZoomRatio,display_size_a.y);vec2 pattern_size_b=vec2(display_size_b.x*toScale/tileZoomRatio,display_size_b.y);float aspect_a=display_size_a.y/v_width;float aspect_b=display_size_b.y/v_width;float dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float x_a=mod(v_linesofar/pattern_size_a.x*aspect_a,1.0);float x_b=mod(v_linesofar/pattern_size_b.x*aspect_b,1.0);float y=0.5*v_normal.y+0.5;vec2 texel_size=1.0/u_texsize;vec2 pos_a=mix(pattern_tl_a*texel_size-texel_size,pattern_br_a*texel_size+texel_size,vec2(x_a,y));vec2 pos_b=mix(pattern_tl_b*texel_size-texel_size,pattern_br_b*texel_size+texel_size,vec2(x_b,y));vec4 color=mix(texture2D(u_image,pos_a),texture2D(u_image,pos_b),u_fade);gl_FragColor=color*alpha*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","\n#define scale 0.015873016\n#define LINE_DISTANCE_SCALE 2.0\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform vec2 u_units_to_pixels;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define mediump float width\n#pragma mapbox: define lowp float floorwidth\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize mediump float width\n#pragma mapbox: initialize lowp float floorwidth\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_linesofar=a_linesofar;v_width2=vec2(outset,inset);v_width=floorwidth;}"),lineSDF:ne("uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;uniform float u_sdfgamma;uniform float u_mix;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float width\n#pragma mapbox: define lowp float floorwidth\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float width\n#pragma mapbox: initialize lowp float floorwidth\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float sdfdist_a=texture2D(u_image,v_tex_a).a;float sdfdist_b=texture2D(u_image,v_tex_b).a;float sdfdist=mix(sdfdist_a,sdfdist_b,u_mix);alpha*=smoothstep(0.5-u_sdfgamma/floorwidth,0.5+u_sdfgamma/floorwidth,sdfdist);gl_FragColor=color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","\n#define scale 0.015873016\n#define LINE_DISTANCE_SCALE 2.0\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_patternscale_a;uniform float u_tex_y_a;uniform vec2 u_patternscale_b;uniform float u_tex_y_b;uniform vec2 u_units_to_pixels;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float width\n#pragma mapbox: define lowp float floorwidth\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float width\n#pragma mapbox: initialize lowp float floorwidth\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_tex_a=vec2(a_linesofar*u_patternscale_a.x/floorwidth,normal.y*u_patternscale_a.y+u_tex_y_a);v_tex_b=vec2(a_linesofar*u_patternscale_b.x/floorwidth,normal.y*u_patternscale_b.y+u_tex_y_b);v_width2=vec2(outset,inset);}"),raster:ne("uniform float u_fade_t;uniform float u_opacity;uniform sampler2D u_image0;uniform sampler2D u_image1;varying vec2 v_pos0;varying vec2 v_pos1;uniform float u_brightness_low;uniform float u_brightness_high;uniform float u_saturation_factor;uniform float u_contrast_factor;uniform vec3 u_spin_weights;void main() {vec4 color0=texture2D(u_image0,v_pos0);vec4 color1=texture2D(u_image1,v_pos1);if (color0.a > 0.0) {color0.rgb=color0.rgb/color0.a;}if (color1.a > 0.0) {color1.rgb=color1.rgb/color1.a;}vec4 color=mix(color0,color1,u_fade_t);color.a*=u_opacity;vec3 rgb=color.rgb;rgb=vec3(dot(rgb,u_spin_weights.xyz),dot(rgb,u_spin_weights.zxy),dot(rgb,u_spin_weights.yzx));float average=(color.r+color.g+color.b)/3.0;rgb+=(average-rgb)*u_saturation_factor;rgb=(rgb-0.5)*u_contrast_factor+0.5;vec3 u_high_vec=vec3(u_brightness_low,u_brightness_low,u_brightness_low);vec3 u_low_vec=vec3(u_brightness_high,u_brightness_high,u_brightness_high);gl_FragColor=vec4(mix(u_high_vec,u_low_vec,rgb)*color.a,color.a);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","uniform mat4 u_matrix;uniform vec2 u_tl_parent;uniform float u_scale_parent;uniform float u_buffer_scale;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos0;varying vec2 v_pos1;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos0=(((a_texture_pos/8192.0)-0.5)/u_buffer_scale )+0.5;v_pos1=(v_pos0*u_scale_parent)+u_tl_parent;}"),symbolIcon:ne("uniform sampler2D u_texture;varying vec2 v_tex;varying float v_fade_opacity;\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\nlowp float alpha=opacity*v_fade_opacity;gl_FragColor=texture2D(u_texture,v_tex)*alpha;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform highp float u_camera_to_center_distance;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform float u_fade_change;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform vec2 u_texsize;varying vec2 v_tex;varying float v_fade_opacity;\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;vec2 a_minFontScale=a_pixeloffset.zw/256.0;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\ncamera_to_anchor_distance/u_camera_to_center_distance :\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*max(a_minFontScale,fontScale)+a_pxoffset/16.0),z,1.0);v_tex=a_tex/u_texsize;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float visibility=calculate_visibility(projectedPoint);v_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));}"),symbolSDF:ne("#define SDF_PX 8.0\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;uniform bool u_is_text;varying vec2 v_data0;varying vec3 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nfloat EDGE_GAMMA=0.105/u_device_pixel_ratio;vec2 tex=v_data0.xy;float gamma_scale=v_data1.x;float size=v_data1.y;float fade_opacity=v_data1[2];float fontScale=u_is_text ? size/24.0 : size;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float inner_edge=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);inner_edge=inner_edge+gamma*gamma_scale;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(inner_edge-gamma_scaled,inner_edge+gamma_scaled,dist);if (u_is_halo) {lowp float halo_edge=(6.0-halo_width/fontScale)/SDF_PX;alpha=min(smoothstep(halo_edge-gamma_scaled,halo_edge+gamma_scaled,dist),1.0-alpha);}gl_FragColor=color*(alpha*opacity*fade_opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;varying vec2 v_data0;varying vec3 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\ncamera_to_anchor_distance/u_camera_to_center_distance :\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale+a_pxoffset),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0=a_tex/u_texsize;v_data1=vec3(gamma_scale,size,interpolated_fade_opacity);}"),symbolTextAndIcon:ne("#define SDF_PX 8.0\n#define SDF 1.0\n#define ICON 0.0\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform sampler2D u_texture_icon;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;varying vec4 v_data0;varying vec4 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nfloat fade_opacity=v_data1[2];if (v_data1.w==ICON) {vec2 tex_icon=v_data0.zw;lowp float alpha=opacity*fade_opacity;gl_FragColor=texture2D(u_texture_icon,tex_icon)*alpha;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\nreturn;}vec2 tex=v_data0.xy;float EDGE_GAMMA=0.105/u_device_pixel_ratio;float gamma_scale=v_data1.x;float size=v_data1.y;float fontScale=size/24.0;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float buff=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);buff=(6.0-halo_width/fontScale)/SDF_PX;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(buff-gamma_scaled,buff+gamma_scaled,dist);gl_FragColor=color*(alpha*opacity*fade_opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}","const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;uniform vec2 u_texsize_icon;varying vec4 v_data0;varying vec4 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);float is_sdf=a_size[0]-2.0*a_size_min;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\ncamera_to_anchor_distance/u_camera_to_center_distance :\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=size/24.0;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0.xy=a_tex/u_texsize;v_data0.zw=a_tex/u_texsize_icon;v_data1=vec4(gamma_scale,size,interpolated_fade_opacity,is_sdf);}"),terrain:ne("uniform sampler2D u_texture;varying vec2 v_texture_pos;void main() {gl_FragColor=texture2D(u_texture,v_texture_pos);}",oe),terrainDepth:ne("varying float v_depth;const highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitMsk=vec4(0.,vec3(1./256.0));highp vec4 pack(highp float value) {highp vec4 comp=fract(value*bitSh);comp-=comp.xxyz*bitMsk;return comp;}void main() {gl_FragColor=pack(v_depth);}",oe),terrainCoords:ne("precision mediump float;uniform sampler2D u_texture;uniform float u_terrain_coords_id;varying vec2 v_texture_pos;void main() {vec4 rgba=texture2D(u_texture,v_texture_pos);gl_FragColor=vec4(rgba.r,rgba.g,rgba.b,u_terrain_coords_id);}",oe)};function ne(t,e){const i=/#pragma mapbox: ([\w]+) ([\w]+) ([\w]+) ([\w]+)/g,s=e.match(/attribute ([\w]+) ([\w]+)/g),a=t.match(/uniform ([\w]+) ([\w]+)([\s]*)([\w]*)/g),o=e.match(/uniform ([\w]+) ([\w]+)([\s]*)([\w]*)/g),r=o?o.concat(a):a,n={};return {fragmentSource:t=t.replace(i,((t,e,i,s,a)=>(n[a]=!0,"define"===e?`\n#ifndef HAS_UNIFORM_u_${a}\nvarying ${i} ${s} ${a};\n#else\nuniform ${i} ${s} u_${a};\n#endif\n`:`\n#ifdef HAS_UNIFORM_u_${a}\n ${i} ${s} ${a} = u_${a};\n#endif\n`))),vertexSource:e=e.replace(i,((t,e,i,s,a)=>{const o="float"===s?"vec2":"vec4",r=a.match(/color/)?"color":o;return n[a]?"define"===e?`\n#ifndef HAS_UNIFORM_u_${a}\nuniform lowp float u_${a}_t;\nattribute ${i} ${o} a_${a};\nvarying ${i} ${s} ${a};\n#else\nuniform ${i} ${s} u_${a};\n#endif\n`:"vec4"===r?`\n#ifndef HAS_UNIFORM_u_${a}\n ${a} = a_${a};\n#else\n ${i} ${s} ${a} = u_${a};\n#endif\n`:`\n#ifndef HAS_UNIFORM_u_${a}\n ${a} = unpack_mix_${r}(a_${a}, u_${a}_t);\n#else\n ${i} ${s} ${a} = u_${a};\n#endif\n`:"define"===e?`\n#ifndef HAS_UNIFORM_u_${a}\nuniform lowp float u_${a}_t;\nattribute ${i} ${o} a_${a};\n#else\nuniform ${i} ${s} u_${a};\n#endif\n`:"vec4"===r?`\n#ifndef HAS_UNIFORM_u_${a}\n ${i} ${s} ${a} = a_${a};\n#else\n ${i} ${s} ${a} = u_${a};\n#endif\n`:`\n#ifndef HAS_UNIFORM_u_${a}\n ${i} ${s} ${a} = unpack_mix_${r}(a_${a}, u_${a}_t);\n#else\n ${i} ${s} ${a} = u_${a};\n#endif\n`})),staticAttributes:s,staticUniforms:r}}class le{constructor(){this.boundProgram=null,this.boundLayoutVertexBuffer=null,this.boundPaintVertexBuffers=[],this.boundIndexBuffer=null,this.boundVertexOffset=null,this.boundDynamicVertexBuffer=null,this.vao=null;}bind(t,e,i,s,a,o,r,n,l){this.context=t;let h=this.boundPaintVertexBuffers.length!==s.length;for(let t=0;!h&&t({u_depth:new t.aL(e,i.u_depth),u_terrain:new t.aL(e,i.u_terrain),u_terrain_dim:new t.aM(e,i.u_terrain_dim),u_terrain_matrix:new t.aN(e,i.u_terrain_matrix),u_terrain_unpack:new t.aO(e,i.u_terrain_unpack),u_terrain_exaggeration:new t.aM(e,i.u_terrain_exaggeration)}))(e,b),this.binderUniforms=s?s.getUniforms(e,b):[];}draw(t,e,i,s,a,o,r,n,l,h,c,u,d,_,p,m,f,g){const v=t.gl;if(this.failedToCreate)return;if(t.program.set(this.program),t.setDepthMode(i),t.setStencilMode(s),t.setColorMode(a),t.setCullFace(o),n){t.activeTexture.set(v.TEXTURE2),v.bindTexture(v.TEXTURE_2D,n.depthTexture),t.activeTexture.set(v.TEXTURE3),v.bindTexture(v.TEXTURE_2D,n.texture);for(const t in this.terrainUniforms)this.terrainUniforms[t].set(n[t]);}for(const t in this.fixedUniforms)this.fixedUniforms[t].set(r[t]);p&&p.setUniforms(t,this.binderUniforms,d,{zoom:_});let x=0;switch(e){case v.LINES:x=2;break;case v.TRIANGLES:x=3;break;case v.LINE_STRIP:x=1;}for(const i of u.get()){const s=i.vaos||(i.vaos={});(s[l]||(s[l]=new le)).bind(t,this,h,p?p.getPaintVertexBuffers():[],c,i.vertexOffset,m,f,g),v.drawElements(e,i.primitiveLength*x,v.UNSIGNED_SHORT,i.primitiveOffset*x*2);}}}function ue(t,e,i){const s=1/St(i,1,e.transform.tileZoom),a=Math.pow(2,i.tileID.overscaledZ),o=i.tileSize*Math.pow(2,e.transform.tileZoom)/a,r=o*(i.tileID.canonical.x+i.tileID.wrap*a),n=o*i.tileID.canonical.y;return {u_image:0,u_texsize:i.imageAtlasTexture.size,u_scale:[s,t.fromScale,t.toScale],u_fade:t.t,u_pixel_coord_upper:[r>>16,n>>16],u_pixel_coord_lower:[65535&r,65535&n]}}const de=(e,i,s,a)=>{const o=i.style.light,r=o.properties.get("position"),n=[r.x,r.y,r.z],l=function(){var e=new t.A(9);return t.A!=Float32Array&&(e[1]=0,e[2]=0,e[3]=0,e[5]=0,e[6]=0,e[7]=0),e[0]=1,e[4]=1,e[8]=1,e}();"viewport"===o.properties.get("anchor")&&function(t,e){var i=Math.sin(e),s=Math.cos(e);t[0]=s,t[1]=i,t[2]=0,t[3]=-i,t[4]=s,t[5]=0,t[6]=0,t[7]=0,t[8]=1;}(l,-i.transform.angle),function(t,e,i){var s=e[0],a=e[1],o=e[2];t[0]=s*i[0]+a*i[3]+o*i[6],t[1]=s*i[1]+a*i[4]+o*i[7],t[2]=s*i[2]+a*i[5]+o*i[8];}(n,n,l);const h=o.properties.get("color");return {u_matrix:e,u_lightpos:n,u_lightintensity:o.properties.get("intensity"),u_lightcolor:[h.r,h.g,h.b],u_vertical_gradient:+s,u_opacity:a}},_e=(e,i,s,a,o,r,n)=>t.e(de(e,i,s,a),ue(r,i,n),{u_height_factor:-Math.pow(2,o.overscaledZ)/n.tileSize/8}),pe=t=>({u_matrix:t}),me=(e,i,s,a)=>t.e(pe(e),ue(s,i,a)),fe=(t,e)=>({u_matrix:t,u_world:e}),ge=(e,i,s,a,o)=>t.e(me(e,i,s,a),{u_world:o}),ve=(t,e,i,s)=>{const a=t.transform;let o,r;if("map"===s.paint.get("circle-pitch-alignment")){const t=St(i,1,a.zoom);o=!0,r=[t,t];}else o=!1,r=a.pixelsToGLUnits;return {u_camera_to_center_distance:a.cameraToCenterDistance,u_scale_with_map:+("map"===s.paint.get("circle-pitch-scale")),u_matrix:t.translatePosMatrix(e.posMatrix,i,s.paint.get("circle-translate"),s.paint.get("circle-translate-anchor")),u_pitch_with_map:+o,u_device_pixel_ratio:t.pixelRatio,u_extrude_scale:r}},xe=(t,e,i)=>{const s=St(i,1,e.zoom),a=Math.pow(2,e.zoom-i.tileID.overscaledZ),o=i.tileID.overscaleFactor();return {u_matrix:t,u_camera_to_center_distance:e.cameraToCenterDistance,u_pixels_to_tile_units:s,u_extrude_scale:[e.pixelsToGLUnits[0]/(s*a),e.pixelsToGLUnits[1]/(s*a)],u_overscale_factor:o}},ye=(t,e,i=1)=>({u_matrix:t,u_color:e,u_overlay:0,u_overlay_scale:i}),be=t=>({u_matrix:t}),we=(t,e,i,s)=>({u_matrix:t,u_extrude_scale:St(e,1,i),u_intensity:s});function Te(e,i){const s=Math.pow(2,i.canonical.z),a=i.canonical.y;return [new t.U(0,a/s).toLngLat().lat,new t.U(0,(a+1)/s).toLngLat().lat]}const Ie=(t,e,i,s)=>{const a=t.transform;return {u_matrix:De(t,e,i,s),u_ratio:1/St(e,1,a.zoom),u_device_pixel_ratio:t.pixelRatio,u_units_to_pixels:[1/a.pixelsToGLUnits[0],1/a.pixelsToGLUnits[1]]}},Ee=(e,i,s,a,o)=>t.e(Ie(e,i,s,o),{u_image:0,u_image_height:a}),Se=(t,e,i,s,a)=>{const o=t.transform,r=Pe(e,o);return {u_matrix:De(t,e,i,a),u_texsize:e.imageAtlasTexture.size,u_ratio:1/St(e,1,o.zoom),u_device_pixel_ratio:t.pixelRatio,u_image:0,u_scale:[r,s.fromScale,s.toScale],u_fade:s.t,u_units_to_pixels:[1/o.pixelsToGLUnits[0],1/o.pixelsToGLUnits[1]]}},Ce=(e,i,s,a,o,r)=>{const n=e.lineAtlas,l=Pe(i,e.transform),h="round"===s.layout.get("line-cap"),c=n.getDash(a.from,h),u=n.getDash(a.to,h),d=c.width*o.fromScale,_=u.width*o.toScale;return t.e(Ie(e,i,s,r),{u_patternscale_a:[l/d,-c.height/2],u_patternscale_b:[l/_,-u.height/2],u_sdfgamma:n.width/(256*Math.min(d,_)*e.pixelRatio)/2,u_image:0,u_tex_y_a:c.y,u_tex_y_b:u.y,u_mix:o.t})};function Pe(t,e){return 1/St(t,1,e.tileZoom)}function De(t,e,i,s){return t.translatePosMatrix(s?s.posMatrix:e.tileID.posMatrix,e,i.paint.get("line-translate"),i.paint.get("line-translate-anchor"))}const Me=(t,e,i,s,a)=>{return {u_matrix:t,u_tl_parent:e,u_scale_parent:i,u_buffer_scale:1,u_fade_t:s.mix,u_opacity:s.opacity*a.paint.get("raster-opacity"),u_image0:0,u_image1:1,u_brightness_low:a.paint.get("raster-brightness-min"),u_brightness_high:a.paint.get("raster-brightness-max"),u_saturation_factor:(r=a.paint.get("raster-saturation"),r>0?1-1/(1.001-r):-r),u_contrast_factor:(o=a.paint.get("raster-contrast"),o>0?1/(1-o):1+o),u_spin_weights:ze(a.paint.get("raster-hue-rotate"))};var o,r;};function ze(t){t*=Math.PI/180;const e=Math.sin(t),i=Math.cos(t);return [(2*i+1)/3,(-Math.sqrt(3)*e-i+1)/3,(Math.sqrt(3)*e-i+1)/3]}const Le=(t,e,i,s,a,o,r,n,l,h)=>{const c=a.transform;return {u_is_size_zoom_constant:+("constant"===t||"source"===t),u_is_size_feature_constant:+("constant"===t||"camera"===t),u_size_t:e?e.uSizeT:0,u_size:e?e.uSize:0,u_camera_to_center_distance:c.cameraToCenterDistance,u_pitch:c.pitch/360*2*Math.PI,u_rotate_symbol:+i,u_aspect_ratio:c.width/c.height,u_fade_change:a.options.fadeDuration?a.symbolFadeChange:1,u_matrix:o,u_label_plane_matrix:r,u_coord_matrix:n,u_is_text:+l,u_pitch_with_map:+s,u_texsize:h,u_texture:0}},Ae=(e,i,s,a,o,r,n,l,h,c,u)=>{const d=o.transform;return t.e(Le(e,i,s,a,o,r,n,l,h,c),{u_gamma_scale:a?Math.cos(d._pitch)*d.cameraToCenterDistance:1,u_device_pixel_ratio:o.pixelRatio,u_is_halo:+u})},Re=(e,i,s,a,o,r,n,l,h,c)=>t.e(Ae(e,i,s,a,o,r,n,l,!0,h,!0),{u_texsize_icon:c,u_texture_icon:1}),ke=(t,e,i)=>({u_matrix:t,u_opacity:e,u_color:i}),Fe=(e,i,s,a,o,r)=>t.e(function(t,e,i,s){const a=i.imageManager.getPattern(t.from.toString()),o=i.imageManager.getPattern(t.to.toString()),{width:r,height:n}=i.imageManager.getPixelSize(),l=Math.pow(2,s.tileID.overscaledZ),h=s.tileSize*Math.pow(2,i.transform.tileZoom)/l,c=h*(s.tileID.canonical.x+s.tileID.wrap*l),u=h*s.tileID.canonical.y;return {u_image:0,u_pattern_tl_a:a.tl,u_pattern_br_a:a.br,u_pattern_tl_b:o.tl,u_pattern_br_b:o.br,u_texsize:[r,n],u_mix:e.t,u_pattern_size_a:a.displaySize,u_pattern_size_b:o.displaySize,u_scale_a:e.fromScale,u_scale_b:e.toScale,u_tile_units_to_pixels:1/St(s,1,i.transform.tileZoom),u_pixel_coord_upper:[c>>16,u>>16],u_pixel_coord_lower:[65535&c,65535&u]}}(a,r,s,o),{u_matrix:e,u_opacity:i}),Be={fillExtrusion:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_lightpos:new t.aP(e,i.u_lightpos),u_lightintensity:new t.aM(e,i.u_lightintensity),u_lightcolor:new t.aP(e,i.u_lightcolor),u_vertical_gradient:new t.aM(e,i.u_vertical_gradient),u_opacity:new t.aM(e,i.u_opacity)}),fillExtrusionPattern:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_lightpos:new t.aP(e,i.u_lightpos),u_lightintensity:new t.aM(e,i.u_lightintensity),u_lightcolor:new t.aP(e,i.u_lightcolor),u_vertical_gradient:new t.aM(e,i.u_vertical_gradient),u_height_factor:new t.aM(e,i.u_height_factor),u_image:new t.aL(e,i.u_image),u_texsize:new t.aQ(e,i.u_texsize),u_pixel_coord_upper:new t.aQ(e,i.u_pixel_coord_upper),u_pixel_coord_lower:new t.aQ(e,i.u_pixel_coord_lower),u_scale:new t.aP(e,i.u_scale),u_fade:new t.aM(e,i.u_fade),u_opacity:new t.aM(e,i.u_opacity)}),fill:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix)}),fillPattern:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_image:new t.aL(e,i.u_image),u_texsize:new t.aQ(e,i.u_texsize),u_pixel_coord_upper:new t.aQ(e,i.u_pixel_coord_upper),u_pixel_coord_lower:new t.aQ(e,i.u_pixel_coord_lower),u_scale:new t.aP(e,i.u_scale),u_fade:new t.aM(e,i.u_fade)}),fillOutline:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_world:new t.aQ(e,i.u_world)}),fillOutlinePattern:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_world:new t.aQ(e,i.u_world),u_image:new t.aL(e,i.u_image),u_texsize:new t.aQ(e,i.u_texsize),u_pixel_coord_upper:new t.aQ(e,i.u_pixel_coord_upper),u_pixel_coord_lower:new t.aQ(e,i.u_pixel_coord_lower),u_scale:new t.aP(e,i.u_scale),u_fade:new t.aM(e,i.u_fade)}),circle:(e,i)=>({u_camera_to_center_distance:new t.aM(e,i.u_camera_to_center_distance),u_scale_with_map:new t.aL(e,i.u_scale_with_map),u_pitch_with_map:new t.aL(e,i.u_pitch_with_map),u_extrude_scale:new t.aQ(e,i.u_extrude_scale),u_device_pixel_ratio:new t.aM(e,i.u_device_pixel_ratio),u_matrix:new t.aN(e,i.u_matrix)}),collisionBox:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_camera_to_center_distance:new t.aM(e,i.u_camera_to_center_distance),u_pixels_to_tile_units:new t.aM(e,i.u_pixels_to_tile_units),u_extrude_scale:new t.aQ(e,i.u_extrude_scale),u_overscale_factor:new t.aM(e,i.u_overscale_factor)}),collisionCircle:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_inv_matrix:new t.aN(e,i.u_inv_matrix),u_camera_to_center_distance:new t.aM(e,i.u_camera_to_center_distance),u_viewport_size:new t.aQ(e,i.u_viewport_size)}),debug:(e,i)=>({u_color:new t.aR(e,i.u_color),u_matrix:new t.aN(e,i.u_matrix),u_overlay:new t.aL(e,i.u_overlay),u_overlay_scale:new t.aM(e,i.u_overlay_scale)}),clippingMask:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix)}),heatmap:(e,i)=>({u_extrude_scale:new t.aM(e,i.u_extrude_scale),u_intensity:new t.aM(e,i.u_intensity),u_matrix:new t.aN(e,i.u_matrix)}),heatmapTexture:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_world:new t.aQ(e,i.u_world),u_image:new t.aL(e,i.u_image),u_color_ramp:new t.aL(e,i.u_color_ramp),u_opacity:new t.aM(e,i.u_opacity)}),hillshade:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_image:new t.aL(e,i.u_image),u_latrange:new t.aQ(e,i.u_latrange),u_light:new t.aQ(e,i.u_light),u_shadow:new t.aR(e,i.u_shadow),u_highlight:new t.aR(e,i.u_highlight),u_accent:new t.aR(e,i.u_accent)}),hillshadePrepare:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_image:new t.aL(e,i.u_image),u_dimension:new t.aQ(e,i.u_dimension),u_zoom:new t.aM(e,i.u_zoom),u_unpack:new t.aO(e,i.u_unpack)}),line:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_ratio:new t.aM(e,i.u_ratio),u_device_pixel_ratio:new t.aM(e,i.u_device_pixel_ratio),u_units_to_pixels:new t.aQ(e,i.u_units_to_pixels)}),lineGradient:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_ratio:new t.aM(e,i.u_ratio),u_device_pixel_ratio:new t.aM(e,i.u_device_pixel_ratio),u_units_to_pixels:new t.aQ(e,i.u_units_to_pixels),u_image:new t.aL(e,i.u_image),u_image_height:new t.aM(e,i.u_image_height)}),linePattern:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_texsize:new t.aQ(e,i.u_texsize),u_ratio:new t.aM(e,i.u_ratio),u_device_pixel_ratio:new t.aM(e,i.u_device_pixel_ratio),u_image:new t.aL(e,i.u_image),u_units_to_pixels:new t.aQ(e,i.u_units_to_pixels),u_scale:new t.aP(e,i.u_scale),u_fade:new t.aM(e,i.u_fade)}),lineSDF:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_ratio:new t.aM(e,i.u_ratio),u_device_pixel_ratio:new t.aM(e,i.u_device_pixel_ratio),u_units_to_pixels:new t.aQ(e,i.u_units_to_pixels),u_patternscale_a:new t.aQ(e,i.u_patternscale_a),u_patternscale_b:new t.aQ(e,i.u_patternscale_b),u_sdfgamma:new t.aM(e,i.u_sdfgamma),u_image:new t.aL(e,i.u_image),u_tex_y_a:new t.aM(e,i.u_tex_y_a),u_tex_y_b:new t.aM(e,i.u_tex_y_b),u_mix:new t.aM(e,i.u_mix)}),raster:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_tl_parent:new t.aQ(e,i.u_tl_parent),u_scale_parent:new t.aM(e,i.u_scale_parent),u_buffer_scale:new t.aM(e,i.u_buffer_scale),u_fade_t:new t.aM(e,i.u_fade_t),u_opacity:new t.aM(e,i.u_opacity),u_image0:new t.aL(e,i.u_image0),u_image1:new t.aL(e,i.u_image1),u_brightness_low:new t.aM(e,i.u_brightness_low),u_brightness_high:new t.aM(e,i.u_brightness_high),u_saturation_factor:new t.aM(e,i.u_saturation_factor),u_contrast_factor:new t.aM(e,i.u_contrast_factor),u_spin_weights:new t.aP(e,i.u_spin_weights)}),symbolIcon:(e,i)=>({u_is_size_zoom_constant:new t.aL(e,i.u_is_size_zoom_constant),u_is_size_feature_constant:new t.aL(e,i.u_is_size_feature_constant),u_size_t:new t.aM(e,i.u_size_t),u_size:new t.aM(e,i.u_size),u_camera_to_center_distance:new t.aM(e,i.u_camera_to_center_distance),u_pitch:new t.aM(e,i.u_pitch),u_rotate_symbol:new t.aL(e,i.u_rotate_symbol),u_aspect_ratio:new t.aM(e,i.u_aspect_ratio),u_fade_change:new t.aM(e,i.u_fade_change),u_matrix:new t.aN(e,i.u_matrix),u_label_plane_matrix:new t.aN(e,i.u_label_plane_matrix),u_coord_matrix:new t.aN(e,i.u_coord_matrix),u_is_text:new t.aL(e,i.u_is_text),u_pitch_with_map:new t.aL(e,i.u_pitch_with_map),u_texsize:new t.aQ(e,i.u_texsize),u_texture:new t.aL(e,i.u_texture)}),symbolSDF:(e,i)=>({u_is_size_zoom_constant:new t.aL(e,i.u_is_size_zoom_constant),u_is_size_feature_constant:new t.aL(e,i.u_is_size_feature_constant),u_size_t:new t.aM(e,i.u_size_t),u_size:new t.aM(e,i.u_size),u_camera_to_center_distance:new t.aM(e,i.u_camera_to_center_distance),u_pitch:new t.aM(e,i.u_pitch),u_rotate_symbol:new t.aL(e,i.u_rotate_symbol),u_aspect_ratio:new t.aM(e,i.u_aspect_ratio),u_fade_change:new t.aM(e,i.u_fade_change),u_matrix:new t.aN(e,i.u_matrix),u_label_plane_matrix:new t.aN(e,i.u_label_plane_matrix),u_coord_matrix:new t.aN(e,i.u_coord_matrix),u_is_text:new t.aL(e,i.u_is_text),u_pitch_with_map:new t.aL(e,i.u_pitch_with_map),u_texsize:new t.aQ(e,i.u_texsize),u_texture:new t.aL(e,i.u_texture),u_gamma_scale:new t.aM(e,i.u_gamma_scale),u_device_pixel_ratio:new t.aM(e,i.u_device_pixel_ratio),u_is_halo:new t.aL(e,i.u_is_halo)}),symbolTextAndIcon:(e,i)=>({u_is_size_zoom_constant:new t.aL(e,i.u_is_size_zoom_constant),u_is_size_feature_constant:new t.aL(e,i.u_is_size_feature_constant),u_size_t:new t.aM(e,i.u_size_t),u_size:new t.aM(e,i.u_size),u_camera_to_center_distance:new t.aM(e,i.u_camera_to_center_distance),u_pitch:new t.aM(e,i.u_pitch),u_rotate_symbol:new t.aL(e,i.u_rotate_symbol),u_aspect_ratio:new t.aM(e,i.u_aspect_ratio),u_fade_change:new t.aM(e,i.u_fade_change),u_matrix:new t.aN(e,i.u_matrix),u_label_plane_matrix:new t.aN(e,i.u_label_plane_matrix),u_coord_matrix:new t.aN(e,i.u_coord_matrix),u_is_text:new t.aL(e,i.u_is_text),u_pitch_with_map:new t.aL(e,i.u_pitch_with_map),u_texsize:new t.aQ(e,i.u_texsize),u_texsize_icon:new t.aQ(e,i.u_texsize_icon),u_texture:new t.aL(e,i.u_texture),u_texture_icon:new t.aL(e,i.u_texture_icon),u_gamma_scale:new t.aM(e,i.u_gamma_scale),u_device_pixel_ratio:new t.aM(e,i.u_device_pixel_ratio),u_is_halo:new t.aL(e,i.u_is_halo)}),background:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_opacity:new t.aM(e,i.u_opacity),u_color:new t.aR(e,i.u_color)}),backgroundPattern:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_opacity:new t.aM(e,i.u_opacity),u_image:new t.aL(e,i.u_image),u_pattern_tl_a:new t.aQ(e,i.u_pattern_tl_a),u_pattern_br_a:new t.aQ(e,i.u_pattern_br_a),u_pattern_tl_b:new t.aQ(e,i.u_pattern_tl_b),u_pattern_br_b:new t.aQ(e,i.u_pattern_br_b),u_texsize:new t.aQ(e,i.u_texsize),u_mix:new t.aM(e,i.u_mix),u_pattern_size_a:new t.aQ(e,i.u_pattern_size_a),u_pattern_size_b:new t.aQ(e,i.u_pattern_size_b),u_scale_a:new t.aM(e,i.u_scale_a),u_scale_b:new t.aM(e,i.u_scale_b),u_pixel_coord_upper:new t.aQ(e,i.u_pixel_coord_upper),u_pixel_coord_lower:new t.aQ(e,i.u_pixel_coord_lower),u_tile_units_to_pixels:new t.aM(e,i.u_tile_units_to_pixels)}),terrain:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_texture:new t.aL(e,i.u_texture),u_ele_delta:new t.aM(e,i.u_ele_delta)}),terrainDepth:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_ele_delta:new t.aM(e,i.u_ele_delta)}),terrainCoords:(e,i)=>({u_matrix:new t.aN(e,i.u_matrix),u_texture:new t.aL(e,i.u_texture),u_terrain_coords_id:new t.aM(e,i.u_terrain_coords_id),u_ele_delta:new t.aM(e,i.u_ele_delta)})};class Oe{constructor(t,e,i){this.context=t;const s=t.gl;this.buffer=s.createBuffer(),this.dynamicDraw=Boolean(i),this.context.unbindVAO(),t.bindElementBuffer.set(this.buffer),s.bufferData(s.ELEMENT_ARRAY_BUFFER,e.arrayBuffer,this.dynamicDraw?s.DYNAMIC_DRAW:s.STATIC_DRAW),this.dynamicDraw||delete e.arrayBuffer;}bind(){this.context.bindElementBuffer.set(this.buffer);}updateData(t){const e=this.context.gl;if(!this.dynamicDraw)throw new Error("Attempted to update data while not in dynamic mode.");this.context.unbindVAO(),this.bind(),e.bufferSubData(e.ELEMENT_ARRAY_BUFFER,0,t.arrayBuffer);}destroy(){this.buffer&&(this.context.gl.deleteBuffer(this.buffer),delete this.buffer);}}const Ne={Int8:"BYTE",Uint8:"UNSIGNED_BYTE",Int16:"SHORT",Uint16:"UNSIGNED_SHORT",Int32:"INT",Uint32:"UNSIGNED_INT",Float32:"FLOAT"};class Ue{constructor(t,e,i,s){this.length=e.length,this.attributes=i,this.itemSize=e.bytesPerElement,this.dynamicDraw=s,this.context=t;const a=t.gl;this.buffer=a.createBuffer(),t.bindVertexBuffer.set(this.buffer),a.bufferData(a.ARRAY_BUFFER,e.arrayBuffer,this.dynamicDraw?a.DYNAMIC_DRAW:a.STATIC_DRAW),this.dynamicDraw||delete e.arrayBuffer;}bind(){this.context.bindVertexBuffer.set(this.buffer);}updateData(t){if(t.length!==this.length)throw new Error(`Length of new data is ${t.length}, which doesn't match current length of ${this.length}`);const e=this.context.gl;this.bind(),e.bufferSubData(e.ARRAY_BUFFER,0,t.arrayBuffer);}enableAttributes(t,e){for(let i=0;i0){const i=t.Z(),s=v;t.aU(i,g.placementInvProjMatrix,e.transform.glCoordMatrix),t.aU(i,i,g.placementViewportMatrix),u.push({circleArray:y,circleOffset:_,transform:s,invTransform:i,coord:m}),d+=y.length/4,_=d;}x&&c.draw(l,h.LINES,Pi.disabled,Mi.disabled,e.colorModeForRenderPass(),zi.disabled,xe(v,e.transform,f),e.style.map.terrain&&e.style.map.terrain.getTerrainData(m),s.id,x.layoutVertexBuffer,x.indexBuffer,x.segments,null,e.transform.zoom,null,null,x.collisionVertexBuffer);}if(!n||!u.length)return;const p=e.useProgram("collisionCircle"),m=new t.aV;m.resize(4*d),m._trim();let f=0;for(const t of u)for(let e=0;e=0&&(m[f.associatedIconIndex]={shiftedAnchor:E,angle:S});}else wt(f.numGlyphs,_);}if(c){p.clear();const i=e.icon.placedSymbolArray;for(let e=0;ee.style.map.terrain.getElevation(l,t,i):null,i="map"===s.layout.get("text-rotation-alignment");dt(h,l.posMatrix,e,o,F,B,g,c,i,t);}const U=e.translatePosMatrix(l.posMatrix,a,r,n),Z=v||o&&T||N?Ri:F,G=e.translatePosMatrix(B,a,r,n,!0),j=_&&0!==s.paint.get(o?"text-halo-width":"icon-halo-width").constantOr(1);let V;V=_?h.iconsInText?Re(w.kind,C,x,g,e,U,Z,G,D,A):Ae(w.kind,C,x,g,e,U,Z,G,o,D,!0):Le(w.kind,C,x,g,e,U,Z,G,o,D);const q={program:S,buffers:u,uniformValues:V,atlasTexture:M,atlasTextureIcon:R,atlasInterpolation:z,atlasInterpolationIcon:L,isSDF:_,hasHalo:j};if(y&&h.canOverlap){b=!0;const e=u.segments.get();for(const i of e)I.push({segments:new t.S([i]),sortKey:i.sortKey,state:q,terrainData:P});}else I.push({segments:u.segments,sortKey:0,state:q,terrainData:P});}b&&I.sort(((t,e)=>t.sortKey-e.sortKey));for(const t of I){const i=t.state;if(_.activeTexture.set(p.TEXTURE0),i.atlasTexture.bind(i.atlasInterpolation,p.CLAMP_TO_EDGE),i.atlasTextureIcon&&(_.activeTexture.set(p.TEXTURE1),i.atlasTextureIcon&&i.atlasTextureIcon.bind(i.atlasInterpolationIcon,p.CLAMP_TO_EDGE)),i.isSDF){const a=i.uniformValues;i.hasHalo&&(a.u_is_halo=1,Ni(i.buffers,t.segments,s,e,i.program,w,u,d,a,t.terrainData)),a.u_is_halo=0;}Ni(i.buffers,t.segments,s,e,i.program,w,u,d,i.uniformValues,t.terrainData);}}function Ni(t,e,i,s,a,o,r,n,l,h){const c=s.context;a.draw(c,c.gl.TRIANGLES,o,r,n,zi.disabled,l,h,i.id,t.layoutVertexBuffer,t.indexBuffer,e,i.paint,s.transform.zoom,t.programConfigurations.get(i.id),t.dynamicLayoutVertexBuffer,t.opacityVertexBuffer);}function Ui(t,e,i,s,a){if(!i||!s||!s.imageAtlas)return;const o=s.imageAtlas.patternPositions;let r=o[i.to.toString()],n=o[i.from.toString()];if(!r&&n&&(r=n),!n&&r&&(n=r),!r||!n){const t=a.getPaintProperty(e);r=o[t],n=o[t];}r&&n&&t.setConstantPatternPositions(r,n);}function Zi(t,e,i,s,a,o,r){const n=t.context.gl,l="fill-pattern",h=i.paint.get(l),c=h&&h.constantOr(1),u=i.getCrossfadeParameters();let d,_,p,m,f;r?(_=c&&!i.getPaintProperty("fill-outline-color")?"fillOutlinePattern":"fillOutline",d=n.LINES):(_=c?"fillPattern":"fill",d=n.TRIANGLES);const g=h.constantOr(null);for(const h of s){const s=e.getTile(h);if(c&&!s.patternsLoaded())continue;const v=s.getBucket(i);if(!v)continue;const x=v.programConfigurations.get(i.id),y=t.useProgram(_,x),b=t.style.map.terrain&&t.style.map.terrain.getTerrainData(h);c&&(t.context.activeTexture.set(n.TEXTURE0),s.imageAtlasTexture.bind(n.LINEAR,n.CLAMP_TO_EDGE),x.updatePaintBuffers(u)),Ui(x,l,g,s,i);const w=b?h:null,T=t.translatePosMatrix(w?w.posMatrix:h.posMatrix,s,i.paint.get("fill-translate"),i.paint.get("fill-translate-anchor"));if(r){m=v.indexBuffer2,f=v.segments2;const e=[n.drawingBufferWidth,n.drawingBufferHeight];p="fillOutlinePattern"===_&&c?ge(T,t,u,s,e):fe(T,e);}else m=v.indexBuffer,f=v.segments,p=c?me(T,t,u,s):pe(T);y.draw(t.context,d,a,t.stencilModeForClipping(h),o,zi.disabled,p,b,i.id,v.layoutVertexBuffer,m,f,i.paint,t.transform.zoom,x);}}function Gi(t,e,i,s,a,o,r){const n=t.context,l=n.gl,h="fill-extrusion-pattern",c=i.paint.get(h),u=c.constantOr(1),d=i.getCrossfadeParameters(),_=i.paint.get("fill-extrusion-opacity"),p=c.constantOr(null);for(const c of s){const s=e.getTile(c),m=s.getBucket(i);if(!m)continue;const f=t.style.map.terrain&&t.style.map.terrain.getTerrainData(c),g=m.programConfigurations.get(i.id),v=t.useProgram(u?"fillExtrusionPattern":"fillExtrusion",g);u&&(t.context.activeTexture.set(l.TEXTURE0),s.imageAtlasTexture.bind(l.LINEAR,l.CLAMP_TO_EDGE),g.updatePaintBuffers(d)),Ui(g,h,p,s,i);const x=t.translatePosMatrix(c.posMatrix,s,i.paint.get("fill-extrusion-translate"),i.paint.get("fill-extrusion-translate-anchor")),y=i.paint.get("fill-extrusion-vertical-gradient"),b=u?_e(x,t,y,_,c,d,s):de(x,t,y,_);v.draw(n,n.gl.TRIANGLES,a,o,r,zi.backCCW,b,f,i.id,m.layoutVertexBuffer,m.indexBuffer,m.segments,i.paint,t.transform.zoom,g,t.style.map.terrain&&m.centroidVertexBuffer);}}function ji(t,e,i,s,a,o,r){const n=t.context,l=n.gl,h=i.fbo;if(!h)return;const c=t.useProgram("hillshade"),u=t.style.map.terrain&&t.style.map.terrain.getTerrainData(e);n.activeTexture.set(l.TEXTURE0),l.bindTexture(l.TEXTURE_2D,h.colorAttachment.get()),c.draw(n,l.TRIANGLES,a,o,r,zi.disabled,((t,e,i,s)=>{const a=i.paint.get("hillshade-shadow-color"),o=i.paint.get("hillshade-highlight-color"),r=i.paint.get("hillshade-accent-color");let n=i.paint.get("hillshade-illumination-direction")*(Math.PI/180);"viewport"===i.paint.get("hillshade-illumination-anchor")&&(n-=t.transform.angle);const l=!t.options.moving;return {u_matrix:s?s.posMatrix:t.transform.calculatePosMatrix(e.tileID.toUnwrapped(),l),u_image:0,u_latrange:Te(0,e.tileID),u_light:[i.paint.get("hillshade-exaggeration"),n],u_shadow:a,u_highlight:o,u_accent:r}})(t,i,s,u?e:null),u,s.id,t.rasterBoundsBuffer,t.quadTriangleIndexBuffer,t.rasterBoundsSegments);}function Vi(e,i,s,a,o,r){const n=e.context,l=n.gl,h=i.dem;if(h&&h.data){const c=h.dim,u=h.stride,d=h.getPixels();if(n.activeTexture.set(l.TEXTURE1),n.pixelStoreUnpackPremultiplyAlpha.set(!1),i.demTexture=i.demTexture||e.getTileTexture(u),i.demTexture){const t=i.demTexture;t.update(d,{premultiply:!1}),t.bind(l.NEAREST,l.CLAMP_TO_EDGE);}else i.demTexture=new x(n,d,l.RGBA,{premultiply:!1}),i.demTexture.bind(l.NEAREST,l.CLAMP_TO_EDGE);n.activeTexture.set(l.TEXTURE0);let _=i.fbo;if(!_){const t=new x(n,{width:c,height:c,data:null},l.RGBA);t.bind(l.LINEAR,l.CLAMP_TO_EDGE),_=i.fbo=n.createFramebuffer(c,c,!0,!1),_.colorAttachment.set(t.texture);}n.bindFramebuffer.set(_.framebuffer),n.viewport.set([0,0,c,c]),e.useProgram("hillshadePrepare").draw(n,l.TRIANGLES,a,o,r,zi.disabled,((e,i)=>{const s=i.stride,a=t.Z();return t.aS(a,0,t.N,-t.N,0,0,1),t.$(a,a,[0,-t.N,0]),{u_matrix:a,u_image:1,u_dimension:[s,s],u_zoom:e.overscaledZ,u_unpack:i.getUnpackVector()}})(i.tileID,h),null,s.id,e.rasterBoundsBuffer,e.quadTriangleIndexBuffer,e.rasterBoundsSegments),i.needsHillshadePrepare=!1;}}function qi(e,i,s,a,o,r){const n=a.paint.get("raster-fade-duration");if(!r&&n>0){const a=t.h.now(),r=(a-e.timeAdded)/n,l=i?(a-i.timeAdded)/n:-1,h=s.getSource(),c=o.coveringZoomLevel({tileSize:h.tileSize,roundZoom:h.roundZoom}),u=!i||Math.abs(i.tileID.overscaledZ-c)>Math.abs(e.tileID.overscaledZ-c),d=u&&e.refreshedUponExpiration?1:t.ad(u?r:1-l,0,1);return e.refreshedUponExpiration&&r>=1&&(e.refreshedUponExpiration=!1),i?{opacity:1,mix:1-d}:{opacity:d,mix:0}}return {opacity:1,mix:0}}const $i=new t.aT(1,0,0,1),Wi=new t.aT(0,1,0,1),Hi=new t.aT(0,0,1,1),Xi=new t.aT(1,0,1,1),Ki=new t.aT(0,1,1,1);function Qi(t,e,i,s){Ji(t,0,e+i/2,t.transform.width,i,s);}function Yi(t,e,i,s){Ji(t,e-i/2,0,i,t.transform.height,s);}function Ji(t,e,i,s,a,o){const r=t.context,n=r.gl;n.enable(n.SCISSOR_TEST),n.scissor(e*t.pixelRatio,i*t.pixelRatio,s*t.pixelRatio,a*t.pixelRatio),r.clear({color:o}),n.disable(n.SCISSOR_TEST);}function ts(e,i,s){const a=e.context,o=a.gl,r=s.posMatrix,n=e.useProgram("debug"),l=Pi.disabled,h=Mi.disabled,c=e.colorModeForRenderPass(),u="$debug",d=e.style.map.terrain&&e.style.map.terrain.getTerrainData(s);a.activeTexture.set(o.TEXTURE0);const _=i.getTileByID(s.key).latestRawTileData,p=Math.floor((_&&_.byteLength||0)/1024),m=i.getTile(s).tileSize,f=512/Math.min(m,512)*(s.overscaledZ/e.transform.zoom)*.5;let g=s.canonical.toString();s.overscaledZ!==s.canonical.z&&(g+=` => ${s.overscaledZ}`),function(t,e){t.initDebugOverlayCanvas();const i=t.debugOverlayCanvas,s=t.context.gl,a=t.debugOverlayCanvas.getContext("2d");a.clearRect(0,0,i.width,i.height),a.shadowColor="white",a.shadowBlur=2,a.lineWidth=1.5,a.strokeStyle="white",a.textBaseline="top",a.font="bold 36px Open Sans, sans-serif",a.fillText(e,5,5),a.strokeText(e,5,5),t.debugOverlayTexture.update(i),t.debugOverlayTexture.bind(s.LINEAR,s.CLAMP_TO_EDGE);}(e,`${g} ${p}kB`),n.draw(a,o.TRIANGLES,l,h,Si.alphaBlended,zi.disabled,ye(r,t.aT.transparent,f),null,u,e.debugBuffer,e.quadTriangleIndexBuffer,e.debugSegments),n.draw(a,o.LINE_STRIP,l,h,c,zi.disabled,ye(r,t.aT.red),d,u,e.debugBuffer,e.tileBorderIndexBuffer,e.debugSegments);}function es(t,e,i){const s=t.context,a=s.gl,o=t.colorModeForRenderPass(),r=new Pi(a.LEQUAL,Pi.ReadWrite,t.depthRangeFor3D),n=t.useProgram("terrain"),l=e.getTerrainMesh();s.bindFramebuffer.set(null),s.viewport.set([0,0,t.width,t.height]);for(const h of i){const i=t.renderToTexture.getTexture(h),c=e.getTerrainData(h.tileID);s.activeTexture.set(a.TEXTURE0),a.bindTexture(a.TEXTURE_2D,i.texture);const u={u_matrix:t.transform.calculatePosMatrix(h.tileID.toUnwrapped()),u_texture:0,u_ele_delta:e.getMeshFrameDelta(t.transform.zoom)};n.draw(s,a.TRIANGLES,r,Mi.disabled,o,zi.backCCW,u,c,"terrain",l.vertexBuffer,l.indexBuffer,l.segments);}}class is{constructor(e,i){this.context=new Ci(e),this.transform=i,this._tileTextures={},this.terrainFacilitator={dirty:!0,matrix:t.Z(),renderTime:0},this.setup(),this.numSublayers=K.maxUnderzooming+K.maxOverzooming+1,this.depthEpsilon=1/Math.pow(2,16),this.crossTileSymbolIndex=new Yt;}resize(t,e,i){if(this.width=Math.floor(t*i),this.height=Math.floor(e*i),this.pixelRatio=i,this.context.viewport.set([0,0,this.width,this.height]),this.style)for(const t of this.style._order)this.style._layers[t].resize();}setup(){const e=this.context,i=new t.a_;i.emplaceBack(0,0),i.emplaceBack(t.N,0),i.emplaceBack(0,t.N),i.emplaceBack(t.N,t.N),this.tileExtentBuffer=e.createVertexBuffer(i,ae.members),this.tileExtentSegments=t.S.simpleSegment(0,0,4,2);const s=new t.a_;s.emplaceBack(0,0),s.emplaceBack(t.N,0),s.emplaceBack(0,t.N),s.emplaceBack(t.N,t.N),this.debugBuffer=e.createVertexBuffer(s,ae.members),this.debugSegments=t.S.simpleSegment(0,0,4,5);const a=new t.V;a.emplaceBack(0,0,0,0),a.emplaceBack(t.N,0,t.N,0),a.emplaceBack(0,t.N,0,t.N),a.emplaceBack(t.N,t.N,t.N,t.N),this.rasterBoundsBuffer=e.createVertexBuffer(a,O.members),this.rasterBoundsSegments=t.S.simpleSegment(0,0,4,2);const o=new t.a_;o.emplaceBack(0,0),o.emplaceBack(1,0),o.emplaceBack(0,1),o.emplaceBack(1,1),this.viewportBuffer=e.createVertexBuffer(o,ae.members),this.viewportSegments=t.S.simpleSegment(0,0,4,2);const r=new t.a$;r.emplaceBack(0),r.emplaceBack(1),r.emplaceBack(3),r.emplaceBack(2),r.emplaceBack(0),this.tileBorderIndexBuffer=e.createIndexBuffer(r);const n=new t.b0;n.emplaceBack(0,1,2),n.emplaceBack(2,1,3),this.quadTriangleIndexBuffer=e.createIndexBuffer(n);const l=this.context.gl;this.stencilClearMode=new Mi({func:l.ALWAYS,mask:0},0,255,l.ZERO,l.ZERO,l.ZERO);}clearStencil(){const e=this.context,i=e.gl;this.nextStencilID=1,this.currentStencilSource=void 0;const s=t.Z();t.aS(s,0,this.width,this.height,0,0,1),t.a0(s,s,[i.drawingBufferWidth,i.drawingBufferHeight,0]),this.useProgram("clippingMask").draw(e,i.TRIANGLES,Pi.disabled,this.stencilClearMode,Si.disabled,zi.disabled,be(s),null,"$clipping",this.viewportBuffer,this.quadTriangleIndexBuffer,this.viewportSegments);}_renderTileClippingMasks(t,e){if(this.currentStencilSource===t.source||!t.isTileClipped()||!e||!e.length)return;this.currentStencilSource=t.source;const i=this.context,s=i.gl;this.nextStencilID+e.length>256&&this.clearStencil(),i.setColorMode(Si.disabled),i.setDepthMode(Pi.disabled);const a=this.useProgram("clippingMask");this._tileClippingMaskIDs={};for(const t of e){const e=this._tileClippingMaskIDs[t.key]=this.nextStencilID++,o=this.style.map.terrain&&this.style.map.terrain.getTerrainData(t);a.draw(i,s.TRIANGLES,Pi.disabled,new Mi({func:s.ALWAYS,mask:0},e,255,s.KEEP,s.KEEP,s.REPLACE),Si.disabled,zi.disabled,be(t.posMatrix),o,"$clipping",this.tileExtentBuffer,this.quadTriangleIndexBuffer,this.tileExtentSegments);}}stencilModeFor3D(){this.currentStencilSource=void 0,this.nextStencilID+1>256&&this.clearStencil();const t=this.nextStencilID++,e=this.context.gl;return new Mi({func:e.NOTEQUAL,mask:255},t,255,e.KEEP,e.KEEP,e.REPLACE)}stencilModeForClipping(t){const e=this.context.gl;return new Mi({func:e.EQUAL,mask:255},this._tileClippingMaskIDs[t.key],0,e.KEEP,e.KEEP,e.REPLACE)}stencilConfigForOverlap(t){const e=this.context.gl,i=t.sort(((t,e)=>e.overscaledZ-t.overscaledZ)),s=i[i.length-1].overscaledZ,a=i[0].overscaledZ-s+1;if(a>1){this.currentStencilSource=void 0,this.nextStencilID+a>256&&this.clearStencil();const t={};for(let i=0;i=0;this.currentLayer--){const t=this.style._layers[s[this.currentLayer]],e=a[t.source],i=o[t.source];this._renderTileClippingMasks(t,i),this.renderLayer(this,e,t,i);}for(this.renderPass="translucent",this.currentLayer=0;this.currentLayeri.source&&!i.isHidden(e)?[t.sourceCaches[i.source]]:[])),a=s.filter((t=>"vector"===t.getSource().type)),o=s.filter((t=>"vector"!==t.getSource().type)),r=t=>{(!i||i.getSource().maxzoomr(t))),i||o.forEach((t=>r(t))),i}(this.style,this.transform.zoom);t&&function(t,e,i){for(let s=0;si.style.map.terrain.getElevation(o,t,e):null);}}}(a,e,s,i,s.layout.get("text-rotation-alignment"),s.layout.get("text-pitch-alignment"),o),0!==s.paint.get("icon-opacity").constantOr(1)&&Oi(e,i,s,a,!1,s.paint.get("icon-translate"),s.paint.get("icon-translate-anchor"),s.layout.get("icon-rotation-alignment"),s.layout.get("icon-pitch-alignment"),s.layout.get("icon-keep-upright"),r,n),0!==s.paint.get("text-opacity").constantOr(1)&&Oi(e,i,s,a,!0,s.paint.get("text-translate"),s.paint.get("text-translate-anchor"),s.layout.get("text-rotation-alignment"),s.layout.get("text-pitch-alignment"),s.layout.get("text-keep-upright"),r,n),i.map.showCollisionBoxes&&(Ai(e,i,s,a,s.paint.get("text-translate"),s.paint.get("text-translate-anchor"),!0),Ai(e,i,s,a,s.paint.get("icon-translate"),s.paint.get("icon-translate-anchor"),!1));}(e,i,s,a,this.style.placement.variableOffsets);break;case"circle":!function(e,i,s,a){if("translucent"!==e.renderPass)return;const o=s.paint.get("circle-opacity"),r=s.paint.get("circle-stroke-width"),n=s.paint.get("circle-stroke-opacity"),l=!s.layout.get("circle-sort-key").isConstant();if(0===o.constantOr(1)&&(0===r.constantOr(1)||0===n.constantOr(1)))return;const h=e.context,c=h.gl,u=e.depthModeForSublayer(0,Pi.ReadOnly),d=Mi.disabled,_=e.colorModeForRenderPass(),p=[];for(let o=0;ot.sortKey-e.sortKey));for(const t of p){const{programConfiguration:i,program:a,layoutVertexBuffer:o,indexBuffer:r,uniformValues:n,terrainData:l}=t.state;a.draw(h,c.TRIANGLES,u,d,_,zi.disabled,n,l,s.id,o,r,t.segments,s.paint,e.transform.zoom,i);}}(e,i,s,a);break;case"heatmap":!function(e,i,s,a){if(0!==s.paint.get("heatmap-opacity"))if("offscreen"===e.renderPass){const o=e.context,r=o.gl,n=Mi.disabled,l=new Si([r.ONE,r.ONE],t.aT.transparent,[!0,!0,!0,!0]);!function(t,e,i){const s=t.gl;t.activeTexture.set(s.TEXTURE1),t.viewport.set([0,0,e.width/4,e.height/4]);let a=i.heatmapFbo;if(a)s.bindTexture(s.TEXTURE_2D,a.colorAttachment.get()),t.bindFramebuffer.set(a.framebuffer);else {const o=s.createTexture();s.bindTexture(s.TEXTURE_2D,o),s.texParameteri(s.TEXTURE_2D,s.TEXTURE_WRAP_S,s.CLAMP_TO_EDGE),s.texParameteri(s.TEXTURE_2D,s.TEXTURE_WRAP_T,s.CLAMP_TO_EDGE),s.texParameteri(s.TEXTURE_2D,s.TEXTURE_MIN_FILTER,s.LINEAR),s.texParameteri(s.TEXTURE_2D,s.TEXTURE_MAG_FILTER,s.LINEAR),a=i.heatmapFbo=t.createFramebuffer(e.width/4,e.height/4,!1,!1),function(t,e,i,s){var a,o;const r=t.gl,n=null!==(a=t.HALF_FLOAT)&&void 0!==a?a:r.UNSIGNED_BYTE,l=null!==(o=t.RGBA16F)&&void 0!==o?o:r.RGBA;r.texImage2D(r.TEXTURE_2D,0,l,e.width/4,e.height/4,0,r.RGBA,n,null),s.colorAttachment.set(i);}(t,e,o,a);}}(o,e,s),o.clear({color:t.aT.transparent});for(let t=0;t{const o=t.Z();t.aS(o,0,e.width,e.height,0,0,1);const r=e.context.gl;return {u_matrix:o,u_world:[r.drawingBufferWidth,r.drawingBufferHeight],u_image:0,u_color_ramp:1,u_opacity:i.paint.get("heatmap-opacity")}})(e,i),null,i.id,e.viewportBuffer,e.quadTriangleIndexBuffer,e.viewportSegments,i.paint,e.transform.zoom);}(e,s));}(e,i,s,a);break;case"line":!function(e,i,s,a){if("translucent"!==e.renderPass)return;const o=s.paint.get("line-opacity"),r=s.paint.get("line-width");if(0===o.constantOr(1)||0===r.constantOr(1))return;const n=e.depthModeForSublayer(0,Pi.ReadOnly),l=e.colorModeForRenderPass(),h=s.paint.get("line-dasharray"),c=s.paint.get("line-pattern"),u=c.constantOr(1),d=s.paint.get("line-gradient"),_=s.getCrossfadeParameters(),p=u?"linePattern":h?"lineSDF":d?"lineGradient":"line",m=e.context,f=m.gl;let g=!0;for(const o of a){const a=i.getTile(o);if(u&&!a.patternsLoaded())continue;const r=a.getBucket(s);if(!r)continue;const v=r.programConfigurations.get(s.id),y=e.context.program.get(),b=e.useProgram(p,v),w=g||b.program!==y,T=e.style.map.terrain&&e.style.map.terrain.getTerrainData(o),I=c.constantOr(null);if(I&&a.imageAtlas){const t=a.imageAtlas,e=t.patternPositions[I.to.toString()],i=t.patternPositions[I.from.toString()];e&&i&&v.setConstantPatternPositions(e,i);}const E=T?o:null,S=u?Se(e,a,s,_,E):h?Ce(e,a,s,h,_,E):d?Ee(e,a,s,r.lineClipsArray.length,E):Ie(e,a,s,E);if(u)m.activeTexture.set(f.TEXTURE0),a.imageAtlasTexture.bind(f.LINEAR,f.CLAMP_TO_EDGE),v.updatePaintBuffers(_);else if(h&&(w||e.lineAtlas.dirty))m.activeTexture.set(f.TEXTURE0),e.lineAtlas.bind(m);else if(d){const a=r.gradients[s.id];let n=a.texture;if(s.gradientVersion!==a.version){let l=256;if(s.stepInterpolant){const s=i.getSource().maxzoom,a=o.canonical.z===s?Math.ceil(1<0?e.pop():null}isPatternMissing(t){if(!t)return !1;if(!t.from||!t.to)return !0;const e=this.imageManager.getPattern(t.from.toString()),i=this.imageManager.getPattern(t.to.toString());return !e||!i}useProgram(t,e){this.cache=this.cache||{};const i=t+(e?e.cacheKey:"")+(this._showOverdrawInspector?"/overdraw":"")+(this.style.map.terrain?"/terrain":"");return this.cache[i]||(this.cache[i]=new ce(this.context,re[t],e,Be[t],this._showOverdrawInspector,this.style.map.terrain)),this.cache[i]}setCustomLayerDefaults(){this.context.unbindVAO(),this.context.cullFace.setDefault(),this.context.activeTexture.setDefault(),this.context.pixelStoreUnpack.setDefault(),this.context.pixelStoreUnpackPremultiplyAlpha.setDefault(),this.context.pixelStoreUnpackFlipY.setDefault();}setBaseState(){const t=this.context.gl;this.context.cullFace.set(!1),this.context.viewport.set([0,0,this.width,this.height]),this.context.blendEquation.set(t.FUNC_ADD);}initDebugOverlayCanvas(){null==this.debugOverlayCanvas&&(this.debugOverlayCanvas=document.createElement("canvas"),this.debugOverlayCanvas.width=512,this.debugOverlayCanvas.height=512,this.debugOverlayTexture=new x(this.context,this.debugOverlayCanvas,this.context.gl.RGBA));}destroy(){this.debugOverlayTexture&&this.debugOverlayTexture.destroy();}overLimit(){const{drawingBufferWidth:t,drawingBufferHeight:e}=this.context.gl;return this.width!==t||this.height!==e}}class ss{constructor(t,e){this.points=t,this.planes=e;}static fromInvProjectionMatrix(e,i,s){const a=Math.pow(2,s),o=[[-1,1,-1,1],[1,1,-1,1],[1,-1,-1,1],[-1,-1,-1,1],[-1,1,1,1],[1,1,1,1],[1,-1,1,1],[-1,-1,1,1]].map((s=>{const o=1/(s=t.ag([],s,e))[3]/i*a;return t.b3(s,s,[o,o,1/s[3],o])})),r=[[0,1,2],[6,5,4],[0,3,7],[2,1,5],[3,2,6],[0,4,5]].map((t=>{const e=function(t,e){var i=e[0],s=e[1],a=e[2],o=i*i+s*s+a*a;return o>0&&(o=1/Math.sqrt(o)),t[0]=e[0]*o,t[1]=e[1]*o,t[2]=e[2]*o,t}([],function(t,e,i){var s=e[0],a=e[1],o=e[2],r=i[0],n=i[1],l=i[2];return t[0]=a*l-o*n,t[1]=o*r-s*l,t[2]=s*n-a*r,t}([],m([],o[t[0]],o[t[1]]),m([],o[t[2]],o[t[1]]))),i=-((s=e)[0]*(a=o[t[1]])[0]+s[1]*a[1]+s[2]*a[2]);var s,a;return e.concat(i)}));return new ss(o,r)}}class as{constructor(t,e){this.min=t,this.max=e,this.center=function(t,e,i){return t[0]=.5*e[0],t[1]=.5*e[1],t[2]=.5*e[2],t}([],function(t,e,i){return t[0]=e[0]+i[0],t[1]=e[1]+i[1],t[2]=e[2]+i[2],t}([],this.min,this.max));}quadrant(t){const e=[t%2==0,t<2],i=_(this.min),s=_(this.max);for(let t=0;t=0&&r++;if(0===r)return 0;r!==i.length&&(s=!1);}if(s)return 2;for(let t=0;t<3;t++){let i=Number.MAX_VALUE,s=-Number.MAX_VALUE;for(let a=0;athis.max[t]-this.min[t])return 0}return 1}}class os{constructor(t=0,e=0,i=0,s=0){if(isNaN(t)||t<0||isNaN(e)||e<0||isNaN(i)||i<0||isNaN(s)||s<0)throw new Error("Invalid value for edge-insets, top, bottom, left and right must all be numbers");this.top=t,this.bottom=e,this.left=i,this.right=s;}interpolate(e,i,s){return null!=i.top&&null!=e.top&&(this.top=t.B.number(e.top,i.top,s)),null!=i.bottom&&null!=e.bottom&&(this.bottom=t.B.number(e.bottom,i.bottom,s)),null!=i.left&&null!=e.left&&(this.left=t.B.number(e.left,i.left,s)),null!=i.right&&null!=e.right&&(this.right=t.B.number(e.right,i.right,s)),this}getCenter(e,i){const s=t.ad((this.left+e-this.right)/2,0,e),a=t.ad((this.top+i-this.bottom)/2,0,i);return new t.P(s,a)}equals(t){return this.top===t.top&&this.bottom===t.bottom&&this.left===t.left&&this.right===t.right}clone(){return new os(this.top,this.bottom,this.left,this.right)}toJSON(){return {top:this.top,bottom:this.bottom,left:this.left,right:this.right}}}class rs{constructor(e,i,s,a,o){this.tileSize=512,this.maxValidLatitude=85.051129,this._renderWorldCopies=void 0===o||!!o,this._minZoom=e||0,this._maxZoom=i||22,this._minPitch=null==s?0:s,this._maxPitch=null==a?60:a,this.setMaxBounds(),this.width=0,this.height=0,this._center=new t.L(0,0),this._elevation=0,this.zoom=0,this.angle=0,this._fov=.6435011087932844,this._pitch=0,this._unmodified=!0,this._edgeInsets=new os,this._posMatrixCache={},this._alignedPosMatrixCache={},this._minEleveationForCurrentTile=0;}clone(){const t=new rs(this._minZoom,this._maxZoom,this._minPitch,this.maxPitch,this._renderWorldCopies);return t.apply(this),t}apply(t){this.tileSize=t.tileSize,this.latRange=t.latRange,this.width=t.width,this.height=t.height,this._center=t._center,this._elevation=t._elevation,this._minEleveationForCurrentTile=t._minEleveationForCurrentTile,this.zoom=t.zoom,this.angle=t.angle,this._fov=t._fov,this._pitch=t._pitch,this._unmodified=t._unmodified,this._edgeInsets=t._edgeInsets.clone(),this._calcMatrices();}get minZoom(){return this._minZoom}set minZoom(t){this._minZoom!==t&&(this._minZoom=t,this.zoom=Math.max(this.zoom,t));}get maxZoom(){return this._maxZoom}set maxZoom(t){this._maxZoom!==t&&(this._maxZoom=t,this.zoom=Math.min(this.zoom,t));}get minPitch(){return this._minPitch}set minPitch(t){this._minPitch!==t&&(this._minPitch=t,this.pitch=Math.max(this.pitch,t));}get maxPitch(){return this._maxPitch}set maxPitch(t){this._maxPitch!==t&&(this._maxPitch=t,this.pitch=Math.min(this.pitch,t));}get renderWorldCopies(){return this._renderWorldCopies}set renderWorldCopies(t){void 0===t?t=!0:null===t&&(t=!1),this._renderWorldCopies=t;}get worldSize(){return this.tileSize*this.scale}get centerOffset(){return this.centerPoint._sub(this.size._div(2))}get size(){return new t.P(this.width,this.height)}get bearing(){return -this.angle/Math.PI*180}set bearing(e){const i=-t.b5(e,-180,180)*Math.PI/180;this.angle!==i&&(this._unmodified=!1,this.angle=i,this._calcMatrices(),this.rotationMatrix=function(){var e=new t.A(4);return t.A!=Float32Array&&(e[1]=0,e[2]=0),e[0]=1,e[3]=1,e}(),function(t,e,i){var s=e[0],a=e[1],o=e[2],r=e[3],n=Math.sin(i),l=Math.cos(i);t[0]=s*l+o*n,t[1]=a*l+r*n,t[2]=s*-n+o*l,t[3]=a*-n+r*l;}(this.rotationMatrix,this.rotationMatrix,this.angle));}get pitch(){return this._pitch/Math.PI*180}set pitch(e){const i=t.ad(e,this.minPitch,this.maxPitch)/180*Math.PI;this._pitch!==i&&(this._unmodified=!1,this._pitch=i,this._calcMatrices());}get fov(){return this._fov/Math.PI*180}set fov(t){t=Math.max(.01,Math.min(60,t)),this._fov!==t&&(this._unmodified=!1,this._fov=t/180*Math.PI,this._calcMatrices());}get zoom(){return this._zoom}set zoom(t){const e=Math.min(Math.max(t,this.minZoom),this.maxZoom);this._zoom!==e&&(this._unmodified=!1,this._zoom=e,this.tileZoom=Math.max(0,Math.floor(e)),this.scale=this.zoomScale(e),this._constrain(),this._calcMatrices());}get center(){return this._center}set center(t){t.lat===this._center.lat&&t.lng===this._center.lng||(this._unmodified=!1,this._center=t,this._constrain(),this._calcMatrices());}get elevation(){return this._elevation}set elevation(t){t!==this._elevation&&(this._elevation=t,this._constrain(),this._calcMatrices());}get padding(){return this._edgeInsets.toJSON()}set padding(t){this._edgeInsets.equals(t)||(this._unmodified=!1,this._edgeInsets.interpolate(this._edgeInsets,t,1),this._calcMatrices());}get centerPoint(){return this._edgeInsets.getCenter(this.width,this.height)}isPaddingEqual(t){return this._edgeInsets.equals(t)}interpolatePadding(t,e,i){this._unmodified=!1,this._edgeInsets.interpolate(t,e,i),this._constrain(),this._calcMatrices();}coveringZoomLevel(t){const e=(t.roundZoom?Math.round:Math.floor)(this.zoom+this.scaleZoom(this.tileSize/t.tileSize));return Math.max(0,e)}getVisibleUnwrappedCoordinates(e){const i=[new t.b6(0,e)];if(this._renderWorldCopies){const s=this.pointCoordinate(new t.P(0,0)),a=this.pointCoordinate(new t.P(this.width,0)),o=this.pointCoordinate(new t.P(this.width,this.height)),r=this.pointCoordinate(new t.P(0,this.height)),n=Math.floor(Math.min(s.x,a.x,o.x,r.x)),l=Math.floor(Math.max(s.x,a.x,o.x,r.x)),h=1;for(let s=n-h;s<=l+h;s++)0!==s&&i.push(new t.b6(s,e));}return i}coveringTiles(e){var i,s;let a=this.coveringZoomLevel(e);const o=a;if(void 0!==e.minzoom&&ae.maxzoom&&(a=e.maxzoom);const r=this.pointCoordinate(this.getCameraPoint()),n=t.U.fromLngLat(this.center),l=Math.pow(2,a),h=[l*r.x,l*r.y,0],c=[l*n.x,l*n.y,0],u=ss.fromInvProjectionMatrix(this.invProjMatrix,this.worldSize,a);let d=e.minzoom||0;!e.terrain&&this.pitch<=60&&this._edgeInsets.top<.1&&(d=a);const _=e.terrain?2/Math.min(this.tileSize,e.tileSize)*this.tileSize:3,p=t=>({aabb:new as([t*l,0,0],[(t+1)*l,l,0]),zoom:0,x:0,y:0,wrap:t,fullyVisible:!1}),m=[],g=[],v=a,x=e.reparseOverscaled?o:a;if(this._renderWorldCopies)for(let t=1;t<=3;t++)m.push(p(-t)),m.push(p(t));for(m.push(p(0));m.length>0;){const a=m.pop(),o=a.x,r=a.y;let n=a.fullyVisible;if(!n){const t=a.aabb.intersects(u);if(0===t)continue;n=2===t;}const l=e.terrain?h:c,p=a.aabb.distanceX(l),y=a.aabb.distanceY(l),b=Math.max(Math.abs(p),Math.abs(y));if(a.zoom===v||b>_+(1<=d){const e=v-a.zoom,i=h[0]-.5-(o<>1),u=a.zoom+1;let d=a.aabb.quadrant(l);if(e.terrain){const o=new t.O(u,a.wrap,u,h,c),r=e.terrain.getMinMaxElevation(o),n=null!==(i=r.minElevation)&&void 0!==i?i:this.elevation,l=null!==(s=r.maxElevation)&&void 0!==s?s:this.elevation;d=new as([d.min[0],d.min[1],n],[d.max[0],d.max[1],l]);}m.push({aabb:d,zoom:u,x:h,y:c,wrap:a.wrap,fullyVisible:n});}}return g.sort(((t,e)=>t.distanceSq-e.distanceSq)).map((t=>t.tileID))}resize(t,e){this.width=t,this.height=e,this.pixelsToGLUnits=[2/t,-2/e],this._constrain(),this._calcMatrices();}get unmodified(){return this._unmodified}zoomScale(t){return Math.pow(2,t)}scaleZoom(t){return Math.log(t)/Math.LN2}project(e){const i=t.ad(e.lat,-this.maxValidLatitude,this.maxValidLatitude);return new t.P(t.G(e.lng)*this.worldSize,t.H(i)*this.worldSize)}unproject(e){return new t.U(e.x/this.worldSize,e.y/this.worldSize).toLngLat()}get point(){return this.project(this.center)}getCameraPosition(){return {lngLat:this.pointLocation(this.getCameraPoint()),altitude:Math.cos(this._pitch)*this.cameraToCenterDistance/this._pixelPerMeter+this.elevation}}recalculateZoom(e){const i=this.pointLocation(this.centerPoint,e),s=e.getElevationForLngLatZoom(i,this.tileZoom);if(!(this.elevation-s))return;const a=this.getCameraPosition(),o=t.U.fromLngLat(a.lngLat,a.altitude),r=t.U.fromLngLat(i,s),n=o.x-r.x,l=o.y-r.y,h=o.z-r.z,c=Math.sqrt(n*n+l*l+h*h),u=this.scaleZoom(this.cameraToCenterDistance/c/this.tileSize);this._elevation=s,this._center=i,this.zoom=u;}setLocationAtPoint(e,i){const s=this.pointCoordinate(i),a=this.pointCoordinate(this.centerPoint),o=this.locationCoordinate(e),r=new t.U(o.x-(s.x-a.x),o.y-(s.y-a.y));this.center=this.coordinateLocation(r),this._renderWorldCopies&&(this.center=this.center.wrap());}locationPoint(t,e){return e?this.coordinatePoint(this.locationCoordinate(t),e.getElevationForLngLatZoom(t,this.tileZoom),this.pixelMatrix3D):this.coordinatePoint(this.locationCoordinate(t))}pointLocation(t,e){return this.coordinateLocation(this.pointCoordinate(t,e))}locationCoordinate(e){return t.U.fromLngLat(e)}coordinateLocation(t){return t&&t.toLngLat()}pointCoordinate(e,i){if(i){const t=i.pointCoordinate(e);if(null!=t)return t}const s=[e.x,e.y,0,1],a=[e.x,e.y,1,1];t.ag(s,s,this.pixelMatrixInverse),t.ag(a,a,this.pixelMatrixInverse);const o=s[3],r=a[3],n=s[1]/o,l=a[1]/r,h=s[2]/o,c=a[2]/r,u=h===c?0:(0-h)/(c-h);return new t.U(t.B.number(s[0]/o,a[0]/r,u)/this.worldSize,t.B.number(n,l,u)/this.worldSize)}coordinatePoint(e,i=0,s=this.pixelMatrix){const a=[e.x*this.worldSize,e.y*this.worldSize,i,1];return t.ag(a,a,s),new t.P(a[0]/a[3],a[1]/a[3])}getBounds(){const e=Math.max(0,this.height/2-this.getHorizon());return (new L).extend(this.pointLocation(new t.P(0,e))).extend(this.pointLocation(new t.P(this.width,e))).extend(this.pointLocation(new t.P(this.width,this.height))).extend(this.pointLocation(new t.P(0,this.height)))}getMaxBounds(){return this.latRange&&2===this.latRange.length&&this.lngRange&&2===this.lngRange.length?new L([this.lngRange[0],this.latRange[0]],[this.lngRange[1],this.latRange[1]]):null}getHorizon(){return Math.tan(Math.PI/2-this._pitch)*this.cameraToCenterDistance*.85}setMaxBounds(t){t?(this.lngRange=[t.getWest(),t.getEast()],this.latRange=[t.getSouth(),t.getNorth()],this._constrain()):(this.lngRange=null,this.latRange=[-this.maxValidLatitude,this.maxValidLatitude]);}calculatePosMatrix(e,i=!1){const s=e.key,a=i?this._alignedPosMatrixCache:this._posMatrixCache;if(a[s])return a[s];const o=e.canonical,r=this.worldSize/this.zoomScale(o.z),n=o.x+Math.pow(2,o.z)*e.wrap,l=t.ao(new Float64Array(16));return t.$(l,l,[n*r,o.y*r,0]),t.a0(l,l,[r/t.N,r/t.N,1]),t.a1(l,i?this.alignedProjMatrix:this.projMatrix,l),a[s]=new Float32Array(l),a[s]}customLayerMatrix(){return this.mercatorMatrix.slice()}_constrain(){if(!this.center||!this.width||!this.height||this._constraining)return;this._constraining=!0;let e,i,s,a,o=-90,r=90,n=-180,l=180;const h=this.size,c=this._unmodified;if(this.latRange){const i=this.latRange;o=t.H(i[1])*this.worldSize,r=t.H(i[0])*this.worldSize,e=r-or&&(a=r-e);}if(this.lngRange){const e=(n+l)/2,i=t.b5(u.x,e-this.worldSize/2,e+this.worldSize/2),a=h.x/2;i-al&&(s=l-a);}void 0===s&&void 0===a||(this.center=this.unproject(new t.P(void 0!==s?s:u.x,void 0!==a?a:u.y)).wrap()),this._unmodified=c,this._constraining=!1;}_calcMatrices(){if(!this.height)return;const e=this.centerOffset,i=this.point.x,s=this.point.y;this.cameraToCenterDistance=.5/Math.tan(this._fov/2)*this.height,this._pixelPerMeter=t.b7(1,this.center.lat)*this.worldSize;let a=t.ao(new Float64Array(16));t.a0(a,a,[this.width/2,-this.height/2,1]),t.$(a,a,[1,-1,0]),this.labelPlaneMatrix=a,a=t.ao(new Float64Array(16)),t.a0(a,a,[1,-1,1]),t.$(a,a,[-1,-1,0]),t.a0(a,a,[2/this.width,2/this.height,1]),this.glCoordMatrix=a;const o=this.cameraToCenterDistance+this._elevation*this._pixelPerMeter/Math.cos(this._pitch),r=Math.min(this.elevation,this._minEleveationForCurrentTile),n=o-r*this._pixelPerMeter/Math.cos(this._pitch),l=r<0?n:o,h=Math.PI/2+this._pitch,c=this._fov*(.5+e.y/this.height),u=Math.sin(c)*l/Math.sin(t.ad(Math.PI-h-c,.01,Math.PI-.01)),d=this.getHorizon(),_=2*Math.atan(d/this.cameraToCenterDistance)*(.5+e.y/(2*d)),p=Math.sin(_)*l/Math.sin(t.ad(Math.PI-h-_,.01,Math.PI-.01)),m=Math.min(u,p),f=1.01*(Math.cos(Math.PI/2-this._pitch)*m+l),g=this.height/50;a=new Float64Array(16),t.b8(a,this._fov,this.width/this.height,g,f),a[8]=2*-e.x/this.width,a[9]=2*e.y/this.height,t.a0(a,a,[1,-1,1]),t.$(a,a,[0,0,-this.cameraToCenterDistance]),t.b9(a,a,this._pitch),t.ae(a,a,this.angle),t.$(a,a,[-i,-s,0]),this.mercatorMatrix=t.a0([],a,[this.worldSize,this.worldSize,this.worldSize]),t.a0(a,a,[1,1,this._pixelPerMeter]),this.pixelMatrix=t.a1(new Float64Array(16),this.labelPlaneMatrix,a),t.$(a,a,[0,0,-this.elevation]),this.projMatrix=a,this.invProjMatrix=t.as([],a),this.pixelMatrix3D=t.a1(new Float64Array(16),this.labelPlaneMatrix,a);const v=this.width%2/2,x=this.height%2/2,y=Math.cos(this.angle),b=Math.sin(this.angle),w=i-Math.round(i)+y*v+b*x,T=s-Math.round(s)+y*x+b*v,I=new Float64Array(a);if(t.$(I,I,[w>.5?w-1:w,T>.5?T-1:T,0]),this.alignedProjMatrix=I,a=t.as(new Float64Array(16),this.pixelMatrix),!a)throw new Error("failed to invert matrix");this.pixelMatrixInverse=a,this._posMatrixCache={},this._alignedPosMatrixCache={};}maxPitchScaleFactor(){if(!this.pixelMatrixInverse)return 1;const e=this.pointCoordinate(new t.P(0,0)),i=[e.x*this.worldSize,e.y*this.worldSize,0,1];return t.ag(i,i,this.pixelMatrix)[3]/this.cameraToCenterDistance}getCameraPoint(){const e=Math.tan(this._pitch)*(this.cameraToCenterDistance||1);return this.centerPoint.add(new t.P(0,e))}getCameraQueryGeometry(e){const i=this.getCameraPoint();if(1===e.length)return [e[0],i];{let s=i.x,a=i.y,o=i.x,r=i.y;for(const t of e)s=Math.min(s,t.x),a=Math.min(a,t.y),o=Math.max(o,t.x),r=Math.max(r,t.y);return [new t.P(s,a),new t.P(o,a),new t.P(o,r),new t.P(s,r),new t.P(s,a)]}}}function ns(t,e){let i,s=!1,a=null,o=null;const r=()=>{a=null,s&&(t.apply(o,i),a=setTimeout(r,e),s=!1);};return (...t)=>(s=!0,o=this,i=t,a||r(),a)}class ls{constructor(t){this._getCurrentHash=()=>{const t=window.location.hash.replace("#","");if(this._hashName){let e;return t.split("&").map((t=>t.split("="))).forEach((t=>{t[0]===this._hashName&&(e=t);})),(e&&e[1]||"").split("/")}return t.split("/")},this._onHashChange=()=>{const t=this._getCurrentHash();if(t.length>=3&&!t.some((t=>isNaN(t)))){const e=this._map.dragRotate.isEnabled()&&this._map.touchZoomRotate.isEnabled()?+(t[3]||0):this._map.getBearing();return this._map.jumpTo({center:[+t[2],+t[1]],zoom:+t[0],bearing:e,pitch:+(t[4]||0)}),!0}return !1},this._updateHashUnthrottled=()=>{const t=window.location.href.replace(/(#.+)?$/,this.getHashString());try{window.history.replaceState(window.history.state,null,t);}catch(t){}},this._updateHash=ns(this._updateHashUnthrottled,300),this._hashName=t&&encodeURIComponent(t);}addTo(t){return this._map=t,addEventListener("hashchange",this._onHashChange,!1),this._map.on("moveend",this._updateHash),this}remove(){return removeEventListener("hashchange",this._onHashChange,!1),this._map.off("moveend",this._updateHash),clearTimeout(this._updateHash()),delete this._map,this}getHashString(t){const e=this._map.getCenter(),i=Math.round(100*this._map.getZoom())/100,s=Math.ceil((i*Math.LN2+Math.log(512/360/.5))/Math.LN10),a=Math.pow(10,s),o=Math.round(e.lng*a)/a,r=Math.round(e.lat*a)/a,n=this._map.getBearing(),l=this._map.getPitch();let h="";if(h+=t?`/${o}/${r}/${i}`:`${i}/${r}/${o}`,(n||l)&&(h+="/"+Math.round(10*n)/10),l&&(h+=`/${Math.round(l)}`),this._hashName){const t=this._hashName;let e=!1;const i=window.location.hash.slice(1).split("&").map((i=>{const s=i.split("=")[0];return s===t?(e=!0,`${s}=${h}`):i})).filter((t=>t));return e||i.push(`${t}=${h}`),`#${i.join("&")}`}return `#${h}`}}const hs={linearity:.3,easing:t.ba(0,0,.3,1)},cs=t.e({deceleration:2500,maxSpeed:1400},hs),us=t.e({deceleration:20,maxSpeed:1400},hs),ds=t.e({deceleration:1e3,maxSpeed:360},hs),_s=t.e({deceleration:1e3,maxSpeed:90},hs);class ps{constructor(t){this._map=t,this.clear();}clear(){this._inertiaBuffer=[];}record(e){this._drainInertiaBuffer(),this._inertiaBuffer.push({time:t.h.now(),settings:e});}_drainInertiaBuffer(){const e=this._inertiaBuffer,i=t.h.now();for(;e.length>0&&i-e[0].time>160;)e.shift();}_onMoveEnd(e){if(this._drainInertiaBuffer(),this._inertiaBuffer.length<2)return;const i={zoom:0,bearing:0,pitch:0,pan:new t.P(0,0),pinchAround:void 0,around:void 0};for(const{settings:t}of this._inertiaBuffer)i.zoom+=t.zoomDelta||0,i.bearing+=t.bearingDelta||0,i.pitch+=t.pitchDelta||0,t.panDelta&&i.pan._add(t.panDelta),t.around&&(i.around=t.around),t.pinchAround&&(i.pinchAround=t.pinchAround);const s=this._inertiaBuffer[this._inertiaBuffer.length-1].time-this._inertiaBuffer[0].time,a={};if(i.pan.mag()){const o=fs(i.pan.mag(),s,t.e({},cs,e||{}));a.offset=i.pan.mult(o.amount/i.pan.mag()),a.center=this._map.transform.center,ms(a,o);}if(i.zoom){const t=fs(i.zoom,s,us);a.zoom=this._map.transform.zoom+t.amount,ms(a,t);}if(i.bearing){const e=fs(i.bearing,s,ds);a.bearing=this._map.transform.bearing+t.ad(e.amount,-179,179),ms(a,e);}if(i.pitch){const t=fs(i.pitch,s,_s);a.pitch=this._map.transform.pitch+t.amount,ms(a,t);}if(a.zoom||a.bearing){const t=void 0===i.pinchAround?i.around:i.pinchAround;a.around=t?this._map.unproject(t):this._map.getCenter();}return this.clear(),t.e(a,{noMoveStart:!0})}}function ms(t,e){(!t.duration||t.durations.unproject(t))),l=r.reduce(((t,e,i,s)=>t.add(e.div(s.length))),new t.P(0,0));super(e,{points:r,point:l,lngLats:n,lngLat:s.unproject(l),originalEvent:a}),this._defaultPrevented=!1;}}class xs extends t.k{preventDefault(){this._defaultPrevented=!0;}get defaultPrevented(){return this._defaultPrevented}constructor(t,e,i){super(t,{originalEvent:i}),this._defaultPrevented=!1;}}class ys{constructor(t,e){this._map=t,this._clickTolerance=e.clickTolerance;}reset(){delete this._mousedownPos;}wheel(t){return this._firePreventable(new xs(t.type,this._map,t))}mousedown(t,e){return this._mousedownPos=e,this._firePreventable(new gs(t.type,this._map,t))}mouseup(t){this._map.fire(new gs(t.type,this._map,t));}click(t,e){this._mousedownPos&&this._mousedownPos.dist(e)>=this._clickTolerance||this._map.fire(new gs(t.type,this._map,t));}dblclick(t){return this._firePreventable(new gs(t.type,this._map,t))}mouseover(t){this._map.fire(new gs(t.type,this._map,t));}mouseout(t){this._map.fire(new gs(t.type,this._map,t));}touchstart(t){return this._firePreventable(new vs(t.type,this._map,t))}touchmove(t){this._map.fire(new vs(t.type,this._map,t));}touchend(t){this._map.fire(new vs(t.type,this._map,t));}touchcancel(t){this._map.fire(new vs(t.type,this._map,t));}_firePreventable(t){if(this._map.fire(t),t.defaultPrevented)return {}}isEnabled(){return !0}isActive(){return !1}enable(){}disable(){}}class bs{constructor(t){this._map=t;}reset(){this._delayContextMenu=!1,this._ignoreContextMenu=!0,delete this._contextMenuEvent;}mousemove(t){this._map.fire(new gs(t.type,this._map,t));}mousedown(){this._delayContextMenu=!0,this._ignoreContextMenu=!1;}mouseup(){this._delayContextMenu=!1,this._contextMenuEvent&&(this._map.fire(new gs("contextmenu",this._map,this._contextMenuEvent)),delete this._contextMenuEvent);}contextmenu(t){this._delayContextMenu?this._contextMenuEvent=t:this._ignoreContextMenu||this._map.fire(new gs(t.type,this._map,t)),this._map.listens("contextmenu")&&t.preventDefault();}isEnabled(){return !0}isActive(){return !1}enable(){}disable(){}}class ws{constructor(t){this._map=t;}get transform(){return this._map._requestedCameraState||this._map.transform}get center(){return {lng:this.transform.center.lng,lat:this.transform.center.lat}}get zoom(){return this.transform.zoom}get pitch(){return this.transform.pitch}get bearing(){return this.transform.bearing}unproject(e){return this.transform.pointLocation(t.P.convert(e),this._map.terrain)}}class Ts{constructor(t,e){this._map=t,this._tr=new ws(t),this._el=t.getCanvasContainer(),this._container=t.getContainer(),this._clickTolerance=e.clickTolerance||1;}isEnabled(){return !!this._enabled}isActive(){return !!this._active}enable(){this.isEnabled()||(this._enabled=!0);}disable(){this.isEnabled()&&(this._enabled=!1);}mousedown(t,e){this.isEnabled()&&t.shiftKey&&0===t.button&&(i.disableDrag(),this._startPos=this._lastPos=e,this._active=!0);}mousemoveWindow(t,e){if(!this._active)return;const s=e;if(this._lastPos.equals(s)||!this._box&&s.dist(this._startPos)t.fitScreenCoordinates(a,o,this._tr.bearing,{linear:!0})};this._fireEvent("boxzoomcancel",e);}keydown(t){this._active&&27===t.keyCode&&(this.reset(),this._fireEvent("boxzoomcancel",t));}reset(){this._active=!1,this._container.classList.remove("maplibregl-crosshair"),this._box&&(i.remove(this._box),this._box=null),i.enableDrag(),delete this._startPos,delete this._lastPos;}_fireEvent(e,i){return this._map.fire(new t.k(e,{originalEvent:i}))}}function Is(t,e){if(t.length!==e.length)throw new Error(`The number of touches and points are not equal - touches ${t.length}, points ${e.length}`);const i={};for(let s=0;sthis.numTouches)&&(this.aborted=!0),this.aborted||(void 0===this.startTime&&(this.startTime=e.timeStamp),s.length===this.numTouches&&(this.centroid=function(e){const i=new t.P(0,0);for(const t of e)i._add(t);return i.div(e.length)}(i),this.touches=Is(s,i)));}touchmove(t,e,i){if(this.aborted||!this.centroid)return;const s=Is(i,e);for(const t in this.touches){const e=s[t];(!e||e.dist(this.touches[t])>30)&&(this.aborted=!0);}}touchend(t,e,i){if((!this.centroid||t.timeStamp-this.startTime>500)&&(this.aborted=!0),0===i.length){const t=!this.aborted&&this.centroid;if(this.reset(),t)return t}}}class Ss{constructor(t){this.singleTap=new Es(t),this.numTaps=t.numTaps,this.reset();}reset(){this.lastTime=1/0,delete this.lastTap,this.count=0,this.singleTap.reset();}touchstart(t,e,i){this.singleTap.touchstart(t,e,i);}touchmove(t,e,i){this.singleTap.touchmove(t,e,i);}touchend(t,e,i){const s=this.singleTap.touchend(t,e,i);if(s){const e=t.timeStamp-this.lastTime<500,i=!this.lastTap||this.lastTap.dist(s)<30;if(e&&i||this.reset(),this.count++,this.lastTime=t.timeStamp,this.lastTap=s,this.count===this.numTaps)return this.reset(),s}}}class Cs{constructor(t){this._tr=new ws(t),this._zoomIn=new Ss({numTouches:1,numTaps:2}),this._zoomOut=new Ss({numTouches:2,numTaps:1}),this.reset();}reset(){this._active=!1,this._zoomIn.reset(),this._zoomOut.reset();}touchstart(t,e,i){this._zoomIn.touchstart(t,e,i),this._zoomOut.touchstart(t,e,i);}touchmove(t,e,i){this._zoomIn.touchmove(t,e,i),this._zoomOut.touchmove(t,e,i);}touchend(t,e,i){const s=this._zoomIn.touchend(t,e,i),a=this._zoomOut.touchend(t,e,i),o=this._tr;return s?(this._active=!0,t.preventDefault(),setTimeout((()=>this.reset()),0),{cameraAnimation:e=>e.easeTo({duration:300,zoom:o.zoom+1,around:o.unproject(s)},{originalEvent:t})}):a?(this._active=!0,t.preventDefault(),setTimeout((()=>this.reset()),0),{cameraAnimation:e=>e.easeTo({duration:300,zoom:o.zoom-1,around:o.unproject(a)},{originalEvent:t})}):void 0}touchcancel(){this.reset();}enable(){this._enabled=!0;}disable(){this._enabled=!1,this.reset();}isEnabled(){return this._enabled}isActive(){return this._active}}class Ps{constructor(t){this._enabled=!!t.enable,this._moveStateManager=t.moveStateManager,this._clickTolerance=t.clickTolerance||1,this._moveFunction=t.move,this._activateOnStart=!!t.activateOnStart,t.assignEvents(this),this.reset();}reset(t){this._active=!1,this._moved=!1,delete this._lastPoint,this._moveStateManager.endMove(t);}_move(...t){const e=this._moveFunction(...t);if(e.bearingDelta||e.pitchDelta||e.around||e.panDelta)return this._active=!0,e}dragStart(t,e){this.isEnabled()&&!this._lastPoint&&this._moveStateManager.isValidStartEvent(t)&&(this._moveStateManager.startMove(t),this._lastPoint=e.length?e[0]:e,this._activateOnStart&&this._lastPoint&&(this._active=!0));}dragMove(t,e){if(!this.isEnabled())return;const i=this._lastPoint;if(!i)return;if(t.preventDefault(),!this._moveStateManager.isValidMoveEvent(t))return void this.reset(t);const s=e.length?e[0]:e;return !this._moved&&s.dist(i){t.mousedown=t.dragStart,t.mousemoveWindow=t.dragMove,t.mouseup=t.dragEnd,t.contextmenu=function(t){t.preventDefault();};},As=({enable:t,clickTolerance:e,bearingDegreesPerPixelMoved:s=.8})=>{const a=new Ms({checkCorrectEvent:t=>0===i.mouseButton(t)&&t.ctrlKey||2===i.mouseButton(t)});return new Ps({clickTolerance:e,move:(t,e)=>({bearingDelta:(e.x-t.x)*s}),moveStateManager:a,enable:t,assignEvents:Ls})},Rs=({enable:t,clickTolerance:e,pitchDegreesPerPixelMoved:s=-.5})=>{const a=new Ms({checkCorrectEvent:t=>0===i.mouseButton(t)&&t.ctrlKey||2===i.mouseButton(t)});return new Ps({clickTolerance:e,move:(t,e)=>({pitchDelta:(e.y-t.y)*s}),moveStateManager:a,enable:t,assignEvents:Ls})};class ks{constructor(t,e){this._minTouches=t.cooperativeGestures?2:1,this._clickTolerance=t.clickTolerance||1,this._map=e,this.reset();}reset(){this._active=!1,this._touches={},this._sum=new t.P(0,0),setTimeout((()=>{this._cancelCooperativeMessage=!1;}),200);}touchstart(t,e,i){return this._calculateTransform(t,e,i)}touchmove(t,e,i){if(this._map._cooperativeGestures&&(2===this._minTouches&&i.length<2&&!this._cancelCooperativeMessage?this._map._onCooperativeGesture(t,!1,i.length):this._cancelCooperativeMessage||(this._cancelCooperativeMessage=!0)),this._active&&!(i.length0&&(this._active=!0);const a=Is(s,i),o=new t.P(0,0),r=new t.P(0,0);let n=0;for(const t in a){const e=a[t],i=this._touches[t];i&&(o._add(e),r._add(e.sub(i)),n++,a[t]=e);}if(this._touches=a,nMath.abs(t.x)}class js extends Fs{constructor(t){super(),this._map=t;}reset(){super.reset(),this._valid=void 0,delete this._firstMove,delete this._lastPoints;}touchstart(t,e,i){super.touchstart(t,e,i),this._currentTouchCount=i.length;}_start(t){this._lastPoints=t,Gs(t[0].sub(t[1]))&&(this._valid=!1);}_move(t,e,i){if(this._map._cooperativeGestures&&this._currentTouchCount<3)return;const s=t[0].sub(this._lastPoints[0]),a=t[1].sub(this._lastPoints[1]);return this._valid=this.gestureBeginsVertically(s,a,i.timeStamp),this._valid?(this._lastPoints=t,this._active=!0,{pitchDelta:(s.y+a.y)/2*-.5}):void 0}gestureBeginsVertically(t,e,i){if(void 0!==this._valid)return this._valid;const s=t.mag()>=2,a=e.mag()>=2;if(!s&&!a)return;if(!s||!a)return void 0===this._firstMove&&(this._firstMove=i),i-this._firstMove<100&&void 0;const o=t.y>0==e.y>0;return Gs(t)&&Gs(e)&&o}}const Vs={panStep:100,bearingStep:15,pitchStep:10};class qs{constructor(t){this._tr=new ws(t);const e=Vs;this._panStep=e.panStep,this._bearingStep=e.bearingStep,this._pitchStep=e.pitchStep,this._rotationDisabled=!1;}reset(){this._active=!1;}keydown(t){if(t.altKey||t.ctrlKey||t.metaKey)return;let e=0,i=0,s=0,a=0,o=0;switch(t.keyCode){case 61:case 107:case 171:case 187:e=1;break;case 189:case 109:case 173:e=-1;break;case 37:t.shiftKey?i=-1:(t.preventDefault(),a=-1);break;case 39:t.shiftKey?i=1:(t.preventDefault(),a=1);break;case 38:t.shiftKey?s=1:(t.preventDefault(),o=-1);break;case 40:t.shiftKey?s=-1:(t.preventDefault(),o=1);break;default:return}return this._rotationDisabled&&(i=0,s=0),{cameraAnimation:r=>{const n=this._tr;r.easeTo({duration:300,easeId:"keyboardHandler",easing:$s,zoom:e?Math.round(n.zoom)+e*(t.shiftKey?2:1):n.zoom,bearing:n.bearing+i*this._bearingStep,pitch:n.pitch+s*this._pitchStep,offset:[-a*this._panStep,-o*this._panStep],center:n.center},{originalEvent:t});}}}enable(){this._enabled=!0;}disable(){this._enabled=!1,this.reset();}isEnabled(){return this._enabled}isActive(){return this._active}disableRotation(){this._rotationDisabled=!0;}enableRotation(){this._rotationDisabled=!1;}}function $s(t){return t*(2-t)}const Ws=4.000244140625;class Hs{constructor(t,e){this._onTimeout=t=>{this._type="wheel",this._delta-=this._lastValue,this._active||this._start(t);},this._map=t,this._tr=new ws(t),this._el=t.getCanvasContainer(),this._triggerRenderFrame=e,this._delta=0,this._defaultZoomRate=.01,this._wheelZoomRate=.0022222222222222222;}setZoomRate(t){this._defaultZoomRate=t;}setWheelZoomRate(t){this._wheelZoomRate=t;}isEnabled(){return !!this._enabled}isActive(){return !!this._active||void 0!==this._finishTimeout}isZooming(){return !!this._zooming}enable(t){this.isEnabled()||(this._enabled=!0,this._aroundCenter=!!t&&"center"===t.around);}disable(){this.isEnabled()&&(this._enabled=!1);}wheel(e){if(!this.isEnabled())return;if(this._map._cooperativeGestures){if(!e[this._map._metaKey])return;e.preventDefault();}let i=e.deltaMode===WheelEvent.DOM_DELTA_LINE?40*e.deltaY:e.deltaY;const s=t.h.now(),a=s-(this._lastWheelEventTime||0);this._lastWheelEventTime=s,0!==i&&i%Ws==0?this._type="wheel":0!==i&&Math.abs(i)<4?this._type="trackpad":a>400?(this._type=null,this._lastValue=i,this._timeout=setTimeout(this._onTimeout,40,e)):this._type||(this._type=Math.abs(a*i)<200?"trackpad":"wheel",this._timeout&&(clearTimeout(this._timeout),this._timeout=null,i+=this._lastValue)),e.shiftKey&&i&&(i/=4),this._type&&(this._lastWheelEvent=e,this._delta-=i,this._active||this._start(e)),e.preventDefault();}_start(e){if(!this._delta)return;this._frameId&&(this._frameId=null),this._active=!0,this.isZooming()||(this._zooming=!0),this._finishTimeout&&(clearTimeout(this._finishTimeout),delete this._finishTimeout);const s=i.mousePos(this._el,e),a=this._tr;this._around=t.L.convert(this._aroundCenter?a.center:a.unproject(s)),this._aroundPoint=a.transform.locationPoint(this._around),this._frameId||(this._frameId=!0,this._triggerRenderFrame());}renderFrame(){if(!this._frameId)return;if(this._frameId=null,!this.isActive())return;const e=this._tr.transform;if(0!==this._delta){const t="wheel"===this._type&&Math.abs(this._delta)>Ws?this._wheelZoomRate:this._defaultZoomRate;let i=2/(1+Math.exp(-Math.abs(this._delta*t)));this._delta<0&&0!==i&&(i=1/i);const s="number"==typeof this._targetZoom?e.zoomScale(this._targetZoom):e.scale;this._targetZoom=Math.min(e.maxZoom,Math.max(e.minZoom,e.scaleZoom(s*i))),"wheel"===this._type&&(this._startZoom=e.zoom,this._easing=this._smoothOutEasing(200)),this._delta=0;}const i="number"==typeof this._targetZoom?this._targetZoom:e.zoom,s=this._startZoom,a=this._easing;let o,r=!1;if("wheel"===this._type&&s&&a){const e=Math.min((t.h.now()-this._lastWheelEventTime)/200,1),n=a(e);o=t.B.number(s,i,n),e<1?this._frameId||(this._frameId=!0):r=!0;}else o=i,r=!0;return this._active=!0,r&&(this._active=!1,this._finishTimeout=setTimeout((()=>{this._zooming=!1,this._triggerRenderFrame(),delete this._targetZoom,delete this._finishTimeout;}),200)),{noInertia:!0,needsRenderFrame:!r,zoomDelta:o-e.zoom,around:this._aroundPoint,originalEvent:this._lastWheelEvent}}_smoothOutEasing(e){let i=t.bb;if(this._prevEase){const e=this._prevEase,s=(t.h.now()-e.start)/e.duration,a=e.easing(s+.01)-e.easing(s),o=.27/Math.sqrt(a*a+1e-4)*.01,r=Math.sqrt(.0729-o*o);i=t.ba(o,r,.25,1);}return this._prevEase={start:t.h.now(),duration:e,easing:i},i}reset(){this._active=!1,this._zooming=!1,delete this._targetZoom,this._finishTimeout&&(clearTimeout(this._finishTimeout),delete this._finishTimeout);}}class Xs{constructor(t,e){this._clickZoom=t,this._tapZoom=e;}enable(){this._clickZoom.enable(),this._tapZoom.enable();}disable(){this._clickZoom.disable(),this._tapZoom.disable();}isEnabled(){return this._clickZoom.isEnabled()&&this._tapZoom.isEnabled()}isActive(){return this._clickZoom.isActive()||this._tapZoom.isActive()}}class Ks{constructor(t){this._tr=new ws(t),this.reset();}reset(){this._active=!1;}dblclick(t,e){return t.preventDefault(),{cameraAnimation:i=>{i.easeTo({duration:300,zoom:this._tr.zoom+(t.shiftKey?-1:1),around:this._tr.unproject(e)},{originalEvent:t});}}}enable(){this._enabled=!0;}disable(){this._enabled=!1,this.reset();}isEnabled(){return this._enabled}isActive(){return this._active}}class Qs{constructor(){this._tap=new Ss({numTouches:1,numTaps:1}),this.reset();}reset(){this._active=!1,delete this._swipePoint,delete this._swipeTouch,delete this._tapTime,delete this._tapPoint,this._tap.reset();}touchstart(t,e,i){if(!this._swipePoint)if(this._tapTime){const s=e[0],a=t.timeStamp-this._tapTime<500,o=this._tapPoint.dist(s)<30;a&&o?i.length>0&&(this._swipePoint=s,this._swipeTouch=i[0].identifier):this.reset();}else this._tap.touchstart(t,e,i);}touchmove(t,e,i){if(this._tapTime){if(this._swipePoint){if(i[0].identifier!==this._swipeTouch)return;const s=e[0],a=s.y-this._swipePoint.y;return this._swipePoint=s,t.preventDefault(),this._active=!0,{zoomDelta:a/128}}}else this._tap.touchmove(t,e,i);}touchend(t,e,i){if(this._tapTime)this._swipePoint&&0===i.length&&this.reset();else {const s=this._tap.touchend(t,e,i);s&&(this._tapTime=t.timeStamp,this._tapPoint=s);}}touchcancel(){this.reset();}enable(){this._enabled=!0;}disable(){this._enabled=!1,this.reset();}isEnabled(){return this._enabled}isActive(){return this._active}}class Ys{constructor(t,e,i){this._el=t,this._mousePan=e,this._touchPan=i;}enable(t){this._inertiaOptions=t||{},this._mousePan.enable(),this._touchPan.enable(),this._el.classList.add("maplibregl-touch-drag-pan");}disable(){this._mousePan.disable(),this._touchPan.disable(),this._el.classList.remove("maplibregl-touch-drag-pan");}isEnabled(){return this._mousePan.isEnabled()&&this._touchPan.isEnabled()}isActive(){return this._mousePan.isActive()||this._touchPan.isActive()}}class Js{constructor(t,e,i){this._pitchWithRotate=t.pitchWithRotate,this._mouseRotate=e,this._mousePitch=i;}enable(){this._mouseRotate.enable(),this._pitchWithRotate&&this._mousePitch.enable();}disable(){this._mouseRotate.disable(),this._mousePitch.disable();}isEnabled(){return this._mouseRotate.isEnabled()&&(!this._pitchWithRotate||this._mousePitch.isEnabled())}isActive(){return this._mouseRotate.isActive()||this._mousePitch.isActive()}}class ta{constructor(t,e,i,s){this._el=t,this._touchZoom=e,this._touchRotate=i,this._tapDragZoom=s,this._rotationDisabled=!1,this._enabled=!0;}enable(t){this._touchZoom.enable(t),this._rotationDisabled||this._touchRotate.enable(t),this._tapDragZoom.enable(),this._el.classList.add("maplibregl-touch-zoom-rotate");}disable(){this._touchZoom.disable(),this._touchRotate.disable(),this._tapDragZoom.disable(),this._el.classList.remove("maplibregl-touch-zoom-rotate");}isEnabled(){return this._touchZoom.isEnabled()&&(this._rotationDisabled||this._touchRotate.isEnabled())&&this._tapDragZoom.isEnabled()}isActive(){return this._touchZoom.isActive()||this._touchRotate.isActive()||this._tapDragZoom.isActive()}disableRotation(){this._rotationDisabled=!0,this._touchRotate.disable();}enableRotation(){this._rotationDisabled=!1,this._touchZoom.isEnabled()&&this._touchRotate.enable();}}const ea=t=>t.zoom||t.drag||t.pitch||t.rotate;class ia extends t.k{}function sa(t){return t.panDelta&&t.panDelta.mag()||t.zoomDelta||t.bearingDelta||t.pitchDelta}class aa{constructor(t,e){this.handleWindowEvent=t=>{this.handleEvent(t,`${t.type}Window`);},this.handleEvent=(t,e)=>{if("blur"===t.type)return void this.stop(!0);this._updatingCamera=!0;const s="renderFrame"===t.type?void 0:t,a={needsRenderFrame:!1},o={},r={},n=t.touches,l=n?this._getMapTouches(n):void 0,h=l?i.touchPos(this._el,l):i.mousePos(this._el,t);for(const{handlerName:i,handler:n,allowed:c}of this._handlers){if(!n.isEnabled())continue;let u;this._blockedByActive(r,c,i)?n.reset():n[e||t.type]&&(u=n[e||t.type](t,h,l),this.mergeHandlerResult(a,o,u,i,s),u&&u.needsRenderFrame&&this._triggerRenderFrame()),(u||n.isActive())&&(r[i]=n);}const c={};for(const t in this._previousActiveHandlers)r[t]||(c[t]=s);this._previousActiveHandlers=r,(Object.keys(c).length||sa(a))&&(this._changes.push([a,o,c]),this._triggerRenderFrame()),(Object.keys(r).length||sa(a))&&this._map._stop(!0),this._updatingCamera=!1;const{cameraAnimation:u}=a;u&&(this._inertia.clear(),this._fireEvents({},{},!0),this._changes=[],u(this._map));},this._map=t,this._el=this._map.getCanvasContainer(),this._handlers=[],this._handlersById={},this._changes=[],this._inertia=new ps(t),this._bearingSnap=e.bearingSnap,this._previousActiveHandlers={},this._eventsInProgress={},this._addDefaultHandlers(e);const s=this._el;this._listeners=[[s,"touchstart",{passive:!0}],[s,"touchmove",{passive:!1}],[s,"touchend",void 0],[s,"touchcancel",void 0],[s,"mousedown",void 0],[s,"mousemove",void 0],[s,"mouseup",void 0],[document,"mousemove",{capture:!0}],[document,"mouseup",void 0],[s,"mouseover",void 0],[s,"mouseout",void 0],[s,"dblclick",void 0],[s,"click",void 0],[s,"keydown",{capture:!1}],[s,"keyup",void 0],[s,"wheel",{passive:!1}],[s,"contextmenu",void 0],[window,"blur",void 0]];for(const[t,e,s]of this._listeners)i.addEventListener(t,e,t===document?this.handleWindowEvent:this.handleEvent,s);}destroy(){for(const[t,e,s]of this._listeners)i.removeEventListener(t,e,t===document?this.handleWindowEvent:this.handleEvent,s);}_addDefaultHandlers(t){const e=this._map,s=e.getCanvasContainer();this._add("mapEvent",new ys(e,t));const a=e.boxZoom=new Ts(e,t);this._add("boxZoom",a),t.interactive&&t.boxZoom&&a.enable();const o=new Cs(e),r=new Ks(e);e.doubleClickZoom=new Xs(r,o),this._add("tapZoom",o),this._add("clickZoom",r),t.interactive&&t.doubleClickZoom&&e.doubleClickZoom.enable();const n=new Qs;this._add("tapDragZoom",n);const l=e.touchPitch=new js(e);this._add("touchPitch",l),t.interactive&&t.touchPitch&&e.touchPitch.enable(t.touchPitch);const h=As(t),c=Rs(t);e.dragRotate=new Js(t,h,c),this._add("mouseRotate",h,["mousePitch"]),this._add("mousePitch",c,["mouseRotate"]),t.interactive&&t.dragRotate&&e.dragRotate.enable();const u=(({enable:t,clickTolerance:e})=>{const s=new Ms({checkCorrectEvent:t=>0===i.mouseButton(t)&&!t.ctrlKey});return new Ps({clickTolerance:e,move:(t,e)=>({around:e,panDelta:e.sub(t)}),activateOnStart:!0,moveStateManager:s,enable:t,assignEvents:Ls})})(t),d=new ks(t,e);e.dragPan=new Ys(s,u,d),this._add("mousePan",u),this._add("touchPan",d,["touchZoom","touchRotate"]),t.interactive&&t.dragPan&&e.dragPan.enable(t.dragPan);const _=new Zs,p=new Ns;e.touchZoomRotate=new ta(s,p,_,n),this._add("touchRotate",_,["touchPan","touchZoom"]),this._add("touchZoom",p,["touchPan","touchRotate"]),t.interactive&&t.touchZoomRotate&&e.touchZoomRotate.enable(t.touchZoomRotate);const m=e.scrollZoom=new Hs(e,(()=>this._triggerRenderFrame()));this._add("scrollZoom",m,["mousePan"]),t.interactive&&t.scrollZoom&&e.scrollZoom.enable(t.scrollZoom);const f=e.keyboard=new qs(e);this._add("keyboard",f),t.interactive&&t.keyboard&&e.keyboard.enable(),this._add("blockableMapEvent",new bs(e));}_add(t,e,i){this._handlers.push({handlerName:t,handler:e,allowed:i}),this._handlersById[t]=e;}stop(t){if(!this._updatingCamera){for(const{handler:t}of this._handlers)t.reset();this._inertia.clear(),this._fireEvents({},{},t),this._changes=[];}}isActive(){for(const{handler:t}of this._handlers)if(t.isActive())return !0;return !1}isZooming(){return !!this._eventsInProgress.zoom||this._map.scrollZoom.isZooming()}isRotating(){return !!this._eventsInProgress.rotate}isMoving(){return Boolean(ea(this._eventsInProgress))||this.isZooming()}_blockedByActive(t,e,i){for(const s in t)if(s!==i&&(!e||e.indexOf(s)<0))return !0;return !1}_getMapTouches(t){const e=[];for(const i of t)this._el.contains(i.target)&&e.push(i);return e}mergeHandlerResult(e,i,s,a,o){if(!s)return;t.e(e,s);const r={handlerName:a,originalEvent:s.originalEvent||o};void 0!==s.zoomDelta&&(i.zoom=r),void 0!==s.panDelta&&(i.drag=r),void 0!==s.pitchDelta&&(i.pitch=r),void 0!==s.bearingDelta&&(i.rotate=r);}_applyChanges(){const e={},i={},s={};for(const[a,o,r]of this._changes)a.panDelta&&(e.panDelta=(e.panDelta||new t.P(0,0))._add(a.panDelta)),a.zoomDelta&&(e.zoomDelta=(e.zoomDelta||0)+a.zoomDelta),a.bearingDelta&&(e.bearingDelta=(e.bearingDelta||0)+a.bearingDelta),a.pitchDelta&&(e.pitchDelta=(e.pitchDelta||0)+a.pitchDelta),void 0!==a.around&&(e.around=a.around),void 0!==a.pinchAround&&(e.pinchAround=a.pinchAround),a.noInertia&&(e.noInertia=a.noInertia),t.e(i,o),t.e(s,r);this._updateMapTransform(e,i,s),this._changes=[];}_updateMapTransform(t,e,i){const s=this._map,a=s._getTransformForUpdate(),o=s.terrain;if(!(sa(t)||o&&this._terrainMovement))return this._fireEvents(e,i,!0);let{panDelta:r,zoomDelta:n,bearingDelta:l,pitchDelta:h,around:c,pinchAround:u}=t;void 0!==u&&(c=u),s._stop(!0),c=c||s.transform.centerPoint;const d=a.pointLocation(r?c.sub(r):c);l&&(a.bearing+=l),h&&(a.pitch+=h),n&&(a.zoom+=n),o?this._terrainMovement||!e.drag&&!e.zoom?e.drag&&this._terrainMovement?a.center=a.pointLocation(a.centerPoint.sub(r)):a.setLocationAtPoint(d,c):(this._terrainMovement=!0,this._map._elevationFreeze=!0,a.setLocationAtPoint(d,c),this._map.once("moveend",(()=>{this._map._elevationFreeze=!1,this._terrainMovement=!1,a.recalculateZoom(s.terrain);}))):a.setLocationAtPoint(d,c),s._applyUpdatedTransform(a),this._map._update(),t.noInertia||this._inertia.record(t),this._fireEvents(e,i,!0);}_fireEvents(e,i,s){const a=ea(this._eventsInProgress),o=ea(e),r={};for(const t in e){const{originalEvent:i}=e[t];this._eventsInProgress[t]||(r[`${t}start`]=i),this._eventsInProgress[t]=e[t];}!a&&o&&this._fireEvent("movestart",o.originalEvent);for(const t in r)this._fireEvent(t,r[t]);o&&this._fireEvent("move",o.originalEvent);for(const t in e){const{originalEvent:i}=e[t];this._fireEvent(t,i);}const n={};let l;for(const t in this._eventsInProgress){const{handlerName:e,originalEvent:s}=this._eventsInProgress[t];this._handlersById[e].isActive()||(delete this._eventsInProgress[t],l=i[e]||s,n[`${t}end`]=l);}for(const t in n)this._fireEvent(t,n[t]);const h=ea(this._eventsInProgress);if(s&&(a||o)&&!h){this._updatingCamera=!0;const e=this._inertia._onMoveEnd(this._map.dragPan._inertiaOptions),i=t=>0!==t&&-this._bearingSnap{delete this._frameId,this.handleEvent(new ia("renderFrame",{timeStamp:t})),this._applyChanges();}))}_triggerRenderFrame(){void 0===this._frameId&&(this._frameId=this._requestFrame());}}class oa extends t.E{constructor(e,i){super(),this._renderFrameCallback=()=>{const e=Math.min((t.h.now()-this._easeStart)/this._easeOptions.duration,1);this._onEaseFrame(this._easeOptions.easing(e)),e<1&&this._easeFrameId?this._easeFrameId=this._requestRenderFrame(this._renderFrameCallback):this.stop();},this._moving=!1,this._zooming=!1,this.transform=e,this._bearingSnap=i.bearingSnap,this.on("moveend",(()=>{delete this._requestedCameraState;}));}getCenter(){return new t.L(this.transform.center.lng,this.transform.center.lat)}setCenter(t,e){return this.jumpTo({center:t},e)}panBy(e,i,s){return e=t.P.convert(e).mult(-1),this.panTo(this.transform.center,t.e({offset:e},i),s)}panTo(e,i,s){return this.easeTo(t.e({center:e},i),s)}getZoom(){return this.transform.zoom}setZoom(t,e){return this.jumpTo({zoom:t},e),this}zoomTo(e,i,s){return this.easeTo(t.e({zoom:e},i),s)}zoomIn(t,e){return this.zoomTo(this.getZoom()+1,t,e),this}zoomOut(t,e){return this.zoomTo(this.getZoom()-1,t,e),this}getBearing(){return this.transform.bearing}setBearing(t,e){return this.jumpTo({bearing:t},e),this}getPadding(){return this.transform.padding}setPadding(t,e){return this.jumpTo({padding:t},e),this}rotateTo(e,i,s){return this.easeTo(t.e({bearing:e},i),s)}resetNorth(e,i){return this.rotateTo(0,t.e({duration:1e3},e),i),this}resetNorthPitch(e,i){return this.easeTo(t.e({bearing:0,pitch:0,duration:1e3},e),i),this}snapToNorth(t,e){return Math.abs(this.getBearing()){if(this._zooming&&(s.zoom=t.B.number(a,l,p)),this._rotating&&(s.bearing=t.B.number(o,h,p)),this._pitching&&(s.pitch=t.B.number(r,c,p)),this._padding&&(s.interpolatePadding(n,u,p),_=s.centerPoint.add(d)),this.terrain&&!e.freezeElevation&&this._updateElevation(p),x)s.setLocationAtPoint(x,y);else {const t=s.zoomScale(s.zoom-a),e=l>a?Math.min(2,v):Math.max(.5,v),i=Math.pow(e,1-p),o=s.unproject(f.add(g.mult(p*i)).mult(t));s.setLocationAtPoint(s.renderWorldCopies?o.wrap():o,_);}this._applyUpdatedTransform(s),this._fireMoveEvents(i);}),(t=>{this.terrain&&this._finalizeElevation(),this._afterEase(i,t);}),e),this}_prepareEase(e,i,s={}){this._moving=!0,i||s.moving||this.fire(new t.k("movestart",e)),this._zooming&&!s.zooming&&this.fire(new t.k("zoomstart",e)),this._rotating&&!s.rotating&&this.fire(new t.k("rotatestart",e)),this._pitching&&!s.pitching&&this.fire(new t.k("pitchstart",e));}_prepareElevation(t){this._elevationCenter=t,this._elevationStart=this.transform.elevation,this._elevationTarget=this.terrain.getElevationForLngLatZoom(t,this.transform.tileZoom),this._elevationFreeze=!0;}_updateElevation(e){this.transform._minEleveationForCurrentTile=this.terrain.getMinTileElevationForLngLatZoom(this._elevationCenter,this.transform.tileZoom);const i=this.terrain.getElevationForLngLatZoom(this._elevationCenter,this.transform.tileZoom);if(e<1&&i!==this._elevationTarget){const t=this._elevationTarget-this._elevationStart;this._elevationStart+=e*(t-(i-(t*e+this._elevationStart))/(1-e)),this._elevationTarget=i;}this.transform.elevation=t.B.number(this._elevationStart,this._elevationTarget,e);}_finalizeElevation(){this._elevationFreeze=!1,this.transform.recalculateZoom(this.terrain);}_getTransformForUpdate(){return this.transformCameraUpdate?(this._requestedCameraState||(this._requestedCameraState=this.transform.clone()),this._requestedCameraState):this.transform}_applyUpdatedTransform(t){if(!this.transformCameraUpdate)return;const e=t.clone(),{center:i,zoom:s,pitch:a,bearing:o,elevation:r}=this.transformCameraUpdate(e);i&&(e.center=i),void 0!==s&&(e.zoom=s),void 0!==a&&(e.pitch=a),void 0!==o&&(e.bearing=o),void 0!==r&&(e.elevation=r),this.transform.apply(e);}_fireMoveEvents(e){this.fire(new t.k("move",e)),this._zooming&&this.fire(new t.k("zoom",e)),this._rotating&&this.fire(new t.k("rotate",e)),this._pitching&&this.fire(new t.k("pitch",e));}_afterEase(e,i){if(this._easeId&&i&&this._easeId===i)return;delete this._easeId;const s=this._zooming,a=this._rotating,o=this._pitching;this._moving=!1,this._zooming=!1,this._rotating=!1,this._pitching=!1,this._padding=!1,s&&this.fire(new t.k("zoomend",e)),a&&this.fire(new t.k("rotateend",e)),o&&this.fire(new t.k("pitchend",e)),this.fire(new t.k("moveend",e));}flyTo(e,i){if(!e.essential&&t.h.prefersReducedMotion){const s=t.F(e,["center","zoom","bearing","pitch","around"]);return this.jumpTo(s,i)}this.stop(),e=t.e({offset:[0,0],speed:1.2,curve:1.42,easing:t.bb},e);const s=this._getTransformForUpdate(),a=this.getZoom(),o=this.getBearing(),r=this.getPitch(),n=this.getPadding(),l="zoom"in e?t.ad(+e.zoom,s.minZoom,s.maxZoom):a,h="bearing"in e?this._normalizeBearing(e.bearing,o):o,c="pitch"in e?+e.pitch:r,u="padding"in e?e.padding:s.padding,d=s.zoomScale(l-a),_=t.P.convert(e.offset);let p=s.centerPoint.add(_);const m=s.pointLocation(p),f=t.L.convert(e.center||m);this._normalizeCenter(f);const g=s.project(m),v=s.project(f).sub(g);let x=e.curve;const y=Math.max(s.width,s.height),b=y/d,w=v.mag();if("minZoom"in e){const i=t.ad(Math.min(e.minZoom,a,l),s.minZoom,s.maxZoom),o=y/s.zoomScale(i-a);x=Math.sqrt(o/w*2);}const T=x*x;function I(t){const e=(b*b-y*y+(t?-1:1)*T*T*w*w)/(2*(t?b:y)*T*w);return Math.log(Math.sqrt(e*e+1)-e)}function E(t){return (Math.exp(t)-Math.exp(-t))/2}function S(t){return (Math.exp(t)+Math.exp(-t))/2}const C=I(!1);let P=function(t){return S(C)/S(C+x*t)},D=function(t){return y*((S(C)*(E(e=C+x*t)/S(e))-E(C))/T)/w;var e;},M=(I(!0)-C)/x;if(Math.abs(w)<1e-6||!isFinite(M)){if(Math.abs(y-b)<1e-6)return this.easeTo(e,i);const t=be.maxDuration&&(e.duration=0),this._zooming=!0,this._rotating=o!==h,this._pitching=c!==r,this._padding=!s.isPaddingEqual(u),this._prepareEase(i,!1),this.terrain&&this._prepareElevation(f),this._ease((d=>{const m=d*M,x=1/P(m);s.zoom=1===d?l:a+s.scaleZoom(x),this._rotating&&(s.bearing=t.B.number(o,h,d)),this._pitching&&(s.pitch=t.B.number(r,c,d)),this._padding&&(s.interpolatePadding(n,u,d),p=s.centerPoint.add(_)),this.terrain&&!e.freezeElevation&&this._updateElevation(d);const y=1===d?f:s.unproject(g.add(v.mult(D(m))).mult(x));s.setLocationAtPoint(s.renderWorldCopies?y.wrap():y,p),this._applyUpdatedTransform(s),this._fireMoveEvents(i);}),(()=>{this.terrain&&this._finalizeElevation(),this._afterEase(i);}),e),this}isEasing(){return !!this._easeFrameId}stop(){return this._stop()}_stop(t,e){if(this._easeFrameId&&(this._cancelRenderFrame(this._easeFrameId),delete this._easeFrameId,delete this._onEaseFrame),this._onEaseEnd){const t=this._onEaseEnd;delete this._onEaseEnd,t.call(this,e);}if(!t){const t=this.handlers;t&&t.stop(!1);}return this}_ease(e,i,s){!1===s.animate||0===s.duration?(e(1),i()):(this._easeStart=t.h.now(),this._easeOptions=s,this._onEaseFrame=e,this._onEaseEnd=i,this._easeFrameId=this._requestRenderFrame(this._renderFrameCallback));}_normalizeBearing(e,i){e=t.b5(e,-180,180);const s=Math.abs(e-i);return Math.abs(e-360-i)180?-360:i<-180?360:0;}queryTerrainElevation(e){return this.terrain?this.terrain.getElevationForLngLatZoom(t.L.convert(e),this.transform.tileZoom)-this.transform.elevation:null}}class ra{constructor(t={}){this._toggleAttribution=()=>{this._container.classList.contains("maplibregl-compact")&&(this._container.classList.contains("maplibregl-compact-show")?(this._container.setAttribute("open",""),this._container.classList.remove("maplibregl-compact-show")):(this._container.classList.add("maplibregl-compact-show"),this._container.removeAttribute("open")));},this._updateData=t=>{!t||"metadata"!==t.sourceDataType&&"visibility"!==t.sourceDataType&&"style"!==t.dataType&&"terrain"!==t.type||this._updateAttributions();},this._updateCompact=()=>{this._map.getCanvasContainer().offsetWidth<=640||this._compact?!1===this._compact?this._container.setAttribute("open",""):this._container.classList.contains("maplibregl-compact")||this._container.classList.contains("maplibregl-attrib-empty")||(this._container.setAttribute("open",""),this._container.classList.add("maplibregl-compact","maplibregl-compact-show")):(this._container.setAttribute("open",""),this._container.classList.contains("maplibregl-compact")&&this._container.classList.remove("maplibregl-compact","maplibregl-compact-show"));},this._updateCompactMinimize=()=>{this._container.classList.contains("maplibregl-compact")&&this._container.classList.contains("maplibregl-compact-show")&&this._container.classList.remove("maplibregl-compact-show");},this.options=t;}getDefaultPosition(){return "bottom-right"}onAdd(t){return this._map=t,this._compact=this.options&&this.options.compact,this._container=i.create("details","maplibregl-ctrl maplibregl-ctrl-attrib"),this._compactButton=i.create("summary","maplibregl-ctrl-attrib-button",this._container),this._compactButton.addEventListener("click",this._toggleAttribution),this._setElementTitle(this._compactButton,"ToggleAttribution"),this._innerContainer=i.create("div","maplibregl-ctrl-attrib-inner",this._container),this._updateAttributions(),this._updateCompact(),this._map.on("styledata",this._updateData),this._map.on("sourcedata",this._updateData),this._map.on("terrain",this._updateData),this._map.on("resize",this._updateCompact),this._map.on("drag",this._updateCompactMinimize),this._container}onRemove(){i.remove(this._container),this._map.off("styledata",this._updateData),this._map.off("sourcedata",this._updateData),this._map.off("terrain",this._updateData),this._map.off("resize",this._updateCompact),this._map.off("drag",this._updateCompactMinimize),this._map=void 0,this._compact=void 0,this._attribHTML=void 0;}_setElementTitle(t,e){const i=this._map._getUIString(`AttributionControl.${e}`);t.title=i,t.setAttribute("aria-label",i);}_updateAttributions(){if(!this._map.style)return;let t=[];if(this.options.customAttribution&&(Array.isArray(this.options.customAttribution)?t=t.concat(this.options.customAttribution.map((t=>"string"!=typeof t?"":t))):"string"==typeof this.options.customAttribution&&t.push(this.options.customAttribution)),this._map.style.stylesheet){const t=this._map.style.stylesheet;this.styleOwner=t.owner,this.styleId=t.id;}const e=this._map.style.sourceCaches;for(const i in e){const s=e[i];if(s.used||s.usedForTerrain){const e=s.getSource();e.attribution&&t.indexOf(e.attribution)<0&&t.push(e.attribution);}}t=t.filter((t=>String(t).trim())),t.sort(((t,e)=>t.length-e.length)),t=t.filter(((e,i)=>{for(let s=i+1;s=0)return !1;return !0}));const i=t.join(" | ");i!==this._attribHTML&&(this._attribHTML=i,t.length?(this._innerContainer.innerHTML=i,this._container.classList.remove("maplibregl-attrib-empty")):this._container.classList.add("maplibregl-attrib-empty"),this._updateCompact(),this._editLink=null);}}class na{constructor(t={}){this._updateCompact=()=>{const t=this._container.children;if(t.length){const e=t[0];this._map.getCanvasContainer().offsetWidth<=640||this._compact?!1!==this._compact&&e.classList.add("maplibregl-compact"):e.classList.remove("maplibregl-compact");}},this.options=t;}getDefaultPosition(){return "bottom-left"}onAdd(t){this._map=t,this._compact=this.options&&this.options.compact,this._container=i.create("div","maplibregl-ctrl");const e=i.create("a","maplibregl-ctrl-logo");return e.target="_blank",e.rel="noopener nofollow",e.href="https://maplibre.org/",e.setAttribute("aria-label",this._map._getUIString("LogoControl.Title")),e.setAttribute("rel","noopener nofollow"),this._container.appendChild(e),this._container.style.display="block",this._map.on("resize",this._updateCompact),this._updateCompact(),this._container}onRemove(){i.remove(this._container),this._map.off("resize",this._updateCompact),this._map=void 0,this._compact=void 0;}}class la{constructor(){this._queue=[],this._id=0,this._cleared=!1,this._currentlyRunning=!1;}add(t){const e=++this._id;return this._queue.push({callback:t,id:e,cancelled:!1}),e}remove(t){const e=this._currentlyRunning,i=e?this._queue.concat(e):this._queue;for(const e of i)if(e.id===t)return void(e.cancelled=!0)}run(t=0){if(this._currentlyRunning)throw new Error("Attempting to run(), but is already running.");const e=this._currentlyRunning=this._queue;this._queue=[];for(const i of e)if(!i.cancelled&&(i.callback(t),this._cleared))break;this._cleared=!1,this._currentlyRunning=!1;}clear(){this._currentlyRunning&&(this._cleared=!0),this._queue=[];}}const ha={"AttributionControl.ToggleAttribution":"Toggle attribution","AttributionControl.MapFeedback":"Map feedback","FullscreenControl.Enter":"Enter fullscreen","FullscreenControl.Exit":"Exit fullscreen","GeolocateControl.FindMyLocation":"Find my location","GeolocateControl.LocationNotAvailable":"Location not available","LogoControl.Title":"Mapbox logo","NavigationControl.ResetBearing":"Reset bearing to north","NavigationControl.ZoomIn":"Zoom in","NavigationControl.ZoomOut":"Zoom out","ScaleControl.Feet":"ft","ScaleControl.Meters":"m","ScaleControl.Kilometers":"km","ScaleControl.Miles":"mi","ScaleControl.NauticalMiles":"nm","TerrainControl.enableTerrain":"Enable terrain","TerrainControl.disableTerrain":"Disable terrain"};var ca=t.Q([{name:"a_pos3d",type:"Int16",components:3}]);class ua extends t.E{constructor(t){super(),this.sourceCache=t,this._tiles={},this._renderableTilesKeys=[],this._sourceTileCache={},this.minzoom=0,this.maxzoom=22,this.tileSize=512,this.deltaZoom=1,t.usedForTerrain=!0,t.tileSize=this.tileSize*2**this.deltaZoom;}destruct(){this.sourceCache.usedForTerrain=!1,this.sourceCache.tileSize=null;}update(e,i){this.sourceCache.update(e,i),this._renderableTilesKeys=[];const s={};for(const a of e.coveringTiles({tileSize:this.tileSize,minzoom:this.minzoom,maxzoom:this.maxzoom,reparseOverscaled:!1,terrain:i}))s[a.key]=!0,this._renderableTilesKeys.push(a.key),this._tiles[a.key]||(a.posMatrix=new Float64Array(16),t.aS(a.posMatrix,0,t.N,0,t.N,0,1),this._tiles[a.key]=new W(a,this.tileSize));for(const t in this._tiles)s[t]||delete this._tiles[t];}freeRtt(t){for(const e in this._tiles){const i=this._tiles[e];(!t||i.tileID.equals(t)||i.tileID.isChildOf(t)||t.isChildOf(i.tileID))&&(i.rtt=[]);}}getRenderableTiles(){return this._renderableTilesKeys.map((t=>this.getTileByID(t)))}getTileByID(t){return this._tiles[t]}getTerrainCoords(e){const i={};for(const s of this._renderableTilesKeys){const a=this._tiles[s].tileID;if(a.canonical.equals(e.canonical)){const a=e.clone();a.posMatrix=new Float64Array(16),t.aS(a.posMatrix,0,t.N,0,t.N,0,1),i[s]=a;}else if(a.canonical.isChildOf(e.canonical)){const o=e.clone();o.posMatrix=new Float64Array(16);const r=a.canonical.z-e.canonical.z,n=a.canonical.x-(a.canonical.x>>r<>r<>r;t.aS(o.posMatrix,0,h,0,h,0,1),t.$(o.posMatrix,o.posMatrix,[-n*h,-l*h,0]),i[s]=o;}else if(e.canonical.isChildOf(a.canonical)){const o=e.clone();o.posMatrix=new Float64Array(16);const r=e.canonical.z-a.canonical.z,n=e.canonical.x-(e.canonical.x>>r<>r<>r;t.aS(o.posMatrix,0,t.N,0,t.N,0,1),t.$(o.posMatrix,o.posMatrix,[n*h,l*h,0]),t.a0(o.posMatrix,o.posMatrix,[1/2**r,1/2**r,0]),i[s]=o;}}return i}getSourceTile(t,e){const i=this.sourceCache._source;let s=t.overscaledZ-this.deltaZoom;if(s>i.maxzoom&&(s=i.maxzoom),s=i.minzoom&&(!a||!a.dem);)a=this.sourceCache.getTileByID(t.scaledTo(s--).key);return a}tilesAfterTime(t=Date.now()){return Object.values(this._tiles).filter((e=>e.timeAdded>=t))}}class da{constructor(t,e,i){this.painter=t,this.sourceCache=new ua(e),this.options=i,this.exaggeration="number"==typeof i.exaggeration?i.exaggeration:1,this.qualityFactor=2,this.meshSize=128,this._demMatrixCache={},this.coordsIndex=[],this._coordsTextureSize=1024;}getDEMElevation(e,i,s,a=t.N){var o;if(!(i>=0&&i=0&&se.canonical.z&&(e.canonical.z>=s?a=e.canonical.z-s:t.w("cannot calculate elevation if elevation maxzoom > source.maxzoom"));const o=e.canonical.x-(e.canonical.x>>a<>a<>8<<4|t>>8,i[e+3]=0;const s=new t.R({width:this._coordsTextureSize,height:this._coordsTextureSize},new Uint8Array(i.buffer)),a=new x(e,s,e.gl.RGBA,{premultiply:!1});return a.bind(e.gl.NEAREST,e.gl.CLAMP_TO_EDGE),this._coordsTexture=a,a}pointCoordinate(e){const i=new Uint8Array(4),s=this.painter.context,a=s.gl;s.bindFramebuffer.set(this.getFramebuffer("coords").framebuffer),a.readPixels(e.x,this.painter.height/devicePixelRatio-e.y-1,1,1,a.RGBA,a.UNSIGNED_BYTE,i),s.bindFramebuffer.set(null);const o=i[0]+(i[2]>>4<<8),r=i[1]+((15&i[2])<<8),n=this.coordsIndex[255-i[3]],l=n&&this.sourceCache.getTileByID(n);if(!l)return null;const h=this._coordsTextureSize,c=(1<0&&Math.sign(o)<0||!s&&Math.sign(a)<0&&Math.sign(o)>0?(a=360*Math.sign(o)+a,t.G(a)):i}}class _a{constructor(t,e,i){this._context=t,this._size=e,this._tileSize=i,this._objects=[],this._recentlyUsed=[],this._stamp=0;}destruct(){for(const t of this._objects)t.texture.destroy(),t.fbo.destroy();}_createObject(t){const e=this._context.createFramebuffer(this._tileSize,this._tileSize,!0,!0),i=new x(this._context,{width:this._tileSize,height:this._tileSize,data:null},this._context.gl.RGBA);return i.bind(this._context.gl.LINEAR,this._context.gl.CLAMP_TO_EDGE),e.depthAttachment.set(this._context.createRenderbuffer(this._context.gl.DEPTH_STENCIL,this._tileSize,this._tileSize)),e.colorAttachment.set(i.texture),{id:t,fbo:e,texture:i,stamp:-1,inUse:!1}}getObjectForId(t){return this._objects[t]}useObject(t){t.inUse=!0,this._recentlyUsed=this._recentlyUsed.filter((e=>t.id!==e)),this._recentlyUsed.push(t.id);}stampObject(t){t.stamp=++this._stamp;}getOrCreateFreeObject(){for(const t of this._recentlyUsed)if(!this._objects[t].inUse)return this._objects[t];if(this._objects.length>=this._size)throw new Error("No free RenderPool available, call freeAllObjects() required!");const t=this._createObject(this._objects.length);return this._objects.push(t),t}freeObject(t){t.inUse=!1;}freeAllObjects(){for(const t of this._objects)this.freeObject(t);}isFull(){return !(this._objects.length!t.inUse))}}const pa={background:!0,fill:!0,line:!0,raster:!0,hillshade:!0};class ma{constructor(t,e){this.painter=t,this.terrain=e,this.pool=new _a(t.context,30,e.sourceCache.tileSize*e.qualityFactor);}destruct(){this.pool.destruct();}getTexture(t){return this.pool.getObjectForId(t.rtt[this._stacks.length-1].id).texture}prepareForRender(t,e){this._stacks=[],this._prevType=null,this._rttTiles=[],this._renderableTiles=this.terrain.sourceCache.getRenderableTiles(),this._renderableLayerIds=t._order.filter((i=>!t._layers[i].isHidden(e))),this._coordsDescendingInv={};for(const e in t.sourceCaches){this._coordsDescendingInv[e]={};const i=t.sourceCaches[e].getVisibleCoordinates();for(const t of i){const i=this.terrain.sourceCache.getTerrainCoords(t);for(const t in i)this._coordsDescendingInv[e][t]||(this._coordsDescendingInv[e][t]=[]),this._coordsDescendingInv[e][t].push(i[t]);}}this._coordsDescendingInvStr={};for(const e of t._order){const i=t._layers[e],s=i.source;if(pa[i.type]&&!this._coordsDescendingInvStr[s]){this._coordsDescendingInvStr[s]={};for(const t in this._coordsDescendingInv[s])this._coordsDescendingInvStr[s][t]=this._coordsDescendingInv[s][t].map((t=>t.key)).sort().join();}}for(const t of this._renderableTiles)for(const e in this._coordsDescendingInvStr){const i=this._coordsDescendingInvStr[e][t.tileID.key];i&&i!==t.rttCoords[e]&&(t.rtt=[]);}}renderLayer(e){if(e.isHidden(this.painter.transform.zoom))return !1;const i=e.type,s=this.painter,a=this._renderableLayerIds[this._renderableLayerIds.length-1]===e.id;if(pa[i]&&(this._prevType&&pa[this._prevType]||this._stacks.push([]),this._prevType=i,this._stacks[this._stacks.length-1].push(e.id),!a))return !0;if(pa[this._prevType]||pa[i]&&a){this._prevType=i;const e=this._stacks.length-1,a=this._stacks[e]||[];for(const i of this._renderableTiles){if(this.pool.isFull()&&(es(this.painter,this.terrain,this._rttTiles),this._rttTiles=[],this.pool.freeAllObjects()),this._rttTiles.push(i),i.rtt[e]){const t=this.pool.getObjectForId(i.rtt[e].id);if(t.stamp===i.rtt[e].stamp){this.pool.useObject(t);continue}}const o=this.pool.getOrCreateFreeObject();this.pool.useObject(o),this.pool.stampObject(o),i.rtt[e]={id:o.id,stamp:o.stamp},s.context.bindFramebuffer.set(o.fbo.framebuffer),s.context.clear({color:t.aT.transparent,stencil:0}),s.currentStencilSource=void 0;for(let t=0;t{t.touchstart=t.dragStart,t.touchmoveWindow=t.dragMove,t.touchend=t.dragEnd;},xa={showCompass:!0,showZoom:!0,visualizePitch:!1};class ya{constructor(e,s,a=!1){this.mousedown=e=>{this.startMouse(t.e({},e,{ctrlKey:!0,preventDefault:()=>e.preventDefault()}),i.mousePos(this.element,e)),i.addEventListener(window,"mousemove",this.mousemove),i.addEventListener(window,"mouseup",this.mouseup);},this.mousemove=t=>{this.moveMouse(t,i.mousePos(this.element,t));},this.mouseup=t=>{this.mouseRotate.dragEnd(t),this.mousePitch&&this.mousePitch.dragEnd(t),this.offTemp();},this.touchstart=t=>{1!==t.targetTouches.length?this.reset():(this._startPos=this._lastPos=i.touchPos(this.element,t.targetTouches)[0],this.startTouch(t,this._startPos),i.addEventListener(window,"touchmove",this.touchmove,{passive:!1}),i.addEventListener(window,"touchend",this.touchend));},this.touchmove=t=>{1!==t.targetTouches.length?this.reset():(this._lastPos=i.touchPos(this.element,t.targetTouches)[0],this.moveTouch(t,this._lastPos));},this.touchend=t=>{0===t.targetTouches.length&&this._startPos&&this._lastPos&&this._startPos.dist(this._lastPos){this.mouseRotate.reset(),this.mousePitch&&this.mousePitch.reset(),this.touchRotate.reset(),this.touchPitch&&this.touchPitch.reset(),delete this._startPos,delete this._lastPos,this.offTemp();},this._clickTolerance=10;const o=e.dragRotate._mouseRotate.getClickTolerance(),r=e.dragRotate._mousePitch.getClickTolerance();this.element=s,this.mouseRotate=As({clickTolerance:o,enable:!0}),this.touchRotate=(({enable:t,clickTolerance:e,bearingDegreesPerPixelMoved:i=.8})=>{const s=new zs;return new Ps({clickTolerance:e,move:(t,e)=>({bearingDelta:(e.x-t.x)*i}),moveStateManager:s,enable:t,assignEvents:va})})({clickTolerance:o,enable:!0}),this.map=e,a&&(this.mousePitch=Rs({clickTolerance:r,enable:!0}),this.touchPitch=(({enable:t,clickTolerance:e,pitchDegreesPerPixelMoved:i=-.5})=>{const s=new zs;return new Ps({clickTolerance:e,move:(t,e)=>({pitchDelta:(e.y-t.y)*i}),moveStateManager:s,enable:t,assignEvents:va})})({clickTolerance:r,enable:!0})),i.addEventListener(s,"mousedown",this.mousedown),i.addEventListener(s,"touchstart",this.touchstart,{passive:!1}),i.addEventListener(s,"touchcancel",this.reset);}startMouse(t,e){this.mouseRotate.dragStart(t,e),this.mousePitch&&this.mousePitch.dragStart(t,e),i.disableDrag();}startTouch(t,e){this.touchRotate.dragStart(t,e),this.touchPitch&&this.touchPitch.dragStart(t,e),i.disableDrag();}moveMouse(t,e){const i=this.map,{bearingDelta:s}=this.mouseRotate.dragMove(t,e)||{};if(s&&i.setBearing(i.getBearing()+s),this.mousePitch){const{pitchDelta:s}=this.mousePitch.dragMove(t,e)||{};s&&i.setPitch(i.getPitch()+s);}}moveTouch(t,e){const i=this.map,{bearingDelta:s}=this.touchRotate.dragMove(t,e)||{};if(s&&i.setBearing(i.getBearing()+s),this.touchPitch){const{pitchDelta:s}=this.touchPitch.dragMove(t,e)||{};s&&i.setPitch(i.getPitch()+s);}}off(){const t=this.element;i.removeEventListener(t,"mousedown",this.mousedown),i.removeEventListener(t,"touchstart",this.touchstart,{passive:!1}),i.removeEventListener(window,"touchmove",this.touchmove,{passive:!1}),i.removeEventListener(window,"touchend",this.touchend),i.removeEventListener(t,"touchcancel",this.reset),this.offTemp();}offTemp(){i.enableDrag(),i.removeEventListener(window,"mousemove",this.mousemove),i.removeEventListener(window,"mouseup",this.mouseup),i.removeEventListener(window,"touchmove",this.touchmove,{passive:!1}),i.removeEventListener(window,"touchend",this.touchend);}}let ba;function wa(e,i,s){if(e=new t.L(e.lng,e.lat),i){const a=new t.L(e.lng-360,e.lat),o=new t.L(e.lng+360,e.lat),r=s.locationPoint(e).distSqr(i);s.locationPoint(a).distSqr(i)180;){const t=s.locationPoint(e);if(t.x>=0&&t.y>=0&&t.x<=s.width&&t.y<=s.height)break;e.lng>s.center.lng?e.lng-=360:e.lng+=360;}return e}const Ta={center:"translate(-50%,-50%)",top:"translate(-50%,0)","top-left":"translate(0,0)","top-right":"translate(-100%,0)",bottom:"translate(-50%,-100%)","bottom-left":"translate(0,-100%)","bottom-right":"translate(-100%,-100%)",left:"translate(0,-50%)",right:"translate(-100%,-50%)"};function Ia(t,e,i){const s=t.classList;for(const t in Ta)s.remove(`maplibregl-${i}-anchor-${t}`);s.add(`maplibregl-${i}-anchor-${e}`);}class Ea extends t.E{constructor(e){if(super(),this._onKeyPress=t=>{const e=t.code,i=t.charCode||t.keyCode;"Space"!==e&&"Enter"!==e&&32!==i&&13!==i||this.togglePopup();},this._onMapClick=t=>{const e=t.originalEvent.target,i=this._element;this._popup&&(e===i||i.contains(e))&&this.togglePopup();},this._update=t=>{if(!this._map)return;const e=this._map.loaded()&&!this._map.isMoving();("terrain"===(null==t?void 0:t.type)||"render"===(null==t?void 0:t.type)&&!e)&&this._map.once("render",this._update),this._map.transform.renderWorldCopies&&(this._lngLat=wa(this._lngLat,this._pos,this._map.transform)),this._pos=this._map.project(this._lngLat)._add(this._offset);let s="";"viewport"===this._rotationAlignment||"auto"===this._rotationAlignment?s=`rotateZ(${this._rotation}deg)`:"map"===this._rotationAlignment&&(s=`rotateZ(${this._rotation-this._map.getBearing()}deg)`);let a="";"viewport"===this._pitchAlignment||"auto"===this._pitchAlignment?a="rotateX(0deg)":"map"===this._pitchAlignment&&(a=`rotateX(${this._map.getPitch()}deg)`),t&&"moveend"!==t.type||(this._pos=this._pos.round()),i.setTransform(this._element,`${Ta[this._anchor]} translate(${this._pos.x}px, ${this._pos.y}px) ${a} ${s}`),this._map.terrain&&!this._opacityTimeout&&(this._opacityTimeout=setTimeout((()=>{const t=this._map.unproject(this._pos),e=40075016.686*Math.abs(Math.cos(this._lngLat.lat*Math.PI/180))/Math.pow(2,this._map.transform.tileZoom+8);this._element.style.opacity=t.distanceTo(this._lngLat)>20*e?"0.2":"1.0",this._opacityTimeout=null;}),100));},this._onMove=e=>{if(!this._isDragging){const t=this._clickTolerance||this._map._clickTolerance;this._isDragging=e.point.dist(this._pointerdownPos)>=t;}this._isDragging&&(this._pos=e.point.sub(this._positionDelta),this._lngLat=this._map.unproject(this._pos),this.setLngLat(this._lngLat),this._element.style.pointerEvents="none","pending"===this._state&&(this._state="active",this.fire(new t.k("dragstart"))),this.fire(new t.k("drag")));},this._onUp=()=>{this._element.style.pointerEvents="auto",this._positionDelta=null,this._pointerdownPos=null,this._isDragging=!1,this._map.off("mousemove",this._onMove),this._map.off("touchmove",this._onMove),"active"===this._state&&this.fire(new t.k("dragend")),this._state="inactive";},this._addDragHandler=t=>{this._element.contains(t.originalEvent.target)&&(t.preventDefault(),this._positionDelta=t.point.sub(this._pos).add(this._offset),this._pointerdownPos=t.point,this._state="pending",this._map.on("mousemove",this._onMove),this._map.on("touchmove",this._onMove),this._map.once("mouseup",this._onUp),this._map.once("touchend",this._onUp));},this._anchor=e&&e.anchor||"center",this._color=e&&e.color||"#3FB1CE",this._scale=e&&e.scale||1,this._draggable=e&&e.draggable||!1,this._clickTolerance=e&&e.clickTolerance||0,this._isDragging=!1,this._state="inactive",this._rotation=e&&e.rotation||0,this._rotationAlignment=e&&e.rotationAlignment||"auto",this._pitchAlignment=e&&e.pitchAlignment&&"auto"!==e.pitchAlignment?e.pitchAlignment:this._rotationAlignment,e&&e.element)this._element=e.element,this._offset=t.P.convert(e&&e.offset||[0,0]);else {this._defaultMarker=!0,this._element=i.create("div"),this._element.setAttribute("aria-label","Map marker");const s=i.createNS("http://www.w3.org/2000/svg","svg"),a=41,o=27;s.setAttributeNS(null,"display","block"),s.setAttributeNS(null,"height",`${a}px`),s.setAttributeNS(null,"width",`${o}px`),s.setAttributeNS(null,"viewBox",`0 0 ${o} ${a}`);const r=i.createNS("http://www.w3.org/2000/svg","g");r.setAttributeNS(null,"stroke","none"),r.setAttributeNS(null,"stroke-width","1"),r.setAttributeNS(null,"fill","none"),r.setAttributeNS(null,"fill-rule","evenodd");const n=i.createNS("http://www.w3.org/2000/svg","g");n.setAttributeNS(null,"fill-rule","nonzero");const l=i.createNS("http://www.w3.org/2000/svg","g");l.setAttributeNS(null,"transform","translate(3.0, 29.0)"),l.setAttributeNS(null,"fill","#000000");const h=[{rx:"10.5",ry:"5.25002273"},{rx:"10.5",ry:"5.25002273"},{rx:"9.5",ry:"4.77275007"},{rx:"8.5",ry:"4.29549936"},{rx:"7.5",ry:"3.81822308"},{rx:"6.5",ry:"3.34094679"},{rx:"5.5",ry:"2.86367051"},{rx:"4.5",ry:"2.38636864"}];for(const t of h){const e=i.createNS("http://www.w3.org/2000/svg","ellipse");e.setAttributeNS(null,"opacity","0.04"),e.setAttributeNS(null,"cx","10.5"),e.setAttributeNS(null,"cy","5.80029008"),e.setAttributeNS(null,"rx",t.rx),e.setAttributeNS(null,"ry",t.ry),l.appendChild(e);}const c=i.createNS("http://www.w3.org/2000/svg","g");c.setAttributeNS(null,"fill",this._color);const u=i.createNS("http://www.w3.org/2000/svg","path");u.setAttributeNS(null,"d","M27,13.5 C27,19.074644 20.250001,27.000002 14.75,34.500002 C14.016665,35.500004 12.983335,35.500004 12.25,34.500002 C6.7499993,27.000002 0,19.222562 0,13.5 C0,6.0441559 6.0441559,0 13.5,0 C20.955844,0 27,6.0441559 27,13.5 Z"),c.appendChild(u);const d=i.createNS("http://www.w3.org/2000/svg","g");d.setAttributeNS(null,"opacity","0.25"),d.setAttributeNS(null,"fill","#000000");const _=i.createNS("http://www.w3.org/2000/svg","path");_.setAttributeNS(null,"d","M13.5,0 C6.0441559,0 0,6.0441559 0,13.5 C0,19.222562 6.7499993,27 12.25,34.5 C13,35.522727 14.016664,35.500004 14.75,34.5 C20.250001,27 27,19.074644 27,13.5 C27,6.0441559 20.955844,0 13.5,0 Z M13.5,1 C20.415404,1 26,6.584596 26,13.5 C26,15.898657 24.495584,19.181431 22.220703,22.738281 C19.945823,26.295132 16.705119,30.142167 13.943359,33.908203 C13.743445,34.180814 13.612715,34.322738 13.5,34.441406 C13.387285,34.322738 13.256555,34.180814 13.056641,33.908203 C10.284481,30.127985 7.4148684,26.314159 5.015625,22.773438 C2.6163816,19.232715 1,15.953538 1,13.5 C1,6.584596 6.584596,1 13.5,1 Z"),d.appendChild(_);const p=i.createNS("http://www.w3.org/2000/svg","g");p.setAttributeNS(null,"transform","translate(6.0, 7.0)"),p.setAttributeNS(null,"fill","#FFFFFF");const m=i.createNS("http://www.w3.org/2000/svg","g");m.setAttributeNS(null,"transform","translate(8.0, 8.0)");const f=i.createNS("http://www.w3.org/2000/svg","circle");f.setAttributeNS(null,"fill","#000000"),f.setAttributeNS(null,"opacity","0.25"),f.setAttributeNS(null,"cx","5.5"),f.setAttributeNS(null,"cy","5.5"),f.setAttributeNS(null,"r","5.4999962");const g=i.createNS("http://www.w3.org/2000/svg","circle");g.setAttributeNS(null,"fill","#FFFFFF"),g.setAttributeNS(null,"cx","5.5"),g.setAttributeNS(null,"cy","5.5"),g.setAttributeNS(null,"r","5.4999962"),m.appendChild(f),m.appendChild(g),n.appendChild(l),n.appendChild(c),n.appendChild(d),n.appendChild(p),n.appendChild(m),s.appendChild(n),s.setAttributeNS(null,"height",a*this._scale+"px"),s.setAttributeNS(null,"width",o*this._scale+"px"),this._element.appendChild(s),this._offset=t.P.convert(e&&e.offset||[0,-14]);}if(this._element.classList.add("maplibregl-marker"),this._element.addEventListener("dragstart",(t=>{t.preventDefault();})),this._element.addEventListener("mousedown",(t=>{t.preventDefault();})),Ia(this._element,this._anchor,"marker"),e&&e.className)for(const t of e.className.split(" "))this._element.classList.add(t);this._popup=null;}addTo(t){return this.remove(),this._map=t,t.getCanvasContainer().appendChild(this._element),t.on("move",this._update),t.on("moveend",this._update),t.on("terrain",this._update),this.setDraggable(this._draggable),this._update(),this._map.on("click",this._onMapClick),this}remove(){return this._opacityTimeout&&(clearTimeout(this._opacityTimeout),delete this._opacityTimeout),this._map&&(this._map.off("click",this._onMapClick),this._map.off("move",this._update),this._map.off("moveend",this._update),this._map.off("mousedown",this._addDragHandler),this._map.off("touchstart",this._addDragHandler),this._map.off("mouseup",this._onUp),this._map.off("touchend",this._onUp),this._map.off("mousemove",this._onMove),this._map.off("touchmove",this._onMove),delete this._map),i.remove(this._element),this._popup&&this._popup.remove(),this}getLngLat(){return this._lngLat}setLngLat(e){return this._lngLat=t.L.convert(e),this._pos=null,this._popup&&this._popup.setLngLat(this._lngLat),this._update(),this}getElement(){return this._element}setPopup(t){if(this._popup&&(this._popup.remove(),this._popup=null,this._element.removeEventListener("keypress",this._onKeyPress),this._originalTabIndex||this._element.removeAttribute("tabindex")),t){if(!("offset"in t.options)){const e=38.1,i=13.5,s=Math.abs(i)/Math.SQRT2;t.options.offset=this._defaultMarker?{top:[0,0],"top-left":[0,0],"top-right":[0,0],bottom:[0,-e],"bottom-left":[s,-1*(e-i+s)],"bottom-right":[-s,-1*(e-i+s)],left:[i,-1*(e-i)],right:[-i,-1*(e-i)]}:this._offset;}this._popup=t,this._lngLat&&this._popup.setLngLat(this._lngLat),this._originalTabIndex=this._element.getAttribute("tabindex"),this._originalTabIndex||this._element.setAttribute("tabindex","0"),this._element.addEventListener("keypress",this._onKeyPress);}return this}getPopup(){return this._popup}togglePopup(){const t=this._popup;return t?(t.isOpen()?t.remove():t.addTo(this._map),this):this}getOffset(){return this._offset}setOffset(e){return this._offset=t.P.convert(e),this._update(),this}addClassName(t){this._element.classList.add(t);}removeClassName(t){this._element.classList.remove(t);}toggleClassName(t){return this._element.classList.toggle(t)}setDraggable(t){return this._draggable=!!t,this._map&&(t?(this._map.on("mousedown",this._addDragHandler),this._map.on("touchstart",this._addDragHandler)):(this._map.off("mousedown",this._addDragHandler),this._map.off("touchstart",this._addDragHandler))),this}isDraggable(){return this._draggable}setRotation(t){return this._rotation=t||0,this._update(),this}getRotation(){return this._rotation}setRotationAlignment(t){return this._rotationAlignment=t||"auto",this._update(),this}getRotationAlignment(){return this._rotationAlignment}setPitchAlignment(t){return this._pitchAlignment=t&&"auto"!==t?t:this._rotationAlignment,this._update(),this}getPitchAlignment(){return this._pitchAlignment}}const Sa={positionOptions:{enableHighAccuracy:!1,maximumAge:0,timeout:6e3},fitBoundsOptions:{maxZoom:15},trackUserLocation:!1,showAccuracyCircle:!0,showUserLocation:!0};let Ca=0,Pa=!1;const Da={maxWidth:100,unit:"metric"};function Ma(t,e,i){const s=i&&i.maxWidth||100,a=t._container.clientHeight/2,o=t.unproject([0,a]),r=t.unproject([s,a]),n=o.distanceTo(r);if(i&&"imperial"===i.unit){const i=3.2808*n;i>5280?za(e,s,i/5280,t._getUIString("ScaleControl.Miles")):za(e,s,i,t._getUIString("ScaleControl.Feet"));}else i&&"nautical"===i.unit?za(e,s,n/1852,t._getUIString("ScaleControl.NauticalMiles")):n>=1e3?za(e,s,n/1e3,t._getUIString("ScaleControl.Kilometers")):za(e,s,n,t._getUIString("ScaleControl.Meters"));}function za(t,e,i,s){const a=function(t){const e=Math.pow(10,`${Math.floor(t)}`.length-1);let i=t/e;return i=i>=10?10:i>=5?5:i>=3?3:i>=2?2:i>=1?1:function(t){const e=Math.pow(10,Math.ceil(-Math.log(t)/Math.LN10));return Math.round(t*e)/e}(i),e*i}(i);t.style.width=e*(a/i)+"px",t.innerHTML=`${a} ${s}`;}const La={closeButton:!0,closeOnClick:!0,focusAfterOpen:!0,className:"",maxWidth:"240px"},Aa=["a[href]","[tabindex]:not([tabindex='-1'])","[contenteditable]:not([contenteditable='false'])","button:not([disabled])","input:not([disabled])","select:not([disabled])","textarea:not([disabled])"].join(", ");function Ra(e){if(e){if("number"==typeof e){const i=Math.round(Math.abs(e)/Math.SQRT2);return {center:new t.P(0,0),top:new t.P(0,e),"top-left":new t.P(i,i),"top-right":new t.P(-i,i),bottom:new t.P(0,-e),"bottom-left":new t.P(i,-i),"bottom-right":new t.P(-i,-i),left:new t.P(e,0),right:new t.P(-e,0)}}if(e instanceof t.P||Array.isArray(e)){const i=t.P.convert(e);return {center:i,top:i,"top-left":i,"top-right":i,bottom:i,"bottom-left":i,"bottom-right":i,left:i,right:i}}return {center:t.P.convert(e.center||[0,0]),top:t.P.convert(e.top||[0,0]),"top-left":t.P.convert(e["top-left"]||[0,0]),"top-right":t.P.convert(e["top-right"]||[0,0]),bottom:t.P.convert(e.bottom||[0,0]),"bottom-left":t.P.convert(e["bottom-left"]||[0,0]),"bottom-right":t.P.convert(e["bottom-right"]||[0,0]),left:t.P.convert(e.left||[0,0]),right:t.P.convert(e.right||[0,0])}}return Ra(new t.P(0,0))}const ka={extend:(e,...i)=>t.e(e,...i),run(t){t();},logToElement(t,e=!1,i="log"){const s=window.document.getElementById(i);s&&(e&&(s.innerHTML=""),s.innerHTML+=`
${t}`);}},Fa=e;class Ba{static get version(){return Fa}static get workerCount(){return tt.workerCount}static set workerCount(t){tt.workerCount=t;}static get maxParallelImageRequests(){return t.c.MAX_PARALLEL_IMAGE_REQUESTS}static set maxParallelImageRequests(e){t.c.MAX_PARALLEL_IMAGE_REQUESTS=e;}static get workerUrl(){return t.c.WORKER_URL}static set workerUrl(e){t.c.WORKER_URL=e;}static addProtocol(e,i){t.c.REGISTERED_PROTOCOLS[e]=i;}static removeProtocol(e){delete t.c.REGISTERED_PROTOCOLS[e];}}return Ba.Map=class extends oa{constructor(e){if(t.bg.mark(t.bh.create),null!=(e=t.e({},ga,e)).minZoom&&null!=e.maxZoom&&e.minZoom>e.maxZoom)throw new Error("maxZoom must be greater than or equal to minZoom");if(null!=e.minPitch&&null!=e.maxPitch&&e.minPitch>e.maxPitch)throw new Error("maxPitch must be greater than or equal to minPitch");if(null!=e.minPitch&&e.minPitch<0)throw new Error("minPitch must be greater than or equal to 0");if(null!=e.maxPitch&&e.maxPitch>85)throw new Error("maxPitch must be less than or equal to 85");if(super(new rs(e.minZoom,e.maxZoom,e.minPitch,e.maxPitch,e.renderWorldCopies),{bearingSnap:e.bearingSnap}),this._cooperativeGesturesOnWheel=t=>{this._onCooperativeGesture(t,t[this._metaKey],1);},this._contextLost=e=>{e.preventDefault(),this._frame&&(this._frame.cancel(),this._frame=null),this.fire(new t.k("webglcontextlost",{originalEvent:e}));},this._contextRestored=e=>{this._setupPainter(),this.resize(),this._update(),this.fire(new t.k("webglcontextrestored",{originalEvent:e}));},this._onMapScroll=t=>{if(t.target===this._container)return this._container.scrollTop=0,this._container.scrollLeft=0,!1},this._onWindowOnline=()=>{this._update();},this._interactive=e.interactive,this._cooperativeGestures=e.cooperativeGestures,this._metaKey=0===navigator.platform.indexOf("Mac")?"metaKey":"ctrlKey",this._maxTileCacheSize=e.maxTileCacheSize,this._maxTileCacheZoomLevels=e.maxTileCacheZoomLevels,this._failIfMajorPerformanceCaveat=e.failIfMajorPerformanceCaveat,this._preserveDrawingBuffer=e.preserveDrawingBuffer,this._antialias=e.antialias,this._trackResize=e.trackResize,this._bearingSnap=e.bearingSnap,this._refreshExpiredTiles=e.refreshExpiredTiles,this._fadeDuration=e.fadeDuration,this._crossSourceCollisions=e.crossSourceCollisions,this._crossFadingFactor=1,this._collectResourceTiming=e.collectResourceTiming,this._renderTaskQueue=new la,this._controls=[],this._mapId=t.a2(),this._locale=t.e({},ha,e.locale),this._clickTolerance=e.clickTolerance,this._overridePixelRatio=e.pixelRatio,this._maxCanvasSize=e.maxCanvasSize,this.transformCameraUpdate=e.transformCameraUpdate,this._imageQueueHandle=h.addThrottleControl((()=>this.isMoving())),this._requestManager=new u(e.transformRequest),"string"==typeof e.container){if(this._container=document.getElementById(e.container),!this._container)throw new Error(`Container '${e.container}' not found.`)}else {if(!(e.container instanceof HTMLElement))throw new Error("Invalid type: 'container' must be a String or HTMLElement.");this._container=e.container;}if(e.maxBounds&&this.setMaxBounds(e.maxBounds),this._setupContainer(),this._setupPainter(),this.on("move",(()=>this._update(!1))),this.on("moveend",(()=>this._update(!1))),this.on("zoom",(()=>this._update(!0))),this.on("terrain",(()=>{this.painter.terrainFacilitator.dirty=!0,this._update(!0);})),this.once("idle",(()=>{this._idleTriggered=!0;})),"undefined"!=typeof window){addEventListener("online",this._onWindowOnline,!1);let t=!1;const e=ns((t=>{this._trackResize&&!this._removed&&this.resize(t)._update();}),50);this._resizeObserver=new ResizeObserver((i=>{t?e(i):t=!0;})),this._resizeObserver.observe(this._container);}this.handlers=new aa(this,e),this._cooperativeGestures&&this._setupCooperativeGestures(),this._hash=e.hash&&new ls("string"==typeof e.hash&&e.hash||void 0).addTo(this),this._hash&&this._hash._onHashChange()||(this.jumpTo({center:e.center,zoom:e.zoom,bearing:e.bearing,pitch:e.pitch}),e.bounds&&(this.resize(),this.fitBounds(e.bounds,t.e({},e.fitBoundsOptions,{duration:0})))),this.resize(),this._localIdeographFontFamily=e.localIdeographFontFamily,this._validateStyle=e.validateStyle,e.style&&this.setStyle(e.style,{localIdeographFontFamily:e.localIdeographFontFamily}),e.attributionControl&&this.addControl(new ra({customAttribution:e.customAttribution})),e.maplibreLogo&&this.addControl(new na,e.logoPosition),this.on("style.load",(()=>{this.transform.unmodified&&this.jumpTo(this.style.stylesheet);})),this.on("data",(e=>{this._update("style"===e.dataType),this.fire(new t.k(`${e.dataType}data`,e));})),this.on("dataloading",(e=>{this.fire(new t.k(`${e.dataType}dataloading`,e));})),this.on("dataabort",(e=>{this.fire(new t.k("sourcedataabort",e));}));}_getMapId(){return this._mapId}addControl(e,i){if(void 0===i&&(i=e.getDefaultPosition?e.getDefaultPosition():"top-right"),!e||!e.onAdd)return this.fire(new t.j(new Error("Invalid argument to map.addControl(). Argument must be a control with onAdd and onRemove methods.")));const s=e.onAdd(this);this._controls.push(e);const a=this._controlPositions[i];return -1!==i.indexOf("bottom")?a.insertBefore(s,a.firstChild):a.appendChild(s),this}removeControl(e){if(!e||!e.onRemove)return this.fire(new t.j(new Error("Invalid argument to map.removeControl(). Argument must be a control with onAdd and onRemove methods.")));const i=this._controls.indexOf(e);return i>-1&&this._controls.splice(i,1),e.onRemove(this),this}hasControl(t){return this._controls.indexOf(t)>-1}calculateCameraOptionsFromTo(t,e,i,s){return null==s&&this.terrain&&(s=this.terrain.getElevationForLngLatZoom(i,this.transform.tileZoom)),super.calculateCameraOptionsFromTo(t,e,i,s)}resize(e){var i;const s=this._containerDimensions(),a=s[0],o=s[1],r=this._getClampedPixelRatio(a,o);if(this._resizeCanvas(a,o,r),this.painter.resize(a,o,r),this.painter.overLimit()){const t=this.painter.context.gl;this._maxCanvasSize=[t.drawingBufferWidth,t.drawingBufferHeight];const e=this._getClampedPixelRatio(a,o);this._resizeCanvas(a,o,e),this.painter.resize(a,o,e);}this.transform.resize(a,o),null===(i=this._requestedCameraState)||void 0===i||i.resize(a,o);const n=!this._moving;return n&&(this.stop(),this.fire(new t.k("movestart",e)).fire(new t.k("move",e))),this.fire(new t.k("resize",e)),n&&this.fire(new t.k("moveend",e)),this}_getClampedPixelRatio(t,e){const{0:i,1:s}=this._maxCanvasSize,a=this.getPixelRatio(),o=t*a,r=e*a;return Math.min(o>i?i/o:1,r>s?s/r:1)*a}getPixelRatio(){var t;return null!==(t=this._overridePixelRatio)&&void 0!==t?t:devicePixelRatio}setPixelRatio(t){this._overridePixelRatio=t,this.resize();}getBounds(){return this.transform.getBounds()}getMaxBounds(){return this.transform.getMaxBounds()}setMaxBounds(t){return this.transform.setMaxBounds(L.convert(t)),this._update()}setMinZoom(t){if((t=null==t?-2:t)>=-2&&t<=this.transform.maxZoom)return this.transform.minZoom=t,this._update(),this.getZoom()=this.transform.minZoom)return this.transform.maxZoom=t,this._update(),this.getZoom()>t&&this.setZoom(t),this;throw new Error("maxZoom must be greater than the current minZoom")}getMaxZoom(){return this.transform.maxZoom}setMinPitch(t){if((t=null==t?0:t)<0)throw new Error("minPitch must be greater than or equal to 0");if(t>=0&&t<=this.transform.maxPitch)return this.transform.minPitch=t,this._update(),this.getPitch()85)throw new Error("maxPitch must be less than or equal to 85");if(t>=this.transform.minPitch)return this.transform.maxPitch=t,this._update(),this.getPitch()>t&&this.setPitch(t),this;throw new Error("maxPitch must be greater than the current minPitch")}getMaxPitch(){return this.transform.maxPitch}getRenderWorldCopies(){return this.transform.renderWorldCopies}setRenderWorldCopies(t){return this.transform.renderWorldCopies=t,this._update()}getCooperativeGestures(){return this._cooperativeGestures}setCooperativeGestures(t){return this._cooperativeGestures=t,this._cooperativeGestures?this._setupCooperativeGestures():this._destroyCooperativeGestures(),this}project(e){return this.transform.locationPoint(t.L.convert(e),this.style&&this.terrain)}unproject(e){return this.transform.pointLocation(t.P.convert(e),this.terrain)}isMoving(){var t;return this._moving||(null===(t=this.handlers)||void 0===t?void 0:t.isMoving())}isZooming(){var t;return this._zooming||(null===(t=this.handlers)||void 0===t?void 0:t.isZooming())}isRotating(){var t;return this._rotating||(null===(t=this.handlers)||void 0===t?void 0:t.isRotating())}_createDelegatedListener(t,e,i){if("mouseenter"===t||"mouseover"===t){let s=!1;const a=a=>{const o=this.getLayer(e)?this.queryRenderedFeatures(a.point,{layers:[e]}):[];o.length?s||(s=!0,i.call(this,new gs(t,this,a.originalEvent,{features:o}))):s=!1;};return {layer:e,listener:i,delegates:{mousemove:a,mouseout:()=>{s=!1;}}}}if("mouseleave"===t||"mouseout"===t){let s=!1;const a=a=>{(this.getLayer(e)?this.queryRenderedFeatures(a.point,{layers:[e]}):[]).length?s=!0:s&&(s=!1,i.call(this,new gs(t,this,a.originalEvent)));},o=e=>{s&&(s=!1,i.call(this,new gs(t,this,e.originalEvent)));};return {layer:e,listener:i,delegates:{mousemove:a,mouseout:o}}}{const s=t=>{const s=this.getLayer(e)?this.queryRenderedFeatures(t.point,{layers:[e]}):[];s.length&&(t.features=s,i.call(this,t),delete t.features);};return {layer:e,listener:i,delegates:{[t]:s}}}}on(t,e,i){if(void 0===i)return super.on(t,e);const s=this._createDelegatedListener(t,e,i);this._delegatedListeners=this._delegatedListeners||{},this._delegatedListeners[t]=this._delegatedListeners[t]||[],this._delegatedListeners[t].push(s);for(const t in s.delegates)this.on(t,s.delegates[t]);return this}once(t,e,i){if(void 0===i)return super.once(t,e);const s=this._createDelegatedListener(t,e,i);for(const t in s.delegates)this.once(t,s.delegates[t]);return this}off(t,e,i){return void 0===i?super.off(t,e):(this._delegatedListeners&&this._delegatedListeners[t]&&(s=>{const a=this._delegatedListeners[t];for(let t=0;tthis._updateStyle(t,e)));const i=this.style&&e.transformStyle?this.style.serialize():void 0;return this.style&&(this.style.setEventedParent(null),this.style._remove(!t)),t?(this.style=new se(this,e||{}),this.style.setEventedParent(this,{style:this.style}),"string"==typeof t?this.style.loadURL(t,e,i):this.style.loadJSON(t,e,i),this):(delete this.style,this)}_lazyInitEmptyStyle(){this.style||(this.style=new se(this,{}),this.style.setEventedParent(this,{style:this.style}),this.style.loadEmpty());}_diffStyle(e,i){if("string"==typeof e){const s=this._requestManager.transformRequest(e,c.Style);t.f(s,((e,s)=>{e?this.fire(new t.j(e)):s&&this._updateDiff(s,i);}));}else "object"==typeof e&&this._updateDiff(e,i);}_updateDiff(e,i){try{this.style.setState(e,i)&&this._update(!0);}catch(s){t.w(`Unable to perform style diff: ${s.message||s.error||s}. Rebuilding the style from scratch.`),this._updateStyle(e,i);}}getStyle(){if(this.style)return this.style.serialize()}isStyleLoaded(){return this.style?this.style.loaded():t.w("There is no style added to the map.")}addSource(t,e){return this._lazyInitEmptyStyle(),this.style.addSource(t,e),this._update(!0)}isSourceLoaded(e){const i=this.style&&this.style.sourceCaches[e];if(void 0!==i)return i.loaded();this.fire(new t.j(new Error(`There is no source with ID '${e}'`)));}setTerrain(e){if(this.style._checkLoaded(),this._terrainDataCallback&&this.style.off("data",this._terrainDataCallback),e){const i=this.style.sourceCaches[e.source];if(!i)throw new Error(`cannot load terrain, because there exists no source with ID: ${e.source}`);for(const i in this.style._layers){const s=this.style._layers[i];"hillshade"===s.type&&s.source===e.source&&t.w("You are using the same source for a hillshade layer and for 3D terrain. Please consider using two separate sources to improve rendering quality.");}this.terrain=new da(this.painter,i,e),this.painter.renderToTexture=new ma(this.painter,this.terrain),this.transform._minEleveationForCurrentTile=this.terrain.getMinTileElevationForLngLatZoom(this.transform.center,this.transform.tileZoom),this.transform.elevation=this.terrain.getElevationForLngLatZoom(this.transform.center,this.transform.tileZoom),this._terrainDataCallback=t=>{"style"===t.dataType?this.terrain.sourceCache.freeRtt():"source"===t.dataType&&t.tile&&(t.sourceId!==e.source||this._elevationFreeze||(this.transform._minEleveationForCurrentTile=this.terrain.getMinTileElevationForLngLatZoom(this.transform.center,this.transform.tileZoom),this.transform.elevation=this.terrain.getElevationForLngLatZoom(this.transform.center,this.transform.tileZoom)),this.terrain.sourceCache.freeRtt(t.tile.tileID));},this.style.on("data",this._terrainDataCallback);}else this.terrain&&this.terrain.sourceCache.destruct(),this.terrain=null,this.painter.renderToTexture&&this.painter.renderToTexture.destruct(),this.painter.renderToTexture=null,this.transform._minEleveationForCurrentTile=0,this.transform.elevation=0;return this.fire(new t.k("terrain",{terrain:e})),this}getTerrain(){var t,e;return null!==(e=null===(t=this.terrain)||void 0===t?void 0:t.options)&&void 0!==e?e:null}areTilesLoaded(){const t=this.style&&this.style.sourceCaches;for(const e in t){const i=t[e]._tiles;for(const t in i){const e=i[t];if("loaded"!==e.state&&"errored"!==e.state)return !1}}return !0}addSourceType(t,e,i){return this._lazyInitEmptyStyle(),this.style.addSourceType(t,e,i)}removeSource(t){return this.style.removeSource(t),this._update(!0)}getSource(t){return this.style.getSource(t)}addImage(e,i,s={}){const{pixelRatio:a=1,sdf:o=!1,stretchX:r,stretchY:n,content:l}=s;if(this._lazyInitEmptyStyle(),!(i instanceof HTMLImageElement||t.a(i))){if(void 0===i.width||void 0===i.height)return this.fire(new t.j(new Error("Invalid arguments to map.addImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, or object with `width`, `height`, and `data` properties with the same format as `ImageData`")));{const{width:s,height:h,data:c}=i,u=i;return this.style.addImage(e,{data:new t.R({width:s,height:h},new Uint8Array(c)),pixelRatio:a,stretchX:r,stretchY:n,content:l,sdf:o,version:0,userImage:u}),u.onAdd&&u.onAdd(this,e),this}}{const{width:s,height:h,data:c}=t.h.getImageData(i);this.style.addImage(e,{data:new t.R({width:s,height:h},c),pixelRatio:a,stretchX:r,stretchY:n,content:l,sdf:o,version:0});}}updateImage(e,i){const s=this.style.getImage(e);if(!s)return this.fire(new t.j(new Error("The map has no image with that id. If you are adding a new image use `map.addImage(...)` instead.")));const a=i instanceof HTMLImageElement||t.a(i)?t.h.getImageData(i):i,{width:o,height:r,data:n}=a;if(void 0===o||void 0===r)return this.fire(new t.j(new Error("Invalid arguments to map.updateImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, or object with `width`, `height`, and `data` properties with the same format as `ImageData`")));if(o!==s.data.width||r!==s.data.height)return this.fire(new t.j(new Error("The width and height of the updated image must be that same as the previous version of the image")));const l=!(i instanceof HTMLImageElement||t.a(i));return s.data.replace(n,l),this.style.updateImage(e,s),this}getImage(t){return this.style.getImage(t)}hasImage(e){return e?!!this.style.getImage(e):(this.fire(new t.j(new Error("Missing required image id"))),!1)}removeImage(t){this.style.removeImage(t);}loadImage(t,e){h.getImage(this._requestManager.transformRequest(t,c.Image),e);}listImages(){return this.style.listImages()}addLayer(t,e){return this._lazyInitEmptyStyle(),this.style.addLayer(t,e),this._update(!0)}moveLayer(t,e){return this.style.moveLayer(t,e),this._update(!0)}removeLayer(t){return this.style.removeLayer(t),this._update(!0)}getLayer(t){return this.style.getLayer(t)}getLayersOrder(){return this.style.getLayersOrder()}setLayerZoomRange(t,e,i){return this.style.setLayerZoomRange(t,e,i),this._update(!0)}setFilter(t,e,i={}){return this.style.setFilter(t,e,i),this._update(!0)}getFilter(t){return this.style.getFilter(t)}setPaintProperty(t,e,i,s={}){return this.style.setPaintProperty(t,e,i,s),this._update(!0)}getPaintProperty(t,e){return this.style.getPaintProperty(t,e)}setLayoutProperty(t,e,i,s={}){return this.style.setLayoutProperty(t,e,i,s),this._update(!0)}getLayoutProperty(t,e){return this.style.getLayoutProperty(t,e)}setGlyphs(t,e={}){return this._lazyInitEmptyStyle(),this.style.setGlyphs(t,e),this._update(!0)}getGlyphs(){return this.style.getGlyphsUrl()}addSprite(t,e,i={}){return this._lazyInitEmptyStyle(),this.style.addSprite(t,e,i,(t=>{t||this._update(!0);})),this}removeSprite(t){return this._lazyInitEmptyStyle(),this.style.removeSprite(t),this._update(!0)}getSprite(){return this.style.getSprite()}setSprite(t,e={}){return this._lazyInitEmptyStyle(),this.style.setSprite(t,e,(t=>{t||this._update(!0);})),this}setLight(t,e={}){return this._lazyInitEmptyStyle(),this.style.setLight(t,e),this._update(!0)}getLight(){return this.style.getLight()}setFeatureState(t,e){return this.style.setFeatureState(t,e),this._update()}removeFeatureState(t,e){return this.style.removeFeatureState(t,e),this._update()}getFeatureState(t){return this.style.getFeatureState(t)}getContainer(){return this._container}getCanvasContainer(){return this._canvasContainer}getCanvas(){return this._canvas}_containerDimensions(){let t=0,e=0;return this._container&&(t=this._container.clientWidth||400,e=this._container.clientHeight||300),[t,e]}_setupContainer(){const t=this._container;t.classList.add("maplibregl-map");const e=this._canvasContainer=i.create("div","maplibregl-canvas-container",t);this._interactive&&e.classList.add("maplibregl-interactive"),this._canvas=i.create("canvas","maplibregl-canvas",e),this._canvas.addEventListener("webglcontextlost",this._contextLost,!1),this._canvas.addEventListener("webglcontextrestored",this._contextRestored,!1),this._canvas.setAttribute("tabindex","0"),this._canvas.setAttribute("aria-label","Map"),this._canvas.setAttribute("role","region");const s=this._containerDimensions(),a=this._getClampedPixelRatio(s[0],s[1]);this._resizeCanvas(s[0],s[1],a);const o=this._controlContainer=i.create("div","maplibregl-control-container",t),r=this._controlPositions={};["top-left","top-right","bottom-left","bottom-right"].forEach((t=>{r[t]=i.create("div",`maplibregl-ctrl-${t} `,o);})),this._container.addEventListener("scroll",this._onMapScroll,!1);}_setupCooperativeGestures(){this._cooperativeGesturesScreen=i.create("div","maplibregl-cooperative-gesture-screen",this._container);let t="boolean"!=typeof this._cooperativeGestures&&this._cooperativeGestures.windowsHelpText?this._cooperativeGestures.windowsHelpText:"Use Ctrl + scroll to zoom the map";0===navigator.platform.indexOf("Mac")&&(t="boolean"!=typeof this._cooperativeGestures&&this._cooperativeGestures.macHelpText?this._cooperativeGestures.macHelpText:"Use ⌘ + scroll to zoom the map"),this._cooperativeGesturesScreen.innerHTML=`\n
${t}
\n
${"boolean"!=typeof this._cooperativeGestures&&this._cooperativeGestures.mobileHelpText?this._cooperativeGestures.mobileHelpText:"Use two fingers to move the map"}
\n `,this._cooperativeGesturesScreen.setAttribute("aria-hidden","true"),this._canvasContainer.addEventListener("wheel",this._cooperativeGesturesOnWheel,!1),this._canvasContainer.classList.add("maplibregl-cooperative-gestures");}_destroyCooperativeGestures(){i.remove(this._cooperativeGesturesScreen),this._canvasContainer.removeEventListener("wheel",this._cooperativeGesturesOnWheel,!1),this._canvasContainer.classList.remove("maplibregl-cooperative-gestures");}_resizeCanvas(t,e,i){this._canvas.width=Math.floor(i*t),this._canvas.height=Math.floor(i*e),this._canvas.style.width=`${t}px`,this._canvas.style.height=`${e}px`;}_setupPainter(){const t={alpha:!0,stencil:!0,depth:!0,failIfMajorPerformanceCaveat:this._failIfMajorPerformanceCaveat,preserveDrawingBuffer:this._preserveDrawingBuffer,antialias:this._antialias||!1};let e=null;this._canvas.addEventListener("webglcontextcreationerror",(i=>{e={requestedAttributes:t},i&&(e.statusMessage=i.statusMessage,e.type=i.type);}),{once:!0});const i=this._canvas.getContext("webgl2",t)||this._canvas.getContext("webgl",t);if(!i){const t="Failed to initialize WebGL";throw e?(e.message=t,new Error(JSON.stringify(e))):new Error(t)}this.painter=new is(i,this.transform),s.testSupport(i);}_onCooperativeGesture(t,e,i){return !e&&i<2&&(this._cooperativeGesturesScreen.classList.add("maplibregl-show"),setTimeout((()=>{this._cooperativeGesturesScreen.classList.remove("maplibregl-show");}),100)),!1}loaded(){return !this._styleDirty&&!this._sourcesDirty&&!!this.style&&this.style.loaded()}_update(t){return this.style&&this.style._loaded?(this._styleDirty=this._styleDirty||t,this._sourcesDirty=!0,this.triggerRepaint(),this):this}_requestRenderFrame(t){return this._update(),this._renderTaskQueue.add(t)}_cancelRenderFrame(t){this._renderTaskQueue.remove(t);}_render(e){const i=this._idleTriggered?this._fadeDuration:0;if(this.painter.context.setDirty(),this.painter.setBaseState(),this._renderTaskQueue.run(e),this._removed)return;let s=!1;if(this.style&&this._styleDirty){this._styleDirty=!1;const e=this.transform.zoom,a=t.h.now();this.style.zoomHistory.update(e,a);const o=new t.a8(e,{now:a,fadeDuration:i,zoomHistory:this.style.zoomHistory,transition:this.style.getTransition()}),r=o.crossFadingFactor();1===r&&r===this._crossFadingFactor||(s=!0,this._crossFadingFactor=r),this.style.update(o);}this.style&&this._sourcesDirty&&(this._sourcesDirty=!1,this.style._updateSources(this.transform)),this.terrain?(this.terrain.sourceCache.update(this.transform,this.terrain),this.transform._minEleveationForCurrentTile=this.terrain.getMinTileElevationForLngLatZoom(this.transform.center,this.transform.tileZoom),this._elevationFreeze||(this.transform.elevation=this.terrain.getElevationForLngLatZoom(this.transform.center,this.transform.tileZoom))):(this.transform._minEleveationForCurrentTile=0,this.transform.elevation=0),this._placementDirty=this.style&&this.style._updatePlacement(this.painter.transform,this.showCollisionBoxes,i,this._crossSourceCollisions),this.painter.render(this.style,{showTileBoundaries:this.showTileBoundaries,showOverdrawInspector:this._showOverdrawInspector,rotating:this.isRotating(),zooming:this.isZooming(),moving:this.isMoving(),fadeDuration:i,showPadding:this.showPadding}),this.fire(new t.k("render")),this.loaded()&&!this._loaded&&(this._loaded=!0,t.bg.mark(t.bh.load),this.fire(new t.k("load"))),this.style&&(this.style.hasTransitions()||s)&&(this._styleDirty=!0),this.style&&!this._placementDirty&&this.style._releaseSymbolFadeTiles();const a=this._sourcesDirty||this._styleDirty||this._placementDirty;return a||this._repaint?this.triggerRepaint():!this.isMoving()&&this.loaded()&&this.fire(new t.k("idle")),!this._loaded||this._fullyLoaded||a||(this._fullyLoaded=!0,t.bg.mark(t.bh.fullLoad)),this}redraw(){return this.style&&(this._frame&&(this._frame.cancel(),this._frame=null),this._render(0)),this}remove(){var e;this._hash&&this._hash.remove();for(const t of this._controls)t.onRemove(this);this._controls=[],this._frame&&(this._frame.cancel(),this._frame=null),this._renderTaskQueue.clear(),this.painter.destroy(),this.handlers.destroy(),delete this.handlers,this.setStyle(null),"undefined"!=typeof window&&removeEventListener("online",this._onWindowOnline,!1),h.removeThrottleControl(this._imageQueueHandle),null===(e=this._resizeObserver)||void 0===e||e.disconnect();const s=this.painter.context.gl.getExtension("WEBGL_lose_context");s&&s.loseContext(),this._canvas.removeEventListener("webglcontextrestored",this._contextRestored,!1),this._canvas.removeEventListener("webglcontextlost",this._contextLost,!1),i.remove(this._canvasContainer),i.remove(this._controlContainer),this._cooperativeGestures&&this._destroyCooperativeGestures(),this._container.classList.remove("maplibregl-map"),t.bg.clearMetrics(),this._removed=!0,this.fire(new t.k("remove"));}triggerRepaint(){this.style&&!this._frame&&(this._frame=t.h.frame((e=>{t.bg.frame(e),this._frame=null,this._render(e);})));}get showTileBoundaries(){return !!this._showTileBoundaries}set showTileBoundaries(t){this._showTileBoundaries!==t&&(this._showTileBoundaries=t,this._update());}get showPadding(){return !!this._showPadding}set showPadding(t){this._showPadding!==t&&(this._showPadding=t,this._update());}get showCollisionBoxes(){return !!this._showCollisionBoxes}set showCollisionBoxes(t){this._showCollisionBoxes!==t&&(this._showCollisionBoxes=t,t?this.style._generateCollisionBoxes():this._update());}get showOverdrawInspector(){return !!this._showOverdrawInspector}set showOverdrawInspector(t){this._showOverdrawInspector!==t&&(this._showOverdrawInspector=t,this._update());}get repaint(){return !!this._repaint}set repaint(t){this._repaint!==t&&(this._repaint=t,this.triggerRepaint());}get vertices(){return !!this._vertices}set vertices(t){this._vertices=t,this._update();}get version(){return fa}getCameraTargetElevation(){return this.transform.elevation}},Ba.NavigationControl=class{constructor(e){this._updateZoomButtons=()=>{const t=this._map.getZoom(),e=t===this._map.getMaxZoom(),i=t===this._map.getMinZoom();this._zoomInButton.disabled=e,this._zoomOutButton.disabled=i,this._zoomInButton.setAttribute("aria-disabled",e.toString()),this._zoomOutButton.setAttribute("aria-disabled",i.toString());},this._rotateCompassArrow=()=>{const t=this.options.visualizePitch?`scale(${1/Math.pow(Math.cos(this._map.transform.pitch*(Math.PI/180)),.5)}) rotateX(${this._map.transform.pitch}deg) rotateZ(${this._map.transform.angle*(180/Math.PI)}deg)`:`rotate(${this._map.transform.angle*(180/Math.PI)}deg)`;this._compassIcon.style.transform=t;},this._setButtonTitle=(t,e)=>{const i=this._map._getUIString(`NavigationControl.${e}`);t.title=i,t.setAttribute("aria-label",i);},this.options=t.e({},xa,e),this._container=i.create("div","maplibregl-ctrl maplibregl-ctrl-group"),this._container.addEventListener("contextmenu",(t=>t.preventDefault())),this.options.showZoom&&(this._zoomInButton=this._createButton("maplibregl-ctrl-zoom-in",(t=>this._map.zoomIn({},{originalEvent:t}))),i.create("span","maplibregl-ctrl-icon",this._zoomInButton).setAttribute("aria-hidden","true"),this._zoomOutButton=this._createButton("maplibregl-ctrl-zoom-out",(t=>this._map.zoomOut({},{originalEvent:t}))),i.create("span","maplibregl-ctrl-icon",this._zoomOutButton).setAttribute("aria-hidden","true")),this.options.showCompass&&(this._compass=this._createButton("maplibregl-ctrl-compass",(t=>{this.options.visualizePitch?this._map.resetNorthPitch({},{originalEvent:t}):this._map.resetNorth({},{originalEvent:t});})),this._compassIcon=i.create("span","maplibregl-ctrl-icon",this._compass),this._compassIcon.setAttribute("aria-hidden","true"));}onAdd(t){return this._map=t,this.options.showZoom&&(this._setButtonTitle(this._zoomInButton,"ZoomIn"),this._setButtonTitle(this._zoomOutButton,"ZoomOut"),this._map.on("zoom",this._updateZoomButtons),this._updateZoomButtons()),this.options.showCompass&&(this._setButtonTitle(this._compass,"ResetBearing"),this.options.visualizePitch&&this._map.on("pitch",this._rotateCompassArrow),this._map.on("rotate",this._rotateCompassArrow),this._rotateCompassArrow(),this._handler=new ya(this._map,this._compass,this.options.visualizePitch)),this._container}onRemove(){i.remove(this._container),this.options.showZoom&&this._map.off("zoom",this._updateZoomButtons),this.options.showCompass&&(this.options.visualizePitch&&this._map.off("pitch",this._rotateCompassArrow),this._map.off("rotate",this._rotateCompassArrow),this._handler.off(),delete this._handler),delete this._map;}_createButton(t,e){const s=i.create("button",t,this._container);return s.type="button",s.addEventListener("click",e),s}},Ba.GeolocateControl=class extends t.E{constructor(e){super(),this._onSuccess=e=>{if(this._map){if(this._isOutOfMapMaxBounds(e))return this._setErrorState(),this.fire(new t.k("outofmaxbounds",e)),this._updateMarker(),void this._finish();if(this.options.trackUserLocation)switch(this._lastKnownPosition=e,this._watchState){case"WAITING_ACTIVE":case"ACTIVE_LOCK":case"ACTIVE_ERROR":this._watchState="ACTIVE_LOCK",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active-error"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-active");break;case"BACKGROUND":case"BACKGROUND_ERROR":this._watchState="BACKGROUND",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background-error"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-background");break;default:throw new Error(`Unexpected watchState ${this._watchState}`)}this.options.showUserLocation&&"OFF"!==this._watchState&&this._updateMarker(e),this.options.trackUserLocation&&"ACTIVE_LOCK"!==this._watchState||this._updateCamera(e),this.options.showUserLocation&&this._dotElement.classList.remove("maplibregl-user-location-dot-stale"),this.fire(new t.k("geolocate",e)),this._finish();}},this._updateCamera=e=>{const i=new t.L(e.coords.longitude,e.coords.latitude),s=e.coords.accuracy,a=this._map.getBearing(),o=t.e({bearing:a},this.options.fitBoundsOptions),r=L.fromLngLat(i,s);this._map.fitBounds(r,o,{geolocateSource:!0});},this._updateMarker=e=>{if(e){const i=new t.L(e.coords.longitude,e.coords.latitude);this._accuracyCircleMarker.setLngLat(i).addTo(this._map),this._userLocationDotMarker.setLngLat(i).addTo(this._map),this._accuracy=e.coords.accuracy,this.options.showUserLocation&&this.options.showAccuracyCircle&&this._updateCircleRadius();}else this._userLocationDotMarker.remove(),this._accuracyCircleMarker.remove();},this._onZoom=()=>{this.options.showUserLocation&&this.options.showAccuracyCircle&&this._updateCircleRadius();},this._onError=e=>{if(this._map){if(this.options.trackUserLocation)if(1===e.code){this._watchState="OFF",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active-error"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background-error"),this._geolocateButton.disabled=!0;const t=this._map._getUIString("GeolocateControl.LocationNotAvailable");this._geolocateButton.title=t,this._geolocateButton.setAttribute("aria-label",t),void 0!==this._geolocationWatchID&&this._clearWatch();}else {if(3===e.code&&Pa)return;this._setErrorState();}"OFF"!==this._watchState&&this.options.showUserLocation&&this._dotElement.classList.add("maplibregl-user-location-dot-stale"),this.fire(new t.k("error",e)),this._finish();}},this._finish=()=>{this._timeoutId&&clearTimeout(this._timeoutId),this._timeoutId=void 0;},this._setupUI=e=>{if(this._map){if(this._container.addEventListener("contextmenu",(t=>t.preventDefault())),this._geolocateButton=i.create("button","maplibregl-ctrl-geolocate",this._container),i.create("span","maplibregl-ctrl-icon",this._geolocateButton).setAttribute("aria-hidden","true"),this._geolocateButton.type="button",!1===e){t.w("Geolocation support is not available so the GeolocateControl will be disabled.");const e=this._map._getUIString("GeolocateControl.LocationNotAvailable");this._geolocateButton.disabled=!0,this._geolocateButton.title=e,this._geolocateButton.setAttribute("aria-label",e);}else {const t=this._map._getUIString("GeolocateControl.FindMyLocation");this._geolocateButton.title=t,this._geolocateButton.setAttribute("aria-label",t);}this.options.trackUserLocation&&(this._geolocateButton.setAttribute("aria-pressed","false"),this._watchState="OFF"),this.options.showUserLocation&&(this._dotElement=i.create("div","maplibregl-user-location-dot"),this._userLocationDotMarker=new Ea({element:this._dotElement}),this._circleElement=i.create("div","maplibregl-user-location-accuracy-circle"),this._accuracyCircleMarker=new Ea({element:this._circleElement,pitchAlignment:"map"}),this.options.trackUserLocation&&(this._watchState="OFF"),this._map.on("zoom",this._onZoom)),this._geolocateButton.addEventListener("click",this.trigger.bind(this)),this._setup=!0,this.options.trackUserLocation&&this._map.on("movestart",(e=>{e.geolocateSource||"ACTIVE_LOCK"!==this._watchState||e.originalEvent&&"resize"===e.originalEvent.type||(this._watchState="BACKGROUND",this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-background"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active"),this.fire(new t.k("trackuserlocationend")));}));}},this.options=t.e({},Sa,e);}onAdd(t){return this._map=t,this._container=i.create("div","maplibregl-ctrl maplibregl-ctrl-group"),function(t,e=!1){void 0===ba||e?void 0!==window.navigator.permissions?window.navigator.permissions.query({name:"geolocation"}).then((e=>{ba="denied"!==e.state,t(ba);})).catch((()=>{ba=!!window.navigator.geolocation,t(ba);})):(ba=!!window.navigator.geolocation,t(ba)):t(ba);}(this._setupUI),this._container}onRemove(){void 0!==this._geolocationWatchID&&(window.navigator.geolocation.clearWatch(this._geolocationWatchID),this._geolocationWatchID=void 0),this.options.showUserLocation&&this._userLocationDotMarker&&this._userLocationDotMarker.remove(),this.options.showAccuracyCircle&&this._accuracyCircleMarker&&this._accuracyCircleMarker.remove(),i.remove(this._container),this._map.off("zoom",this._onZoom),this._map=void 0,Ca=0,Pa=!1;}_isOutOfMapMaxBounds(t){const e=this._map.getMaxBounds(),i=t.coords;return e&&(i.longitudee.getEast()||i.latitudee.getNorth())}_setErrorState(){switch(this._watchState){case"WAITING_ACTIVE":this._watchState="ACTIVE_ERROR",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-active-error");break;case"ACTIVE_LOCK":this._watchState="ACTIVE_ERROR",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-active-error"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-waiting");break;case"BACKGROUND":this._watchState="BACKGROUND_ERROR",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-background-error"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-waiting");break;case"ACTIVE_ERROR":break;default:throw new Error(`Unexpected watchState ${this._watchState}`)}}_updateCircleRadius(){const t=this._map.getBounds(),e=t.getSouthEast(),i=t.getNorthEast(),s=e.distanceTo(i),a=Math.ceil(this._accuracy/(s/this._map._container.clientHeight)*2);this._circleElement.style.width=`${a}px`,this._circleElement.style.height=`${a}px`;}trigger(){if(!this._setup)return t.w("Geolocate control triggered before added to a map"),!1;if(this.options.trackUserLocation){switch(this._watchState){case"OFF":this._watchState="WAITING_ACTIVE",this.fire(new t.k("trackuserlocationstart"));break;case"WAITING_ACTIVE":case"ACTIVE_LOCK":case"ACTIVE_ERROR":case"BACKGROUND_ERROR":Ca--,Pa=!1,this._watchState="OFF",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-active-error"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background"),this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background-error"),this.fire(new t.k("trackuserlocationend"));break;case"BACKGROUND":this._watchState="ACTIVE_LOCK",this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-background"),this._lastKnownPosition&&this._updateCamera(this._lastKnownPosition),this.fire(new t.k("trackuserlocationstart"));break;default:throw new Error(`Unexpected watchState ${this._watchState}`)}switch(this._watchState){case"WAITING_ACTIVE":this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-active");break;case"ACTIVE_LOCK":this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-active");break;case"OFF":break;default:throw new Error(`Unexpected watchState ${this._watchState}`)}if("OFF"===this._watchState&&void 0!==this._geolocationWatchID)this._clearWatch();else if(void 0===this._geolocationWatchID){let t;this._geolocateButton.classList.add("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.setAttribute("aria-pressed","true"),Ca++,Ca>1?(t={maximumAge:6e5,timeout:0},Pa=!0):(t=this.options.positionOptions,Pa=!1),this._geolocationWatchID=window.navigator.geolocation.watchPosition(this._onSuccess,this._onError,t);}}else window.navigator.geolocation.getCurrentPosition(this._onSuccess,this._onError,this.options.positionOptions),this._timeoutId=setTimeout(this._finish,1e4);return !0}_clearWatch(){window.navigator.geolocation.clearWatch(this._geolocationWatchID),this._geolocationWatchID=void 0,this._geolocateButton.classList.remove("maplibregl-ctrl-geolocate-waiting"),this._geolocateButton.setAttribute("aria-pressed","false"),this.options.showUserLocation&&this._updateMarker(null);}},Ba.AttributionControl=ra,Ba.LogoControl=na,Ba.ScaleControl=class{constructor(e){this._onMove=()=>{Ma(this._map,this._container,this.options);},this.setUnit=t=>{this.options.unit=t,Ma(this._map,this._container,this.options);},this.options=t.e({},Da,e);}getDefaultPosition(){return "bottom-left"}onAdd(t){return this._map=t,this._container=i.create("div","maplibregl-ctrl maplibregl-ctrl-scale",t.getContainer()),this._map.on("move",this._onMove),this._onMove(),this._container}onRemove(){i.remove(this._container),this._map.off("move",this._onMove),this._map=void 0;}},Ba.FullscreenControl=class extends t.E{constructor(e={}){super(),this._onFullscreenChange=()=>{(window.document.fullscreenElement||window.document.mozFullScreenElement||window.document.webkitFullscreenElement||window.document.msFullscreenElement)===this._container!==this._fullscreen&&this._handleFullscreenChange();},this._onClickFullscreen=()=>{this._isFullscreen()?this._exitFullscreen():this._requestFullscreen();},this._fullscreen=!1,e&&e.container&&(e.container instanceof HTMLElement?this._container=e.container:t.w("Full screen control 'container' must be a DOM element.")),"onfullscreenchange"in document?this._fullscreenchange="fullscreenchange":"onmozfullscreenchange"in document?this._fullscreenchange="mozfullscreenchange":"onwebkitfullscreenchange"in document?this._fullscreenchange="webkitfullscreenchange":"onmsfullscreenchange"in document&&(this._fullscreenchange="MSFullscreenChange");}onAdd(t){return this._map=t,this._container||(this._container=this._map.getContainer()),this._controlContainer=i.create("div","maplibregl-ctrl maplibregl-ctrl-group"),this._setupUI(),this._controlContainer}onRemove(){i.remove(this._controlContainer),this._map=null,window.document.removeEventListener(this._fullscreenchange,this._onFullscreenChange);}_setupUI(){const t=this._fullscreenButton=i.create("button","maplibregl-ctrl-fullscreen",this._controlContainer);i.create("span","maplibregl-ctrl-icon",t).setAttribute("aria-hidden","true"),t.type="button",this._updateTitle(),this._fullscreenButton.addEventListener("click",this._onClickFullscreen),window.document.addEventListener(this._fullscreenchange,this._onFullscreenChange);}_updateTitle(){const t=this._getTitle();this._fullscreenButton.setAttribute("aria-label",t),this._fullscreenButton.title=t;}_getTitle(){return this._map._getUIString(this._isFullscreen()?"FullscreenControl.Exit":"FullscreenControl.Enter")}_isFullscreen(){return this._fullscreen}_handleFullscreenChange(){this._fullscreen=!this._fullscreen,this._fullscreenButton.classList.toggle("maplibregl-ctrl-shrink"),this._fullscreenButton.classList.toggle("maplibregl-ctrl-fullscreen"),this._updateTitle(),this._fullscreen?(this.fire(new t.k("fullscreenstart")),this._map._cooperativeGestures&&(this._prevCooperativeGestures=this._map._cooperativeGestures,this._map.setCooperativeGestures())):(this.fire(new t.k("fullscreenend")),this._prevCooperativeGestures&&(this._map.setCooperativeGestures(this._prevCooperativeGestures),delete this._prevCooperativeGestures));}_exitFullscreen(){window.document.exitFullscreen?window.document.exitFullscreen():window.document.mozCancelFullScreen?window.document.mozCancelFullScreen():window.document.msExitFullscreen?window.document.msExitFullscreen():window.document.webkitCancelFullScreen?window.document.webkitCancelFullScreen():this._togglePseudoFullScreen();}_requestFullscreen(){this._container.requestFullscreen?this._container.requestFullscreen():this._container.mozRequestFullScreen?this._container.mozRequestFullScreen():this._container.msRequestFullscreen?this._container.msRequestFullscreen():this._container.webkitRequestFullscreen?this._container.webkitRequestFullscreen():this._togglePseudoFullScreen();}_togglePseudoFullScreen(){this._container.classList.toggle("maplibregl-pseudo-fullscreen"),this._handleFullscreenChange(),this._map.resize();}},Ba.TerrainControl=class{constructor(t){this._toggleTerrain=()=>{this._map.getTerrain()?this._map.setTerrain(null):this._map.setTerrain(this.options),this._updateTerrainIcon();},this._updateTerrainIcon=()=>{this._terrainButton.classList.remove("maplibregl-ctrl-terrain"),this._terrainButton.classList.remove("maplibregl-ctrl-terrain-enabled"),this._map.terrain?(this._terrainButton.classList.add("maplibregl-ctrl-terrain-enabled"),this._terrainButton.title=this._map._getUIString("TerrainControl.disableTerrain")):(this._terrainButton.classList.add("maplibregl-ctrl-terrain"),this._terrainButton.title=this._map._getUIString("TerrainControl.enableTerrain"));},this.options=t;}onAdd(t){return this._map=t,this._container=i.create("div","maplibregl-ctrl maplibregl-ctrl-group"),this._terrainButton=i.create("button","maplibregl-ctrl-terrain",this._container),i.create("span","maplibregl-ctrl-icon",this._terrainButton).setAttribute("aria-hidden","true"),this._terrainButton.type="button",this._terrainButton.addEventListener("click",this._toggleTerrain),this._updateTerrainIcon(),this._map.on("terrain",this._updateTerrainIcon),this._container}onRemove(){i.remove(this._container),this._map.off("terrain",this._updateTerrainIcon),this._map=void 0;}},Ba.Popup=class extends t.E{constructor(e){super(),this.remove=()=>(this._content&&i.remove(this._content),this._container&&(i.remove(this._container),delete this._container),this._map&&(this._map.off("move",this._update),this._map.off("move",this._onClose),this._map.off("click",this._onClose),this._map.off("remove",this.remove),this._map.off("mousemove",this._onMouseMove),this._map.off("mouseup",this._onMouseUp),this._map.off("drag",this._onDrag),delete this._map),this.fire(new t.k("close")),this),this._onMouseUp=t=>{this._update(t.point);},this._onMouseMove=t=>{this._update(t.point);},this._onDrag=t=>{this._update(t.point);},this._update=t=>{if(!this._map||!this._lngLat&&!this._trackPointer||!this._content)return;if(!this._container){if(this._container=i.create("div","maplibregl-popup",this._map.getContainer()),this._tip=i.create("div","maplibregl-popup-tip",this._container),this._container.appendChild(this._content),this.options.className)for(const t of this.options.className.split(" "))this._container.classList.add(t);this._trackPointer&&this._container.classList.add("maplibregl-popup-track-pointer");}if(this.options.maxWidth&&this._container.style.maxWidth!==this.options.maxWidth&&(this._container.style.maxWidth=this.options.maxWidth),this._map.transform.renderWorldCopies&&!this._trackPointer&&(this._lngLat=wa(this._lngLat,this._pos,this._map.transform)),this._trackPointer&&!t)return;const e=this._pos=this._trackPointer&&t?t:this._map.project(this._lngLat);let s=this.options.anchor;const a=Ra(this.options.offset);if(!s){const t=this._container.offsetWidth,i=this._container.offsetHeight;let o;o=e.y+a.bottom.ythis._map.transform.height-i?["bottom"]:[],e.xthis._map.transform.width-t/2&&o.push("right"),s=0===o.length?"bottom":o.join("-");}const o=e.add(a[s]).round();i.setTransform(this._container,`${Ta[s]} translate(${o.x}px,${o.y}px)`),Ia(this._container,s,"popup");},this._onClose=()=>{this.remove();},this.options=t.e(Object.create(La),e);}addTo(e){return this._map&&this.remove(),this._map=e,this.options.closeOnClick&&this._map.on("click",this._onClose),this.options.closeOnMove&&this._map.on("move",this._onClose),this._map.on("remove",this.remove),this._update(),this._focusFirstElement(),this._trackPointer?(this._map.on("mousemove",this._onMouseMove),this._map.on("mouseup",this._onMouseUp),this._container&&this._container.classList.add("maplibregl-popup-track-pointer"),this._map._canvasContainer.classList.add("maplibregl-track-pointer")):this._map.on("move",this._update),this.fire(new t.k("open")),this}isOpen(){return !!this._map}getLngLat(){return this._lngLat}setLngLat(e){return this._lngLat=t.L.convert(e),this._pos=null,this._trackPointer=!1,this._update(),this._map&&(this._map.on("move",this._update),this._map.off("mousemove",this._onMouseMove),this._container&&this._container.classList.remove("maplibregl-popup-track-pointer"),this._map._canvasContainer.classList.remove("maplibregl-track-pointer")),this}trackPointer(){return this._trackPointer=!0,this._pos=null,this._update(),this._map&&(this._map.off("move",this._update),this._map.on("mousemove",this._onMouseMove),this._map.on("drag",this._onDrag),this._container&&this._container.classList.add("maplibregl-popup-track-pointer"),this._map._canvasContainer.classList.add("maplibregl-track-pointer")),this}getElement(){return this._container}setText(t){return this.setDOMContent(document.createTextNode(t))}setHTML(t){const e=document.createDocumentFragment(),i=document.createElement("body");let s;for(i.innerHTML=t;s=i.firstChild,s;)e.appendChild(s);return this.setDOMContent(e)}getMaxWidth(){var t;return null===(t=this._container)||void 0===t?void 0:t.style.maxWidth}setMaxWidth(t){return this.options.maxWidth=t,this._update(),this}setDOMContent(t){if(this._content)for(;this._content.hasChildNodes();)this._content.firstChild&&this._content.removeChild(this._content.firstChild);else this._content=i.create("div","maplibregl-popup-content",this._container);return this._content.appendChild(t),this._createCloseButton(),this._update(),this._focusFirstElement(),this}addClassName(t){this._container&&this._container.classList.add(t);}removeClassName(t){this._container&&this._container.classList.remove(t);}setOffset(t){return this.options.offset=t,this._update(),this}toggleClassName(t){if(this._container)return this._container.classList.toggle(t)}_createCloseButton(){this.options.closeButton&&(this._closeButton=i.create("button","maplibregl-popup-close-button",this._content),this._closeButton.type="button",this._closeButton.setAttribute("aria-label","Close popup"),this._closeButton.innerHTML="×",this._closeButton.addEventListener("click",this._onClose));}_focusFirstElement(){if(!this.options.focusAfterOpen||!this._container)return;const t=this._container.querySelector(Aa);t&&t.focus();}},Ba.Marker=Ea,Ba.Style=se,Ba.LngLat=t.L,Ba.LngLatBounds=L,Ba.Point=t.P,Ba.MercatorCoordinate=t.U,Ba.Evented=t.E,Ba.AJAXError=t.bi,Ba.config=t.c,Ba.CanvasSource=Z,Ba.GeoJSONSource=B,Ba.ImageSource=N,Ba.RasterDEMTileSource=F,Ba.RasterTileSource=k,Ba.VectorTileSource=R,Ba.VideoSource=U,Ba.setRTLTextPlugin=t.bj,Ba.getRTLTextPluginStatus=t.bk,Ba.prewarm=function(){st().acquire(J);},Ba.clearPrewarmedResources=function(){const t=it;t&&(t.isPreloaded()&&1===t.numActive()?(t.release(J),it=null):console.warn("Could not clear WebWorkers since there are active Map instances that still reference it. The pre-warmed WebWorker pool can only be cleared when all map instances have been removed with map.remove()"));},ka.extend(Ba,{isSafari:t.ac,getPerformanceMetrics:t.bg.getPerformanceMetrics}),Ba})); + +// +// Our custom intro provides a specialized "define()" function, called by the +// AMD modules below, that sets up the worker blob URL and then executes the +// main module, storing its exported value as 'maplibregl' + + +var maplibregl$1 = maplibregl; + +return maplibregl$1; + +})); +//# sourceMappingURL=maplibre-gl.js.map diff --git a/web/libraries/maplibre-gl/dist/maplibre-gl.js.map b/web/libraries/maplibre-gl/dist/maplibre-gl.js.map new file mode 100644 index 00000000..24ea7c9e --- /dev/null +++ b/web/libraries/maplibre-gl/dist/maplibre-gl.js.map @@ -0,0 +1 @@ +{"version":3,"file":"maplibre-gl.js","sources":["../node_modules/tslib/tslib.es6.js","../node_modules/@mapbox/point-geometry/index.js","../node_modules/@mapbox/unitbezier/index.js","../src/util/offscreen_canvas_supported.ts","../src/util/offscreen_canvas_distorted.ts","../src/util/util.ts","../src/util/browser.ts","../src/util/config.ts","../src/util/ajax.ts","../src/util/evented.ts","../node_modules/@maplibre/maplibre-gl-style-spec/dist/index.mjs","../src/style/validate_style.ts","../src/util/transferable_grid_index.ts","../src/util/web_worker_transfer.ts","../src/style/zoom_history.ts","../src/util/is_char_in_unicode_block.ts","../src/util/script_detection.ts","../src/source/rtl_text_plugin.ts","../src/style/evaluation_parameters.ts","../src/style/properties.ts","../src/style/style_layer.ts","../src/util/struct_array.ts","../src/data/array_types.g.ts","../src/data/bucket/circle_attributes.ts","../src/data/segment.ts","../src/shaders/encode_attribute.ts","../src/data/bucket/pattern_attributes.ts","../node_modules/murmurhash-js/murmurhash3_gc.js","../node_modules/murmurhash-js/murmurhash2_gc.js","../node_modules/murmurhash-js/index.js","../src/data/feature_position_map.ts","../src/render/uniform_binding.ts","../src/data/program_configuration.ts","../src/data/extent.ts","../src/data/load_geometry.ts","../src/data/evaluation_feature.ts","../src/data/bucket/circle_bucket.ts","../src/util/intersection_tests.ts","../src/style/query_utils.ts","../src/style/style_layer/circle_style_layer_properties.g.ts","../node_modules/gl-matrix/esm/common.js","../node_modules/gl-matrix/esm/mat4.js","../node_modules/gl-matrix/esm/vec4.js","../src/style/style_layer/circle_style_layer.ts","../src/data/bucket/heatmap_bucket.ts","../src/style/style_layer/heatmap_style_layer_properties.g.ts","../src/util/image.ts","../src/util/color_ramp.ts","../src/style/style_layer/heatmap_style_layer.ts","../src/style/style_layer/hillshade_style_layer_properties.g.ts","../src/style/style_layer/hillshade_style_layer.ts","../src/data/bucket/fill_attributes.ts","../node_modules/earcut/src/earcut.js","../node_modules/quickselect/index.js","../src/util/classify_rings.ts","../src/data/bucket/pattern_bucket_features.ts","../src/data/bucket/fill_bucket.ts","../src/style/style_layer/fill_style_layer_properties.g.ts","../src/style/style_layer/fill_style_layer.ts","../src/data/bucket/fill_extrusion_attributes.ts","../node_modules/@mapbox/vector-tile/lib/vectortilefeature.js","../node_modules/@mapbox/vector-tile/lib/vectortilelayer.js","../node_modules/@mapbox/vector-tile/lib/vectortile.js","../node_modules/@mapbox/vector-tile/index.js","../src/data/bucket/fill_extrusion_bucket.ts","../src/style/style_layer/fill_extrusion_style_layer_properties.g.ts","../src/style/style_layer/fill_extrusion_style_layer.ts","../src/data/bucket/line_attributes.ts","../src/data/bucket/line_attributes_ext.ts","../src/data/bucket/line_bucket.ts","../src/style/style_layer/line_style_layer_properties.g.ts","../src/style/style_layer/line_style_layer.ts","../src/data/bucket/symbol_attributes.ts","../src/symbol/transform_text.ts","../src/util/verticalize_punctuation.ts","../src/symbol/one_em.ts","../node_modules/pbf/index.js","../node_modules/ieee754/index.js","../src/style/parse_glyph_pbf.ts","../node_modules/potpack/index.js","../src/render/image_atlas.ts","../src/symbol/shaping.ts","../src/symbol/symbol_size.ts","../src/style/style_layer/overlap_mode.ts","../src/data/bucket/symbol_bucket.ts","../src/symbol/merge_lines.ts","../src/style/style_layer/symbol_style_layer_properties.g.ts","../src/style/format_section_override.ts","../src/style/style_layer/symbol_style_layer.ts","../src/util/resolve_tokens.ts","../src/style/style_layer/background_style_layer_properties.g.ts","../src/style/style_layer/background_style_layer.ts","../src/style/style_layer/raster_style_layer_properties.g.ts","../src/style/style_layer/raster_style_layer.ts","../src/style/style_layer/custom_style_layer.ts","../src/util/throttled_invoker.ts","../src/geo/lng_lat.ts","../src/geo/mercator_coordinate.ts","../node_modules/@mapbox/whoots-js/index.mjs","../src/source/tile_id.ts","../src/data/dem_data.ts","../src/util/dictionary_coder.ts","../src/util/vectortile_to_geojson.ts","../src/data/feature_index.ts","../src/symbol/clip_line.ts","../src/symbol/anchor.ts","../src/symbol/check_max_angle.ts","../src/symbol/get_anchors.ts","../src/symbol/quads.ts","../src/symbol/collision_feature.ts","../node_modules/tinyqueue/index.js","../src/util/find_pole_of_inaccessibility.ts","../src/style/style_layer/variable_text_anchor.ts","../src/symbol/symbol_layout.ts","../node_modules/kdbush/index.js","../src/util/performance.ts","../src/util/actor.ts","../src/style/create_style_layer.ts","../src/style/style_layer_index.ts","../src/render/glyph_atlas.ts","../src/source/worker_tile.ts","../src/source/vector_tile_worker_source.ts","../src/source/raster_dem_tile_worker_source.ts","../node_modules/@mapbox/geojson-rewind/index.js","../src/source/geojson_wrapper.ts","../node_modules/vt-pbf/lib/geojson_wrapper.js","../node_modules/vt-pbf/index.js","../node_modules/supercluster/index.js","../node_modules/geojson-vt/src/simplify.js","../node_modules/geojson-vt/src/feature.js","../node_modules/geojson-vt/src/convert.js","../node_modules/geojson-vt/src/clip.js","../node_modules/geojson-vt/src/wrap.js","../node_modules/geojson-vt/src/transform.js","../node_modules/geojson-vt/src/tile.js","../node_modules/geojson-vt/src/index.js","../src/source/geojson_source_diff.ts","../src/source/geojson_worker_source.ts","../src/source/worker.ts","../src/util/dom.ts","../src/util/webp_supported.ts","../src/util/image_request.ts","../src/util/request_manager.ts","../node_modules/gl-matrix/esm/vec3.js","../node_modules/gl-matrix/esm/vec2.js","../src/util/style.ts","../src/style/load_sprite.ts","../src/render/texture.ts","../src/style/style_image.ts","../src/render/image_manager.ts","../node_modules/@mapbox/tiny-sdf/index.js","../src/render/glyph_manager.ts","../src/style/load_glyph_range.ts","../src/style/light.ts","../src/render/line_atlas.ts","../src/util/dispatcher.ts","../src/source/load_tilejson.ts","../src/geo/lng_lat_bounds.ts","../src/source/tile_bounds.ts","../src/source/vector_tile_source.ts","../src/source/raster_tile_source.ts","../src/source/raster_dem_tile_source.ts","../src/source/geojson_source.ts","../src/data/raster_bounds_attributes.ts","../src/source/image_source.ts","../src/source/video_source.ts","../src/source/canvas_source.ts","../src/source/source.ts","../src/source/query_features.ts","../src/source/tile.ts","../src/data/bucket.ts","../src/source/tile_cache.ts","../src/source/source_state.ts","../src/source/source_cache.ts","../src/util/worker_pool.ts","../src/util/web_worker.ts","../src/util/global_worker_pool.ts","../src/symbol/path_interpolator.ts","../src/symbol/grid_index.ts","../src/symbol/projection.ts","../src/symbol/collision_index.ts","../src/source/pixels_to_tile_units.ts","../src/symbol/placement.ts","../src/style/pauseable_placement.ts","../src/symbol/cross_tile_symbol_index.ts","../src/style/style.ts","../src/data/pos_attributes.ts","../src/shaders/terrain.vertex.glsl.g.ts","../src/shaders/shaders.ts","../src/shaders/_prelude.fragment.glsl.g.ts","../src/shaders/_prelude.vertex.glsl.g.ts","../src/shaders/background.fragment.glsl.g.ts","../src/shaders/background.vertex.glsl.g.ts","../src/shaders/background_pattern.fragment.glsl.g.ts","../src/shaders/background_pattern.vertex.glsl.g.ts","../src/shaders/circle.fragment.glsl.g.ts","../src/shaders/circle.vertex.glsl.g.ts","../src/shaders/clipping_mask.fragment.glsl.g.ts","../src/shaders/clipping_mask.vertex.glsl.g.ts","../src/shaders/heatmap.fragment.glsl.g.ts","../src/shaders/heatmap.vertex.glsl.g.ts","../src/shaders/heatmap_texture.fragment.glsl.g.ts","../src/shaders/heatmap_texture.vertex.glsl.g.ts","../src/shaders/collision_box.fragment.glsl.g.ts","../src/shaders/collision_box.vertex.glsl.g.ts","../src/shaders/collision_circle.fragment.glsl.g.ts","../src/shaders/collision_circle.vertex.glsl.g.ts","../src/shaders/debug.fragment.glsl.g.ts","../src/shaders/debug.vertex.glsl.g.ts","../src/shaders/fill.fragment.glsl.g.ts","../src/shaders/fill.vertex.glsl.g.ts","../src/shaders/fill_outline.fragment.glsl.g.ts","../src/shaders/fill_outline.vertex.glsl.g.ts","../src/shaders/fill_outline_pattern.fragment.glsl.g.ts","../src/shaders/fill_outline_pattern.vertex.glsl.g.ts","../src/shaders/fill_pattern.fragment.glsl.g.ts","../src/shaders/fill_pattern.vertex.glsl.g.ts","../src/shaders/fill_extrusion.fragment.glsl.g.ts","../src/shaders/fill_extrusion.vertex.glsl.g.ts","../src/shaders/fill_extrusion_pattern.fragment.glsl.g.ts","../src/shaders/fill_extrusion_pattern.vertex.glsl.g.ts","../src/shaders/hillshade_prepare.fragment.glsl.g.ts","../src/shaders/hillshade_prepare.vertex.glsl.g.ts","../src/shaders/hillshade.fragment.glsl.g.ts","../src/shaders/hillshade.vertex.glsl.g.ts","../src/shaders/line.fragment.glsl.g.ts","../src/shaders/line.vertex.glsl.g.ts","../src/shaders/line_gradient.fragment.glsl.g.ts","../src/shaders/line_gradient.vertex.glsl.g.ts","../src/shaders/line_pattern.fragment.glsl.g.ts","../src/shaders/line_pattern.vertex.glsl.g.ts","../src/shaders/line_sdf.fragment.glsl.g.ts","../src/shaders/line_sdf.vertex.glsl.g.ts","../src/shaders/raster.fragment.glsl.g.ts","../src/shaders/raster.vertex.glsl.g.ts","../src/shaders/symbol_icon.fragment.glsl.g.ts","../src/shaders/symbol_icon.vertex.glsl.g.ts","../src/shaders/symbol_sdf.fragment.glsl.g.ts","../src/shaders/symbol_sdf.vertex.glsl.g.ts","../src/shaders/symbol_text_and_icon.fragment.glsl.g.ts","../src/shaders/symbol_text_and_icon.vertex.glsl.g.ts","../src/shaders/terrain.fragment.glsl.g.ts","../src/shaders/terrain_depth.fragment.glsl.g.ts","../src/shaders/terrain_coords.fragment.glsl.g.ts","../src/render/vertex_array_object.ts","../src/render/program.ts","../src/render/program/terrain_program.ts","../src/render/program/pattern.ts","../src/render/program/fill_extrusion_program.ts","../node_modules/gl-matrix/esm/mat3.js","../src/render/program/fill_program.ts","../src/render/program/circle_program.ts","../src/render/program/collision_program.ts","../src/render/program/debug_program.ts","../src/render/program/clipping_mask_program.ts","../src/render/program/heatmap_program.ts","../src/render/program/hillshade_program.ts","../src/render/program/line_program.ts","../src/render/program/raster_program.ts","../src/render/program/symbol_program.ts","../src/render/program/background_program.ts","../src/render/program/program_uniforms.ts","../src/gl/index_buffer.ts","../src/gl/vertex_buffer.ts","../src/gl/webgl2.ts","../src/gl/value.ts","../src/gl/framebuffer.ts","../src/gl/color_mode.ts","../src/gl/context.ts","../src/gl/depth_mode.ts","../src/gl/stencil_mode.ts","../src/gl/cull_face_mode.ts","../src/render/draw_collision_debug.ts","../src/render/draw_symbol.ts","../src/render/update_pattern_positions_in_program.ts","../src/render/draw_fill.ts","../src/render/draw_fill_extrusion.ts","../src/render/draw_hillshade.ts","../src/render/draw_raster.ts","../src/render/draw_debug.ts","../src/render/draw_terrain.ts","../src/render/painter.ts","../src/render/draw_circle.ts","../src/render/draw_heatmap.ts","../src/render/draw_line.ts","../src/render/draw_background.ts","../src/render/draw_custom.ts","../src/util/primitives.ts","../src/geo/edge_insets.ts","../src/geo/transform.ts","../node_modules/gl-matrix/esm/mat2.js","../src/util/throttle.ts","../src/ui/hash.ts","../src/ui/handler_inertia.ts","../src/ui/events.ts","../src/ui/handler/map_event.ts","../src/ui/handler/transform-provider.ts","../src/ui/handler/box_zoom.ts","../src/ui/handler/handler_util.ts","../src/ui/handler/tap_recognizer.ts","../src/ui/handler/tap_zoom.ts","../src/ui/handler/drag_handler.ts","../src/ui/handler/drag_move_state_manager.ts","../src/ui/handler/mouse.ts","../src/ui/handler/touch_pan.ts","../src/ui/handler/two_fingers_touch.ts","../src/ui/handler/keyboard.ts","../src/ui/handler/scroll_zoom.ts","../src/ui/handler/shim/dblclick_zoom.ts","../src/ui/handler/click_zoom.ts","../src/ui/handler/tap_drag_zoom.ts","../src/ui/handler/shim/drag_pan.ts","../src/ui/handler/shim/drag_rotate.ts","../src/ui/handler/shim/two_fingers_touch.ts","../src/ui/handler_manager.ts","../src/ui/camera.ts","../src/ui/control/attribution_control.ts","../src/ui/control/logo_control.ts","../src/util/task_queue.ts","../src/ui/default_locale.ts","../src/data/pos3d_attributes.ts","../src/source/terrain_source_cache.ts","../src/render/terrain.ts","../src/gl/render_pool.ts","../src/render/render_to_texture.ts","../src/ui/map.ts","../src/ui/handler/one_finger_touch_drag.ts","../src/ui/control/navigation_control.ts","../src/util/geolocation_support.ts","../src/util/smart_wrap.ts","../src/ui/anchor.ts","../src/ui/marker.ts","../src/ui/control/geolocate_control.ts","../src/ui/control/scale_control.ts","../src/ui/popup.ts","../src/util/debug.ts","../src/index.ts","../src/ui/control/fullscreen_control.ts","../src/ui/control/terrain_control.ts","../build/rollup/maplibregl.js"],"sourcesContent":["/******************************************************************************\r\nCopyright (c) Microsoft Corporation.\r\n\r\nPermission to use, copy, modify, and/or distribute this software for any\r\npurpose with or without fee is hereby granted.\r\n\r\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\r\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\r\nAND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\r\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\r\nLOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\r\nOTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\r\nPERFORMANCE OF THIS SOFTWARE.\r\n***************************************************************************** */\r\n/* global Reflect, Promise, SuppressedError, Symbol */\r\n\r\nvar extendStatics = function(d, b) {\r\n extendStatics = Object.setPrototypeOf ||\r\n ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\r\n function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };\r\n return extendStatics(d, b);\r\n};\r\n\r\nexport function __extends(d, b) {\r\n if (typeof b !== \"function\" && b !== null)\r\n throw new TypeError(\"Class extends value \" + String(b) + \" is not a constructor or null\");\r\n extendStatics(d, b);\r\n function __() { this.constructor = d; }\r\n d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\r\n}\r\n\r\nexport var __assign = function() {\r\n __assign = Object.assign || function __assign(t) {\r\n for (var s, i = 1, n = arguments.length; i < n; i++) {\r\n s = arguments[i];\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];\r\n }\r\n return t;\r\n }\r\n return __assign.apply(this, arguments);\r\n}\r\n\r\nexport function __rest(s, e) {\r\n var t = {};\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\r\n t[p] = s[p];\r\n if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\r\n for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\r\n if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\r\n t[p[i]] = s[p[i]];\r\n }\r\n return t;\r\n}\r\n\r\nexport function __decorate(decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n}\r\n\r\nexport function __param(paramIndex, decorator) {\r\n return function (target, key) { decorator(target, key, paramIndex); }\r\n}\r\n\r\nexport function __esDecorate(ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {\r\n function accept(f) { if (f !== void 0 && typeof f !== \"function\") throw new TypeError(\"Function expected\"); return f; }\r\n var kind = contextIn.kind, key = kind === \"getter\" ? \"get\" : kind === \"setter\" ? \"set\" : \"value\";\r\n var target = !descriptorIn && ctor ? contextIn[\"static\"] ? ctor : ctor.prototype : null;\r\n var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});\r\n var _, done = false;\r\n for (var i = decorators.length - 1; i >= 0; i--) {\r\n var context = {};\r\n for (var p in contextIn) context[p] = p === \"access\" ? {} : contextIn[p];\r\n for (var p in contextIn.access) context.access[p] = contextIn.access[p];\r\n context.addInitializer = function (f) { if (done) throw new TypeError(\"Cannot add initializers after decoration has completed\"); extraInitializers.push(accept(f || null)); };\r\n var result = (0, decorators[i])(kind === \"accessor\" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);\r\n if (kind === \"accessor\") {\r\n if (result === void 0) continue;\r\n if (result === null || typeof result !== \"object\") throw new TypeError(\"Object expected\");\r\n if (_ = accept(result.get)) descriptor.get = _;\r\n if (_ = accept(result.set)) descriptor.set = _;\r\n if (_ = accept(result.init)) initializers.unshift(_);\r\n }\r\n else if (_ = accept(result)) {\r\n if (kind === \"field\") initializers.unshift(_);\r\n else descriptor[key] = _;\r\n }\r\n }\r\n if (target) Object.defineProperty(target, contextIn.name, descriptor);\r\n done = true;\r\n};\r\n\r\nexport function __runInitializers(thisArg, initializers, value) {\r\n var useValue = arguments.length > 2;\r\n for (var i = 0; i < initializers.length; i++) {\r\n value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);\r\n }\r\n return useValue ? value : void 0;\r\n};\r\n\r\nexport function __propKey(x) {\r\n return typeof x === \"symbol\" ? x : \"\".concat(x);\r\n};\r\n\r\nexport function __setFunctionName(f, name, prefix) {\r\n if (typeof name === \"symbol\") name = name.description ? \"[\".concat(name.description, \"]\") : \"\";\r\n return Object.defineProperty(f, \"name\", { configurable: true, value: prefix ? \"\".concat(prefix, \" \", name) : name });\r\n};\r\n\r\nexport function __metadata(metadataKey, metadataValue) {\r\n if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(metadataKey, metadataValue);\r\n}\r\n\r\nexport function __awaiter(thisArg, _arguments, P, generator) {\r\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\r\n return new (P || (P = Promise))(function (resolve, reject) {\r\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\r\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\r\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\r\n step((generator = generator.apply(thisArg, _arguments || [])).next());\r\n });\r\n}\r\n\r\nexport function __generator(thisArg, body) {\r\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\r\n return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\r\n function verb(n) { return function (v) { return step([n, v]); }; }\r\n function step(op) {\r\n if (f) throw new TypeError(\"Generator is already executing.\");\r\n while (g && (g = 0, op[0] && (_ = 0)), _) try {\r\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\r\n if (y = 0, t) op = [op[0] & 2, t.value];\r\n switch (op[0]) {\r\n case 0: case 1: t = op; break;\r\n case 4: _.label++; return { value: op[1], done: false };\r\n case 5: _.label++; y = op[1]; op = [0]; continue;\r\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\r\n default:\r\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\r\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\r\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\r\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\r\n if (t[2]) _.ops.pop();\r\n _.trys.pop(); continue;\r\n }\r\n op = body.call(thisArg, _);\r\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\r\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\r\n }\r\n}\r\n\r\nexport var __createBinding = Object.create ? (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n var desc = Object.getOwnPropertyDescriptor(m, k);\r\n if (!desc || (\"get\" in desc ? !m.__esModule : desc.writable || desc.configurable)) {\r\n desc = { enumerable: true, get: function() { return m[k]; } };\r\n }\r\n Object.defineProperty(o, k2, desc);\r\n}) : (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n o[k2] = m[k];\r\n});\r\n\r\nexport function __exportStar(m, o) {\r\n for (var p in m) if (p !== \"default\" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p);\r\n}\r\n\r\nexport function __values(o) {\r\n var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\r\n if (m) return m.call(o);\r\n if (o && typeof o.length === \"number\") return {\r\n next: function () {\r\n if (o && i >= o.length) o = void 0;\r\n return { value: o && o[i++], done: !o };\r\n }\r\n };\r\n throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\r\n}\r\n\r\nexport function __read(o, n) {\r\n var m = typeof Symbol === \"function\" && o[Symbol.iterator];\r\n if (!m) return o;\r\n var i = m.call(o), r, ar = [], e;\r\n try {\r\n while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\r\n }\r\n catch (error) { e = { error: error }; }\r\n finally {\r\n try {\r\n if (r && !r.done && (m = i[\"return\"])) m.call(i);\r\n }\r\n finally { if (e) throw e.error; }\r\n }\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spread() {\r\n for (var ar = [], i = 0; i < arguments.length; i++)\r\n ar = ar.concat(__read(arguments[i]));\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spreadArrays() {\r\n for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\r\n for (var r = Array(s), k = 0, i = 0; i < il; i++)\r\n for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\r\n r[k] = a[j];\r\n return r;\r\n}\r\n\r\nexport function __spreadArray(to, from, pack) {\r\n if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {\r\n if (ar || !(i in from)) {\r\n if (!ar) ar = Array.prototype.slice.call(from, 0, i);\r\n ar[i] = from[i];\r\n }\r\n }\r\n return to.concat(ar || Array.prototype.slice.call(from));\r\n}\r\n\r\nexport function __await(v) {\r\n return this instanceof __await ? (this.v = v, this) : new __await(v);\r\n}\r\n\r\nexport function __asyncGenerator(thisArg, _arguments, generator) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var g = generator.apply(thisArg, _arguments || []), i, q = [];\r\n return i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i;\r\n function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }\r\n function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\r\n function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\r\n function fulfill(value) { resume(\"next\", value); }\r\n function reject(value) { resume(\"throw\", value); }\r\n function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\r\n}\r\n\r\nexport function __asyncDelegator(o) {\r\n var i, p;\r\n return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\r\n function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: false } : f ? f(v) : v; } : f; }\r\n}\r\n\r\nexport function __asyncValues(o) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var m = o[Symbol.asyncIterator], i;\r\n return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\r\n function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\r\n function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\r\n}\r\n\r\nexport function __makeTemplateObject(cooked, raw) {\r\n if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\r\n return cooked;\r\n};\r\n\r\nvar __setModuleDefault = Object.create ? (function(o, v) {\r\n Object.defineProperty(o, \"default\", { enumerable: true, value: v });\r\n}) : function(o, v) {\r\n o[\"default\"] = v;\r\n};\r\n\r\nexport function __importStar(mod) {\r\n if (mod && mod.__esModule) return mod;\r\n var result = {};\r\n if (mod != null) for (var k in mod) if (k !== \"default\" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);\r\n __setModuleDefault(result, mod);\r\n return result;\r\n}\r\n\r\nexport function __importDefault(mod) {\r\n return (mod && mod.__esModule) ? mod : { default: mod };\r\n}\r\n\r\nexport function __classPrivateFieldGet(receiver, state, kind, f) {\r\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a getter\");\r\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot read private member from an object whose class did not declare it\");\r\n return kind === \"m\" ? f : kind === \"a\" ? f.call(receiver) : f ? f.value : state.get(receiver);\r\n}\r\n\r\nexport function __classPrivateFieldSet(receiver, state, value, kind, f) {\r\n if (kind === \"m\") throw new TypeError(\"Private method is not writable\");\r\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a setter\");\r\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot write private member to an object whose class did not declare it\");\r\n return (kind === \"a\" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;\r\n}\r\n\r\nexport function __classPrivateFieldIn(state, receiver) {\r\n if (receiver === null || (typeof receiver !== \"object\" && typeof receiver !== \"function\")) throw new TypeError(\"Cannot use 'in' operator on non-object\");\r\n return typeof state === \"function\" ? receiver === state : state.has(receiver);\r\n}\r\n\r\nexport function __addDisposableResource(env, value, async) {\r\n if (value !== null && value !== void 0) {\r\n if (typeof value !== \"object\" && typeof value !== \"function\") throw new TypeError(\"Object expected.\");\r\n var dispose;\r\n if (async) {\r\n if (!Symbol.asyncDispose) throw new TypeError(\"Symbol.asyncDispose is not defined.\");\r\n dispose = value[Symbol.asyncDispose];\r\n }\r\n if (dispose === void 0) {\r\n if (!Symbol.dispose) throw new TypeError(\"Symbol.dispose is not defined.\");\r\n dispose = value[Symbol.dispose];\r\n }\r\n if (typeof dispose !== \"function\") throw new TypeError(\"Object not disposable.\");\r\n env.stack.push({ value: value, dispose: dispose, async: async });\r\n }\r\n else if (async) {\r\n env.stack.push({ async: true });\r\n }\r\n return value;\r\n}\r\n\r\nvar _SuppressedError = typeof SuppressedError === \"function\" ? SuppressedError : function (error, suppressed, message) {\r\n var e = new Error(message);\r\n return e.name = \"SuppressedError\", e.error = error, e.suppressed = suppressed, e;\r\n};\r\n\r\nexport function __disposeResources(env) {\r\n function fail(e) {\r\n env.error = env.hasError ? new _SuppressedError(e, env.error, \"An error was suppressed during disposal.\") : e;\r\n env.hasError = true;\r\n }\r\n function next() {\r\n while (env.stack.length) {\r\n var rec = env.stack.pop();\r\n try {\r\n var result = rec.dispose && rec.dispose.call(rec.value);\r\n if (rec.async) return Promise.resolve(result).then(next, function(e) { fail(e); return next(); });\r\n }\r\n catch (e) {\r\n fail(e);\r\n }\r\n }\r\n if (env.hasError) throw env.error;\r\n }\r\n return next();\r\n}\r\n\r\nexport default {\r\n __extends: __extends,\r\n __assign: __assign,\r\n __rest: __rest,\r\n __decorate: __decorate,\r\n __param: __param,\r\n __metadata: __metadata,\r\n __awaiter: __awaiter,\r\n __generator: __generator,\r\n __createBinding: __createBinding,\r\n __exportStar: __exportStar,\r\n __values: __values,\r\n __read: __read,\r\n __spread: __spread,\r\n __spreadArrays: __spreadArrays,\r\n __spreadArray: __spreadArray,\r\n __await: __await,\r\n __asyncGenerator: __asyncGenerator,\r\n __asyncDelegator: __asyncDelegator,\r\n __asyncValues: __asyncValues,\r\n __makeTemplateObject: __makeTemplateObject,\r\n __importStar: __importStar,\r\n __importDefault: __importDefault,\r\n __classPrivateFieldGet: __classPrivateFieldGet,\r\n __classPrivateFieldSet: __classPrivateFieldSet,\r\n __classPrivateFieldIn: __classPrivateFieldIn,\r\n __addDisposableResource: __addDisposableResource,\r\n __disposeResources: __disposeResources,\r\n};\r\n","'use strict';\n\nmodule.exports = Point;\n\n/**\n * A standalone point geometry with useful accessor, comparison, and\n * modification methods.\n *\n * @class Point\n * @param {Number} x the x-coordinate. this could be longitude or screen\n * pixels, or any other sort of unit.\n * @param {Number} y the y-coordinate. this could be latitude or screen\n * pixels, or any other sort of unit.\n * @example\n * var point = new Point(-77, 38);\n */\nfunction Point(x, y) {\n this.x = x;\n this.y = y;\n}\n\nPoint.prototype = {\n\n /**\n * Clone this point, returning a new point that can be modified\n * without affecting the old one.\n * @return {Point} the clone\n */\n clone: function() { return new Point(this.x, this.y); },\n\n /**\n * Add this point's x & y coordinates to another point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n add: function(p) { return this.clone()._add(p); },\n\n /**\n * Subtract this point's x & y coordinates to from point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n sub: function(p) { return this.clone()._sub(p); },\n\n /**\n * Multiply this point's x & y coordinates by point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n multByPoint: function(p) { return this.clone()._multByPoint(p); },\n\n /**\n * Divide this point's x & y coordinates by point,\n * yielding a new point.\n * @param {Point} p the other point\n * @return {Point} output point\n */\n divByPoint: function(p) { return this.clone()._divByPoint(p); },\n\n /**\n * Multiply this point's x & y coordinates by a factor,\n * yielding a new point.\n * @param {Point} k factor\n * @return {Point} output point\n */\n mult: function(k) { return this.clone()._mult(k); },\n\n /**\n * Divide this point's x & y coordinates by a factor,\n * yielding a new point.\n * @param {Point} k factor\n * @return {Point} output point\n */\n div: function(k) { return this.clone()._div(k); },\n\n /**\n * Rotate this point around the 0, 0 origin by an angle a,\n * given in radians\n * @param {Number} a angle to rotate around, in radians\n * @return {Point} output point\n */\n rotate: function(a) { return this.clone()._rotate(a); },\n\n /**\n * Rotate this point around p point by an angle a,\n * given in radians\n * @param {Number} a angle to rotate around, in radians\n * @param {Point} p Point to rotate around\n * @return {Point} output point\n */\n rotateAround: function(a,p) { return this.clone()._rotateAround(a,p); },\n\n /**\n * Multiply this point by a 4x1 transformation matrix\n * @param {Array} m transformation matrix\n * @return {Point} output point\n */\n matMult: function(m) { return this.clone()._matMult(m); },\n\n /**\n * Calculate this point but as a unit vector from 0, 0, meaning\n * that the distance from the resulting point to the 0, 0\n * coordinate will be equal to 1 and the angle from the resulting\n * point to the 0, 0 coordinate will be the same as before.\n * @return {Point} unit vector point\n */\n unit: function() { return this.clone()._unit(); },\n\n /**\n * Compute a perpendicular point, where the new y coordinate\n * is the old x coordinate and the new x coordinate is the old y\n * coordinate multiplied by -1\n * @return {Point} perpendicular point\n */\n perp: function() { return this.clone()._perp(); },\n\n /**\n * Return a version of this point with the x & y coordinates\n * rounded to integers.\n * @return {Point} rounded point\n */\n round: function() { return this.clone()._round(); },\n\n /**\n * Return the magitude of this point: this is the Euclidean\n * distance from the 0, 0 coordinate to this point's x and y\n * coordinates.\n * @return {Number} magnitude\n */\n mag: function() {\n return Math.sqrt(this.x * this.x + this.y * this.y);\n },\n\n /**\n * Judge whether this point is equal to another point, returning\n * true or false.\n * @param {Point} other the other point\n * @return {boolean} whether the points are equal\n */\n equals: function(other) {\n return this.x === other.x &&\n this.y === other.y;\n },\n\n /**\n * Calculate the distance from this point to another point\n * @param {Point} p the other point\n * @return {Number} distance\n */\n dist: function(p) {\n return Math.sqrt(this.distSqr(p));\n },\n\n /**\n * Calculate the distance from this point to another point,\n * without the square root step. Useful if you're comparing\n * relative distances.\n * @param {Point} p the other point\n * @return {Number} distance\n */\n distSqr: function(p) {\n var dx = p.x - this.x,\n dy = p.y - this.y;\n return dx * dx + dy * dy;\n },\n\n /**\n * Get the angle from the 0, 0 coordinate to this point, in radians\n * coordinates.\n * @return {Number} angle\n */\n angle: function() {\n return Math.atan2(this.y, this.x);\n },\n\n /**\n * Get the angle from this point to another point, in radians\n * @param {Point} b the other point\n * @return {Number} angle\n */\n angleTo: function(b) {\n return Math.atan2(this.y - b.y, this.x - b.x);\n },\n\n /**\n * Get the angle between this point and another point, in radians\n * @param {Point} b the other point\n * @return {Number} angle\n */\n angleWith: function(b) {\n return this.angleWithSep(b.x, b.y);\n },\n\n /*\n * Find the angle of the two vectors, solving the formula for\n * the cross product a x b = |a||b|sin(θ) for θ.\n * @param {Number} x the x-coordinate\n * @param {Number} y the y-coordinate\n * @return {Number} the angle in radians\n */\n angleWithSep: function(x, y) {\n return Math.atan2(\n this.x * y - this.y * x,\n this.x * x + this.y * y);\n },\n\n _matMult: function(m) {\n var x = m[0] * this.x + m[1] * this.y,\n y = m[2] * this.x + m[3] * this.y;\n this.x = x;\n this.y = y;\n return this;\n },\n\n _add: function(p) {\n this.x += p.x;\n this.y += p.y;\n return this;\n },\n\n _sub: function(p) {\n this.x -= p.x;\n this.y -= p.y;\n return this;\n },\n\n _mult: function(k) {\n this.x *= k;\n this.y *= k;\n return this;\n },\n\n _div: function(k) {\n this.x /= k;\n this.y /= k;\n return this;\n },\n\n _multByPoint: function(p) {\n this.x *= p.x;\n this.y *= p.y;\n return this;\n },\n\n _divByPoint: function(p) {\n this.x /= p.x;\n this.y /= p.y;\n return this;\n },\n\n _unit: function() {\n this._div(this.mag());\n return this;\n },\n\n _perp: function() {\n var y = this.y;\n this.y = this.x;\n this.x = -y;\n return this;\n },\n\n _rotate: function(angle) {\n var cos = Math.cos(angle),\n sin = Math.sin(angle),\n x = cos * this.x - sin * this.y,\n y = sin * this.x + cos * this.y;\n this.x = x;\n this.y = y;\n return this;\n },\n\n _rotateAround: function(angle, p) {\n var cos = Math.cos(angle),\n sin = Math.sin(angle),\n x = p.x + cos * (this.x - p.x) - sin * (this.y - p.y),\n y = p.y + sin * (this.x - p.x) + cos * (this.y - p.y);\n this.x = x;\n this.y = y;\n return this;\n },\n\n _round: function() {\n this.x = Math.round(this.x);\n this.y = Math.round(this.y);\n return this;\n }\n};\n\n/**\n * Construct a point from an array if necessary, otherwise if the input\n * is already a Point, or an unknown type, return it unchanged\n * @param {Array|Point|*} a any kind of input value\n * @return {Point} constructed point, or passed-through value.\n * @example\n * // this\n * var point = Point.convert([0, 1]);\n * // is equivalent to\n * var point = new Point(0, 1);\n */\nPoint.convert = function (a) {\n if (a instanceof Point) {\n return a;\n }\n if (Array.isArray(a)) {\n return new Point(a[0], a[1]);\n }\n return a;\n};\n","'use strict';\n\nmodule.exports = UnitBezier;\n\nfunction UnitBezier(p1x, p1y, p2x, p2y) {\n // Calculate the polynomial coefficients, implicit first and last control points are (0,0) and (1,1).\n this.cx = 3.0 * p1x;\n this.bx = 3.0 * (p2x - p1x) - this.cx;\n this.ax = 1.0 - this.cx - this.bx;\n\n this.cy = 3.0 * p1y;\n this.by = 3.0 * (p2y - p1y) - this.cy;\n this.ay = 1.0 - this.cy - this.by;\n\n this.p1x = p1x;\n this.p1y = p1y;\n this.p2x = p2x;\n this.p2y = p2y;\n}\n\nUnitBezier.prototype = {\n sampleCurveX: function (t) {\n // `ax t^3 + bx t^2 + cx t' expanded using Horner's rule.\n return ((this.ax * t + this.bx) * t + this.cx) * t;\n },\n\n sampleCurveY: function (t) {\n return ((this.ay * t + this.by) * t + this.cy) * t;\n },\n\n sampleCurveDerivativeX: function (t) {\n return (3.0 * this.ax * t + 2.0 * this.bx) * t + this.cx;\n },\n\n solveCurveX: function (x, epsilon) {\n if (epsilon === undefined) epsilon = 1e-6;\n\n if (x < 0.0) return 0.0;\n if (x > 1.0) return 1.0;\n\n var t = x;\n\n // First try a few iterations of Newton's method - normally very fast.\n for (var i = 0; i < 8; i++) {\n var x2 = this.sampleCurveX(t) - x;\n if (Math.abs(x2) < epsilon) return t;\n\n var d2 = this.sampleCurveDerivativeX(t);\n if (Math.abs(d2) < 1e-6) break;\n\n t = t - x2 / d2;\n }\n\n // Fall back to the bisection method for reliability.\n var t0 = 0.0;\n var t1 = 1.0;\n t = x;\n\n for (i = 0; i < 20; i++) {\n x2 = this.sampleCurveX(t);\n if (Math.abs(x2 - x) < epsilon) break;\n\n if (x > x2) {\n t0 = t;\n } else {\n t1 = t;\n }\n\n t = (t1 - t0) * 0.5 + t0;\n }\n\n return t;\n },\n\n solve: function (x, epsilon) {\n return this.sampleCurveY(this.solveCurveX(x, epsilon));\n }\n};\n","let supportsOffscreenCanvas: boolean;\n\nexport function offscreenCanvasSupported(): boolean {\n if (supportsOffscreenCanvas == null) {\n supportsOffscreenCanvas = typeof OffscreenCanvas !== 'undefined' &&\n new OffscreenCanvas(1, 1).getContext('2d') &&\n typeof createImageBitmap === 'function';\n }\n\n return supportsOffscreenCanvas;\n}\n","import {offscreenCanvasSupported} from './offscreen_canvas_supported';\n\nlet offscreenCanvasDistorted: boolean;\n\n/**\n * Some browsers don't return the exact pixels from a canvas to prevent user fingerprinting (see #3185).\n * This function writes pixels to an OffscreenCanvas and reads them back using getImageData, returning false\n * if they don't match.\n *\n * @returns true if the browser supports OffscreenCanvas but it distorts getImageData results, false otherwise.\n */\nexport function isOffscreenCanvasDistorted(): boolean {\n if (offscreenCanvasDistorted == null) {\n offscreenCanvasDistorted = false;\n if (offscreenCanvasSupported()) {\n const size = 5;\n const canvas = new OffscreenCanvas(size, size);\n const context = canvas.getContext('2d', {willReadFrequently: true});\n if (context) {\n // fill each pixel with an RGB value that should make the byte at index i equal to i (except alpha channel):\n // [0, 1, 2, 255, 4, 5, 6, 255, 8, 9, 10, 255, ...]\n for (let i = 0; i < size * size; i++) {\n const base = i * 4;\n context.fillStyle = `rgb(${base},${base + 1},${base + 2})`;\n context.fillRect(i % size, Math.floor(i / size), 1, 1);\n }\n const data = context.getImageData(0, 0, size, size).data;\n for (let i = 0; i < size * size * 4; i++) {\n if (i % 4 !== 3 && data[i] !== i) {\n offscreenCanvasDistorted = true;\n break;\n }\n }\n }\n }\n }\n\n return offscreenCanvasDistorted || false;\n}\n","import Point from '@mapbox/point-geometry';\nimport UnitBezier from '@mapbox/unitbezier';\nimport type {Callback} from '../types/callback';\nimport {isOffscreenCanvasDistorted} from './offscreen_canvas_distorted';\nimport type {Size} from './image';\n\n/**\n * Given a value `t` that varies between 0 and 1, return\n * an interpolation function that eases between 0 and 1 in a pleasing\n * cubic in-out fashion.\n */\nexport function easeCubicInOut(t: number): number {\n if (t <= 0) return 0;\n if (t >= 1) return 1;\n const t2 = t * t,\n t3 = t2 * t;\n return 4 * (t < 0.5 ? t3 : 3 * (t - t2) + t3 - 0.75);\n}\n\n/**\n * Given given (x, y), (x1, y1) control points for a bezier curve,\n * return a function that interpolates along that curve.\n *\n * @param p1x - control point 1 x coordinate\n * @param p1y - control point 1 y coordinate\n * @param p2x - control point 2 x coordinate\n * @param p2y - control point 2 y coordinate\n */\nexport function bezier(p1x: number, p1y: number, p2x: number, p2y: number): (t: number) => number {\n const bezier = new UnitBezier(p1x, p1y, p2x, p2y);\n return function(t: number) {\n return bezier.solve(t);\n };\n}\n\n/**\n * A default bezier-curve powered easing function with\n * control points (0.25, 0.1) and (0.25, 1)\n */\nexport const defaultEasing = bezier(0.25, 0.1, 0.25, 1);\n\n/**\n * constrain n to the given range via min + max\n *\n * @param n - value\n * @param min - the minimum value to be returned\n * @param max - the maximum value to be returned\n * @returns the clamped value\n */\nexport function clamp(n: number, min: number, max: number): number {\n return Math.min(max, Math.max(min, n));\n}\n\n/**\n * constrain n to the given range, excluding the minimum, via modular arithmetic\n *\n * @param n - value\n * @param min - the minimum value to be returned, exclusive\n * @param max - the maximum value to be returned, inclusive\n * @returns constrained number\n */\nexport function wrap(n: number, min: number, max: number): number {\n const d = max - min;\n const w = ((n - min) % d + d) % d + min;\n return (w === min) ? max : w;\n}\n\n/**\n * Call an asynchronous function on an array of arguments,\n * calling `callback` with the completed results of all calls.\n *\n * @param array - input to each call of the async function.\n * @param fn - an async function with signature (data, callback)\n * @param callback - a callback run after all async work is done.\n * called with an array, containing the results of each async call.\n */\nexport function asyncAll(\n array: Array,\n fn: (item: Item, fnCallback: Callback) => void,\n callback: Callback>\n) {\n if (!array.length) { return callback(null, []); }\n let remaining = array.length;\n const results = new Array(array.length);\n let error = null;\n array.forEach((item, i) => {\n fn(item, (err, result) => {\n if (err) error = err;\n results[i] = (result as any as Result); // https://github.com/facebook/flow/issues/2123\n if (--remaining === 0) callback(error, results);\n });\n });\n}\n\n/**\n * Compute the difference between the keys in one object and the keys\n * in another object.\n *\n * @returns keys difference\n */\nexport function keysDifference(\n obj: {[key: string]: S},\n other: {[key: string]: T}\n): Array {\n const difference = [];\n for (const i in obj) {\n if (!(i in other)) {\n difference.push(i);\n }\n }\n return difference;\n}\n\n/**\n * Given a destination object and optionally many source objects,\n * copy all properties from the source objects into the destination.\n * The last source object given overrides properties from previous\n * source objects.\n *\n * @param dest - destination object\n * @param sources - sources from which properties are pulled\n */\nexport function extend(dest: any, ...sources: Array): any {\n for (const src of sources) {\n for (const k in src) {\n dest[k] = src[k];\n }\n }\n return dest;\n}\n\n/**\n * Given an object and a number of properties as strings, return version\n * of that object with only those properties.\n *\n * @param src - the object\n * @param properties - an array of property names chosen\n * to appear on the resulting object.\n * @returns object with limited properties.\n * @example\n * ```ts\n * let foo = { name: 'Charlie', age: 10 };\n * let justName = pick(foo, ['name']); // justName = { name: 'Charlie' }\n * ```\n */\nexport function pick(src: any, properties: Array): any {\n const result = {};\n for (let i = 0; i < properties.length; i++) {\n const k = properties[i];\n if (k in src) {\n result[k] = src[k];\n }\n }\n return result;\n}\n\nlet id = 1;\n\n/**\n * Return a unique numeric id, starting at 1 and incrementing with\n * each call.\n *\n * @returns unique numeric id.\n */\nexport function uniqueId(): number {\n return id++;\n}\n\n/**\n * Return whether a given value is a power of two\n */\nexport function isPowerOfTwo(value: number): boolean {\n return (Math.log(value) / Math.LN2) % 1 === 0;\n}\n\n/**\n * Return the next power of two, or the input value if already a power of two\n */\nexport function nextPowerOfTwo(value: number): number {\n if (value <= 1) return 1;\n return Math.pow(2, Math.ceil(Math.log(value) / Math.LN2));\n}\n\n/**\n * Create an object by mapping all the values of an existing object while\n * preserving their keys.\n */\nexport function mapObject(input: any, iterator: Function, context?: any): any {\n const output = {};\n for (const key in input) {\n output[key] = iterator.call(context || this, input[key], key, input);\n }\n return output;\n}\n\n/**\n * Create an object by filtering out values of an existing object.\n */\nexport function filterObject(input: any, iterator: Function, context?: any): any {\n const output = {};\n for (const key in input) {\n if (iterator.call(context || this, input[key], key, input)) {\n output[key] = input[key];\n }\n }\n return output;\n}\n\n/**\n * Deeply compares two object literals.\n * @param a - first object literal to be compared\n * @param b - second object literal to be compared\n * @returns true if the two object literals are deeply equal, false otherwise\n */\nexport function deepEqual(a?: unknown | null, b?: unknown | null): boolean {\n if (Array.isArray(a)) {\n if (!Array.isArray(b) || a.length !== b.length) return false;\n for (let i = 0; i < a.length; i++) {\n if (!deepEqual(a[i], b[i])) return false;\n }\n return true;\n }\n if (typeof a === 'object' && a !== null && b !== null) {\n if (!(typeof b === 'object')) return false;\n const keys = Object.keys(a);\n if (keys.length !== Object.keys(b).length) return false;\n for (const key in a) {\n if (!deepEqual(a[key], b[key])) return false;\n }\n return true;\n }\n return a === b;\n}\n\n/**\n * Deeply clones two objects.\n */\nexport function clone(input: T): T {\n if (Array.isArray(input)) {\n return input.map(clone) as any as T;\n } else if (typeof input === 'object' && input) {\n return mapObject(input, clone) as any as T;\n } else {\n return input;\n }\n}\n\n/**\n * Check if two arrays have at least one common element.\n */\nexport function arraysIntersect(a: Array, b: Array): boolean {\n for (let l = 0; l < a.length; l++) {\n if (b.indexOf(a[l]) >= 0) return true;\n }\n return false;\n}\n\n/**\n * Print a warning message to the console and ensure duplicate warning messages\n * are not printed.\n */\nconst warnOnceHistory: {[key: string]: boolean} = {};\n\nexport function warnOnce(message: string): void {\n if (!warnOnceHistory[message]) {\n // console isn't defined in some WebWorkers, see #2558\n if (typeof console !== 'undefined') console.warn(message);\n warnOnceHistory[message] = true;\n }\n}\n\n/**\n * Indicates if the provided Points are in a counter clockwise (true) or clockwise (false) order\n *\n * @returns true for a counter clockwise set of points\n */\n// http://bryceboe.com/2006/10/23/line-segment-intersection-algorithm/\nexport function isCounterClockwise(a: Point, b: Point, c: Point): boolean {\n return (c.y - a.y) * (b.x - a.x) > (b.y - a.y) * (c.x - a.x);\n}\n\n/**\n * For two lines a and b in 2d space, defined by any two points along the lines,\n * find the intersection point, or return null if the lines are parallel\n *\n * @param a1 - First point on line a\n * @param a2 - Second point on line a\n * @param b1 - First point on line b\n * @param b2 - Second point on line b\n *\n * @returns the intersection point of the two lines or null if they are parallel\n */\nexport function findLineIntersection(a1: Point, a2: Point, b1: Point, b2: Point): Point | null {\n const aDeltaY = a2.y - a1.y;\n const aDeltaX = a2.x - a1.x;\n const bDeltaY = b2.y - b1.y;\n const bDeltaX = b2.x - b1.x;\n\n const denominator = (bDeltaY * aDeltaX) - (bDeltaX * aDeltaY);\n\n if (denominator === 0) {\n // Lines are parallel\n return null;\n }\n\n const originDeltaY = a1.y - b1.y;\n const originDeltaX = a1.x - b1.x;\n const aInterpolation = (bDeltaX * originDeltaY - bDeltaY * originDeltaX) / denominator;\n\n // Find intersection by projecting out from origin of first segment\n return new Point(a1.x + (aInterpolation * aDeltaX), a1.y + (aInterpolation * aDeltaY));\n}\n\n/**\n * Returns the signed area for the polygon ring. Positive areas are exterior rings and\n * have a clockwise winding. Negative areas are interior rings and have a counter clockwise\n * ordering.\n *\n * @param ring - Exterior or interior ring\n */\nexport function calculateSignedArea(ring: Array): number {\n let sum = 0;\n for (let i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) {\n p1 = ring[i];\n p2 = ring[j];\n sum += (p2.x - p1.x) * (p1.y + p2.y);\n }\n return sum;\n}\n\n/**\n * Detects closed polygons, first + last point are equal\n *\n * @param points - array of points\n * @returns `true` if the points are a closed polygon\n */\nexport function isClosedPolygon(points: Array): boolean {\n // If it is 2 points that are the same then it is a point\n // If it is 3 points with start and end the same then it is a line\n if (points.length < 4)\n return false;\n\n const p1 = points[0];\n const p2 = points[points.length - 1];\n\n if (Math.abs(p1.x - p2.x) > 0 ||\n Math.abs(p1.y - p2.y) > 0) {\n return false;\n }\n\n // polygon simplification can produce polygons with zero area and more than 3 points\n return Math.abs(calculateSignedArea(points)) > 0.01;\n}\n\n/**\n * Converts spherical coordinates to cartesian coordinates.\n *\n * @param spherical - Spherical coordinates, in [radial, azimuthal, polar]\n * @returns cartesian coordinates in [x, y, z]\n */\n\nexport function sphericalToCartesian([r, azimuthal, polar]: [number, number, number]): {\n x: number;\n y: number;\n z: number;\n} {\n // We abstract \"north\"/\"up\" (compass-wise) to be 0° when really this is 90° (π/2):\n // correct for that here\n azimuthal += 90;\n\n // Convert azimuthal and polar angles to radians\n azimuthal *= Math.PI / 180;\n polar *= Math.PI / 180;\n\n return {\n x: r * Math.cos(azimuthal) * Math.sin(polar),\n y: r * Math.sin(azimuthal) * Math.sin(polar),\n z: r * Math.cos(polar)\n };\n}\n\n/**\n * Returns true if the when run in the web-worker context.\n *\n * @returns `true` if the when run in the web-worker context.\n */\nexport function isWorker(): boolean {\n // @ts-ignore\n return typeof WorkerGlobalScope !== 'undefined' && typeof self !== 'undefined' && self instanceof WorkerGlobalScope;\n}\n\n/**\n * Parses data from 'Cache-Control' headers.\n *\n * @param cacheControl - Value of 'Cache-Control' header\n * @returns object containing parsed header info.\n */\n\nexport function parseCacheControl(cacheControl: string): any {\n // Taken from [Wreck](https://github.com/hapijs/wreck)\n const re = /(?:^|(?:\\s*\\,\\s*))([^\\x00-\\x20\\(\\)<>@\\,;\\:\\\\\"\\/\\[\\]\\?\\=\\{\\}\\x7F]+)(?:\\=(?:([^\\x00-\\x20\\(\\)<>@\\,;\\:\\\\\"\\/\\[\\]\\?\\=\\{\\}\\x7F]+)|(?:\\\"((?:[^\"\\\\]|\\\\.)*)\\\")))?/g;\n\n const header = {};\n cacheControl.replace(re, ($0, $1, $2, $3) => {\n const value = $2 || $3;\n header[$1] = value ? value.toLowerCase() : true;\n return '';\n });\n\n if (header['max-age']) {\n const maxAge = parseInt(header['max-age'], 10);\n if (isNaN(maxAge)) delete header['max-age'];\n else header['max-age'] = maxAge;\n }\n\n return header;\n}\n\nlet _isSafari = null;\n\n/**\n * Returns true when run in WebKit derived browsers.\n * This is used as a workaround for a memory leak in Safari caused by using Transferable objects to\n * transfer data between WebWorkers and the main thread.\n * https://github.com/mapbox/mapbox-gl-js/issues/8771\n *\n * This should be removed once the underlying Safari issue is fixed.\n *\n * @param scope - Since this function is used both on the main thread and WebWorker context,\n * let the calling scope pass in the global scope object.\n * @returns `true` when run in WebKit derived browsers.\n */\nexport function isSafari(scope: any): boolean {\n if (_isSafari == null) {\n const userAgent = scope.navigator ? scope.navigator.userAgent : null;\n _isSafari = !!scope.safari ||\n !!(userAgent && (/\\b(iPad|iPhone|iPod)\\b/.test(userAgent) || (!!userAgent.match('Safari') && !userAgent.match('Chrome'))));\n }\n return _isSafari;\n}\n\nexport function storageAvailable(type: string): boolean {\n try {\n const storage = window[type];\n storage.setItem('_mapbox_test_', 1);\n storage.removeItem('_mapbox_test_');\n return true;\n } catch (e) {\n return false;\n }\n}\n\n// The following methods are from https://developer.mozilla.org/en-US/docs/Web/API/WindowBase64/Base64_encoding_and_decoding#The_Unicode_Problem\n//Unicode compliant base64 encoder for strings\nexport function b64EncodeUnicode(str: string) {\n return btoa(\n encodeURIComponent(str).replace(/%([0-9A-F]{2})/g,\n (match, p1) => {\n return String.fromCharCode(Number('0x' + p1)); //eslint-disable-line\n }\n )\n );\n}\n\n// Unicode compliant decoder for base64-encoded strings\nexport function b64DecodeUnicode(str: string) {\n return decodeURIComponent(atob(str).split('').map((c) => {\n return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2); //eslint-disable-line\n }).join(''));\n}\n\nexport function isImageBitmap(image: any): image is ImageBitmap {\n return typeof ImageBitmap !== 'undefined' && image instanceof ImageBitmap;\n}\n\n/**\n * Converts an ArrayBuffer to an ImageBitmap.\n *\n * Used mostly for testing purposes only, because mocking libs don't know how to work with ArrayBuffers, but work\n * perfectly fine with ImageBitmaps. Might also be used for environments (other than testing) not supporting\n * ArrayBuffers.\n *\n * @param data - Data to convert\n * @param callback - A callback executed after the conversion is finished. Invoked with error (if any) as the first argument and resulting image bitmap (when no error) as the second\n */\nexport function arrayBufferToImageBitmap(data: ArrayBuffer, callback: (err?: Error | null, image?: ImageBitmap | null) => void) {\n const blob: Blob = new Blob([new Uint8Array(data)], {type: 'image/png'});\n createImageBitmap(blob).then((imgBitmap) => {\n callback(null, imgBitmap);\n }).catch((e) => {\n callback(new Error(`Could not load image because of ${e.message}. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.`));\n });\n}\n\nconst transparentPngUrl = '';\n\n/**\n * Converts an ArrayBuffer to an HTMLImageElement.\n *\n * Used mostly for testing purposes only, because mocking libs don't know how to work with ArrayBuffers, but work\n * perfectly fine with ImageBitmaps. Might also be used for environments (other than testing) not supporting\n * ArrayBuffers.\n *\n * @param data - Data to convert\n * @param callback - A callback executed after the conversion is finished. Invoked with error (if any) as the first argument and resulting image element (when no error) as the second\n */\nexport function arrayBufferToImage(data: ArrayBuffer, callback: (err?: Error | null, image?: HTMLImageElement | null) => void) {\n const img: HTMLImageElement = new Image();\n img.onload = () => {\n callback(null, img);\n URL.revokeObjectURL(img.src);\n // prevent image dataURI memory leak in Safari;\n // but don't free the image immediately because it might be uploaded in the next frame\n // https://github.com/mapbox/mapbox-gl-js/issues/10226\n img.onload = null;\n window.requestAnimationFrame(() => { img.src = transparentPngUrl; });\n };\n img.onerror = () => callback(new Error('Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.'));\n const blob: Blob = new Blob([new Uint8Array(data)], {type: 'image/png'});\n img.src = data.byteLength ? URL.createObjectURL(blob) : transparentPngUrl;\n}\n\n/**\n * Computes the webcodecs VideoFrame API options to select a rectangle out of\n * an image and write it into the destination rectangle.\n *\n * Rect (x/y/width/height) select the overlapping rectangle from the source image\n * and layout (offset/stride) write that overlapping rectangle to the correct place\n * in the destination image.\n *\n * Offset is the byte offset in the dest image that the first pixel appears at\n * and stride is the number of bytes to the start of the next row:\n * ┌───────────┐\n * │ dest │\n * │ ┌───┼───────┐\n * │offset→│▓▓▓│ source│\n * │ │▓▓▓│ │\n * │ └───┼───────┘\n * │stride ⇠╌╌╌│\n * │╌╌╌╌╌╌→ │\n * └───────────┘\n *\n * @param image - source image containing a width and height attribute\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns the layout and rect options to pass into VideoFrame API\n */\nfunction computeVideoFrameParameters(image: Size, x: number, y: number, width: number, height: number): VideoFrameCopyToOptions {\n const destRowOffset = Math.max(-x, 0) * 4;\n const firstSourceRow = Math.max(0, y);\n const firstDestRow = firstSourceRow - y;\n const offset = firstDestRow * width * 4 + destRowOffset;\n const stride = width * 4;\n\n const sourceLeft = Math.max(0, x);\n const sourceTop = Math.max(0, y);\n const sourceRight = Math.min(image.width, x + width);\n const sourceBottom = Math.min(image.height, y + height);\n return {\n rect: {\n x: sourceLeft,\n y: sourceTop,\n width: sourceRight - sourceLeft,\n height: sourceBottom - sourceTop\n },\n layout: [{offset, stride}]\n };\n}\n\n/**\n * Reads pixels from an ImageBitmap/Image/canvas using webcodec VideoFrame API.\n *\n * @param data - image, imagebitmap, or canvas to parse\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns a promise containing the parsed RGBA pixel values of the image, or the error if an error occurred\n */\nexport async function readImageUsingVideoFrame(\n image: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas,\n x: number, y: number, width: number, height: number\n): Promise {\n if (typeof VideoFrame === 'undefined') {\n throw new Error('VideoFrame not supported');\n }\n const frame = new VideoFrame(image, {timestamp: 0});\n try {\n const format = frame?.format;\n if (!format || !(format.startsWith('BGR') || format.startsWith('RGB'))) {\n throw new Error(`Unrecognized format ${format}`);\n }\n const swapBR = format.startsWith('BGR');\n const result = new Uint8ClampedArray(width * height * 4);\n await frame.copyTo(result, computeVideoFrameParameters(image, x, y, width, height));\n if (swapBR) {\n for (let i = 0; i < result.length; i += 4) {\n const tmp = result[i];\n result[i] = result[i + 2];\n result[i + 2] = tmp;\n }\n }\n return result;\n } finally {\n frame.close();\n }\n}\n\nlet offscreenCanvas: OffscreenCanvas;\nlet offscreenCanvasContext: OffscreenCanvasRenderingContext2D;\n\n/**\n * Reads pixels from an ImageBitmap/Image/canvas using OffscreenCanvas\n *\n * @param data - image, imagebitmap, or canvas to parse\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns a promise containing the parsed RGBA pixel values of the image, or the error if an error occurred\n */\nexport function readImageDataUsingOffscreenCanvas(\n imgBitmap: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas,\n x: number, y: number, width: number, height: number\n): Uint8ClampedArray {\n const origWidth = imgBitmap.width;\n const origHeight = imgBitmap.height;\n // Lazily initialize OffscreenCanvas\n if (!offscreenCanvas || !offscreenCanvasContext) {\n // Dem tiles are typically 256x256\n offscreenCanvas = new OffscreenCanvas(origWidth, origHeight);\n offscreenCanvasContext = offscreenCanvas.getContext('2d', {willReadFrequently: true});\n }\n\n offscreenCanvas.width = origWidth;\n offscreenCanvas.height = origHeight;\n\n offscreenCanvasContext.drawImage(imgBitmap, 0, 0, origWidth, origHeight);\n const imgData = offscreenCanvasContext.getImageData(x, y, width, height);\n offscreenCanvasContext.clearRect(0, 0, origWidth, origHeight);\n return imgData.data;\n}\n\n/**\n * Reads RGBA pixels from an preferring OffscreenCanvas, but falling back to VideoFrame if supported and\n * the browser is mangling OffscreenCanvas getImageData results.\n *\n * @param data - image, imagebitmap, or canvas to parse\n * @param x - top-left x coordinate to read from the image\n * @param y - top-left y coordinate to read from the image\n * @param width - width of the rectangle to read from the image\n * @param height - height of the rectangle to read from the image\n * @returns a promise containing the parsed RGBA pixel values of the image\n */\nexport async function getImageData(\n image: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas,\n x: number, y: number, width: number, height: number\n): Promise {\n if (isOffscreenCanvasDistorted()) {\n try {\n return await readImageUsingVideoFrame(image, x, y, width, height);\n } catch (e) {\n // fall back to OffscreenCanvas\n }\n }\n return readImageDataUsingOffscreenCanvas(image, x, y, width, height);\n}\n","import type {Cancelable} from '../types/cancelable';\n\nconst now = typeof performance !== 'undefined' && performance && performance.now ?\n performance.now.bind(performance) :\n Date.now.bind(Date);\n\nlet linkEl;\n\nlet reducedMotionQuery: MediaQueryList;\n\n/** */\nexport const browser = {\n /**\n * Provides a function that outputs milliseconds: either performance.now()\n * or a fallback to Date.now()\n */\n now,\n\n frame(fn: (paintStartTimestamp: number) => void): Cancelable {\n const frame = requestAnimationFrame(fn);\n return {cancel: () => cancelAnimationFrame(frame)};\n },\n\n getImageData(img: HTMLImageElement | ImageBitmap, padding: number = 0): ImageData {\n const context = this.getImageCanvasContext(img);\n return context.getImageData(-padding, -padding, img.width as number + 2 * padding, img.height as number + 2 * padding);\n },\n\n getImageCanvasContext(img: HTMLImageElement | ImageBitmap): CanvasRenderingContext2D {\n const canvas = window.document.createElement('canvas') as HTMLCanvasElement;\n const context = canvas.getContext('2d', {willReadFrequently: true});\n if (!context) {\n throw new Error('failed to create canvas 2d context');\n }\n canvas.width = img.width as number;\n canvas.height = img.height as number;\n context.drawImage(img, 0, 0, img.width as number, img.height as number);\n return context;\n },\n\n resolveURL(path: string) {\n if (!linkEl) linkEl = document.createElement('a');\n linkEl.href = path;\n return linkEl.href;\n },\n\n hardwareConcurrency: typeof navigator !== 'undefined' && navigator.hardwareConcurrency || 4,\n\n get prefersReducedMotion(): boolean {\n // In case your test crashes when checking matchMedia, call setMatchMedia from 'src/util/test/util'\n if (!matchMedia) return false;\n //Lazily initialize media query\n if (reducedMotionQuery == null) {\n reducedMotionQuery = matchMedia('(prefers-reduced-motion: reduce)');\n }\n return reducedMotionQuery.matches;\n },\n};\n","import type {Cancelable} from '../types/cancelable';\nimport type {RequestParameters, ResponseCallback} from './ajax';\n\n/**\n * This is a global config object used to store the configuration\n * It is available in the workers as well.\n * Only serializable data should be stored in it.\n */\ntype Config = {\n MAX_PARALLEL_IMAGE_REQUESTS: number;\n MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME: number;\n MAX_TILE_CACHE_ZOOM_LEVELS: number;\n REGISTERED_PROTOCOLS: {[x: string]: (requestParameters: RequestParameters, callback: ResponseCallback) => Cancelable};\n WORKER_URL: string;\n};\n\nexport const config: Config = {\n MAX_PARALLEL_IMAGE_REQUESTS: 16,\n MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME: 8,\n MAX_TILE_CACHE_ZOOM_LEVELS: 5,\n REGISTERED_PROTOCOLS: {},\n WORKER_URL: ''\n};\n","import {extend, warnOnce, isWorker} from './util';\nimport {config} from './config';\n\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\n\n/**\n * A `RequestParameters` object to be returned from Map.options.transformRequest callbacks.\n * @example\n * ```ts\n * // use transformRequest to modify requests that begin with `http://myHost`\n * transformRequest: function(url, resourceType) {\n * if (resourceType === 'Source' && url.indexOf('http://myHost') > -1) {\n * return {\n * url: url.replace('http', 'https'),\n * headers: { 'my-custom-header': true },\n * credentials: 'include' // Include cookies for cross-origin requests\n * }\n * }\n * }\n * ```\n */\nexport type RequestParameters = {\n /**\n * The URL to be requested.\n */\n url: string;\n /**\n * The headers to be sent with the request.\n */\n headers?: any;\n /**\n * Request method `'GET' | 'POST' | 'PUT'`.\n */\n method?: 'GET' | 'POST' | 'PUT';\n /**\n * Request body.\n */\n body?: string;\n /**\n * Response body type to be returned `'string' | 'json' | 'arrayBuffer'`.\n */\n type?: 'string' | 'json' | 'arrayBuffer' | 'image';\n /**\n * `'same-origin'|'include'` Use 'include' to send cookies with cross-origin requests.\n */\n credentials?: 'same-origin' | 'include';\n /**\n * If `true`, Resource Timing API information will be collected for these transformed requests and returned in a resourceTiming property of relevant data events.\n */\n collectResourceTiming?: boolean;\n /**\n * Parameters supported only by browser fetch API. Property of the Request interface contains the cache mode of the request. It controls how the request will interact with the browser's HTTP cache. (https://developer.mozilla.org/en-US/docs/Web/API/Request/cache)\n */\n cache?: RequestCache;\n};\n\n/**\n * The response callback used in various places\n */\nexport type ResponseCallback = (\n error?: Error | null,\n data?: T | null,\n cacheControl?: string | null,\n expires?: string | null\n) => void;\n\n/**\n * An error thrown when a HTTP request results in an error response.\n */\nexport class AJAXError extends Error {\n /**\n * The response's HTTP status code.\n */\n status: number;\n\n /**\n * The response's HTTP status text.\n */\n statusText: string;\n\n /**\n * The request's URL.\n */\n url: string;\n\n /**\n * The response's body.\n */\n body: Blob;\n\n /**\n * @param status - The response's HTTP status code.\n * @param statusText - The response's HTTP status text.\n * @param url - The request's URL.\n * @param body - The response's body.\n */\n constructor(status: number, statusText: string, url: string, body: Blob) {\n super(`AJAXError: ${statusText} (${status}): ${url}`);\n this.status = status;\n this.statusText = statusText;\n this.url = url;\n this.body = body;\n }\n}\n\n// Ensure that we're sending the correct referrer from blob URL worker bundles.\n// For files loaded from the local file system, `location.origin` will be set\n// to the string(!) \"null\" (Firefox), or \"file://\" (Chrome, Safari, Edge, IE),\n// and we will set an empty referrer. Otherwise, we're using the document's URL.\n/* global self */\nexport const getReferrer = isWorker() ?\n () => (self as any).worker && (self as any).worker.referrer :\n () => (window.location.protocol === 'blob:' ? window.parent : window).location.href;\n\nexport const getProtocolAction = url => config.REGISTERED_PROTOCOLS[url.substring(0, url.indexOf('://'))];\n\n// Determines whether a URL is a file:// URL. This is obviously the case if it begins\n// with file://. Relative URLs are also file:// URLs iff the original document was loaded\n// via a file:// URL.\nconst isFileURL = url => /^file:/.test(url) || (/^file:/.test(getReferrer()) && !/^\\w+:/.test(url));\n\nfunction makeFetchRequest(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n const controller = new AbortController();\n const request = new Request(requestParameters.url, {\n method: requestParameters.method || 'GET',\n body: requestParameters.body,\n credentials: requestParameters.credentials,\n headers: requestParameters.headers,\n cache: requestParameters.cache,\n referrer: getReferrer(),\n signal: controller.signal\n });\n let complete = false;\n let aborted = false;\n\n if (requestParameters.type === 'json') {\n request.headers.set('Accept', 'application/json');\n }\n\n const validateOrFetch = (err, cachedResponse?, responseIsFresh?) => {\n if (aborted) return;\n\n if (err) {\n // Do fetch in case of cache error.\n // HTTP pages in Edge trigger a security error that can be ignored.\n if (err.message !== 'SecurityError') {\n warnOnce(err);\n }\n }\n\n if (cachedResponse && responseIsFresh) {\n return finishRequest(cachedResponse);\n }\n\n if (cachedResponse) {\n // We can't do revalidation with 'If-None-Match' because then the\n // request doesn't have simple cors headers.\n }\n\n fetch(request).then(response => {\n if (response.ok) {\n return finishRequest(response);\n\n } else {\n return response.blob().then(body => callback(new AJAXError(response.status, response.statusText, requestParameters.url, body)));\n }\n }).catch(error => {\n if (error.code === 20) {\n // silence expected AbortError\n return;\n }\n callback(new Error(error.message));\n });\n };\n\n const finishRequest = (response) => {\n (\n (requestParameters.type === 'arrayBuffer' || requestParameters.type === 'image') ? response.arrayBuffer() :\n requestParameters.type === 'json' ? response.json() :\n response.text()\n ).then(result => {\n if (aborted) return;\n complete = true;\n callback(null, result, response.headers.get('Cache-Control'), response.headers.get('Expires'));\n }).catch(err => {\n if (!aborted) callback(new Error(err.message));\n });\n };\n\n validateOrFetch(null, null);\n\n return {cancel: () => {\n aborted = true;\n if (!complete) controller.abort();\n }};\n}\n\nfunction makeXMLHttpRequest(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n const xhr: XMLHttpRequest = new XMLHttpRequest();\n\n xhr.open(requestParameters.method || 'GET', requestParameters.url, true);\n if (requestParameters.type === 'arrayBuffer' || requestParameters.type === 'image') {\n xhr.responseType = 'arraybuffer';\n }\n for (const k in requestParameters.headers) {\n xhr.setRequestHeader(k, requestParameters.headers[k]);\n }\n if (requestParameters.type === 'json') {\n xhr.responseType = 'text';\n xhr.setRequestHeader('Accept', 'application/json');\n }\n xhr.withCredentials = requestParameters.credentials === 'include';\n xhr.onerror = () => {\n callback(new Error(xhr.statusText));\n };\n xhr.onload = () => {\n if (((xhr.status >= 200 && xhr.status < 300) || xhr.status === 0) && xhr.response !== null) {\n let data: unknown = xhr.response;\n if (requestParameters.type === 'json') {\n // We're manually parsing JSON here to get better error messages.\n try {\n data = JSON.parse(xhr.response);\n } catch (err) {\n return callback(err);\n }\n }\n callback(null, data, xhr.getResponseHeader('Cache-Control'), xhr.getResponseHeader('Expires'));\n } else {\n const body = new Blob([xhr.response], {type: xhr.getResponseHeader('Content-Type')});\n callback(new AJAXError(xhr.status, xhr.statusText, requestParameters.url, body));\n }\n };\n xhr.send(requestParameters.body);\n return {cancel: () => xhr.abort()};\n}\n\nexport const makeRequest = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n // We're trying to use the Fetch API if possible. However, in some situations we can't use it:\n // - IE11 doesn't support it at all. In this case, we dispatch the request to the main thread so\n // that we can get an accruate referrer header.\n // - Safari exposes window.AbortController, but it doesn't work actually abort any requests in\n // some versions (see https://bugs.webkit.org/show_bug.cgi?id=174980#c2)\n // - Requests for resources with the file:// URI scheme don't work with the Fetch API either. In\n // this case we unconditionally use XHR on the current thread since referrers don't matter.\n if (/:\\/\\//.test(requestParameters.url) && !(/^https?:|^file:/.test(requestParameters.url))) {\n if (isWorker() && (self as any).worker && (self as any).worker.actor) {\n return (self as any).worker.actor.send('getResource', requestParameters, callback);\n }\n if (!isWorker()) {\n const action = getProtocolAction(requestParameters.url) || makeFetchRequest;\n return action(requestParameters, callback);\n }\n }\n if (!isFileURL(requestParameters.url)) {\n if (fetch && Request && AbortController && Object.prototype.hasOwnProperty.call(Request.prototype, 'signal')) {\n return makeFetchRequest(requestParameters, callback);\n }\n if (isWorker() && (self as any).worker && (self as any).worker.actor) {\n const queueOnMainThread = true;\n return (self as any).worker.actor.send('getResource', requestParameters, callback, undefined, queueOnMainThread);\n }\n }\n return makeXMLHttpRequest(requestParameters, callback);\n};\n\nexport const getJSON = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n return makeRequest(extend(requestParameters, {type: 'json'}), callback);\n};\n\nexport const getArrayBuffer = function(\n requestParameters: RequestParameters,\n callback: ResponseCallback\n): Cancelable {\n return makeRequest(extend(requestParameters, {type: 'arrayBuffer'}), callback);\n};\n\nexport const postData = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable {\n return makeRequest(extend(requestParameters, {method: 'POST'}), callback);\n};\n\nexport function sameOrigin(inComingUrl: string) {\n // URL class should be available everywhere\n // https://developer.mozilla.org/en-US/docs/Web/API/URL\n // In addtion, a relative URL \"/foo\" or \"./foo\" will throw exception in its ctor,\n // try-catch is expansive so just use a heuristic check to avoid it\n // also check data URL\n if (!inComingUrl ||\n inComingUrl.indexOf('://') <= 0 || // relative URL\n inComingUrl.indexOf('data:image/') === 0 || // data image URL\n inComingUrl.indexOf('blob:') === 0) { // blob\n return true;\n }\n const urlObj = new URL(inComingUrl);\n const locationObj = window.location;\n return urlObj.protocol === locationObj.protocol && urlObj.host === locationObj.host;\n}\n/**\n * A type used to store the tile's expiration date and cache control definition\n */\nexport type ExpiryData = {cacheControl?: string | null; expires?: Date | string | null};\nexport const getVideo = function(urls: Array, callback: Callback): Cancelable {\n const video: HTMLVideoElement = window.document.createElement('video');\n video.muted = true;\n video.onloadstart = function() {\n callback(null, video);\n };\n for (let i = 0; i < urls.length; i++) {\n const s: HTMLSourceElement = window.document.createElement('source');\n if (!sameOrigin(urls[i])) {\n video.crossOrigin = 'Anonymous';\n }\n s.src = urls[i];\n video.appendChild(s);\n }\n return {cancel: () => {}};\n};\n","import {extend} from './util';\n\n/**\n * A listener method used as a callback to events\n */\nexport type Listener = (a: any) => any;\n\ntype Listeners = {[_: string]: Array};\n\nfunction _addEventListener(type: string, listener: Listener, listenerList: Listeners) {\n const listenerExists = listenerList[type] && listenerList[type].indexOf(listener) !== -1;\n if (!listenerExists) {\n listenerList[type] = listenerList[type] || [];\n listenerList[type].push(listener);\n }\n}\n\nfunction _removeEventListener(type: string, listener: Listener, listenerList: Listeners) {\n if (listenerList && listenerList[type]) {\n const index = listenerList[type].indexOf(listener);\n if (index !== -1) {\n listenerList[type].splice(index, 1);\n }\n }\n}\n\n/**\n * The event class\n */\nexport class Event {\n readonly type: string;\n\n constructor(type: string, data: any = {}) {\n extend(this, data);\n this.type = type;\n }\n}\n\ninterface ErrorLike {\n message: string;\n}\n\n/**\n * An error event\n */\nexport class ErrorEvent extends Event {\n error: ErrorLike;\n\n constructor(error: ErrorLike, data: any = {}) {\n super('error', extend({error}, data));\n }\n}\n\n/**\n * Methods mixed in to other classes for event capabilities.\n *\n * @group Event Related\n */\nexport class Evented {\n _listeners: Listeners;\n _oneTimeListeners: Listeners;\n _eventedParent: Evented;\n _eventedParentData: any | (() => any);\n\n /**\n * Adds a listener to a specified event type.\n *\n * @param type - The event type to add a listen for.\n * @param listener - The function to be called when the event is fired.\n * The listener function is called with the data object passed to `fire`,\n * extended with `target` and `type` properties.\n * @returns `this`\n */\n on(type: string, listener: Listener): this {\n this._listeners = this._listeners || {};\n _addEventListener(type, listener, this._listeners);\n\n return this;\n }\n\n /**\n * Removes a previously registered event listener.\n *\n * @param type - The event type to remove listeners for.\n * @param listener - The listener function to remove.\n * @returns `this`\n */\n off(type: string, listener: Listener) {\n _removeEventListener(type, listener, this._listeners);\n _removeEventListener(type, listener, this._oneTimeListeners);\n\n return this;\n }\n\n /**\n * Adds a listener that will be called only once to a specified event type.\n *\n * The listener will be called first time the event fires after the listener is registered.\n *\n * @param type - The event type to listen for.\n * @param listener - The function to be called when the event is fired the first time.\n * @returns `this` or a promise if a listener is not provided\n */\n once(type: string, listener?: Listener): this | Promise {\n if (!listener) {\n return new Promise((resolve) => this.once(type, resolve));\n }\n this._oneTimeListeners = this._oneTimeListeners || {};\n _addEventListener(type, listener, this._oneTimeListeners);\n\n return this;\n }\n\n fire(event: Event | string, properties?: any) {\n // Compatibility with (type: string, properties: Object) signature from previous versions.\n // See https://github.com/mapbox/mapbox-gl-js/issues/6522,\n // https://github.com/mapbox/mapbox-gl-draw/issues/766\n if (typeof event === 'string') {\n event = new Event(event, properties || {});\n }\n\n const type = event.type;\n\n if (this.listens(type)) {\n (event as any).target = this;\n\n // make sure adding or removing listeners inside other listeners won't cause an infinite loop\n const listeners = this._listeners && this._listeners[type] ? this._listeners[type].slice() : [];\n for (const listener of listeners) {\n listener.call(this, event);\n }\n\n const oneTimeListeners = this._oneTimeListeners && this._oneTimeListeners[type] ? this._oneTimeListeners[type].slice() : [];\n for (const listener of oneTimeListeners) {\n _removeEventListener(type, listener, this._oneTimeListeners);\n listener.call(this, event);\n }\n\n const parent = this._eventedParent;\n if (parent) {\n extend(\n event,\n typeof this._eventedParentData === 'function' ? this._eventedParentData() : this._eventedParentData\n );\n parent.fire(event);\n }\n\n // To ensure that no error events are dropped, print them to the\n // console if they have no listeners.\n } else if (event instanceof ErrorEvent) {\n console.error(event.error);\n }\n\n return this;\n }\n\n /**\n * Returns a true if this instance of Evented or any forwardeed instances of Evented have a listener for the specified type.\n *\n * @param type - The event type\n * @returns `true` if there is at least one registered listener for specified event type, `false` otherwise\n */\n listens(type: string): boolean {\n return (\n (this._listeners && this._listeners[type] && this._listeners[type].length > 0) ||\n (this._oneTimeListeners && this._oneTimeListeners[type] && this._oneTimeListeners[type].length > 0) ||\n (this._eventedParent && this._eventedParent.listens(type))\n );\n }\n\n /**\n * Bubble all events fired by this instance of Evented to this parent instance of Evented.\n * @returns `this`\n */\n setEventedParent(parent?: Evented | null, data?: any | (() => any)) {\n this._eventedParent = parent;\n this._eventedParentData = data;\n\n return this;\n }\n}\n","import UnitBezier from '@mapbox/unitbezier';\n\nvar $version = 8;\nvar $root = {\n\tversion: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: [\n\t\t\t8\n\t\t]\n\t},\n\tname: {\n\t\ttype: \"string\"\n\t},\n\tmetadata: {\n\t\ttype: \"*\"\n\t},\n\tcenter: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\"\n\t},\n\tzoom: {\n\t\ttype: \"number\"\n\t},\n\tbearing: {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\tunits: \"degrees\"\n\t},\n\tpitch: {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tunits: \"degrees\"\n\t},\n\tlight: {\n\t\ttype: \"light\"\n\t},\n\tterrain: {\n\t\ttype: \"terrain\"\n\t},\n\tsources: {\n\t\trequired: true,\n\t\ttype: \"sources\"\n\t},\n\tsprite: {\n\t\ttype: \"sprite\"\n\t},\n\tglyphs: {\n\t\ttype: \"string\"\n\t},\n\ttransition: {\n\t\ttype: \"transition\"\n\t},\n\tlayers: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tvalue: \"layer\"\n\t}\n};\nvar sources = {\n\t\"*\": {\n\t\ttype: \"source\"\n\t}\n};\nvar source = [\n\t\"source_vector\",\n\t\"source_raster\",\n\t\"source_raster_dem\",\n\t\"source_geojson\",\n\t\"source_video\",\n\t\"source_image\"\n];\nvar source_vector = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvector: {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\ttype: \"string\"\n\t},\n\ttiles: {\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tbounds: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t-180,\n\t\t\t-85.051129,\n\t\t\t180,\n\t\t\t85.051129\n\t\t]\n\t},\n\tscheme: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\txyz: {\n\t\t\t},\n\t\t\ttms: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"xyz\"\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 22\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tpromoteId: {\n\t\ttype: \"promoteId\"\n\t},\n\tvolatile: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\t\"*\": {\n\t\ttype: \"*\"\n\t}\n};\nvar source_raster = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\traster: {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\ttype: \"string\"\n\t},\n\ttiles: {\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tbounds: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t-180,\n\t\t\t-85.051129,\n\t\t\t180,\n\t\t\t85.051129\n\t\t]\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 22\n\t},\n\ttileSize: {\n\t\ttype: \"number\",\n\t\t\"default\": 512,\n\t\tunits: \"pixels\"\n\t},\n\tscheme: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\txyz: {\n\t\t\t},\n\t\t\ttms: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"xyz\"\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tvolatile: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\t\"*\": {\n\t\ttype: \"*\"\n\t}\n};\nvar source_raster_dem = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\t\"raster-dem\": {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\ttype: \"string\"\n\t},\n\ttiles: {\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tbounds: {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t-180,\n\t\t\t-85.051129,\n\t\t\t180,\n\t\t\t85.051129\n\t\t]\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 22\n\t},\n\ttileSize: {\n\t\ttype: \"number\",\n\t\t\"default\": 512,\n\t\tunits: \"pixels\"\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tencoding: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tterrarium: {\n\t\t\t},\n\t\t\tmapbox: {\n\t\t\t},\n\t\t\tcustom: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"mapbox\"\n\t},\n\tredFactor: {\n\t\ttype: \"number\",\n\t\t\"default\": 1\n\t},\n\tblueFactor: {\n\t\ttype: \"number\",\n\t\t\"default\": 1\n\t},\n\tgreenFactor: {\n\t\ttype: \"number\",\n\t\t\"default\": 1\n\t},\n\tbaseShift: {\n\t\ttype: \"number\",\n\t\t\"default\": 0\n\t},\n\tvolatile: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\t\"*\": {\n\t\ttype: \"*\"\n\t}\n};\nvar source_geojson = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tgeojson: {\n\t\t\t}\n\t\t}\n\t},\n\tdata: {\n\t\trequired: true,\n\t\ttype: \"*\"\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\t\"default\": 18\n\t},\n\tattribution: {\n\t\ttype: \"string\"\n\t},\n\tbuffer: {\n\t\ttype: \"number\",\n\t\t\"default\": 128,\n\t\tmaximum: 512,\n\t\tminimum: 0\n\t},\n\tfilter: {\n\t\ttype: \"*\"\n\t},\n\ttolerance: {\n\t\ttype: \"number\",\n\t\t\"default\": 0.375\n\t},\n\tcluster: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\tclusterRadius: {\n\t\ttype: \"number\",\n\t\t\"default\": 50,\n\t\tminimum: 0\n\t},\n\tclusterMaxZoom: {\n\t\ttype: \"number\"\n\t},\n\tclusterMinPoints: {\n\t\ttype: \"number\"\n\t},\n\tclusterProperties: {\n\t\ttype: \"*\"\n\t},\n\tlineMetrics: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\tgenerateId: {\n\t\ttype: \"boolean\",\n\t\t\"default\": false\n\t},\n\tpromoteId: {\n\t\ttype: \"promoteId\"\n\t}\n};\nvar source_video = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvideo: {\n\t\t\t}\n\t\t}\n\t},\n\turls: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tvalue: \"string\"\n\t},\n\tcoordinates: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tlength: 4,\n\t\tvalue: {\n\t\t\ttype: \"array\",\n\t\t\tlength: 2,\n\t\t\tvalue: \"number\"\n\t\t}\n\t}\n};\nvar source_image = {\n\ttype: {\n\t\trequired: true,\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\timage: {\n\t\t\t}\n\t\t}\n\t},\n\turl: {\n\t\trequired: true,\n\t\ttype: \"string\"\n\t},\n\tcoordinates: {\n\t\trequired: true,\n\t\ttype: \"array\",\n\t\tlength: 4,\n\t\tvalue: {\n\t\t\ttype: \"array\",\n\t\t\tlength: 2,\n\t\t\tvalue: \"number\"\n\t\t}\n\t}\n};\nvar layer = {\n\tid: {\n\t\ttype: \"string\",\n\t\trequired: true\n\t},\n\ttype: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tfill: {\n\t\t\t},\n\t\t\tline: {\n\t\t\t},\n\t\t\tsymbol: {\n\t\t\t},\n\t\t\tcircle: {\n\t\t\t},\n\t\t\theatmap: {\n\t\t\t},\n\t\t\t\"fill-extrusion\": {\n\t\t\t},\n\t\t\traster: {\n\t\t\t},\n\t\t\thillshade: {\n\t\t\t},\n\t\t\tbackground: {\n\t\t\t}\n\t\t},\n\t\trequired: true\n\t},\n\tmetadata: {\n\t\ttype: \"*\"\n\t},\n\tsource: {\n\t\ttype: \"string\"\n\t},\n\t\"source-layer\": {\n\t\ttype: \"string\"\n\t},\n\tminzoom: {\n\t\ttype: \"number\",\n\t\tminimum: 0,\n\t\tmaximum: 24\n\t},\n\tmaxzoom: {\n\t\ttype: \"number\",\n\t\tminimum: 0,\n\t\tmaximum: 24\n\t},\n\tfilter: {\n\t\ttype: \"filter\"\n\t},\n\tlayout: {\n\t\ttype: \"layout\"\n\t},\n\tpaint: {\n\t\ttype: \"paint\"\n\t}\n};\nvar layout = [\n\t\"layout_fill\",\n\t\"layout_line\",\n\t\"layout_circle\",\n\t\"layout_heatmap\",\n\t\"layout_fill-extrusion\",\n\t\"layout_symbol\",\n\t\"layout_raster\",\n\t\"layout_hillshade\",\n\t\"layout_background\"\n];\nvar layout_background = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_fill = {\n\t\"fill-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_circle = {\n\t\"circle-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_heatmap = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_line = {\n\t\"line-cap\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tbutt: {\n\t\t\t},\n\t\t\tround: {\n\t\t\t},\n\t\t\tsquare: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"butt\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-join\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tbevel: {\n\t\t\t},\n\t\t\tround: {\n\t\t\t},\n\t\t\tmiter: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"miter\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-miter-limit\": {\n\t\ttype: \"number\",\n\t\t\"default\": 2,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"line-join\": \"miter\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-round-limit\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1.05,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"line-join\": \"round\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_symbol = {\n\t\"symbol-placement\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tpoint: {\n\t\t\t},\n\t\t\tline: {\n\t\t\t},\n\t\t\t\"line-center\": {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"point\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"symbol-spacing\": {\n\t\ttype: \"number\",\n\t\t\"default\": 250,\n\t\tminimum: 1,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"symbol-placement\": \"line\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"symbol-avoid-edges\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"symbol-sort-key\": {\n\t\ttype: \"number\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"symbol-z-order\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tauto: {\n\t\t\t},\n\t\t\t\"viewport-y\": {\n\t\t\t},\n\t\t\tsource: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-allow-overlap\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t{\n\t\t\t\t\"!\": \"icon-overlap\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-overlap\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnever: {\n\t\t\t},\n\t\t\talways: {\n\t\t\t},\n\t\t\tcooperative: {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-ignore-placement\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-optional\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-rotation-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-size\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tunits: \"factor of the original icon size\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-text-fit\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnone: {\n\t\t\t},\n\t\t\twidth: {\n\t\t\t},\n\t\t\theight: {\n\t\t\t},\n\t\t\tboth: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"none\",\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-text-fit-padding\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 4,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0,\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"icon-text-fit\": [\n\t\t\t\t\t\"both\",\n\t\t\t\t\t\"width\",\n\t\t\t\t\t\"height\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-image\": {\n\t\ttype: \"resolvedImage\",\n\t\ttokens: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-rotate\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\tunits: \"degrees\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-padding\": {\n\t\ttype: \"padding\",\n\t\t\"default\": [\n\t\t\t2\n\t\t],\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-keep-upright\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t{\n\t\t\t\t\"icon-rotation-alignment\": \"map\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"line\",\n\t\t\t\t\t\"line-center\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-offset\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t},\n\t\t\ttop: {\n\t\t\t},\n\t\t\tbottom: {\n\t\t\t},\n\t\t\t\"top-left\": {\n\t\t\t},\n\t\t\t\"top-right\": {\n\t\t\t},\n\t\t\t\"bottom-left\": {\n\t\t\t},\n\t\t\t\"bottom-right\": {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"center\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-pitch-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-pitch-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-rotation-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t},\n\t\t\t\"viewport-glyph\": {\n\t\t\t},\n\t\t\tauto: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"auto\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-field\": {\n\t\ttype: \"formatted\",\n\t\t\"default\": \"\",\n\t\ttokens: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-font\": {\n\t\ttype: \"array\",\n\t\tvalue: \"string\",\n\t\t\"default\": [\n\t\t\t\"Open Sans Regular\",\n\t\t\t\"Arial Unicode MS Regular\"\n\t\t],\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-size\": {\n\t\ttype: \"number\",\n\t\t\"default\": 16,\n\t\tminimum: 0,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-max-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 10,\n\t\tminimum: 0,\n\t\tunits: \"ems\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-line-height\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1.2,\n\t\tunits: \"ems\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-letter-spacing\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tunits: \"ems\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-justify\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tauto: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"center\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-radial-offset\": {\n\t\ttype: \"number\",\n\t\tunits: \"ems\",\n\t\t\"default\": 0,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\t\"property-type\": \"data-driven\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t}\n\t},\n\t\"text-variable-anchor\": {\n\t\ttype: \"array\",\n\t\tvalue: \"enum\",\n\t\tvalues: {\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t},\n\t\t\ttop: {\n\t\t\t},\n\t\t\tbottom: {\n\t\t\t},\n\t\t\t\"top-left\": {\n\t\t\t},\n\t\t\t\"top-right\": {\n\t\t\t},\n\t\t\t\"bottom-left\": {\n\t\t\t},\n\t\t\t\"bottom-right\": {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"point\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-variable-anchor-offset\": {\n\t\ttype: \"variableAnchorOffsetCollection\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"point\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tcenter: {\n\t\t\t},\n\t\t\tleft: {\n\t\t\t},\n\t\t\tright: {\n\t\t\t},\n\t\t\ttop: {\n\t\t\t},\n\t\t\tbottom: {\n\t\t\t},\n\t\t\t\"top-left\": {\n\t\t\t},\n\t\t\t\"top-right\": {\n\t\t\t},\n\t\t\t\"bottom-left\": {\n\t\t\t},\n\t\t\t\"bottom-right\": {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"center\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"!\": \"text-variable-anchor\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-max-angle\": {\n\t\ttype: \"number\",\n\t\t\"default\": 45,\n\t\tunits: \"degrees\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"line\",\n\t\t\t\t\t\"line-center\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-writing-mode\": {\n\t\ttype: \"array\",\n\t\tvalue: \"enum\",\n\t\tvalues: {\n\t\t\thorizontal: {\n\t\t\t},\n\t\t\tvertical: {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"point\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-rotate\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\tunits: \"degrees\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-padding\": {\n\t\ttype: \"number\",\n\t\t\"default\": 2,\n\t\tminimum: 0,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-keep-upright\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": true,\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"text-rotation-alignment\": \"map\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"symbol-placement\": [\n\t\t\t\t\t\"line\",\n\t\t\t\t\t\"line-center\"\n\t\t\t\t]\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-transform\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnone: {\n\t\t\t},\n\t\t\tuppercase: {\n\t\t\t},\n\t\t\tlowercase: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"none\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-offset\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tunits: \"ems\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"!\": \"text-radial-offset\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-allow-overlap\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t{\n\t\t\t\t\"!\": \"text-overlap\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-overlap\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tnever: {\n\t\t\t},\n\t\t\talways: {\n\t\t\t},\n\t\t\tcooperative: {\n\t\t\t}\n\t\t},\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-ignore-placement\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-optional\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": false,\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_raster = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar layout_hillshade = {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n};\nvar filter = {\n\ttype: \"array\",\n\tvalue: \"*\"\n};\nvar filter_operator = {\n\ttype: \"enum\",\n\tvalues: {\n\t\t\"==\": {\n\t\t},\n\t\t\"!=\": {\n\t\t},\n\t\t\">\": {\n\t\t},\n\t\t\">=\": {\n\t\t},\n\t\t\"<\": {\n\t\t},\n\t\t\"<=\": {\n\t\t},\n\t\t\"in\": {\n\t\t},\n\t\t\"!in\": {\n\t\t},\n\t\tall: {\n\t\t},\n\t\tany: {\n\t\t},\n\t\tnone: {\n\t\t},\n\t\thas: {\n\t\t},\n\t\t\"!has\": {\n\t\t},\n\t\twithin: {\n\t\t}\n\t}\n};\nvar geometry_type = {\n\ttype: \"enum\",\n\tvalues: {\n\t\tPoint: {\n\t\t},\n\t\tLineString: {\n\t\t},\n\t\tPolygon: {\n\t\t}\n\t}\n};\nvar function_stop = {\n\ttype: \"array\",\n\tminimum: 0,\n\tmaximum: 24,\n\tvalue: [\n\t\t\"number\",\n\t\t\"color\"\n\t],\n\tlength: 2\n};\nvar expression$1 = {\n\ttype: \"array\",\n\tvalue: \"*\",\n\tminimum: 1\n};\nvar light = {\n\tanchor: {\n\t\ttype: \"enum\",\n\t\t\"default\": \"viewport\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"property-type\": \"data-constant\",\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t}\n\t},\n\tposition: {\n\t\ttype: \"array\",\n\t\t\"default\": [\n\t\t\t1.15,\n\t\t\t210,\n\t\t\t30\n\t\t],\n\t\tlength: 3,\n\t\tvalue: \"number\",\n\t\t\"property-type\": \"data-constant\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t}\n\t},\n\tcolor: {\n\t\ttype: \"color\",\n\t\t\"property-type\": \"data-constant\",\n\t\t\"default\": \"#ffffff\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\ttransition: true\n\t},\n\tintensity: {\n\t\ttype: \"number\",\n\t\t\"property-type\": \"data-constant\",\n\t\t\"default\": 0.5,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\ttransition: true\n\t}\n};\nvar terrain = {\n\tsource: {\n\t\ttype: \"string\",\n\t\trequired: true\n\t},\n\texaggeration: {\n\t\ttype: \"number\",\n\t\tminimum: 0,\n\t\t\"default\": 1\n\t}\n};\nvar paint = [\n\t\"paint_fill\",\n\t\"paint_line\",\n\t\"paint_circle\",\n\t\"paint_heatmap\",\n\t\"paint_fill-extrusion\",\n\t\"paint_symbol\",\n\t\"paint_raster\",\n\t\"paint_hillshade\",\n\t\"paint_background\"\n];\nvar paint_fill = {\n\t\"fill-antialias\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"fill-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-outline-color\": {\n\t\ttype: \"color\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"fill-pattern\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"fill-antialias\": true\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"fill-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded-data-driven\"\n\t}\n};\nvar paint_line = {\n\t\"line-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"line-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"line-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"line-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-gap-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-offset\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"line-dasharray\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"line widths\",\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"line-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded\"\n\t},\n\t\"line-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded-data-driven\"\n\t},\n\t\"line-gradient\": {\n\t\ttype: \"color\",\n\t\ttransition: false,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"line-dasharray\"\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"!\": \"line-pattern\"\n\t\t\t},\n\t\t\t{\n\t\t\t\tsource: \"geojson\",\n\t\t\t\thas: {\n\t\t\t\t\tlineMetrics: true\n\t\t\t\t}\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"line-progress\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"color-ramp\"\n\t}\n};\nvar paint_circle = {\n\t\"circle-radius\": {\n\t\ttype: \"number\",\n\t\t\"default\": 5,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"circle-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-pitch-scale\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-pitch-alignment\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"viewport\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"circle-stroke-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-stroke-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"circle-stroke-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t}\n};\nvar paint_heatmap = {\n\t\"heatmap-radius\": {\n\t\ttype: \"number\",\n\t\t\"default\": 30,\n\t\tminimum: 1,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"heatmap-weight\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"heatmap-intensity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"heatmap-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": [\n\t\t\t\"interpolate\",\n\t\t\t[\n\t\t\t\t\"linear\"\n\t\t\t],\n\t\t\t[\n\t\t\t\t\"heatmap-density\"\n\t\t\t],\n\t\t\t0,\n\t\t\t\"rgba(0, 0, 255, 0)\",\n\t\t\t0.1,\n\t\t\t\"royalblue\",\n\t\t\t0.3,\n\t\t\t\"cyan\",\n\t\t\t0.5,\n\t\t\t\"lime\",\n\t\t\t0.7,\n\t\t\t\"yellow\",\n\t\t\t1,\n\t\t\t\"red\"\n\t\t],\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"heatmap-density\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"color-ramp\"\n\t},\n\t\"heatmap-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_symbol = {\n\t\"icon-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-halo-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"rgba(0, 0, 0, 0)\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-halo-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-halo-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"icon-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"icon-image\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"icon-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"icon-image\",\n\t\t\t\"icon-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\toverridable: true,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-halo-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"rgba(0, 0, 0, 0)\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-halo-width\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-halo-blur\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"text-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\trequires: [\n\t\t\t\"text-field\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"text-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"text-field\",\n\t\t\t\"text-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_raster = {\n\t\"raster-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-hue-rotate\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tperiod: 360,\n\t\ttransition: true,\n\t\tunits: \"degrees\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-brightness-min\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-brightness-max\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-saturation\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: -1,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-contrast\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: -1,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-resampling\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tlinear: {\n\t\t\t},\n\t\t\tnearest: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"linear\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"raster-fade-duration\": {\n\t\ttype: \"number\",\n\t\t\"default\": 300,\n\t\tminimum: 0,\n\t\ttransition: false,\n\t\tunits: \"milliseconds\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_hillshade = {\n\t\"hillshade-illumination-direction\": {\n\t\ttype: \"number\",\n\t\t\"default\": 335,\n\t\tminimum: 0,\n\t\tmaximum: 359,\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-illumination-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"viewport\",\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-exaggeration\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0.5,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-shadow-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-highlight-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#FFFFFF\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"hillshade-accent-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar paint_background = {\n\t\"background-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"background-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"background-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded\"\n\t},\n\t\"background-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n};\nvar transition = {\n\tduration: {\n\t\ttype: \"number\",\n\t\t\"default\": 300,\n\t\tminimum: 0,\n\t\tunits: \"milliseconds\"\n\t},\n\tdelay: {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tunits: \"milliseconds\"\n\t}\n};\nvar promoteId = {\n\t\"*\": {\n\t\ttype: \"string\"\n\t}\n};\nvar v8Spec = {\n\t$version: $version,\n\t$root: $root,\n\tsources: sources,\n\tsource: source,\n\tsource_vector: source_vector,\n\tsource_raster: source_raster,\n\tsource_raster_dem: source_raster_dem,\n\tsource_geojson: source_geojson,\n\tsource_video: source_video,\n\tsource_image: source_image,\n\tlayer: layer,\n\tlayout: layout,\n\tlayout_background: layout_background,\n\tlayout_fill: layout_fill,\n\tlayout_circle: layout_circle,\n\tlayout_heatmap: layout_heatmap,\n\t\"layout_fill-extrusion\": {\n\tvisibility: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tvisible: {\n\t\t\t},\n\t\t\tnone: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"visible\",\n\t\t\"property-type\": \"constant\"\n\t}\n},\n\tlayout_line: layout_line,\n\tlayout_symbol: layout_symbol,\n\tlayout_raster: layout_raster,\n\tlayout_hillshade: layout_hillshade,\n\tfilter: filter,\n\tfilter_operator: filter_operator,\n\tgeometry_type: geometry_type,\n\t\"function\": {\n\texpression: {\n\t\ttype: \"expression\"\n\t},\n\tstops: {\n\t\ttype: \"array\",\n\t\tvalue: \"function_stop\"\n\t},\n\tbase: {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0\n\t},\n\tproperty: {\n\t\ttype: \"string\",\n\t\t\"default\": \"$zoom\"\n\t},\n\ttype: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tidentity: {\n\t\t\t},\n\t\t\texponential: {\n\t\t\t},\n\t\t\tinterval: {\n\t\t\t},\n\t\t\tcategorical: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"exponential\"\n\t},\n\tcolorSpace: {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\trgb: {\n\t\t\t},\n\t\t\tlab: {\n\t\t\t},\n\t\t\thcl: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"rgb\"\n\t},\n\t\"default\": {\n\t\ttype: \"*\",\n\t\trequired: false\n\t}\n},\n\tfunction_stop: function_stop,\n\texpression: expression$1,\n\tlight: light,\n\tterrain: terrain,\n\tpaint: paint,\n\tpaint_fill: paint_fill,\n\t\"paint_fill-extrusion\": {\n\t\"fill-extrusion-opacity\": {\n\t\ttype: \"number\",\n\t\t\"default\": 1,\n\t\tminimum: 0,\n\t\tmaximum: 1,\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-extrusion-color\": {\n\t\ttype: \"color\",\n\t\t\"default\": \"#000000\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t{\n\t\t\t\t\"!\": \"fill-extrusion-pattern\"\n\t\t\t}\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-extrusion-translate\": {\n\t\ttype: \"array\",\n\t\tvalue: \"number\",\n\t\tlength: 2,\n\t\t\"default\": [\n\t\t\t0,\n\t\t\t0\n\t\t],\n\t\ttransition: true,\n\t\tunits: \"pixels\",\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-extrusion-translate-anchor\": {\n\t\ttype: \"enum\",\n\t\tvalues: {\n\t\t\tmap: {\n\t\t\t},\n\t\t\tviewport: {\n\t\t\t}\n\t\t},\n\t\t\"default\": \"map\",\n\t\trequires: [\n\t\t\t\"fill-extrusion-translate\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t},\n\t\"fill-extrusion-pattern\": {\n\t\ttype: \"resolvedImage\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"cross-faded-data-driven\"\n\t},\n\t\"fill-extrusion-height\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tunits: \"meters\",\n\t\ttransition: true,\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-extrusion-base\": {\n\t\ttype: \"number\",\n\t\t\"default\": 0,\n\t\tminimum: 0,\n\t\tunits: \"meters\",\n\t\ttransition: true,\n\t\trequires: [\n\t\t\t\"fill-extrusion-height\"\n\t\t],\n\t\texpression: {\n\t\t\tinterpolated: true,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\",\n\t\t\t\t\"feature\",\n\t\t\t\t\"feature-state\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-driven\"\n\t},\n\t\"fill-extrusion-vertical-gradient\": {\n\t\ttype: \"boolean\",\n\t\t\"default\": true,\n\t\ttransition: false,\n\t\texpression: {\n\t\t\tinterpolated: false,\n\t\t\tparameters: [\n\t\t\t\t\"zoom\"\n\t\t\t]\n\t\t},\n\t\t\"property-type\": \"data-constant\"\n\t}\n},\n\tpaint_line: paint_line,\n\tpaint_circle: paint_circle,\n\tpaint_heatmap: paint_heatmap,\n\tpaint_symbol: paint_symbol,\n\tpaint_raster: paint_raster,\n\tpaint_hillshade: paint_hillshade,\n\tpaint_background: paint_background,\n\ttransition: transition,\n\t\"property-type\": {\n\t\"data-driven\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"cross-faded\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"cross-faded-data-driven\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"color-ramp\": {\n\t\ttype: \"property-type\"\n\t},\n\t\"data-constant\": {\n\t\ttype: \"property-type\"\n\t},\n\tconstant: {\n\t\ttype: \"property-type\"\n\t}\n},\n\tpromoteId: promoteId\n};\n\nconst refProperties = ['type', 'source', 'source-layer', 'minzoom', 'maxzoom', 'filter', 'layout'];\n\nfunction deref(layer, parent) {\n const result = {};\n for (const k in layer) {\n if (k !== 'ref') {\n result[k] = layer[k];\n }\n }\n refProperties.forEach((k) => {\n if (k in parent) {\n result[k] = parent[k];\n }\n });\n return result;\n}\n/**\n * Given an array of layers, some of which may contain `ref` properties\n * whose value is the `id` of another property, return a new array where\n * such layers have been augmented with the 'type', 'source', etc. properties\n * from the parent layer, and the `ref` property has been removed.\n *\n * The input is not modified. The output may contain references to portions\n * of the input.\n *\n * @private\n * @param {Array} layers\n * @returns {Array}\n */\nfunction derefLayers(layers) {\n layers = layers.slice();\n const map = Object.create(null);\n for (let i = 0; i < layers.length; i++) {\n map[layers[i].id] = layers[i];\n }\n for (let i = 0; i < layers.length; i++) {\n if ('ref' in layers[i]) {\n layers[i] = deref(layers[i], map[layers[i].ref]);\n }\n }\n return layers;\n}\n\n/**\n * Deeply compares two object literals.\n *\n * @private\n */\nfunction deepEqual(a, b) {\n if (Array.isArray(a)) {\n if (!Array.isArray(b) || a.length !== b.length)\n return false;\n for (let i = 0; i < a.length; i++) {\n if (!deepEqual(a[i], b[i]))\n return false;\n }\n return true;\n }\n if (typeof a === 'object' && a !== null && b !== null) {\n if (!(typeof b === 'object'))\n return false;\n const keys = Object.keys(a);\n if (keys.length !== Object.keys(b).length)\n return false;\n for (const key in a) {\n if (!deepEqual(a[key], b[key]))\n return false;\n }\n return true;\n }\n return a === b;\n}\n\nconst operations = {\n /*\n * { command: 'setStyle', args: [stylesheet] }\n */\n setStyle: 'setStyle',\n /*\n * { command: 'addLayer', args: [layer, 'beforeLayerId'] }\n */\n addLayer: 'addLayer',\n /*\n * { command: 'removeLayer', args: ['layerId'] }\n */\n removeLayer: 'removeLayer',\n /*\n * { command: 'setPaintProperty', args: ['layerId', 'prop', value] }\n */\n setPaintProperty: 'setPaintProperty',\n /*\n * { command: 'setLayoutProperty', args: ['layerId', 'prop', value] }\n */\n setLayoutProperty: 'setLayoutProperty',\n /*\n * { command: 'setFilter', args: ['layerId', filter] }\n */\n setFilter: 'setFilter',\n /*\n * { command: 'addSource', args: ['sourceId', source] }\n */\n addSource: 'addSource',\n /*\n * { command: 'removeSource', args: ['sourceId'] }\n */\n removeSource: 'removeSource',\n /*\n * { command: 'setGeoJSONSourceData', args: ['sourceId', data] }\n */\n setGeoJSONSourceData: 'setGeoJSONSourceData',\n /*\n * { command: 'setLayerZoomRange', args: ['layerId', 0, 22] }\n */\n setLayerZoomRange: 'setLayerZoomRange',\n /*\n * { command: 'setLayerProperty', args: ['layerId', 'prop', value] }\n */\n setLayerProperty: 'setLayerProperty',\n /*\n * { command: 'setCenter', args: [[lon, lat]] }\n */\n setCenter: 'setCenter',\n /*\n * { command: 'setZoom', args: [zoom] }\n */\n setZoom: 'setZoom',\n /*\n * { command: 'setBearing', args: [bearing] }\n */\n setBearing: 'setBearing',\n /*\n * { command: 'setPitch', args: [pitch] }\n */\n setPitch: 'setPitch',\n /*\n * { command: 'setSprite', args: ['spriteUrl'] }\n */\n setSprite: 'setSprite',\n /*\n * { command: 'setGlyphs', args: ['glyphsUrl'] }\n */\n setGlyphs: 'setGlyphs',\n /*\n * { command: 'setTransition', args: [transition] }\n */\n setTransition: 'setTransition',\n /*\n * { command: 'setLighting', args: [lightProperties] }\n */\n setLight: 'setLight'\n};\nfunction addSource(sourceId, after, commands) {\n commands.push({ command: operations.addSource, args: [sourceId, after[sourceId]] });\n}\nfunction removeSource(sourceId, commands, sourcesRemoved) {\n commands.push({ command: operations.removeSource, args: [sourceId] });\n sourcesRemoved[sourceId] = true;\n}\nfunction updateSource(sourceId, after, commands, sourcesRemoved) {\n removeSource(sourceId, commands, sourcesRemoved);\n addSource(sourceId, after, commands);\n}\nfunction canUpdateGeoJSON(before, after, sourceId) {\n let prop;\n for (prop in before[sourceId]) {\n if (!Object.prototype.hasOwnProperty.call(before[sourceId], prop))\n continue;\n if (prop !== 'data' && !deepEqual(before[sourceId][prop], after[sourceId][prop])) {\n return false;\n }\n }\n for (prop in after[sourceId]) {\n if (!Object.prototype.hasOwnProperty.call(after[sourceId], prop))\n continue;\n if (prop !== 'data' && !deepEqual(before[sourceId][prop], after[sourceId][prop])) {\n return false;\n }\n }\n return true;\n}\nfunction diffSources(before, after, commands, sourcesRemoved) {\n before = before || {};\n after = after || {};\n let sourceId;\n // look for sources to remove\n for (sourceId in before) {\n if (!Object.prototype.hasOwnProperty.call(before, sourceId))\n continue;\n if (!Object.prototype.hasOwnProperty.call(after, sourceId)) {\n removeSource(sourceId, commands, sourcesRemoved);\n }\n }\n // look for sources to add/update\n for (sourceId in after) {\n if (!Object.prototype.hasOwnProperty.call(after, sourceId))\n continue;\n if (!Object.prototype.hasOwnProperty.call(before, sourceId)) {\n addSource(sourceId, after, commands);\n }\n else if (!deepEqual(before[sourceId], after[sourceId])) {\n if (before[sourceId].type === 'geojson' && after[sourceId].type === 'geojson' && canUpdateGeoJSON(before, after, sourceId)) {\n commands.push({ command: operations.setGeoJSONSourceData, args: [sourceId, after[sourceId].data] });\n }\n else {\n // no update command, must remove then add\n updateSource(sourceId, after, commands, sourcesRemoved);\n }\n }\n }\n}\nfunction diffLayerPropertyChanges(before, after, commands, layerId, klass, command) {\n before = before || {};\n after = after || {};\n let prop;\n for (prop in before) {\n if (!Object.prototype.hasOwnProperty.call(before, prop))\n continue;\n if (!deepEqual(before[prop], after[prop])) {\n commands.push({ command, args: [layerId, prop, after[prop], klass] });\n }\n }\n for (prop in after) {\n if (!Object.prototype.hasOwnProperty.call(after, prop) || Object.prototype.hasOwnProperty.call(before, prop))\n continue;\n if (!deepEqual(before[prop], after[prop])) {\n commands.push({ command, args: [layerId, prop, after[prop], klass] });\n }\n }\n}\nfunction pluckId(layer) {\n return layer.id;\n}\nfunction indexById(group, layer) {\n group[layer.id] = layer;\n return group;\n}\nfunction diffLayers(before, after, commands) {\n before = before || [];\n after = after || [];\n // order of layers by id\n const beforeOrder = before.map(pluckId);\n const afterOrder = after.map(pluckId);\n // index of layer by id\n const beforeIndex = before.reduce(indexById, {});\n const afterIndex = after.reduce(indexById, {});\n // track order of layers as if they have been mutated\n const tracker = beforeOrder.slice();\n // layers that have been added do not need to be diffed\n const clean = Object.create(null);\n let i, d, layerId, beforeLayer, afterLayer, insertBeforeLayerId, prop;\n // remove layers\n for (i = 0, d = 0; i < beforeOrder.length; i++) {\n layerId = beforeOrder[i];\n if (!Object.prototype.hasOwnProperty.call(afterIndex, layerId)) {\n commands.push({ command: operations.removeLayer, args: [layerId] });\n tracker.splice(tracker.indexOf(layerId, d), 1);\n }\n else {\n // limit where in tracker we need to look for a match\n d++;\n }\n }\n // add/reorder layers\n for (i = 0, d = 0; i < afterOrder.length; i++) {\n // work backwards as insert is before an existing layer\n layerId = afterOrder[afterOrder.length - 1 - i];\n if (tracker[tracker.length - 1 - i] === layerId)\n continue;\n if (Object.prototype.hasOwnProperty.call(beforeIndex, layerId)) {\n // remove the layer before we insert at the correct position\n commands.push({ command: operations.removeLayer, args: [layerId] });\n tracker.splice(tracker.lastIndexOf(layerId, tracker.length - d), 1);\n }\n else {\n // limit where in tracker we need to look for a match\n d++;\n }\n // add layer at correct position\n insertBeforeLayerId = tracker[tracker.length - i];\n commands.push({ command: operations.addLayer, args: [afterIndex[layerId], insertBeforeLayerId] });\n tracker.splice(tracker.length - i, 0, layerId);\n clean[layerId] = true;\n }\n // update layers\n for (i = 0; i < afterOrder.length; i++) {\n layerId = afterOrder[i];\n beforeLayer = beforeIndex[layerId];\n afterLayer = afterIndex[layerId];\n // no need to update if previously added (new or moved)\n if (clean[layerId] || deepEqual(beforeLayer, afterLayer))\n continue;\n // If source, source-layer, or type have changes, then remove the layer\n // and add it back 'from scratch'.\n if (!deepEqual(beforeLayer.source, afterLayer.source) || !deepEqual(beforeLayer['source-layer'], afterLayer['source-layer']) || !deepEqual(beforeLayer.type, afterLayer.type)) {\n commands.push({ command: operations.removeLayer, args: [layerId] });\n // we add the layer back at the same position it was already in, so\n // there's no need to update the `tracker`\n insertBeforeLayerId = tracker[tracker.lastIndexOf(layerId) + 1];\n commands.push({ command: operations.addLayer, args: [afterLayer, insertBeforeLayerId] });\n continue;\n }\n // layout, paint, filter, minzoom, maxzoom\n diffLayerPropertyChanges(beforeLayer.layout, afterLayer.layout, commands, layerId, null, operations.setLayoutProperty);\n diffLayerPropertyChanges(beforeLayer.paint, afterLayer.paint, commands, layerId, null, operations.setPaintProperty);\n if (!deepEqual(beforeLayer.filter, afterLayer.filter)) {\n commands.push({ command: operations.setFilter, args: [layerId, afterLayer.filter] });\n }\n if (!deepEqual(beforeLayer.minzoom, afterLayer.minzoom) || !deepEqual(beforeLayer.maxzoom, afterLayer.maxzoom)) {\n commands.push({ command: operations.setLayerZoomRange, args: [layerId, afterLayer.minzoom, afterLayer.maxzoom] });\n }\n // handle all other layer props, including paint.*\n for (prop in beforeLayer) {\n if (!Object.prototype.hasOwnProperty.call(beforeLayer, prop))\n continue;\n if (prop === 'layout' || prop === 'paint' || prop === 'filter' ||\n prop === 'metadata' || prop === 'minzoom' || prop === 'maxzoom')\n continue;\n if (prop.indexOf('paint.') === 0) {\n diffLayerPropertyChanges(beforeLayer[prop], afterLayer[prop], commands, layerId, prop.slice(6), operations.setPaintProperty);\n }\n else if (!deepEqual(beforeLayer[prop], afterLayer[prop])) {\n commands.push({ command: operations.setLayerProperty, args: [layerId, prop, afterLayer[prop]] });\n }\n }\n for (prop in afterLayer) {\n if (!Object.prototype.hasOwnProperty.call(afterLayer, prop) || Object.prototype.hasOwnProperty.call(beforeLayer, prop))\n continue;\n if (prop === 'layout' || prop === 'paint' || prop === 'filter' ||\n prop === 'metadata' || prop === 'minzoom' || prop === 'maxzoom')\n continue;\n if (prop.indexOf('paint.') === 0) {\n diffLayerPropertyChanges(beforeLayer[prop], afterLayer[prop], commands, layerId, prop.slice(6), operations.setPaintProperty);\n }\n else if (!deepEqual(beforeLayer[prop], afterLayer[prop])) {\n commands.push({ command: operations.setLayerProperty, args: [layerId, prop, afterLayer[prop]] });\n }\n }\n }\n}\n/**\n * Diff two stylesheet\n *\n * Creates semanticly aware diffs that can easily be applied at runtime.\n * Operations produced by the diff closely resemble the maplibre-gl-js API. Any\n * error creating the diff will fall back to the 'setStyle' operation.\n *\n * Example diff:\n * [\n * { command: 'setConstant', args: ['@water', '#0000FF'] },\n * { command: 'setPaintProperty', args: ['background', 'background-color', 'black'] }\n * ]\n *\n * @private\n * @param {*} [before] stylesheet to compare from\n * @param {*} after stylesheet to compare to\n * @returns Array list of changes\n */\nfunction diffStyles(before, after) {\n if (!before)\n return [{ command: operations.setStyle, args: [after] }];\n let commands = [];\n try {\n // Handle changes to top-level properties\n if (!deepEqual(before.version, after.version)) {\n return [{ command: operations.setStyle, args: [after] }];\n }\n if (!deepEqual(before.center, after.center)) {\n commands.push({ command: operations.setCenter, args: [after.center] });\n }\n if (!deepEqual(before.zoom, after.zoom)) {\n commands.push({ command: operations.setZoom, args: [after.zoom] });\n }\n if (!deepEqual(before.bearing, after.bearing)) {\n commands.push({ command: operations.setBearing, args: [after.bearing] });\n }\n if (!deepEqual(before.pitch, after.pitch)) {\n commands.push({ command: operations.setPitch, args: [after.pitch] });\n }\n if (!deepEqual(before.sprite, after.sprite)) {\n commands.push({ command: operations.setSprite, args: [after.sprite] });\n }\n if (!deepEqual(before.glyphs, after.glyphs)) {\n commands.push({ command: operations.setGlyphs, args: [after.glyphs] });\n }\n if (!deepEqual(before.transition, after.transition)) {\n commands.push({ command: operations.setTransition, args: [after.transition] });\n }\n if (!deepEqual(before.light, after.light)) {\n commands.push({ command: operations.setLight, args: [after.light] });\n }\n // Handle changes to `sources`\n // If a source is to be removed, we also--before the removeSource\n // command--need to remove all the style layers that depend on it.\n const sourcesRemoved = {};\n // First collect the {add,remove}Source commands\n const removeOrAddSourceCommands = [];\n diffSources(before.sources, after.sources, removeOrAddSourceCommands, sourcesRemoved);\n // Push a removeLayer command for each style layer that depends on a\n // source that's being removed.\n // Also, exclude any such layers them from the input to `diffLayers`\n // below, so that diffLayers produces the appropriate `addLayers`\n // command\n const beforeLayers = [];\n if (before.layers) {\n before.layers.forEach((layer) => {\n if (sourcesRemoved[layer.source]) {\n commands.push({ command: operations.removeLayer, args: [layer.id] });\n }\n else {\n beforeLayers.push(layer);\n }\n });\n }\n commands = commands.concat(removeOrAddSourceCommands);\n // Handle changes to `layers`\n diffLayers(beforeLayers, after.layers, commands);\n }\n catch (e) {\n // fall back to setStyle\n console.warn('Unable to compute style diff:', e);\n commands = [{ command: operations.setStyle, args: [after] }];\n }\n return commands;\n}\n\n// Note: Do not inherit from Error. It breaks when transpiling to ES5.\nclass ValidationError {\n constructor(key, value, message, identifier) {\n this.message = (key ? `${key}: ` : '') + message;\n if (identifier)\n this.identifier = identifier;\n if (value !== null && value !== undefined && value.__line__) {\n this.line = value.__line__;\n }\n }\n}\n\n// Note: Do not inherit from Error. It breaks when transpiling to ES5.\nclass ParsingError {\n constructor(error) {\n this.error = error;\n this.message = error.message;\n const match = error.message.match(/line (\\d+)/);\n this.line = match ? parseInt(match[1], 10) : 0;\n }\n}\n\nfunction extendBy(output, ...inputs) {\n for (const input of inputs) {\n for (const k in input) {\n output[k] = input[k];\n }\n }\n return output;\n}\n\nclass ExpressionParsingError extends Error {\n constructor(key, message) {\n super(message);\n this.message = message;\n this.key = key;\n }\n}\n\n/**\n * Tracks `let` bindings during expression parsing.\n * @private\n */\nclass Scope {\n constructor(parent, bindings = []) {\n this.parent = parent;\n this.bindings = {};\n for (const [name, expression] of bindings) {\n this.bindings[name] = expression;\n }\n }\n concat(bindings) {\n return new Scope(this, bindings);\n }\n get(name) {\n if (this.bindings[name]) {\n return this.bindings[name];\n }\n if (this.parent) {\n return this.parent.get(name);\n }\n throw new Error(`${name} not found in scope.`);\n }\n has(name) {\n if (this.bindings[name])\n return true;\n return this.parent ? this.parent.has(name) : false;\n }\n}\n\nconst NullType = { kind: 'null' };\nconst NumberType = { kind: 'number' };\nconst StringType = { kind: 'string' };\nconst BooleanType = { kind: 'boolean' };\nconst ColorType = { kind: 'color' };\nconst ObjectType = { kind: 'object' };\nconst ValueType = { kind: 'value' };\nconst ErrorType = { kind: 'error' };\nconst CollatorType = { kind: 'collator' };\nconst FormattedType = { kind: 'formatted' };\nconst PaddingType = { kind: 'padding' };\nconst ResolvedImageType = { kind: 'resolvedImage' };\nconst VariableAnchorOffsetCollectionType = { kind: 'variableAnchorOffsetCollection' };\nfunction array$1(itemType, N) {\n return {\n kind: 'array',\n itemType,\n N\n };\n}\nfunction toString$1(type) {\n if (type.kind === 'array') {\n const itemType = toString$1(type.itemType);\n return typeof type.N === 'number' ?\n `array<${itemType}, ${type.N}>` :\n type.itemType.kind === 'value' ? 'array' : `array<${itemType}>`;\n }\n else {\n return type.kind;\n }\n}\nconst valueMemberTypes = [\n NullType,\n NumberType,\n StringType,\n BooleanType,\n ColorType,\n FormattedType,\n ObjectType,\n array$1(ValueType),\n PaddingType,\n ResolvedImageType,\n VariableAnchorOffsetCollectionType\n];\n/**\n * Returns null if `t` is a subtype of `expected`; otherwise returns an\n * error message.\n * @private\n */\nfunction checkSubtype(expected, t) {\n if (t.kind === 'error') {\n // Error is a subtype of every type\n return null;\n }\n else if (expected.kind === 'array') {\n if (t.kind === 'array' &&\n ((t.N === 0 && t.itemType.kind === 'value') || !checkSubtype(expected.itemType, t.itemType)) &&\n (typeof expected.N !== 'number' || expected.N === t.N)) {\n return null;\n }\n }\n else if (expected.kind === t.kind) {\n return null;\n }\n else if (expected.kind === 'value') {\n for (const memberType of valueMemberTypes) {\n if (!checkSubtype(memberType, t)) {\n return null;\n }\n }\n }\n return `Expected ${toString$1(expected)} but found ${toString$1(t)} instead.`;\n}\nfunction isValidType(provided, allowedTypes) {\n return allowedTypes.some(t => t.kind === provided.kind);\n}\nfunction isValidNativeType(provided, allowedTypes) {\n return allowedTypes.some(t => {\n if (t === 'null') {\n return provided === null;\n }\n else if (t === 'array') {\n return Array.isArray(provided);\n }\n else if (t === 'object') {\n return provided && !Array.isArray(provided) && typeof provided === 'object';\n }\n else {\n return t === typeof provided;\n }\n });\n}\n/**\n * Verify whether the specified type is of the same type as the specified sample.\n *\n * @param provided Type to verify\n * @param sample Sample type to reference\n * @returns `true` if both objects are of the same type, `false` otherwise\n * @example basic types\n * if (verifyType(outputType, ValueType)) {\n * // type narrowed to:\n * outputType.kind; // 'value'\n * }\n * @example array types\n * if (verifyType(outputType, array(NumberType))) {\n * // type narrowed to:\n * outputType.kind; // 'array'\n * outputType.itemType; // NumberTypeT\n * outputType.itemType.kind; // 'number'\n * }\n */\nfunction verifyType(provided, sample) {\n if (provided.kind === 'array' && sample.kind === 'array') {\n return provided.itemType.kind === sample.itemType.kind && typeof provided.N === 'number';\n }\n return provided.kind === sample.kind;\n}\n\n// See https://observablehq.com/@mbostock/lab-and-rgb\nconst Xn = 0.96422, Yn = 1, Zn = 0.82521, t0 = 4 / 29, t1 = 6 / 29, t2 = 3 * t1 * t1, t3 = t1 * t1 * t1, deg2rad = Math.PI / 180, rad2deg = 180 / Math.PI;\nfunction constrainAngle(angle) {\n angle = angle % 360;\n if (angle < 0) {\n angle += 360;\n }\n return angle;\n}\nfunction rgbToLab([r, g, b, alpha]) {\n r = rgb2xyz(r);\n g = rgb2xyz(g);\n b = rgb2xyz(b);\n let x, z;\n const y = xyz2lab((0.2225045 * r + 0.7168786 * g + 0.0606169 * b) / Yn);\n if (r === g && g === b) {\n x = z = y;\n }\n else {\n x = xyz2lab((0.4360747 * r + 0.3850649 * g + 0.1430804 * b) / Xn);\n z = xyz2lab((0.0139322 * r + 0.0971045 * g + 0.7141733 * b) / Zn);\n }\n const l = 116 * y - 16;\n return [(l < 0) ? 0 : l, 500 * (x - y), 200 * (y - z), alpha];\n}\nfunction rgb2xyz(x) {\n return (x <= 0.04045) ? x / 12.92 : Math.pow((x + 0.055) / 1.055, 2.4);\n}\nfunction xyz2lab(t) {\n return (t > t3) ? Math.pow(t, 1 / 3) : t / t2 + t0;\n}\nfunction labToRgb([l, a, b, alpha]) {\n let y = (l + 16) / 116, x = isNaN(a) ? y : y + a / 500, z = isNaN(b) ? y : y - b / 200;\n y = Yn * lab2xyz(y);\n x = Xn * lab2xyz(x);\n z = Zn * lab2xyz(z);\n return [\n xyz2rgb(3.1338561 * x - 1.6168667 * y - 0.4906146 * z),\n xyz2rgb(-0.9787684 * x + 1.9161415 * y + 0.0334540 * z),\n xyz2rgb(0.0719453 * x - 0.2289914 * y + 1.4052427 * z),\n alpha,\n ];\n}\nfunction xyz2rgb(x) {\n x = (x <= 0.00304) ? 12.92 * x : 1.055 * Math.pow(x, 1 / 2.4) - 0.055;\n return (x < 0) ? 0 : (x > 1) ? 1 : x; // clip to 0..1 range\n}\nfunction lab2xyz(t) {\n return (t > t1) ? t * t * t : t2 * (t - t0);\n}\nfunction rgbToHcl(rgbColor) {\n const [l, a, b, alpha] = rgbToLab(rgbColor);\n const c = Math.sqrt(a * a + b * b);\n const h = Math.round(c * 10000) ? constrainAngle(Math.atan2(b, a) * rad2deg) : NaN;\n return [h, c, l, alpha];\n}\nfunction hclToRgb([h, c, l, alpha]) {\n h = isNaN(h) ? 0 : h * deg2rad;\n return labToRgb([l, Math.cos(h) * c, Math.sin(h) * c, alpha]);\n}\n// https://drafts.csswg.org/css-color-4/#hsl-to-rgb\nfunction hslToRgb([h, s, l, alpha]) {\n h = constrainAngle(h);\n s /= 100;\n l /= 100;\n function f(n) {\n const k = (n + h / 30) % 12;\n const a = s * Math.min(l, 1 - l);\n return l - a * Math.max(-1, Math.min(k - 3, 9 - k, 1));\n }\n return [f(0), f(8), f(4), alpha];\n}\n\n/**\n * CSS color parser compliant with CSS Color 4 Specification.\n * Supports: named colors, `transparent` keyword, all rgb hex notations,\n * rgb(), rgba(), hsl() and hsla() functions.\n * Does not round the parsed values to integers from the range 0..255.\n *\n * Syntax:\n *\n * = | \n * = | \n *\n * rgb() = rgb( {3} [ / ]? ) | rgb( {3} [ / ]? )\n * rgb() = rgb( #{3} , ? ) | rgb( #{3} , ? )\n *\n * hsl() = hsl( [ / ]? )\n * hsl() = hsl( , , , ? )\n *\n * Caveats:\n * - - with optional `deg` suffix; `grad`, `rad`, `turn` are not supported\n * - `none` keyword is not supported\n * - comments inside rgb()/hsl() are not supported\n * - legacy color syntax rgba() is supported with an identical grammar and behavior to rgb()\n * - legacy color syntax hsla() is supported with an identical grammar and behavior to hsl()\n *\n * @param input CSS color string to parse.\n * @returns Color in sRGB color space, with `red`, `green`, `blue`\n * and `alpha` channels normalized to the range 0..1,\n * or `undefined` if the input is not a valid color string.\n */\nfunction parseCssColor(input) {\n input = input.toLowerCase().trim();\n if (input === 'transparent') {\n return [0, 0, 0, 0];\n }\n // 'white', 'black', 'blue'\n const namedColorsMatch = namedColors[input];\n if (namedColorsMatch) {\n const [r, g, b] = namedColorsMatch;\n return [r / 255, g / 255, b / 255, 1];\n }\n // #f0c, #f0cf, #ff00cc, #ff00ccff\n if (input.startsWith('#')) {\n const hexRegexp = /^#(?:[0-9a-f]{3,4}|[0-9a-f]{6}|[0-9a-f]{8})$/;\n if (hexRegexp.test(input)) {\n const step = input.length < 6 ? 1 : 2;\n let i = 1;\n return [\n parseHex(input.slice(i, i += step)),\n parseHex(input.slice(i, i += step)),\n parseHex(input.slice(i, i += step)),\n parseHex(input.slice(i, i + step) || 'ff'),\n ];\n }\n }\n // rgb(128 0 0), rgb(50% 0% 0%), rgba(255,0,255,0.6), rgb(255 0 255 / 60%), rgb(100% 0% 100% /.6)\n if (input.startsWith('rgb')) {\n const rgbRegExp = /^rgba?\\(\\s*([\\de.+-]+)(%)?(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)(%)?(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)(%)?(?:\\s*([,\\/])\\s*([\\de.+-]+)(%)?)?\\s*\\)$/;\n const rgbMatch = input.match(rgbRegExp);\n if (rgbMatch) {\n const [_, // eslint-disable-line @typescript-eslint/no-unused-vars\n r, // \n rp, // % (optional)\n f1, // , (optional)\n g, // \n gp, // % (optional)\n f2, // , (optional)\n b, // \n bp, // % (optional)\n f3, // ,|/ (optional)\n a, // (optional)\n ap, // % (optional)\n ] = rgbMatch;\n const argFormat = [f1 || ' ', f2 || ' ', f3].join('');\n if (argFormat === ' ' ||\n argFormat === ' /' ||\n argFormat === ',,' ||\n argFormat === ',,,') {\n const valFormat = [rp, gp, bp].join('');\n const maxValue = (valFormat === '%%%') ? 100 :\n (valFormat === '') ? 255 : 0;\n if (maxValue) {\n const rgba = [\n clamp(+r / maxValue, 0, 1),\n clamp(+g / maxValue, 0, 1),\n clamp(+b / maxValue, 0, 1),\n a ? parseAlpha(+a, ap) : 1,\n ];\n if (validateNumbers(rgba)) {\n return rgba;\n }\n // invalid numbers\n }\n // values must be all numbers or all percentages\n }\n return; // comma optional syntax requires no commas at all\n }\n }\n // hsl(120 50% 80%), hsla(120deg,50%,80%,.9), hsl(12e1 50% 80% / 90%)\n const hslRegExp = /^hsla?\\(\\s*([\\de.+-]+)(?:deg)?(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)%(?:\\s+|\\s*(,)\\s*)([\\de.+-]+)%(?:\\s*([,\\/])\\s*([\\de.+-]+)(%)?)?\\s*\\)$/;\n const hslMatch = input.match(hslRegExp);\n if (hslMatch) {\n const [_, // eslint-disable-line @typescript-eslint/no-unused-vars\n h, // \n f1, // , (optional)\n s, // \n f2, // , (optional)\n l, // \n f3, // ,|/ (optional)\n a, // (optional)\n ap, // % (optional)\n ] = hslMatch;\n const argFormat = [f1 || ' ', f2 || ' ', f3].join('');\n if (argFormat === ' ' ||\n argFormat === ' /' ||\n argFormat === ',,' ||\n argFormat === ',,,') {\n const hsla = [\n +h,\n clamp(+s, 0, 100),\n clamp(+l, 0, 100),\n a ? parseAlpha(+a, ap) : 1,\n ];\n if (validateNumbers(hsla)) {\n return hslToRgb(hsla);\n }\n // invalid numbers\n }\n // comma optional syntax requires no commas at all\n }\n}\nfunction parseHex(hex) {\n return parseInt(hex.padEnd(2, hex), 16) / 255;\n}\nfunction parseAlpha(a, asPercentage) {\n return clamp(asPercentage ? (a / 100) : a, 0, 1);\n}\nfunction clamp(n, min, max) {\n return Math.min(Math.max(min, n), max);\n}\n/**\n * The regular expression for numeric values is not super specific, and it may\n * happen that it will accept a value that is not a valid number. In order to\n * detect and eliminate such values this function exists.\n *\n * @param array Array of uncertain numbers.\n * @returns `true` if the specified array contains only valid numbers, `false` otherwise.\n */\nfunction validateNumbers(array) {\n return !array.some(Number.isNaN);\n}\n/**\n * To generate:\n * - visit {@link https://www.w3.org/TR/css-color-4/#named-colors}\n * - run in the console:\n * @example\n * copy(`{\\n${[...document.querySelector('.named-color-table tbody').children].map((tr) => `${tr.cells[2].textContent.trim()}: [${tr.cells[4].textContent.trim().split(/\\s+/).join(', ')}],`).join('\\n')}\\n}`);\n */\nconst namedColors = {\n aliceblue: [240, 248, 255],\n antiquewhite: [250, 235, 215],\n aqua: [0, 255, 255],\n aquamarine: [127, 255, 212],\n azure: [240, 255, 255],\n beige: [245, 245, 220],\n bisque: [255, 228, 196],\n black: [0, 0, 0],\n blanchedalmond: [255, 235, 205],\n blue: [0, 0, 255],\n blueviolet: [138, 43, 226],\n brown: [165, 42, 42],\n burlywood: [222, 184, 135],\n cadetblue: [95, 158, 160],\n chartreuse: [127, 255, 0],\n chocolate: [210, 105, 30],\n coral: [255, 127, 80],\n cornflowerblue: [100, 149, 237],\n cornsilk: [255, 248, 220],\n crimson: [220, 20, 60],\n cyan: [0, 255, 255],\n darkblue: [0, 0, 139],\n darkcyan: [0, 139, 139],\n darkgoldenrod: [184, 134, 11],\n darkgray: [169, 169, 169],\n darkgreen: [0, 100, 0],\n darkgrey: [169, 169, 169],\n darkkhaki: [189, 183, 107],\n darkmagenta: [139, 0, 139],\n darkolivegreen: [85, 107, 47],\n darkorange: [255, 140, 0],\n darkorchid: [153, 50, 204],\n darkred: [139, 0, 0],\n darksalmon: [233, 150, 122],\n darkseagreen: [143, 188, 143],\n darkslateblue: [72, 61, 139],\n darkslategray: [47, 79, 79],\n darkslategrey: [47, 79, 79],\n darkturquoise: [0, 206, 209],\n darkviolet: [148, 0, 211],\n deeppink: [255, 20, 147],\n deepskyblue: [0, 191, 255],\n dimgray: [105, 105, 105],\n dimgrey: [105, 105, 105],\n dodgerblue: [30, 144, 255],\n firebrick: [178, 34, 34],\n floralwhite: [255, 250, 240],\n forestgreen: [34, 139, 34],\n fuchsia: [255, 0, 255],\n gainsboro: [220, 220, 220],\n ghostwhite: [248, 248, 255],\n gold: [255, 215, 0],\n goldenrod: [218, 165, 32],\n gray: [128, 128, 128],\n green: [0, 128, 0],\n greenyellow: [173, 255, 47],\n grey: [128, 128, 128],\n honeydew: [240, 255, 240],\n hotpink: [255, 105, 180],\n indianred: [205, 92, 92],\n indigo: [75, 0, 130],\n ivory: [255, 255, 240],\n khaki: [240, 230, 140],\n lavender: [230, 230, 250],\n lavenderblush: [255, 240, 245],\n lawngreen: [124, 252, 0],\n lemonchiffon: [255, 250, 205],\n lightblue: [173, 216, 230],\n lightcoral: [240, 128, 128],\n lightcyan: [224, 255, 255],\n lightgoldenrodyellow: [250, 250, 210],\n lightgray: [211, 211, 211],\n lightgreen: [144, 238, 144],\n lightgrey: [211, 211, 211],\n lightpink: [255, 182, 193],\n lightsalmon: [255, 160, 122],\n lightseagreen: [32, 178, 170],\n lightskyblue: [135, 206, 250],\n lightslategray: [119, 136, 153],\n lightslategrey: [119, 136, 153],\n lightsteelblue: [176, 196, 222],\n lightyellow: [255, 255, 224],\n lime: [0, 255, 0],\n limegreen: [50, 205, 50],\n linen: [250, 240, 230],\n magenta: [255, 0, 255],\n maroon: [128, 0, 0],\n mediumaquamarine: [102, 205, 170],\n mediumblue: [0, 0, 205],\n mediumorchid: [186, 85, 211],\n mediumpurple: [147, 112, 219],\n mediumseagreen: [60, 179, 113],\n mediumslateblue: [123, 104, 238],\n mediumspringgreen: [0, 250, 154],\n mediumturquoise: [72, 209, 204],\n mediumvioletred: [199, 21, 133],\n midnightblue: [25, 25, 112],\n mintcream: [245, 255, 250],\n mistyrose: [255, 228, 225],\n moccasin: [255, 228, 181],\n navajowhite: [255, 222, 173],\n navy: [0, 0, 128],\n oldlace: [253, 245, 230],\n olive: [128, 128, 0],\n olivedrab: [107, 142, 35],\n orange: [255, 165, 0],\n orangered: [255, 69, 0],\n orchid: [218, 112, 214],\n palegoldenrod: [238, 232, 170],\n palegreen: [152, 251, 152],\n paleturquoise: [175, 238, 238],\n palevioletred: [219, 112, 147],\n papayawhip: [255, 239, 213],\n peachpuff: [255, 218, 185],\n peru: [205, 133, 63],\n pink: [255, 192, 203],\n plum: [221, 160, 221],\n powderblue: [176, 224, 230],\n purple: [128, 0, 128],\n rebeccapurple: [102, 51, 153],\n red: [255, 0, 0],\n rosybrown: [188, 143, 143],\n royalblue: [65, 105, 225],\n saddlebrown: [139, 69, 19],\n salmon: [250, 128, 114],\n sandybrown: [244, 164, 96],\n seagreen: [46, 139, 87],\n seashell: [255, 245, 238],\n sienna: [160, 82, 45],\n silver: [192, 192, 192],\n skyblue: [135, 206, 235],\n slateblue: [106, 90, 205],\n slategray: [112, 128, 144],\n slategrey: [112, 128, 144],\n snow: [255, 250, 250],\n springgreen: [0, 255, 127],\n steelblue: [70, 130, 180],\n tan: [210, 180, 140],\n teal: [0, 128, 128],\n thistle: [216, 191, 216],\n tomato: [255, 99, 71],\n turquoise: [64, 224, 208],\n violet: [238, 130, 238],\n wheat: [245, 222, 179],\n white: [255, 255, 255],\n whitesmoke: [245, 245, 245],\n yellow: [255, 255, 0],\n yellowgreen: [154, 205, 50],\n};\n\n/**\n * Color representation used by WebGL.\n * Defined in sRGB color space and pre-blended with alpha.\n * @private\n */\nclass Color {\n /**\n * @param r Red component premultiplied by `alpha` 0..1\n * @param g Green component premultiplied by `alpha` 0..1\n * @param b Blue component premultiplied by `alpha` 0..1\n * @param [alpha=1] Alpha component 0..1\n * @param [premultiplied=true] Whether the `r`, `g` and `b` values have already\n * been multiplied by alpha. If `true` nothing happens if `false` then they will\n * be multiplied automatically.\n */\n constructor(r, g, b, alpha = 1, premultiplied = true) {\n this.r = r;\n this.g = g;\n this.b = b;\n this.a = alpha;\n if (!premultiplied) {\n this.r *= alpha;\n this.g *= alpha;\n this.b *= alpha;\n if (!alpha) {\n // alpha = 0 erases completely rgb channels. This behavior is not desirable\n // if this particular color is later used in color interpolation.\n // Because of that, a reference to original color is saved.\n this.overwriteGetter('rgb', [r, g, b, alpha]);\n }\n }\n }\n /**\n * Parses CSS color strings and converts colors to sRGB color space if needed.\n * Officially supported color formats:\n * - keyword, e.g. 'aquamarine' or 'steelblue'\n * - hex (with 3, 4, 6 or 8 digits), e.g. '#f0f' or '#e9bebea9'\n * - rgb and rgba, e.g. 'rgb(0,240,120)' or 'rgba(0%,94%,47%,0.1)' or 'rgb(0 240 120 / .3)'\n * - hsl and hsla, e.g. 'hsl(0,0%,83%)' or 'hsla(0,0%,83%,.5)' or 'hsl(0 0% 83% / 20%)'\n *\n * @param input CSS color string to parse.\n * @returns A `Color` instance, or `undefined` if the input is not a valid color string.\n */\n static parse(input) {\n // in zoom-and-property function input could be an instance of Color class\n if (input instanceof Color) {\n return input;\n }\n if (typeof input !== 'string') {\n return;\n }\n const rgba = parseCssColor(input);\n if (rgba) {\n return new Color(...rgba, false);\n }\n }\n /**\n * Used in color interpolation and by 'to-rgba' expression.\n *\n * @returns Gien color, with reversed alpha blending, in sRGB color space.\n */\n get rgb() {\n const { r, g, b, a } = this;\n const f = a || Infinity; // reverse alpha blending factor\n return this.overwriteGetter('rgb', [r / f, g / f, b / f, a]);\n }\n /**\n * Used in color interpolation.\n *\n * @returns Gien color, with reversed alpha blending, in HCL color space.\n */\n get hcl() {\n return this.overwriteGetter('hcl', rgbToHcl(this.rgb));\n }\n /**\n * Used in color interpolation.\n *\n * @returns Gien color, with reversed alpha blending, in LAB color space.\n */\n get lab() {\n return this.overwriteGetter('lab', rgbToLab(this.rgb));\n }\n /**\n * Lazy getter pattern. When getter is called for the first time lazy value\n * is calculated and then overwrites getter function in given object instance.\n *\n * @example:\n * const redColor = Color.parse('red');\n * let x = redColor.hcl; // this will invoke `get hcl()`, which will calculate\n * // the value of red in HCL space and invoke this `overwriteGetter` function\n * // which in turn will set a field with a key 'hcl' in the `redColor` object.\n * // In other words it will override `get hcl()` from its `Color` prototype\n * // with its own property: hcl = [calculated red value in hcl].\n * let y = redColor.hcl; // next call will no longer invoke getter but simply\n * // return the previously calculated value\n * x === y; // true - `x` is exactly the same object as `y`\n *\n * @param getterKey Getter key\n * @param lazyValue Lazily calculated value to be memoized by current instance\n * @private\n */\n overwriteGetter(getterKey, lazyValue) {\n Object.defineProperty(this, getterKey, { value: lazyValue });\n return lazyValue;\n }\n /**\n * Used by 'to-string' expression.\n *\n * @returns Serialized color in format `rgba(r,g,b,a)`\n * where r,g,b are numbers within 0..255 and alpha is number within 1..0\n *\n * @example\n * var purple = new Color.parse('purple');\n * purple.toString; // = \"rgba(128,0,128,1)\"\n * var translucentGreen = new Color.parse('rgba(26, 207, 26, .73)');\n * translucentGreen.toString(); // = \"rgba(26,207,26,0.73)\"\n */\n toString() {\n const [r, g, b, a] = this.rgb;\n return `rgba(${[r, g, b].map(n => Math.round(n * 255)).join(',')},${a})`;\n }\n}\nColor.black = new Color(0, 0, 0, 1);\nColor.white = new Color(1, 1, 1, 1);\nColor.transparent = new Color(0, 0, 0, 0);\nColor.red = new Color(1, 0, 0, 1);\n\n// Flow type declarations for Intl cribbed from\n// https://github.com/facebook/flow/issues/1270\nclass Collator {\n constructor(caseSensitive, diacriticSensitive, locale) {\n if (caseSensitive)\n this.sensitivity = diacriticSensitive ? 'variant' : 'case';\n else\n this.sensitivity = diacriticSensitive ? 'accent' : 'base';\n this.locale = locale;\n this.collator = new Intl.Collator(this.locale ? this.locale : [], { sensitivity: this.sensitivity, usage: 'search' });\n }\n compare(lhs, rhs) {\n return this.collator.compare(lhs, rhs);\n }\n resolvedLocale() {\n // We create a Collator without \"usage: search\" because we don't want\n // the search options encoded in our result (e.g. \"en-u-co-search\")\n return new Intl.Collator(this.locale ? this.locale : [])\n .resolvedOptions().locale;\n }\n}\n\nclass FormattedSection {\n constructor(text, image, scale, fontStack, textColor) {\n this.text = text;\n this.image = image;\n this.scale = scale;\n this.fontStack = fontStack;\n this.textColor = textColor;\n }\n}\nclass Formatted {\n constructor(sections) {\n this.sections = sections;\n }\n static fromString(unformatted) {\n return new Formatted([new FormattedSection(unformatted, null, null, null, null)]);\n }\n isEmpty() {\n if (this.sections.length === 0)\n return true;\n return !this.sections.some(section => section.text.length !== 0 ||\n (section.image && section.image.name.length !== 0));\n }\n static factory(text) {\n if (text instanceof Formatted) {\n return text;\n }\n else {\n return Formatted.fromString(text);\n }\n }\n toString() {\n if (this.sections.length === 0)\n return '';\n return this.sections.map(section => section.text).join('');\n }\n}\n\n/**\n * A set of four numbers representing padding around a box. Create instances from\n * bare arrays or numeric values using the static method `Padding.parse`.\n * @private\n */\nclass Padding {\n constructor(values) {\n this.values = values.slice();\n }\n /**\n * Numeric padding values\n * @param input A padding value\n * @returns A `Padding` instance, or `undefined` if the input is not a valid padding value.\n */\n static parse(input) {\n if (input instanceof Padding) {\n return input;\n }\n // Backwards compatibility: bare number is treated the same as array with single value.\n // Padding applies to all four sides.\n if (typeof input === 'number') {\n return new Padding([input, input, input, input]);\n }\n if (!Array.isArray(input)) {\n return undefined;\n }\n if (input.length < 1 || input.length > 4) {\n return undefined;\n }\n for (const val of input) {\n if (typeof val !== 'number') {\n return undefined;\n }\n }\n // Expand shortcut properties into explicit 4-sided values\n switch (input.length) {\n case 1:\n input = [input[0], input[0], input[0], input[0]];\n break;\n case 2:\n input = [input[0], input[1], input[0], input[1]];\n break;\n case 3:\n input = [input[0], input[1], input[2], input[1]];\n break;\n }\n return new Padding(input);\n }\n toString() {\n return JSON.stringify(this.values);\n }\n}\n\n/** Set of valid anchor positions, as a set for validation */\nconst anchors = new Set(['center', 'left', 'right', 'top', 'bottom', 'top-left', 'top-right', 'bottom-left', 'bottom-right']);\n/**\n * Utility class to assist managing values for text-variable-anchor-offset property. Create instances from\n * bare arrays using the static method `VariableAnchorOffsetCollection.parse`.\n * @private\n */\nclass VariableAnchorOffsetCollection {\n constructor(values) {\n this.values = values.slice();\n }\n static parse(input) {\n if (input instanceof VariableAnchorOffsetCollection) {\n return input;\n }\n if (!Array.isArray(input) ||\n input.length < 1 ||\n input.length % 2 !== 0) {\n return undefined;\n }\n for (let i = 0; i < input.length; i += 2) {\n // Elements in even positions should be anchor positions; Elements in odd positions should be offset values\n const anchorValue = input[i];\n const offsetValue = input[i + 1];\n if (typeof anchorValue !== 'string' || !anchors.has(anchorValue)) {\n return undefined;\n }\n if (!Array.isArray(offsetValue) || offsetValue.length !== 2 || typeof offsetValue[0] !== 'number' || typeof offsetValue[1] !== 'number') {\n return undefined;\n }\n }\n return new VariableAnchorOffsetCollection(input);\n }\n toString() {\n return JSON.stringify(this.values);\n }\n}\n\nclass ResolvedImage {\n constructor(options) {\n this.name = options.name;\n this.available = options.available;\n }\n toString() {\n return this.name;\n }\n static fromString(name) {\n if (!name)\n return null; // treat empty values as no image\n return new ResolvedImage({ name, available: false });\n }\n}\n\nfunction validateRGBA(r, g, b, a) {\n if (!(typeof r === 'number' && r >= 0 && r <= 255 &&\n typeof g === 'number' && g >= 0 && g <= 255 &&\n typeof b === 'number' && b >= 0 && b <= 255)) {\n const value = typeof a === 'number' ? [r, g, b, a] : [r, g, b];\n return `Invalid rgba value [${value.join(', ')}]: 'r', 'g', and 'b' must be between 0 and 255.`;\n }\n if (!(typeof a === 'undefined' || (typeof a === 'number' && a >= 0 && a <= 1))) {\n return `Invalid rgba value [${[r, g, b, a].join(', ')}]: 'a' must be between 0 and 1.`;\n }\n return null;\n}\nfunction isValue(mixed) {\n if (mixed === null ||\n typeof mixed === 'string' ||\n typeof mixed === 'boolean' ||\n typeof mixed === 'number' ||\n mixed instanceof Color ||\n mixed instanceof Collator ||\n mixed instanceof Formatted ||\n mixed instanceof Padding ||\n mixed instanceof VariableAnchorOffsetCollection ||\n mixed instanceof ResolvedImage) {\n return true;\n }\n else if (Array.isArray(mixed)) {\n for (const item of mixed) {\n if (!isValue(item)) {\n return false;\n }\n }\n return true;\n }\n else if (typeof mixed === 'object') {\n for (const key in mixed) {\n if (!isValue(mixed[key])) {\n return false;\n }\n }\n return true;\n }\n else {\n return false;\n }\n}\nfunction typeOf(value) {\n if (value === null) {\n return NullType;\n }\n else if (typeof value === 'string') {\n return StringType;\n }\n else if (typeof value === 'boolean') {\n return BooleanType;\n }\n else if (typeof value === 'number') {\n return NumberType;\n }\n else if (value instanceof Color) {\n return ColorType;\n }\n else if (value instanceof Collator) {\n return CollatorType;\n }\n else if (value instanceof Formatted) {\n return FormattedType;\n }\n else if (value instanceof Padding) {\n return PaddingType;\n }\n else if (value instanceof VariableAnchorOffsetCollection) {\n return VariableAnchorOffsetCollectionType;\n }\n else if (value instanceof ResolvedImage) {\n return ResolvedImageType;\n }\n else if (Array.isArray(value)) {\n const length = value.length;\n let itemType;\n for (const item of value) {\n const t = typeOf(item);\n if (!itemType) {\n itemType = t;\n }\n else if (itemType === t) {\n continue;\n }\n else {\n itemType = ValueType;\n break;\n }\n }\n return array$1(itemType || ValueType, length);\n }\n else {\n return ObjectType;\n }\n}\nfunction toString(value) {\n const type = typeof value;\n if (value === null) {\n return '';\n }\n else if (type === 'string' || type === 'number' || type === 'boolean') {\n return String(value);\n }\n else if (value instanceof Color || value instanceof Formatted || value instanceof Padding || value instanceof VariableAnchorOffsetCollection || value instanceof ResolvedImage) {\n return value.toString();\n }\n else {\n return JSON.stringify(value);\n }\n}\n\nclass Literal {\n constructor(type, value) {\n this.type = type;\n this.value = value;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error(`'literal' expression requires exactly one argument, but found ${args.length - 1} instead.`);\n if (!isValue(args[1]))\n return context.error('invalid value');\n const value = args[1];\n let type = typeOf(value);\n // special case: infer the item type if possible for zero-length arrays\n const expected = context.expectedType;\n if (type.kind === 'array' &&\n type.N === 0 &&\n expected &&\n expected.kind === 'array' &&\n (typeof expected.N !== 'number' || expected.N === 0)) {\n type = expected;\n }\n return new Literal(type, value);\n }\n evaluate() {\n return this.value;\n }\n eachChild() { }\n outputDefined() {\n return true;\n }\n}\n\nclass RuntimeError {\n constructor(message) {\n this.name = 'ExpressionEvaluationError';\n this.message = message;\n }\n toJSON() {\n return this.message;\n }\n}\n\nconst types$1 = {\n string: StringType,\n number: NumberType,\n boolean: BooleanType,\n object: ObjectType\n};\nclass Assertion {\n constructor(type, args) {\n this.type = type;\n this.args = args;\n }\n static parse(args, context) {\n if (args.length < 2)\n return context.error('Expected at least one argument.');\n let i = 1;\n let type;\n const name = args[0];\n if (name === 'array') {\n let itemType;\n if (args.length > 2) {\n const type = args[1];\n if (typeof type !== 'string' || !(type in types$1) || type === 'object')\n return context.error('The item type argument of \"array\" must be one of string, number, boolean', 1);\n itemType = types$1[type];\n i++;\n }\n else {\n itemType = ValueType;\n }\n let N;\n if (args.length > 3) {\n if (args[2] !== null &&\n (typeof args[2] !== 'number' ||\n args[2] < 0 ||\n args[2] !== Math.floor(args[2]))) {\n return context.error('The length argument to \"array\" must be a positive integer literal', 2);\n }\n N = args[2];\n i++;\n }\n type = array$1(itemType, N);\n }\n else {\n if (!types$1[name])\n throw new Error(`Types doesn't contain name = ${name}`);\n type = types$1[name];\n }\n const parsed = [];\n for (; i < args.length; i++) {\n const input = context.parse(args[i], i, ValueType);\n if (!input)\n return null;\n parsed.push(input);\n }\n return new Assertion(type, parsed);\n }\n evaluate(ctx) {\n for (let i = 0; i < this.args.length; i++) {\n const value = this.args[i].evaluate(ctx);\n const error = checkSubtype(this.type, typeOf(value));\n if (!error) {\n return value;\n }\n else if (i === this.args.length - 1) {\n throw new RuntimeError(`Expected value to be of type ${toString$1(this.type)}, but found ${toString$1(typeOf(value))} instead.`);\n }\n }\n throw new Error();\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return this.args.every(arg => arg.outputDefined());\n }\n}\n\nconst types = {\n 'to-boolean': BooleanType,\n 'to-color': ColorType,\n 'to-number': NumberType,\n 'to-string': StringType\n};\n/**\n * Special form for error-coalescing coercion expressions \"to-number\",\n * \"to-color\". Since these coercions can fail at runtime, they accept multiple\n * arguments, only evaluating one at a time until one succeeds.\n *\n * @private\n */\nclass Coercion {\n constructor(type, args) {\n this.type = type;\n this.args = args;\n }\n static parse(args, context) {\n if (args.length < 2)\n return context.error('Expected at least one argument.');\n const name = args[0];\n if (!types[name])\n throw new Error(`Can't parse ${name} as it is not part of the known types`);\n if ((name === 'to-boolean' || name === 'to-string') && args.length !== 2)\n return context.error('Expected one argument.');\n const type = types[name];\n const parsed = [];\n for (let i = 1; i < args.length; i++) {\n const input = context.parse(args[i], i, ValueType);\n if (!input)\n return null;\n parsed.push(input);\n }\n return new Coercion(type, parsed);\n }\n evaluate(ctx) {\n switch (this.type.kind) {\n case 'boolean':\n return Boolean(this.args[0].evaluate(ctx));\n case 'color': {\n let input;\n let error;\n for (const arg of this.args) {\n input = arg.evaluate(ctx);\n error = null;\n if (input instanceof Color) {\n return input;\n }\n else if (typeof input === 'string') {\n const c = ctx.parseColor(input);\n if (c)\n return c;\n }\n else if (Array.isArray(input)) {\n if (input.length < 3 || input.length > 4) {\n error = `Invalid rbga value ${JSON.stringify(input)}: expected an array containing either three or four numeric values.`;\n }\n else {\n error = validateRGBA(input[0], input[1], input[2], input[3]);\n }\n if (!error) {\n return new Color(input[0] / 255, input[1] / 255, input[2] / 255, input[3]);\n }\n }\n }\n throw new RuntimeError(error || `Could not parse color from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`);\n }\n case 'padding': {\n let input;\n for (const arg of this.args) {\n input = arg.evaluate(ctx);\n const pad = Padding.parse(input);\n if (pad) {\n return pad;\n }\n }\n throw new RuntimeError(`Could not parse padding from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`);\n }\n case 'variableAnchorOffsetCollection': {\n let input;\n for (const arg of this.args) {\n input = arg.evaluate(ctx);\n const coll = VariableAnchorOffsetCollection.parse(input);\n if (coll) {\n return coll;\n }\n }\n throw new RuntimeError(`Could not parse variableAnchorOffsetCollection from value '${typeof input === 'string' ? input : JSON.stringify(input)}'`);\n }\n case 'number': {\n let value = null;\n for (const arg of this.args) {\n value = arg.evaluate(ctx);\n if (value === null)\n return 0;\n const num = Number(value);\n if (isNaN(num))\n continue;\n return num;\n }\n throw new RuntimeError(`Could not convert ${JSON.stringify(value)} to number.`);\n }\n case 'formatted':\n // There is no explicit 'to-formatted' but this coercion can be implicitly\n // created by properties that expect the 'formatted' type.\n return Formatted.fromString(toString(this.args[0].evaluate(ctx)));\n case 'resolvedImage':\n return ResolvedImage.fromString(toString(this.args[0].evaluate(ctx)));\n default:\n return toString(this.args[0].evaluate(ctx));\n }\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return this.args.every(arg => arg.outputDefined());\n }\n}\n\nconst geometryTypes = ['Unknown', 'Point', 'LineString', 'Polygon'];\nclass EvaluationContext {\n constructor() {\n this.globals = null;\n this.feature = null;\n this.featureState = null;\n this.formattedSection = null;\n this._parseColorCache = {};\n this.availableImages = null;\n this.canonical = null;\n }\n id() {\n return this.feature && 'id' in this.feature ? this.feature.id : null;\n }\n geometryType() {\n return this.feature ? typeof this.feature.type === 'number' ? geometryTypes[this.feature.type] : this.feature.type : null;\n }\n geometry() {\n return this.feature && 'geometry' in this.feature ? this.feature.geometry : null;\n }\n canonicalID() {\n return this.canonical;\n }\n properties() {\n return this.feature && this.feature.properties || {};\n }\n parseColor(input) {\n let cached = this._parseColorCache[input];\n if (!cached) {\n cached = this._parseColorCache[input] = Color.parse(input);\n }\n return cached;\n }\n}\n\n/**\n * State associated parsing at a given point in an expression tree.\n * @private\n */\nclass ParsingContext {\n constructor(registry, isConstantFunc, path = [], expectedType, scope = new Scope(), errors = []) {\n this.registry = registry;\n this.path = path;\n this.key = path.map(part => `[${part}]`).join('');\n this.scope = scope;\n this.errors = errors;\n this.expectedType = expectedType;\n this._isConstant = isConstantFunc;\n }\n /**\n * @param expr the JSON expression to parse\n * @param index the optional argument index if this expression is an argument of a parent expression that's being parsed\n * @param options\n * @param options.omitTypeAnnotations set true to omit inferred type annotations. Caller beware: with this option set, the parsed expression's type will NOT satisfy `expectedType` if it would normally be wrapped in an inferred annotation.\n * @private\n */\n parse(expr, index, expectedType, bindings, options = {}) {\n if (index) {\n return this.concat(index, expectedType, bindings)._parse(expr, options);\n }\n return this._parse(expr, options);\n }\n _parse(expr, options) {\n if (expr === null || typeof expr === 'string' || typeof expr === 'boolean' || typeof expr === 'number') {\n expr = ['literal', expr];\n }\n function annotate(parsed, type, typeAnnotation) {\n if (typeAnnotation === 'assert') {\n return new Assertion(type, [parsed]);\n }\n else if (typeAnnotation === 'coerce') {\n return new Coercion(type, [parsed]);\n }\n else {\n return parsed;\n }\n }\n if (Array.isArray(expr)) {\n if (expr.length === 0) {\n return this.error('Expected an array with at least one element. If you wanted a literal array, use [\"literal\", []].');\n }\n const op = expr[0];\n if (typeof op !== 'string') {\n this.error(`Expression name must be a string, but found ${typeof op} instead. If you wanted a literal array, use [\"literal\", [...]].`, 0);\n return null;\n }\n const Expr = this.registry[op];\n if (Expr) {\n let parsed = Expr.parse(expr, this);\n if (!parsed)\n return null;\n if (this.expectedType) {\n const expected = this.expectedType;\n const actual = parsed.type;\n // When we expect a number, string, boolean, or array but have a value, wrap it in an assertion.\n // When we expect a color or formatted string, but have a string or value, wrap it in a coercion.\n // Otherwise, we do static type-checking.\n //\n // These behaviors are overridable for:\n // * The \"coalesce\" operator, which needs to omit type annotations.\n // * String-valued properties (e.g. `text-field`), where coercion is more convenient than assertion.\n //\n if ((expected.kind === 'string' || expected.kind === 'number' || expected.kind === 'boolean' || expected.kind === 'object' || expected.kind === 'array') && actual.kind === 'value') {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'assert');\n }\n else if ((expected.kind === 'color' || expected.kind === 'formatted' || expected.kind === 'resolvedImage') && (actual.kind === 'value' || actual.kind === 'string')) {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce');\n }\n else if (expected.kind === 'padding' && (actual.kind === 'value' || actual.kind === 'number' || actual.kind === 'array')) {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce');\n }\n else if (expected.kind === 'variableAnchorOffsetCollection' && (actual.kind === 'value' || actual.kind === 'array')) {\n parsed = annotate(parsed, expected, options.typeAnnotation || 'coerce');\n }\n else if (this.checkSubtype(expected, actual)) {\n return null;\n }\n }\n // If an expression's arguments are all literals, we can evaluate\n // it immediately and replace it with a literal value in the\n // parsed/compiled result. Expressions that expect an image should\n // not be resolved here so we can later get the available images.\n if (!(parsed instanceof Literal) && (parsed.type.kind !== 'resolvedImage') && this._isConstant(parsed)) {\n const ec = new EvaluationContext();\n try {\n parsed = new Literal(parsed.type, parsed.evaluate(ec));\n }\n catch (e) {\n this.error(e.message);\n return null;\n }\n }\n return parsed;\n }\n return this.error(`Unknown expression \"${op}\". If you wanted a literal array, use [\"literal\", [...]].`, 0);\n }\n else if (typeof expr === 'undefined') {\n return this.error('\\'undefined\\' value invalid. Use null instead.');\n }\n else if (typeof expr === 'object') {\n return this.error('Bare objects invalid. Use [\"literal\", {...}] instead.');\n }\n else {\n return this.error(`Expected an array, but found ${typeof expr} instead.`);\n }\n }\n /**\n * Returns a copy of this context suitable for parsing the subexpression at\n * index `index`, optionally appending to 'let' binding map.\n *\n * Note that `errors` property, intended for collecting errors while\n * parsing, is copied by reference rather than cloned.\n * @private\n */\n concat(index, expectedType, bindings) {\n const path = typeof index === 'number' ? this.path.concat(index) : this.path;\n const scope = bindings ? this.scope.concat(bindings) : this.scope;\n return new ParsingContext(this.registry, this._isConstant, path, expectedType || null, scope, this.errors);\n }\n /**\n * Push a parsing (or type checking) error into the `this.errors`\n * @param error The message\n * @param keys Optionally specify the source of the error at a child\n * of the current expression at `this.key`.\n * @private\n */\n error(error, ...keys) {\n const key = `${this.key}${keys.map(k => `[${k}]`).join('')}`;\n this.errors.push(new ExpressionParsingError(key, error));\n }\n /**\n * Returns null if `t` is a subtype of `expected`; otherwise returns an\n * error message and also pushes it to `this.errors`.\n * @param expected The expected type\n * @param t The actual type\n * @returns null if `t` is a subtype of `expected`; otherwise returns an error message\n */\n checkSubtype(expected, t) {\n const error = checkSubtype(expected, t);\n if (error)\n this.error(error);\n return error;\n }\n}\n\nclass CollatorExpression {\n constructor(caseSensitive, diacriticSensitive, locale) {\n this.type = CollatorType;\n this.locale = locale;\n this.caseSensitive = caseSensitive;\n this.diacriticSensitive = diacriticSensitive;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error('Expected one argument.');\n const options = args[1];\n if (typeof options !== 'object' || Array.isArray(options))\n return context.error('Collator options argument must be an object.');\n const caseSensitive = context.parse(options['case-sensitive'] === undefined ? false : options['case-sensitive'], 1, BooleanType);\n if (!caseSensitive)\n return null;\n const diacriticSensitive = context.parse(options['diacritic-sensitive'] === undefined ? false : options['diacritic-sensitive'], 1, BooleanType);\n if (!diacriticSensitive)\n return null;\n let locale = null;\n if (options['locale']) {\n locale = context.parse(options['locale'], 1, StringType);\n if (!locale)\n return null;\n }\n return new CollatorExpression(caseSensitive, diacriticSensitive, locale);\n }\n evaluate(ctx) {\n return new Collator(this.caseSensitive.evaluate(ctx), this.diacriticSensitive.evaluate(ctx), this.locale ? this.locale.evaluate(ctx) : null);\n }\n eachChild(fn) {\n fn(this.caseSensitive);\n fn(this.diacriticSensitive);\n if (this.locale) {\n fn(this.locale);\n }\n }\n outputDefined() {\n // Technically the set of possible outputs is the combinatoric set of Collators produced\n // by all possible outputs of locale/caseSensitive/diacriticSensitive\n // But for the primary use of Collators in comparison operators, we ignore the Collator's\n // possible outputs anyway, so we can get away with leaving this false for now.\n return false;\n }\n}\n\nconst EXTENT = 8192;\nfunction updateBBox(bbox, coord) {\n bbox[0] = Math.min(bbox[0], coord[0]);\n bbox[1] = Math.min(bbox[1], coord[1]);\n bbox[2] = Math.max(bbox[2], coord[0]);\n bbox[3] = Math.max(bbox[3], coord[1]);\n}\nfunction mercatorXfromLng(lng) {\n return (180 + lng) / 360;\n}\nfunction mercatorYfromLat(lat) {\n return (180 - (180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360)))) / 360;\n}\nfunction boxWithinBox(bbox1, bbox2) {\n if (bbox1[0] <= bbox2[0])\n return false;\n if (bbox1[2] >= bbox2[2])\n return false;\n if (bbox1[1] <= bbox2[1])\n return false;\n if (bbox1[3] >= bbox2[3])\n return false;\n return true;\n}\nfunction getTileCoordinates(p, canonical) {\n const x = mercatorXfromLng(p[0]);\n const y = mercatorYfromLat(p[1]);\n const tilesAtZoom = Math.pow(2, canonical.z);\n return [Math.round(x * tilesAtZoom * EXTENT), Math.round(y * tilesAtZoom * EXTENT)];\n}\nfunction onBoundary(p, p1, p2) {\n const x1 = p[0] - p1[0];\n const y1 = p[1] - p1[1];\n const x2 = p[0] - p2[0];\n const y2 = p[1] - p2[1];\n return (x1 * y2 - x2 * y1 === 0) && (x1 * x2 <= 0) && (y1 * y2 <= 0);\n}\nfunction rayIntersect(p, p1, p2) {\n return ((p1[1] > p[1]) !== (p2[1] > p[1])) && (p[0] < (p2[0] - p1[0]) * (p[1] - p1[1]) / (p2[1] - p1[1]) + p1[0]);\n}\n// ray casting algorithm for detecting if point is in polygon\nfunction pointWithinPolygon(point, rings) {\n let inside = false;\n for (let i = 0, len = rings.length; i < len; i++) {\n const ring = rings[i];\n for (let j = 0, len2 = ring.length; j < len2 - 1; j++) {\n if (onBoundary(point, ring[j], ring[j + 1]))\n return false;\n if (rayIntersect(point, ring[j], ring[j + 1]))\n inside = !inside;\n }\n }\n return inside;\n}\nfunction pointWithinPolygons(point, polygons) {\n for (let i = 0; i < polygons.length; i++) {\n if (pointWithinPolygon(point, polygons[i]))\n return true;\n }\n return false;\n}\nfunction perp(v1, v2) {\n return (v1[0] * v2[1] - v1[1] * v2[0]);\n}\n// check if p1 and p2 are in different sides of line segment q1->q2\nfunction twoSided(p1, p2, q1, q2) {\n // q1->p1 (x1, y1), q1->p2 (x2, y2), q1->q2 (x3, y3)\n const x1 = p1[0] - q1[0];\n const y1 = p1[1] - q1[1];\n const x2 = p2[0] - q1[0];\n const y2 = p2[1] - q1[1];\n const x3 = q2[0] - q1[0];\n const y3 = q2[1] - q1[1];\n const det1 = (x1 * y3 - x3 * y1);\n const det2 = (x2 * y3 - x3 * y2);\n if ((det1 > 0 && det2 < 0) || (det1 < 0 && det2 > 0))\n return true;\n return false;\n}\n// a, b are end points for line segment1, c and d are end points for line segment2\nfunction lineIntersectLine(a, b, c, d) {\n // check if two segments are parallel or not\n // precondition is end point a, b is inside polygon, if line a->b is\n // parallel to polygon edge c->d, then a->b won't intersect with c->d\n const vectorP = [b[0] - a[0], b[1] - a[1]];\n const vectorQ = [d[0] - c[0], d[1] - c[1]];\n if (perp(vectorQ, vectorP) === 0)\n return false;\n // If lines are intersecting with each other, the relative location should be:\n // a and b lie in different sides of segment c->d\n // c and d lie in different sides of segment a->b\n if (twoSided(a, b, c, d) && twoSided(c, d, a, b))\n return true;\n return false;\n}\nfunction lineIntersectPolygon(p1, p2, polygon) {\n for (const ring of polygon) {\n // loop through every edge of the ring\n for (let j = 0; j < ring.length - 1; ++j) {\n if (lineIntersectLine(p1, p2, ring[j], ring[j + 1])) {\n return true;\n }\n }\n }\n return false;\n}\nfunction lineStringWithinPolygon(line, polygon) {\n // First, check if geometry points of line segments are all inside polygon\n for (let i = 0; i < line.length; ++i) {\n if (!pointWithinPolygon(line[i], polygon)) {\n return false;\n }\n }\n // Second, check if there is line segment intersecting polygon edge\n for (let i = 0; i < line.length - 1; ++i) {\n if (lineIntersectPolygon(line[i], line[i + 1], polygon)) {\n return false;\n }\n }\n return true;\n}\nfunction lineStringWithinPolygons(line, polygons) {\n for (let i = 0; i < polygons.length; i++) {\n if (lineStringWithinPolygon(line, polygons[i]))\n return true;\n }\n return false;\n}\nfunction getTilePolygon(coordinates, bbox, canonical) {\n const polygon = [];\n for (let i = 0; i < coordinates.length; i++) {\n const ring = [];\n for (let j = 0; j < coordinates[i].length; j++) {\n const coord = getTileCoordinates(coordinates[i][j], canonical);\n updateBBox(bbox, coord);\n ring.push(coord);\n }\n polygon.push(ring);\n }\n return polygon;\n}\nfunction getTilePolygons(coordinates, bbox, canonical) {\n const polygons = [];\n for (let i = 0; i < coordinates.length; i++) {\n const polygon = getTilePolygon(coordinates[i], bbox, canonical);\n polygons.push(polygon);\n }\n return polygons;\n}\nfunction updatePoint(p, bbox, polyBBox, worldSize) {\n if (p[0] < polyBBox[0] || p[0] > polyBBox[2]) {\n const halfWorldSize = worldSize * 0.5;\n let shift = (p[0] - polyBBox[0] > halfWorldSize) ? -worldSize : (polyBBox[0] - p[0] > halfWorldSize) ? worldSize : 0;\n if (shift === 0) {\n shift = (p[0] - polyBBox[2] > halfWorldSize) ? -worldSize : (polyBBox[2] - p[0] > halfWorldSize) ? worldSize : 0;\n }\n p[0] += shift;\n }\n updateBBox(bbox, p);\n}\nfunction resetBBox(bbox) {\n bbox[0] = bbox[1] = Infinity;\n bbox[2] = bbox[3] = -Infinity;\n}\nfunction getTilePoints(geometry, pointBBox, polyBBox, canonical) {\n const worldSize = Math.pow(2, canonical.z) * EXTENT;\n const shifts = [canonical.x * EXTENT, canonical.y * EXTENT];\n const tilePoints = [];\n for (const points of geometry) {\n for (const point of points) {\n const p = [point.x + shifts[0], point.y + shifts[1]];\n updatePoint(p, pointBBox, polyBBox, worldSize);\n tilePoints.push(p);\n }\n }\n return tilePoints;\n}\nfunction getTileLines(geometry, lineBBox, polyBBox, canonical) {\n const worldSize = Math.pow(2, canonical.z) * EXTENT;\n const shifts = [canonical.x * EXTENT, canonical.y * EXTENT];\n const tileLines = [];\n for (const line of geometry) {\n const tileLine = [];\n for (const point of line) {\n const p = [point.x + shifts[0], point.y + shifts[1]];\n updateBBox(lineBBox, p);\n tileLine.push(p);\n }\n tileLines.push(tileLine);\n }\n if (lineBBox[2] - lineBBox[0] <= worldSize / 2) {\n resetBBox(lineBBox);\n for (const line of tileLines) {\n for (const p of line) {\n updatePoint(p, lineBBox, polyBBox, worldSize);\n }\n }\n }\n return tileLines;\n}\nfunction pointsWithinPolygons(ctx, polygonGeometry) {\n const pointBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const polyBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const canonical = ctx.canonicalID();\n if (polygonGeometry.type === 'Polygon') {\n const tilePolygon = getTilePolygon(polygonGeometry.coordinates, polyBBox, canonical);\n const tilePoints = getTilePoints(ctx.geometry(), pointBBox, polyBBox, canonical);\n if (!boxWithinBox(pointBBox, polyBBox))\n return false;\n for (const point of tilePoints) {\n if (!pointWithinPolygon(point, tilePolygon))\n return false;\n }\n }\n if (polygonGeometry.type === 'MultiPolygon') {\n const tilePolygons = getTilePolygons(polygonGeometry.coordinates, polyBBox, canonical);\n const tilePoints = getTilePoints(ctx.geometry(), pointBBox, polyBBox, canonical);\n if (!boxWithinBox(pointBBox, polyBBox))\n return false;\n for (const point of tilePoints) {\n if (!pointWithinPolygons(point, tilePolygons))\n return false;\n }\n }\n return true;\n}\nfunction linesWithinPolygons(ctx, polygonGeometry) {\n const lineBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const polyBBox = [Infinity, Infinity, -Infinity, -Infinity];\n const canonical = ctx.canonicalID();\n if (polygonGeometry.type === 'Polygon') {\n const tilePolygon = getTilePolygon(polygonGeometry.coordinates, polyBBox, canonical);\n const tileLines = getTileLines(ctx.geometry(), lineBBox, polyBBox, canonical);\n if (!boxWithinBox(lineBBox, polyBBox))\n return false;\n for (const line of tileLines) {\n if (!lineStringWithinPolygon(line, tilePolygon))\n return false;\n }\n }\n if (polygonGeometry.type === 'MultiPolygon') {\n const tilePolygons = getTilePolygons(polygonGeometry.coordinates, polyBBox, canonical);\n const tileLines = getTileLines(ctx.geometry(), lineBBox, polyBBox, canonical);\n if (!boxWithinBox(lineBBox, polyBBox))\n return false;\n for (const line of tileLines) {\n if (!lineStringWithinPolygons(line, tilePolygons))\n return false;\n }\n }\n return true;\n}\nclass Within {\n constructor(geojson, geometries) {\n this.type = BooleanType;\n this.geojson = geojson;\n this.geometries = geometries;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error(`'within' expression requires exactly one argument, but found ${args.length - 1} instead.`);\n if (isValue(args[1])) {\n const geojson = args[1];\n if (geojson.type === 'FeatureCollection') {\n for (let i = 0; i < geojson.features.length; ++i) {\n const type = geojson.features[i].geometry.type;\n if (type === 'Polygon' || type === 'MultiPolygon') {\n return new Within(geojson, geojson.features[i].geometry);\n }\n }\n }\n else if (geojson.type === 'Feature') {\n const type = geojson.geometry.type;\n if (type === 'Polygon' || type === 'MultiPolygon') {\n return new Within(geojson, geojson.geometry);\n }\n }\n else if (geojson.type === 'Polygon' || geojson.type === 'MultiPolygon') {\n return new Within(geojson, geojson);\n }\n }\n return context.error('\\'within\\' expression requires valid geojson object that contains polygon geometry type.');\n }\n evaluate(ctx) {\n if (ctx.geometry() != null && ctx.canonicalID() != null) {\n if (ctx.geometryType() === 'Point') {\n return pointsWithinPolygons(ctx, this.geometries);\n }\n else if (ctx.geometryType() === 'LineString') {\n return linesWithinPolygons(ctx, this.geometries);\n }\n }\n return false;\n }\n eachChild() { }\n outputDefined() {\n return true;\n }\n}\n\nclass Var {\n constructor(name, boundExpression) {\n this.type = boundExpression.type;\n this.name = name;\n this.boundExpression = boundExpression;\n }\n static parse(args, context) {\n if (args.length !== 2 || typeof args[1] !== 'string')\n return context.error('\\'var\\' expression requires exactly one string literal argument.');\n const name = args[1];\n if (!context.scope.has(name)) {\n return context.error(`Unknown variable \"${name}\". Make sure \"${name}\" has been bound in an enclosing \"let\" expression before using it.`, 1);\n }\n return new Var(name, context.scope.get(name));\n }\n evaluate(ctx) {\n return this.boundExpression.evaluate(ctx);\n }\n eachChild() { }\n outputDefined() {\n return false;\n }\n}\n\nclass CompoundExpression {\n constructor(name, type, evaluate, args) {\n this.name = name;\n this.type = type;\n this._evaluate = evaluate;\n this.args = args;\n }\n evaluate(ctx) {\n return this._evaluate(ctx, this.args);\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return false;\n }\n static parse(args, context) {\n const op = args[0];\n const definition = CompoundExpression.definitions[op];\n if (!definition) {\n return context.error(`Unknown expression \"${op}\". If you wanted a literal array, use [\"literal\", [...]].`, 0);\n }\n // Now check argument types against each signature\n const type = Array.isArray(definition) ?\n definition[0] : definition.type;\n const availableOverloads = Array.isArray(definition) ?\n [[definition[1], definition[2]]] :\n definition.overloads;\n const overloads = availableOverloads.filter(([signature]) => (!Array.isArray(signature) || // varags\n signature.length === args.length - 1 // correct param count\n ));\n let signatureContext = null;\n for (const [params, evaluate] of overloads) {\n // Use a fresh context for each attempted signature so that, if\n // we eventually succeed, we haven't polluted `context.errors`.\n signatureContext = new ParsingContext(context.registry, isExpressionConstant, context.path, null, context.scope);\n // First parse all the args, potentially coercing to the\n // types expected by this overload.\n const parsedArgs = [];\n let argParseFailed = false;\n for (let i = 1; i < args.length; i++) {\n const arg = args[i];\n const expectedType = Array.isArray(params) ?\n params[i - 1] :\n params.type;\n const parsed = signatureContext.parse(arg, 1 + parsedArgs.length, expectedType);\n if (!parsed) {\n argParseFailed = true;\n break;\n }\n parsedArgs.push(parsed);\n }\n if (argParseFailed) {\n // Couldn't coerce args of this overload to expected type, move\n // on to next one.\n continue;\n }\n if (Array.isArray(params)) {\n if (params.length !== parsedArgs.length) {\n signatureContext.error(`Expected ${params.length} arguments, but found ${parsedArgs.length} instead.`);\n continue;\n }\n }\n for (let i = 0; i < parsedArgs.length; i++) {\n const expected = Array.isArray(params) ? params[i] : params.type;\n const arg = parsedArgs[i];\n signatureContext.concat(i + 1).checkSubtype(expected, arg.type);\n }\n if (signatureContext.errors.length === 0) {\n return new CompoundExpression(op, type, evaluate, parsedArgs);\n }\n }\n if (overloads.length === 1) {\n context.errors.push(...signatureContext.errors);\n }\n else {\n const expected = overloads.length ? overloads : availableOverloads;\n const signatures = expected\n .map(([params]) => stringifySignature(params))\n .join(' | ');\n const actualTypes = [];\n // For error message, re-parse arguments without trying to\n // apply any coercions\n for (let i = 1; i < args.length; i++) {\n const parsed = context.parse(args[i], 1 + actualTypes.length);\n if (!parsed)\n return null;\n actualTypes.push(toString$1(parsed.type));\n }\n context.error(`Expected arguments of type ${signatures}, but found (${actualTypes.join(', ')}) instead.`);\n }\n return null;\n }\n static register(registry, definitions) {\n CompoundExpression.definitions = definitions;\n for (const name in definitions) {\n registry[name] = CompoundExpression;\n }\n }\n}\nfunction stringifySignature(signature) {\n if (Array.isArray(signature)) {\n return `(${signature.map(toString$1).join(', ')})`;\n }\n else {\n return `(${toString$1(signature.type)}...)`;\n }\n}\nfunction isExpressionConstant(expression) {\n if (expression instanceof Var) {\n return isExpressionConstant(expression.boundExpression);\n }\n else if (expression instanceof CompoundExpression && expression.name === 'error') {\n return false;\n }\n else if (expression instanceof CollatorExpression) {\n // Although the results of a Collator expression with fixed arguments\n // generally shouldn't change between executions, we can't serialize them\n // as constant expressions because results change based on environment.\n return false;\n }\n else if (expression instanceof Within) {\n return false;\n }\n const isTypeAnnotation = expression instanceof Coercion ||\n expression instanceof Assertion;\n let childrenConstant = true;\n expression.eachChild(child => {\n // We can _almost_ assume that if `expressions` children are constant,\n // they would already have been evaluated to Literal values when they\n // were parsed. Type annotations are the exception, because they might\n // have been inferred and added after a child was parsed.\n // So we recurse into isConstant() for the children of type annotations,\n // but otherwise simply check whether they are Literals.\n if (isTypeAnnotation) {\n childrenConstant = childrenConstant && isExpressionConstant(child);\n }\n else {\n childrenConstant = childrenConstant && child instanceof Literal;\n }\n });\n if (!childrenConstant) {\n return false;\n }\n return isFeatureConstant(expression) &&\n isGlobalPropertyConstant(expression, ['zoom', 'heatmap-density', 'line-progress', 'accumulated', 'is-supported-script']);\n}\nfunction isFeatureConstant(e) {\n if (e instanceof CompoundExpression) {\n if (e.name === 'get' && e.args.length === 1) {\n return false;\n }\n else if (e.name === 'feature-state') {\n return false;\n }\n else if (e.name === 'has' && e.args.length === 1) {\n return false;\n }\n else if (e.name === 'properties' ||\n e.name === 'geometry-type' ||\n e.name === 'id') {\n return false;\n }\n else if (/^filter-/.test(e.name)) {\n return false;\n }\n }\n if (e instanceof Within) {\n return false;\n }\n let result = true;\n e.eachChild(arg => {\n if (result && !isFeatureConstant(arg)) {\n result = false;\n }\n });\n return result;\n}\nfunction isStateConstant(e) {\n if (e instanceof CompoundExpression) {\n if (e.name === 'feature-state') {\n return false;\n }\n }\n let result = true;\n e.eachChild(arg => {\n if (result && !isStateConstant(arg)) {\n result = false;\n }\n });\n return result;\n}\nfunction isGlobalPropertyConstant(e, properties) {\n if (e instanceof CompoundExpression && properties.indexOf(e.name) >= 0) {\n return false;\n }\n let result = true;\n e.eachChild((arg) => {\n if (result && !isGlobalPropertyConstant(arg, properties)) {\n result = false;\n }\n });\n return result;\n}\n\n/**\n * Returns the index of the last stop <= input, or 0 if it doesn't exist.\n * @private\n */\nfunction findStopLessThanOrEqualTo(stops, input) {\n const lastIndex = stops.length - 1;\n let lowerIndex = 0;\n let upperIndex = lastIndex;\n let currentIndex = 0;\n let currentValue, nextValue;\n while (lowerIndex <= upperIndex) {\n currentIndex = Math.floor((lowerIndex + upperIndex) / 2);\n currentValue = stops[currentIndex];\n nextValue = stops[currentIndex + 1];\n if (currentValue <= input) {\n if (currentIndex === lastIndex || input < nextValue) { // Search complete\n return currentIndex;\n }\n lowerIndex = currentIndex + 1;\n }\n else if (currentValue > input) {\n upperIndex = currentIndex - 1;\n }\n else {\n throw new RuntimeError('Input is not a number.');\n }\n }\n return 0;\n}\n\nclass Step {\n constructor(type, input, stops) {\n this.type = type;\n this.input = input;\n this.labels = [];\n this.outputs = [];\n for (const [label, expression] of stops) {\n this.labels.push(label);\n this.outputs.push(expression);\n }\n }\n static parse(args, context) {\n if (args.length - 1 < 4) {\n return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`);\n }\n if ((args.length - 1) % 2 !== 0) {\n return context.error('Expected an even number of arguments.');\n }\n const input = context.parse(args[1], 1, NumberType);\n if (!input)\n return null;\n const stops = [];\n let outputType = null;\n if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n for (let i = 1; i < args.length; i += 2) {\n const label = i === 1 ? -Infinity : args[i];\n const value = args[i + 1];\n const labelKey = i;\n const valueKey = i + 1;\n if (typeof label !== 'number') {\n return context.error('Input/output pairs for \"step\" expressions must be defined using literal numeric values (not computed expressions) for the input values.', labelKey);\n }\n if (stops.length && stops[stops.length - 1][0] >= label) {\n return context.error('Input/output pairs for \"step\" expressions must be arranged with input values in strictly ascending order.', labelKey);\n }\n const parsed = context.parse(value, valueKey, outputType);\n if (!parsed)\n return null;\n outputType = outputType || parsed.type;\n stops.push([label, parsed]);\n }\n return new Step(outputType, input, stops);\n }\n evaluate(ctx) {\n const labels = this.labels;\n const outputs = this.outputs;\n if (labels.length === 1) {\n return outputs[0].evaluate(ctx);\n }\n const value = this.input.evaluate(ctx);\n if (value <= labels[0]) {\n return outputs[0].evaluate(ctx);\n }\n const stopCount = labels.length;\n if (value >= labels[stopCount - 1]) {\n return outputs[stopCount - 1].evaluate(ctx);\n }\n const index = findStopLessThanOrEqualTo(labels, value);\n return outputs[index].evaluate(ctx);\n }\n eachChild(fn) {\n fn(this.input);\n for (const expression of this.outputs) {\n fn(expression);\n }\n }\n outputDefined() {\n return this.outputs.every(out => out.outputDefined());\n }\n}\n\n/**\n * Checks whether the specified color space is one of the supported interpolation color spaces.\n *\n * @param colorSpace Color space key to verify.\n * @returns `true` if the specified color space is one of the supported\n * interpolation color spaces, `false` otherwise\n */\nfunction isSupportedInterpolationColorSpace(colorSpace) {\n return colorSpace === 'rgb' || colorSpace === 'hcl' || colorSpace === 'lab';\n}\n/**\n * @param interpolationType Interpolation type\n * @returns interpolation fn\n * @deprecated use `interpolate[type]` instead\n */\nconst interpolateFactory = (interpolationType) => {\n switch (interpolationType) {\n case 'number': return number;\n case 'color': return color;\n case 'array': return array;\n case 'padding': return padding;\n case 'variableAnchorOffsetCollection': return variableAnchorOffsetCollection;\n }\n};\nfunction number(from, to, t) {\n return from + t * (to - from);\n}\nfunction color(from, to, t, spaceKey = 'rgb') {\n switch (spaceKey) {\n case 'rgb': {\n const [r, g, b, alpha] = array(from.rgb, to.rgb, t);\n return new Color(r, g, b, alpha, false);\n }\n case 'hcl': {\n const [hue0, chroma0, light0, alphaF] = from.hcl;\n const [hue1, chroma1, light1, alphaT] = to.hcl;\n // https://github.com/gka/chroma.js/blob/cd1b3c0926c7a85cbdc3b1453b3a94006de91a92/src/interpolator/_hsx.js\n let hue, chroma;\n if (!isNaN(hue0) && !isNaN(hue1)) {\n let dh = hue1 - hue0;\n if (hue1 > hue0 && dh > 180) {\n dh -= 360;\n }\n else if (hue1 < hue0 && hue0 - hue1 > 180) {\n dh += 360;\n }\n hue = hue0 + t * dh;\n }\n else if (!isNaN(hue0)) {\n hue = hue0;\n if (light1 === 1 || light1 === 0)\n chroma = chroma0;\n }\n else if (!isNaN(hue1)) {\n hue = hue1;\n if (light0 === 1 || light0 === 0)\n chroma = chroma1;\n }\n else {\n hue = NaN;\n }\n const [r, g, b, alpha] = hclToRgb([\n hue,\n chroma !== null && chroma !== void 0 ? chroma : number(chroma0, chroma1, t),\n number(light0, light1, t),\n number(alphaF, alphaT, t),\n ]);\n return new Color(r, g, b, alpha, false);\n }\n case 'lab': {\n const [r, g, b, alpha] = labToRgb(array(from.lab, to.lab, t));\n return new Color(r, g, b, alpha, false);\n }\n }\n}\nfunction array(from, to, t) {\n return from.map((d, i) => {\n return number(d, to[i], t);\n });\n}\nfunction padding(from, to, t) {\n return new Padding(array(from.values, to.values, t));\n}\nfunction variableAnchorOffsetCollection(from, to, t) {\n const fromValues = from.values;\n const toValues = to.values;\n if (fromValues.length !== toValues.length) {\n throw new RuntimeError(`Cannot interpolate values of different length. from: ${from.toString()}, to: ${to.toString()}`);\n }\n const output = [];\n for (let i = 0; i < fromValues.length; i += 2) {\n // Anchor entries must match\n if (fromValues[i] !== toValues[i]) {\n throw new RuntimeError(`Cannot interpolate values containing mismatched anchors. from[${i}]: ${fromValues[i]}, to[${i}]: ${toValues[i]}`);\n }\n output.push(fromValues[i]);\n // Interpolate the offset values for each anchor\n const [fx, fy] = fromValues[i + 1];\n const [tx, ty] = toValues[i + 1];\n output.push([number(fx, tx, t), number(fy, ty, t)]);\n }\n return new VariableAnchorOffsetCollection(output);\n}\nconst interpolate = {\n number,\n color,\n array,\n padding,\n variableAnchorOffsetCollection\n};\n\nclass Interpolate {\n constructor(type, operator, interpolation, input, stops) {\n this.type = type;\n this.operator = operator;\n this.interpolation = interpolation;\n this.input = input;\n this.labels = [];\n this.outputs = [];\n for (const [label, expression] of stops) {\n this.labels.push(label);\n this.outputs.push(expression);\n }\n }\n static interpolationFactor(interpolation, input, lower, upper) {\n let t = 0;\n if (interpolation.name === 'exponential') {\n t = exponentialInterpolation(input, interpolation.base, lower, upper);\n }\n else if (interpolation.name === 'linear') {\n t = exponentialInterpolation(input, 1, lower, upper);\n }\n else if (interpolation.name === 'cubic-bezier') {\n const c = interpolation.controlPoints;\n const ub = new UnitBezier(c[0], c[1], c[2], c[3]);\n t = ub.solve(exponentialInterpolation(input, 1, lower, upper));\n }\n return t;\n }\n static parse(args, context) {\n let [operator, interpolation, input, ...rest] = args;\n if (!Array.isArray(interpolation) || interpolation.length === 0) {\n return context.error('Expected an interpolation type expression.', 1);\n }\n if (interpolation[0] === 'linear') {\n interpolation = { name: 'linear' };\n }\n else if (interpolation[0] === 'exponential') {\n const base = interpolation[1];\n if (typeof base !== 'number')\n return context.error('Exponential interpolation requires a numeric base.', 1, 1);\n interpolation = {\n name: 'exponential',\n base\n };\n }\n else if (interpolation[0] === 'cubic-bezier') {\n const controlPoints = interpolation.slice(1);\n if (controlPoints.length !== 4 ||\n controlPoints.some(t => typeof t !== 'number' || t < 0 || t > 1)) {\n return context.error('Cubic bezier interpolation requires four numeric arguments with values between 0 and 1.', 1);\n }\n interpolation = {\n name: 'cubic-bezier',\n controlPoints: controlPoints\n };\n }\n else {\n return context.error(`Unknown interpolation type ${String(interpolation[0])}`, 1, 0);\n }\n if (args.length - 1 < 4) {\n return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`);\n }\n if ((args.length - 1) % 2 !== 0) {\n return context.error('Expected an even number of arguments.');\n }\n input = context.parse(input, 2, NumberType);\n if (!input)\n return null;\n const stops = [];\n let outputType = null;\n if (operator === 'interpolate-hcl' || operator === 'interpolate-lab') {\n outputType = ColorType;\n }\n else if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n for (let i = 0; i < rest.length; i += 2) {\n const label = rest[i];\n const value = rest[i + 1];\n const labelKey = i + 3;\n const valueKey = i + 4;\n if (typeof label !== 'number') {\n return context.error('Input/output pairs for \"interpolate\" expressions must be defined using literal numeric values (not computed expressions) for the input values.', labelKey);\n }\n if (stops.length && stops[stops.length - 1][0] >= label) {\n return context.error('Input/output pairs for \"interpolate\" expressions must be arranged with input values in strictly ascending order.', labelKey);\n }\n const parsed = context.parse(value, valueKey, outputType);\n if (!parsed)\n return null;\n outputType = outputType || parsed.type;\n stops.push([label, parsed]);\n }\n if (!verifyType(outputType, NumberType) &&\n !verifyType(outputType, ColorType) &&\n !verifyType(outputType, PaddingType) &&\n !verifyType(outputType, VariableAnchorOffsetCollectionType) &&\n !verifyType(outputType, array$1(NumberType))) {\n return context.error(`Type ${toString$1(outputType)} is not interpolatable.`);\n }\n return new Interpolate(outputType, operator, interpolation, input, stops);\n }\n evaluate(ctx) {\n const labels = this.labels;\n const outputs = this.outputs;\n if (labels.length === 1) {\n return outputs[0].evaluate(ctx);\n }\n const value = this.input.evaluate(ctx);\n if (value <= labels[0]) {\n return outputs[0].evaluate(ctx);\n }\n const stopCount = labels.length;\n if (value >= labels[stopCount - 1]) {\n return outputs[stopCount - 1].evaluate(ctx);\n }\n const index = findStopLessThanOrEqualTo(labels, value);\n const lower = labels[index];\n const upper = labels[index + 1];\n const t = Interpolate.interpolationFactor(this.interpolation, value, lower, upper);\n const outputLower = outputs[index].evaluate(ctx);\n const outputUpper = outputs[index + 1].evaluate(ctx);\n switch (this.operator) {\n case 'interpolate':\n return interpolate[this.type.kind](outputLower, outputUpper, t);\n case 'interpolate-hcl':\n return interpolate.color(outputLower, outputUpper, t, 'hcl');\n case 'interpolate-lab':\n return interpolate.color(outputLower, outputUpper, t, 'lab');\n }\n }\n eachChild(fn) {\n fn(this.input);\n for (const expression of this.outputs) {\n fn(expression);\n }\n }\n outputDefined() {\n return this.outputs.every(out => out.outputDefined());\n }\n}\n/**\n * Returns a ratio that can be used to interpolate between exponential function\n * stops.\n * How it works: Two consecutive stop values define a (scaled and shifted) exponential function `f(x) = a * base^x + b`, where `base` is the user-specified base,\n * and `a` and `b` are constants affording sufficient degrees of freedom to fit\n * the function to the given stops.\n *\n * Here's a bit of algebra that lets us compute `f(x)` directly from the stop\n * values without explicitly solving for `a` and `b`:\n *\n * First stop value: `f(x0) = y0 = a * base^x0 + b`\n * Second stop value: `f(x1) = y1 = a * base^x1 + b`\n * => `y1 - y0 = a(base^x1 - base^x0)`\n * => `a = (y1 - y0)/(base^x1 - base^x0)`\n *\n * Desired value: `f(x) = y = a * base^x + b`\n * => `f(x) = y0 + a * (base^x - base^x0)`\n *\n * From the above, we can replace the `a` in `a * (base^x - base^x0)` and do a\n * little algebra:\n * ```\n * a * (base^x - base^x0) = (y1 - y0)/(base^x1 - base^x0) * (base^x - base^x0)\n * = (y1 - y0) * (base^x - base^x0) / (base^x1 - base^x0)\n * ```\n *\n * If we let `(base^x - base^x0) / (base^x1 base^x0)`, then we have\n * `f(x) = y0 + (y1 - y0) * ratio`. In other words, `ratio` may be treated as\n * an interpolation factor between the two stops' output values.\n *\n * (Note: a slightly different form for `ratio`,\n * `(base^(x-x0) - 1) / (base^(x1-x0) - 1) `, is equivalent, but requires fewer\n * expensive `Math.pow()` operations.)\n *\n * @private\n*/\nfunction exponentialInterpolation(input, base, lowerValue, upperValue) {\n const difference = upperValue - lowerValue;\n const progress = input - lowerValue;\n if (difference === 0) {\n return 0;\n }\n else if (base === 1) {\n return progress / difference;\n }\n else {\n return (Math.pow(base, progress) - 1) / (Math.pow(base, difference) - 1);\n }\n}\n\nclass Coalesce {\n constructor(type, args) {\n this.type = type;\n this.args = args;\n }\n static parse(args, context) {\n if (args.length < 2) {\n return context.error('Expectected at least one argument.');\n }\n let outputType = null;\n const expectedType = context.expectedType;\n if (expectedType && expectedType.kind !== 'value') {\n outputType = expectedType;\n }\n const parsedArgs = [];\n for (const arg of args.slice(1)) {\n const parsed = context.parse(arg, 1 + parsedArgs.length, outputType, undefined, { typeAnnotation: 'omit' });\n if (!parsed)\n return null;\n outputType = outputType || parsed.type;\n parsedArgs.push(parsed);\n }\n if (!outputType)\n throw new Error('No output type');\n // Above, we parse arguments without inferred type annotation so that\n // they don't produce a runtime error for `null` input, which would\n // preempt the desired null-coalescing behavior.\n // Thus, if any of our arguments would have needed an annotation, we\n // need to wrap the enclosing coalesce expression with it instead.\n const needsAnnotation = expectedType &&\n parsedArgs.some(arg => checkSubtype(expectedType, arg.type));\n return needsAnnotation ?\n new Coalesce(ValueType, parsedArgs) :\n new Coalesce(outputType, parsedArgs);\n }\n evaluate(ctx) {\n let result = null;\n let argCount = 0;\n let requestedImageName;\n for (const arg of this.args) {\n argCount++;\n result = arg.evaluate(ctx);\n // we need to keep track of the first requested image in a coalesce statement\n // if coalesce can't find a valid image, we return the first image name so styleimagemissing can fire\n if (result && result instanceof ResolvedImage && !result.available) {\n if (!requestedImageName) {\n requestedImageName = result.name;\n }\n result = null;\n if (argCount === this.args.length) {\n result = requestedImageName;\n }\n }\n if (result !== null)\n break;\n }\n return result;\n }\n eachChild(fn) {\n this.args.forEach(fn);\n }\n outputDefined() {\n return this.args.every(arg => arg.outputDefined());\n }\n}\n\nclass Let {\n constructor(bindings, result) {\n this.type = result.type;\n this.bindings = [].concat(bindings);\n this.result = result;\n }\n evaluate(ctx) {\n return this.result.evaluate(ctx);\n }\n eachChild(fn) {\n for (const binding of this.bindings) {\n fn(binding[1]);\n }\n fn(this.result);\n }\n static parse(args, context) {\n if (args.length < 4)\n return context.error(`Expected at least 3 arguments, but found ${args.length - 1} instead.`);\n const bindings = [];\n for (let i = 1; i < args.length - 1; i += 2) {\n const name = args[i];\n if (typeof name !== 'string') {\n return context.error(`Expected string, but found ${typeof name} instead.`, i);\n }\n if (/[^a-zA-Z0-9_]/.test(name)) {\n return context.error('Variable names must contain only alphanumeric characters or \\'_\\'.', i);\n }\n const value = context.parse(args[i + 1], i + 1);\n if (!value)\n return null;\n bindings.push([name, value]);\n }\n const result = context.parse(args[args.length - 1], args.length - 1, context.expectedType, bindings);\n if (!result)\n return null;\n return new Let(bindings, result);\n }\n outputDefined() {\n return this.result.outputDefined();\n }\n}\n\nclass At {\n constructor(type, index, input) {\n this.type = type;\n this.index = index;\n this.input = input;\n }\n static parse(args, context) {\n if (args.length !== 3)\n return context.error(`Expected 2 arguments, but found ${args.length - 1} instead.`);\n const index = context.parse(args[1], 1, NumberType);\n const input = context.parse(args[2], 2, array$1(context.expectedType || ValueType));\n if (!index || !input)\n return null;\n const t = input.type;\n return new At(t.itemType, index, input);\n }\n evaluate(ctx) {\n const index = this.index.evaluate(ctx);\n const array = this.input.evaluate(ctx);\n if (index < 0) {\n throw new RuntimeError(`Array index out of bounds: ${index} < 0.`);\n }\n if (index >= array.length) {\n throw new RuntimeError(`Array index out of bounds: ${index} > ${array.length - 1}.`);\n }\n if (index !== Math.floor(index)) {\n throw new RuntimeError(`Array index must be an integer, but found ${index} instead.`);\n }\n return array[index];\n }\n eachChild(fn) {\n fn(this.index);\n fn(this.input);\n }\n outputDefined() {\n return false;\n }\n}\n\nclass In {\n constructor(needle, haystack) {\n this.type = BooleanType;\n this.needle = needle;\n this.haystack = haystack;\n }\n static parse(args, context) {\n if (args.length !== 3) {\n return context.error(`Expected 2 arguments, but found ${args.length - 1} instead.`);\n }\n const needle = context.parse(args[1], 1, ValueType);\n const haystack = context.parse(args[2], 2, ValueType);\n if (!needle || !haystack)\n return null;\n if (!isValidType(needle.type, [BooleanType, StringType, NumberType, NullType, ValueType])) {\n return context.error(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(needle.type)} instead`);\n }\n return new In(needle, haystack);\n }\n evaluate(ctx) {\n const needle = this.needle.evaluate(ctx);\n const haystack = this.haystack.evaluate(ctx);\n if (!haystack)\n return false;\n if (!isValidNativeType(needle, ['boolean', 'string', 'number', 'null'])) {\n throw new RuntimeError(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(typeOf(needle))} instead.`);\n }\n if (!isValidNativeType(haystack, ['string', 'array'])) {\n throw new RuntimeError(`Expected second argument to be of type array or string, but found ${toString$1(typeOf(haystack))} instead.`);\n }\n return haystack.indexOf(needle) >= 0;\n }\n eachChild(fn) {\n fn(this.needle);\n fn(this.haystack);\n }\n outputDefined() {\n return true;\n }\n}\n\nclass IndexOf {\n constructor(needle, haystack, fromIndex) {\n this.type = NumberType;\n this.needle = needle;\n this.haystack = haystack;\n this.fromIndex = fromIndex;\n }\n static parse(args, context) {\n if (args.length <= 2 || args.length >= 5) {\n return context.error(`Expected 3 or 4 arguments, but found ${args.length - 1} instead.`);\n }\n const needle = context.parse(args[1], 1, ValueType);\n const haystack = context.parse(args[2], 2, ValueType);\n if (!needle || !haystack)\n return null;\n if (!isValidType(needle.type, [BooleanType, StringType, NumberType, NullType, ValueType])) {\n return context.error(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(needle.type)} instead`);\n }\n if (args.length === 4) {\n const fromIndex = context.parse(args[3], 3, NumberType);\n if (!fromIndex)\n return null;\n return new IndexOf(needle, haystack, fromIndex);\n }\n else {\n return new IndexOf(needle, haystack);\n }\n }\n evaluate(ctx) {\n const needle = this.needle.evaluate(ctx);\n const haystack = this.haystack.evaluate(ctx);\n if (!isValidNativeType(needle, ['boolean', 'string', 'number', 'null'])) {\n throw new RuntimeError(`Expected first argument to be of type boolean, string, number or null, but found ${toString$1(typeOf(needle))} instead.`);\n }\n if (!isValidNativeType(haystack, ['string', 'array'])) {\n throw new RuntimeError(`Expected second argument to be of type array or string, but found ${toString$1(typeOf(haystack))} instead.`);\n }\n if (this.fromIndex) {\n const fromIndex = this.fromIndex.evaluate(ctx);\n return haystack.indexOf(needle, fromIndex);\n }\n return haystack.indexOf(needle);\n }\n eachChild(fn) {\n fn(this.needle);\n fn(this.haystack);\n if (this.fromIndex) {\n fn(this.fromIndex);\n }\n }\n outputDefined() {\n return false;\n }\n}\n\nclass Match {\n constructor(inputType, outputType, input, cases, outputs, otherwise) {\n this.inputType = inputType;\n this.type = outputType;\n this.input = input;\n this.cases = cases;\n this.outputs = outputs;\n this.otherwise = otherwise;\n }\n static parse(args, context) {\n if (args.length < 5)\n return context.error(`Expected at least 4 arguments, but found only ${args.length - 1}.`);\n if (args.length % 2 !== 1)\n return context.error('Expected an even number of arguments.');\n let inputType;\n let outputType;\n if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n const cases = {};\n const outputs = [];\n for (let i = 2; i < args.length - 1; i += 2) {\n let labels = args[i];\n const value = args[i + 1];\n if (!Array.isArray(labels)) {\n labels = [labels];\n }\n const labelContext = context.concat(i);\n if (labels.length === 0) {\n return labelContext.error('Expected at least one branch label.');\n }\n for (const label of labels) {\n if (typeof label !== 'number' && typeof label !== 'string') {\n return labelContext.error('Branch labels must be numbers or strings.');\n }\n else if (typeof label === 'number' && Math.abs(label) > Number.MAX_SAFE_INTEGER) {\n return labelContext.error(`Branch labels must be integers no larger than ${Number.MAX_SAFE_INTEGER}.`);\n }\n else if (typeof label === 'number' && Math.floor(label) !== label) {\n return labelContext.error('Numeric branch labels must be integer values.');\n }\n else if (!inputType) {\n inputType = typeOf(label);\n }\n else if (labelContext.checkSubtype(inputType, typeOf(label))) {\n return null;\n }\n if (typeof cases[String(label)] !== 'undefined') {\n return labelContext.error('Branch labels must be unique.');\n }\n cases[String(label)] = outputs.length;\n }\n const result = context.parse(value, i, outputType);\n if (!result)\n return null;\n outputType = outputType || result.type;\n outputs.push(result);\n }\n const input = context.parse(args[1], 1, ValueType);\n if (!input)\n return null;\n const otherwise = context.parse(args[args.length - 1], args.length - 1, outputType);\n if (!otherwise)\n return null;\n if (input.type.kind !== 'value' && context.concat(1).checkSubtype(inputType, input.type)) {\n return null;\n }\n return new Match(inputType, outputType, input, cases, outputs, otherwise);\n }\n evaluate(ctx) {\n const input = this.input.evaluate(ctx);\n const output = (typeOf(input) === this.inputType && this.outputs[this.cases[input]]) || this.otherwise;\n return output.evaluate(ctx);\n }\n eachChild(fn) {\n fn(this.input);\n this.outputs.forEach(fn);\n fn(this.otherwise);\n }\n outputDefined() {\n return this.outputs.every(out => out.outputDefined()) && this.otherwise.outputDefined();\n }\n}\n\nclass Case {\n constructor(type, branches, otherwise) {\n this.type = type;\n this.branches = branches;\n this.otherwise = otherwise;\n }\n static parse(args, context) {\n if (args.length < 4)\n return context.error(`Expected at least 3 arguments, but found only ${args.length - 1}.`);\n if (args.length % 2 !== 0)\n return context.error('Expected an odd number of arguments.');\n let outputType;\n if (context.expectedType && context.expectedType.kind !== 'value') {\n outputType = context.expectedType;\n }\n const branches = [];\n for (let i = 1; i < args.length - 1; i += 2) {\n const test = context.parse(args[i], i, BooleanType);\n if (!test)\n return null;\n const result = context.parse(args[i + 1], i + 1, outputType);\n if (!result)\n return null;\n branches.push([test, result]);\n outputType = outputType || result.type;\n }\n const otherwise = context.parse(args[args.length - 1], args.length - 1, outputType);\n if (!otherwise)\n return null;\n if (!outputType)\n throw new Error('Can\\'t infer output type');\n return new Case(outputType, branches, otherwise);\n }\n evaluate(ctx) {\n for (const [test, expression] of this.branches) {\n if (test.evaluate(ctx)) {\n return expression.evaluate(ctx);\n }\n }\n return this.otherwise.evaluate(ctx);\n }\n eachChild(fn) {\n for (const [test, expression] of this.branches) {\n fn(test);\n fn(expression);\n }\n fn(this.otherwise);\n }\n outputDefined() {\n return this.branches.every(([_, out]) => out.outputDefined()) && this.otherwise.outputDefined();\n }\n}\n\nclass Slice {\n constructor(type, input, beginIndex, endIndex) {\n this.type = type;\n this.input = input;\n this.beginIndex = beginIndex;\n this.endIndex = endIndex;\n }\n static parse(args, context) {\n if (args.length <= 2 || args.length >= 5) {\n return context.error(`Expected 3 or 4 arguments, but found ${args.length - 1} instead.`);\n }\n const input = context.parse(args[1], 1, ValueType);\n const beginIndex = context.parse(args[2], 2, NumberType);\n if (!input || !beginIndex)\n return null;\n if (!isValidType(input.type, [array$1(ValueType), StringType, ValueType])) {\n return context.error(`Expected first argument to be of type array or string, but found ${toString$1(input.type)} instead`);\n }\n if (args.length === 4) {\n const endIndex = context.parse(args[3], 3, NumberType);\n if (!endIndex)\n return null;\n return new Slice(input.type, input, beginIndex, endIndex);\n }\n else {\n return new Slice(input.type, input, beginIndex);\n }\n }\n evaluate(ctx) {\n const input = this.input.evaluate(ctx);\n const beginIndex = this.beginIndex.evaluate(ctx);\n if (!isValidNativeType(input, ['string', 'array'])) {\n throw new RuntimeError(`Expected first argument to be of type array or string, but found ${toString$1(typeOf(input))} instead.`);\n }\n if (this.endIndex) {\n const endIndex = this.endIndex.evaluate(ctx);\n return input.slice(beginIndex, endIndex);\n }\n return input.slice(beginIndex);\n }\n eachChild(fn) {\n fn(this.input);\n fn(this.beginIndex);\n if (this.endIndex) {\n fn(this.endIndex);\n }\n }\n outputDefined() {\n return false;\n }\n}\n\nfunction isComparableType(op, type) {\n if (op === '==' || op === '!=') {\n // equality operator\n return type.kind === 'boolean' ||\n type.kind === 'string' ||\n type.kind === 'number' ||\n type.kind === 'null' ||\n type.kind === 'value';\n }\n else {\n // ordering operator\n return type.kind === 'string' ||\n type.kind === 'number' ||\n type.kind === 'value';\n }\n}\nfunction eq(ctx, a, b) { return a === b; }\nfunction neq(ctx, a, b) { return a !== b; }\nfunction lt(ctx, a, b) { return a < b; }\nfunction gt(ctx, a, b) { return a > b; }\nfunction lteq(ctx, a, b) { return a <= b; }\nfunction gteq(ctx, a, b) { return a >= b; }\nfunction eqCollate(ctx, a, b, c) { return c.compare(a, b) === 0; }\nfunction neqCollate(ctx, a, b, c) { return !eqCollate(ctx, a, b, c); }\nfunction ltCollate(ctx, a, b, c) { return c.compare(a, b) < 0; }\nfunction gtCollate(ctx, a, b, c) { return c.compare(a, b) > 0; }\nfunction lteqCollate(ctx, a, b, c) { return c.compare(a, b) <= 0; }\nfunction gteqCollate(ctx, a, b, c) { return c.compare(a, b) >= 0; }\n/**\n * Special form for comparison operators, implementing the signatures:\n * - (T, T, ?Collator) => boolean\n * - (T, value, ?Collator) => boolean\n * - (value, T, ?Collator) => boolean\n *\n * For inequalities, T must be either value, string, or number. For ==/!=, it\n * can also be boolean or null.\n *\n * Equality semantics are equivalent to Javascript's strict equality (===/!==)\n * -- i.e., when the arguments' types don't match, == evaluates to false, != to\n * true.\n *\n * When types don't match in an ordering comparison, a runtime error is thrown.\n *\n * @private\n */\nfunction makeComparison(op, compareBasic, compareWithCollator) {\n const isOrderComparison = op !== '==' && op !== '!=';\n return class Comparison {\n constructor(lhs, rhs, collator) {\n this.type = BooleanType;\n this.lhs = lhs;\n this.rhs = rhs;\n this.collator = collator;\n this.hasUntypedArgument = lhs.type.kind === 'value' || rhs.type.kind === 'value';\n }\n static parse(args, context) {\n if (args.length !== 3 && args.length !== 4)\n return context.error('Expected two or three arguments.');\n const op = args[0];\n let lhs = context.parse(args[1], 1, ValueType);\n if (!lhs)\n return null;\n if (!isComparableType(op, lhs.type)) {\n return context.concat(1).error(`\"${op}\" comparisons are not supported for type '${toString$1(lhs.type)}'.`);\n }\n let rhs = context.parse(args[2], 2, ValueType);\n if (!rhs)\n return null;\n if (!isComparableType(op, rhs.type)) {\n return context.concat(2).error(`\"${op}\" comparisons are not supported for type '${toString$1(rhs.type)}'.`);\n }\n if (lhs.type.kind !== rhs.type.kind &&\n lhs.type.kind !== 'value' &&\n rhs.type.kind !== 'value') {\n return context.error(`Cannot compare types '${toString$1(lhs.type)}' and '${toString$1(rhs.type)}'.`);\n }\n if (isOrderComparison) {\n // typing rules specific to less/greater than operators\n if (lhs.type.kind === 'value' && rhs.type.kind !== 'value') {\n // (value, T)\n lhs = new Assertion(rhs.type, [lhs]);\n }\n else if (lhs.type.kind !== 'value' && rhs.type.kind === 'value') {\n // (T, value)\n rhs = new Assertion(lhs.type, [rhs]);\n }\n }\n let collator = null;\n if (args.length === 4) {\n if (lhs.type.kind !== 'string' &&\n rhs.type.kind !== 'string' &&\n lhs.type.kind !== 'value' &&\n rhs.type.kind !== 'value') {\n return context.error('Cannot use collator to compare non-string types.');\n }\n collator = context.parse(args[3], 3, CollatorType);\n if (!collator)\n return null;\n }\n return new Comparison(lhs, rhs, collator);\n }\n evaluate(ctx) {\n const lhs = this.lhs.evaluate(ctx);\n const rhs = this.rhs.evaluate(ctx);\n if (isOrderComparison && this.hasUntypedArgument) {\n const lt = typeOf(lhs);\n const rt = typeOf(rhs);\n // check that type is string or number, and equal\n if (lt.kind !== rt.kind || !(lt.kind === 'string' || lt.kind === 'number')) {\n throw new RuntimeError(`Expected arguments for \"${op}\" to be (string, string) or (number, number), but found (${lt.kind}, ${rt.kind}) instead.`);\n }\n }\n if (this.collator && !isOrderComparison && this.hasUntypedArgument) {\n const lt = typeOf(lhs);\n const rt = typeOf(rhs);\n if (lt.kind !== 'string' || rt.kind !== 'string') {\n return compareBasic(ctx, lhs, rhs);\n }\n }\n return this.collator ?\n compareWithCollator(ctx, lhs, rhs, this.collator.evaluate(ctx)) :\n compareBasic(ctx, lhs, rhs);\n }\n eachChild(fn) {\n fn(this.lhs);\n fn(this.rhs);\n if (this.collator) {\n fn(this.collator);\n }\n }\n outputDefined() {\n return true;\n }\n };\n}\nconst Equals = makeComparison('==', eq, eqCollate);\nconst NotEquals = makeComparison('!=', neq, neqCollate);\nconst LessThan = makeComparison('<', lt, ltCollate);\nconst GreaterThan = makeComparison('>', gt, gtCollate);\nconst LessThanOrEqual = makeComparison('<=', lteq, lteqCollate);\nconst GreaterThanOrEqual = makeComparison('>=', gteq, gteqCollate);\n\nclass NumberFormat {\n constructor(number, locale, currency, minFractionDigits, maxFractionDigits) {\n this.type = StringType;\n this.number = number;\n this.locale = locale;\n this.currency = currency;\n this.minFractionDigits = minFractionDigits;\n this.maxFractionDigits = maxFractionDigits;\n }\n static parse(args, context) {\n if (args.length !== 3)\n return context.error('Expected two arguments.');\n const number = context.parse(args[1], 1, NumberType);\n if (!number)\n return null;\n const options = args[2];\n if (typeof options !== 'object' || Array.isArray(options))\n return context.error('NumberFormat options argument must be an object.');\n let locale = null;\n if (options['locale']) {\n locale = context.parse(options['locale'], 1, StringType);\n if (!locale)\n return null;\n }\n let currency = null;\n if (options['currency']) {\n currency = context.parse(options['currency'], 1, StringType);\n if (!currency)\n return null;\n }\n let minFractionDigits = null;\n if (options['min-fraction-digits']) {\n minFractionDigits = context.parse(options['min-fraction-digits'], 1, NumberType);\n if (!minFractionDigits)\n return null;\n }\n let maxFractionDigits = null;\n if (options['max-fraction-digits']) {\n maxFractionDigits = context.parse(options['max-fraction-digits'], 1, NumberType);\n if (!maxFractionDigits)\n return null;\n }\n return new NumberFormat(number, locale, currency, minFractionDigits, maxFractionDigits);\n }\n evaluate(ctx) {\n return new Intl.NumberFormat(this.locale ? this.locale.evaluate(ctx) : [], {\n style: this.currency ? 'currency' : 'decimal',\n currency: this.currency ? this.currency.evaluate(ctx) : undefined,\n minimumFractionDigits: this.minFractionDigits ? this.minFractionDigits.evaluate(ctx) : undefined,\n maximumFractionDigits: this.maxFractionDigits ? this.maxFractionDigits.evaluate(ctx) : undefined,\n }).format(this.number.evaluate(ctx));\n }\n eachChild(fn) {\n fn(this.number);\n if (this.locale) {\n fn(this.locale);\n }\n if (this.currency) {\n fn(this.currency);\n }\n if (this.minFractionDigits) {\n fn(this.minFractionDigits);\n }\n if (this.maxFractionDigits) {\n fn(this.maxFractionDigits);\n }\n }\n outputDefined() {\n return false;\n }\n}\n\nclass FormatExpression {\n constructor(sections) {\n this.type = FormattedType;\n this.sections = sections;\n }\n static parse(args, context) {\n if (args.length < 2) {\n return context.error('Expected at least one argument.');\n }\n const firstArg = args[1];\n if (!Array.isArray(firstArg) && typeof firstArg === 'object') {\n return context.error('First argument must be an image or text section.');\n }\n const sections = [];\n let nextTokenMayBeObject = false;\n for (let i = 1; i <= args.length - 1; ++i) {\n const arg = args[i];\n if (nextTokenMayBeObject && typeof arg === 'object' && !Array.isArray(arg)) {\n nextTokenMayBeObject = false;\n let scale = null;\n if (arg['font-scale']) {\n scale = context.parse(arg['font-scale'], 1, NumberType);\n if (!scale)\n return null;\n }\n let font = null;\n if (arg['text-font']) {\n font = context.parse(arg['text-font'], 1, array$1(StringType));\n if (!font)\n return null;\n }\n let textColor = null;\n if (arg['text-color']) {\n textColor = context.parse(arg['text-color'], 1, ColorType);\n if (!textColor)\n return null;\n }\n const lastExpression = sections[sections.length - 1];\n lastExpression.scale = scale;\n lastExpression.font = font;\n lastExpression.textColor = textColor;\n }\n else {\n const content = context.parse(args[i], 1, ValueType);\n if (!content)\n return null;\n const kind = content.type.kind;\n if (kind !== 'string' && kind !== 'value' && kind !== 'null' && kind !== 'resolvedImage')\n return context.error('Formatted text type must be \\'string\\', \\'value\\', \\'image\\' or \\'null\\'.');\n nextTokenMayBeObject = true;\n sections.push({ content, scale: null, font: null, textColor: null });\n }\n }\n return new FormatExpression(sections);\n }\n evaluate(ctx) {\n const evaluateSection = section => {\n const evaluatedContent = section.content.evaluate(ctx);\n if (typeOf(evaluatedContent) === ResolvedImageType) {\n return new FormattedSection('', evaluatedContent, null, null, null);\n }\n return new FormattedSection(toString(evaluatedContent), null, section.scale ? section.scale.evaluate(ctx) : null, section.font ? section.font.evaluate(ctx).join(',') : null, section.textColor ? section.textColor.evaluate(ctx) : null);\n };\n return new Formatted(this.sections.map(evaluateSection));\n }\n eachChild(fn) {\n for (const section of this.sections) {\n fn(section.content);\n if (section.scale) {\n fn(section.scale);\n }\n if (section.font) {\n fn(section.font);\n }\n if (section.textColor) {\n fn(section.textColor);\n }\n }\n }\n outputDefined() {\n // Technically the combinatoric set of all children\n // Usually, this.text will be undefined anyway\n return false;\n }\n}\n\nclass ImageExpression {\n constructor(input) {\n this.type = ResolvedImageType;\n this.input = input;\n }\n static parse(args, context) {\n if (args.length !== 2) {\n return context.error('Expected two arguments.');\n }\n const name = context.parse(args[1], 1, StringType);\n if (!name)\n return context.error('No image name provided.');\n return new ImageExpression(name);\n }\n evaluate(ctx) {\n const evaluatedImageName = this.input.evaluate(ctx);\n const value = ResolvedImage.fromString(evaluatedImageName);\n if (value && ctx.availableImages)\n value.available = ctx.availableImages.indexOf(evaluatedImageName) > -1;\n return value;\n }\n eachChild(fn) {\n fn(this.input);\n }\n outputDefined() {\n // The output of image is determined by the list of available images in the evaluation context\n return false;\n }\n}\n\nclass Length {\n constructor(input) {\n this.type = NumberType;\n this.input = input;\n }\n static parse(args, context) {\n if (args.length !== 2)\n return context.error(`Expected 1 argument, but found ${args.length - 1} instead.`);\n const input = context.parse(args[1], 1);\n if (!input)\n return null;\n if (input.type.kind !== 'array' && input.type.kind !== 'string' && input.type.kind !== 'value')\n return context.error(`Expected argument of type string or array, but found ${toString$1(input.type)} instead.`);\n return new Length(input);\n }\n evaluate(ctx) {\n const input = this.input.evaluate(ctx);\n if (typeof input === 'string') {\n return input.length;\n }\n else if (Array.isArray(input)) {\n return input.length;\n }\n else {\n throw new RuntimeError(`Expected value to be of type string or array, but found ${toString$1(typeOf(input))} instead.`);\n }\n }\n eachChild(fn) {\n fn(this.input);\n }\n outputDefined() {\n return false;\n }\n}\n\nconst expressions = {\n // special forms\n '==': Equals,\n '!=': NotEquals,\n '>': GreaterThan,\n '<': LessThan,\n '>=': GreaterThanOrEqual,\n '<=': LessThanOrEqual,\n 'array': Assertion,\n 'at': At,\n 'boolean': Assertion,\n 'case': Case,\n 'coalesce': Coalesce,\n 'collator': CollatorExpression,\n 'format': FormatExpression,\n 'image': ImageExpression,\n 'in': In,\n 'index-of': IndexOf,\n 'interpolate': Interpolate,\n 'interpolate-hcl': Interpolate,\n 'interpolate-lab': Interpolate,\n 'length': Length,\n 'let': Let,\n 'literal': Literal,\n 'match': Match,\n 'number': Assertion,\n 'number-format': NumberFormat,\n 'object': Assertion,\n 'slice': Slice,\n 'step': Step,\n 'string': Assertion,\n 'to-boolean': Coercion,\n 'to-color': Coercion,\n 'to-number': Coercion,\n 'to-string': Coercion,\n 'var': Var,\n 'within': Within\n};\nfunction rgba(ctx, [r, g, b, a]) {\n r = r.evaluate(ctx);\n g = g.evaluate(ctx);\n b = b.evaluate(ctx);\n const alpha = a ? a.evaluate(ctx) : 1;\n const error = validateRGBA(r, g, b, alpha);\n if (error)\n throw new RuntimeError(error);\n return new Color(r / 255, g / 255, b / 255, alpha, false);\n}\nfunction has(key, obj) {\n return key in obj;\n}\nfunction get(key, obj) {\n const v = obj[key];\n return typeof v === 'undefined' ? null : v;\n}\nfunction binarySearch(v, a, i, j) {\n while (i <= j) {\n const m = (i + j) >> 1;\n if (a[m] === v)\n return true;\n if (a[m] > v)\n j = m - 1;\n else\n i = m + 1;\n }\n return false;\n}\nfunction varargs(type) {\n return { type };\n}\nCompoundExpression.register(expressions, {\n 'error': [\n ErrorType,\n [StringType],\n (ctx, [v]) => { throw new RuntimeError(v.evaluate(ctx)); }\n ],\n 'typeof': [\n StringType,\n [ValueType],\n (ctx, [v]) => toString$1(typeOf(v.evaluate(ctx)))\n ],\n 'to-rgba': [\n array$1(NumberType, 4),\n [ColorType],\n (ctx, [v]) => {\n const [r, g, b, a] = v.evaluate(ctx).rgb;\n return [r * 255, g * 255, b * 255, a];\n },\n ],\n 'rgb': [\n ColorType,\n [NumberType, NumberType, NumberType],\n rgba\n ],\n 'rgba': [\n ColorType,\n [NumberType, NumberType, NumberType, NumberType],\n rgba\n ],\n 'has': {\n type: BooleanType,\n overloads: [\n [\n [StringType],\n (ctx, [key]) => has(key.evaluate(ctx), ctx.properties())\n ], [\n [StringType, ObjectType],\n (ctx, [key, obj]) => has(key.evaluate(ctx), obj.evaluate(ctx))\n ]\n ]\n },\n 'get': {\n type: ValueType,\n overloads: [\n [\n [StringType],\n (ctx, [key]) => get(key.evaluate(ctx), ctx.properties())\n ], [\n [StringType, ObjectType],\n (ctx, [key, obj]) => get(key.evaluate(ctx), obj.evaluate(ctx))\n ]\n ]\n },\n 'feature-state': [\n ValueType,\n [StringType],\n (ctx, [key]) => get(key.evaluate(ctx), ctx.featureState || {})\n ],\n 'properties': [\n ObjectType,\n [],\n (ctx) => ctx.properties()\n ],\n 'geometry-type': [\n StringType,\n [],\n (ctx) => ctx.geometryType()\n ],\n 'id': [\n ValueType,\n [],\n (ctx) => ctx.id()\n ],\n 'zoom': [\n NumberType,\n [],\n (ctx) => ctx.globals.zoom\n ],\n 'heatmap-density': [\n NumberType,\n [],\n (ctx) => ctx.globals.heatmapDensity || 0\n ],\n 'line-progress': [\n NumberType,\n [],\n (ctx) => ctx.globals.lineProgress || 0\n ],\n 'accumulated': [\n ValueType,\n [],\n (ctx) => ctx.globals.accumulated === undefined ? null : ctx.globals.accumulated\n ],\n '+': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => {\n let result = 0;\n for (const arg of args) {\n result += arg.evaluate(ctx);\n }\n return result;\n }\n ],\n '*': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => {\n let result = 1;\n for (const arg of args) {\n result *= arg.evaluate(ctx);\n }\n return result;\n }\n ],\n '-': {\n type: NumberType,\n overloads: [\n [\n [NumberType, NumberType],\n (ctx, [a, b]) => a.evaluate(ctx) - b.evaluate(ctx)\n ], [\n [NumberType],\n (ctx, [a]) => -a.evaluate(ctx)\n ]\n ]\n },\n '/': [\n NumberType,\n [NumberType, NumberType],\n (ctx, [a, b]) => a.evaluate(ctx) / b.evaluate(ctx)\n ],\n '%': [\n NumberType,\n [NumberType, NumberType],\n (ctx, [a, b]) => a.evaluate(ctx) % b.evaluate(ctx)\n ],\n 'ln2': [\n NumberType,\n [],\n () => Math.LN2\n ],\n 'pi': [\n NumberType,\n [],\n () => Math.PI\n ],\n 'e': [\n NumberType,\n [],\n () => Math.E\n ],\n '^': [\n NumberType,\n [NumberType, NumberType],\n (ctx, [b, e]) => Math.pow(b.evaluate(ctx), e.evaluate(ctx))\n ],\n 'sqrt': [\n NumberType,\n [NumberType],\n (ctx, [x]) => Math.sqrt(x.evaluate(ctx))\n ],\n 'log10': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.log(n.evaluate(ctx)) / Math.LN10\n ],\n 'ln': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.log(n.evaluate(ctx))\n ],\n 'log2': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.log(n.evaluate(ctx)) / Math.LN2\n ],\n 'sin': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.sin(n.evaluate(ctx))\n ],\n 'cos': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.cos(n.evaluate(ctx))\n ],\n 'tan': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.tan(n.evaluate(ctx))\n ],\n 'asin': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.asin(n.evaluate(ctx))\n ],\n 'acos': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.acos(n.evaluate(ctx))\n ],\n 'atan': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.atan(n.evaluate(ctx))\n ],\n 'min': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => Math.min(...args.map(arg => arg.evaluate(ctx)))\n ],\n 'max': [\n NumberType,\n varargs(NumberType),\n (ctx, args) => Math.max(...args.map(arg => arg.evaluate(ctx)))\n ],\n 'abs': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.abs(n.evaluate(ctx))\n ],\n 'round': [\n NumberType,\n [NumberType],\n (ctx, [n]) => {\n const v = n.evaluate(ctx);\n // Javascript's Math.round() rounds towards +Infinity for halfway\n // values, even when they're negative. It's more common to round\n // away from 0 (e.g., this is what python and C++ do)\n return v < 0 ? -Math.round(-v) : Math.round(v);\n }\n ],\n 'floor': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.floor(n.evaluate(ctx))\n ],\n 'ceil': [\n NumberType,\n [NumberType],\n (ctx, [n]) => Math.ceil(n.evaluate(ctx))\n ],\n 'filter-==': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => ctx.properties()[k.value] === v.value\n ],\n 'filter-id-==': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => ctx.id() === v.value\n ],\n 'filter-type-==': [\n BooleanType,\n [StringType],\n (ctx, [v]) => ctx.geometryType() === v.value\n ],\n 'filter-<': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a < b;\n }\n ],\n 'filter-id-<': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a < b;\n }\n ],\n 'filter->': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a > b;\n }\n ],\n 'filter-id->': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a > b;\n }\n ],\n 'filter-<=': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a <= b;\n }\n ],\n 'filter-id-<=': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a <= b;\n }\n ],\n 'filter->=': [\n BooleanType,\n [StringType, ValueType],\n (ctx, [k, v]) => {\n const a = ctx.properties()[k.value];\n const b = v.value;\n return typeof a === typeof b && a >= b;\n }\n ],\n 'filter-id->=': [\n BooleanType,\n [ValueType],\n (ctx, [v]) => {\n const a = ctx.id();\n const b = v.value;\n return typeof a === typeof b && a >= b;\n }\n ],\n 'filter-has': [\n BooleanType,\n [ValueType],\n (ctx, [k]) => k.value in ctx.properties()\n ],\n 'filter-has-id': [\n BooleanType,\n [],\n (ctx) => (ctx.id() !== null && ctx.id() !== undefined)\n ],\n 'filter-type-in': [\n BooleanType,\n [array$1(StringType)],\n (ctx, [v]) => v.value.indexOf(ctx.geometryType()) >= 0\n ],\n 'filter-id-in': [\n BooleanType,\n [array$1(ValueType)],\n (ctx, [v]) => v.value.indexOf(ctx.id()) >= 0\n ],\n 'filter-in-small': [\n BooleanType,\n [StringType, array$1(ValueType)],\n // assumes v is an array literal\n (ctx, [k, v]) => v.value.indexOf(ctx.properties()[k.value]) >= 0\n ],\n 'filter-in-large': [\n BooleanType,\n [StringType, array$1(ValueType)],\n // assumes v is a array literal with values sorted in ascending order and of a single type\n (ctx, [k, v]) => binarySearch(ctx.properties()[k.value], v.value, 0, v.value.length - 1)\n ],\n 'all': {\n type: BooleanType,\n overloads: [\n [\n [BooleanType, BooleanType],\n (ctx, [a, b]) => a.evaluate(ctx) && b.evaluate(ctx)\n ],\n [\n varargs(BooleanType),\n (ctx, args) => {\n for (const arg of args) {\n if (!arg.evaluate(ctx))\n return false;\n }\n return true;\n }\n ]\n ]\n },\n 'any': {\n type: BooleanType,\n overloads: [\n [\n [BooleanType, BooleanType],\n (ctx, [a, b]) => a.evaluate(ctx) || b.evaluate(ctx)\n ],\n [\n varargs(BooleanType),\n (ctx, args) => {\n for (const arg of args) {\n if (arg.evaluate(ctx))\n return true;\n }\n return false;\n }\n ]\n ]\n },\n '!': [\n BooleanType,\n [BooleanType],\n (ctx, [b]) => !b.evaluate(ctx)\n ],\n 'is-supported-script': [\n BooleanType,\n [StringType],\n // At parse time this will always return true, so we need to exclude this expression with isGlobalPropertyConstant\n (ctx, [s]) => {\n const isSupportedScript = ctx.globals && ctx.globals.isSupportedScript;\n if (isSupportedScript) {\n return isSupportedScript(s.evaluate(ctx));\n }\n return true;\n }\n ],\n 'upcase': [\n StringType,\n [StringType],\n (ctx, [s]) => s.evaluate(ctx).toUpperCase()\n ],\n 'downcase': [\n StringType,\n [StringType],\n (ctx, [s]) => s.evaluate(ctx).toLowerCase()\n ],\n 'concat': [\n StringType,\n varargs(ValueType),\n (ctx, args) => args.map(arg => toString(arg.evaluate(ctx))).join('')\n ],\n 'resolved-locale': [\n StringType,\n [CollatorType],\n (ctx, [collator]) => collator.evaluate(ctx).resolvedLocale()\n ]\n});\n\nfunction success(value) {\n return { result: 'success', value };\n}\nfunction error(value) {\n return { result: 'error', value };\n}\n\nfunction supportsPropertyExpression(spec) {\n return spec['property-type'] === 'data-driven' || spec['property-type'] === 'cross-faded-data-driven';\n}\nfunction supportsZoomExpression(spec) {\n return !!spec.expression && spec.expression.parameters.indexOf('zoom') > -1;\n}\nfunction supportsInterpolation(spec) {\n return !!spec.expression && spec.expression.interpolated;\n}\n\nfunction getType(val) {\n if (val instanceof Number) {\n return 'number';\n }\n else if (val instanceof String) {\n return 'string';\n }\n else if (val instanceof Boolean) {\n return 'boolean';\n }\n else if (Array.isArray(val)) {\n return 'array';\n }\n else if (val === null) {\n return 'null';\n }\n else {\n return typeof val;\n }\n}\n\nfunction isFunction(value) {\n return typeof value === 'object' && value !== null && !Array.isArray(value);\n}\nfunction identityFunction(x) {\n return x;\n}\nfunction createFunction(parameters, propertySpec) {\n const isColor = propertySpec.type === 'color';\n const zoomAndFeatureDependent = parameters.stops && typeof parameters.stops[0][0] === 'object';\n const featureDependent = zoomAndFeatureDependent || parameters.property !== undefined;\n const zoomDependent = zoomAndFeatureDependent || !featureDependent;\n const type = parameters.type || (supportsInterpolation(propertySpec) ? 'exponential' : 'interval');\n if (isColor || propertySpec.type === 'padding') {\n const parseFn = isColor ? Color.parse : Padding.parse;\n parameters = extendBy({}, parameters);\n if (parameters.stops) {\n parameters.stops = parameters.stops.map((stop) => {\n return [stop[0], parseFn(stop[1])];\n });\n }\n if (parameters.default) {\n parameters.default = parseFn(parameters.default);\n }\n else {\n parameters.default = parseFn(propertySpec.default);\n }\n }\n if (parameters.colorSpace && !isSupportedInterpolationColorSpace(parameters.colorSpace)) {\n throw new Error(`Unknown color space: \"${parameters.colorSpace}\"`);\n }\n let innerFun;\n let hashedStops;\n let categoricalKeyType;\n if (type === 'exponential') {\n innerFun = evaluateExponentialFunction;\n }\n else if (type === 'interval') {\n innerFun = evaluateIntervalFunction;\n }\n else if (type === 'categorical') {\n innerFun = evaluateCategoricalFunction;\n // For categorical functions, generate an Object as a hashmap of the stops for fast searching\n hashedStops = Object.create(null);\n for (const stop of parameters.stops) {\n hashedStops[stop[0]] = stop[1];\n }\n // Infer key type based on first stop key-- used to encforce strict type checking later\n categoricalKeyType = typeof parameters.stops[0][0];\n }\n else if (type === 'identity') {\n innerFun = evaluateIdentityFunction;\n }\n else {\n throw new Error(`Unknown function type \"${type}\"`);\n }\n if (zoomAndFeatureDependent) {\n const featureFunctions = {};\n const zoomStops = [];\n for (let s = 0; s < parameters.stops.length; s++) {\n const stop = parameters.stops[s];\n const zoom = stop[0].zoom;\n if (featureFunctions[zoom] === undefined) {\n featureFunctions[zoom] = {\n zoom,\n type: parameters.type,\n property: parameters.property,\n default: parameters.default,\n stops: []\n };\n zoomStops.push(zoom);\n }\n featureFunctions[zoom].stops.push([stop[0].value, stop[1]]);\n }\n const featureFunctionStops = [];\n for (const z of zoomStops) {\n featureFunctionStops.push([featureFunctions[z].zoom, createFunction(featureFunctions[z], propertySpec)]);\n }\n const interpolationType = { name: 'linear' };\n return {\n kind: 'composite',\n interpolationType,\n interpolationFactor: Interpolate.interpolationFactor.bind(undefined, interpolationType),\n zoomStops: featureFunctionStops.map(s => s[0]),\n evaluate({ zoom }, properties) {\n return evaluateExponentialFunction({\n stops: featureFunctionStops,\n base: parameters.base\n }, propertySpec, zoom).evaluate(zoom, properties);\n }\n };\n }\n else if (zoomDependent) {\n const interpolationType = type === 'exponential' ?\n { name: 'exponential', base: parameters.base !== undefined ? parameters.base : 1 } : null;\n return {\n kind: 'camera',\n interpolationType,\n interpolationFactor: Interpolate.interpolationFactor.bind(undefined, interpolationType),\n zoomStops: parameters.stops.map(s => s[0]),\n evaluate: ({ zoom }) => innerFun(parameters, propertySpec, zoom, hashedStops, categoricalKeyType)\n };\n }\n else {\n return {\n kind: 'source',\n evaluate(_, feature) {\n const value = feature && feature.properties ? feature.properties[parameters.property] : undefined;\n if (value === undefined) {\n return coalesce$1(parameters.default, propertySpec.default);\n }\n return innerFun(parameters, propertySpec, value, hashedStops, categoricalKeyType);\n }\n };\n }\n}\nfunction coalesce$1(a, b, c) {\n if (a !== undefined)\n return a;\n if (b !== undefined)\n return b;\n if (c !== undefined)\n return c;\n}\nfunction evaluateCategoricalFunction(parameters, propertySpec, input, hashedStops, keyType) {\n const evaluated = typeof input === keyType ? hashedStops[input] : undefined; // Enforce strict typing on input\n return coalesce$1(evaluated, parameters.default, propertySpec.default);\n}\nfunction evaluateIntervalFunction(parameters, propertySpec, input) {\n // Edge cases\n if (getType(input) !== 'number')\n return coalesce$1(parameters.default, propertySpec.default);\n const n = parameters.stops.length;\n if (n === 1)\n return parameters.stops[0][1];\n if (input <= parameters.stops[0][0])\n return parameters.stops[0][1];\n if (input >= parameters.stops[n - 1][0])\n return parameters.stops[n - 1][1];\n const index = findStopLessThanOrEqualTo(parameters.stops.map((stop) => stop[0]), input);\n return parameters.stops[index][1];\n}\nfunction evaluateExponentialFunction(parameters, propertySpec, input) {\n const base = parameters.base !== undefined ? parameters.base : 1;\n // Edge cases\n if (getType(input) !== 'number')\n return coalesce$1(parameters.default, propertySpec.default);\n const n = parameters.stops.length;\n if (n === 1)\n return parameters.stops[0][1];\n if (input <= parameters.stops[0][0])\n return parameters.stops[0][1];\n if (input >= parameters.stops[n - 1][0])\n return parameters.stops[n - 1][1];\n const index = findStopLessThanOrEqualTo(parameters.stops.map((stop) => stop[0]), input);\n const t = interpolationFactor(input, base, parameters.stops[index][0], parameters.stops[index + 1][0]);\n const outputLower = parameters.stops[index][1];\n const outputUpper = parameters.stops[index + 1][1];\n const interp = interpolate[propertySpec.type] || identityFunction;\n if (typeof outputLower.evaluate === 'function') {\n return {\n evaluate(...args) {\n const evaluatedLower = outputLower.evaluate.apply(undefined, args);\n const evaluatedUpper = outputUpper.evaluate.apply(undefined, args);\n // Special case for fill-outline-color, which has no spec default.\n if (evaluatedLower === undefined || evaluatedUpper === undefined) {\n return undefined;\n }\n return interp(evaluatedLower, evaluatedUpper, t, parameters.colorSpace);\n }\n };\n }\n return interp(outputLower, outputUpper, t, parameters.colorSpace);\n}\nfunction evaluateIdentityFunction(parameters, propertySpec, input) {\n switch (propertySpec.type) {\n case 'color':\n input = Color.parse(input);\n break;\n case 'formatted':\n input = Formatted.fromString(input.toString());\n break;\n case 'resolvedImage':\n input = ResolvedImage.fromString(input.toString());\n break;\n case 'padding':\n input = Padding.parse(input);\n break;\n default:\n if (getType(input) !== propertySpec.type && (propertySpec.type !== 'enum' || !propertySpec.values[input])) {\n input = undefined;\n }\n }\n return coalesce$1(input, parameters.default, propertySpec.default);\n}\n/**\n * Returns a ratio that can be used to interpolate between exponential function\n * stops.\n *\n * How it works:\n * Two consecutive stop values define a (scaled and shifted) exponential\n * function `f(x) = a * base^x + b`, where `base` is the user-specified base,\n * and `a` and `b` are constants affording sufficient degrees of freedom to fit\n * the function to the given stops.\n *\n * Here's a bit of algebra that lets us compute `f(x)` directly from the stop\n * values without explicitly solving for `a` and `b`:\n *\n * First stop value: `f(x0) = y0 = a * base^x0 + b`\n * Second stop value: `f(x1) = y1 = a * base^x1 + b`\n * => `y1 - y0 = a(base^x1 - base^x0)`\n * => `a = (y1 - y0)/(base^x1 - base^x0)`\n *\n * Desired value: `f(x) = y = a * base^x + b`\n * => `f(x) = y0 + a * (base^x - base^x0)`\n *\n * From the above, we can replace the `a` in `a * (base^x - base^x0)` and do a\n * little algebra:\n * ```\n * a * (base^x - base^x0) = (y1 - y0)/(base^x1 - base^x0) * (base^x - base^x0)\n * = (y1 - y0) * (base^x - base^x0) / (base^x1 - base^x0)\n * ```\n *\n * If we let `(base^x - base^x0) / (base^x1 base^x0)`, then we have\n * `f(x) = y0 + (y1 - y0) * ratio`. In other words, `ratio` may be treated as\n * an interpolation factor between the two stops' output values.\n *\n * (Note: a slightly different form for `ratio`,\n * `(base^(x-x0) - 1) / (base^(x1-x0) - 1) `, is equivalent, but requires fewer\n * expensive `Math.pow()` operations.)\n *\n * @private\n */\nfunction interpolationFactor(input, base, lowerValue, upperValue) {\n const difference = upperValue - lowerValue;\n const progress = input - lowerValue;\n if (difference === 0) {\n return 0;\n }\n else if (base === 1) {\n return progress / difference;\n }\n else {\n return (Math.pow(base, progress) - 1) / (Math.pow(base, difference) - 1);\n }\n}\n\nclass StyleExpression {\n constructor(expression, propertySpec) {\n this.expression = expression;\n this._warningHistory = {};\n this._evaluator = new EvaluationContext();\n this._defaultValue = propertySpec ? getDefaultValue(propertySpec) : null;\n this._enumValues = propertySpec && propertySpec.type === 'enum' ? propertySpec.values : null;\n }\n evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) {\n this._evaluator.globals = globals;\n this._evaluator.feature = feature;\n this._evaluator.featureState = featureState;\n this._evaluator.canonical = canonical;\n this._evaluator.availableImages = availableImages || null;\n this._evaluator.formattedSection = formattedSection;\n return this.expression.evaluate(this._evaluator);\n }\n evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) {\n this._evaluator.globals = globals;\n this._evaluator.feature = feature || null;\n this._evaluator.featureState = featureState || null;\n this._evaluator.canonical = canonical;\n this._evaluator.availableImages = availableImages || null;\n this._evaluator.formattedSection = formattedSection || null;\n try {\n const val = this.expression.evaluate(this._evaluator);\n // eslint-disable-next-line no-self-compare\n if (val === null || val === undefined || (typeof val === 'number' && val !== val)) {\n return this._defaultValue;\n }\n if (this._enumValues && !(val in this._enumValues)) {\n throw new RuntimeError(`Expected value to be one of ${Object.keys(this._enumValues).map(v => JSON.stringify(v)).join(', ')}, but found ${JSON.stringify(val)} instead.`);\n }\n return val;\n }\n catch (e) {\n if (!this._warningHistory[e.message]) {\n this._warningHistory[e.message] = true;\n if (typeof console !== 'undefined') {\n console.warn(e.message);\n }\n }\n return this._defaultValue;\n }\n }\n}\nfunction isExpression(expression) {\n return Array.isArray(expression) && expression.length > 0 &&\n typeof expression[0] === 'string' && expression[0] in expressions;\n}\n/**\n * Parse and typecheck the given style spec JSON expression. If\n * options.defaultValue is provided, then the resulting StyleExpression's\n * `evaluate()` method will handle errors by logging a warning (once per\n * message) and returning the default value. Otherwise, it will throw\n * evaluation errors.\n *\n * @private\n */\nfunction createExpression(expression, propertySpec) {\n const parser = new ParsingContext(expressions, isExpressionConstant, [], propertySpec ? getExpectedType(propertySpec) : undefined);\n // For string-valued properties, coerce to string at the top level rather than asserting.\n const parsed = parser.parse(expression, undefined, undefined, undefined, propertySpec && propertySpec.type === 'string' ? { typeAnnotation: 'coerce' } : undefined);\n if (!parsed) {\n return error(parser.errors);\n }\n return success(new StyleExpression(parsed, propertySpec));\n}\nclass ZoomConstantExpression {\n constructor(kind, expression) {\n this.kind = kind;\n this._styleExpression = expression;\n this.isStateDependent = kind !== 'constant' && !isStateConstant(expression.expression);\n }\n evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluate(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n}\nclass ZoomDependentExpression {\n constructor(kind, expression, zoomStops, interpolationType) {\n this.kind = kind;\n this.zoomStops = zoomStops;\n this._styleExpression = expression;\n this.isStateDependent = kind !== 'camera' && !isStateConstant(expression.expression);\n this.interpolationType = interpolationType;\n }\n evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluateWithoutErrorHandling(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n evaluate(globals, feature, featureState, canonical, availableImages, formattedSection) {\n return this._styleExpression.evaluate(globals, feature, featureState, canonical, availableImages, formattedSection);\n }\n interpolationFactor(input, lower, upper) {\n if (this.interpolationType) {\n return Interpolate.interpolationFactor(this.interpolationType, input, lower, upper);\n }\n else {\n return 0;\n }\n }\n}\nfunction isZoomExpression(expression) {\n return expression._styleExpression !== undefined;\n}\nfunction createPropertyExpression(expressionInput, propertySpec) {\n const expression = createExpression(expressionInput, propertySpec);\n if (expression.result === 'error') {\n return expression;\n }\n const parsed = expression.value.expression;\n const isFeatureConstantResult = isFeatureConstant(parsed);\n if (!isFeatureConstantResult && !supportsPropertyExpression(propertySpec)) {\n return error([new ExpressionParsingError('', 'data expressions not supported')]);\n }\n const isZoomConstant = isGlobalPropertyConstant(parsed, ['zoom']);\n if (!isZoomConstant && !supportsZoomExpression(propertySpec)) {\n return error([new ExpressionParsingError('', 'zoom expressions not supported')]);\n }\n const zoomCurve = findZoomCurve(parsed);\n if (!zoomCurve && !isZoomConstant) {\n return error([new ExpressionParsingError('', '\"zoom\" expression may only be used as input to a top-level \"step\" or \"interpolate\" expression.')]);\n }\n else if (zoomCurve instanceof ExpressionParsingError) {\n return error([zoomCurve]);\n }\n else if (zoomCurve instanceof Interpolate && !supportsInterpolation(propertySpec)) {\n return error([new ExpressionParsingError('', '\"interpolate\" expressions cannot be used with this property')]);\n }\n if (!zoomCurve) {\n return success(isFeatureConstantResult ?\n new ZoomConstantExpression('constant', expression.value) :\n new ZoomConstantExpression('source', expression.value));\n }\n const interpolationType = zoomCurve instanceof Interpolate ? zoomCurve.interpolation : undefined;\n return success(isFeatureConstantResult ?\n new ZoomDependentExpression('camera', expression.value, zoomCurve.labels, interpolationType) :\n new ZoomDependentExpression('composite', expression.value, zoomCurve.labels, interpolationType));\n}\n// serialization wrapper for old-style stop functions normalized to the\n// expression interface\nclass StylePropertyFunction {\n constructor(parameters, specification) {\n this._parameters = parameters;\n this._specification = specification;\n extendBy(this, createFunction(this._parameters, this._specification));\n }\n static deserialize(serialized) {\n return new StylePropertyFunction(serialized._parameters, serialized._specification);\n }\n static serialize(input) {\n return {\n _parameters: input._parameters,\n _specification: input._specification\n };\n }\n}\nfunction normalizePropertyExpression(value, specification) {\n if (isFunction(value)) {\n return new StylePropertyFunction(value, specification);\n }\n else if (isExpression(value)) {\n const expression = createPropertyExpression(value, specification);\n if (expression.result === 'error') {\n // this should have been caught in validation\n throw new Error(expression.value.map(err => `${err.key}: ${err.message}`).join(', '));\n }\n return expression.value;\n }\n else {\n let constant = value;\n if (specification.type === 'color' && typeof value === 'string') {\n constant = Color.parse(value);\n }\n else if (specification.type === 'padding' && (typeof value === 'number' || Array.isArray(value))) {\n constant = Padding.parse(value);\n }\n else if (specification.type === 'variableAnchorOffsetCollection' && Array.isArray(value)) {\n constant = VariableAnchorOffsetCollection.parse(value);\n }\n return {\n kind: 'constant',\n evaluate: () => constant\n };\n }\n}\n// Zoom-dependent expressions may only use [\"zoom\"] as the input to a top-level \"step\" or \"interpolate\"\n// expression (collectively referred to as a \"curve\"). The curve may be wrapped in one or more \"let\" or\n// \"coalesce\" expressions.\nfunction findZoomCurve(expression) {\n let result = null;\n if (expression instanceof Let) {\n result = findZoomCurve(expression.result);\n }\n else if (expression instanceof Coalesce) {\n for (const arg of expression.args) {\n result = findZoomCurve(arg);\n if (result) {\n break;\n }\n }\n }\n else if ((expression instanceof Step || expression instanceof Interpolate) &&\n expression.input instanceof CompoundExpression &&\n expression.input.name === 'zoom') {\n result = expression;\n }\n if (result instanceof ExpressionParsingError) {\n return result;\n }\n expression.eachChild((child) => {\n const childResult = findZoomCurve(child);\n if (childResult instanceof ExpressionParsingError) {\n result = childResult;\n }\n else if (!result && childResult) {\n result = new ExpressionParsingError('', '\"zoom\" expression may only be used as input to a top-level \"step\" or \"interpolate\" expression.');\n }\n else if (result && childResult && result !== childResult) {\n result = new ExpressionParsingError('', 'Only one zoom-based \"step\" or \"interpolate\" subexpression may be used in an expression.');\n }\n });\n return result;\n}\nfunction getExpectedType(spec) {\n const types = {\n color: ColorType,\n string: StringType,\n number: NumberType,\n enum: StringType,\n boolean: BooleanType,\n formatted: FormattedType,\n padding: PaddingType,\n resolvedImage: ResolvedImageType,\n variableAnchorOffsetCollection: VariableAnchorOffsetCollectionType\n };\n if (spec.type === 'array') {\n return array$1(types[spec.value] || ValueType, spec.length);\n }\n return types[spec.type];\n}\nfunction getDefaultValue(spec) {\n if (spec.type === 'color' && isFunction(spec.default)) {\n // Special case for heatmap-color: it uses the 'default:' to define a\n // default color ramp, but createExpression expects a simple value to fall\n // back to in case of runtime errors\n return new Color(0, 0, 0, 0);\n }\n else if (spec.type === 'color') {\n return Color.parse(spec.default) || null;\n }\n else if (spec.type === 'padding') {\n return Padding.parse(spec.default) || null;\n }\n else if (spec.type === 'variableAnchorOffsetCollection') {\n return VariableAnchorOffsetCollection.parse(spec.default) || null;\n }\n else if (spec.default === undefined) {\n return null;\n }\n else {\n return spec.default;\n }\n}\n\nfunction isExpressionFilter(filter) {\n if (filter === true || filter === false) {\n return true;\n }\n if (!Array.isArray(filter) || filter.length === 0) {\n return false;\n }\n switch (filter[0]) {\n case 'has':\n return filter.length >= 2 && filter[1] !== '$id' && filter[1] !== '$type';\n case 'in':\n return filter.length >= 3 && (typeof filter[1] !== 'string' || Array.isArray(filter[2]));\n case '!in':\n case '!has':\n case 'none':\n return false;\n case '==':\n case '!=':\n case '>':\n case '>=':\n case '<':\n case '<=':\n return filter.length !== 3 || (Array.isArray(filter[1]) || Array.isArray(filter[2]));\n case 'any':\n case 'all':\n for (const f of filter.slice(1)) {\n if (!isExpressionFilter(f) && typeof f !== 'boolean') {\n return false;\n }\n }\n return true;\n default:\n return true;\n }\n}\nconst filterSpec = {\n 'type': 'boolean',\n 'default': false,\n 'transition': false,\n 'property-type': 'data-driven',\n 'expression': {\n 'interpolated': false,\n 'parameters': ['zoom', 'feature']\n }\n};\n/**\n * Given a filter expressed as nested arrays, return a new function\n * that evaluates whether a given feature (with a .properties or .tags property)\n * passes its test.\n *\n * @private\n * @param {Array} filter MapLibre filter\n * @returns {Function} filter-evaluating function\n */\nfunction createFilter(filter) {\n if (filter === null || filter === undefined) {\n return { filter: () => true, needGeometry: false };\n }\n if (!isExpressionFilter(filter)) {\n filter = convertFilter$1(filter);\n }\n const compiled = createExpression(filter, filterSpec);\n if (compiled.result === 'error') {\n throw new Error(compiled.value.map(err => `${err.key}: ${err.message}`).join(', '));\n }\n else {\n const needGeometry = geometryNeeded(filter);\n return { filter: (globalProperties, feature, canonical) => compiled.value.evaluate(globalProperties, feature, {}, canonical),\n needGeometry };\n }\n}\n// Comparison function to sort numbers and strings\nfunction compare(a, b) {\n return a < b ? -1 : a > b ? 1 : 0;\n}\nfunction geometryNeeded(filter) {\n if (!Array.isArray(filter))\n return false;\n if (filter[0] === 'within')\n return true;\n for (let index = 1; index < filter.length; index++) {\n if (geometryNeeded(filter[index]))\n return true;\n }\n return false;\n}\nfunction convertFilter$1(filter) {\n if (!filter)\n return true;\n const op = filter[0];\n if (filter.length <= 1)\n return (op !== 'any');\n const converted = op === '==' ? convertComparisonOp$1(filter[1], filter[2], '==') :\n op === '!=' ? convertNegation(convertComparisonOp$1(filter[1], filter[2], '==')) :\n op === '<' ||\n op === '>' ||\n op === '<=' ||\n op === '>=' ? convertComparisonOp$1(filter[1], filter[2], op) :\n op === 'any' ? convertDisjunctionOp(filter.slice(1)) :\n op === 'all' ? ['all'].concat(filter.slice(1).map(convertFilter$1)) :\n op === 'none' ? ['all'].concat(filter.slice(1).map(convertFilter$1).map(convertNegation)) :\n op === 'in' ? convertInOp$1(filter[1], filter.slice(2)) :\n op === '!in' ? convertNegation(convertInOp$1(filter[1], filter.slice(2))) :\n op === 'has' ? convertHasOp$1(filter[1]) :\n op === '!has' ? convertNegation(convertHasOp$1(filter[1])) :\n op === 'within' ? filter :\n true;\n return converted;\n}\nfunction convertComparisonOp$1(property, value, op) {\n switch (property) {\n case '$type':\n return [`filter-type-${op}`, value];\n case '$id':\n return [`filter-id-${op}`, value];\n default:\n return [`filter-${op}`, property, value];\n }\n}\nfunction convertDisjunctionOp(filters) {\n return ['any'].concat(filters.map(convertFilter$1));\n}\nfunction convertInOp$1(property, values) {\n if (values.length === 0) {\n return false;\n }\n switch (property) {\n case '$type':\n return ['filter-type-in', ['literal', values]];\n case '$id':\n return ['filter-id-in', ['literal', values]];\n default:\n if (values.length > 200 && !values.some(v => typeof v !== typeof values[0])) {\n return ['filter-in-large', property, ['literal', values.sort(compare)]];\n }\n else {\n return ['filter-in-small', property, ['literal', values]];\n }\n }\n}\nfunction convertHasOp$1(property) {\n switch (property) {\n case '$type':\n return true;\n case '$id':\n return ['filter-has-id'];\n default:\n return ['filter-has', property];\n }\n}\nfunction convertNegation(filter) {\n return ['!', filter];\n}\n\n/*\n * Convert the given filter to an expression, storing the expected types for\n * any feature properties referenced in expectedTypes.\n *\n * These expected types are needed in order to construct preflight type checks\n * needed for handling 'any' filters. A preflight type check is necessary in\n * order to mimic legacy filters' semantics around expected type mismatches.\n * For example, consider the legacy filter:\n *\n * [\"any\", [\"all\", [\">\", \"y\", 0], [\">\", \"y\", 0]], [\">\", \"x\", 0]]\n *\n * Naively, we might convert this to the expression:\n *\n * [\"any\", [\"all\", [\">\", [\"get\", \"y\"], 0], [\">\", [\"get\", \"z\"], 0]], [\">\", [\"get\", \"x\"], 0]]\n *\n * But if we tried to evaluate this against, say `{x: 1, y: null, z: 0}`, the\n * [\">\", [\"get\", \"y\"], 0] would cause an evaluation error, leading to the\n * entire filter returning false. Legacy filter semantics, though, ask for\n * [\">\", \"y\", 0] to simply return `false` when `y` is of the wrong type,\n * allowing the subsequent terms of the outer \"any\" expression to be evaluated\n * (resulting, in this case, in a `true` value, because x > 0).\n *\n * We account for this by inserting a preflight type-checking expression before\n * each \"any\" term, allowing us to avoid evaluating the actual converted filter\n * if any type mismatches would cause it to produce an evalaution error:\n *\n * [\"any\",\n * [\"case\",\n * [\"all\", [\"==\", [\"typeof\", [\"get\", \"y\"]], \"number\"], [\"==\", [\"typeof\", [\"get\", \"z\"], \"number]],\n * [\"all\", [\">\", [\"get\", \"y\"], 0], [\">\", [\"get\", \"z\"], 0]],\n * false\n * ],\n * [\"case\",\n * [\"==\", [\"typeof\", [\"get\", \"x\"], \"number\"]],\n * [\">\", [\"get\", \"x\"], 0],\n * false\n * ]\n * ]\n *\n * An alternative, possibly more direct approach would be to use type checks\n * in the conversion of each comparison operator, so that the converted version\n * of each individual ==, >=, etc. would mimic the legacy filter semantics. The\n * downside of this approach is that it can lead to many more type checks than\n * would otherwise be necessary: outside the context of an \"any\" expression,\n * bailing out due to a runtime type error (expression semantics) and returning\n * false (legacy filter semantics) are equivalent: they cause the filter to\n * produce a `false` result.\n */\nfunction convertFilter(filter, expectedTypes = {}) {\n if (isExpressionFilter(filter))\n return filter;\n if (!filter)\n return true;\n const legacyFilter = filter;\n const legacyOp = legacyFilter[0];\n if (filter.length <= 1)\n return (legacyOp !== 'any');\n switch (legacyOp) {\n case '==':\n case '!=':\n case '<':\n case '>':\n case '<=':\n case '>=': {\n const [, property, value] = filter;\n return convertComparisonOp(property, value, legacyOp, expectedTypes);\n }\n case 'any': {\n const [, ...conditions] = legacyFilter;\n const children = conditions.map((f) => {\n const types = {};\n const child = convertFilter(f, types);\n const typechecks = runtimeTypeChecks(types);\n return typechecks === true ? child : ['case', typechecks, child, false];\n });\n return ['any', ...children];\n }\n case 'all': {\n const [, ...conditions] = legacyFilter;\n const children = conditions.map(f => convertFilter(f, expectedTypes));\n return children.length > 1 ? ['all', ...children] : children[0];\n }\n case 'none': {\n const [, ...conditions] = legacyFilter;\n return ['!', convertFilter(['any', ...conditions], {})];\n }\n case 'in': {\n const [, property, ...values] = legacyFilter;\n return convertInOp(property, values);\n }\n case '!in': {\n const [, property, ...values] = legacyFilter;\n return convertInOp(property, values, true);\n }\n case 'has':\n return convertHasOp(legacyFilter[1]);\n case '!has':\n return ['!', convertHasOp(legacyFilter[1])];\n default:\n return true;\n }\n}\n// Given a set of feature properties and an expected type for each one,\n// construct an boolean expression that tests whether each property has the\n// right type.\n// E.g.: for {name: 'string', population: 'number'}, return\n// [ 'all',\n// ['==', ['typeof', ['get', 'name'], 'string']],\n// ['==', ['typeof', ['get', 'population'], 'number]]\n// ]\nfunction runtimeTypeChecks(expectedTypes) {\n const conditions = [];\n for (const property in expectedTypes) {\n const get = property === '$id' ? ['id'] : ['get', property];\n conditions.push(['==', ['typeof', get], expectedTypes[property]]);\n }\n if (conditions.length === 0)\n return true;\n if (conditions.length === 1)\n return conditions[0];\n return ['all', ...conditions];\n}\nfunction convertComparisonOp(property, value, op, expectedTypes) {\n let get;\n if (property === '$type') {\n return [op, ['geometry-type'], value];\n }\n else if (property === '$id') {\n get = ['id'];\n }\n else {\n get = ['get', property];\n }\n if (expectedTypes && value !== null) {\n const type = typeof value;\n expectedTypes[property] = type;\n }\n if (op === '==' && property !== '$id' && value === null) {\n return [\n 'all',\n ['has', property],\n ['==', get, null]\n ];\n }\n else if (op === '!=' && property !== '$id' && value === null) {\n return [\n 'any',\n ['!', ['has', property]],\n ['!=', get, null]\n ];\n }\n return [op, get, value];\n}\nfunction convertInOp(property, values, negate = false) {\n if (values.length === 0)\n return negate;\n let get;\n if (property === '$type') {\n get = ['geometry-type'];\n }\n else if (property === '$id') {\n get = ['id'];\n }\n else {\n get = ['get', property];\n }\n // Determine if the list of values to be searched is homogenously typed.\n // If so (and if the type is string or number), then we can use a\n // [match, input, [...values], true, false] construction rather than a\n // bunch of `==` tests.\n let uniformTypes = true;\n const type = typeof values[0];\n for (const value of values) {\n if (typeof value !== type) {\n uniformTypes = false;\n break;\n }\n }\n if (uniformTypes && (type === 'string' || type === 'number')) {\n // Match expressions must have unique values.\n const uniqueValues = values.sort().filter((v, i) => i === 0 || values[i - 1] !== v);\n return ['match', get, uniqueValues, !negate, negate];\n }\n if (negate) {\n return ['all', ...values.map(v => ['!=', get, v])];\n }\n else {\n return ['any', ...values.map(v => ['==', get, v])];\n }\n}\nfunction convertHasOp(property) {\n if (property === '$type') {\n return true;\n }\n else if (property === '$id') {\n return ['!=', ['id'], null];\n }\n else {\n return ['has', property];\n }\n}\n\nfunction convertLiteral(value) {\n return typeof value === 'object' ? ['literal', value] : value;\n}\nfunction convertFunction(parameters, propertySpec) {\n let stops = parameters.stops;\n if (!stops) {\n // identity function\n return convertIdentityFunction(parameters, propertySpec);\n }\n const zoomAndFeatureDependent = stops && typeof stops[0][0] === 'object';\n const featureDependent = zoomAndFeatureDependent || parameters.property !== undefined;\n const zoomDependent = zoomAndFeatureDependent || !featureDependent;\n stops = stops.map((stop) => {\n if (!featureDependent && propertySpec.tokens && typeof stop[1] === 'string') {\n return [stop[0], convertTokenString(stop[1])];\n }\n return [stop[0], convertLiteral(stop[1])];\n });\n if (zoomAndFeatureDependent) {\n return convertZoomAndPropertyFunction(parameters, propertySpec, stops);\n }\n else if (zoomDependent) {\n return convertZoomFunction(parameters, propertySpec, stops);\n }\n else {\n return convertPropertyFunction(parameters, propertySpec, stops);\n }\n}\nfunction convertIdentityFunction(parameters, propertySpec) {\n const get = ['get', parameters.property];\n if (parameters.default === undefined) {\n // By default, expressions for string-valued properties get coerced. To preserve\n // legacy function semantics, insert an explicit assertion instead.\n return propertySpec.type === 'string' ? ['string', get] : get;\n }\n else if (propertySpec.type === 'enum') {\n return [\n 'match',\n get,\n Object.keys(propertySpec.values),\n get,\n parameters.default\n ];\n }\n else {\n const expression = [propertySpec.type === 'color' ? 'to-color' : propertySpec.type, get, convertLiteral(parameters.default)];\n if (propertySpec.type === 'array') {\n expression.splice(1, 0, propertySpec.value, propertySpec.length || null);\n }\n return expression;\n }\n}\nfunction getInterpolateOperator(parameters) {\n switch (parameters.colorSpace) {\n case 'hcl': return 'interpolate-hcl';\n case 'lab': return 'interpolate-lab';\n default: return 'interpolate';\n }\n}\nfunction convertZoomAndPropertyFunction(parameters, propertySpec, stops) {\n const featureFunctionParameters = {};\n const featureFunctionStops = {};\n const zoomStops = [];\n for (let s = 0; s < stops.length; s++) {\n const stop = stops[s];\n const zoom = stop[0].zoom;\n if (featureFunctionParameters[zoom] === undefined) {\n featureFunctionParameters[zoom] = {\n zoom,\n type: parameters.type,\n property: parameters.property,\n default: parameters.default,\n };\n featureFunctionStops[zoom] = [];\n zoomStops.push(zoom);\n }\n featureFunctionStops[zoom].push([stop[0].value, stop[1]]);\n }\n // the interpolation type for the zoom dimension of a zoom-and-property\n // function is determined directly from the style property specification\n // for which it's being used: linear for interpolatable properties, step\n // otherwise.\n const functionType = getFunctionType({}, propertySpec);\n if (functionType === 'exponential') {\n const expression = [getInterpolateOperator(parameters), ['linear'], ['zoom']];\n for (const z of zoomStops) {\n const output = convertPropertyFunction(featureFunctionParameters[z], propertySpec, featureFunctionStops[z]);\n appendStopPair(expression, z, output, false);\n }\n return expression;\n }\n else {\n const expression = ['step', ['zoom']];\n for (const z of zoomStops) {\n const output = convertPropertyFunction(featureFunctionParameters[z], propertySpec, featureFunctionStops[z]);\n appendStopPair(expression, z, output, true);\n }\n fixupDegenerateStepCurve(expression);\n return expression;\n }\n}\nfunction coalesce(a, b) {\n if (a !== undefined)\n return a;\n if (b !== undefined)\n return b;\n}\nfunction getFallback(parameters, propertySpec) {\n const defaultValue = convertLiteral(coalesce(parameters.default, propertySpec.default));\n /*\n * Some fields with type: resolvedImage have an undefined default.\n * Because undefined is an invalid value for resolvedImage, set fallback to\n * an empty string instead of undefined to ensure output\n * passes validation.\n */\n if (defaultValue === undefined && propertySpec.type === 'resolvedImage') {\n return '';\n }\n return defaultValue;\n}\nfunction convertPropertyFunction(parameters, propertySpec, stops) {\n const type = getFunctionType(parameters, propertySpec);\n const get = ['get', parameters.property];\n if (type === 'categorical' && typeof stops[0][0] === 'boolean') {\n const expression = ['case'];\n for (const stop of stops) {\n expression.push(['==', get, stop[0]], stop[1]);\n }\n expression.push(getFallback(parameters, propertySpec));\n return expression;\n }\n else if (type === 'categorical') {\n const expression = ['match', get];\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], false);\n }\n expression.push(getFallback(parameters, propertySpec));\n return expression;\n }\n else if (type === 'interval') {\n const expression = ['step', ['number', get]];\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], true);\n }\n fixupDegenerateStepCurve(expression);\n return parameters.default === undefined ? expression : [\n 'case',\n ['==', ['typeof', get], 'number'],\n expression,\n convertLiteral(parameters.default)\n ];\n }\n else if (type === 'exponential') {\n const base = parameters.base !== undefined ? parameters.base : 1;\n const expression = [\n getInterpolateOperator(parameters),\n base === 1 ? ['linear'] : ['exponential', base],\n ['number', get]\n ];\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], false);\n }\n return parameters.default === undefined ? expression : [\n 'case',\n ['==', ['typeof', get], 'number'],\n expression,\n convertLiteral(parameters.default)\n ];\n }\n else {\n throw new Error(`Unknown property function type ${type}`);\n }\n}\nfunction convertZoomFunction(parameters, propertySpec, stops, input = ['zoom']) {\n const type = getFunctionType(parameters, propertySpec);\n let expression;\n let isStep = false;\n if (type === 'interval') {\n expression = ['step', input];\n isStep = true;\n }\n else if (type === 'exponential') {\n const base = parameters.base !== undefined ? parameters.base : 1;\n expression = [getInterpolateOperator(parameters), base === 1 ? ['linear'] : ['exponential', base], input];\n }\n else {\n throw new Error(`Unknown zoom function type \"${type}\"`);\n }\n for (const stop of stops) {\n appendStopPair(expression, stop[0], stop[1], isStep);\n }\n fixupDegenerateStepCurve(expression);\n return expression;\n}\nfunction fixupDegenerateStepCurve(expression) {\n // degenerate step curve (i.e. a constant function): add a noop stop\n if (expression[0] === 'step' && expression.length === 3) {\n expression.push(0);\n expression.push(expression[3]);\n }\n}\nfunction appendStopPair(curve, input, output, isStep) {\n // Skip duplicate stop values. They were not validated for functions, but they are for expressions.\n // https://github.com/mapbox/mapbox-gl-js/issues/4107\n if (curve.length > 3 && input === curve[curve.length - 2]) {\n return;\n }\n // step curves don't get the first input value, as it is redundant.\n if (!(isStep && curve.length === 2)) {\n curve.push(input);\n }\n curve.push(output);\n}\nfunction getFunctionType(parameters, propertySpec) {\n if (parameters.type) {\n return parameters.type;\n }\n else {\n return propertySpec.expression.interpolated ? 'exponential' : 'interval';\n }\n}\n// \"String with {name} token\" => [\"concat\", \"String with \", [\"get\", \"name\"], \" token\"]\nfunction convertTokenString(s) {\n const result = ['concat'];\n const re = /{([^{}]+)}/g;\n let pos = 0;\n for (let match = re.exec(s); match !== null; match = re.exec(s)) {\n const literal = s.slice(pos, re.lastIndex - match[0].length);\n pos = re.lastIndex;\n if (literal.length > 0)\n result.push(literal);\n result.push(['get', match[1]]);\n }\n if (result.length === 1) {\n return s;\n }\n if (pos < s.length) {\n result.push(s.slice(pos));\n }\n else if (result.length === 2) {\n return ['to-string', result[1]];\n }\n return result;\n}\n\nfunction getPropertyReference(propertyName) {\n for (let i = 0; i < v8Spec.layout.length; i++) {\n for (const key in v8Spec[v8Spec.layout[i]]) {\n if (key === propertyName)\n return v8Spec[v8Spec.layout[i]][key];\n }\n }\n for (let i = 0; i < v8Spec.paint.length; i++) {\n for (const key in v8Spec[v8Spec.paint[i]]) {\n if (key === propertyName)\n return v8Spec[v8Spec.paint[i]][key];\n }\n }\n return null;\n}\nfunction eachSource(style, callback) {\n for (const k in style.sources) {\n callback(style.sources[k]);\n }\n}\nfunction eachLayer(style, callback) {\n for (const layer of style.layers) {\n callback(layer);\n }\n}\nfunction eachProperty(style, options, callback) {\n function inner(layer, propertyType) {\n const properties = layer[propertyType];\n if (!properties)\n return;\n Object.keys(properties).forEach((key) => {\n callback({\n path: [layer.id, propertyType, key],\n key,\n value: properties[key],\n reference: getPropertyReference(key),\n set(x) {\n properties[key] = x;\n }\n });\n });\n }\n eachLayer(style, (layer) => {\n if (options.paint) {\n inner(layer, 'paint');\n }\n if (options.layout) {\n inner(layer, 'layout');\n }\n });\n}\n\nfunction stringify(obj) {\n const type = typeof obj;\n if (type === 'number' || type === 'boolean' || type === 'string' || obj === undefined || obj === null)\n return JSON.stringify(obj);\n if (Array.isArray(obj)) {\n let str = '[';\n for (const val of obj) {\n str += `${stringify(val)},`;\n }\n return `${str}]`;\n }\n const keys = Object.keys(obj).sort();\n let str = '{';\n for (let i = 0; i < keys.length; i++) {\n str += `${JSON.stringify(keys[i])}:${stringify(obj[keys[i]])},`;\n }\n return `${str}}`;\n}\nfunction getKey(layer) {\n let key = '';\n for (const k of refProperties) {\n key += `/${stringify(layer[k])}`;\n }\n return key;\n}\n/**\n * Given an array of layers, return an array of arrays of layers where all\n * layers in each group have identical layout-affecting properties. These\n * are the properties that were formerly used by explicit `ref` mechanism\n * for layers: 'type', 'source', 'source-layer', 'minzoom', 'maxzoom',\n * 'filter', and 'layout'.\n *\n * The input is not modified. The output layers are references to the\n * input layers.\n *\n * @private\n * @param {Array} layers\n * @param {Object} [cachedKeys] - an object to keep already calculated keys.\n * @returns {Array>}\n */\nfunction groupByLayout(layers, cachedKeys) {\n const groups = {};\n for (let i = 0; i < layers.length; i++) {\n const k = (cachedKeys && cachedKeys[layers[i].id]) || getKey(layers[i]);\n // update the cache if there is one\n if (cachedKeys)\n cachedKeys[layers[i].id] = k;\n let group = groups[k];\n if (!group) {\n group = groups[k] = [];\n }\n group.push(layers[i]);\n }\n const result = [];\n for (const k in groups) {\n result.push(groups[k]);\n }\n return result;\n}\n\nfunction emptyStyle() {\n const style = {};\n const version = v8Spec['$version'];\n for (const styleKey in v8Spec['$root']) {\n const spec = v8Spec['$root'][styleKey];\n if (spec.required) {\n let value = null;\n if (styleKey === 'version') {\n value = version;\n }\n else {\n if (spec.type === 'array') {\n value = [];\n }\n else {\n value = {};\n }\n }\n if (value != null) {\n style[styleKey] = value;\n }\n }\n }\n return style;\n}\n\nfunction validateConstants(options) {\n const key = options.key;\n const constants = options.value;\n if (constants) {\n return [new ValidationError(key, constants, 'constants have been deprecated as of v8')];\n }\n else {\n return [];\n }\n}\n\n// Turn jsonlint-lines-primitives objects into primitive objects\nfunction unbundle(value) {\n if (value instanceof Number || value instanceof String || value instanceof Boolean) {\n return value.valueOf();\n }\n else {\n return value;\n }\n}\nfunction deepUnbundle(value) {\n if (Array.isArray(value)) {\n return value.map(deepUnbundle);\n }\n else if (value instanceof Object && !(value instanceof Number || value instanceof String || value instanceof Boolean)) {\n const unbundledValue = {};\n for (const key in value) {\n unbundledValue[key] = deepUnbundle(value[key]);\n }\n return unbundledValue;\n }\n return unbundle(value);\n}\n\nfunction validateObject(options) {\n const key = options.key;\n const object = options.value;\n const elementSpecs = options.valueSpec || {};\n const elementValidators = options.objectElementValidators || {};\n const style = options.style;\n const styleSpec = options.styleSpec;\n const validateSpec = options.validateSpec;\n let errors = [];\n const type = getType(object);\n if (type !== 'object') {\n return [new ValidationError(key, object, `object expected, ${type} found`)];\n }\n for (const objectKey in object) {\n const elementSpecKey = objectKey.split('.')[0]; // treat 'paint.*' as 'paint'\n const elementSpec = elementSpecs[elementSpecKey] || elementSpecs['*'];\n let validateElement;\n if (elementValidators[elementSpecKey]) {\n validateElement = elementValidators[elementSpecKey];\n }\n else if (elementSpecs[elementSpecKey]) {\n validateElement = validateSpec;\n }\n else if (elementValidators['*']) {\n validateElement = elementValidators['*'];\n }\n else if (elementSpecs['*']) {\n validateElement = validateSpec;\n }\n else {\n errors.push(new ValidationError(key, object[objectKey], `unknown property \"${objectKey}\"`));\n continue;\n }\n errors = errors.concat(validateElement({\n key: (key ? `${key}.` : key) + objectKey,\n value: object[objectKey],\n valueSpec: elementSpec,\n style,\n styleSpec,\n object,\n objectKey,\n validateSpec,\n }, object));\n }\n for (const elementSpecKey in elementSpecs) {\n // Don't check `required` when there's a custom validator for that property.\n if (elementValidators[elementSpecKey]) {\n continue;\n }\n if (elementSpecs[elementSpecKey].required && elementSpecs[elementSpecKey]['default'] === undefined && object[elementSpecKey] === undefined) {\n errors.push(new ValidationError(key, object, `missing required property \"${elementSpecKey}\"`));\n }\n }\n return errors;\n}\n\nfunction validateArray(options) {\n const array = options.value;\n const arraySpec = options.valueSpec;\n const validateSpec = options.validateSpec;\n const style = options.style;\n const styleSpec = options.styleSpec;\n const key = options.key;\n const validateArrayElement = options.arrayElementValidator || validateSpec;\n if (getType(array) !== 'array') {\n return [new ValidationError(key, array, `array expected, ${getType(array)} found`)];\n }\n if (arraySpec.length && array.length !== arraySpec.length) {\n return [new ValidationError(key, array, `array length ${arraySpec.length} expected, length ${array.length} found`)];\n }\n if (arraySpec['min-length'] && array.length < arraySpec['min-length']) {\n return [new ValidationError(key, array, `array length at least ${arraySpec['min-length']} expected, length ${array.length} found`)];\n }\n let arrayElementSpec = {\n 'type': arraySpec.value,\n 'values': arraySpec.values\n };\n if (styleSpec.$version < 7) {\n arrayElementSpec['function'] = arraySpec.function;\n }\n if (getType(arraySpec.value) === 'object') {\n arrayElementSpec = arraySpec.value;\n }\n let errors = [];\n for (let i = 0; i < array.length; i++) {\n errors = errors.concat(validateArrayElement({\n array,\n arrayIndex: i,\n value: array[i],\n valueSpec: arrayElementSpec,\n validateSpec: options.validateSpec,\n style,\n styleSpec,\n key: `${key}[${i}]`\n }));\n }\n return errors;\n}\n\nfunction validateNumber(options) {\n const key = options.key;\n const value = options.value;\n const valueSpec = options.valueSpec;\n let type = getType(value);\n // eslint-disable-next-line no-self-compare\n if (type === 'number' && value !== value) {\n type = 'NaN';\n }\n if (type !== 'number') {\n return [new ValidationError(key, value, `number expected, ${type} found`)];\n }\n if ('minimum' in valueSpec && value < valueSpec.minimum) {\n return [new ValidationError(key, value, `${value} is less than the minimum value ${valueSpec.minimum}`)];\n }\n if ('maximum' in valueSpec && value > valueSpec.maximum) {\n return [new ValidationError(key, value, `${value} is greater than the maximum value ${valueSpec.maximum}`)];\n }\n return [];\n}\n\nfunction validateFunction(options) {\n const functionValueSpec = options.valueSpec;\n const functionType = unbundle(options.value.type);\n let stopKeyType;\n let stopDomainValues = {};\n let previousStopDomainValue;\n let previousStopDomainZoom;\n const isZoomFunction = functionType !== 'categorical' && options.value.property === undefined;\n const isPropertyFunction = !isZoomFunction;\n const isZoomAndPropertyFunction = getType(options.value.stops) === 'array' &&\n getType(options.value.stops[0]) === 'array' &&\n getType(options.value.stops[0][0]) === 'object';\n const errors = validateObject({\n key: options.key,\n value: options.value,\n valueSpec: options.styleSpec.function,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec,\n objectElementValidators: {\n stops: validateFunctionStops,\n default: validateFunctionDefault\n }\n });\n if (functionType === 'identity' && isZoomFunction) {\n errors.push(new ValidationError(options.key, options.value, 'missing required property \"property\"'));\n }\n if (functionType !== 'identity' && !options.value.stops) {\n errors.push(new ValidationError(options.key, options.value, 'missing required property \"stops\"'));\n }\n if (functionType === 'exponential' && options.valueSpec.expression && !supportsInterpolation(options.valueSpec)) {\n errors.push(new ValidationError(options.key, options.value, 'exponential functions not supported'));\n }\n if (options.styleSpec.$version >= 8) {\n if (isPropertyFunction && !supportsPropertyExpression(options.valueSpec)) {\n errors.push(new ValidationError(options.key, options.value, 'property functions not supported'));\n }\n else if (isZoomFunction && !supportsZoomExpression(options.valueSpec)) {\n errors.push(new ValidationError(options.key, options.value, 'zoom functions not supported'));\n }\n }\n if ((functionType === 'categorical' || isZoomAndPropertyFunction) && options.value.property === undefined) {\n errors.push(new ValidationError(options.key, options.value, '\"property\" property is required'));\n }\n return errors;\n function validateFunctionStops(options) {\n if (functionType === 'identity') {\n return [new ValidationError(options.key, options.value, 'identity function may not have a \"stops\" property')];\n }\n let errors = [];\n const value = options.value;\n errors = errors.concat(validateArray({\n key: options.key,\n value,\n valueSpec: options.valueSpec,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec,\n arrayElementValidator: validateFunctionStop\n }));\n if (getType(value) === 'array' && value.length === 0) {\n errors.push(new ValidationError(options.key, value, 'array must have at least one stop'));\n }\n return errors;\n }\n function validateFunctionStop(options) {\n let errors = [];\n const value = options.value;\n const key = options.key;\n if (getType(value) !== 'array') {\n return [new ValidationError(key, value, `array expected, ${getType(value)} found`)];\n }\n if (value.length !== 2) {\n return [new ValidationError(key, value, `array length 2 expected, length ${value.length} found`)];\n }\n if (isZoomAndPropertyFunction) {\n if (getType(value[0]) !== 'object') {\n return [new ValidationError(key, value, `object expected, ${getType(value[0])} found`)];\n }\n if (value[0].zoom === undefined) {\n return [new ValidationError(key, value, 'object stop key must have zoom')];\n }\n if (value[0].value === undefined) {\n return [new ValidationError(key, value, 'object stop key must have value')];\n }\n if (previousStopDomainZoom && previousStopDomainZoom > unbundle(value[0].zoom)) {\n return [new ValidationError(key, value[0].zoom, 'stop zoom values must appear in ascending order')];\n }\n if (unbundle(value[0].zoom) !== previousStopDomainZoom) {\n previousStopDomainZoom = unbundle(value[0].zoom);\n previousStopDomainValue = undefined;\n stopDomainValues = {};\n }\n errors = errors.concat(validateObject({\n key: `${key}[0]`,\n value: value[0],\n valueSpec: { zoom: {} },\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec,\n objectElementValidators: { zoom: validateNumber, value: validateStopDomainValue }\n }));\n }\n else {\n errors = errors.concat(validateStopDomainValue({\n key: `${key}[0]`,\n value: value[0],\n valueSpec: {},\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec\n }, value));\n }\n if (isExpression(deepUnbundle(value[1]))) {\n return errors.concat([new ValidationError(`${key}[1]`, value[1], 'expressions are not allowed in function stops.')]);\n }\n return errors.concat(options.validateSpec({\n key: `${key}[1]`,\n value: value[1],\n valueSpec: functionValueSpec,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec\n }));\n }\n function validateStopDomainValue(options, stop) {\n const type = getType(options.value);\n const value = unbundle(options.value);\n const reportValue = options.value !== null ? options.value : stop;\n if (!stopKeyType) {\n stopKeyType = type;\n }\n else if (type !== stopKeyType) {\n return [new ValidationError(options.key, reportValue, `${type} stop domain type must match previous stop domain type ${stopKeyType}`)];\n }\n if (type !== 'number' && type !== 'string' && type !== 'boolean') {\n return [new ValidationError(options.key, reportValue, 'stop domain value must be a number, string, or boolean')];\n }\n if (type !== 'number' && functionType !== 'categorical') {\n let message = `number expected, ${type} found`;\n if (supportsPropertyExpression(functionValueSpec) && functionType === undefined) {\n message += '\\nIf you intended to use a categorical function, specify `\"type\": \"categorical\"`.';\n }\n return [new ValidationError(options.key, reportValue, message)];\n }\n if (functionType === 'categorical' && type === 'number' && (!isFinite(value) || Math.floor(value) !== value)) {\n return [new ValidationError(options.key, reportValue, `integer expected, found ${value}`)];\n }\n if (functionType !== 'categorical' && type === 'number' && previousStopDomainValue !== undefined && value < previousStopDomainValue) {\n return [new ValidationError(options.key, reportValue, 'stop domain values must appear in ascending order')];\n }\n else {\n previousStopDomainValue = value;\n }\n if (functionType === 'categorical' && value in stopDomainValues) {\n return [new ValidationError(options.key, reportValue, 'stop domain values must be unique')];\n }\n else {\n stopDomainValues[value] = true;\n }\n return [];\n }\n function validateFunctionDefault(options) {\n return options.validateSpec({\n key: options.key,\n value: options.value,\n valueSpec: functionValueSpec,\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec: options.styleSpec\n });\n }\n}\n\nfunction validateExpression(options) {\n const expression = (options.expressionContext === 'property' ? createPropertyExpression : createExpression)(deepUnbundle(options.value), options.valueSpec);\n if (expression.result === 'error') {\n return expression.value.map((error) => {\n return new ValidationError(`${options.key}${error.key}`, options.value, error.message);\n });\n }\n const expressionObj = expression.value.expression || expression.value._styleExpression.expression;\n if (options.expressionContext === 'property' && (options.propertyKey === 'text-font') &&\n !expressionObj.outputDefined()) {\n return [new ValidationError(options.key, options.value, `Invalid data expression for \"${options.propertyKey}\". Output values must be contained as literals within the expression.`)];\n }\n if (options.expressionContext === 'property' && options.propertyType === 'layout' &&\n (!isStateConstant(expressionObj))) {\n return [new ValidationError(options.key, options.value, '\"feature-state\" data expressions are not supported with layout properties.')];\n }\n if (options.expressionContext === 'filter' && !isStateConstant(expressionObj)) {\n return [new ValidationError(options.key, options.value, '\"feature-state\" data expressions are not supported with filters.')];\n }\n if (options.expressionContext && options.expressionContext.indexOf('cluster') === 0) {\n if (!isGlobalPropertyConstant(expressionObj, ['zoom', 'feature-state'])) {\n return [new ValidationError(options.key, options.value, '\"zoom\" and \"feature-state\" expressions are not supported with cluster properties.')];\n }\n if (options.expressionContext === 'cluster-initial' && !isFeatureConstant(expressionObj)) {\n return [new ValidationError(options.key, options.value, 'Feature data expressions are not supported with initial expression part of cluster properties.')];\n }\n }\n return [];\n}\n\nfunction validateBoolean(options) {\n const value = options.value;\n const key = options.key;\n const type = getType(value);\n if (type !== 'boolean') {\n return [new ValidationError(key, value, `boolean expected, ${type} found`)];\n }\n return [];\n}\n\nfunction validateColor(options) {\n const key = options.key;\n const value = options.value;\n const type = getType(value);\n if (type !== 'string') {\n return [new ValidationError(key, value, `color expected, ${type} found`)];\n }\n if (!Color.parse(String(value))) { // cast String object to string primitive\n return [new ValidationError(key, value, `color expected, \"${value}\" found`)];\n }\n return [];\n}\n\nfunction validateEnum(options) {\n const key = options.key;\n const value = options.value;\n const valueSpec = options.valueSpec;\n const errors = [];\n if (Array.isArray(valueSpec.values)) { // <=v7\n if (valueSpec.values.indexOf(unbundle(value)) === -1) {\n errors.push(new ValidationError(key, value, `expected one of [${valueSpec.values.join(', ')}], ${JSON.stringify(value)} found`));\n }\n }\n else { // >=v8\n if (Object.keys(valueSpec.values).indexOf(unbundle(value)) === -1) {\n errors.push(new ValidationError(key, value, `expected one of [${Object.keys(valueSpec.values).join(', ')}], ${JSON.stringify(value)} found`));\n }\n }\n return errors;\n}\n\nfunction validateFilter(options) {\n if (isExpressionFilter(deepUnbundle(options.value))) {\n return validateExpression(extendBy({}, options, {\n expressionContext: 'filter',\n valueSpec: { value: 'boolean' }\n }));\n }\n else {\n return validateNonExpressionFilter(options);\n }\n}\nfunction validateNonExpressionFilter(options) {\n const value = options.value;\n const key = options.key;\n if (getType(value) !== 'array') {\n return [new ValidationError(key, value, `array expected, ${getType(value)} found`)];\n }\n const styleSpec = options.styleSpec;\n let type;\n let errors = [];\n if (value.length < 1) {\n return [new ValidationError(key, value, 'filter array must have at least 1 element')];\n }\n errors = errors.concat(validateEnum({\n key: `${key}[0]`,\n value: value[0],\n valueSpec: styleSpec.filter_operator,\n style: options.style,\n styleSpec: options.styleSpec\n }));\n switch (unbundle(value[0])) {\n case '<':\n case '<=':\n case '>':\n case '>=':\n if (value.length >= 2 && unbundle(value[1]) === '$type') {\n errors.push(new ValidationError(key, value, `\"$type\" cannot be use with operator \"${value[0]}\"`));\n }\n /* falls through */\n case '==':\n case '!=':\n if (value.length !== 3) {\n errors.push(new ValidationError(key, value, `filter array for operator \"${value[0]}\" must have 3 elements`));\n }\n /* falls through */\n case 'in':\n case '!in':\n if (value.length >= 2) {\n type = getType(value[1]);\n if (type !== 'string') {\n errors.push(new ValidationError(`${key}[1]`, value[1], `string expected, ${type} found`));\n }\n }\n for (let i = 2; i < value.length; i++) {\n type = getType(value[i]);\n if (unbundle(value[1]) === '$type') {\n errors = errors.concat(validateEnum({\n key: `${key}[${i}]`,\n value: value[i],\n valueSpec: styleSpec.geometry_type,\n style: options.style,\n styleSpec: options.styleSpec\n }));\n }\n else if (type !== 'string' && type !== 'number' && type !== 'boolean') {\n errors.push(new ValidationError(`${key}[${i}]`, value[i], `string, number, or boolean expected, ${type} found`));\n }\n }\n break;\n case 'any':\n case 'all':\n case 'none':\n for (let i = 1; i < value.length; i++) {\n errors = errors.concat(validateNonExpressionFilter({\n key: `${key}[${i}]`,\n value: value[i],\n style: options.style,\n styleSpec: options.styleSpec\n }));\n }\n break;\n case 'has':\n case '!has':\n type = getType(value[1]);\n if (value.length !== 2) {\n errors.push(new ValidationError(key, value, `filter array for \"${value[0]}\" operator must have 2 elements`));\n }\n else if (type !== 'string') {\n errors.push(new ValidationError(`${key}[1]`, value[1], `string expected, ${type} found`));\n }\n break;\n case 'within':\n type = getType(value[1]);\n if (value.length !== 2) {\n errors.push(new ValidationError(key, value, `filter array for \"${value[0]}\" operator must have 2 elements`));\n }\n else if (type !== 'object') {\n errors.push(new ValidationError(`${key}[1]`, value[1], `object expected, ${type} found`));\n }\n break;\n }\n return errors;\n}\n\nfunction validateProperty(options, propertyType) {\n const key = options.key;\n const validateSpec = options.validateSpec;\n const style = options.style;\n const styleSpec = options.styleSpec;\n const value = options.value;\n const propertyKey = options.objectKey;\n const layerSpec = styleSpec[`${propertyType}_${options.layerType}`];\n if (!layerSpec)\n return [];\n const transitionMatch = propertyKey.match(/^(.*)-transition$/);\n if (propertyType === 'paint' && transitionMatch && layerSpec[transitionMatch[1]] && layerSpec[transitionMatch[1]].transition) {\n return validateSpec({\n key,\n value,\n valueSpec: styleSpec.transition,\n style,\n styleSpec\n });\n }\n const valueSpec = options.valueSpec || layerSpec[propertyKey];\n if (!valueSpec) {\n return [new ValidationError(key, value, `unknown property \"${propertyKey}\"`)];\n }\n let tokenMatch;\n if (getType(value) === 'string' && supportsPropertyExpression(valueSpec) && !valueSpec.tokens && (tokenMatch = /^{([^}]+)}$/.exec(value))) {\n return [new ValidationError(key, value, `\"${propertyKey}\" does not support interpolation syntax\\n` +\n `Use an identity property function instead: \\`{ \"type\": \"identity\", \"property\": ${JSON.stringify(tokenMatch[1])} }\\`.`)];\n }\n const errors = [];\n if (options.layerType === 'symbol') {\n if (propertyKey === 'text-field' && style && !style.glyphs) {\n errors.push(new ValidationError(key, value, 'use of \"text-field\" requires a style \"glyphs\" property'));\n }\n if (propertyKey === 'text-font' && isFunction(deepUnbundle(value)) && unbundle(value.type) === 'identity') {\n errors.push(new ValidationError(key, value, '\"text-font\" does not support identity functions'));\n }\n }\n return errors.concat(validateSpec({\n key: options.key,\n value,\n valueSpec,\n style,\n styleSpec,\n expressionContext: 'property',\n propertyType,\n propertyKey\n }));\n}\n\nfunction validatePaintProperty(options) {\n return validateProperty(options, 'paint');\n}\n\nfunction validateLayoutProperty(options) {\n return validateProperty(options, 'layout');\n}\n\nfunction validateLayer(options) {\n let errors = [];\n const layer = options.value;\n const key = options.key;\n const style = options.style;\n const styleSpec = options.styleSpec;\n if (!layer.type && !layer.ref) {\n errors.push(new ValidationError(key, layer, 'either \"type\" or \"ref\" is required'));\n }\n let type = unbundle(layer.type);\n const ref = unbundle(layer.ref);\n if (layer.id) {\n const layerId = unbundle(layer.id);\n for (let i = 0; i < options.arrayIndex; i++) {\n const otherLayer = style.layers[i];\n if (unbundle(otherLayer.id) === layerId) {\n errors.push(new ValidationError(key, layer.id, `duplicate layer id \"${layer.id}\", previously used at line ${otherLayer.id.__line__}`));\n }\n }\n }\n if ('ref' in layer) {\n ['type', 'source', 'source-layer', 'filter', 'layout'].forEach((p) => {\n if (p in layer) {\n errors.push(new ValidationError(key, layer[p], `\"${p}\" is prohibited for ref layers`));\n }\n });\n let parent;\n style.layers.forEach((layer) => {\n if (unbundle(layer.id) === ref)\n parent = layer;\n });\n if (!parent) {\n errors.push(new ValidationError(key, layer.ref, `ref layer \"${ref}\" not found`));\n }\n else if (parent.ref) {\n errors.push(new ValidationError(key, layer.ref, 'ref cannot reference another ref layer'));\n }\n else {\n type = unbundle(parent.type);\n }\n }\n else if (type !== 'background') {\n if (!layer.source) {\n errors.push(new ValidationError(key, layer, 'missing required property \"source\"'));\n }\n else {\n const source = style.sources && style.sources[layer.source];\n const sourceType = source && unbundle(source.type);\n if (!source) {\n errors.push(new ValidationError(key, layer.source, `source \"${layer.source}\" not found`));\n }\n else if (sourceType === 'vector' && type === 'raster') {\n errors.push(new ValidationError(key, layer.source, `layer \"${layer.id}\" requires a raster source`));\n }\n else if (sourceType !== 'raster-dem' && type === 'hillshade') {\n errors.push(new ValidationError(key, layer.source, `layer \"${layer.id}\" requires a raster-dem source`));\n }\n else if (sourceType === 'raster' && type !== 'raster') {\n errors.push(new ValidationError(key, layer.source, `layer \"${layer.id}\" requires a vector source`));\n }\n else if (sourceType === 'vector' && !layer['source-layer']) {\n errors.push(new ValidationError(key, layer, `layer \"${layer.id}\" must specify a \"source-layer\"`));\n }\n else if (sourceType === 'raster-dem' && type !== 'hillshade') {\n errors.push(new ValidationError(key, layer.source, 'raster-dem source can only be used with layer type \\'hillshade\\'.'));\n }\n else if (type === 'line' && layer.paint && layer.paint['line-gradient'] &&\n (sourceType !== 'geojson' || !source.lineMetrics)) {\n errors.push(new ValidationError(key, layer, `layer \"${layer.id}\" specifies a line-gradient, which requires a GeoJSON source with \\`lineMetrics\\` enabled.`));\n }\n }\n }\n errors = errors.concat(validateObject({\n key,\n value: layer,\n valueSpec: styleSpec.layer,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n objectElementValidators: {\n '*'() {\n return [];\n },\n // We don't want to enforce the spec's `\"requires\": true` for backward compatibility with refs;\n // the actual requirement is validated above. See https://github.com/mapbox/mapbox-gl-js/issues/5772.\n type() {\n return options.validateSpec({\n key: `${key}.type`,\n value: layer.type,\n valueSpec: styleSpec.layer.type,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n object: layer,\n objectKey: 'type'\n });\n },\n filter: validateFilter,\n layout(options) {\n return validateObject({\n layer,\n key: options.key,\n value: options.value,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n objectElementValidators: {\n '*'(options) {\n return validateLayoutProperty(extendBy({ layerType: type }, options));\n }\n }\n });\n },\n paint(options) {\n return validateObject({\n layer,\n key: options.key,\n value: options.value,\n style: options.style,\n styleSpec: options.styleSpec,\n validateSpec: options.validateSpec,\n objectElementValidators: {\n '*'(options) {\n return validatePaintProperty(extendBy({ layerType: type }, options));\n }\n }\n });\n }\n }\n }));\n return errors;\n}\n\nfunction validateString(options) {\n const value = options.value;\n const key = options.key;\n const type = getType(value);\n if (type !== 'string') {\n return [new ValidationError(key, value, `string expected, ${type} found`)];\n }\n return [];\n}\n\nfunction validateRasterDEMSource(options) {\n var _a;\n const sourceName = (_a = options.sourceName) !== null && _a !== void 0 ? _a : '';\n const rasterDEM = options.value;\n const styleSpec = options.styleSpec;\n const rasterDEMSpec = styleSpec.source_raster_dem;\n const style = options.style;\n let errors = [];\n const rootType = getType(rasterDEM);\n if (rasterDEM === undefined) {\n return errors;\n }\n else if (rootType !== 'object') {\n errors.push(new ValidationError('source_raster_dem', rasterDEM, `object expected, ${rootType} found`));\n return errors;\n }\n const encoding = unbundle(rasterDEM.encoding);\n const isCustomEncoding = encoding === 'custom';\n const customEncodingKeys = ['redFactor', 'greenFactor', 'blueFactor', 'baseShift'];\n const encodingName = options.value.encoding ? `\"${options.value.encoding}\"` : 'Default';\n for (const key in rasterDEM) {\n if (!isCustomEncoding && customEncodingKeys.includes(key)) {\n errors.push(new ValidationError(key, rasterDEM[key], `In \"${sourceName}\": \"${key}\" is only valid when \"encoding\" is set to \"custom\". ${encodingName} encoding found`));\n }\n else if (rasterDEMSpec[key]) {\n errors = errors.concat(options.validateSpec({\n key,\n value: rasterDEM[key],\n valueSpec: rasterDEMSpec[key],\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else {\n errors.push(new ValidationError(key, rasterDEM[key], `unknown property \"${key}\"`));\n }\n }\n return errors;\n}\n\nconst objectElementValidators = {\n promoteId: validatePromoteId\n};\nfunction validateSource(options) {\n const value = options.value;\n const key = options.key;\n const styleSpec = options.styleSpec;\n const style = options.style;\n const validateSpec = options.validateSpec;\n if (!value.type) {\n return [new ValidationError(key, value, '\"type\" is required')];\n }\n const type = unbundle(value.type);\n let errors;\n switch (type) {\n case 'vector':\n case 'raster':\n errors = validateObject({\n key,\n value,\n valueSpec: styleSpec[`source_${type.replace('-', '_')}`],\n style: options.style,\n styleSpec,\n objectElementValidators,\n validateSpec,\n });\n return errors;\n case 'raster-dem':\n errors = validateRasterDEMSource({\n sourceName: key,\n value,\n style: options.style,\n styleSpec,\n validateSpec,\n });\n return errors;\n case 'geojson':\n errors = validateObject({\n key,\n value,\n valueSpec: styleSpec.source_geojson,\n style,\n styleSpec,\n validateSpec,\n objectElementValidators\n });\n if (value.cluster) {\n for (const prop in value.clusterProperties) {\n const [operator, mapExpr] = value.clusterProperties[prop];\n const reduceExpr = typeof operator === 'string' ? [operator, ['accumulated'], ['get', prop]] : operator;\n errors.push(...validateExpression({\n key: `${key}.${prop}.map`,\n value: mapExpr,\n validateSpec,\n expressionContext: 'cluster-map'\n }));\n errors.push(...validateExpression({\n key: `${key}.${prop}.reduce`,\n value: reduceExpr,\n validateSpec,\n expressionContext: 'cluster-reduce'\n }));\n }\n }\n return errors;\n case 'video':\n return validateObject({\n key,\n value,\n valueSpec: styleSpec.source_video,\n style,\n validateSpec,\n styleSpec\n });\n case 'image':\n return validateObject({\n key,\n value,\n valueSpec: styleSpec.source_image,\n style,\n validateSpec,\n styleSpec\n });\n case 'canvas':\n return [new ValidationError(key, null, 'Please use runtime APIs to add canvas sources, rather than including them in stylesheets.', 'source.canvas')];\n default:\n return validateEnum({\n key: `${key}.type`,\n value: value.type,\n valueSpec: { values: ['vector', 'raster', 'raster-dem', 'geojson', 'video', 'image'] },\n style,\n validateSpec,\n styleSpec\n });\n }\n}\nfunction validatePromoteId({ key, value }) {\n if (getType(value) === 'string') {\n return validateString({ key, value });\n }\n else {\n const errors = [];\n for (const prop in value) {\n errors.push(...validateString({ key: `${key}.${prop}`, value: value[prop] }));\n }\n return errors;\n }\n}\n\nfunction validateLight(options) {\n const light = options.value;\n const styleSpec = options.styleSpec;\n const lightSpec = styleSpec.light;\n const style = options.style;\n let errors = [];\n const rootType = getType(light);\n if (light === undefined) {\n return errors;\n }\n else if (rootType !== 'object') {\n errors = errors.concat([new ValidationError('light', light, `object expected, ${rootType} found`)]);\n return errors;\n }\n for (const key in light) {\n const transitionMatch = key.match(/^(.*)-transition$/);\n if (transitionMatch && lightSpec[transitionMatch[1]] && lightSpec[transitionMatch[1]].transition) {\n errors = errors.concat(options.validateSpec({\n key,\n value: light[key],\n valueSpec: styleSpec.transition,\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else if (lightSpec[key]) {\n errors = errors.concat(options.validateSpec({\n key,\n value: light[key],\n valueSpec: lightSpec[key],\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else {\n errors = errors.concat([new ValidationError(key, light[key], `unknown property \"${key}\"`)]);\n }\n }\n return errors;\n}\n\nfunction validateTerrain(options) {\n const terrain = options.value;\n const styleSpec = options.styleSpec;\n const terrainSpec = styleSpec.terrain;\n const style = options.style;\n let errors = [];\n const rootType = getType(terrain);\n if (terrain === undefined) {\n return errors;\n }\n else if (rootType !== 'object') {\n errors = errors.concat([new ValidationError('terrain', terrain, `object expected, ${rootType} found`)]);\n return errors;\n }\n for (const key in terrain) {\n if (terrainSpec[key]) {\n errors = errors.concat(options.validateSpec({\n key,\n value: terrain[key],\n valueSpec: terrainSpec[key],\n validateSpec: options.validateSpec,\n style,\n styleSpec\n }));\n }\n else {\n errors = errors.concat([new ValidationError(key, terrain[key], `unknown property \"${key}\"`)]);\n }\n }\n return errors;\n}\n\nfunction validateFormatted(options) {\n if (validateString(options).length === 0) {\n return [];\n }\n return validateExpression(options);\n}\n\nfunction validateImage(options) {\n if (validateString(options).length === 0) {\n return [];\n }\n return validateExpression(options);\n}\n\nfunction validatePadding(options) {\n const key = options.key;\n const value = options.value;\n const type = getType(value);\n if (type === 'array') {\n if (value.length < 1 || value.length > 4) {\n return [new ValidationError(key, value, `padding requires 1 to 4 values; ${value.length} values found`)];\n }\n const arrayElementSpec = {\n type: 'number'\n };\n let errors = [];\n for (let i = 0; i < value.length; i++) {\n errors = errors.concat(options.validateSpec({\n key: `${key}[${i}]`,\n value: value[i],\n validateSpec: options.validateSpec,\n valueSpec: arrayElementSpec\n }));\n }\n return errors;\n }\n else {\n return validateNumber({\n key,\n value,\n valueSpec: {}\n });\n }\n}\n\nfunction validateVariableAnchorOffsetCollection(options) {\n const key = options.key;\n const value = options.value;\n const type = getType(value);\n const styleSpec = options.styleSpec;\n if (type !== 'array' || value.length < 1 || value.length % 2 !== 0) {\n return [new ValidationError(key, value, 'variableAnchorOffsetCollection requires a non-empty array of even length')];\n }\n let errors = [];\n for (let i = 0; i < value.length; i += 2) {\n // Elements in even positions should be values from text-anchor enum\n errors = errors.concat(validateEnum({\n key: `${key}[${i}]`,\n value: value[i],\n valueSpec: styleSpec['layout_symbol']['text-anchor']\n }));\n // Elements in odd positions should be points (2-element numeric arrays)\n errors = errors.concat(validateArray({\n key: `${key}[${i + 1}]`,\n value: value[i + 1],\n valueSpec: {\n length: 2,\n value: 'number'\n },\n validateSpec: options.validateSpec,\n style: options.style,\n styleSpec\n }));\n }\n return errors;\n}\n\nfunction validateSprite(options) {\n let errors = [];\n const sprite = options.value;\n const key = options.key;\n if (!Array.isArray(sprite)) {\n return validateString({\n key,\n value: sprite\n });\n }\n else {\n const allSpriteIds = [];\n const allSpriteURLs = [];\n for (const i in sprite) {\n if (sprite[i].id && allSpriteIds.includes(sprite[i].id))\n errors.push(new ValidationError(key, sprite, `all the sprites' ids must be unique, but ${sprite[i].id} is duplicated`));\n allSpriteIds.push(sprite[i].id);\n if (sprite[i].url && allSpriteURLs.includes(sprite[i].url))\n errors.push(new ValidationError(key, sprite, `all the sprites' URLs must be unique, but ${sprite[i].url} is duplicated`));\n allSpriteURLs.push(sprite[i].url);\n const pairSpec = {\n id: {\n type: 'string',\n required: true,\n },\n url: {\n type: 'string',\n required: true,\n }\n };\n errors = errors.concat(validateObject({\n key: `${key}[${i}]`,\n value: sprite[i],\n valueSpec: pairSpec,\n validateSpec: options.validateSpec,\n }));\n }\n return errors;\n }\n}\n\nconst VALIDATORS = {\n '*'() {\n return [];\n },\n 'array': validateArray,\n 'boolean': validateBoolean,\n 'number': validateNumber,\n 'color': validateColor,\n 'constants': validateConstants,\n 'enum': validateEnum,\n 'filter': validateFilter,\n 'function': validateFunction,\n 'layer': validateLayer,\n 'object': validateObject,\n 'source': validateSource,\n 'light': validateLight,\n 'terrain': validateTerrain,\n 'string': validateString,\n 'formatted': validateFormatted,\n 'resolvedImage': validateImage,\n 'padding': validatePadding,\n 'variableAnchorOffsetCollection': validateVariableAnchorOffsetCollection,\n 'sprite': validateSprite,\n};\n// Main recursive validation function. Tracks:\n//\n// - key: string representing location of validation in style tree. Used only\n// for more informative error reporting.\n// - value: current value from style being evaluated. May be anything from a\n// high level object that needs to be descended into deeper or a simple\n// scalar value.\n// - valueSpec: current spec being evaluated. Tracks value.\n// - styleSpec: current full spec being evaluated.\nfunction validate(options) {\n const value = options.value;\n const valueSpec = options.valueSpec;\n const styleSpec = options.styleSpec;\n options.validateSpec = validate;\n if (valueSpec.expression && isFunction(unbundle(value))) {\n return validateFunction(options);\n }\n else if (valueSpec.expression && isExpression(deepUnbundle(value))) {\n return validateExpression(options);\n }\n else if (valueSpec.type && VALIDATORS[valueSpec.type]) {\n return VALIDATORS[valueSpec.type](options);\n }\n else {\n const valid = validateObject(extendBy({}, options, {\n valueSpec: valueSpec.type ? styleSpec[valueSpec.type] : valueSpec\n }));\n return valid;\n }\n}\n\nfunction validateGlyphsUrl(options) {\n const value = options.value;\n const key = options.key;\n const errors = validateString(options);\n if (errors.length)\n return errors;\n if (value.indexOf('{fontstack}') === -1) {\n errors.push(new ValidationError(key, value, '\"glyphs\" url must include a \"{fontstack}\" token'));\n }\n if (value.indexOf('{range}') === -1) {\n errors.push(new ValidationError(key, value, '\"glyphs\" url must include a \"{range}\" token'));\n }\n return errors;\n}\n\n/**\n * Validate a MapLibre style against the style specification. This entrypoint,\n * `maplibre-gl-style-spec/lib/validate_style.min`, is designed to produce as\n * small a browserify bundle as possible by omitting unnecessary functionality\n * and legacy style specifications.\n *\n * @private\n * @param {Object} style The style to be validated.\n * @param {Object} [styleSpec] The style specification to validate against.\n * If omitted, the latest style spec is used.\n * @returns {Array}\n * @example\n * var validate = require('maplibre-gl-style-spec/lib/validate_style.min');\n * var errors = validate(style);\n */\nfunction validateStyleMin(style, styleSpec = v8Spec) {\n let errors = [];\n errors = errors.concat(validate({\n key: '',\n value: style,\n valueSpec: styleSpec.$root,\n styleSpec,\n style,\n validateSpec: validate,\n objectElementValidators: {\n glyphs: validateGlyphsUrl,\n '*'() {\n return [];\n }\n }\n }));\n if (style['constants']) {\n errors = errors.concat(validateConstants({\n key: 'constants',\n value: style['constants'],\n style,\n styleSpec,\n validateSpec: validate,\n }));\n }\n return sortErrors(errors);\n}\nvalidateStyleMin.source = wrapCleanErrors(injectValidateSpec(validateSource));\nvalidateStyleMin.sprite = wrapCleanErrors(injectValidateSpec(validateSprite));\nvalidateStyleMin.glyphs = wrapCleanErrors(injectValidateSpec(validateGlyphsUrl));\nvalidateStyleMin.light = wrapCleanErrors(injectValidateSpec(validateLight));\nvalidateStyleMin.terrain = wrapCleanErrors(injectValidateSpec(validateTerrain));\nvalidateStyleMin.layer = wrapCleanErrors(injectValidateSpec(validateLayer));\nvalidateStyleMin.filter = wrapCleanErrors(injectValidateSpec(validateFilter));\nvalidateStyleMin.paintProperty = wrapCleanErrors(injectValidateSpec(validatePaintProperty));\nvalidateStyleMin.layoutProperty = wrapCleanErrors(injectValidateSpec(validateLayoutProperty));\nfunction injectValidateSpec(validator) {\n return function (options) {\n return validator({\n ...options,\n validateSpec: validate,\n });\n };\n}\nfunction sortErrors(errors) {\n return [].concat(errors).sort((a, b) => {\n return a.line - b.line;\n });\n}\nfunction wrapCleanErrors(inner) {\n return function (...args) {\n return sortErrors(inner.apply(this, args));\n };\n}\n\nconst v8 = v8Spec;\nconst expression = {\n StyleExpression,\n StylePropertyFunction,\n ZoomConstantExpression,\n ZoomDependentExpression,\n createExpression,\n createPropertyExpression,\n isExpression,\n isExpressionFilter,\n isZoomExpression,\n normalizePropertyExpression,\n};\nconst styleFunction = {\n convertFunction,\n createFunction,\n isFunction\n};\nconst visit = { eachLayer, eachProperty, eachSource };\n\nexport { Color, ColorType, CompoundExpression, EvaluationContext, FormatExpression, Formatted, FormattedSection, FormattedType, Interpolate, Literal, NullType, Padding, ParsingError, ResolvedImage, Step, StyleExpression, StylePropertyFunction, ValidationError, VariableAnchorOffsetCollection, ZoomConstantExpression, ZoomDependentExpression, convertFilter, convertFunction, createExpression, createFunction, createPropertyExpression, derefLayers, diffStyles as diff, emptyStyle, expression, expressions, createFilter as featureFilter, styleFunction as function, groupByLayout, interpolateFactory, interpolate as interpolates, isExpression, isFunction, isZoomExpression, v8Spec as latest, normalizePropertyExpression, operations, supportsPropertyExpression, toString$1 as toString, typeOf, v8, validateStyleMin, visit };\n//# sourceMappingURL=index.mjs.map\n","import {validateStyleMin} from '@maplibre/maplibre-gl-style-spec';\nimport {ErrorEvent} from '../util/evented';\n\nimport type {Evented} from '../util/evented';\n\ntype ValidationError = {\n message: string;\n line: number;\n identifier?: string;\n};\n\nexport type Validator = (a: any) => ReadonlyArray;\n\ntype ValidateStyle = {\n source: Validator;\n sprite: Validator;\n glyphs: Validator;\n layer: Validator;\n light: Validator;\n terrain: Validator;\n filter: Validator;\n paintProperty: Validator;\n layoutProperty: Validator;\n (b: any, a?: any | null): ReadonlyArray;\n};\n\nexport const validateStyle = (validateStyleMin as unknown as ValidateStyle);\n\nexport const validateSource = validateStyle.source;\nexport const validateLight = validateStyle.light;\nexport const validateTerrain = validateStyle.terrain;\nexport const validateFilter = validateStyle.filter;\nexport const validatePaintProperty = validateStyle.paintProperty;\nexport const validateLayoutProperty = validateStyle.layoutProperty;\n\nexport function emitValidationErrors(\n emitter: Evented,\n errors?: ReadonlyArray<{\n message: string;\n identifier?: string;\n }> | null\n): boolean {\n let hasErrors = false;\n if (errors && errors.length) {\n for (const error of errors) {\n emitter.fire(new ErrorEvent(new Error(error.message)));\n hasErrors = true;\n }\n }\n return hasErrors;\n}\n","/*\nThis file was copied from https://github.com/mapbox/grid-index and was\nmigrated from JavaScript to TypeScript.\n\nCopyright (c) 2016, Mapbox\n\nPermission to use, copy, modify, and/or distribute this software for any purpose\nwith or without fee is hereby granted, provided that the above copyright notice\nand this permission notice appear in all copies.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND\nFITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS\nOF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER\nTORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF\nTHIS SOFTWARE.\n*/\n\nconst NUM_PARAMS = 3;\n\nexport type SerializedGrid = {\n buffer: ArrayBuffer;\n};\n\nexport class TransferableGridIndex {\n cells: number[][];\n arrayBuffer: ArrayBuffer;\n d: number;\n keys: number[];\n bboxes: number[];\n n: number;\n extent: number;\n padding: number;\n scale: any;\n uid: number;\n min: number;\n max: number;\n\n constructor(extent: number | ArrayBuffer, n?: number, padding?: number) {\n const cells = this.cells = [];\n\n if (extent instanceof ArrayBuffer) {\n this.arrayBuffer = extent;\n const array = new Int32Array(this.arrayBuffer);\n extent = array[0];\n n = array[1];\n padding = array[2];\n\n this.d = n + 2 * padding;\n for (let k = 0; k < this.d * this.d; k++) {\n const start = array[NUM_PARAMS + k];\n const end = array[NUM_PARAMS + k + 1];\n cells.push(start === end ? null : array.subarray(start, end));\n }\n const keysOffset = array[NUM_PARAMS + cells.length];\n const bboxesOffset = array[NUM_PARAMS + cells.length + 1];\n this.keys = array.subarray(keysOffset, bboxesOffset) as any as number[];\n this.bboxes = array.subarray(bboxesOffset) as any as number[];\n\n this.insert = this._insertReadonly;\n\n } else {\n this.d = n + 2 * padding;\n for (let i = 0; i < this.d * this.d; i++) {\n cells.push([]);\n }\n this.keys = [];\n this.bboxes = [];\n }\n\n this.n = n;\n this.extent = extent;\n this.padding = padding;\n this.scale = n / extent;\n this.uid = 0;\n\n const p = (padding / n) * extent;\n this.min = -p;\n this.max = extent + p;\n }\n\n insert(key: number, x1: number, y1: number, x2: number, y2: number) {\n this._forEachCell(x1, y1, x2, y2, this._insertCell, this.uid++, undefined, undefined);\n this.keys.push(key);\n this.bboxes.push(x1);\n this.bboxes.push(y1);\n this.bboxes.push(x2);\n this.bboxes.push(y2);\n }\n\n _insertReadonly() {\n throw new Error('Cannot insert into a GridIndex created from an ArrayBuffer.');\n }\n\n _insertCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) {\n this.cells[cellIndex].push(uid);\n }\n\n query(x1: number, y1: number, x2: number, y2: number, intersectionTest?: Function): number[] {\n const min = this.min;\n const max = this.max;\n if (x1 <= min && y1 <= min && max <= x2 && max <= y2 && !intersectionTest) {\n // We use `Array#slice` because `this.keys` may be a `Int32Array` and\n // some browsers (Safari and IE) do not support `TypedArray#slice`\n // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/TypedArray/slice#Browser_compatibility\n return Array.prototype.slice.call(this.keys);\n\n } else {\n const result = [];\n const seenUids = {};\n this._forEachCell(x1, y1, x2, y2, this._queryCell, result, seenUids, intersectionTest);\n return result;\n }\n }\n\n _queryCell(x1: number, y1: number, x2: number, y2:number, cellIndex:number, result, seenUids, intersectionTest: Function) {\n const cell = this.cells[cellIndex];\n if (cell !== null) {\n const keys = this.keys;\n const bboxes = this.bboxes;\n for (let u = 0; u < cell.length; u++) {\n const uid = cell[u];\n if (seenUids[uid] === undefined) {\n const offset = uid * 4;\n if (intersectionTest ?\n intersectionTest(bboxes[offset + 0], bboxes[offset + 1], bboxes[offset + 2], bboxes[offset + 3]) :\n ((x1 <= bboxes[offset + 2]) &&\n (y1 <= bboxes[offset + 3]) &&\n (x2 >= bboxes[offset + 0]) &&\n (y2 >= bboxes[offset + 1]))) {\n seenUids[uid] = true;\n result.push(keys[uid]);\n } else {\n seenUids[uid] = false;\n }\n }\n }\n }\n }\n\n _forEachCell(x1: number, y1: number, x2:number, y2:number, fn: Function, arg1, arg2, intersectionTest) {\n const cx1 = this._convertToCellCoord(x1);\n const cy1 = this._convertToCellCoord(y1);\n const cx2 = this._convertToCellCoord(x2);\n const cy2 = this._convertToCellCoord(y2);\n for (let x = cx1; x <= cx2; x++) {\n for (let y = cy1; y <= cy2; y++) {\n const cellIndex = this.d * y + x;\n if (intersectionTest && !intersectionTest(\n this._convertFromCellCoord(x),\n this._convertFromCellCoord(y),\n this._convertFromCellCoord(x + 1),\n this._convertFromCellCoord(y + 1))) continue;\n if (fn.call(this, x1, y1, x2, y2, cellIndex, arg1, arg2, intersectionTest)) return;\n }\n }\n }\n\n _convertFromCellCoord (x) {\n return (x - this.padding) / this.scale;\n }\n\n _convertToCellCoord(x) {\n return Math.max(0, Math.min(this.d - 1, Math.floor(x * this.scale) + this.padding));\n }\n\n toArrayBuffer(): ArrayBuffer {\n if (this.arrayBuffer) return this.arrayBuffer;\n\n const cells = this.cells;\n\n const metadataLength = NUM_PARAMS + this.cells.length + 1 + 1;\n let totalCellLength = 0;\n for (let i = 0; i < this.cells.length; i++) {\n totalCellLength += this.cells[i].length;\n }\n\n const array = new Int32Array(metadataLength + totalCellLength + this.keys.length + this.bboxes.length);\n array[0] = this.extent;\n array[1] = this.n;\n array[2] = this.padding;\n\n let offset = metadataLength;\n for (let k = 0; k < cells.length; k++) {\n const cell = cells[k];\n array[NUM_PARAMS + k] = offset;\n array.set(cell, offset);\n offset += cell.length;\n }\n\n array[NUM_PARAMS + cells.length] = offset;\n array.set(this.keys, offset);\n offset += this.keys.length;\n\n array[NUM_PARAMS + cells.length + 1] = offset;\n array.set(this.bboxes, offset);\n offset += this.bboxes.length;\n\n return array.buffer;\n }\n\n public static serialize(grid: TransferableGridIndex, transferables?: Array): SerializedGrid {\n const buffer = grid.toArrayBuffer();\n if (transferables) {\n transferables.push(buffer);\n }\n return {buffer};\n }\n\n public static deserialize(serialized: SerializedGrid): TransferableGridIndex {\n return new TransferableGridIndex(serialized.buffer);\n }\n}\n","import {TransferableGridIndex} from './transferable_grid_index';\nimport {Color, CompoundExpression, expressions, ResolvedImage, StylePropertyFunction,\n StyleExpression, ZoomDependentExpression, ZoomConstantExpression} from '@maplibre/maplibre-gl-style-spec';\n\nimport {AJAXError} from './ajax';\n\nimport type {Transferable} from '../types/transferable';\nimport {isImageBitmap} from './util';\n\ntype SerializedObject = {\n [_: string]: S;\n};\n\nexport type Serialized = null | void | boolean | number | string | Boolean | Number | String | Date | RegExp | ArrayBuffer | ArrayBufferView | ImageData | ImageBitmap | Blob | Array | SerializedObject;\n\ntype Registry = {\n [_: string]: {\n klass: {\n new (...args: any): any;\n deserialize?: (input: Serialized) => unknown;\n };\n omit: ReadonlyArray;\n shallow: ReadonlyArray;\n };\n};\n\n/**\n * Register options\n */\ntype RegisterOptions = {\n /**\n * List of properties to omit from serialization (e.g., cached/computed properties)\n */\n omit?: ReadonlyArray;\n /**\n * List of properties that should be serialized by a simple shallow copy, rather than by a recursive call to serialize().\n */\n shallow?: ReadonlyArray;\n};\n\nconst registry: Registry = {};\n\n/**\n * Register the given class as serializable.\n *\n * @param options - the registration options\n */\nexport function register(\n name: string,\n klass: {\n new (...args: any): T;\n },\n options: RegisterOptions = {}\n) {\n if (registry[name]) throw new Error(`${name} is already registered.`);\n ((Object.defineProperty as any))(klass, '_classRegistryKey', {\n value: name,\n writeable: false\n });\n registry[name] = {\n klass,\n omit: options.omit as ReadonlyArray || [],\n shallow: options.shallow as ReadonlyArray || []\n };\n}\n\nregister('Object', Object);\nregister('TransferableGridIndex', TransferableGridIndex);\n\nregister('Color', Color);\nregister('Error', Error);\nregister('AJAXError', AJAXError);\nregister('ResolvedImage', ResolvedImage);\n\nregister('StylePropertyFunction', StylePropertyFunction);\nregister('StyleExpression', StyleExpression, {omit: ['_evaluator']});\n\nregister('ZoomDependentExpression', ZoomDependentExpression);\nregister('ZoomConstantExpression', ZoomConstantExpression);\nregister('CompoundExpression', CompoundExpression, {omit: ['_evaluate']});\nfor (const name in expressions) {\n if ((expressions[name] as any)._classRegistryKey) continue;\n register(`Expression_${name}`, expressions[name]);\n}\n\nfunction isArrayBuffer(value: any): value is ArrayBuffer {\n return value && typeof ArrayBuffer !== 'undefined' &&\n (value instanceof ArrayBuffer || (value.constructor && value.constructor.name === 'ArrayBuffer'));\n}\n\n/**\n * Serialize the given object for transfer to or from a web worker.\n *\n * For non-builtin types, recursively serialize each property (possibly\n * omitting certain properties - see register()), and package the result along\n * with the constructor's `name` so that the appropriate constructor can be\n * looked up in `deserialize()`.\n *\n * If a `transferables` array is provided, add any transferable objects (i.e.,\n * any ArrayBuffers or ArrayBuffer views) to the list. (If a copy is needed,\n * this should happen in the client code, before using serialize().)\n */\nexport function serialize(input: unknown, transferables?: Array | null): Serialized {\n if (input === null ||\n input === undefined ||\n typeof input === 'boolean' ||\n typeof input === 'number' ||\n typeof input === 'string' ||\n input instanceof Boolean ||\n input instanceof Number ||\n input instanceof String ||\n input instanceof Date ||\n input instanceof RegExp ||\n input instanceof Blob) {\n return input;\n }\n\n if (isArrayBuffer(input)) {\n if (transferables) {\n transferables.push(input);\n }\n return input;\n }\n\n if (isImageBitmap(input)) {\n if (transferables) {\n transferables.push(input);\n }\n return input;\n }\n\n if (ArrayBuffer.isView(input)) {\n const view = input;\n if (transferables) {\n transferables.push(view.buffer);\n }\n return view;\n }\n\n if (input instanceof ImageData) {\n if (transferables) {\n transferables.push(input.data.buffer);\n }\n return input;\n }\n\n if (Array.isArray(input)) {\n const serialized: Array = [];\n for (const item of input) {\n serialized.push(serialize(item, transferables));\n }\n return serialized;\n }\n\n if (typeof input === 'object') {\n const klass = (input.constructor as any);\n const name = klass._classRegistryKey;\n if (!name) {\n throw new Error('can\\'t serialize object of unregistered class');\n }\n if (!registry[name]) throw new Error(`${name} is not registered.`);\n\n const properties: SerializedObject = klass.serialize ?\n // (Temporary workaround) allow a class to provide static\n // `serialize()` and `deserialize()` methods to bypass the generic\n // approach.\n // This temporary workaround lets us use the generic serialization\n // approach for objects whose members include instances of dynamic\n // StructArray types. Once we refactor StructArray to be static,\n // we can remove this complexity.\n (klass.serialize(input, transferables) as SerializedObject) : {};\n\n if (!klass.serialize) {\n for (const key in input) {\n if (!input.hasOwnProperty(key)) continue; // eslint-disable-line no-prototype-builtins\n if (registry[name].omit.indexOf(key) >= 0) continue;\n const property = input[key];\n properties[key] = registry[name].shallow.indexOf(key) >= 0 ?\n property :\n serialize(property, transferables);\n }\n if (input instanceof Error) {\n properties.message = input.message;\n }\n } else {\n if (transferables && properties === transferables[transferables.length - 1]) {\n throw new Error('statically serialized object won\\'t survive transfer of $name property');\n }\n }\n\n if (properties.$name) {\n throw new Error('$name property is reserved for worker serialization logic.');\n }\n if (name !== 'Object') {\n properties.$name = name;\n }\n\n return properties;\n }\n\n throw new Error(`can't serialize object of type ${typeof input}`);\n}\n\nexport function deserialize(input: Serialized): unknown {\n if (input === null ||\n input === undefined ||\n typeof input === 'boolean' ||\n typeof input === 'number' ||\n typeof input === 'string' ||\n input instanceof Boolean ||\n input instanceof Number ||\n input instanceof String ||\n input instanceof Date ||\n input instanceof RegExp ||\n input instanceof Blob ||\n isArrayBuffer(input) ||\n isImageBitmap(input) ||\n ArrayBuffer.isView(input) ||\n input instanceof ImageData) {\n return input;\n }\n\n if (Array.isArray(input)) {\n return input.map(deserialize);\n }\n\n if (typeof input === 'object') {\n const name = input.$name || 'Object';\n if (!registry[name]) {\n throw new Error(`can't deserialize unregistered class ${name}`);\n }\n const {klass} = registry[name];\n if (!klass) {\n throw new Error(`can't deserialize unregistered class ${name}`);\n }\n\n if (klass.deserialize) {\n return klass.deserialize(input);\n }\n\n const result = Object.create(klass.prototype);\n\n for (const key of Object.keys(input)) {\n if (key === '$name') continue;\n const value = (input as SerializedObject)[key];\n result[key] = registry[name].shallow.indexOf(key) >= 0 ? value : deserialize(value);\n }\n\n return result;\n }\n\n throw new Error(`can't deserialize object of type ${typeof input}`);\n}\n","export class ZoomHistory {\n lastZoom: number;\n lastFloorZoom: number;\n lastIntegerZoom: number;\n lastIntegerZoomTime: number;\n first: boolean;\n\n constructor() {\n this.first = true;\n }\n\n update(z: number, now: number) {\n const floorZ = Math.floor(z);\n\n if (this.first) {\n this.first = false;\n this.lastIntegerZoom = floorZ;\n this.lastIntegerZoomTime = 0;\n this.lastZoom = z;\n this.lastFloorZoom = floorZ;\n return true;\n }\n\n if (this.lastFloorZoom > floorZ) {\n this.lastIntegerZoom = floorZ + 1;\n this.lastIntegerZoomTime = now;\n } else if (this.lastFloorZoom < floorZ) {\n this.lastIntegerZoom = floorZ;\n this.lastIntegerZoomTime = now;\n }\n\n if (z !== this.lastZoom) {\n this.lastZoom = z;\n this.lastFloorZoom = floorZ;\n return true;\n }\n\n return false;\n }\n}\n","// The following table comes from .\n// Keep it synchronized with .\n\ntype UnicodeBlockLookup = {[key: string]: (char: number) => boolean};\n\nexport const unicodeBlockLookup: UnicodeBlockLookup = {\n // 'Basic Latin': (char) => char >= 0x0000 && char <= 0x007F,\n 'Latin-1 Supplement': (char) => char >= 0x0080 && char <= 0x00FF,\n // 'Latin Extended-A': (char) => char >= 0x0100 && char <= 0x017F,\n // 'Latin Extended-B': (char) => char >= 0x0180 && char <= 0x024F,\n // 'IPA Extensions': (char) => char >= 0x0250 && char <= 0x02AF,\n // 'Spacing Modifier Letters': (char) => char >= 0x02B0 && char <= 0x02FF,\n // 'Combining Diacritical Marks': (char) => char >= 0x0300 && char <= 0x036F,\n // 'Greek and Coptic': (char) => char >= 0x0370 && char <= 0x03FF,\n // 'Cyrillic': (char) => char >= 0x0400 && char <= 0x04FF,\n // 'Cyrillic Supplement': (char) => char >= 0x0500 && char <= 0x052F,\n // 'Armenian': (char) => char >= 0x0530 && char <= 0x058F,\n //'Hebrew': (char) => char >= 0x0590 && char <= 0x05FF,\n 'Arabic': (char) => char >= 0x0600 && char <= 0x06FF,\n //'Syriac': (char) => char >= 0x0700 && char <= 0x074F,\n 'Arabic Supplement': (char) => char >= 0x0750 && char <= 0x077F,\n // 'Thaana': (char) => char >= 0x0780 && char <= 0x07BF,\n // 'NKo': (char) => char >= 0x07C0 && char <= 0x07FF,\n // 'Samaritan': (char) => char >= 0x0800 && char <= 0x083F,\n // 'Mandaic': (char) => char >= 0x0840 && char <= 0x085F,\n // 'Syriac Supplement': (char) => char >= 0x0860 && char <= 0x086F,\n 'Arabic Extended-A': (char) => char >= 0x08A0 && char <= 0x08FF,\n // 'Devanagari': (char) => char >= 0x0900 && char <= 0x097F,\n // 'Bengali': (char) => char >= 0x0980 && char <= 0x09FF,\n // 'Gurmukhi': (char) => char >= 0x0A00 && char <= 0x0A7F,\n // 'Gujarati': (char) => char >= 0x0A80 && char <= 0x0AFF,\n // 'Oriya': (char) => char >= 0x0B00 && char <= 0x0B7F,\n // 'Tamil': (char) => char >= 0x0B80 && char <= 0x0BFF,\n // 'Telugu': (char) => char >= 0x0C00 && char <= 0x0C7F,\n // 'Kannada': (char) => char >= 0x0C80 && char <= 0x0CFF,\n // 'Malayalam': (char) => char >= 0x0D00 && char <= 0x0D7F,\n // 'Sinhala': (char) => char >= 0x0D80 && char <= 0x0DFF,\n // 'Thai': (char) => char >= 0x0E00 && char <= 0x0E7F,\n // 'Lao': (char) => char >= 0x0E80 && char <= 0x0EFF,\n // 'Tibetan': (char) => char >= 0x0F00 && char <= 0x0FFF,\n // 'Myanmar': (char) => char >= 0x1000 && char <= 0x109F,\n // 'Georgian': (char) => char >= 0x10A0 && char <= 0x10FF,\n 'Hangul Jamo': (char) => char >= 0x1100 && char <= 0x11FF,\n // 'Ethiopic': (char) => char >= 0x1200 && char <= 0x137F,\n // 'Ethiopic Supplement': (char) => char >= 0x1380 && char <= 0x139F,\n // 'Cherokee': (char) => char >= 0x13A0 && char <= 0x13FF,\n 'Unified Canadian Aboriginal Syllabics': (char) => char >= 0x1400 && char <= 0x167F,\n // 'Ogham': (char) => char >= 0x1680 && char <= 0x169F,\n // 'Runic': (char) => char >= 0x16A0 && char <= 0x16FF,\n // 'Tagalog': (char) => char >= 0x1700 && char <= 0x171F,\n // 'Hanunoo': (char) => char >= 0x1720 && char <= 0x173F,\n // 'Buhid': (char) => char >= 0x1740 && char <= 0x175F,\n // 'Tagbanwa': (char) => char >= 0x1760 && char <= 0x177F,\n 'Khmer': (char) => char >= 0x1780 && char <= 0x17FF,\n // 'Mongolian': (char) => char >= 0x1800 && char <= 0x18AF,\n 'Unified Canadian Aboriginal Syllabics Extended': (char) => char >= 0x18B0 && char <= 0x18FF,\n // 'Limbu': (char) => char >= 0x1900 && char <= 0x194F,\n // 'Tai Le': (char) => char >= 0x1950 && char <= 0x197F,\n // 'New Tai Lue': (char) => char >= 0x1980 && char <= 0x19DF,\n // 'Khmer Symbols': (char) => char >= 0x19E0 && char <= 0x19FF,\n // 'Buginese': (char) => char >= 0x1A00 && char <= 0x1A1F,\n // 'Tai Tham': (char) => char >= 0x1A20 && char <= 0x1AAF,\n // 'Combining Diacritical Marks Extended': (char) => char >= 0x1AB0 && char <= 0x1AFF,\n // 'Balinese': (char) => char >= 0x1B00 && char <= 0x1B7F,\n // 'Sundanese': (char) => char >= 0x1B80 && char <= 0x1BBF,\n // 'Batak': (char) => char >= 0x1BC0 && char <= 0x1BFF,\n // 'Lepcha': (char) => char >= 0x1C00 && char <= 0x1C4F,\n // 'Ol Chiki': (char) => char >= 0x1C50 && char <= 0x1C7F,\n // 'Cyrillic Extended-C': (char) => char >= 0x1C80 && char <= 0x1C8F,\n // 'Georgian Extended': (char) => char >= 0x1C90 && char <= 0x1CBF,\n // 'Sundanese Supplement': (char) => char >= 0x1CC0 && char <= 0x1CCF,\n // 'Vedic Extensions': (char) => char >= 0x1CD0 && char <= 0x1CFF,\n // 'Phonetic Extensions': (char) => char >= 0x1D00 && char <= 0x1D7F,\n // 'Phonetic Extensions Supplement': (char) => char >= 0x1D80 && char <= 0x1DBF,\n // 'Combining Diacritical Marks Supplement': (char) => char >= 0x1DC0 && char <= 0x1DFF,\n // 'Latin Extended Additional': (char) => char >= 0x1E00 && char <= 0x1EFF,\n // 'Greek Extended': (char) => char >= 0x1F00 && char <= 0x1FFF,\n 'General Punctuation': (char) => char >= 0x2000 && char <= 0x206F,\n // 'Superscripts and Subscripts': (char) => char >= 0x2070 && char <= 0x209F,\n // 'Currency Symbols': (char) => char >= 0x20A0 && char <= 0x20CF,\n // 'Combining Diacritical Marks for Symbols': (char) => char >= 0x20D0 && char <= 0x20FF,\n 'Letterlike Symbols': (char) => char >= 0x2100 && char <= 0x214F,\n 'Number Forms': (char) => char >= 0x2150 && char <= 0x218F,\n // 'Arrows': (char) => char >= 0x2190 && char <= 0x21FF,\n // 'Mathematical Operators': (char) => char >= 0x2200 && char <= 0x22FF,\n 'Miscellaneous Technical': (char) => char >= 0x2300 && char <= 0x23FF,\n 'Control Pictures': (char) => char >= 0x2400 && char <= 0x243F,\n 'Optical Character Recognition': (char) => char >= 0x2440 && char <= 0x245F,\n 'Enclosed Alphanumerics': (char) => char >= 0x2460 && char <= 0x24FF,\n // 'Box Drawing': (char) => char >= 0x2500 && char <= 0x257F,\n // 'Block Elements': (char) => char >= 0x2580 && char <= 0x259F,\n 'Geometric Shapes': (char) => char >= 0x25A0 && char <= 0x25FF,\n 'Miscellaneous Symbols': (char) => char >= 0x2600 && char <= 0x26FF,\n // 'Dingbats': (char) => char >= 0x2700 && char <= 0x27BF,\n // 'Miscellaneous Mathematical Symbols-A': (char) => char >= 0x27C0 && char <= 0x27EF,\n // 'Supplemental Arrows-A': (char) => char >= 0x27F0 && char <= 0x27FF,\n // 'Braille Patterns': (char) => char >= 0x2800 && char <= 0x28FF,\n // 'Supplemental Arrows-B': (char) => char >= 0x2900 && char <= 0x297F,\n // 'Miscellaneous Mathematical Symbols-B': (char) => char >= 0x2980 && char <= 0x29FF,\n // 'Supplemental Mathematical Operators': (char) => char >= 0x2A00 && char <= 0x2AFF,\n 'Miscellaneous Symbols and Arrows': (char) => char >= 0x2B00 && char <= 0x2BFF,\n // 'Glagolitic': (char) => char >= 0x2C00 && char <= 0x2C5F,\n // 'Latin Extended-C': (char) => char >= 0x2C60 && char <= 0x2C7F,\n // 'Coptic': (char) => char >= 0x2C80 && char <= 0x2CFF,\n // 'Georgian Supplement': (char) => char >= 0x2D00 && char <= 0x2D2F,\n // 'Tifinagh': (char) => char >= 0x2D30 && char <= 0x2D7F,\n // 'Ethiopic Extended': (char) => char >= 0x2D80 && char <= 0x2DDF,\n // 'Cyrillic Extended-A': (char) => char >= 0x2DE0 && char <= 0x2DFF,\n // 'Supplemental Punctuation': (char) => char >= 0x2E00 && char <= 0x2E7F,\n 'CJK Radicals Supplement': (char) => char >= 0x2E80 && char <= 0x2EFF,\n 'Kangxi Radicals': (char) => char >= 0x2F00 && char <= 0x2FDF,\n 'Ideographic Description Characters': (char) => char >= 0x2FF0 && char <= 0x2FFF,\n 'CJK Symbols and Punctuation': (char) => char >= 0x3000 && char <= 0x303F,\n 'Hiragana': (char) => char >= 0x3040 && char <= 0x309F,\n 'Katakana': (char) => char >= 0x30A0 && char <= 0x30FF,\n 'Bopomofo': (char) => char >= 0x3100 && char <= 0x312F,\n 'Hangul Compatibility Jamo': (char) => char >= 0x3130 && char <= 0x318F,\n 'Kanbun': (char) => char >= 0x3190 && char <= 0x319F,\n 'Bopomofo Extended': (char) => char >= 0x31A0 && char <= 0x31BF,\n 'CJK Strokes': (char) => char >= 0x31C0 && char <= 0x31EF,\n 'Katakana Phonetic Extensions': (char) => char >= 0x31F0 && char <= 0x31FF,\n 'Enclosed CJK Letters and Months': (char) => char >= 0x3200 && char <= 0x32FF,\n 'CJK Compatibility': (char) => char >= 0x3300 && char <= 0x33FF,\n 'CJK Unified Ideographs Extension A': (char) => char >= 0x3400 && char <= 0x4DBF,\n 'Yijing Hexagram Symbols': (char) => char >= 0x4DC0 && char <= 0x4DFF,\n 'CJK Unified Ideographs': (char) => char >= 0x4E00 && char <= 0x9FFF,\n 'Yi Syllables': (char) => char >= 0xA000 && char <= 0xA48F,\n 'Yi Radicals': (char) => char >= 0xA490 && char <= 0xA4CF,\n // 'Lisu': (char) => char >= 0xA4D0 && char <= 0xA4FF,\n // 'Vai': (char) => char >= 0xA500 && char <= 0xA63F,\n // 'Cyrillic Extended-B': (char) => char >= 0xA640 && char <= 0xA69F,\n // 'Bamum': (char) => char >= 0xA6A0 && char <= 0xA6FF,\n // 'Modifier Tone Letters': (char) => char >= 0xA700 && char <= 0xA71F,\n // 'Latin Extended-D': (char) => char >= 0xA720 && char <= 0xA7FF,\n // 'Syloti Nagri': (char) => char >= 0xA800 && char <= 0xA82F,\n // 'Common Indic Number Forms': (char) => char >= 0xA830 && char <= 0xA83F,\n // 'Phags-pa': (char) => char >= 0xA840 && char <= 0xA87F,\n // 'Saurashtra': (char) => char >= 0xA880 && char <= 0xA8DF,\n // 'Devanagari Extended': (char) => char >= 0xA8E0 && char <= 0xA8FF,\n // 'Kayah Li': (char) => char >= 0xA900 && char <= 0xA92F,\n // 'Rejang': (char) => char >= 0xA930 && char <= 0xA95F,\n 'Hangul Jamo Extended-A': (char) => char >= 0xA960 && char <= 0xA97F,\n // 'Javanese': (char) => char >= 0xA980 && char <= 0xA9DF,\n // 'Myanmar Extended-B': (char) => char >= 0xA9E0 && char <= 0xA9FF,\n // 'Cham': (char) => char >= 0xAA00 && char <= 0xAA5F,\n // 'Myanmar Extended-A': (char) => char >= 0xAA60 && char <= 0xAA7F,\n // 'Tai Viet': (char) => char >= 0xAA80 && char <= 0xAADF,\n // 'Meetei Mayek Extensions': (char) => char >= 0xAAE0 && char <= 0xAAFF,\n // 'Ethiopic Extended-A': (char) => char >= 0xAB00 && char <= 0xAB2F,\n // 'Latin Extended-E': (char) => char >= 0xAB30 && char <= 0xAB6F,\n // 'Cherokee Supplement': (char) => char >= 0xAB70 && char <= 0xABBF,\n // 'Meetei Mayek': (char) => char >= 0xABC0 && char <= 0xABFF,\n 'Hangul Syllables': (char) => char >= 0xAC00 && char <= 0xD7AF,\n 'Hangul Jamo Extended-B': (char) => char >= 0xD7B0 && char <= 0xD7FF,\n // 'High Surrogates': (char) => char >= 0xD800 && char <= 0xDB7F,\n // 'High Private Use Surrogates': (char) => char >= 0xDB80 && char <= 0xDBFF,\n // 'Low Surrogates': (char) => char >= 0xDC00 && char <= 0xDFFF,\n 'Private Use Area': (char) => char >= 0xE000 && char <= 0xF8FF,\n 'CJK Compatibility Ideographs': (char) => char >= 0xF900 && char <= 0xFAFF,\n // 'Alphabetic Presentation Forms': (char) => char >= 0xFB00 && char <= 0xFB4F,\n 'Arabic Presentation Forms-A': (char) => char >= 0xFB50 && char <= 0xFDFF,\n // 'Variation Selectors': (char) => char >= 0xFE00 && char <= 0xFE0F,\n 'Vertical Forms': (char) => char >= 0xFE10 && char <= 0xFE1F,\n // 'Combining Half Marks': (char) => char >= 0xFE20 && char <= 0xFE2F,\n 'CJK Compatibility Forms': (char) => char >= 0xFE30 && char <= 0xFE4F,\n 'Small Form Variants': (char) => char >= 0xFE50 && char <= 0xFE6F,\n 'Arabic Presentation Forms-B': (char) => char >= 0xFE70 && char <= 0xFEFF,\n 'Halfwidth and Fullwidth Forms': (char) => char >= 0xFF00 && char <= 0xFFEF\n // 'Specials': (char) => char >= 0xFFF0 && char <= 0xFFFF,\n // 'Linear B Syllabary': (char) => char >= 0x10000 && char <= 0x1007F,\n // 'Linear B Ideograms': (char) => char >= 0x10080 && char <= 0x100FF,\n // 'Aegean Numbers': (char) => char >= 0x10100 && char <= 0x1013F,\n // 'Ancient Greek Numbers': (char) => char >= 0x10140 && char <= 0x1018F,\n // 'Ancient Symbols': (char) => char >= 0x10190 && char <= 0x101CF,\n // 'Phaistos Disc': (char) => char >= 0x101D0 && char <= 0x101FF,\n // 'Lycian': (char) => char >= 0x10280 && char <= 0x1029F,\n // 'Carian': (char) => char >= 0x102A0 && char <= 0x102DF,\n // 'Coptic Epact Numbers': (char) => char >= 0x102E0 && char <= 0x102FF,\n // 'Old Italic': (char) => char >= 0x10300 && char <= 0x1032F,\n // 'Gothic': (char) => char >= 0x10330 && char <= 0x1034F,\n // 'Old Permic': (char) => char >= 0x10350 && char <= 0x1037F,\n // 'Ugaritic': (char) => char >= 0x10380 && char <= 0x1039F,\n // 'Old Persian': (char) => char >= 0x103A0 && char <= 0x103DF,\n // 'Deseret': (char) => char >= 0x10400 && char <= 0x1044F,\n // 'Shavian': (char) => char >= 0x10450 && char <= 0x1047F,\n // 'Osmanya': (char) => char >= 0x10480 && char <= 0x104AF,\n // 'Osage': (char) => char >= 0x104B0 && char <= 0x104FF,\n // 'Elbasan': (char) => char >= 0x10500 && char <= 0x1052F,\n // 'Caucasian Albanian': (char) => char >= 0x10530 && char <= 0x1056F,\n // 'Linear A': (char) => char >= 0x10600 && char <= 0x1077F,\n // 'Cypriot Syllabary': (char) => char >= 0x10800 && char <= 0x1083F,\n // 'Imperial Aramaic': (char) => char >= 0x10840 && char <= 0x1085F,\n // 'Palmyrene': (char) => char >= 0x10860 && char <= 0x1087F,\n // 'Nabataean': (char) => char >= 0x10880 && char <= 0x108AF,\n // 'Hatran': (char) => char >= 0x108E0 && char <= 0x108FF,\n // 'Phoenician': (char) => char >= 0x10900 && char <= 0x1091F,\n // 'Lydian': (char) => char >= 0x10920 && char <= 0x1093F,\n // 'Meroitic Hieroglyphs': (char) => char >= 0x10980 && char <= 0x1099F,\n // 'Meroitic Cursive': (char) => char >= 0x109A0 && char <= 0x109FF,\n // 'Kharoshthi': (char) => char >= 0x10A00 && char <= 0x10A5F,\n // 'Old South Arabian': (char) => char >= 0x10A60 && char <= 0x10A7F,\n // 'Old North Arabian': (char) => char >= 0x10A80 && char <= 0x10A9F,\n // 'Manichaean': (char) => char >= 0x10AC0 && char <= 0x10AFF,\n // 'Avestan': (char) => char >= 0x10B00 && char <= 0x10B3F,\n // 'Inscriptional Parthian': (char) => char >= 0x10B40 && char <= 0x10B5F,\n // 'Inscriptional Pahlavi': (char) => char >= 0x10B60 && char <= 0x10B7F,\n // 'Psalter Pahlavi': (char) => char >= 0x10B80 && char <= 0x10BAF,\n // 'Old Turkic': (char) => char >= 0x10C00 && char <= 0x10C4F,\n // 'Old Hungarian': (char) => char >= 0x10C80 && char <= 0x10CFF,\n // 'Hanifi Rohingya': (char) => char >= 0x10D00 && char <= 0x10D3F,\n // 'Rumi Numeral Symbols': (char) => char >= 0x10E60 && char <= 0x10E7F,\n // 'Old Sogdian': (char) => char >= 0x10F00 && char <= 0x10F2F,\n // 'Sogdian': (char) => char >= 0x10F30 && char <= 0x10F6F,\n // 'Elymaic': (char) => char >= 0x10FE0 && char <= 0x10FFF,\n // 'Brahmi': (char) => char >= 0x11000 && char <= 0x1107F,\n // 'Kaithi': (char) => char >= 0x11080 && char <= 0x110CF,\n // 'Sora Sompeng': (char) => char >= 0x110D0 && char <= 0x110FF,\n // 'Chakma': (char) => char >= 0x11100 && char <= 0x1114F,\n // 'Mahajani': (char) => char >= 0x11150 && char <= 0x1117F,\n // 'Sharada': (char) => char >= 0x11180 && char <= 0x111DF,\n // 'Sinhala Archaic Numbers': (char) => char >= 0x111E0 && char <= 0x111FF,\n // 'Khojki': (char) => char >= 0x11200 && char <= 0x1124F,\n // 'Multani': (char) => char >= 0x11280 && char <= 0x112AF,\n // 'Khudawadi': (char) => char >= 0x112B0 && char <= 0x112FF,\n // 'Grantha': (char) => char >= 0x11300 && char <= 0x1137F,\n // 'Newa': (char) => char >= 0x11400 && char <= 0x1147F,\n // 'Tirhuta': (char) => char >= 0x11480 && char <= 0x114DF,\n // 'Siddham': (char) => char >= 0x11580 && char <= 0x115FF,\n // 'Modi': (char) => char >= 0x11600 && char <= 0x1165F,\n // 'Mongolian Supplement': (char) => char >= 0x11660 && char <= 0x1167F,\n // 'Takri': (char) => char >= 0x11680 && char <= 0x116CF,\n // 'Ahom': (char) => char >= 0x11700 && char <= 0x1173F,\n // 'Dogra': (char) => char >= 0x11800 && char <= 0x1184F,\n // 'Warang Citi': (char) => char >= 0x118A0 && char <= 0x118FF,\n // 'Nandinagari': (char) => char >= 0x119A0 && char <= 0x119FF,\n // 'Zanabazar Square': (char) => char >= 0x11A00 && char <= 0x11A4F,\n // 'Soyombo': (char) => char >= 0x11A50 && char <= 0x11AAF,\n // 'Pau Cin Hau': (char) => char >= 0x11AC0 && char <= 0x11AFF,\n // 'Bhaiksuki': (char) => char >= 0x11C00 && char <= 0x11C6F,\n // 'Marchen': (char) => char >= 0x11C70 && char <= 0x11CBF,\n // 'Masaram Gondi': (char) => char >= 0x11D00 && char <= 0x11D5F,\n // 'Gunjala Gondi': (char) => char >= 0x11D60 && char <= 0x11DAF,\n // 'Makasar': (char) => char >= 0x11EE0 && char <= 0x11EFF,\n // 'Tamil Supplement': (char) => char >= 0x11FC0 && char <= 0x11FFF,\n // 'Cuneiform': (char) => char >= 0x12000 && char <= 0x123FF,\n // 'Cuneiform Numbers and Punctuation': (char) => char >= 0x12400 && char <= 0x1247F,\n // 'Early Dynastic Cuneiform': (char) => char >= 0x12480 && char <= 0x1254F,\n // 'Egyptian Hieroglyphs': (char) => char >= 0x13000 && char <= 0x1342F,\n // 'Egyptian Hieroglyph Format Controls': (char) => char >= 0x13430 && char <= 0x1343F,\n // 'Anatolian Hieroglyphs': (char) => char >= 0x14400 && char <= 0x1467F,\n // 'Bamum Supplement': (char) => char >= 0x16800 && char <= 0x16A3F,\n // 'Mro': (char) => char >= 0x16A40 && char <= 0x16A6F,\n // 'Bassa Vah': (char) => char >= 0x16AD0 && char <= 0x16AFF,\n // 'Pahawh Hmong': (char) => char >= 0x16B00 && char <= 0x16B8F,\n // 'Medefaidrin': (char) => char >= 0x16E40 && char <= 0x16E9F,\n // 'Miao': (char) => char >= 0x16F00 && char <= 0x16F9F,\n // 'Ideographic Symbols and Punctuation': (char) => char >= 0x16FE0 && char <= 0x16FFF,\n // 'Tangut': (char) => char >= 0x17000 && char <= 0x187FF,\n // 'Tangut Components': (char) => char >= 0x18800 && char <= 0x18AFF,\n // 'Kana Supplement': (char) => char >= 0x1B000 && char <= 0x1B0FF,\n // 'Kana Extended-A': (char) => char >= 0x1B100 && char <= 0x1B12F,\n // 'Small Kana Extension': (char) => char >= 0x1B130 && char <= 0x1B16F,\n // 'Nushu': (char) => char >= 0x1B170 && char <= 0x1B2FF,\n // 'Duployan': (char) => char >= 0x1BC00 && char <= 0x1BC9F,\n // 'Shorthand Format Controls': (char) => char >= 0x1BCA0 && char <= 0x1BCAF,\n // 'Byzantine Musical Symbols': (char) => char >= 0x1D000 && char <= 0x1D0FF,\n // 'Musical Symbols': (char) => char >= 0x1D100 && char <= 0x1D1FF,\n // 'Ancient Greek Musical Notation': (char) => char >= 0x1D200 && char <= 0x1D24F,\n // 'Mayan Numerals': (char) => char >= 0x1D2E0 && char <= 0x1D2FF,\n // 'Tai Xuan Jing Symbols': (char) => char >= 0x1D300 && char <= 0x1D35F,\n // 'Counting Rod Numerals': (char) => char >= 0x1D360 && char <= 0x1D37F,\n // 'Mathematical Alphanumeric Symbols': (char) => char >= 0x1D400 && char <= 0x1D7FF,\n // 'Sutton SignWriting': (char) => char >= 0x1D800 && char <= 0x1DAAF,\n // 'Glagolitic Supplement': (char) => char >= 0x1E000 && char <= 0x1E02F,\n // 'Nyiakeng Puachue Hmong': (char) => char >= 0x1E100 && char <= 0x1E14F,\n // 'Wancho': (char) => char >= 0x1E2C0 && char <= 0x1E2FF,\n // 'Mende Kikakui': (char) => char >= 0x1E800 && char <= 0x1E8DF,\n // 'Adlam': (char) => char >= 0x1E900 && char <= 0x1E95F,\n // 'Indic Siyaq Numbers': (char) => char >= 0x1EC70 && char <= 0x1ECBF,\n // 'Ottoman Siyaq Numbers': (char) => char >= 0x1ED00 && char <= 0x1ED4F,\n // 'Arabic Mathematical Alphabetic Symbols': (char) => char >= 0x1EE00 && char <= 0x1EEFF,\n // 'Mahjong Tiles': (char) => char >= 0x1F000 && char <= 0x1F02F,\n // 'Domino Tiles': (char) => char >= 0x1F030 && char <= 0x1F09F,\n // 'Playing Cards': (char) => char >= 0x1F0A0 && char <= 0x1F0FF,\n // 'Enclosed Alphanumeric Supplement': (char) => char >= 0x1F100 && char <= 0x1F1FF,\n // 'Enclosed Ideographic Supplement': (char) => char >= 0x1F200 && char <= 0x1F2FF,\n // 'Miscellaneous Symbols and Pictographs': (char) => char >= 0x1F300 && char <= 0x1F5FF,\n // 'Emoticons': (char) => char >= 0x1F600 && char <= 0x1F64F,\n // 'Ornamental Dingbats': (char) => char >= 0x1F650 && char <= 0x1F67F,\n // 'Transport and Map Symbols': (char) => char >= 0x1F680 && char <= 0x1F6FF,\n // 'Alchemical Symbols': (char) => char >= 0x1F700 && char <= 0x1F77F,\n // 'Geometric Shapes Extended': (char) => char >= 0x1F780 && char <= 0x1F7FF,\n // 'Supplemental Arrows-C': (char) => char >= 0x1F800 && char <= 0x1F8FF,\n // 'Supplemental Symbols and Pictographs': (char) => char >= 0x1F900 && char <= 0x1F9FF,\n // 'Chess Symbols': (char) => char >= 0x1FA00 && char <= 0x1FA6F,\n // 'Symbols and Pictographs Extended-A': (char) => char >= 0x1FA70 && char <= 0x1FAFF,\n // 'CJK Unified Ideographs Extension B': (char) => char >= 0x20000 && char <= 0x2A6DF,\n // 'CJK Unified Ideographs Extension C': (char) => char >= 0x2A700 && char <= 0x2B73F,\n // 'CJK Unified Ideographs Extension D': (char) => char >= 0x2B740 && char <= 0x2B81F,\n // 'CJK Unified Ideographs Extension E': (char) => char >= 0x2B820 && char <= 0x2CEAF,\n // 'CJK Unified Ideographs Extension F': (char) => char >= 0x2CEB0 && char <= 0x2EBEF,\n // 'CJK Compatibility Ideographs Supplement': (char) => char >= 0x2F800 && char <= 0x2FA1F,\n // 'Tags': (char) => char >= 0xE0000 && char <= 0xE007F,\n // 'Variation Selectors Supplement': (char) => char >= 0xE0100 && char <= 0xE01EF,\n // 'Supplementary Private Use Area-A': (char) => char >= 0xF0000 && char <= 0xFFFFF,\n // 'Supplementary Private Use Area-B': (char) => char >= 0x100000 && char <= 0x10FFFF,\n};\n","/* eslint-disable new-cap */\n\nimport {unicodeBlockLookup as isChar} from './is_char_in_unicode_block';\n\nexport function allowsIdeographicBreaking(chars: string) {\n for (const char of chars) {\n if (!charAllowsIdeographicBreaking(char.charCodeAt(0))) return false;\n }\n return true;\n}\n\nexport function allowsVerticalWritingMode(chars: string) {\n for (const char of chars) {\n if (charHasUprightVerticalOrientation(char.charCodeAt(0))) return true;\n }\n return false;\n}\n\nexport function allowsLetterSpacing(chars: string) {\n for (const char of chars) {\n if (!charAllowsLetterSpacing(char.charCodeAt(0))) return false;\n }\n return true;\n}\n\nexport function charAllowsLetterSpacing(char: number) {\n if (isChar['Arabic'](char)) return false;\n if (isChar['Arabic Supplement'](char)) return false;\n if (isChar['Arabic Extended-A'](char)) return false;\n if (isChar['Arabic Presentation Forms-A'](char)) return false;\n if (isChar['Arabic Presentation Forms-B'](char)) return false;\n\n return true;\n}\n\nexport function charAllowsIdeographicBreaking(char: number) {\n // Return early for characters outside all ideographic ranges.\n if (char < 0x2E80) return false;\n\n if (isChar['Bopomofo Extended'](char)) return true;\n if (isChar['Bopomofo'](char)) return true;\n if (isChar['CJK Compatibility Forms'](char)) return true;\n if (isChar['CJK Compatibility Ideographs'](char)) return true;\n if (isChar['CJK Compatibility'](char)) return true;\n if (isChar['CJK Radicals Supplement'](char)) return true;\n if (isChar['CJK Strokes'](char)) return true;\n if (isChar['CJK Symbols and Punctuation'](char)) return true;\n if (isChar['CJK Unified Ideographs Extension A'](char)) return true;\n if (isChar['CJK Unified Ideographs'](char)) return true;\n if (isChar['Enclosed CJK Letters and Months'](char)) return true;\n if (isChar['Halfwidth and Fullwidth Forms'](char)) return true;\n if (isChar['Hiragana'](char)) return true;\n if (isChar['Ideographic Description Characters'](char)) return true;\n if (isChar['Kangxi Radicals'](char)) return true;\n if (isChar['Katakana Phonetic Extensions'](char)) return true;\n if (isChar['Katakana'](char)) return true;\n if (isChar['Vertical Forms'](char)) return true;\n if (isChar['Yi Radicals'](char)) return true;\n if (isChar['Yi Syllables'](char)) return true;\n\n return false;\n}\n\n// The following logic comes from\n// .\n// Keep it synchronized with\n// .\n// The data file denotes with “U” or “Tu” any codepoint that may be drawn\n// upright in vertical text but does not distinguish between upright and\n// “neutral” characters.\n\n// Blocks in the Unicode supplementary planes are excluded from this module due\n// to .\n\n/**\n * Returns true if the given Unicode codepoint identifies a character with\n * upright orientation.\n *\n * A character has upright orientation if it is drawn upright (unrotated)\n * whether the line is oriented horizontally or vertically, even if both\n * adjacent characters can be rotated. For example, a Chinese character is\n * always drawn upright. An uprightly oriented character causes an adjacent\n * “neutral” character to be drawn upright as well.\n */\nexport function charHasUprightVerticalOrientation(char: number) {\n if (char === 0x02EA /* modifier letter yin departing tone mark */ ||\n char === 0x02EB /* modifier letter yang departing tone mark */) {\n return true;\n }\n\n // Return early for characters outside all ranges whose characters remain\n // upright in vertical writing mode.\n if (char < 0x1100) return false;\n\n if (isChar['Bopomofo Extended'](char)) return true;\n if (isChar['Bopomofo'](char)) return true;\n if (isChar['CJK Compatibility Forms'](char)) {\n if (!((char >= 0xFE49 /* dashed overline */ && char <= 0xFE4F) /* wavy low line */)) {\n return true;\n }\n }\n if (isChar['CJK Compatibility Ideographs'](char)) return true;\n if (isChar['CJK Compatibility'](char)) return true;\n if (isChar['CJK Radicals Supplement'](char)) return true;\n if (isChar['CJK Strokes'](char)) return true;\n if (isChar['CJK Symbols and Punctuation'](char)) {\n if (!((char >= 0x3008 /* left angle bracket */ && char <= 0x3011) /* right black lenticular bracket */) &&\n !((char >= 0x3014 /* left tortoise shell bracket */ && char <= 0x301F) /* low double prime quotation mark */) &&\n char !== 0x3030 /* wavy dash */) {\n return true;\n }\n }\n if (isChar['CJK Unified Ideographs Extension A'](char)) return true;\n if (isChar['CJK Unified Ideographs'](char)) return true;\n if (isChar['Enclosed CJK Letters and Months'](char)) return true;\n if (isChar['Hangul Compatibility Jamo'](char)) return true;\n if (isChar['Hangul Jamo Extended-A'](char)) return true;\n if (isChar['Hangul Jamo Extended-B'](char)) return true;\n if (isChar['Hangul Jamo'](char)) return true;\n if (isChar['Hangul Syllables'](char)) return true;\n if (isChar['Hiragana'](char)) return true;\n if (isChar['Ideographic Description Characters'](char)) return true;\n if (isChar['Kanbun'](char)) return true;\n if (isChar['Kangxi Radicals'](char)) return true;\n if (isChar['Katakana Phonetic Extensions'](char)) return true;\n if (isChar['Katakana'](char)) {\n if (char !== 0x30FC /* katakana-hiragana prolonged sound mark */) {\n return true;\n }\n }\n if (isChar['Halfwidth and Fullwidth Forms'](char)) {\n if (char !== 0xFF08 /* fullwidth left parenthesis */ &&\n char !== 0xFF09 /* fullwidth right parenthesis */ &&\n char !== 0xFF0D /* fullwidth hyphen-minus */ &&\n !((char >= 0xFF1A /* fullwidth colon */ && char <= 0xFF1E) /* fullwidth greater-than sign */) &&\n char !== 0xFF3B /* fullwidth left square bracket */ &&\n char !== 0xFF3D /* fullwidth right square bracket */ &&\n char !== 0xFF3F /* fullwidth low line */ &&\n !(char >= 0xFF5B /* fullwidth left curly bracket */ && char <= 0xFFDF) &&\n char !== 0xFFE3 /* fullwidth macron */ &&\n !(char >= 0xFFE8 /* halfwidth forms light vertical */ && char <= 0xFFEF)) {\n return true;\n }\n }\n if (isChar['Small Form Variants'](char)) {\n if (!((char >= 0xFE58 /* small em dash */ && char <= 0xFE5E) /* small right tortoise shell bracket */) &&\n !((char >= 0xFE63 /* small hyphen-minus */ && char <= 0xFE66) /* small equals sign */)) {\n return true;\n }\n }\n if (isChar['Unified Canadian Aboriginal Syllabics'](char)) return true;\n if (isChar['Unified Canadian Aboriginal Syllabics Extended'](char)) return true;\n if (isChar['Vertical Forms'](char)) return true;\n if (isChar['Yijing Hexagram Symbols'](char)) return true;\n if (isChar['Yi Syllables'](char)) return true;\n if (isChar['Yi Radicals'](char)) return true;\n\n return false;\n}\n\n/**\n * Returns true if the given Unicode codepoint identifies a character with\n * neutral orientation.\n *\n * A character has neutral orientation if it may be drawn rotated or unrotated\n * when the line is oriented vertically, depending on the orientation of the\n * adjacent characters. For example, along a verticlly oriented line, the vulgar\n * fraction ½ is drawn upright among Chinese characters but rotated among Latin\n * letters. A neutrally oriented character does not influence whether an\n * adjacent character is drawn upright or rotated.\n */\nexport function charHasNeutralVerticalOrientation(char: number) {\n if (isChar['Latin-1 Supplement'](char)) {\n if (char === 0x00A7 /* section sign */ ||\n char === 0x00A9 /* copyright sign */ ||\n char === 0x00AE /* registered sign */ ||\n char === 0x00B1 /* plus-minus sign */ ||\n char === 0x00BC /* vulgar fraction one quarter */ ||\n char === 0x00BD /* vulgar fraction one half */ ||\n char === 0x00BE /* vulgar fraction three quarters */ ||\n char === 0x00D7 /* multiplication sign */ ||\n char === 0x00F7 /* division sign */) {\n return true;\n }\n }\n if (isChar['General Punctuation'](char)) {\n if (char === 0x2016 /* double vertical line */ ||\n char === 0x2020 /* dagger */ ||\n char === 0x2021 /* double dagger */ ||\n char === 0x2030 /* per mille sign */ ||\n char === 0x2031 /* per ten thousand sign */ ||\n char === 0x203B /* reference mark */ ||\n char === 0x203C /* double exclamation mark */ ||\n char === 0x2042 /* asterism */ ||\n char === 0x2047 /* double question mark */ ||\n char === 0x2048 /* question exclamation mark */ ||\n char === 0x2049 /* exclamation question mark */ ||\n char === 0x2051 /* two asterisks aligned vertically */) {\n return true;\n }\n }\n if (isChar['Letterlike Symbols'](char)) return true;\n if (isChar['Number Forms'](char)) return true;\n if (isChar['Miscellaneous Technical'](char)) {\n if ((char >= 0x2300 /* diameter sign */ && char <= 0x2307 /* wavy line */) ||\n (char >= 0x230C /* bottom right crop */ && char <= 0x231F /* bottom right corner */) ||\n (char >= 0x2324 /* up arrowhead between two horizontal bars */ && char <= 0x2328 /* keyboard */) ||\n char === 0x232B /* erase to the left */ ||\n (char >= 0x237D /* shouldered open box */ && char <= 0x239A /* clear screen symbol */) ||\n (char >= 0x23BE /* dentistry symbol light vertical and top right */ && char <= 0x23CD /* square foot */) ||\n char === 0x23CF /* eject symbol */ ||\n (char >= 0x23D1 /* metrical breve */ && char <= 0x23DB /* fuse */) ||\n (char >= 0x23E2 /* white trapezium */ && char <= 0x23FF)) {\n return true;\n }\n }\n if (isChar['Control Pictures'](char) && char !== 0x2423 /* open box */) return true;\n if (isChar['Optical Character Recognition'](char)) return true;\n if (isChar['Enclosed Alphanumerics'](char)) return true;\n if (isChar['Geometric Shapes'](char)) return true;\n if (isChar['Miscellaneous Symbols'](char)) {\n if (!((char >= 0x261A /* black left pointing index */ && char <= 0x261F) /* white down pointing index */)) {\n return true;\n }\n }\n if (isChar['Miscellaneous Symbols and Arrows'](char)) {\n if ((char >= 0x2B12 /* square with top half black */ && char <= 0x2B2F /* white vertical ellipse */) ||\n (char >= 0x2B50 /* white medium star */ && char <= 0x2B59 /* heavy circled saltire */) ||\n (char >= 0x2BB8 /* upwards white arrow from bar with horizontal bar */ && char <= 0x2BEB)) {\n return true;\n }\n }\n if (isChar['CJK Symbols and Punctuation'](char)) return true;\n if (isChar['Katakana'](char)) return true;\n if (isChar['Private Use Area'](char)) return true;\n if (isChar['CJK Compatibility Forms'](char)) return true;\n if (isChar['Small Form Variants'](char)) return true;\n if (isChar['Halfwidth and Fullwidth Forms'](char)) return true;\n\n if (char === 0x221E /* infinity */ ||\n char === 0x2234 /* therefore */ ||\n char === 0x2235 /* because */ ||\n (char >= 0x2700 /* black safety scissors */ && char <= 0x2767 /* rotated floral heart bullet */) ||\n (char >= 0x2776 /* dingbat negative circled digit one */ && char <= 0x2793 /* dingbat negative circled sans-serif number ten */) ||\n char === 0xFFFC /* object replacement character */ ||\n char === 0xFFFD /* replacement character */) {\n return true;\n }\n\n return false;\n}\n\n/**\n * Returns true if the given Unicode codepoint identifies a character with\n * rotated orientation.\n *\n * A character has rotated orientation if it is drawn rotated when the line is\n * oriented vertically, even if both adjacent characters are upright. For\n * example, a Latin letter is drawn rotated along a vertical line. A rotated\n * character causes an adjacent “neutral” character to be drawn rotated as well.\n */\nexport function charHasRotatedVerticalOrientation(char: number) {\n return !(charHasUprightVerticalOrientation(char) ||\n charHasNeutralVerticalOrientation(char));\n}\n\nexport function charInComplexShapingScript(char: number) {\n return isChar['Arabic'](char) ||\n isChar['Arabic Supplement'](char) ||\n isChar['Arabic Extended-A'](char) ||\n isChar['Arabic Presentation Forms-A'](char) ||\n isChar['Arabic Presentation Forms-B'](char);\n}\n\nexport function charInRTLScript(char: number) {\n // Main blocks for Hebrew, Arabic, Thaana and other RTL scripts\n return (char >= 0x0590 && char <= 0x08FF) ||\n isChar['Arabic Presentation Forms-A'](char) ||\n isChar['Arabic Presentation Forms-B'](char);\n}\n\nexport function charInSupportedScript(char: number, canRenderRTL: boolean) {\n // This is a rough heuristic: whether we \"can render\" a script\n // actually depends on the properties of the font being used\n // and whether differences from the ideal rendering are considered\n // semantically significant.\n\n // Even in Latin script, we \"can't render\" combinations such as the fi\n // ligature, but we don't consider that semantically significant.\n if (!canRenderRTL && charInRTLScript(char)) {\n return false;\n }\n if ((char >= 0x0900 && char <= 0x0DFF) ||\n // Main blocks for Indic scripts and Sinhala\n (char >= 0x0F00 && char <= 0x109F) ||\n // Main blocks for Tibetan and Myanmar\n isChar['Khmer'](char)) {\n // These blocks cover common scripts that require\n // complex text shaping, based on unicode script metadata:\n // http://www.unicode.org/repos/cldr/trunk/common/properties/scriptMetadata.txt\n // where \"Web Rank <= 32\" \"Shaping Required = YES\"\n return false;\n }\n return true;\n}\n\nexport function stringContainsRTLText(chars: string): boolean {\n for (const char of chars) {\n if (charInRTLScript(char.charCodeAt(0))) {\n return true;\n }\n }\n return false;\n}\n\nexport function isStringInSupportedScript(chars: string, canRenderRTL: boolean) {\n for (const char of chars) {\n if (!charInSupportedScript(char.charCodeAt(0), canRenderRTL)) {\n return false;\n }\n }\n return true;\n}\n","import {getArrayBuffer} from '../util/ajax';\nimport {browser} from '../util/browser';\nimport {Event, Evented} from '../util/evented';\nimport {isWorker} from '../util/util';\n\nconst status = {\n unavailable: 'unavailable', // Not loaded\n deferred: 'deferred', // The plugin URL has been specified, but loading has been deferred\n loading: 'loading', // request in-flight\n loaded: 'loaded',\n error: 'error'\n};\n\nexport type PluginState = {\n pluginStatus: typeof status[keyof typeof status];\n pluginURL: string;\n};\n\n/**\n * An error callback\n */\ntype ErrorCallback = (error?: Error | null) => void;\ntype PluginStateSyncCallback = (state: PluginState) => void;\nlet _completionCallback = null;\n\n//Variables defining the current state of the plugin\nlet pluginStatus = status.unavailable;\nlet pluginURL = null;\n\nexport const triggerPluginCompletionEvent = function(error: Error | string) {\n // NetworkError's are not correctly reflected by the plugin status which prevents reloading plugin\n if (error && typeof error === 'string' && error.indexOf('NetworkError') > -1) {\n pluginStatus = status.error;\n }\n\n if (_completionCallback) {\n _completionCallback(error);\n }\n};\n\nfunction sendPluginStateToWorker() {\n evented.fire(new Event('pluginStateChange', {pluginStatus, pluginURL}));\n}\n\nexport const evented = new Evented();\n\nexport const getRTLTextPluginStatus = function () {\n return pluginStatus;\n};\n\nexport const registerForPluginStateChange = function(callback: PluginStateSyncCallback) {\n // Do an initial sync of the state\n callback({pluginStatus, pluginURL});\n // Listen for all future state changes\n evented.on('pluginStateChange', callback);\n return callback;\n};\n\nexport const clearRTLTextPlugin = function() {\n pluginStatus = status.unavailable;\n pluginURL = null;\n _completionCallback = null;\n};\n\nexport const setRTLTextPlugin = function(url: string, callback: ErrorCallback, deferred: boolean = false) {\n if (pluginStatus === status.deferred || pluginStatus === status.loading || pluginStatus === status.loaded) {\n throw new Error('setRTLTextPlugin cannot be called multiple times.');\n }\n pluginURL = browser.resolveURL(url);\n pluginStatus = status.deferred;\n _completionCallback = callback;\n sendPluginStateToWorker();\n\n //Start downloading the plugin immediately if not intending to lazy-load\n if (!deferred) {\n downloadRTLTextPlugin();\n }\n};\n\nexport const downloadRTLTextPlugin = function() {\n if (pluginStatus !== status.deferred || !pluginURL) {\n throw new Error('rtl-text-plugin cannot be downloaded unless a pluginURL is specified');\n }\n pluginStatus = status.loading;\n sendPluginStateToWorker();\n if (pluginURL) {\n getArrayBuffer({url: pluginURL}, (error) => {\n if (error) {\n triggerPluginCompletionEvent(error);\n } else {\n pluginStatus = status.loaded;\n sendPluginStateToWorker();\n }\n });\n }\n};\n\nexport const plugin: {\n applyArabicShaping: Function;\n processBidirectionalText: ((b: string, a: Array) => Array);\n processStyledBidirectionalText: ((c: string, b: Array, a: Array) => Array<[string, Array]>);\n isLoaded: () => boolean;\n isLoading: () => boolean;\n setState: (state: PluginState) => void;\n isParsed: () => boolean;\n getPluginURL: () => string;\n} = {\n applyArabicShaping: null,\n processBidirectionalText: null,\n processStyledBidirectionalText: null,\n isLoaded() {\n return pluginStatus === status.loaded || // Main Thread: loaded if the completion callback returned successfully\n plugin.applyArabicShaping != null; // Web-worker: loaded if the plugin functions have been compiled\n },\n isLoading() { // Main Thread Only: query the loading status, this function does not return the correct value in the worker context.\n return pluginStatus === status.loading;\n },\n setState(state: PluginState) { // Worker thread only: this tells the worker threads that the plugin is available on the Main thread\n if (!isWorker()) throw new Error('Cannot set the state of the rtl-text-plugin when not in the web-worker context');\n\n pluginStatus = state.pluginStatus;\n pluginURL = state.pluginURL;\n },\n isParsed(): boolean {\n if (!isWorker()) throw new Error('rtl-text-plugin is only parsed on the worker-threads');\n\n return plugin.applyArabicShaping != null &&\n plugin.processBidirectionalText != null &&\n plugin.processStyledBidirectionalText != null;\n },\n getPluginURL(): string {\n if (!isWorker()) throw new Error('rtl-text-plugin url can only be queried from the worker threads');\n return pluginURL;\n }\n};\n\nexport const lazyLoadRTLTextPlugin = function() {\n if (!plugin.isLoading() &&\n !plugin.isLoaded() &&\n getRTLTextPluginStatus() === 'deferred'\n ) {\n downloadRTLTextPlugin();\n }\n};\n","import {ZoomHistory} from './zoom_history';\nimport {isStringInSupportedScript} from '../util/script_detection';\nimport {plugin as rtlTextPlugin} from '../source/rtl_text_plugin';\n\nimport type {TransitionSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type CrossfadeParameters = {\n fromScale: number;\n toScale: number;\n t: number;\n};\n\n/**\n * @internal\n * A parameter that can be evaluated to a value\n */\nexport class EvaluationParameters {\n zoom: number;\n now: number;\n fadeDuration: number;\n zoomHistory: ZoomHistory;\n transition: TransitionSpecification;\n\n // \"options\" may also be another EvaluationParameters to copy, see CrossFadedProperty.possiblyEvaluate\n constructor(zoom: number, options?: any) {\n this.zoom = zoom;\n\n if (options) {\n this.now = options.now;\n this.fadeDuration = options.fadeDuration;\n this.zoomHistory = options.zoomHistory;\n this.transition = options.transition;\n } else {\n this.now = 0;\n this.fadeDuration = 0;\n this.zoomHistory = new ZoomHistory();\n this.transition = {};\n }\n }\n\n isSupportedScript(str: string): boolean {\n return isStringInSupportedScript(str, rtlTextPlugin.isLoaded());\n }\n\n crossFadingFactor() {\n if (this.fadeDuration === 0) {\n return 1;\n } else {\n return Math.min((this.now - this.zoomHistory.lastIntegerZoomTime) / this.fadeDuration, 1);\n }\n }\n\n getCrossfadeParameters(): CrossfadeParameters {\n const z = this.zoom;\n const fraction = z - Math.floor(z);\n const t = this.crossFadingFactor();\n\n return z > this.zoomHistory.lastIntegerZoom ?\n {fromScale: 2, toScale: 1, t: fraction + (1 - fraction) * t} :\n {fromScale: 0.5, toScale: 1, t: 1 - (1 - t) * fraction};\n }\n}\n","import {clone, extend, easeCubicInOut} from '../util/util';\nimport {interpolates, Color, StylePropertySpecification, normalizePropertyExpression,\n Feature,\n FeatureState,\n StylePropertyExpression,\n SourceExpression,\n CompositeExpression, TransitionSpecification,\n PropertyValueSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport {register} from '../util/web_worker_transfer';\nimport {EvaluationParameters} from './evaluation_parameters';\n\nimport {CanonicalTileID} from '../source/tile_id';\n\ntype TimePoint = number;\n\n/**\n * A from-to type\n */\nexport type CrossFaded = {\n to: T;\n from: T;\n};\n\n/**\n * @internal\n * Implementations of the `Property` interface:\n *\n * * Hold metadata about a property that's independent of any specific value: stuff like the type of the value,\n * the default value, etc. This comes from the style specification JSON.\n * * Define behavior that needs to be polymorphic across different properties: \"possibly evaluating\"\n * an input value (see below), and interpolating between two possibly-evaluted values.\n *\n * The type `T` is the fully-evaluated value type (e.g. `number`, `string`, `Color`).\n * The type `R` is the intermediate \"possibly evaluated\" value type. See below.\n *\n * There are two main implementations of the interface -- one for properties that allow data-driven values,\n * and one for properties that don't. There are a few \"special case\" implementations as well: one for properties\n * which cross-fade between two values rather than interpolating, one for `heatmap-color` and `line-gradient`,\n * and one for `light-position`.\n */\nexport interface Property {\n specification: StylePropertySpecification;\n possiblyEvaluate(\n value: PropertyValue,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): R;\n interpolate(a: R, b: R, t: number): R;\n}\n\n/**\n * @internal\n * `PropertyValue` represents the value part of a property key-value unit. It's used to represent both\n * paint and layout property values, and regardless of whether or not their property supports data-driven\n * expressions.\n *\n * `PropertyValue` stores the raw input value as seen in a style or a runtime styling API call, i.e. one of the\n * following:\n *\n * * A constant value of the type appropriate for the property\n * * A function which produces a value of that type (but functions are quasi-deprecated in favor of expressions)\n * * An expression which produces a value of that type\n * * \"undefined\"/\"not present\", in which case the property is assumed to take on its default value.\n *\n * In addition to storing the original input value, `PropertyValue` also stores a normalized representation,\n * effectively treating functions as if they are expressions, and constant or default values as if they are\n * (constant) expressions.\n */\nexport class PropertyValue {\n property: Property;\n value: PropertyValueSpecification | void;\n expression: StylePropertyExpression;\n\n constructor(property: Property, value: PropertyValueSpecification | void) {\n this.property = property;\n this.value = value;\n this.expression = normalizePropertyExpression(value === undefined ? property.specification.default : value, property.specification);\n }\n\n isDataDriven(): boolean {\n return this.expression.kind === 'source' || this.expression.kind === 'composite';\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): R {\n return this.property.possiblyEvaluate(this, parameters, canonical, availableImages);\n }\n}\n\nexport type TransitionParameters = {\n now: TimePoint;\n transition: TransitionSpecification;\n};\n\n/**\n * @internal\n * Paint properties are _transitionable_: they can change in a fluid manner, interpolating or cross-fading between\n * old and new value. The duration of the transition, and the delay before it begins, is configurable.\n *\n * `TransitionablePropertyValue` is a compositional class that stores both the property value and that transition\n * configuration.\n *\n * A `TransitionablePropertyValue` can calculate the next step in the evaluation chain for paint property values:\n * `TransitioningPropertyValue`.\n */\nclass TransitionablePropertyValue {\n property: Property;\n value: PropertyValue;\n transition: TransitionSpecification | void;\n\n constructor(property: Property) {\n this.property = property;\n this.value = new PropertyValue(property, undefined);\n }\n\n transitioned(parameters: TransitionParameters, prior: TransitioningPropertyValue): TransitioningPropertyValue {\n return new TransitioningPropertyValue(this.property, this.value, prior,\n extend({}, parameters.transition, this.transition), parameters.now);\n }\n\n untransitioned(): TransitioningPropertyValue {\n return new TransitioningPropertyValue(this.property, this.value, null, {}, 0);\n }\n}\n\n/**\n * @internal\n * `Transitionable` stores a map of all (property name, `TransitionablePropertyValue`) pairs for paint properties of a\n * given layer type. It can calculate the `TransitioningPropertyValue`s for all of them at once, producing a\n * `Transitioning` instance for the same set of properties.\n */\nexport class Transitionable {\n _properties: Properties;\n _values: {[K in keyof Props]: TransitionablePropertyValue};\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = (Object.create(properties.defaultTransitionablePropertyValues) as any);\n }\n\n getValue(name: S): PropertyValueSpecification | void {\n return clone(this._values[name].value.value);\n }\n\n setValue(name: S, value: PropertyValueSpecification | void) {\n if (!Object.prototype.hasOwnProperty.call(this._values, name)) {\n this._values[name] = new TransitionablePropertyValue(this._values[name].property);\n }\n // Note that we do not _remove_ an own property in the case where a value is being reset\n // to the default: the transition might still be non-default.\n this._values[name].value = new PropertyValue(this._values[name].property, value === null ? undefined : clone(value));\n }\n\n getTransition(name: S): TransitionSpecification | void {\n return clone(this._values[name].transition);\n }\n\n setTransition(name: S, value: TransitionSpecification | void) {\n if (!Object.prototype.hasOwnProperty.call(this._values, name)) {\n this._values[name] = new TransitionablePropertyValue(this._values[name].property);\n }\n this._values[name].transition = clone(value) || undefined;\n }\n\n serialize() {\n const result: any = {};\n for (const property of Object.keys(this._values)) {\n const value = this.getValue(property as keyof Props);\n if (value !== undefined) {\n result[property] = value;\n }\n\n const transition = this.getTransition(property as keyof Props);\n if (transition !== undefined) {\n result[`${property}-transition`] = transition;\n }\n }\n return result;\n }\n\n transitioned(parameters: TransitionParameters, prior: Transitioning): Transitioning {\n const result = new Transitioning(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].transitioned(parameters, prior._values[property]);\n }\n return result;\n }\n\n untransitioned(): Transitioning {\n const result = new Transitioning(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].untransitioned();\n }\n return result;\n }\n}\n\n/**\n * @internal\n * `TransitioningPropertyValue` implements the first of two intermediate steps in the evaluation chain of a paint\n * property value. In this step, transitions between old and new values are handled: as long as the transition is in\n * progress, `TransitioningPropertyValue` maintains a reference to the prior value, and interpolates between it and\n * the new value based on the current time and the configured transition duration and delay. The product is the next\n * step in the evaluation chain: the \"possibly evaluated\" result type `R`. See below for more on this concept.\n */\nclass TransitioningPropertyValue {\n property: Property;\n value: PropertyValue;\n prior: TransitioningPropertyValue;\n begin: TimePoint;\n end: TimePoint;\n\n constructor(property: Property,\n value: PropertyValue,\n prior: TransitioningPropertyValue,\n transition: TransitionSpecification,\n now: TimePoint) {\n this.property = property;\n this.value = value;\n this.begin = now + transition.delay || 0;\n this.end = this.begin + transition.duration || 0;\n if (property.specification.transition && (transition.delay || transition.duration)) {\n this.prior = prior;\n }\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical: CanonicalTileID,\n availableImages: Array\n ): R {\n const now = parameters.now || 0;\n const finalValue = this.value.possiblyEvaluate(parameters, canonical, availableImages);\n const prior = this.prior;\n if (!prior) {\n // No prior value.\n return finalValue;\n } else if (now > this.end) {\n // Transition from prior value is now complete.\n this.prior = null;\n return finalValue;\n } else if (this.value.isDataDriven()) {\n // Transitions to data-driven properties are not supported.\n // We snap immediately to the data-driven value so that, when we perform layout,\n // we see the data-driven function and can use it to populate vertex buffers.\n this.prior = null;\n return finalValue;\n } else if (now < this.begin) {\n // Transition hasn't started yet.\n return prior.possiblyEvaluate(parameters, canonical, availableImages);\n } else {\n // Interpolate between recursively-calculated prior value and final.\n const t = (now - this.begin) / (this.end - this.begin);\n return this.property.interpolate(prior.possiblyEvaluate(parameters, canonical, availableImages), finalValue, easeCubicInOut(t));\n }\n }\n}\n\n/**\n * @internal\n * `Transitioning` stores a map of all (property name, `TransitioningPropertyValue`) pairs for paint properties of a\n * given layer type. It can calculate the possibly-evaluated values for all of them at once, producing a\n * `PossiblyEvaluated` instance for the same set of properties.\n */\nexport class Transitioning {\n _properties: Properties;\n _values: {[K in keyof Props]: PossiblyEvaluatedPropertyValue};\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = (Object.create(properties.defaultTransitioningPropertyValues) as any);\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluated {\n const result = new PossiblyEvaluated(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].possiblyEvaluate(parameters, canonical, availableImages);\n }\n return result;\n }\n\n hasTransition() {\n for (const property of Object.keys(this._values)) {\n if (this._values[property].prior) {\n return true;\n }\n }\n return false;\n }\n}\n\n// ------- Layout -------\n\n/**\n * Because layout properties are not transitionable, they have a simpler representation and evaluation chain than\n * paint properties: `PropertyValue`s are possibly evaluated, producing possibly evaluated values, which are then\n * fully evaluated.\n *\n * `Layout` stores a map of all (property name, `PropertyValue`) pairs for layout properties of a\n * given layer type. It can calculate the possibly-evaluated values for all of them at once, producing a\n * `PossiblyEvaluated` instance for the same set of properties.\n */\nexport class Layout {\n _properties: Properties;\n _values: {[K in keyof Props]: PropertyValue>};\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = (Object.create(properties.defaultPropertyValues) as any);\n }\n\n hasValue(name: S) {\n return this._values[name].value !== undefined;\n }\n\n getValue(name: S) {\n return clone(this._values[name].value);\n }\n\n setValue(name: S, value: any) {\n this._values[name] = new PropertyValue(this._values[name].property, value === null ? undefined : clone(value)) as any;\n }\n\n serialize() {\n const result: any = {};\n for (const property of Object.keys(this._values)) {\n const value = this.getValue(property as keyof Props);\n if (value !== undefined) {\n result[property] = value;\n }\n }\n return result;\n }\n\n possiblyEvaluate(\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluated {\n const result = new PossiblyEvaluated(this._properties);\n for (const property of Object.keys(this._values)) {\n result._values[property] = this._values[property].possiblyEvaluate(parameters, canonical, availableImages);\n }\n return result;\n }\n}\n\n// ------- PossiblyEvaluated -------\n\n/**\n * \"Possibly evaluated value\" is an intermediate stage in the evaluation chain for both paint and layout property\n * values. The purpose of this stage is to optimize away unnecessary recalculations for data-driven properties. Code\n * which uses data-driven property values must assume that the value is dependent on feature data, and request that it\n * be evaluated for each feature. But when that property value is in fact a constant or camera function, the calculation\n * will not actually depend on the feature, and we can benefit from returning the prior result of having done the\n * evaluation once, ahead of time, in an intermediate step whose inputs are just the value and \"global\" parameters\n * such as current zoom level.\n *\n * `PossiblyEvaluatedValue` represents the three possible outcomes of this step: if the input value was a constant or\n * camera expression, then the \"possibly evaluated\" result is a constant value. Otherwise, the input value was either\n * a source or composite expression, and we must defer final evaluation until supplied a feature. We separate\n * the source and composite cases because they are handled differently when generating GL attributes, buffers, and\n * uniforms.\n *\n * Note that `PossiblyEvaluatedValue` (and `PossiblyEvaluatedPropertyValue`, below) are _not_ used for properties that\n * do not allow data-driven values. For such properties, we know that the \"possibly evaluated\" result is always a constant\n * scalar value. See below.\n */\ntype PossiblyEvaluatedValue = {\n kind: 'constant';\n value: T;\n} | SourceExpression | CompositeExpression;\n\n/**\n * @internal\n * `PossiblyEvaluatedPropertyValue` is used for data-driven paint and layout property values. It holds a\n * `PossiblyEvaluatedValue` and the `GlobalProperties` that were used to generate it. You're not allowed to supply\n * a different set of `GlobalProperties` when performing the final evaluation because they would be ignored in the\n * case where the input value was a constant or camera function.\n */\nexport class PossiblyEvaluatedPropertyValue {\n property: DataDrivenProperty;\n value: PossiblyEvaluatedValue;\n parameters: EvaluationParameters;\n\n constructor(property: DataDrivenProperty, value: PossiblyEvaluatedValue, parameters: EvaluationParameters) {\n this.property = property;\n this.value = value;\n this.parameters = parameters;\n }\n\n isConstant(): boolean {\n return this.value.kind === 'constant';\n }\n\n constantOr(value: T): T {\n if (this.value.kind === 'constant') {\n return this.value.value;\n } else {\n return value;\n }\n }\n\n evaluate(\n feature: Feature,\n featureState: FeatureState,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): T {\n return this.property.evaluate(this.value, this.parameters, feature, featureState, canonical, availableImages);\n }\n}\n\n/**\n * @internal\n * `PossiblyEvaluated` stores a map of all (property name, `R`) pairs for paint or layout properties of a\n * given layer type.\n */\nexport class PossiblyEvaluated {\n _properties: Properties;\n _values: PossibleEvaluatedProps;\n\n constructor(properties: Properties) {\n this._properties = properties;\n this._values = Object.create(properties.defaultPossiblyEvaluatedValues);\n }\n\n get(name: S): PossibleEvaluatedProps[S] {\n return this._values[name];\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for properties that do not permit data-driven (source or composite) expressions.\n * This restriction allows us to declare statically that the result of possibly evaluating this kind of property\n * is in fact always the scalar type `T`, and can be used without further evaluating the value on a per-feature basis.\n */\nexport class DataConstantProperty implements Property {\n specification: StylePropertySpecification;\n\n constructor(specification: StylePropertySpecification) {\n this.specification = specification;\n }\n\n possiblyEvaluate(value: PropertyValue, parameters: EvaluationParameters): T {\n if (value.isDataDriven()) throw new Error('Value should not be data driven');\n return value.expression.evaluate(parameters);\n }\n\n interpolate(a: T, b: T, t: number): T {\n const interpolationType = this.specification.type as keyof typeof interpolates;\n const interpolationFn = interpolates[interpolationType] as ((from: T, to: T, t: number) => T) | undefined;\n if (interpolationFn) {\n return interpolationFn(a, b, t);\n } else {\n return a;\n }\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for properties that permit data-driven (source or composite) expressions.\n * The result of possibly evaluating this kind of property is `PossiblyEvaluatedPropertyValue`; obtaining\n * a scalar value `T` requires further evaluation on a per-feature basis.\n */\nexport class DataDrivenProperty implements Property> {\n specification: StylePropertySpecification;\n overrides: any;\n\n constructor(specification: StylePropertySpecification, overrides?: any) {\n this.specification = specification;\n this.overrides = overrides;\n }\n\n possiblyEvaluate(\n value: PropertyValue>,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluatedPropertyValue {\n if (value.expression.kind === 'constant' || value.expression.kind === 'camera') {\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: value.expression.evaluate(parameters, null, {}, canonical, availableImages)}, parameters);\n } else {\n return new PossiblyEvaluatedPropertyValue(this, value.expression, parameters);\n }\n }\n\n interpolate(\n a: PossiblyEvaluatedPropertyValue,\n b: PossiblyEvaluatedPropertyValue,\n t: number\n ): PossiblyEvaluatedPropertyValue {\n // If either possibly-evaluated value is non-constant, give up: we aren't able to interpolate data-driven values.\n if (a.value.kind !== 'constant' || b.value.kind !== 'constant') {\n return a;\n }\n\n // Special case hack solely for fill-outline-color. The undefined value is subsequently handled in\n // FillStyleLayer#recalculate, which sets fill-outline-color to the fill-color value if the former\n // is a PossiblyEvaluatedPropertyValue containing a constant undefined value. In addition to the\n // return value here, the other source of a PossiblyEvaluatedPropertyValue containing a constant\n // undefined value is the \"default value\" for fill-outline-color held in\n // `Properties#defaultPossiblyEvaluatedValues`, which serves as the prototype of\n // `PossiblyEvaluated#_values`.\n if (a.value.value === undefined || b.value.value === undefined) {\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: undefined}, a.parameters);\n }\n\n const interpolationType = this.specification.type as keyof typeof interpolates;\n const interpolationFn = interpolates[interpolationType] as ((from: T, to: T, t: number) => T) | undefined;\n if (interpolationFn) {\n const interpolatedValue = interpolationFn(a.value.value, b.value.value, t);\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: interpolatedValue}, a.parameters);\n } else {\n return a;\n }\n }\n\n evaluate(\n value: PossiblyEvaluatedValue,\n parameters: EvaluationParameters,\n feature: Feature,\n featureState: FeatureState,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): T {\n if (value.kind === 'constant') {\n return value.value;\n } else {\n return value.evaluate(parameters, feature, featureState, canonical, availableImages);\n }\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for data driven `line-pattern` which are transitioned by cross-fading\n * rather than interpolation.\n */\n\nexport class CrossFadedDataDrivenProperty extends DataDrivenProperty> {\n\n possiblyEvaluate(\n value: PropertyValue, PossiblyEvaluatedPropertyValue>>,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): PossiblyEvaluatedPropertyValue> {\n if (value.value === undefined) {\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: undefined}, parameters);\n } else if (value.expression.kind === 'constant') {\n const evaluatedValue = value.expression.evaluate(parameters, null, {}, canonical, availableImages);\n const isImageExpression = value.property.specification.type as any === 'resolvedImage';\n const constantValue = isImageExpression && typeof evaluatedValue !== 'string' ? evaluatedValue.name : evaluatedValue;\n const constant = this._calculate(constantValue, constantValue, constantValue, parameters);\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: constant}, parameters);\n } else if (value.expression.kind === 'camera') {\n const cameraVal = this._calculate(\n value.expression.evaluate({zoom: parameters.zoom - 1.0}),\n value.expression.evaluate({zoom: parameters.zoom}),\n value.expression.evaluate({zoom: parameters.zoom + 1.0}),\n parameters);\n return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: cameraVal}, parameters);\n } else {\n // source or composite expression\n return new PossiblyEvaluatedPropertyValue(this, value.expression, parameters);\n }\n }\n\n evaluate(\n value: PossiblyEvaluatedValue>,\n globals: EvaluationParameters,\n feature: Feature,\n featureState: FeatureState,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): CrossFaded {\n if (value.kind === 'source') {\n const constant = value.evaluate(globals, feature, featureState, canonical, availableImages);\n return this._calculate(constant, constant, constant, globals);\n } else if (value.kind === 'composite') {\n return this._calculate(\n value.evaluate({zoom: Math.floor(globals.zoom) - 1.0}, feature, featureState),\n value.evaluate({zoom: Math.floor(globals.zoom)}, feature, featureState),\n value.evaluate({zoom: Math.floor(globals.zoom) + 1.0}, feature, featureState),\n globals);\n } else {\n return value.value;\n }\n }\n\n _calculate(min: T, mid: T, max: T, parameters: EvaluationParameters): CrossFaded {\n const z = parameters.zoom;\n return z > parameters.zoomHistory.lastIntegerZoom ? {from: min, to: mid} : {from: max, to: mid};\n }\n\n interpolate(a: PossiblyEvaluatedPropertyValue>): PossiblyEvaluatedPropertyValue> {\n return a;\n }\n}\n/**\n * @internal\n * An implementation of `Property` for `*-pattern` and `line-dasharray`, which are transitioned by cross-fading\n * rather than interpolation.\n */\nexport class CrossFadedProperty implements Property> {\n specification: StylePropertySpecification;\n\n constructor(specification: StylePropertySpecification) {\n this.specification = specification;\n }\n\n possiblyEvaluate(\n value: PropertyValue>,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): CrossFaded {\n if (value.value === undefined) {\n return undefined;\n } else if (value.expression.kind === 'constant') {\n const constant = value.expression.evaluate(parameters, null, {}, canonical, availableImages);\n return this._calculate(constant, constant, constant, parameters);\n } else {\n return this._calculate(\n value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom - 1.0), parameters)),\n value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom), parameters)),\n value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom + 1.0), parameters)),\n parameters);\n }\n }\n\n _calculate(min: T, mid: T, max: T, parameters: EvaluationParameters): CrossFaded {\n const z = parameters.zoom;\n return z > parameters.zoomHistory.lastIntegerZoom ? {from: min, to: mid} : {from: max, to: mid};\n }\n\n interpolate(a?: CrossFaded | null): CrossFaded {\n return a;\n }\n}\n\n/**\n * @internal\n * An implementation of `Property` for `heatmap-color` and `line-gradient`. Interpolation is a no-op, and\n * evaluation returns a boolean value in order to indicate its presence, but the real\n * evaluation happens in StyleLayer classes.\n */\n\nexport class ColorRampProperty implements Property {\n specification: StylePropertySpecification;\n\n constructor(specification: StylePropertySpecification) {\n this.specification = specification;\n }\n\n possiblyEvaluate(\n value: PropertyValue,\n parameters: EvaluationParameters,\n canonical?: CanonicalTileID,\n availableImages?: Array\n ): boolean {\n return !!value.expression.evaluate(parameters, null, {}, canonical, availableImages);\n }\n\n interpolate(): boolean { return false; }\n}\n\n/**\n * @internal\n * `Properties` holds objects containing default values for the layout or paint property set of a given\n * layer type. These objects are immutable, and they are used as the prototypes for the `_values` members of\n * `Transitionable`, `Transitioning`, `Layout`, and `PossiblyEvaluated`. This allows these classes to avoid\n * doing work in the common case where a property has no explicit value set and should be considered to take\n * on the default value: using `for (const property of Object.keys(this._values))`, they can iterate over\n * only the _own_ properties of `_values`, skipping repeated calculation of transitions and possible/final\n * evaluations for defaults, the result of which will always be the same.\n */\nexport class Properties {\n properties: Props;\n defaultPropertyValues: {[K in keyof Props]: PropertyValue};\n defaultTransitionablePropertyValues: {[K in keyof Props]: TransitionablePropertyValue};\n defaultTransitioningPropertyValues: {[K in keyof Props]: TransitioningPropertyValue};\n defaultPossiblyEvaluatedValues: {[K in keyof Props]: PossiblyEvaluatedPropertyValue};\n overridableProperties: Array;\n\n constructor(properties: Props) {\n this.properties = properties;\n this.defaultPropertyValues = ({} as any);\n this.defaultTransitionablePropertyValues = ({} as any);\n this.defaultTransitioningPropertyValues = ({} as any);\n this.defaultPossiblyEvaluatedValues = ({} as any);\n this.overridableProperties = ([] as any);\n\n for (const property in properties) {\n const prop = properties[property] as any;\n if (prop.specification.overridable) {\n this.overridableProperties.push(property);\n }\n const defaultPropertyValue = this.defaultPropertyValues[property] =\n new PropertyValue(prop, undefined);\n const defaultTransitionablePropertyValue = this.defaultTransitionablePropertyValues[property] =\n new TransitionablePropertyValue(prop);\n this.defaultTransitioningPropertyValues[property] =\n defaultTransitionablePropertyValue.untransitioned();\n this.defaultPossiblyEvaluatedValues[property] =\n defaultPropertyValue.possiblyEvaluate({} as any);\n }\n }\n}\n\nregister('DataDrivenProperty', DataDrivenProperty);\nregister('DataConstantProperty', DataConstantProperty);\nregister('CrossFadedDataDrivenProperty', CrossFadedDataDrivenProperty);\nregister('CrossFadedProperty', CrossFadedProperty);\nregister('ColorRampProperty', ColorRampProperty);\n","import {filterObject} from '../util/util';\n\nimport {latest as styleSpec, supportsPropertyExpression} from '@maplibre/maplibre-gl-style-spec';\nimport {\n validateStyle,\n validateLayoutProperty,\n validatePaintProperty,\n emitValidationErrors\n} from './validate_style';\nimport {Evented} from '../util/evented';\nimport {Layout, Transitionable, Transitioning, Properties, PossiblyEvaluated, PossiblyEvaluatedPropertyValue} from './properties';\n\nimport type {Bucket} from '../data/bucket';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureFilter, FeatureState,\n LayerSpecification,\n FilterSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {TransitionParameters, PropertyValue} from './properties';\nimport {EvaluationParameters} from './evaluation_parameters';\nimport type {CrossfadeParameters} from './evaluation_parameters';\n\nimport type {Transform} from '../geo/transform';\nimport type {CustomLayerInterface} from './style_layer/custom_style_layer';\nimport type {Map} from '../ui/map';\nimport type {StyleSetterOptions} from './style';\nimport {mat4} from 'gl-matrix';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nconst TRANSITION_SUFFIX = '-transition';\n\n/**\n * A base class for style layers\n */\nexport abstract class StyleLayer extends Evented {\n id: string;\n metadata: unknown;\n type: LayerSpecification['type'] | CustomLayerInterface['type'];\n source: string;\n sourceLayer: string;\n minzoom: number;\n maxzoom: number;\n filter: FilterSpecification | void;\n visibility: 'visible' | 'none' | void;\n _crossfadeParameters: CrossfadeParameters;\n\n _unevaluatedLayout: Layout;\n readonly layout: unknown;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n readonly paint: unknown;\n\n _featureFilter: FeatureFilter;\n\n readonly onAdd: ((map: Map) => void);\n readonly onRemove: ((map: Map) => void);\n\n queryRadius?(bucket: Bucket): number;\n queryIntersectsFeature?(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number,\n pixelPosMatrix: mat4\n ): boolean | number;\n\n constructor(layer: LayerSpecification | CustomLayerInterface, properties: Readonly<{\n layout?: Properties;\n paint?: Properties;\n }>) {\n super();\n\n this.id = layer.id;\n this.type = layer.type;\n this._featureFilter = {filter: () => true, needGeometry: false};\n\n if (layer.type === 'custom') return;\n\n layer = (layer as any as LayerSpecification);\n\n this.metadata = layer.metadata;\n this.minzoom = layer.minzoom;\n this.maxzoom = layer.maxzoom;\n\n if (layer.type !== 'background') {\n this.source = layer.source;\n this.sourceLayer = layer['source-layer'];\n this.filter = layer.filter;\n }\n\n if (properties.layout) {\n this._unevaluatedLayout = new Layout(properties.layout);\n }\n\n if (properties.paint) {\n this._transitionablePaint = new Transitionable(properties.paint);\n\n for (const property in layer.paint) {\n this.setPaintProperty(property, layer.paint[property], {validate: false});\n }\n for (const property in layer.layout) {\n this.setLayoutProperty(property, layer.layout[property], {validate: false});\n }\n\n this._transitioningPaint = this._transitionablePaint.untransitioned();\n //$FlowFixMe\n this.paint = new PossiblyEvaluated(properties.paint);\n }\n }\n\n getCrossfadeParameters() {\n return this._crossfadeParameters;\n }\n\n getLayoutProperty(name: string) {\n if (name === 'visibility') {\n return this.visibility;\n }\n\n return this._unevaluatedLayout.getValue(name);\n }\n\n setLayoutProperty(name: string, value: any, options: StyleSetterOptions = {}) {\n if (value !== null && value !== undefined) {\n const key = `layers.${this.id}.layout.${name}`;\n if (this._validate(validateLayoutProperty, key, name, value, options)) {\n return;\n }\n }\n\n if (name === 'visibility') {\n this.visibility = value;\n return;\n }\n\n this._unevaluatedLayout.setValue(name, value);\n }\n\n getPaintProperty(name: string) {\n if (name.endsWith(TRANSITION_SUFFIX)) {\n return this._transitionablePaint.getTransition(name.slice(0, -TRANSITION_SUFFIX.length));\n } else {\n return this._transitionablePaint.getValue(name);\n }\n }\n\n setPaintProperty(name: string, value: unknown, options: StyleSetterOptions = {}) {\n if (value !== null && value !== undefined) {\n const key = `layers.${this.id}.paint.${name}`;\n if (this._validate(validatePaintProperty, key, name, value, options)) {\n return false;\n }\n }\n\n if (name.endsWith(TRANSITION_SUFFIX)) {\n this._transitionablePaint.setTransition(name.slice(0, -TRANSITION_SUFFIX.length), (value as any) || undefined);\n return false;\n } else {\n const transitionable = this._transitionablePaint._values[name];\n const isCrossFadedProperty = transitionable.property.specification['property-type'] === 'cross-faded-data-driven';\n const wasDataDriven = transitionable.value.isDataDriven();\n const oldValue = transitionable.value;\n\n this._transitionablePaint.setValue(name, value);\n this._handleSpecialPaintPropertyUpdate(name);\n\n const newValue = this._transitionablePaint._values[name].value;\n const isDataDriven = newValue.isDataDriven();\n\n // if a cross-faded value is changed, we need to make sure the new icons get added to each tile's iconAtlas\n // so a call to _updateLayer is necessary, and we return true from this function so it gets called in\n // Style#setPaintProperty\n return isDataDriven || wasDataDriven || isCrossFadedProperty || this._handleOverridablePaintPropertyUpdate(name, oldValue, newValue);\n }\n }\n\n _handleSpecialPaintPropertyUpdate(_: string) {\n // No-op; can be overridden by derived classes.\n }\n\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n _handleOverridablePaintPropertyUpdate(name: string, oldValue: PropertyValue, newValue: PropertyValue): boolean {\n // No-op; can be overridden by derived classes.\n return false;\n }\n\n isHidden(zoom: number) {\n if (this.minzoom && zoom < this.minzoom) return true;\n if (this.maxzoom && zoom >= this.maxzoom) return true;\n return this.visibility === 'none';\n }\n\n updateTransitions(parameters: TransitionParameters) {\n this._transitioningPaint = this._transitionablePaint.transitioned(parameters, this._transitioningPaint);\n }\n\n hasTransition() {\n return this._transitioningPaint.hasTransition();\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n if (parameters.getCrossfadeParameters) {\n this._crossfadeParameters = parameters.getCrossfadeParameters();\n }\n\n if (this._unevaluatedLayout) {\n (this as any).layout = this._unevaluatedLayout.possiblyEvaluate(parameters, undefined, availableImages);\n }\n\n (this as any).paint = this._transitioningPaint.possiblyEvaluate(parameters, undefined, availableImages);\n }\n\n serialize(): LayerSpecification {\n const output: LayerSpecification = {\n 'id': this.id,\n 'type': this.type as LayerSpecification['type'],\n 'source': this.source,\n 'source-layer': this.sourceLayer,\n 'metadata': this.metadata,\n 'minzoom': this.minzoom,\n 'maxzoom': this.maxzoom,\n 'filter': this.filter as FilterSpecification,\n 'layout': this._unevaluatedLayout && this._unevaluatedLayout.serialize(),\n 'paint': this._transitionablePaint && this._transitionablePaint.serialize()\n };\n\n if (this.visibility) {\n output.layout = output.layout || {};\n output.layout.visibility = this.visibility;\n }\n\n return filterObject(output, (value, key) => {\n return value !== undefined &&\n !(key === 'layout' && !Object.keys(value).length) &&\n !(key === 'paint' && !Object.keys(value).length);\n });\n }\n\n _validate(validate: Function, key: string, name: string, value: unknown, options: StyleSetterOptions = {}) {\n if (options && options.validate === false) {\n return false;\n }\n return emitValidationErrors(this, validate.call(validateStyle, {\n key,\n layerType: this.type,\n objectKey: name,\n value,\n styleSpec,\n // Workaround for https://github.com/mapbox/mapbox-gl-js/issues/2407\n style: {glyphs: true, sprite: true}\n }));\n }\n\n is3D() {\n return false;\n }\n\n isTileClipped() {\n return false;\n }\n\n hasOffscreenPass() {\n return false;\n }\n\n resize() {\n // noop\n }\n\n isStateDependent() {\n for (const property in (this as any).paint._values) {\n const value = (this as any).paint.get(property);\n if (!(value instanceof PossiblyEvaluatedPropertyValue) || !supportsPropertyExpression(value.property.specification)) {\n continue;\n }\n\n if ((value.value.kind === 'source' || value.value.kind === 'composite') &&\n value.value.isStateDependent) {\n return true;\n }\n }\n return false;\n }\n}\n","// Note: all \"sizes\" are measured in bytes\n\nimport type {Transferable} from '../types/transferable';\n\n/**\n * @internal\n * A view type size\n */\nconst viewTypes = {\n 'Int8': Int8Array,\n 'Uint8': Uint8Array,\n 'Int16': Int16Array,\n 'Uint16': Uint16Array,\n 'Int32': Int32Array,\n 'Uint32': Uint32Array,\n 'Float32': Float32Array\n};\n\n/**\n * @internal\n * A view type size\n */\nexport type ViewType = keyof typeof viewTypes;\n\n/** @internal */\nclass Struct {\n _pos1: number;\n _pos2: number;\n _pos4: number;\n _pos8: number;\n readonly _structArray: StructArray;\n\n // The following properties are defined on the prototype of sub classes.\n size: number;\n\n /**\n * @param structArray - The StructArray the struct is stored in\n * @param index - The index of the struct in the StructArray.\n */\n constructor(structArray: StructArray, index: number) {\n (this as any)._structArray = structArray;\n this._pos1 = index * this.size;\n this._pos2 = this._pos1 / 2;\n this._pos4 = this._pos1 / 4;\n this._pos8 = this._pos1 / 8;\n }\n}\n\nconst DEFAULT_CAPACITY = 128;\nconst RESIZE_MULTIPLIER = 5;\n\n/**\n * @internal\n * A struct array memeber\n */\nexport type StructArrayMember = {\n name: string;\n type: ViewType;\n components: number;\n offset: number;\n};\n\nexport type StructArrayLayout = {\n members: Array;\n size: number;\n alignment: number;\n};\n\n/**\n * An array that can be desialized\n */\nexport type SerializedStructArray = {\n length: number;\n arrayBuffer: ArrayBuffer;\n};\n\n/**\n * @internal\n * `StructArray` provides an abstraction over `ArrayBuffer` and `TypedArray`\n * making it behave like an array of typed structs.\n *\n * Conceptually, a StructArray is comprised of elements, i.e., instances of its\n * associated struct type. Each particular struct type, together with an\n * alignment size, determines the memory layout of a StructArray whose elements\n * are of that type. Thus, for each such layout that we need, we have\n * a corresponding StructArrayLayout class, inheriting from StructArray and\n * implementing `emplaceBack()` and `_refreshViews()`.\n *\n * In some cases, where we need to access particular elements of a StructArray,\n * we implement a more specific subclass that inherits from one of the\n * StructArrayLayouts and adds a `get(i): T` accessor that returns a structured\n * object whose properties are proxies into the underlying memory space for the\n * i-th element. This affords the convience of working with (seemingly) plain\n * Javascript objects without the overhead of serializing/deserializing them\n * into ArrayBuffers for efficient web worker transfer.\n */\nabstract class StructArray {\n capacity: number;\n length: number;\n isTransferred: boolean;\n arrayBuffer: ArrayBuffer;\n uint8: Uint8Array;\n\n // The following properties are defined on the prototype.\n members: Array;\n bytesPerElement: number;\n abstract emplaceBack(...v: number[]);\n abstract emplace(i: number, ...v: number[]);\n\n constructor() {\n this.isTransferred = false;\n this.capacity = -1;\n this.resize(0);\n }\n\n /**\n * Serialize a StructArray instance. Serializes both the raw data and the\n * metadata needed to reconstruct the StructArray base class during\n * deserialization.\n */\n static serialize(array: StructArray, transferables?: Array): SerializedStructArray {\n\n array._trim();\n\n if (transferables) {\n array.isTransferred = true;\n transferables.push(array.arrayBuffer);\n }\n\n return {\n length: array.length,\n arrayBuffer: array.arrayBuffer,\n };\n }\n\n static deserialize(input: SerializedStructArray) {\n const structArray = Object.create(this.prototype);\n structArray.arrayBuffer = input.arrayBuffer;\n structArray.length = input.length;\n structArray.capacity = input.arrayBuffer.byteLength / structArray.bytesPerElement;\n structArray._refreshViews();\n return structArray;\n }\n\n /**\n * Resize the array to discard unused capacity.\n */\n _trim() {\n if (this.length !== this.capacity) {\n this.capacity = this.length;\n this.arrayBuffer = this.arrayBuffer.slice(0, this.length * this.bytesPerElement);\n this._refreshViews();\n }\n }\n\n /**\n * Resets the length of the array to 0 without de-allocating capcacity.\n */\n clear() {\n this.length = 0;\n }\n\n /**\n * Resize the array.\n * If `n` is greater than the current length then additional elements with undefined values are added.\n * If `n` is less than the current length then the array will be reduced to the first `n` elements.\n * @param n - The new size of the array.\n */\n resize(n: number) {\n this.reserve(n);\n this.length = n;\n }\n\n /**\n * Indicate a planned increase in size, so that any necessary allocation may\n * be done once, ahead of time.\n * @param n - The expected size of the array.\n */\n reserve(n: number) {\n if (n > this.capacity) {\n this.capacity = Math.max(n, Math.floor(this.capacity * RESIZE_MULTIPLIER), DEFAULT_CAPACITY);\n this.arrayBuffer = new ArrayBuffer(this.capacity * this.bytesPerElement);\n\n const oldUint8Array = this.uint8;\n this._refreshViews();\n if (oldUint8Array) this.uint8.set(oldUint8Array);\n }\n }\n\n /**\n * Create TypedArray views for the current ArrayBuffer.\n */\n _refreshViews() {\n throw new Error('_refreshViews() must be implemented by each concrete StructArray layout');\n }\n}\n\n/**\n * Given a list of member fields, create a full StructArrayLayout, in\n * particular calculating the correct byte offset for each field. This data\n * is used at build time to generate StructArrayLayout_*#emplaceBack() and\n * other accessors, and at runtime for binding vertex buffer attributes.\n */\nfunction createLayout(\n members: Array<{\n name: string;\n type: ViewType;\n readonly components?: number;\n }>,\n alignment: number = 1\n): StructArrayLayout {\n\n let offset = 0;\n let maxSize = 0;\n const layoutMembers = members.map((member) => {\n const typeSize = sizeOf(member.type);\n const memberOffset = offset = align(offset, Math.max(alignment, typeSize));\n const components = member.components || 1;\n\n maxSize = Math.max(maxSize, typeSize);\n offset += typeSize * components;\n\n return {\n name: member.name,\n type: member.type,\n components,\n offset: memberOffset,\n };\n });\n\n const size = align(offset, Math.max(maxSize, alignment));\n\n return {\n members: layoutMembers,\n size,\n alignment\n };\n}\n\nfunction sizeOf(type: ViewType): number {\n return viewTypes[type].BYTES_PER_ELEMENT;\n}\n\nfunction align(offset: number, size: number): number {\n return Math.ceil(offset / size) * size;\n}\n\nexport {StructArray, Struct, viewTypes, createLayout};\n","// This file is generated. Edit build/generate-struct-arrays.ts, then run `npm run codegen`.\n\nimport {Struct, StructArray} from '../util/struct_array';\nimport {register} from '../util/web_worker_transfer';\nimport Point from '@mapbox/point-geometry';\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n *\n */\nclass StructArrayLayout2i4 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1);\n }\n\n public emplace(i: number, v0: number, v1: number) {\n const o2 = i * 2;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n return i;\n }\n}\n\nStructArrayLayout2i4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout2i4', StructArrayLayout2i4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[3]\n *\n */\nclass StructArrayLayout3i6 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o2 = i * 3;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout3i6.prototype.bytesPerElement = 6;\nregister('StructArrayLayout3i6', StructArrayLayout3i6);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[4]\n *\n */\nclass StructArrayLayout4i8 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number) {\n const o2 = i * 4;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n return i;\n }\n}\n\nStructArrayLayout4i8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout4i8', StructArrayLayout4i8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Int16[4]\n *\n */\nclass StructArrayLayout2i4i12 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const o2 = i * 6;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n return i;\n }\n}\n\nStructArrayLayout2i4i12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout2i4i12', StructArrayLayout2i4i12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Uint8[4]\n *\n */\nclass StructArrayLayout2i4ub8 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const o2 = i * 4;\n const o1 = i * 8;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.uint8[o1 + 4] = v2;\n this.uint8[o1 + 5] = v3;\n this.uint8[o1 + 6] = v4;\n this.uint8[o1 + 7] = v5;\n return i;\n }\n}\n\nStructArrayLayout2i4ub8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout2i4ub8', StructArrayLayout2i4ub8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[2]\n *\n */\nclass StructArrayLayout2f8 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1);\n }\n\n public emplace(i: number, v0: number, v1: number) {\n const o4 = i * 2;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n return i;\n }\n}\n\nStructArrayLayout2f8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout2f8', StructArrayLayout2f8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[10]\n *\n */\nclass StructArrayLayout10ui20 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number) {\n const o2 = i * 10;\n this.uint16[o2 + 0] = v0;\n this.uint16[o2 + 1] = v1;\n this.uint16[o2 + 2] = v2;\n this.uint16[o2 + 3] = v3;\n this.uint16[o2 + 4] = v4;\n this.uint16[o2 + 5] = v5;\n this.uint16[o2 + 6] = v6;\n this.uint16[o2 + 7] = v7;\n this.uint16[o2 + 8] = v8;\n this.uint16[o2 + 9] = v9;\n return i;\n }\n}\n\nStructArrayLayout10ui20.prototype.bytesPerElement = 20;\nregister('StructArrayLayout10ui20', StructArrayLayout10ui20);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[4]\n * [8]: Uint16[4]\n * [16]: Int16[4]\n *\n */\nclass StructArrayLayout4i4ui4i24 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number) {\n const o2 = i * 12;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.uint16[o2 + 4] = v4;\n this.uint16[o2 + 5] = v5;\n this.uint16[o2 + 6] = v6;\n this.uint16[o2 + 7] = v7;\n this.int16[o2 + 8] = v8;\n this.int16[o2 + 9] = v9;\n this.int16[o2 + 10] = v10;\n this.int16[o2 + 11] = v11;\n return i;\n }\n}\n\nStructArrayLayout4i4ui4i24.prototype.bytesPerElement = 24;\nregister('StructArrayLayout4i4ui4i24', StructArrayLayout4i4ui4i24);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[3]\n *\n */\nclass StructArrayLayout3f12 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o4 = i * 3;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout3f12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout3f12', StructArrayLayout3f12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint32[1]\n *\n */\nclass StructArrayLayout1ul4 extends StructArray {\n uint8: Uint8Array;\n uint32: Uint32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0);\n }\n\n public emplace(i: number, v0: number) {\n const o4 = i * 1;\n this.uint32[o4 + 0] = v0;\n return i;\n }\n}\n\nStructArrayLayout1ul4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout1ul4', StructArrayLayout1ul4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[6]\n * [12]: Uint32[1]\n * [16]: Uint16[2]\n *\n */\nclass StructArrayLayout6i1ul2ui20 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint32: Uint32Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number) {\n const o2 = i * 10;\n const o4 = i * 5;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n this.uint32[o4 + 3] = v6;\n this.uint16[o2 + 8] = v7;\n this.uint16[o2 + 9] = v8;\n return i;\n }\n}\n\nStructArrayLayout6i1ul2ui20.prototype.bytesPerElement = 20;\nregister('StructArrayLayout6i1ul2ui20', StructArrayLayout6i1ul2ui20);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Int16[2]\n * [8]: Int16[2]\n *\n */\nclass StructArrayLayout2i2i2i12 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) {\n const o2 = i * 6;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n return i;\n }\n}\n\nStructArrayLayout2i2i2i12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout2i2i2i12', StructArrayLayout2i2i2i12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[2]\n * [8]: Float32[1]\n * [12]: Int16[2]\n *\n */\nclass StructArrayLayout2f1f2i16 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n int16: Int16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number) {\n const o4 = i * 4;\n const o2 = i * 8;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n this.int16[o2 + 6] = v3;\n this.int16[o2 + 7] = v4;\n return i;\n }\n}\n\nStructArrayLayout2f1f2i16.prototype.bytesPerElement = 16;\nregister('StructArrayLayout2f1f2i16', StructArrayLayout2f1f2i16);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint8[2]\n * [4]: Float32[2]\n *\n */\nclass StructArrayLayout2ub2f12 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number) {\n const o1 = i * 12;\n const o4 = i * 3;\n this.uint8[o1 + 0] = v0;\n this.uint8[o1 + 1] = v1;\n this.float32[o4 + 1] = v2;\n this.float32[o4 + 2] = v3;\n return i;\n }\n}\n\nStructArrayLayout2ub2f12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout2ub2f12', StructArrayLayout2ub2f12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[3]\n *\n */\nclass StructArrayLayout3ui6 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o2 = i * 3;\n this.uint16[o2 + 0] = v0;\n this.uint16[o2 + 1] = v1;\n this.uint16[o2 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout3ui6.prototype.bytesPerElement = 6;\nregister('StructArrayLayout3ui6', StructArrayLayout3ui6);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[2]\n * [4]: Uint16[2]\n * [8]: Uint32[3]\n * [20]: Uint16[3]\n * [28]: Float32[2]\n * [36]: Uint8[3]\n * [40]: Uint32[1]\n * [44]: Int16[1]\n *\n */\nclass StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint16: Uint16Array;\n uint32: Uint32Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number) {\n const o2 = i * 24;\n const o4 = i * 12;\n const o1 = i * 48;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.uint16[o2 + 2] = v2;\n this.uint16[o2 + 3] = v3;\n this.uint32[o4 + 2] = v4;\n this.uint32[o4 + 3] = v5;\n this.uint32[o4 + 4] = v6;\n this.uint16[o2 + 10] = v7;\n this.uint16[o2 + 11] = v8;\n this.uint16[o2 + 12] = v9;\n this.float32[o4 + 7] = v10;\n this.float32[o4 + 8] = v11;\n this.uint8[o1 + 36] = v12;\n this.uint8[o1 + 37] = v13;\n this.uint8[o1 + 38] = v14;\n this.uint32[o4 + 10] = v15;\n this.int16[o2 + 22] = v16;\n return i;\n }\n}\n\nStructArrayLayout2i2ui3ul3ui2f3ub1ul1i48.prototype.bytesPerElement = 48;\nregister('StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48', StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Int16[8]\n * [16]: Uint16[15]\n * [48]: Uint32[1]\n * [52]: Float32[2]\n * [60]: Uint16[2]\n *\n */\nclass StructArrayLayout8i15ui1ul2f2ui64 extends StructArray {\n uint8: Uint8Array;\n int16: Int16Array;\n uint16: Uint16Array;\n uint32: Uint32Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.int16 = new Int16Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number, v17: number, v18: number, v19: number, v20: number, v21: number, v22: number, v23: number, v24: number, v25: number, v26: number, v27: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number, v17: number, v18: number, v19: number, v20: number, v21: number, v22: number, v23: number, v24: number, v25: number, v26: number, v27: number) {\n const o2 = i * 32;\n const o4 = i * 16;\n this.int16[o2 + 0] = v0;\n this.int16[o2 + 1] = v1;\n this.int16[o2 + 2] = v2;\n this.int16[o2 + 3] = v3;\n this.int16[o2 + 4] = v4;\n this.int16[o2 + 5] = v5;\n this.int16[o2 + 6] = v6;\n this.int16[o2 + 7] = v7;\n this.uint16[o2 + 8] = v8;\n this.uint16[o2 + 9] = v9;\n this.uint16[o2 + 10] = v10;\n this.uint16[o2 + 11] = v11;\n this.uint16[o2 + 12] = v12;\n this.uint16[o2 + 13] = v13;\n this.uint16[o2 + 14] = v14;\n this.uint16[o2 + 15] = v15;\n this.uint16[o2 + 16] = v16;\n this.uint16[o2 + 17] = v17;\n this.uint16[o2 + 18] = v18;\n this.uint16[o2 + 19] = v19;\n this.uint16[o2 + 20] = v20;\n this.uint16[o2 + 21] = v21;\n this.uint16[o2 + 22] = v22;\n this.uint32[o4 + 12] = v23;\n this.float32[o4 + 13] = v24;\n this.float32[o4 + 14] = v25;\n this.uint16[o2 + 30] = v26;\n this.uint16[o2 + 31] = v27;\n return i;\n }\n}\n\nStructArrayLayout8i15ui1ul2f2ui64.prototype.bytesPerElement = 64;\nregister('StructArrayLayout8i15ui1ul2f2ui64', StructArrayLayout8i15ui1ul2f2ui64);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[1]\n *\n */\nclass StructArrayLayout1f4 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0);\n }\n\n public emplace(i: number, v0: number) {\n const o4 = i * 1;\n this.float32[o4 + 0] = v0;\n return i;\n }\n}\n\nStructArrayLayout1f4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout1f4', StructArrayLayout1f4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[1]\n * [4]: Float32[2]\n *\n */\nclass StructArrayLayout1ui2f12 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o2 = i * 6;\n const o4 = i * 3;\n this.uint16[o2 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n return i;\n }\n}\n\nStructArrayLayout1ui2f12.prototype.bytesPerElement = 12;\nregister('StructArrayLayout1ui2f12', StructArrayLayout1ui2f12);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint32[1]\n * [4]: Uint16[2]\n *\n */\nclass StructArrayLayout1ul2ui8 extends StructArray {\n uint8: Uint8Array;\n uint32: Uint32Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint32 = new Uint32Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number) {\n const o4 = i * 2;\n const o2 = i * 4;\n this.uint32[o4 + 0] = v0;\n this.uint16[o2 + 2] = v1;\n this.uint16[o2 + 3] = v2;\n return i;\n }\n}\n\nStructArrayLayout1ul2ui8.prototype.bytesPerElement = 8;\nregister('StructArrayLayout1ul2ui8', StructArrayLayout1ul2ui8);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[2]\n *\n */\nclass StructArrayLayout2ui4 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1);\n }\n\n public emplace(i: number, v0: number, v1: number) {\n const o2 = i * 2;\n this.uint16[o2 + 0] = v0;\n this.uint16[o2 + 1] = v1;\n return i;\n }\n}\n\nStructArrayLayout2ui4.prototype.bytesPerElement = 4;\nregister('StructArrayLayout2ui4', StructArrayLayout2ui4);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Uint16[1]\n *\n */\nclass StructArrayLayout1ui2 extends StructArray {\n uint8: Uint8Array;\n uint16: Uint16Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.uint16 = new Uint16Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0);\n }\n\n public emplace(i: number, v0: number) {\n const o2 = i * 1;\n this.uint16[o2 + 0] = v0;\n return i;\n }\n}\n\nStructArrayLayout1ui2.prototype.bytesPerElement = 2;\nregister('StructArrayLayout1ui2', StructArrayLayout1ui2);\n\n/**\n * @internal\n * Implementation of the StructArray layout:\n * [0]: Float32[4]\n *\n */\nclass StructArrayLayout4f16 extends StructArray {\n uint8: Uint8Array;\n float32: Float32Array;\n\n _refreshViews() {\n this.uint8 = new Uint8Array(this.arrayBuffer);\n this.float32 = new Float32Array(this.arrayBuffer);\n }\n\n public emplaceBack(v0: number, v1: number, v2: number, v3: number) {\n const i = this.length;\n this.resize(i + 1);\n return this.emplace(i, v0, v1, v2, v3);\n }\n\n public emplace(i: number, v0: number, v1: number, v2: number, v3: number) {\n const o4 = i * 4;\n this.float32[o4 + 0] = v0;\n this.float32[o4 + 1] = v1;\n this.float32[o4 + 2] = v2;\n this.float32[o4 + 3] = v3;\n return i;\n }\n}\n\nStructArrayLayout4f16.prototype.bytesPerElement = 16;\nregister('StructArrayLayout4f16', StructArrayLayout4f16);\n\n/** @internal */\nclass CollisionBoxStruct extends Struct {\n _structArray: CollisionBoxArray;\n get anchorPointX() { return this._structArray.int16[this._pos2 + 0]; }\n get anchorPointY() { return this._structArray.int16[this._pos2 + 1]; }\n get x1() { return this._structArray.int16[this._pos2 + 2]; }\n get y1() { return this._structArray.int16[this._pos2 + 3]; }\n get x2() { return this._structArray.int16[this._pos2 + 4]; }\n get y2() { return this._structArray.int16[this._pos2 + 5]; }\n get featureIndex() { return this._structArray.uint32[this._pos4 + 3]; }\n get sourceLayerIndex() { return this._structArray.uint16[this._pos2 + 8]; }\n get bucketIndex() { return this._structArray.uint16[this._pos2 + 9]; }\n get anchorPoint() { return new Point(this.anchorPointX, this.anchorPointY); }\n}\n\nCollisionBoxStruct.prototype.size = 20;\n\nexport type CollisionBox = CollisionBoxStruct;\n\n/** @internal */\nexport class CollisionBoxArray extends StructArrayLayout6i1ul2ui20 {\n /**\n * Return the CollisionBoxStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): CollisionBoxStruct {\n return new CollisionBoxStruct(this, index);\n }\n}\n\nregister('CollisionBoxArray', CollisionBoxArray);\n\n/** @internal */\nclass PlacedSymbolStruct extends Struct {\n _structArray: PlacedSymbolArray;\n get anchorX() { return this._structArray.int16[this._pos2 + 0]; }\n get anchorY() { return this._structArray.int16[this._pos2 + 1]; }\n get glyphStartIndex() { return this._structArray.uint16[this._pos2 + 2]; }\n get numGlyphs() { return this._structArray.uint16[this._pos2 + 3]; }\n get vertexStartIndex() { return this._structArray.uint32[this._pos4 + 2]; }\n get lineStartIndex() { return this._structArray.uint32[this._pos4 + 3]; }\n get lineLength() { return this._structArray.uint32[this._pos4 + 4]; }\n get segment() { return this._structArray.uint16[this._pos2 + 10]; }\n get lowerSize() { return this._structArray.uint16[this._pos2 + 11]; }\n get upperSize() { return this._structArray.uint16[this._pos2 + 12]; }\n get lineOffsetX() { return this._structArray.float32[this._pos4 + 7]; }\n get lineOffsetY() { return this._structArray.float32[this._pos4 + 8]; }\n get writingMode() { return this._structArray.uint8[this._pos1 + 36]; }\n get placedOrientation() { return this._structArray.uint8[this._pos1 + 37]; }\n set placedOrientation(x: number) { this._structArray.uint8[this._pos1 + 37] = x; }\n get hidden() { return this._structArray.uint8[this._pos1 + 38]; }\n set hidden(x: number) { this._structArray.uint8[this._pos1 + 38] = x; }\n get crossTileID() { return this._structArray.uint32[this._pos4 + 10]; }\n set crossTileID(x: number) { this._structArray.uint32[this._pos4 + 10] = x; }\n get associatedIconIndex() { return this._structArray.int16[this._pos2 + 22]; }\n}\n\nPlacedSymbolStruct.prototype.size = 48;\n\nexport type PlacedSymbol = PlacedSymbolStruct;\n\n/** @internal */\nexport class PlacedSymbolArray extends StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 {\n /**\n * Return the PlacedSymbolStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): PlacedSymbolStruct {\n return new PlacedSymbolStruct(this, index);\n }\n}\n\nregister('PlacedSymbolArray', PlacedSymbolArray);\n\n/** @internal */\nclass SymbolInstanceStruct extends Struct {\n _structArray: SymbolInstanceArray;\n get anchorX() { return this._structArray.int16[this._pos2 + 0]; }\n get anchorY() { return this._structArray.int16[this._pos2 + 1]; }\n get rightJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 2]; }\n get centerJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 3]; }\n get leftJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 4]; }\n get verticalPlacedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 5]; }\n get placedIconSymbolIndex() { return this._structArray.int16[this._pos2 + 6]; }\n get verticalPlacedIconSymbolIndex() { return this._structArray.int16[this._pos2 + 7]; }\n get key() { return this._structArray.uint16[this._pos2 + 8]; }\n get textBoxStartIndex() { return this._structArray.uint16[this._pos2 + 9]; }\n get textBoxEndIndex() { return this._structArray.uint16[this._pos2 + 10]; }\n get verticalTextBoxStartIndex() { return this._structArray.uint16[this._pos2 + 11]; }\n get verticalTextBoxEndIndex() { return this._structArray.uint16[this._pos2 + 12]; }\n get iconBoxStartIndex() { return this._structArray.uint16[this._pos2 + 13]; }\n get iconBoxEndIndex() { return this._structArray.uint16[this._pos2 + 14]; }\n get verticalIconBoxStartIndex() { return this._structArray.uint16[this._pos2 + 15]; }\n get verticalIconBoxEndIndex() { return this._structArray.uint16[this._pos2 + 16]; }\n get featureIndex() { return this._structArray.uint16[this._pos2 + 17]; }\n get numHorizontalGlyphVertices() { return this._structArray.uint16[this._pos2 + 18]; }\n get numVerticalGlyphVertices() { return this._structArray.uint16[this._pos2 + 19]; }\n get numIconVertices() { return this._structArray.uint16[this._pos2 + 20]; }\n get numVerticalIconVertices() { return this._structArray.uint16[this._pos2 + 21]; }\n get useRuntimeCollisionCircles() { return this._structArray.uint16[this._pos2 + 22]; }\n get crossTileID() { return this._structArray.uint32[this._pos4 + 12]; }\n set crossTileID(x: number) { this._structArray.uint32[this._pos4 + 12] = x; }\n get textBoxScale() { return this._structArray.float32[this._pos4 + 13]; }\n get collisionCircleDiameter() { return this._structArray.float32[this._pos4 + 14]; }\n get textAnchorOffsetStartIndex() { return this._structArray.uint16[this._pos2 + 30]; }\n get textAnchorOffsetEndIndex() { return this._structArray.uint16[this._pos2 + 31]; }\n}\n\nSymbolInstanceStruct.prototype.size = 64;\n\nexport type SymbolInstance = SymbolInstanceStruct;\n\n/** @internal */\nexport class SymbolInstanceArray extends StructArrayLayout8i15ui1ul2f2ui64 {\n /**\n * Return the SymbolInstanceStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): SymbolInstanceStruct {\n return new SymbolInstanceStruct(this, index);\n }\n}\n\nregister('SymbolInstanceArray', SymbolInstanceArray);\n\n/** @internal */\nexport class GlyphOffsetArray extends StructArrayLayout1f4 {\n getoffsetX(index: number) { return this.float32[index * 1 + 0]; }\n}\n\nregister('GlyphOffsetArray', GlyphOffsetArray);\n\n/** @internal */\nexport class SymbolLineVertexArray extends StructArrayLayout3i6 {\n getx(index: number) { return this.int16[index * 3 + 0]; }\n gety(index: number) { return this.int16[index * 3 + 1]; }\n gettileUnitDistanceFromAnchor(index: number) { return this.int16[index * 3 + 2]; }\n}\n\nregister('SymbolLineVertexArray', SymbolLineVertexArray);\n\n/** @internal */\nclass TextAnchorOffsetStruct extends Struct {\n _structArray: TextAnchorOffsetArray;\n get textAnchor() { return this._structArray.uint16[this._pos2 + 0]; }\n get textOffset0() { return this._structArray.float32[this._pos4 + 1]; }\n get textOffset1() { return this._structArray.float32[this._pos4 + 2]; }\n}\n\nTextAnchorOffsetStruct.prototype.size = 12;\n\nexport type TextAnchorOffset = TextAnchorOffsetStruct;\n\n/** @internal */\nexport class TextAnchorOffsetArray extends StructArrayLayout1ui2f12 {\n /**\n * Return the TextAnchorOffsetStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): TextAnchorOffsetStruct {\n return new TextAnchorOffsetStruct(this, index);\n }\n}\n\nregister('TextAnchorOffsetArray', TextAnchorOffsetArray);\n\n/** @internal */\nclass FeatureIndexStruct extends Struct {\n _structArray: FeatureIndexArray;\n get featureIndex() { return this._structArray.uint32[this._pos4 + 0]; }\n get sourceLayerIndex() { return this._structArray.uint16[this._pos2 + 2]; }\n get bucketIndex() { return this._structArray.uint16[this._pos2 + 3]; }\n}\n\nFeatureIndexStruct.prototype.size = 8;\n\nexport type FeatureIndex = FeatureIndexStruct;\n\n/** @internal */\nexport class FeatureIndexArray extends StructArrayLayout1ul2ui8 {\n /**\n * Return the FeatureIndexStruct at the given location in the array.\n * @param index The index of the element.\n */\n get(index: number): FeatureIndexStruct {\n return new FeatureIndexStruct(this, index);\n }\n}\n\nregister('FeatureIndexArray', FeatureIndexArray);\n\nexport class PosArray extends StructArrayLayout2i4 {}\nexport class Pos3dArray extends StructArrayLayout3i6 {}\nexport class RasterBoundsArray extends StructArrayLayout4i8 {}\nexport class CircleLayoutArray extends StructArrayLayout2i4 {}\nexport class FillLayoutArray extends StructArrayLayout2i4 {}\nexport class FillExtrusionLayoutArray extends StructArrayLayout2i4i12 {}\nexport class HeatmapLayoutArray extends StructArrayLayout2i4 {}\nexport class LineLayoutArray extends StructArrayLayout2i4ub8 {}\nexport class LineExtLayoutArray extends StructArrayLayout2f8 {}\nexport class PatternLayoutArray extends StructArrayLayout10ui20 {}\nexport class SymbolLayoutArray extends StructArrayLayout4i4ui4i24 {}\nexport class SymbolDynamicLayoutArray extends StructArrayLayout3f12 {}\nexport class SymbolOpacityArray extends StructArrayLayout1ul4 {}\nexport class CollisionBoxLayoutArray extends StructArrayLayout2i2i2i12 {}\nexport class CollisionCircleLayoutArray extends StructArrayLayout2f1f2i16 {}\nexport class CollisionVertexArray extends StructArrayLayout2ub2f12 {}\nexport class QuadTriangleArray extends StructArrayLayout3ui6 {}\nexport class TriangleIndexArray extends StructArrayLayout3ui6 {}\nexport class LineIndexArray extends StructArrayLayout2ui4 {}\nexport class LineStripIndexArray extends StructArrayLayout1ui2 {}\nexport {\n StructArrayLayout2i4,\n StructArrayLayout3i6,\n StructArrayLayout4i8,\n StructArrayLayout2i4i12,\n StructArrayLayout2i4ub8,\n StructArrayLayout2f8,\n StructArrayLayout10ui20,\n StructArrayLayout4i4ui4i24,\n StructArrayLayout3f12,\n StructArrayLayout1ul4,\n StructArrayLayout6i1ul2ui20,\n StructArrayLayout2i2i2i12,\n StructArrayLayout2f1f2i16,\n StructArrayLayout2ub2f12,\n StructArrayLayout3ui6,\n StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48,\n StructArrayLayout8i15ui1ul2f2ui64,\n StructArrayLayout1f4,\n StructArrayLayout1ui2f12,\n StructArrayLayout1ul2ui8,\n StructArrayLayout2ui4,\n StructArrayLayout1ui2,\n StructArrayLayout4f16\n};\n","import {createLayout} from '../../util/struct_array';\n\nconst layout = createLayout([\n {name: 'a_pos', components: 2, type: 'Int16'}\n], 4);\n\nexport default layout;\nexport const {members, size, alignment} = layout;\n","import {warnOnce} from '../util/util';\n\nimport {register} from '../util/web_worker_transfer';\n\nimport type {VertexArrayObject} from '../render/vertex_array_object';\nimport type {StructArray} from '../util/struct_array';\n\n/**\n * @internal\n * A single segment of a vector\n */\nexport type Segment = {\n sortKey?: number;\n vertexOffset: number;\n primitiveOffset: number;\n vertexLength: number;\n primitiveLength: number;\n vaos: {[_: string]: VertexArrayObject};\n};\n\n/**\n * @internal\n * Used for calculations on vector segments\n */\nexport class SegmentVector {\n static MAX_VERTEX_ARRAY_LENGTH: number;\n segments: Array;\n\n constructor(segments: Array = []) {\n this.segments = segments;\n }\n\n prepareSegment(\n numVertices: number,\n layoutVertexArray: StructArray,\n indexArray: StructArray,\n sortKey?: number\n ): Segment {\n let segment: Segment = this.segments[this.segments.length - 1];\n if (numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) warnOnce(`Max vertices per segment is ${SegmentVector.MAX_VERTEX_ARRAY_LENGTH}: bucket requested ${numVertices}`);\n if (!segment || segment.vertexLength + numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH || segment.sortKey !== sortKey) {\n segment = ({\n vertexOffset: layoutVertexArray.length,\n primitiveOffset: indexArray.length,\n vertexLength: 0,\n primitiveLength: 0\n } as any);\n if (sortKey !== undefined) segment.sortKey = sortKey;\n this.segments.push(segment);\n }\n return segment;\n }\n\n get() {\n return this.segments;\n }\n\n destroy() {\n for (const segment of this.segments) {\n for (const k in segment.vaos) {\n segment.vaos[k].destroy();\n }\n }\n }\n\n static simpleSegment(\n vertexOffset: number,\n primitiveOffset: number,\n vertexLength: number,\n primitiveLength: number\n ): SegmentVector {\n return new SegmentVector([{\n vertexOffset,\n primitiveOffset,\n vertexLength,\n primitiveLength,\n vaos: {},\n sortKey: 0\n }]);\n }\n}\n\n/**\n * The maximum size of a vertex array. This limit is imposed by WebGL's 16 bit\n * addressing of vertex buffers.\n */\nSegmentVector.MAX_VERTEX_ARRAY_LENGTH = Math.pow(2, 16) - 1;\n\nregister('SegmentVector', SegmentVector);\n","import {clamp} from '../util/util';\n\n/**\n * Packs two numbers, interpreted as 8-bit unsigned integers, into a single\n * float. Unpack them in the shader using the `unpack_float()` function,\n * defined in _prelude.vertex.glsl\n */\nexport function packUint8ToFloat(a: number, b: number) {\n // coerce a and b to 8-bit ints\n a = clamp(Math.floor(a), 0, 255);\n b = clamp(Math.floor(b), 0, 255);\n return 256 * a + b;\n}\n","import {createLayout} from '../../util/struct_array';\n\nexport const patternAttributes = createLayout([\n // [tl.x, tl.y, br.x, br.y]\n {name: 'a_pattern_from', components: 4, type: 'Uint16'},\n {name: 'a_pattern_to', components: 4, type: 'Uint16'},\n {name: 'a_pixel_ratio_from', components: 1, type: 'Uint16'},\n {name: 'a_pixel_ratio_to', components: 1, type: 'Uint16'},\n]);\n","/**\n * JS Implementation of MurmurHash3 (r136) (as of May 20, 2011)\n * \n * @author Gary Court\n * @see http://github.com/garycourt/murmurhash-js\n * @author Austin Appleby\n * @see http://sites.google.com/site/murmurhash/\n * \n * @param {string} key ASCII only\n * @param {number} seed Positive integer only\n * @return {number} 32-bit positive integer hash \n */\n\nfunction murmurhash3_32_gc(key, seed) {\n\tvar remainder, bytes, h1, h1b, c1, c1b, c2, c2b, k1, i;\n\t\n\tremainder = key.length & 3; // key.length % 4\n\tbytes = key.length - remainder;\n\th1 = seed;\n\tc1 = 0xcc9e2d51;\n\tc2 = 0x1b873593;\n\ti = 0;\n\t\n\twhile (i < bytes) {\n\t \tk1 = \n\t \t ((key.charCodeAt(i) & 0xff)) |\n\t \t ((key.charCodeAt(++i) & 0xff) << 8) |\n\t \t ((key.charCodeAt(++i) & 0xff) << 16) |\n\t \t ((key.charCodeAt(++i) & 0xff) << 24);\n\t\t++i;\n\t\t\n\t\tk1 = ((((k1 & 0xffff) * c1) + ((((k1 >>> 16) * c1) & 0xffff) << 16))) & 0xffffffff;\n\t\tk1 = (k1 << 15) | (k1 >>> 17);\n\t\tk1 = ((((k1 & 0xffff) * c2) + ((((k1 >>> 16) * c2) & 0xffff) << 16))) & 0xffffffff;\n\n\t\th1 ^= k1;\n h1 = (h1 << 13) | (h1 >>> 19);\n\t\th1b = ((((h1 & 0xffff) * 5) + ((((h1 >>> 16) * 5) & 0xffff) << 16))) & 0xffffffff;\n\t\th1 = (((h1b & 0xffff) + 0x6b64) + ((((h1b >>> 16) + 0xe654) & 0xffff) << 16));\n\t}\n\t\n\tk1 = 0;\n\t\n\tswitch (remainder) {\n\t\tcase 3: k1 ^= (key.charCodeAt(i + 2) & 0xff) << 16;\n\t\tcase 2: k1 ^= (key.charCodeAt(i + 1) & 0xff) << 8;\n\t\tcase 1: k1 ^= (key.charCodeAt(i) & 0xff);\n\t\t\n\t\tk1 = (((k1 & 0xffff) * c1) + ((((k1 >>> 16) * c1) & 0xffff) << 16)) & 0xffffffff;\n\t\tk1 = (k1 << 15) | (k1 >>> 17);\n\t\tk1 = (((k1 & 0xffff) * c2) + ((((k1 >>> 16) * c2) & 0xffff) << 16)) & 0xffffffff;\n\t\th1 ^= k1;\n\t}\n\t\n\th1 ^= key.length;\n\n\th1 ^= h1 >>> 16;\n\th1 = (((h1 & 0xffff) * 0x85ebca6b) + ((((h1 >>> 16) * 0x85ebca6b) & 0xffff) << 16)) & 0xffffffff;\n\th1 ^= h1 >>> 13;\n\th1 = ((((h1 & 0xffff) * 0xc2b2ae35) + ((((h1 >>> 16) * 0xc2b2ae35) & 0xffff) << 16))) & 0xffffffff;\n\th1 ^= h1 >>> 16;\n\n\treturn h1 >>> 0;\n}\n\nif(typeof module !== \"undefined\") {\n module.exports = murmurhash3_32_gc\n}","/**\n * JS Implementation of MurmurHash2\n * \n * @author Gary Court\n * @see http://github.com/garycourt/murmurhash-js\n * @author Austin Appleby\n * @see http://sites.google.com/site/murmurhash/\n * \n * @param {string} str ASCII only\n * @param {number} seed Positive integer only\n * @return {number} 32-bit positive integer hash\n */\n\nfunction murmurhash2_32_gc(str, seed) {\n var\n l = str.length,\n h = seed ^ l,\n i = 0,\n k;\n \n while (l >= 4) {\n \tk = \n \t ((str.charCodeAt(i) & 0xff)) |\n \t ((str.charCodeAt(++i) & 0xff) << 8) |\n \t ((str.charCodeAt(++i) & 0xff) << 16) |\n \t ((str.charCodeAt(++i) & 0xff) << 24);\n \n k = (((k & 0xffff) * 0x5bd1e995) + ((((k >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n k ^= k >>> 24;\n k = (((k & 0xffff) * 0x5bd1e995) + ((((k >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n\n\th = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16)) ^ k;\n\n l -= 4;\n ++i;\n }\n \n switch (l) {\n case 3: h ^= (str.charCodeAt(i + 2) & 0xff) << 16;\n case 2: h ^= (str.charCodeAt(i + 1) & 0xff) << 8;\n case 1: h ^= (str.charCodeAt(i) & 0xff);\n h = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n }\n\n h ^= h >>> 13;\n h = (((h & 0xffff) * 0x5bd1e995) + ((((h >>> 16) * 0x5bd1e995) & 0xffff) << 16));\n h ^= h >>> 15;\n\n return h >>> 0;\n}\n\nif(typeof module !== undefined) {\n module.exports = murmurhash2_32_gc\n}\n","var murmur3 = require(\"./murmurhash3_gc.js\")\nvar murmur2 = require(\"./murmurhash2_gc.js\")\n\nmodule.exports = murmur3\nmodule.exports.murmur3 = murmur3\nmodule.exports.murmur2 = murmur2\n","import murmur3 from 'murmurhash-js';\nimport {register} from '../util/web_worker_transfer';\n\ntype SerializedFeaturePositionMap = {\n ids: Float64Array;\n positions: Uint32Array;\n};\n\ntype FeaturePosition = {\n index: number;\n start: number;\n end: number;\n};\n\n// A transferable data structure that maps feature ids to their indices and buffer offsets\nexport class FeaturePositionMap {\n ids: Array;\n positions: Array;\n indexed: boolean;\n\n constructor() {\n this.ids = [];\n this.positions = [];\n this.indexed = false;\n }\n\n add(id: unknown, index: number, start: number, end: number) {\n this.ids.push(getNumericId(id));\n this.positions.push(index, start, end);\n }\n\n getPositions(id: unknown): Array {\n if (!this.indexed) throw new Error('Trying to get index, but feature positions are not indexed');\n\n const intId = getNumericId(id);\n\n // binary search for the first occurrence of id in this.ids;\n // relies on ids/positions being sorted by id, which happens in serialization\n let i = 0;\n let j = this.ids.length - 1;\n while (i < j) {\n const m = (i + j) >> 1;\n if (this.ids[m] >= intId) {\n j = m;\n } else {\n i = m + 1;\n }\n }\n const positions = [];\n while (this.ids[i] === intId) {\n const index = this.positions[3 * i];\n const start = this.positions[3 * i + 1];\n const end = this.positions[3 * i + 2];\n positions.push({index, start, end});\n i++;\n }\n return positions;\n }\n\n static serialize(map: FeaturePositionMap, transferables: Array): SerializedFeaturePositionMap {\n const ids = new Float64Array(map.ids);\n const positions = new Uint32Array(map.positions);\n\n sort(ids, positions, 0, ids.length - 1);\n\n if (transferables) {\n transferables.push(ids.buffer, positions.buffer);\n }\n\n return {ids, positions};\n }\n\n static deserialize(obj: SerializedFeaturePositionMap): FeaturePositionMap {\n const map = new FeaturePositionMap();\n // after transferring, we only use these arrays statically (no pushes),\n // so TypedArray vs Array distinction that flow points out doesn't matter\n map.ids = (obj.ids as any);\n map.positions = (obj.positions as any);\n map.indexed = true;\n return map;\n }\n}\n\nfunction getNumericId(value: unknown) {\n const numValue = +value;\n if (!isNaN(numValue) && numValue <= Number.MAX_SAFE_INTEGER) {\n return numValue;\n }\n return murmur3(String(value));\n}\n\n// custom quicksort that sorts ids, indices and offsets together (by ids)\n// uses Hoare partitioning & manual tail call optimization to avoid worst case scenarios\nfunction sort(ids, positions, left, right) {\n while (left < right) {\n const pivot = ids[(left + right) >> 1];\n let i = left - 1;\n let j = right + 1;\n\n while (true) {\n do i++; while (ids[i] < pivot);\n do j--; while (ids[j] > pivot);\n if (i >= j) break;\n swap(ids, i, j);\n swap(positions, 3 * i, 3 * j);\n swap(positions, 3 * i + 1, 3 * j + 1);\n swap(positions, 3 * i + 2, 3 * j + 2);\n }\n\n if (j - left < right - j) {\n sort(ids, positions, left, j);\n left = j + 1;\n } else {\n sort(ids, positions, j + 1, right);\n right = j;\n }\n }\n}\n\nfunction swap(arr, i, j) {\n const tmp = arr[i];\n arr[i] = arr[j];\n arr[j] = tmp;\n}\n\nregister('FeaturePositionMap', FeaturePositionMap);\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {Context} from '../gl/context';\nimport {mat4, vec2, vec3, vec4} from 'gl-matrix';\n\ntype $ObjMap any> = {\n [K in keyof T]: F extends (v: T[K]) => infer R ? R : never;\n};\n\nexport type UniformValues = $ObjMap(u: Uniform) => V>;\nexport type UniformLocations = {[_: string]: WebGLUniformLocation};\n\n/**\n * @internal\n * A base uniform abstract class\n */\nabstract class Uniform {\n gl: WebGLRenderingContext|WebGL2RenderingContext;\n location: WebGLUniformLocation;\n current: T;\n\n constructor(context: Context, location: WebGLUniformLocation) {\n this.gl = context.gl;\n this.location = location;\n }\n\n abstract set(v: T): void;\n}\n\nclass Uniform1i extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = 0;\n }\n\n set(v: number): void {\n if (this.current !== v) {\n this.current = v;\n this.gl.uniform1i(this.location, v);\n }\n }\n}\n\nclass Uniform1f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = 0;\n }\n\n set(v: number): void {\n if (this.current !== v) {\n this.current = v;\n this.gl.uniform1f(this.location, v);\n }\n }\n}\n\nclass Uniform2f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = [0, 0];\n }\n\n set(v: vec2): void {\n if (v[0] !== this.current[0] || v[1] !== this.current[1]) {\n this.current = v;\n this.gl.uniform2f(this.location, v[0], v[1]);\n }\n }\n}\n\nclass Uniform3f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = [0, 0, 0];\n }\n\n set(v: vec3): void {\n if (v[0] !== this.current[0] || v[1] !== this.current[1] || v[2] !== this.current[2]) {\n this.current = v;\n this.gl.uniform3f(this.location, v[0], v[1], v[2]);\n }\n }\n}\n\nclass Uniform4f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = [0, 0, 0, 0];\n }\n\n set(v: vec4): void {\n if (v[0] !== this.current[0] || v[1] !== this.current[1] ||\n v[2] !== this.current[2] || v[3] !== this.current[3]) {\n this.current = v;\n this.gl.uniform4f(this.location, v[0], v[1], v[2], v[3]);\n }\n }\n}\n\nclass UniformColor extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = Color.transparent;\n }\n\n set(v: Color): void {\n if (v.r !== this.current.r || v.g !== this.current.g ||\n v.b !== this.current.b || v.a !== this.current.a) {\n this.current = v;\n this.gl.uniform4f(this.location, v.r, v.g, v.b, v.a);\n }\n }\n}\n\nconst emptyMat4 = new Float32Array(16) as mat4;\nclass UniformMatrix4f extends Uniform {\n constructor(context: Context, location: WebGLUniformLocation) {\n super(context, location);\n this.current = emptyMat4;\n }\n\n set(v: mat4): void {\n // The vast majority of matrix comparisons that will trip this set\n // happen at i=12 or i=0, so we check those first to avoid lots of\n // unnecessary iteration:\n if (v[12] !== this.current[12] || v[0] !== this.current[0]) {\n this.current = v;\n this.gl.uniformMatrix4fv(this.location, false, v);\n return;\n }\n for (let i = 1; i < 16; i++) {\n if (v[i] !== this.current[i]) {\n this.current = v;\n this.gl.uniformMatrix4fv(this.location, false, v);\n break;\n }\n }\n }\n}\n\nexport {\n Uniform,\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f,\n Uniform4f,\n UniformColor,\n UniformMatrix4f\n};\n\n/**\n * @internal\n * A uniform bindings\n */\nexport type UniformBindings = {[_: string]: Uniform};\n","import {packUint8ToFloat} from '../shaders/encode_attribute';\nimport {Color, supportsPropertyExpression} from '@maplibre/maplibre-gl-style-spec';\nimport {register} from '../util/web_worker_transfer';\nimport {PossiblyEvaluatedPropertyValue} from '../style/properties';\nimport {StructArrayLayout1f4, StructArrayLayout2f8, StructArrayLayout4f16, PatternLayoutArray} from './array_types.g';\nimport {clamp} from '../util/util';\nimport {patternAttributes} from './bucket/pattern_attributes';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {FeaturePositionMap} from './feature_position_map';\nimport {Uniform, Uniform1f, UniformColor, Uniform4f} from '../render/uniform_binding';\n\nimport type {UniformLocations} from '../render/uniform_binding';\n\nimport type {CanonicalTileID} from '../source/tile_id';\nimport type {Context} from '../gl/context';\nimport type {TypedStyleLayer} from '../style/style_layer/typed_style_layer';\nimport type {CrossfadeParameters} from '../style/evaluation_parameters';\nimport type {StructArray, StructArrayMember} from '../util/struct_array';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {ImagePosition} from '../render/image_atlas';\nimport type {\n Feature,\n FeatureState,\n GlobalProperties,\n SourceExpression,\n CompositeExpression,\n FormattedSection\n} from '@maplibre/maplibre-gl-style-spec';\nimport type {FeatureStates} from '../source/source_state';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nexport type BinderUniform = {\n name: string;\n property: string;\n binding: Uniform;\n};\n\nfunction packColor(color: Color): [number, number] {\n return [\n packUint8ToFloat(255 * color.r, 255 * color.g),\n packUint8ToFloat(255 * color.b, 255 * color.a)\n ];\n}\n\n/**\n * `Binder` is the interface definition for the strategies for constructing,\n * uploading, and binding paint property data as GLSL attributes. Most style-\n * spec properties have a 1:1 relationship to shader attribute/uniforms, but\n * some require multiple values per feature to be passed to the GPU, and in\n * those cases we bind multiple attributes/uniforms.\n *\n * It has three implementations, one for each of the three strategies we use:\n *\n * * For _constant_ properties -- those whose value is a constant, or the constant\n * result of evaluating a camera expression at a particular camera position -- we\n * don't need a vertex attribute buffer, and instead use a uniform.\n * * For data expressions, we use a vertex buffer with a single attribute value,\n * the evaluated result of the source function for the given feature.\n * * For composite expressions, we use a vertex buffer with two attributes: min and\n * max values covering the range of zooms at which we expect the tile to be\n * displayed. These values are calculated by evaluating the composite expression for\n * the given feature at strategically chosen zoom levels. In addition to this\n * attribute data, we also use a uniform value which the shader uses to interpolate\n * between the min and max value at the final displayed zoom level. The use of a\n * uniform allows us to cheaply update the value on every frame.\n *\n * Note that the shader source varies depending on whether we're using a uniform or\n * attribute. We dynamically compile shaders at runtime to accommodate this.\n */\ninterface AttributeBinder {\n populatePaintArray(\n length: number,\n feature: Feature,\n imagePositions: {[_: string]: ImagePosition},\n canonical?: CanonicalTileID,\n formattedSection?: FormattedSection\n ): void;\n updatePaintArray(\n start: number,\n length: number,\n feature: Feature,\n featureState: FeatureState,\n imagePositions: {[_: string]: ImagePosition}\n ): void;\n upload(a: Context): void;\n destroy(): void;\n}\n\ninterface UniformBinder {\n uniformNames: Array;\n setUniform(\n uniform: Uniform,\n globals: GlobalProperties,\n currentValue: PossiblyEvaluatedPropertyValue,\n uniformName: string\n ): void;\n getBinding(context: Context, location: WebGLUniformLocation, name: string): Partial>;\n}\n\nclass ConstantBinder implements UniformBinder {\n value: unknown;\n type: string;\n uniformNames: Array;\n\n constructor(value: unknown, names: Array, type: string) {\n this.value = value;\n this.uniformNames = names.map(name => `u_${name}`);\n this.type = type;\n }\n\n setUniform(\n uniform: Uniform,\n globals: GlobalProperties,\n currentValue: PossiblyEvaluatedPropertyValue\n ): void {\n uniform.set(currentValue.constantOr(this.value));\n }\n\n getBinding(context: Context, location: WebGLUniformLocation, _: string): Partial> {\n return (this.type === 'color') ?\n new UniformColor(context, location) :\n new Uniform1f(context, location);\n }\n}\n\nclass CrossFadedConstantBinder implements UniformBinder {\n uniformNames: Array;\n patternFrom: Array;\n patternTo: Array;\n pixelRatioFrom: number;\n pixelRatioTo: number;\n\n constructor(value: unknown, names: Array) {\n this.uniformNames = names.map(name => `u_${name}`);\n this.patternFrom = null;\n this.patternTo = null;\n this.pixelRatioFrom = 1.0;\n this.pixelRatioTo = 1.0;\n }\n\n setConstantPatternPositions(posTo: ImagePosition, posFrom: ImagePosition) {\n this.pixelRatioFrom = posFrom.pixelRatio;\n this.pixelRatioTo = posTo.pixelRatio;\n this.patternFrom = posFrom.tlbr;\n this.patternTo = posTo.tlbr;\n }\n\n setUniform(uniform: Uniform, globals: GlobalProperties, currentValue: PossiblyEvaluatedPropertyValue, uniformName: string) {\n const pos =\n uniformName === 'u_pattern_to' ? this.patternTo :\n uniformName === 'u_pattern_from' ? this.patternFrom :\n uniformName === 'u_pixel_ratio_to' ? this.pixelRatioTo :\n uniformName === 'u_pixel_ratio_from' ? this.pixelRatioFrom : null;\n if (pos) uniform.set(pos);\n }\n\n getBinding(context: Context, location: WebGLUniformLocation, name: string): Partial> {\n return name.substr(0, 9) === 'u_pattern' ?\n new Uniform4f(context, location) :\n new Uniform1f(context, location);\n }\n}\n\nclass SourceExpressionBinder implements AttributeBinder {\n expression: SourceExpression;\n type: string;\n maxValue: number;\n\n paintVertexArray: StructArray;\n paintVertexAttributes: Array;\n paintVertexBuffer: VertexBuffer;\n\n constructor(expression: SourceExpression, names: Array, type: string, PaintVertexArray: {\n new (...args: any): StructArray;\n }) {\n this.expression = expression;\n this.type = type;\n this.maxValue = 0;\n this.paintVertexAttributes = names.map((name) => ({\n name: `a_${name}`,\n type: 'Float32',\n components: type === 'color' ? 2 : 1,\n offset: 0\n }));\n this.paintVertexArray = new PaintVertexArray();\n }\n\n populatePaintArray(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) {\n const start = this.paintVertexArray.length;\n const value = this.expression.evaluate(new EvaluationParameters(0), feature, {}, canonical, [], formattedSection);\n this.paintVertexArray.resize(newLength);\n this._setPaintValue(start, newLength, value);\n }\n\n updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState) {\n const value = this.expression.evaluate({zoom: 0}, feature, featureState);\n this._setPaintValue(start, end, value);\n }\n\n _setPaintValue(start, end, value) {\n if (this.type === 'color') {\n const color = packColor(value);\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, color[0], color[1]);\n }\n } else {\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, value);\n }\n this.maxValue = Math.max(this.maxValue, Math.abs(value));\n }\n }\n\n upload(context: Context) {\n if (this.paintVertexArray && this.paintVertexArray.arrayBuffer) {\n if (this.paintVertexBuffer && this.paintVertexBuffer.buffer) {\n this.paintVertexBuffer.updateData(this.paintVertexArray);\n } else {\n this.paintVertexBuffer = context.createVertexBuffer(this.paintVertexArray, this.paintVertexAttributes, this.expression.isStateDependent);\n }\n }\n }\n\n destroy() {\n if (this.paintVertexBuffer) {\n this.paintVertexBuffer.destroy();\n }\n }\n}\n\nclass CompositeExpressionBinder implements AttributeBinder, UniformBinder {\n expression: CompositeExpression;\n uniformNames: Array;\n type: string;\n useIntegerZoom: boolean;\n zoom: number;\n maxValue: number;\n\n paintVertexArray: StructArray;\n paintVertexAttributes: Array;\n paintVertexBuffer: VertexBuffer;\n\n constructor(expression: CompositeExpression, names: Array, type: string, useIntegerZoom: boolean, zoom: number, PaintVertexArray: {\n new (...args: any): StructArray;\n }) {\n this.expression = expression;\n this.uniformNames = names.map(name => `u_${name}_t`);\n this.type = type;\n this.useIntegerZoom = useIntegerZoom;\n this.zoom = zoom;\n this.maxValue = 0;\n this.paintVertexAttributes = names.map((name) => ({\n name: `a_${name}`,\n type: 'Float32',\n components: type === 'color' ? 4 : 2,\n offset: 0\n }));\n this.paintVertexArray = new PaintVertexArray();\n }\n\n populatePaintArray(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) {\n const min = this.expression.evaluate(new EvaluationParameters(this.zoom), feature, {}, canonical, [], formattedSection);\n const max = this.expression.evaluate(new EvaluationParameters(this.zoom + 1), feature, {}, canonical, [], formattedSection);\n const start = this.paintVertexArray.length;\n this.paintVertexArray.resize(newLength);\n this._setPaintValue(start, newLength, min, max);\n }\n\n updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState) {\n const min = this.expression.evaluate({zoom: this.zoom}, feature, featureState);\n const max = this.expression.evaluate({zoom: this.zoom + 1}, feature, featureState);\n this._setPaintValue(start, end, min, max);\n }\n\n _setPaintValue(start, end, min, max) {\n if (this.type === 'color') {\n const minColor = packColor(min);\n const maxColor = packColor(max);\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, minColor[0], minColor[1], maxColor[0], maxColor[1]);\n }\n } else {\n for (let i = start; i < end; i++) {\n this.paintVertexArray.emplace(i, min, max);\n }\n this.maxValue = Math.max(this.maxValue, Math.abs(min), Math.abs(max));\n }\n }\n\n upload(context: Context) {\n if (this.paintVertexArray && this.paintVertexArray.arrayBuffer) {\n if (this.paintVertexBuffer && this.paintVertexBuffer.buffer) {\n this.paintVertexBuffer.updateData(this.paintVertexArray);\n } else {\n this.paintVertexBuffer = context.createVertexBuffer(this.paintVertexArray, this.paintVertexAttributes, this.expression.isStateDependent);\n }\n }\n }\n\n destroy() {\n if (this.paintVertexBuffer) {\n this.paintVertexBuffer.destroy();\n }\n }\n\n setUniform(uniform: Uniform, globals: GlobalProperties): void {\n const currentZoom = this.useIntegerZoom ? Math.floor(globals.zoom) : globals.zoom;\n const factor = clamp(this.expression.interpolationFactor(currentZoom, this.zoom, this.zoom + 1), 0, 1);\n uniform.set(factor);\n }\n\n getBinding(context: Context, location: WebGLUniformLocation, _: string): Uniform1f {\n return new Uniform1f(context, location);\n }\n}\n\nclass CrossFadedCompositeBinder implements AttributeBinder {\n expression: CompositeExpression;\n type: string;\n useIntegerZoom: boolean;\n zoom: number;\n layerId: string;\n\n zoomInPaintVertexArray: StructArray;\n zoomOutPaintVertexArray: StructArray;\n zoomInPaintVertexBuffer: VertexBuffer;\n zoomOutPaintVertexBuffer: VertexBuffer;\n paintVertexAttributes: Array;\n\n constructor(expression: CompositeExpression, type: string, useIntegerZoom: boolean, zoom: number, PaintVertexArray: {\n new (...args: any): StructArray;\n }, layerId: string) {\n this.expression = expression;\n this.type = type;\n this.useIntegerZoom = useIntegerZoom;\n this.zoom = zoom;\n this.layerId = layerId;\n\n this.zoomInPaintVertexArray = new PaintVertexArray();\n this.zoomOutPaintVertexArray = new PaintVertexArray();\n }\n\n populatePaintArray(length: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}) {\n const start = this.zoomInPaintVertexArray.length;\n this.zoomInPaintVertexArray.resize(length);\n this.zoomOutPaintVertexArray.resize(length);\n this._setPaintValues(start, length, feature.patterns && feature.patterns[this.layerId], imagePositions);\n }\n\n updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState, imagePositions: {[_: string]: ImagePosition}) {\n this._setPaintValues(start, end, feature.patterns && feature.patterns[this.layerId], imagePositions);\n }\n\n _setPaintValues(start, end, patterns, positions) {\n if (!positions || !patterns) return;\n\n const {min, mid, max} = patterns;\n const imageMin = positions[min];\n const imageMid = positions[mid];\n const imageMax = positions[max];\n if (!imageMin || !imageMid || !imageMax) return;\n\n // We populate two paint arrays because, for cross-faded properties, we don't know which direction\n // we're cross-fading to at layout time. In order to keep vertex attributes to a minimum and not pass\n // unnecessary vertex data to the shaders, we determine which to upload at draw time.\n for (let i = start; i < end; i++) {\n this.zoomInPaintVertexArray.emplace(i,\n imageMid.tl[0], imageMid.tl[1], imageMid.br[0], imageMid.br[1],\n imageMin.tl[0], imageMin.tl[1], imageMin.br[0], imageMin.br[1],\n imageMid.pixelRatio,\n imageMin.pixelRatio,\n );\n this.zoomOutPaintVertexArray.emplace(i,\n imageMid.tl[0], imageMid.tl[1], imageMid.br[0], imageMid.br[1],\n imageMax.tl[0], imageMax.tl[1], imageMax.br[0], imageMax.br[1],\n imageMid.pixelRatio,\n imageMax.pixelRatio,\n );\n }\n }\n\n upload(context: Context) {\n if (this.zoomInPaintVertexArray && this.zoomInPaintVertexArray.arrayBuffer && this.zoomOutPaintVertexArray && this.zoomOutPaintVertexArray.arrayBuffer) {\n this.zoomInPaintVertexBuffer = context.createVertexBuffer(this.zoomInPaintVertexArray, patternAttributes.members, this.expression.isStateDependent);\n this.zoomOutPaintVertexBuffer = context.createVertexBuffer(this.zoomOutPaintVertexArray, patternAttributes.members, this.expression.isStateDependent);\n }\n }\n\n destroy() {\n if (this.zoomOutPaintVertexBuffer) this.zoomOutPaintVertexBuffer.destroy();\n if (this.zoomInPaintVertexBuffer) this.zoomInPaintVertexBuffer.destroy();\n }\n}\n\n/**\n * @internal\n * ProgramConfiguration contains the logic for binding style layer properties and tile\n * layer feature data into GL program uniforms and vertex attributes.\n *\n * Non-data-driven property values are bound to shader uniforms. Data-driven property\n * values are bound to vertex attributes. In order to support a uniform GLSL syntax over\n * both, [Mapbox GL Shaders](https://github.com/mapbox/mapbox-gl-shaders) defines a `#pragma`\n * abstraction, which ProgramConfiguration is responsible for implementing. At runtime,\n * it examines the attributes of a particular layer, combines this with fixed knowledge\n * about how layers of the particular type are implemented, and determines which uniforms\n * and vertex attributes will be required. It can then substitute the appropriate text\n * into the shader source code, create and link a program, and bind the uniforms and\n * vertex attributes in preparation for drawing.\n *\n * When a vector tile is parsed, this same configuration information is used to\n * populate the attribute buffers needed for data-driven styling using the zoom\n * level and feature property data.\n */\nexport class ProgramConfiguration {\n binders: {[_: string]: AttributeBinder | UniformBinder};\n cacheKey: string;\n\n _buffers: Array;\n\n constructor(layer: TypedStyleLayer, zoom: number, filterProperties: (_: string) => boolean) {\n this.binders = {};\n this._buffers = [];\n\n const keys = [];\n\n for (const property in layer.paint._values) {\n if (!filterProperties(property)) continue;\n const value = (layer.paint as any).get(property);\n if (!(value instanceof PossiblyEvaluatedPropertyValue) || !supportsPropertyExpression(value.property.specification)) {\n continue;\n }\n const names = paintAttributeNames(property, layer.type);\n const expression = value.value;\n const type = value.property.specification.type;\n const useIntegerZoom = (value.property as any).useIntegerZoom;\n const propType = value.property.specification['property-type'];\n const isCrossFaded = propType === 'cross-faded' || propType === 'cross-faded-data-driven';\n\n if (expression.kind === 'constant') {\n this.binders[property] = isCrossFaded ?\n new CrossFadedConstantBinder(expression.value, names) :\n new ConstantBinder(expression.value, names, type);\n keys.push(`/u_${property}`);\n\n } else if (expression.kind === 'source' || isCrossFaded) {\n const StructArrayLayout = layoutType(property, type, 'source');\n this.binders[property] = isCrossFaded ?\n new CrossFadedCompositeBinder(expression as CompositeExpression, type, useIntegerZoom, zoom, StructArrayLayout, layer.id) :\n new SourceExpressionBinder(expression as SourceExpression, names, type, StructArrayLayout);\n keys.push(`/a_${property}`);\n\n } else {\n const StructArrayLayout = layoutType(property, type, 'composite');\n this.binders[property] = new CompositeExpressionBinder(expression, names, type, useIntegerZoom, zoom, StructArrayLayout);\n keys.push(`/z_${property}`);\n }\n }\n\n this.cacheKey = keys.sort().join('');\n }\n\n getMaxValue(property: string): number {\n const binder = this.binders[property];\n return binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder ? binder.maxValue : 0;\n }\n\n populatePaintArrays(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder)\n (binder as AttributeBinder).populatePaintArray(newLength, feature, imagePositions, canonical, formattedSection);\n }\n }\n setConstantPatternPositions(posTo: ImagePosition, posFrom: ImagePosition) {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof CrossFadedConstantBinder)\n binder.setConstantPatternPositions(posTo, posFrom);\n }\n }\n\n updatePaintArrays(\n featureStates: FeatureStates,\n featureMap: FeaturePositionMap,\n vtLayer: VectorTileLayer,\n layer: TypedStyleLayer,\n imagePositions: {[_: string]: ImagePosition}\n ): boolean {\n let dirty: boolean = false;\n for (const id in featureStates) {\n const positions = featureMap.getPositions(id);\n\n for (const pos of positions) {\n const feature = vtLayer.feature(pos.index);\n\n for (const property in this.binders) {\n const binder = this.binders[property];\n if ((binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder ||\n binder instanceof CrossFadedCompositeBinder) && (binder as any).expression.isStateDependent === true) {\n //AHM: Remove after https://github.com/mapbox/mapbox-gl-js/issues/6255\n const value = (layer.paint as any).get(property);\n (binder as any).expression = value.value;\n (binder as AttributeBinder).updatePaintArray(pos.start, pos.end, feature, featureStates[id], imagePositions);\n dirty = true;\n }\n }\n }\n }\n return dirty;\n }\n\n defines(): Array {\n const result = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder) {\n result.push(...binder.uniformNames.map(name => `#define HAS_UNIFORM_${name}`));\n }\n }\n return result;\n }\n\n getBinderAttributes(): Array {\n const result = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder) {\n for (let i = 0; i < binder.paintVertexAttributes.length; i++) {\n result.push(binder.paintVertexAttributes[i].name);\n }\n } else if (binder instanceof CrossFadedCompositeBinder) {\n for (let i = 0; i < patternAttributes.members.length; i++) {\n result.push(patternAttributes.members[i].name);\n }\n }\n }\n return result;\n }\n\n getBinderUniforms(): Array {\n const uniforms = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder || binder instanceof CompositeExpressionBinder) {\n for (const uniformName of binder.uniformNames) {\n uniforms.push(uniformName);\n }\n }\n }\n return uniforms;\n }\n\n getPaintVertexBuffers(): Array {\n return this._buffers;\n }\n\n getUniforms(context: Context, locations: UniformLocations): Array {\n const uniforms = [];\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder || binder instanceof CompositeExpressionBinder) {\n for (const name of binder.uniformNames) {\n if (locations[name]) {\n const binding = binder.getBinding(context, locations[name], name);\n uniforms.push({name, property, binding});\n }\n }\n }\n }\n return uniforms;\n }\n\n setUniforms(\n context: Context,\n binderUniforms: Array,\n properties: any,\n globals: GlobalProperties\n ) {\n // Uniform state bindings are owned by the Program, but we set them\n // from within the ProgramConfiguraton's binder members.\n for (const {name, property, binding} of binderUniforms) {\n (this.binders[property] as any).setUniform(binding, globals, properties.get(property), name);\n }\n }\n\n updatePaintBuffers(crossfade?: CrossfadeParameters) {\n this._buffers = [];\n\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (crossfade && binder instanceof CrossFadedCompositeBinder) {\n const patternVertexBuffer = crossfade.fromScale === 2 ? binder.zoomInPaintVertexBuffer : binder.zoomOutPaintVertexBuffer;\n if (patternVertexBuffer) this._buffers.push(patternVertexBuffer);\n\n } else if ((binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder) && binder.paintVertexBuffer) {\n this._buffers.push(binder.paintVertexBuffer);\n }\n }\n }\n\n upload(context: Context) {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder)\n binder.upload(context);\n }\n this.updatePaintBuffers();\n }\n\n destroy() {\n for (const property in this.binders) {\n const binder = this.binders[property];\n if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder)\n binder.destroy();\n }\n }\n}\n\nexport class ProgramConfigurationSet {\n programConfigurations: {[_: string]: ProgramConfiguration};\n needsUpload: boolean;\n _featureMap: FeaturePositionMap;\n _bufferOffset: number;\n\n constructor(layers: ReadonlyArray, zoom: number, filterProperties: (_: string) => boolean = () => true) {\n this.programConfigurations = {};\n for (const layer of layers) {\n this.programConfigurations[layer.id] = new ProgramConfiguration(layer, zoom, filterProperties);\n }\n this.needsUpload = false;\n this._featureMap = new FeaturePositionMap();\n this._bufferOffset = 0;\n }\n\n populatePaintArrays(length: number, feature: Feature, index: number, imagePositions: {[_: string]: ImagePosition}, canonical: CanonicalTileID, formattedSection?: FormattedSection) {\n for (const key in this.programConfigurations) {\n this.programConfigurations[key].populatePaintArrays(length, feature, imagePositions, canonical, formattedSection);\n }\n\n if (feature.id !== undefined) {\n this._featureMap.add(feature.id, index, this._bufferOffset, length);\n }\n this._bufferOffset = length;\n\n this.needsUpload = true;\n }\n\n updatePaintArrays(featureStates: FeatureStates, vtLayer: VectorTileLayer, layers: ReadonlyArray, imagePositions: {[_: string]: ImagePosition}) {\n for (const layer of layers) {\n this.needsUpload = this.programConfigurations[layer.id].updatePaintArrays(featureStates, this._featureMap, vtLayer, layer, imagePositions) || this.needsUpload;\n }\n }\n\n get(layerId: string) {\n return this.programConfigurations[layerId];\n }\n\n upload(context: Context) {\n if (!this.needsUpload) return;\n for (const layerId in this.programConfigurations) {\n this.programConfigurations[layerId].upload(context);\n }\n this.needsUpload = false;\n }\n\n destroy() {\n for (const layerId in this.programConfigurations) {\n this.programConfigurations[layerId].destroy();\n }\n }\n}\n\nfunction paintAttributeNames(property, type) {\n const attributeNameExceptions = {\n 'text-opacity': ['opacity'],\n 'icon-opacity': ['opacity'],\n 'text-color': ['fill_color'],\n 'icon-color': ['fill_color'],\n 'text-halo-color': ['halo_color'],\n 'icon-halo-color': ['halo_color'],\n 'text-halo-blur': ['halo_blur'],\n 'icon-halo-blur': ['halo_blur'],\n 'text-halo-width': ['halo_width'],\n 'icon-halo-width': ['halo_width'],\n 'line-gap-width': ['gapwidth'],\n 'line-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'],\n 'fill-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'],\n 'fill-extrusion-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'],\n };\n\n return attributeNameExceptions[property] || [property.replace(`${type}-`, '').replace(/-/g, '_')];\n}\n\nfunction getLayoutException(property) {\n const propertyExceptions = {\n 'line-pattern': {\n 'source': PatternLayoutArray,\n 'composite': PatternLayoutArray\n },\n 'fill-pattern': {\n 'source': PatternLayoutArray,\n 'composite': PatternLayoutArray\n },\n 'fill-extrusion-pattern': {\n 'source': PatternLayoutArray,\n 'composite': PatternLayoutArray\n }\n };\n\n return propertyExceptions[property];\n}\n\nfunction layoutType(property, type, binderType) {\n const defaultLayouts = {\n 'color': {\n 'source': StructArrayLayout2f8,\n 'composite': StructArrayLayout4f16\n },\n 'number': {\n 'source': StructArrayLayout1f4,\n 'composite': StructArrayLayout2f8\n }\n };\n\n const layoutException = getLayoutException(property);\n return layoutException && layoutException[binderType] || defaultLayouts[type][binderType];\n}\n\nregister('ConstantBinder', ConstantBinder);\nregister('CrossFadedConstantBinder', CrossFadedConstantBinder);\nregister('SourceExpressionBinder', SourceExpressionBinder);\nregister('CrossFadedCompositeBinder', CrossFadedCompositeBinder);\nregister('CompositeExpressionBinder', CompositeExpressionBinder);\nregister('ProgramConfiguration', ProgramConfiguration, {omit: ['_buffers']});\nregister('ProgramConfigurationSet', ProgramConfigurationSet);\n","/**\n * The maximum value of a coordinate in the internal tile coordinate system. Coordinates of\n * all source features normalized to this extent upon load.\n *\n * The value is a consequence of the following:\n *\n * * Vertex buffer store positions as signed 16 bit integers.\n * * One bit is lost for signedness to support tile buffers.\n * * One bit is lost because the line vertex buffer used to pack 1 bit of other data into the int.\n * * One bit is lost to support features extending past the extent on the right edge of the tile.\n * * This leaves us with 2^13 = 8192\n */\nexport const EXTENT = 8192;\n","import {warnOnce, clamp} from '../util/util';\n\nimport {EXTENT} from './extent';\n\nimport type Point from '@mapbox/point-geometry';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\n// These bounds define the minimum and maximum supported coordinate values.\n// While visible coordinates are within [0, EXTENT], tiles may theoretically\n// contain coordinates within [-Infinity, Infinity]. Our range is limited by the\n// number of bits used to represent the coordinate.\nconst BITS = 15;\nconst MAX = Math.pow(2, BITS - 1) - 1;\nconst MIN = -MAX - 1;\n\n/**\n * Loads a geometry from a VectorTileFeature and scales it to the common extent\n * used internally.\n * @param feature - the vector tile feature to load\n */\nexport function loadGeometry(feature: VectorTileFeature): Array> {\n const scale = EXTENT / feature.extent;\n const geometry = feature.loadGeometry();\n for (let r = 0; r < geometry.length; r++) {\n const ring = geometry[r];\n for (let p = 0; p < ring.length; p++) {\n const point = ring[p];\n // round here because mapbox-gl-native uses integers to represent\n // points and we need to do the same to avoid renering differences.\n const x = Math.round(point.x * scale);\n const y = Math.round(point.y * scale);\n\n point.x = clamp(x, MIN, MAX);\n point.y = clamp(y, MIN, MAX);\n\n if (x < point.x || x > point.x + 1 || y < point.y || y > point.y + 1) {\n // warn when exceeding allowed extent except for the 1-px-off case\n // https://github.com/mapbox/mapbox-gl-js/issues/8992\n warnOnce('Geometry exceeds allowed extent, reduce your vector tile buffer size');\n }\n }\n }\n return geometry;\n}\n","import {loadGeometry} from './load_geometry';\nimport type Point from '@mapbox/point-geometry';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\nimport type {Feature} from '@maplibre/maplibre-gl-style-spec';\n\ntype EvaluationFeature = Feature & { geometry: Array> };\n/**\n * Construct a new feature based on a VectorTileFeature for expression evaluation, the geometry of which\n * will be loaded based on necessity.\n * @param feature - the feature to evaluate\n * @param needGeometry - if set to true this will load the geometry\n */\nexport function toEvaluationFeature(feature: VectorTileFeature, needGeometry: boolean): EvaluationFeature {\n return {type: feature.type,\n id: feature.id,\n properties: feature.properties,\n geometry: needGeometry ? loadGeometry(feature) : []};\n}\n","import {CircleLayoutArray} from '../array_types.g';\n\nimport {members as layoutAttributes} from './circle_attributes';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray} from '../index_array_type';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EXTENT} from '../extent';\nimport {register} from '../../util/web_worker_transfer';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {CircleStyleLayer} from '../../style/style_layer/circle_style_layer';\nimport type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nfunction addCircleVertex(layoutVertexArray, x, y, extrudeX, extrudeY) {\n layoutVertexArray.emplaceBack(\n (x * 2) + ((extrudeX + 1) / 2),\n (y * 2) + ((extrudeY + 1) / 2));\n}\n\n/**\n * @internal\n * Circles are represented by two triangles.\n *\n * Each corner has a pos that is the center of the circle and an extrusion\n * vector that is where it points.\n */\nexport class CircleBucket implements Bucket {\n index: number;\n zoom: number;\n overscaling: number;\n layerIds: Array;\n layers: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n\n layoutVertexArray: CircleLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n uploaded: boolean;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n\n this.layoutVertexArray = new CircleLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.segments = new SegmentVector();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n const styleLayer = this.layers[0];\n const bucketFeatures: BucketFeature[] = [];\n let circleSortKey = null;\n let sortFeaturesByKey = false;\n\n // Heatmap layers are handled in this bucket and have no evaluated properties, so we check our access\n if (styleLayer.type === 'circle') {\n circleSortKey = (styleLayer as CircleStyleLayer).layout.get('circle-sort-key');\n sortFeaturesByKey = !circleSortKey.isConstant();\n }\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const sortKey = sortFeaturesByKey ?\n circleSortKey.evaluate(evaluationFeature, {}, canonical) :\n undefined;\n\n const bucketFeature: BucketFeature = {\n id,\n properties: feature.properties,\n type: feature.type,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n patterns: {},\n sortKey\n };\n\n bucketFeatures.push(bucketFeature);\n\n }\n\n if (sortFeaturesByKey) {\n bucketFeatures.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const bucketFeature of bucketFeatures) {\n const {geometry, index, sourceLayerIndex} = bucketFeature;\n const feature = features[index].feature;\n\n this.addFeature(bucketFeature, geometry, index, canonical);\n options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0;\n }\n\n uploadPending() {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID) {\n for (const ring of geometry) {\n for (const point of ring) {\n const x = point.x;\n const y = point.y;\n\n // Do not include points that are outside the tile boundaries.\n if (x < 0 || x >= EXTENT || y < 0 || y >= EXTENT) continue;\n\n // this geometry will be of the Point type, and we'll derive\n // two triangles from it.\n //\n // ┌─────────┐\n // │ 3 2 │\n // │ │\n // │ 0 1 │\n // └─────────┘\n\n const segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray, feature.sortKey);\n const index = segment.vertexLength;\n\n addCircleVertex(this.layoutVertexArray, x, y, -1, -1);\n addCircleVertex(this.layoutVertexArray, x, y, 1, -1);\n addCircleVertex(this.layoutVertexArray, x, y, 1, 1);\n addCircleVertex(this.layoutVertexArray, x, y, -1, 1);\n\n this.indexArray.emplaceBack(index, index + 1, index + 2);\n this.indexArray.emplaceBack(index, index + 3, index + 2);\n\n segment.vertexLength += 4;\n segment.primitiveLength += 2;\n }\n }\n\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, {}, canonical);\n }\n}\n\nregister('CircleBucket', CircleBucket, {omit: ['layers']});\n","import {isCounterClockwise} from './util';\n\nimport Point from '@mapbox/point-geometry';\n\nexport {polygonIntersectsBufferedPoint, polygonIntersectsMultiPolygon, polygonIntersectsBufferedMultiLine, polygonIntersectsPolygon, distToSegmentSquared, polygonIntersectsBox};\n\ntype Line = Array;\ntype MultiLine = Array;\ntype Ring = Array;\ntype Polygon = Array;\ntype MultiPolygon = Array;\n\nfunction polygonIntersectsPolygon(polygonA: Polygon, polygonB: Polygon) {\n for (let i = 0; i < polygonA.length; i++) {\n if (polygonContainsPoint(polygonB, polygonA[i])) return true;\n }\n\n for (let i = 0; i < polygonB.length; i++) {\n if (polygonContainsPoint(polygonA, polygonB[i])) return true;\n }\n\n if (lineIntersectsLine(polygonA, polygonB)) return true;\n\n return false;\n}\n\nfunction polygonIntersectsBufferedPoint(polygon: Polygon, point: Point, radius: number) {\n if (polygonContainsPoint(polygon, point)) return true;\n if (pointIntersectsBufferedLine(point, polygon, radius)) return true;\n return false;\n}\n\nfunction polygonIntersectsMultiPolygon(polygon: Polygon, multiPolygon: MultiPolygon) {\n\n if (polygon.length === 1) {\n return multiPolygonContainsPoint(multiPolygon, polygon[0]);\n }\n\n for (let m = 0; m < multiPolygon.length; m++) {\n const ring = multiPolygon[m];\n for (let n = 0; n < ring.length; n++) {\n if (polygonContainsPoint(polygon, ring[n])) return true;\n }\n }\n\n for (let i = 0; i < polygon.length; i++) {\n if (multiPolygonContainsPoint(multiPolygon, polygon[i])) return true;\n }\n\n for (let k = 0; k < multiPolygon.length; k++) {\n if (lineIntersectsLine(polygon, multiPolygon[k])) return true;\n }\n\n return false;\n}\n\nfunction polygonIntersectsBufferedMultiLine(polygon: Polygon, multiLine: MultiLine, radius: number) {\n for (let i = 0; i < multiLine.length; i++) {\n const line = multiLine[i];\n\n if (polygon.length >= 3) {\n for (let k = 0; k < line.length; k++) {\n if (polygonContainsPoint(polygon, line[k])) return true;\n }\n }\n\n if (lineIntersectsBufferedLine(polygon, line, radius)) return true;\n }\n return false;\n}\n\nfunction lineIntersectsBufferedLine(lineA: Line, lineB: Line, radius: number) {\n\n if (lineA.length > 1) {\n if (lineIntersectsLine(lineA, lineB)) return true;\n\n // Check whether any point in either line is within radius of the other line\n for (let j = 0; j < lineB.length; j++) {\n if (pointIntersectsBufferedLine(lineB[j], lineA, radius)) return true;\n }\n }\n\n for (let k = 0; k < lineA.length; k++) {\n if (pointIntersectsBufferedLine(lineA[k], lineB, radius)) return true;\n }\n\n return false;\n}\n\nfunction lineIntersectsLine(lineA: Line, lineB: Line) {\n if (lineA.length === 0 || lineB.length === 0) return false;\n for (let i = 0; i < lineA.length - 1; i++) {\n const a0 = lineA[i];\n const a1 = lineA[i + 1];\n for (let j = 0; j < lineB.length - 1; j++) {\n const b0 = lineB[j];\n const b1 = lineB[j + 1];\n if (lineSegmentIntersectsLineSegment(a0, a1, b0, b1)) return true;\n }\n }\n return false;\n}\n\nfunction lineSegmentIntersectsLineSegment(a0: Point, a1: Point, b0: Point, b1: Point) {\n return isCounterClockwise(a0, b0, b1) !== isCounterClockwise(a1, b0, b1) &&\n isCounterClockwise(a0, a1, b0) !== isCounterClockwise(a0, a1, b1);\n}\n\nfunction pointIntersectsBufferedLine(p: Point, line: Line, radius: number) {\n const radiusSquared = radius * radius;\n\n if (line.length === 1) return p.distSqr(line[0]) < radiusSquared;\n\n for (let i = 1; i < line.length; i++) {\n // Find line segments that have a distance <= radius^2 to p\n // In that case, we treat the line as \"containing point p\".\n const v = line[i - 1], w = line[i];\n if (distToSegmentSquared(p, v, w) < radiusSquared) return true;\n }\n return false;\n}\n\n// Code from http://stackoverflow.com/a/1501725/331379.\nfunction distToSegmentSquared(p: Point, v: Point, w: Point) {\n const l2 = v.distSqr(w);\n if (l2 === 0) return p.distSqr(v);\n const t = ((p.x - v.x) * (w.x - v.x) + (p.y - v.y) * (w.y - v.y)) / l2;\n if (t < 0) return p.distSqr(v);\n if (t > 1) return p.distSqr(w);\n return p.distSqr(w.sub(v)._mult(t)._add(v));\n}\n\n// point in polygon ray casting algorithm\nfunction multiPolygonContainsPoint(rings: Array, p: Point) {\n let c = false,\n ring, p1, p2;\n\n for (let k = 0; k < rings.length; k++) {\n ring = rings[k];\n for (let i = 0, j = ring.length - 1; i < ring.length; j = i++) {\n p1 = ring[i];\n p2 = ring[j];\n if (((p1.y > p.y) !== (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) {\n c = !c;\n }\n }\n }\n return c;\n}\n\nfunction polygonContainsPoint(ring: Ring, p: Point) {\n let c = false;\n for (let i = 0, j = ring.length - 1; i < ring.length; j = i++) {\n const p1 = ring[i];\n const p2 = ring[j];\n if (((p1.y > p.y) !== (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) {\n c = !c;\n }\n }\n return c;\n}\n\nfunction polygonIntersectsBox(ring: Ring, boxX1: number, boxY1: number, boxX2: number, boxY2: number) {\n for (const p of ring) {\n if (boxX1 <= p.x &&\n boxY1 <= p.y &&\n boxX2 >= p.x &&\n boxY2 >= p.y) return true;\n }\n\n const corners = [\n new Point(boxX1, boxY1),\n new Point(boxX1, boxY2),\n new Point(boxX2, boxY2),\n new Point(boxX2, boxY1)];\n\n if (ring.length > 2) {\n for (const corner of corners) {\n if (polygonContainsPoint(ring, corner)) return true;\n }\n }\n\n for (let i = 0; i < ring.length - 1; i++) {\n const p1 = ring[i];\n const p2 = ring[i + 1];\n if (edgeIntersectsBox(p1, p2, corners)) return true;\n }\n\n return false;\n}\n\nfunction edgeIntersectsBox(e1: Point, e2: Point, corners: Array) {\n const tl = corners[0];\n const br = corners[2];\n // the edge and box do not intersect in either the x or y dimensions\n if (((e1.x < tl.x) && (e2.x < tl.x)) ||\n ((e1.x > br.x) && (e2.x > br.x)) ||\n ((e1.y < tl.y) && (e2.y < tl.y)) ||\n ((e1.y > br.y) && (e2.y > br.y))) return false;\n\n // check if all corners of the box are on the same side of the edge\n const dir = isCounterClockwise(e1, e2, corners[0]);\n return dir !== isCounterClockwise(e1, e2, corners[1]) ||\n dir !== isCounterClockwise(e1, e2, corners[2]) ||\n dir !== isCounterClockwise(e1, e2, corners[3]);\n}\n","import Point from '@mapbox/point-geometry';\n\nimport type {PossiblyEvaluatedPropertyValue} from './properties';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {CircleBucket} from '../data/bucket/circle_bucket';\nimport type {LineBucket} from '../data/bucket/line_bucket';\n\nexport function getMaximumPaintValue(\n property: string,\n layer: StyleLayer,\n bucket: CircleBucket | LineBucket\n): number {\n const value = ((layer.paint as any).get(property) as PossiblyEvaluatedPropertyValue).value;\n if (value.kind === 'constant') {\n return value.value;\n } else {\n return bucket.programConfigurations.get(layer.id).getMaxValue(property);\n }\n}\n\nexport function translateDistance(translate: [number, number]) {\n return Math.sqrt(translate[0] * translate[0] + translate[1] * translate[1]);\n}\n\nexport function translate(queryGeometry: Array,\n translate: [number, number],\n translateAnchor: 'viewport' | 'map',\n bearing: number,\n pixelsToTileUnits: number) {\n if (!translate[0] && !translate[1]) {\n return queryGeometry;\n }\n const pt = Point.convert(translate)._mult(pixelsToTileUnits);\n\n if (translateAnchor === 'viewport') {\n pt._rotate(-bearing);\n }\n\n const translated = [];\n for (let i = 0; i < queryGeometry.length; i++) {\n const point = queryGeometry[i];\n translated.push(point.sub(pt));\n }\n return translated;\n}\n\nexport function offsetLine(rings: Array>, offset: number) {\n const newRings: Array> = [];\n for (let ringIndex = 0; ringIndex < rings.length; ringIndex++) {\n const ring = rings[ringIndex];\n const newRing: Array = [];\n for (let index = 0; index < ring.length; index++) {\n const a = ring[index - 1];\n const b = ring[index];\n const c = ring[index + 1];\n const aToB = index === 0 ? new Point(0, 0) : b.sub(a)._unit()._perp();\n const bToC = index === ring.length - 1 ? new Point(0, 0) : c.sub(b)._unit()._perp();\n const extrude = aToB._add(bToC)._unit();\n\n const cosHalfAngle = extrude.x * bToC.x + extrude.y * bToC.y;\n if (cosHalfAngle !== 0) {\n extrude._mult(1 / cosHalfAngle);\n }\n\n newRing.push(extrude._mult(offset)._add(b));\n }\n newRings.push(newRing);\n }\n return newRings;\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type CircleLayoutProps = {\n \"circle-sort-key\": DataDrivenProperty,\n};\n\nexport type CircleLayoutPropsPossiblyEvaluated = {\n \"circle-sort-key\": PossiblyEvaluatedPropertyValue,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"circle-sort-key\": new DataDrivenProperty(styleSpec[\"layout_circle\"][\"circle-sort-key\"] as any as StylePropertySpecification),\n});\n\nexport type CirclePaintProps = {\n \"circle-radius\": DataDrivenProperty,\n \"circle-color\": DataDrivenProperty,\n \"circle-blur\": DataDrivenProperty,\n \"circle-opacity\": DataDrivenProperty,\n \"circle-translate\": DataConstantProperty<[number, number]>,\n \"circle-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"circle-pitch-scale\": DataConstantProperty<\"map\" | \"viewport\">,\n \"circle-pitch-alignment\": DataConstantProperty<\"map\" | \"viewport\">,\n \"circle-stroke-width\": DataDrivenProperty,\n \"circle-stroke-color\": DataDrivenProperty,\n \"circle-stroke-opacity\": DataDrivenProperty,\n};\n\nexport type CirclePaintPropsPossiblyEvaluated = {\n \"circle-radius\": PossiblyEvaluatedPropertyValue,\n \"circle-color\": PossiblyEvaluatedPropertyValue,\n \"circle-blur\": PossiblyEvaluatedPropertyValue,\n \"circle-opacity\": PossiblyEvaluatedPropertyValue,\n \"circle-translate\": [number, number],\n \"circle-translate-anchor\": \"map\" | \"viewport\",\n \"circle-pitch-scale\": \"map\" | \"viewport\",\n \"circle-pitch-alignment\": \"map\" | \"viewport\",\n \"circle-stroke-width\": PossiblyEvaluatedPropertyValue,\n \"circle-stroke-color\": PossiblyEvaluatedPropertyValue,\n \"circle-stroke-opacity\": PossiblyEvaluatedPropertyValue,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"circle-radius\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-radius\"] as any as StylePropertySpecification),\n \"circle-color\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-color\"] as any as StylePropertySpecification),\n \"circle-blur\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-blur\"] as any as StylePropertySpecification),\n \"circle-opacity\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-opacity\"] as any as StylePropertySpecification),\n \"circle-translate\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-translate\"] as any as StylePropertySpecification),\n \"circle-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-translate-anchor\"] as any as StylePropertySpecification),\n \"circle-pitch-scale\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-pitch-scale\"] as any as StylePropertySpecification),\n \"circle-pitch-alignment\": new DataConstantProperty(styleSpec[\"paint_circle\"][\"circle-pitch-alignment\"] as any as StylePropertySpecification),\n \"circle-stroke-width\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-stroke-width\"] as any as StylePropertySpecification),\n \"circle-stroke-color\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-stroke-color\"] as any as StylePropertySpecification),\n \"circle-stroke-opacity\": new DataDrivenProperty(styleSpec[\"paint_circle\"][\"circle-stroke-opacity\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","/**\n * Common utilities\n * @module glMatrix\n */\n// Configuration Constants\nexport var EPSILON = 0.000001;\nexport var ARRAY_TYPE = typeof Float32Array !== 'undefined' ? Float32Array : Array;\nexport var RANDOM = Math.random;\n/**\n * Sets the type of array used when creating new vectors and matrices\n *\n * @param {Float32ArrayConstructor | ArrayConstructor} type Array type, such as Float32Array or Array\n */\n\nexport function setMatrixArrayType(type) {\n ARRAY_TYPE = type;\n}\nvar degree = Math.PI / 180;\n/**\n * Convert Degree To Radian\n *\n * @param {Number} a Angle in Degrees\n */\n\nexport function toRadian(a) {\n return a * degree;\n}\n/**\n * Tests whether or not the arguments have approximately the same value, within an absolute\n * or relative tolerance of glMatrix.EPSILON (an absolute tolerance is used for values less\n * than or equal to 1.0, and a relative tolerance is used for larger values)\n *\n * @param {Number} a The first number to test.\n * @param {Number} b The second number to test.\n * @returns {Boolean} True if the numbers are approximately equal, false otherwise.\n */\n\nexport function equals(a, b) {\n return Math.abs(a - b) <= EPSILON * Math.max(1.0, Math.abs(a), Math.abs(b));\n}\nif (!Math.hypot) Math.hypot = function () {\n var y = 0,\n i = arguments.length;\n\n while (i--) {\n y += arguments[i] * arguments[i];\n }\n\n return Math.sqrt(y);\n};","import * as glMatrix from \"./common.js\";\n/**\n * 4x4 Matrix
Format: column-major, when typed out it looks like row-major
The matrices are being post multiplied.\n * @module mat4\n */\n\n/**\n * Creates a new identity mat4\n *\n * @returns {mat4} a new 4x4 matrix\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(16);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n }\n\n out[0] = 1;\n out[5] = 1;\n out[10] = 1;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a new mat4 initialized with values from an existing matrix\n *\n * @param {ReadonlyMat4} a matrix to clone\n * @returns {mat4} a new 4x4 matrix\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(16);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n out[9] = a[9];\n out[10] = a[10];\n out[11] = a[11];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n return out;\n}\n/**\n * Copy the values from one mat4 to another\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n out[9] = a[9];\n out[10] = a[10];\n out[11] = a[11];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n return out;\n}\n/**\n * Create a new mat4 with the given values\n *\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m03 Component in column 0, row 3 position (index 3)\n * @param {Number} m10 Component in column 1, row 0 position (index 4)\n * @param {Number} m11 Component in column 1, row 1 position (index 5)\n * @param {Number} m12 Component in column 1, row 2 position (index 6)\n * @param {Number} m13 Component in column 1, row 3 position (index 7)\n * @param {Number} m20 Component in column 2, row 0 position (index 8)\n * @param {Number} m21 Component in column 2, row 1 position (index 9)\n * @param {Number} m22 Component in column 2, row 2 position (index 10)\n * @param {Number} m23 Component in column 2, row 3 position (index 11)\n * @param {Number} m30 Component in column 3, row 0 position (index 12)\n * @param {Number} m31 Component in column 3, row 1 position (index 13)\n * @param {Number} m32 Component in column 3, row 2 position (index 14)\n * @param {Number} m33 Component in column 3, row 3 position (index 15)\n * @returns {mat4} A new mat4\n */\n\nexport function fromValues(m00, m01, m02, m03, m10, m11, m12, m13, m20, m21, m22, m23, m30, m31, m32, m33) {\n var out = new glMatrix.ARRAY_TYPE(16);\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m03;\n out[4] = m10;\n out[5] = m11;\n out[6] = m12;\n out[7] = m13;\n out[8] = m20;\n out[9] = m21;\n out[10] = m22;\n out[11] = m23;\n out[12] = m30;\n out[13] = m31;\n out[14] = m32;\n out[15] = m33;\n return out;\n}\n/**\n * Set the components of a mat4 to the given values\n *\n * @param {mat4} out the receiving matrix\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m03 Component in column 0, row 3 position (index 3)\n * @param {Number} m10 Component in column 1, row 0 position (index 4)\n * @param {Number} m11 Component in column 1, row 1 position (index 5)\n * @param {Number} m12 Component in column 1, row 2 position (index 6)\n * @param {Number} m13 Component in column 1, row 3 position (index 7)\n * @param {Number} m20 Component in column 2, row 0 position (index 8)\n * @param {Number} m21 Component in column 2, row 1 position (index 9)\n * @param {Number} m22 Component in column 2, row 2 position (index 10)\n * @param {Number} m23 Component in column 2, row 3 position (index 11)\n * @param {Number} m30 Component in column 3, row 0 position (index 12)\n * @param {Number} m31 Component in column 3, row 1 position (index 13)\n * @param {Number} m32 Component in column 3, row 2 position (index 14)\n * @param {Number} m33 Component in column 3, row 3 position (index 15)\n * @returns {mat4} out\n */\n\nexport function set(out, m00, m01, m02, m03, m10, m11, m12, m13, m20, m21, m22, m23, m30, m31, m32, m33) {\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m03;\n out[4] = m10;\n out[5] = m11;\n out[6] = m12;\n out[7] = m13;\n out[8] = m20;\n out[9] = m21;\n out[10] = m22;\n out[11] = m23;\n out[12] = m30;\n out[13] = m31;\n out[14] = m32;\n out[15] = m33;\n return out;\n}\n/**\n * Set a mat4 to the identity matrix\n *\n * @param {mat4} out the receiving matrix\n * @returns {mat4} out\n */\n\nexport function identity(out) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = 1;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 1;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Transpose the values of a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function transpose(out, a) {\n // If we are transposing ourselves we can skip a few steps but have to cache some values\n if (out === a) {\n var a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a12 = a[6],\n a13 = a[7];\n var a23 = a[11];\n out[1] = a[4];\n out[2] = a[8];\n out[3] = a[12];\n out[4] = a01;\n out[6] = a[9];\n out[7] = a[13];\n out[8] = a02;\n out[9] = a12;\n out[11] = a[14];\n out[12] = a03;\n out[13] = a13;\n out[14] = a23;\n } else {\n out[0] = a[0];\n out[1] = a[4];\n out[2] = a[8];\n out[3] = a[12];\n out[4] = a[1];\n out[5] = a[5];\n out[6] = a[9];\n out[7] = a[13];\n out[8] = a[2];\n out[9] = a[6];\n out[10] = a[10];\n out[11] = a[14];\n out[12] = a[3];\n out[13] = a[7];\n out[14] = a[11];\n out[15] = a[15];\n }\n\n return out;\n}\n/**\n * Inverts a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function invert(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n var b00 = a00 * a11 - a01 * a10;\n var b01 = a00 * a12 - a02 * a10;\n var b02 = a00 * a13 - a03 * a10;\n var b03 = a01 * a12 - a02 * a11;\n var b04 = a01 * a13 - a03 * a11;\n var b05 = a02 * a13 - a03 * a12;\n var b06 = a20 * a31 - a21 * a30;\n var b07 = a20 * a32 - a22 * a30;\n var b08 = a20 * a33 - a23 * a30;\n var b09 = a21 * a32 - a22 * a31;\n var b10 = a21 * a33 - a23 * a31;\n var b11 = a22 * a33 - a23 * a32; // Calculate the determinant\n\n var det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det;\n out[1] = (a02 * b10 - a01 * b11 - a03 * b09) * det;\n out[2] = (a31 * b05 - a32 * b04 + a33 * b03) * det;\n out[3] = (a22 * b04 - a21 * b05 - a23 * b03) * det;\n out[4] = (a12 * b08 - a10 * b11 - a13 * b07) * det;\n out[5] = (a00 * b11 - a02 * b08 + a03 * b07) * det;\n out[6] = (a32 * b02 - a30 * b05 - a33 * b01) * det;\n out[7] = (a20 * b05 - a22 * b02 + a23 * b01) * det;\n out[8] = (a10 * b10 - a11 * b08 + a13 * b06) * det;\n out[9] = (a01 * b08 - a00 * b10 - a03 * b06) * det;\n out[10] = (a30 * b04 - a31 * b02 + a33 * b00) * det;\n out[11] = (a21 * b02 - a20 * b04 - a23 * b00) * det;\n out[12] = (a11 * b07 - a10 * b09 - a12 * b06) * det;\n out[13] = (a00 * b09 - a01 * b07 + a02 * b06) * det;\n out[14] = (a31 * b01 - a30 * b03 - a32 * b00) * det;\n out[15] = (a20 * b03 - a21 * b01 + a22 * b00) * det;\n return out;\n}\n/**\n * Calculates the adjugate of a mat4\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the source matrix\n * @returns {mat4} out\n */\n\nexport function adjoint(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n out[0] = a11 * (a22 * a33 - a23 * a32) - a21 * (a12 * a33 - a13 * a32) + a31 * (a12 * a23 - a13 * a22);\n out[1] = -(a01 * (a22 * a33 - a23 * a32) - a21 * (a02 * a33 - a03 * a32) + a31 * (a02 * a23 - a03 * a22));\n out[2] = a01 * (a12 * a33 - a13 * a32) - a11 * (a02 * a33 - a03 * a32) + a31 * (a02 * a13 - a03 * a12);\n out[3] = -(a01 * (a12 * a23 - a13 * a22) - a11 * (a02 * a23 - a03 * a22) + a21 * (a02 * a13 - a03 * a12));\n out[4] = -(a10 * (a22 * a33 - a23 * a32) - a20 * (a12 * a33 - a13 * a32) + a30 * (a12 * a23 - a13 * a22));\n out[5] = a00 * (a22 * a33 - a23 * a32) - a20 * (a02 * a33 - a03 * a32) + a30 * (a02 * a23 - a03 * a22);\n out[6] = -(a00 * (a12 * a33 - a13 * a32) - a10 * (a02 * a33 - a03 * a32) + a30 * (a02 * a13 - a03 * a12));\n out[7] = a00 * (a12 * a23 - a13 * a22) - a10 * (a02 * a23 - a03 * a22) + a20 * (a02 * a13 - a03 * a12);\n out[8] = a10 * (a21 * a33 - a23 * a31) - a20 * (a11 * a33 - a13 * a31) + a30 * (a11 * a23 - a13 * a21);\n out[9] = -(a00 * (a21 * a33 - a23 * a31) - a20 * (a01 * a33 - a03 * a31) + a30 * (a01 * a23 - a03 * a21));\n out[10] = a00 * (a11 * a33 - a13 * a31) - a10 * (a01 * a33 - a03 * a31) + a30 * (a01 * a13 - a03 * a11);\n out[11] = -(a00 * (a11 * a23 - a13 * a21) - a10 * (a01 * a23 - a03 * a21) + a20 * (a01 * a13 - a03 * a11));\n out[12] = -(a10 * (a21 * a32 - a22 * a31) - a20 * (a11 * a32 - a12 * a31) + a30 * (a11 * a22 - a12 * a21));\n out[13] = a00 * (a21 * a32 - a22 * a31) - a20 * (a01 * a32 - a02 * a31) + a30 * (a01 * a22 - a02 * a21);\n out[14] = -(a00 * (a11 * a32 - a12 * a31) - a10 * (a01 * a32 - a02 * a31) + a30 * (a01 * a12 - a02 * a11));\n out[15] = a00 * (a11 * a22 - a12 * a21) - a10 * (a01 * a22 - a02 * a21) + a20 * (a01 * a12 - a02 * a11);\n return out;\n}\n/**\n * Calculates the determinant of a mat4\n *\n * @param {ReadonlyMat4} a the source matrix\n * @returns {Number} determinant of a\n */\n\nexport function determinant(a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n var b00 = a00 * a11 - a01 * a10;\n var b01 = a00 * a12 - a02 * a10;\n var b02 = a00 * a13 - a03 * a10;\n var b03 = a01 * a12 - a02 * a11;\n var b04 = a01 * a13 - a03 * a11;\n var b05 = a02 * a13 - a03 * a12;\n var b06 = a20 * a31 - a21 * a30;\n var b07 = a20 * a32 - a22 * a30;\n var b08 = a20 * a33 - a23 * a30;\n var b09 = a21 * a32 - a22 * a31;\n var b10 = a21 * a33 - a23 * a31;\n var b11 = a22 * a33 - a23 * a32; // Calculate the determinant\n\n return b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n}\n/**\n * Multiplies two mat4s\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @returns {mat4} out\n */\n\nexport function multiply(out, a, b) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15]; // Cache only the current line of the second matrix\n\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n out[0] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[1] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[2] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[3] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n b0 = b[4];\n b1 = b[5];\n b2 = b[6];\n b3 = b[7];\n out[4] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[5] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[6] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[7] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n b0 = b[8];\n b1 = b[9];\n b2 = b[10];\n b3 = b[11];\n out[8] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[9] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[10] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[11] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n b0 = b[12];\n b1 = b[13];\n b2 = b[14];\n b3 = b[15];\n out[12] = b0 * a00 + b1 * a10 + b2 * a20 + b3 * a30;\n out[13] = b0 * a01 + b1 * a11 + b2 * a21 + b3 * a31;\n out[14] = b0 * a02 + b1 * a12 + b2 * a22 + b3 * a32;\n out[15] = b0 * a03 + b1 * a13 + b2 * a23 + b3 * a33;\n return out;\n}\n/**\n * Translate a mat4 by the given vector\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to translate\n * @param {ReadonlyVec3} v vector to translate by\n * @returns {mat4} out\n */\n\nexport function translate(out, a, v) {\n var x = v[0],\n y = v[1],\n z = v[2];\n var a00, a01, a02, a03;\n var a10, a11, a12, a13;\n var a20, a21, a22, a23;\n\n if (a === out) {\n out[12] = a[0] * x + a[4] * y + a[8] * z + a[12];\n out[13] = a[1] * x + a[5] * y + a[9] * z + a[13];\n out[14] = a[2] * x + a[6] * y + a[10] * z + a[14];\n out[15] = a[3] * x + a[7] * y + a[11] * z + a[15];\n } else {\n a00 = a[0];\n a01 = a[1];\n a02 = a[2];\n a03 = a[3];\n a10 = a[4];\n a11 = a[5];\n a12 = a[6];\n a13 = a[7];\n a20 = a[8];\n a21 = a[9];\n a22 = a[10];\n a23 = a[11];\n out[0] = a00;\n out[1] = a01;\n out[2] = a02;\n out[3] = a03;\n out[4] = a10;\n out[5] = a11;\n out[6] = a12;\n out[7] = a13;\n out[8] = a20;\n out[9] = a21;\n out[10] = a22;\n out[11] = a23;\n out[12] = a00 * x + a10 * y + a20 * z + a[12];\n out[13] = a01 * x + a11 * y + a21 * z + a[13];\n out[14] = a02 * x + a12 * y + a22 * z + a[14];\n out[15] = a03 * x + a13 * y + a23 * z + a[15];\n }\n\n return out;\n}\n/**\n * Scales the mat4 by the dimensions in the given vec3 not using vectorization\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to scale\n * @param {ReadonlyVec3} v the vec3 to scale the matrix by\n * @returns {mat4} out\n **/\n\nexport function scale(out, a, v) {\n var x = v[0],\n y = v[1],\n z = v[2];\n out[0] = a[0] * x;\n out[1] = a[1] * x;\n out[2] = a[2] * x;\n out[3] = a[3] * x;\n out[4] = a[4] * y;\n out[5] = a[5] * y;\n out[6] = a[6] * y;\n out[7] = a[7] * y;\n out[8] = a[8] * z;\n out[9] = a[9] * z;\n out[10] = a[10] * z;\n out[11] = a[11] * z;\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n return out;\n}\n/**\n * Rotates a mat4 by the given angle around the given axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @param {ReadonlyVec3} axis the axis to rotate around\n * @returns {mat4} out\n */\n\nexport function rotate(out, a, rad, axis) {\n var x = axis[0],\n y = axis[1],\n z = axis[2];\n var len = Math.hypot(x, y, z);\n var s, c, t;\n var a00, a01, a02, a03;\n var a10, a11, a12, a13;\n var a20, a21, a22, a23;\n var b00, b01, b02;\n var b10, b11, b12;\n var b20, b21, b22;\n\n if (len < glMatrix.EPSILON) {\n return null;\n }\n\n len = 1 / len;\n x *= len;\n y *= len;\n z *= len;\n s = Math.sin(rad);\n c = Math.cos(rad);\n t = 1 - c;\n a00 = a[0];\n a01 = a[1];\n a02 = a[2];\n a03 = a[3];\n a10 = a[4];\n a11 = a[5];\n a12 = a[6];\n a13 = a[7];\n a20 = a[8];\n a21 = a[9];\n a22 = a[10];\n a23 = a[11]; // Construct the elements of the rotation matrix\n\n b00 = x * x * t + c;\n b01 = y * x * t + z * s;\n b02 = z * x * t - y * s;\n b10 = x * y * t - z * s;\n b11 = y * y * t + c;\n b12 = z * y * t + x * s;\n b20 = x * z * t + y * s;\n b21 = y * z * t - x * s;\n b22 = z * z * t + c; // Perform rotation-specific matrix multiplication\n\n out[0] = a00 * b00 + a10 * b01 + a20 * b02;\n out[1] = a01 * b00 + a11 * b01 + a21 * b02;\n out[2] = a02 * b00 + a12 * b01 + a22 * b02;\n out[3] = a03 * b00 + a13 * b01 + a23 * b02;\n out[4] = a00 * b10 + a10 * b11 + a20 * b12;\n out[5] = a01 * b10 + a11 * b11 + a21 * b12;\n out[6] = a02 * b10 + a12 * b11 + a22 * b12;\n out[7] = a03 * b10 + a13 * b11 + a23 * b12;\n out[8] = a00 * b20 + a10 * b21 + a20 * b22;\n out[9] = a01 * b20 + a11 * b21 + a21 * b22;\n out[10] = a02 * b20 + a12 * b21 + a22 * b22;\n out[11] = a03 * b20 + a13 * b21 + a23 * b22;\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged last row\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n }\n\n return out;\n}\n/**\n * Rotates a matrix by the given angle around the X axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function rotateX(out, a, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n var a10 = a[4];\n var a11 = a[5];\n var a12 = a[6];\n var a13 = a[7];\n var a20 = a[8];\n var a21 = a[9];\n var a22 = a[10];\n var a23 = a[11];\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged rows\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n } // Perform axis-specific matrix multiplication\n\n\n out[4] = a10 * c + a20 * s;\n out[5] = a11 * c + a21 * s;\n out[6] = a12 * c + a22 * s;\n out[7] = a13 * c + a23 * s;\n out[8] = a20 * c - a10 * s;\n out[9] = a21 * c - a11 * s;\n out[10] = a22 * c - a12 * s;\n out[11] = a23 * c - a13 * s;\n return out;\n}\n/**\n * Rotates a matrix by the given angle around the Y axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function rotateY(out, a, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n var a00 = a[0];\n var a01 = a[1];\n var a02 = a[2];\n var a03 = a[3];\n var a20 = a[8];\n var a21 = a[9];\n var a22 = a[10];\n var a23 = a[11];\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged rows\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n } // Perform axis-specific matrix multiplication\n\n\n out[0] = a00 * c - a20 * s;\n out[1] = a01 * c - a21 * s;\n out[2] = a02 * c - a22 * s;\n out[3] = a03 * c - a23 * s;\n out[8] = a00 * s + a20 * c;\n out[9] = a01 * s + a21 * c;\n out[10] = a02 * s + a22 * c;\n out[11] = a03 * s + a23 * c;\n return out;\n}\n/**\n * Rotates a matrix by the given angle around the Z axis\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function rotateZ(out, a, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n var a00 = a[0];\n var a01 = a[1];\n var a02 = a[2];\n var a03 = a[3];\n var a10 = a[4];\n var a11 = a[5];\n var a12 = a[6];\n var a13 = a[7];\n\n if (a !== out) {\n // If the source and destination differ, copy the unchanged last row\n out[8] = a[8];\n out[9] = a[9];\n out[10] = a[10];\n out[11] = a[11];\n out[12] = a[12];\n out[13] = a[13];\n out[14] = a[14];\n out[15] = a[15];\n } // Perform axis-specific matrix multiplication\n\n\n out[0] = a00 * c + a10 * s;\n out[1] = a01 * c + a11 * s;\n out[2] = a02 * c + a12 * s;\n out[3] = a03 * c + a13 * s;\n out[4] = a10 * c - a00 * s;\n out[5] = a11 * c - a01 * s;\n out[6] = a12 * c - a02 * s;\n out[7] = a13 * c - a03 * s;\n return out;\n}\n/**\n * Creates a matrix from a vector translation\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, dest, vec);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {ReadonlyVec3} v Translation vector\n * @returns {mat4} out\n */\n\nexport function fromTranslation(out, v) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = 1;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 1;\n out[11] = 0;\n out[12] = v[0];\n out[13] = v[1];\n out[14] = v[2];\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a vector scaling\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.scale(dest, dest, vec);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {ReadonlyVec3} v Scaling vector\n * @returns {mat4} out\n */\n\nexport function fromScaling(out, v) {\n out[0] = v[0];\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = v[1];\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = v[2];\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a given angle around a given axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotate(dest, dest, rad, axis);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @param {ReadonlyVec3} axis the axis to rotate around\n * @returns {mat4} out\n */\n\nexport function fromRotation(out, rad, axis) {\n var x = axis[0],\n y = axis[1],\n z = axis[2];\n var len = Math.hypot(x, y, z);\n var s, c, t;\n\n if (len < glMatrix.EPSILON) {\n return null;\n }\n\n len = 1 / len;\n x *= len;\n y *= len;\n z *= len;\n s = Math.sin(rad);\n c = Math.cos(rad);\n t = 1 - c; // Perform rotation-specific matrix multiplication\n\n out[0] = x * x * t + c;\n out[1] = y * x * t + z * s;\n out[2] = z * x * t - y * s;\n out[3] = 0;\n out[4] = x * y * t - z * s;\n out[5] = y * y * t + c;\n out[6] = z * y * t + x * s;\n out[7] = 0;\n out[8] = x * z * t + y * s;\n out[9] = y * z * t - x * s;\n out[10] = z * z * t + c;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from the given angle around the X axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotateX(dest, dest, rad);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function fromXRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad); // Perform axis-specific matrix multiplication\n\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = c;\n out[6] = s;\n out[7] = 0;\n out[8] = 0;\n out[9] = -s;\n out[10] = c;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from the given angle around the Y axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotateY(dest, dest, rad);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function fromYRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad); // Perform axis-specific matrix multiplication\n\n out[0] = c;\n out[1] = 0;\n out[2] = -s;\n out[3] = 0;\n out[4] = 0;\n out[5] = 1;\n out[6] = 0;\n out[7] = 0;\n out[8] = s;\n out[9] = 0;\n out[10] = c;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from the given angle around the Z axis\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.rotateZ(dest, dest, rad);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat4} out\n */\n\nexport function fromZRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad); // Perform axis-specific matrix multiplication\n\n out[0] = c;\n out[1] = s;\n out[2] = 0;\n out[3] = 0;\n out[4] = -s;\n out[5] = c;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 1;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a quaternion rotation and vector translation\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, vec);\n * let quatMat = mat4.create();\n * quat4.toMat4(quat, quatMat);\n * mat4.multiply(dest, quatMat);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {ReadonlyVec3} v Translation vector\n * @returns {mat4} out\n */\n\nexport function fromRotationTranslation(out, q, v) {\n // Quaternion math\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var xy = x * y2;\n var xz = x * z2;\n var yy = y * y2;\n var yz = y * z2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n out[0] = 1 - (yy + zz);\n out[1] = xy + wz;\n out[2] = xz - wy;\n out[3] = 0;\n out[4] = xy - wz;\n out[5] = 1 - (xx + zz);\n out[6] = yz + wx;\n out[7] = 0;\n out[8] = xz + wy;\n out[9] = yz - wx;\n out[10] = 1 - (xx + yy);\n out[11] = 0;\n out[12] = v[0];\n out[13] = v[1];\n out[14] = v[2];\n out[15] = 1;\n return out;\n}\n/**\n * Creates a new mat4 from a dual quat.\n *\n * @param {mat4} out Matrix\n * @param {ReadonlyQuat2} a Dual Quaternion\n * @returns {mat4} mat4 receiving operation result\n */\n\nexport function fromQuat2(out, a) {\n var translation = new glMatrix.ARRAY_TYPE(3);\n var bx = -a[0],\n by = -a[1],\n bz = -a[2],\n bw = a[3],\n ax = a[4],\n ay = a[5],\n az = a[6],\n aw = a[7];\n var magnitude = bx * bx + by * by + bz * bz + bw * bw; //Only scale if it makes sense\n\n if (magnitude > 0) {\n translation[0] = (ax * bw + aw * bx + ay * bz - az * by) * 2 / magnitude;\n translation[1] = (ay * bw + aw * by + az * bx - ax * bz) * 2 / magnitude;\n translation[2] = (az * bw + aw * bz + ax * by - ay * bx) * 2 / magnitude;\n } else {\n translation[0] = (ax * bw + aw * bx + ay * bz - az * by) * 2;\n translation[1] = (ay * bw + aw * by + az * bx - ax * bz) * 2;\n translation[2] = (az * bw + aw * bz + ax * by - ay * bx) * 2;\n }\n\n fromRotationTranslation(out, a, translation);\n return out;\n}\n/**\n * Returns the translation vector component of a transformation\n * matrix. If a matrix is built with fromRotationTranslation,\n * the returned vector will be the same as the translation vector\n * originally supplied.\n * @param {vec3} out Vector to receive translation component\n * @param {ReadonlyMat4} mat Matrix to be decomposed (input)\n * @return {vec3} out\n */\n\nexport function getTranslation(out, mat) {\n out[0] = mat[12];\n out[1] = mat[13];\n out[2] = mat[14];\n return out;\n}\n/**\n * Returns the scaling factor component of a transformation\n * matrix. If a matrix is built with fromRotationTranslationScale\n * with a normalized Quaternion paramter, the returned vector will be\n * the same as the scaling vector\n * originally supplied.\n * @param {vec3} out Vector to receive scaling factor component\n * @param {ReadonlyMat4} mat Matrix to be decomposed (input)\n * @return {vec3} out\n */\n\nexport function getScaling(out, mat) {\n var m11 = mat[0];\n var m12 = mat[1];\n var m13 = mat[2];\n var m21 = mat[4];\n var m22 = mat[5];\n var m23 = mat[6];\n var m31 = mat[8];\n var m32 = mat[9];\n var m33 = mat[10];\n out[0] = Math.hypot(m11, m12, m13);\n out[1] = Math.hypot(m21, m22, m23);\n out[2] = Math.hypot(m31, m32, m33);\n return out;\n}\n/**\n * Returns a quaternion representing the rotational component\n * of a transformation matrix. If a matrix is built with\n * fromRotationTranslation, the returned quaternion will be the\n * same as the quaternion originally supplied.\n * @param {quat} out Quaternion to receive the rotation component\n * @param {ReadonlyMat4} mat Matrix to be decomposed (input)\n * @return {quat} out\n */\n\nexport function getRotation(out, mat) {\n var scaling = new glMatrix.ARRAY_TYPE(3);\n getScaling(scaling, mat);\n var is1 = 1 / scaling[0];\n var is2 = 1 / scaling[1];\n var is3 = 1 / scaling[2];\n var sm11 = mat[0] * is1;\n var sm12 = mat[1] * is2;\n var sm13 = mat[2] * is3;\n var sm21 = mat[4] * is1;\n var sm22 = mat[5] * is2;\n var sm23 = mat[6] * is3;\n var sm31 = mat[8] * is1;\n var sm32 = mat[9] * is2;\n var sm33 = mat[10] * is3;\n var trace = sm11 + sm22 + sm33;\n var S = 0;\n\n if (trace > 0) {\n S = Math.sqrt(trace + 1.0) * 2;\n out[3] = 0.25 * S;\n out[0] = (sm23 - sm32) / S;\n out[1] = (sm31 - sm13) / S;\n out[2] = (sm12 - sm21) / S;\n } else if (sm11 > sm22 && sm11 > sm33) {\n S = Math.sqrt(1.0 + sm11 - sm22 - sm33) * 2;\n out[3] = (sm23 - sm32) / S;\n out[0] = 0.25 * S;\n out[1] = (sm12 + sm21) / S;\n out[2] = (sm31 + sm13) / S;\n } else if (sm22 > sm33) {\n S = Math.sqrt(1.0 + sm22 - sm11 - sm33) * 2;\n out[3] = (sm31 - sm13) / S;\n out[0] = (sm12 + sm21) / S;\n out[1] = 0.25 * S;\n out[2] = (sm23 + sm32) / S;\n } else {\n S = Math.sqrt(1.0 + sm33 - sm11 - sm22) * 2;\n out[3] = (sm12 - sm21) / S;\n out[0] = (sm31 + sm13) / S;\n out[1] = (sm23 + sm32) / S;\n out[2] = 0.25 * S;\n }\n\n return out;\n}\n/**\n * Creates a matrix from a quaternion rotation, vector translation and vector scale\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, vec);\n * let quatMat = mat4.create();\n * quat4.toMat4(quat, quatMat);\n * mat4.multiply(dest, quatMat);\n * mat4.scale(dest, scale)\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {ReadonlyVec3} v Translation vector\n * @param {ReadonlyVec3} s Scaling vector\n * @returns {mat4} out\n */\n\nexport function fromRotationTranslationScale(out, q, v, s) {\n // Quaternion math\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var xy = x * y2;\n var xz = x * z2;\n var yy = y * y2;\n var yz = y * z2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n var sx = s[0];\n var sy = s[1];\n var sz = s[2];\n out[0] = (1 - (yy + zz)) * sx;\n out[1] = (xy + wz) * sx;\n out[2] = (xz - wy) * sx;\n out[3] = 0;\n out[4] = (xy - wz) * sy;\n out[5] = (1 - (xx + zz)) * sy;\n out[6] = (yz + wx) * sy;\n out[7] = 0;\n out[8] = (xz + wy) * sz;\n out[9] = (yz - wx) * sz;\n out[10] = (1 - (xx + yy)) * sz;\n out[11] = 0;\n out[12] = v[0];\n out[13] = v[1];\n out[14] = v[2];\n out[15] = 1;\n return out;\n}\n/**\n * Creates a matrix from a quaternion rotation, vector translation and vector scale, rotating and scaling around the given origin\n * This is equivalent to (but much faster than):\n *\n * mat4.identity(dest);\n * mat4.translate(dest, vec);\n * mat4.translate(dest, origin);\n * let quatMat = mat4.create();\n * quat4.toMat4(quat, quatMat);\n * mat4.multiply(dest, quatMat);\n * mat4.scale(dest, scale)\n * mat4.translate(dest, negativeOrigin);\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {quat4} q Rotation quaternion\n * @param {ReadonlyVec3} v Translation vector\n * @param {ReadonlyVec3} s Scaling vector\n * @param {ReadonlyVec3} o The origin vector around which to scale and rotate\n * @returns {mat4} out\n */\n\nexport function fromRotationTranslationScaleOrigin(out, q, v, s, o) {\n // Quaternion math\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var xy = x * y2;\n var xz = x * z2;\n var yy = y * y2;\n var yz = y * z2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n var sx = s[0];\n var sy = s[1];\n var sz = s[2];\n var ox = o[0];\n var oy = o[1];\n var oz = o[2];\n var out0 = (1 - (yy + zz)) * sx;\n var out1 = (xy + wz) * sx;\n var out2 = (xz - wy) * sx;\n var out4 = (xy - wz) * sy;\n var out5 = (1 - (xx + zz)) * sy;\n var out6 = (yz + wx) * sy;\n var out8 = (xz + wy) * sz;\n var out9 = (yz - wx) * sz;\n var out10 = (1 - (xx + yy)) * sz;\n out[0] = out0;\n out[1] = out1;\n out[2] = out2;\n out[3] = 0;\n out[4] = out4;\n out[5] = out5;\n out[6] = out6;\n out[7] = 0;\n out[8] = out8;\n out[9] = out9;\n out[10] = out10;\n out[11] = 0;\n out[12] = v[0] + ox - (out0 * ox + out4 * oy + out8 * oz);\n out[13] = v[1] + oy - (out1 * ox + out5 * oy + out9 * oz);\n out[14] = v[2] + oz - (out2 * ox + out6 * oy + out10 * oz);\n out[15] = 1;\n return out;\n}\n/**\n * Calculates a 4x4 matrix from the given quaternion\n *\n * @param {mat4} out mat4 receiving operation result\n * @param {ReadonlyQuat} q Quaternion to create matrix from\n *\n * @returns {mat4} out\n */\n\nexport function fromQuat(out, q) {\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var yx = y * x2;\n var yy = y * y2;\n var zx = z * x2;\n var zy = z * y2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n out[0] = 1 - yy - zz;\n out[1] = yx + wz;\n out[2] = zx - wy;\n out[3] = 0;\n out[4] = yx - wz;\n out[5] = 1 - xx - zz;\n out[6] = zy + wx;\n out[7] = 0;\n out[8] = zx + wy;\n out[9] = zy - wx;\n out[10] = 1 - xx - yy;\n out[11] = 0;\n out[12] = 0;\n out[13] = 0;\n out[14] = 0;\n out[15] = 1;\n return out;\n}\n/**\n * Generates a frustum matrix with the given bounds\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {Number} left Left bound of the frustum\n * @param {Number} right Right bound of the frustum\n * @param {Number} bottom Bottom bound of the frustum\n * @param {Number} top Top bound of the frustum\n * @param {Number} near Near bound of the frustum\n * @param {Number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function frustum(out, left, right, bottom, top, near, far) {\n var rl = 1 / (right - left);\n var tb = 1 / (top - bottom);\n var nf = 1 / (near - far);\n out[0] = near * 2 * rl;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = near * 2 * tb;\n out[6] = 0;\n out[7] = 0;\n out[8] = (right + left) * rl;\n out[9] = (top + bottom) * tb;\n out[10] = (far + near) * nf;\n out[11] = -1;\n out[12] = 0;\n out[13] = 0;\n out[14] = far * near * 2 * nf;\n out[15] = 0;\n return out;\n}\n/**\n * Generates a perspective projection matrix with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [-1, 1],\n * which matches WebGL/OpenGL's clip volume.\n * Passing null/undefined/no value for far will generate infinite projection matrix.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} fovy Vertical field of view in radians\n * @param {number} aspect Aspect ratio. typically viewport width/height\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum, can be null or Infinity\n * @returns {mat4} out\n */\n\nexport function perspectiveNO(out, fovy, aspect, near, far) {\n var f = 1.0 / Math.tan(fovy / 2),\n nf;\n out[0] = f / aspect;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = f;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[11] = -1;\n out[12] = 0;\n out[13] = 0;\n out[15] = 0;\n\n if (far != null && far !== Infinity) {\n nf = 1 / (near - far);\n out[10] = (far + near) * nf;\n out[14] = 2 * far * near * nf;\n } else {\n out[10] = -1;\n out[14] = -2 * near;\n }\n\n return out;\n}\n/**\n * Alias for {@link mat4.perspectiveNO}\n * @function\n */\n\nexport var perspective = perspectiveNO;\n/**\n * Generates a perspective projection matrix suitable for WebGPU with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [0, 1],\n * which matches WebGPU/Vulkan/DirectX/Metal's clip volume.\n * Passing null/undefined/no value for far will generate infinite projection matrix.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} fovy Vertical field of view in radians\n * @param {number} aspect Aspect ratio. typically viewport width/height\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum, can be null or Infinity\n * @returns {mat4} out\n */\n\nexport function perspectiveZO(out, fovy, aspect, near, far) {\n var f = 1.0 / Math.tan(fovy / 2),\n nf;\n out[0] = f / aspect;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = f;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[11] = -1;\n out[12] = 0;\n out[13] = 0;\n out[15] = 0;\n\n if (far != null && far !== Infinity) {\n nf = 1 / (near - far);\n out[10] = far * nf;\n out[14] = far * near * nf;\n } else {\n out[10] = -1;\n out[14] = -near;\n }\n\n return out;\n}\n/**\n * Generates a perspective projection matrix with the given field of view.\n * This is primarily useful for generating projection matrices to be used\n * with the still experiemental WebVR API.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {Object} fov Object containing the following values: upDegrees, downDegrees, leftDegrees, rightDegrees\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function perspectiveFromFieldOfView(out, fov, near, far) {\n var upTan = Math.tan(fov.upDegrees * Math.PI / 180.0);\n var downTan = Math.tan(fov.downDegrees * Math.PI / 180.0);\n var leftTan = Math.tan(fov.leftDegrees * Math.PI / 180.0);\n var rightTan = Math.tan(fov.rightDegrees * Math.PI / 180.0);\n var xScale = 2.0 / (leftTan + rightTan);\n var yScale = 2.0 / (upTan + downTan);\n out[0] = xScale;\n out[1] = 0.0;\n out[2] = 0.0;\n out[3] = 0.0;\n out[4] = 0.0;\n out[5] = yScale;\n out[6] = 0.0;\n out[7] = 0.0;\n out[8] = -((leftTan - rightTan) * xScale * 0.5);\n out[9] = (upTan - downTan) * yScale * 0.5;\n out[10] = far / (near - far);\n out[11] = -1.0;\n out[12] = 0.0;\n out[13] = 0.0;\n out[14] = far * near / (near - far);\n out[15] = 0.0;\n return out;\n}\n/**\n * Generates a orthogonal projection matrix with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [-1, 1],\n * which matches WebGL/OpenGL's clip volume.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} left Left bound of the frustum\n * @param {number} right Right bound of the frustum\n * @param {number} bottom Bottom bound of the frustum\n * @param {number} top Top bound of the frustum\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function orthoNO(out, left, right, bottom, top, near, far) {\n var lr = 1 / (left - right);\n var bt = 1 / (bottom - top);\n var nf = 1 / (near - far);\n out[0] = -2 * lr;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = -2 * bt;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = 2 * nf;\n out[11] = 0;\n out[12] = (left + right) * lr;\n out[13] = (top + bottom) * bt;\n out[14] = (far + near) * nf;\n out[15] = 1;\n return out;\n}\n/**\n * Alias for {@link mat4.orthoNO}\n * @function\n */\n\nexport var ortho = orthoNO;\n/**\n * Generates a orthogonal projection matrix with the given bounds.\n * The near/far clip planes correspond to a normalized device coordinate Z range of [0, 1],\n * which matches WebGPU/Vulkan/DirectX/Metal's clip volume.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {number} left Left bound of the frustum\n * @param {number} right Right bound of the frustum\n * @param {number} bottom Bottom bound of the frustum\n * @param {number} top Top bound of the frustum\n * @param {number} near Near bound of the frustum\n * @param {number} far Far bound of the frustum\n * @returns {mat4} out\n */\n\nexport function orthoZO(out, left, right, bottom, top, near, far) {\n var lr = 1 / (left - right);\n var bt = 1 / (bottom - top);\n var nf = 1 / (near - far);\n out[0] = -2 * lr;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 0;\n out[5] = -2 * bt;\n out[6] = 0;\n out[7] = 0;\n out[8] = 0;\n out[9] = 0;\n out[10] = nf;\n out[11] = 0;\n out[12] = (left + right) * lr;\n out[13] = (top + bottom) * bt;\n out[14] = near * nf;\n out[15] = 1;\n return out;\n}\n/**\n * Generates a look-at matrix with the given eye position, focal point, and up axis.\n * If you want a matrix that actually makes an object look at another object, you should use targetTo instead.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {ReadonlyVec3} eye Position of the viewer\n * @param {ReadonlyVec3} center Point the viewer is looking at\n * @param {ReadonlyVec3} up vec3 pointing up\n * @returns {mat4} out\n */\n\nexport function lookAt(out, eye, center, up) {\n var x0, x1, x2, y0, y1, y2, z0, z1, z2, len;\n var eyex = eye[0];\n var eyey = eye[1];\n var eyez = eye[2];\n var upx = up[0];\n var upy = up[1];\n var upz = up[2];\n var centerx = center[0];\n var centery = center[1];\n var centerz = center[2];\n\n if (Math.abs(eyex - centerx) < glMatrix.EPSILON && Math.abs(eyey - centery) < glMatrix.EPSILON && Math.abs(eyez - centerz) < glMatrix.EPSILON) {\n return identity(out);\n }\n\n z0 = eyex - centerx;\n z1 = eyey - centery;\n z2 = eyez - centerz;\n len = 1 / Math.hypot(z0, z1, z2);\n z0 *= len;\n z1 *= len;\n z2 *= len;\n x0 = upy * z2 - upz * z1;\n x1 = upz * z0 - upx * z2;\n x2 = upx * z1 - upy * z0;\n len = Math.hypot(x0, x1, x2);\n\n if (!len) {\n x0 = 0;\n x1 = 0;\n x2 = 0;\n } else {\n len = 1 / len;\n x0 *= len;\n x1 *= len;\n x2 *= len;\n }\n\n y0 = z1 * x2 - z2 * x1;\n y1 = z2 * x0 - z0 * x2;\n y2 = z0 * x1 - z1 * x0;\n len = Math.hypot(y0, y1, y2);\n\n if (!len) {\n y0 = 0;\n y1 = 0;\n y2 = 0;\n } else {\n len = 1 / len;\n y0 *= len;\n y1 *= len;\n y2 *= len;\n }\n\n out[0] = x0;\n out[1] = y0;\n out[2] = z0;\n out[3] = 0;\n out[4] = x1;\n out[5] = y1;\n out[6] = z1;\n out[7] = 0;\n out[8] = x2;\n out[9] = y2;\n out[10] = z2;\n out[11] = 0;\n out[12] = -(x0 * eyex + x1 * eyey + x2 * eyez);\n out[13] = -(y0 * eyex + y1 * eyey + y2 * eyez);\n out[14] = -(z0 * eyex + z1 * eyey + z2 * eyez);\n out[15] = 1;\n return out;\n}\n/**\n * Generates a matrix that makes something look at something else.\n *\n * @param {mat4} out mat4 frustum matrix will be written into\n * @param {ReadonlyVec3} eye Position of the viewer\n * @param {ReadonlyVec3} center Point the viewer is looking at\n * @param {ReadonlyVec3} up vec3 pointing up\n * @returns {mat4} out\n */\n\nexport function targetTo(out, eye, target, up) {\n var eyex = eye[0],\n eyey = eye[1],\n eyez = eye[2],\n upx = up[0],\n upy = up[1],\n upz = up[2];\n var z0 = eyex - target[0],\n z1 = eyey - target[1],\n z2 = eyez - target[2];\n var len = z0 * z0 + z1 * z1 + z2 * z2;\n\n if (len > 0) {\n len = 1 / Math.sqrt(len);\n z0 *= len;\n z1 *= len;\n z2 *= len;\n }\n\n var x0 = upy * z2 - upz * z1,\n x1 = upz * z0 - upx * z2,\n x2 = upx * z1 - upy * z0;\n len = x0 * x0 + x1 * x1 + x2 * x2;\n\n if (len > 0) {\n len = 1 / Math.sqrt(len);\n x0 *= len;\n x1 *= len;\n x2 *= len;\n }\n\n out[0] = x0;\n out[1] = x1;\n out[2] = x2;\n out[3] = 0;\n out[4] = z1 * x2 - z2 * x1;\n out[5] = z2 * x0 - z0 * x2;\n out[6] = z0 * x1 - z1 * x0;\n out[7] = 0;\n out[8] = z0;\n out[9] = z1;\n out[10] = z2;\n out[11] = 0;\n out[12] = eyex;\n out[13] = eyey;\n out[14] = eyez;\n out[15] = 1;\n return out;\n}\n/**\n * Returns a string representation of a mat4\n *\n * @param {ReadonlyMat4} a matrix to represent as a string\n * @returns {String} string representation of the matrix\n */\n\nexport function str(a) {\n return \"mat4(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \", \" + a[4] + \", \" + a[5] + \", \" + a[6] + \", \" + a[7] + \", \" + a[8] + \", \" + a[9] + \", \" + a[10] + \", \" + a[11] + \", \" + a[12] + \", \" + a[13] + \", \" + a[14] + \", \" + a[15] + \")\";\n}\n/**\n * Returns Frobenius norm of a mat4\n *\n * @param {ReadonlyMat4} a the matrix to calculate Frobenius norm of\n * @returns {Number} Frobenius norm\n */\n\nexport function frob(a) {\n return Math.hypot(a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8], a[9], a[10], a[11], a[12], a[13], a[14], a[15]);\n}\n/**\n * Adds two mat4's\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @returns {mat4} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n out[4] = a[4] + b[4];\n out[5] = a[5] + b[5];\n out[6] = a[6] + b[6];\n out[7] = a[7] + b[7];\n out[8] = a[8] + b[8];\n out[9] = a[9] + b[9];\n out[10] = a[10] + b[10];\n out[11] = a[11] + b[11];\n out[12] = a[12] + b[12];\n out[13] = a[13] + b[13];\n out[14] = a[14] + b[14];\n out[15] = a[15] + b[15];\n return out;\n}\n/**\n * Subtracts matrix b from matrix a\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @returns {mat4} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n out[4] = a[4] - b[4];\n out[5] = a[5] - b[5];\n out[6] = a[6] - b[6];\n out[7] = a[7] - b[7];\n out[8] = a[8] - b[8];\n out[9] = a[9] - b[9];\n out[10] = a[10] - b[10];\n out[11] = a[11] - b[11];\n out[12] = a[12] - b[12];\n out[13] = a[13] - b[13];\n out[14] = a[14] - b[14];\n out[15] = a[15] - b[15];\n return out;\n}\n/**\n * Multiply each element of the matrix by a scalar.\n *\n * @param {mat4} out the receiving matrix\n * @param {ReadonlyMat4} a the matrix to scale\n * @param {Number} b amount to scale the matrix's elements by\n * @returns {mat4} out\n */\n\nexport function multiplyScalar(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n out[4] = a[4] * b;\n out[5] = a[5] * b;\n out[6] = a[6] * b;\n out[7] = a[7] * b;\n out[8] = a[8] * b;\n out[9] = a[9] * b;\n out[10] = a[10] * b;\n out[11] = a[11] * b;\n out[12] = a[12] * b;\n out[13] = a[13] * b;\n out[14] = a[14] * b;\n out[15] = a[15] * b;\n return out;\n}\n/**\n * Adds two mat4's after multiplying each element of the second operand by a scalar value.\n *\n * @param {mat4} out the receiving vector\n * @param {ReadonlyMat4} a the first operand\n * @param {ReadonlyMat4} b the second operand\n * @param {Number} scale the amount to scale b's elements by before adding\n * @returns {mat4} out\n */\n\nexport function multiplyScalarAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n out[4] = a[4] + b[4] * scale;\n out[5] = a[5] + b[5] * scale;\n out[6] = a[6] + b[6] * scale;\n out[7] = a[7] + b[7] * scale;\n out[8] = a[8] + b[8] * scale;\n out[9] = a[9] + b[9] * scale;\n out[10] = a[10] + b[10] * scale;\n out[11] = a[11] + b[11] * scale;\n out[12] = a[12] + b[12] * scale;\n out[13] = a[13] + b[13] * scale;\n out[14] = a[14] + b[14] * scale;\n out[15] = a[15] + b[15] * scale;\n return out;\n}\n/**\n * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyMat4} a The first matrix.\n * @param {ReadonlyMat4} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5] && a[6] === b[6] && a[7] === b[7] && a[8] === b[8] && a[9] === b[9] && a[10] === b[10] && a[11] === b[11] && a[12] === b[12] && a[13] === b[13] && a[14] === b[14] && a[15] === b[15];\n}\n/**\n * Returns whether or not the matrices have approximately the same elements in the same position.\n *\n * @param {ReadonlyMat4} a The first matrix.\n * @param {ReadonlyMat4} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var a4 = a[4],\n a5 = a[5],\n a6 = a[6],\n a7 = a[7];\n var a8 = a[8],\n a9 = a[9],\n a10 = a[10],\n a11 = a[11];\n var a12 = a[12],\n a13 = a[13],\n a14 = a[14],\n a15 = a[15];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n var b4 = b[4],\n b5 = b[5],\n b6 = b[6],\n b7 = b[7];\n var b8 = b[8],\n b9 = b[9],\n b10 = b[10],\n b11 = b[11];\n var b12 = b[12],\n b13 = b[13],\n b14 = b[14],\n b15 = b[15];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5)) && Math.abs(a6 - b6) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a6), Math.abs(b6)) && Math.abs(a7 - b7) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a7), Math.abs(b7)) && Math.abs(a8 - b8) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a8), Math.abs(b8)) && Math.abs(a9 - b9) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a9), Math.abs(b9)) && Math.abs(a10 - b10) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a10), Math.abs(b10)) && Math.abs(a11 - b11) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a11), Math.abs(b11)) && Math.abs(a12 - b12) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a12), Math.abs(b12)) && Math.abs(a13 - b13) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a13), Math.abs(b13)) && Math.abs(a14 - b14) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a14), Math.abs(b14)) && Math.abs(a15 - b15) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a15), Math.abs(b15));\n}\n/**\n * Alias for {@link mat4.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link mat4.subtract}\n * @function\n */\n\nexport var sub = subtract;","import * as glMatrix from \"./common.js\";\n/**\n * 4 Dimensional Vector\n * @module vec4\n */\n\n/**\n * Creates a new, empty vec4\n *\n * @returns {vec4} a new 4D vector\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(4);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[0] = 0;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n }\n\n return out;\n}\n/**\n * Creates a new vec4 initialized with values from an existing vector\n *\n * @param {ReadonlyVec4} a vector to clone\n * @returns {vec4} a new 4D vector\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Creates a new vec4 initialized with the given values\n *\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @param {Number} w W component\n * @returns {vec4} a new 4D vector\n */\n\nexport function fromValues(x, y, z, w) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = x;\n out[1] = y;\n out[2] = z;\n out[3] = w;\n return out;\n}\n/**\n * Copy the values from one vec4 to another\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the source vector\n * @returns {vec4} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Set the components of a vec4 to the given values\n *\n * @param {vec4} out the receiving vector\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @param {Number} w W component\n * @returns {vec4} out\n */\n\nexport function set(out, x, y, z, w) {\n out[0] = x;\n out[1] = y;\n out[2] = z;\n out[3] = w;\n return out;\n}\n/**\n * Adds two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n return out;\n}\n/**\n * Subtracts vector b from vector a\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n return out;\n}\n/**\n * Multiplies two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function multiply(out, a, b) {\n out[0] = a[0] * b[0];\n out[1] = a[1] * b[1];\n out[2] = a[2] * b[2];\n out[3] = a[3] * b[3];\n return out;\n}\n/**\n * Divides two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function divide(out, a, b) {\n out[0] = a[0] / b[0];\n out[1] = a[1] / b[1];\n out[2] = a[2] / b[2];\n out[3] = a[3] / b[3];\n return out;\n}\n/**\n * Math.ceil the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to ceil\n * @returns {vec4} out\n */\n\nexport function ceil(out, a) {\n out[0] = Math.ceil(a[0]);\n out[1] = Math.ceil(a[1]);\n out[2] = Math.ceil(a[2]);\n out[3] = Math.ceil(a[3]);\n return out;\n}\n/**\n * Math.floor the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to floor\n * @returns {vec4} out\n */\n\nexport function floor(out, a) {\n out[0] = Math.floor(a[0]);\n out[1] = Math.floor(a[1]);\n out[2] = Math.floor(a[2]);\n out[3] = Math.floor(a[3]);\n return out;\n}\n/**\n * Returns the minimum of two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function min(out, a, b) {\n out[0] = Math.min(a[0], b[0]);\n out[1] = Math.min(a[1], b[1]);\n out[2] = Math.min(a[2], b[2]);\n out[3] = Math.min(a[3], b[3]);\n return out;\n}\n/**\n * Returns the maximum of two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {vec4} out\n */\n\nexport function max(out, a, b) {\n out[0] = Math.max(a[0], b[0]);\n out[1] = Math.max(a[1], b[1]);\n out[2] = Math.max(a[2], b[2]);\n out[3] = Math.max(a[3], b[3]);\n return out;\n}\n/**\n * Math.round the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to round\n * @returns {vec4} out\n */\n\nexport function round(out, a) {\n out[0] = Math.round(a[0]);\n out[1] = Math.round(a[1]);\n out[2] = Math.round(a[2]);\n out[3] = Math.round(a[3]);\n return out;\n}\n/**\n * Scales a vec4 by a scalar number\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the vector to scale\n * @param {Number} b amount to scale the vector by\n * @returns {vec4} out\n */\n\nexport function scale(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n return out;\n}\n/**\n * Adds two vec4's after scaling the second operand by a scalar value\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @param {Number} scale the amount to scale b by before adding\n * @returns {vec4} out\n */\n\nexport function scaleAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n return out;\n}\n/**\n * Calculates the euclidian distance between two vec4's\n *\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {Number} distance between a and b\n */\n\nexport function distance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n var w = b[3] - a[3];\n return Math.hypot(x, y, z, w);\n}\n/**\n * Calculates the squared euclidian distance between two vec4's\n *\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {Number} squared distance between a and b\n */\n\nexport function squaredDistance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n var w = b[3] - a[3];\n return x * x + y * y + z * z + w * w;\n}\n/**\n * Calculates the length of a vec4\n *\n * @param {ReadonlyVec4} a vector to calculate length of\n * @returns {Number} length of a\n */\n\nexport function length(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var w = a[3];\n return Math.hypot(x, y, z, w);\n}\n/**\n * Calculates the squared length of a vec4\n *\n * @param {ReadonlyVec4} a vector to calculate squared length of\n * @returns {Number} squared length of a\n */\n\nexport function squaredLength(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var w = a[3];\n return x * x + y * y + z * z + w * w;\n}\n/**\n * Negates the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to negate\n * @returns {vec4} out\n */\n\nexport function negate(out, a) {\n out[0] = -a[0];\n out[1] = -a[1];\n out[2] = -a[2];\n out[3] = -a[3];\n return out;\n}\n/**\n * Returns the inverse of the components of a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to invert\n * @returns {vec4} out\n */\n\nexport function inverse(out, a) {\n out[0] = 1.0 / a[0];\n out[1] = 1.0 / a[1];\n out[2] = 1.0 / a[2];\n out[3] = 1.0 / a[3];\n return out;\n}\n/**\n * Normalize a vec4\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a vector to normalize\n * @returns {vec4} out\n */\n\nexport function normalize(out, a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var w = a[3];\n var len = x * x + y * y + z * z + w * w;\n\n if (len > 0) {\n len = 1 / Math.sqrt(len);\n }\n\n out[0] = x * len;\n out[1] = y * len;\n out[2] = z * len;\n out[3] = w * len;\n return out;\n}\n/**\n * Calculates the dot product of two vec4's\n *\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @returns {Number} dot product of a and b\n */\n\nexport function dot(a, b) {\n return a[0] * b[0] + a[1] * b[1] + a[2] * b[2] + a[3] * b[3];\n}\n/**\n * Returns the cross-product of three vectors in a 4-dimensional space\n *\n * @param {ReadonlyVec4} result the receiving vector\n * @param {ReadonlyVec4} U the first vector\n * @param {ReadonlyVec4} V the second vector\n * @param {ReadonlyVec4} W the third vector\n * @returns {vec4} result\n */\n\nexport function cross(out, u, v, w) {\n var A = v[0] * w[1] - v[1] * w[0],\n B = v[0] * w[2] - v[2] * w[0],\n C = v[0] * w[3] - v[3] * w[0],\n D = v[1] * w[2] - v[2] * w[1],\n E = v[1] * w[3] - v[3] * w[1],\n F = v[2] * w[3] - v[3] * w[2];\n var G = u[0];\n var H = u[1];\n var I = u[2];\n var J = u[3];\n out[0] = H * F - I * E + J * D;\n out[1] = -(G * F) + I * C - J * B;\n out[2] = G * E - H * C + J * A;\n out[3] = -(G * D) + H * B - I * A;\n return out;\n}\n/**\n * Performs a linear interpolation between two vec4's\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the first operand\n * @param {ReadonlyVec4} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec4} out\n */\n\nexport function lerp(out, a, b, t) {\n var ax = a[0];\n var ay = a[1];\n var az = a[2];\n var aw = a[3];\n out[0] = ax + t * (b[0] - ax);\n out[1] = ay + t * (b[1] - ay);\n out[2] = az + t * (b[2] - az);\n out[3] = aw + t * (b[3] - aw);\n return out;\n}\n/**\n * Generates a random vector with the given scale\n *\n * @param {vec4} out the receiving vector\n * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned\n * @returns {vec4} out\n */\n\nexport function random(out, scale) {\n scale = scale || 1.0; // Marsaglia, George. Choosing a Point from the Surface of a\n // Sphere. Ann. Math. Statist. 43 (1972), no. 2, 645--646.\n // http://projecteuclid.org/euclid.aoms/1177692644;\n\n var v1, v2, v3, v4;\n var s1, s2;\n\n do {\n v1 = glMatrix.RANDOM() * 2 - 1;\n v2 = glMatrix.RANDOM() * 2 - 1;\n s1 = v1 * v1 + v2 * v2;\n } while (s1 >= 1);\n\n do {\n v3 = glMatrix.RANDOM() * 2 - 1;\n v4 = glMatrix.RANDOM() * 2 - 1;\n s2 = v3 * v3 + v4 * v4;\n } while (s2 >= 1);\n\n var d = Math.sqrt((1 - s1) / s2);\n out[0] = scale * v1;\n out[1] = scale * v2;\n out[2] = scale * v3 * d;\n out[3] = scale * v4 * d;\n return out;\n}\n/**\n * Transforms the vec4 with a mat4.\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the vector to transform\n * @param {ReadonlyMat4} m matrix to transform with\n * @returns {vec4} out\n */\n\nexport function transformMat4(out, a, m) {\n var x = a[0],\n y = a[1],\n z = a[2],\n w = a[3];\n out[0] = m[0] * x + m[4] * y + m[8] * z + m[12] * w;\n out[1] = m[1] * x + m[5] * y + m[9] * z + m[13] * w;\n out[2] = m[2] * x + m[6] * y + m[10] * z + m[14] * w;\n out[3] = m[3] * x + m[7] * y + m[11] * z + m[15] * w;\n return out;\n}\n/**\n * Transforms the vec4 with a quat\n *\n * @param {vec4} out the receiving vector\n * @param {ReadonlyVec4} a the vector to transform\n * @param {ReadonlyQuat} q quaternion to transform with\n * @returns {vec4} out\n */\n\nexport function transformQuat(out, a, q) {\n var x = a[0],\n y = a[1],\n z = a[2];\n var qx = q[0],\n qy = q[1],\n qz = q[2],\n qw = q[3]; // calculate quat * vec\n\n var ix = qw * x + qy * z - qz * y;\n var iy = qw * y + qz * x - qx * z;\n var iz = qw * z + qx * y - qy * x;\n var iw = -qx * x - qy * y - qz * z; // calculate result * inverse quat\n\n out[0] = ix * qw + iw * -qx + iy * -qz - iz * -qy;\n out[1] = iy * qw + iw * -qy + iz * -qx - ix * -qz;\n out[2] = iz * qw + iw * -qz + ix * -qy - iy * -qx;\n out[3] = a[3];\n return out;\n}\n/**\n * Set the components of a vec4 to zero\n *\n * @param {vec4} out the receiving vector\n * @returns {vec4} out\n */\n\nexport function zero(out) {\n out[0] = 0.0;\n out[1] = 0.0;\n out[2] = 0.0;\n out[3] = 0.0;\n return out;\n}\n/**\n * Returns a string representation of a vector\n *\n * @param {ReadonlyVec4} a vector to represent as a string\n * @returns {String} string representation of the vector\n */\n\nexport function str(a) {\n return \"vec4(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \")\";\n}\n/**\n * Returns whether or not the vectors have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyVec4} a The first vector.\n * @param {ReadonlyVec4} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3];\n}\n/**\n * Returns whether or not the vectors have approximately the same elements in the same position.\n *\n * @param {ReadonlyVec4} a The first vector.\n * @param {ReadonlyVec4} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3));\n}\n/**\n * Alias for {@link vec4.subtract}\n * @function\n */\n\nexport var sub = subtract;\n/**\n * Alias for {@link vec4.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link vec4.divide}\n * @function\n */\n\nexport var div = divide;\n/**\n * Alias for {@link vec4.distance}\n * @function\n */\n\nexport var dist = distance;\n/**\n * Alias for {@link vec4.squaredDistance}\n * @function\n */\n\nexport var sqrDist = squaredDistance;\n/**\n * Alias for {@link vec4.length}\n * @function\n */\n\nexport var len = length;\n/**\n * Alias for {@link vec4.squaredLength}\n * @function\n */\n\nexport var sqrLen = squaredLength;\n/**\n * Perform some operation over an array of vec4s.\n *\n * @param {Array} a the array of vectors to iterate over\n * @param {Number} stride Number of elements between the start of each vec4. If 0 assumes tightly packed\n * @param {Number} offset Number of elements to skip at the beginning of the array\n * @param {Number} count Number of vec4s to iterate over. If 0 iterates over entire array\n * @param {Function} fn Function to call for each vector in the array\n * @param {Object} [arg] additional argument to pass to fn\n * @returns {Array} a\n * @function\n */\n\nexport var forEach = function () {\n var vec = create();\n return function (a, stride, offset, count, fn, arg) {\n var i, l;\n\n if (!stride) {\n stride = 4;\n }\n\n if (!offset) {\n offset = 0;\n }\n\n if (count) {\n l = Math.min(count * stride + offset, a.length);\n } else {\n l = a.length;\n }\n\n for (i = offset; i < l; i += stride) {\n vec[0] = a[i];\n vec[1] = a[i + 1];\n vec[2] = a[i + 2];\n vec[3] = a[i + 3];\n fn(vec, vec, arg);\n a[i] = vec[0];\n a[i + 1] = vec[1];\n a[i + 2] = vec[2];\n a[i + 3] = vec[3];\n }\n\n return a;\n };\n}();","import {StyleLayer} from '../style_layer';\n\nimport {CircleBucket} from '../../data/bucket/circle_bucket';\nimport {polygonIntersectsBufferedPoint} from '../../util/intersection_tests';\nimport {getMaximumPaintValue, translateDistance, translate} from '../query_utils';\nimport properties, {CircleLayoutPropsPossiblyEvaluated, CirclePaintPropsPossiblyEvaluated} from './circle_style_layer_properties.g';\nimport {Transitionable, Transitioning, Layout, PossiblyEvaluated} from '../properties';\nimport {mat4, vec4} from 'gl-matrix';\nimport Point from '@mapbox/point-geometry';\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Transform} from '../../geo/transform';\nimport type {Bucket, BucketParameters} from '../../data/bucket';\nimport type {CircleLayoutProps, CirclePaintProps} from './circle_style_layer_properties.g';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\n/**\n * A style layer that defines a circle\n */\nexport class CircleStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n createBucket(parameters: BucketParameters) {\n return new CircleBucket(parameters);\n }\n\n queryRadius(bucket: Bucket): number {\n const circleBucket: CircleBucket = (bucket as any);\n return getMaximumPaintValue('circle-radius', this, circleBucket) +\n getMaximumPaintValue('circle-stroke-width', this, circleBucket) +\n translateDistance(this.paint.get('circle-translate'));\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number,\n pixelPosMatrix: mat4\n ): boolean {\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('circle-translate'),\n this.paint.get('circle-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n const radius = this.paint.get('circle-radius').evaluate(feature, featureState);\n const stroke = this.paint.get('circle-stroke-width').evaluate(feature, featureState);\n const size = radius + stroke;\n\n // For pitch-alignment: map, compare feature geometry to query geometry in the plane of the tile\n // // Otherwise, compare geometry in the plane of the viewport\n // // A circle with fixed scaling relative to the viewport gets larger in tile space as it moves into the distance\n // // A circle with fixed scaling relative to the map gets smaller in viewport space as it moves into the distance\n const alignWithMap = this.paint.get('circle-pitch-alignment') === 'map';\n const transformedPolygon = alignWithMap ? translatedPolygon : projectQueryGeometry(translatedPolygon, pixelPosMatrix);\n const transformedSize = alignWithMap ? size * pixelsToTileUnits : size;\n\n for (const ring of geometry) {\n for (const point of ring) {\n\n const transformedPoint = alignWithMap ? point : projectPoint(point, pixelPosMatrix);\n\n let adjustedSize = transformedSize;\n const projectedCenter = vec4.transformMat4([] as any, [point.x, point.y, 0, 1], pixelPosMatrix);\n if (this.paint.get('circle-pitch-scale') === 'viewport' && this.paint.get('circle-pitch-alignment') === 'map') {\n adjustedSize *= projectedCenter[3] / transform.cameraToCenterDistance;\n } else if (this.paint.get('circle-pitch-scale') === 'map' && this.paint.get('circle-pitch-alignment') === 'viewport') {\n adjustedSize *= transform.cameraToCenterDistance / projectedCenter[3];\n }\n\n if (polygonIntersectsBufferedPoint(transformedPolygon, transformedPoint, adjustedSize)) return true;\n }\n }\n\n return false;\n }\n}\n\nfunction projectPoint(p: Point, pixelPosMatrix: mat4) {\n const point = vec4.transformMat4([] as any, [p.x, p.y, 0, 1], pixelPosMatrix);\n return new Point(point[0] / point[3], point[1] / point[3]);\n}\n\nfunction projectQueryGeometry(queryGeometry: Array, pixelPosMatrix: mat4) {\n return queryGeometry.map((p) => {\n return projectPoint(p, pixelPosMatrix);\n });\n}\n","import {CircleBucket} from './circle_bucket';\nimport {register} from '../../util/web_worker_transfer';\n\nimport type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer';\n\nexport class HeatmapBucket extends CircleBucket {\n // Needed for flow to accept omit: ['layers'] below, due to\n // https://github.com/facebook/flow/issues/4262\n layers: Array;\n}\n\nregister('HeatmapBucket', HeatmapBucket, {omit: ['layers']});\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type HeatmapPaintProps = {\n \"heatmap-radius\": DataDrivenProperty,\n \"heatmap-weight\": DataDrivenProperty,\n \"heatmap-intensity\": DataConstantProperty,\n \"heatmap-color\": ColorRampProperty,\n \"heatmap-opacity\": DataConstantProperty,\n};\n\nexport type HeatmapPaintPropsPossiblyEvaluated = {\n \"heatmap-radius\": PossiblyEvaluatedPropertyValue,\n \"heatmap-weight\": PossiblyEvaluatedPropertyValue,\n \"heatmap-intensity\": number,\n \"heatmap-color\": ColorRampProperty,\n \"heatmap-opacity\": number,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"heatmap-radius\": new DataDrivenProperty(styleSpec[\"paint_heatmap\"][\"heatmap-radius\"] as any as StylePropertySpecification),\n \"heatmap-weight\": new DataDrivenProperty(styleSpec[\"paint_heatmap\"][\"heatmap-weight\"] as any as StylePropertySpecification),\n \"heatmap-intensity\": new DataConstantProperty(styleSpec[\"paint_heatmap\"][\"heatmap-intensity\"] as any as StylePropertySpecification),\n \"heatmap-color\": new ColorRampProperty(styleSpec[\"paint_heatmap\"][\"heatmap-color\"] as any as StylePropertySpecification),\n \"heatmap-opacity\": new DataConstantProperty(styleSpec[\"paint_heatmap\"][\"heatmap-opacity\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {register} from './web_worker_transfer';\n\nexport type Size = {\n width: number;\n height: number;\n};\n\ntype Point2D = {\n x: number;\n y: number;\n};\n\nfunction createImage(image: any, {\n width,\n height\n}: Size, channels: number, data?: Uint8Array | Uint8ClampedArray) {\n if (!data) {\n data = new Uint8Array(width * height * channels);\n } else if (data instanceof Uint8ClampedArray) {\n data = new Uint8Array(data.buffer);\n } else if (data.length !== width * height * channels) {\n throw new RangeError(`mismatched image size. expected: ${data.length} but got: ${width * height * channels}`);\n }\n image.width = width;\n image.height = height;\n image.data = data;\n return image;\n}\n\nfunction resizeImage(image: any, {\n width,\n height\n}: Size, channels: number) {\n if (width === image.width && height === image.height) {\n return;\n }\n\n const newImage = createImage({}, {width, height}, channels);\n\n copyImage(image, newImage, {x: 0, y: 0}, {x: 0, y: 0}, {\n width: Math.min(image.width, width),\n height: Math.min(image.height, height)\n }, channels);\n\n image.width = width;\n image.height = height;\n image.data = newImage.data;\n}\n\nfunction copyImage(srcImg: any, dstImg: any, srcPt: Point2D, dstPt: Point2D, size: Size, channels: number) {\n if (size.width === 0 || size.height === 0) {\n return dstImg;\n }\n\n if (size.width > srcImg.width ||\n size.height > srcImg.height ||\n srcPt.x > srcImg.width - size.width ||\n srcPt.y > srcImg.height - size.height) {\n throw new RangeError('out of range source coordinates for image copy');\n }\n\n if (size.width > dstImg.width ||\n size.height > dstImg.height ||\n dstPt.x > dstImg.width - size.width ||\n dstPt.y > dstImg.height - size.height) {\n throw new RangeError('out of range destination coordinates for image copy');\n }\n\n const srcData = srcImg.data;\n const dstData = dstImg.data;\n\n if (srcData === dstData) throw new Error('srcData equals dstData, so image is already copied');\n\n for (let y = 0; y < size.height; y++) {\n const srcOffset = ((srcPt.y + y) * srcImg.width + srcPt.x) * channels;\n const dstOffset = ((dstPt.y + y) * dstImg.width + dstPt.x) * channels;\n for (let i = 0; i < size.width * channels; i++) {\n dstData[dstOffset + i] = srcData[srcOffset + i];\n }\n }\n return dstImg;\n}\n\n/**\n * @internal\n * An image with alpha color value\n */\nexport class AlphaImage {\n width: number;\n height: number;\n data: Uint8Array;\n\n constructor(size: Size, data?: Uint8Array | Uint8ClampedArray) {\n createImage(this, size, 1, data);\n }\n\n resize(size: Size) {\n resizeImage(this, size, 1);\n }\n\n clone() {\n return new AlphaImage({width: this.width, height: this.height}, new Uint8Array(this.data));\n }\n\n static copy(srcImg: AlphaImage, dstImg: AlphaImage, srcPt: Point2D, dstPt: Point2D, size: Size) {\n copyImage(srcImg, dstImg, srcPt, dstPt, size, 1);\n }\n}\n\n/**\n * An object to store image data not premultiplied, because ImageData is not premultiplied.\n * UNPACK_PREMULTIPLY_ALPHA_WEBGL must be used when uploading to a texture.\n */\nexport class RGBAImage {\n width: number;\n height: number;\n\n /**\n * data must be a Uint8Array instead of Uint8ClampedArray because texImage2D does not support Uint8ClampedArray in all browsers.\n */\n data: Uint8Array;\n\n constructor(size: Size, data?: Uint8Array | Uint8ClampedArray) {\n createImage(this, size, 4, data);\n }\n\n resize(size: Size) {\n resizeImage(this, size, 4);\n }\n\n replace(data: Uint8Array | Uint8ClampedArray, copy?: boolean) {\n if (copy) {\n this.data.set(data);\n } else if (data instanceof Uint8ClampedArray) {\n this.data = new Uint8Array(data.buffer);\n } else {\n this.data = data;\n }\n }\n\n clone() {\n return new RGBAImage({width: this.width, height: this.height}, new Uint8Array(this.data));\n }\n\n static copy(srcImg: RGBAImage | ImageData, dstImg: RGBAImage, srcPt: Point2D, dstPt: Point2D, size: Size) {\n copyImage(srcImg, dstImg, srcPt, dstPt, size, 4);\n }\n}\n\nregister('AlphaImage', AlphaImage);\nregister('RGBAImage', RGBAImage);\n","import {RGBAImage} from './image';\nimport {isPowerOfTwo} from './util';\n\nimport type {StylePropertyExpression} from '@maplibre/maplibre-gl-style-spec';\n\nexport type ColorRampParams = {\n expression: StylePropertyExpression;\n evaluationKey: string;\n resolution?: number;\n image?: RGBAImage;\n clips?: Array;\n};\n\n/**\n * Given an expression that should evaluate to a color ramp,\n * return a RGBA image representing that ramp expression.\n */\nexport function renderColorRamp(params: ColorRampParams): RGBAImage {\n const evaluationGlobals = {};\n const width = params.resolution || 256;\n const height = params.clips ? params.clips.length : 1;\n const image = params.image || new RGBAImage({width, height});\n\n if (!isPowerOfTwo(width)) throw new Error(`width is not a power of 2 - ${width}`);\n\n const renderPixel = (stride, index, progress) => {\n evaluationGlobals[params.evaluationKey] = progress;\n const pxColor = params.expression.evaluate(evaluationGlobals as any);\n // the colors are being unpremultiplied because Color uses\n // premultiplied values, and the Texture class expects unpremultiplied ones\n image.data[stride + index + 0] = Math.floor(pxColor.r * 255 / pxColor.a);\n image.data[stride + index + 1] = Math.floor(pxColor.g * 255 / pxColor.a);\n image.data[stride + index + 2] = Math.floor(pxColor.b * 255 / pxColor.a);\n image.data[stride + index + 3] = Math.floor(pxColor.a * 255);\n };\n\n if (!params.clips) {\n for (let i = 0, j = 0; i < width; i++, j += 4) {\n const progress = i / (width - 1);\n\n renderPixel(0, j, progress);\n }\n } else {\n for (let clip = 0, stride = 0; clip < height; ++clip, stride += width * 4) {\n for (let i = 0, j = 0; i < width; i++, j += 4) {\n // Remap progress between clips\n const progress = i / (width - 1);\n const {start, end} = params.clips[clip];\n const evaluationProgress = start * (1 - progress) + end * progress;\n renderPixel(stride, j, evaluationProgress);\n }\n }\n }\n\n return image;\n}\n","import {StyleLayer} from '../style_layer';\n\nimport {HeatmapBucket} from '../../data/bucket/heatmap_bucket';\nimport {RGBAImage} from '../../util/image';\nimport properties, {HeatmapPaintPropsPossiblyEvaluated} from './heatmap_style_layer_properties.g';\nimport {renderColorRamp} from '../../util/color_ramp';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {Texture} from '../../render/texture';\nimport type {Framebuffer} from '../../gl/framebuffer';\nimport type {HeatmapPaintProps} from './heatmap_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A style layer that defines a heatmap\n */\nexport class HeatmapStyleLayer extends StyleLayer {\n\n heatmapFbo: Framebuffer;\n colorRamp: RGBAImage;\n colorRampTexture: Texture;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n createBucket(options: any) {\n return new HeatmapBucket(options);\n }\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n\n // make sure color ramp texture is generated for default heatmap color too\n this._updateColorRamp();\n }\n\n _handleSpecialPaintPropertyUpdate(name: string) {\n if (name === 'heatmap-color') {\n this._updateColorRamp();\n }\n }\n\n _updateColorRamp() {\n const expression = this._transitionablePaint._values['heatmap-color'].value.expression;\n this.colorRamp = renderColorRamp({\n expression,\n evaluationKey: 'heatmapDensity',\n image: this.colorRamp\n });\n this.colorRampTexture = null;\n }\n\n resize() {\n if (this.heatmapFbo) {\n this.heatmapFbo.destroy();\n this.heatmapFbo = null;\n }\n }\n\n queryRadius(): number {\n return 0;\n }\n\n queryIntersectsFeature(): boolean {\n return false;\n }\n\n hasOffscreenPass() {\n return this.paint.get('heatmap-opacity') !== 0 && this.visibility !== 'none';\n }\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type HillshadePaintProps = {\n \"hillshade-illumination-direction\": DataConstantProperty,\n \"hillshade-illumination-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"hillshade-exaggeration\": DataConstantProperty,\n \"hillshade-shadow-color\": DataConstantProperty,\n \"hillshade-highlight-color\": DataConstantProperty,\n \"hillshade-accent-color\": DataConstantProperty,\n};\n\nexport type HillshadePaintPropsPossiblyEvaluated = {\n \"hillshade-illumination-direction\": number,\n \"hillshade-illumination-anchor\": \"map\" | \"viewport\",\n \"hillshade-exaggeration\": number,\n \"hillshade-shadow-color\": Color,\n \"hillshade-highlight-color\": Color,\n \"hillshade-accent-color\": Color,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"hillshade-illumination-direction\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-illumination-direction\"] as any as StylePropertySpecification),\n \"hillshade-illumination-anchor\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-illumination-anchor\"] as any as StylePropertySpecification),\n \"hillshade-exaggeration\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-exaggeration\"] as any as StylePropertySpecification),\n \"hillshade-shadow-color\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-shadow-color\"] as any as StylePropertySpecification),\n \"hillshade-highlight-color\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-highlight-color\"] as any as StylePropertySpecification),\n \"hillshade-accent-color\": new DataConstantProperty(styleSpec[\"paint_hillshade\"][\"hillshade-accent-color\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport properties, {HillshadePaintPropsPossiblyEvaluated} from './hillshade_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {HillshadePaintProps} from './hillshade_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport class HillshadeStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n hasOffscreenPass() {\n return this.paint.get('hillshade-exaggeration') !== 0 && this.visibility !== 'none';\n }\n}\n","import {createLayout} from '../../util/struct_array';\n\nconst layout = createLayout([\n {name: 'a_pos', components: 2, type: 'Int16'}\n], 4);\n\nexport default layout;\nexport const {members, size, alignment} = layout;\n","'use strict';\n\nmodule.exports = earcut;\nmodule.exports.default = earcut;\n\nfunction earcut(data, holeIndices, dim) {\n\n dim = dim || 2;\n\n var hasHoles = holeIndices && holeIndices.length,\n outerLen = hasHoles ? holeIndices[0] * dim : data.length,\n outerNode = linkedList(data, 0, outerLen, dim, true),\n triangles = [];\n\n if (!outerNode || outerNode.next === outerNode.prev) return triangles;\n\n var minX, minY, maxX, maxY, x, y, invSize;\n\n if (hasHoles) outerNode = eliminateHoles(data, holeIndices, outerNode, dim);\n\n // if the shape is not too simple, we'll use z-order curve hash later; calculate polygon bbox\n if (data.length > 80 * dim) {\n minX = maxX = data[0];\n minY = maxY = data[1];\n\n for (var i = dim; i < outerLen; i += dim) {\n x = data[i];\n y = data[i + 1];\n if (x < minX) minX = x;\n if (y < minY) minY = y;\n if (x > maxX) maxX = x;\n if (y > maxY) maxY = y;\n }\n\n // minX, minY and invSize are later used to transform coords into integers for z-order calculation\n invSize = Math.max(maxX - minX, maxY - minY);\n invSize = invSize !== 0 ? 32767 / invSize : 0;\n }\n\n earcutLinked(outerNode, triangles, dim, minX, minY, invSize, 0);\n\n return triangles;\n}\n\n// create a circular doubly linked list from polygon points in the specified winding order\nfunction linkedList(data, start, end, dim, clockwise) {\n var i, last;\n\n if (clockwise === (signedArea(data, start, end, dim) > 0)) {\n for (i = start; i < end; i += dim) last = insertNode(i, data[i], data[i + 1], last);\n } else {\n for (i = end - dim; i >= start; i -= dim) last = insertNode(i, data[i], data[i + 1], last);\n }\n\n if (last && equals(last, last.next)) {\n removeNode(last);\n last = last.next;\n }\n\n return last;\n}\n\n// eliminate colinear or duplicate points\nfunction filterPoints(start, end) {\n if (!start) return start;\n if (!end) end = start;\n\n var p = start,\n again;\n do {\n again = false;\n\n if (!p.steiner && (equals(p, p.next) || area(p.prev, p, p.next) === 0)) {\n removeNode(p);\n p = end = p.prev;\n if (p === p.next) break;\n again = true;\n\n } else {\n p = p.next;\n }\n } while (again || p !== end);\n\n return end;\n}\n\n// main ear slicing loop which triangulates a polygon (given as a linked list)\nfunction earcutLinked(ear, triangles, dim, minX, minY, invSize, pass) {\n if (!ear) return;\n\n // interlink polygon nodes in z-order\n if (!pass && invSize) indexCurve(ear, minX, minY, invSize);\n\n var stop = ear,\n prev, next;\n\n // iterate through ears, slicing them one by one\n while (ear.prev !== ear.next) {\n prev = ear.prev;\n next = ear.next;\n\n if (invSize ? isEarHashed(ear, minX, minY, invSize) : isEar(ear)) {\n // cut off the triangle\n triangles.push(prev.i / dim | 0);\n triangles.push(ear.i / dim | 0);\n triangles.push(next.i / dim | 0);\n\n removeNode(ear);\n\n // skipping the next vertex leads to less sliver triangles\n ear = next.next;\n stop = next.next;\n\n continue;\n }\n\n ear = next;\n\n // if we looped through the whole remaining polygon and can't find any more ears\n if (ear === stop) {\n // try filtering points and slicing again\n if (!pass) {\n earcutLinked(filterPoints(ear), triangles, dim, minX, minY, invSize, 1);\n\n // if this didn't work, try curing all small self-intersections locally\n } else if (pass === 1) {\n ear = cureLocalIntersections(filterPoints(ear), triangles, dim);\n earcutLinked(ear, triangles, dim, minX, minY, invSize, 2);\n\n // as a last resort, try splitting the remaining polygon into two\n } else if (pass === 2) {\n splitEarcut(ear, triangles, dim, minX, minY, invSize);\n }\n\n break;\n }\n }\n}\n\n// check whether a polygon node forms a valid ear with adjacent nodes\nfunction isEar(ear) {\n var a = ear.prev,\n b = ear,\n c = ear.next;\n\n if (area(a, b, c) >= 0) return false; // reflex, can't be an ear\n\n // now make sure we don't have other points inside the potential ear\n var ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y;\n\n // triangle bbox; min & max are calculated like this for speed\n var x0 = ax < bx ? (ax < cx ? ax : cx) : (bx < cx ? bx : cx),\n y0 = ay < by ? (ay < cy ? ay : cy) : (by < cy ? by : cy),\n x1 = ax > bx ? (ax > cx ? ax : cx) : (bx > cx ? bx : cx),\n y1 = ay > by ? (ay > cy ? ay : cy) : (by > cy ? by : cy);\n\n var p = c.next;\n while (p !== a) {\n if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) &&\n area(p.prev, p, p.next) >= 0) return false;\n p = p.next;\n }\n\n return true;\n}\n\nfunction isEarHashed(ear, minX, minY, invSize) {\n var a = ear.prev,\n b = ear,\n c = ear.next;\n\n if (area(a, b, c) >= 0) return false; // reflex, can't be an ear\n\n var ax = a.x, bx = b.x, cx = c.x, ay = a.y, by = b.y, cy = c.y;\n\n // triangle bbox; min & max are calculated like this for speed\n var x0 = ax < bx ? (ax < cx ? ax : cx) : (bx < cx ? bx : cx),\n y0 = ay < by ? (ay < cy ? ay : cy) : (by < cy ? by : cy),\n x1 = ax > bx ? (ax > cx ? ax : cx) : (bx > cx ? bx : cx),\n y1 = ay > by ? (ay > cy ? ay : cy) : (by > cy ? by : cy);\n\n // z-order range for the current triangle bbox;\n var minZ = zOrder(x0, y0, minX, minY, invSize),\n maxZ = zOrder(x1, y1, minX, minY, invSize);\n\n var p = ear.prevZ,\n n = ear.nextZ;\n\n // look for points inside the triangle in both directions\n while (p && p.z >= minZ && n && n.z <= maxZ) {\n if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 && p !== a && p !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) && area(p.prev, p, p.next) >= 0) return false;\n p = p.prevZ;\n\n if (n.x >= x0 && n.x <= x1 && n.y >= y0 && n.y <= y1 && n !== a && n !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) && area(n.prev, n, n.next) >= 0) return false;\n n = n.nextZ;\n }\n\n // look for remaining points in decreasing z-order\n while (p && p.z >= minZ) {\n if (p.x >= x0 && p.x <= x1 && p.y >= y0 && p.y <= y1 && p !== a && p !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, p.x, p.y) && area(p.prev, p, p.next) >= 0) return false;\n p = p.prevZ;\n }\n\n // look for remaining points in increasing z-order\n while (n && n.z <= maxZ) {\n if (n.x >= x0 && n.x <= x1 && n.y >= y0 && n.y <= y1 && n !== a && n !== c &&\n pointInTriangle(ax, ay, bx, by, cx, cy, n.x, n.y) && area(n.prev, n, n.next) >= 0) return false;\n n = n.nextZ;\n }\n\n return true;\n}\n\n// go through all polygon nodes and cure small local self-intersections\nfunction cureLocalIntersections(start, triangles, dim) {\n var p = start;\n do {\n var a = p.prev,\n b = p.next.next;\n\n if (!equals(a, b) && intersects(a, p, p.next, b) && locallyInside(a, b) && locallyInside(b, a)) {\n\n triangles.push(a.i / dim | 0);\n triangles.push(p.i / dim | 0);\n triangles.push(b.i / dim | 0);\n\n // remove two nodes involved\n removeNode(p);\n removeNode(p.next);\n\n p = start = b;\n }\n p = p.next;\n } while (p !== start);\n\n return filterPoints(p);\n}\n\n// try splitting polygon into two and triangulate them independently\nfunction splitEarcut(start, triangles, dim, minX, minY, invSize) {\n // look for a valid diagonal that divides the polygon into two\n var a = start;\n do {\n var b = a.next.next;\n while (b !== a.prev) {\n if (a.i !== b.i && isValidDiagonal(a, b)) {\n // split the polygon in two by the diagonal\n var c = splitPolygon(a, b);\n\n // filter colinear points around the cuts\n a = filterPoints(a, a.next);\n c = filterPoints(c, c.next);\n\n // run earcut on each half\n earcutLinked(a, triangles, dim, minX, minY, invSize, 0);\n earcutLinked(c, triangles, dim, minX, minY, invSize, 0);\n return;\n }\n b = b.next;\n }\n a = a.next;\n } while (a !== start);\n}\n\n// link every hole into the outer loop, producing a single-ring polygon without holes\nfunction eliminateHoles(data, holeIndices, outerNode, dim) {\n var queue = [],\n i, len, start, end, list;\n\n for (i = 0, len = holeIndices.length; i < len; i++) {\n start = holeIndices[i] * dim;\n end = i < len - 1 ? holeIndices[i + 1] * dim : data.length;\n list = linkedList(data, start, end, dim, false);\n if (list === list.next) list.steiner = true;\n queue.push(getLeftmost(list));\n }\n\n queue.sort(compareX);\n\n // process holes from left to right\n for (i = 0; i < queue.length; i++) {\n outerNode = eliminateHole(queue[i], outerNode);\n }\n\n return outerNode;\n}\n\nfunction compareX(a, b) {\n return a.x - b.x;\n}\n\n// find a bridge between vertices that connects hole with an outer ring and and link it\nfunction eliminateHole(hole, outerNode) {\n var bridge = findHoleBridge(hole, outerNode);\n if (!bridge) {\n return outerNode;\n }\n\n var bridgeReverse = splitPolygon(bridge, hole);\n\n // filter collinear points around the cuts\n filterPoints(bridgeReverse, bridgeReverse.next);\n return filterPoints(bridge, bridge.next);\n}\n\n// David Eberly's algorithm for finding a bridge between hole and outer polygon\nfunction findHoleBridge(hole, outerNode) {\n var p = outerNode,\n hx = hole.x,\n hy = hole.y,\n qx = -Infinity,\n m;\n\n // find a segment intersected by a ray from the hole's leftmost point to the left;\n // segment's endpoint with lesser x will be potential connection point\n do {\n if (hy <= p.y && hy >= p.next.y && p.next.y !== p.y) {\n var x = p.x + (hy - p.y) * (p.next.x - p.x) / (p.next.y - p.y);\n if (x <= hx && x > qx) {\n qx = x;\n m = p.x < p.next.x ? p : p.next;\n if (x === hx) return m; // hole touches outer segment; pick leftmost endpoint\n }\n }\n p = p.next;\n } while (p !== outerNode);\n\n if (!m) return null;\n\n // look for points inside the triangle of hole point, segment intersection and endpoint;\n // if there are no points found, we have a valid connection;\n // otherwise choose the point of the minimum angle with the ray as connection point\n\n var stop = m,\n mx = m.x,\n my = m.y,\n tanMin = Infinity,\n tan;\n\n p = m;\n\n do {\n if (hx >= p.x && p.x >= mx && hx !== p.x &&\n pointInTriangle(hy < my ? hx : qx, hy, mx, my, hy < my ? qx : hx, hy, p.x, p.y)) {\n\n tan = Math.abs(hy - p.y) / (hx - p.x); // tangential\n\n if (locallyInside(p, hole) &&\n (tan < tanMin || (tan === tanMin && (p.x > m.x || (p.x === m.x && sectorContainsSector(m, p)))))) {\n m = p;\n tanMin = tan;\n }\n }\n\n p = p.next;\n } while (p !== stop);\n\n return m;\n}\n\n// whether sector in vertex m contains sector in vertex p in the same coordinates\nfunction sectorContainsSector(m, p) {\n return area(m.prev, m, p.prev) < 0 && area(p.next, m, m.next) < 0;\n}\n\n// interlink polygon nodes in z-order\nfunction indexCurve(start, minX, minY, invSize) {\n var p = start;\n do {\n if (p.z === 0) p.z = zOrder(p.x, p.y, minX, minY, invSize);\n p.prevZ = p.prev;\n p.nextZ = p.next;\n p = p.next;\n } while (p !== start);\n\n p.prevZ.nextZ = null;\n p.prevZ = null;\n\n sortLinked(p);\n}\n\n// Simon Tatham's linked list merge sort algorithm\n// http://www.chiark.greenend.org.uk/~sgtatham/algorithms/listsort.html\nfunction sortLinked(list) {\n var i, p, q, e, tail, numMerges, pSize, qSize,\n inSize = 1;\n\n do {\n p = list;\n list = null;\n tail = null;\n numMerges = 0;\n\n while (p) {\n numMerges++;\n q = p;\n pSize = 0;\n for (i = 0; i < inSize; i++) {\n pSize++;\n q = q.nextZ;\n if (!q) break;\n }\n qSize = inSize;\n\n while (pSize > 0 || (qSize > 0 && q)) {\n\n if (pSize !== 0 && (qSize === 0 || !q || p.z <= q.z)) {\n e = p;\n p = p.nextZ;\n pSize--;\n } else {\n e = q;\n q = q.nextZ;\n qSize--;\n }\n\n if (tail) tail.nextZ = e;\n else list = e;\n\n e.prevZ = tail;\n tail = e;\n }\n\n p = q;\n }\n\n tail.nextZ = null;\n inSize *= 2;\n\n } while (numMerges > 1);\n\n return list;\n}\n\n// z-order of a point given coords and inverse of the longer side of data bbox\nfunction zOrder(x, y, minX, minY, invSize) {\n // coords are transformed into non-negative 15-bit integer range\n x = (x - minX) * invSize | 0;\n y = (y - minY) * invSize | 0;\n\n x = (x | (x << 8)) & 0x00FF00FF;\n x = (x | (x << 4)) & 0x0F0F0F0F;\n x = (x | (x << 2)) & 0x33333333;\n x = (x | (x << 1)) & 0x55555555;\n\n y = (y | (y << 8)) & 0x00FF00FF;\n y = (y | (y << 4)) & 0x0F0F0F0F;\n y = (y | (y << 2)) & 0x33333333;\n y = (y | (y << 1)) & 0x55555555;\n\n return x | (y << 1);\n}\n\n// find the leftmost node of a polygon ring\nfunction getLeftmost(start) {\n var p = start,\n leftmost = start;\n do {\n if (p.x < leftmost.x || (p.x === leftmost.x && p.y < leftmost.y)) leftmost = p;\n p = p.next;\n } while (p !== start);\n\n return leftmost;\n}\n\n// check if a point lies within a convex triangle\nfunction pointInTriangle(ax, ay, bx, by, cx, cy, px, py) {\n return (cx - px) * (ay - py) >= (ax - px) * (cy - py) &&\n (ax - px) * (by - py) >= (bx - px) * (ay - py) &&\n (bx - px) * (cy - py) >= (cx - px) * (by - py);\n}\n\n// check if a diagonal between two polygon nodes is valid (lies in polygon interior)\nfunction isValidDiagonal(a, b) {\n return a.next.i !== b.i && a.prev.i !== b.i && !intersectsPolygon(a, b) && // dones't intersect other edges\n (locallyInside(a, b) && locallyInside(b, a) && middleInside(a, b) && // locally visible\n (area(a.prev, a, b.prev) || area(a, b.prev, b)) || // does not create opposite-facing sectors\n equals(a, b) && area(a.prev, a, a.next) > 0 && area(b.prev, b, b.next) > 0); // special zero-length case\n}\n\n// signed area of a triangle\nfunction area(p, q, r) {\n return (q.y - p.y) * (r.x - q.x) - (q.x - p.x) * (r.y - q.y);\n}\n\n// check if two points are equal\nfunction equals(p1, p2) {\n return p1.x === p2.x && p1.y === p2.y;\n}\n\n// check if two segments intersect\nfunction intersects(p1, q1, p2, q2) {\n var o1 = sign(area(p1, q1, p2));\n var o2 = sign(area(p1, q1, q2));\n var o3 = sign(area(p2, q2, p1));\n var o4 = sign(area(p2, q2, q1));\n\n if (o1 !== o2 && o3 !== o4) return true; // general case\n\n if (o1 === 0 && onSegment(p1, p2, q1)) return true; // p1, q1 and p2 are collinear and p2 lies on p1q1\n if (o2 === 0 && onSegment(p1, q2, q1)) return true; // p1, q1 and q2 are collinear and q2 lies on p1q1\n if (o3 === 0 && onSegment(p2, p1, q2)) return true; // p2, q2 and p1 are collinear and p1 lies on p2q2\n if (o4 === 0 && onSegment(p2, q1, q2)) return true; // p2, q2 and q1 are collinear and q1 lies on p2q2\n\n return false;\n}\n\n// for collinear points p, q, r, check if point q lies on segment pr\nfunction onSegment(p, q, r) {\n return q.x <= Math.max(p.x, r.x) && q.x >= Math.min(p.x, r.x) && q.y <= Math.max(p.y, r.y) && q.y >= Math.min(p.y, r.y);\n}\n\nfunction sign(num) {\n return num > 0 ? 1 : num < 0 ? -1 : 0;\n}\n\n// check if a polygon diagonal intersects any polygon segments\nfunction intersectsPolygon(a, b) {\n var p = a;\n do {\n if (p.i !== a.i && p.next.i !== a.i && p.i !== b.i && p.next.i !== b.i &&\n intersects(p, p.next, a, b)) return true;\n p = p.next;\n } while (p !== a);\n\n return false;\n}\n\n// check if a polygon diagonal is locally inside the polygon\nfunction locallyInside(a, b) {\n return area(a.prev, a, a.next) < 0 ?\n area(a, b, a.next) >= 0 && area(a, a.prev, b) >= 0 :\n area(a, b, a.prev) < 0 || area(a, a.next, b) < 0;\n}\n\n// check if the middle point of a polygon diagonal is inside the polygon\nfunction middleInside(a, b) {\n var p = a,\n inside = false,\n px = (a.x + b.x) / 2,\n py = (a.y + b.y) / 2;\n do {\n if (((p.y > py) !== (p.next.y > py)) && p.next.y !== p.y &&\n (px < (p.next.x - p.x) * (py - p.y) / (p.next.y - p.y) + p.x))\n inside = !inside;\n p = p.next;\n } while (p !== a);\n\n return inside;\n}\n\n// link two polygon vertices with a bridge; if the vertices belong to the same ring, it splits polygon into two;\n// if one belongs to the outer ring and another to a hole, it merges it into a single ring\nfunction splitPolygon(a, b) {\n var a2 = new Node(a.i, a.x, a.y),\n b2 = new Node(b.i, b.x, b.y),\n an = a.next,\n bp = b.prev;\n\n a.next = b;\n b.prev = a;\n\n a2.next = an;\n an.prev = a2;\n\n b2.next = a2;\n a2.prev = b2;\n\n bp.next = b2;\n b2.prev = bp;\n\n return b2;\n}\n\n// create a node and optionally link it with previous one (in a circular doubly linked list)\nfunction insertNode(i, x, y, last) {\n var p = new Node(i, x, y);\n\n if (!last) {\n p.prev = p;\n p.next = p;\n\n } else {\n p.next = last.next;\n p.prev = last;\n last.next.prev = p;\n last.next = p;\n }\n return p;\n}\n\nfunction removeNode(p) {\n p.next.prev = p.prev;\n p.prev.next = p.next;\n\n if (p.prevZ) p.prevZ.nextZ = p.nextZ;\n if (p.nextZ) p.nextZ.prevZ = p.prevZ;\n}\n\nfunction Node(i, x, y) {\n // vertex index in coordinates array\n this.i = i;\n\n // vertex coordinates\n this.x = x;\n this.y = y;\n\n // previous and next vertex nodes in a polygon ring\n this.prev = null;\n this.next = null;\n\n // z-order curve value\n this.z = 0;\n\n // previous and next nodes in z-order\n this.prevZ = null;\n this.nextZ = null;\n\n // indicates whether this is a steiner point\n this.steiner = false;\n}\n\n// return a percentage difference between the polygon area and its triangulation area;\n// used to verify correctness of triangulation\nearcut.deviation = function (data, holeIndices, dim, triangles) {\n var hasHoles = holeIndices && holeIndices.length;\n var outerLen = hasHoles ? holeIndices[0] * dim : data.length;\n\n var polygonArea = Math.abs(signedArea(data, 0, outerLen, dim));\n if (hasHoles) {\n for (var i = 0, len = holeIndices.length; i < len; i++) {\n var start = holeIndices[i] * dim;\n var end = i < len - 1 ? holeIndices[i + 1] * dim : data.length;\n polygonArea -= Math.abs(signedArea(data, start, end, dim));\n }\n }\n\n var trianglesArea = 0;\n for (i = 0; i < triangles.length; i += 3) {\n var a = triangles[i] * dim;\n var b = triangles[i + 1] * dim;\n var c = triangles[i + 2] * dim;\n trianglesArea += Math.abs(\n (data[a] - data[c]) * (data[b + 1] - data[a + 1]) -\n (data[a] - data[b]) * (data[c + 1] - data[a + 1]));\n }\n\n return polygonArea === 0 && trianglesArea === 0 ? 0 :\n Math.abs((trianglesArea - polygonArea) / polygonArea);\n};\n\nfunction signedArea(data, start, end, dim) {\n var sum = 0;\n for (var i = start, j = end - dim; i < end; i += dim) {\n sum += (data[j] - data[i]) * (data[i + 1] + data[j + 1]);\n j = i;\n }\n return sum;\n}\n\n// turn a polygon in a multi-dimensional array form (e.g. as in GeoJSON) into a form Earcut accepts\nearcut.flatten = function (data) {\n var dim = data[0][0].length,\n result = {vertices: [], holes: [], dimensions: dim},\n holeIndex = 0;\n\n for (var i = 0; i < data.length; i++) {\n for (var j = 0; j < data[i].length; j++) {\n for (var d = 0; d < dim; d++) result.vertices.push(data[i][j][d]);\n }\n if (i > 0) {\n holeIndex += data[i - 1].length;\n result.holes.push(holeIndex);\n }\n }\n return result;\n};\n","\nexport default function quickselect(arr, k, left, right, compare) {\n quickselectStep(arr, k, left || 0, right || (arr.length - 1), compare || defaultCompare);\n}\n\nfunction quickselectStep(arr, k, left, right, compare) {\n\n while (right > left) {\n if (right - left > 600) {\n var n = right - left + 1;\n var m = k - left + 1;\n var z = Math.log(n);\n var s = 0.5 * Math.exp(2 * z / 3);\n var sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);\n var newLeft = Math.max(left, Math.floor(k - m * s / n + sd));\n var newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));\n quickselectStep(arr, k, newLeft, newRight, compare);\n }\n\n var t = arr[k];\n var i = left;\n var j = right;\n\n swap(arr, left, k);\n if (compare(arr[right], t) > 0) swap(arr, left, right);\n\n while (i < j) {\n swap(arr, i, j);\n i++;\n j--;\n while (compare(arr[i], t) < 0) i++;\n while (compare(arr[j], t) > 0) j--;\n }\n\n if (compare(arr[left], t) === 0) swap(arr, left, j);\n else {\n j++;\n swap(arr, j, right);\n }\n\n if (j <= k) left = j + 1;\n if (k <= j) right = j - 1;\n }\n}\n\nfunction swap(arr, i, j) {\n var tmp = arr[i];\n arr[i] = arr[j];\n arr[j] = tmp;\n}\n\nfunction defaultCompare(a, b) {\n return a < b ? -1 : a > b ? 1 : 0;\n}\n","import quickselect from 'quickselect';\n\nimport {calculateSignedArea} from './util';\n\nimport type Point from '@mapbox/point-geometry';\n\n// classifies an array of rings into polygons with outer rings and holes\nexport function classifyRings(rings: Array>, maxRings: number) {\n const len = rings.length;\n\n if (len <= 1) return [rings];\n\n const polygons = [];\n let polygon,\n ccw;\n\n for (let i = 0; i < len; i++) {\n const area = calculateSignedArea(rings[i]);\n if (area === 0) continue;\n\n (rings[i] as any).area = Math.abs(area);\n\n if (ccw === undefined) ccw = area < 0;\n\n if (ccw === area < 0) {\n if (polygon) polygons.push(polygon);\n polygon = [rings[i]];\n\n } else {\n (polygon as any).push(rings[i]);\n }\n }\n if (polygon) polygons.push(polygon);\n\n // Earcut performance degrades with the # of rings in a polygon. For this\n // reason, we limit strip out all but the `maxRings` largest rings.\n if (maxRings > 1) {\n for (let j = 0; j < polygons.length; j++) {\n if (polygons[j].length <= maxRings) continue;\n quickselect(polygons[j], maxRings, 1, polygons[j].length - 1, compareAreas);\n polygons[j] = polygons[j].slice(0, maxRings);\n }\n }\n\n return polygons;\n}\n\nfunction compareAreas(a, b) {\n return b.area - a.area;\n}\n","import type {FillStyleLayer} from '../../style/style_layer/fill_style_layer';\nimport type {FillExtrusionStyleLayer} from '../../style/style_layer/fill_extrusion_style_layer';\nimport type {LineStyleLayer} from '../../style/style_layer/line_style_layer';\n\nimport type {\n BucketFeature,\n PopulateParameters\n} from '../bucket';\nimport {PossiblyEvaluated} from '../../style/properties';\n\ntype PatternStyleLayers = Array | Array | Array;\n\nexport function hasPattern(type: string, layers: PatternStyleLayers, options: PopulateParameters) {\n const patterns = options.patternDependencies;\n let hasPattern = false;\n\n for (const layer of layers) {\n const patternProperty = (layer.paint as PossiblyEvaluated).get(`${type}-pattern`);\n if (!patternProperty.isConstant()) {\n hasPattern = true;\n }\n\n const constantPattern = patternProperty.constantOr(null);\n if (constantPattern) {\n hasPattern = true;\n patterns[constantPattern.to] = true;\n patterns[constantPattern.from] = true;\n }\n }\n\n return hasPattern;\n}\n\nexport function addPatternDependencies(type: string, layers: PatternStyleLayers, patternFeature: BucketFeature, zoom: number, options: PopulateParameters) {\n const patterns = options.patternDependencies;\n for (const layer of layers) {\n const patternProperty = (layer.paint as PossiblyEvaluated).get(`${type}-pattern`);\n\n const patternPropertyValue = patternProperty.value;\n if (patternPropertyValue.kind !== 'constant') {\n let min = patternPropertyValue.evaluate({zoom: zoom - 1}, patternFeature, {}, options.availableImages);\n let mid = patternPropertyValue.evaluate({zoom}, patternFeature, {}, options.availableImages);\n let max = patternPropertyValue.evaluate({zoom: zoom + 1}, patternFeature, {}, options.availableImages);\n min = min && min.name ? min.name : min;\n mid = mid && mid.name ? mid.name : mid;\n max = max && max.name ? max.name : max;\n // add to patternDependencies\n patterns[min] = true;\n patterns[mid] = true;\n patterns[max] = true;\n\n // save for layout\n patternFeature.patterns[layer.id] = {min, mid, max};\n }\n }\n return patternFeature;\n}\n","import {FillLayoutArray} from '../array_types.g';\n\nimport {members as layoutAttributes} from './fill_attributes';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {LineIndexArray, TriangleIndexArray} from '../index_array_type';\nimport earcut from 'earcut';\nimport {classifyRings} from '../../util/classify_rings';\nconst EARCUT_MAX_RINGS = 500;\nimport {register} from '../../util/web_worker_transfer';\nimport {hasPattern, addPatternDependencies} from './pattern_bucket_features';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {FillStyleLayer} from '../../style/style_layer/fill_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nexport class FillBucket implements Bucket {\n index: number;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n patternFeatures: Array;\n\n layoutVertexArray: FillLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n indexArray2: LineIndexArray;\n indexBuffer2: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n segments2: SegmentVector;\n uploaded: boolean;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n this.patternFeatures = [];\n\n this.layoutVertexArray = new FillLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.indexArray2 = new LineIndexArray();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.segments = new SegmentVector();\n this.segments2 = new SegmentVector();\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n this.hasPattern = hasPattern('fill', this.layers, options);\n const fillSortKey = this.layers[0].layout.get('fill-sort-key');\n const sortFeaturesByKey = !fillSortKey.isConstant();\n const bucketFeatures: BucketFeature[] = [];\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const sortKey = sortFeaturesByKey ?\n fillSortKey.evaluate(evaluationFeature, {}, canonical, options.availableImages) :\n undefined;\n\n const bucketFeature: BucketFeature = {\n id,\n properties: feature.properties,\n type: feature.type,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n patterns: {},\n sortKey\n };\n\n bucketFeatures.push(bucketFeature);\n }\n\n if (sortFeaturesByKey) {\n bucketFeatures.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const bucketFeature of bucketFeatures) {\n const {geometry, index, sourceLayerIndex} = bucketFeature;\n\n if (this.hasPattern) {\n const patternFeature = addPatternDependencies('fill', this.layers, bucketFeature, this.zoom, options);\n // pattern features are added only once the pattern is loaded into the image atlas\n // so are stored during populate until later updated with positions by tile worker in addFeatures\n this.patternFeatures.push(patternFeature);\n } else {\n this.addFeature(bucketFeature, geometry, index, canonical, {});\n }\n\n const feature = features[index].feature;\n options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {\n [_: string]: ImagePosition;\n }) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {\n [_: string]: ImagePosition;\n }) {\n for (const feature of this.patternFeatures) {\n this.addFeature(feature, feature.geometry, feature.index, canonical, imagePositions);\n }\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0;\n }\n\n uploadPending(): boolean {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n upload(context: Context) {\n if (!this.uploaded) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n this.indexBuffer2 = context.createIndexBuffer(this.indexArray2);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.indexBuffer2.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n this.segments2.destroy();\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {\n [_: string]: ImagePosition;\n }) {\n for (const polygon of classifyRings(geometry, EARCUT_MAX_RINGS)) {\n let numVertices = 0;\n for (const ring of polygon) {\n numVertices += ring.length;\n }\n\n const triangleSegment = this.segments.prepareSegment(numVertices, this.layoutVertexArray, this.indexArray);\n const triangleIndex = triangleSegment.vertexLength;\n\n const flattened = [];\n const holeIndices = [];\n\n for (const ring of polygon) {\n if (ring.length === 0) {\n continue;\n }\n\n if (ring !== polygon[0]) {\n holeIndices.push(flattened.length / 2);\n }\n\n const lineSegment = this.segments2.prepareSegment(ring.length, this.layoutVertexArray, this.indexArray2);\n const lineIndex = lineSegment.vertexLength;\n\n this.layoutVertexArray.emplaceBack(ring[0].x, ring[0].y);\n this.indexArray2.emplaceBack(lineIndex + ring.length - 1, lineIndex);\n flattened.push(ring[0].x);\n flattened.push(ring[0].y);\n\n for (let i = 1; i < ring.length; i++) {\n this.layoutVertexArray.emplaceBack(ring[i].x, ring[i].y);\n this.indexArray2.emplaceBack(lineIndex + i - 1, lineIndex + i);\n flattened.push(ring[i].x);\n flattened.push(ring[i].y);\n }\n\n lineSegment.vertexLength += ring.length;\n lineSegment.primitiveLength += ring.length;\n }\n\n const indices = earcut(flattened, holeIndices);\n\n for (let i = 0; i < indices.length; i += 3) {\n this.indexArray.emplaceBack(\n triangleIndex + indices[i],\n triangleIndex + indices[i + 1],\n triangleIndex + indices[i + 2]);\n }\n\n triangleSegment.vertexLength += numVertices;\n triangleSegment.primitiveLength += indices.length / 3;\n }\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical);\n }\n}\n\nregister('FillBucket', FillBucket, {omit: ['layers', 'patternFeatures']});\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type FillLayoutProps = {\n \"fill-sort-key\": DataDrivenProperty,\n};\n\nexport type FillLayoutPropsPossiblyEvaluated = {\n \"fill-sort-key\": PossiblyEvaluatedPropertyValue,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"fill-sort-key\": new DataDrivenProperty(styleSpec[\"layout_fill\"][\"fill-sort-key\"] as any as StylePropertySpecification),\n});\n\nexport type FillPaintProps = {\n \"fill-antialias\": DataConstantProperty,\n \"fill-opacity\": DataDrivenProperty,\n \"fill-color\": DataDrivenProperty,\n \"fill-outline-color\": DataDrivenProperty,\n \"fill-translate\": DataConstantProperty<[number, number]>,\n \"fill-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"fill-pattern\": CrossFadedDataDrivenProperty,\n};\n\nexport type FillPaintPropsPossiblyEvaluated = {\n \"fill-antialias\": boolean,\n \"fill-opacity\": PossiblyEvaluatedPropertyValue,\n \"fill-color\": PossiblyEvaluatedPropertyValue,\n \"fill-outline-color\": PossiblyEvaluatedPropertyValue,\n \"fill-translate\": [number, number],\n \"fill-translate-anchor\": \"map\" | \"viewport\",\n \"fill-pattern\": PossiblyEvaluatedPropertyValue>,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"fill-antialias\": new DataConstantProperty(styleSpec[\"paint_fill\"][\"fill-antialias\"] as any as StylePropertySpecification),\n \"fill-opacity\": new DataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-opacity\"] as any as StylePropertySpecification),\n \"fill-color\": new DataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-color\"] as any as StylePropertySpecification),\n \"fill-outline-color\": new DataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-outline-color\"] as any as StylePropertySpecification),\n \"fill-translate\": new DataConstantProperty(styleSpec[\"paint_fill\"][\"fill-translate\"] as any as StylePropertySpecification),\n \"fill-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_fill\"][\"fill-translate-anchor\"] as any as StylePropertySpecification),\n \"fill-pattern\": new CrossFadedDataDrivenProperty(styleSpec[\"paint_fill\"][\"fill-pattern\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import {StyleLayer} from '../style_layer';\n\nimport {FillBucket} from '../../data/bucket/fill_bucket';\nimport {polygonIntersectsMultiPolygon} from '../../util/intersection_tests';\nimport {translateDistance, translate} from '../query_utils';\nimport properties, {FillLayoutPropsPossiblyEvaluated, FillPaintPropsPossiblyEvaluated} from './fill_style_layer_properties.g';\nimport {Transitionable, Transitioning, Layout, PossiblyEvaluated} from '../properties';\n\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {BucketParameters} from '../../data/bucket';\nimport type Point from '@mapbox/point-geometry';\nimport type {FillLayoutProps, FillPaintProps} from './fill_style_layer_properties.g';\nimport type {EvaluationParameters} from '../evaluation_parameters';\nimport type {Transform} from '../../geo/transform';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nexport class FillStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n super.recalculate(parameters, availableImages);\n\n const outlineColor = this.paint._values['fill-outline-color'];\n if (outlineColor.value.kind === 'constant' && outlineColor.value.value === undefined) {\n this.paint._values['fill-outline-color'] = this.paint._values['fill-color'];\n }\n }\n\n createBucket(parameters: BucketParameters) {\n return new FillBucket(parameters);\n }\n\n queryRadius(): number {\n return translateDistance(this.paint.get('fill-translate'));\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number\n ): boolean {\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('fill-translate'),\n this.paint.get('fill-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n return polygonIntersectsMultiPolygon(translatedPolygon, geometry);\n }\n\n isTileClipped() {\n return true;\n }\n}\n","import {createLayout} from '../../util/struct_array';\n\nconst layout = createLayout([\n {name: 'a_pos', components: 2, type: 'Int16'},\n {name: 'a_normal_ed', components: 4, type: 'Int16'},\n], 4);\n\nexport const centroidAttributes = createLayout([\n {name: 'a_centroid', components: 2, type: 'Int16'}\n], 4);\n\nexport default layout;\nexport const {members, size, alignment} = layout;\n","'use strict';\n\nvar Point = require('@mapbox/point-geometry');\n\nmodule.exports = VectorTileFeature;\n\nfunction VectorTileFeature(pbf, end, extent, keys, values) {\n // Public\n this.properties = {};\n this.extent = extent;\n this.type = 0;\n\n // Private\n this._pbf = pbf;\n this._geometry = -1;\n this._keys = keys;\n this._values = values;\n\n pbf.readFields(readFeature, this, end);\n}\n\nfunction readFeature(tag, feature, pbf) {\n if (tag == 1) feature.id = pbf.readVarint();\n else if (tag == 2) readTag(pbf, feature);\n else if (tag == 3) feature.type = pbf.readVarint();\n else if (tag == 4) feature._geometry = pbf.pos;\n}\n\nfunction readTag(pbf, feature) {\n var end = pbf.readVarint() + pbf.pos;\n\n while (pbf.pos < end) {\n var key = feature._keys[pbf.readVarint()],\n value = feature._values[pbf.readVarint()];\n feature.properties[key] = value;\n }\n}\n\nVectorTileFeature.types = ['Unknown', 'Point', 'LineString', 'Polygon'];\n\nVectorTileFeature.prototype.loadGeometry = function() {\n var pbf = this._pbf;\n pbf.pos = this._geometry;\n\n var end = pbf.readVarint() + pbf.pos,\n cmd = 1,\n length = 0,\n x = 0,\n y = 0,\n lines = [],\n line;\n\n while (pbf.pos < end) {\n if (length <= 0) {\n var cmdLen = pbf.readVarint();\n cmd = cmdLen & 0x7;\n length = cmdLen >> 3;\n }\n\n length--;\n\n if (cmd === 1 || cmd === 2) {\n x += pbf.readSVarint();\n y += pbf.readSVarint();\n\n if (cmd === 1) { // moveTo\n if (line) lines.push(line);\n line = [];\n }\n\n line.push(new Point(x, y));\n\n } else if (cmd === 7) {\n\n // Workaround for https://github.com/mapbox/mapnik-vector-tile/issues/90\n if (line) {\n line.push(line[0].clone()); // closePolygon\n }\n\n } else {\n throw new Error('unknown command ' + cmd);\n }\n }\n\n if (line) lines.push(line);\n\n return lines;\n};\n\nVectorTileFeature.prototype.bbox = function() {\n var pbf = this._pbf;\n pbf.pos = this._geometry;\n\n var end = pbf.readVarint() + pbf.pos,\n cmd = 1,\n length = 0,\n x = 0,\n y = 0,\n x1 = Infinity,\n x2 = -Infinity,\n y1 = Infinity,\n y2 = -Infinity;\n\n while (pbf.pos < end) {\n if (length <= 0) {\n var cmdLen = pbf.readVarint();\n cmd = cmdLen & 0x7;\n length = cmdLen >> 3;\n }\n\n length--;\n\n if (cmd === 1 || cmd === 2) {\n x += pbf.readSVarint();\n y += pbf.readSVarint();\n if (x < x1) x1 = x;\n if (x > x2) x2 = x;\n if (y < y1) y1 = y;\n if (y > y2) y2 = y;\n\n } else if (cmd !== 7) {\n throw new Error('unknown command ' + cmd);\n }\n }\n\n return [x1, y1, x2, y2];\n};\n\nVectorTileFeature.prototype.toGeoJSON = function(x, y, z) {\n var size = this.extent * Math.pow(2, z),\n x0 = this.extent * x,\n y0 = this.extent * y,\n coords = this.loadGeometry(),\n type = VectorTileFeature.types[this.type],\n i, j;\n\n function project(line) {\n for (var j = 0; j < line.length; j++) {\n var p = line[j], y2 = 180 - (p.y + y0) * 360 / size;\n line[j] = [\n (p.x + x0) * 360 / size - 180,\n 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90\n ];\n }\n }\n\n switch (this.type) {\n case 1:\n var points = [];\n for (i = 0; i < coords.length; i++) {\n points[i] = coords[i][0];\n }\n coords = points;\n project(coords);\n break;\n\n case 2:\n for (i = 0; i < coords.length; i++) {\n project(coords[i]);\n }\n break;\n\n case 3:\n coords = classifyRings(coords);\n for (i = 0; i < coords.length; i++) {\n for (j = 0; j < coords[i].length; j++) {\n project(coords[i][j]);\n }\n }\n break;\n }\n\n if (coords.length === 1) {\n coords = coords[0];\n } else {\n type = 'Multi' + type;\n }\n\n var result = {\n type: \"Feature\",\n geometry: {\n type: type,\n coordinates: coords\n },\n properties: this.properties\n };\n\n if ('id' in this) {\n result.id = this.id;\n }\n\n return result;\n};\n\n// classifies an array of rings into polygons with outer rings and holes\n\nfunction classifyRings(rings) {\n var len = rings.length;\n\n if (len <= 1) return [rings];\n\n var polygons = [],\n polygon,\n ccw;\n\n for (var i = 0; i < len; i++) {\n var area = signedArea(rings[i]);\n if (area === 0) continue;\n\n if (ccw === undefined) ccw = area < 0;\n\n if (ccw === area < 0) {\n if (polygon) polygons.push(polygon);\n polygon = [rings[i]];\n\n } else {\n polygon.push(rings[i]);\n }\n }\n if (polygon) polygons.push(polygon);\n\n return polygons;\n}\n\nfunction signedArea(ring) {\n var sum = 0;\n for (var i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) {\n p1 = ring[i];\n p2 = ring[j];\n sum += (p2.x - p1.x) * (p1.y + p2.y);\n }\n return sum;\n}\n","'use strict';\n\nvar VectorTileFeature = require('./vectortilefeature.js');\n\nmodule.exports = VectorTileLayer;\n\nfunction VectorTileLayer(pbf, end) {\n // Public\n this.version = 1;\n this.name = null;\n this.extent = 4096;\n this.length = 0;\n\n // Private\n this._pbf = pbf;\n this._keys = [];\n this._values = [];\n this._features = [];\n\n pbf.readFields(readLayer, this, end);\n\n this.length = this._features.length;\n}\n\nfunction readLayer(tag, layer, pbf) {\n if (tag === 15) layer.version = pbf.readVarint();\n else if (tag === 1) layer.name = pbf.readString();\n else if (tag === 5) layer.extent = pbf.readVarint();\n else if (tag === 2) layer._features.push(pbf.pos);\n else if (tag === 3) layer._keys.push(pbf.readString());\n else if (tag === 4) layer._values.push(readValueMessage(pbf));\n}\n\nfunction readValueMessage(pbf) {\n var value = null,\n end = pbf.readVarint() + pbf.pos;\n\n while (pbf.pos < end) {\n var tag = pbf.readVarint() >> 3;\n\n value = tag === 1 ? pbf.readString() :\n tag === 2 ? pbf.readFloat() :\n tag === 3 ? pbf.readDouble() :\n tag === 4 ? pbf.readVarint64() :\n tag === 5 ? pbf.readVarint() :\n tag === 6 ? pbf.readSVarint() :\n tag === 7 ? pbf.readBoolean() : null;\n }\n\n return value;\n}\n\n// return feature `i` from this layer as a `VectorTileFeature`\nVectorTileLayer.prototype.feature = function(i) {\n if (i < 0 || i >= this._features.length) throw new Error('feature index out of bounds');\n\n this._pbf.pos = this._features[i];\n\n var end = this._pbf.readVarint() + this._pbf.pos;\n return new VectorTileFeature(this._pbf, end, this.extent, this._keys, this._values);\n};\n","'use strict';\n\nvar VectorTileLayer = require('./vectortilelayer');\n\nmodule.exports = VectorTile;\n\nfunction VectorTile(pbf, end) {\n this.layers = pbf.readFields(readTile, {}, end);\n}\n\nfunction readTile(tag, layers, pbf) {\n if (tag === 3) {\n var layer = new VectorTileLayer(pbf, pbf.readVarint() + pbf.pos);\n if (layer.length) layers[layer.name] = layer;\n }\n}\n\n","module.exports.VectorTile = require('./lib/vectortile.js');\nmodule.exports.VectorTileFeature = require('./lib/vectortilefeature.js');\nmodule.exports.VectorTileLayer = require('./lib/vectortilelayer.js');\n","import {FillExtrusionLayoutArray, PosArray} from '../array_types.g';\n\nimport {members as layoutAttributes, centroidAttributes} from './fill_extrusion_attributes';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray} from '../index_array_type';\nimport {EXTENT} from '../extent';\nimport earcut from 'earcut';\nimport mvt from '@mapbox/vector-tile';\nconst vectorTileFeatureTypes = mvt.VectorTileFeature.types;\nimport {classifyRings} from '../../util/classify_rings';\nconst EARCUT_MAX_RINGS = 500;\nimport {register} from '../../util/web_worker_transfer';\nimport {hasPattern, addPatternDependencies} from './pattern_bucket_features';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\n\nimport type {FillExtrusionStyleLayer} from '../../style/style_layer/fill_extrusion_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type Point from '@mapbox/point-geometry';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nconst FACTOR = Math.pow(2, 13);\n\nfunction addVertex(vertexArray, x, y, nx, ny, nz, t, e) {\n vertexArray.emplaceBack(\n // a_pos\n x,\n y,\n // a_normal_ed: 3-component normal and 1-component edgedistance\n Math.floor(nx * FACTOR) * 2 + t,\n ny * FACTOR * 2,\n nz * FACTOR * 2,\n // edgedistance (used for wrapping patterns around extrusion sides)\n Math.round(e)\n );\n}\n\nexport class FillExtrusionBucket implements Bucket {\n index: number;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n\n layoutVertexArray: FillExtrusionLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n centroidVertexArray: PosArray;\n centroidVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n uploaded: boolean;\n features: Array;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n\n this.layoutVertexArray = new FillExtrusionLayoutArray();\n this.centroidVertexArray = new PosArray();\n this.indexArray = new TriangleIndexArray();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.segments = new SegmentVector();\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n this.features = [];\n this.hasPattern = hasPattern('fill-extrusion', this.layers, options);\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const bucketFeature: BucketFeature = {\n id,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n properties: feature.properties,\n type: feature.type,\n patterns: {}\n };\n\n if (this.hasPattern) {\n this.features.push(addPatternDependencies('fill-extrusion', this.layers, bucketFeature, this.zoom, options));\n } else {\n this.addFeature(bucketFeature, bucketFeature.geometry, index, canonical, {});\n }\n\n options.featureIndex.insert(feature, bucketFeature.geometry, index, sourceLayerIndex, this.index, true);\n }\n }\n\n addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n for (const feature of this.features) {\n const {geometry} = feature;\n this.addFeature(feature, geometry, feature.index, canonical, imagePositions);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0 && this.centroidVertexArray.length === 0;\n }\n\n uploadPending() {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.centroidVertexBuffer = context.createVertexBuffer(this.centroidVertexArray, centroidAttributes.members, true);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n this.centroidVertexBuffer.destroy();\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n const centroid = {x: 0, y: 0, vertexCount: 0};\n for (const polygon of classifyRings(geometry, EARCUT_MAX_RINGS)) {\n let numVertices = 0;\n for (const ring of polygon) {\n numVertices += ring.length;\n }\n let segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray);\n\n for (const ring of polygon) {\n if (ring.length === 0) {\n continue;\n }\n\n if (isEntirelyOutside(ring)) {\n continue;\n }\n\n let edgeDistance = 0;\n\n for (let p = 0; p < ring.length; p++) {\n const p1 = ring[p];\n\n if (p >= 1) {\n const p2 = ring[p - 1];\n\n if (!isBoundaryEdge(p1, p2)) {\n if (segment.vertexLength + 4 > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) {\n segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray);\n }\n\n const perp = p1.sub(p2)._perp()._unit();\n const dist = p2.dist(p1);\n if (edgeDistance + dist > 32768) edgeDistance = 0;\n\n addVertex(this.layoutVertexArray, p1.x, p1.y, perp.x, perp.y, 0, 0, edgeDistance);\n addVertex(this.layoutVertexArray, p1.x, p1.y, perp.x, perp.y, 0, 1, edgeDistance);\n centroid.x += 2 * p1.x;\n centroid.y += 2 * p1.y;\n centroid.vertexCount += 2;\n\n edgeDistance += dist;\n\n addVertex(this.layoutVertexArray, p2.x, p2.y, perp.x, perp.y, 0, 0, edgeDistance);\n addVertex(this.layoutVertexArray, p2.x, p2.y, perp.x, perp.y, 0, 1, edgeDistance);\n centroid.x += 2 * p2.x;\n centroid.y += 2 * p2.y;\n centroid.vertexCount += 2;\n\n const bottomRight = segment.vertexLength;\n\n // ┌──────┐\n // │ 0 1 │ Counter-clockwise winding order.\n // │ │ Triangle 1: 0 => 2 => 1\n // │ 2 3 │ Triangle 2: 1 => 2 => 3\n // └──────┘\n this.indexArray.emplaceBack(bottomRight, bottomRight + 2, bottomRight + 1);\n this.indexArray.emplaceBack(bottomRight + 1, bottomRight + 2, bottomRight + 3);\n\n segment.vertexLength += 4;\n segment.primitiveLength += 2;\n }\n }\n }\n\n }\n\n if (segment.vertexLength + numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) {\n segment = this.segments.prepareSegment(numVertices, this.layoutVertexArray, this.indexArray);\n }\n\n //Only triangulate and draw the area of the feature if it is a polygon\n //Other feature types (e.g. LineString) do not have area, so triangulation is pointless / undefined\n if (vectorTileFeatureTypes[feature.type] !== 'Polygon')\n continue;\n\n const flattened = [];\n const holeIndices = [];\n const triangleIndex = segment.vertexLength;\n\n for (const ring of polygon) {\n if (ring.length === 0) {\n continue;\n }\n\n if (ring !== polygon[0]) {\n holeIndices.push(flattened.length / 2);\n }\n\n for (let i = 0; i < ring.length; i++) {\n const p = ring[i];\n\n addVertex(this.layoutVertexArray, p.x, p.y, 0, 0, 1, 1, 0);\n centroid.x += p.x;\n centroid.y += p.y;\n centroid.vertexCount += 1;\n\n flattened.push(p.x);\n flattened.push(p.y);\n }\n\n }\n\n const indices = earcut(flattened, holeIndices);\n\n for (let j = 0; j < indices.length; j += 3) {\n // Counter-clockwise winding order.\n this.indexArray.emplaceBack(\n triangleIndex + indices[j],\n triangleIndex + indices[j + 2],\n triangleIndex + indices[j + 1]);\n }\n\n segment.primitiveLength += indices.length / 3;\n segment.vertexLength += numVertices;\n }\n\n // remember polygon centroid to calculate elevation in GPU\n for (let i = 0; i < centroid.vertexCount; i++) {\n this.centroidVertexArray.emplaceBack(\n Math.floor(centroid.x / centroid.vertexCount),\n Math.floor(centroid.y / centroid.vertexCount)\n );\n }\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical);\n }\n}\n\nregister('FillExtrusionBucket', FillExtrusionBucket, {omit: ['layers', 'features']});\n\nfunction isBoundaryEdge(p1, p2) {\n return (p1.x === p2.x && (p1.x < 0 || p1.x > EXTENT)) ||\n (p1.y === p2.y && (p1.y < 0 || p1.y > EXTENT));\n}\n\nfunction isEntirelyOutside(ring) {\n return ring.every(p => p.x < 0) ||\n ring.every(p => p.x > EXTENT) ||\n ring.every(p => p.y < 0) ||\n ring.every(p => p.y > EXTENT);\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type FillExtrusionPaintProps = {\n \"fill-extrusion-opacity\": DataConstantProperty,\n \"fill-extrusion-color\": DataDrivenProperty,\n \"fill-extrusion-translate\": DataConstantProperty<[number, number]>,\n \"fill-extrusion-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"fill-extrusion-pattern\": CrossFadedDataDrivenProperty,\n \"fill-extrusion-height\": DataDrivenProperty,\n \"fill-extrusion-base\": DataDrivenProperty,\n \"fill-extrusion-vertical-gradient\": DataConstantProperty,\n};\n\nexport type FillExtrusionPaintPropsPossiblyEvaluated = {\n \"fill-extrusion-opacity\": number,\n \"fill-extrusion-color\": PossiblyEvaluatedPropertyValue,\n \"fill-extrusion-translate\": [number, number],\n \"fill-extrusion-translate-anchor\": \"map\" | \"viewport\",\n \"fill-extrusion-pattern\": PossiblyEvaluatedPropertyValue>,\n \"fill-extrusion-height\": PossiblyEvaluatedPropertyValue,\n \"fill-extrusion-base\": PossiblyEvaluatedPropertyValue,\n \"fill-extrusion-vertical-gradient\": boolean,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"fill-extrusion-opacity\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-opacity\"] as any as StylePropertySpecification),\n \"fill-extrusion-color\": new DataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-color\"] as any as StylePropertySpecification),\n \"fill-extrusion-translate\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-translate\"] as any as StylePropertySpecification),\n \"fill-extrusion-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-translate-anchor\"] as any as StylePropertySpecification),\n \"fill-extrusion-pattern\": new CrossFadedDataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-pattern\"] as any as StylePropertySpecification),\n \"fill-extrusion-height\": new DataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-height\"] as any as StylePropertySpecification),\n \"fill-extrusion-base\": new DataDrivenProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-base\"] as any as StylePropertySpecification),\n \"fill-extrusion-vertical-gradient\": new DataConstantProperty(styleSpec[\"paint_fill-extrusion\"][\"fill-extrusion-vertical-gradient\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport {FillExtrusionBucket} from '../../data/bucket/fill_extrusion_bucket';\nimport {polygonIntersectsPolygon, polygonIntersectsMultiPolygon} from '../../util/intersection_tests';\nimport {translateDistance, translate} from '../query_utils';\nimport properties, {FillExtrusionPaintPropsPossiblyEvaluated} from './fill_extrusion_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\nimport {mat4, vec4} from 'gl-matrix';\nimport Point from '@mapbox/point-geometry';\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {BucketParameters} from '../../data/bucket';\nimport type {FillExtrusionPaintProps} from './fill_extrusion_style_layer_properties.g';\nimport type {Transform} from '../../geo/transform';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nexport class Point3D extends Point {\n z: number;\n}\n\nexport class FillExtrusionStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n createBucket(parameters: BucketParameters) {\n return new FillExtrusionBucket(parameters);\n }\n\n queryRadius(): number {\n return translateDistance(this.paint.get('fill-extrusion-translate'));\n }\n\n is3D(): boolean {\n return true;\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number,\n pixelPosMatrix: mat4\n ): boolean | number {\n\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('fill-extrusion-translate'),\n this.paint.get('fill-extrusion-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n\n const height = this.paint.get('fill-extrusion-height').evaluate(feature, featureState);\n const base = this.paint.get('fill-extrusion-base').evaluate(feature, featureState);\n\n const projectedQueryGeometry = projectQueryGeometry(translatedPolygon, pixelPosMatrix, transform, 0);\n\n const projected = projectExtrusion(geometry, base, height, pixelPosMatrix);\n const projectedBase = projected[0];\n const projectedTop = projected[1];\n return checkIntersection(projectedBase, projectedTop, projectedQueryGeometry);\n }\n}\n\nfunction dot(a, b) {\n return a.x * b.x + a.y * b.y;\n}\n\nexport function getIntersectionDistance(projectedQueryGeometry: Array, projectedFace: Array) {\n\n if (projectedQueryGeometry.length === 1) {\n // For point queries calculate the z at which the point intersects the face\n // using barycentric coordinates.\n\n // Find the barycentric coordinates of the projected point within the first\n // triangle of the face, using only the xy plane. It doesn't matter if the\n // point is outside the first triangle because all the triangles in the face\n // are in the same plane.\n //\n // Check whether points are coincident and use other points if they are.\n let i = 0;\n const a = projectedFace[i++];\n let b;\n while (!b || a.equals(b)) {\n b = projectedFace[i++];\n if (!b) return Infinity;\n }\n\n // Loop until point `c` is not colinear with points `a` and `b`.\n for (; i < projectedFace.length; i++) {\n const c = projectedFace[i];\n\n const p = projectedQueryGeometry[0];\n\n const ab = b.sub(a);\n const ac = c.sub(a);\n const ap = p.sub(a);\n\n const dotABAB = dot(ab, ab);\n const dotABAC = dot(ab, ac);\n const dotACAC = dot(ac, ac);\n const dotAPAB = dot(ap, ab);\n const dotAPAC = dot(ap, ac);\n const denom = dotABAB * dotACAC - dotABAC * dotABAC;\n\n const v = (dotACAC * dotAPAB - dotABAC * dotAPAC) / denom;\n const w = (dotABAB * dotAPAC - dotABAC * dotAPAB) / denom;\n const u = 1 - v - w;\n\n // Use the barycentric weighting along with the original triangle z coordinates to get the point of intersection.\n const distance = a.z * u + b.z * v + c.z * w;\n\n if (isFinite(distance)) return distance;\n }\n\n return Infinity;\n\n } else {\n // The counts as closest is less clear when the query is a box. This\n // returns the distance to the nearest point on the face, whether it is\n // within the query or not. It could be more correct to return the\n // distance to the closest point within the query box but this would be\n // more complicated and expensive to calculate with little benefit.\n let closestDistance = Infinity;\n for (const p of projectedFace) {\n closestDistance = Math.min(closestDistance, p.z);\n }\n return closestDistance;\n }\n}\n\nfunction checkIntersection(projectedBase: Array>, projectedTop: Array>, projectedQueryGeometry: Array) {\n let closestDistance = Infinity;\n\n if (polygonIntersectsMultiPolygon(projectedQueryGeometry, projectedTop)) {\n closestDistance = getIntersectionDistance(projectedQueryGeometry, projectedTop[0]);\n }\n\n for (let r = 0; r < projectedTop.length; r++) {\n const ringTop = projectedTop[r];\n const ringBase = projectedBase[r];\n for (let p = 0; p < ringTop.length - 1; p++) {\n const topA = ringTop[p];\n const topB = ringTop[p + 1];\n const baseA = ringBase[p];\n const baseB = ringBase[p + 1];\n const face = [topA, topB, baseB, baseA, topA];\n if (polygonIntersectsPolygon(projectedQueryGeometry, face)) {\n closestDistance = Math.min(closestDistance, getIntersectionDistance(projectedQueryGeometry, face));\n }\n }\n }\n\n return closestDistance === Infinity ? false : closestDistance;\n}\n\n/*\n * Project the geometry using matrix `m`. This is essentially doing\n * `vec4.transformMat4([], [p.x, p.y, z, 1], m)` but the multiplication\n * is inlined so that parts of the projection that are the same across\n * different points can only be done once. This produced a measurable\n * performance improvement.\n */\nfunction projectExtrusion(geometry: Array>, zBase: number, zTop: number, m: mat4): [Array>, Array>] {\n const projectedBase = [] as Array>;\n const projectedTop = [] as Array>;\n const baseXZ = m[8] * zBase;\n const baseYZ = m[9] * zBase;\n const baseZZ = m[10] * zBase;\n const baseWZ = m[11] * zBase;\n const topXZ = m[8] * zTop;\n const topYZ = m[9] * zTop;\n const topZZ = m[10] * zTop;\n const topWZ = m[11] * zTop;\n\n for (const r of geometry) {\n const ringBase = [] as Array;\n const ringTop = [] as Array;\n for (const p of r) {\n const x = p.x;\n const y = p.y;\n\n const sX = m[0] * x + m[4] * y + m[12];\n const sY = m[1] * x + m[5] * y + m[13];\n const sZ = m[2] * x + m[6] * y + m[14];\n const sW = m[3] * x + m[7] * y + m[15];\n\n const baseX = sX + baseXZ;\n const baseY = sY + baseYZ;\n const baseZ = sZ + baseZZ;\n const baseW = sW + baseWZ;\n\n const topX = sX + topXZ;\n const topY = sY + topYZ;\n const topZ = sZ + topZZ;\n const topW = sW + topWZ;\n\n const b = new Point(baseX / baseW, baseY / baseW) as Point3D;\n b.z = baseZ / baseW;\n ringBase.push(b);\n\n const t = new Point(topX / topW, topY / topW) as Point3D;\n t.z = topZ / topW;\n ringTop.push(t);\n }\n projectedBase.push(ringBase);\n projectedTop.push(ringTop);\n }\n return [projectedBase, projectedTop];\n}\n\nfunction projectQueryGeometry(queryGeometry: Array, pixelPosMatrix: mat4, transform: Transform, z: number) {\n const projectedQueryGeometry = [];\n for (const p of queryGeometry) {\n const v = [p.x, p.y, z, 1] as vec4;\n vec4.transformMat4(v, v, pixelPosMatrix);\n projectedQueryGeometry.push(new Point(v[0] / v[3], v[1] / v[3]));\n }\n return projectedQueryGeometry;\n}\n","import {createLayout} from '../../util/struct_array';\n\nexport const lineLayoutAttributes = createLayout([\n {name: 'a_pos_normal', components: 2, type: 'Int16'},\n {name: 'a_data', components: 4, type: 'Uint8'}\n], 4);\n\nexport const {members, size, alignment} = lineLayoutAttributes;\n","import {createLayout} from '../../util/struct_array';\n\nexport const lineLayoutAttributesExt = createLayout([\n {name: 'a_uv_x', components: 1, type: 'Float32'},\n {name: 'a_split_index', components: 1, type: 'Float32'},\n]);\n\nexport const {members, size, alignment} = lineLayoutAttributesExt;\n","import {LineLayoutArray, LineExtLayoutArray} from '../array_types.g';\n\nimport {members as layoutAttributes} from './line_attributes';\nimport {members as layoutAttributesExt} from './line_attributes_ext';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray} from '../index_array_type';\nimport {EXTENT} from '../extent';\nimport mvt from '@mapbox/vector-tile';\nconst vectorTileFeatureTypes = mvt.VectorTileFeature.types;\nimport {register} from '../../util/web_worker_transfer';\nimport {hasPattern, addPatternDependencies} from './pattern_bucket_features';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\n\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n BucketFeature,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {LineStyleLayer} from '../../style/style_layer/line_style_layer';\nimport type Point from '@mapbox/point-geometry';\nimport type {Segment} from '../segment';\nimport {RGBAImage} from '../../util/image';\nimport type {Context} from '../../gl/context';\nimport type {Texture} from '../../render/texture';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\n// NOTE ON EXTRUDE SCALE:\n// scale the extrusion vector so that the normal length is this value.\n// contains the \"texture\" normals (-1..1). this is distinct from the extrude\n// normals for line joins, because the x-value remains 0 for the texture\n// normal array, while the extrude normal actually moves the vertex to create\n// the acute/bevelled line join.\nconst EXTRUDE_SCALE = 63;\n\n/*\n * Sharp corners cause dashed lines to tilt because the distance along the line\n * is the same at both the inner and outer corners. To improve the appearance of\n * dashed lines we add extra points near sharp corners so that a smaller part\n * of the line is tilted.\n *\n * COS_HALF_SHARP_CORNER controls how sharp a corner has to be for us to add an\n * extra vertex. The default is 75 degrees.\n *\n * The newly created vertices are placed SHARP_CORNER_OFFSET pixels from the corner.\n */\nconst COS_HALF_SHARP_CORNER = Math.cos(75 / 2 * (Math.PI / 180));\nconst SHARP_CORNER_OFFSET = 15;\n\n// Angle per triangle for approximating round line joins.\nconst DEG_PER_TRIANGLE = 20;\n\n// The number of bits that is used to store the line distance in the buffer.\nconst LINE_DISTANCE_BUFFER_BITS = 15;\n\n// We don't have enough bits for the line distance as we'd like to have, so\n// use this value to scale the line distance (in tile units) down to a smaller\n// value. This lets us store longer distances while sacrificing precision.\nconst LINE_DISTANCE_SCALE = 1 / 2;\n\n// The maximum line distance, in tile units, that fits in the buffer.\nconst MAX_LINE_DISTANCE = Math.pow(2, LINE_DISTANCE_BUFFER_BITS - 1) / LINE_DISTANCE_SCALE;\n\ntype LineClips = {\n start: number;\n end: number;\n};\n\ntype GradientTexture = {\n texture?: Texture;\n gradient?: RGBAImage;\n version?: number;\n};\n\n/**\n * @internal\n * Line bucket class\n */\nexport class LineBucket implements Bucket {\n distance: number;\n totalDistance: number;\n maxLineLength: number;\n scaledDistance: number;\n lineClips?: LineClips;\n\n e1: number;\n e2: number;\n\n index: number;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n gradients: {[x: string]: GradientTexture};\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n patternFeatures: Array;\n lineClipsArray: Array;\n\n layoutVertexArray: LineLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n layoutVertexArray2: LineExtLayoutArray;\n layoutVertexBuffer2: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n hasPattern: boolean;\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n uploaded: boolean;\n\n constructor(options: BucketParameters) {\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.hasPattern = false;\n this.patternFeatures = [];\n this.lineClipsArray = [];\n this.gradients = {};\n this.layers.forEach(layer => {\n this.gradients[layer.id] = {};\n });\n\n this.layoutVertexArray = new LineLayoutArray();\n this.layoutVertexArray2 = new LineExtLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom);\n this.segments = new SegmentVector();\n this.maxLineLength = 0;\n\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n this.hasPattern = hasPattern('line', this.layers, options);\n const lineSortKey = this.layers[0].layout.get('line-sort-key');\n const sortFeaturesByKey = !lineSortKey.isConstant();\n const bucketFeatures: BucketFeature[] = [];\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n const needGeometry = this.layers[0]._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n\n if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue;\n\n const sortKey = sortFeaturesByKey ?\n lineSortKey.evaluate(evaluationFeature, {}, canonical) :\n undefined;\n\n const bucketFeature: BucketFeature = {\n id,\n properties: feature.properties,\n type: feature.type,\n sourceLayerIndex,\n index,\n geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature),\n patterns: {},\n sortKey\n };\n\n bucketFeatures.push(bucketFeature);\n }\n\n if (sortFeaturesByKey) {\n bucketFeatures.sort((a, b) => {\n return (a.sortKey) - (b.sortKey);\n });\n }\n\n for (const bucketFeature of bucketFeatures) {\n const {geometry, index, sourceLayerIndex} = bucketFeature;\n\n if (this.hasPattern) {\n const patternBucketFeature = addPatternDependencies('line', this.layers, bucketFeature, this.zoom, options);\n // pattern features are added only once the pattern is loaded into the image atlas\n // so are stored during populate until later updated with positions by tile worker in addFeatures\n this.patternFeatures.push(patternBucketFeature);\n } else {\n this.addFeature(bucketFeature, geometry, index, canonical, {});\n }\n\n const feature = features[index].feature;\n options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index);\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions);\n }\n\n addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n for (const feature of this.patternFeatures) {\n this.addFeature(feature, feature.geometry, feature.index, canonical, imagePositions);\n }\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0;\n }\n\n uploadPending() {\n return !this.uploaded || this.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded) {\n if (this.layoutVertexArray2.length !== 0) {\n this.layoutVertexBuffer2 = context.createVertexBuffer(this.layoutVertexArray2, layoutAttributesExt);\n }\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n }\n this.programConfigurations.upload(context);\n this.uploaded = true;\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n }\n\n lineFeatureClips(feature: BucketFeature): LineClips | undefined {\n if (!!feature.properties && Object.prototype.hasOwnProperty.call(feature.properties, 'mapbox_clip_start') && Object.prototype.hasOwnProperty.call(feature.properties, 'mapbox_clip_end')) {\n const start = +feature.properties['mapbox_clip_start'];\n const end = +feature.properties['mapbox_clip_end'];\n return {start, end};\n }\n }\n\n addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) {\n const layout = this.layers[0].layout;\n const join = layout.get('line-join').evaluate(feature, {});\n const cap = layout.get('line-cap');\n const miterLimit = layout.get('line-miter-limit');\n const roundLimit = layout.get('line-round-limit');\n this.lineClips = this.lineFeatureClips(feature);\n\n for (const line of geometry) {\n this.addLine(line, feature, join, cap, miterLimit, roundLimit);\n }\n\n this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical);\n }\n\n addLine(vertices: Array, feature: BucketFeature, join: string, cap: string, miterLimit: number, roundLimit: number) {\n this.distance = 0;\n this.scaledDistance = 0;\n this.totalDistance = 0;\n\n if (this.lineClips) {\n this.lineClipsArray.push(this.lineClips);\n // Calculate the total distance, in tile units, of this tiled line feature\n for (let i = 0; i < vertices.length - 1; i++) {\n this.totalDistance += vertices[i].dist(vertices[i + 1]);\n }\n this.updateScaledDistance();\n this.maxLineLength = Math.max(this.maxLineLength, this.totalDistance);\n }\n\n const isPolygon = vectorTileFeatureTypes[feature.type] === 'Polygon';\n\n // If the line has duplicate vertices at the ends, adjust start/length to remove them.\n let len = vertices.length;\n while (len >= 2 && vertices[len - 1].equals(vertices[len - 2])) {\n len--;\n }\n let first = 0;\n while (first < len - 1 && vertices[first].equals(vertices[first + 1])) {\n first++;\n }\n\n // Ignore invalid geometry.\n if (len < (isPolygon ? 3 : 2)) return;\n\n if (join === 'bevel') miterLimit = 1.05;\n\n const sharpCornerOffset = this.overscaling <= 16 ?\n SHARP_CORNER_OFFSET * EXTENT / (512 * this.overscaling) :\n 0;\n\n // we could be more precise, but it would only save a negligible amount of space\n const segment = this.segments.prepareSegment(len * 10, this.layoutVertexArray, this.indexArray);\n\n let currentVertex: Point;\n let prevVertex: Point;\n let nextVertex: Point;\n let prevNormal: Point;\n let nextNormal: Point;\n\n // the last two vertices added\n this.e1 = this.e2 = -1;\n\n if (isPolygon) {\n currentVertex = vertices[len - 2];\n nextNormal = vertices[first].sub(currentVertex)._unit()._perp();\n }\n\n for (let i = first; i < len; i++) {\n\n nextVertex = i === len - 1 ?\n (isPolygon ? vertices[first + 1] : undefined) : // if it's a polygon, treat the last vertex like the first\n vertices[i + 1]; // just the next vertex\n\n // if two consecutive vertices exist, skip the current one\n if (nextVertex && vertices[i].equals(nextVertex)) continue;\n\n if (nextNormal) prevNormal = nextNormal;\n if (currentVertex) prevVertex = currentVertex;\n\n currentVertex = vertices[i];\n\n // Calculate the normal towards the next vertex in this line. In case\n // there is no next vertex, pretend that the line is continuing straight,\n // meaning that we are just using the previous normal.\n nextNormal = nextVertex ? nextVertex.sub(currentVertex)._unit()._perp() : prevNormal;\n\n // If we still don't have a previous normal, this is the beginning of a\n // non-closed line, so we're doing a straight \"join\".\n prevNormal = prevNormal || nextNormal;\n\n // Determine the normal of the join extrusion. It is the angle bisector\n // of the segments between the previous line and the next line.\n // In the case of 180° angles, the prev and next normals cancel each other out:\n // prevNormal + nextNormal = (0, 0), its magnitude is 0, so the unit vector would be\n // undefined. In that case, we're keeping the joinNormal at (0, 0), so that the cosHalfAngle\n // below will also become 0 and miterLength will become Infinity.\n let joinNormal = prevNormal.add(nextNormal);\n if (joinNormal.x !== 0 || joinNormal.y !== 0) {\n joinNormal._unit();\n }\n /* joinNormal prevNormal\n * ↖ ↑\n * .________. prevVertex\n * |\n * nextNormal ← | currentVertex\n * |\n * nextVertex !\n *\n */\n\n // calculate cosines of the angle (and its half) using dot product\n const cosAngle = prevNormal.x * nextNormal.x + prevNormal.y * nextNormal.y;\n const cosHalfAngle = joinNormal.x * nextNormal.x + joinNormal.y * nextNormal.y;\n\n // Calculate the length of the miter (the ratio of the miter to the width)\n // as the inverse of cosine of the angle between next and join normals\n const miterLength = cosHalfAngle !== 0 ? 1 / cosHalfAngle : Infinity;\n\n // approximate angle from cosine\n const approxAngle = 2 * Math.sqrt(2 - 2 * cosHalfAngle);\n\n const isSharpCorner = cosHalfAngle < COS_HALF_SHARP_CORNER && prevVertex && nextVertex;\n const lineTurnsLeft = prevNormal.x * nextNormal.y - prevNormal.y * nextNormal.x > 0;\n\n if (isSharpCorner && i > first) {\n const prevSegmentLength = currentVertex.dist(prevVertex);\n if (prevSegmentLength > 2 * sharpCornerOffset) {\n const newPrevVertex = currentVertex.sub(currentVertex.sub(prevVertex)._mult(sharpCornerOffset / prevSegmentLength)._round());\n this.updateDistance(prevVertex, newPrevVertex);\n this.addCurrentVertex(newPrevVertex, prevNormal, 0, 0, segment);\n prevVertex = newPrevVertex;\n }\n }\n\n // The join if a middle vertex, otherwise the cap.\n const middleVertex = prevVertex && nextVertex;\n let currentJoin = middleVertex ? join : isPolygon ? 'butt' : cap;\n\n if (middleVertex && currentJoin === 'round') {\n if (miterLength < roundLimit) {\n currentJoin = 'miter';\n } else if (miterLength <= 2) {\n currentJoin = 'fakeround';\n }\n }\n\n if (currentJoin === 'miter' && miterLength > miterLimit) {\n currentJoin = 'bevel';\n }\n\n if (currentJoin === 'bevel') {\n // The maximum extrude length is 128 / 63 = 2 times the width of the line\n // so if miterLength >= 2 we need to draw a different type of bevel here.\n if (miterLength > 2) currentJoin = 'flipbevel';\n\n // If the miterLength is really small and the line bevel wouldn't be visible,\n // just draw a miter join to save a triangle.\n if (miterLength < miterLimit) currentJoin = 'miter';\n }\n\n // Calculate how far along the line the currentVertex is\n if (prevVertex) this.updateDistance(prevVertex, currentVertex);\n\n if (currentJoin === 'miter') {\n\n joinNormal._mult(miterLength);\n this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment);\n\n } else if (currentJoin === 'flipbevel') {\n // miter is too big, flip the direction to make a beveled join\n\n if (miterLength > 100) {\n // Almost parallel lines\n joinNormal = nextNormal.mult(-1);\n\n } else {\n const bevelLength = miterLength * prevNormal.add(nextNormal).mag() / prevNormal.sub(nextNormal).mag();\n joinNormal._perp()._mult(bevelLength * (lineTurnsLeft ? -1 : 1));\n }\n this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment);\n this.addCurrentVertex(currentVertex, joinNormal.mult(-1), 0, 0, segment);\n\n } else if (currentJoin === 'bevel' || currentJoin === 'fakeround') {\n const offset = -Math.sqrt(miterLength * miterLength - 1);\n const offsetA = lineTurnsLeft ? offset : 0;\n const offsetB = lineTurnsLeft ? 0 : offset;\n\n // Close previous segment with a bevel\n if (prevVertex) {\n this.addCurrentVertex(currentVertex, prevNormal, offsetA, offsetB, segment);\n }\n\n if (currentJoin === 'fakeround') {\n // The join angle is sharp enough that a round join would be visible.\n // Bevel joins fill the gap between segments with a single pie slice triangle.\n // Create a round join by adding multiple pie slices. The join isn't actually round, but\n // it looks like it is at the sizes we render lines at.\n\n // pick the number of triangles for approximating round join by based on the angle between normals\n const n = Math.round((approxAngle * 180 / Math.PI) / DEG_PER_TRIANGLE);\n\n for (let m = 1; m < n; m++) {\n let t = m / n;\n if (t !== 0.5) {\n // approximate spherical interpolation https://observablehq.com/@mourner/approximating-geometric-slerp\n const t2 = t - 0.5;\n const A = 1.0904 + cosAngle * (-3.2452 + cosAngle * (3.55645 - cosAngle * 1.43519));\n const B = 0.848013 + cosAngle * (-1.06021 + cosAngle * 0.215638);\n t = t + t * t2 * (t - 1) * (A * t2 * t2 + B);\n }\n const extrude = nextNormal.sub(prevNormal)._mult(t)._add(prevNormal)._unit()._mult(lineTurnsLeft ? -1 : 1);\n this.addHalfVertex(currentVertex, extrude.x, extrude.y, false, lineTurnsLeft, 0, segment);\n }\n }\n\n if (nextVertex) {\n // Start next segment\n this.addCurrentVertex(currentVertex, nextNormal, -offsetA, -offsetB, segment);\n }\n\n } else if (currentJoin === 'butt') {\n this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment); // butt cap\n\n } else if (currentJoin === 'square') {\n const offset = prevVertex ? 1 : -1; // closing or starting square cap\n this.addCurrentVertex(currentVertex, joinNormal, offset, offset, segment);\n\n } else if (currentJoin === 'round') {\n\n if (prevVertex) {\n // Close previous segment with butt\n this.addCurrentVertex(currentVertex, prevNormal, 0, 0, segment);\n\n // Add round cap or linejoin at end of segment\n this.addCurrentVertex(currentVertex, prevNormal, 1, 1, segment, true);\n }\n if (nextVertex) {\n // Add round cap before first segment\n this.addCurrentVertex(currentVertex, nextNormal, -1, -1, segment, true);\n\n // Start next segment with a butt\n this.addCurrentVertex(currentVertex, nextNormal, 0, 0, segment);\n }\n }\n\n if (isSharpCorner && i < len - 1) {\n const nextSegmentLength = currentVertex.dist(nextVertex);\n if (nextSegmentLength > 2 * sharpCornerOffset) {\n const newCurrentVertex = currentVertex.add(nextVertex.sub(currentVertex)._mult(sharpCornerOffset / nextSegmentLength)._round());\n this.updateDistance(currentVertex, newCurrentVertex);\n this.addCurrentVertex(newCurrentVertex, nextNormal, 0, 0, segment);\n currentVertex = newCurrentVertex;\n }\n }\n }\n }\n\n /**\n * Add two vertices to the buffers.\n *\n * @param p - the line vertex to add buffer vertices for\n * @param normal - vertex normal\n * @param endLeft - extrude to shift the left vertex along the line\n * @param endRight - extrude to shift the left vertex along the line\n * @param segment - the segment object to add the vertex to\n * @param round - whether this is a round cap\n */\n addCurrentVertex(p: Point, normal: Point, endLeft: number, endRight: number, segment: Segment, round: boolean = false) {\n // left and right extrude vectors, perpendicularly shifted by endLeft/endRight\n const leftX = normal.x + normal.y * endLeft;\n const leftY = normal.y - normal.x * endLeft;\n const rightX = -normal.x + normal.y * endRight;\n const rightY = -normal.y - normal.x * endRight;\n\n this.addHalfVertex(p, leftX, leftY, round, false, endLeft, segment);\n this.addHalfVertex(p, rightX, rightY, round, true, -endRight, segment);\n\n // There is a maximum \"distance along the line\" that we can store in the buffers.\n // When we get close to the distance, reset it to zero and add the vertex again with\n // a distance of zero. The max distance is determined by the number of bits we allocate\n // to `linesofar`.\n if (this.distance > MAX_LINE_DISTANCE / 2 && this.totalDistance === 0) {\n this.distance = 0;\n this.updateScaledDistance();\n this.addCurrentVertex(p, normal, endLeft, endRight, segment, round);\n }\n }\n\n addHalfVertex({x, y}: Point, extrudeX: number, extrudeY: number, round: boolean, up: boolean, dir: number, segment: Segment) {\n const totalDistance = this.lineClips ? this.scaledDistance * (MAX_LINE_DISTANCE - 1) : this.scaledDistance;\n // scale down so that we can store longer distances while sacrificing precision.\n const linesofarScaled = totalDistance * LINE_DISTANCE_SCALE;\n\n this.layoutVertexArray.emplaceBack(\n // a_pos_normal\n // Encode round/up the least significant bits\n (x << 1) + (round ? 1 : 0),\n (y << 1) + (up ? 1 : 0),\n // a_data\n // add 128 to store a byte in an unsigned byte\n Math.round(EXTRUDE_SCALE * extrudeX) + 128,\n Math.round(EXTRUDE_SCALE * extrudeY) + 128,\n // Encode the -1/0/1 direction value into the first two bits of .z of a_data.\n // Combine it with the lower 6 bits of `linesofarScaled` (shifted by 2 bits to make\n // room for the direction value). The upper 8 bits of `linesofarScaled` are placed in\n // the `w` component.\n ((dir === 0 ? 0 : (dir < 0 ? -1 : 1)) + 1) | ((linesofarScaled & 0x3F) << 2),\n linesofarScaled >> 6);\n\n // Constructs a second vertex buffer with higher precision line progress\n if (this.lineClips) {\n const progressRealigned = this.scaledDistance - this.lineClips.start;\n const endClipRealigned = this.lineClips.end - this.lineClips.start;\n const uvX = progressRealigned / endClipRealigned;\n this.layoutVertexArray2.emplaceBack(uvX, this.lineClipsArray.length);\n }\n\n const e = segment.vertexLength++;\n if (this.e1 >= 0 && this.e2 >= 0) {\n this.indexArray.emplaceBack(this.e1, this.e2, e);\n segment.primitiveLength++;\n }\n if (up) {\n this.e2 = e;\n } else {\n this.e1 = e;\n }\n }\n\n updateScaledDistance() {\n // Knowing the ratio of the full linestring covered by this tiled feature, as well\n // as the total distance (in tile units) of this tiled feature, and the distance\n // (in tile units) of the current vertex, we can determine the relative distance\n // of this vertex along the full linestring feature and scale it to [0, 2^15)\n this.scaledDistance = this.lineClips ?\n this.lineClips.start + (this.lineClips.end - this.lineClips.start) * this.distance / this.totalDistance :\n this.distance;\n }\n\n updateDistance(prev: Point, next: Point) {\n this.distance += prev.dist(next);\n this.updateScaledDistance();\n }\n}\n\nregister('LineBucket', LineBucket, {omit: ['layers', 'patternFeatures']});\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type LineLayoutProps = {\n \"line-cap\": DataConstantProperty<\"butt\" | \"round\" | \"square\">,\n \"line-join\": DataDrivenProperty<\"bevel\" | \"round\" | \"miter\">,\n \"line-miter-limit\": DataConstantProperty,\n \"line-round-limit\": DataConstantProperty,\n \"line-sort-key\": DataDrivenProperty,\n};\n\nexport type LineLayoutPropsPossiblyEvaluated = {\n \"line-cap\": \"butt\" | \"round\" | \"square\",\n \"line-join\": PossiblyEvaluatedPropertyValue<\"bevel\" | \"round\" | \"miter\">,\n \"line-miter-limit\": number,\n \"line-round-limit\": number,\n \"line-sort-key\": PossiblyEvaluatedPropertyValue,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"line-cap\": new DataConstantProperty(styleSpec[\"layout_line\"][\"line-cap\"] as any as StylePropertySpecification),\n \"line-join\": new DataDrivenProperty(styleSpec[\"layout_line\"][\"line-join\"] as any as StylePropertySpecification),\n \"line-miter-limit\": new DataConstantProperty(styleSpec[\"layout_line\"][\"line-miter-limit\"] as any as StylePropertySpecification),\n \"line-round-limit\": new DataConstantProperty(styleSpec[\"layout_line\"][\"line-round-limit\"] as any as StylePropertySpecification),\n \"line-sort-key\": new DataDrivenProperty(styleSpec[\"layout_line\"][\"line-sort-key\"] as any as StylePropertySpecification),\n});\n\nexport type LinePaintProps = {\n \"line-opacity\": DataDrivenProperty,\n \"line-color\": DataDrivenProperty,\n \"line-translate\": DataConstantProperty<[number, number]>,\n \"line-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"line-width\": DataDrivenProperty,\n \"line-gap-width\": DataDrivenProperty,\n \"line-offset\": DataDrivenProperty,\n \"line-blur\": DataDrivenProperty,\n \"line-dasharray\": CrossFadedProperty>,\n \"line-pattern\": CrossFadedDataDrivenProperty,\n \"line-gradient\": ColorRampProperty,\n};\n\nexport type LinePaintPropsPossiblyEvaluated = {\n \"line-opacity\": PossiblyEvaluatedPropertyValue,\n \"line-color\": PossiblyEvaluatedPropertyValue,\n \"line-translate\": [number, number],\n \"line-translate-anchor\": \"map\" | \"viewport\",\n \"line-width\": PossiblyEvaluatedPropertyValue,\n \"line-gap-width\": PossiblyEvaluatedPropertyValue,\n \"line-offset\": PossiblyEvaluatedPropertyValue,\n \"line-blur\": PossiblyEvaluatedPropertyValue,\n \"line-dasharray\": CrossFaded>,\n \"line-pattern\": PossiblyEvaluatedPropertyValue>,\n \"line-gradient\": ColorRampProperty,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"line-opacity\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-opacity\"] as any as StylePropertySpecification),\n \"line-color\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-color\"] as any as StylePropertySpecification),\n \"line-translate\": new DataConstantProperty(styleSpec[\"paint_line\"][\"line-translate\"] as any as StylePropertySpecification),\n \"line-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_line\"][\"line-translate-anchor\"] as any as StylePropertySpecification),\n \"line-width\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-width\"] as any as StylePropertySpecification),\n \"line-gap-width\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-gap-width\"] as any as StylePropertySpecification),\n \"line-offset\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-offset\"] as any as StylePropertySpecification),\n \"line-blur\": new DataDrivenProperty(styleSpec[\"paint_line\"][\"line-blur\"] as any as StylePropertySpecification),\n \"line-dasharray\": new CrossFadedProperty(styleSpec[\"paint_line\"][\"line-dasharray\"] as any as StylePropertySpecification),\n \"line-pattern\": new CrossFadedDataDrivenProperty(styleSpec[\"paint_line\"][\"line-pattern\"] as any as StylePropertySpecification),\n \"line-gradient\": new ColorRampProperty(styleSpec[\"paint_line\"][\"line-gradient\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import Point from '@mapbox/point-geometry';\n\nimport {StyleLayer} from '../style_layer';\nimport {LineBucket} from '../../data/bucket/line_bucket';\nimport {polygonIntersectsBufferedMultiLine} from '../../util/intersection_tests';\nimport {getMaximumPaintValue, translateDistance, translate, offsetLine} from '../query_utils';\nimport properties, {LineLayoutPropsPossiblyEvaluated, LinePaintPropsPossiblyEvaluated} from './line_style_layer_properties.g';\nimport {extend} from '../../util/util';\nimport {EvaluationParameters} from '../evaluation_parameters';\nimport {Transitionable, Transitioning, Layout, PossiblyEvaluated, DataDrivenProperty} from '../properties';\n\nimport {isZoomExpression, Step} from '@maplibre/maplibre-gl-style-spec';\nimport type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Bucket, BucketParameters} from '../../data/bucket';\nimport type {LineLayoutProps, LinePaintProps} from './line_style_layer_properties.g';\nimport type {Transform} from '../../geo/transform';\nimport type {VectorTileFeature} from '@mapbox/vector-tile';\n\nexport class LineFloorwidthProperty extends DataDrivenProperty {\n useIntegerZoom: true;\n\n possiblyEvaluate(value, parameters) {\n parameters = new EvaluationParameters(Math.floor(parameters.zoom), {\n now: parameters.now,\n fadeDuration: parameters.fadeDuration,\n zoomHistory: parameters.zoomHistory,\n transition: parameters.transition\n });\n return super.possiblyEvaluate(value, parameters);\n }\n\n evaluate(value, globals, feature, featureState) {\n globals = extend({}, globals, {zoom: Math.floor(globals.zoom)});\n return super.evaluate(value, globals, feature, featureState);\n }\n}\n\nlet lineFloorwidthProperty: LineFloorwidthProperty;\n\nexport class LineStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n gradientVersion: number;\n stepInterpolant: boolean;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n this.gradientVersion = 0;\n if (!lineFloorwidthProperty) {\n lineFloorwidthProperty =\n new LineFloorwidthProperty(properties.paint.properties['line-width'].specification);\n lineFloorwidthProperty.useIntegerZoom = true;\n }\n }\n\n _handleSpecialPaintPropertyUpdate(name: string) {\n if (name === 'line-gradient') {\n const expression = this.gradientExpression();\n if (isZoomExpression(expression)) {\n this.stepInterpolant = expression._styleExpression.expression instanceof Step;\n } else {\n this.stepInterpolant = false;\n }\n this.gradientVersion = (this.gradientVersion + 1) % Number.MAX_SAFE_INTEGER;\n }\n }\n\n gradientExpression() {\n return this._transitionablePaint._values['line-gradient'].value.expression;\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n super.recalculate(parameters, availableImages);\n (this.paint._values as any)['line-floorwidth'] =\n lineFloorwidthProperty.possiblyEvaluate(this._transitioningPaint._values['line-width'].value, parameters);\n }\n\n createBucket(parameters: BucketParameters) {\n return new LineBucket(parameters);\n }\n\n queryRadius(bucket: Bucket): number {\n const lineBucket: LineBucket = (bucket as any);\n const width = getLineWidth(\n getMaximumPaintValue('line-width', this, lineBucket),\n getMaximumPaintValue('line-gap-width', this, lineBucket));\n const offset = getMaximumPaintValue('line-offset', this, lineBucket);\n return width / 2 + Math.abs(offset) + translateDistance(this.paint.get('line-translate'));\n }\n\n queryIntersectsFeature(\n queryGeometry: Array,\n feature: VectorTileFeature,\n featureState: FeatureState,\n geometry: Array>,\n zoom: number,\n transform: Transform,\n pixelsToTileUnits: number\n ): boolean {\n const translatedPolygon = translate(queryGeometry,\n this.paint.get('line-translate'),\n this.paint.get('line-translate-anchor'),\n transform.angle, pixelsToTileUnits);\n const halfWidth = pixelsToTileUnits / 2 * getLineWidth(\n this.paint.get('line-width').evaluate(feature, featureState),\n this.paint.get('line-gap-width').evaluate(feature, featureState));\n const lineOffset = this.paint.get('line-offset').evaluate(feature, featureState);\n if (lineOffset) {\n geometry = offsetLine(geometry, lineOffset * pixelsToTileUnits);\n }\n\n return polygonIntersectsBufferedMultiLine(translatedPolygon, geometry, halfWidth);\n }\n\n isTileClipped() {\n return true;\n }\n}\n\nfunction getLineWidth(lineWidth, lineGapWidth) {\n if (lineGapWidth > 0) {\n return lineGapWidth + 2 * lineWidth;\n } else {\n return lineWidth;\n }\n}\n","import {createLayout} from '../../util/struct_array';\n\nexport const symbolLayoutAttributes = createLayout([\n {name: 'a_pos_offset', components: 4, type: 'Int16'},\n {name: 'a_data', components: 4, type: 'Uint16'},\n {name: 'a_pixeloffset', components: 4, type: 'Int16'}\n], 4);\n\nexport const dynamicLayoutAttributes = createLayout([\n {name: 'a_projected_pos', components: 3, type: 'Float32'}\n], 4);\n\nexport const placementOpacityAttributes = createLayout([\n {name: 'a_fade_opacity', components: 1, type: 'Uint32'}\n], 4);\n\nexport const collisionVertexAttributes = createLayout([\n {name: 'a_placed', components: 2, type: 'Uint8'},\n {name: 'a_shift', components: 2, type: 'Float32'}\n]);\n\nexport const collisionBox = createLayout([\n // the box is centered around the anchor point\n {type: 'Int16', name: 'anchorPointX'},\n {type: 'Int16', name: 'anchorPointY'},\n\n // distances to the edges from the anchor\n {type: 'Int16', name: 'x1'},\n {type: 'Int16', name: 'y1'},\n {type: 'Int16', name: 'x2'},\n {type: 'Int16', name: 'y2'},\n\n // the index of the feature in the original vectortile\n {type: 'Uint32', name: 'featureIndex'},\n // the source layer the feature appears in\n {type: 'Uint16', name: 'sourceLayerIndex'},\n // the bucket the feature appears in\n {type: 'Uint16', name: 'bucketIndex'},\n]);\n\nexport const collisionBoxLayout = createLayout([ // used to render collision boxes for debugging purposes\n {name: 'a_pos', components: 2, type: 'Int16'},\n {name: 'a_anchor_pos', components: 2, type: 'Int16'},\n {name: 'a_extrude', components: 2, type: 'Int16'}\n], 4);\n\nexport const collisionCircleLayout = createLayout([ // used to render collision circles for debugging purposes\n {name: 'a_pos', components: 2, type: 'Float32'},\n {name: 'a_radius', components: 1, type: 'Float32'},\n {name: 'a_flags', components: 2, type: 'Int16'}\n], 4);\n\nexport const quadTriangle = createLayout([\n {name: 'triangle', components: 3, type: 'Uint16'},\n]);\n\nexport const placement = createLayout([\n {type: 'Int16', name: 'anchorX'},\n {type: 'Int16', name: 'anchorY'},\n {type: 'Uint16', name: 'glyphStartIndex'},\n {type: 'Uint16', name: 'numGlyphs'},\n {type: 'Uint32', name: 'vertexStartIndex'},\n {type: 'Uint32', name: 'lineStartIndex'},\n {type: 'Uint32', name: 'lineLength'},\n {type: 'Uint16', name: 'segment'},\n {type: 'Uint16', name: 'lowerSize'},\n {type: 'Uint16', name: 'upperSize'},\n {type: 'Float32', name: 'lineOffsetX'},\n {type: 'Float32', name: 'lineOffsetY'},\n {type: 'Uint8', name: 'writingMode'},\n {type: 'Uint8', name: 'placedOrientation'},\n {type: 'Uint8', name: 'hidden'},\n {type: 'Uint32', name: 'crossTileID'},\n {type: 'Int16', name: 'associatedIconIndex'}\n]);\n\nexport const symbolInstance = createLayout([\n {type: 'Int16', name: 'anchorX'},\n {type: 'Int16', name: 'anchorY'},\n {type: 'Int16', name: 'rightJustifiedTextSymbolIndex'},\n {type: 'Int16', name: 'centerJustifiedTextSymbolIndex'},\n {type: 'Int16', name: 'leftJustifiedTextSymbolIndex'},\n {type: 'Int16', name: 'verticalPlacedTextSymbolIndex'},\n {type: 'Int16', name: 'placedIconSymbolIndex'},\n {type: 'Int16', name: 'verticalPlacedIconSymbolIndex'},\n {type: 'Uint16', name: 'key'},\n {type: 'Uint16', name: 'textBoxStartIndex'},\n {type: 'Uint16', name: 'textBoxEndIndex'},\n {type: 'Uint16', name: 'verticalTextBoxStartIndex'},\n {type: 'Uint16', name: 'verticalTextBoxEndIndex'},\n {type: 'Uint16', name: 'iconBoxStartIndex'},\n {type: 'Uint16', name: 'iconBoxEndIndex'},\n {type: 'Uint16', name: 'verticalIconBoxStartIndex'},\n {type: 'Uint16', name: 'verticalIconBoxEndIndex'},\n {type: 'Uint16', name: 'featureIndex'},\n {type: 'Uint16', name: 'numHorizontalGlyphVertices'},\n {type: 'Uint16', name: 'numVerticalGlyphVertices'},\n {type: 'Uint16', name: 'numIconVertices'},\n {type: 'Uint16', name: 'numVerticalIconVertices'},\n {type: 'Uint16', name: 'useRuntimeCollisionCircles'},\n {type: 'Uint32', name: 'crossTileID'},\n {type: 'Float32', name: 'textBoxScale'},\n {type: 'Float32', name: 'collisionCircleDiameter'},\n {type: 'Uint16', name: 'textAnchorOffsetStartIndex'},\n {type: 'Uint16', name: 'textAnchorOffsetEndIndex'}\n]);\n\nexport const glyphOffset = createLayout([\n {type: 'Float32', name: 'offsetX'}\n]);\n\nexport const lineVertex = createLayout([\n {type: 'Int16', name: 'x'},\n {type: 'Int16', name: 'y'},\n {type: 'Int16', name: 'tileUnitDistanceFromAnchor'}\n]);\n\nexport const textAnchorOffset = createLayout([\n {type: 'Uint16', name: 'textAnchor'},\n {type: 'Float32', components: 2, name: 'textOffset'}\n]);\n","import {plugin as rtlTextPlugin} from '../source/rtl_text_plugin';\n\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\nimport type {Feature} from '@maplibre/maplibre-gl-style-spec';\nimport {Formatted} from '@maplibre/maplibre-gl-style-spec';\n\nfunction transformTextInternal(text: string, layer: SymbolStyleLayer, feature: Feature) {\n const transform = layer.layout.get('text-transform').evaluate(feature, {});\n if (transform === 'uppercase') {\n text = text.toLocaleUpperCase();\n } else if (transform === 'lowercase') {\n text = text.toLocaleLowerCase();\n }\n\n if (rtlTextPlugin.applyArabicShaping) {\n text = rtlTextPlugin.applyArabicShaping(text);\n }\n\n return text;\n}\n\nexport function transformText(text: Formatted, layer: SymbolStyleLayer, feature: Feature): Formatted {\n text.sections.forEach(section => {\n section.text = transformTextInternal(section.text, layer, feature);\n });\n return text;\n}\n","import {charHasRotatedVerticalOrientation} from './script_detection';\n\nexport const verticalizedCharacterMap = {\n '!': '︕',\n '#': '#',\n '$': '$',\n '%': '%',\n '&': '&',\n '(': '︵',\n ')': '︶',\n '*': '*',\n '+': '+',\n ',': '︐',\n '-': '︲',\n '.': '・',\n '/': '/',\n ':': '︓',\n ';': '︔',\n '<': '︿',\n '=': '=',\n '>': '﹀',\n '?': '︖',\n '@': '@',\n '[': '﹇',\n '\\\\': '\',\n ']': '﹈',\n '^': '^',\n '_': '︳',\n '`': '`',\n '{': '︷',\n '|': '―',\n '}': '︸',\n '~': '~',\n '¢': '¢',\n '£': '£',\n '¥': '¥',\n '¦': '¦',\n '¬': '¬',\n '¯': ' ̄',\n '–': '︲',\n '—': '︱',\n '‘': '﹃',\n '’': '﹄',\n '“': '﹁',\n '”': '﹂',\n '…': '︙',\n '‧': '・',\n '₩': '₩',\n '、': '︑',\n '。': '︒',\n '〈': '︿',\n '〉': '﹀',\n '《': '︽',\n '》': '︾',\n '「': '﹁',\n '」': '﹂',\n '『': '﹃',\n '』': '﹄',\n '【': '︻',\n '】': '︼',\n '〔': '︹',\n '〕': '︺',\n '〖': '︗',\n '〗': '︘',\n '!': '︕',\n '(': '︵',\n ')': '︶',\n ',': '︐',\n '-': '︲',\n '.': '・',\n ':': '︓',\n ';': '︔',\n '<': '︿',\n '>': '﹀',\n '?': '︖',\n '[': '﹇',\n ']': '﹈',\n '_': '︳',\n '{': '︷',\n '|': '―',\n '}': '︸',\n '⦅': '︵',\n '⦆': '︶',\n '。': '︒',\n '「': '﹁',\n '」': '﹂'\n};\n\nexport function verticalizePunctuation(input: string) {\n let output = '';\n\n for (let i = 0; i < input.length; i++) {\n const nextCharCode = input.charCodeAt(i + 1) || null;\n const prevCharCode = input.charCodeAt(i - 1) || null;\n\n const canReplacePunctuation = (\n (!nextCharCode || !charHasRotatedVerticalOrientation(nextCharCode) || verticalizedCharacterMap[input[i + 1]]) &&\n (!prevCharCode || !charHasRotatedVerticalOrientation(prevCharCode) || verticalizedCharacterMap[input[i - 1]])\n );\n\n if (canReplacePunctuation && verticalizedCharacterMap[input[i]]) {\n output += verticalizedCharacterMap[input[i]];\n } else {\n output += input[i];\n }\n }\n\n return output;\n}\n\n","// ONE_EM constant used to go between \"em\" units used in style spec and \"points\" used internally for layout\n\nexport default 24;\n","'use strict';\n\nmodule.exports = Pbf;\n\nvar ieee754 = require('ieee754');\n\nfunction Pbf(buf) {\n this.buf = ArrayBuffer.isView && ArrayBuffer.isView(buf) ? buf : new Uint8Array(buf || 0);\n this.pos = 0;\n this.type = 0;\n this.length = this.buf.length;\n}\n\nPbf.Varint = 0; // varint: int32, int64, uint32, uint64, sint32, sint64, bool, enum\nPbf.Fixed64 = 1; // 64-bit: double, fixed64, sfixed64\nPbf.Bytes = 2; // length-delimited: string, bytes, embedded messages, packed repeated fields\nPbf.Fixed32 = 5; // 32-bit: float, fixed32, sfixed32\n\nvar SHIFT_LEFT_32 = (1 << 16) * (1 << 16),\n SHIFT_RIGHT_32 = 1 / SHIFT_LEFT_32;\n\n// Threshold chosen based on both benchmarking and knowledge about browser string\n// data structures (which currently switch structure types at 12 bytes or more)\nvar TEXT_DECODER_MIN_LENGTH = 12;\nvar utf8TextDecoder = typeof TextDecoder === 'undefined' ? null : new TextDecoder('utf8');\n\nPbf.prototype = {\n\n destroy: function() {\n this.buf = null;\n },\n\n // === READING =================================================================\n\n readFields: function(readField, result, end) {\n end = end || this.length;\n\n while (this.pos < end) {\n var val = this.readVarint(),\n tag = val >> 3,\n startPos = this.pos;\n\n this.type = val & 0x7;\n readField(tag, result, this);\n\n if (this.pos === startPos) this.skip(val);\n }\n return result;\n },\n\n readMessage: function(readField, result) {\n return this.readFields(readField, result, this.readVarint() + this.pos);\n },\n\n readFixed32: function() {\n var val = readUInt32(this.buf, this.pos);\n this.pos += 4;\n return val;\n },\n\n readSFixed32: function() {\n var val = readInt32(this.buf, this.pos);\n this.pos += 4;\n return val;\n },\n\n // 64-bit int handling is based on github.com/dpw/node-buffer-more-ints (MIT-licensed)\n\n readFixed64: function() {\n var val = readUInt32(this.buf, this.pos) + readUInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;\n this.pos += 8;\n return val;\n },\n\n readSFixed64: function() {\n var val = readUInt32(this.buf, this.pos) + readInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;\n this.pos += 8;\n return val;\n },\n\n readFloat: function() {\n var val = ieee754.read(this.buf, this.pos, true, 23, 4);\n this.pos += 4;\n return val;\n },\n\n readDouble: function() {\n var val = ieee754.read(this.buf, this.pos, true, 52, 8);\n this.pos += 8;\n return val;\n },\n\n readVarint: function(isSigned) {\n var buf = this.buf,\n val, b;\n\n b = buf[this.pos++]; val = b & 0x7f; if (b < 0x80) return val;\n b = buf[this.pos++]; val |= (b & 0x7f) << 7; if (b < 0x80) return val;\n b = buf[this.pos++]; val |= (b & 0x7f) << 14; if (b < 0x80) return val;\n b = buf[this.pos++]; val |= (b & 0x7f) << 21; if (b < 0x80) return val;\n b = buf[this.pos]; val |= (b & 0x0f) << 28;\n\n return readVarintRemainder(val, isSigned, this);\n },\n\n readVarint64: function() { // for compatibility with v2.0.1\n return this.readVarint(true);\n },\n\n readSVarint: function() {\n var num = this.readVarint();\n return num % 2 === 1 ? (num + 1) / -2 : num / 2; // zigzag encoding\n },\n\n readBoolean: function() {\n return Boolean(this.readVarint());\n },\n\n readString: function() {\n var end = this.readVarint() + this.pos;\n var pos = this.pos;\n this.pos = end;\n\n if (end - pos >= TEXT_DECODER_MIN_LENGTH && utf8TextDecoder) {\n // longer strings are fast with the built-in browser TextDecoder API\n return readUtf8TextDecoder(this.buf, pos, end);\n }\n // short strings are fast with our custom implementation\n return readUtf8(this.buf, pos, end);\n },\n\n readBytes: function() {\n var end = this.readVarint() + this.pos,\n buffer = this.buf.subarray(this.pos, end);\n this.pos = end;\n return buffer;\n },\n\n // verbose for performance reasons; doesn't affect gzipped size\n\n readPackedVarint: function(arr, isSigned) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readVarint(isSigned));\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readVarint(isSigned));\n return arr;\n },\n readPackedSVarint: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readSVarint());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readSVarint());\n return arr;\n },\n readPackedBoolean: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readBoolean());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readBoolean());\n return arr;\n },\n readPackedFloat: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readFloat());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readFloat());\n return arr;\n },\n readPackedDouble: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readDouble());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readDouble());\n return arr;\n },\n readPackedFixed32: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readFixed32());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readFixed32());\n return arr;\n },\n readPackedSFixed32: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed32());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readSFixed32());\n return arr;\n },\n readPackedFixed64: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readFixed64());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readFixed64());\n return arr;\n },\n readPackedSFixed64: function(arr) {\n if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed64());\n var end = readPackedEnd(this);\n arr = arr || [];\n while (this.pos < end) arr.push(this.readSFixed64());\n return arr;\n },\n\n skip: function(val) {\n var type = val & 0x7;\n if (type === Pbf.Varint) while (this.buf[this.pos++] > 0x7f) {}\n else if (type === Pbf.Bytes) this.pos = this.readVarint() + this.pos;\n else if (type === Pbf.Fixed32) this.pos += 4;\n else if (type === Pbf.Fixed64) this.pos += 8;\n else throw new Error('Unimplemented type: ' + type);\n },\n\n // === WRITING =================================================================\n\n writeTag: function(tag, type) {\n this.writeVarint((tag << 3) | type);\n },\n\n realloc: function(min) {\n var length = this.length || 16;\n\n while (length < this.pos + min) length *= 2;\n\n if (length !== this.length) {\n var buf = new Uint8Array(length);\n buf.set(this.buf);\n this.buf = buf;\n this.length = length;\n }\n },\n\n finish: function() {\n this.length = this.pos;\n this.pos = 0;\n return this.buf.subarray(0, this.length);\n },\n\n writeFixed32: function(val) {\n this.realloc(4);\n writeInt32(this.buf, val, this.pos);\n this.pos += 4;\n },\n\n writeSFixed32: function(val) {\n this.realloc(4);\n writeInt32(this.buf, val, this.pos);\n this.pos += 4;\n },\n\n writeFixed64: function(val) {\n this.realloc(8);\n writeInt32(this.buf, val & -1, this.pos);\n writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4);\n this.pos += 8;\n },\n\n writeSFixed64: function(val) {\n this.realloc(8);\n writeInt32(this.buf, val & -1, this.pos);\n writeInt32(this.buf, Math.floor(val * SHIFT_RIGHT_32), this.pos + 4);\n this.pos += 8;\n },\n\n writeVarint: function(val) {\n val = +val || 0;\n\n if (val > 0xfffffff || val < 0) {\n writeBigVarint(val, this);\n return;\n }\n\n this.realloc(4);\n\n this.buf[this.pos++] = val & 0x7f | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return;\n this.buf[this.pos++] = ((val >>>= 7) & 0x7f) | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return;\n this.buf[this.pos++] = ((val >>>= 7) & 0x7f) | (val > 0x7f ? 0x80 : 0); if (val <= 0x7f) return;\n this.buf[this.pos++] = (val >>> 7) & 0x7f;\n },\n\n writeSVarint: function(val) {\n this.writeVarint(val < 0 ? -val * 2 - 1 : val * 2);\n },\n\n writeBoolean: function(val) {\n this.writeVarint(Boolean(val));\n },\n\n writeString: function(str) {\n str = String(str);\n this.realloc(str.length * 4);\n\n this.pos++; // reserve 1 byte for short string length\n\n var startPos = this.pos;\n // write the string directly to the buffer and see how much was written\n this.pos = writeUtf8(this.buf, str, this.pos);\n var len = this.pos - startPos;\n\n if (len >= 0x80) makeRoomForExtraLength(startPos, len, this);\n\n // finally, write the message length in the reserved place and restore the position\n this.pos = startPos - 1;\n this.writeVarint(len);\n this.pos += len;\n },\n\n writeFloat: function(val) {\n this.realloc(4);\n ieee754.write(this.buf, val, this.pos, true, 23, 4);\n this.pos += 4;\n },\n\n writeDouble: function(val) {\n this.realloc(8);\n ieee754.write(this.buf, val, this.pos, true, 52, 8);\n this.pos += 8;\n },\n\n writeBytes: function(buffer) {\n var len = buffer.length;\n this.writeVarint(len);\n this.realloc(len);\n for (var i = 0; i < len; i++) this.buf[this.pos++] = buffer[i];\n },\n\n writeRawMessage: function(fn, obj) {\n this.pos++; // reserve 1 byte for short message length\n\n // write the message directly to the buffer and see how much was written\n var startPos = this.pos;\n fn(obj, this);\n var len = this.pos - startPos;\n\n if (len >= 0x80) makeRoomForExtraLength(startPos, len, this);\n\n // finally, write the message length in the reserved place and restore the position\n this.pos = startPos - 1;\n this.writeVarint(len);\n this.pos += len;\n },\n\n writeMessage: function(tag, fn, obj) {\n this.writeTag(tag, Pbf.Bytes);\n this.writeRawMessage(fn, obj);\n },\n\n writePackedVarint: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedVarint, arr); },\n writePackedSVarint: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSVarint, arr); },\n writePackedBoolean: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedBoolean, arr); },\n writePackedFloat: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFloat, arr); },\n writePackedDouble: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedDouble, arr); },\n writePackedFixed32: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFixed32, arr); },\n writePackedSFixed32: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSFixed32, arr); },\n writePackedFixed64: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedFixed64, arr); },\n writePackedSFixed64: function(tag, arr) { if (arr.length) this.writeMessage(tag, writePackedSFixed64, arr); },\n\n writeBytesField: function(tag, buffer) {\n this.writeTag(tag, Pbf.Bytes);\n this.writeBytes(buffer);\n },\n writeFixed32Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed32);\n this.writeFixed32(val);\n },\n writeSFixed32Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed32);\n this.writeSFixed32(val);\n },\n writeFixed64Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed64);\n this.writeFixed64(val);\n },\n writeSFixed64Field: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed64);\n this.writeSFixed64(val);\n },\n writeVarintField: function(tag, val) {\n this.writeTag(tag, Pbf.Varint);\n this.writeVarint(val);\n },\n writeSVarintField: function(tag, val) {\n this.writeTag(tag, Pbf.Varint);\n this.writeSVarint(val);\n },\n writeStringField: function(tag, str) {\n this.writeTag(tag, Pbf.Bytes);\n this.writeString(str);\n },\n writeFloatField: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed32);\n this.writeFloat(val);\n },\n writeDoubleField: function(tag, val) {\n this.writeTag(tag, Pbf.Fixed64);\n this.writeDouble(val);\n },\n writeBooleanField: function(tag, val) {\n this.writeVarintField(tag, Boolean(val));\n }\n};\n\nfunction readVarintRemainder(l, s, p) {\n var buf = p.buf,\n h, b;\n\n b = buf[p.pos++]; h = (b & 0x70) >> 4; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 3; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 10; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 17; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x7f) << 24; if (b < 0x80) return toNum(l, h, s);\n b = buf[p.pos++]; h |= (b & 0x01) << 31; if (b < 0x80) return toNum(l, h, s);\n\n throw new Error('Expected varint not more than 10 bytes');\n}\n\nfunction readPackedEnd(pbf) {\n return pbf.type === Pbf.Bytes ?\n pbf.readVarint() + pbf.pos : pbf.pos + 1;\n}\n\nfunction toNum(low, high, isSigned) {\n if (isSigned) {\n return high * 0x100000000 + (low >>> 0);\n }\n\n return ((high >>> 0) * 0x100000000) + (low >>> 0);\n}\n\nfunction writeBigVarint(val, pbf) {\n var low, high;\n\n if (val >= 0) {\n low = (val % 0x100000000) | 0;\n high = (val / 0x100000000) | 0;\n } else {\n low = ~(-val % 0x100000000);\n high = ~(-val / 0x100000000);\n\n if (low ^ 0xffffffff) {\n low = (low + 1) | 0;\n } else {\n low = 0;\n high = (high + 1) | 0;\n }\n }\n\n if (val >= 0x10000000000000000 || val < -0x10000000000000000) {\n throw new Error('Given varint doesn\\'t fit into 10 bytes');\n }\n\n pbf.realloc(10);\n\n writeBigVarintLow(low, high, pbf);\n writeBigVarintHigh(high, pbf);\n}\n\nfunction writeBigVarintLow(low, high, pbf) {\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos++] = low & 0x7f | 0x80; low >>>= 7;\n pbf.buf[pbf.pos] = low & 0x7f;\n}\n\nfunction writeBigVarintHigh(high, pbf) {\n var lsb = (high & 0x07) << 4;\n\n pbf.buf[pbf.pos++] |= lsb | ((high >>>= 3) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f | ((high >>>= 7) ? 0x80 : 0); if (!high) return;\n pbf.buf[pbf.pos++] = high & 0x7f;\n}\n\nfunction makeRoomForExtraLength(startPos, len, pbf) {\n var extraLen =\n len <= 0x3fff ? 1 :\n len <= 0x1fffff ? 2 :\n len <= 0xfffffff ? 3 : Math.floor(Math.log(len) / (Math.LN2 * 7));\n\n // if 1 byte isn't enough for encoding message length, shift the data to the right\n pbf.realloc(extraLen);\n for (var i = pbf.pos - 1; i >= startPos; i--) pbf.buf[i + extraLen] = pbf.buf[i];\n}\n\nfunction writePackedVarint(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeVarint(arr[i]); }\nfunction writePackedSVarint(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSVarint(arr[i]); }\nfunction writePackedFloat(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFloat(arr[i]); }\nfunction writePackedDouble(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeDouble(arr[i]); }\nfunction writePackedBoolean(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeBoolean(arr[i]); }\nfunction writePackedFixed32(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFixed32(arr[i]); }\nfunction writePackedSFixed32(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSFixed32(arr[i]); }\nfunction writePackedFixed64(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeFixed64(arr[i]); }\nfunction writePackedSFixed64(arr, pbf) { for (var i = 0; i < arr.length; i++) pbf.writeSFixed64(arr[i]); }\n\n// Buffer code below from https://github.com/feross/buffer, MIT-licensed\n\nfunction readUInt32(buf, pos) {\n return ((buf[pos]) |\n (buf[pos + 1] << 8) |\n (buf[pos + 2] << 16)) +\n (buf[pos + 3] * 0x1000000);\n}\n\nfunction writeInt32(buf, val, pos) {\n buf[pos] = val;\n buf[pos + 1] = (val >>> 8);\n buf[pos + 2] = (val >>> 16);\n buf[pos + 3] = (val >>> 24);\n}\n\nfunction readInt32(buf, pos) {\n return ((buf[pos]) |\n (buf[pos + 1] << 8) |\n (buf[pos + 2] << 16)) +\n (buf[pos + 3] << 24);\n}\n\nfunction readUtf8(buf, pos, end) {\n var str = '';\n var i = pos;\n\n while (i < end) {\n var b0 = buf[i];\n var c = null; // codepoint\n var bytesPerSequence =\n b0 > 0xEF ? 4 :\n b0 > 0xDF ? 3 :\n b0 > 0xBF ? 2 : 1;\n\n if (i + bytesPerSequence > end) break;\n\n var b1, b2, b3;\n\n if (bytesPerSequence === 1) {\n if (b0 < 0x80) {\n c = b0;\n }\n } else if (bytesPerSequence === 2) {\n b1 = buf[i + 1];\n if ((b1 & 0xC0) === 0x80) {\n c = (b0 & 0x1F) << 0x6 | (b1 & 0x3F);\n if (c <= 0x7F) {\n c = null;\n }\n }\n } else if (bytesPerSequence === 3) {\n b1 = buf[i + 1];\n b2 = buf[i + 2];\n if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80) {\n c = (b0 & 0xF) << 0xC | (b1 & 0x3F) << 0x6 | (b2 & 0x3F);\n if (c <= 0x7FF || (c >= 0xD800 && c <= 0xDFFF)) {\n c = null;\n }\n }\n } else if (bytesPerSequence === 4) {\n b1 = buf[i + 1];\n b2 = buf[i + 2];\n b3 = buf[i + 3];\n if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80 && (b3 & 0xC0) === 0x80) {\n c = (b0 & 0xF) << 0x12 | (b1 & 0x3F) << 0xC | (b2 & 0x3F) << 0x6 | (b3 & 0x3F);\n if (c <= 0xFFFF || c >= 0x110000) {\n c = null;\n }\n }\n }\n\n if (c === null) {\n c = 0xFFFD;\n bytesPerSequence = 1;\n\n } else if (c > 0xFFFF) {\n c -= 0x10000;\n str += String.fromCharCode(c >>> 10 & 0x3FF | 0xD800);\n c = 0xDC00 | c & 0x3FF;\n }\n\n str += String.fromCharCode(c);\n i += bytesPerSequence;\n }\n\n return str;\n}\n\nfunction readUtf8TextDecoder(buf, pos, end) {\n return utf8TextDecoder.decode(buf.subarray(pos, end));\n}\n\nfunction writeUtf8(buf, str, pos) {\n for (var i = 0, c, lead; i < str.length; i++) {\n c = str.charCodeAt(i); // code point\n\n if (c > 0xD7FF && c < 0xE000) {\n if (lead) {\n if (c < 0xDC00) {\n buf[pos++] = 0xEF;\n buf[pos++] = 0xBF;\n buf[pos++] = 0xBD;\n lead = c;\n continue;\n } else {\n c = lead - 0xD800 << 10 | c - 0xDC00 | 0x10000;\n lead = null;\n }\n } else {\n if (c > 0xDBFF || (i + 1 === str.length)) {\n buf[pos++] = 0xEF;\n buf[pos++] = 0xBF;\n buf[pos++] = 0xBD;\n } else {\n lead = c;\n }\n continue;\n }\n } else if (lead) {\n buf[pos++] = 0xEF;\n buf[pos++] = 0xBF;\n buf[pos++] = 0xBD;\n lead = null;\n }\n\n if (c < 0x80) {\n buf[pos++] = c;\n } else {\n if (c < 0x800) {\n buf[pos++] = c >> 0x6 | 0xC0;\n } else {\n if (c < 0x10000) {\n buf[pos++] = c >> 0xC | 0xE0;\n } else {\n buf[pos++] = c >> 0x12 | 0xF0;\n buf[pos++] = c >> 0xC & 0x3F | 0x80;\n }\n buf[pos++] = c >> 0x6 & 0x3F | 0x80;\n }\n buf[pos++] = c & 0x3F | 0x80;\n }\n }\n return pos;\n}\n","/*! ieee754. BSD-3-Clause License. Feross Aboukhadijeh */\nexports.read = function (buffer, offset, isLE, mLen, nBytes) {\n var e, m\n var eLen = (nBytes * 8) - mLen - 1\n var eMax = (1 << eLen) - 1\n var eBias = eMax >> 1\n var nBits = -7\n var i = isLE ? (nBytes - 1) : 0\n var d = isLE ? -1 : 1\n var s = buffer[offset + i]\n\n i += d\n\n e = s & ((1 << (-nBits)) - 1)\n s >>= (-nBits)\n nBits += eLen\n for (; nBits > 0; e = (e * 256) + buffer[offset + i], i += d, nBits -= 8) {}\n\n m = e & ((1 << (-nBits)) - 1)\n e >>= (-nBits)\n nBits += mLen\n for (; nBits > 0; m = (m * 256) + buffer[offset + i], i += d, nBits -= 8) {}\n\n if (e === 0) {\n e = 1 - eBias\n } else if (e === eMax) {\n return m ? NaN : ((s ? -1 : 1) * Infinity)\n } else {\n m = m + Math.pow(2, mLen)\n e = e - eBias\n }\n return (s ? -1 : 1) * m * Math.pow(2, e - mLen)\n}\n\nexports.write = function (buffer, value, offset, isLE, mLen, nBytes) {\n var e, m, c\n var eLen = (nBytes * 8) - mLen - 1\n var eMax = (1 << eLen) - 1\n var eBias = eMax >> 1\n var rt = (mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0)\n var i = isLE ? 0 : (nBytes - 1)\n var d = isLE ? 1 : -1\n var s = value < 0 || (value === 0 && 1 / value < 0) ? 1 : 0\n\n value = Math.abs(value)\n\n if (isNaN(value) || value === Infinity) {\n m = isNaN(value) ? 1 : 0\n e = eMax\n } else {\n e = Math.floor(Math.log(value) / Math.LN2)\n if (value * (c = Math.pow(2, -e)) < 1) {\n e--\n c *= 2\n }\n if (e + eBias >= 1) {\n value += rt / c\n } else {\n value += rt * Math.pow(2, 1 - eBias)\n }\n if (value * c >= 2) {\n e++\n c /= 2\n }\n\n if (e + eBias >= eMax) {\n m = 0\n e = eMax\n } else if (e + eBias >= 1) {\n m = ((value * c) - 1) * Math.pow(2, mLen)\n e = e + eBias\n } else {\n m = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen)\n e = 0\n }\n }\n\n for (; mLen >= 8; buffer[offset + i] = m & 0xff, i += d, m /= 256, mLen -= 8) {}\n\n e = (e << mLen) | m\n eLen += mLen\n for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8) {}\n\n buffer[offset + i - d] |= s * 128\n}\n","import {AlphaImage} from '../util/image';\n\nimport Protobuf from 'pbf';\nconst border = 3;\n\nimport type {StyleGlyph} from './style_glyph';\n\nfunction readFontstacks(tag: number, glyphs: Array, pbf: Protobuf) {\n if (tag === 1) {\n pbf.readMessage(readFontstack, glyphs);\n }\n}\n\nfunction readFontstack(tag: number, glyphs: Array, pbf: Protobuf) {\n if (tag === 3) {\n const {id, bitmap, width, height, left, top, advance} = pbf.readMessage(readGlyph, {});\n glyphs.push({\n id,\n bitmap: new AlphaImage({\n width: width + 2 * border,\n height: height + 2 * border\n }, bitmap),\n metrics: {width, height, left, top, advance}\n });\n }\n}\n\nfunction readGlyph(tag: number, glyph: any, pbf: Protobuf) {\n if (tag === 1) glyph.id = pbf.readVarint();\n else if (tag === 2) glyph.bitmap = pbf.readBytes();\n else if (tag === 3) glyph.width = pbf.readVarint();\n else if (tag === 4) glyph.height = pbf.readVarint();\n else if (tag === 5) glyph.left = pbf.readSVarint();\n else if (tag === 6) glyph.top = pbf.readSVarint();\n else if (tag === 7) glyph.advance = pbf.readVarint();\n}\n\nexport function parseGlyphPbf(data: ArrayBuffer | Uint8Array): Array {\n return new Protobuf(data).readFields(readFontstacks, []);\n}\n\nexport const GLYPH_PBF_BORDER = border;\n","\nexport default function potpack(boxes) {\n\n // calculate total box area and maximum box width\n let area = 0;\n let maxWidth = 0;\n\n for (const box of boxes) {\n area += box.w * box.h;\n maxWidth = Math.max(maxWidth, box.w);\n }\n\n // sort the boxes for insertion by height, descending\n boxes.sort((a, b) => b.h - a.h);\n\n // aim for a squarish resulting container,\n // slightly adjusted for sub-100% space utilization\n const startWidth = Math.max(Math.ceil(Math.sqrt(area / 0.95)), maxWidth);\n\n // start with a single empty space, unbounded at the bottom\n const spaces = [{x: 0, y: 0, w: startWidth, h: Infinity}];\n\n let width = 0;\n let height = 0;\n\n for (const box of boxes) {\n // look through spaces backwards so that we check smaller spaces first\n for (let i = spaces.length - 1; i >= 0; i--) {\n const space = spaces[i];\n\n // look for empty spaces that can accommodate the current box\n if (box.w > space.w || box.h > space.h) continue;\n\n // found the space; add the box to its top-left corner\n // |-------|-------|\n // | box | |\n // |_______| |\n // | space |\n // |_______________|\n box.x = space.x;\n box.y = space.y;\n\n height = Math.max(height, box.y + box.h);\n width = Math.max(width, box.x + box.w);\n\n if (box.w === space.w && box.h === space.h) {\n // space matches the box exactly; remove it\n const last = spaces.pop();\n if (i < spaces.length) spaces[i] = last;\n\n } else if (box.h === space.h) {\n // space matches the box height; update it accordingly\n // |-------|---------------|\n // | box | updated space |\n // |_______|_______________|\n space.x += box.w;\n space.w -= box.w;\n\n } else if (box.w === space.w) {\n // space matches the box width; update it accordingly\n // |---------------|\n // | box |\n // |_______________|\n // | updated space |\n // |_______________|\n space.y += box.h;\n space.h -= box.h;\n\n } else {\n // otherwise the box splits the space into two spaces\n // |-------|-----------|\n // | box | new space |\n // |_______|___________|\n // | updated space |\n // |___________________|\n spaces.push({\n x: space.x + box.w,\n y: space.y,\n w: space.w - box.w,\n h: box.h\n });\n space.y += box.h;\n space.h -= box.h;\n }\n break;\n }\n }\n\n return {\n w: width, // container width\n h: height, // container height\n fill: (area / (width * height)) || 0 // space utilization\n };\n}\n","/* eslint-disable key-spacing */\nimport {RGBAImage} from '../util/image';\nimport {register} from '../util/web_worker_transfer';\nimport potpack from 'potpack';\n\nimport type {StyleImage} from '../style/style_image';\nimport type {ImageManager} from './image_manager';\nimport type {Texture} from './texture';\nimport type {Rect} from './glyph_atlas';\n\nconst IMAGE_PADDING: number = 1;\nexport {IMAGE_PADDING};\n\nexport class ImagePosition {\n paddedRect: Rect;\n pixelRatio: number;\n version: number;\n stretchY: Array<[number, number]>;\n stretchX: Array<[number, number]>;\n content: [number, number, number, number];\n\n constructor(paddedRect: Rect, {\n pixelRatio,\n version,\n stretchX,\n stretchY,\n content\n }: StyleImage) {\n this.paddedRect = paddedRect;\n this.pixelRatio = pixelRatio;\n this.stretchX = stretchX;\n this.stretchY = stretchY;\n this.content = content;\n this.version = version;\n }\n\n get tl(): [number, number] {\n return [\n this.paddedRect.x + IMAGE_PADDING,\n this.paddedRect.y + IMAGE_PADDING\n ];\n }\n\n get br(): [number, number] {\n return [\n this.paddedRect.x + this.paddedRect.w - IMAGE_PADDING,\n this.paddedRect.y + this.paddedRect.h - IMAGE_PADDING\n ];\n }\n\n get tlbr(): Array {\n return this.tl.concat(this.br);\n }\n\n get displaySize(): [number, number] {\n return [\n (this.paddedRect.w - IMAGE_PADDING * 2) / this.pixelRatio,\n (this.paddedRect.h - IMAGE_PADDING * 2) / this.pixelRatio\n ];\n }\n}\n\n/**\n * @internal\n * A class holding all the images\n */\nexport class ImageAtlas {\n image: RGBAImage;\n iconPositions: {[_: string]: ImagePosition};\n patternPositions: {[_: string]: ImagePosition};\n haveRenderCallbacks: Array;\n uploaded: boolean;\n\n constructor(icons: {[_: string]: StyleImage}, patterns: {[_: string]: StyleImage}) {\n const iconPositions = {}, patternPositions = {};\n this.haveRenderCallbacks = [];\n\n const bins = [];\n\n this.addImages(icons, iconPositions, bins);\n this.addImages(patterns, patternPositions, bins);\n\n const {w, h} = potpack(bins);\n const image = new RGBAImage({width: w || 1, height: h || 1});\n\n for (const id in icons) {\n const src = icons[id];\n const bin = iconPositions[id].paddedRect;\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x: bin.x + IMAGE_PADDING, y: bin.y + IMAGE_PADDING}, src.data);\n }\n\n for (const id in patterns) {\n const src = patterns[id];\n const bin = patternPositions[id].paddedRect;\n const x = bin.x + IMAGE_PADDING,\n y = bin.y + IMAGE_PADDING,\n w = src.data.width,\n h = src.data.height;\n\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x, y}, src.data);\n // Add 1 pixel wrapped padding on each side of the image.\n RGBAImage.copy(src.data, image, {x: 0, y: h - 1}, {x, y: y - 1}, {width: w, height: 1}); // T\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x, y: y + h}, {width: w, height: 1}); // B\n RGBAImage.copy(src.data, image, {x: w - 1, y: 0}, {x: x - 1, y}, {width: 1, height: h}); // L\n RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x: x + w, y}, {width: 1, height: h}); // R\n }\n\n this.image = image;\n this.iconPositions = iconPositions;\n this.patternPositions = patternPositions;\n }\n\n addImages(images: {[_: string]: StyleImage}, positions: {[_: string]: ImagePosition}, bins: Array) {\n for (const id in images) {\n const src = images[id];\n const bin = {\n x: 0,\n y: 0,\n w: src.data.width + 2 * IMAGE_PADDING,\n h: src.data.height + 2 * IMAGE_PADDING,\n };\n bins.push(bin);\n positions[id] = new ImagePosition(bin, src);\n\n if (src.hasRenderCallback) {\n this.haveRenderCallbacks.push(id);\n }\n }\n }\n\n patchUpdatedImages(imageManager: ImageManager, texture: Texture) {\n imageManager.dispatchRenderCallbacks(this.haveRenderCallbacks);\n for (const name in imageManager.updatedImages) {\n this.patchUpdatedImage(this.iconPositions[name], imageManager.getImage(name), texture);\n this.patchUpdatedImage(this.patternPositions[name], imageManager.getImage(name), texture);\n }\n }\n\n patchUpdatedImage(position: ImagePosition, image: StyleImage, texture: Texture) {\n if (!position || !image) return;\n\n if (position.version === image.version) return;\n\n position.version = image.version;\n const [x, y] = position.tl;\n texture.update(image.data, undefined, {x, y});\n }\n\n}\n\nregister('ImagePosition', ImagePosition);\nregister('ImageAtlas', ImageAtlas);\n","import {\n charHasUprightVerticalOrientation,\n charAllowsIdeographicBreaking,\n charInComplexShapingScript\n} from '../util/script_detection';\nimport {verticalizePunctuation} from '../util/verticalize_punctuation';\nimport {plugin as rtlTextPlugin} from '../source/rtl_text_plugin';\nimport ONE_EM from './one_em';\nimport {warnOnce} from '../util/util';\n\nimport type {StyleGlyph, GlyphMetrics} from '../style/style_glyph';\nimport {GLYPH_PBF_BORDER} from '../style/parse_glyph_pbf';\nimport type {ImagePosition} from '../render/image_atlas';\nimport {IMAGE_PADDING} from '../render/image_atlas';\nimport type {Rect, GlyphPosition} from '../render/glyph_atlas';\nimport {Formatted, FormattedSection} from '@maplibre/maplibre-gl-style-spec';\n\nenum WritingMode {\n none = 0,\n horizontal = 1,\n vertical = 2,\n horizontalOnly = 3\n}\n\nconst SHAPING_DEFAULT_OFFSET = -17;\nexport {shapeText, shapeIcon, fitIconToText, getAnchorAlignment, WritingMode, SHAPING_DEFAULT_OFFSET};\n\n// The position of a glyph relative to the text's anchor point.\nexport type PositionedGlyph = {\n glyph: number;\n imageName: string | null;\n x: number;\n y: number;\n vertical: boolean;\n scale: number;\n fontStack: string;\n sectionIndex: number;\n metrics: GlyphMetrics;\n rect: Rect | null;\n};\n\nexport type PositionedLine = {\n positionedGlyphs: Array;\n lineOffset: number;\n};\n\n// A collection of positioned glyphs and some metadata\nexport type Shaping = {\n positionedLines: Array;\n top: number;\n bottom: number;\n left: number;\n right: number;\n writingMode: WritingMode.horizontal | WritingMode.vertical;\n text: string;\n iconsInText: boolean;\n verticalizable: boolean;\n};\n\nfunction isEmpty(positionedLines: Array) {\n for (const line of positionedLines) {\n if (line.positionedGlyphs.length !== 0) {\n return false;\n }\n }\n return true;\n}\n\nexport type SymbolAnchor = 'center' | 'left' | 'right' | 'top' | 'bottom' | 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right';\nexport type TextJustify = 'left' | 'center' | 'right';\n\n// Max number of images in label is 6401 U+E000–U+F8FF that covers\n// Basic Multilingual Plane Unicode Private Use Area (PUA).\nconst PUAbegin = 0xE000;\nconst PUAend = 0xF8FF;\n\nclass SectionOptions {\n // Text options\n scale: number;\n fontStack: string;\n // Image options\n imageName: string | null;\n\n constructor() {\n this.scale = 1.0;\n this.fontStack = '';\n this.imageName = null;\n }\n\n static forText(scale: number | null, fontStack: string) {\n const textOptions = new SectionOptions();\n textOptions.scale = scale || 1;\n textOptions.fontStack = fontStack;\n return textOptions;\n }\n\n static forImage(imageName: string) {\n const imageOptions = new SectionOptions();\n imageOptions.imageName = imageName;\n return imageOptions;\n }\n\n}\n\nclass TaggedString {\n text: string;\n sectionIndex: Array; // maps each character in 'text' to its corresponding entry in 'sections'\n sections: Array;\n imageSectionID: number | null;\n\n constructor() {\n this.text = '';\n this.sectionIndex = [];\n this.sections = [];\n this.imageSectionID = null;\n }\n\n static fromFeature(text: Formatted, defaultFontStack: string) {\n const result = new TaggedString();\n for (let i = 0; i < text.sections.length; i++) {\n const section = text.sections[i];\n if (!section.image) {\n result.addTextSection(section, defaultFontStack);\n } else {\n result.addImageSection(section);\n }\n }\n return result;\n }\n\n length(): number {\n return this.text.length;\n }\n\n getSection(index: number): SectionOptions {\n return this.sections[this.sectionIndex[index]];\n }\n\n getSectionIndex(index: number): number {\n return this.sectionIndex[index];\n }\n\n getCharCode(index: number): number {\n return this.text.charCodeAt(index);\n }\n\n verticalizePunctuation() {\n this.text = verticalizePunctuation(this.text);\n }\n\n trim() {\n let beginningWhitespace = 0;\n for (let i = 0;\n i < this.text.length && whitespace[this.text.charCodeAt(i)];\n i++) {\n beginningWhitespace++;\n }\n let trailingWhitespace = this.text.length;\n for (let i = this.text.length - 1;\n i >= 0 && i >= beginningWhitespace && whitespace[this.text.charCodeAt(i)];\n i--) {\n trailingWhitespace--;\n }\n this.text = this.text.substring(beginningWhitespace, trailingWhitespace);\n this.sectionIndex = this.sectionIndex.slice(beginningWhitespace, trailingWhitespace);\n }\n\n substring(start: number, end: number): TaggedString {\n const substring = new TaggedString();\n substring.text = this.text.substring(start, end);\n substring.sectionIndex = this.sectionIndex.slice(start, end);\n substring.sections = this.sections;\n return substring;\n }\n\n toString(): string {\n return this.text;\n }\n\n getMaxScale() {\n return this.sectionIndex.reduce((max, index) => Math.max(max, this.sections[index].scale), 0);\n }\n\n addTextSection(section: FormattedSection, defaultFontStack: string) {\n this.text += section.text;\n this.sections.push(SectionOptions.forText(section.scale, section.fontStack || defaultFontStack));\n const index = this.sections.length - 1;\n for (let i = 0; i < section.text.length; ++i) {\n this.sectionIndex.push(index);\n }\n }\n\n addImageSection(section: FormattedSection) {\n const imageName = section.image ? section.image.name : '';\n if (imageName.length === 0) {\n warnOnce('Can\\'t add FormattedSection with an empty image.');\n return;\n }\n\n const nextImageSectionCharCode = this.getNextImageSectionCharCode();\n if (!nextImageSectionCharCode) {\n warnOnce(`Reached maximum number of images ${PUAend - PUAbegin + 2}`);\n return;\n }\n\n this.text += String.fromCharCode(nextImageSectionCharCode);\n this.sections.push(SectionOptions.forImage(imageName));\n this.sectionIndex.push(this.sections.length - 1);\n }\n\n getNextImageSectionCharCode(): number | null {\n if (!this.imageSectionID) {\n this.imageSectionID = PUAbegin;\n return this.imageSectionID;\n }\n\n if (this.imageSectionID >= PUAend) return null;\n return ++this.imageSectionID;\n }\n}\n\nfunction breakLines(input: TaggedString, lineBreakPoints: Array): Array {\n const lines = [];\n const text = input.text;\n let start = 0;\n for (const lineBreak of lineBreakPoints) {\n lines.push(input.substring(start, lineBreak));\n start = lineBreak;\n }\n\n if (start < text.length) {\n lines.push(input.substring(start, text.length));\n }\n return lines;\n}\n\nfunction shapeText(\n text: Formatted,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n glyphPositions: {\n [_: string]: {\n [_: number]: GlyphPosition;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n defaultFontStack: string,\n maxWidth: number,\n lineHeight: number,\n textAnchor: SymbolAnchor,\n textJustify: TextJustify,\n spacing: number,\n translate: [number, number],\n writingMode: WritingMode.horizontal | WritingMode.vertical,\n allowVerticalPlacement: boolean,\n symbolPlacement: string,\n layoutTextSize: number,\n layoutTextSizeThisZoom: number\n): Shaping | false {\n const logicalInput = TaggedString.fromFeature(text, defaultFontStack);\n\n if (writingMode === WritingMode.vertical) {\n logicalInput.verticalizePunctuation();\n }\n\n let lines: Array;\n\n const {processBidirectionalText, processStyledBidirectionalText} = rtlTextPlugin;\n if (processBidirectionalText && logicalInput.sections.length === 1) {\n // Bidi doesn't have to be style-aware\n lines = [];\n const untaggedLines =\n processBidirectionalText(logicalInput.toString(),\n determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize));\n for (const line of untaggedLines) {\n const taggedLine = new TaggedString();\n taggedLine.text = line;\n taggedLine.sections = logicalInput.sections;\n for (let i = 0; i < line.length; i++) {\n taggedLine.sectionIndex.push(0);\n }\n lines.push(taggedLine);\n }\n } else if (processStyledBidirectionalText) {\n // Need version of mapbox-gl-rtl-text with style support for combining RTL text\n // with formatting\n lines = [];\n const processedLines =\n processStyledBidirectionalText(logicalInput.text,\n logicalInput.sectionIndex,\n determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize));\n for (const line of processedLines) {\n const taggedLine = new TaggedString();\n taggedLine.text = line[0];\n taggedLine.sectionIndex = line[1];\n taggedLine.sections = logicalInput.sections;\n lines.push(taggedLine);\n }\n } else {\n lines = breakLines(logicalInput, determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize));\n }\n\n const positionedLines = [];\n const shaping = {\n positionedLines,\n text: logicalInput.toString(),\n top: translate[1],\n bottom: translate[1],\n left: translate[0],\n right: translate[0],\n writingMode,\n iconsInText: false,\n verticalizable: false\n };\n\n shapeLines(shaping, glyphMap, glyphPositions, imagePositions, lines, lineHeight, textAnchor, textJustify, writingMode, spacing, allowVerticalPlacement, layoutTextSizeThisZoom);\n if (isEmpty(positionedLines)) return false;\n\n return shaping;\n}\n\n// using computed properties due to https://github.com/facebook/flow/issues/380\n/* eslint no-useless-computed-key: 0 */\n\nconst whitespace: {\n [_: number]: boolean;\n} = {\n [0x09]: true, // tab\n [0x0a]: true, // newline\n [0x0b]: true, // vertical tab\n [0x0c]: true, // form feed\n [0x0d]: true, // carriage return\n [0x20]: true, // space\n};\n\nconst breakable: {\n [_: number]: boolean;\n} = {\n [0x0a]: true, // newline\n [0x20]: true, // space\n [0x26]: true, // ampersand\n [0x28]: true, // left parenthesis\n [0x29]: true, // right parenthesis\n [0x2b]: true, // plus sign\n [0x2d]: true, // hyphen-minus\n [0x2f]: true, // solidus\n [0xad]: true, // soft hyphen\n [0xb7]: true, // middle dot\n [0x200b]: true, // zero-width space\n [0x2010]: true, // hyphen\n [0x2013]: true, // en dash\n [0x2027]: true // interpunct\n // Many other characters may be reasonable breakpoints\n // Consider \"neutral orientation\" characters at scriptDetection.charHasNeutralVerticalOrientation\n // See https://github.com/mapbox/mapbox-gl-js/issues/3658\n};\n\nfunction getGlyphAdvance(\n codePoint: number,\n section: SectionOptions,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n spacing: number,\n layoutTextSize: number\n): number {\n if (!section.imageName) {\n const positions = glyphMap[section.fontStack];\n const glyph = positions && positions[codePoint];\n if (!glyph) return 0;\n return glyph.metrics.advance * section.scale + spacing;\n } else {\n const imagePosition = imagePositions[section.imageName];\n if (!imagePosition) return 0;\n return imagePosition.displaySize[0] * section.scale * ONE_EM / layoutTextSize + spacing;\n }\n}\n\nfunction determineAverageLineWidth(logicalInput: TaggedString,\n spacing: number,\n maxWidth: number,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n layoutTextSize: number) {\n let totalWidth = 0;\n\n for (let index = 0; index < logicalInput.length(); index++) {\n const section = logicalInput.getSection(index);\n totalWidth += getGlyphAdvance(logicalInput.getCharCode(index), section, glyphMap, imagePositions, spacing, layoutTextSize);\n }\n\n const lineCount = Math.max(1, Math.ceil(totalWidth / maxWidth));\n return totalWidth / lineCount;\n}\n\nfunction calculateBadness(lineWidth: number,\n targetWidth: number,\n penalty: number,\n isLastBreak: boolean) {\n const raggedness = Math.pow(lineWidth - targetWidth, 2);\n if (isLastBreak) {\n // Favor finals lines shorter than average over longer than average\n if (lineWidth < targetWidth) {\n return raggedness / 2;\n } else {\n return raggedness * 2;\n }\n }\n\n return raggedness + Math.abs(penalty) * penalty;\n}\n\nfunction calculatePenalty(codePoint: number, nextCodePoint: number, penalizableIdeographicBreak: boolean) {\n let penalty = 0;\n // Force break on newline\n if (codePoint === 0x0a) {\n penalty -= 10000;\n }\n // Penalize breaks between characters that allow ideographic breaking because\n // they are less preferable than breaks at spaces (or zero width spaces).\n if (penalizableIdeographicBreak) {\n penalty += 150;\n }\n\n // Penalize open parenthesis at end of line\n if (codePoint === 0x28 || codePoint === 0xff08) {\n penalty += 50;\n }\n\n // Penalize close parenthesis at beginning of line\n if (nextCodePoint === 0x29 || nextCodePoint === 0xff09) {\n penalty += 50;\n }\n return penalty;\n}\n\ntype Break = {\n index: number;\n x: number;\n priorBreak: Break;\n badness: number;\n};\n\nfunction evaluateBreak(\n breakIndex: number,\n breakX: number,\n targetWidth: number,\n potentialBreaks: Array,\n penalty: number,\n isLastBreak: boolean\n): Break {\n // We could skip evaluating breaks where the line length (breakX - priorBreak.x) > maxWidth\n // ...but in fact we allow lines longer than maxWidth (if there's no break points)\n // ...and when targetWidth and maxWidth are close, strictly enforcing maxWidth can give\n // more lopsided results.\n\n let bestPriorBreak: Break = null;\n let bestBreakBadness = calculateBadness(breakX, targetWidth, penalty, isLastBreak);\n\n for (const potentialBreak of potentialBreaks) {\n const lineWidth = breakX - potentialBreak.x;\n const breakBadness =\n calculateBadness(lineWidth, targetWidth, penalty, isLastBreak) + potentialBreak.badness;\n if (breakBadness <= bestBreakBadness) {\n bestPriorBreak = potentialBreak;\n bestBreakBadness = breakBadness;\n }\n }\n\n return {\n index: breakIndex,\n x: breakX,\n priorBreak: bestPriorBreak,\n badness: bestBreakBadness\n };\n}\n\nfunction leastBadBreaks(lastLineBreak?: Break | null): Array {\n if (!lastLineBreak) {\n return [];\n }\n return leastBadBreaks(lastLineBreak.priorBreak).concat(lastLineBreak.index);\n}\n\nfunction determineLineBreaks(\n logicalInput: TaggedString,\n spacing: number,\n maxWidth: number,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n symbolPlacement: string,\n layoutTextSize: number\n): Array {\n if (symbolPlacement !== 'point')\n return [];\n\n if (!logicalInput)\n return [];\n\n const potentialLineBreaks = [];\n const targetWidth = determineAverageLineWidth(logicalInput, spacing, maxWidth, glyphMap, imagePositions, layoutTextSize);\n\n const hasServerSuggestedBreakpoints = logicalInput.text.indexOf('\\u200b') >= 0;\n\n let currentX = 0;\n\n for (let i = 0; i < logicalInput.length(); i++) {\n const section = logicalInput.getSection(i);\n const codePoint = logicalInput.getCharCode(i);\n if (!whitespace[codePoint]) currentX += getGlyphAdvance(codePoint, section, glyphMap, imagePositions, spacing, layoutTextSize);\n\n // Ideographic characters, spaces, and word-breaking punctuation that often appear without\n // surrounding spaces.\n if ((i < logicalInput.length() - 1)) {\n const ideographicBreak = charAllowsIdeographicBreaking(codePoint);\n if (breakable[codePoint] || ideographicBreak || section.imageName) {\n\n potentialLineBreaks.push(\n evaluateBreak(\n i + 1,\n currentX,\n targetWidth,\n potentialLineBreaks,\n calculatePenalty(codePoint, logicalInput.getCharCode(i + 1), ideographicBreak && hasServerSuggestedBreakpoints),\n false));\n }\n }\n }\n\n return leastBadBreaks(\n evaluateBreak(\n logicalInput.length(),\n currentX,\n targetWidth,\n potentialLineBreaks,\n 0,\n true));\n}\n\nfunction getAnchorAlignment(anchor: SymbolAnchor) {\n let horizontalAlign = 0.5, verticalAlign = 0.5;\n\n switch (anchor) {\n case 'right':\n case 'top-right':\n case 'bottom-right':\n horizontalAlign = 1;\n break;\n case 'left':\n case 'top-left':\n case 'bottom-left':\n horizontalAlign = 0;\n break;\n }\n\n switch (anchor) {\n case 'bottom':\n case 'bottom-right':\n case 'bottom-left':\n verticalAlign = 1;\n break;\n case 'top':\n case 'top-right':\n case 'top-left':\n verticalAlign = 0;\n break;\n }\n\n return {horizontalAlign, verticalAlign};\n}\n\nfunction shapeLines(shaping: Shaping,\n glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n },\n glyphPositions: {\n [_: string]: {\n [_: number]: GlyphPosition;\n };\n },\n imagePositions: {[_: string]: ImagePosition},\n lines: Array,\n lineHeight: number,\n textAnchor: SymbolAnchor,\n textJustify: TextJustify,\n writingMode: WritingMode.horizontal | WritingMode.vertical,\n spacing: number,\n allowVerticalPlacement: boolean,\n layoutTextSizeThisZoom: number) {\n\n let x = 0;\n let y = SHAPING_DEFAULT_OFFSET;\n\n let maxLineLength = 0;\n let maxLineHeight = 0;\n\n const justify =\n textJustify === 'right' ? 1 :\n textJustify === 'left' ? 0 : 0.5;\n\n let lineIndex = 0;\n for (const line of lines) {\n line.trim();\n\n const lineMaxScale = line.getMaxScale();\n const maxLineOffset = (lineMaxScale - 1) * ONE_EM;\n const positionedLine = {positionedGlyphs: [], lineOffset: 0};\n shaping.positionedLines[lineIndex] = positionedLine;\n const positionedGlyphs = positionedLine.positionedGlyphs;\n let lineOffset = 0.0;\n\n if (!line.length()) {\n y += lineHeight; // Still need a line feed after empty line\n ++lineIndex;\n continue;\n }\n\n for (let i = 0; i < line.length(); i++) {\n const section = line.getSection(i);\n const sectionIndex = line.getSectionIndex(i);\n const codePoint = line.getCharCode(i);\n let baselineOffset = 0.0;\n let metrics = null;\n let rect = null;\n let imageName = null;\n let verticalAdvance = ONE_EM;\n const vertical = !(writingMode === WritingMode.horizontal ||\n // Don't verticalize glyphs that have no upright orientation if vertical placement is disabled.\n (!allowVerticalPlacement && !charHasUprightVerticalOrientation(codePoint)) ||\n // If vertical placement is enabled, don't verticalize glyphs that\n // are from complex text layout script, or whitespaces.\n (allowVerticalPlacement && (whitespace[codePoint] || charInComplexShapingScript(codePoint))));\n\n if (!section.imageName) {\n const positions = glyphPositions[section.fontStack];\n const glyphPosition = positions && positions[codePoint];\n if (glyphPosition && glyphPosition.rect) {\n rect = glyphPosition.rect;\n metrics = glyphPosition.metrics;\n } else {\n const glyphs = glyphMap[section.fontStack];\n const glyph = glyphs && glyphs[codePoint];\n if (!glyph) continue;\n metrics = glyph.metrics;\n }\n\n // We don't know the baseline, but since we're laying out\n // at 24 points, we can calculate how much it will move when\n // we scale up or down.\n baselineOffset = (lineMaxScale - section.scale) * ONE_EM;\n } else {\n const imagePosition = imagePositions[section.imageName];\n if (!imagePosition) continue;\n imageName = section.imageName;\n shaping.iconsInText = shaping.iconsInText || true;\n rect = imagePosition.paddedRect;\n const size = imagePosition.displaySize;\n // If needed, allow to set scale factor for an image using\n // alias \"image-scale\" that could be alias for \"font-scale\"\n // when FormattedSection is an image section.\n section.scale = section.scale * ONE_EM / layoutTextSizeThisZoom;\n\n metrics = {width: size[0],\n height: size[1],\n left: IMAGE_PADDING,\n top: -GLYPH_PBF_BORDER,\n advance: vertical ? size[1] : size[0]};\n\n // Difference between one EM and an image size.\n // Aligns bottom of an image to a baseline level.\n const imageOffset = ONE_EM - size[1] * section.scale;\n baselineOffset = maxLineOffset + imageOffset;\n verticalAdvance = metrics.advance;\n\n // Difference between height of an image and one EM at max line scale.\n // Pushes current line down if an image size is over 1 EM at max line scale.\n const offset = vertical ? size[0] * section.scale - ONE_EM * lineMaxScale :\n size[1] * section.scale - ONE_EM * lineMaxScale;\n if (offset > 0 && offset > lineOffset) {\n lineOffset = offset;\n }\n }\n\n if (!vertical) {\n positionedGlyphs.push({glyph: codePoint, imageName, x, y: y + baselineOffset, vertical, scale: section.scale, fontStack: section.fontStack, sectionIndex, metrics, rect});\n x += metrics.advance * section.scale + spacing;\n } else {\n shaping.verticalizable = true;\n positionedGlyphs.push({glyph: codePoint, imageName, x, y: y + baselineOffset, vertical, scale: section.scale, fontStack: section.fontStack, sectionIndex, metrics, rect});\n x += verticalAdvance * section.scale + spacing;\n }\n }\n\n // Only justify if we placed at least one glyph\n if (positionedGlyphs.length !== 0) {\n const lineLength = x - spacing;\n maxLineLength = Math.max(lineLength, maxLineLength);\n justifyLine(positionedGlyphs, 0, positionedGlyphs.length - 1, justify, lineOffset);\n }\n\n x = 0;\n const currentLineHeight = lineHeight * lineMaxScale + lineOffset;\n positionedLine.lineOffset = Math.max(lineOffset, maxLineOffset);\n y += currentLineHeight;\n maxLineHeight = Math.max(currentLineHeight, maxLineHeight);\n ++lineIndex;\n }\n\n // Calculate the bounding box and justify / align text block.\n const height = y - SHAPING_DEFAULT_OFFSET;\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(textAnchor);\n align(shaping.positionedLines, justify, horizontalAlign, verticalAlign, maxLineLength, maxLineHeight, lineHeight, height, lines.length);\n\n shaping.top += -verticalAlign * height;\n shaping.bottom = shaping.top + height;\n shaping.left += -horizontalAlign * maxLineLength;\n shaping.right = shaping.left + maxLineLength;\n}\n\n// justify right = 1, left = 0, center = 0.5\nfunction justifyLine(positionedGlyphs: Array,\n start: number,\n end: number,\n justify: 1 | 0 | 0.5,\n lineOffset: number) {\n if (!justify && !lineOffset)\n return;\n\n const lastPositionedGlyph = positionedGlyphs[end];\n const lastAdvance = lastPositionedGlyph.metrics.advance * lastPositionedGlyph.scale;\n const lineIndent = (positionedGlyphs[end].x + lastAdvance) * justify;\n\n for (let j = start; j <= end; j++) {\n positionedGlyphs[j].x -= lineIndent;\n positionedGlyphs[j].y += lineOffset;\n }\n}\n\nfunction align(positionedLines: Array,\n justify: number,\n horizontalAlign: number,\n verticalAlign: number,\n maxLineLength: number,\n maxLineHeight: number,\n lineHeight: number,\n blockHeight: number,\n lineCount: number) {\n const shiftX = (justify - horizontalAlign) * maxLineLength;\n let shiftY = 0;\n\n if (maxLineHeight !== lineHeight) {\n shiftY = -blockHeight * verticalAlign - SHAPING_DEFAULT_OFFSET;\n } else {\n shiftY = (-verticalAlign * lineCount + 0.5) * lineHeight;\n }\n\n for (const line of positionedLines) {\n for (const positionedGlyph of line.positionedGlyphs) {\n positionedGlyph.x += shiftX;\n positionedGlyph.y += shiftY;\n }\n }\n}\n\nexport type PositionedIcon = {\n image: ImagePosition;\n top: number;\n bottom: number;\n left: number;\n right: number;\n collisionPadding?: [number, number, number, number];\n};\n\nfunction shapeIcon(\n image: ImagePosition,\n iconOffset: [number, number],\n iconAnchor: SymbolAnchor\n): PositionedIcon {\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(iconAnchor);\n const dx = iconOffset[0];\n const dy = iconOffset[1];\n const x1 = dx - image.displaySize[0] * horizontalAlign;\n const x2 = x1 + image.displaySize[0];\n const y1 = dy - image.displaySize[1] * verticalAlign;\n const y2 = y1 + image.displaySize[1];\n return {image, top: y1, bottom: y2, left: x1, right: x2};\n}\n\nfunction fitIconToText(\n shapedIcon: PositionedIcon,\n shapedText: Shaping,\n textFit: string,\n padding: [number, number, number, number],\n iconOffset: [number, number],\n fontScale: number\n): PositionedIcon {\n\n const image = shapedIcon.image;\n\n let collisionPadding;\n if (image.content) {\n const content = image.content;\n const pixelRatio = image.pixelRatio || 1;\n collisionPadding = [\n content[0] / pixelRatio,\n content[1] / pixelRatio,\n image.displaySize[0] - content[2] / pixelRatio,\n image.displaySize[1] - content[3] / pixelRatio\n ];\n }\n\n // We don't respect the icon-anchor, because icon-text-fit is set. Instead,\n // the icon will be centered on the text, then stretched in the given\n // dimensions.\n\n const textLeft = shapedText.left * fontScale;\n const textRight = shapedText.right * fontScale;\n\n let top, right, bottom, left;\n if (textFit === 'width' || textFit === 'both') {\n // Stretched horizontally to the text width\n left = iconOffset[0] + textLeft - padding[3];\n right = iconOffset[0] + textRight + padding[1];\n } else {\n // Centered on the text\n left = iconOffset[0] + (textLeft + textRight - image.displaySize[0]) / 2;\n right = left + image.displaySize[0];\n }\n\n const textTop = shapedText.top * fontScale;\n const textBottom = shapedText.bottom * fontScale;\n if (textFit === 'height' || textFit === 'both') {\n // Stretched vertically to the text height\n top = iconOffset[1] + textTop - padding[0];\n bottom = iconOffset[1] + textBottom + padding[2];\n } else {\n // Centered on the text\n top = iconOffset[1] + (textTop + textBottom - image.displaySize[1]) / 2;\n bottom = top + image.displaySize[1];\n }\n\n return {image, top, right, bottom, left, collisionPadding};\n}\n","import {Interpolate, interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {clamp} from '../util/util';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\n\nimport type {PropertyValue, PossiblyEvaluatedPropertyValue} from '../style/properties';\nimport type {InterpolationType} from '@maplibre/maplibre-gl-style-spec';\n\nconst MAX_GLYPH_ICON_SIZE = 255;\nconst SIZE_PACK_FACTOR = 128;\nconst MAX_PACKED_SIZE = MAX_GLYPH_ICON_SIZE * SIZE_PACK_FACTOR;\n\nexport {getSizeData, evaluateSizeForFeature, evaluateSizeForZoom, SIZE_PACK_FACTOR, MAX_GLYPH_ICON_SIZE, MAX_PACKED_SIZE};\n\nexport type SizeData = {\n kind: 'constant';\n layoutSize: number;\n} | {\n kind: 'source';\n} | {\n kind: 'camera';\n minZoom: number;\n maxZoom: number;\n minSize: number;\n maxSize: number;\n interpolationType: InterpolationType;\n} | {\n kind: 'composite';\n minZoom: number;\n maxZoom: number;\n interpolationType: InterpolationType;\n};\n\nexport type EvaluatedZoomSize = {uSizeT: number; uSize: number};\n\n// For {text,icon}-size, get the bucket-level data that will be needed by\n// the painter to set symbol-size-related uniforms\nfunction getSizeData(\n tileZoom: number,\n value: PropertyValue>\n): SizeData {\n const {expression} = value;\n\n if (expression.kind === 'constant') {\n const layoutSize = expression.evaluate(new EvaluationParameters(tileZoom + 1));\n return {kind: 'constant', layoutSize};\n\n } else if (expression.kind === 'source') {\n return {kind: 'source'};\n\n } else {\n const {zoomStops, interpolationType} = expression;\n\n // calculate covering zoom stops for zoom-dependent values\n let lower = 0;\n while (lower < zoomStops.length && zoomStops[lower] <= tileZoom) lower++;\n lower = Math.max(0, lower - 1);\n let upper = lower;\n while (upper < zoomStops.length && zoomStops[upper] < tileZoom + 1) upper++;\n upper = Math.min(zoomStops.length - 1, upper);\n\n const minZoom = zoomStops[lower];\n const maxZoom = zoomStops[upper];\n\n // We'd like to be able to use CameraExpression or CompositeExpression in these\n // return types rather than ExpressionSpecification, but the former are not\n // transferrable across Web Worker boundaries.\n if (expression.kind === 'composite') {\n return {kind: 'composite', minZoom, maxZoom, interpolationType};\n }\n\n // for camera functions, also save off the function values\n // evaluated at the covering zoom levels\n const minSize = expression.evaluate(new EvaluationParameters(minZoom));\n const maxSize = expression.evaluate(new EvaluationParameters(maxZoom));\n\n return {kind: 'camera', minZoom, maxZoom, minSize, maxSize, interpolationType};\n }\n}\n\nfunction evaluateSizeForFeature(sizeData: SizeData,\n {\n uSize,\n uSizeT\n }: {\n uSize: number;\n uSizeT: number;\n },\n {\n lowerSize,\n upperSize\n }: {\n lowerSize: number;\n upperSize: number;\n }): number {\n if (sizeData.kind === 'source') {\n return lowerSize / SIZE_PACK_FACTOR;\n } else if (sizeData.kind === 'composite') {\n return interpolates.number(lowerSize / SIZE_PACK_FACTOR, upperSize / SIZE_PACK_FACTOR, uSizeT);\n }\n return uSize;\n}\n\nfunction evaluateSizeForZoom(sizeData: SizeData, zoom: number): EvaluatedZoomSize {\n let uSizeT = 0;\n let uSize = 0;\n\n if (sizeData.kind === 'constant') {\n uSize = sizeData.layoutSize;\n\n } else if (sizeData.kind !== 'source') {\n const {interpolationType, minZoom, maxZoom} = sizeData;\n\n // Even though we could get the exact value of the camera function\n // at z = tr.zoom, we intentionally do not: instead, we interpolate\n // between the camera function values at a pair of zoom stops covering\n // [tileZoom, tileZoom + 1] in order to be consistent with this\n // restriction on composite functions\n const t = !interpolationType ? 0 : clamp(\n Interpolate.interpolationFactor(interpolationType, zoom, minZoom, maxZoom), 0, 1);\n\n if (sizeData.kind === 'camera') {\n uSize = interpolates.number(sizeData.minSize, sizeData.maxSize, t);\n } else {\n uSizeT = t;\n }\n }\n\n return {uSizeT, uSize};\n}\n","import {SymbolLayoutPropsPossiblyEvaluated} from './symbol_style_layer_properties.g';\nimport type {SymbolLayoutProps} from './symbol_style_layer_properties.g';\nimport {PossiblyEvaluated} from '../properties';\n\n/**\n * The overlap mode for properties like `icon-overlap`and `text-overlap`\n */\nexport type OverlapMode = 'never' | 'always' | 'cooperative';\n\nexport function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'icon-overlap', allowOverlapProp: 'icon-allow-overlap'): OverlapMode;\nexport function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'text-overlap', allowOverlapProp: 'text-allow-overlap'): OverlapMode;\nexport function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'icon-overlap' | 'text-overlap', allowOverlapProp: 'icon-allow-overlap' | 'text-allow-overlap'): OverlapMode {\n let result: OverlapMode = 'never';\n const overlap = layout.get(overlapProp);\n\n if (overlap) {\n // if -overlap is set, use it\n result = overlap;\n } else if (layout.get(allowOverlapProp)) {\n // fall back to -allow-overlap, with false='never', true='always'\n result = 'always';\n }\n\n return result;\n}\n","import {\n symbolLayoutAttributes,\n collisionVertexAttributes,\n collisionBoxLayout,\n dynamicLayoutAttributes,\n} from './symbol_attributes';\n\nimport {SymbolLayoutArray,\n SymbolDynamicLayoutArray,\n SymbolOpacityArray,\n CollisionBoxLayoutArray,\n CollisionVertexArray,\n PlacedSymbolArray,\n SymbolInstanceArray,\n GlyphOffsetArray,\n SymbolLineVertexArray,\n TextAnchorOffsetArray\n} from '../array_types.g';\n\nimport Point from '@mapbox/point-geometry';\nimport {SegmentVector} from '../segment';\nimport {ProgramConfigurationSet} from '../program_configuration';\nimport {TriangleIndexArray, LineIndexArray} from '../index_array_type';\nimport {transformText} from '../../symbol/transform_text';\nimport {mergeLines} from '../../symbol/merge_lines';\nimport {allowsVerticalWritingMode, stringContainsRTLText} from '../../util/script_detection';\nimport {WritingMode} from '../../symbol/shaping';\nimport {loadGeometry} from '../load_geometry';\nimport {toEvaluationFeature} from '../evaluation_feature';\nimport mvt from '@mapbox/vector-tile';\nconst vectorTileFeatureTypes = mvt.VectorTileFeature.types;\nimport {verticalizedCharacterMap} from '../../util/verticalize_punctuation';\nimport {Anchor} from '../../symbol/anchor';\nimport {getSizeData, MAX_PACKED_SIZE} from '../../symbol/symbol_size';\n\nimport {register} from '../../util/web_worker_transfer';\nimport {EvaluationParameters} from '../../style/evaluation_parameters';\nimport {Formatted, ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport {plugin as globalRTLTextPlugin, getRTLTextPluginStatus} from '../../source/rtl_text_plugin';\nimport {mat4} from 'gl-matrix';\nimport {getOverlapMode} from '../../style/style_layer/overlap_mode';\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport type {\n Bucket,\n BucketParameters,\n IndexedFeature,\n PopulateParameters\n} from '../bucket';\nimport type {CollisionBoxArray, CollisionBox, SymbolInstance} from '../array_types.g';\nimport type {StructArray, StructArrayMember, ViewType} from '../../util/struct_array';\nimport type {SymbolStyleLayer} from '../../style/style_layer/symbol_style_layer';\nimport type {Context} from '../../gl/context';\nimport type {IndexBuffer} from '../../gl/index_buffer';\nimport type {VertexBuffer} from '../../gl/vertex_buffer';\nimport type {SymbolQuad} from '../../symbol/quads';\nimport type {SizeData} from '../../symbol/symbol_size';\nimport type {FeatureStates} from '../../source/source_state';\nimport type {ImagePosition} from '../../render/image_atlas';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\n\nexport type SingleCollisionBox = {\n x1: number;\n y1: number;\n x2: number;\n y2: number;\n anchorPointX: number;\n anchorPointY: number;\n};\n\nexport type CollisionArrays = {\n textBox?: SingleCollisionBox;\n verticalTextBox?: SingleCollisionBox;\n iconBox?: SingleCollisionBox;\n verticalIconBox?: SingleCollisionBox;\n textFeatureIndex?: number;\n verticalTextFeatureIndex?: number;\n iconFeatureIndex?: number;\n verticalIconFeatureIndex?: number;\n};\n\nexport type SymbolFeature = {\n sortKey: number | void;\n text: Formatted | void;\n icon: ResolvedImage;\n index: number;\n sourceLayerIndex: number;\n geometry: Array>;\n properties: any;\n type: 'Unknown' | 'Point' | 'LineString' | 'Polygon';\n id?: any;\n};\n\nexport type SortKeyRange = {\n sortKey: number;\n symbolInstanceStart: number;\n symbolInstanceEnd: number;\n};\n\n// Opacity arrays are frequently updated but don't contain a lot of information, so we pack them\n// tight. Each Uint32 is actually four duplicate Uint8s for the four corners of a glyph\n// 7 bits are for the current opacity, and the lowest bit is the target opacity\n\n// actually defined in symbol_attributes.js\n// const placementOpacityAttributes = [\n// { name: 'a_fade_opacity', components: 1, type: 'Uint32' }\n// ];\nconst shaderOpacityAttributes = [\n {name: 'a_fade_opacity', components: 1, type: 'Uint8' as ViewType, offset: 0}\n];\n\nfunction addVertex(\n array: StructArray,\n anchorX: number,\n anchorY: number,\n ox: number,\n oy: number,\n tx: number,\n ty: number,\n sizeVertex: number,\n isSDF: boolean,\n pixelOffsetX: number,\n pixelOffsetY: number,\n minFontScaleX: number,\n minFontScaleY: number\n) {\n const aSizeX = sizeVertex ? Math.min(MAX_PACKED_SIZE, Math.round(sizeVertex[0])) : 0;\n const aSizeY = sizeVertex ? Math.min(MAX_PACKED_SIZE, Math.round(sizeVertex[1])) : 0;\n array.emplaceBack(\n // a_pos_offset\n anchorX,\n anchorY,\n Math.round(ox * 32),\n Math.round(oy * 32),\n\n // a_data\n tx, // x coordinate of symbol on glyph atlas texture\n ty, // y coordinate of symbol on glyph atlas texture\n (aSizeX << 1) + (isSDF ? 1 : 0),\n aSizeY,\n pixelOffsetX * 16,\n pixelOffsetY * 16,\n minFontScaleX * 256,\n minFontScaleY * 256\n );\n}\n\nfunction addDynamicAttributes(dynamicLayoutVertexArray: StructArray, p: Point, angle: number) {\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle);\n}\n\nfunction containsRTLText(formattedText: Formatted): boolean {\n for (const section of formattedText.sections) {\n if (stringContainsRTLText(section.text)) {\n return true;\n }\n }\n return false;\n}\n\nexport class SymbolBuffers {\n layoutVertexArray: SymbolLayoutArray;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray;\n indexBuffer: IndexBuffer;\n\n programConfigurations: ProgramConfigurationSet;\n segments: SegmentVector;\n\n dynamicLayoutVertexArray: SymbolDynamicLayoutArray;\n dynamicLayoutVertexBuffer: VertexBuffer;\n\n opacityVertexArray: SymbolOpacityArray;\n opacityVertexBuffer: VertexBuffer;\n hasVisibleVertices: boolean;\n\n collisionVertexArray: CollisionVertexArray;\n collisionVertexBuffer: VertexBuffer;\n\n placedSymbolArray: PlacedSymbolArray;\n\n constructor(programConfigurations: ProgramConfigurationSet) {\n this.layoutVertexArray = new SymbolLayoutArray();\n this.indexArray = new TriangleIndexArray();\n this.programConfigurations = programConfigurations;\n this.segments = new SegmentVector();\n this.dynamicLayoutVertexArray = new SymbolDynamicLayoutArray();\n this.opacityVertexArray = new SymbolOpacityArray();\n this.hasVisibleVertices = false;\n this.placedSymbolArray = new PlacedSymbolArray();\n }\n\n isEmpty() {\n return this.layoutVertexArray.length === 0 &&\n this.indexArray.length === 0 &&\n this.dynamicLayoutVertexArray.length === 0 &&\n this.opacityVertexArray.length === 0;\n }\n\n upload(context: Context, dynamicIndexBuffer: boolean, upload?: boolean, update?: boolean) {\n if (this.isEmpty()) {\n return;\n }\n\n if (upload) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, symbolLayoutAttributes.members);\n this.indexBuffer = context.createIndexBuffer(this.indexArray, dynamicIndexBuffer);\n this.dynamicLayoutVertexBuffer = context.createVertexBuffer(this.dynamicLayoutVertexArray, dynamicLayoutAttributes.members, true);\n this.opacityVertexBuffer = context.createVertexBuffer(this.opacityVertexArray, shaderOpacityAttributes, true);\n // This is a performance hack so that we can write to opacityVertexArray with uint32s\n // even though the shaders read uint8s\n this.opacityVertexBuffer.itemSize = 1;\n }\n if (upload || update) {\n this.programConfigurations.upload(context);\n }\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.programConfigurations.destroy();\n this.segments.destroy();\n this.dynamicLayoutVertexBuffer.destroy();\n this.opacityVertexBuffer.destroy();\n }\n}\n\nregister('SymbolBuffers', SymbolBuffers);\n\nclass CollisionBuffers {\n layoutVertexArray: StructArray;\n layoutAttributes: Array;\n layoutVertexBuffer: VertexBuffer;\n\n indexArray: TriangleIndexArray | LineIndexArray;\n indexBuffer: IndexBuffer;\n\n segments: SegmentVector;\n\n collisionVertexArray: CollisionVertexArray;\n collisionVertexBuffer: VertexBuffer;\n\n constructor(LayoutArray: {\n new (...args: any): StructArray;\n },\n layoutAttributes: Array,\n IndexArray: {\n new (...args: any): TriangleIndexArray | LineIndexArray;\n }) {\n this.layoutVertexArray = new LayoutArray();\n this.layoutAttributes = layoutAttributes;\n this.indexArray = new IndexArray();\n this.segments = new SegmentVector();\n this.collisionVertexArray = new CollisionVertexArray();\n }\n\n upload(context: Context) {\n this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, this.layoutAttributes);\n this.indexBuffer = context.createIndexBuffer(this.indexArray);\n this.collisionVertexBuffer = context.createVertexBuffer(this.collisionVertexArray, collisionVertexAttributes.members, true);\n }\n\n destroy() {\n if (!this.layoutVertexBuffer) return;\n this.layoutVertexBuffer.destroy();\n this.indexBuffer.destroy();\n this.segments.destroy();\n this.collisionVertexBuffer.destroy();\n }\n}\n\nregister('CollisionBuffers', CollisionBuffers);\n\n/**\n * @internal\n * Unlike other buckets, which simply implement #addFeature with type-specific\n * logic for (essentially) triangulating feature geometries, SymbolBucket\n * requires specialized behavior:\n *\n * 1. WorkerTile#parse(), the logical owner of the bucket creation process,\n * calls SymbolBucket#populate(), which resolves text and icon tokens on\n * each feature, adds each glyphs and symbols needed to the passed-in\n * collections options.glyphDependencies and options.iconDependencies, and\n * stores the feature data for use in subsequent step (this.features).\n *\n * 2. WorkerTile asynchronously requests from the main thread all of the glyphs\n * and icons needed (by this bucket and any others). When glyphs and icons\n * have been received, the WorkerTile creates a CollisionIndex and invokes:\n *\n * 3. performSymbolLayout(bucket, stacks, icons) perform texts shaping and\n * layout on a Symbol Bucket. This step populates:\n * `this.symbolInstances`: metadata on generated symbols\n * `this.collisionBoxArray`: collision data for use by foreground\n * `this.text`: SymbolBuffers for text symbols\n * `this.icons`: SymbolBuffers for icons\n * `this.iconCollisionBox`: Debug SymbolBuffers for icon collision boxes\n * `this.textCollisionBox`: Debug SymbolBuffers for text collision boxes\n * The results are sent to the foreground for rendering\n *\n * 4. performSymbolPlacement(bucket, collisionIndex) is run on the foreground,\n * and uses the CollisionIndex along with current camera settings to determine\n * which symbols can actually show on the map. Collided symbols are hidden\n * using a dynamic \"OpacityVertexArray\".\n */\nexport class SymbolBucket implements Bucket {\n static MAX_GLYPHS: number;\n static addDynamicAttributes: typeof addDynamicAttributes;\n\n collisionBoxArray: CollisionBoxArray;\n zoom: number;\n overscaling: number;\n layers: Array;\n layerIds: Array;\n stateDependentLayers: Array;\n stateDependentLayerIds: Array;\n\n index: number;\n sdfIcons: boolean;\n iconsInText: boolean;\n iconsNeedLinear: boolean;\n bucketInstanceId: number;\n justReloaded: boolean;\n hasPattern: boolean;\n\n textSizeData: SizeData;\n iconSizeData: SizeData;\n\n glyphOffsetArray: GlyphOffsetArray;\n lineVertexArray: SymbolLineVertexArray;\n features: Array;\n symbolInstances: SymbolInstanceArray;\n textAnchorOffsets: TextAnchorOffsetArray;\n collisionArrays: Array;\n sortKeyRanges: Array;\n pixelRatio: number;\n tilePixelRatio: number;\n compareText: {[_: string]: Array};\n fadeStartTime: number;\n sortFeaturesByKey: boolean;\n sortFeaturesByY: boolean;\n canOverlap: boolean;\n sortedAngle: number;\n featureSortOrder: Array;\n\n collisionCircleArray: Array;\n placementInvProjMatrix: mat4;\n placementViewportMatrix: mat4;\n\n text: SymbolBuffers;\n icon: SymbolBuffers;\n textCollisionBox: CollisionBuffers;\n iconCollisionBox: CollisionBuffers;\n uploaded: boolean;\n sourceLayerIndex: number;\n sourceID: string;\n symbolInstanceIndexes: Array;\n writingModes: WritingMode[];\n allowVerticalPlacement: boolean;\n hasRTLText: boolean;\n\n constructor(options: BucketParameters) {\n this.collisionBoxArray = options.collisionBoxArray;\n this.zoom = options.zoom;\n this.overscaling = options.overscaling;\n this.layers = options.layers;\n this.layerIds = this.layers.map(layer => layer.id);\n this.index = options.index;\n this.pixelRatio = options.pixelRatio;\n this.sourceLayerIndex = options.sourceLayerIndex;\n this.hasPattern = false;\n this.hasRTLText = false;\n this.sortKeyRanges = [];\n\n this.collisionCircleArray = [];\n this.placementInvProjMatrix = mat4.identity([] as any);\n this.placementViewportMatrix = mat4.identity([] as any);\n\n const layer = this.layers[0];\n const unevaluatedLayoutValues = layer._unevaluatedLayout._values;\n\n this.textSizeData = getSizeData(this.zoom, unevaluatedLayoutValues['text-size']);\n this.iconSizeData = getSizeData(this.zoom, unevaluatedLayoutValues['icon-size']);\n\n const layout = this.layers[0].layout;\n const sortKey = layout.get('symbol-sort-key');\n const zOrder = layout.get('symbol-z-order');\n this.canOverlap =\n getOverlapMode(layout, 'text-overlap', 'text-allow-overlap') !== 'never' ||\n getOverlapMode(layout, 'icon-overlap', 'icon-allow-overlap') !== 'never' ||\n layout.get('text-ignore-placement') ||\n layout.get('icon-ignore-placement');\n this.sortFeaturesByKey = zOrder !== 'viewport-y' && !sortKey.isConstant();\n const zOrderByViewportY = zOrder === 'viewport-y' || (zOrder === 'auto' && !this.sortFeaturesByKey);\n this.sortFeaturesByY = zOrderByViewportY && this.canOverlap;\n\n if (layout.get('symbol-placement') === 'point') {\n this.writingModes = layout.get('text-writing-mode').map(wm => WritingMode[wm]);\n }\n\n this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id);\n\n this.sourceID = options.sourceID;\n }\n\n createArrays() {\n this.text = new SymbolBuffers(new ProgramConfigurationSet(this.layers, this.zoom, property => /^text/.test(property)));\n this.icon = new SymbolBuffers(new ProgramConfigurationSet(this.layers, this.zoom, property => /^icon/.test(property)));\n\n this.glyphOffsetArray = new GlyphOffsetArray();\n this.lineVertexArray = new SymbolLineVertexArray();\n this.symbolInstances = new SymbolInstanceArray();\n this.textAnchorOffsets = new TextAnchorOffsetArray();\n }\n\n calculateGlyphDependencies(text: string, stack: {[_: number]: boolean}, textAlongLine: boolean, allowVerticalPlacement: boolean, doesAllowVerticalWritingMode: boolean) {\n for (let i = 0; i < text.length; i++) {\n stack[text.charCodeAt(i)] = true;\n if ((textAlongLine || allowVerticalPlacement) && doesAllowVerticalWritingMode) {\n const verticalChar = verticalizedCharacterMap[text.charAt(i)];\n if (verticalChar) {\n stack[verticalChar.charCodeAt(0)] = true;\n }\n }\n }\n }\n\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) {\n const layer = this.layers[0];\n const layout = layer.layout;\n\n const textFont = layout.get('text-font');\n const textField = layout.get('text-field');\n const iconImage = layout.get('icon-image');\n const hasText =\n (textField.value.kind !== 'constant' ||\n (textField.value.value instanceof Formatted && !textField.value.value.isEmpty()) ||\n textField.value.value.toString().length > 0) &&\n (textFont.value.kind !== 'constant' || textFont.value.value.length > 0);\n // we should always resolve the icon-image value if the property was defined in the style\n // this allows us to fire the styleimagemissing event if image evaluation returns null\n // the only way to distinguish between null returned from a coalesce statement with no valid images\n // and null returned because icon-image wasn't defined is to check whether or not iconImage.parameters is an empty object\n const hasIcon = iconImage.value.kind !== 'constant' || !!iconImage.value.value || Object.keys(iconImage.parameters).length > 0;\n const symbolSortKey = layout.get('symbol-sort-key');\n\n this.features = [];\n\n if (!hasText && !hasIcon) {\n return;\n }\n\n const icons = options.iconDependencies;\n const stacks = options.glyphDependencies;\n const availableImages = options.availableImages;\n const globalProperties = new EvaluationParameters(this.zoom);\n\n for (const {feature, id, index, sourceLayerIndex} of features) {\n\n const needGeometry = layer._featureFilter.needGeometry;\n const evaluationFeature = toEvaluationFeature(feature, needGeometry);\n if (!layer._featureFilter.filter(globalProperties, evaluationFeature, canonical)) {\n continue;\n }\n\n if (!needGeometry) evaluationFeature.geometry = loadGeometry(feature);\n\n let text: Formatted | void;\n if (hasText) {\n // Expression evaluation will automatically coerce to Formatted\n // but plain string token evaluation skips that pathway so do the\n // conversion here.\n const resolvedTokens = layer.getValueAndResolveTokens('text-field', evaluationFeature, canonical, availableImages);\n const formattedText = Formatted.factory(resolvedTokens);\n if (containsRTLText(formattedText)) {\n this.hasRTLText = true;\n }\n if (\n !this.hasRTLText || // non-rtl text so can proceed safely\n getRTLTextPluginStatus() === 'unavailable' || // We don't intend to lazy-load the rtl text plugin, so proceed with incorrect shaping\n this.hasRTLText && globalRTLTextPlugin.isParsed() // Use the rtlText plugin to shape text\n ) {\n text = transformText(formattedText, layer, evaluationFeature);\n }\n }\n\n let icon: ResolvedImage;\n if (hasIcon) {\n // Expression evaluation will automatically coerce to Image\n // but plain string token evaluation skips that pathway so do the\n // conversion here.\n const resolvedTokens = layer.getValueAndResolveTokens('icon-image', evaluationFeature, canonical, availableImages);\n if (resolvedTokens instanceof ResolvedImage) {\n icon = resolvedTokens;\n } else {\n icon = ResolvedImage.fromString(resolvedTokens);\n }\n }\n\n if (!text && !icon) {\n continue;\n }\n const sortKey = this.sortFeaturesByKey ?\n symbolSortKey.evaluate(evaluationFeature, {}, canonical) :\n undefined;\n\n const symbolFeature: SymbolFeature = {\n id,\n text,\n icon,\n index,\n sourceLayerIndex,\n geometry: evaluationFeature.geometry,\n properties: feature.properties,\n type: vectorTileFeatureTypes[feature.type],\n sortKey\n };\n this.features.push(symbolFeature);\n\n if (icon) {\n icons[icon.name] = true;\n }\n\n if (text) {\n const fontStack = textFont.evaluate(evaluationFeature, {}, canonical).join(',');\n const textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point';\n this.allowVerticalPlacement = this.writingModes && this.writingModes.indexOf(WritingMode.vertical) >= 0;\n for (const section of text.sections) {\n if (!section.image) {\n const doesAllowVerticalWritingMode = allowsVerticalWritingMode(text.toString());\n const sectionFont = section.fontStack || fontStack;\n const sectionStack = stacks[sectionFont] = stacks[sectionFont] || {};\n this.calculateGlyphDependencies(section.text, sectionStack, textAlongLine, this.allowVerticalPlacement, doesAllowVerticalWritingMode);\n } else {\n // Add section image to the list of dependencies.\n icons[section.image.name] = true;\n }\n }\n }\n }\n\n if (layout.get('symbol-placement') === 'line') {\n // Merge adjacent lines with the same text to improve labelling.\n // It's better to place labels on one long line than on many short segments.\n this.features = mergeLines(this.features);\n }\n\n if (this.sortFeaturesByKey) {\n this.features.sort((a, b) => {\n // a.sortKey is always a number when sortFeaturesByKey is true\n return (a.sortKey as number) - (b.sortKey as number);\n });\n }\n }\n\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) {\n if (!this.stateDependentLayers.length) return;\n this.text.programConfigurations.updatePaintArrays(states, vtLayer, this.layers, imagePositions);\n this.icon.programConfigurations.updatePaintArrays(states, vtLayer, this.layers, imagePositions);\n }\n\n isEmpty() {\n // When the bucket encounters only rtl-text but the plugin isnt loaded, no symbol instances will be created.\n // In order for the bucket to be serialized, and not discarded as an empty bucket both checks are necessary.\n return this.symbolInstances.length === 0 && !this.hasRTLText;\n }\n\n uploadPending() {\n return !this.uploaded || this.text.programConfigurations.needsUpload || this.icon.programConfigurations.needsUpload;\n }\n\n upload(context: Context) {\n if (!this.uploaded && this.hasDebugData()) {\n this.textCollisionBox.upload(context);\n this.iconCollisionBox.upload(context);\n }\n this.text.upload(context, this.sortFeaturesByY, !this.uploaded, this.text.programConfigurations.needsUpload);\n this.icon.upload(context, this.sortFeaturesByY, !this.uploaded, this.icon.programConfigurations.needsUpload);\n this.uploaded = true;\n }\n\n destroyDebugData() {\n this.textCollisionBox.destroy();\n this.iconCollisionBox.destroy();\n }\n\n destroy() {\n this.text.destroy();\n this.icon.destroy();\n\n if (this.hasDebugData()) {\n this.destroyDebugData();\n }\n }\n\n addToLineVertexArray(anchor: Anchor, line: any) {\n const lineStartIndex = this.lineVertexArray.length;\n if (anchor.segment !== undefined) {\n let sumForwardLength = anchor.dist(line[anchor.segment + 1]);\n let sumBackwardLength = anchor.dist(line[anchor.segment]);\n const vertices = {};\n for (let i = anchor.segment + 1; i < line.length; i++) {\n vertices[i] = {x: line[i].x, y: line[i].y, tileUnitDistanceFromAnchor: sumForwardLength};\n if (i < line.length - 1) {\n sumForwardLength += line[i + 1].dist(line[i]);\n }\n }\n for (let i = anchor.segment || 0; i >= 0; i--) {\n vertices[i] = {x: line[i].x, y: line[i].y, tileUnitDistanceFromAnchor: sumBackwardLength};\n if (i > 0) {\n sumBackwardLength += line[i - 1].dist(line[i]);\n }\n }\n for (let i = 0; i < line.length; i++) {\n const vertex = vertices[i];\n this.lineVertexArray.emplaceBack(vertex.x, vertex.y, vertex.tileUnitDistanceFromAnchor);\n }\n }\n return {\n lineStartIndex,\n lineLength: this.lineVertexArray.length - lineStartIndex\n };\n }\n\n addSymbols(arrays: SymbolBuffers,\n quads: Array,\n sizeVertex: any,\n lineOffset: [number, number],\n alongLine: boolean,\n feature: SymbolFeature,\n writingMode: WritingMode,\n labelAnchor: Anchor,\n lineStartIndex: number,\n lineLength: number,\n associatedIconIndex: number,\n canonical: CanonicalTileID) {\n const indexArray = arrays.indexArray;\n const layoutVertexArray = arrays.layoutVertexArray;\n\n const segment = arrays.segments.prepareSegment(4 * quads.length, layoutVertexArray, indexArray, this.canOverlap ? feature.sortKey as number : undefined);\n const glyphOffsetArrayStart = this.glyphOffsetArray.length;\n const vertexStartIndex = segment.vertexLength;\n\n const angle = (this.allowVerticalPlacement && writingMode === WritingMode.vertical) ? Math.PI / 2 : 0;\n\n const sections = feature.text && feature.text.sections;\n\n for (let i = 0; i < quads.length; i++) {\n const {tl, tr, bl, br, tex, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY, glyphOffset, isSDF, sectionIndex} = quads[i];\n const index = segment.vertexLength;\n\n const y = glyphOffset[1];\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, tl.x, y + tl.y, tex.x, tex.y, sizeVertex, isSDF, pixelOffsetTL.x, pixelOffsetTL.y, minFontScaleX, minFontScaleY);\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, tr.x, y + tr.y, tex.x + tex.w, tex.y, sizeVertex, isSDF, pixelOffsetBR.x, pixelOffsetTL.y, minFontScaleX, minFontScaleY);\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, bl.x, y + bl.y, tex.x, tex.y + tex.h, sizeVertex, isSDF, pixelOffsetTL.x, pixelOffsetBR.y, minFontScaleX, minFontScaleY);\n addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, br.x, y + br.y, tex.x + tex.w, tex.y + tex.h, sizeVertex, isSDF, pixelOffsetBR.x, pixelOffsetBR.y, minFontScaleX, minFontScaleY);\n\n addDynamicAttributes(arrays.dynamicLayoutVertexArray, labelAnchor, angle);\n\n indexArray.emplaceBack(index, index + 1, index + 2);\n indexArray.emplaceBack(index + 1, index + 2, index + 3);\n\n segment.vertexLength += 4;\n segment.primitiveLength += 2;\n\n this.glyphOffsetArray.emplaceBack(glyphOffset[0]);\n\n if (i === quads.length - 1 || sectionIndex !== quads[i + 1].sectionIndex) {\n arrays.programConfigurations.populatePaintArrays(layoutVertexArray.length, feature, feature.index, {}, canonical, sections && sections[sectionIndex]);\n }\n }\n\n arrays.placedSymbolArray.emplaceBack(\n labelAnchor.x, labelAnchor.y,\n glyphOffsetArrayStart,\n this.glyphOffsetArray.length - glyphOffsetArrayStart,\n vertexStartIndex,\n lineStartIndex,\n lineLength,\n labelAnchor.segment,\n sizeVertex ? sizeVertex[0] : 0,\n sizeVertex ? sizeVertex[1] : 0,\n lineOffset[0], lineOffset[1],\n writingMode,\n // placedOrientation is null initially; will be updated to horizontal(1)/vertical(2) if placed\n 0,\n false as unknown as number,\n // The crossTileID is only filled/used on the foreground for dynamic text anchors\n 0,\n associatedIconIndex\n );\n }\n\n _addCollisionDebugVertex(layoutVertexArray: StructArray, collisionVertexArray: StructArray, point: Point, anchorX: number, anchorY: number, extrude: Point) {\n collisionVertexArray.emplaceBack(0, 0);\n return layoutVertexArray.emplaceBack(\n // pos\n point.x,\n point.y,\n // a_anchor_pos\n anchorX,\n anchorY,\n // extrude\n Math.round(extrude.x),\n Math.round(extrude.y));\n }\n\n addCollisionDebugVertices(x1: number, y1: number, x2: number, y2: number, arrays: CollisionBuffers, boxAnchorPoint: Point, symbolInstance: SymbolInstance) {\n const segment = arrays.segments.prepareSegment(4, arrays.layoutVertexArray, arrays.indexArray);\n const index = segment.vertexLength;\n\n const layoutVertexArray = arrays.layoutVertexArray;\n const collisionVertexArray = arrays.collisionVertexArray;\n\n const anchorX = symbolInstance.anchorX;\n const anchorY = symbolInstance.anchorY;\n\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x1, y1));\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x2, y1));\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x2, y2));\n this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x1, y2));\n\n segment.vertexLength += 4;\n\n const indexArray = arrays.indexArray as LineIndexArray;\n indexArray.emplaceBack(index, index + 1);\n indexArray.emplaceBack(index + 1, index + 2);\n indexArray.emplaceBack(index + 2, index + 3);\n indexArray.emplaceBack(index + 3, index);\n\n segment.primitiveLength += 4;\n }\n\n addDebugCollisionBoxes(startIndex: number, endIndex: number, symbolInstance: SymbolInstance, isText: boolean) {\n for (let b = startIndex; b < endIndex; b++) {\n const box: CollisionBox = this.collisionBoxArray.get(b);\n const x1 = box.x1;\n const y1 = box.y1;\n const x2 = box.x2;\n const y2 = box.y2;\n\n this.addCollisionDebugVertices(x1, y1, x2, y2,\n isText ? this.textCollisionBox : this.iconCollisionBox,\n box.anchorPoint, symbolInstance);\n }\n }\n\n generateCollisionDebugBuffers() {\n if (this.hasDebugData()) {\n this.destroyDebugData();\n }\n\n this.textCollisionBox = new CollisionBuffers(CollisionBoxLayoutArray, collisionBoxLayout.members, LineIndexArray);\n this.iconCollisionBox = new CollisionBuffers(CollisionBoxLayoutArray, collisionBoxLayout.members, LineIndexArray);\n\n for (let i = 0; i < this.symbolInstances.length; i++) {\n const symbolInstance = this.symbolInstances.get(i);\n this.addDebugCollisionBoxes(symbolInstance.textBoxStartIndex, symbolInstance.textBoxEndIndex, symbolInstance, true);\n this.addDebugCollisionBoxes(symbolInstance.verticalTextBoxStartIndex, symbolInstance.verticalTextBoxEndIndex, symbolInstance, true);\n this.addDebugCollisionBoxes(symbolInstance.iconBoxStartIndex, symbolInstance.iconBoxEndIndex, symbolInstance, false);\n this.addDebugCollisionBoxes(symbolInstance.verticalIconBoxStartIndex, symbolInstance.verticalIconBoxEndIndex, symbolInstance, false);\n }\n }\n\n // These flat arrays are meant to be quicker to iterate over than the source\n // CollisionBoxArray\n _deserializeCollisionBoxesForSymbol(\n collisionBoxArray: CollisionBoxArray,\n textStartIndex: number,\n textEndIndex: number,\n verticalTextStartIndex: number,\n verticalTextEndIndex: number,\n iconStartIndex: number,\n iconEndIndex: number,\n verticalIconStartIndex: number,\n verticalIconEndIndex: number\n ): CollisionArrays {\n\n const collisionArrays = {} as CollisionArrays;\n for (let k = textStartIndex; k < textEndIndex; k++) {\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.textBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.textFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n for (let k = verticalTextStartIndex; k < verticalTextEndIndex; k++) {\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.verticalTextBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.verticalTextFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n for (let k = iconStartIndex; k < iconEndIndex; k++) {\n // An icon can only have one box now, so this indexing is a bit vestigial...\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.iconBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.iconFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n for (let k = verticalIconStartIndex; k < verticalIconEndIndex; k++) {\n // An icon can only have one box now, so this indexing is a bit vestigial...\n const box: CollisionBox = collisionBoxArray.get(k);\n collisionArrays.verticalIconBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY};\n collisionArrays.verticalIconFeatureIndex = box.featureIndex;\n break; // Only one box allowed per instance\n }\n return collisionArrays;\n }\n\n deserializeCollisionBoxes(collisionBoxArray: CollisionBoxArray) {\n this.collisionArrays = [];\n for (let i = 0; i < this.symbolInstances.length; i++) {\n const symbolInstance = this.symbolInstances.get(i);\n this.collisionArrays.push(this._deserializeCollisionBoxesForSymbol(\n collisionBoxArray,\n symbolInstance.textBoxStartIndex,\n symbolInstance.textBoxEndIndex,\n symbolInstance.verticalTextBoxStartIndex,\n symbolInstance.verticalTextBoxEndIndex,\n symbolInstance.iconBoxStartIndex,\n symbolInstance.iconBoxEndIndex,\n symbolInstance.verticalIconBoxStartIndex,\n symbolInstance.verticalIconBoxEndIndex\n ));\n }\n }\n\n hasTextData() {\n return this.text.segments.get().length > 0;\n }\n\n hasIconData() {\n return this.icon.segments.get().length > 0;\n }\n\n hasDebugData() {\n return this.textCollisionBox && this.iconCollisionBox;\n }\n\n hasTextCollisionBoxData() {\n return this.hasDebugData() && this.textCollisionBox.segments.get().length > 0;\n }\n\n hasIconCollisionBoxData() {\n return this.hasDebugData() && this.iconCollisionBox.segments.get().length > 0;\n }\n\n addIndicesForPlacedSymbol(iconOrText: SymbolBuffers, placedSymbolIndex: number) {\n const placedSymbol = iconOrText.placedSymbolArray.get(placedSymbolIndex);\n\n const endIndex = placedSymbol.vertexStartIndex + placedSymbol.numGlyphs * 4;\n for (let vertexIndex = placedSymbol.vertexStartIndex; vertexIndex < endIndex; vertexIndex += 4) {\n iconOrText.indexArray.emplaceBack(vertexIndex, vertexIndex + 1, vertexIndex + 2);\n iconOrText.indexArray.emplaceBack(vertexIndex + 1, vertexIndex + 2, vertexIndex + 3);\n }\n }\n\n getSortedSymbolIndexes(angle: number) {\n if (this.sortedAngle === angle && this.symbolInstanceIndexes !== undefined) {\n return this.symbolInstanceIndexes;\n }\n const sin = Math.sin(angle);\n const cos = Math.cos(angle);\n const rotatedYs = [];\n const featureIndexes = [];\n const result = [];\n\n for (let i = 0; i < this.symbolInstances.length; ++i) {\n result.push(i);\n const symbolInstance = this.symbolInstances.get(i);\n rotatedYs.push(Math.round(sin * symbolInstance.anchorX + cos * symbolInstance.anchorY) | 0);\n featureIndexes.push(symbolInstance.featureIndex);\n }\n\n result.sort((aIndex, bIndex) => {\n return (rotatedYs[aIndex] - rotatedYs[bIndex]) ||\n (featureIndexes[bIndex] - featureIndexes[aIndex]);\n });\n\n return result;\n }\n\n addToSortKeyRanges(symbolInstanceIndex: number, sortKey: number) {\n const last = this.sortKeyRanges[this.sortKeyRanges.length - 1];\n if (last && last.sortKey === sortKey) {\n last.symbolInstanceEnd = symbolInstanceIndex + 1;\n } else {\n this.sortKeyRanges.push({\n sortKey,\n symbolInstanceStart: symbolInstanceIndex,\n symbolInstanceEnd: symbolInstanceIndex + 1\n });\n }\n }\n\n sortFeatures(angle: number) {\n if (!this.sortFeaturesByY) return;\n if (this.sortedAngle === angle) return;\n\n // The current approach to sorting doesn't sort across segments so don't try.\n // Sorting within segments separately seemed not to be worth the complexity.\n if (this.text.segments.get().length > 1 || this.icon.segments.get().length > 1) return;\n\n // If the symbols are allowed to overlap sort them by their vertical screen position.\n // The index array buffer is rewritten to reference the (unchanged) vertices in the\n // sorted order.\n\n // To avoid sorting the actual symbolInstance array we sort an array of indexes.\n this.symbolInstanceIndexes = this.getSortedSymbolIndexes(angle);\n this.sortedAngle = angle;\n\n this.text.indexArray.clear();\n this.icon.indexArray.clear();\n\n this.featureSortOrder = [];\n\n for (const i of this.symbolInstanceIndexes) {\n const symbolInstance = this.symbolInstances.get(i);\n this.featureSortOrder.push(symbolInstance.featureIndex);\n\n [\n symbolInstance.rightJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.leftJustifiedTextSymbolIndex\n ].forEach((index, i, array) => {\n // Only add a given index the first time it shows up,\n // to avoid duplicate opacity entries when multiple justifications\n // share the same glyphs.\n if (index >= 0 && array.indexOf(index) === i) {\n this.addIndicesForPlacedSymbol(this.text, index);\n }\n });\n\n if (symbolInstance.verticalPlacedTextSymbolIndex >= 0) {\n this.addIndicesForPlacedSymbol(this.text, symbolInstance.verticalPlacedTextSymbolIndex);\n }\n\n if (symbolInstance.placedIconSymbolIndex >= 0) {\n this.addIndicesForPlacedSymbol(this.icon, symbolInstance.placedIconSymbolIndex);\n }\n\n if (symbolInstance.verticalPlacedIconSymbolIndex >= 0) {\n this.addIndicesForPlacedSymbol(this.icon, symbolInstance.verticalPlacedIconSymbolIndex);\n }\n }\n\n if (this.text.indexBuffer) this.text.indexBuffer.updateData(this.text.indexArray);\n if (this.icon.indexBuffer) this.icon.indexBuffer.updateData(this.icon.indexArray);\n }\n}\n\nregister('SymbolBucket', SymbolBucket, {\n omit: ['layers', 'collisionBoxArray', 'features', 'compareText']\n});\n\n// this constant is based on the size of StructArray indexes used in a symbol\n// bucket--namely, glyphOffsetArrayStart\n// eg the max valid UInt16 is 65,535\n// See https://github.com/mapbox/mapbox-gl-js/issues/2907 for motivation\n// lineStartIndex and textBoxStartIndex could potentially be concerns\n// but we expect there to be many fewer boxes/lines than glyphs\nSymbolBucket.MAX_GLYPHS = 65535;\n\nSymbolBucket.addDynamicAttributes = addDynamicAttributes;\n\nexport {addDynamicAttributes};\n","import type {SymbolFeature} from '../data/bucket/symbol_bucket';\n\nexport function mergeLines(features: Array): Array {\n const leftIndex: {[_: string]: number} = {};\n const rightIndex: {[_: string]: number} = {};\n const mergedFeatures = [];\n let mergedIndex = 0;\n\n function add(k) {\n mergedFeatures.push(features[k]);\n mergedIndex++;\n }\n\n function mergeFromRight(leftKey: string, rightKey: string, geom) {\n const i = rightIndex[leftKey];\n delete rightIndex[leftKey];\n rightIndex[rightKey] = i;\n\n mergedFeatures[i].geometry[0].pop();\n mergedFeatures[i].geometry[0] = mergedFeatures[i].geometry[0].concat(geom[0]);\n return i;\n }\n\n function mergeFromLeft(leftKey: string, rightKey: string, geom) {\n const i = leftIndex[rightKey];\n delete leftIndex[rightKey];\n leftIndex[leftKey] = i;\n\n mergedFeatures[i].geometry[0].shift();\n mergedFeatures[i].geometry[0] = geom[0].concat(mergedFeatures[i].geometry[0]);\n return i;\n }\n\n function getKey(text, geom, onRight?) {\n const point = onRight ? geom[0][geom[0].length - 1] : geom[0][0];\n return `${text}:${point.x}:${point.y}`;\n }\n\n for (let k = 0; k < features.length; k++) {\n const feature = features[k];\n const geom = feature.geometry;\n const text = feature.text ? feature.text.toString() : null;\n\n if (!text) {\n add(k);\n continue;\n }\n\n const leftKey = getKey(text, geom),\n rightKey = getKey(text, geom, true);\n\n if ((leftKey in rightIndex) && (rightKey in leftIndex) && (rightIndex[leftKey] !== leftIndex[rightKey])) {\n // found lines with the same text adjacent to both ends of the current line, merge all three\n const j = mergeFromLeft(leftKey, rightKey, geom);\n const i = mergeFromRight(leftKey, rightKey, mergedFeatures[j].geometry);\n\n delete leftIndex[leftKey];\n delete rightIndex[rightKey];\n\n rightIndex[getKey(text, mergedFeatures[i].geometry, true)] = i;\n mergedFeatures[j].geometry = null;\n\n } else if (leftKey in rightIndex) {\n // found mergeable line adjacent to the start of the current line, merge\n mergeFromRight(leftKey, rightKey, geom);\n\n } else if (rightKey in leftIndex) {\n // found mergeable line adjacent to the end of the current line, merge\n mergeFromLeft(leftKey, rightKey, geom);\n\n } else {\n // no adjacent lines, add as a new item\n add(k);\n leftIndex[leftKey] = mergedIndex - 1;\n rightIndex[rightKey] = mergedIndex - 1;\n }\n }\n\n return mergedFeatures.filter((f) => f.geometry);\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n ColorType\n } from '@maplibre/maplibre-gl-style-spec';\n \nexport type SymbolLayoutProps = {\n \"symbol-placement\": DataConstantProperty<\"point\" | \"line\" | \"line-center\">,\n \"symbol-spacing\": DataConstantProperty,\n \"symbol-avoid-edges\": DataConstantProperty,\n \"symbol-sort-key\": DataDrivenProperty,\n \"symbol-z-order\": DataConstantProperty<\"auto\" | \"viewport-y\" | \"source\">,\n \"icon-allow-overlap\": DataConstantProperty,\n \"icon-overlap\": DataConstantProperty<\"never\" | \"always\" | \"cooperative\">,\n \"icon-ignore-placement\": DataConstantProperty,\n \"icon-optional\": DataConstantProperty,\n \"icon-rotation-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"auto\">,\n \"icon-size\": DataDrivenProperty,\n \"icon-text-fit\": DataConstantProperty<\"none\" | \"width\" | \"height\" | \"both\">,\n \"icon-text-fit-padding\": DataConstantProperty<[number, number, number, number]>,\n \"icon-image\": DataDrivenProperty,\n \"icon-rotate\": DataDrivenProperty,\n \"icon-padding\": DataDrivenProperty,\n \"icon-keep-upright\": DataConstantProperty,\n \"icon-offset\": DataDrivenProperty<[number, number]>,\n \"icon-anchor\": DataDrivenProperty<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"icon-pitch-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"auto\">,\n \"text-pitch-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"auto\">,\n \"text-rotation-alignment\": DataConstantProperty<\"map\" | \"viewport\" | \"viewport-glyph\" | \"auto\">,\n \"text-field\": DataDrivenProperty,\n \"text-font\": DataDrivenProperty>,\n \"text-size\": DataDrivenProperty,\n \"text-max-width\": DataDrivenProperty,\n \"text-line-height\": DataConstantProperty,\n \"text-letter-spacing\": DataDrivenProperty,\n \"text-justify\": DataDrivenProperty<\"auto\" | \"left\" | \"center\" | \"right\">,\n \"text-radial-offset\": DataDrivenProperty,\n \"text-variable-anchor\": DataConstantProperty>,\n \"text-variable-anchor-offset\": DataDrivenProperty,\n \"text-anchor\": DataDrivenProperty<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"text-max-angle\": DataConstantProperty,\n \"text-writing-mode\": DataConstantProperty>,\n \"text-rotate\": DataDrivenProperty,\n \"text-padding\": DataConstantProperty,\n \"text-keep-upright\": DataConstantProperty,\n \"text-transform\": DataDrivenProperty<\"none\" | \"uppercase\" | \"lowercase\">,\n \"text-offset\": DataDrivenProperty<[number, number]>,\n \"text-allow-overlap\": DataConstantProperty,\n \"text-overlap\": DataConstantProperty<\"never\" | \"always\" | \"cooperative\">,\n \"text-ignore-placement\": DataConstantProperty,\n \"text-optional\": DataConstantProperty,\n};\n\nexport type SymbolLayoutPropsPossiblyEvaluated = {\n \"symbol-placement\": \"point\" | \"line\" | \"line-center\",\n \"symbol-spacing\": number,\n \"symbol-avoid-edges\": boolean,\n \"symbol-sort-key\": PossiblyEvaluatedPropertyValue,\n \"symbol-z-order\": \"auto\" | \"viewport-y\" | \"source\",\n \"icon-allow-overlap\": boolean,\n \"icon-overlap\": \"never\" | \"always\" | \"cooperative\",\n \"icon-ignore-placement\": boolean,\n \"icon-optional\": boolean,\n \"icon-rotation-alignment\": \"map\" | \"viewport\" | \"auto\",\n \"icon-size\": PossiblyEvaluatedPropertyValue,\n \"icon-text-fit\": \"none\" | \"width\" | \"height\" | \"both\",\n \"icon-text-fit-padding\": [number, number, number, number],\n \"icon-image\": PossiblyEvaluatedPropertyValue,\n \"icon-rotate\": PossiblyEvaluatedPropertyValue,\n \"icon-padding\": PossiblyEvaluatedPropertyValue,\n \"icon-keep-upright\": boolean,\n \"icon-offset\": PossiblyEvaluatedPropertyValue<[number, number]>,\n \"icon-anchor\": PossiblyEvaluatedPropertyValue<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"icon-pitch-alignment\": \"map\" | \"viewport\" | \"auto\",\n \"text-pitch-alignment\": \"map\" | \"viewport\" | \"auto\",\n \"text-rotation-alignment\": \"map\" | \"viewport\" | \"viewport-glyph\" | \"auto\",\n \"text-field\": PossiblyEvaluatedPropertyValue,\n \"text-font\": PossiblyEvaluatedPropertyValue>,\n \"text-size\": PossiblyEvaluatedPropertyValue,\n \"text-max-width\": PossiblyEvaluatedPropertyValue,\n \"text-line-height\": number,\n \"text-letter-spacing\": PossiblyEvaluatedPropertyValue,\n \"text-justify\": PossiblyEvaluatedPropertyValue<\"auto\" | \"left\" | \"center\" | \"right\">,\n \"text-radial-offset\": PossiblyEvaluatedPropertyValue,\n \"text-variable-anchor\": Array<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"text-variable-anchor-offset\": PossiblyEvaluatedPropertyValue,\n \"text-anchor\": PossiblyEvaluatedPropertyValue<\"center\" | \"left\" | \"right\" | \"top\" | \"bottom\" | \"top-left\" | \"top-right\" | \"bottom-left\" | \"bottom-right\">,\n \"text-max-angle\": number,\n \"text-writing-mode\": Array<\"horizontal\" | \"vertical\">,\n \"text-rotate\": PossiblyEvaluatedPropertyValue,\n \"text-padding\": number,\n \"text-keep-upright\": boolean,\n \"text-transform\": PossiblyEvaluatedPropertyValue<\"none\" | \"uppercase\" | \"lowercase\">,\n \"text-offset\": PossiblyEvaluatedPropertyValue<[number, number]>,\n \"text-allow-overlap\": boolean,\n \"text-overlap\": \"never\" | \"always\" | \"cooperative\",\n \"text-ignore-placement\": boolean,\n \"text-optional\": boolean,\n};\n\nlet layout: Properties;\nconst getLayout = () => layout = layout || new Properties({\n \"symbol-placement\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-placement\"] as any as StylePropertySpecification),\n \"symbol-spacing\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-spacing\"] as any as StylePropertySpecification),\n \"symbol-avoid-edges\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-avoid-edges\"] as any as StylePropertySpecification),\n \"symbol-sort-key\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"symbol-sort-key\"] as any as StylePropertySpecification),\n \"symbol-z-order\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"symbol-z-order\"] as any as StylePropertySpecification),\n \"icon-allow-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-allow-overlap\"] as any as StylePropertySpecification),\n \"icon-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-overlap\"] as any as StylePropertySpecification),\n \"icon-ignore-placement\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-ignore-placement\"] as any as StylePropertySpecification),\n \"icon-optional\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-optional\"] as any as StylePropertySpecification),\n \"icon-rotation-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-rotation-alignment\"] as any as StylePropertySpecification),\n \"icon-size\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-size\"] as any as StylePropertySpecification),\n \"icon-text-fit\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-text-fit\"] as any as StylePropertySpecification),\n \"icon-text-fit-padding\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-text-fit-padding\"] as any as StylePropertySpecification),\n \"icon-image\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-image\"] as any as StylePropertySpecification),\n \"icon-rotate\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-rotate\"] as any as StylePropertySpecification),\n \"icon-padding\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-padding\"] as any as StylePropertySpecification),\n \"icon-keep-upright\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-keep-upright\"] as any as StylePropertySpecification),\n \"icon-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-offset\"] as any as StylePropertySpecification),\n \"icon-anchor\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"icon-anchor\"] as any as StylePropertySpecification),\n \"icon-pitch-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"icon-pitch-alignment\"] as any as StylePropertySpecification),\n \"text-pitch-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-pitch-alignment\"] as any as StylePropertySpecification),\n \"text-rotation-alignment\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-rotation-alignment\"] as any as StylePropertySpecification),\n \"text-field\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-field\"] as any as StylePropertySpecification),\n \"text-font\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-font\"] as any as StylePropertySpecification),\n \"text-size\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-size\"] as any as StylePropertySpecification),\n \"text-max-width\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-max-width\"] as any as StylePropertySpecification),\n \"text-line-height\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-line-height\"] as any as StylePropertySpecification),\n \"text-letter-spacing\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-letter-spacing\"] as any as StylePropertySpecification),\n \"text-justify\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-justify\"] as any as StylePropertySpecification),\n \"text-radial-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-radial-offset\"] as any as StylePropertySpecification),\n \"text-variable-anchor\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-variable-anchor\"] as any as StylePropertySpecification),\n \"text-variable-anchor-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-variable-anchor-offset\"] as any as StylePropertySpecification),\n \"text-anchor\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-anchor\"] as any as StylePropertySpecification),\n \"text-max-angle\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-max-angle\"] as any as StylePropertySpecification),\n \"text-writing-mode\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-writing-mode\"] as any as StylePropertySpecification),\n \"text-rotate\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-rotate\"] as any as StylePropertySpecification),\n \"text-padding\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-padding\"] as any as StylePropertySpecification),\n \"text-keep-upright\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-keep-upright\"] as any as StylePropertySpecification),\n \"text-transform\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-transform\"] as any as StylePropertySpecification),\n \"text-offset\": new DataDrivenProperty(styleSpec[\"layout_symbol\"][\"text-offset\"] as any as StylePropertySpecification),\n \"text-allow-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-allow-overlap\"] as any as StylePropertySpecification),\n \"text-overlap\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-overlap\"] as any as StylePropertySpecification),\n \"text-ignore-placement\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-ignore-placement\"] as any as StylePropertySpecification),\n \"text-optional\": new DataConstantProperty(styleSpec[\"layout_symbol\"][\"text-optional\"] as any as StylePropertySpecification),\n});\n\nexport type SymbolPaintProps = {\n \"icon-opacity\": DataDrivenProperty,\n \"icon-color\": DataDrivenProperty,\n \"icon-halo-color\": DataDrivenProperty,\n \"icon-halo-width\": DataDrivenProperty,\n \"icon-halo-blur\": DataDrivenProperty,\n \"icon-translate\": DataConstantProperty<[number, number]>,\n \"icon-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n \"text-opacity\": DataDrivenProperty,\n \"text-color\": DataDrivenProperty,\n \"text-halo-color\": DataDrivenProperty,\n \"text-halo-width\": DataDrivenProperty,\n \"text-halo-blur\": DataDrivenProperty,\n \"text-translate\": DataConstantProperty<[number, number]>,\n \"text-translate-anchor\": DataConstantProperty<\"map\" | \"viewport\">,\n};\n\nexport type SymbolPaintPropsPossiblyEvaluated = {\n \"icon-opacity\": PossiblyEvaluatedPropertyValue,\n \"icon-color\": PossiblyEvaluatedPropertyValue,\n \"icon-halo-color\": PossiblyEvaluatedPropertyValue,\n \"icon-halo-width\": PossiblyEvaluatedPropertyValue,\n \"icon-halo-blur\": PossiblyEvaluatedPropertyValue,\n \"icon-translate\": [number, number],\n \"icon-translate-anchor\": \"map\" | \"viewport\",\n \"text-opacity\": PossiblyEvaluatedPropertyValue,\n \"text-color\": PossiblyEvaluatedPropertyValue,\n \"text-halo-color\": PossiblyEvaluatedPropertyValue,\n \"text-halo-width\": PossiblyEvaluatedPropertyValue,\n \"text-halo-blur\": PossiblyEvaluatedPropertyValue,\n \"text-translate\": [number, number],\n \"text-translate-anchor\": \"map\" | \"viewport\",\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"icon-opacity\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-opacity\"] as any as StylePropertySpecification),\n \"icon-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-color\"] as any as StylePropertySpecification),\n \"icon-halo-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-halo-color\"] as any as StylePropertySpecification),\n \"icon-halo-width\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-halo-width\"] as any as StylePropertySpecification),\n \"icon-halo-blur\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"icon-halo-blur\"] as any as StylePropertySpecification),\n \"icon-translate\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"icon-translate\"] as any as StylePropertySpecification),\n \"icon-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"icon-translate-anchor\"] as any as StylePropertySpecification),\n \"text-opacity\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-opacity\"] as any as StylePropertySpecification),\n \"text-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-color\"] as any as StylePropertySpecification, { runtimeType: ColorType, getOverride: (o) => o.textColor, hasOverride: (o) => !!o.textColor }),\n \"text-halo-color\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-halo-color\"] as any as StylePropertySpecification),\n \"text-halo-width\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-halo-width\"] as any as StylePropertySpecification),\n \"text-halo-blur\": new DataDrivenProperty(styleSpec[\"paint_symbol\"][\"text-halo-blur\"] as any as StylePropertySpecification),\n \"text-translate\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"text-translate\"] as any as StylePropertySpecification),\n \"text-translate-anchor\": new DataConstantProperty(styleSpec[\"paint_symbol\"][\"text-translate-anchor\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() }, get layout() { return getLayout() } });","import type {Expression, EvaluationContext, Type, ZoomConstantExpression} from '@maplibre/maplibre-gl-style-spec';\nimport {NullType} from '@maplibre/maplibre-gl-style-spec';\nimport {PossiblyEvaluatedPropertyValue} from './properties';\nimport {register} from '../util/web_worker_transfer';\n\n// This is an internal expression class. It is only used in GL JS and\n// has GL JS dependencies which can break the standalone style-spec module\nexport class FormatSectionOverride implements Expression {\n type: Type;\n defaultValue: PossiblyEvaluatedPropertyValue;\n\n constructor(defaultValue: PossiblyEvaluatedPropertyValue) {\n if (defaultValue.property.overrides === undefined) throw new Error('overrides must be provided to instantiate FormatSectionOverride class');\n this.type = defaultValue.property.overrides ? defaultValue.property.overrides.runtimeType : NullType;\n this.defaultValue = defaultValue;\n }\n\n evaluate(ctx: EvaluationContext) {\n if (ctx.formattedSection) {\n const overrides = this.defaultValue.property.overrides;\n if (overrides && overrides.hasOverride(ctx.formattedSection)) {\n return overrides.getOverride(ctx.formattedSection);\n }\n }\n\n if (ctx.feature && ctx.featureState) {\n return this.defaultValue.evaluate(ctx.feature, ctx.featureState);\n }\n\n return this.defaultValue.property.specification.default;\n }\n\n eachChild(fn: (_: Expression) => void) {\n if (!this.defaultValue.isConstant()) {\n const expr: ZoomConstantExpression<'source'> = (this.defaultValue.value as any);\n fn(expr._styleExpression.expression);\n }\n }\n\n // Cannot be statically evaluated, as the output depends on the evaluation context.\n outputDefined() {\n return false;\n }\n\n serialize() {\n return null;\n }\n}\n\nregister('FormatSectionOverride', FormatSectionOverride, {omit: ['defaultValue']});\n","import {StyleLayer} from '../style_layer';\n\nimport {SymbolBucket, SymbolFeature} from '../../data/bucket/symbol_bucket';\nimport {resolveTokens} from '../../util/resolve_tokens';\nimport properties, {SymbolLayoutPropsPossiblyEvaluated, SymbolPaintPropsPossiblyEvaluated} from './symbol_style_layer_properties.g';\n\nimport {\n Transitionable,\n Transitioning,\n Layout,\n PossiblyEvaluated,\n PossiblyEvaluatedPropertyValue,\n PropertyValue\n} from '../properties';\n\nimport {\n isExpression,\n StyleExpression,\n ZoomConstantExpression,\n ZoomDependentExpression,\n FormattedType,\n typeOf,\n Formatted,\n FormatExpression,\n Literal} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {BucketParameters} from '../../data/bucket';\nimport type {SymbolLayoutProps, SymbolPaintProps} from './symbol_style_layer_properties.g';\nimport type {EvaluationParameters} from '../evaluation_parameters';\nimport type {Expression, Feature, SourceExpression, LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {CanonicalTileID} from '../../source/tile_id';\nimport {FormatSectionOverride} from '../format_section_override';\n\nexport class SymbolStyleLayer extends StyleLayer {\n _unevaluatedLayout: Layout;\n layout: PossiblyEvaluated;\n\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n\n recalculate(parameters: EvaluationParameters, availableImages: Array) {\n super.recalculate(parameters, availableImages);\n\n if (this.layout.get('icon-rotation-alignment') === 'auto') {\n if (this.layout.get('symbol-placement') !== 'point') {\n this.layout._values['icon-rotation-alignment'] = 'map';\n } else {\n this.layout._values['icon-rotation-alignment'] = 'viewport';\n }\n }\n\n if (this.layout.get('text-rotation-alignment') === 'auto') {\n if (this.layout.get('symbol-placement') !== 'point') {\n this.layout._values['text-rotation-alignment'] = 'map';\n } else {\n this.layout._values['text-rotation-alignment'] = 'viewport';\n }\n }\n\n // If unspecified, `*-pitch-alignment` inherits `*-rotation-alignment`\n if (this.layout.get('text-pitch-alignment') === 'auto') {\n this.layout._values['text-pitch-alignment'] = this.layout.get('text-rotation-alignment') === 'map' ? 'map' : 'viewport';\n }\n if (this.layout.get('icon-pitch-alignment') === 'auto') {\n this.layout._values['icon-pitch-alignment'] = this.layout.get('icon-rotation-alignment');\n }\n\n if (this.layout.get('symbol-placement') === 'point') {\n const writingModes = this.layout.get('text-writing-mode');\n if (writingModes) {\n // remove duplicates, preserving order\n const deduped = [];\n for (const m of writingModes) {\n if (deduped.indexOf(m) < 0) deduped.push(m);\n }\n this.layout._values['text-writing-mode'] = deduped;\n } else {\n this.layout._values['text-writing-mode'] = ['horizontal'];\n }\n }\n\n this._setPaintOverrides();\n }\n\n getValueAndResolveTokens(name: any, feature: Feature, canonical: CanonicalTileID, availableImages: Array) {\n const value = this.layout.get(name).evaluate(feature, {}, canonical, availableImages);\n const unevaluated = this._unevaluatedLayout._values[name];\n if (!unevaluated.isDataDriven() && !isExpression(unevaluated.value) && value) {\n return resolveTokens(feature.properties, value);\n }\n\n return value;\n }\n\n createBucket(parameters: BucketParameters) {\n return new SymbolBucket(parameters);\n }\n\n queryRadius(): number {\n return 0;\n }\n\n queryIntersectsFeature(): boolean {\n throw new Error('Should take a different path in FeatureIndex');\n }\n\n _setPaintOverrides() {\n for (const overridable of properties.paint.overridableProperties) {\n if (!SymbolStyleLayer.hasPaintOverride(this.layout, overridable)) {\n continue;\n }\n const overriden = this.paint.get(overridable as keyof SymbolPaintPropsPossiblyEvaluated) as PossiblyEvaluatedPropertyValue;\n const override = new FormatSectionOverride(overriden);\n const styleExpression = new StyleExpression(override, overriden.property.specification);\n let expression = null;\n if (overriden.value.kind === 'constant' || overriden.value.kind === 'source') {\n expression = new ZoomConstantExpression('source', styleExpression) as SourceExpression;\n } else {\n expression = new ZoomDependentExpression('composite',\n styleExpression,\n overriden.value.zoomStops);\n }\n this.paint._values[overridable] = new PossiblyEvaluatedPropertyValue(overriden.property,\n expression,\n overriden.parameters);\n }\n }\n\n _handleOverridablePaintPropertyUpdate(name: string, oldValue: PropertyValue, newValue: PropertyValue): boolean {\n if (!this.layout || oldValue.isDataDriven() || newValue.isDataDriven()) {\n return false;\n }\n return SymbolStyleLayer.hasPaintOverride(this.layout, name);\n }\n\n static hasPaintOverride(layout: PossiblyEvaluated, propertyName: string): boolean {\n const textField = layout.get('text-field');\n const property = properties.paint.properties[propertyName];\n let hasOverrides = false;\n\n const checkSections = (sections) => {\n for (const section of sections) {\n if (property.overrides && property.overrides.hasOverride(section)) {\n hasOverrides = true;\n return;\n }\n }\n };\n\n if (textField.value.kind === 'constant' && textField.value.value instanceof Formatted) {\n checkSections(textField.value.value.sections);\n } else if (textField.value.kind === 'source') {\n\n const checkExpression = (expression: Expression) => {\n if (hasOverrides) return;\n\n if (expression instanceof Literal && typeOf(expression.value) === FormattedType) {\n const formatted: Formatted = (expression.value as any);\n checkSections(formatted.sections);\n } else if (expression instanceof FormatExpression) {\n checkSections(expression.sections);\n } else {\n expression.eachChild(checkExpression);\n }\n };\n\n const expr: ZoomConstantExpression<'source'> = (textField.value as any);\n if (expr._styleExpression) {\n checkExpression(expr._styleExpression.expression);\n }\n }\n\n return hasOverrides;\n }\n}\n\nexport type SymbolPadding = [number, number, number, number];\n\nexport function getIconPadding(layout: PossiblyEvaluated, feature: SymbolFeature, canonical: CanonicalTileID, pixelRatio = 1): SymbolPadding {\n // Support text-padding in addition to icon-padding? Unclear how to apply asymmetric text-padding to the radius for collision circles.\n const result = layout.get('icon-padding').evaluate(feature, {}, canonical);\n const values = result && result.values;\n\n return [\n values[0] * pixelRatio,\n values[1] * pixelRatio,\n values[2] * pixelRatio,\n values[3] * pixelRatio,\n ];\n}\n","/**\n * Replace tokens in a string template with values in an object\n *\n * @param properties - a key/value relationship between tokens and replacements\n * @param text - the template string\n * @returns the template with tokens replaced\n */\nexport function resolveTokens(\n properties: {\n readonly [x: string]: unknown;\n } | null,\n text: string\n): string {\n return text.replace(/{([^{}]+)}/g, (match, key: string) => {\n return properties && key in properties ? String(properties[key]) : '';\n });\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type BackgroundPaintProps = {\n \"background-color\": DataConstantProperty,\n \"background-pattern\": CrossFadedProperty,\n \"background-opacity\": DataConstantProperty,\n};\n\nexport type BackgroundPaintPropsPossiblyEvaluated = {\n \"background-color\": Color,\n \"background-pattern\": CrossFaded,\n \"background-opacity\": number,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"background-color\": new DataConstantProperty(styleSpec[\"paint_background\"][\"background-color\"] as any as StylePropertySpecification),\n \"background-pattern\": new CrossFadedProperty(styleSpec[\"paint_background\"][\"background-pattern\"] as any as StylePropertySpecification),\n \"background-opacity\": new DataConstantProperty(styleSpec[\"paint_background\"][\"background-opacity\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport properties, {BackgroundPaintPropsPossiblyEvaluated} from './background_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {BackgroundPaintProps} from './background_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport class BackgroundStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n}\n","// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'.\n/* eslint-disable */\n\nimport {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {\n Properties,\n DataConstantProperty,\n DataDrivenProperty,\n CrossFadedDataDrivenProperty,\n CrossFadedProperty,\n ColorRampProperty,\n PossiblyEvaluatedPropertyValue,\n CrossFaded\n} from '../properties';\n\nimport type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec';\n\n\nexport type RasterPaintProps = {\n \"raster-opacity\": DataConstantProperty,\n \"raster-hue-rotate\": DataConstantProperty,\n \"raster-brightness-min\": DataConstantProperty,\n \"raster-brightness-max\": DataConstantProperty,\n \"raster-saturation\": DataConstantProperty,\n \"raster-contrast\": DataConstantProperty,\n \"raster-resampling\": DataConstantProperty<\"linear\" | \"nearest\">,\n \"raster-fade-duration\": DataConstantProperty,\n};\n\nexport type RasterPaintPropsPossiblyEvaluated = {\n \"raster-opacity\": number,\n \"raster-hue-rotate\": number,\n \"raster-brightness-min\": number,\n \"raster-brightness-max\": number,\n \"raster-saturation\": number,\n \"raster-contrast\": number,\n \"raster-resampling\": \"linear\" | \"nearest\",\n \"raster-fade-duration\": number,\n};\n\nlet paint: Properties;\nconst getPaint = () => paint = paint || new Properties({\n \"raster-opacity\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-opacity\"] as any as StylePropertySpecification),\n \"raster-hue-rotate\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-hue-rotate\"] as any as StylePropertySpecification),\n \"raster-brightness-min\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-brightness-min\"] as any as StylePropertySpecification),\n \"raster-brightness-max\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-brightness-max\"] as any as StylePropertySpecification),\n \"raster-saturation\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-saturation\"] as any as StylePropertySpecification),\n \"raster-contrast\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-contrast\"] as any as StylePropertySpecification),\n \"raster-resampling\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-resampling\"] as any as StylePropertySpecification),\n \"raster-fade-duration\": new DataConstantProperty(styleSpec[\"paint_raster\"][\"raster-fade-duration\"] as any as StylePropertySpecification),\n});\n\nexport default ({ get paint() { return getPaint() } });","import {StyleLayer} from '../style_layer';\n\nimport properties, {RasterPaintPropsPossiblyEvaluated} from './raster_style_layer_properties.g';\nimport {Transitionable, Transitioning, PossiblyEvaluated} from '../properties';\n\nimport type {RasterPaintProps} from './raster_style_layer_properties.g';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport class RasterStyleLayer extends StyleLayer {\n _transitionablePaint: Transitionable;\n _transitioningPaint: Transitioning;\n paint: PossiblyEvaluated;\n\n constructor(layer: LayerSpecification) {\n super(layer, properties);\n }\n}\n","import {StyleLayer} from '../style_layer';\nimport type {Map} from '../../ui/map';\nimport {mat4} from 'gl-matrix';\nimport {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * @param gl - The map's gl context.\n * @param matrix - The map's camera matrix. It projects spherical mercator\n * coordinates to gl coordinates. The spherical mercator coordinate `[0, 0]` represents the\n * top left corner of the mercator world and `[1, 1]` represents the bottom right corner. When\n * the `renderingMode` is `\"3d\"`, the z coordinate is conformal. A box with identical x, y, and z\n * lengths in mercator units would be rendered as a cube. {@link MercatorCoordinate.fromLngLat}\n * can be used to project a `LngLat` to a mercator coordinate.\n */\ntype CustomRenderMethod = (gl: WebGLRenderingContext|WebGL2RenderingContext, matrix: mat4) => void;\n\n/**\n * Interface for custom style layers. This is a specification for\n * implementers to model: it is not an exported method or class.\n *\n * Custom layers allow a user to render directly into the map's GL context using the map's camera.\n * These layers can be added between any regular layers using {@link Map#addLayer}.\n *\n * Custom layers must have a unique `id` and must have the `type` of `\"custom\"`.\n * They must implement `render` and may implement `prerender`, `onAdd` and `onRemove`.\n * They can trigger rendering using {@link Map#triggerRepaint}\n * and they should appropriately handle {@link MapContextEvent} with `webglcontextlost` and `webglcontextrestored`.\n *\n * The `renderingMode` property controls whether the layer is treated as a `\"2d\"` or `\"3d\"` map layer. Use:\n * - `\"renderingMode\": \"3d\"` to use the depth buffer and share it with other layers\n * - `\"renderingMode\": \"2d\"` to add a layer with no depth. If you need to use the depth buffer for a `\"2d\"` layer you must use an offscreen\n * framebuffer and {@link CustomLayerInterface#prerender}\n *\n * @example\n * Custom layer implemented as ES6 class\n * ```ts\n * class NullIslandLayer {\n * constructor() {\n * this.id = 'null-island';\n * this.type = 'custom';\n * this.renderingMode = '2d';\n * }\n *\n * onAdd(map, gl) {\n * const vertexSource = `\n * uniform mat4 u_matrix;\n * void main() {\n * gl_Position = u_matrix * vec4(0.5, 0.5, 0.0, 1.0);\n * gl_PointSize = 20.0;\n * }`;\n *\n * const fragmentSource = `\n * void main() {\n * fragColor = vec4(1.0, 0.0, 0.0, 1.0);\n * }`;\n *\n * const vertexShader = gl.createShader(gl.VERTEX_SHADER);\n * gl.shaderSource(vertexShader, vertexSource);\n * gl.compileShader(vertexShader);\n * const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);\n * gl.shaderSource(fragmentShader, fragmentSource);\n * gl.compileShader(fragmentShader);\n *\n * this.program = gl.createProgram();\n * gl.attachShader(this.program, vertexShader);\n * gl.attachShader(this.program, fragmentShader);\n * gl.linkProgram(this.program);\n * }\n *\n * render(gl, matrix) {\n * gl.useProgram(this.program);\n * gl.uniformMatrix4fv(gl.getUniformLocation(this.program, \"u_matrix\"), false, matrix);\n * gl.drawArrays(gl.POINTS, 0, 1);\n * }\n * }\n *\n * map.on('load', function() {\n * map.addLayer(new NullIslandLayer());\n * });\n * ```\n */\nexport interface CustomLayerInterface {\n /**\n * A unique layer id.\n */\n id: string;\n /**\n * The layer's type. Must be `\"custom\"`.\n */\n type: 'custom';\n /**\n * Either `\"2d\"` or `\"3d\"`. Defaults to `\"2d\"`.\n */\n renderingMode?: '2d' | '3d';\n /**\n * Called during a render frame allowing the layer to draw into the GL context.\n *\n * The layer can assume blending and depth state is set to allow the layer to properly\n * blend and clip other layers. The layer cannot make any other assumptions about the\n * current GL state.\n *\n * If the layer needs to render to a texture, it should implement the `prerender` method\n * to do this and only use the `render` method for drawing directly into the main framebuffer.\n *\n * The blend function is set to `gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA)`. This expects\n * colors to be provided in premultiplied alpha form where the `r`, `g` and `b` values are already\n * multiplied by the `a` value. If you are unable to provide colors in premultiplied form you\n * may want to change the blend function to\n * `gl.blendFuncSeparate(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA, gl.ONE, gl.ONE_MINUS_SRC_ALPHA)`.\n */\n render: CustomRenderMethod;\n /**\n * Optional method called during a render frame to allow a layer to prepare resources or render into a texture.\n *\n * The layer cannot make any assumptions about the current GL state and must bind a framebuffer before rendering.\n */\n prerender?: CustomRenderMethod;\n /**\n * Optional method called when the layer has been added to the Map with {@link Map#addLayer}. This\n * gives the layer a chance to initialize gl resources and register event listeners.\n *\n * @param map - The Map this custom layer was just added to.\n * @param gl - The gl context for the map.\n */\n onAdd?(map: Map, gl: WebGLRenderingContext | WebGL2RenderingContext): void;\n /**\n * Optional method called when the layer has been removed from the Map with {@link Map#removeLayer}. This\n * gives the layer a chance to clean up gl resources and event listeners.\n *\n * @param map - The Map this custom layer was just added to.\n * @param gl - The gl context for the map.\n */\n onRemove?(map: Map, gl: WebGLRenderingContext | WebGL2RenderingContext): void;\n}\n\nexport function validateCustomStyleLayer(layerObject: CustomLayerInterface) {\n const errors = [];\n const id = layerObject.id;\n\n if (id === undefined) {\n errors.push({\n message: `layers.${id}: missing required property \"id\"`\n });\n }\n\n if (layerObject.render === undefined) {\n errors.push({\n message: `layers.${id}: missing required method \"render\"`\n });\n }\n\n if (layerObject.renderingMode &&\n layerObject.renderingMode !== '2d' &&\n layerObject.renderingMode !== '3d') {\n errors.push({\n message: `layers.${id}: property \"renderingMode\" must be either \"2d\" or \"3d\"`\n });\n }\n\n return errors;\n}\n\nexport class CustomStyleLayer extends StyleLayer {\n\n implementation: CustomLayerInterface;\n\n constructor(implementation: CustomLayerInterface) {\n super(implementation, {});\n this.implementation = implementation;\n }\n\n is3D() {\n return this.implementation.renderingMode === '3d';\n }\n\n hasOffscreenPass() {\n return this.implementation.prerender !== undefined;\n }\n\n recalculate() {}\n updateTransitions() {}\n hasTransition() { return false; }\n\n serialize(): LayerSpecification {\n throw new Error('Custom layers cannot be serialized');\n }\n\n onAdd = (map: Map) => {\n if (this.implementation.onAdd) {\n this.implementation.onAdd(map, map.painter.context.gl);\n }\n };\n\n onRemove = (map: Map) => {\n if (this.implementation.onRemove) {\n this.implementation.onRemove(map, map.painter.context.gl);\n }\n };\n}\n","/**\n * Invokes the wrapped function in a non-blocking way when trigger() is called. Invocation requests\n * are ignored until the function was actually invoked.\n */\nexport class ThrottledInvoker {\n _channel: MessageChannel;\n _triggered: boolean;\n _callback: Function;\n\n constructor(callback: Function) {\n this._callback = callback;\n this._triggered = false;\n if (typeof MessageChannel !== 'undefined') {\n this._channel = new MessageChannel();\n this._channel.port2.onmessage = () => {\n this._triggered = false;\n this._callback();\n };\n }\n }\n\n trigger() {\n if (!this._triggered) {\n this._triggered = true;\n if (this._channel) {\n this._channel.port1.postMessage(true);\n } else {\n setTimeout(() => {\n this._triggered = false;\n this._callback();\n }, 0);\n }\n }\n }\n\n remove() {\n delete this._channel;\n this._callback = () => {};\n }\n}\n","import {wrap} from '../util/util';\n\n/*\n* Approximate radius of the earth in meters.\n* Uses the WGS-84 approximation. The radius at the equator is ~6378137 and at the poles is ~6356752. https://en.wikipedia.org/wiki/World_Geodetic_System#WGS84\n* 6371008.8 is one published \"average radius\" see https://en.wikipedia.org/wiki/Earth_radius#Mean_radius, or ftp://athena.fsv.cvut.cz/ZFG/grs80-Moritz.pdf p.4\n*/\nexport const earthRadius = 6371008.8;\n\n/**\n * A {@link LngLat} object, an array of two numbers representing longitude and latitude,\n * or an object with `lng` and `lat` or `lon` and `lat` properties.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let v1 = new maplibregl.LngLat(-122.420679, 37.772537);\n * let v2 = [-122.420679, 37.772537];\n * let v3 = {lon: -122.420679, lat: 37.772537};\n * ```\n */\nexport type LngLatLike = LngLat | {\n lng: number;\n lat: number;\n} | {\n lon: number;\n lat: number;\n} | [number, number];\n\n/**\n * A `LngLat` object represents a given longitude and latitude coordinate, measured in degrees.\n * These coordinates are based on the [WGS84 (EPSG:4326) standard](https://en.wikipedia.org/wiki/World_Geodetic_System#WGS84).\n *\n * MapLibre GL JS uses longitude, latitude coordinate order (as opposed to latitude, longitude) to match the\n * [GeoJSON specification](https://tools.ietf.org/html/rfc7946).\n *\n * Note that any MapLibre GL JS method that accepts a `LngLat` object as an argument or option\n * can also accept an `Array` of two numbers and will perform an implicit conversion.\n * This flexible type is documented as {@link LngLatLike}.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(-123.9749, 40.7736);\n * ll.lng; // = -123.9749\n * ```\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Create a timeline animation](https://maplibre.org/maplibre-gl-js/docs/examples/timeline-animation/)\n */\nexport class LngLat {\n lng: number;\n lat: number;\n\n /**\n * @param lng - Longitude, measured in degrees.\n * @param lat - Latitude, measured in degrees.\n */\n constructor(lng: number, lat: number) {\n if (isNaN(lng) || isNaN(lat)) {\n throw new Error(`Invalid LngLat object: (${lng}, ${lat})`);\n }\n this.lng = +lng;\n this.lat = +lat;\n if (this.lat > 90 || this.lat < -90) {\n throw new Error('Invalid LngLat latitude value: must be between -90 and 90');\n }\n }\n\n /**\n * Returns a new `LngLat` object whose longitude is wrapped to the range (-180, 180).\n *\n * @returns The wrapped `LngLat` object.\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(286.0251, 40.7736);\n * let wrapped = ll.wrap();\n * wrapped.lng; // = -73.9749\n * ```\n */\n wrap() {\n return new LngLat(wrap(this.lng, -180, 180), this.lat);\n }\n\n /**\n * Returns the coordinates represented as an array of two numbers.\n *\n * @returns The coordinates represented as an array of longitude and latitude.\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(-73.9749, 40.7736);\n * ll.toArray(); // = [-73.9749, 40.7736]\n * ```\n */\n toArray(): [number, number] {\n return [this.lng, this.lat];\n }\n\n /**\n * Returns the coordinates represent as a string.\n *\n * @returns The coordinates represented as a string of the format `'LngLat(lng, lat)'`.\n * @example\n * ```ts\n * let ll = new maplibregl.LngLat(-73.9749, 40.7736);\n * ll.toString(); // = \"LngLat(-73.9749, 40.7736)\"\n * ```\n */\n toString(): string {\n return `LngLat(${this.lng}, ${this.lat})`;\n }\n\n /**\n * Returns the approximate distance between a pair of coordinates in meters\n * Uses the Haversine Formula (from R.W. Sinnott, \"Virtues of the Haversine\", Sky and Telescope, vol. 68, no. 2, 1984, p. 159)\n *\n * @param lngLat - coordinates to compute the distance to\n * @returns Distance in meters between the two coordinates.\n * @example\n * ```ts\n * let new_york = new maplibregl.LngLat(-74.0060, 40.7128);\n * let los_angeles = new maplibregl.LngLat(-118.2437, 34.0522);\n * new_york.distanceTo(los_angeles); // = 3935751.690893987, \"true distance\" using a non-spherical approximation is ~3966km\n * ```\n */\n distanceTo(lngLat: LngLat): number {\n const rad = Math.PI / 180;\n const lat1 = this.lat * rad;\n const lat2 = lngLat.lat * rad;\n const a = Math.sin(lat1) * Math.sin(lat2) + Math.cos(lat1) * Math.cos(lat2) * Math.cos((lngLat.lng - this.lng) * rad);\n\n const maxMeters = earthRadius * Math.acos(Math.min(a, 1));\n return maxMeters;\n }\n\n /**\n * Converts an array of two numbers or an object with `lng` and `lat` or `lon` and `lat` properties\n * to a `LngLat` object.\n *\n * If a `LngLat` object is passed in, the function returns it unchanged.\n *\n * @param input - An array of two numbers or object to convert, or a `LngLat` object to return.\n * @returns A new `LngLat` object, if a conversion occurred, or the original `LngLat` object.\n * @example\n * ```ts\n * let arr = [-73.9749, 40.7736];\n * let ll = maplibregl.LngLat.convert(arr);\n * ll; // = LngLat {lng: -73.9749, lat: 40.7736}\n * ```\n */\n static convert(input: LngLatLike): LngLat {\n if (input instanceof LngLat) {\n return input;\n }\n if (Array.isArray(input) && (input.length === 2 || input.length === 3)) {\n return new LngLat(Number(input[0]), Number(input[1]));\n }\n if (!Array.isArray(input) && typeof input === 'object' && input !== null) {\n return new LngLat(\n // flow can't refine this to have one of lng or lat, so we have to cast to any\n Number('lng' in input ? (input as any).lng : (input as any).lon),\n Number(input.lat)\n );\n }\n throw new Error('`LngLatLike` argument must be specified as a LngLat instance, an object {lng: , lat: }, an object {lon: , lat: }, or an array of [, ]');\n }\n}\n","import {LngLat, earthRadius} from '../geo/lng_lat';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport {IMercatorCoordinate} from '@maplibre/maplibre-gl-style-spec';\n\n/*\n * The average circumference of the world in meters.\n */\nconst earthCircumfrence = 2 * Math.PI * earthRadius; // meters\n\n/*\n * The circumference at a line of latitude in meters.\n */\nfunction circumferenceAtLatitude(latitude: number) {\n return earthCircumfrence * Math.cos(latitude * Math.PI / 180);\n}\n\nexport function mercatorXfromLng(lng: number) {\n return (180 + lng) / 360;\n}\n\nexport function mercatorYfromLat(lat: number) {\n return (180 - (180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360)))) / 360;\n}\n\nexport function mercatorZfromAltitude(altitude: number, lat: number) {\n return altitude / circumferenceAtLatitude(lat);\n}\n\nexport function lngFromMercatorX(x: number) {\n return x * 360 - 180;\n}\n\nexport function latFromMercatorY(y: number) {\n const y2 = 180 - y * 360;\n return 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90;\n}\n\nexport function altitudeFromMercatorZ(z: number, y: number) {\n return z * circumferenceAtLatitude(latFromMercatorY(y));\n}\n\n/**\n * Determine the Mercator scale factor for a given latitude, see\n * https://en.wikipedia.org/wiki/Mercator_projection#Scale_factor\n *\n * At the equator the scale factor will be 1, which increases at higher latitudes.\n *\n * @param lat - Latitude\n * @returns scale factor\n */\nexport function mercatorScale(lat: number) {\n return 1 / Math.cos(lat * Math.PI / 180);\n}\n\n/**\n * A `MercatorCoordinate` object represents a projected three dimensional position.\n *\n * `MercatorCoordinate` uses the web mercator projection ([EPSG:3857](https://epsg.io/3857)) with slightly different units:\n * - the size of 1 unit is the width of the projected world instead of the \"mercator meter\"\n * - the origin of the coordinate space is at the north-west corner instead of the middle\n *\n * For example, `MercatorCoordinate(0, 0, 0)` is the north-west corner of the mercator world and\n * `MercatorCoordinate(1, 1, 0)` is the south-east corner. If you are familiar with\n * [vector tiles](https://github.com/mapbox/vector-tile-spec) it may be helpful to think\n * of the coordinate space as the `0/0/0` tile with an extent of `1`.\n *\n * The `z` dimension of `MercatorCoordinate` is conformal. A cube in the mercator coordinate space would be rendered as a cube.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let nullIsland = new maplibregl.MercatorCoordinate(0.5, 0.5, 0);\n * ```\n * @see [Add a custom style layer](https://maplibre.org/maplibre-gl-js/docs/examples/custom-style-layer/)\n */\nexport class MercatorCoordinate implements IMercatorCoordinate {\n x: number;\n y: number;\n z: number;\n\n /**\n * @param x - The x component of the position.\n * @param y - The y component of the position.\n * @param z - The z component of the position.\n */\n constructor(x: number, y: number, z: number = 0) {\n this.x = +x;\n this.y = +y;\n this.z = +z;\n }\n\n /**\n * Project a `LngLat` to a `MercatorCoordinate`.\n *\n * @param lngLatLike - The location to project.\n * @param altitude - The altitude in meters of the position.\n * @returns The projected mercator coordinate.\n * @example\n * ```ts\n * let coord = maplibregl.MercatorCoordinate.fromLngLat({ lng: 0, lat: 0}, 0);\n * coord; // MercatorCoordinate(0.5, 0.5, 0)\n * ```\n */\n static fromLngLat(lngLatLike: LngLatLike, altitude: number = 0): MercatorCoordinate {\n const lngLat = LngLat.convert(lngLatLike);\n\n return new MercatorCoordinate(\n mercatorXfromLng(lngLat.lng),\n mercatorYfromLat(lngLat.lat),\n mercatorZfromAltitude(altitude, lngLat.lat));\n }\n\n /**\n * Returns the `LngLat` for the coordinate.\n *\n * @returns The `LngLat` object.\n * @example\n * ```ts\n * let coord = new maplibregl.MercatorCoordinate(0.5, 0.5, 0);\n * let lngLat = coord.toLngLat(); // LngLat(0, 0)\n * ```\n */\n toLngLat() {\n return new LngLat(\n lngFromMercatorX(this.x),\n latFromMercatorY(this.y));\n }\n\n /**\n * Returns the altitude in meters of the coordinate.\n *\n * @returns The altitude in meters.\n * @example\n * ```ts\n * let coord = new maplibregl.MercatorCoordinate(0, 0, 0.02);\n * coord.toAltitude(); // 6914.281956295339\n * ```\n */\n toAltitude(): number {\n return altitudeFromMercatorZ(this.z, this.y);\n }\n\n /**\n * Returns the distance of 1 meter in `MercatorCoordinate` units at this latitude.\n *\n * For coordinates in real world units using meters, this naturally provides the scale\n * to transform into `MercatorCoordinate`s.\n *\n * @returns Distance of 1 meter in `MercatorCoordinate` units.\n */\n meterInMercatorCoordinateUnits(): number {\n // 1 meter / circumference at equator in meters * Mercator projection scale factor at this latitude\n return 1 / earthCircumfrence * mercatorScale(latFromMercatorY(this.y));\n }\n}\n","export { getURL, getTileBBox, getMercCoords };\n\n\n/**\n * getURL\n *\n * @param {String} baseUrl Base url of the WMS server\n * @param {String} layer Layer name\n * @param {Number} x Tile coordinate x\n * @param {Number} y Tile coordinate y\n * @param {Number} z Tile zoom\n * @param {Object} [options]\n * @param {String} [options.format='image/png']\n * @param {String} [options.service='WMS']\n * @param {String} [options.version='1.1.1']\n * @param {String} [options.request='GetMap']\n * @param {String} [options.srs='EPSG:3857']\n * @param {Number} [options.width='256']\n * @param {Number} [options.height='256']\n * @returns {String} url\n * @example\n * var baseUrl = 'http://geodata.state.nj.us/imagerywms/Natural2015';\n * var layer = 'Natural2015';\n * var url = whoots.getURL(baseUrl, layer, 154308, 197167, 19);\n */\nfunction getURL(baseUrl, layer, x, y, z, options) {\n options = options || {};\n\n var url = baseUrl + '?' + [\n 'bbox=' + getTileBBox(x, y, z),\n 'format=' + (options.format || 'image/png'),\n 'service=' + (options.service || 'WMS'),\n 'version=' + (options.version || '1.1.1'),\n 'request=' + (options.request || 'GetMap'),\n 'srs=' + (options.srs || 'EPSG:3857'),\n 'width=' + (options.width || 256),\n 'height=' + (options.height || 256),\n 'layers=' + layer\n ].join('&');\n\n return url;\n}\n\n\n/**\n * getTileBBox\n *\n * @param {Number} x Tile coordinate x\n * @param {Number} y Tile coordinate y\n * @param {Number} z Tile zoom\n * @returns {String} String of the bounding box\n */\nfunction getTileBBox(x, y, z) {\n // for Google/OSM tile scheme we need to alter the y\n y = (Math.pow(2, z) - y - 1);\n\n var min = getMercCoords(x * 256, y * 256, z),\n max = getMercCoords((x + 1) * 256, (y + 1) * 256, z);\n\n return min[0] + ',' + min[1] + ',' + max[0] + ',' + max[1];\n}\n\n\n/**\n * getMercCoords\n *\n * @param {Number} x Pixel coordinate x\n * @param {Number} y Pixel coordinate y\n * @param {Number} z Tile zoom\n * @returns {Array} [x, y]\n */\nfunction getMercCoords(x, y, z) {\n var resolution = (2 * Math.PI * 6378137 / 256) / Math.pow(2, z),\n merc_x = (x * resolution - 2 * Math.PI * 6378137 / 2.0),\n merc_y = (y * resolution - 2 * Math.PI * 6378137 / 2.0);\n\n return [merc_x, merc_y];\n}\n","import {getTileBBox} from '@mapbox/whoots-js';\nimport {EXTENT} from '../data/extent';\nimport Point from '@mapbox/point-geometry';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\nimport {register} from '../util/web_worker_transfer';\nimport {mat4} from 'gl-matrix';\nimport {ICanonicalTileID, IMercatorCoordinate} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A canonical way to define a tile ID\n */\nexport class CanonicalTileID implements ICanonicalTileID {\n z: number;\n x: number;\n y: number;\n key: string;\n\n constructor(z: number, x: number, y: number) {\n\n if (z < 0 || z > 25 || y < 0 || y >= Math.pow(2, z) || x < 0 || x >= Math.pow(2, z)) {\n throw new Error(`x=${x}, y=${y}, z=${z} outside of bounds. 0<=x<${Math.pow(2, z)}, 0<=y<${Math.pow(2, z)} 0<=z<=25 `);\n }\n\n this.z = z;\n this.x = x;\n this.y = y;\n this.key = calculateKey(0, z, z, x, y);\n }\n\n equals(id: ICanonicalTileID) {\n return this.z === id.z && this.x === id.x && this.y === id.y;\n }\n\n // given a list of urls, choose a url template and return a tile URL\n url(urls: Array, pixelRatio: number, scheme?: string | null) {\n const bbox = getTileBBox(this.x, this.y, this.z);\n const quadkey = getQuadkey(this.z, this.x, this.y);\n\n return urls[(this.x + this.y) % urls.length]\n .replace(/{prefix}/g, (this.x % 16).toString(16) + (this.y % 16).toString(16))\n .replace(/{z}/g, String(this.z))\n .replace(/{x}/g, String(this.x))\n .replace(/{y}/g, String(scheme === 'tms' ? (Math.pow(2, this.z) - this.y - 1) : this.y))\n .replace(/{ratio}/g, pixelRatio > 1 ? '@2x' : '')\n .replace(/{quadkey}/g, quadkey)\n .replace(/{bbox-epsg-3857}/g, bbox);\n }\n\n isChildOf(parent: ICanonicalTileID) {\n const dz = this.z - parent.z;\n return dz > 0 && parent.x === (this.x >> dz) && parent.y === (this.y >> dz);\n }\n\n getTilePoint(coord: IMercatorCoordinate) {\n const tilesAtZoom = Math.pow(2, this.z);\n return new Point(\n (coord.x * tilesAtZoom - this.x) * EXTENT,\n (coord.y * tilesAtZoom - this.y) * EXTENT);\n }\n\n toString() {\n return `${this.z}/${this.x}/${this.y}`;\n }\n}\n\n/**\n * @internal\n * An unwrapped tile identifier\n */\nexport class UnwrappedTileID {\n wrap: number;\n canonical: CanonicalTileID;\n key: string;\n\n constructor(wrap: number, canonical: CanonicalTileID) {\n this.wrap = wrap;\n this.canonical = canonical;\n this.key = calculateKey(wrap, canonical.z, canonical.z, canonical.x, canonical.y);\n }\n}\n\n/**\n * An overscaled tile identifier\n */\nexport class OverscaledTileID {\n overscaledZ: number;\n wrap: number;\n canonical: CanonicalTileID;\n key: string;\n posMatrix: mat4;\n\n constructor(overscaledZ: number, wrap: number, z: number, x: number, y: number) {\n if (overscaledZ < z) throw new Error(`overscaledZ should be >= z; overscaledZ = ${overscaledZ}; z = ${z}`);\n this.overscaledZ = overscaledZ;\n this.wrap = wrap;\n this.canonical = new CanonicalTileID(z, +x, +y);\n this.key = calculateKey(wrap, overscaledZ, z, x, y);\n }\n\n clone() {\n return new OverscaledTileID(this.overscaledZ, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y);\n }\n\n equals(id: OverscaledTileID) {\n return this.overscaledZ === id.overscaledZ && this.wrap === id.wrap && this.canonical.equals(id.canonical);\n }\n\n scaledTo(targetZ: number) {\n if (targetZ > this.overscaledZ) throw new Error(`targetZ > this.overscaledZ; targetZ = ${targetZ}; overscaledZ = ${this.overscaledZ}`);\n const zDifference = this.canonical.z - targetZ;\n if (targetZ > this.canonical.z) {\n return new OverscaledTileID(targetZ, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y);\n } else {\n return new OverscaledTileID(targetZ, this.wrap, targetZ, this.canonical.x >> zDifference, this.canonical.y >> zDifference);\n }\n }\n\n /*\n * calculateScaledKey is an optimization:\n * when withWrap == true, implements the same as this.scaledTo(z).key,\n * when withWrap == false, implements the same as this.scaledTo(z).wrapped().key.\n */\n calculateScaledKey(targetZ: number, withWrap: boolean): string {\n if (targetZ > this.overscaledZ) throw new Error(`targetZ > this.overscaledZ; targetZ = ${targetZ}; overscaledZ = ${this.overscaledZ}`);\n const zDifference = this.canonical.z - targetZ;\n if (targetZ > this.canonical.z) {\n return calculateKey(this.wrap * +withWrap, targetZ, this.canonical.z, this.canonical.x, this.canonical.y);\n } else {\n return calculateKey(this.wrap * +withWrap, targetZ, targetZ, this.canonical.x >> zDifference, this.canonical.y >> zDifference);\n }\n }\n\n isChildOf(parent: OverscaledTileID) {\n if (parent.wrap !== this.wrap) {\n // We can't be a child if we're in a different world copy\n return false;\n }\n const zDifference = this.canonical.z - parent.canonical.z;\n // We're first testing for z == 0, to avoid a 32 bit shift, which is undefined.\n return parent.overscaledZ === 0 || (\n parent.overscaledZ < this.overscaledZ &&\n parent.canonical.x === (this.canonical.x >> zDifference) &&\n parent.canonical.y === (this.canonical.y >> zDifference));\n }\n\n children(sourceMaxZoom: number) {\n if (this.overscaledZ >= sourceMaxZoom) {\n // return a single tile coord representing a an overscaled tile\n return [new OverscaledTileID(this.overscaledZ + 1, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y)];\n }\n\n const z = this.canonical.z + 1;\n const x = this.canonical.x * 2;\n const y = this.canonical.y * 2;\n return [\n new OverscaledTileID(z, this.wrap, z, x, y),\n new OverscaledTileID(z, this.wrap, z, x + 1, y),\n new OverscaledTileID(z, this.wrap, z, x, y + 1),\n new OverscaledTileID(z, this.wrap, z, x + 1, y + 1)\n ];\n }\n\n isLessThan(rhs: OverscaledTileID) {\n if (this.wrap < rhs.wrap) return true;\n if (this.wrap > rhs.wrap) return false;\n\n if (this.overscaledZ < rhs.overscaledZ) return true;\n if (this.overscaledZ > rhs.overscaledZ) return false;\n\n if (this.canonical.x < rhs.canonical.x) return true;\n if (this.canonical.x > rhs.canonical.x) return false;\n\n if (this.canonical.y < rhs.canonical.y) return true;\n return false;\n }\n\n wrapped() {\n return new OverscaledTileID(this.overscaledZ, 0, this.canonical.z, this.canonical.x, this.canonical.y);\n }\n\n unwrapTo(wrap: number) {\n return new OverscaledTileID(this.overscaledZ, wrap, this.canonical.z, this.canonical.x, this.canonical.y);\n }\n\n overscaleFactor() {\n return Math.pow(2, this.overscaledZ - this.canonical.z);\n }\n\n toUnwrapped() {\n return new UnwrappedTileID(this.wrap, this.canonical);\n }\n\n toString() {\n return `${this.overscaledZ}/${this.canonical.x}/${this.canonical.y}`;\n }\n\n getTilePoint(coord: MercatorCoordinate) {\n return this.canonical.getTilePoint(new MercatorCoordinate(coord.x - this.wrap, coord.y));\n }\n}\n\nfunction calculateKey(wrap: number, overscaledZ: number, z: number, x: number, y: number): string {\n wrap *= 2;\n if (wrap < 0) wrap = wrap * -1 - 1;\n const dim = 1 << z;\n return (dim * dim * wrap + dim * y + x).toString(36) + z.toString(36) + overscaledZ.toString(36);\n}\n\nfunction getQuadkey(z, x, y) {\n let quadkey = '', mask;\n for (let i = z; i > 0; i--) {\n mask = 1 << (i - 1);\n quadkey += ((x & mask ? 1 : 0) + (y & mask ? 2 : 0));\n }\n return quadkey;\n}\n\nregister('CanonicalTileID', CanonicalTileID);\nregister('OverscaledTileID', OverscaledTileID, {omit: ['posMatrix']});\n","import {RGBAImage} from '../util/image';\n\nimport {warnOnce} from '../util/util';\nimport {register} from '../util/web_worker_transfer';\n\n// DEMData is a data structure for decoding, backfilling, and storing elevation data for processing in the hillshade shaders\n// data can be populated either from a pngraw image tile or from serliazed data sent back from a worker. When data is initially\n// loaded from a image tile, we decode the pixel values using the appropriate decoding formula, but we store the\n// elevation data as an Int32 value. we add 65536 (2^16) to eliminate negative values and enable the use of\n// integer overflow when creating the texture used in the hillshadePrepare step.\n\n// DEMData also handles the backfilling of data from a tile's neighboring tiles. This is necessary because we use a pixel's 8\n// surrounding pixel values to compute the slope at that pixel, and we cannot accurately calculate the slope at pixels on a\n// tile's edge without backfilling from neighboring tiles.\n\nexport type DEMEncoding = 'mapbox' | 'terrarium' | 'custom'\n\nexport class DEMData {\n uid: string;\n data: Uint32Array;\n stride: number;\n dim: number;\n min: number;\n max: number;\n redFactor: number;\n greenFactor: number;\n blueFactor: number;\n baseShift: number;\n\n // RGBAImage data has uniform 1px padding on all sides: square tile edge size defines stride\n // and dim is calculated as stride - 2.\n constructor(uid: string, data: RGBAImage, encoding: DEMEncoding, redFactor = 1.0, greenFactor = 1.0, blueFactor = 1.0, baseShift = 0.0) {\n this.uid = uid;\n if (data.height !== data.width) throw new RangeError('DEM tiles must be square');\n if (encoding && !['mapbox', 'terrarium', 'custom'].includes(encoding)) {\n warnOnce(`\"${encoding}\" is not a valid encoding type. Valid types include \"mapbox\", \"terrarium\" and \"custom\".`);\n return;\n }\n this.stride = data.height;\n const dim = this.dim = data.height - 2;\n this.data = new Uint32Array(data.data.buffer);\n switch (encoding) {\n case 'terrarium':\n // unpacking formula for mapzen terrarium:\n // https://aws.amazon.com/public-datasets/terrain/\n this.redFactor = 256.0;\n this.greenFactor = 1.0;\n this.blueFactor = 1.0 / 256.0;\n this.baseShift = 32768.0;\n break;\n case 'custom':\n this.redFactor = redFactor;\n this.greenFactor = greenFactor;\n this.blueFactor = blueFactor;\n this.baseShift = baseShift;\n break;\n case 'mapbox':\n default:\n // unpacking formula for mapbox.terrain-rgb:\n // https://www.mapbox.com/help/access-elevation-data/#mapbox-terrain-rgb\n this.redFactor = 6553.6;\n this.greenFactor = 25.6;\n this.blueFactor = 0.1;\n this.baseShift = 10000.0;\n break;\n }\n\n // in order to avoid flashing seams between tiles, here we are initially populating a 1px border of pixels around the image\n // with the data of the nearest pixel from the image. this data is eventually replaced when the tile's neighboring\n // tiles are loaded and the accurate data can be backfilled using DEMData#backfillBorder\n for (let x = 0; x < dim; x++) {\n // left vertical border\n this.data[this._idx(-1, x)] = this.data[this._idx(0, x)];\n // right vertical border\n this.data[this._idx(dim, x)] = this.data[this._idx(dim - 1, x)];\n // left horizontal border\n this.data[this._idx(x, -1)] = this.data[this._idx(x, 0)];\n // right horizontal border\n this.data[this._idx(x, dim)] = this.data[this._idx(x, dim - 1)];\n }\n // corners\n this.data[this._idx(-1, -1)] = this.data[this._idx(0, 0)];\n this.data[this._idx(dim, -1)] = this.data[this._idx(dim - 1, 0)];\n this.data[this._idx(-1, dim)] = this.data[this._idx(0, dim - 1)];\n this.data[this._idx(dim, dim)] = this.data[this._idx(dim - 1, dim - 1)];\n\n // calculate min/max values\n this.min = Number.MAX_SAFE_INTEGER;\n this.max = Number.MIN_SAFE_INTEGER;\n for (let x = 0; x < dim; x++) {\n for (let y = 0; y < dim; y++) {\n const ele = this.get(x, y);\n if (ele > this.max) this.max = ele;\n if (ele < this.min) this.min = ele;\n }\n }\n }\n\n get(x: number, y: number) {\n const pixels = new Uint8Array(this.data.buffer);\n const index = this._idx(x, y) * 4;\n return this.unpack(pixels[index], pixels[index + 1], pixels[index + 2]);\n }\n\n getUnpackVector() {\n return [this.redFactor, this.greenFactor, this.blueFactor, this.baseShift];\n }\n\n _idx(x: number, y: number) {\n if (x < -1 || x >= this.dim + 1 || y < -1 || y >= this.dim + 1) throw new RangeError('out of range source coordinates for DEM data');\n return (y + 1) * this.stride + (x + 1);\n }\n\n unpack(r: number, g: number, b: number) {\n return (r * this.redFactor + g * this.greenFactor + b * this.blueFactor - this.baseShift);\n }\n\n getPixels() {\n return new RGBAImage({width: this.stride, height: this.stride}, new Uint8Array(this.data.buffer));\n }\n\n backfillBorder(borderTile: DEMData, dx: number, dy: number) {\n if (this.dim !== borderTile.dim) throw new Error('dem dimension mismatch');\n\n let xMin = dx * this.dim,\n xMax = dx * this.dim + this.dim,\n yMin = dy * this.dim,\n yMax = dy * this.dim + this.dim;\n\n switch (dx) {\n case -1:\n xMin = xMax - 1;\n break;\n case 1:\n xMax = xMin + 1;\n break;\n }\n\n switch (dy) {\n case -1:\n yMin = yMax - 1;\n break;\n case 1:\n yMax = yMin + 1;\n break;\n }\n\n const ox = -dx * this.dim;\n const oy = -dy * this.dim;\n for (let y = yMin; y < yMax; y++) {\n for (let x = xMin; x < xMax; x++) {\n this.data[this._idx(x, y)] = borderTile.data[this._idx(x + ox, y + oy)];\n }\n }\n }\n}\n\nregister('DEMData', DEMData);\n","export class DictionaryCoder {\n _stringToNumber: {[_: string]: number};\n _numberToString: Array;\n\n constructor(strings: Array) {\n this._stringToNumber = {};\n this._numberToString = [];\n for (let i = 0; i < strings.length; i++) {\n const string = strings[i];\n this._stringToNumber[string] = i;\n this._numberToString[i] = string;\n }\n }\n\n encode(string: string) {\n return this._stringToNumber[string];\n }\n\n decode(n: number) {\n if (n >= this._numberToString.length) throw new Error(`Out of bounds. Index requested n=${n} can't be >= this._numberToString.length ${this._numberToString.length}`);\n return this._numberToString[n];\n }\n}\n","import type {VectorTileFeature} from '@mapbox/vector-tile';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A helper for type to omit a property from a type\n */\ntype DistributiveKeys = T extends T ? keyof T : never;\n/**\n * A helper for type to omit a property from a type\n */\ntype DistributiveOmit> = T extends unknown\n ? Omit\n : never;\n\n/**\n * An extended geojson feature used by the events to return data to the listener\n */\nexport type MapGeoJSONFeature = GeoJSONFeature & {\n layer: DistributiveOmit & {source: string};\n source: string;\n sourceLayer?: string;\n state: { [key: string]: any };\n}\n\n/**\n * A geojson feature\n */\nexport class GeoJSONFeature {\n type: 'Feature';\n _geometry: GeoJSON.Geometry;\n properties: { [name: string]: any };\n id: number | string | undefined;\n\n _vectorTileFeature: VectorTileFeature;\n\n constructor(vectorTileFeature: VectorTileFeature, z: number, x: number, y: number, id: string | number | undefined) {\n this.type = 'Feature';\n\n this._vectorTileFeature = vectorTileFeature;\n (vectorTileFeature as any)._z = z;\n (vectorTileFeature as any)._x = x;\n (vectorTileFeature as any)._y = y;\n\n this.properties = vectorTileFeature.properties;\n this.id = id;\n }\n\n get geometry(): GeoJSON.Geometry {\n if (this._geometry === undefined) {\n this._geometry = this._vectorTileFeature.toGeoJSON(\n (this._vectorTileFeature as any)._x,\n (this._vectorTileFeature as any)._y,\n (this._vectorTileFeature as any)._z).geometry;\n }\n return this._geometry;\n }\n\n set geometry(g: GeoJSON.Geometry) {\n this._geometry = g;\n }\n\n toJSON() {\n const json: any = {\n geometry: this.geometry\n };\n for (const i in this) {\n if (i === '_geometry' || i === '_vectorTileFeature') continue;\n json[i] = (this)[i];\n }\n return json;\n }\n}\n","import Point from '@mapbox/point-geometry';\nimport {loadGeometry} from './load_geometry';\nimport {toEvaluationFeature} from './evaluation_feature';\nimport {EXTENT} from './extent';\nimport {featureFilter} from '@maplibre/maplibre-gl-style-spec';\nimport {TransferableGridIndex} from '../util/transferable_grid_index';\nimport {DictionaryCoder} from '../util/dictionary_coder';\nimport vt from '@mapbox/vector-tile';\nimport Protobuf from 'pbf';\nimport {GeoJSONFeature} from '../util/vectortile_to_geojson';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\nimport {arraysIntersect, mapObject, extend} from '../util/util';\nimport {OverscaledTileID} from '../source/tile_id';\nimport {register} from '../util/web_worker_transfer';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {SourceFeatureState} from '../source/source_state';\nimport {polygonIntersectsBox} from '../util/intersection_tests';\nimport {PossiblyEvaluated} from '../style/properties';\nimport {FeatureIndexArray} from './array_types.g';\nimport {mat4} from 'gl-matrix';\n\nimport type {StyleLayer} from '../style/style_layer';\nimport type {FeatureFilter, FeatureState, FilterSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Transform} from '../geo/transform';\nimport type {VectorTileFeature, VectorTileLayer} from '@mapbox/vector-tile';\n\ntype QueryParameters = {\n scale: number;\n pixelPosMatrix: mat4;\n transform: Transform;\n tileSize: number;\n queryGeometry: Array;\n cameraQueryGeometry: Array;\n queryPadding: number;\n params: {\n filter: FilterSpecification;\n layers: Array;\n availableImages: Array;\n };\n};\n\n/**\n * @internal\n * An in memory index class to allow fast interaction with features\n */\nexport class FeatureIndex {\n tileID: OverscaledTileID;\n x: number;\n y: number;\n z: number;\n grid: TransferableGridIndex;\n grid3D: TransferableGridIndex;\n featureIndexArray: FeatureIndexArray;\n promoteId?: PromoteIdSpecification;\n\n rawTileData: ArrayBuffer;\n bucketLayerIDs: Array>;\n\n vtLayers: {[_: string]: VectorTileLayer};\n sourceLayerCoder: DictionaryCoder;\n\n constructor(tileID: OverscaledTileID, promoteId?: PromoteIdSpecification | null) {\n this.tileID = tileID;\n this.x = tileID.canonical.x;\n this.y = tileID.canonical.y;\n this.z = tileID.canonical.z;\n this.grid = new TransferableGridIndex(EXTENT, 16, 0);\n this.grid3D = new TransferableGridIndex(EXTENT, 16, 0);\n this.featureIndexArray = new FeatureIndexArray();\n this.promoteId = promoteId;\n }\n\n insert(feature: VectorTileFeature, geometry: Array>, featureIndex: number, sourceLayerIndex: number, bucketIndex: number, is3D?: boolean) {\n const key = this.featureIndexArray.length;\n this.featureIndexArray.emplaceBack(featureIndex, sourceLayerIndex, bucketIndex);\n\n const grid = is3D ? this.grid3D : this.grid;\n\n for (let r = 0; r < geometry.length; r++) {\n const ring = geometry[r];\n\n const bbox = [Infinity, Infinity, -Infinity, -Infinity];\n for (let i = 0; i < ring.length; i++) {\n const p = ring[i];\n bbox[0] = Math.min(bbox[0], p.x);\n bbox[1] = Math.min(bbox[1], p.y);\n bbox[2] = Math.max(bbox[2], p.x);\n bbox[3] = Math.max(bbox[3], p.y);\n }\n\n if (bbox[0] < EXTENT &&\n bbox[1] < EXTENT &&\n bbox[2] >= 0 &&\n bbox[3] >= 0) {\n grid.insert(key, bbox[0], bbox[1], bbox[2], bbox[3]);\n }\n }\n }\n\n loadVTLayers(): {[_: string]: VectorTileLayer} {\n if (!this.vtLayers) {\n this.vtLayers = new vt.VectorTile(new Protobuf(this.rawTileData)).layers;\n this.sourceLayerCoder = new DictionaryCoder(this.vtLayers ? Object.keys(this.vtLayers).sort() : ['_geojsonTileLayer']);\n }\n return this.vtLayers;\n }\n\n // Finds non-symbol features in this tile at a particular position.\n query(\n args: QueryParameters,\n styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n sourceFeatureState: SourceFeatureState\n ): {[_: string]: Array<{featureIndex: number; feature: GeoJSONFeature}>} {\n this.loadVTLayers();\n\n const params = args.params || {} as { filter: any; layers: string[]; availableImages: string[] },\n pixelsToTileUnits = EXTENT / args.tileSize / args.scale,\n filter = featureFilter(params.filter);\n\n const queryGeometry = args.queryGeometry;\n const queryPadding = args.queryPadding * pixelsToTileUnits;\n\n const bounds = getBounds(queryGeometry);\n const matching = this.grid.query(bounds.minX - queryPadding, bounds.minY - queryPadding, bounds.maxX + queryPadding, bounds.maxY + queryPadding);\n\n const cameraBounds = getBounds(args.cameraQueryGeometry);\n const matching3D = this.grid3D.query(\n cameraBounds.minX - queryPadding, cameraBounds.minY - queryPadding, cameraBounds.maxX + queryPadding, cameraBounds.maxY + queryPadding,\n (bx1, by1, bx2, by2) => {\n return polygonIntersectsBox(args.cameraQueryGeometry, bx1 - queryPadding, by1 - queryPadding, bx2 + queryPadding, by2 + queryPadding);\n });\n\n for (const key of matching3D) {\n matching.push(key);\n }\n\n matching.sort(topDownFeatureComparator);\n\n const result = {};\n let previousIndex;\n for (let k = 0; k < matching.length; k++) {\n const index = matching[k];\n\n // don't check the same feature more than once\n if (index === previousIndex) continue;\n previousIndex = index;\n\n const match = this.featureIndexArray.get(index);\n let featureGeometry = null;\n this.loadMatchingFeature(\n result,\n match.bucketIndex,\n match.sourceLayerIndex,\n match.featureIndex,\n filter,\n params.layers,\n params.availableImages,\n styleLayers,\n serializedLayers,\n sourceFeatureState,\n (feature: VectorTileFeature, styleLayer: StyleLayer, featureState: FeatureState) => {\n if (!featureGeometry) {\n featureGeometry = loadGeometry(feature);\n }\n\n return styleLayer.queryIntersectsFeature(queryGeometry, feature, featureState, featureGeometry, this.z, args.transform, pixelsToTileUnits, args.pixelPosMatrix);\n }\n );\n }\n\n return result;\n }\n\n loadMatchingFeature(\n result: {\n [_: string]: Array<{\n featureIndex: number;\n feature: GeoJSONFeature;\n intersectionZ?: boolean | number;\n }>;\n },\n bucketIndex: number,\n sourceLayerIndex: number,\n featureIndex: number,\n filter: FeatureFilter,\n filterLayerIDs: Array,\n availableImages: Array,\n styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n sourceFeatureState?: SourceFeatureState,\n intersectionTest?: (\n feature: VectorTileFeature,\n styleLayer: StyleLayer,\n featureState: any,\n id: string | number | void\n ) => boolean | number) {\n\n const layerIDs = this.bucketLayerIDs[bucketIndex];\n if (filterLayerIDs && !arraysIntersect(filterLayerIDs, layerIDs))\n return;\n\n const sourceLayerName = this.sourceLayerCoder.decode(sourceLayerIndex);\n const sourceLayer = this.vtLayers[sourceLayerName];\n const feature = sourceLayer.feature(featureIndex);\n\n if (filter.needGeometry) {\n const evaluationFeature = toEvaluationFeature(feature, true);\n if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), evaluationFeature, this.tileID.canonical)) {\n return;\n }\n } else if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), feature)) {\n return;\n }\n\n const id = this.getId(feature, sourceLayerName);\n\n for (let l = 0; l < layerIDs.length; l++) {\n const layerID = layerIDs[l];\n\n if (filterLayerIDs && filterLayerIDs.indexOf(layerID) < 0) {\n continue;\n }\n\n const styleLayer = styleLayers[layerID];\n\n if (!styleLayer) continue;\n\n let featureState = {};\n if (id && sourceFeatureState) {\n // `feature-state` expression evaluation requires feature state to be available\n featureState = sourceFeatureState.getState(styleLayer.sourceLayer || '_geojsonTileLayer', id);\n }\n\n const serializedLayer = extend({}, serializedLayers[layerID]);\n\n serializedLayer.paint = evaluateProperties(serializedLayer.paint, styleLayer.paint, feature, featureState, availableImages);\n serializedLayer.layout = evaluateProperties(serializedLayer.layout, styleLayer.layout, feature, featureState, availableImages);\n\n const intersectionZ = !intersectionTest || intersectionTest(feature, styleLayer, featureState);\n if (!intersectionZ) {\n // Only applied for non-symbol features\n continue;\n }\n\n const geojsonFeature = new GeoJSONFeature(feature, this.z, this.x, this.y, id) as MapGeoJSONFeature;\n geojsonFeature.layer = serializedLayer;\n let layerResult = result[layerID];\n if (layerResult === undefined) {\n layerResult = result[layerID] = [];\n }\n layerResult.push({featureIndex, feature: geojsonFeature, intersectionZ});\n }\n }\n\n // Given a set of symbol indexes that have already been looked up,\n // return a matching set of GeoJSONFeatures\n lookupSymbolFeatures(symbolFeatureIndexes: Array,\n serializedLayers: {[_: string]: StyleLayer},\n bucketIndex: number,\n sourceLayerIndex: number,\n filterSpec: FilterSpecification,\n filterLayerIDs: Array,\n availableImages: Array,\n styleLayers: {[_: string]: StyleLayer}) {\n const result = {};\n this.loadVTLayers();\n\n const filter = featureFilter(filterSpec);\n\n for (const symbolFeatureIndex of symbolFeatureIndexes) {\n this.loadMatchingFeature(\n result,\n bucketIndex,\n sourceLayerIndex,\n symbolFeatureIndex,\n filter,\n filterLayerIDs,\n availableImages,\n styleLayers,\n serializedLayers\n );\n\n }\n return result;\n }\n\n hasLayer(id: string) {\n for (const layerIDs of this.bucketLayerIDs) {\n for (const layerID of layerIDs) {\n if (id === layerID) return true;\n }\n }\n\n return false;\n }\n\n getId(feature: VectorTileFeature, sourceLayerId: string): string | number {\n let id: string | number = feature.id;\n if (this.promoteId) {\n const propName = typeof this.promoteId === 'string' ? this.promoteId : this.promoteId[sourceLayerId];\n id = feature.properties[propName] as string | number;\n if (typeof id === 'boolean') id = Number(id);\n }\n return id;\n }\n}\n\nregister(\n 'FeatureIndex',\n FeatureIndex,\n {omit: ['rawTileData', 'sourceLayerCoder']}\n);\n\nfunction evaluateProperties(serializedProperties, styleLayerProperties, feature, featureState, availableImages) {\n return mapObject(serializedProperties, (property, key) => {\n const prop = styleLayerProperties instanceof PossiblyEvaluated ? styleLayerProperties.get(key) : null;\n return prop && prop.evaluate ? prop.evaluate(feature, featureState, availableImages) : prop;\n });\n}\n\nfunction getBounds(geometry: Array) {\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n for (const p of geometry) {\n minX = Math.min(minX, p.x);\n minY = Math.min(minY, p.y);\n maxX = Math.max(maxX, p.x);\n maxY = Math.max(maxY, p.y);\n }\n return {minX, minY, maxX, maxY};\n}\n\nfunction topDownFeatureComparator(a, b) {\n return b - a;\n}\n","import Point from '@mapbox/point-geometry';\n\n/**\n * Returns the part of a multiline that intersects with the provided rectangular box.\n *\n * @param lines - the lines to check\n * @param x1 - the left edge of the box\n * @param y1 - the top edge of the box\n * @param x2 - the right edge of the box\n * @param y2 - the bottom edge of the box\n * @returns lines\n */\nexport function clipLine(lines: Array>, x1: number, y1: number, x2: number, y2: number): Array> {\n const clippedLines = [];\n\n for (let l = 0; l < lines.length; l++) {\n const line = lines[l];\n let clippedLine;\n\n for (let i = 0; i < line.length - 1; i++) {\n let p0 = line[i];\n let p1 = line[i + 1];\n\n if (p0.x < x1 && p1.x < x1) {\n continue;\n } else if (p0.x < x1) {\n p0 = new Point(x1, p0.y + (p1.y - p0.y) * ((x1 - p0.x) / (p1.x - p0.x)))._round();\n } else if (p1.x < x1) {\n p1 = new Point(x1, p0.y + (p1.y - p0.y) * ((x1 - p0.x) / (p1.x - p0.x)))._round();\n }\n\n if (p0.y < y1 && p1.y < y1) {\n continue;\n } else if (p0.y < y1) {\n p0 = new Point(p0.x + (p1.x - p0.x) * ((y1 - p0.y) / (p1.y - p0.y)), y1)._round();\n } else if (p1.y < y1) {\n p1 = new Point(p0.x + (p1.x - p0.x) * ((y1 - p0.y) / (p1.y - p0.y)), y1)._round();\n }\n\n if (p0.x >= x2 && p1.x >= x2) {\n continue;\n } else if (p0.x >= x2) {\n p0 = new Point(x2, p0.y + (p1.y - p0.y) * ((x2 - p0.x) / (p1.x - p0.x)))._round();\n } else if (p1.x >= x2) {\n p1 = new Point(x2, p0.y + (p1.y - p0.y) * ((x2 - p0.x) / (p1.x - p0.x)))._round();\n }\n\n if (p0.y >= y2 && p1.y >= y2) {\n continue;\n } else if (p0.y >= y2) {\n p0 = new Point(p0.x + (p1.x - p0.x) * ((y2 - p0.y) / (p1.y - p0.y)), y2)._round();\n } else if (p1.y >= y2) {\n p1 = new Point(p0.x + (p1.x - p0.x) * ((y2 - p0.y) / (p1.y - p0.y)), y2)._round();\n }\n\n if (!clippedLine || !p0.equals(clippedLine[clippedLine.length - 1])) {\n clippedLine = [p0];\n clippedLines.push(clippedLine);\n }\n\n clippedLine.push(p1);\n }\n }\n\n return clippedLines;\n}\n","import Point from '@mapbox/point-geometry';\n\nimport {register} from '../util/web_worker_transfer';\n\nexport class Anchor extends Point {\n angle: any;\n segment?: number;\n\n constructor(x: number, y: number, angle: number, segment?: number) {\n super(x, y);\n this.angle = angle;\n if (segment !== undefined) {\n this.segment = segment;\n }\n }\n\n clone() {\n return new Anchor(this.x, this.y, this.angle, this.segment);\n }\n}\n\nregister('Anchor', Anchor);\n","import type Point from '@mapbox/point-geometry';\nimport type {Anchor} from './anchor';\n\n/**\n * Labels placed around really sharp angles aren't readable. Check if any\n * part of the potential label has a combined angle that is too big.\n *\n * @param line - The line to check\n * @param anchor - The point on the line around which the label is anchored.\n * @param labelLength - The length of the label in geometry units.\n * @param windowSize - The check fails if the combined angles within a part of the line that is `windowSize` long is too big.\n * @param maxAngle - The maximum combined angle that any window along the label is allowed to have.\n *\n * @returns whether the label should be placed\n */\nexport function checkMaxAngle(line: Array, anchor: Anchor, labelLength: number, windowSize: number, maxAngle: number) {\n\n // horizontal labels and labels with length 0 always pass\n if (anchor.segment === undefined || labelLength === 0) return true;\n\n let p = anchor;\n let index = anchor.segment + 1;\n let anchorDistance = 0;\n\n // move backwards along the line to the first segment the label appears on\n while (anchorDistance > -labelLength / 2) {\n index--;\n\n // there isn't enough room for the label after the beginning of the line\n if (index < 0) return false;\n\n anchorDistance -= line[index].dist(p);\n p = line[index];\n }\n\n anchorDistance += line[index].dist(line[index + 1]);\n index++;\n\n // store recent corners and their total angle difference\n const recentCorners = [];\n let recentAngleDelta = 0;\n\n // move forwards by the length of the label and check angles along the way\n while (anchorDistance < labelLength / 2) {\n const prev = line[index - 1];\n const current = line[index];\n const next = line[index + 1];\n\n // there isn't enough room for the label before the end of the line\n if (!next) return false;\n\n let angleDelta = prev.angleTo(current) - current.angleTo(next);\n // restrict angle to -pi..pi range\n angleDelta = Math.abs(((angleDelta + 3 * Math.PI) % (Math.PI * 2)) - Math.PI);\n\n recentCorners.push({\n distance: anchorDistance,\n angleDelta\n });\n recentAngleDelta += angleDelta;\n\n // remove corners that are far enough away from the list of recent anchors\n while (anchorDistance - recentCorners[0].distance > windowSize) {\n recentAngleDelta -= recentCorners.shift().angleDelta;\n }\n\n // the sum of angles within the window area exceeds the maximum allowed value. check fails.\n if (recentAngleDelta > maxAngle) return false;\n\n index++;\n anchorDistance += current.dist(next);\n }\n\n // no part of the line had an angle greater than the maximum allowed. check passes.\n return true;\n}\n","import {interpolates} from '@maplibre/maplibre-gl-style-spec';\n\nimport {Anchor} from '../symbol/anchor';\nimport {checkMaxAngle} from './check_max_angle';\n\nimport type Point from '@mapbox/point-geometry';\nimport type {Shaping, PositionedIcon} from './shaping';\n\nexport {getAnchors, getCenterAnchor};\n\nfunction getLineLength(line: Array): number {\n let lineLength = 0;\n for (let k = 0; k < line.length - 1; k++) {\n lineLength += line[k].dist(line[k + 1]);\n }\n return lineLength;\n}\n\nfunction getAngleWindowSize(\n shapedText: Shaping,\n glyphSize: number,\n boxScale: number\n): number {\n return shapedText ?\n 3 / 5 * glyphSize * boxScale :\n 0;\n}\n\nfunction getShapedLabelLength(shapedText?: Shaping | null, shapedIcon?: PositionedIcon | null): number {\n return Math.max(\n shapedText ? shapedText.right - shapedText.left : 0,\n shapedIcon ? shapedIcon.right - shapedIcon.left : 0);\n}\n\nfunction getCenterAnchor(line: Array,\n maxAngle: number,\n shapedText: Shaping,\n shapedIcon: PositionedIcon,\n glyphSize: number,\n boxScale: number) {\n const angleWindowSize = getAngleWindowSize(shapedText, glyphSize, boxScale);\n const labelLength = getShapedLabelLength(shapedText, shapedIcon) * boxScale;\n\n let prevDistance = 0;\n const centerDistance = getLineLength(line) / 2;\n\n for (let i = 0; i < line.length - 1; i++) {\n\n const a = line[i],\n b = line[i + 1];\n\n const segmentDistance = a.dist(b);\n\n if (prevDistance + segmentDistance > centerDistance) {\n // The center is on this segment\n const t = (centerDistance - prevDistance) / segmentDistance,\n x = interpolates.number(a.x, b.x, t),\n y = interpolates.number(a.y, b.y, t);\n\n const anchor = new Anchor(x, y, b.angleTo(a), i);\n anchor._round();\n if (!angleWindowSize || checkMaxAngle(line, anchor, labelLength, angleWindowSize, maxAngle)) {\n return anchor;\n } else {\n return;\n }\n }\n\n prevDistance += segmentDistance;\n }\n}\n\nfunction getAnchors(line: Array,\n spacing: number,\n maxAngle: number,\n shapedText: Shaping,\n shapedIcon: PositionedIcon,\n glyphSize: number,\n boxScale: number,\n overscaling: number,\n tileExtent: number) {\n\n // Resample a line to get anchor points for labels and check that each\n // potential label passes text-max-angle check and has enough froom to fit\n // on the line.\n\n const angleWindowSize = getAngleWindowSize(shapedText, glyphSize, boxScale);\n const shapedLabelLength = getShapedLabelLength(shapedText, shapedIcon);\n const labelLength = shapedLabelLength * boxScale;\n\n // Is the line continued from outside the tile boundary?\n const isLineContinued = line[0].x === 0 || line[0].x === tileExtent || line[0].y === 0 || line[0].y === tileExtent;\n\n // Is the label long, relative to the spacing?\n // If so, adjust the spacing so there is always a minimum space of `spacing / 4` between label edges.\n if (spacing - labelLength < spacing / 4) {\n spacing = labelLength + spacing / 4;\n }\n\n // Offset the first anchor by:\n // Either half the label length plus a fixed extra offset if the line is not continued\n // Or half the spacing if the line is continued.\n\n // For non-continued lines, add a bit of fixed extra offset to avoid collisions at T intersections.\n const fixedExtraOffset = glyphSize * 2;\n\n const offset = !isLineContinued ?\n ((shapedLabelLength / 2 + fixedExtraOffset) * boxScale * overscaling) % spacing :\n (spacing / 2 * overscaling) % spacing;\n\n return resample(line, offset, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, false, tileExtent);\n}\n\nfunction resample(line, offset, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, placeAtMiddle, tileExtent) {\n\n const halfLabelLength = labelLength / 2;\n const lineLength = getLineLength(line);\n\n let distance = 0,\n markedDistance = offset - spacing;\n\n let anchors = [];\n\n for (let i = 0; i < line.length - 1; i++) {\n\n const a = line[i],\n b = line[i + 1];\n\n const segmentDist = a.dist(b),\n angle = b.angleTo(a);\n\n while (markedDistance + spacing < distance + segmentDist) {\n markedDistance += spacing;\n\n const t = (markedDistance - distance) / segmentDist,\n x = interpolates.number(a.x, b.x, t),\n y = interpolates.number(a.y, b.y, t);\n\n // Check that the point is within the tile boundaries and that\n // the label would fit before the beginning and end of the line\n // if placed at this point.\n if (x >= 0 && x < tileExtent && y >= 0 && y < tileExtent &&\n markedDistance - halfLabelLength >= 0 &&\n markedDistance + halfLabelLength <= lineLength) {\n const anchor = new Anchor(x, y, angle, i);\n anchor._round();\n\n if (!angleWindowSize || checkMaxAngle(line, anchor, labelLength, angleWindowSize, maxAngle)) {\n anchors.push(anchor);\n }\n }\n }\n\n distance += segmentDist;\n }\n\n if (!placeAtMiddle && !anchors.length && !isLineContinued) {\n // The first attempt at finding anchors at which labels can be placed failed.\n // Try again, but this time just try placing one anchor at the middle of the line.\n // This has the most effect for short lines in overscaled tiles, since the\n // initial offset used in overscaled tiles is calculated to align labels with positions in\n // parent tiles instead of placing the label as close to the beginning as possible.\n anchors = resample(line, distance / 2, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, true, tileExtent);\n }\n\n return anchors;\n}\n","import Point from '@mapbox/point-geometry';\n\nimport {GLYPH_PBF_BORDER} from '../style/parse_glyph_pbf';\n\nimport type {Anchor} from './anchor';\nimport type {PositionedIcon, Shaping} from './shaping';\nimport {SHAPING_DEFAULT_OFFSET} from './shaping';\nimport {IMAGE_PADDING} from '../render/image_atlas';\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\nimport type {Feature} from '@maplibre/maplibre-gl-style-spec';\nimport type {StyleImage} from '../style/style_image';\nimport ONE_EM from './one_em';\nimport {Rect} from '../render/glyph_atlas';\n\n/**\n * A textured quad for rendering a single icon or glyph.\n *\n * The zoom range the glyph can be shown is defined by minScale and maxScale.\n *\n * @param tl - The offset of the top left corner from the anchor.\n * @param tr - The offset of the top right corner from the anchor.\n * @param bl - The offset of the bottom left corner from the anchor.\n * @param br - The offset of the bottom right corner from the anchor.\n * @param tex - The texture coordinates.\n */\nexport type SymbolQuad = {\n tl: Point;\n tr: Point;\n bl: Point;\n br: Point;\n tex: {\n x: number;\n y: number;\n w: number;\n h: number;\n };\n pixelOffsetTL: Point;\n pixelOffsetBR: Point;\n writingMode: any | void;\n glyphOffset: [number, number];\n sectionIndex: number;\n isSDF: boolean;\n minFontScaleX: number;\n minFontScaleY: number;\n};\n\n// If you have a 10px icon that isn't perfectly aligned to the pixel grid it will cover 11 actual\n// pixels. The quad needs to be padded to account for this, otherwise they'll look slightly clipped\n// on one edge in some cases.\nconst border = IMAGE_PADDING;\n\n/**\n * Create the quads used for rendering an icon.\n */\nexport function getIconQuads(\n shapedIcon: PositionedIcon,\n iconRotate: number,\n isSDFIcon: boolean,\n hasIconTextFit: boolean\n): Array {\n const quads = [];\n\n const image = shapedIcon.image;\n const pixelRatio = image.pixelRatio;\n const imageWidth = image.paddedRect.w - 2 * border;\n const imageHeight = image.paddedRect.h - 2 * border;\n\n const iconWidth = shapedIcon.right - shapedIcon.left;\n const iconHeight = shapedIcon.bottom - shapedIcon.top;\n\n const stretchX = image.stretchX || [[0, imageWidth]];\n const stretchY = image.stretchY || [[0, imageHeight]];\n\n const reduceRanges = (sum, range) => sum + range[1] - range[0];\n const stretchWidth = stretchX.reduce(reduceRanges, 0);\n const stretchHeight = stretchY.reduce(reduceRanges, 0);\n const fixedWidth = imageWidth - stretchWidth;\n const fixedHeight = imageHeight - stretchHeight;\n\n let stretchOffsetX = 0;\n let stretchContentWidth = stretchWidth;\n let stretchOffsetY = 0;\n let stretchContentHeight = stretchHeight;\n let fixedOffsetX = 0;\n let fixedContentWidth = fixedWidth;\n let fixedOffsetY = 0;\n let fixedContentHeight = fixedHeight;\n\n if (image.content && hasIconTextFit) {\n const content = image.content;\n stretchOffsetX = sumWithinRange(stretchX, 0, content[0]);\n stretchOffsetY = sumWithinRange(stretchY, 0, content[1]);\n stretchContentWidth = sumWithinRange(stretchX, content[0], content[2]);\n stretchContentHeight = sumWithinRange(stretchY, content[1], content[3]);\n fixedOffsetX = content[0] - stretchOffsetX;\n fixedOffsetY = content[1] - stretchOffsetY;\n fixedContentWidth = content[2] - content[0] - stretchContentWidth;\n fixedContentHeight = content[3] - content[1] - stretchContentHeight;\n }\n\n const makeBox = (left, top, right, bottom) => {\n\n const leftEm = getEmOffset(left.stretch - stretchOffsetX, stretchContentWidth, iconWidth, shapedIcon.left);\n const leftPx = getPxOffset(left.fixed - fixedOffsetX, fixedContentWidth, left.stretch, stretchWidth);\n\n const topEm = getEmOffset(top.stretch - stretchOffsetY, stretchContentHeight, iconHeight, shapedIcon.top);\n const topPx = getPxOffset(top.fixed - fixedOffsetY, fixedContentHeight, top.stretch, stretchHeight);\n\n const rightEm = getEmOffset(right.stretch - stretchOffsetX, stretchContentWidth, iconWidth, shapedIcon.left);\n const rightPx = getPxOffset(right.fixed - fixedOffsetX, fixedContentWidth, right.stretch, stretchWidth);\n\n const bottomEm = getEmOffset(bottom.stretch - stretchOffsetY, stretchContentHeight, iconHeight, shapedIcon.top);\n const bottomPx = getPxOffset(bottom.fixed - fixedOffsetY, fixedContentHeight, bottom.stretch, stretchHeight);\n\n const tl = new Point(leftEm, topEm);\n const tr = new Point(rightEm, topEm);\n const br = new Point(rightEm, bottomEm);\n const bl = new Point(leftEm, bottomEm);\n const pixelOffsetTL = new Point(leftPx / pixelRatio, topPx / pixelRatio);\n const pixelOffsetBR = new Point(rightPx / pixelRatio, bottomPx / pixelRatio);\n\n const angle = iconRotate * Math.PI / 180;\n\n if (angle) {\n const sin = Math.sin(angle),\n cos = Math.cos(angle),\n matrix = [cos, -sin, sin, cos];\n\n tl._matMult(matrix);\n tr._matMult(matrix);\n bl._matMult(matrix);\n br._matMult(matrix);\n }\n\n const x1 = left.stretch + left.fixed;\n const x2 = right.stretch + right.fixed;\n const y1 = top.stretch + top.fixed;\n const y2 = bottom.stretch + bottom.fixed;\n\n const subRect = {\n x: image.paddedRect.x + border + x1,\n y: image.paddedRect.y + border + y1,\n w: x2 - x1,\n h: y2 - y1\n };\n\n const minFontScaleX = fixedContentWidth / pixelRatio / iconWidth;\n const minFontScaleY = fixedContentHeight / pixelRatio / iconHeight;\n\n // Icon quad is padded, so texture coordinates also need to be padded.\n return {tl, tr, bl, br, tex: subRect, writingMode: undefined, glyphOffset: [0, 0], sectionIndex: 0, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY, isSDF: isSDFIcon};\n };\n\n if (!hasIconTextFit || (!image.stretchX && !image.stretchY)) {\n quads.push(makeBox(\n {fixed: 0, stretch: -1},\n {fixed: 0, stretch: -1},\n {fixed: 0, stretch: imageWidth + 1},\n {fixed: 0, stretch: imageHeight + 1}));\n } else {\n const xCuts = stretchZonesToCuts(stretchX, fixedWidth, stretchWidth);\n const yCuts = stretchZonesToCuts(stretchY, fixedHeight, stretchHeight);\n\n for (let xi = 0; xi < xCuts.length - 1; xi++) {\n const x1 = xCuts[xi];\n const x2 = xCuts[xi + 1];\n for (let yi = 0; yi < yCuts.length - 1; yi++) {\n const y1 = yCuts[yi];\n const y2 = yCuts[yi + 1];\n quads.push(makeBox(x1, y1, x2, y2));\n }\n }\n }\n\n return quads;\n}\n\nfunction sumWithinRange(ranges, min, max) {\n let sum = 0;\n for (const range of ranges) {\n sum += Math.max(min, Math.min(max, range[1])) - Math.max(min, Math.min(max, range[0]));\n }\n return sum;\n}\n\nfunction stretchZonesToCuts(stretchZones, fixedSize, stretchSize) {\n const cuts = [{fixed: -border, stretch: 0}];\n\n for (const [c1, c2] of stretchZones) {\n const last = cuts[cuts.length - 1];\n cuts.push({\n fixed: c1 - last.stretch,\n stretch: last.stretch\n });\n cuts.push({\n fixed: c1 - last.stretch,\n stretch: last.stretch + (c2 - c1)\n });\n }\n cuts.push({\n fixed: fixedSize + border,\n stretch: stretchSize\n });\n return cuts;\n}\n\nfunction getEmOffset(stretchOffset, stretchSize, iconSize, iconOffset) {\n return stretchOffset / stretchSize * iconSize + iconOffset;\n}\n\nfunction getPxOffset(fixedOffset, fixedSize, stretchOffset, stretchSize) {\n return fixedOffset - fixedSize * stretchOffset / stretchSize;\n}\n\n/**\n * Create the quads used for rendering a text label.\n */\nexport function getGlyphQuads(\n anchor: Anchor,\n shaping: Shaping,\n textOffset: [number, number],\n layer: SymbolStyleLayer,\n alongLine: boolean,\n feature: Feature,\n imageMap: {[_: string]: StyleImage},\n allowVerticalPlacement: boolean\n): Array {\n\n const textRotate = layer.layout.get('text-rotate').evaluate(feature, {}) * Math.PI / 180;\n const quads = [];\n\n for (const line of shaping.positionedLines) {\n for (const positionedGlyph of line.positionedGlyphs) {\n if (!positionedGlyph.rect) continue;\n const textureRect: Rect = positionedGlyph.rect || {} as Rect;\n\n // The rects have an additional buffer that is not included in their size.\n const glyphPadding = 1.0;\n let rectBuffer = GLYPH_PBF_BORDER + glyphPadding;\n let isSDF = true;\n let pixelRatio = 1.0;\n let lineOffset = 0.0;\n\n const rotateVerticalGlyph = (alongLine || allowVerticalPlacement) && positionedGlyph.vertical;\n const halfAdvance = positionedGlyph.metrics.advance * positionedGlyph.scale / 2;\n\n // Align images and scaled glyphs in the middle of a vertical line.\n if (allowVerticalPlacement && shaping.verticalizable) {\n const scaledGlyphOffset = (positionedGlyph.scale - 1) * ONE_EM;\n const imageOffset = (ONE_EM - positionedGlyph.metrics.width * positionedGlyph.scale) / 2;\n lineOffset = line.lineOffset / 2 - (positionedGlyph.imageName ? -imageOffset : scaledGlyphOffset);\n }\n\n if (positionedGlyph.imageName) {\n const image = imageMap[positionedGlyph.imageName];\n isSDF = image.sdf;\n pixelRatio = image.pixelRatio;\n rectBuffer = IMAGE_PADDING / pixelRatio;\n }\n\n const glyphOffset = alongLine ?\n [positionedGlyph.x + halfAdvance, positionedGlyph.y] :\n [0, 0];\n\n let builtInOffset: [number, number] = alongLine ?\n [0, 0] :\n [positionedGlyph.x + halfAdvance + textOffset[0], positionedGlyph.y + textOffset[1] - lineOffset];\n\n let verticalizedLabelOffset = [0, 0] as [number, number];\n if (rotateVerticalGlyph) {\n // Vertical POI labels that are rotated 90deg CW and whose glyphs must preserve upright orientation\n // need to be rotated 90deg CCW. After a quad is rotated, it is translated to the original built-in offset.\n verticalizedLabelOffset = builtInOffset;\n builtInOffset = [0, 0];\n }\n\n const textureScale = positionedGlyph.metrics.isDoubleResolution ? 2 : 1;\n\n const x1 = (positionedGlyph.metrics.left - rectBuffer) * positionedGlyph.scale - halfAdvance + builtInOffset[0];\n const y1 = (-positionedGlyph.metrics.top - rectBuffer) * positionedGlyph.scale + builtInOffset[1];\n const x2 = x1 + textureRect.w / textureScale * positionedGlyph.scale / pixelRatio;\n const y2 = y1 + textureRect.h / textureScale * positionedGlyph.scale / pixelRatio;\n\n const tl = new Point(x1, y1);\n const tr = new Point(x2, y1);\n const bl = new Point(x1, y2);\n const br = new Point(x2, y2);\n\n if (rotateVerticalGlyph) {\n // Vertical-supporting glyphs are laid out in 24x24 point boxes (1 square em)\n // In horizontal orientation, the y values for glyphs are below the midline\n // and we use a \"yOffset\" of -17 to pull them up to the middle.\n // By rotating counter-clockwise around the point at the center of the left\n // edge of a 24x24 layout box centered below the midline, we align the center\n // of the glyphs with the horizontal midline, so the yOffset is no longer\n // necessary, but we also pull the glyph to the left along the x axis.\n // The y coordinate includes baseline yOffset, thus needs to be accounted\n // for when glyph is rotated and translated.\n const center = new Point(-halfAdvance, halfAdvance - SHAPING_DEFAULT_OFFSET);\n const verticalRotation = -Math.PI / 2;\n\n // xHalfWidthOffsetCorrection is a difference between full-width and half-width\n // advance, should be 0 for full-width glyphs and will pull up half-width glyphs.\n const xHalfWidthOffsetCorrection = ONE_EM / 2 - halfAdvance;\n const yImageOffsetCorrection = positionedGlyph.imageName ? xHalfWidthOffsetCorrection : 0.0;\n const halfWidthOffsetCorrection = new Point(5 - SHAPING_DEFAULT_OFFSET - xHalfWidthOffsetCorrection, -yImageOffsetCorrection);\n const verticalOffsetCorrection = new Point(...verticalizedLabelOffset);\n tl._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection);\n tr._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection);\n bl._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection);\n br._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection);\n }\n\n if (textRotate) {\n const sin = Math.sin(textRotate),\n cos = Math.cos(textRotate),\n matrix = [cos, -sin, sin, cos];\n\n tl._matMult(matrix);\n tr._matMult(matrix);\n bl._matMult(matrix);\n br._matMult(matrix);\n }\n\n const pixelOffsetTL = new Point(0, 0);\n const pixelOffsetBR = new Point(0, 0);\n const minFontScaleX = 0;\n const minFontScaleY = 0;\n quads.push({tl, tr, bl, br, tex: textureRect, writingMode: shaping.writingMode, glyphOffset, sectionIndex: positionedGlyph.sectionIndex, isSDF, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY});\n }\n }\n\n return quads;\n}\n","import type {CollisionBoxArray} from '../data/array_types.g';\nimport Point from '@mapbox/point-geometry';\nimport type {Anchor} from './anchor';\nimport {SymbolPadding} from '../style/style_layer/symbol_style_layer';\n\n/**\n * A CollisionFeature represents the area of the tile covered by a single label.\n * It is used with CollisionIndex to check if the label overlaps with any\n * previous labels. A CollisionFeature is mostly just a set of CollisionBox\n * objects.\n */\nexport class CollisionFeature {\n boxStartIndex: number;\n boxEndIndex: number;\n circleDiameter: number;\n\n /**\n * Create a CollisionFeature, adding its collision box data to the given collisionBoxArray in the process.\n * For line aligned labels a collision circle diameter is computed instead.\n *\n * @param anchor - The point along the line around which the label is anchored.\n * @param shaped - The text or icon shaping results.\n * @param boxScale - A magic number used to convert from glyph metrics units to geometry units.\n * @param padding - The amount of padding to add around the label edges.\n * @param alignLine - Whether the label is aligned with the line or the viewport.\n */\n constructor(collisionBoxArray: CollisionBoxArray,\n anchor: Anchor,\n featureIndex: number,\n sourceLayerIndex: number,\n bucketIndex: number,\n shaped: any,\n boxScale: number,\n padding: SymbolPadding,\n alignLine: boolean,\n rotate: number) {\n\n this.boxStartIndex = collisionBoxArray.length;\n\n if (alignLine) {\n // Compute height of the shape in glyph metrics and apply collision padding.\n // Note that the pixel based 'text-padding' is applied at runtime\n let top = shaped.top;\n let bottom = shaped.bottom;\n const collisionPadding = shaped.collisionPadding;\n\n if (collisionPadding) {\n top -= collisionPadding[1];\n bottom += collisionPadding[3];\n }\n\n let height = bottom - top;\n\n if (height > 0) {\n // set minimum box height to avoid very many small labels\n height = Math.max(10, height);\n this.circleDiameter = height;\n }\n } else {\n // margin is in CSS order: [top, right, bottom, left]\n let y1 = shaped.top * boxScale - padding[0];\n let y2 = shaped.bottom * boxScale + padding[2];\n let x1 = shaped.left * boxScale - padding[3];\n let x2 = shaped.right * boxScale + padding[1];\n\n const collisionPadding = shaped.collisionPadding;\n if (collisionPadding) {\n x1 -= collisionPadding[0] * boxScale;\n y1 -= collisionPadding[1] * boxScale;\n x2 += collisionPadding[2] * boxScale;\n y2 += collisionPadding[3] * boxScale;\n }\n\n if (rotate) {\n // Account for *-rotate in point collision boxes\n // See https://github.com/mapbox/mapbox-gl-js/issues/6075\n // Doesn't account for icon-text-fit\n\n const tl = new Point(x1, y1);\n const tr = new Point(x2, y1);\n const bl = new Point(x1, y2);\n const br = new Point(x2, y2);\n\n const rotateRadians = rotate * Math.PI / 180;\n\n tl._rotate(rotateRadians);\n tr._rotate(rotateRadians);\n bl._rotate(rotateRadians);\n br._rotate(rotateRadians);\n\n // Collision features require an \"on-axis\" geometry,\n // so take the envelope of the rotated geometry\n // (may be quite large for wide labels rotated 45 degrees)\n x1 = Math.min(tl.x, tr.x, bl.x, br.x);\n x2 = Math.max(tl.x, tr.x, bl.x, br.x);\n y1 = Math.min(tl.y, tr.y, bl.y, br.y);\n y2 = Math.max(tl.y, tr.y, bl.y, br.y);\n }\n collisionBoxArray.emplaceBack(anchor.x, anchor.y, x1, y1, x2, y2, featureIndex, sourceLayerIndex, bucketIndex);\n }\n\n this.boxEndIndex = collisionBoxArray.length;\n }\n}\n","\nexport default class TinyQueue {\n constructor(data = [], compare = defaultCompare) {\n this.data = data;\n this.length = this.data.length;\n this.compare = compare;\n\n if (this.length > 0) {\n for (let i = (this.length >> 1) - 1; i >= 0; i--) this._down(i);\n }\n }\n\n push(item) {\n this.data.push(item);\n this.length++;\n this._up(this.length - 1);\n }\n\n pop() {\n if (this.length === 0) return undefined;\n\n const top = this.data[0];\n const bottom = this.data.pop();\n this.length--;\n\n if (this.length > 0) {\n this.data[0] = bottom;\n this._down(0);\n }\n\n return top;\n }\n\n peek() {\n return this.data[0];\n }\n\n _up(pos) {\n const {data, compare} = this;\n const item = data[pos];\n\n while (pos > 0) {\n const parent = (pos - 1) >> 1;\n const current = data[parent];\n if (compare(item, current) >= 0) break;\n data[pos] = current;\n pos = parent;\n }\n\n data[pos] = item;\n }\n\n _down(pos) {\n const {data, compare} = this;\n const halfLength = this.length >> 1;\n const item = data[pos];\n\n while (pos < halfLength) {\n let left = (pos << 1) + 1;\n let best = data[left];\n const right = left + 1;\n\n if (right < this.length && compare(data[right], best) < 0) {\n left = right;\n best = data[right];\n }\n if (compare(best, item) >= 0) break;\n\n data[pos] = best;\n pos = left;\n }\n\n data[pos] = item;\n }\n}\n\nfunction defaultCompare(a, b) {\n return a < b ? -1 : a > b ? 1 : 0;\n}\n","import Queue from 'tinyqueue';\n\nimport Point from '@mapbox/point-geometry';\nimport {distToSegmentSquared} from './intersection_tests';\n\n/**\n * Finds an approximation of a polygon's Pole Of Inaccessibiliy https://en.wikipedia.org/wiki/Pole_of_inaccessibility\n * This is a copy of http://github.com/mapbox/polylabel adapted to use Points\n *\n * @param polygonRings - first item in array is the outer ring followed optionally by the list of holes, should be an element of the result of util/classify_rings\n * @param precision - Specified in input coordinate units. If 0 returns after first run, if `> 0` repeatedly narrows the search space until the radius of the area searched for the best pole is less than precision\n * @param debug - Print some statistics to the console during execution\n * @returns Pole of Inaccessibiliy.\n */\nexport function findPoleOfInaccessibility(\n polygonRings: Array>,\n precision: number = 1,\n debug: boolean = false\n): Point {\n // find the bounding box of the outer ring\n let minX = Infinity, minY = Infinity, maxX = -Infinity, maxY = -Infinity;\n const outerRing = polygonRings[0];\n for (let i = 0; i < outerRing.length; i++) {\n const p = outerRing[i];\n if (!i || p.x < minX) minX = p.x;\n if (!i || p.y < minY) minY = p.y;\n if (!i || p.x > maxX) maxX = p.x;\n if (!i || p.y > maxY) maxY = p.y;\n }\n\n const width = maxX - minX;\n const height = maxY - minY;\n const cellSize = Math.min(width, height);\n let h = cellSize / 2;\n\n // a priority queue of cells in order of their \"potential\" (max distance to polygon)\n const cellQueue = new Queue([], compareMax);\n\n if (cellSize === 0) return new Point(minX, minY);\n\n // cover polygon with initial cells\n for (let x = minX; x < maxX; x += cellSize) {\n for (let y = minY; y < maxY; y += cellSize) {\n cellQueue.push(new Cell(x + h, y + h, h, polygonRings));\n }\n }\n\n // take centroid as the first best guess\n let bestCell = getCentroidCell(polygonRings);\n let numProbes = cellQueue.length;\n\n while (cellQueue.length) {\n // pick the most promising cell from the queue\n const cell = cellQueue.pop();\n\n // update the best cell if we found a better one\n if (cell.d > bestCell.d || !bestCell.d) {\n bestCell = cell;\n if (debug) console.log('found best %d after %d probes', Math.round(1e4 * cell.d) / 1e4, numProbes);\n }\n\n // do not drill down further if there's no chance of a better solution\n if (cell.max - bestCell.d <= precision) continue;\n\n // split the cell into four cells\n h = cell.h / 2;\n cellQueue.push(new Cell(cell.p.x - h, cell.p.y - h, h, polygonRings));\n cellQueue.push(new Cell(cell.p.x + h, cell.p.y - h, h, polygonRings));\n cellQueue.push(new Cell(cell.p.x - h, cell.p.y + h, h, polygonRings));\n cellQueue.push(new Cell(cell.p.x + h, cell.p.y + h, h, polygonRings));\n numProbes += 4;\n }\n\n if (debug) {\n console.log(`num probes: ${numProbes}`);\n console.log(`best distance: ${bestCell.d}`);\n }\n\n return bestCell.p;\n}\n\nfunction compareMax(a, b) {\n return b.max - a.max;\n}\n\nfunction Cell(x, y, h, polygon) {\n this.p = new Point(x, y);\n this.h = h; // half the cell size\n this.d = pointToPolygonDist(this.p, polygon); // distance from cell center to polygon\n this.max = this.d + this.h * Math.SQRT2; // max distance to polygon within a cell\n}\n\n// signed distance from point to polygon outline (negative if point is outside)\nfunction pointToPolygonDist(p, polygon) {\n let inside = false;\n let minDistSq = Infinity;\n\n for (let k = 0; k < polygon.length; k++) {\n const ring = polygon[k];\n\n for (let i = 0, len = ring.length, j = len - 1; i < len; j = i++) {\n const a = ring[i];\n const b = ring[j];\n\n if ((a.y > p.y !== b.y > p.y) &&\n (p.x < (b.x - a.x) * (p.y - a.y) / (b.y - a.y) + a.x)) inside = !inside;\n\n minDistSq = Math.min(minDistSq, distToSegmentSquared(p, a, b));\n }\n }\n\n return (inside ? 1 : -1) * Math.sqrt(minDistSq);\n}\n\n// get polygon centroid\nfunction getCentroidCell(polygon) {\n let area = 0;\n let x = 0;\n let y = 0;\n const points = polygon[0];\n for (let i = 0, len = points.length, j = len - 1; i < len; j = i++) {\n const a = points[i];\n const b = points[j];\n const f = a.x * b.y - b.x * a.y;\n x += (a.x + b.x) * f;\n y += (a.y + b.y) * f;\n area += f * 3;\n }\n return new Cell(x / area, y / area, 0, polygon);\n}\n","import {VariableAnchorOffsetCollection, VariableAnchorOffsetCollectionSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport {SymbolFeature} from '../../data/bucket/symbol_bucket';\nimport {CanonicalTileID} from '../../source/tile_id';\nimport ONE_EM from '../../symbol/one_em';\nimport {SymbolStyleLayer} from './symbol_style_layer';\n\nexport enum TextAnchorEnum {\n 'center' = 1,\n 'left' = 2,\n 'right' = 3,\n 'top' = 4,\n 'bottom' = 5,\n 'top-left' = 6,\n 'top-right' = 7,\n 'bottom-left' = 8,\n 'bottom-right' = 9\n}\n\nexport type TextAnchor = keyof typeof TextAnchorEnum;\n\n// The radial offset is to the edge of the text box\n// In the horizontal direction, the edge of the text box is where glyphs start\n// But in the vertical direction, the glyphs appear to \"start\" at the baseline\n// We don't actually load baseline data, but we assume an offset of ONE_EM - 17\n// (see \"yOffset\" in shaping.js)\nconst baselineOffset = 7;\nexport const INVALID_TEXT_OFFSET = Number.POSITIVE_INFINITY;\n\nexport function evaluateVariableOffset(anchor: TextAnchor, offset: [number, number]): [number, number] {\n\n function fromRadialOffset(anchor: TextAnchor, radialOffset: number): [number, number] {\n let x = 0, y = 0;\n if (radialOffset < 0) radialOffset = 0; // Ignore negative offset.\n // solve for r where r^2 + r^2 = radialOffset^2\n const hypotenuse = radialOffset / Math.SQRT2;\n switch (anchor) {\n case 'top-right':\n case 'top-left':\n y = hypotenuse - baselineOffset;\n break;\n case 'bottom-right':\n case 'bottom-left':\n y = -hypotenuse + baselineOffset;\n break;\n case 'bottom':\n y = -radialOffset + baselineOffset;\n break;\n case 'top':\n y = radialOffset - baselineOffset;\n break;\n }\n\n switch (anchor) {\n case 'top-right':\n case 'bottom-right':\n x = -hypotenuse;\n break;\n case 'top-left':\n case 'bottom-left':\n x = hypotenuse;\n break;\n case 'left':\n x = radialOffset;\n break;\n case 'right':\n x = -radialOffset;\n break;\n }\n\n return [x, y];\n }\n\n function fromTextOffset(anchor: TextAnchor, offsetX: number, offsetY: number): [number, number] {\n let x = 0, y = 0;\n // Use absolute offset values.\n offsetX = Math.abs(offsetX);\n offsetY = Math.abs(offsetY);\n\n switch (anchor) {\n case 'top-right':\n case 'top-left':\n case 'top':\n y = offsetY - baselineOffset;\n break;\n case 'bottom-right':\n case 'bottom-left':\n case 'bottom':\n y = -offsetY + baselineOffset;\n break;\n }\n\n switch (anchor) {\n case 'top-right':\n case 'bottom-right':\n case 'right':\n x = -offsetX;\n break;\n case 'top-left':\n case 'bottom-left':\n case 'left':\n x = offsetX;\n break;\n }\n\n return [x, y];\n }\n\n return (offset[1] !== INVALID_TEXT_OFFSET) ? fromTextOffset(anchor, offset[0], offset[1]) : fromRadialOffset(anchor, offset[0]);\n}\n\n// Helper to support both text-variable-anchor and text-variable-anchor-offset. Offset values converted from EMs to PXs\nexport function getTextVariableAnchorOffset(layer: SymbolStyleLayer, feature: SymbolFeature, canonical: CanonicalTileID): VariableAnchorOffsetCollection | null {\n const layout = layer.layout;\n // If style specifies text-variable-anchor-offset, just return it\n const variableAnchorOffset = layout.get('text-variable-anchor-offset')?.evaluate(feature, {}, canonical);\n\n if (variableAnchorOffset) {\n const sourceValues = variableAnchorOffset.values;\n const destValues: VariableAnchorOffsetCollectionSpecification = [];\n\n // Convert offsets from EM to PX, and apply baseline shift\n for (let i = 0; i < sourceValues.length; i += 2) {\n const anchor = destValues[i] = sourceValues[i] as TextAnchor;\n const offset = (sourceValues[i + 1] as [number, number]).map(t => t * ONE_EM) as [number, number];\n\n if (anchor.startsWith('top')) {\n offset[1] -= baselineOffset;\n } else if (anchor.startsWith('bottom')) {\n offset[1] += baselineOffset;\n }\n\n destValues[i + 1] = offset;\n }\n\n return new VariableAnchorOffsetCollection(destValues);\n }\n\n // If style specifies text-variable-anchor, convert to the new format\n const variableAnchor = layout.get('text-variable-anchor');\n\n if (variableAnchor) {\n let textOffset: [number, number];\n const unevaluatedLayout = layer._unevaluatedLayout;\n\n // The style spec says don't use `text-offset` and `text-radial-offset` together\n // but doesn't actually specify what happens if you use both. We go with the radial offset.\n if (unevaluatedLayout.getValue('text-radial-offset') !== undefined) {\n textOffset = [layout.get('text-radial-offset').evaluate(feature, {}, canonical) * ONE_EM, INVALID_TEXT_OFFSET];\n } else {\n textOffset = layout.get('text-offset').evaluate(feature, {}, canonical).map(t => t * ONE_EM) as [number, number];\n }\n\n const anchorOffsets: VariableAnchorOffsetCollectionSpecification = [];\n\n for (const anchor of variableAnchor) {\n anchorOffsets.push(anchor, evaluateVariableOffset(anchor, textOffset));\n }\n\n return new VariableAnchorOffsetCollection(anchorOffsets);\n }\n\n return null;\n}\n","import {Anchor} from './anchor';\n\nimport {getAnchors, getCenterAnchor} from './get_anchors';\nimport {clipLine} from './clip_line';\nimport {shapeText, shapeIcon, WritingMode, fitIconToText} from './shaping';\nimport {getGlyphQuads, getIconQuads} from './quads';\nimport {CollisionFeature} from './collision_feature';\nimport {warnOnce} from '../util/util';\nimport {\n allowsVerticalWritingMode,\n allowsLetterSpacing\n} from '../util/script_detection';\nimport {findPoleOfInaccessibility} from '../util/find_pole_of_inaccessibility';\nimport {classifyRings} from '../util/classify_rings';\nimport {EXTENT} from '../data/extent';\nimport {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {SIZE_PACK_FACTOR, MAX_PACKED_SIZE, MAX_GLYPH_ICON_SIZE} from './symbol_size';\nimport ONE_EM from './one_em';\nimport type {CanonicalTileID} from '../source/tile_id';\nimport type {Shaping, PositionedIcon, TextJustify} from './shaping';\nimport type {CollisionBoxArray, TextAnchorOffsetArray} from '../data/array_types.g';\nimport type {SymbolFeature} from '../data/bucket/symbol_bucket';\nimport type {StyleImage} from '../style/style_image';\nimport type {StyleGlyph} from '../style/style_glyph';\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\nimport type {ImagePosition} from '../render/image_atlas';\nimport type {GlyphPosition} from '../render/glyph_atlas';\nimport type {PossiblyEvaluatedPropertyValue} from '../style/properties';\n\nimport Point from '@mapbox/point-geometry';\nimport murmur3 from 'murmurhash-js';\nimport {getIconPadding, SymbolPadding} from '../style/style_layer/symbol_style_layer';\nimport {VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec';\nimport {getTextVariableAnchorOffset, evaluateVariableOffset, INVALID_TEXT_OFFSET, TextAnchor, TextAnchorEnum} from '../style/style_layer/variable_text_anchor';\n\n// The symbol layout process needs `text-size` evaluated at up to five different zoom levels, and\n// `icon-size` at up to three:\n//\n// 1. `text-size` at the zoom level of the bucket. Used to calculate a per-feature size for source `text-size`\n// expressions, and to calculate the box dimensions for icon-text-fit.\n// 2. `icon-size` at the zoom level of the bucket. Used to calculate a per-feature size for source `icon-size`\n// expressions.\n// 3. `text-size` and `icon-size` at the zoom level of the bucket, plus one. Used to calculate collision boxes.\n// 4. `text-size` at zoom level 18. Used for something line-symbol-placement-related.\n// 5. For composite `*-size` expressions: two zoom levels of curve stops that \"cover\" the zoom level of the\n// bucket. These go into a vertex buffer and are used by the shader to interpolate the size at render time.\n//\n// (1) and (2) are stored in `bucket.layers[0].layout`. The remainder are below.\n//\ntype Sizes = {\n layoutTextSize: PossiblyEvaluatedPropertyValue; // (3),\n layoutIconSize: PossiblyEvaluatedPropertyValue; // (3),\n textMaxSize: PossiblyEvaluatedPropertyValue; // (4),\n compositeTextSizes: [PossiblyEvaluatedPropertyValue, PossiblyEvaluatedPropertyValue]; // (5),\n compositeIconSizes: [PossiblyEvaluatedPropertyValue, PossiblyEvaluatedPropertyValue]; // (5)\n};\n\ntype ShapedTextOrientations = {\n vertical: Shaping | false;\n horizontal: Record;\n};\n\nexport function performSymbolLayout(args: {\n bucket: SymbolBucket;\n glyphMap: {\n [_: string]: {\n [x: number]: StyleGlyph;\n };\n };\n glyphPositions: {\n [_: string]: {\n [x: number]: GlyphPosition;\n };\n };\n imageMap: {[_: string]: StyleImage};\n imagePositions: {[_: string]: ImagePosition};\n showCollisionBoxes: boolean;\n canonical: CanonicalTileID;\n}) {\n args.bucket.createArrays();\n\n const tileSize = 512 * args.bucket.overscaling;\n args.bucket.tilePixelRatio = EXTENT / tileSize;\n args.bucket.compareText = {};\n args.bucket.iconsNeedLinear = false;\n\n const layer = args.bucket.layers[0];\n const layout = layer.layout;\n const unevaluatedLayoutValues = layer._unevaluatedLayout._values;\n\n const sizes: Sizes = {\n // Filled in below, if *SizeData.kind is 'composite'\n // compositeIconSizes: undefined,\n // compositeTextSizes: undefined,\n layoutIconSize: unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(args.bucket.zoom + 1), args.canonical),\n layoutTextSize: unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(args.bucket.zoom + 1), args.canonical),\n textMaxSize: unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(18))\n } as Sizes;\n\n if (args.bucket.textSizeData.kind === 'composite') {\n const {minZoom, maxZoom} = args.bucket.textSizeData;\n sizes.compositeTextSizes = [\n unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(minZoom), args.canonical),\n unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), args.canonical)\n ];\n }\n\n if (args.bucket.iconSizeData.kind === 'composite') {\n const {minZoom, maxZoom} = args.bucket.iconSizeData;\n sizes.compositeIconSizes = [\n unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(minZoom), args.canonical),\n unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), args.canonical)\n ];\n }\n\n const lineHeight = layout.get('text-line-height') * ONE_EM;\n const textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point';\n const keepUpright = layout.get('text-keep-upright');\n const textSize = layout.get('text-size');\n\n for (const feature of args.bucket.features) {\n const fontstack = layout.get('text-font').evaluate(feature, {}, args.canonical).join(',');\n const layoutTextSizeThisZoom = textSize.evaluate(feature, {}, args.canonical);\n const layoutTextSize = sizes.layoutTextSize.evaluate(feature, {}, args.canonical);\n const layoutIconSize = sizes.layoutIconSize.evaluate(feature, {}, args.canonical);\n\n const shapedTextOrientations: ShapedTextOrientations = {\n horizontal: {} as Record,\n vertical: undefined\n };\n const text = feature.text;\n let textOffset: [number, number] = [0, 0];\n if (text) {\n const unformattedText = text.toString();\n const spacing = layout.get('text-letter-spacing').evaluate(feature, {}, args.canonical) * ONE_EM;\n const spacingIfAllowed = allowsLetterSpacing(unformattedText) ? spacing : 0;\n\n const textAnchor = layout.get('text-anchor').evaluate(feature, {}, args.canonical);\n const variableAnchorOffset = getTextVariableAnchorOffset(layer, feature, args.canonical);\n\n if (!variableAnchorOffset) {\n const radialOffset = layout.get('text-radial-offset').evaluate(feature, {}, args.canonical);\n // Layers with variable anchors use the `text-radial-offset` property and the [x, y] offset vector\n // is calculated at placement time instead of layout time\n if (radialOffset) {\n // The style spec says don't use `text-offset` and `text-radial-offset` together\n // but doesn't actually specify what happens if you use both. We go with the radial offset.\n textOffset = evaluateVariableOffset(textAnchor, [radialOffset * ONE_EM, INVALID_TEXT_OFFSET]);\n } else {\n textOffset = (layout.get('text-offset').evaluate(feature, {}, args.canonical).map(t => t * ONE_EM) as [number, number]);\n }\n }\n\n let textJustify = textAlongLine ?\n 'center' :\n layout.get('text-justify').evaluate(feature, {}, args.canonical);\n\n const symbolPlacement = layout.get('symbol-placement');\n const maxWidth = symbolPlacement === 'point' ?\n layout.get('text-max-width').evaluate(feature, {}, args.canonical) * ONE_EM :\n 0;\n\n const addVerticalShapingForPointLabelIfNeeded = () => {\n if (args.bucket.allowVerticalPlacement && allowsVerticalWritingMode(unformattedText)) {\n // Vertical POI label placement is meant to be used for scripts that support vertical\n // writing mode, thus, default left justification is used. If Latin\n // scripts would need to be supported, this should take into account other justifications.\n shapedTextOrientations.vertical = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor,\n 'left', spacingIfAllowed, textOffset, WritingMode.vertical, true, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n }\n };\n\n // If this layer uses text-variable-anchor, generate shapings for all justification possibilities.\n if (!textAlongLine && variableAnchorOffset) {\n const justifications = new Set();\n\n if (textJustify === 'auto') {\n for (let i = 0; i < variableAnchorOffset.values.length; i += 2) {\n justifications.add(getAnchorJustification(variableAnchorOffset.values[i] as TextAnchor));\n }\n } else {\n justifications.add(textJustify);\n }\n\n let singleLine = false;\n for (const justification of justifications) {\n if (shapedTextOrientations.horizontal[justification]) continue;\n if (singleLine) {\n // If the shaping for the first justification was only a single line, we\n // can re-use it for the other justifications\n shapedTextOrientations.horizontal[justification] = shapedTextOrientations.horizontal[0];\n } else {\n // If using text-variable-anchor for the layer, we use a center anchor for all shapings and apply\n // the offsets for the anchor in the placement step.\n const shaping = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, 'center',\n justification, spacingIfAllowed, textOffset, WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n if (shaping) {\n shapedTextOrientations.horizontal[justification] = shaping;\n singleLine = shaping.positionedLines.length === 1;\n }\n }\n }\n\n addVerticalShapingForPointLabelIfNeeded();\n } else {\n if (textJustify === 'auto') {\n textJustify = getAnchorJustification(textAnchor);\n }\n\n // Horizontal point or line label.\n const shaping = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify, spacingIfAllowed,\n textOffset, WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n if (shaping) shapedTextOrientations.horizontal[textJustify] = shaping;\n\n // Vertical point label (if allowVerticalPlacement is enabled).\n addVerticalShapingForPointLabelIfNeeded();\n\n // Verticalized line label.\n if (allowsVerticalWritingMode(unformattedText) && textAlongLine && keepUpright) {\n shapedTextOrientations.vertical = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify,\n spacingIfAllowed, textOffset, WritingMode.vertical, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom);\n }\n }\n }\n\n let shapedIcon;\n let isSDFIcon = false;\n if (feature.icon && feature.icon.name) {\n const image = args.imageMap[feature.icon.name];\n if (image) {\n shapedIcon = shapeIcon(\n args.imagePositions[feature.icon.name],\n layout.get('icon-offset').evaluate(feature, {}, args.canonical),\n layout.get('icon-anchor').evaluate(feature, {}, args.canonical));\n // null/undefined SDF property treated same as default (false)\n isSDFIcon = !!image.sdf;\n if (args.bucket.sdfIcons === undefined) {\n args.bucket.sdfIcons = isSDFIcon;\n } else if (args.bucket.sdfIcons !== isSDFIcon) {\n warnOnce('Style sheet warning: Cannot mix SDF and non-SDF icons in one buffer');\n }\n if (image.pixelRatio !== args.bucket.pixelRatio) {\n args.bucket.iconsNeedLinear = true;\n } else if (layout.get('icon-rotate').constantOr(1) !== 0) {\n args.bucket.iconsNeedLinear = true;\n }\n }\n }\n\n const shapedText = getDefaultHorizontalShaping(shapedTextOrientations.horizontal) || shapedTextOrientations.vertical;\n args.bucket.iconsInText = shapedText ? shapedText.iconsInText : false;\n if (shapedText || shapedIcon) {\n addFeature(args.bucket, feature, shapedTextOrientations, shapedIcon, args.imageMap, sizes, layoutTextSize, layoutIconSize, textOffset, isSDFIcon, args.canonical);\n }\n }\n\n if (args.showCollisionBoxes) {\n args.bucket.generateCollisionDebugBuffers();\n }\n}\n\n// Choose the justification that matches the direction of the TextAnchor\nexport function getAnchorJustification(anchor: TextAnchor): TextJustify {\n switch (anchor) {\n case 'right':\n case 'top-right':\n case 'bottom-right':\n return 'right';\n case 'left':\n case 'top-left':\n case 'bottom-left':\n return 'left';\n }\n return 'center';\n}\n\n/**\n * Given a feature and its shaped text and icon data, add a 'symbol\n * instance' for each _possible_ placement of the symbol feature.\n * (At render timePlaceSymbols#place() selects which of these instances to\n * show or hide based on collisions with symbols in other layers.)\n */\nfunction addFeature(bucket: SymbolBucket,\n feature: SymbolFeature,\n shapedTextOrientations: ShapedTextOrientations,\n shapedIcon: PositionedIcon,\n imageMap: {[_: string]: StyleImage},\n sizes: Sizes,\n layoutTextSize: number,\n layoutIconSize: number,\n textOffset: [number, number],\n isSDFIcon: boolean, canonical: CanonicalTileID) {\n // To reduce the number of labels that jump around when zooming we need\n // to use a text-size value that is the same for all zoom levels.\n // bucket calculates text-size at a high zoom level so that all tiles can\n // use the same value when calculating anchor positions.\n let textMaxSize = sizes.textMaxSize.evaluate(feature, {});\n if (textMaxSize === undefined) {\n textMaxSize = layoutTextSize;\n }\n const layout = bucket.layers[0].layout;\n const iconOffset = layout.get('icon-offset').evaluate(feature, {}, canonical);\n const defaultHorizontalShaping = getDefaultHorizontalShaping(shapedTextOrientations.horizontal);\n const glyphSize = 24,\n fontScale = layoutTextSize / glyphSize,\n textBoxScale = bucket.tilePixelRatio * fontScale,\n textMaxBoxScale = bucket.tilePixelRatio * textMaxSize / glyphSize,\n iconBoxScale = bucket.tilePixelRatio * layoutIconSize,\n symbolMinDistance = bucket.tilePixelRatio * layout.get('symbol-spacing'),\n textPadding = layout.get('text-padding') * bucket.tilePixelRatio,\n iconPadding = getIconPadding(layout, feature, canonical, bucket.tilePixelRatio),\n textMaxAngle = layout.get('text-max-angle') / 180 * Math.PI,\n textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point',\n iconAlongLine = layout.get('icon-rotation-alignment') === 'map' && layout.get('symbol-placement') !== 'point',\n symbolPlacement = layout.get('symbol-placement'),\n textRepeatDistance = symbolMinDistance / 2;\n\n const iconTextFit = layout.get('icon-text-fit');\n let verticallyShapedIcon;\n // Adjust shaped icon size when icon-text-fit is used.\n if (shapedIcon && iconTextFit !== 'none') {\n if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) {\n verticallyShapedIcon = fitIconToText(shapedIcon, shapedTextOrientations.vertical, iconTextFit,\n layout.get('icon-text-fit-padding'), iconOffset, fontScale);\n }\n if (defaultHorizontalShaping) {\n shapedIcon = fitIconToText(shapedIcon, defaultHorizontalShaping, iconTextFit,\n layout.get('icon-text-fit-padding'), iconOffset, fontScale);\n }\n }\n\n const addSymbolAtAnchor = (line, anchor) => {\n if (anchor.x < 0 || anchor.x >= EXTENT || anchor.y < 0 || anchor.y >= EXTENT) {\n // Symbol layers are drawn across tile boundaries, We filter out symbols\n // outside our tile boundaries (which may be included in vector tile buffers)\n // to prevent double-drawing symbols.\n return;\n }\n\n addSymbol(bucket, anchor, line, shapedTextOrientations, shapedIcon, imageMap, verticallyShapedIcon, bucket.layers[0],\n bucket.collisionBoxArray, feature.index, feature.sourceLayerIndex, bucket.index,\n textBoxScale, [textPadding, textPadding, textPadding, textPadding], textAlongLine, textOffset,\n iconBoxScale, iconPadding, iconAlongLine, iconOffset,\n feature, sizes, isSDFIcon, canonical, layoutTextSize);\n };\n\n if (symbolPlacement === 'line') {\n for (const line of clipLine(feature.geometry, 0, 0, EXTENT, EXTENT)) {\n const anchors = getAnchors(\n line,\n symbolMinDistance,\n textMaxAngle,\n shapedTextOrientations.vertical || defaultHorizontalShaping,\n shapedIcon,\n glyphSize,\n textMaxBoxScale,\n bucket.overscaling,\n EXTENT\n );\n for (const anchor of anchors) {\n const shapedText = defaultHorizontalShaping;\n if (!shapedText || !anchorIsTooClose(bucket, shapedText.text, textRepeatDistance, anchor)) {\n addSymbolAtAnchor(line, anchor);\n }\n }\n }\n } else if (symbolPlacement === 'line-center') {\n // No clipping, multiple lines per feature are allowed\n // \"lines\" with only one point are ignored as in clipLines\n for (const line of feature.geometry) {\n if (line.length > 1) {\n const anchor = getCenterAnchor(\n line,\n textMaxAngle,\n shapedTextOrientations.vertical || defaultHorizontalShaping,\n shapedIcon,\n glyphSize,\n textMaxBoxScale);\n if (anchor) {\n addSymbolAtAnchor(line, anchor);\n }\n }\n }\n } else if (feature.type === 'Polygon') {\n for (const polygon of classifyRings(feature.geometry, 0)) {\n // 16 here represents 2 pixels\n const poi = findPoleOfInaccessibility(polygon, 16);\n addSymbolAtAnchor(polygon[0], new Anchor(poi.x, poi.y, 0));\n }\n } else if (feature.type === 'LineString') {\n // https://github.com/mapbox/mapbox-gl-js/issues/3808\n for (const line of feature.geometry) {\n addSymbolAtAnchor(line, new Anchor(line[0].x, line[0].y, 0));\n }\n } else if (feature.type === 'Point') {\n for (const points of feature.geometry) {\n for (const point of points) {\n addSymbolAtAnchor([point], new Anchor(point.x, point.y, 0));\n }\n }\n }\n}\n\nfunction addTextVariableAnchorOffsets(textAnchorOffsets: TextAnchorOffsetArray, variableAnchorOffset: VariableAnchorOffsetCollection): [number, number] {\n const startIndex = textAnchorOffsets.length;\n const values = variableAnchorOffset?.values;\n\n if (values?.length > 0) {\n for (let i = 0; i < values.length; i += 2) {\n const anchor = TextAnchorEnum[values[i] as TextAnchor];\n const offset = values[i + 1] as [number, number];\n\n textAnchorOffsets.emplaceBack(anchor, offset[0], offset[1]);\n }\n }\n\n return [startIndex, textAnchorOffsets.length];\n}\n\nfunction addTextVertices(bucket: SymbolBucket,\n anchor: Point,\n shapedText: Shaping,\n imageMap: {[_: string]: StyleImage},\n layer: SymbolStyleLayer,\n textAlongLine: boolean,\n feature: SymbolFeature,\n textOffset: [number, number],\n lineArray: {\n lineStartIndex: number;\n lineLength: number;\n },\n writingMode: WritingMode,\n placementTypes: Array<'vertical' | 'center' | 'left' | 'right'>,\n placedTextSymbolIndices: {[_: string]: number},\n placedIconIndex: number,\n sizes: Sizes,\n canonical: CanonicalTileID) {\n const glyphQuads = getGlyphQuads(anchor, shapedText, textOffset,\n layer, textAlongLine, feature, imageMap, bucket.allowVerticalPlacement);\n\n const sizeData = bucket.textSizeData;\n let textSizeData = null;\n\n if (sizeData.kind === 'source') {\n textSizeData = [\n SIZE_PACK_FACTOR * layer.layout.get('text-size').evaluate(feature, {})\n ];\n if (textSizeData[0] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"text-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"text-size\".`);\n }\n } else if (sizeData.kind === 'composite') {\n textSizeData = [\n SIZE_PACK_FACTOR * sizes.compositeTextSizes[0].evaluate(feature, {}, canonical),\n SIZE_PACK_FACTOR * sizes.compositeTextSizes[1].evaluate(feature, {}, canonical)\n ];\n if (textSizeData[0] > MAX_PACKED_SIZE || textSizeData[1] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"text-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"text-size\".`);\n }\n }\n\n bucket.addSymbols(\n bucket.text,\n glyphQuads,\n textSizeData,\n textOffset,\n textAlongLine,\n feature,\n writingMode,\n anchor,\n lineArray.lineStartIndex,\n lineArray.lineLength,\n placedIconIndex,\n canonical);\n\n // The placedSymbolArray is used at render time in drawTileSymbols\n // These indices allow access to the array at collision detection time\n for (const placementType of placementTypes) {\n placedTextSymbolIndices[placementType] = bucket.text.placedSymbolArray.length - 1;\n }\n\n return glyphQuads.length * 4;\n}\n\nfunction getDefaultHorizontalShaping(\n horizontalShaping: Record\n): Shaping | null {\n // We don't care which shaping we get because this is used for collision purposes\n // and all the justifications have the same collision box\n for (const justification in horizontalShaping) {\n return horizontalShaping[justification];\n }\n return null;\n}\n\n/**\n * Add a single label & icon placement.\n */\nfunction addSymbol(bucket: SymbolBucket,\n anchor: Anchor,\n line: Array,\n shapedTextOrientations: ShapedTextOrientations,\n shapedIcon: PositionedIcon | void,\n imageMap: {[_: string]: StyleImage},\n verticallyShapedIcon: PositionedIcon | void,\n layer: SymbolStyleLayer,\n collisionBoxArray: CollisionBoxArray,\n featureIndex: number,\n sourceLayerIndex: number,\n bucketIndex: number,\n textBoxScale: number,\n textPadding: SymbolPadding,\n textAlongLine: boolean,\n textOffset: [number, number],\n iconBoxScale: number,\n iconPadding: SymbolPadding,\n iconAlongLine: boolean,\n iconOffset: [number, number],\n feature: SymbolFeature,\n sizes: Sizes,\n isSDFIcon: boolean,\n canonical: CanonicalTileID,\n layoutTextSize: number) {\n const lineArray = bucket.addToLineVertexArray(anchor, line);\n\n let textCollisionFeature, iconCollisionFeature, verticalTextCollisionFeature, verticalIconCollisionFeature;\n\n let numIconVertices = 0;\n let numVerticalIconVertices = 0;\n let numHorizontalGlyphVertices = 0;\n let numVerticalGlyphVertices = 0;\n let placedIconSymbolIndex = -1;\n let verticalPlacedIconSymbolIndex = -1;\n const placedTextSymbolIndices: {[k: string]: number} = {};\n let key = murmur3('');\n\n if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) {\n const textRotation = layer.layout.get('text-rotate').evaluate(feature, {}, canonical);\n const verticalTextRotation = textRotation + 90.0;\n const verticalShaping = shapedTextOrientations.vertical;\n verticalTextCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticalShaping, textBoxScale, textPadding, textAlongLine, verticalTextRotation);\n\n if (verticallyShapedIcon) {\n verticalIconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticallyShapedIcon, iconBoxScale, iconPadding, textAlongLine, verticalTextRotation);\n }\n }\n\n //Place icon first, so text can have a reference to its index in the placed symbol array.\n //Text symbols can lazily shift at render-time because of variable anchor placement.\n //If the style specifies an `icon-text-fit` then the icon would have to shift along with it.\n // For more info check `updateVariableAnchors` in `draw_symbol.js` .\n if (shapedIcon) {\n const iconRotate = layer.layout.get('icon-rotate').evaluate(feature, {});\n const hasIconTextFit = layer.layout.get('icon-text-fit') !== 'none';\n const iconQuads = getIconQuads(shapedIcon, iconRotate, isSDFIcon, hasIconTextFit);\n const verticalIconQuads = verticallyShapedIcon ? getIconQuads(verticallyShapedIcon, iconRotate, isSDFIcon, hasIconTextFit) : undefined;\n iconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shapedIcon, iconBoxScale, iconPadding, /*align boxes to line*/false, iconRotate);\n\n numIconVertices = iconQuads.length * 4;\n\n const sizeData = bucket.iconSizeData;\n let iconSizeData = null;\n\n if (sizeData.kind === 'source') {\n iconSizeData = [\n SIZE_PACK_FACTOR * layer.layout.get('icon-size').evaluate(feature, {})\n ];\n if (iconSizeData[0] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"icon-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"icon-size\".`);\n }\n } else if (sizeData.kind === 'composite') {\n iconSizeData = [\n SIZE_PACK_FACTOR * sizes.compositeIconSizes[0].evaluate(feature, {}, canonical),\n SIZE_PACK_FACTOR * sizes.compositeIconSizes[1].evaluate(feature, {}, canonical)\n ];\n if (iconSizeData[0] > MAX_PACKED_SIZE || iconSizeData[1] > MAX_PACKED_SIZE) {\n warnOnce(`${bucket.layerIds[0]}: Value for \"icon-size\" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your \"icon-size\".`);\n }\n }\n\n bucket.addSymbols(\n bucket.icon,\n iconQuads,\n iconSizeData,\n iconOffset,\n iconAlongLine,\n feature,\n WritingMode.none,\n anchor,\n lineArray.lineStartIndex,\n lineArray.lineLength,\n // The icon itself does not have an associated symbol since the text isnt placed yet\n -1, canonical);\n\n placedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1;\n\n if (verticalIconQuads) {\n numVerticalIconVertices = verticalIconQuads.length * 4;\n\n bucket.addSymbols(\n bucket.icon,\n verticalIconQuads,\n iconSizeData,\n iconOffset,\n iconAlongLine,\n feature,\n WritingMode.vertical,\n anchor,\n lineArray.lineStartIndex,\n lineArray.lineLength,\n // The icon itself does not have an associated symbol since the text isnt placed yet\n -1, canonical);\n\n verticalPlacedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1;\n }\n }\n\n const justifications = Object.keys(shapedTextOrientations.horizontal) as TextJustify[];\n for (const justification of justifications) {\n const shaping = shapedTextOrientations.horizontal[justification];\n\n if (!textCollisionFeature) {\n key = murmur3(shaping.text);\n const textRotate = layer.layout.get('text-rotate').evaluate(feature, {}, canonical);\n // As a collision approximation, we can use either the vertical or any of the horizontal versions of the feature\n // We're counting on all versions having similar dimensions\n textCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shaping, textBoxScale, textPadding, textAlongLine, textRotate);\n }\n\n const singleLine = shaping.positionedLines.length === 1;\n numHorizontalGlyphVertices += addTextVertices(\n bucket, anchor, shaping, imageMap, layer, textAlongLine, feature, textOffset, lineArray,\n shapedTextOrientations.vertical ? WritingMode.horizontal : WritingMode.horizontalOnly,\n singleLine ? justifications : [justification],\n placedTextSymbolIndices, placedIconSymbolIndex, sizes, canonical);\n\n if (singleLine) {\n break;\n }\n }\n\n if (shapedTextOrientations.vertical) {\n numVerticalGlyphVertices += addTextVertices(\n bucket, anchor, shapedTextOrientations.vertical, imageMap, layer, textAlongLine, feature,\n textOffset, lineArray, WritingMode.vertical, ['vertical'], placedTextSymbolIndices, verticalPlacedIconSymbolIndex, sizes, canonical);\n }\n\n const textBoxStartIndex = textCollisionFeature ? textCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const textBoxEndIndex = textCollisionFeature ? textCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n const verticalTextBoxStartIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const verticalTextBoxEndIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n const iconBoxStartIndex = iconCollisionFeature ? iconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const iconBoxEndIndex = iconCollisionFeature ? iconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n const verticalIconBoxStartIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length;\n const verticalIconBoxEndIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length;\n\n // Check if runtime collision circles should be used for any of the collision features.\n // It is enough to choose the tallest feature shape as circles are always placed on a line.\n // All measurements are in glyph metrics and later converted into pixels using proper font size \"layoutTextSize\"\n let collisionCircleDiameter = -1;\n\n const getCollisionCircleHeight = (feature: CollisionFeature, prevHeight: number): number => {\n if (feature && feature.circleDiameter)\n return Math.max(feature.circleDiameter, prevHeight);\n return prevHeight;\n };\n\n collisionCircleDiameter = getCollisionCircleHeight(textCollisionFeature, collisionCircleDiameter);\n collisionCircleDiameter = getCollisionCircleHeight(verticalTextCollisionFeature, collisionCircleDiameter);\n collisionCircleDiameter = getCollisionCircleHeight(iconCollisionFeature, collisionCircleDiameter);\n collisionCircleDiameter = getCollisionCircleHeight(verticalIconCollisionFeature, collisionCircleDiameter);\n const useRuntimeCollisionCircles = (collisionCircleDiameter > -1) ? 1 : 0;\n\n // Convert circle collision height into pixels\n if (useRuntimeCollisionCircles)\n collisionCircleDiameter *= layoutTextSize / ONE_EM;\n\n if (bucket.glyphOffsetArray.length >= SymbolBucket.MAX_GLYPHS) warnOnce(\n 'Too many glyphs being rendered in a tile. See https://github.com/mapbox/mapbox-gl-js/issues/2907'\n );\n\n if (feature.sortKey !== undefined) {\n bucket.addToSortKeyRanges(bucket.symbolInstances.length, feature.sortKey as number);\n }\n\n const variableAnchorOffset = getTextVariableAnchorOffset(layer, feature, canonical);\n const [textAnchorOffsetStartIndex, textAnchorOffsetEndIndex] = addTextVariableAnchorOffsets(bucket.textAnchorOffsets, variableAnchorOffset);\n\n bucket.symbolInstances.emplaceBack(\n anchor.x,\n anchor.y,\n placedTextSymbolIndices.right >= 0 ? placedTextSymbolIndices.right : -1,\n placedTextSymbolIndices.center >= 0 ? placedTextSymbolIndices.center : -1,\n placedTextSymbolIndices.left >= 0 ? placedTextSymbolIndices.left : -1,\n placedTextSymbolIndices.vertical || -1,\n placedIconSymbolIndex,\n verticalPlacedIconSymbolIndex,\n key,\n textBoxStartIndex,\n textBoxEndIndex,\n verticalTextBoxStartIndex,\n verticalTextBoxEndIndex,\n iconBoxStartIndex,\n iconBoxEndIndex,\n verticalIconBoxStartIndex,\n verticalIconBoxEndIndex,\n featureIndex,\n numHorizontalGlyphVertices,\n numVerticalGlyphVertices,\n numIconVertices,\n numVerticalIconVertices,\n useRuntimeCollisionCircles,\n 0,\n textBoxScale,\n collisionCircleDiameter,\n textAnchorOffsetStartIndex,\n textAnchorOffsetEndIndex);\n}\n\nfunction anchorIsTooClose(bucket: SymbolBucket, text: string, repeatDistance: number, anchor: Point) {\n const compareText = bucket.compareText;\n if (!(text in compareText)) {\n compareText[text] = [];\n } else {\n const otherAnchors = compareText[text];\n for (let k = otherAnchors.length - 1; k >= 0; k--) {\n if (anchor.dist(otherAnchors[k]) < repeatDistance) {\n // If it's within repeatDistance of one anchor, stop looking\n return true;\n }\n }\n }\n // If anchor is not within repeatDistance of any other anchor, add to array\n compareText[text].push(anchor);\n return false;\n}\n","\nconst ARRAY_TYPES = [\n Int8Array, Uint8Array, Uint8ClampedArray, Int16Array, Uint16Array,\n Int32Array, Uint32Array, Float32Array, Float64Array\n];\n\n/** @typedef {Int8ArrayConstructor | Uint8ArrayConstructor | Uint8ClampedArrayConstructor | Int16ArrayConstructor | Uint16ArrayConstructor | Int32ArrayConstructor | Uint32ArrayConstructor | Float32ArrayConstructor | Float64ArrayConstructor} TypedArrayConstructor */\n\nconst VERSION = 1; // serialized format version\nconst HEADER_SIZE = 8;\n\nexport default class KDBush {\n\n /**\n * Creates an index from raw `ArrayBuffer` data.\n * @param {ArrayBuffer} data\n */\n static from(data) {\n if (!(data instanceof ArrayBuffer)) {\n throw new Error('Data must be an instance of ArrayBuffer.');\n }\n const [magic, versionAndType] = new Uint8Array(data, 0, 2);\n if (magic !== 0xdb) {\n throw new Error('Data does not appear to be in a KDBush format.');\n }\n const version = versionAndType >> 4;\n if (version !== VERSION) {\n throw new Error(`Got v${version} data when expected v${VERSION}.`);\n }\n const ArrayType = ARRAY_TYPES[versionAndType & 0x0f];\n if (!ArrayType) {\n throw new Error('Unrecognized array type.');\n }\n const [nodeSize] = new Uint16Array(data, 2, 1);\n const [numItems] = new Uint32Array(data, 4, 1);\n\n return new KDBush(numItems, nodeSize, ArrayType, data);\n }\n\n /**\n * Creates an index that will hold a given number of items.\n * @param {number} numItems\n * @param {number} [nodeSize=64] Size of the KD-tree node (64 by default).\n * @param {TypedArrayConstructor} [ArrayType=Float64Array] The array type used for coordinates storage (`Float64Array` by default).\n * @param {ArrayBuffer} [data] (For internal use only)\n */\n constructor(numItems, nodeSize = 64, ArrayType = Float64Array, data) {\n if (isNaN(numItems) || numItems < 0) throw new Error(`Unpexpected numItems value: ${numItems}.`);\n\n this.numItems = +numItems;\n this.nodeSize = Math.min(Math.max(+nodeSize, 2), 65535);\n this.ArrayType = ArrayType;\n this.IndexArrayType = numItems < 65536 ? Uint16Array : Uint32Array;\n\n const arrayTypeIndex = ARRAY_TYPES.indexOf(this.ArrayType);\n const coordsByteSize = numItems * 2 * this.ArrayType.BYTES_PER_ELEMENT;\n const idsByteSize = numItems * this.IndexArrayType.BYTES_PER_ELEMENT;\n const padCoords = (8 - idsByteSize % 8) % 8;\n\n if (arrayTypeIndex < 0) {\n throw new Error(`Unexpected typed array class: ${ArrayType}.`);\n }\n\n if (data && (data instanceof ArrayBuffer)) { // reconstruct an index from a buffer\n this.data = data;\n this.ids = new this.IndexArrayType(this.data, HEADER_SIZE, numItems);\n this.coords = new this.ArrayType(this.data, HEADER_SIZE + idsByteSize + padCoords, numItems * 2);\n this._pos = numItems * 2;\n this._finished = true;\n } else { // initialize a new index\n this.data = new ArrayBuffer(HEADER_SIZE + coordsByteSize + idsByteSize + padCoords);\n this.ids = new this.IndexArrayType(this.data, HEADER_SIZE, numItems);\n this.coords = new this.ArrayType(this.data, HEADER_SIZE + idsByteSize + padCoords, numItems * 2);\n this._pos = 0;\n this._finished = false;\n\n // set header\n new Uint8Array(this.data, 0, 2).set([0xdb, (VERSION << 4) + arrayTypeIndex]);\n new Uint16Array(this.data, 2, 1)[0] = nodeSize;\n new Uint32Array(this.data, 4, 1)[0] = numItems;\n }\n }\n\n /**\n * Add a point to the index.\n * @param {number} x\n * @param {number} y\n * @returns {number} An incremental index associated with the added item (starting from `0`).\n */\n add(x, y) {\n const index = this._pos >> 1;\n this.ids[index] = index;\n this.coords[this._pos++] = x;\n this.coords[this._pos++] = y;\n return index;\n }\n\n /**\n * Perform indexing of the added points.\n */\n finish() {\n const numAdded = this._pos >> 1;\n if (numAdded !== this.numItems) {\n throw new Error(`Added ${numAdded} items when expected ${this.numItems}.`);\n }\n // kd-sort both arrays for efficient search\n sort(this.ids, this.coords, this.nodeSize, 0, this.numItems - 1, 0);\n\n this._finished = true;\n return this;\n }\n\n /**\n * Search the index for items within a given bounding box.\n * @param {number} minX\n * @param {number} minY\n * @param {number} maxX\n * @param {number} maxY\n * @returns {number[]} An array of indices correponding to the found items.\n */\n range(minX, minY, maxX, maxY) {\n if (!this._finished) throw new Error('Data not yet indexed - call index.finish().');\n\n const {ids, coords, nodeSize} = this;\n const stack = [0, ids.length - 1, 0];\n const result = [];\n\n // recursively search for items in range in the kd-sorted arrays\n while (stack.length) {\n const axis = stack.pop() || 0;\n const right = stack.pop() || 0;\n const left = stack.pop() || 0;\n\n // if we reached \"tree node\", search linearly\n if (right - left <= nodeSize) {\n for (let i = left; i <= right; i++) {\n const x = coords[2 * i];\n const y = coords[2 * i + 1];\n if (x >= minX && x <= maxX && y >= minY && y <= maxY) result.push(ids[i]);\n }\n continue;\n }\n\n // otherwise find the middle index\n const m = (left + right) >> 1;\n\n // include the middle item if it's in range\n const x = coords[2 * m];\n const y = coords[2 * m + 1];\n if (x >= minX && x <= maxX && y >= minY && y <= maxY) result.push(ids[m]);\n\n // queue search in halves that intersect the query\n if (axis === 0 ? minX <= x : minY <= y) {\n stack.push(left);\n stack.push(m - 1);\n stack.push(1 - axis);\n }\n if (axis === 0 ? maxX >= x : maxY >= y) {\n stack.push(m + 1);\n stack.push(right);\n stack.push(1 - axis);\n }\n }\n\n return result;\n }\n\n /**\n * Search the index for items within a given radius.\n * @param {number} qx\n * @param {number} qy\n * @param {number} r Query radius.\n * @returns {number[]} An array of indices correponding to the found items.\n */\n within(qx, qy, r) {\n if (!this._finished) throw new Error('Data not yet indexed - call index.finish().');\n\n const {ids, coords, nodeSize} = this;\n const stack = [0, ids.length - 1, 0];\n const result = [];\n const r2 = r * r;\n\n // recursively search for items within radius in the kd-sorted arrays\n while (stack.length) {\n const axis = stack.pop() || 0;\n const right = stack.pop() || 0;\n const left = stack.pop() || 0;\n\n // if we reached \"tree node\", search linearly\n if (right - left <= nodeSize) {\n for (let i = left; i <= right; i++) {\n if (sqDist(coords[2 * i], coords[2 * i + 1], qx, qy) <= r2) result.push(ids[i]);\n }\n continue;\n }\n\n // otherwise find the middle index\n const m = (left + right) >> 1;\n\n // include the middle item if it's in range\n const x = coords[2 * m];\n const y = coords[2 * m + 1];\n if (sqDist(x, y, qx, qy) <= r2) result.push(ids[m]);\n\n // queue search in halves that intersect the query\n if (axis === 0 ? qx - r <= x : qy - r <= y) {\n stack.push(left);\n stack.push(m - 1);\n stack.push(1 - axis);\n }\n if (axis === 0 ? qx + r >= x : qy + r >= y) {\n stack.push(m + 1);\n stack.push(right);\n stack.push(1 - axis);\n }\n }\n\n return result;\n }\n}\n\n/**\n * @param {Uint16Array | Uint32Array} ids\n * @param {InstanceType} coords\n * @param {number} nodeSize\n * @param {number} left\n * @param {number} right\n * @param {number} axis\n */\nfunction sort(ids, coords, nodeSize, left, right, axis) {\n if (right - left <= nodeSize) return;\n\n const m = (left + right) >> 1; // middle index\n\n // sort ids and coords around the middle index so that the halves lie\n // either left/right or top/bottom correspondingly (taking turns)\n select(ids, coords, m, left, right, axis);\n\n // recursively kd-sort first half and second half on the opposite axis\n sort(ids, coords, nodeSize, left, m - 1, 1 - axis);\n sort(ids, coords, nodeSize, m + 1, right, 1 - axis);\n}\n\n/**\n * Custom Floyd-Rivest selection algorithm: sort ids and coords so that\n * [left..k-1] items are smaller than k-th item (on either x or y axis)\n * @param {Uint16Array | Uint32Array} ids\n * @param {InstanceType} coords\n * @param {number} k\n * @param {number} left\n * @param {number} right\n * @param {number} axis\n */\nfunction select(ids, coords, k, left, right, axis) {\n\n while (right > left) {\n if (right - left > 600) {\n const n = right - left + 1;\n const m = k - left + 1;\n const z = Math.log(n);\n const s = 0.5 * Math.exp(2 * z / 3);\n const sd = 0.5 * Math.sqrt(z * s * (n - s) / n) * (m - n / 2 < 0 ? -1 : 1);\n const newLeft = Math.max(left, Math.floor(k - m * s / n + sd));\n const newRight = Math.min(right, Math.floor(k + (n - m) * s / n + sd));\n select(ids, coords, k, newLeft, newRight, axis);\n }\n\n const t = coords[2 * k + axis];\n let i = left;\n let j = right;\n\n swapItem(ids, coords, left, k);\n if (coords[2 * right + axis] > t) swapItem(ids, coords, left, right);\n\n while (i < j) {\n swapItem(ids, coords, i, j);\n i++;\n j--;\n while (coords[2 * i + axis] < t) i++;\n while (coords[2 * j + axis] > t) j--;\n }\n\n if (coords[2 * left + axis] === t) swapItem(ids, coords, left, j);\n else {\n j++;\n swapItem(ids, coords, j, right);\n }\n\n if (j <= k) left = j + 1;\n if (k <= j) right = j - 1;\n }\n}\n\n/**\n * @param {Uint16Array | Uint32Array} ids\n * @param {InstanceType} coords\n * @param {number} i\n * @param {number} j\n */\nfunction swapItem(ids, coords, i, j) {\n swap(ids, i, j);\n swap(coords, 2 * i, 2 * j);\n swap(coords, 2 * i + 1, 2 * j + 1);\n}\n\n/**\n * @param {InstanceType} arr\n * @param {number} i\n * @param {number} j\n */\nfunction swap(arr, i, j) {\n const tmp = arr[i];\n arr[i] = arr[j];\n arr[j] = tmp;\n}\n\n/**\n * @param {number} ax\n * @param {number} ay\n * @param {number} bx\n * @param {number} by\n */\nfunction sqDist(ax, ay, bx, by) {\n const dx = ax - bx;\n const dy = ay - by;\n return dx * dx + dy * dy;\n}\n","import type {RequestParameters} from '../util/ajax';\n\nexport type PerformanceMetrics = {\n loadTime: number;\n fullLoadTime: number;\n fps: number;\n percentDroppedFrames: number;\n totalFrames: number;\n};\n\nexport enum PerformanceMarkers {\n create = 'create',\n load = 'load',\n fullLoad = 'fullLoad'\n}\n\nlet lastFrameTime = null;\nlet frameTimes = [];\n\nconst minFramerateTarget = 60;\nconst frameTimeTarget = 1000 / minFramerateTarget;\n\nconst loadTimeKey = 'loadTime';\nconst fullLoadTimeKey = 'fullLoadTime';\n\nexport const PerformanceUtils = {\n mark(marker: PerformanceMarkers) {\n performance.mark(marker);\n },\n frame(timestamp: number) {\n const currTimestamp = timestamp;\n if (lastFrameTime != null) {\n const frameTime = currTimestamp - lastFrameTime;\n frameTimes.push(frameTime);\n }\n lastFrameTime = currTimestamp;\n },\n clearMetrics() {\n lastFrameTime = null;\n frameTimes = [];\n performance.clearMeasures(loadTimeKey);\n performance.clearMeasures(fullLoadTimeKey);\n\n for (const marker in PerformanceMarkers) {\n performance.clearMarks(PerformanceMarkers[marker]);\n }\n },\n\n getPerformanceMetrics(): PerformanceMetrics {\n performance.measure(loadTimeKey, PerformanceMarkers.create, PerformanceMarkers.load);\n performance.measure(fullLoadTimeKey, PerformanceMarkers.create, PerformanceMarkers.fullLoad);\n const loadTime = performance.getEntriesByName(loadTimeKey)[0].duration;\n const fullLoadTime = performance.getEntriesByName(fullLoadTimeKey)[0].duration;\n const totalFrames = frameTimes.length;\n\n const avgFrameTime = frameTimes.reduce((prev, curr) => prev + curr, 0) / totalFrames / 1000;\n const fps = 1 / avgFrameTime;\n\n // count frames that missed our framerate target\n const droppedFrames = frameTimes\n .filter((frameTime) => frameTime > frameTimeTarget)\n .reduce((acc, curr) => {\n return acc + (curr - frameTimeTarget) / frameTimeTarget;\n }, 0);\n const percentDroppedFrames = (droppedFrames / (totalFrames + droppedFrames)) * 100;\n\n return {\n loadTime,\n fullLoadTime,\n fps,\n percentDroppedFrames,\n totalFrames\n };\n }\n};\n\n/**\n * @internal\n * Safe wrapper for the performance resource timing API in web workers with graceful degradation\n */\nexport class RequestPerformance {\n _marks: {\n start: string;\n end: string;\n measure: string;\n };\n\n constructor (request: RequestParameters) {\n this._marks = {\n start: [request.url, 'start'].join('#'),\n end: [request.url, 'end'].join('#'),\n measure: request.url.toString()\n };\n\n performance.mark(this._marks.start);\n }\n\n finish() {\n performance.mark(this._marks.end);\n let resourceTimingData = performance.getEntriesByName(this._marks.measure);\n\n // fallback if web worker implementation of perf.getEntriesByName returns empty\n if (resourceTimingData.length === 0) {\n performance.measure(this._marks.measure, this._marks.start, this._marks.end);\n resourceTimingData = performance.getEntriesByName(this._marks.measure);\n\n // cleanup\n performance.clearMarks(this._marks.start);\n performance.clearMarks(this._marks.end);\n performance.clearMeasures(this._marks.measure);\n }\n\n return resourceTimingData;\n }\n}\n\nexport default performance;\n","import {isWorker} from './util';\nimport {serialize, deserialize, Serialized} from './web_worker_transfer';\nimport {ThrottledInvoker} from './throttled_invoker';\n\nimport type {Transferable} from '../types/transferable';\nimport type {Cancelable} from '../types/cancelable';\nimport type {WorkerSource} from '../source/worker_source';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {Callback} from '../types/callback';\nimport type {StyleGlyph} from '../style/style_glyph';\n\nexport interface ActorTarget {\n addEventListener: typeof window.addEventListener;\n removeEventListener: typeof window.removeEventListener;\n postMessage: typeof window.postMessage;\n terminate?: () => void;\n}\n\nexport interface WorkerSourceProvider {\n getWorkerSource(mapId: string | number, sourceType: string, sourceName: string): WorkerSource;\n}\n\nexport interface GlyphsProvider {\n getGlyphs(mapId: string, params: {\n stacks: {[_: string]: Array};\n source: string;\n tileID: OverscaledTileID;\n type: string;\n },\n callback: Callback<{[_: string]: {[_: number]: StyleGlyph}}>\n );\n}\n\nexport type MessageType = '' | '' |\n'geojson.getClusterExpansionZoom' | 'geojson.getClusterChildren' | 'geojson.getClusterLeaves' | 'geojson.loadData' |\n'removeSource' | 'loadWorkerSource' | 'loadDEMTile' | 'removeDEMTile' |\n'removeTile' | 'reloadTile' | 'abortTile' | 'loadTile' | 'getTile' |\n'getGlyphs' | 'getImages' | 'setImages' |\n'syncRTLPluginState' | 'setReferrer' | 'setLayers' | 'updateLayers';\n\nexport type MessageData = {\n id: string;\n type: MessageType;\n data?: Serialized;\n targetMapId?: string | number | null;\n mustQueue?: boolean;\n error?: Serialized | null;\n hasCallback?: boolean;\n sourceMapId: string | number | null;\n}\n\nexport type Message = {\n data: MessageData;\n}\n\n/**\n * An implementation of the [Actor design pattern](http://en.wikipedia.org/wiki/Actor_model)\n * that maintains the relationship between asynchronous tasks and the objects\n * that spin them off - in this case, tasks like parsing parts of styles,\n * owned by the styles\n */\nexport class Actor {\n target: ActorTarget;\n parent: WorkerSourceProvider | GlyphsProvider;\n mapId: string | number | null;\n callbacks: { [x: number]: Function};\n name: string;\n tasks: { [x: number]: MessageData };\n taskQueue: Array;\n cancelCallbacks: { [x: number]: () => void };\n invoker: ThrottledInvoker;\n globalScope: ActorTarget;\n\n /**\n * @param target - The target\n * @param parent - The parent\n * @param mapId - A unique identifier for the Map instance using this Actor.\n */\n constructor(target: ActorTarget, parent: WorkerSourceProvider | GlyphsProvider, mapId?: string | number) {\n this.target = target;\n this.parent = parent;\n this.mapId = mapId;\n this.callbacks = {};\n this.tasks = {};\n this.taskQueue = [];\n this.cancelCallbacks = {};\n this.invoker = new ThrottledInvoker(this.process);\n this.target.addEventListener('message', this.receive, false);\n this.globalScope = isWorker() ? target : window;\n }\n\n /**\n * Sends a message from a main-thread map to a Worker or from a Worker back to\n * a main-thread map instance.\n *\n * @param type - The name of the target method to invoke or '[source-type].[source-name].name' for a method on a WorkerSource.\n * @param targetMapId - A particular mapId to which to send this message.\n */\n send(\n type: MessageType,\n data: unknown,\n callback?: Function | null,\n targetMapId?: string | null,\n mustQueue: boolean = false\n ): Cancelable {\n // We're using a string ID instead of numbers because they are being used as object keys\n // anyway, and thus stringified implicitly. We use random IDs because an actor may receive\n // message from multiple other actors which could run in different execution context. A\n // linearly increasing ID could produce collisions.\n const id = Math.round((Math.random() * 1e18)).toString(36).substring(0, 10);\n if (callback) {\n this.callbacks[id] = callback;\n }\n const buffers: Array = [];\n const message: MessageData = {\n id,\n type,\n hasCallback: !!callback,\n targetMapId,\n mustQueue,\n sourceMapId: this.mapId,\n data: serialize(data, buffers)\n };\n\n this.target.postMessage(message, {transfer: buffers});\n return {\n cancel: () => {\n if (callback) {\n // Set the callback to null so that it never fires after the request is aborted.\n delete this.callbacks[id];\n }\n const cancelMessage: MessageData = {\n id,\n type: '',\n targetMapId,\n sourceMapId: this.mapId\n };\n this.target.postMessage(cancelMessage);\n }\n };\n }\n\n receive = (message: Message) => {\n const data = message.data;\n const id = data.id;\n\n if (!id) {\n return;\n }\n\n if (data.targetMapId && this.mapId !== data.targetMapId) {\n return;\n }\n\n if (data.type === '') {\n // Remove the original request from the queue. This is only possible if it\n // hasn't been kicked off yet. The id will remain in the queue, but because\n // there is no associated task, it will be dropped once it's time to execute it.\n delete this.tasks[id];\n const cancel = this.cancelCallbacks[id];\n delete this.cancelCallbacks[id];\n if (cancel) {\n cancel();\n }\n } else {\n if (isWorker() || data.mustQueue) {\n // In workers, store the tasks that we need to process before actually processing them. This\n // is necessary because we want to keep receiving messages, and in particular,\n // messages. Some tasks may take a while in the worker thread, so before\n // executing the next task in our queue, postMessage preempts this and \n // messages can be processed. We're using a MessageChannel object to get throttle the\n // process() flow to one at a time.\n this.tasks[id] = data;\n this.taskQueue.push(id);\n this.invoker.trigger();\n } else {\n // In the main thread, process messages immediately so that other work does not slip in\n // between getting partial data back from workers.\n this.processTask(id, data);\n }\n }\n };\n\n process = () => {\n if (!this.taskQueue.length) {\n return;\n }\n const id = this.taskQueue.shift();\n const task = this.tasks[id];\n delete this.tasks[id];\n // Schedule another process call if we know there's more to process _before_ invoking the\n // current task. This is necessary so that processing continues even if the current task\n // doesn't execute successfully.\n if (this.taskQueue.length) {\n this.invoker.trigger();\n }\n if (!task) {\n // If the task ID doesn't have associated task data anymore, it was canceled.\n return;\n }\n\n this.processTask(id, task);\n };\n\n processTask(id: string, task: MessageData) {\n if (task.type === '') {\n // The done() function in the counterpart has been called, and we are now\n // firing the callback in the originating actor, if there is one.\n const callback = this.callbacks[id];\n delete this.callbacks[id];\n if (callback) {\n // If we get a response, but don't have a callback, the request was canceled.\n if (task.error) {\n callback(deserialize(task.error));\n } else {\n callback(null, deserialize(task.data));\n }\n }\n } else {\n let completed = false;\n const buffers: Array = [];\n const done = task.hasCallback ? (err: Error, data?: any) => {\n completed = true;\n delete this.cancelCallbacks[id];\n const responseMessage: MessageData = {\n id,\n type: '',\n sourceMapId: this.mapId,\n error: err ? serialize(err) : null,\n data: serialize(data, buffers)\n };\n this.target.postMessage(responseMessage, {transfer: buffers});\n } : (_) => {\n completed = true;\n };\n\n let callback: Cancelable = null;\n const params = deserialize(task.data);\n if (this.parent[task.type]) {\n // task.type == 'loadTile', 'removeTile', etc.\n callback = this.parent[task.type](task.sourceMapId, params, done);\n } else if ('getWorkerSource' in this.parent) {\n // task.type == sourcetype.method\n const keys = task.type.split('.');\n const scope = this.parent.getWorkerSource(task.sourceMapId, keys[0], (params as any).source);\n callback = scope[keys[1]](params, done);\n } else {\n // No function was found.\n done(new Error(`Could not find function ${task.type}`));\n }\n\n if (!completed && callback && callback.cancel) {\n // Allows canceling the task as long as it hasn't been completed yet.\n this.cancelCallbacks[id] = callback.cancel;\n }\n }\n }\n\n remove() {\n this.invoker.remove();\n this.target.removeEventListener('message', this.receive, false);\n }\n}\n","import {CircleStyleLayer} from './style_layer/circle_style_layer';\nimport {HeatmapStyleLayer} from './style_layer/heatmap_style_layer';\nimport {HillshadeStyleLayer} from './style_layer/hillshade_style_layer';\nimport {FillStyleLayer} from './style_layer/fill_style_layer';\nimport {FillExtrusionStyleLayer} from './style_layer/fill_extrusion_style_layer';\nimport {LineStyleLayer} from './style_layer/line_style_layer';\nimport {SymbolStyleLayer} from './style_layer/symbol_style_layer';\nimport {BackgroundStyleLayer} from './style_layer/background_style_layer';\nimport {RasterStyleLayer} from './style_layer/raster_style_layer';\nimport {CustomStyleLayer, type CustomLayerInterface} from './style_layer/custom_style_layer';\n\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport function createStyleLayer(layer: LayerSpecification | CustomLayerInterface) {\n if (layer.type === 'custom') {\n return new CustomStyleLayer(layer);\n }\n switch (layer.type) {\n case 'background':\n return new BackgroundStyleLayer(layer);\n case 'circle':\n return new CircleStyleLayer(layer);\n case 'fill':\n return new FillStyleLayer(layer);\n case 'fill-extrusion':\n return new FillExtrusionStyleLayer(layer);\n case 'heatmap':\n return new HeatmapStyleLayer(layer);\n case 'hillshade':\n return new HillshadeStyleLayer(layer);\n case 'line':\n return new LineStyleLayer(layer);\n case 'raster':\n return new RasterStyleLayer(layer);\n case 'symbol':\n return new SymbolStyleLayer(layer);\n }\n}\n\n","import {StyleLayer} from './style_layer';\nimport {createStyleLayer} from './create_style_layer';\n\nimport {featureFilter, groupByLayout} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {TypedStyleLayer} from './style_layer/typed_style_layer';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type LayerConfigs = {[_: string]: LayerSpecification};\nexport type Family = Array;\n\nexport class StyleLayerIndex {\n familiesBySource: {\n [source: string]: {\n [sourceLayer: string]: Array>;\n };\n };\n keyCache: {[source: string]: string};\n\n _layerConfigs: LayerConfigs;\n _layers: {[_: string]: StyleLayer};\n\n constructor(layerConfigs?: Array | null) {\n this.keyCache = {};\n if (layerConfigs) {\n this.replace(layerConfigs);\n }\n }\n\n replace(layerConfigs: Array) {\n this._layerConfigs = {};\n this._layers = {};\n this.update(layerConfigs, []);\n }\n\n update(layerConfigs: Array, removedIds: Array) {\n for (const layerConfig of layerConfigs) {\n this._layerConfigs[layerConfig.id] = layerConfig;\n\n const layer = this._layers[layerConfig.id] = createStyleLayer(layerConfig);\n layer._featureFilter = featureFilter(layer.filter);\n if (this.keyCache[layerConfig.id])\n delete this.keyCache[layerConfig.id];\n }\n for (const id of removedIds) {\n delete this.keyCache[id];\n delete this._layerConfigs[id];\n delete this._layers[id];\n }\n\n this.familiesBySource = {};\n\n const groups = groupByLayout(Object.values(this._layerConfigs), this.keyCache);\n\n for (const layerConfigs of groups) {\n const layers = layerConfigs.map((layerConfig) => this._layers[layerConfig.id]);\n\n const layer = layers[0];\n if (layer.visibility === 'none') {\n continue;\n }\n\n const sourceId = layer.source || '';\n let sourceGroup = this.familiesBySource[sourceId];\n if (!sourceGroup) {\n sourceGroup = this.familiesBySource[sourceId] = {};\n }\n\n const sourceLayerId = layer.sourceLayer || '_geojsonTileLayer';\n let sourceLayerFamilies = sourceGroup[sourceLayerId];\n if (!sourceLayerFamilies) {\n sourceLayerFamilies = sourceGroup[sourceLayerId] = [];\n }\n\n sourceLayerFamilies.push(layers);\n }\n }\n}\n","import {AlphaImage} from '../util/image';\nimport {register} from '../util/web_worker_transfer';\nimport potpack from 'potpack';\n\nimport type {GlyphMetrics, StyleGlyph} from '../style/style_glyph';\n\nconst padding = 1;\n\n/**\n * A rectangle type with postion, width and height.\n */\nexport type Rect = {\n x: number;\n y: number;\n w: number;\n h: number;\n};\n\n/**\n * The glyph's position\n */\nexport type GlyphPosition = {\n rect: Rect;\n metrics: GlyphMetrics;\n};\n\n/**\n * The glyphs' positions\n */\nexport type GlyphPositions = {\n [_: string]: {\n [_: number]: GlyphPosition;\n };\n};\n\nexport class GlyphAtlas {\n image: AlphaImage;\n positions: GlyphPositions;\n\n constructor(stacks: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n }) {\n const positions = {};\n const bins = [];\n\n for (const stack in stacks) {\n const glyphs = stacks[stack];\n const stackPositions = positions[stack] = {};\n\n for (const id in glyphs) {\n const src = glyphs[+id];\n if (!src || src.bitmap.width === 0 || src.bitmap.height === 0) continue;\n\n const bin = {\n x: 0,\n y: 0,\n w: src.bitmap.width + 2 * padding,\n h: src.bitmap.height + 2 * padding\n };\n bins.push(bin);\n stackPositions[id] = {rect: bin, metrics: src.metrics};\n }\n }\n\n const {w, h} = potpack(bins);\n const image = new AlphaImage({width: w || 1, height: h || 1});\n\n for (const stack in stacks) {\n const glyphs = stacks[stack];\n\n for (const id in glyphs) {\n const src = glyphs[+id];\n if (!src || src.bitmap.width === 0 || src.bitmap.height === 0) continue;\n const bin = positions[stack][id].rect;\n AlphaImage.copy(src.bitmap, image, {x: 0, y: 0}, {x: bin.x + padding, y: bin.y + padding}, src.bitmap);\n }\n }\n\n this.image = image;\n this.positions = positions;\n }\n}\n\nregister('GlyphAtlas', GlyphAtlas);\n","import {FeatureIndex} from '../data/feature_index';\nimport {performSymbolLayout} from '../symbol/symbol_layout';\nimport {CollisionBoxArray} from '../data/array_types.g';\nimport {DictionaryCoder} from '../util/dictionary_coder';\nimport {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {LineBucket} from '../data/bucket/line_bucket';\nimport {FillBucket} from '../data/bucket/fill_bucket';\nimport {FillExtrusionBucket} from '../data/bucket/fill_extrusion_bucket';\nimport {warnOnce, mapObject} from '../util/util';\nimport {ImageAtlas} from '../render/image_atlas';\nimport {GlyphAtlas} from '../render/glyph_atlas';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {OverscaledTileID} from './tile_id';\n\nimport type {Bucket} from '../data/bucket';\nimport type {Actor} from '../util/actor';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {StyleLayerIndex} from '../style/style_layer_index';\nimport type {StyleImage} from '../style/style_image';\nimport type {StyleGlyph} from '../style/style_glyph';\nimport type {\n WorkerTileParameters,\n WorkerTileCallback,\n} from '../source/worker_source';\nimport type {PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {VectorTile} from '@mapbox/vector-tile';\nimport {Cancelable} from '../types/cancelable';\n\nexport class WorkerTile {\n tileID: OverscaledTileID;\n uid: string;\n zoom: number;\n pixelRatio: number;\n tileSize: number;\n source: string;\n promoteId: PromoteIdSpecification;\n overscaling: number;\n showCollisionBoxes: boolean;\n collectResourceTiming: boolean;\n returnDependencies: boolean;\n\n status: 'parsing' | 'done';\n data: VectorTile;\n collisionBoxArray: CollisionBoxArray;\n\n abort: (() => void);\n vectorTile: VectorTile;\n inFlightDependencies: Cancelable[];\n dependencySentinel: number;\n\n constructor(params: WorkerTileParameters) {\n this.tileID = new OverscaledTileID(params.tileID.overscaledZ, params.tileID.wrap, params.tileID.canonical.z, params.tileID.canonical.x, params.tileID.canonical.y);\n this.uid = params.uid;\n this.zoom = params.zoom;\n this.pixelRatio = params.pixelRatio;\n this.tileSize = params.tileSize;\n this.source = params.source;\n this.overscaling = this.tileID.overscaleFactor();\n this.showCollisionBoxes = params.showCollisionBoxes;\n this.collectResourceTiming = !!params.collectResourceTiming;\n this.returnDependencies = !!params.returnDependencies;\n this.promoteId = params.promoteId;\n this.inFlightDependencies = [];\n this.dependencySentinel = -1;\n }\n\n parse(data: VectorTile, layerIndex: StyleLayerIndex, availableImages: Array, actor: Actor, callback: WorkerTileCallback) {\n this.status = 'parsing';\n this.data = data;\n\n this.collisionBoxArray = new CollisionBoxArray();\n const sourceLayerCoder = new DictionaryCoder(Object.keys(data.layers).sort());\n\n const featureIndex = new FeatureIndex(this.tileID, this.promoteId);\n featureIndex.bucketLayerIDs = [];\n\n const buckets: {[_: string]: Bucket} = {};\n\n const options = {\n featureIndex,\n iconDependencies: {},\n patternDependencies: {},\n glyphDependencies: {},\n availableImages\n };\n\n const layerFamilies = layerIndex.familiesBySource[this.source];\n for (const sourceLayerId in layerFamilies) {\n const sourceLayer = data.layers[sourceLayerId];\n if (!sourceLayer) {\n continue;\n }\n\n if (sourceLayer.version === 1) {\n warnOnce(`Vector tile source \"${this.source}\" layer \"${sourceLayerId}\" ` +\n 'does not use vector tile spec v2 and therefore may have some rendering errors.');\n }\n\n const sourceLayerIndex = sourceLayerCoder.encode(sourceLayerId);\n const features = [];\n for (let index = 0; index < sourceLayer.length; index++) {\n const feature = sourceLayer.feature(index);\n const id = featureIndex.getId(feature, sourceLayerId);\n features.push({feature, id, index, sourceLayerIndex});\n }\n\n for (const family of layerFamilies[sourceLayerId]) {\n const layer = family[0];\n\n if (layer.source !== this.source) {\n warnOnce(`layer.source = ${layer.source} does not equal this.source = ${this.source}`);\n }\n if (layer.minzoom && this.zoom < Math.floor(layer.minzoom)) continue;\n if (layer.maxzoom && this.zoom >= layer.maxzoom) continue;\n if (layer.visibility === 'none') continue;\n\n recalculateLayers(family, this.zoom, availableImages);\n\n const bucket = buckets[layer.id] = layer.createBucket({\n index: featureIndex.bucketLayerIDs.length,\n layers: family,\n zoom: this.zoom,\n pixelRatio: this.pixelRatio,\n overscaling: this.overscaling,\n collisionBoxArray: this.collisionBoxArray,\n sourceLayerIndex,\n sourceID: this.source\n });\n\n bucket.populate(features, options, this.tileID.canonical);\n featureIndex.bucketLayerIDs.push(family.map((l) => l.id));\n }\n }\n\n let error: Error;\n let glyphMap: {\n [_: string]: {\n [_: number]: StyleGlyph;\n };\n };\n let iconMap: {[_: string]: StyleImage};\n let patternMap: {[_: string]: StyleImage};\n\n const stacks = mapObject(options.glyphDependencies, (glyphs) => Object.keys(glyphs).map(Number));\n\n this.inFlightDependencies.forEach((request) => request?.cancel());\n this.inFlightDependencies = [];\n\n // cancelling seems to be not sufficient, we seems to still manage to get a callback hit, so use a sentinel to drop stale results\n const dependencySentinel = ++this.dependencySentinel;\n if (Object.keys(stacks).length) {\n this.inFlightDependencies.push(actor.send('getGlyphs', {uid: this.uid, stacks, source: this.source, tileID: this.tileID, type: 'glyphs'}, (err, result) => {\n if (dependencySentinel !== this.dependencySentinel) {\n return;\n }\n if (!error) {\n error = err;\n glyphMap = result;\n maybePrepare.call(this);\n }\n }));\n } else {\n glyphMap = {};\n }\n\n const icons = Object.keys(options.iconDependencies);\n if (icons.length) {\n this.inFlightDependencies.push(actor.send('getImages', {icons, source: this.source, tileID: this.tileID, type: 'icons'}, (err, result) => {\n if (dependencySentinel !== this.dependencySentinel) {\n return;\n }\n if (!error) {\n error = err;\n iconMap = result;\n maybePrepare.call(this);\n }\n }));\n } else {\n iconMap = {};\n }\n\n const patterns = Object.keys(options.patternDependencies);\n if (patterns.length) {\n this.inFlightDependencies.push(actor.send('getImages', {icons: patterns, source: this.source, tileID: this.tileID, type: 'patterns'}, (err, result) => {\n if (dependencySentinel !== this.dependencySentinel) {\n return;\n }\n if (!error) {\n error = err;\n patternMap = result;\n maybePrepare.call(this);\n }\n }));\n } else {\n patternMap = {};\n }\n\n maybePrepare.call(this);\n\n function maybePrepare() {\n if (error) {\n return callback(error);\n } else if (glyphMap && iconMap && patternMap) {\n const glyphAtlas = new GlyphAtlas(glyphMap);\n const imageAtlas = new ImageAtlas(iconMap, patternMap);\n\n for (const key in buckets) {\n const bucket = buckets[key];\n if (bucket instanceof SymbolBucket) {\n recalculateLayers(bucket.layers, this.zoom, availableImages);\n performSymbolLayout({\n bucket,\n glyphMap,\n glyphPositions: glyphAtlas.positions,\n imageMap: iconMap,\n imagePositions: imageAtlas.iconPositions,\n showCollisionBoxes: this.showCollisionBoxes,\n canonical: this.tileID.canonical\n });\n } else if (bucket.hasPattern &&\n (bucket instanceof LineBucket ||\n bucket instanceof FillBucket ||\n bucket instanceof FillExtrusionBucket)) {\n recalculateLayers(bucket.layers, this.zoom, availableImages);\n bucket.addFeatures(options, this.tileID.canonical, imageAtlas.patternPositions);\n }\n }\n\n this.status = 'done';\n callback(null, {\n buckets: Object.values(buckets).filter(b => !b.isEmpty()),\n featureIndex,\n collisionBoxArray: this.collisionBoxArray,\n glyphAtlasImage: glyphAtlas.image,\n imageAtlas,\n // Only used for benchmarking:\n glyphMap: this.returnDependencies ? glyphMap : null,\n iconMap: this.returnDependencies ? iconMap : null,\n glyphPositions: this.returnDependencies ? glyphAtlas.positions : null\n });\n }\n }\n }\n}\n\nfunction recalculateLayers(layers: ReadonlyArray, zoom: number, availableImages: Array) {\n // Layers are shared and may have been used by a WorkerTile with a different zoom.\n const parameters = new EvaluationParameters(zoom);\n for (const layer of layers) {\n layer.recalculate(parameters, availableImages);\n }\n}\n","import {ExpiryData, getArrayBuffer} from '../util/ajax';\n\nimport vt from '@mapbox/vector-tile';\nimport Protobuf from 'pbf';\nimport {WorkerTile} from './worker_tile';\nimport {extend} from '../util/util';\nimport {RequestPerformance} from '../util/performance';\n\nimport type {\n WorkerSource,\n WorkerTileParameters,\n WorkerTileCallback,\n TileParameters\n} from '../source/worker_source';\n\nimport type {Actor} from '../util/actor';\nimport type {StyleLayerIndex} from '../style/style_layer_index';\nimport type {Callback} from '../types/callback';\nimport type {VectorTile} from '@mapbox/vector-tile';\n\nexport type LoadVectorTileResult = {\n vectorTile: VectorTile;\n rawData: ArrayBuffer;\n resourceTiming?: Array;\n} & ExpiryData;\n\ntype FetchingState = {\n rawTileData: ArrayBuffer;\n cacheControl: ExpiryData;\n resourceTiming: any;\n}\n\n/**\n * The callback when finished loading vector data\n */\nexport type LoadVectorDataCallback = Callback;\n\nexport type AbortVectorData = () => void;\nexport type LoadVectorData = (params: WorkerTileParameters, callback: LoadVectorDataCallback) => AbortVectorData | void;\n\n/**\n * Loads a vector tile\n */\nfunction loadVectorTile(params: WorkerTileParameters, callback: LoadVectorDataCallback) {\n const request = getArrayBuffer(params.request, (err?: Error | null, data?: ArrayBuffer | null, cacheControl?: string | null, expires?: string | null) => {\n if (err) {\n callback(err);\n } else if (data) {\n try {\n const vectorTile = new vt.VectorTile(new Protobuf(data));\n callback(null, {\n vectorTile,\n rawData: data,\n cacheControl,\n expires\n });\n } catch (ex) {\n const bytes = new Uint8Array(data);\n const isGzipped = bytes[0] === 0x1f && bytes[1] === 0x8b;\n let errorMessage = `Unable to parse the tile at ${params.request.url}, `;\n if (isGzipped) {\n errorMessage += 'please make sure the data is not gzipped and that you have configured the relevant header in the server';\n } else {\n errorMessage += `got error: ${ex.messge}`;\n }\n callback(new Error(errorMessage));\n }\n }\n });\n return () => {\n request.cancel();\n callback();\n };\n}\n\n/**\n * The {@link WorkerSource} implementation that supports {@link VectorTileSource}.\n * This class is designed to be easily reused to support custom source types\n * for data formats that can be parsed/converted into an in-memory VectorTile\n * representation. To do so, create it with\n * `new VectorTileWorkerSource(actor, styleLayers, customLoadVectorDataFunction)`.\n */\nexport class VectorTileWorkerSource implements WorkerSource {\n actor: Actor;\n layerIndex: StyleLayerIndex;\n availableImages: Array;\n loadVectorData: LoadVectorData;\n fetching: {[_: string]: FetchingState };\n loading: {[_: string]: WorkerTile};\n loaded: {[_: string]: WorkerTile};\n\n /**\n * @param loadVectorData - Optional method for custom loading of a VectorTile\n * object based on parameters passed from the main-thread Source. See\n * {@link VectorTileWorkerSource#loadTile}. The default implementation simply\n * loads the pbf at `params.url`.\n */\n constructor(actor: Actor, layerIndex: StyleLayerIndex, availableImages: Array, loadVectorData?: LoadVectorData | null) {\n this.actor = actor;\n this.layerIndex = layerIndex;\n this.availableImages = availableImages;\n this.loadVectorData = loadVectorData || loadVectorTile;\n this.fetching = {};\n this.loading = {};\n this.loaded = {};\n }\n\n /**\n * Implements {@link WorkerSource#loadTile}. Delegates to\n * {@link VectorTileWorkerSource#loadVectorData} (which by default expects\n * a `params.url` property) for fetching and producing a VectorTile object.\n */\n loadTile(params: WorkerTileParameters, callback: WorkerTileCallback) {\n const uid = params.uid;\n\n if (!this.loading)\n this.loading = {};\n\n const perf = (params && params.request && params.request.collectResourceTiming) ?\n new RequestPerformance(params.request) : false;\n\n const workerTile = this.loading[uid] = new WorkerTile(params);\n workerTile.abort = this.loadVectorData(params, (err, response) => {\n delete this.loading[uid];\n\n if (err || !response) {\n workerTile.status = 'done';\n this.loaded[uid] = workerTile;\n return callback(err);\n }\n\n const rawTileData = response.rawData;\n const cacheControl = {} as ExpiryData;\n if (response.expires) cacheControl.expires = response.expires;\n if (response.cacheControl) cacheControl.cacheControl = response.cacheControl;\n\n const resourceTiming = {} as {resourceTiming: any};\n if (perf) {\n const resourceTimingData = perf.finish();\n // it's necessary to eval the result of getEntriesByName() here via parse/stringify\n // late evaluation in the main thread causes TypeError: illegal invocation\n if (resourceTimingData)\n resourceTiming.resourceTiming = JSON.parse(JSON.stringify(resourceTimingData));\n }\n\n workerTile.vectorTile = response.vectorTile;\n workerTile.parse(response.vectorTile, this.layerIndex, this.availableImages, this.actor, (err, result) => {\n delete this.fetching[uid];\n if (err || !result) return callback(err);\n\n // Transferring a copy of rawTileData because the worker needs to retain its copy.\n callback(null, extend({rawTileData: rawTileData.slice(0)}, result, cacheControl, resourceTiming));\n });\n\n this.loaded = this.loaded || {};\n this.loaded[uid] = workerTile;\n // keep the original fetching state so that reload tile can pick it up if the original parse is cancelled by reloads' parse\n this.fetching[uid] = {rawTileData, cacheControl, resourceTiming};\n }) as AbortVectorData;\n }\n\n /**\n * Implements {@link WorkerSource#reloadTile}.\n */\n reloadTile(params: WorkerTileParameters, callback: WorkerTileCallback) {\n const loaded = this.loaded;\n const uid = params.uid;\n if (loaded && loaded[uid]) {\n const workerTile = loaded[uid];\n workerTile.showCollisionBoxes = params.showCollisionBoxes;\n if (workerTile.status === 'parsing') {\n workerTile.parse(workerTile.vectorTile, this.layerIndex, this.availableImages, this.actor, (err, result) => {\n if (err || !result) return callback(err, result);\n\n // if we have cancelled the original parse, make sure to pass the rawTileData from the original fetch\n let parseResult;\n if (this.fetching[uid]) {\n const {rawTileData, cacheControl, resourceTiming} = this.fetching[uid];\n delete this.fetching[uid];\n parseResult = extend({rawTileData: rawTileData.slice(0)}, result, cacheControl, resourceTiming);\n } else {\n parseResult = result;\n }\n\n callback(null, parseResult);\n });\n } else if (workerTile.status === 'done') {\n // if there was no vector tile data on the initial load, don't try and re-parse tile\n if (workerTile.vectorTile) {\n workerTile.parse(workerTile.vectorTile, this.layerIndex, this.availableImages, this.actor, callback);\n } else {\n callback();\n }\n }\n }\n }\n\n /**\n * Implements {@link WorkerSource#abortTile}.\n *\n * @param params - The tile parameters\n * @param callback - The callback\n */\n abortTile(params: TileParameters, callback: WorkerTileCallback) {\n const loading = this.loading,\n uid = params.uid;\n if (loading && loading[uid] && loading[uid].abort) {\n loading[uid].abort();\n delete loading[uid];\n }\n callback();\n }\n\n /**\n * Implements {@link WorkerSource#removeTile}.\n *\n * @param params - The tile parameters\n * @param callback - The callback\n */\n removeTile(params: TileParameters, callback: WorkerTileCallback) {\n const loaded = this.loaded,\n uid = params.uid;\n if (loaded && loaded[uid]) {\n delete loaded[uid];\n }\n callback();\n }\n}\n","import {DEMData} from '../data/dem_data';\nimport {RGBAImage} from '../util/image';\nimport type {Actor} from '../util/actor';\nimport type {\n WorkerDEMTileParameters,\n WorkerDEMTileCallback,\n TileParameters\n} from './worker_source';\nimport {getImageData, isImageBitmap} from '../util/util';\n\nexport class RasterDEMTileWorkerSource {\n actor: Actor;\n loaded: {[_: string]: DEMData};\n\n constructor() {\n this.loaded = {};\n }\n\n async loadTile(params: WorkerDEMTileParameters, callback: WorkerDEMTileCallback) {\n const {uid, encoding, rawImageData, redFactor, greenFactor, blueFactor, baseShift} = params;\n const width = rawImageData.width + 2;\n const height = rawImageData.height + 2;\n const imagePixels: RGBAImage = isImageBitmap(rawImageData) ?\n new RGBAImage({width, height}, await getImageData(rawImageData, -1, -1, width, height)) :\n rawImageData;\n const dem = new DEMData(uid, imagePixels, encoding, redFactor, greenFactor, blueFactor, baseShift);\n this.loaded = this.loaded || {};\n this.loaded[uid] = dem;\n callback(null, dem);\n }\n\n removeTile(params: TileParameters) {\n const loaded = this.loaded,\n uid = params.uid;\n if (loaded && loaded[uid]) {\n delete loaded[uid];\n }\n }\n}\n","\nmodule.exports = rewind;\n\nfunction rewind(gj, outer) {\n var type = gj && gj.type, i;\n\n if (type === 'FeatureCollection') {\n for (i = 0; i < gj.features.length; i++) rewind(gj.features[i], outer);\n\n } else if (type === 'GeometryCollection') {\n for (i = 0; i < gj.geometries.length; i++) rewind(gj.geometries[i], outer);\n\n } else if (type === 'Feature') {\n rewind(gj.geometry, outer);\n\n } else if (type === 'Polygon') {\n rewindRings(gj.coordinates, outer);\n\n } else if (type === 'MultiPolygon') {\n for (i = 0; i < gj.coordinates.length; i++) rewindRings(gj.coordinates[i], outer);\n }\n\n return gj;\n}\n\nfunction rewindRings(rings, outer) {\n if (rings.length === 0) return;\n\n rewindRing(rings[0], outer);\n for (var i = 1; i < rings.length; i++) {\n rewindRing(rings[i], !outer);\n }\n}\n\nfunction rewindRing(ring, dir) {\n var area = 0, err = 0;\n for (var i = 0, len = ring.length, j = len - 1; i < len; j = i++) {\n var k = (ring[i][0] - ring[j][0]) * (ring[j][1] + ring[i][1]);\n var m = area + k;\n err += Math.abs(area) >= Math.abs(k) ? area - m + k : k - m + area;\n area = m;\n }\n if (area + err >= 0 !== !!dir) ring.reverse();\n}\n","import Point from '@mapbox/point-geometry';\n\nimport mvt from '@mapbox/vector-tile';\nimport type {VectorTileFeature, VectorTileLayer, VectorTile} from '@mapbox/vector-tile';\nconst toGeoJSON = mvt.VectorTileFeature.prototype.toGeoJSON;\nimport {EXTENT} from '../data/extent';\nimport type {TileFeature, AnyProps} from 'supercluster';\nimport type {Feature as GeoJSONVTFeature} from 'geojson-vt';\n\nexport type Feature = TileFeature | GeoJSONVTFeature;\n\nclass FeatureWrapper implements VectorTileFeature {\n _feature: Feature;\n\n extent: number;\n type: Feature['type'];\n id: number;\n properties: {[_: string]: string | number | boolean};\n\n constructor(feature: Feature) {\n this._feature = feature;\n\n this.extent = EXTENT;\n this.type = feature.type;\n this.properties = feature.tags;\n\n // If the feature has a top-level `id` property, copy it over, but only\n // if it can be coerced to an integer, because this wrapper is used for\n // serializing geojson feature data into vector tile PBF data, and the\n // vector tile spec only supports integer values for feature ids --\n // allowing non-integer values here results in a non-compliant PBF\n // that causes an exception when it is parsed with vector-tile-js\n if ('id' in feature && !isNaN(feature.id as any)) {\n this.id = parseInt(feature.id, 10);\n }\n }\n\n loadGeometry() {\n if (this._feature.type === 1) {\n const geometry = [];\n for (const point of this._feature.geometry) {\n geometry.push([new Point(point[0], point[1])]);\n }\n return geometry;\n } else {\n const geometry = [];\n for (const ring of this._feature.geometry) {\n const newRing = [];\n for (const point of ring) {\n newRing.push(new Point(point[0], point[1]));\n }\n geometry.push(newRing);\n }\n return geometry;\n }\n }\n\n toGeoJSON(x: number, y: number, z: number) {\n return toGeoJSON.call(this, x, y, z);\n }\n}\n\nexport class GeoJSONWrapper implements VectorTile, VectorTileLayer {\n layers: {[_: string]: VectorTileLayer};\n name: string;\n extent: number;\n length: number;\n _features: Array;\n\n constructor(features: Array) {\n this.layers = {'_geojsonTileLayer': this};\n this.name = '_geojsonTileLayer';\n this.extent = EXTENT;\n this.length = features.length;\n this._features = features;\n }\n\n feature(i: number): VectorTileFeature {\n return new FeatureWrapper(this._features[i]);\n }\n}\n","'use strict'\n\nvar Point = require('@mapbox/point-geometry')\nvar VectorTileFeature = require('@mapbox/vector-tile').VectorTileFeature\n\nmodule.exports = GeoJSONWrapper\n\n// conform to vectortile api\nfunction GeoJSONWrapper (features, options) {\n this.options = options || {}\n this.features = features\n this.length = features.length\n}\n\nGeoJSONWrapper.prototype.feature = function (i) {\n return new FeatureWrapper(this.features[i], this.options.extent)\n}\n\nfunction FeatureWrapper (feature, extent) {\n this.id = typeof feature.id === 'number' ? feature.id : undefined\n this.type = feature.type\n this.rawGeometry = feature.type === 1 ? [feature.geometry] : feature.geometry\n this.properties = feature.tags\n this.extent = extent || 4096\n}\n\nFeatureWrapper.prototype.loadGeometry = function () {\n var rings = this.rawGeometry\n this.geometry = []\n\n for (var i = 0; i < rings.length; i++) {\n var ring = rings[i]\n var newRing = []\n for (var j = 0; j < ring.length; j++) {\n newRing.push(new Point(ring[j][0], ring[j][1]))\n }\n this.geometry.push(newRing)\n }\n return this.geometry\n}\n\nFeatureWrapper.prototype.bbox = function () {\n if (!this.geometry) this.loadGeometry()\n\n var rings = this.geometry\n var x1 = Infinity\n var x2 = -Infinity\n var y1 = Infinity\n var y2 = -Infinity\n\n for (var i = 0; i < rings.length; i++) {\n var ring = rings[i]\n\n for (var j = 0; j < ring.length; j++) {\n var coord = ring[j]\n\n x1 = Math.min(x1, coord.x)\n x2 = Math.max(x2, coord.x)\n y1 = Math.min(y1, coord.y)\n y2 = Math.max(y2, coord.y)\n }\n }\n\n return [x1, y1, x2, y2]\n}\n\nFeatureWrapper.prototype.toGeoJSON = VectorTileFeature.prototype.toGeoJSON\n","var Pbf = require('pbf')\nvar GeoJSONWrapper = require('./lib/geojson_wrapper')\n\nmodule.exports = fromVectorTileJs\nmodule.exports.fromVectorTileJs = fromVectorTileJs\nmodule.exports.fromGeojsonVt = fromGeojsonVt\nmodule.exports.GeoJSONWrapper = GeoJSONWrapper\n\n/**\n * Serialize a vector-tile-js-created tile to pbf\n *\n * @param {Object} tile\n * @return {Buffer} uncompressed, pbf-serialized tile data\n */\nfunction fromVectorTileJs (tile) {\n var out = new Pbf()\n writeTile(tile, out)\n return out.finish()\n}\n\n/**\n * Serialized a geojson-vt-created tile to pbf.\n *\n * @param {Object} layers - An object mapping layer names to geojson-vt-created vector tile objects\n * @param {Object} [options] - An object specifying the vector-tile specification version and extent that were used to create `layers`.\n * @param {Number} [options.version=1] - Version of vector-tile spec used\n * @param {Number} [options.extent=4096] - Extent of the vector tile\n * @return {Buffer} uncompressed, pbf-serialized tile data\n */\nfunction fromGeojsonVt (layers, options) {\n options = options || {}\n var l = {}\n for (var k in layers) {\n l[k] = new GeoJSONWrapper(layers[k].features, options)\n l[k].name = k\n l[k].version = options.version\n l[k].extent = options.extent\n }\n return fromVectorTileJs({ layers: l })\n}\n\nfunction writeTile (tile, pbf) {\n for (var key in tile.layers) {\n pbf.writeMessage(3, writeLayer, tile.layers[key])\n }\n}\n\nfunction writeLayer (layer, pbf) {\n pbf.writeVarintField(15, layer.version || 1)\n pbf.writeStringField(1, layer.name || '')\n pbf.writeVarintField(5, layer.extent || 4096)\n\n var i\n var context = {\n keys: [],\n values: [],\n keycache: {},\n valuecache: {}\n }\n\n for (i = 0; i < layer.length; i++) {\n context.feature = layer.feature(i)\n pbf.writeMessage(2, writeFeature, context)\n }\n\n var keys = context.keys\n for (i = 0; i < keys.length; i++) {\n pbf.writeStringField(3, keys[i])\n }\n\n var values = context.values\n for (i = 0; i < values.length; i++) {\n pbf.writeMessage(4, writeValue, values[i])\n }\n}\n\nfunction writeFeature (context, pbf) {\n var feature = context.feature\n\n if (feature.id !== undefined) {\n pbf.writeVarintField(1, feature.id)\n }\n\n pbf.writeMessage(2, writeProperties, context)\n pbf.writeVarintField(3, feature.type)\n pbf.writeMessage(4, writeGeometry, feature)\n}\n\nfunction writeProperties (context, pbf) {\n var feature = context.feature\n var keys = context.keys\n var values = context.values\n var keycache = context.keycache\n var valuecache = context.valuecache\n\n for (var key in feature.properties) {\n var value = feature.properties[key]\n\n var keyIndex = keycache[key]\n if (value === null) continue // don't encode null value properties\n\n if (typeof keyIndex === 'undefined') {\n keys.push(key)\n keyIndex = keys.length - 1\n keycache[key] = keyIndex\n }\n pbf.writeVarint(keyIndex)\n\n var type = typeof value\n if (type !== 'string' && type !== 'boolean' && type !== 'number') {\n value = JSON.stringify(value)\n }\n var valueKey = type + ':' + value\n var valueIndex = valuecache[valueKey]\n if (typeof valueIndex === 'undefined') {\n values.push(value)\n valueIndex = values.length - 1\n valuecache[valueKey] = valueIndex\n }\n pbf.writeVarint(valueIndex)\n }\n}\n\nfunction command (cmd, length) {\n return (length << 3) + (cmd & 0x7)\n}\n\nfunction zigzag (num) {\n return (num << 1) ^ (num >> 31)\n}\n\nfunction writeGeometry (feature, pbf) {\n var geometry = feature.loadGeometry()\n var type = feature.type\n var x = 0\n var y = 0\n var rings = geometry.length\n for (var r = 0; r < rings; r++) {\n var ring = geometry[r]\n var count = 1\n if (type === 1) {\n count = ring.length\n }\n pbf.writeVarint(command(1, count)) // moveto\n // do not write polygon closing path as lineto\n var lineCount = type === 3 ? ring.length - 1 : ring.length\n for (var i = 0; i < lineCount; i++) {\n if (i === 1 && type !== 1) {\n pbf.writeVarint(command(2, lineCount - 1)) // lineto\n }\n var dx = ring[i].x - x\n var dy = ring[i].y - y\n pbf.writeVarint(zigzag(dx))\n pbf.writeVarint(zigzag(dy))\n x += dx\n y += dy\n }\n if (type === 3) {\n pbf.writeVarint(command(7, 1)) // closepath\n }\n }\n}\n\nfunction writeValue (value, pbf) {\n var type = typeof value\n if (type === 'string') {\n pbf.writeStringField(1, value)\n } else if (type === 'boolean') {\n pbf.writeBooleanField(7, value)\n } else if (type === 'number') {\n if (value % 1 !== 0) {\n pbf.writeDoubleField(3, value)\n } else if (value < 0) {\n pbf.writeSVarintField(6, value)\n } else {\n pbf.writeVarintField(5, value)\n }\n }\n}\n","\nimport KDBush from 'kdbush';\n\nconst defaultOptions = {\n minZoom: 0, // min zoom to generate clusters on\n maxZoom: 16, // max zoom level to cluster the points on\n minPoints: 2, // minimum points to form a cluster\n radius: 40, // cluster radius in pixels\n extent: 512, // tile extent (radius is calculated relative to it)\n nodeSize: 64, // size of the KD-tree leaf node, affects performance\n log: false, // whether to log timing info\n\n // whether to generate numeric ids for input features (in vector tiles)\n generateId: false,\n\n // a reduce function for calculating custom cluster properties\n reduce: null, // (accumulated, props) => { accumulated.sum += props.sum; }\n\n // properties to use for individual points when running the reducer\n map: props => props // props => ({sum: props.my_value})\n};\n\nconst fround = Math.fround || (tmp => ((x) => { tmp[0] = +x; return tmp[0]; }))(new Float32Array(1));\n\nconst OFFSET_ZOOM = 2;\nconst OFFSET_ID = 3;\nconst OFFSET_PARENT = 4;\nconst OFFSET_NUM = 5;\nconst OFFSET_PROP = 6;\n\nexport default class Supercluster {\n constructor(options) {\n this.options = Object.assign(Object.create(defaultOptions), options);\n this.trees = new Array(this.options.maxZoom + 1);\n this.stride = this.options.reduce ? 7 : 6;\n this.clusterProps = [];\n }\n\n load(points) {\n const {log, minZoom, maxZoom} = this.options;\n\n if (log) console.time('total time');\n\n const timerId = `prepare ${ points.length } points`;\n if (log) console.time(timerId);\n\n this.points = points;\n\n // generate a cluster object for each point and index input points into a KD-tree\n const data = [];\n\n for (let i = 0; i < points.length; i++) {\n const p = points[i];\n if (!p.geometry) continue;\n\n const [lng, lat] = p.geometry.coordinates;\n const x = fround(lngX(lng));\n const y = fround(latY(lat));\n // store internal point/cluster data in flat numeric arrays for performance\n data.push(\n x, y, // projected point coordinates\n Infinity, // the last zoom the point was processed at\n i, // index of the source feature in the original input array\n -1, // parent cluster id\n 1 // number of points in a cluster\n );\n if (this.options.reduce) data.push(0); // noop\n }\n let tree = this.trees[maxZoom + 1] = this._createTree(data);\n\n if (log) console.timeEnd(timerId);\n\n // cluster points on max zoom, then cluster the results on previous zoom, etc.;\n // results in a cluster hierarchy across zoom levels\n for (let z = maxZoom; z >= minZoom; z--) {\n const now = +Date.now();\n\n // create a new set of clusters for the zoom and index them with a KD-tree\n tree = this.trees[z] = this._createTree(this._cluster(tree, z));\n\n if (log) console.log('z%d: %d clusters in %dms', z, tree.numItems, +Date.now() - now);\n }\n\n if (log) console.timeEnd('total time');\n\n return this;\n }\n\n getClusters(bbox, zoom) {\n let minLng = ((bbox[0] + 180) % 360 + 360) % 360 - 180;\n const minLat = Math.max(-90, Math.min(90, bbox[1]));\n let maxLng = bbox[2] === 180 ? 180 : ((bbox[2] + 180) % 360 + 360) % 360 - 180;\n const maxLat = Math.max(-90, Math.min(90, bbox[3]));\n\n if (bbox[2] - bbox[0] >= 360) {\n minLng = -180;\n maxLng = 180;\n } else if (minLng > maxLng) {\n const easternHem = this.getClusters([minLng, minLat, 180, maxLat], zoom);\n const westernHem = this.getClusters([-180, minLat, maxLng, maxLat], zoom);\n return easternHem.concat(westernHem);\n }\n\n const tree = this.trees[this._limitZoom(zoom)];\n const ids = tree.range(lngX(minLng), latY(maxLat), lngX(maxLng), latY(minLat));\n const data = tree.data;\n const clusters = [];\n for (const id of ids) {\n const k = this.stride * id;\n clusters.push(data[k + OFFSET_NUM] > 1 ? getClusterJSON(data, k, this.clusterProps) : this.points[data[k + OFFSET_ID]]);\n }\n return clusters;\n }\n\n getChildren(clusterId) {\n const originId = this._getOriginId(clusterId);\n const originZoom = this._getOriginZoom(clusterId);\n const errorMsg = 'No cluster with the specified id.';\n\n const tree = this.trees[originZoom];\n if (!tree) throw new Error(errorMsg);\n\n const data = tree.data;\n if (originId * this.stride >= data.length) throw new Error(errorMsg);\n\n const r = this.options.radius / (this.options.extent * Math.pow(2, originZoom - 1));\n const x = data[originId * this.stride];\n const y = data[originId * this.stride + 1];\n const ids = tree.within(x, y, r);\n const children = [];\n for (const id of ids) {\n const k = id * this.stride;\n if (data[k + OFFSET_PARENT] === clusterId) {\n children.push(data[k + OFFSET_NUM] > 1 ? getClusterJSON(data, k, this.clusterProps) : this.points[data[k + OFFSET_ID]]);\n }\n }\n\n if (children.length === 0) throw new Error(errorMsg);\n\n return children;\n }\n\n getLeaves(clusterId, limit, offset) {\n limit = limit || 10;\n offset = offset || 0;\n\n const leaves = [];\n this._appendLeaves(leaves, clusterId, limit, offset, 0);\n\n return leaves;\n }\n\n getTile(z, x, y) {\n const tree = this.trees[this._limitZoom(z)];\n const z2 = Math.pow(2, z);\n const {extent, radius} = this.options;\n const p = radius / extent;\n const top = (y - p) / z2;\n const bottom = (y + 1 + p) / z2;\n\n const tile = {\n features: []\n };\n\n this._addTileFeatures(\n tree.range((x - p) / z2, top, (x + 1 + p) / z2, bottom),\n tree.data, x, y, z2, tile);\n\n if (x === 0) {\n this._addTileFeatures(\n tree.range(1 - p / z2, top, 1, bottom),\n tree.data, z2, y, z2, tile);\n }\n if (x === z2 - 1) {\n this._addTileFeatures(\n tree.range(0, top, p / z2, bottom),\n tree.data, -1, y, z2, tile);\n }\n\n return tile.features.length ? tile : null;\n }\n\n getClusterExpansionZoom(clusterId) {\n let expansionZoom = this._getOriginZoom(clusterId) - 1;\n while (expansionZoom <= this.options.maxZoom) {\n const children = this.getChildren(clusterId);\n expansionZoom++;\n if (children.length !== 1) break;\n clusterId = children[0].properties.cluster_id;\n }\n return expansionZoom;\n }\n\n _appendLeaves(result, clusterId, limit, offset, skipped) {\n const children = this.getChildren(clusterId);\n\n for (const child of children) {\n const props = child.properties;\n\n if (props && props.cluster) {\n if (skipped + props.point_count <= offset) {\n // skip the whole cluster\n skipped += props.point_count;\n } else {\n // enter the cluster\n skipped = this._appendLeaves(result, props.cluster_id, limit, offset, skipped);\n // exit the cluster\n }\n } else if (skipped < offset) {\n // skip a single point\n skipped++;\n } else {\n // add a single point\n result.push(child);\n }\n if (result.length === limit) break;\n }\n\n return skipped;\n }\n\n _createTree(data) {\n const tree = new KDBush(data.length / this.stride | 0, this.options.nodeSize, Float32Array);\n for (let i = 0; i < data.length; i += this.stride) tree.add(data[i], data[i + 1]);\n tree.finish();\n tree.data = data;\n return tree;\n }\n\n _addTileFeatures(ids, data, x, y, z2, tile) {\n for (const i of ids) {\n const k = i * this.stride;\n const isCluster = data[k + OFFSET_NUM] > 1;\n\n let tags, px, py;\n if (isCluster) {\n tags = getClusterProperties(data, k, this.clusterProps);\n px = data[k];\n py = data[k + 1];\n } else {\n const p = this.points[data[k + OFFSET_ID]];\n tags = p.properties;\n const [lng, lat] = p.geometry.coordinates;\n px = lngX(lng);\n py = latY(lat);\n }\n\n const f = {\n type: 1,\n geometry: [[\n Math.round(this.options.extent * (px * z2 - x)),\n Math.round(this.options.extent * (py * z2 - y))\n ]],\n tags\n };\n\n // assign id\n let id;\n if (isCluster || this.options.generateId) {\n // optionally generate id for points\n id = data[k + OFFSET_ID];\n } else {\n // keep id if already assigned\n id = this.points[data[k + OFFSET_ID]].id;\n }\n\n if (id !== undefined) f.id = id;\n\n tile.features.push(f);\n }\n }\n\n _limitZoom(z) {\n return Math.max(this.options.minZoom, Math.min(Math.floor(+z), this.options.maxZoom + 1));\n }\n\n _cluster(tree, zoom) {\n const {radius, extent, reduce, minPoints} = this.options;\n const r = radius / (extent * Math.pow(2, zoom));\n const data = tree.data;\n const nextData = [];\n const stride = this.stride;\n\n // loop through each point\n for (let i = 0; i < data.length; i += stride) {\n // if we've already visited the point at this zoom level, skip it\n if (data[i + OFFSET_ZOOM] <= zoom) continue;\n data[i + OFFSET_ZOOM] = zoom;\n\n // find all nearby points\n const x = data[i];\n const y = data[i + 1];\n const neighborIds = tree.within(data[i], data[i + 1], r);\n\n const numPointsOrigin = data[i + OFFSET_NUM];\n let numPoints = numPointsOrigin;\n\n // count the number of points in a potential cluster\n for (const neighborId of neighborIds) {\n const k = neighborId * stride;\n // filter out neighbors that are already processed\n if (data[k + OFFSET_ZOOM] > zoom) numPoints += data[k + OFFSET_NUM];\n }\n\n // if there were neighbors to merge, and there are enough points to form a cluster\n if (numPoints > numPointsOrigin && numPoints >= minPoints) {\n let wx = x * numPointsOrigin;\n let wy = y * numPointsOrigin;\n\n let clusterProperties;\n let clusterPropIndex = -1;\n\n // encode both zoom and point index on which the cluster originated -- offset by total length of features\n const id = ((i / stride | 0) << 5) + (zoom + 1) + this.points.length;\n\n for (const neighborId of neighborIds) {\n const k = neighborId * stride;\n\n if (data[k + OFFSET_ZOOM] <= zoom) continue;\n data[k + OFFSET_ZOOM] = zoom; // save the zoom (so it doesn't get processed twice)\n\n const numPoints2 = data[k + OFFSET_NUM];\n wx += data[k] * numPoints2; // accumulate coordinates for calculating weighted center\n wy += data[k + 1] * numPoints2;\n\n data[k + OFFSET_PARENT] = id;\n\n if (reduce) {\n if (!clusterProperties) {\n clusterProperties = this._map(data, i, true);\n clusterPropIndex = this.clusterProps.length;\n this.clusterProps.push(clusterProperties);\n }\n reduce(clusterProperties, this._map(data, k));\n }\n }\n\n data[i + OFFSET_PARENT] = id;\n nextData.push(wx / numPoints, wy / numPoints, Infinity, id, -1, numPoints);\n if (reduce) nextData.push(clusterPropIndex);\n\n } else { // left points as unclustered\n for (let j = 0; j < stride; j++) nextData.push(data[i + j]);\n\n if (numPoints > 1) {\n for (const neighborId of neighborIds) {\n const k = neighborId * stride;\n if (data[k + OFFSET_ZOOM] <= zoom) continue;\n data[k + OFFSET_ZOOM] = zoom;\n for (let j = 0; j < stride; j++) nextData.push(data[k + j]);\n }\n }\n }\n }\n\n return nextData;\n }\n\n // get index of the point from which the cluster originated\n _getOriginId(clusterId) {\n return (clusterId - this.points.length) >> 5;\n }\n\n // get zoom of the point from which the cluster originated\n _getOriginZoom(clusterId) {\n return (clusterId - this.points.length) % 32;\n }\n\n _map(data, i, clone) {\n if (data[i + OFFSET_NUM] > 1) {\n const props = this.clusterProps[data[i + OFFSET_PROP]];\n return clone ? Object.assign({}, props) : props;\n }\n const original = this.points[data[i + OFFSET_ID]].properties;\n const result = this.options.map(original);\n return clone && result === original ? Object.assign({}, result) : result;\n }\n}\n\nfunction getClusterJSON(data, i, clusterProps) {\n return {\n type: 'Feature',\n id: data[i + OFFSET_ID],\n properties: getClusterProperties(data, i, clusterProps),\n geometry: {\n type: 'Point',\n coordinates: [xLng(data[i]), yLat(data[i + 1])]\n }\n };\n}\n\nfunction getClusterProperties(data, i, clusterProps) {\n const count = data[i + OFFSET_NUM];\n const abbrev =\n count >= 10000 ? `${Math.round(count / 1000) }k` :\n count >= 1000 ? `${Math.round(count / 100) / 10 }k` : count;\n const propIndex = data[i + OFFSET_PROP];\n const properties = propIndex === -1 ? {} : Object.assign({}, clusterProps[propIndex]);\n return Object.assign(properties, {\n cluster: true,\n cluster_id: data[i + OFFSET_ID],\n point_count: count,\n point_count_abbreviated: abbrev\n });\n}\n\n// longitude/latitude to spherical mercator in [0..1] range\nfunction lngX(lng) {\n return lng / 360 + 0.5;\n}\nfunction latY(lat) {\n const sin = Math.sin(lat * Math.PI / 180);\n const y = (0.5 - 0.25 * Math.log((1 + sin) / (1 - sin)) / Math.PI);\n return y < 0 ? 0 : y > 1 ? 1 : y;\n}\n\n// spherical mercator to longitude/latitude\nfunction xLng(x) {\n return (x - 0.5) * 360;\n}\nfunction yLat(y) {\n const y2 = (180 - y * 360) * Math.PI / 180;\n return 360 * Math.atan(Math.exp(y2)) / Math.PI - 90;\n}\n","\n// calculate simplification data using optimized Douglas-Peucker algorithm\n\nexport default function simplify(coords, first, last, sqTolerance) {\n var maxSqDist = sqTolerance;\n var mid = (last - first) >> 1;\n var minPosToMid = last - first;\n var index;\n\n var ax = coords[first];\n var ay = coords[first + 1];\n var bx = coords[last];\n var by = coords[last + 1];\n\n for (var i = first + 3; i < last; i += 3) {\n var d = getSqSegDist(coords[i], coords[i + 1], ax, ay, bx, by);\n\n if (d > maxSqDist) {\n index = i;\n maxSqDist = d;\n\n } else if (d === maxSqDist) {\n // a workaround to ensure we choose a pivot close to the middle of the list,\n // reducing recursion depth, for certain degenerate inputs\n // https://github.com/mapbox/geojson-vt/issues/104\n var posToMid = Math.abs(i - mid);\n if (posToMid < minPosToMid) {\n index = i;\n minPosToMid = posToMid;\n }\n }\n }\n\n if (maxSqDist > sqTolerance) {\n if (index - first > 3) simplify(coords, first, index, sqTolerance);\n coords[index + 2] = maxSqDist;\n if (last - index > 3) simplify(coords, index, last, sqTolerance);\n }\n}\n\n// square distance from a point to a segment\nfunction getSqSegDist(px, py, x, y, bx, by) {\n\n var dx = bx - x;\n var dy = by - y;\n\n if (dx !== 0 || dy !== 0) {\n\n var t = ((px - x) * dx + (py - y) * dy) / (dx * dx + dy * dy);\n\n if (t > 1) {\n x = bx;\n y = by;\n\n } else if (t > 0) {\n x += dx * t;\n y += dy * t;\n }\n }\n\n dx = px - x;\n dy = py - y;\n\n return dx * dx + dy * dy;\n}\n","\nexport default function createFeature(id, type, geom, tags) {\n var feature = {\n id: typeof id === 'undefined' ? null : id,\n type: type,\n geometry: geom,\n tags: tags,\n minX: Infinity,\n minY: Infinity,\n maxX: -Infinity,\n maxY: -Infinity\n };\n calcBBox(feature);\n return feature;\n}\n\nfunction calcBBox(feature) {\n var geom = feature.geometry;\n var type = feature.type;\n\n if (type === 'Point' || type === 'MultiPoint' || type === 'LineString') {\n calcLineBBox(feature, geom);\n\n } else if (type === 'Polygon' || type === 'MultiLineString') {\n for (var i = 0; i < geom.length; i++) {\n calcLineBBox(feature, geom[i]);\n }\n\n } else if (type === 'MultiPolygon') {\n for (i = 0; i < geom.length; i++) {\n for (var j = 0; j < geom[i].length; j++) {\n calcLineBBox(feature, geom[i][j]);\n }\n }\n }\n}\n\nfunction calcLineBBox(feature, geom) {\n for (var i = 0; i < geom.length; i += 3) {\n feature.minX = Math.min(feature.minX, geom[i]);\n feature.minY = Math.min(feature.minY, geom[i + 1]);\n feature.maxX = Math.max(feature.maxX, geom[i]);\n feature.maxY = Math.max(feature.maxY, geom[i + 1]);\n }\n}\n","\nimport simplify from './simplify';\nimport createFeature from './feature';\n\n// converts GeoJSON feature into an intermediate projected JSON vector format with simplification data\n\nexport default function convert(data, options) {\n var features = [];\n if (data.type === 'FeatureCollection') {\n for (var i = 0; i < data.features.length; i++) {\n convertFeature(features, data.features[i], options, i);\n }\n\n } else if (data.type === 'Feature') {\n convertFeature(features, data, options);\n\n } else {\n // single geometry or a geometry collection\n convertFeature(features, {geometry: data}, options);\n }\n\n return features;\n}\n\nfunction convertFeature(features, geojson, options, index) {\n if (!geojson.geometry) return;\n\n var coords = geojson.geometry.coordinates;\n var type = geojson.geometry.type;\n var tolerance = Math.pow(options.tolerance / ((1 << options.maxZoom) * options.extent), 2);\n var geometry = [];\n var id = geojson.id;\n if (options.promoteId) {\n id = geojson.properties[options.promoteId];\n } else if (options.generateId) {\n id = index || 0;\n }\n if (type === 'Point') {\n convertPoint(coords, geometry);\n\n } else if (type === 'MultiPoint') {\n for (var i = 0; i < coords.length; i++) {\n convertPoint(coords[i], geometry);\n }\n\n } else if (type === 'LineString') {\n convertLine(coords, geometry, tolerance, false);\n\n } else if (type === 'MultiLineString') {\n if (options.lineMetrics) {\n // explode into linestrings to be able to track metrics\n for (i = 0; i < coords.length; i++) {\n geometry = [];\n convertLine(coords[i], geometry, tolerance, false);\n features.push(createFeature(id, 'LineString', geometry, geojson.properties));\n }\n return;\n } else {\n convertLines(coords, geometry, tolerance, false);\n }\n\n } else if (type === 'Polygon') {\n convertLines(coords, geometry, tolerance, true);\n\n } else if (type === 'MultiPolygon') {\n for (i = 0; i < coords.length; i++) {\n var polygon = [];\n convertLines(coords[i], polygon, tolerance, true);\n geometry.push(polygon);\n }\n } else if (type === 'GeometryCollection') {\n for (i = 0; i < geojson.geometry.geometries.length; i++) {\n convertFeature(features, {\n id: id,\n geometry: geojson.geometry.geometries[i],\n properties: geojson.properties\n }, options, index);\n }\n return;\n } else {\n throw new Error('Input data is not a valid GeoJSON object.');\n }\n\n features.push(createFeature(id, type, geometry, geojson.properties));\n}\n\nfunction convertPoint(coords, out) {\n out.push(projectX(coords[0]));\n out.push(projectY(coords[1]));\n out.push(0);\n}\n\nfunction convertLine(ring, out, tolerance, isPolygon) {\n var x0, y0;\n var size = 0;\n\n for (var j = 0; j < ring.length; j++) {\n var x = projectX(ring[j][0]);\n var y = projectY(ring[j][1]);\n\n out.push(x);\n out.push(y);\n out.push(0);\n\n if (j > 0) {\n if (isPolygon) {\n size += (x0 * y - x * y0) / 2; // area\n } else {\n size += Math.sqrt(Math.pow(x - x0, 2) + Math.pow(y - y0, 2)); // length\n }\n }\n x0 = x;\n y0 = y;\n }\n\n var last = out.length - 3;\n out[2] = 1;\n simplify(out, 0, last, tolerance);\n out[last + 2] = 1;\n\n out.size = Math.abs(size);\n out.start = 0;\n out.end = out.size;\n}\n\nfunction convertLines(rings, out, tolerance, isPolygon) {\n for (var i = 0; i < rings.length; i++) {\n var geom = [];\n convertLine(rings[i], geom, tolerance, isPolygon);\n out.push(geom);\n }\n}\n\nfunction projectX(x) {\n return x / 360 + 0.5;\n}\n\nfunction projectY(y) {\n var sin = Math.sin(y * Math.PI / 180);\n var y2 = 0.5 - 0.25 * Math.log((1 + sin) / (1 - sin)) / Math.PI;\n return y2 < 0 ? 0 : y2 > 1 ? 1 : y2;\n}\n","\nimport createFeature from './feature';\n\n/* clip features between two axis-parallel lines:\n * | |\n * ___|___ | /\n * / | \\____|____/\n * | |\n */\n\nexport default function clip(features, scale, k1, k2, axis, minAll, maxAll, options) {\n\n k1 /= scale;\n k2 /= scale;\n\n if (minAll >= k1 && maxAll < k2) return features; // trivial accept\n else if (maxAll < k1 || minAll >= k2) return null; // trivial reject\n\n var clipped = [];\n\n for (var i = 0; i < features.length; i++) {\n\n var feature = features[i];\n var geometry = feature.geometry;\n var type = feature.type;\n\n var min = axis === 0 ? feature.minX : feature.minY;\n var max = axis === 0 ? feature.maxX : feature.maxY;\n\n if (min >= k1 && max < k2) { // trivial accept\n clipped.push(feature);\n continue;\n } else if (max < k1 || min >= k2) { // trivial reject\n continue;\n }\n\n var newGeometry = [];\n\n if (type === 'Point' || type === 'MultiPoint') {\n clipPoints(geometry, newGeometry, k1, k2, axis);\n\n } else if (type === 'LineString') {\n clipLine(geometry, newGeometry, k1, k2, axis, false, options.lineMetrics);\n\n } else if (type === 'MultiLineString') {\n clipLines(geometry, newGeometry, k1, k2, axis, false);\n\n } else if (type === 'Polygon') {\n clipLines(geometry, newGeometry, k1, k2, axis, true);\n\n } else if (type === 'MultiPolygon') {\n for (var j = 0; j < geometry.length; j++) {\n var polygon = [];\n clipLines(geometry[j], polygon, k1, k2, axis, true);\n if (polygon.length) {\n newGeometry.push(polygon);\n }\n }\n }\n\n if (newGeometry.length) {\n if (options.lineMetrics && type === 'LineString') {\n for (j = 0; j < newGeometry.length; j++) {\n clipped.push(createFeature(feature.id, type, newGeometry[j], feature.tags));\n }\n continue;\n }\n\n if (type === 'LineString' || type === 'MultiLineString') {\n if (newGeometry.length === 1) {\n type = 'LineString';\n newGeometry = newGeometry[0];\n } else {\n type = 'MultiLineString';\n }\n }\n if (type === 'Point' || type === 'MultiPoint') {\n type = newGeometry.length === 3 ? 'Point' : 'MultiPoint';\n }\n\n clipped.push(createFeature(feature.id, type, newGeometry, feature.tags));\n }\n }\n\n return clipped.length ? clipped : null;\n}\n\nfunction clipPoints(geom, newGeom, k1, k2, axis) {\n for (var i = 0; i < geom.length; i += 3) {\n var a = geom[i + axis];\n\n if (a >= k1 && a <= k2) {\n newGeom.push(geom[i]);\n newGeom.push(geom[i + 1]);\n newGeom.push(geom[i + 2]);\n }\n }\n}\n\nfunction clipLine(geom, newGeom, k1, k2, axis, isPolygon, trackMetrics) {\n\n var slice = newSlice(geom);\n var intersect = axis === 0 ? intersectX : intersectY;\n var len = geom.start;\n var segLen, t;\n\n for (var i = 0; i < geom.length - 3; i += 3) {\n var ax = geom[i];\n var ay = geom[i + 1];\n var az = geom[i + 2];\n var bx = geom[i + 3];\n var by = geom[i + 4];\n var a = axis === 0 ? ax : ay;\n var b = axis === 0 ? bx : by;\n var exited = false;\n\n if (trackMetrics) segLen = Math.sqrt(Math.pow(ax - bx, 2) + Math.pow(ay - by, 2));\n\n if (a < k1) {\n // ---|--> | (line enters the clip region from the left)\n if (b > k1) {\n t = intersect(slice, ax, ay, bx, by, k1);\n if (trackMetrics) slice.start = len + segLen * t;\n }\n } else if (a > k2) {\n // | <--|--- (line enters the clip region from the right)\n if (b < k2) {\n t = intersect(slice, ax, ay, bx, by, k2);\n if (trackMetrics) slice.start = len + segLen * t;\n }\n } else {\n addPoint(slice, ax, ay, az);\n }\n if (b < k1 && a >= k1) {\n // <--|--- | or <--|-----|--- (line exits the clip region on the left)\n t = intersect(slice, ax, ay, bx, by, k1);\n exited = true;\n }\n if (b > k2 && a <= k2) {\n // | ---|--> or ---|-----|--> (line exits the clip region on the right)\n t = intersect(slice, ax, ay, bx, by, k2);\n exited = true;\n }\n\n if (!isPolygon && exited) {\n if (trackMetrics) slice.end = len + segLen * t;\n newGeom.push(slice);\n slice = newSlice(geom);\n }\n\n if (trackMetrics) len += segLen;\n }\n\n // add the last point\n var last = geom.length - 3;\n ax = geom[last];\n ay = geom[last + 1];\n az = geom[last + 2];\n a = axis === 0 ? ax : ay;\n if (a >= k1 && a <= k2) addPoint(slice, ax, ay, az);\n\n // close the polygon if its endpoints are not the same after clipping\n last = slice.length - 3;\n if (isPolygon && last >= 3 && (slice[last] !== slice[0] || slice[last + 1] !== slice[1])) {\n addPoint(slice, slice[0], slice[1], slice[2]);\n }\n\n // add the final slice\n if (slice.length) {\n newGeom.push(slice);\n }\n}\n\nfunction newSlice(line) {\n var slice = [];\n slice.size = line.size;\n slice.start = line.start;\n slice.end = line.end;\n return slice;\n}\n\nfunction clipLines(geom, newGeom, k1, k2, axis, isPolygon) {\n for (var i = 0; i < geom.length; i++) {\n clipLine(geom[i], newGeom, k1, k2, axis, isPolygon, false);\n }\n}\n\nfunction addPoint(out, x, y, z) {\n out.push(x);\n out.push(y);\n out.push(z);\n}\n\nfunction intersectX(out, ax, ay, bx, by, x) {\n var t = (x - ax) / (bx - ax);\n out.push(x);\n out.push(ay + (by - ay) * t);\n out.push(1);\n return t;\n}\n\nfunction intersectY(out, ax, ay, bx, by, y) {\n var t = (y - ay) / (by - ay);\n out.push(ax + (bx - ax) * t);\n out.push(y);\n out.push(1);\n return t;\n}\n","\nimport clip from './clip';\nimport createFeature from './feature';\n\nexport default function wrap(features, options) {\n var buffer = options.buffer / options.extent;\n var merged = features;\n var left = clip(features, 1, -1 - buffer, buffer, 0, -1, 2, options); // left world copy\n var right = clip(features, 1, 1 - buffer, 2 + buffer, 0, -1, 2, options); // right world copy\n\n if (left || right) {\n merged = clip(features, 1, -buffer, 1 + buffer, 0, -1, 2, options) || []; // center world copy\n\n if (left) merged = shiftFeatureCoords(left, 1).concat(merged); // merge left into center\n if (right) merged = merged.concat(shiftFeatureCoords(right, -1)); // merge right into center\n }\n\n return merged;\n}\n\nfunction shiftFeatureCoords(features, offset) {\n var newFeatures = [];\n\n for (var i = 0; i < features.length; i++) {\n var feature = features[i],\n type = feature.type;\n\n var newGeometry;\n\n if (type === 'Point' || type === 'MultiPoint' || type === 'LineString') {\n newGeometry = shiftCoords(feature.geometry, offset);\n\n } else if (type === 'MultiLineString' || type === 'Polygon') {\n newGeometry = [];\n for (var j = 0; j < feature.geometry.length; j++) {\n newGeometry.push(shiftCoords(feature.geometry[j], offset));\n }\n } else if (type === 'MultiPolygon') {\n newGeometry = [];\n for (j = 0; j < feature.geometry.length; j++) {\n var newPolygon = [];\n for (var k = 0; k < feature.geometry[j].length; k++) {\n newPolygon.push(shiftCoords(feature.geometry[j][k], offset));\n }\n newGeometry.push(newPolygon);\n }\n }\n\n newFeatures.push(createFeature(feature.id, type, newGeometry, feature.tags));\n }\n\n return newFeatures;\n}\n\nfunction shiftCoords(points, offset) {\n var newPoints = [];\n newPoints.size = points.size;\n\n if (points.start !== undefined) {\n newPoints.start = points.start;\n newPoints.end = points.end;\n }\n\n for (var i = 0; i < points.length; i += 3) {\n newPoints.push(points[i] + offset, points[i + 1], points[i + 2]);\n }\n return newPoints;\n}\n","\n// Transforms the coordinates of each feature in the given tile from\n// mercator-projected space into (extent x extent) tile space.\nexport default function transformTile(tile, extent) {\n if (tile.transformed) return tile;\n\n var z2 = 1 << tile.z,\n tx = tile.x,\n ty = tile.y,\n i, j, k;\n\n for (i = 0; i < tile.features.length; i++) {\n var feature = tile.features[i],\n geom = feature.geometry,\n type = feature.type;\n\n feature.geometry = [];\n\n if (type === 1) {\n for (j = 0; j < geom.length; j += 2) {\n feature.geometry.push(transformPoint(geom[j], geom[j + 1], extent, z2, tx, ty));\n }\n } else {\n for (j = 0; j < geom.length; j++) {\n var ring = [];\n for (k = 0; k < geom[j].length; k += 2) {\n ring.push(transformPoint(geom[j][k], geom[j][k + 1], extent, z2, tx, ty));\n }\n feature.geometry.push(ring);\n }\n }\n }\n\n tile.transformed = true;\n\n return tile;\n}\n\nfunction transformPoint(x, y, extent, z2, tx, ty) {\n return [\n Math.round(extent * (x * z2 - tx)),\n Math.round(extent * (y * z2 - ty))];\n}\n","\nexport default function createTile(features, z, tx, ty, options) {\n var tolerance = z === options.maxZoom ? 0 : options.tolerance / ((1 << z) * options.extent);\n var tile = {\n features: [],\n numPoints: 0,\n numSimplified: 0,\n numFeatures: 0,\n source: null,\n x: tx,\n y: ty,\n z: z,\n transformed: false,\n minX: 2,\n minY: 1,\n maxX: -1,\n maxY: 0\n };\n for (var i = 0; i < features.length; i++) {\n tile.numFeatures++;\n addFeature(tile, features[i], tolerance, options);\n\n var minX = features[i].minX;\n var minY = features[i].minY;\n var maxX = features[i].maxX;\n var maxY = features[i].maxY;\n\n if (minX < tile.minX) tile.minX = minX;\n if (minY < tile.minY) tile.minY = minY;\n if (maxX > tile.maxX) tile.maxX = maxX;\n if (maxY > tile.maxY) tile.maxY = maxY;\n }\n return tile;\n}\n\nfunction addFeature(tile, feature, tolerance, options) {\n\n var geom = feature.geometry,\n type = feature.type,\n simplified = [];\n\n if (type === 'Point' || type === 'MultiPoint') {\n for (var i = 0; i < geom.length; i += 3) {\n simplified.push(geom[i]);\n simplified.push(geom[i + 1]);\n tile.numPoints++;\n tile.numSimplified++;\n }\n\n } else if (type === 'LineString') {\n addLine(simplified, geom, tile, tolerance, false, false);\n\n } else if (type === 'MultiLineString' || type === 'Polygon') {\n for (i = 0; i < geom.length; i++) {\n addLine(simplified, geom[i], tile, tolerance, type === 'Polygon', i === 0);\n }\n\n } else if (type === 'MultiPolygon') {\n\n for (var k = 0; k < geom.length; k++) {\n var polygon = geom[k];\n for (i = 0; i < polygon.length; i++) {\n addLine(simplified, polygon[i], tile, tolerance, true, i === 0);\n }\n }\n }\n\n if (simplified.length) {\n var tags = feature.tags || null;\n if (type === 'LineString' && options.lineMetrics) {\n tags = {};\n for (var key in feature.tags) tags[key] = feature.tags[key];\n tags['mapbox_clip_start'] = geom.start / geom.size;\n tags['mapbox_clip_end'] = geom.end / geom.size;\n }\n var tileFeature = {\n geometry: simplified,\n type: type === 'Polygon' || type === 'MultiPolygon' ? 3 :\n type === 'LineString' || type === 'MultiLineString' ? 2 : 1,\n tags: tags\n };\n if (feature.id !== null) {\n tileFeature.id = feature.id;\n }\n tile.features.push(tileFeature);\n }\n}\n\nfunction addLine(result, geom, tile, tolerance, isPolygon, isOuter) {\n var sqTolerance = tolerance * tolerance;\n\n if (tolerance > 0 && (geom.size < (isPolygon ? sqTolerance : tolerance))) {\n tile.numPoints += geom.length / 3;\n return;\n }\n\n var ring = [];\n\n for (var i = 0; i < geom.length; i += 3) {\n if (tolerance === 0 || geom[i + 2] > sqTolerance) {\n tile.numSimplified++;\n ring.push(geom[i]);\n ring.push(geom[i + 1]);\n }\n tile.numPoints++;\n }\n\n if (isPolygon) rewind(ring, isOuter);\n\n result.push(ring);\n}\n\nfunction rewind(ring, clockwise) {\n var area = 0;\n for (var i = 0, len = ring.length, j = len - 2; i < len; j = i, i += 2) {\n area += (ring[i] - ring[j]) * (ring[i + 1] + ring[j + 1]);\n }\n if (area > 0 === clockwise) {\n for (i = 0, len = ring.length; i < len / 2; i += 2) {\n var x = ring[i];\n var y = ring[i + 1];\n ring[i] = ring[len - 2 - i];\n ring[i + 1] = ring[len - 1 - i];\n ring[len - 2 - i] = x;\n ring[len - 1 - i] = y;\n }\n }\n}\n","\nimport convert from './convert'; // GeoJSON conversion and preprocessing\nimport clip from './clip'; // stripe clipping algorithm\nimport wrap from './wrap'; // date line processing\nimport transform from './transform'; // coordinate transformation\nimport createTile from './tile'; // final simplified tile generation\n\nexport default function geojsonvt(data, options) {\n return new GeoJSONVT(data, options);\n}\n\nfunction GeoJSONVT(data, options) {\n options = this.options = extend(Object.create(this.options), options);\n\n var debug = options.debug;\n\n if (debug) console.time('preprocess data');\n\n if (options.maxZoom < 0 || options.maxZoom > 24) throw new Error('maxZoom should be in the 0-24 range');\n if (options.promoteId && options.generateId) throw new Error('promoteId and generateId cannot be used together.');\n\n var features = convert(data, options);\n\n this.tiles = {};\n this.tileCoords = [];\n\n if (debug) {\n console.timeEnd('preprocess data');\n console.log('index: maxZoom: %d, maxPoints: %d', options.indexMaxZoom, options.indexMaxPoints);\n console.time('generate tiles');\n this.stats = {};\n this.total = 0;\n }\n\n features = wrap(features, options);\n\n // start slicing from the top tile down\n if (features.length) this.splitTile(features, 0, 0, 0);\n\n if (debug) {\n if (features.length) console.log('features: %d, points: %d', this.tiles[0].numFeatures, this.tiles[0].numPoints);\n console.timeEnd('generate tiles');\n console.log('tiles generated:', this.total, JSON.stringify(this.stats));\n }\n}\n\nGeoJSONVT.prototype.options = {\n maxZoom: 14, // max zoom to preserve detail on\n indexMaxZoom: 5, // max zoom in the tile index\n indexMaxPoints: 100000, // max number of points per tile in the tile index\n tolerance: 3, // simplification tolerance (higher means simpler)\n extent: 4096, // tile extent\n buffer: 64, // tile buffer on each side\n lineMetrics: false, // whether to calculate line metrics\n promoteId: null, // name of a feature property to be promoted to feature.id\n generateId: false, // whether to generate feature ids. Cannot be used with promoteId\n debug: 0 // logging level (0, 1 or 2)\n};\n\nGeoJSONVT.prototype.splitTile = function (features, z, x, y, cz, cx, cy) {\n\n var stack = [features, z, x, y],\n options = this.options,\n debug = options.debug;\n\n // avoid recursion by using a processing queue\n while (stack.length) {\n y = stack.pop();\n x = stack.pop();\n z = stack.pop();\n features = stack.pop();\n\n var z2 = 1 << z,\n id = toID(z, x, y),\n tile = this.tiles[id];\n\n if (!tile) {\n if (debug > 1) console.time('creation');\n\n tile = this.tiles[id] = createTile(features, z, x, y, options);\n this.tileCoords.push({z: z, x: x, y: y});\n\n if (debug) {\n if (debug > 1) {\n console.log('tile z%d-%d-%d (features: %d, points: %d, simplified: %d)',\n z, x, y, tile.numFeatures, tile.numPoints, tile.numSimplified);\n console.timeEnd('creation');\n }\n var key = 'z' + z;\n this.stats[key] = (this.stats[key] || 0) + 1;\n this.total++;\n }\n }\n\n // save reference to original geometry in tile so that we can drill down later if we stop now\n tile.source = features;\n\n // if it's the first-pass tiling\n if (!cz) {\n // stop tiling if we reached max zoom, or if the tile is too simple\n if (z === options.indexMaxZoom || tile.numPoints <= options.indexMaxPoints) continue;\n\n // if a drilldown to a specific tile\n } else {\n // stop tiling if we reached base zoom or our target tile zoom\n if (z === options.maxZoom || z === cz) continue;\n\n // stop tiling if it's not an ancestor of the target tile\n var m = 1 << (cz - z);\n if (x !== Math.floor(cx / m) || y !== Math.floor(cy / m)) continue;\n }\n\n // if we slice further down, no need to keep source geometry\n tile.source = null;\n\n if (features.length === 0) continue;\n\n if (debug > 1) console.time('clipping');\n\n // values we'll use for clipping\n var k1 = 0.5 * options.buffer / options.extent,\n k2 = 0.5 - k1,\n k3 = 0.5 + k1,\n k4 = 1 + k1,\n tl, bl, tr, br, left, right;\n\n tl = bl = tr = br = null;\n\n left = clip(features, z2, x - k1, x + k3, 0, tile.minX, tile.maxX, options);\n right = clip(features, z2, x + k2, x + k4, 0, tile.minX, tile.maxX, options);\n features = null;\n\n if (left) {\n tl = clip(left, z2, y - k1, y + k3, 1, tile.minY, tile.maxY, options);\n bl = clip(left, z2, y + k2, y + k4, 1, tile.minY, tile.maxY, options);\n left = null;\n }\n\n if (right) {\n tr = clip(right, z2, y - k1, y + k3, 1, tile.minY, tile.maxY, options);\n br = clip(right, z2, y + k2, y + k4, 1, tile.minY, tile.maxY, options);\n right = null;\n }\n\n if (debug > 1) console.timeEnd('clipping');\n\n stack.push(tl || [], z + 1, x * 2, y * 2);\n stack.push(bl || [], z + 1, x * 2, y * 2 + 1);\n stack.push(tr || [], z + 1, x * 2 + 1, y * 2);\n stack.push(br || [], z + 1, x * 2 + 1, y * 2 + 1);\n }\n};\n\nGeoJSONVT.prototype.getTile = function (z, x, y) {\n var options = this.options,\n extent = options.extent,\n debug = options.debug;\n\n if (z < 0 || z > 24) return null;\n\n var z2 = 1 << z;\n x = ((x % z2) + z2) % z2; // wrap tile x coordinate\n\n var id = toID(z, x, y);\n if (this.tiles[id]) return transform(this.tiles[id], extent);\n\n if (debug > 1) console.log('drilling down to z%d-%d-%d', z, x, y);\n\n var z0 = z,\n x0 = x,\n y0 = y,\n parent;\n\n while (!parent && z0 > 0) {\n z0--;\n x0 = Math.floor(x0 / 2);\n y0 = Math.floor(y0 / 2);\n parent = this.tiles[toID(z0, x0, y0)];\n }\n\n if (!parent || !parent.source) return null;\n\n // if we found a parent tile containing the original geometry, we can drill down from it\n if (debug > 1) console.log('found parent tile z%d-%d-%d', z0, x0, y0);\n\n if (debug > 1) console.time('drilling down');\n this.splitTile(parent.source, z0, x0, y0, z, x, y);\n if (debug > 1) console.timeEnd('drilling down');\n\n return this.tiles[id] ? transform(this.tiles[id], extent) : null;\n};\n\nfunction toID(z, x, y) {\n return (((1 << z) * y + x) * 32) + z;\n}\n\nfunction extend(dest, src) {\n for (var i in src) dest[i] = src[i];\n return dest;\n}\n","/**\n * A way to indentify a feature, either by string or by number\n */\nexport type GeoJSONFeatureId = number | string;\n\n/**\n * The geojson source diff object\n */\nexport type GeoJSONSourceDiff = {\n /**\n * When set to `true` it will remove all features\n */\n removeAll?: boolean;\n /**\n * An array of features IDs to remove\n */\n remove?: Array;\n /**\n * An array of features to add\n */\n add?: Array;\n /**\n * An array of update objects\n */\n update?: Array;\n}\n\n/**\n * A geojson feature diff object\n */\nexport type GeoJSONFeatureDiff = {\n /**\n * The feature ID\n */\n id: GeoJSONFeatureId;\n /**\n * If it's a new geometry, place it here\n */\n newGeometry?: GeoJSON.Geometry;\n /**\n * Setting to `true` will remove all preperties\n */\n removeAllProperties?: boolean;\n /**\n * The properties keys to remove\n */\n removeProperties?: Array;\n /**\n * The properties to add or update along side their values\n */\n addOrUpdateProperties?: Array<{key: string; value: any}>;\n}\n\nexport type UpdateableGeoJSON = GeoJSON.Feature | GeoJSON.FeatureCollection | undefined;\n\nfunction getFeatureId(feature: GeoJSON.Feature, promoteId?: string): GeoJSONFeatureId | undefined {\n return promoteId ? feature.properties[promoteId] : feature.id;\n}\n\nexport function isUpdateableGeoJSON(data: GeoJSON.GeoJSON | undefined, promoteId?: string): data is UpdateableGeoJSON {\n // null can be updated\n if (data == null) {\n return true;\n }\n\n // a single feature with an id can be updated, need to explicitly check against null because 0 is a valid feature id that is falsy\n if (data.type === 'Feature') {\n return getFeatureId(data, promoteId) != null;\n }\n\n // a feature collection can be updated if every feature has an id, and the ids are all unique\n // this prevents us from silently dropping features if ids get reused\n if (data.type === 'FeatureCollection') {\n const seenIds = new Set();\n for (const feature of data.features) {\n const id = getFeatureId(feature, promoteId);\n if (id == null) {\n return false;\n }\n\n if (seenIds.has(id)) {\n return false;\n }\n\n seenIds.add(id);\n }\n\n return true;\n }\n\n return false;\n}\n\nexport function toUpdateable(data: UpdateableGeoJSON, promoteId?: string) {\n const result = new Map();\n if (data == null) {\n // empty result\n } else if (data.type === 'Feature') {\n result.set(getFeatureId(data, promoteId)!, data);\n } else {\n for (const feature of data.features) {\n result.set(getFeatureId(feature, promoteId)!, feature);\n }\n }\n\n return result;\n}\n\n// mutates updateable\nexport function applySourceDiff(updateable: Map, diff: GeoJSONSourceDiff, promoteId?: string): void {\n if (diff.removeAll) {\n updateable.clear();\n }\n\n if (diff.remove) {\n for (const id of diff.remove) {\n updateable.delete(id);\n }\n }\n\n if (diff.add) {\n for (const feature of diff.add) {\n const id = getFeatureId(feature, promoteId);\n\n if (id != null) {\n updateable.set(id, feature);\n }\n }\n }\n\n if (diff.update) {\n for (const update of diff.update) {\n let feature = updateable.get(update.id);\n\n if (feature == null) {\n continue;\n }\n\n // be careful to clone the feature and/or properties objects to avoid mutating our input\n const cloneFeature = update.newGeometry || update.removeAllProperties;\n // note: removeAllProperties gives us a new properties object, so we can skip the clone step\n const cloneProperties = !update.removeAllProperties && (update.removeProperties?.length > 0 || update.addOrUpdateProperties?.length > 0);\n if (cloneFeature || cloneProperties) {\n feature = {...feature};\n updateable.set(update.id, feature);\n if (cloneProperties) {\n feature.properties = {...feature.properties};\n }\n }\n\n if (update.newGeometry) {\n feature.geometry = update.newGeometry;\n }\n\n if (update.removeAllProperties) {\n feature.properties = {};\n } else if (update.removeProperties?.length > 0) {\n for (const prop of update.removeProperties) {\n if (Object.prototype.hasOwnProperty.call(feature.properties, prop)) {\n delete feature.properties[prop];\n }\n }\n }\n\n if (update.addOrUpdateProperties?.length > 0) {\n for (const {key, value} of update.addOrUpdateProperties) {\n feature.properties[key] = value;\n }\n }\n }\n }\n}\n","import {getJSON} from '../util/ajax';\n\nimport {RequestPerformance} from '../util/performance';\nimport rewind from '@mapbox/geojson-rewind';\nimport {GeoJSONWrapper} from './geojson_wrapper';\nimport vtpbf from 'vt-pbf';\nimport Supercluster, {type Options as SuperclusterOptions, type ClusterProperties} from 'supercluster';\nimport geojsonvt, {type Options as GeoJSONVTOptions} from 'geojson-vt';\nimport {VectorTileWorkerSource} from './vector_tile_worker_source';\nimport {createExpression} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {\n WorkerTileParameters,\n WorkerTileCallback,\n} from '../source/worker_source';\n\nimport type {Actor} from '../util/actor';\nimport type {StyleLayerIndex} from '../style/style_layer_index';\n\nimport type {LoadVectorDataCallback} from './vector_tile_worker_source';\nimport type {RequestParameters, ResponseCallback} from '../util/ajax';\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\nimport {isUpdateableGeoJSON, type GeoJSONSourceDiff, applySourceDiff, toUpdateable, GeoJSONFeatureId} from './geojson_source_diff';\n\nexport type LoadGeoJSONParameters = {\n request?: RequestParameters;\n /**\n * Literal GeoJSON data. Must be provided if `request.url` is not.\n */\n data?: string;\n dataDiff?: GeoJSONSourceDiff;\n source: string;\n cluster: boolean;\n superclusterOptions?: SuperclusterOptions;\n geojsonVtOptions?: GeoJSONVTOptions;\n clusterProperties?: ClusterProperties;\n filter?: Array;\n promoteId?: string;\n};\n\nexport type LoadGeoJSON = (params: LoadGeoJSONParameters, callback: ResponseCallback) => Cancelable;\n\ntype GeoJSONIndex = ReturnType | Supercluster;\n\n/**\n * The {@link WorkerSource} implementation that supports {@link GeoJSONSource}.\n * This class is designed to be easily reused to support custom source types\n * for data formats that can be parsed/converted into an in-memory GeoJSON\n * representation. To do so, create it with\n * `new GeoJSONWorkerSource(actor, layerIndex, customLoadGeoJSONFunction)`.\n * For a full example, see [mapbox-gl-topojson](https://github.com/developmentseed/mapbox-gl-topojson).\n */\nexport class GeoJSONWorkerSource extends VectorTileWorkerSource {\n _pendingCallback: Callback<{\n resourceTiming?: {[_: string]: Array};\n abandoned?: boolean;\n }>;\n _pendingRequest: Cancelable;\n _geoJSONIndex: GeoJSONIndex;\n _dataUpdateable = new Map();\n\n /**\n * @param loadGeoJSON - Optional method for custom loading/parsing of\n * GeoJSON based on parameters passed from the main-thread Source.\n * See {@link GeoJSONWorkerSource#loadGeoJSON}.\n */\n constructor(actor: Actor, layerIndex: StyleLayerIndex, availableImages: Array, loadGeoJSON?: LoadGeoJSON | null) {\n super(actor, layerIndex, availableImages);\n this.loadVectorData = this.loadGeoJSONTile;\n if (loadGeoJSON) {\n this.loadGeoJSON = loadGeoJSON;\n }\n }\n\n loadGeoJSONTile(params: WorkerTileParameters, callback: LoadVectorDataCallback): (() => void) | void {\n const canonical = params.tileID.canonical;\n\n if (!this._geoJSONIndex) {\n return callback(null, null); // we couldn't load the file\n }\n\n const geoJSONTile = this._geoJSONIndex.getTile(canonical.z, canonical.x, canonical.y);\n if (!geoJSONTile) {\n return callback(null, null); // nothing in the given tile\n }\n\n const geojsonWrapper = new GeoJSONWrapper(geoJSONTile.features);\n // Encode the geojson-vt tile into binary vector tile form. This\n // is a convenience that allows `FeatureIndex` to operate the same way\n // across `VectorTileSource` and `GeoJSONSource` data.\n let pbf = vtpbf(geojsonWrapper);\n if (pbf.byteOffset !== 0 || pbf.byteLength !== pbf.buffer.byteLength) {\n // Compatibility with node Buffer (https://github.com/mapbox/pbf/issues/35)\n pbf = new Uint8Array(pbf);\n }\n\n callback(null, {\n vectorTile: geojsonWrapper,\n rawData: pbf.buffer\n });\n }\n\n /**\n * Fetches (if appropriate), parses, and index geojson data into tiles. This\n * preparatory method must be called before {@link GeoJSONWorkerSource#loadTile}\n * can correctly serve up tiles.\n *\n * Defers to {@link GeoJSONWorkerSource#loadGeoJSON} for the fetching/parsing,\n * expecting `callback(error, data)` to be called with either an error or a\n * parsed GeoJSON object.\n *\n * When a `loadData` request comes in while a previous one is being processed,\n * the previous one is aborted.\n *\n * @param params - the parameters\n * @param callback - the callback for completion or error\n */\n loadData(params: LoadGeoJSONParameters, callback: Callback<{\n resourceTiming?: {[_: string]: Array};\n abandoned?: boolean;\n }>) {\n this._pendingRequest?.cancel();\n if (this._pendingCallback) {\n // Tell the foreground the previous call has been abandoned\n this._pendingCallback(null, {abandoned: true});\n }\n\n const perf = (params && params.request && params.request.collectResourceTiming) ?\n new RequestPerformance(params.request) : false;\n\n this._pendingCallback = callback;\n this._pendingRequest = this.loadGeoJSON(params, (err?: Error | null, data?: any | null) => {\n delete this._pendingCallback;\n delete this._pendingRequest;\n\n if (err || !data) {\n return callback(err);\n } else if (typeof data !== 'object') {\n return callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`));\n } else {\n rewind(data, true);\n\n try {\n if (params.filter) {\n const compiled = createExpression(params.filter, {type: 'boolean', 'property-type': 'data-driven', overridable: false, transition: false} as any);\n if (compiled.result === 'error')\n throw new Error(compiled.value.map(err => `${err.key}: ${err.message}`).join(', '));\n\n const features = data.features.filter(feature => compiled.value.evaluate({zoom: 0}, feature));\n data = {type: 'FeatureCollection', features};\n }\n\n this._geoJSONIndex = params.cluster ?\n new Supercluster(getSuperclusterOptions(params)).load(data.features) :\n geojsonvt(data, params.geojsonVtOptions);\n } catch (err) {\n return callback(err);\n }\n\n this.loaded = {};\n\n const result = {} as { resourceTiming: any };\n if (perf) {\n const resourceTimingData = perf.finish();\n // it's necessary to eval the result of getEntriesByName() here via parse/stringify\n // late evaluation in the main thread causes TypeError: illegal invocation\n if (resourceTimingData) {\n result.resourceTiming = {};\n result.resourceTiming[params.source] = JSON.parse(JSON.stringify(resourceTimingData));\n }\n }\n callback(null, result);\n }\n });\n }\n\n /**\n * Implements {@link WorkerSource#reloadTile}.\n *\n * If the tile is loaded, uses the implementation in VectorTileWorkerSource.\n * Otherwise, such as after a setData() call, we load the tile fresh.\n *\n * @param params - the parameters\n * @param callback - the callback for completion or error\n */\n reloadTile(params: WorkerTileParameters, callback: WorkerTileCallback) {\n const loaded = this.loaded,\n uid = params.uid;\n\n if (loaded && loaded[uid]) {\n return super.reloadTile(params, callback);\n } else {\n return this.loadTile(params, callback);\n }\n }\n\n /**\n * Fetch and parse GeoJSON according to the given params. Calls `callback`\n * with `(err, data)`, where `data` is a parsed GeoJSON object.\n *\n * GeoJSON is loaded and parsed from `params.url` if it exists, or else\n * expected as a literal (string or object) `params.data`.\n *\n * @param params - the parameters\n * @param callback - the callback for completion or error\n * @returns A Cancelable object.\n */\n loadGeoJSON = (params: LoadGeoJSONParameters, callback: ResponseCallback): Cancelable => {\n const {promoteId} = params;\n // Because of same origin issues, urls must either include an explicit\n // origin or absolute path.\n // ie: /foo/bar.json or http://example.com/bar.json\n // but not ../foo/bar.json\n if (params.request) {\n return getJSON(params.request, (\n error?: Error,\n data?: any,\n cacheControl?: string,\n expires?: string\n ) => {\n this._dataUpdateable = isUpdateableGeoJSON(data, promoteId) ? toUpdateable(data, promoteId) : undefined;\n callback(error, data, cacheControl, expires);\n });\n } else if (typeof params.data === 'string') {\n try {\n const parsed = JSON.parse(params.data);\n this._dataUpdateable = isUpdateableGeoJSON(parsed, promoteId) ? toUpdateable(parsed, promoteId) : undefined;\n callback(null, parsed);\n } catch (e) {\n callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`));\n }\n } else if (params.dataDiff) {\n if (this._dataUpdateable) {\n applySourceDiff(this._dataUpdateable, params.dataDiff, promoteId);\n callback(null, {type: 'FeatureCollection', features: Array.from(this._dataUpdateable.values())});\n } else {\n callback(new Error(`Cannot update existing geojson data in ${params.source}`));\n }\n } else {\n callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`));\n }\n\n return {cancel: () => {}};\n };\n\n removeSource(params: {\n source: string;\n }, callback: WorkerTileCallback) {\n if (this._pendingCallback) {\n // Don't leak callbacks\n this._pendingCallback(null, {abandoned: true});\n }\n callback();\n }\n\n getClusterExpansionZoom(params: {\n clusterId: number;\n }, callback: Callback) {\n try {\n callback(null, (this._geoJSONIndex as Supercluster).getClusterExpansionZoom(params.clusterId));\n } catch (e) {\n callback(e);\n }\n }\n\n getClusterChildren(params: {\n clusterId: number;\n }, callback: Callback>) {\n try {\n callback(null, (this._geoJSONIndex as Supercluster).getChildren(params.clusterId));\n } catch (e) {\n callback(e);\n }\n }\n\n getClusterLeaves(params: {\n clusterId: number;\n limit: number;\n offset: number;\n }, callback: Callback>) {\n try {\n callback(null, (this._geoJSONIndex as Supercluster).getLeaves(params.clusterId, params.limit, params.offset));\n } catch (e) {\n callback(e);\n }\n }\n}\n\nfunction getSuperclusterOptions({superclusterOptions, clusterProperties}: LoadGeoJSONParameters) {\n if (!clusterProperties || !superclusterOptions) return superclusterOptions;\n\n const mapExpressions = {};\n const reduceExpressions = {};\n const globals = {accumulated: null, zoom: 0};\n const feature = {properties: null};\n const propertyNames = Object.keys(clusterProperties);\n\n for (const key of propertyNames) {\n const [operator, mapExpression] = clusterProperties[key];\n\n const mapExpressionParsed = createExpression(mapExpression);\n const reduceExpressionParsed = createExpression(\n typeof operator === 'string' ? [operator, ['accumulated'], ['get', key]] : operator);\n\n mapExpressions[key] = mapExpressionParsed.value;\n reduceExpressions[key] = reduceExpressionParsed.value;\n }\n\n superclusterOptions.map = (pointProperties) => {\n feature.properties = pointProperties;\n const properties = {};\n for (const key of propertyNames) {\n properties[key] = mapExpressions[key].evaluate(globals, feature);\n }\n return properties;\n };\n superclusterOptions.reduce = (accumulated, clusterProperties) => {\n feature.properties = clusterProperties;\n for (const key of propertyNames) {\n globals.accumulated = accumulated[key];\n accumulated[key] = reduceExpressions[key].evaluate(globals, feature);\n }\n };\n\n return superclusterOptions;\n}\n","import {Actor, ActorTarget} from '../util/actor';\nimport {StyleLayerIndex} from '../style/style_layer_index';\nimport {VectorTileWorkerSource} from './vector_tile_worker_source';\nimport {RasterDEMTileWorkerSource} from './raster_dem_tile_worker_source';\nimport {GeoJSONWorkerSource} from './geojson_worker_source';\nimport {plugin as globalRTLTextPlugin} from './rtl_text_plugin';\nimport {isWorker} from '../util/util';\n\nimport type {\n WorkerSource,\n WorkerTileParameters,\n WorkerDEMTileParameters,\n WorkerTileCallback,\n WorkerDEMTileCallback,\n TileParameters\n} from '../source/worker_source';\n\nimport type {WorkerGlobalScopeInterface} from '../util/web_worker';\nimport type {Callback} from '../types/callback';\nimport type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {PluginState} from './rtl_text_plugin';\n\n/**\n * The Worker class responsidble for background thread related execution\n */\nexport default class Worker {\n self: WorkerGlobalScopeInterface & ActorTarget;\n actor: Actor;\n layerIndexes: {[_: string]: StyleLayerIndex};\n availableImages: {[_: string]: Array};\n workerSourceTypes: {\n [_: string]: {\n new (...args: any): WorkerSource;\n };\n };\n workerSources: {\n [_: string]: {\n [_: string]: {\n [_: string]: WorkerSource;\n };\n };\n };\n demWorkerSources: {\n [_: string]: {\n [_: string]: RasterDEMTileWorkerSource;\n };\n };\n referrer: string;\n\n constructor(self: WorkerGlobalScopeInterface & ActorTarget) {\n this.self = self;\n this.actor = new Actor(self, this);\n\n this.layerIndexes = {};\n this.availableImages = {};\n\n this.workerSourceTypes = {\n vector: VectorTileWorkerSource,\n geojson: GeoJSONWorkerSource\n };\n\n // [mapId][sourceType][sourceName] => worker source instance\n this.workerSources = {};\n this.demWorkerSources = {};\n\n this.self.registerWorkerSource = (name: string, WorkerSource: {\n new (...args: any): WorkerSource;\n }) => {\n if (this.workerSourceTypes[name]) {\n throw new Error(`Worker source with name \"${name}\" already registered.`);\n }\n this.workerSourceTypes[name] = WorkerSource;\n };\n\n // This is invoked by the RTL text plugin when the download via the `importScripts` call has finished, and the code has been parsed.\n this.self.registerRTLTextPlugin = (rtlTextPlugin: {\n applyArabicShaping: Function;\n processBidirectionalText: ((b: string, a: Array) => Array);\n processStyledBidirectionalText?: ((c: string, b: Array, a: Array) => Array<[string, Array]>);\n }) => {\n if (globalRTLTextPlugin.isParsed()) {\n throw new Error('RTL text plugin already registered.');\n }\n globalRTLTextPlugin['applyArabicShaping'] = rtlTextPlugin.applyArabicShaping;\n globalRTLTextPlugin['processBidirectionalText'] = rtlTextPlugin.processBidirectionalText;\n globalRTLTextPlugin['processStyledBidirectionalText'] = rtlTextPlugin.processStyledBidirectionalText;\n };\n }\n\n setReferrer(mapID: string, referrer: string) {\n this.referrer = referrer;\n }\n\n setImages(mapId: string, images: Array, callback: WorkerTileCallback) {\n this.availableImages[mapId] = images;\n for (const workerSource in this.workerSources[mapId]) {\n const ws = this.workerSources[mapId][workerSource];\n for (const source in ws) {\n ws[source].availableImages = images;\n }\n }\n callback();\n }\n\n setLayers(mapId: string, layers: Array, callback: WorkerTileCallback) {\n this.getLayerIndex(mapId).replace(layers);\n callback();\n }\n\n updateLayers(mapId: string, params: {\n layers: Array;\n removedIds: Array;\n }, callback: WorkerTileCallback) {\n this.getLayerIndex(mapId).update(params.layers, params.removedIds);\n callback();\n }\n\n loadTile(mapId: string, params: WorkerTileParameters & {\n type: string;\n }, callback: WorkerTileCallback) {\n this.getWorkerSource(mapId, params.type, params.source).loadTile(params, callback);\n }\n\n loadDEMTile(mapId: string, params: WorkerDEMTileParameters, callback: WorkerDEMTileCallback) {\n this.getDEMWorkerSource(mapId, params.source).loadTile(params, callback);\n }\n\n reloadTile(mapId: string, params: WorkerTileParameters & {\n type: string;\n }, callback: WorkerTileCallback) {\n this.getWorkerSource(mapId, params.type, params.source).reloadTile(params, callback);\n }\n\n abortTile(mapId: string, params: TileParameters & {\n type: string;\n }, callback: WorkerTileCallback) {\n this.getWorkerSource(mapId, params.type, params.source).abortTile(params, callback);\n }\n\n removeTile(mapId: string, params: TileParameters & {\n type: string;\n }, callback: WorkerTileCallback) {\n this.getWorkerSource(mapId, params.type, params.source).removeTile(params, callback);\n }\n\n removeDEMTile(mapId: string, params: TileParameters) {\n this.getDEMWorkerSource(mapId, params.source).removeTile(params);\n }\n\n removeSource(mapId: string, params: {\n source: string;\n } & {\n type: string;\n }, callback: WorkerTileCallback) {\n\n if (!this.workerSources[mapId] ||\n !this.workerSources[mapId][params.type] ||\n !this.workerSources[mapId][params.type][params.source]) {\n return;\n }\n\n const worker = this.workerSources[mapId][params.type][params.source];\n delete this.workerSources[mapId][params.type][params.source];\n\n if (worker.removeSource !== undefined) {\n worker.removeSource(params, callback);\n } else {\n callback();\n }\n }\n\n /**\n * Load a {@link WorkerSource} script at params.url. The script is run\n * (using importScripts) with `registerWorkerSource` in scope, which is a\n * function taking `(name, workerSourceObject)`.\n */\n loadWorkerSource(map: string, params: {\n url: string;\n }, callback: Callback) {\n try {\n this.self.importScripts(params.url);\n callback();\n } catch (e) {\n callback(e.toString());\n }\n }\n\n syncRTLPluginState(map: string, state: PluginState, callback: Callback) {\n try {\n globalRTLTextPlugin.setState(state);\n const pluginURL = globalRTLTextPlugin.getPluginURL();\n if (\n globalRTLTextPlugin.isLoaded() &&\n !globalRTLTextPlugin.isParsed() &&\n pluginURL != null // Not possible when `isLoaded` is true, but keeps flow happy\n ) {\n this.self.importScripts(pluginURL);\n const complete = globalRTLTextPlugin.isParsed();\n const error = complete ? undefined : new Error(`RTL Text Plugin failed to import scripts from ${pluginURL}`);\n callback(error, complete);\n }\n } catch (e) {\n callback(e.toString());\n }\n }\n\n getAvailableImages(mapId: string) {\n let availableImages = this.availableImages[mapId];\n\n if (!availableImages) {\n availableImages = [];\n }\n\n return availableImages;\n }\n\n getLayerIndex(mapId: string) {\n let layerIndexes = this.layerIndexes[mapId];\n if (!layerIndexes) {\n layerIndexes = this.layerIndexes[mapId] = new StyleLayerIndex();\n }\n return layerIndexes;\n }\n\n getWorkerSource(mapId: string, sourceType: string, sourceName: string): WorkerSource {\n if (!this.workerSources[mapId])\n this.workerSources[mapId] = {};\n if (!this.workerSources[mapId][sourceType])\n this.workerSources[mapId][sourceType] = {};\n\n if (!this.workerSources[mapId][sourceType][sourceName]) {\n // use a wrapped actor so that we can attach a target mapId param\n // to any messages invoked by the WorkerSource\n const actor = {\n send: (type, data, callback) => {\n this.actor.send(type, data, callback, mapId);\n }\n };\n this.workerSources[mapId][sourceType][sourceName] = new (this.workerSourceTypes[sourceType] as any)((actor as any), this.getLayerIndex(mapId), this.getAvailableImages(mapId));\n }\n\n return this.workerSources[mapId][sourceType][sourceName];\n }\n\n getDEMWorkerSource(mapId: string, source: string) {\n if (!this.demWorkerSources[mapId])\n this.demWorkerSources[mapId] = {};\n\n if (!this.demWorkerSources[mapId][source]) {\n this.demWorkerSources[mapId][source] = new RasterDEMTileWorkerSource();\n }\n\n return this.demWorkerSources[mapId][source];\n }\n}\n\nif (isWorker()) {\n (self as any).worker = new Worker(self as any);\n}\n","import Point from '@mapbox/point-geometry';\n\nexport class DOM {\n private static readonly docStyle = typeof window !== 'undefined' && window.document && window.document.documentElement.style;\n\n private static userSelect: string;\n\n private static selectProp = DOM.testProp(['userSelect', 'MozUserSelect', 'WebkitUserSelect', 'msUserSelect']);\n\n private static transformProp = DOM.testProp(['transform', 'WebkitTransform']);\n\n private static testProp(props: string[]): string {\n if (!DOM.docStyle) return props[0];\n for (let i = 0; i < props.length; i++) {\n if (props[i] in DOM.docStyle) {\n return props[i];\n }\n }\n return props[0];\n }\n\n public static create(tagName: K, className?: string, container?: HTMLElement): HTMLElementTagNameMap[K] {\n const el = window.document.createElement(tagName);\n if (className !== undefined) el.className = className;\n if (container) container.appendChild(el);\n return el;\n }\n\n public static createNS(namespaceURI: string, tagName: string) {\n const el = window.document.createElementNS(namespaceURI, tagName);\n return el;\n }\n\n public static disableDrag() {\n if (DOM.docStyle && DOM.selectProp) {\n DOM.userSelect = DOM.docStyle[DOM.selectProp];\n DOM.docStyle[DOM.selectProp] = 'none';\n }\n }\n\n public static enableDrag() {\n if (DOM.docStyle && DOM.selectProp) {\n DOM.docStyle[DOM.selectProp] = DOM.userSelect;\n }\n }\n\n public static setTransform(el: HTMLElement, value: string) {\n el.style[DOM.transformProp] = value;\n }\n\n public static addEventListener(target: HTMLElement | Window | Document, type: string, callback: EventListenerOrEventListenerObject, options: {\n passive?: boolean;\n capture?: boolean;\n } = {}) {\n if ('passive' in options) {\n target.addEventListener(type, callback, options);\n } else {\n target.addEventListener(type, callback, options.capture);\n }\n }\n\n public static removeEventListener(target: HTMLElement | Window | Document, type: string, callback: EventListenerOrEventListenerObject, options: {\n passive?: boolean;\n capture?: boolean;\n } = {}) {\n if ('passive' in options) {\n target.removeEventListener(type, callback, options);\n } else {\n target.removeEventListener(type, callback, options.capture);\n }\n }\n\n // Suppress the next click, but only if it's immediate.\n private static suppressClickInternal(e) {\n e.preventDefault();\n e.stopPropagation();\n window.removeEventListener('click', DOM.suppressClickInternal, true);\n }\n\n public static suppressClick() {\n window.addEventListener('click', DOM.suppressClickInternal, true);\n window.setTimeout(() => {\n window.removeEventListener('click', DOM.suppressClickInternal, true);\n }, 0);\n }\n\n public static mousePos(el: HTMLElement, e: MouseEvent | Touch) {\n const rect = el.getBoundingClientRect();\n return new Point(\n e.clientX - rect.left - el.clientLeft,\n e.clientY - rect.top - el.clientTop\n );\n }\n\n public static touchPos(el: HTMLElement, touches: TouchList) {\n const rect = el.getBoundingClientRect();\n const points: Point[] = [];\n for (let i = 0; i < touches.length; i++) {\n points.push(new Point(\n touches[i].clientX - rect.left - el.clientLeft,\n touches[i].clientY - rect.top - el.clientTop\n ));\n }\n return points;\n }\n\n public static mouseButton(e: MouseEvent) {\n return e.button;\n }\n\n public static remove(node: HTMLElement) {\n if (node.parentNode) {\n node.parentNode.removeChild(node);\n }\n }\n}\n","export const webpSupported = {\n supported: false,\n testSupport\n};\n\nlet glForTesting: WebGLRenderingContext|WebGL2RenderingContext;\nlet webpCheckComplete = false;\nlet webpImgTest;\nlet webpImgTestOnloadComplete = false;\n\nif (typeof document !== 'undefined') {\n webpImgTest = document.createElement('img');\n webpImgTest.onload = function() {\n if (glForTesting) testWebpTextureUpload(glForTesting);\n glForTesting = null;\n webpImgTestOnloadComplete = true;\n };\n webpImgTest.onerror = function() {\n webpCheckComplete = true;\n glForTesting = null;\n };\n webpImgTest.src = '';\n}\n\nfunction testSupport(gl: WebGLRenderingContext | WebGL2RenderingContext) {\n if (webpCheckComplete || !webpImgTest) return;\n\n // HTMLImageElement.complete is set when an image is done loading it's source\n // regardless of whether the load was successful or not.\n // It's possible for an error to set HTMLImageElement.complete to true which would trigger\n // testWebpTextureUpload and mistakenly set exported.supported to true in browsers which don't support webp\n // To avoid this, we set a flag in the image's onload handler and only call testWebpTextureUpload\n // after a successful image load event.\n if (webpImgTestOnloadComplete) {\n testWebpTextureUpload(gl);\n } else {\n glForTesting = gl;\n\n }\n}\n\nfunction testWebpTextureUpload(gl: WebGLRenderingContext|WebGL2RenderingContext) {\n // Edge 18 supports WebP but not uploading a WebP image to a gl texture\n // Test support for this before allowing WebP images.\n // https://github.com/mapbox/mapbox-gl-js/issues/7671\n const texture = gl.createTexture();\n gl.bindTexture(gl.TEXTURE_2D, texture);\n\n try {\n gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, webpImgTest);\n\n // The error does not get triggered in Edge if the context is lost\n if (gl.isContextLost()) return;\n\n webpSupported.supported = true;\n } catch (e) {\n // Catch \"Unspecified Error.\" in Edge 18.\n }\n\n gl.deleteTexture(texture);\n\n webpCheckComplete = true;\n}\n","import type {Cancelable} from '../types/cancelable';\nimport {RequestParameters, ExpiryData, makeRequest, sameOrigin, getProtocolAction} from './ajax';\nimport type {Callback} from '../types/callback';\n\nimport {arrayBufferToImageBitmap, arrayBufferToImage, extend, isWorker, isImageBitmap} from './util';\nimport {webpSupported} from './webp_supported';\nimport {config} from './config';\n\n/**\n * The callback that is being called after an image was fetched\n */\nexport type GetImageCallback = (error?: Error | null, image?: HTMLImageElement | ImageBitmap | null, expiry?: ExpiryData | null) => void;\n\ntype ImageQueueThrottleControlCallback = () => boolean;\n\nexport type ImageRequestQueueItem = Cancelable & {\n requestParameters: RequestParameters;\n supportImageRefresh: boolean;\n callback: GetImageCallback;\n cancelled: boolean;\n completed: boolean;\n innerRequest?: Cancelable;\n}\n\ntype ImageQueueThrottleCallbackDictionary = {\n [Key: number]: ImageQueueThrottleControlCallback;\n}\n\ntype HTMLImageElementWithPriority = HTMLImageElement &\n{\n // fetchPriority is experimental property supported on Chromium browsers from Version 102\n // By default images are downloaded with priority low, whereas fetch request downloads with priority high\n // https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/fetchPriority\n fetchPriority?: 'auto' | 'high' | 'low';\n};\n\n/**\n * By default, the image queue is self driven, meaning as soon as one requested item is processed,\n * it will move on to next one as quickly as it can while limiting\n * the number of concurrent requests to MAX_PARALLEL_IMAGE_REQUESTS. The default behavior\n * ensures that static views of the map can be rendered with minimal delay.\n *\n * However, the default behavior can prevent dynamic views of the map from rendering\n * smoothly in that many requests can finish in one render frame, putting too much pressure on GPU.\n *\n * When the view of the map is moving dynamically, smoother frame rates can be achieved\n * by throttling the number of items processed by the queue per frame. This can be\n * accomplished by using {@link addThrottleControl} to allow the caller to\n * use a lambda function to determine when the queue should be throttled (e.g. when isMoving())\n * and manually calling {@link processQueue} in the render loop.\n */\nexport namespace ImageRequest {\n let imageRequestQueue : ImageRequestQueueItem[];\n let currentParallelImageRequests:number;\n\n let throttleControlCallbackHandleCounter: number;\n let throttleControlCallbacks: ImageQueueThrottleCallbackDictionary;\n\n /**\n * Reset the image request queue, removing all pending requests.\n */\n export const resetRequestQueue = (): void => {\n imageRequestQueue = [];\n currentParallelImageRequests = 0;\n throttleControlCallbackHandleCounter = 0;\n throttleControlCallbacks = {};\n };\n\n /**\n * Install a callback to control when image queue throttling is desired.\n * (e.g. when the map view is moving)\n * @param callback - The callback function to install\n * @returns handle that identifies the installed callback.\n */\n export const addThrottleControl = (callback: ImageQueueThrottleControlCallback): number => {\n const handle = throttleControlCallbackHandleCounter++;\n throttleControlCallbacks[handle] = callback;\n return handle;\n };\n\n /**\n * Remove a previously installed callback by passing in the handle returned\n * by {@link addThrottleControl}.\n * @param callbackHandle - The handle for the callback to remove.\n */\n export const removeThrottleControl = (callbackHandle: number): void => {\n delete throttleControlCallbacks[callbackHandle];\n // Try updating the queue\n processQueue();\n };\n\n /**\n * Check to see if any of the installed callbacks are requesting the queue\n * to be throttled.\n * @returns `true` if any callback is causing the queue to be throttled.\n */\n const isThrottled = (): boolean => {\n const allControlKeys = Object.keys(throttleControlCallbacks);\n let throttleingRequested = false;\n if (allControlKeys.length > 0) {\n for (const key of allControlKeys) {\n throttleingRequested = throttleControlCallbacks[key]();\n if (throttleingRequested) {\n break;\n }\n }\n }\n return throttleingRequested;\n };\n\n /**\n * Request to load an image.\n * @param requestParameters - Request parameters.\n * @param callback - Callback to issue when the request completes.\n * @param supportImageRefresh - `true`, if the image request need to support refresh based on cache headers.\n * @returns Cancelable request.\n */\n export const getImage = (\n requestParameters: RequestParameters,\n callback: GetImageCallback,\n supportImageRefresh: boolean = true\n ): ImageRequestQueueItem => {\n if (webpSupported.supported) {\n if (!requestParameters.headers) {\n requestParameters.headers = {};\n }\n requestParameters.headers.accept = 'image/webp,*/*';\n }\n\n const request:ImageRequestQueueItem = {\n requestParameters,\n supportImageRefresh,\n callback,\n cancelled: false,\n completed: false,\n cancel: () => {\n if (!request.completed && !request.cancelled) {\n request.cancelled = true;\n\n // Only reduce currentParallelImageRequests, if the image request was issued.\n if (request.innerRequest) {\n request.innerRequest.cancel();\n currentParallelImageRequests--;\n }\n\n // in the case of cancelling, it WILL move on\n processQueue();\n }\n }\n };\n\n imageRequestQueue.push(request);\n processQueue();\n return request;\n };\n\n const arrayBufferToCanvasImageSource = (data: ArrayBuffer, callback: Callback) => {\n const imageBitmapSupported = typeof createImageBitmap === 'function';\n if (imageBitmapSupported) {\n arrayBufferToImageBitmap(data, callback);\n } else {\n arrayBufferToImage(data, callback);\n }\n };\n\n const doImageRequest = (itemInQueue: ImageRequestQueueItem): Cancelable => {\n const {requestParameters, supportImageRefresh, callback} = itemInQueue;\n extend(requestParameters, {type: 'image'});\n\n // - If refreshExpiredTiles is false, then we can use HTMLImageElement to download raster images.\n // - Fetch/XHR (via MakeRequest API) will be used to download images for following scenarios:\n // 1. Style image sprite will had a issue with HTMLImageElement as described\n // here: https://github.com/mapbox/mapbox-gl-js/issues/1470\n // 2. If refreshExpiredTiles is true (default), then in order to read the image cache header,\n // fetch/XHR request will be required\n // - For any special case handling like use of AddProtocol, worker initiated request or additional headers\n // let makeRequest handle it.\n // - HtmlImageElement request automatically adds accept header for all the browser supported images\n const canUseHTMLImageElement = supportImageRefresh === false &&\n !isWorker() &&\n !getProtocolAction(requestParameters.url) &&\n (!requestParameters.headers ||\n Object.keys(requestParameters.headers).reduce((acc, item) => acc && item === 'accept', true));\n\n const action = canUseHTMLImageElement ? getImageUsingHtmlImage : makeRequest;\n return action(\n requestParameters,\n (err?: Error | null,\n data?: HTMLImageElement | ImageBitmap | ArrayBuffer | null,\n cacheControl?: string | null,\n expires?: string | null) => {\n onImageResponse(itemInQueue, callback, err, data, cacheControl, expires);\n });\n };\n\n const onImageResponse = (\n itemInQueue: ImageRequestQueueItem,\n callback:GetImageCallback,\n err?: Error | null,\n data?: HTMLImageElement | ImageBitmap | ArrayBuffer | null,\n cacheControl?: string | null,\n expires?: string | null): void => {\n if (err) {\n callback(err);\n } else if (data instanceof HTMLImageElement || isImageBitmap(data)) {\n // User using addProtocol can directly return HTMLImageElement/ImageBitmap type\n // If HtmlImageElement is used to get image then response type will be HTMLImageElement\n callback(null, data);\n } else if (data) {\n const decoratedCallback = (imgErr?: Error | null, imgResult?: CanvasImageSource | null) => {\n if (imgErr != null) {\n callback(imgErr);\n } else if (imgResult != null) {\n callback(null, imgResult as (HTMLImageElement | ImageBitmap), {cacheControl, expires});\n }\n };\n arrayBufferToCanvasImageSource(data, decoratedCallback);\n }\n if (!itemInQueue.cancelled) {\n itemInQueue.completed = true;\n currentParallelImageRequests--;\n\n processQueue();\n }\n };\n\n /**\n * Process some number of items in the image request queue.\n */\n const processQueue = (): void => {\n\n const maxImageRequests = isThrottled() ?\n config.MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME :\n config.MAX_PARALLEL_IMAGE_REQUESTS;\n\n // limit concurrent image loads to help with raster sources performance on big screens\n for (let numImageRequests = currentParallelImageRequests;\n numImageRequests < maxImageRequests && imageRequestQueue.length > 0;\n numImageRequests++) {\n\n const topItemInQueue: ImageRequestQueueItem = imageRequestQueue.shift();\n if (topItemInQueue.cancelled) {\n numImageRequests--;\n continue;\n }\n\n const innerRequest = doImageRequest(topItemInQueue);\n\n currentParallelImageRequests++;\n\n topItemInQueue.innerRequest = innerRequest;\n }\n };\n\n const getImageUsingHtmlImage = (requestParameters: RequestParameters, callback: GetImageCallback): Cancelable => {\n const image = new Image() as HTMLImageElementWithPriority;\n const url = requestParameters.url;\n let requestCancelled = false;\n const credentials = requestParameters.credentials;\n if (credentials && credentials === 'include') {\n image.crossOrigin = 'use-credentials';\n } else if ((credentials && credentials === 'same-origin') || !sameOrigin(url)) {\n image.crossOrigin = 'anonymous';\n }\n\n image.fetchPriority = 'high';\n image.onload = () => {\n callback(null, image);\n image.onerror = image.onload = null;\n };\n image.onerror = () => {\n if (!requestCancelled) {\n callback(new Error('Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.'));\n }\n image.onerror = image.onload = null;\n };\n image.src = url;\n return {\n cancel: () => {\n requestCancelled = true;\n // Set src to '' to actually cancel the request\n image.src = '';\n }\n };\n };\n}\n\nImageRequest.resetRequestQueue();\n","import type {RequestParameters} from './ajax';\n\n/**\n * A type of MapLibre resource.\n */\nexport const enum ResourceType {\n Glyphs = 'Glyphs',\n Image = 'Image',\n Source = 'Source',\n SpriteImage = 'SpriteImage',\n SpriteJSON = 'SpriteJSON',\n Style = 'Style',\n Tile = 'Tile',\n Unknown = 'Unknown',\n}\n\n/**\n * This function is used to tranform a request.\n * It is used just before executing the relevant request.\n */\nexport type RequestTransformFunction = (url: string, resourceType?: ResourceType) => RequestParameters | undefined;\n\ntype UrlObject = {\n protocol: string;\n authority: string;\n path: string;\n params: Array;\n};\n\nexport class RequestManager {\n _transformRequestFn: RequestTransformFunction;\n\n constructor(transformRequestFn?: RequestTransformFunction) {\n this._transformRequestFn = transformRequestFn;\n }\n\n transformRequest(url: string, type: ResourceType) {\n if (this._transformRequestFn) {\n return this._transformRequestFn(url, type) || {url};\n }\n\n return {url};\n }\n\n normalizeSpriteURL(url: string, format: string, extension: string): string {\n const urlObject = parseUrl(url);\n urlObject.path += `${format}${extension}`;\n return formatUrl(urlObject);\n }\n\n setTransformRequest(transformRequest: RequestTransformFunction) {\n this._transformRequestFn = transformRequest;\n }\n}\n\nconst urlRe = /^(\\w+):\\/\\/([^/?]*)(\\/[^?]+)?\\??(.+)?/;\n\nfunction parseUrl(url: string): UrlObject {\n const parts = url.match(urlRe);\n if (!parts) {\n throw new Error(`Unable to parse URL \"${url}\"`);\n }\n return {\n protocol: parts[1],\n authority: parts[2],\n path: parts[3] || '/',\n params: parts[4] ? parts[4].split('&') : []\n };\n}\n\nfunction formatUrl(obj: UrlObject): string {\n const params = obj.params.length ? `?${obj.params.join('&')}` : '';\n return `${obj.protocol}://${obj.authority}${obj.path}${params}`;\n}\n","import * as glMatrix from \"./common.js\";\n/**\n * 3 Dimensional Vector\n * @module vec3\n */\n\n/**\n * Creates a new, empty vec3\n *\n * @returns {vec3} a new 3D vector\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(3);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[0] = 0;\n out[1] = 0;\n out[2] = 0;\n }\n\n return out;\n}\n/**\n * Creates a new vec3 initialized with values from an existing vector\n *\n * @param {ReadonlyVec3} a vector to clone\n * @returns {vec3} a new 3D vector\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(3);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n return out;\n}\n/**\n * Calculates the length of a vec3\n *\n * @param {ReadonlyVec3} a vector to calculate length of\n * @returns {Number} length of a\n */\n\nexport function length(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n return Math.hypot(x, y, z);\n}\n/**\n * Creates a new vec3 initialized with the given values\n *\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @returns {vec3} a new 3D vector\n */\n\nexport function fromValues(x, y, z) {\n var out = new glMatrix.ARRAY_TYPE(3);\n out[0] = x;\n out[1] = y;\n out[2] = z;\n return out;\n}\n/**\n * Copy the values from one vec3 to another\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the source vector\n * @returns {vec3} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n return out;\n}\n/**\n * Set the components of a vec3 to the given values\n *\n * @param {vec3} out the receiving vector\n * @param {Number} x X component\n * @param {Number} y Y component\n * @param {Number} z Z component\n * @returns {vec3} out\n */\n\nexport function set(out, x, y, z) {\n out[0] = x;\n out[1] = y;\n out[2] = z;\n return out;\n}\n/**\n * Adds two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n return out;\n}\n/**\n * Subtracts vector b from vector a\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n return out;\n}\n/**\n * Multiplies two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function multiply(out, a, b) {\n out[0] = a[0] * b[0];\n out[1] = a[1] * b[1];\n out[2] = a[2] * b[2];\n return out;\n}\n/**\n * Divides two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function divide(out, a, b) {\n out[0] = a[0] / b[0];\n out[1] = a[1] / b[1];\n out[2] = a[2] / b[2];\n return out;\n}\n/**\n * Math.ceil the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to ceil\n * @returns {vec3} out\n */\n\nexport function ceil(out, a) {\n out[0] = Math.ceil(a[0]);\n out[1] = Math.ceil(a[1]);\n out[2] = Math.ceil(a[2]);\n return out;\n}\n/**\n * Math.floor the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to floor\n * @returns {vec3} out\n */\n\nexport function floor(out, a) {\n out[0] = Math.floor(a[0]);\n out[1] = Math.floor(a[1]);\n out[2] = Math.floor(a[2]);\n return out;\n}\n/**\n * Returns the minimum of two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function min(out, a, b) {\n out[0] = Math.min(a[0], b[0]);\n out[1] = Math.min(a[1], b[1]);\n out[2] = Math.min(a[2], b[2]);\n return out;\n}\n/**\n * Returns the maximum of two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function max(out, a, b) {\n out[0] = Math.max(a[0], b[0]);\n out[1] = Math.max(a[1], b[1]);\n out[2] = Math.max(a[2], b[2]);\n return out;\n}\n/**\n * Math.round the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to round\n * @returns {vec3} out\n */\n\nexport function round(out, a) {\n out[0] = Math.round(a[0]);\n out[1] = Math.round(a[1]);\n out[2] = Math.round(a[2]);\n return out;\n}\n/**\n * Scales a vec3 by a scalar number\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the vector to scale\n * @param {Number} b amount to scale the vector by\n * @returns {vec3} out\n */\n\nexport function scale(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n return out;\n}\n/**\n * Adds two vec3's after scaling the second operand by a scalar value\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @param {Number} scale the amount to scale b by before adding\n * @returns {vec3} out\n */\n\nexport function scaleAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n return out;\n}\n/**\n * Calculates the euclidian distance between two vec3's\n *\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {Number} distance between a and b\n */\n\nexport function distance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n return Math.hypot(x, y, z);\n}\n/**\n * Calculates the squared euclidian distance between two vec3's\n *\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {Number} squared distance between a and b\n */\n\nexport function squaredDistance(a, b) {\n var x = b[0] - a[0];\n var y = b[1] - a[1];\n var z = b[2] - a[2];\n return x * x + y * y + z * z;\n}\n/**\n * Calculates the squared length of a vec3\n *\n * @param {ReadonlyVec3} a vector to calculate squared length of\n * @returns {Number} squared length of a\n */\n\nexport function squaredLength(a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n return x * x + y * y + z * z;\n}\n/**\n * Negates the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to negate\n * @returns {vec3} out\n */\n\nexport function negate(out, a) {\n out[0] = -a[0];\n out[1] = -a[1];\n out[2] = -a[2];\n return out;\n}\n/**\n * Returns the inverse of the components of a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to invert\n * @returns {vec3} out\n */\n\nexport function inverse(out, a) {\n out[0] = 1.0 / a[0];\n out[1] = 1.0 / a[1];\n out[2] = 1.0 / a[2];\n return out;\n}\n/**\n * Normalize a vec3\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a vector to normalize\n * @returns {vec3} out\n */\n\nexport function normalize(out, a) {\n var x = a[0];\n var y = a[1];\n var z = a[2];\n var len = x * x + y * y + z * z;\n\n if (len > 0) {\n //TODO: evaluate use of glm_invsqrt here?\n len = 1 / Math.sqrt(len);\n }\n\n out[0] = a[0] * len;\n out[1] = a[1] * len;\n out[2] = a[2] * len;\n return out;\n}\n/**\n * Calculates the dot product of two vec3's\n *\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {Number} dot product of a and b\n */\n\nexport function dot(a, b) {\n return a[0] * b[0] + a[1] * b[1] + a[2] * b[2];\n}\n/**\n * Computes the cross product of two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @returns {vec3} out\n */\n\nexport function cross(out, a, b) {\n var ax = a[0],\n ay = a[1],\n az = a[2];\n var bx = b[0],\n by = b[1],\n bz = b[2];\n out[0] = ay * bz - az * by;\n out[1] = az * bx - ax * bz;\n out[2] = ax * by - ay * bx;\n return out;\n}\n/**\n * Performs a linear interpolation between two vec3's\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec3} out\n */\n\nexport function lerp(out, a, b, t) {\n var ax = a[0];\n var ay = a[1];\n var az = a[2];\n out[0] = ax + t * (b[0] - ax);\n out[1] = ay + t * (b[1] - ay);\n out[2] = az + t * (b[2] - az);\n return out;\n}\n/**\n * Performs a hermite interpolation with two control points\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @param {ReadonlyVec3} c the third operand\n * @param {ReadonlyVec3} d the fourth operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec3} out\n */\n\nexport function hermite(out, a, b, c, d, t) {\n var factorTimes2 = t * t;\n var factor1 = factorTimes2 * (2 * t - 3) + 1;\n var factor2 = factorTimes2 * (t - 2) + t;\n var factor3 = factorTimes2 * (t - 1);\n var factor4 = factorTimes2 * (3 - 2 * t);\n out[0] = a[0] * factor1 + b[0] * factor2 + c[0] * factor3 + d[0] * factor4;\n out[1] = a[1] * factor1 + b[1] * factor2 + c[1] * factor3 + d[1] * factor4;\n out[2] = a[2] * factor1 + b[2] * factor2 + c[2] * factor3 + d[2] * factor4;\n return out;\n}\n/**\n * Performs a bezier interpolation with two control points\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the first operand\n * @param {ReadonlyVec3} b the second operand\n * @param {ReadonlyVec3} c the third operand\n * @param {ReadonlyVec3} d the fourth operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec3} out\n */\n\nexport function bezier(out, a, b, c, d, t) {\n var inverseFactor = 1 - t;\n var inverseFactorTimesTwo = inverseFactor * inverseFactor;\n var factorTimes2 = t * t;\n var factor1 = inverseFactorTimesTwo * inverseFactor;\n var factor2 = 3 * t * inverseFactorTimesTwo;\n var factor3 = 3 * factorTimes2 * inverseFactor;\n var factor4 = factorTimes2 * t;\n out[0] = a[0] * factor1 + b[0] * factor2 + c[0] * factor3 + d[0] * factor4;\n out[1] = a[1] * factor1 + b[1] * factor2 + c[1] * factor3 + d[1] * factor4;\n out[2] = a[2] * factor1 + b[2] * factor2 + c[2] * factor3 + d[2] * factor4;\n return out;\n}\n/**\n * Generates a random vector with the given scale\n *\n * @param {vec3} out the receiving vector\n * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned\n * @returns {vec3} out\n */\n\nexport function random(out, scale) {\n scale = scale || 1.0;\n var r = glMatrix.RANDOM() * 2.0 * Math.PI;\n var z = glMatrix.RANDOM() * 2.0 - 1.0;\n var zScale = Math.sqrt(1.0 - z * z) * scale;\n out[0] = Math.cos(r) * zScale;\n out[1] = Math.sin(r) * zScale;\n out[2] = z * scale;\n return out;\n}\n/**\n * Transforms the vec3 with a mat4.\n * 4th vector component is implicitly '1'\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the vector to transform\n * @param {ReadonlyMat4} m matrix to transform with\n * @returns {vec3} out\n */\n\nexport function transformMat4(out, a, m) {\n var x = a[0],\n y = a[1],\n z = a[2];\n var w = m[3] * x + m[7] * y + m[11] * z + m[15];\n w = w || 1.0;\n out[0] = (m[0] * x + m[4] * y + m[8] * z + m[12]) / w;\n out[1] = (m[1] * x + m[5] * y + m[9] * z + m[13]) / w;\n out[2] = (m[2] * x + m[6] * y + m[10] * z + m[14]) / w;\n return out;\n}\n/**\n * Transforms the vec3 with a mat3.\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the vector to transform\n * @param {ReadonlyMat3} m the 3x3 matrix to transform with\n * @returns {vec3} out\n */\n\nexport function transformMat3(out, a, m) {\n var x = a[0],\n y = a[1],\n z = a[2];\n out[0] = x * m[0] + y * m[3] + z * m[6];\n out[1] = x * m[1] + y * m[4] + z * m[7];\n out[2] = x * m[2] + y * m[5] + z * m[8];\n return out;\n}\n/**\n * Transforms the vec3 with a quat\n * Can also be used for dual quaternions. (Multiply it with the real part)\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec3} a the vector to transform\n * @param {ReadonlyQuat} q quaternion to transform with\n * @returns {vec3} out\n */\n\nexport function transformQuat(out, a, q) {\n // benchmarks: https://jsperf.com/quaternion-transform-vec3-implementations-fixed\n var qx = q[0],\n qy = q[1],\n qz = q[2],\n qw = q[3];\n var x = a[0],\n y = a[1],\n z = a[2]; // var qvec = [qx, qy, qz];\n // var uv = vec3.cross([], qvec, a);\n\n var uvx = qy * z - qz * y,\n uvy = qz * x - qx * z,\n uvz = qx * y - qy * x; // var uuv = vec3.cross([], qvec, uv);\n\n var uuvx = qy * uvz - qz * uvy,\n uuvy = qz * uvx - qx * uvz,\n uuvz = qx * uvy - qy * uvx; // vec3.scale(uv, uv, 2 * w);\n\n var w2 = qw * 2;\n uvx *= w2;\n uvy *= w2;\n uvz *= w2; // vec3.scale(uuv, uuv, 2);\n\n uuvx *= 2;\n uuvy *= 2;\n uuvz *= 2; // return vec3.add(out, a, vec3.add(out, uv, uuv));\n\n out[0] = x + uvx + uuvx;\n out[1] = y + uvy + uuvy;\n out[2] = z + uvz + uuvz;\n return out;\n}\n/**\n * Rotate a 3D vector around the x-axis\n * @param {vec3} out The receiving vec3\n * @param {ReadonlyVec3} a The vec3 point to rotate\n * @param {ReadonlyVec3} b The origin of the rotation\n * @param {Number} rad The angle of rotation in radians\n * @returns {vec3} out\n */\n\nexport function rotateX(out, a, b, rad) {\n var p = [],\n r = []; //Translate point to the origin\n\n p[0] = a[0] - b[0];\n p[1] = a[1] - b[1];\n p[2] = a[2] - b[2]; //perform rotation\n\n r[0] = p[0];\n r[1] = p[1] * Math.cos(rad) - p[2] * Math.sin(rad);\n r[2] = p[1] * Math.sin(rad) + p[2] * Math.cos(rad); //translate to correct position\n\n out[0] = r[0] + b[0];\n out[1] = r[1] + b[1];\n out[2] = r[2] + b[2];\n return out;\n}\n/**\n * Rotate a 3D vector around the y-axis\n * @param {vec3} out The receiving vec3\n * @param {ReadonlyVec3} a The vec3 point to rotate\n * @param {ReadonlyVec3} b The origin of the rotation\n * @param {Number} rad The angle of rotation in radians\n * @returns {vec3} out\n */\n\nexport function rotateY(out, a, b, rad) {\n var p = [],\n r = []; //Translate point to the origin\n\n p[0] = a[0] - b[0];\n p[1] = a[1] - b[1];\n p[2] = a[2] - b[2]; //perform rotation\n\n r[0] = p[2] * Math.sin(rad) + p[0] * Math.cos(rad);\n r[1] = p[1];\n r[2] = p[2] * Math.cos(rad) - p[0] * Math.sin(rad); //translate to correct position\n\n out[0] = r[0] + b[0];\n out[1] = r[1] + b[1];\n out[2] = r[2] + b[2];\n return out;\n}\n/**\n * Rotate a 3D vector around the z-axis\n * @param {vec3} out The receiving vec3\n * @param {ReadonlyVec3} a The vec3 point to rotate\n * @param {ReadonlyVec3} b The origin of the rotation\n * @param {Number} rad The angle of rotation in radians\n * @returns {vec3} out\n */\n\nexport function rotateZ(out, a, b, rad) {\n var p = [],\n r = []; //Translate point to the origin\n\n p[0] = a[0] - b[0];\n p[1] = a[1] - b[1];\n p[2] = a[2] - b[2]; //perform rotation\n\n r[0] = p[0] * Math.cos(rad) - p[1] * Math.sin(rad);\n r[1] = p[0] * Math.sin(rad) + p[1] * Math.cos(rad);\n r[2] = p[2]; //translate to correct position\n\n out[0] = r[0] + b[0];\n out[1] = r[1] + b[1];\n out[2] = r[2] + b[2];\n return out;\n}\n/**\n * Get the angle between two 3D vectors\n * @param {ReadonlyVec3} a The first operand\n * @param {ReadonlyVec3} b The second operand\n * @returns {Number} The angle in radians\n */\n\nexport function angle(a, b) {\n var ax = a[0],\n ay = a[1],\n az = a[2],\n bx = b[0],\n by = b[1],\n bz = b[2],\n mag1 = Math.sqrt(ax * ax + ay * ay + az * az),\n mag2 = Math.sqrt(bx * bx + by * by + bz * bz),\n mag = mag1 * mag2,\n cosine = mag && dot(a, b) / mag;\n return Math.acos(Math.min(Math.max(cosine, -1), 1));\n}\n/**\n * Set the components of a vec3 to zero\n *\n * @param {vec3} out the receiving vector\n * @returns {vec3} out\n */\n\nexport function zero(out) {\n out[0] = 0.0;\n out[1] = 0.0;\n out[2] = 0.0;\n return out;\n}\n/**\n * Returns a string representation of a vector\n *\n * @param {ReadonlyVec3} a vector to represent as a string\n * @returns {String} string representation of the vector\n */\n\nexport function str(a) {\n return \"vec3(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \")\";\n}\n/**\n * Returns whether or not the vectors have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyVec3} a The first vector.\n * @param {ReadonlyVec3} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2];\n}\n/**\n * Returns whether or not the vectors have approximately the same elements in the same position.\n *\n * @param {ReadonlyVec3} a The first vector.\n * @param {ReadonlyVec3} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2));\n}\n/**\n * Alias for {@link vec3.subtract}\n * @function\n */\n\nexport var sub = subtract;\n/**\n * Alias for {@link vec3.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link vec3.divide}\n * @function\n */\n\nexport var div = divide;\n/**\n * Alias for {@link vec3.distance}\n * @function\n */\n\nexport var dist = distance;\n/**\n * Alias for {@link vec3.squaredDistance}\n * @function\n */\n\nexport var sqrDist = squaredDistance;\n/**\n * Alias for {@link vec3.length}\n * @function\n */\n\nexport var len = length;\n/**\n * Alias for {@link vec3.squaredLength}\n * @function\n */\n\nexport var sqrLen = squaredLength;\n/**\n * Perform some operation over an array of vec3s.\n *\n * @param {Array} a the array of vectors to iterate over\n * @param {Number} stride Number of elements between the start of each vec3. If 0 assumes tightly packed\n * @param {Number} offset Number of elements to skip at the beginning of the array\n * @param {Number} count Number of vec3s to iterate over. If 0 iterates over entire array\n * @param {Function} fn Function to call for each vector in the array\n * @param {Object} [arg] additional argument to pass to fn\n * @returns {Array} a\n * @function\n */\n\nexport var forEach = function () {\n var vec = create();\n return function (a, stride, offset, count, fn, arg) {\n var i, l;\n\n if (!stride) {\n stride = 3;\n }\n\n if (!offset) {\n offset = 0;\n }\n\n if (count) {\n l = Math.min(count * stride + offset, a.length);\n } else {\n l = a.length;\n }\n\n for (i = offset; i < l; i += stride) {\n vec[0] = a[i];\n vec[1] = a[i + 1];\n vec[2] = a[i + 2];\n fn(vec, vec, arg);\n a[i] = vec[0];\n a[i + 1] = vec[1];\n a[i + 2] = vec[2];\n }\n\n return a;\n };\n}();","import * as glMatrix from \"./common.js\";\n/**\n * 2 Dimensional Vector\n * @module vec2\n */\n\n/**\n * Creates a new, empty vec2\n *\n * @returns {vec2} a new 2D vector\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(2);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[0] = 0;\n out[1] = 0;\n }\n\n return out;\n}\n/**\n * Creates a new vec2 initialized with values from an existing vector\n *\n * @param {ReadonlyVec2} a vector to clone\n * @returns {vec2} a new 2D vector\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(2);\n out[0] = a[0];\n out[1] = a[1];\n return out;\n}\n/**\n * Creates a new vec2 initialized with the given values\n *\n * @param {Number} x X component\n * @param {Number} y Y component\n * @returns {vec2} a new 2D vector\n */\n\nexport function fromValues(x, y) {\n var out = new glMatrix.ARRAY_TYPE(2);\n out[0] = x;\n out[1] = y;\n return out;\n}\n/**\n * Copy the values from one vec2 to another\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the source vector\n * @returns {vec2} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n return out;\n}\n/**\n * Set the components of a vec2 to the given values\n *\n * @param {vec2} out the receiving vector\n * @param {Number} x X component\n * @param {Number} y Y component\n * @returns {vec2} out\n */\n\nexport function set(out, x, y) {\n out[0] = x;\n out[1] = y;\n return out;\n}\n/**\n * Adds two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n return out;\n}\n/**\n * Subtracts vector b from vector a\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n return out;\n}\n/**\n * Multiplies two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function multiply(out, a, b) {\n out[0] = a[0] * b[0];\n out[1] = a[1] * b[1];\n return out;\n}\n/**\n * Divides two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function divide(out, a, b) {\n out[0] = a[0] / b[0];\n out[1] = a[1] / b[1];\n return out;\n}\n/**\n * Math.ceil the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to ceil\n * @returns {vec2} out\n */\n\nexport function ceil(out, a) {\n out[0] = Math.ceil(a[0]);\n out[1] = Math.ceil(a[1]);\n return out;\n}\n/**\n * Math.floor the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to floor\n * @returns {vec2} out\n */\n\nexport function floor(out, a) {\n out[0] = Math.floor(a[0]);\n out[1] = Math.floor(a[1]);\n return out;\n}\n/**\n * Returns the minimum of two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function min(out, a, b) {\n out[0] = Math.min(a[0], b[0]);\n out[1] = Math.min(a[1], b[1]);\n return out;\n}\n/**\n * Returns the maximum of two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec2} out\n */\n\nexport function max(out, a, b) {\n out[0] = Math.max(a[0], b[0]);\n out[1] = Math.max(a[1], b[1]);\n return out;\n}\n/**\n * Math.round the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to round\n * @returns {vec2} out\n */\n\nexport function round(out, a) {\n out[0] = Math.round(a[0]);\n out[1] = Math.round(a[1]);\n return out;\n}\n/**\n * Scales a vec2 by a scalar number\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to scale\n * @param {Number} b amount to scale the vector by\n * @returns {vec2} out\n */\n\nexport function scale(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n return out;\n}\n/**\n * Adds two vec2's after scaling the second operand by a scalar value\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @param {Number} scale the amount to scale b by before adding\n * @returns {vec2} out\n */\n\nexport function scaleAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n return out;\n}\n/**\n * Calculates the euclidian distance between two vec2's\n *\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {Number} distance between a and b\n */\n\nexport function distance(a, b) {\n var x = b[0] - a[0],\n y = b[1] - a[1];\n return Math.hypot(x, y);\n}\n/**\n * Calculates the squared euclidian distance between two vec2's\n *\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {Number} squared distance between a and b\n */\n\nexport function squaredDistance(a, b) {\n var x = b[0] - a[0],\n y = b[1] - a[1];\n return x * x + y * y;\n}\n/**\n * Calculates the length of a vec2\n *\n * @param {ReadonlyVec2} a vector to calculate length of\n * @returns {Number} length of a\n */\n\nexport function length(a) {\n var x = a[0],\n y = a[1];\n return Math.hypot(x, y);\n}\n/**\n * Calculates the squared length of a vec2\n *\n * @param {ReadonlyVec2} a vector to calculate squared length of\n * @returns {Number} squared length of a\n */\n\nexport function squaredLength(a) {\n var x = a[0],\n y = a[1];\n return x * x + y * y;\n}\n/**\n * Negates the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to negate\n * @returns {vec2} out\n */\n\nexport function negate(out, a) {\n out[0] = -a[0];\n out[1] = -a[1];\n return out;\n}\n/**\n * Returns the inverse of the components of a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to invert\n * @returns {vec2} out\n */\n\nexport function inverse(out, a) {\n out[0] = 1.0 / a[0];\n out[1] = 1.0 / a[1];\n return out;\n}\n/**\n * Normalize a vec2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a vector to normalize\n * @returns {vec2} out\n */\n\nexport function normalize(out, a) {\n var x = a[0],\n y = a[1];\n var len = x * x + y * y;\n\n if (len > 0) {\n //TODO: evaluate use of glm_invsqrt here?\n len = 1 / Math.sqrt(len);\n }\n\n out[0] = a[0] * len;\n out[1] = a[1] * len;\n return out;\n}\n/**\n * Calculates the dot product of two vec2's\n *\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {Number} dot product of a and b\n */\n\nexport function dot(a, b) {\n return a[0] * b[0] + a[1] * b[1];\n}\n/**\n * Computes the cross product of two vec2's\n * Note that the cross product must by definition produce a 3D vector\n *\n * @param {vec3} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @returns {vec3} out\n */\n\nexport function cross(out, a, b) {\n var z = a[0] * b[1] - a[1] * b[0];\n out[0] = out[1] = 0;\n out[2] = z;\n return out;\n}\n/**\n * Performs a linear interpolation between two vec2's\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the first operand\n * @param {ReadonlyVec2} b the second operand\n * @param {Number} t interpolation amount, in the range [0-1], between the two inputs\n * @returns {vec2} out\n */\n\nexport function lerp(out, a, b, t) {\n var ax = a[0],\n ay = a[1];\n out[0] = ax + t * (b[0] - ax);\n out[1] = ay + t * (b[1] - ay);\n return out;\n}\n/**\n * Generates a random vector with the given scale\n *\n * @param {vec2} out the receiving vector\n * @param {Number} [scale] Length of the resulting vector. If ommitted, a unit vector will be returned\n * @returns {vec2} out\n */\n\nexport function random(out, scale) {\n scale = scale || 1.0;\n var r = glMatrix.RANDOM() * 2.0 * Math.PI;\n out[0] = Math.cos(r) * scale;\n out[1] = Math.sin(r) * scale;\n return out;\n}\n/**\n * Transforms the vec2 with a mat2\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to transform\n * @param {ReadonlyMat2} m matrix to transform with\n * @returns {vec2} out\n */\n\nexport function transformMat2(out, a, m) {\n var x = a[0],\n y = a[1];\n out[0] = m[0] * x + m[2] * y;\n out[1] = m[1] * x + m[3] * y;\n return out;\n}\n/**\n * Transforms the vec2 with a mat2d\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to transform\n * @param {ReadonlyMat2d} m matrix to transform with\n * @returns {vec2} out\n */\n\nexport function transformMat2d(out, a, m) {\n var x = a[0],\n y = a[1];\n out[0] = m[0] * x + m[2] * y + m[4];\n out[1] = m[1] * x + m[3] * y + m[5];\n return out;\n}\n/**\n * Transforms the vec2 with a mat3\n * 3rd vector component is implicitly '1'\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to transform\n * @param {ReadonlyMat3} m matrix to transform with\n * @returns {vec2} out\n */\n\nexport function transformMat3(out, a, m) {\n var x = a[0],\n y = a[1];\n out[0] = m[0] * x + m[3] * y + m[6];\n out[1] = m[1] * x + m[4] * y + m[7];\n return out;\n}\n/**\n * Transforms the vec2 with a mat4\n * 3rd vector component is implicitly '0'\n * 4th vector component is implicitly '1'\n *\n * @param {vec2} out the receiving vector\n * @param {ReadonlyVec2} a the vector to transform\n * @param {ReadonlyMat4} m matrix to transform with\n * @returns {vec2} out\n */\n\nexport function transformMat4(out, a, m) {\n var x = a[0];\n var y = a[1];\n out[0] = m[0] * x + m[4] * y + m[12];\n out[1] = m[1] * x + m[5] * y + m[13];\n return out;\n}\n/**\n * Rotate a 2D vector\n * @param {vec2} out The receiving vec2\n * @param {ReadonlyVec2} a The vec2 point to rotate\n * @param {ReadonlyVec2} b The origin of the rotation\n * @param {Number} rad The angle of rotation in radians\n * @returns {vec2} out\n */\n\nexport function rotate(out, a, b, rad) {\n //Translate point to the origin\n var p0 = a[0] - b[0],\n p1 = a[1] - b[1],\n sinC = Math.sin(rad),\n cosC = Math.cos(rad); //perform rotation and translate to correct position\n\n out[0] = p0 * cosC - p1 * sinC + b[0];\n out[1] = p0 * sinC + p1 * cosC + b[1];\n return out;\n}\n/**\n * Get the angle between two 2D vectors\n * @param {ReadonlyVec2} a The first operand\n * @param {ReadonlyVec2} b The second operand\n * @returns {Number} The angle in radians\n */\n\nexport function angle(a, b) {\n var x1 = a[0],\n y1 = a[1],\n x2 = b[0],\n y2 = b[1],\n // mag is the product of the magnitudes of a and b\n mag = Math.sqrt(x1 * x1 + y1 * y1) * Math.sqrt(x2 * x2 + y2 * y2),\n // mag &&.. short circuits if mag == 0\n cosine = mag && (x1 * x2 + y1 * y2) / mag; // Math.min(Math.max(cosine, -1), 1) clamps the cosine between -1 and 1\n\n return Math.acos(Math.min(Math.max(cosine, -1), 1));\n}\n/**\n * Set the components of a vec2 to zero\n *\n * @param {vec2} out the receiving vector\n * @returns {vec2} out\n */\n\nexport function zero(out) {\n out[0] = 0.0;\n out[1] = 0.0;\n return out;\n}\n/**\n * Returns a string representation of a vector\n *\n * @param {ReadonlyVec2} a vector to represent as a string\n * @returns {String} string representation of the vector\n */\n\nexport function str(a) {\n return \"vec2(\" + a[0] + \", \" + a[1] + \")\";\n}\n/**\n * Returns whether or not the vectors exactly have the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyVec2} a The first vector.\n * @param {ReadonlyVec2} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1];\n}\n/**\n * Returns whether or not the vectors have approximately the same elements in the same position.\n *\n * @param {ReadonlyVec2} a The first vector.\n * @param {ReadonlyVec2} b The second vector.\n * @returns {Boolean} True if the vectors are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1];\n var b0 = b[0],\n b1 = b[1];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1));\n}\n/**\n * Alias for {@link vec2.length}\n * @function\n */\n\nexport var len = length;\n/**\n * Alias for {@link vec2.subtract}\n * @function\n */\n\nexport var sub = subtract;\n/**\n * Alias for {@link vec2.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link vec2.divide}\n * @function\n */\n\nexport var div = divide;\n/**\n * Alias for {@link vec2.distance}\n * @function\n */\n\nexport var dist = distance;\n/**\n * Alias for {@link vec2.squaredDistance}\n * @function\n */\n\nexport var sqrDist = squaredDistance;\n/**\n * Alias for {@link vec2.squaredLength}\n * @function\n */\n\nexport var sqrLen = squaredLength;\n/**\n * Perform some operation over an array of vec2s.\n *\n * @param {Array} a the array of vectors to iterate over\n * @param {Number} stride Number of elements between the start of each vec2. If 0 assumes tightly packed\n * @param {Number} offset Number of elements to skip at the beginning of the array\n * @param {Number} count Number of vec2s to iterate over. If 0 iterates over entire array\n * @param {Function} fn Function to call for each vector in the array\n * @param {Object} [arg] additional argument to pass to fn\n * @returns {Array} a\n * @function\n */\n\nexport var forEach = function () {\n var vec = create();\n return function (a, stride, offset, count, fn, arg) {\n var i, l;\n\n if (!stride) {\n stride = 2;\n }\n\n if (!offset) {\n offset = 0;\n }\n\n if (count) {\n l = Math.min(count * stride + offset, a.length);\n } else {\n l = a.length;\n }\n\n for (i = offset; i < l; i += stride) {\n vec[0] = a[i];\n vec[1] = a[i + 1];\n fn(vec, vec, arg);\n a[i] = vec[0];\n a[i + 1] = vec[1];\n }\n\n return a;\n };\n}();","import {SpriteSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * Takes a SpriteSpecification value and returns it in its array form. If `undefined` is passed as an input value, an\n * empty array is returned.\n * duplicated entries with identical id/url will be removed in returned array\n * @param sprite - optional sprite to coerce\n * @returns an empty array in case `undefined` is passed; id-url pairs otherwise\n */\nexport function coerceSpriteToArray(sprite?: SpriteSpecification): {id: string; url: string}[] {\n const resultArray: {id: string; url: string}[] = [];\n\n if (typeof sprite === 'string') {\n resultArray.push({id: 'default', url: sprite});\n } else if (sprite && sprite.length > 0) {\n const dedupArray: string[] = [];\n for (const {id, url} of sprite) {\n const key = `${id}${url}`;\n if (dedupArray.indexOf(key) === -1) {\n dedupArray.push(key);\n resultArray.push({id, url});\n }\n }\n }\n\n return resultArray;\n\n}\n","import {getJSON} from '../util/ajax';\nimport {ImageRequest} from '../util/image_request';\nimport {ResourceType} from '../util/request_manager';\n\nimport {browser} from '../util/browser';\nimport {coerceSpriteToArray} from '../util/style';\n\nimport type {SpriteSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {StyleImage} from './style_image';\nimport type {RequestManager} from '../util/request_manager';\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\n\nexport function loadSprite(\n originalSprite: SpriteSpecification,\n requestManager: RequestManager,\n pixelRatio: number,\n callback: Callback<{[spriteName: string]: {[id: string]: StyleImage}}>\n): Cancelable {\n const spriteArray = coerceSpriteToArray(originalSprite);\n const spriteArrayLength = spriteArray.length;\n const format = pixelRatio > 1 ? '@2x' : '';\n\n const combinedRequestsMap: {[requestKey: string]: Cancelable} = {};\n const jsonsMap: {[id: string]: any} = {};\n const imagesMap: {[id: string]: (HTMLImageElement | ImageBitmap)} = {};\n\n for (const {id, url} of spriteArray) {\n const jsonRequestParameters = requestManager.transformRequest(requestManager.normalizeSpriteURL(url, format, '.json'), ResourceType.SpriteJSON);\n const jsonRequestKey = `${id}_${jsonRequestParameters.url}`; // use id_url as requestMap key to make sure it is unique\n combinedRequestsMap[jsonRequestKey] = getJSON(jsonRequestParameters, (err?: Error | null, data?: any | null) => {\n delete combinedRequestsMap[jsonRequestKey];\n jsonsMap[id] = data;\n doOnceCompleted(callback, jsonsMap, imagesMap, err, spriteArrayLength);\n });\n\n const imageRequestParameters = requestManager.transformRequest(requestManager.normalizeSpriteURL(url, format, '.png'), ResourceType.SpriteImage);\n const imageRequestKey = `${id}_${imageRequestParameters.url}`; // use id_url as requestMap key to make sure it is unique\n combinedRequestsMap[imageRequestKey] = ImageRequest.getImage(imageRequestParameters, (err, img) => {\n delete combinedRequestsMap[imageRequestKey];\n imagesMap[id] = img;\n doOnceCompleted(callback, jsonsMap, imagesMap, err, spriteArrayLength);\n });\n }\n\n return {\n cancel() {\n for (const requst of Object.values(combinedRequestsMap)) {\n requst.cancel();\n }\n }\n };\n}\n\n/**\n * @param callbackFunc - the callback function (both erro and success)\n * @param jsonsMap - JSON data map\n * @param imagesMap - image data map\n * @param err - error object\n * @param expectedResultCounter - number of expected JSON or Image results when everything is finished, respectively.\n */\nfunction doOnceCompleted(\n callbackFunc:Callback<{[spriteName: string]: {[id: string]: StyleImage}}>,\n jsonsMap:{[id: string]: any},\n imagesMap:{[id: string]: (HTMLImageElement | ImageBitmap)},\n err: Error,\n expectedResultCounter: number): void {\n\n if (err) {\n callbackFunc(err);\n return;\n }\n\n if (expectedResultCounter !== Object.values(jsonsMap).length || expectedResultCounter !== Object.values(imagesMap).length) {\n // not done yet, nothing to do\n return;\n }\n\n const result = {} as {[spriteName: string]: {[id: string]: StyleImage}};\n for (const spriteName in jsonsMap) {\n result[spriteName] = {};\n\n const context = browser.getImageCanvasContext(imagesMap[spriteName]);\n const json = jsonsMap[spriteName];\n\n for (const id in json) {\n const {width, height, x, y, sdf, pixelRatio, stretchX, stretchY, content} = json[id];\n const spriteData = {width, height, x, y, context};\n result[spriteName][id] = {data: null, pixelRatio, sdf, stretchX, stretchY, content, spriteData};\n }\n }\n\n callbackFunc(null, result);\n}\n","import type {Context} from '../gl/context';\nimport type {RGBAImage, AlphaImage} from '../util/image';\nimport {isImageBitmap} from '../util/util';\n\nexport type TextureFormat = WebGLRenderingContextBase['RGBA'] | WebGLRenderingContextBase['ALPHA'];\nexport type TextureFilter = WebGLRenderingContextBase['LINEAR'] | WebGLRenderingContextBase['LINEAR_MIPMAP_NEAREST'] | WebGLRenderingContextBase['NEAREST'];\nexport type TextureWrap = WebGLRenderingContextBase['REPEAT'] | WebGLRenderingContextBase['CLAMP_TO_EDGE'] | WebGLRenderingContextBase['MIRRORED_REPEAT'];\n\ntype EmptyImage = {\n width: number;\n height: number;\n data: null;\n};\n\ntype DataTextureImage = RGBAImage | AlphaImage | EmptyImage;\nexport type TextureImage = TexImageSource | DataTextureImage;\n\n/**\n * @internal\n * A `Texture` GL related object\n */\nexport class Texture {\n context: Context;\n size: [number, number];\n texture: WebGLTexture;\n format: TextureFormat;\n filter: TextureFilter;\n wrap: TextureWrap;\n useMipmap: boolean;\n\n constructor(context: Context, image: TextureImage, format: TextureFormat, options?: {\n premultiply?: boolean;\n useMipmap?: boolean;\n } | null) {\n this.context = context;\n this.format = format;\n this.texture = context.gl.createTexture();\n this.update(image, options);\n }\n\n update(image: TextureImage, options?: {\n premultiply?: boolean;\n useMipmap?: boolean;\n } | null, position?: {\n x: number;\n y: number;\n }) {\n const {width, height} = image as {width: number; height: number};\n const resize = (!this.size || this.size[0] !== width || this.size[1] !== height) && !position;\n const {context} = this;\n const {gl} = context;\n\n this.useMipmap = Boolean(options && options.useMipmap);\n gl.bindTexture(gl.TEXTURE_2D, this.texture);\n\n context.pixelStoreUnpackFlipY.set(false);\n context.pixelStoreUnpack.set(1);\n context.pixelStoreUnpackPremultiplyAlpha.set(this.format === gl.RGBA && (!options || options.premultiply !== false));\n\n if (resize) {\n this.size = [width, height];\n\n if (image instanceof HTMLImageElement || image instanceof HTMLCanvasElement || image instanceof HTMLVideoElement || image instanceof ImageData || isImageBitmap(image)) {\n gl.texImage2D(gl.TEXTURE_2D, 0, this.format, this.format, gl.UNSIGNED_BYTE, image);\n } else {\n gl.texImage2D(gl.TEXTURE_2D, 0, this.format, width, height, 0, this.format, gl.UNSIGNED_BYTE, (image as DataTextureImage).data);\n }\n\n } else {\n const {x, y} = position || {x: 0, y: 0};\n if (image instanceof HTMLImageElement || image instanceof HTMLCanvasElement || image instanceof HTMLVideoElement || image instanceof ImageData || isImageBitmap(image)) {\n gl.texSubImage2D(gl.TEXTURE_2D, 0, x, y, gl.RGBA, gl.UNSIGNED_BYTE, image);\n } else {\n gl.texSubImage2D(gl.TEXTURE_2D, 0, x, y, width, height, gl.RGBA, gl.UNSIGNED_BYTE, (image as DataTextureImage).data);\n }\n }\n\n if (this.useMipmap && this.isSizePowerOfTwo()) {\n gl.generateMipmap(gl.TEXTURE_2D);\n }\n }\n\n bind(filter: TextureFilter, wrap: TextureWrap, minFilter?: TextureFilter | null) {\n const {context} = this;\n const {gl} = context;\n gl.bindTexture(gl.TEXTURE_2D, this.texture);\n\n if (minFilter === gl.LINEAR_MIPMAP_NEAREST && !this.isSizePowerOfTwo()) {\n minFilter = gl.LINEAR;\n }\n\n if (filter !== this.filter) {\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, filter);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, minFilter || filter);\n this.filter = filter;\n }\n\n if (wrap !== this.wrap) {\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, wrap);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, wrap);\n this.wrap = wrap;\n }\n }\n\n isSizePowerOfTwo() {\n return this.size[0] === this.size[1] && (Math.log(this.size[0]) / Math.LN2) % 1 === 0;\n }\n\n destroy() {\n const {gl} = this.context;\n gl.deleteTexture(this.texture);\n this.texture = null;\n }\n}\n","import {RGBAImage} from '../util/image';\n\nimport type {Map} from '../ui/map';\n\n/**\n * The sprite data\n */\nexport type SpriteOnDemandStyleImage = {\n width: number;\n height: number;\n x: number;\n y: number;\n context: CanvasRenderingContext2D;\n};\n\n/**\n * The style's image metadata\n */\nexport type StyleImageData = {\n data: RGBAImage;\n version?: number;\n hasRenderCallback?: boolean;\n userImage?: StyleImageInterface;\n spriteData?: SpriteOnDemandStyleImage;\n};\n\n/**\n * The style's image metadata\n */\nexport type StyleImageMetadata = {\n /**\n * The ratio of pixels in the image to physical pixels on the screen\n */\n pixelRatio: number;\n /**\n * Whether the image should be interpreted as an SDF image\n */\n sdf: boolean;\n /**\n * If `icon-text-fit` is used in a layer with this image, this option defines the part(s) of the image that can be stretched horizontally.\n */\n stretchX?: Array<[number, number]>;\n /**\n * If `icon-text-fit` is used in a layer with this image, this option defines the part(s) of the image that can be stretched vertically.\n */\n stretchY?: Array<[number, number]>;\n /**\n * If `icon-text-fit` is used in a layer with this image, this option defines the part of the image that can be covered by the content in `text-field`.\n */\n content?: [number, number, number, number];\n};\n\n/**\n * the style's image, including data and metedata\n */\nexport type StyleImage = StyleImageData & StyleImageMetadata;\n\n/**\n * Interface for dynamically generated style images. This is a specification for\n * implementers to model: it is not an exported method or class.\n *\n * Images implementing this interface can be redrawn for every frame. They can be used to animate\n * icons and patterns or make them respond to user input. Style images can implement a\n * {@link StyleImageInterface#render} method. The method is called every frame and\n * can be used to update the image.\n *\n * @see [Add an animated icon to the map.](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-animated/)\n *\n * @example\n * ```ts\n * let flashingSquare = {\n * width: 64,\n * height: 64,\n * data: new Uint8Array(64 * 64 * 4),\n *\n * onAdd: function(map) {\n * this.map = map;\n * },\n *\n * render: function() {\n * // keep repainting while the icon is on the map\n * this.map.triggerRepaint();\n *\n * // alternate between black and white based on the time\n * let value = Math.round(Date.now() / 1000) % 2 === 0 ? 255 : 0;\n *\n * // check if image needs to be changed\n * if (value !== this.previousValue) {\n * this.previousValue = value;\n *\n * let bytesPerPixel = 4;\n * for (let x = 0; x < this.width; x++) {\n * for (let y = 0; y < this.height; y++) {\n * let offset = (y * this.width + x) * bytesPerPixel;\n * this.data[offset + 0] = value;\n * this.data[offset + 1] = value;\n * this.data[offset + 2] = value;\n * this.data[offset + 3] = 255;\n * }\n * }\n *\n * // return true to indicate that the image changed\n * return true;\n * }\n * }\n * }\n *\n * map.addImage('flashing_square', flashingSquare);\n * ```\n */\n\nexport interface StyleImageInterface {\n width: number;\n height: number;\n data: Uint8Array | Uint8ClampedArray;\n /**\n * This method is called once before every frame where the icon will be used.\n * The method can optionally update the image's `data` member with a new image.\n *\n * If the method updates the image it must return `true` to commit the change.\n * If the method returns `false` or nothing the image is assumed to not have changed.\n *\n * If updates are infrequent it maybe easier to use {@link Map#updateImage} to update\n * the image instead of implementing this method.\n *\n * @returns `true` if this method updated the image. `false` if the image was not changed.\n */\n render?: () => boolean;\n /**\n * Optional method called when the layer has been added to the Map with {@link Map#addImage}.\n *\n * @param map - The Map this custom layer was just added to.\n */\n onAdd?: (map: Map, id: string) => void;\n /**\n * Optional method called when the icon is removed from the map with {@link Map#removeImage}.\n * This gives the image a chance to clean up resources and event listeners.\n */\n onRemove?: () => void;\n}\n\nexport function renderStyleImage(image: StyleImage) {\n const {userImage} = image;\n if (userImage && userImage.render) {\n const updated = userImage.render();\n if (updated) {\n image.data.replace(new Uint8Array(userImage.data.buffer));\n return true;\n }\n }\n return false;\n}\n","/* eslint-disable key-spacing */\nimport potpack from 'potpack';\n\nimport {Event, ErrorEvent, Evented} from '../util/evented';\nimport {RGBAImage} from '../util/image';\nimport {ImagePosition} from './image_atlas';\nimport {Texture} from './texture';\nimport {renderStyleImage} from '../style/style_image';\nimport {warnOnce} from '../util/util';\n\nimport type {StyleImage} from '../style/style_image';\nimport type {Context} from '../gl/context';\nimport type {PotpackBox} from 'potpack';\nimport type {Callback} from '../types/callback';\n\ntype Pattern = {\n bin: PotpackBox;\n position: ImagePosition;\n};\n\n// When copied into the atlas texture, image data is padded by one pixel on each side. Icon\n// images are padded with fully transparent pixels, while pattern images are padded with a\n// copy of the image data wrapped from the opposite side. In both cases, this ensures the\n// correct behavior of GL_LINEAR texture sampling mode.\nconst padding = 1;\n\n/*\n ImageManager does three things:\n\n 1. Tracks requests for icon images from tile workers and sends responses when the requests are fulfilled.\n 2. Builds a texture atlas for pattern images.\n 3. Rerenders renderable images once per frame\n\n These are disparate responsibilities and should eventually be handled by different classes. When we implement\n data-driven support for `*-pattern`, we'll likely use per-bucket pattern atlases, and that would be a good time\n to refactor this.\n*/\nexport class ImageManager extends Evented {\n images: {[_: string]: StyleImage};\n updatedImages: {[_: string]: boolean};\n callbackDispatchedThisFrame: {[_: string]: boolean};\n loaded: boolean;\n requestors: Array<{\n ids: Array;\n callback: Callback<{[_: string]: StyleImage}>;\n }>;\n\n patterns: {[_: string]: Pattern};\n atlasImage: RGBAImage;\n atlasTexture: Texture;\n dirty: boolean;\n\n constructor() {\n super();\n this.images = {};\n this.updatedImages = {};\n this.callbackDispatchedThisFrame = {};\n this.loaded = false;\n this.requestors = [];\n\n this.patterns = {};\n this.atlasImage = new RGBAImage({width: 1, height: 1});\n this.dirty = true;\n }\n\n isLoaded() {\n return this.loaded;\n }\n\n setLoaded(loaded: boolean) {\n if (this.loaded === loaded) {\n return;\n }\n\n this.loaded = loaded;\n\n if (loaded) {\n for (const {ids, callback} of this.requestors) {\n this._notify(ids, callback);\n }\n this.requestors = [];\n }\n }\n\n getImage(id: string): StyleImage {\n const image = this.images[id];\n\n // Extract sprite image data on demand\n if (image && !image.data && image.spriteData) {\n const spriteData = image.spriteData;\n image.data = new RGBAImage({\n width: spriteData.width,\n height: spriteData.height\n }, spriteData.context.getImageData(\n spriteData.x,\n spriteData.y,\n spriteData.width,\n spriteData.height).data);\n image.spriteData = null;\n }\n\n return image;\n }\n\n addImage(id: string, image: StyleImage) {\n if (this.images[id]) throw new Error(`Image id ${id} already exist, use updateImage instead`);\n if (this._validate(id, image)) {\n this.images[id] = image;\n }\n }\n\n _validate(id: string, image: StyleImage) {\n let valid = true;\n const data = image.data || image.spriteData;\n if (!this._validateStretch(image.stretchX, data && data.width)) {\n this.fire(new ErrorEvent(new Error(`Image \"${id}\" has invalid \"stretchX\" value`)));\n valid = false;\n }\n if (!this._validateStretch(image.stretchY, data && data.height)) {\n this.fire(new ErrorEvent(new Error(`Image \"${id}\" has invalid \"stretchY\" value`)));\n valid = false;\n }\n if (!this._validateContent(image.content, image)) {\n this.fire(new ErrorEvent(new Error(`Image \"${id}\" has invalid \"content\" value`)));\n valid = false;\n }\n return valid;\n }\n\n _validateStretch(stretch: Array<[number, number]>, size: number) {\n if (!stretch) return true;\n let last = 0;\n for (const part of stretch) {\n if (part[0] < last || part[1] < part[0] || size < part[1]) return false;\n last = part[1];\n }\n return true;\n }\n\n _validateContent(content: [number, number, number, number], image: StyleImage) {\n if (!content) return true;\n if (content.length !== 4) return false;\n const spriteData = image.spriteData;\n const width = (spriteData && spriteData.width) || image.data.width;\n const height = (spriteData && spriteData.height) || image.data.height;\n if (content[0] < 0 || width < content[0]) return false;\n if (content[1] < 0 || height < content[1]) return false;\n if (content[2] < 0 || width < content[2]) return false;\n if (content[3] < 0 || height < content[3]) return false;\n if (content[2] < content[0]) return false;\n if (content[3] < content[1]) return false;\n return true;\n }\n\n updateImage(id: string, image: StyleImage, validate = true) {\n const oldImage = this.getImage(id);\n if (validate && (oldImage.data.width !== image.data.width || oldImage.data.height !== image.data.height)) {\n throw new Error(`size mismatch between old image (${oldImage.data.width}x${oldImage.data.height}) and new image (${image.data.width}x${image.data.height}).`);\n }\n image.version = oldImage.version + 1;\n this.images[id] = image;\n this.updatedImages[id] = true;\n }\n\n removeImage(id: string) {\n const image = this.images[id];\n delete this.images[id];\n delete this.patterns[id];\n\n if (image.userImage && image.userImage.onRemove) {\n image.userImage.onRemove();\n }\n }\n\n listImages(): Array {\n return Object.keys(this.images);\n }\n\n getImages(ids: Array, callback: Callback<{[_: string]: StyleImage}>) {\n // If the sprite has been loaded, or if all the icon dependencies are already present\n // (i.e. if they've been added via runtime styling), then notify the requestor immediately.\n // Otherwise, delay notification until the sprite is loaded. At that point, if any of the\n // dependencies are still unavailable, we'll just assume they are permanently missing.\n let hasAllDependencies = true;\n if (!this.isLoaded()) {\n for (const id of ids) {\n if (!this.images[id]) {\n hasAllDependencies = false;\n }\n }\n }\n if (this.isLoaded() || hasAllDependencies) {\n this._notify(ids, callback);\n } else {\n this.requestors.push({ids, callback});\n }\n }\n\n _notify(ids: Array, callback: Callback<{[_: string]: StyleImage}>) {\n const response = {};\n\n for (const id of ids) {\n let image = this.getImage(id);\n\n if (!image) {\n this.fire(new Event('styleimagemissing', {id}));\n //Try to acquire image again in case styleimagemissing has populated it\n image = this.getImage(id);\n }\n\n if (image) {\n // Clone the image so that our own copy of its ArrayBuffer doesn't get transferred.\n response[id] = {\n data: image.data.clone(),\n pixelRatio: image.pixelRatio,\n sdf: image.sdf,\n version: image.version,\n stretchX: image.stretchX,\n stretchY: image.stretchY,\n content: image.content,\n hasRenderCallback: Boolean(image.userImage && image.userImage.render)\n };\n } else {\n warnOnce(`Image \"${id}\" could not be loaded. Please make sure you have added the image with map.addImage() or a \"sprite\" property in your style. You can provide missing images by listening for the \"styleimagemissing\" map event.`);\n }\n }\n\n callback(null, response);\n }\n\n // Pattern stuff\n\n getPixelSize() {\n const {width, height} = this.atlasImage;\n return {width, height};\n }\n\n getPattern(id: string): ImagePosition {\n const pattern = this.patterns[id];\n\n const image = this.getImage(id);\n if (!image) {\n return null;\n }\n\n if (pattern && pattern.position.version === image.version) {\n return pattern.position;\n }\n\n if (!pattern) {\n const w = image.data.width + padding * 2;\n const h = image.data.height + padding * 2;\n const bin = {w, h, x: 0, y: 0};\n const position = new ImagePosition(bin, image);\n this.patterns[id] = {bin, position};\n } else {\n pattern.position.version = image.version;\n }\n\n this._updatePatternAtlas();\n\n return this.patterns[id].position;\n }\n\n bind(context: Context) {\n const gl = context.gl;\n if (!this.atlasTexture) {\n this.atlasTexture = new Texture(context, this.atlasImage, gl.RGBA);\n } else if (this.dirty) {\n this.atlasTexture.update(this.atlasImage);\n this.dirty = false;\n }\n\n this.atlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n }\n\n _updatePatternAtlas() {\n const bins = [];\n for (const id in this.patterns) {\n bins.push(this.patterns[id].bin);\n }\n\n const {w, h} = potpack(bins);\n\n const dst = this.atlasImage;\n dst.resize({width: w || 1, height: h || 1});\n\n for (const id in this.patterns) {\n const {bin} = this.patterns[id];\n const x = bin.x + padding;\n const y = bin.y + padding;\n const src = this.getImage(id).data;\n const w = src.width;\n const h = src.height;\n\n RGBAImage.copy(src, dst, {x: 0, y: 0}, {x, y}, {width: w, height: h});\n\n // Add 1 pixel wrapped padding on each side of the image.\n RGBAImage.copy(src, dst, {x: 0, y: h - 1}, {x, y: y - 1}, {width: w, height: 1}); // T\n RGBAImage.copy(src, dst, {x: 0, y: 0}, {x, y: y + h}, {width: w, height: 1}); // B\n RGBAImage.copy(src, dst, {x: w - 1, y: 0}, {x: x - 1, y}, {width: 1, height: h}); // L\n RGBAImage.copy(src, dst, {x: 0, y: 0}, {x: x + w, y}, {width: 1, height: h}); // R\n }\n\n this.dirty = true;\n }\n\n beginFrame() {\n this.callbackDispatchedThisFrame = {};\n }\n\n dispatchRenderCallbacks(ids: Array) {\n for (const id of ids) {\n\n // the callback for the image was already dispatched for a different frame\n if (this.callbackDispatchedThisFrame[id]) continue;\n this.callbackDispatchedThisFrame[id] = true;\n\n const image = this.getImage(id);\n if (!image) warnOnce(`Image with ID: \"${id}\" was not found`);\n\n const updated = renderStyleImage(image);\n if (updated) {\n this.updateImage(id, image);\n }\n }\n }\n}\n","const INF = 1e20;\n\nexport default class TinySDF {\n constructor({\n fontSize = 24,\n buffer = 3,\n radius = 8,\n cutoff = 0.25,\n fontFamily = 'sans-serif',\n fontWeight = 'normal',\n fontStyle = 'normal'\n } = {}) {\n this.buffer = buffer;\n this.cutoff = cutoff;\n this.radius = radius;\n\n // make the canvas size big enough to both have the specified buffer around the glyph\n // for \"halo\", and account for some glyphs possibly being larger than their font size\n const size = this.size = fontSize + buffer * 4;\n\n const canvas = this._createCanvas(size);\n const ctx = this.ctx = canvas.getContext('2d', {willReadFrequently: true});\n ctx.font = `${fontStyle} ${fontWeight} ${fontSize}px ${fontFamily}`;\n\n ctx.textBaseline = 'alphabetic';\n ctx.textAlign = 'left'; // Necessary so that RTL text doesn't have different alignment\n ctx.fillStyle = 'black';\n\n // temporary arrays for the distance transform\n this.gridOuter = new Float64Array(size * size);\n this.gridInner = new Float64Array(size * size);\n this.f = new Float64Array(size);\n this.z = new Float64Array(size + 1);\n this.v = new Uint16Array(size);\n }\n\n _createCanvas(size) {\n const canvas = document.createElement('canvas');\n canvas.width = canvas.height = size;\n return canvas;\n }\n\n draw(char) {\n const {\n width: glyphAdvance,\n actualBoundingBoxAscent,\n actualBoundingBoxDescent,\n actualBoundingBoxLeft,\n actualBoundingBoxRight\n } = this.ctx.measureText(char);\n\n // The integer/pixel part of the top alignment is encoded in metrics.glyphTop\n // The remainder is implicitly encoded in the rasterization\n const glyphTop = Math.ceil(actualBoundingBoxAscent);\n const glyphLeft = 0;\n\n // If the glyph overflows the canvas size, it will be clipped at the bottom/right\n const glyphWidth = Math.max(0, Math.min(this.size - this.buffer, Math.ceil(actualBoundingBoxRight - actualBoundingBoxLeft)));\n const glyphHeight = Math.min(this.size - this.buffer, glyphTop + Math.ceil(actualBoundingBoxDescent));\n\n const width = glyphWidth + 2 * this.buffer;\n const height = glyphHeight + 2 * this.buffer;\n\n const len = Math.max(width * height, 0);\n const data = new Uint8ClampedArray(len);\n const glyph = {data, width, height, glyphWidth, glyphHeight, glyphTop, glyphLeft, glyphAdvance};\n if (glyphWidth === 0 || glyphHeight === 0) return glyph;\n\n const {ctx, buffer, gridInner, gridOuter} = this;\n ctx.clearRect(buffer, buffer, glyphWidth, glyphHeight);\n ctx.fillText(char, buffer, buffer + glyphTop);\n const imgData = ctx.getImageData(buffer, buffer, glyphWidth, glyphHeight);\n\n // Initialize grids outside the glyph range to alpha 0\n gridOuter.fill(INF, 0, len);\n gridInner.fill(0, 0, len);\n\n for (let y = 0; y < glyphHeight; y++) {\n for (let x = 0; x < glyphWidth; x++) {\n const a = imgData.data[4 * (y * glyphWidth + x) + 3] / 255; // alpha value\n if (a === 0) continue; // empty pixels\n\n const j = (y + buffer) * width + x + buffer;\n\n if (a === 1) { // fully drawn pixels\n gridOuter[j] = 0;\n gridInner[j] = INF;\n\n } else { // aliased pixels\n const d = 0.5 - a;\n gridOuter[j] = d > 0 ? d * d : 0;\n gridInner[j] = d < 0 ? d * d : 0;\n }\n }\n }\n\n edt(gridOuter, 0, 0, width, height, width, this.f, this.v, this.z);\n edt(gridInner, buffer, buffer, glyphWidth, glyphHeight, width, this.f, this.v, this.z);\n\n for (let i = 0; i < len; i++) {\n const d = Math.sqrt(gridOuter[i]) - Math.sqrt(gridInner[i]);\n data[i] = Math.round(255 - 255 * (d / this.radius + this.cutoff));\n }\n\n return glyph;\n }\n}\n\n// 2D Euclidean squared distance transform by Felzenszwalb & Huttenlocher https://cs.brown.edu/~pff/papers/dt-final.pdf\nfunction edt(data, x0, y0, width, height, gridSize, f, v, z) {\n for (let x = x0; x < x0 + width; x++) edt1d(data, y0 * gridSize + x, gridSize, height, f, v, z);\n for (let y = y0; y < y0 + height; y++) edt1d(data, y * gridSize + x0, 1, width, f, v, z);\n}\n\n// 1D squared distance transform\nfunction edt1d(grid, offset, stride, length, f, v, z) {\n v[0] = 0;\n z[0] = -INF;\n z[1] = INF;\n f[0] = grid[offset];\n\n for (let q = 1, k = 0, s = 0; q < length; q++) {\n f[q] = grid[offset + q * stride];\n const q2 = q * q;\n do {\n const r = v[k];\n s = (f[q] - f[r] + q2 - r * r) / (q - r) / 2;\n } while (s <= z[k] && --k > -1);\n\n k++;\n v[k] = q;\n z[k] = s;\n z[k + 1] = INF;\n }\n\n for (let q = 0, k = 0; q < length; q++) {\n while (z[k + 1] < q) k++;\n const r = v[k];\n const qr = q - r;\n grid[offset + q * stride] = f[r] + qr * qr;\n }\n}\n","import {loadGlyphRange} from '../style/load_glyph_range';\n\nimport TinySDF from '@mapbox/tiny-sdf';\nimport {unicodeBlockLookup} from '../util/is_char_in_unicode_block';\nimport {asyncAll} from '../util/util';\nimport {AlphaImage} from '../util/image';\n\nimport type {StyleGlyph} from '../style/style_glyph';\nimport type {RequestManager} from '../util/request_manager';\nimport type {Callback} from '../types/callback';\n\ntype Entry = {\n // null means we've requested the range, but the glyph wasn't included in the result.\n glyphs: {\n [id: number]: StyleGlyph | null;\n };\n requests: {\n [range: number]: Array>;\n };\n ranges: {\n [range: number]: boolean | null;\n };\n tinySDF?: TinySDF;\n};\n\nexport class GlyphManager {\n requestManager: RequestManager;\n localIdeographFontFamily: string;\n entries: {\n [_: string]: Entry;\n };\n url: string;\n\n // exposed as statics to enable stubbing in unit tests\n static loadGlyphRange = loadGlyphRange;\n static TinySDF = TinySDF;\n\n constructor(requestManager: RequestManager, localIdeographFontFamily?: string | null) {\n this.requestManager = requestManager;\n this.localIdeographFontFamily = localIdeographFontFamily;\n this.entries = {};\n }\n\n setURL(url?: string | null) {\n this.url = url;\n }\n\n getGlyphs(glyphs: {\n [stack: string]: Array;\n }, callback: Callback<{\n [stack: string]: {\n [id: number]: StyleGlyph;\n };\n }>) {\n const all = [];\n\n for (const stack in glyphs) {\n for (const id of glyphs[stack]) {\n all.push({stack, id});\n }\n }\n\n asyncAll(all, ({stack, id}, callback: Callback<{\n stack: string;\n id: number;\n glyph: StyleGlyph;\n }>) => {\n let entry = this.entries[stack];\n if (!entry) {\n entry = this.entries[stack] = {\n glyphs: {},\n requests: {},\n ranges: {}\n };\n }\n\n let glyph = entry.glyphs[id];\n if (glyph !== undefined) {\n callback(null, {stack, id, glyph});\n return;\n }\n\n glyph = this._tinySDF(entry, stack, id);\n if (glyph) {\n entry.glyphs[id] = glyph;\n callback(null, {stack, id, glyph});\n return;\n }\n\n const range = Math.floor(id / 256);\n if (range * 256 > 65535) {\n callback(new Error('glyphs > 65535 not supported'));\n return;\n }\n\n if (entry.ranges[range]) {\n callback(null, {stack, id, glyph});\n return;\n }\n\n if (!this.url) {\n callback(new Error('glyphsUrl is not set'));\n return;\n }\n\n let requests = entry.requests[range];\n if (!requests) {\n requests = entry.requests[range] = [];\n GlyphManager.loadGlyphRange(stack, range, this.url, this.requestManager,\n (err, response?: {\n [_: number]: StyleGlyph | null;\n } | null) => {\n if (response) {\n for (const id in response) {\n if (!this._doesCharSupportLocalGlyph(+id)) {\n entry.glyphs[+id] = response[+id];\n }\n }\n entry.ranges[range] = true;\n }\n for (const cb of requests) {\n cb(err, response);\n }\n delete entry.requests[range];\n });\n }\n\n requests.push((err, result?: {\n [_: number]: StyleGlyph | null;\n } | null) => {\n if (err) {\n callback(err);\n } else if (result) {\n callback(null, {stack, id, glyph: result[id] || null});\n }\n });\n }, (err, glyphs?: Array<{\n stack: string;\n id: number;\n glyph: StyleGlyph;\n }> | null) => {\n if (err) {\n callback(err);\n } else if (glyphs) {\n const result = {};\n\n for (const {stack, id, glyph} of glyphs) {\n // Clone the glyph so that our own copy of its ArrayBuffer doesn't get transferred.\n (result[stack] || (result[stack] = {}))[id] = glyph && {\n id: glyph.id,\n bitmap: glyph.bitmap.clone(),\n metrics: glyph.metrics\n };\n }\n\n callback(null, result);\n }\n });\n }\n\n _doesCharSupportLocalGlyph(id: number): boolean {\n /* eslint-disable new-cap */\n return !!this.localIdeographFontFamily &&\n (unicodeBlockLookup['CJK Unified Ideographs'](id) ||\n unicodeBlockLookup['Hangul Syllables'](id) ||\n unicodeBlockLookup['Hiragana'](id) ||\n unicodeBlockLookup['Katakana'](id));\n /* eslint-enable new-cap */\n }\n\n _tinySDF(entry: Entry, stack: string, id: number): StyleGlyph {\n const fontFamily = this.localIdeographFontFamily;\n if (!fontFamily) {\n return;\n }\n\n if (!this._doesCharSupportLocalGlyph(id)) {\n return;\n }\n\n // Client-generated glyphs are rendered at 2x texture scale,\n // because CJK glyphs are more detailed than others.\n const textureScale = 2;\n\n let tinySDF = entry.tinySDF;\n if (!tinySDF) {\n let fontWeight = '400';\n if (/bold/i.test(stack)) {\n fontWeight = '900';\n } else if (/medium/i.test(stack)) {\n fontWeight = '500';\n } else if (/light/i.test(stack)) {\n fontWeight = '200';\n }\n tinySDF = entry.tinySDF = new GlyphManager.TinySDF({\n fontSize: 24 * textureScale,\n buffer: 3 * textureScale,\n radius: 8 * textureScale,\n cutoff: 0.25,\n fontFamily,\n fontWeight\n });\n }\n\n const char = tinySDF.draw(String.fromCharCode(id));\n\n /**\n * TinySDF's \"top\" is the distance from the alphabetic baseline to the top of the glyph.\n * Server-generated fonts specify \"top\" relative to an origin above the em box (the origin\n * comes from FreeType, but I'm unclear on exactly how it's derived)\n * ref: https://github.com/mapbox/sdf-glyph-foundry\n *\n * Server fonts don't yet include baseline information, so we can't line up exactly with them\n * (and they don't line up with each other)\n * ref: https://github.com/mapbox/node-fontnik/pull/160\n *\n * To approximately align TinySDF glyphs with server-provided glyphs, we use this baseline adjustment\n * factor calibrated to be in between DIN Pro and Arial Unicode (but closer to Arial Unicode)\n */\n const topAdjustment = 27.5;\n\n const leftAdjustment = 0.5;\n\n return {\n id,\n bitmap: new AlphaImage({width: char.width || 30 * textureScale, height: char.height || 30 * textureScale}, char.data),\n metrics: {\n width: char.glyphWidth / textureScale || 24,\n height: char.glyphHeight / textureScale || 24,\n left: (char.glyphLeft / textureScale + leftAdjustment) || 0,\n top: char.glyphTop / textureScale - topAdjustment || -8,\n advance: char.glyphAdvance / textureScale || 24,\n isDoubleResolution: true\n }\n };\n }\n}\n","import {getArrayBuffer} from '../util/ajax';\nimport {ResourceType} from '../util/request_manager';\n\nimport {parseGlyphPbf} from './parse_glyph_pbf';\n\nimport type {StyleGlyph} from './style_glyph';\nimport type {RequestManager} from '../util/request_manager';\nimport type {Callback} from '../types/callback';\n\nexport function loadGlyphRange(fontstack: string,\n range: number,\n urlTemplate: string,\n requestManager: RequestManager,\n callback: Callback<{\n [_: number]: StyleGlyph | null;\n }>) {\n const begin = range * 256;\n const end = begin + 255;\n\n const request = requestManager.transformRequest(\n urlTemplate.replace('{fontstack}', fontstack).replace('{range}', `${begin}-${end}`),\n ResourceType.Glyphs\n );\n\n getArrayBuffer(request, (err?: Error | null, data?: ArrayBuffer | null) => {\n if (err) {\n callback(err);\n } else if (data) {\n const glyphs = {};\n\n for (const glyph of parseGlyphPbf(data)) {\n glyphs[glyph.id] = glyph;\n }\n\n callback(null, glyphs);\n }\n });\n}\n","import {interpolates, Color, latest as styleSpec} from '@maplibre/maplibre-gl-style-spec';\n\nimport {extend, sphericalToCartesian} from '../util/util';\nimport {Evented} from '../util/evented';\nimport {\n validateStyle,\n validateLight,\n emitValidationErrors\n} from './validate_style';\n\nimport type {StylePropertySpecification, LightSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {EvaluationParameters} from './evaluation_parameters';\nimport type {StyleSetterOptions} from '../style/style';\nimport {Properties, Transitionable, Transitioning, PossiblyEvaluated, DataConstantProperty} from './properties';\n\nimport type {\n Property,\n PropertyValue,\n TransitionParameters\n} from './properties';\n\ntype LightPosition = {\n x: number;\n y: number;\n z: number;\n};\n\nclass LightPositionProperty implements Property<[number, number, number], LightPosition> {\n specification: StylePropertySpecification;\n\n constructor() {\n this.specification = styleSpec.light.position as StylePropertySpecification;\n }\n\n possiblyEvaluate(\n value: PropertyValue<[number, number, number], LightPosition>,\n parameters: EvaluationParameters\n ): LightPosition {\n return sphericalToCartesian(value.expression.evaluate(parameters));\n }\n\n interpolate(a: LightPosition, b: LightPosition, t: number): LightPosition {\n return {\n x: interpolates.number(a.x, b.x, t),\n y: interpolates.number(a.y, b.y, t),\n z: interpolates.number(a.z, b.z, t),\n };\n }\n}\n\ntype Props = {\n 'anchor': DataConstantProperty<'map' | 'viewport'>;\n 'position': LightPositionProperty;\n 'color': DataConstantProperty;\n 'intensity': DataConstantProperty;\n};\n\ntype PropsPossiblyEvaluated = {\n 'anchor': 'map' | 'viewport';\n 'position': LightPosition;\n 'color': Color;\n 'intensity': number;\n};\n\nconst TRANSITION_SUFFIX = '-transition';\n\nlet lightProperties: Properties;\n\n/*\n * Represents the light used to light extruded features.\n */\nexport class Light extends Evented {\n _transitionable: Transitionable;\n _transitioning: Transitioning;\n properties: PossiblyEvaluated;\n\n constructor(lightOptions?: LightSpecification) {\n super();\n lightProperties = lightProperties || new Properties({\n 'anchor': new DataConstantProperty(styleSpec.light.anchor as StylePropertySpecification),\n 'position': new LightPositionProperty(),\n 'color': new DataConstantProperty(styleSpec.light.color as StylePropertySpecification),\n 'intensity': new DataConstantProperty(styleSpec.light.intensity as StylePropertySpecification),\n });\n this._transitionable = new Transitionable(lightProperties);\n this.setLight(lightOptions);\n this._transitioning = this._transitionable.untransitioned();\n }\n\n getLight(): LightSpecification {\n return this._transitionable.serialize();\n }\n\n setLight(light?: LightSpecification, options: StyleSetterOptions = {}) {\n if (this._validate(validateLight, light, options)) {\n return;\n }\n\n for (const name in light) {\n const value = light[name];\n if (name.endsWith(TRANSITION_SUFFIX)) {\n this._transitionable.setTransition(name.slice(0, -TRANSITION_SUFFIX.length) as keyof Props, value);\n } else {\n this._transitionable.setValue(name as keyof Props, value);\n }\n }\n }\n\n updateTransitions(parameters: TransitionParameters) {\n this._transitioning = this._transitionable.transitioned(parameters, this._transitioning);\n }\n\n hasTransition() {\n return this._transitioning.hasTransition();\n }\n\n recalculate(parameters: EvaluationParameters) {\n this.properties = this._transitioning.possiblyEvaluate(parameters);\n }\n\n _validate(validate: Function, value: unknown, options?: {\n validate?: boolean;\n }) {\n if (options && options.validate === false) {\n return false;\n }\n\n return emitValidationErrors(this, validate.call(validateStyle, extend({\n value,\n // Workaround for https://github.com/mapbox/mapbox-gl-js/issues/2407\n style: {glyphs: true, sprite: true},\n styleSpec\n })));\n }\n}\n","import {warnOnce} from '../util/util';\n\nimport type {Context} from '../gl/context';\n\n/**\n * A dash entry\n */\ntype DashEntry = {\n y: number;\n height: number;\n width: number;\n}\n\n/**\n * @internal\n * A LineAtlas lets us reuse rendered dashed lines\n * by writing many of them to a texture and then fetching their positions\n * using {@link LineAtlas#getDash}.\n *\n * @param width - the width\n * @param height - the height\n */\nexport class LineAtlas {\n width: number;\n height: number;\n nextRow: number;\n bytes: number;\n data: Uint8Array;\n dashEntry: {[_: string]: DashEntry};\n dirty: boolean;\n texture: WebGLTexture;\n\n constructor(width: number, height: number) {\n this.width = width;\n this.height = height;\n this.nextRow = 0;\n\n this.data = new Uint8Array(this.width * this.height);\n\n this.dashEntry = {};\n }\n\n /**\n * Get or create a dash line pattern.\n *\n * @param dasharray - the key (represented by numbers) to get the dash texture\n * @param round - whether to add circle caps in between dash segments\n * @returns position of dash texture in {@link DashEntry}\n */\n getDash(dasharray: Array, round: boolean) {\n const key = dasharray.join(',') + String(round);\n\n if (!this.dashEntry[key]) {\n this.dashEntry[key] = this.addDash(dasharray, round);\n }\n return this.dashEntry[key];\n }\n\n getDashRanges(dasharray: Array, lineAtlasWidth: number, stretch: number) {\n // If dasharray has an odd length, both the first and last parts\n // are dashes and should be joined seamlessly.\n const oddDashArray = dasharray.length % 2 === 1;\n\n const ranges = [];\n\n let left = oddDashArray ? -dasharray[dasharray.length - 1] * stretch : 0;\n let right = dasharray[0] * stretch;\n let isDash = true;\n\n ranges.push({left, right, isDash, zeroLength: dasharray[0] === 0});\n\n let currentDashLength = dasharray[0];\n for (let i = 1; i < dasharray.length; i++) {\n isDash = !isDash;\n\n const dashLength = dasharray[i];\n left = currentDashLength * stretch;\n currentDashLength += dashLength;\n right = currentDashLength * stretch;\n\n ranges.push({left, right, isDash, zeroLength: dashLength === 0});\n }\n\n return ranges;\n }\n\n addRoundDash(ranges: any, stretch: number, n: number) {\n const halfStretch = stretch / 2;\n\n for (let y = -n; y <= n; y++) {\n const row = this.nextRow + n + y;\n const index = this.width * row;\n let currIndex = 0;\n let range = ranges[currIndex];\n\n for (let x = 0; x < this.width; x++) {\n if (x / range.right > 1) { range = ranges[++currIndex]; }\n\n const distLeft = Math.abs(x - range.left);\n const distRight = Math.abs(x - range.right);\n const minDist = Math.min(distLeft, distRight);\n let signedDistance;\n\n const distMiddle = y / n * (halfStretch + 1);\n if (range.isDash) {\n const distEdge = halfStretch - Math.abs(distMiddle);\n signedDistance = Math.sqrt(minDist * minDist + distEdge * distEdge);\n } else {\n signedDistance = halfStretch - Math.sqrt(minDist * minDist + distMiddle * distMiddle);\n }\n\n this.data[index + x] = Math.max(0, Math.min(255, signedDistance + 128));\n }\n }\n }\n\n addRegularDash(ranges: any) {\n\n // Collapse any zero-length range\n // Collapse neighbouring same-type parts into a single part\n for (let i = ranges.length - 1; i >= 0; --i) {\n const part = ranges[i];\n const next = ranges[i + 1];\n if (part.zeroLength) {\n ranges.splice(i, 1);\n } else if (next && next.isDash === part.isDash) {\n next.left = part.left;\n ranges.splice(i, 1);\n }\n }\n\n // Combine the first and last parts if possible\n const first = ranges[0];\n const last = ranges[ranges.length - 1];\n if (first.isDash === last.isDash) {\n first.left = last.left - this.width;\n last.right = first.right + this.width;\n }\n\n const index = this.width * this.nextRow;\n let currIndex = 0;\n let range = ranges[currIndex];\n\n for (let x = 0; x < this.width; x++) {\n if (x / range.right > 1) {\n range = ranges[++currIndex];\n }\n\n const distLeft = Math.abs(x - range.left);\n const distRight = Math.abs(x - range.right);\n\n const minDist = Math.min(distLeft, distRight);\n const signedDistance = range.isDash ? minDist : -minDist;\n\n this.data[index + x] = Math.max(0, Math.min(255, signedDistance + 128));\n }\n }\n\n addDash(dasharray: Array, round: boolean): DashEntry {\n const n = round ? 7 : 0;\n const height = 2 * n + 1;\n\n if (this.nextRow + height > this.height) {\n warnOnce('LineAtlas out of space');\n return null;\n }\n\n let length = 0;\n for (let i = 0; i < dasharray.length; i++) { length += dasharray[i]; }\n\n if (length !== 0) {\n const stretch = this.width / length;\n const ranges = this.getDashRanges(dasharray, this.width, stretch);\n\n if (round) {\n this.addRoundDash(ranges, stretch, n);\n } else {\n this.addRegularDash(ranges);\n }\n }\n\n const dashEntry = {\n y: (this.nextRow + n + 0.5) / this.height,\n height: 2 * n / this.height,\n width: length\n };\n\n this.nextRow += height;\n this.dirty = true;\n\n return dashEntry;\n }\n\n bind(context: Context) {\n const gl = context.gl;\n if (!this.texture) {\n this.texture = gl.createTexture();\n gl.bindTexture(gl.TEXTURE_2D, this.texture);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.REPEAT);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.REPEAT);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n gl.texImage2D(gl.TEXTURE_2D, 0, gl.ALPHA, this.width, this.height, 0, gl.ALPHA, gl.UNSIGNED_BYTE, this.data);\n\n } else {\n gl.bindTexture(gl.TEXTURE_2D, this.texture);\n\n if (this.dirty) {\n this.dirty = false;\n gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, this.width, this.height, gl.ALPHA, gl.UNSIGNED_BYTE, this.data);\n }\n }\n }\n}\n","import {asyncAll} from './util';\nimport {Actor, GlyphsProvider, MessageType} from './actor';\n\nimport type {WorkerPool} from './worker_pool';\nimport type {WorkerSource} from '../source/worker_source'; /* eslint-disable-line */ // this is used for the docs' import\n/**\n * Responsible for sending messages from a {@link Source} to an associated\n * {@link WorkerSource}.\n */\nexport class Dispatcher {\n workerPool: WorkerPool;\n actors: Array;\n currentActor: number;\n id: string | number;\n\n constructor(workerPool: WorkerPool, parent: GlyphsProvider, mapId: string | number) {\n this.workerPool = workerPool;\n this.actors = [];\n this.currentActor = 0;\n this.id = mapId;\n const workers = this.workerPool.acquire(mapId);\n for (let i = 0; i < workers.length; i++) {\n const worker = workers[i];\n const actor = new Actor(worker, parent, mapId);\n actor.name = `Worker ${i}`;\n this.actors.push(actor);\n }\n if (!this.actors.length) throw new Error('No actors found');\n }\n\n /**\n * Broadcast a message to all Workers.\n */\n broadcast(type: MessageType, data: unknown, cb?: (...args: any[]) => any) {\n cb = cb || function () {};\n asyncAll(this.actors, (actor, done) => {\n actor.send(type, data, done);\n }, cb);\n }\n\n /**\n * Acquires an actor to dispatch messages to. The actors are distributed in round-robin fashion.\n * @returns An actor object backed by a web worker for processing messages.\n */\n getActor(): Actor {\n this.currentActor = (this.currentActor + 1) % this.actors.length;\n return this.actors[this.currentActor];\n }\n\n remove(mapRemoved: boolean = true) {\n this.actors.forEach((actor) => { actor.remove(); });\n this.actors = [];\n if (mapRemoved) this.workerPool.release(this.id);\n }\n}\n","import {pick, extend} from '../util/util';\n\nimport {getJSON} from '../util/ajax';\nimport {ResourceType} from '../util/request_manager';\nimport {browser} from '../util/browser';\n\nimport type {RequestManager} from '../util/request_manager';\nimport type {Callback} from '../types/callback';\nimport type {TileJSON} from '../types/tilejson';\nimport type {Cancelable} from '../types/cancelable';\nimport type {RasterDEMSourceSpecification, RasterSourceSpecification, VectorSourceSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport function loadTileJson(\n options: RasterSourceSpecification | RasterDEMSourceSpecification | VectorSourceSpecification,\n requestManager: RequestManager,\n callback: Callback\n): Cancelable {\n const loaded = function(err: Error, tileJSON: any) {\n if (err) {\n return callback(err);\n } else if (tileJSON) {\n const result: any = pick(\n // explicit source options take precedence over TileJSON\n extend(tileJSON, options),\n ['tiles', 'minzoom', 'maxzoom', 'attribution', 'bounds', 'scheme', 'tileSize', 'encoding']\n );\n\n if (tileJSON.vector_layers) {\n result.vectorLayers = tileJSON.vector_layers;\n result.vectorLayerIds = result.vectorLayers.map((layer) => { return layer.id; });\n }\n\n callback(null, result);\n }\n };\n\n if (options.url) {\n return getJSON(requestManager.transformRequest(options.url, ResourceType.Source), loaded);\n } else {\n return browser.frame(() => loaded(null, options));\n }\n}\n","import {LngLat} from './lng_lat';\nimport type {LngLatLike} from './lng_lat';\n\n/**\n * A {@link LngLatBounds} object, an array of {@link LngLatLike} objects in [sw, ne] order,\n * or an array of numbers in [west, south, east, north] order.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let v1 = new maplibregl.LngLatBounds(\n * new maplibregl.LngLat(-73.9876, 40.7661),\n * new maplibregl.LngLat(-73.9397, 40.8002)\n * );\n * let v2 = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002])\n * let v3 = [[-73.9876, 40.7661], [-73.9397, 40.8002]];\n * ```\n */\nexport type LngLatBoundsLike = LngLatBounds | [LngLatLike, LngLatLike] | [number, number, number, number];\n\n/**\n * A `LngLatBounds` object represents a geographical bounding box,\n * defined by its southwest and northeast points in longitude and latitude.\n *\n * If no arguments are provided to the constructor, a `null` bounding box is created.\n *\n * Note that any Mapbox GL method that accepts a `LngLatBounds` object as an argument or option\n * can also accept an `Array` of two {@link LngLatLike} constructs and will perform an implicit conversion.\n * This flexible type is documented as {@link LngLatBoundsLike}.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let sw = new maplibregl.LngLat(-73.9876, 40.7661);\n * let ne = new maplibregl.LngLat(-73.9397, 40.8002);\n * let llb = new maplibregl.LngLatBounds(sw, ne);\n * ```\n */\nexport class LngLatBounds {\n _ne: LngLat;\n _sw: LngLat;\n\n /**\n * @param sw - The southwest corner of the bounding box.\n * OR array of 4 numbers in the order of west, south, east, north\n * OR array of 2 LngLatLike: [sw,ne]\n * @param ne - The northeast corner of the bounding box.\n * @example\n * ```ts\n * let sw = new maplibregl.LngLat(-73.9876, 40.7661);\n * let ne = new maplibregl.LngLat(-73.9397, 40.8002);\n * let llb = new maplibregl.LngLatBounds(sw, ne);\n * ```\n * OR\n * ```ts\n * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661, -73.9397, 40.8002]);\n * ```\n * OR\n * ```ts\n * let llb = new maplibregl.LngLatBounds([sw, ne]);\n * ```\n */\n constructor(sw?: LngLatLike | [number, number, number, number] | [LngLatLike, LngLatLike], ne?: LngLatLike) {\n if (!sw) {\n // noop\n } else if (ne) {\n this.setSouthWest(sw).setNorthEast(ne);\n } else if (Array.isArray(sw)) {\n if (sw.length === 4) {\n // 4 element array: west, south, east, north\n this.setSouthWest([sw[0], sw[1]]).setNorthEast([sw[2], sw[3]]);\n } else {\n this.setSouthWest(sw[0] as LngLatLike).setNorthEast(sw[1] as LngLatLike);\n }\n }\n }\n\n /**\n * Set the northeast corner of the bounding box\n *\n * @param ne - a {@link LngLatLike} object describing the northeast corner of the bounding box.\n * @returns `this`\n */\n setNorthEast(ne: LngLatLike): this {\n this._ne = ne instanceof LngLat ? new LngLat(ne.lng, ne.lat) : LngLat.convert(ne);\n return this;\n }\n\n /**\n * Set the southwest corner of the bounding box\n *\n * @param sw - a {@link LngLatLike} object describing the southwest corner of the bounding box.\n * @returns `this`\n */\n setSouthWest(sw: LngLatLike): this {\n this._sw = sw instanceof LngLat ? new LngLat(sw.lng, sw.lat) : LngLat.convert(sw);\n return this;\n }\n\n /**\n * Extend the bounds to include a given LngLatLike or LngLatBoundsLike.\n *\n * @param obj - object to extend to\n * @returns `this`\n */\n extend(obj: LngLatLike | LngLatBoundsLike): this {\n const sw = this._sw,\n ne = this._ne;\n let sw2, ne2;\n\n if (obj instanceof LngLat) {\n sw2 = obj;\n ne2 = obj;\n\n } else if (obj instanceof LngLatBounds) {\n sw2 = obj._sw;\n ne2 = obj._ne;\n\n if (!sw2 || !ne2) return this;\n\n } else {\n if (Array.isArray(obj)) {\n if (obj.length === 4 || (obj as any[]).every(Array.isArray)) {\n const lngLatBoundsObj = (obj as any as LngLatBoundsLike);\n return this.extend(LngLatBounds.convert(lngLatBoundsObj));\n } else {\n const lngLatObj = (obj as any as LngLatLike);\n return this.extend(LngLat.convert(lngLatObj));\n }\n\n } else if (obj && ('lng' in obj || 'lon' in obj) && 'lat' in obj) {\n return this.extend(LngLat.convert(obj));\n }\n\n return this;\n }\n\n if (!sw && !ne) {\n this._sw = new LngLat(sw2.lng, sw2.lat);\n this._ne = new LngLat(ne2.lng, ne2.lat);\n\n } else {\n sw.lng = Math.min(sw2.lng, sw.lng);\n sw.lat = Math.min(sw2.lat, sw.lat);\n ne.lng = Math.max(ne2.lng, ne.lng);\n ne.lat = Math.max(ne2.lat, ne.lat);\n }\n\n return this;\n }\n\n /**\n * Returns the geographical coordinate equidistant from the bounding box's corners.\n *\n * @returns The bounding box's center.\n * @example\n * ```ts\n * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]);\n * llb.getCenter(); // = LngLat {lng: -73.96365, lat: 40.78315}\n * ```\n */\n getCenter(): LngLat {\n return new LngLat((this._sw.lng + this._ne.lng) / 2, (this._sw.lat + this._ne.lat) / 2);\n }\n\n /**\n * Returns the southwest corner of the bounding box.\n *\n * @returns The southwest corner of the bounding box.\n */\n getSouthWest(): LngLat { return this._sw; }\n\n /**\n * Returns the northeast corner of the bounding box.\n *\n * @returns The northeast corner of the bounding box.\n */\n getNorthEast(): LngLat { return this._ne; }\n\n /**\n * Returns the northwest corner of the bounding box.\n *\n * @returns The northwest corner of the bounding box.\n */\n getNorthWest(): LngLat { return new LngLat(this.getWest(), this.getNorth()); }\n\n /**\n * Returns the southeast corner of the bounding box.\n *\n * @returns The southeast corner of the bounding box.\n */\n getSouthEast(): LngLat { return new LngLat(this.getEast(), this.getSouth()); }\n\n /**\n * Returns the west edge of the bounding box.\n *\n * @returns The west edge of the bounding box.\n */\n getWest(): number { return this._sw.lng; }\n\n /**\n * Returns the south edge of the bounding box.\n *\n * @returns The south edge of the bounding box.\n */\n getSouth(): number { return this._sw.lat; }\n\n /**\n * Returns the east edge of the bounding box.\n *\n * @returns The east edge of the bounding box.\n */\n getEast(): number { return this._ne.lng; }\n\n /**\n * Returns the north edge of the bounding box.\n *\n * @returns The north edge of the bounding box.\n */\n getNorth(): number { return this._ne.lat; }\n\n /**\n * Returns the bounding box represented as an array.\n *\n * @returns The bounding box represented as an array, consisting of the\n * southwest and northeast coordinates of the bounding represented as arrays of numbers.\n * @example\n * ```ts\n * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]);\n * llb.toArray(); // = [[-73.9876, 40.7661], [-73.9397, 40.8002]]\n * ```\n */\n toArray() {\n return [this._sw.toArray(), this._ne.toArray()];\n }\n\n /**\n * Return the bounding box represented as a string.\n *\n * @returns The bounding box represents as a string of the format\n * `'LngLatBounds(LngLat(lng, lat), LngLat(lng, lat))'`.\n * @example\n * ```ts\n * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]);\n * llb.toString(); // = \"LngLatBounds(LngLat(-73.9876, 40.7661), LngLat(-73.9397, 40.8002))\"\n * ```\n */\n toString() {\n return `LngLatBounds(${this._sw.toString()}, ${this._ne.toString()})`;\n }\n\n /**\n * Check if the bounding box is an empty/`null`-type box.\n *\n * @returns True if bounds have been defined, otherwise false.\n */\n isEmpty() {\n return !(this._sw && this._ne);\n }\n\n /**\n * Check if the point is within the bounding box.\n *\n * @param lnglat - geographic point to check against.\n * @returns `true` if the point is within the bounding box.\n * @example\n * ```ts\n * let llb = new maplibregl.LngLatBounds(\n * new maplibregl.LngLat(-73.9876, 40.7661),\n * new maplibregl.LngLat(-73.9397, 40.8002)\n * );\n *\n * let ll = new maplibregl.LngLat(-73.9567, 40.7789);\n *\n * console.log(llb.contains(ll)); // = true\n * ```\n */\n contains(lnglat: LngLatLike) {\n const {lng, lat} = LngLat.convert(lnglat);\n\n const containsLatitude = this._sw.lat <= lat && lat <= this._ne.lat;\n let containsLongitude = this._sw.lng <= lng && lng <= this._ne.lng;\n if (this._sw.lng > this._ne.lng) { // wrapped coordinates\n containsLongitude = this._sw.lng >= lng && lng >= this._ne.lng;\n }\n\n return containsLatitude && containsLongitude;\n }\n\n /**\n * Converts an array to a `LngLatBounds` object.\n *\n * If a `LngLatBounds` object is passed in, the function returns it unchanged.\n *\n * Internally, the function calls `LngLat#convert` to convert arrays to `LngLat` values.\n *\n * @param input - An array of two coordinates to convert, or a `LngLatBounds` object to return.\n * @returns A new `LngLatBounds` object, if a conversion occurred, or the original `LngLatBounds` object.\n * @example\n * ```ts\n * let arr = [[-73.9876, 40.7661], [-73.9397, 40.8002]];\n * let llb = maplibregl.LngLatBounds.convert(arr); // = LngLatBounds {_sw: LngLat {lng: -73.9876, lat: 40.7661}, _ne: LngLat {lng: -73.9397, lat: 40.8002}}\n * ```\n */\n static convert(input: LngLatBoundsLike | null): LngLatBounds {\n if (input instanceof LngLatBounds) return input;\n if (!input) return input as null;\n return new LngLatBounds(input);\n }\n\n /**\n * Returns a `LngLatBounds` from the coordinates extended by a given `radius`. The returned `LngLatBounds` completely contains the `radius`.\n *\n * @param center - center coordinates of the new bounds.\n * @param radius - Distance in meters from the coordinates to extend the bounds.\n * @returns A new `LngLatBounds` object representing the coordinates extended by the `radius`.\n * @example\n * ```ts\n * let center = new maplibregl.LngLat(-73.9749, 40.7736);\n * maplibregl.LngLatBounds.fromLngLat(100).toArray(); // = [[-73.97501862141328, 40.77351016847229], [-73.97478137858673, 40.77368983152771]]\n * ```\n */\n static fromLngLat(center: LngLat, radius:number = 0): LngLatBounds {\n const earthCircumferenceInMetersAtEquator = 40075017;\n const latAccuracy = 360 * radius / earthCircumferenceInMetersAtEquator,\n lngAccuracy = latAccuracy / Math.cos((Math.PI / 180) * center.lat);\n\n return new LngLatBounds(new LngLat(center.lng - lngAccuracy, center.lat - latAccuracy),\n new LngLat(center.lng + lngAccuracy, center.lat + latAccuracy));\n }\n}\n","import {LngLatBounds, LngLatBoundsLike} from '../geo/lng_lat_bounds';\nimport {mercatorXfromLng, mercatorYfromLat} from '../geo/mercator_coordinate';\n\nimport type {CanonicalTileID} from './tile_id';\n\nexport class TileBounds {\n bounds: LngLatBounds;\n minzoom: number;\n maxzoom: number;\n\n constructor(bounds: [number, number, number, number], minzoom?: number | null, maxzoom?: number | null) {\n this.bounds = LngLatBounds.convert(this.validateBounds(bounds));\n this.minzoom = minzoom || 0;\n this.maxzoom = maxzoom || 24;\n }\n\n validateBounds(bounds: [number, number, number, number]): LngLatBoundsLike {\n // make sure the bounds property contains valid longitude and latitudes\n if (!Array.isArray(bounds) || bounds.length !== 4) return [-180, -90, 180, 90];\n return [Math.max(-180, bounds[0]), Math.max(-90, bounds[1]), Math.min(180, bounds[2]), Math.min(90, bounds[3])];\n }\n\n contains(tileID: CanonicalTileID) {\n const worldSize = Math.pow(2, tileID.z);\n const level = {\n minX: Math.floor(mercatorXfromLng(this.bounds.getWest()) * worldSize),\n minY: Math.floor(mercatorYfromLat(this.bounds.getNorth()) * worldSize),\n maxX: Math.ceil(mercatorXfromLng(this.bounds.getEast()) * worldSize),\n maxY: Math.ceil(mercatorYfromLat(this.bounds.getSouth()) * worldSize)\n };\n const hit = tileID.x >= level.minX && tileID.x < level.maxX && tileID.y >= level.minY && tileID.y < level.maxY;\n return hit;\n }\n}\n","import {Event, ErrorEvent, Evented} from '../util/evented';\n\nimport {extend, pick} from '../util/util';\nimport {loadTileJson} from './load_tilejson';\nimport {TileBounds} from './tile_bounds';\nimport {ResourceType} from '../util/request_manager';\n\nimport type {Source} from './source';\nimport type {OverscaledTileID} from './tile_id';\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\nimport type {VectorSourceSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\n\nexport type VectorTileSourceOptions = VectorSourceSpecification & {\n collectResourceTiming?: boolean;\n}\n\n/**\n * A source containing vector tiles in [Mapbox Vector Tile format](https://docs.mapbox.com/vector-tiles/reference/).\n * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * map.addSource('some id', {\n * type: 'vector',\n * url: 'https://demotiles.maplibre.org/tiles/tiles.json'\n * });\n * ```\n *\n * @example\n * ```ts\n * map.addSource('some id', {\n * type: 'vector',\n * tiles: ['https://d25uarhxywzl1j.cloudfront.net/v0.1/{z}/{x}/{y}.mvt'],\n * minzoom: 6,\n * maxzoom: 14\n * });\n * ```\n *\n * @example\n * ```ts\n * map.getSource('some id').setUrl(\"https://demotiles.maplibre.org/tiles/tiles.json\");\n * ```\n *\n * @example\n * ```ts\n * map.getSource('some id').setTiles(['https://d25uarhxywzl1j.cloudfront.net/v0.1/{z}/{x}/{y}.mvt']);\n * ```\n * @see [Add a vector tile source](https://maplibre.org/maplibre-gl-js/docs/examples/vector-source/)\n */\nexport class VectorTileSource extends Evented implements Source {\n type: 'vector';\n id: string;\n minzoom: number;\n maxzoom: number;\n url: string;\n scheme: string;\n tileSize: number;\n promoteId: PromoteIdSpecification;\n\n _options: VectorSourceSpecification;\n _collectResourceTiming: boolean;\n dispatcher: Dispatcher;\n map: Map;\n bounds: [number, number, number, number];\n tiles: Array;\n tileBounds: TileBounds;\n reparseOverscaled: boolean;\n isTileClipped: boolean;\n _tileJSONRequest: Cancelable;\n _loaded: boolean;\n\n constructor(id: string, options: VectorTileSourceOptions, dispatcher: Dispatcher, eventedParent: Evented) {\n super();\n this.id = id;\n this.dispatcher = dispatcher;\n\n this.type = 'vector';\n this.minzoom = 0;\n this.maxzoom = 22;\n this.scheme = 'xyz';\n this.tileSize = 512;\n this.reparseOverscaled = true;\n this.isTileClipped = true;\n this._loaded = false;\n\n extend(this, pick(options, ['url', 'scheme', 'tileSize', 'promoteId']));\n this._options = extend({type: 'vector'}, options);\n\n this._collectResourceTiming = options.collectResourceTiming;\n\n if (this.tileSize !== 512) {\n throw new Error('vector tile sources must have a tileSize of 512');\n }\n\n this.setEventedParent(eventedParent);\n }\n\n load = () => {\n this._loaded = false;\n this.fire(new Event('dataloading', {dataType: 'source'}));\n this._tileJSONRequest = loadTileJson(this._options, this.map._requestManager, (err, tileJSON) => {\n this._tileJSONRequest = null;\n this._loaded = true;\n this.map.style.sourceCaches[this.id].clearTiles();\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (tileJSON) {\n extend(this, tileJSON);\n if (tileJSON.bounds) this.tileBounds = new TileBounds(tileJSON.bounds, this.minzoom, this.maxzoom);\n\n // `content` is included here to prevent a race condition where `Style#_updateSources` is called\n // before the TileJSON arrives. this makes sure the tiles needed are loaded once TileJSON arrives\n // ref: https://github.com/mapbox/mapbox-gl-js/pull/4347#discussion_r104418088\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'}));\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'}));\n }\n });\n };\n\n loaded(): boolean {\n return this._loaded;\n }\n\n hasTile(tileID: OverscaledTileID) {\n return !this.tileBounds || this.tileBounds.contains(tileID.canonical);\n }\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n }\n\n setSourceProperty(callback: Function) {\n if (this._tileJSONRequest) {\n this._tileJSONRequest.cancel();\n }\n\n callback();\n\n this.load();\n }\n\n /**\n * Sets the source `tiles` property and re-renders the map.\n *\n * @param tiles - An array of one or more tile source URLs, as in the TileJSON spec.\n * @returns `this`\n */\n setTiles(tiles: Array): this {\n this.setSourceProperty(() => {\n this._options.tiles = tiles;\n });\n\n return this;\n }\n\n /**\n * Sets the source `url` property and re-renders the map.\n *\n * @param url - A URL to a TileJSON resource. Supported protocols are `http:` and `https:`.\n * @returns `this`\n */\n setUrl(url: string): this {\n this.setSourceProperty(() => {\n this.url = url;\n this._options.url = url;\n });\n\n return this;\n }\n\n onRemove() {\n if (this._tileJSONRequest) {\n this._tileJSONRequest.cancel();\n this._tileJSONRequest = null;\n }\n }\n\n serialize = (): VectorSourceSpecification => {\n return extend({}, this._options);\n };\n\n loadTile(tile: Tile, callback: Callback) {\n const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme);\n const params = {\n request: this.map._requestManager.transformRequest(url, ResourceType.Tile),\n uid: tile.uid,\n tileID: tile.tileID,\n zoom: tile.tileID.overscaledZ,\n tileSize: this.tileSize * tile.tileID.overscaleFactor(),\n type: this.type,\n source: this.id,\n pixelRatio: this.map.getPixelRatio(),\n showCollisionBoxes: this.map.showCollisionBoxes,\n promoteId: this.promoteId\n };\n params.request.collectResourceTiming = this._collectResourceTiming;\n\n if (!tile.actor || tile.state === 'expired') {\n tile.actor = this.dispatcher.getActor();\n tile.request = tile.actor.send('loadTile', params, done.bind(this));\n } else if (tile.state === 'loading') {\n // schedule tile reloading after it has been loaded\n tile.reloadCallback = callback;\n } else {\n tile.request = tile.actor.send('reloadTile', params, done.bind(this));\n }\n\n function done(err, data) {\n delete tile.request;\n\n if (tile.aborted)\n return callback(null);\n\n if (err && err.status !== 404) {\n return callback(err);\n }\n\n if (data && data.resourceTiming)\n tile.resourceTiming = data.resourceTiming;\n\n if (this.map._refreshExpiredTiles && data) tile.setExpiryData(data);\n tile.loadVectorData(data, this.map.painter);\n\n callback(null);\n\n if (tile.reloadCallback) {\n this.loadTile(tile, tile.reloadCallback);\n tile.reloadCallback = null;\n }\n }\n }\n\n abortTile(tile: Tile) {\n if (tile.request) {\n tile.request.cancel();\n delete tile.request;\n }\n if (tile.actor) {\n tile.actor.send('abortTile', {uid: tile.uid, type: this.type, source: this.id}, undefined);\n }\n }\n\n unloadTile(tile: Tile) {\n tile.unloadVectorData();\n if (tile.actor) {\n tile.actor.send('removeTile', {uid: tile.uid, type: this.type, source: this.id}, undefined);\n }\n }\n\n hasTransition() {\n return false;\n }\n}\n","import {extend, pick} from '../util/util';\n\nimport {ImageRequest} from '../util/image_request';\n\nimport {ResourceType} from '../util/request_manager';\nimport {Event, ErrorEvent, Evented} from '../util/evented';\nimport {loadTileJson} from './load_tilejson';\nimport {TileBounds} from './tile_bounds';\nimport {Texture} from '../render/texture';\n\nimport type {Source} from './source';\nimport type {OverscaledTileID} from './tile_id';\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {Cancelable} from '../types/cancelable';\nimport type {\n RasterSourceSpecification,\n RasterDEMSourceSpecification\n} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A source containing raster tiles (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * map.addSource('raster-source', {\n * 'type': 'raster',\n * 'tiles': ['https://tiles.stadiamaps.com/tiles/stamen_watercolor/{z}/{x}/{y}.jpg'],\n * 'tileSize': 256,\n * });\n * ```\n *\n * @example\n * ```ts\n * map.addSource('wms-test-source', {\n * 'type': 'raster',\n * // use the tiles option to specify a WMS tile source URL\n * 'tiles': [\n * 'https://img.nj.gov/imagerywms/Natural2015?bbox={bbox-epsg-3857}&format=image/png&service=WMS&version=1.1.1&request=GetMap&srs=EPSG:3857&transparent=true&width=256&height=256&layers=Natural2015'\n * ],\n * 'tileSize': 256\n * });\n * ```\n * @see [Add a raster tile source](https://maplibre.org/maplibre-gl-js/docs/examples/map-tiles/)\n * @see [Add a WMS source](https://maplibre.org/maplibre-gl-js/docs/examples/wms/)\n * @see [Display a satellite map](https://maplibre.org/maplibre-gl-js/docs/examples/satellite-map/)\n */\nexport class RasterTileSource extends Evented implements Source {\n type: 'raster' | 'raster-dem';\n id: string;\n minzoom: number;\n maxzoom: number;\n url: string;\n scheme: string;\n tileSize: number;\n\n bounds: [number, number, number, number];\n tileBounds: TileBounds;\n roundZoom: boolean;\n dispatcher: Dispatcher;\n map: Map;\n tiles: Array;\n\n _loaded: boolean;\n _options: RasterSourceSpecification | RasterDEMSourceSpecification;\n _tileJSONRequest: Cancelable;\n\n constructor(id: string, options: RasterSourceSpecification | RasterDEMSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super();\n this.id = id;\n this.dispatcher = dispatcher;\n this.setEventedParent(eventedParent);\n\n this.type = 'raster';\n this.minzoom = 0;\n this.maxzoom = 22;\n this.roundZoom = true;\n this.scheme = 'xyz';\n this.tileSize = 512;\n this._loaded = false;\n\n this._options = extend({type: 'raster'}, options);\n extend(this, pick(options, ['url', 'scheme', 'tileSize']));\n }\n\n load() {\n this._loaded = false;\n this.fire(new Event('dataloading', {dataType: 'source'}));\n this._tileJSONRequest = loadTileJson(this._options, this.map._requestManager, (err, tileJSON) => {\n this._tileJSONRequest = null;\n this._loaded = true;\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (tileJSON) {\n extend(this, tileJSON);\n if (tileJSON.bounds) this.tileBounds = new TileBounds(tileJSON.bounds, this.minzoom, this.maxzoom);\n\n // `content` is included here to prevent a race condition where `Style#_updateSources` is called\n // before the TileJSON arrives. this makes sure the tiles needed are loaded once TileJSON arrives\n // ref: https://github.com/mapbox/mapbox-gl-js/pull/4347#discussion_r104418088\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'}));\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'}));\n }\n });\n }\n\n loaded(): boolean {\n return this._loaded;\n }\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n }\n\n onRemove() {\n if (this._tileJSONRequest) {\n this._tileJSONRequest.cancel();\n this._tileJSONRequest = null;\n }\n }\n\n setSourceProperty(callback: Function) {\n if (this._tileJSONRequest) {\n this._tileJSONRequest.cancel();\n }\n\n callback();\n\n this.load();\n }\n\n /**\n * Sets the source `tiles` property and re-renders the map.\n *\n * @param tiles - An array of one or more tile source URLs, as in the raster tiles spec (See the [Style Specification](https://maplibre.org/maplibre-style-spec/)\n * @returns `this`\n */\n setTiles(tiles: Array): this {\n this.setSourceProperty(() => {\n this._options.tiles = tiles;\n });\n\n return this;\n }\n\n serialize() {\n return extend({}, this._options);\n }\n\n hasTile(tileID: OverscaledTileID) {\n return !this.tileBounds || this.tileBounds.contains(tileID.canonical);\n }\n\n loadTile(tile: Tile, callback: Callback) {\n const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme);\n tile.request = ImageRequest.getImage(this.map._requestManager.transformRequest(url, ResourceType.Tile), (err, img, expiry) => {\n delete tile.request;\n\n if (tile.aborted) {\n tile.state = 'unloaded';\n callback(null);\n } else if (err) {\n tile.state = 'errored';\n callback(err);\n } else if (img) {\n if (this.map._refreshExpiredTiles && expiry) tile.setExpiryData(expiry);\n\n const context = this.map.painter.context;\n const gl = context.gl;\n tile.texture = this.map.painter.getTileTexture(img.width);\n if (tile.texture) {\n tile.texture.update(img, {useMipmap: true});\n } else {\n tile.texture = new Texture(context, img, gl.RGBA, {useMipmap: true});\n tile.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST);\n\n if (context.extTextureFilterAnisotropic) {\n gl.texParameterf(gl.TEXTURE_2D, context.extTextureFilterAnisotropic.TEXTURE_MAX_ANISOTROPY_EXT, context.extTextureFilterAnisotropicMax);\n }\n }\n\n tile.state = 'loaded';\n\n callback(null);\n }\n }, this.map._refreshExpiredTiles);\n }\n\n abortTile(tile: Tile, callback: Callback) {\n if (tile.request) {\n tile.request.cancel();\n delete tile.request;\n }\n callback();\n }\n\n unloadTile(tile: Tile, callback: Callback) {\n if (tile.texture) this.map.painter.saveTileTexture(tile.texture);\n callback();\n }\n\n hasTransition() {\n return false;\n }\n}\n","import {ImageRequest} from '../util/image_request';\nimport {ResourceType} from '../util/request_manager';\nimport {extend, isImageBitmap, readImageUsingVideoFrame} from '../util/util';\nimport {Evented} from '../util/evented';\nimport {browser} from '../util/browser';\nimport {offscreenCanvasSupported} from '../util/offscreen_canvas_supported';\nimport {OverscaledTileID} from './tile_id';\nimport {RasterTileSource} from './raster_tile_source';\n// ensure DEMData is registered for worker transfer on main thread:\nimport '../data/dem_data';\nimport type {DEMEncoding} from '../data/dem_data';\n\nimport type {Source} from './source';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {RasterDEMSourceSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {ExpiryData} from '../util/ajax';\nimport {isOffscreenCanvasDistorted} from '../util/offscreen_canvas_distorted';\nimport {RGBAImage} from '../util/image';\n\n/**\n * A source containing raster DEM tiles (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.)\n * This source can be used to show hillshading and 3D terrain\n *\n * @group Sources\n *\n * @example\n * ```ts\n * map.addSource('raster-dem-source', {\n * type: 'raster-dem',\n * url: 'https://demotiles.maplibre.org/terrain-tiles/tiles.json',\n * tileSize: 256\n * });\n * ```\n * @see [3D Terrain](https://maplibre.org/maplibre-gl-js/docs/examples/3d-terrain/)\n */\nexport class RasterDEMTileSource extends RasterTileSource implements Source {\n encoding: DEMEncoding;\n redFactor?: number;\n greenFactor?: number;\n blueFactor?: number;\n baseShift?: number;\n\n constructor(id: string, options: RasterDEMSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super(id, options, dispatcher, eventedParent);\n this.type = 'raster-dem';\n this.maxzoom = 22;\n this._options = extend({type: 'raster-dem'}, options);\n this.encoding = options.encoding || 'mapbox';\n this.redFactor = options.redFactor;\n this.greenFactor = options.greenFactor;\n this.blueFactor = options.blueFactor;\n this.baseShift = options.baseShift;\n }\n\n loadTile(tile: Tile, callback: Callback) {\n const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme);\n const request = this.map._requestManager.transformRequest(url, ResourceType.Tile);\n tile.neighboringTiles = this._getNeighboringTiles(tile.tileID);\n tile.request = ImageRequest.getImage(request, async (err: Error, img: (HTMLImageElement | ImageBitmap), expiry: ExpiryData) => {\n delete tile.request;\n if (tile.aborted) {\n tile.state = 'unloaded';\n callback(null);\n } else if (err) {\n tile.state = 'errored';\n callback(err);\n } else if (img) {\n if (this.map._refreshExpiredTiles) tile.setExpiryData(expiry);\n const transfer = isImageBitmap(img) && offscreenCanvasSupported();\n const rawImageData = transfer ? img : await readImageNow(img);\n const params = {\n uid: tile.uid,\n coord: tile.tileID,\n source: this.id,\n rawImageData,\n encoding: this.encoding,\n redFactor: this.redFactor,\n greenFactor: this.greenFactor,\n blueFactor: this.blueFactor,\n baseShift: this.baseShift\n };\n\n if (!tile.actor || tile.state === 'expired') {\n tile.actor = this.dispatcher.getActor();\n tile.actor.send('loadDEMTile', params, done);\n }\n }\n }, this.map._refreshExpiredTiles);\n\n async function readImageNow(img: ImageBitmap | HTMLImageElement): Promise {\n if (typeof VideoFrame !== 'undefined' && isOffscreenCanvasDistorted()) {\n const width = img.width + 2;\n const height = img.height + 2;\n try {\n return new RGBAImage({width, height}, await readImageUsingVideoFrame(img, -1, -1, width, height));\n } catch (e) {\n // fall-back to browser canvas decoding\n }\n }\n return browser.getImageData(img, 1);\n }\n\n function done(err, data) {\n if (err) {\n tile.state = 'errored';\n callback(err);\n }\n\n if (data) {\n tile.dem = data;\n tile.needsHillshadePrepare = true;\n tile.needsTerrainPrepare = true;\n tile.state = 'loaded';\n callback(null);\n }\n }\n }\n\n _getNeighboringTiles(tileID: OverscaledTileID) {\n const canonical = tileID.canonical;\n const dim = Math.pow(2, canonical.z);\n\n const px = (canonical.x - 1 + dim) % dim;\n const pxw = canonical.x === 0 ? tileID.wrap - 1 : tileID.wrap;\n const nx = (canonical.x + 1 + dim) % dim;\n const nxw = canonical.x + 1 === dim ? tileID.wrap + 1 : tileID.wrap;\n\n const neighboringTiles = {};\n // add adjacent tiles\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y).key] = {backfilled: false};\n\n // Add upper neighboringTiles\n if (canonical.y > 0) {\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y - 1).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, tileID.wrap, canonical.z, canonical.x, canonical.y - 1).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y - 1).key] = {backfilled: false};\n }\n // Add lower neighboringTiles\n if (canonical.y + 1 < dim) {\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y + 1).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, tileID.wrap, canonical.z, canonical.x, canonical.y + 1).key] = {backfilled: false};\n neighboringTiles[new OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y + 1).key] = {backfilled: false};\n }\n\n return neighboringTiles;\n }\n\n unloadTile(tile: Tile) {\n if (tile.demTexture) this.map.painter.saveTileTexture(tile.demTexture);\n if (tile.fbo) {\n tile.fbo.destroy();\n delete tile.fbo;\n }\n if (tile.dem) delete tile.dem;\n delete tile.neighboringTiles;\n\n tile.state = 'unloaded';\n if (tile.actor) {\n tile.actor.send('removeDEMTile', {uid: tile.uid, source: this.id});\n }\n }\n}\n","import {Event, ErrorEvent, Evented} from '../util/evented';\n\nimport {extend} from '../util/util';\nimport {EXTENT} from '../data/extent';\nimport {ResourceType} from '../util/request_manager';\nimport {browser} from '../util/browser';\n\nimport type {Source} from './source';\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Actor} from '../util/actor';\nimport type {Callback} from '../types/callback';\nimport type {GeoJSONSourceSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {GeoJSONSourceDiff} from './geojson_source_diff';\nimport type {Options, ClusterProperties} from 'supercluster';\n\nexport type GeoJSONSourceOptions = GeoJSONSourceSpecification & {\n workerOptions?: WorkerOptions;\n collectResourceTiming?: boolean;\n}\n\nexport type GeoJsonSourceOptions = {\n data?: GeoJSON.GeoJSON | string | undefined;\n cluster?: boolean;\n clusterMaxZoom?: number;\n clusterRadius?: number;\n clusterMinPoints?: number;\n generateId?: boolean;\n}\nexport type WorkerOptions = {\n source?: string;\n cluster?: boolean;\n geojsonVtOptions?: {\n buffer?: number;\n tolerance?: number;\n extent?: number;\n maxZoom?: number;\n linemetrics?: boolean;\n generateId?: boolean;\n };\n superclusterOptions?: Options;\n clusterProperties?: ClusterProperties;\n fliter?: any;\n promoteId?: any;\n collectResourceTiming?: boolean;\n}\n\n/**\n * The cluster options to set\n */\nexport type SetClusterOptions = {\n /**\n * Whether or not to cluster\n */\n cluster?: boolean;\n /**\n * The cluster's max zoom\n */\n clusterMaxZoom?: number;\n /**\n * The cluster's radius\n */\n clusterRadius?: number;\n}\n\n/**\n * A source containing GeoJSON.\n * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-geojson) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * map.addSource('some id', {\n * type: 'geojson',\n * data: 'https://d2ad6b4ur7yvpq.cloudfront.net/naturalearth-3.3.0/ne_10m_ports.geojson'\n * });\n * ```\n *\n * @example\n * ```ts\n * map.addSource('some id', {\n * type: 'geojson',\n * data: {\n * \"type\": \"FeatureCollection\",\n * \"features\": [{\n * \"type\": \"Feature\",\n * \"properties\": {},\n * \"geometry\": {\n * \"type\": \"Point\",\n * \"coordinates\": [\n * -76.53063297271729,\n * 39.18174077994108\n * ]\n * }\n * }]\n * }\n * });\n * ```\n *\n * @example\n * ```ts\n * map.getSource('some id').setData({\n * \"type\": \"FeatureCollection\",\n * \"features\": [{\n * \"type\": \"Feature\",\n * \"properties\": { \"name\": \"Null Island\" },\n * \"geometry\": {\n * \"type\": \"Point\",\n * \"coordinates\": [ 0, 0 ]\n * }\n * }]\n * });\n * ```\n * @see [Draw GeoJSON points](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-markers/)\n * @see [Add a GeoJSON line](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-line/)\n * @see [Create a heatmap from points](https://maplibre.org/maplibre-gl-js/docs/examples/heatmap/)\n * @see [Create and style clusters](https://maplibre.org/maplibre-gl-js/docs/examples/cluster/)\n */\nexport class GeoJSONSource extends Evented implements Source {\n type: 'geojson';\n id: string;\n minzoom: number;\n maxzoom: number;\n tileSize: number;\n attribution: string;\n promoteId: PromoteIdSpecification;\n\n isTileClipped: boolean;\n reparseOverscaled: boolean;\n _data: GeoJSON.GeoJSON | string | undefined;\n _options: GeoJsonSourceOptions;\n workerOptions: WorkerOptions;\n map: Map;\n actor: Actor;\n _pendingLoads: number;\n _collectResourceTiming: boolean;\n _removed: boolean;\n\n /** @internal */\n constructor(id: string, options: GeoJSONSourceOptions, dispatcher: Dispatcher, eventedParent: Evented) {\n super();\n\n this.id = id;\n\n // `type` is a property rather than a constant to make it easy for 3rd\n // parties to use GeoJSONSource to build their own source types.\n this.type = 'geojson';\n\n this.minzoom = 0;\n this.maxzoom = 18;\n this.tileSize = 512;\n this.isTileClipped = true;\n this.reparseOverscaled = true;\n this._removed = false;\n this._pendingLoads = 0;\n\n this.actor = dispatcher.getActor();\n this.setEventedParent(eventedParent);\n\n this._data = (options.data as any);\n this._options = extend({}, options);\n\n this._collectResourceTiming = options.collectResourceTiming;\n\n if (options.maxzoom !== undefined) this.maxzoom = options.maxzoom;\n if (options.type) this.type = options.type;\n if (options.attribution) this.attribution = options.attribution;\n this.promoteId = options.promoteId;\n\n const scale = EXTENT / this.tileSize;\n\n // sent to the worker, along with `url: ...` or `data: literal geojson`,\n // so that it can load/parse/index the geojson data\n // extending with `options.workerOptions` helps to make it easy for\n // third-party sources to hack/reuse GeoJSONSource.\n this.workerOptions = extend({\n source: this.id,\n cluster: options.cluster || false,\n geojsonVtOptions: {\n buffer: (options.buffer !== undefined ? options.buffer : 128) * scale,\n tolerance: (options.tolerance !== undefined ? options.tolerance : 0.375) * scale,\n extent: EXTENT,\n maxZoom: this.maxzoom,\n lineMetrics: options.lineMetrics || false,\n generateId: options.generateId || false\n },\n superclusterOptions: {\n maxZoom: options.clusterMaxZoom !== undefined ? options.clusterMaxZoom : this.maxzoom - 1,\n minPoints: Math.max(2, options.clusterMinPoints || 2),\n extent: EXTENT,\n radius: (options.clusterRadius || 50) * scale,\n log: false,\n generateId: options.generateId || false\n },\n clusterProperties: options.clusterProperties,\n filter: options.filter\n }, options.workerOptions);\n\n // send the promoteId to the worker to have more flexible updates, but only if it is a string\n if (typeof this.promoteId === 'string') {\n this.workerOptions.promoteId = this.promoteId;\n }\n }\n\n load = () => {\n this._updateWorkerData();\n };\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n }\n\n /**\n * Sets the GeoJSON data and re-renders the map.\n *\n * @param data - A GeoJSON data object or a URL to one. The latter is preferable in the case of large GeoJSON files.\n * @returns `this`\n */\n setData(data: GeoJSON.GeoJSON | string): this {\n this._data = data;\n this._updateWorkerData();\n\n return this;\n }\n\n /**\n * Updates the source's GeoJSON, and re-renders the map.\n *\n * For sources with lots of features, this method can be used to make updates more quickly.\n *\n * This approach requires unique IDs for every feature in the source. The IDs can either be specified on the feature,\n * or by using the promoteId option to specify which property should be used as the ID.\n *\n * It is an error to call updateData on a source that did not have unique IDs for each of its features already.\n *\n * Updates are applied on a best-effort basis, updating an ID that does not exist will not result in an error.\n *\n * @param diff - The changes that need to be applied.\n * @returns `this`\n */\n updateData(diff: GeoJSONSourceDiff): this {\n this._updateWorkerData(diff);\n\n return this;\n }\n\n /**\n * To disable/enable clustering on the source options\n * @param options - The options to set\n * @returns `this`\n * @example\n * ```ts\n * map.getSource('some id').setClusterOptions({cluster: false});\n * map.getSource('some id').setClusterOptions({cluster: false, clusterRadius: 50, clusterMaxZoom: 14});\n * ```\n */\n setClusterOptions(options: SetClusterOptions): this {\n this.workerOptions.cluster = options.cluster;\n if (options) {\n if (options.clusterRadius !== undefined) this.workerOptions.superclusterOptions.radius = options.clusterRadius;\n if (options.clusterMaxZoom !== undefined) this.workerOptions.superclusterOptions.maxZoom = options.clusterMaxZoom;\n }\n this._updateWorkerData();\n return this;\n }\n\n /**\n * For clustered sources, fetches the zoom at which the given cluster expands.\n *\n * @param clusterId - The value of the cluster's `cluster_id` property.\n * @param callback - A callback to be called when the zoom value is retrieved (`(error, zoom) => { ... }`).\n * @returns `this`\n */\n getClusterExpansionZoom(clusterId: number, callback: Callback): this {\n this.actor.send('geojson.getClusterExpansionZoom', {clusterId, source: this.id}, callback);\n return this;\n }\n\n /**\n * For clustered sources, fetches the children of the given cluster on the next zoom level (as an array of GeoJSON features).\n *\n * @param clusterId - The value of the cluster's `cluster_id` property.\n * @param callback - A callback to be called when the features are retrieved (`(error, features) => { ... }`).\n * @returns `this`\n */\n getClusterChildren(clusterId: number, callback: Callback>): this {\n this.actor.send('geojson.getClusterChildren', {clusterId, source: this.id}, callback);\n return this;\n }\n\n /**\n * For clustered sources, fetches the original points that belong to the cluster (as an array of GeoJSON features).\n *\n * @param clusterId - The value of the cluster's `cluster_id` property.\n * @param limit - The maximum number of features to return.\n * @param offset - The number of features to skip (e.g. for pagination).\n * @param callback - A callback to be called when the features are retrieved (`(error, features) => { ... }`).\n * @returns `this`\n * @example\n * Retrieve cluster leaves on click\n * ```ts\n * map.on('click', 'clusters', function(e) {\n * let features = map.queryRenderedFeatures(e.point, {\n * layers: ['clusters']\n * });\n *\n * let clusterId = features[0].properties.cluster_id;\n * let pointCount = features[0].properties.point_count;\n * let clusterSource = map.getSource('clusters');\n *\n * clusterSource.getClusterLeaves(clusterId, pointCount, 0, function(error, features) {\n * // Print cluster leaves in the console\n * console.log('Cluster leaves:', error, features);\n * })\n * });\n * ```\n */\n getClusterLeaves(clusterId: number, limit: number, offset: number, callback: Callback>): this {\n this.actor.send('geojson.getClusterLeaves', {\n source: this.id,\n clusterId,\n limit,\n offset\n }, callback);\n return this;\n }\n\n /**\n * Responsible for invoking WorkerSource's geojson.loadData target, which\n * handles loading the geojson data and preparing to serve it up as tiles,\n * using geojson-vt or supercluster as appropriate.\n * @param diff - the diff object\n */\n _updateWorkerData(diff?: GeoJSONSourceDiff) {\n const options = extend({}, this.workerOptions);\n if (diff) {\n options.dataDiff = diff;\n } else if (typeof this._data === 'string') {\n options.request = this.map._requestManager.transformRequest(browser.resolveURL(this._data as string), ResourceType.Source);\n options.request.collectResourceTiming = this._collectResourceTiming;\n } else {\n options.data = JSON.stringify(this._data);\n }\n\n this._pendingLoads++;\n this.fire(new Event('dataloading', {dataType: 'source'}));\n\n // target {this.type}.loadData rather than literally geojson.loadData,\n // so that other geojson-like source types can easily reuse this\n // implementation\n this.actor.send(`${this.type}.loadData`, options, (err, result) => {\n this._pendingLoads--;\n\n if (this._removed || (result && result.abandoned)) {\n this.fire(new Event('dataabort', {dataType: 'source'}));\n return;\n }\n\n let resourceTiming = null;\n if (result && result.resourceTiming && result.resourceTiming[this.id])\n resourceTiming = result.resourceTiming[this.id].slice(0);\n\n if (err) {\n this.fire(new ErrorEvent(err));\n return;\n }\n\n const data: any = {dataType: 'source'};\n if (this._collectResourceTiming && resourceTiming && resourceTiming.length > 0)\n extend(data, {resourceTiming});\n\n // although GeoJSON sources contain no metadata, we fire this event to let the SourceCache\n // know its ok to start requesting tiles.\n this.fire(new Event('data', {...data, sourceDataType: 'metadata'}));\n this.fire(new Event('data', {...data, sourceDataType: 'content'}));\n });\n }\n\n loaded(): boolean {\n return this._pendingLoads === 0;\n }\n\n loadTile(tile: Tile, callback: Callback) {\n const message = !tile.actor ? 'loadTile' : 'reloadTile';\n tile.actor = this.actor;\n const params = {\n type: this.type,\n uid: tile.uid,\n tileID: tile.tileID,\n zoom: tile.tileID.overscaledZ,\n maxZoom: this.maxzoom,\n tileSize: this.tileSize,\n source: this.id,\n pixelRatio: this.map.getPixelRatio(),\n showCollisionBoxes: this.map.showCollisionBoxes,\n promoteId: this.promoteId\n };\n\n tile.request = this.actor.send(message, params, (err, data) => {\n delete tile.request;\n tile.unloadVectorData();\n\n if (tile.aborted) {\n return callback(null);\n }\n\n if (err) {\n return callback(err);\n }\n\n tile.loadVectorData(data, this.map.painter, message === 'reloadTile');\n\n return callback(null);\n });\n }\n\n abortTile(tile: Tile) {\n if (tile.request) {\n tile.request.cancel();\n delete tile.request;\n }\n tile.aborted = true;\n }\n\n unloadTile(tile: Tile) {\n tile.unloadVectorData();\n this.actor.send('removeTile', {uid: tile.uid, type: this.type, source: this.id});\n }\n\n onRemove() {\n this._removed = true;\n this.actor.send('removeSource', {type: this.type, source: this.id});\n }\n\n serialize = (): GeoJSONSourceSpecification => {\n return extend({}, this._options, {\n type: this.type,\n data: this._data\n });\n };\n\n hasTransition() {\n return false;\n }\n}\n","import {createLayout} from '../util/struct_array';\n\nexport default createLayout([\n {name: 'a_pos', type: 'Int16', components: 2},\n {name: 'a_texture_pos', type: 'Int16', components: 2}\n]);\n","import {CanonicalTileID} from './tile_id';\nimport {Event, ErrorEvent, Evented} from '../util/evented';\nimport {ImageRequest} from '../util/image_request';\nimport {ResourceType} from '../util/request_manager';\nimport {EXTENT} from '../data/extent';\nimport {RasterBoundsArray} from '../data/array_types.g';\nimport rasterBoundsAttributes from '../data/raster_bounds_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {Texture} from '../render/texture';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\n\nimport type {Source} from './source';\nimport type {CanvasSourceSpecification} from './canvas_source';\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Tile} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {\n ImageSourceSpecification,\n VideoSourceSpecification\n} from '@maplibre/maplibre-gl-style-spec';\nimport {Cancelable} from '../types/cancelable';\n\n/**\n * Four geographical coordinates,\n * represented as arrays of longitude and latitude numbers, which define the corners of the image.\n * The coordinates start at the top left corner of the image and proceed in clockwise order.\n * They do not have to represent a rectangle.\n */\nexport type Coordinates = [[number, number], [number, number], [number, number], [number, number]];\n\n/**\n * The options object for the {@link ImageSource#updateImage} method\n */\nexport type UpdateImageOptions = {\n /**\n * Required image URL.\n */\n url: string;\n /**\n * The image coordinates\n */\n coordinates?: Coordinates;\n}\n\n/**\n * A data source containing an image.\n * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-image) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * // add to map\n * map.addSource('some id', {\n * type: 'image',\n * url: 'https://www.maplibre.org/images/foo.png',\n * coordinates: [\n * [-76.54, 39.18],\n * [-76.52, 39.18],\n * [-76.52, 39.17],\n * [-76.54, 39.17]\n * ]\n * });\n *\n * // update coordinates\n * let mySource = map.getSource('some id');\n * mySource.setCoordinates([\n * [-76.54335737228394, 39.18579907229748],\n * [-76.52803659439087, 39.1838364847587],\n * [-76.5295386314392, 39.17683392507606],\n * [-76.54520273208618, 39.17876344106642]\n * ]);\n *\n * // update url and coordinates simultaneously\n * mySource.updateImage({\n * url: 'https://www.maplibre.org/images/bar.png',\n * coordinates: [\n * [-76.54335737228394, 39.18579907229748],\n * [-76.52803659439087, 39.1838364847587],\n * [-76.5295386314392, 39.17683392507606],\n * [-76.54520273208618, 39.17876344106642]\n * ]\n * })\n *\n * map.removeSource('some id'); // remove\n * ```\n */\nexport class ImageSource extends Evented implements Source {\n type: string;\n id: string;\n minzoom: number;\n maxzoom: number;\n tileSize: number;\n url: string;\n\n coordinates: Coordinates;\n tiles: {[_: string]: Tile};\n options: any;\n dispatcher: Dispatcher;\n map: Map;\n texture: Texture | null;\n image: HTMLImageElement | ImageBitmap;\n tileID: CanonicalTileID;\n _boundsArray: RasterBoundsArray;\n boundsBuffer: VertexBuffer;\n boundsSegments: SegmentVector;\n _loaded: boolean;\n _request: Cancelable;\n\n /** @internal */\n constructor(id: string, options: ImageSourceSpecification | VideoSourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super();\n this.id = id;\n this.dispatcher = dispatcher;\n this.coordinates = options.coordinates;\n\n this.type = 'image';\n this.minzoom = 0;\n this.maxzoom = 22;\n this.tileSize = 512;\n this.tiles = {};\n this._loaded = false;\n\n this.setEventedParent(eventedParent);\n\n this.options = options;\n }\n\n load = (newCoordinates?: Coordinates, successCallback?: () => void) => {\n this._loaded = false;\n this.fire(new Event('dataloading', {dataType: 'source'}));\n\n this.url = this.options.url;\n\n this._request = ImageRequest.getImage(this.map._requestManager.transformRequest(this.url, ResourceType.Image), (err, image) => {\n this._request = null;\n this._loaded = true;\n\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (image) {\n this.image = image;\n if (newCoordinates) {\n this.coordinates = newCoordinates;\n }\n if (successCallback) {\n successCallback();\n }\n this._finishLoading();\n }\n });\n };\n\n loaded(): boolean {\n return this._loaded;\n }\n\n /**\n * Updates the image URL and, optionally, the coordinates. To avoid having the image flash after changing,\n * set the `raster-fade-duration` paint property on the raster layer to 0.\n *\n * @param options - The options object.\n * @returns `this`\n */\n updateImage(options: UpdateImageOptions): this {\n if (!options.url) {\n return this;\n }\n\n if (this._request) {\n this._request.cancel();\n this._request = null;\n }\n\n this.options.url = options.url;\n this.load(options.coordinates, () => { this.texture = null; });\n return this;\n }\n\n _finishLoading() {\n if (this.map) {\n this.setCoordinates(this.coordinates);\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'}));\n }\n }\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n }\n\n onRemove() {\n if (this._request) {\n this._request.cancel();\n this._request = null;\n }\n }\n\n /**\n * Sets the image's coordinates and re-renders the map.\n *\n * @param coordinates - Four geographical coordinates,\n * represented as arrays of longitude and latitude numbers, which define the corners of the image.\n * The coordinates start at the top left corner of the image and proceed in clockwise order.\n * They do not have to represent a rectangle.\n * @returns `this`\n */\n setCoordinates(coordinates: Coordinates): this {\n this.coordinates = coordinates;\n\n // Calculate which mercator tile is suitable for rendering the video in\n // and create a buffer with the corner coordinates. These coordinates\n // may be outside the tile, because raster tiles aren't clipped when rendering.\n\n // transform the geo coordinates into (zoom 0) tile space coordinates\n const cornerCoords = coordinates.map(MercatorCoordinate.fromLngLat);\n\n // Compute the coordinates of the tile we'll use to hold this image's\n // render data\n this.tileID = getCoordinatesCenterTileID(cornerCoords);\n\n // Constrain min/max zoom to our tile's zoom level in order to force\n // SourceCache to request this tile (no matter what the map's zoom\n // level)\n this.minzoom = this.maxzoom = this.tileID.z;\n\n // Transform the corner coordinates into the coordinate space of our\n // tile.\n const tileCoords = cornerCoords.map((coord) => this.tileID.getTilePoint(coord)._round());\n\n this._boundsArray = new RasterBoundsArray();\n this._boundsArray.emplaceBack(tileCoords[0].x, tileCoords[0].y, 0, 0);\n this._boundsArray.emplaceBack(tileCoords[1].x, tileCoords[1].y, EXTENT, 0);\n this._boundsArray.emplaceBack(tileCoords[3].x, tileCoords[3].y, 0, EXTENT);\n this._boundsArray.emplaceBack(tileCoords[2].x, tileCoords[2].y, EXTENT, EXTENT);\n\n if (this.boundsBuffer) {\n this.boundsBuffer.destroy();\n delete this.boundsBuffer;\n }\n\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'}));\n return this;\n }\n\n prepare = () => {\n if (Object.keys(this.tiles).length === 0 || !this.image) {\n return;\n }\n\n const context = this.map.painter.context;\n const gl = context.gl;\n\n if (!this.boundsBuffer) {\n this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members);\n }\n\n if (!this.boundsSegments) {\n this.boundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n }\n\n if (!this.texture) {\n this.texture = new Texture(context, this.image, gl.RGBA);\n this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n }\n\n let newTilesLoaded = false;\n for (const w in this.tiles) {\n const tile = this.tiles[w];\n if (tile.state !== 'loaded') {\n tile.state = 'loaded';\n tile.texture = this.texture;\n newTilesLoaded = true;\n }\n }\n\n if (newTilesLoaded) {\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'idle', sourceId: this.id}));\n }\n };\n\n loadTile(tile: Tile, callback: Callback) {\n // We have a single tile -- whose coordinates are this.tileID -- that\n // covers the image we want to render. If that's the one being\n // requested, set it up with the image; otherwise, mark the tile as\n // `errored` to indicate that we have no data for it.\n // If the world wraps, we may have multiple \"wrapped\" copies of the\n // single tile.\n if (this.tileID && this.tileID.equals(tile.tileID.canonical)) {\n this.tiles[String(tile.tileID.wrap)] = tile;\n tile.buckets = {};\n callback(null);\n } else {\n tile.state = 'errored';\n callback(null);\n }\n }\n\n serialize = (): ImageSourceSpecification | VideoSourceSpecification | CanvasSourceSpecification => {\n return {\n type: 'image',\n url: this.options.url,\n coordinates: this.coordinates\n };\n };\n\n hasTransition() {\n return false;\n }\n}\n\n/**\n * Given a list of coordinates, get their center as a coordinate.\n *\n * @returns centerpoint\n * @internal\n */\nexport function getCoordinatesCenterTileID(coords: Array) {\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n\n for (const coord of coords) {\n minX = Math.min(minX, coord.x);\n minY = Math.min(minY, coord.y);\n maxX = Math.max(maxX, coord.x);\n maxY = Math.max(maxY, coord.y);\n }\n\n const dx = maxX - minX;\n const dy = maxY - minY;\n const dMax = Math.max(dx, dy);\n const zoom = Math.max(0, Math.floor(-Math.log(dMax) / Math.LN2));\n const tilesAtZoom = Math.pow(2, zoom);\n\n return new CanonicalTileID(\n zoom,\n Math.floor((minX + maxX) / 2 * tilesAtZoom),\n Math.floor((minY + maxY) / 2 * tilesAtZoom));\n}\n","import {getVideo} from '../util/ajax';\nimport {ResourceType} from '../util/request_manager';\n\nimport {ImageSource} from './image_source';\nimport rasterBoundsAttributes from '../data/raster_bounds_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {Texture} from '../render/texture';\nimport {Event, ErrorEvent} from '../util/evented';\nimport {ValidationError} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Evented} from '../util/evented';\nimport type {VideoSourceSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A data source containing video.\n * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-video) for detailed documentation of options.)\n *\n * @group Sources\n *\n * @example\n * ```ts\n * // add to map\n * map.addSource('some id', {\n * type: 'video',\n * url: [\n * 'https://www.mapbox.com/blog/assets/baltimore-smoke.mp4',\n * 'https://www.mapbox.com/blog/assets/baltimore-smoke.webm'\n * ],\n * coordinates: [\n * [-76.54, 39.18],\n * [-76.52, 39.18],\n * [-76.52, 39.17],\n * [-76.54, 39.17]\n * ]\n * });\n *\n * // update\n * let mySource = map.getSource('some id');\n * mySource.setCoordinates([\n * [-76.54335737228394, 39.18579907229748],\n * [-76.52803659439087, 39.1838364847587],\n * [-76.5295386314392, 39.17683392507606],\n * [-76.54520273208618, 39.17876344106642]\n * ]);\n *\n * map.removeSource('some id'); // remove\n * ```\n * @see [Add a video](https://maplibre.org/maplibre-gl-js/docs/examples/video-on-a-map/)\n *\n * Note that when rendered as a raster layer, the layer's `raster-fade-duration` property will cause the video to fade in.\n * This happens when playback is started, paused and resumed, or when the video's coordinates are updated. To avoid this behavior,\n * set the layer's `raster-fade-duration` property to `0`.\n */\nexport class VideoSource extends ImageSource {\n options: VideoSourceSpecification;\n urls: Array;\n video: HTMLVideoElement;\n roundZoom: boolean;\n\n constructor(id: string, options: VideoSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super(id, options, dispatcher, eventedParent);\n this.roundZoom = true;\n this.type = 'video';\n this.options = options;\n }\n\n load = () => {\n this._loaded = false;\n const options = this.options;\n\n this.urls = [];\n for (const url of options.urls) {\n this.urls.push(this.map._requestManager.transformRequest(url, ResourceType.Source).url);\n }\n\n getVideo(this.urls, (err, video) => {\n this._loaded = true;\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (video) {\n this.video = video;\n this.video.loop = true;\n\n // Start repainting when video starts playing. hasTransition() will then return\n // true to trigger additional frames as long as the videos continues playing.\n this.video.addEventListener('playing', () => {\n this.map.triggerRepaint();\n });\n\n if (this.map) {\n this.video.play();\n }\n\n this._finishLoading();\n }\n });\n };\n\n /**\n * Pauses the video.\n */\n pause() {\n if (this.video) {\n this.video.pause();\n }\n }\n\n /**\n * Plays the video.\n */\n play() {\n if (this.video) {\n this.video.play();\n }\n }\n\n /**\n * Sets playback to a timestamp, in seconds.\n */\n seek(seconds: number) {\n if (this.video) {\n const seekableRange = this.video.seekable;\n if (seconds < seekableRange.start(0) || seconds > seekableRange.end(0)) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${this.id}`, null, `Playback for this video can be set only between the ${seekableRange.start(0)} and ${seekableRange.end(0)}-second mark.`)));\n } else this.video.currentTime = seconds;\n }\n }\n\n /**\n * Returns the HTML `video` element.\n *\n * @returns The HTML `video` element.\n */\n getVideo(): HTMLVideoElement {\n return this.video;\n }\n\n onAdd(map: Map) {\n if (this.map) return;\n this.map = map;\n this.load();\n if (this.video) {\n this.video.play();\n this.setCoordinates(this.coordinates);\n }\n }\n\n /**\n * Sets the video's coordinates and re-renders the map.\n *\n * @returns `this`\n */\n prepare = (): this => {\n if (Object.keys(this.tiles).length === 0 || this.video.readyState < 2) {\n return; // not enough data for current position\n }\n\n const context = this.map.painter.context;\n const gl = context.gl;\n\n if (!this.boundsBuffer) {\n this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members);\n }\n\n if (!this.boundsSegments) {\n this.boundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n }\n\n if (!this.texture) {\n this.texture = new Texture(context, this.video, gl.RGBA);\n this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n } else if (!this.video.paused) {\n this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, gl.RGBA, gl.UNSIGNED_BYTE, this.video);\n }\n\n let newTilesLoaded = false;\n for (const w in this.tiles) {\n const tile = this.tiles[w];\n if (tile.state !== 'loaded') {\n tile.state = 'loaded';\n tile.texture = this.texture;\n newTilesLoaded = true;\n }\n }\n\n if (newTilesLoaded) {\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'idle', sourceId: this.id}));\n }\n };\n\n serialize = (): VideoSourceSpecification => {\n return {\n type: 'video',\n urls: this.urls,\n coordinates: this.coordinates\n };\n };\n\n hasTransition() {\n return this.video && !this.video.paused;\n }\n}\n","import {ImageSource} from './image_source';\n\nimport rasterBoundsAttributes from '../data/raster_bounds_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {Texture} from '../render/texture';\nimport {Event, ErrorEvent} from '../util/evented';\nimport {ValidationError} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {Map} from '../ui/map';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Evented} from '../util/evented';\n\n/**\n * Options to add a canvas source type to the map.\n */\nexport type CanvasSourceSpecification = {\n /**\n * Source type. Must be `\"canvas\"`.\n */\n type: 'canvas';\n /**\n * Four geographical coordinates denoting where to place the corners of the canvas, specified in `[longitude, latitude]` pairs.\n */\n coordinates: [[number, number], [number, number], [number, number], [number, number]];\n /**\n * Whether the canvas source is animated. If the canvas is static (i.e. pixels do not need to be re-read on every frame), `animate` should be set to `false` to improve performance.\n * @defaultValue true\n */\n animate?: boolean;\n /**\n * Canvas source from which to read pixels. Can be a string representing the ID of the canvas element, or the `HTMLCanvasElement` itself.\n */\n canvas?: string | HTMLCanvasElement;\n};\n\n/**\n * A data source containing the contents of an HTML canvas. See {@link CanvasSourceSpecification} for detailed documentation of options.\n *\n * @group Sources\n *\n * @example\n * ```ts\n * // add to map\n * map.addSource('some id', {\n * type: 'canvas',\n * canvas: 'idOfMyHTMLCanvas',\n * animate: true,\n * coordinates: [\n * [-76.54, 39.18],\n * [-76.52, 39.18],\n * [-76.52, 39.17],\n * [-76.54, 39.17]\n * ]\n * });\n *\n * // update\n * let mySource = map.getSource('some id');\n * mySource.setCoordinates([\n * [-76.54335737228394, 39.18579907229748],\n * [-76.52803659439087, 39.1838364847587],\n * [-76.5295386314392, 39.17683392507606],\n * [-76.54520273208618, 39.17876344106642]\n * ]);\n *\n * map.removeSource('some id'); // remove\n * ```\n */\nexport class CanvasSource extends ImageSource {\n options: CanvasSourceSpecification;\n animate: boolean;\n canvas: HTMLCanvasElement;\n width: number;\n height: number;\n /**\n * Enables animation. The image will be copied from the canvas to the map on each frame.\n */\n play: () => void;\n /**\n * Disables animation. The map will display a static copy of the canvas image.\n */\n pause: () => void;\n _playing: boolean;\n\n /** @internal */\n constructor(id: string, options: CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) {\n super(id, options, dispatcher, eventedParent);\n\n // We build in some validation here, since canvas sources aren't included in the style spec:\n if (!options.coordinates) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, 'missing required property \"coordinates\"')));\n } else if (!Array.isArray(options.coordinates) || options.coordinates.length !== 4 ||\n options.coordinates.some(c => !Array.isArray(c) || c.length !== 2 || c.some(l => typeof l !== 'number'))) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, '\"coordinates\" property must be an array of 4 longitude/latitude array pairs')));\n }\n\n if (options.animate && typeof options.animate !== 'boolean') {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, 'optional \"animate\" property must be a boolean value')));\n }\n\n if (!options.canvas) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, 'missing required property \"canvas\"')));\n } else if (typeof options.canvas !== 'string' && !(options.canvas instanceof HTMLCanvasElement)) {\n this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, '\"canvas\" must be either a string representing the ID of the canvas element from which to read, or an HTMLCanvasElement instance')));\n }\n\n this.options = options;\n this.animate = options.animate !== undefined ? options.animate : true;\n }\n\n load = () => {\n this._loaded = true;\n if (!this.canvas) {\n this.canvas = (this.options.canvas instanceof HTMLCanvasElement) ?\n this.options.canvas :\n document.getElementById(this.options.canvas) as HTMLCanvasElement;\n // cast to HTMLCanvasElement in else of ternary\n // should we do a safety check and throw if it's not actually HTMLCanvasElement?\n }\n this.width = this.canvas.width;\n this.height = this.canvas.height;\n\n if (this._hasInvalidDimensions()) {\n this.fire(new ErrorEvent(new Error('Canvas dimensions cannot be less than or equal to zero.')));\n return;\n }\n\n this.play = function() {\n this._playing = true;\n this.map.triggerRepaint();\n };\n\n this.pause = function() {\n if (this._playing) {\n this.prepare();\n this._playing = false;\n }\n };\n\n this._finishLoading();\n };\n\n /**\n * Returns the HTML `canvas` element.\n *\n * @returns The HTML `canvas` element.\n */\n getCanvas(): HTMLCanvasElement {\n return this.canvas;\n }\n\n onAdd(map: Map) {\n this.map = map;\n this.load();\n if (this.canvas) {\n if (this.animate) this.play();\n }\n }\n\n onRemove() {\n this.pause();\n }\n\n prepare = () => {\n let resize = false;\n if (this.canvas.width !== this.width) {\n this.width = this.canvas.width;\n resize = true;\n }\n if (this.canvas.height !== this.height) {\n this.height = this.canvas.height;\n resize = true;\n }\n\n if (this._hasInvalidDimensions()) return;\n\n if (Object.keys(this.tiles).length === 0) return; // not enough data for current position\n\n const context = this.map.painter.context;\n const gl = context.gl;\n\n if (!this.boundsBuffer) {\n this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members);\n }\n\n if (!this.boundsSegments) {\n this.boundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n }\n\n if (!this.texture) {\n this.texture = new Texture(context, this.canvas, gl.RGBA, {premultiply: true});\n } else if (resize || this._playing) {\n this.texture.update(this.canvas, {premultiply: true});\n }\n\n let newTilesLoaded = false;\n for (const w in this.tiles) {\n const tile = this.tiles[w];\n if (tile.state !== 'loaded') {\n tile.state = 'loaded';\n tile.texture = this.texture;\n newTilesLoaded = true;\n }\n }\n\n if (newTilesLoaded) {\n this.fire(new Event('data', {dataType: 'source', sourceDataType: 'idle', sourceId: this.id}));\n }\n };\n\n serialize = (): CanvasSourceSpecification => {\n return {\n type: 'canvas',\n coordinates: this.coordinates\n };\n };\n\n hasTransition() {\n return this._playing;\n }\n\n _hasInvalidDimensions() {\n for (const x of [this.canvas.width, this.canvas.height]) {\n if (isNaN(x) || x <= 0) return true;\n }\n return false;\n }\n}\n","import {VectorTileSource} from '../source/vector_tile_source';\nimport {RasterTileSource} from '../source/raster_tile_source';\nimport {RasterDEMTileSource} from '../source/raster_dem_tile_source';\nimport {GeoJSONSource} from '../source/geojson_source';\nimport {VideoSource} from '../source/video_source';\nimport {ImageSource} from '../source/image_source';\nimport {CanvasSource} from '../source/canvas_source';\n\nimport type {SourceSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Event, Evented} from '../util/evented';\nimport type {Map} from '../ui/map';\nimport type {Tile} from './tile';\nimport type {OverscaledTileID, CanonicalTileID} from './tile_id';\nimport type {Callback} from '../types/callback';\nimport type {CanvasSourceSpecification} from '../source/canvas_source';\n\nconst registeredSources = {} as {[key:string]: SourceClass};\n\n/**\n * The `Source` interface must be implemented by each source type, including \"core\" types (`vector`, `raster`,\n * `video`, etc.) and all custom, third-party types.\n *\n * @event `data` - Fired with `{dataType: 'source', sourceDataType: 'metadata'}` to indicate that any necessary metadata\n * has been loaded so that it's okay to call `loadTile`; and with `{dataType: 'source', sourceDataType: 'content'}`\n * to indicate that the source data has changed, so that any current caches should be flushed.\n *\n * @group Sources\n */\nexport interface Source {\n readonly type: string;\n /**\n * The id for the source. Must not be used by any existing source.\n */\n id: string;\n minzoom: number;\n maxzoom: number;\n tileSize: number;\n attribution?: string;\n /**\n * `true` if zoom levels are rounded to the nearest integer in the source data, `false` if they are floor-ed to the nearest integer.\n */\n roundZoom?: boolean;\n /**\n * `false` if tiles can be drawn outside their boundaries, `true` if they cannot.\n */\n isTileClipped?: boolean;\n tileID?: CanonicalTileID;\n /**\n * `true` if tiles should be sent back to the worker for each overzoomed zoom level, `false` if not.\n */\n reparseOverscaled?: boolean;\n vectorLayerIds?: Array;\n hasTransition(): boolean;\n loaded(): boolean;\n fire(event: Event): unknown;\n readonly onAdd?: (map: Map) => void;\n readonly onRemove?: (map: Map) => void;\n loadTile(tile: Tile, callback: Callback): void;\n readonly hasTile?: (tileID: OverscaledTileID) => boolean;\n readonly abortTile?: (tile: Tile, callback: Callback) => void;\n readonly unloadTile?: (tile: Tile, callback: Callback) => void;\n /**\n * @returns A plain (stringifiable) JS object representing the current state of the source.\n * Creating a source using the returned object as the `options` should result in a Source that is\n * equivalent to this one.\n */\n serialize(): any;\n readonly prepare?: () => void;\n}\n\n/**\n * A supporting type to the source definition\n */\ntype SourceStatics = {\n /*\n * An optional URL to a script which, when run by a Worker, registers a {@link WorkerSource}\n * implementation for this Source type by calling `self.registerWorkerSource(workerSource: WorkerSource)`.\n */\n workerSourceURL?: URL;\n};\n\n/**\n * A general definition of a {@link Source} class for factory usage\n */\nexport type SourceClass = {\n new (id: string, specification: SourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented): Source;\n} & SourceStatics;\n\n/**\n * Creates a tiled data source instance given an options object.\n *\n * @param id - The id for the source. Must not be used by any existing source.\n * @param specification - Source options, specific to the source type (except for `options.type`, which is always required).\n * @param source - A source definition object compliant with\n * [`maplibre-gl-style-spec`](https://maplibre.org/maplibre-style-spec/#sources) or, for a third-party source type,\n * with that type's requirements.\n * @param dispatcher - A {@link Dispatcher} instance, which can be used to send messages to the workers.\n * @returns a newly created source\n */\nexport const create = (id: string, specification: SourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented): Source => {\n\n const Class = getSourceType(specification.type);\n const source = new Class(id, specification, dispatcher, eventedParent);\n\n if (source.id !== id) {\n throw new Error(`Expected Source id to be ${id} instead of ${source.id}`);\n }\n\n return source;\n};\n\nexport const getSourceType = (name: string): SourceClass => {\n switch (name) {\n case 'geojson':\n return GeoJSONSource;\n case 'image':\n return ImageSource;\n case 'raster':\n return RasterTileSource;\n case 'raster-dem':\n return RasterDEMTileSource;\n case 'vector':\n return VectorTileSource;\n case 'video':\n return VideoSource;\n case 'canvas':\n return CanvasSource;\n }\n return registeredSources[name];\n};\n\nexport const setSourceType = (name: string, type: SourceClass) => {\n registeredSources[name] = type;\n};\n\nexport interface Actor {\n send(type: string, data: any, callback: Callback): void;\n}\n","import type {SourceCache} from './source_cache';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {CollisionIndex} from '../symbol/collision_index';\nimport type {Transform} from '../geo/transform';\nimport type {RetainedQueryData} from '../symbol/placement';\nimport type {FilterSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\nimport type Point from '@mapbox/point-geometry';\nimport {mat4} from 'gl-matrix';\n\n/**\n * Options to pass to query the map for the rendered features\n */\nexport type QueryRenderedFeaturesOptions = {\n /**\n * An array of [style layer IDs](https://maplibre.org/maplibre-style-spec/#layer-id) for the query to inspect.\n * Only features within these layers will be returned. If this parameter is undefined, all layers will be checked.\n */\n layers?: Array;\n /**\n * A [filter](https://maplibre.org/maplibre-style-spec/layers/#filter) to limit query results.\n */\n filter?: FilterSpecification;\n /**\n * An array of string representing the available images\n */\n availableImages?: Array;\n /**\n * Whether to check if the [options.filter] conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function.\n */\n validate?: boolean;\n};\n\n/**\n * The options object related to the {@link Map#queryRenderedFeatures} method\n */\nexport type QuerySourceFeatureOptions = {\n /**\n * The name of the source layer to query. *For vector tile sources, this parameter is required.* For GeoJSON sources, it is ignored.\n */\n sourceLayer?: string;\n /**\n * A [filter](https://maplibre.org/maplibre-style-spec/layers/#filter)\n * to limit query results.\n */\n filter?: FilterSpecification;\n /**\n * Whether to check if the [parameters.filter] conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function.\n * @defaultValue true\n */\n validate?: boolean;\n}\n\n/*\n * Returns a matrix that can be used to convert from tile coordinates to viewport pixel coordinates.\n */\nfunction getPixelPosMatrix(transform, tileID) {\n const t = mat4.create();\n mat4.translate(t, t, [1, 1, 0]);\n mat4.scale(t, t, [transform.width * 0.5, transform.height * 0.5, 1]);\n return mat4.multiply(t, t, transform.calculatePosMatrix(tileID.toUnwrapped()));\n}\n\nfunction queryIncludes3DLayer(layers: Array, styleLayers: {[_: string]: StyleLayer}, sourceID: string) {\n if (layers) {\n for (const layerID of layers) {\n const layer = styleLayers[layerID];\n if (layer && layer.source === sourceID && layer.type === 'fill-extrusion') {\n return true;\n }\n }\n } else {\n for (const key in styleLayers) {\n const layer = styleLayers[key];\n if (layer.source === sourceID && layer.type === 'fill-extrusion') {\n return true;\n }\n }\n }\n return false;\n}\n\nexport function queryRenderedFeatures(\n sourceCache: SourceCache,\n styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n queryGeometry: Array,\n params: QueryRenderedFeaturesOptions,\n transform: Transform\n): { [key: string]: Array<{featureIndex: number; feature: MapGeoJSONFeature}> } {\n\n const has3DLayer = queryIncludes3DLayer(params && params.layers, styleLayers, sourceCache.id);\n const maxPitchScaleFactor = transform.maxPitchScaleFactor();\n const tilesIn = sourceCache.tilesIn(queryGeometry, maxPitchScaleFactor, has3DLayer);\n\n tilesIn.sort(sortTilesIn);\n const renderedFeatureLayers = [];\n for (const tileIn of tilesIn) {\n renderedFeatureLayers.push({\n wrappedTileID: tileIn.tileID.wrapped().key,\n queryResults: tileIn.tile.queryRenderedFeatures(\n styleLayers,\n serializedLayers,\n sourceCache._state,\n tileIn.queryGeometry,\n tileIn.cameraQueryGeometry,\n tileIn.scale,\n params,\n transform,\n maxPitchScaleFactor,\n getPixelPosMatrix(sourceCache.transform, tileIn.tileID))\n });\n }\n\n const result = mergeRenderedFeatureLayers(renderedFeatureLayers);\n\n // Merge state from SourceCache into the results\n for (const layerID in result) {\n result[layerID].forEach((featureWrapper) => {\n const feature = featureWrapper.feature as MapGeoJSONFeature;\n const state = sourceCache.getFeatureState(feature.layer['source-layer'], feature.id);\n feature.source = feature.layer.source;\n if (feature.layer['source-layer']) {\n feature.sourceLayer = feature.layer['source-layer'];\n }\n feature.state = state;\n });\n }\n return result;\n}\n\nexport function queryRenderedSymbols(styleLayers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: StyleLayer},\n sourceCaches: {[_: string]: SourceCache},\n queryGeometry: Array,\n params: QueryRenderedFeaturesOptions,\n collisionIndex: CollisionIndex,\n retainedQueryData: {\n [_: number]: RetainedQueryData;\n }) {\n const result = {};\n const renderedSymbols = collisionIndex.queryRenderedSymbols(queryGeometry);\n const bucketQueryData = [];\n for (const bucketInstanceId of Object.keys(renderedSymbols).map(Number)) {\n bucketQueryData.push(retainedQueryData[bucketInstanceId]);\n }\n bucketQueryData.sort(sortTilesIn);\n\n for (const queryData of bucketQueryData) {\n const bucketSymbols = queryData.featureIndex.lookupSymbolFeatures(\n renderedSymbols[queryData.bucketInstanceId],\n serializedLayers,\n queryData.bucketIndex,\n queryData.sourceLayerIndex,\n params.filter,\n params.layers,\n params.availableImages,\n styleLayers);\n\n for (const layerID in bucketSymbols) {\n const resultFeatures = result[layerID] = result[layerID] || [];\n const layerSymbols = bucketSymbols[layerID];\n layerSymbols.sort((a, b) => {\n // Match topDownFeatureComparator from FeatureIndex, but using\n // most recent sorting of features from bucket.sortFeatures\n const featureSortOrder = queryData.featureSortOrder;\n if (featureSortOrder) {\n // queryRenderedSymbols documentation says we'll return features in\n // \"top-to-bottom\" rendering order (aka last-to-first).\n // Actually there can be multiple symbol instances per feature, so\n // we sort each feature based on the first matching symbol instance.\n const sortedA = featureSortOrder.indexOf(a.featureIndex);\n const sortedB = featureSortOrder.indexOf(b.featureIndex);\n return sortedB - sortedA;\n } else {\n // Bucket hasn't been re-sorted based on angle, so use the\n // reverse of the order the features appeared in the data.\n return b.featureIndex - a.featureIndex;\n }\n });\n for (const symbolFeature of layerSymbols) {\n resultFeatures.push(symbolFeature);\n }\n }\n }\n\n // Merge state from SourceCache into the results\n for (const layerName in result) {\n result[layerName].forEach((featureWrapper) => {\n const feature = featureWrapper.feature;\n const layer = styleLayers[layerName];\n const sourceCache = sourceCaches[layer.source];\n const state = sourceCache.getFeatureState(feature.layer['source-layer'], feature.id);\n feature.source = feature.layer.source;\n if (feature.layer['source-layer']) {\n feature.sourceLayer = feature.layer['source-layer'];\n }\n feature.state = state;\n });\n }\n return result;\n}\n\nexport function querySourceFeatures(sourceCache: SourceCache, params: QuerySourceFeatureOptions) {\n const tiles = sourceCache.getRenderableIds().map((id) => {\n return sourceCache.getTileByID(id);\n });\n\n const result = [];\n\n const dataTiles = {};\n for (let i = 0; i < tiles.length; i++) {\n const tile = tiles[i];\n const dataID = tile.tileID.canonical.key;\n if (!dataTiles[dataID]) {\n dataTiles[dataID] = true;\n tile.querySourceFeatures(result, params);\n }\n }\n\n return result;\n}\n\nfunction sortTilesIn(a, b) {\n const idA = a.tileID;\n const idB = b.tileID;\n return (idA.overscaledZ - idB.overscaledZ) || (idA.canonical.y - idB.canonical.y) || (idA.wrap - idB.wrap) || (idA.canonical.x - idB.canonical.x);\n}\n\nfunction mergeRenderedFeatureLayers(tiles) {\n // Merge results from all tiles, but if two tiles share the same\n // wrapped ID, don't duplicate features between the two tiles\n const result = {};\n const wrappedIDLayerMap = {};\n for (const tile of tiles) {\n const queryResults = tile.queryResults;\n const wrappedID = tile.wrappedTileID;\n const wrappedIDLayers = wrappedIDLayerMap[wrappedID] = wrappedIDLayerMap[wrappedID] || {};\n for (const layerID in queryResults) {\n const tileFeatures = queryResults[layerID];\n const wrappedIDFeatures = wrappedIDLayers[layerID] = wrappedIDLayers[layerID] || {};\n const resultFeatures = result[layerID] = result[layerID] || [];\n for (const tileFeature of tileFeatures) {\n if (!wrappedIDFeatures[tileFeature.featureIndex]) {\n wrappedIDFeatures[tileFeature.featureIndex] = true;\n resultFeatures.push(tileFeature);\n }\n }\n }\n }\n return result;\n}\n","import {uniqueId, parseCacheControl} from '../util/util';\nimport {deserialize as deserializeBucket} from '../data/bucket';\nimport '../data/feature_index';\nimport type {FeatureIndex} from '../data/feature_index';\nimport {GeoJSONFeature} from '../util/vectortile_to_geojson';\nimport {featureFilter} from '@maplibre/maplibre-gl-style-spec';\nimport {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {CollisionBoxArray} from '../data/array_types.g';\nimport {Texture} from '../render/texture';\nimport {browser} from '../util/browser';\nimport {toEvaluationFeature} from '../data/evaluation_feature';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {SourceFeatureState} from '../source/source_state';\nimport {lazyLoadRTLTextPlugin} from './rtl_text_plugin';\n\nconst CLOCK_SKEW_RETRY_TIMEOUT = 30000;\n\nimport type {Bucket} from '../data/bucket';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {WorkerTileResult} from './worker_source';\nimport type {Actor} from '../util/actor';\nimport type {DEMData} from '../data/dem_data';\nimport type {AlphaImage} from '../util/image';\nimport type {ImageAtlas} from '../render/image_atlas';\nimport type {ImageManager} from '../render/image_manager';\nimport type {Context} from '../gl/context';\nimport type {OverscaledTileID} from './tile_id';\nimport type {Framebuffer} from '../gl/framebuffer';\nimport type {Transform} from '../geo/transform';\nimport type {LayerFeatureStates} from './source_state';\nimport type {Cancelable} from '../types/cancelable';\nimport type {FilterSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type Point from '@mapbox/point-geometry';\nimport {mat4} from 'gl-matrix';\nimport type {VectorTileLayer} from '@mapbox/vector-tile';\nimport {ExpiryData} from '../util/ajax';\n\n/**\n * The tile's state, can be:\n * - `loading` Tile data is in the process of loading.\n * - `loaded` Tile data has been loaded. Tile can be rendered.\n * - `reloading` Tile data has been loaded and is being updated. Tile can be rendered.\n * - `unloaded` Tile data has been deleted.\n * - `errored` Tile data was not loaded because of an error.\n * - `expired` Tile data was previously loaded, but has expired per its HTTP headers and is in the process of refreshing.\n */\nexport type TileState = 'loading' | 'loaded' | 'reloading' | 'unloaded' | 'errored' | 'expired';\n\n/**\n * @internal\n * A tile object is the combination of a Coordinate, which defines\n * its place, as well as a unique ID and data tracking for its content\n */\nexport class Tile {\n tileID: OverscaledTileID;\n uid: number;\n uses: number;\n tileSize: number;\n buckets: {[_: string]: Bucket};\n latestFeatureIndex: FeatureIndex;\n latestRawTileData: ArrayBuffer;\n imageAtlas: ImageAtlas;\n imageAtlasTexture: Texture;\n glyphAtlasImage: AlphaImage;\n glyphAtlasTexture: Texture;\n expirationTime: any;\n expiredRequestCount: number;\n state: TileState;\n timeAdded: number = 0;\n fadeEndTime: number = 0;\n collisionBoxArray: CollisionBoxArray;\n redoWhenDone: boolean;\n showCollisionBoxes: boolean;\n placementSource: any;\n actor: Actor;\n vtLayers: {[_: string]: VectorTileLayer};\n\n neighboringTiles: any;\n dem: DEMData;\n demMatrix: mat4;\n aborted: boolean;\n needsHillshadePrepare: boolean;\n needsTerrainPrepare: boolean;\n request: Cancelable;\n texture: any;\n fbo: Framebuffer;\n demTexture: Texture;\n refreshedUponExpiration: boolean;\n reloadCallback: any;\n resourceTiming: Array;\n queryPadding: number;\n\n symbolFadeHoldUntil: number;\n hasSymbolBuckets: boolean;\n hasRTLText: boolean;\n dependencies: any;\n rtt: Array<{id: number; stamp: number}>;\n rttCoords: {[_:string]: string};\n\n /**\n * @param tileID - the tile ID\n * @param size - The tile size\n */\n constructor(tileID: OverscaledTileID, size: number) {\n this.tileID = tileID;\n this.uid = uniqueId();\n this.uses = 0;\n this.tileSize = size;\n this.buckets = {};\n this.expirationTime = null;\n this.queryPadding = 0;\n this.hasSymbolBuckets = false;\n this.hasRTLText = false;\n this.dependencies = {};\n this.rtt = [];\n this.rttCoords = {};\n\n // Counts the number of times a response was already expired when\n // received. We're using this to add a delay when making a new request\n // so we don't have to keep retrying immediately in case of a server\n // serving expired tiles.\n this.expiredRequestCount = 0;\n\n this.state = 'loading';\n }\n\n registerFadeDuration(duration: number) {\n const fadeEndTime = duration + this.timeAdded;\n\n if (fadeEndTime < this.fadeEndTime) {\n return;\n }\n\n this.fadeEndTime = fadeEndTime;\n }\n\n wasRequested() {\n return this.state === 'errored' || this.state === 'loaded' || this.state === 'reloading';\n }\n\n clearTextures(painter: any) {\n if (this.demTexture) painter.saveTileTexture(this.demTexture);\n this.demTexture = null;\n }\n\n /**\n * Given a data object with a 'buffers' property, load it into\n * this tile's elementGroups and buffers properties and set loaded\n * to true. If the data is null, like in the case of an empty\n * GeoJSON tile, no-op but still set loaded to true.\n * @param data - The data from the worker\n * @param painter - the painter\n * @param justReloaded - `true` to just reload\n */\n loadVectorData(data: WorkerTileResult, painter: any, justReloaded?: boolean | null) {\n if (this.hasData()) {\n this.unloadVectorData();\n }\n\n this.state = 'loaded';\n\n // empty GeoJSON tile\n if (!data) {\n this.collisionBoxArray = new CollisionBoxArray();\n return;\n }\n\n if (data.featureIndex) {\n this.latestFeatureIndex = data.featureIndex;\n if (data.rawTileData) {\n // Only vector tiles have rawTileData, and they won't update it for\n // 'reloadTile'\n this.latestRawTileData = data.rawTileData;\n this.latestFeatureIndex.rawTileData = data.rawTileData;\n } else if (this.latestRawTileData) {\n // If rawTileData hasn't updated, hold onto a pointer to the last\n // one we received\n this.latestFeatureIndex.rawTileData = this.latestRawTileData;\n }\n }\n this.collisionBoxArray = data.collisionBoxArray;\n this.buckets = deserializeBucket(data.buckets, painter.style);\n\n this.hasSymbolBuckets = false;\n for (const id in this.buckets) {\n const bucket = this.buckets[id];\n if (bucket instanceof SymbolBucket) {\n this.hasSymbolBuckets = true;\n if (justReloaded) {\n bucket.justReloaded = true;\n } else {\n break;\n }\n }\n }\n\n this.hasRTLText = false;\n if (this.hasSymbolBuckets) {\n for (const id in this.buckets) {\n const bucket = this.buckets[id];\n if (bucket instanceof SymbolBucket) {\n if (bucket.hasRTLText) {\n this.hasRTLText = true;\n lazyLoadRTLTextPlugin();\n break;\n }\n }\n }\n }\n\n this.queryPadding = 0;\n for (const id in this.buckets) {\n const bucket = this.buckets[id];\n this.queryPadding = Math.max(this.queryPadding, painter.style.getLayer(id).queryRadius(bucket));\n }\n\n if (data.imageAtlas) {\n this.imageAtlas = data.imageAtlas;\n }\n if (data.glyphAtlasImage) {\n this.glyphAtlasImage = data.glyphAtlasImage;\n }\n }\n\n /**\n * Release any data or WebGL resources referenced by this tile.\n */\n unloadVectorData() {\n for (const id in this.buckets) {\n this.buckets[id].destroy();\n }\n this.buckets = {};\n\n if (this.imageAtlasTexture) {\n this.imageAtlasTexture.destroy();\n }\n\n if (this.imageAtlas) {\n this.imageAtlas = null;\n }\n\n if (this.glyphAtlasTexture) {\n this.glyphAtlasTexture.destroy();\n }\n\n this.latestFeatureIndex = null;\n this.state = 'unloaded';\n }\n\n getBucket(layer: StyleLayer) {\n return this.buckets[layer.id];\n }\n\n upload(context: Context) {\n for (const id in this.buckets) {\n const bucket = this.buckets[id];\n if (bucket.uploadPending()) {\n bucket.upload(context);\n }\n }\n\n const gl = context.gl;\n if (this.imageAtlas && !this.imageAtlas.uploaded) {\n this.imageAtlasTexture = new Texture(context, this.imageAtlas.image, gl.RGBA);\n this.imageAtlas.uploaded = true;\n }\n\n if (this.glyphAtlasImage) {\n this.glyphAtlasTexture = new Texture(context, this.glyphAtlasImage, gl.ALPHA);\n this.glyphAtlasImage = null;\n }\n }\n\n prepare(imageManager: ImageManager) {\n if (this.imageAtlas) {\n this.imageAtlas.patchUpdatedImages(imageManager, this.imageAtlasTexture);\n }\n }\n\n // Queries non-symbol features rendered for this tile.\n // Symbol features are queried globally\n queryRenderedFeatures(\n layers: {[_: string]: StyleLayer},\n serializedLayers: {[_: string]: any},\n sourceFeatureState: SourceFeatureState,\n queryGeometry: Array,\n cameraQueryGeometry: Array,\n scale: number,\n params: {\n filter: FilterSpecification;\n layers: Array;\n availableImages: Array;\n },\n transform: Transform,\n maxPitchScaleFactor: number,\n pixelPosMatrix: mat4\n ): {[_: string]: Array<{featureIndex: number; feature: GeoJSONFeature}>} {\n if (!this.latestFeatureIndex || !this.latestFeatureIndex.rawTileData)\n return {};\n\n return this.latestFeatureIndex.query({\n queryGeometry,\n cameraQueryGeometry,\n scale,\n tileSize: this.tileSize,\n pixelPosMatrix,\n transform,\n params,\n queryPadding: this.queryPadding * maxPitchScaleFactor\n }, layers, serializedLayers, sourceFeatureState);\n }\n\n querySourceFeatures(result: Array, params?: {\n sourceLayer?: string;\n filter?: FilterSpecification;\n validate?: boolean;\n }) {\n const featureIndex = this.latestFeatureIndex;\n if (!featureIndex || !featureIndex.rawTileData) return;\n\n const vtLayers = featureIndex.loadVTLayers();\n\n const sourceLayer = params && params.sourceLayer ? params.sourceLayer : '';\n const layer = vtLayers._geojsonTileLayer || vtLayers[sourceLayer];\n\n if (!layer) return;\n\n const filter = featureFilter(params && params.filter);\n const {z, x, y} = this.tileID.canonical;\n const coord = {z, x, y};\n\n for (let i = 0; i < layer.length; i++) {\n const feature = layer.feature(i);\n if (filter.needGeometry) {\n const evaluationFeature = toEvaluationFeature(feature, true);\n if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), evaluationFeature, this.tileID.canonical)) continue;\n } else if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), feature)) {\n continue;\n }\n const id = featureIndex.getId(feature, sourceLayer);\n const geojsonFeature = new GeoJSONFeature(feature, z, x, y, id);\n (geojsonFeature as any).tile = coord;\n result.push(geojsonFeature);\n }\n }\n\n hasData() {\n return this.state === 'loaded' || this.state === 'reloading' || this.state === 'expired';\n }\n\n patternsLoaded() {\n return this.imageAtlas && !!Object.keys(this.imageAtlas.patternPositions).length;\n }\n\n setExpiryData(data: ExpiryData) {\n const prior = this.expirationTime;\n\n if (data.cacheControl) {\n const parsedCC = parseCacheControl(data.cacheControl);\n if (parsedCC['max-age']) this.expirationTime = Date.now() + parsedCC['max-age'] * 1000;\n } else if (data.expires) {\n this.expirationTime = new Date(data.expires).getTime();\n }\n\n if (this.expirationTime) {\n const now = Date.now();\n let isExpired = false;\n\n if (this.expirationTime > now) {\n isExpired = false;\n } else if (!prior) {\n isExpired = true;\n } else if (this.expirationTime < prior) {\n // Expiring date is going backwards:\n // fall back to exponential backoff\n isExpired = true;\n\n } else {\n const delta = this.expirationTime - prior;\n\n if (!delta) {\n // Server is serving the same expired resource over and over: fall\n // back to exponential backoff.\n isExpired = true;\n\n } else {\n // Assume that either the client or the server clock is wrong and\n // try to interpolate a valid expiration date (from the client POV)\n // observing a minimum timeout.\n this.expirationTime = now + Math.max(delta, CLOCK_SKEW_RETRY_TIMEOUT);\n\n }\n }\n\n if (isExpired) {\n this.expiredRequestCount++;\n this.state = 'expired';\n } else {\n this.expiredRequestCount = 0;\n }\n }\n }\n\n getExpiryTimeout() {\n if (this.expirationTime) {\n if (this.expiredRequestCount) {\n return 1000 * (1 << Math.min(this.expiredRequestCount - 1, 31));\n } else {\n // Max value for `setTimeout` implementations is a 32 bit integer; cap this accordingly\n return Math.min(this.expirationTime - new Date().getTime(), Math.pow(2, 31) - 1);\n }\n }\n }\n\n setFeatureState(states: LayerFeatureStates, painter: any) {\n if (!this.latestFeatureIndex ||\n !this.latestFeatureIndex.rawTileData ||\n Object.keys(states).length === 0) {\n return;\n }\n\n const vtLayers = this.latestFeatureIndex.loadVTLayers();\n\n for (const id in this.buckets) {\n if (!painter.style.hasLayer(id)) continue;\n\n const bucket = this.buckets[id];\n // Buckets are grouped by common source-layer\n const sourceLayerId = bucket.layers[0]['sourceLayer'] || '_geojsonTileLayer';\n const sourceLayer = vtLayers[sourceLayerId];\n const sourceLayerStates = states[sourceLayerId];\n if (!sourceLayer || !sourceLayerStates || Object.keys(sourceLayerStates).length === 0) continue;\n\n bucket.update(sourceLayerStates, sourceLayer, this.imageAtlas && this.imageAtlas.patternPositions || {});\n const layer = painter && painter.style && painter.style.getLayer(id);\n if (layer) {\n this.queryPadding = Math.max(this.queryPadding, layer.queryRadius(bucket));\n }\n }\n }\n\n holdingForFade(): boolean {\n return this.symbolFadeHoldUntil !== undefined;\n }\n\n symbolFadeFinished(): boolean {\n return !this.symbolFadeHoldUntil || this.symbolFadeHoldUntil < browser.now();\n }\n\n clearFadeHold() {\n this.symbolFadeHoldUntil = undefined;\n }\n\n setHoldDuration(duration: number) {\n this.symbolFadeHoldUntil = browser.now() + duration;\n }\n\n setDependencies(namespace: string, dependencies: Array) {\n const index = {};\n for (const dep of dependencies) {\n index[dep] = true;\n }\n this.dependencies[namespace] = index;\n }\n\n hasDependency(namespaces: Array, keys: Array) {\n for (const namespace of namespaces) {\n const dependencies = this.dependencies[namespace];\n if (dependencies) {\n for (const key of keys) {\n if (dependencies[key]) {\n return true;\n }\n }\n }\n }\n return false;\n }\n}\n","import type {CollisionBoxArray} from './array_types.g';\nimport type {Style} from '../style/style';\nimport type {TypedStyleLayer} from '../style/style_layer/typed_style_layer';\nimport type {FeatureIndex} from './feature_index';\nimport type {Context} from '../gl/context';\nimport type {FeatureStates} from '../source/source_state';\nimport type {ImagePosition} from '../render/image_atlas';\nimport type {CanonicalTileID} from '../source/tile_id';\nimport type {VectorTileFeature, VectorTileLayer} from '@mapbox/vector-tile';\nimport Point from '@mapbox/point-geometry';\n\nexport type BucketParameters = {\n index: number;\n layers: Array;\n zoom: number;\n pixelRatio: number;\n overscaling: number;\n collisionBoxArray: CollisionBoxArray;\n sourceLayerIndex: number;\n sourceID: string;\n};\n\nexport type PopulateParameters = {\n featureIndex: FeatureIndex;\n iconDependencies: {};\n patternDependencies: {};\n glyphDependencies: {};\n availableImages: Array;\n};\n\nexport type IndexedFeature = {\n feature: VectorTileFeature;\n id: number | string;\n index: number;\n sourceLayerIndex: number;\n};\n\nexport type BucketFeature = {\n index: number;\n sourceLayerIndex: number;\n geometry: Array>;\n properties: any;\n type: 0 | 1 | 2 | 3;\n id?: any;\n readonly patterns: {\n [_: string]: {\n 'min': string;\n 'mid': string;\n 'max': string;\n };\n };\n sortKey?: number;\n};\n\n/**\n * The `Bucket` interface is the single point of knowledge about turning vector\n * tiles into WebGL buffers.\n *\n * `Bucket` is an abstract interface. An implementation exists for each style layer type.\n * Create a bucket via the `StyleLayer#createBucket` method.\n *\n * The concrete bucket types, using layout options from the style layer,\n * transform feature geometries into vertex and index data for use by the\n * vertex shader. They also (via `ProgramConfiguration`) use feature\n * properties and the zoom level to populate the attributes needed for\n * data-driven styling.\n *\n * Buckets are designed to be built on a worker thread and then serialized and\n * transferred back to the main thread for rendering. On the worker side, a\n * bucket's vertex, index, and attribute data is stored in `bucket.arrays: ArrayGroup`.\n * When a bucket's data is serialized and sent back to the main thread,\n * is gets deserialized (using `new Bucket(serializedBucketData)`, with\n * the array data now stored in `bucket.buffers: BufferGroup`. BufferGroups\n * hold the same data as ArrayGroups, but are tuned for consumption by WebGL.\n */\nexport interface Bucket {\n layerIds: Array;\n hasPattern: boolean;\n readonly layers: Array;\n readonly stateDependentLayers: Array;\n readonly stateDependentLayerIds: Array;\n populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID): void;\n update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}): void;\n isEmpty(): boolean;\n upload(context: Context): void;\n uploadPending(): boolean;\n /**\n * Release the WebGL resources associated with the buffers. Note that because\n * buckets are shared between layers having the same layout properties, they\n * must be destroyed in groups (all buckets for a tile, or all symbol buckets).\n */\n destroy(): void;\n}\n\nexport function deserialize(input: Array, style: Style): {[_: string]: Bucket} {\n const output = {};\n\n // Guard against the case where the map's style has been set to null while\n // this bucket has been parsing.\n if (!style) return output;\n\n for (const bucket of input) {\n const layers = bucket.layerIds\n .map((id) => style.getLayer(id))\n .filter(Boolean);\n\n if (layers.length === 0) {\n continue;\n }\n\n // look up StyleLayer objects from layer ids (since we don't\n // want to waste time serializing/copying them from the worker)\n (bucket as any).layers = layers;\n if (bucket.stateDependentLayerIds) {\n (bucket as any).stateDependentLayers = bucket.stateDependentLayerIds.map((lId) => layers.filter((l) => l.id === lId)[0]);\n }\n for (const layer of layers) {\n output[layer.id] = bucket;\n }\n }\n\n return output;\n}\n","import {OverscaledTileID} from './tile_id';\nimport type {Tile} from './tile';\n\n/**\n * @internal\n * A [least-recently-used cache](http://en.wikipedia.org/wiki/Cache_algorithms)\n * with hash lookup made possible by keeping a list of keys in parallel to\n * an array of dictionary of values\n */\nexport class TileCache {\n max: number;\n data: {\n [key: string]: Array<{\n value: Tile;\n timeout: ReturnType;\n }>;\n };\n order: Array;\n onRemove: (element: Tile) => void;\n /**\n * @param max - number of permitted values\n * @param onRemove - callback called with items when they expire\n */\n constructor(max: number, onRemove: (element: Tile) => void) {\n this.max = max;\n this.onRemove = onRemove;\n this.reset();\n }\n\n /**\n * Clear the cache\n *\n * @returns this cache\n */\n reset() {\n for (const key in this.data) {\n for (const removedData of this.data[key]) {\n if (removedData.timeout) clearTimeout(removedData.timeout);\n this.onRemove(removedData.value);\n }\n }\n\n this.data = {};\n this.order = [];\n\n return this;\n }\n\n /**\n * Add a key, value combination to the cache, trimming its size if this pushes\n * it over max length.\n *\n * @param tileID - lookup key for the item\n * @param data - tile data\n *\n * @returns this cache\n */\n add(tileID: OverscaledTileID, data: Tile, expiryTimeout: number | void) {\n const key = tileID.wrapped().key;\n if (this.data[key] === undefined) {\n this.data[key] = [];\n }\n\n const dataWrapper = {\n value: data,\n timeout: undefined\n };\n\n if (expiryTimeout !== undefined) {\n dataWrapper.timeout = setTimeout(() => {\n this.remove(tileID, dataWrapper);\n }, expiryTimeout as number);\n }\n\n this.data[key].push(dataWrapper);\n this.order.push(key);\n\n if (this.order.length > this.max) {\n const removedData = this._getAndRemoveByKey(this.order[0]);\n if (removedData) this.onRemove(removedData);\n }\n\n return this;\n }\n\n /**\n * Determine whether the value attached to `key` is present\n *\n * @param tileID - the key to be looked-up\n * @returns whether the cache has this value\n */\n has(tileID: OverscaledTileID): boolean {\n return tileID.wrapped().key in this.data;\n }\n\n /**\n * Get the value attached to a specific key and remove data from cache.\n * If the key is not found, returns `null`\n *\n * @param tileID - the key to look up\n * @returns the tile data, or null if it isn't found\n */\n getAndRemove(tileID: OverscaledTileID): Tile {\n if (!this.has(tileID)) { return null; }\n return this._getAndRemoveByKey(tileID.wrapped().key);\n }\n\n /*\n * Get and remove the value with the specified key.\n */\n _getAndRemoveByKey(key: string): Tile {\n const data = this.data[key].shift();\n if (data.timeout) clearTimeout(data.timeout);\n\n if (this.data[key].length === 0) {\n delete this.data[key];\n }\n this.order.splice(this.order.indexOf(key), 1);\n\n return data.value;\n }\n\n /*\n * Get the value with the specified (wrapped tile) key.\n */\n getByKey(key: string): Tile {\n const data = this.data[key];\n return data ? data[0].value : null;\n }\n\n /**\n * Get the value attached to a specific key without removing data\n * from the cache. If the key is not found, returns `null`\n *\n * @param tileID - the key to look up\n * @returns the tile data, or null if it isn't found\n */\n get(tileID: OverscaledTileID): Tile {\n if (!this.has(tileID)) { return null; }\n\n const data = this.data[tileID.wrapped().key][0];\n return data.value;\n }\n\n /**\n * Remove a key/value combination from the cache.\n *\n * @param tileID - the key for the pair to delete\n * @param value - If a value is provided, remove that exact version of the value.\n * @returns this cache\n */\n remove(tileID: OverscaledTileID, value?: {\n value: Tile;\n timeout: ReturnType;\n }) {\n if (!this.has(tileID)) { return this; }\n const key = tileID.wrapped().key;\n\n const dataIndex = value === undefined ? 0 : this.data[key].indexOf(value);\n const data = this.data[key][dataIndex];\n this.data[key].splice(dataIndex, 1);\n if (data.timeout) clearTimeout(data.timeout);\n if (this.data[key].length === 0) {\n delete this.data[key];\n }\n this.onRemove(data.value);\n this.order.splice(this.order.indexOf(key), 1);\n\n return this;\n }\n\n /**\n * Change the max size of the cache.\n *\n * @param max - the max size of the cache\n * @returns this cache\n */\n setMaxSize(max: number): TileCache {\n this.max = max;\n\n while (this.order.length > this.max) {\n const removedData = this._getAndRemoveByKey(this.order[0]);\n if (removedData) this.onRemove(removedData);\n }\n\n return this;\n }\n\n /**\n * Remove entries that do not pass a filter function. Used for removing\n * stale tiles from the cache.\n *\n * @param filterFn - Determines whether the tile is filtered. If the supplied function returns false, the tile will be filtered out.\n */\n filter(filterFn: (tile: Tile) => boolean) {\n const removed = [];\n for (const key in this.data) {\n for (const entry of this.data[key]) {\n if (!filterFn(entry.value)) {\n removed.push(entry);\n }\n }\n }\n for (const r of removed) {\n this.remove(r.value.tileID, r);\n }\n }\n}\n","import {extend} from '../util/util';\nimport {Tile} from './tile';\nimport type {FeatureState} from '@maplibre/maplibre-gl-style-spec';\n\nexport type FeatureStates = {[featureId: string]: FeatureState};\nexport type LayerFeatureStates = {[layer: string]: FeatureStates};\n\n/**\n * @internal\n * SourceFeatureState manages the state and pending changes\n * to features in a source, separated by source layer.\n * stateChanges and deletedStates batch all changes to the tile (updates and removes, respectively)\n * between coalesce() events. addFeatureState() and removeFeatureState() also update their counterpart's\n * list of changes, such that coalesce() can apply the proper state changes while agnostic to the order of operations.\n * In deletedStates, all null's denote complete removal of state at that scope\n*/\nexport class SourceFeatureState {\n state: LayerFeatureStates;\n stateChanges: LayerFeatureStates;\n deletedStates: {};\n\n constructor() {\n this.state = {};\n this.stateChanges = {};\n this.deletedStates = {};\n }\n\n updateState(sourceLayer: string, featureId: number | string, newState: any) {\n const feature = String(featureId);\n this.stateChanges[sourceLayer] = this.stateChanges[sourceLayer] || {};\n this.stateChanges[sourceLayer][feature] = this.stateChanges[sourceLayer][feature] || {};\n extend(this.stateChanges[sourceLayer][feature], newState);\n\n if (this.deletedStates[sourceLayer] === null) {\n this.deletedStates[sourceLayer] = {};\n for (const ft in this.state[sourceLayer]) {\n if (ft !== feature) this.deletedStates[sourceLayer][ft] = null;\n }\n } else {\n const featureDeletionQueued = this.deletedStates[sourceLayer] && this.deletedStates[sourceLayer][feature] === null;\n if (featureDeletionQueued) {\n this.deletedStates[sourceLayer][feature] = {};\n for (const prop in this.state[sourceLayer][feature]) {\n if (!newState[prop]) this.deletedStates[sourceLayer][feature][prop] = null;\n }\n } else {\n for (const key in newState) {\n const deletionInQueue = this.deletedStates[sourceLayer] && this.deletedStates[sourceLayer][feature] && this.deletedStates[sourceLayer][feature][key] === null;\n if (deletionInQueue) delete this.deletedStates[sourceLayer][feature][key];\n }\n }\n }\n }\n\n removeFeatureState(sourceLayer: string, featureId?: number | string, key?: string) {\n const sourceLayerDeleted = this.deletedStates[sourceLayer] === null;\n if (sourceLayerDeleted) return;\n\n const feature = String(featureId);\n\n this.deletedStates[sourceLayer] = this.deletedStates[sourceLayer] || {};\n\n if (key && featureId !== undefined) {\n if (this.deletedStates[sourceLayer][feature] !== null) {\n this.deletedStates[sourceLayer][feature] = this.deletedStates[sourceLayer][feature] || {};\n this.deletedStates[sourceLayer][feature][key] = null;\n }\n } else if (featureId !== undefined) {\n const updateInQueue = this.stateChanges[sourceLayer] && this.stateChanges[sourceLayer][feature];\n if (updateInQueue) {\n this.deletedStates[sourceLayer][feature] = {};\n for (key in this.stateChanges[sourceLayer][feature]) this.deletedStates[sourceLayer][feature][key] = null;\n\n } else {\n this.deletedStates[sourceLayer][feature] = null;\n }\n } else {\n this.deletedStates[sourceLayer] = null;\n }\n\n }\n\n getState(sourceLayer: string, featureId: number | string) {\n const feature = String(featureId);\n const base = this.state[sourceLayer] || {};\n const changes = this.stateChanges[sourceLayer] || {};\n\n const reconciledState = extend({}, base[feature], changes[feature]);\n\n //return empty object if the whole source layer is awaiting deletion\n if (this.deletedStates[sourceLayer] === null) return {};\n else if (this.deletedStates[sourceLayer]) {\n const featureDeletions = this.deletedStates[sourceLayer][featureId];\n if (featureDeletions === null) return {};\n for (const prop in featureDeletions) delete reconciledState[prop];\n }\n return reconciledState;\n }\n\n initializeTileState(tile: Tile, painter: any) {\n tile.setFeatureState(this.state, painter);\n }\n\n coalesceChanges(tiles: {\n [_ in any]: Tile;\n }, painter: any) {\n //track changes with full state objects, but only for features that got modified\n const featuresChanged: LayerFeatureStates = {};\n\n for (const sourceLayer in this.stateChanges) {\n this.state[sourceLayer] = this.state[sourceLayer] || {};\n const layerStates = {};\n for (const feature in this.stateChanges[sourceLayer]) {\n if (!this.state[sourceLayer][feature]) this.state[sourceLayer][feature] = {};\n extend(this.state[sourceLayer][feature], this.stateChanges[sourceLayer][feature]);\n layerStates[feature] = this.state[sourceLayer][feature];\n }\n featuresChanged[sourceLayer] = layerStates;\n }\n\n for (const sourceLayer in this.deletedStates) {\n this.state[sourceLayer] = this.state[sourceLayer] || {};\n const layerStates = {};\n\n if (this.deletedStates[sourceLayer] === null) {\n for (const ft in this.state[sourceLayer]) {\n layerStates[ft] = {};\n this.state[sourceLayer][ft] = {};\n }\n } else {\n for (const feature in this.deletedStates[sourceLayer]) {\n const deleteWholeFeatureState = this.deletedStates[sourceLayer][feature] === null;\n if (deleteWholeFeatureState) this.state[sourceLayer][feature] = {};\n else {\n for (const key of Object.keys(this.deletedStates[sourceLayer][feature])) {\n delete this.state[sourceLayer][feature][key];\n }\n }\n layerStates[feature] = this.state[sourceLayer][feature];\n }\n }\n\n featuresChanged[sourceLayer] = featuresChanged[sourceLayer] || {};\n extend(featuresChanged[sourceLayer], layerStates);\n }\n\n this.stateChanges = {};\n this.deletedStates = {};\n\n if (Object.keys(featuresChanged).length === 0) return;\n\n for (const id in tiles) {\n const tile = tiles[id];\n tile.setFeatureState(featuresChanged, painter);\n }\n }\n}\n","import {create as createSource} from './source';\n\nimport {Tile} from './tile';\nimport {Event, ErrorEvent, Evented} from '../util/evented';\nimport {TileCache} from './tile_cache';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\nimport {keysDifference} from '../util/util';\nimport {EXTENT} from '../data/extent';\nimport {Context} from '../gl/context';\nimport Point from '@mapbox/point-geometry';\nimport {browser} from '../util/browser';\nimport {OverscaledTileID} from './tile_id';\nimport {SourceFeatureState} from './source_state';\n\nimport type {Source} from './source';\nimport type {Map} from '../ui/map';\nimport type {Style} from '../style/style';\nimport type {Dispatcher} from '../util/dispatcher';\nimport type {Transform} from '../geo/transform';\nimport type {TileState} from './tile';\nimport type {Callback} from '../types/callback';\nimport type {SourceSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {MapSourceDataEvent} from '../ui/events';\nimport {Terrain} from '../render/terrain';\nimport {config} from '../util/config';\n\n/**\n * @internal\n * `SourceCache` is responsible for\n *\n * - creating an instance of `Source`\n * - forwarding events from `Source`\n * - caching tiles loaded from an instance of `Source`\n * - loading the tiles needed to render a given viewport\n * - unloading the cached tiles not needed to render a given viewport\n */\nexport class SourceCache extends Evented {\n id: string;\n dispatcher: Dispatcher;\n map: Map;\n style: Style;\n\n _source: Source;\n _sourceLoaded: boolean;\n _sourceErrored: boolean;\n _tiles: {[_: string]: Tile};\n _prevLng: number;\n _cache: TileCache;\n _timers: {\n [_ in any]: ReturnType;\n };\n _cacheTimers: {\n [_ in any]: ReturnType;\n };\n _maxTileCacheSize: number;\n _maxTileCacheZoomLevels: number;\n _paused: boolean;\n _shouldReloadOnResume: boolean;\n _coveredTiles: {[_: string]: boolean};\n transform: Transform;\n terrain: Terrain;\n used: boolean;\n usedForTerrain: boolean;\n tileSize: number;\n _state: SourceFeatureState;\n _loadedParentTiles: {[_: string]: Tile};\n _didEmitContent: boolean;\n _updated: boolean;\n\n static maxUnderzooming: number;\n static maxOverzooming: number;\n\n constructor(id: string, options: SourceSpecification, dispatcher: Dispatcher) {\n super();\n this.id = id;\n this.dispatcher = dispatcher;\n\n this.on('data', (e: MapSourceDataEvent) => {\n // this._sourceLoaded signifies that the TileJSON is loaded if applicable.\n // if the source type does not come with a TileJSON, the flag signifies the\n // source data has loaded (i.e geojson has been tiled on the worker and is ready)\n if (e.dataType === 'source' && e.sourceDataType === 'metadata') this._sourceLoaded = true;\n\n // for sources with mutable data, this event fires when the underlying data\n // to a source is changed. (i.e. GeoJSONSource#setData and ImageSource#serCoordinates)\n if (this._sourceLoaded && !this._paused && e.dataType === 'source' && e.sourceDataType === 'content') {\n this.reload();\n if (this.transform) {\n this.update(this.transform, this.terrain);\n }\n\n this._didEmitContent = true;\n }\n });\n\n this.on('dataloading', () => {\n this._sourceErrored = false;\n });\n\n this.on('error', () => {\n // Only set _sourceErrored if the source does not have pending loads.\n this._sourceErrored = this._source.loaded();\n });\n\n this._source = createSource(id, options, dispatcher, this);\n\n this._tiles = {};\n this._cache = new TileCache(0, this._unloadTile.bind(this));\n this._timers = {};\n this._cacheTimers = {};\n this._maxTileCacheSize = null;\n this._maxTileCacheZoomLevels = null;\n this._loadedParentTiles = {};\n\n this._coveredTiles = {};\n this._state = new SourceFeatureState();\n this._didEmitContent = false;\n this._updated = false;\n }\n\n onAdd(map: Map) {\n this.map = map;\n this._maxTileCacheSize = map ? map._maxTileCacheSize : null;\n this._maxTileCacheZoomLevels = map ? map._maxTileCacheZoomLevels : null;\n if (this._source && this._source.onAdd) {\n this._source.onAdd(map);\n }\n }\n\n onRemove(map: Map) {\n this.clearTiles();\n if (this._source && this._source.onRemove) {\n this._source.onRemove(map);\n }\n }\n\n /**\n * Return true if no tile data is pending, tiles will not change unless\n * an additional API call is received.\n */\n loaded(): boolean {\n if (this._sourceErrored) { return true; }\n if (!this._sourceLoaded) { return false; }\n if (!this._source.loaded()) { return false; }\n if ((this.used !== undefined || this.usedForTerrain !== undefined) && !this.used && !this.usedForTerrain) { return true; }\n // do not consider as loaded if the update hasn't been called yet (we do not know if we will have any tiles to fetch)\n if (!this._updated) { return false; }\n\n for (const t in this._tiles) {\n const tile = this._tiles[t];\n if (tile.state !== 'loaded' && tile.state !== 'errored')\n return false;\n }\n return true;\n }\n\n getSource(): Source {\n return this._source;\n }\n\n pause() {\n this._paused = true;\n }\n\n resume() {\n if (!this._paused) return;\n const shouldReload = this._shouldReloadOnResume;\n this._paused = false;\n this._shouldReloadOnResume = false;\n if (shouldReload) this.reload();\n if (this.transform) this.update(this.transform, this.terrain);\n }\n\n _loadTile(tile: Tile, callback: Callback) {\n return this._source.loadTile(tile, callback);\n }\n\n _unloadTile(tile: Tile) {\n if (this._source.unloadTile)\n return this._source.unloadTile(tile, () => {});\n }\n\n _abortTile(tile: Tile) {\n if (this._source.abortTile)\n this._source.abortTile(tile, () => {});\n\n this._source.fire(new Event('dataabort', {tile, coord: tile.tileID, dataType: 'source'}));\n }\n\n serialize() {\n return this._source.serialize();\n }\n\n prepare(context: Context) {\n if (this._source.prepare) {\n this._source.prepare();\n }\n\n this._state.coalesceChanges(this._tiles, this.map ? this.map.painter : null);\n for (const i in this._tiles) {\n const tile = this._tiles[i];\n tile.upload(context);\n tile.prepare(this.map.style.imageManager);\n }\n }\n\n /**\n * Return all tile ids ordered with z-order, and cast to numbers\n */\n getIds(): Array {\n return (Object.values(this._tiles) as any).map((tile: Tile) => tile.tileID).sort(compareTileId).map(id => id.key);\n }\n\n getRenderableIds(symbolLayer?: boolean): Array {\n const renderables: Array = [];\n for (const id in this._tiles) {\n if (this._isIdRenderable(id, symbolLayer)) renderables.push(this._tiles[id]);\n }\n if (symbolLayer) {\n return renderables.sort((a_: Tile, b_: Tile) => {\n const a = a_.tileID;\n const b = b_.tileID;\n const rotatedA = (new Point(a.canonical.x, a.canonical.y))._rotate(this.transform.angle);\n const rotatedB = (new Point(b.canonical.x, b.canonical.y))._rotate(this.transform.angle);\n return a.overscaledZ - b.overscaledZ || rotatedB.y - rotatedA.y || rotatedB.x - rotatedA.x;\n }).map(tile => tile.tileID.key);\n }\n return renderables.map(tile => tile.tileID).sort(compareTileId).map(id => id.key);\n }\n\n hasRenderableParent(tileID: OverscaledTileID) {\n const parentTile = this.findLoadedParent(tileID, 0);\n if (parentTile) {\n return this._isIdRenderable(parentTile.tileID.key);\n }\n return false;\n }\n\n _isIdRenderable(id: string, symbolLayer?: boolean) {\n return this._tiles[id] && this._tiles[id].hasData() &&\n !this._coveredTiles[id] && (symbolLayer || !this._tiles[id].holdingForFade());\n }\n\n reload() {\n if (this._paused) {\n this._shouldReloadOnResume = true;\n return;\n }\n\n this._cache.reset();\n\n for (const i in this._tiles) {\n if (this._tiles[i].state !== 'errored') this._reloadTile(i, 'reloading');\n }\n }\n\n _reloadTile(id: string, state: TileState) {\n const tile = this._tiles[id];\n\n // this potentially does not address all underlying\n // issues https://github.com/mapbox/mapbox-gl-js/issues/4252\n // - hard to tell without repro steps\n if (!tile) return;\n\n // The difference between \"loading\" tiles and \"reloading\" or \"expired\"\n // tiles is that \"reloading\"/\"expired\" tiles are \"renderable\".\n // Therefore, a \"loading\" tile cannot become a \"reloading\" tile without\n // first becoming a \"loaded\" tile.\n if (tile.state !== 'loading') {\n tile.state = state;\n }\n\n this._loadTile(tile, this._tileLoaded.bind(this, tile, id, state));\n }\n\n _tileLoaded(tile: Tile, id: string, previousState: TileState, err?: Error | null) {\n if (err) {\n tile.state = 'errored';\n if ((err as any).status !== 404) this._source.fire(new ErrorEvent(err, {tile}));\n // continue to try loading parent/children tiles if a tile doesn't exist (404)\n else this.update(this.transform, this.terrain);\n return;\n }\n\n tile.timeAdded = browser.now();\n if (previousState === 'expired') tile.refreshedUponExpiration = true;\n this._setTileReloadTimer(id, tile);\n if (this.getSource().type === 'raster-dem' && tile.dem) this._backfillDEM(tile);\n this._state.initializeTileState(tile, this.map ? this.map.painter : null);\n\n if (!tile.aborted) {\n this._source.fire(new Event('data', {dataType: 'source', tile, coord: tile.tileID}));\n }\n }\n\n /**\n * For raster terrain source, backfill DEM to eliminate visible tile boundaries\n */\n _backfillDEM(tile: Tile) {\n const renderables = this.getRenderableIds();\n for (let i = 0; i < renderables.length; i++) {\n const borderId = renderables[i];\n if (tile.neighboringTiles && tile.neighboringTiles[borderId]) {\n const borderTile = this.getTileByID(borderId);\n fillBorder(tile, borderTile);\n fillBorder(borderTile, tile);\n }\n }\n\n function fillBorder(tile, borderTile) {\n tile.needsHillshadePrepare = true;\n tile.needsTerrainPrepare = true;\n let dx = borderTile.tileID.canonical.x - tile.tileID.canonical.x;\n const dy = borderTile.tileID.canonical.y - tile.tileID.canonical.y;\n const dim = Math.pow(2, tile.tileID.canonical.z);\n const borderId = borderTile.tileID.key;\n if (dx === 0 && dy === 0) return;\n\n if (Math.abs(dy) > 1) {\n return;\n }\n if (Math.abs(dx) > 1) {\n // Adjust the delta coordinate for world wraparound.\n if (Math.abs(dx + dim) === 1) {\n dx += dim;\n } else if (Math.abs(dx - dim) === 1) {\n dx -= dim;\n }\n }\n if (!borderTile.dem || !tile.dem) return;\n tile.dem.backfillBorder(borderTile.dem, dx, dy);\n if (tile.neighboringTiles && tile.neighboringTiles[borderId])\n tile.neighboringTiles[borderId].backfilled = true;\n }\n }\n /**\n * Get a specific tile by TileID\n */\n getTile(tileID: OverscaledTileID): Tile {\n return this.getTileByID(tileID.key);\n }\n\n /**\n * Get a specific tile by id\n */\n getTileByID(id: string): Tile {\n return this._tiles[id];\n }\n\n /**\n * For a given set of tiles, retain children that are loaded and have a zoom\n * between `zoom` (exclusive) and `maxCoveringZoom` (inclusive)\n */\n _retainLoadedChildren(\n idealTiles: {\n [_ in any]: OverscaledTileID;\n },\n zoom: number,\n maxCoveringZoom: number,\n retain: {\n [_ in any]: OverscaledTileID;\n }\n ) {\n for (const id in this._tiles) {\n let tile = this._tiles[id];\n\n // only consider renderable tiles up to maxCoveringZoom\n if (retain[id] ||\n !tile.hasData() ||\n tile.tileID.overscaledZ <= zoom ||\n tile.tileID.overscaledZ > maxCoveringZoom\n ) continue;\n\n // loop through parents and retain the topmost loaded one if found\n let topmostLoadedID = tile.tileID;\n while (tile && tile.tileID.overscaledZ > zoom + 1) {\n const parentID = tile.tileID.scaledTo(tile.tileID.overscaledZ - 1);\n\n tile = this._tiles[parentID.key];\n\n if (tile && tile.hasData()) {\n topmostLoadedID = parentID;\n }\n }\n\n // loop through ancestors of the topmost loaded child to see if there's one that needed it\n let tileID = topmostLoadedID;\n while (tileID.overscaledZ > zoom) {\n tileID = tileID.scaledTo(tileID.overscaledZ - 1);\n\n if (idealTiles[tileID.key]) {\n // found a parent that needed a loaded child; retain that child\n retain[topmostLoadedID.key] = topmostLoadedID;\n break;\n }\n }\n }\n }\n\n /**\n * Find a loaded parent of the given tile (up to minCoveringZoom)\n */\n findLoadedParent(tileID: OverscaledTileID, minCoveringZoom: number): Tile {\n if (tileID.key in this._loadedParentTiles) {\n const parent = this._loadedParentTiles[tileID.key];\n if (parent && parent.tileID.overscaledZ >= minCoveringZoom) {\n return parent;\n } else {\n return null;\n }\n }\n for (let z = tileID.overscaledZ - 1; z >= minCoveringZoom; z--) {\n const parentTileID = tileID.scaledTo(z);\n const tile = this._getLoadedTile(parentTileID);\n if (tile) {\n return tile;\n }\n }\n }\n\n _getLoadedTile(tileID: OverscaledTileID): Tile {\n const tile = this._tiles[tileID.key];\n if (tile && tile.hasData()) {\n return tile;\n }\n // TileCache ignores wrap in lookup.\n const cachedTile = this._cache.getByKey(tileID.wrapped().key);\n return cachedTile;\n }\n\n /**\n * Resizes the tile cache based on the current viewport's size\n * or the maxTileCacheSize option passed during map creation\n *\n * Larger viewports use more tiles and need larger caches. Larger viewports\n * are more likely to be found on devices with more memory and on pages where\n * the map is more important.\n */\n updateCacheSize(transform: Transform) {\n const widthInTiles = Math.ceil(transform.width / this._source.tileSize) + 1;\n const heightInTiles = Math.ceil(transform.height / this._source.tileSize) + 1;\n const approxTilesInView = widthInTiles * heightInTiles;\n const commonZoomRange = this._maxTileCacheZoomLevels === null ?\n config.MAX_TILE_CACHE_ZOOM_LEVELS : this._maxTileCacheZoomLevels;\n const viewDependentMaxSize = Math.floor(approxTilesInView * commonZoomRange);\n const maxSize = typeof this._maxTileCacheSize === 'number' ?\n Math.min(this._maxTileCacheSize, viewDependentMaxSize) : viewDependentMaxSize;\n\n this._cache.setMaxSize(maxSize);\n }\n\n handleWrapJump(lng: number) {\n // On top of the regular z/x/y values, TileIDs have a `wrap` value that specify\n // which cppy of the world the tile belongs to. For example, at `lng: 10` you\n // might render z/x/y/0 while at `lng: 370` you would render z/x/y/1.\n //\n // When lng values get wrapped (going from `lng: 370` to `long: 10`) you expect\n // to see the same thing on the screen (370 degrees and 10 degrees is the same\n // place in the world) but all the TileIDs will have different wrap values.\n //\n // In order to make this transition seamless, we calculate the rounded difference of\n // \"worlds\" between the last frame and the current frame. If the map panned by\n // a world, then we can assign all the tiles new TileIDs with updated wrap values.\n // For example, assign z/x/y/1 a new id: z/x/y/0. It is the same tile, just rendered\n // in a different position.\n //\n // This enables us to reuse the tiles at more ideal locations and prevent flickering.\n const prevLng = this._prevLng === undefined ? lng : this._prevLng;\n const lngDifference = lng - prevLng;\n const worldDifference = lngDifference / 360;\n const wrapDelta = Math.round(worldDifference);\n this._prevLng = lng;\n\n if (wrapDelta) {\n const tiles: {[_: string]: Tile} = {};\n for (const key in this._tiles) {\n const tile = this._tiles[key];\n tile.tileID = tile.tileID.unwrapTo(tile.tileID.wrap + wrapDelta);\n tiles[tile.tileID.key] = tile;\n }\n this._tiles = tiles;\n\n // Reset tile reload timers\n for (const id in this._timers) {\n clearTimeout(this._timers[id]);\n delete this._timers[id];\n }\n for (const id in this._tiles) {\n const tile = this._tiles[id];\n this._setTileReloadTimer(id, tile);\n }\n }\n }\n\n /**\n * Removes tiles that are outside the viewport and adds new tiles that\n * are inside the viewport.\n */\n update(transform: Transform, terrain?: Terrain) {\n this.transform = transform;\n this.terrain = terrain;\n if (!this._sourceLoaded || this._paused) { return; }\n\n this.updateCacheSize(transform);\n this.handleWrapJump(this.transform.center.lng);\n\n // Covered is a list of retained tiles who's areas are fully covered by other,\n // better, retained tiles. They are not drawn separately.\n this._coveredTiles = {};\n\n let idealTileIDs;\n if (!this.used && !this.usedForTerrain) {\n idealTileIDs = [];\n } else if (this._source.tileID) {\n idealTileIDs = transform.getVisibleUnwrappedCoordinates(this._source.tileID)\n .map((unwrapped) => new OverscaledTileID(unwrapped.canonical.z, unwrapped.wrap, unwrapped.canonical.z, unwrapped.canonical.x, unwrapped.canonical.y));\n } else {\n idealTileIDs = transform.coveringTiles({\n tileSize: this.usedForTerrain ? this.tileSize : this._source.tileSize,\n minzoom: this._source.minzoom,\n maxzoom: this._source.maxzoom,\n roundZoom: this.usedForTerrain ? false : this._source.roundZoom,\n reparseOverscaled: this._source.reparseOverscaled,\n terrain\n });\n\n if (this._source.hasTile) {\n idealTileIDs = idealTileIDs.filter((coord) => (this._source.hasTile as any)(coord));\n }\n }\n\n // Determine the overzooming/underzooming amounts.\n const zoom = transform.coveringZoomLevel(this._source);\n const minCoveringZoom = Math.max(zoom - SourceCache.maxOverzooming, this._source.minzoom);\n const maxCoveringZoom = Math.max(zoom + SourceCache.maxUnderzooming, this._source.minzoom);\n\n // When sourcecache is used for terrain also load parent tiles to avoid flickering when zooming out\n if (this.usedForTerrain) {\n const parents = {};\n for (const tileID of idealTileIDs) {\n if (tileID.canonical.z > this._source.minzoom) {\n const parent = tileID.scaledTo(tileID.canonical.z - 1);\n parents[parent.key] = parent;\n // load very low zoom to calculate tile visibility in transform.coveringTiles and high zoomlevels correct\n const parent2 = tileID.scaledTo(Math.max(this._source.minzoom, Math.min(tileID.canonical.z, 5)));\n parents[parent2.key] = parent2;\n }\n }\n idealTileIDs = idealTileIDs.concat(Object.values(parents));\n }\n\n const noPendingDataEmissions = idealTileIDs.length === 0 && !this._updated && this._didEmitContent;\n this._updated = true;\n // if we won't have any tiles to fetch and content is already emitted\n // there will be no more data emissions, so we need to emit the event with isSourceLoaded = true\n if (noPendingDataEmissions) {\n this.fire(new Event('data', {sourceDataType: 'idle', dataType: 'source', sourceId: this.id}));\n }\n\n // Retain is a list of tiles that we shouldn't delete, even if they are not\n // the most ideal tile for the current viewport. This may include tiles like\n // parent or child tiles that are *already* loaded.\n const retain = this._updateRetainedTiles(idealTileIDs, zoom);\n\n if (isRasterType(this._source.type)) {\n const parentsForFading: {[_: string]: OverscaledTileID} = {};\n const fadingTiles = {};\n const ids = Object.keys(retain);\n const now = browser.now();\n for (const id of ids) {\n const tileID = retain[id];\n\n const tile = this._tiles[id];\n\n // when fadeEndTime is 0, the tile is created but registerFadeDuration\n // has not been called, therefore must be kept in fadingTiles dictionary\n // for next round of rendering\n if (!tile || (tile.fadeEndTime !== 0 && tile.fadeEndTime <= now)) {\n continue;\n }\n\n // if the tile is loaded but still fading in, find parents to cross-fade with it\n const parentTile = this.findLoadedParent(tileID, minCoveringZoom);\n if (parentTile) {\n this._addTile(parentTile.tileID);\n parentsForFading[parentTile.tileID.key] = parentTile.tileID;\n }\n\n fadingTiles[id] = tileID;\n }\n\n // for tiles that are still fading in, also find children to cross-fade with\n this._retainLoadedChildren(fadingTiles, zoom, maxCoveringZoom, retain);\n\n for (const id in parentsForFading) {\n if (!retain[id]) {\n // If a tile is only needed for fading, mark it as covered so that it isn't rendered on it's own.\n this._coveredTiles[id] = true;\n retain[id] = parentsForFading[id];\n }\n }\n\n // disable fading logic in terrain3D mode to avoid rendering two tiles on the same place\n if (terrain) {\n const idealRasterTileIDs: {[_: string]: OverscaledTileID} = {};\n const missingTileIDs: {[_: string]: OverscaledTileID} = {};\n for (const tileID of idealTileIDs) {\n if (this._tiles[tileID.key].hasData())\n idealRasterTileIDs[tileID.key] = tileID;\n else\n missingTileIDs[tileID.key] = tileID;\n }\n // search for a complete set of children for each missing tile\n for (const key in missingTileIDs) {\n const children = missingTileIDs[key].children(this._source.maxzoom);\n if (this._tiles[children[0].key] && this._tiles[children[1].key] && this._tiles[children[2].key] && this._tiles[children[3].key]) {\n idealRasterTileIDs[children[0].key] = retain[children[0].key] = children[0];\n idealRasterTileIDs[children[1].key] = retain[children[1].key] = children[1];\n idealRasterTileIDs[children[2].key] = retain[children[2].key] = children[2];\n idealRasterTileIDs[children[3].key] = retain[children[3].key] = children[3];\n delete missingTileIDs[key];\n }\n }\n // search for parent for each missing tile\n for (const key in missingTileIDs) {\n const parent = this.findLoadedParent(missingTileIDs[key], this._source.minzoom);\n if (parent) {\n idealRasterTileIDs[parent.tileID.key] = retain[parent.tileID.key] = parent.tileID;\n // remove idealTiles which would be rendered twice\n for (const key in idealRasterTileIDs) {\n if (idealRasterTileIDs[key].isChildOf(parent.tileID)) delete idealRasterTileIDs[key];\n }\n }\n }\n // cover all tiles which are not needed\n for (const key in this._tiles) {\n if (!idealRasterTileIDs[key]) this._coveredTiles[key] = true;\n }\n }\n }\n\n for (const retainedId in retain) {\n // Make sure retained tiles always clear any existing fade holds\n // so that if they're removed again their fade timer starts fresh.\n this._tiles[retainedId].clearFadeHold();\n }\n\n // Remove the tiles we don't need anymore.\n const remove = keysDifference(this._tiles, retain);\n for (const tileID of remove) {\n const tile = this._tiles[tileID];\n if (tile.hasSymbolBuckets && !tile.holdingForFade()) {\n tile.setHoldDuration(this.map._fadeDuration);\n } else if (!tile.hasSymbolBuckets || tile.symbolFadeFinished()) {\n this._removeTile(tileID);\n }\n }\n\n // Construct a cache of loaded parents\n this._updateLoadedParentTileCache();\n }\n\n releaseSymbolFadeTiles() {\n for (const id in this._tiles) {\n if (this._tiles[id].holdingForFade()) {\n this._removeTile(id);\n }\n }\n }\n\n _updateRetainedTiles(idealTileIDs: Array, zoom: number): {[_: string]: OverscaledTileID} {\n const retain: {[_: string]: OverscaledTileID} = {};\n const checked: {[_: string]: boolean} = {};\n const minCoveringZoom = Math.max(zoom - SourceCache.maxOverzooming, this._source.minzoom);\n const maxCoveringZoom = Math.max(zoom + SourceCache.maxUnderzooming, this._source.minzoom);\n\n const missingTiles = {};\n for (const tileID of idealTileIDs) {\n const tile = this._addTile(tileID);\n\n // retain the tile even if it's not loaded because it's an ideal tile.\n retain[tileID.key] = tileID;\n\n if (tile.hasData()) continue;\n\n if (zoom < this._source.maxzoom) {\n // save missing tiles that potentially have loaded children\n missingTiles[tileID.key] = tileID;\n }\n }\n\n // retain any loaded children of ideal tiles up to maxCoveringZoom\n this._retainLoadedChildren(missingTiles, zoom, maxCoveringZoom, retain);\n\n for (const tileID of idealTileIDs) {\n let tile = this._tiles[tileID.key];\n\n if (tile.hasData()) continue;\n\n // The tile we require is not yet loaded or does not exist;\n // Attempt to find children that fully cover it.\n\n if (zoom + 1 > this._source.maxzoom) {\n // We're looking for an overzoomed child tile.\n const childCoord = tileID.children(this._source.maxzoom)[0];\n const childTile = this.getTile(childCoord);\n if (!!childTile && childTile.hasData()) {\n retain[childCoord.key] = childCoord;\n continue; // tile is covered by overzoomed child\n }\n } else {\n // check if all 4 immediate children are loaded (i.e. the missing ideal tile is covered)\n const children = tileID.children(this._source.maxzoom);\n\n if (retain[children[0].key] &&\n retain[children[1].key] &&\n retain[children[2].key] &&\n retain[children[3].key]) continue; // tile is covered by children\n }\n\n // We couldn't find child tiles that entirely cover the ideal tile; look for parents now.\n\n // As we ascend up the tile pyramid of the ideal tile, we check whether the parent\n // tile has been previously requested (and errored because we only loop over tiles with no data)\n // in order to determine if we need to request its parent.\n let parentWasRequested = tile.wasRequested();\n\n for (let overscaledZ = tileID.overscaledZ - 1; overscaledZ >= minCoveringZoom; --overscaledZ) {\n const parentId = tileID.scaledTo(overscaledZ);\n\n // Break parent tile ascent if this route has been previously checked by another child.\n if (checked[parentId.key]) break;\n checked[parentId.key] = true;\n\n tile = this.getTile(parentId);\n if (!tile && parentWasRequested) {\n tile = this._addTile(parentId);\n }\n if (tile) {\n const hasData = tile.hasData();\n if (parentWasRequested || hasData) {\n retain[parentId.key] = parentId;\n }\n // Save the current values, since they're the parent of the next iteration\n // of the parent tile ascent loop.\n parentWasRequested = tile.wasRequested();\n if (hasData) break;\n }\n }\n }\n\n return retain;\n }\n\n _updateLoadedParentTileCache() {\n this._loadedParentTiles = {};\n\n for (const tileKey in this._tiles) {\n const path = [];\n let parentTile: Tile;\n let currentId = this._tiles[tileKey].tileID;\n\n // Find the closest loaded ancestor by traversing the tile tree towards the root and\n // caching results along the way\n while (currentId.overscaledZ > 0) {\n\n // Do we have a cached result from previous traversals?\n if (currentId.key in this._loadedParentTiles) {\n parentTile = this._loadedParentTiles[currentId.key];\n break;\n }\n\n path.push(currentId.key);\n\n // Is the parent loaded?\n const parentId = currentId.scaledTo(currentId.overscaledZ - 1);\n parentTile = this._getLoadedTile(parentId);\n if (parentTile) {\n break;\n }\n\n currentId = parentId;\n }\n\n // Cache the result of this traversal to all newly visited tiles\n for (const key of path) {\n this._loadedParentTiles[key] = parentTile;\n }\n }\n }\n\n /**\n * Add a tile, given its coordinate, to the pyramid.\n */\n _addTile(tileID: OverscaledTileID): Tile {\n let tile = this._tiles[tileID.key];\n if (tile)\n return tile;\n\n tile = this._cache.getAndRemove(tileID);\n if (tile) {\n this._setTileReloadTimer(tileID.key, tile);\n // set the tileID because the cached tile could have had a different wrap value\n tile.tileID = tileID;\n this._state.initializeTileState(tile, this.map ? this.map.painter : null);\n if (this._cacheTimers[tileID.key]) {\n clearTimeout(this._cacheTimers[tileID.key]);\n delete this._cacheTimers[tileID.key];\n this._setTileReloadTimer(tileID.key, tile);\n }\n }\n\n const cached = tile;\n\n if (!tile) {\n tile = new Tile(tileID, this._source.tileSize * tileID.overscaleFactor());\n this._loadTile(tile, this._tileLoaded.bind(this, tile, tileID.key, tile.state));\n }\n\n tile.uses++;\n this._tiles[tileID.key] = tile;\n if (!cached) {\n this._source.fire(new Event('dataloading', {tile, coord: tile.tileID, dataType: 'source'}));\n }\n\n return tile;\n }\n\n _setTileReloadTimer(id: string, tile: Tile) {\n if (id in this._timers) {\n clearTimeout(this._timers[id]);\n delete this._timers[id];\n }\n\n const expiryTimeout = tile.getExpiryTimeout();\n if (expiryTimeout) {\n this._timers[id] = setTimeout(() => {\n this._reloadTile(id, 'expired');\n delete this._timers[id];\n }, expiryTimeout);\n }\n }\n\n /**\n * Remove a tile, given its id, from the pyramid\n */\n _removeTile(id: string) {\n const tile = this._tiles[id];\n if (!tile)\n return;\n\n tile.uses--;\n delete this._tiles[id];\n if (this._timers[id]) {\n clearTimeout(this._timers[id]);\n delete this._timers[id];\n }\n\n if (tile.uses > 0)\n return;\n\n if (tile.hasData() && tile.state !== 'reloading') {\n this._cache.add(tile.tileID, tile, tile.getExpiryTimeout());\n } else {\n tile.aborted = true;\n this._abortTile(tile);\n this._unloadTile(tile);\n }\n }\n\n /**\n * Remove all tiles from this pyramid\n */\n clearTiles() {\n this._shouldReloadOnResume = false;\n this._paused = false;\n\n for (const id in this._tiles)\n this._removeTile(id);\n\n this._cache.reset();\n }\n\n /**\n * Search through our current tiles and attempt to find the tiles that\n * cover the given bounds.\n * @param pointQueryGeometry - coordinates of the corners of bounding rectangle\n * @returns result items have `{tile, minX, maxX, minY, maxY}`, where min/max bounding values are the given bounds transformed in into the coordinate space of this tile.\n */\n tilesIn(pointQueryGeometry: Array, maxPitchScaleFactor: number, has3DLayer: boolean): any[] {\n\n const tileResults = [];\n\n const transform = this.transform;\n if (!transform) return tileResults;\n\n const cameraPointQueryGeometry = has3DLayer ?\n transform.getCameraQueryGeometry(pointQueryGeometry) :\n pointQueryGeometry;\n\n const queryGeometry = pointQueryGeometry.map((p: Point) => transform.pointCoordinate(p, this.terrain));\n const cameraQueryGeometry = cameraPointQueryGeometry.map((p: Point) => transform.pointCoordinate(p, this.terrain));\n\n const ids = this.getIds();\n\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n\n for (const p of cameraQueryGeometry) {\n minX = Math.min(minX, p.x);\n minY = Math.min(minY, p.y);\n maxX = Math.max(maxX, p.x);\n maxY = Math.max(maxY, p.y);\n }\n\n for (let i = 0; i < ids.length; i++) {\n const tile = this._tiles[ids[i]];\n if (tile.holdingForFade()) {\n // Tiles held for fading are covered by tiles that are closer to ideal\n continue;\n }\n const tileID = tile.tileID;\n const scale = Math.pow(2, transform.zoom - tile.tileID.overscaledZ);\n const queryPadding = maxPitchScaleFactor * tile.queryPadding * EXTENT / tile.tileSize / scale;\n\n const tileSpaceBounds = [\n tileID.getTilePoint(new MercatorCoordinate(minX, minY)),\n tileID.getTilePoint(new MercatorCoordinate(maxX, maxY))\n ];\n\n if (tileSpaceBounds[0].x - queryPadding < EXTENT && tileSpaceBounds[0].y - queryPadding < EXTENT &&\n tileSpaceBounds[1].x + queryPadding >= 0 && tileSpaceBounds[1].y + queryPadding >= 0) {\n\n const tileSpaceQueryGeometry: Array = queryGeometry.map((c) => tileID.getTilePoint(c));\n const tileSpaceCameraQueryGeometry = cameraQueryGeometry.map((c) => tileID.getTilePoint(c));\n\n tileResults.push({\n tile,\n tileID,\n queryGeometry: tileSpaceQueryGeometry,\n cameraQueryGeometry: tileSpaceCameraQueryGeometry,\n scale\n });\n }\n }\n\n return tileResults;\n }\n\n getVisibleCoordinates(symbolLayer?: boolean): Array {\n const coords = this.getRenderableIds(symbolLayer).map((id) => this._tiles[id].tileID);\n for (const coord of coords) {\n coord.posMatrix = this.transform.calculatePosMatrix(coord.toUnwrapped());\n }\n return coords;\n }\n\n hasTransition() {\n if (this._source.hasTransition()) {\n return true;\n }\n\n if (isRasterType(this._source.type)) {\n const now = browser.now();\n for (const id in this._tiles) {\n const tile = this._tiles[id];\n if (tile.fadeEndTime >= now) {\n return true;\n }\n }\n }\n\n return false;\n }\n\n /**\n * Set the value of a particular state for a feature\n */\n setFeatureState(sourceLayer: string, featureId: number | string, state: any) {\n sourceLayer = sourceLayer || '_geojsonTileLayer';\n this._state.updateState(sourceLayer, featureId, state);\n }\n\n /**\n * Resets the value of a particular state key for a feature\n */\n removeFeatureState(sourceLayer?: string, featureId?: number | string, key?: string) {\n sourceLayer = sourceLayer || '_geojsonTileLayer';\n this._state.removeFeatureState(sourceLayer, featureId, key);\n }\n\n /**\n * Get the entire state object for a feature\n */\n getFeatureState(sourceLayer: string, featureId: number | string) {\n sourceLayer = sourceLayer || '_geojsonTileLayer';\n return this._state.getState(sourceLayer, featureId);\n }\n\n /**\n * Sets the set of keys that the tile depends on. This allows tiles to\n * be reloaded when their dependencies change.\n */\n setDependencies(tileKey: string, namespace: string, dependencies: Array) {\n const tile = this._tiles[tileKey];\n if (tile) {\n tile.setDependencies(namespace, dependencies);\n }\n }\n\n /**\n * Reloads all tiles that depend on the given keys.\n */\n reloadTilesForDependencies(namespaces: Array, keys: Array) {\n for (const id in this._tiles) {\n const tile = this._tiles[id];\n if (tile.hasDependency(namespaces, keys)) {\n this._reloadTile(id, 'reloading');\n }\n }\n this._cache.filter(tile => !tile.hasDependency(namespaces, keys));\n }\n}\n\nSourceCache.maxOverzooming = 10;\nSourceCache.maxUnderzooming = 3;\n\nfunction compareTileId(a: OverscaledTileID, b: OverscaledTileID): number {\n // Different copies of the world are sorted based on their distance to the center.\n // Wrap values are converted to unsigned distances by reserving odd number for copies\n // with negative wrap and even numbers for copies with positive wrap.\n const aWrap = Math.abs(a.wrap * 2) - +(a.wrap < 0);\n const bWrap = Math.abs(b.wrap * 2) - +(b.wrap < 0);\n return a.overscaledZ - b.overscaledZ || bWrap - aWrap || b.canonical.y - a.canonical.y || b.canonical.x - a.canonical.x;\n}\n\nfunction isRasterType(type) {\n return type === 'raster' || type === 'image' || type === 'video';\n}\n","import {workerFactory} from './web_worker';\nimport {browser} from './browser';\nimport {isSafari} from './util';\nimport {ActorTarget} from './actor';\n\nexport const PRELOAD_POOL_ID = 'mapboxgl_preloaded_worker_pool';\n\n/**\n * Constructs a worker pool.\n */\nexport class WorkerPool {\n static workerCount: number;\n\n active: {\n [_ in number | string]: boolean;\n };\n workers: Array;\n\n constructor() {\n this.active = {};\n }\n\n acquire(mapId: number | string): Array {\n if (!this.workers) {\n // Lazily look up the value of mapboxgl.workerCount so that\n // client code has had a chance to set it.\n this.workers = [];\n while (this.workers.length < WorkerPool.workerCount) {\n this.workers.push(workerFactory());\n }\n }\n\n this.active[mapId] = true;\n return this.workers.slice();\n }\n\n release(mapId: number | string) {\n delete this.active[mapId];\n if (this.numActive() === 0) {\n this.workers.forEach((w) => {\n w.terminate();\n });\n this.workers = null;\n }\n }\n\n isPreloaded(): boolean {\n return !!this.active[PRELOAD_POOL_ID];\n }\n\n numActive(): number {\n return Object.keys(this.active).length;\n }\n}\n\n// Based on results from A/B testing: https://github.com/maplibre/maplibre-gl-js/pull/2354\nconst availableLogicalProcessors = Math.floor(browser.hardwareConcurrency / 2);\nWorkerPool.workerCount = isSafari(globalThis) ? Math.max(Math.min(availableLogicalProcessors, 3), 1) : 1;\n","import {config} from './config';\n\nimport type {WorkerSource} from '../source/worker_source';\n\nexport interface WorkerGlobalScopeInterface {\n importScripts(...urls: Array): void;\n registerWorkerSource: (\n b: string,\n a: {\n new(...args: any): WorkerSource;\n }\n ) => void;\n registerRTLTextPlugin: (_: any) => void;\n}\n\nexport function workerFactory() {\n return new Worker(config.WORKER_URL);\n}\n","import {WorkerPool, PRELOAD_POOL_ID} from './worker_pool';\n\nlet globalWorkerPool;\n\n/**\n * Creates (if necessary) and returns the single, global WorkerPool instance\n * to be shared across each Map\n */\nexport function getGlobalWorkerPool() {\n if (!globalWorkerPool) {\n globalWorkerPool = new WorkerPool();\n }\n return globalWorkerPool;\n}\n\nexport function prewarm() {\n const workerPool = getGlobalWorkerPool();\n workerPool.acquire(PRELOAD_POOL_ID);\n}\n\nexport function clearPrewarmedResources() {\n const pool = globalWorkerPool;\n if (pool) {\n // Remove the pool only if all maps that referenced the preloaded global worker pool have been removed.\n if (pool.isPreloaded() && pool.numActive() === 1) {\n pool.release(PRELOAD_POOL_ID);\n globalWorkerPool = null;\n } else {\n console.warn('Could not clear WebWorkers since there are active Map instances that still reference it. The pre-warmed WebWorker pool can only be cleared when all map instances have been removed with map.remove()');\n }\n }\n}\n","import {clamp} from '../util/util';\nimport Point from '@mapbox/point-geometry';\n\nexport class PathInterpolator {\n points: Array;\n length: number;\n paddedLength: number;\n padding: number;\n _distances: Array;\n\n constructor(points_?: Array | null, padding_?: number | null) {\n this.reset(points_, padding_);\n }\n\n reset(points_?: Array | null, padding_?: number | null) {\n this.points = points_ || [];\n\n // Compute cumulative distance from first point to every other point in the segment.\n // Last entry in the array is total length of the path\n this._distances = [0.0];\n\n for (let i = 1; i < this.points.length; i++) {\n this._distances[i] = this._distances[i - 1] + this.points[i].dist(this.points[i - 1]);\n }\n\n this.length = this._distances[this._distances.length - 1];\n this.padding = Math.min(padding_ || 0, this.length * 0.5);\n this.paddedLength = this.length - this.padding * 2.0;\n }\n\n lerp(t: number): Point {\n if (this.points.length === 1) {\n return this.points[0];\n }\n\n t = clamp(t, 0, 1);\n\n // Find the correct segment [p0, p1] where p0 <= x < p1\n let currentIndex = 1;\n let distOfCurrentIdx = this._distances[currentIndex];\n const distToTarget = t * this.paddedLength + this.padding;\n\n while (distOfCurrentIdx < distToTarget && currentIndex < this._distances.length) {\n distOfCurrentIdx = this._distances[++currentIndex];\n }\n\n // Interpolate between the two points of the segment\n const idxOfPrevPoint = currentIndex - 1;\n const distOfPrevIdx = this._distances[idxOfPrevPoint];\n const segmentLength = distOfCurrentIdx - distOfPrevIdx;\n const segmentT = segmentLength > 0 ? (distToTarget - distOfPrevIdx) / segmentLength : 0;\n\n return this.points[idxOfPrevPoint].mult(1.0 - segmentT).add(this.points[currentIndex].mult(segmentT));\n }\n}\n","import type {OverlapMode} from '../style/style_layer/overlap_mode';\n\ntype QueryArgs = {\n hitTest: boolean;\n overlapMode?: OverlapMode;\n circle?: {\n x: number;\n y: number;\n radius: number;\n };\n seenUids: {\n box: {\n [_: number]: boolean;\n };\n circle: {\n [_: number]: boolean;\n };\n };\n};\n\ntype QueryResult = {\n key: T;\n x1: number;\n y1: number;\n x2: number;\n y2: number;\n};\n\n/**\n * A key for the grid\n */\nexport type GridKey = {\n overlapMode?: OverlapMode;\n}\n\nfunction overlapAllowed(overlapA: OverlapMode, overlapB: OverlapMode): boolean {\n let allowed = true;\n\n if (overlapA === 'always') {\n // symbol A using 'always' overlap - allowed to overlap anything.\n } else if (overlapA === 'never' || overlapB === 'never') {\n // symbol A using 'never' overlap - can't overlap anything\n // symbol A using 'cooperative' overlap - can overlap 'always' or 'cooperative' symbol; can't overlap 'never'\n allowed = false;\n }\n\n return allowed;\n}\n\n/**\n * @internal\n * GridIndex is a data structure for testing the intersection of\n * circles and rectangles in a 2d plane.\n * It is optimized for rapid insertion and querying.\n * GridIndex splits the plane into a set of \"cells\" and keeps track\n * of which geometries intersect with each cell. At query time,\n * full geometry comparisons are only done for items that share\n * at least one cell. As long as the geometries are relatively\n * uniformly distributed across the plane, this greatly reduces\n * the number of comparisons necessary.\n */\nexport class GridIndex {\n circleKeys: Array;\n boxKeys: Array;\n boxCells: Array>;\n circleCells: Array>;\n bboxes: Array;\n circles: Array;\n xCellCount: number;\n yCellCount: number;\n width: number;\n height: number;\n xScale: number;\n yScale: number;\n boxUid: number;\n circleUid: number;\n\n constructor (width: number, height: number, cellSize: number) {\n const boxCells = this.boxCells = [];\n const circleCells = this.circleCells = [];\n\n // More cells -> fewer geometries to check per cell, but items tend\n // to be split across more cells.\n // Sweet spot allows most small items to fit in one cell\n this.xCellCount = Math.ceil(width / cellSize);\n this.yCellCount = Math.ceil(height / cellSize);\n\n for (let i = 0; i < this.xCellCount * this.yCellCount; i++) {\n boxCells.push([]);\n circleCells.push([]);\n }\n this.circleKeys = [];\n this.boxKeys = [];\n this.bboxes = [];\n this.circles = [];\n\n this.width = width;\n this.height = height;\n this.xScale = this.xCellCount / width;\n this.yScale = this.yCellCount / height;\n this.boxUid = 0;\n this.circleUid = 0;\n }\n\n keysLength() {\n return this.boxKeys.length + this.circleKeys.length;\n }\n\n insert(key: T, x1: number, y1: number, x2: number, y2: number) {\n this._forEachCell(x1, y1, x2, y2, this._insertBoxCell, this.boxUid++);\n this.boxKeys.push(key);\n this.bboxes.push(x1);\n this.bboxes.push(y1);\n this.bboxes.push(x2);\n this.bboxes.push(y2);\n }\n\n insertCircle(key: T, x: number, y: number, radius: number) {\n // Insert circle into grid for all cells in the circumscribing square\n // It's more than necessary (by a factor of 4/PI), but fast to insert\n this._forEachCell(x - radius, y - radius, x + radius, y + radius, this._insertCircleCell, this.circleUid++);\n this.circleKeys.push(key);\n this.circles.push(x);\n this.circles.push(y);\n this.circles.push(radius);\n }\n\n private _insertBoxCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) {\n this.boxCells[cellIndex].push(uid);\n }\n\n private _insertCircleCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) {\n this.circleCells[cellIndex].push(uid);\n }\n\n private _query(x1: number, y1: number, x2: number, y2: number, hitTest: boolean, overlapMode: OverlapMode, predicate?: (key: T) => boolean): Array> {\n if (x2 < 0 || x1 > this.width || y2 < 0 || y1 > this.height) {\n return [];\n }\n const result: Array> = [];\n if (x1 <= 0 && y1 <= 0 && this.width <= x2 && this.height <= y2) {\n if (hitTest) {\n // Covers the entire grid, so collides with everything\n return [{\n key: null,\n x1,\n y1,\n x2,\n y2\n }];\n }\n for (let boxUid = 0; boxUid < this.boxKeys.length; boxUid++) {\n result.push({\n key: this.boxKeys[boxUid],\n x1: this.bboxes[boxUid * 4],\n y1: this.bboxes[boxUid * 4 + 1],\n x2: this.bboxes[boxUid * 4 + 2],\n y2: this.bboxes[boxUid * 4 + 3]\n });\n }\n for (let circleUid = 0; circleUid < this.circleKeys.length; circleUid++) {\n const x = this.circles[circleUid * 3];\n const y = this.circles[circleUid * 3 + 1];\n const radius = this.circles[circleUid * 3 + 2];\n result.push({\n key: this.circleKeys[circleUid],\n x1: x - radius,\n y1: y - radius,\n x2: x + radius,\n y2: y + radius\n });\n }\n } else {\n const queryArgs: QueryArgs = {\n hitTest,\n overlapMode,\n seenUids: {box: {}, circle: {}}\n };\n this._forEachCell(x1, y1, x2, y2, this._queryCell, result, queryArgs, predicate);\n }\n\n return result;\n }\n\n query(x1: number, y1: number, x2: number, y2: number): Array> {\n return this._query(x1, y1, x2, y2, false, null);\n }\n\n hitTest(x1: number, y1: number, x2: number, y2: number, overlapMode: OverlapMode, predicate?: (key: T) => boolean): boolean {\n return this._query(x1, y1, x2, y2, true, overlapMode, predicate).length > 0;\n }\n\n hitTestCircle(x: number, y: number, radius: number, overlapMode: OverlapMode, predicate?: (key: T) => boolean): boolean {\n // Insert circle into grid for all cells in the circumscribing square\n // It's more than necessary (by a factor of 4/PI), but fast to insert\n const x1 = x - radius;\n const x2 = x + radius;\n const y1 = y - radius;\n const y2 = y + radius;\n if (x2 < 0 || x1 > this.width || y2 < 0 || y1 > this.height) {\n return false;\n }\n\n // Box query early exits if the bounding box is larger than the grid, but we don't do\n // the equivalent calculation for circle queries because early exit is less likely\n // and the calculation is more expensive\n const result: boolean[] = [];\n const queryArgs: QueryArgs = {\n hitTest: true,\n overlapMode,\n circle: {x, y, radius},\n seenUids: {box: {}, circle: {}}\n };\n this._forEachCell(x1, y1, x2, y2, this._queryCellCircle, result, queryArgs, predicate);\n return result.length > 0;\n }\n\n private _queryCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, result: Array>, queryArgs: QueryArgs, predicate?: (key: T) => boolean): boolean {\n const {seenUids, hitTest, overlapMode} = queryArgs;\n const boxCell = this.boxCells[cellIndex];\n\n if (boxCell !== null) {\n const bboxes = this.bboxes;\n for (const boxUid of boxCell) {\n if (!seenUids.box[boxUid]) {\n seenUids.box[boxUid] = true;\n const offset = boxUid * 4;\n const key = this.boxKeys[boxUid];\n\n if ((x1 <= bboxes[offset + 2]) &&\n (y1 <= bboxes[offset + 3]) &&\n (x2 >= bboxes[offset + 0]) &&\n (y2 >= bboxes[offset + 1]) &&\n (!predicate || predicate(key))) {\n if (!hitTest || !overlapAllowed(overlapMode, key.overlapMode)) {\n result.push({\n key,\n x1: bboxes[offset],\n y1: bboxes[offset + 1],\n x2: bboxes[offset + 2],\n y2: bboxes[offset + 3]\n });\n if (hitTest) {\n // true return value stops the query after first match\n return true;\n }\n }\n }\n }\n }\n }\n const circleCell = this.circleCells[cellIndex];\n if (circleCell !== null) {\n const circles = this.circles;\n for (const circleUid of circleCell) {\n if (!seenUids.circle[circleUid]) {\n seenUids.circle[circleUid] = true;\n const offset = circleUid * 3;\n const key = this.circleKeys[circleUid];\n\n if (this._circleAndRectCollide(\n circles[offset],\n circles[offset + 1],\n circles[offset + 2],\n x1,\n y1,\n x2,\n y2) &&\n (!predicate || predicate(key))) {\n if (!hitTest || !overlapAllowed(overlapMode, key.overlapMode)) {\n const x = circles[offset];\n const y = circles[offset + 1];\n const radius = circles[offset + 2];\n result.push({\n key,\n x1: x - radius,\n y1: y - radius,\n x2: x + radius,\n y2: y + radius\n });\n if (hitTest) {\n // true return value stops the query after first match\n return true;\n }\n }\n }\n }\n }\n }\n\n // false return to continue query\n return false;\n }\n\n private _queryCellCircle(x1: number, y1: number, x2: number, y2: number, cellIndex: number, result: Array, queryArgs: QueryArgs, predicate?: (key: T) => boolean): boolean {\n const {circle, seenUids, overlapMode} = queryArgs;\n const boxCell = this.boxCells[cellIndex];\n\n if (boxCell !== null) {\n const bboxes = this.bboxes;\n for (const boxUid of boxCell) {\n if (!seenUids.box[boxUid]) {\n seenUids.box[boxUid] = true;\n const offset = boxUid * 4;\n const key = this.boxKeys[boxUid];\n if (this._circleAndRectCollide(\n circle.x,\n circle.y,\n circle.radius,\n bboxes[offset + 0],\n bboxes[offset + 1],\n bboxes[offset + 2],\n bboxes[offset + 3]) &&\n (!predicate || predicate(key)) &&\n !overlapAllowed(overlapMode, key.overlapMode)) {\n result.push(true);\n return true;\n }\n }\n }\n }\n\n const circleCell = this.circleCells[cellIndex];\n if (circleCell !== null) {\n const circles = this.circles;\n for (const circleUid of circleCell) {\n if (!seenUids.circle[circleUid]) {\n seenUids.circle[circleUid] = true;\n const offset = circleUid * 3;\n const key = this.circleKeys[circleUid];\n if (this._circlesCollide(\n circles[offset],\n circles[offset + 1],\n circles[offset + 2],\n circle.x,\n circle.y,\n circle.radius) &&\n (!predicate || predicate(key)) &&\n !overlapAllowed(overlapMode, key.overlapMode)) {\n result.push(true);\n return true;\n }\n }\n }\n }\n }\n\n private _forEachCell(\n x1: number,\n y1: number,\n x2: number,\n y2: number,\n fn: (x1: number, y1: number, x2: number, y2: number, cellIndex: number, arg1: TArg, arg2?: QueryArgs, predicate?: (key: T) => boolean) => boolean | void,\n arg1: TArg,\n arg2?: QueryArgs,\n predicate?: (key: T) => boolean) {\n const cx1 = this._convertToXCellCoord(x1);\n const cy1 = this._convertToYCellCoord(y1);\n const cx2 = this._convertToXCellCoord(x2);\n const cy2 = this._convertToYCellCoord(y2);\n\n for (let x = cx1; x <= cx2; x++) {\n for (let y = cy1; y <= cy2; y++) {\n const cellIndex = this.xCellCount * y + x;\n if (fn.call(this, x1, y1, x2, y2, cellIndex, arg1, arg2, predicate)) return;\n }\n }\n }\n\n private _convertToXCellCoord(x: number) {\n return Math.max(0, Math.min(this.xCellCount - 1, Math.floor(x * this.xScale)));\n }\n\n private _convertToYCellCoord(y: number) {\n return Math.max(0, Math.min(this.yCellCount - 1, Math.floor(y * this.yScale)));\n }\n\n private _circlesCollide(x1: number, y1: number, r1: number, x2: number, y2: number, r2: number): boolean {\n const dx = x2 - x1;\n const dy = y2 - y1;\n const bothRadii = r1 + r2;\n return (bothRadii * bothRadii) > (dx * dx + dy * dy);\n }\n\n private _circleAndRectCollide(\n circleX: number,\n circleY: number,\n radius: number,\n x1: number,\n y1: number,\n x2: number,\n y2: number\n ): boolean {\n const halfRectWidth = (x2 - x1) / 2;\n const distX = Math.abs(circleX - (x1 + halfRectWidth));\n if (distX > (halfRectWidth + radius)) {\n return false;\n }\n\n const halfRectHeight = (y2 - y1) / 2;\n const distY = Math.abs(circleY - (y1 + halfRectHeight));\n if (distY > (halfRectHeight + radius)) {\n return false;\n }\n\n if (distX <= halfRectWidth || distY <= halfRectHeight) {\n return true;\n }\n\n const dx = distX - halfRectWidth;\n const dy = distY - halfRectHeight;\n return (dx * dx + dy * dy <= (radius * radius));\n }\n}\n","import Point from '@mapbox/point-geometry';\n\nimport {mat4, vec4} from 'gl-matrix';\nimport * as symbolSize from './symbol_size';\nimport {addDynamicAttributes} from '../data/bucket/symbol_bucket';\n\nimport type {Painter} from '../render/painter';\nimport type {Transform} from '../geo/transform';\nimport type {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport type {\n GlyphOffsetArray,\n SymbolLineVertexArray,\n SymbolDynamicLayoutArray\n} from '../data/array_types.g';\nimport {WritingMode} from '../symbol/shaping';\nimport {findLineIntersection} from '../util/util';\n\nexport {updateLineLabels, hideGlyphs, getLabelPlaneMatrix, getGlCoordMatrix, project, getPerspectiveRatio, placeFirstAndLastGlyph, placeGlyphAlongLine, xyTransformMat4, projectVertexToViewport, findOffsetIntersectionPoint, transformToOffsetNormal};\n\n/*\n * # Overview of coordinate spaces\n *\n * ## Tile coordinate spaces\n * Each label has an anchor. Some labels have corresponding line geometries.\n * The points for both anchors and lines are stored in tile units. Each tile has it's own\n * coordinate space going from (0, 0) at the top left to (EXTENT, EXTENT) at the bottom right.\n *\n * ## GL coordinate space\n * At the end of everything, the vertex shader needs to produce a position in GL coordinate space,\n * which is (-1, 1) at the top left and (1, -1) in the bottom right.\n *\n * ## Map pixel coordinate spaces\n * Each tile has a pixel coordinate space. It's just the tile units scaled so that one unit is\n * whatever counts as 1 pixel at the current zoom.\n * This space is used for pitch-alignment=map, rotation-alignment=map\n *\n * ## Rotated map pixel coordinate spaces\n * Like the above, but rotated so axis of the space are aligned with the viewport instead of the tile.\n * This space is used for pitch-alignment=map, rotation-alignment=viewport\n *\n * ## Viewport pixel coordinate space\n * (0, 0) is at the top left of the canvas and (pixelWidth, pixelHeight) is at the bottom right corner\n * of the canvas. This space is used for pitch-alignment=viewport\n *\n *\n * # Vertex projection\n * It goes roughly like this:\n * 1. project the anchor and line from tile units into the correct label coordinate space\n * - map pixel space pitch-alignment=map rotation-alignment=map\n * - rotated map pixel space pitch-alignment=map rotation-alignment=viewport\n * - viewport pixel space pitch-alignment=viewport rotation-alignment=*\n * 2. if the label follows a line, find the point along the line that is the correct distance from the anchor.\n * 3. add the glyph's corner offset to the point from step 3\n * 4. convert from the label coordinate space to gl coordinates\n *\n * For horizontal labels we want to do step 1 in the shader for performance reasons (no cpu work).\n * This is what `u_label_plane_matrix` is used for.\n * For labels aligned with lines we have to steps 1 and 2 on the cpu since we need access to the line geometry.\n * This is what `updateLineLabels(...)` does.\n * Since the conversion is handled on the cpu we just set `u_label_plane_matrix` to an identity matrix.\n *\n * Steps 3 and 4 are done in the shaders for all labels.\n */\n\n/*\n * Returns a matrix for converting from tile units to the correct label coordinate space.\n */\nfunction getLabelPlaneMatrix(posMatrix: mat4,\n pitchWithMap: boolean,\n rotateWithMap: boolean,\n transform: Transform,\n pixelsToTileUnits: number) {\n const m = mat4.create();\n if (pitchWithMap) {\n mat4.scale(m, m, [1 / pixelsToTileUnits, 1 / pixelsToTileUnits, 1]);\n if (!rotateWithMap) {\n mat4.rotateZ(m, m, transform.angle);\n }\n } else {\n mat4.multiply(m, transform.labelPlaneMatrix, posMatrix);\n }\n return m;\n}\n\n/*\n * Returns a matrix for converting from the correct label coordinate space to gl coords.\n */\nfunction getGlCoordMatrix(posMatrix: mat4,\n pitchWithMap: boolean,\n rotateWithMap: boolean,\n transform: Transform,\n pixelsToTileUnits: number) {\n if (pitchWithMap) {\n const m = mat4.clone(posMatrix);\n mat4.scale(m, m, [pixelsToTileUnits, pixelsToTileUnits, 1]);\n if (!rotateWithMap) {\n mat4.rotateZ(m, m, -transform.angle);\n }\n return m;\n } else {\n return transform.glCoordMatrix;\n }\n}\n\nfunction project(point: Point, matrix: mat4, getElevation?: (x: number, y: number) => number) {\n let pos;\n if (getElevation) { // slow because of handle z-index\n pos = [point.x, point.y, getElevation(point.x, point.y), 1] as vec4;\n vec4.transformMat4(pos, pos, matrix);\n } else { // fast because of ignore z-index\n pos = [point.x, point.y, 0, 1] as vec4;\n xyTransformMat4(pos, pos, matrix);\n }\n const w = pos[3];\n return {\n point: new Point(pos[0] / w, pos[1] / w),\n signedDistanceFromCamera: w\n };\n}\n\nfunction getPerspectiveRatio(cameraToCenterDistance: number, signedDistanceFromCamera: number): number {\n return 0.5 + 0.5 * (cameraToCenterDistance / signedDistanceFromCamera);\n}\n\nfunction isVisible(anchorPos: vec4,\n clippingBuffer: [number, number]) {\n const x = anchorPos[0] / anchorPos[3];\n const y = anchorPos[1] / anchorPos[3];\n const inPaddedViewport = (\n x >= -clippingBuffer[0] &&\n x <= clippingBuffer[0] &&\n y >= -clippingBuffer[1] &&\n y <= clippingBuffer[1]);\n return inPaddedViewport;\n}\n\n/*\n * Update the `dynamicLayoutVertexBuffer` for the buffer with the correct glyph positions for the current map view.\n * This is only run on labels that are aligned with lines. Horizontal labels are handled entirely in the shader.\n */\nfunction updateLineLabels(bucket: SymbolBucket,\n posMatrix: mat4,\n painter: Painter,\n isText: boolean,\n labelPlaneMatrix: mat4,\n glCoordMatrix: mat4,\n pitchWithMap: boolean,\n keepUpright: boolean,\n rotateToLine: boolean,\n getElevation: (x: number, y: number) => number) {\n\n const sizeData = isText ? bucket.textSizeData : bucket.iconSizeData;\n const partiallyEvaluatedSize = symbolSize.evaluateSizeForZoom(sizeData, painter.transform.zoom);\n\n const clippingBuffer: [number, number] = [256 / painter.width * 2 + 1, 256 / painter.height * 2 + 1];\n\n const dynamicLayoutVertexArray = isText ?\n bucket.text.dynamicLayoutVertexArray :\n bucket.icon.dynamicLayoutVertexArray;\n dynamicLayoutVertexArray.clear();\n\n const lineVertexArray = bucket.lineVertexArray;\n const placedSymbols = isText ? bucket.text.placedSymbolArray : bucket.icon.placedSymbolArray;\n\n const aspectRatio = painter.transform.width / painter.transform.height;\n\n let useVertical = false;\n\n for (let s = 0; s < placedSymbols.length; s++) {\n const symbol = placedSymbols.get(s);\n\n // Don't do calculations for vertical glyphs unless the previous symbol was horizontal\n // and we determined that vertical glyphs were necessary.\n // Also don't do calculations for symbols that are collided and fully faded out\n if (symbol.hidden || symbol.writingMode === WritingMode.vertical && !useVertical) {\n hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray);\n continue;\n }\n // Awkward... but we're counting on the paired \"vertical\" symbol coming immediately after its horizontal counterpart\n useVertical = false;\n\n let anchorPos;\n if (getElevation) { // slow because of handle z-index\n anchorPos = [symbol.anchorX, symbol.anchorY, getElevation(symbol.anchorX, symbol.anchorY), 1] as vec4;\n vec4.transformMat4(anchorPos, anchorPos, posMatrix);\n } else { // fast because of ignore z-index\n anchorPos = [symbol.anchorX, symbol.anchorY, 0, 1] as vec4;\n xyTransformMat4(anchorPos, anchorPos, posMatrix);\n }\n\n // Don't bother calculating the correct point for invisible labels.\n if (!isVisible(anchorPos, clippingBuffer)) {\n hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray);\n continue;\n }\n\n const cameraToAnchorDistance = anchorPos[3];\n const perspectiveRatio = getPerspectiveRatio(painter.transform.cameraToCenterDistance, cameraToAnchorDistance);\n\n const fontSize = symbolSize.evaluateSizeForFeature(sizeData, partiallyEvaluatedSize, symbol);\n const pitchScaledFontSize = pitchWithMap ? fontSize / perspectiveRatio : fontSize * perspectiveRatio;\n\n const tileAnchorPoint = new Point(symbol.anchorX, symbol.anchorY);\n const anchorPoint = project(tileAnchorPoint, labelPlaneMatrix, getElevation).point;\n const projectionCache = {projections: {}, offsets: {}};\n\n const placeUnflipped: any = placeGlyphsAlongLine(symbol, pitchScaledFontSize, false /*unflipped*/, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix,\n bucket.glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation);\n\n useVertical = placeUnflipped.useVertical;\n\n if (placeUnflipped.notEnoughRoom || useVertical ||\n (placeUnflipped.needsFlipping &&\n (placeGlyphsAlongLine(symbol, pitchScaledFontSize, true /*flipped*/, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix,\n bucket.glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation) as any).notEnoughRoom)) {\n hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray);\n }\n }\n\n if (isText) {\n bucket.text.dynamicLayoutVertexBuffer.updateData(dynamicLayoutVertexArray);\n } else {\n bucket.icon.dynamicLayoutVertexBuffer.updateData(dynamicLayoutVertexArray);\n }\n}\n\ntype FirstAndLastGlyphPlacement = {\n first: PlacedGlyph;\n last: PlacedGlyph;\n} | null;\n\n/*\n * Place the first and last glyph of a line label, projected to the label plane.\n * This function is called both during collision detection (to determine the label's size)\n * and during line label rendering (to make sure the label fits on the line geometry with\n * the current camera position, which may differ from the position used during collision detection).\n *\n * Calling this function has the effect of populating the \"projectionCache\" with all projected\n * vertex locations the label will need, making future calls to placeGlyphAlongLine (for all the\n * intermediate glyphs) much cheaper.\n *\n * Returns null if the label can't fit on the geometry\n */\nfunction placeFirstAndLastGlyph(fontScale: number, glyphOffsetArray: GlyphOffsetArray, lineOffsetX: number, lineOffsetY: number, flip: boolean, anchorPoint: Point, tileAnchorPoint: Point, symbol: any, lineVertexArray: SymbolLineVertexArray, labelPlaneMatrix: mat4, projectionCache: ProjectionCache, rotateToLine: boolean, getElevation: (x: number, y: number) => number): FirstAndLastGlyphPlacement {\n const glyphEndIndex = symbol.glyphStartIndex + symbol.numGlyphs;\n const lineStartIndex = symbol.lineStartIndex;\n const lineEndIndex = symbol.lineStartIndex + symbol.lineLength;\n\n const firstGlyphOffset = glyphOffsetArray.getoffsetX(symbol.glyphStartIndex);\n const lastGlyphOffset = glyphOffsetArray.getoffsetX(glyphEndIndex - 1);\n\n const firstPlacedGlyph = placeGlyphAlongLine(fontScale * firstGlyphOffset, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment,\n lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation);\n if (!firstPlacedGlyph)\n return null;\n\n const lastPlacedGlyph = placeGlyphAlongLine(fontScale * lastGlyphOffset, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment,\n lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation);\n if (!lastPlacedGlyph)\n return null;\n\n return {first: firstPlacedGlyph, last: lastPlacedGlyph};\n}\n\nfunction requiresOrientationChange(writingMode, firstPoint, lastPoint, aspectRatio) {\n if (writingMode === WritingMode.horizontal) {\n // On top of choosing whether to flip, choose whether to render this version of the glyphs or the alternate\n // vertical glyphs. We can't just filter out vertical glyphs in the horizontal range because the horizontal\n // and vertical versions can have slightly different projections which could lead to angles where both or\n // neither showed.\n const rise = Math.abs(lastPoint.y - firstPoint.y);\n const run = Math.abs(lastPoint.x - firstPoint.x) * aspectRatio;\n if (rise > run) {\n return {useVertical: true};\n }\n }\n\n if (writingMode === WritingMode.vertical ? firstPoint.y < lastPoint.y : firstPoint.x > lastPoint.x) {\n // Includes \"horizontalOnly\" case for labels without vertical glyphs\n return {needsFlipping: true};\n }\n\n return null;\n}\n\n/*\n* Place first and last glyph along the line projected to label plane, and if they fit\n* iterate through all the intermediate glyphs, calculating their label plane positions\n* from the projected line.\n*\n* Finally, add resulting glyph position calculations to dynamicLayoutVertexArray for\n* upload to the GPU\n*/\nfunction placeGlyphsAlongLine(symbol, fontSize, flip, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix, glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation) {\n const fontScale = fontSize / 24;\n const lineOffsetX = symbol.lineOffsetX * fontScale;\n const lineOffsetY = symbol.lineOffsetY * fontScale;\n\n let placedGlyphs;\n if (symbol.numGlyphs > 1) {\n const glyphEndIndex = symbol.glyphStartIndex + symbol.numGlyphs;\n const lineStartIndex = symbol.lineStartIndex;\n const lineEndIndex = symbol.lineStartIndex + symbol.lineLength;\n\n // Place the first and the last glyph in the label first, so we can figure out\n // the overall orientation of the label and determine whether it needs to be flipped in keepUpright mode\n const firstAndLastGlyph = placeFirstAndLastGlyph(fontScale, glyphOffsetArray, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation);\n if (!firstAndLastGlyph) {\n return {notEnoughRoom: true};\n }\n const firstPoint = project(firstAndLastGlyph.first.point, glCoordMatrix, getElevation).point;\n const lastPoint = project(firstAndLastGlyph.last.point, glCoordMatrix, getElevation).point;\n\n if (keepUpright && !flip) {\n const orientationChange = requiresOrientationChange(symbol.writingMode, firstPoint, lastPoint, aspectRatio);\n if (orientationChange) {\n return orientationChange;\n }\n }\n\n placedGlyphs = [firstAndLastGlyph.first];\n for (let glyphIndex = symbol.glyphStartIndex + 1; glyphIndex < glyphEndIndex - 1; glyphIndex++) {\n // Since first and last glyph fit on the line, we're sure that the rest of the glyphs can be placed\n placedGlyphs.push(placeGlyphAlongLine(fontScale * glyphOffsetArray.getoffsetX(glyphIndex), lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment,\n lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation));\n }\n placedGlyphs.push(firstAndLastGlyph.last);\n } else {\n // Only a single glyph to place\n // So, determine whether to flip based on projected angle of the line segment it's on\n if (keepUpright && !flip) {\n const a = project(tileAnchorPoint, posMatrix, getElevation).point;\n const tileVertexIndex = (symbol.lineStartIndex + symbol.segment + 1);\n const tileSegmentEnd = new Point(lineVertexArray.getx(tileVertexIndex), lineVertexArray.gety(tileVertexIndex));\n const projectedVertex = project(tileSegmentEnd, posMatrix, getElevation);\n // We know the anchor will be in the viewport, but the end of the line segment may be\n // behind the plane of the camera, in which case we can use a point at any arbitrary (closer)\n // point on the segment.\n const b = (projectedVertex.signedDistanceFromCamera > 0) ?\n projectedVertex.point :\n projectTruncatedLineSegment(tileAnchorPoint, tileSegmentEnd, a, 1, posMatrix, getElevation);\n\n const orientationChange = requiresOrientationChange(symbol.writingMode, a, b, aspectRatio);\n if (orientationChange) {\n return orientationChange;\n }\n }\n const singleGlyph = placeGlyphAlongLine(fontScale * glyphOffsetArray.getoffsetX(symbol.glyphStartIndex), lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment,\n symbol.lineStartIndex, symbol.lineStartIndex + symbol.lineLength, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation);\n if (!singleGlyph)\n return {notEnoughRoom: true};\n\n placedGlyphs = [singleGlyph];\n }\n\n for (const glyph of placedGlyphs) {\n addDynamicAttributes(dynamicLayoutVertexArray, glyph.point, glyph.angle);\n }\n return {};\n}\n\nfunction projectTruncatedLineSegment(previousTilePoint: Point, currentTilePoint: Point, previousProjectedPoint: Point, minimumLength: number, projectionMatrix: mat4, getElevation: (x: number, y: number) => number) {\n // We are assuming \"previousTilePoint\" won't project to a point within one unit of the camera plane\n // If it did, that would mean our label extended all the way out from within the viewport to a (very distant)\n // point near the plane of the camera. We wouldn't be able to render the label anyway once it crossed the\n // plane of the camera.\n const projectedUnitVertex = project(previousTilePoint.add(previousTilePoint.sub(currentTilePoint)._unit()), projectionMatrix, getElevation).point;\n const projectedUnitSegment = previousProjectedPoint.sub(projectedUnitVertex);\n\n return previousProjectedPoint.add(projectedUnitSegment._mult(minimumLength / projectedUnitSegment.mag()));\n}\n\ntype IndexToPointCache = { [lineIndex: number]: Point };\n\n/**\n * We calculate label-plane projected points for line vertices as we place glyphs along the line\n * Since we will use the same vertices for potentially many glyphs, cache the results for this bucket\n * over the course of the render. Each vertex location also potentially has one offset equivalent\n * for us to hold onto. The vertex indices are per-symbol-bucket.\n */\ntype ProjectionCache = {\n /**\n * tile-unit vertices projected into label-plane units\n */\n projections: IndexToPointCache;\n /**\n * label-plane vertices which have been shifted to follow an offset line\n */\n offsets: IndexToPointCache;\n};\n\n/**\n * Arguments necessary to project a vertex to the label plane\n */\ntype ProjectionArgs = {\n /**\n * Used to cache results, save cost if projecting the same vertex multiple times\n */\n projectionCache: ProjectionCache;\n /**\n * The array of tile-unit vertices transferred from worker\n */\n lineVertexArray: SymbolLineVertexArray;\n /**\n * Label plane projection matrix\n */\n labelPlaneMatrix: mat4;\n /**\n * Function to get elevation at a point\n * @param x - the x coordinate\n * @param y - the y coordinate\n */\n getElevation: (x: number, y: number) => number;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n tileAnchorPoint: Point;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n distanceFromAnchor: number;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n previousVertex: Point;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n direction: number;\n /**\n * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera\n */\n absOffsetX: number;\n};\n\n/**\n * Transform a vertex from tile coordinates to label plane coordinates\n * @param index - index of vertex to project\n * @param projectionArgs - necessary data to project a vertex\n * @returns the vertex projected to the label plane\n */\nfunction projectVertexToViewport(index: number, projectionArgs: ProjectionArgs): Point {\n const {projectionCache, lineVertexArray, labelPlaneMatrix, tileAnchorPoint, distanceFromAnchor, getElevation, previousVertex, direction, absOffsetX} = projectionArgs;\n if (projectionCache.projections[index]) {\n return projectionCache.projections[index];\n }\n const currentVertex = new Point(lineVertexArray.getx(index), lineVertexArray.gety(index));\n const projection = project(currentVertex, labelPlaneMatrix, getElevation);\n if (projection.signedDistanceFromCamera > 0) {\n projectionCache.projections[index] = projection.point;\n return projection.point;\n }\n\n // The vertex is behind the plane of the camera, so we can't project it\n // Instead, we'll create a vertex along the line that's far enough to include the glyph\n const previousLineVertexIndex = index - direction;\n const previousTilePoint = distanceFromAnchor === 0 ?\n tileAnchorPoint :\n new Point(lineVertexArray.getx(previousLineVertexIndex), lineVertexArray.gety(previousLineVertexIndex));\n // Don't cache because the new vertex might not be far enough out for future glyphs on the same segment\n return projectTruncatedLineSegment(previousTilePoint, currentVertex, previousVertex, absOffsetX - distanceFromAnchor + 1, labelPlaneMatrix, getElevation);\n}\n\n/**\n * Calculate the normal vector for a line segment\n * @param segmentVector - will be mutated as a tiny optimization\n * @param offset - magnitude of resulting vector\n * @param direction - direction of line traversal\n * @returns a normal vector from the segment, with magnitude equal to offset amount\n */\nfunction transformToOffsetNormal(segmentVector: Point, offset: number, direction: number): Point {\n return segmentVector._unit()._perp()._mult(offset * direction);\n}\n\n/**\n * Construct offset line segments for the current segment and the next segment, then extend/shrink\n * the segments until they intersect. If the segments are parallel, then they will touch with no modification.\n *\n * @param index - Index of the current vertex\n * @param prevToCurrentOffsetNormal - Normal vector of the line segment from the previous vertex to the current vertex\n * @param currentVertex - Current (non-offset) vertex projected to the label plane\n * @param lineStartIndex - Beginning index for the line this label is on\n * @param lineEndIndex - End index for the line this label is on\n * @param offsetPreviousVertex - The previous vertex projected to the label plane, and then offset along the previous segments normal\n * @param lineOffsetY - Magnitude of the offset\n * @param projectionArgs - Necessary data for tile-to-label-plane projection\n * @returns The point at which the current and next line segments intersect, once offset and extended/shrunk to their meeting point\n */\nfunction findOffsetIntersectionPoint(index: number, prevToCurrentOffsetNormal: Point, currentVertex: Point, lineStartIndex: number, lineEndIndex: number, offsetPreviousVertex: Point, lineOffsetY: number, projectionArgs: ProjectionArgs) {\n const {projectionCache, direction} = projectionArgs;\n if (projectionCache.offsets[index]) {\n return projectionCache.offsets[index];\n }\n\n const offsetCurrentVertex = currentVertex.add(prevToCurrentOffsetNormal);\n\n if (index + direction < lineStartIndex || index + direction >= lineEndIndex) {\n // This is the end of the line, no intersection to calculate\n projectionCache.offsets[index] = offsetCurrentVertex;\n return offsetCurrentVertex;\n }\n // Offset the vertices for the next segment\n const nextVertex = projectVertexToViewport(index + direction, projectionArgs);\n const currentToNextOffsetNormal = transformToOffsetNormal(nextVertex.sub(currentVertex), lineOffsetY, direction);\n const offsetNextSegmentBegin = currentVertex.add(currentToNextOffsetNormal);\n const offsetNextSegmentEnd = nextVertex.add(currentToNextOffsetNormal);\n\n // find the intersection of these two lines\n // if the lines are parallel, offsetCurrent/offsetNextBegin will touch\n projectionCache.offsets[index] = findLineIntersection(offsetPreviousVertex, offsetCurrentVertex, offsetNextSegmentBegin, offsetNextSegmentEnd) || offsetCurrentVertex;\n\n return projectionCache.offsets[index];\n}\n\n/**\n * Placed Glyph type\n */\ntype PlacedGlyph = {\n /**\n * The point at which the glyph should be placed, in label plane coordinates\n */\n point: Point;\n /**\n * The angle at which the glyph should be placed\n */\n angle: number;\n /**\n * The label-plane path used to reach this glyph: used only for collision detection\n */\n path: Array;\n};\n\n/*\n * Place a single glyph along its line, projected into the label plane, by iterating outward\n * from the anchor point until the distance traversed in the label plane equals the glyph's\n * offsetX. Returns null if the glyph can't fit on the line geometry.\n */\nfunction placeGlyphAlongLine(\n offsetX: number,\n lineOffsetX: number,\n lineOffsetY: number,\n flip: boolean,\n anchorPoint: Point,\n tileAnchorPoint: Point,\n anchorSegment: number,\n lineStartIndex: number,\n lineEndIndex: number,\n lineVertexArray: SymbolLineVertexArray,\n labelPlaneMatrix: mat4,\n projectionCache: ProjectionCache,\n rotateToLine: boolean,\n getElevation: (x: number, y: number) => number): PlacedGlyph | null {\n\n const combinedOffsetX = flip ?\n offsetX - lineOffsetX :\n offsetX + lineOffsetX;\n\n let direction = combinedOffsetX > 0 ? 1 : -1;\n\n let angle = 0;\n if (flip) {\n // The label needs to be flipped to keep text upright.\n // Iterate in the reverse direction.\n direction *= -1;\n angle = Math.PI;\n }\n\n if (direction < 0) angle += Math.PI;\n\n let currentIndex = direction > 0 ?\n lineStartIndex + anchorSegment :\n lineStartIndex + anchorSegment + 1;\n\n let currentVertex = anchorPoint;\n let previousVertex = anchorPoint;\n\n // offsetPrev and intersectionPoint are analogous to previousVertex and currentVertex\n // but if there's a line offset they are calculated in parallel as projection happens\n let offsetIntersectionPoint: Point;\n let offsetPreviousVertex: Point;\n\n let distanceFromAnchor = 0;\n let currentSegmentDistance = 0;\n const absOffsetX = Math.abs(combinedOffsetX);\n const pathVertices: Array = [];\n\n let currentLineSegment: Point;\n while (distanceFromAnchor + currentSegmentDistance <= absOffsetX) {\n currentIndex += direction;\n\n // offset does not fit on the projected line\n if (currentIndex < lineStartIndex || currentIndex >= lineEndIndex)\n return null;\n\n // accumulate values from last iteration\n distanceFromAnchor += currentSegmentDistance;\n previousVertex = currentVertex;\n offsetPreviousVertex = offsetIntersectionPoint;\n\n const projectionArgs: ProjectionArgs = {\n projectionCache,\n lineVertexArray,\n labelPlaneMatrix,\n tileAnchorPoint,\n distanceFromAnchor,\n getElevation,\n previousVertex,\n direction,\n absOffsetX\n };\n\n // find next vertex in viewport space\n currentVertex = projectVertexToViewport(currentIndex, projectionArgs);\n if (lineOffsetY === 0) {\n // Store vertices for collision detection and update current segment geometry\n pathVertices.push(previousVertex);\n currentLineSegment = currentVertex.sub(previousVertex);\n } else {\n // Calculate the offset for this section\n let prevToCurrentOffsetNormal;\n const prevToCurrent = currentVertex.sub(previousVertex);\n if (prevToCurrent.mag() === 0) {\n // We are starting with our anchor point directly on the vertex, so look one vertex ahead\n // to calculate a normal\n const nextVertex = projectVertexToViewport(currentIndex + direction, projectionArgs);\n prevToCurrentOffsetNormal = transformToOffsetNormal(nextVertex.sub(currentVertex), lineOffsetY, direction);\n } else {\n prevToCurrentOffsetNormal = transformToOffsetNormal(prevToCurrent, lineOffsetY, direction);\n }\n // Initialize offsetPrev on our first iteration, after that it will be pre-calculated\n if (!offsetPreviousVertex)\n offsetPreviousVertex = previousVertex.add(prevToCurrentOffsetNormal);\n\n offsetIntersectionPoint = findOffsetIntersectionPoint(currentIndex, prevToCurrentOffsetNormal, currentVertex, lineStartIndex, lineEndIndex, offsetPreviousVertex, lineOffsetY, projectionArgs);\n\n pathVertices.push(offsetPreviousVertex);\n currentLineSegment = offsetIntersectionPoint.sub(offsetPreviousVertex);\n }\n currentSegmentDistance = currentLineSegment.mag();\n }\n\n // The point is on the current segment. Interpolate to find it.\n const segmentInterpolationT = (absOffsetX - distanceFromAnchor) / currentSegmentDistance;\n const p = currentLineSegment._mult(segmentInterpolationT)._add(offsetPreviousVertex || previousVertex);\n\n const segmentAngle = angle + Math.atan2(currentVertex.y - previousVertex.y, currentVertex.x - previousVertex.x);\n\n pathVertices.push(p);\n\n return {\n point: p,\n angle: rotateToLine ? segmentAngle : 0.0,\n path: pathVertices\n };\n}\n\nconst hiddenGlyphAttributes = new Float32Array([-Infinity, -Infinity, 0, -Infinity, -Infinity, 0, -Infinity, -Infinity, 0, -Infinity, -Infinity, 0]);\n\n// Hide them by moving them offscreen. We still need to add them to the buffer\n// because the dynamic buffer is paired with a static buffer that doesn't get updated.\nfunction hideGlyphs(num: number, dynamicLayoutVertexArray: SymbolDynamicLayoutArray) {\n for (let i = 0; i < num; i++) {\n const offset = dynamicLayoutVertexArray.length;\n dynamicLayoutVertexArray.resize(offset + 4);\n // Since all hidden glyphs have the same attributes, we can build up the array faster with a single call to Float32Array.set\n // for each set of four vertices, instead of calling addDynamicAttributes for each vertex.\n dynamicLayoutVertexArray.float32.set(hiddenGlyphAttributes, offset * 3);\n }\n}\n\n// For line label layout, we're not using z output and our w input is always 1\n// This custom matrix transformation ignores those components to make projection faster\nfunction xyTransformMat4(out: vec4, a: vec4, m: mat4) {\n const x = a[0], y = a[1];\n out[0] = m[0] * x + m[4] * y + m[12];\n out[1] = m[1] * x + m[5] * y + m[13];\n out[3] = m[3] * x + m[7] * y + m[15];\n return out;\n}\n","import Point from '@mapbox/point-geometry';\nimport {clipLine} from './clip_line';\nimport {PathInterpolator} from './path_interpolator';\n\nimport * as intersectionTests from '../util/intersection_tests';\nimport {GridIndex} from './grid_index';\nimport {mat4, vec4} from 'gl-matrix';\nimport ONE_EM from '../symbol/one_em';\n\nimport * as projection from '../symbol/projection';\n\nimport type {Transform} from '../geo/transform';\nimport type {SingleCollisionBox} from '../data/bucket/symbol_bucket';\nimport type {\n GlyphOffsetArray,\n SymbolLineVertexArray\n} from '../data/array_types.g';\nimport type {OverlapMode} from '../style/style_layer/overlap_mode';\n\n// When a symbol crosses the edge that causes it to be included in\n// collision detection, it will cause changes in the symbols around\n// it. This constant specifies how many pixels to pad the edge of\n// the viewport for collision detection so that the bulk of the changes\n// occur offscreen. Making this constant greater increases label\n// stability, but it's expensive.\nconst viewportPadding = 100;\n\nexport type FeatureKey = {\n bucketInstanceId: number;\n featureIndex: number;\n collisionGroupID: number;\n overlapMode: OverlapMode;\n};\n\n/**\n * @internal\n * A collision index used to prevent symbols from overlapping. It keep tracks of\n * where previous symbols have been placed and is used to check if a new\n * symbol overlaps with any previously added symbols.\n *\n * There are two steps to insertion: first placeCollisionBox/Circles checks if\n * there's room for a symbol, then insertCollisionBox/Circles actually puts the\n * symbol in the index. The two step process allows paired symbols to be inserted\n * together even if they overlap.\n */\nexport class CollisionIndex {\n grid: GridIndex;\n ignoredGrid: GridIndex;\n transform: Transform;\n pitchfactor: number;\n screenRightBoundary: number;\n screenBottomBoundary: number;\n gridRightBoundary: number;\n gridBottomBoundary: number;\n\n // With perspectiveRatio the fontsize is calculated for tilted maps (near = bigger, far = smaller).\n // The cutoff defines a threshold to no longer render labels near the horizon.\n perspectiveRatioCutoff: number;\n\n constructor(\n transform: Transform,\n grid = new GridIndex(transform.width + 2 * viewportPadding, transform.height + 2 * viewportPadding, 25),\n ignoredGrid = new GridIndex(transform.width + 2 * viewportPadding, transform.height + 2 * viewportPadding, 25)\n ) {\n this.transform = transform;\n\n this.grid = grid;\n this.ignoredGrid = ignoredGrid;\n this.pitchfactor = Math.cos(transform._pitch) * transform.cameraToCenterDistance;\n\n this.screenRightBoundary = transform.width + viewportPadding;\n this.screenBottomBoundary = transform.height + viewportPadding;\n this.gridRightBoundary = transform.width + 2 * viewportPadding;\n this.gridBottomBoundary = transform.height + 2 * viewportPadding;\n\n this.perspectiveRatioCutoff = 0.6;\n }\n\n placeCollisionBox(\n collisionBox: SingleCollisionBox,\n overlapMode: OverlapMode,\n textPixelRatio: number,\n posMatrix: mat4,\n collisionGroupPredicate?: (key: FeatureKey) => boolean,\n getElevation?: (x: number, y: number) => number\n ): {\n box: Array;\n offscreen: boolean;\n } {\n const projectedPoint = this.projectAndGetPerspectiveRatio(posMatrix, collisionBox.anchorPointX, collisionBox.anchorPointY, getElevation);\n const tileToViewport = textPixelRatio * projectedPoint.perspectiveRatio;\n const tlX = collisionBox.x1 * tileToViewport + projectedPoint.point.x;\n const tlY = collisionBox.y1 * tileToViewport + projectedPoint.point.y;\n const brX = collisionBox.x2 * tileToViewport + projectedPoint.point.x;\n const brY = collisionBox.y2 * tileToViewport + projectedPoint.point.y;\n\n if (!this.isInsideGrid(tlX, tlY, brX, brY) ||\n (overlapMode !== 'always' && this.grid.hitTest(tlX, tlY, brX, brY, overlapMode, collisionGroupPredicate)) ||\n projectedPoint.perspectiveRatio < this.perspectiveRatioCutoff) {\n return {\n box: [],\n offscreen: false\n };\n }\n\n return {\n box: [tlX, tlY, brX, brY],\n offscreen: this.isOffscreen(tlX, tlY, brX, brY)\n };\n }\n\n placeCollisionCircles(\n overlapMode: OverlapMode,\n symbol: any,\n lineVertexArray: SymbolLineVertexArray,\n glyphOffsetArray: GlyphOffsetArray,\n fontSize: number,\n posMatrix: mat4,\n labelPlaneMatrix: mat4,\n labelToScreenMatrix: mat4,\n showCollisionCircles: boolean,\n pitchWithMap: boolean,\n collisionGroupPredicate: (key: FeatureKey) => boolean,\n circlePixelDiameter: number,\n textPixelPadding: number,\n getElevation: (x: number, y: number) => number\n ): {\n circles: Array;\n offscreen: boolean;\n collisionDetected: boolean;\n } {\n const placedCollisionCircles = [];\n\n const tileUnitAnchorPoint = new Point(symbol.anchorX, symbol.anchorY);\n const screenAnchorPoint = projection.project(tileUnitAnchorPoint, posMatrix, getElevation);\n const perspectiveRatio = projection.getPerspectiveRatio(this.transform.cameraToCenterDistance, screenAnchorPoint.signedDistanceFromCamera);\n const labelPlaneFontSize = pitchWithMap ? fontSize / perspectiveRatio : fontSize * perspectiveRatio;\n const labelPlaneFontScale = labelPlaneFontSize / ONE_EM;\n\n const labelPlaneAnchorPoint = projection.project(tileUnitAnchorPoint, labelPlaneMatrix, getElevation).point;\n\n const projectionCache = {projections: {}, offsets: {}};\n const lineOffsetX = symbol.lineOffsetX * labelPlaneFontScale;\n const lineOffsetY = symbol.lineOffsetY * labelPlaneFontScale;\n\n const firstAndLastGlyph = projection.placeFirstAndLastGlyph(\n labelPlaneFontScale,\n glyphOffsetArray,\n lineOffsetX,\n lineOffsetY,\n /*flip*/ false,\n labelPlaneAnchorPoint,\n tileUnitAnchorPoint,\n symbol,\n lineVertexArray,\n labelPlaneMatrix,\n projectionCache,\n false,\n getElevation);\n\n let collisionDetected = false;\n let inGrid = false;\n let entirelyOffscreen = true;\n\n if (firstAndLastGlyph) {\n const radius = circlePixelDiameter * 0.5 * perspectiveRatio + textPixelPadding;\n const screenPlaneMin = new Point(-viewportPadding, -viewportPadding);\n const screenPlaneMax = new Point(this.screenRightBoundary, this.screenBottomBoundary);\n const interpolator = new PathInterpolator();\n\n // Construct a projected path from projected line vertices. Anchor points are ignored and removed\n const first = firstAndLastGlyph.first;\n const last = firstAndLastGlyph.last;\n\n let projectedPath = [];\n for (let i = first.path.length - 1; i >= 1; i--) {\n projectedPath.push(first.path[i]);\n }\n for (let i = 1; i < last.path.length; i++) {\n projectedPath.push(last.path[i]);\n }\n\n // Tolerate a slightly longer distance than one diameter between two adjacent circles\n const circleDist = radius * 2.5;\n\n // The path might need to be converted into screen space if a pitched map is used as the label space\n if (labelToScreenMatrix) {\n const screenSpacePath = projectedPath.map(p => projection.project(p, labelToScreenMatrix, getElevation));\n\n // Do not try to place collision circles if even of the points is behind the camera.\n // This is a plausible scenario with big camera pitch angles\n if (screenSpacePath.some(point => point.signedDistanceFromCamera <= 0)) {\n projectedPath = [];\n } else {\n projectedPath = screenSpacePath.map(p => p.point);\n }\n }\n\n let segments = [];\n\n if (projectedPath.length > 0) {\n // Quickly check if the path is fully inside or outside of the padded collision region.\n // For overlapping paths we'll only create collision circles for the visible segments\n const minPoint = projectedPath[0].clone();\n const maxPoint = projectedPath[0].clone();\n\n for (let i = 1; i < projectedPath.length; i++) {\n minPoint.x = Math.min(minPoint.x, projectedPath[i].x);\n minPoint.y = Math.min(minPoint.y, projectedPath[i].y);\n maxPoint.x = Math.max(maxPoint.x, projectedPath[i].x);\n maxPoint.y = Math.max(maxPoint.y, projectedPath[i].y);\n }\n\n if (minPoint.x >= screenPlaneMin.x && maxPoint.x <= screenPlaneMax.x &&\n minPoint.y >= screenPlaneMin.y && maxPoint.y <= screenPlaneMax.y) {\n // Quad fully visible\n segments = [projectedPath];\n } else if (maxPoint.x < screenPlaneMin.x || minPoint.x > screenPlaneMax.x ||\n maxPoint.y < screenPlaneMin.y || minPoint.y > screenPlaneMax.y) {\n // Not visible\n segments = [];\n } else {\n segments = clipLine([projectedPath], screenPlaneMin.x, screenPlaneMin.y, screenPlaneMax.x, screenPlaneMax.y);\n }\n }\n\n for (const seg of segments) {\n // interpolate positions for collision circles. Add a small padding to both ends of the segment\n interpolator.reset(seg, radius * 0.25);\n\n let numCircles = 0;\n\n if (interpolator.length <= 0.5 * radius) {\n numCircles = 1;\n } else {\n numCircles = Math.ceil(interpolator.paddedLength / circleDist) + 1;\n }\n\n for (let i = 0; i < numCircles; i++) {\n const t = i / Math.max(numCircles - 1, 1);\n const circlePosition = interpolator.lerp(t);\n\n // add viewport padding to the position and perform initial collision check\n const centerX = circlePosition.x + viewportPadding;\n const centerY = circlePosition.y + viewportPadding;\n\n placedCollisionCircles.push(centerX, centerY, radius, 0);\n\n const x1 = centerX - radius;\n const y1 = centerY - radius;\n const x2 = centerX + radius;\n const y2 = centerY + radius;\n\n entirelyOffscreen = entirelyOffscreen && this.isOffscreen(x1, y1, x2, y2);\n inGrid = inGrid || this.isInsideGrid(x1, y1, x2, y2);\n\n if (overlapMode !== 'always' && this.grid.hitTestCircle(centerX, centerY, radius, overlapMode, collisionGroupPredicate)) {\n // Don't early exit if we're showing the debug circles because we still want to calculate\n // which circles are in use\n collisionDetected = true;\n if (!showCollisionCircles) {\n return {\n circles: [],\n offscreen: false,\n collisionDetected\n };\n }\n }\n }\n }\n }\n\n return {\n circles: ((!showCollisionCircles && collisionDetected) || !inGrid || perspectiveRatio < this.perspectiveRatioCutoff) ? [] : placedCollisionCircles,\n offscreen: entirelyOffscreen,\n collisionDetected\n };\n }\n\n /**\n * Because the geometries in the CollisionIndex are an approximation of the shape of\n * symbols on the map, we use the CollisionIndex to look up the symbol part of\n * `queryRenderedFeatures`.\n */\n queryRenderedSymbols(viewportQueryGeometry: Array) {\n if (viewportQueryGeometry.length === 0 || (this.grid.keysLength() === 0 && this.ignoredGrid.keysLength() === 0)) {\n return {};\n }\n\n const query = [];\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n for (const point of viewportQueryGeometry) {\n const gridPoint = new Point(point.x + viewportPadding, point.y + viewportPadding);\n minX = Math.min(minX, gridPoint.x);\n minY = Math.min(minY, gridPoint.y);\n maxX = Math.max(maxX, gridPoint.x);\n maxY = Math.max(maxY, gridPoint.y);\n query.push(gridPoint);\n }\n\n const features = this.grid.query(minX, minY, maxX, maxY)\n .concat(this.ignoredGrid.query(minX, minY, maxX, maxY));\n\n const seenFeatures = {};\n const result = {};\n\n for (const feature of features) {\n const featureKey = feature.key;\n // Skip already seen features.\n if (seenFeatures[featureKey.bucketInstanceId] === undefined) {\n seenFeatures[featureKey.bucketInstanceId] = {};\n }\n if (seenFeatures[featureKey.bucketInstanceId][featureKey.featureIndex]) {\n continue;\n }\n\n // Check if query intersects with the feature box\n // \"Collision Circles\" for line labels are treated as boxes here\n // Since there's no actual collision taking place, the circle vs. square\n // distinction doesn't matter as much, and box geometry is easier\n // to work with.\n const bbox = [\n new Point(feature.x1, feature.y1),\n new Point(feature.x2, feature.y1),\n new Point(feature.x2, feature.y2),\n new Point(feature.x1, feature.y2)\n ];\n if (!intersectionTests.polygonIntersectsPolygon(query, bbox)) {\n continue;\n }\n\n seenFeatures[featureKey.bucketInstanceId][featureKey.featureIndex] = true;\n if (result[featureKey.bucketInstanceId] === undefined) {\n result[featureKey.bucketInstanceId] = [];\n }\n result[featureKey.bucketInstanceId].push(featureKey.featureIndex);\n }\n\n return result;\n }\n\n insertCollisionBox(collisionBox: Array, overlapMode: OverlapMode, ignorePlacement: boolean, bucketInstanceId: number, featureIndex: number, collisionGroupID: number) {\n const grid = ignorePlacement ? this.ignoredGrid : this.grid;\n\n const key = {bucketInstanceId, featureIndex, collisionGroupID, overlapMode};\n grid.insert(key, collisionBox[0], collisionBox[1], collisionBox[2], collisionBox[3]);\n }\n\n insertCollisionCircles(collisionCircles: Array, overlapMode: OverlapMode, ignorePlacement: boolean, bucketInstanceId: number, featureIndex: number, collisionGroupID: number) {\n const grid = ignorePlacement ? this.ignoredGrid : this.grid;\n\n const key = {bucketInstanceId, featureIndex, collisionGroupID, overlapMode};\n for (let k = 0; k < collisionCircles.length; k += 4) {\n grid.insertCircle(key, collisionCircles[k], collisionCircles[k + 1], collisionCircles[k + 2]);\n }\n }\n\n projectAndGetPerspectiveRatio(posMatrix: mat4, x: number, y: number, getElevation?: (x: number, y: number) => number) {\n let p;\n if (getElevation) { // slow because of handle z-index\n p = [x, y, getElevation(x, y), 1] as vec4;\n vec4.transformMat4(p, p, posMatrix);\n } else { // fast because of ignore z-index\n p = [x, y, 0, 1] as vec4;\n projection.xyTransformMat4(p, p, posMatrix);\n }\n const a = new Point(\n (((p[0] / p[3] + 1) / 2) * this.transform.width) + viewportPadding,\n (((-p[1] / p[3] + 1) / 2) * this.transform.height) + viewportPadding\n );\n return {\n point: a,\n // See perspective ratio comment in symbol_sdf.vertex\n // We're doing collision detection in viewport space so we need\n // to scale down boxes in the distance\n perspectiveRatio: 0.5 + 0.5 * (this.transform.cameraToCenterDistance / p[3])\n };\n }\n\n isOffscreen(x1: number, y1: number, x2: number, y2: number) {\n return x2 < viewportPadding || x1 >= this.screenRightBoundary || y2 < viewportPadding || y1 > this.screenBottomBoundary;\n }\n\n isInsideGrid(x1: number, y1: number, x2: number, y2: number) {\n return x2 >= 0 && x1 < this.gridRightBoundary && y2 >= 0 && y1 < this.gridBottomBoundary;\n }\n\n /*\n * Returns a matrix for transforming collision shapes to viewport coordinate space.\n * Use this function to render e.g. collision circles on the screen.\n * example transformation: clipPos = glCoordMatrix * viewportMatrix * circle_pos\n */\n getViewportMatrix() {\n const m = mat4.identity([] as any);\n mat4.translate(m, m, [-viewportPadding, -viewportPadding, 0.0]);\n return m;\n }\n}\n","import {EXTENT} from '../data/extent';\n\nimport type {OverscaledTileID} from './tile_id';\n\n/**\n * Converts a pixel value at a the given zoom level to tile units.\n *\n * The shaders mostly calculate everything in tile units so style\n * properties need to be converted from pixels to tile units using this.\n *\n * For example, a translation by 30 pixels at zoom 6.5 will be a\n * translation by pixelsToTileUnits(30, 6.5) tile units.\n *\n * @returns value in tile units\n */\nexport function pixelsToTileUnits(\n tile: {\n tileID: OverscaledTileID;\n tileSize: number;\n },\n pixelValue: number,\n z: number\n): number {\n return pixelValue * (EXTENT / (tile.tileSize * Math.pow(2, z - tile.tileID.overscaledZ)));\n}\n","import {CollisionIndex} from './collision_index';\nimport type {FeatureKey} from './collision_index';\nimport {EXTENT} from '../data/extent';\nimport * as symbolSize from './symbol_size';\nimport * as projection from './projection';\nimport {getAnchorJustification} from './symbol_layout';\nimport {getAnchorAlignment, WritingMode} from './shaping';\nimport {mat4} from 'gl-matrix';\nimport {pixelsToTileUnits} from '../source/pixels_to_tile_units';\nimport Point from '@mapbox/point-geometry';\nimport type {Transform} from '../geo/transform';\nimport type {StyleLayer} from '../style/style_layer';\nimport {PossiblyEvaluated} from '../style/properties';\nimport type {SymbolLayoutProps, SymbolLayoutPropsPossiblyEvaluated} from '../style/style_layer/symbol_style_layer_properties.g';\nimport {getOverlapMode, OverlapMode} from '../style/style_layer/overlap_mode';\n\nimport type {Tile} from '../source/tile';\nimport {SymbolBucket, CollisionArrays, SingleCollisionBox} from '../data/bucket/symbol_bucket';\n\nimport type {CollisionBoxArray, CollisionVertexArray, SymbolInstance, TextAnchorOffset} from '../data/array_types.g';\nimport type {FeatureIndex} from '../data/feature_index';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport {Terrain} from '../render/terrain';\nimport {warnOnce} from '../util/util';\nimport {TextAnchor, TextAnchorEnum} from '../style/style_layer/variable_text_anchor';\n\nclass OpacityState {\n opacity: number;\n placed: boolean;\n constructor(prevState: OpacityState, increment: number, placed: boolean, skipFade?: boolean | null) {\n if (prevState) {\n this.opacity = Math.max(0, Math.min(1, prevState.opacity + (prevState.placed ? increment : -increment)));\n } else {\n this.opacity = (skipFade && placed) ? 1 : 0;\n }\n this.placed = placed;\n }\n isHidden() {\n return this.opacity === 0 && !this.placed;\n }\n}\n\nclass JointOpacityState {\n text: OpacityState;\n icon: OpacityState;\n constructor(prevState: JointOpacityState, increment: number, placedText: boolean, placedIcon: boolean, skipFade?: boolean | null) {\n this.text = new OpacityState(prevState ? prevState.text : null, increment, placedText, skipFade);\n this.icon = new OpacityState(prevState ? prevState.icon : null, increment, placedIcon, skipFade);\n }\n isHidden() {\n return this.text.isHidden() && this.icon.isHidden();\n }\n}\n\nclass JointPlacement {\n text: boolean;\n icon: boolean;\n // skipFade = outside viewport, but within CollisionIndex::viewportPadding px of the edge\n // Because these symbols aren't onscreen yet, we can skip the \"fade in\" animation,\n // and if a subsequent viewport change brings them into view, they'll be fully\n // visible right away.\n skipFade: boolean;\n constructor(text: boolean, icon: boolean, skipFade: boolean) {\n this.text = text;\n this.icon = icon;\n this.skipFade = skipFade;\n }\n}\n\nclass CollisionCircleArray {\n // Stores collision circles and placement matrices of a bucket for debug rendering.\n invProjMatrix: mat4;\n viewportMatrix: mat4;\n circles: Array;\n\n constructor() {\n this.invProjMatrix = mat4.create();\n this.viewportMatrix = mat4.create();\n this.circles = [];\n }\n}\n\nexport class RetainedQueryData {\n bucketInstanceId: number;\n featureIndex: FeatureIndex;\n sourceLayerIndex: number;\n bucketIndex: number;\n tileID: OverscaledTileID;\n featureSortOrder: Array;\n constructor(bucketInstanceId: number,\n featureIndex: FeatureIndex,\n sourceLayerIndex: number,\n bucketIndex: number,\n tileID: OverscaledTileID) {\n this.bucketInstanceId = bucketInstanceId;\n this.featureIndex = featureIndex;\n this.sourceLayerIndex = sourceLayerIndex;\n this.bucketIndex = bucketIndex;\n this.tileID = tileID;\n }\n}\n\ntype CollisionGroup = {\n ID: number;\n predicate?: (key: FeatureKey) => boolean;\n};\n\nclass CollisionGroups {\n collisionGroups: {[groupName: string]: CollisionGroup};\n maxGroupID: number;\n crossSourceCollisions: boolean;\n\n constructor(crossSourceCollisions: boolean) {\n this.crossSourceCollisions = crossSourceCollisions;\n this.maxGroupID = 0;\n this.collisionGroups = {};\n }\n\n get(sourceID: string) {\n // The predicate/groupID mechanism allows for arbitrary grouping,\n // but the current interface defines one source == one group when\n // crossSourceCollisions == true.\n if (!this.crossSourceCollisions) {\n if (!this.collisionGroups[sourceID]) {\n const nextGroupID = ++this.maxGroupID;\n this.collisionGroups[sourceID] = {\n ID: nextGroupID,\n predicate: (key) => {\n return key.collisionGroupID === nextGroupID;\n }\n };\n }\n return this.collisionGroups[sourceID];\n } else {\n return {ID: 0, predicate: null};\n }\n }\n}\n\nfunction calculateVariableLayoutShift(\n anchor: TextAnchor,\n width: number,\n height: number,\n textOffset: [number, number],\n textBoxScale: number\n): Point {\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(anchor);\n const shiftX = -(horizontalAlign - 0.5) * width;\n const shiftY = -(verticalAlign - 0.5) * height;\n return new Point(\n shiftX + textOffset[0] * textBoxScale,\n shiftY + textOffset[1] * textBoxScale\n );\n}\n\nfunction shiftVariableCollisionBox(collisionBox: SingleCollisionBox,\n shiftX: number, shiftY: number,\n rotateWithMap: boolean, pitchWithMap: boolean,\n angle: number) {\n const {x1, x2, y1, y2, anchorPointX, anchorPointY} = collisionBox;\n const rotatedOffset = new Point(shiftX, shiftY);\n if (rotateWithMap) {\n rotatedOffset._rotate(pitchWithMap ? angle : -angle);\n }\n return {\n x1: x1 + rotatedOffset.x,\n y1: y1 + rotatedOffset.y,\n x2: x2 + rotatedOffset.x,\n y2: y2 + rotatedOffset.y,\n // symbol anchor point stays the same regardless of text-anchor\n anchorPointX,\n anchorPointY\n };\n}\n\nexport type VariableOffset = {\n textOffset: [number, number];\n width: number;\n height: number;\n anchor: TextAnchor;\n textBoxScale: number;\n prevAnchor?: TextAnchor;\n};\n\ntype TileLayerParameters = {\n bucket: SymbolBucket;\n layout: PossiblyEvaluated;\n posMatrix: mat4;\n textLabelPlaneMatrix: mat4;\n labelToScreenMatrix: mat4;\n scale: number;\n textPixelRatio: number;\n holdingForFade: boolean;\n collisionBoxArray: CollisionBoxArray;\n partiallyEvaluatedTextSize: {\n uSize: number;\n uSizeT: number;\n };\n collisionGroup: CollisionGroup;\n};\n\nexport type BucketPart = {\n sortKey?: number | void;\n symbolInstanceStart: number;\n symbolInstanceEnd: number;\n parameters: TileLayerParameters;\n};\n\nexport type CrossTileID = string | number;\n\nexport class Placement {\n transform: Transform;\n terrain: Terrain;\n collisionIndex: CollisionIndex;\n placements: {\n [_ in CrossTileID]: JointPlacement;\n };\n opacities: {\n [_ in CrossTileID]: JointOpacityState;\n };\n variableOffsets: {\n [_ in CrossTileID]: VariableOffset;\n };\n placedOrientations: {\n [_ in CrossTileID]: number;\n };\n commitTime: number;\n prevZoomAdjustment: number;\n lastPlacementChangeTime: number;\n stale: boolean;\n fadeDuration: number;\n retainedQueryData: {\n [_: number]: RetainedQueryData;\n };\n collisionGroups: CollisionGroups;\n prevPlacement: Placement;\n zoomAtLastRecencyCheck: number;\n collisionCircleArrays: {\n [k in any]: CollisionCircleArray;\n };\n\n constructor(transform: Transform, terrain: Terrain, fadeDuration: number, crossSourceCollisions: boolean, prevPlacement?: Placement) {\n this.transform = transform.clone();\n this.terrain = terrain;\n this.collisionIndex = new CollisionIndex(this.transform);\n this.placements = {};\n this.opacities = {};\n this.variableOffsets = {};\n this.stale = false;\n this.commitTime = 0;\n this.fadeDuration = fadeDuration;\n this.retainedQueryData = {};\n this.collisionGroups = new CollisionGroups(crossSourceCollisions);\n this.collisionCircleArrays = {};\n\n this.prevPlacement = prevPlacement;\n if (prevPlacement) {\n prevPlacement.prevPlacement = undefined; // Only hold on to one placement back\n }\n\n this.placedOrientations = {};\n }\n\n getBucketParts(results: Array, styleLayer: StyleLayer, tile: Tile, sortAcrossTiles: boolean) {\n const symbolBucket = (tile.getBucket(styleLayer) as SymbolBucket);\n const bucketFeatureIndex = tile.latestFeatureIndex;\n if (!symbolBucket || !bucketFeatureIndex || styleLayer.id !== symbolBucket.layerIds[0])\n return;\n\n const collisionBoxArray = tile.collisionBoxArray;\n\n const layout = symbolBucket.layers[0].layout;\n\n const scale = Math.pow(2, this.transform.zoom - tile.tileID.overscaledZ);\n const textPixelRatio = tile.tileSize / EXTENT;\n\n const posMatrix = this.transform.calculatePosMatrix(tile.tileID.toUnwrapped());\n\n const pitchWithMap = layout.get('text-pitch-alignment') === 'map';\n const rotateWithMap = layout.get('text-rotation-alignment') === 'map';\n const pixelsToTiles = pixelsToTileUnits(tile, 1, this.transform.zoom);\n\n const textLabelPlaneMatrix = projection.getLabelPlaneMatrix(posMatrix,\n pitchWithMap,\n rotateWithMap,\n this.transform,\n pixelsToTiles);\n\n let labelToScreenMatrix = null;\n\n if (pitchWithMap) {\n const glMatrix = projection.getGlCoordMatrix(\n posMatrix,\n pitchWithMap,\n rotateWithMap,\n this.transform,\n pixelsToTiles);\n\n labelToScreenMatrix = mat4.multiply([] as any, this.transform.labelPlaneMatrix, glMatrix);\n }\n\n // As long as this placement lives, we have to hold onto this bucket's\n // matching FeatureIndex/data for querying purposes\n this.retainedQueryData[symbolBucket.bucketInstanceId] = new RetainedQueryData(\n symbolBucket.bucketInstanceId,\n bucketFeatureIndex,\n symbolBucket.sourceLayerIndex,\n symbolBucket.index,\n tile.tileID\n );\n\n const parameters = {\n bucket: symbolBucket,\n layout,\n posMatrix,\n textLabelPlaneMatrix,\n labelToScreenMatrix,\n scale,\n textPixelRatio,\n holdingForFade: tile.holdingForFade(),\n collisionBoxArray,\n partiallyEvaluatedTextSize: symbolSize.evaluateSizeForZoom(symbolBucket.textSizeData, this.transform.zoom),\n collisionGroup: this.collisionGroups.get(symbolBucket.sourceID)\n };\n\n if (sortAcrossTiles) {\n for (const range of symbolBucket.sortKeyRanges) {\n const {sortKey, symbolInstanceStart, symbolInstanceEnd} = range;\n results.push({sortKey, symbolInstanceStart, symbolInstanceEnd, parameters});\n }\n } else {\n results.push({\n symbolInstanceStart: 0,\n symbolInstanceEnd: symbolBucket.symbolInstances.length,\n parameters\n });\n }\n }\n\n attemptAnchorPlacement(\n textAnchorOffset: TextAnchorOffset,\n textBox: SingleCollisionBox,\n width: number,\n height: number,\n textBoxScale: number,\n rotateWithMap: boolean,\n pitchWithMap: boolean,\n textPixelRatio: number,\n posMatrix: mat4,\n collisionGroup: CollisionGroup,\n textOverlapMode: OverlapMode,\n symbolInstance: SymbolInstance,\n bucket: SymbolBucket,\n orientation: number,\n iconBox?: SingleCollisionBox | null,\n getElevation?: (x: number, y: number) => number\n ): {\n shift: Point;\n placedGlyphBoxes: {\n box: Array;\n offscreen: boolean;\n };\n } {\n\n const anchor = TextAnchorEnum[textAnchorOffset.textAnchor] as TextAnchor;\n const textOffset = [textAnchorOffset.textOffset0, textAnchorOffset.textOffset1] as [number, number];\n const shift = calculateVariableLayoutShift(anchor, width, height, textOffset, textBoxScale);\n\n const placedGlyphBoxes = this.collisionIndex.placeCollisionBox(\n shiftVariableCollisionBox(\n textBox, shift.x, shift.y,\n rotateWithMap, pitchWithMap, this.transform.angle),\n textOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation);\n\n if (iconBox) {\n const placedIconBoxes = this.collisionIndex.placeCollisionBox(\n shiftVariableCollisionBox(\n iconBox, shift.x, shift.y,\n rotateWithMap, pitchWithMap, this.transform.angle),\n textOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation);\n if (placedIconBoxes.box.length === 0) return;\n }\n\n if (placedGlyphBoxes.box.length > 0) {\n let prevAnchor;\n // If this label was placed in the previous placement, record the anchor position\n // to allow us to animate the transition\n if (this.prevPlacement &&\n this.prevPlacement.variableOffsets[symbolInstance.crossTileID] &&\n this.prevPlacement.placements[symbolInstance.crossTileID] &&\n this.prevPlacement.placements[symbolInstance.crossTileID].text) {\n prevAnchor = this.prevPlacement.variableOffsets[symbolInstance.crossTileID].anchor;\n }\n if (symbolInstance.crossTileID === 0) throw new Error('symbolInstance.crossTileID can\\'t be 0');\n this.variableOffsets[symbolInstance.crossTileID] = {\n textOffset,\n width,\n height,\n anchor,\n textBoxScale,\n prevAnchor\n };\n this.markUsedJustification(bucket, anchor, symbolInstance, orientation);\n\n if (bucket.allowVerticalPlacement) {\n this.markUsedOrientation(bucket, orientation, symbolInstance);\n this.placedOrientations[symbolInstance.crossTileID] = orientation;\n }\n\n return {shift, placedGlyphBoxes};\n }\n }\n\n placeLayerBucketPart(bucketPart: BucketPart, seenCrossTileIDs: {\n [k in string | number]: boolean;\n }, showCollisionBoxes: boolean) {\n\n const {\n bucket,\n layout,\n posMatrix,\n textLabelPlaneMatrix,\n labelToScreenMatrix,\n textPixelRatio,\n holdingForFade,\n collisionBoxArray,\n partiallyEvaluatedTextSize,\n collisionGroup\n } = bucketPart.parameters;\n\n const textOptional = layout.get('text-optional');\n const iconOptional = layout.get('icon-optional');\n const textOverlapMode = getOverlapMode(layout, 'text-overlap', 'text-allow-overlap');\n const textAlwaysOverlap = textOverlapMode === 'always';\n const iconOverlapMode = getOverlapMode(layout, 'icon-overlap', 'icon-allow-overlap');\n const iconAlwaysOverlap = iconOverlapMode === 'always';\n const rotateWithMap = layout.get('text-rotation-alignment') === 'map';\n const pitchWithMap = layout.get('text-pitch-alignment') === 'map';\n const hasIconTextFit = layout.get('icon-text-fit') !== 'none';\n const zOrderByViewportY = layout.get('symbol-z-order') === 'viewport-y';\n\n // This logic is similar to the \"defaultOpacityState\" logic below in updateBucketOpacities\n // If we know a symbol is always supposed to show, force it to be marked visible even if\n // it wasn't placed into the collision index (because some or all of it was outside the range\n // of the collision grid).\n // There is a subtle edge case here we're accepting:\n // Symbol A has text-allow-overlap: true, icon-allow-overlap: true, icon-optional: false\n // A's icon is outside the grid, so doesn't get placed\n // A's text would be inside grid, but doesn't get placed because of icon-optional: false\n // We still show A because of the allow-overlap settings.\n // Symbol B has allow-overlap: false, and gets placed where A's text would be\n // On panning in, there is a short period when Symbol B and Symbol A will overlap\n // This is the reverse of our normal policy of \"fade in on pan\", but should look like any other\n // collision and hopefully not be too noticeable.\n // See https://github.com/mapbox/mapbox-gl-js/issues/7172\n const alwaysShowText = textAlwaysOverlap && (iconAlwaysOverlap || !bucket.hasIconData() || iconOptional);\n const alwaysShowIcon = iconAlwaysOverlap && (textAlwaysOverlap || !bucket.hasTextData() || textOptional);\n\n if (!bucket.collisionArrays && collisionBoxArray) {\n bucket.deserializeCollisionBoxes(collisionBoxArray);\n }\n\n const tileID = this.retainedQueryData[bucket.bucketInstanceId].tileID;\n const getElevation = this.terrain ? (x: number, y: number) => this.terrain.getElevation(tileID, x, y) : null;\n\n const placeSymbol = (symbolInstance: SymbolInstance, collisionArrays: CollisionArrays) => {\n if (seenCrossTileIDs[symbolInstance.crossTileID]) return;\n if (holdingForFade) {\n // Mark all symbols from this tile as \"not placed\", but don't add to seenCrossTileIDs, because we don't\n // know yet if we have a duplicate in a parent tile that _should_ be placed.\n this.placements[symbolInstance.crossTileID] = new JointPlacement(false, false, false);\n return;\n }\n\n let placeText = false;\n let placeIcon = false;\n let offscreen = true;\n let shift = null;\n\n let placed = {box: null, offscreen: null};\n let placedVerticalText = {box: null, offscreen: null};\n\n let placedGlyphBoxes = null;\n let placedGlyphCircles = null;\n let placedIconBoxes = null;\n let textFeatureIndex = 0;\n let verticalTextFeatureIndex = 0;\n let iconFeatureIndex = 0;\n\n if (collisionArrays.textFeatureIndex) {\n textFeatureIndex = collisionArrays.textFeatureIndex;\n } else if (symbolInstance.useRuntimeCollisionCircles) {\n textFeatureIndex = symbolInstance.featureIndex;\n }\n if (collisionArrays.verticalTextFeatureIndex) {\n verticalTextFeatureIndex = collisionArrays.verticalTextFeatureIndex;\n }\n\n const textBox = collisionArrays.textBox;\n if (textBox) {\n\n const updatePreviousOrientationIfNotPlaced = (isPlaced) => {\n let previousOrientation = WritingMode.horizontal;\n if (bucket.allowVerticalPlacement && !isPlaced && this.prevPlacement) {\n const prevPlacedOrientation = this.prevPlacement.placedOrientations[symbolInstance.crossTileID];\n if (prevPlacedOrientation) {\n this.placedOrientations[symbolInstance.crossTileID] = prevPlacedOrientation;\n previousOrientation = prevPlacedOrientation;\n this.markUsedOrientation(bucket, previousOrientation, symbolInstance);\n }\n }\n return previousOrientation;\n };\n\n const placeTextForPlacementModes = (placeHorizontalFn, placeVerticalFn) => {\n if (bucket.allowVerticalPlacement && symbolInstance.numVerticalGlyphVertices > 0 && collisionArrays.verticalTextBox) {\n for (const placementMode of bucket.writingModes) {\n if (placementMode === WritingMode.vertical) {\n placed = placeVerticalFn();\n placedVerticalText = placed;\n } else {\n placed = placeHorizontalFn();\n }\n if (placed && placed.box && placed.box.length) break;\n }\n } else {\n placed = placeHorizontalFn();\n }\n };\n\n const textAnchorOffsetStart = symbolInstance.textAnchorOffsetStartIndex;\n const textAnchorOffsetEnd = symbolInstance.textAnchorOffsetEndIndex;\n\n // If start+end indices match, text-variable-anchor is not in play.\n if (textAnchorOffsetEnd === textAnchorOffsetStart) {\n const placeBox = (collisionTextBox, orientation) => {\n const placedFeature = this.collisionIndex.placeCollisionBox(\n collisionTextBox,\n textOverlapMode,\n textPixelRatio,\n posMatrix,\n collisionGroup.predicate,\n getElevation\n );\n if (placedFeature && placedFeature.box && placedFeature.box.length) {\n this.markUsedOrientation(bucket, orientation, symbolInstance);\n this.placedOrientations[symbolInstance.crossTileID] = orientation;\n }\n return placedFeature;\n };\n\n const placeHorizontal = () => {\n return placeBox(textBox, WritingMode.horizontal);\n };\n\n const placeVertical = () => {\n const verticalTextBox = collisionArrays.verticalTextBox;\n if (bucket.allowVerticalPlacement && symbolInstance.numVerticalGlyphVertices > 0 && verticalTextBox) {\n return placeBox(verticalTextBox, WritingMode.vertical);\n }\n return {box: null, offscreen: null};\n };\n\n placeTextForPlacementModes(placeHorizontal, placeVertical);\n updatePreviousOrientationIfNotPlaced(placed && placed.box && placed.box.length);\n\n } else {\n // If this symbol was in the last placement, prefer placement using same anchor, if it's still available\n let prevAnchor = TextAnchorEnum[this.prevPlacement?.variableOffsets[symbolInstance.crossTileID]?.anchor];\n\n const placeBoxForVariableAnchors = (collisionTextBox, collisionIconBox, orientation) => {\n const width = collisionTextBox.x2 - collisionTextBox.x1;\n const height = collisionTextBox.y2 - collisionTextBox.y1;\n const textBoxScale = symbolInstance.textBoxScale;\n const variableIconBox = hasIconTextFit && (iconOverlapMode === 'never') ? collisionIconBox : null;\n\n let placedBox: {\n box: Array;\n offscreen: boolean;\n } = {box: [], offscreen: false};\n let placementPasses = (textOverlapMode === 'never') ? 1 : 2;\n let overlapMode: OverlapMode = 'never';\n\n if (prevAnchor) {\n placementPasses++;\n }\n\n for (let pass = 0; pass < placementPasses; pass++) {\n for (let i = textAnchorOffsetStart; i < textAnchorOffsetEnd; i++) {\n const textAnchorOffset = bucket.textAnchorOffsets.get(i);\n\n if (prevAnchor && textAnchorOffset.textAnchor !== prevAnchor) {\n continue;\n }\n\n const result = this.attemptAnchorPlacement(\n textAnchorOffset, collisionTextBox, width, height,\n textBoxScale, rotateWithMap, pitchWithMap, textPixelRatio, posMatrix,\n collisionGroup, overlapMode, symbolInstance, bucket, orientation, variableIconBox, getElevation);\n\n if (result) {\n placedBox = result.placedGlyphBoxes;\n if (placedBox && placedBox.box && placedBox.box.length) {\n placeText = true;\n shift = result.shift;\n return placedBox;\n }\n }\n }\n\n if (prevAnchor) {\n prevAnchor = null;\n } else {\n overlapMode = textOverlapMode;\n }\n }\n\n return placedBox;\n };\n\n const placeHorizontal = () => {\n return placeBoxForVariableAnchors(textBox, collisionArrays.iconBox, WritingMode.horizontal);\n };\n\n const placeVertical = () => {\n const verticalTextBox = collisionArrays.verticalTextBox;\n const wasPlaced = placed && placed.box && placed.box.length;\n if (bucket.allowVerticalPlacement && !wasPlaced && symbolInstance.numVerticalGlyphVertices > 0 && verticalTextBox) {\n return placeBoxForVariableAnchors(verticalTextBox, collisionArrays.verticalIconBox, WritingMode.vertical);\n }\n return {box: null, offscreen: null};\n };\n\n placeTextForPlacementModes(placeHorizontal, placeVertical);\n\n if (placed) {\n placeText = placed.box;\n offscreen = placed.offscreen;\n }\n\n const prevOrientation = updatePreviousOrientationIfNotPlaced(placed && placed.box);\n\n // If we didn't get placed, we still need to copy our position from the last placement for\n // fade animations\n if (!placeText && this.prevPlacement) {\n const prevOffset = this.prevPlacement.variableOffsets[symbolInstance.crossTileID];\n if (prevOffset) {\n this.variableOffsets[symbolInstance.crossTileID] = prevOffset;\n this.markUsedJustification(bucket, prevOffset.anchor, symbolInstance, prevOrientation);\n }\n }\n\n }\n }\n\n placedGlyphBoxes = placed;\n placeText = placedGlyphBoxes && placedGlyphBoxes.box && placedGlyphBoxes.box.length > 0;\n\n offscreen = placedGlyphBoxes && placedGlyphBoxes.offscreen;\n\n if (symbolInstance.useRuntimeCollisionCircles) {\n const placedSymbol = bucket.text.placedSymbolArray.get(symbolInstance.centerJustifiedTextSymbolIndex);\n const fontSize = symbolSize.evaluateSizeForFeature(bucket.textSizeData, partiallyEvaluatedTextSize, placedSymbol);\n\n const textPixelPadding = layout.get('text-padding');\n const circlePixelDiameter = symbolInstance.collisionCircleDiameter;\n\n placedGlyphCircles = this.collisionIndex.placeCollisionCircles(\n textOverlapMode,\n placedSymbol,\n bucket.lineVertexArray,\n bucket.glyphOffsetArray,\n fontSize,\n posMatrix,\n textLabelPlaneMatrix,\n labelToScreenMatrix,\n showCollisionBoxes,\n pitchWithMap,\n collisionGroup.predicate,\n circlePixelDiameter,\n textPixelPadding,\n getElevation\n );\n\n if (placedGlyphCircles.circles.length && placedGlyphCircles.collisionDetected && !showCollisionBoxes) {\n warnOnce('Collisions detected, but collision boxes are not shown');\n }\n\n // If text-overlap is set to 'always', force \"placedCircles\" to true\n // In theory there should always be at least one circle placed\n // in this case, but for now quirks in text-anchor\n // and text-offset may prevent that from being true.\n placeText = textAlwaysOverlap || (placedGlyphCircles.circles.length > 0 && !placedGlyphCircles.collisionDetected);\n offscreen = offscreen && placedGlyphCircles.offscreen;\n }\n\n if (collisionArrays.iconFeatureIndex) {\n iconFeatureIndex = collisionArrays.iconFeatureIndex;\n }\n\n if (collisionArrays.iconBox) {\n const placeIconFeature = iconBox => {\n const shiftedIconBox = hasIconTextFit && shift ?\n shiftVariableCollisionBox(\n iconBox, shift.x, shift.y,\n rotateWithMap, pitchWithMap, this.transform.angle) :\n iconBox;\n return this.collisionIndex.placeCollisionBox(shiftedIconBox,\n iconOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation);\n };\n\n if (placedVerticalText && placedVerticalText.box && placedVerticalText.box.length && collisionArrays.verticalIconBox) {\n placedIconBoxes = placeIconFeature(collisionArrays.verticalIconBox);\n placeIcon = placedIconBoxes.box.length > 0;\n } else {\n placedIconBoxes = placeIconFeature(collisionArrays.iconBox);\n placeIcon = placedIconBoxes.box.length > 0;\n }\n offscreen = offscreen && placedIconBoxes.offscreen;\n }\n\n const iconWithoutText = textOptional ||\n (symbolInstance.numHorizontalGlyphVertices === 0 && symbolInstance.numVerticalGlyphVertices === 0);\n const textWithoutIcon = iconOptional || symbolInstance.numIconVertices === 0;\n\n // Combine the scales for icons and text.\n if (!iconWithoutText && !textWithoutIcon) {\n placeIcon = placeText = placeIcon && placeText;\n } else if (!textWithoutIcon) {\n placeText = placeIcon && placeText;\n } else if (!iconWithoutText) {\n placeIcon = placeIcon && placeText;\n }\n\n if (placeText && placedGlyphBoxes && placedGlyphBoxes.box) {\n if (placedVerticalText && placedVerticalText.box && verticalTextFeatureIndex) {\n this.collisionIndex.insertCollisionBox(\n placedGlyphBoxes.box,\n textOverlapMode,\n layout.get('text-ignore-placement'),\n bucket.bucketInstanceId,\n verticalTextFeatureIndex,\n collisionGroup.ID);\n } else {\n this.collisionIndex.insertCollisionBox(\n placedGlyphBoxes.box,\n textOverlapMode,\n layout.get('text-ignore-placement'),\n bucket.bucketInstanceId,\n textFeatureIndex,\n collisionGroup.ID);\n }\n\n }\n if (placeIcon && placedIconBoxes) {\n this.collisionIndex.insertCollisionBox(\n placedIconBoxes.box,\n iconOverlapMode,\n layout.get('icon-ignore-placement'),\n bucket.bucketInstanceId,\n iconFeatureIndex,\n collisionGroup.ID);\n }\n if (placedGlyphCircles) {\n if (placeText) {\n this.collisionIndex.insertCollisionCircles(\n placedGlyphCircles.circles,\n textOverlapMode,\n layout.get('text-ignore-placement'),\n bucket.bucketInstanceId,\n textFeatureIndex,\n collisionGroup.ID);\n }\n\n if (showCollisionBoxes) {\n const id = bucket.bucketInstanceId;\n let circleArray = this.collisionCircleArrays[id];\n\n // Group collision circles together by bucket. Circles can't be pushed forward for rendering yet as the symbol placement\n // for a bucket is not guaranteed to be complete before the commit-function has been called\n if (circleArray === undefined)\n circleArray = this.collisionCircleArrays[id] = new CollisionCircleArray();\n\n for (let i = 0; i < placedGlyphCircles.circles.length; i += 4) {\n circleArray.circles.push(placedGlyphCircles.circles[i + 0]); // x\n circleArray.circles.push(placedGlyphCircles.circles[i + 1]); // y\n circleArray.circles.push(placedGlyphCircles.circles[i + 2]); // radius\n circleArray.circles.push(placedGlyphCircles.collisionDetected ? 1 : 0); // collisionDetected-flag\n }\n }\n }\n\n if (symbolInstance.crossTileID === 0) throw new Error('symbolInstance.crossTileID can\\'t be 0');\n if (bucket.bucketInstanceId === 0) throw new Error('bucket.bucketInstanceId can\\'t be 0');\n\n this.placements[symbolInstance.crossTileID] = new JointPlacement(placeText || alwaysShowText, placeIcon || alwaysShowIcon, offscreen || bucket.justReloaded);\n seenCrossTileIDs[symbolInstance.crossTileID] = true;\n };\n\n if (zOrderByViewportY) {\n if (bucketPart.symbolInstanceStart !== 0) throw new Error('bucket.bucketInstanceId should be 0');\n const symbolIndexes = bucket.getSortedSymbolIndexes(this.transform.angle);\n for (let i = symbolIndexes.length - 1; i >= 0; --i) {\n const symbolIndex = symbolIndexes[i];\n placeSymbol(bucket.symbolInstances.get(symbolIndex), bucket.collisionArrays[symbolIndex]);\n }\n } else {\n for (let i = bucketPart.symbolInstanceStart; i < bucketPart.symbolInstanceEnd; i++) {\n placeSymbol(bucket.symbolInstances.get(i), bucket.collisionArrays[i]);\n }\n }\n\n if (showCollisionBoxes && bucket.bucketInstanceId in this.collisionCircleArrays) {\n const circleArray = this.collisionCircleArrays[bucket.bucketInstanceId];\n\n // Store viewport and inverse projection matrices per bucket\n mat4.invert(circleArray.invProjMatrix, posMatrix);\n circleArray.viewportMatrix = this.collisionIndex.getViewportMatrix();\n }\n\n bucket.justReloaded = false;\n }\n\n markUsedJustification(bucket: SymbolBucket, placedAnchor: TextAnchor, symbolInstance: SymbolInstance, orientation: number) {\n const justifications = {\n 'left': symbolInstance.leftJustifiedTextSymbolIndex,\n 'center': symbolInstance.centerJustifiedTextSymbolIndex,\n 'right': symbolInstance.rightJustifiedTextSymbolIndex\n };\n\n let autoIndex;\n if (orientation === WritingMode.vertical) {\n autoIndex = symbolInstance.verticalPlacedTextSymbolIndex;\n } else {\n autoIndex = justifications[getAnchorJustification(placedAnchor)];\n }\n\n const indexes = [\n symbolInstance.leftJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.rightJustifiedTextSymbolIndex,\n symbolInstance.verticalPlacedTextSymbolIndex\n ];\n\n for (const index of indexes) {\n if (index >= 0) {\n if (autoIndex >= 0 && index !== autoIndex) {\n // There are multiple justifications and this one isn't it: shift offscreen\n bucket.text.placedSymbolArray.get(index).crossTileID = 0;\n } else {\n // Either this is the chosen justification or the justification is hardwired: use this one\n bucket.text.placedSymbolArray.get(index).crossTileID = symbolInstance.crossTileID;\n }\n }\n }\n }\n\n markUsedOrientation(bucket: SymbolBucket, orientation: number, symbolInstance: SymbolInstance) {\n const horizontal = (orientation === WritingMode.horizontal || orientation === WritingMode.horizontalOnly) ? orientation : 0;\n const vertical = orientation === WritingMode.vertical ? orientation : 0;\n\n const horizontalIndexes = [\n symbolInstance.leftJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.rightJustifiedTextSymbolIndex\n ];\n\n for (const index of horizontalIndexes) {\n bucket.text.placedSymbolArray.get(index).placedOrientation = horizontal;\n }\n\n if (symbolInstance.verticalPlacedTextSymbolIndex) {\n bucket.text.placedSymbolArray.get(symbolInstance.verticalPlacedTextSymbolIndex).placedOrientation = vertical;\n }\n }\n\n commit(now: number): void {\n this.commitTime = now;\n this.zoomAtLastRecencyCheck = this.transform.zoom;\n\n const prevPlacement = this.prevPlacement;\n let placementChanged = false;\n\n this.prevZoomAdjustment = prevPlacement ? prevPlacement.zoomAdjustment(this.transform.zoom) : 0;\n const increment = prevPlacement ? prevPlacement.symbolFadeChange(now) : 1;\n\n const prevOpacities = prevPlacement ? prevPlacement.opacities : {};\n const prevOffsets = prevPlacement ? prevPlacement.variableOffsets : {};\n const prevOrientations = prevPlacement ? prevPlacement.placedOrientations : {};\n\n // add the opacities from the current placement, and copy their current values from the previous placement\n for (const crossTileID in this.placements) {\n const jointPlacement = this.placements[crossTileID];\n const prevOpacity = prevOpacities[crossTileID];\n if (prevOpacity) {\n this.opacities[crossTileID] = new JointOpacityState(prevOpacity, increment, jointPlacement.text, jointPlacement.icon);\n placementChanged = placementChanged ||\n jointPlacement.text !== prevOpacity.text.placed ||\n jointPlacement.icon !== prevOpacity.icon.placed;\n } else {\n this.opacities[crossTileID] = new JointOpacityState(null, increment, jointPlacement.text, jointPlacement.icon, jointPlacement.skipFade);\n placementChanged = placementChanged || jointPlacement.text || jointPlacement.icon;\n }\n }\n\n // copy and update values from the previous placement that aren't in the current placement but haven't finished fading\n for (const crossTileID in prevOpacities) {\n const prevOpacity = prevOpacities[crossTileID];\n if (!this.opacities[crossTileID]) {\n const jointOpacity = new JointOpacityState(prevOpacity, increment, false, false);\n if (!jointOpacity.isHidden()) {\n this.opacities[crossTileID] = jointOpacity;\n placementChanged = placementChanged || prevOpacity.text.placed || prevOpacity.icon.placed;\n }\n }\n }\n for (const crossTileID in prevOffsets) {\n if (!this.variableOffsets[crossTileID] && this.opacities[crossTileID] && !this.opacities[crossTileID].isHidden()) {\n this.variableOffsets[crossTileID] = prevOffsets[crossTileID];\n }\n }\n\n for (const crossTileID in prevOrientations) {\n if (!this.placedOrientations[crossTileID] && this.opacities[crossTileID] && !this.opacities[crossTileID].isHidden()) {\n this.placedOrientations[crossTileID] = prevOrientations[crossTileID];\n }\n }\n\n // this.lastPlacementChangeTime is the time of the last commit() that\n // resulted in a placement change -- in other words, the start time of\n // the last symbol fade animation\n if (prevPlacement && prevPlacement.lastPlacementChangeTime === undefined) {\n throw new Error('Last placement time for previous placement is not defined');\n }\n if (placementChanged) {\n this.lastPlacementChangeTime = now;\n } else if (typeof this.lastPlacementChangeTime !== 'number') {\n this.lastPlacementChangeTime = prevPlacement ? prevPlacement.lastPlacementChangeTime : now;\n }\n }\n\n updateLayerOpacities(styleLayer: StyleLayer, tiles: Array) {\n const seenCrossTileIDs = {};\n for (const tile of tiles) {\n const symbolBucket = tile.getBucket(styleLayer) as SymbolBucket;\n if (symbolBucket && tile.latestFeatureIndex && styleLayer.id === symbolBucket.layerIds[0]) {\n this.updateBucketOpacities(symbolBucket, seenCrossTileIDs, tile.collisionBoxArray);\n }\n }\n }\n\n updateBucketOpacities(bucket: SymbolBucket, seenCrossTileIDs: {\n [k in string | number]: boolean;\n }, collisionBoxArray?: CollisionBoxArray | null) {\n if (bucket.hasTextData()) {\n bucket.text.opacityVertexArray.clear();\n bucket.text.hasVisibleVertices = false;\n }\n if (bucket.hasIconData()) {\n bucket.icon.opacityVertexArray.clear();\n bucket.icon.hasVisibleVertices = false;\n }\n if (bucket.hasIconCollisionBoxData()) bucket.iconCollisionBox.collisionVertexArray.clear();\n if (bucket.hasTextCollisionBoxData()) bucket.textCollisionBox.collisionVertexArray.clear();\n\n const layer = bucket.layers[0];\n const layout = layer.layout;\n const duplicateOpacityState = new JointOpacityState(null, 0, false, false, true);\n const textAllowOverlap = layout.get('text-allow-overlap');\n const iconAllowOverlap = layout.get('icon-allow-overlap');\n const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset');\n const rotateWithMap = layout.get('text-rotation-alignment') === 'map';\n const pitchWithMap = layout.get('text-pitch-alignment') === 'map';\n const hasIconTextFit = layout.get('icon-text-fit') !== 'none';\n // If allow-overlap is true, we can show symbols before placement runs on them\n // But we have to wait for placement if we potentially depend on a paired icon/text\n // with allow-overlap: false.\n // See https://github.com/mapbox/mapbox-gl-js/issues/7032\n const defaultOpacityState = new JointOpacityState(null, 0,\n textAllowOverlap && (iconAllowOverlap || !bucket.hasIconData() || layout.get('icon-optional')),\n iconAllowOverlap && (textAllowOverlap || !bucket.hasTextData() || layout.get('text-optional')),\n true);\n\n if (!bucket.collisionArrays && collisionBoxArray && ((bucket.hasIconCollisionBoxData() || bucket.hasTextCollisionBoxData()))) {\n bucket.deserializeCollisionBoxes(collisionBoxArray);\n }\n\n const addOpacities = (iconOrText, numVertices: number, opacity: number) => {\n for (let i = 0; i < numVertices / 4; i++) {\n iconOrText.opacityVertexArray.emplaceBack(opacity);\n }\n iconOrText.hasVisibleVertices = iconOrText.hasVisibleVertices || (opacity !== PACKED_HIDDEN_OPACITY);\n };\n\n for (let s = 0; s < bucket.symbolInstances.length; s++) {\n const symbolInstance = bucket.symbolInstances.get(s);\n const {\n numHorizontalGlyphVertices,\n numVerticalGlyphVertices,\n crossTileID\n } = symbolInstance;\n\n const isDuplicate = seenCrossTileIDs[crossTileID];\n\n let opacityState = this.opacities[crossTileID];\n if (isDuplicate) {\n opacityState = duplicateOpacityState;\n } else if (!opacityState) {\n opacityState = defaultOpacityState;\n // store the state so that future placements use it as a starting point\n this.opacities[crossTileID] = opacityState;\n }\n\n seenCrossTileIDs[crossTileID] = true;\n\n const hasText = numHorizontalGlyphVertices > 0 || numVerticalGlyphVertices > 0;\n const hasIcon = symbolInstance.numIconVertices > 0;\n\n const placedOrientation = this.placedOrientations[symbolInstance.crossTileID];\n const horizontalHidden = placedOrientation === WritingMode.vertical;\n const verticalHidden = placedOrientation === WritingMode.horizontal || placedOrientation === WritingMode.horizontalOnly;\n\n if (hasText) {\n const packedOpacity = packOpacity(opacityState.text);\n // Vertical text fades in/out on collision the same way as corresponding\n // horizontal text. Switch between vertical/horizontal should be instantaneous\n const horizontalOpacity = horizontalHidden ? PACKED_HIDDEN_OPACITY : packedOpacity;\n addOpacities(bucket.text, numHorizontalGlyphVertices, horizontalOpacity);\n const verticalOpacity = verticalHidden ? PACKED_HIDDEN_OPACITY : packedOpacity;\n addOpacities(bucket.text, numVerticalGlyphVertices, verticalOpacity);\n\n // If this label is completely faded, mark it so that we don't have to calculate\n // its position at render time. If this layer has variable placement, shift the various\n // symbol instances appropriately so that symbols from buckets that have yet to be placed\n // offset appropriately.\n const symbolHidden = opacityState.text.isHidden();\n [\n symbolInstance.rightJustifiedTextSymbolIndex,\n symbolInstance.centerJustifiedTextSymbolIndex,\n symbolInstance.leftJustifiedTextSymbolIndex\n ].forEach(index => {\n if (index >= 0) {\n bucket.text.placedSymbolArray.get(index).hidden = symbolHidden || horizontalHidden ? 1 : 0;\n }\n });\n\n if (symbolInstance.verticalPlacedTextSymbolIndex >= 0) {\n bucket.text.placedSymbolArray.get(symbolInstance.verticalPlacedTextSymbolIndex).hidden = symbolHidden || verticalHidden ? 1 : 0;\n }\n\n const prevOffset = this.variableOffsets[symbolInstance.crossTileID];\n if (prevOffset) {\n this.markUsedJustification(bucket, prevOffset.anchor, symbolInstance, placedOrientation);\n }\n\n const prevOrientation = this.placedOrientations[symbolInstance.crossTileID];\n if (prevOrientation) {\n this.markUsedJustification(bucket, 'left', symbolInstance, prevOrientation);\n this.markUsedOrientation(bucket, prevOrientation, symbolInstance);\n }\n }\n\n if (hasIcon) {\n const packedOpacity = packOpacity(opacityState.icon);\n\n const useHorizontal = !(hasIconTextFit && symbolInstance.verticalPlacedIconSymbolIndex && horizontalHidden);\n\n if (symbolInstance.placedIconSymbolIndex >= 0) {\n const horizontalOpacity = useHorizontal ? packedOpacity : PACKED_HIDDEN_OPACITY;\n addOpacities(bucket.icon, symbolInstance.numIconVertices, horizontalOpacity);\n bucket.icon.placedSymbolArray.get(symbolInstance.placedIconSymbolIndex).hidden =\n (opacityState.icon.isHidden() as any);\n }\n\n if (symbolInstance.verticalPlacedIconSymbolIndex >= 0) {\n const verticalOpacity = !useHorizontal ? packedOpacity : PACKED_HIDDEN_OPACITY;\n addOpacities(bucket.icon, symbolInstance.numVerticalIconVertices, verticalOpacity);\n bucket.icon.placedSymbolArray.get(symbolInstance.verticalPlacedIconSymbolIndex).hidden =\n (opacityState.icon.isHidden() as any);\n }\n }\n\n if (bucket.hasIconCollisionBoxData() || bucket.hasTextCollisionBoxData()) {\n const collisionArrays = bucket.collisionArrays[s];\n if (collisionArrays) {\n let shift = new Point(0, 0);\n if (collisionArrays.textBox || collisionArrays.verticalTextBox) {\n let used = true;\n if (hasVariablePlacement) {\n const variableOffset = this.variableOffsets[crossTileID];\n if (variableOffset) {\n // This will show either the currently placed position or the last\n // successfully placed position (so you can visualize what collision\n // just made the symbol disappear, and the most likely place for the\n // symbol to come back)\n shift = calculateVariableLayoutShift(variableOffset.anchor,\n variableOffset.width,\n variableOffset.height,\n variableOffset.textOffset,\n variableOffset.textBoxScale);\n if (rotateWithMap) {\n shift._rotate(pitchWithMap ? this.transform.angle : -this.transform.angle);\n }\n } else {\n // No offset -> this symbol hasn't been placed since coming on-screen\n // No single box is particularly meaningful and all of them would be too noisy\n // Use the center box just to show something's there, but mark it \"not used\"\n used = false;\n }\n }\n\n if (collisionArrays.textBox) {\n updateCollisionVertices(bucket.textCollisionBox.collisionVertexArray, opacityState.text.placed, !used || horizontalHidden, shift.x, shift.y);\n }\n if (collisionArrays.verticalTextBox) {\n updateCollisionVertices(bucket.textCollisionBox.collisionVertexArray, opacityState.text.placed, !used || verticalHidden, shift.x, shift.y);\n }\n }\n\n const verticalIconUsed = Boolean(!verticalHidden && collisionArrays.verticalIconBox);\n\n if (collisionArrays.iconBox) {\n updateCollisionVertices(bucket.iconCollisionBox.collisionVertexArray, opacityState.icon.placed, verticalIconUsed,\n hasIconTextFit ? shift.x : 0,\n hasIconTextFit ? shift.y : 0);\n }\n\n if (collisionArrays.verticalIconBox) {\n updateCollisionVertices(bucket.iconCollisionBox.collisionVertexArray, opacityState.icon.placed, !verticalIconUsed,\n hasIconTextFit ? shift.x : 0,\n hasIconTextFit ? shift.y : 0);\n }\n }\n }\n }\n\n bucket.sortFeatures(this.transform.angle);\n if (this.retainedQueryData[bucket.bucketInstanceId]) {\n this.retainedQueryData[bucket.bucketInstanceId].featureSortOrder = bucket.featureSortOrder;\n }\n\n if (bucket.hasTextData() && bucket.text.opacityVertexBuffer) {\n bucket.text.opacityVertexBuffer.updateData(bucket.text.opacityVertexArray);\n }\n if (bucket.hasIconData() && bucket.icon.opacityVertexBuffer) {\n bucket.icon.opacityVertexBuffer.updateData(bucket.icon.opacityVertexArray);\n }\n if (bucket.hasIconCollisionBoxData() && bucket.iconCollisionBox.collisionVertexBuffer) {\n bucket.iconCollisionBox.collisionVertexBuffer.updateData(bucket.iconCollisionBox.collisionVertexArray);\n }\n if (bucket.hasTextCollisionBoxData() && bucket.textCollisionBox.collisionVertexBuffer) {\n bucket.textCollisionBox.collisionVertexBuffer.updateData(bucket.textCollisionBox.collisionVertexArray);\n }\n\n if (bucket.text.opacityVertexArray.length !== bucket.text.layoutVertexArray.length / 4) throw new Error(`bucket.text.opacityVertexArray.length (= ${bucket.text.opacityVertexArray.length}) !== bucket.text.layoutVertexArray.length (= ${bucket.text.layoutVertexArray.length}) / 4`);\n if (bucket.icon.opacityVertexArray.length !== bucket.icon.layoutVertexArray.length / 4) throw new Error(`bucket.icon.opacityVertexArray.length (= ${bucket.icon.opacityVertexArray.length}) !== bucket.icon.layoutVertexArray.length (= ${bucket.icon.layoutVertexArray.length}) / 4`);\n\n // Push generated collision circles to the bucket for debug rendering\n if (bucket.bucketInstanceId in this.collisionCircleArrays) {\n const instance = this.collisionCircleArrays[bucket.bucketInstanceId];\n\n bucket.placementInvProjMatrix = instance.invProjMatrix;\n bucket.placementViewportMatrix = instance.viewportMatrix;\n bucket.collisionCircleArray = instance.circles;\n\n delete this.collisionCircleArrays[bucket.bucketInstanceId];\n }\n }\n\n symbolFadeChange(now: number) {\n return this.fadeDuration === 0 ?\n 1 :\n ((now - this.commitTime) / this.fadeDuration + this.prevZoomAdjustment);\n }\n\n zoomAdjustment(zoom: number) {\n // When zooming out quickly, labels can overlap each other. This\n // adjustment is used to reduce the interval between placement calculations\n // and to reduce the fade duration when zooming out quickly. Discovering the\n // collisions more quickly and fading them more quickly reduces the unwanted effect.\n return Math.max(0, (this.transform.zoom - zoom) / 1.5);\n }\n\n hasTransitions(now: number) {\n return this.stale ||\n now - this.lastPlacementChangeTime < this.fadeDuration;\n }\n\n stillRecent(now: number, zoom: number) {\n // The adjustment makes placement more frequent when zooming.\n // This condition applies the adjustment only after the map has\n // stopped zooming. This avoids adding extra jank while zooming.\n const durationAdjustment = this.zoomAtLastRecencyCheck === zoom ?\n (1 - this.zoomAdjustment(zoom)) :\n 1;\n this.zoomAtLastRecencyCheck = zoom;\n\n return this.commitTime + this.fadeDuration * durationAdjustment > now;\n }\n\n setStale() {\n this.stale = true;\n }\n}\n\nfunction updateCollisionVertices(collisionVertexArray: CollisionVertexArray, placed: boolean, notUsed: boolean | number, shiftX?: number, shiftY?: number) {\n collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0);\n collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0);\n collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0);\n collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0);\n}\n\n// All four vertices for a glyph will have the same opacity state\n// So we pack the opacity into a uint8, and then repeat it four times\n// to make a single uint32 that we can upload for each glyph in the\n// label.\nconst shift25 = Math.pow(2, 25);\nconst shift24 = Math.pow(2, 24);\nconst shift17 = Math.pow(2, 17);\nconst shift16 = Math.pow(2, 16);\nconst shift9 = Math.pow(2, 9);\nconst shift8 = Math.pow(2, 8);\nconst shift1 = Math.pow(2, 1);\nfunction packOpacity(opacityState: OpacityState): number {\n if (opacityState.opacity === 0 && !opacityState.placed) {\n return 0;\n } else if (opacityState.opacity === 1 && opacityState.placed) {\n return 4294967295;\n }\n const targetBit = opacityState.placed ? 1 : 0;\n const opacityBits = Math.floor(opacityState.opacity * 127);\n return opacityBits * shift25 + targetBit * shift24 +\n opacityBits * shift17 + targetBit * shift16 +\n opacityBits * shift9 + targetBit * shift8 +\n opacityBits * shift1 + targetBit;\n}\n\nconst PACKED_HIDDEN_OPACITY = 0;\n","import {browser} from '../util/browser';\n\nimport {Placement} from '../symbol/placement';\n\nimport type {Transform} from '../geo/transform';\nimport type {StyleLayer} from './style_layer';\nimport type {SymbolStyleLayer} from './style_layer/symbol_style_layer';\nimport type {Tile} from '../source/tile';\nimport type {BucketPart} from '../symbol/placement';\nimport {Terrain} from '../render/terrain';\n\nclass LayerPlacement {\n _sortAcrossTiles: boolean;\n _currentTileIndex: number;\n _currentPartIndex: number;\n _seenCrossTileIDs: {\n [k in string | number]: boolean;\n };\n _bucketParts: Array;\n\n constructor(styleLayer: SymbolStyleLayer) {\n this._sortAcrossTiles = styleLayer.layout.get('symbol-z-order') !== 'viewport-y' &&\n !styleLayer.layout.get('symbol-sort-key').isConstant();\n\n this._currentTileIndex = 0;\n this._currentPartIndex = 0;\n this._seenCrossTileIDs = {};\n this._bucketParts = [];\n }\n\n continuePlacement(tiles: Array, placement: Placement, showCollisionBoxes: boolean, styleLayer: StyleLayer, shouldPausePlacement: () => boolean) {\n\n const bucketParts = this._bucketParts;\n\n while (this._currentTileIndex < tiles.length) {\n const tile = tiles[this._currentTileIndex];\n placement.getBucketParts(bucketParts, styleLayer, tile, this._sortAcrossTiles);\n\n this._currentTileIndex++;\n if (shouldPausePlacement()) {\n return true;\n }\n }\n\n if (this._sortAcrossTiles) {\n this._sortAcrossTiles = false;\n bucketParts.sort((a, b) => (a.sortKey as any as number) - (b.sortKey as any as number));\n }\n\n while (this._currentPartIndex < bucketParts.length) {\n const bucketPart = bucketParts[this._currentPartIndex];\n placement.placeLayerBucketPart(bucketPart, this._seenCrossTileIDs, showCollisionBoxes);\n\n this._currentPartIndex++;\n if (shouldPausePlacement()) {\n return true;\n }\n }\n return false;\n }\n}\n\nexport class PauseablePlacement {\n placement: Placement;\n _done: boolean;\n _currentPlacementIndex: number;\n _forceFullPlacement: boolean;\n _showCollisionBoxes: boolean;\n _inProgressLayer: LayerPlacement;\n\n constructor(\n transform: Transform,\n terrain: Terrain,\n order: Array,\n forceFullPlacement: boolean,\n showCollisionBoxes: boolean,\n fadeDuration: number,\n crossSourceCollisions: boolean,\n prevPlacement?: Placement\n ) {\n this.placement = new Placement(transform, terrain, fadeDuration, crossSourceCollisions, prevPlacement);\n this._currentPlacementIndex = order.length - 1;\n this._forceFullPlacement = forceFullPlacement;\n this._showCollisionBoxes = showCollisionBoxes;\n this._done = false;\n }\n\n isDone() {\n return this._done;\n }\n\n continuePlacement(\n order: Array,\n layers: {[_: string]: StyleLayer},\n layerTiles: {[_: string]: Array}\n ) {\n const startTime = browser.now();\n\n const shouldPausePlacement = () => {\n return this._forceFullPlacement ? false : (browser.now() - startTime) > 2;\n };\n\n while (this._currentPlacementIndex >= 0) {\n const layerId = order[this._currentPlacementIndex];\n const layer = layers[layerId];\n const placementZoom = this.placement.collisionIndex.transform.zoom;\n if (layer.type === 'symbol' &&\n (!layer.minzoom || layer.minzoom <= placementZoom) &&\n (!layer.maxzoom || layer.maxzoom > placementZoom)) {\n\n if (!this._inProgressLayer) {\n this._inProgressLayer = new LayerPlacement(layer as any as SymbolStyleLayer);\n }\n\n const pausePlacement = this._inProgressLayer.continuePlacement(layerTiles[layer.source], this.placement, this._showCollisionBoxes, layer, shouldPausePlacement);\n\n if (pausePlacement) {\n // We didn't finish placing all layers within 2ms,\n // but we can keep rendering with a partial placement\n // We'll resume here on the next frame\n return;\n }\n\n delete this._inProgressLayer;\n }\n\n this._currentPlacementIndex--;\n }\n\n this._done = true;\n }\n\n commit(now: number) {\n this.placement.commit(now);\n return this.placement;\n }\n}\n","import KDBush from 'kdbush';\nimport {EXTENT} from '../data/extent';\n\nimport {SymbolInstanceArray} from '../data/array_types.g';\n\nimport type {SymbolInstance} from '../data/array_types.g';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {Tile} from '../source/tile';\n\n/*\n The CrossTileSymbolIndex generally works on the assumption that\n a conceptual \"unique symbol\" can be identified by the text of\n the label combined with the anchor point. The goal is to assign\n these conceptual \"unique symbols\" a shared crossTileID that can be\n used by Placement to keep fading opacity states consistent and to\n deduplicate labels.\n\n The CrossTileSymbolIndex indexes all the current symbol instances and\n their crossTileIDs. When a symbol bucket gets added or updated, the\n index assigns a crossTileID to each of it's symbol instances by either\n matching it with an existing id or assigning a new one.\n*/\n\n// Round anchor positions to roughly 4 pixel grid\nconst roundingFactor = 512 / EXTENT / 2;\n\nexport const KDBUSH_THRESHHOLD = 128;\n\ninterface SymbolsByKeyEntry {\n index?: KDBush;\n positions?: {x: number; y: number}[];\n crossTileIDs: number[];\n}\n\nclass TileLayerIndex {\n _symbolsByKey: Record = {};\n\n constructor(public tileID: OverscaledTileID, symbolInstances: SymbolInstanceArray, public bucketInstanceId: number) {\n // group the symbolInstances by key\n const symbolInstancesByKey = new Map();\n for (let i = 0; i < symbolInstances.length; i++) {\n const symbolInstance = symbolInstances.get(i);\n const key = symbolInstance.key;\n const instances = symbolInstancesByKey.get(key);\n if (instances) {\n // This tile may have multiple symbol instances with the same key\n // Store each one along with its coordinates\n instances.push(symbolInstance);\n } else {\n symbolInstancesByKey.set(key, [symbolInstance]);\n }\n }\n\n // index the SymbolInstances in this each bucket\n for (const [key, symbols] of symbolInstancesByKey) {\n const positions = symbols.map(symbolInstance => ({x: Math.floor(symbolInstance.anchorX * roundingFactor), y: Math.floor(symbolInstance.anchorY * roundingFactor)}));\n const crossTileIDs = symbols.map(v => v.crossTileID);\n const entry: SymbolsByKeyEntry = {positions, crossTileIDs};\n\n // once we get too many symbols for a given key, it becomes much faster to index it before queries\n if (entry.positions.length > KDBUSH_THRESHHOLD) {\n\n const index = new KDBush(entry.positions.length, 16, Uint16Array);\n for (const {x, y} of entry.positions) index.add(x, y);\n index.finish();\n\n // clear all references to the original positions data\n delete entry.positions;\n entry.index = index;\n }\n\n this._symbolsByKey[key] = entry;\n }\n }\n\n // Converts the coordinates of the input symbol instance into coordinates that be can compared\n // against other symbols in this index. Coordinates are:\n // (1) local-tile-based (so after correction we get x,y values relative to our local anchorX/Y)\n // (2) converted to the z-scale of this TileLayerIndex\n // (3) down-sampled by \"roundingFactor\" from tile coordinate precision in order to be\n // more tolerant of small differences between tiles.\n getScaledCoordinates(symbolInstance: SymbolInstance, childTileID: OverscaledTileID): {x: number; y: number} {\n const {x: localX, y: localY, z: localZ} = this.tileID.canonical;\n const {x, y, z} = childTileID.canonical;\n\n const zDifference = z - localZ;\n const scale = roundingFactor / Math.pow(2, zDifference);\n const xWorld = (x * EXTENT + symbolInstance.anchorX) * scale;\n const yWorld = (y * EXTENT + symbolInstance.anchorY) * scale;\n const xOffset = localX * EXTENT * roundingFactor;\n const yOffset = localY * EXTENT * roundingFactor;\n const result = {\n x: Math.floor(xWorld - xOffset),\n y: Math.floor(yWorld - yOffset)\n };\n\n return result;\n }\n\n findMatches(symbolInstances: SymbolInstanceArray, newTileID: OverscaledTileID, zoomCrossTileIDs: {\n [crossTileID: number]: boolean;\n }) {\n const tolerance = this.tileID.canonical.z < newTileID.canonical.z ? 1 : Math.pow(2, this.tileID.canonical.z - newTileID.canonical.z);\n\n for (let i = 0; i < symbolInstances.length; i++) {\n const symbolInstance = symbolInstances.get(i);\n if (symbolInstance.crossTileID) {\n // already has a match, skip\n continue;\n }\n\n const entry = this._symbolsByKey[symbolInstance.key];\n if (!entry) {\n // No symbol with this key in this bucket\n continue;\n }\n\n const scaledSymbolCoord = this.getScaledCoordinates(symbolInstance, newTileID);\n\n if (entry.index) {\n // Return any symbol with the same keys whose coordinates are within 1\n // grid unit. (with a 4px grid, this covers a 12px by 12px area)\n const indexes = entry.index.range(\n scaledSymbolCoord.x - tolerance,\n scaledSymbolCoord.y - tolerance,\n scaledSymbolCoord.x + tolerance,\n scaledSymbolCoord.y + tolerance).sort();\n\n for (const i of indexes) {\n const crossTileID = entry.crossTileIDs[i];\n\n if (!zoomCrossTileIDs[crossTileID]) {\n // Once we've marked ourselves duplicate against this parent symbol,\n // don't let any other symbols at the same zoom level duplicate against\n // the same parent (see issue #5993)\n zoomCrossTileIDs[crossTileID] = true;\n symbolInstance.crossTileID = crossTileID;\n break;\n }\n }\n } else if (entry.positions) {\n for (let i = 0; i < entry.positions.length; i++) {\n const thisTileSymbol = entry.positions[i];\n const crossTileID = entry.crossTileIDs[i];\n\n // Return any symbol with the same keys whose coordinates are within 1\n // grid unit. (with a 4px grid, this covers a 12px by 12px area)\n if (Math.abs(thisTileSymbol.x - scaledSymbolCoord.x) <= tolerance &&\n Math.abs(thisTileSymbol.y - scaledSymbolCoord.y) <= tolerance &&\n !zoomCrossTileIDs[crossTileID]) {\n // Once we've marked ourselves duplicate against this parent symbol,\n // don't let any other symbols at the same zoom level duplicate against\n // the same parent (see issue #5993)\n zoomCrossTileIDs[crossTileID] = true;\n symbolInstance.crossTileID = crossTileID;\n break;\n }\n }\n }\n }\n }\n\n getCrossTileIDsLists() {\n return Object.values(this._symbolsByKey).map(({crossTileIDs}) => crossTileIDs);\n }\n}\n\nclass CrossTileIDs {\n maxCrossTileID: number;\n constructor() {\n this.maxCrossTileID = 0;\n }\n generate() {\n return ++this.maxCrossTileID;\n }\n}\n\nclass CrossTileSymbolLayerIndex {\n indexes: {\n [zoom in string | number]: {\n [tileId in string | number]: TileLayerIndex;\n };\n };\n usedCrossTileIDs: {\n [zoom in string | number]: {\n [crossTileID: number]: boolean;\n };\n };\n lng: number;\n\n constructor() {\n this.indexes = {};\n this.usedCrossTileIDs = {};\n this.lng = 0;\n }\n\n /*\n * Sometimes when a user pans across the antimeridian the longitude value gets wrapped.\n * To prevent labels from flashing out and in we adjust the tileID values in the indexes\n * so that they match the new wrapped version of the map.\n */\n handleWrapJump(lng: number) {\n const wrapDelta = Math.round((lng - this.lng) / 360);\n if (wrapDelta !== 0) {\n for (const zoom in this.indexes) {\n const zoomIndexes = this.indexes[zoom];\n const newZoomIndex = {};\n for (const key in zoomIndexes) {\n // change the tileID's wrap and add it to a new index\n const index = zoomIndexes[key];\n index.tileID = index.tileID.unwrapTo(index.tileID.wrap + wrapDelta);\n newZoomIndex[index.tileID.key] = index;\n }\n this.indexes[zoom] = newZoomIndex;\n }\n }\n this.lng = lng;\n }\n\n addBucket(tileID: OverscaledTileID, bucket: SymbolBucket, crossTileIDs: CrossTileIDs) {\n if (this.indexes[tileID.overscaledZ] &&\n this.indexes[tileID.overscaledZ][tileID.key]) {\n if (this.indexes[tileID.overscaledZ][tileID.key].bucketInstanceId ===\n bucket.bucketInstanceId) {\n return false;\n } else {\n // We're replacing this bucket with an updated version\n // Remove the old bucket's \"used crossTileIDs\" now so that\n // the new bucket can claim them.\n // The old index entries themselves stick around until\n // 'removeStaleBuckets' is called.\n this.removeBucketCrossTileIDs(tileID.overscaledZ,\n this.indexes[tileID.overscaledZ][tileID.key]);\n }\n }\n\n for (let i = 0; i < bucket.symbolInstances.length; i++) {\n const symbolInstance = bucket.symbolInstances.get(i);\n symbolInstance.crossTileID = 0;\n }\n\n if (!this.usedCrossTileIDs[tileID.overscaledZ]) {\n this.usedCrossTileIDs[tileID.overscaledZ] = {};\n }\n const zoomCrossTileIDs = this.usedCrossTileIDs[tileID.overscaledZ];\n\n for (const zoom in this.indexes) {\n const zoomIndexes = this.indexes[zoom];\n if (Number(zoom) > tileID.overscaledZ) {\n for (const id in zoomIndexes) {\n const childIndex = zoomIndexes[id];\n if (childIndex.tileID.isChildOf(tileID)) {\n childIndex.findMatches(bucket.symbolInstances, tileID, zoomCrossTileIDs);\n }\n }\n } else {\n const parentCoord = tileID.scaledTo(Number(zoom));\n const parentIndex = zoomIndexes[parentCoord.key];\n if (parentIndex) {\n parentIndex.findMatches(bucket.symbolInstances, tileID, zoomCrossTileIDs);\n }\n }\n }\n\n for (let i = 0; i < bucket.symbolInstances.length; i++) {\n const symbolInstance = bucket.symbolInstances.get(i);\n if (!symbolInstance.crossTileID) {\n // symbol did not match any known symbol, assign a new id\n symbolInstance.crossTileID = crossTileIDs.generate();\n zoomCrossTileIDs[symbolInstance.crossTileID] = true;\n }\n }\n\n if (this.indexes[tileID.overscaledZ] === undefined) {\n this.indexes[tileID.overscaledZ] = {};\n }\n this.indexes[tileID.overscaledZ][tileID.key] = new TileLayerIndex(tileID, bucket.symbolInstances, bucket.bucketInstanceId);\n\n return true;\n }\n\n removeBucketCrossTileIDs(zoom: string | number, removedBucket: TileLayerIndex) {\n for (const crossTileIDs of removedBucket.getCrossTileIDsLists()) {\n for (const crossTileID of crossTileIDs) {\n delete this.usedCrossTileIDs[zoom][crossTileID];\n }\n }\n }\n\n removeStaleBuckets(currentIDs: {\n [k in string | number]: boolean;\n }) {\n let tilesChanged = false;\n for (const z in this.indexes) {\n const zoomIndexes = this.indexes[z];\n for (const tileKey in zoomIndexes) {\n if (!currentIDs[zoomIndexes[tileKey].bucketInstanceId]) {\n this.removeBucketCrossTileIDs(z, zoomIndexes[tileKey]);\n delete zoomIndexes[tileKey];\n tilesChanged = true;\n }\n }\n }\n return tilesChanged;\n }\n}\n\nexport class CrossTileSymbolIndex {\n layerIndexes: {[layerId: string]: CrossTileSymbolLayerIndex};\n crossTileIDs: CrossTileIDs;\n maxBucketInstanceId: number;\n bucketsInCurrentPlacement: {[_: number]: boolean};\n\n constructor() {\n this.layerIndexes = {};\n this.crossTileIDs = new CrossTileIDs();\n this.maxBucketInstanceId = 0;\n this.bucketsInCurrentPlacement = {};\n }\n\n addLayer(styleLayer: StyleLayer, tiles: Array, lng: number) {\n let layerIndex = this.layerIndexes[styleLayer.id];\n if (layerIndex === undefined) {\n layerIndex = this.layerIndexes[styleLayer.id] = new CrossTileSymbolLayerIndex();\n }\n\n let symbolBucketsChanged = false;\n const currentBucketIDs = {};\n\n layerIndex.handleWrapJump(lng);\n\n for (const tile of tiles) {\n const symbolBucket = (tile.getBucket(styleLayer) as any as SymbolBucket);\n if (!symbolBucket || styleLayer.id !== symbolBucket.layerIds[0])\n continue;\n\n if (!symbolBucket.bucketInstanceId) {\n symbolBucket.bucketInstanceId = ++this.maxBucketInstanceId;\n }\n\n if (layerIndex.addBucket(tile.tileID, symbolBucket, this.crossTileIDs)) {\n symbolBucketsChanged = true;\n }\n currentBucketIDs[symbolBucket.bucketInstanceId] = true;\n }\n\n if (layerIndex.removeStaleBuckets(currentBucketIDs)) {\n symbolBucketsChanged = true;\n }\n\n return symbolBucketsChanged;\n }\n\n pruneUnusedLayers(usedLayers: Array) {\n const usedLayerMap = {};\n usedLayers.forEach((usedLayer) => {\n usedLayerMap[usedLayer] = true;\n });\n for (const layerId in this.layerIndexes) {\n if (!usedLayerMap[layerId]) {\n delete this.layerIndexes[layerId];\n }\n }\n }\n}\n","import {Event, ErrorEvent, Evented} from '../util/evented';\nimport {StyleLayer} from './style_layer';\nimport {createStyleLayer} from './create_style_layer';\nimport {loadSprite} from './load_sprite';\nimport {ImageManager} from '../render/image_manager';\nimport {GlyphManager} from '../render/glyph_manager';\nimport {Light} from './light';\nimport {LineAtlas} from '../render/line_atlas';\nimport {pick, clone, extend, deepEqual, filterObject, mapObject} from '../util/util';\nimport {coerceSpriteToArray} from '../util/style';\nimport {getJSON, getReferrer, makeRequest} from '../util/ajax';\nimport {ResourceType} from '../util/request_manager';\nimport {browser} from '../util/browser';\nimport {Dispatcher} from '../util/dispatcher';\nimport {validateStyle, emitValidationErrors as _emitValidationErrors} from './validate_style';\nimport {getSourceType, setSourceType, Source} from '../source/source';\nimport type {SourceClass} from '../source/source';\nimport {QueryRenderedFeaturesOptions, QuerySourceFeatureOptions, queryRenderedFeatures, queryRenderedSymbols, querySourceFeatures} from '../source/query_features';\nimport {SourceCache} from '../source/source_cache';\nimport {GeoJSONSource} from '../source/geojson_source';\nimport {latest as styleSpec, derefLayers as deref, emptyStyle, diff as diffStyles, operations as diffOperations} from '@maplibre/maplibre-gl-style-spec';\nimport {getGlobalWorkerPool} from '../util/global_worker_pool';\nimport {\n registerForPluginStateChange,\n evented as rtlTextPluginEvented,\n triggerPluginCompletionEvent\n} from '../source/rtl_text_plugin';\nimport {PauseablePlacement} from './pauseable_placement';\nimport {ZoomHistory} from './zoom_history';\nimport {CrossTileSymbolIndex} from '../symbol/cross_tile_symbol_index';\nimport {validateCustomStyleLayer} from './style_layer/custom_style_layer';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\n\n// We're skipping validation errors with the `source.canvas` identifier in order\n// to continue to allow canvas sources to be added at runtime/updated in\n// smart setStyle (see https://github.com/mapbox/mapbox-gl-js/pull/6424):\nconst emitValidationErrors = (evented: Evented, errors?: ReadonlyArray<{\n message: string;\n identifier?: string;\n}> | null) =>\n _emitValidationErrors(evented, errors && errors.filter(error => error.identifier !== 'source.canvas'));\n\nimport type {Map} from '../ui/map';\nimport type {Transform} from '../geo/transform';\nimport type {StyleImage} from './style_image';\nimport type {StyleGlyph} from './style_glyph';\nimport type {Callback} from '../types/callback';\nimport type {EvaluationParameters} from './evaluation_parameters';\nimport type {Placement} from '../symbol/placement';\nimport type {Cancelable} from '../types/cancelable';\nimport type {RequestParameters, ResponseCallback} from '../util/ajax';\nimport type {\n LayerSpecification,\n FilterSpecification,\n StyleSpecification,\n LightSpecification,\n SourceSpecification,\n SpriteSpecification,\n} from '@maplibre/maplibre-gl-style-spec';\nimport type {CustomLayerInterface} from './style_layer/custom_style_layer';\nimport type {Validator} from './validate_style';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nconst supportedDiffOperations = pick(diffOperations, [\n 'addLayer',\n 'removeLayer',\n 'setPaintProperty',\n 'setLayoutProperty',\n 'setFilter',\n 'addSource',\n 'removeSource',\n 'setLayerZoomRange',\n 'setLight',\n 'setTransition',\n 'setGeoJSONSourceData',\n 'setGlyphs',\n 'setSprite',\n]);\n\nconst ignoredDiffOperations = pick(diffOperations, [\n 'setCenter',\n 'setZoom',\n 'setBearing',\n 'setPitch'\n]);\n\nconst empty = emptyStyle() as StyleSpecification;\n/**\n * A feature identifier that is bound to a source\n */\nexport type FeatureIdentifier = {\n /**\n * Unique id of the feature.\n */\n id?: string | number | undefined;\n /**\n * The id of the vector or GeoJSON source for the feature.\n */\n source: string;\n /**\n * *For vector tile sources, `sourceLayer` is required.*\n */\n sourceLayer?: string | undefined;\n};\n\n/**\n * The options object related to the {@link Map}'s style related methods\n */\nexport type StyleOptions = {\n /**\n * If false, style validation will be skipped. Useful in production environment.\n */\n validate?: boolean;\n /**\n * Defines a CSS\n * font-family for locally overriding generation of glyphs in the 'CJK Unified Ideographs', 'Hiragana', 'Katakana' and 'Hangul Syllables' ranges.\n * In these ranges, font settings from the map's style will be ignored, except for font-weight keywords (light/regular/medium/bold).\n * Set to `false`, to enable font settings from the map's style for these glyph ranges.\n * Forces a full update.\n */\n localIdeographFontFamily?: string;\n};\n\n/**\n * Supporting type to add validation to another style related type\n */\nexport type StyleSetterOptions = {\n /**\n * Whether to check if the filter conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function.\n */\n validate?: boolean;\n};\n\n/**\n * Part of {@link Map#setStyle} options, transformStyle is a convenience function that allows to modify a style after it is fetched but before it is committed to the map state\n * this function exposes previous and next styles, it can be commonly used to support a range of functionalities like:\n * when previous style carries certain 'state' that needs to be carried over to a new style gracefully\n * when a desired style is a certain combination of previous and incoming style\n * when an incoming style requires modification based on external state\n *\n * @param previousStyle - The current style.\n * @param nextStyle - The next style.\n * @returns resulting style that will to be applied to the map\n *\n * @example\n * ```ts\n * map.setStyle('https://demotiles.maplibre.org/style.json', {\n * transformStyle: (previousStyle, nextStyle) => ({\n * ...nextStyle,\n * sources: {\n * ...nextStyle.sources,\n * // copy a source from previous style\n * 'osm': previousStyle.sources.osm\n * },\n * layers: [\n * // background layer\n * nextStyle.layers[0],\n * // copy a layer from previous style\n * previousStyle.layers[0],\n * // other layers from the next style\n * ...nextStyle.layers.slice(1).map(layer => {\n * // hide the layers we don't need from demotiles style\n * if (layer.id.startsWith('geolines')) {\n * layer.layout = {...layer.layout || {}, visibility: 'none'};\n * // filter out US polygons\n * } else if (layer.id.startsWith('coastline') || layer.id.startsWith('countries')) {\n * layer.filter = ['!=', ['get', 'ADM0_A3'], 'USA'];\n * }\n * return layer;\n * })\n * ]\n * })\n * });\n * ```\n */\nexport type TransformStyleFunction = (previous: StyleSpecification | undefined, next: StyleSpecification) => StyleSpecification;\n\n/**\n * The options object related to the {@link Map}'s style related methods\n */\nexport type StyleSwapOptions = {\n /**\n * If false, force a 'full' update, removing the current style\n * and building the given one instead of attempting a diff-based update.\n */\n diff?: boolean;\n /**\n * TransformStyleFunction is a convenience function\n * that allows to modify a style after it is fetched but before it is committed to the map state. Refer to {@link TransformStyleFunction}.\n */\n transformStyle?: TransformStyleFunction;\n}\n\n/**\n * Specifies a layer to be added to a {@link Style}. In addition to a standard {@link LayerSpecification}\n * or a {@link CustomLayerInterface}, a {@link LayerSpecification} with an embedded {@link SourceSpecification} can also be provided.\n */\nexport type AddLayerObject = LayerSpecification | (Omit & {source: SourceSpecification}) | CustomLayerInterface;\n\n/**\n * The Style base class\n */\nexport class Style extends Evented {\n map: Map;\n stylesheet: StyleSpecification;\n dispatcher: Dispatcher;\n imageManager: ImageManager;\n glyphManager: GlyphManager;\n lineAtlas: LineAtlas;\n light: Light;\n\n _request: Cancelable;\n _spriteRequest: Cancelable;\n _layers: {[_: string]: StyleLayer};\n _serializedLayers: {[_: string]: LayerSpecification};\n _order: Array;\n sourceCaches: {[_: string]: SourceCache};\n zoomHistory: ZoomHistory;\n _loaded: boolean;\n _rtlTextPluginCallback: (a: any) => any;\n _changed: boolean;\n _updatedSources: {[_: string]: 'clear' | 'reload'};\n _updatedLayers: {[_: string]: true};\n _removedLayers: {[_: string]: StyleLayer};\n _changedImages: {[_: string]: true};\n _glyphsDidChange: boolean;\n _updatedPaintProps: {[layer: string]: true};\n _layerOrderChanged: boolean;\n // image ids of images loaded from style's sprite\n _spritesImagesIds: {[spriteId: string]: string[]};\n // image ids of all images loaded (sprite + user)\n _availableImages: Array;\n\n crossTileSymbolIndex: CrossTileSymbolIndex;\n pauseablePlacement: PauseablePlacement;\n placement: Placement;\n z: number;\n\n static registerForPluginStateChange: typeof registerForPluginStateChange;\n\n constructor(map: Map, options: StyleOptions = {}) {\n super();\n\n this.map = map;\n this.dispatcher = new Dispatcher(getGlobalWorkerPool(), this, map._getMapId());\n this.imageManager = new ImageManager();\n this.imageManager.setEventedParent(this);\n this.glyphManager = new GlyphManager(map._requestManager, options.localIdeographFontFamily);\n this.lineAtlas = new LineAtlas(256, 512);\n this.crossTileSymbolIndex = new CrossTileSymbolIndex();\n\n this._spritesImagesIds = {};\n this._layers = {};\n\n this._order = [];\n this.sourceCaches = {};\n this.zoomHistory = new ZoomHistory();\n this._loaded = false;\n this._availableImages = [];\n\n this._resetUpdates();\n\n this.dispatcher.broadcast('setReferrer', getReferrer());\n\n const self = this;\n this._rtlTextPluginCallback = Style.registerForPluginStateChange((event) => {\n const state = {\n pluginStatus: event.pluginStatus,\n pluginURL: event.pluginURL\n };\n self.dispatcher.broadcast('syncRTLPluginState', state, (err, results) => {\n triggerPluginCompletionEvent(err);\n if (results) {\n const allComplete = results.every((elem) => elem);\n if (allComplete) {\n for (const id in self.sourceCaches) {\n const sourceType = self.sourceCaches[id].getSource().type;\n if (sourceType === 'vector' || sourceType === 'geojson') {\n // Non-vector sources don't have any symbols buckets to reload when the RTL text plugin loads\n // They also load more quickly, so they're more likely to have already displaying tiles\n // that would be unnecessarily booted by the plugin load event\n self.sourceCaches[id].reload(); // Should be a no-op if the plugin loads before any tiles load\n }\n }\n }\n }\n\n });\n });\n\n this.on('data', (event) => {\n if (event.dataType !== 'source' || event.sourceDataType !== 'metadata') {\n return;\n }\n\n const sourceCache = this.sourceCaches[event.sourceId];\n if (!sourceCache) {\n return;\n }\n\n const source = sourceCache.getSource();\n if (!source || !source.vectorLayerIds) {\n return;\n }\n\n for (const layerId in this._layers) {\n const layer = this._layers[layerId];\n if (layer.source === source.id) {\n this._validateLayer(layer);\n }\n }\n });\n }\n\n loadURL(url: string, options: StyleSwapOptions & StyleSetterOptions = {}, previousStyle?: StyleSpecification) {\n this.fire(new Event('dataloading', {dataType: 'style'}));\n\n options.validate = typeof options.validate === 'boolean' ?\n options.validate : true;\n\n const request = this.map._requestManager.transformRequest(url, ResourceType.Style);\n this._request = getJSON(request, (error?: Error | null, json?: any | null) => {\n this._request = null;\n if (error) {\n this.fire(new ErrorEvent(error));\n } else if (json) {\n this._load(json, options, previousStyle);\n }\n });\n }\n\n loadJSON(json: StyleSpecification, options: StyleSetterOptions & StyleSwapOptions = {}, previousStyle?: StyleSpecification) {\n this.fire(new Event('dataloading', {dataType: 'style'}));\n\n this._request = browser.frame(() => {\n this._request = null;\n options.validate = options.validate !== false;\n this._load(json, options, previousStyle);\n });\n }\n\n loadEmpty() {\n this.fire(new Event('dataloading', {dataType: 'style'}));\n this._load(empty, {validate: false});\n }\n\n _load(json: StyleSpecification, options: StyleSwapOptions & StyleSetterOptions, previousStyle?: StyleSpecification) {\n const nextState = options.transformStyle ? options.transformStyle(previousStyle, json) : json;\n if (options.validate && emitValidationErrors(this, validateStyle(nextState))) {\n return;\n }\n\n this._loaded = true;\n this.stylesheet = nextState;\n\n for (const id in nextState.sources) {\n this.addSource(id, nextState.sources[id], {validate: false});\n }\n\n if (nextState.sprite) {\n this._loadSprite(nextState.sprite);\n } else {\n this.imageManager.setLoaded(true);\n }\n\n this.glyphManager.setURL(nextState.glyphs);\n this._createLayers();\n\n this.light = new Light(this.stylesheet.light);\n\n this.map.setTerrain(this.stylesheet.terrain ?? null);\n\n this.fire(new Event('data', {dataType: 'style'}));\n this.fire(new Event('style.load'));\n }\n\n private _createLayers() {\n const dereferencedLayers = deref(this.stylesheet.layers);\n\n // Broadcast layers to workers first, so that expensive style processing (createStyleLayer)\n // can happen in parallel on both main and worker threads.\n this.dispatcher.broadcast('setLayers', dereferencedLayers);\n\n this._order = dereferencedLayers.map((layer) => layer.id);\n this._layers = {};\n\n // reset serialization field, to be populated only when needed\n this._serializedLayers = null;\n for (const layer of dereferencedLayers) {\n const styledLayer = createStyleLayer(layer);\n styledLayer.setEventedParent(this, {layer: {id: layer.id}});\n this._layers[layer.id] = styledLayer;\n }\n }\n\n _loadSprite(sprite: SpriteSpecification, isUpdate: boolean = false, completion: (err: Error) => void = undefined) {\n this.imageManager.setLoaded(false);\n\n this._spriteRequest = loadSprite(sprite, this.map._requestManager, this.map.getPixelRatio(), (err, images) => {\n this._spriteRequest = null;\n if (err) {\n this.fire(new ErrorEvent(err));\n } else if (images) {\n for (const spriteId in images) {\n this._spritesImagesIds[spriteId] = [];\n\n // remove old sprite's loaded images (for the same sprite id) that are not in new sprite\n const imagesToRemove = this._spritesImagesIds[spriteId] ? this._spritesImagesIds[spriteId].filter(id => !(id in images)) : [];\n for (const id of imagesToRemove) {\n this.imageManager.removeImage(id);\n this._changedImages[id] = true;\n }\n\n for (const id in images[spriteId]) {\n // don't prefix images of the \"default\" sprite\n const imageId = spriteId === 'default' ? id : `${spriteId}:${id}`;\n // save all the sprite's images' ids to be able to delete them in `removeSprite`\n this._spritesImagesIds[spriteId].push(imageId);\n if (imageId in this.imageManager.images) {\n this.imageManager.updateImage(imageId, images[spriteId][id], false);\n } else {\n this.imageManager.addImage(imageId, images[spriteId][id]);\n }\n\n if (isUpdate) {\n this._changedImages[imageId] = true;\n }\n }\n }\n }\n\n this.imageManager.setLoaded(true);\n this._availableImages = this.imageManager.listImages();\n\n if (isUpdate) {\n this._changed = true;\n }\n\n this.dispatcher.broadcast('setImages', this._availableImages);\n this.fire(new Event('data', {dataType: 'style'}));\n\n if (completion) {\n completion(err);\n }\n });\n }\n\n _unloadSprite() {\n for (const id of Object.values(this._spritesImagesIds).flat()) {\n this.imageManager.removeImage(id);\n this._changedImages[id] = true;\n }\n\n this._spritesImagesIds = {};\n this._availableImages = this.imageManager.listImages();\n this._changed = true;\n this.dispatcher.broadcast('setImages', this._availableImages);\n this.fire(new Event('data', {dataType: 'style'}));\n }\n\n _validateLayer(layer: StyleLayer) {\n const sourceCache = this.sourceCaches[layer.source];\n if (!sourceCache) {\n return;\n }\n\n const sourceLayer = layer.sourceLayer;\n if (!sourceLayer) {\n return;\n }\n\n const source = sourceCache.getSource();\n if (source.type === 'geojson' || (source.vectorLayerIds && source.vectorLayerIds.indexOf(sourceLayer) === -1)) {\n this.fire(new ErrorEvent(new Error(\n `Source layer \"${sourceLayer}\" ` +\n `does not exist on source \"${source.id}\" ` +\n `as specified by style layer \"${layer.id}\".`\n )));\n }\n }\n\n loaded() {\n if (!this._loaded)\n return false;\n\n if (Object.keys(this._updatedSources).length)\n return false;\n\n for (const id in this.sourceCaches)\n if (!this.sourceCaches[id].loaded())\n return false;\n\n if (!this.imageManager.isLoaded())\n return false;\n\n return true;\n }\n\n /**\n * take an array of string IDs, and based on this._layers, generate an array of LayerSpecification\n * @param ids - an array of string IDs, for which serialized layers will be generated. If omitted, all serialized layers will be returned\n * @returns generated result\n */\n private _serializeByIds(ids?: Array): Array {\n\n const serializedLayersDictionary = this._serializedAllLayers();\n if (!ids || ids.length === 0) {\n return Object.values(serializedLayersDictionary);\n }\n\n const serializedLayers = [];\n for (const id of ids) {\n // this check will skip all custom layers\n if (serializedLayersDictionary[id]) {\n serializedLayers.push(serializedLayersDictionary[id]);\n }\n }\n\n return serializedLayers;\n }\n\n /**\n * Lazy initialization of this._serializedLayers dictionary and return it\n * @returns this._serializedLayers dictionary\n */\n private _serializedAllLayers(): {[_: string]: LayerSpecification} {\n let serializedLayers = this._serializedLayers;\n if (serializedLayers) {\n return serializedLayers;\n }\n\n serializedLayers = this._serializedLayers = {};\n const allLayerIds: string [] = Object.keys(this._layers);\n for (const layerId of allLayerIds) {\n const layer = this._layers[layerId];\n if (layer.type !== 'custom') {\n serializedLayers[layerId] = layer.serialize();\n }\n }\n\n return serializedLayers;\n }\n\n hasTransitions() {\n if (this.light && this.light.hasTransition()) {\n return true;\n }\n\n for (const id in this.sourceCaches) {\n if (this.sourceCaches[id].hasTransition()) {\n return true;\n }\n }\n\n for (const id in this._layers) {\n if (this._layers[id].hasTransition()) {\n return true;\n }\n }\n\n return false;\n }\n\n _checkLoaded() {\n if (!this._loaded) {\n throw new Error('Style is not done loading.');\n }\n }\n\n /**\n * @internal\n * Apply queued style updates in a batch and recalculate zoom-dependent paint properties.\n */\n update(parameters: EvaluationParameters) {\n if (!this._loaded) {\n return;\n }\n\n const changed = this._changed;\n if (this._changed) {\n const updatedIds = Object.keys(this._updatedLayers);\n const removedIds = Object.keys(this._removedLayers);\n\n if (updatedIds.length || removedIds.length) {\n this._updateWorkerLayers(updatedIds, removedIds);\n }\n for (const id in this._updatedSources) {\n const action = this._updatedSources[id];\n\n if (action === 'reload') {\n this._reloadSource(id);\n } else if (action === 'clear') {\n this._clearSource(id);\n } else {\n throw new Error(`Invalid action ${action}`);\n }\n }\n\n this._updateTilesForChangedImages();\n this._updateTilesForChangedGlyphs();\n\n for (const id in this._updatedPaintProps) {\n this._layers[id].updateTransitions(parameters);\n }\n\n this.light.updateTransitions(parameters);\n\n this._resetUpdates();\n }\n\n const sourcesUsedBefore = {};\n\n for (const sourceId in this.sourceCaches) {\n const sourceCache = this.sourceCaches[sourceId];\n sourcesUsedBefore[sourceId] = sourceCache.used;\n sourceCache.used = false;\n }\n\n for (const layerId of this._order) {\n const layer = this._layers[layerId];\n\n layer.recalculate(parameters, this._availableImages);\n if (!layer.isHidden(parameters.zoom) && layer.source) {\n this.sourceCaches[layer.source].used = true;\n }\n }\n\n for (const sourceId in sourcesUsedBefore) {\n const sourceCache = this.sourceCaches[sourceId];\n if (sourcesUsedBefore[sourceId] !== sourceCache.used) {\n sourceCache.fire(new Event('data', {sourceDataType: 'visibility', dataType: 'source', sourceId}));\n }\n }\n\n this.light.recalculate(parameters);\n this.z = parameters.zoom;\n\n if (changed) {\n this.fire(new Event('data', {dataType: 'style'}));\n }\n\n }\n\n /*\n * Apply any queued image changes.\n */\n _updateTilesForChangedImages() {\n const changedImages = Object.keys(this._changedImages);\n if (changedImages.length) {\n for (const name in this.sourceCaches) {\n this.sourceCaches[name].reloadTilesForDependencies(['icons', 'patterns'], changedImages);\n }\n this._changedImages = {};\n }\n }\n\n _updateTilesForChangedGlyphs() {\n if (this._glyphsDidChange) {\n for (const name in this.sourceCaches) {\n this.sourceCaches[name].reloadTilesForDependencies(['glyphs'], ['']);\n }\n this._glyphsDidChange = false;\n }\n }\n\n _updateWorkerLayers(updatedIds: Array, removedIds: Array) {\n this.dispatcher.broadcast('updateLayers', {\n layers: this._serializeByIds(updatedIds),\n removedIds\n });\n }\n\n _resetUpdates() {\n this._changed = false;\n\n this._updatedLayers = {};\n this._removedLayers = {};\n\n this._updatedSources = {};\n this._updatedPaintProps = {};\n\n this._changedImages = {};\n this._glyphsDidChange = false;\n }\n\n /**\n * Update this style's state to match the given style JSON, performing only\n * the necessary mutations.\n *\n * May throw an Error ('Unimplemented: METHOD') if the mapbox-gl-style-spec\n * diff algorithm produces an operation that is not supported.\n *\n * @returns true if any changes were made; false otherwise\n */\n setState(nextState: StyleSpecification, options: StyleSwapOptions = {}) {\n this._checkLoaded();\n\n const serializedStyle = this.serialize();\n nextState = options.transformStyle ? options.transformStyle(serializedStyle, nextState) : nextState;\n if (emitValidationErrors(this, validateStyle(nextState))) return false;\n\n nextState = clone(nextState);\n nextState.layers = deref(nextState.layers);\n\n const changes = diffStyles(serializedStyle, nextState)\n .filter(op => !(op.command in ignoredDiffOperations));\n\n if (changes.length === 0) {\n return false;\n }\n\n const unimplementedOps = changes.filter(op => !(op.command in supportedDiffOperations));\n if (unimplementedOps.length > 0) {\n throw new Error(`Unimplemented: ${unimplementedOps.map(op => op.command).join(', ')}.`);\n }\n\n for (const op of changes) {\n if (op.command === 'setTransition') {\n // `transition` is always read directly off of\n // `this.stylesheet`, which we update below\n continue;\n }\n (this as any)[op.command].apply(this, op.args);\n }\n\n this.stylesheet = nextState;\n\n // reset serialization field, to be populated only when needed\n this._serializedLayers = null;\n\n return true;\n }\n\n addImage(id: string, image: StyleImage) {\n if (this.getImage(id)) {\n return this.fire(new ErrorEvent(new Error(`An image named \"${id}\" already exists.`)));\n }\n this.imageManager.addImage(id, image);\n this._afterImageUpdated(id);\n }\n\n updateImage(id: string, image: StyleImage) {\n this.imageManager.updateImage(id, image);\n }\n\n getImage(id: string): StyleImage {\n return this.imageManager.getImage(id);\n }\n\n removeImage(id: string) {\n if (!this.getImage(id)) {\n return this.fire(new ErrorEvent(new Error(`An image named \"${id}\" does not exist.`)));\n }\n this.imageManager.removeImage(id);\n this._afterImageUpdated(id);\n }\n\n _afterImageUpdated(id: string) {\n this._availableImages = this.imageManager.listImages();\n this._changedImages[id] = true;\n this._changed = true;\n this.dispatcher.broadcast('setImages', this._availableImages);\n this.fire(new Event('data', {dataType: 'style'}));\n }\n\n listImages() {\n this._checkLoaded();\n\n return this.imageManager.listImages();\n }\n\n addSource(id: string, source: SourceSpecification, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n if (this.sourceCaches[id] !== undefined) {\n throw new Error(`Source \"${id}\" already exists.`);\n }\n\n if (!source.type) {\n throw new Error(`The type property must be defined, but only the following properties were given: ${Object.keys(source).join(', ')}.`);\n }\n\n const builtIns = ['vector', 'raster', 'geojson', 'video', 'image'];\n const shouldValidate = builtIns.indexOf(source.type) >= 0;\n if (shouldValidate && this._validate(validateStyle.source, `sources.${id}`, source, null, options)) return;\n\n if (this.map && this.map._collectResourceTiming) (source as any).collectResourceTiming = true;\n const sourceCache = this.sourceCaches[id] = new SourceCache(id, source, this.dispatcher);\n sourceCache.style = this;\n sourceCache.setEventedParent(this, () => ({\n isSourceLoaded: sourceCache.loaded(),\n source: sourceCache.serialize(),\n sourceId: id\n }));\n\n sourceCache.onAdd(this.map);\n this._changed = true;\n }\n\n /**\n * Remove a source from this stylesheet, given its id.\n * @param id - id of the source to remove\n * @throws if no source is found with the given ID\n * @returns `this`.\n */\n removeSource(id: string): this {\n this._checkLoaded();\n\n if (this.sourceCaches[id] === undefined) {\n throw new Error('There is no source with this ID');\n }\n for (const layerId in this._layers) {\n if (this._layers[layerId].source === id) {\n return this.fire(new ErrorEvent(new Error(`Source \"${id}\" cannot be removed while layer \"${layerId}\" is using it.`)));\n }\n }\n\n const sourceCache = this.sourceCaches[id];\n delete this.sourceCaches[id];\n delete this._updatedSources[id];\n sourceCache.fire(new Event('data', {sourceDataType: 'metadata', dataType: 'source', sourceId: id}));\n sourceCache.setEventedParent(null);\n sourceCache.onRemove(this.map);\n this._changed = true;\n }\n\n /**\n * Set the data of a GeoJSON source, given its id.\n * @param id - id of the source\n * @param data - GeoJSON source\n */\n setGeoJSONSourceData(id: string, data: GeoJSON.GeoJSON | string) {\n this._checkLoaded();\n\n if (this.sourceCaches[id] === undefined) throw new Error(`There is no source with this ID=${id}`);\n const geojsonSource: GeoJSONSource = (this.sourceCaches[id].getSource() as any);\n if (geojsonSource.type !== 'geojson') throw new Error(`geojsonSource.type is ${geojsonSource.type}, which is !== 'geojson`);\n\n geojsonSource.setData(data);\n this._changed = true;\n }\n\n /**\n * Get a source by ID.\n * @param id - ID of the desired source\n * @returns source\n */\n getSource(id: string): Source | undefined {\n return this.sourceCaches[id] && this.sourceCaches[id].getSource();\n }\n\n /**\n * Add a layer to the map style. The layer will be inserted before the layer with\n * ID `before`, or appended if `before` is omitted.\n * @param layerObject - The style layer to add.\n * @param before - ID of an existing layer to insert before\n * @param options - Style setter options.\n * @returns `this`.\n */\n addLayer(layerObject: AddLayerObject, before?: string, options: StyleSetterOptions = {}): this {\n this._checkLoaded();\n\n const id = layerObject.id;\n\n if (this.getLayer(id)) {\n this.fire(new ErrorEvent(new Error(`Layer \"${id}\" already exists on this map.`)));\n return;\n }\n\n let layer: ReturnType;\n if (layerObject.type === 'custom') {\n\n if (emitValidationErrors(this, validateCustomStyleLayer(layerObject))) return;\n\n layer = createStyleLayer(layerObject);\n\n } else {\n if ('source' in layerObject && typeof layerObject.source === 'object') {\n this.addSource(id, layerObject.source);\n layerObject = clone(layerObject);\n layerObject = extend(layerObject, {source: id});\n }\n\n // this layer is not in the style.layers array, so we pass an impossible array index\n if (this._validate(validateStyle.layer,\n `layers.${id}`, layerObject, {arrayIndex: -1}, options)) return;\n\n layer = createStyleLayer(layerObject as LayerSpecification | CustomLayerInterface);\n this._validateLayer(layer);\n\n layer.setEventedParent(this, {layer: {id}});\n }\n\n const index = before ? this._order.indexOf(before) : this._order.length;\n if (before && index === -1) {\n this.fire(new ErrorEvent(new Error(`Cannot add layer \"${id}\" before non-existing layer \"${before}\".`)));\n return;\n }\n\n this._order.splice(index, 0, id);\n this._layerOrderChanged = true;\n\n this._layers[id] = layer;\n\n if (this._removedLayers[id] && layer.source && layer.type !== 'custom') {\n // If, in the current batch, we have already removed this layer\n // and we are now re-adding it with a different `type`, then we\n // need to clear (rather than just reload) the underlying source's\n // tiles. Otherwise, tiles marked 'reloading' will have buckets /\n // buffers that are set up for the _previous_ version of this\n // layer, causing, e.g.:\n // https://github.com/mapbox/mapbox-gl-js/issues/3633\n const removed = this._removedLayers[id];\n delete this._removedLayers[id];\n if (removed.type !== layer.type) {\n this._updatedSources[layer.source] = 'clear';\n } else {\n this._updatedSources[layer.source] = 'reload';\n this.sourceCaches[layer.source].pause();\n }\n }\n this._updateLayer(layer);\n\n if (layer.onAdd) {\n layer.onAdd(this.map);\n }\n }\n\n /**\n * Moves a layer to a different z-position. The layer will be inserted before the layer with\n * ID `before`, or appended if `before` is omitted.\n * @param id - ID of the layer to move\n * @param before - ID of an existing layer to insert before\n */\n moveLayer(id: string, before?: string) {\n this._checkLoaded();\n this._changed = true;\n\n const layer = this._layers[id];\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`The layer '${id}' does not exist in the map's style and cannot be moved.`)));\n return;\n }\n\n if (id === before) {\n return;\n }\n\n const index = this._order.indexOf(id);\n this._order.splice(index, 1);\n\n const newIndex = before ? this._order.indexOf(before) : this._order.length;\n if (before && newIndex === -1) {\n this.fire(new ErrorEvent(new Error(`Cannot move layer \"${id}\" before non-existing layer \"${before}\".`)));\n return;\n }\n this._order.splice(newIndex, 0, id);\n\n this._layerOrderChanged = true;\n }\n\n /**\n * Remove the layer with the given id from the style.\n *\n * If no such layer exists, an `error` event is fired.\n *\n * @param id - id of the layer to remove\n * @event `error` - Fired if the layer does not exist\n */\n removeLayer(id: string) {\n this._checkLoaded();\n\n const layer = this._layers[id];\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot remove non-existing layer \"${id}\".`)));\n return;\n }\n\n layer.setEventedParent(null);\n\n const index = this._order.indexOf(id);\n this._order.splice(index, 1);\n\n this._layerOrderChanged = true;\n this._changed = true;\n this._removedLayers[id] = layer;\n delete this._layers[id];\n\n if (this._serializedLayers) {\n delete this._serializedLayers[id];\n }\n delete this._updatedLayers[id];\n delete this._updatedPaintProps[id];\n\n if (layer.onRemove) {\n layer.onRemove(this.map);\n }\n }\n\n /**\n * Return the style layer object with the given `id`.\n *\n * @param id - id of the desired layer\n * @returns a layer, if one with the given `id` exists\n */\n getLayer(id: string): StyleLayer | undefined {\n return this._layers[id];\n }\n\n /**\n * Return the ids of all layers currently in the style, including custom layers, in order.\n *\n * @returns ids of layers, in order\n */\n getLayersOrder(): string[] {\n return [...this._order];\n }\n\n /**\n * Checks if a specific layer is present within the style.\n *\n * @param id - the id of the desired layer\n * @returns a boolean specifying if the given layer is present\n */\n hasLayer(id: string): boolean {\n return id in this._layers;\n }\n\n setLayerZoomRange(layerId: string, minzoom?: number | null, maxzoom?: number | null) {\n this._checkLoaded();\n\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot set the zoom range of non-existing layer \"${layerId}\".`)));\n return;\n }\n\n if (layer.minzoom === minzoom && layer.maxzoom === maxzoom) return;\n\n if (minzoom != null) {\n layer.minzoom = minzoom;\n }\n if (maxzoom != null) {\n layer.maxzoom = maxzoom;\n }\n this._updateLayer(layer);\n }\n\n setFilter(layerId: string, filter?: FilterSpecification | null, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot filter non-existing layer \"${layerId}\".`)));\n return;\n }\n\n if (deepEqual(layer.filter, filter)) {\n return;\n }\n\n if (filter === null || filter === undefined) {\n layer.filter = undefined;\n this._updateLayer(layer);\n return;\n }\n\n if (this._validate(validateStyle.filter, `layers.${layer.id}.filter`, filter, null, options)) {\n return;\n }\n\n layer.filter = clone(filter);\n this._updateLayer(layer);\n }\n\n /**\n * Get a layer's filter object\n * @param layer - the layer to inspect\n * @returns the layer's filter, if any\n */\n getFilter(layer: string): FilterSpecification | void {\n return clone(this.getLayer(layer).filter);\n }\n\n setLayoutProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot style non-existing layer \"${layerId}\".`)));\n return;\n }\n\n if (deepEqual(layer.getLayoutProperty(name), value)) return;\n\n layer.setLayoutProperty(name, value, options);\n this._updateLayer(layer);\n }\n\n /**\n * Get a layout property's value from a given layer\n * @param layerId - the layer to inspect\n * @param name - the name of the layout property\n * @returns the property value\n */\n getLayoutProperty(layerId: string, name: string) {\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot get style of non-existing layer \"${layerId}\".`)));\n return;\n }\n\n return layer.getLayoutProperty(name);\n }\n\n setPaintProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n const layer = this.getLayer(layerId);\n if (!layer) {\n this.fire(new ErrorEvent(new Error(`Cannot style non-existing layer \"${layerId}\".`)));\n return;\n }\n\n if (deepEqual(layer.getPaintProperty(name), value)) return;\n\n const requiresRelayout = layer.setPaintProperty(name, value, options);\n if (requiresRelayout) {\n this._updateLayer(layer);\n }\n\n this._changed = true;\n this._updatedPaintProps[layerId] = true;\n }\n\n getPaintProperty(layer: string, name: string) {\n return this.getLayer(layer).getPaintProperty(name);\n }\n\n setFeatureState(target: FeatureIdentifier, state: any) {\n this._checkLoaded();\n const sourceId = target.source;\n const sourceLayer = target.sourceLayer;\n const sourceCache = this.sourceCaches[sourceId];\n\n if (sourceCache === undefined) {\n this.fire(new ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`)));\n return;\n }\n const sourceType = sourceCache.getSource().type;\n if (sourceType === 'geojson' && sourceLayer) {\n this.fire(new ErrorEvent(new Error('GeoJSON sources cannot have a sourceLayer parameter.')));\n return;\n }\n if (sourceType === 'vector' && !sourceLayer) {\n this.fire(new ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.')));\n return;\n }\n if (target.id === undefined) {\n this.fire(new ErrorEvent(new Error('The feature id parameter must be provided.')));\n }\n\n sourceCache.setFeatureState(sourceLayer, target.id, state);\n }\n\n removeFeatureState(target: FeatureIdentifier, key?: string) {\n this._checkLoaded();\n const sourceId = target.source;\n const sourceCache = this.sourceCaches[sourceId];\n\n if (sourceCache === undefined) {\n this.fire(new ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`)));\n return;\n }\n\n const sourceType = sourceCache.getSource().type;\n const sourceLayer = sourceType === 'vector' ? target.sourceLayer : undefined;\n\n if (sourceType === 'vector' && !sourceLayer) {\n this.fire(new ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.')));\n return;\n }\n\n if (key && (typeof target.id !== 'string' && typeof target.id !== 'number')) {\n this.fire(new ErrorEvent(new Error('A feature id is required to remove its specific state property.')));\n return;\n }\n\n sourceCache.removeFeatureState(sourceLayer, target.id, key);\n }\n\n getFeatureState(target: FeatureIdentifier) {\n this._checkLoaded();\n const sourceId = target.source;\n const sourceLayer = target.sourceLayer;\n const sourceCache = this.sourceCaches[sourceId];\n\n if (sourceCache === undefined) {\n this.fire(new ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`)));\n return;\n }\n const sourceType = sourceCache.getSource().type;\n if (sourceType === 'vector' && !sourceLayer) {\n this.fire(new ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.')));\n return;\n }\n if (target.id === undefined) {\n this.fire(new ErrorEvent(new Error('The feature id parameter must be provided.')));\n }\n\n return sourceCache.getFeatureState(sourceLayer, target.id);\n }\n\n getTransition() {\n return extend({duration: 300, delay: 0}, this.stylesheet && this.stylesheet.transition);\n }\n\n serialize(): StyleSpecification {\n // We return undefined before we're loaded, following the pattern of Map.getStyle() before\n // the Style object is initialized.\n // Internally, Style._validate() calls Style.serialize() but callers are responsible for\n // calling Style._checkLoaded() first if their validation requires the style to be loaded.\n if (!this._loaded) return;\n\n const sources = mapObject(this.sourceCaches, (source) => source.serialize());\n const layers = this._serializeByIds(this._order);\n const terrain = this.map.getTerrain() || undefined;\n const myStyleSheet = this.stylesheet;\n\n return filterObject({\n version: myStyleSheet.version,\n name: myStyleSheet.name,\n metadata: myStyleSheet.metadata,\n light: myStyleSheet.light,\n center: myStyleSheet.center,\n zoom: myStyleSheet.zoom,\n bearing: myStyleSheet.bearing,\n pitch: myStyleSheet.pitch,\n sprite: myStyleSheet.sprite,\n glyphs: myStyleSheet.glyphs,\n transition: myStyleSheet.transition,\n sources,\n layers,\n terrain\n },\n (value) => { return value !== undefined; });\n }\n\n _updateLayer(layer: StyleLayer) {\n this._updatedLayers[layer.id] = true;\n if (layer.source && !this._updatedSources[layer.source] &&\n //Skip for raster layers (https://github.com/mapbox/mapbox-gl-js/issues/7865)\n this.sourceCaches[layer.source].getSource().type !== 'raster') {\n this._updatedSources[layer.source] = 'reload';\n this.sourceCaches[layer.source].pause();\n }\n\n // upon updating, serilized layer dictionary should be reset.\n // When needed, it will be populated with the correct copy again.\n this._serializedLayers = null;\n this._changed = true;\n }\n\n _flattenAndSortRenderedFeatures(sourceResults: Array<{ [key: string]: Array<{featureIndex: number; feature: MapGeoJSONFeature}> }>) {\n // Feature order is complicated.\n // The order between features in two 2D layers is always determined by layer order.\n // The order between features in two 3D layers is always determined by depth.\n // The order between a feature in a 2D layer and a 3D layer is tricky:\n // Most often layer order determines the feature order in this case. If\n // a line layer is above a extrusion layer the line feature will be rendered\n // above the extrusion. If the line layer is below the extrusion layer,\n // it will be rendered below it.\n //\n // There is a weird case though.\n // You have layers in this order: extrusion_layer_a, line_layer, extrusion_layer_b\n // Each layer has a feature that overlaps the other features.\n // The feature in extrusion_layer_a is closer than the feature in extrusion_layer_b so it is rendered above.\n // The feature in line_layer is rendered above extrusion_layer_a.\n // This means that that the line_layer feature is above the extrusion_layer_b feature despite\n // it being in an earlier layer.\n\n const isLayer3D = layerId => this._layers[layerId].type === 'fill-extrusion';\n\n const layerIndex = {};\n const features3D = [];\n for (let l = this._order.length - 1; l >= 0; l--) {\n const layerId = this._order[l];\n if (isLayer3D(layerId)) {\n layerIndex[layerId] = l;\n for (const sourceResult of sourceResults) {\n const layerFeatures = sourceResult[layerId];\n if (layerFeatures) {\n for (const featureWrapper of layerFeatures) {\n features3D.push(featureWrapper);\n }\n }\n }\n }\n }\n\n features3D.sort((a, b) => {\n return b.intersectionZ - a.intersectionZ;\n });\n\n const features = [];\n for (let l = this._order.length - 1; l >= 0; l--) {\n const layerId = this._order[l];\n\n if (isLayer3D(layerId)) {\n // add all 3D features that are in or above the current layer\n for (let i = features3D.length - 1; i >= 0; i--) {\n const topmost3D = features3D[i].feature;\n if (layerIndex[topmost3D.layer.id] < l) break;\n features.push(topmost3D);\n features3D.pop();\n }\n } else {\n for (const sourceResult of sourceResults) {\n const layerFeatures = sourceResult[layerId];\n if (layerFeatures) {\n for (const featureWrapper of layerFeatures) {\n features.push(featureWrapper.feature);\n }\n }\n }\n }\n }\n\n return features;\n }\n\n queryRenderedFeatures(queryGeometry: any, params: QueryRenderedFeaturesOptions, transform: Transform) {\n if (params && params.filter) {\n this._validate(validateStyle.filter, 'queryRenderedFeatures.filter', params.filter, null, params);\n }\n\n const includedSources = {};\n if (params && params.layers) {\n if (!Array.isArray(params.layers)) {\n this.fire(new ErrorEvent(new Error('parameters.layers must be an Array.')));\n return [];\n }\n for (const layerId of params.layers) {\n const layer = this._layers[layerId];\n if (!layer) {\n // this layer is not in the style.layers array\n this.fire(new ErrorEvent(new Error(`The layer '${layerId}' does not exist in the map's style and cannot be queried for features.`)));\n return [];\n }\n includedSources[layer.source] = true;\n }\n }\n\n const sourceResults = [];\n\n params.availableImages = this._availableImages;\n\n // LayerSpecification is serialized StyleLayer, and this casting is safe.\n const serializedLayers = this._serializedAllLayers() as {[_: string]: StyleLayer};\n\n for (const id in this.sourceCaches) {\n if (params.layers && !includedSources[id]) continue;\n sourceResults.push(\n queryRenderedFeatures(\n this.sourceCaches[id],\n this._layers,\n serializedLayers,\n queryGeometry,\n params,\n transform)\n );\n }\n\n if (this.placement) {\n // If a placement has run, query against its CollisionIndex\n // for symbol results, and treat it as an extra source to merge\n sourceResults.push(\n queryRenderedSymbols(\n this._layers,\n serializedLayers,\n this.sourceCaches,\n queryGeometry,\n params,\n this.placement.collisionIndex,\n this.placement.retainedQueryData)\n );\n }\n\n return this._flattenAndSortRenderedFeatures(sourceResults);\n }\n\n querySourceFeatures(\n sourceID: string,\n params?: QuerySourceFeatureOptions\n ) {\n if (params && params.filter) {\n this._validate(validateStyle.filter, 'querySourceFeatures.filter', params.filter, null, params);\n }\n const sourceCache = this.sourceCaches[sourceID];\n return sourceCache ? querySourceFeatures(sourceCache, params) : [];\n }\n\n addSourceType(name: string, SourceType: SourceClass, callback: Callback) {\n if (getSourceType(name)) {\n return callback(new Error(`A source type called \"${name}\" already exists.`));\n }\n\n setSourceType(name, SourceType);\n\n if (!SourceType.workerSourceURL) {\n return callback(null, null);\n }\n\n this.dispatcher.broadcast('loadWorkerSource', {\n name,\n url: SourceType.workerSourceURL\n }, callback);\n }\n\n getLight() {\n return this.light.getLight();\n }\n\n setLight(lightOptions: LightSpecification, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n\n const light = this.light.getLight();\n let _update = false;\n for (const key in lightOptions) {\n if (!deepEqual(lightOptions[key], light[key])) {\n _update = true;\n break;\n }\n }\n if (!_update) return;\n\n const parameters = {\n now: browser.now(),\n transition: extend({\n duration: 300,\n delay: 0\n }, this.stylesheet.transition)\n };\n\n this.light.setLight(lightOptions, options);\n this.light.updateTransitions(parameters);\n }\n\n _validate(validate: Validator, key: string, value: any, props: any, options: {\n validate?: boolean;\n } = {}) {\n if (options && options.validate === false) {\n return false;\n }\n return emitValidationErrors(this, validate.call(validateStyle, extend({\n key,\n style: this.serialize(),\n value,\n styleSpec\n }, props)));\n }\n\n _remove(mapRemoved: boolean = true) {\n if (this._request) {\n this._request.cancel();\n this._request = null;\n }\n if (this._spriteRequest) {\n this._spriteRequest.cancel();\n this._spriteRequest = null;\n }\n rtlTextPluginEvented.off('pluginStateChange', this._rtlTextPluginCallback);\n for (const layerId in this._layers) {\n const layer: StyleLayer = this._layers[layerId];\n layer.setEventedParent(null);\n }\n for (const id in this.sourceCaches) {\n const sourceCache = this.sourceCaches[id];\n sourceCache.setEventedParent(null);\n sourceCache.onRemove(this.map);\n }\n this.imageManager.setEventedParent(null);\n this.setEventedParent(null);\n this.dispatcher.remove(mapRemoved);\n }\n\n _clearSource(id: string) {\n this.sourceCaches[id].clearTiles();\n }\n\n _reloadSource(id: string) {\n this.sourceCaches[id].resume();\n this.sourceCaches[id].reload();\n }\n\n _updateSources(transform: Transform) {\n for (const id in this.sourceCaches) {\n this.sourceCaches[id].update(transform, this.map.terrain);\n }\n }\n\n _generateCollisionBoxes() {\n for (const id in this.sourceCaches) {\n this._reloadSource(id);\n }\n }\n\n _updatePlacement(transform: Transform, showCollisionBoxes: boolean, fadeDuration: number, crossSourceCollisions: boolean, forceFullPlacement: boolean = false) {\n let symbolBucketsChanged = false;\n let placementCommitted = false;\n\n const layerTiles = {};\n\n for (const layerID of this._order) {\n const styleLayer = this._layers[layerID];\n if (styleLayer.type !== 'symbol') continue;\n\n if (!layerTiles[styleLayer.source]) {\n const sourceCache = this.sourceCaches[styleLayer.source];\n layerTiles[styleLayer.source] = sourceCache.getRenderableIds(true)\n .map((id) => sourceCache.getTileByID(id))\n .sort((a, b) => (b.tileID.overscaledZ - a.tileID.overscaledZ) || (a.tileID.isLessThan(b.tileID) ? -1 : 1));\n }\n\n const layerBucketsChanged = this.crossTileSymbolIndex.addLayer(styleLayer, layerTiles[styleLayer.source], transform.center.lng);\n symbolBucketsChanged = symbolBucketsChanged || layerBucketsChanged;\n }\n this.crossTileSymbolIndex.pruneUnusedLayers(this._order);\n\n // Anything that changes our \"in progress\" layer and tile indices requires us\n // to start over. When we start over, we do a full placement instead of incremental\n // to prevent starvation.\n // We need to restart placement to keep layer indices in sync.\n // Also force full placement when fadeDuration === 0 to ensure that newly loaded\n // tiles will fully display symbols in their first frame\n forceFullPlacement = forceFullPlacement || this._layerOrderChanged || fadeDuration === 0;\n\n if (forceFullPlacement || !this.pauseablePlacement || (this.pauseablePlacement.isDone() && !this.placement.stillRecent(browser.now(), transform.zoom))) {\n this.pauseablePlacement = new PauseablePlacement(transform, this.map.terrain, this._order, forceFullPlacement, showCollisionBoxes, fadeDuration, crossSourceCollisions, this.placement);\n this._layerOrderChanged = false;\n }\n\n if (this.pauseablePlacement.isDone()) {\n // the last placement finished running, but the next one hasn’t\n // started yet because of the `stillRecent` check immediately\n // above, so mark it stale to ensure that we request another\n // render frame\n this.placement.setStale();\n } else {\n this.pauseablePlacement.continuePlacement(this._order, this._layers, layerTiles);\n\n if (this.pauseablePlacement.isDone()) {\n this.placement = this.pauseablePlacement.commit(browser.now());\n placementCommitted = true;\n }\n\n if (symbolBucketsChanged) {\n // since the placement gets split over multiple frames it is possible\n // these buckets were processed before they were changed and so the\n // placement is already stale while it is in progress\n this.pauseablePlacement.placement.setStale();\n }\n }\n\n if (placementCommitted || symbolBucketsChanged) {\n for (const layerID of this._order) {\n const styleLayer = this._layers[layerID];\n if (styleLayer.type !== 'symbol') continue;\n this.placement.updateLayerOpacities(styleLayer, layerTiles[styleLayer.source]);\n }\n }\n\n // needsRender is false when we have just finished a placement that didn't change the visibility of any symbols\n const needsRerender = !this.pauseablePlacement.isDone() || this.placement.hasTransitions(browser.now());\n return needsRerender;\n }\n\n _releaseSymbolFadeTiles() {\n for (const id in this.sourceCaches) {\n this.sourceCaches[id].releaseSymbolFadeTiles();\n }\n }\n\n // Callbacks from web workers\n\n getImages(\n mapId: string,\n params: {\n icons: Array;\n source: string;\n tileID: OverscaledTileID;\n type: string;\n },\n callback: Callback<{[_: string]: StyleImage}>\n ) {\n this.imageManager.getImages(params.icons, callback);\n\n // Apply queued image changes before setting the tile's dependencies so that the tile\n // is not reloaded unnecessarily. Without this forced update the reload could happen in cases\n // like this one:\n // - icons contains \"my-image\"\n // - imageManager.getImages(...) triggers `onstyleimagemissing`\n // - the user adds \"my-image\" within the callback\n // - addImage adds \"my-image\" to this._changedImages\n // - the next frame triggers a reload of this tile even though it already has the latest version\n this._updateTilesForChangedImages();\n\n const sourceCache = this.sourceCaches[params.source];\n if (sourceCache) {\n sourceCache.setDependencies(params.tileID.key, params.type, params.icons);\n }\n }\n\n getGlyphs(\n mapId: string,\n params: {\n stacks: {[_: string]: Array};\n source: string;\n tileID: OverscaledTileID;\n type: string;\n },\n callback: Callback<{[_: string]: {[_: number]: StyleGlyph}}>\n ) {\n this.glyphManager.getGlyphs(params.stacks, callback);\n const sourceCache = this.sourceCaches[params.source];\n if (sourceCache) {\n // we are not setting stacks as dependencies since for now\n // we just need to know which tiles have glyph dependencies\n sourceCache.setDependencies(params.tileID.key, params.type, ['']);\n }\n }\n\n getResource(mapId: string, params: RequestParameters, callback: ResponseCallback): Cancelable {\n return makeRequest(params, callback);\n }\n\n getGlyphsUrl() {\n return this.stylesheet.glyphs || null;\n }\n\n setGlyphs(glyphsUrl: string | null, options: StyleSetterOptions = {}) {\n this._checkLoaded();\n if (glyphsUrl && this._validate(validateStyle.glyphs, 'glyphs', glyphsUrl, null, options)) {\n return;\n }\n\n this._glyphsDidChange = true;\n this.stylesheet.glyphs = glyphsUrl;\n this.glyphManager.entries = {};\n this.glyphManager.setURL(glyphsUrl);\n }\n\n /**\n * Add a sprite.\n *\n * @param id - The id of the desired sprite\n * @param url - The url to load the desired sprite from\n * @param options - The style setter options\n * @param completion - The completion handler\n */\n addSprite(id: string, url: string, options: StyleSetterOptions = {}, completion?: (err: Error) => void) {\n this._checkLoaded();\n\n const spriteToAdd = [{id, url}];\n const updatedSprite = [\n ...coerceSpriteToArray(this.stylesheet.sprite),\n ...spriteToAdd\n ];\n\n if (this._validate(validateStyle.sprite, 'sprite', updatedSprite, null, options)) return;\n\n this.stylesheet.sprite = updatedSprite;\n this._loadSprite(spriteToAdd, true, completion);\n }\n\n /**\n * Remove a sprite by its id. When the last sprite is removed, the whole `this.stylesheet.sprite` object becomes\n * `undefined`. This falsy `undefined` value later prevents attempts to load the sprite when it's absent.\n *\n * @param id - the id of the sprite to remove\n */\n removeSprite(id: string) {\n this._checkLoaded();\n\n const internalSpriteRepresentation = coerceSpriteToArray(this.stylesheet.sprite);\n\n if (!internalSpriteRepresentation.find(sprite => sprite.id === id)) {\n this.fire(new ErrorEvent(new Error(`Sprite \"${id}\" doesn't exists on this map.`)));\n return;\n }\n\n if (this._spritesImagesIds[id]) {\n for (const imageId of this._spritesImagesIds[id]) {\n this.imageManager.removeImage(imageId);\n this._changedImages[imageId] = true;\n }\n }\n\n internalSpriteRepresentation.splice(internalSpriteRepresentation.findIndex(sprite => sprite.id === id), 1);\n this.stylesheet.sprite = internalSpriteRepresentation.length > 0 ? internalSpriteRepresentation : undefined;\n\n delete this._spritesImagesIds[id];\n this._availableImages = this.imageManager.listImages();\n this._changed = true;\n this.dispatcher.broadcast('setImages', this._availableImages);\n this.fire(new Event('data', {dataType: 'style'}));\n }\n\n /**\n * Get the current sprite value.\n *\n * @returns empty array when no sprite is set; id-url pairs otherwise\n */\n getSprite() {\n return coerceSpriteToArray(this.stylesheet.sprite);\n }\n\n /**\n * Set a new value for the style's sprite.\n *\n * @param sprite - new sprite value\n * @param options - style setter options\n * @param completion - the completion handler\n */\n setSprite(sprite: SpriteSpecification, options: StyleSetterOptions = {}, completion?: (err: Error) => void) {\n this._checkLoaded();\n\n if (sprite && this._validate(validateStyle.sprite, 'sprite', sprite, null, options)) {\n return;\n }\n\n this.stylesheet.sprite = sprite;\n\n if (sprite) {\n this._loadSprite(sprite, true, completion);\n } else {\n this._unloadSprite();\n if (completion) {\n completion(null);\n }\n }\n }\n}\n\nStyle.registerForPluginStateChange = registerForPluginStateChange;\n","import {createLayout} from '../util/struct_array';\n\nexport default createLayout([\n {name: 'a_pos', type: 'Int16', components: 2}\n]);\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec3 a_pos3d;uniform mat4 u_matrix;uniform float u_ele_delta;varying vec2 v_texture_pos;varying float v_depth;void main() {float extent=8192.0;float ele_delta=a_pos3d.z==1.0 ? u_ele_delta : 0.0;v_texture_pos=a_pos3d.xy/extent;gl_Position=u_matrix*vec4(a_pos3d.xy,get_elevation(a_pos3d.xy)-ele_delta,1.0);v_depth=gl_Position.z/gl_Position.w;}';\n","\n// Disable Flow annotations here because Flow doesn't support importing GLSL files\n\nimport preludeFrag from './_prelude.fragment.glsl.g';\nimport preludeVert from './_prelude.vertex.glsl.g';\nimport backgroundFrag from './background.fragment.glsl.g';\nimport backgroundVert from './background.vertex.glsl.g';\nimport backgroundPatternFrag from './background_pattern.fragment.glsl.g';\nimport backgroundPatternVert from './background_pattern.vertex.glsl.g';\nimport circleFrag from './circle.fragment.glsl.g';\nimport circleVert from './circle.vertex.glsl.g';\nimport clippingMaskFrag from './clipping_mask.fragment.glsl.g';\nimport clippingMaskVert from './clipping_mask.vertex.glsl.g';\nimport heatmapFrag from './heatmap.fragment.glsl.g';\nimport heatmapVert from './heatmap.vertex.glsl.g';\nimport heatmapTextureFrag from './heatmap_texture.fragment.glsl.g';\nimport heatmapTextureVert from './heatmap_texture.vertex.glsl.g';\nimport collisionBoxFrag from './collision_box.fragment.glsl.g';\nimport collisionBoxVert from './collision_box.vertex.glsl.g';\nimport collisionCircleFrag from './collision_circle.fragment.glsl.g';\nimport collisionCircleVert from './collision_circle.vertex.glsl.g';\nimport debugFrag from './debug.fragment.glsl.g';\nimport debugVert from './debug.vertex.glsl.g';\nimport fillFrag from './fill.fragment.glsl.g';\nimport fillVert from './fill.vertex.glsl.g';\nimport fillOutlineFrag from './fill_outline.fragment.glsl.g';\nimport fillOutlineVert from './fill_outline.vertex.glsl.g';\nimport fillOutlinePatternFrag from './fill_outline_pattern.fragment.glsl.g';\nimport fillOutlinePatternVert from './fill_outline_pattern.vertex.glsl.g';\nimport fillPatternFrag from './fill_pattern.fragment.glsl.g';\nimport fillPatternVert from './fill_pattern.vertex.glsl.g';\nimport fillExtrusionFrag from './fill_extrusion.fragment.glsl.g';\nimport fillExtrusionVert from './fill_extrusion.vertex.glsl.g';\nimport fillExtrusionPatternFrag from './fill_extrusion_pattern.fragment.glsl.g';\nimport fillExtrusionPatternVert from './fill_extrusion_pattern.vertex.glsl.g';\nimport hillshadePrepareFrag from './hillshade_prepare.fragment.glsl.g';\nimport hillshadePrepareVert from './hillshade_prepare.vertex.glsl.g';\nimport hillshadeFrag from './hillshade.fragment.glsl.g';\nimport hillshadeVert from './hillshade.vertex.glsl.g';\nimport lineFrag from './line.fragment.glsl.g';\nimport lineVert from './line.vertex.glsl.g';\nimport lineGradientFrag from './line_gradient.fragment.glsl.g';\nimport lineGradientVert from './line_gradient.vertex.glsl.g';\nimport linePatternFrag from './line_pattern.fragment.glsl.g';\nimport linePatternVert from './line_pattern.vertex.glsl.g';\nimport lineSDFFrag from './line_sdf.fragment.glsl.g';\nimport lineSDFVert from './line_sdf.vertex.glsl.g';\nimport rasterFrag from './raster.fragment.glsl.g';\nimport rasterVert from './raster.vertex.glsl.g';\nimport symbolIconFrag from './symbol_icon.fragment.glsl.g';\nimport symbolIconVert from './symbol_icon.vertex.glsl.g';\nimport symbolSDFFrag from './symbol_sdf.fragment.glsl.g';\nimport symbolSDFVert from './symbol_sdf.vertex.glsl.g';\nimport symbolTextAndIconFrag from './symbol_text_and_icon.fragment.glsl.g';\nimport symbolTextAndIconVert from './symbol_text_and_icon.vertex.glsl.g';\nimport terrainDepthFrag from './terrain_depth.fragment.glsl.g';\nimport terrainCoordsFrag from './terrain_coords.fragment.glsl.g';\nimport terrainFrag from './terrain.fragment.glsl.g';\nimport terrainVert from './terrain.vertex.glsl.g';\n\nexport const shaders = {\n prelude: compile(preludeFrag, preludeVert),\n background: compile(backgroundFrag, backgroundVert),\n backgroundPattern: compile(backgroundPatternFrag, backgroundPatternVert),\n circle: compile(circleFrag, circleVert),\n clippingMask: compile(clippingMaskFrag, clippingMaskVert),\n heatmap: compile(heatmapFrag, heatmapVert),\n heatmapTexture: compile(heatmapTextureFrag, heatmapTextureVert),\n collisionBox: compile(collisionBoxFrag, collisionBoxVert),\n collisionCircle: compile(collisionCircleFrag, collisionCircleVert),\n debug: compile(debugFrag, debugVert),\n fill: compile(fillFrag, fillVert),\n fillOutline: compile(fillOutlineFrag, fillOutlineVert),\n fillOutlinePattern: compile(fillOutlinePatternFrag, fillOutlinePatternVert),\n fillPattern: compile(fillPatternFrag, fillPatternVert),\n fillExtrusion: compile(fillExtrusionFrag, fillExtrusionVert),\n fillExtrusionPattern: compile(fillExtrusionPatternFrag, fillExtrusionPatternVert),\n hillshadePrepare: compile(hillshadePrepareFrag, hillshadePrepareVert),\n hillshade: compile(hillshadeFrag, hillshadeVert),\n line: compile(lineFrag, lineVert),\n lineGradient: compile(lineGradientFrag, lineGradientVert),\n linePattern: compile(linePatternFrag, linePatternVert),\n lineSDF: compile(lineSDFFrag, lineSDFVert),\n raster: compile(rasterFrag, rasterVert),\n symbolIcon: compile(symbolIconFrag, symbolIconVert),\n symbolSDF: compile(symbolSDFFrag, symbolSDFVert),\n symbolTextAndIcon: compile(symbolTextAndIconFrag, symbolTextAndIconVert),\n terrain: compile(terrainFrag, terrainVert),\n terrainDepth: compile(terrainDepthFrag, terrainVert),\n terrainCoords: compile(terrainCoordsFrag, terrainVert)\n};\n\n// Expand #pragmas to #ifdefs.\n\nfunction compile(fragmentSource, vertexSource) {\n const re = /#pragma mapbox: ([\\w]+) ([\\w]+) ([\\w]+) ([\\w]+)/g;\n\n const staticAttributes = vertexSource.match(/attribute ([\\w]+) ([\\w]+)/g);\n const fragmentUniforms = fragmentSource.match(/uniform ([\\w]+) ([\\w]+)([\\s]*)([\\w]*)/g);\n const vertexUniforms = vertexSource.match(/uniform ([\\w]+) ([\\w]+)([\\s]*)([\\w]*)/g);\n const staticUniforms = vertexUniforms ? vertexUniforms.concat(fragmentUniforms) : fragmentUniforms;\n\n const fragmentPragmas = {};\n\n fragmentSource = fragmentSource.replace(re, (match, operation, precision, type, name) => {\n fragmentPragmas[name] = true;\n if (operation === 'define') {\n return `\n#ifndef HAS_UNIFORM_u_${name}\nvarying ${precision} ${type} ${name};\n#else\nuniform ${precision} ${type} u_${name};\n#endif\n`;\n } else /* if (operation === 'initialize') */ {\n return `\n#ifdef HAS_UNIFORM_u_${name}\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n }\n });\n\n vertexSource = vertexSource.replace(re, (match, operation, precision, type, name) => {\n const attrType = type === 'float' ? 'vec2' : 'vec4';\n const unpackType = name.match(/color/) ? 'color' : attrType;\n\n if (fragmentPragmas[name]) {\n if (operation === 'define') {\n return `\n#ifndef HAS_UNIFORM_u_${name}\nuniform lowp float u_${name}_t;\nattribute ${precision} ${attrType} a_${name};\nvarying ${precision} ${type} ${name};\n#else\nuniform ${precision} ${type} u_${name};\n#endif\n`;\n } else /* if (operation === 'initialize') */ {\n if (unpackType === 'vec4') {\n // vec4 attributes are only used for cross-faded properties, and are not packed\n return `\n#ifndef HAS_UNIFORM_u_${name}\n ${name} = a_${name};\n#else\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n } else {\n return `\n#ifndef HAS_UNIFORM_u_${name}\n ${name} = unpack_mix_${unpackType}(a_${name}, u_${name}_t);\n#else\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n }\n }\n } else {\n if (operation === 'define') {\n return `\n#ifndef HAS_UNIFORM_u_${name}\nuniform lowp float u_${name}_t;\nattribute ${precision} ${attrType} a_${name};\n#else\nuniform ${precision} ${type} u_${name};\n#endif\n`;\n } else /* if (operation === 'initialize') */ {\n if (unpackType === 'vec4') {\n // vec4 attributes are only used for cross-faded properties, and are not packed\n return `\n#ifndef HAS_UNIFORM_u_${name}\n ${precision} ${type} ${name} = a_${name};\n#else\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n } else /* */ {\n return `\n#ifndef HAS_UNIFORM_u_${name}\n ${precision} ${type} ${name} = unpack_mix_${unpackType}(a_${name}, u_${name}_t);\n#else\n ${precision} ${type} ${name} = u_${name};\n#endif\n`;\n }\n }\n }\n });\n\n return {fragmentSource, vertexSource, staticAttributes, staticUniforms};\n}\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision mediump float;\\n#else\\n#if !defined(lowp)\\n#define lowp\\n#endif\\n#if !defined(mediump)\\n#define mediump\\n#endif\\n#if !defined(highp)\\n#define highp\\n#endif\\n#endif\\n';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision highp float;\\n#else\\n#if !defined(lowp)\\n#define lowp\\n#endif\\n#if !defined(mediump)\\n#define mediump\\n#endif\\n#if !defined(highp)\\n#define highp\\n#endif\\n#endif\\nvec2 unpack_float(const float packedValue) {int packedIntValue=int(packedValue);int v0=packedIntValue/256;return vec2(v0,packedIntValue-v0*256);}vec2 unpack_opacity(const float packedOpacity) {int intOpacity=int(packedOpacity)/2;return vec2(float(intOpacity)/127.0,mod(packedOpacity,2.0));}vec4 decode_color(const vec2 encodedColor) {return vec4(unpack_float(encodedColor[0])/255.0,unpack_float(encodedColor[1])/255.0\\n);}float unpack_mix_vec2(const vec2 packedValue,const float t) {return mix(packedValue[0],packedValue[1],t);}vec4 unpack_mix_color(const vec4 packedColors,const float t) {vec4 minColor=decode_color(vec2(packedColors[0],packedColors[1]));vec4 maxColor=decode_color(vec2(packedColors[2],packedColors[3]));return mix(minColor,maxColor,t);}vec2 get_pattern_pos(const vec2 pixel_coord_upper,const vec2 pixel_coord_lower,const vec2 pattern_size,const float tile_units_to_pixels,const vec2 pos) {vec2 offset=mod(mod(mod(pixel_coord_upper,pattern_size)*256.0,pattern_size)*256.0+pixel_coord_lower,pattern_size);return (tile_units_to_pixels*pos+offset)/pattern_size;}\\n#ifdef TERRAIN3D\\nuniform sampler2D u_terrain;uniform float u_terrain_dim;uniform mat4 u_terrain_matrix;uniform vec4 u_terrain_unpack;uniform float u_terrain_exaggeration;uniform highp sampler2D u_depth;\\n#endif\\nconst highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitShifts=vec4(1.)/bitSh;highp float unpack(highp vec4 color) {return dot(color,bitShifts);}highp float depthOpacity(vec3 frag) {\\n#ifdef TERRAIN3D\\nhighp float d=unpack(texture2D(u_depth,frag.xy*0.5+0.5))+0.0001-frag.z;return 1.0-max(0.0,min(1.0,-d*500.0));\\n#else\\nreturn 1.0;\\n#endif\\n}float calculate_visibility(vec4 pos) {\\n#ifdef TERRAIN3D\\nvec3 frag=pos.xyz/pos.w;highp float d=depthOpacity(frag);if (d > 0.95) return 1.0;return (d+depthOpacity(frag+vec3(0.0,0.01,0.0)))/2.0;\\n#else\\nreturn 1.0;\\n#endif\\n}float ele(vec2 pos) {\\n#ifdef TERRAIN3D\\nvec4 rgb=(texture2D(u_terrain,pos)*255.0)*u_terrain_unpack;return rgb.r+rgb.g+rgb.b-u_terrain_unpack.a;\\n#else\\nreturn 0.0;\\n#endif\\n}float get_elevation(vec2 pos) {\\n#ifdef TERRAIN3D\\nvec2 coord=(u_terrain_matrix*vec4(pos,0.0,1.0)).xy*u_terrain_dim+1.0;vec2 f=fract(coord);vec2 c=(floor(coord)+0.5)/(u_terrain_dim+2.0);float d=1.0/(u_terrain_dim+2.0);float tl=ele(c);float tr=ele(c+vec2(d,0.0));float bl=ele(c+vec2(0.0,d));float br=ele(c+vec2(d,d));float elevation=mix(mix(tl,tr,f.x),mix(bl,br,f.x),f.y);return elevation*u_terrain_exaggeration;\\n#else\\nreturn 0.0;\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform vec4 u_color;uniform float u_opacity;void main() {gl_FragColor=u_color*u_opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform vec2 u_pattern_tl_a;uniform vec2 u_pattern_br_a;uniform vec2 u_pattern_tl_b;uniform vec2 u_pattern_br_b;uniform vec2 u_texsize;uniform float u_mix;uniform float u_opacity;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(u_pattern_tl_a/u_texsize,u_pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(u_pattern_tl_b/u_texsize,u_pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_mix)*u_opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_pattern_size_a;uniform vec2 u_pattern_size_b;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_scale_a;uniform float u_scale_b;uniform float u_tile_units_to_pixels;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_a*u_pattern_size_a,u_tile_units_to_pixels,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_b*u_pattern_size_b,u_tile_units_to_pixels,a_pos);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying vec3 v_data;varying float v_visibility;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define mediump float radius\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define highp vec4 stroke_color\\n#pragma mapbox: define mediump float stroke_width\\n#pragma mapbox: define lowp float stroke_opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize mediump float radius\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize highp vec4 stroke_color\\n#pragma mapbox: initialize mediump float stroke_width\\n#pragma mapbox: initialize lowp float stroke_opacity\\nvec2 extrude=v_data.xy;float extrude_length=length(extrude);lowp float antialiasblur=v_data.z;float antialiased_blur=-max(blur,antialiasblur);float opacity_t=smoothstep(0.0,antialiased_blur,extrude_length-1.0);float color_t=stroke_width < 0.01 ? 0.0 : smoothstep(antialiased_blur,0.0,extrude_length-radius/(radius+stroke_width));gl_FragColor=v_visibility*opacity_t*mix(color*opacity,stroke_color*stroke_opacity,color_t);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform bool u_scale_with_map;uniform bool u_pitch_with_map;uniform vec2 u_extrude_scale;uniform lowp float u_device_pixel_ratio;uniform highp float u_camera_to_center_distance;attribute vec2 a_pos;varying vec3 v_data;varying float v_visibility;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define mediump float radius\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define highp vec4 stroke_color\\n#pragma mapbox: define mediump float stroke_width\\n#pragma mapbox: define lowp float stroke_opacity\\nvoid main(void) {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize mediump float radius\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize highp vec4 stroke_color\\n#pragma mapbox: initialize mediump float stroke_width\\n#pragma mapbox: initialize lowp float stroke_opacity\\nvec2 extrude=vec2(mod(a_pos,2.0)*2.0-1.0);vec2 circle_center=floor(a_pos*0.5);float ele=get_elevation(circle_center);v_visibility=calculate_visibility(u_matrix*vec4(circle_center,ele,1.0));if (u_pitch_with_map) {vec2 corner_position=circle_center;if (u_scale_with_map) {corner_position+=extrude*(radius+stroke_width)*u_extrude_scale;} else {vec4 projected_center=u_matrix*vec4(circle_center,0,1);corner_position+=extrude*(radius+stroke_width)*u_extrude_scale*(projected_center.w/u_camera_to_center_distance);}gl_Position=u_matrix*vec4(corner_position,ele,1);} else {gl_Position=u_matrix*vec4(circle_center,ele,1);if (u_scale_with_map) {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*u_camera_to_center_distance;} else {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*gl_Position.w;}}lowp float antialiasblur=1.0/u_device_pixel_ratio/(radius+stroke_width);v_data=vec3(extrude.x,extrude.y,antialiasblur);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'void main() {gl_FragColor=vec4(1.0);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform highp float u_intensity;varying vec2 v_extrude;\\n#pragma mapbox: define highp float weight\\n#define GAUSS_COEF 0.3989422804014327\\nvoid main() {\\n#pragma mapbox: initialize highp float weight\\nfloat d=-0.5*3.0*3.0*dot(v_extrude,v_extrude);float val=weight*u_intensity*GAUSS_COEF*exp(d);gl_FragColor=vec4(val,1.0,1.0,1.0);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform float u_extrude_scale;uniform float u_opacity;uniform float u_intensity;attribute vec2 a_pos;varying vec2 v_extrude;\\n#pragma mapbox: define highp float weight\\n#pragma mapbox: define mediump float radius\\nconst highp float ZERO=1.0/255.0/16.0;\\n#define GAUSS_COEF 0.3989422804014327\\nvoid main(void) {\\n#pragma mapbox: initialize highp float weight\\n#pragma mapbox: initialize mediump float radius\\nvec2 unscaled_extrude=vec2(mod(a_pos,2.0)*2.0-1.0);float S=sqrt(-2.0*log(ZERO/weight/u_intensity/GAUSS_COEF))/3.0;v_extrude=S*unscaled_extrude;vec2 extrude=v_extrude*radius*u_extrude_scale;vec4 pos=vec4(floor(a_pos*0.5)+extrude,0,1);gl_Position=u_matrix*pos;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform sampler2D u_image;uniform sampler2D u_color_ramp;uniform float u_opacity;varying vec2 v_pos;void main() {float t=texture2D(u_image,v_pos).r;vec4 color=texture2D(u_color_ramp,vec2(t,0.5));gl_FragColor=color*u_opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(0.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_world;attribute vec2 a_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos*u_world,0,1);v_pos.x=a_pos.x;v_pos.y=1.0-a_pos.y;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying float v_placed;varying float v_notUsed;void main() {float alpha=0.5;gl_FragColor=vec4(1.0,0.0,0.0,1.0)*alpha;if (v_placed > 0.5) {gl_FragColor=vec4(0.0,0.0,1.0,0.5)*alpha;}if (v_notUsed > 0.5) {gl_FragColor*=.1;}}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;attribute vec2 a_anchor_pos;attribute vec2 a_extrude;attribute vec2 a_placed;attribute vec2 a_shift;uniform mat4 u_matrix;uniform vec2 u_extrude_scale;uniform float u_camera_to_center_distance;varying float v_placed;varying float v_notUsed;void main() {vec4 projectedPoint=u_matrix*vec4(a_anchor_pos,0,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);gl_Position=u_matrix*vec4(a_pos,get_elevation(a_pos),1.0);gl_Position.xy+=(a_extrude+a_shift)*u_extrude_scale*gl_Position.w*collision_perspective_ratio;v_placed=a_placed.x;v_notUsed=a_placed.y;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;void main() {float alpha=0.5*min(v_perspective_ratio,1.0);float stroke_radius=0.9*max(v_perspective_ratio,1.0);float distance_to_center=length(v_extrude);float distance_to_edge=abs(distance_to_center-v_radius);float opacity_t=smoothstep(-stroke_radius,0.0,-distance_to_edge);vec4 color=mix(vec4(0.0,0.0,1.0,0.5),vec4(1.0,0.0,0.0,1.0),v_collision);gl_FragColor=color*alpha*opacity_t;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;attribute float a_radius;attribute vec2 a_flags;uniform mat4 u_matrix;uniform mat4 u_inv_matrix;uniform vec2 u_viewport_size;uniform float u_camera_to_center_distance;varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;vec3 toTilePosition(vec2 screenPos) {vec4 rayStart=u_inv_matrix*vec4(screenPos,-1.0,1.0);vec4 rayEnd =u_inv_matrix*vec4(screenPos, 1.0,1.0);rayStart.xyz/=rayStart.w;rayEnd.xyz /=rayEnd.w;highp float t=(0.0-rayStart.z)/(rayEnd.z-rayStart.z);return mix(rayStart.xyz,rayEnd.xyz,t);}void main() {vec2 quadCenterPos=a_pos;float radius=a_radius;float collision=a_flags.x;float vertexIdx=a_flags.y;vec2 quadVertexOffset=vec2(mix(-1.0,1.0,float(vertexIdx >=2.0)),mix(-1.0,1.0,float(vertexIdx >=1.0 && vertexIdx <=2.0)));vec2 quadVertexExtent=quadVertexOffset*radius;vec3 tilePos=toTilePosition(quadCenterPos);vec4 clipPos=u_matrix*vec4(tilePos,1.0);highp float camera_to_anchor_distance=clipPos.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);float padding_factor=1.2;v_radius=radius;v_extrude=quadVertexExtent*padding_factor;v_perspective_ratio=collision_perspective_ratio;v_collision=collision;gl_Position=vec4(clipPos.xyz/clipPos.w,1.0)+vec4(quadVertexExtent*padding_factor/u_viewport_size*2.0,0.0,0.0);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform highp vec4 u_color;uniform sampler2D u_overlay;varying vec2 v_uv;void main() {vec4 overlay_color=texture2D(u_overlay,v_uv);gl_FragColor=mix(u_color,overlay_color,overlay_color.a);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;varying vec2 v_uv;uniform mat4 u_matrix;uniform float u_overlay_scale;void main() {v_uv=a_pos/8192.0;gl_Position=u_matrix*vec4(a_pos*u_overlay_scale,get_elevation(a_pos),1);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float opacity\\ngl_FragColor=color*opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;uniform mat4 u_matrix;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float opacity\\ngl_Position=u_matrix*vec4(a_pos,0,1);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying vec2 v_pos;\\n#pragma mapbox: define highp vec4 outline_color\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 outline_color\\n#pragma mapbox: initialize lowp float opacity\\nfloat dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=outline_color*(alpha*opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'attribute vec2 a_pos;uniform mat4 u_matrix;uniform vec2 u_world;varying vec2 v_pos;\\n#pragma mapbox: define highp vec4 outline_color\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 outline_color\\n#pragma mapbox: initialize lowp float opacity\\ngl_Position=u_matrix*vec4(a_pos,0,1);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform vec2 u_texsize;uniform sampler2D u_image;uniform float u_fade;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);float dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=mix(color1,color2,u_fade)*alpha*opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_world;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;gl_Position=u_matrix*vec4(a_pos,0,1);vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,a_pos);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision highp float;\\n#endif\\nuniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_fade)*opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileZoomRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileZoomRatio,a_pos);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying vec4 v_color;void main() {gl_FragColor=v_color;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;uniform float u_vertical_gradient;uniform lowp float u_opacity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\\n#ifdef TERRAIN3D\\nattribute vec2 a_centroid;\\n#endif\\nvarying vec4 v_color;\\n#pragma mapbox: define highp float base\\n#pragma mapbox: define highp float height\\n#pragma mapbox: define highp vec4 color\\nvoid main() {\\n#pragma mapbox: initialize highp float base\\n#pragma mapbox: initialize highp float height\\n#pragma mapbox: initialize highp vec4 color\\nvec3 normal=a_normal_ed.xyz;\\n#ifdef TERRAIN3D\\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\\n#else\\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\\n#endif\\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);gl_Position=u_matrix*vec4(a_pos,t > 0.0 ? height : base,1);float colorvalue=color.r*0.2126+color.g*0.7152+color.b*0.0722;v_color=vec4(0.0,0.0,0.0,1.0);vec4 ambientlight=vec4(0.03,0.03,0.03,1.0);color+=ambientlight;float directional=clamp(dot(normal/16384.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((1.0-colorvalue+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_color.r+=clamp(color.r*directional*u_lightcolor.r,mix(0.0,0.3,1.0-u_lightcolor.r),1.0);v_color.g+=clamp(color.g*directional*u_lightcolor.g,mix(0.0,0.3,1.0-u_lightcolor.g),1.0);v_color.b+=clamp(color.b*directional*u_lightcolor.b,mix(0.0,0.3,1.0-u_lightcolor.b),1.0);v_color*=u_opacity;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\\n#pragma mapbox: define lowp float base\\n#pragma mapbox: define lowp float height\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float base\\n#pragma mapbox: initialize lowp float height\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);vec4 mixedColor=mix(color1,color2,u_fade);gl_FragColor=mixedColor*v_lighting;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_height_factor;uniform vec3 u_scale;uniform float u_vertical_gradient;uniform lowp float u_opacity;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\\n#ifdef TERRAIN3D\\nattribute vec2 a_centroid;\\n#endif\\nvarying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\\n#pragma mapbox: define lowp float base\\n#pragma mapbox: define lowp float height\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float base\\n#pragma mapbox: initialize lowp float height\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec3 normal=a_normal_ed.xyz;float edgedistance=a_normal_ed.w;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;\\n#ifdef TERRAIN3D\\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\\n#else\\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\\n#endif\\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);float z=t > 0.0 ? height : base;gl_Position=u_matrix*vec4(a_pos,z,1);vec2 pos=normal.x==1.0 && normal.y==0.0 && normal.z==16384.0\\n? a_pos\\n: vec2(edgedistance,z*u_height_factor);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,pos);v_lighting=vec4(0.0,0.0,0.0,1.0);float directional=clamp(dot(normal/16383.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((0.5+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_lighting.rgb+=clamp(directional*u_lightcolor,mix(vec3(0.0),vec3(0.3),1.0-u_lightcolor),vec3(1.0));v_lighting*=u_opacity;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision highp float;\\n#endif\\nuniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_dimension;uniform float u_zoom;uniform vec4 u_unpack;float getElevation(vec2 coord,float bias) {vec4 data=texture2D(u_image,coord)*255.0;data.a=-1.0;return dot(data,u_unpack)/4.0;}void main() {vec2 epsilon=1.0/u_dimension;float a=getElevation(v_pos+vec2(-epsilon.x,-epsilon.y),0.0);float b=getElevation(v_pos+vec2(0,-epsilon.y),0.0);float c=getElevation(v_pos+vec2(epsilon.x,-epsilon.y),0.0);float d=getElevation(v_pos+vec2(-epsilon.x,0),0.0);float e=getElevation(v_pos,0.0);float f=getElevation(v_pos+vec2(epsilon.x,0),0.0);float g=getElevation(v_pos+vec2(-epsilon.x,epsilon.y),0.0);float h=getElevation(v_pos+vec2(0,epsilon.y),0.0);float i=getElevation(v_pos+vec2(epsilon.x,epsilon.y),0.0);float exaggerationFactor=u_zoom < 2.0 ? 0.4 : u_zoom < 4.5 ? 0.35 : 0.3;float exaggeration=u_zoom < 15.0 ? (u_zoom-15.0)*exaggerationFactor : 0.0;vec2 deriv=vec2((c+f+f+i)-(a+d+d+g),(g+h+h+i)-(a+b+b+c))/pow(2.0,exaggeration+(19.2562-u_zoom));gl_FragColor=clamp(vec4(deriv.x/2.0+0.5,deriv.y/2.0+0.5,1.0,1.0),0.0,1.0);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_dimension;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);highp vec2 epsilon=1.0/u_dimension;float scale=(u_dimension.x-2.0)/u_dimension.x;v_pos=(a_texture_pos/8192.0)*scale+epsilon;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_latrange;uniform vec2 u_light;uniform vec4 u_shadow;uniform vec4 u_highlight;uniform vec4 u_accent;\\n#define PI 3.141592653589793\\nvoid main() {vec4 pixel=texture2D(u_image,v_pos);vec2 deriv=((pixel.rg*2.0)-1.0);float scaleFactor=cos(radians((u_latrange[0]-u_latrange[1])*(1.0-v_pos.y)+u_latrange[1]));float slope=atan(1.25*length(deriv)/scaleFactor);float aspect=deriv.x !=0.0 ? atan(deriv.y,-deriv.x) : PI/2.0*(deriv.y > 0.0 ? 1.0 :-1.0);float intensity=u_light.x;float azimuth=u_light.y+PI;float base=1.875-intensity*1.75;float maxValue=0.5*PI;float scaledSlope=intensity !=0.5 ? ((pow(base,slope)-1.0)/(pow(base,maxValue)-1.0))*maxValue : slope;float accent=cos(scaledSlope);vec4 accent_color=(1.0-accent)*u_accent*clamp(intensity*2.0,0.0,1.0);float shade=abs(mod((aspect+azimuth)/PI+0.5,2.0)-1.0);vec4 shade_color=mix(u_shadow,u_highlight,shade)*sin(scaledSlope)*clamp(intensity*2.0,0.0,1.0);gl_FragColor=accent_color*(1.0-shade_color.a)+shade_color;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos=a_texture_pos/8192.0;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform lowp float u_device_pixel_ratio;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);gl_FragColor=color*(alpha*opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '\\n#define scale 0.015873016\\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform vec2 u_units_to_pixels;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp float v_linesofar;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define mediump float gapwidth\\n#pragma mapbox: define lowp float offset\\n#pragma mapbox: define mediump float width\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump float gapwidth\\n#pragma mapbox: initialize lowp float offset\\n#pragma mapbox: initialize mediump float width\\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;v_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*2.0;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\\n#ifdef TERRAIN3D\\nv_gamma_scale=1.0;\\n#else\\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\\n#endif\\nv_width2=vec2(outset,inset);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;varying highp vec2 v_uv;\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);vec4 color=texture2D(u_image,v_uv);gl_FragColor=color*(alpha*opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '\\n#define scale 0.015873016\\nattribute vec2 a_pos_normal;attribute vec4 a_data;attribute float a_uv_x;attribute float a_split_index;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_units_to_pixels;uniform float u_image_height;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp vec2 v_uv;\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define mediump float gapwidth\\n#pragma mapbox: define lowp float offset\\n#pragma mapbox: define mediump float width\\nvoid main() {\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump float gapwidth\\n#pragma mapbox: initialize lowp float offset\\n#pragma mapbox: initialize mediump float width\\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;highp float texel_height=1.0/u_image_height;highp float half_texel_height=0.5*texel_height;v_uv=vec2(a_uv_x,a_split_index*texel_height-half_texel_height);vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\\n#ifdef TERRAIN3D\\nv_gamma_scale=1.0;\\n#else\\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\\n#endif\\nv_width2=vec2(outset,inset);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#ifdef GL_ES\\nprecision highp float;\\n#endif\\nuniform lowp float u_device_pixel_ratio;uniform vec2 u_texsize;uniform float u_fade;uniform mediump vec3 u_scale;uniform sampler2D u_image;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;vec2 pattern_size_a=vec2(display_size_a.x*fromScale/tileZoomRatio,display_size_a.y);vec2 pattern_size_b=vec2(display_size_b.x*toScale/tileZoomRatio,display_size_b.y);float aspect_a=display_size_a.y/v_width;float aspect_b=display_size_b.y/v_width;float dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float x_a=mod(v_linesofar/pattern_size_a.x*aspect_a,1.0);float x_b=mod(v_linesofar/pattern_size_b.x*aspect_b,1.0);float y=0.5*v_normal.y+0.5;vec2 texel_size=1.0/u_texsize;vec2 pos_a=mix(pattern_tl_a*texel_size-texel_size,pattern_br_a*texel_size+texel_size,vec2(x_a,y));vec2 pos_b=mix(pattern_tl_b*texel_size-texel_size,pattern_br_b*texel_size+texel_size,vec2(x_b,y));vec4 color=mix(texture2D(u_image,pos_a),texture2D(u_image,pos_b),u_fade);gl_FragColor=color*alpha*opacity;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '\\n#define scale 0.015873016\\n#define LINE_DISTANCE_SCALE 2.0\\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform vec2 u_units_to_pixels;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float offset\\n#pragma mapbox: define mediump float gapwidth\\n#pragma mapbox: define mediump float width\\n#pragma mapbox: define lowp float floorwidth\\n#pragma mapbox: define lowp vec4 pattern_from\\n#pragma mapbox: define lowp vec4 pattern_to\\n#pragma mapbox: define lowp float pixel_ratio_from\\n#pragma mapbox: define lowp float pixel_ratio_to\\nvoid main() {\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float offset\\n#pragma mapbox: initialize mediump float gapwidth\\n#pragma mapbox: initialize mediump float width\\n#pragma mapbox: initialize lowp float floorwidth\\n#pragma mapbox: initialize mediump vec4 pattern_from\\n#pragma mapbox: initialize mediump vec4 pattern_to\\n#pragma mapbox: initialize lowp float pixel_ratio_from\\n#pragma mapbox: initialize lowp float pixel_ratio_to\\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\\n#ifdef TERRAIN3D\\nv_gamma_scale=1.0;\\n#else\\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\\n#endif\\nv_linesofar=a_linesofar;v_width2=vec2(outset,inset);v_width=floorwidth;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;uniform float u_sdfgamma;uniform float u_mix;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define mediump float width\\n#pragma mapbox: define lowp float floorwidth\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump float width\\n#pragma mapbox: initialize lowp float floorwidth\\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float sdfdist_a=texture2D(u_image,v_tex_a).a;float sdfdist_b=texture2D(u_image,v_tex_b).a;float sdfdist=mix(sdfdist_a,sdfdist_b,u_mix);alpha*=smoothstep(0.5-u_sdfgamma/floorwidth,0.5+u_sdfgamma/floorwidth,sdfdist);gl_FragColor=color*(alpha*opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '\\n#define scale 0.015873016\\n#define LINE_DISTANCE_SCALE 2.0\\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_patternscale_a;uniform float u_tex_y_a;uniform vec2 u_patternscale_b;uniform float u_tex_y_b;uniform vec2 u_units_to_pixels;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\\n#pragma mapbox: define highp vec4 color\\n#pragma mapbox: define lowp float blur\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define mediump float gapwidth\\n#pragma mapbox: define lowp float offset\\n#pragma mapbox: define mediump float width\\n#pragma mapbox: define lowp float floorwidth\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 color\\n#pragma mapbox: initialize lowp float blur\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize mediump float gapwidth\\n#pragma mapbox: initialize lowp float offset\\n#pragma mapbox: initialize mediump float width\\n#pragma mapbox: initialize lowp float floorwidth\\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\\n#ifdef TERRAIN3D\\nv_gamma_scale=1.0;\\n#else\\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\\n#endif\\nv_tex_a=vec2(a_linesofar*u_patternscale_a.x/floorwidth,normal.y*u_patternscale_a.y+u_tex_y_a);v_tex_b=vec2(a_linesofar*u_patternscale_b.x/floorwidth,normal.y*u_patternscale_b.y+u_tex_y_b);v_width2=vec2(outset,inset);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform float u_fade_t;uniform float u_opacity;uniform sampler2D u_image0;uniform sampler2D u_image1;varying vec2 v_pos0;varying vec2 v_pos1;uniform float u_brightness_low;uniform float u_brightness_high;uniform float u_saturation_factor;uniform float u_contrast_factor;uniform vec3 u_spin_weights;void main() {vec4 color0=texture2D(u_image0,v_pos0);vec4 color1=texture2D(u_image1,v_pos1);if (color0.a > 0.0) {color0.rgb=color0.rgb/color0.a;}if (color1.a > 0.0) {color1.rgb=color1.rgb/color1.a;}vec4 color=mix(color0,color1,u_fade_t);color.a*=u_opacity;vec3 rgb=color.rgb;rgb=vec3(dot(rgb,u_spin_weights.xyz),dot(rgb,u_spin_weights.zxy),dot(rgb,u_spin_weights.yzx));float average=(color.r+color.g+color.b)/3.0;rgb+=(average-rgb)*u_saturation_factor;rgb=(rgb-0.5)*u_contrast_factor+0.5;vec3 u_high_vec=vec3(u_brightness_low,u_brightness_low,u_brightness_low);vec3 u_low_vec=vec3(u_brightness_high,u_brightness_high,u_brightness_high);gl_FragColor=vec4(mix(u_high_vec,u_low_vec,rgb)*color.a,color.a);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform mat4 u_matrix;uniform vec2 u_tl_parent;uniform float u_scale_parent;uniform float u_buffer_scale;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos0;varying vec2 v_pos1;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos0=(((a_texture_pos/8192.0)-0.5)/u_buffer_scale )+0.5;v_pos1=(v_pos0*u_scale_parent)+u_tl_parent;}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform sampler2D u_texture;varying vec2 v_tex;varying float v_fade_opacity;\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\nlowp float alpha=opacity*v_fade_opacity;gl_FragColor=texture2D(u_texture,v_tex)*alpha;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform highp float u_camera_to_center_distance;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform float u_fade_change;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform vec2 u_texsize;varying vec2 v_tex;varying float v_fade_opacity;\\n#pragma mapbox: define lowp float opacity\\nvoid main() {\\n#pragma mapbox: initialize lowp float opacity\\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;vec2 a_minFontScale=a_pixeloffset.zw/256.0;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\\ncamera_to_anchor_distance/u_camera_to_center_distance :\\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*max(a_minFontScale,fontScale)+a_pxoffset/16.0),z,1.0);v_tex=a_tex/u_texsize;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float visibility=calculate_visibility(projectedPoint);v_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#define SDF_PX 8.0\\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;uniform bool u_is_text;varying vec2 v_data0;varying vec3 v_data1;\\n#pragma mapbox: define highp vec4 fill_color\\n#pragma mapbox: define highp vec4 halo_color\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float halo_width\\n#pragma mapbox: define lowp float halo_blur\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 fill_color\\n#pragma mapbox: initialize highp vec4 halo_color\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float halo_width\\n#pragma mapbox: initialize lowp float halo_blur\\nfloat EDGE_GAMMA=0.105/u_device_pixel_ratio;vec2 tex=v_data0.xy;float gamma_scale=v_data1.x;float size=v_data1.y;float fade_opacity=v_data1[2];float fontScale=u_is_text ? size/24.0 : size;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float inner_edge=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);inner_edge=inner_edge+gamma*gamma_scale;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(inner_edge-gamma_scaled,inner_edge+gamma_scaled,dist);if (u_is_halo) {lowp float halo_edge=(6.0-halo_width/fontScale)/SDF_PX;alpha=min(smoothstep(halo_edge-gamma_scaled,halo_edge+gamma_scaled,dist),1.0-alpha);}gl_FragColor=color*(alpha*opacity*fade_opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;varying vec2 v_data0;varying vec3 v_data1;\\n#pragma mapbox: define highp vec4 fill_color\\n#pragma mapbox: define highp vec4 halo_color\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float halo_width\\n#pragma mapbox: define lowp float halo_blur\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 fill_color\\n#pragma mapbox: initialize highp vec4 halo_color\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float halo_width\\n#pragma mapbox: initialize lowp float halo_blur\\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\\ncamera_to_anchor_distance/u_camera_to_center_distance :\\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale+a_pxoffset),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0=a_tex/u_texsize;v_data1=vec3(gamma_scale,size,interpolated_fade_opacity);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default '#define SDF_PX 8.0\\n#define SDF 1.0\\n#define ICON 0.0\\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform sampler2D u_texture_icon;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;varying vec4 v_data0;varying vec4 v_data1;\\n#pragma mapbox: define highp vec4 fill_color\\n#pragma mapbox: define highp vec4 halo_color\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float halo_width\\n#pragma mapbox: define lowp float halo_blur\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 fill_color\\n#pragma mapbox: initialize highp vec4 halo_color\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float halo_width\\n#pragma mapbox: initialize lowp float halo_blur\\nfloat fade_opacity=v_data1[2];if (v_data1.w==ICON) {vec2 tex_icon=v_data0.zw;lowp float alpha=opacity*fade_opacity;gl_FragColor=texture2D(u_texture_icon,tex_icon)*alpha;\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\nreturn;}vec2 tex=v_data0.xy;float EDGE_GAMMA=0.105/u_device_pixel_ratio;float gamma_scale=v_data1.x;float size=v_data1.y;float fontScale=size/24.0;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float buff=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);buff=(6.0-halo_width/fontScale)/SDF_PX;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(buff-gamma_scaled,buff+gamma_scaled,dist);gl_FragColor=color*(alpha*opacity*fade_opacity);\\n#ifdef OVERDRAW_INSPECTOR\\ngl_FragColor=vec4(1.0);\\n#endif\\n}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;uniform vec2 u_texsize_icon;varying vec4 v_data0;varying vec4 v_data1;\\n#pragma mapbox: define highp vec4 fill_color\\n#pragma mapbox: define highp vec4 halo_color\\n#pragma mapbox: define lowp float opacity\\n#pragma mapbox: define lowp float halo_width\\n#pragma mapbox: define lowp float halo_blur\\nvoid main() {\\n#pragma mapbox: initialize highp vec4 fill_color\\n#pragma mapbox: initialize highp vec4 halo_color\\n#pragma mapbox: initialize lowp float opacity\\n#pragma mapbox: initialize lowp float halo_width\\n#pragma mapbox: initialize lowp float halo_blur\\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);float is_sdf=a_size[0]-2.0*a_size_min;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\\ncamera_to_anchor_distance/u_camera_to_center_distance :\\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=size/24.0;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0.xy=a_tex/u_texsize;v_data0.zw=a_tex/u_texsize_icon;v_data1=vec4(gamma_scale,size,interpolated_fade_opacity,is_sdf);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'uniform sampler2D u_texture;varying vec2 v_texture_pos;void main() {gl_FragColor=texture2D(u_texture,v_texture_pos);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'varying float v_depth;const highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitMsk=vec4(0.,vec3(1./256.0));highp vec4 pack(highp float value) {highp vec4 comp=fract(value*bitSh);comp-=comp.xxyz*bitMsk;return comp;}void main() {gl_FragColor=pack(v_depth);}';\n","// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`.\nexport default 'precision mediump float;uniform sampler2D u_texture;uniform float u_terrain_coords_id;varying vec2 v_texture_pos;void main() {vec4 rgba=texture2D(u_texture,v_texture_pos);gl_FragColor=vec4(rgba.r,rgba.g,rgba.b,u_terrain_coords_id);}';\n","\nimport type {Program} from './program';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {IndexBuffer} from '../gl/index_buffer';\nimport type {Context} from '../gl/context';\n\n/**\n * @internal\n * A vertex array object used to pass data to the webgl code\n */\nexport class VertexArrayObject {\n context: Context;\n boundProgram: Program;\n boundLayoutVertexBuffer: VertexBuffer;\n boundPaintVertexBuffers: Array;\n boundIndexBuffer: IndexBuffer;\n boundVertexOffset: number;\n boundDynamicVertexBuffer: VertexBuffer;\n boundDynamicVertexBuffer2: VertexBuffer;\n boundDynamicVertexBuffer3: VertexBuffer;\n vao: any;\n\n constructor() {\n this.boundProgram = null;\n this.boundLayoutVertexBuffer = null;\n this.boundPaintVertexBuffers = [];\n this.boundIndexBuffer = null;\n this.boundVertexOffset = null;\n this.boundDynamicVertexBuffer = null;\n this.vao = null;\n }\n\n bind(context: Context,\n program: Program,\n layoutVertexBuffer: VertexBuffer,\n paintVertexBuffers: Array,\n indexBuffer?: IndexBuffer | null,\n vertexOffset?: number | null,\n dynamicVertexBuffer?: VertexBuffer | null,\n dynamicVertexBuffer2?: VertexBuffer | null,\n dynamicVertexBuffer3?: VertexBuffer | null) {\n\n this.context = context;\n\n let paintBuffersDiffer = this.boundPaintVertexBuffers.length !== paintVertexBuffers.length;\n for (let i = 0; !paintBuffersDiffer && i < paintVertexBuffers.length; i++) {\n if (this.boundPaintVertexBuffers[i] !== paintVertexBuffers[i]) {\n paintBuffersDiffer = true;\n }\n }\n\n const isFreshBindRequired = (\n !this.vao ||\n this.boundProgram !== program ||\n this.boundLayoutVertexBuffer !== layoutVertexBuffer ||\n paintBuffersDiffer ||\n this.boundIndexBuffer !== indexBuffer ||\n this.boundVertexOffset !== vertexOffset ||\n this.boundDynamicVertexBuffer !== dynamicVertexBuffer ||\n this.boundDynamicVertexBuffer2 !== dynamicVertexBuffer2 ||\n this.boundDynamicVertexBuffer3 !== dynamicVertexBuffer3\n );\n\n if (isFreshBindRequired) {\n this.freshBind(program, layoutVertexBuffer, paintVertexBuffers, indexBuffer, vertexOffset, dynamicVertexBuffer, dynamicVertexBuffer2, dynamicVertexBuffer3);\n } else {\n context.bindVertexArray.set(this.vao);\n\n if (dynamicVertexBuffer) {\n // The buffer may have been updated. Rebind to upload data.\n dynamicVertexBuffer.bind();\n }\n\n if (indexBuffer && indexBuffer.dynamicDraw) {\n indexBuffer.bind();\n }\n\n if (dynamicVertexBuffer2) {\n dynamicVertexBuffer2.bind();\n }\n\n if (dynamicVertexBuffer3) {\n dynamicVertexBuffer3.bind();\n }\n }\n }\n\n freshBind(program: Program,\n layoutVertexBuffer: VertexBuffer,\n paintVertexBuffers: Array,\n indexBuffer?: IndexBuffer | null,\n vertexOffset?: number | null,\n dynamicVertexBuffer?: VertexBuffer | null,\n dynamicVertexBuffer2?: VertexBuffer | null,\n dynamicVertexBuffer3?: VertexBuffer | null) {\n\n const numNextAttributes = program.numAttributes;\n\n const context = this.context;\n const gl = context.gl;\n\n if (this.vao) this.destroy();\n this.vao = context.createVertexArray();\n context.bindVertexArray.set(this.vao);\n\n // store the arguments so that we can verify them when the vao is bound again\n this.boundProgram = program;\n this.boundLayoutVertexBuffer = layoutVertexBuffer;\n this.boundPaintVertexBuffers = paintVertexBuffers;\n this.boundIndexBuffer = indexBuffer;\n this.boundVertexOffset = vertexOffset;\n this.boundDynamicVertexBuffer = dynamicVertexBuffer;\n this.boundDynamicVertexBuffer2 = dynamicVertexBuffer2;\n this.boundDynamicVertexBuffer3 = dynamicVertexBuffer3;\n\n layoutVertexBuffer.enableAttributes(gl, program);\n for (const vertexBuffer of paintVertexBuffers) {\n vertexBuffer.enableAttributes(gl, program);\n }\n\n if (dynamicVertexBuffer) {\n dynamicVertexBuffer.enableAttributes(gl, program);\n }\n if (dynamicVertexBuffer2) {\n dynamicVertexBuffer2.enableAttributes(gl, program);\n }\n if (dynamicVertexBuffer3) {\n dynamicVertexBuffer3.enableAttributes(gl, program);\n }\n\n layoutVertexBuffer.bind();\n layoutVertexBuffer.setVertexAttribPointers(gl, program, vertexOffset);\n for (const vertexBuffer of paintVertexBuffers) {\n vertexBuffer.bind();\n vertexBuffer.setVertexAttribPointers(gl, program, vertexOffset);\n }\n\n if (dynamicVertexBuffer) {\n dynamicVertexBuffer.bind();\n dynamicVertexBuffer.setVertexAttribPointers(gl, program, vertexOffset);\n }\n if (indexBuffer) {\n indexBuffer.bind();\n }\n if (dynamicVertexBuffer2) {\n dynamicVertexBuffer2.bind();\n dynamicVertexBuffer2.setVertexAttribPointers(gl, program, vertexOffset);\n }\n if (dynamicVertexBuffer3) {\n dynamicVertexBuffer3.bind();\n dynamicVertexBuffer3.setVertexAttribPointers(gl, program, vertexOffset);\n }\n\n context.currentNumAttributes = numNextAttributes;\n }\n\n destroy() {\n if (this.vao) {\n this.context.deleteVertexArray(this.vao);\n this.vao = null;\n }\n }\n}\n","import {shaders} from '../shaders/shaders';\nimport {ProgramConfiguration} from '../data/program_configuration';\nimport {VertexArrayObject} from './vertex_array_object';\nimport {Context} from '../gl/context';\n\nimport type {SegmentVector} from '../data/segment';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {IndexBuffer} from '../gl/index_buffer';\nimport type {DepthMode} from '../gl/depth_mode';\nimport type {StencilMode} from '../gl/stencil_mode';\nimport type {ColorMode} from '../gl/color_mode';\nimport type {CullFaceMode} from '../gl/cull_face_mode';\nimport type {UniformBindings, UniformValues, UniformLocations} from './uniform_binding';\nimport type {BinderUniform} from '../data/program_configuration';\nimport {terrainPreludeUniforms, TerrainPreludeUniformsType} from './program/terrain_program';\nimport type {TerrainData} from '../render/terrain';\nimport {Terrain} from '../render/terrain';\n\nexport type DrawMode = WebGLRenderingContextBase['LINES'] | WebGLRenderingContextBase['TRIANGLES'] | WebGL2RenderingContext['LINE_STRIP'];\n\nfunction getTokenizedAttributesAndUniforms(array: Array): Array {\n const result = [];\n\n for (let i = 0; i < array.length; i++) {\n if (array[i] === null) continue;\n const token = array[i].split(' ');\n result.push(token.pop());\n }\n return result;\n}\n\n/**\n * @internal\n * A webgl program to execute in the GPU space\n */\nexport class Program {\n program: WebGLProgram;\n attributes: {[_: string]: number};\n numAttributes: number;\n fixedUniforms: Us;\n terrainUniforms: TerrainPreludeUniformsType;\n binderUniforms: Array;\n failedToCreate: boolean;\n\n constructor(context: Context,\n source: {\n fragmentSource: string;\n vertexSource: string;\n staticAttributes: Array;\n staticUniforms: Array;\n },\n configuration: ProgramConfiguration,\n fixedUniforms: (b: Context, a: UniformLocations) => Us,\n showOverdrawInspector: boolean,\n terrain: Terrain) {\n\n const gl = context.gl;\n this.program = gl.createProgram();\n\n const staticAttrInfo = getTokenizedAttributesAndUniforms(source.staticAttributes);\n const dynamicAttrInfo = configuration ? configuration.getBinderAttributes() : [];\n const allAttrInfo = staticAttrInfo.concat(dynamicAttrInfo);\n\n const preludeUniformsInfo = shaders.prelude.staticUniforms ? getTokenizedAttributesAndUniforms(shaders.prelude.staticUniforms) : [];\n const staticUniformsInfo = source.staticUniforms ? getTokenizedAttributesAndUniforms(source.staticUniforms) : [];\n const dynamicUniformsInfo = configuration ? configuration.getBinderUniforms() : [];\n // remove duplicate uniforms\n const uniformList = preludeUniformsInfo.concat(staticUniformsInfo).concat(dynamicUniformsInfo);\n const allUniformsInfo = [];\n for (const uniform of uniformList) {\n if (allUniformsInfo.indexOf(uniform) < 0) allUniformsInfo.push(uniform);\n }\n\n const defines = configuration ? configuration.defines() : [];\n if (showOverdrawInspector) {\n defines.push('#define OVERDRAW_INSPECTOR;');\n }\n if (terrain) {\n defines.push('#define TERRAIN3D;');\n }\n\n const fragmentSource = defines.concat(shaders.prelude.fragmentSource, source.fragmentSource).join('\\n');\n const vertexSource = defines.concat(shaders.prelude.vertexSource, source.vertexSource).join('\\n');\n\n const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);\n if (gl.isContextLost()) {\n this.failedToCreate = true;\n return;\n }\n gl.shaderSource(fragmentShader, fragmentSource);\n gl.compileShader(fragmentShader);\n\n if (!gl.getShaderParameter(fragmentShader, gl.COMPILE_STATUS)) {\n throw new Error(`Could not compile fragment shader: ${gl.getShaderInfoLog(fragmentShader)}`);\n }\n\n gl.attachShader(this.program, fragmentShader);\n\n const vertexShader = gl.createShader(gl.VERTEX_SHADER);\n if (gl.isContextLost()) {\n this.failedToCreate = true;\n return;\n }\n gl.shaderSource(vertexShader, vertexSource);\n gl.compileShader(vertexShader);\n\n if (!gl.getShaderParameter(vertexShader, gl.COMPILE_STATUS)) {\n throw new Error(`Could not compile vertex shader: ${gl.getShaderInfoLog(vertexShader)}`);\n }\n\n gl.attachShader(this.program, vertexShader);\n\n this.attributes = {};\n const uniformLocations = {};\n\n this.numAttributes = allAttrInfo.length;\n\n for (let i = 0; i < this.numAttributes; i++) {\n if (allAttrInfo[i]) {\n gl.bindAttribLocation(this.program, i, allAttrInfo[i]);\n this.attributes[allAttrInfo[i]] = i;\n }\n }\n\n gl.linkProgram(this.program);\n\n if (!gl.getProgramParameter(this.program, gl.LINK_STATUS)) {\n throw new Error(`Program failed to link: ${gl.getProgramInfoLog(this.program)}`);\n }\n\n gl.deleteShader(vertexShader);\n gl.deleteShader(fragmentShader);\n\n for (let it = 0; it < allUniformsInfo.length; it++) {\n const uniform = allUniformsInfo[it];\n if (uniform && !uniformLocations[uniform]) {\n const uniformLocation = gl.getUniformLocation(this.program, uniform);\n if (uniformLocation) {\n uniformLocations[uniform] = uniformLocation;\n }\n }\n }\n\n this.fixedUniforms = fixedUniforms(context, uniformLocations);\n this.terrainUniforms = terrainPreludeUniforms(context, uniformLocations);\n this.binderUniforms = configuration ? configuration.getUniforms(context, uniformLocations) : [];\n }\n\n draw(context: Context,\n drawMode: DrawMode,\n depthMode: Readonly,\n stencilMode: Readonly,\n colorMode: Readonly,\n cullFaceMode: Readonly,\n uniformValues: UniformValues,\n terrain: TerrainData,\n layerID: string,\n layoutVertexBuffer: VertexBuffer,\n indexBuffer: IndexBuffer,\n segments: SegmentVector,\n currentProperties?: any,\n zoom?: number | null,\n configuration?: ProgramConfiguration | null,\n dynamicLayoutBuffer?: VertexBuffer | null,\n dynamicLayoutBuffer2?: VertexBuffer | null,\n dynamicLayoutBuffer3?: VertexBuffer | null) {\n\n const gl = context.gl;\n\n if (this.failedToCreate) return;\n\n context.program.set(this.program);\n context.setDepthMode(depthMode);\n context.setStencilMode(stencilMode);\n context.setColorMode(colorMode);\n context.setCullFace(cullFaceMode);\n\n // set variables used by the 3d functions defined in _prelude.vertex.glsl\n if (terrain) {\n context.activeTexture.set(gl.TEXTURE2);\n gl.bindTexture(gl.TEXTURE_2D, terrain.depthTexture);\n context.activeTexture.set(gl.TEXTURE3);\n gl.bindTexture(gl.TEXTURE_2D, terrain.texture);\n for (const name in this.terrainUniforms) {\n this.terrainUniforms[name].set(terrain[name]);\n }\n }\n\n for (const name in this.fixedUniforms) {\n this.fixedUniforms[name].set(uniformValues[name]);\n }\n\n if (configuration) {\n configuration.setUniforms(context, this.binderUniforms, currentProperties, {zoom: (zoom as any)});\n }\n\n let primitiveSize = 0;\n switch (drawMode) {\n case gl.LINES:\n primitiveSize = 2;\n break;\n case gl.TRIANGLES:\n primitiveSize = 3;\n break;\n case gl.LINE_STRIP:\n primitiveSize = 1;\n break;\n }\n\n for (const segment of segments.get()) {\n const vaos = segment.vaos || (segment.vaos = {});\n const vao: VertexArrayObject = vaos[layerID] || (vaos[layerID] = new VertexArrayObject());\n\n vao.bind(\n context,\n this,\n layoutVertexBuffer,\n configuration ? configuration.getPaintVertexBuffers() : [],\n indexBuffer,\n segment.vertexOffset,\n dynamicLayoutBuffer,\n dynamicLayoutBuffer2,\n dynamicLayoutBuffer3\n );\n\n gl.drawElements(\n drawMode,\n segment.primitiveLength * primitiveSize,\n gl.UNSIGNED_SHORT,\n segment.primitiveOffset * primitiveSize * 2);\n }\n }\n}\n","import {\n Uniform1i,\n Uniform1f,\n Uniform4f,\n UniformMatrix4f\n} from '../uniform_binding';\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../../render/uniform_binding';\nimport {mat4} from 'gl-matrix';\n\nexport type TerrainPreludeUniformsType = {\n 'u_depth': Uniform1i;\n 'u_terrain': Uniform1i;\n 'u_terrain_dim': Uniform1f;\n 'u_terrain_matrix': UniformMatrix4f;\n 'u_terrain_unpack': Uniform4f;\n 'u_terrain_exaggeration': Uniform1f;\n};\n\nexport type TerrainUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_texture': Uniform1i;\n 'u_ele_delta': Uniform1f;\n};\n\nexport type TerrainDepthUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_ele_delta': Uniform1f;\n};\n\nexport type TerrainCoordsUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_texture': Uniform1i;\n 'u_terrain_coords_id': Uniform1f;\n 'u_ele_delta': Uniform1f;\n};\n\nconst terrainPreludeUniforms = (context: Context, locations: UniformLocations): TerrainPreludeUniformsType => ({\n 'u_depth': new Uniform1i(context, locations.u_depth),\n 'u_terrain': new Uniform1i(context, locations.u_terrain),\n 'u_terrain_dim': new Uniform1f(context, locations.u_terrain_dim),\n 'u_terrain_matrix': new UniformMatrix4f(context, locations.u_terrain_matrix),\n 'u_terrain_unpack': new Uniform4f(context, locations.u_terrain_unpack),\n 'u_terrain_exaggeration': new Uniform1f(context, locations.u_terrain_exaggeration)\n});\n\nconst terrainUniforms = (context: Context, locations: UniformLocations): TerrainUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_texture': new Uniform1i(context, locations.u_texture),\n 'u_ele_delta': new Uniform1f(context, locations.u_ele_delta)\n});\n\nconst terrainDepthUniforms = (context: Context, locations: UniformLocations): TerrainDepthUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_ele_delta': new Uniform1f(context, locations.u_ele_delta)\n});\n\nconst terrainCoordsUniforms = (context: Context, locations: UniformLocations): TerrainCoordsUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_texture': new Uniform1i(context, locations.u_texture),\n 'u_terrain_coords_id': new Uniform1f(context, locations.u_terrain_coords_id),\n 'u_ele_delta': new Uniform1f(context, locations.u_ele_delta)\n});\n\nconst terrainUniformValues = (\n matrix: mat4,\n eleDelta: number\n): UniformValues => ({\n 'u_matrix': matrix,\n 'u_texture': 0,\n 'u_ele_delta': eleDelta\n});\n\nconst terrainDepthUniformValues = (\n matrix: mat4,\n eleDelta: number\n): UniformValues => ({\n 'u_matrix': matrix,\n 'u_ele_delta': eleDelta\n});\n\nconst terrainCoordsUniformValues = (\n matrix: mat4,\n coordsId: number,\n eleDelta: number\n): UniformValues => ({\n 'u_matrix': matrix,\n 'u_terrain_coords_id': coordsId / 255,\n 'u_texture': 0,\n 'u_ele_delta': eleDelta\n});\n\nexport {terrainUniforms, terrainDepthUniforms, terrainCoordsUniforms, terrainPreludeUniforms, terrainUniformValues, terrainDepthUniformValues, terrainCoordsUniformValues};\n","import {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f\n} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\n\nimport type {Painter} from '../painter';\nimport type {OverscaledTileID} from '../../source/tile_id';\nimport type {CrossFaded} from '../../style/properties';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport type {UniformValues} from '../uniform_binding';\nimport type {Tile} from '../../source/tile';\nimport type {ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\n\ntype BackgroundPatternUniformsType = {\n 'u_image': Uniform1i;\n 'u_pattern_tl_a': Uniform2f;\n 'u_pattern_br_a': Uniform2f;\n 'u_pattern_tl_b': Uniform2f;\n 'u_pattern_br_b': Uniform2f;\n 'u_texsize': Uniform2f;\n 'u_mix': Uniform1f;\n 'u_pattern_size_a': Uniform2f;\n 'u_pattern_size_b': Uniform2f;\n 'u_scale_a': Uniform1f;\n 'u_scale_b': Uniform1f;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_tile_units_to_pixels': Uniform1f;\n};\n\nexport type PatternUniformsType = {\n // pattern uniforms:\n 'u_image': Uniform1i;\n 'u_texsize': Uniform2f;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n};\n\nfunction patternUniformValues(crossfade: CrossfadeParameters, painter: Painter, tile: Tile): UniformValues {\n\n const tileRatio = 1 / pixelsToTileUnits(tile, 1, painter.transform.tileZoom);\n\n const numTiles = Math.pow(2, tile.tileID.overscaledZ);\n const tileSizeAtNearestZoom = tile.tileSize * Math.pow(2, painter.transform.tileZoom) / numTiles;\n\n const pixelX = tileSizeAtNearestZoom * (tile.tileID.canonical.x + tile.tileID.wrap * numTiles);\n const pixelY = tileSizeAtNearestZoom * tile.tileID.canonical.y;\n\n return {\n 'u_image': 0,\n 'u_texsize': tile.imageAtlasTexture.size,\n 'u_scale': [tileRatio, crossfade.fromScale, crossfade.toScale],\n 'u_fade': crossfade.t,\n // split the pixel coord into two pairs of 16 bit numbers. The glsl spec only guarantees 16 bits of precision.\n 'u_pixel_coord_upper': [pixelX >> 16, pixelY >> 16],\n 'u_pixel_coord_lower': [pixelX & 0xFFFF, pixelY & 0xFFFF]\n };\n}\n\nfunction bgPatternUniformValues(\n image: CrossFaded,\n crossfade: CrossfadeParameters,\n painter: Painter,\n tile: {\n tileID: OverscaledTileID;\n tileSize: number;\n }\n): UniformValues {\n const imagePosA = painter.imageManager.getPattern(image.from.toString());\n const imagePosB = painter.imageManager.getPattern(image.to.toString());\n const {width, height} = painter.imageManager.getPixelSize();\n\n const numTiles = Math.pow(2, tile.tileID.overscaledZ);\n const tileSizeAtNearestZoom = tile.tileSize * Math.pow(2, painter.transform.tileZoom) / numTiles;\n\n const pixelX = tileSizeAtNearestZoom * (tile.tileID.canonical.x + tile.tileID.wrap * numTiles);\n const pixelY = tileSizeAtNearestZoom * tile.tileID.canonical.y;\n\n return {\n 'u_image': 0,\n 'u_pattern_tl_a': (imagePosA as any).tl,\n 'u_pattern_br_a': (imagePosA as any).br,\n 'u_pattern_tl_b': (imagePosB as any).tl,\n 'u_pattern_br_b': (imagePosB as any).br,\n 'u_texsize': [width, height],\n 'u_mix': crossfade.t,\n 'u_pattern_size_a': (imagePosA as any).displaySize,\n 'u_pattern_size_b': (imagePosB as any).displaySize,\n 'u_scale_a': crossfade.fromScale,\n 'u_scale_b': crossfade.toScale,\n 'u_tile_units_to_pixels': 1 / pixelsToTileUnits(tile, 1, painter.transform.tileZoom),\n // split the pixel coord into two pairs of 16 bit numbers. The glsl spec only guarantees 16 bits of precision.\n 'u_pixel_coord_upper': [pixelX >> 16, pixelY >> 16],\n 'u_pixel_coord_lower': [pixelX & 0xFFFF, pixelY & 0xFFFF]\n };\n}\nexport {bgPatternUniformValues, patternUniformValues};\n","import {patternUniformValues} from './pattern';\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f,\n UniformMatrix4f\n} from '../uniform_binding';\n\nimport {mat3, mat4, vec3} from 'gl-matrix';\nimport {extend} from '../../util/util';\n\nimport type {Context} from '../../gl/context';\nimport type {Painter} from '../painter';\nimport type {OverscaledTileID} from '../../source/tile_id';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport type {Tile} from '../../source/tile';\n\nexport type FillExtrusionUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_lightpos': Uniform3f;\n 'u_lightintensity': Uniform1f;\n 'u_lightcolor': Uniform3f;\n 'u_vertical_gradient': Uniform1f;\n 'u_opacity': Uniform1f;\n};\n\nexport type FillExtrusionPatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_lightpos': Uniform3f;\n 'u_lightintensity': Uniform1f;\n 'u_lightcolor': Uniform3f;\n 'u_height_factor': Uniform1f;\n 'u_vertical_gradient': Uniform1f;\n // pattern uniforms:\n 'u_texsize': Uniform2f;\n 'u_image': Uniform1i;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n 'u_opacity': Uniform1f;\n};\n\nconst fillExtrusionUniforms = (context: Context, locations: UniformLocations): FillExtrusionUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_lightpos': new Uniform3f(context, locations.u_lightpos),\n 'u_lightintensity': new Uniform1f(context, locations.u_lightintensity),\n 'u_lightcolor': new Uniform3f(context, locations.u_lightcolor),\n 'u_vertical_gradient': new Uniform1f(context, locations.u_vertical_gradient),\n 'u_opacity': new Uniform1f(context, locations.u_opacity)\n});\n\nconst fillExtrusionPatternUniforms = (context: Context, locations: UniformLocations): FillExtrusionPatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_lightpos': new Uniform3f(context, locations.u_lightpos),\n 'u_lightintensity': new Uniform1f(context, locations.u_lightintensity),\n 'u_lightcolor': new Uniform3f(context, locations.u_lightcolor),\n 'u_vertical_gradient': new Uniform1f(context, locations.u_vertical_gradient),\n 'u_height_factor': new Uniform1f(context, locations.u_height_factor),\n // pattern uniforms\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper),\n 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower),\n 'u_scale': new Uniform3f(context, locations.u_scale),\n 'u_fade': new Uniform1f(context, locations.u_fade),\n 'u_opacity': new Uniform1f(context, locations.u_opacity)\n});\n\nconst fillExtrusionUniformValues = (\n matrix: mat4,\n painter: Painter,\n shouldUseVerticalGradient: boolean,\n opacity: number\n): UniformValues => {\n const light = painter.style.light;\n const _lp = light.properties.get('position');\n const lightPos = [_lp.x, _lp.y, _lp.z] as vec3;\n const lightMat = mat3.create();\n if (light.properties.get('anchor') === 'viewport') {\n mat3.fromRotation(lightMat, -painter.transform.angle);\n }\n vec3.transformMat3(lightPos, lightPos, lightMat);\n\n const lightColor = light.properties.get('color');\n\n return {\n 'u_matrix': matrix,\n 'u_lightpos': lightPos,\n 'u_lightintensity': light.properties.get('intensity'),\n 'u_lightcolor': [lightColor.r, lightColor.g, lightColor.b],\n 'u_vertical_gradient': +shouldUseVerticalGradient,\n 'u_opacity': opacity\n };\n};\n\nconst fillExtrusionPatternUniformValues = (\n matrix: mat4,\n painter: Painter,\n shouldUseVerticalGradient: boolean,\n opacity: number,\n coord: OverscaledTileID,\n crossfade: CrossfadeParameters,\n tile: Tile\n): UniformValues => {\n return extend(fillExtrusionUniformValues(matrix, painter, shouldUseVerticalGradient, opacity),\n patternUniformValues(crossfade, painter, tile),\n {\n 'u_height_factor': -Math.pow(2, coord.overscaledZ) / tile.tileSize / 8\n });\n};\n\nexport {\n fillExtrusionUniforms,\n fillExtrusionPatternUniforms,\n fillExtrusionUniformValues,\n fillExtrusionPatternUniformValues\n};\n","import * as glMatrix from \"./common.js\";\n/**\n * 3x3 Matrix\n * @module mat3\n */\n\n/**\n * Creates a new identity mat3\n *\n * @returns {mat3} a new 3x3 matrix\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(9);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n }\n\n out[0] = 1;\n out[4] = 1;\n out[8] = 1;\n return out;\n}\n/**\n * Copies the upper-left 3x3 values into the given mat3.\n *\n * @param {mat3} out the receiving 3x3 matrix\n * @param {ReadonlyMat4} a the source 4x4 matrix\n * @returns {mat3} out\n */\n\nexport function fromMat4(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[4];\n out[4] = a[5];\n out[5] = a[6];\n out[6] = a[8];\n out[7] = a[9];\n out[8] = a[10];\n return out;\n}\n/**\n * Creates a new mat3 initialized with values from an existing matrix\n *\n * @param {ReadonlyMat3} a matrix to clone\n * @returns {mat3} a new 3x3 matrix\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(9);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n return out;\n}\n/**\n * Copy the values from one mat3 to another\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the source matrix\n * @returns {mat3} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n out[4] = a[4];\n out[5] = a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n return out;\n}\n/**\n * Create a new mat3 with the given values\n *\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m10 Component in column 1, row 0 position (index 3)\n * @param {Number} m11 Component in column 1, row 1 position (index 4)\n * @param {Number} m12 Component in column 1, row 2 position (index 5)\n * @param {Number} m20 Component in column 2, row 0 position (index 6)\n * @param {Number} m21 Component in column 2, row 1 position (index 7)\n * @param {Number} m22 Component in column 2, row 2 position (index 8)\n * @returns {mat3} A new mat3\n */\n\nexport function fromValues(m00, m01, m02, m10, m11, m12, m20, m21, m22) {\n var out = new glMatrix.ARRAY_TYPE(9);\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m10;\n out[4] = m11;\n out[5] = m12;\n out[6] = m20;\n out[7] = m21;\n out[8] = m22;\n return out;\n}\n/**\n * Set the components of a mat3 to the given values\n *\n * @param {mat3} out the receiving matrix\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m02 Component in column 0, row 2 position (index 2)\n * @param {Number} m10 Component in column 1, row 0 position (index 3)\n * @param {Number} m11 Component in column 1, row 1 position (index 4)\n * @param {Number} m12 Component in column 1, row 2 position (index 5)\n * @param {Number} m20 Component in column 2, row 0 position (index 6)\n * @param {Number} m21 Component in column 2, row 1 position (index 7)\n * @param {Number} m22 Component in column 2, row 2 position (index 8)\n * @returns {mat3} out\n */\n\nexport function set(out, m00, m01, m02, m10, m11, m12, m20, m21, m22) {\n out[0] = m00;\n out[1] = m01;\n out[2] = m02;\n out[3] = m10;\n out[4] = m11;\n out[5] = m12;\n out[6] = m20;\n out[7] = m21;\n out[8] = m22;\n return out;\n}\n/**\n * Set a mat3 to the identity matrix\n *\n * @param {mat3} out the receiving matrix\n * @returns {mat3} out\n */\n\nexport function identity(out) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 1;\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 1;\n return out;\n}\n/**\n * Transpose the values of a mat3\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the source matrix\n * @returns {mat3} out\n */\n\nexport function transpose(out, a) {\n // If we are transposing ourselves we can skip a few steps but have to cache some values\n if (out === a) {\n var a01 = a[1],\n a02 = a[2],\n a12 = a[5];\n out[1] = a[3];\n out[2] = a[6];\n out[3] = a01;\n out[5] = a[7];\n out[6] = a02;\n out[7] = a12;\n } else {\n out[0] = a[0];\n out[1] = a[3];\n out[2] = a[6];\n out[3] = a[1];\n out[4] = a[4];\n out[5] = a[7];\n out[6] = a[2];\n out[7] = a[5];\n out[8] = a[8];\n }\n\n return out;\n}\n/**\n * Inverts a mat3\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the source matrix\n * @returns {mat3} out\n */\n\nexport function invert(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2];\n var a10 = a[3],\n a11 = a[4],\n a12 = a[5];\n var a20 = a[6],\n a21 = a[7],\n a22 = a[8];\n var b01 = a22 * a11 - a12 * a21;\n var b11 = -a22 * a10 + a12 * a20;\n var b21 = a21 * a10 - a11 * a20; // Calculate the determinant\n\n var det = a00 * b01 + a01 * b11 + a02 * b21;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = b01 * det;\n out[1] = (-a22 * a01 + a02 * a21) * det;\n out[2] = (a12 * a01 - a02 * a11) * det;\n out[3] = b11 * det;\n out[4] = (a22 * a00 - a02 * a20) * det;\n out[5] = (-a12 * a00 + a02 * a10) * det;\n out[6] = b21 * det;\n out[7] = (-a21 * a00 + a01 * a20) * det;\n out[8] = (a11 * a00 - a01 * a10) * det;\n return out;\n}\n/**\n * Calculates the adjugate of a mat3\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the source matrix\n * @returns {mat3} out\n */\n\nexport function adjoint(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2];\n var a10 = a[3],\n a11 = a[4],\n a12 = a[5];\n var a20 = a[6],\n a21 = a[7],\n a22 = a[8];\n out[0] = a11 * a22 - a12 * a21;\n out[1] = a02 * a21 - a01 * a22;\n out[2] = a01 * a12 - a02 * a11;\n out[3] = a12 * a20 - a10 * a22;\n out[4] = a00 * a22 - a02 * a20;\n out[5] = a02 * a10 - a00 * a12;\n out[6] = a10 * a21 - a11 * a20;\n out[7] = a01 * a20 - a00 * a21;\n out[8] = a00 * a11 - a01 * a10;\n return out;\n}\n/**\n * Calculates the determinant of a mat3\n *\n * @param {ReadonlyMat3} a the source matrix\n * @returns {Number} determinant of a\n */\n\nexport function determinant(a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2];\n var a10 = a[3],\n a11 = a[4],\n a12 = a[5];\n var a20 = a[6],\n a21 = a[7],\n a22 = a[8];\n return a00 * (a22 * a11 - a12 * a21) + a01 * (-a22 * a10 + a12 * a20) + a02 * (a21 * a10 - a11 * a20);\n}\n/**\n * Multiplies two mat3's\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the first operand\n * @param {ReadonlyMat3} b the second operand\n * @returns {mat3} out\n */\n\nexport function multiply(out, a, b) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2];\n var a10 = a[3],\n a11 = a[4],\n a12 = a[5];\n var a20 = a[6],\n a21 = a[7],\n a22 = a[8];\n var b00 = b[0],\n b01 = b[1],\n b02 = b[2];\n var b10 = b[3],\n b11 = b[4],\n b12 = b[5];\n var b20 = b[6],\n b21 = b[7],\n b22 = b[8];\n out[0] = b00 * a00 + b01 * a10 + b02 * a20;\n out[1] = b00 * a01 + b01 * a11 + b02 * a21;\n out[2] = b00 * a02 + b01 * a12 + b02 * a22;\n out[3] = b10 * a00 + b11 * a10 + b12 * a20;\n out[4] = b10 * a01 + b11 * a11 + b12 * a21;\n out[5] = b10 * a02 + b11 * a12 + b12 * a22;\n out[6] = b20 * a00 + b21 * a10 + b22 * a20;\n out[7] = b20 * a01 + b21 * a11 + b22 * a21;\n out[8] = b20 * a02 + b21 * a12 + b22 * a22;\n return out;\n}\n/**\n * Translate a mat3 by the given vector\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the matrix to translate\n * @param {ReadonlyVec2} v vector to translate by\n * @returns {mat3} out\n */\n\nexport function translate(out, a, v) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a10 = a[3],\n a11 = a[4],\n a12 = a[5],\n a20 = a[6],\n a21 = a[7],\n a22 = a[8],\n x = v[0],\n y = v[1];\n out[0] = a00;\n out[1] = a01;\n out[2] = a02;\n out[3] = a10;\n out[4] = a11;\n out[5] = a12;\n out[6] = x * a00 + y * a10 + a20;\n out[7] = x * a01 + y * a11 + a21;\n out[8] = x * a02 + y * a12 + a22;\n return out;\n}\n/**\n * Rotates a mat3 by the given angle\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat3} out\n */\n\nexport function rotate(out, a, rad) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a10 = a[3],\n a11 = a[4],\n a12 = a[5],\n a20 = a[6],\n a21 = a[7],\n a22 = a[8],\n s = Math.sin(rad),\n c = Math.cos(rad);\n out[0] = c * a00 + s * a10;\n out[1] = c * a01 + s * a11;\n out[2] = c * a02 + s * a12;\n out[3] = c * a10 - s * a00;\n out[4] = c * a11 - s * a01;\n out[5] = c * a12 - s * a02;\n out[6] = a20;\n out[7] = a21;\n out[8] = a22;\n return out;\n}\n/**\n * Scales the mat3 by the dimensions in the given vec2\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the matrix to rotate\n * @param {ReadonlyVec2} v the vec2 to scale the matrix by\n * @returns {mat3} out\n **/\n\nexport function scale(out, a, v) {\n var x = v[0],\n y = v[1];\n out[0] = x * a[0];\n out[1] = x * a[1];\n out[2] = x * a[2];\n out[3] = y * a[3];\n out[4] = y * a[4];\n out[5] = y * a[5];\n out[6] = a[6];\n out[7] = a[7];\n out[8] = a[8];\n return out;\n}\n/**\n * Creates a matrix from a vector translation\n * This is equivalent to (but much faster than):\n *\n * mat3.identity(dest);\n * mat3.translate(dest, dest, vec);\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {ReadonlyVec2} v Translation vector\n * @returns {mat3} out\n */\n\nexport function fromTranslation(out, v) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = 1;\n out[5] = 0;\n out[6] = v[0];\n out[7] = v[1];\n out[8] = 1;\n return out;\n}\n/**\n * Creates a matrix from a given angle\n * This is equivalent to (but much faster than):\n *\n * mat3.identity(dest);\n * mat3.rotate(dest, dest, rad);\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat3} out\n */\n\nexport function fromRotation(out, rad) {\n var s = Math.sin(rad),\n c = Math.cos(rad);\n out[0] = c;\n out[1] = s;\n out[2] = 0;\n out[3] = -s;\n out[4] = c;\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 1;\n return out;\n}\n/**\n * Creates a matrix from a vector scaling\n * This is equivalent to (but much faster than):\n *\n * mat3.identity(dest);\n * mat3.scale(dest, dest, vec);\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {ReadonlyVec2} v Scaling vector\n * @returns {mat3} out\n */\n\nexport function fromScaling(out, v) {\n out[0] = v[0];\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = v[1];\n out[5] = 0;\n out[6] = 0;\n out[7] = 0;\n out[8] = 1;\n return out;\n}\n/**\n * Copies the values from a mat2d into a mat3\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat2d} a the matrix to copy\n * @returns {mat3} out\n **/\n\nexport function fromMat2d(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = 0;\n out[3] = a[2];\n out[4] = a[3];\n out[5] = 0;\n out[6] = a[4];\n out[7] = a[5];\n out[8] = 1;\n return out;\n}\n/**\n * Calculates a 3x3 matrix from the given quaternion\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {ReadonlyQuat} q Quaternion to create matrix from\n *\n * @returns {mat3} out\n */\n\nexport function fromQuat(out, q) {\n var x = q[0],\n y = q[1],\n z = q[2],\n w = q[3];\n var x2 = x + x;\n var y2 = y + y;\n var z2 = z + z;\n var xx = x * x2;\n var yx = y * x2;\n var yy = y * y2;\n var zx = z * x2;\n var zy = z * y2;\n var zz = z * z2;\n var wx = w * x2;\n var wy = w * y2;\n var wz = w * z2;\n out[0] = 1 - yy - zz;\n out[3] = yx - wz;\n out[6] = zx + wy;\n out[1] = yx + wz;\n out[4] = 1 - xx - zz;\n out[7] = zy - wx;\n out[2] = zx - wy;\n out[5] = zy + wx;\n out[8] = 1 - xx - yy;\n return out;\n}\n/**\n * Calculates a 3x3 normal matrix (transpose inverse) from the 4x4 matrix\n *\n * @param {mat3} out mat3 receiving operation result\n * @param {ReadonlyMat4} a Mat4 to derive the normal matrix from\n *\n * @returns {mat3} out\n */\n\nexport function normalFromMat4(out, a) {\n var a00 = a[0],\n a01 = a[1],\n a02 = a[2],\n a03 = a[3];\n var a10 = a[4],\n a11 = a[5],\n a12 = a[6],\n a13 = a[7];\n var a20 = a[8],\n a21 = a[9],\n a22 = a[10],\n a23 = a[11];\n var a30 = a[12],\n a31 = a[13],\n a32 = a[14],\n a33 = a[15];\n var b00 = a00 * a11 - a01 * a10;\n var b01 = a00 * a12 - a02 * a10;\n var b02 = a00 * a13 - a03 * a10;\n var b03 = a01 * a12 - a02 * a11;\n var b04 = a01 * a13 - a03 * a11;\n var b05 = a02 * a13 - a03 * a12;\n var b06 = a20 * a31 - a21 * a30;\n var b07 = a20 * a32 - a22 * a30;\n var b08 = a20 * a33 - a23 * a30;\n var b09 = a21 * a32 - a22 * a31;\n var b10 = a21 * a33 - a23 * a31;\n var b11 = a22 * a33 - a23 * a32; // Calculate the determinant\n\n var det = b00 * b11 - b01 * b10 + b02 * b09 + b03 * b08 - b04 * b07 + b05 * b06;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = (a11 * b11 - a12 * b10 + a13 * b09) * det;\n out[1] = (a12 * b08 - a10 * b11 - a13 * b07) * det;\n out[2] = (a10 * b10 - a11 * b08 + a13 * b06) * det;\n out[3] = (a02 * b10 - a01 * b11 - a03 * b09) * det;\n out[4] = (a00 * b11 - a02 * b08 + a03 * b07) * det;\n out[5] = (a01 * b08 - a00 * b10 - a03 * b06) * det;\n out[6] = (a31 * b05 - a32 * b04 + a33 * b03) * det;\n out[7] = (a32 * b02 - a30 * b05 - a33 * b01) * det;\n out[8] = (a30 * b04 - a31 * b02 + a33 * b00) * det;\n return out;\n}\n/**\n * Generates a 2D projection matrix with the given bounds\n *\n * @param {mat3} out mat3 frustum matrix will be written into\n * @param {number} width Width of your gl context\n * @param {number} height Height of gl context\n * @returns {mat3} out\n */\n\nexport function projection(out, width, height) {\n out[0] = 2 / width;\n out[1] = 0;\n out[2] = 0;\n out[3] = 0;\n out[4] = -2 / height;\n out[5] = 0;\n out[6] = -1;\n out[7] = 1;\n out[8] = 1;\n return out;\n}\n/**\n * Returns a string representation of a mat3\n *\n * @param {ReadonlyMat3} a matrix to represent as a string\n * @returns {String} string representation of the matrix\n */\n\nexport function str(a) {\n return \"mat3(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \", \" + a[4] + \", \" + a[5] + \", \" + a[6] + \", \" + a[7] + \", \" + a[8] + \")\";\n}\n/**\n * Returns Frobenius norm of a mat3\n *\n * @param {ReadonlyMat3} a the matrix to calculate Frobenius norm of\n * @returns {Number} Frobenius norm\n */\n\nexport function frob(a) {\n return Math.hypot(a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8]);\n}\n/**\n * Adds two mat3's\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the first operand\n * @param {ReadonlyMat3} b the second operand\n * @returns {mat3} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n out[4] = a[4] + b[4];\n out[5] = a[5] + b[5];\n out[6] = a[6] + b[6];\n out[7] = a[7] + b[7];\n out[8] = a[8] + b[8];\n return out;\n}\n/**\n * Subtracts matrix b from matrix a\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the first operand\n * @param {ReadonlyMat3} b the second operand\n * @returns {mat3} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n out[4] = a[4] - b[4];\n out[5] = a[5] - b[5];\n out[6] = a[6] - b[6];\n out[7] = a[7] - b[7];\n out[8] = a[8] - b[8];\n return out;\n}\n/**\n * Multiply each element of the matrix by a scalar.\n *\n * @param {mat3} out the receiving matrix\n * @param {ReadonlyMat3} a the matrix to scale\n * @param {Number} b amount to scale the matrix's elements by\n * @returns {mat3} out\n */\n\nexport function multiplyScalar(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n out[4] = a[4] * b;\n out[5] = a[5] * b;\n out[6] = a[6] * b;\n out[7] = a[7] * b;\n out[8] = a[8] * b;\n return out;\n}\n/**\n * Adds two mat3's after multiplying each element of the second operand by a scalar value.\n *\n * @param {mat3} out the receiving vector\n * @param {ReadonlyMat3} a the first operand\n * @param {ReadonlyMat3} b the second operand\n * @param {Number} scale the amount to scale b's elements by before adding\n * @returns {mat3} out\n */\n\nexport function multiplyScalarAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n out[4] = a[4] + b[4] * scale;\n out[5] = a[5] + b[5] * scale;\n out[6] = a[6] + b[6] * scale;\n out[7] = a[7] + b[7] * scale;\n out[8] = a[8] + b[8] * scale;\n return out;\n}\n/**\n * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyMat3} a The first matrix.\n * @param {ReadonlyMat3} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3] && a[4] === b[4] && a[5] === b[5] && a[6] === b[6] && a[7] === b[7] && a[8] === b[8];\n}\n/**\n * Returns whether or not the matrices have approximately the same elements in the same position.\n *\n * @param {ReadonlyMat3} a The first matrix.\n * @param {ReadonlyMat3} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3],\n a4 = a[4],\n a5 = a[5],\n a6 = a[6],\n a7 = a[7],\n a8 = a[8];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3],\n b4 = b[4],\n b5 = b[5],\n b6 = b[6],\n b7 = b[7],\n b8 = b[8];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3)) && Math.abs(a4 - b4) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a4), Math.abs(b4)) && Math.abs(a5 - b5) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a5), Math.abs(b5)) && Math.abs(a6 - b6) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a6), Math.abs(b6)) && Math.abs(a7 - b7) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a7), Math.abs(b7)) && Math.abs(a8 - b8) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a8), Math.abs(b8));\n}\n/**\n * Alias for {@link mat3.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link mat3.subtract}\n * @function\n */\n\nexport var sub = subtract;","import {patternUniformValues} from './pattern';\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n Uniform3f,\n UniformMatrix4f\n} from '../uniform_binding';\nimport {extend} from '../../util/util';\n\nimport type {Painter} from '../painter';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Context} from '../../gl/context';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport type {Tile} from '../../source/tile';\nimport {mat4} from 'gl-matrix';\n\nexport type FillUniformsType = {\n 'u_matrix': UniformMatrix4f;\n};\n\nexport type FillOutlineUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_world': Uniform2f;\n};\n\nexport type FillPatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n // pattern uniforms:\n 'u_texsize': Uniform2f;\n 'u_image': Uniform1i;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n};\n\nexport type FillOutlinePatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_world': Uniform2f;\n // pattern uniforms:\n 'u_texsize': Uniform2f;\n 'u_image': Uniform1i;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n};\n\nconst fillUniforms = (context: Context, locations: UniformLocations): FillUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix)\n});\n\nconst fillPatternUniforms = (context: Context, locations: UniformLocations): FillPatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper),\n 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower),\n 'u_scale': new Uniform3f(context, locations.u_scale),\n 'u_fade': new Uniform1f(context, locations.u_fade)\n});\n\nconst fillOutlineUniforms = (context: Context, locations: UniformLocations): FillOutlineUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_world': new Uniform2f(context, locations.u_world)\n});\n\nconst fillOutlinePatternUniforms = (context: Context, locations: UniformLocations): FillOutlinePatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_world': new Uniform2f(context, locations.u_world),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper),\n 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower),\n 'u_scale': new Uniform3f(context, locations.u_scale),\n 'u_fade': new Uniform1f(context, locations.u_fade)\n});\n\nconst fillUniformValues = (matrix: mat4): UniformValues => ({\n 'u_matrix': matrix\n});\n\nconst fillPatternUniformValues = (\n matrix: mat4,\n painter: Painter,\n crossfade: CrossfadeParameters,\n tile: Tile\n): UniformValues => extend(\n fillUniformValues(matrix),\n patternUniformValues(crossfade, painter, tile)\n);\n\nconst fillOutlineUniformValues = (matrix: mat4, drawingBufferSize: [number, number]): UniformValues => ({\n 'u_matrix': matrix,\n 'u_world': drawingBufferSize\n});\n\nconst fillOutlinePatternUniformValues = (\n matrix: mat4,\n painter: Painter,\n crossfade: CrossfadeParameters,\n tile: Tile,\n drawingBufferSize: [number, number]\n): UniformValues => extend(\n fillPatternUniformValues(matrix, painter, crossfade, tile),\n {\n 'u_world': drawingBufferSize\n }\n);\n\nexport {\n fillUniforms,\n fillPatternUniforms,\n fillOutlineUniforms,\n fillOutlinePatternUniforms,\n fillUniformValues,\n fillPatternUniformValues,\n fillOutlineUniformValues,\n fillOutlinePatternUniformValues\n};\n","import {Uniform1i, Uniform1f, Uniform2f, UniformMatrix4f} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {OverscaledTileID} from '../../source/tile_id';\nimport type {Tile} from '../../source/tile';\nimport type {CircleStyleLayer} from '../../style/style_layer/circle_style_layer';\nimport type {Painter} from '../painter';\n\nexport type CircleUniformsType = {\n 'u_camera_to_center_distance': Uniform1f;\n 'u_scale_with_map': Uniform1i;\n 'u_pitch_with_map': Uniform1i;\n 'u_extrude_scale': Uniform2f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n};\n\nconst circleUniforms = (context: Context, locations: UniformLocations): CircleUniformsType => ({\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_scale_with_map': new Uniform1i(context, locations.u_scale_with_map),\n 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map),\n 'u_extrude_scale': new Uniform2f(context, locations.u_extrude_scale),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix)\n});\n\nconst circleUniformValues = (\n painter: Painter,\n coord: OverscaledTileID,\n tile: Tile,\n layer: CircleStyleLayer\n): UniformValues => {\n const transform = painter.transform;\n\n let pitchWithMap: boolean, extrudeScale: [number, number];\n if (layer.paint.get('circle-pitch-alignment') === 'map') {\n const pixelRatio = pixelsToTileUnits(tile, 1, transform.zoom);\n pitchWithMap = true;\n extrudeScale = [pixelRatio, pixelRatio];\n } else {\n pitchWithMap = false;\n extrudeScale = transform.pixelsToGLUnits;\n }\n\n return {\n 'u_camera_to_center_distance': transform.cameraToCenterDistance,\n 'u_scale_with_map': +(layer.paint.get('circle-pitch-scale') === 'map'),\n 'u_matrix': painter.translatePosMatrix(\n coord.posMatrix,\n tile,\n layer.paint.get('circle-translate'),\n layer.paint.get('circle-translate-anchor')),\n 'u_pitch_with_map': +(pitchWithMap),\n 'u_device_pixel_ratio': painter.pixelRatio,\n 'u_extrude_scale': extrudeScale\n };\n};\n\nexport {circleUniforms, circleUniformValues};\n","import {Uniform1f, Uniform2f, UniformMatrix4f} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Transform} from '../../geo/transform';\nimport type {Tile} from '../../source/tile';\nimport {mat4} from 'gl-matrix';\n\nexport type CollisionUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_pixels_to_tile_units': Uniform1f;\n 'u_extrude_scale': Uniform2f;\n 'u_overscale_factor': Uniform1f;\n};\n\nexport type CollisionCircleUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_inv_matrix': UniformMatrix4f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_viewport_size': Uniform2f;\n};\n\nconst collisionUniforms = (context: Context, locations: UniformLocations): CollisionUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_pixels_to_tile_units': new Uniform1f(context, locations.u_pixels_to_tile_units),\n 'u_extrude_scale': new Uniform2f(context, locations.u_extrude_scale),\n 'u_overscale_factor': new Uniform1f(context, locations.u_overscale_factor)\n});\n\nconst collisionCircleUniforms = (context: Context, locations: UniformLocations): CollisionCircleUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_inv_matrix': new UniformMatrix4f(context, locations.u_inv_matrix),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_viewport_size': new Uniform2f(context, locations.u_viewport_size)\n});\n\nconst collisionUniformValues = (matrix: mat4, transform: Transform, tile: Tile): UniformValues => {\n const pixelRatio = pixelsToTileUnits(tile, 1, transform.zoom);\n const scale = Math.pow(2, transform.zoom - tile.tileID.overscaledZ);\n const overscaleFactor = tile.tileID.overscaleFactor();\n return {\n 'u_matrix': matrix,\n 'u_camera_to_center_distance': transform.cameraToCenterDistance,\n 'u_pixels_to_tile_units': pixelRatio,\n 'u_extrude_scale': [transform.pixelsToGLUnits[0] / (pixelRatio * scale),\n transform.pixelsToGLUnits[1] / (pixelRatio * scale)],\n 'u_overscale_factor': overscaleFactor\n };\n};\n\nconst collisionCircleUniformValues = (matrix: mat4, invMatrix: mat4, transform: Transform): UniformValues => {\n return {\n 'u_matrix': matrix,\n 'u_inv_matrix': invMatrix,\n 'u_camera_to_center_distance': transform.cameraToCenterDistance,\n 'u_viewport_size': [transform.width, transform.height]\n };\n};\n\nexport {collisionUniforms, collisionUniformValues, collisionCircleUniforms, collisionCircleUniformValues};\n","import {UniformColor, UniformMatrix4f, Uniform1i, Uniform1f} from '../uniform_binding';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {mat4} from 'gl-matrix';\n\nexport type DebugUniformsType = {\n 'u_color': UniformColor;\n 'u_matrix': UniformMatrix4f;\n 'u_overlay': Uniform1i;\n 'u_overlay_scale': Uniform1f;\n};\n\nconst debugUniforms = (context: Context, locations: UniformLocations): DebugUniformsType => ({\n 'u_color': new UniformColor(context, locations.u_color),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_overlay': new Uniform1i(context, locations.u_overlay),\n 'u_overlay_scale': new Uniform1f(context, locations.u_overlay_scale)\n});\n\nconst debugUniformValues = (matrix: mat4, color: Color, scaleRatio: number = 1): UniformValues => ({\n 'u_matrix': matrix,\n 'u_color': color,\n 'u_overlay': 0,\n 'u_overlay_scale': scaleRatio\n});\n\nexport {debugUniforms, debugUniformValues};\n","import {UniformMatrix4f} from '../uniform_binding';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport {mat4} from 'gl-matrix';\n\nexport type ClippingMaskUniformsType = {\n 'u_matrix': UniformMatrix4f;\n};\n\nconst clippingMaskUniforms = (context: Context, locations: UniformLocations): ClippingMaskUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix)\n});\n\nconst clippingMaskUniformValues = (matrix: mat4): UniformValues => ({\n 'u_matrix': matrix\n});\n\nexport {clippingMaskUniforms, clippingMaskUniformValues};\n","import {mat4} from 'gl-matrix';\n\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n UniformMatrix4f\n} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\n\nimport type {Context} from '../../gl/context';\nimport type {Tile} from '../../source/tile';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Painter} from '../painter';\nimport type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer';\n\nexport type HeatmapUniformsType = {\n 'u_extrude_scale': Uniform1f;\n 'u_intensity': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n};\n\nexport type HeatmapTextureUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_world': Uniform2f;\n 'u_image': Uniform1i;\n 'u_color_ramp': Uniform1i;\n 'u_opacity': Uniform1f;\n};\n\nconst heatmapUniforms = (context: Context, locations: UniformLocations): HeatmapUniformsType => ({\n 'u_extrude_scale': new Uniform1f(context, locations.u_extrude_scale),\n 'u_intensity': new Uniform1f(context, locations.u_intensity),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix)\n});\n\nconst heatmapTextureUniforms = (context: Context, locations: UniformLocations): HeatmapTextureUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_world': new Uniform2f(context, locations.u_world),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_color_ramp': new Uniform1i(context, locations.u_color_ramp),\n 'u_opacity': new Uniform1f(context, locations.u_opacity)\n});\n\nconst heatmapUniformValues = (matrix: mat4, tile: Tile, zoom: number, intensity: number): UniformValues => ({\n 'u_matrix': matrix,\n 'u_extrude_scale': pixelsToTileUnits(tile, 1, zoom),\n 'u_intensity': intensity\n});\n\nconst heatmapTextureUniformValues = (\n painter: Painter,\n layer: HeatmapStyleLayer,\n textureUnit: number,\n colorRampUnit: number\n): UniformValues => {\n const matrix = mat4.create();\n mat4.ortho(matrix, 0, painter.width, painter.height, 0, 0, 1);\n\n const gl = painter.context.gl;\n\n return {\n 'u_matrix': matrix,\n 'u_world': [gl.drawingBufferWidth, gl.drawingBufferHeight],\n 'u_image': textureUnit,\n 'u_color_ramp': colorRampUnit,\n 'u_opacity': layer.paint.get('heatmap-opacity')\n };\n};\n\nexport {\n heatmapUniforms,\n heatmapTextureUniforms,\n heatmapUniformValues,\n heatmapTextureUniformValues\n};\n","import {mat4} from 'gl-matrix';\n\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n UniformColor,\n UniformMatrix4f,\n Uniform4f\n} from '../uniform_binding';\nimport {EXTENT} from '../../data/extent';\nimport {MercatorCoordinate} from '../../geo/mercator_coordinate';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Tile} from '../../source/tile';\nimport type {Painter} from '../painter';\nimport type {HillshadeStyleLayer} from '../../style/style_layer/hillshade_style_layer';\nimport type {DEMData} from '../../data/dem_data';\nimport type {OverscaledTileID} from '../../source/tile_id';\n\nexport type HillshadeUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_image': Uniform1i;\n 'u_latrange': Uniform2f;\n 'u_light': Uniform2f;\n 'u_shadow': UniformColor;\n 'u_highlight': UniformColor;\n 'u_accent': UniformColor;\n};\n\nexport type HillshadePrepareUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_image': Uniform1i;\n 'u_dimension': Uniform2f;\n 'u_zoom': Uniform1f;\n 'u_unpack': Uniform4f;\n};\n\nconst hillshadeUniforms = (context: Context, locations: UniformLocations): HillshadeUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_latrange': new Uniform2f(context, locations.u_latrange),\n 'u_light': new Uniform2f(context, locations.u_light),\n 'u_shadow': new UniformColor(context, locations.u_shadow),\n 'u_highlight': new UniformColor(context, locations.u_highlight),\n 'u_accent': new UniformColor(context, locations.u_accent)\n});\n\nconst hillshadePrepareUniforms = (context: Context, locations: UniformLocations): HillshadePrepareUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_dimension': new Uniform2f(context, locations.u_dimension),\n 'u_zoom': new Uniform1f(context, locations.u_zoom),\n 'u_unpack': new Uniform4f(context, locations.u_unpack)\n});\n\nconst hillshadeUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: HillshadeStyleLayer,\n coord: OverscaledTileID\n): UniformValues => {\n const shadow = layer.paint.get('hillshade-shadow-color');\n const highlight = layer.paint.get('hillshade-highlight-color');\n const accent = layer.paint.get('hillshade-accent-color');\n\n let azimuthal = layer.paint.get('hillshade-illumination-direction') * (Math.PI / 180);\n // modify azimuthal angle by map rotation if light is anchored at the viewport\n if (layer.paint.get('hillshade-illumination-anchor') === 'viewport') {\n azimuthal -= painter.transform.angle;\n }\n const align = !painter.options.moving;\n return {\n 'u_matrix': coord ? coord.posMatrix : painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped(), align),\n 'u_image': 0,\n 'u_latrange': getTileLatRange(painter, tile.tileID),\n 'u_light': [layer.paint.get('hillshade-exaggeration'), azimuthal],\n 'u_shadow': shadow,\n 'u_highlight': highlight,\n 'u_accent': accent\n };\n};\n\nconst hillshadeUniformPrepareValues = (tileID: OverscaledTileID, dem: DEMData): UniformValues => {\n\n const stride = dem.stride;\n const matrix = mat4.create();\n // Flip rendering at y axis.\n mat4.ortho(matrix, 0, EXTENT, -EXTENT, 0, 0, 1);\n mat4.translate(matrix, matrix, [0, -EXTENT, 0]);\n\n return {\n 'u_matrix': matrix,\n 'u_image': 1,\n 'u_dimension': [stride, stride],\n 'u_zoom': tileID.overscaledZ,\n 'u_unpack': dem.getUnpackVector()\n };\n};\n\nfunction getTileLatRange(painter: Painter, tileID: OverscaledTileID) {\n // for scaling the magnitude of a points slope by its latitude\n const tilesAtZoom = Math.pow(2, tileID.canonical.z);\n const y = tileID.canonical.y;\n return [\n new MercatorCoordinate(0, y / tilesAtZoom).toLngLat().lat,\n new MercatorCoordinate(0, (y + 1) / tilesAtZoom).toLngLat().lat];\n}\n\nexport {\n hillshadeUniforms,\n hillshadePrepareUniforms,\n hillshadeUniformValues,\n hillshadeUniformPrepareValues\n};\n","import {Uniform1i, Uniform1f, Uniform2f, Uniform3f, UniformMatrix4f} from '../uniform_binding';\nimport {pixelsToTileUnits} from '../../source/pixels_to_tile_units';\nimport {extend} from '../../util/util';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Transform} from '../../geo/transform';\nimport type {Tile} from '../../source/tile';\nimport type {CrossFaded} from '../../style/properties';\nimport type {LineStyleLayer} from '../../style/style_layer/line_style_layer';\nimport type {Painter} from '../painter';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport {OverscaledTileID} from '../../source/tile_id';\n\nexport type LineUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_ratio': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_units_to_pixels': Uniform2f;\n};\n\nexport type LineGradientUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_ratio': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_units_to_pixels': Uniform2f;\n 'u_image': Uniform1i;\n 'u_image_height': Uniform1f;\n};\n\nexport type LinePatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_texsize': Uniform2f;\n 'u_ratio': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_units_to_pixels': Uniform2f;\n 'u_image': Uniform1i;\n 'u_scale': Uniform3f;\n 'u_fade': Uniform1f;\n};\n\nexport type LineSDFUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_ratio': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_units_to_pixels': Uniform2f;\n 'u_patternscale_a': Uniform2f;\n 'u_patternscale_b': Uniform2f;\n 'u_sdfgamma': Uniform1f;\n 'u_image': Uniform1i;\n 'u_tex_y_a': Uniform1f;\n 'u_tex_y_b': Uniform1f;\n 'u_mix': Uniform1f;\n};\n\nconst lineUniforms = (context: Context, locations: UniformLocations): LineUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_ratio': new Uniform1f(context, locations.u_ratio),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels)\n});\n\nconst lineGradientUniforms = (context: Context, locations: UniformLocations): LineGradientUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_ratio': new Uniform1f(context, locations.u_ratio),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_image_height': new Uniform1f(context, locations.u_image_height)\n});\n\nconst linePatternUniforms = (context: Context, locations: UniformLocations): LinePatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_ratio': new Uniform1f(context, locations.u_ratio),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels),\n 'u_scale': new Uniform3f(context, locations.u_scale),\n 'u_fade': new Uniform1f(context, locations.u_fade)\n});\n\nconst lineSDFUniforms = (context: Context, locations: UniformLocations): LineSDFUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_ratio': new Uniform1f(context, locations.u_ratio),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels),\n 'u_patternscale_a': new Uniform2f(context, locations.u_patternscale_a),\n 'u_patternscale_b': new Uniform2f(context, locations.u_patternscale_b),\n 'u_sdfgamma': new Uniform1f(context, locations.u_sdfgamma),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_tex_y_a': new Uniform1f(context, locations.u_tex_y_a),\n 'u_tex_y_b': new Uniform1f(context, locations.u_tex_y_b),\n 'u_mix': new Uniform1f(context, locations.u_mix)\n});\n\nconst lineUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: LineStyleLayer,\n coord: OverscaledTileID\n): UniformValues => {\n const transform = painter.transform;\n\n return {\n 'u_matrix': calculateMatrix(painter, tile, layer, coord),\n 'u_ratio': 1 / pixelsToTileUnits(tile, 1, transform.zoom),\n 'u_device_pixel_ratio': painter.pixelRatio,\n 'u_units_to_pixels': [\n 1 / transform.pixelsToGLUnits[0],\n 1 / transform.pixelsToGLUnits[1]\n ]\n };\n};\n\nconst lineGradientUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: LineStyleLayer,\n imageHeight: number,\n coord: OverscaledTileID\n): UniformValues => {\n return extend(lineUniformValues(painter, tile, layer, coord), {\n 'u_image': 0,\n 'u_image_height': imageHeight,\n });\n};\n\nconst linePatternUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: LineStyleLayer,\n crossfade: CrossfadeParameters,\n coord: OverscaledTileID\n): UniformValues => {\n const transform = painter.transform;\n const tileZoomRatio = calculateTileRatio(tile, transform);\n return {\n 'u_matrix': calculateMatrix(painter, tile, layer, coord),\n 'u_texsize': tile.imageAtlasTexture.size,\n // camera zoom ratio\n 'u_ratio': 1 / pixelsToTileUnits(tile, 1, transform.zoom),\n 'u_device_pixel_ratio': painter.pixelRatio,\n 'u_image': 0,\n 'u_scale': [tileZoomRatio, crossfade.fromScale, crossfade.toScale],\n 'u_fade': crossfade.t,\n 'u_units_to_pixels': [\n 1 / transform.pixelsToGLUnits[0],\n 1 / transform.pixelsToGLUnits[1]\n ]\n };\n};\n\nconst lineSDFUniformValues = (\n painter: Painter,\n tile: Tile,\n layer: LineStyleLayer,\n dasharray: CrossFaded>,\n crossfade: CrossfadeParameters,\n coord: OverscaledTileID\n): UniformValues => {\n const transform = painter.transform;\n const lineAtlas = painter.lineAtlas;\n const tileRatio = calculateTileRatio(tile, transform);\n\n const round = layer.layout.get('line-cap') === 'round';\n\n const posA = lineAtlas.getDash(dasharray.from, round);\n const posB = lineAtlas.getDash(dasharray.to, round);\n\n const widthA = posA.width * crossfade.fromScale;\n const widthB = posB.width * crossfade.toScale;\n\n return extend(lineUniformValues(painter, tile, layer, coord), {\n 'u_patternscale_a': [tileRatio / widthA, -posA.height / 2],\n 'u_patternscale_b': [tileRatio / widthB, -posB.height / 2],\n 'u_sdfgamma': lineAtlas.width / (Math.min(widthA, widthB) * 256 * painter.pixelRatio) / 2,\n 'u_image': 0,\n 'u_tex_y_a': posA.y,\n 'u_tex_y_b': posB.y,\n 'u_mix': crossfade.t\n });\n};\n\nfunction calculateTileRatio(tile: Tile, transform: Transform) {\n return 1 / pixelsToTileUnits(tile, 1, transform.tileZoom);\n}\n\nfunction calculateMatrix(painter: Painter, tile: Tile, layer: LineStyleLayer, coord: OverscaledTileID) {\n return painter.translatePosMatrix(\n coord ? coord.posMatrix : tile.tileID.posMatrix,\n tile,\n layer.paint.get('line-translate'),\n layer.paint.get('line-translate-anchor')\n );\n}\n\nexport {\n lineUniforms,\n lineGradientUniforms,\n linePatternUniforms,\n lineSDFUniforms,\n lineUniformValues,\n lineGradientUniformValues,\n linePatternUniformValues,\n lineSDFUniformValues\n};\n","import {Uniform1i, Uniform1f, Uniform2f, Uniform3f, UniformMatrix4f} from '../uniform_binding';\n\nimport type {Context} from '../../gl/context';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {RasterStyleLayer} from '../../style/style_layer/raster_style_layer';\nimport {mat4} from 'gl-matrix';\n\nexport type RasterUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_tl_parent': Uniform2f;\n 'u_scale_parent': Uniform1f;\n 'u_buffer_scale': Uniform1f;\n 'u_fade_t': Uniform1f;\n 'u_opacity': Uniform1f;\n 'u_image0': Uniform1i;\n 'u_image1': Uniform1i;\n 'u_brightness_low': Uniform1f;\n 'u_brightness_high': Uniform1f;\n 'u_saturation_factor': Uniform1f;\n 'u_contrast_factor': Uniform1f;\n 'u_spin_weights': Uniform3f;\n};\n\nconst rasterUniforms = (context: Context, locations: UniformLocations): RasterUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_tl_parent': new Uniform2f(context, locations.u_tl_parent),\n 'u_scale_parent': new Uniform1f(context, locations.u_scale_parent),\n 'u_buffer_scale': new Uniform1f(context, locations.u_buffer_scale),\n 'u_fade_t': new Uniform1f(context, locations.u_fade_t),\n 'u_opacity': new Uniform1f(context, locations.u_opacity),\n 'u_image0': new Uniform1i(context, locations.u_image0),\n 'u_image1': new Uniform1i(context, locations.u_image1),\n 'u_brightness_low': new Uniform1f(context, locations.u_brightness_low),\n 'u_brightness_high': new Uniform1f(context, locations.u_brightness_high),\n 'u_saturation_factor': new Uniform1f(context, locations.u_saturation_factor),\n 'u_contrast_factor': new Uniform1f(context, locations.u_contrast_factor),\n 'u_spin_weights': new Uniform3f(context, locations.u_spin_weights)\n});\n\nconst rasterUniformValues = (\n matrix: mat4,\n parentTL: [number, number],\n parentScaleBy: number,\n fade: {\n mix: number;\n opacity: number;\n },\n layer: RasterStyleLayer\n): UniformValues => ({\n 'u_matrix': matrix,\n 'u_tl_parent': parentTL,\n 'u_scale_parent': parentScaleBy,\n 'u_buffer_scale': 1,\n 'u_fade_t': fade.mix,\n 'u_opacity': fade.opacity * layer.paint.get('raster-opacity'),\n 'u_image0': 0,\n 'u_image1': 1,\n 'u_brightness_low': layer.paint.get('raster-brightness-min'),\n 'u_brightness_high': layer.paint.get('raster-brightness-max'),\n 'u_saturation_factor': saturationFactor(layer.paint.get('raster-saturation')),\n 'u_contrast_factor': contrastFactor(layer.paint.get('raster-contrast')),\n 'u_spin_weights': spinWeights(layer.paint.get('raster-hue-rotate'))\n});\n\nfunction spinWeights(angle) {\n angle *= Math.PI / 180;\n const s = Math.sin(angle);\n const c = Math.cos(angle);\n return [\n (2 * c + 1) / 3,\n (-Math.sqrt(3) * s - c + 1) / 3,\n (Math.sqrt(3) * s - c + 1) / 3\n ];\n}\n\nfunction contrastFactor(contrast) {\n return contrast > 0 ?\n 1 / (1 - contrast) :\n 1 + contrast;\n}\n\nfunction saturationFactor(saturation) {\n return saturation > 0 ?\n 1 - 1 / (1.001 - saturation) :\n -saturation;\n}\n\nexport {rasterUniforms, rasterUniformValues};\n","import {Uniform1i, Uniform1f, Uniform2f, UniformMatrix4f} from '../uniform_binding';\nimport {extend} from '../../util/util';\n\nimport type {Context} from '../../gl/context';\nimport type {Painter} from '../painter';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport {mat4} from 'gl-matrix';\n\nexport type SymbolIconUniformsType = {\n 'u_is_size_zoom_constant': Uniform1i;\n 'u_is_size_feature_constant': Uniform1i;\n 'u_size_t': Uniform1f;\n 'u_size': Uniform1f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_pitch': Uniform1f;\n 'u_rotate_symbol': Uniform1i;\n 'u_aspect_ratio': Uniform1f;\n 'u_fade_change': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n 'u_label_plane_matrix': UniformMatrix4f;\n 'u_coord_matrix': UniformMatrix4f;\n 'u_is_text': Uniform1i;\n 'u_pitch_with_map': Uniform1i;\n 'u_texsize': Uniform2f;\n 'u_texture': Uniform1i;\n};\n\nexport type SymbolSDFUniformsType = {\n 'u_is_size_zoom_constant': Uniform1i;\n 'u_is_size_feature_constant': Uniform1i;\n 'u_size_t': Uniform1f;\n 'u_size': Uniform1f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_pitch': Uniform1f;\n 'u_rotate_symbol': Uniform1i;\n 'u_aspect_ratio': Uniform1f;\n 'u_fade_change': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n 'u_label_plane_matrix': UniformMatrix4f;\n 'u_coord_matrix': UniformMatrix4f;\n 'u_is_text': Uniform1i;\n 'u_pitch_with_map': Uniform1i;\n 'u_texsize': Uniform2f;\n 'u_texture': Uniform1i;\n 'u_gamma_scale': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_is_halo': Uniform1i;\n};\n\nexport type symbolTextAndIconUniformsType = {\n 'u_is_size_zoom_constant': Uniform1i;\n 'u_is_size_feature_constant': Uniform1i;\n 'u_size_t': Uniform1f;\n 'u_size': Uniform1f;\n 'u_camera_to_center_distance': Uniform1f;\n 'u_pitch': Uniform1f;\n 'u_rotate_symbol': Uniform1i;\n 'u_aspect_ratio': Uniform1f;\n 'u_fade_change': Uniform1f;\n 'u_matrix': UniformMatrix4f;\n 'u_label_plane_matrix': UniformMatrix4f;\n 'u_coord_matrix': UniformMatrix4f;\n 'u_is_text': Uniform1i;\n 'u_pitch_with_map': Uniform1i;\n 'u_texsize': Uniform2f;\n 'u_texsize_icon': Uniform2f;\n 'u_texture': Uniform1i;\n 'u_texture_icon': Uniform1i;\n 'u_gamma_scale': Uniform1f;\n 'u_device_pixel_ratio': Uniform1f;\n 'u_is_halo': Uniform1i;\n};\n\nconst symbolIconUniforms = (context: Context, locations: UniformLocations): SymbolIconUniformsType => ({\n 'u_is_size_zoom_constant': new Uniform1i(context, locations.u_is_size_zoom_constant),\n 'u_is_size_feature_constant': new Uniform1i(context, locations.u_is_size_feature_constant),\n 'u_size_t': new Uniform1f(context, locations.u_size_t),\n 'u_size': new Uniform1f(context, locations.u_size),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_pitch': new Uniform1f(context, locations.u_pitch),\n 'u_rotate_symbol': new Uniform1i(context, locations.u_rotate_symbol),\n 'u_aspect_ratio': new Uniform1f(context, locations.u_aspect_ratio),\n 'u_fade_change': new Uniform1f(context, locations.u_fade_change),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_label_plane_matrix': new UniformMatrix4f(context, locations.u_label_plane_matrix),\n 'u_coord_matrix': new UniformMatrix4f(context, locations.u_coord_matrix),\n 'u_is_text': new Uniform1i(context, locations.u_is_text),\n 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_texture': new Uniform1i(context, locations.u_texture)\n});\n\nconst symbolSDFUniforms = (context: Context, locations: UniformLocations): SymbolSDFUniformsType => ({\n 'u_is_size_zoom_constant': new Uniform1i(context, locations.u_is_size_zoom_constant),\n 'u_is_size_feature_constant': new Uniform1i(context, locations.u_is_size_feature_constant),\n 'u_size_t': new Uniform1f(context, locations.u_size_t),\n 'u_size': new Uniform1f(context, locations.u_size),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_pitch': new Uniform1f(context, locations.u_pitch),\n 'u_rotate_symbol': new Uniform1i(context, locations.u_rotate_symbol),\n 'u_aspect_ratio': new Uniform1f(context, locations.u_aspect_ratio),\n 'u_fade_change': new Uniform1f(context, locations.u_fade_change),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_label_plane_matrix': new UniformMatrix4f(context, locations.u_label_plane_matrix),\n 'u_coord_matrix': new UniformMatrix4f(context, locations.u_coord_matrix),\n 'u_is_text': new Uniform1i(context, locations.u_is_text),\n 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_texture': new Uniform1i(context, locations.u_texture),\n 'u_gamma_scale': new Uniform1f(context, locations.u_gamma_scale),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_is_halo': new Uniform1i(context, locations.u_is_halo)\n});\n\nconst symbolTextAndIconUniforms = (context: Context, locations: UniformLocations): symbolTextAndIconUniformsType => ({\n 'u_is_size_zoom_constant': new Uniform1i(context, locations.u_is_size_zoom_constant),\n 'u_is_size_feature_constant': new Uniform1i(context, locations.u_is_size_feature_constant),\n 'u_size_t': new Uniform1f(context, locations.u_size_t),\n 'u_size': new Uniform1f(context, locations.u_size),\n 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance),\n 'u_pitch': new Uniform1f(context, locations.u_pitch),\n 'u_rotate_symbol': new Uniform1i(context, locations.u_rotate_symbol),\n 'u_aspect_ratio': new Uniform1f(context, locations.u_aspect_ratio),\n 'u_fade_change': new Uniform1f(context, locations.u_fade_change),\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_label_plane_matrix': new UniformMatrix4f(context, locations.u_label_plane_matrix),\n 'u_coord_matrix': new UniformMatrix4f(context, locations.u_coord_matrix),\n 'u_is_text': new Uniform1i(context, locations.u_is_text),\n 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_texsize_icon': new Uniform2f(context, locations.u_texsize_icon),\n 'u_texture': new Uniform1i(context, locations.u_texture),\n 'u_texture_icon': new Uniform1i(context, locations.u_texture_icon),\n 'u_gamma_scale': new Uniform1f(context, locations.u_gamma_scale),\n 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio),\n 'u_is_halo': new Uniform1i(context, locations.u_is_halo)\n});\n\nconst symbolIconUniformValues = (\n functionType: string,\n size: {\n uSizeT: number;\n uSize: number;\n },\n rotateInShader: boolean,\n pitchWithMap: boolean,\n painter: Painter,\n matrix: mat4,\n labelPlaneMatrix: mat4,\n glCoordMatrix: mat4,\n isText: boolean,\n texSize: [number, number]\n): UniformValues => {\n const transform = painter.transform;\n\n return {\n 'u_is_size_zoom_constant': +(functionType === 'constant' || functionType === 'source'),\n 'u_is_size_feature_constant': +(functionType === 'constant' || functionType === 'camera'),\n 'u_size_t': size ? size.uSizeT : 0,\n 'u_size': size ? size.uSize : 0,\n 'u_camera_to_center_distance': transform.cameraToCenterDistance,\n 'u_pitch': transform.pitch / 360 * 2 * Math.PI,\n 'u_rotate_symbol': +rotateInShader,\n 'u_aspect_ratio': transform.width / transform.height,\n 'u_fade_change': painter.options.fadeDuration ? painter.symbolFadeChange : 1,\n 'u_matrix': matrix,\n 'u_label_plane_matrix': labelPlaneMatrix,\n 'u_coord_matrix': glCoordMatrix,\n 'u_is_text': +isText,\n 'u_pitch_with_map': +pitchWithMap,\n 'u_texsize': texSize,\n 'u_texture': 0\n };\n};\n\nconst symbolSDFUniformValues = (\n functionType: string,\n size: {\n uSizeT: number;\n uSize: number;\n },\n rotateInShader: boolean,\n pitchWithMap: boolean,\n painter: Painter,\n matrix: mat4,\n labelPlaneMatrix: mat4,\n glCoordMatrix: mat4,\n isText: boolean,\n texSize: [number, number],\n isHalo: boolean\n): UniformValues => {\n const transform = painter.transform;\n\n return extend(symbolIconUniformValues(functionType, size,\n rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix,\n glCoordMatrix, isText, texSize), {\n 'u_gamma_scale': (pitchWithMap ? Math.cos(transform._pitch) * transform.cameraToCenterDistance : 1),\n 'u_device_pixel_ratio': painter.pixelRatio,\n 'u_is_halo': +isHalo\n });\n};\n\nconst symbolTextAndIconUniformValues = (\n functionType: string,\n size: {\n uSizeT: number;\n uSize: number;\n },\n rotateInShader: boolean,\n pitchWithMap: boolean,\n painter: Painter,\n matrix: mat4,\n labelPlaneMatrix: mat4,\n glCoordMatrix: mat4,\n texSizeSDF: [number, number],\n texSizeIcon: [number, number]\n): UniformValues => {\n return extend(symbolSDFUniformValues(functionType, size,\n rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix,\n glCoordMatrix, true, texSizeSDF, true), {\n 'u_texsize_icon': texSizeIcon,\n 'u_texture_icon': 1\n });\n};\n\nexport {symbolIconUniforms, symbolSDFUniforms, symbolIconUniformValues, symbolSDFUniformValues, symbolTextAndIconUniformValues, symbolTextAndIconUniforms};\n","import {bgPatternUniformValues} from './pattern';\nimport {\n Uniform1i,\n Uniform1f,\n Uniform2f,\n UniformColor,\n UniformMatrix4f\n} from '../uniform_binding';\nimport {extend} from '../../util/util';\n\nimport type {Painter} from '../painter';\nimport type {UniformValues, UniformLocations} from '../uniform_binding';\nimport type {Context} from '../../gl/context';\nimport type {Color, ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport type {CrossFaded} from '../../style/properties';\nimport type {CrossfadeParameters} from '../../style/evaluation_parameters';\nimport type {OverscaledTileID} from '../../source/tile_id';\nimport {mat4} from 'gl-matrix';\n\nexport type BackgroundUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_opacity': Uniform1f;\n 'u_color': UniformColor;\n};\n\nexport type BackgroundPatternUniformsType = {\n 'u_matrix': UniformMatrix4f;\n 'u_opacity': Uniform1f;\n // pattern uniforms:\n 'u_image': Uniform1i;\n 'u_pattern_tl_a': Uniform2f;\n 'u_pattern_br_a': Uniform2f;\n 'u_pattern_tl_b': Uniform2f;\n 'u_pattern_br_b': Uniform2f;\n 'u_texsize': Uniform2f;\n 'u_mix': Uniform1f;\n 'u_pattern_size_a': Uniform2f;\n 'u_pattern_size_b': Uniform2f;\n 'u_scale_a': Uniform1f;\n 'u_scale_b': Uniform1f;\n 'u_pixel_coord_upper': Uniform2f;\n 'u_pixel_coord_lower': Uniform2f;\n 'u_tile_units_to_pixels': Uniform1f;\n};\n\nconst backgroundUniforms = (context: Context, locations: UniformLocations): BackgroundUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_opacity': new Uniform1f(context, locations.u_opacity),\n 'u_color': new UniformColor(context, locations.u_color)\n});\n\nconst backgroundPatternUniforms = (context: Context, locations: UniformLocations): BackgroundPatternUniformsType => ({\n 'u_matrix': new UniformMatrix4f(context, locations.u_matrix),\n 'u_opacity': new Uniform1f(context, locations.u_opacity),\n 'u_image': new Uniform1i(context, locations.u_image),\n 'u_pattern_tl_a': new Uniform2f(context, locations.u_pattern_tl_a),\n 'u_pattern_br_a': new Uniform2f(context, locations.u_pattern_br_a),\n 'u_pattern_tl_b': new Uniform2f(context, locations.u_pattern_tl_b),\n 'u_pattern_br_b': new Uniform2f(context, locations.u_pattern_br_b),\n 'u_texsize': new Uniform2f(context, locations.u_texsize),\n 'u_mix': new Uniform1f(context, locations.u_mix),\n 'u_pattern_size_a': new Uniform2f(context, locations.u_pattern_size_a),\n 'u_pattern_size_b': new Uniform2f(context, locations.u_pattern_size_b),\n 'u_scale_a': new Uniform1f(context, locations.u_scale_a),\n 'u_scale_b': new Uniform1f(context, locations.u_scale_b),\n 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper),\n 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower),\n 'u_tile_units_to_pixels': new Uniform1f(context, locations.u_tile_units_to_pixels)\n});\n\nconst backgroundUniformValues = (matrix: mat4, opacity: number, color: Color): UniformValues => ({\n 'u_matrix': matrix,\n 'u_opacity': opacity,\n 'u_color': color\n});\n\nconst backgroundPatternUniformValues = (\n matrix: mat4,\n opacity: number,\n painter: Painter,\n image: CrossFaded,\n tile: {\n tileID: OverscaledTileID;\n tileSize: number;\n },\n crossfade: CrossfadeParameters\n): UniformValues => extend(\n bgPatternUniformValues(image, crossfade, painter, tile),\n {\n 'u_matrix': matrix,\n 'u_opacity': opacity\n }\n);\n\nexport {\n backgroundUniforms,\n backgroundPatternUniforms,\n backgroundUniformValues,\n backgroundPatternUniformValues\n};\n","import {fillExtrusionUniforms, fillExtrusionPatternUniforms} from './fill_extrusion_program';\nimport {fillUniforms, fillPatternUniforms, fillOutlineUniforms, fillOutlinePatternUniforms} from './fill_program';\nimport {circleUniforms} from './circle_program';\nimport {collisionUniforms, collisionCircleUniforms} from './collision_program';\nimport {debugUniforms} from './debug_program';\nimport {clippingMaskUniforms} from './clipping_mask_program';\nimport {heatmapUniforms, heatmapTextureUniforms} from './heatmap_program';\nimport {hillshadeUniforms, hillshadePrepareUniforms} from './hillshade_program';\nimport {lineUniforms, lineGradientUniforms, linePatternUniforms, lineSDFUniforms} from './line_program';\nimport {rasterUniforms} from './raster_program';\nimport {symbolIconUniforms, symbolSDFUniforms, symbolTextAndIconUniforms} from './symbol_program';\nimport {backgroundUniforms, backgroundPatternUniforms} from './background_program';\nimport {terrainUniforms, terrainDepthUniforms, terrainCoordsUniforms} from './terrain_program';\n\nexport const programUniforms = {\n fillExtrusion: fillExtrusionUniforms,\n fillExtrusionPattern: fillExtrusionPatternUniforms,\n fill: fillUniforms,\n fillPattern: fillPatternUniforms,\n fillOutline: fillOutlineUniforms,\n fillOutlinePattern: fillOutlinePatternUniforms,\n circle: circleUniforms,\n collisionBox: collisionUniforms,\n collisionCircle: collisionCircleUniforms,\n debug: debugUniforms,\n clippingMask: clippingMaskUniforms,\n heatmap: heatmapUniforms,\n heatmapTexture: heatmapTextureUniforms,\n hillshade: hillshadeUniforms,\n hillshadePrepare: hillshadePrepareUniforms,\n line: lineUniforms,\n lineGradient: lineGradientUniforms,\n linePattern: linePatternUniforms,\n lineSDF: lineSDFUniforms,\n raster: rasterUniforms,\n symbolIcon: symbolIconUniforms,\n symbolSDF: symbolSDFUniforms,\n symbolTextAndIcon: symbolTextAndIconUniforms,\n background: backgroundUniforms,\n backgroundPattern: backgroundPatternUniforms,\n terrain: terrainUniforms,\n terrainDepth: terrainDepthUniforms,\n terrainCoords: terrainCoordsUniforms\n};\n","\nimport type {StructArray} from '../util/struct_array';\nimport type {TriangleIndexArray, LineIndexArray, LineStripIndexArray} from '../data/index_array_type';\nimport type {Context} from '../gl/context';\n\n/**\n * @internal\n * an index buffer class\n */\nexport class IndexBuffer {\n context: Context;\n buffer: WebGLBuffer;\n dynamicDraw: boolean;\n\n constructor(context: Context, array: TriangleIndexArray | LineIndexArray | LineStripIndexArray, dynamicDraw?: boolean) {\n this.context = context;\n const gl = context.gl;\n this.buffer = gl.createBuffer();\n this.dynamicDraw = Boolean(dynamicDraw);\n\n // The bound index buffer is part of vertex array object state. We don't want to\n // modify whatever VAO happens to be currently bound, so make sure the default\n // vertex array provided by the context is bound instead.\n this.context.unbindVAO();\n\n context.bindElementBuffer.set(this.buffer);\n gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, array.arrayBuffer, this.dynamicDraw ? gl.DYNAMIC_DRAW : gl.STATIC_DRAW);\n\n if (!this.dynamicDraw) {\n delete array.arrayBuffer;\n }\n }\n\n bind() {\n this.context.bindElementBuffer.set(this.buffer);\n }\n\n updateData(array: StructArray) {\n const gl = this.context.gl;\n if (!this.dynamicDraw) throw new Error('Attempted to update data while not in dynamic mode.');\n // The right VAO will get this buffer re-bound later in VertexArrayObject#bind\n // See https://github.com/mapbox/mapbox-gl-js/issues/5620\n this.context.unbindVAO();\n this.bind();\n gl.bufferSubData(gl.ELEMENT_ARRAY_BUFFER, 0, array.arrayBuffer);\n }\n\n destroy() {\n const gl = this.context.gl;\n if (this.buffer) {\n gl.deleteBuffer(this.buffer);\n delete this.buffer;\n }\n }\n}\n","\nimport type {\n StructArray,\n StructArrayMember\n} from '../util/struct_array';\n\nimport type {Program} from '../render/program';\nimport type {Context} from '../gl/context';\n\n/**\n * An Enum for AttributeType\n */\nconst AttributeType = {\n Int8: 'BYTE',\n Uint8: 'UNSIGNED_BYTE',\n Int16: 'SHORT',\n Uint16: 'UNSIGNED_SHORT',\n Int32: 'INT',\n Uint32: 'UNSIGNED_INT',\n Float32: 'FLOAT'\n};\n\n/**\n * @internal\n * The `VertexBuffer` class turns a `StructArray` into a WebGL buffer. Each member of the StructArray's\n * Struct type is converted to a WebGL attribute.\n */\nexport class VertexBuffer {\n length: number;\n attributes: ReadonlyArray;\n itemSize: number;\n dynamicDraw: boolean;\n context: Context;\n buffer: WebGLBuffer;\n\n /**\n * @param dynamicDraw - Whether this buffer will be repeatedly updated.\n */\n constructor(context: Context, array: StructArray, attributes: ReadonlyArray, dynamicDraw?: boolean) {\n this.length = array.length;\n this.attributes = attributes;\n this.itemSize = array.bytesPerElement;\n this.dynamicDraw = dynamicDraw;\n\n this.context = context;\n const gl = context.gl;\n this.buffer = gl.createBuffer();\n context.bindVertexBuffer.set(this.buffer);\n gl.bufferData(gl.ARRAY_BUFFER, array.arrayBuffer, this.dynamicDraw ? gl.DYNAMIC_DRAW : gl.STATIC_DRAW);\n\n if (!this.dynamicDraw) {\n delete array.arrayBuffer;\n }\n }\n\n bind() {\n this.context.bindVertexBuffer.set(this.buffer);\n }\n\n updateData(array: StructArray) {\n if (array.length !== this.length) throw new Error(`Length of new data is ${array.length}, which doesn't match current length of ${this.length}`);\n const gl = this.context.gl;\n this.bind();\n gl.bufferSubData(gl.ARRAY_BUFFER, 0, array.arrayBuffer);\n }\n\n enableAttributes(gl: WebGLRenderingContext|WebGL2RenderingContext, program: Program) {\n for (let j = 0; j < this.attributes.length; j++) {\n const member = this.attributes[j];\n const attribIndex: number | void = program.attributes[member.name];\n if (attribIndex !== undefined) {\n gl.enableVertexAttribArray(attribIndex);\n }\n }\n }\n\n /**\n * Set the attribute pointers in a WebGL context\n * @param gl - The WebGL context\n * @param program - The active WebGL program\n * @param vertexOffset - Index of the starting vertex of the segment\n */\n setVertexAttribPointers(gl: WebGLRenderingContext|WebGL2RenderingContext, program: Program, vertexOffset?: number | null) {\n for (let j = 0; j < this.attributes.length; j++) {\n const member = this.attributes[j];\n const attribIndex: number | void = program.attributes[member.name];\n\n if (attribIndex !== undefined) {\n gl.vertexAttribPointer(\n attribIndex,\n member.components,\n (gl as any)[AttributeType[member.type]],\n false,\n this.itemSize,\n member.offset + (this.itemSize * (vertexOffset || 0))\n );\n }\n }\n }\n\n /**\n * Destroy the GL buffer bound to the given WebGL context\n */\n destroy() {\n const gl = this.context.gl;\n if (this.buffer) {\n gl.deleteBuffer(this.buffer);\n delete this.buffer;\n }\n }\n}\n","const cache = new WeakMap();\nexport function isWebGL2(\n gl: WebGLRenderingContext | WebGL2RenderingContext\n): gl is WebGL2RenderingContext {\n if (cache.has(gl)) {\n return cache.get(gl);\n } else {\n const value = gl.getParameter(gl.VERSION)?.startsWith('WebGL 2.0');\n cache.set(gl, value);\n return value;\n }\n}\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {isWebGL2} from './webgl2';\n\nimport type {Context} from './context';\nimport type {\n BlendFuncType,\n BlendEquationType,\n ColorMaskType,\n DepthRangeType,\n DepthMaskType,\n StencilFuncType,\n StencilOpType,\n DepthFuncType,\n TextureUnitType,\n ViewportType,\n CullFaceModeType,\n FrontFaceType,\n} from './types';\n\nexport interface IValue {\n current: T;\n default: T;\n dirty: boolean;\n get(): T;\n setDefault(): void;\n set(value: T): void;\n}\n\nclass BaseValue implements IValue {\n gl: WebGLRenderingContext|WebGL2RenderingContext;\n current: T;\n default: T;\n dirty: boolean;\n\n constructor(context: Context) {\n this.gl = context.gl;\n this.default = this.getDefault();\n this.current = this.default;\n this.dirty = false;\n }\n\n get(): T {\n return this.current;\n }\n set(value: T) { // eslint-disable-line\n // overridden in child classes;\n }\n\n getDefault(): T {\n return this.default; // overriden in child classes\n }\n setDefault() {\n this.set(this.default);\n }\n}\n\nexport class ClearColor extends BaseValue {\n getDefault(): Color {\n return Color.transparent;\n }\n set(v: Color) {\n const c = this.current;\n if (v.r === c.r && v.g === c.g && v.b === c.b && v.a === c.a && !this.dirty) return;\n this.gl.clearColor(v.r, v.g, v.b, v.a);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ClearDepth extends BaseValue {\n getDefault(): number {\n return 1;\n }\n set(v: number) {\n if (v === this.current && !this.dirty) return;\n this.gl.clearDepth(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ClearStencil extends BaseValue {\n getDefault(): number {\n return 0;\n }\n set(v: number) {\n if (v === this.current && !this.dirty) return;\n this.gl.clearStencil(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ColorMask extends BaseValue {\n getDefault(): ColorMaskType {\n return [true, true, true, true];\n }\n set(v: ColorMaskType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && v[3] === c[3] && !this.dirty) return;\n this.gl.colorMask(v[0], v[1], v[2], v[3]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthMask extends BaseValue {\n getDefault(): DepthMaskType {\n return true;\n }\n set(v: DepthMaskType): void {\n if (v === this.current && !this.dirty) return;\n this.gl.depthMask(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class StencilMask extends BaseValue {\n getDefault(): number {\n return 0xFF;\n }\n set(v: number): void {\n if (v === this.current && !this.dirty) return;\n this.gl.stencilMask(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class StencilFunc extends BaseValue {\n getDefault(): StencilFuncType {\n return {\n func: this.gl.ALWAYS,\n ref: 0,\n mask: 0xFF\n };\n }\n set(v: StencilFuncType): void {\n const c = this.current;\n if (v.func === c.func && v.ref === c.ref && v.mask === c.mask && !this.dirty) return;\n this.gl.stencilFunc(v.func, v.ref, v.mask);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class StencilOp extends BaseValue {\n getDefault(): StencilOpType {\n const gl = this.gl;\n return [gl.KEEP, gl.KEEP, gl.KEEP];\n }\n set(v: StencilOpType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && !this.dirty) return;\n this.gl.stencilOp(v[0], v[1], v[2]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class StencilTest extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n if (v) {\n gl.enable(gl.STENCIL_TEST);\n } else {\n gl.disable(gl.STENCIL_TEST);\n }\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthRange extends BaseValue {\n getDefault(): DepthRangeType {\n return [0, 1];\n }\n set(v: DepthRangeType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && !this.dirty) return;\n this.gl.depthRange(v[0], v[1]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthTest extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n if (v) {\n gl.enable(gl.DEPTH_TEST);\n } else {\n gl.disable(gl.DEPTH_TEST);\n }\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthFunc extends BaseValue {\n getDefault(): DepthFuncType {\n return this.gl.LESS;\n }\n set(v: DepthFuncType) {\n if (v === this.current && !this.dirty) return;\n this.gl.depthFunc(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class Blend extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n if (v) {\n gl.enable(gl.BLEND);\n } else {\n gl.disable(gl.BLEND);\n }\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BlendFunc extends BaseValue {\n getDefault(): BlendFuncType {\n const gl = this.gl;\n return [gl.ONE, gl.ZERO];\n }\n set(v: BlendFuncType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && !this.dirty) return;\n this.gl.blendFunc(v[0], v[1]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BlendColor extends BaseValue {\n getDefault(): Color {\n return Color.transparent;\n }\n set(v: Color) {\n const c = this.current;\n if (v.r === c.r && v.g === c.g && v.b === c.b && v.a === c.a && !this.dirty) return;\n this.gl.blendColor(v.r, v.g, v.b, v.a);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BlendEquation extends BaseValue {\n getDefault(): BlendEquationType {\n return this.gl.FUNC_ADD;\n }\n set(v: BlendEquationType) {\n if (v === this.current && !this.dirty) return;\n this.gl.blendEquation(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class CullFace extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n if (v) {\n gl.enable(gl.CULL_FACE);\n } else {\n gl.disable(gl.CULL_FACE);\n }\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class CullFaceSide extends BaseValue {\n getDefault(): CullFaceModeType {\n return this.gl.BACK;\n }\n set(v: CullFaceModeType) {\n if (v === this.current && !this.dirty) return;\n this.gl.cullFace(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class FrontFace extends BaseValue {\n getDefault(): FrontFaceType {\n return this.gl.CCW;\n }\n set(v: FrontFaceType) {\n if (v === this.current && !this.dirty) return;\n this.gl.frontFace(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ProgramValue extends BaseValue {\n getDefault(): WebGLProgram {\n return null;\n }\n set(v?: WebGLProgram | null) {\n if (v === this.current && !this.dirty) return;\n this.gl.useProgram(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class ActiveTextureUnit extends BaseValue {\n getDefault(): TextureUnitType {\n return this.gl.TEXTURE0;\n }\n set(v: TextureUnitType) {\n if (v === this.current && !this.dirty) return;\n this.gl.activeTexture(v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class Viewport extends BaseValue {\n getDefault(): ViewportType {\n const gl = this.gl;\n return [0, 0, gl.drawingBufferWidth, gl.drawingBufferHeight];\n }\n set(v: ViewportType) {\n const c = this.current;\n if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && v[3] === c[3] && !this.dirty) return;\n this.gl.viewport(v[0], v[1], v[2], v[3]);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindFramebuffer extends BaseValue {\n getDefault(): WebGLFramebuffer {\n return null;\n }\n set(v?: WebGLFramebuffer | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.bindFramebuffer(gl.FRAMEBUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindRenderbuffer extends BaseValue {\n getDefault(): WebGLRenderbuffer {\n return null;\n }\n set(v?: WebGLRenderbuffer | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.bindRenderbuffer(gl.RENDERBUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindTexture extends BaseValue {\n getDefault(): WebGLTexture {\n return null;\n }\n set(v?: WebGLTexture | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.bindTexture(gl.TEXTURE_2D, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindVertexBuffer extends BaseValue {\n getDefault(): WebGLBuffer {\n return null;\n }\n set(v?: WebGLBuffer | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.bindBuffer(gl.ARRAY_BUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindElementBuffer extends BaseValue {\n getDefault(): WebGLBuffer {\n return null;\n }\n set(v?: WebGLBuffer | null) {\n // Always rebind\n const gl = this.gl;\n gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class BindVertexArray extends BaseValue {\n getDefault(): WebGLVertexArrayObject | null {\n return null;\n }\n set(v: WebGLVertexArrayObject | null) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n\n if (isWebGL2(gl)) {\n gl.bindVertexArray(v);\n } else {\n gl.getExtension('OES_vertex_array_object')?.bindVertexArrayOES(v);\n }\n\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class PixelStoreUnpack extends BaseValue {\n getDefault(): number {\n return 4;\n }\n set(v: number) {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.pixelStorei(gl.UNPACK_ALIGNMENT, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class PixelStoreUnpackPremultiplyAlpha extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean): void {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, ((v as any)));\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class PixelStoreUnpackFlipY extends BaseValue {\n getDefault(): boolean {\n return false;\n }\n set(v: boolean): void {\n if (v === this.current && !this.dirty) return;\n const gl = this.gl;\n gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, ((v as any)));\n this.current = v;\n this.dirty = false;\n }\n}\n\nclass FramebufferAttachment extends BaseValue {\n parent: WebGLFramebuffer;\n context: Context;\n\n constructor(context: Context, parent: WebGLFramebuffer) {\n super(context);\n this.context = context;\n this.parent = parent;\n }\n getDefault() {\n return null;\n }\n}\n\nexport class ColorAttachment extends FramebufferAttachment {\n setDirty() {\n this.dirty = true;\n }\n set(v?: WebGLTexture | null): void {\n if (v === this.current && !this.dirty) return;\n this.context.bindFramebuffer.set(this.parent);\n // note: it's possible to attach a renderbuffer to the color\n // attachment point, but thus far MBGL only uses textures for color\n const gl = this.gl;\n gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, v, 0);\n\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthAttachment extends FramebufferAttachment {\n set(v?: WebGLRenderbuffer | null): void {\n if (v === this.current && !this.dirty) return;\n this.context.bindFramebuffer.set(this.parent);\n // note: it's possible to attach a texture to the depth attachment\n // point, but thus far MBGL only uses renderbuffers for depth\n const gl = this.gl;\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_ATTACHMENT, gl.RENDERBUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n\nexport class DepthStencilAttachment extends FramebufferAttachment {\n set(v?: WebGLRenderbuffer | null): void {\n if (v === this.current && !this.dirty) return;\n this.context.bindFramebuffer.set(this.parent);\n // note: it's possible to attach a texture to the depth attachment\n // point, but thus far MBGL only uses renderbuffers for depth\n const gl = this.gl;\n gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, v);\n this.current = v;\n this.dirty = false;\n }\n}\n","import {ColorAttachment, DepthAttachment, DepthStencilAttachment} from './value';\n\nimport type {Context} from './context';\n\n/**\n * @internal\n * A framebuffer holder object\n */\nexport class Framebuffer {\n context: Context;\n width: number;\n height: number;\n framebuffer: WebGLFramebuffer;\n colorAttachment: ColorAttachment;\n depthAttachment: DepthAttachment;\n\n constructor(context: Context, width: number, height: number, hasDepth: boolean, hasStencil: boolean) {\n this.context = context;\n this.width = width;\n this.height = height;\n const gl = context.gl;\n const fbo = this.framebuffer = gl.createFramebuffer();\n\n this.colorAttachment = new ColorAttachment(context, fbo);\n if (hasDepth) {\n this.depthAttachment = hasStencil ? new DepthStencilAttachment(context, fbo) : new DepthAttachment(context, fbo);\n } else if (hasStencil) {\n throw new Error('Stencil cannot be setted without depth');\n }\n if (gl.checkFramebufferStatus(gl.FRAMEBUFFER) !== gl.FRAMEBUFFER_COMPLETE) {\n throw new Error('Framebuffer is not complete');\n }\n }\n\n destroy() {\n const gl = this.context.gl;\n\n const texture = this.colorAttachment.get();\n if (texture) gl.deleteTexture(texture);\n\n if (this.depthAttachment) {\n const renderbuffer = this.depthAttachment.get();\n if (renderbuffer) gl.deleteRenderbuffer(renderbuffer);\n }\n\n gl.deleteFramebuffer(this.framebuffer);\n }\n}\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\n\nimport type {BlendFuncType, ColorMaskType} from './types';\n\nconst ZERO = 0x0000;\nconst ONE = 0x0001;\nconst ONE_MINUS_SRC_ALPHA = 0x0303;\n\nexport class ColorMode {\n blendFunction: BlendFuncType;\n blendColor: Color;\n mask: ColorMaskType;\n\n constructor(blendFunction: BlendFuncType, blendColor: Color, mask: ColorMaskType) {\n this.blendFunction = blendFunction;\n this.blendColor = blendColor;\n this.mask = mask;\n }\n\n static Replace: BlendFuncType;\n\n static disabled: Readonly;\n static unblended: Readonly;\n static alphaBlended: Readonly;\n}\n\nColorMode.Replace = [ONE, ZERO];\n\nColorMode.disabled = new ColorMode(ColorMode.Replace, Color.transparent, [false, false, false, false]);\nColorMode.unblended = new ColorMode(ColorMode.Replace, Color.transparent, [true, true, true, true]);\nColorMode.alphaBlended = new ColorMode([ONE, ONE_MINUS_SRC_ALPHA], Color.transparent, [true, true, true, true]);\n","import {IndexBuffer} from './index_buffer';\n\nimport {VertexBuffer} from './vertex_buffer';\nimport {Framebuffer} from './framebuffer';\nimport {DepthMode} from './depth_mode';\nimport {StencilMode} from './stencil_mode';\nimport {ColorMode} from './color_mode';\nimport {CullFaceMode} from './cull_face_mode';\nimport {deepEqual} from '../util/util';\nimport {ClearColor, ClearDepth, ClearStencil, ColorMask, DepthMask, StencilMask, StencilFunc, StencilOp, StencilTest, DepthRange, DepthTest, DepthFunc, Blend, BlendFunc, BlendColor, BlendEquation, CullFace, CullFaceSide, FrontFace, ProgramValue, ActiveTextureUnit, Viewport, BindFramebuffer, BindRenderbuffer, BindTexture, BindVertexBuffer, BindElementBuffer, BindVertexArray, PixelStoreUnpack, PixelStoreUnpackPremultiplyAlpha, PixelStoreUnpackFlipY} from './value';\n\nimport type {TriangleIndexArray, LineIndexArray, LineStripIndexArray} from '../data/index_array_type';\nimport type {\n StructArray,\n StructArrayMember\n} from '../util/struct_array';\nimport type {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {isWebGL2} from './webgl2';\n\ntype ClearArgs = {\n color?: Color;\n depth?: number;\n stencil?: number;\n};\n\n/**\n * @internal\n * A webgl wrapper class to allow injection, mocking and abstaction\n */\nexport class Context {\n gl: WebGLRenderingContext | WebGL2RenderingContext;\n\n currentNumAttributes: number;\n maxTextureSize: number;\n\n clearColor: ClearColor;\n clearDepth: ClearDepth;\n clearStencil: ClearStencil;\n colorMask: ColorMask;\n depthMask: DepthMask;\n stencilMask: StencilMask;\n stencilFunc: StencilFunc;\n stencilOp: StencilOp;\n stencilTest: StencilTest;\n depthRange: DepthRange;\n depthTest: DepthTest;\n depthFunc: DepthFunc;\n blend: Blend;\n blendFunc: BlendFunc;\n blendColor: BlendColor;\n blendEquation: BlendEquation;\n cullFace: CullFace;\n cullFaceSide: CullFaceSide;\n frontFace: FrontFace;\n program: ProgramValue;\n activeTexture: ActiveTextureUnit;\n viewport: Viewport;\n bindFramebuffer: BindFramebuffer;\n bindRenderbuffer: BindRenderbuffer;\n bindTexture: BindTexture;\n bindVertexBuffer: BindVertexBuffer;\n bindElementBuffer: BindElementBuffer;\n bindVertexArray: BindVertexArray;\n pixelStoreUnpack: PixelStoreUnpack;\n pixelStoreUnpackPremultiplyAlpha: PixelStoreUnpackPremultiplyAlpha;\n pixelStoreUnpackFlipY: PixelStoreUnpackFlipY;\n\n // eslint-disable-next-line camelcase\n extTextureFilterAnisotropic: EXT_texture_filter_anisotropic | null;\n extTextureFilterAnisotropicMax?: GLfloat;\n HALF_FLOAT?: GLenum;\n RGBA16F?: GLenum;\n RGB16F?: GLenum;\n\n constructor(gl: WebGLRenderingContext | WebGL2RenderingContext) {\n this.gl = gl;\n this.clearColor = new ClearColor(this);\n this.clearDepth = new ClearDepth(this);\n this.clearStencil = new ClearStencil(this);\n this.colorMask = new ColorMask(this);\n this.depthMask = new DepthMask(this);\n this.stencilMask = new StencilMask(this);\n this.stencilFunc = new StencilFunc(this);\n this.stencilOp = new StencilOp(this);\n this.stencilTest = new StencilTest(this);\n this.depthRange = new DepthRange(this);\n this.depthTest = new DepthTest(this);\n this.depthFunc = new DepthFunc(this);\n this.blend = new Blend(this);\n this.blendFunc = new BlendFunc(this);\n this.blendColor = new BlendColor(this);\n this.blendEquation = new BlendEquation(this);\n this.cullFace = new CullFace(this);\n this.cullFaceSide = new CullFaceSide(this);\n this.frontFace = new FrontFace(this);\n this.program = new ProgramValue(this);\n this.activeTexture = new ActiveTextureUnit(this);\n this.viewport = new Viewport(this);\n this.bindFramebuffer = new BindFramebuffer(this);\n this.bindRenderbuffer = new BindRenderbuffer(this);\n this.bindTexture = new BindTexture(this);\n this.bindVertexBuffer = new BindVertexBuffer(this);\n this.bindElementBuffer = new BindElementBuffer(this);\n this.bindVertexArray = new BindVertexArray(this);\n this.pixelStoreUnpack = new PixelStoreUnpack(this);\n this.pixelStoreUnpackPremultiplyAlpha = new PixelStoreUnpackPremultiplyAlpha(this);\n this.pixelStoreUnpackFlipY = new PixelStoreUnpackFlipY(this);\n\n this.extTextureFilterAnisotropic = (\n gl.getExtension('EXT_texture_filter_anisotropic') ||\n gl.getExtension('MOZ_EXT_texture_filter_anisotropic') ||\n gl.getExtension('WEBKIT_EXT_texture_filter_anisotropic')\n );\n\n if (this.extTextureFilterAnisotropic) {\n this.extTextureFilterAnisotropicMax = gl.getParameter(this.extTextureFilterAnisotropic.MAX_TEXTURE_MAX_ANISOTROPY_EXT);\n }\n\n this.maxTextureSize = gl.getParameter(gl.MAX_TEXTURE_SIZE);\n\n if (isWebGL2(gl)) {\n this.HALF_FLOAT = gl.HALF_FLOAT;\n const extColorBufferHalfFloat = gl.getExtension('EXT_color_buffer_half_float');\n this.RGBA16F = gl.RGBA16F ?? extColorBufferHalfFloat?.RGBA16F_EXT;\n this.RGB16F = gl.RGB16F ?? extColorBufferHalfFloat?.RGB16F_EXT;\n gl.getExtension('EXT_color_buffer_float');\n } else {\n gl.getExtension('EXT_color_buffer_half_float');\n gl.getExtension('OES_texture_half_float_linear');\n const extTextureHalfFloat = gl.getExtension('OES_texture_half_float');\n this.HALF_FLOAT = extTextureHalfFloat?.HALF_FLOAT_OES;\n }\n }\n\n setDefault() {\n this.unbindVAO();\n\n this.clearColor.setDefault();\n this.clearDepth.setDefault();\n this.clearStencil.setDefault();\n this.colorMask.setDefault();\n this.depthMask.setDefault();\n this.stencilMask.setDefault();\n this.stencilFunc.setDefault();\n this.stencilOp.setDefault();\n this.stencilTest.setDefault();\n this.depthRange.setDefault();\n this.depthTest.setDefault();\n this.depthFunc.setDefault();\n this.blend.setDefault();\n this.blendFunc.setDefault();\n this.blendColor.setDefault();\n this.blendEquation.setDefault();\n this.cullFace.setDefault();\n this.cullFaceSide.setDefault();\n this.frontFace.setDefault();\n this.program.setDefault();\n this.activeTexture.setDefault();\n this.bindFramebuffer.setDefault();\n this.pixelStoreUnpack.setDefault();\n this.pixelStoreUnpackPremultiplyAlpha.setDefault();\n this.pixelStoreUnpackFlipY.setDefault();\n }\n\n setDirty() {\n this.clearColor.dirty = true;\n this.clearDepth.dirty = true;\n this.clearStencil.dirty = true;\n this.colorMask.dirty = true;\n this.depthMask.dirty = true;\n this.stencilMask.dirty = true;\n this.stencilFunc.dirty = true;\n this.stencilOp.dirty = true;\n this.stencilTest.dirty = true;\n this.depthRange.dirty = true;\n this.depthTest.dirty = true;\n this.depthFunc.dirty = true;\n this.blend.dirty = true;\n this.blendFunc.dirty = true;\n this.blendColor.dirty = true;\n this.blendEquation.dirty = true;\n this.cullFace.dirty = true;\n this.cullFaceSide.dirty = true;\n this.frontFace.dirty = true;\n this.program.dirty = true;\n this.activeTexture.dirty = true;\n this.viewport.dirty = true;\n this.bindFramebuffer.dirty = true;\n this.bindRenderbuffer.dirty = true;\n this.bindTexture.dirty = true;\n this.bindVertexBuffer.dirty = true;\n this.bindElementBuffer.dirty = true;\n this.bindVertexArray.dirty = true;\n this.pixelStoreUnpack.dirty = true;\n this.pixelStoreUnpackPremultiplyAlpha.dirty = true;\n this.pixelStoreUnpackFlipY.dirty = true;\n }\n\n createIndexBuffer(array: TriangleIndexArray | LineIndexArray | LineStripIndexArray, dynamicDraw?: boolean) {\n return new IndexBuffer(this, array, dynamicDraw);\n }\n\n createVertexBuffer(array: StructArray, attributes: ReadonlyArray, dynamicDraw?: boolean) {\n return new VertexBuffer(this, array, attributes, dynamicDraw);\n }\n\n createRenderbuffer(storageFormat: number, width: number, height: number) {\n const gl = this.gl;\n\n const rbo = gl.createRenderbuffer();\n this.bindRenderbuffer.set(rbo);\n gl.renderbufferStorage(gl.RENDERBUFFER, storageFormat, width, height);\n this.bindRenderbuffer.set(null);\n\n return rbo;\n }\n\n createFramebuffer(width: number, height: number, hasDepth: boolean, hasStencil: boolean) {\n return new Framebuffer(this, width, height, hasDepth, hasStencil);\n }\n\n clear({\n color,\n depth,\n stencil\n }: ClearArgs) {\n const gl = this.gl;\n let mask = 0;\n\n if (color) {\n mask |= gl.COLOR_BUFFER_BIT;\n this.clearColor.set(color);\n this.colorMask.set([true, true, true, true]);\n }\n\n if (typeof depth !== 'undefined') {\n mask |= gl.DEPTH_BUFFER_BIT;\n\n // Workaround for platforms where clearDepth doesn't seem to work\n // without resetting the depthRange. See https://github.com/mapbox/mapbox-gl-js/issues/3437\n this.depthRange.set([0, 1]);\n\n this.clearDepth.set(depth);\n this.depthMask.set(true);\n }\n\n if (typeof stencil !== 'undefined') {\n mask |= gl.STENCIL_BUFFER_BIT;\n this.clearStencil.set(stencil);\n this.stencilMask.set(0xFF);\n }\n\n gl.clear(mask);\n }\n\n setCullFace(cullFaceMode: Readonly) {\n if (cullFaceMode.enable === false) {\n this.cullFace.set(false);\n } else {\n this.cullFace.set(true);\n this.cullFaceSide.set(cullFaceMode.mode);\n this.frontFace.set(cullFaceMode.frontFace);\n }\n }\n\n setDepthMode(depthMode: Readonly) {\n if (depthMode.func === this.gl.ALWAYS && !depthMode.mask) {\n this.depthTest.set(false);\n } else {\n this.depthTest.set(true);\n this.depthFunc.set(depthMode.func);\n this.depthMask.set(depthMode.mask);\n this.depthRange.set(depthMode.range);\n }\n }\n\n setStencilMode(stencilMode: Readonly) {\n if (stencilMode.test.func === this.gl.ALWAYS && !stencilMode.mask) {\n this.stencilTest.set(false);\n } else {\n this.stencilTest.set(true);\n this.stencilMask.set(stencilMode.mask);\n this.stencilOp.set([stencilMode.fail, stencilMode.depthFail, stencilMode.pass]);\n this.stencilFunc.set({\n func: stencilMode.test.func,\n ref: stencilMode.ref,\n mask: stencilMode.test.mask\n });\n }\n }\n\n setColorMode(colorMode: Readonly) {\n if (deepEqual(colorMode.blendFunction, ColorMode.Replace)) {\n this.blend.set(false);\n } else {\n this.blend.set(true);\n this.blendFunc.set(colorMode.blendFunction);\n this.blendColor.set(colorMode.blendColor);\n }\n\n this.colorMask.set(colorMode.mask);\n }\n\n createVertexArray(): WebGLVertexArrayObject | undefined {\n if (isWebGL2(this.gl))\n return this.gl.createVertexArray();\n return this.gl.getExtension('OES_vertex_array_object')?.createVertexArrayOES();\n }\n\n deleteVertexArray(x: WebGLVertexArrayObject | undefined) {\n if (isWebGL2(this.gl))\n return this.gl.deleteVertexArray(x);\n return this.gl.getExtension('OES_vertex_array_object')?.deleteVertexArrayOES(x);\n }\n\n unbindVAO() {\n // Unbinding the VAO prevents other things (custom layers, new buffer creation) from\n // unintentionally changing the state of the last VAO used.\n this.bindVertexArray.set(null);\n }\n}\n","import type {DepthFuncType, DepthMaskType, DepthRangeType} from './types';\n\nconst ALWAYS = 0x0207;\n\nexport class DepthMode {\n func: DepthFuncType;\n mask: DepthMaskType;\n range: DepthRangeType;\n\n // DepthMask enums\n static ReadOnly: boolean;\n static ReadWrite: boolean;\n\n constructor(depthFunc: DepthFuncType, depthMask: DepthMaskType, depthRange: DepthRangeType) {\n this.func = depthFunc;\n this.mask = depthMask;\n this.range = depthRange;\n }\n\n static disabled: Readonly;\n}\n\nDepthMode.ReadOnly = false;\nDepthMode.ReadWrite = true;\n\nDepthMode.disabled = new DepthMode(ALWAYS, DepthMode.ReadOnly, [0, 1]);\n","import type {StencilOpConstant, StencilTestGL} from './types';\n\nconst ALWAYS = 0x0207;\nconst KEEP = 0x1E00;\n\nexport class StencilMode {\n test: StencilTestGL;\n ref: number;\n mask: number;\n fail: StencilOpConstant;\n depthFail: StencilOpConstant;\n pass: StencilOpConstant;\n\n constructor(test: StencilTestGL, ref: number, mask: number, fail: StencilOpConstant,\n depthFail: StencilOpConstant, pass: StencilOpConstant) {\n this.test = test;\n this.ref = ref;\n this.mask = mask;\n this.fail = fail;\n this.depthFail = depthFail;\n this.pass = pass;\n }\n\n static disabled: Readonly;\n}\n\nStencilMode.disabled = new StencilMode({func: ALWAYS, mask: 0}, 0, 0, KEEP, KEEP, KEEP);\n","import type {CullFaceModeType, FrontFaceType} from './types';\n\nconst BACK = 0x0405;\nconst CCW = 0x0901;\n\nexport class CullFaceMode {\n enable: boolean;\n mode: CullFaceModeType;\n frontFace: FrontFaceType;\n\n constructor(enable: boolean, mode: CullFaceModeType, frontFace: FrontFaceType) {\n this.enable = enable;\n this.mode = mode;\n this.frontFace = frontFace;\n }\n\n static disabled: Readonly;\n static backCCW: Readonly;\n}\n\nCullFaceMode.disabled = new CullFaceMode(false, BACK, CCW);\nCullFaceMode.backCCW = new CullFaceMode(true, BACK, CCW);\n","import type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {SymbolBucket} from '../data/bucket/symbol_bucket';\nimport {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {collisionUniformValues, collisionCircleUniformValues} from './program/collision_program';\n\nimport {QuadTriangleArray, CollisionCircleLayoutArray} from '../data/array_types.g';\nimport {collisionCircleLayout} from '../data/bucket/symbol_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {mat4} from 'gl-matrix';\nimport {VertexBuffer} from '../gl/vertex_buffer';\nimport {IndexBuffer} from '../gl/index_buffer';\n\ntype TileBatch = {\n circleArray: Array;\n circleOffset: number;\n transform: mat4;\n invTransform: mat4;\n coord: OverscaledTileID;\n};\n\nlet quadTriangles: QuadTriangleArray;\n\nexport function drawCollisionDebug(painter: Painter, sourceCache: SourceCache, layer: StyleLayer, coords: Array, translate: [number, number], translateAnchor: 'map' | 'viewport', isText: boolean) {\n const context = painter.context;\n const gl = context.gl;\n const program = painter.useProgram('collisionBox');\n const tileBatches: Array = [];\n let circleCount = 0;\n let circleOffset = 0;\n\n for (let i = 0; i < coords.length; i++) {\n const coord = coords[i];\n const tile = sourceCache.getTile(coord);\n const bucket: SymbolBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n let posMatrix = coord.posMatrix;\n if (translate[0] !== 0 || translate[1] !== 0) {\n posMatrix = painter.translatePosMatrix(coord.posMatrix, tile, translate, translateAnchor);\n }\n const buffers = isText ? bucket.textCollisionBox : bucket.iconCollisionBox;\n // Get collision circle data of this bucket\n const circleArray: Array = bucket.collisionCircleArray;\n if (circleArray.length > 0) {\n // We need to know the projection matrix that was used for projecting collision circles to the screen.\n // This might vary between buckets as the symbol placement is a continuous process. This matrix is\n // required for transforming points from previous screen space to the current one\n const invTransform = mat4.create();\n const transform = posMatrix;\n\n mat4.mul(invTransform, bucket.placementInvProjMatrix, painter.transform.glCoordMatrix);\n mat4.mul(invTransform, invTransform, bucket.placementViewportMatrix);\n\n tileBatches.push({\n circleArray,\n circleOffset,\n transform,\n invTransform,\n coord\n });\n\n circleCount += circleArray.length / 4; // 4 values per circle\n circleOffset = circleCount;\n }\n if (!buffers) continue;\n program.draw(context, gl.LINES,\n DepthMode.disabled, StencilMode.disabled,\n painter.colorModeForRenderPass(),\n CullFaceMode.disabled,\n collisionUniformValues(\n posMatrix,\n painter.transform,\n tile),\n painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord),\n layer.id, buffers.layoutVertexBuffer, buffers.indexBuffer,\n buffers.segments, null, painter.transform.zoom, null, null,\n buffers.collisionVertexBuffer);\n }\n\n if (!isText || !tileBatches.length) {\n return;\n }\n\n // Render collision circles\n const circleProgram = painter.useProgram('collisionCircle');\n\n // Construct vertex data\n const vertexData = new CollisionCircleLayoutArray();\n vertexData.resize(circleCount * 4);\n vertexData._trim();\n\n let vertexOffset = 0;\n\n for (const batch of tileBatches) {\n for (let i = 0; i < batch.circleArray.length / 4; i++) {\n const circleIdx = i * 4;\n const x = batch.circleArray[circleIdx + 0];\n const y = batch.circleArray[circleIdx + 1];\n const radius = batch.circleArray[circleIdx + 2];\n const collision = batch.circleArray[circleIdx + 3];\n\n // 4 floats per vertex, 4 vertices per quad\n vertexData.emplace(vertexOffset++, x, y, radius, collision, 0);\n vertexData.emplace(vertexOffset++, x, y, radius, collision, 1);\n vertexData.emplace(vertexOffset++, x, y, radius, collision, 2);\n vertexData.emplace(vertexOffset++, x, y, radius, collision, 3);\n }\n }\n if (!quadTriangles || quadTriangles.length < circleCount * 2) {\n quadTriangles = createQuadTriangles(circleCount);\n }\n\n const indexBuffer: IndexBuffer = context.createIndexBuffer(quadTriangles, true);\n const vertexBuffer: VertexBuffer = context.createVertexBuffer(vertexData, collisionCircleLayout.members, true);\n\n // Render batches\n for (const batch of tileBatches) {\n const uniforms = collisionCircleUniformValues(\n batch.transform,\n batch.invTransform,\n painter.transform\n );\n\n circleProgram.draw(\n context,\n gl.TRIANGLES,\n DepthMode.disabled,\n StencilMode.disabled,\n painter.colorModeForRenderPass(),\n CullFaceMode.disabled,\n uniforms,\n painter.style.map.terrain && painter.style.map.terrain.getTerrainData(batch.coord),\n layer.id,\n vertexBuffer,\n indexBuffer,\n SegmentVector.simpleSegment(0, batch.circleOffset * 2, batch.circleArray.length, batch.circleArray.length / 2),\n null,\n painter.transform.zoom,\n null,\n null,\n null);\n }\n\n vertexBuffer.destroy();\n indexBuffer.destroy();\n}\n\nfunction createQuadTriangles(quadCount: number): QuadTriangleArray {\n const triCount = quadCount * 2;\n const array = new QuadTriangleArray();\n\n array.resize(triCount);\n array._trim();\n\n // Two triangles and 4 vertices per quad.\n for (let i = 0; i < triCount; i++) {\n const idx = i * 6;\n\n array.uint16[idx + 0] = i * 4 + 0;\n array.uint16[idx + 1] = i * 4 + 1;\n array.uint16[idx + 2] = i * 4 + 2;\n array.uint16[idx + 3] = i * 4 + 2;\n array.uint16[idx + 4] = i * 4 + 3;\n array.uint16[idx + 5] = i * 4 + 0;\n }\n\n return array;\n}\n","import Point from '@mapbox/point-geometry';\nimport {drawCollisionDebug} from './draw_collision_debug';\n\nimport {SegmentVector} from '../data/segment';\nimport {pixelsToTileUnits} from '../source/pixels_to_tile_units';\nimport * as symbolProjection from '../symbol/projection';\nimport {EvaluatedZoomSize, evaluateSizeForFeature, evaluateSizeForZoom} from '../symbol/symbol_size';\nimport {mat4} from 'gl-matrix';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {addDynamicAttributes} from '../data/bucket/symbol_bucket';\n\nimport {getAnchorAlignment, WritingMode} from '../symbol/shaping';\nimport ONE_EM from '../symbol/one_em';\n\nimport {\n SymbolIconUniformsType,\n symbolIconUniformValues,\n symbolSDFUniformValues,\n symbolTextAndIconUniformValues\n} from './program/symbol_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer';\n\nimport type {Texture} from '../render/texture';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport type {UniformValues} from './uniform_binding';\nimport type {SymbolSDFUniformsType} from '../render/program/symbol_program';\nimport type {CrossTileID, VariableOffset} from '../symbol/placement';\nimport type {SymbolBucket, SymbolBuffers} from '../data/bucket/symbol_bucket';\nimport type {TerrainData} from '../render/terrain';\nimport type {SymbolLayerSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport type {Transform} from '../geo/transform';\nimport type {ColorMode} from '../gl/color_mode';\nimport type {Program} from './program';\nimport type {TextAnchor} from '../style/style_layer/variable_text_anchor';\n\ntype SymbolTileRenderState = {\n segments: SegmentVector;\n sortKey: number;\n terrainData: TerrainData;\n state: {\n program: Program;\n buffers: SymbolBuffers;\n uniformValues: UniformValues;\n atlasTexture: Texture;\n atlasTextureIcon: Texture | null;\n atlasInterpolation: GLenum;\n atlasInterpolationIcon: GLenum;\n isSDF: boolean;\n hasHalo: boolean;\n };\n};\n\nconst identityMat4 = mat4.identity(new Float32Array(16));\n\nexport function drawSymbols(painter: Painter, sourceCache: SourceCache, layer: SymbolStyleLayer, coords: Array, variableOffsets: {\n [_ in CrossTileID]: VariableOffset;\n}) {\n if (painter.renderPass !== 'translucent') return;\n\n // Disable the stencil test so that labels aren't clipped to tile boundaries.\n const stencilMode = StencilMode.disabled;\n const colorMode = painter.colorModeForRenderPass();\n const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset');\n\n //Compute variable-offsets before painting since icons and text data positioning\n //depend on each other in this case.\n if (hasVariablePlacement) {\n updateVariableAnchors(coords, painter, layer, sourceCache,\n layer.layout.get('text-rotation-alignment'),\n layer.layout.get('text-pitch-alignment'),\n variableOffsets\n );\n }\n\n if (layer.paint.get('icon-opacity').constantOr(1) !== 0) {\n drawLayerSymbols(painter, sourceCache, layer, coords, false,\n layer.paint.get('icon-translate'),\n layer.paint.get('icon-translate-anchor'),\n layer.layout.get('icon-rotation-alignment'),\n layer.layout.get('icon-pitch-alignment'),\n layer.layout.get('icon-keep-upright'),\n stencilMode, colorMode\n );\n }\n\n if (layer.paint.get('text-opacity').constantOr(1) !== 0) {\n drawLayerSymbols(painter, sourceCache, layer, coords, true,\n layer.paint.get('text-translate'),\n layer.paint.get('text-translate-anchor'),\n layer.layout.get('text-rotation-alignment'),\n layer.layout.get('text-pitch-alignment'),\n layer.layout.get('text-keep-upright'),\n stencilMode, colorMode\n );\n }\n\n if (sourceCache.map.showCollisionBoxes) {\n drawCollisionDebug(painter, sourceCache, layer, coords, layer.paint.get('text-translate'),\n layer.paint.get('text-translate-anchor'), true);\n drawCollisionDebug(painter, sourceCache, layer, coords, layer.paint.get('icon-translate'),\n layer.paint.get('icon-translate-anchor'), false);\n }\n}\n\nfunction calculateVariableRenderShift(\n anchor: TextAnchor,\n width: number,\n height: number,\n textOffset: [number, number],\n textBoxScale: number,\n renderTextSize: number): Point {\n const {horizontalAlign, verticalAlign} = getAnchorAlignment(anchor);\n const shiftX = -(horizontalAlign - 0.5) * width;\n const shiftY = -(verticalAlign - 0.5) * height;\n return new Point(\n (shiftX / textBoxScale + textOffset[0]) * renderTextSize,\n (shiftY / textBoxScale + textOffset[1]) * renderTextSize\n );\n}\n\nfunction updateVariableAnchors(coords: Array,\n painter: Painter,\n layer:SymbolStyleLayer, sourceCache: SourceCache,\n rotationAlignment: SymbolLayerSpecification['layout']['text-rotation-alignment'],\n pitchAlignment: SymbolLayerSpecification['layout']['text-pitch-alignment'],\n variableOffsets: {[_ in CrossTileID]: VariableOffset}) {\n const tr = painter.transform;\n const rotateWithMap = rotationAlignment === 'map';\n const pitchWithMap = pitchAlignment === 'map';\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n const bucket = tile.getBucket(layer) as SymbolBucket;\n if (!bucket || !bucket.text || !bucket.text.segments.get().length) continue;\n\n const sizeData = bucket.textSizeData;\n const size = evaluateSizeForZoom(sizeData, tr.zoom);\n\n const pixelToTileScale = pixelsToTileUnits(tile, 1, painter.transform.zoom);\n const labelPlaneMatrix = symbolProjection.getLabelPlaneMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, pixelToTileScale);\n const updateTextFitIcon = layer.layout.get('icon-text-fit') !== 'none' && bucket.hasIconData();\n\n if (size) {\n const tileScale = Math.pow(2, tr.zoom - tile.tileID.overscaledZ);\n const getElevation = painter.style.map.terrain ? (x: number, y: number) => painter.style.map.terrain.getElevation(coord, x, y) : null;\n updateVariableAnchorsForBucket(bucket, rotateWithMap, pitchWithMap, variableOffsets,\n tr, labelPlaneMatrix, coord.posMatrix, tileScale, size, updateTextFitIcon, getElevation);\n }\n }\n}\n\nfunction updateVariableAnchorsForBucket(\n bucket: SymbolBucket,\n rotateWithMap: boolean,\n pitchWithMap: boolean,\n variableOffsets: {[_ in CrossTileID]: VariableOffset},\n transform: Transform,\n labelPlaneMatrix: mat4,\n posMatrix: mat4,\n tileScale: number,\n size: EvaluatedZoomSize,\n updateTextFitIcon: boolean,\n getElevation: (x: number, y: number) => number) {\n const placedSymbols = bucket.text.placedSymbolArray;\n const dynamicTextLayoutVertexArray = bucket.text.dynamicLayoutVertexArray;\n const dynamicIconLayoutVertexArray = bucket.icon.dynamicLayoutVertexArray;\n const placedTextShifts = {};\n\n dynamicTextLayoutVertexArray.clear();\n for (let s = 0; s < placedSymbols.length; s++) {\n const symbol = placedSymbols.get(s);\n const skipOrientation = bucket.allowVerticalPlacement && !symbol.placedOrientation;\n const variableOffset = (!symbol.hidden && symbol.crossTileID && !skipOrientation) ? variableOffsets[symbol.crossTileID] : null;\n\n if (!variableOffset) {\n // These symbols are from a justification that is not being used, or a label that wasn't placed\n // so we don't need to do the extra math to figure out what incremental shift to apply.\n symbolProjection.hideGlyphs(symbol.numGlyphs, dynamicTextLayoutVertexArray);\n } else {\n const tileAnchor = new Point(symbol.anchorX, symbol.anchorY);\n const projectedAnchor = symbolProjection.project(tileAnchor, pitchWithMap ? posMatrix : labelPlaneMatrix, getElevation);\n const perspectiveRatio = symbolProjection.getPerspectiveRatio(transform.cameraToCenterDistance, projectedAnchor.signedDistanceFromCamera);\n let renderTextSize = evaluateSizeForFeature(bucket.textSizeData, size, symbol) * perspectiveRatio / ONE_EM;\n if (pitchWithMap) {\n // Go from size in pixels to equivalent size in tile units\n renderTextSize *= bucket.tilePixelRatio / tileScale;\n }\n\n const {width, height, anchor, textOffset, textBoxScale} = variableOffset;\n\n const shift = calculateVariableRenderShift(\n anchor, width, height, textOffset, textBoxScale, renderTextSize);\n\n // Usual case is that we take the projected anchor and add the pixel-based shift\n // calculated above. In the (somewhat weird) case of pitch-aligned text, we add an equivalent\n // tile-unit based shift to the anchor before projecting to the label plane.\n const shiftedAnchor = pitchWithMap ?\n symbolProjection.project(tileAnchor.add(shift), labelPlaneMatrix, getElevation).point :\n projectedAnchor.point.add(rotateWithMap ?\n shift.rotate(-transform.angle) :\n shift);\n\n const angle = (bucket.allowVerticalPlacement && symbol.placedOrientation === WritingMode.vertical) ? Math.PI / 2 : 0;\n for (let g = 0; g < symbol.numGlyphs; g++) {\n addDynamicAttributes(dynamicTextLayoutVertexArray, shiftedAnchor, angle);\n }\n //Only offset horizontal text icons\n if (updateTextFitIcon && symbol.associatedIconIndex >= 0) {\n placedTextShifts[symbol.associatedIconIndex] = {shiftedAnchor, angle};\n }\n }\n }\n\n if (updateTextFitIcon) {\n dynamicIconLayoutVertexArray.clear();\n const placedIcons = bucket.icon.placedSymbolArray;\n for (let i = 0; i < placedIcons.length; i++) {\n const placedIcon = placedIcons.get(i);\n if (placedIcon.hidden) {\n symbolProjection.hideGlyphs(placedIcon.numGlyphs, dynamicIconLayoutVertexArray);\n } else {\n const shift = placedTextShifts[i];\n if (!shift) {\n symbolProjection.hideGlyphs(placedIcon.numGlyphs, dynamicIconLayoutVertexArray);\n } else {\n for (let g = 0; g < placedIcon.numGlyphs; g++) {\n addDynamicAttributes(dynamicIconLayoutVertexArray, shift.shiftedAnchor, shift.angle);\n }\n }\n }\n }\n bucket.icon.dynamicLayoutVertexBuffer.updateData(dynamicIconLayoutVertexArray);\n }\n bucket.text.dynamicLayoutVertexBuffer.updateData(dynamicTextLayoutVertexArray);\n}\n\nfunction getSymbolProgramName(isSDF: boolean, isText: boolean, bucket: SymbolBucket) {\n if (bucket.iconsInText && isText) {\n return 'symbolTextAndIcon';\n } else if (isSDF) {\n return 'symbolSDF';\n } else {\n return 'symbolIcon';\n }\n}\n\nfunction drawLayerSymbols(\n painter: Painter,\n sourceCache: SourceCache,\n layer: SymbolStyleLayer,\n coords: Array,\n isText: boolean,\n translate: [number, number],\n translateAnchor: 'map' | 'viewport',\n rotationAlignment: SymbolLayerSpecification['layout']['text-rotation-alignment'],\n pitchAlignment: SymbolLayerSpecification['layout']['text-pitch-alignment'],\n keepUpright: boolean,\n stencilMode: StencilMode,\n colorMode: Readonly) {\n\n const context = painter.context;\n const gl = context.gl;\n const tr = painter.transform;\n\n const rotateWithMap = rotationAlignment === 'map';\n const pitchWithMap = pitchAlignment === 'map';\n const alongLine = rotationAlignment !== 'viewport' && layer.layout.get('symbol-placement') !== 'point';\n // Line label rotation happens in `updateLineLabels`\n // Pitched point labels are automatically rotated by the labelPlaneMatrix projection\n // Unpitched point labels need to have their rotation applied after projection\n const rotateInShader = rotateWithMap && !pitchWithMap && !alongLine;\n\n const hasSortKey = !layer.layout.get('symbol-sort-key').isConstant();\n let sortFeaturesByKey = false;\n\n const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n\n const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset');\n\n const tileRenderState: Array = [];\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n const bucket = tile.getBucket(layer) as SymbolBucket;\n if (!bucket) continue;\n const buffers = isText ? bucket.text : bucket.icon;\n\n if (!buffers || !buffers.segments.get().length || !buffers.hasVisibleVertices) continue;\n const programConfiguration = buffers.programConfigurations.get(layer.id);\n\n const isSDF = isText || bucket.sdfIcons;\n\n const sizeData = isText ? bucket.textSizeData : bucket.iconSizeData;\n const transformed = pitchWithMap || tr.pitch !== 0;\n\n const program = painter.useProgram(getSymbolProgramName(isSDF, isText, bucket), programConfiguration);\n const size = evaluateSizeForZoom(sizeData, tr.zoom);\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n let texSize: [number, number];\n let texSizeIcon: [number, number] = [0, 0];\n let atlasTexture: Texture;\n let atlasInterpolation: GLenum;\n let atlasTextureIcon = null;\n let atlasInterpolationIcon: GLenum;\n if (isText) {\n atlasTexture = tile.glyphAtlasTexture;\n atlasInterpolation = gl.LINEAR;\n texSize = tile.glyphAtlasTexture.size;\n if (bucket.iconsInText) {\n texSizeIcon = tile.imageAtlasTexture.size;\n atlasTextureIcon = tile.imageAtlasTexture;\n const zoomDependentSize = sizeData.kind === 'composite' || sizeData.kind === 'camera';\n atlasInterpolationIcon = transformed || painter.options.rotating || painter.options.zooming || zoomDependentSize ? gl.LINEAR : gl.NEAREST;\n }\n } else {\n const iconScaled = layer.layout.get('icon-size').constantOr(0) !== 1 || bucket.iconsNeedLinear;\n atlasTexture = tile.imageAtlasTexture;\n atlasInterpolation = isSDF || painter.options.rotating || painter.options.zooming || iconScaled || transformed ?\n gl.LINEAR :\n gl.NEAREST;\n texSize = tile.imageAtlasTexture.size;\n }\n\n const s = pixelsToTileUnits(tile, 1, painter.transform.zoom);\n const labelPlaneMatrix = symbolProjection.getLabelPlaneMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, s);\n const glCoordMatrix = symbolProjection.getGlCoordMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, s);\n\n const hasVariableAnchors = hasVariablePlacement && bucket.hasTextData();\n const updateTextFitIcon = layer.layout.get('icon-text-fit') !== 'none' &&\n hasVariableAnchors &&\n bucket.hasIconData();\n\n if (alongLine) {\n const getElevation = painter.style.map.terrain ? (x: number, y: number) => painter.style.map.terrain.getElevation(coord, x, y) : null;\n const rotateToLine = layer.layout.get('text-rotation-alignment') === 'map';\n symbolProjection.updateLineLabels(bucket, coord.posMatrix, painter, isText, labelPlaneMatrix, glCoordMatrix, pitchWithMap, keepUpright, rotateToLine, getElevation);\n }\n\n const matrix = painter.translatePosMatrix(coord.posMatrix, tile, translate, translateAnchor),\n uLabelPlaneMatrix = (alongLine || (isText && hasVariablePlacement) || updateTextFitIcon) ? identityMat4 : labelPlaneMatrix,\n uglCoordMatrix = painter.translatePosMatrix(glCoordMatrix, tile, translate, translateAnchor, true);\n\n const hasHalo = isSDF && layer.paint.get(isText ? 'text-halo-width' : 'icon-halo-width').constantOr(1) !== 0;\n\n let uniformValues: UniformValues;\n if (isSDF) {\n if (!bucket.iconsInText) {\n uniformValues = symbolSDFUniformValues(sizeData.kind,\n size, rotateInShader, pitchWithMap, painter, matrix,\n uLabelPlaneMatrix, uglCoordMatrix, isText, texSize, true);\n } else {\n uniformValues = symbolTextAndIconUniformValues(sizeData.kind,\n size, rotateInShader, pitchWithMap, painter, matrix,\n uLabelPlaneMatrix, uglCoordMatrix, texSize, texSizeIcon);\n }\n } else {\n uniformValues = symbolIconUniformValues(sizeData.kind,\n size, rotateInShader, pitchWithMap, painter, matrix,\n uLabelPlaneMatrix, uglCoordMatrix, isText, texSize);\n }\n\n const state = {\n program,\n buffers,\n uniformValues,\n atlasTexture,\n atlasTextureIcon,\n atlasInterpolation,\n atlasInterpolationIcon,\n isSDF,\n hasHalo\n };\n\n if (hasSortKey && bucket.canOverlap) {\n sortFeaturesByKey = true;\n const oldSegments = buffers.segments.get();\n for (const segment of oldSegments) {\n tileRenderState.push({\n segments: new SegmentVector([segment]),\n sortKey: segment.sortKey,\n state,\n terrainData\n });\n }\n } else {\n tileRenderState.push({\n segments: buffers.segments,\n sortKey: 0,\n state,\n terrainData\n });\n }\n }\n\n if (sortFeaturesByKey) {\n tileRenderState.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const segmentState of tileRenderState) {\n const state = segmentState.state;\n\n context.activeTexture.set(gl.TEXTURE0);\n // @ts-ignore\n state.atlasTexture.bind(state.atlasInterpolation, gl.CLAMP_TO_EDGE);\n if (state.atlasTextureIcon) {\n context.activeTexture.set(gl.TEXTURE1);\n if (state.atlasTextureIcon) {\n // @ts-ignore\n state.atlasTextureIcon.bind(state.atlasInterpolationIcon, gl.CLAMP_TO_EDGE);\n }\n }\n\n if (state.isSDF) {\n const uniformValues = state.uniformValues;\n if (state.hasHalo) {\n uniformValues['u_is_halo'] = 1;\n drawSymbolElements(state.buffers, segmentState.segments, layer, painter, state.program, depthMode, stencilMode, colorMode, uniformValues, segmentState.terrainData);\n }\n uniformValues['u_is_halo'] = 0;\n }\n drawSymbolElements(state.buffers, segmentState.segments, layer, painter, state.program, depthMode, stencilMode, colorMode, state.uniformValues, segmentState.terrainData);\n }\n}\n\nfunction drawSymbolElements(\n buffers: SymbolBuffers,\n segments: SegmentVector,\n layer: SymbolStyleLayer,\n painter: Painter,\n program: Program,\n depthMode: Readonly,\n stencilMode: StencilMode,\n colorMode: Readonly,\n uniformValues: UniformValues,\n terrainData: TerrainData) {\n const context = painter.context;\n const gl = context.gl;\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id, buffers.layoutVertexBuffer,\n buffers.indexBuffer, segments, layer.paint,\n painter.transform.zoom, buffers.programConfigurations.get(layer.id),\n buffers.dynamicLayoutVertexBuffer, buffers.opacityVertexBuffer);\n}\n","import type {CrossFaded} from '../style/properties';\nimport type {ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport type {Tile} from '../source/tile';\nimport type {ProgramConfiguration} from '../data/program_configuration';\nimport type {FillExtrusionStyleLayer} from '../style/style_layer/fill_extrusion_style_layer';\nimport type {FillStyleLayer} from '../style/style_layer/fill_style_layer';\n\n/**\n * A simple helper shared by draw_fill and draw_fill_extrusions to find the correct pattern positions AND update program.\n * For transtionable properties, especially 'fill-pattern' and 'fill-extrusion-pattern', while rendering certain frames\n * tile.imageAtlas has been updated by worker to hold the new pattern only, but rendering code is still looking for the previous image.\n * The mismatch was causing setConstantPatternPositions method not being called and pixelRatio was always the\n * default of 1, instead of actual values set by original map.addImage.\n *\n * @param programConfiguration - to be used to set pattern position and device pixel ratio.\n * @param propertyName - 'fill-pattern' or 'fill-extrusion-pattern' property key\n * @param constantPattern - either 'fill-pattern' or 'fill-extrusion-pattern' property value\n * @param tile - current tile being drawn\n * @param layer - current layer being rendered\n */\nexport function updatePatternPositionsInProgram(\n programConfiguration: ProgramConfiguration,\n propertyName: 'fill-pattern' | 'fill-extrusion-pattern',\n constantPattern: CrossFaded,\n tile: Tile,\n layer: FillStyleLayer | FillExtrusionStyleLayer): void {\n\n if (!constantPattern || !tile || !tile.imageAtlas) {\n return;\n }\n\n const patternPositions = tile.imageAtlas.patternPositions;\n let posTo = patternPositions[constantPattern.to.toString()];\n let posFrom = patternPositions[constantPattern.from.toString()];\n\n // https://github.com/maplibre/maplibre-gl-js/issues/3377\n if (!posTo && posFrom) posTo = posFrom;\n if (!posFrom && posTo) posFrom = posTo;\n\n // try again in case patternPositions has been updated by worker\n if (!posTo || !posFrom) {\n const transitioned = layer.getPaintProperty(propertyName) as string;\n posTo = patternPositions[transitioned];\n posFrom = patternPositions[transitioned];\n }\n\n if (posTo && posFrom) {\n programConfiguration.setConstantPatternPositions(posTo, posFrom);\n }\n}\n","import {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {\n fillUniformValues,\n fillPatternUniformValues,\n fillOutlineUniformValues,\n fillOutlinePatternUniformValues\n} from './program/fill_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {FillStyleLayer} from '../style/style_layer/fill_style_layer';\nimport type {FillBucket} from '../data/bucket/fill_bucket';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport {updatePatternPositionsInProgram} from './update_pattern_positions_in_program';\n\nexport function drawFill(painter: Painter, sourceCache: SourceCache, layer: FillStyleLayer, coords: Array) {\n const color = layer.paint.get('fill-color');\n const opacity = layer.paint.get('fill-opacity');\n\n if (opacity.constantOr(1) === 0) {\n return;\n }\n\n const colorMode = painter.colorModeForRenderPass();\n\n const pattern = layer.paint.get('fill-pattern');\n const pass = painter.opaquePassEnabledForLayer() &&\n (!pattern.constantOr(1 as any) &&\n color.constantOr(Color.transparent).a === 1 &&\n opacity.constantOr(0) === 1) ? 'opaque' : 'translucent';\n\n // Draw fill\n if (painter.renderPass === pass) {\n const depthMode = painter.depthModeForSublayer(\n 1, painter.renderPass === 'opaque' ? DepthMode.ReadWrite : DepthMode.ReadOnly);\n drawFillTiles(painter, sourceCache, layer, coords, depthMode, colorMode, false);\n }\n\n // Draw stroke\n if (painter.renderPass === 'translucent' && layer.paint.get('fill-antialias')) {\n\n // If we defined a different color for the fill outline, we are\n // going to ignore the bits in 0x07 and just care about the global\n // clipping mask.\n // Otherwise, we only want to drawFill the antialiased parts that are\n // *outside* the current shape. This is important in case the fill\n // or stroke color is translucent. If we wouldn't clip to outside\n // the current shape, some pixels from the outline stroke overlapped\n // the (non-antialiased) fill.\n const depthMode = painter.depthModeForSublayer(\n layer.getPaintProperty('fill-outline-color') ? 2 : 0, DepthMode.ReadOnly);\n drawFillTiles(painter, sourceCache, layer, coords, depthMode, colorMode, true);\n }\n}\n\nfunction drawFillTiles(\n painter: Painter,\n sourceCache: SourceCache,\n layer: FillStyleLayer,\n coords: Array,\n depthMode: Readonly,\n colorMode: Readonly,\n isOutline: boolean) {\n const gl = painter.context.gl;\n const fillPropertyName = 'fill-pattern';\n const patternProperty = layer.paint.get(fillPropertyName);\n const image = patternProperty && patternProperty.constantOr(1 as any);\n const crossfade = layer.getCrossfadeParameters();\n let drawMode, programName, uniformValues, indexBuffer, segments;\n\n if (!isOutline) {\n programName = image ? 'fillPattern' : 'fill';\n drawMode = gl.TRIANGLES;\n } else {\n programName = image && !layer.getPaintProperty('fill-outline-color') ? 'fillOutlinePattern' : 'fillOutline';\n drawMode = gl.LINES;\n }\n\n const constantPattern = patternProperty.constantOr(null);\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n if (image && !tile.patternsLoaded()) continue;\n\n const bucket: FillBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const program = painter.useProgram(programName, programConfiguration);\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n if (image) {\n painter.context.activeTexture.set(gl.TEXTURE0);\n tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n programConfiguration.updatePaintBuffers(crossfade);\n }\n\n updatePatternPositionsInProgram(programConfiguration, fillPropertyName, constantPattern, tile, layer);\n\n const terrainCoord = terrainData ? coord : null;\n const posMatrix = terrainCoord ? terrainCoord.posMatrix : coord.posMatrix;\n const tileMatrix = painter.translatePosMatrix(posMatrix, tile,\n layer.paint.get('fill-translate'), layer.paint.get('fill-translate-anchor'));\n\n if (!isOutline) {\n indexBuffer = bucket.indexBuffer;\n segments = bucket.segments;\n uniformValues = image ?\n fillPatternUniformValues(tileMatrix, painter, crossfade, tile) :\n fillUniformValues(tileMatrix);\n } else {\n indexBuffer = bucket.indexBuffer2;\n segments = bucket.segments2;\n const drawingBufferSize = [gl.drawingBufferWidth, gl.drawingBufferHeight] as [number, number];\n uniformValues = (programName === 'fillOutlinePattern' && image) ?\n fillOutlinePatternUniformValues(tileMatrix, painter, crossfade, tile, drawingBufferSize) :\n fillOutlineUniformValues(tileMatrix, drawingBufferSize);\n }\n\n program.draw(painter.context, drawMode, depthMode,\n painter.stencilModeForClipping(coord), colorMode, CullFaceMode.disabled, uniformValues, terrainData,\n layer.id, bucket.layoutVertexBuffer, indexBuffer, segments,\n layer.paint, painter.transform.zoom, programConfiguration);\n }\n}\n","import {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {\n fillExtrusionUniformValues,\n fillExtrusionPatternUniformValues,\n} from './program/fill_extrusion_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {FillExtrusionStyleLayer} from '../style/style_layer/fill_extrusion_style_layer';\nimport type {FillExtrusionBucket} from '../data/bucket/fill_extrusion_bucket';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nimport {updatePatternPositionsInProgram} from './update_pattern_positions_in_program';\n\nexport function drawFillExtrusion(painter: Painter, source: SourceCache, layer: FillExtrusionStyleLayer, coords: Array) {\n const opacity = layer.paint.get('fill-extrusion-opacity');\n if (opacity === 0) {\n return;\n }\n\n if (painter.renderPass === 'translucent') {\n const depthMode = new DepthMode(painter.context.gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D);\n\n if (opacity === 1 && !layer.paint.get('fill-extrusion-pattern').constantOr(1 as any)) {\n const colorMode = painter.colorModeForRenderPass();\n drawExtrusionTiles(painter, source, layer, coords, depthMode, StencilMode.disabled, colorMode);\n\n } else {\n // Draw transparent buildings in two passes so that only the closest surface is drawn.\n // First draw all the extrusions into only the depth buffer. No colors are drawn.\n drawExtrusionTiles(painter, source, layer, coords, depthMode,\n StencilMode.disabled,\n ColorMode.disabled);\n\n // Then draw all the extrusions a second type, only coloring fragments if they have the\n // same depth value as the closest fragment in the previous pass. Use the stencil buffer\n // to prevent the second draw in cases where we have coincident polygons.\n drawExtrusionTiles(painter, source, layer, coords, depthMode,\n painter.stencilModeFor3D(),\n painter.colorModeForRenderPass());\n }\n }\n}\n\nfunction drawExtrusionTiles(\n painter: Painter,\n source: SourceCache,\n layer: FillExtrusionStyleLayer,\n coords: OverscaledTileID[],\n depthMode: DepthMode,\n stencilMode: Readonly,\n colorMode: Readonly) {\n const context = painter.context;\n const gl = context.gl;\n const fillPropertyName = 'fill-extrusion-pattern';\n const patternProperty = layer.paint.get(fillPropertyName);\n const image = patternProperty.constantOr(1 as any);\n const crossfade = layer.getCrossfadeParameters();\n const opacity = layer.paint.get('fill-extrusion-opacity');\n const constantPattern = patternProperty.constantOr(null);\n for (const coord of coords) {\n const tile = source.getTile(coord);\n const bucket: FillExtrusionBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const program = painter.useProgram(image ? 'fillExtrusionPattern' : 'fillExtrusion', programConfiguration);\n\n if (image) {\n painter.context.activeTexture.set(gl.TEXTURE0);\n tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n programConfiguration.updatePaintBuffers(crossfade);\n }\n\n updatePatternPositionsInProgram(programConfiguration, fillPropertyName, constantPattern, tile, layer);\n\n const matrix = painter.translatePosMatrix(\n coord.posMatrix,\n tile,\n layer.paint.get('fill-extrusion-translate'),\n layer.paint.get('fill-extrusion-translate-anchor'));\n\n const shouldUseVerticalGradient = layer.paint.get('fill-extrusion-vertical-gradient');\n const uniformValues = image ?\n fillExtrusionPatternUniformValues(matrix, painter, shouldUseVerticalGradient, opacity, coord, crossfade, tile) :\n fillExtrusionUniformValues(matrix, painter, shouldUseVerticalGradient, opacity);\n\n program.draw(context, context.gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.backCCW,\n uniformValues, terrainData, layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer,\n bucket.segments, layer.paint, painter.transform.zoom,\n programConfiguration, painter.style.map.terrain && bucket.centroidVertexBuffer);\n }\n}\n","import {Texture} from './texture';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {Tile} from '../source/tile';\nimport {\n hillshadeUniformValues,\n hillshadeUniformPrepareValues\n} from './program/hillshade_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {HillshadeStyleLayer} from '../style/style_layer/hillshade_style_layer';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nexport function drawHillshade(painter: Painter, sourceCache: SourceCache, layer: HillshadeStyleLayer, tileIDs: Array) {\n if (painter.renderPass !== 'offscreen' && painter.renderPass !== 'translucent') return;\n\n const context = painter.context;\n\n const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n const colorMode = painter.colorModeForRenderPass();\n\n const [stencilModes, coords] = painter.renderPass === 'translucent' ?\n painter.stencilConfigForOverlap(tileIDs) : [{}, tileIDs];\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n if (typeof tile.needsHillshadePrepare !== 'undefined' && tile.needsHillshadePrepare && painter.renderPass === 'offscreen') {\n prepareHillshade(painter, tile, layer, depthMode, StencilMode.disabled, colorMode);\n } else if (painter.renderPass === 'translucent') {\n renderHillshade(painter, coord, tile, layer, depthMode, stencilModes[coord.overscaledZ], colorMode);\n }\n }\n\n context.viewport.set([0, 0, painter.width, painter.height]);\n}\n\nfunction renderHillshade(\n painter: Painter,\n coord: OverscaledTileID,\n tile: Tile,\n layer: HillshadeStyleLayer,\n depthMode: Readonly,\n stencilMode: Readonly,\n colorMode: Readonly) {\n const context = painter.context;\n const gl = context.gl;\n const fbo = tile.fbo;\n if (!fbo) return;\n\n const program = painter.useProgram('hillshade');\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n context.activeTexture.set(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get());\n\n const terrainCoord = terrainData ? coord : null;\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n hillshadeUniformValues(painter, tile, layer, terrainCoord), terrainData, layer.id, painter.rasterBoundsBuffer,\n painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments);\n\n}\n\n// hillshade rendering is done in two steps. the prepare step first calculates the slope of the terrain in the x and y\n// directions for each pixel, and saves those values to a framebuffer texture in the r and g channels.\nfunction prepareHillshade(\n painter: Painter,\n tile: Tile,\n layer: HillshadeStyleLayer,\n depthMode: Readonly,\n stencilMode: Readonly,\n colorMode: Readonly) {\n const context = painter.context;\n const gl = context.gl;\n const dem = tile.dem;\n if (dem && dem.data) {\n const tileSize = dem.dim;\n const textureStride = dem.stride;\n\n const pixelData = dem.getPixels();\n context.activeTexture.set(gl.TEXTURE1);\n\n context.pixelStoreUnpackPremultiplyAlpha.set(false);\n tile.demTexture = tile.demTexture || painter.getTileTexture(textureStride);\n if (tile.demTexture) {\n const demTexture = tile.demTexture;\n demTexture.update(pixelData, {premultiply: false});\n demTexture.bind(gl.NEAREST, gl.CLAMP_TO_EDGE);\n } else {\n tile.demTexture = new Texture(context, pixelData, gl.RGBA, {premultiply: false});\n tile.demTexture.bind(gl.NEAREST, gl.CLAMP_TO_EDGE);\n }\n\n context.activeTexture.set(gl.TEXTURE0);\n\n let fbo = tile.fbo;\n\n if (!fbo) {\n const renderTexture = new Texture(context, {width: tileSize, height: tileSize, data: null}, gl.RGBA);\n renderTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n\n fbo = tile.fbo = context.createFramebuffer(tileSize, tileSize, true, false);\n fbo.colorAttachment.set(renderTexture.texture);\n }\n\n context.bindFramebuffer.set(fbo.framebuffer);\n context.viewport.set([0, 0, tileSize, tileSize]);\n\n painter.useProgram('hillshadePrepare').draw(context, gl.TRIANGLES,\n depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n hillshadeUniformPrepareValues(tile.tileID, dem),\n null, layer.id, painter.rasterBoundsBuffer,\n painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments);\n\n tile.needsHillshadePrepare = false;\n }\n}\n","import {clamp} from '../util/util';\n\nimport {ImageSource} from '../source/image_source';\nimport {browser} from '../util/browser';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {rasterUniformValues} from './program/raster_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {RasterStyleLayer} from '../style/style_layer/raster_style_layer';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nexport function drawRaster(painter: Painter, sourceCache: SourceCache, layer: RasterStyleLayer, tileIDs: Array) {\n if (painter.renderPass !== 'translucent') return;\n if (layer.paint.get('raster-opacity') === 0) return;\n if (!tileIDs.length) return;\n\n const context = painter.context;\n const gl = context.gl;\n const source = sourceCache.getSource();\n const program = painter.useProgram('raster');\n\n const colorMode = painter.colorModeForRenderPass();\n\n const [stencilModes, coords] = source instanceof ImageSource ? [{}, tileIDs] :\n painter.stencilConfigForOverlap(tileIDs);\n\n const minTileZ = coords[coords.length - 1].overscaledZ;\n\n const align = !painter.options.moving;\n for (const coord of coords) {\n // Set the lower zoom level to sublayer 0, and higher zoom levels to higher sublayers\n // Use gl.LESS to prevent double drawing in areas where tiles overlap.\n const depthMode = painter.depthModeForSublayer(coord.overscaledZ - minTileZ,\n layer.paint.get('raster-opacity') === 1 ? DepthMode.ReadWrite : DepthMode.ReadOnly, gl.LESS);\n\n const tile = sourceCache.getTile(coord);\n\n tile.registerFadeDuration(layer.paint.get('raster-fade-duration'));\n\n const parentTile = sourceCache.findLoadedParent(coord, 0),\n fade = getFadeValues(tile, parentTile, sourceCache, layer, painter.transform, painter.style.map.terrain);\n\n let parentScaleBy, parentTL;\n\n const textureFilter = layer.paint.get('raster-resampling') === 'nearest' ? gl.NEAREST : gl.LINEAR;\n\n context.activeTexture.set(gl.TEXTURE0);\n tile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST);\n\n context.activeTexture.set(gl.TEXTURE1);\n\n if (parentTile) {\n parentTile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST);\n parentScaleBy = Math.pow(2, parentTile.tileID.overscaledZ - tile.tileID.overscaledZ);\n parentTL = [tile.tileID.canonical.x * parentScaleBy % 1, tile.tileID.canonical.y * parentScaleBy % 1];\n\n } else {\n tile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST);\n }\n\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n const terrainCoord = terrainData ? coord : null;\n const posMatrix = terrainCoord ? terrainCoord.posMatrix : painter.transform.calculatePosMatrix(coord.toUnwrapped(), align);\n const uniformValues = rasterUniformValues(posMatrix, parentTL || [0, 0], parentScaleBy || 1, fade, layer);\n\n if (source instanceof ImageSource) {\n program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id, source.boundsBuffer,\n painter.quadTriangleIndexBuffer, source.boundsSegments);\n } else {\n program.draw(context, gl.TRIANGLES, depthMode, stencilModes[coord.overscaledZ], colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id, painter.rasterBoundsBuffer,\n painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments);\n }\n }\n}\n\nfunction getFadeValues(tile, parentTile, sourceCache, layer, transform, terrain) {\n const fadeDuration = layer.paint.get('raster-fade-duration');\n\n if (!terrain && fadeDuration > 0) {\n const now = browser.now();\n const sinceTile = (now - tile.timeAdded) / fadeDuration;\n const sinceParent = parentTile ? (now - parentTile.timeAdded) / fadeDuration : -1;\n\n const source = sourceCache.getSource();\n const idealZ = transform.coveringZoomLevel({\n tileSize: source.tileSize,\n roundZoom: source.roundZoom\n });\n\n // if no parent or parent is older, fade in; if parent is younger, fade out\n const fadeIn = !parentTile || Math.abs(parentTile.tileID.overscaledZ - idealZ) > Math.abs(tile.tileID.overscaledZ - idealZ);\n\n const childOpacity = (fadeIn && tile.refreshedUponExpiration) ? 1 : clamp(fadeIn ? sinceTile : 1 - sinceParent, 0, 1);\n\n // we don't crossfade tiles that were just refreshed upon expiring:\n // once they're old enough to pass the crossfading threshold\n // (fadeDuration), unset the `refreshedUponExpiration` flag so we don't\n // incorrectly fail to crossfade them when zooming\n if (tile.refreshedUponExpiration && sinceTile >= 1) tile.refreshedUponExpiration = false;\n\n if (parentTile) {\n return {\n opacity: 1,\n mix: 1 - childOpacity\n };\n } else {\n return {\n opacity: childOpacity,\n mix: 0\n };\n }\n } else {\n return {\n opacity: 1,\n mix: 0\n };\n }\n}\n","import {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {debugUniformValues} from './program/debug_program';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {ColorMode} from '../gl/color_mode';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport {Style} from '../style/style';\n\nconst topColor = new Color(1, 0, 0, 1);\nconst btmColor = new Color(0, 1, 0, 1);\nconst leftColor = new Color(0, 0, 1, 1);\nconst rightColor = new Color(1, 0, 1, 1);\nconst centerColor = new Color(0, 1, 1, 1);\n\nexport function drawDebugPadding(painter: Painter) {\n const padding = painter.transform.padding;\n const lineWidth = 3;\n // Top\n drawHorizontalLine(painter, painter.transform.height - (padding.top || 0), lineWidth, topColor);\n // Bottom\n drawHorizontalLine(painter, padding.bottom || 0, lineWidth, btmColor);\n // Left\n drawVerticalLine(painter, padding.left || 0, lineWidth, leftColor);\n // Right\n drawVerticalLine(painter, painter.transform.width - (padding.right || 0), lineWidth, rightColor);\n // Center\n const center = painter.transform.centerPoint;\n drawCrosshair(painter, center.x, painter.transform.height - center.y, centerColor);\n}\n\nfunction drawCrosshair(painter: Painter, x: number, y: number, color: Color) {\n const size = 20;\n const lineWidth = 2;\n //Vertical line\n drawDebugSSRect(painter, x - lineWidth / 2, y - size / 2, lineWidth, size, color);\n //Horizontal line\n drawDebugSSRect(painter, x - size / 2, y - lineWidth / 2, size, lineWidth, color);\n}\n\nfunction drawHorizontalLine(painter: Painter, y: number, lineWidth: number, color: Color) {\n drawDebugSSRect(painter, 0, y + lineWidth / 2, painter.transform.width, lineWidth, color);\n}\n\nfunction drawVerticalLine(painter: Painter, x: number, lineWidth: number, color: Color) {\n drawDebugSSRect(painter, x - lineWidth / 2, 0, lineWidth, painter.transform.height, color);\n}\n\nfunction drawDebugSSRect(painter: Painter, x: number, y: number, width: number, height: number, color: Color) {\n const context = painter.context;\n const gl = context.gl;\n\n gl.enable(gl.SCISSOR_TEST);\n gl.scissor(x * painter.pixelRatio, y * painter.pixelRatio, width * painter.pixelRatio, height * painter.pixelRatio);\n context.clear({color});\n gl.disable(gl.SCISSOR_TEST);\n}\n\nexport function drawDebug(painter: Painter, sourceCache: SourceCache, coords: Array) {\n for (let i = 0; i < coords.length; i++) {\n drawDebugTile(painter, sourceCache, coords[i]);\n }\n}\n\nfunction drawDebugTile(painter: Painter, sourceCache: SourceCache, coord: OverscaledTileID) {\n const context = painter.context;\n const gl = context.gl;\n\n const posMatrix = coord.posMatrix;\n const program = painter.useProgram('debug');\n\n const depthMode = DepthMode.disabled;\n const stencilMode = StencilMode.disabled;\n const colorMode = painter.colorModeForRenderPass();\n const id = '$debug';\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n context.activeTexture.set(gl.TEXTURE0);\n\n const tileRawData = sourceCache.getTileByID(coord.key).latestRawTileData;\n const tileByteLength = (tileRawData && tileRawData.byteLength) || 0;\n const tileSizeKb = Math.floor(tileByteLength / 1024);\n const tileSize = sourceCache.getTile(coord).tileSize;\n const scaleRatio = (512 / Math.min(tileSize, 512) * (coord.overscaledZ / painter.transform.zoom)) * 0.5;\n let tileIdText = coord.canonical.toString();\n if (coord.overscaledZ !== coord.canonical.z) {\n tileIdText += ` => ${coord.overscaledZ}`;\n }\n const tileLabel = `${tileIdText} ${tileSizeKb}kB`;\n drawTextToOverlay(painter, tileLabel);\n\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, ColorMode.alphaBlended, CullFaceMode.disabled,\n debugUniformValues(posMatrix, Color.transparent, scaleRatio), null, id,\n painter.debugBuffer, painter.quadTriangleIndexBuffer, painter.debugSegments);\n program.draw(context, gl.LINE_STRIP, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n debugUniformValues(posMatrix, Color.red), terrainData, id,\n painter.debugBuffer, painter.tileBorderIndexBuffer, painter.debugSegments);\n}\n\nfunction drawTextToOverlay(painter: Painter, text: string) {\n painter.initDebugOverlayCanvas();\n const canvas = painter.debugOverlayCanvas;\n const gl = painter.context.gl;\n const ctx2d = painter.debugOverlayCanvas.getContext('2d');\n ctx2d.clearRect(0, 0, canvas.width, canvas.height);\n\n ctx2d.shadowColor = 'white';\n ctx2d.shadowBlur = 2;\n ctx2d.lineWidth = 1.5;\n ctx2d.strokeStyle = 'white';\n ctx2d.textBaseline = 'top';\n ctx2d.font = `bold ${36}px Open Sans, sans-serif`;\n ctx2d.fillText(text, 5, 5);\n ctx2d.strokeText(text, 5, 5);\n\n painter.debugOverlayTexture.update(canvas);\n painter.debugOverlayTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n}\n\nexport function selectDebugSource(style: Style, zoom: number): SourceCache | null {\n // Use vector source with highest maxzoom\n // Else use source with highest maxzoom of any type\n let selectedSource: SourceCache = null;\n const layers = Object.values(style._layers);\n const sources = layers.flatMap((layer) => {\n if (layer.source && !layer.isHidden(zoom)) {\n const sourceCache = style.sourceCaches[layer.source];\n return [sourceCache];\n } else {\n return [];\n }\n });\n const vectorSources = sources.filter((source) => source.getSource().type === 'vector');\n const otherSources = sources.filter((source) => source.getSource().type !== 'vector');\n const considerSource = (source: SourceCache) => {\n if (!selectedSource || (selectedSource.getSource().maxzoom < source.getSource().maxzoom)) {\n selectedSource = source;\n }\n };\n vectorSources.forEach((source) => considerSource(source));\n if (!selectedSource) {\n otherSources.forEach((source) => considerSource(source));\n }\n return selectedSource;\n}\n","import {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {terrainUniformValues, terrainDepthUniformValues, terrainCoordsUniformValues} from './program/terrain_program';\nimport type {Painter} from './painter';\nimport type {Tile} from '../source/tile';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {ColorMode} from '../gl/color_mode';\nimport {Terrain} from './terrain';\n\n/**\n * Redraw the Depth Framebuffer\n * @param painter - the painter\n * @param terrain - the terrain\n */\nfunction drawDepth(painter: Painter, terrain: Terrain) {\n const context = painter.context;\n const gl = context.gl;\n const colorMode = ColorMode.unblended;\n const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, [0, 1]);\n const mesh = terrain.getTerrainMesh();\n const tiles = terrain.sourceCache.getRenderableTiles();\n const program = painter.useProgram('terrainDepth');\n context.bindFramebuffer.set(terrain.getFramebuffer('depth').framebuffer);\n context.viewport.set([0, 0, painter.width / devicePixelRatio, painter.height / devicePixelRatio]);\n context.clear({color: Color.transparent, depth: 1});\n for (const tile of tiles) {\n const terrainData = terrain.getTerrainData(tile.tileID);\n const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped());\n const uniformValues = terrainDepthUniformValues(posMatrix, terrain.getMeshFrameDelta(painter.transform.zoom));\n program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments);\n }\n context.bindFramebuffer.set(null);\n context.viewport.set([0, 0, painter.width, painter.height]);\n}\n\n/**\n * Redraw the Coords Framebuffers\n * @param painter - the painter\n * @param terrain - the terrain\n */\nfunction drawCoords(painter: Painter, terrain: Terrain) {\n const context = painter.context;\n const gl = context.gl;\n const colorMode = ColorMode.unblended;\n const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, [0, 1]);\n const mesh = terrain.getTerrainMesh();\n const coords = terrain.getCoordsTexture();\n const tiles = terrain.sourceCache.getRenderableTiles();\n\n // draw tile-coords into framebuffer\n const program = painter.useProgram('terrainCoords');\n context.bindFramebuffer.set(terrain.getFramebuffer('coords').framebuffer);\n context.viewport.set([0, 0, painter.width / devicePixelRatio, painter.height / devicePixelRatio]);\n context.clear({color: Color.transparent, depth: 1});\n terrain.coordsIndex = [];\n for (const tile of tiles) {\n const terrainData = terrain.getTerrainData(tile.tileID);\n context.activeTexture.set(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, coords.texture);\n const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped());\n const uniformValues = terrainCoordsUniformValues(posMatrix, 255 - terrain.coordsIndex.length, terrain.getMeshFrameDelta(painter.transform.zoom));\n program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments);\n terrain.coordsIndex.push(tile.tileID.key);\n }\n context.bindFramebuffer.set(null);\n context.viewport.set([0, 0, painter.width, painter.height]);\n}\n\nfunction drawTerrain(painter: Painter, terrain: Terrain, tiles: Array) {\n const context = painter.context;\n const gl = context.gl;\n const colorMode = painter.colorModeForRenderPass();\n const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D);\n const program = painter.useProgram('terrain');\n const mesh = terrain.getTerrainMesh();\n\n context.bindFramebuffer.set(null);\n context.viewport.set([0, 0, painter.width, painter.height]);\n\n for (const tile of tiles) {\n const texture = painter.renderToTexture.getTexture(tile);\n const terrainData = terrain.getTerrainData(tile.tileID);\n context.activeTexture.set(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, texture.texture);\n const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped());\n const uniformValues = terrainUniformValues(posMatrix, terrain.getMeshFrameDelta(painter.transform.zoom));\n program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments);\n }\n\n}\n\nexport {\n drawTerrain,\n drawDepth,\n drawCoords\n};\n","import {browser} from '../util/browser';\nimport {mat4, vec3} from 'gl-matrix';\nimport {SourceCache} from '../source/source_cache';\nimport {EXTENT} from '../data/extent';\nimport {pixelsToTileUnits} from '../source/pixels_to_tile_units';\nimport {SegmentVector} from '../data/segment';\nimport {RasterBoundsArray, PosArray, TriangleIndexArray, LineStripIndexArray} from '../data/array_types.g';\nimport rasterBoundsAttributes from '../data/raster_bounds_attributes';\nimport posAttributes from '../data/pos_attributes';\nimport {ProgramConfiguration} from '../data/program_configuration';\nimport {CrossTileSymbolIndex} from '../symbol/cross_tile_symbol_index';\nimport {shaders} from '../shaders/shaders';\nimport {Program} from './program';\nimport {programUniforms} from './program/program_uniforms';\nimport {Context} from '../gl/context';\nimport {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Texture} from './texture';\nimport {clippingMaskUniformValues} from './program/clipping_mask_program';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {drawSymbols} from './draw_symbol';\nimport {drawCircles} from './draw_circle';\nimport {drawHeatmap} from './draw_heatmap';\nimport {drawLine} from './draw_line';\nimport {drawFill} from './draw_fill';\nimport {drawFillExtrusion} from './draw_fill_extrusion';\nimport {drawHillshade} from './draw_hillshade';\nimport {drawRaster} from './draw_raster';\nimport {drawBackground} from './draw_background';\nimport {drawDebug, drawDebugPadding, selectDebugSource} from './draw_debug';\nimport {drawCustom} from './draw_custom';\nimport {drawDepth, drawCoords} from './draw_terrain';\nimport {OverscaledTileID} from '../source/tile_id';\n\nimport type {Transform} from '../geo/transform';\nimport type {Tile} from '../source/tile';\nimport type {Style} from '../style/style';\nimport type {StyleLayer} from '../style/style_layer';\nimport type {CrossFaded} from '../style/properties';\nimport type {LineAtlas} from './line_atlas';\nimport type {ImageManager} from './image_manager';\nimport type {GlyphManager} from './glyph_manager';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {IndexBuffer} from '../gl/index_buffer';\nimport type {DepthRangeType, DepthMaskType, DepthFuncType} from '../gl/types';\nimport type {ResolvedImage} from '@maplibre/maplibre-gl-style-spec';\nimport {RenderToTexture} from './render_to_texture';\n\nexport type RenderPass = 'offscreen' | 'opaque' | 'translucent';\n\ntype PainterOptions = {\n showOverdrawInspector: boolean;\n showTileBoundaries: boolean;\n showPadding: boolean;\n rotating: boolean;\n zooming: boolean;\n moving: boolean;\n fadeDuration: number;\n};\n\n/**\n * @internal\n * Initialize a new painter object.\n */\nexport class Painter {\n context: Context;\n transform: Transform;\n renderToTexture: RenderToTexture;\n _tileTextures: {\n [_: number]: Array;\n };\n numSublayers: number;\n depthEpsilon: number;\n emptyProgramConfiguration: ProgramConfiguration;\n width: number;\n height: number;\n pixelRatio: number;\n tileExtentBuffer: VertexBuffer;\n tileExtentSegments: SegmentVector;\n debugBuffer: VertexBuffer;\n debugSegments: SegmentVector;\n rasterBoundsBuffer: VertexBuffer;\n rasterBoundsSegments: SegmentVector;\n viewportBuffer: VertexBuffer;\n viewportSegments: SegmentVector;\n quadTriangleIndexBuffer: IndexBuffer;\n tileBorderIndexBuffer: IndexBuffer;\n _tileClippingMaskIDs: {[_: string]: number};\n stencilClearMode: StencilMode;\n style: Style;\n options: PainterOptions;\n lineAtlas: LineAtlas;\n imageManager: ImageManager;\n glyphManager: GlyphManager;\n depthRangeFor3D: DepthRangeType;\n opaquePassCutoff: number;\n renderPass: RenderPass;\n currentLayer: number;\n currentStencilSource: string;\n nextStencilID: number;\n id: string;\n _showOverdrawInspector: boolean;\n cache: {[_: string]: Program};\n crossTileSymbolIndex: CrossTileSymbolIndex;\n symbolFadeChange: number;\n debugOverlayTexture: Texture;\n debugOverlayCanvas: HTMLCanvasElement;\n // this object stores the current camera-matrix and the last render time\n // of the terrain-facilitators. e.g. depth & coords framebuffers\n // every time the camera-matrix changes the terrain-facilitators will be redrawn.\n terrainFacilitator: {dirty: boolean; matrix: mat4; renderTime: number};\n\n constructor(gl: WebGLRenderingContext | WebGL2RenderingContext, transform: Transform) {\n this.context = new Context(gl);\n this.transform = transform;\n this._tileTextures = {};\n this.terrainFacilitator = {dirty: true, matrix: mat4.create(), renderTime: 0};\n\n this.setup();\n\n // Within each layer there are multiple distinct z-planes that can be drawn to.\n // This is implemented using the WebGL depth buffer.\n this.numSublayers = SourceCache.maxUnderzooming + SourceCache.maxOverzooming + 1;\n this.depthEpsilon = 1 / Math.pow(2, 16);\n\n this.crossTileSymbolIndex = new CrossTileSymbolIndex();\n }\n\n /*\n * Update the GL viewport, projection matrix, and transforms to compensate\n * for a new width and height value.\n */\n resize(width: number, height: number, pixelRatio: number) {\n this.width = Math.floor(width * pixelRatio);\n this.height = Math.floor(height * pixelRatio);\n this.pixelRatio = pixelRatio;\n this.context.viewport.set([0, 0, this.width, this.height]);\n\n if (this.style) {\n for (const layerId of this.style._order) {\n this.style._layers[layerId].resize();\n }\n }\n }\n\n setup() {\n const context = this.context;\n\n const tileExtentArray = new PosArray();\n tileExtentArray.emplaceBack(0, 0);\n tileExtentArray.emplaceBack(EXTENT, 0);\n tileExtentArray.emplaceBack(0, EXTENT);\n tileExtentArray.emplaceBack(EXTENT, EXTENT);\n this.tileExtentBuffer = context.createVertexBuffer(tileExtentArray, posAttributes.members);\n this.tileExtentSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n\n const debugArray = new PosArray();\n debugArray.emplaceBack(0, 0);\n debugArray.emplaceBack(EXTENT, 0);\n debugArray.emplaceBack(0, EXTENT);\n debugArray.emplaceBack(EXTENT, EXTENT);\n this.debugBuffer = context.createVertexBuffer(debugArray, posAttributes.members);\n this.debugSegments = SegmentVector.simpleSegment(0, 0, 4, 5);\n\n const rasterBoundsArray = new RasterBoundsArray();\n rasterBoundsArray.emplaceBack(0, 0, 0, 0);\n rasterBoundsArray.emplaceBack(EXTENT, 0, EXTENT, 0);\n rasterBoundsArray.emplaceBack(0, EXTENT, 0, EXTENT);\n rasterBoundsArray.emplaceBack(EXTENT, EXTENT, EXTENT, EXTENT);\n this.rasterBoundsBuffer = context.createVertexBuffer(rasterBoundsArray, rasterBoundsAttributes.members);\n this.rasterBoundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n\n const viewportArray = new PosArray();\n viewportArray.emplaceBack(0, 0);\n viewportArray.emplaceBack(1, 0);\n viewportArray.emplaceBack(0, 1);\n viewportArray.emplaceBack(1, 1);\n this.viewportBuffer = context.createVertexBuffer(viewportArray, posAttributes.members);\n this.viewportSegments = SegmentVector.simpleSegment(0, 0, 4, 2);\n\n const tileLineStripIndices = new LineStripIndexArray();\n tileLineStripIndices.emplaceBack(0);\n tileLineStripIndices.emplaceBack(1);\n tileLineStripIndices.emplaceBack(3);\n tileLineStripIndices.emplaceBack(2);\n tileLineStripIndices.emplaceBack(0);\n this.tileBorderIndexBuffer = context.createIndexBuffer(tileLineStripIndices);\n\n const quadTriangleIndices = new TriangleIndexArray();\n quadTriangleIndices.emplaceBack(0, 1, 2);\n quadTriangleIndices.emplaceBack(2, 1, 3);\n this.quadTriangleIndexBuffer = context.createIndexBuffer(quadTriangleIndices);\n\n const gl = this.context.gl;\n this.stencilClearMode = new StencilMode({func: gl.ALWAYS, mask: 0}, 0x0, 0xFF, gl.ZERO, gl.ZERO, gl.ZERO);\n }\n\n /*\n * Reset the drawing canvas by clearing the stencil buffer so that we can draw\n * new tiles at the same location, while retaining previously drawn pixels.\n */\n clearStencil() {\n const context = this.context;\n const gl = context.gl;\n\n this.nextStencilID = 1;\n this.currentStencilSource = undefined;\n\n // As a temporary workaround for https://github.com/mapbox/mapbox-gl-js/issues/5490,\n // pending an upstream fix, we draw a fullscreen stencil=0 clipping mask here,\n // effectively clearing the stencil buffer: once an upstream patch lands, remove\n // this function in favor of context.clear({ stencil: 0x0 })\n\n const matrix = mat4.create();\n mat4.ortho(matrix, 0, this.width, this.height, 0, 0, 1);\n mat4.scale(matrix, matrix, [gl.drawingBufferWidth, gl.drawingBufferHeight, 0]);\n\n this.useProgram('clippingMask').draw(context, gl.TRIANGLES,\n DepthMode.disabled, this.stencilClearMode, ColorMode.disabled, CullFaceMode.disabled,\n clippingMaskUniformValues(matrix), null,\n '$clipping', this.viewportBuffer,\n this.quadTriangleIndexBuffer, this.viewportSegments);\n }\n\n _renderTileClippingMasks(layer: StyleLayer, tileIDs: Array) {\n if (this.currentStencilSource === layer.source || !layer.isTileClipped() || !tileIDs || !tileIDs.length) return;\n\n this.currentStencilSource = layer.source;\n\n const context = this.context;\n const gl = context.gl;\n\n if (this.nextStencilID + tileIDs.length > 256) {\n // we'll run out of fresh IDs so we need to clear and start from scratch\n this.clearStencil();\n }\n\n context.setColorMode(ColorMode.disabled);\n context.setDepthMode(DepthMode.disabled);\n\n const program = this.useProgram('clippingMask');\n\n this._tileClippingMaskIDs = {};\n\n for (const tileID of tileIDs) {\n const id = this._tileClippingMaskIDs[tileID.key] = this.nextStencilID++;\n const terrainData = this.style.map.terrain && this.style.map.terrain.getTerrainData(tileID);\n\n program.draw(context, gl.TRIANGLES, DepthMode.disabled,\n // Tests will always pass, and ref value will be written to stencil buffer.\n new StencilMode({func: gl.ALWAYS, mask: 0}, id, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE),\n ColorMode.disabled, CullFaceMode.disabled, clippingMaskUniformValues(tileID.posMatrix),\n terrainData, '$clipping', this.tileExtentBuffer,\n this.quadTriangleIndexBuffer, this.tileExtentSegments);\n }\n }\n\n stencilModeFor3D(): StencilMode {\n this.currentStencilSource = undefined;\n\n if (this.nextStencilID + 1 > 256) {\n this.clearStencil();\n }\n\n const id = this.nextStencilID++;\n const gl = this.context.gl;\n return new StencilMode({func: gl.NOTEQUAL, mask: 0xFF}, id, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE);\n }\n\n stencilModeForClipping(tileID: OverscaledTileID): StencilMode {\n const gl = this.context.gl;\n return new StencilMode({func: gl.EQUAL, mask: 0xFF}, this._tileClippingMaskIDs[tileID.key], 0x00, gl.KEEP, gl.KEEP, gl.REPLACE);\n }\n\n /*\n * Sort coordinates by Z as drawing tiles is done in Z-descending order.\n * All children with the same Z write the same stencil value. Children\n * stencil values are greater than parent's. This is used only for raster\n * and raster-dem tiles, which are already clipped to tile boundaries, to\n * mask area of tile overlapped by children tiles.\n * Stencil ref values continue range used in _tileClippingMaskIDs.\n *\n * Returns [StencilMode for tile overscaleZ map, sortedCoords].\n */\n stencilConfigForOverlap(tileIDs: Array): [{\n [_: number]: Readonly;\n }, Array] {\n const gl = this.context.gl;\n const coords = tileIDs.sort((a, b) => b.overscaledZ - a.overscaledZ);\n const minTileZ = coords[coords.length - 1].overscaledZ;\n const stencilValues = coords[0].overscaledZ - minTileZ + 1;\n if (stencilValues > 1) {\n this.currentStencilSource = undefined;\n if (this.nextStencilID + stencilValues > 256) {\n this.clearStencil();\n }\n const zToStencilMode = {};\n for (let i = 0; i < stencilValues; i++) {\n zToStencilMode[i + minTileZ] = new StencilMode({func: gl.GEQUAL, mask: 0xFF}, i + this.nextStencilID, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE);\n }\n this.nextStencilID += stencilValues;\n return [zToStencilMode, coords];\n }\n return [{[minTileZ]: StencilMode.disabled}, coords];\n }\n\n colorModeForRenderPass(): Readonly {\n const gl = this.context.gl;\n if (this._showOverdrawInspector) {\n const numOverdrawSteps = 8;\n const a = 1 / numOverdrawSteps;\n\n return new ColorMode([gl.CONSTANT_COLOR, gl.ONE], new Color(a, a, a, 0), [true, true, true, true]);\n } else if (this.renderPass === 'opaque') {\n return ColorMode.unblended;\n } else {\n return ColorMode.alphaBlended;\n }\n }\n\n depthModeForSublayer(n: number, mask: DepthMaskType, func?: DepthFuncType | null): Readonly {\n if (!this.opaquePassEnabledForLayer()) return DepthMode.disabled;\n const depth = 1 - ((1 + this.currentLayer) * this.numSublayers + n) * this.depthEpsilon;\n return new DepthMode(func || this.context.gl.LEQUAL, mask, [depth, depth]);\n }\n\n /*\n * The opaque pass and 3D layers both use the depth buffer.\n * Layers drawn above 3D layers need to be drawn using the\n * painter's algorithm so that they appear above 3D features.\n * This returns true for layers that can be drawn using the\n * opaque pass.\n */\n opaquePassEnabledForLayer() {\n return this.currentLayer < this.opaquePassCutoff;\n }\n\n render(style: Style, options: PainterOptions) {\n this.style = style;\n this.options = options;\n\n this.lineAtlas = style.lineAtlas;\n this.imageManager = style.imageManager;\n this.glyphManager = style.glyphManager;\n\n this.symbolFadeChange = style.placement.symbolFadeChange(browser.now());\n\n this.imageManager.beginFrame();\n\n const layerIds = this.style._order;\n const sourceCaches = this.style.sourceCaches;\n\n const coordsAscending: {[_: string]: Array} = {};\n const coordsDescending: {[_: string]: Array} = {};\n const coordsDescendingSymbol: {[_: string]: Array} = {};\n\n for (const id in sourceCaches) {\n const sourceCache = sourceCaches[id];\n if (sourceCache.used) {\n sourceCache.prepare(this.context);\n }\n\n coordsAscending[id] = sourceCache.getVisibleCoordinates();\n coordsDescending[id] = coordsAscending[id].slice().reverse();\n coordsDescendingSymbol[id] = sourceCache.getVisibleCoordinates(true).reverse();\n }\n\n this.opaquePassCutoff = Infinity;\n for (let i = 0; i < layerIds.length; i++) {\n const layerId = layerIds[i];\n if (this.style._layers[layerId].is3D()) {\n this.opaquePassCutoff = i;\n break;\n }\n }\n\n if (this.renderToTexture) {\n this.renderToTexture.prepareForRender(this.style, this.transform.zoom);\n // this is disabled, because render-to-texture is rendering all layers from bottom to top.\n this.opaquePassCutoff = 0;\n\n // update coords/depth-framebuffer on camera movement, or tile reloading\n const newTiles = this.style.map.terrain.sourceCache.tilesAfterTime(this.terrainFacilitator.renderTime);\n if (this.terrainFacilitator.dirty || !mat4.equals(this.terrainFacilitator.matrix, this.transform.projMatrix) || newTiles.length) {\n mat4.copy(this.terrainFacilitator.matrix, this.transform.projMatrix);\n this.terrainFacilitator.renderTime = Date.now();\n this.terrainFacilitator.dirty = false;\n drawDepth(this, this.style.map.terrain);\n drawCoords(this, this.style.map.terrain);\n }\n }\n\n // Offscreen pass ===============================================\n // We first do all rendering that requires rendering to a separate\n // framebuffer, and then save those for rendering back to the map\n // later: in doing this we avoid doing expensive framebuffer restores.\n this.renderPass = 'offscreen';\n\n for (const layerId of layerIds) {\n const layer = this.style._layers[layerId];\n if (!layer.hasOffscreenPass() || layer.isHidden(this.transform.zoom)) continue;\n\n const coords = coordsDescending[layer.source];\n if (layer.type !== 'custom' && !coords.length) continue;\n\n this.renderLayer(this, sourceCaches[layer.source], layer, coords);\n }\n\n // Rebind the main framebuffer now that all offscreen layers have been rendered:\n this.context.bindFramebuffer.set(null);\n\n // Clear buffers in preparation for drawing to the main framebuffer\n this.context.clear({color: options.showOverdrawInspector ? Color.black : Color.transparent, depth: 1});\n this.clearStencil();\n\n this._showOverdrawInspector = options.showOverdrawInspector;\n this.depthRangeFor3D = [0, 1 - ((style._order.length + 2) * this.numSublayers * this.depthEpsilon)];\n\n // Opaque pass ===============================================\n // Draw opaque layers top-to-bottom first.\n if (!this.renderToTexture) {\n this.renderPass = 'opaque';\n\n for (this.currentLayer = layerIds.length - 1; this.currentLayer >= 0; this.currentLayer--) {\n const layer = this.style._layers[layerIds[this.currentLayer]];\n const sourceCache = sourceCaches[layer.source];\n const coords = coordsAscending[layer.source];\n\n this._renderTileClippingMasks(layer, coords);\n this.renderLayer(this, sourceCache, layer, coords);\n }\n }\n\n // Translucent pass ===============================================\n // Draw all other layers bottom-to-top.\n this.renderPass = 'translucent';\n\n for (this.currentLayer = 0; this.currentLayer < layerIds.length; this.currentLayer++) {\n const layer = this.style._layers[layerIds[this.currentLayer]];\n const sourceCache = sourceCaches[layer.source];\n\n if (this.renderToTexture && this.renderToTexture.renderLayer(layer)) continue;\n\n // For symbol layers in the translucent pass, we add extra tiles to the renderable set\n // for cross-tile symbol fading. Symbol layers don't use tile clipping, so no need to render\n // separate clipping masks\n const coords = (layer.type === 'symbol' ? coordsDescendingSymbol : coordsDescending)[layer.source];\n\n this._renderTileClippingMasks(layer, coordsAscending[layer.source]);\n this.renderLayer(this, sourceCache, layer, coords);\n }\n\n if (this.options.showTileBoundaries) {\n const selectedSource = selectDebugSource(this.style, this.transform.zoom);\n if (selectedSource) {\n drawDebug(this, selectedSource, selectedSource.getVisibleCoordinates());\n }\n }\n\n if (this.options.showPadding) {\n drawDebugPadding(this);\n }\n\n // Set defaults for most GL values so that anyone using the state after the render\n // encounters more expected values.\n this.context.setDefault();\n }\n\n renderLayer(painter: Painter, sourceCache: SourceCache, layer: StyleLayer, coords: Array) {\n if (layer.isHidden(this.transform.zoom)) return;\n if (layer.type !== 'background' && layer.type !== 'custom' && !(coords || []).length) return;\n this.id = layer.id;\n\n switch (layer.type) {\n case 'symbol':\n drawSymbols(painter, sourceCache, layer as any, coords, this.style.placement.variableOffsets);\n break;\n case 'circle':\n drawCircles(painter, sourceCache, layer as any, coords);\n break;\n case 'heatmap':\n drawHeatmap(painter, sourceCache, layer as any, coords);\n break;\n case 'line':\n drawLine(painter, sourceCache, layer as any, coords);\n break;\n case 'fill':\n drawFill(painter, sourceCache, layer as any, coords);\n break;\n case 'fill-extrusion':\n drawFillExtrusion(painter, sourceCache, layer as any, coords);\n break;\n case 'hillshade':\n drawHillshade(painter, sourceCache, layer as any, coords);\n break;\n case 'raster':\n drawRaster(painter, sourceCache, layer as any, coords);\n break;\n case 'background':\n drawBackground(painter, sourceCache, layer as any, coords);\n break;\n case 'custom':\n drawCustom(painter, sourceCache, layer as any);\n break;\n }\n }\n\n /**\n * Transform a matrix to incorporate the *-translate and *-translate-anchor properties into it.\n * @param inViewportPixelUnitsUnits - True when the units accepted by the matrix are in viewport pixels instead of tile units.\n * @returns matrix\n */\n translatePosMatrix(matrix: mat4, tile: Tile, translate: [number, number], translateAnchor: 'map' | 'viewport', inViewportPixelUnitsUnits?: boolean): mat4 {\n if (!translate[0] && !translate[1]) return matrix;\n\n const angle = inViewportPixelUnitsUnits ?\n (translateAnchor === 'map' ? this.transform.angle : 0) :\n (translateAnchor === 'viewport' ? -this.transform.angle : 0);\n\n if (angle) {\n const sinA = Math.sin(angle);\n const cosA = Math.cos(angle);\n translate = [\n translate[0] * cosA - translate[1] * sinA,\n translate[0] * sinA + translate[1] * cosA\n ];\n }\n\n const translation = [\n inViewportPixelUnitsUnits ? translate[0] : pixelsToTileUnits(tile, translate[0], this.transform.zoom),\n inViewportPixelUnitsUnits ? translate[1] : pixelsToTileUnits(tile, translate[1], this.transform.zoom),\n 0\n ] as vec3;\n\n const translatedMatrix = new Float32Array(16);\n mat4.translate(translatedMatrix, matrix, translation);\n return translatedMatrix;\n }\n\n saveTileTexture(texture: Texture) {\n const textures = this._tileTextures[texture.size[0]];\n if (!textures) {\n this._tileTextures[texture.size[0]] = [texture];\n } else {\n textures.push(texture);\n }\n }\n\n getTileTexture(size: number) {\n const textures = this._tileTextures[size];\n return textures && textures.length > 0 ? textures.pop() : null;\n }\n\n /**\n * Checks whether a pattern image is needed, and if it is, whether it is not loaded.\n *\n * @returns true if a needed image is missing and rendering needs to be skipped.\n */\n isPatternMissing(image?: CrossFaded | null): boolean {\n if (!image) return false;\n if (!image.from || !image.to) return true;\n const imagePosA = this.imageManager.getPattern(image.from.toString());\n const imagePosB = this.imageManager.getPattern(image.to.toString());\n return !imagePosA || !imagePosB;\n }\n\n useProgram(name: string, programConfiguration?: ProgramConfiguration | null): Program {\n this.cache = this.cache || {};\n const key = name +\n (programConfiguration ? programConfiguration.cacheKey : '') +\n (this._showOverdrawInspector ? '/overdraw' : '') +\n (this.style.map.terrain ? '/terrain' : '');\n if (!this.cache[key]) {\n this.cache[key] = new Program(\n this.context,\n shaders[name],\n programConfiguration,\n programUniforms[name],\n this._showOverdrawInspector,\n this.style.map.terrain\n );\n }\n return this.cache[key];\n }\n\n /*\n * Reset some GL state to default values to avoid hard-to-debug bugs\n * in custom layers.\n */\n setCustomLayerDefaults() {\n // Prevent custom layers from unintentionally modify the last VAO used.\n // All other state is state is restored on it's own, but for VAOs it's\n // simpler to unbind so that we don't have to track the state of VAOs.\n this.context.unbindVAO();\n\n // The default values for this state is meaningful and often expected.\n // Leaving this state dirty could cause a lot of confusion for users.\n this.context.cullFace.setDefault();\n this.context.activeTexture.setDefault();\n this.context.pixelStoreUnpack.setDefault();\n this.context.pixelStoreUnpackPremultiplyAlpha.setDefault();\n this.context.pixelStoreUnpackFlipY.setDefault();\n }\n\n /*\n * Set GL state that is shared by all layers.\n */\n setBaseState() {\n const gl = this.context.gl;\n this.context.cullFace.set(false);\n this.context.viewport.set([0, 0, this.width, this.height]);\n this.context.blendEquation.set(gl.FUNC_ADD);\n }\n\n initDebugOverlayCanvas() {\n if (this.debugOverlayCanvas == null) {\n this.debugOverlayCanvas = document.createElement('canvas');\n this.debugOverlayCanvas.width = 512;\n this.debugOverlayCanvas.height = 512;\n const gl = this.context.gl;\n this.debugOverlayTexture = new Texture(this.context, this.debugOverlayCanvas, gl.RGBA);\n }\n }\n\n destroy() {\n if (this.debugOverlayTexture) {\n this.debugOverlayTexture.destroy();\n }\n }\n\n /*\n * Return true if drawing buffer size is != from requested size.\n * That means that we've reached GL limits somehow.\n * Note: drawing buffer size changes only when canvas size changes\n */\n overLimit() {\n const {drawingBufferWidth, drawingBufferHeight} = this.context.gl;\n return this.width !== drawingBufferWidth || this.height !== drawingBufferHeight;\n }\n}\n","import {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Program} from './program';\nimport {circleUniformValues} from './program/circle_program';\nimport {SegmentVector} from '../data/segment';\nimport {OverscaledTileID} from '../source/tile_id';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {CircleStyleLayer} from '../style/style_layer/circle_style_layer';\nimport type {CircleBucket} from '../data/bucket/circle_bucket';\nimport type {ProgramConfiguration} from '../data/program_configuration';\nimport type {VertexBuffer} from '../gl/vertex_buffer';\nimport type {IndexBuffer} from '../gl/index_buffer';\nimport type {UniformValues} from './uniform_binding';\nimport type {CircleUniformsType} from './program/circle_program';\nimport type {TerrainData} from '../render/terrain';\n\ntype TileRenderState = {\n programConfiguration: ProgramConfiguration;\n program: Program;\n layoutVertexBuffer: VertexBuffer;\n indexBuffer: IndexBuffer;\n uniformValues: UniformValues;\n terrainData: TerrainData;\n};\n\ntype SegmentsTileRenderState = {\n segments: SegmentVector;\n sortKey: number;\n state: TileRenderState;\n};\n\nexport function drawCircles(painter: Painter, sourceCache: SourceCache, layer: CircleStyleLayer, coords: Array) {\n if (painter.renderPass !== 'translucent') return;\n\n const opacity = layer.paint.get('circle-opacity');\n const strokeWidth = layer.paint.get('circle-stroke-width');\n const strokeOpacity = layer.paint.get('circle-stroke-opacity');\n const sortFeaturesByKey = !layer.layout.get('circle-sort-key').isConstant();\n\n if (opacity.constantOr(1) === 0 && (strokeWidth.constantOr(1) === 0 || strokeOpacity.constantOr(1) === 0)) {\n return;\n }\n\n const context = painter.context;\n const gl = context.gl;\n\n const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n // Turn off stencil testing to allow circles to be drawn across boundaries,\n // so that large circles are not clipped to tiles\n const stencilMode = StencilMode.disabled;\n const colorMode = painter.colorModeForRenderPass();\n\n const segmentsRenderStates: Array = [];\n\n for (let i = 0; i < coords.length; i++) {\n const coord = coords[i];\n\n const tile = sourceCache.getTile(coord);\n const bucket: CircleBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const program = painter.useProgram('circle', programConfiguration);\n const layoutVertexBuffer = bucket.layoutVertexBuffer;\n const indexBuffer = bucket.indexBuffer;\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n const uniformValues = circleUniformValues(painter, coord, tile, layer);\n\n const state: TileRenderState = {\n programConfiguration,\n program,\n layoutVertexBuffer,\n indexBuffer,\n uniformValues,\n terrainData\n };\n\n if (sortFeaturesByKey) {\n const oldSegments = bucket.segments.get();\n for (const segment of oldSegments) {\n segmentsRenderStates.push({\n segments: new SegmentVector([segment]),\n sortKey: (segment.sortKey as any as number),\n state\n });\n }\n } else {\n segmentsRenderStates.push({\n segments: bucket.segments,\n sortKey: 0,\n state\n });\n }\n\n }\n\n if (sortFeaturesByKey) {\n segmentsRenderStates.sort((a, b) => a.sortKey - b.sortKey);\n }\n\n for (const segmentsState of segmentsRenderStates) {\n const {programConfiguration, program, layoutVertexBuffer, indexBuffer, uniformValues, terrainData} = segmentsState.state;\n const segments = segmentsState.segments;\n\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id,\n layoutVertexBuffer, indexBuffer, segments,\n layer.paint, painter.transform.zoom, programConfiguration);\n }\n}\n","import {Texture} from './texture';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\nimport {ColorMode} from '../gl/color_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Context} from '../gl/context';\nimport {Framebuffer} from '../gl/framebuffer';\nimport {\n heatmapUniformValues,\n heatmapTextureUniformValues\n} from './program/heatmap_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {HeatmapStyleLayer} from '../style/style_layer/heatmap_style_layer';\nimport type {HeatmapBucket} from '../data/bucket/heatmap_bucket';\nimport type {OverscaledTileID} from '../source/tile_id';\n\nexport function drawHeatmap(painter: Painter, sourceCache: SourceCache, layer: HeatmapStyleLayer, coords: Array) {\n if (layer.paint.get('heatmap-opacity') === 0) {\n return;\n }\n\n if (painter.renderPass === 'offscreen') {\n const context = painter.context;\n const gl = context.gl;\n\n // Allow kernels to be drawn across boundaries, so that\n // large kernels are not clipped to tiles\n const stencilMode = StencilMode.disabled;\n // Turn on additive blending for kernels, which is a key aspect of kernel density estimation formula\n const colorMode = new ColorMode([gl.ONE, gl.ONE], Color.transparent, [true, true, true, true]);\n\n bindFramebuffer(context, painter, layer);\n\n context.clear({color: Color.transparent});\n\n for (let i = 0; i < coords.length; i++) {\n const coord = coords[i];\n\n // Skip tiles that have uncovered parents to avoid flickering; we don't need\n // to use complex tile masking here because the change between zoom levels is subtle,\n // so it's fine to simply render the parent until all its 4 children are loaded\n if (sourceCache.hasRenderableParent(coord)) continue;\n\n const tile = sourceCache.getTile(coord);\n const bucket: HeatmapBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const program = painter.useProgram('heatmap', programConfiguration);\n const {zoom} = painter.transform;\n\n program.draw(context, gl.TRIANGLES, DepthMode.disabled, stencilMode, colorMode, CullFaceMode.disabled,\n heatmapUniformValues(coord.posMatrix, tile, zoom, layer.paint.get('heatmap-intensity')), null,\n layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer,\n bucket.segments, layer.paint, painter.transform.zoom,\n programConfiguration);\n }\n\n context.viewport.set([0, 0, painter.width, painter.height]);\n\n } else if (painter.renderPass === 'translucent') {\n painter.context.setColorMode(painter.colorModeForRenderPass());\n renderTextureToMap(painter, layer);\n }\n}\n\nfunction bindFramebuffer(context: Context, painter: Painter, layer: HeatmapStyleLayer) {\n const gl = context.gl;\n context.activeTexture.set(gl.TEXTURE1);\n\n // Use a 4x downscaled screen texture for better performance\n context.viewport.set([0, 0, painter.width / 4, painter.height / 4]);\n\n let fbo = layer.heatmapFbo;\n\n if (!fbo) {\n const texture = gl.createTexture();\n gl.bindTexture(gl.TEXTURE_2D, texture);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);\n gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);\n\n fbo = layer.heatmapFbo = context.createFramebuffer(painter.width / 4, painter.height / 4, false, false);\n\n bindTextureToFramebuffer(context, painter, texture, fbo);\n\n } else {\n gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get());\n context.bindFramebuffer.set(fbo.framebuffer);\n }\n}\n\nfunction bindTextureToFramebuffer(context: Context, painter: Painter, texture: WebGLTexture, fbo: Framebuffer) {\n const gl = context.gl;\n // Use the higher precision half-float texture where available (producing much smoother looking heatmaps);\n // Otherwise, fall back to a low precision texture\n\n const numType = context.HALF_FLOAT ?? gl.UNSIGNED_BYTE;\n const internalFormat = context.RGBA16F ?? gl.RGBA;\n\n gl.texImage2D(gl.TEXTURE_2D, 0, internalFormat, painter.width / 4, painter.height / 4, 0, gl.RGBA, numType, null);\n fbo.colorAttachment.set(texture);\n}\n\nfunction renderTextureToMap(painter: Painter, layer: HeatmapStyleLayer) {\n const context = painter.context;\n const gl = context.gl;\n\n // Here we bind two different textures from which we'll sample in drawing\n // heatmaps: the kernel texture, prepared in the offscreen pass, and a\n // color ramp texture.\n const fbo = layer.heatmapFbo;\n if (!fbo) return;\n context.activeTexture.set(gl.TEXTURE0);\n gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get());\n\n context.activeTexture.set(gl.TEXTURE1);\n let colorRampTexture = layer.colorRampTexture;\n if (!colorRampTexture) {\n colorRampTexture = layer.colorRampTexture = new Texture(context, layer.colorRamp, gl.RGBA);\n }\n colorRampTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n\n painter.useProgram('heatmapTexture').draw(context, gl.TRIANGLES,\n DepthMode.disabled, StencilMode.disabled, painter.colorModeForRenderPass(), CullFaceMode.disabled,\n heatmapTextureUniformValues(painter, layer, 0, 1), null,\n layer.id, painter.viewportBuffer, painter.quadTriangleIndexBuffer,\n painter.viewportSegments, layer.paint, painter.transform.zoom);\n}\n","import {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {Texture} from './texture';\nimport {\n lineUniformValues,\n linePatternUniformValues,\n lineSDFUniformValues,\n lineGradientUniformValues\n} from './program/line_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {LineStyleLayer} from '../style/style_layer/line_style_layer';\nimport type {LineBucket} from '../data/bucket/line_bucket';\nimport type {OverscaledTileID} from '../source/tile_id';\nimport {clamp, nextPowerOfTwo} from '../util/util';\nimport {renderColorRamp} from '../util/color_ramp';\nimport {EXTENT} from '../data/extent';\n\nexport function drawLine(painter: Painter, sourceCache: SourceCache, layer: LineStyleLayer, coords: Array) {\n if (painter.renderPass !== 'translucent') return;\n\n const opacity = layer.paint.get('line-opacity');\n const width = layer.paint.get('line-width');\n if (opacity.constantOr(1) === 0 || width.constantOr(1) === 0) return;\n\n const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n const colorMode = painter.colorModeForRenderPass();\n\n const dasharray = layer.paint.get('line-dasharray');\n const patternProperty = layer.paint.get('line-pattern');\n const image = patternProperty.constantOr(1 as any);\n\n const gradient = layer.paint.get('line-gradient');\n const crossfade = layer.getCrossfadeParameters();\n\n const programId =\n image ? 'linePattern' :\n dasharray ? 'lineSDF' :\n gradient ? 'lineGradient' : 'line';\n\n const context = painter.context;\n const gl = context.gl;\n\n let firstTile = true;\n\n for (const coord of coords) {\n const tile = sourceCache.getTile(coord);\n\n if (image && !tile.patternsLoaded()) continue;\n\n const bucket: LineBucket = (tile.getBucket(layer) as any);\n if (!bucket) continue;\n\n const programConfiguration = bucket.programConfigurations.get(layer.id);\n const prevProgram = painter.context.program.get();\n const program = painter.useProgram(programId, programConfiguration);\n const programChanged = firstTile || program.program !== prevProgram;\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord);\n\n const constantPattern = patternProperty.constantOr(null);\n if (constantPattern && tile.imageAtlas) {\n const atlas = tile.imageAtlas;\n const posTo = atlas.patternPositions[constantPattern.to.toString()];\n const posFrom = atlas.patternPositions[constantPattern.from.toString()];\n if (posTo && posFrom) programConfiguration.setConstantPatternPositions(posTo, posFrom);\n }\n\n const terrainCoord = terrainData ? coord : null;\n const uniformValues = image ? linePatternUniformValues(painter, tile, layer, crossfade, terrainCoord) :\n dasharray ? lineSDFUniformValues(painter, tile, layer, dasharray, crossfade, terrainCoord) :\n gradient ? lineGradientUniformValues(painter, tile, layer, bucket.lineClipsArray.length, terrainCoord) :\n lineUniformValues(painter, tile, layer, terrainCoord);\n\n if (image) {\n context.activeTexture.set(gl.TEXTURE0);\n tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE);\n programConfiguration.updatePaintBuffers(crossfade);\n } else if (dasharray && (programChanged || painter.lineAtlas.dirty)) {\n context.activeTexture.set(gl.TEXTURE0);\n painter.lineAtlas.bind(context);\n } else if (gradient) {\n const layerGradient = bucket.gradients[layer.id];\n let gradientTexture = layerGradient.texture;\n if (layer.gradientVersion !== layerGradient.version) {\n let textureResolution = 256;\n if (layer.stepInterpolant) {\n const sourceMaxZoom = sourceCache.getSource().maxzoom;\n const potentialOverzoom = coord.canonical.z === sourceMaxZoom ?\n Math.ceil(1 << (painter.transform.maxZoom - coord.canonical.z)) : 1;\n const lineLength = bucket.maxLineLength / EXTENT;\n // Logical pixel tile size is 512px, and 1024px right before current zoom + 1\n const maxTilePixelSize = 1024;\n // Maximum possible texture coverage heuristic, bound by hardware max texture size\n const maxTextureCoverage = lineLength * maxTilePixelSize * potentialOverzoom;\n textureResolution = clamp(nextPowerOfTwo(maxTextureCoverage), 256, context.maxTextureSize);\n }\n layerGradient.gradient = renderColorRamp({\n expression: layer.gradientExpression(),\n evaluationKey: 'lineProgress',\n resolution: textureResolution,\n image: layerGradient.gradient || undefined,\n clips: bucket.lineClipsArray\n });\n if (layerGradient.texture) {\n layerGradient.texture.update(layerGradient.gradient);\n } else {\n layerGradient.texture = new Texture(context, layerGradient.gradient, gl.RGBA);\n }\n layerGradient.version = layer.gradientVersion;\n gradientTexture = layerGradient.texture;\n }\n context.activeTexture.set(gl.TEXTURE0);\n gradientTexture.bind(layer.stepInterpolant ? gl.NEAREST : gl.LINEAR, gl.CLAMP_TO_EDGE);\n }\n\n program.draw(context, gl.TRIANGLES, depthMode,\n painter.stencilModeForClipping(coord), colorMode, CullFaceMode.disabled, uniformValues, terrainData,\n layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer, bucket.segments,\n layer.paint, painter.transform.zoom, programConfiguration, bucket.layoutVertexBuffer2);\n\n firstTile = false;\n // once refactored so that bound texture state is managed, we'll also be able to remove this firstTile/programChanged logic\n }\n}\n","import {StencilMode} from '../gl/stencil_mode';\nimport {DepthMode} from '../gl/depth_mode';\nimport {CullFaceMode} from '../gl/cull_face_mode';\nimport {\n backgroundUniformValues,\n backgroundPatternUniformValues\n} from './program/background_program';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {BackgroundStyleLayer} from '../style/style_layer/background_style_layer';\nimport {OverscaledTileID} from '../source/tile_id';\n\nexport function drawBackground(painter: Painter, sourceCache: SourceCache, layer: BackgroundStyleLayer, coords?: Array) {\n const color = layer.paint.get('background-color');\n const opacity = layer.paint.get('background-opacity');\n\n if (opacity === 0) return;\n\n const context = painter.context;\n const gl = context.gl;\n const transform = painter.transform;\n const tileSize = transform.tileSize;\n const image = layer.paint.get('background-pattern');\n if (painter.isPatternMissing(image)) return;\n\n const pass = (!image && color.a === 1 && opacity === 1 && painter.opaquePassEnabledForLayer()) ? 'opaque' : 'translucent';\n if (painter.renderPass !== pass) return;\n\n const stencilMode = StencilMode.disabled;\n const depthMode = painter.depthModeForSublayer(0, pass === 'opaque' ? DepthMode.ReadWrite : DepthMode.ReadOnly);\n const colorMode = painter.colorModeForRenderPass();\n const program = painter.useProgram(image ? 'backgroundPattern' : 'background');\n const tileIDs = coords ? coords : transform.coveringTiles({tileSize, terrain: painter.style.map.terrain});\n\n if (image) {\n context.activeTexture.set(gl.TEXTURE0);\n painter.imageManager.bind(painter.context);\n }\n\n const crossfade = layer.getCrossfadeParameters();\n for (const tileID of tileIDs) {\n const matrix = coords ? tileID.posMatrix : painter.transform.calculatePosMatrix(tileID.toUnwrapped());\n const uniformValues = image ?\n backgroundPatternUniformValues(matrix, opacity, painter, image, {tileID, tileSize}, crossfade) :\n backgroundUniformValues(matrix, opacity, color);\n const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(tileID);\n\n program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled,\n uniformValues, terrainData, layer.id, painter.tileExtentBuffer,\n painter.quadTriangleIndexBuffer, painter.tileExtentSegments);\n }\n}\n","import {DepthMode} from '../gl/depth_mode';\nimport {StencilMode} from '../gl/stencil_mode';\n\nimport type {Painter} from './painter';\nimport type {SourceCache} from '../source/source_cache';\nimport type {CustomStyleLayer} from '../style/style_layer/custom_style_layer';\n\nexport function drawCustom(painter: Painter, sourceCache: SourceCache, layer: CustomStyleLayer) {\n\n const context = painter.context;\n const implementation = layer.implementation;\n\n if (painter.renderPass === 'offscreen') {\n\n const prerender = implementation.prerender;\n if (prerender) {\n painter.setCustomLayerDefaults();\n context.setColorMode(painter.colorModeForRenderPass());\n\n prerender.call(implementation, context.gl, painter.transform.customLayerMatrix());\n\n context.setDirty();\n painter.setBaseState();\n }\n\n } else if (painter.renderPass === 'translucent') {\n\n painter.setCustomLayerDefaults();\n\n context.setColorMode(painter.colorModeForRenderPass());\n context.setStencilMode(StencilMode.disabled);\n\n const depthMode = implementation.renderingMode === '3d' ?\n new DepthMode(painter.context.gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D) :\n painter.depthModeForSublayer(0, DepthMode.ReadOnly);\n\n context.setDepthMode(depthMode);\n\n implementation.render(context.gl, painter.transform.customLayerMatrix());\n\n context.setDirty();\n painter.setBaseState();\n context.bindFramebuffer.set(null);\n }\n}\n","import {mat4, vec3, vec4} from 'gl-matrix';\n\nclass Frustum {\n\n constructor(public points: vec4[], public planes: vec4[]) { }\n\n public static fromInvProjectionMatrix(invProj: mat4, worldSize: number, zoom: number): Frustum {\n const clipSpaceCorners = [\n [-1, 1, -1, 1],\n [1, 1, -1, 1],\n [1, -1, -1, 1],\n [-1, -1, -1, 1],\n [-1, 1, 1, 1],\n [1, 1, 1, 1],\n [1, -1, 1, 1],\n [-1, -1, 1, 1]\n ];\n\n const scale = Math.pow(2, zoom);\n\n // Transform frustum corner points from clip space to tile space, Z to meters\n const frustumCoords = clipSpaceCorners.map(v => {\n v = vec4.transformMat4([] as any, v as any, invProj) as any;\n const s = 1.0 / v[3] / worldSize * scale;\n return vec4.mul(v as any, v as any, [s, s, 1.0 / v[3], s] as vec4);\n });\n\n const frustumPlanePointIndices = [\n [0, 1, 2], // near\n [6, 5, 4], // far\n [0, 3, 7], // left\n [2, 1, 5], // right\n [3, 2, 6], // bottom\n [0, 4, 5] // top\n ];\n\n const frustumPlanes = frustumPlanePointIndices.map((p: number[]) => {\n const a = vec3.sub([] as any, frustumCoords[p[0]] as vec3, frustumCoords[p[1]] as vec3);\n const b = vec3.sub([] as any, frustumCoords[p[2]] as vec3, frustumCoords[p[1]] as vec3);\n const n = vec3.normalize([] as any, vec3.cross([] as any, a, b)) as any;\n const d = -vec3.dot(n, frustumCoords[p[1]] as vec3);\n return n.concat(d);\n });\n\n return new Frustum(frustumCoords, frustumPlanes);\n }\n}\n\nclass Aabb {\n min: vec3;\n max: vec3;\n center: vec3;\n\n constructor(min_: vec3, max_: vec3) {\n this.min = min_;\n this.max = max_;\n this.center = vec3.scale([] as any, vec3.add([] as any, this.min, this.max), 0.5);\n }\n\n quadrant(index: number): Aabb {\n const split = [(index % 2) === 0, index < 2];\n const qMin = vec3.clone(this.min);\n const qMax = vec3.clone(this.max);\n for (let axis = 0; axis < split.length; axis++) {\n qMin[axis] = split[axis] ? this.min[axis] : this.center[axis];\n qMax[axis] = split[axis] ? this.center[axis] : this.max[axis];\n }\n // Elevation is always constant, hence quadrant.max.z = this.max.z\n qMax[2] = this.max[2];\n return new Aabb(qMin, qMax);\n }\n\n distanceX(point: Array): number {\n const pointOnAabb = Math.max(Math.min(this.max[0], point[0]), this.min[0]);\n return pointOnAabb - point[0];\n }\n\n distanceY(point: Array): number {\n const pointOnAabb = Math.max(Math.min(this.max[1], point[1]), this.min[1]);\n return pointOnAabb - point[1];\n }\n\n // Performs a frustum-aabb intersection test. Returns 0 if there's no intersection,\n // 1 if shapes are intersecting and 2 if the aabb if fully inside the frustum.\n intersects(frustum: Frustum): number {\n // Execute separating axis test between two convex objects to find intersections\n // Each frustum plane together with 3 major axes define the separating axes\n\n const aabbPoints = [\n [this.min[0], this.min[1], this.min[2], 1],\n [this.max[0], this.min[1], this.min[2], 1],\n [this.max[0], this.max[1], this.min[2], 1],\n [this.min[0], this.max[1], this.min[2], 1],\n [this.min[0], this.min[1], this.max[2], 1],\n [this.max[0], this.min[1], this.max[2], 1],\n [this.max[0], this.max[1], this.max[2], 1],\n [this.min[0], this.max[1], this.max[2], 1]\n ];\n\n let fullyInside = true;\n\n for (let p = 0; p < frustum.planes.length; p++) {\n const plane = frustum.planes[p];\n let pointsInside = 0;\n\n for (let i = 0; i < aabbPoints.length; i++) {\n if (vec4.dot(plane, aabbPoints[i] as any) >= 0) {\n pointsInside++;\n }\n }\n\n if (pointsInside === 0)\n return 0;\n\n if (pointsInside !== aabbPoints.length)\n fullyInside = false;\n }\n\n if (fullyInside)\n return 2;\n\n for (let axis = 0; axis < 3; axis++) {\n let projMin = Number.MAX_VALUE;\n let projMax = -Number.MAX_VALUE;\n\n for (let p = 0; p < frustum.points.length; p++) {\n const projectedPoint = frustum.points[p][axis] - this.min[axis];\n\n projMin = Math.min(projMin, projectedPoint);\n projMax = Math.max(projMax, projectedPoint);\n }\n\n if (projMax < 0 || projMin > this.max[axis] - this.min[axis])\n return 0;\n }\n\n return 1;\n }\n}\nexport {\n Aabb,\n Frustum\n};\n","import {interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport Point from '@mapbox/point-geometry';\nimport {clamp} from '../util/util';\n\n/**\n * An `EdgeInset` object represents screen space padding applied to the edges of the viewport.\n * This shifts the apprent center or the vanishing point of the map. This is useful for adding floating UI elements\n * on top of the map and having the vanishing point shift as UI elements resize.\n *\n * @group Geography and Geometry\n */\nexport class EdgeInsets {\n /**\n * @defaultValue 0\n */\n top: number;\n /**\n * @defaultValue 0\n */\n bottom: number;\n /**\n * @defaultValue 0\n */\n left: number;\n /**\n * @defaultValue 0\n */\n right: number;\n\n constructor(top: number = 0, bottom: number = 0, left: number = 0, right: number = 0) {\n if (isNaN(top) || top < 0 ||\n isNaN(bottom) || bottom < 0 ||\n isNaN(left) || left < 0 ||\n isNaN(right) || right < 0\n ) {\n throw new Error('Invalid value for edge-insets, top, bottom, left and right must all be numbers');\n }\n\n this.top = top;\n this.bottom = bottom;\n this.left = left;\n this.right = right;\n }\n\n /**\n * Interpolates the inset in-place.\n * This maintains the current inset value for any inset not present in `target`.\n * @param start - interpolation start\n * @param target - interpolation target\n * @param t - interpolation step/weight\n * @returns the insets\n */\n interpolate(start: PaddingOptions | EdgeInsets, target: PaddingOptions, t: number): EdgeInsets {\n if (target.top != null && start.top != null) this.top = interpolates.number(start.top, target.top, t);\n if (target.bottom != null && start.bottom != null) this.bottom = interpolates.number(start.bottom, target.bottom, t);\n if (target.left != null && start.left != null) this.left = interpolates.number(start.left, target.left, t);\n if (target.right != null && start.right != null) this.right = interpolates.number(start.right, target.right, t);\n\n return this;\n }\n\n /**\n * Utility method that computes the new apprent center or vanishing point after applying insets.\n * This is in pixels and with the top left being (0.0) and +y being downwards.\n *\n * @param width - the width\n * @param height - the height\n * @returns the point\n */\n getCenter(width: number, height: number): Point {\n // Clamp insets so they never overflow width/height and always calculate a valid center\n const x = clamp((this.left + width - this.right) / 2, 0, width);\n const y = clamp((this.top + height - this.bottom) / 2, 0, height);\n\n return new Point(x, y);\n }\n\n equals(other: PaddingOptions): boolean {\n return this.top === other.top &&\n this.bottom === other.bottom &&\n this.left === other.left &&\n this.right === other.right;\n }\n\n clone(): EdgeInsets {\n return new EdgeInsets(this.top, this.bottom, this.left, this.right);\n }\n\n /**\n * Returns the current state as json, useful when you want to have a\n * read-only representation of the inset.\n *\n * @returns state as json\n */\n toJSON(): PaddingOptions {\n return {\n top: this.top,\n bottom: this.bottom,\n left: this.left,\n right: this.right\n };\n }\n}\n\n/**\n * Options for setting padding on calls to methods such as {@link Map#fitBounds}, {@link Map#fitScreenCoordinates}, and {@link Map#setPadding}. Adjust these options to set the amount of padding in pixels added to the edges of the canvas. Set a uniform padding on all edges or individual values for each edge. All properties of this object must be\n * non-negative integers.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let bbox = [[-79, 43], [-73, 45]];\n * map.fitBounds(bbox, {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n *\n * @example\n * ```ts\n * let bbox = [[-79, 43], [-73, 45]];\n * map.fitBounds(bbox, {\n * padding: 20\n * });\n * ```\n * @see [Fit to the bounds of a LineString](https://maplibre.org/maplibre-gl-js/docs/examples/zoomto-linestring/)\n * @see [Fit a map to a bounding box](https://maplibre.org/maplibre-gl-js/docs/examples/fitbounds/)\n */\nexport type PaddingOptions = {\n /**\n * Padding in pixels from the top of the map canvas.\n */\n top: number;\n /**\n * Padding in pixels from the bottom of the map canvas.\n */\n bottom: number;\n /**\n * Padding in pixels from the left of the map canvas.\n */\n right: number;\n /**\n * Padding in pixels from the right of the map canvas.\n */\n left: number;\n};\n","import {LngLat} from './lng_lat';\nimport {LngLatBounds} from './lng_lat_bounds';\nimport {MercatorCoordinate, mercatorXfromLng, mercatorYfromLat, mercatorZfromAltitude} from './mercator_coordinate';\nimport Point from '@mapbox/point-geometry';\nimport {wrap, clamp} from '../util/util';\nimport {interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {EXTENT} from '../data/extent';\nimport {vec3, vec4, mat4, mat2, vec2} from 'gl-matrix';\nimport {Aabb, Frustum} from '../util/primitives';\nimport {EdgeInsets} from './edge_insets';\n\nimport {UnwrappedTileID, OverscaledTileID, CanonicalTileID} from '../source/tile_id';\nimport type {PaddingOptions} from './edge_insets';\nimport {Terrain} from '../render/terrain';\n\n/**\n * @internal\n * A single transform, generally used for a single tile to be\n * scaled, rotated, and zoomed.\n */\nexport class Transform {\n tileSize: number;\n tileZoom: number;\n lngRange: [number, number];\n latRange: [number, number];\n maxValidLatitude: number;\n scale: number;\n width: number;\n height: number;\n angle: number;\n rotationMatrix: mat2;\n pixelsToGLUnits: [number, number];\n cameraToCenterDistance: number;\n mercatorMatrix: mat4;\n projMatrix: mat4;\n invProjMatrix: mat4;\n alignedProjMatrix: mat4;\n pixelMatrix: mat4;\n pixelMatrix3D: mat4;\n pixelMatrixInverse: mat4;\n glCoordMatrix: mat4;\n labelPlaneMatrix: mat4;\n _fov: number;\n _pitch: number;\n _zoom: number;\n _unmodified: boolean;\n _renderWorldCopies: boolean;\n _minZoom: number;\n _maxZoom: number;\n _minPitch: number;\n _maxPitch: number;\n _center: LngLat;\n _elevation: number;\n _pixelPerMeter: number;\n _edgeInsets: EdgeInsets;\n _constraining: boolean;\n _posMatrixCache: {[_: string]: mat4};\n _alignedPosMatrixCache: {[_: string]: mat4};\n _minEleveationForCurrentTile: number;\n\n constructor(minZoom?: number, maxZoom?: number, minPitch?: number, maxPitch?: number, renderWorldCopies?: boolean) {\n this.tileSize = 512; // constant\n this.maxValidLatitude = 85.051129; // constant\n\n this._renderWorldCopies = renderWorldCopies === undefined ? true : !!renderWorldCopies;\n this._minZoom = minZoom || 0;\n this._maxZoom = maxZoom || 22;\n\n this._minPitch = (minPitch === undefined || minPitch === null) ? 0 : minPitch;\n this._maxPitch = (maxPitch === undefined || maxPitch === null) ? 60 : maxPitch;\n\n this.setMaxBounds();\n\n this.width = 0;\n this.height = 0;\n this._center = new LngLat(0, 0);\n this._elevation = 0;\n this.zoom = 0;\n this.angle = 0;\n this._fov = 0.6435011087932844;\n this._pitch = 0;\n this._unmodified = true;\n this._edgeInsets = new EdgeInsets();\n this._posMatrixCache = {};\n this._alignedPosMatrixCache = {};\n this._minEleveationForCurrentTile = 0;\n }\n\n clone(): Transform {\n const clone = new Transform(this._minZoom, this._maxZoom, this._minPitch, this.maxPitch, this._renderWorldCopies);\n clone.apply(this);\n return clone;\n }\n\n apply(that: Transform) {\n this.tileSize = that.tileSize;\n this.latRange = that.latRange;\n this.width = that.width;\n this.height = that.height;\n this._center = that._center;\n this._elevation = that._elevation;\n this._minEleveationForCurrentTile = that._minEleveationForCurrentTile;\n this.zoom = that.zoom;\n this.angle = that.angle;\n this._fov = that._fov;\n this._pitch = that._pitch;\n this._unmodified = that._unmodified;\n this._edgeInsets = that._edgeInsets.clone();\n this._calcMatrices();\n }\n\n get minZoom(): number { return this._minZoom; }\n set minZoom(zoom: number) {\n if (this._minZoom === zoom) return;\n this._minZoom = zoom;\n this.zoom = Math.max(this.zoom, zoom);\n }\n\n get maxZoom(): number { return this._maxZoom; }\n set maxZoom(zoom: number) {\n if (this._maxZoom === zoom) return;\n this._maxZoom = zoom;\n this.zoom = Math.min(this.zoom, zoom);\n }\n\n get minPitch(): number { return this._minPitch; }\n set minPitch(pitch: number) {\n if (this._minPitch === pitch) return;\n this._minPitch = pitch;\n this.pitch = Math.max(this.pitch, pitch);\n }\n\n get maxPitch(): number { return this._maxPitch; }\n set maxPitch(pitch: number) {\n if (this._maxPitch === pitch) return;\n this._maxPitch = pitch;\n this.pitch = Math.min(this.pitch, pitch);\n }\n\n get renderWorldCopies(): boolean { return this._renderWorldCopies; }\n set renderWorldCopies(renderWorldCopies: boolean) {\n if (renderWorldCopies === undefined) {\n renderWorldCopies = true;\n } else if (renderWorldCopies === null) {\n renderWorldCopies = false;\n }\n\n this._renderWorldCopies = renderWorldCopies;\n }\n\n get worldSize(): number {\n return this.tileSize * this.scale;\n }\n\n get centerOffset(): Point {\n return this.centerPoint._sub(this.size._div(2));\n }\n\n get size(): Point {\n return new Point(this.width, this.height);\n }\n\n get bearing(): number {\n return -this.angle / Math.PI * 180;\n }\n set bearing(bearing: number) {\n const b = -wrap(bearing, -180, 180) * Math.PI / 180;\n if (this.angle === b) return;\n this._unmodified = false;\n this.angle = b;\n this._calcMatrices();\n\n // 2x2 matrix for rotating points\n this.rotationMatrix = mat2.create();\n mat2.rotate(this.rotationMatrix, this.rotationMatrix, this.angle);\n }\n\n get pitch(): number {\n return this._pitch / Math.PI * 180;\n }\n set pitch(pitch: number) {\n const p = clamp(pitch, this.minPitch, this.maxPitch) / 180 * Math.PI;\n if (this._pitch === p) return;\n this._unmodified = false;\n this._pitch = p;\n this._calcMatrices();\n }\n\n get fov(): number {\n return this._fov / Math.PI * 180;\n }\n set fov(fov: number) {\n fov = Math.max(0.01, Math.min(60, fov));\n if (this._fov === fov) return;\n this._unmodified = false;\n this._fov = fov / 180 * Math.PI;\n this._calcMatrices();\n }\n\n get zoom(): number { return this._zoom; }\n set zoom(zoom: number) {\n const constrainedZoom = Math.min(Math.max(zoom, this.minZoom), this.maxZoom);\n if (this._zoom === constrainedZoom) return;\n this._unmodified = false;\n this._zoom = constrainedZoom;\n this.tileZoom = Math.max(0, Math.floor(constrainedZoom));\n this.scale = this.zoomScale(constrainedZoom);\n this._constrain();\n this._calcMatrices();\n }\n\n get center(): LngLat { return this._center; }\n set center(center: LngLat) {\n if (center.lat === this._center.lat && center.lng === this._center.lng) return;\n this._unmodified = false;\n this._center = center;\n this._constrain();\n this._calcMatrices();\n }\n\n get elevation(): number { return this._elevation; }\n set elevation(elevation: number) {\n if (elevation === this._elevation) return;\n this._elevation = elevation;\n this._constrain();\n this._calcMatrices();\n }\n\n get padding(): PaddingOptions { return this._edgeInsets.toJSON(); }\n set padding(padding: PaddingOptions) {\n if (this._edgeInsets.equals(padding)) return;\n this._unmodified = false;\n //Update edge-insets inplace\n this._edgeInsets.interpolate(this._edgeInsets, padding, 1);\n this._calcMatrices();\n }\n\n /**\n * The center of the screen in pixels with the top-left corner being (0,0)\n * and +y axis pointing downwards. This accounts for padding.\n */\n get centerPoint(): Point {\n return this._edgeInsets.getCenter(this.width, this.height);\n }\n\n /**\n * Returns if the padding params match\n *\n * @param padding - the padding to check against\n * @returns true if they are equal, false otherwise\n */\n isPaddingEqual(padding: PaddingOptions): boolean {\n return this._edgeInsets.equals(padding);\n }\n\n /**\n * Helper method to update edge-insets in place\n *\n * @param start - the starting padding\n * @param target - the target padding\n * @param t - the step/weight\n */\n interpolatePadding(start: PaddingOptions, target: PaddingOptions, t: number) {\n this._unmodified = false;\n this._edgeInsets.interpolate(start, target, t);\n this._constrain();\n this._calcMatrices();\n }\n\n /**\n * Return a zoom level that will cover all tiles the transform\n * @param options - the options\n * @returns zoom level An integer zoom level at which all tiles will be visible.\n */\n coveringZoomLevel(options: {\n /**\n * Target zoom level. If true, the value will be rounded to the closest integer. Otherwise the value will be floored.\n */\n roundZoom?: boolean;\n /**\n * Tile size, expressed in screen pixels.\n */\n tileSize: number;\n }): number {\n const z = (options.roundZoom ? Math.round : Math.floor)(\n this.zoom + this.scaleZoom(this.tileSize / options.tileSize)\n );\n // At negative zoom levels load tiles from z0 because negative tile zoom levels don't exist.\n return Math.max(0, z);\n }\n\n /**\n * Return any \"wrapped\" copies of a given tile coordinate that are visible\n * in the current view.\n */\n getVisibleUnwrappedCoordinates(tileID: CanonicalTileID) {\n const result = [new UnwrappedTileID(0, tileID)];\n if (this._renderWorldCopies) {\n const utl = this.pointCoordinate(new Point(0, 0));\n const utr = this.pointCoordinate(new Point(this.width, 0));\n const ubl = this.pointCoordinate(new Point(this.width, this.height));\n const ubr = this.pointCoordinate(new Point(0, this.height));\n const w0 = Math.floor(Math.min(utl.x, utr.x, ubl.x, ubr.x));\n const w1 = Math.floor(Math.max(utl.x, utr.x, ubl.x, ubr.x));\n\n // Add an extra copy of the world on each side to properly render ImageSources and CanvasSources.\n // Both sources draw outside the tile boundaries of the tile that \"contains them\" so we need\n // to add extra copies on both sides in case offscreen tiles need to draw into on-screen ones.\n const extraWorldCopy = 1;\n\n for (let w = w0 - extraWorldCopy; w <= w1 + extraWorldCopy; w++) {\n if (w === 0) continue;\n result.push(new UnwrappedTileID(w, tileID));\n }\n }\n return result;\n }\n\n /**\n * Return all coordinates that could cover this transform for a covering\n * zoom level.\n * @param options - the options\n * @returns OverscaledTileIDs\n */\n coveringTiles(\n options: {\n tileSize: number;\n minzoom?: number;\n maxzoom?: number;\n roundZoom?: boolean;\n reparseOverscaled?: boolean;\n renderWorldCopies?: boolean;\n terrain?: Terrain;\n }\n ): Array {\n let z = this.coveringZoomLevel(options);\n const actualZ = z;\n\n if (options.minzoom !== undefined && z < options.minzoom) return [];\n if (options.maxzoom !== undefined && z > options.maxzoom) z = options.maxzoom;\n\n const cameraCoord = this.pointCoordinate(this.getCameraPoint());\n const centerCoord = MercatorCoordinate.fromLngLat(this.center);\n const numTiles = Math.pow(2, z);\n const cameraPoint = [numTiles * cameraCoord.x, numTiles * cameraCoord.y, 0];\n const centerPoint = [numTiles * centerCoord.x, numTiles * centerCoord.y, 0];\n const cameraFrustum = Frustum.fromInvProjectionMatrix(this.invProjMatrix, this.worldSize, z);\n\n // No change of LOD behavior for pitch lower than 60 and when there is no top padding: return only tile ids from the requested zoom level\n let minZoom = options.minzoom || 0;\n // Use 0.1 as an epsilon to avoid for explicit == 0.0 floating point checks\n if (!options.terrain && this.pitch <= 60.0 && this._edgeInsets.top < 0.1)\n minZoom = z;\n\n // There should always be a certain number of maximum zoom level tiles surrounding the center location in 2D or in front of the camera in 3D\n const radiusOfMaxLvlLodInTiles = options.terrain ? 2 / Math.min(this.tileSize, options.tileSize) * this.tileSize : 3;\n\n const newRootTile = (wrap: number): any => {\n return {\n aabb: new Aabb([wrap * numTiles, 0, 0], [(wrap + 1) * numTiles, numTiles, 0]),\n zoom: 0,\n x: 0,\n y: 0,\n wrap,\n fullyVisible: false\n };\n };\n\n // Do a depth-first traversal to find visible tiles and proper levels of detail\n const stack = [];\n const result = [];\n const maxZoom = z;\n const overscaledZ = options.reparseOverscaled ? actualZ : z;\n\n if (this._renderWorldCopies) {\n // Render copy of the globe thrice on both sides\n for (let i = 1; i <= 3; i++) {\n stack.push(newRootTile(-i));\n stack.push(newRootTile(i));\n }\n }\n\n stack.push(newRootTile(0));\n\n while (stack.length > 0) {\n const it = stack.pop();\n const x = it.x;\n const y = it.y;\n let fullyVisible = it.fullyVisible;\n\n // Visibility of a tile is not required if any of its ancestor if fully inside the frustum\n if (!fullyVisible) {\n const intersectResult = it.aabb.intersects(cameraFrustum);\n\n if (intersectResult === 0)\n continue;\n\n fullyVisible = intersectResult === 2;\n }\n\n const refPoint = options.terrain ? cameraPoint : centerPoint;\n const distanceX = it.aabb.distanceX(refPoint);\n const distanceY = it.aabb.distanceY(refPoint);\n const longestDim = Math.max(Math.abs(distanceX), Math.abs(distanceY));\n\n // We're using distance based heuristics to determine if a tile should be split into quadrants or not.\n // radiusOfMaxLvlLodInTiles defines that there's always a certain number of maxLevel tiles next to the map center.\n // Using the fact that a parent node in quadtree is twice the size of its children (per dimension)\n // we can define distance thresholds for each relative level:\n // f(k) = offset + 2 + 4 + 8 + 16 + ... + 2^k. This is the same as \"offset+2^(k+1)-2\"\n const distToSplit = radiusOfMaxLvlLodInTiles + (1 << (maxZoom - it.zoom)) - 2;\n\n // Have we reached the target depth or is the tile too far away to be any split further?\n if (it.zoom === maxZoom || (longestDim > distToSplit && it.zoom >= minZoom)) {\n const dz = maxZoom - it.zoom, dx = cameraPoint[0] - 0.5 - (x << dz), dy = cameraPoint[1] - 0.5 - (y << dz);\n result.push({\n tileID: new OverscaledTileID(it.zoom === maxZoom ? overscaledZ : it.zoom, it.wrap, it.zoom, x, y),\n distanceSq: vec2.sqrLen([centerPoint[0] - 0.5 - x, centerPoint[1] - 0.5 - y]),\n // this variable is currently not used, but may be important to reduce the amount of loaded tiles\n tileDistanceToCamera: Math.sqrt(dx * dx + dy * dy)\n });\n continue;\n }\n\n for (let i = 0; i < 4; i++) {\n const childX = (x << 1) + (i % 2);\n const childY = (y << 1) + (i >> 1);\n const childZ = it.zoom + 1;\n let quadrant = it.aabb.quadrant(i);\n if (options.terrain) {\n const tileID = new OverscaledTileID(childZ, it.wrap, childZ, childX, childY);\n const minMax = options.terrain.getMinMaxElevation(tileID);\n const minElevation = minMax.minElevation ?? this.elevation;\n const maxElevation = minMax.maxElevation ?? this.elevation;\n quadrant = new Aabb(\n [quadrant.min[0], quadrant.min[1], minElevation] as vec3,\n [quadrant.max[0], quadrant.max[1], maxElevation] as vec3\n );\n }\n stack.push({aabb: quadrant, zoom: childZ, x: childX, y: childY, wrap: it.wrap, fullyVisible});\n }\n }\n\n return result.sort((a, b) => a.distanceSq - b.distanceSq).map(a => a.tileID);\n }\n\n resize(width: number, height: number) {\n this.width = width;\n this.height = height;\n\n this.pixelsToGLUnits = [2 / width, -2 / height];\n this._constrain();\n this._calcMatrices();\n }\n\n get unmodified(): boolean { return this._unmodified; }\n\n zoomScale(zoom: number) { return Math.pow(2, zoom); }\n scaleZoom(scale: number) { return Math.log(scale) / Math.LN2; }\n\n project(lnglat: LngLat) {\n const lat = clamp(lnglat.lat, -this.maxValidLatitude, this.maxValidLatitude);\n return new Point(\n mercatorXfromLng(lnglat.lng) * this.worldSize,\n mercatorYfromLat(lat) * this.worldSize);\n }\n\n unproject(point: Point): LngLat {\n return new MercatorCoordinate(point.x / this.worldSize, point.y / this.worldSize).toLngLat();\n }\n\n get point(): Point { return this.project(this.center); }\n\n /**\n * get the camera position in LngLat and altitudes in meter\n * @returns An object with lngLat & altitude.\n */\n getCameraPosition(): {\n lngLat: LngLat;\n altitude: number;\n } {\n const lngLat = this.pointLocation(this.getCameraPoint());\n const altitude = Math.cos(this._pitch) * this.cameraToCenterDistance / this._pixelPerMeter;\n return {lngLat, altitude: altitude + this.elevation};\n }\n\n /**\n * This method works in combination with freezeElevation activated.\n * freezeElevtion is enabled during map-panning because during this the camera should sit in constant height.\n * After panning finished, call this method to recalculate the zoomlevel for the current camera-height in current terrain.\n * @param terrain - the terrain\n */\n recalculateZoom(terrain: Terrain) {\n // find position the camera is looking on\n const center = this.pointLocation(this.centerPoint, terrain);\n const elevation = terrain.getElevationForLngLatZoom(center, this.tileZoom);\n const deltaElevation = this.elevation - elevation;\n if (!deltaElevation) return;\n\n // calculate mercator distance between camera & target\n const cameraPosition = this.getCameraPosition();\n const camera = MercatorCoordinate.fromLngLat(cameraPosition.lngLat, cameraPosition.altitude);\n const target = MercatorCoordinate.fromLngLat(center, elevation);\n const dx = camera.x - target.x, dy = camera.y - target.y, dz = camera.z - target.z;\n const distance = Math.sqrt(dx * dx + dy * dy + dz * dz);\n\n // from this distance we calculate the new zoomlevel\n const zoom = this.scaleZoom(this.cameraToCenterDistance / distance / this.tileSize);\n\n // update matrices\n this._elevation = elevation;\n this._center = center;\n this.zoom = zoom;\n }\n\n setLocationAtPoint(lnglat: LngLat, point: Point) {\n const a = this.pointCoordinate(point);\n const b = this.pointCoordinate(this.centerPoint);\n const loc = this.locationCoordinate(lnglat);\n const newCenter = new MercatorCoordinate(\n loc.x - (a.x - b.x),\n loc.y - (a.y - b.y));\n this.center = this.coordinateLocation(newCenter);\n if (this._renderWorldCopies) {\n this.center = this.center.wrap();\n }\n }\n\n /**\n * Given a location, return the screen point that corresponds to it\n * @param lnglat - location\n * @param terrain - optional terrain\n * @returns screen point\n */\n locationPoint(lnglat: LngLat, terrain?: Terrain): Point {\n return terrain ?\n this.coordinatePoint(this.locationCoordinate(lnglat), terrain.getElevationForLngLatZoom(lnglat, this.tileZoom), this.pixelMatrix3D) :\n this.coordinatePoint(this.locationCoordinate(lnglat));\n }\n\n /**\n * Given a point on screen, return its lnglat\n * @param p - screen point\n * @param terrain - optional terrain\n * @returns lnglat location\n */\n pointLocation(p: Point, terrain?: Terrain): LngLat {\n return this.coordinateLocation(this.pointCoordinate(p, terrain));\n }\n\n /**\n * Given a geographical lnglat, return an unrounded\n * coordinate that represents it at this transform's zoom level.\n * @param lnglat - the location\n * @returns The mercator coordinate\n */\n locationCoordinate(lnglat: LngLat): MercatorCoordinate {\n return MercatorCoordinate.fromLngLat(lnglat);\n }\n\n /**\n * Given a Coordinate, return its geographical position.\n * @param coord - mercator coordivates\n * @returns lng and lat\n */\n coordinateLocation(coord: MercatorCoordinate): LngLat {\n return coord && coord.toLngLat();\n }\n\n /**\n * Given a Point, return its mercator coordinate.\n * @param p - the point\n * @param terrain - optional terrain\n * @returns lnglat\n */\n pointCoordinate(p: Point, terrain?: Terrain): MercatorCoordinate {\n // get point-coordinate from terrain coordinates framebuffer\n if (terrain) {\n const coordinate = terrain.pointCoordinate(p);\n if (coordinate != null) {\n return coordinate;\n }\n }\n\n // calculate point-coordinate on flat earth\n const targetZ = 0;\n // since we don't know the correct projected z value for the point,\n // unproject two points to get a line and then find the point on that\n // line with z=0\n\n const coord0 = [p.x, p.y, 0, 1] as any;\n const coord1 = [p.x, p.y, 1, 1] as any;\n\n vec4.transformMat4(coord0, coord0, this.pixelMatrixInverse);\n vec4.transformMat4(coord1, coord1, this.pixelMatrixInverse);\n\n const w0 = coord0[3];\n const w1 = coord1[3];\n const x0 = coord0[0] / w0;\n const x1 = coord1[0] / w1;\n const y0 = coord0[1] / w0;\n const y1 = coord1[1] / w1;\n const z0 = coord0[2] / w0;\n const z1 = coord1[2] / w1;\n\n const t = z0 === z1 ? 0 : (targetZ - z0) / (z1 - z0);\n\n return new MercatorCoordinate(\n interpolates.number(x0, x1, t) / this.worldSize,\n interpolates.number(y0, y1, t) / this.worldSize);\n }\n\n /**\n * Given a coordinate, return the screen point that corresponds to it\n * @param coord - the coordinates\n * @param elevation - the elevation\n * @param pixelMatrix - the pixel matrix\n * @returns screen point\n */\n coordinatePoint(coord: MercatorCoordinate, elevation: number = 0, pixelMatrix = this.pixelMatrix): Point {\n const p = [coord.x * this.worldSize, coord.y * this.worldSize, elevation, 1] as any;\n vec4.transformMat4(p, p, pixelMatrix);\n return new Point(p[0] / p[3], p[1] / p[3]);\n }\n\n /**\n * Returns the map's geographical bounds. When the bearing or pitch is non-zero, the visible region is not\n * an axis-aligned rectangle, and the result is the smallest bounds that encompasses the visible region.\n * @returns Returns a {@link LngLatBounds} object describing the map's geographical bounds.\n */\n getBounds(): LngLatBounds {\n const top = Math.max(0, this.height / 2 - this.getHorizon());\n return new LngLatBounds()\n .extend(this.pointLocation(new Point(0, top)))\n .extend(this.pointLocation(new Point(this.width, top)))\n .extend(this.pointLocation(new Point(this.width, this.height)))\n .extend(this.pointLocation(new Point(0, this.height)));\n }\n\n /**\n * Returns the maximum geographical bounds the map is constrained to, or `null` if none set.\n * @returns max bounds\n */\n getMaxBounds(): LngLatBounds | null {\n if (!this.latRange || this.latRange.length !== 2 ||\n !this.lngRange || this.lngRange.length !== 2) return null;\n\n return new LngLatBounds([this.lngRange[0], this.latRange[0]], [this.lngRange[1], this.latRange[1]]);\n }\n\n /**\n * Calculate pixel height of the visible horizon in relation to map-center (e.g. height/2),\n * multiplied by a static factor to simulate the earth-radius.\n * The calculated value is the horizontal line from the camera-height to sea-level.\n * @returns Horizon above center in pixels.\n */\n getHorizon(): number {\n return Math.tan(Math.PI / 2 - this._pitch) * this.cameraToCenterDistance * 0.85;\n }\n\n /**\n * Sets or clears the map's geographical constraints.\n * @param bounds - A {@link LngLatBounds} object describing the new geographic boundaries of the map.\n */\n setMaxBounds(bounds?: LngLatBounds | null) {\n if (bounds) {\n this.lngRange = [bounds.getWest(), bounds.getEast()];\n this.latRange = [bounds.getSouth(), bounds.getNorth()];\n this._constrain();\n } else {\n this.lngRange = null;\n this.latRange = [-this.maxValidLatitude, this.maxValidLatitude];\n }\n }\n\n /**\n * Calculate the posMatrix that, given a tile coordinate, would be used to display the tile on a map.\n * @param unwrappedTileID - the tile ID\n */\n calculatePosMatrix(unwrappedTileID: UnwrappedTileID, aligned: boolean = false): mat4 {\n const posMatrixKey = unwrappedTileID.key;\n const cache = aligned ? this._alignedPosMatrixCache : this._posMatrixCache;\n if (cache[posMatrixKey]) {\n return cache[posMatrixKey];\n }\n\n const canonical = unwrappedTileID.canonical;\n const scale = this.worldSize / this.zoomScale(canonical.z);\n const unwrappedX = canonical.x + Math.pow(2, canonical.z) * unwrappedTileID.wrap;\n\n const posMatrix = mat4.identity(new Float64Array(16) as any);\n mat4.translate(posMatrix, posMatrix, [unwrappedX * scale, canonical.y * scale, 0]);\n mat4.scale(posMatrix, posMatrix, [scale / EXTENT, scale / EXTENT, 1]);\n mat4.multiply(posMatrix, aligned ? this.alignedProjMatrix : this.projMatrix, posMatrix);\n\n cache[posMatrixKey] = new Float32Array(posMatrix);\n return cache[posMatrixKey];\n }\n\n customLayerMatrix(): mat4 {\n return this.mercatorMatrix.slice() as any;\n }\n\n _constrain() {\n if (!this.center || !this.width || !this.height || this._constraining) return;\n\n this._constraining = true;\n\n let minY = -90;\n let maxY = 90;\n let minX = -180;\n let maxX = 180;\n let sy, sx, x2, y2;\n const size = this.size,\n unmodified = this._unmodified;\n\n if (this.latRange) {\n const latRange = this.latRange;\n minY = mercatorYfromLat(latRange[1]) * this.worldSize;\n maxY = mercatorYfromLat(latRange[0]) * this.worldSize;\n sy = maxY - minY < size.y ? size.y / (maxY - minY) : 0;\n }\n\n if (this.lngRange) {\n const lngRange = this.lngRange;\n\n minX = wrap(\n mercatorXfromLng(lngRange[0]) * this.worldSize,\n 0,\n this.worldSize\n );\n maxX = wrap(\n mercatorXfromLng(lngRange[1]) * this.worldSize,\n 0,\n this.worldSize\n );\n\n if (maxX < minX) maxX += this.worldSize;\n\n sx = maxX - minX < size.x ? size.x / (maxX - minX) : 0;\n }\n\n const point = this.point;\n\n // how much the map should scale to fit the screen into given latitude/longitude ranges\n const s = Math.max(sx || 0, sy || 0);\n\n if (s) {\n this.center = this.unproject(new Point(\n sx ? (maxX + minX) / 2 : point.x,\n sy ? (maxY + minY) / 2 : point.y));\n this.zoom += this.scaleZoom(s);\n this._unmodified = unmodified;\n this._constraining = false;\n return;\n }\n\n if (this.latRange) {\n const y = point.y,\n h2 = size.y / 2;\n\n if (y - h2 < minY) y2 = minY + h2;\n if (y + h2 > maxY) y2 = maxY - h2;\n }\n\n if (this.lngRange) {\n const centerX = (minX + maxX) / 2;\n const x = wrap(point.x, centerX - this.worldSize / 2, centerX + this.worldSize / 2);\n const w2 = size.x / 2;\n\n if (x - w2 < minX) x2 = minX + w2;\n if (x + w2 > maxX) x2 = maxX - w2;\n }\n\n // pan the map if the screen goes off the range\n if (x2 !== undefined || y2 !== undefined) {\n this.center = this.unproject(new Point(\n x2 !== undefined ? x2 : point.x,\n y2 !== undefined ? y2 : point.y)).wrap();\n }\n\n this._unmodified = unmodified;\n this._constraining = false;\n }\n\n _calcMatrices() {\n if (!this.height) return;\n\n const halfFov = this._fov / 2;\n const offset = this.centerOffset;\n const x = this.point.x, y = this.point.y;\n this.cameraToCenterDistance = 0.5 / Math.tan(halfFov) * this.height;\n this._pixelPerMeter = mercatorZfromAltitude(1, this.center.lat) * this.worldSize;\n\n let m = mat4.identity(new Float64Array(16) as any);\n mat4.scale(m, m, [this.width / 2, -this.height / 2, 1]);\n mat4.translate(m, m, [1, -1, 0]);\n this.labelPlaneMatrix = m;\n\n m = mat4.identity(new Float64Array(16) as any);\n mat4.scale(m, m, [1, -1, 1]);\n mat4.translate(m, m, [-1, -1, 0]);\n mat4.scale(m, m, [2 / this.width, 2 / this.height, 1]);\n this.glCoordMatrix = m;\n\n // Calculate the camera to sea-level distance in pixel in respect of terrain\n const cameraToSeaLevelDistance = this.cameraToCenterDistance + this._elevation * this._pixelPerMeter / Math.cos(this._pitch);\n // In case of negative minimum elevation (e.g. the dead see, under the sea maps) use a lower plane for calculation\n const minElevation = Math.min(this.elevation, this._minEleveationForCurrentTile);\n const cameraToLowestPointDistance = cameraToSeaLevelDistance - minElevation * this._pixelPerMeter / Math.cos(this._pitch);\n const lowestPlane = minElevation < 0 ? cameraToLowestPointDistance : cameraToSeaLevelDistance;\n\n // Find the distance from the center point [width/2 + offset.x, height/2 + offset.y] to the\n // center top point [width/2 + offset.x, 0] in Z units, using the law of sines.\n // 1 Z unit is equivalent to 1 horizontal px at the center of the map\n // (the distance between[width/2, height/2] and [width/2 + 1, height/2])\n const groundAngle = Math.PI / 2 + this._pitch;\n const fovAboveCenter = this._fov * (0.5 + offset.y / this.height);\n const topHalfSurfaceDistance = Math.sin(fovAboveCenter) * lowestPlane / Math.sin(clamp(Math.PI - groundAngle - fovAboveCenter, 0.01, Math.PI - 0.01));\n\n // Find the distance from the center point to the horizon\n const horizon = this.getHorizon();\n const horizonAngle = Math.atan(horizon / this.cameraToCenterDistance);\n const fovCenterToHorizon = 2 * horizonAngle * (0.5 + offset.y / (horizon * 2));\n const topHalfSurfaceDistanceHorizon = Math.sin(fovCenterToHorizon) * lowestPlane / Math.sin(clamp(Math.PI - groundAngle - fovCenterToHorizon, 0.01, Math.PI - 0.01));\n\n // Calculate z distance of the farthest fragment that should be rendered.\n // Add a bit extra to avoid precision problems when a fragment's distance is exactly `furthestDistance`\n const topHalfMinDistance = Math.min(topHalfSurfaceDistance, topHalfSurfaceDistanceHorizon);\n const farZ = (Math.cos(Math.PI / 2 - this._pitch) * topHalfMinDistance + lowestPlane) * 1.01;\n\n // The larger the value of nearZ is\n // - the more depth precision is available for features (good)\n // - clipping starts appearing sooner when the camera is close to 3d features (bad)\n //\n // Smaller values worked well for mapbox-gl-js but deckgl was encountering precision issues\n // when rendering it's layers using custom layers. This value was experimentally chosen and\n // seems to solve z-fighting issues in deckgl while not clipping buildings too close to the camera.\n const nearZ = this.height / 50;\n\n // matrix for conversion from location to GL coordinates (-1 .. 1)\n m = new Float64Array(16) as any;\n mat4.perspective(m, this._fov, this.width / this.height, nearZ, farZ);\n\n // Apply center of perspective offset\n m[8] = -offset.x * 2 / this.width;\n m[9] = offset.y * 2 / this.height;\n\n mat4.scale(m, m, [1, -1, 1]);\n mat4.translate(m, m, [0, 0, -this.cameraToCenterDistance]);\n mat4.rotateX(m, m, this._pitch);\n mat4.rotateZ(m, m, this.angle);\n mat4.translate(m, m, [-x, -y, 0]);\n\n // The mercatorMatrix can be used to transform points from mercator coordinates\n // ([0, 0] nw, [1, 1] se) to GL coordinates.\n this.mercatorMatrix = mat4.scale([] as any, m, [this.worldSize, this.worldSize, this.worldSize]);\n\n // scale vertically to meters per pixel (inverse of ground resolution):\n mat4.scale(m, m, [1, 1, this._pixelPerMeter]);\n\n // matrix for conversion from location to screen coordinates in 2D\n this.pixelMatrix = mat4.multiply(new Float64Array(16) as any, this.labelPlaneMatrix, m);\n\n // matrix for conversion from location to GL coordinates (-1 .. 1)\n mat4.translate(m, m, [0, 0, -this.elevation]); // elevate camera over terrain\n this.projMatrix = m;\n this.invProjMatrix = mat4.invert([] as any, m);\n\n // matrix for conversion from location to screen coordinates in 2D\n this.pixelMatrix3D = mat4.multiply(new Float64Array(16) as any, this.labelPlaneMatrix, m);\n\n // Make a second projection matrix that is aligned to a pixel grid for rendering raster tiles.\n // We're rounding the (floating point) x/y values to achieve to avoid rendering raster images to fractional\n // coordinates. Additionally, we adjust by half a pixel in either direction in case that viewport dimension\n // is an odd integer to preserve rendering to the pixel grid. We're rotating this shift based on the angle\n // of the transformation so that 0°, 90°, 180°, and 270° rasters are crisp, and adjust the shift so that\n // it is always <= 0.5 pixels.\n const xShift = (this.width % 2) / 2, yShift = (this.height % 2) / 2,\n angleCos = Math.cos(this.angle), angleSin = Math.sin(this.angle),\n dx = x - Math.round(x) + angleCos * xShift + angleSin * yShift,\n dy = y - Math.round(y) + angleCos * yShift + angleSin * xShift;\n const alignedM = new Float64Array(m) as any as mat4;\n mat4.translate(alignedM, alignedM, [dx > 0.5 ? dx - 1 : dx, dy > 0.5 ? dy - 1 : dy, 0]);\n this.alignedProjMatrix = alignedM;\n\n // inverse matrix for conversion from screen coordinaes to location\n m = mat4.invert(new Float64Array(16) as any, this.pixelMatrix);\n if (!m) throw new Error('failed to invert matrix');\n this.pixelMatrixInverse = m;\n\n this._posMatrixCache = {};\n this._alignedPosMatrixCache = {};\n }\n\n maxPitchScaleFactor() {\n // calcMatrices hasn't run yet\n if (!this.pixelMatrixInverse) return 1;\n\n const coord = this.pointCoordinate(new Point(0, 0));\n const p = [coord.x * this.worldSize, coord.y * this.worldSize, 0, 1] as vec4;\n const topPoint = vec4.transformMat4(p, p, this.pixelMatrix);\n return topPoint[3] / this.cameraToCenterDistance;\n }\n\n /**\n * The camera looks at the map from a 3D (lng, lat, altitude) location. Let's use `cameraLocation`\n * as the name for the location under the camera and on the surface of the earth (lng, lat, 0).\n * `cameraPoint` is the projected position of the `cameraLocation`.\n *\n * This point is useful to us because only fill-extrusions that are between `cameraPoint` and\n * the query point on the surface of the earth can extend and intersect the query.\n *\n * When the map is not pitched the `cameraPoint` is equivalent to the center of the map because\n * the camera is right above the center of the map.\n */\n getCameraPoint() {\n const pitch = this._pitch;\n const yOffset = Math.tan(pitch) * (this.cameraToCenterDistance || 1);\n return this.centerPoint.add(new Point(0, yOffset));\n }\n\n /**\n * When the map is pitched, some of the 3D features that intersect a query will not intersect\n * the query at the surface of the earth. Instead the feature may be closer and only intersect\n * the query because it extrudes into the air.\n * @param queryGeometry - For point queries, the line from the query point to the \"camera point\",\n * for other geometries, the envelope of the query geometry and the \"camera point\"\n * @returns a geometry that includes all of the original query as well as all possible ares of the\n * screen where the *base* of a visible extrusion could be.\n *\n */\n getCameraQueryGeometry(queryGeometry: Array): Array {\n const c = this.getCameraPoint();\n\n if (queryGeometry.length === 1) {\n return [queryGeometry[0], c];\n } else {\n let minX = c.x;\n let minY = c.y;\n let maxX = c.x;\n let maxY = c.y;\n for (const p of queryGeometry) {\n minX = Math.min(minX, p.x);\n minY = Math.min(minY, p.y);\n maxX = Math.max(maxX, p.x);\n maxY = Math.max(maxY, p.y);\n }\n return [\n new Point(minX, minY),\n new Point(maxX, minY),\n new Point(maxX, maxY),\n new Point(minX, maxY),\n new Point(minX, minY)\n ];\n }\n }\n}\n","import * as glMatrix from \"./common.js\";\n/**\n * 2x2 Matrix\n * @module mat2\n */\n\n/**\n * Creates a new identity mat2\n *\n * @returns {mat2} a new 2x2 matrix\n */\n\nexport function create() {\n var out = new glMatrix.ARRAY_TYPE(4);\n\n if (glMatrix.ARRAY_TYPE != Float32Array) {\n out[1] = 0;\n out[2] = 0;\n }\n\n out[0] = 1;\n out[3] = 1;\n return out;\n}\n/**\n * Creates a new mat2 initialized with values from an existing matrix\n *\n * @param {ReadonlyMat2} a matrix to clone\n * @returns {mat2} a new 2x2 matrix\n */\n\nexport function clone(a) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Copy the values from one mat2 to another\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the source matrix\n * @returns {mat2} out\n */\n\nexport function copy(out, a) {\n out[0] = a[0];\n out[1] = a[1];\n out[2] = a[2];\n out[3] = a[3];\n return out;\n}\n/**\n * Set a mat2 to the identity matrix\n *\n * @param {mat2} out the receiving matrix\n * @returns {mat2} out\n */\n\nexport function identity(out) {\n out[0] = 1;\n out[1] = 0;\n out[2] = 0;\n out[3] = 1;\n return out;\n}\n/**\n * Create a new mat2 with the given values\n *\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m10 Component in column 1, row 0 position (index 2)\n * @param {Number} m11 Component in column 1, row 1 position (index 3)\n * @returns {mat2} out A new 2x2 matrix\n */\n\nexport function fromValues(m00, m01, m10, m11) {\n var out = new glMatrix.ARRAY_TYPE(4);\n out[0] = m00;\n out[1] = m01;\n out[2] = m10;\n out[3] = m11;\n return out;\n}\n/**\n * Set the components of a mat2 to the given values\n *\n * @param {mat2} out the receiving matrix\n * @param {Number} m00 Component in column 0, row 0 position (index 0)\n * @param {Number} m01 Component in column 0, row 1 position (index 1)\n * @param {Number} m10 Component in column 1, row 0 position (index 2)\n * @param {Number} m11 Component in column 1, row 1 position (index 3)\n * @returns {mat2} out\n */\n\nexport function set(out, m00, m01, m10, m11) {\n out[0] = m00;\n out[1] = m01;\n out[2] = m10;\n out[3] = m11;\n return out;\n}\n/**\n * Transpose the values of a mat2\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the source matrix\n * @returns {mat2} out\n */\n\nexport function transpose(out, a) {\n // If we are transposing ourselves we can skip a few steps but have to cache\n // some values\n if (out === a) {\n var a1 = a[1];\n out[1] = a[2];\n out[2] = a1;\n } else {\n out[0] = a[0];\n out[1] = a[2];\n out[2] = a[1];\n out[3] = a[3];\n }\n\n return out;\n}\n/**\n * Inverts a mat2\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the source matrix\n * @returns {mat2} out\n */\n\nexport function invert(out, a) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3]; // Calculate the determinant\n\n var det = a0 * a3 - a2 * a1;\n\n if (!det) {\n return null;\n }\n\n det = 1.0 / det;\n out[0] = a3 * det;\n out[1] = -a1 * det;\n out[2] = -a2 * det;\n out[3] = a0 * det;\n return out;\n}\n/**\n * Calculates the adjugate of a mat2\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the source matrix\n * @returns {mat2} out\n */\n\nexport function adjoint(out, a) {\n // Caching this value is nessecary if out == a\n var a0 = a[0];\n out[0] = a[3];\n out[1] = -a[1];\n out[2] = -a[2];\n out[3] = a0;\n return out;\n}\n/**\n * Calculates the determinant of a mat2\n *\n * @param {ReadonlyMat2} a the source matrix\n * @returns {Number} determinant of a\n */\n\nexport function determinant(a) {\n return a[0] * a[3] - a[2] * a[1];\n}\n/**\n * Multiplies two mat2's\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the first operand\n * @param {ReadonlyMat2} b the second operand\n * @returns {mat2} out\n */\n\nexport function multiply(out, a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n out[0] = a0 * b0 + a2 * b1;\n out[1] = a1 * b0 + a3 * b1;\n out[2] = a0 * b2 + a2 * b3;\n out[3] = a1 * b2 + a3 * b3;\n return out;\n}\n/**\n * Rotates a mat2 by the given angle\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the matrix to rotate\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat2} out\n */\n\nexport function rotate(out, a, rad) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n out[0] = a0 * c + a2 * s;\n out[1] = a1 * c + a3 * s;\n out[2] = a0 * -s + a2 * c;\n out[3] = a1 * -s + a3 * c;\n return out;\n}\n/**\n * Scales the mat2 by the dimensions in the given vec2\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the matrix to rotate\n * @param {ReadonlyVec2} v the vec2 to scale the matrix by\n * @returns {mat2} out\n **/\n\nexport function scale(out, a, v) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var v0 = v[0],\n v1 = v[1];\n out[0] = a0 * v0;\n out[1] = a1 * v0;\n out[2] = a2 * v1;\n out[3] = a3 * v1;\n return out;\n}\n/**\n * Creates a matrix from a given angle\n * This is equivalent to (but much faster than):\n *\n * mat2.identity(dest);\n * mat2.rotate(dest, dest, rad);\n *\n * @param {mat2} out mat2 receiving operation result\n * @param {Number} rad the angle to rotate the matrix by\n * @returns {mat2} out\n */\n\nexport function fromRotation(out, rad) {\n var s = Math.sin(rad);\n var c = Math.cos(rad);\n out[0] = c;\n out[1] = s;\n out[2] = -s;\n out[3] = c;\n return out;\n}\n/**\n * Creates a matrix from a vector scaling\n * This is equivalent to (but much faster than):\n *\n * mat2.identity(dest);\n * mat2.scale(dest, dest, vec);\n *\n * @param {mat2} out mat2 receiving operation result\n * @param {ReadonlyVec2} v Scaling vector\n * @returns {mat2} out\n */\n\nexport function fromScaling(out, v) {\n out[0] = v[0];\n out[1] = 0;\n out[2] = 0;\n out[3] = v[1];\n return out;\n}\n/**\n * Returns a string representation of a mat2\n *\n * @param {ReadonlyMat2} a matrix to represent as a string\n * @returns {String} string representation of the matrix\n */\n\nexport function str(a) {\n return \"mat2(\" + a[0] + \", \" + a[1] + \", \" + a[2] + \", \" + a[3] + \")\";\n}\n/**\n * Returns Frobenius norm of a mat2\n *\n * @param {ReadonlyMat2} a the matrix to calculate Frobenius norm of\n * @returns {Number} Frobenius norm\n */\n\nexport function frob(a) {\n return Math.hypot(a[0], a[1], a[2], a[3]);\n}\n/**\n * Returns L, D and U matrices (Lower triangular, Diagonal and Upper triangular) by factorizing the input matrix\n * @param {ReadonlyMat2} L the lower triangular matrix\n * @param {ReadonlyMat2} D the diagonal matrix\n * @param {ReadonlyMat2} U the upper triangular matrix\n * @param {ReadonlyMat2} a the input matrix to factorize\n */\n\nexport function LDU(L, D, U, a) {\n L[2] = a[2] / a[0];\n U[0] = a[0];\n U[1] = a[1];\n U[3] = a[3] - L[2] * U[1];\n return [L, D, U];\n}\n/**\n * Adds two mat2's\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the first operand\n * @param {ReadonlyMat2} b the second operand\n * @returns {mat2} out\n */\n\nexport function add(out, a, b) {\n out[0] = a[0] + b[0];\n out[1] = a[1] + b[1];\n out[2] = a[2] + b[2];\n out[3] = a[3] + b[3];\n return out;\n}\n/**\n * Subtracts matrix b from matrix a\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the first operand\n * @param {ReadonlyMat2} b the second operand\n * @returns {mat2} out\n */\n\nexport function subtract(out, a, b) {\n out[0] = a[0] - b[0];\n out[1] = a[1] - b[1];\n out[2] = a[2] - b[2];\n out[3] = a[3] - b[3];\n return out;\n}\n/**\n * Returns whether or not the matrices have exactly the same elements in the same position (when compared with ===)\n *\n * @param {ReadonlyMat2} a The first matrix.\n * @param {ReadonlyMat2} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function exactEquals(a, b) {\n return a[0] === b[0] && a[1] === b[1] && a[2] === b[2] && a[3] === b[3];\n}\n/**\n * Returns whether or not the matrices have approximately the same elements in the same position.\n *\n * @param {ReadonlyMat2} a The first matrix.\n * @param {ReadonlyMat2} b The second matrix.\n * @returns {Boolean} True if the matrices are equal, false otherwise.\n */\n\nexport function equals(a, b) {\n var a0 = a[0],\n a1 = a[1],\n a2 = a[2],\n a3 = a[3];\n var b0 = b[0],\n b1 = b[1],\n b2 = b[2],\n b3 = b[3];\n return Math.abs(a0 - b0) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a0), Math.abs(b0)) && Math.abs(a1 - b1) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a1), Math.abs(b1)) && Math.abs(a2 - b2) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a2), Math.abs(b2)) && Math.abs(a3 - b3) <= glMatrix.EPSILON * Math.max(1.0, Math.abs(a3), Math.abs(b3));\n}\n/**\n * Multiply each element of the matrix by a scalar.\n *\n * @param {mat2} out the receiving matrix\n * @param {ReadonlyMat2} a the matrix to scale\n * @param {Number} b amount to scale the matrix's elements by\n * @returns {mat2} out\n */\n\nexport function multiplyScalar(out, a, b) {\n out[0] = a[0] * b;\n out[1] = a[1] * b;\n out[2] = a[2] * b;\n out[3] = a[3] * b;\n return out;\n}\n/**\n * Adds two mat2's after multiplying each element of the second operand by a scalar value.\n *\n * @param {mat2} out the receiving vector\n * @param {ReadonlyMat2} a the first operand\n * @param {ReadonlyMat2} b the second operand\n * @param {Number} scale the amount to scale b's elements by before adding\n * @returns {mat2} out\n */\n\nexport function multiplyScalarAndAdd(out, a, b, scale) {\n out[0] = a[0] + b[0] * scale;\n out[1] = a[1] + b[1] * scale;\n out[2] = a[2] + b[2] * scale;\n out[3] = a[3] + b[3] * scale;\n return out;\n}\n/**\n * Alias for {@link mat2.multiply}\n * @function\n */\n\nexport var mul = multiply;\n/**\n * Alias for {@link mat2.subtract}\n * @function\n */\n\nexport var sub = subtract;","/**\n * Throttle the given function to run at most every `period` milliseconds.\n */\nexport function throttle void>(fn: T, time: number): (...args: Parameters) => ReturnType {\n let pending = false;\n let timerId: ReturnType = null;\n let lastCallContext = null;\n let lastCallArgs: Parameters;\n\n const later = () => {\n timerId = null;\n if (pending) {\n fn.apply(lastCallContext, lastCallArgs);\n timerId = setTimeout(later, time);\n pending = false;\n }\n };\n\n return (...args: Parameters) => {\n pending = true;\n lastCallContext = this;\n lastCallArgs = args;\n if (!timerId) {\n later();\n }\n return timerId;\n };\n}\n","import {throttle} from '../util/throttle';\n\nimport type {Map} from './map';\n\n/**\n * Adds the map's position to its page's location hash.\n * Passed as an option to the map object.\n *\n * @group Markers and Controls\n */\nexport class Hash {\n _map: Map;\n _hashName: string;\n\n constructor(hashName?: string | null) {\n this._hashName = hashName && encodeURIComponent(hashName);\n }\n\n /**\n * Map element to listen for coordinate changes\n *\n * @param map - The map object\n * @returns `this`\n */\n addTo(map: Map) {\n this._map = map;\n addEventListener('hashchange', this._onHashChange, false);\n this._map.on('moveend', this._updateHash);\n return this;\n }\n\n /**\n * Removes hash\n *\n * @returns `this`\n */\n remove() {\n removeEventListener('hashchange', this._onHashChange, false);\n this._map.off('moveend', this._updateHash);\n clearTimeout(this._updateHash());\n\n delete this._map;\n return this;\n }\n\n getHashString(mapFeedback?: boolean) {\n const center = this._map.getCenter(),\n zoom = Math.round(this._map.getZoom() * 100) / 100,\n // derived from equation: 512px * 2^z / 360 / 10^d < 0.5px\n precision = Math.ceil((zoom * Math.LN2 + Math.log(512 / 360 / 0.5)) / Math.LN10),\n m = Math.pow(10, precision),\n lng = Math.round(center.lng * m) / m,\n lat = Math.round(center.lat * m) / m,\n bearing = this._map.getBearing(),\n pitch = this._map.getPitch();\n let hash = '';\n if (mapFeedback) {\n // new map feedback site has some constraints that don't allow\n // us to use the same hash format as we do for the Map hash option.\n hash += `/${lng}/${lat}/${zoom}`;\n } else {\n hash += `${zoom}/${lat}/${lng}`;\n }\n\n if (bearing || pitch) hash += (`/${Math.round(bearing * 10) / 10}`);\n if (pitch) hash += (`/${Math.round(pitch)}`);\n\n if (this._hashName) {\n const hashName = this._hashName;\n let found = false;\n const parts = window.location.hash.slice(1).split('&').map(part => {\n const key = part.split('=')[0];\n if (key === hashName) {\n found = true;\n return `${key}=${hash}`;\n }\n return part;\n }).filter(a => a);\n if (!found) {\n parts.push(`${hashName}=${hash}`);\n }\n return `#${parts.join('&')}`;\n }\n\n return `#${hash}`;\n }\n\n _getCurrentHash = () => {\n // Get the current hash from location, stripped from its number sign\n const hash = window.location.hash.replace('#', '');\n if (this._hashName) {\n // Split the parameter-styled hash into parts and find the value we need\n let keyval;\n hash.split('&').map(\n part => part.split('=')\n ).forEach(part => {\n if (part[0] === this._hashName) {\n keyval = part;\n }\n });\n return (keyval ? keyval[1] || '' : '').split('/');\n }\n return hash.split('/');\n };\n\n _onHashChange = () => {\n const loc = this._getCurrentHash();\n if (loc.length >= 3 && !loc.some(v => isNaN(v))) {\n const bearing = this._map.dragRotate.isEnabled() && this._map.touchZoomRotate.isEnabled() ? +(loc[3] || 0) : this._map.getBearing();\n this._map.jumpTo({\n center: [+loc[2], +loc[1]],\n zoom: +loc[0],\n bearing,\n pitch: +(loc[4] || 0)\n });\n return true;\n }\n return false;\n };\n\n _updateHashUnthrottled = () => {\n // Replace if already present, else append the updated hash string\n const location = window.location.href.replace(/(#.+)?$/, this.getHashString());\n try {\n window.history.replaceState(window.history.state, null, location);\n } catch (SecurityError) {\n // IE11 does not allow this if the page is within an iframe created\n // with iframe.contentWindow.document.write(...).\n // https://github.com/mapbox/mapbox-gl-js/issues/7410\n }\n };\n\n /**\n * Mobile Safari doesn't allow updating the hash more than 100 times per 30 seconds.\n */\n _updateHash: () => ReturnType = throttle(this._updateHashUnthrottled, 30 * 1000 / 100);\n}\n","import {browser} from '../util/browser';\nimport type {Map} from './map';\nimport {bezier, clamp, extend} from '../util/util';\nimport Point from '@mapbox/point-geometry';\nimport type {DragPanOptions} from './handler/shim/drag_pan';\n\nconst defaultInertiaOptions = {\n linearity: 0.3,\n easing: bezier(0, 0, 0.3, 1),\n};\n\nconst defaultPanInertiaOptions = extend({\n deceleration: 2500,\n maxSpeed: 1400\n}, defaultInertiaOptions);\n\nconst defaultZoomInertiaOptions = extend({\n deceleration: 20,\n maxSpeed: 1400\n}, defaultInertiaOptions);\n\nconst defaultBearingInertiaOptions = extend({\n deceleration: 1000,\n maxSpeed: 360\n}, defaultInertiaOptions);\n\nconst defaultPitchInertiaOptions = extend({\n deceleration: 1000,\n maxSpeed: 90\n}, defaultInertiaOptions);\n\nexport type InertiaOptions = {\n linearity: number;\n easing: (t: number) => number;\n deceleration: number;\n maxSpeed: number;\n};\n\nexport type InputEvent = MouseEvent | TouchEvent | KeyboardEvent | WheelEvent;\n\nexport class HandlerInertia {\n _map: Map;\n _inertiaBuffer: Array<{\n time: number;\n settings: any;\n }>;\n\n constructor(map: Map) {\n this._map = map;\n this.clear();\n }\n\n clear() {\n this._inertiaBuffer = [];\n }\n\n record(settings: any) {\n this._drainInertiaBuffer();\n this._inertiaBuffer.push({time: browser.now(), settings});\n }\n\n _drainInertiaBuffer() {\n const inertia = this._inertiaBuffer,\n now = browser.now(),\n cutoff = 160; //msec\n\n while (inertia.length > 0 && now - inertia[0].time > cutoff)\n inertia.shift();\n }\n\n _onMoveEnd(panInertiaOptions?: DragPanOptions | boolean) {\n this._drainInertiaBuffer();\n if (this._inertiaBuffer.length < 2) {\n return;\n }\n\n const deltas = {\n zoom: 0,\n bearing: 0,\n pitch: 0,\n pan: new Point(0, 0),\n pinchAround: undefined,\n around: undefined\n };\n\n for (const {settings} of this._inertiaBuffer) {\n deltas.zoom += settings.zoomDelta || 0;\n deltas.bearing += settings.bearingDelta || 0;\n deltas.pitch += settings.pitchDelta || 0;\n if (settings.panDelta) deltas.pan._add(settings.panDelta);\n if (settings.around) deltas.around = settings.around;\n if (settings.pinchAround) deltas.pinchAround = settings.pinchAround;\n }\n\n const lastEntry = this._inertiaBuffer[this._inertiaBuffer.length - 1];\n const duration = (lastEntry.time - this._inertiaBuffer[0].time);\n\n const easeOptions = {} as any;\n\n if (deltas.pan.mag()) {\n const result = calculateEasing(deltas.pan.mag(), duration, extend({}, defaultPanInertiaOptions, panInertiaOptions || {}));\n easeOptions.offset = deltas.pan.mult(result.amount / deltas.pan.mag());\n easeOptions.center = this._map.transform.center;\n extendDuration(easeOptions, result);\n }\n\n if (deltas.zoom) {\n const result = calculateEasing(deltas.zoom, duration, defaultZoomInertiaOptions);\n easeOptions.zoom = this._map.transform.zoom + result.amount;\n extendDuration(easeOptions, result);\n }\n\n if (deltas.bearing) {\n const result = calculateEasing(deltas.bearing, duration, defaultBearingInertiaOptions);\n easeOptions.bearing = this._map.transform.bearing + clamp(result.amount, -179, 179);\n extendDuration(easeOptions, result);\n }\n\n if (deltas.pitch) {\n const result = calculateEasing(deltas.pitch, duration, defaultPitchInertiaOptions);\n easeOptions.pitch = this._map.transform.pitch + result.amount;\n extendDuration(easeOptions, result);\n }\n\n if (easeOptions.zoom || easeOptions.bearing) {\n const last = deltas.pinchAround === undefined ? deltas.around : deltas.pinchAround;\n easeOptions.around = last ? this._map.unproject(last) : this._map.getCenter();\n }\n\n this.clear();\n return extend(easeOptions, {\n noMoveStart: true\n });\n\n }\n}\n\n// Unfortunately zoom, bearing, etc can't have different durations and easings so\n// we need to choose one. We use the longest duration and it's corresponding easing.\nfunction extendDuration(easeOptions, result) {\n if (!easeOptions.duration || easeOptions.duration < result.duration) {\n easeOptions.duration = result.duration;\n easeOptions.easing = result.easing;\n }\n}\n\nfunction calculateEasing(amount, inertiaDuration: number, inertiaOptions) {\n const {maxSpeed, linearity, deceleration} = inertiaOptions;\n const speed = clamp(\n amount * linearity / (inertiaDuration / 1000),\n -maxSpeed,\n maxSpeed);\n const duration = Math.abs(speed) / (deceleration * linearity);\n return {\n easing: inertiaOptions.easing,\n duration: duration * 1000,\n amount: speed * (duration / 2)\n };\n}\n","import {Event} from '../util/evented';\n\nimport {DOM} from '../util/dom';\nimport Point from '@mapbox/point-geometry';\nimport {extend} from '../util/util';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\n\nimport type {Map} from './map';\nimport type {LngLat} from '../geo/lng_lat';\nimport {SourceSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * An event from the mouse relevant to a specific layer.\n *\n * @group Event Related\n */\nexport type MapLayerMouseEvent = MapMouseEvent & { features?: MapGeoJSONFeature[] };\n\n/**\n * An event from a touch device relevat to a specific layer.\n *\n * @group Event Related\n */\nexport type MapLayerTouchEvent = MapTouchEvent & { features?: MapGeoJSONFeature[] };\n\n/**\n * The source event data type\n */\nexport type MapSourceDataType = 'content' | 'metadata' | 'visibility' | 'idle';\n\n/**\n * `MapLayerEventType` - a mapping between the event name and the event.\n * **Note:** These events are compatible with the optional `layerId` parameter.\n * If `layerId` is included as the second argument in {@link Map#on}, the event listener will fire only when the\n * event action contains a visible portion of the specified layer.\n * The following example can be used for all the events.\n *\n * @group Event Related\n * @example\n * ```ts\n * // Initialize the map\n * let map = new maplibregl.Map({ // map options });\n * // Set an event listener for a specific layer\n * map.on('the-event-name', 'poi-label', function(e) {\n * console.log('An event has occurred on a visible portion of the poi-label layer');\n * });\n * ```\n */\nexport type MapLayerEventType = {\n /**\n * Fired when a pointing device (usually a mouse) is pressed and released contains a visible portion of the specified layer.\n *\n * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/)\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n */\n click: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed and released twice contains a visible portion of the specified layer.\n *\n * **Note:** Under normal conditions, this event will be preceded by two `click` events.\n */\n dblclick: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed while inside a visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n mousedown: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is released while inside a visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n mouseup: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is moved while the cursor is inside a visible portion of the specified layer.\n * As you move the cursor across the layer, the event will fire every time the cursor changes position within that layer.\n *\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on over](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n */\n mousemove: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) enters a visible portion of a specified layer from\n * outside that layer or outside the map canvas.\n *\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n */\n mouseenter: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) leaves a visible portion of a specified layer, or leaves\n * the map canvas.\n *\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n */\n mouseleave: MapLayerMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is moved inside a visible portion of the specified layer.\n *\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n */\n mouseover: MapLayerMouseEvent;\n /**\n * Fired when a point device (usually a mouse) leaves the visible portion of the specified layer.\n */\n mouseout: MapLayerMouseEvent;\n /**\n * Fired when the right button of the mouse is clicked or the context menu key is pressed within visible portion of the specified layer.\n */\n contextmenu: MapLayerMouseEvent;\n /**\n * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchstart: MapLayerTouchEvent;\n /**\n * Fired when a [`touchend`](https://developer.mozilla.org/en-US/docs/Web/Events/touchend) event occurs within the visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchend: MapLayerTouchEvent;\n /**\n * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the visible portion of the specified layer.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchcancel: MapLayerTouchEvent;\n};\n\n/**\n * `MapEventType` - a mapping between the event name and the event value.\n * These events are used with the {@link Map#on} method.\n * When using a `layerId` with {@link Map#on} method, please refer to {@link MapLayerEventType}.\n * The following example can be used for all the events.\n *\n * @group Event Related\n * @example\n * ```ts\n * // Initialize the map\n * let map = new maplibregl.Map({ // map options });\n * // Set an event listener\n * map.on('the-event-name', () => {\n * console.log('An event has occurred!');\n * });\n * ```\n */\nexport type MapEventType = {\n /**\n * Fired when an error occurs. This is GL JS's primary error reporting\n * mechanism. We use an event instead of `throw` to better accommodate\n * asynchronous operations. If no listeners are bound to the `error` event, the\n * error will be printed to the console.\n */\n error: ErrorEvent;\n /**\n * @event `load` Fired immediately after all necessary resources have been downloaded\n * and the first visually complete rendering of the map has occurred.\n *\n * @see [Draw GeoJSON points](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-markers/)\n * @see [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/)\n * @see [Animate a point](https://maplibre.org/maplibre-gl-js/docs/examples/animate-point-along-line/)\n */\n load: MapLibreEvent;\n /**\n * Fired after the last frame rendered before the map enters an\n * \"idle\" state:\n *\n * - No camera transitions are in progress\n * - All currently requested tiles have loaded\n * - All fade/transition animations have completed\n */\n idle: MapLibreEvent;\n /**\n * Fired immediately after the map has been removed with {@link Map#remove}.\n */\n remove: MapLibreEvent;\n /**\n * Fired whenever the map is drawn to the screen, as the result of\n *\n * - a change to the map's position, zoom, pitch, or bearing\n * - a change to the map's style\n * - a change to a GeoJSON source\n * - the loading of a vector tile, GeoJSON file, glyph, or sprite\n */\n render: MapLibreEvent;\n /**\n * Fired immediately after the map has been resized.\n */\n resize: MapLibreEvent;\n /**\n * Fired when the WebGL context is lost.\n */\n webglcontextlost: MapContextEvent;\n /**\n * Fired when the WebGL context is restored.\n */\n webglcontextrestored: MapContextEvent;\n /**\n * Fired when any map data (style, source, tile, etc) begins loading or\n * changing asynchronously. All `dataloading` events are followed by a `data`,\n * `dataabort` or `error` event.\n */\n dataloading: MapDataEvent;\n /**\n * Fired when any map data loads or changes. See {@link MapDataEvent} for more information.\n * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/)\n */\n data: MapDataEvent;\n tiledataloading: MapDataEvent;\n /**\n * Fired when one of the map's sources begins loading or changing asynchronously.\n * All `sourcedataloading` events are followed by a `sourcedata`, `sourcedataabort` or `error` event.\n */\n sourcedataloading: MapSourceDataEvent;\n /**\n * Fired when the map's style begins loading or changing asynchronously.\n * All `styledataloading` events are followed by a `styledata`\n * or `error` event.\n */\n styledataloading: MapStyleDataEvent;\n /**\n * Fired when one of the map's sources loads or changes, including if a tile belonging\n * to a source loads or changes.\n */\n sourcedata: MapSourceDataEvent;\n /**\n * Fired when the map's style loads or changes.\n */\n styledata: MapStyleDataEvent;\n /**\n * Fired when an icon or pattern needed by the style is missing. The missing image can\n * be added with {@link Map#addImage} within this event listener callback to prevent the image from\n * being skipped. This event can be used to dynamically generate icons and patterns.\n * @see [Generate and add a missing icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-missing-generated/)\n */\n styleimagemissing: MapStyleImageMissingEvent;\n /**\n * Fired when a request for one of the map's sources' tiles or data is aborted.\n */\n dataabort: MapDataEvent;\n /**\n * Fired when a request for one of the map's sources' data is aborted.\n */\n sourcedataabort: MapSourceDataEvent;\n /**\n * Fired when the user cancels a \"box zoom\" interaction, or when the bounding box does not meet the minimum size threshold.\n * See {@link BoxZoomHandler}.\n */\n boxzoomcancel: MapLibreZoomEvent;\n /**\n * Fired when a \"box zoom\" interaction starts. See {@link BoxZoomHandler}.\n */\n boxzoomstart: MapLibreZoomEvent;\n /**\n * Fired when a \"box zoom\" interaction ends. See {@link BoxZoomHandler}.\n */\n boxzoomend: MapLibreZoomEvent;\n /**\n * Fired when a [`touchcancel`](https://developer.mozilla.org/en-US/docs/Web/Events/touchcancel) event occurs within the map.\n */\n touchcancel: MapTouchEvent;\n /**\n * Fired when a [`touchmove`](https://developer.mozilla.org/en-US/docs/Web/Events/touchmove) event occurs within the map.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchmove: MapTouchEvent;\n /**\n * Fired when a [`touchend`](https://developer.mozilla.org/en-US/docs/Web/Events/touchend) event occurs within the map.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchend: MapTouchEvent;\n /**\n * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the map.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n touchstart: MapTouchEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed and released at the same point on the map.\n *\n * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/)\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n */\n click: MapMouseEvent;\n /**\n * Fired when the right button of the mouse is clicked or the context menu key is pressed within the map.\n */\n contextmenu: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed and released twice at the same point on the map in rapid succession.\n *\n * **Note:** Under normal conditions, this event will be preceded by two `click` events.\n */\n dblclick: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is moved while the cursor is inside the map.\n * As you move the cursor across the map, the event will fire every time the cursor changes position within the map.\n *\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on over](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n */\n mousemove: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is released within the map.\n *\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n mouseup: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is pressed within the map.\n *\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n mousedown: MapMouseEvent;\n /**\n * Fired when a point device (usually a mouse) leaves the map's canvas.\n */\n mouseout: MapMouseEvent;\n /**\n * Fired when a pointing device (usually a mouse) is moved within the map.\n * As you move the cursor across a web page containing a map,\n * the event will fire each time it enters the map or any child elements.\n *\n * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/)\n * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n */\n mouseover: MapMouseEvent;\n /**\n * Fired just before the map begins a transition from one\n * view to another, as the result of either user interaction or methods such as {@link Map#jumpTo}.\n *\n */\n movestart: MapLibreEvent;\n /**\n * Fired repeatedly during an animated transition from one view to\n * another, as the result of either user interaction or methods such as {@link Map#flyTo}.\n *\n * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/)\n */\n move: MapLibreEvent;\n /**\n * Fired just after the map completes a transition from one\n * view to another, as the result of either user interaction or methods such as {@link Map#jumpTo}.\n *\n * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/)\n */\n moveend: MapLibreEvent;\n /**\n * Fired just before the map begins a transition from one zoom level to another,\n * as the result of either user interaction or methods such as {@link Map#flyTo}.\n */\n zoomstart: MapLibreEvent;\n /**\n * Fired repeatedly during an animated transition from one zoom level to another,\n * as the result of either user interaction or methods such as {@link Map#flyTo}.\n */\n zoom: MapLibreEvent;\n /**\n * Fired just after the map completes a transition from one zoom level to another,\n * as the result of either user interaction or methods such as {@link Map#flyTo}.\n */\n zoomend: MapLibreEvent;\n /**\n * Fired when a \"drag to rotate\" interaction starts. See {@link DragRotateHandler}.\n */\n rotatestart: MapLibreEvent;\n /**\n * Fired repeatedly during a \"drag to rotate\" interaction. See {@link DragRotateHandler}.\n */\n rotate: MapLibreEvent;\n /**\n * Fired when a \"drag to rotate\" interaction ends. See {@link DragRotateHandler}.\n */\n rotateend: MapLibreEvent;\n /**\n * Fired when a \"drag to pan\" interaction starts. See {@link DragPanHandler}.\n */\n dragstart: MapLibreEvent;\n /**\n * Fired repeatedly during a \"drag to pan\" interaction. See {@link DragPanHandler}.\n */\n drag: MapLibreEvent;\n /**\n * Fired when a \"drag to pan\" interaction ends. See {@link DragPanHandler}.\n * @see [Create a draggable marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/)\n */\n dragend: MapLibreEvent;\n /**\n * Fired whenever the map's pitch (tilt) begins a change as\n * the result of either user interaction or methods such as {@link Map#flyTo} .\n */\n pitchstart: MapLibreEvent;\n /**\n * Fired repeatedly during the map's pitch (tilt) animation between\n * one state and another as the result of either user interaction\n * or methods such as {@link Map#flyTo}.\n */\n pitch: MapLibreEvent;\n /**\n * Fired immediately after the map's pitch (tilt) finishes changing as\n * the result of either user interaction or methods such as {@link Map#flyTo}.\n */\n pitchend: MapLibreEvent;\n /**\n * Fired when a [`wheel`](https://developer.mozilla.org/en-US/docs/Web/Events/wheel) event occurs within the map.\n */\n wheel: MapWheelEvent;\n /**\n * Fired when terrain is changed\n */\n terrain: MapTerrainEvent;\n};\n\n/**\n * The base event for MapLibre\n *\n * @group Event Related\n */\nexport type MapLibreEvent = {\n type: keyof MapEventType | keyof MapLayerEventType;\n target: Map;\n originalEvent: TOrig;\n}\n\n/**\n * The style data event\n *\n * @group Event Related\n */\nexport type MapStyleDataEvent = MapLibreEvent & {\n dataType: 'style';\n}\n\n/**\n * The source data event interface\n *\n * @group Event Related\n */\nexport type MapSourceDataEvent = MapLibreEvent & {\n dataType: 'source';\n /**\n * True if the event has a `dataType` of `source` and the source has no outstanding network requests.\n */\n isSourceLoaded: boolean;\n /**\n * The [style spec representation of the source](https://maplibre.org/maplibre-style-spec/#sources) if the event has a `dataType` of `source`.\n */\n source: SourceSpecification;\n sourceId: string;\n sourceDataType: MapSourceDataType;\n /**\n * The tile being loaded or changed, if the event has a `dataType` of `source` and\n * the event is related to loading of a tile.\n */\n tile: any;\n}\n/**\n * `MapMouseEvent` is the event type for mouse-related map events.\n * @example\n * ```ts\n * // The `click` event is an example of a `MapMouseEvent`.\n * // Set up an event listener on the map.\n * map.on('click', function(e) {\n * // The event object (e) contains information like the\n * // coordinates of the point on the map that was clicked.\n * console.log('A click event has occurred at ' + e.lngLat);\n * });\n * ```\n */\nexport class MapMouseEvent extends Event implements MapLibreEvent {\n /**\n * The event type\n */\n type: 'mousedown' | 'mouseup' | 'click' | 'dblclick' | 'mousemove' | 'mouseover' | 'mouseenter' | 'mouseleave' | 'mouseout' | 'contextmenu';\n\n /**\n * The `Map` object that fired the event.\n */\n target: Map;\n\n /**\n * The DOM event which caused the map event.\n */\n originalEvent: MouseEvent;\n\n /**\n * The pixel coordinates of the mouse cursor, relative to the map and measured from the top left corner.\n */\n point: Point;\n\n /**\n * The geographic location on the map of the mouse cursor.\n */\n lngLat: LngLat;\n\n /**\n * Prevents subsequent default processing of the event by the map.\n *\n * Calling this method will prevent the following default map behaviors:\n *\n * * On `mousedown` events, the behavior of {@link DragPanHandler}\n * * On `mousedown` events, the behavior of {@link DragRotateHandler}\n * * On `mousedown` events, the behavior of {@link BoxZoomHandler}\n * * On `dblclick` events, the behavior of {@link DoubleClickZoomHandler}\n *\n */\n preventDefault() {\n this._defaultPrevented = true;\n }\n\n /**\n * `true` if `preventDefault` has been called.\n */\n get defaultPrevented(): boolean {\n return this._defaultPrevented;\n }\n\n _defaultPrevented: boolean;\n\n constructor(type: string, map: Map, originalEvent: MouseEvent, data: any = {}) {\n const point = DOM.mousePos(map.getCanvasContainer(), originalEvent);\n const lngLat = map.unproject(point);\n super(type, extend({point, lngLat, originalEvent}, data));\n this._defaultPrevented = false;\n this.target = map;\n }\n}\n\n/**\n * `MapTouchEvent` is the event type for touch-related map events.\n *\n * @group Event Related\n */\nexport class MapTouchEvent extends Event implements MapLibreEvent {\n /**\n * The event type.\n */\n type: 'touchstart' | 'touchmove' | 'touchend' | 'touchcancel';\n\n /**\n * The `Map` object that fired the event.\n */\n target: Map;\n\n /**\n * The DOM event which caused the map event.\n */\n originalEvent: TouchEvent;\n\n /**\n * The geographic location on the map of the center of the touch event points.\n */\n lngLat: LngLat;\n\n /**\n * The pixel coordinates of the center of the touch event points, relative to the map and measured from the top left\n * corner.\n */\n point: Point;\n\n /**\n * The array of pixel coordinates corresponding to a\n * [touch event's `touches`](https://developer.mozilla.org/en-US/docs/Web/API/TouchEvent/touches) property.\n */\n points: Array;\n\n /**\n * The geographical locations on the map corresponding to a\n * [touch event's `touches`](https://developer.mozilla.org/en-US/docs/Web/API/TouchEvent/touches) property.\n */\n lngLats: Array;\n\n /**\n * Prevents subsequent default processing of the event by the map.\n *\n * Calling this method will prevent the following default map behaviors:\n *\n * * On `touchstart` events, the behavior of {@link DragPanHandler}\n * * On `touchstart` events, the behavior of {@link TwoFingersTouchZoomRotateHandler}\n *\n */\n preventDefault() {\n this._defaultPrevented = true;\n }\n\n /**\n * `true` if `preventDefault` has been called.\n */\n get defaultPrevented(): boolean {\n return this._defaultPrevented;\n }\n\n _defaultPrevented: boolean;\n\n constructor(type: string, map: Map, originalEvent: TouchEvent) {\n const touches = type === 'touchend' ? originalEvent.changedTouches : originalEvent.touches;\n const points = DOM.touchPos(map.getCanvasContainer(), touches);\n const lngLats = points.map((t) => map.unproject(t));\n const point = points.reduce((prev, curr, i, arr) => {\n return prev.add(curr.div(arr.length));\n }, new Point(0, 0));\n const lngLat = map.unproject(point);\n super(type, {points, point, lngLats, lngLat, originalEvent});\n this._defaultPrevented = false;\n }\n}\n\n/**\n * `MapWheelEvent` is the event type for the `wheel` map event.\n *\n * @group Event Related\n */\nexport class MapWheelEvent extends Event {\n /**\n * The event type.\n */\n type: 'wheel';\n\n /**\n * The `Map` object that fired the event.\n */\n target: Map;\n\n /**\n * The DOM event which caused the map event.\n */\n originalEvent: WheelEvent;\n\n /**\n * Prevents subsequent default processing of the event by the map.\n *\n * Calling this method will prevent the behavior of {@link ScrollZoomHandler}.\n */\n preventDefault() {\n this._defaultPrevented = true;\n }\n\n /**\n * `true` if `preventDefault` has been called.\n */\n get defaultPrevented(): boolean {\n return this._defaultPrevented;\n }\n\n _defaultPrevented: boolean;\n\n /** */\n constructor(type: string, map: Map, originalEvent: WheelEvent) {\n super(type, {originalEvent});\n this._defaultPrevented = false;\n }\n}\n\n/**\n * A `MapLibreZoomEvent` is the event type for the boxzoom-related map events emitted by the {@link BoxZoomHandler}.\n *\n * @group Event Related\n */\nexport type MapLibreZoomEvent = {\n /**\n * The type of boxzoom event. One of `boxzoomstart`, `boxzoomend` or `boxzoomcancel`\n */\n type: 'boxzoomstart' | 'boxzoomend' | 'boxzoomcancel';\n /**\n * The `Map` instance that triggered the event\n */\n target: Map;\n /**\n * The DOM event that triggered the boxzoom event. Can be a `MouseEvent` or `KeyboardEvent`\n */\n originalEvent: MouseEvent;\n};\n\n/**\n * A `MapDataEvent` object is emitted with the `data`\n * and `dataloading` events. Possible values for\n * `dataType`s are:\n *\n * - `'source'`: The non-tile data associated with any source\n * - `'style'`: The [style](https://maplibre.org/maplibre-style-spec/) used by the map\n *\n * Possible values for `sourceDataType`s are:\n *\n * - `'metadata'`: indicates that any necessary source metadata has been loaded (such as TileJSON) and it is ok to start loading tiles\n * - `'content'`: indicates the source data has changed (such as when source.setData() has been called on GeoJSONSource)\n * - `'visibility'`: send when the source becomes used when at least one of its layers becomes visible in style sense (inside the layer's zoom range and with layout.visibility set to 'visible')\n * - `'idle'`: indicates that no new source data has been fetched (but the source has done loading)\n *\n * @group Event Related\n *\n * @example\n * ```ts\n * // The sourcedata event is an example of MapDataEvent.\n * // Set up an event listener on the map.\n * map.on('sourcedata', function(e) {\n * if (e.isSourceLoaded) {\n * // Do something when the source has finished loading\n * }\n * });\n * ```\n */\nexport type MapDataEvent = {\n /**\n * The event type.\n */\n type: string;\n /**\n * The type of data that has changed. One of `'source'`, `'style'`.\n */\n dataType: string;\n /**\n * Included if the event has a `dataType` of `source` and the event signals that internal data has been received or changed. Possible values are `metadata`, `content`, `visibility` and `idle`.\n */\n sourceDataType: MapSourceDataType;\n};\n\n/**\n * The terrain event\n *\n * @group Event Related\n */\nexport type MapTerrainEvent = {\n type: 'terrain';\n};\n\n/**\n * An event related to the web gl context\n *\n * @group Event Related\n */\nexport type MapContextEvent = {\n type: 'webglcontextlost' | 'webglcontextrestored';\n originalEvent: WebGLContextEvent;\n};\n\n/**\n * The style image missing event\n *\n * @group Event Related\n *\n * @see [Generate and add a missing icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-missing-generated/)\n */\nexport type MapStyleImageMissingEvent = MapLibreEvent & {\n type: 'styleimagemissing';\n id: string;\n}\n","import {MapMouseEvent, MapTouchEvent, MapWheelEvent} from '../events';\nimport {Handler} from '../handler_manager';\nimport type {Map} from '../map';\nimport type Point from '@mapbox/point-geometry';\n\nexport class MapEventHandler implements Handler {\n\n _mousedownPos: Point;\n _clickTolerance: number;\n _map: Map;\n\n constructor(map: Map, options: {\n clickTolerance: number;\n }) {\n this._map = map;\n this._clickTolerance = options.clickTolerance;\n }\n\n reset() {\n delete this._mousedownPos;\n }\n\n wheel(e: WheelEvent) {\n // If mapEvent.preventDefault() is called by the user, prevent handlers such as:\n // - ScrollZoom\n return this._firePreventable(new MapWheelEvent(e.type, this._map, e));\n }\n\n mousedown(e: MouseEvent, point: Point) {\n this._mousedownPos = point;\n // If mapEvent.preventDefault() is called by the user, prevent handlers such as:\n // - MousePan\n // - MouseRotate\n // - MousePitch\n // - DblclickHandler\n return this._firePreventable(new MapMouseEvent(e.type, this._map, e));\n }\n\n mouseup(e: MouseEvent) {\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n click(e: MouseEvent, point: Point) {\n if (this._mousedownPos && this._mousedownPos.dist(point) >= this._clickTolerance) return;\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n dblclick(e: MouseEvent) {\n // If mapEvent.preventDefault() is called by the user, prevent handlers such as:\n // - DblClickZoom\n return this._firePreventable(new MapMouseEvent(e.type, this._map, e));\n }\n\n mouseover(e: MouseEvent) {\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n mouseout(e: MouseEvent) {\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n touchstart(e: TouchEvent) {\n // If mapEvent.preventDefault() is called by the user, prevent handlers such as:\n // - TouchPan\n // - TouchZoom\n // - TouchRotate\n // - TouchPitch\n // - TapZoom\n // - SwipeZoom\n return this._firePreventable(new MapTouchEvent(e.type, this._map, e));\n }\n\n touchmove(e: TouchEvent) {\n this._map.fire(new MapTouchEvent(e.type, this._map, e));\n }\n\n touchend(e: TouchEvent) {\n this._map.fire(new MapTouchEvent(e.type, this._map, e));\n }\n\n touchcancel(e: TouchEvent) {\n this._map.fire(new MapTouchEvent(e.type, this._map, e));\n }\n\n _firePreventable(mapEvent: MapMouseEvent | MapTouchEvent | MapWheelEvent) {\n this._map.fire(mapEvent);\n if (mapEvent.defaultPrevented) {\n // returning an object marks the handler as active and resets other handlers\n return {};\n }\n }\n\n isEnabled() {\n return true;\n }\n\n isActive() {\n return false;\n }\n enable() {}\n disable() {}\n}\n\nexport class BlockableMapEventHandler {\n _map: Map;\n _delayContextMenu: boolean;\n _ignoreContextMenu: boolean;\n _contextMenuEvent: MouseEvent;\n\n constructor(map: Map) {\n this._map = map;\n }\n\n reset() {\n this._delayContextMenu = false;\n this._ignoreContextMenu = true;\n delete this._contextMenuEvent;\n }\n\n mousemove(e: MouseEvent) {\n // mousemove map events should not be fired when interaction handlers (pan, rotate, etc) are active\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n mousedown() {\n this._delayContextMenu = true;\n this._ignoreContextMenu = false;\n }\n\n mouseup() {\n this._delayContextMenu = false;\n if (this._contextMenuEvent) {\n this._map.fire(new MapMouseEvent('contextmenu', this._map, this._contextMenuEvent));\n delete this._contextMenuEvent;\n }\n }\n contextmenu(e: MouseEvent) {\n if (this._delayContextMenu) {\n // Mac: contextmenu fired on mousedown; we save it until mouseup for consistency's sake\n this._contextMenuEvent = e;\n } else if (!this._ignoreContextMenu) {\n // Windows: contextmenu fired on mouseup, so fire event now\n this._map.fire(new MapMouseEvent(e.type, this._map, e));\n }\n\n // prevent browser context menu when necessary\n if (this._map.listens('contextmenu')) {\n e.preventDefault();\n }\n }\n\n isEnabled() {\n return true;\n }\n\n isActive() {\n return false;\n }\n enable() {}\n disable() {}\n}\n","import type {Map} from '../map';\nimport type {PointLike} from '../camera';\nimport type {Transform} from '../../geo/transform';\nimport Point from '@mapbox/point-geometry';\nimport {LngLat} from '../../geo/lng_lat';\n\n/**\n * @internal\n * Shared utilities for the Handler classes to access the correct camera state.\n * If Camera.transformCameraUpdate is specified, the \"desired state\" of camera may differ from the state used for rendering.\n * The handlers need the \"desired state\" to track accumulated changes.\n */\nexport class TransformProvider {\n _map: Map;\n\n constructor(map: Map) {\n this._map = map;\n }\n\n get transform(): Transform {\n return this._map._requestedCameraState || this._map.transform;\n }\n\n get center() {\n return {lng: this.transform.center.lng, lat: this.transform.center.lat};\n }\n\n get zoom() {\n return this.transform.zoom;\n }\n\n get pitch() {\n return this.transform.pitch;\n }\n\n get bearing() {\n return this.transform.bearing;\n }\n\n unproject(point: PointLike): LngLat {\n return this.transform.pointLocation(Point.convert(point), this._map.terrain);\n }\n}\n","import {DOM} from '../../util/dom';\n\nimport {Event} from '../../util/evented';\nimport {TransformProvider} from './transform-provider';\n\nimport type {Map} from '../map';\nimport type Point from '@mapbox/point-geometry';\nimport {Handler} from '../handler_manager';\n\n/**\n * The `BoxZoomHandler` allows the user to zoom the map to fit within a bounding box.\n * The bounding box is defined by clicking and holding `shift` while dragging the cursor.\n *\n * @group Handlers\n */\nexport class BoxZoomHandler implements Handler {\n _map: Map;\n _tr: TransformProvider;\n _el: HTMLElement;\n _container: HTMLElement;\n _enabled: boolean;\n _active: boolean;\n _startPos: Point;\n _lastPos: Point;\n _box: HTMLElement;\n _clickTolerance: number;\n\n /** @internal */\n constructor(map: Map, options: {\n clickTolerance: number;\n }) {\n this._map = map;\n this._tr = new TransformProvider(map);\n this._el = map.getCanvasContainer();\n this._container = map.getContainer();\n this._clickTolerance = options.clickTolerance || 1;\n }\n\n /**\n * Returns a Boolean indicating whether the \"box zoom\" interaction is enabled.\n *\n * @returns `true` if the \"box zoom\" interaction is enabled.\n */\n isEnabled() {\n return !!this._enabled;\n }\n\n /**\n * Returns a Boolean indicating whether the \"box zoom\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"box zoom\" interaction is active.\n */\n isActive() {\n return !!this._active;\n }\n\n /**\n * Enables the \"box zoom\" interaction.\n *\n * @example\n * ```ts\n * map.boxZoom.enable();\n * ```\n */\n enable() {\n if (this.isEnabled()) return;\n this._enabled = true;\n }\n\n /**\n * Disables the \"box zoom\" interaction.\n *\n * @example\n * ```ts\n * map.boxZoom.disable();\n * ```\n */\n disable() {\n if (!this.isEnabled()) return;\n this._enabled = false;\n }\n\n mousedown(e: MouseEvent, point: Point) {\n if (!this.isEnabled()) return;\n if (!(e.shiftKey && e.button === 0)) return;\n\n DOM.disableDrag();\n this._startPos = this._lastPos = point;\n this._active = true;\n }\n\n mousemoveWindow(e: MouseEvent, point: Point) {\n if (!this._active) return;\n\n const pos = point;\n\n if (this._lastPos.equals(pos) || (!this._box && pos.dist(this._startPos) < this._clickTolerance)) {\n return;\n }\n\n const p0 = this._startPos;\n this._lastPos = pos;\n\n if (!this._box) {\n this._box = DOM.create('div', 'maplibregl-boxzoom', this._container);\n this._container.classList.add('maplibregl-crosshair');\n this._fireEvent('boxzoomstart', e);\n }\n\n const minX = Math.min(p0.x, pos.x),\n maxX = Math.max(p0.x, pos.x),\n minY = Math.min(p0.y, pos.y),\n maxY = Math.max(p0.y, pos.y);\n\n DOM.setTransform(this._box, `translate(${minX}px,${minY}px)`);\n\n this._box.style.width = `${maxX - minX}px`;\n this._box.style.height = `${maxY - minY}px`;\n }\n\n mouseupWindow(e: MouseEvent, point: Point) {\n if (!this._active) return;\n\n if (e.button !== 0) return;\n\n const p0 = this._startPos,\n p1 = point;\n\n this.reset();\n\n DOM.suppressClick();\n\n if (p0.x === p1.x && p0.y === p1.y) {\n this._fireEvent('boxzoomcancel', e);\n } else {\n this._map.fire(new Event('boxzoomend', {originalEvent: e}));\n return {\n cameraAnimation: map => map.fitScreenCoordinates(p0, p1, this._tr.bearing, {linear: true})\n };\n }\n }\n\n keydown(e: KeyboardEvent) {\n if (!this._active) return;\n\n if (e.keyCode === 27) {\n this.reset();\n this._fireEvent('boxzoomcancel', e);\n }\n }\n\n reset() {\n this._active = false;\n\n this._container.classList.remove('maplibregl-crosshair');\n\n if (this._box) {\n DOM.remove(this._box);\n this._box = null;\n }\n\n DOM.enableDrag();\n\n delete this._startPos;\n delete this._lastPos;\n }\n\n _fireEvent(type: string, e: any) {\n return this._map.fire(new Event(type, {originalEvent: e}));\n }\n}\n","import Point from '@mapbox/point-geometry';\n\nexport function indexTouches(touches: Array, points: Array) {\n if (touches.length !== points.length) throw new Error(`The number of touches and points are not equal - touches ${touches.length}, points ${points.length}`);\n const obj = {};\n for (let i = 0; i < touches.length; i++) {\n obj[touches[i].identifier] = points[i];\n }\n return obj;\n}\n","import Point from '@mapbox/point-geometry';\nimport {indexTouches} from './handler_util';\n\nfunction getCentroid(points: Array) {\n const sum = new Point(0, 0);\n for (const point of points) {\n sum._add(point);\n }\n return sum.div(points.length);\n}\n\nexport const MAX_TAP_INTERVAL = 500;\nconst MAX_TOUCH_TIME = 500;\nexport const MAX_DIST = 30;\n\nexport class SingleTapRecognizer {\n\n numTouches: number;\n centroid: Point;\n startTime: number;\n aborted: boolean;\n touches: {\n [k in number | string]: Point;\n };\n\n constructor(options: {\n numTouches: number;\n }) {\n this.reset();\n this.numTouches = options.numTouches;\n }\n\n reset() {\n delete this.centroid;\n delete this.startTime;\n delete this.touches;\n this.aborted = false;\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n\n if (this.centroid || mapTouches.length > this.numTouches) {\n this.aborted = true;\n }\n if (this.aborted) {\n return;\n }\n\n if (this.startTime === undefined) {\n this.startTime = e.timeStamp;\n }\n\n if (mapTouches.length === this.numTouches) {\n this.centroid = getCentroid(points);\n this.touches = indexTouches(mapTouches, points);\n }\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n if (this.aborted || !this.centroid) return;\n\n const newTouches = indexTouches(mapTouches, points);\n for (const id in this.touches) {\n const prevPos = this.touches[id];\n const pos = newTouches[id];\n if (!pos || pos.dist(prevPos) > MAX_DIST) {\n this.aborted = true;\n }\n }\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this.centroid || e.timeStamp - this.startTime > MAX_TOUCH_TIME) {\n this.aborted = true;\n }\n\n if (mapTouches.length === 0) {\n const centroid = !this.aborted && this.centroid;\n this.reset();\n if (centroid) return centroid;\n }\n }\n\n}\n\nexport class TapRecognizer {\n\n singleTap: SingleTapRecognizer;\n numTaps: number;\n lastTime: number;\n lastTap: Point;\n count: number;\n\n constructor(options: {\n numTaps: number;\n numTouches: number;\n }) {\n this.singleTap = new SingleTapRecognizer(options);\n this.numTaps = options.numTaps;\n this.reset();\n }\n\n reset() {\n this.lastTime = Infinity;\n delete this.lastTap;\n this.count = 0;\n this.singleTap.reset();\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n this.singleTap.touchstart(e, points, mapTouches);\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n this.singleTap.touchmove(e, points, mapTouches);\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n const tap = this.singleTap.touchend(e, points, mapTouches);\n if (tap) {\n const soonEnough = e.timeStamp - this.lastTime < MAX_TAP_INTERVAL;\n const closeEnough = !this.lastTap || this.lastTap.dist(tap) < MAX_DIST;\n\n if (!soonEnough || !closeEnough) {\n this.reset();\n }\n\n this.count++;\n this.lastTime = e.timeStamp;\n this.lastTap = tap;\n\n if (this.count === this.numTaps) {\n this.reset();\n return tap;\n }\n }\n }\n}\n","import {TapRecognizer} from './tap_recognizer';\nimport type Point from '@mapbox/point-geometry';\nimport type {Map} from '../map';\nimport {TransformProvider} from './transform-provider';\nimport {Handler} from '../handler_manager';\n\nexport class TapZoomHandler implements Handler {\n _tr: TransformProvider;\n _enabled: boolean;\n _active: boolean;\n _zoomIn: TapRecognizer;\n _zoomOut: TapRecognizer;\n\n constructor(map: Map) {\n this._tr = new TransformProvider(map);\n this._zoomIn = new TapRecognizer({\n numTouches: 1,\n numTaps: 2\n });\n\n this._zoomOut = new TapRecognizer({\n numTouches: 2,\n numTaps: 1\n });\n\n this.reset();\n }\n\n reset() {\n this._active = false;\n this._zoomIn.reset();\n this._zoomOut.reset();\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n this._zoomIn.touchstart(e, points, mapTouches);\n this._zoomOut.touchstart(e, points, mapTouches);\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n this._zoomIn.touchmove(e, points, mapTouches);\n this._zoomOut.touchmove(e, points, mapTouches);\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n const zoomInPoint = this._zoomIn.touchend(e, points, mapTouches);\n const zoomOutPoint = this._zoomOut.touchend(e, points, mapTouches);\n const tr = this._tr;\n\n if (zoomInPoint) {\n this._active = true;\n e.preventDefault();\n setTimeout(() => this.reset(), 0);\n return {\n cameraAnimation: (map: Map) => map.easeTo({\n duration: 300,\n zoom: tr.zoom + 1,\n around: tr.unproject(zoomInPoint)\n }, {originalEvent: e})\n };\n } else if (zoomOutPoint) {\n this._active = true;\n e.preventDefault();\n setTimeout(() => this.reset(), 0);\n return {\n cameraAnimation: (map: Map) => map.easeTo({\n duration: 300,\n zoom: tr.zoom - 1,\n around: tr.unproject(zoomOutPoint)\n }, {originalEvent: e})\n };\n }\n }\n\n touchcancel() {\n this.reset();\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n}\n","import {DOM} from '../../util/dom';\nimport type Point from '@mapbox/point-geometry';\nimport {DragMoveStateManager} from './drag_move_state_manager';\nimport {Handler} from '../handler_manager';\n\ninterface DragMovementResult {\n bearingDelta?: number;\n pitchDelta?: number;\n around?: Point;\n panDelta?: Point;\n}\n\nexport interface DragPanResult extends DragMovementResult {\n around: Point;\n panDelta: Point;\n}\n\nexport interface DragRotateResult extends DragMovementResult {\n bearingDelta: number;\n}\n\nexport interface DragPitchResult extends DragMovementResult {\n pitchDelta: number;\n}\n\ntype DragMoveFunction = (lastPoint: Point, point: Point) => T;\n\nexport interface DragMoveHandler extends Handler {\n dragStart: (e: E, point: Point) => void;\n dragMove: (e: E, point: Point) => T | void;\n dragEnd: (e: E) => void;\n getClickTolerance: () => number;\n}\n\nexport type DragMoveHandlerOptions = {\n /**\n * If the movement is shorter than this value, consider it a click.\n */\n clickTolerance: number;\n /**\n * The move function to run on a valid movement.\n */\n move: DragMoveFunction;\n /**\n * A class used to manage the state of the drag event - start, checking valid moves, end. See the class documentation for more details.\n */\n moveStateManager: DragMoveStateManager;\n /**\n * A method used to assign the dragStart, dragMove, and dragEnd methods to the relevant event handlers, as well as assigning the contextmenu handler\n * @param handler - the handler\n */\n assignEvents: (handler: DragMoveHandler) => void;\n /**\n * Should the move start on the \"start\" event, or should it start on the first valid move.\n */\n activateOnStart?: boolean;\n /**\n * If true, handler will be enabled during construction\n */\n enable?: boolean;\n}\n\n/**\n * A generic class to create handlers for drag events, from both mouse and touch events.\n */\nexport class DragHandler implements DragMoveHandler {\n // Event handlers that may be assigned by the implementations of this class\n contextmenu?: Handler['contextmenu'];\n mousedown?: Handler['mousedown'];\n mousemoveWindow?: Handler['mousemoveWindow'];\n mouseup?: Handler['mouseup'];\n touchstart?: Handler['touchstart'];\n touchmoveWindow?: Handler['touchmoveWindow'];\n touchend?: Handler['touchend'];\n\n _clickTolerance: number;\n _moveFunction: DragMoveFunction;\n _activateOnStart: boolean;\n _active: boolean;\n _enabled: boolean;\n _moved: boolean;\n _lastPoint: Point | null;\n _moveStateManager: DragMoveStateManager;\n\n constructor(options: DragMoveHandlerOptions) {\n this._enabled = !!options.enable;\n this._moveStateManager = options.moveStateManager;\n this._clickTolerance = options.clickTolerance || 1;\n this._moveFunction = options.move;\n this._activateOnStart = !!options.activateOnStart;\n\n options.assignEvents(this);\n\n this.reset();\n }\n\n reset(e?: E) {\n this._active = false;\n this._moved = false;\n delete this._lastPoint;\n this._moveStateManager.endMove(e);\n }\n\n _move(...params: Parameters>) {\n const move = this._moveFunction(...params);\n if (move.bearingDelta || move.pitchDelta || move.around || move.panDelta) {\n this._active = true;\n return move;\n }\n }\n\n dragStart(e: E, point: Point);\n dragStart(e: E, point: Point[]);\n dragStart(e: E, point: Point | Point[]) {\n if (!this.isEnabled() || this._lastPoint) return;\n\n if (!this._moveStateManager.isValidStartEvent(e)) return;\n this._moveStateManager.startMove(e);\n\n this._lastPoint = point['length'] ? point[0] : point;\n\n if (this._activateOnStart && this._lastPoint) this._active = true;\n }\n\n dragMove(e: E, point: Point);\n dragMove(e: E, point: Point[]);\n dragMove(e: E, point: Point | Point[]) {\n if (!this.isEnabled()) return;\n const lastPoint = this._lastPoint;\n if (!lastPoint) return;\n e.preventDefault();\n\n if (!this._moveStateManager.isValidMoveEvent(e)) {\n this.reset(e);\n return;\n }\n\n const movePoint = point['length'] ? point[0] : point;\n\n if (!this._moved && movePoint.dist(lastPoint) < this._clickTolerance) return;\n this._moved = true;\n this._lastPoint = movePoint;\n\n return this._move(lastPoint, movePoint);\n }\n\n dragEnd(e: E) {\n if (!this.isEnabled() || !this._lastPoint) return;\n if (!this._moveStateManager.isValidEndEvent(e)) return;\n if (this._moved) DOM.suppressClick();\n this.reset(e);\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n\n getClickTolerance() {\n return this._clickTolerance;\n }\n}\n","import {DOM} from '../../util/dom';\n\nconst LEFT_BUTTON = 0;\nconst RIGHT_BUTTON = 2;\n\n// the values for each button in MouseEvent.buttons\nconst BUTTONS_FLAGS = {\n [LEFT_BUTTON]: 1,\n [RIGHT_BUTTON]: 2\n};\n\nfunction buttonNoLongerPressed(e: MouseEvent, button: number) {\n const flag = BUTTONS_FLAGS[button];\n return e.buttons === undefined || (e.buttons & flag) !== flag;\n}\n\n/*\n * Drag events are initiated by specific interaction which needs to be tracked until it ends.\n * This requires some state management:\n * 1. registering the initiating event,\n * 2. tracking that it was not canceled / not confusing it with another event firing.\n * 3. recognizing the ending event and cleaning up any internal state\n *\n * Concretely, we implement two state managers:\n * 1. MouseMoveStateManager\n * Receives a functions that is used to recognize mouse events that should be registered as the\n * relevant drag interactions - i.e. dragging with the right mouse button, or while CTRL is pressed.\n * 2. OneFingerTouchMoveStateManager\n * Checks if a drag event is using one finger, and continuously tracking that this is the same event\n * (i.e. to make sure not additional finger has started interacting with the screen before raising\n * the first finger).\n */\nexport interface DragMoveStateManager {\n startMove: (e: E) => void;\n endMove: (e?: E) => void;\n isValidStartEvent: (e: E) => boolean;\n isValidMoveEvent: (e: E) => boolean;\n isValidEndEvent: (e?: E) => boolean;\n}\n\nexport class MouseMoveStateManager implements DragMoveStateManager {\n _eventButton: number | undefined;\n _correctEvent: (e: MouseEvent) => boolean;\n\n constructor(options: {\n checkCorrectEvent: (e: MouseEvent) => boolean;\n }) {\n this._correctEvent = options.checkCorrectEvent;\n }\n\n startMove(e: MouseEvent) {\n const eventButton = DOM.mouseButton(e);\n this._eventButton = eventButton;\n }\n\n endMove(_e?: MouseEvent) {\n delete this._eventButton;\n }\n\n isValidStartEvent(e: MouseEvent) {\n return this._correctEvent(e);\n }\n\n isValidMoveEvent(e: MouseEvent) {\n // Some browsers don't fire a `mouseup` when the mouseup occurs outside\n // the window or iframe:\n // https://github.com/mapbox/mapbox-gl-js/issues/4622\n //\n // If the button is no longer pressed during this `mousemove` it may have\n // been released outside of the window or iframe.\n return !buttonNoLongerPressed(e, this._eventButton);\n }\n\n isValidEndEvent(e: MouseEvent) {\n const eventButton = DOM.mouseButton(e);\n return eventButton === this._eventButton;\n }\n}\n\nexport class OneFingerTouchMoveStateManager implements DragMoveStateManager {\n _firstTouch: number | undefined;\n\n constructor() {\n this._firstTouch = undefined;\n }\n\n _isOneFingerTouch(e: TouchEvent) {\n return e.targetTouches.length === 1;\n }\n\n _isSameTouchEvent(e: TouchEvent) {\n return e.targetTouches[0].identifier === this._firstTouch;\n }\n\n startMove(e: TouchEvent) {\n const firstTouch = e.targetTouches[0].identifier;\n this._firstTouch = firstTouch;\n }\n\n endMove(_e?: TouchEvent) {\n delete this._firstTouch;\n }\n\n isValidStartEvent(e: TouchEvent) {\n return this._isOneFingerTouch(e);\n }\n\n isValidMoveEvent(e: TouchEvent) {\n return this._isOneFingerTouch(e) && this._isSameTouchEvent(e);\n }\n\n isValidEndEvent(e: TouchEvent) {\n return this._isOneFingerTouch(e) && this._isSameTouchEvent(e);\n }\n}\n","import type Point from '@mapbox/point-geometry';\n\nimport {DOM} from '../../util/dom';\nimport {DragMoveHandler, DragPanResult, DragRotateResult, DragPitchResult, DragHandler} from './drag_handler';\nimport {MouseMoveStateManager} from './drag_move_state_manager';\n\nexport interface MousePanHandler extends DragMoveHandler {}\nexport interface MouseRotateHandler extends DragMoveHandler {}\nexport interface MousePitchHandler extends DragMoveHandler {}\n\nconst LEFT_BUTTON = 0;\nconst RIGHT_BUTTON = 2;\n\nconst assignEvents = (handler: DragHandler) => {\n handler.mousedown = handler.dragStart;\n handler.mousemoveWindow = handler.dragMove;\n handler.mouseup = handler.dragEnd;\n handler.contextmenu = function(e: MouseEvent) {\n e.preventDefault();\n };\n};\n\nexport const generateMousePanHandler = ({enable, clickTolerance,}: {\n clickTolerance: number;\n enable?: boolean;\n}): MousePanHandler => {\n const mouseMoveStateManager = new MouseMoveStateManager({\n checkCorrectEvent: (e: MouseEvent) => DOM.mouseButton(e) === LEFT_BUTTON && !e.ctrlKey,\n });\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({around: point, panDelta: point.sub(lastPoint)}),\n activateOnStart: true,\n moveStateManager: mouseMoveStateManager,\n enable,\n assignEvents,\n });\n};\n\nexport const generateMouseRotationHandler = ({enable, clickTolerance, bearingDegreesPerPixelMoved = 0.8}: {\n clickTolerance: number;\n bearingDegreesPerPixelMoved?: number;\n enable?: boolean;\n}): MouseRotateHandler => {\n const mouseMoveStateManager = new MouseMoveStateManager({\n checkCorrectEvent: (e: MouseEvent): boolean =>\n (DOM.mouseButton(e) === LEFT_BUTTON && e.ctrlKey) ||\n (DOM.mouseButton(e) === RIGHT_BUTTON),\n });\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({bearingDelta: (point.x - lastPoint.x) * bearingDegreesPerPixelMoved}),\n // prevent browser context menu when necessary; we don't allow it with rotation\n // because we can't discern rotation gesture start from contextmenu on Mac\n moveStateManager: mouseMoveStateManager,\n enable,\n assignEvents,\n });\n};\n\nexport const generateMousePitchHandler = ({enable, clickTolerance, pitchDegreesPerPixelMoved = -0.5}: {\n clickTolerance: number;\n pitchDegreesPerPixelMoved?: number;\n enable?: boolean;\n}): MousePitchHandler => {\n const mouseMoveStateManager = new MouseMoveStateManager({\n checkCorrectEvent: (e: MouseEvent): boolean =>\n (DOM.mouseButton(e) === LEFT_BUTTON && e.ctrlKey) ||\n (DOM.mouseButton(e) === RIGHT_BUTTON),\n });\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({pitchDelta: (point.y - lastPoint.y) * pitchDegreesPerPixelMoved}),\n // prevent browser context menu when necessary; we don't allow it with rotation\n // because we can't discern rotation gesture start from contextmenu on Mac\n moveStateManager: mouseMoveStateManager,\n enable,\n assignEvents,\n });\n};\n","import Point from '@mapbox/point-geometry';\nimport {indexTouches} from './handler_util';\nimport type {Map} from '../map';\nimport {GestureOptions} from '../map';\nimport {Handler} from '../handler_manager';\n\nexport class TouchPanHandler implements Handler {\n\n _enabled: boolean;\n _active: boolean;\n _touches: {\n [k in string | number]: Point;\n };\n _minTouches: number;\n _clickTolerance: number;\n _sum: Point;\n _map: Map;\n _cancelCooperativeMessage: boolean;\n\n constructor(options: {\n clickTolerance: number;\n cooperativeGestures: boolean | GestureOptions;\n }, map: Map) {\n this._minTouches = options.cooperativeGestures ? 2 : 1;\n this._clickTolerance = options.clickTolerance || 1;\n this._map = map;\n this.reset();\n }\n\n reset() {\n this._active = false;\n this._touches = {};\n this._sum = new Point(0, 0);\n\n // Put a delay on the cooperative gesture message so it's less twitchy\n setTimeout(() => {\n this._cancelCooperativeMessage = false;\n }, 200);\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n return this._calculateTransform(e, points, mapTouches);\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n if (this._map._cooperativeGestures) {\n if (this._minTouches === 2 && mapTouches.length < 2 && !this._cancelCooperativeMessage) {\n // If coop gesture enabled, show panning info to user\n this._map._onCooperativeGesture(e, false, mapTouches.length);\n } else if (!this._cancelCooperativeMessage) {\n // If user is successfully navigating, we don't need this warning until the touch resets\n this._cancelCooperativeMessage = true;\n }\n }\n if (!this._active || mapTouches.length < this._minTouches) return;\n e.preventDefault();\n return this._calculateTransform(e, points, mapTouches);\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n this._calculateTransform(e, points, mapTouches);\n\n if (this._active && mapTouches.length < this._minTouches) {\n this.reset();\n }\n }\n\n touchcancel() {\n this.reset();\n }\n\n _calculateTransform(e: TouchEvent, points: Array, mapTouches: Array) {\n if (mapTouches.length > 0) this._active = true;\n\n const touches = indexTouches(mapTouches, points);\n\n const touchPointSum = new Point(0, 0);\n const touchDeltaSum = new Point(0, 0);\n let touchDeltaCount = 0;\n\n for (const identifier in touches) {\n const point = touches[identifier];\n const prevPoint = this._touches[identifier];\n if (prevPoint) {\n touchPointSum._add(point);\n touchDeltaSum._add(point.sub(prevPoint));\n touchDeltaCount++;\n touches[identifier] = point;\n }\n }\n\n this._touches = touches;\n\n if (touchDeltaCount < this._minTouches || !touchDeltaSum.mag()) return;\n\n const panDelta = touchDeltaSum.div(touchDeltaCount);\n this._sum._add(panDelta);\n if (this._sum.mag() < this._clickTolerance) return;\n\n const around = touchPointSum.div(touchDeltaCount);\n\n return {\n around,\n panDelta\n };\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n}\n","import Point from '@mapbox/point-geometry';\nimport {DOM} from '../../util/dom';\nimport type {Map} from '../map';\nimport {Handler} from '../handler_manager';\n\n/**\n * An options object sent to the enable function of some of the handlers\n */\nexport type AroundCenterOptions = {\n /**\n * If \"center\" is passed, map will zoom around the center of map\n */\n around: 'center';\n}\n\n/**\n * The `TwoFingersTouchHandler`s allows the user to zoom, pitch and rotate the map using two fingers\n *\n * @group Handlers\n */\nabstract class TwoFingersTouchHandler implements Handler {\n\n _enabled: boolean;\n _active: boolean;\n _firstTwoTouches: [number, number];\n _vector: Point;\n _startVector: Point;\n _aroundCenter: boolean;\n\n /** @internal */\n constructor() {\n this.reset();\n }\n\n reset() {\n this._active = false;\n delete this._firstTwoTouches;\n }\n\n abstract _start(points: [Point, Point]);\n abstract _move(points: [Point, Point], pinchAround: Point, e: TouchEvent);\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n //log('touchstart', points, e.target.innerHTML, e.targetTouches.length ? e.targetTouches[0].target.innerHTML: undefined);\n if (this._firstTwoTouches || mapTouches.length < 2) return;\n\n this._firstTwoTouches = [\n mapTouches[0].identifier,\n mapTouches[1].identifier\n ];\n\n // implemented by child classes\n this._start([points[0], points[1]]);\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this._firstTwoTouches) return;\n\n e.preventDefault();\n\n const [idA, idB] = this._firstTwoTouches;\n const a = getTouchById(mapTouches, points, idA);\n const b = getTouchById(mapTouches, points, idB);\n if (!a || !b) return;\n const pinchAround = this._aroundCenter ? null : a.add(b).div(2);\n\n // implemented by child classes\n return this._move([a, b], pinchAround, e);\n\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this._firstTwoTouches) return;\n\n const [idA, idB] = this._firstTwoTouches;\n const a = getTouchById(mapTouches, points, idA);\n const b = getTouchById(mapTouches, points, idB);\n if (a && b) return;\n\n if (this._active) DOM.suppressClick();\n\n this.reset();\n }\n\n touchcancel() {\n this.reset();\n }\n\n /**\n * Enables the \"drag to pitch\" interaction.\n *\n * @example\n * ```ts\n * map.touchPitch.enable();\n * ```\n */\n enable(options?: AroundCenterOptions | boolean | null) {\n this._enabled = true;\n this._aroundCenter = !!options && (options as AroundCenterOptions).around === 'center';\n }\n\n /**\n * Disables the \"drag to pitch\" interaction.\n *\n * @example\n * ```ts\n * map.touchPitch.disable();\n * ```\n */\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to pitch\" interaction is enabled.\n *\n * @returns `true` if the \"drag to pitch\" interaction is enabled.\n */\n isEnabled() {\n return this._enabled;\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to pitch\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"drag to pitch\" interaction is active.\n */\n isActive() {\n return this._active;\n }\n}\n\nfunction getTouchById(mapTouches: Array, points: Array, identifier: number) {\n for (let i = 0; i < mapTouches.length; i++) {\n if (mapTouches[i].identifier === identifier) return points[i];\n }\n}\n\n/* ZOOM */\n\nconst ZOOM_THRESHOLD = 0.1;\n\nfunction getZoomDelta(distance, lastDistance) {\n return Math.log(distance / lastDistance) / Math.LN2;\n}\n\n/**\n * The `TwoFingersTouchHandler`s allows the user to zoom the map two fingers\n *\n * @group Handlers\n */\nexport class TwoFingersTouchZoomHandler extends TwoFingersTouchHandler {\n\n _distance: number;\n _startDistance: number;\n\n reset() {\n super.reset();\n delete this._distance;\n delete this._startDistance;\n }\n\n _start(points: [Point, Point]) {\n this._startDistance = this._distance = points[0].dist(points[1]);\n }\n\n _move(points: [Point, Point], pinchAround: Point) {\n const lastDistance = this._distance;\n this._distance = points[0].dist(points[1]);\n if (!this._active && Math.abs(getZoomDelta(this._distance, this._startDistance)) < ZOOM_THRESHOLD) return;\n this._active = true;\n return {\n zoomDelta: getZoomDelta(this._distance, lastDistance),\n pinchAround\n };\n }\n}\n\n/* ROTATE */\n\nconst ROTATION_THRESHOLD = 25; // pixels along circumference of touch circle\n\nfunction getBearingDelta(a, b) {\n return a.angleWith(b) * 180 / Math.PI;\n}\n\n/**\n * The `TwoFingersTouchHandler`s allows the user to rotate the map two fingers\n *\n * @group Handlers\n */\nexport class TwoFingersTouchRotateHandler extends TwoFingersTouchHandler {\n _minDiameter: number;\n\n reset() {\n super.reset();\n delete this._minDiameter;\n delete this._startVector;\n delete this._vector;\n }\n\n _start(points: [Point, Point]) {\n this._startVector = this._vector = points[0].sub(points[1]);\n this._minDiameter = points[0].dist(points[1]);\n }\n\n _move(points: [Point, Point], pinchAround: Point) {\n const lastVector = this._vector;\n this._vector = points[0].sub(points[1]);\n\n if (!this._active && this._isBelowThreshold(this._vector)) return;\n this._active = true;\n\n return {\n bearingDelta: getBearingDelta(this._vector, lastVector),\n pinchAround\n };\n }\n\n _isBelowThreshold(vector: Point) {\n /*\n * The threshold before a rotation actually happens is configured in\n * pixels along the circumference of the circle formed by the two fingers.\n * This makes the threshold in degrees larger when the fingers are close\n * together and smaller when the fingers are far apart.\n *\n * Use the smallest diameter from the whole gesture to reduce sensitivity\n * when pinching in and out.\n */\n\n this._minDiameter = Math.min(this._minDiameter, vector.mag());\n const circumference = Math.PI * this._minDiameter;\n const threshold = ROTATION_THRESHOLD / circumference * 360;\n\n const bearingDeltaSinceStart = getBearingDelta(vector, this._startVector);\n return Math.abs(bearingDeltaSinceStart) < threshold;\n }\n}\n\n/* PITCH */\n\nfunction isVertical(vector) {\n return Math.abs(vector.y) > Math.abs(vector.x);\n}\n\nconst ALLOWED_SINGLE_TOUCH_TIME = 100;\n\n/**\n * The `TwoFingersTouchPitchHandler` allows the user to pitch the map by dragging up and down with two fingers.\n *\n * @group Handlers\n */\nexport class TwoFingersTouchPitchHandler extends TwoFingersTouchHandler {\n\n _valid: boolean | void;\n _firstMove: number;\n _lastPoints: [Point, Point];\n _map: Map;\n _currentTouchCount: number;\n\n constructor(map: Map) {\n super();\n this._map = map;\n }\n\n reset() {\n super.reset();\n this._valid = undefined;\n delete this._firstMove;\n delete this._lastPoints;\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n super.touchstart(e, points, mapTouches);\n this._currentTouchCount = mapTouches.length;\n }\n\n _start(points: [Point, Point]) {\n this._lastPoints = points;\n if (isVertical(points[0].sub(points[1]))) {\n // fingers are more horizontal than vertical\n this._valid = false;\n\n }\n }\n\n _move(points: [Point, Point], center: Point, e: TouchEvent) {\n // If cooperative gestures is enabled, we need a 3-finger minimum for this gesture to register\n if (this._map._cooperativeGestures && this._currentTouchCount < 3) {\n return;\n }\n\n const vectorA = points[0].sub(this._lastPoints[0]);\n const vectorB = points[1].sub(this._lastPoints[1]);\n\n this._valid = this.gestureBeginsVertically(vectorA, vectorB, e.timeStamp);\n if (!this._valid) return;\n\n this._lastPoints = points;\n this._active = true;\n const yDeltaAverage = (vectorA.y + vectorB.y) / 2;\n const degreesPerPixelMoved = -0.5;\n return {\n pitchDelta: yDeltaAverage * degreesPerPixelMoved\n };\n }\n\n gestureBeginsVertically(vectorA: Point, vectorB: Point, timeStamp: number) {\n if (this._valid !== undefined) return this._valid;\n\n const threshold = 2;\n const movedA = vectorA.mag() >= threshold;\n const movedB = vectorB.mag() >= threshold;\n\n // neither finger has moved a meaningful amount, wait\n if (!movedA && !movedB) return;\n\n // One finger has moved and the other has not.\n // If enough time has passed, decide it is not a pitch.\n if (!movedA || !movedB) {\n if (this._firstMove === undefined) {\n this._firstMove = timeStamp;\n }\n\n if (timeStamp - this._firstMove < ALLOWED_SINGLE_TOUCH_TIME) {\n // still waiting for a movement from the second finger\n return undefined;\n } else {\n return false;\n }\n }\n\n const isSameDirection = vectorA.y > 0 === vectorB.y > 0;\n return isVertical(vectorA) && isVertical(vectorB) && isSameDirection;\n }\n}\n","import {Handler} from '../handler_manager';\nimport type {Map} from '../map';\nimport {TransformProvider} from './transform-provider';\n\nconst defaultOptions = {\n panStep: 100,\n bearingStep: 15,\n pitchStep: 10\n};\n\n/**\n * The `KeyboardHandler` allows the user to zoom, rotate, and pan the map using\n * the following keyboard shortcuts:\n *\n * - `=` / `+`: Increase the zoom level by 1.\n * - `Shift-=` / `Shift-+`: Increase the zoom level by 2.\n * - `-`: Decrease the zoom level by 1.\n * - `Shift--`: Decrease the zoom level by 2.\n * - Arrow keys: Pan by 100 pixels.\n * - `Shift+⇢`: Increase the rotation by 15 degrees.\n * - `Shift+⇠`: Decrease the rotation by 15 degrees.\n * - `Shift+⇡`: Increase the pitch by 10 degrees.\n * - `Shift+⇣`: Decrease the pitch by 10 degrees.\n *\n * @group Handlers\n */\nexport class KeyboardHandler implements Handler {\n _tr: TransformProvider;\n _enabled: boolean;\n _active: boolean;\n _panStep: number;\n _bearingStep: number;\n _pitchStep: number;\n _rotationDisabled: boolean;\n\n /** @internal */\n constructor(map: Map) {\n this._tr = new TransformProvider(map);\n const stepOptions = defaultOptions;\n this._panStep = stepOptions.panStep;\n this._bearingStep = stepOptions.bearingStep;\n this._pitchStep = stepOptions.pitchStep;\n this._rotationDisabled = false;\n }\n\n reset() {\n this._active = false;\n }\n\n keydown(e: KeyboardEvent) {\n if (e.altKey || e.ctrlKey || e.metaKey) return;\n\n let zoomDir = 0;\n let bearingDir = 0;\n let pitchDir = 0;\n let xDir = 0;\n let yDir = 0;\n\n switch (e.keyCode) {\n case 61:\n case 107:\n case 171:\n case 187:\n zoomDir = 1;\n break;\n\n case 189:\n case 109:\n case 173:\n zoomDir = -1;\n break;\n\n case 37:\n if (e.shiftKey) {\n bearingDir = -1;\n } else {\n e.preventDefault();\n xDir = -1;\n }\n break;\n\n case 39:\n if (e.shiftKey) {\n bearingDir = 1;\n } else {\n e.preventDefault();\n xDir = 1;\n }\n break;\n\n case 38:\n if (e.shiftKey) {\n pitchDir = 1;\n } else {\n e.preventDefault();\n yDir = -1;\n }\n break;\n\n case 40:\n if (e.shiftKey) {\n pitchDir = -1;\n } else {\n e.preventDefault();\n yDir = 1;\n }\n break;\n\n default:\n return;\n }\n\n if (this._rotationDisabled) {\n bearingDir = 0;\n pitchDir = 0;\n }\n\n return {\n cameraAnimation: (map: Map) => {\n const tr = this._tr;\n map.easeTo({\n duration: 300,\n easeId: 'keyboardHandler',\n easing: easeOut,\n\n zoom: zoomDir ? Math.round(tr.zoom) + zoomDir * (e.shiftKey ? 2 : 1) : tr.zoom,\n bearing: tr.bearing + bearingDir * this._bearingStep,\n pitch: tr.pitch + pitchDir * this._pitchStep,\n offset: [-xDir * this._panStep, -yDir * this._panStep],\n center: tr.center\n }, {originalEvent: e});\n }\n };\n }\n\n /**\n * Enables the \"keyboard rotate and zoom\" interaction.\n *\n * @example\n * ```ts\n * map.keyboard.enable();\n * ```\n */\n enable() {\n this._enabled = true;\n }\n\n /**\n * Disables the \"keyboard rotate and zoom\" interaction.\n *\n * @example\n * ```ts\n * map.keyboard.disable();\n * ```\n */\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n /**\n * Returns a Boolean indicating whether the \"keyboard rotate and zoom\"\n * interaction is enabled.\n *\n * @returns `true` if the \"keyboard rotate and zoom\"\n * interaction is enabled.\n */\n isEnabled() {\n return this._enabled;\n }\n\n /**\n * Returns true if the handler is enabled and has detected the start of a\n * zoom/rotate gesture.\n *\n * @returns `true` if the handler is enabled and has detected the\n * start of a zoom/rotate gesture.\n */\n isActive() {\n return this._active;\n }\n\n /**\n * Disables the \"keyboard pan/rotate\" interaction, leaving the\n * \"keyboard zoom\" interaction enabled.\n *\n * @example\n * ```ts\n * map.keyboard.disableRotation();\n * ```\n */\n disableRotation() {\n this._rotationDisabled = true;\n }\n\n /**\n * Enables the \"keyboard pan/rotate\" interaction.\n *\n * @example\n * ```ts\n * map.keyboard.enable();\n * map.keyboard.enableRotation();\n * ```\n */\n enableRotation() {\n this._rotationDisabled = false;\n }\n}\n\nfunction easeOut(t: number) {\n return t * (2 - t);\n}\n","import {DOM} from '../../util/dom';\n\nimport {defaultEasing, bezier} from '../../util/util';\nimport {browser} from '../../util/browser';\nimport {interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {LngLat} from '../../geo/lng_lat';\nimport {TransformProvider} from './transform-provider';\n\nimport type {Map} from '../map';\nimport type Point from '@mapbox/point-geometry';\nimport type {AroundCenterOptions} from './two_fingers_touch';\nimport {Handler} from '../handler_manager';\n\n// deltaY value for mouse scroll wheel identification\nconst wheelZoomDelta = 4.000244140625;\n\n// These magic numbers control the rate of zoom. Trackpad events fire at a greater\n// frequency than mouse scroll wheel, so reduce the zoom rate per wheel tick\nconst defaultZoomRate = 1 / 100;\nconst wheelZoomRate = 1 / 450;\n\n// upper bound on how much we scale the map in any single render frame; this\n// is used to limit zoom rate in the case of very fast scrolling\nconst maxScalePerFrame = 2;\n\n/**\n * The `ScrollZoomHandler` allows the user to zoom the map by scrolling.\n *\n * @group Handlers\n */\nexport class ScrollZoomHandler implements Handler {\n _map: Map;\n _tr: TransformProvider;\n _el: HTMLElement;\n _enabled: boolean;\n _active: boolean;\n _zooming: boolean;\n _aroundCenter: boolean;\n _around: LngLat;\n _aroundPoint: Point;\n _type: 'wheel' | 'trackpad' | null;\n _lastValue: number;\n _timeout: ReturnType; // used for delayed-handling of a single wheel movement\n _finishTimeout: ReturnType; // used to delay final '{move,zoom}end' events\n\n _lastWheelEvent: any;\n _lastWheelEventTime: number;\n\n _startZoom: number;\n _targetZoom: number;\n _delta: number;\n _easing: ((a: number) => number);\n _prevEase: {\n start: number;\n duration: number;\n easing: (_: number) => number;\n };\n\n _frameId: boolean;\n _triggerRenderFrame: () => void;\n\n _defaultZoomRate: number;\n _wheelZoomRate: number;\n\n /** @internal */\n constructor(map: Map, triggerRenderFrame: () => void) {\n this._map = map;\n this._tr = new TransformProvider(map);\n this._el = map.getCanvasContainer();\n this._triggerRenderFrame = triggerRenderFrame;\n\n this._delta = 0;\n\n this._defaultZoomRate = defaultZoomRate;\n this._wheelZoomRate = wheelZoomRate;\n }\n\n /**\n * Set the zoom rate of a trackpad\n * @param zoomRate - 1/100 The rate used to scale trackpad movement to a zoom value.\n * @example\n * Speed up trackpad zoom\n * ```ts\n * map.scrollZoom.setZoomRate(1/25);\n * ```\n */\n setZoomRate(zoomRate: number) {\n this._defaultZoomRate = zoomRate;\n }\n\n /**\n * Set the zoom rate of a mouse wheel\n * @param wheelZoomRate - 1/450 The rate used to scale mouse wheel movement to a zoom value.\n * @example\n * Slow down zoom of mouse wheel\n * ```ts\n * map.scrollZoom.setWheelZoomRate(1/600);\n * ```\n */\n setWheelZoomRate(wheelZoomRate: number) {\n this._wheelZoomRate = wheelZoomRate;\n }\n\n /**\n * Returns a Boolean indicating whether the \"scroll to zoom\" interaction is enabled.\n * @returns `true` if the \"scroll to zoom\" interaction is enabled.\n */\n isEnabled() {\n return !!this._enabled;\n }\n\n /*\n * Active state is turned on and off with every scroll wheel event and is set back to false before the map\n * render is called, so _active is not a good candidate for determining if a scroll zoom animation is in\n * progress.\n */\n isActive() {\n return !!this._active || this._finishTimeout !== undefined;\n }\n\n isZooming() {\n return !!this._zooming;\n }\n\n /**\n * Enables the \"scroll to zoom\" interaction.\n *\n * @param options - Options object.\n * @example\n * ```ts\n * map.scrollZoom.enable();\n * map.scrollZoom.enable({ around: 'center' })\n * ```\n */\n enable(options?: AroundCenterOptions | boolean) {\n if (this.isEnabled()) return;\n this._enabled = true;\n this._aroundCenter = !!options && (options as AroundCenterOptions).around === 'center';\n }\n\n /**\n * Disables the \"scroll to zoom\" interaction.\n *\n * @example\n * ```ts\n * map.scrollZoom.disable();\n * ```\n */\n disable() {\n if (!this.isEnabled()) return;\n this._enabled = false;\n }\n\n wheel(e: WheelEvent) {\n if (!this.isEnabled()) return;\n if (this._map._cooperativeGestures) {\n if (e[this._map._metaKey]) {\n e.preventDefault();\n } else {\n return;\n }\n }\n let value = e.deltaMode === WheelEvent.DOM_DELTA_LINE ? e.deltaY * 40 : e.deltaY;\n const now = browser.now(),\n timeDelta = now - (this._lastWheelEventTime || 0);\n\n this._lastWheelEventTime = now;\n\n if (value !== 0 && (value % wheelZoomDelta) === 0) {\n // This one is definitely a mouse wheel event.\n this._type = 'wheel';\n\n } else if (value !== 0 && Math.abs(value) < 4) {\n // This one is definitely a trackpad event because it is so small.\n this._type = 'trackpad';\n\n } else if (timeDelta > 400) {\n // This is likely a new scroll action.\n this._type = null;\n this._lastValue = value;\n\n // Start a timeout in case this was a singular event, and dely it by up to 40ms.\n this._timeout = setTimeout(this._onTimeout, 40, e);\n\n } else if (!this._type) {\n // This is a repeating event, but we don't know the type of event just yet.\n // If the delta per time is small, we assume it's a fast trackpad; otherwise we switch into wheel mode.\n this._type = (Math.abs(timeDelta * value) < 200) ? 'trackpad' : 'wheel';\n\n // Make sure our delayed event isn't fired again, because we accumulate\n // the previous event (which was less than 40ms ago) into this event.\n if (this._timeout) {\n clearTimeout(this._timeout);\n this._timeout = null;\n value += this._lastValue;\n }\n }\n\n // Slow down zoom if shift key is held for more precise zooming\n if (e.shiftKey && value) value = value / 4;\n\n // Only fire the callback if we actually know what type of scrolling device the user uses.\n if (this._type) {\n this._lastWheelEvent = e;\n this._delta -= value;\n if (!this._active) {\n this._start(e);\n }\n }\n\n e.preventDefault();\n }\n\n _onTimeout = (initialEvent: MouseEvent) => {\n this._type = 'wheel';\n this._delta -= this._lastValue;\n if (!this._active) {\n this._start(initialEvent);\n }\n };\n\n _start(e: MouseEvent) {\n if (!this._delta) return;\n\n if (this._frameId) {\n this._frameId = null;\n }\n\n this._active = true;\n if (!this.isZooming()) {\n this._zooming = true;\n }\n\n if (this._finishTimeout) {\n clearTimeout(this._finishTimeout);\n delete this._finishTimeout;\n }\n\n const pos = DOM.mousePos(this._el, e);\n const tr = this._tr;\n\n this._around = LngLat.convert(this._aroundCenter ? tr.center : tr.unproject(pos));\n this._aroundPoint = tr.transform.locationPoint(this._around);\n if (!this._frameId) {\n this._frameId = true;\n this._triggerRenderFrame();\n }\n }\n\n renderFrame() {\n if (!this._frameId) return;\n this._frameId = null;\n\n if (!this.isActive()) return;\n const tr = this._tr.transform;\n\n // if we've had scroll events since the last render frame, consume the\n // accumulated delta, and update the target zoom level accordingly\n if (this._delta !== 0) {\n // For trackpad events and single mouse wheel ticks, use the default zoom rate\n const zoomRate = (this._type === 'wheel' && Math.abs(this._delta) > wheelZoomDelta) ? this._wheelZoomRate : this._defaultZoomRate;\n // Scale by sigmoid of scroll wheel delta.\n let scale = maxScalePerFrame / (1 + Math.exp(-Math.abs(this._delta * zoomRate)));\n\n if (this._delta < 0 && scale !== 0) {\n scale = 1 / scale;\n }\n\n const fromScale = typeof this._targetZoom === 'number' ? tr.zoomScale(this._targetZoom) : tr.scale;\n this._targetZoom = Math.min(tr.maxZoom, Math.max(tr.minZoom, tr.scaleZoom(fromScale * scale)));\n\n // if this is a mouse wheel, refresh the starting zoom and easing\n // function we're using to smooth out the zooming between wheel\n // events\n if (this._type === 'wheel') {\n this._startZoom = tr.zoom;\n this._easing = this._smoothOutEasing(200);\n }\n\n this._delta = 0;\n }\n\n const targetZoom = typeof this._targetZoom === 'number' ?\n this._targetZoom : tr.zoom;\n const startZoom = this._startZoom;\n const easing = this._easing;\n\n let finished = false;\n let zoom;\n if (this._type === 'wheel' && startZoom && easing) {\n\n const t = Math.min((browser.now() - this._lastWheelEventTime) / 200, 1);\n const k = easing(t);\n zoom = interpolates.number(startZoom, targetZoom, k);\n if (t < 1) {\n if (!this._frameId) {\n this._frameId = true;\n }\n } else {\n finished = true;\n }\n } else {\n zoom = targetZoom;\n finished = true;\n }\n\n this._active = true;\n\n if (finished) {\n this._active = false;\n this._finishTimeout = setTimeout(() => {\n this._zooming = false;\n this._triggerRenderFrame();\n delete this._targetZoom;\n delete this._finishTimeout;\n }, 200);\n }\n\n return {\n noInertia: true,\n needsRenderFrame: !finished,\n zoomDelta: zoom - tr.zoom,\n around: this._aroundPoint,\n originalEvent: this._lastWheelEvent\n };\n }\n\n _smoothOutEasing(duration: number) {\n let easing = defaultEasing;\n\n if (this._prevEase) {\n const currentEase = this._prevEase;\n const t = (browser.now() - currentEase.start) / currentEase.duration;\n const speed = currentEase.easing(t + 0.01) - currentEase.easing(t);\n\n // Quick hack to make new bezier that is continuous with last\n const x = 0.27 / Math.sqrt(speed * speed + 0.0001) * 0.01;\n const y = Math.sqrt(0.27 * 0.27 - x * x);\n\n easing = bezier(x, y, 0.25, 1);\n }\n\n this._prevEase = {\n start: browser.now(),\n duration,\n easing\n };\n\n return easing;\n }\n\n reset() {\n this._active = false;\n this._zooming = false;\n delete this._targetZoom;\n if (this._finishTimeout) {\n clearTimeout(this._finishTimeout);\n delete this._finishTimeout;\n }\n }\n}\n","import type {ClickZoomHandler} from '../click_zoom';\nimport type {TapZoomHandler} from './../tap_zoom';\n\n/**\n * The `DoubleClickZoomHandler` allows the user to zoom the map at a point by\n * double clicking or double tapping.\n *\n * @group Handlers\n */\nexport class DoubleClickZoomHandler {\n\n _clickZoom: ClickZoomHandler;\n _tapZoom: TapZoomHandler;\n\n /** @internal */\n constructor(clickZoom: ClickZoomHandler, TapZoom: TapZoomHandler) {\n this._clickZoom = clickZoom;\n this._tapZoom = TapZoom;\n }\n\n /**\n * Enables the \"double click to zoom\" interaction.\n *\n * @example\n * ```ts\n * map.doubleClickZoom.enable();\n * ```\n */\n enable() {\n this._clickZoom.enable();\n this._tapZoom.enable();\n }\n\n /**\n * Disables the \"double click to zoom\" interaction.\n *\n * @example\n * ```ts\n * map.doubleClickZoom.disable();\n * ```\n */\n disable() {\n this._clickZoom.disable();\n this._tapZoom.disable();\n }\n\n /**\n * Returns a Boolean indicating whether the \"double click to zoom\" interaction is enabled.\n *\n * @returns `true` if the \"double click to zoom\" interaction is enabled.\n */\n isEnabled() {\n return this._clickZoom.isEnabled() && this._tapZoom.isEnabled();\n }\n\n /**\n * Returns a Boolean indicating whether the \"double click to zoom\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"double click to zoom\" interaction is active.\n */\n isActive() {\n return this._clickZoom.isActive() || this._tapZoom.isActive();\n }\n}\n","import type Point from '@mapbox/point-geometry';\nimport type {Map} from '../map';\nimport {TransformProvider} from './transform-provider';\nimport {Handler} from '../handler_manager';\n\n/**\n * The `ClickZoomHandler` allows the user to zoom the map at a point by double clicking\n * It is used by other handlers\n */\nexport class ClickZoomHandler implements Handler {\n\n _tr: TransformProvider;\n _enabled: boolean;\n _active: boolean;\n\n /** @internal */\n constructor(map: Map) {\n this._tr = new TransformProvider(map);\n this.reset();\n }\n\n reset() {\n this._active = false;\n }\n\n dblclick(e: MouseEvent, point: Point) {\n e.preventDefault();\n return {\n cameraAnimation: (map: Map) => {\n map.easeTo({\n duration: 300,\n zoom: this._tr.zoom + (e.shiftKey ? -1 : 1),\n around: this._tr.unproject(point)\n }, {originalEvent: e});\n }\n };\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n}\n","import {Handler} from '../handler_manager';\nimport {TapRecognizer, MAX_TAP_INTERVAL, MAX_DIST} from './tap_recognizer';\nimport type Point from '@mapbox/point-geometry';\n\nexport class TapDragZoomHandler implements Handler {\n\n _enabled: boolean;\n _active: boolean;\n _swipePoint: Point;\n _swipeTouch: number;\n _tapTime: number;\n _tapPoint: Point;\n _tap: TapRecognizer;\n\n constructor() {\n\n this._tap = new TapRecognizer({\n numTouches: 1,\n numTaps: 1\n });\n\n this.reset();\n }\n\n reset() {\n this._active = false;\n delete this._swipePoint;\n delete this._swipeTouch;\n delete this._tapTime;\n delete this._tapPoint;\n this._tap.reset();\n }\n\n touchstart(e: TouchEvent, points: Array, mapTouches: Array) {\n if (this._swipePoint) return;\n\n if (!this._tapTime) {\n this._tap.touchstart(e, points, mapTouches);\n } else {\n const swipePoint = points[0];\n\n const soonEnough = e.timeStamp - this._tapTime < MAX_TAP_INTERVAL;\n const closeEnough = this._tapPoint.dist(swipePoint) < MAX_DIST;\n\n if (!soonEnough || !closeEnough) {\n this.reset();\n } else if (mapTouches.length > 0) {\n this._swipePoint = swipePoint;\n this._swipeTouch = mapTouches[0].identifier;\n }\n }\n }\n\n touchmove(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this._tapTime) {\n this._tap.touchmove(e, points, mapTouches);\n } else if (this._swipePoint) {\n if (mapTouches[0].identifier !== this._swipeTouch) {\n return;\n }\n\n const newSwipePoint = points[0];\n const dist = newSwipePoint.y - this._swipePoint.y;\n this._swipePoint = newSwipePoint;\n\n e.preventDefault();\n this._active = true;\n\n return {\n zoomDelta: dist / 128\n };\n }\n }\n\n touchend(e: TouchEvent, points: Array, mapTouches: Array) {\n if (!this._tapTime) {\n const point = this._tap.touchend(e, points, mapTouches);\n if (point) {\n this._tapTime = e.timeStamp;\n this._tapPoint = point;\n }\n } else if (this._swipePoint) {\n if (mapTouches.length === 0) {\n this.reset();\n }\n }\n }\n\n touchcancel() {\n this.reset();\n }\n\n enable() {\n this._enabled = true;\n }\n\n disable() {\n this._enabled = false;\n this.reset();\n }\n\n isEnabled() {\n return this._enabled;\n }\n\n isActive() {\n return this._active;\n }\n}\n","import type {MousePanHandler} from '../mouse';\nimport type {TouchPanHandler} from './../touch_pan';\n\n/**\n * A {@link DragPanHandler} options object\n */\nexport type DragPanOptions = {\n /**\n * factor used to scale the drag velocity\n * @defaultValue 0\n */\n linearity?: number;\n /**\n * easing function applled to `map.panTo` when applying the drag.\n * @param t - the easing function\n * @defaultValue bezier(0, 0, 0.3, 1)\n */\n easing?: (t: number) => number;\n /**\n * the maximum value of the drag velocity.\n * @defaultValue 1400\n */\n deceleration?: number;\n /**\n * the rate at which the speed reduces after the pan ends.\n * @defaultValue 2500\n */\n maxSpeed?: number;\n};\n\n/**\n * The `DragPanHandler` allows the user to pan the map by clicking and dragging\n * the cursor.\n *\n * @group Handlers\n */\nexport class DragPanHandler {\n\n _el: HTMLElement;\n _mousePan: MousePanHandler;\n _touchPan: TouchPanHandler;\n _inertiaOptions: DragPanOptions | boolean;\n\n /** @internal */\n constructor(el: HTMLElement, mousePan: MousePanHandler, touchPan: TouchPanHandler) {\n this._el = el;\n this._mousePan = mousePan;\n this._touchPan = touchPan;\n }\n\n /**\n * Enables the \"drag to pan\" interaction.\n *\n * @param options - Options object\n * @example\n * ```ts\n * map.dragPan.enable();\n * map.dragPan.enable({\n * linearity: 0.3,\n * easing: bezier(0, 0, 0.3, 1),\n * maxSpeed: 1400,\n * deceleration: 2500,\n * });\n * ```\n */\n enable(options?: DragPanOptions | boolean) {\n this._inertiaOptions = options || {};\n this._mousePan.enable();\n this._touchPan.enable();\n this._el.classList.add('maplibregl-touch-drag-pan');\n }\n\n /**\n * Disables the \"drag to pan\" interaction.\n *\n * @example\n * ```ts\n * map.dragPan.disable();\n * ```\n */\n disable() {\n this._mousePan.disable();\n this._touchPan.disable();\n this._el.classList.remove('maplibregl-touch-drag-pan');\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to pan\" interaction is enabled.\n *\n * @returns `true` if the \"drag to pan\" interaction is enabled.\n */\n isEnabled() {\n return this._mousePan.isEnabled() && this._touchPan.isEnabled();\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to pan\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"drag to pan\" interaction is active.\n */\n isActive() {\n return this._mousePan.isActive() || this._touchPan.isActive();\n }\n}\n","import type {MousePitchHandler, MouseRotateHandler} from '../mouse';\n\nexport type DragRotateHandlerOptions = {\n /**\n * Control the map pitch in addition to the bearing\n * @defaultValue true\n */\n pitchWithRotate: boolean;\n}\n\n/**\n * The `DragRotateHandler` allows the user to rotate the map by clicking and\n * dragging the cursor while holding the right mouse button or `ctrl` key.\n *\n * @group Handlers\n */\nexport class DragRotateHandler {\n\n _mouseRotate: MouseRotateHandler;\n _mousePitch: MousePitchHandler;\n _pitchWithRotate: boolean;\n\n /** @internal */\n constructor(options: DragRotateHandlerOptions, mouseRotate: MouseRotateHandler, mousePitch: MousePitchHandler) {\n this._pitchWithRotate = options.pitchWithRotate;\n this._mouseRotate = mouseRotate;\n this._mousePitch = mousePitch;\n }\n\n /**\n * Enables the \"drag to rotate\" interaction.\n *\n * @example\n * ```ts\n * map.dragRotate.enable();\n * ```\n */\n enable() {\n this._mouseRotate.enable();\n if (this._pitchWithRotate) this._mousePitch.enable();\n }\n\n /**\n * Disables the \"drag to rotate\" interaction.\n *\n * @example\n * ```ts\n * map.dragRotate.disable();\n * ```\n */\n disable() {\n this._mouseRotate.disable();\n this._mousePitch.disable();\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to rotate\" interaction is enabled.\n *\n * @returns `true` if the \"drag to rotate\" interaction is enabled.\n */\n isEnabled() {\n return this._mouseRotate.isEnabled() && (!this._pitchWithRotate || this._mousePitch.isEnabled());\n }\n\n /**\n * Returns a Boolean indicating whether the \"drag to rotate\" interaction is active, i.e. currently being used.\n *\n * @returns `true` if the \"drag to rotate\" interaction is active.\n */\n isActive() {\n return this._mouseRotate.isActive() || this._mousePitch.isActive();\n }\n}\n","import type {TwoFingersTouchZoomHandler, TwoFingersTouchRotateHandler, AroundCenterOptions} from '../two_fingers_touch';\nimport type {TapDragZoomHandler} from '../tap_drag_zoom';\n\n/**\n * The `TwoFingersTouchZoomRotateHandler` allows the user to zoom and rotate the map by\n * pinching on a touchscreen.\n *\n * They can zoom with one finger by double tapping and dragging. On the second tap,\n * hold the finger down and drag up or down to zoom in or out.\n *\n * @group Handlers\n */\nexport class TwoFingersTouchZoomRotateHandler {\n\n _el: HTMLElement;\n _touchZoom: TwoFingersTouchZoomHandler;\n _touchRotate: TwoFingersTouchRotateHandler;\n _tapDragZoom: TapDragZoomHandler;\n _rotationDisabled: boolean;\n _enabled: boolean;\n\n /** @internal */\n constructor(el: HTMLElement, touchZoom: TwoFingersTouchZoomHandler, touchRotate: TwoFingersTouchRotateHandler, tapDragZoom: TapDragZoomHandler) {\n this._el = el;\n this._touchZoom = touchZoom;\n this._touchRotate = touchRotate;\n this._tapDragZoom = tapDragZoom;\n this._rotationDisabled = false;\n this._enabled = true;\n }\n\n /**\n * Enables the \"pinch to rotate and zoom\" interaction.\n *\n * @param options - Options object.\n *\n * @example\n * ```ts\n * map.touchZoomRotate.enable();\n * map.touchZoomRotate.enable({ around: 'center' });\n * ```\n */\n enable(options?: AroundCenterOptions | boolean | null) {\n this._touchZoom.enable(options);\n if (!this._rotationDisabled) this._touchRotate.enable(options);\n this._tapDragZoom.enable();\n this._el.classList.add('maplibregl-touch-zoom-rotate');\n }\n\n /**\n * Disables the \"pinch to rotate and zoom\" interaction.\n *\n * @example\n * ```ts\n * map.touchZoomRotate.disable();\n * ```\n */\n disable() {\n this._touchZoom.disable();\n this._touchRotate.disable();\n this._tapDragZoom.disable();\n this._el.classList.remove('maplibregl-touch-zoom-rotate');\n }\n\n /**\n * Returns a Boolean indicating whether the \"pinch to rotate and zoom\" interaction is enabled.\n *\n * @returns `true` if the \"pinch to rotate and zoom\" interaction is enabled.\n */\n isEnabled() {\n return this._touchZoom.isEnabled() &&\n (this._rotationDisabled || this._touchRotate.isEnabled()) &&\n this._tapDragZoom.isEnabled();\n }\n\n /**\n * Returns true if the handler is enabled and has detected the start of a zoom/rotate gesture.\n *\n * @returns `true` if the handler is active, `false` otherwise\n */\n isActive() {\n return this._touchZoom.isActive() || this._touchRotate.isActive() || this._tapDragZoom.isActive();\n }\n\n /**\n * Disables the \"pinch to rotate\" interaction, leaving the \"pinch to zoom\"\n * interaction enabled.\n *\n * @example\n * ```ts\n * map.touchZoomRotate.disableRotation();\n * ```\n */\n disableRotation() {\n this._rotationDisabled = true;\n this._touchRotate.disable();\n }\n\n /**\n * Enables the \"pinch to rotate\" interaction.\n *\n * @example\n * ```ts\n * map.touchZoomRotate.enable();\n * map.touchZoomRotate.enableRotation();\n * ```\n */\n enableRotation() {\n this._rotationDisabled = false;\n if (this._touchZoom.isEnabled()) this._touchRotate.enable();\n }\n}\n","import {Event} from '../util/evented';\nimport {DOM} from '../util/dom';\nimport {Map, CompleteMapOptions} from './map';\nimport {HandlerInertia} from './handler_inertia';\nimport {MapEventHandler, BlockableMapEventHandler} from './handler/map_event';\nimport {BoxZoomHandler} from './handler/box_zoom';\nimport {TapZoomHandler} from './handler/tap_zoom';\nimport {generateMouseRotationHandler, generateMousePitchHandler, generateMousePanHandler} from './handler/mouse';\nimport {TouchPanHandler} from './handler/touch_pan';\nimport {TwoFingersTouchZoomHandler, TwoFingersTouchRotateHandler, TwoFingersTouchPitchHandler} from './handler/two_fingers_touch';\nimport {KeyboardHandler} from './handler/keyboard';\nimport {ScrollZoomHandler} from './handler/scroll_zoom';\nimport {DoubleClickZoomHandler} from './handler/shim/dblclick_zoom';\nimport {ClickZoomHandler} from './handler/click_zoom';\nimport {TapDragZoomHandler} from './handler/tap_drag_zoom';\nimport {DragPanHandler} from './handler/shim/drag_pan';\nimport {DragRotateHandler} from './handler/shim/drag_rotate';\nimport {TwoFingersTouchZoomRotateHandler} from './handler/shim/two_fingers_touch';\nimport {extend} from '../util/util';\nimport {browser} from '../util/browser';\nimport Point from '@mapbox/point-geometry';\n\nexport type InputEvent = MouseEvent | TouchEvent | KeyboardEvent | WheelEvent;\n\nconst isMoving = p => p.zoom || p.drag || p.pitch || p.rotate;\n\nclass RenderFrameEvent extends Event {\n type: 'renderFrame';\n timeStamp: number;\n}\n\n/**\n * Handlers interpret dom events and return camera changes that should be\n * applied to the map (`HandlerResult`s). The camera changes are all deltas.\n * The handler itself should have no knowledge of the map's current state.\n * This makes it easier to merge multiple results and keeps handlers simpler.\n * For example, if there is a mousedown and mousemove, the mousePan handler\n * would return a `panDelta` on the mousemove.\n */\nexport interface Handler {\n enable(): void;\n disable(): void;\n isEnabled(): boolean;\n isActive(): boolean;\n /**\n * `reset` can be called by the manager at any time and must reset everything to it's original state\n */\n reset(): void;\n // Handlers can optionally implement these methods.\n // They are called with dom events whenever those dom evens are received.\n readonly touchstart?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly touchmove?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly touchmoveWindow?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly touchend?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly touchcancel?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void;\n readonly mousedown?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly mousemove?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly mousemoveWindow?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly mouseup?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly mouseupWindow?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly dblclick?: (e: MouseEvent, point: Point) => HandlerResult | void;\n readonly contextmenu?: (e: MouseEvent) => HandlerResult | void;\n readonly wheel?: (e: WheelEvent, point: Point) => HandlerResult | void;\n readonly keydown?: (e: KeyboardEvent) => HandlerResult | void;\n readonly keyup?: (e: KeyboardEvent) => HandlerResult | void;\n /**\n * `renderFrame` is the only non-dom event. It is called during render\n * frames and can be used to smooth camera changes (see scroll handler).\n */\n readonly renderFrame?: () => HandlerResult | void;\n}\n\n/**\n * All handler methods that are called with events can optionally return a `HandlerResult`.\n */\nexport type HandlerResult = {\n panDelta?: Point;\n zoomDelta?: number;\n bearingDelta?: number;\n pitchDelta?: number;\n /**\n * the point to not move when changing the camera\n */\n around?: Point | null;\n /**\n * same as above, except for pinch actions, which are given higher priority\n */\n pinchAround?: Point | null;\n /**\n * A method that can fire a one-off easing by directly changing the map's camera.\n */\n cameraAnimation?: (map: Map) => any;\n /**\n * The last three properties are needed by only one handler: scrollzoom.\n * The DOM event to be used as the `originalEvent` on any camera change events.\n */\n originalEvent?: Event;\n /**\n * Makes the manager trigger a frame, allowing the handler to return multiple results over time (see scrollzoom).\n */\n needsRenderFrame?: boolean;\n /**\n * The camera changes won't get recorded for inertial zooming.\n */\n noInertia?: boolean;\n};\n\nexport type EventInProgress = {\n handlerName: string;\n originalEvent: Event;\n}\n\nexport type EventsInProgress = {\n zoom?: EventInProgress;\n pitch?: EventInProgress;\n rotate?: EventInProgress;\n drag?: EventInProgress;\n}\n\nfunction hasChange(result: HandlerResult) {\n return (result.panDelta && result.panDelta.mag()) || result.zoomDelta || result.bearingDelta || result.pitchDelta;\n}\n\nexport class HandlerManager {\n _map: Map;\n _el: HTMLElement;\n _handlers: Array<{\n handlerName: string;\n handler: Handler;\n allowed: Array;\n }>;\n _eventsInProgress: EventsInProgress;\n _frameId: number;\n _inertia: HandlerInertia;\n _bearingSnap: number;\n _handlersById: {[x: string]: Handler};\n _updatingCamera: boolean;\n _changes: Array<[HandlerResult, EventsInProgress, {[handlerName: string]: Event}]>;\n _terrainMovement: boolean;\n _zoom: {handlerName: string};\n _previousActiveHandlers: {[x: string]: Handler};\n _listeners: Array<[Window | Document | HTMLElement, string, {\n passive?: boolean;\n capture?: boolean;\n } | undefined]>;\n\n constructor(map: Map, options: CompleteMapOptions) {\n this._map = map;\n this._el = this._map.getCanvasContainer();\n this._handlers = [];\n this._handlersById = {};\n this._changes = [];\n\n this._inertia = new HandlerInertia(map);\n this._bearingSnap = options.bearingSnap;\n this._previousActiveHandlers = {};\n\n // Track whether map is currently moving, to compute start/move/end events\n this._eventsInProgress = {};\n\n this._addDefaultHandlers(options);\n\n const el = this._el;\n\n this._listeners = [\n // This needs to be `passive: true` so that a double tap fires two\n // pairs of touchstart/end events in iOS Safari 13. If this is set to\n // `passive: false` then the second pair of events is only fired if\n // preventDefault() is called on the first touchstart. Calling preventDefault()\n // undesirably prevents click events.\n [el, 'touchstart', {passive: true}],\n // This needs to be `passive: false` so that scrolls and pinches can be\n // prevented in browsers that don't support `touch-actions: none`, for example iOS Safari 12.\n [el, 'touchmove', {passive: false}],\n [el, 'touchend', undefined],\n [el, 'touchcancel', undefined],\n\n [el, 'mousedown', undefined],\n [el, 'mousemove', undefined],\n [el, 'mouseup', undefined],\n\n // Bind window-level event listeners for move and up/end events. In the absence of\n // the pointer capture API, which is not supported by all necessary platforms,\n // window-level event listeners give us the best shot at capturing events that\n // fall outside the map canvas element. Use `{capture: true}` for the move event\n // to prevent map move events from being fired during a drag.\n [document, 'mousemove', {capture: true}],\n [document, 'mouseup', undefined],\n\n [el, 'mouseover', undefined],\n [el, 'mouseout', undefined],\n [el, 'dblclick', undefined],\n [el, 'click', undefined],\n\n [el, 'keydown', {capture: false}],\n [el, 'keyup', undefined],\n\n [el, 'wheel', {passive: false}],\n [el, 'contextmenu', undefined],\n\n [window, 'blur', undefined]\n ];\n\n for (const [target, type, listenerOptions] of this._listeners) {\n DOM.addEventListener(target, type, target === document ? this.handleWindowEvent : this.handleEvent, listenerOptions);\n }\n }\n\n destroy() {\n for (const [target, type, listenerOptions] of this._listeners) {\n DOM.removeEventListener(target, type, target === document ? this.handleWindowEvent : this.handleEvent, listenerOptions);\n }\n }\n\n _addDefaultHandlers(options: CompleteMapOptions) {\n const map = this._map;\n const el = map.getCanvasContainer();\n this._add('mapEvent', new MapEventHandler(map, options));\n\n const boxZoom = map.boxZoom = new BoxZoomHandler(map, options);\n this._add('boxZoom', boxZoom);\n if (options.interactive && options.boxZoom) {\n boxZoom.enable();\n }\n\n const tapZoom = new TapZoomHandler(map);\n const clickZoom = new ClickZoomHandler(map);\n map.doubleClickZoom = new DoubleClickZoomHandler(clickZoom, tapZoom);\n this._add('tapZoom', tapZoom);\n this._add('clickZoom', clickZoom);\n if (options.interactive && options.doubleClickZoom) {\n map.doubleClickZoom.enable();\n }\n\n const tapDragZoom = new TapDragZoomHandler();\n this._add('tapDragZoom', tapDragZoom);\n\n const touchPitch = map.touchPitch = new TwoFingersTouchPitchHandler(map);\n this._add('touchPitch', touchPitch);\n if (options.interactive && options.touchPitch) {\n map.touchPitch.enable(options.touchPitch);\n }\n\n const mouseRotate = generateMouseRotationHandler(options);\n const mousePitch = generateMousePitchHandler(options);\n map.dragRotate = new DragRotateHandler(options, mouseRotate, mousePitch);\n this._add('mouseRotate', mouseRotate, ['mousePitch']);\n this._add('mousePitch', mousePitch, ['mouseRotate']);\n if (options.interactive && options.dragRotate) {\n map.dragRotate.enable();\n }\n\n const mousePan = generateMousePanHandler(options);\n const touchPan = new TouchPanHandler(options, map);\n map.dragPan = new DragPanHandler(el, mousePan, touchPan);\n this._add('mousePan', mousePan);\n this._add('touchPan', touchPan, ['touchZoom', 'touchRotate']);\n if (options.interactive && options.dragPan) {\n map.dragPan.enable(options.dragPan);\n }\n\n const touchRotate = new TwoFingersTouchRotateHandler();\n const touchZoom = new TwoFingersTouchZoomHandler();\n map.touchZoomRotate = new TwoFingersTouchZoomRotateHandler(el, touchZoom, touchRotate, tapDragZoom);\n this._add('touchRotate', touchRotate, ['touchPan', 'touchZoom']);\n this._add('touchZoom', touchZoom, ['touchPan', 'touchRotate']);\n if (options.interactive && options.touchZoomRotate) {\n map.touchZoomRotate.enable(options.touchZoomRotate);\n }\n\n const scrollZoom = map.scrollZoom = new ScrollZoomHandler(map, () => this._triggerRenderFrame());\n this._add('scrollZoom', scrollZoom, ['mousePan']);\n if (options.interactive && options.scrollZoom) {\n map.scrollZoom.enable(options.scrollZoom);\n }\n\n const keyboard = map.keyboard = new KeyboardHandler(map);\n this._add('keyboard', keyboard);\n if (options.interactive && options.keyboard) {\n map.keyboard.enable();\n }\n\n this._add('blockableMapEvent', new BlockableMapEventHandler(map));\n }\n\n _add(handlerName: string, handler: Handler, allowed?: Array) {\n this._handlers.push({handlerName, handler, allowed});\n this._handlersById[handlerName] = handler;\n }\n\n stop(allowEndAnimation: boolean) {\n // do nothing if this method was triggered by a gesture update\n if (this._updatingCamera) return;\n\n for (const {handler} of this._handlers) {\n handler.reset();\n }\n this._inertia.clear();\n this._fireEvents({}, {}, allowEndAnimation);\n this._changes = [];\n }\n\n isActive() {\n for (const {handler} of this._handlers) {\n if (handler.isActive()) return true;\n }\n return false;\n }\n\n isZooming() {\n return !!this._eventsInProgress.zoom || this._map.scrollZoom.isZooming();\n }\n isRotating() {\n return !!this._eventsInProgress.rotate;\n }\n\n isMoving() {\n return Boolean(isMoving(this._eventsInProgress)) || this.isZooming();\n }\n\n _blockedByActive(activeHandlers: {[x: string]: Handler}, allowed: Array, myName: string) {\n for (const name in activeHandlers) {\n if (name === myName) continue;\n if (!allowed || allowed.indexOf(name) < 0) {\n return true;\n }\n }\n return false;\n }\n\n handleWindowEvent = (e: { type: 'mousemove' | 'mouseup' | 'touchmove'}) => {\n this.handleEvent(e, `${e.type}Window`);\n };\n\n _getMapTouches(touches: TouchList) {\n const mapTouches = [];\n for (const t of touches) {\n const target = (t.target as any as Node);\n if (this._el.contains(target)) {\n mapTouches.push(t);\n }\n }\n return mapTouches as any as TouchList;\n }\n\n handleEvent = (e: Event, eventName?: keyof Handler) => {\n\n if (e.type === 'blur') {\n this.stop(true);\n return;\n }\n\n this._updatingCamera = true;\n\n const inputEvent = e.type === 'renderFrame' ? undefined : e as InputEvent;\n\n /*\n * We don't call e.preventDefault() for any events by default.\n * Handlers are responsible for calling it where necessary.\n */\n\n const mergedHandlerResult: HandlerResult = {needsRenderFrame: false};\n const eventsInProgress: EventsInProgress = {};\n const activeHandlers = {};\n const eventTouches = (e as TouchEvent).touches;\n\n const mapTouches = eventTouches ? this._getMapTouches(eventTouches) : undefined;\n const points = mapTouches ? DOM.touchPos(this._el, mapTouches) : DOM.mousePos(this._el, ((e as MouseEvent)));\n\n for (const {handlerName, handler, allowed} of this._handlers) {\n if (!handler.isEnabled()) continue;\n\n let data: HandlerResult;\n if (this._blockedByActive(activeHandlers, allowed, handlerName)) {\n handler.reset();\n\n } else {\n if (handler[eventName || e.type]) {\n data = handler[eventName || e.type](e, points, mapTouches);\n this.mergeHandlerResult(mergedHandlerResult, eventsInProgress, data, handlerName, inputEvent);\n if (data && data.needsRenderFrame) {\n this._triggerRenderFrame();\n }\n }\n }\n\n if (data || handler.isActive()) {\n activeHandlers[handlerName] = handler;\n }\n }\n\n const deactivatedHandlers: {[handlerName: string]: Event} = {};\n for (const name in this._previousActiveHandlers) {\n if (!activeHandlers[name]) {\n deactivatedHandlers[name] = inputEvent;\n }\n }\n this._previousActiveHandlers = activeHandlers;\n\n if (Object.keys(deactivatedHandlers).length || hasChange(mergedHandlerResult)) {\n this._changes.push([mergedHandlerResult, eventsInProgress, deactivatedHandlers]);\n this._triggerRenderFrame();\n }\n\n if (Object.keys(activeHandlers).length || hasChange(mergedHandlerResult)) {\n this._map._stop(true);\n }\n\n this._updatingCamera = false;\n\n const {cameraAnimation} = mergedHandlerResult;\n if (cameraAnimation) {\n this._inertia.clear();\n this._fireEvents({}, {}, true);\n this._changes = [];\n cameraAnimation(this._map);\n }\n };\n\n mergeHandlerResult(mergedHandlerResult: HandlerResult,\n eventsInProgress: EventsInProgress,\n handlerResult: HandlerResult,\n name: string,\n e?: InputEvent) {\n if (!handlerResult) return;\n\n extend(mergedHandlerResult, handlerResult);\n\n const eventData = {handlerName: name, originalEvent: handlerResult.originalEvent || e};\n\n // track which handler changed which camera property\n if (handlerResult.zoomDelta !== undefined) {\n eventsInProgress.zoom = eventData;\n }\n if (handlerResult.panDelta !== undefined) {\n eventsInProgress.drag = eventData;\n }\n if (handlerResult.pitchDelta !== undefined) {\n eventsInProgress.pitch = eventData;\n }\n if (handlerResult.bearingDelta !== undefined) {\n eventsInProgress.rotate = eventData;\n }\n\n }\n\n _applyChanges() {\n const combined: HandlerResult = {};\n const combinedEventsInProgress: EventsInProgress = {};\n const combinedDeactivatedHandlers = {};\n\n for (const [change, eventsInProgress, deactivatedHandlers] of this._changes) {\n\n if (change.panDelta) combined.panDelta = (combined.panDelta || new Point(0, 0))._add(change.panDelta);\n if (change.zoomDelta) combined.zoomDelta = (combined.zoomDelta || 0) + change.zoomDelta;\n if (change.bearingDelta) combined.bearingDelta = (combined.bearingDelta || 0) + change.bearingDelta;\n if (change.pitchDelta) combined.pitchDelta = (combined.pitchDelta || 0) + change.pitchDelta;\n if (change.around !== undefined) combined.around = change.around;\n if (change.pinchAround !== undefined) combined.pinchAround = change.pinchAround;\n if (change.noInertia) combined.noInertia = change.noInertia;\n\n extend(combinedEventsInProgress, eventsInProgress);\n extend(combinedDeactivatedHandlers, deactivatedHandlers);\n }\n\n this._updateMapTransform(combined, combinedEventsInProgress, combinedDeactivatedHandlers);\n this._changes = [];\n }\n\n _updateMapTransform(combinedResult: HandlerResult,\n combinedEventsInProgress: EventsInProgress,\n deactivatedHandlers: {[handlerName: string]: Event}) {\n const map = this._map;\n const tr = map._getTransformForUpdate();\n const terrain = map.terrain;\n\n if (!hasChange(combinedResult) && !(terrain && this._terrainMovement)) {\n return this._fireEvents(combinedEventsInProgress, deactivatedHandlers, true);\n }\n\n let {panDelta, zoomDelta, bearingDelta, pitchDelta, around, pinchAround} = combinedResult;\n\n if (pinchAround !== undefined) {\n around = pinchAround;\n }\n\n // stop any ongoing camera animations (easeTo, flyTo)\n map._stop(true);\n\n around = around || map.transform.centerPoint;\n const loc = tr.pointLocation(panDelta ? around.sub(panDelta) : around);\n if (bearingDelta) tr.bearing += bearingDelta;\n if (pitchDelta) tr.pitch += pitchDelta;\n if (zoomDelta) tr.zoom += zoomDelta;\n\n if (!terrain) {\n tr.setLocationAtPoint(loc, around);\n } else {\n // when 3d-terrain is enabled act a little different:\n // - dragging do not drag the picked point itself, instead it drags the map by pixel-delta.\n // With this approach it is no longer possible to pick a point from somewhere near\n // the horizon to the center in one move.\n // So this logic avoids the problem, that in such cases you easily loose orientation.\n if (!this._terrainMovement &&\n (combinedEventsInProgress.drag || combinedEventsInProgress.zoom)) {\n // When starting to drag or move, flag it and register moveend to clear flagging\n this._terrainMovement = true;\n this._map._elevationFreeze = true;\n tr.setLocationAtPoint(loc, around);\n this._map.once('moveend', () => {\n this._map._elevationFreeze = false;\n this._terrainMovement = false;\n tr.recalculateZoom(map.terrain);\n });\n } else if (combinedEventsInProgress.drag && this._terrainMovement) {\n // drag map\n tr.center = tr.pointLocation(tr.centerPoint.sub(panDelta));\n } else {\n tr.setLocationAtPoint(loc, around);\n }\n }\n\n map._applyUpdatedTransform(tr);\n\n this._map._update();\n if (!combinedResult.noInertia) this._inertia.record(combinedResult);\n this._fireEvents(combinedEventsInProgress, deactivatedHandlers, true);\n\n }\n\n _fireEvents(newEventsInProgress: EventsInProgress, deactivatedHandlers: {[handlerName: string]: Event}, allowEndAnimation: boolean) {\n\n const wasMoving = isMoving(this._eventsInProgress);\n const nowMoving = isMoving(newEventsInProgress);\n\n const startEvents = {};\n\n for (const eventName in newEventsInProgress) {\n const {originalEvent} = newEventsInProgress[eventName];\n if (!this._eventsInProgress[eventName]) {\n startEvents[`${eventName}start`] = originalEvent;\n }\n this._eventsInProgress[eventName] = newEventsInProgress[eventName];\n }\n\n // fire start events only after this._eventsInProgress has been updated\n if (!wasMoving && nowMoving) {\n this._fireEvent('movestart', nowMoving.originalEvent);\n }\n\n for (const name in startEvents) {\n this._fireEvent(name, startEvents[name]);\n }\n\n if (nowMoving) {\n this._fireEvent('move', nowMoving.originalEvent);\n }\n\n for (const eventName in newEventsInProgress) {\n const {originalEvent} = newEventsInProgress[eventName];\n this._fireEvent(eventName, originalEvent);\n }\n\n const endEvents = {};\n\n let originalEndEvent;\n for (const eventName in this._eventsInProgress) {\n const {handlerName, originalEvent} = this._eventsInProgress[eventName];\n if (!this._handlersById[handlerName].isActive()) {\n delete this._eventsInProgress[eventName];\n originalEndEvent = deactivatedHandlers[handlerName] || originalEvent;\n endEvents[`${eventName}end`] = originalEndEvent;\n }\n }\n\n for (const name in endEvents) {\n this._fireEvent(name, endEvents[name]);\n }\n\n const stillMoving = isMoving(this._eventsInProgress);\n if (allowEndAnimation && (wasMoving || nowMoving) && !stillMoving) {\n this._updatingCamera = true;\n const inertialEase = this._inertia._onMoveEnd(this._map.dragPan._inertiaOptions);\n\n const shouldSnapToNorth = bearing => bearing !== 0 && -this._bearingSnap < bearing && bearing < this._bearingSnap;\n\n if (inertialEase && (inertialEase.essential || !browser.prefersReducedMotion)) {\n if (shouldSnapToNorth(inertialEase.bearing || this._map.getBearing())) {\n inertialEase.bearing = 0;\n }\n inertialEase.freezeElevation = true;\n this._map.easeTo(inertialEase, {originalEvent: originalEndEvent});\n } else {\n this._map.fire(new Event('moveend', {originalEvent: originalEndEvent}));\n if (shouldSnapToNorth(this._map.getBearing())) {\n this._map.resetNorth();\n }\n }\n this._updatingCamera = false;\n }\n\n }\n\n _fireEvent(type: string, e?: Event) {\n this._map.fire(new Event(type, e ? {originalEvent: e} : {}));\n }\n\n _requestFrame() {\n this._map.triggerRepaint();\n return this._map._renderTaskQueue.add(timeStamp => {\n delete this._frameId;\n this.handleEvent(new RenderFrameEvent('renderFrame', {timeStamp}));\n this._applyChanges();\n });\n }\n\n _triggerRenderFrame() {\n if (this._frameId === undefined) {\n this._frameId = this._requestFrame();\n }\n }\n}\n","import {extend, warnOnce, clamp, wrap, defaultEasing, pick} from '../util/util';\nimport {interpolates} from '@maplibre/maplibre-gl-style-spec';\nimport {browser} from '../util/browser';\nimport {LngLat} from '../geo/lng_lat';\nimport {LngLatBounds} from '../geo/lng_lat_bounds';\nimport Point from '@mapbox/point-geometry';\nimport {Event, Evented} from '../util/evented';\nimport {Terrain} from '../render/terrain';\n\nimport type {Transform} from '../geo/transform';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport type {LngLatBoundsLike} from '../geo/lng_lat_bounds';\nimport type {TaskID} from '../util/task_queue';\nimport type {PaddingOptions} from '../geo/edge_insets';\nimport {MercatorCoordinate} from '../geo/mercator_coordinate';\n\n/**\n * A [Point](https://github.com/mapbox/point-geometry) or an array of two numbers representing `x` and `y` screen coordinates in pixels.\n *\n * @group Geography and Geometry\n *\n * @example\n * ```ts\n * let p1 = new Point(-77, 38); // a PointLike which is a Point\n * let p2 = [-77, 38]; // a PointLike which is an array of two numbers\n * ```\n */\nexport type PointLike = Point | [number, number];\n\n/**\n * A helper to allow require of at least one propery\n */\nexport type RequireAtLeastOne = { [K in keyof T]-?: Required> & Partial>>; }[keyof T];\n\n/**\n * Options common to {@link Map#jumpTo}, {@link Map#easeTo}, and {@link Map#flyTo}, controlling the desired location,\n * zoom, bearing, and pitch of the camera. All properties are optional, and when a property is omitted, the current\n * camera value for that property will remain unchanged.\n *\n * @example\n * Set the map's initial perspective with CameraOptions\n * ```ts\n * let map = new maplibregl.Map({\n * container: 'map',\n * style: 'https://demotiles.maplibre.org/style.json',\n * center: [-73.5804, 45.53483],\n * pitch: 60,\n * bearing: -60,\n * zoom: 10\n * });\n * ```\n * @see [Set pitch and bearing](https://maplibre.org/maplibre-gl-js/docs/examples/set-perspective/)\n * @see [Jump to a series of locations](https://maplibre.org/maplibre-gl-js/docs/examples/jump-to/)\n * @see [Fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto/)\n * @see [Display buildings in 3D](https://maplibre.org/maplibre-gl-js/docs/examples/3d-buildings/)\n */\nexport type CameraOptions = CenterZoomBearing & {\n /**\n * The desired pitch in degrees. The pitch is the angle towards the horizon\n * measured in degrees with a range between 0 and 60 degrees. For example, pitch: 0 provides the appearance\n * of looking straight down at the map, while pitch: 60 tilts the user's perspective towards the horizon.\n * Increasing the pitch value is often used to display 3D objects.\n */\n pitch?: number;\n /**\n * If `zoom` is specified, `around` determines the point around which the zoom is centered.\n */\n around?: LngLatLike;\n};\n\n/**\n * Holds center, zoom and bearing properties\n */\nexport type CenterZoomBearing = {\n /**\n * The desired center.\n */\n center?: LngLatLike;\n /**\n * The desired zoom level.\n */\n zoom?: number;\n /**\n * The desired bearing in degrees. The bearing is the compass direction that\n * is \"up\". For example, `bearing: 90` orients the map so that east is up.\n */\n bearing?: number;\n}\n\n/**\n * The options object related to the {@link Map#jumpTo} method\n */\nexport type JumpToOptions = CameraOptions & {\n /**\n * Dimensions in pixels applied on each side of the viewport for shifting the vanishing point.\n */\n padding?: PaddingOptions;\n}\n\n/**\n * A options object for the {@link Map#cameraForBounds} method\n */\nexport type CameraForBoundsOptions = CameraOptions & {\n /**\n * The amount of padding in pixels to add to the given bounds.\n */\n padding?: number | RequireAtLeastOne;\n /**\n * The center of the given bounds relative to the map's center, measured in pixels.\n * @defaultValue [0, 0]\n */\n offset?: PointLike;\n /**\n * The maximum zoom level to allow when the camera would transition to the specified bounds.\n */\n maxZoom?: number;\n}\n\n/**\n * The {@link Map#flyTo} options object\n */\nexport type FlyToOptions = AnimationOptions & CameraOptions & {\n /**\n * The zooming \"curve\" that will occur along the\n * flight path. A high value maximizes zooming for an exaggerated animation, while a low\n * value minimizes zooming for an effect closer to {@link Map#easeTo}. 1.42 is the average\n * value selected by participants in the user study discussed in\n * [van Wijk (2003)](https://www.win.tue.nl/~vanwijk/zoompan.pdf). A value of\n * `Math.pow(6, 0.25)` would be equivalent to the root mean squared average velocity. A\n * value of 1 would produce a circular motion.\n * @defaultValue 1.42\n */\n curve?: number;\n /**\n * The zero-based zoom level at the peak of the flight path. If\n * `options.curve` is specified, this option is ignored.\n */\n minZoom?: number;\n /**\n * The average speed of the animation defined in relation to\n * `options.curve`. A speed of 1.2 means that the map appears to move along the flight path\n * by 1.2 times `options.curve` screenfuls every second. A _screenful_ is the map's visible span.\n * It does not correspond to a fixed physical distance, but varies by zoom level.\n * @defaultValue 1.2\n */\n speed?: number;\n /**\n * The average speed of the animation measured in screenfuls\n * per second, assuming a linear timing curve. If `options.speed` is specified, this option is ignored.\n */\n screenSpeed?: number;\n /**\n * The animation's maximum duration, measured in milliseconds.\n * If duration exceeds maximum duration, it resets to 0.\n */\n maxDuration?: number;\n /**\n * The amount of padding in pixels to add to the given bounds.\n */\n padding?: number | RequireAtLeastOne;\n}\n\nexport type EaseToOptions = AnimationOptions & CameraOptions & {\n delayEndEvents?: number;\n padding?: number | RequireAtLeastOne;\n}\n\n/**\n * Options for {@link Map#fitBounds} method\n */\nexport type FitBoundsOptions = FlyToOptions & {\n /**\n * If `true`, the map transitions using {@link Map#easeTo}. If `false`, the map transitions using {@link Map#flyTo}.\n * See those functions and {@link AnimationOptions} for information about options available.\n * @defaultValue false\n */\n linear?: boolean;\n /**\n * The center of the given bounds relative to the map's center, measured in pixels.\n * @defaultValue [0, 0]\n */\n offset?: PointLike;\n /**\n * The maximum zoom level to allow when the map view transitions to the specified bounds.\n */\n maxZoom?: number;\n}\n\n/**\n * Options common to map movement methods that involve animation, such as {@link Map#panBy} and\n * {@link Map#easeTo}, controlling the duration and easing function of the animation. All properties\n * are optional.\n *\n */\nexport type AnimationOptions = {\n /**\n * The animation's duration, measured in milliseconds.\n */\n duration?: number;\n /**\n * A function taking a time in the range 0..1 and returning a number where 0 is\n * the initial state and 1 is the final state.\n */\n easing?: (_: number) => number;\n /**\n * of the target center relative to real map container center at the end of animation.\n */\n offset?: PointLike;\n /**\n * If `false`, no animation will occur.\n */\n animate?: boolean;\n /**\n * If `true`, then the animation is considered essential and will not be affected by\n * [`prefers-reduced-motion`](https://developer.mozilla.org/en-US/docs/Web/CSS/\\@media/prefers-reduced-motion).\n */\n essential?: boolean;\n /**\n * Default false. Needed in 3D maps to let the camera stay in a constant\n * height based on sea-level. After the animation finished the zoom-level will be recalculated in respect of\n * the distance from the camera to the center-coordinate-altitude.\n */\n freezeElevation?: boolean;\n};\n\n/**\n * A callback hook that allows manipulating the camera and being notified about camera updates before they happen\n */\nexport type CameraUpdateTransformFunction = (next: {\n center: LngLat;\n zoom: number;\n pitch: number;\n bearing: number;\n elevation: number;\n}) => {\n center?: LngLat;\n zoom?: number;\n pitch?: number;\n bearing?: number;\n elevation?: number;\n};\n\nexport abstract class Camera extends Evented {\n transform: Transform;\n terrain: Terrain;\n\n _moving: boolean;\n _zooming: boolean;\n _rotating: boolean;\n _pitching: boolean;\n _padding: boolean;\n\n _bearingSnap: number;\n _easeStart: number;\n _easeOptions: {\n duration?: number;\n easing?: (_: number) => number;\n };\n _easeId: string | void;\n\n _onEaseFrame: (_: number) => void;\n _onEaseEnd: (easeId?: string) => void;\n _easeFrameId: TaskID;\n\n /**\n * @internal\n * holds the geographical coordinate of the target\n */\n _elevationCenter: LngLat;\n /**\n * @internal\n * holds the targ altitude value, = center elevation of the target.\n * This value may changes during flight, because new terrain-tiles loads during flight.\n */\n _elevationTarget: number;\n /**\n * @internal\n * holds the start altitude value, = center elevation before animation begins\n * this value will recalculated during flight in respect of changing _elevationTarget values,\n * so the linear interpolation between start and target keeps smooth and without jumps.\n */\n _elevationStart: number;\n /**\n * @internal\n * Saves the current state of the elevation freeze - this is used during map movement to prevent \"rocky\" camera movement.\n */\n _elevationFreeze: boolean;\n /**\n * @internal\n * Used to track accumulated changes during continuous interaction\n */\n _requestedCameraState?: Transform;\n /**\n * A callback used to defer camera updates or apply arbitrary constraints.\n * If specified, this Camera instance can be used as a stateless component in React etc.\n */\n transformCameraUpdate: CameraUpdateTransformFunction | null;\n\n abstract _requestRenderFrame(a: () => void): TaskID;\n abstract _cancelRenderFrame(_: TaskID): void;\n\n constructor(transform: Transform, options: {\n bearingSnap: number;\n }) {\n super();\n this._moving = false;\n this._zooming = false;\n this.transform = transform;\n this._bearingSnap = options.bearingSnap;\n\n this.on('moveend', () => {\n delete this._requestedCameraState;\n });\n }\n\n /**\n * Returns the map's geographical centerpoint.\n *\n * @returns The map's geographical centerpoint.\n * @example\n * Return a LngLat object such as `{lng: 0, lat: 0}`\n * ```ts\n * let center = map.getCenter();\n * // access longitude and latitude values directly\n * let {lng, lat} = map.getCenter();\n * ```\n */\n getCenter(): LngLat { return new LngLat(this.transform.center.lng, this.transform.center.lat); }\n\n /**\n * Sets the map's geographical centerpoint. Equivalent to `jumpTo({center: center})`.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param center - The centerpoint to set.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * map.setCenter([-74, 38]);\n * ```\n */\n setCenter(center: LngLatLike, eventData?: any) {\n return this.jumpTo({center}, eventData);\n }\n\n /**\n * Pans the map by the specified offset.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param offset - `x` and `y` coordinates by which to pan the map.\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/)\n */\n panBy(offset: PointLike, options?: AnimationOptions, eventData?: any): this {\n offset = Point.convert(offset).mult(-1);\n return this.panTo(this.transform.center, extend({offset}, options), eventData);\n }\n\n /**\n * Pans the map to the specified location with an animated transition.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param lnglat - The location to pan the map to.\n * @param options - Options describing the destination and animation of the transition.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * map.panTo([-74, 38]);\n * // Specify that the panTo animation should last 5000 milliseconds.\n * map.panTo([-74, 38], {duration: 5000});\n * ```\n * @see [Update a feature in realtime](https://maplibre.org/maplibre-gl-js/docs/examples/live-update-feature/)\n */\n panTo(lnglat: LngLatLike, options?: AnimationOptions, eventData?: any): this {\n return this.easeTo(extend({\n center: lnglat\n }, options), eventData);\n }\n\n /**\n * Returns the map's current zoom level.\n *\n * @returns The map's current zoom level.\n * @example\n * ```ts\n * map.getZoom();\n * ```\n */\n getZoom(): number { return this.transform.zoom; }\n\n /**\n * Sets the map's zoom level. Equivalent to `jumpTo({zoom: zoom})`.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`.\n *\n * @param zoom - The zoom level to set (0-20).\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Zoom to the zoom level 5 without an animated transition\n * ```ts\n * map.setZoom(5);\n * ```\n */\n setZoom(zoom: number, eventData?: any): this {\n this.jumpTo({zoom}, eventData);\n return this;\n }\n\n /**\n * Zooms the map to the specified zoom level, with an animated transition.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`.\n *\n * @param zoom - The zoom level to transition to.\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * // Zoom to the zoom level 5 without an animated transition\n * map.zoomTo(5);\n * // Zoom to the zoom level 8 with an animated transition\n * map.zoomTo(8, {\n * duration: 2000,\n * offset: [100, 50]\n * });\n * ```\n */\n zoomTo(zoom: number, options?: AnimationOptions | null, eventData?: any): this {\n return this.easeTo(extend({\n zoom\n }, options), eventData);\n }\n\n /**\n * Increases the map's zoom level by 1.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Zoom the map in one level with a custom animation duration\n * ```ts\n * map.zoomIn({duration: 1000});\n * ```\n */\n zoomIn(options?: AnimationOptions, eventData?: any): this {\n this.zoomTo(this.getZoom() + 1, options, eventData);\n return this;\n }\n\n /**\n * Decreases the map's zoom level by 1.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Zoom the map out one level with a custom animation offset\n * ```ts\n * map.zoomOut({offset: [80, 60]});\n * ```\n */\n zoomOut(options?: AnimationOptions, eventData?: any): this {\n this.zoomTo(this.getZoom() - 1, options, eventData);\n return this;\n }\n\n /**\n * Returns the map's current bearing. The bearing is the compass direction that is \"up\"; for example, a bearing\n * of 90° orients the map so that east is up.\n *\n * @returns The map's current bearing.\n * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/)\n */\n getBearing(): number { return this.transform.bearing; }\n\n /**\n * Sets the map's bearing (rotation). The bearing is the compass direction that is \"up\"; for example, a bearing\n * of 90° orients the map so that east is up.\n *\n * Equivalent to `jumpTo({bearing: bearing})`.\n *\n * Triggers the following events: `movestart`, `moveend`, and `rotate`.\n *\n * @param bearing - The desired bearing.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Rotate the map to 90 degrees\n * ```ts\n * map.setBearing(90);\n * ```\n */\n setBearing(bearing: number, eventData?: any): this {\n this.jumpTo({bearing}, eventData);\n return this;\n }\n\n /**\n * Returns the current padding applied around the map viewport.\n *\n * @returns The current padding around the map viewport.\n */\n getPadding(): PaddingOptions { return this.transform.padding; }\n\n /**\n * Sets the padding in pixels around the viewport.\n *\n * Equivalent to `jumpTo({padding: padding})`.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param padding - The desired padding.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * Sets a left padding of 300px, and a top padding of 50px\n * ```ts\n * map.setPadding({ left: 300, top: 50 });\n * ```\n */\n setPadding(padding: PaddingOptions, eventData?: any): this {\n this.jumpTo({padding}, eventData);\n return this;\n }\n\n /**\n * Rotates the map to the specified bearing, with an animated transition. The bearing is the compass direction\n * that is \"up\"; for example, a bearing of 90° orients the map so that east is up.\n *\n * Triggers the following events: `movestart`, `moveend`, and `rotate`.\n *\n * @param bearing - The desired bearing.\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n rotateTo(bearing: number, options?: AnimationOptions, eventData?: any): this {\n return this.easeTo(extend({\n bearing\n }, options), eventData);\n }\n\n /**\n * Rotates the map so that north is up (0° bearing), with an animated transition.\n *\n * Triggers the following events: `movestart`, `moveend`, and `rotate`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n resetNorth(options?: AnimationOptions, eventData?: any): this {\n this.rotateTo(0, extend({duration: 1000}, options), eventData);\n return this;\n }\n\n /**\n * Rotates and pitches the map so that north is up (0° bearing) and pitch is 0°, with an animated transition.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `pitchstart`, `pitch`, `pitchend`, and `rotate`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n resetNorthPitch(options?: AnimationOptions, eventData?: any): this {\n this.easeTo(extend({\n bearing: 0,\n pitch: 0,\n duration: 1000\n }, options), eventData);\n return this;\n }\n\n /**\n * Snaps the map so that north is up (0° bearing), if the current bearing is close enough to it (i.e. within the\n * `bearingSnap` threshold).\n *\n * Triggers the following events: `movestart`, `moveend`, and `rotate`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n snapToNorth(options?: AnimationOptions, eventData?: any): this {\n if (Math.abs(this.getBearing()) < this._bearingSnap) {\n return this.resetNorth(options, eventData);\n }\n return this;\n }\n\n /**\n * Returns the map's current pitch (tilt).\n *\n * @returns The map's current pitch, measured in degrees away from the plane of the screen.\n */\n getPitch(): number { return this.transform.pitch; }\n\n /**\n * Sets the map's pitch (tilt). Equivalent to `jumpTo({pitch: pitch})`.\n *\n * Triggers the following events: `movestart`, `moveend`, `pitchstart`, and `pitchend`.\n *\n * @param pitch - The pitch to set, measured in degrees away from the plane of the screen (0-60).\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n */\n setPitch(pitch: number, eventData?: any): this {\n this.jumpTo({pitch}, eventData);\n return this;\n }\n\n /**\n * @param bounds - Calculate the center for these bounds in the viewport and use\n * the highest zoom level up to and including `Map#getMaxZoom()` that fits\n * in the viewport. LngLatBounds represent a box that is always axis-aligned with bearing 0.\n * @param options - Options object\n * @returns If map is able to fit to provided bounds, returns `center`, `zoom`, and `bearing`.\n * If map is unable to fit, method will warn and return undefined.\n * @example\n * ```ts\n * let bbox = [[-79, 43], [-73, 45]];\n * let newCameraTransform = map.cameraForBounds(bbox, {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n */\n cameraForBounds(bounds: LngLatBoundsLike, options?: CameraForBoundsOptions): CenterZoomBearing {\n bounds = LngLatBounds.convert(bounds);\n const bearing = options && options.bearing || 0;\n return this._cameraForBoxAndBearing(bounds.getNorthWest(), bounds.getSouthEast(), bearing, options);\n }\n\n /**\n * @internal\n * Calculate the center of these two points in the viewport and use\n * the highest zoom level up to and including `Map#getMaxZoom()` that fits\n * the points in the viewport at the specified bearing.\n * @param p0 - First point\n * @param p1 - Second point\n * @param bearing - Desired map bearing at end of animation, in degrees\n * @param options - the camera options\n * @returns If map is able to fit to provided bounds, returns `center`, `zoom`, and `bearing`.\n * If map is unable to fit, method will warn and return undefined.\n * @example\n * ```ts\n * let p0 = [-79, 43];\n * let p1 = [-73, 45];\n * let bearing = 90;\n * let newCameraTransform = map._cameraForBoxAndBearing(p0, p1, bearing, {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n */\n _cameraForBoxAndBearing(p0: LngLatLike, p1: LngLatLike, bearing: number, options?: CameraForBoundsOptions): CenterZoomBearing {\n const defaultPadding = {\n top: 0,\n bottom: 0,\n right: 0,\n left: 0\n };\n options = extend({\n padding: defaultPadding,\n offset: [0, 0],\n maxZoom: this.transform.maxZoom\n }, options);\n\n if (typeof options.padding === 'number') {\n const p = options.padding;\n options.padding = {\n top: p,\n bottom: p,\n right: p,\n left: p\n };\n }\n\n options.padding = extend(defaultPadding, options.padding) as PaddingOptions;\n const tr = this.transform;\n const edgePadding = tr.padding;\n\n // We want to calculate the upper right and lower left of the box defined by p0 and p1\n // in a coordinate system rotate to match the destination bearing.\n const p0world = tr.project(LngLat.convert(p0));\n const p1world = tr.project(LngLat.convert(p1));\n const p0rotated = p0world.rotate(-bearing * Math.PI / 180);\n const p1rotated = p1world.rotate(-bearing * Math.PI / 180);\n\n const upperRight = new Point(Math.max(p0rotated.x, p1rotated.x), Math.max(p0rotated.y, p1rotated.y));\n const lowerLeft = new Point(Math.min(p0rotated.x, p1rotated.x), Math.min(p0rotated.y, p1rotated.y));\n\n // Calculate zoom: consider the original bbox and padding.\n const size = upperRight.sub(lowerLeft);\n const scaleX = (tr.width - (edgePadding.left + edgePadding.right + options.padding.left + options.padding.right)) / size.x;\n const scaleY = (tr.height - (edgePadding.top + edgePadding.bottom + options.padding.top + options.padding.bottom)) / size.y;\n\n if (scaleY < 0 || scaleX < 0) {\n warnOnce(\n 'Map cannot fit within canvas with the given bounds, padding, and/or offset.'\n );\n return undefined;\n }\n\n const zoom = Math.min(tr.scaleZoom(tr.scale * Math.min(scaleX, scaleY)), options.maxZoom);\n\n // Calculate center: apply the zoom, the configured offset, as well as offset that exists as a result of padding.\n const offset = Point.convert(options.offset);\n const paddingOffsetX = (options.padding.left - options.padding.right) / 2;\n const paddingOffsetY = (options.padding.top - options.padding.bottom) / 2;\n const paddingOffset = new Point(paddingOffsetX, paddingOffsetY);\n const rotatedPaddingOffset = paddingOffset.rotate(bearing * Math.PI / 180);\n const offsetAtInitialZoom = offset.add(rotatedPaddingOffset);\n const offsetAtFinalZoom = offsetAtInitialZoom.mult(tr.scale / tr.zoomScale(zoom));\n\n const center = tr.unproject(p0world.add(p1world).div(2).sub(offsetAtFinalZoom));\n\n return {\n center,\n zoom,\n bearing\n };\n }\n\n /**\n * Pans and zooms the map to contain its visible area within the specified geographical bounds.\n * This function will also reset the map's bearing to 0 if bearing is nonzero.\n *\n * Triggers the following events: `movestart` and `moveend`.\n *\n * @param bounds - Center these bounds in the viewport and use the highest\n * zoom level up to and including `Map#getMaxZoom()` that fits them in the viewport.\n * @param options- Options supports all properties from {@link AnimationOptions} and {@link CameraOptions} in addition to the fields below.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * let bbox = [[-79, 43], [-73, 45]];\n * map.fitBounds(bbox, {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n * @see [Fit a map to a bounding box](https://maplibre.org/maplibre-gl-js/docs/examples/fitbounds/)\n */\n fitBounds(bounds: LngLatBoundsLike, options?: FitBoundsOptions, eventData?: any): this {\n return this._fitInternal(\n this.cameraForBounds(bounds, options),\n options,\n eventData);\n }\n\n /**\n * Pans, rotates and zooms the map to to fit the box made by points p0 and p1\n * once the map is rotated to the specified bearing. To zoom without rotating,\n * pass in the current map bearing.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend` and `rotate`.\n *\n * @param p0 - First point on screen, in pixel coordinates\n * @param p1 - Second point on screen, in pixel coordinates\n * @param bearing - Desired map bearing at end of animation, in degrees\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * let p0 = [220, 400];\n * let p1 = [500, 900];\n * map.fitScreenCoordinates(p0, p1, map.getBearing(), {\n * padding: {top: 10, bottom:25, left: 15, right: 5}\n * });\n * ```\n * @see Used by {@link BoxZoomHandler}\n */\n fitScreenCoordinates(p0: PointLike, p1: PointLike, bearing: number, options?: FitBoundsOptions, eventData?: any): this {\n return this._fitInternal(\n this._cameraForBoxAndBearing(\n this.transform.pointLocation(Point.convert(p0)),\n this.transform.pointLocation(Point.convert(p1)),\n bearing,\n options),\n options,\n eventData);\n }\n\n _fitInternal(calculatedOptions?: CenterZoomBearing, options?: FitBoundsOptions, eventData?: any): this {\n // cameraForBounds warns + returns undefined if unable to fit:\n if (!calculatedOptions) return this;\n\n options = extend(calculatedOptions, options);\n // Explicitly remove the padding field because, calculatedOptions already accounts for padding by setting zoom and center accordingly.\n delete options.padding;\n\n return options.linear ?\n this.easeTo(options, eventData) :\n this.flyTo(options, eventData);\n }\n\n /**\n * Changes any combination of center, zoom, bearing, and pitch, without\n * an animated transition. The map will retain its current values for any\n * details not specified in `options`.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`,\n * `pitch`, `pitchend`, and `rotate`.\n *\n * @param options - Options object\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * // jump to coordinates at current zoom\n * map.jumpTo({center: [0, 0]});\n * // jump with zoom, pitch, and bearing options\n * map.jumpTo({\n * center: [0, 0],\n * zoom: 8,\n * pitch: 45,\n * bearing: 90\n * });\n * ```\n * @see [Jump to a series of locations](https://maplibre.org/maplibre-gl-js/docs/examples/jump-to/)\n * @see [Update a feature in realtime](https://maplibre.org/maplibre-gl-js/docs/examples/live-update-feature/)\n */\n jumpTo(options: JumpToOptions, eventData?: any): this {\n this.stop();\n\n const tr = this._getTransformForUpdate();\n let zoomChanged = false,\n bearingChanged = false,\n pitchChanged = false;\n\n if ('zoom' in options && tr.zoom !== +options.zoom) {\n zoomChanged = true;\n tr.zoom = +options.zoom;\n }\n\n if (options.center !== undefined) {\n tr.center = LngLat.convert(options.center);\n }\n\n if ('bearing' in options && tr.bearing !== +options.bearing) {\n bearingChanged = true;\n tr.bearing = +options.bearing;\n }\n\n if ('pitch' in options && tr.pitch !== +options.pitch) {\n pitchChanged = true;\n tr.pitch = +options.pitch;\n }\n\n if (options.padding != null && !tr.isPaddingEqual(options.padding)) {\n tr.padding = options.padding;\n }\n this._applyUpdatedTransform(tr);\n\n this.fire(new Event('movestart', eventData))\n .fire(new Event('move', eventData));\n\n if (zoomChanged) {\n this.fire(new Event('zoomstart', eventData))\n .fire(new Event('zoom', eventData))\n .fire(new Event('zoomend', eventData));\n }\n\n if (bearingChanged) {\n this.fire(new Event('rotatestart', eventData))\n .fire(new Event('rotate', eventData))\n .fire(new Event('rotateend', eventData));\n }\n\n if (pitchChanged) {\n this.fire(new Event('pitchstart', eventData))\n .fire(new Event('pitch', eventData))\n .fire(new Event('pitchend', eventData));\n }\n\n return this.fire(new Event('moveend', eventData));\n }\n\n /**\n * Calculates pitch, zoom and bearing for looking at `newCenter` with the camera position being `newCenter`\n * and returns them as {@link CameraOptions}.\n * @param from - The camera to look from\n * @param altitudeFrom - The altitude of the camera to look from\n * @param to - The center to look at\n * @param altitudeTo - Optional altitude of the center to look at. If none given the ground height will be used.\n * @returns the calculated camera options\n */\n calculateCameraOptionsFromTo(from: LngLat, altitudeFrom: number, to: LngLat, altitudeTo: number = 0): CameraOptions {\n const fromMerc = MercatorCoordinate.fromLngLat(from, altitudeFrom);\n const toMerc = MercatorCoordinate.fromLngLat(to, altitudeTo);\n const dx = toMerc.x - fromMerc.x;\n const dy = toMerc.y - fromMerc.y;\n const dz = toMerc.z - fromMerc.z;\n\n const distance3D = Math.hypot(dx, dy, dz);\n if (distance3D === 0) throw new Error('Can\\'t calculate camera options with same From and To');\n\n const groundDistance = Math.hypot(dx, dy);\n\n const zoom = this.transform.scaleZoom(this.transform.cameraToCenterDistance / distance3D / this.transform.tileSize);\n const bearing = (Math.atan2(dx, -dy) * 180) / Math.PI;\n let pitch = (Math.acos(groundDistance / distance3D) * 180) / Math.PI;\n pitch = dz < 0 ? 90 - pitch : 90 + pitch;\n\n return {\n center: toMerc.toLngLat(),\n zoom,\n pitch,\n bearing\n };\n }\n\n /**\n * Changes any combination of `center`, `zoom`, `bearing`, `pitch`, and `padding` with an animated transition\n * between old and new values. The map will retain its current values for any\n * details not specified in `options`.\n *\n * Note: The transition will happen instantly if the user has enabled\n * the `reduced motion` accessibility feature enabled in their operating system,\n * unless `options` includes `essential: true`.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`,\n * `pitch`, `pitchend`, and `rotate`.\n *\n * @param options - Options describing the destination and animation of the transition.\n * Accepts {@link CameraOptions} and {@link AnimationOptions}.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/)\n */\n easeTo(options: EaseToOptions & {\n easeId?: string;\n noMoveStart?: boolean;\n }, eventData?: any): this {\n this._stop(false, options.easeId);\n\n options = extend({\n offset: [0, 0],\n duration: 500,\n easing: defaultEasing\n }, options);\n\n if (options.animate === false || (!options.essential && browser.prefersReducedMotion)) options.duration = 0;\n\n const tr = this._getTransformForUpdate(),\n startZoom = this.getZoom(),\n startBearing = this.getBearing(),\n startPitch = this.getPitch(),\n startPadding = this.getPadding(),\n\n zoom = 'zoom' in options ? +options.zoom : startZoom,\n bearing = 'bearing' in options ? this._normalizeBearing(options.bearing, startBearing) : startBearing,\n pitch = 'pitch' in options ? +options.pitch : startPitch,\n padding = 'padding' in options ? options.padding : tr.padding;\n\n const offsetAsPoint = Point.convert(options.offset);\n let pointAtOffset = tr.centerPoint.add(offsetAsPoint);\n const locationAtOffset = tr.pointLocation(pointAtOffset);\n const center = LngLat.convert(options.center || locationAtOffset);\n this._normalizeCenter(center);\n\n const from = tr.project(locationAtOffset);\n const delta = tr.project(center).sub(from);\n const finalScale = tr.zoomScale(zoom - startZoom);\n\n let around, aroundPoint;\n\n if (options.around) {\n around = LngLat.convert(options.around);\n aroundPoint = tr.locationPoint(around);\n }\n\n const currently = {\n moving: this._moving,\n zooming: this._zooming,\n rotating: this._rotating,\n pitching: this._pitching\n };\n\n this._zooming = this._zooming || (zoom !== startZoom);\n this._rotating = this._rotating || (startBearing !== bearing);\n this._pitching = this._pitching || (pitch !== startPitch);\n this._padding = !tr.isPaddingEqual(padding as PaddingOptions);\n\n this._easeId = options.easeId;\n this._prepareEase(eventData, options.noMoveStart, currently);\n if (this.terrain) this._prepareElevation(center);\n\n this._ease((k) => {\n if (this._zooming) {\n tr.zoom = interpolates.number(startZoom, zoom, k);\n }\n if (this._rotating) {\n tr.bearing = interpolates.number(startBearing, bearing, k);\n }\n if (this._pitching) {\n tr.pitch = interpolates.number(startPitch, pitch, k);\n }\n if (this._padding) {\n tr.interpolatePadding(startPadding, padding as PaddingOptions, k);\n // When padding is being applied, Transform#centerPoint is changing continuously,\n // thus we need to recalculate offsetPoint every frame\n pointAtOffset = tr.centerPoint.add(offsetAsPoint);\n }\n\n if (this.terrain && !options.freezeElevation) this._updateElevation(k);\n\n if (around) {\n tr.setLocationAtPoint(around, aroundPoint);\n } else {\n const scale = tr.zoomScale(tr.zoom - startZoom);\n const base = zoom > startZoom ?\n Math.min(2, finalScale) :\n Math.max(0.5, finalScale);\n const speedup = Math.pow(base, 1 - k);\n const newCenter = tr.unproject(from.add(delta.mult(k * speedup)).mult(scale));\n tr.setLocationAtPoint(tr.renderWorldCopies ? newCenter.wrap() : newCenter, pointAtOffset);\n }\n\n this._applyUpdatedTransform(tr);\n\n this._fireMoveEvents(eventData);\n\n }, (interruptingEaseId?: string) => {\n if (this.terrain) this._finalizeElevation();\n this._afterEase(eventData, interruptingEaseId);\n }, options as any);\n\n return this;\n }\n\n _prepareEase(eventData: any, noMoveStart: boolean, currently: any = {}) {\n this._moving = true;\n if (!noMoveStart && !currently.moving) {\n this.fire(new Event('movestart', eventData));\n }\n if (this._zooming && !currently.zooming) {\n this.fire(new Event('zoomstart', eventData));\n }\n if (this._rotating && !currently.rotating) {\n this.fire(new Event('rotatestart', eventData));\n }\n if (this._pitching && !currently.pitching) {\n this.fire(new Event('pitchstart', eventData));\n }\n }\n\n _prepareElevation(center: LngLat) {\n this._elevationCenter = center;\n this._elevationStart = this.transform.elevation;\n this._elevationTarget = this.terrain.getElevationForLngLatZoom(center, this.transform.tileZoom);\n this._elevationFreeze = true;\n }\n\n _updateElevation(k: number) {\n this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this._elevationCenter, this.transform.tileZoom);\n const elevation = this.terrain.getElevationForLngLatZoom(this._elevationCenter, this.transform.tileZoom);\n // target terrain updated during flight, slowly move camera to new height\n if (k < 1 && elevation !== this._elevationTarget) {\n const pitch1 = this._elevationTarget - this._elevationStart;\n const pitch2 = (elevation - (pitch1 * k + this._elevationStart)) / (1 - k);\n this._elevationStart += k * (pitch1 - pitch2);\n this._elevationTarget = elevation;\n }\n this.transform.elevation = interpolates.number(this._elevationStart, this._elevationTarget, k);\n }\n\n _finalizeElevation() {\n this._elevationFreeze = false;\n this.transform.recalculateZoom(this.terrain);\n }\n\n /**\n * @internal\n * Called when the camera is about to be manipulated.\n * If `transformCameraUpdate` is specified, a copy of the current transform is created to track the accumulated changes.\n * This underlying transform represents the \"desired state\" proposed by input handlers / animations / UI controls.\n * It may differ from the state used for rendering (`this.transform`).\n * @returns Transform to apply changes to\n */\n _getTransformForUpdate(): Transform {\n if (!this.transformCameraUpdate) return this.transform;\n\n if (!this._requestedCameraState) {\n this._requestedCameraState = this.transform.clone();\n }\n return this._requestedCameraState;\n }\n\n /**\n * @internal\n * Called after the camera is done being manipulated.\n * @param tr - the requested camera end state\n * Call `transformCameraUpdate` if present, and then apply the \"approved\" changes.\n */\n _applyUpdatedTransform(tr: Transform) {\n if (!this.transformCameraUpdate) return;\n\n const nextTransform = tr.clone();\n const {\n center,\n zoom,\n pitch,\n bearing,\n elevation\n } = this.transformCameraUpdate(nextTransform);\n if (center) nextTransform.center = center;\n if (zoom !== undefined) nextTransform.zoom = zoom;\n if (pitch !== undefined) nextTransform.pitch = pitch;\n if (bearing !== undefined) nextTransform.bearing = bearing;\n if (elevation !== undefined) nextTransform.elevation = elevation;\n this.transform.apply(nextTransform);\n }\n\n _fireMoveEvents(eventData?: any) {\n this.fire(new Event('move', eventData));\n if (this._zooming) {\n this.fire(new Event('zoom', eventData));\n }\n if (this._rotating) {\n this.fire(new Event('rotate', eventData));\n }\n if (this._pitching) {\n this.fire(new Event('pitch', eventData));\n }\n }\n\n _afterEase(eventData?: any, easeId?: string) {\n // if this easing is being stopped to start another easing with\n // the same id then don't fire any events to avoid extra start/stop events\n if (this._easeId && easeId && this._easeId === easeId) {\n return;\n }\n delete this._easeId;\n\n const wasZooming = this._zooming;\n const wasRotating = this._rotating;\n const wasPitching = this._pitching;\n this._moving = false;\n this._zooming = false;\n this._rotating = false;\n this._pitching = false;\n this._padding = false;\n\n if (wasZooming) {\n this.fire(new Event('zoomend', eventData));\n }\n if (wasRotating) {\n this.fire(new Event('rotateend', eventData));\n }\n if (wasPitching) {\n this.fire(new Event('pitchend', eventData));\n }\n this.fire(new Event('moveend', eventData));\n }\n\n /**\n * Changes any combination of center, zoom, bearing, and pitch, animating the transition along a curve that\n * evokes flight. The animation seamlessly incorporates zooming and panning to help\n * the user maintain her bearings even after traversing a great distance.\n *\n * Note: The animation will be skipped, and this will behave equivalently to `jumpTo`\n * if the user has the `reduced motion` accessibility feature enabled in their operating system,\n * unless 'options' includes `essential: true`.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`,\n * `pitch`, `pitchend`, and `rotate`.\n *\n * @param options - Options describing the destination and animation of the transition.\n * Accepts {@link CameraOptions}, {@link AnimationOptions},\n * and the following additional options.\n * @param eventData - Additional properties to be added to event objects of events triggered by this method.\n * @returns `this`\n * @example\n * ```ts\n * // fly with default options to null island\n * map.flyTo({center: [0, 0], zoom: 9});\n * // using flyTo options\n * map.flyTo({\n * center: [0, 0],\n * zoom: 9,\n * speed: 0.2,\n * curve: 1,\n * easing(t) {\n * return t;\n * }\n * });\n * ```\n * @see [Fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto/)\n * @see [Slowly fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto-options/)\n * @see [Fly to a location based on scroll position](https://maplibre.org/maplibre-gl-js/docs/examples/scroll-fly-to/)\n */\n flyTo(options: FlyToOptions, eventData?: any): this {\n // Fall through to jumpTo if user has set prefers-reduced-motion\n if (!options.essential && browser.prefersReducedMotion) {\n const coercedOptions = pick(options, ['center', 'zoom', 'bearing', 'pitch', 'around']) as CameraOptions;\n return this.jumpTo(coercedOptions, eventData);\n }\n\n // This method implements an “optimal path” animation, as detailed in:\n //\n // Van Wijk, Jarke J.; Nuij, Wim A. A. “Smooth and efficient zooming and panning.” INFOVIS\n // ’03. pp. 15–22. .\n //\n // Where applicable, local variable documentation begins with the associated variable or\n // function in van Wijk (2003).\n\n this.stop();\n\n options = extend({\n offset: [0, 0],\n speed: 1.2,\n curve: 1.42,\n easing: defaultEasing\n }, options);\n\n const tr = this._getTransformForUpdate(),\n startZoom = this.getZoom(),\n startBearing = this.getBearing(),\n startPitch = this.getPitch(),\n startPadding = this.getPadding();\n\n const zoom = 'zoom' in options ? clamp(+options.zoom, tr.minZoom, tr.maxZoom) : startZoom;\n const bearing = 'bearing' in options ? this._normalizeBearing(options.bearing, startBearing) : startBearing;\n const pitch = 'pitch' in options ? +options.pitch : startPitch;\n const padding = 'padding' in options ? options.padding : tr.padding;\n\n const scale = tr.zoomScale(zoom - startZoom);\n const offsetAsPoint = Point.convert(options.offset);\n let pointAtOffset = tr.centerPoint.add(offsetAsPoint);\n const locationAtOffset = tr.pointLocation(pointAtOffset);\n const center = LngLat.convert(options.center || locationAtOffset);\n this._normalizeCenter(center);\n\n const from = tr.project(locationAtOffset);\n const delta = tr.project(center).sub(from);\n\n let rho = options.curve;\n\n // w₀: Initial visible span, measured in pixels at the initial scale.\n const w0 = Math.max(tr.width, tr.height),\n // w₁: Final visible span, measured in pixels with respect to the initial scale.\n w1 = w0 / scale,\n // Length of the flight path as projected onto the ground plane, measured in pixels from\n // the world image origin at the initial scale.\n u1 = delta.mag();\n\n if ('minZoom' in options) {\n const minZoom = clamp(Math.min(options.minZoom, startZoom, zoom), tr.minZoom, tr.maxZoom);\n // wm: Maximum visible span, measured in pixels with respect to the initial\n // scale.\n const wMax = w0 / tr.zoomScale(minZoom - startZoom);\n rho = Math.sqrt(wMax / u1 * 2);\n }\n\n // ρ²\n const rho2 = rho * rho;\n\n /**\n * rᵢ: Returns the zoom-out factor at one end of the animation.\n *\n * @param descent - `true` for the descent, `false` for the ascent\n */\n function zoomOutFactor(descent: boolean) {\n const b = (w1 * w1 - w0 * w0 + (descent ? -1 : 1) * rho2 * rho2 * u1 * u1) / (2 * (descent ? w1 : w0) * rho2 * u1);\n return Math.log(Math.sqrt(b * b + 1) - b);\n }\n\n function sinh(n) { return (Math.exp(n) - Math.exp(-n)) / 2; }\n function cosh(n) { return (Math.exp(n) + Math.exp(-n)) / 2; }\n function tanh(n) { return sinh(n) / cosh(n); }\n\n // r₀: Zoom-out factor during ascent.\n const r0 = zoomOutFactor(false);\n\n // w(s): Returns the visible span on the ground, measured in pixels with respect to the\n // initial scale. Assumes an angular field of view of 2 arctan ½ ≈ 53°.\n let w: (_: number) => number = function (s) {\n return (cosh(r0) / cosh(r0 + rho * s));\n };\n\n // u(s): Returns the distance along the flight path as projected onto the ground plane,\n // measured in pixels from the world image origin at the initial scale.\n let u: (_: number) => number = function (s) {\n return w0 * ((cosh(r0) * tanh(r0 + rho * s) - sinh(r0)) / rho2) / u1;\n };\n\n // S: Total length of the flight path, measured in ρ-screenfuls.\n let S = (zoomOutFactor(true) - r0) / rho;\n\n // When u₀ = u₁, the optimal path doesn’t require both ascent and descent.\n if (Math.abs(u1) < 0.000001 || !isFinite(S)) {\n // Perform a more or less instantaneous transition if the path is too short.\n if (Math.abs(w0 - w1) < 0.000001) return this.easeTo(options, eventData);\n\n const k = w1 < w0 ? -1 : 1;\n S = Math.abs(Math.log(w1 / w0)) / rho;\n\n u = function() { return 0; };\n w = function(s) { return Math.exp(k * rho * s); };\n }\n\n if ('duration' in options) {\n options.duration = +options.duration;\n } else {\n const V = 'screenSpeed' in options ? +options.screenSpeed / rho : +options.speed;\n options.duration = 1000 * S / V;\n }\n\n if (options.maxDuration && options.duration > options.maxDuration) {\n options.duration = 0;\n }\n\n this._zooming = true;\n this._rotating = (startBearing !== bearing);\n this._pitching = (pitch !== startPitch);\n this._padding = !tr.isPaddingEqual(padding as PaddingOptions);\n\n this._prepareEase(eventData, false);\n if (this.terrain) this._prepareElevation(center);\n\n this._ease((k) => {\n // s: The distance traveled along the flight path, measured in ρ-screenfuls.\n const s = k * S;\n const scale = 1 / w(s);\n tr.zoom = k === 1 ? zoom : startZoom + tr.scaleZoom(scale);\n\n if (this._rotating) {\n tr.bearing = interpolates.number(startBearing, bearing, k);\n }\n if (this._pitching) {\n tr.pitch = interpolates.number(startPitch, pitch, k);\n }\n if (this._padding) {\n tr.interpolatePadding(startPadding, padding as PaddingOptions, k);\n // When padding is being applied, Transform#centerPoint is changing continuously,\n // thus we need to recalculate offsetPoint every frame\n pointAtOffset = tr.centerPoint.add(offsetAsPoint);\n }\n\n if (this.terrain && !options.freezeElevation) this._updateElevation(k);\n\n const newCenter = k === 1 ? center : tr.unproject(from.add(delta.mult(u(s))).mult(scale));\n tr.setLocationAtPoint(tr.renderWorldCopies ? newCenter.wrap() : newCenter, pointAtOffset);\n\n this._applyUpdatedTransform(tr);\n\n this._fireMoveEvents(eventData);\n\n }, () => {\n if (this.terrain) this._finalizeElevation();\n this._afterEase(eventData);\n }, options);\n\n return this;\n }\n\n isEasing() {\n return !!this._easeFrameId;\n }\n\n /**\n * Stops any animated transition underway.\n *\n * @returns `this`\n */\n stop(): this {\n return this._stop();\n }\n\n _stop(allowGestures?: boolean, easeId?: string): this {\n if (this._easeFrameId) {\n this._cancelRenderFrame(this._easeFrameId);\n delete this._easeFrameId;\n delete this._onEaseFrame;\n }\n\n if (this._onEaseEnd) {\n // The _onEaseEnd function might emit events which trigger new\n // animation, which sets a new _onEaseEnd. Ensure we don't delete\n // it unintentionally.\n const onEaseEnd = this._onEaseEnd;\n delete this._onEaseEnd;\n onEaseEnd.call(this, easeId);\n }\n if (!allowGestures) {\n const handlers = (this as any).handlers;\n if (handlers) handlers.stop(false);\n }\n return this;\n }\n\n _ease(frame: (_: number) => void,\n finish: () => void,\n options: {\n animate?: boolean;\n duration?: number;\n easing?: (_: number) => number;\n }) {\n if (options.animate === false || options.duration === 0) {\n frame(1);\n finish();\n } else {\n this._easeStart = browser.now();\n this._easeOptions = options;\n this._onEaseFrame = frame;\n this._onEaseEnd = finish;\n this._easeFrameId = this._requestRenderFrame(this._renderFrameCallback);\n }\n }\n\n // Callback for map._requestRenderFrame\n _renderFrameCallback = () => {\n const t = Math.min((browser.now() - this._easeStart) / this._easeOptions.duration, 1);\n this._onEaseFrame(this._easeOptions.easing(t));\n\n // if _stop is called during _onEaseFrame from _fireMoveEvents we should avoid a new _requestRenderFrame, checking it by ensuring _easeFrameId was not deleted\n if (t < 1 && this._easeFrameId) {\n this._easeFrameId = this._requestRenderFrame(this._renderFrameCallback);\n } else {\n this.stop();\n }\n };\n\n // convert bearing so that it's numerically close to the current one so that it interpolates properly\n _normalizeBearing(bearing: number, currentBearing: number) {\n bearing = wrap(bearing, -180, 180);\n const diff = Math.abs(bearing - currentBearing);\n if (Math.abs(bearing - 360 - currentBearing) < diff) bearing -= 360;\n if (Math.abs(bearing + 360 - currentBearing) < diff) bearing += 360;\n return bearing;\n }\n\n // If a path crossing the antimeridian would be shorter, extend the final coordinate so that\n // interpolating between the two endpoints will cross it.\n _normalizeCenter(center: LngLat) {\n const tr = this.transform;\n if (!tr.renderWorldCopies || tr.lngRange) return;\n\n const delta = center.lng - tr.center.lng;\n center.lng +=\n delta > 180 ? -360 :\n delta < -180 ? 360 : 0;\n }\n\n /**\n * Query the current elevation of location. It return null if terrain is not enabled. the elevation is in meters relative to mean sea-level\n * @param lngLatLike - [x,y] or LngLat coordinates of the location\n * @returns elevation in meters\n */\n queryTerrainElevation(lngLatLike: LngLatLike): number | null {\n if (!this.terrain) {\n return null;\n }\n const elevation = this.terrain.getElevationForLngLatZoom(LngLat.convert(lngLatLike), this.transform.tileZoom);\n /**\n * Different zoomlevels with different terrain-tiles the elevation-values are not the same.\n * map.transform.elevation variable with the center-altitude.\n * In maplibre the proj-matrix is translated by this value in negative z-direction.\n * So we need to add this value to the elevation to get the correct value.\n */\n return elevation - this.transform.elevation;\n }\n}\n","import {DOM} from '../../util/dom';\n\nimport type {Map} from '../map';\nimport type {ControlPosition, IControl} from './control';\nimport type {MapDataEvent} from '../events';\nimport type {StyleSpecification} from '@maplibre/maplibre-gl-style-spec';\n/**\n * The {@link AttributionControl} options\n */\ntype AttributionOptions = {\n /**\n * If `true`, the attribution control will always collapse when moving the map. If `false`,\n * force the expanded attribution control. The default is a responsive attribution that collapses when the user moves the map on maps less than 640 pixels wide.\n * **Attribution should not be collapsed if it can comfortably fit on the map. `compact` should only be used to modify default attribution when map size makes it impossible to fit default attribution and when the automatic compact resizing for default settings are not sufficient.**\n */\n compact?: boolean;\n /**\n * Attributions to show in addition to any other attributions.\n */\n customAttribution?: string | Array;\n};\n\n/**\n * An `AttributionControl` control presents the map's attribution information. By default, the attribution control is expanded (regardless of map width).\n * @group Markers and Controls\n * @example\n * ```ts\n * let map = new maplibregl.Map({attributionControl: false})\n * .addControl(new maplibregl.AttributionControl({\n * compact: true\n * }));\n * ```\n */\nexport class AttributionControl implements IControl {\n options: AttributionOptions;\n _map: Map;\n _compact: boolean;\n _container: HTMLElement;\n _innerContainer: HTMLElement;\n _compactButton: HTMLElement;\n _editLink: HTMLAnchorElement;\n _attribHTML: string;\n styleId: string;\n styleOwner: string;\n\n /**\n * @param options - the attribution options\n */\n constructor(options: AttributionOptions = {}) {\n this.options = options;\n }\n\n getDefaultPosition(): ControlPosition {\n return 'bottom-right';\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._compact = this.options && this.options.compact;\n this._container = DOM.create('details', 'maplibregl-ctrl maplibregl-ctrl-attrib');\n this._compactButton = DOM.create('summary', 'maplibregl-ctrl-attrib-button', this._container);\n this._compactButton.addEventListener('click', this._toggleAttribution);\n this._setElementTitle(this._compactButton, 'ToggleAttribution');\n this._innerContainer = DOM.create('div', 'maplibregl-ctrl-attrib-inner', this._container);\n\n this._updateAttributions();\n this._updateCompact();\n\n this._map.on('styledata', this._updateData);\n this._map.on('sourcedata', this._updateData);\n this._map.on('terrain', this._updateData);\n this._map.on('resize', this._updateCompact);\n this._map.on('drag', this._updateCompactMinimize);\n\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._container);\n\n this._map.off('styledata', this._updateData);\n this._map.off('sourcedata', this._updateData);\n this._map.off('terrain', this._updateData);\n this._map.off('resize', this._updateCompact);\n this._map.off('drag', this._updateCompactMinimize);\n\n this._map = undefined;\n this._compact = undefined;\n this._attribHTML = undefined;\n }\n\n _setElementTitle(element: HTMLElement, title: string) {\n const str = this._map._getUIString(`AttributionControl.${title}`);\n element.title = str;\n element.setAttribute('aria-label', str);\n }\n\n _toggleAttribution = () => {\n if (this._container.classList.contains('maplibregl-compact')) {\n if (this._container.classList.contains('maplibregl-compact-show')) {\n this._container.setAttribute('open', '');\n this._container.classList.remove('maplibregl-compact-show');\n } else {\n this._container.classList.add('maplibregl-compact-show');\n this._container.removeAttribute('open');\n }\n }\n };\n\n _updateData = (e: MapDataEvent) => {\n if (e && (e.sourceDataType === 'metadata' || e.sourceDataType === 'visibility' || e.dataType === 'style' || e.type === 'terrain')) {\n this._updateAttributions();\n }\n };\n\n _updateAttributions() {\n if (!this._map.style) return;\n let attributions: Array = [];\n if (this.options.customAttribution) {\n if (Array.isArray(this.options.customAttribution)) {\n attributions = attributions.concat(\n this.options.customAttribution.map(attribution => {\n if (typeof attribution !== 'string') return '';\n return attribution;\n })\n );\n } else if (typeof this.options.customAttribution === 'string') {\n attributions.push(this.options.customAttribution);\n }\n }\n\n if (this._map.style.stylesheet) {\n const stylesheet = this._map.style.stylesheet as StyleSpecification & { owner: string; id: string };\n this.styleOwner = stylesheet.owner;\n this.styleId = stylesheet.id;\n }\n\n const sourceCaches = this._map.style.sourceCaches;\n for (const id in sourceCaches) {\n const sourceCache = sourceCaches[id];\n if (sourceCache.used || sourceCache.usedForTerrain) {\n const source = sourceCache.getSource();\n if (source.attribution && attributions.indexOf(source.attribution) < 0) {\n attributions.push(source.attribution);\n }\n }\n }\n\n // remove any entries that are whitespace\n attributions = attributions.filter(e => String(e).trim());\n\n // remove any entries that are substrings of another entry.\n // first sort by length so that substrings come first\n attributions.sort((a, b) => a.length - b.length);\n attributions = attributions.filter((attrib, i) => {\n for (let j = i + 1; j < attributions.length; j++) {\n if (attributions[j].indexOf(attrib) >= 0) { return false; }\n }\n return true;\n });\n\n // check if attribution string is different to minimize DOM changes\n const attribHTML = attributions.join(' | ');\n if (attribHTML === this._attribHTML) return;\n\n this._attribHTML = attribHTML;\n\n if (attributions.length) {\n this._innerContainer.innerHTML = attribHTML;\n this._container.classList.remove('maplibregl-attrib-empty');\n } else {\n this._container.classList.add('maplibregl-attrib-empty');\n }\n this._updateCompact();\n // remove old DOM node from _editLink\n this._editLink = null;\n }\n\n _updateCompact = () => {\n if (this._map.getCanvasContainer().offsetWidth <= 640 || this._compact) {\n if (this._compact === false) {\n this._container.setAttribute('open', '');\n } else if (!this._container.classList.contains('maplibregl-compact') && !this._container.classList.contains('maplibregl-attrib-empty')) {\n this._container.setAttribute('open', '');\n this._container.classList.add('maplibregl-compact', 'maplibregl-compact-show');\n }\n } else {\n this._container.setAttribute('open', '');\n if (this._container.classList.contains('maplibregl-compact')) {\n this._container.classList.remove('maplibregl-compact', 'maplibregl-compact-show');\n }\n }\n };\n\n _updateCompactMinimize = () => {\n if (this._container.classList.contains('maplibregl-compact')) {\n if (this._container.classList.contains('maplibregl-compact-show')) {\n this._container.classList.remove('maplibregl-compact-show');\n }\n }\n };\n}\n","import {DOM} from '../../util/dom';\n\nimport type {Map} from '../map';\nimport type {ControlPosition, IControl} from './control';\n\n/**\n * The {@link LogoControl} options object\n */\ntype LogoOptions = {\n /**\n * If `true`, force a compact logo.\n * If `false`, force the full logo. The default is a responsive logo that collapses when the map is less than 640 pixels wide.\n */\n compact?: boolean;\n};\n\n/**\n * A `LogoControl` is a control that adds the watermark.\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * map.addControl(new maplibregl.LogoControl({compact: false}));\n * ```\n **/\nexport class LogoControl implements IControl {\n options: LogoOptions;\n _map: Map;\n _compact: boolean;\n _container: HTMLElement;\n\n constructor(options: LogoOptions = {}) {\n this.options = options;\n }\n\n getDefaultPosition(): ControlPosition {\n return 'bottom-left';\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._compact = this.options && this.options.compact;\n this._container = DOM.create('div', 'maplibregl-ctrl');\n const anchor = DOM.create('a', 'maplibregl-ctrl-logo');\n anchor.target = '_blank';\n anchor.rel = 'noopener nofollow';\n anchor.href = 'https://maplibre.org/';\n anchor.setAttribute('aria-label', this._map._getUIString('LogoControl.Title'));\n anchor.setAttribute('rel', 'noopener nofollow');\n this._container.appendChild(anchor);\n this._container.style.display = 'block';\n\n this._map.on('resize', this._updateCompact);\n this._updateCompact();\n\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._container);\n this._map.off('resize', this._updateCompact);\n this._map = undefined;\n this._compact = undefined;\n }\n\n _updateCompact = () => {\n const containerChildren = this._container.children;\n if (containerChildren.length) {\n const anchor = containerChildren[0];\n if (this._map.getCanvasContainer().offsetWidth <= 640 || this._compact) {\n if (this._compact !== false) {\n anchor.classList.add('maplibregl-compact');\n }\n } else {\n anchor.classList.remove('maplibregl-compact');\n }\n }\n };\n\n}\n","export type TaskID = number; // can't mark opaque due to https://github.com/flowtype/flow-remove-types/pull/61\n\ntype Task = {\n callback: (timeStamp: number) => void;\n id: TaskID;\n cancelled: boolean;\n};\n\nexport class TaskQueue {\n _queue: Array;\n _id: TaskID;\n _cleared: boolean;\n _currentlyRunning: Array | false;\n\n constructor() {\n this._queue = [];\n this._id = 0;\n this._cleared = false;\n this._currentlyRunning = false;\n }\n\n add(callback: (timeStamp: number) => void): TaskID {\n const id = ++this._id;\n const queue = this._queue;\n queue.push({callback, id, cancelled: false});\n return id;\n }\n\n remove(id: TaskID) {\n const running = this._currentlyRunning;\n const queue = running ? this._queue.concat(running) : this._queue;\n for (const task of queue) {\n if (task.id === id) {\n task.cancelled = true;\n return;\n }\n }\n }\n\n run(timeStamp: number = 0) {\n if (this._currentlyRunning) throw new Error('Attempting to run(), but is already running.');\n const queue = this._currentlyRunning = this._queue;\n\n // Tasks queued by callbacks in the current queue should be executed\n // on the next run, not the current run.\n this._queue = [];\n\n for (const task of queue) {\n if (task.cancelled) continue;\n task.callback(timeStamp);\n if (this._cleared) break;\n }\n\n this._cleared = false;\n this._currentlyRunning = false;\n }\n\n clear() {\n if (this._currentlyRunning) {\n this._cleared = true;\n }\n this._queue = [];\n }\n}\n","export const defaultLocale = {\n 'AttributionControl.ToggleAttribution': 'Toggle attribution',\n 'AttributionControl.MapFeedback': 'Map feedback',\n 'FullscreenControl.Enter': 'Enter fullscreen',\n 'FullscreenControl.Exit': 'Exit fullscreen',\n 'GeolocateControl.FindMyLocation': 'Find my location',\n 'GeolocateControl.LocationNotAvailable': 'Location not available',\n 'LogoControl.Title': 'Mapbox logo',\n 'NavigationControl.ResetBearing': 'Reset bearing to north',\n 'NavigationControl.ZoomIn': 'Zoom in',\n 'NavigationControl.ZoomOut': 'Zoom out',\n 'ScaleControl.Feet': 'ft',\n 'ScaleControl.Meters': 'm',\n 'ScaleControl.Kilometers': 'km',\n 'ScaleControl.Miles': 'mi',\n 'ScaleControl.NauticalMiles': 'nm',\n 'TerrainControl.enableTerrain': 'Enable terrain',\n 'TerrainControl.disableTerrain': 'Disable terrain'\n};\n","import {createLayout} from '../util/struct_array';\n\nexport default createLayout([\n {name: 'a_pos3d', type: 'Int16', components: 3}\n]);\n","import {OverscaledTileID} from './tile_id';\nimport {Tile} from './tile';\nimport {EXTENT} from '../data/extent';\nimport {mat4} from 'gl-matrix';\nimport {Evented} from '../util/evented';\nimport type {Transform} from '../geo/transform';\nimport type {SourceCache} from '../source/source_cache';\nimport {Terrain} from '../render/terrain';\n\n/**\n * @internal\n * This class is a helper for the Terrain-class, it:\n * - loads raster-dem tiles\n * - manages all renderToTexture tiles.\n * - caches previous rendered tiles.\n * - finds all necessary renderToTexture tiles for a OverscaledTileID area\n * - finds the corresponding raster-dem tile for OverscaledTileID\n */\nexport class TerrainSourceCache extends Evented {\n /**\n * source-cache for the raster-dem source.\n */\n sourceCache: SourceCache;\n /**\n * stores all render-to-texture tiles.\n */\n _tiles: {[_: string]: Tile};\n /**\n * contains a list of tileID-keys for the current scene. (only for performance)\n */\n _renderableTilesKeys: Array;\n /**\n * raster-dem-tile for a TileID cache.\n */\n _sourceTileCache: {[_: string]: string};\n /**\n * minimum zoomlevel to render the terrain.\n */\n minzoom: number;\n /**\n * maximum zoomlevel to render the terrain.\n */\n maxzoom: number;\n /**\n * render-to-texture tileSize in scene.\n */\n tileSize: number;\n /**\n * raster-dem tiles will load for performance the actualZoom - deltaZoom zoom-level.\n */\n deltaZoom: number;\n\n constructor(sourceCache: SourceCache) {\n super();\n this.sourceCache = sourceCache;\n this._tiles = {};\n this._renderableTilesKeys = [];\n this._sourceTileCache = {};\n this.minzoom = 0;\n this.maxzoom = 22;\n this.tileSize = 512;\n this.deltaZoom = 1;\n sourceCache.usedForTerrain = true;\n sourceCache.tileSize = this.tileSize * 2 ** this.deltaZoom;\n }\n\n destruct() {\n this.sourceCache.usedForTerrain = false;\n this.sourceCache.tileSize = null;\n }\n\n /**\n * Load Terrain Tiles, create internal render-to-texture tiles, free GPU memory.\n * @param transform - the operation to do\n * @param terrain - the terrain\n */\n update(transform: Transform, terrain: Terrain): void {\n // load raster-dem tiles for the current scene.\n this.sourceCache.update(transform, terrain);\n // create internal render-to-texture tiles for the current scene.\n this._renderableTilesKeys = [];\n const keys = {};\n for (const tileID of transform.coveringTiles({\n tileSize: this.tileSize,\n minzoom: this.minzoom,\n maxzoom: this.maxzoom,\n reparseOverscaled: false,\n terrain\n })) {\n keys[tileID.key] = true;\n this._renderableTilesKeys.push(tileID.key);\n if (!this._tiles[tileID.key]) {\n tileID.posMatrix = new Float64Array(16) as any;\n mat4.ortho(tileID.posMatrix, 0, EXTENT, 0, EXTENT, 0, 1);\n this._tiles[tileID.key] = new Tile(tileID, this.tileSize);\n }\n }\n // free unused tiles\n for (const key in this._tiles) {\n if (!keys[key]) delete this._tiles[key];\n }\n }\n\n /**\n * Free render to texture cache\n * @param tileID - optional, free only corresponding to tileID.\n */\n freeRtt(tileID?: OverscaledTileID) {\n for (const key in this._tiles) {\n const tile = this._tiles[key];\n if (!tileID || tile.tileID.equals(tileID) || tile.tileID.isChildOf(tileID) || tileID.isChildOf(tile.tileID))\n tile.rtt = [];\n }\n }\n\n /**\n * get a list of tiles, which are loaded and should be rendered in the current scene\n * @returns the renderable tiles\n */\n getRenderableTiles(): Array {\n return this._renderableTilesKeys.map(key => this.getTileByID(key));\n }\n\n /**\n * get terrain tile by the TileID key\n * @param id - the tile id\n * @returns the tile\n */\n getTileByID(id: string): Tile {\n return this._tiles[id];\n }\n\n /**\n * Searches for the corresponding current renderable terrain-tiles\n * @param tileID - the tile to look for\n * @returns the tiles that were found\n */\n getTerrainCoords(tileID: OverscaledTileID): Record {\n const coords = {};\n for (const key of this._renderableTilesKeys) {\n const _tileID = this._tiles[key].tileID;\n if (_tileID.canonical.equals(tileID.canonical)) {\n const coord = tileID.clone();\n coord.posMatrix = new Float64Array(16) as any;\n mat4.ortho(coord.posMatrix, 0, EXTENT, 0, EXTENT, 0, 1);\n coords[key] = coord;\n } else if (_tileID.canonical.isChildOf(tileID.canonical)) {\n const coord = tileID.clone();\n coord.posMatrix = new Float64Array(16) as any;\n const dz = _tileID.canonical.z - tileID.canonical.z;\n const dx = _tileID.canonical.x - (_tileID.canonical.x >> dz << dz);\n const dy = _tileID.canonical.y - (_tileID.canonical.y >> dz << dz);\n const size = EXTENT >> dz;\n mat4.ortho(coord.posMatrix, 0, size, 0, size, 0, 1);\n mat4.translate(coord.posMatrix, coord.posMatrix, [-dx * size, -dy * size, 0]);\n coords[key] = coord;\n } else if (tileID.canonical.isChildOf(_tileID.canonical)) {\n const coord = tileID.clone();\n coord.posMatrix = new Float64Array(16) as any;\n const dz = tileID.canonical.z - _tileID.canonical.z;\n const dx = tileID.canonical.x - (tileID.canonical.x >> dz << dz);\n const dy = tileID.canonical.y - (tileID.canonical.y >> dz << dz);\n const size = EXTENT >> dz;\n mat4.ortho(coord.posMatrix, 0, EXTENT, 0, EXTENT, 0, 1);\n mat4.translate(coord.posMatrix, coord.posMatrix, [dx * size, dy * size, 0]);\n mat4.scale(coord.posMatrix, coord.posMatrix, [1 / (2 ** dz), 1 / (2 ** dz), 0]);\n coords[key] = coord;\n }\n }\n return coords;\n }\n\n /**\n * find the covering raster-dem tile\n * @param tileID - the tile to look for\n * @param searchForDEM - Optinal parameter to search for (parent) souretiles with loaded dem.\n * @returns the tile\n */\n getSourceTile(tileID: OverscaledTileID, searchForDEM?: boolean): Tile {\n const source = this.sourceCache._source;\n let z = tileID.overscaledZ - this.deltaZoom;\n if (z > source.maxzoom) z = source.maxzoom;\n if (z < source.minzoom) return null;\n // cache for tileID to terrain-tileID\n if (!this._sourceTileCache[tileID.key])\n this._sourceTileCache[tileID.key] = tileID.scaledTo(z).key;\n let tile = this.sourceCache.getTileByID(this._sourceTileCache[tileID.key]);\n // during tile-loading phase look if parent tiles (with loaded dem) are available.\n if (!(tile && tile.dem) && searchForDEM)\n while (z >= source.minzoom && !(tile && tile.dem))\n tile = this.sourceCache.getTileByID(tileID.scaledTo(z--).key);\n return tile;\n }\n\n /**\n * get a list of tiles, loaded after a spezific time. This is used to update depth & coords framebuffers.\n * @param time - the time\n * @returns the relevant tiles\n */\n tilesAfterTime(time = Date.now()): Array {\n return Object.values(this._tiles).filter(t => t.timeAdded >= time);\n }\n}\n","\nimport {Tile} from '../source/tile';\nimport {mat4, vec2} from 'gl-matrix';\nimport {OverscaledTileID} from '../source/tile_id';\nimport {RGBAImage} from '../util/image';\nimport {warnOnce} from '../util/util';\nimport {Pos3dArray, TriangleIndexArray} from '../data/array_types.g';\nimport pos3dAttributes from '../data/pos3d_attributes';\nimport {SegmentVector} from '../data/segment';\nimport {VertexBuffer} from '../gl/vertex_buffer';\nimport {IndexBuffer} from '../gl/index_buffer';\nimport {Painter} from './painter';\nimport {Texture} from '../render/texture';\nimport type {Framebuffer} from '../gl/framebuffer';\nimport Point from '@mapbox/point-geometry';\nimport {MercatorCoordinate, lngFromMercatorX, mercatorXfromLng} from '../geo/mercator_coordinate';\nimport {TerrainSourceCache} from '../source/terrain_source_cache';\nimport {SourceCache} from '../source/source_cache';\nimport {EXTENT} from '../data/extent';\nimport type {TerrainSpecification} from '@maplibre/maplibre-gl-style-spec';\nimport {LngLat, earthRadius} from '../geo/lng_lat';\n\n/**\n * @internal\n * A terrain GPU related object\n */\nexport type TerrainData = {\n 'u_depth': number;\n 'u_terrain': number;\n 'u_terrain_dim': number;\n 'u_terrain_matrix': mat4;\n 'u_terrain_unpack': number[];\n 'u_terrain_exaggeration': number;\n texture: WebGLTexture;\n depthTexture: WebGLTexture;\n tile: Tile;\n}\n\n/**\n * @internal\n * A terrain mesh object\n */\nexport type TerrainMesh = {\n indexBuffer: IndexBuffer;\n vertexBuffer: VertexBuffer;\n segments: SegmentVector;\n}\n\n/**\n * @internal\n * This is the main class which handles most of the 3D Terrain logic. It has the following topics:\n * 1) loads raster-dem tiles via the internal sourceCache this.sourceCache\n * 2) creates a depth-framebuffer, which is used to calculate the visibility of coordinates\n * 3) creates a coords-framebuffer, which is used the get to tile-coordinate for a screen-pixel\n * 4) stores all render-to-texture tiles in the this.sourceCache._tiles\n * 5) calculates the elevation for a specific tile-coordinate\n * 6) creates a terrain-mesh\n *\n * A note about the GPU resource-usage:\n * Framebuffers:\n * - one for the depth & coords framebuffer with the size of the map-div.\n * - one for rendering a tile to texture with the size of tileSize (= 512x512).\n * Textures:\n * - one texture for an empty raster-dem tile with size 1x1\n * - one texture for an empty depth-buffer, when terrain is disabled with size 1x1\n * - one texture for an each loaded raster-dem with size of the source.tileSize\n * - one texture for the coords-framebuffer with the size of the map-div.\n * - one texture for the depth-framebuffer with the size of the map-div.\n * - one texture for the encoded tile-coords with the size 2*tileSize (=1024x1024)\n * - finally for each render-to-texture tile (= this._tiles) a set of textures\n * for each render stack (The stack-concept is documented in painter.ts).\n * Normally there exists 1-3 Textures per tile, depending on the stylesheet.\n * Each Textures has the size 2*tileSize (= 1024x1024). Also there exists a\n * cache of the last 150 newest rendered tiles.\n *\n */\nexport class Terrain {\n /**\n * The style this terrain crresponds to\n */\n painter: Painter;\n /**\n * the sourcecache this terrain is based on\n */\n sourceCache: TerrainSourceCache;\n /**\n * the TerrainSpecification object passed to this instance\n */\n options: TerrainSpecification;\n /**\n * define the meshSize per tile.\n */\n meshSize: number;\n /**\n * multiplicator for the elevation. Used to make terrain more \"extreme\".\n */\n exaggeration: number;\n /**\n * to not see pixels in the render-to-texture tiles it is good to render them bigger\n * this number is the multiplicator (must be a power of 2) for the current tileSize.\n * So to get good results with not too much memory footprint a value of 2 should be fine.\n */\n qualityFactor: number;\n /**\n * holds the framebuffer object in size of the screen to render the coords & depth into a texture.\n */\n _fbo: Framebuffer;\n _fboCoordsTexture: Texture;\n _fboDepthTexture: Texture;\n _emptyDepthTexture: Texture;\n /**\n * GL Objects for the terrain-mesh\n * The mesh is a regular mesh, which has the advantage that it can be reused for all tiles.\n */\n _mesh: TerrainMesh;\n /**\n * coords index contains a list of tileID.keys. This index is used to identify\n * the tile via the alpha-cannel in the coords-texture.\n * As the alpha-channel has 1 Byte a max of 255 tiles can rendered without an error.\n */\n coordsIndex: Array;\n /**\n * tile-coords encoded in the rgb channel, _coordsIndex is in the alpha-channel.\n */\n _coordsTexture: Texture;\n /**\n * accuracy of the coords. 2 * tileSize should be enoughth.\n */\n _coordsTextureSize: number;\n /**\n * variables for an empty dem texture, which is used while the raster-dem tile is loading.\n */\n _emptyDemUnpack: number[];\n _emptyDemTexture: Texture;\n _emptyDemMatrix: mat4;\n /**\n * as of overzooming of raster-dem tiles in high zoomlevels, this cache contains\n * matrices to transform from vector-tile coords to raster-dem-tile coords.\n */\n _demMatrixCache: {[_: string]: { matrix: mat4; coord: OverscaledTileID }};\n\n constructor(painter: Painter, sourceCache: SourceCache, options: TerrainSpecification) {\n this.painter = painter;\n this.sourceCache = new TerrainSourceCache(sourceCache);\n this.options = options;\n this.exaggeration = typeof options.exaggeration === 'number' ? options.exaggeration : 1.0;\n this.qualityFactor = 2;\n this.meshSize = 128;\n this._demMatrixCache = {};\n this.coordsIndex = [];\n this._coordsTextureSize = 1024;\n }\n\n /**\n * get the elevation-value from original dem-data for a given tile-coordinate\n * @param tileID - the tile to get elevation for\n * @param x - between 0 .. EXTENT\n * @param y - between 0 .. EXTENT\n * @param extent - optional, default 8192\n * @returns the elevation\n */\n getDEMElevation(tileID: OverscaledTileID, x: number, y: number, extent: number = EXTENT): number {\n if (!(x >= 0 && x < extent && y >= 0 && y < extent)) return 0;\n const terrain = this.getTerrainData(tileID);\n const dem = terrain.tile?.dem;\n if (!dem)\n return 0;\n\n const pos = vec2.transformMat4([] as any, [x / extent * EXTENT, y / extent * EXTENT], terrain.u_terrain_matrix);\n const coord = [pos[0] * dem.dim, pos[1] * dem.dim];\n\n // bilinear interpolation\n const cx = Math.floor(coord[0]),\n cy = Math.floor(coord[1]),\n tx = coord[0] - cx,\n ty = coord[1] - cy;\n return (\n dem.get(cx, cy) * (1 - tx) * (1 - ty) +\n dem.get(cx + 1, cy) * (tx) * (1 - ty) +\n dem.get(cx, cy + 1) * (1 - tx) * (ty) +\n dem.get(cx + 1, cy + 1) * (tx) * (ty)\n );\n }\n\n /**\n * Get the elevation for given {@link LngLat} in respect of exaggeration.\n * @param lnglat - the location\n * @param zoom - the zoom\n * @returns the elevation\n */\n getElevationForLngLatZoom(lnglat: LngLat, zoom: number) {\n const {tileID, mercatorX, mercatorY} = this._getOverscaledTileIDFromLngLatZoom(lnglat, zoom);\n return this.getElevation(tileID, mercatorX % EXTENT, mercatorY % EXTENT, EXTENT);\n }\n\n /**\n * Get the elevation for given coordinate in respect of exaggeration.\n * @param tileID - the tile id\n * @param x - between 0 .. EXTENT\n * @param y - between 0 .. EXTENT\n * @param extent - optional, default 8192\n * @returns the elevation\n */\n getElevation(tileID: OverscaledTileID, x: number, y: number, extent: number = EXTENT): number {\n return this.getDEMElevation(tileID, x, y, extent) * this.exaggeration;\n }\n\n /**\n * returns a Terrain Object for a tile. Unless the tile corresponds to data (e.g. tile is loading), return a flat dem object\n * @param tileID - the tile to get the terrain for\n * @returns the terrain data to use in the program\n */\n getTerrainData(tileID: OverscaledTileID): TerrainData {\n // create empty DEM Objects, which will used while raster-dem tiles are loading.\n // creates an empty depth-buffer texture which is needed, during the initialization process of the 3d mesh..\n if (!this._emptyDemTexture) {\n const context = this.painter.context;\n const image = new RGBAImage({width: 1, height: 1}, new Uint8Array(1 * 4));\n this._emptyDepthTexture = new Texture(context, image, context.gl.RGBA, {premultiply: false});\n this._emptyDemUnpack = [0, 0, 0, 0];\n this._emptyDemTexture = new Texture(context, new RGBAImage({width: 1, height: 1}), context.gl.RGBA, {premultiply: false});\n this._emptyDemTexture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE);\n this._emptyDemMatrix = mat4.identity([] as any);\n }\n // find covering dem tile and prepare demTexture\n const sourceTile = this.sourceCache.getSourceTile(tileID, true);\n if (sourceTile && sourceTile.dem && (!sourceTile.demTexture || sourceTile.needsTerrainPrepare)) {\n const context = this.painter.context;\n sourceTile.demTexture = this.painter.getTileTexture(sourceTile.dem.stride);\n if (sourceTile.demTexture) sourceTile.demTexture.update(sourceTile.dem.getPixels(), {premultiply: false});\n else sourceTile.demTexture = new Texture(context, sourceTile.dem.getPixels(), context.gl.RGBA, {premultiply: false});\n sourceTile.demTexture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE);\n sourceTile.needsTerrainPrepare = false;\n }\n // create matrix for lookup in dem data\n const matrixKey = sourceTile && (sourceTile + sourceTile.tileID.key) + tileID.key;\n if (matrixKey && !this._demMatrixCache[matrixKey]) {\n const maxzoom = this.sourceCache.sourceCache._source.maxzoom;\n let dz = tileID.canonical.z - sourceTile.tileID.canonical.z;\n if (tileID.overscaledZ > tileID.canonical.z) {\n if (tileID.canonical.z >= maxzoom) dz = tileID.canonical.z - maxzoom;\n else warnOnce('cannot calculate elevation if elevation maxzoom > source.maxzoom');\n }\n const dx = tileID.canonical.x - (tileID.canonical.x >> dz << dz);\n const dy = tileID.canonical.y - (tileID.canonical.y >> dz << dz);\n const demMatrix = mat4.fromScaling(new Float64Array(16) as any, [1 / (EXTENT << dz), 1 / (EXTENT << dz), 0]);\n mat4.translate(demMatrix, demMatrix, [dx * EXTENT, dy * EXTENT, 0]);\n this._demMatrixCache[tileID.key] = {matrix: demMatrix, coord: tileID};\n }\n // return uniform values & textures\n return {\n 'u_depth': 2,\n 'u_terrain': 3,\n 'u_terrain_dim': sourceTile && sourceTile.dem && sourceTile.dem.dim || 1,\n 'u_terrain_matrix': matrixKey ? this._demMatrixCache[tileID.key].matrix : this._emptyDemMatrix,\n 'u_terrain_unpack': sourceTile && sourceTile.dem && sourceTile.dem.getUnpackVector() || this._emptyDemUnpack,\n 'u_terrain_exaggeration': this.exaggeration,\n texture: (sourceTile && sourceTile.demTexture || this._emptyDemTexture).texture,\n depthTexture: (this._fboDepthTexture || this._emptyDepthTexture).texture,\n tile: sourceTile\n };\n }\n\n /**\n * get a framebuffer as big as the map-div, which will be used to render depth & coords into a texture\n * @param texture - the texture\n * @returns the frame buffer\n */\n getFramebuffer(texture: string): Framebuffer {\n const painter = this.painter;\n const width = painter.width / devicePixelRatio;\n const height = painter.height / devicePixelRatio;\n if (this._fbo && (this._fbo.width !== width || this._fbo.height !== height)) {\n this._fbo.destroy();\n this._fboCoordsTexture.destroy();\n this._fboDepthTexture.destroy();\n delete this._fbo;\n delete this._fboDepthTexture;\n delete this._fboCoordsTexture;\n }\n if (!this._fboCoordsTexture) {\n this._fboCoordsTexture = new Texture(painter.context, {width, height, data: null}, painter.context.gl.RGBA, {premultiply: false});\n this._fboCoordsTexture.bind(painter.context.gl.NEAREST, painter.context.gl.CLAMP_TO_EDGE);\n }\n if (!this._fboDepthTexture) {\n this._fboDepthTexture = new Texture(painter.context, {width, height, data: null}, painter.context.gl.RGBA, {premultiply: false});\n this._fboDepthTexture.bind(painter.context.gl.NEAREST, painter.context.gl.CLAMP_TO_EDGE);\n }\n if (!this._fbo) {\n this._fbo = painter.context.createFramebuffer(width, height, true, false);\n this._fbo.depthAttachment.set(painter.context.createRenderbuffer(painter.context.gl.DEPTH_COMPONENT16, width, height));\n }\n this._fbo.colorAttachment.set(texture === 'coords' ? this._fboCoordsTexture.texture : this._fboDepthTexture.texture);\n return this._fbo;\n }\n\n /**\n * create coords texture, needed to grab coordinates from canvas\n * encode coords coordinate into 4 bytes:\n * - 8 lower bits for x\n * - 8 lower bits for y\n * - 4 higher bits for x\n * - 4 higher bits for y\n * - 8 bits for coordsIndex (1 .. 255) (= number of terraintile), is later setted in draw_terrain uniform value\n * @returns the texture\n */\n getCoordsTexture(): Texture {\n const context = this.painter.context;\n if (this._coordsTexture) return this._coordsTexture;\n const data = new Uint8Array(this._coordsTextureSize * this._coordsTextureSize * 4);\n for (let y = 0, i = 0; y < this._coordsTextureSize; y++) for (let x = 0; x < this._coordsTextureSize; x++, i += 4) {\n data[i + 0] = x & 255;\n data[i + 1] = y & 255;\n data[i + 2] = ((x >> 8) << 4) | (y >> 8);\n data[i + 3] = 0;\n }\n const image = new RGBAImage({width: this._coordsTextureSize, height: this._coordsTextureSize}, new Uint8Array(data.buffer));\n const texture = new Texture(context, image, context.gl.RGBA, {premultiply: false});\n texture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE);\n this._coordsTexture = texture;\n return texture;\n }\n\n /**\n * Reads a pixel from the coords-framebuffer and translate this to mercator.\n * @param p - Screen-Coordinate\n * @returns mercator coordinate for a screen pixel\n */\n pointCoordinate(p: Point): MercatorCoordinate {\n const rgba = new Uint8Array(4);\n const context = this.painter.context, gl = context.gl;\n // grab coordinate pixel from coordinates framebuffer\n context.bindFramebuffer.set(this.getFramebuffer('coords').framebuffer);\n gl.readPixels(p.x, this.painter.height / devicePixelRatio - p.y - 1, 1, 1, gl.RGBA, gl.UNSIGNED_BYTE, rgba);\n context.bindFramebuffer.set(null);\n // decode coordinates (encoding see getCoordsTexture)\n const x = rgba[0] + ((rgba[2] >> 4) << 8);\n const y = rgba[1] + ((rgba[2] & 15) << 8);\n const tileID = this.coordsIndex[255 - rgba[3]];\n const tile = tileID && this.sourceCache.getTileByID(tileID);\n if (!tile) return null;\n const coordsSize = this._coordsTextureSize;\n const worldSize = (1 << tile.tileID.canonical.z) * coordsSize;\n const mercatorX = (tile.tileID.canonical.x * coordsSize + x) / worldSize;\n return new MercatorCoordinate(\n this._allowMercatorOverflow(p, mercatorX),\n (tile.tileID.canonical.y * coordsSize + y) / worldSize,\n this.getElevation(tile.tileID, x, y, coordsSize)\n );\n }\n\n /**\n * create a regular mesh which will be used by all terrain-tiles\n * @returns the created regular mesh\n */\n getTerrainMesh(): TerrainMesh {\n if (this._mesh) return this._mesh;\n const context = this.painter.context;\n const vertexArray = new Pos3dArray();\n const indexArray = new TriangleIndexArray();\n const meshSize = this.meshSize;\n const delta = EXTENT / meshSize;\n const meshSize2 = meshSize * meshSize;\n for (let y = 0; y <= meshSize; y++) for (let x = 0; x <= meshSize; x++)\n vertexArray.emplaceBack(x * delta, y * delta, 0);\n for (let y = 0; y < meshSize2; y += meshSize + 1) for (let x = 0; x < meshSize; x++) {\n indexArray.emplaceBack(x + y, meshSize + x + y + 1, meshSize + x + y + 2);\n indexArray.emplaceBack(x + y, meshSize + x + y + 2, x + y + 1);\n }\n // add an extra frame around the mesh to avoid stiching on tile boundaries with different zoomlevels\n // first code-block is for top-bottom frame and second for left-right frame\n const offsetTop = vertexArray.length, offsetBottom = offsetTop + (meshSize + 1) * 2;\n for (const y of [0, 1]) for (let x = 0; x <= meshSize; x++) for (const z of [0, 1])\n vertexArray.emplaceBack(x * delta, y * EXTENT, z);\n for (let x = 0; x < meshSize * 2; x += 2) {\n indexArray.emplaceBack(offsetBottom + x, offsetBottom + x + 1, offsetBottom + x + 3);\n indexArray.emplaceBack(offsetBottom + x, offsetBottom + x + 3, offsetBottom + x + 2);\n indexArray.emplaceBack(offsetTop + x, offsetTop + x + 3, offsetTop + x + 1);\n indexArray.emplaceBack(offsetTop + x, offsetTop + x + 2, offsetTop + x + 3);\n }\n const offsetLeft = vertexArray.length, offsetRight = offsetLeft + (meshSize + 1) * 2;\n for (const x of [0, 1]) for (let y = 0; y <= meshSize; y++) for (const z of [0, 1])\n vertexArray.emplaceBack(x * EXTENT, y * delta, z);\n for (let y = 0; y < meshSize * 2; y += 2) {\n indexArray.emplaceBack(offsetLeft + y, offsetLeft + y + 1, offsetLeft + y + 3);\n indexArray.emplaceBack(offsetLeft + y, offsetLeft + y + 3, offsetLeft + y + 2);\n indexArray.emplaceBack(offsetRight + y, offsetRight + y + 3, offsetRight + y + 1);\n indexArray.emplaceBack(offsetRight + y, offsetRight + y + 2, offsetRight + y + 3);\n }\n this._mesh = {\n indexBuffer: context.createIndexBuffer(indexArray),\n vertexBuffer: context.createVertexBuffer(vertexArray, pos3dAttributes.members),\n segments: SegmentVector.simpleSegment(0, 0, vertexArray.length, indexArray.length)\n };\n return this._mesh;\n }\n\n /**\n * Calculates a height of the frame around the terrain-mesh to avoid stiching between\n * tile boundaries in different zoomlevels.\n * @param zoom - current zoomlevel\n * @returns the elevation delta in meters\n */\n getMeshFrameDelta(zoom: number): number {\n // divide by 5 is evaluated by trial & error to get a frame in the right height\n return 2 * Math.PI * earthRadius / Math.pow(2, zoom) / 5;\n }\n\n getMinTileElevationForLngLatZoom(lnglat: LngLat, zoom: number) {\n const {tileID} = this._getOverscaledTileIDFromLngLatZoom(lnglat, zoom);\n return this.getMinMaxElevation(tileID).minElevation ?? 0;\n }\n\n /**\n * Get the minimum and maximum elevation contained in a tile. This includes any\n * exaggeration included in the terrain.\n *\n * @param tileID - ID of the tile to be used as a source for the min/max elevation\n * @returns the minimum and maximum elevation found in the tile, including the terrain's\n * exaggeration\n */\n getMinMaxElevation(tileID: OverscaledTileID): {minElevation: number | null; maxElevation: number | null} {\n const tile = this.getTerrainData(tileID).tile;\n const minMax = {minElevation: null, maxElevation: null};\n if (tile && tile.dem) {\n minMax.minElevation = tile.dem.min * this.exaggeration;\n minMax.maxElevation = tile.dem.max * this.exaggeration;\n }\n return minMax;\n }\n\n _getOverscaledTileIDFromLngLatZoom(lnglat: LngLat, zoom: number): { tileID: OverscaledTileID; mercatorX: number; mercatorY: number} {\n const mercatorCoordinate = MercatorCoordinate.fromLngLat(lnglat.wrap());\n const worldSize = (1 << zoom) * EXTENT;\n const mercatorX = mercatorCoordinate.x * worldSize;\n const mercatorY = mercatorCoordinate.y * worldSize;\n const tileX = Math.floor(mercatorX / EXTENT), tileY = Math.floor(mercatorY / EXTENT);\n const tileID = new OverscaledTileID(zoom, 0, zoom, tileX, tileY);\n return {\n tileID,\n mercatorX,\n mercatorY\n };\n }\n\n _allowMercatorOverflow(p: Point, mercatorX: number): number {\n const inLeftHalf = p.x < (this.painter.width / 2);\n let lng = lngFromMercatorX(mercatorX);\n const centerLng = this.painter.transform.center.lng;\n if (\n (inLeftHalf && Math.sign(lng) > 0 && Math.sign(centerLng) < 0) ||\n (!inLeftHalf && Math.sign(lng) < 0 && Math.sign(centerLng) > 0)\n ) {\n lng = 360 * Math.sign(centerLng) + lng;\n return mercatorXfromLng(lng);\n }\n return mercatorX;\n }\n}\n","import {Texture} from '../render/texture';\nimport {Context} from './context';\nimport {Framebuffer} from './framebuffer';\n\nexport type PoolObject = {\n id: number;\n fbo: Framebuffer;\n texture: Texture;\n stamp: number;\n inUse: boolean;\n};\n/**\n * @internal\n * `RenderPool` is a resource pool for textures and framebuffers\n */\nexport class RenderPool {\n private _objects: Array;\n /**\n * An index array of recently used pool objects.\n * Items that are used recently are last in the array\n */\n private _recentlyUsed: Array;\n private _stamp: number;\n\n constructor(\n private readonly _context: Context,\n private readonly _size: number,\n private readonly _tileSize: number) {\n this._objects = [];\n this._recentlyUsed = [];\n this._stamp = 0;\n }\n\n public destruct() {\n for (const obj of this._objects) {\n obj.texture.destroy();\n obj.fbo.destroy();\n }\n }\n\n private _createObject(id: number): PoolObject {\n const fbo = this._context.createFramebuffer(this._tileSize, this._tileSize, true, true);\n const texture = new Texture(this._context, {width: this._tileSize, height: this._tileSize, data: null}, this._context.gl.RGBA);\n texture.bind(this._context.gl.LINEAR, this._context.gl.CLAMP_TO_EDGE);\n fbo.depthAttachment.set(this._context.createRenderbuffer(this._context.gl.DEPTH_STENCIL, this._tileSize, this._tileSize));\n fbo.colorAttachment.set(texture.texture);\n return {id, fbo, texture, stamp: -1, inUse: false};\n }\n\n public getObjectForId(id: number): PoolObject {\n return this._objects[id];\n }\n\n public useObject(obj: PoolObject) {\n obj.inUse = true;\n this._recentlyUsed = this._recentlyUsed.filter(id => obj.id !== id);\n this._recentlyUsed.push(obj.id);\n }\n\n public stampObject(obj: PoolObject) {\n obj.stamp = ++this._stamp;\n }\n\n public getOrCreateFreeObject(): PoolObject {\n // check for free existing object\n for (const id of this._recentlyUsed) {\n if (!this._objects[id].inUse)\n return this._objects[id];\n }\n if (this._objects.length >= this._size)\n throw new Error('No free RenderPool available, call freeAllObjects() required!');\n // create new object\n const obj = this._createObject(this._objects.length);\n this._objects.push(obj);\n return obj;\n }\n\n public freeObject(obj: PoolObject) {\n obj.inUse = false;\n }\n\n public freeAllObjects() {\n for (const obj of this._objects)\n this.freeObject(obj);\n }\n\n public isFull(): boolean {\n if (this._objects.length < this._size) {\n return false;\n }\n return this._objects.some(o => !o.inUse) === false;\n }\n}\n","import {Painter} from './painter';\nimport {Tile} from '../source/tile';\nimport {Color} from '@maplibre/maplibre-gl-style-spec';\nimport {OverscaledTileID} from '../source/tile_id';\nimport {drawTerrain} from './draw_terrain';\nimport {Style} from '../style/style';\nimport {Terrain} from './terrain';\nimport {RenderPool} from '../gl/render_pool';\nimport {Texture} from './texture';\nimport type {StyleLayer} from '../style/style_layer';\n\n/**\n * lookup table which layers should rendered to texture\n */\nconst LAYERS: { [keyof in StyleLayer['type']]?: boolean } = {\n background: true,\n fill: true,\n line: true,\n raster: true,\n hillshade: true\n};\n\n/**\n * @internal\n * A helper class to help define what should be rendered to texture and how\n */\nexport class RenderToTexture {\n painter: Painter;\n terrain: Terrain;\n pool: RenderPool;\n /**\n * coordsDescendingInv contains a list of all tiles which should be rendered for one render-to-texture tile\n * e.g. render 4 raster-tiles with size 256px to the 512px render-to-texture tile\n */\n _coordsDescendingInv: {[_: string]: {[_:string]: Array}};\n /**\n * create a string representation of all to tiles rendered to render-to-texture tiles\n * this string representation is used to check if tile should be re-rendered.\n */\n _coordsDescendingInvStr: {[_: string]: {[_:string]: string}};\n /**\n * store for render-stacks\n * a render stack is a set of layers which should be rendered into one texture\n * every stylesheet can have multiple stacks. A new stack is created if layers which should\n * not rendered to texture sit inbetween layers which should rendered to texture. e.g. hillshading or symbols\n */\n _stacks: Array>;\n /**\n * remember the previous processed layer to check if a new stack is needed\n */\n _prevType: string;\n /**\n * a list of tiles that can potentially rendered\n */\n _renderableTiles: Array;\n /**\n * a list of tiles that should be rendered to screen in the next render-call\n */\n _rttTiles: Array;\n /**\n * a list of all layer-ids which should be rendered\n */\n _renderableLayerIds: Array;\n\n constructor(painter: Painter, terrain: Terrain) {\n this.painter = painter;\n this.terrain = terrain;\n this.pool = new RenderPool(painter.context, 30, terrain.sourceCache.tileSize * terrain.qualityFactor);\n }\n\n destruct() {\n this.pool.destruct();\n }\n\n getTexture(tile: Tile): Texture {\n return this.pool.getObjectForId(tile.rtt[this._stacks.length - 1].id).texture;\n }\n\n prepareForRender(style: Style, zoom: number) {\n this._stacks = [];\n this._prevType = null;\n this._rttTiles = [];\n this._renderableTiles = this.terrain.sourceCache.getRenderableTiles();\n this._renderableLayerIds = style._order.filter(id => !style._layers[id].isHidden(zoom));\n\n this._coordsDescendingInv = {};\n for (const id in style.sourceCaches) {\n this._coordsDescendingInv[id] = {};\n const tileIDs = style.sourceCaches[id].getVisibleCoordinates();\n for (const tileID of tileIDs) {\n const keys = this.terrain.sourceCache.getTerrainCoords(tileID);\n for (const key in keys) {\n if (!this._coordsDescendingInv[id][key]) this._coordsDescendingInv[id][key] = [];\n this._coordsDescendingInv[id][key].push(keys[key]);\n }\n }\n }\n\n this._coordsDescendingInvStr = {};\n for (const id of style._order) {\n const layer = style._layers[id], source = layer.source;\n if (LAYERS[layer.type]) {\n if (!this._coordsDescendingInvStr[source]) {\n this._coordsDescendingInvStr[source] = {};\n for (const key in this._coordsDescendingInv[source])\n this._coordsDescendingInvStr[source][key] = this._coordsDescendingInv[source][key].map(c => c.key).sort().join();\n }\n }\n }\n\n // check tiles to render\n for (const tile of this._renderableTiles) {\n for (const source in this._coordsDescendingInvStr) {\n // rerender if there are more coords to render than in the last rendering\n const coords = this._coordsDescendingInvStr[source][tile.tileID.key];\n if (coords && coords !== tile.rttCoords[source]) tile.rtt = [];\n }\n }\n }\n\n /**\n * due that switching textures is relatively slow, the render\n * layer-by-layer context is not practicable. To bypass this problem\n * this lines of code stack all layers and later render all at once.\n * Because of the stylesheet possibility to mixing render-to-texture layers\n * and 'live'-layers (f.e. symbols) it is necessary to create more stacks. For example\n * a symbol-layer is in between of fill-layers.\n * @param layer - the layer to render\n * @returns if true layer is rendered to texture, otherwise false\n */\n renderLayer(layer: StyleLayer): boolean {\n if (layer.isHidden(this.painter.transform.zoom)) return false;\n\n const type = layer.type;\n const painter = this.painter;\n const isLastLayer = this._renderableLayerIds[this._renderableLayerIds.length - 1] === layer.id;\n\n // remember background, fill, line & raster layer to render into a stack\n if (LAYERS[type]) {\n // create a new stack if previous layer was not rendered to texture (f.e. symbols)\n if (!this._prevType || !LAYERS[this._prevType]) this._stacks.push([]);\n // push current render-to-texture layer to render-stack\n this._prevType = type;\n this._stacks[this._stacks.length - 1].push(layer.id);\n // rendering is done later, all in once\n if (!isLastLayer) return true;\n }\n\n // in case a stack is finished render all collected stack-layers into a texture\n if (LAYERS[this._prevType] || (LAYERS[type] && isLastLayer)) {\n this._prevType = type;\n const stack = this._stacks.length - 1, layers = this._stacks[stack] || [];\n for (const tile of this._renderableTiles) {\n // if render pool is full draw current tiles to screen and free pool\n if (this.pool.isFull()) {\n drawTerrain(this.painter, this.terrain, this._rttTiles);\n this._rttTiles = [];\n this.pool.freeAllObjects();\n }\n this._rttTiles.push(tile);\n // check for cached PoolObject\n if (tile.rtt[stack]) {\n const obj = this.pool.getObjectForId(tile.rtt[stack].id);\n if (obj.stamp === tile.rtt[stack].stamp) {\n this.pool.useObject(obj);\n continue;\n }\n }\n // get free PoolObject\n const obj = this.pool.getOrCreateFreeObject();\n this.pool.useObject(obj);\n this.pool.stampObject(obj);\n tile.rtt[stack] = {id: obj.id, stamp: obj.stamp};\n // prepare PoolObject for rendering\n painter.context.bindFramebuffer.set(obj.fbo.framebuffer);\n painter.context.clear({color: Color.transparent, stencil: 0});\n painter.currentStencilSource = undefined;\n for (let l = 0; l < layers.length; l++) {\n const layer = painter.style._layers[layers[l]];\n const coords = layer.source ? this._coordsDescendingInv[layer.source][tile.tileID.key] : [tile.tileID];\n painter.context.viewport.set([0, 0, obj.fbo.width, obj.fbo.height]);\n painter._renderTileClippingMasks(layer, coords);\n painter.renderLayer(painter, painter.style.sourceCaches[layer.source], layer, coords);\n if (layer.source) tile.rttCoords[layer.source] = this._coordsDescendingInvStr[layer.source][tile.tileID.key];\n }\n }\n drawTerrain(this.painter, this.terrain, this._rttTiles);\n this._rttTiles = [];\n this.pool.freeAllObjects();\n\n return LAYERS[type];\n }\n\n return false;\n }\n\n}\n","import {extend, warnOnce, uniqueId, isImageBitmap} from '../util/util';\nimport {browser} from '../util/browser';\nimport {DOM} from '../util/dom';\nimport packageJSON from '../../package.json' assert {type: 'json'};\n\nimport {getJSON} from '../util/ajax';\nimport {ImageRequest} from '../util/image_request';\nimport type {GetImageCallback} from '../util/image_request';\n\nimport {RequestManager, ResourceType} from '../util/request_manager';\nimport {Style, StyleSwapOptions} from '../style/style';\nimport {EvaluationParameters} from '../style/evaluation_parameters';\nimport {Painter} from '../render/painter';\nimport {Transform} from '../geo/transform';\nimport {Hash} from './hash';\nimport {HandlerManager} from './handler_manager';\nimport {Camera, CameraOptions, CameraUpdateTransformFunction, FitBoundsOptions} from './camera';\nimport {LngLat} from '../geo/lng_lat';\nimport {LngLatBounds} from '../geo/lng_lat_bounds';\nimport Point from '@mapbox/point-geometry';\nimport {AttributionControl} from './control/attribution_control';\nimport {LogoControl} from './control/logo_control';\n\nimport {RGBAImage} from '../util/image';\nimport {Event, ErrorEvent, Listener} from '../util/evented';\nimport {MapEventType, MapLayerEventType, MapMouseEvent, MapSourceDataEvent, MapStyleDataEvent} from './events';\nimport {TaskQueue} from '../util/task_queue';\nimport {throttle} from '../util/throttle';\nimport {webpSupported} from '../util/webp_supported';\nimport {PerformanceMarkers, PerformanceUtils} from '../util/performance';\nimport {Source, SourceClass} from '../source/source';\nimport {StyleLayer} from '../style/style_layer';\n\nimport type {RequestTransformFunction} from '../util/request_manager';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport type {LngLatBoundsLike} from '../geo/lng_lat_bounds';\nimport type {AddLayerObject, FeatureIdentifier, StyleOptions, StyleSetterOptions} from '../style/style';\nimport type {MapDataEvent} from './events';\nimport type {StyleImage, StyleImageInterface, StyleImageMetadata} from '../style/style_image';\nimport type {PointLike} from './camera';\nimport type {ScrollZoomHandler} from './handler/scroll_zoom';\nimport type {BoxZoomHandler} from './handler/box_zoom';\nimport type {AroundCenterOptions, TwoFingersTouchPitchHandler} from './handler/two_fingers_touch';\nimport type {DragRotateHandler} from './handler/shim/drag_rotate';\nimport {DragPanHandler, DragPanOptions} from './handler/shim/drag_pan';\n\nimport type {KeyboardHandler} from './handler/keyboard';\nimport type {DoubleClickZoomHandler} from './handler/shim/dblclick_zoom';\nimport type {TwoFingersTouchZoomRotateHandler} from './handler/shim/two_fingers_touch';\nimport {defaultLocale} from './default_locale';\nimport type {TaskID} from '../util/task_queue';\nimport type {Cancelable} from '../types/cancelable';\nimport type {\n FilterSpecification,\n StyleSpecification,\n LightSpecification,\n SourceSpecification,\n TerrainSpecification\n} from '@maplibre/maplibre-gl-style-spec';\n\nimport {Callback} from '../types/callback';\nimport type {ControlPosition, IControl} from './control/control';\nimport type {MapGeoJSONFeature} from '../util/vectortile_to_geojson';\nimport {Terrain} from '../render/terrain';\nimport {RenderToTexture} from '../render/render_to_texture';\nimport {config} from '../util/config';\nimport type {QueryRenderedFeaturesOptions, QuerySourceFeatureOptions} from '../source/query_features';\n\nconst version = packageJSON.version;\n\n/**\n * The {@link Map} options object.\n */\nexport type MapOptions = {\n /**\n * If `true`, the map's position (zoom, center latitude, center longitude, bearing, and pitch) will be synced with the hash fragment of the page's URL.\n * For example, `http://path/to/my/page.html#2.59/39.26/53.07/-24.1/60`.\n * An additional string may optionally be provided to indicate a parameter-styled hash,\n * e.g. http://path/to/my/page.html#map=2.59/39.26/53.07/-24.1/60&foo=bar, where foo\n * is a custom parameter and bar is an arbitrary hash distinct from the map hash.\n * @defaultValue false\n */\n hash?: boolean | string;\n /**\n * If `false`, no mouse, touch, or keyboard listeners will be attached to the map, so it will not respond to interaction.\n * @defaultValue true\n */\n interactive?: boolean;\n /**\n * The HTML element in which MapLibre GL JS will render the map, or the element's string `id`. The specified element must have no children.\n */\n container: HTMLElement | string;\n /**\n * The threshold, measured in degrees, that determines when the map's\n * bearing will snap to north. For example, with a `bearingSnap` of 7, if the user rotates\n * the map within 7 degrees of north, the map will automatically snap to exact north.\n * @defaultValue 7\n */\n bearingSnap?: number;\n /**\n * If `true`, an {@link AttributionControl} will be added to the map.\n * @defaultValue true\n */\n attributionControl?: boolean;\n /**\n * Attribution text to show in an {@link AttributionControl}. Only applicable if `options.attributionControl` is `true`.\n */\n customAttribution?: string | Array;\n /**\n * If `true`, the MapLibre logo will be shown.\n * @defaultValue false\n */\n maplibreLogo?: boolean;\n /**\n * A string representing the position of the MapLibre wordmark on the map. Valid options are `top-left`,`top-right`, `bottom-left`, or `bottom-right`.\n * @defaultValue 'bottom-left'\n */\n logoPosition?: ControlPosition;\n /**\n * If `true`, map creation will fail if the performance of MapLibre GL JS would be dramatically worse than expected\n * (i.e. a software renderer would be used).\n * @defaultValue false\n */\n failIfMajorPerformanceCaveat?: boolean;\n /**\n * If `true`, the map's canvas can be exported to a PNG using `map.getCanvas().toDataURL()`. This is `false` by default as a performance optimization.\n * @defaultValue false\n */\n preserveDrawingBuffer?: boolean;\n /**\n * If `true`, the gl context will be created with MSAA antialiasing, which can be useful for antialiasing custom layers. This is `false` by default as a performance optimization.\n */\n antialias?: boolean;\n /**\n * If `false`, the map won't attempt to re-request tiles once they expire per their HTTP `cacheControl`/`expires` headers.\n * @defaultValue true\n */\n refreshExpiredTiles?: boolean;\n /**\n * If set, the map will be constrained to the given bounds.\n */\n maxBounds?: LngLatBoundsLike;\n /**\n * If `true`, the \"scroll to zoom\" interaction is enabled. {@link AroundCenterOptions} are passed as options to {@link ScrollZoomHandler#enable}.\n * @defaultValue true\n */\n scrollZoom?: boolean | AroundCenterOptions;\n /**\n * The minimum zoom level of the map (0-24).\n * @defaultValue 0\n */\n minZoom?: number | null;\n /**\n * The maximum zoom level of the map (0-24).\n * @defaultValue 22\n */\n maxZoom?: number | null;\n /**\n * The minimum pitch of the map (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * @defaultValue 0\n */\n minPitch?: number | null;\n /**\n * The maximum pitch of the map (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * @defaultValue 60\n */\n maxPitch?: number | null;\n /**\n * If `true`, the \"box zoom\" interaction is enabled (see {@link BoxZoomHandler}).\n * @defaultValue true\n */\n boxZoom?: boolean;\n /**\n * If `true`, the \"drag to rotate\" interaction is enabled (see {@link DragRotateHandler}).\n * @defaultValue true\n */\n dragRotate?: boolean;\n /**\n * If `true`, the \"drag to pan\" interaction is enabled. An `Object` value is passed as options to {@link DragPanHandler#enable}.\n * @defaultValue true\n */\n dragPan?: boolean | DragPanOptions;\n /**\n * If `true`, keyboard shortcuts are enabled (see {@link KeyboardHandler}).\n * @defaultValue true\n */\n keyboard?: boolean;\n /**\n * If `true`, the \"double click to zoom\" interaction is enabled (see {@link DoubleClickZoomHandler}).\n * @defaultValue true\n */\n doubleClickZoom?: boolean;\n /**\n * If `true`, the \"pinch to rotate and zoom\" interaction is enabled. An `Object` value is passed as options to {@link TwoFingersTouchZoomRotateHandler#enable}.\n * @defaultValue true\n */\n touchZoomRotate?: boolean | AroundCenterOptions;\n /**\n * If `true`, the \"drag to pitch\" interaction is enabled. An `Object` value is passed as options to {@link TwoFingersTouchPitchHandler#enable}.\n * @defaultValue true\n */\n touchPitch?: boolean | AroundCenterOptions;\n /**\n * If `true` or set to an options object, the map is only accessible on desktop while holding Command/Ctrl and only accessible on mobile with two fingers. Interacting with the map using normal gestures will trigger an informational screen. With this option enabled, \"drag to pitch\" requires a three-finger gesture. Cooperative gestures are disabled when a map enters fullscreen using {@link FullscreenControl}.\n * @defaultValue undefined\n */\n cooperativeGestures?: boolean | GestureOptions;\n /**\n * If `true`, the map will automatically resize when the browser window resizes.\n * @defaultValue true\n */\n trackResize?: boolean;\n /**\n * The initial geographical centerpoint of the map. If `center` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `[0, 0]` Note: MapLibre GL JS uses longitude, latitude coordinate order (as opposed to latitude, longitude) to match GeoJSON.\n * @defaultValue [0, 0]\n */\n center?: LngLatLike;\n /**\n * The initial zoom level of the map. If `zoom` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`.\n * @defaultValue 0\n */\n zoom?: number;\n /**\n * The initial bearing (rotation) of the map, measured in degrees counter-clockwise from north. If `bearing` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`.\n * @defaultValue 0\n */\n bearing?: number;\n /**\n * The initial pitch (tilt) of the map, measured in degrees away from the plane of the screen (0-85). If `pitch` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`. Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * @defaultValue 0\n */\n pitch?: number;\n /**\n * If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`:\n * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire\n * container, there will be blank space beyond 180 and -180 degrees longitude.\n * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the\n * map and the other on the left edge of the map) at every zoom level.\n * @defaultValue true\n */\n renderWorldCopies?: boolean;\n /**\n * The maximum number of tiles stored in the tile cache for a given source. If omitted, the cache will be dynamically sized based on the current viewport which can be set using `maxTileCacheZoomLevels` constructor options.\n * @defaultValue null\n */\n maxTileCacheSize?: number;\n /**\n * The maximum number of zoom levels for which to store tiles for a given source. Tile cache dynamic size is calculated by multiplying `maxTileCacheZoomLevels` with the approximate number of tiles in the viewport for a given source.\n * @defaultValue 5\n */\n maxTileCacheZoomLevels?: number;\n /**\n * A callback run before the Map makes a request for an external URL. The callback can be used to modify the url, set headers, or set the credentials property for cross-origin requests.\n * Expected to return an object with a `url` property and optionally `headers` and `credentials` properties.\n */\n transformRequest?: RequestTransformFunction;\n /**\n * A callback run before the map's camera is moved due to user input or animation. The callback can be used to modify the new center, zoom, pitch and bearing.\n * Expected to return an object containing center, zoom, pitch or bearing values to overwrite.\n */\n transformCameraUpdate?: CameraUpdateTransformFunction;\n /**\n * A patch to apply to the default localization table for UI strings, e.g. control tooltips. The `locale` object maps namespaced UI string IDs to translated strings in the target language; see `src/ui/default_locale.js` for an example with all supported string IDs. The object may specify all UI strings (thereby adding support for a new translation) or only a subset of strings (thereby patching the default translation table).\n * @defaultValue null\n */\n locale?: any;\n /**\n * Controls the duration of the fade-in/fade-out animation for label collisions after initial map load, in milliseconds. This setting affects all symbol layers. This setting does not affect the duration of runtime styling transitions or raster tile cross-fading.\n * @defaultValue 300\n */\n fadeDuration?: number;\n /**\n * If `true`, symbols from multiple sources can collide with each other during collision detection. If `false`, collision detection is run separately for the symbols in each source.\n * @defaultValue true\n */\n crossSourceCollisions?: boolean;\n /**\n * If `true`, Resource Timing API information will be collected for requests made by GeoJSON and Vector Tile web workers (this information is normally inaccessible from the main Javascript thread). Information will be returned in a `resourceTiming` property of relevant `data` events.\n * @defaultValue false\n */\n collectResourceTiming?: boolean;\n /**\n * The max number of pixels a user can shift the mouse pointer during a click for it to be considered a valid click (as opposed to a mouse drag).\n * @defaultValue true\n */\n clickTolerance?: number;\n /**\n * The initial bounds of the map. If `bounds` is specified, it overrides `center` and `zoom` constructor options.\n */\n bounds?: LngLatBoundsLike;\n /**\n * A {@link FitBoundsOptions} options object to use _only_ when fitting the initial `bounds` provided above.\n */\n fitBoundsOptions?: FitBoundsOptions;\n /**\n * Defines a CSS\n * font-family for locally overriding generation of glyphs in the 'CJK Unified Ideographs', 'Hiragana', 'Katakana' and 'Hangul Syllables' ranges.\n * In these ranges, font settings from the map's style will be ignored, except for font-weight keywords (light/regular/medium/bold).\n * Set to `false`, to enable font settings from the map's style for these glyph ranges.\n * The purpose of this option is to avoid bandwidth-intensive glyph server requests. (See [Use locally generated ideographs](https://maplibre.org/maplibre-gl-js/docs/examples/local-ideographs).)\n * @defaultValue 'sans-serif'\n */\n localIdeographFontFamily?: string;\n /**\n * The map's MapLibre style. This must be a JSON object conforming to\n * the schema described in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/),\n * or a URL to such JSON.\n */\n style: StyleSpecification | string;\n /**\n * If `false`, the map's pitch (tilt) control with \"drag to rotate\" interaction will be disabled.\n * @defaultValue true\n */\n pitchWithRotate?: boolean;\n /**\n * The pixel ratio. The canvas' `width` attribute will be `container.clientWidth * pixelRatio` and its `height` attribute will be `container.clientHeight * pixelRatio`. Defaults to `devicePixelRatio` if not specified.\n */\n pixelRatio?: number;\n /**\n * If false, style validation will be skipped. Useful in production environment.\n * @defaultValue true\n */\n validateStyle?: boolean;\n /**\n * The canvas' `width` and `height` max size. The values are passed as an array where the first element is max width and the second element is max height.\n * You shouldn't set this above WebGl `MAX_TEXTURE_SIZE`. Defaults to [4096, 4096].\n */\n maxCanvasSize?: [number, number];\n};\n\n/**\n * An options object for the gesture settings\n * @example\n * ```ts\n * let options = {\n * windowsHelpText: \"Use Ctrl + scroll to zoom the map\",\n * macHelpText: \"Use ⌘ + scroll to zoom the map\",\n * mobileHelpText: \"Use two fingers to move the map\",\n * }\n * ```\n */\nexport type GestureOptions = {\n windowsHelpText?: string;\n macHelpText?: string;\n mobileHelpText?: string;\n};\n\nexport type AddImageOptions = {\n\n}\n\n// See article here: https://medium.com/terria/typescript-transforming-optional-properties-to-required-properties-that-may-be-undefined-7482cb4e1585\ntype Complete = {\n [P in keyof Required]: Pick extends Required> ? T[P] : (T[P] | undefined);\n}\n\n// This type is used inside map since all properties are assigned a default value.\nexport type CompleteMapOptions = Complete;\n\nconst defaultMinZoom = -2;\nconst defaultMaxZoom = 22;\n\n// the default values, but also the valid range\nconst defaultMinPitch = 0;\nconst defaultMaxPitch = 60;\n\n// use this variable to check maxPitch for validity\nconst maxPitchThreshold = 85;\n\nconst defaultOptions = {\n center: [0, 0],\n zoom: 0,\n bearing: 0,\n pitch: 0,\n\n minZoom: defaultMinZoom,\n maxZoom: defaultMaxZoom,\n\n minPitch: defaultMinPitch,\n maxPitch: defaultMaxPitch,\n\n interactive: true,\n scrollZoom: true,\n boxZoom: true,\n dragRotate: true,\n dragPan: true,\n keyboard: true,\n doubleClickZoom: true,\n touchZoomRotate: true,\n touchPitch: true,\n cooperativeGestures: undefined,\n\n bearingSnap: 7,\n clickTolerance: 3,\n pitchWithRotate: true,\n\n hash: false,\n attributionControl: true,\n maplibreLogo: false,\n\n failIfMajorPerformanceCaveat: false,\n preserveDrawingBuffer: false,\n trackResize: true,\n renderWorldCopies: true,\n refreshExpiredTiles: true,\n maxTileCacheSize: null,\n maxTileCacheZoomLevels: config.MAX_TILE_CACHE_ZOOM_LEVELS,\n localIdeographFontFamily: 'sans-serif',\n transformRequest: null,\n transformCameraUpdate: null,\n fadeDuration: 300,\n crossSourceCollisions: true,\n validateStyle: true,\n /**Because GL MAX_TEXTURE_SIZE is usually at least 4096px. */\n maxCanvasSize: [4096, 4096]\n} as CompleteMapOptions;\n\n/**\n * The `Map` object represents the map on your page. It exposes methods\n * and properties that enable you to programmatically change the map,\n * and fires events as users interact with it.\n *\n * You create a `Map` by specifying a `container` and other options, see {@link MapOptions} for the full list.\n * Then MapLibre GL JS initializes the map on the page and returns your `Map` object.\n *\n * @group Main\n *\n * @example\n * ```ts\n * let map = new maplibregl.Map({\n * container: 'map',\n * center: [-122.420679, 37.772537],\n * zoom: 13,\n * style: style_object,\n * hash: true,\n * transformRequest: (url, resourceType)=> {\n * if(resourceType === 'Source' && url.startsWith('http://myHost')) {\n * return {\n * url: url.replace('http', 'https'),\n * headers: { 'my-custom-header': true},\n * credentials: 'include' // Include cookies for cross-origin requests\n * }\n * }\n * }\n * });\n * ```\n * @see [Display a map](https://maplibre.org/maplibre-gl-js/docs/examples/simple-map/)\n */\nexport class Map extends Camera {\n style: Style;\n painter: Painter;\n handlers: HandlerManager;\n\n _container: HTMLElement;\n _canvasContainer: HTMLElement;\n _controlContainer: HTMLElement;\n _controlPositions: {[_: string]: HTMLElement};\n _interactive: boolean;\n _cooperativeGestures: boolean | GestureOptions;\n _cooperativeGesturesScreen: HTMLElement;\n _metaKey: keyof MouseEvent;\n _showTileBoundaries: boolean;\n _showCollisionBoxes: boolean;\n _showPadding: boolean;\n _showOverdrawInspector: boolean;\n _repaint: boolean;\n _vertices: boolean;\n _canvas: HTMLCanvasElement;\n _maxTileCacheSize: number;\n _maxTileCacheZoomLevels: number;\n _frame: Cancelable;\n _styleDirty: boolean;\n _sourcesDirty: boolean;\n _placementDirty: boolean;\n\n _loaded: boolean;\n _idleTriggered: boolean;\n // accounts for placement finishing as well\n _fullyLoaded: boolean;\n _trackResize: boolean;\n _resizeObserver: ResizeObserver;\n _preserveDrawingBuffer: boolean;\n _failIfMajorPerformanceCaveat: boolean;\n _antialias: boolean;\n _refreshExpiredTiles: boolean;\n _hash: Hash;\n _delegatedListeners: any;\n _fadeDuration: number;\n _crossSourceCollisions: boolean;\n _crossFadingFactor: number;\n _collectResourceTiming: boolean;\n _renderTaskQueue: TaskQueue;\n _controls: Array;\n _mapId: number;\n _localIdeographFontFamily: string;\n _validateStyle: boolean;\n _requestManager: RequestManager;\n _locale: any;\n _removed: boolean;\n _clickTolerance: number;\n _overridePixelRatio: number | null;\n _maxCanvasSize: [number, number];\n _terrainDataCallback: (e: MapStyleDataEvent | MapSourceDataEvent) => void;\n\n /**\n * @internal\n * image queue throttling handle. To be used later when clean up\n */\n _imageQueueHandle: number;\n\n /**\n * The map's {@link ScrollZoomHandler}, which implements zooming in and out with a scroll wheel or trackpad.\n * Find more details and examples using `scrollZoom` in the {@link ScrollZoomHandler} section.\n */\n scrollZoom: ScrollZoomHandler;\n\n /**\n * The map's {@link BoxZoomHandler}, which implements zooming using a drag gesture with the Shift key pressed.\n * Find more details and examples using `boxZoom` in the {@link BoxZoomHandler} section.\n */\n boxZoom: BoxZoomHandler;\n\n /**\n * The map's {@link DragRotateHandler}, which implements rotating the map while dragging with the right\n * mouse button or with the Control key pressed. Find more details and examples using `dragRotate`\n * in the {@link DragRotateHandler} section.\n */\n dragRotate: DragRotateHandler;\n\n /**\n * The map's {@link DragPanHandler}, which implements dragging the map with a mouse or touch gesture.\n * Find more details and examples using `dragPan` in the {@link DragPanHandler} section.\n */\n dragPan: DragPanHandler;\n\n /**\n * The map's {@link KeyboardHandler}, which allows the user to zoom, rotate, and pan the map using keyboard\n * shortcuts. Find more details and examples using `keyboard` in the {@link KeyboardHandler} section.\n */\n keyboard: KeyboardHandler;\n\n /**\n * The map's {@link DoubleClickZoomHandler}, which allows the user to zoom by double clicking.\n * Find more details and examples using `doubleClickZoom` in the {@link DoubleClickZoomHandler} section.\n */\n doubleClickZoom: DoubleClickZoomHandler;\n\n /**\n * The map's {@link TwoFingersTouchZoomRotateHandler}, which allows the user to zoom or rotate the map with touch gestures.\n * Find more details and examples using `touchZoomRotate` in the {@link TwoFingersTouchZoomRotateHandler} section.\n */\n touchZoomRotate: TwoFingersTouchZoomRotateHandler;\n\n /**\n * The map's {@link TwoFingersTouchPitchHandler}, which allows the user to pitch the map with touch gestures.\n * Find more details and examples using `touchPitch` in the {@link TwoFingersTouchPitchHandler} section.\n */\n touchPitch: TwoFingersTouchPitchHandler;\n\n constructor(options: MapOptions) {\n PerformanceUtils.mark(PerformanceMarkers.create);\n\n options = extend({}, defaultOptions, options);\n\n if (options.minZoom != null && options.maxZoom != null && options.minZoom > options.maxZoom) {\n throw new Error('maxZoom must be greater than or equal to minZoom');\n }\n\n if (options.minPitch != null && options.maxPitch != null && options.minPitch > options.maxPitch) {\n throw new Error('maxPitch must be greater than or equal to minPitch');\n }\n\n if (options.minPitch != null && options.minPitch < defaultMinPitch) {\n throw new Error(`minPitch must be greater than or equal to ${defaultMinPitch}`);\n }\n\n if (options.maxPitch != null && options.maxPitch > maxPitchThreshold) {\n throw new Error(`maxPitch must be less than or equal to ${maxPitchThreshold}`);\n }\n\n const transform = new Transform(options.minZoom, options.maxZoom, options.minPitch, options.maxPitch, options.renderWorldCopies);\n super(transform, {bearingSnap: options.bearingSnap});\n\n this._interactive = options.interactive;\n this._cooperativeGestures = options.cooperativeGestures;\n this._metaKey = navigator.platform.indexOf('Mac') === 0 ? 'metaKey' : 'ctrlKey';\n this._maxTileCacheSize = options.maxTileCacheSize;\n this._maxTileCacheZoomLevels = options.maxTileCacheZoomLevels;\n this._failIfMajorPerformanceCaveat = options.failIfMajorPerformanceCaveat;\n this._preserveDrawingBuffer = options.preserveDrawingBuffer;\n this._antialias = options.antialias;\n this._trackResize = options.trackResize;\n this._bearingSnap = options.bearingSnap;\n this._refreshExpiredTiles = options.refreshExpiredTiles;\n this._fadeDuration = options.fadeDuration;\n this._crossSourceCollisions = options.crossSourceCollisions;\n this._crossFadingFactor = 1;\n this._collectResourceTiming = options.collectResourceTiming;\n this._renderTaskQueue = new TaskQueue();\n this._controls = [];\n this._mapId = uniqueId();\n this._locale = extend({}, defaultLocale, options.locale);\n this._clickTolerance = options.clickTolerance;\n this._overridePixelRatio = options.pixelRatio;\n this._maxCanvasSize = options.maxCanvasSize;\n this.transformCameraUpdate = options.transformCameraUpdate;\n\n this._imageQueueHandle = ImageRequest.addThrottleControl(() => this.isMoving());\n\n this._requestManager = new RequestManager(options.transformRequest);\n\n if (typeof options.container === 'string') {\n this._container = document.getElementById(options.container);\n if (!this._container) {\n throw new Error(`Container '${options.container}' not found.`);\n }\n } else if (options.container instanceof HTMLElement) {\n this._container = options.container;\n } else {\n throw new Error('Invalid type: \\'container\\' must be a String or HTMLElement.');\n }\n\n if (options.maxBounds) {\n this.setMaxBounds(options.maxBounds);\n }\n\n this._setupContainer();\n this._setupPainter();\n\n this.on('move', () => this._update(false));\n this.on('moveend', () => this._update(false));\n this.on('zoom', () => this._update(true));\n this.on('terrain', () => {\n this.painter.terrainFacilitator.dirty = true;\n this._update(true);\n });\n this.once('idle', () => { this._idleTriggered = true; });\n\n if (typeof window !== 'undefined') {\n addEventListener('online', this._onWindowOnline, false);\n let initialResizeEventCaptured = false;\n const throttledResizeCallback = throttle((entries: ResizeObserverEntry[]) => {\n if (this._trackResize && !this._removed) {\n this.resize(entries)._update();\n }\n }, 50);\n this._resizeObserver = new ResizeObserver((entries) => {\n if (!initialResizeEventCaptured) {\n initialResizeEventCaptured = true;\n return;\n }\n throttledResizeCallback(entries);\n });\n this._resizeObserver.observe(this._container);\n }\n\n this.handlers = new HandlerManager(this, options as CompleteMapOptions);\n\n if (this._cooperativeGestures) {\n this._setupCooperativeGestures();\n }\n\n const hashName = (typeof options.hash === 'string' && options.hash) || undefined;\n this._hash = options.hash && (new Hash(hashName)).addTo(this);\n // don't set position from options if set through hash\n if (!this._hash || !this._hash._onHashChange()) {\n this.jumpTo({\n center: options.center,\n zoom: options.zoom,\n bearing: options.bearing,\n pitch: options.pitch\n });\n\n if (options.bounds) {\n this.resize();\n this.fitBounds(options.bounds, extend({}, options.fitBoundsOptions, {duration: 0}));\n }\n }\n\n this.resize();\n\n this._localIdeographFontFamily = options.localIdeographFontFamily;\n this._validateStyle = options.validateStyle;\n\n if (options.style) this.setStyle(options.style, {localIdeographFontFamily: options.localIdeographFontFamily});\n\n if (options.attributionControl)\n this.addControl(new AttributionControl({customAttribution: options.customAttribution}));\n\n if (options.maplibreLogo)\n this.addControl(new LogoControl(), options.logoPosition);\n\n this.on('style.load', () => {\n if (this.transform.unmodified) {\n this.jumpTo(this.style.stylesheet as any);\n }\n });\n this.on('data', (event: MapDataEvent) => {\n this._update(event.dataType === 'style');\n this.fire(new Event(`${event.dataType}data`, event));\n });\n this.on('dataloading', (event: MapDataEvent) => {\n this.fire(new Event(`${event.dataType}dataloading`, event));\n });\n this.on('dataabort', (event: MapDataEvent) => {\n this.fire(new Event('sourcedataabort', event));\n });\n }\n\n /**\n * @internal\n * Returns a unique number for this map instance which is used for the MapLoadEvent\n * to make sure we only fire one event per instantiated map object.\n * @returns the uniq map ID\n */\n _getMapId() {\n return this._mapId;\n }\n\n /**\n * Adds an {@link IControl} to the map, calling `control.onAdd(this)`.\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param control - The {@link IControl} to add.\n * @param position - position on the map to which the control will be added.\n * Valid values are `'top-left'`, `'top-right'`, `'bottom-left'`, and `'bottom-right'`. Defaults to `'top-right'`.\n * @returns `this`\n * @example\n * Add zoom and rotation controls to the map.\n * ```ts\n * map.addControl(new maplibregl.NavigationControl());\n * ```\n * @see [Display map navigation controls](https://maplibre.org/maplibre-gl-js/docs/examples/navigation/)\n */\n addControl(control: IControl, position?: ControlPosition): Map {\n if (position === undefined) {\n if (control.getDefaultPosition) {\n position = control.getDefaultPosition();\n } else {\n position = 'top-right';\n }\n }\n if (!control || !control.onAdd) {\n return this.fire(new ErrorEvent(new Error(\n 'Invalid argument to map.addControl(). Argument must be a control with onAdd and onRemove methods.')));\n }\n const controlElement = control.onAdd(this);\n this._controls.push(control);\n\n const positionContainer = this._controlPositions[position];\n if (position.indexOf('bottom') !== -1) {\n positionContainer.insertBefore(controlElement, positionContainer.firstChild);\n } else {\n positionContainer.appendChild(controlElement);\n }\n return this;\n }\n\n /**\n * Removes the control from the map.\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param control - The {@link IControl} to remove.\n * @returns `this`\n * @example\n * ```ts\n * // Define a new navigation control.\n * let navigation = new maplibregl.NavigationControl();\n * // Add zoom and rotation controls to the map.\n * map.addControl(navigation);\n * // Remove zoom and rotation controls from the map.\n * map.removeControl(navigation);\n * ```\n */\n removeControl(control: IControl): Map {\n if (!control || !control.onRemove) {\n return this.fire(new ErrorEvent(new Error(\n 'Invalid argument to map.removeControl(). Argument must be a control with onAdd and onRemove methods.')));\n }\n const ci = this._controls.indexOf(control);\n if (ci > -1) this._controls.splice(ci, 1);\n control.onRemove(this);\n return this;\n }\n\n /**\n * Checks if a control exists on the map.\n *\n * @param control - The {@link IControl} to check.\n * @returns true if map contains control.\n * @example\n * ```ts\n * // Define a new navigation control.\n * let navigation = new maplibregl.NavigationControl();\n * // Add zoom and rotation controls to the map.\n * map.addControl(navigation);\n * // Check that the navigation control exists on the map.\n * map.hasControl(navigation);\n * ```\n */\n hasControl(control: IControl): boolean {\n return this._controls.indexOf(control) > -1;\n }\n\n calculateCameraOptionsFromTo(from: LngLat, altitudeFrom: number, to: LngLat, altitudeTo?: number): CameraOptions {\n if (altitudeTo == null && this.terrain) {\n altitudeTo = this.terrain.getElevationForLngLatZoom(to, this.transform.tileZoom);\n }\n return super.calculateCameraOptionsFromTo(from, altitudeFrom, to, altitudeTo);\n }\n\n /**\n * Resizes the map according to the dimensions of its\n * `container` element.\n *\n * Checks if the map container size changed and updates the map if it has changed.\n * This method must be called after the map's `container` is resized programmatically\n * or when the map is shown after being initially hidden with CSS.\n *\n * Triggers the following events: `movestart`, `move`, `moveend`, and `resize`.\n *\n * @param eventData - Additional properties to be passed to `movestart`, `move`, `resize`, and `moveend`\n * events that get triggered as a result of resize. This can be useful for differentiating the\n * source of an event (for example, user-initiated or programmatically-triggered events).\n * @returns `this`\n * @example\n * Resize the map when the map container is shown after being initially hidden with CSS.\n * ```ts\n * let mapDiv = document.getElementById('map');\n * if (mapDiv.style.visibility === true) map.resize();\n * ```\n */\n resize(eventData?: any): Map {\n const dimensions = this._containerDimensions();\n const width = dimensions[0];\n const height = dimensions[1];\n\n const clampedPixelRatio = this._getClampedPixelRatio(width, height);\n this._resizeCanvas(width, height, clampedPixelRatio);\n this.painter.resize(width, height, clampedPixelRatio);\n\n // check if we've reached GL limits, in that case further clamps pixelRatio\n if (this.painter.overLimit()) {\n const gl = this.painter.context.gl;\n // store updated _maxCanvasSize value\n this._maxCanvasSize = [gl.drawingBufferWidth, gl.drawingBufferHeight];\n const clampedPixelRatio = this._getClampedPixelRatio(width, height);\n this._resizeCanvas(width, height, clampedPixelRatio);\n this.painter.resize(width, height, clampedPixelRatio);\n }\n\n this.transform.resize(width, height);\n this._requestedCameraState?.resize(width, height);\n\n const fireMoving = !this._moving;\n if (fireMoving) {\n this.stop();\n this.fire(new Event('movestart', eventData))\n .fire(new Event('move', eventData));\n }\n\n this.fire(new Event('resize', eventData));\n\n if (fireMoving) this.fire(new Event('moveend', eventData));\n\n return this;\n }\n\n /**\n * @internal\n * Return the map's pixel ratio eventually scaled down to respect maxCanvasSize.\n * Internally you should use this and not getPixelRatio().\n */\n _getClampedPixelRatio(width: number, height: number): number {\n const {0: maxCanvasWidth, 1: maxCanvasHeight} = this._maxCanvasSize;\n const pixelRatio = this.getPixelRatio();\n\n const canvasWidth = width * pixelRatio;\n const canvasHeight = height * pixelRatio;\n\n const widthScaleFactor = canvasWidth > maxCanvasWidth ? (maxCanvasWidth / canvasWidth) : 1;\n const heightScaleFactor = canvasHeight > maxCanvasHeight ? (maxCanvasHeight / canvasHeight) : 1;\n\n return Math.min(widthScaleFactor, heightScaleFactor) * pixelRatio;\n }\n\n /**\n * Returns the map's pixel ratio.\n * Note that the pixel ratio actually applied may be lower to respect maxCanvasSize.\n * @returns The pixel ratio.\n */\n getPixelRatio(): number {\n return this._overridePixelRatio ?? devicePixelRatio;\n }\n\n /**\n * Sets the map's pixel ratio. This allows to override `devicePixelRatio`.\n * After this call, the canvas' `width` attribute will be `container.clientWidth * pixelRatio`\n * and its height attribute will be `container.clientHeight * pixelRatio`.\n * Set this to null to disable `devicePixelRatio` override.\n * Note that the pixel ratio actually applied may be lower to respect maxCanvasSize.\n * @param pixelRatio - The pixel ratio.\n */\n setPixelRatio(pixelRatio: number) {\n this._overridePixelRatio = pixelRatio;\n this.resize();\n }\n\n /**\n * Returns the map's geographical bounds. When the bearing or pitch is non-zero, the visible region is not\n * an axis-aligned rectangle, and the result is the smallest bounds that encompasses the visible region.\n * @returns The geographical bounds of the map as {@link LngLatBounds}.\n * @example\n * ```ts\n * let bounds = map.getBounds();\n * ```\n */\n getBounds(): LngLatBounds {\n return this.transform.getBounds();\n }\n\n /**\n * Returns the maximum geographical bounds the map is constrained to, or `null` if none set.\n * @returns The map object.\n * @example\n * ```ts\n * let maxBounds = map.getMaxBounds();\n * ```\n */\n getMaxBounds(): LngLatBounds | null {\n return this.transform.getMaxBounds();\n }\n\n /**\n * Sets or clears the map's geographical bounds.\n *\n * Pan and zoom operations are constrained within these bounds.\n * If a pan or zoom is performed that would\n * display regions outside these bounds, the map will\n * instead display a position and zoom level\n * as close as possible to the operation's request while still\n * remaining within the bounds.\n *\n * @param bounds - The maximum bounds to set. If `null` or `undefined` is provided, the function removes the map's maximum bounds.\n * @returns `this`\n * @example\n * Define bounds that conform to the `LngLatBoundsLike` object as set the max bounds.\n * ```ts\n * let bounds = [\n * [-74.04728, 40.68392], // [west, south]\n * [-73.91058, 40.87764] // [east, north]\n * ];\n * map.setMaxBounds(bounds);\n * ```\n */\n setMaxBounds(bounds?: LngLatBoundsLike | null): Map {\n this.transform.setMaxBounds(LngLatBounds.convert(bounds));\n return this._update();\n }\n\n /**\n * Sets or clears the map's minimum zoom level.\n * If the map's current zoom level is lower than the new minimum,\n * the map will zoom to the new minimum.\n *\n * It is not always possible to zoom out and reach the set `minZoom`.\n * Other factors such as map height may restrict zooming. For example,\n * if the map is 512px tall it will not be possible to zoom below zoom 0\n * no matter what the `minZoom` is set to.\n *\n * A {@link ErrorEvent} event will be fired if minZoom is out of bounds.\n *\n * @param minZoom - The minimum zoom level to set (-2 - 24).\n * If `null` or `undefined` is provided, the function removes the current minimum zoom (i.e. sets it to -2).\n * @returns `this`\n * @example\n * ```ts\n * map.setMinZoom(12.25);\n * ```\n */\n setMinZoom(minZoom?: number | null): Map {\n\n minZoom = minZoom === null || minZoom === undefined ? defaultMinZoom : minZoom;\n\n if (minZoom >= defaultMinZoom && minZoom <= this.transform.maxZoom) {\n this.transform.minZoom = minZoom;\n this._update();\n\n if (this.getZoom() < minZoom) this.setZoom(minZoom);\n\n return this;\n\n } else throw new Error(`minZoom must be between ${defaultMinZoom} and the current maxZoom, inclusive`);\n }\n\n /**\n * Returns the map's minimum allowable zoom level.\n *\n * @returns minZoom\n * @example\n * ```ts\n * let minZoom = map.getMinZoom();\n * ```\n */\n getMinZoom(): number { return this.transform.minZoom; }\n\n /**\n * Sets or clears the map's maximum zoom level.\n * If the map's current zoom level is higher than the new maximum,\n * the map will zoom to the new maximum.\n *\n * A {@link ErrorEvent} event will be fired if minZoom is out of bounds.\n *\n * @param maxZoom - The maximum zoom level to set.\n * If `null` or `undefined` is provided, the function removes the current maximum zoom (sets it to 22).\n * @returns `this`\n * @example\n * ```ts\n * map.setMaxZoom(18.75);\n * ```\n */\n setMaxZoom(maxZoom?: number | null): Map {\n\n maxZoom = maxZoom === null || maxZoom === undefined ? defaultMaxZoom : maxZoom;\n\n if (maxZoom >= this.transform.minZoom) {\n this.transform.maxZoom = maxZoom;\n this._update();\n\n if (this.getZoom() > maxZoom) this.setZoom(maxZoom);\n\n return this;\n\n } else throw new Error('maxZoom must be greater than the current minZoom');\n }\n\n /**\n * Returns the map's maximum allowable zoom level.\n *\n * @returns The maxZoom\n * @example\n * ```ts\n * let maxZoom = map.getMaxZoom();\n * ```\n */\n getMaxZoom(): number { return this.transform.maxZoom; }\n\n /**\n * Sets or clears the map's minimum pitch.\n * If the map's current pitch is lower than the new minimum,\n * the map will pitch to the new minimum.\n *\n * A {@link ErrorEvent} event will be fired if minPitch is out of bounds.\n *\n * @param minPitch - The minimum pitch to set (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * If `null` or `undefined` is provided, the function removes the current minimum pitch (i.e. sets it to 0).\n * @returns `this`\n */\n setMinPitch(minPitch?: number | null): Map {\n\n minPitch = minPitch === null || minPitch === undefined ? defaultMinPitch : minPitch;\n\n if (minPitch < defaultMinPitch) {\n throw new Error(`minPitch must be greater than or equal to ${defaultMinPitch}`);\n }\n\n if (minPitch >= defaultMinPitch && minPitch <= this.transform.maxPitch) {\n this.transform.minPitch = minPitch;\n this._update();\n\n if (this.getPitch() < minPitch) this.setPitch(minPitch);\n\n return this;\n\n } else throw new Error(`minPitch must be between ${defaultMinPitch} and the current maxPitch, inclusive`);\n }\n\n /**\n * Returns the map's minimum allowable pitch.\n *\n * @returns The minPitch\n */\n getMinPitch(): number { return this.transform.minPitch; }\n\n /**\n * Sets or clears the map's maximum pitch.\n * If the map's current pitch is higher than the new maximum,\n * the map will pitch to the new maximum.\n *\n * A {@link ErrorEvent} event will be fired if maxPitch is out of bounds.\n *\n * @param maxPitch - The maximum pitch to set (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project.\n * If `null` or `undefined` is provided, the function removes the current maximum pitch (sets it to 60).\n * @returns `this`\n */\n setMaxPitch(maxPitch?: number | null): Map {\n\n maxPitch = maxPitch === null || maxPitch === undefined ? defaultMaxPitch : maxPitch;\n\n if (maxPitch > maxPitchThreshold) {\n throw new Error(`maxPitch must be less than or equal to ${maxPitchThreshold}`);\n }\n\n if (maxPitch >= this.transform.minPitch) {\n this.transform.maxPitch = maxPitch;\n this._update();\n\n if (this.getPitch() > maxPitch) this.setPitch(maxPitch);\n\n return this;\n\n } else throw new Error('maxPitch must be greater than the current minPitch');\n }\n\n /**\n * Returns the map's maximum allowable pitch.\n *\n * @returns The maxPitch\n */\n getMaxPitch(): number { return this.transform.maxPitch; }\n\n /**\n * Returns the state of `renderWorldCopies`. If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`:\n * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire\n * container, there will be blank space beyond 180 and -180 degrees longitude.\n * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the\n * map and the other on the left edge of the map) at every zoom level.\n * @returns The renderWorldCopies\n * @example\n * ```ts\n * let worldCopiesRendered = map.getRenderWorldCopies();\n * ```\n * @see [Render world copies](https://maplibre.org/maplibre-gl-js/docs/examples/render-world-copies/)\n */\n getRenderWorldCopies(): boolean { return this.transform.renderWorldCopies; }\n\n /**\n * Sets the state of `renderWorldCopies`.\n *\n * @param renderWorldCopies - If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`:\n * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire\n * container, there will be blank space beyond 180 and -180 degrees longitude.\n * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the\n * map and the other on the left edge of the map) at every zoom level.\n *\n * `undefined` is treated as `true`, `null` is treated as `false`.\n * @returns `this`\n * @example\n * ```ts\n * map.setRenderWorldCopies(true);\n * ```\n * @see [Render world copies](https://maplibre.org/maplibre-gl-js/docs/examples/render-world-copies/)\n */\n setRenderWorldCopies(renderWorldCopies?: boolean | null): Map {\n this.transform.renderWorldCopies = renderWorldCopies;\n return this._update();\n }\n\n /**\n * Gets the map's cooperativeGestures option\n *\n * @returns The gestureOptions\n */\n getCooperativeGestures(): boolean | GestureOptions {\n return this._cooperativeGestures;\n }\n\n /**\n * Sets or clears the map's cooperativeGestures option\n *\n * @param gestureOptions - If `true` or set to an options object, map is only accessible on desktop while holding Command/Ctrl and only accessible on mobile with two fingers. Interacting with the map using normal gestures will trigger an informational screen. With this option enabled, \"drag to pitch\" requires a three-finger gesture.\n * @returns `this`\n */\n setCooperativeGestures(gestureOptions?: GestureOptions | boolean | null): Map {\n this._cooperativeGestures = gestureOptions;\n if (this._cooperativeGestures) {\n this._setupCooperativeGestures();\n } else {\n this._destroyCooperativeGestures();\n }\n\n return this;\n }\n\n /**\n * Returns a [Point](https://github.com/mapbox/point-geometry) representing pixel coordinates, relative to the map's `container`,\n * that correspond to the specified geographical location.\n *\n * @param lnglat - The geographical location to project.\n * @returns The [Point](https://github.com/mapbox/point-geometry) corresponding to `lnglat`, relative to the map's `container`.\n * @example\n * ```ts\n * let coordinate = [-122.420679, 37.772537];\n * let point = map.project(coordinate);\n * ```\n */\n project(lnglat: LngLatLike): Point {\n return this.transform.locationPoint(LngLat.convert(lnglat), this.style && this.terrain);\n }\n\n /**\n * Returns a {@link LngLat} representing geographical coordinates that correspond\n * to the specified pixel coordinates.\n *\n * @param point - The pixel coordinates to unproject.\n * @returns The {@link LngLat} corresponding to `point`.\n * @example\n * ```ts\n * map.on('click', function(e) {\n * // When the map is clicked, get the geographic coordinate.\n * let coordinate = map.unproject(e.point);\n * });\n * ```\n */\n unproject(point: PointLike): LngLat {\n return this.transform.pointLocation(Point.convert(point), this.terrain);\n }\n\n /**\n * Returns true if the map is panning, zooming, rotating, or pitching due to a camera animation or user gesture.\n * @returns true if the map is moving.\n * @example\n * ```ts\n * let isMoving = map.isMoving();\n * ```\n */\n isMoving(): boolean {\n return this._moving || this.handlers?.isMoving();\n }\n\n /**\n * Returns true if the map is zooming due to a camera animation or user gesture.\n * @returns true if the map is zooming.\n * @example\n * ```ts\n * let isZooming = map.isZooming();\n * ```\n */\n isZooming(): boolean {\n return this._zooming || this.handlers?.isZooming();\n }\n\n /**\n * Returns true if the map is rotating due to a camera animation or user gesture.\n * @returns true if the map is rotating.\n * @example\n * ```ts\n * map.isRotating();\n * ```\n */\n isRotating(): boolean {\n return this._rotating || this.handlers?.isRotating();\n }\n\n _createDelegatedListener(type: keyof MapEventType | string, layerId: string, listener: Listener): {\n layer: string;\n listener: Listener;\n delegates: {[type in keyof MapEventType]?: (e: any) => void};\n } {\n if (type === 'mouseenter' || type === 'mouseover') {\n let mousein = false;\n const mousemove = (e) => {\n const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, {layers: [layerId]}) : [];\n if (!features.length) {\n mousein = false;\n } else if (!mousein) {\n mousein = true;\n listener.call(this, new MapMouseEvent(type, this, e.originalEvent, {features}));\n }\n };\n const mouseout = () => {\n mousein = false;\n };\n return {layer: layerId, listener, delegates: {mousemove, mouseout}};\n } else if (type === 'mouseleave' || type === 'mouseout') {\n let mousein = false;\n const mousemove = (e) => {\n const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, {layers: [layerId]}) : [];\n if (features.length) {\n mousein = true;\n } else if (mousein) {\n mousein = false;\n listener.call(this, new MapMouseEvent(type, this, e.originalEvent));\n }\n };\n const mouseout = (e) => {\n if (mousein) {\n mousein = false;\n listener.call(this, new MapMouseEvent(type, this, e.originalEvent));\n }\n };\n return {layer: layerId, listener, delegates: {mousemove, mouseout}};\n } else {\n const delegate = (e) => {\n const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, {layers: [layerId]}) : [];\n if (features.length) {\n // Here we need to mutate the original event, so that preventDefault works as expected.\n e.features = features;\n listener.call(this, e);\n delete e.features;\n }\n };\n return {layer: layerId, listener, delegates: {[type]: delegate}};\n }\n }\n\n /**\n * @event\n * Adds a listener for events of a specified type, optionally limited to features in a specified style layer.\n * See {@link MapEventType} and {@link MapLayerEventType} for a full list of events and their description.\n *\n * | Event | Compatible with `layerId` |\n * |------------------------|---------------------------|\n * | `mousedown` | yes |\n * | `mouseup` | yes |\n * | `mouseover` | yes |\n * | `mouseout` | yes |\n * | `mousemove` | yes |\n * | `mouseenter` | yes (required) |\n * | `mouseleave` | yes (required) |\n * | `click` | yes |\n * | `dblclick` | yes |\n * | `contextmenu` | yes |\n * | `touchstart` | yes |\n * | `touchend` | yes |\n * | `touchcancel` | yes |\n * | `wheel` | |\n * | `resize` | |\n * | `remove` | |\n * | `touchmove` | |\n * | `movestart` | |\n * | `move` | |\n * | `moveend` | |\n * | `dragstart` | |\n * | `drag` | |\n * | `dragend` | |\n * | `zoomstart` | |\n * | `zoom` | |\n * | `zoomend` | |\n * | `rotatestart` | |\n * | `rotate` | |\n * | `rotateend` | |\n * | `pitchstart` | |\n * | `pitch` | |\n * | `pitchend` | |\n * | `boxzoomstart` | |\n * | `boxzoomend` | |\n * | `boxzoomcancel` | |\n * | `webglcontextlost` | |\n * | `webglcontextrestored` | |\n * | `load` | |\n * | `render` | |\n * | `idle` | |\n * | `error` | |\n * | `data` | |\n * | `styledata` | |\n * | `sourcedata` | |\n * | `dataloading` | |\n * | `styledataloading` | |\n * | `sourcedataloading` | |\n * | `styleimagemissing` | |\n * | `dataabort` | |\n * | `sourcedataabort` | |\n *\n * @param type - The event type to listen for. Events compatible with the optional `layerId` parameter are triggered\n * when the cursor enters a visible portion of the specified layer from outside that layer or outside the map canvas.\n * @param layer - The ID of a style layer or a listener if no ID is provided. Event will only be triggered if its location\n * is within a visible feature in this layer. The event will have a `features` property containing\n * an array of the matching features. If `layer` is not supplied, the event will not have a `features` property.\n * Please note that many event types are not compatible with the optional `layer` parameter.\n * @param listener - The function to be called when the event is fired.\n * @returns `this`\n * @example\n * ```ts\n * // Set an event listener that will fire\n * // when the map has finished loading\n * map.on('load', function() {\n * // Once the map has finished loading,\n * // add a new layer\n * map.addLayer({\n * id: 'points-of-interest',\n * source: {\n * type: 'vector',\n * url: 'https://maplibre.org/maplibre-style-spec/'\n * },\n * 'source-layer': 'poi_label',\n * type: 'circle',\n * paint: {\n * // MapLibre Style Specification paint properties\n * },\n * layout: {\n * // MapLibre Style Specification layout properties\n * }\n * });\n * });\n * ```\n * @example\n * ```ts\n * // Set an event listener that will fire\n * // when a feature on the countries layer of the map is clicked\n * map.on('click', 'countries', (e) => {\n * new maplibregl.Popup()\n * .setLngLat(e.lngLat)\n * .setHTML(`Country name: ${e.features[0].properties.name}`)\n * .addTo(map);\n * });\n * ```\n * @see [Display popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n * @see [Create a hover effect](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n * @see [Create a draggable marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n on(\n type: T,\n layer: string,\n listener: (ev: MapLayerEventType[T] & Object) => void,\n ): Map;\n /**\n * Overload of the `on` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The listener callback.\n * @returns `this`\n */\n on(type: T, listener: (ev: MapEventType[T] & Object) => void): this;\n /**\n * Overload of the `on` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The listener callback.\n * @returns `this`\n */\n on(type: keyof MapEventType | string, listener: Listener): this;\n on(type: keyof MapEventType | string, layerIdOrListener: string | Listener, listener?: Listener): this {\n if (listener === undefined) {\n return super.on(type, layerIdOrListener as Listener);\n }\n\n const delegatedListener = this._createDelegatedListener(type, layerIdOrListener as string, listener);\n\n this._delegatedListeners = this._delegatedListeners || {};\n this._delegatedListeners[type] = this._delegatedListeners[type] || [];\n this._delegatedListeners[type].push(delegatedListener);\n\n for (const event in delegatedListener.delegates) {\n this.on(event, delegatedListener.delegates[event]);\n }\n\n return this;\n }\n\n /**\n * Adds a listener that will be called only once to a specified event type, optionally limited to features in a specified style layer.\n *\n * @event\n * @param type - The event type to listen for; one of `'mousedown'`, `'mouseup'`, `'click'`, `'dblclick'`,\n * `'mousemove'`, `'mouseenter'`, `'mouseleave'`, `'mouseover'`, `'mouseout'`, `'contextmenu'`, `'touchstart'`,\n * `'touchend'`, or `'touchcancel'`. `mouseenter` and `mouseover` events are triggered when the cursor enters\n * a visible portion of the specified layer from outside that layer or outside the map canvas. `mouseleave`\n * and `mouseout` events are triggered when the cursor leaves a visible portion of the specified layer, or leaves\n * the map canvas.\n * @param layer - The ID of a style layer or a listener if no ID is provided. Only events whose location is within a visible\n * feature in this layer will trigger the listener. The event will have a `features` property containing\n * an array of the matching features.\n * @param listener - The function to be called when the event is fired.\n * @returns `this` if listener is provided, promise otherwise to allow easier usage of async/await\n */\n once(\n type: T,\n layer: string,\n listener?: (ev: MapLayerEventType[T] & Object) => void,\n ): this | Promise;\n /**\n * Overload of the `once` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The listener callback.\n * @returns `this`\n */\n once(type: T, listener?: (ev: MapEventType[T] & Object) => void): this | Promise;\n /**\n * Overload of the `once` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The listener callback.\n * @returns `this`\n */\n once(type: keyof MapEventType | string, listener?: Listener): this | Promise;\n once(type: keyof MapEventType | string, layerIdOrListener: string | Listener, listener?: Listener): this | Promise {\n\n if (listener === undefined) {\n return super.once(type, layerIdOrListener as Listener);\n }\n\n const delegatedListener = this._createDelegatedListener(type, layerIdOrListener as string, listener);\n\n for (const event in delegatedListener.delegates) {\n this.once(event, delegatedListener.delegates[event]);\n }\n\n return this;\n }\n\n /**\n * Removes an event listener for events previously added with `Map#on`.\n *\n * @event\n * @param type - The event type previously used to install the listener.\n * @param layer - The layer ID or listener previously used to install the listener.\n * @param listener - The function previously installed as a listener.\n * @returns `this`\n */\n off(\n type: T,\n layer: string,\n listener: (ev: MapLayerEventType[T] & Object) => void,\n ): this;\n /**\n * Overload of the `off` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The function previously installed as a listener.\n * @returns `this`\n */\n off(type: T, listener: (ev: MapEventType[T] & Object) => void): this;\n /**\n * Overload of the `off` method that allows to listen to events without specifying a layer.\n * @event\n * @param type - The type of the event.\n * @param listener - The function previously installed as a listener.\n * @returns `this`\n */\n off(type: keyof MapEventType | string, listener: Listener): this;\n off(type: keyof MapEventType | string, layerIdOrListener: string | Listener, listener?: Listener): this {\n if (listener === undefined) {\n return super.off(type, layerIdOrListener as Listener);\n }\n\n const removeDelegatedListener = (delegatedListeners) => {\n const listeners = delegatedListeners[type];\n for (let i = 0; i < listeners.length; i++) {\n const delegatedListener = listeners[i];\n if (delegatedListener.layer === layerIdOrListener && delegatedListener.listener === listener) {\n for (const event in delegatedListener.delegates) {\n this.off(((event as any)), delegatedListener.delegates[event]);\n }\n listeners.splice(i, 1);\n return this;\n }\n }\n };\n\n if (this._delegatedListeners && this._delegatedListeners[type]) {\n removeDelegatedListener(this._delegatedListeners);\n }\n\n return this;\n }\n\n /**\n * Returns an array of MapGeoJSONFeature objects\n * representing visible features that satisfy the query parameters.\n *\n * @param geometryOrOptions - (optional) The geometry of the query region:\n * either a single point or southwest and northeast points describing a bounding box.\n * Omitting this parameter (i.e. calling {@link Map#queryRenderedFeatures} with zero arguments,\n * or with only a `options` argument) is equivalent to passing a bounding box encompassing the entire\n * map viewport.\n * The geometryOrOptions can receive a {@link QueryRenderedFeaturesOptions} only to support a situation where the function receives only one parameter which is the options parameter.\n * @param options - (optional) Options object.\n *\n * @returns An array of MapGeoJSONFeature objects.\n *\n * The `properties` value of each returned feature object contains the properties of its source feature. For GeoJSON sources, only\n * string and numeric property values are supported (i.e. `null`, `Array`, and `Object` values are not supported).\n *\n * Each feature includes top-level `layer`, `source`, and `sourceLayer` properties. The `layer` property is an object\n * representing the style layer to which the feature belongs. Layout and paint properties in this object contain values\n * which are fully evaluated for the given zoom level and feature.\n *\n * Only features that are currently rendered are included. Some features will **not** be included, like:\n *\n * - Features from layers whose `visibility` property is `\"none\"`.\n * - Features from layers whose zoom range excludes the current zoom level.\n * - Symbol features that have been hidden due to text or icon collision.\n *\n * Features from all other layers are included, including features that may have no visible\n * contribution to the rendered result; for example, because the layer's opacity or color alpha component is set to\n * 0.\n *\n * The topmost rendered feature appears first in the returned array, and subsequent features are sorted by\n * descending z-order. Features that are rendered multiple times (due to wrapping across the antemeridian at low\n * zoom levels) are returned only once (though subject to the following caveat).\n *\n * Because features come from tiled vector data or GeoJSON data that is converted to tiles internally, feature\n * geometries may be split or duplicated across tile boundaries and, as a result, features may appear multiple\n * times in query results. For example, suppose there is a highway running through the bounding rectangle of a query.\n * The results of the query will be those parts of the highway that lie within the map tiles covering the bounding\n * rectangle, even if the highway extends into other tiles, and the portion of the highway within each map tile\n * will be returned as a separate feature. Similarly, a point feature near a tile boundary may appear in multiple\n * tiles due to tile buffering.\n *\n * @example\n * Find all features at a point\n * ```ts\n * let features = map.queryRenderedFeatures(\n * [20, 35],\n * { layers: ['my-layer-name'] }\n * );\n * ```\n *\n * @example\n * Find all features within a static bounding box\n * ```ts\n * let features = map.queryRenderedFeatures(\n * [[10, 20], [30, 50]],\n * { layers: ['my-layer-name'] }\n * );\n * ```\n *\n * @example\n * Find all features within a bounding box around a point\n * ```ts\n * let width = 10;\n * let height = 20;\n * let features = map.queryRenderedFeatures([\n * [point.x - width / 2, point.y - height / 2],\n * [point.x + width / 2, point.y + height / 2]\n * ], { layers: ['my-layer-name'] });\n * ```\n *\n * @example\n * Query all rendered features from a single layer\n * ```ts\n * let features = map.queryRenderedFeatures({ layers: ['my-layer-name'] });\n * ```\n * @see [Get features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/queryrenderedfeatures/)\n */\n queryRenderedFeatures(geometryOrOptions?: PointLike | [PointLike, PointLike] | QueryRenderedFeaturesOptions, options?: QueryRenderedFeaturesOptions): MapGeoJSONFeature[] {\n if (!this.style) {\n return [];\n }\n let queryGeometry;\n const isGeometry = geometryOrOptions instanceof Point || Array.isArray(geometryOrOptions);\n const geometry = isGeometry ? geometryOrOptions : [[0, 0], [this.transform.width, this.transform.height]];\n options = options || (isGeometry ? {} : geometryOrOptions) || {};\n\n if (geometry instanceof Point || typeof geometry[0] === 'number') {\n queryGeometry = [Point.convert(geometry as PointLike)];\n } else {\n const tl = Point.convert(geometry[0] as PointLike);\n const br = Point.convert(geometry[1] as PointLike);\n queryGeometry = [tl, new Point(br.x, tl.y), br, new Point(tl.x, br.y), tl];\n }\n\n return this.style.queryRenderedFeatures(queryGeometry, options, this.transform);\n }\n\n /**\n * Returns an array of MapGeoJSONFeature objects\n * representing features within the specified vector tile or GeoJSON source that satisfy the query parameters.\n *\n * @param sourceId - The ID of the vector tile or GeoJSON source to query.\n * @param parameters - The options object.\n * @returns An array of MapGeoJSONFeature objects.\n *\n * In contrast to {@link Map#queryRenderedFeatures}, this function returns all features matching the query parameters,\n * whether or not they are rendered by the current style (i.e. visible). The domain of the query includes all currently-loaded\n * vector tiles and GeoJSON source tiles: this function does not check tiles outside the currently\n * visible viewport.\n *\n * Because features come from tiled vector data or GeoJSON data that is converted to tiles internally, feature\n * geometries may be split or duplicated across tile boundaries and, as a result, features may appear multiple\n * times in query results. For example, suppose there is a highway running through the bounding rectangle of a query.\n * The results of the query will be those parts of the highway that lie within the map tiles covering the bounding\n * rectangle, even if the highway extends into other tiles, and the portion of the highway within each map tile\n * will be returned as a separate feature. Similarly, a point feature near a tile boundary may appear in multiple\n * tiles due to tile buffering.\n *\n * @example\n * Find all features in one source layer in a vector source\n * ```ts\n * let features = map.querySourceFeatures('your-source-id', {\n * sourceLayer: 'your-source-layer'\n * });\n * ```\n *\n */\n querySourceFeatures(sourceId: string, parameters?: QuerySourceFeatureOptions | null): MapGeoJSONFeature[] {\n return this.style.querySourceFeatures(sourceId, parameters);\n }\n\n /**\n * Updates the map's MapLibre style object with a new value.\n *\n * If a style is already set when this is used and options.diff is set to true, the map renderer will attempt to compare the given style\n * against the map's current state and perform only the changes necessary to make the map style match the desired state. Changes in sprites\n * (images used for icons and patterns) and glyphs (fonts for label text) **cannot** be diffed. If the sprites or fonts used in the current\n * style and the given style are different in any way, the map renderer will force a full update, removing the current style and building\n * the given one from scratch.\n *\n *\n * @param style - A JSON object conforming to the schema described in the\n * [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/), or a URL to such JSON.\n * @param options - The options object.\n * @returns `this`\n *\n * @example\n * ```ts\n * map.setStyle(\"https://demotiles.maplibre.org/style.json\");\n *\n * map.setStyle('https://demotiles.maplibre.org/style.json', {\n * transformStyle: (previousStyle, nextStyle) => ({\n * ...nextStyle,\n * sources: {\n * ...nextStyle.sources,\n * // copy a source from previous style\n * 'osm': previousStyle.sources.osm\n * },\n * layers: [\n * // background layer\n * nextStyle.layers[0],\n * // copy a layer from previous style\n * previousStyle.layers[0],\n * // other layers from the next style\n * ...nextStyle.layers.slice(1).map(layer => {\n * // hide the layers we don't need from demotiles style\n * if (layer.id.startsWith('geolines')) {\n * layer.layout = {...layer.layout || {}, visibility: 'none'};\n * // filter out US polygons\n * } else if (layer.id.startsWith('coastline') || layer.id.startsWith('countries')) {\n * layer.filter = ['!=', ['get', 'ADM0_A3'], 'USA'];\n * }\n * return layer;\n * })\n * ]\n * })\n * });\n * ```\n */\n setStyle(style: StyleSpecification | string | null, options?: StyleSwapOptions & StyleOptions): this {\n options = extend({},\n {\n localIdeographFontFamily: this._localIdeographFontFamily,\n validate: this._validateStyle\n }, options);\n\n if ((options.diff !== false && options.localIdeographFontFamily === this._localIdeographFontFamily) && this.style && style) {\n this._diffStyle(style, options);\n return this;\n } else {\n this._localIdeographFontFamily = options.localIdeographFontFamily;\n return this._updateStyle(style, options);\n }\n }\n\n /**\n * Updates the requestManager's transform request with a new function\n *\n * @param transformRequest - A callback run before the Map makes a request for an external URL. The callback can be used to modify the url, set headers, or set the credentials property for cross-origin requests.\n * Expected to return an object with a `url` property and optionally `headers` and `credentials` properties\n *\n * @returns `this`\n *\n * @example\n * ```ts\n * map.setTransformRequest((url: string, resourceType: string) => {});\n * ```\n */\n setTransformRequest(transformRequest: RequestTransformFunction): this {\n this._requestManager.setTransformRequest(transformRequest);\n return this;\n }\n\n _getUIString(key: string) {\n const str = this._locale[key];\n if (str == null) {\n throw new Error(`Missing UI string '${key}'`);\n }\n\n return str;\n }\n\n _updateStyle(style: StyleSpecification | string | null, options?: StyleSwapOptions & StyleOptions) {\n // transformStyle relies on having previous style serialized, if it is not loaded yet, delay _updateStyle until previous style is loaded\n if (options.transformStyle && this.style && !this.style._loaded) {\n this.style.once('style.load', () => this._updateStyle(style, options));\n return;\n }\n\n const previousStyle = this.style && options.transformStyle ? this.style.serialize() : undefined;\n if (this.style) {\n this.style.setEventedParent(null);\n\n // Only release workers when map is getting disposed\n this.style._remove(!style);\n }\n\n if (!style) {\n delete this.style;\n return this;\n } else {\n this.style = new Style(this, options || {});\n }\n\n this.style.setEventedParent(this, {style: this.style});\n\n if (typeof style === 'string') {\n this.style.loadURL(style, options, previousStyle);\n } else {\n this.style.loadJSON(style, options, previousStyle);\n }\n\n return this;\n }\n\n _lazyInitEmptyStyle() {\n if (!this.style) {\n this.style = new Style(this, {});\n this.style.setEventedParent(this, {style: this.style});\n this.style.loadEmpty();\n }\n }\n\n _diffStyle(style: StyleSpecification | string, options?: StyleSwapOptions & StyleOptions) {\n if (typeof style === 'string') {\n const url = style;\n const request = this._requestManager.transformRequest(url, ResourceType.Style);\n getJSON(request, (error?: Error | null, json?: any | null) => {\n if (error) {\n this.fire(new ErrorEvent(error));\n } else if (json) {\n this._updateDiff(json, options);\n }\n });\n } else if (typeof style === 'object') {\n this._updateDiff(style, options);\n }\n }\n\n _updateDiff(style: StyleSpecification, options?: StyleSwapOptions & StyleOptions) {\n try {\n if (this.style.setState(style, options)) {\n this._update(true);\n }\n } catch (e) {\n warnOnce(\n `Unable to perform style diff: ${e.message || e.error || e}. Rebuilding the style from scratch.`\n );\n this._updateStyle(style, options);\n }\n }\n\n /**\n * Returns the map's MapLibre style object, a JSON object which can be used to recreate the map's style.\n *\n * @returns The map's style JSON object.\n *\n * @example\n * ```ts\n * let styleJson = map.getStyle();\n * ```\n *\n */\n getStyle(): StyleSpecification {\n if (this.style) {\n return this.style.serialize();\n }\n }\n\n /**\n * Returns a Boolean indicating whether the map's style is fully loaded.\n *\n * @returns A Boolean indicating whether the style is fully loaded.\n *\n * @example\n * ```ts\n * let styleLoadStatus = map.isStyleLoaded();\n * ```\n */\n isStyleLoaded(): boolean | void {\n if (!this.style) return warnOnce('There is no style added to the map.');\n return this.style.loaded();\n }\n\n /**\n * Adds a source to the map's style.\n *\n * Events triggered:\n *\n * Triggers the `source.add` event.\n *\n * @param id - The ID of the source to add. Must not conflict with existing sources.\n * @param source - The source object, conforming to the\n * MapLibre Style Specification's [source definition](https://maplibre.org/maplibre-style-spec/sources) or\n * {@link CanvasSourceSpecification}.\n * @returns `this`\n * @example\n * ```ts\n * map.addSource('my-data', {\n * type: 'vector',\n * url: 'https://demotiles.maplibre.org/tiles/tiles.json'\n * });\n * ```\n * @example\n * ```ts\n * map.addSource('my-data', {\n * \"type\": \"geojson\",\n * \"data\": {\n * \"type\": \"Feature\",\n * \"geometry\": {\n * \"type\": \"Point\",\n * \"coordinates\": [-77.0323, 38.9131]\n * },\n * \"properties\": {\n * \"title\": \"Mapbox DC\",\n * \"marker-symbol\": \"monument\"\n * }\n * }\n * });\n * ```\n * @see GeoJSON source: [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/)\n */\n addSource(id: string, source: SourceSpecification): this {\n this._lazyInitEmptyStyle();\n this.style.addSource(id, source);\n return this._update(true);\n }\n\n /**\n * Returns a Boolean indicating whether the source is loaded. Returns `true` if the source with\n * the given ID in the map's style has no outstanding network requests, otherwise `false`.\n *\n * A {@link ErrorEvent} event will be fired if there is no source wit the specified ID.\n *\n * @param id - The ID of the source to be checked.\n * @returns A Boolean indicating whether the source is loaded.\n * @example\n * ```ts\n * let sourceLoaded = map.isSourceLoaded('bathymetry-data');\n * ```\n */\n isSourceLoaded(id: string): boolean {\n const source = this.style && this.style.sourceCaches[id];\n if (source === undefined) {\n this.fire(new ErrorEvent(new Error(`There is no source with ID '${id}'`)));\n return;\n }\n return source.loaded();\n }\n\n /**\n * Loads a 3D terrain mesh, based on a \"raster-dem\" source.\n *\n * Triggers the `terrain` event.\n *\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setTerrain({ source: 'terrain' });\n * ```\n */\n setTerrain(options: TerrainSpecification | null): this {\n this.style._checkLoaded();\n\n // clear event handlers\n if (this._terrainDataCallback) this.style.off('data', this._terrainDataCallback);\n\n if (!options) {\n // remove terrain\n if (this.terrain) this.terrain.sourceCache.destruct();\n this.terrain = null;\n if (this.painter.renderToTexture) this.painter.renderToTexture.destruct();\n this.painter.renderToTexture = null;\n this.transform._minEleveationForCurrentTile = 0;\n this.transform.elevation = 0;\n } else {\n // add terrain\n const sourceCache = this.style.sourceCaches[options.source];\n if (!sourceCache) throw new Error(`cannot load terrain, because there exists no source with ID: ${options.source}`);\n // Warn once if user is using the same source for hillshade and terrain\n for (const index in this.style._layers) {\n const thisLayer = this.style._layers[index];\n if (thisLayer.type === 'hillshade' && thisLayer.source === options.source) {\n warnOnce('You are using the same source for a hillshade layer and for 3D terrain. Please consider using two separate sources to improve rendering quality.');\n }\n }\n this.terrain = new Terrain(this.painter, sourceCache, options);\n this.painter.renderToTexture = new RenderToTexture(this.painter, this.terrain);\n this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n this._terrainDataCallback = e => {\n if (e.dataType === 'style') {\n this.terrain.sourceCache.freeRtt();\n } else if (e.dataType === 'source' && e.tile) {\n if (e.sourceId === options.source && !this._elevationFreeze) {\n this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n }\n this.terrain.sourceCache.freeRtt(e.tile.tileID);\n }\n };\n this.style.on('data', this._terrainDataCallback);\n }\n\n this.fire(new Event('terrain', {terrain: options}));\n return this;\n }\n\n /**\n * Get the terrain-options if terrain is loaded\n * @returns the TerrainSpecification passed to setTerrain\n * @example\n * ```ts\n * map.getTerrain(); // { source: 'terrain' };\n * ```\n */\n getTerrain(): TerrainSpecification | null {\n return this.terrain?.options ?? null;\n }\n\n /**\n * Returns a Boolean indicating whether all tiles in the viewport from all sources on\n * the style are loaded.\n *\n * @returns A Boolean indicating whether all tiles are loaded.\n * @example\n * ```ts\n * let tilesLoaded = map.areTilesLoaded();\n * ```\n */\n areTilesLoaded(): boolean {\n const sources = this.style && this.style.sourceCaches;\n for (const id in sources) {\n const source = sources[id];\n const tiles = source._tiles;\n for (const t in tiles) {\n const tile = tiles[t];\n if (!(tile.state === 'loaded' || tile.state === 'errored')) return false;\n }\n }\n return true;\n }\n\n /**\n * Adds a [custom source type](#Custom Sources), making it available for use with\n * {@link Map#addSource}.\n * @param name - The name of the source type; source definition objects use this name in the `{type: ...}` field.\n * @param SourceType - A {@link Source} constructor.\n * @param callback - Called when the source type is ready or with an error argument if there is an error.\n */\n addSourceType(name: string, SourceType: SourceClass, callback: Callback) {\n this._lazyInitEmptyStyle();\n return this.style.addSourceType(name, SourceType, callback);\n }\n\n /**\n * Removes a source from the map's style.\n *\n * @param id - The ID of the source to remove.\n * @returns `this`\n * @example\n * ```ts\n * map.removeSource('bathymetry-data');\n * ```\n */\n removeSource(id: string): Map {\n this.style.removeSource(id);\n return this._update(true);\n }\n\n /**\n * Returns the source with the specified ID in the map's style.\n *\n * This method is often used to update a source using the instance members for the relevant\n * source type as defined in [Sources](#sources).\n * For example, setting the `data` for a GeoJSON source or updating the `url` and `coordinates`\n * of an image source.\n *\n * @param id - The ID of the source to get.\n * @returns The style source with the specified ID or `undefined` if the ID\n * corresponds to no existing sources.\n * The shape of the object varies by source type.\n * A list of options for each source type is available on the MapLibre Style Specification's\n * [Sources](https://maplibre.org/maplibre-style-spec/sources/) page.\n * @example\n * ```ts\n * let sourceObject = map.getSource('points');\n * ```\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n * @see [Animate a point](https://maplibre.org/maplibre-gl-js/docs/examples/animate-point-along-line/)\n * @see [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/)\n */\n getSource(id: string): Source | undefined {\n return this.style.getSource(id);\n }\n\n /**\n * Add an image to the style. This image can be displayed on the map like any other icon in the style's\n * sprite using the image's ID with\n * [`icon-image`](https://maplibre.org/maplibre-style-spec/layers/#layout-symbol-icon-image),\n * [`background-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-background-background-pattern),\n * [`fill-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-fill-fill-pattern),\n * or [`line-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-line-line-pattern).\n *\n * A {@link ErrorEvent} event will be fired if the image parameter is invalid or there is not enough space in the sprite to add this image.\n *\n * @param id - The ID of the image.\n * @param image - The image as an `HTMLImageElement`, `ImageData`, `ImageBitmap` or object with `width`, `height`, and `data`\n * properties with the same format as `ImageData`.\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * // If the style's sprite does not already contain an image with ID 'cat',\n * // add the image 'cat-icon.png' to the style's sprite with the ID 'cat'.\n * map.loadImage('https://upload.wikimedia.org/wikipedia/commons/thumb/6/60/Cat_silhouette.svg/400px-Cat_silhouette.svg.png', function(error, image) {\n * if (error) throw error;\n * if (!map.hasImage('cat')) map.addImage('cat', image);\n * });\n *\n * // Add a stretchable image that can be used with `icon-text-fit`\n * // In this example, the image is 600px wide by 400px high.\n * map.loadImage('https://upload.wikimedia.org/wikipedia/commons/8/89/Black_and_White_Boxed_%28bordered%29.png', function(error, image) {\n * if (error) throw error;\n * if (!map.hasImage('border-image')) {\n * map.addImage('border-image', image, {\n * content: [16, 16, 300, 384], // place text over left half of image, avoiding the 16px border\n * stretchX: [[16, 584]], // stretch everything horizontally except the 16px border\n * stretchY: [[16, 384]], // stretch everything vertically except the 16px border\n * });\n * }\n * });\n * ```\n * @see Use `HTMLImageElement`: [Add an icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image/)\n * @see Use `ImageData`: [Add a generated icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-generated/)\n */\n addImage(id: string,\n image: HTMLImageElement | ImageBitmap | ImageData | {\n width: number;\n height: number;\n data: Uint8Array | Uint8ClampedArray;\n } | StyleImageInterface,\n options: Partial = {}): this {\n const {\n pixelRatio = 1,\n sdf = false,\n stretchX,\n stretchY,\n content\n } = options;\n this._lazyInitEmptyStyle();\n const version = 0;\n\n if (image instanceof HTMLImageElement || isImageBitmap(image)) {\n const {width, height, data} = browser.getImageData(image);\n this.style.addImage(id, {data: new RGBAImage({width, height}, data), pixelRatio, stretchX, stretchY, content, sdf, version});\n } else if (image.width === undefined || image.height === undefined) {\n return this.fire(new ErrorEvent(new Error(\n 'Invalid arguments to map.addImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, ' +\n 'or object with `width`, `height`, and `data` properties with the same format as `ImageData`')));\n } else {\n const {width, height, data} = image as ImageData;\n const userImage = (image as any as StyleImageInterface);\n\n this.style.addImage(id, {\n data: new RGBAImage({width, height}, new Uint8Array(data)),\n pixelRatio,\n stretchX,\n stretchY,\n content,\n sdf,\n version,\n userImage\n });\n\n if (userImage.onAdd) {\n userImage.onAdd(this, id);\n }\n return this;\n }\n }\n\n /**\n * Update an existing image in a style. This image can be displayed on the map like any other icon in the style's\n * sprite using the image's ID with\n * [`icon-image`](https://maplibre.org/maplibre-style-spec/layers/#layout-symbol-icon-image),\n * [`background-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-background-background-pattern),\n * [`fill-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-fill-fill-pattern),\n * or [`line-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-line-line-pattern).\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param id - The ID of the image.\n * @param image - The image as an `HTMLImageElement`, `ImageData`, `ImageBitmap` or object with `width`, `height`, and `data`\n * properties with the same format as `ImageData`.\n * @returns `this`\n * @example\n * ```ts\n * // If an image with the ID 'cat' already exists in the style's sprite,\n * // replace that image with a new image, 'other-cat-icon.png'.\n * if (map.hasImage('cat')) map.updateImage('cat', './other-cat-icon.png');\n * ```\n */\n updateImage(id: string,\n image: HTMLImageElement | ImageBitmap | ImageData | {\n width: number;\n height: number;\n data: Uint8Array | Uint8ClampedArray;\n } | StyleImageInterface): this {\n\n const existingImage = this.style.getImage(id);\n if (!existingImage) {\n return this.fire(new ErrorEvent(new Error(\n 'The map has no image with that id. If you are adding a new image use `map.addImage(...)` instead.')));\n }\n const imageData = (image instanceof HTMLImageElement || isImageBitmap(image)) ?\n browser.getImageData(image) :\n image;\n const {width, height, data} = imageData;\n\n if (width === undefined || height === undefined) {\n return this.fire(new ErrorEvent(new Error(\n 'Invalid arguments to map.updateImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, ' +\n 'or object with `width`, `height`, and `data` properties with the same format as `ImageData`')));\n }\n\n if (width !== existingImage.data.width || height !== existingImage.data.height) {\n return this.fire(new ErrorEvent(new Error(\n 'The width and height of the updated image must be that same as the previous version of the image')));\n }\n\n const copy = !(image instanceof HTMLImageElement || isImageBitmap(image));\n existingImage.data.replace(data, copy);\n\n this.style.updateImage(id, existingImage);\n return this;\n }\n\n /**\n * Returns an image, specified by ID, currently available in the map.\n * This includes both images from the style's original sprite\n * and any images that have been added at runtime using {@link Map#addImage}.\n *\n * @param id - The ID of the image.\n * @returns An image in the map with the specified ID.\n *\n * @example\n * ```ts\n * let coffeeShopIcon = map.getImage(\"coffee_cup\");\n * ```\n */\n getImage(id: string): StyleImage {\n return this.style.getImage(id);\n }\n\n /**\n * Check whether or not an image with a specific ID exists in the style. This checks both images\n * in the style's original sprite and any images\n * that have been added at runtime using {@link Map#addImage}.\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param id - The ID of the image.\n *\n * @returns A Boolean indicating whether the image exists.\n * @example\n * Check if an image with the ID 'cat' exists in the style's sprite.\n * ```ts\n * let catIconExists = map.hasImage('cat');\n * ```\n */\n hasImage(id: string): boolean {\n if (!id) {\n this.fire(new ErrorEvent(new Error('Missing required image id')));\n return false;\n }\n\n return !!this.style.getImage(id);\n }\n\n /**\n * Remove an image from a style. This can be an image from the style's original\n * sprite or any images\n * that have been added at runtime using {@link Map#addImage}.\n *\n * @param id - The ID of the image.\n *\n * @example\n * ```ts\n * // If an image with the ID 'cat' exists in\n * // the style's sprite, remove it.\n * if (map.hasImage('cat')) map.removeImage('cat');\n * ```\n */\n removeImage(id: string) {\n this.style.removeImage(id);\n }\n\n /**\n * Load an image from an external URL to be used with {@link Map#addImage}. External\n * domains must support [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/Access_control_CORS).\n *\n * @param url - The URL of the image file. Image file must be in png, webp, or jpg format.\n * @param callback - Expecting `callback(error, data)`. Called when the image has loaded or with an error argument if there is an error.\n *\n * @example\n * Load an image from an external URL.\n * ```ts\n * map.loadImage('http://placekitten.com/50/50', function(error, image) {\n * if (error) throw error;\n * // Add the loaded image to the style's sprite with the ID 'kitten'.\n * map.addImage('kitten', image);\n * });\n * ```\n * @see [Add an icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image/)\n */\n loadImage(url: string, callback: GetImageCallback) {\n ImageRequest.getImage(this._requestManager.transformRequest(url, ResourceType.Image), callback);\n }\n\n /**\n * Returns an Array of strings containing the IDs of all images currently available in the map.\n * This includes both images from the style's original sprite\n * and any images that have been added at runtime using {@link Map#addImage}.\n *\n * @returns An Array of strings containing the names of all sprites/images currently available in the map.\n *\n * @example\n * ```ts\n * let allImages = map.listImages();\n * ```\n */\n listImages(): Array {\n return this.style.listImages();\n }\n\n /**\n * Adds a [MapLibre style layer](https://maplibre.org/maplibre-style-spec/layers)\n * to the map's style.\n *\n * A layer defines how data from a specified source will be styled. Read more about layer types\n * and available paint and layout properties in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/layers).\n *\n * @param layer - The layer to add,\n * conforming to either the MapLibre Style Specification's [layer definition](https://maplibre.org/maplibre-style-spec/layers) or,\n * less commonly, the {@link CustomLayerInterface} specification. Can also be a layer definition with an embedded source definition.\n * The MapLibre Style Specification's layer definition is appropriate for most layers.\n *\n * @param beforeId - The ID of an existing layer to insert the new layer before,\n * resulting in the new layer appearing visually beneath the existing layer.\n * If this argument is not specified, the layer will be appended to the end of the layers array\n * and appear visually above all other layers.\n *\n * @returns `this`\n *\n * @example\n * Add a circle layer with a vector source\n * ```ts\n * map.addLayer({\n * id: 'points-of-interest',\n * source: {\n * type: 'vector',\n * url: 'https://demotiles.maplibre.org/tiles/tiles.json'\n * },\n * 'source-layer': 'poi_label',\n * type: 'circle',\n * paint: {\n * // MapLibre Style Specification paint properties\n * },\n * layout: {\n * // MapLibre Style Specification layout properties\n * }\n * });\n * ```\n *\n * @example\n * Define a source before using it to create a new layer\n * ```ts\n * map.addSource('state-data', {\n * type: 'geojson',\n * data: 'path/to/data.geojson'\n * });\n *\n * map.addLayer({\n * id: 'states',\n * // References the GeoJSON source defined above\n * // and does not require a `source-layer`\n * source: 'state-data',\n * type: 'symbol',\n * layout: {\n * // Set the label content to the\n * // feature's `name` property\n * text-field: ['get', 'name']\n * }\n * });\n * ```\n *\n * @example\n * Add a new symbol layer before an existing layer\n * ```ts\n * map.addLayer({\n * id: 'states',\n * // References a source that's already been defined\n * source: 'state-data',\n * type: 'symbol',\n * layout: {\n * // Set the label content to the\n * // feature's `name` property\n * text-field: ['get', 'name']\n * }\n * // Add the layer before the existing `cities` layer\n * }, 'cities');\n * ```\n * @see [Create and style clusters](https://maplibre.org/maplibre-gl-js/docs/examples/cluster/)\n * @see [Add a vector tile source](https://maplibre.org/maplibre-gl-js/docs/examples/vector-source/)\n * @see [Add a WMS source](https://maplibre.org/maplibre-gl-js/docs/examples/wms/)\n */\n addLayer(layer: AddLayerObject, beforeId?: string) {\n this._lazyInitEmptyStyle();\n this.style.addLayer(layer, beforeId);\n return this._update(true);\n }\n\n /**\n * Moves a layer to a different z-position.\n *\n * @param id - The ID of the layer to move.\n * @param beforeId - The ID of an existing layer to insert the new layer before. When viewing the map, the `id` layer will appear beneath the `beforeId` layer. If `beforeId` is omitted, the layer will be appended to the end of the layers array and appear above all other layers on the map.\n * @returns `this`\n *\n * @example\n * Move a layer with ID 'polygon' before the layer with ID 'country-label'. The `polygon` layer will appear beneath the `country-label` layer on the map.\n * ```ts\n * map.moveLayer('polygon', 'country-label');\n * ```\n */\n moveLayer(id: string, beforeId?: string): this {\n this.style.moveLayer(id, beforeId);\n return this._update(true);\n }\n\n /**\n * Removes the layer with the given ID from the map's style.\n *\n * An {@link ErrorEvent} will be fired if the image parameter is invald.\n *\n * @param id - The ID of the layer to remove\n * @returns `this`\n *\n * @example\n * If a layer with ID 'state-data' exists, remove it.\n * ```ts\n * if (map.getLayer('state-data')) map.removeLayer('state-data');\n * ```\n */\n removeLayer(id: string): this {\n this.style.removeLayer(id);\n return this._update(true);\n }\n\n /**\n * Returns the layer with the specified ID in the map's style.\n *\n * @param id - The ID of the layer to get.\n * @returns The layer with the specified ID, or `undefined`\n * if the ID corresponds to no existing layers.\n *\n * @example\n * ```ts\n * let stateDataLayer = map.getLayer('state-data');\n * ```\n * @see [Filter symbols by toggling a list](https://maplibre.org/maplibre-gl-js/docs/examples/filter-markers/)\n * @see [Filter symbols by text input](https://maplibre.org/maplibre-gl-js/docs/examples/filter-markers-by-input/)\n */\n getLayer(id: string): StyleLayer | undefined {\n return this.style.getLayer(id);\n }\n\n /**\n * Return the ids of all layers currently in the style, including custom layers, in order.\n *\n * @returns ids of layers, in order\n *\n * @example\n * ```ts\n * const orderedLayerIds = map.getLayersOrder();\n * ```\n */\n getLayersOrder(): string[] {\n return this.style.getLayersOrder();\n }\n\n /**\n * Sets the zoom extent for the specified style layer. The zoom extent includes the\n * [minimum zoom level](https://maplibre.org/maplibre-style-spec/layers/#minzoom)\n * and [maximum zoom level](https://maplibre.org/maplibre-style-spec/layers/#maxzoom))\n * at which the layer will be rendered.\n *\n * Note: For style layers using vector sources, style layers cannot be rendered at zoom levels lower than the\n * minimum zoom level of the _source layer_ because the data does not exist at those zoom levels. If the minimum\n * zoom level of the source layer is higher than the minimum zoom level defined in the style layer, the style\n * layer will not be rendered at all zoom levels in the zoom range.\n *\n * @param layerId - The ID of the layer to which the zoom extent will be applied.\n * @param minzoom - The minimum zoom to set (0-24).\n * @param maxzoom - The maximum zoom to set (0-24).\n * @returns `this`\n *\n * @example\n * ```ts\n * map.setLayerZoomRange('my-layer', 2, 5);\n * ```\n */\n setLayerZoomRange(layerId: string, minzoom: number, maxzoom: number): this {\n this.style.setLayerZoomRange(layerId, minzoom, maxzoom);\n return this._update(true);\n }\n\n /**\n * Sets the filter for the specified style layer.\n *\n * Filters control which features a style layer renders from its source.\n * Any feature for which the filter expression evaluates to `true` will be\n * rendered on the map. Those that are false will be hidden.\n *\n * Use `setFilter` to show a subset of your source data.\n *\n * To clear the filter, pass `null` or `undefined` as the second parameter.\n *\n * @param layerId - The ID of the layer to which the filter will be applied.\n * @param filter - The filter, conforming to the MapLibre Style Specification's\n * [filter definition](https://maplibre.org/maplibre-style-spec/layers/#filter). If `null` or `undefined` is provided, the function removes any existing filter from the layer.\n * @param options - Options object.\n * @returns `this`\n *\n * @example\n * Display only features with the 'name' property 'USA'\n * ```ts\n * map.setFilter('my-layer', ['==', ['get', 'name'], 'USA']);\n * ```\n * @example\n * Display only features with five or more 'available-spots'\n * ```ts\n * map.setFilter('bike-docks', ['>=', ['get', 'available-spots'], 5]);\n * ```\n * @example\n * Remove the filter for the 'bike-docks' style layer\n * ```ts\n * map.setFilter('bike-docks', null);\n * ```\n * @see [Create a timeline animation](https://maplibre.org/maplibre-gl-js/docs/examples/timeline-animation/)\n */\n setFilter(layerId: string, filter?: FilterSpecification | null, options: StyleSetterOptions = {}) {\n this.style.setFilter(layerId, filter, options);\n return this._update(true);\n }\n\n /**\n * Returns the filter applied to the specified style layer.\n *\n * @param layerId - The ID of the style layer whose filter to get.\n * @returns The layer's filter.\n */\n getFilter(layerId: string): FilterSpecification | void {\n return this.style.getFilter(layerId);\n }\n\n /**\n * Sets the value of a paint property in the specified style layer.\n *\n * @param layerId - The ID of the layer to set the paint property in.\n * @param name - The name of the paint property to set.\n * @param value - The value of the paint property to set.\n * Must be of a type appropriate for the property, as defined in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/).\n * Pass `null` to unset the existing value.\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setPaintProperty('my-layer', 'fill-color', '#faafee');\n * ```\n * @see [Change a layer's color with buttons](https://maplibre.org/maplibre-gl-js/docs/examples/color-switcher/)\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n setPaintProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}): this {\n this.style.setPaintProperty(layerId, name, value, options);\n return this._update(true);\n }\n\n /**\n * Returns the value of a paint property in the specified style layer.\n *\n * @param layerId - The ID of the layer to get the paint property from.\n * @param name - The name of a paint property to get.\n * @returns The value of the specified paint property.\n */\n getPaintProperty(layerId: string, name: string) {\n return this.style.getPaintProperty(layerId, name);\n }\n\n /**\n * Sets the value of a layout property in the specified style layer.\n *\n * @param layerId - The ID of the layer to set the layout property in.\n * @param name - The name of the layout property to set.\n * @param value - The value of the layout property. Must be of a type appropriate for the property, as defined in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/).\n * @param options - The options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setLayoutProperty('my-layer', 'visibility', 'none');\n * ```\n */\n setLayoutProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}): this {\n this.style.setLayoutProperty(layerId, name, value, options);\n return this._update(true);\n }\n\n /**\n * Returns the value of a layout property in the specified style layer.\n *\n * @param layerId - The ID of the layer to get the layout property from.\n * @param name - The name of the layout property to get.\n * @returns The value of the specified layout property.\n */\n getLayoutProperty(layerId: string, name: string) {\n return this.style.getLayoutProperty(layerId, name);\n }\n\n /**\n * Sets the value of the style's glyphs property.\n *\n * @param glyphsUrl - Glyph URL to set. Must conform to the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/glyphs/).\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setGlyphs('https://demotiles.maplibre.org/font/{fontstack}/{range}.pbf');\n * ```\n */\n setGlyphs(glyphsUrl: string | null, options: StyleSetterOptions = {}): this {\n this._lazyInitEmptyStyle();\n this.style.setGlyphs(glyphsUrl, options);\n return this._update(true);\n }\n\n /**\n * Returns the value of the style's glyphs URL\n *\n * @returns glyphs Style's glyphs url\n */\n getGlyphs(): string | null {\n return this.style.getGlyphsUrl();\n }\n\n /**\n * Adds a sprite to the map's style. Fires the `style` event.\n *\n * @param id - The ID of the sprite to add. Must not conflict with existing sprites.\n * @param url - The URL to load the sprite from\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.addSprite('sprite-two', 'http://example.com/sprite-two');\n * ```\n */\n addSprite(id: string, url: string, options: StyleSetterOptions = {}): this {\n this._lazyInitEmptyStyle();\n this.style.addSprite(id, url, options, (err) => {\n if (!err) {\n this._update(true);\n }\n });\n return this;\n }\n\n /**\n * Removes the sprite from the map's style. Fires the `style` event.\n *\n * @param id - The ID of the sprite to remove. If the sprite is declared as a single URL, the ID must be \"default\".\n * @returns `this`\n * @example\n * ```ts\n * map.removeSprite('sprite-two');\n * map.removeSprite('default');\n * ```\n */\n removeSprite(id: string) {\n this._lazyInitEmptyStyle();\n this.style.removeSprite(id);\n return this._update(true);\n }\n\n /**\n * Returns the as-is value of the style's sprite.\n *\n * @returns style's sprite list of id-url pairs\n */\n getSprite(): {id: string; url: string}[] {\n return this.style.getSprite();\n }\n\n /**\n * Sets the value of the style's sprite property.\n *\n * @param spriteUrl - Sprite URL to set.\n * @param options - Options object.\n * @returns `this`\n * @example\n * ```ts\n * map.setSprite('YOUR_SPRITE_URL');\n * ```\n */\n setSprite(spriteUrl: string | null, options: StyleSetterOptions = {}) {\n this._lazyInitEmptyStyle();\n this.style.setSprite(spriteUrl, options, (err) => {\n if (!err) {\n this._update(true);\n }\n });\n return this;\n }\n\n /**\n * Sets the any combination of light values.\n *\n * @param light - Light properties to set. Must conform to the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/light).\n * @param options - Options object.\n * @returns `this`\n *\n * @example\n * ```ts\n * let layerVisibility = map.getLayoutProperty('my-layer', 'visibility');\n * ```\n */\n setLight(light: LightSpecification, options: StyleSetterOptions = {}) {\n this._lazyInitEmptyStyle();\n this.style.setLight(light, options);\n return this._update(true);\n }\n\n /**\n * Returns the value of the light object.\n *\n * @returns light Light properties of the style.\n */\n getLight(): LightSpecification {\n return this.style.getLight();\n }\n\n /**\n * Sets the `state` of a feature.\n * A feature's `state` is a set of user-defined key-value pairs that are assigned to a feature at runtime.\n * When using this method, the `state` object is merged with any existing key-value pairs in the feature's state.\n * Features are identified by their `feature.id` attribute, which can be any number or string.\n *\n * This method can only be used with sources that have a `feature.id` attribute. The `feature.id` attribute can be defined in three ways:\n * - For vector or GeoJSON sources, including an `id` attribute in the original data file.\n * - For vector or GeoJSON sources, using the [`promoteId`](https://maplibre.org/maplibre-style-spec/sources/#vector-promoteId) option at the time the source is defined.\n * - For GeoJSON sources, using the [`generateId`](https://maplibre.org/maplibre-style-spec/sources/#geojson-generateId) option to auto-assign an `id` based on the feature's index in the source data. If you change feature data using `map.getSource('some id').setData(..)`, you may need to re-apply state taking into account updated `id` values.\n *\n * _Note: You can use the [`feature-state` expression](https://maplibre.org/maplibre-style-spec/expressions/#feature-state) to access the values in a feature's state object for the purposes of styling._\n *\n * @param feature - Feature identifier. Feature objects returned from\n * {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers.\n * @param state - A set of key-value pairs. The values should be valid JSON types.\n * @returns `this`\n *\n * @example\n * ```ts\n * // When the mouse moves over the `my-layer` layer, update\n * // the feature state for the feature under the mouse\n * map.on('mousemove', 'my-layer', function(e) {\n * if (e.features.length > 0) {\n * map.setFeatureState({\n * source: 'my-source',\n * sourceLayer: 'my-source-layer',\n * id: e.features[0].id,\n * }, {\n * hover: true\n * });\n * }\n * });\n * ```\n * @see [Create a hover effect](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/)\n */\n setFeatureState(feature: FeatureIdentifier, state: any): this {\n this.style.setFeatureState(feature, state);\n return this._update();\n }\n\n /**\n * Removes the `state` of a feature, setting it back to the default behavior.\n * If only a `target.source` is specified, it will remove the state for all features from that source.\n * If `target.id` is also specified, it will remove all keys for that feature's state.\n * If `key` is also specified, it removes only that key from that feature's state.\n * Features are identified by their `feature.id` attribute, which can be any number or string.\n *\n * @param target - Identifier of where to remove state. It can be a source, a feature, or a specific key of feature.\n * Feature objects returned from {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers.\n * @param key - (optional) The key in the feature state to reset.\n * @returns `this`\n * @example\n * Reset the entire state object for all features in the `my-source` source\n * ```ts\n * map.removeFeatureState({\n * source: 'my-source'\n * });\n * ```\n *\n * @example\n * When the mouse leaves the `my-layer` layer,\n * reset the entire state object for the\n * feature under the mouse\n * ```ts\n * map.on('mouseleave', 'my-layer', function(e) {\n * map.removeFeatureState({\n * source: 'my-source',\n * sourceLayer: 'my-source-layer',\n * id: e.features[0].id\n * });\n * });\n * ```\n *\n * @example\n * When the mouse leaves the `my-layer` layer,\n * reset only the `hover` key-value pair in the\n * state for the feature under the mouse\n * ```ts\n * map.on('mouseleave', 'my-layer', function(e) {\n * map.removeFeatureState({\n * source: 'my-source',\n * sourceLayer: 'my-source-layer',\n * id: e.features[0].id\n * }, 'hover');\n * });\n * ```\n */\n removeFeatureState(target: FeatureIdentifier, key?: string): this {\n this.style.removeFeatureState(target, key);\n return this._update();\n }\n\n /**\n * Gets the `state` of a feature.\n * A feature's `state` is a set of user-defined key-value pairs that are assigned to a feature at runtime.\n * Features are identified by their `feature.id` attribute, which can be any number or string.\n *\n * _Note: To access the values in a feature's state object for the purposes of styling the feature, use the [`feature-state` expression](https://maplibre.org/maplibre-style-spec/expressions/#feature-state)._\n *\n * @param feature - Feature identifier. Feature objects returned from\n * {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers.\n * @returns The state of the feature: a set of key-value pairs that was assigned to the feature at runtime.\n *\n * @example\n * When the mouse moves over the `my-layer` layer,\n * get the feature state for the feature under the mouse\n * ```ts\n * map.on('mousemove', 'my-layer', function(e) {\n * if (e.features.length > 0) {\n * map.getFeatureState({\n * source: 'my-source',\n * sourceLayer: 'my-source-layer',\n * id: e.features[0].id\n * });\n * }\n * });\n * ```\n */\n getFeatureState(feature: FeatureIdentifier): any {\n return this.style.getFeatureState(feature);\n }\n\n /**\n * Returns the map's containing HTML element.\n *\n * @returns The map's container.\n */\n getContainer(): HTMLElement {\n return this._container;\n }\n\n /**\n * Returns the HTML element containing the map's `` element.\n *\n * If you want to add non-GL overlays to the map, you should append them to this element.\n *\n * This is the element to which event bindings for map interactivity (such as panning and zooming) are\n * attached. It will receive bubbled events from child elements such as the ``, but not from\n * map controls.\n *\n * @returns The container of the map's ``.\n * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/)\n */\n getCanvasContainer(): HTMLElement {\n return this._canvasContainer;\n }\n\n /**\n * Returns the map's `` element.\n *\n * @returns The map's `` element.\n * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/)\n */\n getCanvas(): HTMLCanvasElement {\n return this._canvas;\n }\n\n _containerDimensions() {\n let width = 0;\n let height = 0;\n\n if (this._container) {\n width = this._container.clientWidth || 400;\n height = this._container.clientHeight || 300;\n }\n\n return [width, height];\n }\n\n _setupContainer() {\n const container = this._container;\n container.classList.add('maplibregl-map');\n\n const canvasContainer = this._canvasContainer = DOM.create('div', 'maplibregl-canvas-container', container);\n if (this._interactive) {\n canvasContainer.classList.add('maplibregl-interactive');\n }\n\n this._canvas = DOM.create('canvas', 'maplibregl-canvas', canvasContainer);\n this._canvas.addEventListener('webglcontextlost', this._contextLost, false);\n this._canvas.addEventListener('webglcontextrestored', this._contextRestored, false);\n this._canvas.setAttribute('tabindex', '0');\n this._canvas.setAttribute('aria-label', 'Map');\n this._canvas.setAttribute('role', 'region');\n\n const dimensions = this._containerDimensions();\n const clampedPixelRatio = this._getClampedPixelRatio(dimensions[0], dimensions[1]);\n this._resizeCanvas(dimensions[0], dimensions[1], clampedPixelRatio);\n\n const controlContainer = this._controlContainer = DOM.create('div', 'maplibregl-control-container', container);\n const positions = this._controlPositions = {};\n ['top-left', 'top-right', 'bottom-left', 'bottom-right'].forEach((positionName) => {\n positions[positionName] = DOM.create('div', `maplibregl-ctrl-${positionName} `, controlContainer);\n });\n\n this._container.addEventListener('scroll', this._onMapScroll, false);\n }\n\n _cooperativeGesturesOnWheel = (event: WheelEvent) => {\n this._onCooperativeGesture(event, event[this._metaKey], 1);\n };\n\n _setupCooperativeGestures() {\n const container = this._container;\n this._cooperativeGesturesScreen = DOM.create('div', 'maplibregl-cooperative-gesture-screen', container);\n let desktopMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.windowsHelpText ? this._cooperativeGestures.windowsHelpText : 'Use Ctrl + scroll to zoom the map';\n if (navigator.platform.indexOf('Mac') === 0) {\n desktopMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.macHelpText ? this._cooperativeGestures.macHelpText : 'Use ⌘ + scroll to zoom the map';\n }\n const mobileMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.mobileHelpText ? this._cooperativeGestures.mobileHelpText : 'Use two fingers to move the map';\n this._cooperativeGesturesScreen.innerHTML = `\n
${desktopMessage}
\n
${mobileMessage}
\n `;\n\n // Remove cooperative gesture screen from the accessibility tree since screenreaders cannot interact with the map using gestures\n this._cooperativeGesturesScreen.setAttribute('aria-hidden', 'true');\n\n // Add event to canvas container since gesture container is pointer-events: none\n this._canvasContainer.addEventListener('wheel', this._cooperativeGesturesOnWheel, false);\n\n // Add a cooperative gestures class (enable touch-action: pan-x pan-y;)\n this._canvasContainer.classList.add('maplibregl-cooperative-gestures');\n }\n\n _destroyCooperativeGestures() {\n DOM.remove(this._cooperativeGesturesScreen);\n this._canvasContainer.removeEventListener('wheel', this._cooperativeGesturesOnWheel, false);\n this._canvasContainer.classList.remove('maplibregl-cooperative-gestures');\n }\n\n _resizeCanvas(width: number, height: number, pixelRatio: number) {\n // Request the required canvas size taking the pixelratio into account.\n this._canvas.width = Math.floor(pixelRatio * width);\n this._canvas.height = Math.floor(pixelRatio * height);\n\n // Maintain the same canvas size, potentially downscaling it for HiDPI displays\n this._canvas.style.width = `${width}px`;\n this._canvas.style.height = `${height}px`;\n }\n\n _setupPainter() {\n\n const attributes = {\n alpha: true,\n stencil: true,\n depth: true,\n failIfMajorPerformanceCaveat: this._failIfMajorPerformanceCaveat,\n preserveDrawingBuffer: this._preserveDrawingBuffer,\n antialias: this._antialias || false\n };\n\n let webglcontextcreationerrorDetailObject: any = null;\n this._canvas.addEventListener('webglcontextcreationerror', (args: WebGLContextEvent) => {\n webglcontextcreationerrorDetailObject = {requestedAttributes: attributes};\n if (args) {\n webglcontextcreationerrorDetailObject.statusMessage = args.statusMessage;\n webglcontextcreationerrorDetailObject.type = args.type;\n }\n }, {once: true});\n\n const gl =\n this._canvas.getContext('webgl2', attributes) as WebGL2RenderingContext ||\n this._canvas.getContext('webgl', attributes) as WebGLRenderingContext;\n\n if (!gl) {\n const msg = 'Failed to initialize WebGL';\n if (webglcontextcreationerrorDetailObject) {\n webglcontextcreationerrorDetailObject.message = msg;\n throw new Error(JSON.stringify(webglcontextcreationerrorDetailObject));\n } else {\n throw new Error(msg);\n }\n }\n\n this.painter = new Painter(gl, this.transform);\n\n webpSupported.testSupport(gl);\n }\n\n _contextLost = (event: any) => {\n event.preventDefault();\n if (this._frame) {\n this._frame.cancel();\n this._frame = null;\n }\n this.fire(new Event('webglcontextlost', {originalEvent: event}));\n };\n\n _contextRestored = (event: any) => {\n this._setupPainter();\n this.resize();\n this._update();\n this.fire(new Event('webglcontextrestored', {originalEvent: event}));\n };\n\n _onMapScroll = (event: any) => {\n if (event.target !== this._container) return;\n\n // Revert any scroll which would move the canvas outside of the view\n this._container.scrollTop = 0;\n this._container.scrollLeft = 0;\n return false;\n };\n\n _onCooperativeGesture(event: any, metaPress, touches) {\n if (!metaPress && touches < 2) {\n // Alert user how to scroll/pan\n this._cooperativeGesturesScreen.classList.add('maplibregl-show');\n setTimeout(() => {\n this._cooperativeGesturesScreen.classList.remove('maplibregl-show');\n }, 100);\n }\n return false;\n }\n\n /**\n * Returns a Boolean indicating whether the map is fully loaded.\n *\n * Returns `false` if the style is not yet fully loaded,\n * or if there has been a change to the sources or style that\n * has not yet fully loaded.\n *\n * @returns A Boolean indicating whether the map is fully loaded.\n */\n loaded(): boolean {\n return !this._styleDirty && !this._sourcesDirty && !!this.style && this.style.loaded();\n }\n\n /**\n * @internal\n * Update this map's style and sources, and re-render the map.\n *\n * @param updateStyle - mark the map's style for reprocessing as\n * well as its sources\n * @returns `this`\n */\n _update(updateStyle?: boolean) {\n if (!this.style || !this.style._loaded) return this;\n\n this._styleDirty = this._styleDirty || updateStyle;\n this._sourcesDirty = true;\n this.triggerRepaint();\n\n return this;\n }\n\n /**\n * @internal\n * Request that the given callback be executed during the next render\n * frame. Schedule a render frame if one is not already scheduled.\n *\n * @returns An id that can be used to cancel the callback\n */\n _requestRenderFrame(callback: () => void): TaskID {\n this._update();\n return this._renderTaskQueue.add(callback);\n }\n\n _cancelRenderFrame(id: TaskID) {\n this._renderTaskQueue.remove(id);\n }\n\n /**\n * @internal\n * Call when a (re-)render of the map is required:\n * - The style has changed (`setPaintProperty()`, etc.)\n * - Source data has changed (e.g. tiles have finished loading)\n * - The map has is moving (or just finished moving)\n * - A transition is in progress\n *\n * @param paintStartTimeStamp - The time when the animation frame began executing.\n *\n * @returns `this`\n */\n _render(paintStartTimeStamp: number) {\n const fadeDuration = this._idleTriggered ? this._fadeDuration : 0;\n\n // A custom layer may have used the context asynchronously. Mark the state as dirty.\n this.painter.context.setDirty();\n this.painter.setBaseState();\n\n this._renderTaskQueue.run(paintStartTimeStamp);\n // A task queue callback may have fired a user event which may have removed the map\n if (this._removed) return;\n\n let crossFading = false;\n\n // If the style has changed, the map is being zoomed, or a transition or fade is in progress:\n // - Apply style changes (in a batch)\n // - Recalculate paint properties.\n if (this.style && this._styleDirty) {\n this._styleDirty = false;\n\n const zoom = this.transform.zoom;\n const now = browser.now();\n this.style.zoomHistory.update(zoom, now);\n\n const parameters = new EvaluationParameters(zoom, {\n now,\n fadeDuration,\n zoomHistory: this.style.zoomHistory,\n transition: this.style.getTransition()\n });\n\n const factor = parameters.crossFadingFactor();\n if (factor !== 1 || factor !== this._crossFadingFactor) {\n crossFading = true;\n this._crossFadingFactor = factor;\n }\n\n this.style.update(parameters);\n }\n\n // If we are in _render for any reason other than an in-progress paint\n // transition, update source caches to check for and load any tiles we\n // need for the current transform\n if (this.style && this._sourcesDirty) {\n this._sourcesDirty = false;\n this.style._updateSources(this.transform);\n }\n\n // update terrain stuff\n if (this.terrain) {\n this.terrain.sourceCache.update(this.transform, this.terrain);\n this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n if (!this._elevationFreeze) {\n this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom);\n }\n } else {\n this.transform._minEleveationForCurrentTile = 0;\n this.transform.elevation = 0;\n }\n\n this._placementDirty = this.style && this.style._updatePlacement(this.painter.transform, this.showCollisionBoxes, fadeDuration, this._crossSourceCollisions);\n\n // Actually draw\n this.painter.render(this.style, {\n showTileBoundaries: this.showTileBoundaries,\n showOverdrawInspector: this._showOverdrawInspector,\n rotating: this.isRotating(),\n zooming: this.isZooming(),\n moving: this.isMoving(),\n fadeDuration,\n showPadding: this.showPadding,\n });\n\n this.fire(new Event('render'));\n\n if (this.loaded() && !this._loaded) {\n this._loaded = true;\n PerformanceUtils.mark(PerformanceMarkers.load);\n this.fire(new Event('load'));\n }\n\n if (this.style && (this.style.hasTransitions() || crossFading)) {\n this._styleDirty = true;\n }\n\n if (this.style && !this._placementDirty) {\n // Since no fade operations are in progress, we can release\n // all tiles held for fading. If we didn't do this, the tiles\n // would just sit in the SourceCaches until the next render\n this.style._releaseSymbolFadeTiles();\n }\n\n // Schedule another render frame if it's needed.\n //\n // Even though `_styleDirty` and `_sourcesDirty` are reset in this\n // method, synchronous events fired during Style#update or\n // Style#_updateSources could have caused them to be set again.\n const somethingDirty = this._sourcesDirty || this._styleDirty || this._placementDirty;\n if (somethingDirty || this._repaint) {\n this.triggerRepaint();\n } else if (!this.isMoving() && this.loaded()) {\n this.fire(new Event('idle'));\n }\n\n if (this._loaded && !this._fullyLoaded && !somethingDirty) {\n this._fullyLoaded = true;\n PerformanceUtils.mark(PerformanceMarkers.fullLoad);\n }\n\n return this;\n }\n\n /**\n * Force a synchronous redraw of the map.\n * @returns `this`\n * @example\n * ```ts\n * map.redraw();\n * ```\n */\n redraw(): this {\n if (this.style) {\n // cancel the scheduled update\n if (this._frame) {\n this._frame.cancel();\n this._frame = null;\n }\n this._render(0);\n }\n return this;\n }\n\n /**\n * Clean up and release all internal resources associated with this map.\n *\n * This includes DOM elements, event bindings, web workers, and WebGL resources.\n *\n * Use this method when you are done using the map and wish to ensure that it no\n * longer consumes browser resources. Afterwards, you must not call any other\n * methods on the map.\n */\n remove() {\n if (this._hash) this._hash.remove();\n\n for (const control of this._controls) control.onRemove(this);\n this._controls = [];\n\n if (this._frame) {\n this._frame.cancel();\n this._frame = null;\n }\n this._renderTaskQueue.clear();\n this.painter.destroy();\n this.handlers.destroy();\n delete this.handlers;\n this.setStyle(null);\n if (typeof window !== 'undefined') {\n removeEventListener('online', this._onWindowOnline, false);\n }\n\n ImageRequest.removeThrottleControl(this._imageQueueHandle);\n\n this._resizeObserver?.disconnect();\n const extension = this.painter.context.gl.getExtension('WEBGL_lose_context');\n if (extension) extension.loseContext();\n this._canvas.removeEventListener('webglcontextrestored', this._contextRestored, false);\n this._canvas.removeEventListener('webglcontextlost', this._contextLost, false);\n DOM.remove(this._canvasContainer);\n DOM.remove(this._controlContainer);\n if (this._cooperativeGestures) {\n this._destroyCooperativeGestures();\n }\n this._container.classList.remove('maplibregl-map');\n\n PerformanceUtils.clearMetrics();\n\n this._removed = true;\n this.fire(new Event('remove'));\n }\n\n /**\n * Trigger the rendering of a single frame. Use this method with custom layers to\n * repaint the map when the layer changes. Calling this multiple times before the\n * next frame is rendered will still result in only a single frame being rendered.\n * @example\n * ```ts\n * map.triggerRepaint();\n * ```\n * @see [Add a 3D model](https://maplibre.org/maplibre-gl-js/docs/examples/add-3d-model/)\n * @see [Add an animated icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-animated/)\n */\n triggerRepaint() {\n if (this.style && !this._frame) {\n this._frame = browser.frame((paintStartTimeStamp: number) => {\n PerformanceUtils.frame(paintStartTimeStamp);\n this._frame = null;\n this._render(paintStartTimeStamp);\n });\n }\n }\n\n _onWindowOnline = () => {\n this._update();\n };\n\n /**\n * Gets and sets a Boolean indicating whether the map will render an outline\n * around each tile and the tile ID. These tile boundaries are useful for\n * debugging.\n *\n * The uncompressed file size of the first vector source is drawn in the top left\n * corner of each tile, next to the tile ID.\n *\n * @example\n * ```ts\n * map.showTileBoundaries = true;\n * ```\n */\n get showTileBoundaries(): boolean { return !!this._showTileBoundaries; }\n set showTileBoundaries(value: boolean) {\n if (this._showTileBoundaries === value) return;\n this._showTileBoundaries = value;\n this._update();\n }\n\n /**\n * Gets and sets a Boolean indicating whether the map will visualize\n * the padding offsets.\n */\n get showPadding(): boolean { return !!this._showPadding; }\n set showPadding(value: boolean) {\n if (this._showPadding === value) return;\n this._showPadding = value;\n this._update();\n }\n\n /**\n * Gets and sets a Boolean indicating whether the map will render boxes\n * around all symbols in the data source, revealing which symbols\n * were rendered or which were hidden due to collisions.\n * This information is useful for debugging.\n */\n get showCollisionBoxes(): boolean { return !!this._showCollisionBoxes; }\n set showCollisionBoxes(value: boolean) {\n if (this._showCollisionBoxes === value) return;\n this._showCollisionBoxes = value;\n if (value) {\n // When we turn collision boxes on we have to generate them for existing tiles\n // When we turn them off, there's no cost to leaving existing boxes in place\n this.style._generateCollisionBoxes();\n } else {\n // Otherwise, call an update to remove collision boxes\n this._update();\n }\n }\n\n /**\n * Gets and sets a Boolean indicating whether the map should color-code\n * each fragment to show how many times it has been shaded.\n * White fragments have been shaded 8 or more times.\n * Black fragments have been shaded 0 times.\n * This information is useful for debugging.\n */\n get showOverdrawInspector(): boolean { return !!this._showOverdrawInspector; }\n set showOverdrawInspector(value: boolean) {\n if (this._showOverdrawInspector === value) return;\n this._showOverdrawInspector = value;\n this._update();\n }\n\n /**\n * Gets and sets a Boolean indicating whether the map will\n * continuously repaint. This information is useful for analyzing performance.\n */\n get repaint(): boolean { return !!this._repaint; }\n set repaint(value: boolean) {\n if (this._repaint !== value) {\n this._repaint = value;\n this.triggerRepaint();\n }\n }\n // show vertices\n get vertices(): boolean { return !!this._vertices; }\n set vertices(value: boolean) { this._vertices = value; this._update(); }\n\n /**\n * Returns the package version of the library\n * @returns Package version of the library\n */\n get version(): string {\n return version;\n }\n\n /**\n * Returns the elevation for the point where the camera is looking.\n * This value corresponds to:\n * \"meters above sea level\" * \"exaggeration\"\n * @returns The elevation.\n */\n getCameraTargetElevation(): number {\n return this.transform.elevation;\n }\n}\n","import type Point from '@mapbox/point-geometry';\n\nimport {DragMoveHandler, DragRotateResult, DragPitchResult, DragHandler} from './drag_handler';\nimport {OneFingerTouchMoveStateManager} from './drag_move_state_manager';\n\nexport interface OneFingerTouchRotateHandler extends DragMoveHandler {}\nexport interface OneFingerTouchPitchHandler extends DragMoveHandler {}\n\nconst assignEvents = (handler: DragHandler) => {\n handler.touchstart = handler.dragStart;\n handler.touchmoveWindow = handler.dragMove;\n handler.touchend = handler.dragEnd;\n};\n\nexport const generateOneFingerTouchRotationHandler = ({enable, clickTolerance, bearingDegreesPerPixelMoved = 0.8}: {\n clickTolerance: number;\n bearingDegreesPerPixelMoved?: number;\n enable?: boolean;\n}): OneFingerTouchRotateHandler => {\n const touchMoveStateManager = new OneFingerTouchMoveStateManager();\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({bearingDelta: (point.x - lastPoint.x) * bearingDegreesPerPixelMoved}),\n moveStateManager: touchMoveStateManager,\n enable,\n assignEvents,\n });\n};\n\nexport const generateOneFingerTouchPitchHandler = ({enable, clickTolerance, pitchDegreesPerPixelMoved = -0.5}: {\n clickTolerance: number;\n pitchDegreesPerPixelMoved?: number;\n enable?: boolean;\n}): OneFingerTouchPitchHandler => {\n const touchMoveStateManager = new OneFingerTouchMoveStateManager();\n return new DragHandler({\n clickTolerance,\n move: (lastPoint: Point, point: Point) =>\n ({pitchDelta: (point.y - lastPoint.y) * pitchDegreesPerPixelMoved}),\n moveStateManager: touchMoveStateManager,\n enable,\n assignEvents,\n });\n};\n","import Point from '@mapbox/point-geometry';\n\nimport {DOM} from '../../util/dom';\nimport {extend} from '../../util/util';\nimport {generateMousePitchHandler, generateMouseRotationHandler, MousePitchHandler, MouseRotateHandler} from '../handler/mouse';\nimport {generateOneFingerTouchPitchHandler, generateOneFingerTouchRotationHandler, OneFingerTouchPitchHandler, OneFingerTouchRotateHandler} from '../handler/one_finger_touch_drag';\n\nimport type {Map} from '../map';\nimport type {IControl} from './control';\n\n/**\n * The {@link NavigationControl} options object\n */\ntype NavigationOptions = {\n /**\n * If `true` the compass button is included.\n */\n showCompass?: boolean;\n /**\n * If `true` the zoom-in and zoom-out buttons are included.\n */\n showZoom?: boolean;\n /**\n * If `true` the pitch is visualized by rotating X-axis of compass.\n */\n visualizePitch?: boolean;\n};\n\nconst defaultOptions: NavigationOptions = {\n showCompass: true,\n showZoom: true,\n visualizePitch: false\n};\n\n/**\n * A `NavigationControl` control contains zoom buttons and a compass.\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * let nav = new maplibregl.NavigationControl();\n * map.addControl(nav, 'top-left');\n * ```\n * @see [Display map navigation controls](https://maplibre.org/maplibre-gl-js/docs/examples/navigation/)\n */\nexport class NavigationControl implements IControl {\n _map: Map;\n options: NavigationOptions;\n _container: HTMLElement;\n _zoomInButton: HTMLButtonElement;\n _zoomOutButton: HTMLButtonElement;\n _compass: HTMLButtonElement;\n _compassIcon: HTMLElement;\n _handler: MouseRotateWrapper;\n\n /**\n * @param options - the control's options\n */\n constructor(options?: NavigationOptions) {\n this.options = extend({}, defaultOptions, options);\n\n this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group');\n this._container.addEventListener('contextmenu', (e) => e.preventDefault());\n\n if (this.options.showZoom) {\n this._zoomInButton = this._createButton('maplibregl-ctrl-zoom-in', (e) => this._map.zoomIn({}, {originalEvent: e}));\n DOM.create('span', 'maplibregl-ctrl-icon', this._zoomInButton).setAttribute('aria-hidden', 'true');\n this._zoomOutButton = this._createButton('maplibregl-ctrl-zoom-out', (e) => this._map.zoomOut({}, {originalEvent: e}));\n DOM.create('span', 'maplibregl-ctrl-icon', this._zoomOutButton).setAttribute('aria-hidden', 'true');\n }\n if (this.options.showCompass) {\n this._compass = this._createButton('maplibregl-ctrl-compass', (e) => {\n if (this.options.visualizePitch) {\n this._map.resetNorthPitch({}, {originalEvent: e});\n } else {\n this._map.resetNorth({}, {originalEvent: e});\n }\n });\n this._compassIcon = DOM.create('span', 'maplibregl-ctrl-icon', this._compass);\n this._compassIcon.setAttribute('aria-hidden', 'true');\n }\n }\n\n _updateZoomButtons = () => {\n const zoom = this._map.getZoom();\n const isMax = zoom === this._map.getMaxZoom();\n const isMin = zoom === this._map.getMinZoom();\n this._zoomInButton.disabled = isMax;\n this._zoomOutButton.disabled = isMin;\n this._zoomInButton.setAttribute('aria-disabled', isMax.toString());\n this._zoomOutButton.setAttribute('aria-disabled', isMin.toString());\n };\n\n _rotateCompassArrow = () => {\n const rotate = this.options.visualizePitch ?\n `scale(${1 / Math.pow(Math.cos(this._map.transform.pitch * (Math.PI / 180)), 0.5)}) rotateX(${this._map.transform.pitch}deg) rotateZ(${this._map.transform.angle * (180 / Math.PI)}deg)` :\n `rotate(${this._map.transform.angle * (180 / Math.PI)}deg)`;\n\n this._compassIcon.style.transform = rotate;\n };\n\n onAdd(map: Map) {\n this._map = map;\n if (this.options.showZoom) {\n this._setButtonTitle(this._zoomInButton, 'ZoomIn');\n this._setButtonTitle(this._zoomOutButton, 'ZoomOut');\n this._map.on('zoom', this._updateZoomButtons);\n this._updateZoomButtons();\n }\n if (this.options.showCompass) {\n this._setButtonTitle(this._compass, 'ResetBearing');\n if (this.options.visualizePitch) {\n this._map.on('pitch', this._rotateCompassArrow);\n }\n this._map.on('rotate', this._rotateCompassArrow);\n this._rotateCompassArrow();\n this._handler = new MouseRotateWrapper(this._map, this._compass, this.options.visualizePitch);\n }\n return this._container;\n }\n\n onRemove() {\n DOM.remove(this._container);\n if (this.options.showZoom) {\n this._map.off('zoom', this._updateZoomButtons);\n }\n if (this.options.showCompass) {\n if (this.options.visualizePitch) {\n this._map.off('pitch', this._rotateCompassArrow);\n }\n this._map.off('rotate', this._rotateCompassArrow);\n this._handler.off();\n delete this._handler;\n }\n\n delete this._map;\n }\n\n _createButton(className: string, fn: (e?: any) => unknown) {\n const a = DOM.create('button', className, this._container) as HTMLButtonElement;\n a.type = 'button';\n a.addEventListener('click', fn);\n return a;\n }\n\n _setButtonTitle = (button: HTMLButtonElement, title: string) => {\n const str = this._map._getUIString(`NavigationControl.${title}`);\n button.title = str;\n button.setAttribute('aria-label', str);\n };\n}\n\nclass MouseRotateWrapper {\n\n map: Map;\n _clickTolerance: number;\n element: HTMLElement;\n // Rotation and pitch handlers are separated due to different _clickTolerance values\n mouseRotate: MouseRotateHandler;\n touchRotate: OneFingerTouchRotateHandler;\n mousePitch: MousePitchHandler;\n touchPitch: OneFingerTouchPitchHandler;\n _startPos: Point;\n _lastPos: Point;\n\n constructor(map: Map, element: HTMLElement, pitch: boolean = false) {\n this._clickTolerance = 10;\n const mapRotateTolerance = map.dragRotate._mouseRotate.getClickTolerance();\n const mapPitchTolerance = map.dragRotate._mousePitch.getClickTolerance();\n this.element = element;\n this.mouseRotate = generateMouseRotationHandler({clickTolerance: mapRotateTolerance, enable: true});\n this.touchRotate = generateOneFingerTouchRotationHandler({clickTolerance: mapRotateTolerance, enable: true});\n this.map = map;\n if (pitch) {\n this.mousePitch = generateMousePitchHandler({clickTolerance: mapPitchTolerance, enable: true});\n this.touchPitch = generateOneFingerTouchPitchHandler({clickTolerance: mapPitchTolerance, enable: true});\n }\n\n DOM.addEventListener(element, 'mousedown', this.mousedown);\n DOM.addEventListener(element, 'touchstart', this.touchstart, {passive: false});\n DOM.addEventListener(element, 'touchcancel', this.reset);\n }\n\n startMouse(e: MouseEvent, point: Point) {\n this.mouseRotate.dragStart(e, point);\n if (this.mousePitch) this.mousePitch.dragStart(e, point);\n DOM.disableDrag();\n }\n\n startTouch(e: TouchEvent, point: Point) {\n this.touchRotate.dragStart(e, point);\n if (this.touchPitch) this.touchPitch.dragStart(e, point);\n DOM.disableDrag();\n }\n\n moveMouse(e: MouseEvent, point: Point) {\n const map = this.map;\n const {bearingDelta} = this.mouseRotate.dragMove(e, point) || {};\n if (bearingDelta) map.setBearing(map.getBearing() + bearingDelta);\n if (this.mousePitch) {\n const {pitchDelta} = this.mousePitch.dragMove(e, point) || {};\n if (pitchDelta) map.setPitch(map.getPitch() + pitchDelta);\n }\n }\n\n moveTouch(e: TouchEvent, point: Point) {\n const map = this.map;\n const {bearingDelta} = this.touchRotate.dragMove(e, point) || {};\n if (bearingDelta) map.setBearing(map.getBearing() + bearingDelta);\n if (this.touchPitch) {\n const {pitchDelta} = this.touchPitch.dragMove(e, point) || {};\n if (pitchDelta) map.setPitch(map.getPitch() + pitchDelta);\n }\n }\n\n off() {\n const element = this.element;\n DOM.removeEventListener(element, 'mousedown', this.mousedown);\n DOM.removeEventListener(element, 'touchstart', this.touchstart, {passive: false});\n DOM.removeEventListener(window, 'touchmove', this.touchmove, {passive: false});\n DOM.removeEventListener(window, 'touchend', this.touchend);\n DOM.removeEventListener(element, 'touchcancel', this.reset);\n this.offTemp();\n }\n\n offTemp() {\n DOM.enableDrag();\n DOM.removeEventListener(window, 'mousemove', this.mousemove);\n DOM.removeEventListener(window, 'mouseup', this.mouseup);\n DOM.removeEventListener(window, 'touchmove', this.touchmove, {passive: false});\n DOM.removeEventListener(window, 'touchend', this.touchend);\n }\n\n mousedown = (e: MouseEvent) => {\n this.startMouse(extend({}, e, {ctrlKey: true, preventDefault: () => e.preventDefault()}), DOM.mousePos(this.element, e));\n DOM.addEventListener(window, 'mousemove', this.mousemove);\n DOM.addEventListener(window, 'mouseup', this.mouseup);\n };\n\n mousemove = (e: MouseEvent) => {\n this.moveMouse(e, DOM.mousePos(this.element, e));\n };\n\n mouseup = (e: MouseEvent) => {\n this.mouseRotate.dragEnd(e);\n if (this.mousePitch) this.mousePitch.dragEnd(e);\n this.offTemp();\n };\n\n touchstart = (e: TouchEvent) => {\n if (e.targetTouches.length !== 1) {\n this.reset();\n } else {\n this._startPos = this._lastPos = DOM.touchPos(this.element, e.targetTouches)[0];\n this.startTouch(e, this._startPos);\n DOM.addEventListener(window, 'touchmove', this.touchmove, {passive: false});\n DOM.addEventListener(window, 'touchend', this.touchend);\n }\n };\n\n touchmove = (e: TouchEvent) => {\n if (e.targetTouches.length !== 1) {\n this.reset();\n } else {\n this._lastPos = DOM.touchPos(this.element, e.targetTouches)[0];\n this.moveTouch(e, this._lastPos);\n }\n };\n\n touchend = (e: TouchEvent) => {\n if (e.targetTouches.length === 0 &&\n this._startPos &&\n this._lastPos &&\n this._startPos.dist(this._lastPos) < this._clickTolerance) {\n this.element.click();\n }\n delete this._startPos;\n delete this._lastPos;\n this.offTemp();\n };\n\n reset = () => {\n this.mouseRotate.reset();\n if (this.mousePitch) this.mousePitch.reset();\n this.touchRotate.reset();\n if (this.touchPitch) this.touchPitch.reset();\n delete this._startPos;\n delete this._lastPos;\n this.offTemp();\n };\n}\n","let supportsGeolocation;\n\nexport function checkGeolocationSupport(callback: (supported: boolean) => void, forceRecalculation = false): void {\n if (supportsGeolocation !== undefined && !forceRecalculation) {\n callback(supportsGeolocation);\n } else if (window.navigator.permissions !== undefined) {\n // navigator.permissions has incomplete browser support\n // http://caniuse.com/#feat=permissions-api\n // Test for the case where a browser disables Geolocation because of an\n // insecure origin\n window.navigator.permissions.query({name: 'geolocation'}).then((p) => {\n supportsGeolocation = p.state !== 'denied';\n callback(supportsGeolocation);\n }).catch(() => {\n // Fix for iOS16 which rejects query but still supports geolocation\n supportsGeolocation = !!window.navigator.geolocation;\n callback(supportsGeolocation);\n });\n\n } else {\n supportsGeolocation = !!window.navigator.geolocation;\n callback(supportsGeolocation);\n }\n}\n","import {LngLat} from '../geo/lng_lat';\n\nimport type Point from '@mapbox/point-geometry';\nimport type {Transform} from '../geo/transform';\n\n/**\n * Given a LngLat, prior projected position, and a transform, return a new LngLat shifted\n * n × 360° east or west for some n ≥ 0 such that:\n *\n * * the projected location of the result is on screen, if possible, and secondarily:\n * * the difference between the projected location of the result and the prior position\n * is minimized.\n *\n * The object is to preserve perceived object constancy for Popups and Markers as much as\n * possible; they should avoid shifting large distances across the screen, even when the\n * map center changes by ±360° due to automatic wrapping, and when about to go off screen,\n * should wrap just enough to avoid doing so.\n */\nexport function smartWrap(lngLat: LngLat, priorPos: Point, transform: Transform): LngLat {\n lngLat = new LngLat(lngLat.lng, lngLat.lat);\n\n // First, try shifting one world in either direction, and see if either is closer to the\n // prior position. This preserves object constancy when the map center is auto-wrapped\n // during animations.\n if (priorPos) {\n const left = new LngLat(lngLat.lng - 360, lngLat.lat);\n const right = new LngLat(lngLat.lng + 360, lngLat.lat);\n const delta = transform.locationPoint(lngLat).distSqr(priorPos);\n if (transform.locationPoint(left).distSqr(priorPos) < delta) {\n lngLat = left;\n } else if (transform.locationPoint(right).distSqr(priorPos) < delta) {\n lngLat = right;\n }\n }\n\n // Second, wrap toward the center until the new position is on screen, or we can't get\n // any closer.\n while (Math.abs(lngLat.lng - transform.center.lng) > 180) {\n const pos = transform.locationPoint(lngLat);\n if (pos.x >= 0 && pos.y >= 0 && pos.x <= transform.width && pos.y <= transform.height) {\n break;\n }\n if (lngLat.lng > transform.center.lng) {\n lngLat.lng -= 360;\n } else {\n lngLat.lng += 360;\n }\n }\n\n return lngLat;\n}\n","/**\n * Where to position the anchor.\n * Used by a popup and a marker.\n */\nexport type PositionAnchor = 'center' | 'top' | 'bottom' | 'left' | 'right' | 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right';\n\nexport const anchorTranslate: {\n [_ in PositionAnchor]: string;\n} = {\n 'center': 'translate(-50%,-50%)',\n 'top': 'translate(-50%,0)',\n 'top-left': 'translate(0,0)',\n 'top-right': 'translate(-100%,0)',\n 'bottom': 'translate(-50%,-100%)',\n 'bottom-left': 'translate(0,-100%)',\n 'bottom-right': 'translate(-100%,-100%)',\n 'left': 'translate(0,-50%)',\n 'right': 'translate(-100%,-50%)'\n};\n\nexport function applyAnchorClass(element: HTMLElement, anchor: PositionAnchor, prefix: string) {\n const classList = element.classList;\n for (const key in anchorTranslate) {\n classList.remove(`maplibregl-${prefix}-anchor-${key}`);\n }\n classList.add(`maplibregl-${prefix}-anchor-${anchor}`);\n}\n","import {DOM} from '../util/dom';\nimport {LngLat} from '../geo/lng_lat';\nimport Point from '@mapbox/point-geometry';\nimport {smartWrap} from '../util/smart_wrap';\nimport {anchorTranslate, applyAnchorClass} from './anchor';\nimport type {PositionAnchor} from './anchor';\nimport {Event, Evented} from '../util/evented';\nimport type {Map} from './map';\nimport {Popup, Offset} from './popup';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport type {MapMouseEvent, MapTouchEvent} from './events';\nimport type {PointLike} from './camera';\n\n/**\n * Alignment options of rotation and pitch\n */\ntype Alignment = 'map' | 'viewport' | 'auto';\n\n/**\n * The {@link Marker} options object\n */\ntype MarkerOptions = {\n /**\n * DOM element to use as a marker. The default is a light blue, droplet-shaped SVG marker.\n */\n element?: HTMLElement;\n /**\n * Space-separated CSS class names to add to marker element.\n */\n className?: string;\n /**\n * The offset in pixels as a {@link PointLike} object to apply relative to the element's center. Negatives indicate left and up.\n */\n offset?: PointLike;\n /**\n * A string indicating the part of the Marker that should be positioned closest to the coordinate set via {@link Marker#setLngLat}.\n * Options are `'center'`, `'top'`, `'bottom'`, `'left'`, `'right'`, `'top-left'`, `'top-right'`, `'bottom-left'`, and `'bottom-right'`.\n * @defaultValue 'center'\n * */\n anchor?: PositionAnchor;\n /**\n * The color to use for the default marker if options.element is not provided. The default is light blue.\n * @defaultValue '#3FB1CE'\n */\n color?: string;\n /**\n * The scale to use for the default marker if options.element is not provided. The default scale corresponds to a height of `41px` and a width of `27px`.\n * @defaultValue 1\n */\n scale?: number;\n /**\n * A boolean indicating whether or not a marker is able to be dragged to a new position on the map.\n * @defaultValue false\n */\n draggable?: boolean;\n /**\n * The max number of pixels a user can shift the mouse pointer during a click on the marker for it to be considered a valid click (as opposed to a marker drag). The default is to inherit map's clickTolerance.\n * @defaultValue 0\n */\n clickTolerance?: number;\n /**\n * The rotation angle of the marker in degrees, relative to its respective `rotationAlignment` setting. A positive value will rotate the marker clockwise.\n * @defaultValue 0\n */\n rotation?: number;\n /**\n * `map` aligns the `Marker`'s rotation relative to the map, maintaining a bearing as the map rotates. `viewport` aligns the `Marker`'s rotation relative to the viewport, agnostic to map rotations. `auto` is equivalent to `viewport`.\n * @defaultValue 'auto'\n */\n rotationAlignment?: Alignment;\n /**\n * `map` aligns the `Marker` to the plane of the map. `viewport` aligns the `Marker` to the plane of the viewport. `auto` automatically matches the value of `rotationAlignment`.\n * @defaultValue 'auto'\n */\n pitchAlignment?: Alignment;\n};\n\n/**\n * Creates a marker component\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([30.5, 50.5])\n * .addTo(map);\n * ```\n *\n * @example\n * Set options\n * ```ts\n * let marker = new maplibregl.Marker({\n * color: \"#FFFFFF\",\n * draggable: true\n * }).setLngLat([30.5, 50.5])\n * .addTo(map);\n * ```\n * @see [Add custom icons with Markers](https://maplibre.org/maplibre-gl-js/docs/examples/custom-marker-icons/)\n * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/)\n *\n * ### Events\n *\n * @event `dragstart` Fired when dragging starts, `marker` object that is being dragged\n *\n * @event `drag` Fired while dragging. `marker` object that is being dragged\n *\n * @event `dragend` Fired when the marker is finished being dragged, `marker` object that was dragged\n */\nexport class Marker extends Evented {\n _map: Map;\n _anchor: PositionAnchor;\n _offset: Point;\n _element: HTMLElement;\n _popup: Popup;\n _lngLat: LngLat;\n _pos: Point;\n _color: string;\n _scale: number;\n _defaultMarker: boolean;\n _draggable: boolean;\n _clickTolerance: number;\n _isDragging: boolean;\n _state: 'inactive' | 'pending' | 'active'; // used for handling drag events\n _positionDelta: Point;\n _pointerdownPos: Point;\n _rotation: number;\n _pitchAlignment: Alignment;\n _rotationAlignment: Alignment;\n _originalTabIndex: string; // original tabindex of _element\n _opacityTimeout: ReturnType;\n\n /**\n * @param options - the options\n */\n constructor(options?: MarkerOptions) {\n super();\n\n this._anchor = options && options.anchor || 'center';\n this._color = options && options.color || '#3FB1CE';\n this._scale = options && options.scale || 1;\n this._draggable = options && options.draggable || false;\n this._clickTolerance = options && options.clickTolerance || 0;\n this._isDragging = false;\n this._state = 'inactive';\n this._rotation = options && options.rotation || 0;\n this._rotationAlignment = options && options.rotationAlignment || 'auto';\n this._pitchAlignment = options && options.pitchAlignment && options.pitchAlignment !== 'auto' ? options.pitchAlignment : this._rotationAlignment;\n\n if (!options || !options.element) {\n this._defaultMarker = true;\n this._element = DOM.create('div');\n this._element.setAttribute('aria-label', 'Map marker');\n\n // create default map marker SVG\n const svg = DOM.createNS('http://www.w3.org/2000/svg', 'svg');\n const defaultHeight = 41;\n const defaultWidth = 27;\n svg.setAttributeNS(null, 'display', 'block');\n svg.setAttributeNS(null, 'height', `${defaultHeight}px`);\n svg.setAttributeNS(null, 'width', `${defaultWidth}px`);\n svg.setAttributeNS(null, 'viewBox', `0 0 ${defaultWidth} ${defaultHeight}`);\n\n const markerLarge = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n markerLarge.setAttributeNS(null, 'stroke', 'none');\n markerLarge.setAttributeNS(null, 'stroke-width', '1');\n markerLarge.setAttributeNS(null, 'fill', 'none');\n markerLarge.setAttributeNS(null, 'fill-rule', 'evenodd');\n\n const page1 = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n page1.setAttributeNS(null, 'fill-rule', 'nonzero');\n\n const shadow = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n shadow.setAttributeNS(null, 'transform', 'translate(3.0, 29.0)');\n shadow.setAttributeNS(null, 'fill', '#000000');\n\n const ellipses = [\n {'rx': '10.5', 'ry': '5.25002273'},\n {'rx': '10.5', 'ry': '5.25002273'},\n {'rx': '9.5', 'ry': '4.77275007'},\n {'rx': '8.5', 'ry': '4.29549936'},\n {'rx': '7.5', 'ry': '3.81822308'},\n {'rx': '6.5', 'ry': '3.34094679'},\n {'rx': '5.5', 'ry': '2.86367051'},\n {'rx': '4.5', 'ry': '2.38636864'}\n ];\n\n for (const data of ellipses) {\n const ellipse = DOM.createNS('http://www.w3.org/2000/svg', 'ellipse');\n ellipse.setAttributeNS(null, 'opacity', '0.04');\n ellipse.setAttributeNS(null, 'cx', '10.5');\n ellipse.setAttributeNS(null, 'cy', '5.80029008');\n ellipse.setAttributeNS(null, 'rx', data['rx']);\n ellipse.setAttributeNS(null, 'ry', data['ry']);\n shadow.appendChild(ellipse);\n }\n\n const background = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n background.setAttributeNS(null, 'fill', this._color);\n\n const bgPath = DOM.createNS('http://www.w3.org/2000/svg', 'path');\n bgPath.setAttributeNS(null, 'd', 'M27,13.5 C27,19.074644 20.250001,27.000002 14.75,34.500002 C14.016665,35.500004 12.983335,35.500004 12.25,34.500002 C6.7499993,27.000002 0,19.222562 0,13.5 C0,6.0441559 6.0441559,0 13.5,0 C20.955844,0 27,6.0441559 27,13.5 Z');\n\n background.appendChild(bgPath);\n\n const border = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n border.setAttributeNS(null, 'opacity', '0.25');\n border.setAttributeNS(null, 'fill', '#000000');\n\n const borderPath = DOM.createNS('http://www.w3.org/2000/svg', 'path');\n borderPath.setAttributeNS(null, 'd', 'M13.5,0 C6.0441559,0 0,6.0441559 0,13.5 C0,19.222562 6.7499993,27 12.25,34.5 C13,35.522727 14.016664,35.500004 14.75,34.5 C20.250001,27 27,19.074644 27,13.5 C27,6.0441559 20.955844,0 13.5,0 Z M13.5,1 C20.415404,1 26,6.584596 26,13.5 C26,15.898657 24.495584,19.181431 22.220703,22.738281 C19.945823,26.295132 16.705119,30.142167 13.943359,33.908203 C13.743445,34.180814 13.612715,34.322738 13.5,34.441406 C13.387285,34.322738 13.256555,34.180814 13.056641,33.908203 C10.284481,30.127985 7.4148684,26.314159 5.015625,22.773438 C2.6163816,19.232715 1,15.953538 1,13.5 C1,6.584596 6.584596,1 13.5,1 Z');\n\n border.appendChild(borderPath);\n\n const maki = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n maki.setAttributeNS(null, 'transform', 'translate(6.0, 7.0)');\n maki.setAttributeNS(null, 'fill', '#FFFFFF');\n\n const circleContainer = DOM.createNS('http://www.w3.org/2000/svg', 'g');\n circleContainer.setAttributeNS(null, 'transform', 'translate(8.0, 8.0)');\n\n const circle1 = DOM.createNS('http://www.w3.org/2000/svg', 'circle');\n circle1.setAttributeNS(null, 'fill', '#000000');\n circle1.setAttributeNS(null, 'opacity', '0.25');\n circle1.setAttributeNS(null, 'cx', '5.5');\n circle1.setAttributeNS(null, 'cy', '5.5');\n circle1.setAttributeNS(null, 'r', '5.4999962');\n\n const circle2 = DOM.createNS('http://www.w3.org/2000/svg', 'circle');\n circle2.setAttributeNS(null, 'fill', '#FFFFFF');\n circle2.setAttributeNS(null, 'cx', '5.5');\n circle2.setAttributeNS(null, 'cy', '5.5');\n circle2.setAttributeNS(null, 'r', '5.4999962');\n\n circleContainer.appendChild(circle1);\n circleContainer.appendChild(circle2);\n\n page1.appendChild(shadow);\n page1.appendChild(background);\n page1.appendChild(border);\n page1.appendChild(maki);\n page1.appendChild(circleContainer);\n\n svg.appendChild(page1);\n\n svg.setAttributeNS(null, 'height', `${defaultHeight * this._scale}px`);\n svg.setAttributeNS(null, 'width', `${defaultWidth * this._scale}px`);\n\n this._element.appendChild(svg);\n\n // if no element and no offset option given apply an offset for the default marker\n // the -14 as the y value of the default marker offset was determined as follows\n //\n // the marker tip is at the center of the shadow ellipse from the default svg\n // the y value of the center of the shadow ellipse relative to the svg top left is \"shadow transform translate-y (29.0) + ellipse cy (5.80029008)\"\n // offset to the svg center \"height (41 / 2)\" gives (29.0 + 5.80029008) - (41 / 2) and rounded for an integer pixel offset gives 14\n // negative is used to move the marker up from the center so the tip is at the Marker lngLat\n this._offset = Point.convert(options && options.offset || [0, -14]);\n } else {\n this._element = options.element;\n this._offset = Point.convert(options && options.offset || [0, 0]);\n }\n\n this._element.classList.add('maplibregl-marker');\n this._element.addEventListener('dragstart', (e: DragEvent) => {\n e.preventDefault();\n });\n this._element.addEventListener('mousedown', (e: MouseEvent) => {\n // prevent focusing on click\n e.preventDefault();\n });\n applyAnchorClass(this._element, this._anchor, 'marker');\n\n if (options && options.className) {\n for (const name of options.className.split(' ')) {\n this._element.classList.add(name);\n }\n }\n\n this._popup = null;\n }\n\n /**\n * Attaches the `Marker` to a `Map` object.\n * @param map - The MapLibre GL JS map to add the marker to.\n * @returns `this`\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([30.5, 50.5])\n * .addTo(map); // add the marker to the map\n * ```\n */\n addTo(map: Map): this {\n this.remove();\n this._map = map;\n map.getCanvasContainer().appendChild(this._element);\n map.on('move', this._update);\n map.on('moveend', this._update);\n map.on('terrain', this._update);\n\n this.setDraggable(this._draggable);\n this._update();\n\n // If we attached the `click` listener to the marker element, the popup\n // would close once the event propagated to `map` due to the\n // `Popup#_onClickClose` listener.\n this._map.on('click', this._onMapClick);\n\n return this;\n }\n\n /**\n * Removes the marker from a map\n * @example\n * ```ts\n * let marker = new maplibregl.Marker().addTo(map);\n * marker.remove();\n * ```\n * @returns `this`\n */\n remove(): this {\n if (this._opacityTimeout) {\n clearTimeout(this._opacityTimeout);\n delete this._opacityTimeout;\n }\n if (this._map) {\n this._map.off('click', this._onMapClick);\n this._map.off('move', this._update);\n this._map.off('moveend', this._update);\n this._map.off('mousedown', this._addDragHandler);\n this._map.off('touchstart', this._addDragHandler);\n this._map.off('mouseup', this._onUp);\n this._map.off('touchend', this._onUp);\n this._map.off('mousemove', this._onMove);\n this._map.off('touchmove', this._onMove);\n delete this._map;\n }\n DOM.remove(this._element);\n if (this._popup) this._popup.remove();\n return this;\n }\n\n /**\n * Get the marker's geographical location.\n *\n * The longitude of the result may differ by a multiple of 360 degrees from the longitude previously\n * set by `setLngLat` because `Marker` wraps the anchor longitude across copies of the world to keep\n * the marker on screen.\n *\n * @returns A {@link LngLat} describing the marker's location.\n * @example\n * ```ts\n * // Store the marker's longitude and latitude coordinates in a variable\n * let lngLat = marker.getLngLat();\n * // Print the marker's longitude and latitude values in the console\n * console.log('Longitude: ' + lngLat.lng + ', Latitude: ' + lngLat.lat )\n * ```\n * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/)\n */\n getLngLat(): LngLat {\n return this._lngLat;\n }\n\n /**\n * Set the marker's geographical position and move it.\n * @param lnglat - A {@link LngLat} describing where the marker should be located.\n * @returns `this`\n * @example\n * Create a new marker, set the longitude and latitude, and add it to the map\n * ```ts\n * new maplibregl.Marker()\n * .setLngLat([-65.017, -16.457])\n * .addTo(map);\n * ```\n * @see [Add custom icons with Markers](https://maplibre.org/maplibre-gl-js/docs/examples/custom-marker-icons/)\n * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/)\n */\n setLngLat(lnglat: LngLatLike): this {\n this._lngLat = LngLat.convert(lnglat);\n this._pos = null;\n if (this._popup) this._popup.setLngLat(this._lngLat);\n this._update();\n return this;\n }\n\n /**\n * Returns the `Marker`'s HTML element.\n * @returns element\n */\n getElement(): HTMLElement {\n return this._element;\n }\n\n /**\n * Binds a {@link Popup} to the {@link Marker}.\n * @param popup - An instance of the {@link Popup} class. If undefined or null, any popup\n * set on this {@link Marker} instance is unset.\n * @returns `this`\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([0, 0])\n * .setPopup(new maplibregl.Popup().setHTML(\"

Hello World!

\")) // add popup\n * .addTo(map);\n * ```\n * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/)\n */\n setPopup(popup?: Popup | null): this {\n if (this._popup) {\n this._popup.remove();\n this._popup = null;\n this._element.removeEventListener('keypress', this._onKeyPress);\n\n if (!this._originalTabIndex) {\n this._element.removeAttribute('tabindex');\n }\n }\n\n if (popup) {\n if (!('offset' in popup.options)) {\n const markerHeight = 41 - (5.8 / 2);\n const markerRadius = 13.5;\n const linearOffset = Math.abs(markerRadius) / Math.SQRT2;\n popup.options.offset = this._defaultMarker ? {\n 'top': [0, 0],\n 'top-left': [0, 0],\n 'top-right': [0, 0],\n 'bottom': [0, -markerHeight],\n 'bottom-left': [linearOffset, (markerHeight - markerRadius + linearOffset) * -1],\n 'bottom-right': [-linearOffset, (markerHeight - markerRadius + linearOffset) * -1],\n 'left': [markerRadius, (markerHeight - markerRadius) * -1],\n 'right': [-markerRadius, (markerHeight - markerRadius) * -1]\n } as Offset : this._offset;\n }\n this._popup = popup;\n if (this._lngLat) this._popup.setLngLat(this._lngLat);\n\n this._originalTabIndex = this._element.getAttribute('tabindex');\n if (!this._originalTabIndex) {\n this._element.setAttribute('tabindex', '0');\n }\n this._element.addEventListener('keypress', this._onKeyPress);\n }\n\n return this;\n }\n\n _onKeyPress = (e: KeyboardEvent) => {\n const code = e.code;\n const legacyCode = e.charCode || e.keyCode;\n\n if (\n (code === 'Space') || (code === 'Enter') ||\n (legacyCode === 32) || (legacyCode === 13) // space or enter\n ) {\n this.togglePopup();\n }\n };\n\n _onMapClick = (e: MapMouseEvent) => {\n const targetElement = e.originalEvent.target;\n const element = this._element;\n\n if (this._popup && (targetElement === element || element.contains(targetElement as any))) {\n this.togglePopup();\n }\n };\n\n /**\n * Returns the {@link Popup} instance that is bound to the {@link Marker}.\n * @returns popup\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([0, 0])\n * .setPopup(new maplibregl.Popup().setHTML(\"

Hello World!

\"))\n * .addTo(map);\n *\n * console.log(marker.getPopup()); // return the popup instance\n * ```\n */\n getPopup(): Popup {\n return this._popup;\n }\n\n /**\n * Opens or closes the {@link Popup} instance that is bound to the {@link Marker}, depending on the current state of the {@link Popup}.\n * @returns `this`\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * .setLngLat([0, 0])\n * .setPopup(new maplibregl.Popup().setHTML(\"

Hello World!

\"))\n * .addTo(map);\n *\n * marker.togglePopup(); // toggle popup open or closed\n * ```\n */\n togglePopup(): this {\n const popup = this._popup;\n\n if (!popup) return this;\n else if (popup.isOpen()) popup.remove();\n else popup.addTo(this._map);\n return this;\n }\n\n _update = (e?: { type: 'move' | 'moveend' | 'terrain' | 'render' }) => {\n if (!this._map) return;\n\n const isFullyLoaded = this._map.loaded() && !this._map.isMoving();\n if (e?.type === 'terrain' || (e?.type === 'render' && !isFullyLoaded)) {\n this._map.once('render', this._update);\n }\n\n if (this._map.transform.renderWorldCopies) {\n this._lngLat = smartWrap(this._lngLat, this._pos, this._map.transform);\n }\n\n this._pos = this._map.project(this._lngLat)._add(this._offset);\n\n let rotation = '';\n if (this._rotationAlignment === 'viewport' || this._rotationAlignment === 'auto') {\n rotation = `rotateZ(${this._rotation}deg)`;\n } else if (this._rotationAlignment === 'map') {\n rotation = `rotateZ(${this._rotation - this._map.getBearing()}deg)`;\n }\n\n let pitch = '';\n if (this._pitchAlignment === 'viewport' || this._pitchAlignment === 'auto') {\n pitch = 'rotateX(0deg)';\n } else if (this._pitchAlignment === 'map') {\n pitch = `rotateX(${this._map.getPitch()}deg)`;\n }\n\n // because rounding the coordinates at every `move` event causes stuttered zooming\n // we only round them when _update is called with `moveend` or when its called with\n // no arguments (when the Marker is initialized or Marker#setLngLat is invoked).\n if (!e || e.type === 'moveend') {\n this._pos = this._pos.round();\n }\n\n DOM.setTransform(this._element, `${anchorTranslate[this._anchor]} translate(${this._pos.x}px, ${this._pos.y}px) ${pitch} ${rotation}`);\n\n // in case of 3D, ask the terrain coords-framebuffer for this pos and check if the marker is visible\n // call this logic in setTimeout with a timeout of 100ms to save performance in map-movement\n if (this._map.terrain && !this._opacityTimeout) this._opacityTimeout = setTimeout(() => {\n const lnglat = this._map.unproject(this._pos);\n const metresPerPixel = 40075016.686 * Math.abs(Math.cos(this._lngLat.lat * Math.PI / 180)) / Math.pow(2, this._map.transform.tileZoom + 8);\n this._element.style.opacity = lnglat.distanceTo(this._lngLat) > metresPerPixel * 20 ? '0.2' : '1.0';\n this._opacityTimeout = null;\n }, 100);\n };\n\n /**\n * Get the marker's offset.\n * @returns The marker's screen coordinates in pixels.\n */\n getOffset(): Point {\n return this._offset;\n }\n\n /**\n * Sets the offset of the marker\n * @param offset - The offset in pixels as a {@link PointLike} object to apply relative to the element's center. Negatives indicate left and up.\n * @returns `this`\n */\n setOffset(offset: PointLike): this {\n this._offset = Point.convert(offset);\n this._update();\n return this;\n }\n\n /**\n * Adds a CSS class to the marker element.\n *\n * @param className - on-empty string with CSS class name to add to marker element\n *\n * @example\n * ```\n * let marker = new maplibregl.Marker()\n * marker.addClassName('some-class')\n * ```\n */\n addClassName(className: string) {\n this._element.classList.add(className);\n }\n\n /**\n * Removes a CSS class from the marker element.\n *\n * @param className - Non-empty string with CSS class name to remove from marker element\n *\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * marker.removeClassName('some-class')\n * ```\n */\n removeClassName(className: string) {\n this._element.classList.remove(className);\n }\n\n /**\n * Add or remove the given CSS class on the marker element, depending on whether the element currently has that class.\n *\n * @param className - Non-empty string with CSS class name to add/remove\n *\n * @returns if the class was removed return false, if class was added, then return true\n *\n * @example\n * ```ts\n * let marker = new maplibregl.Marker()\n * marker.toggleClassName('toggleClass')\n * ```\n */\n toggleClassName(className: string): boolean {\n return this._element.classList.toggle(className);\n }\n\n _onMove = (e: MapMouseEvent | MapTouchEvent) => {\n if (!this._isDragging) {\n const clickTolerance = this._clickTolerance || this._map._clickTolerance;\n this._isDragging = e.point.dist(this._pointerdownPos) >= clickTolerance;\n }\n if (!this._isDragging) return;\n\n this._pos = e.point.sub(this._positionDelta);\n this._lngLat = this._map.unproject(this._pos);\n this.setLngLat(this._lngLat);\n // suppress click event so that popups don't toggle on drag\n this._element.style.pointerEvents = 'none';\n\n // make sure dragstart only fires on the first move event after mousedown.\n // this can't be on mousedown because that event doesn't necessarily\n // imply that a drag is about to happen.\n if (this._state === 'pending') {\n this._state = 'active';\n this.fire(new Event('dragstart'));\n }\n this.fire(new Event('drag'));\n };\n\n _onUp = () => {\n // revert to normal pointer event handling\n this._element.style.pointerEvents = 'auto';\n this._positionDelta = null;\n this._pointerdownPos = null;\n this._isDragging = false;\n this._map.off('mousemove', this._onMove);\n this._map.off('touchmove', this._onMove);\n\n // only fire dragend if it was preceded by at least one drag event\n if (this._state === 'active') {\n this.fire(new Event('dragend'));\n }\n\n this._state = 'inactive';\n };\n\n _addDragHandler = (e: MapMouseEvent | MapTouchEvent) => {\n if (this._element.contains(e.originalEvent.target as any)) {\n e.preventDefault();\n\n // We need to calculate the pixel distance between the click point\n // and the marker position, with the offset accounted for. Then we\n // can subtract this distance from the mousemove event's position\n // to calculate the new marker position.\n // If we don't do this, the marker 'jumps' to the click position\n // creating a jarring UX effect.\n this._positionDelta = e.point.sub(this._pos).add(this._offset);\n\n this._pointerdownPos = e.point;\n\n this._state = 'pending';\n this._map.on('mousemove', this._onMove);\n this._map.on('touchmove', this._onMove);\n this._map.once('mouseup', this._onUp);\n this._map.once('touchend', this._onUp);\n }\n };\n\n /**\n * Sets the `draggable` property and functionality of the marker\n * @param shouldBeDraggable - Turns drag functionality on/off\n * @returns `this`\n */\n setDraggable(shouldBeDraggable?: boolean): this {\n this._draggable = !!shouldBeDraggable; // convert possible undefined value to false\n\n // handle case where map may not exist yet\n // e.g. when setDraggable is called before addTo\n if (this._map) {\n if (shouldBeDraggable) {\n this._map.on('mousedown', this._addDragHandler);\n this._map.on('touchstart', this._addDragHandler);\n } else {\n this._map.off('mousedown', this._addDragHandler);\n this._map.off('touchstart', this._addDragHandler);\n }\n }\n\n return this;\n }\n\n /**\n * Returns true if the marker can be dragged\n * @returns True if the marker is draggable.\n */\n isDraggable(): boolean {\n return this._draggable;\n }\n\n /**\n * Sets the `rotation` property of the marker.\n * @param rotation - The rotation angle of the marker (clockwise, in degrees), relative to its respective {@link Marker#setRotationAlignment} setting.\n * @returns `this`\n */\n setRotation(rotation?: number): this {\n this._rotation = rotation || 0;\n this._update();\n return this;\n }\n\n /**\n * Returns the current rotation angle of the marker (in degrees).\n * @returns The current rotation angle of the marker.\n */\n getRotation(): number {\n return this._rotation;\n }\n\n /**\n * Sets the `rotationAlignment` property of the marker.\n * @param alignment - Sets the `rotationAlignment` property of the marker. defaults to 'auto'\n * @returns `this`\n */\n setRotationAlignment(alignment?: Alignment): this {\n this._rotationAlignment = alignment || 'auto';\n this._update();\n return this;\n }\n\n /**\n * Returns the current `rotationAlignment` property of the marker.\n * @returns The current rotational alignment of the marker.\n */\n getRotationAlignment(): Alignment {\n return this._rotationAlignment;\n }\n\n /**\n * Sets the `pitchAlignment` property of the marker.\n * @param alignment - Sets the `pitchAlignment` property of the marker. If alignment is 'auto', it will automatically match `rotationAlignment`.\n * @returns `this`\n */\n setPitchAlignment(alignment?: Alignment): this {\n this._pitchAlignment = alignment && alignment !== 'auto' ? alignment : this._rotationAlignment;\n this._update();\n return this;\n }\n\n /**\n * Returns the current `pitchAlignment` property of the marker.\n * @returns The current pitch alignment of the marker in degrees.\n */\n getPitchAlignment(): Alignment {\n return this._pitchAlignment;\n }\n}\n","import {Event, Evented} from '../../util/evented';\nimport {DOM} from '../../util/dom';\nimport {extend, warnOnce} from '../../util/util';\nimport {checkGeolocationSupport} from '../../util/geolocation_support';\nimport {LngLat} from '../../geo/lng_lat';\nimport {Marker} from '../marker';\n\nimport type {Map} from '../map';\nimport type {FitBoundsOptions} from '../camera';\nimport type {IControl} from './control';\nimport {LngLatBounds} from '../../geo/lng_lat_bounds';\n\n/**\n * The {@link GeolocateControl} options\n */\ntype GeolocateOptions = {\n /**\n * A Geolocation API [PositionOptions](https://developer.mozilla.org/en-US/docs/Web/API/PositionOptions) object.\n * @defaultValue `{enableHighAccuracy: false, timeout: 6000}`\n */\n positionOptions?: PositionOptions;\n /**\n * A options object to use when the map is panned and zoomed to the user's location. The default is to use a `maxZoom` of 15 to limit how far the map will zoom in for very accurate locations.\n */\n fitBoundsOptions?: FitBoundsOptions;\n /**\n * If `true` the Geolocate Control becomes a toggle button and when active the map will receive updates to the user's location as it changes.\n * @defaultValue false\n */\n trackUserLocation?: boolean;\n /**\n * By default, if showUserLocation is `true`, a transparent circle will be drawn around the user location indicating the accuracy (95% confidence level) of the user's location. Set to `false` to disable. Always disabled when showUserLocation is `false`.\n * @defaultValue true\n */\n showAccuracyCircle?: boolean;\n /**\n * By default a dot will be shown on the map at the user's location. Set to `false` to disable.\n * @defaultValue true\n */\n showUserLocation?: boolean;\n};\n\nconst defaultOptions: GeolocateOptions = {\n positionOptions: {\n enableHighAccuracy: false,\n maximumAge: 0,\n timeout: 6000 /* 6 sec */\n },\n fitBoundsOptions: {\n maxZoom: 15\n },\n trackUserLocation: false,\n showAccuracyCircle: true,\n showUserLocation: true\n};\n\nlet numberOfWatches = 0;\nlet noTimeout = false;\n\n/**\n * A `GeolocateControl` control provides a button that uses the browser's geolocation\n * API to locate the user on the map.\n *\n * Not all browsers support geolocation,\n * and some users may disable the feature. Geolocation support for modern\n * browsers including Chrome requires sites to be served over HTTPS. If\n * geolocation support is not available, the GeolocateControl will show\n * as disabled.\n *\n * The zoom level applied will depend on the accuracy of the geolocation provided by the device.\n *\n * The GeolocateControl has two modes. If `trackUserLocation` is `false` (default) the control acts as a button, which when pressed will set the map's camera to target the user location. If the user moves, the map won't update. This is most suited for the desktop. If `trackUserLocation` is `true` the control acts as a toggle button that when active the user's location is actively monitored for changes. In this mode the GeolocateControl has three interaction states:\n * * active - the map's camera automatically updates as the user's location changes, keeping the location dot in the center. Initial state and upon clicking the `GeolocateControl` button.\n * * passive - the user's location dot automatically updates, but the map's camera does not. Occurs upon the user initiating a map movement.\n * * disabled - occurs if Geolocation is not available, disabled or denied.\n *\n * These interaction states can't be controlled programmatically, rather they are set based on user interactions.\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * map.addControl(new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * }));\n * ```\n * @see [Locate the user](https://maplibre.org/maplibre-gl-js/docs/examples/locate-user/)\n *\n * ### Events\n *\n * @event `trackuserlocationend` - Fired when the Geolocate Control changes to the background state, which happens when a user changes the camera during an active position lock. This only applies when trackUserLocation is true. In the background state, the dot on the map will update with location updates but the camera will not.\n *\n * @event `trackuserlocationstart` - Fired when the Geolocate Control changes to the active lock state, which happens either upon first obtaining a successful Geolocation API position for the user (a geolocate event will follow), or the user clicks the geolocate button when in the background state which uses the last known position to recenter the map and enter active lock state (no geolocate event will follow unless the users's location changes).\n *\n * @event `geolocate` - Fired on each Geolocation API position update which returned as success.\n * `data` - The returned [Position](https://developer.mozilla.org/en-US/docs/Web/API/Position) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition).\n *\n * @event `error` - Fired on each Geolocation API position update which returned as an error.\n * `data` - The returned [PositionError](https://developer.mozilla.org/en-US/docs/Web/API/PositionError) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition).\n *\n * @event `outofmaxbounds` Fired on each Geolocation API position update which returned as success but user position is out of map maxBounds.\n * `data` - The returned [Position](https://developer.mozilla.org/en-US/docs/Web/API/Position) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition).\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when a trackuserlocationend event occurs.\n * geolocate.on('trackuserlocationend', function() {\n * console.log('A trackuserlocationend event has occurred.')\n * });\n * ```\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when a trackuserlocationstart event occurs.\n * geolocate.on('trackuserlocationstart', function() {\n * console.log('A trackuserlocationstart event has occurred.')\n * });\n * ```\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when a geolocate event occurs.\n * geolocate.on('geolocate', function() {\n * console.log('A geolocate event has occurred.')\n * });\n * ```\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when an error event occurs.\n * geolocate.on('error', function() {\n * console.log('An error event has occurred.')\n * });\n * ```\n *\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * // Set an event listener that fires\n * // when an outofmaxbounds event occurs.\n * geolocate.on('outofmaxbounds', function() {\n * console.log('An outofmaxbounds event has occurred.')\n * });\n * ```\n */\nexport class GeolocateControl extends Evented implements IControl {\n _map: Map;\n options: GeolocateOptions;\n _container: HTMLElement;\n _dotElement: HTMLElement;\n _circleElement: HTMLElement;\n _geolocateButton: HTMLButtonElement;\n _geolocationWatchID: number;\n _timeoutId: ReturnType;\n /* Geolocate Control Watch States\n * This is the private state of the control.\n *\n * OFF\n * off/inactive\n * WAITING_ACTIVE\n * Geolocate Control was clicked but still waiting for Geolocation API response with user location\n * ACTIVE_LOCK\n * Showing the user location as a dot AND tracking the camera to be fixed to their location. If their location changes the map moves to follow.\n * ACTIVE_ERROR\n * There was en error from the Geolocation API while trying to show and track the user location.\n * BACKGROUND\n * Showing the user location as a dot but the camera doesn't follow their location as it changes.\n * BACKGROUND_ERROR\n * There was an error from the Geolocation API while trying to show (but not track) the user location.\n */\n _watchState: 'OFF' | 'ACTIVE_LOCK' | 'WAITING_ACTIVE' | 'ACTIVE_ERROR' | 'BACKGROUND' | 'BACKGROUND_ERROR';\n _lastKnownPosition: any;\n _userLocationDotMarker: Marker;\n _accuracyCircleMarker: Marker;\n _accuracy: number;\n _setup: boolean; // set to true once the control has been setup\n\n constructor(options: GeolocateOptions) {\n super();\n this.options = extend({}, defaultOptions, options);\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group');\n checkGeolocationSupport(this._setupUI);\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n // clear the geolocation watch if exists\n if (this._geolocationWatchID !== undefined) {\n window.navigator.geolocation.clearWatch(this._geolocationWatchID);\n this._geolocationWatchID = undefined;\n }\n\n // clear the markers from the map\n if (this.options.showUserLocation && this._userLocationDotMarker) {\n this._userLocationDotMarker.remove();\n }\n if (this.options.showAccuracyCircle && this._accuracyCircleMarker) {\n this._accuracyCircleMarker.remove();\n }\n\n DOM.remove(this._container);\n this._map.off('zoom', this._onZoom);\n this._map = undefined;\n numberOfWatches = 0;\n noTimeout = false;\n }\n\n /**\n * Check if the Geolocation API Position is outside the map's maxbounds.\n *\n * @param position - the Geolocation API Position\n * @returns `true` if position is outside the map's maxbounds, otherwise returns `false`.\n */\n _isOutOfMapMaxBounds(position: GeolocationPosition) {\n const bounds = this._map.getMaxBounds();\n const coordinates = position.coords;\n\n return bounds && (\n coordinates.longitude < bounds.getWest() ||\n coordinates.longitude > bounds.getEast() ||\n coordinates.latitude < bounds.getSouth() ||\n coordinates.latitude > bounds.getNorth()\n );\n }\n\n _setErrorState() {\n switch (this._watchState) {\n case 'WAITING_ACTIVE':\n this._watchState = 'ACTIVE_ERROR';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active-error');\n break;\n case 'ACTIVE_LOCK':\n this._watchState = 'ACTIVE_ERROR';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active-error');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting');\n // turn marker grey\n break;\n case 'BACKGROUND':\n this._watchState = 'BACKGROUND_ERROR';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background-error');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting');\n // turn marker grey\n break;\n case 'ACTIVE_ERROR':\n break;\n default:\n throw new Error(`Unexpected watchState ${this._watchState}`);\n }\n }\n\n /**\n * When the Geolocation API returns a new location, update the GeolocateControl.\n *\n * @param position - the Geolocation API Position\n */\n _onSuccess = (position: GeolocationPosition) => {\n if (!this._map) {\n // control has since been removed\n return;\n }\n\n if (this._isOutOfMapMaxBounds(position)) {\n this._setErrorState();\n\n this.fire(new Event('outofmaxbounds', position));\n this._updateMarker();\n this._finish();\n\n return;\n }\n\n if (this.options.trackUserLocation) {\n // keep a record of the position so that if the state is BACKGROUND and the user\n // clicks the button, we can move to ACTIVE_LOCK immediately without waiting for\n // watchPosition to trigger _onSuccess\n this._lastKnownPosition = position;\n\n switch (this._watchState) {\n case 'WAITING_ACTIVE':\n case 'ACTIVE_LOCK':\n case 'ACTIVE_ERROR':\n this._watchState = 'ACTIVE_LOCK';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active');\n break;\n case 'BACKGROUND':\n case 'BACKGROUND_ERROR':\n this._watchState = 'BACKGROUND';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background');\n break;\n default:\n throw new Error(`Unexpected watchState ${this._watchState}`);\n }\n }\n\n // if showUserLocation and the watch state isn't off then update the marker location\n if (this.options.showUserLocation && this._watchState !== 'OFF') {\n this._updateMarker(position);\n }\n\n // if in normal mode (not watch mode), or if in watch mode and the state is active watch\n // then update the camera\n if (!this.options.trackUserLocation || this._watchState === 'ACTIVE_LOCK') {\n this._updateCamera(position);\n }\n\n if (this.options.showUserLocation) {\n this._dotElement.classList.remove('maplibregl-user-location-dot-stale');\n }\n\n this.fire(new Event('geolocate', position));\n this._finish();\n };\n\n /**\n * Update the camera location to center on the current position\n *\n * @param position - the Geolocation API Position\n */\n _updateCamera = (position: GeolocationPosition) => {\n const center = new LngLat(position.coords.longitude, position.coords.latitude);\n const radius = position.coords.accuracy;\n const bearing = this._map.getBearing();\n const options = extend({bearing}, this.options.fitBoundsOptions);\n const newBounds = LngLatBounds.fromLngLat(center, radius);\n\n this._map.fitBounds(newBounds, options, {\n geolocateSource: true // tag this camera change so it won't cause the control to change to background state\n });\n };\n\n /**\n * Update the user location dot Marker to the current position\n *\n * @param position - the Geolocation API Position\n */\n _updateMarker = (position?: GeolocationPosition | null) => {\n if (position) {\n const center = new LngLat(position.coords.longitude, position.coords.latitude);\n this._accuracyCircleMarker.setLngLat(center).addTo(this._map);\n this._userLocationDotMarker.setLngLat(center).addTo(this._map);\n this._accuracy = position.coords.accuracy;\n if (this.options.showUserLocation && this.options.showAccuracyCircle) {\n this._updateCircleRadius();\n }\n } else {\n this._userLocationDotMarker.remove();\n this._accuracyCircleMarker.remove();\n }\n };\n\n _updateCircleRadius() {\n const bounds = this._map.getBounds();\n const southEastPoint = bounds.getSouthEast();\n const northEastPoint = bounds.getNorthEast();\n const mapHeightInMeters = southEastPoint.distanceTo(northEastPoint);\n const mapHeightInPixels = this._map._container.clientHeight;\n const circleDiameter = Math.ceil(2 * (this._accuracy / (mapHeightInMeters / mapHeightInPixels)));\n this._circleElement.style.width = `${circleDiameter}px`;\n this._circleElement.style.height = `${circleDiameter}px`;\n }\n\n _onZoom = () => {\n if (this.options.showUserLocation && this.options.showAccuracyCircle) {\n this._updateCircleRadius();\n }\n };\n\n _onError = (error: GeolocationPositionError) => {\n if (!this._map) {\n // control has since been removed\n return;\n }\n\n if (this.options.trackUserLocation) {\n if (error.code === 1) {\n // PERMISSION_DENIED\n this._watchState = 'OFF';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error');\n this._geolocateButton.disabled = true;\n const title = this._map._getUIString('GeolocateControl.LocationNotAvailable');\n this._geolocateButton.title = title;\n this._geolocateButton.setAttribute('aria-label', title);\n\n if (this._geolocationWatchID !== undefined) {\n this._clearWatch();\n }\n } else if (error.code === 3 && noTimeout) {\n // this represents a forced error state\n // this was triggered to force immediate geolocation when a watch is already present\n // see https://github.com/mapbox/mapbox-gl-js/issues/8214\n // and https://w3c.github.io/geolocation-api/#example-5-forcing-the-user-agent-to-return-a-fresh-cached-position\n return;\n } else {\n this._setErrorState();\n }\n }\n\n if (this._watchState !== 'OFF' && this.options.showUserLocation) {\n this._dotElement.classList.add('maplibregl-user-location-dot-stale');\n }\n\n this.fire(new Event('error', error));\n\n this._finish();\n };\n\n _finish = () => {\n if (this._timeoutId) { clearTimeout(this._timeoutId); }\n this._timeoutId = undefined;\n };\n\n _setupUI = (supported: boolean) => {\n // this method is called asynchronously during onAdd\n // the control could have been removed before reaching here\n if (!this._map) {\n return;\n }\n\n this._container.addEventListener('contextmenu', (e: MouseEvent) => e.preventDefault());\n this._geolocateButton = DOM.create('button', 'maplibregl-ctrl-geolocate', this._container);\n DOM.create('span', 'maplibregl-ctrl-icon', this._geolocateButton).setAttribute('aria-hidden', 'true');\n this._geolocateButton.type = 'button';\n\n if (supported === false) {\n warnOnce('Geolocation support is not available so the GeolocateControl will be disabled.');\n const title = this._map._getUIString('GeolocateControl.LocationNotAvailable');\n this._geolocateButton.disabled = true;\n this._geolocateButton.title = title;\n this._geolocateButton.setAttribute('aria-label', title);\n } else {\n const title = this._map._getUIString('GeolocateControl.FindMyLocation');\n this._geolocateButton.title = title;\n this._geolocateButton.setAttribute('aria-label', title);\n }\n\n if (this.options.trackUserLocation) {\n this._geolocateButton.setAttribute('aria-pressed', 'false');\n this._watchState = 'OFF';\n }\n\n // when showUserLocation is enabled, keep the Geolocate button disabled until the device location marker is setup on the map\n if (this.options.showUserLocation) {\n this._dotElement = DOM.create('div', 'maplibregl-user-location-dot');\n\n this._userLocationDotMarker = new Marker({element: this._dotElement});\n\n this._circleElement = DOM.create('div', 'maplibregl-user-location-accuracy-circle');\n this._accuracyCircleMarker = new Marker({element: this._circleElement, pitchAlignment: 'map'});\n\n if (this.options.trackUserLocation) this._watchState = 'OFF';\n\n this._map.on('zoom', this._onZoom);\n }\n\n this._geolocateButton.addEventListener('click',\n this.trigger.bind(this));\n\n this._setup = true;\n\n // when the camera is changed (and it's not as a result of the Geolocation Control) change\n // the watch mode to background watch, so that the marker is updated but not the camera.\n if (this.options.trackUserLocation) {\n this._map.on('movestart', (event: any) => {\n const fromResize = event.originalEvent && event.originalEvent.type === 'resize';\n if (!event.geolocateSource && this._watchState === 'ACTIVE_LOCK' && !fromResize) {\n this._watchState = 'BACKGROUND';\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n\n this.fire(new Event('trackuserlocationend'));\n }\n });\n }\n };\n\n /**\n * Programmatically request and move the map to the user's location.\n *\n * @returns `false` if called before control was added to a map, otherwise returns `true`.\n * @example\n * ```ts\n * // Initialize the geolocate control.\n * let geolocate = new maplibregl.GeolocateControl({\n * positionOptions: {\n * enableHighAccuracy: true\n * },\n * trackUserLocation: true\n * });\n * // Add the control to the map.\n * map.addControl(geolocate);\n * map.on('load', function() {\n * geolocate.trigger();\n * });\n * ```\n */\n trigger(): boolean {\n if (!this._setup) {\n warnOnce('Geolocate control triggered before added to a map');\n return false;\n }\n if (this.options.trackUserLocation) {\n // update watchState and do any outgoing state cleanup\n switch (this._watchState) {\n case 'OFF':\n // turn on the Geolocate Control\n this._watchState = 'WAITING_ACTIVE';\n\n this.fire(new Event('trackuserlocationstart'));\n break;\n case 'WAITING_ACTIVE':\n case 'ACTIVE_LOCK':\n case 'ACTIVE_ERROR':\n case 'BACKGROUND_ERROR':\n // turn off the Geolocate Control\n numberOfWatches--;\n noTimeout = false;\n this._watchState = 'OFF';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background');\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error');\n\n this.fire(new Event('trackuserlocationend'));\n break;\n case 'BACKGROUND':\n this._watchState = 'ACTIVE_LOCK';\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background');\n // set camera to last known location\n if (this._lastKnownPosition) this._updateCamera(this._lastKnownPosition);\n\n this.fire(new Event('trackuserlocationstart'));\n break;\n default:\n throw new Error(`Unexpected watchState ${this._watchState}`);\n }\n\n // incoming state setup\n switch (this._watchState) {\n case 'WAITING_ACTIVE':\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active');\n break;\n case 'ACTIVE_LOCK':\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active');\n break;\n case 'OFF':\n break;\n default:\n throw new Error(`Unexpected watchState ${this._watchState}`);\n }\n\n // manage geolocation.watchPosition / geolocation.clearWatch\n if (this._watchState === 'OFF' && this._geolocationWatchID !== undefined) {\n // clear watchPosition as we've changed to an OFF state\n this._clearWatch();\n } else if (this._geolocationWatchID === undefined) {\n // enable watchPosition since watchState is not OFF and there is no watchPosition already running\n\n this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.setAttribute('aria-pressed', 'true');\n\n numberOfWatches++;\n let positionOptions;\n if (numberOfWatches > 1) {\n positionOptions = {maximumAge: 600000, timeout: 0};\n noTimeout = true;\n } else {\n positionOptions = this.options.positionOptions;\n noTimeout = false;\n }\n\n this._geolocationWatchID = window.navigator.geolocation.watchPosition(\n this._onSuccess, this._onError, positionOptions);\n }\n } else {\n window.navigator.geolocation.getCurrentPosition(\n this._onSuccess, this._onError, this.options.positionOptions);\n\n // This timeout ensures that we still call finish() even if\n // the user declines to share their location in Firefox\n this._timeoutId = setTimeout(this._finish, 10000 /* 10sec */);\n }\n\n return true;\n }\n\n _clearWatch() {\n window.navigator.geolocation.clearWatch(this._geolocationWatchID);\n\n this._geolocationWatchID = undefined;\n this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting');\n this._geolocateButton.setAttribute('aria-pressed', 'false');\n\n if (this.options.showUserLocation) {\n this._updateMarker(null);\n }\n }\n}\n\n","import {DOM} from '../../util/dom';\nimport {extend} from '../../util/util';\n\nimport type {Map} from '../map';\nimport type {ControlPosition, IControl} from './control';\n\n/**\n * The unit type for length to use for the {@link ScaleControl}\n */\nexport type Unit = 'imperial' | 'metric' | 'nautical';\n\n/**\n * The {@link ScaleControl} options object\n */\ntype ScaleOptions = {\n /**\n * The maximum length of the scale control in pixels.\n * @defaultValue 100\n */\n maxWidth?: number;\n /**\n * Unit of the distance (`'imperial'`, `'metric'` or `'nautical'`).\n * @defaultValue 'metric'\n */\n unit?: Unit;\n};\n\nconst defaultOptions: ScaleOptions = {\n maxWidth: 100,\n unit: 'metric'\n};\n\n/**\n * A `ScaleControl` control displays the ratio of a distance on the map to the corresponding distance on the ground.\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * let scale = new maplibregl.ScaleControl({\n * maxWidth: 80,\n * unit: 'imperial'\n * });\n * map.addControl(scale);\n *\n * scale.setUnit('metric');\n * ```\n */\nexport class ScaleControl implements IControl {\n _map: Map;\n _container: HTMLElement;\n options: ScaleOptions;\n\n constructor(options: ScaleOptions) {\n this.options = extend({}, defaultOptions, options);\n }\n\n getDefaultPosition(): ControlPosition {\n return 'bottom-left';\n }\n\n _onMove = () => {\n updateScale(this._map, this._container, this.options);\n };\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-scale', map.getContainer());\n\n this._map.on('move', this._onMove);\n this._onMove();\n\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._container);\n this._map.off('move', this._onMove);\n this._map = undefined;\n }\n\n /**\n * Set the scale's unit of the distance\n *\n * @param unit - Unit of the distance (`'imperial'`, `'metric'` or `'nautical'`).\n */\n setUnit = (unit: Unit) => {\n this.options.unit = unit;\n updateScale(this._map, this._container, this.options);\n };\n}\n\nfunction updateScale(map, container, options) {\n // A horizontal scale is imagined to be present at center of the map\n // container with maximum length (Default) as 100px.\n // Using spherical law of cosines approximation, the real distance is\n // found between the two coordinates.\n const maxWidth = options && options.maxWidth || 100;\n\n const y = map._container.clientHeight / 2;\n const left = map.unproject([0, y]);\n const right = map.unproject([maxWidth, y]);\n const maxMeters = left.distanceTo(right);\n // The real distance corresponding to 100px scale length is rounded off to\n // near pretty number and the scale length for the same is found out.\n // Default unit of the scale is based on User's locale.\n if (options && options.unit === 'imperial') {\n const maxFeet = 3.2808 * maxMeters;\n if (maxFeet > 5280) {\n const maxMiles = maxFeet / 5280;\n setScale(container, maxWidth, maxMiles, map._getUIString('ScaleControl.Miles'));\n } else {\n setScale(container, maxWidth, maxFeet, map._getUIString('ScaleControl.Feet'));\n }\n } else if (options && options.unit === 'nautical') {\n const maxNauticals = maxMeters / 1852;\n setScale(container, maxWidth, maxNauticals, map._getUIString('ScaleControl.NauticalMiles'));\n } else if (maxMeters >= 1000) {\n setScale(container, maxWidth, maxMeters / 1000, map._getUIString('ScaleControl.Kilometers'));\n } else {\n setScale(container, maxWidth, maxMeters, map._getUIString('ScaleControl.Meters'));\n }\n}\n\nfunction setScale(container, maxWidth, maxDistance, unit) {\n const distance = getRoundNum(maxDistance);\n const ratio = distance / maxDistance;\n container.style.width = `${maxWidth * ratio}px`;\n container.innerHTML = `${distance} ${unit}`;\n}\n\nfunction getDecimalRoundNum(d) {\n const multiplier = Math.pow(10, Math.ceil(-Math.log(d) / Math.LN10));\n return Math.round(d * multiplier) / multiplier;\n}\n\nfunction getRoundNum(num) {\n const pow10 = Math.pow(10, (`${Math.floor(num)}`).length - 1);\n let d = num / pow10;\n\n d = d >= 10 ? 10 :\n d >= 5 ? 5 :\n d >= 3 ? 3 :\n d >= 2 ? 2 :\n d >= 1 ? 1 : getDecimalRoundNum(d);\n\n return pow10 * d;\n}\n","import {extend} from '../util/util';\nimport {Event, Evented} from '../util/evented';\nimport {MapMouseEvent} from '../ui/events';\nimport {DOM} from '../util/dom';\nimport {LngLat} from '../geo/lng_lat';\nimport Point from '@mapbox/point-geometry';\nimport {smartWrap} from '../util/smart_wrap';\nimport {anchorTranslate, applyAnchorClass} from './anchor';\n\nimport type {PositionAnchor} from './anchor';\nimport type {Map} from './map';\nimport type {LngLatLike} from '../geo/lng_lat';\nimport type {PointLike} from './camera';\n\nconst defaultOptions = {\n closeButton: true,\n closeOnClick: true,\n focusAfterOpen: true,\n className: '',\n maxWidth: '240px'\n};\n\n/**\n * A pixel offset specified as:\n * - a single number specifying a distance from the location\n * - a {@link PointLike} specifying a constant offset\n * - an object of {@link Point}s specifying an offset for each anchor position\n * Negative offsets indicate left and up.\n */\nexport type Offset = number | PointLike | {\n [_ in PositionAnchor]: PointLike;\n};\n\nexport type PopupOptions = {\n /**\n * If `true`, a close button will appear in the top right corner of the popup.\n * @defaultValue true\n */\n closeButton?: boolean;\n /**\n * If `true`, the popup will closed when the map is clicked.\n * @defaultValue true\n */\n closeOnClick?: boolean;\n /**\n * If `true`, the popup will closed when the map moves.\n * @defaultValue false\n */\n closeOnMove?: boolean;\n /**\n * If `true`, the popup will try to focus the first focusable element inside the popup.\n * @defaultValue true\n */\n focusAfterOpen?: boolean;\n /**\n * A string indicating the part of the Popup that should\n * be positioned closest to the coordinate set via {@link Popup#setLngLat}.\n * Options are `'center'`, `'top'`, `'bottom'`, `'left'`, `'right'`, `'top-left'`,\n * `'top-right'`, `'bottom-left'`, and `'bottom-right'`. If unset the anchor will be\n * dynamically set to ensure the popup falls within the map container with a preference\n * for `'bottom'`.\n */\n anchor?: PositionAnchor;\n /**\n * A pixel offset applied to the popup's location\n */\n offset?: Offset;\n /**\n * Space-separated CSS class names to add to popup container\n */\n className?: string;\n /**\n * A string that sets the CSS property of the popup's maximum width, eg `'300px'`.\n * To ensure the popup resizes to fit its content, set this property to `'none'`.\n * Available values can be found here: https://developer.mozilla.org/en-US/docs/Web/CSS/max-width\n * @defaultValue '240px'\n */\n maxWidth?: string;\n};\n\nconst focusQuerySelector = [\n 'a[href]',\n '[tabindex]:not([tabindex=\\'-1\\'])',\n '[contenteditable]:not([contenteditable=\\'false\\'])',\n 'button:not([disabled])',\n 'input:not([disabled])',\n 'select:not([disabled])',\n 'textarea:not([disabled])',\n].join(', ');\n\n/**\n * A popup component.\n *\n * @group Markers and Controls\n *\n *\n * @example\n * Create a popup\n * ```ts\n * let popup = new maplibregl.Popup();\n * // Set an event listener that will fire\n * // any time the popup is opened\n * popup.on('open', function(){\n * console.log('popup was opened');\n * });\n * ```\n *\n * @example\n * Create a popup\n * ```ts\n * let popup = new maplibregl.Popup();\n * // Set an event listener that will fire\n * // any time the popup is closed\n * popup.on('close', function(){\n * console.log('popup was closed');\n * });\n * ```\n *\n * @example\n * ```ts\n * let markerHeight = 50, markerRadius = 10, linearOffset = 25;\n * let popupOffsets = {\n * 'top': [0, 0],\n * 'top-left': [0,0],\n * 'top-right': [0,0],\n * 'bottom': [0, -markerHeight],\n * 'bottom-left': [linearOffset, (markerHeight - markerRadius + linearOffset) * -1],\n * 'bottom-right': [-linearOffset, (markerHeight - markerRadius + linearOffset) * -1],\n * 'left': [markerRadius, (markerHeight - markerRadius) * -1],\n * 'right': [-markerRadius, (markerHeight - markerRadius) * -1]\n * };\n * let popup = new maplibregl.Popup({offset: popupOffsets, className: 'my-class'})\n * .setLngLat(e.lngLat)\n * .setHTML(\"

Hello World!

\")\n * .setMaxWidth(\"300px\")\n * .addTo(map);\n * ```\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/)\n *\n * ### Events\n *\n * @event `open` Fired when the popup is opened manually or programmatically. `popup` object that was opened\n *\n * @event `close` Fired when the popup is closed manually or programmatically. `popup` object that was closed\n */\nexport class Popup extends Evented {\n _map: Map;\n options: PopupOptions;\n _content: HTMLElement;\n _container: HTMLElement;\n _closeButton: HTMLButtonElement;\n _tip: HTMLElement;\n _lngLat: LngLat;\n _trackPointer: boolean;\n _pos: Point;\n\n constructor(options?: PopupOptions) {\n super();\n this.options = extend(Object.create(defaultOptions), options);\n }\n\n /**\n * Adds the popup to a map.\n *\n * @param map - The MapLibre GL JS map to add the popup to.\n * @returns `this`\n * @example\n * ```ts\n * new maplibregl.Popup()\n * .setLngLat([0, 0])\n * .setHTML(\"

Null Island

\")\n * .addTo(map);\n * ```\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n * @see [Show polygon information on click](https://maplibre.org/maplibre-gl-js/docs/examples/polygon-popup-on-click/)\n */\n addTo(map: Map): this {\n if (this._map) this.remove();\n\n this._map = map;\n if (this.options.closeOnClick) {\n this._map.on('click', this._onClose);\n }\n\n if (this.options.closeOnMove) {\n this._map.on('move', this._onClose);\n }\n\n this._map.on('remove', this.remove);\n this._update();\n this._focusFirstElement();\n\n if (this._trackPointer) {\n this._map.on('mousemove', this._onMouseMove);\n this._map.on('mouseup', this._onMouseUp);\n if (this._container) {\n this._container.classList.add('maplibregl-popup-track-pointer');\n }\n this._map._canvasContainer.classList.add('maplibregl-track-pointer');\n } else {\n this._map.on('move', this._update);\n }\n\n this.fire(new Event('open'));\n\n return this;\n }\n\n /**\n * @returns `true` if the popup is open, `false` if it is closed.\n */\n isOpen() {\n return !!this._map;\n }\n\n /**\n * Removes the popup from the map it has been added to.\n *\n * @example\n * ```ts\n * let popup = new maplibregl.Popup().addTo(map);\n * popup.remove();\n * ```\n * @returns `this`\n */\n remove = (): this => {\n if (this._content) {\n DOM.remove(this._content);\n }\n\n if (this._container) {\n DOM.remove(this._container);\n delete this._container;\n }\n\n if (this._map) {\n this._map.off('move', this._update);\n this._map.off('move', this._onClose);\n this._map.off('click', this._onClose);\n this._map.off('remove', this.remove);\n this._map.off('mousemove', this._onMouseMove);\n this._map.off('mouseup', this._onMouseUp);\n this._map.off('drag', this._onDrag);\n delete this._map;\n }\n\n this.fire(new Event('close'));\n\n return this;\n };\n\n /**\n * Returns the geographical location of the popup's anchor.\n *\n * The longitude of the result may differ by a multiple of 360 degrees from the longitude previously\n * set by `setLngLat` because `Popup` wraps the anchor longitude across copies of the world to keep\n * the popup on screen.\n *\n * @returns The geographical location of the popup's anchor.\n */\n getLngLat(): LngLat {\n return this._lngLat;\n }\n\n /**\n * Sets the geographical location of the popup's anchor, and moves the popup to it. Replaces trackPointer() behavior.\n *\n * @param lnglat - The geographical location to set as the popup's anchor.\n * @returns `this`\n */\n setLngLat(lnglat: LngLatLike): this {\n this._lngLat = LngLat.convert(lnglat);\n this._pos = null;\n\n this._trackPointer = false;\n\n this._update();\n\n if (this._map) {\n this._map.on('move', this._update);\n this._map.off('mousemove', this._onMouseMove);\n if (this._container) {\n this._container.classList.remove('maplibregl-popup-track-pointer');\n }\n this._map._canvasContainer.classList.remove('maplibregl-track-pointer');\n }\n\n return this;\n }\n\n /**\n * Tracks the popup anchor to the cursor position on screens with a pointer device (it will be hidden on touchscreens). Replaces the `setLngLat` behavior.\n * For most use cases, set `closeOnClick` and `closeButton` to `false`.\n * @example\n * ```ts\n * let popup = new maplibregl.Popup({ closeOnClick: false, closeButton: false })\n * .setHTML(\"

Hello World!

\")\n * .trackPointer()\n * .addTo(map);\n * ```\n * @returns `this`\n */\n trackPointer(): this {\n this._trackPointer = true;\n this._pos = null;\n this._update();\n if (this._map) {\n this._map.off('move', this._update);\n this._map.on('mousemove', this._onMouseMove);\n this._map.on('drag', this._onDrag);\n if (this._container) {\n this._container.classList.add('maplibregl-popup-track-pointer');\n }\n this._map._canvasContainer.classList.add('maplibregl-track-pointer');\n }\n\n return this;\n\n }\n\n /**\n * Returns the `Popup`'s HTML element.\n * @example\n * Change the `Popup` element's font size\n * ```ts\n * let popup = new maplibregl.Popup()\n * .setLngLat([-96, 37.8])\n * .setHTML(\"

Hello World!

\")\n * .addTo(map);\n * let popupElem = popup.getElement();\n * popupElem.style.fontSize = \"25px\";\n * ```\n * @returns element\n */\n getElement(): HTMLElement {\n return this._container;\n }\n\n /**\n * Sets the popup's content to a string of text.\n *\n * This function creates a [Text](https://developer.mozilla.org/en-US/docs/Web/API/Text) node in the DOM,\n * so it cannot insert raw HTML. Use this method for security against XSS\n * if the popup content is user-provided.\n *\n * @param text - Textual content for the popup.\n * @returns `this`\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * .setLngLat(e.lngLat)\n * .setText('Hello, world!')\n * .addTo(map);\n * ```\n */\n setText(text: string): this {\n return this.setDOMContent(document.createTextNode(text));\n }\n\n /**\n * Sets the popup's content to the HTML provided as a string.\n *\n * This method does not perform HTML filtering or sanitization, and must be\n * used only with trusted content. Consider {@link Popup#setText} if\n * the content is an untrusted text string.\n *\n * @param html - A string representing HTML content for the popup.\n * @returns `this`\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * .setLngLat(e.lngLat)\n * .setHTML(\"

Hello World!

\")\n * .addTo(map);\n * ```\n * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/)\n * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/)\n * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/)\n * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/)\n */\n setHTML(html: string): this {\n const frag = document.createDocumentFragment();\n const temp = document.createElement('body');\n let child;\n temp.innerHTML = html;\n while (true) {\n child = temp.firstChild;\n if (!child) break;\n frag.appendChild(child);\n }\n\n return this.setDOMContent(frag);\n }\n\n /**\n * Returns the popup's maximum width.\n *\n * @returns The maximum width of the popup.\n */\n getMaxWidth(): string {\n return this._container?.style.maxWidth;\n }\n\n /**\n * Sets the popup's maximum width. This is setting the CSS property `max-width`.\n * Available values can be found here: https://developer.mozilla.org/en-US/docs/Web/CSS/max-width\n *\n * @param maxWidth - A string representing the value for the maximum width.\n * @returns `this`\n */\n setMaxWidth(maxWidth: string): this {\n this.options.maxWidth = maxWidth;\n this._update();\n return this;\n }\n\n /**\n * Sets the popup's content to the element provided as a DOM node.\n *\n * @param htmlNode - A DOM node to be used as content for the popup.\n * @returns `this`\n * @example\n * Create an element with the popup content\n * ```ts\n * let div = document.createElement('div');\n * div.innerHTML = 'Hello, world!';\n * let popup = new maplibregl.Popup()\n * .setLngLat(e.lngLat)\n * .setDOMContent(div)\n * .addTo(map);\n * ```\n */\n setDOMContent(htmlNode: Node): this {\n if (this._content) {\n // Clear out children first.\n while (this._content.hasChildNodes()) {\n if (this._content.firstChild) {\n this._content.removeChild(this._content.firstChild);\n }\n }\n } else {\n this._content = DOM.create('div', 'maplibregl-popup-content', this._container);\n }\n\n // The close button should be the last tabbable element inside the popup for a good keyboard UX.\n this._content.appendChild(htmlNode);\n this._createCloseButton();\n this._update();\n this._focusFirstElement();\n return this;\n }\n\n /**\n * Adds a CSS class to the popup container element.\n *\n * @param className - Non-empty string with CSS class name to add to popup container\n *\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * popup.addClassName('some-class')\n * ```\n */\n addClassName(className: string) {\n if (this._container) {\n this._container.classList.add(className);\n }\n }\n\n /**\n * Removes a CSS class from the popup container element.\n *\n * @param className - Non-empty string with CSS class name to remove from popup container\n *\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * popup.removeClassName('some-class')\n * ```\n */\n removeClassName(className: string) {\n if (this._container) {\n this._container.classList.remove(className);\n }\n }\n\n /**\n * Sets the popup's offset.\n *\n * @param offset - Sets the popup's offset.\n * @returns `this`\n */\n setOffset (offset?: Offset): this {\n this.options.offset = offset;\n this._update();\n return this;\n }\n\n /**\n * Add or remove the given CSS class on the popup container, depending on whether the container currently has that class.\n *\n * @param className - Non-empty string with CSS class name to add/remove\n *\n * @returns if the class was removed return false, if class was added, then return true, undefined if there is no container\n *\n * @example\n * ```ts\n * let popup = new maplibregl.Popup()\n * popup.toggleClassName('toggleClass')\n * ```\n */\n toggleClassName(className: string): boolean | undefined {\n if (this._container) {\n return this._container.classList.toggle(className);\n }\n }\n\n _createCloseButton() {\n if (this.options.closeButton) {\n this._closeButton = DOM.create('button', 'maplibregl-popup-close-button', this._content);\n this._closeButton.type = 'button';\n this._closeButton.setAttribute('aria-label', 'Close popup');\n this._closeButton.innerHTML = '×';\n this._closeButton.addEventListener('click', this._onClose);\n }\n }\n\n _onMouseUp = (event: MapMouseEvent) => {\n this._update(event.point);\n };\n\n _onMouseMove = (event: MapMouseEvent) => {\n this._update(event.point);\n };\n\n _onDrag = (event: MapMouseEvent) => {\n this._update(event.point);\n };\n\n _update = (cursor?: Point) => {\n const hasPosition = this._lngLat || this._trackPointer;\n\n if (!this._map || !hasPosition || !this._content) { return; }\n\n if (!this._container) {\n this._container = DOM.create('div', 'maplibregl-popup', this._map.getContainer());\n this._tip = DOM.create('div', 'maplibregl-popup-tip', this._container);\n this._container.appendChild(this._content);\n if (this.options.className) {\n for (const name of this.options.className.split(' ')) {\n this._container.classList.add(name);\n }\n }\n\n if (this._trackPointer) {\n this._container.classList.add('maplibregl-popup-track-pointer');\n }\n }\n\n if (this.options.maxWidth && this._container.style.maxWidth !== this.options.maxWidth) {\n this._container.style.maxWidth = this.options.maxWidth;\n }\n\n if (this._map.transform.renderWorldCopies && !this._trackPointer) {\n this._lngLat = smartWrap(this._lngLat, this._pos, this._map.transform);\n }\n\n if (this._trackPointer && !cursor) return;\n\n const pos = this._pos = this._trackPointer && cursor ? cursor : this._map.project(this._lngLat);\n\n let anchor = this.options.anchor;\n const offset = normalizeOffset(this.options.offset);\n\n if (!anchor) {\n const width = this._container.offsetWidth;\n const height = this._container.offsetHeight;\n let anchorComponents;\n\n if (pos.y + offset.bottom.y < height) {\n anchorComponents = ['top'];\n } else if (pos.y > this._map.transform.height - height) {\n anchorComponents = ['bottom'];\n } else {\n anchorComponents = [];\n }\n\n if (pos.x < width / 2) {\n anchorComponents.push('left');\n } else if (pos.x > this._map.transform.width - width / 2) {\n anchorComponents.push('right');\n }\n\n if (anchorComponents.length === 0) {\n anchor = 'bottom';\n } else {\n anchor = (anchorComponents.join('-') as any);\n }\n }\n\n const offsetedPos = pos.add(offset[anchor]).round();\n DOM.setTransform(this._container, `${anchorTranslate[anchor]} translate(${offsetedPos.x}px,${offsetedPos.y}px)`);\n applyAnchorClass(this._container, anchor, 'popup');\n };\n\n _focusFirstElement() {\n if (!this.options.focusAfterOpen || !this._container) return;\n\n const firstFocusable = this._container.querySelector(focusQuerySelector) as HTMLElement;\n\n if (firstFocusable) firstFocusable.focus();\n }\n\n _onClose = () => {\n this.remove();\n };\n}\n\nfunction normalizeOffset(offset?: Offset | null) {\n if (!offset) {\n return normalizeOffset(new Point(0, 0));\n\n } else if (typeof offset === 'number') {\n // input specifies a radius from which to calculate offsets at all positions\n const cornerOffset = Math.round(Math.abs(offset) / Math.SQRT2);\n return {\n 'center': new Point(0, 0),\n 'top': new Point(0, offset),\n 'top-left': new Point(cornerOffset, cornerOffset),\n 'top-right': new Point(-cornerOffset, cornerOffset),\n 'bottom': new Point(0, -offset),\n 'bottom-left': new Point(cornerOffset, -cornerOffset),\n 'bottom-right': new Point(-cornerOffset, -cornerOffset),\n 'left': new Point(offset, 0),\n 'right': new Point(-offset, 0)\n };\n\n } else if (offset instanceof Point || Array.isArray(offset)) {\n // input specifies a single offset to be applied to all positions\n const convertedOffset = Point.convert(offset);\n return {\n 'center': convertedOffset,\n 'top': convertedOffset,\n 'top-left': convertedOffset,\n 'top-right': convertedOffset,\n 'bottom': convertedOffset,\n 'bottom-left': convertedOffset,\n 'bottom-right': convertedOffset,\n 'left': convertedOffset,\n 'right': convertedOffset\n };\n\n } else {\n // input specifies an offset per position\n return {\n 'center': Point.convert(offset['center'] || [0, 0]),\n 'top': Point.convert(offset['top'] || [0, 0]),\n 'top-left': Point.convert(offset['top-left'] || [0, 0]),\n 'top-right': Point.convert(offset['top-right'] || [0, 0]),\n 'bottom': Point.convert(offset['bottom'] || [0, 0]),\n 'bottom-left': Point.convert(offset['bottom-left'] || [0, 0]),\n 'bottom-right': Point.convert(offset['bottom-right'] || [0, 0]),\n 'left': Point.convert(offset['left'] || [0, 0]),\n 'right': Point.convert(offset['right'] || [0, 0])\n };\n }\n}\n","import {extend} from './util';\n\n/**\n * This is a private namespace for utility functions that will get automatically stripped\n * out in production builds.\n */\nexport const Debug = {\n extend(dest: any, ...sources: Array): any {\n return extend(dest, ...sources);\n },\n\n run(fn: () => any) {\n fn();\n },\n\n logToElement(message: string, overwrite: boolean = false, id: string = 'log') {\n const el = window.document.getElementById(id);\n if (el) {\n if (overwrite) el.innerHTML = '';\n el.innerHTML += `
${message}`;\n }\n\n }\n};\n","import packageJSON from '../package.json' assert {type: 'json'};\nimport {Map} from './ui/map';\nimport {NavigationControl} from './ui/control/navigation_control';\nimport {GeolocateControl} from './ui/control/geolocate_control';\nimport {AttributionControl} from './ui/control/attribution_control';\nimport {LogoControl} from './ui/control/logo_control';\nimport {ScaleControl} from './ui/control/scale_control';\nimport {FullscreenControl} from './ui/control/fullscreen_control';\nimport {TerrainControl} from './ui/control/terrain_control';\nimport {Popup} from './ui/popup';\nimport {Marker} from './ui/marker';\nimport {Style} from './style/style';\nimport {LngLat} from './geo/lng_lat';\nimport {LngLatBounds} from './geo/lng_lat_bounds';\nimport Point from '@mapbox/point-geometry';\nimport {MercatorCoordinate} from './geo/mercator_coordinate';\nimport {Evented} from './util/evented';\nimport {config} from './util/config';\nimport {Debug} from './util/debug';\nimport {isSafari} from './util/util';\nimport {setRTLTextPlugin, getRTLTextPluginStatus} from './source/rtl_text_plugin';\nimport {WorkerPool} from './util/worker_pool';\nimport {prewarm, clearPrewarmedResources} from './util/global_worker_pool';\nimport {PerformanceUtils} from './util/performance';\nimport {AJAXError} from './util/ajax';\nimport type {RequestParameters, ResponseCallback} from './util/ajax';\nimport type {Cancelable} from './types/cancelable';\nimport {GeoJSONSource} from './source/geojson_source';\nimport {CanvasSource} from './source/canvas_source';\nimport {ImageSource} from './source/image_source';\nimport {RasterDEMTileSource} from './source/raster_dem_tile_source';\nimport {RasterTileSource} from './source/raster_tile_source';\nimport {VectorTileSource} from './source/vector_tile_source';\nimport {VideoSource} from './source/video_source';\n\nconst version = packageJSON.version;\n\nexport type * from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * `maplibregl` is the global object that allows configurations that are not specific to a map instance\n *\n * @group Main\n */\nclass MapLibreGL {\n static Map = Map;\n static NavigationControl = NavigationControl;\n static GeolocateControl = GeolocateControl;\n static AttributionControl = AttributionControl;\n static LogoControl = LogoControl;\n static ScaleControl = ScaleControl;\n static FullscreenControl = FullscreenControl;\n static TerrainControl = TerrainControl;\n static Popup = Popup;\n static Marker = Marker;\n static Style = Style;\n static LngLat = LngLat;\n static LngLatBounds = LngLatBounds;\n static Point = Point;\n static MercatorCoordinate = MercatorCoordinate;\n static Evented = Evented;\n static AJAXError = AJAXError;\n static config = config;\n static CanvasSource = CanvasSource;\n static GeoJSONSource = GeoJSONSource;\n static ImageSource = ImageSource;\n static RasterDEMTileSource = RasterDEMTileSource;\n static RasterTileSource = RasterTileSource;\n static VectorTileSource = VectorTileSource;\n static VideoSource = VideoSource;\n /**\n * Sets the map's [RTL text plugin](https://www.mapbox.com/mapbox-gl-js/plugins/#mapbox-gl-rtl-text).\n * Necessary for supporting the Arabic and Hebrew languages, which are written right-to-left.\n *\n * @param pluginURL - URL pointing to the Mapbox RTL text plugin source.\n * @param callback - Called with an error argument if there is an error.\n * @param lazy - If set to `true`, mapboxgl will defer loading the plugin until rtl text is encountered,\n * rtl text will then be rendered only after the plugin finishes loading.\n * @example\n * ```ts\n * maplibregl.setRTLTextPlugin('https://unpkg.com/@mapbox/mapbox-gl-rtl-text@0.2.3/mapbox-gl-rtl-text.js');\n * ```\n * @see [Add support for right-to-left scripts](https://maplibre.org/maplibre-gl-js/docs/examples/mapbox-gl-rtl-text/)\n */\n static setRTLTextPlugin = setRTLTextPlugin;\n /**\n * Gets the map's [RTL text plugin](https://www.mapbox.com/mapbox-gl-js/plugins/#mapbox-gl-rtl-text) status.\n * The status can be `unavailable` (i.e. not requested or removed), `loading`, `loaded` or `error`.\n * If the status is `loaded` and the plugin is requested again, an error will be thrown.\n *\n * @example\n * ```ts\n * const pluginStatus = maplibregl.getRTLTextPluginStatus();\n * ```\n */\n static getRTLTextPluginStatus = getRTLTextPluginStatus;\n /**\n * Initializes resources like WebWorkers that can be shared across maps to lower load\n * times in some situations. `maplibregl.workerUrl` and `maplibregl.workerCount`, if being\n * used, must be set before `prewarm()` is called to have an effect.\n *\n * By default, the lifecycle of these resources is managed automatically, and they are\n * lazily initialized when a Map is first created. By invoking `prewarm()`, these\n * resources will be created ahead of time, and will not be cleared when the last Map\n * is removed from the page. This allows them to be re-used by new Map instances that\n * are created later. They can be manually cleared by calling\n * `maplibregl.clearPrewarmedResources()`. This is only necessary if your web page remains\n * active but stops using maps altogether.\n *\n * This is primarily useful when using GL-JS maps in a single page app, wherein a user\n * would navigate between various views that can cause Map instances to constantly be\n * created and destroyed.\n *\n * @example\n * ```ts\n * maplibregl.prewarm()\n * ```\n */\n static prewarm = prewarm;\n /**\n * Clears up resources that have previously been created by `maplibregl.prewarm()`.\n * Note that this is typically not necessary. You should only call this function\n * if you expect the user of your app to not return to a Map view at any point\n * in your application.\n *\n * @example\n * ```ts\n * maplibregl.clearPrewarmedResources()\n * ```\n */\n static clearPrewarmedResources = clearPrewarmedResources;\n /**\n * Returns the package version of the library\n * @returns Package version of the library\n */\n static get version(): string {\n return version;\n }\n\n /**\n * Gets and sets the number of web workers instantiated on a page with GL JS maps.\n * By default, workerCount is 1 except for Safari browser where it is set to half the number of CPU cores (capped at 3).\n * Make sure to set this property before creating any map instances for it to have effect.\n *\n * @returns Number of workers currently configured.\n * @example\n * ```ts\n * maplibregl.workerCount = 2;\n * ```\n */\n static get workerCount(): number {\n return WorkerPool.workerCount;\n }\n\n static set workerCount(count: number) {\n WorkerPool.workerCount = count;\n }\n /**\n * Gets and sets the maximum number of images (raster tiles, sprites, icons) to load in parallel,\n * which affects performance in raster-heavy maps. 16 by default.\n *\n * @returns Number of parallel requests currently configured.\n * @example\n * ```ts\n * maplibregl.maxParallelImageRequests = 10;\n * ```\n */\n static get maxParallelImageRequests(): number {\n return config.MAX_PARALLEL_IMAGE_REQUESTS;\n }\n\n static set maxParallelImageRequests(numRequests: number) {\n config.MAX_PARALLEL_IMAGE_REQUESTS = numRequests;\n }\n\n static get workerUrl(): string {\n return config.WORKER_URL;\n }\n\n static set workerUrl(value: string) {\n config.WORKER_URL = value;\n }\n\n /**\n * Sets a custom load tile function that will be called when using a source that starts with a custom url schema.\n * The example below will be triggered for custom:// urls defined in the sources list in the style definitions.\n * The function passed will receive the request parameters and should call the callback with the resulting request,\n * for example a pbf vector tile, non-compressed, represented as ArrayBuffer.\n *\n * @param customProtocol - the protocol to hook, for example 'custom'\n * @param loadFn - the function to use when trying to fetch a tile specified by the customProtocol\n * @example\n * This will fetch a file using the fetch API (this is obviously a non interesting example...)\n * ```ts\n * maplibregl.addProtocol('custom', (params, callback) => {\n fetch(`https://${params.url.split(\"://\")[1]}`)\n .then(t => {\n if (t.status == 200) {\n t.arrayBuffer().then(arr => {\n callback(null, arr, null, null);\n });\n } else {\n callback(new Error(`Tile fetch error: ${t.statusText}`));\n }\n })\n .catch(e => {\n callback(new Error(e));\n });\n return { cancel: () => { } };\n });\n * // the following is an example of a way to return an error when trying to load a tile\n * maplibregl.addProtocol('custom2', (params, callback) => {\n * callback(new Error('someErrorMessage'));\n * return { cancel: () => { } };\n * });\n * ```\n */\n static addProtocol(customProtocol: string, loadFn: (requestParameters: RequestParameters, callback: ResponseCallback) => Cancelable) {\n config.REGISTERED_PROTOCOLS[customProtocol] = loadFn;\n }\n\n /**\n * Removes a previously added protocol\n *\n * @param customProtocol - the custom protocol to remove registration for\n * @example\n * ```ts\n * maplibregl.removeProtocol('custom');\n * ```\n */\n static removeProtocol(customProtocol: string) {\n delete config.REGISTERED_PROTOCOLS[customProtocol];\n }\n}\n\n//This gets automatically stripped out in production builds.\nDebug.extend(MapLibreGL, {isSafari, getPerformanceMetrics: PerformanceUtils.getPerformanceMetrics});\n\nexport default MapLibreGL;\n","import {DOM} from '../../util/dom';\n\nimport {warnOnce} from '../../util/util';\n\nimport {Event, Evented} from '../../util/evented';\nimport type {Map, GestureOptions} from '../map';\nimport type {IControl} from './control';\n\n/**\n * The {@link FullscreenControl} options\n */\ntype FullscreenOptions = {\n /**\n * `container` is the [compatible DOM element](https://developer.mozilla.org/en-US/docs/Web/API/Element/requestFullScreen#Compatible_elements) which should be made full screen. By default, the map container element will be made full screen.\n */\n container?: HTMLElement;\n};\n\n/**\n * A `FullscreenControl` control contains a button for toggling the map in and out of fullscreen mode.\n * When [requestFullscreen](https://developer.mozilla.org/en-US/docs/Web/API/Element/requestFullscreen) is not supported, fullscreen is handled via CSS properties.\n * The map's `cooperativeGestures` option is temporarily disabled while the map\n * is in fullscreen mode, and is restored when the map exist fullscreen mode.\n *\n * @group Markers and Controls\n * @param options - the full screen control options\n *\n * @example\n * ```ts\n * map.addControl(new maplibregl.FullscreenControl({container: document.querySelector('body')}));\n * ```\n * @see [View a fullscreen map](https://maplibre.org/maplibre-gl-js/docs/examples/fullscreen/)\n *\n * ### Events\n *\n * @event `fullscreenstart` - Fired when fullscreen mode has started\n *\n * @event `fullscreenend` - Fired when fullscreen mode has ended\n */\nexport class FullscreenControl extends Evented implements IControl {\n _map: Map;\n _controlContainer: HTMLElement;\n _fullscreen: boolean;\n _fullscreenchange: string;\n _fullscreenButton: HTMLButtonElement;\n _container: HTMLElement;\n _prevCooperativeGestures: boolean | GestureOptions;\n\n constructor(options: FullscreenOptions = {}) {\n super();\n this._fullscreen = false;\n\n if (options && options.container) {\n if (options.container instanceof HTMLElement) {\n this._container = options.container;\n } else {\n warnOnce('Full screen control \\'container\\' must be a DOM element.');\n }\n }\n\n if ('onfullscreenchange' in document) {\n this._fullscreenchange = 'fullscreenchange';\n } else if ('onmozfullscreenchange' in document) {\n this._fullscreenchange = 'mozfullscreenchange';\n } else if ('onwebkitfullscreenchange' in document) {\n this._fullscreenchange = 'webkitfullscreenchange';\n } else if ('onmsfullscreenchange' in document) {\n this._fullscreenchange = 'MSFullscreenChange';\n }\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map): HTMLElement {\n this._map = map;\n if (!this._container) this._container = this._map.getContainer();\n this._controlContainer = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group');\n this._setupUI();\n return this._controlContainer;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._controlContainer);\n this._map = null;\n window.document.removeEventListener(this._fullscreenchange, this._onFullscreenChange);\n }\n\n _setupUI() {\n const button = this._fullscreenButton = DOM.create('button', (('maplibregl-ctrl-fullscreen')), this._controlContainer);\n DOM.create('span', 'maplibregl-ctrl-icon', button).setAttribute('aria-hidden', 'true');\n button.type = 'button';\n this._updateTitle();\n this._fullscreenButton.addEventListener('click', this._onClickFullscreen);\n window.document.addEventListener(this._fullscreenchange, this._onFullscreenChange);\n }\n\n _updateTitle() {\n const title = this._getTitle();\n this._fullscreenButton.setAttribute('aria-label', title);\n this._fullscreenButton.title = title;\n }\n\n _getTitle() {\n return this._map._getUIString(this._isFullscreen() ? 'FullscreenControl.Exit' : 'FullscreenControl.Enter');\n }\n\n _isFullscreen() {\n return this._fullscreen;\n }\n\n _onFullscreenChange = () => {\n const fullscreenElement =\n window.document.fullscreenElement ||\n (window.document as any).mozFullScreenElement ||\n (window.document as any).webkitFullscreenElement ||\n (window.document as any).msFullscreenElement;\n\n if ((fullscreenElement === this._container) !== this._fullscreen) {\n this._handleFullscreenChange();\n }\n };\n\n _handleFullscreenChange() {\n this._fullscreen = !this._fullscreen;\n this._fullscreenButton.classList.toggle('maplibregl-ctrl-shrink');\n this._fullscreenButton.classList.toggle('maplibregl-ctrl-fullscreen');\n this._updateTitle();\n\n if (this._fullscreen) {\n this.fire(new Event('fullscreenstart'));\n if (this._map._cooperativeGestures) {\n this._prevCooperativeGestures = this._map._cooperativeGestures;\n this._map.setCooperativeGestures();\n }\n } else {\n this.fire(new Event('fullscreenend'));\n if (this._prevCooperativeGestures) {\n this._map.setCooperativeGestures(this._prevCooperativeGestures);\n delete this._prevCooperativeGestures;\n }\n }\n }\n\n _onClickFullscreen = () => {\n if (this._isFullscreen()) {\n this._exitFullscreen();\n } else {\n this._requestFullscreen();\n }\n };\n\n _exitFullscreen() {\n if (window.document.exitFullscreen) {\n (window.document as any).exitFullscreen();\n } else if ((window.document as any).mozCancelFullScreen) {\n (window.document as any).mozCancelFullScreen();\n } else if ((window.document as any).msExitFullscreen) {\n (window.document as any).msExitFullscreen();\n } else if ((window.document as any).webkitCancelFullScreen) {\n (window.document as any).webkitCancelFullScreen();\n } else {\n this._togglePseudoFullScreen();\n }\n }\n\n _requestFullscreen() {\n if (this._container.requestFullscreen) {\n this._container.requestFullscreen();\n } else if ((this._container as any).mozRequestFullScreen) {\n (this._container as any).mozRequestFullScreen();\n } else if ((this._container as any).msRequestFullscreen) {\n (this._container as any).msRequestFullscreen();\n } else if ((this._container as any).webkitRequestFullscreen) {\n (this._container as any).webkitRequestFullscreen();\n } else {\n this._togglePseudoFullScreen();\n }\n }\n\n _togglePseudoFullScreen() {\n this._container.classList.toggle('maplibregl-pseudo-fullscreen');\n this._handleFullscreenChange();\n this._map.resize();\n }\n}\n","import {DOM} from '../../util/dom';\n\nimport type {Map} from '../map';\nimport type {IControl} from './control';\nimport type {TerrainSpecification} from '@maplibre/maplibre-gl-style-spec';\n\n/**\n * A `TerrainControl` control contains a button for turning the terrain on and off.\n *\n * @group Markers and Controls\n *\n * @example\n * ```ts\n * let map = new maplibregl.Map({TerrainControl: false})\n * .addControl(new maplibregl.TerrainControl({\n * source: \"terrain\"\n * }));\n * ```\n */\nexport class TerrainControl implements IControl {\n options: TerrainSpecification;\n _map: Map;\n _container: HTMLElement;\n _terrainButton: HTMLButtonElement;\n\n constructor(options: TerrainSpecification) {\n this.options = options;\n }\n\n /** {@inheritDoc IControl.onAdd} */\n onAdd(map: Map) {\n this._map = map;\n this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group');\n this._terrainButton = DOM.create('button', 'maplibregl-ctrl-terrain', this._container);\n DOM.create('span', 'maplibregl-ctrl-icon', this._terrainButton).setAttribute('aria-hidden', 'true');\n this._terrainButton.type = 'button';\n this._terrainButton.addEventListener('click', this._toggleTerrain);\n\n this._updateTerrainIcon();\n this._map.on('terrain', this._updateTerrainIcon);\n return this._container;\n }\n\n /** {@inheritDoc IControl.onRemove} */\n onRemove() {\n DOM.remove(this._container);\n this._map.off('terrain', this._updateTerrainIcon);\n this._map = undefined;\n }\n\n _toggleTerrain = () => {\n if (this._map.getTerrain()) {\n this._map.setTerrain(null);\n } else {\n this._map.setTerrain(this.options);\n }\n this._updateTerrainIcon();\n };\n\n _updateTerrainIcon = () => {\n this._terrainButton.classList.remove('maplibregl-ctrl-terrain');\n this._terrainButton.classList.remove('maplibregl-ctrl-terrain-enabled');\n if (this._map.terrain) {\n this._terrainButton.classList.add('maplibregl-ctrl-terrain-enabled');\n this._terrainButton.title = this._map._getUIString('TerrainControl.disableTerrain');\n } else {\n this._terrainButton.classList.add('maplibregl-ctrl-terrain');\n this._terrainButton.title = this._map._getUIString('TerrainControl.enableTerrain');\n }\n };\n}\n","//\n// Our custom intro provides a specialized \"define()\" function, called by the\n// AMD modules below, that sets up the worker blob URL and then executes the\n// main module, storing its exported value as 'maplibregl'\n\n// The three \"chunks\" imported here are produced by a first Rollup pass,\n// which outputs them as AMD modules.\n\n// Shared dependencies, i.e.:\n/*\ndefine(['exports'], function (exports) {\n // Code for all common dependencies\n // Each module's exports are attached attached to 'exports' (with\n // names rewritten to avoid collisions, etc.)\n})\n*/\nimport '../../staging/maplibregl/shared';\n\n// Worker and its unique dependencies, i.e.:\n/*\ndefine(['./shared.js'], function (__shared__js) {\n // Code for worker script and its unique dependencies.\n // Expects the output of 'shared' module to be passed in as an argument,\n // since all references to common deps look like, e.g.,\n // __shared__js.shapeText().\n});\n*/\n// When this wrapper function is passed to our custom define() above,\n// it gets stringified, together with the shared wrapper (using\n// Function.toString()), and the resulting string of code is made into a\n// Blob URL that gets used by the main module to create the web workers.\nimport '../../staging/maplibregl/worker';\n\n// Main module and its unique dependencies\n/*\ndefine(['./shared.js'], function (__shared__js) {\n // Code for main GL JS module and its unique dependencies.\n // Expects the output of 'shared' module to be passed in as an argument,\n // since all references to common deps look like, e.g.,\n // __shared__js.shapeText().\n //\n // Returns the actual maplibregl (i.e. src/index.js)\n});\n*/\nimport '../../staging/maplibregl/index';\n\nexport default maplibregl;\n"],"names":["__awaiter","thisArg","_arguments","P","generator","Promise","resolve","reject","fulfilled","value","step","next","e","rejected","result","done","then","apply","SuppressedError","pointGeometry","Point","x","y","this","prototype","clone","add","p","_add","sub","_sub","multByPoint","_multByPoint","divByPoint","_divByPoint","mult","k","_mult","div","_div","rotate","a","_rotate","rotateAround","_rotateAround","matMult","m","_matMult","unit","_unit","perp","_perp","round","_round","mag","Math","sqrt","equals","other","dist","distSqr","dx","dy","angle","atan2","angleTo","b","angleWith","angleWithSep","cos","sin","convert","Array","isArray","unitbezier","UnitBezier","p1x","p1y","p2x","p2y","cx","bx","ax","cy","by","ay","sampleCurveX","t","sampleCurveY","sampleCurveDerivativeX","solveCurveX","epsilon","undefined","i","x2","abs","d2","t0","t1","solve","supportsOffscreenCanvas","offscreenCanvasDistorted","offscreenCanvasSupported","OffscreenCanvas","getContext","createImageBitmap","isOffscreenCanvasDistorted","size","context","willReadFrequently","base","fillStyle","fillRect","floor","data","getImageData","bezier","defaultEasing","clamp","n","min","max","wrap","d","w","extend","dest","sources","src","id","mapObject","input","iterator","output","key","call","filterObject","map","warnOnceHistory","warnOnce","message","console","warn","isCounterClockwise","c","calculateSignedArea","ring","sum","p1","p2","len","length","j","isWorker","WorkerGlobalScope","self","_isSafari","isImageBitmap","image","ImageBitmap","transparentPngUrl","readImageUsingVideoFrame","width","height","VideoFrame","Error","frame","timestamp","format","startsWith","swapBR","Uint8ClampedArray","copyTo","destRowOffset","offset","stride","sourceLeft","sourceTop","rect","layout","computeVideoFrameParameters","tmp","close","offscreenCanvas","offscreenCanvasContext","linkEl","reducedMotionQuery","browser","now","performance","bind","Date","fn","requestAnimationFrame","cancel","cancelAnimationFrame","img","padding","getImageCanvasContext","canvas","window","document","createElement","drawImage","resolveURL","path","href","hardwareConcurrency","navigator","prefersReducedMotion","matchMedia","matches","config","MAX_PARALLEL_IMAGE_REQUESTS","MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME","MAX_TILE_CACHE_ZOOM_LEVELS","REGISTERED_PROTOCOLS","WORKER_URL","AJAXError","constructor","status","statusText","url","body","super","getReferrer","worker","referrer","location","protocol","parent","getProtocolAction","substring","indexOf","makeFetchRequest","requestParameters","callback","controller","AbortController","request","Request","method","credentials","headers","cache","signal","complete","aborted","type","set","fetch","response","ok","arrayBuffer","json","text","get","catch","err","blob","error","code","abort","makeRequest","test","actor","send","Object","hasOwnProperty","xhr","XMLHttpRequest","open","responseType","setRequestHeader","withCredentials","onerror","onload","JSON","parse","getResponseHeader","Blob","makeXMLHttpRequest","getArrayBuffer","sameOrigin","inComingUrl","urlObj","URL","locationObj","host","_addEventListener","listener","listenerList","push","_removeEventListener","index","splice","Event","ErrorEvent","Evented","on","_listeners","off","_oneTimeListeners","once","fire","event","properties","listens","target","listeners","slice","oneTimeListeners","_eventedParent","_eventedParentData","setEventedParent","v8Spec","$version","$root","version","required","values","name","metadata","center","zoom","bearing","default","period","units","pitch","light","terrain","sprite","glyphs","transition","layers","source","source_vector","vector","tiles","bounds","scheme","xyz","tms","minzoom","maxzoom","attribution","promoteId","volatile","source_raster","raster","tileSize","source_raster_dem","encoding","terrarium","mapbox","custom","redFactor","blueFactor","greenFactor","baseShift","source_geojson","geojson","buffer","maximum","minimum","filter","tolerance","cluster","clusterRadius","clusterMaxZoom","clusterMinPoints","clusterProperties","lineMetrics","generateId","source_video","video","urls","coordinates","source_image","layer","fill","line","symbol","circle","heatmap","hillshade","background","paint","layout_background","visibility","visible","none","layout_fill","expression","interpolated","parameters","layout_circle","layout_heatmap","layout_line","butt","square","bevel","miter","requires","layout_symbol","point","auto","never","always","cooperative","viewport","both","tokens","left","right","top","bottom","horizontal","vertical","uppercase","lowercase","layout_raster","layout_hillshade","filter_operator","in","all","any","has","within","geometry_type","LineString","Polygon","function","stops","property","identity","exponential","interval","categorical","colorSpace","rgb","lab","hcl","function_stop","anchor","position","color","intensity","exaggeration","paint_fill","paint_line","paint_circle","paint_heatmap","paint_symbol","overridable","paint_raster","linear","nearest","paint_hillshade","paint_background","duration","delay","constant","refProperties","deref","forEach","deepEqual","keys","operations","setStyle","addLayer","removeLayer","setPaintProperty","setLayoutProperty","setFilter","addSource","removeSource","setGeoJSONSourceData","setLayerZoomRange","setLayerProperty","setCenter","setZoom","setBearing","setPitch","setSprite","setGlyphs","setTransition","setLight","sourceId","after","commands","command","args","sourcesRemoved","updateSource","canUpdateGeoJSON","before","prop","diffLayerPropertyChanges","layerId","klass","pluckId","indexById","group","ValidationError","identifier","__line__","extendBy","inputs","ExpressionParsingError","Scope","bindings","concat","NullType","kind","NumberType","StringType","BooleanType","ColorType","ObjectType","ValueType","CollatorType","FormattedType","PaddingType","ResolvedImageType","VariableAnchorOffsetCollectionType","array$1","itemType","N","toString$1","valueMemberTypes","checkSubtype","expected","memberType","isValidType","provided","allowedTypes","some","isValidNativeType","verifyType","sample","Xn","Zn","t2","t3","deg2rad","PI","rad2deg","constrainAngle","rgbToLab","r","g","alpha","z","xyz2lab","rgb2xyz","l","pow","labToRgb","isNaN","lab2xyz","xyz2rgb","parseHex","hex","parseInt","padEnd","parseAlpha","asPercentage","validateNumbers","array","Number","namedColors","aliceblue","antiquewhite","aqua","aquamarine","azure","beige","bisque","black","blanchedalmond","blue","blueviolet","brown","burlywood","cadetblue","chartreuse","chocolate","coral","cornflowerblue","cornsilk","crimson","cyan","darkblue","darkcyan","darkgoldenrod","darkgray","darkgreen","darkgrey","darkkhaki","darkmagenta","darkolivegreen","darkorange","darkorchid","darkred","darksalmon","darkseagreen","darkslateblue","darkslategray","darkslategrey","darkturquoise","darkviolet","deeppink","deepskyblue","dimgray","dimgrey","dodgerblue","firebrick","floralwhite","forestgreen","fuchsia","gainsboro","ghostwhite","gold","goldenrod","gray","green","greenyellow","grey","honeydew","hotpink","indianred","indigo","ivory","khaki","lavender","lavenderblush","lawngreen","lemonchiffon","lightblue","lightcoral","lightcyan","lightgoldenrodyellow","lightgray","lightgreen","lightgrey","lightpink","lightsalmon","lightseagreen","lightskyblue","lightslategray","lightslategrey","lightsteelblue","lightyellow","lime","limegreen","linen","magenta","maroon","mediumaquamarine","mediumblue","mediumorchid","mediumpurple","mediumseagreen","mediumslateblue","mediumspringgreen","mediumturquoise","mediumvioletred","midnightblue","mintcream","mistyrose","moccasin","navajowhite","navy","oldlace","olive","olivedrab","orange","orangered","orchid","palegoldenrod","palegreen","paleturquoise","palevioletred","papayawhip","peachpuff","peru","pink","plum","powderblue","purple","rebeccapurple","red","rosybrown","royalblue","saddlebrown","salmon","sandybrown","seagreen","seashell","sienna","silver","skyblue","slateblue","slategray","slategrey","snow","springgreen","steelblue","tan","teal","thistle","tomato","turquoise","violet","wheat","white","whitesmoke","yellow","yellowgreen","Color","premultiplied","overwriteGetter","static","rgba","toLowerCase","trim","namedColorsMatch","rgbMatch","match","_","rp","f1","gp","f2","bp","f3","ap","argFormat","join","valFormat","maxValue","hslMatch","h","s","hsla","f","hslToRgb","parseCssColor","Infinity","rgbColor","NaN","rgbToHcl","getterKey","lazyValue","defineProperty","toString","transparent","Collator","caseSensitive","diacriticSensitive","locale","sensitivity","collator","Intl","usage","compare","lhs","rhs","resolvedLocale","resolvedOptions","FormattedSection","scale","fontStack","textColor","Formatted","sections","unformatted","isEmpty","section","fromString","Padding","val","stringify","anchors","Set","VariableAnchorOffsetCollection","anchorValue","offsetValue","ResolvedImage","options","available","validateRGBA","isValue","mixed","item","typeOf","String","Literal","expectedType","evaluate","eachChild","outputDefined","RuntimeError","toJSON","types$1","string","number","boolean","object","Assertion","parsed","ctx","every","arg","types","Coercion","Boolean","parseColor","pad","coll","num","geometryTypes","EvaluationContext","globals","feature","featureState","formattedSection","_parseColorCache","availableImages","canonical","geometryType","geometry","canonicalID","cached","ParsingContext","registry","isConstantFunc","scope","errors","part","_isConstant","expr","_parse","annotate","typeAnnotation","op","Expr","actual","ec","CollatorExpression","EXTENT","updateBBox","bbox","coord","boxWithinBox","bbox1","bbox2","getTileCoordinates","log","tilesAtZoom","onBoundary","x1","y1","y2","pointWithinPolygon","rings","inside","len2","pointWithinPolygons","polygons","twoSided","q1","q2","x3","y3","det1","det2","lineIntersectPolygon","polygon","v1","v2","lineStringWithinPolygon","lineStringWithinPolygons","getTilePolygon","getTilePolygons","updatePoint","polyBBox","worldSize","halfWorldSize","shift","getTilePoints","pointBBox","shifts","tilePoints","points","getTileLines","lineBBox","tileLines","tileLine","Within","geometries","features","polygonGeometry","tilePolygon","tilePolygons","pointsWithinPolygons","linesWithinPolygons","Var","boundExpression","CompoundExpression","_evaluate","definition","definitions","availableOverloads","overloads","signature","signatureContext","params","isExpressionConstant","parsedArgs","argParseFailed","signatures","stringifySignature","actualTypes","isTypeAnnotation","childrenConstant","child","isFeatureConstant","isGlobalPropertyConstant","isStateConstant","findStopLessThanOrEqualTo","lastIndex","currentValue","nextValue","lowerIndex","upperIndex","currentIndex","Step","labels","outputs","label","outputType","labelKey","valueKey","stopCount","out","from","to","interpolate","spaceKey","hue0","chroma0","light0","alphaF","hue1","chroma1","light1","alphaT","hue","chroma","dh","hclToRgb","variableAnchorOffsetCollection","fromValues","toValues","fx","fy","tx","ty","Interpolate","operator","interpolation","lower","upper","exponentialInterpolation","controlPoints","rest","interpolationFactor","outputLower","outputUpper","lowerValue","upperValue","difference","progress","Coalesce","needsAnnotation","requestedImageName","argCount","Let","binding","At","In","needle","haystack","IndexOf","fromIndex","Match","inputType","cases","otherwise","labelContext","MAX_SAFE_INTEGER","Case","branches","Slice","beginIndex","endIndex","isComparableType","eqCollate","makeComparison","compareBasic","compareWithCollator","isOrderComparison","Comparison","hasUntypedArgument","lt","rt","Equals","NotEquals","LessThan","GreaterThan","LessThanOrEqual","GreaterThanOrEqual","NumberFormat","currency","minFractionDigits","maxFractionDigits","style","minimumFractionDigits","maximumFractionDigits","FormatExpression","firstArg","nextTokenMayBeObject","font","lastExpression","content","evaluatedContent","ImageExpression","evaluatedImageName","Length","expressions","at","case","coalesce","let","literal","var","obj","v","varargs","success","supportsPropertyExpression","spec","supportsZoomExpression","supportsInterpolation","getType","isFunction","identityFunction","createFunction","propertySpec","isColor","zoomAndFeatureDependent","zoomDependent","parseFn","stop","innerFun","hashedStops","categoricalKeyType","evaluateExponentialFunction","evaluateIntervalFunction","evaluateCategoricalFunction","create","evaluateIdentityFunction","featureFunctions","zoomStops","featureFunctionStops","interpolationType","coalesce$1","keyType","interp","evaluatedLower","evaluatedUpper","register","typeof","heatmapDensity","lineProgress","accumulated","ln2","LN2","pi","E","log10","LN10","ln","log2","asin","acos","atan","ceil","binarySearch","isSupportedScript","upcase","toUpperCase","downcase","StyleExpression","_warningHistory","_evaluator","_defaultValue","_enumValues","evaluateWithoutErrorHandling","isExpression","createExpression","parser","enum","formatted","resolvedImage","getExpectedType","ZoomConstantExpression","_styleExpression","isStateDependent","ZoomDependentExpression","createPropertyExpression","expressionInput","isFeatureConstantResult","isZoomConstant","zoomCurve","findZoomCurve","StylePropertyFunction","specification","_parameters","_specification","serialized","childResult","isExpressionFilter","filterSpec","createFilter","needGeometry","convertFilter$1","compiled","globalProperties","geometryNeeded","convertComparisonOp$1","convertNegation","filters","convertInOp$1","convertHasOp$1","sort","str","getKey","validateConstants","constants","unbundle","valueOf","deepUnbundle","unbundledValue","validateObject","elementSpecs","valueSpec","elementValidators","objectElementValidators","styleSpec","validateSpec","objectKey","elementSpecKey","split","elementSpec","validateElement","validateArray","arraySpec","validateArrayElement","arrayElementValidator","arrayElementSpec","arrayIndex","validateNumber","validateFunction","functionValueSpec","functionType","stopKeyType","previousStopDomainValue","previousStopDomainZoom","stopDomainValues","isZoomFunction","isPropertyFunction","isZoomAndPropertyFunction","validateFunctionStop","validateStopDomainValue","reportValue","isFinite","validateExpression","expressionContext","expressionObj","propertyKey","propertyType","validateEnum","validateFilter","validateNonExpressionFilter","validateProperty","layerSpec","layerType","transitionMatch","tokenMatch","exec","validatePaintProperty","validateLayoutProperty","validateLayer","ref","otherLayer","sourceType","validateString","validateSource","replace","_a","sourceName","rasterDEM","rasterDEMSpec","rootType","isCustomEncoding","customEncodingKeys","encodingName","includes","validateRasterDEMSource","mapExpr","reduceExpr","validateLight","lightSpec","validateTerrain","terrainSpec","validateSprite","allSpriteIds","allSpriteURLs","VALIDATORS","validate","validateGlyphsUrl","validateStyleMin","sortErrors","injectValidateSpec","validator","wrapCleanErrors","inner","paintProperty","layoutProperty","validateStyle","emitValidationErrors","emitter","hasErrors","TransferableGridIndex","extent","cells","ArrayBuffer","Int32Array","start","end","subarray","bboxesOffset","bboxes","insert","_insertReadonly","uid","_forEachCell","_insertCell","cellIndex","query","intersectionTest","_queryCell","seenUids","cell","u","arg1","arg2","cx1","_convertToCellCoord","cy1","cx2","cy2","_convertFromCellCoord","toArrayBuffer","metadataLength","totalCellLength","grid","transferables","writeable","omit","shallow","_classRegistryKey","isArrayBuffer","serialize","RegExp","isView","view","ImageData","$name","deserialize","ZoomHistory","first","update","floorZ","lastIntegerZoom","lastIntegerZoomTime","lastZoom","lastFloorZoom","unicodeBlockLookup","char","Arabic","Khmer","Hiragana","Katakana","Bopomofo","Kanbun","allowsVerticalWritingMode","chars","charHasUprightVerticalOrientation","charCodeAt","allowsLetterSpacing","charAllowsLetterSpacing","isChar","charHasRotatedVerticalOrientation","charHasNeutralVerticalOrientation","charInRTLScript","charInSupportedScript","canRenderRTL","stringContainsRTLText","_completionCallback","pluginStatus","pluginURL","triggerPluginCompletionEvent","sendPluginStateToWorker","evented","getRTLTextPluginStatus","downloadRTLTextPlugin","plugin","applyArabicShaping","processBidirectionalText","processStyledBidirectionalText","isLoaded","isLoading","setState","state","isParsed","getPluginURL","EvaluationParameters","fadeDuration","zoomHistory","isStringInSupportedScript","rtlTextPlugin","crossFadingFactor","getCrossfadeParameters","fraction","fromScale","toScale","PropertyValue","normalizePropertyExpression","isDataDriven","possiblyEvaluate","TransitionablePropertyValue","transitioned","prior","TransitioningPropertyValue","untransitioned","Transitionable","_properties","_values","defaultTransitionablePropertyValues","getValue","setValue","getTransition","Transitioning","begin","finalValue","easeCubicInOut","defaultTransitioningPropertyValues","PossiblyEvaluated","hasTransition","Layout","defaultPropertyValues","hasValue","PossiblyEvaluatedPropertyValue","isConstant","constantOr","defaultPossiblyEvaluatedValues","DataConstantProperty","interpolationFn","interpolates","DataDrivenProperty","overrides","interpolatedValue","CrossFadedDataDrivenProperty","evaluatedValue","constantValue","_calculate","cameraVal","mid","CrossFadedProperty","ColorRampProperty","Properties","overridableProperties","defaultPropertyValue","defaultTransitionablePropertyValue","TRANSITION_SUFFIX","StyleLayer","_featureFilter","sourceLayer","_unevaluatedLayout","_transitionablePaint","_transitioningPaint","_crossfadeParameters","getLayoutProperty","_validate","getPaintProperty","endsWith","transitionable","isCrossFadedProperty","wasDataDriven","oldValue","_handleSpecialPaintPropertyUpdate","newValue","_handleOverridablePaintPropertyUpdate","isHidden","updateTransitions","recalculate","is3D","isTileClipped","hasOffscreenPass","resize","viewTypes","Int8","Int8Array","Uint8","Uint8Array","Int16","Int16Array","Uint16","Uint16Array","Int32","Uint32","Uint32Array","Float32","Float32Array","Struct","structArray","_structArray","_pos1","_pos2","_pos4","_pos8","StructArray","isTransferred","capacity","_trim","byteLength","bytesPerElement","_refreshViews","clear","reserve","oldUint8Array","uint8","createLayout","members","alignment","maxSize","member","typeSize","BYTES_PER_ELEMENT","memberOffset","align","components","StructArrayLayout2i4","int16","emplaceBack","v0","emplace","o2","StructArrayLayout3i6","StructArrayLayout4i8","v3","StructArrayLayout2i4i12","v4","v5","StructArrayLayout2i4ub8","o1","StructArrayLayout2f8","float32","o4","StructArrayLayout10ui20","uint16","v6","v7","v8","v9","StructArrayLayout4i4ui4i24","v10","v11","StructArrayLayout3f12","StructArrayLayout1ul4","uint32","StructArrayLayout6i1ul2ui20","StructArrayLayout2i2i2i12","StructArrayLayout2f1f2i16","StructArrayLayout2ub2f12","StructArrayLayout3ui6","StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48","v12","v13","v14","v15","v16","StructArrayLayout8i15ui1ul2f2ui64","v17","v18","v19","v20","v21","v22","v23","v24","v25","v26","v27","StructArrayLayout1f4","StructArrayLayout1ui2f12","StructArrayLayout1ul2ui8","StructArrayLayout2ui4","StructArrayLayout1ui2","StructArrayLayout4f16","CollisionBoxStruct","anchorPointX","anchorPointY","featureIndex","sourceLayerIndex","bucketIndex","anchorPoint","CollisionBoxArray","PlacedSymbolStruct","anchorX","anchorY","glyphStartIndex","numGlyphs","vertexStartIndex","lineStartIndex","lineLength","segment","lowerSize","upperSize","lineOffsetX","lineOffsetY","writingMode","placedOrientation","hidden","crossTileID","associatedIconIndex","PlacedSymbolArray","SymbolInstanceStruct","rightJustifiedTextSymbolIndex","centerJustifiedTextSymbolIndex","leftJustifiedTextSymbolIndex","verticalPlacedTextSymbolIndex","placedIconSymbolIndex","verticalPlacedIconSymbolIndex","textBoxStartIndex","textBoxEndIndex","verticalTextBoxStartIndex","verticalTextBoxEndIndex","iconBoxStartIndex","iconBoxEndIndex","verticalIconBoxStartIndex","verticalIconBoxEndIndex","numHorizontalGlyphVertices","numVerticalGlyphVertices","numIconVertices","numVerticalIconVertices","useRuntimeCollisionCircles","textBoxScale","collisionCircleDiameter","textAnchorOffsetStartIndex","textAnchorOffsetEndIndex","SymbolInstanceArray","GlyphOffsetArray","getoffsetX","SymbolLineVertexArray","getx","gety","gettileUnitDistanceFromAnchor","TextAnchorOffsetStruct","textAnchor","textOffset0","textOffset1","TextAnchorOffsetArray","FeatureIndexStruct","FeatureIndexArray","PosArray","CircleLayoutArray","FillLayoutArray","FillExtrusionLayoutArray","LineLayoutArray","LineExtLayoutArray","PatternLayoutArray","SymbolLayoutArray","SymbolDynamicLayoutArray","SymbolOpacityArray","CollisionBoxLayoutArray","CollisionVertexArray","TriangleIndexArray","LineIndexArray","SegmentVector","segments","prepareSegment","numVertices","layoutVertexArray","indexArray","sortKey","MAX_VERTEX_ARRAY_LENGTH","vertexLength","vertexOffset","primitiveOffset","primitiveLength","destroy","vaos","packUint8ToFloat","patternAttributes","seed","remainder","bytes","h1","h1b","c1","c2","k1","murmur3","require$$0","murmur2","murmurhashJsModule","exports","FeaturePositionMap","ids","positions","indexed","getNumericId","getPositions","intId","Float64Array","numValue","pivot","swap","arr","Uniform","gl","Uniform1f","current","uniform1f","Uniform4f","uniform4f","UniformColor","emptyMat4","packColor","ConstantBinder","names","uniformNames","setUniform","uniform","getBinding","CrossFadedConstantBinder","patternFrom","patternTo","pixelRatioFrom","pixelRatioTo","setConstantPatternPositions","posTo","posFrom","pixelRatio","tlbr","uniformName","pos","substr","SourceExpressionBinder","PaintVertexArray","paintVertexAttributes","paintVertexArray","populatePaintArray","newLength","imagePositions","_setPaintValue","updatePaintArray","upload","paintVertexBuffer","updateData","createVertexBuffer","CompositeExpressionBinder","useIntegerZoom","minColor","maxColor","currentZoom","factor","CrossFadedCompositeBinder","zoomInPaintVertexArray","zoomOutPaintVertexArray","_setPaintValues","patterns","imageMin","imageMid","imageMax","tl","br","zoomInPaintVertexBuffer","zoomOutPaintVertexBuffer","ProgramConfiguration","filterProperties","binders","_buffers","paintAttributeNames","propType","isCrossFaded","StructArrayLayout","layoutType","cacheKey","getMaxValue","binder","populatePaintArrays","updatePaintArrays","featureStates","featureMap","vtLayer","dirty","defines","getBinderAttributes","getBinderUniforms","uniforms","getPaintVertexBuffers","getUniforms","locations","setUniforms","binderUniforms","updatePaintBuffers","crossfade","patternVertexBuffer","ProgramConfigurationSet","programConfigurations","needsUpload","_featureMap","_bufferOffset","binderType","defaultLayouts","composite","layoutException","getLayoutException","MAX","BITS","MIN","loadGeometry","toEvaluationFeature","addCircleVertex","extrudeX","extrudeY","CircleBucket","overscaling","layerIds","hasPattern","stateDependentLayerIds","populate","styleLayer","bucketFeatures","circleSortKey","sortFeaturesByKey","evaluationFeature","bucketFeature","addFeature","states","stateDependentLayers","uploadPending","uploaded","layoutVertexBuffer","layoutAttributes","indexBuffer","createIndexBuffer","polygonIntersectsPolygon","polygonA","polygonB","polygonContainsPoint","lineIntersectsLine","polygonIntersectsBufferedPoint","radius","pointIntersectsBufferedLine","polygonIntersectsMultiPolygon","multiPolygon","multiPolygonContainsPoint","lineIntersectsBufferedLine","lineA","lineB","a0","a1","lineSegmentIntersectsLineSegment","b0","b1","radiusSquared","distToSegmentSquared","l2","edgeIntersectsBox","e1","e2","corners","dir","getMaximumPaintValue","bucket","translateDistance","translate","queryGeometry","translateAnchor","pixelsToTileUnits","pt","translated","properties$8","EPSILON","ARRAY_TYPE","multiply","a00","a01","a02","a03","a10","a11","a12","a13","a20","a21","a22","a23","a30","a31","a32","a33","b2","b3","hypot","arguments","mul","transformMat4","glMatrix.ARRAY_TYPE","CircleStyleLayer","createBucket","queryRadius","circleBucket","queryIntersectsFeature","transform","pixelPosMatrix","translatedPolygon","alignWithMap","transformedPolygon","projectPoint","projectQueryGeometry","transformedSize","transformedPoint","adjustedSize","projectedCenter","vec4.transformMat4","cameraToCenterDistance","HeatmapBucket","properties$7","createImage","channels","RangeError","resizeImage","newImage","copyImage","srcImg","dstImg","srcPt","dstPt","srcData","dstData","srcOffset","dstOffset","AlphaImage","RGBAImage","copy","renderColorRamp","evaluationGlobals","resolution","clips","renderPixel","evaluationKey","pxColor","clip","HeatmapStyleLayer","_updateColorRamp","colorRamp","colorRampTexture","heatmapFbo","properties$6","HillshadeStyleLayer","earcut","holeIndices","dim","minX","minY","maxX","maxY","invSize","hasHoles","outerLen","outerNode","linkedList","triangles","prev","list","queue","steiner","getLeftmost","compareX","eliminateHole","eliminateHoles","earcutLinked","clockwise","last","signedArea","insertNode","removeNode","filterPoints","again","area","ear","pass","zOrder","prevZ","nextZ","q","tail","numMerges","pSize","qSize","inSize","sortLinked","indexCurve","isEarHashed","isEar","cureLocalIntersections","splitEarcut","x0","y0","pointInTriangle","minZ","maxZ","intersects","locallyInside","isValidDiagonal","splitPolygon","hole","bridge","hx","hy","qx","mx","my","tanMin","sectorContainsSector","findHoleBridge","bridgeReverse","leftmost","px","py","intersectsPolygon","middleInside","sign","o3","onSegment","a2","Node","an","earcutModule","deviation","polygonArea","trianglesArea","flatten","vertices","holes","dimensions","holeIndex","quickselect","quickselectStep","defaultCompare","exp","sd","classifyRings","maxRings","ccw","compareAreas","patternDependencies","patternProperty","constantPattern","addPatternDependencies","patternFeature","patternPropertyValue","FillBucket","patternFeatures","indexArray2","segments2","fillSortKey","addFeatures","indexBuffer2","triangleSegment","triangleIndex","flattened","lineSegment","lineIndex","indices","properties$5","FillStyleLayer","outlineColor","centroidAttributes","vectortilefeature","VectorTileFeature","pbf","_pbf","_geometry","_keys","readFields","readFeature","tag","readVarint","readTag","cmd","lines","cmdLen","readSVarint","toGeoJSON","coords","project","vectortilelayer","VectorTileLayer","_features","readLayer","readString","readFloat","readDouble","readVarint64","readBoolean","readValueMessage","readTile","vectorTile","VectorTile","require$$1","require$$2","vectorTileFeatureTypes","mvt","FACTOR","addVertex","vertexArray","nx","ny","nz","FillExtrusionBucket","centroidVertexArray","centroidVertexBuffer","centroid","vertexCount","isEntirelyOutside","edgeDistance","isBoundaryEdge","bottomRight","properties$4","FillExtrusionStyleLayer","projectedQueryGeometry","projected","zBase","zTop","projectedBase","projectedTop","baseXZ","baseYZ","baseZZ","baseWZ","topXZ","topYZ","topZZ","topWZ","ringBase","ringTop","sX","sY","sZ","sW","baseZ","baseW","topX","topY","topZ","topW","projectExtrusion","closestDistance","getIntersectionDistance","topA","face","checkIntersection","dot","projectedFace","ab","ac","dotABAB","dotABAC","dotACAC","dotAPAB","dotAPAC","denom","distance","lineLayoutAttributes","lineLayoutAttributesExt","COS_HALF_SHARP_CORNER","MAX_LINE_DISTANCE","LINE_DISTANCE_BUFFER_BITS","LineBucket","lineClipsArray","gradients","layoutVertexArray2","maxLineLength","lineSortKey","patternBucketFeature","layoutVertexBuffer2","layoutAttributesExt","lineFeatureClips","cap","miterLimit","roundLimit","lineClips","addLine","scaledDistance","totalDistance","updateScaledDistance","isPolygon","sharpCornerOffset","currentVertex","prevVertex","nextVertex","prevNormal","nextNormal","joinNormal","cosAngle","cosHalfAngle","miterLength","approxAngle","isSharpCorner","lineTurnsLeft","prevSegmentLength","newPrevVertex","updateDistance","addCurrentVertex","middleVertex","currentJoin","bevelLength","offsetA","offsetB","extrude","addHalfVertex","nextSegmentLength","newCurrentVertex","normal","endLeft","endRight","rightX","rightY","up","linesofarScaled","properties$3","LineFloorwidthProperty","lineFloorwidthProperty","LineStyleLayer","gradientVersion","gradientExpression","stepInterpolant","isZoomExpression","lineBucket","getLineWidth","halfWidth","lineOffset","newRings","ringIndex","newRing","aToB","bToC","offsetLine","multiLine","polygonIntersectsBufferedMultiLine","lineWidth","lineGapWidth","symbolLayoutAttributes","dynamicLayoutAttributes","collisionVertexAttributes","collisionBoxLayout","collisionCircleLayout","transformText","toLocaleUpperCase","toLocaleLowerCase","transformTextInternal","verticalizedCharacterMap","$","ONE_EM","Pbf","ieee754","isLE","mLen","nBytes","eLen","eMax","eBias","nBits","buf","Varint","Fixed64","Bytes","Fixed32","SHIFT_LEFT_32","SHIFT_RIGHT_32","utf8TextDecoder","TextDecoder","readPackedEnd","toNum","low","high","isSigned","makeRoomForExtraLength","startPos","extraLen","realloc","writePackedVarint","writeVarint","writePackedSVarint","writeSVarint","writePackedFloat","writeFloat","writePackedDouble","writeDouble","writePackedBoolean","writeBoolean","writePackedFixed32","writeFixed32","writePackedSFixed32","writeSFixed32","writePackedFixed64","writeFixed64","writePackedSFixed64","writeSFixed64","readUInt32","writeInt32","readInt32","readField","skip","readMessage","readFixed32","readSFixed32","readFixed64","readSFixed64","readVarintRemainder","decode","readUtf8TextDecoder","bytesPerSequence","fromCharCode","readUtf8","readBytes","readPackedVarint","readPackedSVarint","readPackedBoolean","readPackedFloat","readPackedDouble","readPackedFixed32","readPackedSFixed32","readPackedFixed64","readPackedSFixed64","writeTag","finish","writeBigVarintLow","lsb","writeBigVarintHigh","writeBigVarint","writeString","lead","writeUtf8","writeBytes","writeRawMessage","writeMessage","writeBytesField","writeFixed32Field","writeSFixed32Field","writeFixed64Field","writeSFixed64Field","writeVarintField","writeSVarintField","writeStringField","writeFloatField","writeDoubleField","writeBooleanField","border","readFontstacks","readFontstack","bitmap","advance","readGlyph","metrics","glyph","GLYPH_PBF_BORDER","potpack","boxes","maxWidth","box","spaces","space","pop","IMAGE_PADDING","ImagePosition","paddedRect","stretchX","stretchY","displaySize","ImageAtlas","icons","iconPositions","patternPositions","haveRenderCallbacks","bins","addImages","bin","images","hasRenderCallback","patchUpdatedImages","imageManager","texture","dispatchRenderCallbacks","updatedImages","patchUpdatedImage","getImage","WritingMode","ai","SHAPING_DEFAULT_OFFSET","SectionOptions","imageName","textOptions","imageOptions","TaggedString","sectionIndex","imageSectionID","defaultFontStack","addImageSection","addTextSection","getSection","getSectionIndex","getCharCode","verticalizePunctuation","nextCharCode","prevCharCode","beginningWhitespace","whitespace","trailingWhitespace","getMaxScale","reduce","forText","nextImageSectionCharCode","getNextImageSectionCharCode","forImage","shapeText","glyphMap","glyphPositions","lineHeight","textJustify","spacing","allowVerticalPlacement","symbolPlacement","layoutTextSize","layoutTextSizeThisZoom","logicalInput","fromFeature","untaggedLines","determineLineBreaks","taggedLine","processedLines","lineBreakPoints","lineBreak","breakLines","positionedLines","shaping","iconsInText","verticalizable","maxLineHeight","justify","lineMaxScale","maxLineOffset","positionedLine","positionedGlyphs","codePoint","baselineOffset","verticalAdvance","imagePosition","glyphPosition","justifyLine","currentLineHeight","horizontalAlign","verticalAlign","getAnchorAlignment","blockHeight","lineCount","shiftX","shiftY","positionedGlyph","shapeLines","breakable","getGlyphAdvance","calculateBadness","targetWidth","penalty","isLastBreak","raggedness","calculatePenalty","nextCodePoint","penalizableIdeographicBreak","evaluateBreak","breakIndex","breakX","potentialBreaks","bestPriorBreak","bestBreakBadness","potentialBreak","breakBadness","badness","priorBreak","leastBadBreaks","lastLineBreak","potentialLineBreaks","totalWidth","determineAverageLineWidth","hasServerSuggestedBreakpoints","currentX","ideographicBreak","lastPositionedGlyph","lineIndent","shapeIcon","iconOffset","iconAnchor","fitIconToText","shapedIcon","shapedText","textFit","fontScale","collisionPadding","textLeft","textRight","textTop","textBottom","MAX_GLYPH_ICON_SIZE","SIZE_PACK_FACTOR","MAX_PACKED_SIZE","getSizeData","tileZoom","layoutSize","minZoom","maxZoom","minSize","getOverlapMode","overlapProp","allowOverlapProp","overlap","shaderOpacityAttributes","ox","oy","sizeVertex","isSDF","pixelOffsetX","pixelOffsetY","minFontScaleX","minFontScaleY","aSizeX","aSizeY","addDynamicAttributes","dynamicLayoutVertexArray","containsRTLText","formattedText","SymbolBuffers","opacityVertexArray","hasVisibleVertices","placedSymbolArray","dynamicIndexBuffer","dynamicLayoutVertexBuffer","opacityVertexBuffer","itemSize","CollisionBuffers","LayoutArray","IndexArray","collisionVertexArray","collisionVertexBuffer","SymbolBucket","collisionBoxArray","hasRTLText","sortKeyRanges","collisionCircleArray","placementInvProjMatrix","mat4.identity","placementViewportMatrix","unevaluatedLayoutValues","textSizeData","iconSizeData","canOverlap","sortFeaturesByY","writingModes","wm","sourceID","createArrays","icon","glyphOffsetArray","lineVertexArray","symbolInstances","textAnchorOffsets","calculateGlyphDependencies","stack","textAlongLine","doesAllowVerticalWritingMode","verticalChar","charAt","textFont","textField","iconImage","hasText","hasIcon","symbolSortKey","iconDependencies","stacks","glyphDependencies","resolvedTokens","getValueAndResolveTokens","factory","globalRTLTextPlugin","sectionFont","sectionStack","leftIndex","rightIndex","mergedFeatures","mergedIndex","mergeFromRight","leftKey","rightKey","geom","mergeFromLeft","onRight","mergeLines","hasDebugData","textCollisionBox","iconCollisionBox","destroyDebugData","addToLineVertexArray","sumForwardLength","sumBackwardLength","tileUnitDistanceFromAnchor","vertex","addSymbols","arrays","quads","alongLine","labelAnchor","glyphOffsetArrayStart","tr","bl","tex","pixelOffsetTL","pixelOffsetBR","glyphOffset","_addCollisionDebugVertex","addCollisionDebugVertices","boxAnchorPoint","symbolInstance","addDebugCollisionBoxes","startIndex","isText","generateCollisionDebugBuffers","_deserializeCollisionBoxesForSymbol","textStartIndex","textEndIndex","verticalTextStartIndex","verticalTextEndIndex","iconStartIndex","iconEndIndex","verticalIconStartIndex","verticalIconEndIndex","collisionArrays","textBox","textFeatureIndex","verticalTextBox","verticalTextFeatureIndex","iconBox","iconFeatureIndex","verticalIconBox","verticalIconFeatureIndex","deserializeCollisionBoxes","hasTextData","hasIconData","hasTextCollisionBoxData","hasIconCollisionBoxData","addIndicesForPlacedSymbol","iconOrText","placedSymbolIndex","placedSymbol","vertexIndex","getSortedSymbolIndexes","sortedAngle","symbolInstanceIndexes","rotatedYs","featureIndexes","aIndex","bIndex","addToSortKeyRanges","symbolInstanceIndex","symbolInstanceEnd","symbolInstanceStart","sortFeatures","featureSortOrder","MAX_GLYPHS","properties$2","runtimeType","getOverride","o","hasOverride","FormatSectionOverride","defaultValue","SymbolStyleLayer","deduped","_setPaintOverrides","unevaluated","resolveTokens","hasPaintOverride","overriden","override","styleExpression","propertyName","hasOverrides","checkSections","checkExpression","properties$1","BackgroundStyleLayer","RasterStyleLayer","CustomStyleLayer","implementation","onAdd","painter","onRemove","renderingMode","prerender","ThrottledInvoker","_callback","_triggered","MessageChannel","_channel","port2","onmessage","trigger","port1","postMessage","setTimeout","remove","earthRadius","LngLat","lng","lat","toArray","distanceTo","lngLat","rad","lat1","lat2","lon","earthCircumfrence","circumferenceAtLatitude","latitude","mercatorXfromLng","mercatorYfromLat","mercatorZfromAltitude","altitude","lngFromMercatorX","latFromMercatorY","MercatorCoordinate","lngLatLike","toLngLat","toAltitude","meterInMercatorCoordinateUnits","getMercCoords","CanonicalTileID","calculateKey","quadkey","mask","getQuadkey","isChildOf","dz","getTilePoint","UnwrappedTileID","OverscaledTileID","overscaledZ","scaledTo","targetZ","zDifference","calculateScaledKey","withWrap","children","sourceMaxZoom","isLessThan","wrapped","unwrapTo","overscaleFactor","toUnwrapped","DEMData","_idx","MIN_SAFE_INTEGER","ele","pixels","unpack","getUnpackVector","getPixels","backfillBorder","borderTile","xMin","xMax","yMin","yMax","DictionaryCoder","strings","_stringToNumber","_numberToString","encode","GeoJSONFeature","vectorTileFeature","_vectorTileFeature","_z","_x","_y","FeatureIndex","tileID","grid3D","featureIndexArray","loadVTLayers","vtLayers","vt","Protobuf","rawTileData","sourceLayerCoder","styleLayers","serializedLayers","sourceFeatureState","featureFilter","queryPadding","getBounds","matching","cameraBounds","cameraQueryGeometry","matching3D","bx1","by1","bx2","by2","boxX1","boxY1","boxX2","boxY2","corner","polygonIntersectsBox","topDownFeatureComparator","previousIndex","featureGeometry","loadMatchingFeature","filterLayerIDs","layerIDs","bucketLayerIDs","arraysIntersect","sourceLayerName","getId","layerID","getState","serializedLayer","evaluateProperties","intersectionZ","geojsonFeature","layerResult","lookupSymbolFeatures","symbolFeatureIndexes","symbolFeatureIndex","hasLayer","sourceLayerId","serializedProperties","styleLayerProperties","clipLine","clippedLines","clippedLine","p0","Anchor","checkMaxAngle","labelLength","windowSize","maxAngle","anchorDistance","recentCorners","recentAngleDelta","angleDelta","getLineLength","getAngleWindowSize","glyphSize","boxScale","getShapedLabelLength","getCenterAnchor","angleWindowSize","prevDistance","centerDistance","segmentDistance","getAnchors","tileExtent","shapedLabelLength","isLineContinued","resample","placeAtMiddle","halfLabelLength","markedDistance","segmentDist","getIconQuads","iconRotate","isSDFIcon","hasIconTextFit","imageWidth","imageHeight","iconWidth","iconHeight","reduceRanges","range","stretchWidth","stretchHeight","fixedWidth","fixedHeight","stretchOffsetX","stretchContentWidth","stretchOffsetY","stretchContentHeight","fixedOffsetX","fixedContentWidth","fixedOffsetY","fixedContentHeight","sumWithinRange","makeBox","leftEm","getEmOffset","stretch","leftPx","getPxOffset","fixed","topEm","topPx","rightEm","rightPx","bottomEm","bottomPx","matrix","xCuts","stretchZonesToCuts","yCuts","xi","yi","ranges","stretchZones","fixedSize","stretchSize","cuts","stretchOffset","iconSize","fixedOffset","CollisionFeature","shaped","alignLine","boxStartIndex","circleDiameter","rotateRadians","boxEndIndex","TinyQueue","_down","_up","peek","halfLength","best","findPoleOfInaccessibility","polygonRings","precision","debug","outerRing","cellSize","cellQueue","Queue","compareMax","Cell","bestCell","getCentroidCell","numProbes","minDistSq","pointToPolygonDist","SQRT2","TextAnchorEnum","aq","INVALID_TEXT_OFFSET","POSITIVE_INFINITY","evaluateVariableOffset","offsetX","offsetY","fromTextOffset","radialOffset","hypotenuse","fromRadialOffset","getTextVariableAnchorOffset","variableAnchorOffset","sourceValues","destValues","variableAnchor","textOffset","anchorOffsets","getAnchorJustification","shapedTextOrientations","imageMap","sizes","layoutIconSize","textMaxSize","defaultHorizontalShaping","getDefaultHorizontalShaping","tilePixelRatio","textMaxBoxScale","iconBoxScale","symbolMinDistance","textPadding","iconPadding","getIconPadding","textMaxAngle","iconAlongLine","textRepeatDistance","iconTextFit","verticallyShapedIcon","addSymbolAtAnchor","lineArray","textCollisionFeature","iconCollisionFeature","verticalTextCollisionFeature","verticalIconCollisionFeature","placedTextSymbolIndices","verticalTextRotation","iconQuads","verticalIconQuads","sizeData","compositeIconSizes","justifications","justification","textRotate","singleLine","addTextVertices","horizontalOnly","getCollisionCircleHeight","prevHeight","addTextVariableAnchorOffsets","addSymbol","anchorIsTooClose","poi","placementTypes","placedIconIndex","glyphQuads","textureRect","rectBuffer","rotateVerticalGlyph","halfAdvance","sdf","builtInOffset","verticalizedLabelOffset","textureScale","isDoubleResolution","verticalRotation","xHalfWidthOffsetCorrection","halfWidthOffsetCorrection","verticalOffsetCorrection","getGlyphQuads","compositeTextSizes","placementType","horizontalShaping","repeatDistance","compareText","otherAnchors","ARRAY_TYPES","KDBush","magic","versionAndType","ArrayType","nodeSize","numItems","IndexArrayType","arrayTypeIndex","coordsByteSize","idsByteSize","padCoords","_pos","_finished","numAdded","axis","qy","r2","sqDist","select","swapItem","PerformanceMarkers","bh","lastFrameTime","frameTimes","frameTimeTarget","loadTimeKey","fullLoadTimeKey","PerformanceUtils","mark","marker","currTimestamp","clearMetrics","clearMeasures","clearMarks","getPerformanceMetrics","measure","load","fullLoad","loadTime","getEntriesByName","fullLoadTime","totalFrames","fps","curr","droppedFrames","frameTime","acc","percentDroppedFrames","mapId","receive","targetMapId","tasks","cancelCallbacks","mustQueue","taskQueue","invoker","processTask","process","task","callbacks","addEventListener","globalScope","random","buffers","hasCallback","sourceMapId","transfer","completed","responseMessage","getWorkerSource","removeEventListener","muted","onloadstart","crossOrigin","appendChild","removeOrAddSourceCommands","diffSources","beforeLayers","beforeOrder","afterOrder","beforeIndex","afterIndex","tracker","clean","beforeLayer","afterLayer","insertBeforeLayerId","lastIndexOf","diffLayers","layerObject","render","uniform1i","uniformMatrix4fv","uniform3f","uniform2f","near","far","lr","bt","nf","cacheControl","header","$0","$1","$2","$3","maxAge","userAgent","safari","uSizeT","uSize","aDeltaY","aDeltaX","bDeltaY","bDeltaX","denominator","aInterpolation","b00","b01","b02","b03","b04","b05","b06","b07","b08","b09","b10","b11","det","styleKey","imgBitmap","a3","a4","a5","a6","a7","a8","a9","a14","a15","b4","b5","b6","b7","b8","b9","b12","b13","b14","b15","glMatrix.EPSILON","fovy","aspect","deferred","cachedKeys","groups","iconsNeedLinear","keepUpright","textSize","fontstack","unformattedText","spacingIfAllowed","addVerticalShapingForPointLabelIfNeeded","sdfIcons","showCollisionBoxes","_marks","resourceTimingData","origWidth","origHeight","imgData","clearRect","readImageDataUsingOffscreenCanvas","Image","revokeObjectURL","createObjectURL","remaining","results","azimuthal","polar","StyleLayerIndex","layerConfigs","keyCache","_layerConfigs","_layers","removedIds","layerConfig","createStyleLayer","familiesBySource","groupByLayout","sourceGroup","sourceLayerFamilies","GlyphAtlas","stackPositions","WorkerTile","collectResourceTiming","returnDependencies","inFlightDependencies","dependencySentinel","layerIndex","bn","buckets","layerFamilies","family","recalculateLayers","iconMap","patternMap","aH","maybePrepare","glyphAtlas","imageAtlas","performSymbolLayout","bs","glyphAtlasImage","loadVectorTile","expires","bv","rawData","ex","errorMessage","messge","VectorTileWorkerSource","loadVectorData","fetching","loading","loaded","loadTile","perf","RequestPerformance","workerTile","resourceTiming","reloadTile","parseResult","abortTile","removeTile","RasterDEMTileWorkerSource","rawImageData","imagePixels","R","dem","rewindRings","outer","rewindRing","reverse","rewind","gj","bw","bA","geojson_wrapper","GeoJSONWrapper","FeatureWrapper","rawGeometry","tags","bB","fromVectorTileJs","tile","writeLayer","writeTile","keycache","valuecache","writeFeature","writeValue","writeProperties","writeGeometry","keyIndex","valueIndex","zigzag","count","vtPbfModule","fromGeojsonVt","defaultOptions","minPoints","props","fround","OFFSET_ID","OFFSET_NUM","OFFSET_PROP","Supercluster","assign","trees","clusterProps","time","timerId","lngX","latY","tree","_createTree","timeEnd","_cluster","getClusters","minLng","minLat","maxLng","maxLat","easternHem","westernHem","_limitZoom","clusters","getClusterJSON","getChildren","clusterId","originId","_getOriginId","originZoom","_getOriginZoom","errorMsg","getLeaves","limit","leaves","_appendLeaves","getTile","z2","_addTileFeatures","getClusterExpansionZoom","expansionZoom","cluster_id","skipped","point_count","isCluster","getClusterProperties","nextData","neighborIds","numPointsOrigin","numPoints","neighborId","wx","wy","clusterPropIndex","numPoints2","_map","original","yLat","abbrev","propIndex","point_count_abbreviated","simplify","sqTolerance","maxSqDist","minPosToMid","getSqSegDist","posToMid","createFeature","calcLineBBox","calcBBox","convertFeature","convertPoint","convertLine","convertLines","projectX","projectY","k2","minAll","maxAll","clipped","newGeometry","clipPoints","clipLines","newGeom","trackMetrics","segLen","newSlice","intersect","intersectX","intersectY","az","exited","addPoint","shiftFeatureCoords","newFeatures","shiftCoords","newPolygon","newPoints","transformTile","transformed","transformPoint","createTile","numSimplified","numFeatures","simplified","tileFeature","isOuter","GeoJSONVT","tileCoords","indexMaxZoom","indexMaxPoints","stats","total","merged","splitTile","toID","getFeatureId","isUpdateableGeoJSON","seenIds","toUpdateable","Map","cz","k3","k4","z0","GeoJSONWorkerSource","loadGeoJSON","_dataUpdateable","getJSON","dataDiff","updateable","diff","removeAll","delete","cloneProperties","removeAllProperties","removeProperties","_b","addOrUpdateProperties","_c","_d","applySourceDiff","loadGeoJSONTile","_geoJSONIndex","geoJSONTile","geojsonWrapper","_geojsonTileLayer","_feature","vtpbf","byteOffset","loadData","_pendingRequest","_pendingCallback","abandoned","bC","superclusterOptions","mapExpressions","reduceExpressions","propertyNames","mapExpression","mapExpressionParsed","reduceExpressionParsed","pointProperties","getSuperclusterOptions","geojsonvt","geojsonVtOptions","getClusterChildren","getClusterLeaves","Worker","Actor","C","layerIndexes","workerSourceTypes","workerSources","demWorkerSources","registerWorkerSource","WorkerSource","registerRTLTextPlugin","setReferrer","mapID","setImages","workerSource","ws","setLayers","getLayerIndex","updateLayers","loadDEMTile","getDEMWorkerSource","removeDEMTile","loadWorkerSource","importScripts","syncRTLPluginState","getAvailableImages","DOM","docStyle","tagName","className","container","el","namespaceURI","createElementNS","selectProp","userSelect","transformProp","capture","preventDefault","stopPropagation","suppressClickInternal","getBoundingClientRect","clientX","clientLeft","clientY","clientTop","touches","button","node","parentNode","removeChild","documentElement","testProp","webpSupported","supported","testSupport","webpCheckComplete","webpImgTest","webpImgTestOnloadComplete","testWebpTextureUpload","glForTesting","createTexture","bindTexture","TEXTURE_2D","texImage2D","RGBA","UNSIGNED_BYTE","isContextLost","deleteTexture","ImageRequest","ResourceType","imageRequestQueue","currentParallelImageRequests","throttleControlCallbackHandleCounter","throttleControlCallbacks","resetRequestQueue","addThrottleControl","handle","removeThrottleControl","callbackHandle","processQueue","supportImageRefresh","accept","cancelled","innerRequest","doImageRequest","itemInQueue","getImageUsingHtmlImage","onImageResponse","HTMLImageElement","arrayBufferToImageBitmap","arrayBufferToImage","arrayBufferToCanvasImageSource","imgErr","imgResult","maxImageRequests","allControlKeys","throttleingRequested","numImageRequests","topItemInQueue","requestCancelled","fetchPriority","RequestManager","transformRequestFn","_transformRequestFn","transformRequest","normalizeSpriteURL","extension","urlObject","parts","urlRe","authority","parseUrl","formatUrl","setTransformRequest","sqrLen","coerceSpriteToArray","resultArray","dedupArray","doOnceCompleted","callbackFunc","jsonsMap","imagesMap","expectedResultCounter","spriteName","spriteData","Texture","useMipmap","pixelStoreUnpackFlipY","pixelStoreUnpack","pixelStoreUnpackPremultiplyAlpha","premultiply","HTMLCanvasElement","HTMLVideoElement","texSubImage2D","isSizePowerOfTwo","generateMipmap","minFilter","LINEAR_MIPMAP_NEAREST","LINEAR","texParameteri","TEXTURE_MAG_FILTER","TEXTURE_MIN_FILTER","TEXTURE_WRAP_S","TEXTURE_WRAP_T","renderStyleImage","userImage","ImageManager","callbackDispatchedThisFrame","requestors","atlasImage","setLoaded","_notify","addImage","valid","_validateStretch","_validateContent","updateImage","oldImage","removeImage","listImages","getImages","hasAllDependencies","getPixelSize","getPattern","pattern","_updatePatternAtlas","atlasTexture","CLAMP_TO_EDGE","dst","beginFrame","INF","edt","gridSize","edt1d","qr","GlyphManager","requestManager","localIdeographFontFamily","entries","setURL","getGlyphs","asyncAll","entry","requests","_tinySDF","loadGlyphRange","_doesCharSupportLocalGlyph","cb","fontFamily","tinySDF","fontWeight","TinySDF","fontSize","cutoff","draw","glyphWidth","glyphHeight","glyphLeft","glyphTop","glyphAdvance","urlTemplate","Glyphs","parseGlyphPbf","fontStyle","_createCanvas","textBaseline","textAlign","gridOuter","gridInner","actualBoundingBoxAscent","actualBoundingBoxDescent","actualBoundingBoxLeft","actualBoundingBoxRight","measureText","fillText","LightPositionProperty","sphericalToCartesian","B","lightProperties","Light","lightOptions","D","_transitionable","T","_transitioning","getLight","LineAtlas","nextRow","dashEntry","getDash","dasharray","addDash","getDashRanges","lineAtlasWidth","isDash","zeroLength","currentDashLength","dashLength","addRoundDash","halfStretch","currIndex","distLeft","distRight","minDist","signedDistance","distMiddle","distEdge","addRegularDash","ALPHA","REPEAT","Dispatcher","workerPool","actors","currentActor","workers","acquire","broadcast","getActor","mapRemoved","release","loadTileJson","tileJSON","pick","F","vector_layers","vectorLayers","vectorLayerIds","Source","LngLatBounds","sw","ne","setSouthWest","setNorthEast","_ne","L","_sw","sw2","ne2","getCenter","getSouthWest","getNorthEast","getNorthWest","getWest","getNorth","getSouthEast","getEast","getSouth","contains","lnglat","containsLongitude","latAccuracy","lngAccuracy","TileBounds","validateBounds","level","G","H","VectorTileSource","dispatcher","eventedParent","_loaded","dataType","_tileJSONRequest","_options","_requestManager","sourceCaches","clearTiles","tileBounds","sourceDataType","reparseOverscaled","_collectResourceTiming","hasTile","setSourceProperty","setTiles","setUrl","getPixelRatio","Tile","_refreshExpiredTiles","setExpiryData","reloadCallback","unloadTile","unloadVectorData","RasterTileSource","roundZoom","expiry","getTileTexture","extTextureFilterAnisotropic","texParameterf","TEXTURE_MAX_ANISOTROPY_EXT","extTextureFilterAnisotropicMax","saveTileTexture","RasterDEMTileSource","needsHillshadePrepare","needsTerrainPrepare","neighboringTiles","_getNeighboringTiles","J","M","readImageNow","pxw","nxw","backfilled","O","demTexture","fbo","GeoJSONSource","_updateWorkerData","_data","_removed","_pendingLoads","workerOptions","setData","setClusterOptions","rasterBoundsAttributes","ImageSource","newCoordinates","successCallback","_request","_finishLoading","prepare","boundsBuffer","_boundsArray","boundsSegments","simpleSegment","newTilesLoaded","setCoordinates","cornerCoords","U","fromLngLat","dMax","W","getCoordinatesCenterTileID","RasterBoundsArray","VideoSource","getVideo","loop","triggerRepaint","play","readyState","paused","pause","seek","seconds","seekableRange","seekable","Y","currentTime","CanvasSource","getElementById","_hasInvalidDimensions","_playing","animate","getCanvas","registeredSources","getSourceType","getPixelPosMatrix","mat4.create","mat4.translate","mat4.scale","mat4.multiply","calculatePosMatrix","queryRenderedFeatures","sourceCache","has3DLayer","queryIncludes3DLayer","maxPitchScaleFactor","tilesIn","sortTilesIn","renderedFeatureLayers","tileIn","wrappedTileID","queryResults","_state","wrappedIDLayerMap","wrappedID","wrappedIDLayers","tileFeatures","wrappedIDFeatures","resultFeatures","mergeRenderedFeatureLayers","featureWrapper","getFeatureState","idA","idB","timeAdded","fadeEndTime","uniqueId","uses","expirationTime","hasSymbolBuckets","dependencies","rtt","rttCoords","expiredRequestCount","registerFadeDuration","wasRequested","clearTextures","justReloaded","hasData","latestFeatureIndex","latestRawTileData","getLayer","lId","deserializeBucket","lazyLoadRTLTextPlugin","imageAtlasTexture","glyphAtlasTexture","getBucket","querySourceFeatures","patternsLoaded","parsedCC","parseCacheControl","getTime","isExpired","delta","getExpiryTimeout","setFeatureState","sourceLayerStates","holdingForFade","symbolFadeHoldUntil","symbolFadeFinished","clearFadeHold","setHoldDuration","setDependencies","namespace","dep","hasDependency","namespaces","TileCache","reset","removedData","timeout","clearTimeout","order","expiryTimeout","dataWrapper","_getAndRemoveByKey","getAndRemove","getByKey","dataIndex","setMaxSize","filterFn","removed","SourceFeatureState","stateChanges","deletedStates","updateState","featureId","newState","ft","removeFeatureState","reconciledState","featureDeletions","initializeTileState","coalesceChanges","featuresChanged","layerStates","SourceCache","_sourceLoaded","_paused","reload","_didEmitContent","_sourceErrored","_source","createSource","_tiles","_cache","_unloadTile","_timers","_cacheTimers","_maxTileCacheSize","_maxTileCacheZoomLevels","_loadedParentTiles","_coveredTiles","_updated","used","usedForTerrain","getSource","resume","shouldReload","_shouldReloadOnResume","_loadTile","_abortTile","getIds","compareTileId","getRenderableIds","symbolLayer","renderables","_isIdRenderable","a_","b_","rotatedA","rotatedB","hasRenderableParent","parentTile","findLoadedParent","_reloadTile","_tileLoaded","previousState","refreshedUponExpiration","_setTileReloadTimer","_backfillDEM","borderId","getTileByID","fillBorder","_retainLoadedChildren","idealTiles","maxCoveringZoom","retain","topmostLoadedID","parentID","minCoveringZoom","parentTileID","_getLoadedTile","updateCacheSize","widthInTiles","heightInTiles","viewDependentMaxSize","handleWrapJump","wrapDelta","_prevLng","idealTileIDs","getVisibleUnwrappedCoordinates","unwrapped","coveringTiles","coveringZoomLevel","maxOverzooming","maxUnderzooming","parents","parent2","noPendingDataEmissions","_updateRetainedTiles","isRasterType","parentsForFading","fadingTiles","_addTile","idealRasterTileIDs","missingTileIDs","retainedId","keysDifference","_fadeDuration","_removeTile","_updateLoadedParentTileCache","releaseSymbolFadeTiles","checked","missingTiles","childCoord","childTile","parentWasRequested","parentId","tileKey","currentId","pointQueryGeometry","tileResults","cameraPointQueryGeometry","getCameraQueryGeometry","pointCoordinate","tileSpaceBounds","tileSpaceQueryGeometry","tileSpaceCameraQueryGeometry","getVisibleCoordinates","posMatrix","reloadTilesForDependencies","aWrap","bWrap","PRELOAD_POOL_ID","WorkerPool","active","workerCount","numActive","terminate","isPreloaded","availableLogicalProcessors","globalWorkerPool","getGlobalWorkerPool","isSafari","globalThis","PathInterpolator","points_","padding_","_distances","paddedLength","lerp","distOfCurrentIdx","distToTarget","idxOfPrevPoint","distOfPrevIdx","segmentLength","segmentT","overlapAllowed","overlapA","overlapB","allowed","GridIndex","boxCells","circleCells","xCellCount","yCellCount","circleKeys","boxKeys","circles","xScale","yScale","boxUid","circleUid","keysLength","_insertBoxCell","insertCircle","_insertCircleCell","_query","hitTest","overlapMode","predicate","hitTestCircle","_queryCellCircle","queryArgs","boxCell","circleCell","_circleAndRectCollide","_circlesCollide","_convertToXCellCoord","_convertToYCellCoord","r1","bothRadii","circleX","circleY","halfRectWidth","distX","halfRectHeight","distY","getLabelPlaneMatrix","pitchWithMap","rotateWithMap","mat4.rotateZ","labelPlaneMatrix","getGlCoordMatrix","mat4.clone","glCoordMatrix","getElevation","xyTransformMat4","signedDistanceFromCamera","getPerspectiveRatio","isVisible","anchorPos","clippingBuffer","updateLineLabels","rotateToLine","partiallyEvaluatedSize","symbolSize.evaluateSizeForZoom","placedSymbols","aspectRatio","useVertical","hideGlyphs","perspectiveRatio","symbolSize.evaluateSizeForFeature","aj","pitchScaledFontSize","tileAnchorPoint","projectionCache","projections","offsets","placeUnflipped","placeGlyphsAlongLine","notEnoughRoom","needsFlipping","placeFirstAndLastGlyph","flip","glyphEndIndex","lineEndIndex","firstGlyphOffset","lastGlyphOffset","firstPlacedGlyph","placeGlyphAlongLine","lastPlacedGlyph","requiresOrientationChange","firstPoint","lastPoint","placedGlyphs","firstAndLastGlyph","orientationChange","glyphIndex","tileVertexIndex","tileSegmentEnd","projectedVertex","projectTruncatedLineSegment","singleGlyph","ak","previousTilePoint","currentTilePoint","previousProjectedPoint","minimumLength","projectionMatrix","projectedUnitVertex","projectedUnitSegment","projectVertexToViewport","projectionArgs","distanceFromAnchor","previousVertex","direction","absOffsetX","projection","previousLineVertexIndex","transformToOffsetNormal","segmentVector","findOffsetIntersectionPoint","prevToCurrentOffsetNormal","offsetPreviousVertex","offsetCurrentVertex","currentToNextOffsetNormal","offsetNextSegmentBegin","offsetNextSegmentEnd","findLineIntersection","al","anchorSegment","combinedOffsetX","offsetIntersectionPoint","currentSegmentDistance","pathVertices","currentLineSegment","prevToCurrent","segmentAngle","hiddenGlyphAttributes","viewportPadding","CollisionIndex","ignoredGrid","pitchfactor","_pitch","screenRightBoundary","screenBottomBoundary","gridRightBoundary","gridBottomBoundary","perspectiveRatioCutoff","placeCollisionBox","collisionBox","textPixelRatio","collisionGroupPredicate","projectedPoint","projectAndGetPerspectiveRatio","tileToViewport","tlX","tlY","brX","brY","isInsideGrid","offscreen","isOffscreen","placeCollisionCircles","labelToScreenMatrix","showCollisionCircles","circlePixelDiameter","textPixelPadding","placedCollisionCircles","tileUnitAnchorPoint","screenAnchorPoint","projection.project","projection.getPerspectiveRatio","labelPlaneFontScale","labelPlaneAnchorPoint","projection.placeFirstAndLastGlyph","collisionDetected","inGrid","entirelyOffscreen","screenPlaneMin","screenPlaneMax","interpolator","projectedPath","circleDist","screenSpacePath","minPoint","maxPoint","seg","numCircles","circlePosition","centerX","centerY","queryRenderedSymbols","viewportQueryGeometry","gridPoint","seenFeatures","featureKey","bucketInstanceId","intersectionTests.polygonIntersectsPolygon","insertCollisionBox","ignorePlacement","collisionGroupID","insertCollisionCircles","collisionCircles","projection.xyTransformMat4","getViewportMatrix","pixelValue","OpacityState","prevState","increment","placed","skipFade","opacity","JointOpacityState","placedText","placedIcon","JointPlacement","CollisionCircleArray","invProjMatrix","viewportMatrix","RetainedQueryData","CollisionGroups","crossSourceCollisions","maxGroupID","collisionGroups","ID","nextGroupID","calculateVariableLayoutShift","au","shiftVariableCollisionBox","rotatedOffset","Placement","prevPlacement","collisionIndex","placements","opacities","variableOffsets","stale","commitTime","retainedQueryData","collisionCircleArrays","placedOrientations","getBucketParts","sortAcrossTiles","symbolBucket","bucketFeatureIndex","pixelsToTiles","textLabelPlaneMatrix","projection.getLabelPlaneMatrix","glMatrix","projection.getGlCoordMatrix","partiallyEvaluatedTextSize","ah","collisionGroup","attemptAnchorPlacement","textAnchorOffset","textOverlapMode","orientation","placedGlyphBoxes","prevAnchor","markUsedJustification","markUsedOrientation","placeLayerBucketPart","bucketPart","seenCrossTileIDs","textOptional","iconOptional","ar","textAlwaysOverlap","iconOverlapMode","iconAlwaysOverlap","zOrderByViewportY","alwaysShowText","alwaysShowIcon","placeSymbol","placeText","placeIcon","placedVerticalText","placedGlyphCircles","placedIconBoxes","updatePreviousOrientationIfNotPlaced","isPlaced","previousOrientation","prevPlacedOrientation","placeTextForPlacementModes","placeHorizontalFn","placeVerticalFn","placementMode","textAnchorOffsetStart","textAnchorOffsetEnd","placeBox","collisionTextBox","placedFeature","placeBoxForVariableAnchors","collisionIconBox","variableIconBox","placedBox","placementPasses","prevOrientation","prevOffset","placeIconFeature","shiftedIconBox","iconWithoutText","textWithoutIcon","circleArray","symbolIndexes","symbolIndex","mat4.invert","placedAnchor","autoIndex","indexes","horizontalIndexes","commit","zoomAtLastRecencyCheck","placementChanged","prevZoomAdjustment","zoomAdjustment","symbolFadeChange","prevOpacities","prevOffsets","prevOrientations","jointPlacement","prevOpacity","jointOpacity","lastPlacementChangeTime","updateLayerOpacities","updateBucketOpacities","duplicateOpacityState","textAllowOverlap","iconAllowOverlap","hasVariablePlacement","defaultOpacityState","addOpacities","PACKED_HIDDEN_OPACITY","opacityState","horizontalHidden","verticalHidden","packedOpacity","packOpacity","symbolHidden","useHorizontal","variableOffset","updateCollisionVertices","verticalIconUsed","instance","hasTransitions","stillRecent","durationAdjustment","setStale","notUsed","shift25","shift24","shift17","shift16","shift9","shift8","shift1","targetBit","opacityBits","LayerPlacement","_sortAcrossTiles","_currentTileIndex","_currentPartIndex","_seenCrossTileIDs","_bucketParts","continuePlacement","placement","shouldPausePlacement","bucketParts","PauseablePlacement","forceFullPlacement","_currentPlacementIndex","_forceFullPlacement","_showCollisionBoxes","_done","isDone","layerTiles","startTime","placementZoom","_inProgressLayer","roundingFactor","TileLayerIndex","_symbolsByKey","symbolInstancesByKey","instances","symbols","crossTileIDs","getScaledCoordinates","childTileID","localX","localY","localZ","yWorld","yOffset","findMatches","newTileID","zoomCrossTileIDs","scaledSymbolCoord","thisTileSymbol","getCrossTileIDsLists","CrossTileIDs","maxCrossTileID","generate","CrossTileSymbolLayerIndex","usedCrossTileIDs","zoomIndexes","newZoomIndex","addBucket","removeBucketCrossTileIDs","childIndex","parentIndex","removedBucket","removeStaleBuckets","currentIDs","tilesChanged","CrossTileSymbolIndex","maxBucketInstanceId","bucketsInCurrentPlacement","symbolBucketsChanged","currentBucketIDs","pruneUnusedLayers","usedLayers","usedLayerMap","usedLayer","_emitValidationErrors","supportedDiffOperations","diffOperations","ignoredDiffOperations","empty","emptyStyle","aw","Style","_getMapId","glyphManager","lineAtlas","crossTileSymbolIndex","_spritesImagesIds","_order","_availableImages","_resetUpdates","_rtlTextPluginCallback","registerForPluginStateChange","aA","elem","_validateLayer","loadURL","previousStyle","_load","loadJSON","loadEmpty","nextState","transformStyle","stylesheet","_loadSprite","_createLayers","setTerrain","dereferencedLayers","aB","_serializedLayers","styledLayer","isUpdate","completion","_spriteRequest","originalSprite","spriteArray","spriteArrayLength","combinedRequestsMap","jsonRequestParameters","SpriteJSON","jsonRequestKey","imageRequestParameters","SpriteImage","imageRequestKey","requst","loadSprite","spriteId","imagesToRemove","_changedImages","imageId","_changed","_unloadSprite","flat","_updatedSources","_serializeByIds","serializedLayersDictionary","_serializedAllLayers","allLayerIds","_checkLoaded","changed","updatedIds","_updatedLayers","_removedLayers","_updateWorkerLayers","action","_reloadSource","_clearSource","_updateTilesForChangedImages","_updateTilesForChangedGlyphs","_updatedPaintProps","sourcesUsedBefore","changedImages","_glyphsDidChange","serializedStyle","changes","diffStyles","unimplementedOps","_afterImageUpdated","isSourceLoaded","geojsonSource","validateCustomStyleLayer","aF","_layerOrderChanged","_updateLayer","moveLayer","newIndex","getLayersOrder","getFilter","aG","getTerrain","myStyleSheet","_flattenAndSortRenderedFeatures","sourceResults","isLayer3D","features3D","sourceResult","layerFeatures","topmost3D","includedSources","renderedSymbols","bucketQueryData","queryData","bucketSymbols","layerSymbols","sortedA","symbolFeature","layerName","dataTiles","dataID","addSourceType","SourceType","setSourceType","workerSourceURL","_update","_remove","rtlTextPluginEvented","_updateSources","_generateCollisionBoxes","_updatePlacement","placementCommitted","layerBucketsChanged","pauseablePlacement","_releaseSymbolFadeTiles","getResource","getGlyphsUrl","glyphsUrl","addSprite","spriteToAdd","updatedSprite","removeSprite","internalSpriteRepresentation","find","findIndex","getSprite","aK","posAttributes","terrainVert","shaders","prelude","compile","backgroundPattern","clippingMask","heatmapTexture","collisionCircle","fillOutline","fillOutlinePattern","fillPattern","fillExtrusion","fillExtrusionPattern","hillshadePrepare","lineGradient","linePattern","lineSDF","symbolIcon","symbolSDF","symbolTextAndIcon","terrainDepth","terrainCoords","fragmentSource","vertexSource","re","staticAttributes","fragmentUniforms","vertexUniforms","staticUniforms","fragmentPragmas","operation","attrType","unpackType","VertexArrayObject","boundProgram","boundLayoutVertexBuffer","boundPaintVertexBuffers","boundIndexBuffer","boundVertexOffset","boundDynamicVertexBuffer","vao","program","paintVertexBuffers","dynamicVertexBuffer","dynamicVertexBuffer2","dynamicVertexBuffer3","paintBuffersDiffer","boundDynamicVertexBuffer2","boundDynamicVertexBuffer3","freshBind","bindVertexArray","dynamicDraw","numNextAttributes","numAttributes","createVertexArray","enableAttributes","vertexBuffer","setVertexAttribPointers","currentNumAttributes","deleteVertexArray","getTokenizedAttributesAndUniforms","token","Program","configuration","fixedUniforms","showOverdrawInspector","createProgram","staticAttrInfo","dynamicAttrInfo","allAttrInfo","preludeUniformsInfo","staticUniformsInfo","dynamicUniformsInfo","uniformList","allUniformsInfo","fragmentShader","createShader","FRAGMENT_SHADER","failedToCreate","shaderSource","compileShader","getShaderParameter","COMPILE_STATUS","getShaderInfoLog","attachShader","vertexShader","VERTEX_SHADER","attributes","uniformLocations","bindAttribLocation","linkProgram","getProgramParameter","LINK_STATUS","getProgramInfoLog","deleteShader","it","uniformLocation","getUniformLocation","terrainUniforms","u_depth","Uniform1i","aL","u_terrain","u_terrain_dim","aM","u_terrain_matrix","UniformMatrix4f","aN","u_terrain_unpack","aO","u_terrain_exaggeration","terrainPreludeUniforms","drawMode","depthMode","stencilMode","colorMode","cullFaceMode","uniformValues","currentProperties","dynamicLayoutBuffer","dynamicLayoutBuffer2","dynamicLayoutBuffer3","setDepthMode","setStencilMode","setColorMode","setCullFace","activeTexture","TEXTURE2","depthTexture","TEXTURE3","primitiveSize","LINES","TRIANGLES","LINE_STRIP","drawElements","UNSIGNED_SHORT","patternUniformValues","tileRatio","numTiles","tileSizeAtNearestZoom","pixelX","pixelY","u_image","u_texsize","u_scale","u_fade","u_pixel_coord_upper","u_pixel_coord_lower","fillExtrusionUniformValues","shouldUseVerticalGradient","_lp","lightPos","lightMat","mat3.create","mat3.fromRotation","vec3.transformMat3","lightColor","u_matrix","u_lightpos","u_lightintensity","u_lightcolor","u_vertical_gradient","u_opacity","fillExtrusionPatternUniformValues","u_height_factor","fillUniformValues","fillPatternUniformValues","fillOutlineUniformValues","drawingBufferSize","u_world","fillOutlinePatternUniformValues","circleUniformValues","extrudeScale","pixelsToGLUnits","u_camera_to_center_distance","u_scale_with_map","translatePosMatrix","u_pitch_with_map","u_device_pixel_ratio","u_extrude_scale","collisionUniformValues","u_pixels_to_tile_units","u_overscale_factor","debugUniformValues","scaleRatio","u_color","u_overlay","u_overlay_scale","clippingMaskUniformValues","heatmapUniformValues","u_intensity","getTileLatRange","lineUniformValues","calculateMatrix","u_ratio","u_units_to_pixels","lineGradientUniformValues","u_image_height","linePatternUniformValues","tileZoomRatio","calculateTileRatio","lineSDFUniformValues","posA","posB","widthA","widthB","u_patternscale_a","u_patternscale_b","u_sdfgamma","u_tex_y_a","u_tex_y_b","u_mix","rasterUniformValues","parentTL","parentScaleBy","fade","u_tl_parent","u_scale_parent","u_buffer_scale","u_fade_t","mix","u_image0","u_image1","u_brightness_low","u_brightness_high","u_saturation_factor","saturation","u_contrast_factor","contrast","u_spin_weights","spinWeights","symbolIconUniformValues","rotateInShader","texSize","u_is_size_zoom_constant","u_is_size_feature_constant","u_size_t","u_size","u_pitch","u_rotate_symbol","u_aspect_ratio","u_fade_change","u_label_plane_matrix","u_coord_matrix","u_is_text","u_texture","symbolSDFUniformValues","isHalo","u_gamma_scale","u_is_halo","symbolTextAndIconUniformValues","texSizeSDF","texSizeIcon","u_texsize_icon","u_texture_icon","backgroundUniformValues","backgroundPatternUniformValues","imagePosA","imagePosB","u_pattern_tl_a","u_pattern_br_a","u_pattern_tl_b","u_pattern_br_b","u_pattern_size_a","u_pattern_size_b","u_scale_a","u_scale_b","u_tile_units_to_pixels","bgPatternUniformValues","programUniforms","Uniform3f","aP","Uniform2f","aQ","u_inv_matrix","u_viewport_size","aR","u_color_ramp","u_latrange","u_light","u_shadow","u_highlight","u_accent","u_dimension","u_zoom","u_unpack","u_ele_delta","u_terrain_coords_id","IndexBuffer","createBuffer","unbindVAO","bindElementBuffer","bufferData","ELEMENT_ARRAY_BUFFER","DYNAMIC_DRAW","STATIC_DRAW","bufferSubData","deleteBuffer","AttributeType","VertexBuffer","bindVertexBuffer","ARRAY_BUFFER","attribIndex","enableVertexAttribArray","vertexAttribPointer","WeakMap","isWebGL2","getParameter","VERSION","BaseValue","getDefault","setDefault","ClearColor","clearColor","ClearDepth","clearDepth","ClearStencil","clearStencil","ColorMask","colorMask","DepthMask","depthMask","StencilMask","stencilMask","StencilFunc","func","ALWAYS","stencilFunc","StencilOp","KEEP","stencilOp","StencilTest","enable","STENCIL_TEST","disable","DepthRange","depthRange","DepthTest","DEPTH_TEST","DepthFunc","LESS","depthFunc","Blend","BLEND","BlendFunc","ONE","ZERO","blendFunc","BlendColor","blendColor","BlendEquation","FUNC_ADD","blendEquation","CullFace","CULL_FACE","CullFaceSide","BACK","cullFace","FrontFace","CCW","frontFace","ProgramValue","useProgram","ActiveTextureUnit","TEXTURE0","Viewport","drawingBufferWidth","drawingBufferHeight","BindFramebuffer","bindFramebuffer","FRAMEBUFFER","BindRenderbuffer","bindRenderbuffer","RENDERBUFFER","BindTexture","BindVertexBuffer","bindBuffer","BindElementBuffer","BindVertexArray","getExtension","bindVertexArrayOES","PixelStoreUnpack","pixelStorei","UNPACK_ALIGNMENT","PixelStoreUnpackPremultiplyAlpha","UNPACK_PREMULTIPLY_ALPHA_WEBGL","PixelStoreUnpackFlipY","UNPACK_FLIP_Y_WEBGL","FramebufferAttachment","ColorAttachment","setDirty","framebufferTexture2D","COLOR_ATTACHMENT0","DepthAttachment","framebufferRenderbuffer","DEPTH_ATTACHMENT","DepthStencilAttachment","DEPTH_STENCIL_ATTACHMENT","Framebuffer","hasDepth","hasStencil","framebuffer","createFramebuffer","colorAttachment","depthAttachment","checkFramebufferStatus","FRAMEBUFFER_COMPLETE","renderbuffer","deleteRenderbuffer","deleteFramebuffer","ColorMode","blendFunction","Replace","disabled","aT","unblended","alphaBlended","Context","stencilTest","depthTest","blend","cullFaceSide","MAX_TEXTURE_MAX_ANISOTROPY_EXT","maxTextureSize","MAX_TEXTURE_SIZE","HALF_FLOAT","extColorBufferHalfFloat","RGBA16F","RGBA16F_EXT","RGB16F","RGB16F_EXT","extTextureHalfFloat","HALF_FLOAT_OES","createRenderbuffer","storageFormat","rbo","renderbufferStorage","depth","stencil","COLOR_BUFFER_BIT","DEPTH_BUFFER_BIT","STENCIL_BUFFER_BIT","mode","fail","depthFail","createVertexArrayOES","deleteVertexArrayOES","DepthMode","ReadOnly","ReadWrite","StencilMode","CullFaceMode","quadTriangles","drawCollisionDebug","tileBatches","circleCount","circleOffset","invTransform","mat4.mul","aU","colorModeForRenderPass","getTerrainData","circleProgram","vertexData","CollisionCircleLayoutArray","batch","circleIdx","collision","quadCount","triCount","QuadTriangleArray","idx","createQuadTriangles","backCCW","identityMat4","calculateVariableRenderShift","renderTextSize","updateVariableAnchorsForBucket","tileScale","updateTextFitIcon","dynamicTextLayoutVertexArray","dynamicIconLayoutVertexArray","placedTextShifts","tileAnchor","projectedAnchor","symbolProjection.project","symbolProjection.getPerspectiveRatio","evaluateSizeForFeature","shiftedAnchor","symbolProjection.hideGlyphs","placedIcons","getSymbolProgramName","drawLayerSymbols","rotationAlignment","pitchAlignment","hasSortKey","depthModeForSublayer","tileRenderState","programConfiguration","evaluateSizeForZoom","terrainData","atlasInterpolation","atlasInterpolationIcon","atlasTextureIcon","rotating","zooming","NEAREST","iconScaled","symbolProjection.getLabelPlaneMatrix","symbolProjection.getGlCoordMatrix","hasVariableAnchors","symbolProjection.updateLineLabels","uLabelPlaneMatrix","uglCoordMatrix","hasHalo","oldSegments","segmentState","TEXTURE1","drawSymbolElements","updatePatternPositionsInProgram","drawFillTiles","isOutline","fillPropertyName","programName","terrainCoord","tileMatrix","stencilModeForClipping","drawExtrusionTiles","renderHillshade","shadow","highlight","accent","moving","rasterBoundsBuffer","quadTriangleIndexBuffer","rasterBoundsSegments","prepareHillshade","textureStride","pixelData","renderTexture","mat4.ortho","hillshadeUniformPrepareValues","getFadeValues","sinceTile","sinceParent","idealZ","fadeIn","childOpacity","topColor","btmColor","leftColor","rightColor","centerColor","drawHorizontalLine","drawDebugSSRect","drawVerticalLine","SCISSOR_TEST","scissor","drawDebugTile","tileRawData","tileSizeKb","tileIdText","initDebugOverlayCanvas","debugOverlayCanvas","ctx2d","shadowColor","shadowBlur","strokeStyle","strokeText","debugOverlayTexture","drawTextToOverlay","debugBuffer","debugSegments","tileBorderIndexBuffer","drawTerrain","LEQUAL","depthRangeFor3D","mesh","getTerrainMesh","renderToTexture","getTexture","getMeshFrameDelta","Painter","_tileTextures","terrainFacilitator","renderTime","setup","numSublayers","depthEpsilon","tileExtentArray","tileExtentBuffer","tileExtentSegments","debugArray","rasterBoundsArray","viewportArray","viewportBuffer","viewportSegments","tileLineStripIndices","LineStripIndexArray","quadTriangleIndices","stencilClearMode","nextStencilID","currentStencilSource","_renderTileClippingMasks","tileIDs","_tileClippingMaskIDs","REPLACE","stencilModeFor3D","NOTEQUAL","EQUAL","stencilConfigForOverlap","minTileZ","stencilValues","zToStencilMode","GEQUAL","_showOverdrawInspector","CONSTANT_COLOR","renderPass","opaquePassEnabledForLayer","currentLayer","opaquePassCutoff","coordsAscending","coordsDescending","coordsDescendingSymbol","prepareForRender","newTiles","tilesAfterTime","mat4.equals","projMatrix","mat4.copy","getRenderableTiles","getFramebuffer","devicePixelRatio","drawDepth","getCoordsTexture","coordsIndex","drawCoords","renderLayer","showTileBoundaries","selectedSource","flatMap","vectorSources","otherSources","considerSource","selectDebugSource","drawDebug","showPadding","centerPoint","drawCrosshair","drawDebugPadding","pixelToTileScale","updateVariableAnchors","drawSymbols","strokeWidth","strokeOpacity","segmentsRenderStates","segmentsState","drawCircles","numType","internalFormat","bindTextureToFramebuffer","textureUnit","colorRampUnit","renderTextureToMap","drawHeatmap","gradient","programId","firstTile","prevProgram","programChanged","atlas","layerGradient","gradientTexture","textureResolution","potentialOverzoom","ad","nextPowerOfTwo","drawLine","drawFill","drawFillExtrusion","stencilModes","drawHillshade","textureFilter","drawRaster","isPatternMissing","drawBackground","setCustomLayerDefaults","customLayerMatrix","setBaseState","drawCustom","inViewportPixelUnitsUnits","sinA","cosA","translation","translatedMatrix","textures","overLimit","Frustum","planes","invProj","frustumCoords","vec4.mul","frustumPlanes","vec3.normalize","bz","vec3.cross","vec3.sub","Aabb","min_","max_","vec3.scale","vec3.add","quadrant","qMin","vec3.clone","qMax","distanceX","distanceY","frustum","aabbPoints","fullyInside","plane","pointsInside","vec4.dot","projMin","MAX_VALUE","projMax","EdgeInsets","Transform","minPitch","maxPitch","renderWorldCopies","maxValidLatitude","_renderWorldCopies","_minZoom","_maxZoom","_minPitch","_maxPitch","setMaxBounds","_center","_elevation","_fov","_unmodified","_edgeInsets","_posMatrixCache","_alignedPosMatrixCache","_minEleveationForCurrentTile","that","latRange","_calcMatrices","centerOffset","rotationMatrix","mat2.create","mat2.rotate","fov","_zoom","constrainedZoom","zoomScale","_constrain","elevation","isPaddingEqual","interpolatePadding","scaleZoom","utl","utr","ubl","ubr","w0","w1","extraWorldCopy","actualZ","cameraCoord","getCameraPoint","centerCoord","cameraPoint","cameraFrustum","fromInvProjectionMatrix","radiusOfMaxLvlLodInTiles","newRootTile","aabb","fullyVisible","intersectResult","refPoint","longestDim","distanceSq","vec2.sqrLen","tileDistanceToCamera","childX","childY","childZ","minMax","getMinMaxElevation","minElevation","maxElevation","unmodified","unproject","getCameraPosition","pointLocation","_pixelPerMeter","recalculateZoom","getElevationForLngLatZoom","cameraPosition","camera","setLocationAtPoint","loc","locationCoordinate","newCenter","coordinateLocation","locationPoint","coordinatePoint","pixelMatrix3D","coordinate","coord0","coord1","pixelMatrixInverse","z1","pixelMatrix","getHorizon","getMaxBounds","lngRange","unwrappedTileID","aligned","posMatrixKey","unwrappedX","ao","alignedProjMatrix","mercatorMatrix","_constraining","sy","sx","h2","w2","cameraToSeaLevelDistance","cameraToLowestPointDistance","lowestPlane","groundAngle","fovAboveCenter","topHalfSurfaceDistance","horizon","fovCenterToHorizon","topHalfSurfaceDistanceHorizon","topHalfMinDistance","farZ","nearZ","mat4.perspective","mat4.rotateX","xShift","yShift","angleCos","angleSin","alignedM","as","throttle","lastCallArgs","pending","lastCallContext","later","Hash","hashName","_getCurrentHash","hash","_hashName","keyval","_onHashChange","dragRotate","isEnabled","touchZoomRotate","getBearing","jumpTo","_updateHashUnthrottled","getHashString","history","replaceState","SecurityError","_updateHash","encodeURIComponent","addTo","mapFeedback","getZoom","getPitch","found","defaultInertiaOptions","linearity","easing","defaultPanInertiaOptions","deceleration","maxSpeed","defaultZoomInertiaOptions","defaultBearingInertiaOptions","defaultPitchInertiaOptions","HandlerInertia","_inertiaBuffer","record","settings","_drainInertiaBuffer","inertia","_onMoveEnd","panInertiaOptions","deltas","pan","pinchAround","around","zoomDelta","bearingDelta","pitchDelta","panDelta","easeOptions","calculateEasing","amount","extendDuration","noMoveStart","inertiaDuration","inertiaOptions","speed","MapMouseEvent","_defaultPrevented","defaultPrevented","originalEvent","mousePos","getCanvasContainer","MapTouchEvent","changedTouches","touchPos","lngLats","MapWheelEvent","MapEventHandler","_clickTolerance","clickTolerance","_mousedownPos","wheel","_firePreventable","mousedown","mouseup","click","dblclick","mouseover","mouseout","touchstart","touchmove","touchend","touchcancel","mapEvent","isActive","BlockableMapEventHandler","_delayContextMenu","_ignoreContextMenu","_contextMenuEvent","mousemove","contextmenu","TransformProvider","_requestedCameraState","BoxZoomHandler","_tr","_el","_container","getContainer","_enabled","_active","shiftKey","disableDrag","_startPos","_lastPos","mousemoveWindow","_box","classList","_fireEvent","setTransform","mouseupWindow","suppressClick","cameraAnimation","fitScreenCoordinates","keydown","keyCode","enableDrag","indexTouches","SingleTapRecognizer","numTouches","mapTouches","timeStamp","getCentroid","newTouches","TapRecognizer","singleTap","numTaps","lastTime","lastTap","tap","soonEnough","closeEnough","TapZoomHandler","_zoomIn","_zoomOut","zoomInPoint","zoomOutPoint","easeTo","DragHandler","_moveStateManager","moveStateManager","_moveFunction","move","_activateOnStart","activateOnStart","assignEvents","_moved","_lastPoint","endMove","_move","dragStart","isValidStartEvent","startMove","dragMove","isValidMoveEvent","movePoint","dragEnd","isValidEndEvent","getClickTolerance","BUTTONS_FLAGS","MouseMoveStateManager","_correctEvent","checkCorrectEvent","eventButton","mouseButton","_eventButton","_e","flag","buttons","buttonNoLongerPressed","OneFingerTouchMoveStateManager","_firstTouch","_isOneFingerTouch","targetTouches","_isSameTouchEvent","handler","generateMouseRotationHandler","bearingDegreesPerPixelMoved","mouseMoveStateManager","ctrlKey","generateMousePitchHandler","pitchDegreesPerPixelMoved","TouchPanHandler","_minTouches","cooperativeGestures","_touches","_sum","_cancelCooperativeMessage","_calculateTransform","_cooperativeGestures","_onCooperativeGesture","touchPointSum","touchDeltaSum","touchDeltaCount","prevPoint","TwoFingersTouchHandler","_firstTwoTouches","_start","getTouchById","_aroundCenter","getZoomDelta","lastDistance","TwoFingersTouchZoomHandler","_distance","_startDistance","getBearingDelta","TwoFingersTouchRotateHandler","_minDiameter","_startVector","_vector","lastVector","_isBelowThreshold","threshold","bearingDeltaSinceStart","isVertical","TwoFingersTouchPitchHandler","_valid","_firstMove","_lastPoints","_currentTouchCount","vectorA","vectorB","gestureBeginsVertically","movedA","movedB","isSameDirection","panStep","bearingStep","pitchStep","KeyboardHandler","stepOptions","_panStep","_bearingStep","_pitchStep","_rotationDisabled","altKey","metaKey","zoomDir","bearingDir","pitchDir","xDir","yDir","easeId","easeOut","disableRotation","enableRotation","wheelZoomDelta","ScrollZoomHandler","triggerRenderFrame","_onTimeout","initialEvent","_type","_delta","_lastValue","_triggerRenderFrame","_defaultZoomRate","_wheelZoomRate","setZoomRate","zoomRate","setWheelZoomRate","wheelZoomRate","_finishTimeout","isZooming","_zooming","_metaKey","deltaMode","WheelEvent","DOM_DELTA_LINE","deltaY","timeDelta","_lastWheelEventTime","_timeout","_lastWheelEvent","_frameId","_around","_aroundPoint","renderFrame","_targetZoom","_startZoom","_easing","_smoothOutEasing","targetZoom","startZoom","finished","noInertia","needsRenderFrame","_prevEase","currentEase","ba","DoubleClickZoomHandler","clickZoom","TapZoom","_clickZoom","_tapZoom","ClickZoomHandler","TapDragZoomHandler","_tap","_swipePoint","_swipeTouch","_tapTime","_tapPoint","swipePoint","newSwipePoint","DragPanHandler","mousePan","touchPan","_mousePan","_touchPan","_inertiaOptions","DragRotateHandler","mouseRotate","mousePitch","_pitchWithRotate","pitchWithRotate","_mouseRotate","_mousePitch","TwoFingersTouchZoomRotateHandler","touchZoom","touchRotate","tapDragZoom","_touchZoom","_touchRotate","_tapDragZoom","isMoving","drag","RenderFrameEvent","hasChange","HandlerManager","handleWindowEvent","handleEvent","eventName","_updatingCamera","inputEvent","mergedHandlerResult","eventsInProgress","activeHandlers","eventTouches","_getMapTouches","handlerName","_handlers","_blockedByActive","mergeHandlerResult","deactivatedHandlers","_previousActiveHandlers","_changes","_stop","_inertia","_fireEvents","_handlersById","_bearingSnap","bearingSnap","_eventsInProgress","_addDefaultHandlers","passive","listenerOptions","boxZoom","interactive","tapZoom","doubleClickZoom","touchPitch","generateMousePanHandler","dragPan","scrollZoom","keyboard","allowEndAnimation","isRotating","myName","handlerResult","eventData","_applyChanges","combined","combinedEventsInProgress","combinedDeactivatedHandlers","change","_updateMapTransform","combinedResult","_getTransformForUpdate","_terrainMovement","_elevationFreeze","_applyUpdatedTransform","newEventsInProgress","wasMoving","nowMoving","startEvents","endEvents","originalEndEvent","stillMoving","inertialEase","shouldSnapToNorth","essential","resetNorth","freezeElevation","_requestFrame","_renderTaskQueue","Camera","_renderFrameCallback","_easeStart","_easeOptions","_onEaseFrame","_easeFrameId","_requestRenderFrame","_moving","panBy","panTo","zoomTo","zoomIn","zoomOut","getPadding","setPadding","rotateTo","resetNorthPitch","snapToNorth","cameraForBounds","_cameraForBoxAndBearing","defaultPadding","edgePadding","p0world","p1world","p0rotated","p1rotated","upperRight","lowerLeft","scaleX","scaleY","rotatedPaddingOffset","offsetAtFinalZoom","fitBounds","_fitInternal","calculatedOptions","flyTo","zoomChanged","bearingChanged","pitchChanged","calculateCameraOptionsFromTo","altitudeFrom","altitudeTo","fromMerc","toMerc","distance3D","groundDistance","bb","startBearing","startPitch","startPadding","_normalizeBearing","offsetAsPoint","pointAtOffset","locationAtOffset","_normalizeCenter","finalScale","aroundPoint","currently","_rotating","pitching","_pitching","_padding","_easeId","_prepareEase","_prepareElevation","_ease","_updateElevation","speedup","_fireMoveEvents","interruptingEaseId","_finalizeElevation","_afterEase","_elevationCenter","_elevationStart","_elevationTarget","getMinTileElevationForLngLatZoom","pitch1","transformCameraUpdate","nextTransform","wasZooming","wasRotating","wasPitching","coercedOptions","curve","rho","u1","wMax","rho2","zoomOutFactor","descent","sinh","cosh","r0","S","screenSpeed","maxDuration","isEasing","allowGestures","_cancelRenderFrame","_onEaseEnd","onEaseEnd","handlers","currentBearing","queryTerrainElevation","AttributionControl","_toggleAttribution","setAttribute","removeAttribute","_updateData","_updateAttributions","_updateCompact","offsetWidth","_compact","_updateCompactMinimize","getDefaultPosition","compact","_compactButton","_setElementTitle","_innerContainer","_attribHTML","element","title","_getUIString","attributions","customAttribution","styleOwner","owner","styleId","attrib","attribHTML","innerHTML","_editLink","LogoControl","containerChildren","rel","display","TaskQueue","_queue","_id","_cleared","_currentlyRunning","running","run","defaultLocale","pos3dAttributes","TerrainSourceCache","_renderableTilesKeys","_sourceTileCache","deltaZoom","destruct","freeRtt","getTerrainCoords","_tileID","getSourceTile","searchForDEM","Terrain","qualityFactor","meshSize","_demMatrixCache","_coordsTextureSize","getDEMElevation","vec2.transformMat4","mercatorX","mercatorY","_getOverscaledTileIDFromLngLatZoom","_emptyDemTexture","_emptyDepthTexture","_emptyDemUnpack","_emptyDemMatrix","sourceTile","matrixKey","demMatrix","mat4.fromScaling","_fboDepthTexture","_fbo","_fboCoordsTexture","DEPTH_COMPONENT16","_coordsTexture","readPixels","coordsSize","_allowMercatorOverflow","_mesh","Pos3dArray","meshSize2","offsetTop","offsetBottom","offsetLeft","offsetRight","be","mercatorCoordinate","tileX","tileY","inLeftHalf","centerLng","RenderPool","_context","_size","_tileSize","_objects","_recentlyUsed","_stamp","_createObject","DEPTH_STENCIL","stamp","inUse","getObjectForId","useObject","stampObject","getOrCreateFreeObject","freeObject","freeAllObjects","isFull","LAYERS","RenderToTexture","pool","_stacks","_prevType","_rttTiles","_renderableTiles","_renderableLayerIds","_coordsDescendingInv","_coordsDescendingInvStr","isLastLayer","packageJSON","attributionControl","maplibreLogo","failIfMajorPerformanceCaveat","preserveDrawingBuffer","trackResize","refreshExpiredTiles","maxTileCacheSize","maxTileCacheZoomLevels","maxCanvasSize","touchmoveWindow","showCompass","showZoom","visualizePitch","MouseRotateWrapper","startMouse","moveMouse","offTemp","startTouch","moveTouch","mapRotateTolerance","mapPitchTolerance","touchMoveStateManager","generateOneFingerTouchRotationHandler","generateOneFingerTouchPitchHandler","supportsGeolocation","smartWrap","priorPos","anchorTranslate","applyAnchorClass","prefix","Marker","_onKeyPress","legacyCode","charCode","togglePopup","_onMapClick","targetElement","_element","_popup","isFullyLoaded","_lngLat","_offset","rotation","_rotationAlignment","_rotation","_pitchAlignment","_anchor","_opacityTimeout","metresPerPixel","_onMove","_isDragging","_pointerdownPos","_positionDelta","setLngLat","pointerEvents","_onUp","_addDragHandler","_color","_scale","_draggable","draggable","_defaultMarker","svg","createNS","defaultHeight","defaultWidth","setAttributeNS","markerLarge","page1","ellipses","rx","ry","ellipse","bgPath","borderPath","maki","circleContainer","circle1","circle2","setDraggable","getLngLat","getElement","setPopup","popup","_originalTabIndex","markerHeight","markerRadius","linearOffset","getAttribute","getPopup","isOpen","getOffset","setOffset","addClassName","removeClassName","toggleClassName","toggle","shouldBeDraggable","isDraggable","setRotation","getRotation","setRotationAlignment","getRotationAlignment","setPitchAlignment","getPitchAlignment","positionOptions","enableHighAccuracy","maximumAge","fitBoundsOptions","trackUserLocation","showAccuracyCircle","showUserLocation","numberOfWatches","noTimeout","updateScale","clientHeight","maxMeters","maxFeet","setScale","maxDistance","pow10","multiplier","getDecimalRoundNum","getRoundNum","closeButton","closeOnClick","focusAfterOpen","focusQuerySelector","normalizeOffset","cornerOffset","convertedOffset","Debug","logToElement","overwrite","MapLibreGL","maxParallelImageRequests","numRequests","workerUrl","customProtocol","loadFn","_cooperativeGesturesOnWheel","_contextLost","_frame","_contextRestored","_setupPainter","_onMapScroll","scrollTop","scrollLeft","_onWindowOnline","_interactive","platform","_failIfMajorPerformanceCaveat","_preserveDrawingBuffer","_antialias","antialias","_trackResize","_crossSourceCollisions","_crossFadingFactor","_controls","_mapId","_locale","_overridePixelRatio","_maxCanvasSize","_imageQueueHandle","HTMLElement","maxBounds","_setupContainer","_idleTriggered","initialResizeEventCaptured","throttledResizeCallback","_resizeObserver","ResizeObserver","observe","_setupCooperativeGestures","_hash","_localIdeographFontFamily","_validateStyle","addControl","logoPosition","control","controlElement","positionContainer","_controlPositions","insertBefore","firstChild","removeControl","ci","hasControl","_containerDimensions","clampedPixelRatio","_getClampedPixelRatio","_resizeCanvas","fireMoving","maxCanvasWidth","maxCanvasHeight","canvasWidth","canvasHeight","setPixelRatio","setMinZoom","getMinZoom","setMaxZoom","getMaxZoom","setMinPitch","getMinPitch","setMaxPitch","getMaxPitch","getRenderWorldCopies","setRenderWorldCopies","getCooperativeGestures","setCooperativeGestures","gestureOptions","_destroyCooperativeGestures","_createDelegatedListener","mousein","delegates","delegate","layerIdOrListener","delegatedListener","_delegatedListeners","delegatedListeners","geometryOrOptions","isGeometry","_diffStyle","_updateStyle","_lazyInitEmptyStyle","_updateDiff","getStyle","isStyleLoaded","_terrainDataCallback","thisLayer","areTilesLoaded","existingImage","imageData","hasImage","loadImage","beforeId","spriteUrl","_canvasContainer","_canvas","clientWidth","canvasContainer","controlContainer","_controlContainer","positionName","_cooperativeGesturesScreen","desktopMessage","windowsHelpText","macHelpText","mobileHelpText","webglcontextcreationerrorDetailObject","requestedAttributes","statusMessage","msg","metaPress","_styleDirty","_sourcesDirty","updateStyle","_render","paintStartTimeStamp","crossFading","_placementDirty","somethingDirty","_repaint","_fullyLoaded","redraw","disconnect","loseContext","bg","_showTileBoundaries","_showPadding","repaint","_vertices","getCameraTargetElevation","NavigationControl","_updateZoomButtons","isMax","isMin","_zoomInButton","_zoomOutButton","_rotateCompassArrow","_compassIcon","_setButtonTitle","_createButton","_compass","_handler","GeolocateControl","_onSuccess","_isOutOfMapMaxBounds","_setErrorState","_updateMarker","_finish","_lastKnownPosition","_watchState","_geolocateButton","_updateCamera","_dotElement","longitude","accuracy","newBounds","geolocateSource","_accuracyCircleMarker","_userLocationDotMarker","_accuracy","_updateCircleRadius","_onZoom","_onError","_geolocationWatchID","_clearWatch","_timeoutId","_setupUI","_circleElement","_setup","forceRecalculation","permissions","geolocation","checkGeolocationSupport","clearWatch","southEastPoint","northEastPoint","mapHeightInMeters","watchPosition","getCurrentPosition","ScaleControl","setUnit","FullscreenControl","_onFullscreenChange","fullscreenElement","mozFullScreenElement","webkitFullscreenElement","msFullscreenElement","_fullscreen","_handleFullscreenChange","_onClickFullscreen","_isFullscreen","_exitFullscreen","_requestFullscreen","_fullscreenchange","_fullscreenButton","_updateTitle","_getTitle","_prevCooperativeGestures","exitFullscreen","mozCancelFullScreen","msExitFullscreen","webkitCancelFullScreen","_togglePseudoFullScreen","requestFullscreen","mozRequestFullScreen","msRequestFullscreen","webkitRequestFullscreen","TerrainControl","_toggleTerrain","_updateTerrainIcon","_terrainButton","Popup","_content","_onClose","_onMouseMove","_onMouseUp","_onDrag","cursor","_trackPointer","_tip","offsetHeight","anchorComponents","offsetedPos","closeOnMove","_focusFirstElement","trackPointer","setText","setDOMContent","createTextNode","setHTML","html","frag","createDocumentFragment","temp","getMaxWidth","setMaxWidth","htmlNode","hasChildNodes","_createCloseButton","_closeButton","firstFocusable","querySelector","focus","setRTLTextPlugin","prewarm","clearPrewarmedResources"],"mappings":"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","x_google_ignoreList":[0,1,2,10,27,28,29,40,41,42,52,53,60,61,62,63,76,77,79,98,110,114,123,125,126,127,128,129,130,131,132,133,134,135,143,144,150,249,290]} \ No newline at end of file diff --git a/web/libraries/maplibre-gl/dist/package.json b/web/libraries/maplibre-gl/dist/package.json new file mode 100644 index 00000000..a7901014 --- /dev/null +++ b/web/libraries/maplibre-gl/dist/package.json @@ -0,0 +1 @@ +{"name":"maplibre-gl","type":"commonjs","deprecated":"Please install maplibre-gl from parent directory instead"} \ No newline at end of file diff --git a/web/libraries/maplibre-gl/package.json b/web/libraries/maplibre-gl/package.json new file mode 100644 index 00000000..06f5d9ba --- /dev/null +++ b/web/libraries/maplibre-gl/package.json @@ -0,0 +1,190 @@ +{ + "name": "maplibre-gl", + "description": "BSD licensed community fork of mapbox-gl, a WebGL interactive maps library", + "version": "3.6.2", + "main": "dist/maplibre-gl.js", + "style": "dist/maplibre-gl.css", + "license": "BSD-3-Clause", + "funding": "https://github.com/maplibre/maplibre-gl-js?sponsor=1", + "repository": { + "type": "git", + "url": "git://github.com/maplibre/maplibre-gl-js.git" + }, + "types": "dist/maplibre-gl.d.ts", + "type": "module", + "dependencies": { + "@mapbox/geojson-rewind": "^0.5.2", + "@mapbox/jsonlint-lines-primitives": "^2.0.2", + "@mapbox/point-geometry": "^0.1.0", + "@mapbox/tiny-sdf": "^2.0.6", + "@mapbox/unitbezier": "^0.0.1", + "@mapbox/vector-tile": "^1.3.1", + "@mapbox/whoots-js": "^3.1.0", + "@maplibre/maplibre-gl-style-spec": "^19.3.3", + "@types/geojson": "^7946.0.13", + "@types/mapbox__point-geometry": "^0.1.4", + "@types/mapbox__vector-tile": "^1.3.4", + "@types/pbf": "^3.0.5", + "@types/supercluster": "^7.1.3", + "earcut": "^2.2.4", + "geojson-vt": "^3.2.1", + "gl-matrix": "^3.4.3", + "global-prefix": "^3.0.0", + "kdbush": "^4.0.2", + "murmurhash-js": "^1.0.0", + "pbf": "^3.2.1", + "potpack": "^2.0.0", + "quickselect": "^2.0.0", + "supercluster": "^8.0.1", + "tinyqueue": "^2.0.3", + "vt-pbf": "^3.1.3" + }, + "devDependencies": { + "@mapbox/mapbox-gl-rtl-text": "^0.2.3", + "@mapbox/mvt-fixtures": "^3.10.0", + "@rollup/plugin-commonjs": "^25.0.7", + "@rollup/plugin-json": "^6.0.1", + "@rollup/plugin-node-resolve": "^15.2.3", + "@rollup/plugin-replace": "^5.0.5", + "@rollup/plugin-strip": "^3.0.4", + "@rollup/plugin-terser": "^0.4.4", + "@rollup/plugin-typescript": "^11.1.5", + "@types/benchmark": "^2.1.5", + "@types/cssnano": "^5.0.0", + "@types/d3": "^7.4.3", + "@types/diff": "^5.0.8", + "@types/earcut": "^2.1.4", + "@types/eslint": "^8.44.7", + "@types/geojson-vt": "3.2.4", + "@types/gl": "^6.0.5", + "@types/glob": "^8.1.0", + "@types/jest": "^29.5.8", + "@types/jsdom": "^21.1.5", + "@types/minimist": "^1.2.5", + "@types/murmurhash-js": "^1.0.6", + "@types/nise": "^1.4.4", + "@types/node": "^20.9.2", + "@types/offscreencanvas": "^2019.7.3", + "@types/pixelmatch": "^5.2.6", + "@types/pngjs": "^6.0.4", + "@types/react": "^18.2.37", + "@types/react-dom": "^18.2.15", + "@types/request": "^2.48.12", + "@types/shuffle-seed": "^1.1.2", + "@types/window-or-global": "^1.0.6", + "@typescript-eslint/eslint-plugin": "^6.11.0", + "@typescript-eslint/parser": "^6.11.0", + "address": "^2.0.1", + "benchmark": "^2.1.4", + "canvas": "^2.11.2", + "cssnano": "^6.0.1", + "d3": "^7.8.5", + "d3-queue": "^3.0.7", + "devtools-protocol": "^0.0.1226504", + "diff": "^5.1.0", + "dts-bundle-generator": "^8.1.2", + "eslint": "^8.54.0", + "eslint-config-mourner": "^3.0.0", + "eslint-plugin-html": "^7.1.0", + "eslint-plugin-import": "^2.29.0", + "eslint-plugin-jest": "^27.6.0", + "eslint-plugin-react": "^7.33.2", + "eslint-plugin-tsdoc": "0.2.17", + "expect": "^29.7.0", + "gl": "^6.0.2", + "glob": "^10.3.10", + "is-builtin-module": "^3.2.1", + "jest": "^29.7.0", + "jest-canvas-mock": "^2.5.2", + "jest-environment-jsdom": "^29.7.0", + "jsdom": "^22.1.0", + "json-stringify-pretty-compact": "^4.0.0", + "minimist": "^1.2.8", + "mock-geolocation": "^1.0.11", + "nise": "^5.1.5", + "npm-font-open-sans": "^1.1.0", + "npm-run-all": "^4.1.5", + "pdf-merger-js": "^4.3.0", + "pixelmatch": "^5.3.0", + "pngjs": "^7.0.0", + "postcss": "^8.4.31", + "postcss-cli": "^10.1.0", + "postcss-inline-svg": "^6.0.0", + "pretty-bytes": "^6.1.1", + "puppeteer": "^21.5.2", + "react": "^18.2.0", + "react-dom": "^18.2.0", + "rollup": "^4.5.0", + "rollup-plugin-sourcemaps": "^0.6.3", + "rw": "^1.3.3", + "semver": "^7.5.4", + "shuffle-seed": "^1.1.6", + "source-map-explorer": "^2.5.3", + "st": "^3.0.0", + "stylelint": "^15.11.0", + "stylelint-config-standard": "^34.0.0", + "ts-jest": "^29.1.1", + "ts-node": "^10.9.1", + "tslib": "^2.6.2", + "typedoc": "^0.25.3", + "typedoc-plugin-markdown": "^3.17.1", + "typedoc-plugin-missing-exports": "^2.1.0", + "typescript": "^5.2.2" + }, + "overrides": { + "postcss-inline-svg": { + "css-select": "^5.1.0", + "dom-serializer": "^2.0.0", + "htmlparser2": "^8.0.1", + "postcss-value-parser": "^4.2.0" + } + }, + "scripts": { + "generate-dist-package": "npm run tsnode build/generate-dist-package.js", + "generate-shaders": "npm run tsnode build/generate-shaders.ts", + "generate-struct-arrays": "npm run tsnode build/generate-struct-arrays.ts", + "generate-style-code": "npm run tsnode build/generate-style-code.ts", + "generate-typings": "npm run tsnode build/generate-typings.ts", + "generate-docs": "typedoc && npm run tsnode build/generate-docs.ts", + "generate-images": "npm run tsnode build/generate-doc-images.ts", + "build-dist": "run-p --print-label generate-typings build-dev build-prod build-csp build-csp-dev build-css", + "build-dev": "rollup --configPlugin @rollup/plugin-typescript -c --environment BUILD:dev", + "watch-dev": "rollup --configPlugin @rollup/plugin-typescript -c --environment BUILD:dev --watch", + "build-prod": "rollup --configPlugin @rollup/plugin-typescript -c --environment BUILD:production", + "build-csp": "rollup --configPlugin @rollup/plugin-typescript -c rollup.config.csp.ts", + "build-csp-dev": "rollup --configPlugin @rollup/plugin-typescript -c rollup.config.csp.ts --environment BUILD:dev", + "build-css": "postcss -o dist/maplibre-gl.css src/css/maplibre-gl.css", + "watch-css": "postcss --watch -o dist/maplibre-gl.css src/css/maplibre-gl.css", + "build-benchmarks": "npm run build-dev && rollup --configPlugin @rollup/plugin-typescript -c test/bench/rollup_config_benchmarks.ts", + "watch-benchmarks": "rollup --configPlugin @rollup/plugin-typescript -c test/bench/rollup_config_benchmarks.ts --watch", + "start-server": "st --no-cache -H 0.0.0.0 --port 9966 .", + "start-docs": "docker run --rm -it -p 8000:8000 -v ${PWD}:/docs squidfunk/mkdocs-material", + "start": "run-p watch-css watch-dev start-server", + "start-bench": "run-p watch-css watch-benchmarks start-server", + "lint": "eslint --cache --ext .ts,.tsx,.js,.html --ignore-path .gitignore .", + "lint-css": "stylelint src/css/maplibre-gl.css", + "test": "run-p lint lint-css test-render jest", + "jest": "jest", + "jest-ci": "jest --reporters=github-actions --reporters=summary", + "test-build": "jest --selectProjects=build", + "test-integration": "jest --selectProjects=integration", + "test-render": "npm run tsnode test/integration/render/run_render_tests.ts", + "test-unit": "jest --selectProjects=unit", + "test-watch-roots": "jest --watch", + "codegen": "run-p generate-dist-package generate-style-code generate-struct-arrays generate-shaders", + "benchmark": "npm run tsnode test/bench/run-benchmarks.ts", + "gl-stats": "npm run tsnode test/bench/gl-stats.ts", + "prepare": "npm run codegen", + "typecheck": "tsc --noEmit && tsc --project tsconfig.dist.json", + "tsnode": "node --experimental-loader=ts-node/esm --no-warnings" + }, + "files": [ + "build/", + "dist/*", + "src/" + ], + "engines": { + "npm": ">=8.1.0", + "node": ">=16.14.0" + } +} diff --git a/web/libraries/maplibre-gl/src/css/maplibre-gl.css b/web/libraries/maplibre-gl/src/css/maplibre-gl.css new file mode 100644 index 00000000..ccd87a9d --- /dev/null +++ b/web/libraries/maplibre-gl/src/css/maplibre-gl.css @@ -0,0 +1,827 @@ +.maplibregl-map { + font: 12px/20px "Helvetica Neue", Arial, Helvetica, sans-serif; + overflow: hidden; + position: relative; + -webkit-tap-highlight-color: rgb(0 0 0 / 0%); +} + +.maplibregl-canvas { + position: absolute; + left: 0; + top: 0; +} + +.maplibregl-map:fullscreen { + width: 100%; + height: 100%; +} + +.maplibregl-ctrl-group button.maplibregl-ctrl-compass { + touch-action: none; +} + +.maplibregl-canvas-container.maplibregl-interactive, +.maplibregl-ctrl-group button.maplibregl-ctrl-compass { + cursor: grab; + -moz-user-select: none; + -webkit-user-select: none; + -ms-user-select: none; + user-select: none; +} + +.maplibregl-canvas-container.maplibregl-interactive.maplibregl-track-pointer { + cursor: pointer; +} + +.maplibregl-canvas-container.maplibregl-interactive:active, +.maplibregl-ctrl-group button.maplibregl-ctrl-compass:active { + cursor: grabbing; +} + +.maplibregl-canvas-container.maplibregl-touch-zoom-rotate, +.maplibregl-canvas-container.maplibregl-touch-zoom-rotate .maplibregl-canvas { + touch-action: pan-x pan-y; +} + +.maplibregl-canvas-container.maplibregl-touch-drag-pan, +.maplibregl-canvas-container.maplibregl-touch-drag-pan .maplibregl-canvas { + touch-action: pinch-zoom; +} + +.maplibregl-canvas-container.maplibregl-touch-zoom-rotate.maplibregl-touch-drag-pan, +.maplibregl-canvas-container.maplibregl-touch-zoom-rotate.maplibregl-touch-drag-pan .maplibregl-canvas { + touch-action: none; +} + +.maplibregl-canvas-container.maplibregl-touch-drag-pan.maplibregl-cooperative-gestures, +.maplibregl-canvas-container.maplibregl-touch-drag-pan.maplibregl-cooperative-gestures .maplibregl-canvas { + touch-action: pan-x pan-y; +} + +.maplibregl-ctrl-top-left, +.maplibregl-ctrl-top-right, +.maplibregl-ctrl-bottom-left, +.maplibregl-ctrl-bottom-right { position: absolute; pointer-events: none; z-index: 2; } + +.maplibregl-ctrl-top-left { top: 0; left: 0; } + +.maplibregl-ctrl-top-right { top: 0; right: 0; } + +.maplibregl-ctrl-bottom-left { bottom: 0; left: 0; } + +.maplibregl-ctrl-bottom-right { right: 0; bottom: 0; } + +.maplibregl-ctrl { + clear: both; + pointer-events: auto; + + /* workaround for a Safari bug https://github.com/mapbox/mapbox-gl-js/issues/8185 */ + transform: translate(0, 0); +} + +.maplibregl-ctrl-top-left .maplibregl-ctrl { margin: 10px 0 0 10px; float: left; } + +.maplibregl-ctrl-top-right .maplibregl-ctrl { margin: 10px 10px 0 0; float: right; } + +.maplibregl-ctrl-bottom-left .maplibregl-ctrl { margin: 0 0 10px 10px; float: left; } + +.maplibregl-ctrl-bottom-right .maplibregl-ctrl { margin: 0 10px 10px 0; float: right; } + +.maplibregl-ctrl-group { + border-radius: 4px; + background: #fff; +} + +.maplibregl-ctrl-group:not(:empty) { + box-shadow: 0 0 0 2px rgb(0 0 0 / 10%); +} + +@media (-ms-high-contrast: active) { + .maplibregl-ctrl-group:not(:empty) { + box-shadow: 0 0 0 2px ButtonText; + } +} + +.maplibregl-ctrl-group button { + width: 29px; + height: 29px; + display: block; + padding: 0; + outline: none; + border: 0; + box-sizing: border-box; + background-color: transparent; + cursor: pointer; +} + +.maplibregl-ctrl-group button + button { + border-top: 1px solid #ddd; +} + +.maplibregl-ctrl button .maplibregl-ctrl-icon { + display: block; + width: 100%; + height: 100%; + background-repeat: no-repeat; + background-position: center center; +} + +@media (-ms-high-contrast: active) { + .maplibregl-ctrl-icon { + background-color: transparent; + } + + .maplibregl-ctrl-group button + button { + border-top: 1px solid ButtonText; + } +} + +/* https://bugzilla.mozilla.org/show_bug.cgi?id=140562 */ +.maplibregl-ctrl button::-moz-focus-inner { + border: 0; + padding: 0; +} + +.maplibregl-ctrl-attrib-button:focus, +.maplibregl-ctrl-group button:focus { + box-shadow: 0 0 2px 2px rgb(0 150 255 / 100%); +} + +.maplibregl-ctrl button:disabled { + cursor: not-allowed; +} + +.maplibregl-ctrl button:disabled .maplibregl-ctrl-icon { + opacity: 0.25; +} + +.maplibregl-ctrl button:not(:disabled):hover { + background-color: rgb(0 0 0 / 5%); +} + +.maplibregl-ctrl-group button:focus:focus-visible { + box-shadow: 0 0 2px 2px rgb(0 150 255 / 100%); +} + +.maplibregl-ctrl-group button:focus:not(:focus-visible) { + box-shadow: none; +} + +.maplibregl-ctrl-group button:focus:first-child { + border-radius: 4px 4px 0 0; +} + +.maplibregl-ctrl-group button:focus:last-child { + border-radius: 0 0 4px 4px; +} + +.maplibregl-ctrl-group button:focus:only-child { + border-radius: inherit; +} + +.maplibregl-ctrl button.maplibregl-ctrl-zoom-out .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-zoom-out.svg", fill: #333); +} + +.maplibregl-ctrl button.maplibregl-ctrl-zoom-in .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-zoom-in.svg", fill: #333); +} + +@media (-ms-high-contrast: active) { + .maplibregl-ctrl button.maplibregl-ctrl-zoom-out .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-zoom-out.svg", fill: #fff); + } + + .maplibregl-ctrl button.maplibregl-ctrl-zoom-in .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-zoom-in.svg", fill: #fff); + } +} + +@media (-ms-high-contrast: black-on-white) { + .maplibregl-ctrl button.maplibregl-ctrl-zoom-out .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-zoom-out.svg", fill: #000); + } + + .maplibregl-ctrl button.maplibregl-ctrl-zoom-in .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-zoom-in.svg", fill: #000); + } +} + +.maplibregl-ctrl button.maplibregl-ctrl-fullscreen .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-fullscreen.svg", fill: #333); +} + +.maplibregl-ctrl button.maplibregl-ctrl-shrink .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-shrink.svg"); +} + +@media (-ms-high-contrast: active) { + .maplibregl-ctrl button.maplibregl-ctrl-fullscreen .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-fullscreen.svg", fill: #fff); + } + + .maplibregl-ctrl button.maplibregl-ctrl-shrink .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-shrink.svg", fill: #fff); + } +} + +@media (-ms-high-contrast: black-on-white) { + .maplibregl-ctrl button.maplibregl-ctrl-fullscreen .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-fullscreen.svg", fill: #000); + } + + .maplibregl-ctrl button.maplibregl-ctrl-shrink .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-shrink.svg", fill: #000); + } +} + +.maplibregl-ctrl button.maplibregl-ctrl-compass .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-compass.svg", fill: #333); +} + +@media (-ms-high-contrast: active) { + .maplibregl-ctrl button.maplibregl-ctrl-compass .maplibregl-ctrl-icon { + @svg-load ctrl-compass-white url("svg/maplibregl-ctrl-compass.svg") { + fill: #fff; + #south { fill: #999; } + } + + background-image: svg-inline(ctrl-compass-white); + } +} + +@media (-ms-high-contrast: black-on-white) { + .maplibregl-ctrl button.maplibregl-ctrl-compass .maplibregl-ctrl-icon { + background-image: svg-load("svg/maplibregl-ctrl-compass.svg", fill: #000); + } +} + +@svg-load ctrl-terrain url("svg/maplibregl-ctrl-terrain.svg") { + fill: #333; + #stroke { display: none; } +} + +@svg-load ctrl-terrain-enabled url("svg/maplibregl-ctrl-terrain.svg") { + fill: #33b5e5; + #stroke { display: none; } +} + +.maplibregl-ctrl button.maplibregl-ctrl-terrain .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-terrain); +} + +.maplibregl-ctrl button.maplibregl-ctrl-terrain-enabled .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-terrain-enabled); +} + +@svg-load ctrl-geolocate url("svg/maplibregl-ctrl-geolocate.svg") { + fill: #333; + #stroke { display: none; } +} + +@svg-load ctrl-geolocate-white url("svg/maplibregl-ctrl-geolocate.svg") { + fill: #fff; + #stroke { display: none; } +} + +@svg-load ctrl-geolocate-black url("svg/maplibregl-ctrl-geolocate.svg") { + fill: #000; + #stroke { display: none; } +} + +@svg-load ctrl-geolocate-disabled url("svg/maplibregl-ctrl-geolocate.svg") { + fill: #aaa; + #stroke { fill: #f00; } +} + +@svg-load ctrl-geolocate-disabled-white url("svg/maplibregl-ctrl-geolocate.svg") { + fill: #999; + #stroke { fill: #f00; } +} + +@svg-load ctrl-geolocate-disabled-black url("svg/maplibregl-ctrl-geolocate.svg") { + fill: #666; + #stroke { fill: #f00; } +} + +@svg-load ctrl-geolocate-active url("svg/maplibregl-ctrl-geolocate.svg") { + fill: #33b5e5; + #stroke { display: none; } +} + +@svg-load ctrl-geolocate-active-error url("svg/maplibregl-ctrl-geolocate.svg") { + fill: #e58978; + #stroke { display: none; } +} + +@svg-load ctrl-geolocate-background url("svg/maplibregl-ctrl-geolocate.svg") { + fill: #33b5e5; + #stroke { display: none; } + #dot { display: none; } +} + +@svg-load ctrl-geolocate-background-error url("svg/maplibregl-ctrl-geolocate.svg") { + fill: #e54e33; + #stroke { display: none; } + #dot { display: none; } +} + +.maplibregl-ctrl button.maplibregl-ctrl-geolocate .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate); +} + +.maplibregl-ctrl button.maplibregl-ctrl-geolocate:disabled .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-disabled); +} + +.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-active .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-active); +} + +.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-active-error .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-active-error); +} + +.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-background .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-background); +} + +.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-background-error .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-background-error); +} + +.maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-waiting .maplibregl-ctrl-icon { + animation: maplibregl-spin 2s infinite linear; +} + +@media (-ms-high-contrast: active) { + .maplibregl-ctrl button.maplibregl-ctrl-geolocate .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-white); + } + + .maplibregl-ctrl button.maplibregl-ctrl-geolocate:disabled .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-disabled-white); + } + + .maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-active .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-active); + } + + .maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-active-error .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-active-error); + } + + .maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-background .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-background); + } + + .maplibregl-ctrl button.maplibregl-ctrl-geolocate.maplibregl-ctrl-geolocate-background-error .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-background-error); + } +} + +@media (-ms-high-contrast: black-on-white) { + .maplibregl-ctrl button.maplibregl-ctrl-geolocate .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-black); + } + + .maplibregl-ctrl button.maplibregl-ctrl-geolocate:disabled .maplibregl-ctrl-icon { + background-image: svg-inline(ctrl-geolocate-disabled-black); + } +} + +@keyframes maplibregl-spin { + 0% { transform: rotate(0deg); } + 100% { transform: rotate(360deg); } +} + +a.maplibregl-ctrl-logo { + width: 88px; + height: 23px; + margin: 0 0 -4px -4px; + display: block; + background-repeat: no-repeat; + cursor: pointer; + overflow: hidden; + background-image: svg-load("svg/maplibregl-ctrl-logo.svg"); +} + +a.maplibregl-ctrl-logo.maplibregl-compact { + width: 14px; +} + +@media (-ms-high-contrast: active) { + a.maplibregl-ctrl-logo { + @svg-load ctrl-logo-white url("svg/maplibregl-ctrl-logo.svg") { + #outline { opacity: 1; } + #fill { opacity: 1; } + } + + background-color: transparent; + background-image: svg-inline(ctrl-logo-white); + } +} + +@media (-ms-high-contrast: black-on-white) { + a.maplibregl-ctrl-logo { + @svg-load ctrl-logo-black url("svg/maplibregl-ctrl-logo.svg") { + #outline { opacity: 1; fill: #fff; stroke: #fff; } + #fill { opacity: 1; fill: #000; } + } + + background-image: svg-inline(ctrl-logo-black); + } +} + +.maplibregl-ctrl.maplibregl-ctrl-attrib { + padding: 0 5px; + background-color: rgb(255 255 255 / 50%); + margin: 0; +} + +@media screen { + .maplibregl-ctrl-attrib.maplibregl-compact { + min-height: 20px; + padding: 2px 24px 2px 0; + margin: 10px; + position: relative; + background-color: #fff; + border-radius: 12px; + box-sizing: content-box; + } + + .maplibregl-ctrl-attrib.maplibregl-compact-show { + padding: 2px 28px 2px 8px; + visibility: visible; + } + + .maplibregl-ctrl-top-left > .maplibregl-ctrl-attrib.maplibregl-compact-show, + .maplibregl-ctrl-bottom-left > .maplibregl-ctrl-attrib.maplibregl-compact-show { + padding: 2px 8px 2px 28px; + border-radius: 12px; + } + + .maplibregl-ctrl-attrib.maplibregl-compact .maplibregl-ctrl-attrib-inner { + display: none; + } + + .maplibregl-ctrl-attrib-button { + display: none; + cursor: pointer; + position: absolute; + background-image: svg-load("svg/maplibregl-ctrl-attrib.svg"); + background-color: rgb(255 255 255 / 50%); + width: 24px; + height: 24px; + box-sizing: border-box; + border-radius: 12px; + outline: none; + top: 0; + right: 0; + border: 0; + } + + .maplibregl-ctrl-attrib summary.maplibregl-ctrl-attrib-button { + appearance: none; + list-style: none; + } + + .maplibregl-ctrl-attrib summary.maplibregl-ctrl-attrib-button::-webkit-details-marker { + display: none; + } + + .maplibregl-ctrl-top-left .maplibregl-ctrl-attrib-button, + .maplibregl-ctrl-bottom-left .maplibregl-ctrl-attrib-button { + left: 0; + } + + .maplibregl-ctrl-attrib.maplibregl-compact .maplibregl-ctrl-attrib-button, + .maplibregl-ctrl-attrib.maplibregl-compact-show .maplibregl-ctrl-attrib-inner { + display: block; + } + + .maplibregl-ctrl-attrib.maplibregl-compact-show .maplibregl-ctrl-attrib-button { + background-color: rgb(0 0 0 / 5%); + } + + .maplibregl-ctrl-bottom-right > .maplibregl-ctrl-attrib.maplibregl-compact::after { + bottom: 0; + right: 0; + } + + .maplibregl-ctrl-top-right > .maplibregl-ctrl-attrib.maplibregl-compact::after { + top: 0; + right: 0; + } + + .maplibregl-ctrl-top-left > .maplibregl-ctrl-attrib.maplibregl-compact::after { + top: 0; + left: 0; + } + + .maplibregl-ctrl-bottom-left > .maplibregl-ctrl-attrib.maplibregl-compact::after { + bottom: 0; + left: 0; + } +} + +@media screen and (-ms-high-contrast: active) { + .maplibregl-ctrl-attrib.maplibregl-compact::after { + background-image: svg-load("svg/maplibregl-ctrl-attrib.svg", fill=#fff); + } +} + +@media screen and (-ms-high-contrast: black-on-white) { + .maplibregl-ctrl-attrib.maplibregl-compact::after { + background-image: svg-load("svg/maplibregl-ctrl-attrib.svg"); + } +} + +.maplibregl-ctrl-attrib a { + color: rgb(0 0 0 / 75%); + text-decoration: none; +} + +.maplibregl-ctrl-attrib a:hover { + color: inherit; + text-decoration: underline; +} + +.maplibregl-attrib-empty { + display: none; +} + +.maplibregl-ctrl-scale { + background-color: rgb(255 255 255 / 75%); + font-size: 10px; + border-width: medium 2px 2px; + border-style: none solid solid; + border-color: #333; + padding: 0 5px; + color: #333; + box-sizing: border-box; +} + +.maplibregl-popup { + position: absolute; + top: 0; + left: 0; + display: flex; + will-change: transform; + pointer-events: none; +} + +.maplibregl-popup-anchor-top, +.maplibregl-popup-anchor-top-left, +.maplibregl-popup-anchor-top-right { + flex-direction: column; +} + +.maplibregl-popup-anchor-bottom, +.maplibregl-popup-anchor-bottom-left, +.maplibregl-popup-anchor-bottom-right { + flex-direction: column-reverse; +} + +.maplibregl-popup-anchor-left { + flex-direction: row; +} + +.maplibregl-popup-anchor-right { + flex-direction: row-reverse; +} + +.maplibregl-popup-tip { + width: 0; + height: 0; + border: 10px solid transparent; + z-index: 1; +} + +.maplibregl-popup-anchor-top .maplibregl-popup-tip { + align-self: center; + border-top: none; + border-bottom-color: #fff; +} + +.maplibregl-popup-anchor-top-left .maplibregl-popup-tip { + align-self: flex-start; + border-top: none; + border-left: none; + border-bottom-color: #fff; +} + +.maplibregl-popup-anchor-top-right .maplibregl-popup-tip { + align-self: flex-end; + border-top: none; + border-right: none; + border-bottom-color: #fff; +} + +.maplibregl-popup-anchor-bottom .maplibregl-popup-tip { + align-self: center; + border-bottom: none; + border-top-color: #fff; +} + +.maplibregl-popup-anchor-bottom-left .maplibregl-popup-tip { + align-self: flex-start; + border-bottom: none; + border-left: none; + border-top-color: #fff; +} + +.maplibregl-popup-anchor-bottom-right .maplibregl-popup-tip { + align-self: flex-end; + border-bottom: none; + border-right: none; + border-top-color: #fff; +} + +.maplibregl-popup-anchor-left .maplibregl-popup-tip { + align-self: center; + border-left: none; + border-right-color: #fff; +} + +.maplibregl-popup-anchor-right .maplibregl-popup-tip { + align-self: center; + border-right: none; + border-left-color: #fff; +} + +.maplibregl-popup-close-button { + position: absolute; + right: 0; + top: 0; + border: 0; + border-radius: 0 3px 0 0; + cursor: pointer; + background-color: transparent; +} + +.maplibregl-popup-close-button:hover { + background-color: rgb(0 0 0 / 5%); +} + +.maplibregl-popup-content { + position: relative; + background: #fff; + border-radius: 3px; + box-shadow: 0 1px 2px rgb(0 0 0 / 10%); + padding: 15px 10px; + pointer-events: auto; +} + +.maplibregl-popup-anchor-top-left .maplibregl-popup-content { + border-top-left-radius: 0; +} + +.maplibregl-popup-anchor-top-right .maplibregl-popup-content { + border-top-right-radius: 0; +} + +.maplibregl-popup-anchor-bottom-left .maplibregl-popup-content { + border-bottom-left-radius: 0; +} + +.maplibregl-popup-anchor-bottom-right .maplibregl-popup-content { + border-bottom-right-radius: 0; +} + +.maplibregl-popup-track-pointer { + display: none; +} + +.maplibregl-popup-track-pointer * { + pointer-events: none; + user-select: none; +} + +.maplibregl-map:hover .maplibregl-popup-track-pointer { + display: flex; +} + +.maplibregl-map:active .maplibregl-popup-track-pointer { + display: none; +} + +.maplibregl-marker { + position: absolute; + top: 0; + left: 0; + will-change: transform; +} + +.maplibregl-user-location-dot { + background-color: #1da1f2; + width: 15px; + height: 15px; + border-radius: 50%; +} + +.maplibregl-user-location-dot::before { + background-color: #1da1f2; + content: ""; + width: 15px; + height: 15px; + border-radius: 50%; + position: absolute; + animation: maplibregl-user-location-dot-pulse 2s infinite; +} + +.maplibregl-user-location-dot::after { + border-radius: 50%; + border: 2px solid #fff; + content: ""; + height: 19px; + left: -2px; + position: absolute; + top: -2px; + width: 19px; + box-sizing: border-box; + box-shadow: 0 0 3px rgb(0 0 0 / 35%); +} + +@keyframes maplibregl-user-location-dot-pulse { + 0% { transform: scale(1); opacity: 1; } + 70% { transform: scale(3); opacity: 0; } + 100% { transform: scale(1); opacity: 0; } +} + +.maplibregl-user-location-dot-stale { + background-color: #aaa; +} + +.maplibregl-user-location-dot-stale::after { + display: none; +} + +.maplibregl-user-location-accuracy-circle { + background-color: #1da1f233; + width: 1px; + height: 1px; + border-radius: 100%; +} + +.maplibregl-crosshair, +.maplibregl-crosshair .maplibregl-interactive, +.maplibregl-crosshair .maplibregl-interactive:active { + cursor: crosshair; +} + +.maplibregl-boxzoom { + position: absolute; + top: 0; + left: 0; + width: 0; + height: 0; + background: #fff; + border: 2px dotted #202020; + opacity: 0.5; +} + +.maplibregl-cooperative-gesture-screen { + background: rgba(0 0 0 / 40%); + position: absolute; + inset: 0; + display: flex; + justify-content: center; + align-items: center; + color: white; + padding: 1rem; + font-size: 1.4em; + line-height: 1.2; + opacity: 0; + pointer-events: none; + transition: opacity 1s ease 1s; + z-index: 99999; +} + +.maplibregl-cooperative-gesture-screen.maplibregl-show { + opacity: 1; + transition: opacity 0.05s; +} + +.maplibregl-cooperative-gesture-screen .maplibregl-mobile-message { + display: none; +} + +@media (hover: none), (width <= 480px) { + .maplibregl-cooperative-gesture-screen .maplibregl-desktop-message { + display: none; + } + + .maplibregl-cooperative-gesture-screen .maplibregl-mobile-message { + display: block; + } +} + +.maplibregl-pseudo-fullscreen { + position: fixed !important; + width: 100% !important; + height: 100% !important; + top: 0 !important; + left: 0 !important; + z-index: 99999; +} diff --git a/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-attrib.svg b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-attrib.svg new file mode 100644 index 00000000..f96152c9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-attrib.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-compass.svg b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-compass.svg new file mode 100644 index 00000000..8df846a3 --- /dev/null +++ b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-compass.svg @@ -0,0 +1,4 @@ + + + + \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-fullscreen.svg b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-fullscreen.svg new file mode 100644 index 00000000..724408d4 --- /dev/null +++ b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-fullscreen.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-geolocate.svg b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-geolocate.svg new file mode 100644 index 00000000..4b298666 --- /dev/null +++ b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-geolocate.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-logo.svg b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-logo.svg new file mode 100644 index 00000000..419d757f --- /dev/null +++ b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-logo.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-shrink.svg b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-shrink.svg new file mode 100644 index 00000000..fe6df096 --- /dev/null +++ b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-shrink.svg @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-terrain.svg b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-terrain.svg new file mode 100644 index 00000000..af8fefe7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-terrain.svg @@ -0,0 +1,7 @@ + + + + + + + diff --git a/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-zoom-in.svg b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-zoom-in.svg new file mode 100644 index 00000000..98fb23fa --- /dev/null +++ b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-zoom-in.svg @@ -0,0 +1,3 @@ + + + diff --git a/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-zoom-out.svg b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-zoom-out.svg new file mode 100644 index 00000000..4af24dea --- /dev/null +++ b/web/libraries/maplibre-gl/src/css/svg/maplibregl-ctrl-zoom-out.svg @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/data/array_types.g.ts b/web/libraries/maplibre-gl/src/data/array_types.g.ts new file mode 100644 index 00000000..d71623b9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/array_types.g.ts @@ -0,0 +1,1124 @@ +// This file is generated. Edit build/generate-struct-arrays.ts, then run `npm run codegen`. + +import {Struct, StructArray} from '../util/struct_array'; +import {register} from '../util/web_worker_transfer'; +import Point from '@mapbox/point-geometry'; + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[2] + * + */ +class StructArrayLayout2i4 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1); + } + + public emplace(i: number, v0: number, v1: number) { + const o2 = i * 2; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + return i; + } +} + +StructArrayLayout2i4.prototype.bytesPerElement = 4; +register('StructArrayLayout2i4', StructArrayLayout2i4); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[3] + * + */ +class StructArrayLayout3i6 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2); + } + + public emplace(i: number, v0: number, v1: number, v2: number) { + const o2 = i * 3; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + return i; + } +} + +StructArrayLayout3i6.prototype.bytesPerElement = 6; +register('StructArrayLayout3i6', StructArrayLayout3i6); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[4] + * + */ +class StructArrayLayout4i8 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number, v3: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3); + } + + public emplace(i: number, v0: number, v1: number, v2: number, v3: number) { + const o2 = i * 4; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + this.int16[o2 + 3] = v3; + return i; + } +} + +StructArrayLayout4i8.prototype.bytesPerElement = 8; +register('StructArrayLayout4i8', StructArrayLayout4i8); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[2] + * [4]: Int16[4] + * + */ +class StructArrayLayout2i4i12 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5); + } + + public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) { + const o2 = i * 6; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + this.int16[o2 + 3] = v3; + this.int16[o2 + 4] = v4; + this.int16[o2 + 5] = v5; + return i; + } +} + +StructArrayLayout2i4i12.prototype.bytesPerElement = 12; +register('StructArrayLayout2i4i12', StructArrayLayout2i4i12); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[2] + * [4]: Uint8[4] + * + */ +class StructArrayLayout2i4ub8 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5); + } + + public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) { + const o2 = i * 4; + const o1 = i * 8; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.uint8[o1 + 4] = v2; + this.uint8[o1 + 5] = v3; + this.uint8[o1 + 6] = v4; + this.uint8[o1 + 7] = v5; + return i; + } +} + +StructArrayLayout2i4ub8.prototype.bytesPerElement = 8; +register('StructArrayLayout2i4ub8', StructArrayLayout2i4ub8); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Float32[2] + * + */ +class StructArrayLayout2f8 extends StructArray { + uint8: Uint8Array; + float32: Float32Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1); + } + + public emplace(i: number, v0: number, v1: number) { + const o4 = i * 2; + this.float32[o4 + 0] = v0; + this.float32[o4 + 1] = v1; + return i; + } +} + +StructArrayLayout2f8.prototype.bytesPerElement = 8; +register('StructArrayLayout2f8', StructArrayLayout2f8); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint16[10] + * + */ +class StructArrayLayout10ui20 extends StructArray { + uint8: Uint8Array; + uint16: Uint16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9); + } + + public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number) { + const o2 = i * 10; + this.uint16[o2 + 0] = v0; + this.uint16[o2 + 1] = v1; + this.uint16[o2 + 2] = v2; + this.uint16[o2 + 3] = v3; + this.uint16[o2 + 4] = v4; + this.uint16[o2 + 5] = v5; + this.uint16[o2 + 6] = v6; + this.uint16[o2 + 7] = v7; + this.uint16[o2 + 8] = v8; + this.uint16[o2 + 9] = v9; + return i; + } +} + +StructArrayLayout10ui20.prototype.bytesPerElement = 20; +register('StructArrayLayout10ui20', StructArrayLayout10ui20); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[4] + * [8]: Uint16[4] + * [16]: Int16[4] + * + */ +class StructArrayLayout4i4ui4i24 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + uint16: Uint16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11); + } + + public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number) { + const o2 = i * 12; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + this.int16[o2 + 3] = v3; + this.uint16[o2 + 4] = v4; + this.uint16[o2 + 5] = v5; + this.uint16[o2 + 6] = v6; + this.uint16[o2 + 7] = v7; + this.int16[o2 + 8] = v8; + this.int16[o2 + 9] = v9; + this.int16[o2 + 10] = v10; + this.int16[o2 + 11] = v11; + return i; + } +} + +StructArrayLayout4i4ui4i24.prototype.bytesPerElement = 24; +register('StructArrayLayout4i4ui4i24', StructArrayLayout4i4ui4i24); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Float32[3] + * + */ +class StructArrayLayout3f12 extends StructArray { + uint8: Uint8Array; + float32: Float32Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2); + } + + public emplace(i: number, v0: number, v1: number, v2: number) { + const o4 = i * 3; + this.float32[o4 + 0] = v0; + this.float32[o4 + 1] = v1; + this.float32[o4 + 2] = v2; + return i; + } +} + +StructArrayLayout3f12.prototype.bytesPerElement = 12; +register('StructArrayLayout3f12', StructArrayLayout3f12); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint32[1] + * + */ +class StructArrayLayout1ul4 extends StructArray { + uint8: Uint8Array; + uint32: Uint32Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint32 = new Uint32Array(this.arrayBuffer); + } + + public emplaceBack(v0: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0); + } + + public emplace(i: number, v0: number) { + const o4 = i * 1; + this.uint32[o4 + 0] = v0; + return i; + } +} + +StructArrayLayout1ul4.prototype.bytesPerElement = 4; +register('StructArrayLayout1ul4', StructArrayLayout1ul4); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[6] + * [12]: Uint32[1] + * [16]: Uint16[2] + * + */ +class StructArrayLayout6i1ul2ui20 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + uint32: Uint32Array; + uint16: Uint16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + this.uint32 = new Uint32Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8); + } + + public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number) { + const o2 = i * 10; + const o4 = i * 5; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + this.int16[o2 + 3] = v3; + this.int16[o2 + 4] = v4; + this.int16[o2 + 5] = v5; + this.uint32[o4 + 3] = v6; + this.uint16[o2 + 8] = v7; + this.uint16[o2 + 9] = v8; + return i; + } +} + +StructArrayLayout6i1ul2ui20.prototype.bytesPerElement = 20; +register('StructArrayLayout6i1ul2ui20', StructArrayLayout6i1ul2ui20); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[2] + * [4]: Int16[2] + * [8]: Int16[2] + * + */ +class StructArrayLayout2i2i2i12 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5); + } + + public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number) { + const o2 = i * 6; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + this.int16[o2 + 3] = v3; + this.int16[o2 + 4] = v4; + this.int16[o2 + 5] = v5; + return i; + } +} + +StructArrayLayout2i2i2i12.prototype.bytesPerElement = 12; +register('StructArrayLayout2i2i2i12', StructArrayLayout2i2i2i12); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Float32[2] + * [8]: Float32[1] + * [12]: Int16[2] + * + */ +class StructArrayLayout2f1f2i16 extends StructArray { + uint8: Uint8Array; + float32: Float32Array; + int16: Int16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4); + } + + public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number) { + const o4 = i * 4; + const o2 = i * 8; + this.float32[o4 + 0] = v0; + this.float32[o4 + 1] = v1; + this.float32[o4 + 2] = v2; + this.int16[o2 + 6] = v3; + this.int16[o2 + 7] = v4; + return i; + } +} + +StructArrayLayout2f1f2i16.prototype.bytesPerElement = 16; +register('StructArrayLayout2f1f2i16', StructArrayLayout2f1f2i16); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint8[2] + * [4]: Float32[2] + * + */ +class StructArrayLayout2ub2f12 extends StructArray { + uint8: Uint8Array; + float32: Float32Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number, v3: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3); + } + + public emplace(i: number, v0: number, v1: number, v2: number, v3: number) { + const o1 = i * 12; + const o4 = i * 3; + this.uint8[o1 + 0] = v0; + this.uint8[o1 + 1] = v1; + this.float32[o4 + 1] = v2; + this.float32[o4 + 2] = v3; + return i; + } +} + +StructArrayLayout2ub2f12.prototype.bytesPerElement = 12; +register('StructArrayLayout2ub2f12', StructArrayLayout2ub2f12); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint16[3] + * + */ +class StructArrayLayout3ui6 extends StructArray { + uint8: Uint8Array; + uint16: Uint16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2); + } + + public emplace(i: number, v0: number, v1: number, v2: number) { + const o2 = i * 3; + this.uint16[o2 + 0] = v0; + this.uint16[o2 + 1] = v1; + this.uint16[o2 + 2] = v2; + return i; + } +} + +StructArrayLayout3ui6.prototype.bytesPerElement = 6; +register('StructArrayLayout3ui6', StructArrayLayout3ui6); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[2] + * [4]: Uint16[2] + * [8]: Uint32[3] + * [20]: Uint16[3] + * [28]: Float32[2] + * [36]: Uint8[3] + * [40]: Uint32[1] + * [44]: Int16[1] + * + */ +class StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + uint16: Uint16Array; + uint32: Uint32Array; + float32: Float32Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + this.uint32 = new Uint32Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16); + } + + public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number) { + const o2 = i * 24; + const o4 = i * 12; + const o1 = i * 48; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.uint16[o2 + 2] = v2; + this.uint16[o2 + 3] = v3; + this.uint32[o4 + 2] = v4; + this.uint32[o4 + 3] = v5; + this.uint32[o4 + 4] = v6; + this.uint16[o2 + 10] = v7; + this.uint16[o2 + 11] = v8; + this.uint16[o2 + 12] = v9; + this.float32[o4 + 7] = v10; + this.float32[o4 + 8] = v11; + this.uint8[o1 + 36] = v12; + this.uint8[o1 + 37] = v13; + this.uint8[o1 + 38] = v14; + this.uint32[o4 + 10] = v15; + this.int16[o2 + 22] = v16; + return i; + } +} + +StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48.prototype.bytesPerElement = 48; +register('StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48', StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Int16[8] + * [16]: Uint16[15] + * [48]: Uint32[1] + * [52]: Float32[2] + * [60]: Uint16[2] + * + */ +class StructArrayLayout8i15ui1ul2f2ui64 extends StructArray { + uint8: Uint8Array; + int16: Int16Array; + uint16: Uint16Array; + uint32: Uint32Array; + float32: Float32Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.int16 = new Int16Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + this.uint32 = new Uint32Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number, v17: number, v18: number, v19: number, v20: number, v21: number, v22: number, v23: number, v24: number, v25: number, v26: number, v27: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27); + } + + public emplace(i: number, v0: number, v1: number, v2: number, v3: number, v4: number, v5: number, v6: number, v7: number, v8: number, v9: number, v10: number, v11: number, v12: number, v13: number, v14: number, v15: number, v16: number, v17: number, v18: number, v19: number, v20: number, v21: number, v22: number, v23: number, v24: number, v25: number, v26: number, v27: number) { + const o2 = i * 32; + const o4 = i * 16; + this.int16[o2 + 0] = v0; + this.int16[o2 + 1] = v1; + this.int16[o2 + 2] = v2; + this.int16[o2 + 3] = v3; + this.int16[o2 + 4] = v4; + this.int16[o2 + 5] = v5; + this.int16[o2 + 6] = v6; + this.int16[o2 + 7] = v7; + this.uint16[o2 + 8] = v8; + this.uint16[o2 + 9] = v9; + this.uint16[o2 + 10] = v10; + this.uint16[o2 + 11] = v11; + this.uint16[o2 + 12] = v12; + this.uint16[o2 + 13] = v13; + this.uint16[o2 + 14] = v14; + this.uint16[o2 + 15] = v15; + this.uint16[o2 + 16] = v16; + this.uint16[o2 + 17] = v17; + this.uint16[o2 + 18] = v18; + this.uint16[o2 + 19] = v19; + this.uint16[o2 + 20] = v20; + this.uint16[o2 + 21] = v21; + this.uint16[o2 + 22] = v22; + this.uint32[o4 + 12] = v23; + this.float32[o4 + 13] = v24; + this.float32[o4 + 14] = v25; + this.uint16[o2 + 30] = v26; + this.uint16[o2 + 31] = v27; + return i; + } +} + +StructArrayLayout8i15ui1ul2f2ui64.prototype.bytesPerElement = 64; +register('StructArrayLayout8i15ui1ul2f2ui64', StructArrayLayout8i15ui1ul2f2ui64); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Float32[1] + * + */ +class StructArrayLayout1f4 extends StructArray { + uint8: Uint8Array; + float32: Float32Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + + public emplaceBack(v0: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0); + } + + public emplace(i: number, v0: number) { + const o4 = i * 1; + this.float32[o4 + 0] = v0; + return i; + } +} + +StructArrayLayout1f4.prototype.bytesPerElement = 4; +register('StructArrayLayout1f4', StructArrayLayout1f4); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint16[1] + * [4]: Float32[2] + * + */ +class StructArrayLayout1ui2f12 extends StructArray { + uint8: Uint8Array; + uint16: Uint16Array; + float32: Float32Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2); + } + + public emplace(i: number, v0: number, v1: number, v2: number) { + const o2 = i * 6; + const o4 = i * 3; + this.uint16[o2 + 0] = v0; + this.float32[o4 + 1] = v1; + this.float32[o4 + 2] = v2; + return i; + } +} + +StructArrayLayout1ui2f12.prototype.bytesPerElement = 12; +register('StructArrayLayout1ui2f12', StructArrayLayout1ui2f12); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint32[1] + * [4]: Uint16[2] + * + */ +class StructArrayLayout1ul2ui8 extends StructArray { + uint8: Uint8Array; + uint32: Uint32Array; + uint16: Uint16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint32 = new Uint32Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2); + } + + public emplace(i: number, v0: number, v1: number, v2: number) { + const o4 = i * 2; + const o2 = i * 4; + this.uint32[o4 + 0] = v0; + this.uint16[o2 + 2] = v1; + this.uint16[o2 + 3] = v2; + return i; + } +} + +StructArrayLayout1ul2ui8.prototype.bytesPerElement = 8; +register('StructArrayLayout1ul2ui8', StructArrayLayout1ul2ui8); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint16[2] + * + */ +class StructArrayLayout2ui4 extends StructArray { + uint8: Uint8Array; + uint16: Uint16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1); + } + + public emplace(i: number, v0: number, v1: number) { + const o2 = i * 2; + this.uint16[o2 + 0] = v0; + this.uint16[o2 + 1] = v1; + return i; + } +} + +StructArrayLayout2ui4.prototype.bytesPerElement = 4; +register('StructArrayLayout2ui4', StructArrayLayout2ui4); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Uint16[1] + * + */ +class StructArrayLayout1ui2 extends StructArray { + uint8: Uint8Array; + uint16: Uint16Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.uint16 = new Uint16Array(this.arrayBuffer); + } + + public emplaceBack(v0: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0); + } + + public emplace(i: number, v0: number) { + const o2 = i * 1; + this.uint16[o2 + 0] = v0; + return i; + } +} + +StructArrayLayout1ui2.prototype.bytesPerElement = 2; +register('StructArrayLayout1ui2', StructArrayLayout1ui2); + +/** + * @internal + * Implementation of the StructArray layout: + * [0]: Float32[4] + * + */ +class StructArrayLayout4f16 extends StructArray { + uint8: Uint8Array; + float32: Float32Array; + + _refreshViews() { + this.uint8 = new Uint8Array(this.arrayBuffer); + this.float32 = new Float32Array(this.arrayBuffer); + } + + public emplaceBack(v0: number, v1: number, v2: number, v3: number) { + const i = this.length; + this.resize(i + 1); + return this.emplace(i, v0, v1, v2, v3); + } + + public emplace(i: number, v0: number, v1: number, v2: number, v3: number) { + const o4 = i * 4; + this.float32[o4 + 0] = v0; + this.float32[o4 + 1] = v1; + this.float32[o4 + 2] = v2; + this.float32[o4 + 3] = v3; + return i; + } +} + +StructArrayLayout4f16.prototype.bytesPerElement = 16; +register('StructArrayLayout4f16', StructArrayLayout4f16); + +/** @internal */ +class CollisionBoxStruct extends Struct { + _structArray: CollisionBoxArray; + get anchorPointX() { return this._structArray.int16[this._pos2 + 0]; } + get anchorPointY() { return this._structArray.int16[this._pos2 + 1]; } + get x1() { return this._structArray.int16[this._pos2 + 2]; } + get y1() { return this._structArray.int16[this._pos2 + 3]; } + get x2() { return this._structArray.int16[this._pos2 + 4]; } + get y2() { return this._structArray.int16[this._pos2 + 5]; } + get featureIndex() { return this._structArray.uint32[this._pos4 + 3]; } + get sourceLayerIndex() { return this._structArray.uint16[this._pos2 + 8]; } + get bucketIndex() { return this._structArray.uint16[this._pos2 + 9]; } + get anchorPoint() { return new Point(this.anchorPointX, this.anchorPointY); } +} + +CollisionBoxStruct.prototype.size = 20; + +export type CollisionBox = CollisionBoxStruct; + +/** @internal */ +export class CollisionBoxArray extends StructArrayLayout6i1ul2ui20 { + /** + * Return the CollisionBoxStruct at the given location in the array. + * @param index The index of the element. + */ + get(index: number): CollisionBoxStruct { + return new CollisionBoxStruct(this, index); + } +} + +register('CollisionBoxArray', CollisionBoxArray); + +/** @internal */ +class PlacedSymbolStruct extends Struct { + _structArray: PlacedSymbolArray; + get anchorX() { return this._structArray.int16[this._pos2 + 0]; } + get anchorY() { return this._structArray.int16[this._pos2 + 1]; } + get glyphStartIndex() { return this._structArray.uint16[this._pos2 + 2]; } + get numGlyphs() { return this._structArray.uint16[this._pos2 + 3]; } + get vertexStartIndex() { return this._structArray.uint32[this._pos4 + 2]; } + get lineStartIndex() { return this._structArray.uint32[this._pos4 + 3]; } + get lineLength() { return this._structArray.uint32[this._pos4 + 4]; } + get segment() { return this._structArray.uint16[this._pos2 + 10]; } + get lowerSize() { return this._structArray.uint16[this._pos2 + 11]; } + get upperSize() { return this._structArray.uint16[this._pos2 + 12]; } + get lineOffsetX() { return this._structArray.float32[this._pos4 + 7]; } + get lineOffsetY() { return this._structArray.float32[this._pos4 + 8]; } + get writingMode() { return this._structArray.uint8[this._pos1 + 36]; } + get placedOrientation() { return this._structArray.uint8[this._pos1 + 37]; } + set placedOrientation(x: number) { this._structArray.uint8[this._pos1 + 37] = x; } + get hidden() { return this._structArray.uint8[this._pos1 + 38]; } + set hidden(x: number) { this._structArray.uint8[this._pos1 + 38] = x; } + get crossTileID() { return this._structArray.uint32[this._pos4 + 10]; } + set crossTileID(x: number) { this._structArray.uint32[this._pos4 + 10] = x; } + get associatedIconIndex() { return this._structArray.int16[this._pos2 + 22]; } +} + +PlacedSymbolStruct.prototype.size = 48; + +export type PlacedSymbol = PlacedSymbolStruct; + +/** @internal */ +export class PlacedSymbolArray extends StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48 { + /** + * Return the PlacedSymbolStruct at the given location in the array. + * @param index The index of the element. + */ + get(index: number): PlacedSymbolStruct { + return new PlacedSymbolStruct(this, index); + } +} + +register('PlacedSymbolArray', PlacedSymbolArray); + +/** @internal */ +class SymbolInstanceStruct extends Struct { + _structArray: SymbolInstanceArray; + get anchorX() { return this._structArray.int16[this._pos2 + 0]; } + get anchorY() { return this._structArray.int16[this._pos2 + 1]; } + get rightJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 2]; } + get centerJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 3]; } + get leftJustifiedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 4]; } + get verticalPlacedTextSymbolIndex() { return this._structArray.int16[this._pos2 + 5]; } + get placedIconSymbolIndex() { return this._structArray.int16[this._pos2 + 6]; } + get verticalPlacedIconSymbolIndex() { return this._structArray.int16[this._pos2 + 7]; } + get key() { return this._structArray.uint16[this._pos2 + 8]; } + get textBoxStartIndex() { return this._structArray.uint16[this._pos2 + 9]; } + get textBoxEndIndex() { return this._structArray.uint16[this._pos2 + 10]; } + get verticalTextBoxStartIndex() { return this._structArray.uint16[this._pos2 + 11]; } + get verticalTextBoxEndIndex() { return this._structArray.uint16[this._pos2 + 12]; } + get iconBoxStartIndex() { return this._structArray.uint16[this._pos2 + 13]; } + get iconBoxEndIndex() { return this._structArray.uint16[this._pos2 + 14]; } + get verticalIconBoxStartIndex() { return this._structArray.uint16[this._pos2 + 15]; } + get verticalIconBoxEndIndex() { return this._structArray.uint16[this._pos2 + 16]; } + get featureIndex() { return this._structArray.uint16[this._pos2 + 17]; } + get numHorizontalGlyphVertices() { return this._structArray.uint16[this._pos2 + 18]; } + get numVerticalGlyphVertices() { return this._structArray.uint16[this._pos2 + 19]; } + get numIconVertices() { return this._structArray.uint16[this._pos2 + 20]; } + get numVerticalIconVertices() { return this._structArray.uint16[this._pos2 + 21]; } + get useRuntimeCollisionCircles() { return this._structArray.uint16[this._pos2 + 22]; } + get crossTileID() { return this._structArray.uint32[this._pos4 + 12]; } + set crossTileID(x: number) { this._structArray.uint32[this._pos4 + 12] = x; } + get textBoxScale() { return this._structArray.float32[this._pos4 + 13]; } + get collisionCircleDiameter() { return this._structArray.float32[this._pos4 + 14]; } + get textAnchorOffsetStartIndex() { return this._structArray.uint16[this._pos2 + 30]; } + get textAnchorOffsetEndIndex() { return this._structArray.uint16[this._pos2 + 31]; } +} + +SymbolInstanceStruct.prototype.size = 64; + +export type SymbolInstance = SymbolInstanceStruct; + +/** @internal */ +export class SymbolInstanceArray extends StructArrayLayout8i15ui1ul2f2ui64 { + /** + * Return the SymbolInstanceStruct at the given location in the array. + * @param index The index of the element. + */ + get(index: number): SymbolInstanceStruct { + return new SymbolInstanceStruct(this, index); + } +} + +register('SymbolInstanceArray', SymbolInstanceArray); + +/** @internal */ +export class GlyphOffsetArray extends StructArrayLayout1f4 { + getoffsetX(index: number) { return this.float32[index * 1 + 0]; } +} + +register('GlyphOffsetArray', GlyphOffsetArray); + +/** @internal */ +export class SymbolLineVertexArray extends StructArrayLayout3i6 { + getx(index: number) { return this.int16[index * 3 + 0]; } + gety(index: number) { return this.int16[index * 3 + 1]; } + gettileUnitDistanceFromAnchor(index: number) { return this.int16[index * 3 + 2]; } +} + +register('SymbolLineVertexArray', SymbolLineVertexArray); + +/** @internal */ +class TextAnchorOffsetStruct extends Struct { + _structArray: TextAnchorOffsetArray; + get textAnchor() { return this._structArray.uint16[this._pos2 + 0]; } + get textOffset0() { return this._structArray.float32[this._pos4 + 1]; } + get textOffset1() { return this._structArray.float32[this._pos4 + 2]; } +} + +TextAnchorOffsetStruct.prototype.size = 12; + +export type TextAnchorOffset = TextAnchorOffsetStruct; + +/** @internal */ +export class TextAnchorOffsetArray extends StructArrayLayout1ui2f12 { + /** + * Return the TextAnchorOffsetStruct at the given location in the array. + * @param index The index of the element. + */ + get(index: number): TextAnchorOffsetStruct { + return new TextAnchorOffsetStruct(this, index); + } +} + +register('TextAnchorOffsetArray', TextAnchorOffsetArray); + +/** @internal */ +class FeatureIndexStruct extends Struct { + _structArray: FeatureIndexArray; + get featureIndex() { return this._structArray.uint32[this._pos4 + 0]; } + get sourceLayerIndex() { return this._structArray.uint16[this._pos2 + 2]; } + get bucketIndex() { return this._structArray.uint16[this._pos2 + 3]; } +} + +FeatureIndexStruct.prototype.size = 8; + +export type FeatureIndex = FeatureIndexStruct; + +/** @internal */ +export class FeatureIndexArray extends StructArrayLayout1ul2ui8 { + /** + * Return the FeatureIndexStruct at the given location in the array. + * @param index The index of the element. + */ + get(index: number): FeatureIndexStruct { + return new FeatureIndexStruct(this, index); + } +} + +register('FeatureIndexArray', FeatureIndexArray); + +export class PosArray extends StructArrayLayout2i4 {} +export class Pos3dArray extends StructArrayLayout3i6 {} +export class RasterBoundsArray extends StructArrayLayout4i8 {} +export class CircleLayoutArray extends StructArrayLayout2i4 {} +export class FillLayoutArray extends StructArrayLayout2i4 {} +export class FillExtrusionLayoutArray extends StructArrayLayout2i4i12 {} +export class HeatmapLayoutArray extends StructArrayLayout2i4 {} +export class LineLayoutArray extends StructArrayLayout2i4ub8 {} +export class LineExtLayoutArray extends StructArrayLayout2f8 {} +export class PatternLayoutArray extends StructArrayLayout10ui20 {} +export class SymbolLayoutArray extends StructArrayLayout4i4ui4i24 {} +export class SymbolDynamicLayoutArray extends StructArrayLayout3f12 {} +export class SymbolOpacityArray extends StructArrayLayout1ul4 {} +export class CollisionBoxLayoutArray extends StructArrayLayout2i2i2i12 {} +export class CollisionCircleLayoutArray extends StructArrayLayout2f1f2i16 {} +export class CollisionVertexArray extends StructArrayLayout2ub2f12 {} +export class QuadTriangleArray extends StructArrayLayout3ui6 {} +export class TriangleIndexArray extends StructArrayLayout3ui6 {} +export class LineIndexArray extends StructArrayLayout2ui4 {} +export class LineStripIndexArray extends StructArrayLayout1ui2 {} +export { + StructArrayLayout2i4, + StructArrayLayout3i6, + StructArrayLayout4i8, + StructArrayLayout2i4i12, + StructArrayLayout2i4ub8, + StructArrayLayout2f8, + StructArrayLayout10ui20, + StructArrayLayout4i4ui4i24, + StructArrayLayout3f12, + StructArrayLayout1ul4, + StructArrayLayout6i1ul2ui20, + StructArrayLayout2i2i2i12, + StructArrayLayout2f1f2i16, + StructArrayLayout2ub2f12, + StructArrayLayout3ui6, + StructArrayLayout2i2ui3ul3ui2f3ub1ul1i48, + StructArrayLayout8i15ui1ul2f2ui64, + StructArrayLayout1f4, + StructArrayLayout1ui2f12, + StructArrayLayout1ul2ui8, + StructArrayLayout2ui4, + StructArrayLayout1ui2, + StructArrayLayout4f16 +}; diff --git a/web/libraries/maplibre-gl/src/data/bucket.ts b/web/libraries/maplibre-gl/src/data/bucket.ts new file mode 100644 index 00000000..0435acc5 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket.ts @@ -0,0 +1,123 @@ +import type {CollisionBoxArray} from './array_types.g'; +import type {Style} from '../style/style'; +import type {TypedStyleLayer} from '../style/style_layer/typed_style_layer'; +import type {FeatureIndex} from './feature_index'; +import type {Context} from '../gl/context'; +import type {FeatureStates} from '../source/source_state'; +import type {ImagePosition} from '../render/image_atlas'; +import type {CanonicalTileID} from '../source/tile_id'; +import type {VectorTileFeature, VectorTileLayer} from '@mapbox/vector-tile'; +import Point from '@mapbox/point-geometry'; + +export type BucketParameters = { + index: number; + layers: Array; + zoom: number; + pixelRatio: number; + overscaling: number; + collisionBoxArray: CollisionBoxArray; + sourceLayerIndex: number; + sourceID: string; +}; + +export type PopulateParameters = { + featureIndex: FeatureIndex; + iconDependencies: {}; + patternDependencies: {}; + glyphDependencies: {}; + availableImages: Array; +}; + +export type IndexedFeature = { + feature: VectorTileFeature; + id: number | string; + index: number; + sourceLayerIndex: number; +}; + +export type BucketFeature = { + index: number; + sourceLayerIndex: number; + geometry: Array>; + properties: any; + type: 0 | 1 | 2 | 3; + id?: any; + readonly patterns: { + [_: string]: { + 'min': string; + 'mid': string; + 'max': string; + }; + }; + sortKey?: number; +}; + +/** + * The `Bucket` interface is the single point of knowledge about turning vector + * tiles into WebGL buffers. + * + * `Bucket` is an abstract interface. An implementation exists for each style layer type. + * Create a bucket via the `StyleLayer#createBucket` method. + * + * The concrete bucket types, using layout options from the style layer, + * transform feature geometries into vertex and index data for use by the + * vertex shader. They also (via `ProgramConfiguration`) use feature + * properties and the zoom level to populate the attributes needed for + * data-driven styling. + * + * Buckets are designed to be built on a worker thread and then serialized and + * transferred back to the main thread for rendering. On the worker side, a + * bucket's vertex, index, and attribute data is stored in `bucket.arrays: ArrayGroup`. + * When a bucket's data is serialized and sent back to the main thread, + * is gets deserialized (using `new Bucket(serializedBucketData)`, with + * the array data now stored in `bucket.buffers: BufferGroup`. BufferGroups + * hold the same data as ArrayGroups, but are tuned for consumption by WebGL. + */ +export interface Bucket { + layerIds: Array; + hasPattern: boolean; + readonly layers: Array; + readonly stateDependentLayers: Array; + readonly stateDependentLayerIds: Array; + populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID): void; + update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}): void; + isEmpty(): boolean; + upload(context: Context): void; + uploadPending(): boolean; + /** + * Release the WebGL resources associated with the buffers. Note that because + * buckets are shared between layers having the same layout properties, they + * must be destroyed in groups (all buckets for a tile, or all symbol buckets). + */ + destroy(): void; +} + +export function deserialize(input: Array, style: Style): {[_: string]: Bucket} { + const output = {}; + + // Guard against the case where the map's style has been set to null while + // this bucket has been parsing. + if (!style) return output; + + for (const bucket of input) { + const layers = bucket.layerIds + .map((id) => style.getLayer(id)) + .filter(Boolean); + + if (layers.length === 0) { + continue; + } + + // look up StyleLayer objects from layer ids (since we don't + // want to waste time serializing/copying them from the worker) + (bucket as any).layers = layers; + if (bucket.stateDependentLayerIds) { + (bucket as any).stateDependentLayers = bucket.stateDependentLayerIds.map((lId) => layers.filter((l) => l.id === lId)[0]); + } + for (const layer of layers) { + output[layer.id] = bucket; + } + } + + return output; +} diff --git a/web/libraries/maplibre-gl/src/data/bucket/circle_attributes.ts b/web/libraries/maplibre-gl/src/data/bucket/circle_attributes.ts new file mode 100644 index 00000000..ed9ebcef --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/circle_attributes.ts @@ -0,0 +1,8 @@ +import {createLayout} from '../../util/struct_array'; + +const layout = createLayout([ + {name: 'a_pos', components: 2, type: 'Int16'} +], 4); + +export default layout; +export const {members, size, alignment} = layout; diff --git a/web/libraries/maplibre-gl/src/data/bucket/circle_bucket.ts b/web/libraries/maplibre-gl/src/data/bucket/circle_bucket.ts new file mode 100644 index 00000000..0fcbda0a --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/circle_bucket.ts @@ -0,0 +1,197 @@ +import {CircleLayoutArray} from '../array_types.g'; + +import {members as layoutAttributes} from './circle_attributes'; +import {SegmentVector} from '../segment'; +import {ProgramConfigurationSet} from '../program_configuration'; +import {TriangleIndexArray} from '../index_array_type'; +import {loadGeometry} from '../load_geometry'; +import {toEvaluationFeature} from '../evaluation_feature'; +import {EXTENT} from '../extent'; +import {register} from '../../util/web_worker_transfer'; +import {EvaluationParameters} from '../../style/evaluation_parameters'; + +import type {CanonicalTileID} from '../../source/tile_id'; +import type { + Bucket, + BucketParameters, + BucketFeature, + IndexedFeature, + PopulateParameters +} from '../bucket'; +import type {CircleStyleLayer} from '../../style/style_layer/circle_style_layer'; +import type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer'; +import type {Context} from '../../gl/context'; +import type {IndexBuffer} from '../../gl/index_buffer'; +import type {VertexBuffer} from '../../gl/vertex_buffer'; +import type Point from '@mapbox/point-geometry'; +import type {FeatureStates} from '../../source/source_state'; +import type {ImagePosition} from '../../render/image_atlas'; +import type {VectorTileLayer} from '@mapbox/vector-tile'; + +function addCircleVertex(layoutVertexArray, x, y, extrudeX, extrudeY) { + layoutVertexArray.emplaceBack( + (x * 2) + ((extrudeX + 1) / 2), + (y * 2) + ((extrudeY + 1) / 2)); +} + +/** + * @internal + * Circles are represented by two triangles. + * + * Each corner has a pos that is the center of the circle and an extrusion + * vector that is where it points. + */ +export class CircleBucket implements Bucket { + index: number; + zoom: number; + overscaling: number; + layerIds: Array; + layers: Array; + stateDependentLayers: Array; + stateDependentLayerIds: Array; + + layoutVertexArray: CircleLayoutArray; + layoutVertexBuffer: VertexBuffer; + + indexArray: TriangleIndexArray; + indexBuffer: IndexBuffer; + + hasPattern: boolean; + programConfigurations: ProgramConfigurationSet; + segments: SegmentVector; + uploaded: boolean; + + constructor(options: BucketParameters) { + this.zoom = options.zoom; + this.overscaling = options.overscaling; + this.layers = options.layers; + this.layerIds = this.layers.map(layer => layer.id); + this.index = options.index; + this.hasPattern = false; + + this.layoutVertexArray = new CircleLayoutArray(); + this.indexArray = new TriangleIndexArray(); + this.segments = new SegmentVector(); + this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom); + this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id); + } + + populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) { + const styleLayer = this.layers[0]; + const bucketFeatures: BucketFeature[] = []; + let circleSortKey = null; + let sortFeaturesByKey = false; + + // Heatmap layers are handled in this bucket and have no evaluated properties, so we check our access + if (styleLayer.type === 'circle') { + circleSortKey = (styleLayer as CircleStyleLayer).layout.get('circle-sort-key'); + sortFeaturesByKey = !circleSortKey.isConstant(); + } + + for (const {feature, id, index, sourceLayerIndex} of features) { + const needGeometry = this.layers[0]._featureFilter.needGeometry; + const evaluationFeature = toEvaluationFeature(feature, needGeometry); + + if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue; + + const sortKey = sortFeaturesByKey ? + circleSortKey.evaluate(evaluationFeature, {}, canonical) : + undefined; + + const bucketFeature: BucketFeature = { + id, + properties: feature.properties, + type: feature.type, + sourceLayerIndex, + index, + geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature), + patterns: {}, + sortKey + }; + + bucketFeatures.push(bucketFeature); + + } + + if (sortFeaturesByKey) { + bucketFeatures.sort((a, b) => a.sortKey - b.sortKey); + } + + for (const bucketFeature of bucketFeatures) { + const {geometry, index, sourceLayerIndex} = bucketFeature; + const feature = features[index].feature; + + this.addFeature(bucketFeature, geometry, index, canonical); + options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index); + } + } + + update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) { + if (!this.stateDependentLayers.length) return; + this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions); + } + + isEmpty() { + return this.layoutVertexArray.length === 0; + } + + uploadPending() { + return !this.uploaded || this.programConfigurations.needsUpload; + } + + upload(context: Context) { + if (!this.uploaded) { + this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes); + this.indexBuffer = context.createIndexBuffer(this.indexArray); + } + this.programConfigurations.upload(context); + this.uploaded = true; + } + + destroy() { + if (!this.layoutVertexBuffer) return; + this.layoutVertexBuffer.destroy(); + this.indexBuffer.destroy(); + this.programConfigurations.destroy(); + this.segments.destroy(); + } + + addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID) { + for (const ring of geometry) { + for (const point of ring) { + const x = point.x; + const y = point.y; + + // Do not include points that are outside the tile boundaries. + if (x < 0 || x >= EXTENT || y < 0 || y >= EXTENT) continue; + + // this geometry will be of the Point type, and we'll derive + // two triangles from it. + // + // ┌─────────┐ + // │ 3 2 │ + // │ │ + // │ 0 1 │ + // └─────────┘ + + const segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray, feature.sortKey); + const index = segment.vertexLength; + + addCircleVertex(this.layoutVertexArray, x, y, -1, -1); + addCircleVertex(this.layoutVertexArray, x, y, 1, -1); + addCircleVertex(this.layoutVertexArray, x, y, 1, 1); + addCircleVertex(this.layoutVertexArray, x, y, -1, 1); + + this.indexArray.emplaceBack(index, index + 1, index + 2); + this.indexArray.emplaceBack(index, index + 3, index + 2); + + segment.vertexLength += 4; + segment.primitiveLength += 2; + } + } + + this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, {}, canonical); + } +} + +register('CircleBucket', CircleBucket, {omit: ['layers']}); diff --git a/web/libraries/maplibre-gl/src/data/bucket/fill_attributes.ts b/web/libraries/maplibre-gl/src/data/bucket/fill_attributes.ts new file mode 100644 index 00000000..ed9ebcef --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/fill_attributes.ts @@ -0,0 +1,8 @@ +import {createLayout} from '../../util/struct_array'; + +const layout = createLayout([ + {name: 'a_pos', components: 2, type: 'Int16'} +], 4); + +export default layout; +export const {members, size, alignment} = layout; diff --git a/web/libraries/maplibre-gl/src/data/bucket/fill_bucket.test.ts b/web/libraries/maplibre-gl/src/data/bucket/fill_bucket.test.ts new file mode 100644 index 00000000..06bd6056 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/fill_bucket.test.ts @@ -0,0 +1,95 @@ + +import fs from 'fs'; +import path from 'path'; +import Protobuf from 'pbf'; +import {VectorTile} from '@mapbox/vector-tile'; +import Point from '@mapbox/point-geometry'; +import {SegmentVector} from '../segment'; +import {FillBucket} from './fill_bucket'; +import {FillStyleLayer} from '../../style/style_layer/fill_style_layer'; +import {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {EvaluationParameters} from '../../style/evaluation_parameters'; +import {ZoomHistory} from '../../style/zoom_history'; +import {BucketFeature, BucketParameters} from '../bucket'; + +// Load a fill feature from fixture tile. +const vt = new VectorTile(new Protobuf(fs.readFileSync(path.resolve(__dirname, '../../../test/unit/assets/mbsv5-6-18-23.vector.pbf')))); +const feature = vt.layers.water.feature(0); + +function createPolygon(numPoints) { + const points = []; + for (let i = 0; i < numPoints; i++) { + points.push(new Point(2048 + 256 * Math.cos(i / numPoints * 2 * Math.PI), 2048 + 256 * Math.sin(i / numPoints * 2 * Math.PI))); + } + return points; +} + +test('FillBucket', () => { + expect(() => { + const layer = new FillStyleLayer({id: 'test', type: 'fill', layout: {}} as LayerSpecification); + layer.recalculate({zoom: 0, zoomHistory: {} as ZoomHistory} as EvaluationParameters, undefined); + + const bucket = new FillBucket({layers: [layer]} as BucketParameters); + + bucket.addFeature({} as BucketFeature, [[ + new Point(0, 0), + new Point(10, 10) + ]], undefined, undefined, undefined); + + bucket.addFeature({} as BucketFeature, [[ + new Point(0, 0), + new Point(10, 10), + new Point(10, 20) + ]], undefined, undefined, undefined); + + bucket.addFeature(feature as any, feature.loadGeometry(), undefined, undefined, undefined); + }).not.toThrow(); +}); + +test('FillBucket segmentation', () => { + // Stub MAX_VERTEX_ARRAY_LENGTH so we can test features + // breaking across array groups without tests taking a _long_ time. + Object.defineProperty(SegmentVector, 'MAX_VERTEX_ARRAY_LENGTH', {value: 256}); + + const layer = new FillStyleLayer({ + id: 'test', + type: 'fill', + layout: {}, + source: 'source', + paint: { + 'fill-color': ['to-color', ['get', 'foo'], '#000'] + } + } as LayerSpecification); + layer.recalculate({zoom: 0, zoomHistory: {} as ZoomHistory} as EvaluationParameters, undefined); + + const bucket = new FillBucket({layers: [layer]} as BucketParameters); + + // first add an initial, small feature to make sure the next one starts at + // a non-zero offset + bucket.addFeature({} as BucketFeature, [createPolygon(10)], undefined, undefined, undefined); + + // add a feature that will break across the group boundary + bucket.addFeature({} as BucketFeature, [ + createPolygon(128), + createPolygon(128) + ], undefined, undefined, undefined); + + // Each polygon must fit entirely within a segment, so we expect the + // first segment to include the first feature and the first polygon + // of the second feature, and the second segment to include the + // second polygon of the second feature. + expect(bucket.layoutVertexArray).toHaveLength(266); + expect(bucket.segments.get()[0]).toEqual({ + vertexOffset: 0, + vertexLength: 138, + primitiveOffset: 0, + primitiveLength: 134 + }); + expect(bucket.segments.get()[1]).toEqual({ + vertexOffset: 138, + vertexLength: 128, + primitiveOffset: 134, + primitiveLength: 126 + }); + +}); diff --git a/web/libraries/maplibre-gl/src/data/bucket/fill_bucket.ts b/web/libraries/maplibre-gl/src/data/bucket/fill_bucket.ts new file mode 100644 index 00000000..2fc87116 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/fill_bucket.ts @@ -0,0 +1,228 @@ +import {FillLayoutArray} from '../array_types.g'; + +import {members as layoutAttributes} from './fill_attributes'; +import {SegmentVector} from '../segment'; +import {ProgramConfigurationSet} from '../program_configuration'; +import {LineIndexArray, TriangleIndexArray} from '../index_array_type'; +import earcut from 'earcut'; +import {classifyRings} from '../../util/classify_rings'; +const EARCUT_MAX_RINGS = 500; +import {register} from '../../util/web_worker_transfer'; +import {hasPattern, addPatternDependencies} from './pattern_bucket_features'; +import {loadGeometry} from '../load_geometry'; +import {toEvaluationFeature} from '../evaluation_feature'; +import {EvaluationParameters} from '../../style/evaluation_parameters'; + +import type {CanonicalTileID} from '../../source/tile_id'; +import type { + Bucket, + BucketParameters, + BucketFeature, + IndexedFeature, + PopulateParameters +} from '../bucket'; +import type {FillStyleLayer} from '../../style/style_layer/fill_style_layer'; +import type {Context} from '../../gl/context'; +import type {IndexBuffer} from '../../gl/index_buffer'; +import type {VertexBuffer} from '../../gl/vertex_buffer'; +import type Point from '@mapbox/point-geometry'; +import type {FeatureStates} from '../../source/source_state'; +import type {ImagePosition} from '../../render/image_atlas'; +import type {VectorTileLayer} from '@mapbox/vector-tile'; + +export class FillBucket implements Bucket { + index: number; + zoom: number; + overscaling: number; + layers: Array; + layerIds: Array; + stateDependentLayers: Array; + stateDependentLayerIds: Array; + patternFeatures: Array; + + layoutVertexArray: FillLayoutArray; + layoutVertexBuffer: VertexBuffer; + + indexArray: TriangleIndexArray; + indexBuffer: IndexBuffer; + + indexArray2: LineIndexArray; + indexBuffer2: IndexBuffer; + + hasPattern: boolean; + programConfigurations: ProgramConfigurationSet; + segments: SegmentVector; + segments2: SegmentVector; + uploaded: boolean; + + constructor(options: BucketParameters) { + this.zoom = options.zoom; + this.overscaling = options.overscaling; + this.layers = options.layers; + this.layerIds = this.layers.map(layer => layer.id); + this.index = options.index; + this.hasPattern = false; + this.patternFeatures = []; + + this.layoutVertexArray = new FillLayoutArray(); + this.indexArray = new TriangleIndexArray(); + this.indexArray2 = new LineIndexArray(); + this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom); + this.segments = new SegmentVector(); + this.segments2 = new SegmentVector(); + this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id); + } + + populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) { + this.hasPattern = hasPattern('fill', this.layers, options); + const fillSortKey = this.layers[0].layout.get('fill-sort-key'); + const sortFeaturesByKey = !fillSortKey.isConstant(); + const bucketFeatures: BucketFeature[] = []; + + for (const {feature, id, index, sourceLayerIndex} of features) { + const needGeometry = this.layers[0]._featureFilter.needGeometry; + const evaluationFeature = toEvaluationFeature(feature, needGeometry); + + if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue; + + const sortKey = sortFeaturesByKey ? + fillSortKey.evaluate(evaluationFeature, {}, canonical, options.availableImages) : + undefined; + + const bucketFeature: BucketFeature = { + id, + properties: feature.properties, + type: feature.type, + sourceLayerIndex, + index, + geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature), + patterns: {}, + sortKey + }; + + bucketFeatures.push(bucketFeature); + } + + if (sortFeaturesByKey) { + bucketFeatures.sort((a, b) => a.sortKey - b.sortKey); + } + + for (const bucketFeature of bucketFeatures) { + const {geometry, index, sourceLayerIndex} = bucketFeature; + + if (this.hasPattern) { + const patternFeature = addPatternDependencies('fill', this.layers, bucketFeature, this.zoom, options); + // pattern features are added only once the pattern is loaded into the image atlas + // so are stored during populate until later updated with positions by tile worker in addFeatures + this.patternFeatures.push(patternFeature); + } else { + this.addFeature(bucketFeature, geometry, index, canonical, {}); + } + + const feature = features[index].feature; + options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index); + } + } + + update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: { + [_: string]: ImagePosition; + }) { + if (!this.stateDependentLayers.length) return; + this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions); + } + + addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: { + [_: string]: ImagePosition; + }) { + for (const feature of this.patternFeatures) { + this.addFeature(feature, feature.geometry, feature.index, canonical, imagePositions); + } + } + + isEmpty() { + return this.layoutVertexArray.length === 0; + } + + uploadPending(): boolean { + return !this.uploaded || this.programConfigurations.needsUpload; + } + upload(context: Context) { + if (!this.uploaded) { + this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes); + this.indexBuffer = context.createIndexBuffer(this.indexArray); + this.indexBuffer2 = context.createIndexBuffer(this.indexArray2); + } + this.programConfigurations.upload(context); + this.uploaded = true; + } + + destroy() { + if (!this.layoutVertexBuffer) return; + this.layoutVertexBuffer.destroy(); + this.indexBuffer.destroy(); + this.indexBuffer2.destroy(); + this.programConfigurations.destroy(); + this.segments.destroy(); + this.segments2.destroy(); + } + + addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: { + [_: string]: ImagePosition; + }) { + for (const polygon of classifyRings(geometry, EARCUT_MAX_RINGS)) { + let numVertices = 0; + for (const ring of polygon) { + numVertices += ring.length; + } + + const triangleSegment = this.segments.prepareSegment(numVertices, this.layoutVertexArray, this.indexArray); + const triangleIndex = triangleSegment.vertexLength; + + const flattened = []; + const holeIndices = []; + + for (const ring of polygon) { + if (ring.length === 0) { + continue; + } + + if (ring !== polygon[0]) { + holeIndices.push(flattened.length / 2); + } + + const lineSegment = this.segments2.prepareSegment(ring.length, this.layoutVertexArray, this.indexArray2); + const lineIndex = lineSegment.vertexLength; + + this.layoutVertexArray.emplaceBack(ring[0].x, ring[0].y); + this.indexArray2.emplaceBack(lineIndex + ring.length - 1, lineIndex); + flattened.push(ring[0].x); + flattened.push(ring[0].y); + + for (let i = 1; i < ring.length; i++) { + this.layoutVertexArray.emplaceBack(ring[i].x, ring[i].y); + this.indexArray2.emplaceBack(lineIndex + i - 1, lineIndex + i); + flattened.push(ring[i].x); + flattened.push(ring[i].y); + } + + lineSegment.vertexLength += ring.length; + lineSegment.primitiveLength += ring.length; + } + + const indices = earcut(flattened, holeIndices); + + for (let i = 0; i < indices.length; i += 3) { + this.indexArray.emplaceBack( + triangleIndex + indices[i], + triangleIndex + indices[i + 1], + triangleIndex + indices[i + 2]); + } + + triangleSegment.vertexLength += numVertices; + triangleSegment.primitiveLength += indices.length / 3; + } + this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical); + } +} + +register('FillBucket', FillBucket, {omit: ['layers', 'patternFeatures']}); diff --git a/web/libraries/maplibre-gl/src/data/bucket/fill_extrusion_attributes.ts b/web/libraries/maplibre-gl/src/data/bucket/fill_extrusion_attributes.ts new file mode 100644 index 00000000..e4e2936c --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/fill_extrusion_attributes.ts @@ -0,0 +1,13 @@ +import {createLayout} from '../../util/struct_array'; + +const layout = createLayout([ + {name: 'a_pos', components: 2, type: 'Int16'}, + {name: 'a_normal_ed', components: 4, type: 'Int16'}, +], 4); + +export const centroidAttributes = createLayout([ + {name: 'a_centroid', components: 2, type: 'Int16'} +], 4); + +export default layout; +export const {members, size, alignment} = layout; diff --git a/web/libraries/maplibre-gl/src/data/bucket/fill_extrusion_bucket.ts b/web/libraries/maplibre-gl/src/data/bucket/fill_extrusion_bucket.ts new file mode 100644 index 00000000..9dc1a1b2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/fill_extrusion_bucket.ts @@ -0,0 +1,302 @@ +import {FillExtrusionLayoutArray, PosArray} from '../array_types.g'; + +import {members as layoutAttributes, centroidAttributes} from './fill_extrusion_attributes'; +import {SegmentVector} from '../segment'; +import {ProgramConfigurationSet} from '../program_configuration'; +import {TriangleIndexArray} from '../index_array_type'; +import {EXTENT} from '../extent'; +import earcut from 'earcut'; +import mvt from '@mapbox/vector-tile'; +const vectorTileFeatureTypes = mvt.VectorTileFeature.types; +import {classifyRings} from '../../util/classify_rings'; +const EARCUT_MAX_RINGS = 500; +import {register} from '../../util/web_worker_transfer'; +import {hasPattern, addPatternDependencies} from './pattern_bucket_features'; +import {loadGeometry} from '../load_geometry'; +import {toEvaluationFeature} from '../evaluation_feature'; +import {EvaluationParameters} from '../../style/evaluation_parameters'; + +import type {CanonicalTileID} from '../../source/tile_id'; +import type { + Bucket, + BucketParameters, + BucketFeature, + IndexedFeature, + PopulateParameters +} from '../bucket'; + +import type {FillExtrusionStyleLayer} from '../../style/style_layer/fill_extrusion_style_layer'; +import type {Context} from '../../gl/context'; +import type {IndexBuffer} from '../../gl/index_buffer'; +import type {VertexBuffer} from '../../gl/vertex_buffer'; +import type Point from '@mapbox/point-geometry'; +import type {FeatureStates} from '../../source/source_state'; +import type {ImagePosition} from '../../render/image_atlas'; +import type {VectorTileLayer} from '@mapbox/vector-tile'; + +const FACTOR = Math.pow(2, 13); + +function addVertex(vertexArray, x, y, nx, ny, nz, t, e) { + vertexArray.emplaceBack( + // a_pos + x, + y, + // a_normal_ed: 3-component normal and 1-component edgedistance + Math.floor(nx * FACTOR) * 2 + t, + ny * FACTOR * 2, + nz * FACTOR * 2, + // edgedistance (used for wrapping patterns around extrusion sides) + Math.round(e) + ); +} + +export class FillExtrusionBucket implements Bucket { + index: number; + zoom: number; + overscaling: number; + layers: Array; + layerIds: Array; + stateDependentLayers: Array; + stateDependentLayerIds: Array; + + layoutVertexArray: FillExtrusionLayoutArray; + layoutVertexBuffer: VertexBuffer; + + centroidVertexArray: PosArray; + centroidVertexBuffer: VertexBuffer; + + indexArray: TriangleIndexArray; + indexBuffer: IndexBuffer; + + hasPattern: boolean; + programConfigurations: ProgramConfigurationSet; + segments: SegmentVector; + uploaded: boolean; + features: Array; + + constructor(options: BucketParameters) { + this.zoom = options.zoom; + this.overscaling = options.overscaling; + this.layers = options.layers; + this.layerIds = this.layers.map(layer => layer.id); + this.index = options.index; + this.hasPattern = false; + + this.layoutVertexArray = new FillExtrusionLayoutArray(); + this.centroidVertexArray = new PosArray(); + this.indexArray = new TriangleIndexArray(); + this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom); + this.segments = new SegmentVector(); + this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id); + } + + populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) { + this.features = []; + this.hasPattern = hasPattern('fill-extrusion', this.layers, options); + + for (const {feature, id, index, sourceLayerIndex} of features) { + const needGeometry = this.layers[0]._featureFilter.needGeometry; + const evaluationFeature = toEvaluationFeature(feature, needGeometry); + + if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue; + + const bucketFeature: BucketFeature = { + id, + sourceLayerIndex, + index, + geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature), + properties: feature.properties, + type: feature.type, + patterns: {} + }; + + if (this.hasPattern) { + this.features.push(addPatternDependencies('fill-extrusion', this.layers, bucketFeature, this.zoom, options)); + } else { + this.addFeature(bucketFeature, bucketFeature.geometry, index, canonical, {}); + } + + options.featureIndex.insert(feature, bucketFeature.geometry, index, sourceLayerIndex, this.index, true); + } + } + + addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) { + for (const feature of this.features) { + const {geometry} = feature; + this.addFeature(feature, geometry, feature.index, canonical, imagePositions); + } + } + + update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) { + if (!this.stateDependentLayers.length) return; + this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions); + } + + isEmpty() { + return this.layoutVertexArray.length === 0 && this.centroidVertexArray.length === 0; + } + + uploadPending() { + return !this.uploaded || this.programConfigurations.needsUpload; + } + + upload(context: Context) { + if (!this.uploaded) { + this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes); + this.centroidVertexBuffer = context.createVertexBuffer(this.centroidVertexArray, centroidAttributes.members, true); + this.indexBuffer = context.createIndexBuffer(this.indexArray); + } + this.programConfigurations.upload(context); + this.uploaded = true; + } + + destroy() { + if (!this.layoutVertexBuffer) return; + this.layoutVertexBuffer.destroy(); + this.indexBuffer.destroy(); + this.programConfigurations.destroy(); + this.segments.destroy(); + this.centroidVertexBuffer.destroy(); + } + + addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) { + const centroid = {x: 0, y: 0, vertexCount: 0}; + for (const polygon of classifyRings(geometry, EARCUT_MAX_RINGS)) { + let numVertices = 0; + for (const ring of polygon) { + numVertices += ring.length; + } + let segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray); + + for (const ring of polygon) { + if (ring.length === 0) { + continue; + } + + if (isEntirelyOutside(ring)) { + continue; + } + + let edgeDistance = 0; + + for (let p = 0; p < ring.length; p++) { + const p1 = ring[p]; + + if (p >= 1) { + const p2 = ring[p - 1]; + + if (!isBoundaryEdge(p1, p2)) { + if (segment.vertexLength + 4 > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) { + segment = this.segments.prepareSegment(4, this.layoutVertexArray, this.indexArray); + } + + const perp = p1.sub(p2)._perp()._unit(); + const dist = p2.dist(p1); + if (edgeDistance + dist > 32768) edgeDistance = 0; + + addVertex(this.layoutVertexArray, p1.x, p1.y, perp.x, perp.y, 0, 0, edgeDistance); + addVertex(this.layoutVertexArray, p1.x, p1.y, perp.x, perp.y, 0, 1, edgeDistance); + centroid.x += 2 * p1.x; + centroid.y += 2 * p1.y; + centroid.vertexCount += 2; + + edgeDistance += dist; + + addVertex(this.layoutVertexArray, p2.x, p2.y, perp.x, perp.y, 0, 0, edgeDistance); + addVertex(this.layoutVertexArray, p2.x, p2.y, perp.x, perp.y, 0, 1, edgeDistance); + centroid.x += 2 * p2.x; + centroid.y += 2 * p2.y; + centroid.vertexCount += 2; + + const bottomRight = segment.vertexLength; + + // ┌──────┐ + // │ 0 1 │ Counter-clockwise winding order. + // │ │ Triangle 1: 0 => 2 => 1 + // │ 2 3 │ Triangle 2: 1 => 2 => 3 + // └──────┘ + this.indexArray.emplaceBack(bottomRight, bottomRight + 2, bottomRight + 1); + this.indexArray.emplaceBack(bottomRight + 1, bottomRight + 2, bottomRight + 3); + + segment.vertexLength += 4; + segment.primitiveLength += 2; + } + } + } + + } + + if (segment.vertexLength + numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) { + segment = this.segments.prepareSegment(numVertices, this.layoutVertexArray, this.indexArray); + } + + //Only triangulate and draw the area of the feature if it is a polygon + //Other feature types (e.g. LineString) do not have area, so triangulation is pointless / undefined + if (vectorTileFeatureTypes[feature.type] !== 'Polygon') + continue; + + const flattened = []; + const holeIndices = []; + const triangleIndex = segment.vertexLength; + + for (const ring of polygon) { + if (ring.length === 0) { + continue; + } + + if (ring !== polygon[0]) { + holeIndices.push(flattened.length / 2); + } + + for (let i = 0; i < ring.length; i++) { + const p = ring[i]; + + addVertex(this.layoutVertexArray, p.x, p.y, 0, 0, 1, 1, 0); + centroid.x += p.x; + centroid.y += p.y; + centroid.vertexCount += 1; + + flattened.push(p.x); + flattened.push(p.y); + } + + } + + const indices = earcut(flattened, holeIndices); + + for (let j = 0; j < indices.length; j += 3) { + // Counter-clockwise winding order. + this.indexArray.emplaceBack( + triangleIndex + indices[j], + triangleIndex + indices[j + 2], + triangleIndex + indices[j + 1]); + } + + segment.primitiveLength += indices.length / 3; + segment.vertexLength += numVertices; + } + + // remember polygon centroid to calculate elevation in GPU + for (let i = 0; i < centroid.vertexCount; i++) { + this.centroidVertexArray.emplaceBack( + Math.floor(centroid.x / centroid.vertexCount), + Math.floor(centroid.y / centroid.vertexCount) + ); + } + this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical); + } +} + +register('FillExtrusionBucket', FillExtrusionBucket, {omit: ['layers', 'features']}); + +function isBoundaryEdge(p1, p2) { + return (p1.x === p2.x && (p1.x < 0 || p1.x > EXTENT)) || + (p1.y === p2.y && (p1.y < 0 || p1.y > EXTENT)); +} + +function isEntirelyOutside(ring) { + return ring.every(p => p.x < 0) || + ring.every(p => p.x > EXTENT) || + ring.every(p => p.y < 0) || + ring.every(p => p.y > EXTENT); +} diff --git a/web/libraries/maplibre-gl/src/data/bucket/heatmap_bucket.ts b/web/libraries/maplibre-gl/src/data/bucket/heatmap_bucket.ts new file mode 100644 index 00000000..792cd108 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/heatmap_bucket.ts @@ -0,0 +1,12 @@ +import {CircleBucket} from './circle_bucket'; +import {register} from '../../util/web_worker_transfer'; + +import type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer'; + +export class HeatmapBucket extends CircleBucket { + // Needed for flow to accept omit: ['layers'] below, due to + // https://github.com/facebook/flow/issues/4262 + layers: Array; +} + +register('HeatmapBucket', HeatmapBucket, {omit: ['layers']}); diff --git a/web/libraries/maplibre-gl/src/data/bucket/line_attributes.ts b/web/libraries/maplibre-gl/src/data/bucket/line_attributes.ts new file mode 100644 index 00000000..2daca3c1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/line_attributes.ts @@ -0,0 +1,8 @@ +import {createLayout} from '../../util/struct_array'; + +export const lineLayoutAttributes = createLayout([ + {name: 'a_pos_normal', components: 2, type: 'Int16'}, + {name: 'a_data', components: 4, type: 'Uint8'} +], 4); + +export const {members, size, alignment} = lineLayoutAttributes; diff --git a/web/libraries/maplibre-gl/src/data/bucket/line_attributes_ext.ts b/web/libraries/maplibre-gl/src/data/bucket/line_attributes_ext.ts new file mode 100644 index 00000000..5fd4ec45 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/line_attributes_ext.ts @@ -0,0 +1,8 @@ +import {createLayout} from '../../util/struct_array'; + +export const lineLayoutAttributesExt = createLayout([ + {name: 'a_uv_x', components: 1, type: 'Float32'}, + {name: 'a_split_index', components: 1, type: 'Float32'}, +]); + +export const {members, size, alignment} = lineLayoutAttributesExt; diff --git a/web/libraries/maplibre-gl/src/data/bucket/line_bucket.test.ts b/web/libraries/maplibre-gl/src/data/bucket/line_bucket.test.ts new file mode 100644 index 00000000..30fa8713 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/line_bucket.test.ts @@ -0,0 +1,142 @@ +import fs from 'fs'; +import path from 'path'; +import Protobuf from 'pbf'; +import {VectorTile} from '@mapbox/vector-tile'; +import Point from '@mapbox/point-geometry'; +import {SegmentVector} from '../segment'; +import {LineBucket} from './line_bucket'; +import {LineStyleLayer} from '../../style/style_layer/line_style_layer'; +import {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {EvaluationParameters} from '../../style/evaluation_parameters'; +import {BucketFeature, BucketParameters} from '../bucket'; + +// Load a line feature from fixture tile. +const vt = new VectorTile(new Protobuf(fs.readFileSync(path.resolve(__dirname, '../../../test/unit/assets/mbsv5-6-18-23.vector.pbf')))); +const feature = vt.layers.road.feature(0); + +function createLine(numPoints) { + const points = []; + for (let i = 0; i < numPoints; i++) { + points.push(new Point(i / numPoints, i / numPoints)); + } + return points; +} + +describe('LineBucket', () => { + test('LineBucket', () => { + expect(() => { + const layer = new LineStyleLayer({id: 'test', type: 'line'} as LayerSpecification); + layer.recalculate({zoom: 0, zoomHistory: {}} as EvaluationParameters, undefined); + + const bucket = new LineBucket({layers: [layer]} as BucketParameters); + + const line = { + type: 2, + properties: {} + } as BucketFeature; + + const polygon = { + type: 3, + properties: {} + } as BucketFeature; + + bucket.addLine([ + new Point(0, 0) + ], line, undefined, undefined, undefined, undefined); + + bucket.addLine([ + new Point(0, 0) + ], polygon, undefined, undefined, undefined, undefined); + + bucket.addLine([ + new Point(0, 0), + new Point(0, 0) + ], line, undefined, undefined, undefined, undefined); + + bucket.addLine([ + new Point(0, 0), + new Point(0, 0) + ], polygon, undefined, undefined, undefined, undefined); + + bucket.addLine([ + new Point(0, 0), + new Point(10, 10), + new Point(0, 0) + ], line, undefined, undefined, undefined, undefined); + + bucket.addLine([ + new Point(0, 0), + new Point(10, 10), + new Point(0, 0) + ], polygon, undefined, undefined, undefined, undefined); + + bucket.addLine([ + new Point(0, 0), + new Point(10, 10), + new Point(10, 20) + ], line, undefined, undefined, undefined, undefined); + + bucket.addLine([ + new Point(0, 0), + new Point(10, 10), + new Point(10, 20) + ], polygon, undefined, undefined, undefined, undefined); + + bucket.addLine([ + new Point(0, 0), + new Point(10, 10), + new Point(10, 20), + new Point(0, 0) + ], line, undefined, undefined, undefined, undefined); + + bucket.addLine([ + new Point(0, 0), + new Point(10, 10), + new Point(10, 20), + new Point(0, 0) + ], polygon, undefined, undefined, undefined, undefined); + + bucket.addFeature(feature as any, feature.loadGeometry(), undefined, undefined, undefined); + }).not.toThrow(); + }); + + test('LineBucket segmentation', () => { + jest.spyOn(console, 'warn').mockImplementation(() => { }); + + // Stub MAX_VERTEX_ARRAY_LENGTH so we can test features + // breaking across array groups without tests taking a _long_ time. + SegmentVector.MAX_VERTEX_ARRAY_LENGTH = 256; + + const layer = new LineStyleLayer({id: 'test', type: 'line'} as LayerSpecification); + layer.recalculate({zoom: 0, zoomHistory: {}} as EvaluationParameters, undefined); + + const bucket = new LineBucket({layers: [layer]} as BucketParameters); + + // first add an initial, small feature to make sure the next one starts at + // a non-zero offset + bucket.addFeature({} as BucketFeature, [createLine(10)], undefined, undefined, undefined); + + // add a feature that will break across the group boundary + bucket.addFeature({} as BucketFeature, [createLine(128)], undefined, undefined, undefined); + + // Each polygon must fit entirely within a segment, so we expect the + // first segment to include the first feature and the first polygon + // of the second feature, and the second segment to include the + // second polygon of the second feature. + expect(bucket.layoutVertexArray).toHaveLength(276); + expect(bucket.segments.get()).toEqual([{ + vertexOffset: 0, + vertexLength: 20, + primitiveOffset: 0, + primitiveLength: 18 + }, { + vertexOffset: 20, + vertexLength: 256, + primitiveOffset: 18, + primitiveLength: 254 + }]); + + expect(console.warn).toHaveBeenCalledTimes(1); + + }); +}); diff --git a/web/libraries/maplibre-gl/src/data/bucket/line_bucket.ts b/web/libraries/maplibre-gl/src/data/bucket/line_bucket.ts new file mode 100644 index 00000000..ea816f88 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/line_bucket.ts @@ -0,0 +1,592 @@ +import {LineLayoutArray, LineExtLayoutArray} from '../array_types.g'; + +import {members as layoutAttributes} from './line_attributes'; +import {members as layoutAttributesExt} from './line_attributes_ext'; +import {SegmentVector} from '../segment'; +import {ProgramConfigurationSet} from '../program_configuration'; +import {TriangleIndexArray} from '../index_array_type'; +import {EXTENT} from '../extent'; +import mvt from '@mapbox/vector-tile'; +const vectorTileFeatureTypes = mvt.VectorTileFeature.types; +import {register} from '../../util/web_worker_transfer'; +import {hasPattern, addPatternDependencies} from './pattern_bucket_features'; +import {loadGeometry} from '../load_geometry'; +import {toEvaluationFeature} from '../evaluation_feature'; +import {EvaluationParameters} from '../../style/evaluation_parameters'; + +import type {CanonicalTileID} from '../../source/tile_id'; +import type { + Bucket, + BucketParameters, + BucketFeature, + IndexedFeature, + PopulateParameters +} from '../bucket'; +import type {LineStyleLayer} from '../../style/style_layer/line_style_layer'; +import type Point from '@mapbox/point-geometry'; +import type {Segment} from '../segment'; +import {RGBAImage} from '../../util/image'; +import type {Context} from '../../gl/context'; +import type {Texture} from '../../render/texture'; +import type {IndexBuffer} from '../../gl/index_buffer'; +import type {VertexBuffer} from '../../gl/vertex_buffer'; +import type {FeatureStates} from '../../source/source_state'; +import type {ImagePosition} from '../../render/image_atlas'; +import type {VectorTileLayer} from '@mapbox/vector-tile'; + +// NOTE ON EXTRUDE SCALE: +// scale the extrusion vector so that the normal length is this value. +// contains the "texture" normals (-1..1). this is distinct from the extrude +// normals for line joins, because the x-value remains 0 for the texture +// normal array, while the extrude normal actually moves the vertex to create +// the acute/bevelled line join. +const EXTRUDE_SCALE = 63; + +/* + * Sharp corners cause dashed lines to tilt because the distance along the line + * is the same at both the inner and outer corners. To improve the appearance of + * dashed lines we add extra points near sharp corners so that a smaller part + * of the line is tilted. + * + * COS_HALF_SHARP_CORNER controls how sharp a corner has to be for us to add an + * extra vertex. The default is 75 degrees. + * + * The newly created vertices are placed SHARP_CORNER_OFFSET pixels from the corner. + */ +const COS_HALF_SHARP_CORNER = Math.cos(75 / 2 * (Math.PI / 180)); +const SHARP_CORNER_OFFSET = 15; + +// Angle per triangle for approximating round line joins. +const DEG_PER_TRIANGLE = 20; + +// The number of bits that is used to store the line distance in the buffer. +const LINE_DISTANCE_BUFFER_BITS = 15; + +// We don't have enough bits for the line distance as we'd like to have, so +// use this value to scale the line distance (in tile units) down to a smaller +// value. This lets us store longer distances while sacrificing precision. +const LINE_DISTANCE_SCALE = 1 / 2; + +// The maximum line distance, in tile units, that fits in the buffer. +const MAX_LINE_DISTANCE = Math.pow(2, LINE_DISTANCE_BUFFER_BITS - 1) / LINE_DISTANCE_SCALE; + +type LineClips = { + start: number; + end: number; +}; + +type GradientTexture = { + texture?: Texture; + gradient?: RGBAImage; + version?: number; +}; + +/** + * @internal + * Line bucket class + */ +export class LineBucket implements Bucket { + distance: number; + totalDistance: number; + maxLineLength: number; + scaledDistance: number; + lineClips?: LineClips; + + e1: number; + e2: number; + + index: number; + zoom: number; + overscaling: number; + layers: Array; + layerIds: Array; + gradients: {[x: string]: GradientTexture}; + stateDependentLayers: Array; + stateDependentLayerIds: Array; + patternFeatures: Array; + lineClipsArray: Array; + + layoutVertexArray: LineLayoutArray; + layoutVertexBuffer: VertexBuffer; + layoutVertexArray2: LineExtLayoutArray; + layoutVertexBuffer2: VertexBuffer; + + indexArray: TriangleIndexArray; + indexBuffer: IndexBuffer; + + hasPattern: boolean; + programConfigurations: ProgramConfigurationSet; + segments: SegmentVector; + uploaded: boolean; + + constructor(options: BucketParameters) { + this.zoom = options.zoom; + this.overscaling = options.overscaling; + this.layers = options.layers; + this.layerIds = this.layers.map(layer => layer.id); + this.index = options.index; + this.hasPattern = false; + this.patternFeatures = []; + this.lineClipsArray = []; + this.gradients = {}; + this.layers.forEach(layer => { + this.gradients[layer.id] = {}; + }); + + this.layoutVertexArray = new LineLayoutArray(); + this.layoutVertexArray2 = new LineExtLayoutArray(); + this.indexArray = new TriangleIndexArray(); + this.programConfigurations = new ProgramConfigurationSet(options.layers, options.zoom); + this.segments = new SegmentVector(); + this.maxLineLength = 0; + + this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id); + } + + populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) { + this.hasPattern = hasPattern('line', this.layers, options); + const lineSortKey = this.layers[0].layout.get('line-sort-key'); + const sortFeaturesByKey = !lineSortKey.isConstant(); + const bucketFeatures: BucketFeature[] = []; + + for (const {feature, id, index, sourceLayerIndex} of features) { + const needGeometry = this.layers[0]._featureFilter.needGeometry; + const evaluationFeature = toEvaluationFeature(feature, needGeometry); + + if (!this.layers[0]._featureFilter.filter(new EvaluationParameters(this.zoom), evaluationFeature, canonical)) continue; + + const sortKey = sortFeaturesByKey ? + lineSortKey.evaluate(evaluationFeature, {}, canonical) : + undefined; + + const bucketFeature: BucketFeature = { + id, + properties: feature.properties, + type: feature.type, + sourceLayerIndex, + index, + geometry: needGeometry ? evaluationFeature.geometry : loadGeometry(feature), + patterns: {}, + sortKey + }; + + bucketFeatures.push(bucketFeature); + } + + if (sortFeaturesByKey) { + bucketFeatures.sort((a, b) => { + return (a.sortKey) - (b.sortKey); + }); + } + + for (const bucketFeature of bucketFeatures) { + const {geometry, index, sourceLayerIndex} = bucketFeature; + + if (this.hasPattern) { + const patternBucketFeature = addPatternDependencies('line', this.layers, bucketFeature, this.zoom, options); + // pattern features are added only once the pattern is loaded into the image atlas + // so are stored during populate until later updated with positions by tile worker in addFeatures + this.patternFeatures.push(patternBucketFeature); + } else { + this.addFeature(bucketFeature, geometry, index, canonical, {}); + } + + const feature = features[index].feature; + options.featureIndex.insert(feature, geometry, index, sourceLayerIndex, this.index); + } + } + + update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) { + if (!this.stateDependentLayers.length) return; + this.programConfigurations.updatePaintArrays(states, vtLayer, this.stateDependentLayers, imagePositions); + } + + addFeatures(options: PopulateParameters, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) { + for (const feature of this.patternFeatures) { + this.addFeature(feature, feature.geometry, feature.index, canonical, imagePositions); + } + } + + isEmpty() { + return this.layoutVertexArray.length === 0; + } + + uploadPending() { + return !this.uploaded || this.programConfigurations.needsUpload; + } + + upload(context: Context) { + if (!this.uploaded) { + if (this.layoutVertexArray2.length !== 0) { + this.layoutVertexBuffer2 = context.createVertexBuffer(this.layoutVertexArray2, layoutAttributesExt); + } + this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, layoutAttributes); + this.indexBuffer = context.createIndexBuffer(this.indexArray); + } + this.programConfigurations.upload(context); + this.uploaded = true; + } + + destroy() { + if (!this.layoutVertexBuffer) return; + this.layoutVertexBuffer.destroy(); + this.indexBuffer.destroy(); + this.programConfigurations.destroy(); + this.segments.destroy(); + } + + lineFeatureClips(feature: BucketFeature): LineClips | undefined { + if (!!feature.properties && Object.prototype.hasOwnProperty.call(feature.properties, 'mapbox_clip_start') && Object.prototype.hasOwnProperty.call(feature.properties, 'mapbox_clip_end')) { + const start = +feature.properties['mapbox_clip_start']; + const end = +feature.properties['mapbox_clip_end']; + return {start, end}; + } + } + + addFeature(feature: BucketFeature, geometry: Array>, index: number, canonical: CanonicalTileID, imagePositions: {[_: string]: ImagePosition}) { + const layout = this.layers[0].layout; + const join = layout.get('line-join').evaluate(feature, {}); + const cap = layout.get('line-cap'); + const miterLimit = layout.get('line-miter-limit'); + const roundLimit = layout.get('line-round-limit'); + this.lineClips = this.lineFeatureClips(feature); + + for (const line of geometry) { + this.addLine(line, feature, join, cap, miterLimit, roundLimit); + } + + this.programConfigurations.populatePaintArrays(this.layoutVertexArray.length, feature, index, imagePositions, canonical); + } + + addLine(vertices: Array, feature: BucketFeature, join: string, cap: string, miterLimit: number, roundLimit: number) { + this.distance = 0; + this.scaledDistance = 0; + this.totalDistance = 0; + + if (this.lineClips) { + this.lineClipsArray.push(this.lineClips); + // Calculate the total distance, in tile units, of this tiled line feature + for (let i = 0; i < vertices.length - 1; i++) { + this.totalDistance += vertices[i].dist(vertices[i + 1]); + } + this.updateScaledDistance(); + this.maxLineLength = Math.max(this.maxLineLength, this.totalDistance); + } + + const isPolygon = vectorTileFeatureTypes[feature.type] === 'Polygon'; + + // If the line has duplicate vertices at the ends, adjust start/length to remove them. + let len = vertices.length; + while (len >= 2 && vertices[len - 1].equals(vertices[len - 2])) { + len--; + } + let first = 0; + while (first < len - 1 && vertices[first].equals(vertices[first + 1])) { + first++; + } + + // Ignore invalid geometry. + if (len < (isPolygon ? 3 : 2)) return; + + if (join === 'bevel') miterLimit = 1.05; + + const sharpCornerOffset = this.overscaling <= 16 ? + SHARP_CORNER_OFFSET * EXTENT / (512 * this.overscaling) : + 0; + + // we could be more precise, but it would only save a negligible amount of space + const segment = this.segments.prepareSegment(len * 10, this.layoutVertexArray, this.indexArray); + + let currentVertex: Point; + let prevVertex: Point; + let nextVertex: Point; + let prevNormal: Point; + let nextNormal: Point; + + // the last two vertices added + this.e1 = this.e2 = -1; + + if (isPolygon) { + currentVertex = vertices[len - 2]; + nextNormal = vertices[first].sub(currentVertex)._unit()._perp(); + } + + for (let i = first; i < len; i++) { + + nextVertex = i === len - 1 ? + (isPolygon ? vertices[first + 1] : undefined) : // if it's a polygon, treat the last vertex like the first + vertices[i + 1]; // just the next vertex + + // if two consecutive vertices exist, skip the current one + if (nextVertex && vertices[i].equals(nextVertex)) continue; + + if (nextNormal) prevNormal = nextNormal; + if (currentVertex) prevVertex = currentVertex; + + currentVertex = vertices[i]; + + // Calculate the normal towards the next vertex in this line. In case + // there is no next vertex, pretend that the line is continuing straight, + // meaning that we are just using the previous normal. + nextNormal = nextVertex ? nextVertex.sub(currentVertex)._unit()._perp() : prevNormal; + + // If we still don't have a previous normal, this is the beginning of a + // non-closed line, so we're doing a straight "join". + prevNormal = prevNormal || nextNormal; + + // Determine the normal of the join extrusion. It is the angle bisector + // of the segments between the previous line and the next line. + // In the case of 180° angles, the prev and next normals cancel each other out: + // prevNormal + nextNormal = (0, 0), its magnitude is 0, so the unit vector would be + // undefined. In that case, we're keeping the joinNormal at (0, 0), so that the cosHalfAngle + // below will also become 0 and miterLength will become Infinity. + let joinNormal = prevNormal.add(nextNormal); + if (joinNormal.x !== 0 || joinNormal.y !== 0) { + joinNormal._unit(); + } + /* joinNormal prevNormal + * ↖ ↑ + * .________. prevVertex + * | + * nextNormal ← | currentVertex + * | + * nextVertex ! + * + */ + + // calculate cosines of the angle (and its half) using dot product + const cosAngle = prevNormal.x * nextNormal.x + prevNormal.y * nextNormal.y; + const cosHalfAngle = joinNormal.x * nextNormal.x + joinNormal.y * nextNormal.y; + + // Calculate the length of the miter (the ratio of the miter to the width) + // as the inverse of cosine of the angle between next and join normals + const miterLength = cosHalfAngle !== 0 ? 1 / cosHalfAngle : Infinity; + + // approximate angle from cosine + const approxAngle = 2 * Math.sqrt(2 - 2 * cosHalfAngle); + + const isSharpCorner = cosHalfAngle < COS_HALF_SHARP_CORNER && prevVertex && nextVertex; + const lineTurnsLeft = prevNormal.x * nextNormal.y - prevNormal.y * nextNormal.x > 0; + + if (isSharpCorner && i > first) { + const prevSegmentLength = currentVertex.dist(prevVertex); + if (prevSegmentLength > 2 * sharpCornerOffset) { + const newPrevVertex = currentVertex.sub(currentVertex.sub(prevVertex)._mult(sharpCornerOffset / prevSegmentLength)._round()); + this.updateDistance(prevVertex, newPrevVertex); + this.addCurrentVertex(newPrevVertex, prevNormal, 0, 0, segment); + prevVertex = newPrevVertex; + } + } + + // The join if a middle vertex, otherwise the cap. + const middleVertex = prevVertex && nextVertex; + let currentJoin = middleVertex ? join : isPolygon ? 'butt' : cap; + + if (middleVertex && currentJoin === 'round') { + if (miterLength < roundLimit) { + currentJoin = 'miter'; + } else if (miterLength <= 2) { + currentJoin = 'fakeround'; + } + } + + if (currentJoin === 'miter' && miterLength > miterLimit) { + currentJoin = 'bevel'; + } + + if (currentJoin === 'bevel') { + // The maximum extrude length is 128 / 63 = 2 times the width of the line + // so if miterLength >= 2 we need to draw a different type of bevel here. + if (miterLength > 2) currentJoin = 'flipbevel'; + + // If the miterLength is really small and the line bevel wouldn't be visible, + // just draw a miter join to save a triangle. + if (miterLength < miterLimit) currentJoin = 'miter'; + } + + // Calculate how far along the line the currentVertex is + if (prevVertex) this.updateDistance(prevVertex, currentVertex); + + if (currentJoin === 'miter') { + + joinNormal._mult(miterLength); + this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment); + + } else if (currentJoin === 'flipbevel') { + // miter is too big, flip the direction to make a beveled join + + if (miterLength > 100) { + // Almost parallel lines + joinNormal = nextNormal.mult(-1); + + } else { + const bevelLength = miterLength * prevNormal.add(nextNormal).mag() / prevNormal.sub(nextNormal).mag(); + joinNormal._perp()._mult(bevelLength * (lineTurnsLeft ? -1 : 1)); + } + this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment); + this.addCurrentVertex(currentVertex, joinNormal.mult(-1), 0, 0, segment); + + } else if (currentJoin === 'bevel' || currentJoin === 'fakeround') { + const offset = -Math.sqrt(miterLength * miterLength - 1); + const offsetA = lineTurnsLeft ? offset : 0; + const offsetB = lineTurnsLeft ? 0 : offset; + + // Close previous segment with a bevel + if (prevVertex) { + this.addCurrentVertex(currentVertex, prevNormal, offsetA, offsetB, segment); + } + + if (currentJoin === 'fakeround') { + // The join angle is sharp enough that a round join would be visible. + // Bevel joins fill the gap between segments with a single pie slice triangle. + // Create a round join by adding multiple pie slices. The join isn't actually round, but + // it looks like it is at the sizes we render lines at. + + // pick the number of triangles for approximating round join by based on the angle between normals + const n = Math.round((approxAngle * 180 / Math.PI) / DEG_PER_TRIANGLE); + + for (let m = 1; m < n; m++) { + let t = m / n; + if (t !== 0.5) { + // approximate spherical interpolation https://observablehq.com/@mourner/approximating-geometric-slerp + const t2 = t - 0.5; + const A = 1.0904 + cosAngle * (-3.2452 + cosAngle * (3.55645 - cosAngle * 1.43519)); + const B = 0.848013 + cosAngle * (-1.06021 + cosAngle * 0.215638); + t = t + t * t2 * (t - 1) * (A * t2 * t2 + B); + } + const extrude = nextNormal.sub(prevNormal)._mult(t)._add(prevNormal)._unit()._mult(lineTurnsLeft ? -1 : 1); + this.addHalfVertex(currentVertex, extrude.x, extrude.y, false, lineTurnsLeft, 0, segment); + } + } + + if (nextVertex) { + // Start next segment + this.addCurrentVertex(currentVertex, nextNormal, -offsetA, -offsetB, segment); + } + + } else if (currentJoin === 'butt') { + this.addCurrentVertex(currentVertex, joinNormal, 0, 0, segment); // butt cap + + } else if (currentJoin === 'square') { + const offset = prevVertex ? 1 : -1; // closing or starting square cap + this.addCurrentVertex(currentVertex, joinNormal, offset, offset, segment); + + } else if (currentJoin === 'round') { + + if (prevVertex) { + // Close previous segment with butt + this.addCurrentVertex(currentVertex, prevNormal, 0, 0, segment); + + // Add round cap or linejoin at end of segment + this.addCurrentVertex(currentVertex, prevNormal, 1, 1, segment, true); + } + if (nextVertex) { + // Add round cap before first segment + this.addCurrentVertex(currentVertex, nextNormal, -1, -1, segment, true); + + // Start next segment with a butt + this.addCurrentVertex(currentVertex, nextNormal, 0, 0, segment); + } + } + + if (isSharpCorner && i < len - 1) { + const nextSegmentLength = currentVertex.dist(nextVertex); + if (nextSegmentLength > 2 * sharpCornerOffset) { + const newCurrentVertex = currentVertex.add(nextVertex.sub(currentVertex)._mult(sharpCornerOffset / nextSegmentLength)._round()); + this.updateDistance(currentVertex, newCurrentVertex); + this.addCurrentVertex(newCurrentVertex, nextNormal, 0, 0, segment); + currentVertex = newCurrentVertex; + } + } + } + } + + /** + * Add two vertices to the buffers. + * + * @param p - the line vertex to add buffer vertices for + * @param normal - vertex normal + * @param endLeft - extrude to shift the left vertex along the line + * @param endRight - extrude to shift the left vertex along the line + * @param segment - the segment object to add the vertex to + * @param round - whether this is a round cap + */ + addCurrentVertex(p: Point, normal: Point, endLeft: number, endRight: number, segment: Segment, round: boolean = false) { + // left and right extrude vectors, perpendicularly shifted by endLeft/endRight + const leftX = normal.x + normal.y * endLeft; + const leftY = normal.y - normal.x * endLeft; + const rightX = -normal.x + normal.y * endRight; + const rightY = -normal.y - normal.x * endRight; + + this.addHalfVertex(p, leftX, leftY, round, false, endLeft, segment); + this.addHalfVertex(p, rightX, rightY, round, true, -endRight, segment); + + // There is a maximum "distance along the line" that we can store in the buffers. + // When we get close to the distance, reset it to zero and add the vertex again with + // a distance of zero. The max distance is determined by the number of bits we allocate + // to `linesofar`. + if (this.distance > MAX_LINE_DISTANCE / 2 && this.totalDistance === 0) { + this.distance = 0; + this.updateScaledDistance(); + this.addCurrentVertex(p, normal, endLeft, endRight, segment, round); + } + } + + addHalfVertex({x, y}: Point, extrudeX: number, extrudeY: number, round: boolean, up: boolean, dir: number, segment: Segment) { + const totalDistance = this.lineClips ? this.scaledDistance * (MAX_LINE_DISTANCE - 1) : this.scaledDistance; + // scale down so that we can store longer distances while sacrificing precision. + const linesofarScaled = totalDistance * LINE_DISTANCE_SCALE; + + this.layoutVertexArray.emplaceBack( + // a_pos_normal + // Encode round/up the least significant bits + (x << 1) + (round ? 1 : 0), + (y << 1) + (up ? 1 : 0), + // a_data + // add 128 to store a byte in an unsigned byte + Math.round(EXTRUDE_SCALE * extrudeX) + 128, + Math.round(EXTRUDE_SCALE * extrudeY) + 128, + // Encode the -1/0/1 direction value into the first two bits of .z of a_data. + // Combine it with the lower 6 bits of `linesofarScaled` (shifted by 2 bits to make + // room for the direction value). The upper 8 bits of `linesofarScaled` are placed in + // the `w` component. + ((dir === 0 ? 0 : (dir < 0 ? -1 : 1)) + 1) | ((linesofarScaled & 0x3F) << 2), + linesofarScaled >> 6); + + // Constructs a second vertex buffer with higher precision line progress + if (this.lineClips) { + const progressRealigned = this.scaledDistance - this.lineClips.start; + const endClipRealigned = this.lineClips.end - this.lineClips.start; + const uvX = progressRealigned / endClipRealigned; + this.layoutVertexArray2.emplaceBack(uvX, this.lineClipsArray.length); + } + + const e = segment.vertexLength++; + if (this.e1 >= 0 && this.e2 >= 0) { + this.indexArray.emplaceBack(this.e1, this.e2, e); + segment.primitiveLength++; + } + if (up) { + this.e2 = e; + } else { + this.e1 = e; + } + } + + updateScaledDistance() { + // Knowing the ratio of the full linestring covered by this tiled feature, as well + // as the total distance (in tile units) of this tiled feature, and the distance + // (in tile units) of the current vertex, we can determine the relative distance + // of this vertex along the full linestring feature and scale it to [0, 2^15) + this.scaledDistance = this.lineClips ? + this.lineClips.start + (this.lineClips.end - this.lineClips.start) * this.distance / this.totalDistance : + this.distance; + } + + updateDistance(prev: Point, next: Point) { + this.distance += prev.dist(next); + this.updateScaledDistance(); + } +} + +register('LineBucket', LineBucket, {omit: ['layers', 'patternFeatures']}); diff --git a/web/libraries/maplibre-gl/src/data/bucket/pattern_attributes.ts b/web/libraries/maplibre-gl/src/data/bucket/pattern_attributes.ts new file mode 100644 index 00000000..f308f9ba --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/pattern_attributes.ts @@ -0,0 +1,9 @@ +import {createLayout} from '../../util/struct_array'; + +export const patternAttributes = createLayout([ + // [tl.x, tl.y, br.x, br.y] + {name: 'a_pattern_from', components: 4, type: 'Uint16'}, + {name: 'a_pattern_to', components: 4, type: 'Uint16'}, + {name: 'a_pixel_ratio_from', components: 1, type: 'Uint16'}, + {name: 'a_pixel_ratio_to', components: 1, type: 'Uint16'}, +]); diff --git a/web/libraries/maplibre-gl/src/data/bucket/pattern_bucket_features.ts b/web/libraries/maplibre-gl/src/data/bucket/pattern_bucket_features.ts new file mode 100644 index 00000000..11693daa --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/pattern_bucket_features.ts @@ -0,0 +1,57 @@ +import type {FillStyleLayer} from '../../style/style_layer/fill_style_layer'; +import type {FillExtrusionStyleLayer} from '../../style/style_layer/fill_extrusion_style_layer'; +import type {LineStyleLayer} from '../../style/style_layer/line_style_layer'; + +import type { + BucketFeature, + PopulateParameters +} from '../bucket'; +import {PossiblyEvaluated} from '../../style/properties'; + +type PatternStyleLayers = Array | Array | Array; + +export function hasPattern(type: string, layers: PatternStyleLayers, options: PopulateParameters) { + const patterns = options.patternDependencies; + let hasPattern = false; + + for (const layer of layers) { + const patternProperty = (layer.paint as PossiblyEvaluated).get(`${type}-pattern`); + if (!patternProperty.isConstant()) { + hasPattern = true; + } + + const constantPattern = patternProperty.constantOr(null); + if (constantPattern) { + hasPattern = true; + patterns[constantPattern.to] = true; + patterns[constantPattern.from] = true; + } + } + + return hasPattern; +} + +export function addPatternDependencies(type: string, layers: PatternStyleLayers, patternFeature: BucketFeature, zoom: number, options: PopulateParameters) { + const patterns = options.patternDependencies; + for (const layer of layers) { + const patternProperty = (layer.paint as PossiblyEvaluated).get(`${type}-pattern`); + + const patternPropertyValue = patternProperty.value; + if (patternPropertyValue.kind !== 'constant') { + let min = patternPropertyValue.evaluate({zoom: zoom - 1}, patternFeature, {}, options.availableImages); + let mid = patternPropertyValue.evaluate({zoom}, patternFeature, {}, options.availableImages); + let max = patternPropertyValue.evaluate({zoom: zoom + 1}, patternFeature, {}, options.availableImages); + min = min && min.name ? min.name : min; + mid = mid && mid.name ? mid.name : mid; + max = max && max.name ? max.name : max; + // add to patternDependencies + patterns[min] = true; + patterns[mid] = true; + patterns[max] = true; + + // save for layout + patternFeature.patterns[layer.id] = {min, mid, max}; + } + } + return patternFeature; +} diff --git a/web/libraries/maplibre-gl/src/data/bucket/symbol_attributes.ts b/web/libraries/maplibre-gl/src/data/bucket/symbol_attributes.ts new file mode 100644 index 00000000..5b7a53cb --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/symbol_attributes.ts @@ -0,0 +1,121 @@ +import {createLayout} from '../../util/struct_array'; + +export const symbolLayoutAttributes = createLayout([ + {name: 'a_pos_offset', components: 4, type: 'Int16'}, + {name: 'a_data', components: 4, type: 'Uint16'}, + {name: 'a_pixeloffset', components: 4, type: 'Int16'} +], 4); + +export const dynamicLayoutAttributes = createLayout([ + {name: 'a_projected_pos', components: 3, type: 'Float32'} +], 4); + +export const placementOpacityAttributes = createLayout([ + {name: 'a_fade_opacity', components: 1, type: 'Uint32'} +], 4); + +export const collisionVertexAttributes = createLayout([ + {name: 'a_placed', components: 2, type: 'Uint8'}, + {name: 'a_shift', components: 2, type: 'Float32'} +]); + +export const collisionBox = createLayout([ + // the box is centered around the anchor point + {type: 'Int16', name: 'anchorPointX'}, + {type: 'Int16', name: 'anchorPointY'}, + + // distances to the edges from the anchor + {type: 'Int16', name: 'x1'}, + {type: 'Int16', name: 'y1'}, + {type: 'Int16', name: 'x2'}, + {type: 'Int16', name: 'y2'}, + + // the index of the feature in the original vectortile + {type: 'Uint32', name: 'featureIndex'}, + // the source layer the feature appears in + {type: 'Uint16', name: 'sourceLayerIndex'}, + // the bucket the feature appears in + {type: 'Uint16', name: 'bucketIndex'}, +]); + +export const collisionBoxLayout = createLayout([ // used to render collision boxes for debugging purposes + {name: 'a_pos', components: 2, type: 'Int16'}, + {name: 'a_anchor_pos', components: 2, type: 'Int16'}, + {name: 'a_extrude', components: 2, type: 'Int16'} +], 4); + +export const collisionCircleLayout = createLayout([ // used to render collision circles for debugging purposes + {name: 'a_pos', components: 2, type: 'Float32'}, + {name: 'a_radius', components: 1, type: 'Float32'}, + {name: 'a_flags', components: 2, type: 'Int16'} +], 4); + +export const quadTriangle = createLayout([ + {name: 'triangle', components: 3, type: 'Uint16'}, +]); + +export const placement = createLayout([ + {type: 'Int16', name: 'anchorX'}, + {type: 'Int16', name: 'anchorY'}, + {type: 'Uint16', name: 'glyphStartIndex'}, + {type: 'Uint16', name: 'numGlyphs'}, + {type: 'Uint32', name: 'vertexStartIndex'}, + {type: 'Uint32', name: 'lineStartIndex'}, + {type: 'Uint32', name: 'lineLength'}, + {type: 'Uint16', name: 'segment'}, + {type: 'Uint16', name: 'lowerSize'}, + {type: 'Uint16', name: 'upperSize'}, + {type: 'Float32', name: 'lineOffsetX'}, + {type: 'Float32', name: 'lineOffsetY'}, + {type: 'Uint8', name: 'writingMode'}, + {type: 'Uint8', name: 'placedOrientation'}, + {type: 'Uint8', name: 'hidden'}, + {type: 'Uint32', name: 'crossTileID'}, + {type: 'Int16', name: 'associatedIconIndex'} +]); + +export const symbolInstance = createLayout([ + {type: 'Int16', name: 'anchorX'}, + {type: 'Int16', name: 'anchorY'}, + {type: 'Int16', name: 'rightJustifiedTextSymbolIndex'}, + {type: 'Int16', name: 'centerJustifiedTextSymbolIndex'}, + {type: 'Int16', name: 'leftJustifiedTextSymbolIndex'}, + {type: 'Int16', name: 'verticalPlacedTextSymbolIndex'}, + {type: 'Int16', name: 'placedIconSymbolIndex'}, + {type: 'Int16', name: 'verticalPlacedIconSymbolIndex'}, + {type: 'Uint16', name: 'key'}, + {type: 'Uint16', name: 'textBoxStartIndex'}, + {type: 'Uint16', name: 'textBoxEndIndex'}, + {type: 'Uint16', name: 'verticalTextBoxStartIndex'}, + {type: 'Uint16', name: 'verticalTextBoxEndIndex'}, + {type: 'Uint16', name: 'iconBoxStartIndex'}, + {type: 'Uint16', name: 'iconBoxEndIndex'}, + {type: 'Uint16', name: 'verticalIconBoxStartIndex'}, + {type: 'Uint16', name: 'verticalIconBoxEndIndex'}, + {type: 'Uint16', name: 'featureIndex'}, + {type: 'Uint16', name: 'numHorizontalGlyphVertices'}, + {type: 'Uint16', name: 'numVerticalGlyphVertices'}, + {type: 'Uint16', name: 'numIconVertices'}, + {type: 'Uint16', name: 'numVerticalIconVertices'}, + {type: 'Uint16', name: 'useRuntimeCollisionCircles'}, + {type: 'Uint32', name: 'crossTileID'}, + {type: 'Float32', name: 'textBoxScale'}, + {type: 'Float32', name: 'collisionCircleDiameter'}, + {type: 'Uint16', name: 'textAnchorOffsetStartIndex'}, + {type: 'Uint16', name: 'textAnchorOffsetEndIndex'} +]); + +export const glyphOffset = createLayout([ + {type: 'Float32', name: 'offsetX'} +]); + +export const lineVertex = createLayout([ + {type: 'Int16', name: 'x'}, + {type: 'Int16', name: 'y'}, + {type: 'Int16', name: 'tileUnitDistanceFromAnchor'} +]); + +export const textAnchorOffset = createLayout([ + {type: 'Uint16', name: 'textAnchor'}, + {type: 'Float32', components: 2, name: 'textOffset'} +]); diff --git a/web/libraries/maplibre-gl/src/data/bucket/symbol_bucket.test.ts b/web/libraries/maplibre-gl/src/data/bucket/symbol_bucket.test.ts new file mode 100644 index 00000000..f94336a5 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/symbol_bucket.test.ts @@ -0,0 +1,242 @@ +import fs from 'fs'; +import path from 'path'; +import Protobuf from 'pbf'; +import {VectorTile} from '@mapbox/vector-tile'; +import {SymbolBucket} from './symbol_bucket'; +import {CollisionBoxArray} from '../../data/array_types.g'; +import {performSymbolLayout} from '../../symbol/symbol_layout'; +import {Placement} from '../../symbol/placement'; +import {Transform} from '../../geo/transform'; +import {CanonicalTileID, OverscaledTileID} from '../../source/tile_id'; +import {Tile} from '../../source/tile'; +import {CrossTileSymbolIndex} from '../../symbol/cross_tile_symbol_index'; +import {FeatureIndex} from '../../data/feature_index'; +import {createSymbolBucket, createSymbolIconBucket} from '../../../test/unit/lib/create_symbol_layer'; +import {RGBAImage} from '../../util/image'; +import {ImagePosition} from '../../render/image_atlas'; +import {IndexedFeature, PopulateParameters} from '../bucket'; +import {StyleImage} from '../../style/style_image'; +import glyphs from '../../../test/unit/assets/fontstack-glyphs.json' assert {type: 'json'}; +import {StyleGlyph} from '../../style/style_glyph'; + +// Load a point feature from fixture tile. +const vt = new VectorTile(new Protobuf(fs.readFileSync(path.resolve(__dirname, '../../../test/unit/assets/mbsv5-6-18-23.vector.pbf')))); +const feature = vt.layers.place_label.feature(10); + +/*eslint new-cap: 0*/ +const collisionBoxArray = new CollisionBoxArray(); +const transform = new Transform(); +transform.width = 100; +transform.height = 100; +transform.cameraToCenterDistance = 100; + +const stacks = {'Test': glyphs} as any as { + [_: string]: { + [x: number]: StyleGlyph; + }; +}; + +function bucketSetup(text = 'abcde') { + return createSymbolBucket('test', 'Test', text, collisionBoxArray); +} + +function createIndexedFeature(id, index, iconId) { + return { + feature: { + extent: 8192, + type: 1, + id, + properties: { + icon: iconId + }, + loadGeometry () { + return [[{x: 0, y: 0}]]; + } + }, + id, + index, + sourceLayerIndex: 0 + }; +} + +describe('SymbolBucket', () => { + test('SymbolBucket', () => { + const bucketA = bucketSetup() as any as SymbolBucket; + const bucketB = bucketSetup() as any as SymbolBucket; + const options = {iconDependencies: {}, glyphDependencies: {}} as PopulateParameters; + const placement = new Placement(transform, undefined as any, 0, true); + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + const crossTileSymbolIndex = new CrossTileSymbolIndex(); + + // add feature from bucket A + bucketA.populate([{feature} as IndexedFeature], options, undefined as any); + performSymbolLayout( + { + bucket: bucketA, + glyphMap: stacks, + glyphPositions: {} + } as any); + const tileA = new Tile(tileID, 512); + tileA.latestFeatureIndex = new FeatureIndex(tileID); + tileA.buckets = {test: bucketA}; + tileA.collisionBoxArray = collisionBoxArray; + + // add same feature from bucket B + bucketB.populate([{feature} as IndexedFeature], options, undefined as any); + performSymbolLayout({ + bucket: bucketB, glyphMap: stacks, glyphPositions: {} + } as any); + const tileB = new Tile(tileID, 512); + tileB.buckets = {test: bucketB}; + tileB.collisionBoxArray = collisionBoxArray; + + crossTileSymbolIndex.addLayer(bucketA.layers[0], [tileA, tileB], undefined as any); + + const place = (layer, tile) => { + const parts = []; + placement.getBucketParts(parts, layer, tile, false); + for (const part of parts) { + placement.placeLayerBucketPart(part, {}, false); + } + }; + const a = placement.collisionIndex.grid.keysLength(); + place(bucketA.layers[0], tileA); + const b = placement.collisionIndex.grid.keysLength(); + expect(a).not.toBe(b); + + const a2 = placement.collisionIndex.grid.keysLength(); + place(bucketB.layers[0], tileB); + const b2 = placement.collisionIndex.grid.keysLength(); + expect(b2).toBe(a2); + }); + + test('SymbolBucket integer overflow', () => { + const spy = jest.spyOn(console, 'warn').mockImplementation(() => { }); + SymbolBucket.MAX_GLYPHS = 5; + + const bucket = bucketSetup() as any as SymbolBucket; + const options = {iconDependencies: {}, glyphDependencies: {}} as PopulateParameters; + + bucket.populate([{feature} as IndexedFeature], options, undefined as any); + const fakeGlyph = {rect: {w: 10, h: 10}, metrics: {left: 10, top: 10, advance: 10}}; + performSymbolLayout({ + bucket, + glyphMap: stacks, + glyphPositions: {'Test': {97: fakeGlyph, 98: fakeGlyph, 99: fakeGlyph, 100: fakeGlyph, 101: fakeGlyph, 102: fakeGlyph} as any} + } as any); + + expect(spy).toHaveBeenCalledTimes(1); + expect(spy.mock.calls[0][0].includes('Too many glyphs being rendered in a tile.')).toBeTruthy(); + }); + + test('SymbolBucket image undefined sdf', () => { + const spy = jest.spyOn(console, 'warn').mockImplementation(() => { }); + spy.mockReset(); + + const imageMap = { + a: { + data: new RGBAImage({width: 0, height: 0}) + }, + b: { + data: new RGBAImage({width: 0, height: 0}), + sdf: false + } + } as any as { [_: string]: StyleImage }; + const imagePos = { + a: new ImagePosition({x: 0, y: 0, w: 10, h: 10}, 1 as any as StyleImage), + b: new ImagePosition({x: 10, y: 0, w: 10, h: 10}, 1 as any as StyleImage) + }; + const bucket = createSymbolIconBucket('test', 'icon', collisionBoxArray) as any as SymbolBucket; + const options = {iconDependencies: {}, glyphDependencies: {}} as PopulateParameters; + + bucket.populate( + [ + createIndexedFeature(0, 0, 'a'), + createIndexedFeature(1, 1, 'b'), + createIndexedFeature(2, 2, 'a') + ] as any as IndexedFeature[], + options, undefined as any + ); + + const icons = options.iconDependencies as any; + expect(icons.a).toBe(true); + expect(icons.b).toBe(true); + + performSymbolLayout({ + bucket, imageMap, imagePositions: imagePos + } as any); + + // undefined SDF should be treated the same as false SDF - no warning raised + expect(spy).not.toHaveBeenCalledTimes(1); + }); + + test('SymbolBucket image mismatched sdf', () => { + const spy = jest.spyOn(console, 'warn').mockImplementation(() => { }); + spy.mockReset(); + + const imageMap = { + a: { + data: new RGBAImage({width: 0, height: 0}), + sdf: true + }, + b: { + data: new RGBAImage({width: 0, height: 0}), + sdf: false + } + } as any as { [_: string]: StyleImage }; + const imagePos = { + a: new ImagePosition({x: 0, y: 0, w: 10, h: 10}, 1 as any as StyleImage), + b: new ImagePosition({x: 10, y: 0, w: 10, h: 10}, 1 as any as StyleImage) + }; + const bucket = createSymbolIconBucket('test', 'icon', collisionBoxArray) as any as SymbolBucket; + const options = {iconDependencies: {}, glyphDependencies: {}} as PopulateParameters; + + bucket.populate( + [ + createIndexedFeature(0, 0, 'a'), + createIndexedFeature(1, 1, 'b'), + createIndexedFeature(2, 2, 'a') + ] as any as IndexedFeature[], + options, undefined as unknown as CanonicalTileID + ); + + const icons = options.iconDependencies as any; + expect(icons.a).toBe(true); + expect(icons.b).toBe(true); + + performSymbolLayout({bucket, imageMap, imagePositions: imagePos} as any); + + // true SDF and false SDF in same bucket should trigger warning + expect(spy).toHaveBeenCalledTimes(1); + }); + + test('SymbolBucket detects rtl text', () => { + const rtlBucket = bucketSetup('مرحبا'); + const ltrBucket = bucketSetup('hello'); + const options = {iconDependencies: {}, glyphDependencies: {}} as PopulateParameters; + rtlBucket.populate([{feature} as IndexedFeature], options, undefined as any); + ltrBucket.populate([{feature} as IndexedFeature], options, undefined as any); + + expect(rtlBucket.hasRTLText).toBeTruthy(); + expect(ltrBucket.hasRTLText).toBeFalsy(); + }); + + // Test to prevent symbol bucket with rtl from text being culled by worker serialization. + test('SymbolBucket with rtl text is NOT empty even though no symbol instances are created', () => { + const rtlBucket = bucketSetup('مرحبا'); + const options = {iconDependencies: {}, glyphDependencies: {}} as PopulateParameters; + rtlBucket.createArrays(); + rtlBucket.populate([{feature} as IndexedFeature], options, undefined as any); + + expect(rtlBucket.isEmpty()).toBeFalsy(); + expect(rtlBucket.symbolInstances).toHaveLength(0); + }); + + test('SymbolBucket detects rtl text mixed with ltr text', () => { + const mixedBucket = bucketSetup('مرحبا translates to hello'); + const options = {iconDependencies: {}, glyphDependencies: {}} as PopulateParameters; + mixedBucket.populate([{feature} as IndexedFeature], options, undefined as any); + + expect(mixedBucket.hasRTLText).toBeTruthy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/data/bucket/symbol_bucket.ts b/web/libraries/maplibre-gl/src/data/bucket/symbol_bucket.ts new file mode 100644 index 00000000..d6a9ff0e --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/bucket/symbol_bucket.ts @@ -0,0 +1,969 @@ +import { + symbolLayoutAttributes, + collisionVertexAttributes, + collisionBoxLayout, + dynamicLayoutAttributes, +} from './symbol_attributes'; + +import {SymbolLayoutArray, + SymbolDynamicLayoutArray, + SymbolOpacityArray, + CollisionBoxLayoutArray, + CollisionVertexArray, + PlacedSymbolArray, + SymbolInstanceArray, + GlyphOffsetArray, + SymbolLineVertexArray, + TextAnchorOffsetArray +} from '../array_types.g'; + +import Point from '@mapbox/point-geometry'; +import {SegmentVector} from '../segment'; +import {ProgramConfigurationSet} from '../program_configuration'; +import {TriangleIndexArray, LineIndexArray} from '../index_array_type'; +import {transformText} from '../../symbol/transform_text'; +import {mergeLines} from '../../symbol/merge_lines'; +import {allowsVerticalWritingMode, stringContainsRTLText} from '../../util/script_detection'; +import {WritingMode} from '../../symbol/shaping'; +import {loadGeometry} from '../load_geometry'; +import {toEvaluationFeature} from '../evaluation_feature'; +import mvt from '@mapbox/vector-tile'; +const vectorTileFeatureTypes = mvt.VectorTileFeature.types; +import {verticalizedCharacterMap} from '../../util/verticalize_punctuation'; +import {Anchor} from '../../symbol/anchor'; +import {getSizeData, MAX_PACKED_SIZE} from '../../symbol/symbol_size'; + +import {register} from '../../util/web_worker_transfer'; +import {EvaluationParameters} from '../../style/evaluation_parameters'; +import {Formatted, ResolvedImage} from '@maplibre/maplibre-gl-style-spec'; +import {plugin as globalRTLTextPlugin, getRTLTextPluginStatus} from '../../source/rtl_text_plugin'; +import {mat4} from 'gl-matrix'; +import {getOverlapMode} from '../../style/style_layer/overlap_mode'; +import type {CanonicalTileID} from '../../source/tile_id'; +import type { + Bucket, + BucketParameters, + IndexedFeature, + PopulateParameters +} from '../bucket'; +import type {CollisionBoxArray, CollisionBox, SymbolInstance} from '../array_types.g'; +import type {StructArray, StructArrayMember, ViewType} from '../../util/struct_array'; +import type {SymbolStyleLayer} from '../../style/style_layer/symbol_style_layer'; +import type {Context} from '../../gl/context'; +import type {IndexBuffer} from '../../gl/index_buffer'; +import type {VertexBuffer} from '../../gl/vertex_buffer'; +import type {SymbolQuad} from '../../symbol/quads'; +import type {SizeData} from '../../symbol/symbol_size'; +import type {FeatureStates} from '../../source/source_state'; +import type {ImagePosition} from '../../render/image_atlas'; +import type {VectorTileLayer} from '@mapbox/vector-tile'; + +export type SingleCollisionBox = { + x1: number; + y1: number; + x2: number; + y2: number; + anchorPointX: number; + anchorPointY: number; +}; + +export type CollisionArrays = { + textBox?: SingleCollisionBox; + verticalTextBox?: SingleCollisionBox; + iconBox?: SingleCollisionBox; + verticalIconBox?: SingleCollisionBox; + textFeatureIndex?: number; + verticalTextFeatureIndex?: number; + iconFeatureIndex?: number; + verticalIconFeatureIndex?: number; +}; + +export type SymbolFeature = { + sortKey: number | void; + text: Formatted | void; + icon: ResolvedImage; + index: number; + sourceLayerIndex: number; + geometry: Array>; + properties: any; + type: 'Unknown' | 'Point' | 'LineString' | 'Polygon'; + id?: any; +}; + +export type SortKeyRange = { + sortKey: number; + symbolInstanceStart: number; + symbolInstanceEnd: number; +}; + +// Opacity arrays are frequently updated but don't contain a lot of information, so we pack them +// tight. Each Uint32 is actually four duplicate Uint8s for the four corners of a glyph +// 7 bits are for the current opacity, and the lowest bit is the target opacity + +// actually defined in symbol_attributes.js +// const placementOpacityAttributes = [ +// { name: 'a_fade_opacity', components: 1, type: 'Uint32' } +// ]; +const shaderOpacityAttributes = [ + {name: 'a_fade_opacity', components: 1, type: 'Uint8' as ViewType, offset: 0} +]; + +function addVertex( + array: StructArray, + anchorX: number, + anchorY: number, + ox: number, + oy: number, + tx: number, + ty: number, + sizeVertex: number, + isSDF: boolean, + pixelOffsetX: number, + pixelOffsetY: number, + minFontScaleX: number, + minFontScaleY: number +) { + const aSizeX = sizeVertex ? Math.min(MAX_PACKED_SIZE, Math.round(sizeVertex[0])) : 0; + const aSizeY = sizeVertex ? Math.min(MAX_PACKED_SIZE, Math.round(sizeVertex[1])) : 0; + array.emplaceBack( + // a_pos_offset + anchorX, + anchorY, + Math.round(ox * 32), + Math.round(oy * 32), + + // a_data + tx, // x coordinate of symbol on glyph atlas texture + ty, // y coordinate of symbol on glyph atlas texture + (aSizeX << 1) + (isSDF ? 1 : 0), + aSizeY, + pixelOffsetX * 16, + pixelOffsetY * 16, + minFontScaleX * 256, + minFontScaleY * 256 + ); +} + +function addDynamicAttributes(dynamicLayoutVertexArray: StructArray, p: Point, angle: number) { + dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle); + dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle); + dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle); + dynamicLayoutVertexArray.emplaceBack(p.x, p.y, angle); +} + +function containsRTLText(formattedText: Formatted): boolean { + for (const section of formattedText.sections) { + if (stringContainsRTLText(section.text)) { + return true; + } + } + return false; +} + +export class SymbolBuffers { + layoutVertexArray: SymbolLayoutArray; + layoutVertexBuffer: VertexBuffer; + + indexArray: TriangleIndexArray; + indexBuffer: IndexBuffer; + + programConfigurations: ProgramConfigurationSet; + segments: SegmentVector; + + dynamicLayoutVertexArray: SymbolDynamicLayoutArray; + dynamicLayoutVertexBuffer: VertexBuffer; + + opacityVertexArray: SymbolOpacityArray; + opacityVertexBuffer: VertexBuffer; + hasVisibleVertices: boolean; + + collisionVertexArray: CollisionVertexArray; + collisionVertexBuffer: VertexBuffer; + + placedSymbolArray: PlacedSymbolArray; + + constructor(programConfigurations: ProgramConfigurationSet) { + this.layoutVertexArray = new SymbolLayoutArray(); + this.indexArray = new TriangleIndexArray(); + this.programConfigurations = programConfigurations; + this.segments = new SegmentVector(); + this.dynamicLayoutVertexArray = new SymbolDynamicLayoutArray(); + this.opacityVertexArray = new SymbolOpacityArray(); + this.hasVisibleVertices = false; + this.placedSymbolArray = new PlacedSymbolArray(); + } + + isEmpty() { + return this.layoutVertexArray.length === 0 && + this.indexArray.length === 0 && + this.dynamicLayoutVertexArray.length === 0 && + this.opacityVertexArray.length === 0; + } + + upload(context: Context, dynamicIndexBuffer: boolean, upload?: boolean, update?: boolean) { + if (this.isEmpty()) { + return; + } + + if (upload) { + this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, symbolLayoutAttributes.members); + this.indexBuffer = context.createIndexBuffer(this.indexArray, dynamicIndexBuffer); + this.dynamicLayoutVertexBuffer = context.createVertexBuffer(this.dynamicLayoutVertexArray, dynamicLayoutAttributes.members, true); + this.opacityVertexBuffer = context.createVertexBuffer(this.opacityVertexArray, shaderOpacityAttributes, true); + // This is a performance hack so that we can write to opacityVertexArray with uint32s + // even though the shaders read uint8s + this.opacityVertexBuffer.itemSize = 1; + } + if (upload || update) { + this.programConfigurations.upload(context); + } + } + + destroy() { + if (!this.layoutVertexBuffer) return; + this.layoutVertexBuffer.destroy(); + this.indexBuffer.destroy(); + this.programConfigurations.destroy(); + this.segments.destroy(); + this.dynamicLayoutVertexBuffer.destroy(); + this.opacityVertexBuffer.destroy(); + } +} + +register('SymbolBuffers', SymbolBuffers); + +class CollisionBuffers { + layoutVertexArray: StructArray; + layoutAttributes: Array; + layoutVertexBuffer: VertexBuffer; + + indexArray: TriangleIndexArray | LineIndexArray; + indexBuffer: IndexBuffer; + + segments: SegmentVector; + + collisionVertexArray: CollisionVertexArray; + collisionVertexBuffer: VertexBuffer; + + constructor(LayoutArray: { + new (...args: any): StructArray; + }, + layoutAttributes: Array, + IndexArray: { + new (...args: any): TriangleIndexArray | LineIndexArray; + }) { + this.layoutVertexArray = new LayoutArray(); + this.layoutAttributes = layoutAttributes; + this.indexArray = new IndexArray(); + this.segments = new SegmentVector(); + this.collisionVertexArray = new CollisionVertexArray(); + } + + upload(context: Context) { + this.layoutVertexBuffer = context.createVertexBuffer(this.layoutVertexArray, this.layoutAttributes); + this.indexBuffer = context.createIndexBuffer(this.indexArray); + this.collisionVertexBuffer = context.createVertexBuffer(this.collisionVertexArray, collisionVertexAttributes.members, true); + } + + destroy() { + if (!this.layoutVertexBuffer) return; + this.layoutVertexBuffer.destroy(); + this.indexBuffer.destroy(); + this.segments.destroy(); + this.collisionVertexBuffer.destroy(); + } +} + +register('CollisionBuffers', CollisionBuffers); + +/** + * @internal + * Unlike other buckets, which simply implement #addFeature with type-specific + * logic for (essentially) triangulating feature geometries, SymbolBucket + * requires specialized behavior: + * + * 1. WorkerTile#parse(), the logical owner of the bucket creation process, + * calls SymbolBucket#populate(), which resolves text and icon tokens on + * each feature, adds each glyphs and symbols needed to the passed-in + * collections options.glyphDependencies and options.iconDependencies, and + * stores the feature data for use in subsequent step (this.features). + * + * 2. WorkerTile asynchronously requests from the main thread all of the glyphs + * and icons needed (by this bucket and any others). When glyphs and icons + * have been received, the WorkerTile creates a CollisionIndex and invokes: + * + * 3. performSymbolLayout(bucket, stacks, icons) perform texts shaping and + * layout on a Symbol Bucket. This step populates: + * `this.symbolInstances`: metadata on generated symbols + * `this.collisionBoxArray`: collision data for use by foreground + * `this.text`: SymbolBuffers for text symbols + * `this.icons`: SymbolBuffers for icons + * `this.iconCollisionBox`: Debug SymbolBuffers for icon collision boxes + * `this.textCollisionBox`: Debug SymbolBuffers for text collision boxes + * The results are sent to the foreground for rendering + * + * 4. performSymbolPlacement(bucket, collisionIndex) is run on the foreground, + * and uses the CollisionIndex along with current camera settings to determine + * which symbols can actually show on the map. Collided symbols are hidden + * using a dynamic "OpacityVertexArray". + */ +export class SymbolBucket implements Bucket { + static MAX_GLYPHS: number; + static addDynamicAttributes: typeof addDynamicAttributes; + + collisionBoxArray: CollisionBoxArray; + zoom: number; + overscaling: number; + layers: Array; + layerIds: Array; + stateDependentLayers: Array; + stateDependentLayerIds: Array; + + index: number; + sdfIcons: boolean; + iconsInText: boolean; + iconsNeedLinear: boolean; + bucketInstanceId: number; + justReloaded: boolean; + hasPattern: boolean; + + textSizeData: SizeData; + iconSizeData: SizeData; + + glyphOffsetArray: GlyphOffsetArray; + lineVertexArray: SymbolLineVertexArray; + features: Array; + symbolInstances: SymbolInstanceArray; + textAnchorOffsets: TextAnchorOffsetArray; + collisionArrays: Array; + sortKeyRanges: Array; + pixelRatio: number; + tilePixelRatio: number; + compareText: {[_: string]: Array}; + fadeStartTime: number; + sortFeaturesByKey: boolean; + sortFeaturesByY: boolean; + canOverlap: boolean; + sortedAngle: number; + featureSortOrder: Array; + + collisionCircleArray: Array; + placementInvProjMatrix: mat4; + placementViewportMatrix: mat4; + + text: SymbolBuffers; + icon: SymbolBuffers; + textCollisionBox: CollisionBuffers; + iconCollisionBox: CollisionBuffers; + uploaded: boolean; + sourceLayerIndex: number; + sourceID: string; + symbolInstanceIndexes: Array; + writingModes: WritingMode[]; + allowVerticalPlacement: boolean; + hasRTLText: boolean; + + constructor(options: BucketParameters) { + this.collisionBoxArray = options.collisionBoxArray; + this.zoom = options.zoom; + this.overscaling = options.overscaling; + this.layers = options.layers; + this.layerIds = this.layers.map(layer => layer.id); + this.index = options.index; + this.pixelRatio = options.pixelRatio; + this.sourceLayerIndex = options.sourceLayerIndex; + this.hasPattern = false; + this.hasRTLText = false; + this.sortKeyRanges = []; + + this.collisionCircleArray = []; + this.placementInvProjMatrix = mat4.identity([] as any); + this.placementViewportMatrix = mat4.identity([] as any); + + const layer = this.layers[0]; + const unevaluatedLayoutValues = layer._unevaluatedLayout._values; + + this.textSizeData = getSizeData(this.zoom, unevaluatedLayoutValues['text-size']); + this.iconSizeData = getSizeData(this.zoom, unevaluatedLayoutValues['icon-size']); + + const layout = this.layers[0].layout; + const sortKey = layout.get('symbol-sort-key'); + const zOrder = layout.get('symbol-z-order'); + this.canOverlap = + getOverlapMode(layout, 'text-overlap', 'text-allow-overlap') !== 'never' || + getOverlapMode(layout, 'icon-overlap', 'icon-allow-overlap') !== 'never' || + layout.get('text-ignore-placement') || + layout.get('icon-ignore-placement'); + this.sortFeaturesByKey = zOrder !== 'viewport-y' && !sortKey.isConstant(); + const zOrderByViewportY = zOrder === 'viewport-y' || (zOrder === 'auto' && !this.sortFeaturesByKey); + this.sortFeaturesByY = zOrderByViewportY && this.canOverlap; + + if (layout.get('symbol-placement') === 'point') { + this.writingModes = layout.get('text-writing-mode').map(wm => WritingMode[wm]); + } + + this.stateDependentLayerIds = this.layers.filter((l) => l.isStateDependent()).map((l) => l.id); + + this.sourceID = options.sourceID; + } + + createArrays() { + this.text = new SymbolBuffers(new ProgramConfigurationSet(this.layers, this.zoom, property => /^text/.test(property))); + this.icon = new SymbolBuffers(new ProgramConfigurationSet(this.layers, this.zoom, property => /^icon/.test(property))); + + this.glyphOffsetArray = new GlyphOffsetArray(); + this.lineVertexArray = new SymbolLineVertexArray(); + this.symbolInstances = new SymbolInstanceArray(); + this.textAnchorOffsets = new TextAnchorOffsetArray(); + } + + calculateGlyphDependencies(text: string, stack: {[_: number]: boolean}, textAlongLine: boolean, allowVerticalPlacement: boolean, doesAllowVerticalWritingMode: boolean) { + for (let i = 0; i < text.length; i++) { + stack[text.charCodeAt(i)] = true; + if ((textAlongLine || allowVerticalPlacement) && doesAllowVerticalWritingMode) { + const verticalChar = verticalizedCharacterMap[text.charAt(i)]; + if (verticalChar) { + stack[verticalChar.charCodeAt(0)] = true; + } + } + } + } + + populate(features: Array, options: PopulateParameters, canonical: CanonicalTileID) { + const layer = this.layers[0]; + const layout = layer.layout; + + const textFont = layout.get('text-font'); + const textField = layout.get('text-field'); + const iconImage = layout.get('icon-image'); + const hasText = + (textField.value.kind !== 'constant' || + (textField.value.value instanceof Formatted && !textField.value.value.isEmpty()) || + textField.value.value.toString().length > 0) && + (textFont.value.kind !== 'constant' || textFont.value.value.length > 0); + // we should always resolve the icon-image value if the property was defined in the style + // this allows us to fire the styleimagemissing event if image evaluation returns null + // the only way to distinguish between null returned from a coalesce statement with no valid images + // and null returned because icon-image wasn't defined is to check whether or not iconImage.parameters is an empty object + const hasIcon = iconImage.value.kind !== 'constant' || !!iconImage.value.value || Object.keys(iconImage.parameters).length > 0; + const symbolSortKey = layout.get('symbol-sort-key'); + + this.features = []; + + if (!hasText && !hasIcon) { + return; + } + + const icons = options.iconDependencies; + const stacks = options.glyphDependencies; + const availableImages = options.availableImages; + const globalProperties = new EvaluationParameters(this.zoom); + + for (const {feature, id, index, sourceLayerIndex} of features) { + + const needGeometry = layer._featureFilter.needGeometry; + const evaluationFeature = toEvaluationFeature(feature, needGeometry); + if (!layer._featureFilter.filter(globalProperties, evaluationFeature, canonical)) { + continue; + } + + if (!needGeometry) evaluationFeature.geometry = loadGeometry(feature); + + let text: Formatted | void; + if (hasText) { + // Expression evaluation will automatically coerce to Formatted + // but plain string token evaluation skips that pathway so do the + // conversion here. + const resolvedTokens = layer.getValueAndResolveTokens('text-field', evaluationFeature, canonical, availableImages); + const formattedText = Formatted.factory(resolvedTokens); + if (containsRTLText(formattedText)) { + this.hasRTLText = true; + } + if ( + !this.hasRTLText || // non-rtl text so can proceed safely + getRTLTextPluginStatus() === 'unavailable' || // We don't intend to lazy-load the rtl text plugin, so proceed with incorrect shaping + this.hasRTLText && globalRTLTextPlugin.isParsed() // Use the rtlText plugin to shape text + ) { + text = transformText(formattedText, layer, evaluationFeature); + } + } + + let icon: ResolvedImage; + if (hasIcon) { + // Expression evaluation will automatically coerce to Image + // but plain string token evaluation skips that pathway so do the + // conversion here. + const resolvedTokens = layer.getValueAndResolveTokens('icon-image', evaluationFeature, canonical, availableImages); + if (resolvedTokens instanceof ResolvedImage) { + icon = resolvedTokens; + } else { + icon = ResolvedImage.fromString(resolvedTokens); + } + } + + if (!text && !icon) { + continue; + } + const sortKey = this.sortFeaturesByKey ? + symbolSortKey.evaluate(evaluationFeature, {}, canonical) : + undefined; + + const symbolFeature: SymbolFeature = { + id, + text, + icon, + index, + sourceLayerIndex, + geometry: evaluationFeature.geometry, + properties: feature.properties, + type: vectorTileFeatureTypes[feature.type], + sortKey + }; + this.features.push(symbolFeature); + + if (icon) { + icons[icon.name] = true; + } + + if (text) { + const fontStack = textFont.evaluate(evaluationFeature, {}, canonical).join(','); + const textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point'; + this.allowVerticalPlacement = this.writingModes && this.writingModes.indexOf(WritingMode.vertical) >= 0; + for (const section of text.sections) { + if (!section.image) { + const doesAllowVerticalWritingMode = allowsVerticalWritingMode(text.toString()); + const sectionFont = section.fontStack || fontStack; + const sectionStack = stacks[sectionFont] = stacks[sectionFont] || {}; + this.calculateGlyphDependencies(section.text, sectionStack, textAlongLine, this.allowVerticalPlacement, doesAllowVerticalWritingMode); + } else { + // Add section image to the list of dependencies. + icons[section.image.name] = true; + } + } + } + } + + if (layout.get('symbol-placement') === 'line') { + // Merge adjacent lines with the same text to improve labelling. + // It's better to place labels on one long line than on many short segments. + this.features = mergeLines(this.features); + } + + if (this.sortFeaturesByKey) { + this.features.sort((a, b) => { + // a.sortKey is always a number when sortFeaturesByKey is true + return (a.sortKey as number) - (b.sortKey as number); + }); + } + } + + update(states: FeatureStates, vtLayer: VectorTileLayer, imagePositions: {[_: string]: ImagePosition}) { + if (!this.stateDependentLayers.length) return; + this.text.programConfigurations.updatePaintArrays(states, vtLayer, this.layers, imagePositions); + this.icon.programConfigurations.updatePaintArrays(states, vtLayer, this.layers, imagePositions); + } + + isEmpty() { + // When the bucket encounters only rtl-text but the plugin isnt loaded, no symbol instances will be created. + // In order for the bucket to be serialized, and not discarded as an empty bucket both checks are necessary. + return this.symbolInstances.length === 0 && !this.hasRTLText; + } + + uploadPending() { + return !this.uploaded || this.text.programConfigurations.needsUpload || this.icon.programConfigurations.needsUpload; + } + + upload(context: Context) { + if (!this.uploaded && this.hasDebugData()) { + this.textCollisionBox.upload(context); + this.iconCollisionBox.upload(context); + } + this.text.upload(context, this.sortFeaturesByY, !this.uploaded, this.text.programConfigurations.needsUpload); + this.icon.upload(context, this.sortFeaturesByY, !this.uploaded, this.icon.programConfigurations.needsUpload); + this.uploaded = true; + } + + destroyDebugData() { + this.textCollisionBox.destroy(); + this.iconCollisionBox.destroy(); + } + + destroy() { + this.text.destroy(); + this.icon.destroy(); + + if (this.hasDebugData()) { + this.destroyDebugData(); + } + } + + addToLineVertexArray(anchor: Anchor, line: any) { + const lineStartIndex = this.lineVertexArray.length; + if (anchor.segment !== undefined) { + let sumForwardLength = anchor.dist(line[anchor.segment + 1]); + let sumBackwardLength = anchor.dist(line[anchor.segment]); + const vertices = {}; + for (let i = anchor.segment + 1; i < line.length; i++) { + vertices[i] = {x: line[i].x, y: line[i].y, tileUnitDistanceFromAnchor: sumForwardLength}; + if (i < line.length - 1) { + sumForwardLength += line[i + 1].dist(line[i]); + } + } + for (let i = anchor.segment || 0; i >= 0; i--) { + vertices[i] = {x: line[i].x, y: line[i].y, tileUnitDistanceFromAnchor: sumBackwardLength}; + if (i > 0) { + sumBackwardLength += line[i - 1].dist(line[i]); + } + } + for (let i = 0; i < line.length; i++) { + const vertex = vertices[i]; + this.lineVertexArray.emplaceBack(vertex.x, vertex.y, vertex.tileUnitDistanceFromAnchor); + } + } + return { + lineStartIndex, + lineLength: this.lineVertexArray.length - lineStartIndex + }; + } + + addSymbols(arrays: SymbolBuffers, + quads: Array, + sizeVertex: any, + lineOffset: [number, number], + alongLine: boolean, + feature: SymbolFeature, + writingMode: WritingMode, + labelAnchor: Anchor, + lineStartIndex: number, + lineLength: number, + associatedIconIndex: number, + canonical: CanonicalTileID) { + const indexArray = arrays.indexArray; + const layoutVertexArray = arrays.layoutVertexArray; + + const segment = arrays.segments.prepareSegment(4 * quads.length, layoutVertexArray, indexArray, this.canOverlap ? feature.sortKey as number : undefined); + const glyphOffsetArrayStart = this.glyphOffsetArray.length; + const vertexStartIndex = segment.vertexLength; + + const angle = (this.allowVerticalPlacement && writingMode === WritingMode.vertical) ? Math.PI / 2 : 0; + + const sections = feature.text && feature.text.sections; + + for (let i = 0; i < quads.length; i++) { + const {tl, tr, bl, br, tex, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY, glyphOffset, isSDF, sectionIndex} = quads[i]; + const index = segment.vertexLength; + + const y = glyphOffset[1]; + addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, tl.x, y + tl.y, tex.x, tex.y, sizeVertex, isSDF, pixelOffsetTL.x, pixelOffsetTL.y, minFontScaleX, minFontScaleY); + addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, tr.x, y + tr.y, tex.x + tex.w, tex.y, sizeVertex, isSDF, pixelOffsetBR.x, pixelOffsetTL.y, minFontScaleX, minFontScaleY); + addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, bl.x, y + bl.y, tex.x, tex.y + tex.h, sizeVertex, isSDF, pixelOffsetTL.x, pixelOffsetBR.y, minFontScaleX, minFontScaleY); + addVertex(layoutVertexArray, labelAnchor.x, labelAnchor.y, br.x, y + br.y, tex.x + tex.w, tex.y + tex.h, sizeVertex, isSDF, pixelOffsetBR.x, pixelOffsetBR.y, minFontScaleX, minFontScaleY); + + addDynamicAttributes(arrays.dynamicLayoutVertexArray, labelAnchor, angle); + + indexArray.emplaceBack(index, index + 1, index + 2); + indexArray.emplaceBack(index + 1, index + 2, index + 3); + + segment.vertexLength += 4; + segment.primitiveLength += 2; + + this.glyphOffsetArray.emplaceBack(glyphOffset[0]); + + if (i === quads.length - 1 || sectionIndex !== quads[i + 1].sectionIndex) { + arrays.programConfigurations.populatePaintArrays(layoutVertexArray.length, feature, feature.index, {}, canonical, sections && sections[sectionIndex]); + } + } + + arrays.placedSymbolArray.emplaceBack( + labelAnchor.x, labelAnchor.y, + glyphOffsetArrayStart, + this.glyphOffsetArray.length - glyphOffsetArrayStart, + vertexStartIndex, + lineStartIndex, + lineLength, + labelAnchor.segment, + sizeVertex ? sizeVertex[0] : 0, + sizeVertex ? sizeVertex[1] : 0, + lineOffset[0], lineOffset[1], + writingMode, + // placedOrientation is null initially; will be updated to horizontal(1)/vertical(2) if placed + 0, + false as unknown as number, + // The crossTileID is only filled/used on the foreground for dynamic text anchors + 0, + associatedIconIndex + ); + } + + _addCollisionDebugVertex(layoutVertexArray: StructArray, collisionVertexArray: StructArray, point: Point, anchorX: number, anchorY: number, extrude: Point) { + collisionVertexArray.emplaceBack(0, 0); + return layoutVertexArray.emplaceBack( + // pos + point.x, + point.y, + // a_anchor_pos + anchorX, + anchorY, + // extrude + Math.round(extrude.x), + Math.round(extrude.y)); + } + + addCollisionDebugVertices(x1: number, y1: number, x2: number, y2: number, arrays: CollisionBuffers, boxAnchorPoint: Point, symbolInstance: SymbolInstance) { + const segment = arrays.segments.prepareSegment(4, arrays.layoutVertexArray, arrays.indexArray); + const index = segment.vertexLength; + + const layoutVertexArray = arrays.layoutVertexArray; + const collisionVertexArray = arrays.collisionVertexArray; + + const anchorX = symbolInstance.anchorX; + const anchorY = symbolInstance.anchorY; + + this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x1, y1)); + this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x2, y1)); + this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x2, y2)); + this._addCollisionDebugVertex(layoutVertexArray, collisionVertexArray, boxAnchorPoint, anchorX, anchorY, new Point(x1, y2)); + + segment.vertexLength += 4; + + const indexArray = arrays.indexArray as LineIndexArray; + indexArray.emplaceBack(index, index + 1); + indexArray.emplaceBack(index + 1, index + 2); + indexArray.emplaceBack(index + 2, index + 3); + indexArray.emplaceBack(index + 3, index); + + segment.primitiveLength += 4; + } + + addDebugCollisionBoxes(startIndex: number, endIndex: number, symbolInstance: SymbolInstance, isText: boolean) { + for (let b = startIndex; b < endIndex; b++) { + const box: CollisionBox = this.collisionBoxArray.get(b); + const x1 = box.x1; + const y1 = box.y1; + const x2 = box.x2; + const y2 = box.y2; + + this.addCollisionDebugVertices(x1, y1, x2, y2, + isText ? this.textCollisionBox : this.iconCollisionBox, + box.anchorPoint, symbolInstance); + } + } + + generateCollisionDebugBuffers() { + if (this.hasDebugData()) { + this.destroyDebugData(); + } + + this.textCollisionBox = new CollisionBuffers(CollisionBoxLayoutArray, collisionBoxLayout.members, LineIndexArray); + this.iconCollisionBox = new CollisionBuffers(CollisionBoxLayoutArray, collisionBoxLayout.members, LineIndexArray); + + for (let i = 0; i < this.symbolInstances.length; i++) { + const symbolInstance = this.symbolInstances.get(i); + this.addDebugCollisionBoxes(symbolInstance.textBoxStartIndex, symbolInstance.textBoxEndIndex, symbolInstance, true); + this.addDebugCollisionBoxes(symbolInstance.verticalTextBoxStartIndex, symbolInstance.verticalTextBoxEndIndex, symbolInstance, true); + this.addDebugCollisionBoxes(symbolInstance.iconBoxStartIndex, symbolInstance.iconBoxEndIndex, symbolInstance, false); + this.addDebugCollisionBoxes(symbolInstance.verticalIconBoxStartIndex, symbolInstance.verticalIconBoxEndIndex, symbolInstance, false); + } + } + + // These flat arrays are meant to be quicker to iterate over than the source + // CollisionBoxArray + _deserializeCollisionBoxesForSymbol( + collisionBoxArray: CollisionBoxArray, + textStartIndex: number, + textEndIndex: number, + verticalTextStartIndex: number, + verticalTextEndIndex: number, + iconStartIndex: number, + iconEndIndex: number, + verticalIconStartIndex: number, + verticalIconEndIndex: number + ): CollisionArrays { + + const collisionArrays = {} as CollisionArrays; + for (let k = textStartIndex; k < textEndIndex; k++) { + const box: CollisionBox = collisionBoxArray.get(k); + collisionArrays.textBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY}; + collisionArrays.textFeatureIndex = box.featureIndex; + break; // Only one box allowed per instance + } + for (let k = verticalTextStartIndex; k < verticalTextEndIndex; k++) { + const box: CollisionBox = collisionBoxArray.get(k); + collisionArrays.verticalTextBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY}; + collisionArrays.verticalTextFeatureIndex = box.featureIndex; + break; // Only one box allowed per instance + } + for (let k = iconStartIndex; k < iconEndIndex; k++) { + // An icon can only have one box now, so this indexing is a bit vestigial... + const box: CollisionBox = collisionBoxArray.get(k); + collisionArrays.iconBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY}; + collisionArrays.iconFeatureIndex = box.featureIndex; + break; // Only one box allowed per instance + } + for (let k = verticalIconStartIndex; k < verticalIconEndIndex; k++) { + // An icon can only have one box now, so this indexing is a bit vestigial... + const box: CollisionBox = collisionBoxArray.get(k); + collisionArrays.verticalIconBox = {x1: box.x1, y1: box.y1, x2: box.x2, y2: box.y2, anchorPointX: box.anchorPointX, anchorPointY: box.anchorPointY}; + collisionArrays.verticalIconFeatureIndex = box.featureIndex; + break; // Only one box allowed per instance + } + return collisionArrays; + } + + deserializeCollisionBoxes(collisionBoxArray: CollisionBoxArray) { + this.collisionArrays = []; + for (let i = 0; i < this.symbolInstances.length; i++) { + const symbolInstance = this.symbolInstances.get(i); + this.collisionArrays.push(this._deserializeCollisionBoxesForSymbol( + collisionBoxArray, + symbolInstance.textBoxStartIndex, + symbolInstance.textBoxEndIndex, + symbolInstance.verticalTextBoxStartIndex, + symbolInstance.verticalTextBoxEndIndex, + symbolInstance.iconBoxStartIndex, + symbolInstance.iconBoxEndIndex, + symbolInstance.verticalIconBoxStartIndex, + symbolInstance.verticalIconBoxEndIndex + )); + } + } + + hasTextData() { + return this.text.segments.get().length > 0; + } + + hasIconData() { + return this.icon.segments.get().length > 0; + } + + hasDebugData() { + return this.textCollisionBox && this.iconCollisionBox; + } + + hasTextCollisionBoxData() { + return this.hasDebugData() && this.textCollisionBox.segments.get().length > 0; + } + + hasIconCollisionBoxData() { + return this.hasDebugData() && this.iconCollisionBox.segments.get().length > 0; + } + + addIndicesForPlacedSymbol(iconOrText: SymbolBuffers, placedSymbolIndex: number) { + const placedSymbol = iconOrText.placedSymbolArray.get(placedSymbolIndex); + + const endIndex = placedSymbol.vertexStartIndex + placedSymbol.numGlyphs * 4; + for (let vertexIndex = placedSymbol.vertexStartIndex; vertexIndex < endIndex; vertexIndex += 4) { + iconOrText.indexArray.emplaceBack(vertexIndex, vertexIndex + 1, vertexIndex + 2); + iconOrText.indexArray.emplaceBack(vertexIndex + 1, vertexIndex + 2, vertexIndex + 3); + } + } + + getSortedSymbolIndexes(angle: number) { + if (this.sortedAngle === angle && this.symbolInstanceIndexes !== undefined) { + return this.symbolInstanceIndexes; + } + const sin = Math.sin(angle); + const cos = Math.cos(angle); + const rotatedYs = []; + const featureIndexes = []; + const result = []; + + for (let i = 0; i < this.symbolInstances.length; ++i) { + result.push(i); + const symbolInstance = this.symbolInstances.get(i); + rotatedYs.push(Math.round(sin * symbolInstance.anchorX + cos * symbolInstance.anchorY) | 0); + featureIndexes.push(symbolInstance.featureIndex); + } + + result.sort((aIndex, bIndex) => { + return (rotatedYs[aIndex] - rotatedYs[bIndex]) || + (featureIndexes[bIndex] - featureIndexes[aIndex]); + }); + + return result; + } + + addToSortKeyRanges(symbolInstanceIndex: number, sortKey: number) { + const last = this.sortKeyRanges[this.sortKeyRanges.length - 1]; + if (last && last.sortKey === sortKey) { + last.symbolInstanceEnd = symbolInstanceIndex + 1; + } else { + this.sortKeyRanges.push({ + sortKey, + symbolInstanceStart: symbolInstanceIndex, + symbolInstanceEnd: symbolInstanceIndex + 1 + }); + } + } + + sortFeatures(angle: number) { + if (!this.sortFeaturesByY) return; + if (this.sortedAngle === angle) return; + + // The current approach to sorting doesn't sort across segments so don't try. + // Sorting within segments separately seemed not to be worth the complexity. + if (this.text.segments.get().length > 1 || this.icon.segments.get().length > 1) return; + + // If the symbols are allowed to overlap sort them by their vertical screen position. + // The index array buffer is rewritten to reference the (unchanged) vertices in the + // sorted order. + + // To avoid sorting the actual symbolInstance array we sort an array of indexes. + this.symbolInstanceIndexes = this.getSortedSymbolIndexes(angle); + this.sortedAngle = angle; + + this.text.indexArray.clear(); + this.icon.indexArray.clear(); + + this.featureSortOrder = []; + + for (const i of this.symbolInstanceIndexes) { + const symbolInstance = this.symbolInstances.get(i); + this.featureSortOrder.push(symbolInstance.featureIndex); + + [ + symbolInstance.rightJustifiedTextSymbolIndex, + symbolInstance.centerJustifiedTextSymbolIndex, + symbolInstance.leftJustifiedTextSymbolIndex + ].forEach((index, i, array) => { + // Only add a given index the first time it shows up, + // to avoid duplicate opacity entries when multiple justifications + // share the same glyphs. + if (index >= 0 && array.indexOf(index) === i) { + this.addIndicesForPlacedSymbol(this.text, index); + } + }); + + if (symbolInstance.verticalPlacedTextSymbolIndex >= 0) { + this.addIndicesForPlacedSymbol(this.text, symbolInstance.verticalPlacedTextSymbolIndex); + } + + if (symbolInstance.placedIconSymbolIndex >= 0) { + this.addIndicesForPlacedSymbol(this.icon, symbolInstance.placedIconSymbolIndex); + } + + if (symbolInstance.verticalPlacedIconSymbolIndex >= 0) { + this.addIndicesForPlacedSymbol(this.icon, symbolInstance.verticalPlacedIconSymbolIndex); + } + } + + if (this.text.indexBuffer) this.text.indexBuffer.updateData(this.text.indexArray); + if (this.icon.indexBuffer) this.icon.indexBuffer.updateData(this.icon.indexArray); + } +} + +register('SymbolBucket', SymbolBucket, { + omit: ['layers', 'collisionBoxArray', 'features', 'compareText'] +}); + +// this constant is based on the size of StructArray indexes used in a symbol +// bucket--namely, glyphOffsetArrayStart +// eg the max valid UInt16 is 65,535 +// See https://github.com/mapbox/mapbox-gl-js/issues/2907 for motivation +// lineStartIndex and textBoxStartIndex could potentially be concerns +// but we expect there to be many fewer boxes/lines than glyphs +SymbolBucket.MAX_GLYPHS = 65535; + +SymbolBucket.addDynamicAttributes = addDynamicAttributes; + +export {addDynamicAttributes}; diff --git a/web/libraries/maplibre-gl/src/data/dem_data.test.ts b/web/libraries/maplibre-gl/src/data/dem_data.test.ts new file mode 100644 index 00000000..d3c67f87 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/dem_data.test.ts @@ -0,0 +1,237 @@ +import {DEMData} from './dem_data'; +import {RGBAImage} from '../util/image'; +import {serialize, deserialize} from '../util/web_worker_transfer'; + +function createMockImage(height, width) { + // RGBAImage passed to constructor has uniform 1px padding on all sides. + height += 2; + width += 2; + const pixels = new Uint8Array(height * width * 4); + for (let i = 0; i < pixels.length; i++) { + pixels[i] = (i + 1) % 4 === 0 ? 1 : Math.floor(Math.random() * 256); + } + return new RGBAImage({height, width}, pixels); +} + +function createMockClampImage(height, width) { + const pixels = new Uint8ClampedArray(height * width * 4); + for (let i = 0; i < pixels.length; i++) { + pixels[i] = (i + 1) % 4 === 0 ? 1 : Math.floor(Math.random() * 256); + } + return new RGBAImage({height, width}, pixels); +} + +describe('DEMData', () => { + describe('constructor', () => { + test('Uint8Array', () => { + const imageData0 = createMockImage(4, 4); + const dem = new DEMData('0', imageData0, 'mapbox'); + expect(dem.uid).toBe('0'); + expect(dem.dim).toBe(4); + expect(dem.stride).toBe(6); + }); + + test('Uint8ClampedArray', () => { + const imageData0 = createMockClampImage(4, 4); + const dem = new DEMData('0', imageData0, 'mapbox'); + expect(dem).not.toBeNull(); + expect(dem['uid']).toBe('0'); + expect(dem['dim']).toBe(2); + expect(dem['stride']).toBe(4); + }); + + test('otherEncoding', () => { + const spyOnWarnConsole = jest.spyOn(console, 'warn').mockImplementation(); + + const imageData0 = createMockImage(4, 4); + new DEMData('0', imageData0, 'otherEncoding' as any); + + expect(spyOnWarnConsole).toHaveBeenCalledTimes(1); + expect(spyOnWarnConsole.mock.calls).toEqual([['\"otherEncoding\" is not a valid encoding type. Valid types include \"mapbox\", \"terrarium\" and \"custom\".']]); + }); + }); +}); + +function testDEMBorderRegion(dem: DEMData) { + return () => { + let nonempty = true; + for (let x = -1; x < 5; x++) { + for (let y = -1; y < 5; y++) { + if (dem.get(x, y) === -65536) { + nonempty = false; + break; + } + } + } + expect(nonempty).toBeTruthy(); + + let verticalBorderMatch = true; + for (const x of [-1, 4]) { + for (let y = 0; y < 4; y++) { + if (dem.get(x, y) !== dem.get(x < 0 ? x + 1 : x - 1, y)) { + verticalBorderMatch = false; + break; + } + } + } + expect(verticalBorderMatch).toBeTruthy(); + + // horizontal borders empty + let horizontalBorderMatch = true; + for (const y of [-1, 4]) { + for (let x = 0; x < 4; x++) { + if (dem.get(x, y) !== dem.get(x, y < 0 ? y + 1 : y - 1)) { + horizontalBorderMatch = false; + break; + } + } + } + expect(horizontalBorderMatch).toBeTruthy(); + + expect(dem.get(-1, 4) === dem.get(0, 3)).toBeTruthy(); + expect(dem.get(4, 4) === dem.get(3, 3)).toBeTruthy(); + expect(dem.get(-1, -1) === dem.get(0, 0)).toBeTruthy(); + expect(dem.get(4, -1) === dem.get(3, 0)).toBeTruthy(); + }; +} + +function testDEMBackfill(dem0: DEMData, dem1: DEMData) { + return () => { + dem0.backfillBorder(dem1, -1, 0); + for (let y = 0; y < 4; y++) { + // dx = -1, dy = 0, so the left edge of dem1 should equal the right edge of dem0 + expect(dem0.get(-1, y) === dem1.get(3, y)).toBeTruthy(); + } + + dem0.backfillBorder(dem1, 0, -1); + for (let x = 0; x < 4; x++) { + expect(dem0.get(x, -1) === dem1.get(x, 3)).toBeTruthy(); + } + + dem0.backfillBorder(dem1, 1, 0); + for (let y = 0; y < 4; y++) { + expect(dem0.get(4, y) === dem1.get(0, y)).toBeTruthy(); + } + + dem0.backfillBorder(dem1, 0, 1); + for (let x = 0; x < 4; x++) { + expect(dem0.get(x, 4) === dem1.get(x, 0)).toBeTruthy(); + } + + dem0.backfillBorder(dem1, -1, 1); + expect(dem0.get(-1, 4) === dem1.get(3, 0)).toBeTruthy(); + + dem0.backfillBorder(dem1, 1, 1); + expect(dem0.get(4, 4) === dem1.get(0, 0)).toBeTruthy(); + + dem0.backfillBorder(dem1, -1, -1); + expect(dem0.get(-1, -1) === dem1.get(3, 3)).toBeTruthy(); + + dem0.backfillBorder(dem1, 1, -1); + expect(dem0.get(4, -1) === dem1.get(0, 3)).toBeTruthy(); + }; +} + +describe('DEMData#backfillBorder with encoding', () => { + describe('mabox encoding', () => { + const dem0 = new DEMData('0', createMockImage(4, 4), 'mapbox'); + const dem1 = new DEMData('1', createMockImage(4, 4), 'mapbox'); + + test('border region is initially populated with neighboring data', testDEMBorderRegion(dem0)); + test('backfillBorder correctly populates borders with neighboring data', testDEMBackfill(dem0, dem1)); + }); + + describe('terrarium encoding', () => { + const dem0 = new DEMData('0', createMockImage(4, 4), 'terrarium'); + const dem1 = new DEMData('1', createMockImage(4, 4), 'terrarium'); + + test('border region is initially populated with neighboring data', testDEMBorderRegion(dem0)); + test('backfillBorder correctly populates borders with neighboring data', testDEMBackfill(dem0, dem1)); + }); +}); + +function testSerialization(dem0: DEMData, redFactor: number, greenFactor: number, blueFactor: number, baseShift: number) { + return () => { + const serialized = serialize(dem0); + + // calculate min/max values + let min = Number.MAX_SAFE_INTEGER; + let max = Number.MIN_SAFE_INTEGER; + for (let x = 0; x < 4; x++) { + for (let y = 0; y < 4; y++) { + const ele = dem0.get(x, y); + if (ele > max) max = ele; + if (ele < min) min = ele; + } + } + + expect(serialized).toEqual({ + $name: 'DEMData', + uid: '0', + dim: 4, + stride: 6, + data: dem0.data, + redFactor, + greenFactor, + blueFactor, + baseShift, + max, + min, + }); + + const transferrables = []; + serialize(dem0, transferrables); + expect(new Uint32Array(transferrables[0])).toEqual(dem0.data); + }; +} + +function testDeserialization(dem0: DEMData) { + return () => { + const serialized = serialize(dem0); + + const deserialized = deserialize(serialized); + expect(deserialized).toEqual(dem0); + }; +} + +describe('DEMData is correctly serialized and deserialized', () => { + const mapboxDEM = new DEMData('0', createMockImage(4, 4), 'mapbox'); + const terrariumDEM = new DEMData('0', createMockImage(4, 4), 'terrarium'); + const customDEM = new DEMData('0', createMockImage(4, 4), 'custom', 1.0, 2.0, 3.0, 4.0); + test('serialized - mapbox', testSerialization(mapboxDEM, 6553.6, 25.6, 0.1, 10000)); + test('serialized - terrarium', testSerialization(terrariumDEM, 256.0, 1.0, 1.0 / 256.0, 32768.0)); + test('serialized - custom', testSerialization(customDEM, 1.0, 2.0, 3.0, 4.0)); + + test('deserialized - mapbox', testDeserialization(mapboxDEM)); + test('deserialized - terrarium', testDeserialization(terrariumDEM)); + test('deserialized - custom', testDeserialization(customDEM)); +}); + +describe('UnpackVector is correctly returned', () => { + test('terrarium, mapbox and custom', () => { + const mapboxDEM = new DEMData('0', createMockImage(4, 4), 'mapbox'); + const terrariumDEM = new DEMData('0', createMockImage(4, 4), 'terrarium'); + const customDEM = new DEMData('0', createMockImage(4, 4), 'custom', 1.0, 2.0, 3.0, 4.0); + + expect(terrariumDEM.getUnpackVector()).toEqual([256.0, 1.0, 1.0 / 256.0, 32768.0]); + expect(mapboxDEM.getUnpackVector()).toEqual([6553.6, 25.6, 0.1, 10000.0]); + expect(customDEM.getUnpackVector()).toEqual([1.0, 2.0, 3.0, 4.0]); + }); +}); + +function testGetPixels(dem: DEMData, imageData: RGBAImage) { + return () => { + expect(dem.getPixels()).toEqual(imageData); + }; +} + +describe('DEMData#getImage', () => { + const imageData = createMockImage(4, 4); + const mapboxDEM = new DEMData('0', imageData, 'terrarium'); + const terrariumDEM = new DEMData('0', imageData, 'terrarium'); + const customDEM = new DEMData('0', imageData, 'terrarium'); + + test('Image is correctly returned - mapbox', testGetPixels(mapboxDEM, imageData)); + test('Image is correctly returned - terrarium', testGetPixels(terrariumDEM, imageData)); + test('Image is correctly returned - custom', testGetPixels(customDEM, imageData)); +}); diff --git a/web/libraries/maplibre-gl/src/data/dem_data.ts b/web/libraries/maplibre-gl/src/data/dem_data.ts new file mode 100644 index 00000000..b95be92b --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/dem_data.ts @@ -0,0 +1,158 @@ +import {RGBAImage} from '../util/image'; + +import {warnOnce} from '../util/util'; +import {register} from '../util/web_worker_transfer'; + +// DEMData is a data structure for decoding, backfilling, and storing elevation data for processing in the hillshade shaders +// data can be populated either from a pngraw image tile or from serliazed data sent back from a worker. When data is initially +// loaded from a image tile, we decode the pixel values using the appropriate decoding formula, but we store the +// elevation data as an Int32 value. we add 65536 (2^16) to eliminate negative values and enable the use of +// integer overflow when creating the texture used in the hillshadePrepare step. + +// DEMData also handles the backfilling of data from a tile's neighboring tiles. This is necessary because we use a pixel's 8 +// surrounding pixel values to compute the slope at that pixel, and we cannot accurately calculate the slope at pixels on a +// tile's edge without backfilling from neighboring tiles. + +export type DEMEncoding = 'mapbox' | 'terrarium' | 'custom' + +export class DEMData { + uid: string; + data: Uint32Array; + stride: number; + dim: number; + min: number; + max: number; + redFactor: number; + greenFactor: number; + blueFactor: number; + baseShift: number; + + // RGBAImage data has uniform 1px padding on all sides: square tile edge size defines stride + // and dim is calculated as stride - 2. + constructor(uid: string, data: RGBAImage, encoding: DEMEncoding, redFactor = 1.0, greenFactor = 1.0, blueFactor = 1.0, baseShift = 0.0) { + this.uid = uid; + if (data.height !== data.width) throw new RangeError('DEM tiles must be square'); + if (encoding && !['mapbox', 'terrarium', 'custom'].includes(encoding)) { + warnOnce(`"${encoding}" is not a valid encoding type. Valid types include "mapbox", "terrarium" and "custom".`); + return; + } + this.stride = data.height; + const dim = this.dim = data.height - 2; + this.data = new Uint32Array(data.data.buffer); + switch (encoding) { + case 'terrarium': + // unpacking formula for mapzen terrarium: + // https://aws.amazon.com/public-datasets/terrain/ + this.redFactor = 256.0; + this.greenFactor = 1.0; + this.blueFactor = 1.0 / 256.0; + this.baseShift = 32768.0; + break; + case 'custom': + this.redFactor = redFactor; + this.greenFactor = greenFactor; + this.blueFactor = blueFactor; + this.baseShift = baseShift; + break; + case 'mapbox': + default: + // unpacking formula for mapbox.terrain-rgb: + // https://www.mapbox.com/help/access-elevation-data/#mapbox-terrain-rgb + this.redFactor = 6553.6; + this.greenFactor = 25.6; + this.blueFactor = 0.1; + this.baseShift = 10000.0; + break; + } + + // in order to avoid flashing seams between tiles, here we are initially populating a 1px border of pixels around the image + // with the data of the nearest pixel from the image. this data is eventually replaced when the tile's neighboring + // tiles are loaded and the accurate data can be backfilled using DEMData#backfillBorder + for (let x = 0; x < dim; x++) { + // left vertical border + this.data[this._idx(-1, x)] = this.data[this._idx(0, x)]; + // right vertical border + this.data[this._idx(dim, x)] = this.data[this._idx(dim - 1, x)]; + // left horizontal border + this.data[this._idx(x, -1)] = this.data[this._idx(x, 0)]; + // right horizontal border + this.data[this._idx(x, dim)] = this.data[this._idx(x, dim - 1)]; + } + // corners + this.data[this._idx(-1, -1)] = this.data[this._idx(0, 0)]; + this.data[this._idx(dim, -1)] = this.data[this._idx(dim - 1, 0)]; + this.data[this._idx(-1, dim)] = this.data[this._idx(0, dim - 1)]; + this.data[this._idx(dim, dim)] = this.data[this._idx(dim - 1, dim - 1)]; + + // calculate min/max values + this.min = Number.MAX_SAFE_INTEGER; + this.max = Number.MIN_SAFE_INTEGER; + for (let x = 0; x < dim; x++) { + for (let y = 0; y < dim; y++) { + const ele = this.get(x, y); + if (ele > this.max) this.max = ele; + if (ele < this.min) this.min = ele; + } + } + } + + get(x: number, y: number) { + const pixels = new Uint8Array(this.data.buffer); + const index = this._idx(x, y) * 4; + return this.unpack(pixels[index], pixels[index + 1], pixels[index + 2]); + } + + getUnpackVector() { + return [this.redFactor, this.greenFactor, this.blueFactor, this.baseShift]; + } + + _idx(x: number, y: number) { + if (x < -1 || x >= this.dim + 1 || y < -1 || y >= this.dim + 1) throw new RangeError('out of range source coordinates for DEM data'); + return (y + 1) * this.stride + (x + 1); + } + + unpack(r: number, g: number, b: number) { + return (r * this.redFactor + g * this.greenFactor + b * this.blueFactor - this.baseShift); + } + + getPixels() { + return new RGBAImage({width: this.stride, height: this.stride}, new Uint8Array(this.data.buffer)); + } + + backfillBorder(borderTile: DEMData, dx: number, dy: number) { + if (this.dim !== borderTile.dim) throw new Error('dem dimension mismatch'); + + let xMin = dx * this.dim, + xMax = dx * this.dim + this.dim, + yMin = dy * this.dim, + yMax = dy * this.dim + this.dim; + + switch (dx) { + case -1: + xMin = xMax - 1; + break; + case 1: + xMax = xMin + 1; + break; + } + + switch (dy) { + case -1: + yMin = yMax - 1; + break; + case 1: + yMax = yMin + 1; + break; + } + + const ox = -dx * this.dim; + const oy = -dy * this.dim; + for (let y = yMin; y < yMax; y++) { + for (let x = xMin; x < xMax; x++) { + this.data[this._idx(x, y)] = borderTile.data[this._idx(x + ox, y + oy)]; + } + } + } +} + +register('DEMData', DEMData); diff --git a/web/libraries/maplibre-gl/src/data/evaluation_feature.ts b/web/libraries/maplibre-gl/src/data/evaluation_feature.ts new file mode 100644 index 00000000..6594a714 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/evaluation_feature.ts @@ -0,0 +1,18 @@ +import {loadGeometry} from './load_geometry'; +import type Point from '@mapbox/point-geometry'; +import type {VectorTileFeature} from '@mapbox/vector-tile'; +import type {Feature} from '@maplibre/maplibre-gl-style-spec'; + +type EvaluationFeature = Feature & { geometry: Array> }; +/** + * Construct a new feature based on a VectorTileFeature for expression evaluation, the geometry of which + * will be loaded based on necessity. + * @param feature - the feature to evaluate + * @param needGeometry - if set to true this will load the geometry + */ +export function toEvaluationFeature(feature: VectorTileFeature, needGeometry: boolean): EvaluationFeature { + return {type: feature.type, + id: feature.id, + properties: feature.properties, + geometry: needGeometry ? loadGeometry(feature) : []}; +} diff --git a/web/libraries/maplibre-gl/src/data/extent.ts b/web/libraries/maplibre-gl/src/data/extent.ts new file mode 100644 index 00000000..7e92a5e5 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/extent.ts @@ -0,0 +1,13 @@ +/** + * The maximum value of a coordinate in the internal tile coordinate system. Coordinates of + * all source features normalized to this extent upon load. + * + * The value is a consequence of the following: + * + * * Vertex buffer store positions as signed 16 bit integers. + * * One bit is lost for signedness to support tile buffers. + * * One bit is lost because the line vertex buffer used to pack 1 bit of other data into the int. + * * One bit is lost to support features extending past the extent on the right edge of the tile. + * * This leaves us with 2^13 = 8192 + */ +export const EXTENT = 8192; diff --git a/web/libraries/maplibre-gl/src/data/feature_index.ts b/web/libraries/maplibre-gl/src/data/feature_index.ts new file mode 100644 index 00000000..1d8ea9df --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/feature_index.ts @@ -0,0 +1,338 @@ +import Point from '@mapbox/point-geometry'; +import {loadGeometry} from './load_geometry'; +import {toEvaluationFeature} from './evaluation_feature'; +import {EXTENT} from './extent'; +import {featureFilter} from '@maplibre/maplibre-gl-style-spec'; +import {TransferableGridIndex} from '../util/transferable_grid_index'; +import {DictionaryCoder} from '../util/dictionary_coder'; +import vt from '@mapbox/vector-tile'; +import Protobuf from 'pbf'; +import {GeoJSONFeature} from '../util/vectortile_to_geojson'; +import type {MapGeoJSONFeature} from '../util/vectortile_to_geojson'; +import {arraysIntersect, mapObject, extend} from '../util/util'; +import {OverscaledTileID} from '../source/tile_id'; +import {register} from '../util/web_worker_transfer'; +import {EvaluationParameters} from '../style/evaluation_parameters'; +import {SourceFeatureState} from '../source/source_state'; +import {polygonIntersectsBox} from '../util/intersection_tests'; +import {PossiblyEvaluated} from '../style/properties'; +import {FeatureIndexArray} from './array_types.g'; +import {mat4} from 'gl-matrix'; + +import type {StyleLayer} from '../style/style_layer'; +import type {FeatureFilter, FeatureState, FilterSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {Transform} from '../geo/transform'; +import type {VectorTileFeature, VectorTileLayer} from '@mapbox/vector-tile'; + +type QueryParameters = { + scale: number; + pixelPosMatrix: mat4; + transform: Transform; + tileSize: number; + queryGeometry: Array; + cameraQueryGeometry: Array; + queryPadding: number; + params: { + filter: FilterSpecification; + layers: Array; + availableImages: Array; + }; +}; + +/** + * @internal + * An in memory index class to allow fast interaction with features + */ +export class FeatureIndex { + tileID: OverscaledTileID; + x: number; + y: number; + z: number; + grid: TransferableGridIndex; + grid3D: TransferableGridIndex; + featureIndexArray: FeatureIndexArray; + promoteId?: PromoteIdSpecification; + + rawTileData: ArrayBuffer; + bucketLayerIDs: Array>; + + vtLayers: {[_: string]: VectorTileLayer}; + sourceLayerCoder: DictionaryCoder; + + constructor(tileID: OverscaledTileID, promoteId?: PromoteIdSpecification | null) { + this.tileID = tileID; + this.x = tileID.canonical.x; + this.y = tileID.canonical.y; + this.z = tileID.canonical.z; + this.grid = new TransferableGridIndex(EXTENT, 16, 0); + this.grid3D = new TransferableGridIndex(EXTENT, 16, 0); + this.featureIndexArray = new FeatureIndexArray(); + this.promoteId = promoteId; + } + + insert(feature: VectorTileFeature, geometry: Array>, featureIndex: number, sourceLayerIndex: number, bucketIndex: number, is3D?: boolean) { + const key = this.featureIndexArray.length; + this.featureIndexArray.emplaceBack(featureIndex, sourceLayerIndex, bucketIndex); + + const grid = is3D ? this.grid3D : this.grid; + + for (let r = 0; r < geometry.length; r++) { + const ring = geometry[r]; + + const bbox = [Infinity, Infinity, -Infinity, -Infinity]; + for (let i = 0; i < ring.length; i++) { + const p = ring[i]; + bbox[0] = Math.min(bbox[0], p.x); + bbox[1] = Math.min(bbox[1], p.y); + bbox[2] = Math.max(bbox[2], p.x); + bbox[3] = Math.max(bbox[3], p.y); + } + + if (bbox[0] < EXTENT && + bbox[1] < EXTENT && + bbox[2] >= 0 && + bbox[3] >= 0) { + grid.insert(key, bbox[0], bbox[1], bbox[2], bbox[3]); + } + } + } + + loadVTLayers(): {[_: string]: VectorTileLayer} { + if (!this.vtLayers) { + this.vtLayers = new vt.VectorTile(new Protobuf(this.rawTileData)).layers; + this.sourceLayerCoder = new DictionaryCoder(this.vtLayers ? Object.keys(this.vtLayers).sort() : ['_geojsonTileLayer']); + } + return this.vtLayers; + } + + // Finds non-symbol features in this tile at a particular position. + query( + args: QueryParameters, + styleLayers: {[_: string]: StyleLayer}, + serializedLayers: {[_: string]: any}, + sourceFeatureState: SourceFeatureState + ): {[_: string]: Array<{featureIndex: number; feature: GeoJSONFeature}>} { + this.loadVTLayers(); + + const params = args.params || {} as { filter: any; layers: string[]; availableImages: string[] }, + pixelsToTileUnits = EXTENT / args.tileSize / args.scale, + filter = featureFilter(params.filter); + + const queryGeometry = args.queryGeometry; + const queryPadding = args.queryPadding * pixelsToTileUnits; + + const bounds = getBounds(queryGeometry); + const matching = this.grid.query(bounds.minX - queryPadding, bounds.minY - queryPadding, bounds.maxX + queryPadding, bounds.maxY + queryPadding); + + const cameraBounds = getBounds(args.cameraQueryGeometry); + const matching3D = this.grid3D.query( + cameraBounds.minX - queryPadding, cameraBounds.minY - queryPadding, cameraBounds.maxX + queryPadding, cameraBounds.maxY + queryPadding, + (bx1, by1, bx2, by2) => { + return polygonIntersectsBox(args.cameraQueryGeometry, bx1 - queryPadding, by1 - queryPadding, bx2 + queryPadding, by2 + queryPadding); + }); + + for (const key of matching3D) { + matching.push(key); + } + + matching.sort(topDownFeatureComparator); + + const result = {}; + let previousIndex; + for (let k = 0; k < matching.length; k++) { + const index = matching[k]; + + // don't check the same feature more than once + if (index === previousIndex) continue; + previousIndex = index; + + const match = this.featureIndexArray.get(index); + let featureGeometry = null; + this.loadMatchingFeature( + result, + match.bucketIndex, + match.sourceLayerIndex, + match.featureIndex, + filter, + params.layers, + params.availableImages, + styleLayers, + serializedLayers, + sourceFeatureState, + (feature: VectorTileFeature, styleLayer: StyleLayer, featureState: FeatureState) => { + if (!featureGeometry) { + featureGeometry = loadGeometry(feature); + } + + return styleLayer.queryIntersectsFeature(queryGeometry, feature, featureState, featureGeometry, this.z, args.transform, pixelsToTileUnits, args.pixelPosMatrix); + } + ); + } + + return result; + } + + loadMatchingFeature( + result: { + [_: string]: Array<{ + featureIndex: number; + feature: GeoJSONFeature; + intersectionZ?: boolean | number; + }>; + }, + bucketIndex: number, + sourceLayerIndex: number, + featureIndex: number, + filter: FeatureFilter, + filterLayerIDs: Array, + availableImages: Array, + styleLayers: {[_: string]: StyleLayer}, + serializedLayers: {[_: string]: any}, + sourceFeatureState?: SourceFeatureState, + intersectionTest?: ( + feature: VectorTileFeature, + styleLayer: StyleLayer, + featureState: any, + id: string | number | void + ) => boolean | number) { + + const layerIDs = this.bucketLayerIDs[bucketIndex]; + if (filterLayerIDs && !arraysIntersect(filterLayerIDs, layerIDs)) + return; + + const sourceLayerName = this.sourceLayerCoder.decode(sourceLayerIndex); + const sourceLayer = this.vtLayers[sourceLayerName]; + const feature = sourceLayer.feature(featureIndex); + + if (filter.needGeometry) { + const evaluationFeature = toEvaluationFeature(feature, true); + if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), evaluationFeature, this.tileID.canonical)) { + return; + } + } else if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), feature)) { + return; + } + + const id = this.getId(feature, sourceLayerName); + + for (let l = 0; l < layerIDs.length; l++) { + const layerID = layerIDs[l]; + + if (filterLayerIDs && filterLayerIDs.indexOf(layerID) < 0) { + continue; + } + + const styleLayer = styleLayers[layerID]; + + if (!styleLayer) continue; + + let featureState = {}; + if (id && sourceFeatureState) { + // `feature-state` expression evaluation requires feature state to be available + featureState = sourceFeatureState.getState(styleLayer.sourceLayer || '_geojsonTileLayer', id); + } + + const serializedLayer = extend({}, serializedLayers[layerID]); + + serializedLayer.paint = evaluateProperties(serializedLayer.paint, styleLayer.paint, feature, featureState, availableImages); + serializedLayer.layout = evaluateProperties(serializedLayer.layout, styleLayer.layout, feature, featureState, availableImages); + + const intersectionZ = !intersectionTest || intersectionTest(feature, styleLayer, featureState); + if (!intersectionZ) { + // Only applied for non-symbol features + continue; + } + + const geojsonFeature = new GeoJSONFeature(feature, this.z, this.x, this.y, id) as MapGeoJSONFeature; + geojsonFeature.layer = serializedLayer; + let layerResult = result[layerID]; + if (layerResult === undefined) { + layerResult = result[layerID] = []; + } + layerResult.push({featureIndex, feature: geojsonFeature, intersectionZ}); + } + } + + // Given a set of symbol indexes that have already been looked up, + // return a matching set of GeoJSONFeatures + lookupSymbolFeatures(symbolFeatureIndexes: Array, + serializedLayers: {[_: string]: StyleLayer}, + bucketIndex: number, + sourceLayerIndex: number, + filterSpec: FilterSpecification, + filterLayerIDs: Array, + availableImages: Array, + styleLayers: {[_: string]: StyleLayer}) { + const result = {}; + this.loadVTLayers(); + + const filter = featureFilter(filterSpec); + + for (const symbolFeatureIndex of symbolFeatureIndexes) { + this.loadMatchingFeature( + result, + bucketIndex, + sourceLayerIndex, + symbolFeatureIndex, + filter, + filterLayerIDs, + availableImages, + styleLayers, + serializedLayers + ); + + } + return result; + } + + hasLayer(id: string) { + for (const layerIDs of this.bucketLayerIDs) { + for (const layerID of layerIDs) { + if (id === layerID) return true; + } + } + + return false; + } + + getId(feature: VectorTileFeature, sourceLayerId: string): string | number { + let id: string | number = feature.id; + if (this.promoteId) { + const propName = typeof this.promoteId === 'string' ? this.promoteId : this.promoteId[sourceLayerId]; + id = feature.properties[propName] as string | number; + if (typeof id === 'boolean') id = Number(id); + } + return id; + } +} + +register( + 'FeatureIndex', + FeatureIndex, + {omit: ['rawTileData', 'sourceLayerCoder']} +); + +function evaluateProperties(serializedProperties, styleLayerProperties, feature, featureState, availableImages) { + return mapObject(serializedProperties, (property, key) => { + const prop = styleLayerProperties instanceof PossiblyEvaluated ? styleLayerProperties.get(key) : null; + return prop && prop.evaluate ? prop.evaluate(feature, featureState, availableImages) : prop; + }); +} + +function getBounds(geometry: Array) { + let minX = Infinity; + let minY = Infinity; + let maxX = -Infinity; + let maxY = -Infinity; + for (const p of geometry) { + minX = Math.min(minX, p.x); + minY = Math.min(minY, p.y); + maxX = Math.max(maxX, p.x); + maxY = Math.max(maxY, p.y); + } + return {minX, minY, maxX, maxY}; +} + +function topDownFeatureComparator(a, b) { + return b - a; +} diff --git a/web/libraries/maplibre-gl/src/data/feature_position_map.test.ts b/web/libraries/maplibre-gl/src/data/feature_position_map.test.ts new file mode 100644 index 00000000..31a1cff7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/feature_position_map.test.ts @@ -0,0 +1,33 @@ +import {FeaturePositionMap} from './feature_position_map'; +import {serialize, deserialize} from '../util/web_worker_transfer'; + +describe('FeaturePositionMap', () => { + test('Can be queried after serialization/deserialization', () => { + const featureMap = new FeaturePositionMap(); + featureMap.add(7, 1, 0, 1); + featureMap.add(3, 2, 1, 2); + featureMap.add(7, 3, 2, 3); + featureMap.add(4, 4, 3, 4); + featureMap.add(2, 5, 4, 5); + featureMap.add(7, 6, 5, 7); + + const featureMap2 = deserialize(serialize(featureMap, [])) as FeaturePositionMap; + + const compareIndex = (a, b) => a.index - b.index; + + expect(featureMap2.getPositions(7).sort(compareIndex)).toEqual([ + {index: 1, start: 0, end: 1}, + {index: 3, start: 2, end: 3}, + {index: 6, start: 5, end: 7} + ].sort(compareIndex)); + }); + + test('Can not be queried before serialization/deserialization', () => { + const featureMap = new FeaturePositionMap(); + featureMap.add(0, 1, 2, 3); + + expect(() => { + featureMap.getPositions(0); + }).toThrow(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/data/feature_position_map.ts b/web/libraries/maplibre-gl/src/data/feature_position_map.ts new file mode 100644 index 00000000..b5d39b2a --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/feature_position_map.ts @@ -0,0 +1,126 @@ +import murmur3 from 'murmurhash-js'; +import {register} from '../util/web_worker_transfer'; + +type SerializedFeaturePositionMap = { + ids: Float64Array; + positions: Uint32Array; +}; + +type FeaturePosition = { + index: number; + start: number; + end: number; +}; + +// A transferable data structure that maps feature ids to their indices and buffer offsets +export class FeaturePositionMap { + ids: Array; + positions: Array; + indexed: boolean; + + constructor() { + this.ids = []; + this.positions = []; + this.indexed = false; + } + + add(id: unknown, index: number, start: number, end: number) { + this.ids.push(getNumericId(id)); + this.positions.push(index, start, end); + } + + getPositions(id: unknown): Array { + if (!this.indexed) throw new Error('Trying to get index, but feature positions are not indexed'); + + const intId = getNumericId(id); + + // binary search for the first occurrence of id in this.ids; + // relies on ids/positions being sorted by id, which happens in serialization + let i = 0; + let j = this.ids.length - 1; + while (i < j) { + const m = (i + j) >> 1; + if (this.ids[m] >= intId) { + j = m; + } else { + i = m + 1; + } + } + const positions = []; + while (this.ids[i] === intId) { + const index = this.positions[3 * i]; + const start = this.positions[3 * i + 1]; + const end = this.positions[3 * i + 2]; + positions.push({index, start, end}); + i++; + } + return positions; + } + + static serialize(map: FeaturePositionMap, transferables: Array): SerializedFeaturePositionMap { + const ids = new Float64Array(map.ids); + const positions = new Uint32Array(map.positions); + + sort(ids, positions, 0, ids.length - 1); + + if (transferables) { + transferables.push(ids.buffer, positions.buffer); + } + + return {ids, positions}; + } + + static deserialize(obj: SerializedFeaturePositionMap): FeaturePositionMap { + const map = new FeaturePositionMap(); + // after transferring, we only use these arrays statically (no pushes), + // so TypedArray vs Array distinction that flow points out doesn't matter + map.ids = (obj.ids as any); + map.positions = (obj.positions as any); + map.indexed = true; + return map; + } +} + +function getNumericId(value: unknown) { + const numValue = +value; + if (!isNaN(numValue) && numValue <= Number.MAX_SAFE_INTEGER) { + return numValue; + } + return murmur3(String(value)); +} + +// custom quicksort that sorts ids, indices and offsets together (by ids) +// uses Hoare partitioning & manual tail call optimization to avoid worst case scenarios +function sort(ids, positions, left, right) { + while (left < right) { + const pivot = ids[(left + right) >> 1]; + let i = left - 1; + let j = right + 1; + + while (true) { + do i++; while (ids[i] < pivot); + do j--; while (ids[j] > pivot); + if (i >= j) break; + swap(ids, i, j); + swap(positions, 3 * i, 3 * j); + swap(positions, 3 * i + 1, 3 * j + 1); + swap(positions, 3 * i + 2, 3 * j + 2); + } + + if (j - left < right - j) { + sort(ids, positions, left, j); + left = j + 1; + } else { + sort(ids, positions, j + 1, right); + right = j; + } + } +} + +function swap(arr, i, j) { + const tmp = arr[i]; + arr[i] = arr[j]; + arr[j] = tmp; +} + +register('FeaturePositionMap', FeaturePositionMap); diff --git a/web/libraries/maplibre-gl/src/data/index_array_type.ts b/web/libraries/maplibre-gl/src/data/index_array_type.ts new file mode 100644 index 00000000..9d224219 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/index_array_type.ts @@ -0,0 +1,9 @@ +import {LineIndexArray, TriangleIndexArray, LineStripIndexArray} from './array_types.g'; + +/** + * An index array stores Uint16 indices of vertexes in a corresponding vertex array. We use + * three kinds of index arrays: arrays storing groups of three indices, forming triangles; + * arrays storing pairs of indices, forming line segments; and arrays storing single indices, + * forming a line strip. + */ +export {LineIndexArray, TriangleIndexArray, LineStripIndexArray}; diff --git a/web/libraries/maplibre-gl/src/data/load_geometry.test.ts b/web/libraries/maplibre-gl/src/data/load_geometry.test.ts new file mode 100644 index 00000000..23e61f19 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/load_geometry.test.ts @@ -0,0 +1,49 @@ +import fs from 'fs'; +import path from 'path'; +import Protobuf from 'pbf'; +import {VectorTile} from '@mapbox/vector-tile'; +import {loadGeometry} from './load_geometry'; + +// Load a line feature from fixture tile. +const vt = new VectorTile(new Protobuf(fs.readFileSync(path.resolve(__dirname, '../../test/unit/assets/mbsv5-6-18-23.vector.pbf')))); + +describe('loadGeometry', () => { + test('loadGeometry', () => { + const feature = vt.layers.road.feature(0); + const originalGeometry = feature.loadGeometry(); + const scaledGeometry = loadGeometry(feature); + expect(scaledGeometry[0][0].x).toBe(originalGeometry[0][0].x * 2); + expect(scaledGeometry[0][0].y).toBe(originalGeometry[0][0].y * 2); + }); + + test('loadGeometry warns and clamps when exceeding extent', () => { + const feature = vt.layers.road.feature(0); + feature.extent = 2048; + + let numWarnings = 0; + + // Use a custom console.warn to count warnings + const warn = console.warn; + console.warn = function(warning) { + if (warning.match(/Geometry exceeds allowed extent, reduce your vector tile buffer size/)) { + numWarnings++; + } + }; + + const lines = loadGeometry(feature); + + expect(numWarnings).toBe(1); + + let maxValue = -Infinity; + for (const line of lines) { + for (const {x, y} of line) { + maxValue = Math.max(x, y, maxValue); + } + } + expect(maxValue).toBe(16383); + + // Put it back + console.warn = warn; + + }); +}); diff --git a/web/libraries/maplibre-gl/src/data/load_geometry.ts b/web/libraries/maplibre-gl/src/data/load_geometry.ts new file mode 100644 index 00000000..199585be --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/load_geometry.ts @@ -0,0 +1,44 @@ +import {warnOnce, clamp} from '../util/util'; + +import {EXTENT} from './extent'; + +import type Point from '@mapbox/point-geometry'; +import type {VectorTileFeature} from '@mapbox/vector-tile'; + +// These bounds define the minimum and maximum supported coordinate values. +// While visible coordinates are within [0, EXTENT], tiles may theoretically +// contain coordinates within [-Infinity, Infinity]. Our range is limited by the +// number of bits used to represent the coordinate. +const BITS = 15; +const MAX = Math.pow(2, BITS - 1) - 1; +const MIN = -MAX - 1; + +/** + * Loads a geometry from a VectorTileFeature and scales it to the common extent + * used internally. + * @param feature - the vector tile feature to load + */ +export function loadGeometry(feature: VectorTileFeature): Array> { + const scale = EXTENT / feature.extent; + const geometry = feature.loadGeometry(); + for (let r = 0; r < geometry.length; r++) { + const ring = geometry[r]; + for (let p = 0; p < ring.length; p++) { + const point = ring[p]; + // round here because mapbox-gl-native uses integers to represent + // points and we need to do the same to avoid renering differences. + const x = Math.round(point.x * scale); + const y = Math.round(point.y * scale); + + point.x = clamp(x, MIN, MAX); + point.y = clamp(y, MIN, MAX); + + if (x < point.x || x > point.x + 1 || y < point.y || y > point.y + 1) { + // warn when exceeding allowed extent except for the 1-px-off case + // https://github.com/mapbox/mapbox-gl-js/issues/8992 + warnOnce('Geometry exceeds allowed extent, reduce your vector tile buffer size'); + } + } + } + return geometry; +} diff --git a/web/libraries/maplibre-gl/src/data/pos3d_attributes.ts b/web/libraries/maplibre-gl/src/data/pos3d_attributes.ts new file mode 100644 index 00000000..bbbcf996 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/pos3d_attributes.ts @@ -0,0 +1,5 @@ +import {createLayout} from '../util/struct_array'; + +export default createLayout([ + {name: 'a_pos3d', type: 'Int16', components: 3} +]); diff --git a/web/libraries/maplibre-gl/src/data/pos_attributes.ts b/web/libraries/maplibre-gl/src/data/pos_attributes.ts new file mode 100644 index 00000000..37d3cbb5 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/pos_attributes.ts @@ -0,0 +1,5 @@ +import {createLayout} from '../util/struct_array'; + +export default createLayout([ + {name: 'a_pos', type: 'Int16', components: 2} +]); diff --git a/web/libraries/maplibre-gl/src/data/program_configuration.ts b/web/libraries/maplibre-gl/src/data/program_configuration.ts new file mode 100644 index 00000000..e02653aa --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/program_configuration.ts @@ -0,0 +1,735 @@ +import {packUint8ToFloat} from '../shaders/encode_attribute'; +import {Color, supportsPropertyExpression} from '@maplibre/maplibre-gl-style-spec'; +import {register} from '../util/web_worker_transfer'; +import {PossiblyEvaluatedPropertyValue} from '../style/properties'; +import {StructArrayLayout1f4, StructArrayLayout2f8, StructArrayLayout4f16, PatternLayoutArray} from './array_types.g'; +import {clamp} from '../util/util'; +import {patternAttributes} from './bucket/pattern_attributes'; +import {EvaluationParameters} from '../style/evaluation_parameters'; +import {FeaturePositionMap} from './feature_position_map'; +import {Uniform, Uniform1f, UniformColor, Uniform4f} from '../render/uniform_binding'; + +import type {UniformLocations} from '../render/uniform_binding'; + +import type {CanonicalTileID} from '../source/tile_id'; +import type {Context} from '../gl/context'; +import type {TypedStyleLayer} from '../style/style_layer/typed_style_layer'; +import type {CrossfadeParameters} from '../style/evaluation_parameters'; +import type {StructArray, StructArrayMember} from '../util/struct_array'; +import type {VertexBuffer} from '../gl/vertex_buffer'; +import type {ImagePosition} from '../render/image_atlas'; +import type { + Feature, + FeatureState, + GlobalProperties, + SourceExpression, + CompositeExpression, + FormattedSection +} from '@maplibre/maplibre-gl-style-spec'; +import type {FeatureStates} from '../source/source_state'; +import type {VectorTileLayer} from '@mapbox/vector-tile'; + +export type BinderUniform = { + name: string; + property: string; + binding: Uniform; +}; + +function packColor(color: Color): [number, number] { + return [ + packUint8ToFloat(255 * color.r, 255 * color.g), + packUint8ToFloat(255 * color.b, 255 * color.a) + ]; +} + +/** + * `Binder` is the interface definition for the strategies for constructing, + * uploading, and binding paint property data as GLSL attributes. Most style- + * spec properties have a 1:1 relationship to shader attribute/uniforms, but + * some require multiple values per feature to be passed to the GPU, and in + * those cases we bind multiple attributes/uniforms. + * + * It has three implementations, one for each of the three strategies we use: + * + * * For _constant_ properties -- those whose value is a constant, or the constant + * result of evaluating a camera expression at a particular camera position -- we + * don't need a vertex attribute buffer, and instead use a uniform. + * * For data expressions, we use a vertex buffer with a single attribute value, + * the evaluated result of the source function for the given feature. + * * For composite expressions, we use a vertex buffer with two attributes: min and + * max values covering the range of zooms at which we expect the tile to be + * displayed. These values are calculated by evaluating the composite expression for + * the given feature at strategically chosen zoom levels. In addition to this + * attribute data, we also use a uniform value which the shader uses to interpolate + * between the min and max value at the final displayed zoom level. The use of a + * uniform allows us to cheaply update the value on every frame. + * + * Note that the shader source varies depending on whether we're using a uniform or + * attribute. We dynamically compile shaders at runtime to accommodate this. + */ +interface AttributeBinder { + populatePaintArray( + length: number, + feature: Feature, + imagePositions: {[_: string]: ImagePosition}, + canonical?: CanonicalTileID, + formattedSection?: FormattedSection + ): void; + updatePaintArray( + start: number, + length: number, + feature: Feature, + featureState: FeatureState, + imagePositions: {[_: string]: ImagePosition} + ): void; + upload(a: Context): void; + destroy(): void; +} + +interface UniformBinder { + uniformNames: Array; + setUniform( + uniform: Uniform, + globals: GlobalProperties, + currentValue: PossiblyEvaluatedPropertyValue, + uniformName: string + ): void; + getBinding(context: Context, location: WebGLUniformLocation, name: string): Partial>; +} + +class ConstantBinder implements UniformBinder { + value: unknown; + type: string; + uniformNames: Array; + + constructor(value: unknown, names: Array, type: string) { + this.value = value; + this.uniformNames = names.map(name => `u_${name}`); + this.type = type; + } + + setUniform( + uniform: Uniform, + globals: GlobalProperties, + currentValue: PossiblyEvaluatedPropertyValue + ): void { + uniform.set(currentValue.constantOr(this.value)); + } + + getBinding(context: Context, location: WebGLUniformLocation, _: string): Partial> { + return (this.type === 'color') ? + new UniformColor(context, location) : + new Uniform1f(context, location); + } +} + +class CrossFadedConstantBinder implements UniformBinder { + uniformNames: Array; + patternFrom: Array; + patternTo: Array; + pixelRatioFrom: number; + pixelRatioTo: number; + + constructor(value: unknown, names: Array) { + this.uniformNames = names.map(name => `u_${name}`); + this.patternFrom = null; + this.patternTo = null; + this.pixelRatioFrom = 1.0; + this.pixelRatioTo = 1.0; + } + + setConstantPatternPositions(posTo: ImagePosition, posFrom: ImagePosition) { + this.pixelRatioFrom = posFrom.pixelRatio; + this.pixelRatioTo = posTo.pixelRatio; + this.patternFrom = posFrom.tlbr; + this.patternTo = posTo.tlbr; + } + + setUniform(uniform: Uniform, globals: GlobalProperties, currentValue: PossiblyEvaluatedPropertyValue, uniformName: string) { + const pos = + uniformName === 'u_pattern_to' ? this.patternTo : + uniformName === 'u_pattern_from' ? this.patternFrom : + uniformName === 'u_pixel_ratio_to' ? this.pixelRatioTo : + uniformName === 'u_pixel_ratio_from' ? this.pixelRatioFrom : null; + if (pos) uniform.set(pos); + } + + getBinding(context: Context, location: WebGLUniformLocation, name: string): Partial> { + return name.substr(0, 9) === 'u_pattern' ? + new Uniform4f(context, location) : + new Uniform1f(context, location); + } +} + +class SourceExpressionBinder implements AttributeBinder { + expression: SourceExpression; + type: string; + maxValue: number; + + paintVertexArray: StructArray; + paintVertexAttributes: Array; + paintVertexBuffer: VertexBuffer; + + constructor(expression: SourceExpression, names: Array, type: string, PaintVertexArray: { + new (...args: any): StructArray; + }) { + this.expression = expression; + this.type = type; + this.maxValue = 0; + this.paintVertexAttributes = names.map((name) => ({ + name: `a_${name}`, + type: 'Float32', + components: type === 'color' ? 2 : 1, + offset: 0 + })); + this.paintVertexArray = new PaintVertexArray(); + } + + populatePaintArray(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) { + const start = this.paintVertexArray.length; + const value = this.expression.evaluate(new EvaluationParameters(0), feature, {}, canonical, [], formattedSection); + this.paintVertexArray.resize(newLength); + this._setPaintValue(start, newLength, value); + } + + updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState) { + const value = this.expression.evaluate({zoom: 0}, feature, featureState); + this._setPaintValue(start, end, value); + } + + _setPaintValue(start, end, value) { + if (this.type === 'color') { + const color = packColor(value); + for (let i = start; i < end; i++) { + this.paintVertexArray.emplace(i, color[0], color[1]); + } + } else { + for (let i = start; i < end; i++) { + this.paintVertexArray.emplace(i, value); + } + this.maxValue = Math.max(this.maxValue, Math.abs(value)); + } + } + + upload(context: Context) { + if (this.paintVertexArray && this.paintVertexArray.arrayBuffer) { + if (this.paintVertexBuffer && this.paintVertexBuffer.buffer) { + this.paintVertexBuffer.updateData(this.paintVertexArray); + } else { + this.paintVertexBuffer = context.createVertexBuffer(this.paintVertexArray, this.paintVertexAttributes, this.expression.isStateDependent); + } + } + } + + destroy() { + if (this.paintVertexBuffer) { + this.paintVertexBuffer.destroy(); + } + } +} + +class CompositeExpressionBinder implements AttributeBinder, UniformBinder { + expression: CompositeExpression; + uniformNames: Array; + type: string; + useIntegerZoom: boolean; + zoom: number; + maxValue: number; + + paintVertexArray: StructArray; + paintVertexAttributes: Array; + paintVertexBuffer: VertexBuffer; + + constructor(expression: CompositeExpression, names: Array, type: string, useIntegerZoom: boolean, zoom: number, PaintVertexArray: { + new (...args: any): StructArray; + }) { + this.expression = expression; + this.uniformNames = names.map(name => `u_${name}_t`); + this.type = type; + this.useIntegerZoom = useIntegerZoom; + this.zoom = zoom; + this.maxValue = 0; + this.paintVertexAttributes = names.map((name) => ({ + name: `a_${name}`, + type: 'Float32', + components: type === 'color' ? 4 : 2, + offset: 0 + })); + this.paintVertexArray = new PaintVertexArray(); + } + + populatePaintArray(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) { + const min = this.expression.evaluate(new EvaluationParameters(this.zoom), feature, {}, canonical, [], formattedSection); + const max = this.expression.evaluate(new EvaluationParameters(this.zoom + 1), feature, {}, canonical, [], formattedSection); + const start = this.paintVertexArray.length; + this.paintVertexArray.resize(newLength); + this._setPaintValue(start, newLength, min, max); + } + + updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState) { + const min = this.expression.evaluate({zoom: this.zoom}, feature, featureState); + const max = this.expression.evaluate({zoom: this.zoom + 1}, feature, featureState); + this._setPaintValue(start, end, min, max); + } + + _setPaintValue(start, end, min, max) { + if (this.type === 'color') { + const minColor = packColor(min); + const maxColor = packColor(max); + for (let i = start; i < end; i++) { + this.paintVertexArray.emplace(i, minColor[0], minColor[1], maxColor[0], maxColor[1]); + } + } else { + for (let i = start; i < end; i++) { + this.paintVertexArray.emplace(i, min, max); + } + this.maxValue = Math.max(this.maxValue, Math.abs(min), Math.abs(max)); + } + } + + upload(context: Context) { + if (this.paintVertexArray && this.paintVertexArray.arrayBuffer) { + if (this.paintVertexBuffer && this.paintVertexBuffer.buffer) { + this.paintVertexBuffer.updateData(this.paintVertexArray); + } else { + this.paintVertexBuffer = context.createVertexBuffer(this.paintVertexArray, this.paintVertexAttributes, this.expression.isStateDependent); + } + } + } + + destroy() { + if (this.paintVertexBuffer) { + this.paintVertexBuffer.destroy(); + } + } + + setUniform(uniform: Uniform, globals: GlobalProperties): void { + const currentZoom = this.useIntegerZoom ? Math.floor(globals.zoom) : globals.zoom; + const factor = clamp(this.expression.interpolationFactor(currentZoom, this.zoom, this.zoom + 1), 0, 1); + uniform.set(factor); + } + + getBinding(context: Context, location: WebGLUniformLocation, _: string): Uniform1f { + return new Uniform1f(context, location); + } +} + +class CrossFadedCompositeBinder implements AttributeBinder { + expression: CompositeExpression; + type: string; + useIntegerZoom: boolean; + zoom: number; + layerId: string; + + zoomInPaintVertexArray: StructArray; + zoomOutPaintVertexArray: StructArray; + zoomInPaintVertexBuffer: VertexBuffer; + zoomOutPaintVertexBuffer: VertexBuffer; + paintVertexAttributes: Array; + + constructor(expression: CompositeExpression, type: string, useIntegerZoom: boolean, zoom: number, PaintVertexArray: { + new (...args: any): StructArray; + }, layerId: string) { + this.expression = expression; + this.type = type; + this.useIntegerZoom = useIntegerZoom; + this.zoom = zoom; + this.layerId = layerId; + + this.zoomInPaintVertexArray = new PaintVertexArray(); + this.zoomOutPaintVertexArray = new PaintVertexArray(); + } + + populatePaintArray(length: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}) { + const start = this.zoomInPaintVertexArray.length; + this.zoomInPaintVertexArray.resize(length); + this.zoomOutPaintVertexArray.resize(length); + this._setPaintValues(start, length, feature.patterns && feature.patterns[this.layerId], imagePositions); + } + + updatePaintArray(start: number, end: number, feature: Feature, featureState: FeatureState, imagePositions: {[_: string]: ImagePosition}) { + this._setPaintValues(start, end, feature.patterns && feature.patterns[this.layerId], imagePositions); + } + + _setPaintValues(start, end, patterns, positions) { + if (!positions || !patterns) return; + + const {min, mid, max} = patterns; + const imageMin = positions[min]; + const imageMid = positions[mid]; + const imageMax = positions[max]; + if (!imageMin || !imageMid || !imageMax) return; + + // We populate two paint arrays because, for cross-faded properties, we don't know which direction + // we're cross-fading to at layout time. In order to keep vertex attributes to a minimum and not pass + // unnecessary vertex data to the shaders, we determine which to upload at draw time. + for (let i = start; i < end; i++) { + this.zoomInPaintVertexArray.emplace(i, + imageMid.tl[0], imageMid.tl[1], imageMid.br[0], imageMid.br[1], + imageMin.tl[0], imageMin.tl[1], imageMin.br[0], imageMin.br[1], + imageMid.pixelRatio, + imageMin.pixelRatio, + ); + this.zoomOutPaintVertexArray.emplace(i, + imageMid.tl[0], imageMid.tl[1], imageMid.br[0], imageMid.br[1], + imageMax.tl[0], imageMax.tl[1], imageMax.br[0], imageMax.br[1], + imageMid.pixelRatio, + imageMax.pixelRatio, + ); + } + } + + upload(context: Context) { + if (this.zoomInPaintVertexArray && this.zoomInPaintVertexArray.arrayBuffer && this.zoomOutPaintVertexArray && this.zoomOutPaintVertexArray.arrayBuffer) { + this.zoomInPaintVertexBuffer = context.createVertexBuffer(this.zoomInPaintVertexArray, patternAttributes.members, this.expression.isStateDependent); + this.zoomOutPaintVertexBuffer = context.createVertexBuffer(this.zoomOutPaintVertexArray, patternAttributes.members, this.expression.isStateDependent); + } + } + + destroy() { + if (this.zoomOutPaintVertexBuffer) this.zoomOutPaintVertexBuffer.destroy(); + if (this.zoomInPaintVertexBuffer) this.zoomInPaintVertexBuffer.destroy(); + } +} + +/** + * @internal + * ProgramConfiguration contains the logic for binding style layer properties and tile + * layer feature data into GL program uniforms and vertex attributes. + * + * Non-data-driven property values are bound to shader uniforms. Data-driven property + * values are bound to vertex attributes. In order to support a uniform GLSL syntax over + * both, [Mapbox GL Shaders](https://github.com/mapbox/mapbox-gl-shaders) defines a `#pragma` + * abstraction, which ProgramConfiguration is responsible for implementing. At runtime, + * it examines the attributes of a particular layer, combines this with fixed knowledge + * about how layers of the particular type are implemented, and determines which uniforms + * and vertex attributes will be required. It can then substitute the appropriate text + * into the shader source code, create and link a program, and bind the uniforms and + * vertex attributes in preparation for drawing. + * + * When a vector tile is parsed, this same configuration information is used to + * populate the attribute buffers needed for data-driven styling using the zoom + * level and feature property data. + */ +export class ProgramConfiguration { + binders: {[_: string]: AttributeBinder | UniformBinder}; + cacheKey: string; + + _buffers: Array; + + constructor(layer: TypedStyleLayer, zoom: number, filterProperties: (_: string) => boolean) { + this.binders = {}; + this._buffers = []; + + const keys = []; + + for (const property in layer.paint._values) { + if (!filterProperties(property)) continue; + const value = (layer.paint as any).get(property); + if (!(value instanceof PossiblyEvaluatedPropertyValue) || !supportsPropertyExpression(value.property.specification)) { + continue; + } + const names = paintAttributeNames(property, layer.type); + const expression = value.value; + const type = value.property.specification.type; + const useIntegerZoom = (value.property as any).useIntegerZoom; + const propType = value.property.specification['property-type']; + const isCrossFaded = propType === 'cross-faded' || propType === 'cross-faded-data-driven'; + + if (expression.kind === 'constant') { + this.binders[property] = isCrossFaded ? + new CrossFadedConstantBinder(expression.value, names) : + new ConstantBinder(expression.value, names, type); + keys.push(`/u_${property}`); + + } else if (expression.kind === 'source' || isCrossFaded) { + const StructArrayLayout = layoutType(property, type, 'source'); + this.binders[property] = isCrossFaded ? + new CrossFadedCompositeBinder(expression as CompositeExpression, type, useIntegerZoom, zoom, StructArrayLayout, layer.id) : + new SourceExpressionBinder(expression as SourceExpression, names, type, StructArrayLayout); + keys.push(`/a_${property}`); + + } else { + const StructArrayLayout = layoutType(property, type, 'composite'); + this.binders[property] = new CompositeExpressionBinder(expression, names, type, useIntegerZoom, zoom, StructArrayLayout); + keys.push(`/z_${property}`); + } + } + + this.cacheKey = keys.sort().join(''); + } + + getMaxValue(property: string): number { + const binder = this.binders[property]; + return binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder ? binder.maxValue : 0; + } + + populatePaintArrays(newLength: number, feature: Feature, imagePositions: {[_: string]: ImagePosition}, canonical?: CanonicalTileID, formattedSection?: FormattedSection) { + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder) + (binder as AttributeBinder).populatePaintArray(newLength, feature, imagePositions, canonical, formattedSection); + } + } + setConstantPatternPositions(posTo: ImagePosition, posFrom: ImagePosition) { + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof CrossFadedConstantBinder) + binder.setConstantPatternPositions(posTo, posFrom); + } + } + + updatePaintArrays( + featureStates: FeatureStates, + featureMap: FeaturePositionMap, + vtLayer: VectorTileLayer, + layer: TypedStyleLayer, + imagePositions: {[_: string]: ImagePosition} + ): boolean { + let dirty: boolean = false; + for (const id in featureStates) { + const positions = featureMap.getPositions(id); + + for (const pos of positions) { + const feature = vtLayer.feature(pos.index); + + for (const property in this.binders) { + const binder = this.binders[property]; + if ((binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || + binder instanceof CrossFadedCompositeBinder) && (binder as any).expression.isStateDependent === true) { + //AHM: Remove after https://github.com/mapbox/mapbox-gl-js/issues/6255 + const value = (layer.paint as any).get(property); + (binder as any).expression = value.value; + (binder as AttributeBinder).updatePaintArray(pos.start, pos.end, feature, featureStates[id], imagePositions); + dirty = true; + } + } + } + } + return dirty; + } + + defines(): Array { + const result = []; + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder) { + result.push(...binder.uniformNames.map(name => `#define HAS_UNIFORM_${name}`)); + } + } + return result; + } + + getBinderAttributes(): Array { + const result = []; + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder) { + for (let i = 0; i < binder.paintVertexAttributes.length; i++) { + result.push(binder.paintVertexAttributes[i].name); + } + } else if (binder instanceof CrossFadedCompositeBinder) { + for (let i = 0; i < patternAttributes.members.length; i++) { + result.push(patternAttributes.members[i].name); + } + } + } + return result; + } + + getBinderUniforms(): Array { + const uniforms = []; + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder || binder instanceof CompositeExpressionBinder) { + for (const uniformName of binder.uniformNames) { + uniforms.push(uniformName); + } + } + } + return uniforms; + } + + getPaintVertexBuffers(): Array { + return this._buffers; + } + + getUniforms(context: Context, locations: UniformLocations): Array { + const uniforms = []; + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof ConstantBinder || binder instanceof CrossFadedConstantBinder || binder instanceof CompositeExpressionBinder) { + for (const name of binder.uniformNames) { + if (locations[name]) { + const binding = binder.getBinding(context, locations[name], name); + uniforms.push({name, property, binding}); + } + } + } + } + return uniforms; + } + + setUniforms( + context: Context, + binderUniforms: Array, + properties: any, + globals: GlobalProperties + ) { + // Uniform state bindings are owned by the Program, but we set them + // from within the ProgramConfiguraton's binder members. + for (const {name, property, binding} of binderUniforms) { + (this.binders[property] as any).setUniform(binding, globals, properties.get(property), name); + } + } + + updatePaintBuffers(crossfade?: CrossfadeParameters) { + this._buffers = []; + + for (const property in this.binders) { + const binder = this.binders[property]; + if (crossfade && binder instanceof CrossFadedCompositeBinder) { + const patternVertexBuffer = crossfade.fromScale === 2 ? binder.zoomInPaintVertexBuffer : binder.zoomOutPaintVertexBuffer; + if (patternVertexBuffer) this._buffers.push(patternVertexBuffer); + + } else if ((binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder) && binder.paintVertexBuffer) { + this._buffers.push(binder.paintVertexBuffer); + } + } + } + + upload(context: Context) { + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder) + binder.upload(context); + } + this.updatePaintBuffers(); + } + + destroy() { + for (const property in this.binders) { + const binder = this.binders[property]; + if (binder instanceof SourceExpressionBinder || binder instanceof CompositeExpressionBinder || binder instanceof CrossFadedCompositeBinder) + binder.destroy(); + } + } +} + +export class ProgramConfigurationSet { + programConfigurations: {[_: string]: ProgramConfiguration}; + needsUpload: boolean; + _featureMap: FeaturePositionMap; + _bufferOffset: number; + + constructor(layers: ReadonlyArray, zoom: number, filterProperties: (_: string) => boolean = () => true) { + this.programConfigurations = {}; + for (const layer of layers) { + this.programConfigurations[layer.id] = new ProgramConfiguration(layer, zoom, filterProperties); + } + this.needsUpload = false; + this._featureMap = new FeaturePositionMap(); + this._bufferOffset = 0; + } + + populatePaintArrays(length: number, feature: Feature, index: number, imagePositions: {[_: string]: ImagePosition}, canonical: CanonicalTileID, formattedSection?: FormattedSection) { + for (const key in this.programConfigurations) { + this.programConfigurations[key].populatePaintArrays(length, feature, imagePositions, canonical, formattedSection); + } + + if (feature.id !== undefined) { + this._featureMap.add(feature.id, index, this._bufferOffset, length); + } + this._bufferOffset = length; + + this.needsUpload = true; + } + + updatePaintArrays(featureStates: FeatureStates, vtLayer: VectorTileLayer, layers: ReadonlyArray, imagePositions: {[_: string]: ImagePosition}) { + for (const layer of layers) { + this.needsUpload = this.programConfigurations[layer.id].updatePaintArrays(featureStates, this._featureMap, vtLayer, layer, imagePositions) || this.needsUpload; + } + } + + get(layerId: string) { + return this.programConfigurations[layerId]; + } + + upload(context: Context) { + if (!this.needsUpload) return; + for (const layerId in this.programConfigurations) { + this.programConfigurations[layerId].upload(context); + } + this.needsUpload = false; + } + + destroy() { + for (const layerId in this.programConfigurations) { + this.programConfigurations[layerId].destroy(); + } + } +} + +function paintAttributeNames(property, type) { + const attributeNameExceptions = { + 'text-opacity': ['opacity'], + 'icon-opacity': ['opacity'], + 'text-color': ['fill_color'], + 'icon-color': ['fill_color'], + 'text-halo-color': ['halo_color'], + 'icon-halo-color': ['halo_color'], + 'text-halo-blur': ['halo_blur'], + 'icon-halo-blur': ['halo_blur'], + 'text-halo-width': ['halo_width'], + 'icon-halo-width': ['halo_width'], + 'line-gap-width': ['gapwidth'], + 'line-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'], + 'fill-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'], + 'fill-extrusion-pattern': ['pattern_to', 'pattern_from', 'pixel_ratio_to', 'pixel_ratio_from'], + }; + + return attributeNameExceptions[property] || [property.replace(`${type}-`, '').replace(/-/g, '_')]; +} + +function getLayoutException(property) { + const propertyExceptions = { + 'line-pattern': { + 'source': PatternLayoutArray, + 'composite': PatternLayoutArray + }, + 'fill-pattern': { + 'source': PatternLayoutArray, + 'composite': PatternLayoutArray + }, + 'fill-extrusion-pattern': { + 'source': PatternLayoutArray, + 'composite': PatternLayoutArray + } + }; + + return propertyExceptions[property]; +} + +function layoutType(property, type, binderType) { + const defaultLayouts = { + 'color': { + 'source': StructArrayLayout2f8, + 'composite': StructArrayLayout4f16 + }, + 'number': { + 'source': StructArrayLayout1f4, + 'composite': StructArrayLayout2f8 + } + }; + + const layoutException = getLayoutException(property); + return layoutException && layoutException[binderType] || defaultLayouts[type][binderType]; +} + +register('ConstantBinder', ConstantBinder); +register('CrossFadedConstantBinder', CrossFadedConstantBinder); +register('SourceExpressionBinder', SourceExpressionBinder); +register('CrossFadedCompositeBinder', CrossFadedCompositeBinder); +register('CompositeExpressionBinder', CompositeExpressionBinder); +register('ProgramConfiguration', ProgramConfiguration, {omit: ['_buffers']}); +register('ProgramConfigurationSet', ProgramConfigurationSet); diff --git a/web/libraries/maplibre-gl/src/data/raster_bounds_attributes.ts b/web/libraries/maplibre-gl/src/data/raster_bounds_attributes.ts new file mode 100644 index 00000000..6ff07123 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/raster_bounds_attributes.ts @@ -0,0 +1,6 @@ +import {createLayout} from '../util/struct_array'; + +export default createLayout([ + {name: 'a_pos', type: 'Int16', components: 2}, + {name: 'a_texture_pos', type: 'Int16', components: 2} +]); diff --git a/web/libraries/maplibre-gl/src/data/segment.ts b/web/libraries/maplibre-gl/src/data/segment.ts new file mode 100644 index 00000000..07ab4345 --- /dev/null +++ b/web/libraries/maplibre-gl/src/data/segment.ts @@ -0,0 +1,89 @@ +import {warnOnce} from '../util/util'; + +import {register} from '../util/web_worker_transfer'; + +import type {VertexArrayObject} from '../render/vertex_array_object'; +import type {StructArray} from '../util/struct_array'; + +/** + * @internal + * A single segment of a vector + */ +export type Segment = { + sortKey?: number; + vertexOffset: number; + primitiveOffset: number; + vertexLength: number; + primitiveLength: number; + vaos: {[_: string]: VertexArrayObject}; +}; + +/** + * @internal + * Used for calculations on vector segments + */ +export class SegmentVector { + static MAX_VERTEX_ARRAY_LENGTH: number; + segments: Array; + + constructor(segments: Array = []) { + this.segments = segments; + } + + prepareSegment( + numVertices: number, + layoutVertexArray: StructArray, + indexArray: StructArray, + sortKey?: number + ): Segment { + let segment: Segment = this.segments[this.segments.length - 1]; + if (numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH) warnOnce(`Max vertices per segment is ${SegmentVector.MAX_VERTEX_ARRAY_LENGTH}: bucket requested ${numVertices}`); + if (!segment || segment.vertexLength + numVertices > SegmentVector.MAX_VERTEX_ARRAY_LENGTH || segment.sortKey !== sortKey) { + segment = ({ + vertexOffset: layoutVertexArray.length, + primitiveOffset: indexArray.length, + vertexLength: 0, + primitiveLength: 0 + } as any); + if (sortKey !== undefined) segment.sortKey = sortKey; + this.segments.push(segment); + } + return segment; + } + + get() { + return this.segments; + } + + destroy() { + for (const segment of this.segments) { + for (const k in segment.vaos) { + segment.vaos[k].destroy(); + } + } + } + + static simpleSegment( + vertexOffset: number, + primitiveOffset: number, + vertexLength: number, + primitiveLength: number + ): SegmentVector { + return new SegmentVector([{ + vertexOffset, + primitiveOffset, + vertexLength, + primitiveLength, + vaos: {}, + sortKey: 0 + }]); + } +} + +/** + * The maximum size of a vertex array. This limit is imposed by WebGL's 16 bit + * addressing of vertex buffers. + */ +SegmentVector.MAX_VERTEX_ARRAY_LENGTH = Math.pow(2, 16) - 1; + +register('SegmentVector', SegmentVector); diff --git a/web/libraries/maplibre-gl/src/geo/edge_insets.test.ts b/web/libraries/maplibre-gl/src/geo/edge_insets.test.ts new file mode 100644 index 00000000..d9379515 --- /dev/null +++ b/web/libraries/maplibre-gl/src/geo/edge_insets.test.ts @@ -0,0 +1,83 @@ +import expect from 'expect'; +import {EdgeInsets} from '../geo/edge_insets'; + +describe('EdgeInsets', () => { + describe('#constructor', () => { + test('creates an object with default values', () => { + expect(new EdgeInsets() instanceof EdgeInsets).toBeTruthy(); + }); + + test('invalid initialization', () => { + expect(() => { + new EdgeInsets(NaN, 10); + }).toThrow('Invalid value for edge-insets, top, bottom, left and right must all be numbers'); + + expect(() => { + new EdgeInsets(-10, 10, 20, 10); + }).toThrow('Invalid value for edge-insets, top, bottom, left and right must all be numbers'); + }); + + test('valid initialization', () => { + const top = 10; + const bottom = 15; + const left = 26; + const right = 19; + + const inset = new EdgeInsets(top, bottom, left, right); + expect(inset.top).toBe(top); + expect(inset.bottom).toBe(bottom); + expect(inset.left).toBe(left); + expect(inset.right).toBe(right); + }); + }); + + describe('#getCenter', () => { + test('valid input', () => { + const inset = new EdgeInsets(10, 15, 50, 10); + const center = inset.getCenter(600, 400); + expect(center.x).toBe(320); + expect(center.y).toBe(197.5); + }); + + test('center clamping', () => { + const inset = new EdgeInsets(300, 200, 500, 200); + const center = inset.getCenter(600, 400); + + // Midpoint of the overlap when padding overlaps + expect(center.x).toBe(450); + expect(center.y).toBe(250); + }); + }); + + describe('#interpolate', () => { + test('it works', () => { + const inset1 = new EdgeInsets(10, 15, 50, 10); + const inset2 = new EdgeInsets(20, 30, 100, 10); + const inset3 = inset1.interpolate(inset1, inset2, 0.5); + + // inset1 is mutated in-place + expect(inset3).toBe(inset1); + + expect(inset3.top).toBe(15); + expect(inset3.bottom).toBe(22.5); + expect(inset3.left).toBe(75); + expect(inset3.right).toBe(10); + }); + + }); + + test('#equals', () => { + const inset1 = new EdgeInsets(10, 15, 50, 10); + const inset2 = new EdgeInsets(10, 15, 50, 10); + const inset3 = new EdgeInsets(10, 15, 50, 11); + expect(inset1.equals(inset2)).toBeTruthy(); + expect(inset2.equals(inset3)).toBeFalsy(); + }); + + test('#clone', () => { + const inset1 = new EdgeInsets(10, 15, 50, 10); + const inset2 = inset1.clone(); + expect(inset2 === inset1).toBeFalsy(); + expect(inset1.equals(inset2)).toBeTruthy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/geo/edge_insets.ts b/web/libraries/maplibre-gl/src/geo/edge_insets.ts new file mode 100644 index 00000000..82ce736b --- /dev/null +++ b/web/libraries/maplibre-gl/src/geo/edge_insets.ts @@ -0,0 +1,146 @@ +import {interpolates} from '@maplibre/maplibre-gl-style-spec'; +import Point from '@mapbox/point-geometry'; +import {clamp} from '../util/util'; + +/** + * An `EdgeInset` object represents screen space padding applied to the edges of the viewport. + * This shifts the apprent center or the vanishing point of the map. This is useful for adding floating UI elements + * on top of the map and having the vanishing point shift as UI elements resize. + * + * @group Geography and Geometry + */ +export class EdgeInsets { + /** + * @defaultValue 0 + */ + top: number; + /** + * @defaultValue 0 + */ + bottom: number; + /** + * @defaultValue 0 + */ + left: number; + /** + * @defaultValue 0 + */ + right: number; + + constructor(top: number = 0, bottom: number = 0, left: number = 0, right: number = 0) { + if (isNaN(top) || top < 0 || + isNaN(bottom) || bottom < 0 || + isNaN(left) || left < 0 || + isNaN(right) || right < 0 + ) { + throw new Error('Invalid value for edge-insets, top, bottom, left and right must all be numbers'); + } + + this.top = top; + this.bottom = bottom; + this.left = left; + this.right = right; + } + + /** + * Interpolates the inset in-place. + * This maintains the current inset value for any inset not present in `target`. + * @param start - interpolation start + * @param target - interpolation target + * @param t - interpolation step/weight + * @returns the insets + */ + interpolate(start: PaddingOptions | EdgeInsets, target: PaddingOptions, t: number): EdgeInsets { + if (target.top != null && start.top != null) this.top = interpolates.number(start.top, target.top, t); + if (target.bottom != null && start.bottom != null) this.bottom = interpolates.number(start.bottom, target.bottom, t); + if (target.left != null && start.left != null) this.left = interpolates.number(start.left, target.left, t); + if (target.right != null && start.right != null) this.right = interpolates.number(start.right, target.right, t); + + return this; + } + + /** + * Utility method that computes the new apprent center or vanishing point after applying insets. + * This is in pixels and with the top left being (0.0) and +y being downwards. + * + * @param width - the width + * @param height - the height + * @returns the point + */ + getCenter(width: number, height: number): Point { + // Clamp insets so they never overflow width/height and always calculate a valid center + const x = clamp((this.left + width - this.right) / 2, 0, width); + const y = clamp((this.top + height - this.bottom) / 2, 0, height); + + return new Point(x, y); + } + + equals(other: PaddingOptions): boolean { + return this.top === other.top && + this.bottom === other.bottom && + this.left === other.left && + this.right === other.right; + } + + clone(): EdgeInsets { + return new EdgeInsets(this.top, this.bottom, this.left, this.right); + } + + /** + * Returns the current state as json, useful when you want to have a + * read-only representation of the inset. + * + * @returns state as json + */ + toJSON(): PaddingOptions { + return { + top: this.top, + bottom: this.bottom, + left: this.left, + right: this.right + }; + } +} + +/** + * Options for setting padding on calls to methods such as {@link Map#fitBounds}, {@link Map#fitScreenCoordinates}, and {@link Map#setPadding}. Adjust these options to set the amount of padding in pixels added to the edges of the canvas. Set a uniform padding on all edges or individual values for each edge. All properties of this object must be + * non-negative integers. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let bbox = [[-79, 43], [-73, 45]]; + * map.fitBounds(bbox, { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + * + * @example + * ```ts + * let bbox = [[-79, 43], [-73, 45]]; + * map.fitBounds(bbox, { + * padding: 20 + * }); + * ``` + * @see [Fit to the bounds of a LineString](https://maplibre.org/maplibre-gl-js/docs/examples/zoomto-linestring/) + * @see [Fit a map to a bounding box](https://maplibre.org/maplibre-gl-js/docs/examples/fitbounds/) + */ +export type PaddingOptions = { + /** + * Padding in pixels from the top of the map canvas. + */ + top: number; + /** + * Padding in pixels from the bottom of the map canvas. + */ + bottom: number; + /** + * Padding in pixels from the left of the map canvas. + */ + right: number; + /** + * Padding in pixels from the right of the map canvas. + */ + left: number; +}; diff --git a/web/libraries/maplibre-gl/src/geo/lng_lat.test.ts b/web/libraries/maplibre-gl/src/geo/lng_lat.test.ts new file mode 100644 index 00000000..2833a7f1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/geo/lng_lat.test.ts @@ -0,0 +1,65 @@ +import {LngLat} from '../geo/lng_lat'; + +describe('LngLat', () => { + test('#constructor', () => { + expect(new LngLat(0, 0) instanceof LngLat).toBeTruthy(); + + expect(() => { + /*eslint no-new: 0*/ + new LngLat(0, -91); + }).toThrow('Invalid LngLat latitude value: must be between -90 and 90'); + + expect(() => { + /*eslint no-new: 0*/ + new LngLat(0, 91); + }).toThrow('Invalid LngLat latitude value: must be between -90 and 90'); + }); + + test('#convert', () => { + expect(LngLat.convert([0, 10]) instanceof LngLat).toBeTruthy(); + expect(LngLat.convert({lng: 0, lat: 10}) instanceof LngLat).toBeTruthy(); + expect(LngLat.convert({lng: 0, lat: 0}) instanceof LngLat).toBeTruthy(); + expect(LngLat.convert({lon: 0, lat: 10}) instanceof LngLat).toBeTruthy(); + expect(LngLat.convert({lon: 0, lat: 0}) instanceof LngLat).toBeTruthy(); + expect(LngLat.convert(new LngLat(0, 0)) instanceof LngLat).toBeTruthy(); + }); + + test('#wrap', () => { + expect(new LngLat(0, 0).wrap()).toEqual({lng: 0, lat: 0}); + expect(new LngLat(10, 20).wrap()).toEqual({lng: 10, lat: 20}); + expect(new LngLat(360, 0).wrap()).toEqual({lng: 0, lat: 0}); + expect(new LngLat(190, 0).wrap()).toEqual({lng: -170, lat: 0}); + }); + + test('#toArray', () => { + expect(new LngLat(10, 20).toArray()).toEqual([10, 20]); + }); + + test('#toString', () => { + expect(new LngLat(10, 20).toString()).toBe('LngLat(10, 20)'); + }); + + test('#distanceTo', () => { + const newYork = new LngLat(-74.0060, 40.7128); + const losAngeles = new LngLat(-118.2437, 34.0522); + const d = newYork.distanceTo(losAngeles); // 3935751.690893987, "true distance" is 3966km + expect(d > 3935750).toBeTruthy(); + expect(d < 3935752).toBeTruthy(); + }); + + test('#distanceTo to pole', () => { + const newYork = new LngLat(-74.0060, 40.7128); + const northPole = new LngLat(-135, 90); + const d = newYork.distanceTo(northPole); // 5480494.158486183 , "true distance" is 5499km + expect(d > 5480493).toBeTruthy(); + expect(d < 5480495).toBeTruthy(); + }); + + test('#distanceTo to Null Island', () => { + const newYork = new LngLat(-74.0060, 40.7128); + const nullIsland = new LngLat(0, 0); + const d = newYork.distanceTo(nullIsland); // 8667080.125666846 , "true distance" is 8661km + expect(d > 8667079).toBeTruthy(); + expect(d < 8667081).toBeTruthy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/geo/lng_lat.ts b/web/libraries/maplibre-gl/src/geo/lng_lat.ts new file mode 100644 index 00000000..3b2aa37b --- /dev/null +++ b/web/libraries/maplibre-gl/src/geo/lng_lat.ts @@ -0,0 +1,169 @@ +import {wrap} from '../util/util'; + +/* +* Approximate radius of the earth in meters. +* Uses the WGS-84 approximation. The radius at the equator is ~6378137 and at the poles is ~6356752. https://en.wikipedia.org/wiki/World_Geodetic_System#WGS84 +* 6371008.8 is one published "average radius" see https://en.wikipedia.org/wiki/Earth_radius#Mean_radius, or ftp://athena.fsv.cvut.cz/ZFG/grs80-Moritz.pdf p.4 +*/ +export const earthRadius = 6371008.8; + +/** + * A {@link LngLat} object, an array of two numbers representing longitude and latitude, + * or an object with `lng` and `lat` or `lon` and `lat` properties. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let v1 = new maplibregl.LngLat(-122.420679, 37.772537); + * let v2 = [-122.420679, 37.772537]; + * let v3 = {lon: -122.420679, lat: 37.772537}; + * ``` + */ +export type LngLatLike = LngLat | { + lng: number; + lat: number; +} | { + lon: number; + lat: number; +} | [number, number]; + +/** + * A `LngLat` object represents a given longitude and latitude coordinate, measured in degrees. + * These coordinates are based on the [WGS84 (EPSG:4326) standard](https://en.wikipedia.org/wiki/World_Geodetic_System#WGS84). + * + * MapLibre GL JS uses longitude, latitude coordinate order (as opposed to latitude, longitude) to match the + * [GeoJSON specification](https://tools.ietf.org/html/rfc7946). + * + * Note that any MapLibre GL JS method that accepts a `LngLat` object as an argument or option + * can also accept an `Array` of two numbers and will perform an implicit conversion. + * This flexible type is documented as {@link LngLatLike}. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let ll = new maplibregl.LngLat(-123.9749, 40.7736); + * ll.lng; // = -123.9749 + * ``` + * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/) + * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/) + * @see [Create a timeline animation](https://maplibre.org/maplibre-gl-js/docs/examples/timeline-animation/) + */ +export class LngLat { + lng: number; + lat: number; + + /** + * @param lng - Longitude, measured in degrees. + * @param lat - Latitude, measured in degrees. + */ + constructor(lng: number, lat: number) { + if (isNaN(lng) || isNaN(lat)) { + throw new Error(`Invalid LngLat object: (${lng}, ${lat})`); + } + this.lng = +lng; + this.lat = +lat; + if (this.lat > 90 || this.lat < -90) { + throw new Error('Invalid LngLat latitude value: must be between -90 and 90'); + } + } + + /** + * Returns a new `LngLat` object whose longitude is wrapped to the range (-180, 180). + * + * @returns The wrapped `LngLat` object. + * @example + * ```ts + * let ll = new maplibregl.LngLat(286.0251, 40.7736); + * let wrapped = ll.wrap(); + * wrapped.lng; // = -73.9749 + * ``` + */ + wrap() { + return new LngLat(wrap(this.lng, -180, 180), this.lat); + } + + /** + * Returns the coordinates represented as an array of two numbers. + * + * @returns The coordinates represented as an array of longitude and latitude. + * @example + * ```ts + * let ll = new maplibregl.LngLat(-73.9749, 40.7736); + * ll.toArray(); // = [-73.9749, 40.7736] + * ``` + */ + toArray(): [number, number] { + return [this.lng, this.lat]; + } + + /** + * Returns the coordinates represent as a string. + * + * @returns The coordinates represented as a string of the format `'LngLat(lng, lat)'`. + * @example + * ```ts + * let ll = new maplibregl.LngLat(-73.9749, 40.7736); + * ll.toString(); // = "LngLat(-73.9749, 40.7736)" + * ``` + */ + toString(): string { + return `LngLat(${this.lng}, ${this.lat})`; + } + + /** + * Returns the approximate distance between a pair of coordinates in meters + * Uses the Haversine Formula (from R.W. Sinnott, "Virtues of the Haversine", Sky and Telescope, vol. 68, no. 2, 1984, p. 159) + * + * @param lngLat - coordinates to compute the distance to + * @returns Distance in meters between the two coordinates. + * @example + * ```ts + * let new_york = new maplibregl.LngLat(-74.0060, 40.7128); + * let los_angeles = new maplibregl.LngLat(-118.2437, 34.0522); + * new_york.distanceTo(los_angeles); // = 3935751.690893987, "true distance" using a non-spherical approximation is ~3966km + * ``` + */ + distanceTo(lngLat: LngLat): number { + const rad = Math.PI / 180; + const lat1 = this.lat * rad; + const lat2 = lngLat.lat * rad; + const a = Math.sin(lat1) * Math.sin(lat2) + Math.cos(lat1) * Math.cos(lat2) * Math.cos((lngLat.lng - this.lng) * rad); + + const maxMeters = earthRadius * Math.acos(Math.min(a, 1)); + return maxMeters; + } + + /** + * Converts an array of two numbers or an object with `lng` and `lat` or `lon` and `lat` properties + * to a `LngLat` object. + * + * If a `LngLat` object is passed in, the function returns it unchanged. + * + * @param input - An array of two numbers or object to convert, or a `LngLat` object to return. + * @returns A new `LngLat` object, if a conversion occurred, or the original `LngLat` object. + * @example + * ```ts + * let arr = [-73.9749, 40.7736]; + * let ll = maplibregl.LngLat.convert(arr); + * ll; // = LngLat {lng: -73.9749, lat: 40.7736} + * ``` + */ + static convert(input: LngLatLike): LngLat { + if (input instanceof LngLat) { + return input; + } + if (Array.isArray(input) && (input.length === 2 || input.length === 3)) { + return new LngLat(Number(input[0]), Number(input[1])); + } + if (!Array.isArray(input) && typeof input === 'object' && input !== null) { + return new LngLat( + // flow can't refine this to have one of lng or lat, so we have to cast to any + Number('lng' in input ? (input as any).lng : (input as any).lon), + Number(input.lat) + ); + } + throw new Error('`LngLatLike` argument must be specified as a LngLat instance, an object {lng: , lat: }, an object {lon: , lat: }, or an array of [, ]'); + } +} diff --git a/web/libraries/maplibre-gl/src/geo/lng_lat_bounds.test.ts b/web/libraries/maplibre-gl/src/geo/lng_lat_bounds.test.ts new file mode 100644 index 00000000..1b40e02c --- /dev/null +++ b/web/libraries/maplibre-gl/src/geo/lng_lat_bounds.test.ts @@ -0,0 +1,235 @@ +import {LngLat} from './lng_lat'; +import {LngLatBounds} from './lng_lat_bounds'; + +describe('LngLatBounds', () => { + test('#constructor', () => { + const sw = new LngLat(0, 0); + const ne = new LngLat(-10, 10); + const bounds = new LngLatBounds(sw, ne); + expect(bounds.getSouth()).toBe(0); + expect(bounds.getWest()).toBe(0); + expect(bounds.getNorth()).toBe(10); + expect(bounds.getEast()).toBe(-10); + }); + + test('#constructor across dateline', () => { + const sw = new LngLat(170, 0); + const ne = new LngLat(-170, 10); + const bounds = new LngLatBounds(sw, ne); + expect(bounds.getSouth()).toBe(0); + expect(bounds.getWest()).toBe(170); + expect(bounds.getNorth()).toBe(10); + expect(bounds.getEast()).toBe(-170); + }); + + test('#constructor across pole', () => { + const sw = new LngLat(0, 85); + const ne = new LngLat(-10, -85); + const bounds = new LngLatBounds(sw, ne); + expect(bounds.getSouth()).toBe(85); + expect(bounds.getWest()).toBe(0); + expect(bounds.getNorth()).toBe(-85); + expect(bounds.getEast()).toBe(-10); + }); + + test('#constructor no args', () => { + const bounds = new LngLatBounds(); + const t1 = () => { + bounds.getCenter(); + }; + expect(t1).toThrow(); + }); + + test('#extend with coordinate', () => { + const bounds = new LngLatBounds([0, 0], [10, 10]); + bounds.extend([-10, -10]); + + expect(bounds.getSouth()).toBe(-10); + expect(bounds.getWest()).toBe(-10); + expect(bounds.getNorth()).toBe(10); + expect(bounds.getEast()).toBe(10); + + bounds.extend(new LngLat(-15, -15)); + + expect(bounds.getSouth()).toBe(-15); + expect(bounds.getWest()).toBe(-15); + expect(bounds.getNorth()).toBe(10); + expect(bounds.getEast()).toBe(10); + + bounds.extend([-80, -80, 80, 80]); + + expect(bounds.getSouth()).toBe(-80); + expect(bounds.getWest()).toBe(-80); + expect(bounds.getNorth()).toBe(80); + expect(bounds.getEast()).toBe(80); + + bounds.extend({lng: -90, lat: -90}); + + expect(bounds.getSouth()).toBe(-90); + expect(bounds.getWest()).toBe(-90); + expect(bounds.getNorth()).toBe(80); + expect(bounds.getEast()).toBe(80); + + bounds.extend({lon: 90, lat: 90}); + + expect(bounds.getSouth()).toBe(-90); + expect(bounds.getWest()).toBe(-90); + expect(bounds.getNorth()).toBe(90); + expect(bounds.getEast()).toBe(90); + }); + + test('#extend with bounds', () => { + const bounds1 = new LngLatBounds([0, 0], [10, 10]); + const bounds2 = new LngLatBounds([-10, -10], [10, 10]); + + bounds1.extend(bounds2); + + expect(bounds1.getSouth()).toBe(-10); + expect(bounds1.getWest()).toBe(-10); + expect(bounds1.getNorth()).toBe(10); + expect(bounds1.getEast()).toBe(10); + + const bounds4 = new LngLatBounds([-20, -20, 20, 20]); + bounds1.extend(bounds4); + + expect(bounds1.getSouth()).toBe(-20); + expect(bounds1.getWest()).toBe(-20); + expect(bounds1.getNorth()).toBe(20); + expect(bounds1.getEast()).toBe(20); + + const bounds5 = new LngLatBounds(); + bounds1.extend(bounds5); + + expect(bounds1.getSouth()).toBe(-20); + expect(bounds1.getWest()).toBe(-20); + expect(bounds1.getNorth()).toBe(20); + expect(bounds1.getEast()).toBe(20); + }); + + test('#extend with null', () => { + const bounds = new LngLatBounds([0, 0], [10, 10]); + + bounds.extend(null); + + expect(bounds.getSouth()).toBe(0); + expect(bounds.getWest()).toBe(0); + expect(bounds.getNorth()).toBe(10); + expect(bounds.getEast()).toBe(10); + }); + + test('#extend undefined bounding box', () => { + const bounds1 = new LngLatBounds(undefined, undefined); + const bounds2 = new LngLatBounds([-10, -10], [10, 10]); + + bounds1.extend(bounds2); + + expect(bounds1.getSouth()).toBe(-10); + expect(bounds1.getWest()).toBe(-10); + expect(bounds1.getNorth()).toBe(10); + expect(bounds1.getEast()).toBe(10); + }); + + test('#extend same LngLat instance', () => { + const point = new LngLat(0, 0); + const bounds = new LngLatBounds(point, point); + + bounds.extend(new LngLat(15, 15)); + + expect(bounds.getSouth()).toBe(0); + expect(bounds.getWest()).toBe(0); + expect(bounds.getNorth()).toBe(15); + expect(bounds.getEast()).toBe(15); + }); + + test('accessors', () => { + const sw = new LngLat(0, 0); + const ne = new LngLat(-10, -20); + const bounds = new LngLatBounds(sw, ne); + expect(bounds.getCenter()).toEqual(new LngLat(-5, -10)); + expect(bounds.getSouth()).toBe(0); + expect(bounds.getWest()).toBe(0); + expect(bounds.getNorth()).toBe(-20); + expect(bounds.getEast()).toBe(-10); + expect(bounds.getSouthWest()).toEqual(new LngLat(0, 0)); + expect(bounds.getSouthEast()).toEqual(new LngLat(-10, 0)); + expect(bounds.getNorthEast()).toEqual(new LngLat(-10, -20)); + expect(bounds.getNorthWest()).toEqual(new LngLat(0, -20)); + }); + + test('#convert', () => { + const sw = new LngLat(0, 0); + const ne = new LngLat(-10, 10); + const bounds = new LngLatBounds(sw, ne); + expect(LngLatBounds.convert(undefined)).toBeUndefined(); + expect(LngLatBounds.convert(bounds)).toEqual(bounds); + expect(LngLatBounds.convert([sw, ne])).toEqual(bounds); + expect( + LngLatBounds.convert([bounds.getWest(), bounds.getSouth(), bounds.getEast(), bounds.getNorth()]) + ).toEqual(bounds); + }); + + test('#toArray', () => { + const llb = new LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]); + expect(llb.toArray()).toEqual([[-73.9876, 40.7661], [-73.9397, 40.8002]]); + }); + + test('#toString', () => { + const llb = new LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]); + expect(llb.toString()).toBe('LngLatBounds(LngLat(-73.9876, 40.7661), LngLat(-73.9397, 40.8002))'); + }); + + test('#isEmpty', () => { + const nullBounds = new LngLatBounds(); + expect(nullBounds.isEmpty()).toBe(true); + + const sw = new LngLat(0, 0); + const ne = new LngLat(-10, 10); + const bounds = new LngLatBounds(sw, ne); + expect(bounds.isEmpty()).toBe(false); + }); + + test('#fromLngLat', () => { + const center0 = new LngLat(0, 0); + const center1 = new LngLat(-73.9749, 40.7736); + + const center0Radius10 = LngLatBounds.fromLngLat(center0, 10); + const center1Radius10 = LngLatBounds.fromLngLat(center1, 10); + const center1Radius0 = LngLatBounds.fromLngLat(center1); + + expect(center0Radius10.toArray()).toEqual( + [[-0.00008983152770714982, -0.00008983152770714982], [0.00008983152770714982, 0.00008983152770714982]] + ); + expect(center1Radius10.toArray()).toEqual( + [[-73.97501862141328, 40.77351016847229], [-73.97478137858673, 40.77368983152771]] + ); + expect(center1Radius0.toArray()).toEqual([[-73.9749, 40.7736], [-73.9749, 40.7736]]); + }); + + describe('contains', () => { + describe('point', () => { + test('point is in bounds', () => { + const llb = new LngLatBounds([-1, -1], [1, 1]); + const ll = {lng: 0, lat: 0}; + expect(llb.contains(ll)).toBeTruthy(); + }); + + test('point is not in bounds', () => { + const llb = new LngLatBounds([-1, -1], [1, 1]); + const ll = {lng: 3, lat: 3}; + expect(llb.contains(ll)).toBeFalsy(); + }); + + test('point is in bounds that spans dateline', () => { + const llb = new LngLatBounds([190, -10], [170, 10]); + const ll = {lng: 180, lat: 0}; + expect(llb.contains(ll)).toBeTruthy(); + }); + + test('point is not in bounds that spans dateline', () => { + const llb = new LngLatBounds([190, -10], [170, 10]); + const ll = {lng: 0, lat: 0}; + expect(llb.contains(ll)).toBeFalsy(); + }); + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/geo/lng_lat_bounds.ts b/web/libraries/maplibre-gl/src/geo/lng_lat_bounds.ts new file mode 100644 index 00000000..9d75682f --- /dev/null +++ b/web/libraries/maplibre-gl/src/geo/lng_lat_bounds.ts @@ -0,0 +1,333 @@ +import {LngLat} from './lng_lat'; +import type {LngLatLike} from './lng_lat'; + +/** + * A {@link LngLatBounds} object, an array of {@link LngLatLike} objects in [sw, ne] order, + * or an array of numbers in [west, south, east, north] order. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let v1 = new maplibregl.LngLatBounds( + * new maplibregl.LngLat(-73.9876, 40.7661), + * new maplibregl.LngLat(-73.9397, 40.8002) + * ); + * let v2 = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]) + * let v3 = [[-73.9876, 40.7661], [-73.9397, 40.8002]]; + * ``` + */ +export type LngLatBoundsLike = LngLatBounds | [LngLatLike, LngLatLike] | [number, number, number, number]; + +/** + * A `LngLatBounds` object represents a geographical bounding box, + * defined by its southwest and northeast points in longitude and latitude. + * + * If no arguments are provided to the constructor, a `null` bounding box is created. + * + * Note that any Mapbox GL method that accepts a `LngLatBounds` object as an argument or option + * can also accept an `Array` of two {@link LngLatLike} constructs and will perform an implicit conversion. + * This flexible type is documented as {@link LngLatBoundsLike}. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let sw = new maplibregl.LngLat(-73.9876, 40.7661); + * let ne = new maplibregl.LngLat(-73.9397, 40.8002); + * let llb = new maplibregl.LngLatBounds(sw, ne); + * ``` + */ +export class LngLatBounds { + _ne: LngLat; + _sw: LngLat; + + /** + * @param sw - The southwest corner of the bounding box. + * OR array of 4 numbers in the order of west, south, east, north + * OR array of 2 LngLatLike: [sw,ne] + * @param ne - The northeast corner of the bounding box. + * @example + * ```ts + * let sw = new maplibregl.LngLat(-73.9876, 40.7661); + * let ne = new maplibregl.LngLat(-73.9397, 40.8002); + * let llb = new maplibregl.LngLatBounds(sw, ne); + * ``` + * OR + * ```ts + * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661, -73.9397, 40.8002]); + * ``` + * OR + * ```ts + * let llb = new maplibregl.LngLatBounds([sw, ne]); + * ``` + */ + constructor(sw?: LngLatLike | [number, number, number, number] | [LngLatLike, LngLatLike], ne?: LngLatLike) { + if (!sw) { + // noop + } else if (ne) { + this.setSouthWest(sw).setNorthEast(ne); + } else if (Array.isArray(sw)) { + if (sw.length === 4) { + // 4 element array: west, south, east, north + this.setSouthWest([sw[0], sw[1]]).setNorthEast([sw[2], sw[3]]); + } else { + this.setSouthWest(sw[0] as LngLatLike).setNorthEast(sw[1] as LngLatLike); + } + } + } + + /** + * Set the northeast corner of the bounding box + * + * @param ne - a {@link LngLatLike} object describing the northeast corner of the bounding box. + * @returns `this` + */ + setNorthEast(ne: LngLatLike): this { + this._ne = ne instanceof LngLat ? new LngLat(ne.lng, ne.lat) : LngLat.convert(ne); + return this; + } + + /** + * Set the southwest corner of the bounding box + * + * @param sw - a {@link LngLatLike} object describing the southwest corner of the bounding box. + * @returns `this` + */ + setSouthWest(sw: LngLatLike): this { + this._sw = sw instanceof LngLat ? new LngLat(sw.lng, sw.lat) : LngLat.convert(sw); + return this; + } + + /** + * Extend the bounds to include a given LngLatLike or LngLatBoundsLike. + * + * @param obj - object to extend to + * @returns `this` + */ + extend(obj: LngLatLike | LngLatBoundsLike): this { + const sw = this._sw, + ne = this._ne; + let sw2, ne2; + + if (obj instanceof LngLat) { + sw2 = obj; + ne2 = obj; + + } else if (obj instanceof LngLatBounds) { + sw2 = obj._sw; + ne2 = obj._ne; + + if (!sw2 || !ne2) return this; + + } else { + if (Array.isArray(obj)) { + if (obj.length === 4 || (obj as any[]).every(Array.isArray)) { + const lngLatBoundsObj = (obj as any as LngLatBoundsLike); + return this.extend(LngLatBounds.convert(lngLatBoundsObj)); + } else { + const lngLatObj = (obj as any as LngLatLike); + return this.extend(LngLat.convert(lngLatObj)); + } + + } else if (obj && ('lng' in obj || 'lon' in obj) && 'lat' in obj) { + return this.extend(LngLat.convert(obj)); + } + + return this; + } + + if (!sw && !ne) { + this._sw = new LngLat(sw2.lng, sw2.lat); + this._ne = new LngLat(ne2.lng, ne2.lat); + + } else { + sw.lng = Math.min(sw2.lng, sw.lng); + sw.lat = Math.min(sw2.lat, sw.lat); + ne.lng = Math.max(ne2.lng, ne.lng); + ne.lat = Math.max(ne2.lat, ne.lat); + } + + return this; + } + + /** + * Returns the geographical coordinate equidistant from the bounding box's corners. + * + * @returns The bounding box's center. + * @example + * ```ts + * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]); + * llb.getCenter(); // = LngLat {lng: -73.96365, lat: 40.78315} + * ``` + */ + getCenter(): LngLat { + return new LngLat((this._sw.lng + this._ne.lng) / 2, (this._sw.lat + this._ne.lat) / 2); + } + + /** + * Returns the southwest corner of the bounding box. + * + * @returns The southwest corner of the bounding box. + */ + getSouthWest(): LngLat { return this._sw; } + + /** + * Returns the northeast corner of the bounding box. + * + * @returns The northeast corner of the bounding box. + */ + getNorthEast(): LngLat { return this._ne; } + + /** + * Returns the northwest corner of the bounding box. + * + * @returns The northwest corner of the bounding box. + */ + getNorthWest(): LngLat { return new LngLat(this.getWest(), this.getNorth()); } + + /** + * Returns the southeast corner of the bounding box. + * + * @returns The southeast corner of the bounding box. + */ + getSouthEast(): LngLat { return new LngLat(this.getEast(), this.getSouth()); } + + /** + * Returns the west edge of the bounding box. + * + * @returns The west edge of the bounding box. + */ + getWest(): number { return this._sw.lng; } + + /** + * Returns the south edge of the bounding box. + * + * @returns The south edge of the bounding box. + */ + getSouth(): number { return this._sw.lat; } + + /** + * Returns the east edge of the bounding box. + * + * @returns The east edge of the bounding box. + */ + getEast(): number { return this._ne.lng; } + + /** + * Returns the north edge of the bounding box. + * + * @returns The north edge of the bounding box. + */ + getNorth(): number { return this._ne.lat; } + + /** + * Returns the bounding box represented as an array. + * + * @returns The bounding box represented as an array, consisting of the + * southwest and northeast coordinates of the bounding represented as arrays of numbers. + * @example + * ```ts + * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]); + * llb.toArray(); // = [[-73.9876, 40.7661], [-73.9397, 40.8002]] + * ``` + */ + toArray() { + return [this._sw.toArray(), this._ne.toArray()]; + } + + /** + * Return the bounding box represented as a string. + * + * @returns The bounding box represents as a string of the format + * `'LngLatBounds(LngLat(lng, lat), LngLat(lng, lat))'`. + * @example + * ```ts + * let llb = new maplibregl.LngLatBounds([-73.9876, 40.7661], [-73.9397, 40.8002]); + * llb.toString(); // = "LngLatBounds(LngLat(-73.9876, 40.7661), LngLat(-73.9397, 40.8002))" + * ``` + */ + toString() { + return `LngLatBounds(${this._sw.toString()}, ${this._ne.toString()})`; + } + + /** + * Check if the bounding box is an empty/`null`-type box. + * + * @returns True if bounds have been defined, otherwise false. + */ + isEmpty() { + return !(this._sw && this._ne); + } + + /** + * Check if the point is within the bounding box. + * + * @param lnglat - geographic point to check against. + * @returns `true` if the point is within the bounding box. + * @example + * ```ts + * let llb = new maplibregl.LngLatBounds( + * new maplibregl.LngLat(-73.9876, 40.7661), + * new maplibregl.LngLat(-73.9397, 40.8002) + * ); + * + * let ll = new maplibregl.LngLat(-73.9567, 40.7789); + * + * console.log(llb.contains(ll)); // = true + * ``` + */ + contains(lnglat: LngLatLike) { + const {lng, lat} = LngLat.convert(lnglat); + + const containsLatitude = this._sw.lat <= lat && lat <= this._ne.lat; + let containsLongitude = this._sw.lng <= lng && lng <= this._ne.lng; + if (this._sw.lng > this._ne.lng) { // wrapped coordinates + containsLongitude = this._sw.lng >= lng && lng >= this._ne.lng; + } + + return containsLatitude && containsLongitude; + } + + /** + * Converts an array to a `LngLatBounds` object. + * + * If a `LngLatBounds` object is passed in, the function returns it unchanged. + * + * Internally, the function calls `LngLat#convert` to convert arrays to `LngLat` values. + * + * @param input - An array of two coordinates to convert, or a `LngLatBounds` object to return. + * @returns A new `LngLatBounds` object, if a conversion occurred, or the original `LngLatBounds` object. + * @example + * ```ts + * let arr = [[-73.9876, 40.7661], [-73.9397, 40.8002]]; + * let llb = maplibregl.LngLatBounds.convert(arr); // = LngLatBounds {_sw: LngLat {lng: -73.9876, lat: 40.7661}, _ne: LngLat {lng: -73.9397, lat: 40.8002}} + * ``` + */ + static convert(input: LngLatBoundsLike | null): LngLatBounds { + if (input instanceof LngLatBounds) return input; + if (!input) return input as null; + return new LngLatBounds(input); + } + + /** + * Returns a `LngLatBounds` from the coordinates extended by a given `radius`. The returned `LngLatBounds` completely contains the `radius`. + * + * @param center - center coordinates of the new bounds. + * @param radius - Distance in meters from the coordinates to extend the bounds. + * @returns A new `LngLatBounds` object representing the coordinates extended by the `radius`. + * @example + * ```ts + * let center = new maplibregl.LngLat(-73.9749, 40.7736); + * maplibregl.LngLatBounds.fromLngLat(100).toArray(); // = [[-73.97501862141328, 40.77351016847229], [-73.97478137858673, 40.77368983152771]] + * ``` + */ + static fromLngLat(center: LngLat, radius:number = 0): LngLatBounds { + const earthCircumferenceInMetersAtEquator = 40075017; + const latAccuracy = 360 * radius / earthCircumferenceInMetersAtEquator, + lngAccuracy = latAccuracy / Math.cos((Math.PI / 180) * center.lat); + + return new LngLatBounds(new LngLat(center.lng - lngAccuracy, center.lat - latAccuracy), + new LngLat(center.lng + lngAccuracy, center.lat + latAccuracy)); + } +} diff --git a/web/libraries/maplibre-gl/src/geo/mercator_coordinate.test.ts b/web/libraries/maplibre-gl/src/geo/mercator_coordinate.test.ts new file mode 100644 index 00000000..305bb492 --- /dev/null +++ b/web/libraries/maplibre-gl/src/geo/mercator_coordinate.test.ts @@ -0,0 +1,34 @@ +import {LngLat} from './lng_lat'; +import {MercatorCoordinate, mercatorScale} from './mercator_coordinate'; + +describe('LngLat', () => { + test('#constructor', () => { + expect(new MercatorCoordinate(0, 0) instanceof MercatorCoordinate).toBeTruthy(); + expect(new MercatorCoordinate(0, 0, 0) instanceof MercatorCoordinate).toBeTruthy(); + }); + + test('#fromLngLat', () => { + const nullIsland = new LngLat(0, 0); + expect(MercatorCoordinate.fromLngLat(nullIsland)).toEqual({x: 0.5, y: 0.5, z: 0}); + }); + + test('#toLngLat', () => { + const dc = new LngLat(-77, 39); + expect(MercatorCoordinate.fromLngLat(dc, 500).toLngLat()).toEqual({lng: -77, lat: 39}); + }); + + test('#toAltitude', () => { + const dc = new LngLat(-77, 39); + expect(MercatorCoordinate.fromLngLat(dc, 500).toAltitude()).toBe(500); + }); + + test('#mercatorScale', () => { + expect(mercatorScale(0)).toBe(1); + expect(mercatorScale(45)).toBe(1.414213562373095); + }); + + test('#meterInMercatorCoordinateUnits', () => { + const nullIsland = new LngLat(0, 0); + expect(MercatorCoordinate.fromLngLat(nullIsland).meterInMercatorCoordinateUnits()).toBe(2.4981121214570498e-8); + }); +}); diff --git a/web/libraries/maplibre-gl/src/geo/mercator_coordinate.ts b/web/libraries/maplibre-gl/src/geo/mercator_coordinate.ts new file mode 100644 index 00000000..47945cc9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/geo/mercator_coordinate.ts @@ -0,0 +1,156 @@ +import {LngLat, earthRadius} from '../geo/lng_lat'; +import type {LngLatLike} from '../geo/lng_lat'; +import {IMercatorCoordinate} from '@maplibre/maplibre-gl-style-spec'; + +/* + * The average circumference of the world in meters. + */ +const earthCircumfrence = 2 * Math.PI * earthRadius; // meters + +/* + * The circumference at a line of latitude in meters. + */ +function circumferenceAtLatitude(latitude: number) { + return earthCircumfrence * Math.cos(latitude * Math.PI / 180); +} + +export function mercatorXfromLng(lng: number) { + return (180 + lng) / 360; +} + +export function mercatorYfromLat(lat: number) { + return (180 - (180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360)))) / 360; +} + +export function mercatorZfromAltitude(altitude: number, lat: number) { + return altitude / circumferenceAtLatitude(lat); +} + +export function lngFromMercatorX(x: number) { + return x * 360 - 180; +} + +export function latFromMercatorY(y: number) { + const y2 = 180 - y * 360; + return 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90; +} + +export function altitudeFromMercatorZ(z: number, y: number) { + return z * circumferenceAtLatitude(latFromMercatorY(y)); +} + +/** + * Determine the Mercator scale factor for a given latitude, see + * https://en.wikipedia.org/wiki/Mercator_projection#Scale_factor + * + * At the equator the scale factor will be 1, which increases at higher latitudes. + * + * @param lat - Latitude + * @returns scale factor + */ +export function mercatorScale(lat: number) { + return 1 / Math.cos(lat * Math.PI / 180); +} + +/** + * A `MercatorCoordinate` object represents a projected three dimensional position. + * + * `MercatorCoordinate` uses the web mercator projection ([EPSG:3857](https://epsg.io/3857)) with slightly different units: + * - the size of 1 unit is the width of the projected world instead of the "mercator meter" + * - the origin of the coordinate space is at the north-west corner instead of the middle + * + * For example, `MercatorCoordinate(0, 0, 0)` is the north-west corner of the mercator world and + * `MercatorCoordinate(1, 1, 0)` is the south-east corner. If you are familiar with + * [vector tiles](https://github.com/mapbox/vector-tile-spec) it may be helpful to think + * of the coordinate space as the `0/0/0` tile with an extent of `1`. + * + * The `z` dimension of `MercatorCoordinate` is conformal. A cube in the mercator coordinate space would be rendered as a cube. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let nullIsland = new maplibregl.MercatorCoordinate(0.5, 0.5, 0); + * ``` + * @see [Add a custom style layer](https://maplibre.org/maplibre-gl-js/docs/examples/custom-style-layer/) + */ +export class MercatorCoordinate implements IMercatorCoordinate { + x: number; + y: number; + z: number; + + /** + * @param x - The x component of the position. + * @param y - The y component of the position. + * @param z - The z component of the position. + */ + constructor(x: number, y: number, z: number = 0) { + this.x = +x; + this.y = +y; + this.z = +z; + } + + /** + * Project a `LngLat` to a `MercatorCoordinate`. + * + * @param lngLatLike - The location to project. + * @param altitude - The altitude in meters of the position. + * @returns The projected mercator coordinate. + * @example + * ```ts + * let coord = maplibregl.MercatorCoordinate.fromLngLat({ lng: 0, lat: 0}, 0); + * coord; // MercatorCoordinate(0.5, 0.5, 0) + * ``` + */ + static fromLngLat(lngLatLike: LngLatLike, altitude: number = 0): MercatorCoordinate { + const lngLat = LngLat.convert(lngLatLike); + + return new MercatorCoordinate( + mercatorXfromLng(lngLat.lng), + mercatorYfromLat(lngLat.lat), + mercatorZfromAltitude(altitude, lngLat.lat)); + } + + /** + * Returns the `LngLat` for the coordinate. + * + * @returns The `LngLat` object. + * @example + * ```ts + * let coord = new maplibregl.MercatorCoordinate(0.5, 0.5, 0); + * let lngLat = coord.toLngLat(); // LngLat(0, 0) + * ``` + */ + toLngLat() { + return new LngLat( + lngFromMercatorX(this.x), + latFromMercatorY(this.y)); + } + + /** + * Returns the altitude in meters of the coordinate. + * + * @returns The altitude in meters. + * @example + * ```ts + * let coord = new maplibregl.MercatorCoordinate(0, 0, 0.02); + * coord.toAltitude(); // 6914.281956295339 + * ``` + */ + toAltitude(): number { + return altitudeFromMercatorZ(this.z, this.y); + } + + /** + * Returns the distance of 1 meter in `MercatorCoordinate` units at this latitude. + * + * For coordinates in real world units using meters, this naturally provides the scale + * to transform into `MercatorCoordinate`s. + * + * @returns Distance of 1 meter in `MercatorCoordinate` units. + */ + meterInMercatorCoordinateUnits(): number { + // 1 meter / circumference at equator in meters * Mercator projection scale factor at this latitude + return 1 / earthCircumfrence * mercatorScale(latFromMercatorY(this.y)); + } +} diff --git a/web/libraries/maplibre-gl/src/geo/transform.test.ts b/web/libraries/maplibre-gl/src/geo/transform.test.ts new file mode 100644 index 00000000..065a0d4d --- /dev/null +++ b/web/libraries/maplibre-gl/src/geo/transform.test.ts @@ -0,0 +1,464 @@ +import Point from '@mapbox/point-geometry'; +import {Transform} from './transform'; +import {LngLat} from './lng_lat'; +import {OverscaledTileID, CanonicalTileID} from '../source/tile_id'; +import {fixedLngLat, fixedCoord} from '../../test/unit/lib/fixed'; +import type {Terrain} from '../render/terrain'; + +describe('transform', () => { + test('creates a transform', () => { + const transform = new Transform(0, 22, 0, 60, true); + transform.resize(500, 500); + expect(transform.unmodified).toBe(true); + expect(transform.maxValidLatitude).toBe(85.051129); + expect(transform.tileSize).toBe(512); + expect(transform.worldSize).toBe(512); + expect(transform.width).toBe(500); + expect(transform.minZoom).toBe(0); + expect(transform.minPitch).toBe(0); + // Support signed zero + expect(transform.bearing === 0 ? 0 : transform.bearing).toBe(0); + expect(transform.bearing = 1).toBe(1); + expect(transform.bearing).toBe(1); + expect(transform.bearing = 0).toBe(0); + expect(transform.unmodified).toBe(false); + expect(transform.minZoom = 10).toBe(10); + expect(transform.maxZoom = 10).toBe(10); + expect(transform.minZoom).toBe(10); + expect(transform.center).toEqual({lng: 0, lat: 0}); + expect(transform.maxZoom).toBe(10); + expect(transform.minPitch = 10).toBe(10); + expect(transform.maxPitch = 10).toBe(10); + expect(transform.size.equals(new Point(500, 500))).toBe(true); + expect(transform.centerPoint.equals(new Point(250, 250))).toBe(true); + expect(transform.scaleZoom(0)).toBe(-Infinity); + expect(transform.scaleZoom(10)).toBe(3.3219280948873626); + expect(transform.point).toEqual(new Point(262144, 262144)); + expect(transform.height).toBe(500); + expect(fixedLngLat(transform.pointLocation(new Point(250, 250)))).toEqual({lng: 0, lat: 0}); + expect(fixedCoord(transform.pointCoordinate(new Point(250, 250)))).toEqual({x: 0.5, y: 0.5, z: 0}); + expect(transform.locationPoint(new LngLat(0, 0))).toEqual({x: 250, y: 250}); + expect(transform.locationCoordinate(new LngLat(0, 0))).toEqual({x: 0.5, y: 0.5, z: 0}); + }); + + test('does not throw on bad center', () => { + expect(() => { + const transform = new Transform(0, 22, 0, 60, true); + transform.resize(500, 500); + transform.center = new LngLat(50, -90); + }).not.toThrow(); + }); + + test('setLocationAt', () => { + const transform = new Transform(0, 22, 0, 60, true); + transform.resize(500, 500); + transform.zoom = 4; + expect(transform.center).toEqual({lng: 0, lat: 0}); + transform.setLocationAtPoint(new LngLat(13, 10), new Point(15, 45)); + expect(fixedLngLat(transform.pointLocation(new Point(15, 45)))).toEqual({lng: 13, lat: 10}); + }); + + test('setLocationAt tilted', () => { + const transform = new Transform(0, 22, 0, 60, true); + transform.resize(500, 500); + transform.zoom = 4; + transform.pitch = 50; + expect(transform.center).toEqual({lng: 0, lat: 0}); + transform.setLocationAtPoint(new LngLat(13, 10), new Point(15, 45)); + expect(fixedLngLat(transform.pointLocation(new Point(15, 45)))).toEqual({lng: 13, lat: 10}); + }); + + test('has a default zoom', () => { + const transform = new Transform(0, 22, 0, 60, true); + transform.resize(500, 500); + expect(transform.tileZoom).toBe(0); + expect(transform.tileZoom).toBe(transform.zoom); + }); + + test('set zoom inits tileZoom with zoom value', () => { + const transform = new Transform(0, 22, 0, 60); + transform.zoom = 5; + expect(transform.tileZoom).toBe(5); + }); + + test('set zoom clamps tileZoom to non negative value ', () => { + const transform = new Transform(-2, 22, 0, 60); + transform.zoom = -2; + expect(transform.tileZoom).toBe(0); + }); + + test('set fov', () => { + const transform = new Transform(0, 22, 0, 60, true); + transform.fov = 10; + expect(transform.fov).toBe(10); + transform.fov = 10; + expect(transform.fov).toBe(10); + }); + + test('lngRange & latRange constrain zoom and center', () => { + const transform = new Transform(0, 22, 0, 60, true); + transform.center = new LngLat(0, 0); + transform.zoom = 10; + transform.resize(500, 500); + + transform.lngRange = [-5, 5]; + transform.latRange = [-5, 5]; + + transform.zoom = 0; + expect(transform.zoom).toBe(5.1357092861044045); + + transform.center = new LngLat(-50, -30); + expect(transform.center).toEqual(new LngLat(0, -0.0063583052861417855)); + + transform.zoom = 10; + transform.center = new LngLat(-50, -30); + expect(transform.center).toEqual(new LngLat(-4.828338623046875, -4.828969771321582)); + }); + + test('lngRange can constrain zoom and center across meridian', () => { + const transform = new Transform(0, 22, 0, 60, true); + transform.center = new LngLat(180, 0); + transform.zoom = 10; + transform.resize(500, 500); + + // equivalent ranges + const lngRanges: [number, number][] = [ + [175, -175], [175, 185], [-185, -175], [-185, 185] + ]; + + for (const lngRange of lngRanges) { + transform.lngRange = lngRange; + transform.latRange = [-5, 5]; + + transform.zoom = 0; + expect(transform.zoom).toBe(5.1357092861044045); + + transform.center = new LngLat(-50, -30); + expect(transform.center).toEqual(new LngLat(180, -0.0063583052861417855)); + + transform.zoom = 10; + transform.center = new LngLat(-50, -30); + expect(transform.center).toEqual(new LngLat(-175.171661376953125, -4.828969771321582)); + + transform.center = new LngLat(230, 0); + expect(transform.center).toEqual(new LngLat(-175.171661376953125, 0)); + + transform.center = new LngLat(130, 0); + expect(transform.center).toEqual(new LngLat(175.171661376953125, 0)); + } + }); + + describe('coveringTiles', () => { + const options = { + minzoom: 1, + maxzoom: 10, + tileSize: 512 + }; + + const transform = new Transform(0, 22, 0, 60, true); + transform.resize(200, 200); + + test('generell', () => { + + // make slightly off center so that sort order is not subject to precision issues + transform.center = new LngLat(-0.01, 0.01); + + transform.zoom = 0; + expect(transform.coveringTiles(options)).toEqual([]); + + transform.zoom = 1; + expect(transform.coveringTiles(options)).toEqual([ + new OverscaledTileID(1, 0, 1, 0, 0), + new OverscaledTileID(1, 0, 1, 1, 0), + new OverscaledTileID(1, 0, 1, 0, 1), + new OverscaledTileID(1, 0, 1, 1, 1)]); + + transform.zoom = 2.4; + expect(transform.coveringTiles(options)).toEqual([ + new OverscaledTileID(2, 0, 2, 1, 1), + new OverscaledTileID(2, 0, 2, 2, 1), + new OverscaledTileID(2, 0, 2, 1, 2), + new OverscaledTileID(2, 0, 2, 2, 2)]); + + transform.zoom = 10; + expect(transform.coveringTiles(options)).toEqual([ + new OverscaledTileID(10, 0, 10, 511, 511), + new OverscaledTileID(10, 0, 10, 512, 511), + new OverscaledTileID(10, 0, 10, 511, 512), + new OverscaledTileID(10, 0, 10, 512, 512)]); + + transform.zoom = 11; + expect(transform.coveringTiles(options)).toEqual([ + new OverscaledTileID(10, 0, 10, 511, 511), + new OverscaledTileID(10, 0, 10, 512, 511), + new OverscaledTileID(10, 0, 10, 511, 512), + new OverscaledTileID(10, 0, 10, 512, 512)]); + + transform.zoom = 5.1; + transform.pitch = 60.0; + transform.bearing = 32.0; + transform.center = new LngLat(56.90, 48.20); + transform.resize(1024, 768); + expect(transform.coveringTiles(options)).toEqual([ + new OverscaledTileID(5, 0, 5, 21, 11), + new OverscaledTileID(5, 0, 5, 20, 11), + new OverscaledTileID(5, 0, 5, 21, 10), + new OverscaledTileID(5, 0, 5, 20, 10), + new OverscaledTileID(5, 0, 5, 21, 12), + new OverscaledTileID(5, 0, 5, 22, 11), + new OverscaledTileID(5, 0, 5, 20, 12), + new OverscaledTileID(5, 0, 5, 22, 10), + new OverscaledTileID(5, 0, 5, 21, 9), + new OverscaledTileID(5, 0, 5, 20, 9), + new OverscaledTileID(5, 0, 5, 22, 9), + new OverscaledTileID(5, 0, 5, 23, 10), + new OverscaledTileID(5, 0, 5, 21, 8), + new OverscaledTileID(5, 0, 5, 20, 8), + new OverscaledTileID(5, 0, 5, 23, 9), + new OverscaledTileID(5, 0, 5, 22, 8), + new OverscaledTileID(5, 0, 5, 23, 8), + new OverscaledTileID(5, 0, 5, 21, 7), + new OverscaledTileID(5, 0, 5, 20, 7), + new OverscaledTileID(5, 0, 5, 24, 9), + new OverscaledTileID(5, 0, 5, 22, 7) + ]); + + transform.zoom = 8; + transform.pitch = 60; + transform.bearing = 45.0; + transform.center = new LngLat(25.02, 60.15); + transform.resize(300, 50); + expect(transform.coveringTiles(options)).toEqual([ + new OverscaledTileID(8, 0, 8, 145, 74), + new OverscaledTileID(8, 0, 8, 145, 73), + new OverscaledTileID(8, 0, 8, 146, 74) + ]); + + transform.resize(50, 300); + expect(transform.coveringTiles(options)).toEqual([ + new OverscaledTileID(8, 0, 8, 145, 74), + new OverscaledTileID(8, 0, 8, 145, 73), + new OverscaledTileID(8, 0, 8, 146, 74), + new OverscaledTileID(8, 0, 8, 146, 73) + ]); + + transform.zoom = 2; + transform.pitch = 0; + transform.bearing = 0; + transform.resize(300, 300); + }); + + test('calculates tile coverage at w > 0', () => { + transform.center = new LngLat(630.01, 0.01); + expect(transform.coveringTiles(options)).toEqual([ + new OverscaledTileID(2, 2, 2, 1, 1), + new OverscaledTileID(2, 2, 2, 1, 2), + new OverscaledTileID(2, 2, 2, 0, 1), + new OverscaledTileID(2, 2, 2, 0, 2) + ]); + }); + + test('calculates tile coverage at w = -1', () => { + transform.center = new LngLat(-360.01, 0.01); + expect(transform.coveringTiles(options)).toEqual([ + new OverscaledTileID(2, -1, 2, 1, 1), + new OverscaledTileID(2, -1, 2, 1, 2), + new OverscaledTileID(2, -1, 2, 2, 1), + new OverscaledTileID(2, -1, 2, 2, 2) + ]); + }); + + test('calculates tile coverage across meridian', () => { + transform.zoom = 1; + transform.center = new LngLat(-180.01, 0.01); + expect(transform.coveringTiles(options)).toEqual([ + new OverscaledTileID(1, 0, 1, 0, 0), + new OverscaledTileID(1, 0, 1, 0, 1), + new OverscaledTileID(1, -1, 1, 1, 0), + new OverscaledTileID(1, -1, 1, 1, 1) + ]); + }); + + test('only includes tiles for a single world, if renderWorldCopies is set to false', () => { + transform.zoom = 1; + transform.center = new LngLat(-180.01, 0.01); + transform.renderWorldCopies = false; + expect(transform.coveringTiles(options)).toEqual([ + new OverscaledTileID(1, 0, 1, 0, 0), + new OverscaledTileID(1, 0, 1, 0, 1) + ]); + }); + + }); + + test('coveringZoomLevel', () => { + const options = { + minzoom: 1, + maxzoom: 10, + tileSize: 512, + roundZoom: false, + }; + + const transform = new Transform(0, 22, 0, 60, true); + + transform.zoom = 0; + expect(transform.coveringZoomLevel(options)).toBe(0); + + transform.zoom = 0.1; + expect(transform.coveringZoomLevel(options)).toBe(0); + + transform.zoom = 1; + expect(transform.coveringZoomLevel(options)).toBe(1); + + transform.zoom = 2.4; + expect(transform.coveringZoomLevel(options)).toBe(2); + + transform.zoom = 10; + expect(transform.coveringZoomLevel(options)).toBe(10); + + transform.zoom = 11; + expect(transform.coveringZoomLevel(options)).toBe(11); + + transform.zoom = 11.5; + expect(transform.coveringZoomLevel(options)).toBe(11); + + options.tileSize = 256; + + transform.zoom = 0; + expect(transform.coveringZoomLevel(options)).toBe(1); + + transform.zoom = 0.1; + expect(transform.coveringZoomLevel(options)).toBe(1); + + transform.zoom = 1; + expect(transform.coveringZoomLevel(options)).toBe(2); + + transform.zoom = 2.4; + expect(transform.coveringZoomLevel(options)).toBe(3); + + transform.zoom = 10; + expect(transform.coveringZoomLevel(options)).toBe(11); + + transform.zoom = 11; + expect(transform.coveringZoomLevel(options)).toBe(12); + + transform.zoom = 11.5; + expect(transform.coveringZoomLevel(options)).toBe(12); + + options.roundZoom = true; + + expect(transform.coveringZoomLevel(options)).toBe(13); + }); + + test('clamps latitude', () => { + const transform = new Transform(0, 22, 0, 60, true); + + expect(transform.project(new LngLat(0, -90))).toEqual(transform.project(new LngLat(0, -transform.maxValidLatitude))); + expect(transform.project(new LngLat(0, 90))).toEqual(transform.project(new LngLat(0, transform.maxValidLatitude))); + }); + + test('clamps pitch', () => { + const transform = new Transform(0, 22, 0, 60, true); + + transform.pitch = 45; + expect(transform.pitch).toBe(45); + + transform.pitch = -10; + expect(transform.pitch).toBe(0); + + transform.pitch = 90; + expect(transform.pitch).toBe(60); + }); + + test('visibleUnwrappedCoordinates', () => { + const transform = new Transform(0, 22, 0, 60, true); + transform.resize(200, 200); + transform.zoom = 0; + transform.center = new LngLat(-170.01, 0.01); + + let unwrappedCoords = transform.getVisibleUnwrappedCoordinates(new CanonicalTileID(0, 0, 0)); + expect(unwrappedCoords).toHaveLength(4); + + //getVisibleUnwrappedCoordinates should honor _renderWorldCopies + transform._renderWorldCopies = false; + unwrappedCoords = transform.getVisibleUnwrappedCoordinates(new CanonicalTileID(0, 0, 0)); + expect(unwrappedCoords).toHaveLength(1); + }); + + test('maintains high float precision when calculating matrices', () => { + const transform = new Transform(0, 22, 0, 60, true); + transform.resize(200.25, 200.25); + transform.zoom = 20.25; + transform.pitch = 67.25; + transform.center = new LngLat(0.0, 0.0); + transform._calcMatrices(); + + expect(transform.customLayerMatrix()[0].toString().length).toBeGreaterThan(10); + expect(transform.glCoordMatrix[0].toString().length).toBeGreaterThan(10); + expect(transform.maxPitchScaleFactor()).toBeCloseTo(2.366025418080343, 10); + }); + + test('recalcuateZoom', () => { + const transform = new Transform(0, 22, 0, 60, true); + transform.elevation = 200; + transform.center = new LngLat(10.0, 50.0); + transform.zoom = 14; + transform.resize(512, 512); + + // expect same values because of no elevation change + const terrain = { + getElevationForLngLatZoom: () => 200, + pointCoordinate: () => null + }; + transform.recalculateZoom(terrain as any); + expect(transform.zoom).toBe(14); + + // expect new zoom because of elevation change + terrain.getElevationForLngLatZoom = () => 400; + transform.recalculateZoom(terrain as any); + expect(transform.zoom).toBe(14.127997275621933); + expect(transform.elevation).toBe(400); + + expect(transform._center.lng).toBe(10.00000000000071); + expect(transform._center.lat).toBe(50.00000000000017); + + // expect new zoom because of elevation change to point below sea level + terrain.getElevationForLngLatZoom = () => -200; + transform.recalculateZoom(terrain as any); + expect(transform.zoom).toBe(13.773740316343467); + expect(transform.elevation).toBe(-200); + }); + + test('pointCoordinate with terrain when returning null should fall back to 2D', () => { + const transform = new Transform(0, 22, 0, 60, true); + transform.resize(500, 500); + const terrain = { + pointCoordinate: () => null + } as any as Terrain; + const coordinate = transform.pointCoordinate(new Point(0, 0), terrain); + + expect(coordinate).toBeDefined(); + }); + + test('horizon', () => { + const transform = new Transform(0, 22, 0, 85, true); + transform.resize(500, 500); + transform.pitch = 75; + const horizon = transform.getHorizon(); + + expect(horizon).toBeCloseTo(170.8176101748407, 10); + }); + + test('getBounds with horizon', () => { + const transform = new Transform(0, 22, 0, 85, true); + transform.resize(500, 500); + + transform.pitch = 60; + expect(transform.getBounds().getNorthWest().toArray()).toStrictEqual(transform.pointLocation(new Point(0, 0)).toArray()); + + transform.pitch = 75; + const top = Math.max(0, transform.height / 2 - transform.getHorizon()); + expect(top).toBeCloseTo(79.1823898251593, 10); + expect(transform.getBounds().getNorthWest().toArray()).toStrictEqual(transform.pointLocation(new Point(0, top)).toArray()); + }); +}); diff --git a/web/libraries/maplibre-gl/src/geo/transform.ts b/web/libraries/maplibre-gl/src/geo/transform.ts new file mode 100644 index 00000000..b2f8ad96 --- /dev/null +++ b/web/libraries/maplibre-gl/src/geo/transform.ts @@ -0,0 +1,958 @@ +import {LngLat} from './lng_lat'; +import {LngLatBounds} from './lng_lat_bounds'; +import {MercatorCoordinate, mercatorXfromLng, mercatorYfromLat, mercatorZfromAltitude} from './mercator_coordinate'; +import Point from '@mapbox/point-geometry'; +import {wrap, clamp} from '../util/util'; +import {interpolates} from '@maplibre/maplibre-gl-style-spec'; +import {EXTENT} from '../data/extent'; +import {vec3, vec4, mat4, mat2, vec2} from 'gl-matrix'; +import {Aabb, Frustum} from '../util/primitives'; +import {EdgeInsets} from './edge_insets'; + +import {UnwrappedTileID, OverscaledTileID, CanonicalTileID} from '../source/tile_id'; +import type {PaddingOptions} from './edge_insets'; +import {Terrain} from '../render/terrain'; + +/** + * @internal + * A single transform, generally used for a single tile to be + * scaled, rotated, and zoomed. + */ +export class Transform { + tileSize: number; + tileZoom: number; + lngRange: [number, number]; + latRange: [number, number]; + maxValidLatitude: number; + scale: number; + width: number; + height: number; + angle: number; + rotationMatrix: mat2; + pixelsToGLUnits: [number, number]; + cameraToCenterDistance: number; + mercatorMatrix: mat4; + projMatrix: mat4; + invProjMatrix: mat4; + alignedProjMatrix: mat4; + pixelMatrix: mat4; + pixelMatrix3D: mat4; + pixelMatrixInverse: mat4; + glCoordMatrix: mat4; + labelPlaneMatrix: mat4; + _fov: number; + _pitch: number; + _zoom: number; + _unmodified: boolean; + _renderWorldCopies: boolean; + _minZoom: number; + _maxZoom: number; + _minPitch: number; + _maxPitch: number; + _center: LngLat; + _elevation: number; + _pixelPerMeter: number; + _edgeInsets: EdgeInsets; + _constraining: boolean; + _posMatrixCache: {[_: string]: mat4}; + _alignedPosMatrixCache: {[_: string]: mat4}; + _minEleveationForCurrentTile: number; + + constructor(minZoom?: number, maxZoom?: number, minPitch?: number, maxPitch?: number, renderWorldCopies?: boolean) { + this.tileSize = 512; // constant + this.maxValidLatitude = 85.051129; // constant + + this._renderWorldCopies = renderWorldCopies === undefined ? true : !!renderWorldCopies; + this._minZoom = minZoom || 0; + this._maxZoom = maxZoom || 22; + + this._minPitch = (minPitch === undefined || minPitch === null) ? 0 : minPitch; + this._maxPitch = (maxPitch === undefined || maxPitch === null) ? 60 : maxPitch; + + this.setMaxBounds(); + + this.width = 0; + this.height = 0; + this._center = new LngLat(0, 0); + this._elevation = 0; + this.zoom = 0; + this.angle = 0; + this._fov = 0.6435011087932844; + this._pitch = 0; + this._unmodified = true; + this._edgeInsets = new EdgeInsets(); + this._posMatrixCache = {}; + this._alignedPosMatrixCache = {}; + this._minEleveationForCurrentTile = 0; + } + + clone(): Transform { + const clone = new Transform(this._minZoom, this._maxZoom, this._minPitch, this.maxPitch, this._renderWorldCopies); + clone.apply(this); + return clone; + } + + apply(that: Transform) { + this.tileSize = that.tileSize; + this.latRange = that.latRange; + this.width = that.width; + this.height = that.height; + this._center = that._center; + this._elevation = that._elevation; + this._minEleveationForCurrentTile = that._minEleveationForCurrentTile; + this.zoom = that.zoom; + this.angle = that.angle; + this._fov = that._fov; + this._pitch = that._pitch; + this._unmodified = that._unmodified; + this._edgeInsets = that._edgeInsets.clone(); + this._calcMatrices(); + } + + get minZoom(): number { return this._minZoom; } + set minZoom(zoom: number) { + if (this._minZoom === zoom) return; + this._minZoom = zoom; + this.zoom = Math.max(this.zoom, zoom); + } + + get maxZoom(): number { return this._maxZoom; } + set maxZoom(zoom: number) { + if (this._maxZoom === zoom) return; + this._maxZoom = zoom; + this.zoom = Math.min(this.zoom, zoom); + } + + get minPitch(): number { return this._minPitch; } + set minPitch(pitch: number) { + if (this._minPitch === pitch) return; + this._minPitch = pitch; + this.pitch = Math.max(this.pitch, pitch); + } + + get maxPitch(): number { return this._maxPitch; } + set maxPitch(pitch: number) { + if (this._maxPitch === pitch) return; + this._maxPitch = pitch; + this.pitch = Math.min(this.pitch, pitch); + } + + get renderWorldCopies(): boolean { return this._renderWorldCopies; } + set renderWorldCopies(renderWorldCopies: boolean) { + if (renderWorldCopies === undefined) { + renderWorldCopies = true; + } else if (renderWorldCopies === null) { + renderWorldCopies = false; + } + + this._renderWorldCopies = renderWorldCopies; + } + + get worldSize(): number { + return this.tileSize * this.scale; + } + + get centerOffset(): Point { + return this.centerPoint._sub(this.size._div(2)); + } + + get size(): Point { + return new Point(this.width, this.height); + } + + get bearing(): number { + return -this.angle / Math.PI * 180; + } + set bearing(bearing: number) { + const b = -wrap(bearing, -180, 180) * Math.PI / 180; + if (this.angle === b) return; + this._unmodified = false; + this.angle = b; + this._calcMatrices(); + + // 2x2 matrix for rotating points + this.rotationMatrix = mat2.create(); + mat2.rotate(this.rotationMatrix, this.rotationMatrix, this.angle); + } + + get pitch(): number { + return this._pitch / Math.PI * 180; + } + set pitch(pitch: number) { + const p = clamp(pitch, this.minPitch, this.maxPitch) / 180 * Math.PI; + if (this._pitch === p) return; + this._unmodified = false; + this._pitch = p; + this._calcMatrices(); + } + + get fov(): number { + return this._fov / Math.PI * 180; + } + set fov(fov: number) { + fov = Math.max(0.01, Math.min(60, fov)); + if (this._fov === fov) return; + this._unmodified = false; + this._fov = fov / 180 * Math.PI; + this._calcMatrices(); + } + + get zoom(): number { return this._zoom; } + set zoom(zoom: number) { + const constrainedZoom = Math.min(Math.max(zoom, this.minZoom), this.maxZoom); + if (this._zoom === constrainedZoom) return; + this._unmodified = false; + this._zoom = constrainedZoom; + this.tileZoom = Math.max(0, Math.floor(constrainedZoom)); + this.scale = this.zoomScale(constrainedZoom); + this._constrain(); + this._calcMatrices(); + } + + get center(): LngLat { return this._center; } + set center(center: LngLat) { + if (center.lat === this._center.lat && center.lng === this._center.lng) return; + this._unmodified = false; + this._center = center; + this._constrain(); + this._calcMatrices(); + } + + get elevation(): number { return this._elevation; } + set elevation(elevation: number) { + if (elevation === this._elevation) return; + this._elevation = elevation; + this._constrain(); + this._calcMatrices(); + } + + get padding(): PaddingOptions { return this._edgeInsets.toJSON(); } + set padding(padding: PaddingOptions) { + if (this._edgeInsets.equals(padding)) return; + this._unmodified = false; + //Update edge-insets inplace + this._edgeInsets.interpolate(this._edgeInsets, padding, 1); + this._calcMatrices(); + } + + /** + * The center of the screen in pixels with the top-left corner being (0,0) + * and +y axis pointing downwards. This accounts for padding. + */ + get centerPoint(): Point { + return this._edgeInsets.getCenter(this.width, this.height); + } + + /** + * Returns if the padding params match + * + * @param padding - the padding to check against + * @returns true if they are equal, false otherwise + */ + isPaddingEqual(padding: PaddingOptions): boolean { + return this._edgeInsets.equals(padding); + } + + /** + * Helper method to update edge-insets in place + * + * @param start - the starting padding + * @param target - the target padding + * @param t - the step/weight + */ + interpolatePadding(start: PaddingOptions, target: PaddingOptions, t: number) { + this._unmodified = false; + this._edgeInsets.interpolate(start, target, t); + this._constrain(); + this._calcMatrices(); + } + + /** + * Return a zoom level that will cover all tiles the transform + * @param options - the options + * @returns zoom level An integer zoom level at which all tiles will be visible. + */ + coveringZoomLevel(options: { + /** + * Target zoom level. If true, the value will be rounded to the closest integer. Otherwise the value will be floored. + */ + roundZoom?: boolean; + /** + * Tile size, expressed in screen pixels. + */ + tileSize: number; + }): number { + const z = (options.roundZoom ? Math.round : Math.floor)( + this.zoom + this.scaleZoom(this.tileSize / options.tileSize) + ); + // At negative zoom levels load tiles from z0 because negative tile zoom levels don't exist. + return Math.max(0, z); + } + + /** + * Return any "wrapped" copies of a given tile coordinate that are visible + * in the current view. + */ + getVisibleUnwrappedCoordinates(tileID: CanonicalTileID) { + const result = [new UnwrappedTileID(0, tileID)]; + if (this._renderWorldCopies) { + const utl = this.pointCoordinate(new Point(0, 0)); + const utr = this.pointCoordinate(new Point(this.width, 0)); + const ubl = this.pointCoordinate(new Point(this.width, this.height)); + const ubr = this.pointCoordinate(new Point(0, this.height)); + const w0 = Math.floor(Math.min(utl.x, utr.x, ubl.x, ubr.x)); + const w1 = Math.floor(Math.max(utl.x, utr.x, ubl.x, ubr.x)); + + // Add an extra copy of the world on each side to properly render ImageSources and CanvasSources. + // Both sources draw outside the tile boundaries of the tile that "contains them" so we need + // to add extra copies on both sides in case offscreen tiles need to draw into on-screen ones. + const extraWorldCopy = 1; + + for (let w = w0 - extraWorldCopy; w <= w1 + extraWorldCopy; w++) { + if (w === 0) continue; + result.push(new UnwrappedTileID(w, tileID)); + } + } + return result; + } + + /** + * Return all coordinates that could cover this transform for a covering + * zoom level. + * @param options - the options + * @returns OverscaledTileIDs + */ + coveringTiles( + options: { + tileSize: number; + minzoom?: number; + maxzoom?: number; + roundZoom?: boolean; + reparseOverscaled?: boolean; + renderWorldCopies?: boolean; + terrain?: Terrain; + } + ): Array { + let z = this.coveringZoomLevel(options); + const actualZ = z; + + if (options.minzoom !== undefined && z < options.minzoom) return []; + if (options.maxzoom !== undefined && z > options.maxzoom) z = options.maxzoom; + + const cameraCoord = this.pointCoordinate(this.getCameraPoint()); + const centerCoord = MercatorCoordinate.fromLngLat(this.center); + const numTiles = Math.pow(2, z); + const cameraPoint = [numTiles * cameraCoord.x, numTiles * cameraCoord.y, 0]; + const centerPoint = [numTiles * centerCoord.x, numTiles * centerCoord.y, 0]; + const cameraFrustum = Frustum.fromInvProjectionMatrix(this.invProjMatrix, this.worldSize, z); + + // No change of LOD behavior for pitch lower than 60 and when there is no top padding: return only tile ids from the requested zoom level + let minZoom = options.minzoom || 0; + // Use 0.1 as an epsilon to avoid for explicit == 0.0 floating point checks + if (!options.terrain && this.pitch <= 60.0 && this._edgeInsets.top < 0.1) + minZoom = z; + + // There should always be a certain number of maximum zoom level tiles surrounding the center location in 2D or in front of the camera in 3D + const radiusOfMaxLvlLodInTiles = options.terrain ? 2 / Math.min(this.tileSize, options.tileSize) * this.tileSize : 3; + + const newRootTile = (wrap: number): any => { + return { + aabb: new Aabb([wrap * numTiles, 0, 0], [(wrap + 1) * numTiles, numTiles, 0]), + zoom: 0, + x: 0, + y: 0, + wrap, + fullyVisible: false + }; + }; + + // Do a depth-first traversal to find visible tiles and proper levels of detail + const stack = []; + const result = []; + const maxZoom = z; + const overscaledZ = options.reparseOverscaled ? actualZ : z; + + if (this._renderWorldCopies) { + // Render copy of the globe thrice on both sides + for (let i = 1; i <= 3; i++) { + stack.push(newRootTile(-i)); + stack.push(newRootTile(i)); + } + } + + stack.push(newRootTile(0)); + + while (stack.length > 0) { + const it = stack.pop(); + const x = it.x; + const y = it.y; + let fullyVisible = it.fullyVisible; + + // Visibility of a tile is not required if any of its ancestor if fully inside the frustum + if (!fullyVisible) { + const intersectResult = it.aabb.intersects(cameraFrustum); + + if (intersectResult === 0) + continue; + + fullyVisible = intersectResult === 2; + } + + const refPoint = options.terrain ? cameraPoint : centerPoint; + const distanceX = it.aabb.distanceX(refPoint); + const distanceY = it.aabb.distanceY(refPoint); + const longestDim = Math.max(Math.abs(distanceX), Math.abs(distanceY)); + + // We're using distance based heuristics to determine if a tile should be split into quadrants or not. + // radiusOfMaxLvlLodInTiles defines that there's always a certain number of maxLevel tiles next to the map center. + // Using the fact that a parent node in quadtree is twice the size of its children (per dimension) + // we can define distance thresholds for each relative level: + // f(k) = offset + 2 + 4 + 8 + 16 + ... + 2^k. This is the same as "offset+2^(k+1)-2" + const distToSplit = radiusOfMaxLvlLodInTiles + (1 << (maxZoom - it.zoom)) - 2; + + // Have we reached the target depth or is the tile too far away to be any split further? + if (it.zoom === maxZoom || (longestDim > distToSplit && it.zoom >= minZoom)) { + const dz = maxZoom - it.zoom, dx = cameraPoint[0] - 0.5 - (x << dz), dy = cameraPoint[1] - 0.5 - (y << dz); + result.push({ + tileID: new OverscaledTileID(it.zoom === maxZoom ? overscaledZ : it.zoom, it.wrap, it.zoom, x, y), + distanceSq: vec2.sqrLen([centerPoint[0] - 0.5 - x, centerPoint[1] - 0.5 - y]), + // this variable is currently not used, but may be important to reduce the amount of loaded tiles + tileDistanceToCamera: Math.sqrt(dx * dx + dy * dy) + }); + continue; + } + + for (let i = 0; i < 4; i++) { + const childX = (x << 1) + (i % 2); + const childY = (y << 1) + (i >> 1); + const childZ = it.zoom + 1; + let quadrant = it.aabb.quadrant(i); + if (options.terrain) { + const tileID = new OverscaledTileID(childZ, it.wrap, childZ, childX, childY); + const minMax = options.terrain.getMinMaxElevation(tileID); + const minElevation = minMax.minElevation ?? this.elevation; + const maxElevation = minMax.maxElevation ?? this.elevation; + quadrant = new Aabb( + [quadrant.min[0], quadrant.min[1], minElevation] as vec3, + [quadrant.max[0], quadrant.max[1], maxElevation] as vec3 + ); + } + stack.push({aabb: quadrant, zoom: childZ, x: childX, y: childY, wrap: it.wrap, fullyVisible}); + } + } + + return result.sort((a, b) => a.distanceSq - b.distanceSq).map(a => a.tileID); + } + + resize(width: number, height: number) { + this.width = width; + this.height = height; + + this.pixelsToGLUnits = [2 / width, -2 / height]; + this._constrain(); + this._calcMatrices(); + } + + get unmodified(): boolean { return this._unmodified; } + + zoomScale(zoom: number) { return Math.pow(2, zoom); } + scaleZoom(scale: number) { return Math.log(scale) / Math.LN2; } + + project(lnglat: LngLat) { + const lat = clamp(lnglat.lat, -this.maxValidLatitude, this.maxValidLatitude); + return new Point( + mercatorXfromLng(lnglat.lng) * this.worldSize, + mercatorYfromLat(lat) * this.worldSize); + } + + unproject(point: Point): LngLat { + return new MercatorCoordinate(point.x / this.worldSize, point.y / this.worldSize).toLngLat(); + } + + get point(): Point { return this.project(this.center); } + + /** + * get the camera position in LngLat and altitudes in meter + * @returns An object with lngLat & altitude. + */ + getCameraPosition(): { + lngLat: LngLat; + altitude: number; + } { + const lngLat = this.pointLocation(this.getCameraPoint()); + const altitude = Math.cos(this._pitch) * this.cameraToCenterDistance / this._pixelPerMeter; + return {lngLat, altitude: altitude + this.elevation}; + } + + /** + * This method works in combination with freezeElevation activated. + * freezeElevtion is enabled during map-panning because during this the camera should sit in constant height. + * After panning finished, call this method to recalculate the zoomlevel for the current camera-height in current terrain. + * @param terrain - the terrain + */ + recalculateZoom(terrain: Terrain) { + // find position the camera is looking on + const center = this.pointLocation(this.centerPoint, terrain); + const elevation = terrain.getElevationForLngLatZoom(center, this.tileZoom); + const deltaElevation = this.elevation - elevation; + if (!deltaElevation) return; + + // calculate mercator distance between camera & target + const cameraPosition = this.getCameraPosition(); + const camera = MercatorCoordinate.fromLngLat(cameraPosition.lngLat, cameraPosition.altitude); + const target = MercatorCoordinate.fromLngLat(center, elevation); + const dx = camera.x - target.x, dy = camera.y - target.y, dz = camera.z - target.z; + const distance = Math.sqrt(dx * dx + dy * dy + dz * dz); + + // from this distance we calculate the new zoomlevel + const zoom = this.scaleZoom(this.cameraToCenterDistance / distance / this.tileSize); + + // update matrices + this._elevation = elevation; + this._center = center; + this.zoom = zoom; + } + + setLocationAtPoint(lnglat: LngLat, point: Point) { + const a = this.pointCoordinate(point); + const b = this.pointCoordinate(this.centerPoint); + const loc = this.locationCoordinate(lnglat); + const newCenter = new MercatorCoordinate( + loc.x - (a.x - b.x), + loc.y - (a.y - b.y)); + this.center = this.coordinateLocation(newCenter); + if (this._renderWorldCopies) { + this.center = this.center.wrap(); + } + } + + /** + * Given a location, return the screen point that corresponds to it + * @param lnglat - location + * @param terrain - optional terrain + * @returns screen point + */ + locationPoint(lnglat: LngLat, terrain?: Terrain): Point { + return terrain ? + this.coordinatePoint(this.locationCoordinate(lnglat), terrain.getElevationForLngLatZoom(lnglat, this.tileZoom), this.pixelMatrix3D) : + this.coordinatePoint(this.locationCoordinate(lnglat)); + } + + /** + * Given a point on screen, return its lnglat + * @param p - screen point + * @param terrain - optional terrain + * @returns lnglat location + */ + pointLocation(p: Point, terrain?: Terrain): LngLat { + return this.coordinateLocation(this.pointCoordinate(p, terrain)); + } + + /** + * Given a geographical lnglat, return an unrounded + * coordinate that represents it at this transform's zoom level. + * @param lnglat - the location + * @returns The mercator coordinate + */ + locationCoordinate(lnglat: LngLat): MercatorCoordinate { + return MercatorCoordinate.fromLngLat(lnglat); + } + + /** + * Given a Coordinate, return its geographical position. + * @param coord - mercator coordivates + * @returns lng and lat + */ + coordinateLocation(coord: MercatorCoordinate): LngLat { + return coord && coord.toLngLat(); + } + + /** + * Given a Point, return its mercator coordinate. + * @param p - the point + * @param terrain - optional terrain + * @returns lnglat + */ + pointCoordinate(p: Point, terrain?: Terrain): MercatorCoordinate { + // get point-coordinate from terrain coordinates framebuffer + if (terrain) { + const coordinate = terrain.pointCoordinate(p); + if (coordinate != null) { + return coordinate; + } + } + + // calculate point-coordinate on flat earth + const targetZ = 0; + // since we don't know the correct projected z value for the point, + // unproject two points to get a line and then find the point on that + // line with z=0 + + const coord0 = [p.x, p.y, 0, 1] as any; + const coord1 = [p.x, p.y, 1, 1] as any; + + vec4.transformMat4(coord0, coord0, this.pixelMatrixInverse); + vec4.transformMat4(coord1, coord1, this.pixelMatrixInverse); + + const w0 = coord0[3]; + const w1 = coord1[3]; + const x0 = coord0[0] / w0; + const x1 = coord1[0] / w1; + const y0 = coord0[1] / w0; + const y1 = coord1[1] / w1; + const z0 = coord0[2] / w0; + const z1 = coord1[2] / w1; + + const t = z0 === z1 ? 0 : (targetZ - z0) / (z1 - z0); + + return new MercatorCoordinate( + interpolates.number(x0, x1, t) / this.worldSize, + interpolates.number(y0, y1, t) / this.worldSize); + } + + /** + * Given a coordinate, return the screen point that corresponds to it + * @param coord - the coordinates + * @param elevation - the elevation + * @param pixelMatrix - the pixel matrix + * @returns screen point + */ + coordinatePoint(coord: MercatorCoordinate, elevation: number = 0, pixelMatrix = this.pixelMatrix): Point { + const p = [coord.x * this.worldSize, coord.y * this.worldSize, elevation, 1] as any; + vec4.transformMat4(p, p, pixelMatrix); + return new Point(p[0] / p[3], p[1] / p[3]); + } + + /** + * Returns the map's geographical bounds. When the bearing or pitch is non-zero, the visible region is not + * an axis-aligned rectangle, and the result is the smallest bounds that encompasses the visible region. + * @returns Returns a {@link LngLatBounds} object describing the map's geographical bounds. + */ + getBounds(): LngLatBounds { + const top = Math.max(0, this.height / 2 - this.getHorizon()); + return new LngLatBounds() + .extend(this.pointLocation(new Point(0, top))) + .extend(this.pointLocation(new Point(this.width, top))) + .extend(this.pointLocation(new Point(this.width, this.height))) + .extend(this.pointLocation(new Point(0, this.height))); + } + + /** + * Returns the maximum geographical bounds the map is constrained to, or `null` if none set. + * @returns max bounds + */ + getMaxBounds(): LngLatBounds | null { + if (!this.latRange || this.latRange.length !== 2 || + !this.lngRange || this.lngRange.length !== 2) return null; + + return new LngLatBounds([this.lngRange[0], this.latRange[0]], [this.lngRange[1], this.latRange[1]]); + } + + /** + * Calculate pixel height of the visible horizon in relation to map-center (e.g. height/2), + * multiplied by a static factor to simulate the earth-radius. + * The calculated value is the horizontal line from the camera-height to sea-level. + * @returns Horizon above center in pixels. + */ + getHorizon(): number { + return Math.tan(Math.PI / 2 - this._pitch) * this.cameraToCenterDistance * 0.85; + } + + /** + * Sets or clears the map's geographical constraints. + * @param bounds - A {@link LngLatBounds} object describing the new geographic boundaries of the map. + */ + setMaxBounds(bounds?: LngLatBounds | null) { + if (bounds) { + this.lngRange = [bounds.getWest(), bounds.getEast()]; + this.latRange = [bounds.getSouth(), bounds.getNorth()]; + this._constrain(); + } else { + this.lngRange = null; + this.latRange = [-this.maxValidLatitude, this.maxValidLatitude]; + } + } + + /** + * Calculate the posMatrix that, given a tile coordinate, would be used to display the tile on a map. + * @param unwrappedTileID - the tile ID + */ + calculatePosMatrix(unwrappedTileID: UnwrappedTileID, aligned: boolean = false): mat4 { + const posMatrixKey = unwrappedTileID.key; + const cache = aligned ? this._alignedPosMatrixCache : this._posMatrixCache; + if (cache[posMatrixKey]) { + return cache[posMatrixKey]; + } + + const canonical = unwrappedTileID.canonical; + const scale = this.worldSize / this.zoomScale(canonical.z); + const unwrappedX = canonical.x + Math.pow(2, canonical.z) * unwrappedTileID.wrap; + + const posMatrix = mat4.identity(new Float64Array(16) as any); + mat4.translate(posMatrix, posMatrix, [unwrappedX * scale, canonical.y * scale, 0]); + mat4.scale(posMatrix, posMatrix, [scale / EXTENT, scale / EXTENT, 1]); + mat4.multiply(posMatrix, aligned ? this.alignedProjMatrix : this.projMatrix, posMatrix); + + cache[posMatrixKey] = new Float32Array(posMatrix); + return cache[posMatrixKey]; + } + + customLayerMatrix(): mat4 { + return this.mercatorMatrix.slice() as any; + } + + _constrain() { + if (!this.center || !this.width || !this.height || this._constraining) return; + + this._constraining = true; + + let minY = -90; + let maxY = 90; + let minX = -180; + let maxX = 180; + let sy, sx, x2, y2; + const size = this.size, + unmodified = this._unmodified; + + if (this.latRange) { + const latRange = this.latRange; + minY = mercatorYfromLat(latRange[1]) * this.worldSize; + maxY = mercatorYfromLat(latRange[0]) * this.worldSize; + sy = maxY - minY < size.y ? size.y / (maxY - minY) : 0; + } + + if (this.lngRange) { + const lngRange = this.lngRange; + + minX = wrap( + mercatorXfromLng(lngRange[0]) * this.worldSize, + 0, + this.worldSize + ); + maxX = wrap( + mercatorXfromLng(lngRange[1]) * this.worldSize, + 0, + this.worldSize + ); + + if (maxX < minX) maxX += this.worldSize; + + sx = maxX - minX < size.x ? size.x / (maxX - minX) : 0; + } + + const point = this.point; + + // how much the map should scale to fit the screen into given latitude/longitude ranges + const s = Math.max(sx || 0, sy || 0); + + if (s) { + this.center = this.unproject(new Point( + sx ? (maxX + minX) / 2 : point.x, + sy ? (maxY + minY) / 2 : point.y)); + this.zoom += this.scaleZoom(s); + this._unmodified = unmodified; + this._constraining = false; + return; + } + + if (this.latRange) { + const y = point.y, + h2 = size.y / 2; + + if (y - h2 < minY) y2 = minY + h2; + if (y + h2 > maxY) y2 = maxY - h2; + } + + if (this.lngRange) { + const centerX = (minX + maxX) / 2; + const x = wrap(point.x, centerX - this.worldSize / 2, centerX + this.worldSize / 2); + const w2 = size.x / 2; + + if (x - w2 < minX) x2 = minX + w2; + if (x + w2 > maxX) x2 = maxX - w2; + } + + // pan the map if the screen goes off the range + if (x2 !== undefined || y2 !== undefined) { + this.center = this.unproject(new Point( + x2 !== undefined ? x2 : point.x, + y2 !== undefined ? y2 : point.y)).wrap(); + } + + this._unmodified = unmodified; + this._constraining = false; + } + + _calcMatrices() { + if (!this.height) return; + + const halfFov = this._fov / 2; + const offset = this.centerOffset; + const x = this.point.x, y = this.point.y; + this.cameraToCenterDistance = 0.5 / Math.tan(halfFov) * this.height; + this._pixelPerMeter = mercatorZfromAltitude(1, this.center.lat) * this.worldSize; + + let m = mat4.identity(new Float64Array(16) as any); + mat4.scale(m, m, [this.width / 2, -this.height / 2, 1]); + mat4.translate(m, m, [1, -1, 0]); + this.labelPlaneMatrix = m; + + m = mat4.identity(new Float64Array(16) as any); + mat4.scale(m, m, [1, -1, 1]); + mat4.translate(m, m, [-1, -1, 0]); + mat4.scale(m, m, [2 / this.width, 2 / this.height, 1]); + this.glCoordMatrix = m; + + // Calculate the camera to sea-level distance in pixel in respect of terrain + const cameraToSeaLevelDistance = this.cameraToCenterDistance + this._elevation * this._pixelPerMeter / Math.cos(this._pitch); + // In case of negative minimum elevation (e.g. the dead see, under the sea maps) use a lower plane for calculation + const minElevation = Math.min(this.elevation, this._minEleveationForCurrentTile); + const cameraToLowestPointDistance = cameraToSeaLevelDistance - minElevation * this._pixelPerMeter / Math.cos(this._pitch); + const lowestPlane = minElevation < 0 ? cameraToLowestPointDistance : cameraToSeaLevelDistance; + + // Find the distance from the center point [width/2 + offset.x, height/2 + offset.y] to the + // center top point [width/2 + offset.x, 0] in Z units, using the law of sines. + // 1 Z unit is equivalent to 1 horizontal px at the center of the map + // (the distance between[width/2, height/2] and [width/2 + 1, height/2]) + const groundAngle = Math.PI / 2 + this._pitch; + const fovAboveCenter = this._fov * (0.5 + offset.y / this.height); + const topHalfSurfaceDistance = Math.sin(fovAboveCenter) * lowestPlane / Math.sin(clamp(Math.PI - groundAngle - fovAboveCenter, 0.01, Math.PI - 0.01)); + + // Find the distance from the center point to the horizon + const horizon = this.getHorizon(); + const horizonAngle = Math.atan(horizon / this.cameraToCenterDistance); + const fovCenterToHorizon = 2 * horizonAngle * (0.5 + offset.y / (horizon * 2)); + const topHalfSurfaceDistanceHorizon = Math.sin(fovCenterToHorizon) * lowestPlane / Math.sin(clamp(Math.PI - groundAngle - fovCenterToHorizon, 0.01, Math.PI - 0.01)); + + // Calculate z distance of the farthest fragment that should be rendered. + // Add a bit extra to avoid precision problems when a fragment's distance is exactly `furthestDistance` + const topHalfMinDistance = Math.min(topHalfSurfaceDistance, topHalfSurfaceDistanceHorizon); + const farZ = (Math.cos(Math.PI / 2 - this._pitch) * topHalfMinDistance + lowestPlane) * 1.01; + + // The larger the value of nearZ is + // - the more depth precision is available for features (good) + // - clipping starts appearing sooner when the camera is close to 3d features (bad) + // + // Smaller values worked well for mapbox-gl-js but deckgl was encountering precision issues + // when rendering it's layers using custom layers. This value was experimentally chosen and + // seems to solve z-fighting issues in deckgl while not clipping buildings too close to the camera. + const nearZ = this.height / 50; + + // matrix for conversion from location to GL coordinates (-1 .. 1) + m = new Float64Array(16) as any; + mat4.perspective(m, this._fov, this.width / this.height, nearZ, farZ); + + // Apply center of perspective offset + m[8] = -offset.x * 2 / this.width; + m[9] = offset.y * 2 / this.height; + + mat4.scale(m, m, [1, -1, 1]); + mat4.translate(m, m, [0, 0, -this.cameraToCenterDistance]); + mat4.rotateX(m, m, this._pitch); + mat4.rotateZ(m, m, this.angle); + mat4.translate(m, m, [-x, -y, 0]); + + // The mercatorMatrix can be used to transform points from mercator coordinates + // ([0, 0] nw, [1, 1] se) to GL coordinates. + this.mercatorMatrix = mat4.scale([] as any, m, [this.worldSize, this.worldSize, this.worldSize]); + + // scale vertically to meters per pixel (inverse of ground resolution): + mat4.scale(m, m, [1, 1, this._pixelPerMeter]); + + // matrix for conversion from location to screen coordinates in 2D + this.pixelMatrix = mat4.multiply(new Float64Array(16) as any, this.labelPlaneMatrix, m); + + // matrix for conversion from location to GL coordinates (-1 .. 1) + mat4.translate(m, m, [0, 0, -this.elevation]); // elevate camera over terrain + this.projMatrix = m; + this.invProjMatrix = mat4.invert([] as any, m); + + // matrix for conversion from location to screen coordinates in 2D + this.pixelMatrix3D = mat4.multiply(new Float64Array(16) as any, this.labelPlaneMatrix, m); + + // Make a second projection matrix that is aligned to a pixel grid for rendering raster tiles. + // We're rounding the (floating point) x/y values to achieve to avoid rendering raster images to fractional + // coordinates. Additionally, we adjust by half a pixel in either direction in case that viewport dimension + // is an odd integer to preserve rendering to the pixel grid. We're rotating this shift based on the angle + // of the transformation so that 0°, 90°, 180°, and 270° rasters are crisp, and adjust the shift so that + // it is always <= 0.5 pixels. + const xShift = (this.width % 2) / 2, yShift = (this.height % 2) / 2, + angleCos = Math.cos(this.angle), angleSin = Math.sin(this.angle), + dx = x - Math.round(x) + angleCos * xShift + angleSin * yShift, + dy = y - Math.round(y) + angleCos * yShift + angleSin * xShift; + const alignedM = new Float64Array(m) as any as mat4; + mat4.translate(alignedM, alignedM, [dx > 0.5 ? dx - 1 : dx, dy > 0.5 ? dy - 1 : dy, 0]); + this.alignedProjMatrix = alignedM; + + // inverse matrix for conversion from screen coordinaes to location + m = mat4.invert(new Float64Array(16) as any, this.pixelMatrix); + if (!m) throw new Error('failed to invert matrix'); + this.pixelMatrixInverse = m; + + this._posMatrixCache = {}; + this._alignedPosMatrixCache = {}; + } + + maxPitchScaleFactor() { + // calcMatrices hasn't run yet + if (!this.pixelMatrixInverse) return 1; + + const coord = this.pointCoordinate(new Point(0, 0)); + const p = [coord.x * this.worldSize, coord.y * this.worldSize, 0, 1] as vec4; + const topPoint = vec4.transformMat4(p, p, this.pixelMatrix); + return topPoint[3] / this.cameraToCenterDistance; + } + + /** + * The camera looks at the map from a 3D (lng, lat, altitude) location. Let's use `cameraLocation` + * as the name for the location under the camera and on the surface of the earth (lng, lat, 0). + * `cameraPoint` is the projected position of the `cameraLocation`. + * + * This point is useful to us because only fill-extrusions that are between `cameraPoint` and + * the query point on the surface of the earth can extend and intersect the query. + * + * When the map is not pitched the `cameraPoint` is equivalent to the center of the map because + * the camera is right above the center of the map. + */ + getCameraPoint() { + const pitch = this._pitch; + const yOffset = Math.tan(pitch) * (this.cameraToCenterDistance || 1); + return this.centerPoint.add(new Point(0, yOffset)); + } + + /** + * When the map is pitched, some of the 3D features that intersect a query will not intersect + * the query at the surface of the earth. Instead the feature may be closer and only intersect + * the query because it extrudes into the air. + * @param queryGeometry - For point queries, the line from the query point to the "camera point", + * for other geometries, the envelope of the query geometry and the "camera point" + * @returns a geometry that includes all of the original query as well as all possible ares of the + * screen where the *base* of a visible extrusion could be. + * + */ + getCameraQueryGeometry(queryGeometry: Array): Array { + const c = this.getCameraPoint(); + + if (queryGeometry.length === 1) { + return [queryGeometry[0], c]; + } else { + let minX = c.x; + let minY = c.y; + let maxX = c.x; + let maxY = c.y; + for (const p of queryGeometry) { + minX = Math.min(minX, p.x); + minY = Math.min(minY, p.y); + maxX = Math.max(maxX, p.x); + maxY = Math.max(maxY, p.y); + } + return [ + new Point(minX, minY), + new Point(maxX, minY), + new Point(maxX, maxY), + new Point(minX, maxY), + new Point(minX, minY) + ]; + } + } +} diff --git a/web/libraries/maplibre-gl/src/gl/color_mode.ts b/web/libraries/maplibre-gl/src/gl/color_mode.ts new file mode 100644 index 00000000..2e32c577 --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/color_mode.ts @@ -0,0 +1,31 @@ +import {Color} from '@maplibre/maplibre-gl-style-spec'; + +import type {BlendFuncType, ColorMaskType} from './types'; + +const ZERO = 0x0000; +const ONE = 0x0001; +const ONE_MINUS_SRC_ALPHA = 0x0303; + +export class ColorMode { + blendFunction: BlendFuncType; + blendColor: Color; + mask: ColorMaskType; + + constructor(blendFunction: BlendFuncType, blendColor: Color, mask: ColorMaskType) { + this.blendFunction = blendFunction; + this.blendColor = blendColor; + this.mask = mask; + } + + static Replace: BlendFuncType; + + static disabled: Readonly; + static unblended: Readonly; + static alphaBlended: Readonly; +} + +ColorMode.Replace = [ONE, ZERO]; + +ColorMode.disabled = new ColorMode(ColorMode.Replace, Color.transparent, [false, false, false, false]); +ColorMode.unblended = new ColorMode(ColorMode.Replace, Color.transparent, [true, true, true, true]); +ColorMode.alphaBlended = new ColorMode([ONE, ONE_MINUS_SRC_ALPHA], Color.transparent, [true, true, true, true]); diff --git a/web/libraries/maplibre-gl/src/gl/context.ts b/web/libraries/maplibre-gl/src/gl/context.ts new file mode 100644 index 00000000..aa3749ae --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/context.ts @@ -0,0 +1,321 @@ +import {IndexBuffer} from './index_buffer'; + +import {VertexBuffer} from './vertex_buffer'; +import {Framebuffer} from './framebuffer'; +import {DepthMode} from './depth_mode'; +import {StencilMode} from './stencil_mode'; +import {ColorMode} from './color_mode'; +import {CullFaceMode} from './cull_face_mode'; +import {deepEqual} from '../util/util'; +import {ClearColor, ClearDepth, ClearStencil, ColorMask, DepthMask, StencilMask, StencilFunc, StencilOp, StencilTest, DepthRange, DepthTest, DepthFunc, Blend, BlendFunc, BlendColor, BlendEquation, CullFace, CullFaceSide, FrontFace, ProgramValue, ActiveTextureUnit, Viewport, BindFramebuffer, BindRenderbuffer, BindTexture, BindVertexBuffer, BindElementBuffer, BindVertexArray, PixelStoreUnpack, PixelStoreUnpackPremultiplyAlpha, PixelStoreUnpackFlipY} from './value'; + +import type {TriangleIndexArray, LineIndexArray, LineStripIndexArray} from '../data/index_array_type'; +import type { + StructArray, + StructArrayMember +} from '../util/struct_array'; +import type {Color} from '@maplibre/maplibre-gl-style-spec'; +import {isWebGL2} from './webgl2'; + +type ClearArgs = { + color?: Color; + depth?: number; + stencil?: number; +}; + +/** + * @internal + * A webgl wrapper class to allow injection, mocking and abstaction + */ +export class Context { + gl: WebGLRenderingContext | WebGL2RenderingContext; + + currentNumAttributes: number; + maxTextureSize: number; + + clearColor: ClearColor; + clearDepth: ClearDepth; + clearStencil: ClearStencil; + colorMask: ColorMask; + depthMask: DepthMask; + stencilMask: StencilMask; + stencilFunc: StencilFunc; + stencilOp: StencilOp; + stencilTest: StencilTest; + depthRange: DepthRange; + depthTest: DepthTest; + depthFunc: DepthFunc; + blend: Blend; + blendFunc: BlendFunc; + blendColor: BlendColor; + blendEquation: BlendEquation; + cullFace: CullFace; + cullFaceSide: CullFaceSide; + frontFace: FrontFace; + program: ProgramValue; + activeTexture: ActiveTextureUnit; + viewport: Viewport; + bindFramebuffer: BindFramebuffer; + bindRenderbuffer: BindRenderbuffer; + bindTexture: BindTexture; + bindVertexBuffer: BindVertexBuffer; + bindElementBuffer: BindElementBuffer; + bindVertexArray: BindVertexArray; + pixelStoreUnpack: PixelStoreUnpack; + pixelStoreUnpackPremultiplyAlpha: PixelStoreUnpackPremultiplyAlpha; + pixelStoreUnpackFlipY: PixelStoreUnpackFlipY; + + // eslint-disable-next-line camelcase + extTextureFilterAnisotropic: EXT_texture_filter_anisotropic | null; + extTextureFilterAnisotropicMax?: GLfloat; + HALF_FLOAT?: GLenum; + RGBA16F?: GLenum; + RGB16F?: GLenum; + + constructor(gl: WebGLRenderingContext | WebGL2RenderingContext) { + this.gl = gl; + this.clearColor = new ClearColor(this); + this.clearDepth = new ClearDepth(this); + this.clearStencil = new ClearStencil(this); + this.colorMask = new ColorMask(this); + this.depthMask = new DepthMask(this); + this.stencilMask = new StencilMask(this); + this.stencilFunc = new StencilFunc(this); + this.stencilOp = new StencilOp(this); + this.stencilTest = new StencilTest(this); + this.depthRange = new DepthRange(this); + this.depthTest = new DepthTest(this); + this.depthFunc = new DepthFunc(this); + this.blend = new Blend(this); + this.blendFunc = new BlendFunc(this); + this.blendColor = new BlendColor(this); + this.blendEquation = new BlendEquation(this); + this.cullFace = new CullFace(this); + this.cullFaceSide = new CullFaceSide(this); + this.frontFace = new FrontFace(this); + this.program = new ProgramValue(this); + this.activeTexture = new ActiveTextureUnit(this); + this.viewport = new Viewport(this); + this.bindFramebuffer = new BindFramebuffer(this); + this.bindRenderbuffer = new BindRenderbuffer(this); + this.bindTexture = new BindTexture(this); + this.bindVertexBuffer = new BindVertexBuffer(this); + this.bindElementBuffer = new BindElementBuffer(this); + this.bindVertexArray = new BindVertexArray(this); + this.pixelStoreUnpack = new PixelStoreUnpack(this); + this.pixelStoreUnpackPremultiplyAlpha = new PixelStoreUnpackPremultiplyAlpha(this); + this.pixelStoreUnpackFlipY = new PixelStoreUnpackFlipY(this); + + this.extTextureFilterAnisotropic = ( + gl.getExtension('EXT_texture_filter_anisotropic') || + gl.getExtension('MOZ_EXT_texture_filter_anisotropic') || + gl.getExtension('WEBKIT_EXT_texture_filter_anisotropic') + ); + + if (this.extTextureFilterAnisotropic) { + this.extTextureFilterAnisotropicMax = gl.getParameter(this.extTextureFilterAnisotropic.MAX_TEXTURE_MAX_ANISOTROPY_EXT); + } + + this.maxTextureSize = gl.getParameter(gl.MAX_TEXTURE_SIZE); + + if (isWebGL2(gl)) { + this.HALF_FLOAT = gl.HALF_FLOAT; + const extColorBufferHalfFloat = gl.getExtension('EXT_color_buffer_half_float'); + this.RGBA16F = gl.RGBA16F ?? extColorBufferHalfFloat?.RGBA16F_EXT; + this.RGB16F = gl.RGB16F ?? extColorBufferHalfFloat?.RGB16F_EXT; + gl.getExtension('EXT_color_buffer_float'); + } else { + gl.getExtension('EXT_color_buffer_half_float'); + gl.getExtension('OES_texture_half_float_linear'); + const extTextureHalfFloat = gl.getExtension('OES_texture_half_float'); + this.HALF_FLOAT = extTextureHalfFloat?.HALF_FLOAT_OES; + } + } + + setDefault() { + this.unbindVAO(); + + this.clearColor.setDefault(); + this.clearDepth.setDefault(); + this.clearStencil.setDefault(); + this.colorMask.setDefault(); + this.depthMask.setDefault(); + this.stencilMask.setDefault(); + this.stencilFunc.setDefault(); + this.stencilOp.setDefault(); + this.stencilTest.setDefault(); + this.depthRange.setDefault(); + this.depthTest.setDefault(); + this.depthFunc.setDefault(); + this.blend.setDefault(); + this.blendFunc.setDefault(); + this.blendColor.setDefault(); + this.blendEquation.setDefault(); + this.cullFace.setDefault(); + this.cullFaceSide.setDefault(); + this.frontFace.setDefault(); + this.program.setDefault(); + this.activeTexture.setDefault(); + this.bindFramebuffer.setDefault(); + this.pixelStoreUnpack.setDefault(); + this.pixelStoreUnpackPremultiplyAlpha.setDefault(); + this.pixelStoreUnpackFlipY.setDefault(); + } + + setDirty() { + this.clearColor.dirty = true; + this.clearDepth.dirty = true; + this.clearStencil.dirty = true; + this.colorMask.dirty = true; + this.depthMask.dirty = true; + this.stencilMask.dirty = true; + this.stencilFunc.dirty = true; + this.stencilOp.dirty = true; + this.stencilTest.dirty = true; + this.depthRange.dirty = true; + this.depthTest.dirty = true; + this.depthFunc.dirty = true; + this.blend.dirty = true; + this.blendFunc.dirty = true; + this.blendColor.dirty = true; + this.blendEquation.dirty = true; + this.cullFace.dirty = true; + this.cullFaceSide.dirty = true; + this.frontFace.dirty = true; + this.program.dirty = true; + this.activeTexture.dirty = true; + this.viewport.dirty = true; + this.bindFramebuffer.dirty = true; + this.bindRenderbuffer.dirty = true; + this.bindTexture.dirty = true; + this.bindVertexBuffer.dirty = true; + this.bindElementBuffer.dirty = true; + this.bindVertexArray.dirty = true; + this.pixelStoreUnpack.dirty = true; + this.pixelStoreUnpackPremultiplyAlpha.dirty = true; + this.pixelStoreUnpackFlipY.dirty = true; + } + + createIndexBuffer(array: TriangleIndexArray | LineIndexArray | LineStripIndexArray, dynamicDraw?: boolean) { + return new IndexBuffer(this, array, dynamicDraw); + } + + createVertexBuffer(array: StructArray, attributes: ReadonlyArray, dynamicDraw?: boolean) { + return new VertexBuffer(this, array, attributes, dynamicDraw); + } + + createRenderbuffer(storageFormat: number, width: number, height: number) { + const gl = this.gl; + + const rbo = gl.createRenderbuffer(); + this.bindRenderbuffer.set(rbo); + gl.renderbufferStorage(gl.RENDERBUFFER, storageFormat, width, height); + this.bindRenderbuffer.set(null); + + return rbo; + } + + createFramebuffer(width: number, height: number, hasDepth: boolean, hasStencil: boolean) { + return new Framebuffer(this, width, height, hasDepth, hasStencil); + } + + clear({ + color, + depth, + stencil + }: ClearArgs) { + const gl = this.gl; + let mask = 0; + + if (color) { + mask |= gl.COLOR_BUFFER_BIT; + this.clearColor.set(color); + this.colorMask.set([true, true, true, true]); + } + + if (typeof depth !== 'undefined') { + mask |= gl.DEPTH_BUFFER_BIT; + + // Workaround for platforms where clearDepth doesn't seem to work + // without resetting the depthRange. See https://github.com/mapbox/mapbox-gl-js/issues/3437 + this.depthRange.set([0, 1]); + + this.clearDepth.set(depth); + this.depthMask.set(true); + } + + if (typeof stencil !== 'undefined') { + mask |= gl.STENCIL_BUFFER_BIT; + this.clearStencil.set(stencil); + this.stencilMask.set(0xFF); + } + + gl.clear(mask); + } + + setCullFace(cullFaceMode: Readonly) { + if (cullFaceMode.enable === false) { + this.cullFace.set(false); + } else { + this.cullFace.set(true); + this.cullFaceSide.set(cullFaceMode.mode); + this.frontFace.set(cullFaceMode.frontFace); + } + } + + setDepthMode(depthMode: Readonly) { + if (depthMode.func === this.gl.ALWAYS && !depthMode.mask) { + this.depthTest.set(false); + } else { + this.depthTest.set(true); + this.depthFunc.set(depthMode.func); + this.depthMask.set(depthMode.mask); + this.depthRange.set(depthMode.range); + } + } + + setStencilMode(stencilMode: Readonly) { + if (stencilMode.test.func === this.gl.ALWAYS && !stencilMode.mask) { + this.stencilTest.set(false); + } else { + this.stencilTest.set(true); + this.stencilMask.set(stencilMode.mask); + this.stencilOp.set([stencilMode.fail, stencilMode.depthFail, stencilMode.pass]); + this.stencilFunc.set({ + func: stencilMode.test.func, + ref: stencilMode.ref, + mask: stencilMode.test.mask + }); + } + } + + setColorMode(colorMode: Readonly) { + if (deepEqual(colorMode.blendFunction, ColorMode.Replace)) { + this.blend.set(false); + } else { + this.blend.set(true); + this.blendFunc.set(colorMode.blendFunction); + this.blendColor.set(colorMode.blendColor); + } + + this.colorMask.set(colorMode.mask); + } + + createVertexArray(): WebGLVertexArrayObject | undefined { + if (isWebGL2(this.gl)) + return this.gl.createVertexArray(); + return this.gl.getExtension('OES_vertex_array_object')?.createVertexArrayOES(); + } + + deleteVertexArray(x: WebGLVertexArrayObject | undefined) { + if (isWebGL2(this.gl)) + return this.gl.deleteVertexArray(x); + return this.gl.getExtension('OES_vertex_array_object')?.deleteVertexArrayOES(x); + } + + unbindVAO() { + // Unbinding the VAO prevents other things (custom layers, new buffer creation) from + // unintentionally changing the state of the last VAO used. + this.bindVertexArray.set(null); + } +} diff --git a/web/libraries/maplibre-gl/src/gl/cull_face_mode.ts b/web/libraries/maplibre-gl/src/gl/cull_face_mode.ts new file mode 100644 index 00000000..4bd679a2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/cull_face_mode.ts @@ -0,0 +1,22 @@ +import type {CullFaceModeType, FrontFaceType} from './types'; + +const BACK = 0x0405; +const CCW = 0x0901; + +export class CullFaceMode { + enable: boolean; + mode: CullFaceModeType; + frontFace: FrontFaceType; + + constructor(enable: boolean, mode: CullFaceModeType, frontFace: FrontFaceType) { + this.enable = enable; + this.mode = mode; + this.frontFace = frontFace; + } + + static disabled: Readonly; + static backCCW: Readonly; +} + +CullFaceMode.disabled = new CullFaceMode(false, BACK, CCW); +CullFaceMode.backCCW = new CullFaceMode(true, BACK, CCW); diff --git a/web/libraries/maplibre-gl/src/gl/depth_mode.ts b/web/libraries/maplibre-gl/src/gl/depth_mode.ts new file mode 100644 index 00000000..f24ad301 --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/depth_mode.ts @@ -0,0 +1,26 @@ +import type {DepthFuncType, DepthMaskType, DepthRangeType} from './types'; + +const ALWAYS = 0x0207; + +export class DepthMode { + func: DepthFuncType; + mask: DepthMaskType; + range: DepthRangeType; + + // DepthMask enums + static ReadOnly: boolean; + static ReadWrite: boolean; + + constructor(depthFunc: DepthFuncType, depthMask: DepthMaskType, depthRange: DepthRangeType) { + this.func = depthFunc; + this.mask = depthMask; + this.range = depthRange; + } + + static disabled: Readonly; +} + +DepthMode.ReadOnly = false; +DepthMode.ReadWrite = true; + +DepthMode.disabled = new DepthMode(ALWAYS, DepthMode.ReadOnly, [0, 1]); diff --git a/web/libraries/maplibre-gl/src/gl/framebuffer.ts b/web/libraries/maplibre-gl/src/gl/framebuffer.ts new file mode 100644 index 00000000..384d48b9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/framebuffer.ts @@ -0,0 +1,48 @@ +import {ColorAttachment, DepthAttachment, DepthStencilAttachment} from './value'; + +import type {Context} from './context'; + +/** + * @internal + * A framebuffer holder object + */ +export class Framebuffer { + context: Context; + width: number; + height: number; + framebuffer: WebGLFramebuffer; + colorAttachment: ColorAttachment; + depthAttachment: DepthAttachment; + + constructor(context: Context, width: number, height: number, hasDepth: boolean, hasStencil: boolean) { + this.context = context; + this.width = width; + this.height = height; + const gl = context.gl; + const fbo = this.framebuffer = gl.createFramebuffer(); + + this.colorAttachment = new ColorAttachment(context, fbo); + if (hasDepth) { + this.depthAttachment = hasStencil ? new DepthStencilAttachment(context, fbo) : new DepthAttachment(context, fbo); + } else if (hasStencil) { + throw new Error('Stencil cannot be setted without depth'); + } + if (gl.checkFramebufferStatus(gl.FRAMEBUFFER) !== gl.FRAMEBUFFER_COMPLETE) { + throw new Error('Framebuffer is not complete'); + } + } + + destroy() { + const gl = this.context.gl; + + const texture = this.colorAttachment.get(); + if (texture) gl.deleteTexture(texture); + + if (this.depthAttachment) { + const renderbuffer = this.depthAttachment.get(); + if (renderbuffer) gl.deleteRenderbuffer(renderbuffer); + } + + gl.deleteFramebuffer(this.framebuffer); + } +} diff --git a/web/libraries/maplibre-gl/src/gl/index_buffer.ts b/web/libraries/maplibre-gl/src/gl/index_buffer.ts new file mode 100644 index 00000000..f15580b7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/index_buffer.ts @@ -0,0 +1,55 @@ + +import type {StructArray} from '../util/struct_array'; +import type {TriangleIndexArray, LineIndexArray, LineStripIndexArray} from '../data/index_array_type'; +import type {Context} from '../gl/context'; + +/** + * @internal + * an index buffer class + */ +export class IndexBuffer { + context: Context; + buffer: WebGLBuffer; + dynamicDraw: boolean; + + constructor(context: Context, array: TriangleIndexArray | LineIndexArray | LineStripIndexArray, dynamicDraw?: boolean) { + this.context = context; + const gl = context.gl; + this.buffer = gl.createBuffer(); + this.dynamicDraw = Boolean(dynamicDraw); + + // The bound index buffer is part of vertex array object state. We don't want to + // modify whatever VAO happens to be currently bound, so make sure the default + // vertex array provided by the context is bound instead. + this.context.unbindVAO(); + + context.bindElementBuffer.set(this.buffer); + gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, array.arrayBuffer, this.dynamicDraw ? gl.DYNAMIC_DRAW : gl.STATIC_DRAW); + + if (!this.dynamicDraw) { + delete array.arrayBuffer; + } + } + + bind() { + this.context.bindElementBuffer.set(this.buffer); + } + + updateData(array: StructArray) { + const gl = this.context.gl; + if (!this.dynamicDraw) throw new Error('Attempted to update data while not in dynamic mode.'); + // The right VAO will get this buffer re-bound later in VertexArrayObject#bind + // See https://github.com/mapbox/mapbox-gl-js/issues/5620 + this.context.unbindVAO(); + this.bind(); + gl.bufferSubData(gl.ELEMENT_ARRAY_BUFFER, 0, array.arrayBuffer); + } + + destroy() { + const gl = this.context.gl; + if (this.buffer) { + gl.deleteBuffer(this.buffer); + delete this.buffer; + } + } +} diff --git a/web/libraries/maplibre-gl/src/gl/render_pool.test.ts b/web/libraries/maplibre-gl/src/gl/render_pool.test.ts new file mode 100644 index 00000000..067b4094 --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/render_pool.test.ts @@ -0,0 +1,67 @@ +import {Context} from './context'; +import {RenderPool} from './render_pool'; +import gl from 'gl'; + +describe('render pool', () => { + const POOL_SIZE = 3; + + function createAndFillPool(): RenderPool { + const pool = new RenderPool(new Context(gl(1, 1) as any), POOL_SIZE, 512); + for (let i = 0; i < POOL_SIZE; i++) { + pool.useObject(pool.getOrCreateFreeObject()); + } + return pool; + } + + test('create pool should not be full', () => { + const pool = new RenderPool(new Context(gl(1, 1) as any), POOL_SIZE, 512); + expect(pool.isFull()).toBeFalsy(); + }); + + test('create pool should be full', () => { + const pool = createAndFillPool(); + expect(() => pool.getOrCreateFreeObject()).toThrow('No free RenderPool available, call freeAllObjects() required!'); + }); + + test('create pool and fill it', () => { + const pool = createAndFillPool(); + expect(pool.isFull()).toBeTruthy(); + }); + + test('check recently used after using two objects', () => { + const pool = createAndFillPool(); + pool.freeAllObjects(); + const obj0 = pool.getObjectForId(0); + pool.useObject(obj0); + pool.freeAllObjects(); + const obj1 = pool.getOrCreateFreeObject(); + expect(obj1.id).toBe(1); + }); + + test('not full after freeing an object', () => { + const pool = createAndFillPool(); + const obj = pool.getObjectForId(0); + pool.freeObject(obj); + expect(pool.isFull()).toBeFalsy(); + expect(obj.stamp).toBe(-1); + }); + + test('stamp object should get stamped', () => { + const pool = createAndFillPool(); + const obj = pool.getObjectForId(0); + pool.stampObject(obj); + expect(obj.stamp).toBe(1); + }); + + test('free all objects, first object should be the first free object', () => { + const pool = createAndFillPool(); + pool.freeAllObjects(); + expect(pool.getOrCreateFreeObject().id).toBe(0); + }); + + test('destruct should remove textures', () => { + const pool = createAndFillPool(); + pool.destruct(); + expect(pool.getObjectForId(0).texture.texture).toBeNull(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/gl/render_pool.ts b/web/libraries/maplibre-gl/src/gl/render_pool.ts new file mode 100644 index 00000000..abf1a2c7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/render_pool.ts @@ -0,0 +1,93 @@ +import {Texture} from '../render/texture'; +import {Context} from './context'; +import {Framebuffer} from './framebuffer'; + +export type PoolObject = { + id: number; + fbo: Framebuffer; + texture: Texture; + stamp: number; + inUse: boolean; +}; +/** + * @internal + * `RenderPool` is a resource pool for textures and framebuffers + */ +export class RenderPool { + private _objects: Array; + /** + * An index array of recently used pool objects. + * Items that are used recently are last in the array + */ + private _recentlyUsed: Array; + private _stamp: number; + + constructor( + private readonly _context: Context, + private readonly _size: number, + private readonly _tileSize: number) { + this._objects = []; + this._recentlyUsed = []; + this._stamp = 0; + } + + public destruct() { + for (const obj of this._objects) { + obj.texture.destroy(); + obj.fbo.destroy(); + } + } + + private _createObject(id: number): PoolObject { + const fbo = this._context.createFramebuffer(this._tileSize, this._tileSize, true, true); + const texture = new Texture(this._context, {width: this._tileSize, height: this._tileSize, data: null}, this._context.gl.RGBA); + texture.bind(this._context.gl.LINEAR, this._context.gl.CLAMP_TO_EDGE); + fbo.depthAttachment.set(this._context.createRenderbuffer(this._context.gl.DEPTH_STENCIL, this._tileSize, this._tileSize)); + fbo.colorAttachment.set(texture.texture); + return {id, fbo, texture, stamp: -1, inUse: false}; + } + + public getObjectForId(id: number): PoolObject { + return this._objects[id]; + } + + public useObject(obj: PoolObject) { + obj.inUse = true; + this._recentlyUsed = this._recentlyUsed.filter(id => obj.id !== id); + this._recentlyUsed.push(obj.id); + } + + public stampObject(obj: PoolObject) { + obj.stamp = ++this._stamp; + } + + public getOrCreateFreeObject(): PoolObject { + // check for free existing object + for (const id of this._recentlyUsed) { + if (!this._objects[id].inUse) + return this._objects[id]; + } + if (this._objects.length >= this._size) + throw new Error('No free RenderPool available, call freeAllObjects() required!'); + // create new object + const obj = this._createObject(this._objects.length); + this._objects.push(obj); + return obj; + } + + public freeObject(obj: PoolObject) { + obj.inUse = false; + } + + public freeAllObjects() { + for (const obj of this._objects) + this.freeObject(obj); + } + + public isFull(): boolean { + if (this._objects.length < this._size) { + return false; + } + return this._objects.some(o => !o.inUse) === false; + } +} diff --git a/web/libraries/maplibre-gl/src/gl/state.test.ts b/web/libraries/maplibre-gl/src/gl/state.test.ts new file mode 100644 index 00000000..720fd5c0 --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/state.test.ts @@ -0,0 +1,194 @@ +import {ClearColor, ClearDepth, ClearStencil, ColorMask, DepthMask, StencilMask, StencilFunc, StencilOp, StencilTest, DepthRange, DepthTest, DepthFunc, Blend, BlendFunc, BlendColor, ProgramValue, ActiveTextureUnit, Viewport, BindFramebuffer, BindRenderbuffer, BindTexture, BindVertexBuffer, BindElementBuffer, BindVertexArray, PixelStoreUnpack, PixelStoreUnpackPremultiplyAlpha} from './value'; +import {Context} from './context'; +import {Color} from '@maplibre/maplibre-gl-style-spec'; +import {deepEqual} from '../util/util'; +import gl from 'gl'; +import {setupMockWebGLContext} from '../util/test/mock_webgl'; + +const context = new Context(gl(10, 10) as any); + +setupMockWebGLContext(context.gl); + +const valueTest = (Constructor: new (...args:any[]) => any, options) => { + test('#constructor', () => { + const v = new Constructor(context); + expect(v).toBeTruthy(); + const currentV = v.get(); + expect(typeof currentV).not.toBe('undefined'); + }); + + test('#set', () => { + const v = new Constructor(context); + v.set(options.setValue); + const equality = (options.equality) || ((a, b) => deepEqual(a, b)); + expect(equality(v.get(), options.setValue)).toBeTruthy(); + }); + +}; + +describe('ClearColor', () => { + valueTest(ClearColor, { + setValue: new Color(1, 1, 0, 1) + }); +}); + +describe('ClearDepth', () => { + valueTest(ClearDepth, { + setValue: 0.5 + }); +}); + +describe('ClearStencil', () => { + valueTest(ClearStencil, { + setValue: 0.5 + }); +}); + +describe('ColorMask', () => { + valueTest(ColorMask, { + setValue: [false, false, true, true] + }); +}); + +describe('DepthMask', () => { + valueTest(DepthMask, { + setValue: false + }); +}); + +describe('StencilMask', () => { + valueTest(StencilMask, { + setValue: [0x00, 4] + }); +}); + +describe('StencilFunc', () => { + valueTest(StencilFunc, { + setValue: { + func: context.gl.LEQUAL, + ref: 1, + mask: 0xFF + } + }); +}); + +describe('StencilOp', () => { + valueTest(StencilOp, { + setValue: [context.gl.KEEP, context.gl.REPLACE, context.gl.REPLACE] + }); +}); + +describe('StencilTest', () => { + valueTest(StencilTest, { + setValue: true + }); +}); + +describe('DepthRange', () => { + valueTest(DepthRange, { + setValue: [0, 0.1] + }); +}); + +describe('DepthTest', () => { + valueTest(DepthTest, { + setValue: true + }); +}); + +describe('DepthFunc', () => { + valueTest(DepthFunc, { + setValue: context.gl.EQUAL + }); +}); + +describe('Blend', () => { + valueTest(Blend, { + setValue: false + }); +}); + +describe('BlendFunc', () => { + valueTest(BlendFunc, { + setValue: [context.gl.SRC_ALPHA, context.gl.SRC_ALPHA] + }); +}); + +describe('BlendColor', () => { + valueTest(BlendColor, { + setValue: Color.white + }); +}); + +describe('Program', () => { + valueTest(ProgramValue, { + equality: (a, b) => a === b, + setValue: context.gl.createProgram() + }); +}); + +describe('ActiveTextureUnit', () => { + valueTest(ActiveTextureUnit, { + setValue: context.gl.TEXTURE1 + }); +}); + +describe('Viewport', () => { + valueTest(Viewport, { + setValue: [0, 0, 1, 1] + }); +}); + +describe('BindFramebuffer', () => { + valueTest(BindFramebuffer, { + equality: (a, b) => a === b, + setValue: context.gl.createFramebuffer() + }); +}); + +describe('BindRenderbuffer', () => { + valueTest(BindRenderbuffer, { + equality: (a, b) => a === b, + setValue: context.gl.createRenderbuffer() + }); +}); + +describe('BindTexture', () => { + valueTest(BindTexture, { + equality: (a, b) => a === b, + setValue: context.gl.createTexture() + }); +}); + +describe('BindVertexBuffer', () => { + valueTest(BindVertexBuffer, { + equality: (a, b) => a === b, + setValue: context.gl.createBuffer() + }); +}); + +describe('BindElementBuffer', () => { + valueTest(BindElementBuffer, { + equality: (a, b) => a === b, + setValue: context.gl.createBuffer() + }); +}); + +describe('BindVertexArray', () => { + valueTest(BindVertexArray, { + equality: (a, b) => a === b, + setValue: context.createVertexArray() + }); +}); + +describe('PixelStoreUnpack', () => { + valueTest(PixelStoreUnpack, { + setValue: 8 + }); +}); + +describe('PixelStoreUnpackPremultiplyAlpha', () => { + valueTest(PixelStoreUnpackPremultiplyAlpha, { + setValue: true + }); +}); diff --git a/web/libraries/maplibre-gl/src/gl/stencil_mode.ts b/web/libraries/maplibre-gl/src/gl/stencil_mode.ts new file mode 100644 index 00000000..7144eab4 --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/stencil_mode.ts @@ -0,0 +1,27 @@ +import type {StencilOpConstant, StencilTestGL} from './types'; + +const ALWAYS = 0x0207; +const KEEP = 0x1E00; + +export class StencilMode { + test: StencilTestGL; + ref: number; + mask: number; + fail: StencilOpConstant; + depthFail: StencilOpConstant; + pass: StencilOpConstant; + + constructor(test: StencilTestGL, ref: number, mask: number, fail: StencilOpConstant, + depthFail: StencilOpConstant, pass: StencilOpConstant) { + this.test = test; + this.ref = ref; + this.mask = mask; + this.fail = fail; + this.depthFail = depthFail; + this.pass = pass; + } + + static disabled: Readonly; +} + +StencilMode.disabled = new StencilMode({func: ALWAYS, mask: 0}, 0, 0, KEEP, KEEP, KEEP); diff --git a/web/libraries/maplibre-gl/src/gl/types.ts b/web/libraries/maplibre-gl/src/gl/types.ts new file mode 100644 index 00000000..c4acefe7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/types.ts @@ -0,0 +1,59 @@ +type BlendFuncConstant = WebGLRenderingContextBase['ZERO'] | WebGLRenderingContextBase['ONE'] | WebGLRenderingContextBase['SRC_COLOR'] | WebGLRenderingContextBase['ONE_MINUS_SRC_COLOR'] | WebGLRenderingContextBase['DST_COLOR'] | WebGLRenderingContextBase['ONE_MINUS_DST_COLOR'] | WebGLRenderingContextBase['SRC_ALPHA'] | WebGLRenderingContextBase['ONE_MINUS_SRC_ALPHA'] | WebGLRenderingContextBase['DST_ALPHA'] | WebGLRenderingContextBase['ONE_MINUS_DST_ALPHA'] | WebGLRenderingContextBase['CONSTANT_COLOR'] | WebGLRenderingContextBase['ONE_MINUS_CONSTANT_COLOR'] | WebGLRenderingContextBase['CONSTANT_ALPHA'] | WebGLRenderingContextBase['ONE_MINUS_CONSTANT_ALPHA'] | WebGLRenderingContextBase['BLEND_COLOR']; + +export type BlendFuncType = [BlendFuncConstant, BlendFuncConstant]; + +export type BlendEquationType = WebGLRenderingContextBase['FUNC_ADD'] | WebGLRenderingContextBase['FUNC_SUBTRACT'] | WebGLRenderingContextBase['FUNC_REVERSE_SUBTRACT']; + +export type ColorMaskType = [boolean, boolean, boolean, boolean]; + +export type CompareFuncType = WebGLRenderingContextBase['NEVER'] | WebGLRenderingContextBase['LESS'] | WebGLRenderingContextBase['EQUAL'] | WebGLRenderingContextBase['LEQUAL'] | WebGLRenderingContextBase['GREATER'] | WebGLRenderingContextBase['NOTEQUAL'] | WebGLRenderingContextBase['GEQUAL'] | WebGLRenderingContextBase['ALWAYS']; + +export type DepthMaskType = boolean; + +export type DepthRangeType = [number, number]; + +export type DepthFuncType = CompareFuncType; + +export type StencilFuncType = { + func: CompareFuncType; + ref: number; + mask: number; +}; + +export type StencilOpConstant = WebGLRenderingContextBase['KEEP'] | WebGLRenderingContextBase['ZERO'] | WebGLRenderingContextBase['REPLACE'] | WebGLRenderingContextBase['INCR'] | WebGLRenderingContextBase['INCR_WRAP'] | WebGLRenderingContextBase['DECR'] | WebGLRenderingContextBase['DECR_WRAP'] | WebGLRenderingContextBase['INVERT']; + +export type StencilOpType = [StencilOpConstant, StencilOpConstant, StencilOpConstant]; + +export type TextureUnitType = number; + +export type ViewportType = [number, number, number, number]; + +export type StencilTestGL = { + func: WebGLRenderingContextBase['NEVER']; + mask: 0; +} | { + func: WebGLRenderingContextBase['LESS']; + mask: number; +} | { + func: WebGLRenderingContextBase['EQUAL']; + mask: number; +} | { + func: WebGLRenderingContextBase['LEQUAL']; + mask: number; +} | { + func: WebGLRenderingContextBase['GREATER']; + mask: number; +} | { + func: WebGLRenderingContextBase['NOTEQUAL']; + mask: number; +} | { + func: WebGLRenderingContextBase['GEQUAL']; + mask: number; +} | { + func: WebGLRenderingContextBase['ALWAYS']; + mask: 0; +}; + +export type CullFaceModeType = WebGLRenderingContextBase['FRONT'] | WebGLRenderingContextBase['BACK'] | WebGLRenderingContextBase['FRONT_AND_BACK']; + +export type FrontFaceType = WebGLRenderingContextBase['CW'] | WebGLRenderingContextBase['CCW']; diff --git a/web/libraries/maplibre-gl/src/gl/value.ts b/web/libraries/maplibre-gl/src/gl/value.ts new file mode 100644 index 00000000..0e8e1b3c --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/value.ts @@ -0,0 +1,534 @@ +import {Color} from '@maplibre/maplibre-gl-style-spec'; +import {isWebGL2} from './webgl2'; + +import type {Context} from './context'; +import type { + BlendFuncType, + BlendEquationType, + ColorMaskType, + DepthRangeType, + DepthMaskType, + StencilFuncType, + StencilOpType, + DepthFuncType, + TextureUnitType, + ViewportType, + CullFaceModeType, + FrontFaceType, +} from './types'; + +export interface IValue { + current: T; + default: T; + dirty: boolean; + get(): T; + setDefault(): void; + set(value: T): void; +} + +class BaseValue implements IValue { + gl: WebGLRenderingContext|WebGL2RenderingContext; + current: T; + default: T; + dirty: boolean; + + constructor(context: Context) { + this.gl = context.gl; + this.default = this.getDefault(); + this.current = this.default; + this.dirty = false; + } + + get(): T { + return this.current; + } + set(value: T) { // eslint-disable-line + // overridden in child classes; + } + + getDefault(): T { + return this.default; // overriden in child classes + } + setDefault() { + this.set(this.default); + } +} + +export class ClearColor extends BaseValue { + getDefault(): Color { + return Color.transparent; + } + set(v: Color) { + const c = this.current; + if (v.r === c.r && v.g === c.g && v.b === c.b && v.a === c.a && !this.dirty) return; + this.gl.clearColor(v.r, v.g, v.b, v.a); + this.current = v; + this.dirty = false; + } +} + +export class ClearDepth extends BaseValue { + getDefault(): number { + return 1; + } + set(v: number) { + if (v === this.current && !this.dirty) return; + this.gl.clearDepth(v); + this.current = v; + this.dirty = false; + } +} + +export class ClearStencil extends BaseValue { + getDefault(): number { + return 0; + } + set(v: number) { + if (v === this.current && !this.dirty) return; + this.gl.clearStencil(v); + this.current = v; + this.dirty = false; + } +} + +export class ColorMask extends BaseValue { + getDefault(): ColorMaskType { + return [true, true, true, true]; + } + set(v: ColorMaskType) { + const c = this.current; + if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && v[3] === c[3] && !this.dirty) return; + this.gl.colorMask(v[0], v[1], v[2], v[3]); + this.current = v; + this.dirty = false; + } +} + +export class DepthMask extends BaseValue { + getDefault(): DepthMaskType { + return true; + } + set(v: DepthMaskType): void { + if (v === this.current && !this.dirty) return; + this.gl.depthMask(v); + this.current = v; + this.dirty = false; + } +} + +export class StencilMask extends BaseValue { + getDefault(): number { + return 0xFF; + } + set(v: number): void { + if (v === this.current && !this.dirty) return; + this.gl.stencilMask(v); + this.current = v; + this.dirty = false; + } +} + +export class StencilFunc extends BaseValue { + getDefault(): StencilFuncType { + return { + func: this.gl.ALWAYS, + ref: 0, + mask: 0xFF + }; + } + set(v: StencilFuncType): void { + const c = this.current; + if (v.func === c.func && v.ref === c.ref && v.mask === c.mask && !this.dirty) return; + this.gl.stencilFunc(v.func, v.ref, v.mask); + this.current = v; + this.dirty = false; + } +} + +export class StencilOp extends BaseValue { + getDefault(): StencilOpType { + const gl = this.gl; + return [gl.KEEP, gl.KEEP, gl.KEEP]; + } + set(v: StencilOpType) { + const c = this.current; + if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && !this.dirty) return; + this.gl.stencilOp(v[0], v[1], v[2]); + this.current = v; + this.dirty = false; + } +} + +export class StencilTest extends BaseValue { + getDefault(): boolean { + return false; + } + set(v: boolean) { + if (v === this.current && !this.dirty) return; + const gl = this.gl; + if (v) { + gl.enable(gl.STENCIL_TEST); + } else { + gl.disable(gl.STENCIL_TEST); + } + this.current = v; + this.dirty = false; + } +} + +export class DepthRange extends BaseValue { + getDefault(): DepthRangeType { + return [0, 1]; + } + set(v: DepthRangeType) { + const c = this.current; + if (v[0] === c[0] && v[1] === c[1] && !this.dirty) return; + this.gl.depthRange(v[0], v[1]); + this.current = v; + this.dirty = false; + } +} + +export class DepthTest extends BaseValue { + getDefault(): boolean { + return false; + } + set(v: boolean) { + if (v === this.current && !this.dirty) return; + const gl = this.gl; + if (v) { + gl.enable(gl.DEPTH_TEST); + } else { + gl.disable(gl.DEPTH_TEST); + } + this.current = v; + this.dirty = false; + } +} + +export class DepthFunc extends BaseValue { + getDefault(): DepthFuncType { + return this.gl.LESS; + } + set(v: DepthFuncType) { + if (v === this.current && !this.dirty) return; + this.gl.depthFunc(v); + this.current = v; + this.dirty = false; + } +} + +export class Blend extends BaseValue { + getDefault(): boolean { + return false; + } + set(v: boolean) { + if (v === this.current && !this.dirty) return; + const gl = this.gl; + if (v) { + gl.enable(gl.BLEND); + } else { + gl.disable(gl.BLEND); + } + this.current = v; + this.dirty = false; + } +} + +export class BlendFunc extends BaseValue { + getDefault(): BlendFuncType { + const gl = this.gl; + return [gl.ONE, gl.ZERO]; + } + set(v: BlendFuncType) { + const c = this.current; + if (v[0] === c[0] && v[1] === c[1] && !this.dirty) return; + this.gl.blendFunc(v[0], v[1]); + this.current = v; + this.dirty = false; + } +} + +export class BlendColor extends BaseValue { + getDefault(): Color { + return Color.transparent; + } + set(v: Color) { + const c = this.current; + if (v.r === c.r && v.g === c.g && v.b === c.b && v.a === c.a && !this.dirty) return; + this.gl.blendColor(v.r, v.g, v.b, v.a); + this.current = v; + this.dirty = false; + } +} + +export class BlendEquation extends BaseValue { + getDefault(): BlendEquationType { + return this.gl.FUNC_ADD; + } + set(v: BlendEquationType) { + if (v === this.current && !this.dirty) return; + this.gl.blendEquation(v); + this.current = v; + this.dirty = false; + } +} + +export class CullFace extends BaseValue { + getDefault(): boolean { + return false; + } + set(v: boolean) { + if (v === this.current && !this.dirty) return; + const gl = this.gl; + if (v) { + gl.enable(gl.CULL_FACE); + } else { + gl.disable(gl.CULL_FACE); + } + this.current = v; + this.dirty = false; + } +} + +export class CullFaceSide extends BaseValue { + getDefault(): CullFaceModeType { + return this.gl.BACK; + } + set(v: CullFaceModeType) { + if (v === this.current && !this.dirty) return; + this.gl.cullFace(v); + this.current = v; + this.dirty = false; + } +} + +export class FrontFace extends BaseValue { + getDefault(): FrontFaceType { + return this.gl.CCW; + } + set(v: FrontFaceType) { + if (v === this.current && !this.dirty) return; + this.gl.frontFace(v); + this.current = v; + this.dirty = false; + } +} + +export class ProgramValue extends BaseValue { + getDefault(): WebGLProgram { + return null; + } + set(v?: WebGLProgram | null) { + if (v === this.current && !this.dirty) return; + this.gl.useProgram(v); + this.current = v; + this.dirty = false; + } +} + +export class ActiveTextureUnit extends BaseValue { + getDefault(): TextureUnitType { + return this.gl.TEXTURE0; + } + set(v: TextureUnitType) { + if (v === this.current && !this.dirty) return; + this.gl.activeTexture(v); + this.current = v; + this.dirty = false; + } +} + +export class Viewport extends BaseValue { + getDefault(): ViewportType { + const gl = this.gl; + return [0, 0, gl.drawingBufferWidth, gl.drawingBufferHeight]; + } + set(v: ViewportType) { + const c = this.current; + if (v[0] === c[0] && v[1] === c[1] && v[2] === c[2] && v[3] === c[3] && !this.dirty) return; + this.gl.viewport(v[0], v[1], v[2], v[3]); + this.current = v; + this.dirty = false; + } +} + +export class BindFramebuffer extends BaseValue { + getDefault(): WebGLFramebuffer { + return null; + } + set(v?: WebGLFramebuffer | null) { + if (v === this.current && !this.dirty) return; + const gl = this.gl; + gl.bindFramebuffer(gl.FRAMEBUFFER, v); + this.current = v; + this.dirty = false; + } +} + +export class BindRenderbuffer extends BaseValue { + getDefault(): WebGLRenderbuffer { + return null; + } + set(v?: WebGLRenderbuffer | null) { + if (v === this.current && !this.dirty) return; + const gl = this.gl; + gl.bindRenderbuffer(gl.RENDERBUFFER, v); + this.current = v; + this.dirty = false; + } +} + +export class BindTexture extends BaseValue { + getDefault(): WebGLTexture { + return null; + } + set(v?: WebGLTexture | null) { + if (v === this.current && !this.dirty) return; + const gl = this.gl; + gl.bindTexture(gl.TEXTURE_2D, v); + this.current = v; + this.dirty = false; + } +} + +export class BindVertexBuffer extends BaseValue { + getDefault(): WebGLBuffer { + return null; + } + set(v?: WebGLBuffer | null) { + if (v === this.current && !this.dirty) return; + const gl = this.gl; + gl.bindBuffer(gl.ARRAY_BUFFER, v); + this.current = v; + this.dirty = false; + } +} + +export class BindElementBuffer extends BaseValue { + getDefault(): WebGLBuffer { + return null; + } + set(v?: WebGLBuffer | null) { + // Always rebind + const gl = this.gl; + gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, v); + this.current = v; + this.dirty = false; + } +} + +export class BindVertexArray extends BaseValue { + getDefault(): WebGLVertexArrayObject | null { + return null; + } + set(v: WebGLVertexArrayObject | null) { + if (v === this.current && !this.dirty) return; + const gl = this.gl; + + if (isWebGL2(gl)) { + gl.bindVertexArray(v); + } else { + gl.getExtension('OES_vertex_array_object')?.bindVertexArrayOES(v); + } + + this.current = v; + this.dirty = false; + } +} + +export class PixelStoreUnpack extends BaseValue { + getDefault(): number { + return 4; + } + set(v: number) { + if (v === this.current && !this.dirty) return; + const gl = this.gl; + gl.pixelStorei(gl.UNPACK_ALIGNMENT, v); + this.current = v; + this.dirty = false; + } +} + +export class PixelStoreUnpackPremultiplyAlpha extends BaseValue { + getDefault(): boolean { + return false; + } + set(v: boolean): void { + if (v === this.current && !this.dirty) return; + const gl = this.gl; + gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, ((v as any))); + this.current = v; + this.dirty = false; + } +} + +export class PixelStoreUnpackFlipY extends BaseValue { + getDefault(): boolean { + return false; + } + set(v: boolean): void { + if (v === this.current && !this.dirty) return; + const gl = this.gl; + gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, ((v as any))); + this.current = v; + this.dirty = false; + } +} + +class FramebufferAttachment extends BaseValue { + parent: WebGLFramebuffer; + context: Context; + + constructor(context: Context, parent: WebGLFramebuffer) { + super(context); + this.context = context; + this.parent = parent; + } + getDefault() { + return null; + } +} + +export class ColorAttachment extends FramebufferAttachment { + setDirty() { + this.dirty = true; + } + set(v?: WebGLTexture | null): void { + if (v === this.current && !this.dirty) return; + this.context.bindFramebuffer.set(this.parent); + // note: it's possible to attach a renderbuffer to the color + // attachment point, but thus far MBGL only uses textures for color + const gl = this.gl; + gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, v, 0); + + this.current = v; + this.dirty = false; + } +} + +export class DepthAttachment extends FramebufferAttachment { + set(v?: WebGLRenderbuffer | null): void { + if (v === this.current && !this.dirty) return; + this.context.bindFramebuffer.set(this.parent); + // note: it's possible to attach a texture to the depth attachment + // point, but thus far MBGL only uses renderbuffers for depth + const gl = this.gl; + gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_ATTACHMENT, gl.RENDERBUFFER, v); + this.current = v; + this.dirty = false; + } +} + +export class DepthStencilAttachment extends FramebufferAttachment { + set(v?: WebGLRenderbuffer | null): void { + if (v === this.current && !this.dirty) return; + this.context.bindFramebuffer.set(this.parent); + // note: it's possible to attach a texture to the depth attachment + // point, but thus far MBGL only uses renderbuffers for depth + const gl = this.gl; + gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, v); + this.current = v; + this.dirty = false; + } +} diff --git a/web/libraries/maplibre-gl/src/gl/vertex_buffer.test.ts b/web/libraries/maplibre-gl/src/gl/vertex_buffer.test.ts new file mode 100644 index 00000000..341bbab0 --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/vertex_buffer.test.ts @@ -0,0 +1,52 @@ +import {VertexBuffer} from './vertex_buffer'; +import {StructArrayLayout3i6} from '../data/array_types.g'; +import {Context} from '../gl/context'; +import gl from 'gl'; +import {StructArrayMember} from '../util/struct_array'; + +describe('VertexBuffer', () => { + class TestArray extends StructArrayLayout3i6 {} + const attributes = [ + {name: 'map', components: 1, type: 'Int16', offset: 0}, + {name: 'box', components: 2, type: 'Int16', offset: 4} + ] as StructArrayMember[]; + + test('constructs itself', () => { + const context = new Context(gl(10, 10) as any); + const array = new TestArray(); + array.emplaceBack(1, 1, 1); + array.emplaceBack(1, 1, 1); + array.emplaceBack(1, 1, 1); + + const buffer = new VertexBuffer(context, array, attributes); + + expect(buffer.attributes).toEqual([ + {name: 'map', components: 1, type: 'Int16', offset: 0}, + {name: 'box', components: 2, type: 'Int16', offset: 4} + ]); + expect(buffer.itemSize).toBe(6); + expect(buffer).toHaveLength(3); + }); + + test('enableAttributes', () => { + const context = new Context(gl(10, 10) as any); + const array = new TestArray(); + const buffer = new VertexBuffer(context, array, attributes); + const spy = jest.spyOn(context.gl, 'enableVertexAttribArray').mockImplementation(() => {}); + buffer.enableAttributes(context.gl, {attributes: {map: 5, box: 6}} as any); + expect(spy.mock.calls).toEqual([[5], [6]]); + }); + + test('setVertexAttribPointers', () => { + const context = new Context(gl(10, 10) as any); + const array = new TestArray(); + const buffer = new VertexBuffer(context, array, attributes); + const spy = jest.spyOn(context.gl, 'vertexAttribPointer').mockImplementation(() => {}); + buffer.setVertexAttribPointers(context.gl, {attributes: {map: 5, box: 6}} as any, 50); + expect(spy.mock.calls).toEqual([ + [5, 1, context.gl['SHORT'], false, 6, 300], + [6, 2, context.gl['SHORT'], false, 6, 304] + ]); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/gl/vertex_buffer.ts b/web/libraries/maplibre-gl/src/gl/vertex_buffer.ts new file mode 100644 index 00000000..63f55adc --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/vertex_buffer.ts @@ -0,0 +1,111 @@ + +import type { + StructArray, + StructArrayMember +} from '../util/struct_array'; + +import type {Program} from '../render/program'; +import type {Context} from '../gl/context'; + +/** + * An Enum for AttributeType + */ +const AttributeType = { + Int8: 'BYTE', + Uint8: 'UNSIGNED_BYTE', + Int16: 'SHORT', + Uint16: 'UNSIGNED_SHORT', + Int32: 'INT', + Uint32: 'UNSIGNED_INT', + Float32: 'FLOAT' +}; + +/** + * @internal + * The `VertexBuffer` class turns a `StructArray` into a WebGL buffer. Each member of the StructArray's + * Struct type is converted to a WebGL attribute. + */ +export class VertexBuffer { + length: number; + attributes: ReadonlyArray; + itemSize: number; + dynamicDraw: boolean; + context: Context; + buffer: WebGLBuffer; + + /** + * @param dynamicDraw - Whether this buffer will be repeatedly updated. + */ + constructor(context: Context, array: StructArray, attributes: ReadonlyArray, dynamicDraw?: boolean) { + this.length = array.length; + this.attributes = attributes; + this.itemSize = array.bytesPerElement; + this.dynamicDraw = dynamicDraw; + + this.context = context; + const gl = context.gl; + this.buffer = gl.createBuffer(); + context.bindVertexBuffer.set(this.buffer); + gl.bufferData(gl.ARRAY_BUFFER, array.arrayBuffer, this.dynamicDraw ? gl.DYNAMIC_DRAW : gl.STATIC_DRAW); + + if (!this.dynamicDraw) { + delete array.arrayBuffer; + } + } + + bind() { + this.context.bindVertexBuffer.set(this.buffer); + } + + updateData(array: StructArray) { + if (array.length !== this.length) throw new Error(`Length of new data is ${array.length}, which doesn't match current length of ${this.length}`); + const gl = this.context.gl; + this.bind(); + gl.bufferSubData(gl.ARRAY_BUFFER, 0, array.arrayBuffer); + } + + enableAttributes(gl: WebGLRenderingContext|WebGL2RenderingContext, program: Program) { + for (let j = 0; j < this.attributes.length; j++) { + const member = this.attributes[j]; + const attribIndex: number | void = program.attributes[member.name]; + if (attribIndex !== undefined) { + gl.enableVertexAttribArray(attribIndex); + } + } + } + + /** + * Set the attribute pointers in a WebGL context + * @param gl - The WebGL context + * @param program - The active WebGL program + * @param vertexOffset - Index of the starting vertex of the segment + */ + setVertexAttribPointers(gl: WebGLRenderingContext|WebGL2RenderingContext, program: Program, vertexOffset?: number | null) { + for (let j = 0; j < this.attributes.length; j++) { + const member = this.attributes[j]; + const attribIndex: number | void = program.attributes[member.name]; + + if (attribIndex !== undefined) { + gl.vertexAttribPointer( + attribIndex, + member.components, + (gl as any)[AttributeType[member.type]], + false, + this.itemSize, + member.offset + (this.itemSize * (vertexOffset || 0)) + ); + } + } + } + + /** + * Destroy the GL buffer bound to the given WebGL context + */ + destroy() { + const gl = this.context.gl; + if (this.buffer) { + gl.deleteBuffer(this.buffer); + delete this.buffer; + } + } +} diff --git a/web/libraries/maplibre-gl/src/gl/webgl2.ts b/web/libraries/maplibre-gl/src/gl/webgl2.ts new file mode 100644 index 00000000..2b8f18a8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/gl/webgl2.ts @@ -0,0 +1,12 @@ +const cache = new WeakMap(); +export function isWebGL2( + gl: WebGLRenderingContext | WebGL2RenderingContext +): gl is WebGL2RenderingContext { + if (cache.has(gl)) { + return cache.get(gl); + } else { + const value = gl.getParameter(gl.VERSION)?.startsWith('WebGL 2.0'); + cache.set(gl, value); + return value; + } +} diff --git a/web/libraries/maplibre-gl/src/index.test.ts b/web/libraries/maplibre-gl/src/index.test.ts new file mode 100644 index 00000000..628bbe2d --- /dev/null +++ b/web/libraries/maplibre-gl/src/index.test.ts @@ -0,0 +1,129 @@ +import {config} from './util/config'; +import maplibre from './index'; +import {getJSON, getArrayBuffer} from './util/ajax'; +import {ImageRequest} from './util/image_request'; + +describe('maplibre', () => { + beforeEach(() => { + config.REGISTERED_PROTOCOLS = {}; + }); + afterAll(() => { + config.REGISTERED_PROTOCOLS = {}; + }); + + test('workerCount', () => { + expect(typeof maplibre.workerCount === 'number').toBeTruthy(); + }); + + test('addProtocol', () => { + const protocolName = 'custom'; + expect(Object.keys(config.REGISTERED_PROTOCOLS)).toHaveLength(0); + + maplibre.addProtocol(protocolName, () => { return {cancel: () => { }}; }); + expect(Object.keys(config.REGISTERED_PROTOCOLS)[0]).toBe(protocolName); + }); + + test('removeProtocol', () => { + const protocolName = 'custom'; + expect(Object.keys(config.REGISTERED_PROTOCOLS)).toHaveLength(0); + + maplibre.addProtocol(protocolName, () => { return {cancel: () => { }}; }); + expect(Object.keys(config.REGISTERED_PROTOCOLS)[0]).toBe(protocolName); + + maplibre.removeProtocol(protocolName); + expect(Object.keys(config.REGISTERED_PROTOCOLS)).toHaveLength(0); + }); + + test('#addProtocol - getJSON', done => { + let protocolCallbackCalled = false; + maplibre.addProtocol('custom', (reqParam, callback) => { + protocolCallbackCalled = true; + callback(null, {'foo': 'bar'}); + return {cancel: () => {}}; + }); + getJSON({url: 'custom://test/url/json'}, (error, data) => { + expect(error).toBeFalsy(); + expect(data).toEqual({foo: 'bar'}); + expect(protocolCallbackCalled).toBeTruthy(); + done(); + }); + }); + + test('#addProtocol - getArrayBuffer', done => { + let protocolCallbackCalled = false; + maplibre.addProtocol('custom', (reqParam, callback) => { + protocolCallbackCalled = true; + callback(null, new ArrayBuffer(1)); + return {cancel: () => {}}; + }); + getArrayBuffer({url: 'custom://test/url/getArrayBuffer'}, async (error, data) => { + expect(error).toBeFalsy(); + expect(data).toBeInstanceOf(ArrayBuffer); + expect(protocolCallbackCalled).toBeTruthy(); + done(); + }); + }); + + test('#addProtocol - returning ImageBitmap for getImage', done => { + let protocolCallbackCalled = false; + maplibre.addProtocol('custom', (reqParam, callback) => { + protocolCallbackCalled = true; + callback(null, new ImageBitmap()); + return {cancel: () => {}}; + }); + + ImageRequest.getImage({url: 'custom://test/url/getImage'}, async (error, img) => { + expect(error).toBeFalsy(); + expect(img).toBeInstanceOf(ImageBitmap); + expect(protocolCallbackCalled).toBeTruthy(); + done(); + }); + }); + + test('#addProtocol - returning HTMLImageElement for getImage', done => { + let protocolCallbackCalled = false; + maplibre.addProtocol('custom', (reqParam, callback) => { + protocolCallbackCalled = true; + callback(null, new Image()); + return {cancel: () => {}}; + }); + ImageRequest.getImage({url: 'custom://test/url/getImage'}, async (error, img) => { + expect(error).toBeFalsy(); + expect(img).toBeInstanceOf(HTMLImageElement); + expect(protocolCallbackCalled).toBeTruthy(); + done(); + }); + }); + + test('#addProtocol - error', () => { + maplibre.addProtocol('custom', (reqParam, callback) => { + callback(new Error('error')); + return {cancel: () => { }}; + }); + + getJSON({url: 'custom://test/url/json'}, (error) => { + expect(error).toBeTruthy(); + }); + }); + + test('#addProtocol - Cancel request', () => { + let cancelCalled = false; + maplibre.addProtocol('custom', () => { + return {cancel: () => { + cancelCalled = true; + }}; + }); + const request = getJSON({url: 'custom://test/url/json'}, () => { }); + request.cancel(); + expect(cancelCalled).toBeTruthy(); + }); + + test('version', () => { + expect(typeof maplibre.version === 'string').toBeTruthy(); + + // Semver regex: https://gist.github.com/jhorsman/62eeea161a13b80e39f5249281e17c39 + // Backslashes are doubled to escape them + const regexp = new RegExp('^([0-9]+)\\.([0-9]+)\\.([0-9]+)(?:-([0-9A-Za-z-]+(?:\\.[0-9A-Za-z-]+)*))?(?:\\+[0-9A-Za-z-]+)?$'); + expect(regexp.test(maplibre.version)).toBeTruthy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/index.ts b/web/libraries/maplibre-gl/src/index.ts new file mode 100644 index 00000000..0ac560ce --- /dev/null +++ b/web/libraries/maplibre-gl/src/index.ts @@ -0,0 +1,239 @@ +import packageJSON from '../package.json' assert {type: 'json'}; +import {Map} from './ui/map'; +import {NavigationControl} from './ui/control/navigation_control'; +import {GeolocateControl} from './ui/control/geolocate_control'; +import {AttributionControl} from './ui/control/attribution_control'; +import {LogoControl} from './ui/control/logo_control'; +import {ScaleControl} from './ui/control/scale_control'; +import {FullscreenControl} from './ui/control/fullscreen_control'; +import {TerrainControl} from './ui/control/terrain_control'; +import {Popup} from './ui/popup'; +import {Marker} from './ui/marker'; +import {Style} from './style/style'; +import {LngLat} from './geo/lng_lat'; +import {LngLatBounds} from './geo/lng_lat_bounds'; +import Point from '@mapbox/point-geometry'; +import {MercatorCoordinate} from './geo/mercator_coordinate'; +import {Evented} from './util/evented'; +import {config} from './util/config'; +import {Debug} from './util/debug'; +import {isSafari} from './util/util'; +import {setRTLTextPlugin, getRTLTextPluginStatus} from './source/rtl_text_plugin'; +import {WorkerPool} from './util/worker_pool'; +import {prewarm, clearPrewarmedResources} from './util/global_worker_pool'; +import {PerformanceUtils} from './util/performance'; +import {AJAXError} from './util/ajax'; +import type {RequestParameters, ResponseCallback} from './util/ajax'; +import type {Cancelable} from './types/cancelable'; +import {GeoJSONSource} from './source/geojson_source'; +import {CanvasSource} from './source/canvas_source'; +import {ImageSource} from './source/image_source'; +import {RasterDEMTileSource} from './source/raster_dem_tile_source'; +import {RasterTileSource} from './source/raster_tile_source'; +import {VectorTileSource} from './source/vector_tile_source'; +import {VideoSource} from './source/video_source'; + +const version = packageJSON.version; + +export type * from '@maplibre/maplibre-gl-style-spec'; + +/** + * `maplibregl` is the global object that allows configurations that are not specific to a map instance + * + * @group Main + */ +class MapLibreGL { + static Map = Map; + static NavigationControl = NavigationControl; + static GeolocateControl = GeolocateControl; + static AttributionControl = AttributionControl; + static LogoControl = LogoControl; + static ScaleControl = ScaleControl; + static FullscreenControl = FullscreenControl; + static TerrainControl = TerrainControl; + static Popup = Popup; + static Marker = Marker; + static Style = Style; + static LngLat = LngLat; + static LngLatBounds = LngLatBounds; + static Point = Point; + static MercatorCoordinate = MercatorCoordinate; + static Evented = Evented; + static AJAXError = AJAXError; + static config = config; + static CanvasSource = CanvasSource; + static GeoJSONSource = GeoJSONSource; + static ImageSource = ImageSource; + static RasterDEMTileSource = RasterDEMTileSource; + static RasterTileSource = RasterTileSource; + static VectorTileSource = VectorTileSource; + static VideoSource = VideoSource; + /** + * Sets the map's [RTL text plugin](https://www.mapbox.com/mapbox-gl-js/plugins/#mapbox-gl-rtl-text). + * Necessary for supporting the Arabic and Hebrew languages, which are written right-to-left. + * + * @param pluginURL - URL pointing to the Mapbox RTL text plugin source. + * @param callback - Called with an error argument if there is an error. + * @param lazy - If set to `true`, mapboxgl will defer loading the plugin until rtl text is encountered, + * rtl text will then be rendered only after the plugin finishes loading. + * @example + * ```ts + * maplibregl.setRTLTextPlugin('https://unpkg.com/@mapbox/mapbox-gl-rtl-text@0.2.3/mapbox-gl-rtl-text.js'); + * ``` + * @see [Add support for right-to-left scripts](https://maplibre.org/maplibre-gl-js/docs/examples/mapbox-gl-rtl-text/) + */ + static setRTLTextPlugin = setRTLTextPlugin; + /** + * Gets the map's [RTL text plugin](https://www.mapbox.com/mapbox-gl-js/plugins/#mapbox-gl-rtl-text) status. + * The status can be `unavailable` (i.e. not requested or removed), `loading`, `loaded` or `error`. + * If the status is `loaded` and the plugin is requested again, an error will be thrown. + * + * @example + * ```ts + * const pluginStatus = maplibregl.getRTLTextPluginStatus(); + * ``` + */ + static getRTLTextPluginStatus = getRTLTextPluginStatus; + /** + * Initializes resources like WebWorkers that can be shared across maps to lower load + * times in some situations. `maplibregl.workerUrl` and `maplibregl.workerCount`, if being + * used, must be set before `prewarm()` is called to have an effect. + * + * By default, the lifecycle of these resources is managed automatically, and they are + * lazily initialized when a Map is first created. By invoking `prewarm()`, these + * resources will be created ahead of time, and will not be cleared when the last Map + * is removed from the page. This allows them to be re-used by new Map instances that + * are created later. They can be manually cleared by calling + * `maplibregl.clearPrewarmedResources()`. This is only necessary if your web page remains + * active but stops using maps altogether. + * + * This is primarily useful when using GL-JS maps in a single page app, wherein a user + * would navigate between various views that can cause Map instances to constantly be + * created and destroyed. + * + * @example + * ```ts + * maplibregl.prewarm() + * ``` + */ + static prewarm = prewarm; + /** + * Clears up resources that have previously been created by `maplibregl.prewarm()`. + * Note that this is typically not necessary. You should only call this function + * if you expect the user of your app to not return to a Map view at any point + * in your application. + * + * @example + * ```ts + * maplibregl.clearPrewarmedResources() + * ``` + */ + static clearPrewarmedResources = clearPrewarmedResources; + /** + * Returns the package version of the library + * @returns Package version of the library + */ + static get version(): string { + return version; + } + + /** + * Gets and sets the number of web workers instantiated on a page with GL JS maps. + * By default, workerCount is 1 except for Safari browser where it is set to half the number of CPU cores (capped at 3). + * Make sure to set this property before creating any map instances for it to have effect. + * + * @returns Number of workers currently configured. + * @example + * ```ts + * maplibregl.workerCount = 2; + * ``` + */ + static get workerCount(): number { + return WorkerPool.workerCount; + } + + static set workerCount(count: number) { + WorkerPool.workerCount = count; + } + /** + * Gets and sets the maximum number of images (raster tiles, sprites, icons) to load in parallel, + * which affects performance in raster-heavy maps. 16 by default. + * + * @returns Number of parallel requests currently configured. + * @example + * ```ts + * maplibregl.maxParallelImageRequests = 10; + * ``` + */ + static get maxParallelImageRequests(): number { + return config.MAX_PARALLEL_IMAGE_REQUESTS; + } + + static set maxParallelImageRequests(numRequests: number) { + config.MAX_PARALLEL_IMAGE_REQUESTS = numRequests; + } + + static get workerUrl(): string { + return config.WORKER_URL; + } + + static set workerUrl(value: string) { + config.WORKER_URL = value; + } + + /** + * Sets a custom load tile function that will be called when using a source that starts with a custom url schema. + * The example below will be triggered for custom:// urls defined in the sources list in the style definitions. + * The function passed will receive the request parameters and should call the callback with the resulting request, + * for example a pbf vector tile, non-compressed, represented as ArrayBuffer. + * + * @param customProtocol - the protocol to hook, for example 'custom' + * @param loadFn - the function to use when trying to fetch a tile specified by the customProtocol + * @example + * This will fetch a file using the fetch API (this is obviously a non interesting example...) + * ```ts + * maplibregl.addProtocol('custom', (params, callback) => { + fetch(`https://${params.url.split("://")[1]}`) + .then(t => { + if (t.status == 200) { + t.arrayBuffer().then(arr => { + callback(null, arr, null, null); + }); + } else { + callback(new Error(`Tile fetch error: ${t.statusText}`)); + } + }) + .catch(e => { + callback(new Error(e)); + }); + return { cancel: () => { } }; + }); + * // the following is an example of a way to return an error when trying to load a tile + * maplibregl.addProtocol('custom2', (params, callback) => { + * callback(new Error('someErrorMessage')); + * return { cancel: () => { } }; + * }); + * ``` + */ + static addProtocol(customProtocol: string, loadFn: (requestParameters: RequestParameters, callback: ResponseCallback) => Cancelable) { + config.REGISTERED_PROTOCOLS[customProtocol] = loadFn; + } + + /** + * Removes a previously added protocol + * + * @param customProtocol - the custom protocol to remove registration for + * @example + * ```ts + * maplibregl.removeProtocol('custom'); + * ``` + */ + static removeProtocol(customProtocol: string) { + delete config.REGISTERED_PROTOCOLS[customProtocol]; + } +} + +//This gets automatically stripped out in production builds. +Debug.extend(MapLibreGL, {isSafari, getPerformanceMetrics: PerformanceUtils.getPerformanceMetrics}); + +export default MapLibreGL; diff --git a/web/libraries/maplibre-gl/src/render/draw_background.ts b/web/libraries/maplibre-gl/src/render/draw_background.ts new file mode 100644 index 00000000..16ba882c --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_background.ts @@ -0,0 +1,53 @@ +import {StencilMode} from '../gl/stencil_mode'; +import {DepthMode} from '../gl/depth_mode'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import { + backgroundUniformValues, + backgroundPatternUniformValues +} from './program/background_program'; + +import type {Painter} from './painter'; +import type {SourceCache} from '../source/source_cache'; +import type {BackgroundStyleLayer} from '../style/style_layer/background_style_layer'; +import {OverscaledTileID} from '../source/tile_id'; + +export function drawBackground(painter: Painter, sourceCache: SourceCache, layer: BackgroundStyleLayer, coords?: Array) { + const color = layer.paint.get('background-color'); + const opacity = layer.paint.get('background-opacity'); + + if (opacity === 0) return; + + const context = painter.context; + const gl = context.gl; + const transform = painter.transform; + const tileSize = transform.tileSize; + const image = layer.paint.get('background-pattern'); + if (painter.isPatternMissing(image)) return; + + const pass = (!image && color.a === 1 && opacity === 1 && painter.opaquePassEnabledForLayer()) ? 'opaque' : 'translucent'; + if (painter.renderPass !== pass) return; + + const stencilMode = StencilMode.disabled; + const depthMode = painter.depthModeForSublayer(0, pass === 'opaque' ? DepthMode.ReadWrite : DepthMode.ReadOnly); + const colorMode = painter.colorModeForRenderPass(); + const program = painter.useProgram(image ? 'backgroundPattern' : 'background'); + const tileIDs = coords ? coords : transform.coveringTiles({tileSize, terrain: painter.style.map.terrain}); + + if (image) { + context.activeTexture.set(gl.TEXTURE0); + painter.imageManager.bind(painter.context); + } + + const crossfade = layer.getCrossfadeParameters(); + for (const tileID of tileIDs) { + const matrix = coords ? tileID.posMatrix : painter.transform.calculatePosMatrix(tileID.toUnwrapped()); + const uniformValues = image ? + backgroundPatternUniformValues(matrix, opacity, painter, image, {tileID, tileSize}, crossfade) : + backgroundUniformValues(matrix, opacity, color); + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(tileID); + + program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled, + uniformValues, terrainData, layer.id, painter.tileExtentBuffer, + painter.quadTriangleIndexBuffer, painter.tileExtentSegments); + } +} diff --git a/web/libraries/maplibre-gl/src/render/draw_circle.ts b/web/libraries/maplibre-gl/src/render/draw_circle.ts new file mode 100644 index 00000000..be08b0dd --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_circle.ts @@ -0,0 +1,113 @@ +import {StencilMode} from '../gl/stencil_mode'; +import {DepthMode} from '../gl/depth_mode'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import {Program} from './program'; +import {circleUniformValues} from './program/circle_program'; +import {SegmentVector} from '../data/segment'; +import {OverscaledTileID} from '../source/tile_id'; + +import type {Painter} from './painter'; +import type {SourceCache} from '../source/source_cache'; +import type {CircleStyleLayer} from '../style/style_layer/circle_style_layer'; +import type {CircleBucket} from '../data/bucket/circle_bucket'; +import type {ProgramConfiguration} from '../data/program_configuration'; +import type {VertexBuffer} from '../gl/vertex_buffer'; +import type {IndexBuffer} from '../gl/index_buffer'; +import type {UniformValues} from './uniform_binding'; +import type {CircleUniformsType} from './program/circle_program'; +import type {TerrainData} from '../render/terrain'; + +type TileRenderState = { + programConfiguration: ProgramConfiguration; + program: Program; + layoutVertexBuffer: VertexBuffer; + indexBuffer: IndexBuffer; + uniformValues: UniformValues; + terrainData: TerrainData; +}; + +type SegmentsTileRenderState = { + segments: SegmentVector; + sortKey: number; + state: TileRenderState; +}; + +export function drawCircles(painter: Painter, sourceCache: SourceCache, layer: CircleStyleLayer, coords: Array) { + if (painter.renderPass !== 'translucent') return; + + const opacity = layer.paint.get('circle-opacity'); + const strokeWidth = layer.paint.get('circle-stroke-width'); + const strokeOpacity = layer.paint.get('circle-stroke-opacity'); + const sortFeaturesByKey = !layer.layout.get('circle-sort-key').isConstant(); + + if (opacity.constantOr(1) === 0 && (strokeWidth.constantOr(1) === 0 || strokeOpacity.constantOr(1) === 0)) { + return; + } + + const context = painter.context; + const gl = context.gl; + + const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly); + // Turn off stencil testing to allow circles to be drawn across boundaries, + // so that large circles are not clipped to tiles + const stencilMode = StencilMode.disabled; + const colorMode = painter.colorModeForRenderPass(); + + const segmentsRenderStates: Array = []; + + for (let i = 0; i < coords.length; i++) { + const coord = coords[i]; + + const tile = sourceCache.getTile(coord); + const bucket: CircleBucket = (tile.getBucket(layer) as any); + if (!bucket) continue; + + const programConfiguration = bucket.programConfigurations.get(layer.id); + const program = painter.useProgram('circle', programConfiguration); + const layoutVertexBuffer = bucket.layoutVertexBuffer; + const indexBuffer = bucket.indexBuffer; + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + const uniformValues = circleUniformValues(painter, coord, tile, layer); + + const state: TileRenderState = { + programConfiguration, + program, + layoutVertexBuffer, + indexBuffer, + uniformValues, + terrainData + }; + + if (sortFeaturesByKey) { + const oldSegments = bucket.segments.get(); + for (const segment of oldSegments) { + segmentsRenderStates.push({ + segments: new SegmentVector([segment]), + sortKey: (segment.sortKey as any as number), + state + }); + } + } else { + segmentsRenderStates.push({ + segments: bucket.segments, + sortKey: 0, + state + }); + } + + } + + if (sortFeaturesByKey) { + segmentsRenderStates.sort((a, b) => a.sortKey - b.sortKey); + } + + for (const segmentsState of segmentsRenderStates) { + const {programConfiguration, program, layoutVertexBuffer, indexBuffer, uniformValues, terrainData} = segmentsState.state; + const segments = segmentsState.segments; + + program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled, + uniformValues, terrainData, layer.id, + layoutVertexBuffer, indexBuffer, segments, + layer.paint, painter.transform.zoom, programConfiguration); + } +} diff --git a/web/libraries/maplibre-gl/src/render/draw_collision_debug.ts b/web/libraries/maplibre-gl/src/render/draw_collision_debug.ts new file mode 100644 index 00000000..876c5b22 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_collision_debug.ts @@ -0,0 +1,172 @@ +import type {Painter} from './painter'; +import type {SourceCache} from '../source/source_cache'; +import type {StyleLayer} from '../style/style_layer'; +import type {OverscaledTileID} from '../source/tile_id'; +import type {SymbolBucket} from '../data/bucket/symbol_bucket'; +import {DepthMode} from '../gl/depth_mode'; +import {StencilMode} from '../gl/stencil_mode'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import {collisionUniformValues, collisionCircleUniformValues} from './program/collision_program'; + +import {QuadTriangleArray, CollisionCircleLayoutArray} from '../data/array_types.g'; +import {collisionCircleLayout} from '../data/bucket/symbol_attributes'; +import {SegmentVector} from '../data/segment'; +import {mat4} from 'gl-matrix'; +import {VertexBuffer} from '../gl/vertex_buffer'; +import {IndexBuffer} from '../gl/index_buffer'; + +type TileBatch = { + circleArray: Array; + circleOffset: number; + transform: mat4; + invTransform: mat4; + coord: OverscaledTileID; +}; + +let quadTriangles: QuadTriangleArray; + +export function drawCollisionDebug(painter: Painter, sourceCache: SourceCache, layer: StyleLayer, coords: Array, translate: [number, number], translateAnchor: 'map' | 'viewport', isText: boolean) { + const context = painter.context; + const gl = context.gl; + const program = painter.useProgram('collisionBox'); + const tileBatches: Array = []; + let circleCount = 0; + let circleOffset = 0; + + for (let i = 0; i < coords.length; i++) { + const coord = coords[i]; + const tile = sourceCache.getTile(coord); + const bucket: SymbolBucket = (tile.getBucket(layer) as any); + if (!bucket) continue; + let posMatrix = coord.posMatrix; + if (translate[0] !== 0 || translate[1] !== 0) { + posMatrix = painter.translatePosMatrix(coord.posMatrix, tile, translate, translateAnchor); + } + const buffers = isText ? bucket.textCollisionBox : bucket.iconCollisionBox; + // Get collision circle data of this bucket + const circleArray: Array = bucket.collisionCircleArray; + if (circleArray.length > 0) { + // We need to know the projection matrix that was used for projecting collision circles to the screen. + // This might vary between buckets as the symbol placement is a continuous process. This matrix is + // required for transforming points from previous screen space to the current one + const invTransform = mat4.create(); + const transform = posMatrix; + + mat4.mul(invTransform, bucket.placementInvProjMatrix, painter.transform.glCoordMatrix); + mat4.mul(invTransform, invTransform, bucket.placementViewportMatrix); + + tileBatches.push({ + circleArray, + circleOffset, + transform, + invTransform, + coord + }); + + circleCount += circleArray.length / 4; // 4 values per circle + circleOffset = circleCount; + } + if (!buffers) continue; + program.draw(context, gl.LINES, + DepthMode.disabled, StencilMode.disabled, + painter.colorModeForRenderPass(), + CullFaceMode.disabled, + collisionUniformValues( + posMatrix, + painter.transform, + tile), + painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord), + layer.id, buffers.layoutVertexBuffer, buffers.indexBuffer, + buffers.segments, null, painter.transform.zoom, null, null, + buffers.collisionVertexBuffer); + } + + if (!isText || !tileBatches.length) { + return; + } + + // Render collision circles + const circleProgram = painter.useProgram('collisionCircle'); + + // Construct vertex data + const vertexData = new CollisionCircleLayoutArray(); + vertexData.resize(circleCount * 4); + vertexData._trim(); + + let vertexOffset = 0; + + for (const batch of tileBatches) { + for (let i = 0; i < batch.circleArray.length / 4; i++) { + const circleIdx = i * 4; + const x = batch.circleArray[circleIdx + 0]; + const y = batch.circleArray[circleIdx + 1]; + const radius = batch.circleArray[circleIdx + 2]; + const collision = batch.circleArray[circleIdx + 3]; + + // 4 floats per vertex, 4 vertices per quad + vertexData.emplace(vertexOffset++, x, y, radius, collision, 0); + vertexData.emplace(vertexOffset++, x, y, radius, collision, 1); + vertexData.emplace(vertexOffset++, x, y, radius, collision, 2); + vertexData.emplace(vertexOffset++, x, y, radius, collision, 3); + } + } + if (!quadTriangles || quadTriangles.length < circleCount * 2) { + quadTriangles = createQuadTriangles(circleCount); + } + + const indexBuffer: IndexBuffer = context.createIndexBuffer(quadTriangles, true); + const vertexBuffer: VertexBuffer = context.createVertexBuffer(vertexData, collisionCircleLayout.members, true); + + // Render batches + for (const batch of tileBatches) { + const uniforms = collisionCircleUniformValues( + batch.transform, + batch.invTransform, + painter.transform + ); + + circleProgram.draw( + context, + gl.TRIANGLES, + DepthMode.disabled, + StencilMode.disabled, + painter.colorModeForRenderPass(), + CullFaceMode.disabled, + uniforms, + painter.style.map.terrain && painter.style.map.terrain.getTerrainData(batch.coord), + layer.id, + vertexBuffer, + indexBuffer, + SegmentVector.simpleSegment(0, batch.circleOffset * 2, batch.circleArray.length, batch.circleArray.length / 2), + null, + painter.transform.zoom, + null, + null, + null); + } + + vertexBuffer.destroy(); + indexBuffer.destroy(); +} + +function createQuadTriangles(quadCount: number): QuadTriangleArray { + const triCount = quadCount * 2; + const array = new QuadTriangleArray(); + + array.resize(triCount); + array._trim(); + + // Two triangles and 4 vertices per quad. + for (let i = 0; i < triCount; i++) { + const idx = i * 6; + + array.uint16[idx + 0] = i * 4 + 0; + array.uint16[idx + 1] = i * 4 + 1; + array.uint16[idx + 2] = i * 4 + 2; + array.uint16[idx + 3] = i * 4 + 2; + array.uint16[idx + 4] = i * 4 + 3; + array.uint16[idx + 5] = i * 4 + 0; + } + + return array; +} diff --git a/web/libraries/maplibre-gl/src/render/draw_custom.ts b/web/libraries/maplibre-gl/src/render/draw_custom.ts new file mode 100644 index 00000000..f5dc360d --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_custom.ts @@ -0,0 +1,45 @@ +import {DepthMode} from '../gl/depth_mode'; +import {StencilMode} from '../gl/stencil_mode'; + +import type {Painter} from './painter'; +import type {SourceCache} from '../source/source_cache'; +import type {CustomStyleLayer} from '../style/style_layer/custom_style_layer'; + +export function drawCustom(painter: Painter, sourceCache: SourceCache, layer: CustomStyleLayer) { + + const context = painter.context; + const implementation = layer.implementation; + + if (painter.renderPass === 'offscreen') { + + const prerender = implementation.prerender; + if (prerender) { + painter.setCustomLayerDefaults(); + context.setColorMode(painter.colorModeForRenderPass()); + + prerender.call(implementation, context.gl, painter.transform.customLayerMatrix()); + + context.setDirty(); + painter.setBaseState(); + } + + } else if (painter.renderPass === 'translucent') { + + painter.setCustomLayerDefaults(); + + context.setColorMode(painter.colorModeForRenderPass()); + context.setStencilMode(StencilMode.disabled); + + const depthMode = implementation.renderingMode === '3d' ? + new DepthMode(painter.context.gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D) : + painter.depthModeForSublayer(0, DepthMode.ReadOnly); + + context.setDepthMode(depthMode); + + implementation.render(context.gl, painter.transform.customLayerMatrix()); + + context.setDirty(); + painter.setBaseState(); + context.bindFramebuffer.set(null); + } +} diff --git a/web/libraries/maplibre-gl/src/render/draw_debug.test.ts b/web/libraries/maplibre-gl/src/render/draw_debug.test.ts new file mode 100644 index 00000000..ec19eff3 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_debug.test.ts @@ -0,0 +1,122 @@ +import {SourceCache} from '../source/source_cache'; +import {RasterSourceSpecification, SourceSpecification, VectorSourceSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {Style} from '../style/style'; +import {FillStyleLayer} from '../style/style_layer/fill_style_layer'; +import {RasterStyleLayer} from '../style/style_layer/raster_style_layer'; +import {selectDebugSource} from './draw_debug'; + +jest.mock('../style/style'); + +const zoom = 14; + +const defaultSources: { [_: string]: SourceSpecification } = { + 'raster_tiles': { + type: 'raster', + maxzoom: 19, + }, + 'vector_tiles': { + type: 'vector', + maxzoom: 14, + } +}; + +const buildMockStyle = (layers, sources = defaultSources) => { + const style = new Style(null); + style.sourceCaches = Object.fromEntries( + Object.entries(sources).map( + ([id, spec]) => [id, {id, getSource: () => spec} as SourceCache])); + style._layers = layers; + return style; +}; + +describe('selectDebugSource', () => { + test('Decides on vector source if it exists', () => { + const layers = { + '1': new RasterStyleLayer( + {id: '1', type: 'raster', source: 'raster_tiles'}), + '2': new FillStyleLayer( + {id: '2', type: 'fill', source: 'vector_tiles'}), + }; + const mockStyle = buildMockStyle(layers); + const source = selectDebugSource(mockStyle, zoom); + expect(source).toHaveProperty('id', 'vector_tiles'); + }); + + test('Decides raster if vector source not shown at this zoom', () => { + const layers = { + '1': new RasterStyleLayer( + {id: '1', type: 'raster', source: 'raster_tiles'}), + '2': new FillStyleLayer( + {id: '2', type: 'fill', source: 'vector_tiles', maxzoom: 13}), + }; + const mockStyle = buildMockStyle(layers); + const source = selectDebugSource(mockStyle, zoom); + expect(source).toHaveProperty('id', 'raster_tiles'); + }); + + test('Decides raster if vector layer has visibility none', () => { + const layers = { + '1': new RasterStyleLayer( + {id: '1', type: 'raster', source: 'raster_tiles'}), + '2': new FillStyleLayer( + {id: '2', type: 'fill', source: 'vector_tiles', layout: {visibility: 'none'}}), + }; + const style = buildMockStyle(layers); + const source = selectDebugSource(style, zoom); + expect(source).toHaveProperty('id', 'raster_tiles'); + }); + + test('Decides raster if no vector source exists', () => { + const layers = { + '1': new RasterStyleLayer( + {id: '1', type: 'raster', source: 'raster_tiles'}), + }; + const mockStyle = buildMockStyle(layers); + const source = selectDebugSource(mockStyle, zoom); + expect(source).toHaveProperty('id', 'raster_tiles'); + }); + + test('Decides on vector source with highest zoom level', () => { + const sources: { [_: string]: VectorSourceSpecification } = { + 'vector_11': { + type: 'vector', + maxzoom: 11, + }, + 'vector_14': { + type: 'vector', + maxzoom: 14, + } + }; + const layers = { + 'fill_11': new FillStyleLayer( + {id: 'fill_11', type: 'fill', source: 'vector_11'}), + 'fill_14': new FillStyleLayer( + {id: 'fill_14', type: 'fill', source: 'vector_14'}), + }; + const mockStyle = buildMockStyle(layers, sources); + const source = selectDebugSource(mockStyle, zoom); + expect(source).toHaveProperty('id', 'vector_14'); + }); + + test('Decides on raster source with highest zoom level', () => { + const sources: { [_: string]: RasterSourceSpecification } = { + 'raster_11': { + type: 'raster', + maxzoom: 11, + }, + 'raster_14': { + type: 'raster', + maxzoom: 14, + } + }; + const layers = { + 'raster_11': new RasterStyleLayer( + {id: 'raster_11', type: 'raster', source: 'raster_11'}), + 'raster_14': new RasterStyleLayer( + {id: 'raster_14', type: 'raster', source: 'raster_14'}), + }; + const mockStyle = buildMockStyle(layers, sources); + const source = selectDebugSource(mockStyle, zoom); + expect(source).toHaveProperty('id', 'raster_14'); + }); +}); diff --git a/web/libraries/maplibre-gl/src/render/draw_debug.ts b/web/libraries/maplibre-gl/src/render/draw_debug.ts new file mode 100644 index 00000000..5b72587b --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_debug.ts @@ -0,0 +1,148 @@ +import {DepthMode} from '../gl/depth_mode'; +import {StencilMode} from '../gl/stencil_mode'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import {debugUniformValues} from './program/debug_program'; +import {Color} from '@maplibre/maplibre-gl-style-spec'; +import {ColorMode} from '../gl/color_mode'; + +import type {Painter} from './painter'; +import type {SourceCache} from '../source/source_cache'; +import type {OverscaledTileID} from '../source/tile_id'; +import {Style} from '../style/style'; + +const topColor = new Color(1, 0, 0, 1); +const btmColor = new Color(0, 1, 0, 1); +const leftColor = new Color(0, 0, 1, 1); +const rightColor = new Color(1, 0, 1, 1); +const centerColor = new Color(0, 1, 1, 1); + +export function drawDebugPadding(painter: Painter) { + const padding = painter.transform.padding; + const lineWidth = 3; + // Top + drawHorizontalLine(painter, painter.transform.height - (padding.top || 0), lineWidth, topColor); + // Bottom + drawHorizontalLine(painter, padding.bottom || 0, lineWidth, btmColor); + // Left + drawVerticalLine(painter, padding.left || 0, lineWidth, leftColor); + // Right + drawVerticalLine(painter, painter.transform.width - (padding.right || 0), lineWidth, rightColor); + // Center + const center = painter.transform.centerPoint; + drawCrosshair(painter, center.x, painter.transform.height - center.y, centerColor); +} + +function drawCrosshair(painter: Painter, x: number, y: number, color: Color) { + const size = 20; + const lineWidth = 2; + //Vertical line + drawDebugSSRect(painter, x - lineWidth / 2, y - size / 2, lineWidth, size, color); + //Horizontal line + drawDebugSSRect(painter, x - size / 2, y - lineWidth / 2, size, lineWidth, color); +} + +function drawHorizontalLine(painter: Painter, y: number, lineWidth: number, color: Color) { + drawDebugSSRect(painter, 0, y + lineWidth / 2, painter.transform.width, lineWidth, color); +} + +function drawVerticalLine(painter: Painter, x: number, lineWidth: number, color: Color) { + drawDebugSSRect(painter, x - lineWidth / 2, 0, lineWidth, painter.transform.height, color); +} + +function drawDebugSSRect(painter: Painter, x: number, y: number, width: number, height: number, color: Color) { + const context = painter.context; + const gl = context.gl; + + gl.enable(gl.SCISSOR_TEST); + gl.scissor(x * painter.pixelRatio, y * painter.pixelRatio, width * painter.pixelRatio, height * painter.pixelRatio); + context.clear({color}); + gl.disable(gl.SCISSOR_TEST); +} + +export function drawDebug(painter: Painter, sourceCache: SourceCache, coords: Array) { + for (let i = 0; i < coords.length; i++) { + drawDebugTile(painter, sourceCache, coords[i]); + } +} + +function drawDebugTile(painter: Painter, sourceCache: SourceCache, coord: OverscaledTileID) { + const context = painter.context; + const gl = context.gl; + + const posMatrix = coord.posMatrix; + const program = painter.useProgram('debug'); + + const depthMode = DepthMode.disabled; + const stencilMode = StencilMode.disabled; + const colorMode = painter.colorModeForRenderPass(); + const id = '$debug'; + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + + context.activeTexture.set(gl.TEXTURE0); + + const tileRawData = sourceCache.getTileByID(coord.key).latestRawTileData; + const tileByteLength = (tileRawData && tileRawData.byteLength) || 0; + const tileSizeKb = Math.floor(tileByteLength / 1024); + const tileSize = sourceCache.getTile(coord).tileSize; + const scaleRatio = (512 / Math.min(tileSize, 512) * (coord.overscaledZ / painter.transform.zoom)) * 0.5; + let tileIdText = coord.canonical.toString(); + if (coord.overscaledZ !== coord.canonical.z) { + tileIdText += ` => ${coord.overscaledZ}`; + } + const tileLabel = `${tileIdText} ${tileSizeKb}kB`; + drawTextToOverlay(painter, tileLabel); + + program.draw(context, gl.TRIANGLES, depthMode, stencilMode, ColorMode.alphaBlended, CullFaceMode.disabled, + debugUniformValues(posMatrix, Color.transparent, scaleRatio), null, id, + painter.debugBuffer, painter.quadTriangleIndexBuffer, painter.debugSegments); + program.draw(context, gl.LINE_STRIP, depthMode, stencilMode, colorMode, CullFaceMode.disabled, + debugUniformValues(posMatrix, Color.red), terrainData, id, + painter.debugBuffer, painter.tileBorderIndexBuffer, painter.debugSegments); +} + +function drawTextToOverlay(painter: Painter, text: string) { + painter.initDebugOverlayCanvas(); + const canvas = painter.debugOverlayCanvas; + const gl = painter.context.gl; + const ctx2d = painter.debugOverlayCanvas.getContext('2d'); + ctx2d.clearRect(0, 0, canvas.width, canvas.height); + + ctx2d.shadowColor = 'white'; + ctx2d.shadowBlur = 2; + ctx2d.lineWidth = 1.5; + ctx2d.strokeStyle = 'white'; + ctx2d.textBaseline = 'top'; + ctx2d.font = `bold ${36}px Open Sans, sans-serif`; + ctx2d.fillText(text, 5, 5); + ctx2d.strokeText(text, 5, 5); + + painter.debugOverlayTexture.update(canvas); + painter.debugOverlayTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); +} + +export function selectDebugSource(style: Style, zoom: number): SourceCache | null { + // Use vector source with highest maxzoom + // Else use source with highest maxzoom of any type + let selectedSource: SourceCache = null; + const layers = Object.values(style._layers); + const sources = layers.flatMap((layer) => { + if (layer.source && !layer.isHidden(zoom)) { + const sourceCache = style.sourceCaches[layer.source]; + return [sourceCache]; + } else { + return []; + } + }); + const vectorSources = sources.filter((source) => source.getSource().type === 'vector'); + const otherSources = sources.filter((source) => source.getSource().type !== 'vector'); + const considerSource = (source: SourceCache) => { + if (!selectedSource || (selectedSource.getSource().maxzoom < source.getSource().maxzoom)) { + selectedSource = source; + } + }; + vectorSources.forEach((source) => considerSource(source)); + if (!selectedSource) { + otherSources.forEach((source) => considerSource(source)); + } + return selectedSource; +} diff --git a/web/libraries/maplibre-gl/src/render/draw_fill.test.ts b/web/libraries/maplibre-gl/src/render/draw_fill.test.ts new file mode 100644 index 00000000..e0a02762 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_fill.test.ts @@ -0,0 +1,137 @@ +import {mat4} from 'gl-matrix'; +import {OverscaledTileID} from '../source/tile_id'; +import {SourceCache} from '../source/source_cache'; +import {Tile} from '../source/tile'; +import {Painter} from './painter'; +import {Program} from './program'; +import type {ZoomHistory} from '../style/zoom_history'; +import type {Map} from '../ui/map'; +import {Transform} from '../geo/transform'; +import type {EvaluationParameters} from '../style/evaluation_parameters'; +import type {FillLayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {Style} from '../style/style'; +import {FillStyleLayer} from '../style/style_layer/fill_style_layer'; +import {drawFill} from './draw_fill'; +import {FillBucket} from '../data/bucket/fill_bucket'; +import {ProgramConfiguration, ProgramConfigurationSet} from '../data/program_configuration'; + +jest.mock('./painter'); +jest.mock('./program'); +jest.mock('../source/source_cache'); +jest.mock('../source/tile'); +jest.mock('../data/bucket/symbol_bucket'); +jest.mock('../symbol/projection'); + +describe('drawFill', () => { + test('should call programConfiguration.setConstantPatternPositions for transitioning fill-pattern', () => { + + const painterMock: Painter = constructMockPainer(); + const layer: FillStyleLayer = constructMockLayer(); + + const programMock = new Program(null as any, null as any, null as any, null as any, null as any, null as any); + (painterMock.useProgram as jest.Mock).mockReturnValue(programMock); + + const mockTile = constructMockTile(layer); + + const sourceCacheMock = new SourceCache(null as any, null as any, null as any); + (sourceCacheMock.getTile as jest.Mock).mockReturnValue(mockTile); + sourceCacheMock.map = {showCollisionBoxes: false} as any as Map; + + drawFill(painterMock, sourceCacheMock, layer, [mockTile.tileID]); + + // twice: first for fill, second for stroke + expect(programMock.draw).toHaveBeenCalledTimes(2); + + const bucket: FillBucket = (mockTile.getBucket(layer) as any); + const programConfiguration = bucket.programConfigurations.get(layer.id); + + expect(programConfiguration.setConstantPatternPositions).toHaveBeenCalled(); + }); + + function constructMockLayer(): FillStyleLayer { + const layerSpec = { + id: 'mock-layer', + source: 'empty-source', + type: 'fill', + layout: {}, + 'paint': { + 'fill-pattern': 'pattern0' + } + } as FillLayerSpecification; + const layer = new FillStyleLayer(layerSpec); + layer.getCrossfadeParameters = () => ({} as any); + layer.recalculate({zoom: 0, zoomHistory: {} as ZoomHistory} as EvaluationParameters, []); + + // Important: this setup is on purpose -- to NOT match layerspec + // 'fill-pattern': 'pattern0' + // so tile.imageAtlas.patternPositions['pattern0'] would return nothing + // mimicing the transitiong fill-pattern value + layer.getPaintProperty = () => { + return 'pattern1'; + }; + + return layer; + } + + function constructMockPainer(): Painter { + const painterMock = new Painter(null as any, null as any); + painterMock.context = { + gl: {}, + activeTexture: { + set: () => {} + } + } as any; + painterMock.renderPass = 'translucent'; + painterMock.transform = {pitch: 0, labelPlaneMatrix: mat4.create()} as any as Transform; + painterMock.options = {} as any; + painterMock.style = { + map: {} + } as any as Style; + + return painterMock; + } + + function constructMockTile(layer: FillStyleLayer): Tile { + const tileId = new OverscaledTileID(1, 0, 1, 0, 0); + tileId.posMatrix = mat4.create(); + + const tile = new Tile(tileId, 256); + tile.tileID = tileId; + + // Important: this setup is on purpose -- to NOT match layerspec + // 'fill-pattern': 'pattern0' + // so tile.imageAtlas.patternPositions['pattern0'] would return nothing + // mimicing the transitiong fill-pattern value + tile.imageAtlas = { + patternPositions: { + 'pattern1': {} + } + } as any; + tile.imageAtlasTexture = { + bind: () => {} + } as any; + + const bucketMock = constructMockBucket(layer); + + (tile.getBucket as jest.Mock).mockReturnValue(bucketMock); + (tile.patternsLoaded as jest.Mock).mockReturnValue(true); + return tile; + } + + function constructMockBucket(layer: FillStyleLayer) { + const bucketMock = new FillBucket({ + layers: [layer] + } as any); + + const mockProgramConfigurations: ProgramConfigurationSet = {} as any; + const mockProgramConfiguration: ProgramConfiguration = {} as any; + mockProgramConfiguration.updatePaintBuffers = () => {}; + mockProgramConfiguration.setConstantPatternPositions = jest.fn(); + + mockProgramConfigurations.get = () => mockProgramConfiguration; + + bucketMock.programConfigurations = mockProgramConfigurations; + + return bucketMock; + } +}); diff --git a/web/libraries/maplibre-gl/src/render/draw_fill.ts b/web/libraries/maplibre-gl/src/render/draw_fill.ts new file mode 100644 index 00000000..c850dc98 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_fill.ts @@ -0,0 +1,128 @@ +import {Color} from '@maplibre/maplibre-gl-style-spec'; +import {DepthMode} from '../gl/depth_mode'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import {ColorMode} from '../gl/color_mode'; +import { + fillUniformValues, + fillPatternUniformValues, + fillOutlineUniformValues, + fillOutlinePatternUniformValues +} from './program/fill_program'; + +import type {Painter} from './painter'; +import type {SourceCache} from '../source/source_cache'; +import type {FillStyleLayer} from '../style/style_layer/fill_style_layer'; +import type {FillBucket} from '../data/bucket/fill_bucket'; +import type {OverscaledTileID} from '../source/tile_id'; +import {updatePatternPositionsInProgram} from './update_pattern_positions_in_program'; + +export function drawFill(painter: Painter, sourceCache: SourceCache, layer: FillStyleLayer, coords: Array) { + const color = layer.paint.get('fill-color'); + const opacity = layer.paint.get('fill-opacity'); + + if (opacity.constantOr(1) === 0) { + return; + } + + const colorMode = painter.colorModeForRenderPass(); + + const pattern = layer.paint.get('fill-pattern'); + const pass = painter.opaquePassEnabledForLayer() && + (!pattern.constantOr(1 as any) && + color.constantOr(Color.transparent).a === 1 && + opacity.constantOr(0) === 1) ? 'opaque' : 'translucent'; + + // Draw fill + if (painter.renderPass === pass) { + const depthMode = painter.depthModeForSublayer( + 1, painter.renderPass === 'opaque' ? DepthMode.ReadWrite : DepthMode.ReadOnly); + drawFillTiles(painter, sourceCache, layer, coords, depthMode, colorMode, false); + } + + // Draw stroke + if (painter.renderPass === 'translucent' && layer.paint.get('fill-antialias')) { + + // If we defined a different color for the fill outline, we are + // going to ignore the bits in 0x07 and just care about the global + // clipping mask. + // Otherwise, we only want to drawFill the antialiased parts that are + // *outside* the current shape. This is important in case the fill + // or stroke color is translucent. If we wouldn't clip to outside + // the current shape, some pixels from the outline stroke overlapped + // the (non-antialiased) fill. + const depthMode = painter.depthModeForSublayer( + layer.getPaintProperty('fill-outline-color') ? 2 : 0, DepthMode.ReadOnly); + drawFillTiles(painter, sourceCache, layer, coords, depthMode, colorMode, true); + } +} + +function drawFillTiles( + painter: Painter, + sourceCache: SourceCache, + layer: FillStyleLayer, + coords: Array, + depthMode: Readonly, + colorMode: Readonly, + isOutline: boolean) { + const gl = painter.context.gl; + const fillPropertyName = 'fill-pattern'; + const patternProperty = layer.paint.get(fillPropertyName); + const image = patternProperty && patternProperty.constantOr(1 as any); + const crossfade = layer.getCrossfadeParameters(); + let drawMode, programName, uniformValues, indexBuffer, segments; + + if (!isOutline) { + programName = image ? 'fillPattern' : 'fill'; + drawMode = gl.TRIANGLES; + } else { + programName = image && !layer.getPaintProperty('fill-outline-color') ? 'fillOutlinePattern' : 'fillOutline'; + drawMode = gl.LINES; + } + + const constantPattern = patternProperty.constantOr(null); + + for (const coord of coords) { + const tile = sourceCache.getTile(coord); + if (image && !tile.patternsLoaded()) continue; + + const bucket: FillBucket = (tile.getBucket(layer) as any); + if (!bucket) continue; + + const programConfiguration = bucket.programConfigurations.get(layer.id); + const program = painter.useProgram(programName, programConfiguration); + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + + if (image) { + painter.context.activeTexture.set(gl.TEXTURE0); + tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + programConfiguration.updatePaintBuffers(crossfade); + } + + updatePatternPositionsInProgram(programConfiguration, fillPropertyName, constantPattern, tile, layer); + + const terrainCoord = terrainData ? coord : null; + const posMatrix = terrainCoord ? terrainCoord.posMatrix : coord.posMatrix; + const tileMatrix = painter.translatePosMatrix(posMatrix, tile, + layer.paint.get('fill-translate'), layer.paint.get('fill-translate-anchor')); + + if (!isOutline) { + indexBuffer = bucket.indexBuffer; + segments = bucket.segments; + uniformValues = image ? + fillPatternUniformValues(tileMatrix, painter, crossfade, tile) : + fillUniformValues(tileMatrix); + } else { + indexBuffer = bucket.indexBuffer2; + segments = bucket.segments2; + const drawingBufferSize = [gl.drawingBufferWidth, gl.drawingBufferHeight] as [number, number]; + uniformValues = (programName === 'fillOutlinePattern' && image) ? + fillOutlinePatternUniformValues(tileMatrix, painter, crossfade, tile, drawingBufferSize) : + fillOutlineUniformValues(tileMatrix, drawingBufferSize); + } + + program.draw(painter.context, drawMode, depthMode, + painter.stencilModeForClipping(coord), colorMode, CullFaceMode.disabled, uniformValues, terrainData, + layer.id, bucket.layoutVertexBuffer, indexBuffer, segments, + layer.paint, painter.transform.zoom, programConfiguration); + } +} diff --git a/web/libraries/maplibre-gl/src/render/draw_fill_extrusion.ts b/web/libraries/maplibre-gl/src/render/draw_fill_extrusion.ts new file mode 100644 index 00000000..6ce366dd --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_fill_extrusion.ts @@ -0,0 +1,97 @@ +import {DepthMode} from '../gl/depth_mode'; +import {StencilMode} from '../gl/stencil_mode'; +import {ColorMode} from '../gl/color_mode'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import { + fillExtrusionUniformValues, + fillExtrusionPatternUniformValues, +} from './program/fill_extrusion_program'; + +import type {Painter} from './painter'; +import type {SourceCache} from '../source/source_cache'; +import type {FillExtrusionStyleLayer} from '../style/style_layer/fill_extrusion_style_layer'; +import type {FillExtrusionBucket} from '../data/bucket/fill_extrusion_bucket'; +import type {OverscaledTileID} from '../source/tile_id'; + +import {updatePatternPositionsInProgram} from './update_pattern_positions_in_program'; + +export function drawFillExtrusion(painter: Painter, source: SourceCache, layer: FillExtrusionStyleLayer, coords: Array) { + const opacity = layer.paint.get('fill-extrusion-opacity'); + if (opacity === 0) { + return; + } + + if (painter.renderPass === 'translucent') { + const depthMode = new DepthMode(painter.context.gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D); + + if (opacity === 1 && !layer.paint.get('fill-extrusion-pattern').constantOr(1 as any)) { + const colorMode = painter.colorModeForRenderPass(); + drawExtrusionTiles(painter, source, layer, coords, depthMode, StencilMode.disabled, colorMode); + + } else { + // Draw transparent buildings in two passes so that only the closest surface is drawn. + // First draw all the extrusions into only the depth buffer. No colors are drawn. + drawExtrusionTiles(painter, source, layer, coords, depthMode, + StencilMode.disabled, + ColorMode.disabled); + + // Then draw all the extrusions a second type, only coloring fragments if they have the + // same depth value as the closest fragment in the previous pass. Use the stencil buffer + // to prevent the second draw in cases where we have coincident polygons. + drawExtrusionTiles(painter, source, layer, coords, depthMode, + painter.stencilModeFor3D(), + painter.colorModeForRenderPass()); + } + } +} + +function drawExtrusionTiles( + painter: Painter, + source: SourceCache, + layer: FillExtrusionStyleLayer, + coords: OverscaledTileID[], + depthMode: DepthMode, + stencilMode: Readonly, + colorMode: Readonly) { + const context = painter.context; + const gl = context.gl; + const fillPropertyName = 'fill-extrusion-pattern'; + const patternProperty = layer.paint.get(fillPropertyName); + const image = patternProperty.constantOr(1 as any); + const crossfade = layer.getCrossfadeParameters(); + const opacity = layer.paint.get('fill-extrusion-opacity'); + const constantPattern = patternProperty.constantOr(null); + for (const coord of coords) { + const tile = source.getTile(coord); + const bucket: FillExtrusionBucket = (tile.getBucket(layer) as any); + if (!bucket) continue; + + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + const programConfiguration = bucket.programConfigurations.get(layer.id); + const program = painter.useProgram(image ? 'fillExtrusionPattern' : 'fillExtrusion', programConfiguration); + + if (image) { + painter.context.activeTexture.set(gl.TEXTURE0); + tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + programConfiguration.updatePaintBuffers(crossfade); + } + + updatePatternPositionsInProgram(programConfiguration, fillPropertyName, constantPattern, tile, layer); + + const matrix = painter.translatePosMatrix( + coord.posMatrix, + tile, + layer.paint.get('fill-extrusion-translate'), + layer.paint.get('fill-extrusion-translate-anchor')); + + const shouldUseVerticalGradient = layer.paint.get('fill-extrusion-vertical-gradient'); + const uniformValues = image ? + fillExtrusionPatternUniformValues(matrix, painter, shouldUseVerticalGradient, opacity, coord, crossfade, tile) : + fillExtrusionUniformValues(matrix, painter, shouldUseVerticalGradient, opacity); + + program.draw(context, context.gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.backCCW, + uniformValues, terrainData, layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer, + bucket.segments, layer.paint, painter.transform.zoom, + programConfiguration, painter.style.map.terrain && bucket.centroidVertexBuffer); + } +} diff --git a/web/libraries/maplibre-gl/src/render/draw_heatmap.ts b/web/libraries/maplibre-gl/src/render/draw_heatmap.ts new file mode 100644 index 00000000..bf2e6a68 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_heatmap.ts @@ -0,0 +1,133 @@ +import {Texture} from './texture'; +import {Color} from '@maplibre/maplibre-gl-style-spec'; +import {DepthMode} from '../gl/depth_mode'; +import {StencilMode} from '../gl/stencil_mode'; +import {ColorMode} from '../gl/color_mode'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import {Context} from '../gl/context'; +import {Framebuffer} from '../gl/framebuffer'; +import { + heatmapUniformValues, + heatmapTextureUniformValues +} from './program/heatmap_program'; + +import type {Painter} from './painter'; +import type {SourceCache} from '../source/source_cache'; +import type {HeatmapStyleLayer} from '../style/style_layer/heatmap_style_layer'; +import type {HeatmapBucket} from '../data/bucket/heatmap_bucket'; +import type {OverscaledTileID} from '../source/tile_id'; + +export function drawHeatmap(painter: Painter, sourceCache: SourceCache, layer: HeatmapStyleLayer, coords: Array) { + if (layer.paint.get('heatmap-opacity') === 0) { + return; + } + + if (painter.renderPass === 'offscreen') { + const context = painter.context; + const gl = context.gl; + + // Allow kernels to be drawn across boundaries, so that + // large kernels are not clipped to tiles + const stencilMode = StencilMode.disabled; + // Turn on additive blending for kernels, which is a key aspect of kernel density estimation formula + const colorMode = new ColorMode([gl.ONE, gl.ONE], Color.transparent, [true, true, true, true]); + + bindFramebuffer(context, painter, layer); + + context.clear({color: Color.transparent}); + + for (let i = 0; i < coords.length; i++) { + const coord = coords[i]; + + // Skip tiles that have uncovered parents to avoid flickering; we don't need + // to use complex tile masking here because the change between zoom levels is subtle, + // so it's fine to simply render the parent until all its 4 children are loaded + if (sourceCache.hasRenderableParent(coord)) continue; + + const tile = sourceCache.getTile(coord); + const bucket: HeatmapBucket = (tile.getBucket(layer) as any); + if (!bucket) continue; + + const programConfiguration = bucket.programConfigurations.get(layer.id); + const program = painter.useProgram('heatmap', programConfiguration); + const {zoom} = painter.transform; + + program.draw(context, gl.TRIANGLES, DepthMode.disabled, stencilMode, colorMode, CullFaceMode.disabled, + heatmapUniformValues(coord.posMatrix, tile, zoom, layer.paint.get('heatmap-intensity')), null, + layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer, + bucket.segments, layer.paint, painter.transform.zoom, + programConfiguration); + } + + context.viewport.set([0, 0, painter.width, painter.height]); + + } else if (painter.renderPass === 'translucent') { + painter.context.setColorMode(painter.colorModeForRenderPass()); + renderTextureToMap(painter, layer); + } +} + +function bindFramebuffer(context: Context, painter: Painter, layer: HeatmapStyleLayer) { + const gl = context.gl; + context.activeTexture.set(gl.TEXTURE1); + + // Use a 4x downscaled screen texture for better performance + context.viewport.set([0, 0, painter.width / 4, painter.height / 4]); + + let fbo = layer.heatmapFbo; + + if (!fbo) { + const texture = gl.createTexture(); + gl.bindTexture(gl.TEXTURE_2D, texture); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR); + + fbo = layer.heatmapFbo = context.createFramebuffer(painter.width / 4, painter.height / 4, false, false); + + bindTextureToFramebuffer(context, painter, texture, fbo); + + } else { + gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get()); + context.bindFramebuffer.set(fbo.framebuffer); + } +} + +function bindTextureToFramebuffer(context: Context, painter: Painter, texture: WebGLTexture, fbo: Framebuffer) { + const gl = context.gl; + // Use the higher precision half-float texture where available (producing much smoother looking heatmaps); + // Otherwise, fall back to a low precision texture + + const numType = context.HALF_FLOAT ?? gl.UNSIGNED_BYTE; + const internalFormat = context.RGBA16F ?? gl.RGBA; + + gl.texImage2D(gl.TEXTURE_2D, 0, internalFormat, painter.width / 4, painter.height / 4, 0, gl.RGBA, numType, null); + fbo.colorAttachment.set(texture); +} + +function renderTextureToMap(painter: Painter, layer: HeatmapStyleLayer) { + const context = painter.context; + const gl = context.gl; + + // Here we bind two different textures from which we'll sample in drawing + // heatmaps: the kernel texture, prepared in the offscreen pass, and a + // color ramp texture. + const fbo = layer.heatmapFbo; + if (!fbo) return; + context.activeTexture.set(gl.TEXTURE0); + gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get()); + + context.activeTexture.set(gl.TEXTURE1); + let colorRampTexture = layer.colorRampTexture; + if (!colorRampTexture) { + colorRampTexture = layer.colorRampTexture = new Texture(context, layer.colorRamp, gl.RGBA); + } + colorRampTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + + painter.useProgram('heatmapTexture').draw(context, gl.TRIANGLES, + DepthMode.disabled, StencilMode.disabled, painter.colorModeForRenderPass(), CullFaceMode.disabled, + heatmapTextureUniformValues(painter, layer, 0, 1), null, + layer.id, painter.viewportBuffer, painter.quadTriangleIndexBuffer, + painter.viewportSegments, layer.paint, painter.transform.zoom); +} diff --git a/web/libraries/maplibre-gl/src/render/draw_hillshade.ts b/web/libraries/maplibre-gl/src/render/draw_hillshade.ts new file mode 100644 index 00000000..39f46412 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_hillshade.ts @@ -0,0 +1,119 @@ +import {Texture} from './texture'; +import {StencilMode} from '../gl/stencil_mode'; +import {DepthMode} from '../gl/depth_mode'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import {ColorMode} from '../gl/color_mode'; +import {Tile} from '../source/tile'; +import { + hillshadeUniformValues, + hillshadeUniformPrepareValues +} from './program/hillshade_program'; + +import type {Painter} from './painter'; +import type {SourceCache} from '../source/source_cache'; +import type {HillshadeStyleLayer} from '../style/style_layer/hillshade_style_layer'; +import type {OverscaledTileID} from '../source/tile_id'; + +export function drawHillshade(painter: Painter, sourceCache: SourceCache, layer: HillshadeStyleLayer, tileIDs: Array) { + if (painter.renderPass !== 'offscreen' && painter.renderPass !== 'translucent') return; + + const context = painter.context; + + const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly); + const colorMode = painter.colorModeForRenderPass(); + + const [stencilModes, coords] = painter.renderPass === 'translucent' ? + painter.stencilConfigForOverlap(tileIDs) : [{}, tileIDs]; + + for (const coord of coords) { + const tile = sourceCache.getTile(coord); + if (typeof tile.needsHillshadePrepare !== 'undefined' && tile.needsHillshadePrepare && painter.renderPass === 'offscreen') { + prepareHillshade(painter, tile, layer, depthMode, StencilMode.disabled, colorMode); + } else if (painter.renderPass === 'translucent') { + renderHillshade(painter, coord, tile, layer, depthMode, stencilModes[coord.overscaledZ], colorMode); + } + } + + context.viewport.set([0, 0, painter.width, painter.height]); +} + +function renderHillshade( + painter: Painter, + coord: OverscaledTileID, + tile: Tile, + layer: HillshadeStyleLayer, + depthMode: Readonly, + stencilMode: Readonly, + colorMode: Readonly) { + const context = painter.context; + const gl = context.gl; + const fbo = tile.fbo; + if (!fbo) return; + + const program = painter.useProgram('hillshade'); + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + + context.activeTexture.set(gl.TEXTURE0); + gl.bindTexture(gl.TEXTURE_2D, fbo.colorAttachment.get()); + + const terrainCoord = terrainData ? coord : null; + program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled, + hillshadeUniformValues(painter, tile, layer, terrainCoord), terrainData, layer.id, painter.rasterBoundsBuffer, + painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments); + +} + +// hillshade rendering is done in two steps. the prepare step first calculates the slope of the terrain in the x and y +// directions for each pixel, and saves those values to a framebuffer texture in the r and g channels. +function prepareHillshade( + painter: Painter, + tile: Tile, + layer: HillshadeStyleLayer, + depthMode: Readonly, + stencilMode: Readonly, + colorMode: Readonly) { + const context = painter.context; + const gl = context.gl; + const dem = tile.dem; + if (dem && dem.data) { + const tileSize = dem.dim; + const textureStride = dem.stride; + + const pixelData = dem.getPixels(); + context.activeTexture.set(gl.TEXTURE1); + + context.pixelStoreUnpackPremultiplyAlpha.set(false); + tile.demTexture = tile.demTexture || painter.getTileTexture(textureStride); + if (tile.demTexture) { + const demTexture = tile.demTexture; + demTexture.update(pixelData, {premultiply: false}); + demTexture.bind(gl.NEAREST, gl.CLAMP_TO_EDGE); + } else { + tile.demTexture = new Texture(context, pixelData, gl.RGBA, {premultiply: false}); + tile.demTexture.bind(gl.NEAREST, gl.CLAMP_TO_EDGE); + } + + context.activeTexture.set(gl.TEXTURE0); + + let fbo = tile.fbo; + + if (!fbo) { + const renderTexture = new Texture(context, {width: tileSize, height: tileSize, data: null}, gl.RGBA); + renderTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + + fbo = tile.fbo = context.createFramebuffer(tileSize, tileSize, true, false); + fbo.colorAttachment.set(renderTexture.texture); + } + + context.bindFramebuffer.set(fbo.framebuffer); + context.viewport.set([0, 0, tileSize, tileSize]); + + painter.useProgram('hillshadePrepare').draw(context, gl.TRIANGLES, + depthMode, stencilMode, colorMode, CullFaceMode.disabled, + hillshadeUniformPrepareValues(tile.tileID, dem), + null, layer.id, painter.rasterBoundsBuffer, + painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments); + + tile.needsHillshadePrepare = false; + } +} diff --git a/web/libraries/maplibre-gl/src/render/draw_line.ts b/web/libraries/maplibre-gl/src/render/draw_line.ts new file mode 100644 index 00000000..28516064 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_line.ts @@ -0,0 +1,125 @@ +import {DepthMode} from '../gl/depth_mode'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import {Texture} from './texture'; +import { + lineUniformValues, + linePatternUniformValues, + lineSDFUniformValues, + lineGradientUniformValues +} from './program/line_program'; + +import type {Painter} from './painter'; +import type {SourceCache} from '../source/source_cache'; +import type {LineStyleLayer} from '../style/style_layer/line_style_layer'; +import type {LineBucket} from '../data/bucket/line_bucket'; +import type {OverscaledTileID} from '../source/tile_id'; +import {clamp, nextPowerOfTwo} from '../util/util'; +import {renderColorRamp} from '../util/color_ramp'; +import {EXTENT} from '../data/extent'; + +export function drawLine(painter: Painter, sourceCache: SourceCache, layer: LineStyleLayer, coords: Array) { + if (painter.renderPass !== 'translucent') return; + + const opacity = layer.paint.get('line-opacity'); + const width = layer.paint.get('line-width'); + if (opacity.constantOr(1) === 0 || width.constantOr(1) === 0) return; + + const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly); + const colorMode = painter.colorModeForRenderPass(); + + const dasharray = layer.paint.get('line-dasharray'); + const patternProperty = layer.paint.get('line-pattern'); + const image = patternProperty.constantOr(1 as any); + + const gradient = layer.paint.get('line-gradient'); + const crossfade = layer.getCrossfadeParameters(); + + const programId = + image ? 'linePattern' : + dasharray ? 'lineSDF' : + gradient ? 'lineGradient' : 'line'; + + const context = painter.context; + const gl = context.gl; + + let firstTile = true; + + for (const coord of coords) { + const tile = sourceCache.getTile(coord); + + if (image && !tile.patternsLoaded()) continue; + + const bucket: LineBucket = (tile.getBucket(layer) as any); + if (!bucket) continue; + + const programConfiguration = bucket.programConfigurations.get(layer.id); + const prevProgram = painter.context.program.get(); + const program = painter.useProgram(programId, programConfiguration); + const programChanged = firstTile || program.program !== prevProgram; + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + + const constantPattern = patternProperty.constantOr(null); + if (constantPattern && tile.imageAtlas) { + const atlas = tile.imageAtlas; + const posTo = atlas.patternPositions[constantPattern.to.toString()]; + const posFrom = atlas.patternPositions[constantPattern.from.toString()]; + if (posTo && posFrom) programConfiguration.setConstantPatternPositions(posTo, posFrom); + } + + const terrainCoord = terrainData ? coord : null; + const uniformValues = image ? linePatternUniformValues(painter, tile, layer, crossfade, terrainCoord) : + dasharray ? lineSDFUniformValues(painter, tile, layer, dasharray, crossfade, terrainCoord) : + gradient ? lineGradientUniformValues(painter, tile, layer, bucket.lineClipsArray.length, terrainCoord) : + lineUniformValues(painter, tile, layer, terrainCoord); + + if (image) { + context.activeTexture.set(gl.TEXTURE0); + tile.imageAtlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + programConfiguration.updatePaintBuffers(crossfade); + } else if (dasharray && (programChanged || painter.lineAtlas.dirty)) { + context.activeTexture.set(gl.TEXTURE0); + painter.lineAtlas.bind(context); + } else if (gradient) { + const layerGradient = bucket.gradients[layer.id]; + let gradientTexture = layerGradient.texture; + if (layer.gradientVersion !== layerGradient.version) { + let textureResolution = 256; + if (layer.stepInterpolant) { + const sourceMaxZoom = sourceCache.getSource().maxzoom; + const potentialOverzoom = coord.canonical.z === sourceMaxZoom ? + Math.ceil(1 << (painter.transform.maxZoom - coord.canonical.z)) : 1; + const lineLength = bucket.maxLineLength / EXTENT; + // Logical pixel tile size is 512px, and 1024px right before current zoom + 1 + const maxTilePixelSize = 1024; + // Maximum possible texture coverage heuristic, bound by hardware max texture size + const maxTextureCoverage = lineLength * maxTilePixelSize * potentialOverzoom; + textureResolution = clamp(nextPowerOfTwo(maxTextureCoverage), 256, context.maxTextureSize); + } + layerGradient.gradient = renderColorRamp({ + expression: layer.gradientExpression(), + evaluationKey: 'lineProgress', + resolution: textureResolution, + image: layerGradient.gradient || undefined, + clips: bucket.lineClipsArray + }); + if (layerGradient.texture) { + layerGradient.texture.update(layerGradient.gradient); + } else { + layerGradient.texture = new Texture(context, layerGradient.gradient, gl.RGBA); + } + layerGradient.version = layer.gradientVersion; + gradientTexture = layerGradient.texture; + } + context.activeTexture.set(gl.TEXTURE0); + gradientTexture.bind(layer.stepInterpolant ? gl.NEAREST : gl.LINEAR, gl.CLAMP_TO_EDGE); + } + + program.draw(context, gl.TRIANGLES, depthMode, + painter.stencilModeForClipping(coord), colorMode, CullFaceMode.disabled, uniformValues, terrainData, + layer.id, bucket.layoutVertexBuffer, bucket.indexBuffer, bucket.segments, + layer.paint, painter.transform.zoom, programConfiguration, bucket.layoutVertexBuffer2); + + firstTile = false; + // once refactored so that bound texture state is managed, we'll also be able to remove this firstTile/programChanged logic + } +} diff --git a/web/libraries/maplibre-gl/src/render/draw_raster.ts b/web/libraries/maplibre-gl/src/render/draw_raster.ts new file mode 100644 index 00000000..2bb28331 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_raster.ts @@ -0,0 +1,123 @@ +import {clamp} from '../util/util'; + +import {ImageSource} from '../source/image_source'; +import {browser} from '../util/browser'; +import {StencilMode} from '../gl/stencil_mode'; +import {DepthMode} from '../gl/depth_mode'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import {rasterUniformValues} from './program/raster_program'; + +import type {Painter} from './painter'; +import type {SourceCache} from '../source/source_cache'; +import type {RasterStyleLayer} from '../style/style_layer/raster_style_layer'; +import type {OverscaledTileID} from '../source/tile_id'; + +export function drawRaster(painter: Painter, sourceCache: SourceCache, layer: RasterStyleLayer, tileIDs: Array) { + if (painter.renderPass !== 'translucent') return; + if (layer.paint.get('raster-opacity') === 0) return; + if (!tileIDs.length) return; + + const context = painter.context; + const gl = context.gl; + const source = sourceCache.getSource(); + const program = painter.useProgram('raster'); + + const colorMode = painter.colorModeForRenderPass(); + + const [stencilModes, coords] = source instanceof ImageSource ? [{}, tileIDs] : + painter.stencilConfigForOverlap(tileIDs); + + const minTileZ = coords[coords.length - 1].overscaledZ; + + const align = !painter.options.moving; + for (const coord of coords) { + // Set the lower zoom level to sublayer 0, and higher zoom levels to higher sublayers + // Use gl.LESS to prevent double drawing in areas where tiles overlap. + const depthMode = painter.depthModeForSublayer(coord.overscaledZ - minTileZ, + layer.paint.get('raster-opacity') === 1 ? DepthMode.ReadWrite : DepthMode.ReadOnly, gl.LESS); + + const tile = sourceCache.getTile(coord); + + tile.registerFadeDuration(layer.paint.get('raster-fade-duration')); + + const parentTile = sourceCache.findLoadedParent(coord, 0), + fade = getFadeValues(tile, parentTile, sourceCache, layer, painter.transform, painter.style.map.terrain); + + let parentScaleBy, parentTL; + + const textureFilter = layer.paint.get('raster-resampling') === 'nearest' ? gl.NEAREST : gl.LINEAR; + + context.activeTexture.set(gl.TEXTURE0); + tile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST); + + context.activeTexture.set(gl.TEXTURE1); + + if (parentTile) { + parentTile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST); + parentScaleBy = Math.pow(2, parentTile.tileID.overscaledZ - tile.tileID.overscaledZ); + parentTL = [tile.tileID.canonical.x * parentScaleBy % 1, tile.tileID.canonical.y * parentScaleBy % 1]; + + } else { + tile.texture.bind(textureFilter, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST); + } + + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + const terrainCoord = terrainData ? coord : null; + const posMatrix = terrainCoord ? terrainCoord.posMatrix : painter.transform.calculatePosMatrix(coord.toUnwrapped(), align); + const uniformValues = rasterUniformValues(posMatrix, parentTL || [0, 0], parentScaleBy || 1, fade, layer); + + if (source instanceof ImageSource) { + program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.disabled, + uniformValues, terrainData, layer.id, source.boundsBuffer, + painter.quadTriangleIndexBuffer, source.boundsSegments); + } else { + program.draw(context, gl.TRIANGLES, depthMode, stencilModes[coord.overscaledZ], colorMode, CullFaceMode.disabled, + uniformValues, terrainData, layer.id, painter.rasterBoundsBuffer, + painter.quadTriangleIndexBuffer, painter.rasterBoundsSegments); + } + } +} + +function getFadeValues(tile, parentTile, sourceCache, layer, transform, terrain) { + const fadeDuration = layer.paint.get('raster-fade-duration'); + + if (!terrain && fadeDuration > 0) { + const now = browser.now(); + const sinceTile = (now - tile.timeAdded) / fadeDuration; + const sinceParent = parentTile ? (now - parentTile.timeAdded) / fadeDuration : -1; + + const source = sourceCache.getSource(); + const idealZ = transform.coveringZoomLevel({ + tileSize: source.tileSize, + roundZoom: source.roundZoom + }); + + // if no parent or parent is older, fade in; if parent is younger, fade out + const fadeIn = !parentTile || Math.abs(parentTile.tileID.overscaledZ - idealZ) > Math.abs(tile.tileID.overscaledZ - idealZ); + + const childOpacity = (fadeIn && tile.refreshedUponExpiration) ? 1 : clamp(fadeIn ? sinceTile : 1 - sinceParent, 0, 1); + + // we don't crossfade tiles that were just refreshed upon expiring: + // once they're old enough to pass the crossfading threshold + // (fadeDuration), unset the `refreshedUponExpiration` flag so we don't + // incorrectly fail to crossfade them when zooming + if (tile.refreshedUponExpiration && sinceTile >= 1) tile.refreshedUponExpiration = false; + + if (parentTile) { + return { + opacity: 1, + mix: 1 - childOpacity + }; + } else { + return { + opacity: childOpacity, + mix: 0 + }; + } + } else { + return { + opacity: 1, + mix: 0 + }; + } +} diff --git a/web/libraries/maplibre-gl/src/render/draw_symbol.test.ts b/web/libraries/maplibre-gl/src/render/draw_symbol.test.ts new file mode 100644 index 00000000..a7f85142 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_symbol.test.ts @@ -0,0 +1,223 @@ +import {mat4} from 'gl-matrix'; +import {OverscaledTileID} from '../source/tile_id'; +import {SymbolBucket} from '../data/bucket/symbol_bucket'; +import {SourceCache} from '../source/source_cache'; +import {Tile} from '../source/tile'; +import {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer'; +import {Painter} from './painter'; +import {Program} from './program'; +import {drawSymbols} from './draw_symbol'; +import * as symbolProjection from '../symbol/projection'; +import type {ZoomHistory} from '../style/zoom_history'; +import type {Map} from '../ui/map'; +import {Transform} from '../geo/transform'; +import type {EvaluationParameters} from '../style/evaluation_parameters'; +import type {SymbolLayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {Style} from '../style/style'; + +jest.mock('./painter'); +jest.mock('./program'); +jest.mock('../source/source_cache'); +jest.mock('../source/tile'); +jest.mock('../data/bucket/symbol_bucket'); +jest.mock('../symbol/projection'); + +describe('drawSymbol', () => { + test('should not do anything', () => { + const mockPainter = new Painter(null, null); + mockPainter.renderPass = 'opaque'; + + drawSymbols(mockPainter, null, null, null, null); + + expect(mockPainter.colorModeForRenderPass).not.toHaveBeenCalled(); + }); + + test('should call program.draw', () => { + + const painterMock = new Painter(null, null); + painterMock.context = { + gl: {}, + activeTexture: { + set: () => { } + } + } as any; + painterMock.renderPass = 'translucent'; + painterMock.transform = {pitch: 0, labelPlaneMatrix: mat4.create()} as any as Transform; + painterMock.options = {} as any; + painterMock.style = { + map: {} + } as any as Style; + + const layerSpec = { + id: 'mock-layer', + source: 'empty-source', + type: 'symbol', + layout: {}, + paint: { + 'text-opacity': 1 + } + } as SymbolLayerSpecification; + const layer = new SymbolStyleLayer(layerSpec); + layer.recalculate({zoom: 0, zoomHistory: {} as ZoomHistory} as EvaluationParameters, []); + + const tileId = new OverscaledTileID(1, 0, 1, 0, 0); + tileId.posMatrix = mat4.create(); + const programMock = new Program(null, null, null, null, null, null); + (painterMock.useProgram as jest.Mock).mockReturnValue(programMock); + const bucketMock = new SymbolBucket(null); + bucketMock.icon = { + programConfigurations: { + get: () => { } + }, + segments: { + get: () => [1] + }, + hasVisibleVertices: true + } as any; + bucketMock.iconSizeData = { + kind: 'constant', + layoutSize: 1 + }; + const tile = new Tile(tileId, 256); + tile.tileID = tileId; + tile.imageAtlasTexture = { + bind: () => { } + } as any; + (tile.getBucket as jest.Mock).mockReturnValue(bucketMock); + const sourceCacheMock = new SourceCache(null, null, null); + (sourceCacheMock.getTile as jest.Mock).mockReturnValue(tile); + sourceCacheMock.map = {showCollisionBoxes: false} as any as Map; + + drawSymbols(painterMock, sourceCacheMock, layer, [tileId], null); + + expect(programMock.draw).toHaveBeenCalledTimes(1); + }); + + test('should call updateLineLabels with rotateToLine === false if text-rotation-alignment is viewport-glyph', () => { + + const painterMock = new Painter(null, null); + painterMock.context = { + gl: {}, + activeTexture: { + set: () => { } + } + } as any; + painterMock.renderPass = 'translucent'; + painterMock.transform = {pitch: 0, labelPlaneMatrix: mat4.create()} as any as Transform; + painterMock.options = {} as any; + + const layerSpec = { + id: 'mock-layer', + source: 'empty-source', + type: 'symbol', + layout: { + 'text-rotation-alignment': 'viewport-glyph', + 'text-field': 'ABC', + 'symbol-placement': 'line', + }, + paint: { + 'text-opacity': 1 + } + } as SymbolLayerSpecification; + const layer = new SymbolStyleLayer(layerSpec); + layer.recalculate({zoom: 0, zoomHistory: {} as ZoomHistory} as EvaluationParameters, []); + + const tileId = new OverscaledTileID(1, 0, 1, 0, 0); + tileId.posMatrix = mat4.create(); + const programMock = new Program(null, null, null, null, null, null); + (painterMock.useProgram as jest.Mock).mockReturnValue(programMock); + const bucketMock = new SymbolBucket(null); + bucketMock.icon = { + programConfigurations: { + get: () => { } + }, + segments: { + get: () => [1] + }, + hasVisibleVertices: true + } as any; + bucketMock.iconSizeData = { + kind: 'constant', + layoutSize: 1 + }; + const tile = new Tile(tileId, 256); + tile.tileID = tileId; + tile.imageAtlasTexture = { + bind: () => { } + } as any; + (tile.getBucket as jest.Mock).mockReturnValue(bucketMock); + const sourceCacheMock = new SourceCache(null, null, null); + (sourceCacheMock.getTile as jest.Mock).mockReturnValue(tile); + sourceCacheMock.map = {showCollisionBoxes: false} as any as Map; + painterMock.style = { + map: {} + } as any as Style; + + const spy = jest.spyOn(symbolProjection, 'updateLineLabels'); + drawSymbols(painterMock, sourceCacheMock, layer, [tileId], null); + + expect(spy.mock.calls[0][9]).toBeFalsy(); // rotateToLine === false + }); + + test('transparent tile optimization should prevent program.draw from being called', () => { + + const painterMock = new Painter(null, null); + painterMock.context = { + gl: {}, + activeTexture: { + set: () => { } + } + } as any; + painterMock.renderPass = 'translucent'; + painterMock.transform = {pitch: 0, labelPlaneMatrix: mat4.create()} as any as Transform; + painterMock.options = {} as any; + painterMock.style = { + map: {} + } as any as Style; + + const layerSpec = { + id: 'mock-layer', + source: 'empty-source', + type: 'symbol', + layout: {}, + paint: { + 'text-opacity': 1 + } + } as SymbolLayerSpecification; + const layer = new SymbolStyleLayer(layerSpec); + layer.recalculate({zoom: 0, zoomHistory: {} as ZoomHistory} as EvaluationParameters, []); + + const tileId = new OverscaledTileID(1, 0, 1, 0, 0); + tileId.posMatrix = mat4.create(); + const programMock = new Program(null, null, null, null, null, null); + (painterMock.useProgram as jest.Mock).mockReturnValue(programMock); + const bucketMock = new SymbolBucket(null); + bucketMock.icon = { + programConfigurations: { + get: () => { } + }, + segments: { + get: () => [1] + }, + hasVisibleVertices: false // nark this bucket as having no visible vertices + } as any; + bucketMock.iconSizeData = { + kind: 'constant', + layoutSize: 1 + }; + const tile = new Tile(tileId, 256); + tile.tileID = tileId; + tile.imageAtlasTexture = { + bind: () => { } + } as any; + (tile.getBucket as jest.Mock).mockReturnValue(bucketMock); + const sourceCacheMock = new SourceCache(null, null, null); + (sourceCacheMock.getTile as jest.Mock).mockReturnValue(tile); + sourceCacheMock.map = {showCollisionBoxes: false} as any as Map; + + drawSymbols(painterMock, sourceCacheMock, layer, [tileId], null); + + expect(programMock.draw).toHaveBeenCalledTimes(0); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/render/draw_symbol.ts b/web/libraries/maplibre-gl/src/render/draw_symbol.ts new file mode 100644 index 00000000..11a20034 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_symbol.ts @@ -0,0 +1,449 @@ +import Point from '@mapbox/point-geometry'; +import {drawCollisionDebug} from './draw_collision_debug'; + +import {SegmentVector} from '../data/segment'; +import {pixelsToTileUnits} from '../source/pixels_to_tile_units'; +import * as symbolProjection from '../symbol/projection'; +import {EvaluatedZoomSize, evaluateSizeForFeature, evaluateSizeForZoom} from '../symbol/symbol_size'; +import {mat4} from 'gl-matrix'; +import {StencilMode} from '../gl/stencil_mode'; +import {DepthMode} from '../gl/depth_mode'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import {addDynamicAttributes} from '../data/bucket/symbol_bucket'; + +import {getAnchorAlignment, WritingMode} from '../symbol/shaping'; +import ONE_EM from '../symbol/one_em'; + +import { + SymbolIconUniformsType, + symbolIconUniformValues, + symbolSDFUniformValues, + symbolTextAndIconUniformValues +} from './program/symbol_program'; + +import type {Painter} from './painter'; +import type {SourceCache} from '../source/source_cache'; +import type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer'; + +import type {Texture} from '../render/texture'; +import type {OverscaledTileID} from '../source/tile_id'; +import type {UniformValues} from './uniform_binding'; +import type {SymbolSDFUniformsType} from '../render/program/symbol_program'; +import type {CrossTileID, VariableOffset} from '../symbol/placement'; +import type {SymbolBucket, SymbolBuffers} from '../data/bucket/symbol_bucket'; +import type {TerrainData} from '../render/terrain'; +import type {SymbolLayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {Transform} from '../geo/transform'; +import type {ColorMode} from '../gl/color_mode'; +import type {Program} from './program'; +import type {TextAnchor} from '../style/style_layer/variable_text_anchor'; + +type SymbolTileRenderState = { + segments: SegmentVector; + sortKey: number; + terrainData: TerrainData; + state: { + program: Program; + buffers: SymbolBuffers; + uniformValues: UniformValues; + atlasTexture: Texture; + atlasTextureIcon: Texture | null; + atlasInterpolation: GLenum; + atlasInterpolationIcon: GLenum; + isSDF: boolean; + hasHalo: boolean; + }; +}; + +const identityMat4 = mat4.identity(new Float32Array(16)); + +export function drawSymbols(painter: Painter, sourceCache: SourceCache, layer: SymbolStyleLayer, coords: Array, variableOffsets: { + [_ in CrossTileID]: VariableOffset; +}) { + if (painter.renderPass !== 'translucent') return; + + // Disable the stencil test so that labels aren't clipped to tile boundaries. + const stencilMode = StencilMode.disabled; + const colorMode = painter.colorModeForRenderPass(); + const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset'); + + //Compute variable-offsets before painting since icons and text data positioning + //depend on each other in this case. + if (hasVariablePlacement) { + updateVariableAnchors(coords, painter, layer, sourceCache, + layer.layout.get('text-rotation-alignment'), + layer.layout.get('text-pitch-alignment'), + variableOffsets + ); + } + + if (layer.paint.get('icon-opacity').constantOr(1) !== 0) { + drawLayerSymbols(painter, sourceCache, layer, coords, false, + layer.paint.get('icon-translate'), + layer.paint.get('icon-translate-anchor'), + layer.layout.get('icon-rotation-alignment'), + layer.layout.get('icon-pitch-alignment'), + layer.layout.get('icon-keep-upright'), + stencilMode, colorMode + ); + } + + if (layer.paint.get('text-opacity').constantOr(1) !== 0) { + drawLayerSymbols(painter, sourceCache, layer, coords, true, + layer.paint.get('text-translate'), + layer.paint.get('text-translate-anchor'), + layer.layout.get('text-rotation-alignment'), + layer.layout.get('text-pitch-alignment'), + layer.layout.get('text-keep-upright'), + stencilMode, colorMode + ); + } + + if (sourceCache.map.showCollisionBoxes) { + drawCollisionDebug(painter, sourceCache, layer, coords, layer.paint.get('text-translate'), + layer.paint.get('text-translate-anchor'), true); + drawCollisionDebug(painter, sourceCache, layer, coords, layer.paint.get('icon-translate'), + layer.paint.get('icon-translate-anchor'), false); + } +} + +function calculateVariableRenderShift( + anchor: TextAnchor, + width: number, + height: number, + textOffset: [number, number], + textBoxScale: number, + renderTextSize: number): Point { + const {horizontalAlign, verticalAlign} = getAnchorAlignment(anchor); + const shiftX = -(horizontalAlign - 0.5) * width; + const shiftY = -(verticalAlign - 0.5) * height; + return new Point( + (shiftX / textBoxScale + textOffset[0]) * renderTextSize, + (shiftY / textBoxScale + textOffset[1]) * renderTextSize + ); +} + +function updateVariableAnchors(coords: Array, + painter: Painter, + layer:SymbolStyleLayer, sourceCache: SourceCache, + rotationAlignment: SymbolLayerSpecification['layout']['text-rotation-alignment'], + pitchAlignment: SymbolLayerSpecification['layout']['text-pitch-alignment'], + variableOffsets: {[_ in CrossTileID]: VariableOffset}) { + const tr = painter.transform; + const rotateWithMap = rotationAlignment === 'map'; + const pitchWithMap = pitchAlignment === 'map'; + + for (const coord of coords) { + const tile = sourceCache.getTile(coord); + const bucket = tile.getBucket(layer) as SymbolBucket; + if (!bucket || !bucket.text || !bucket.text.segments.get().length) continue; + + const sizeData = bucket.textSizeData; + const size = evaluateSizeForZoom(sizeData, tr.zoom); + + const pixelToTileScale = pixelsToTileUnits(tile, 1, painter.transform.zoom); + const labelPlaneMatrix = symbolProjection.getLabelPlaneMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, pixelToTileScale); + const updateTextFitIcon = layer.layout.get('icon-text-fit') !== 'none' && bucket.hasIconData(); + + if (size) { + const tileScale = Math.pow(2, tr.zoom - tile.tileID.overscaledZ); + const getElevation = painter.style.map.terrain ? (x: number, y: number) => painter.style.map.terrain.getElevation(coord, x, y) : null; + updateVariableAnchorsForBucket(bucket, rotateWithMap, pitchWithMap, variableOffsets, + tr, labelPlaneMatrix, coord.posMatrix, tileScale, size, updateTextFitIcon, getElevation); + } + } +} + +function updateVariableAnchorsForBucket( + bucket: SymbolBucket, + rotateWithMap: boolean, + pitchWithMap: boolean, + variableOffsets: {[_ in CrossTileID]: VariableOffset}, + transform: Transform, + labelPlaneMatrix: mat4, + posMatrix: mat4, + tileScale: number, + size: EvaluatedZoomSize, + updateTextFitIcon: boolean, + getElevation: (x: number, y: number) => number) { + const placedSymbols = bucket.text.placedSymbolArray; + const dynamicTextLayoutVertexArray = bucket.text.dynamicLayoutVertexArray; + const dynamicIconLayoutVertexArray = bucket.icon.dynamicLayoutVertexArray; + const placedTextShifts = {}; + + dynamicTextLayoutVertexArray.clear(); + for (let s = 0; s < placedSymbols.length; s++) { + const symbol = placedSymbols.get(s); + const skipOrientation = bucket.allowVerticalPlacement && !symbol.placedOrientation; + const variableOffset = (!symbol.hidden && symbol.crossTileID && !skipOrientation) ? variableOffsets[symbol.crossTileID] : null; + + if (!variableOffset) { + // These symbols are from a justification that is not being used, or a label that wasn't placed + // so we don't need to do the extra math to figure out what incremental shift to apply. + symbolProjection.hideGlyphs(symbol.numGlyphs, dynamicTextLayoutVertexArray); + } else { + const tileAnchor = new Point(symbol.anchorX, symbol.anchorY); + const projectedAnchor = symbolProjection.project(tileAnchor, pitchWithMap ? posMatrix : labelPlaneMatrix, getElevation); + const perspectiveRatio = symbolProjection.getPerspectiveRatio(transform.cameraToCenterDistance, projectedAnchor.signedDistanceFromCamera); + let renderTextSize = evaluateSizeForFeature(bucket.textSizeData, size, symbol) * perspectiveRatio / ONE_EM; + if (pitchWithMap) { + // Go from size in pixels to equivalent size in tile units + renderTextSize *= bucket.tilePixelRatio / tileScale; + } + + const {width, height, anchor, textOffset, textBoxScale} = variableOffset; + + const shift = calculateVariableRenderShift( + anchor, width, height, textOffset, textBoxScale, renderTextSize); + + // Usual case is that we take the projected anchor and add the pixel-based shift + // calculated above. In the (somewhat weird) case of pitch-aligned text, we add an equivalent + // tile-unit based shift to the anchor before projecting to the label plane. + const shiftedAnchor = pitchWithMap ? + symbolProjection.project(tileAnchor.add(shift), labelPlaneMatrix, getElevation).point : + projectedAnchor.point.add(rotateWithMap ? + shift.rotate(-transform.angle) : + shift); + + const angle = (bucket.allowVerticalPlacement && symbol.placedOrientation === WritingMode.vertical) ? Math.PI / 2 : 0; + for (let g = 0; g < symbol.numGlyphs; g++) { + addDynamicAttributes(dynamicTextLayoutVertexArray, shiftedAnchor, angle); + } + //Only offset horizontal text icons + if (updateTextFitIcon && symbol.associatedIconIndex >= 0) { + placedTextShifts[symbol.associatedIconIndex] = {shiftedAnchor, angle}; + } + } + } + + if (updateTextFitIcon) { + dynamicIconLayoutVertexArray.clear(); + const placedIcons = bucket.icon.placedSymbolArray; + for (let i = 0; i < placedIcons.length; i++) { + const placedIcon = placedIcons.get(i); + if (placedIcon.hidden) { + symbolProjection.hideGlyphs(placedIcon.numGlyphs, dynamicIconLayoutVertexArray); + } else { + const shift = placedTextShifts[i]; + if (!shift) { + symbolProjection.hideGlyphs(placedIcon.numGlyphs, dynamicIconLayoutVertexArray); + } else { + for (let g = 0; g < placedIcon.numGlyphs; g++) { + addDynamicAttributes(dynamicIconLayoutVertexArray, shift.shiftedAnchor, shift.angle); + } + } + } + } + bucket.icon.dynamicLayoutVertexBuffer.updateData(dynamicIconLayoutVertexArray); + } + bucket.text.dynamicLayoutVertexBuffer.updateData(dynamicTextLayoutVertexArray); +} + +function getSymbolProgramName(isSDF: boolean, isText: boolean, bucket: SymbolBucket) { + if (bucket.iconsInText && isText) { + return 'symbolTextAndIcon'; + } else if (isSDF) { + return 'symbolSDF'; + } else { + return 'symbolIcon'; + } +} + +function drawLayerSymbols( + painter: Painter, + sourceCache: SourceCache, + layer: SymbolStyleLayer, + coords: Array, + isText: boolean, + translate: [number, number], + translateAnchor: 'map' | 'viewport', + rotationAlignment: SymbolLayerSpecification['layout']['text-rotation-alignment'], + pitchAlignment: SymbolLayerSpecification['layout']['text-pitch-alignment'], + keepUpright: boolean, + stencilMode: StencilMode, + colorMode: Readonly) { + + const context = painter.context; + const gl = context.gl; + const tr = painter.transform; + + const rotateWithMap = rotationAlignment === 'map'; + const pitchWithMap = pitchAlignment === 'map'; + const alongLine = rotationAlignment !== 'viewport' && layer.layout.get('symbol-placement') !== 'point'; + // Line label rotation happens in `updateLineLabels` + // Pitched point labels are automatically rotated by the labelPlaneMatrix projection + // Unpitched point labels need to have their rotation applied after projection + const rotateInShader = rotateWithMap && !pitchWithMap && !alongLine; + + const hasSortKey = !layer.layout.get('symbol-sort-key').isConstant(); + let sortFeaturesByKey = false; + + const depthMode = painter.depthModeForSublayer(0, DepthMode.ReadOnly); + + const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset'); + + const tileRenderState: Array = []; + + for (const coord of coords) { + const tile = sourceCache.getTile(coord); + const bucket = tile.getBucket(layer) as SymbolBucket; + if (!bucket) continue; + const buffers = isText ? bucket.text : bucket.icon; + + if (!buffers || !buffers.segments.get().length || !buffers.hasVisibleVertices) continue; + const programConfiguration = buffers.programConfigurations.get(layer.id); + + const isSDF = isText || bucket.sdfIcons; + + const sizeData = isText ? bucket.textSizeData : bucket.iconSizeData; + const transformed = pitchWithMap || tr.pitch !== 0; + + const program = painter.useProgram(getSymbolProgramName(isSDF, isText, bucket), programConfiguration); + const size = evaluateSizeForZoom(sizeData, tr.zoom); + const terrainData = painter.style.map.terrain && painter.style.map.terrain.getTerrainData(coord); + + let texSize: [number, number]; + let texSizeIcon: [number, number] = [0, 0]; + let atlasTexture: Texture; + let atlasInterpolation: GLenum; + let atlasTextureIcon = null; + let atlasInterpolationIcon: GLenum; + if (isText) { + atlasTexture = tile.glyphAtlasTexture; + atlasInterpolation = gl.LINEAR; + texSize = tile.glyphAtlasTexture.size; + if (bucket.iconsInText) { + texSizeIcon = tile.imageAtlasTexture.size; + atlasTextureIcon = tile.imageAtlasTexture; + const zoomDependentSize = sizeData.kind === 'composite' || sizeData.kind === 'camera'; + atlasInterpolationIcon = transformed || painter.options.rotating || painter.options.zooming || zoomDependentSize ? gl.LINEAR : gl.NEAREST; + } + } else { + const iconScaled = layer.layout.get('icon-size').constantOr(0) !== 1 || bucket.iconsNeedLinear; + atlasTexture = tile.imageAtlasTexture; + atlasInterpolation = isSDF || painter.options.rotating || painter.options.zooming || iconScaled || transformed ? + gl.LINEAR : + gl.NEAREST; + texSize = tile.imageAtlasTexture.size; + } + + const s = pixelsToTileUnits(tile, 1, painter.transform.zoom); + const labelPlaneMatrix = symbolProjection.getLabelPlaneMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, s); + const glCoordMatrix = symbolProjection.getGlCoordMatrix(coord.posMatrix, pitchWithMap, rotateWithMap, painter.transform, s); + + const hasVariableAnchors = hasVariablePlacement && bucket.hasTextData(); + const updateTextFitIcon = layer.layout.get('icon-text-fit') !== 'none' && + hasVariableAnchors && + bucket.hasIconData(); + + if (alongLine) { + const getElevation = painter.style.map.terrain ? (x: number, y: number) => painter.style.map.terrain.getElevation(coord, x, y) : null; + const rotateToLine = layer.layout.get('text-rotation-alignment') === 'map'; + symbolProjection.updateLineLabels(bucket, coord.posMatrix, painter, isText, labelPlaneMatrix, glCoordMatrix, pitchWithMap, keepUpright, rotateToLine, getElevation); + } + + const matrix = painter.translatePosMatrix(coord.posMatrix, tile, translate, translateAnchor), + uLabelPlaneMatrix = (alongLine || (isText && hasVariablePlacement) || updateTextFitIcon) ? identityMat4 : labelPlaneMatrix, + uglCoordMatrix = painter.translatePosMatrix(glCoordMatrix, tile, translate, translateAnchor, true); + + const hasHalo = isSDF && layer.paint.get(isText ? 'text-halo-width' : 'icon-halo-width').constantOr(1) !== 0; + + let uniformValues: UniformValues; + if (isSDF) { + if (!bucket.iconsInText) { + uniformValues = symbolSDFUniformValues(sizeData.kind, + size, rotateInShader, pitchWithMap, painter, matrix, + uLabelPlaneMatrix, uglCoordMatrix, isText, texSize, true); + } else { + uniformValues = symbolTextAndIconUniformValues(sizeData.kind, + size, rotateInShader, pitchWithMap, painter, matrix, + uLabelPlaneMatrix, uglCoordMatrix, texSize, texSizeIcon); + } + } else { + uniformValues = symbolIconUniformValues(sizeData.kind, + size, rotateInShader, pitchWithMap, painter, matrix, + uLabelPlaneMatrix, uglCoordMatrix, isText, texSize); + } + + const state = { + program, + buffers, + uniformValues, + atlasTexture, + atlasTextureIcon, + atlasInterpolation, + atlasInterpolationIcon, + isSDF, + hasHalo + }; + + if (hasSortKey && bucket.canOverlap) { + sortFeaturesByKey = true; + const oldSegments = buffers.segments.get(); + for (const segment of oldSegments) { + tileRenderState.push({ + segments: new SegmentVector([segment]), + sortKey: segment.sortKey, + state, + terrainData + }); + } + } else { + tileRenderState.push({ + segments: buffers.segments, + sortKey: 0, + state, + terrainData + }); + } + } + + if (sortFeaturesByKey) { + tileRenderState.sort((a, b) => a.sortKey - b.sortKey); + } + + for (const segmentState of tileRenderState) { + const state = segmentState.state; + + context.activeTexture.set(gl.TEXTURE0); + // @ts-ignore + state.atlasTexture.bind(state.atlasInterpolation, gl.CLAMP_TO_EDGE); + if (state.atlasTextureIcon) { + context.activeTexture.set(gl.TEXTURE1); + if (state.atlasTextureIcon) { + // @ts-ignore + state.atlasTextureIcon.bind(state.atlasInterpolationIcon, gl.CLAMP_TO_EDGE); + } + } + + if (state.isSDF) { + const uniformValues = state.uniformValues; + if (state.hasHalo) { + uniformValues['u_is_halo'] = 1; + drawSymbolElements(state.buffers, segmentState.segments, layer, painter, state.program, depthMode, stencilMode, colorMode, uniformValues, segmentState.terrainData); + } + uniformValues['u_is_halo'] = 0; + } + drawSymbolElements(state.buffers, segmentState.segments, layer, painter, state.program, depthMode, stencilMode, colorMode, state.uniformValues, segmentState.terrainData); + } +} + +function drawSymbolElements( + buffers: SymbolBuffers, + segments: SegmentVector, + layer: SymbolStyleLayer, + painter: Painter, + program: Program, + depthMode: Readonly, + stencilMode: StencilMode, + colorMode: Readonly, + uniformValues: UniformValues, + terrainData: TerrainData) { + const context = painter.context; + const gl = context.gl; + program.draw(context, gl.TRIANGLES, depthMode, stencilMode, colorMode, CullFaceMode.disabled, + uniformValues, terrainData, layer.id, buffers.layoutVertexBuffer, + buffers.indexBuffer, segments, layer.paint, + painter.transform.zoom, buffers.programConfigurations.get(layer.id), + buffers.dynamicLayoutVertexBuffer, buffers.opacityVertexBuffer); +} diff --git a/web/libraries/maplibre-gl/src/render/draw_terrain.ts b/web/libraries/maplibre-gl/src/render/draw_terrain.ts new file mode 100644 index 00000000..688c5781 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/draw_terrain.ts @@ -0,0 +1,97 @@ +import {StencilMode} from '../gl/stencil_mode'; +import {DepthMode} from '../gl/depth_mode'; +import {terrainUniformValues, terrainDepthUniformValues, terrainCoordsUniformValues} from './program/terrain_program'; +import type {Painter} from './painter'; +import type {Tile} from '../source/tile'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import {Color} from '@maplibre/maplibre-gl-style-spec'; +import {ColorMode} from '../gl/color_mode'; +import {Terrain} from './terrain'; + +/** + * Redraw the Depth Framebuffer + * @param painter - the painter + * @param terrain - the terrain + */ +function drawDepth(painter: Painter, terrain: Terrain) { + const context = painter.context; + const gl = context.gl; + const colorMode = ColorMode.unblended; + const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, [0, 1]); + const mesh = terrain.getTerrainMesh(); + const tiles = terrain.sourceCache.getRenderableTiles(); + const program = painter.useProgram('terrainDepth'); + context.bindFramebuffer.set(terrain.getFramebuffer('depth').framebuffer); + context.viewport.set([0, 0, painter.width / devicePixelRatio, painter.height / devicePixelRatio]); + context.clear({color: Color.transparent, depth: 1}); + for (const tile of tiles) { + const terrainData = terrain.getTerrainData(tile.tileID); + const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped()); + const uniformValues = terrainDepthUniformValues(posMatrix, terrain.getMeshFrameDelta(painter.transform.zoom)); + program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments); + } + context.bindFramebuffer.set(null); + context.viewport.set([0, 0, painter.width, painter.height]); +} + +/** + * Redraw the Coords Framebuffers + * @param painter - the painter + * @param terrain - the terrain + */ +function drawCoords(painter: Painter, terrain: Terrain) { + const context = painter.context; + const gl = context.gl; + const colorMode = ColorMode.unblended; + const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, [0, 1]); + const mesh = terrain.getTerrainMesh(); + const coords = terrain.getCoordsTexture(); + const tiles = terrain.sourceCache.getRenderableTiles(); + + // draw tile-coords into framebuffer + const program = painter.useProgram('terrainCoords'); + context.bindFramebuffer.set(terrain.getFramebuffer('coords').framebuffer); + context.viewport.set([0, 0, painter.width / devicePixelRatio, painter.height / devicePixelRatio]); + context.clear({color: Color.transparent, depth: 1}); + terrain.coordsIndex = []; + for (const tile of tiles) { + const terrainData = terrain.getTerrainData(tile.tileID); + context.activeTexture.set(gl.TEXTURE0); + gl.bindTexture(gl.TEXTURE_2D, coords.texture); + const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped()); + const uniformValues = terrainCoordsUniformValues(posMatrix, 255 - terrain.coordsIndex.length, terrain.getMeshFrameDelta(painter.transform.zoom)); + program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments); + terrain.coordsIndex.push(tile.tileID.key); + } + context.bindFramebuffer.set(null); + context.viewport.set([0, 0, painter.width, painter.height]); +} + +function drawTerrain(painter: Painter, terrain: Terrain, tiles: Array) { + const context = painter.context; + const gl = context.gl; + const colorMode = painter.colorModeForRenderPass(); + const depthMode = new DepthMode(gl.LEQUAL, DepthMode.ReadWrite, painter.depthRangeFor3D); + const program = painter.useProgram('terrain'); + const mesh = terrain.getTerrainMesh(); + + context.bindFramebuffer.set(null); + context.viewport.set([0, 0, painter.width, painter.height]); + + for (const tile of tiles) { + const texture = painter.renderToTexture.getTexture(tile); + const terrainData = terrain.getTerrainData(tile.tileID); + context.activeTexture.set(gl.TEXTURE0); + gl.bindTexture(gl.TEXTURE_2D, texture.texture); + const posMatrix = painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped()); + const uniformValues = terrainUniformValues(posMatrix, terrain.getMeshFrameDelta(painter.transform.zoom)); + program.draw(context, gl.TRIANGLES, depthMode, StencilMode.disabled, colorMode, CullFaceMode.backCCW, uniformValues, terrainData, 'terrain', mesh.vertexBuffer, mesh.indexBuffer, mesh.segments); + } + +} + +export { + drawTerrain, + drawDepth, + drawCoords +}; diff --git a/web/libraries/maplibre-gl/src/render/glyph_atlas.ts b/web/libraries/maplibre-gl/src/render/glyph_atlas.ts new file mode 100644 index 00000000..0f80b336 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/glyph_atlas.ts @@ -0,0 +1,86 @@ +import {AlphaImage} from '../util/image'; +import {register} from '../util/web_worker_transfer'; +import potpack from 'potpack'; + +import type {GlyphMetrics, StyleGlyph} from '../style/style_glyph'; + +const padding = 1; + +/** + * A rectangle type with postion, width and height. + */ +export type Rect = { + x: number; + y: number; + w: number; + h: number; +}; + +/** + * The glyph's position + */ +export type GlyphPosition = { + rect: Rect; + metrics: GlyphMetrics; +}; + +/** + * The glyphs' positions + */ +export type GlyphPositions = { + [_: string]: { + [_: number]: GlyphPosition; + }; +}; + +export class GlyphAtlas { + image: AlphaImage; + positions: GlyphPositions; + + constructor(stacks: { + [_: string]: { + [_: number]: StyleGlyph; + }; + }) { + const positions = {}; + const bins = []; + + for (const stack in stacks) { + const glyphs = stacks[stack]; + const stackPositions = positions[stack] = {}; + + for (const id in glyphs) { + const src = glyphs[+id]; + if (!src || src.bitmap.width === 0 || src.bitmap.height === 0) continue; + + const bin = { + x: 0, + y: 0, + w: src.bitmap.width + 2 * padding, + h: src.bitmap.height + 2 * padding + }; + bins.push(bin); + stackPositions[id] = {rect: bin, metrics: src.metrics}; + } + } + + const {w, h} = potpack(bins); + const image = new AlphaImage({width: w || 1, height: h || 1}); + + for (const stack in stacks) { + const glyphs = stacks[stack]; + + for (const id in glyphs) { + const src = glyphs[+id]; + if (!src || src.bitmap.width === 0 || src.bitmap.height === 0) continue; + const bin = positions[stack][id].rect; + AlphaImage.copy(src.bitmap, image, {x: 0, y: 0}, {x: bin.x + padding, y: bin.y + padding}, src.bitmap); + } + } + + this.image = image; + this.positions = positions; + } +} + +register('GlyphAtlas', GlyphAtlas); diff --git a/web/libraries/maplibre-gl/src/render/glyph_manager.test.ts b/web/libraries/maplibre-gl/src/render/glyph_manager.test.ts new file mode 100644 index 00000000..65c0ff72 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/glyph_manager.test.ts @@ -0,0 +1,160 @@ +import {parseGlyphPbf} from '../style/parse_glyph_pbf'; +import {GlyphManager} from './glyph_manager'; +import fs from 'fs'; +import {RequestManager} from '../util/request_manager'; + +const glyphs = {}; +for (const glyph of parseGlyphPbf(fs.readFileSync('./test/unit/assets/0-255.pbf'))) { + glyphs[glyph.id] = glyph; +} + +const identityTransform = ((url) => ({url})) as any as RequestManager; + +const createLoadGlyphRangeStub = () => { + return jest.spyOn(GlyphManager, 'loadGlyphRange').mockImplementation((stack, range, urlTemplate, transform, callback) => { + expect(stack).toBe('Arial Unicode MS'); + expect(range).toBe(0); + expect(urlTemplate).toBe('https://localhost/fonts/v1/{fontstack}/{range}.pbf'); + expect(transform).toBe(identityTransform); + setTimeout(() => callback(null, glyphs), 0); + }); +}; + +const createGlyphManager = (font?) => { + const manager = new GlyphManager(identityTransform, font); + manager.setURL('https://localhost/fonts/v1/{fontstack}/{range}.pbf'); + return manager; +}; + +afterEach(() => { + jest.clearAllMocks(); +}); + +describe('GlyphManager', () => { + + test('GlyphManager requests 0-255 PBF', done => { + createLoadGlyphRangeStub(); + const manager = createGlyphManager(); + + manager.getGlyphs({'Arial Unicode MS': [55]}, (err, glyphs) => { + expect(err).toBeFalsy(); + expect(glyphs['Arial Unicode MS']['55'].metrics.advance).toBe(12); + done(); + }); + }); + + test('GlyphManager doesn\'t request twice 0-255 PBF if a glyph is missing', done => { + const stub = createLoadGlyphRangeStub(); + const manager = createGlyphManager(); + + manager.getGlyphs({'Arial Unicode MS': [0.5]}, (err) => { + expect(err).toBeFalsy(); + expect(manager.entries['Arial Unicode MS'].ranges[0]).toBe(true); + expect(stub).toHaveBeenCalledTimes(1); + + // We remove all requests as in getGlyphs code. + delete manager.entries['Arial Unicode MS'].requests[0]; + + manager.getGlyphs({'Arial Unicode MS': [0.5]}, (err) => { + expect(err).toBeFalsy(); + expect(manager.entries['Arial Unicode MS'].ranges[0]).toBe(true); + expect(stub).toHaveBeenCalledTimes(1); + done(); + }); + }); + }); + + test('GlyphManager requests remote CJK PBF', done => { + jest.spyOn(GlyphManager, 'loadGlyphRange').mockImplementation((stack, range, urlTemplate, transform, callback) => { + setTimeout(() => callback(null, glyphs), 0); + }); + + const manager = createGlyphManager(); + + manager.getGlyphs({'Arial Unicode MS': [0x5e73]}, (err, glyphs) => { + expect(err).toBeFalsy(); + expect(glyphs['Arial Unicode MS'][0x5e73]).toBeNull(); // The fixture returns a PBF without the glyph we requested + done(); + }); + }); + + test('GlyphManager does not cache CJK chars that should be rendered locally', done => { + jest.spyOn(GlyphManager, 'loadGlyphRange').mockImplementation((stack, range, urlTemplate, transform, callback) => { + const overlappingGlyphs = {}; + const start = range * 256; + const end = start + 256; + for (let i = start, j = 0; i < end; i++, j++) { + overlappingGlyphs[i] = glyphs[j]; + } + setTimeout(() => callback(null, overlappingGlyphs), 0); + }); + + const manager = createGlyphManager('sans-serif'); + + //Request char that overlaps Katakana range + manager.getGlyphs({'Arial Unicode MS': [0x3005]}, (err, glyphs) => { + expect(err).toBeFalsy(); + expect(glyphs['Arial Unicode MS'][0x3005]).not.toBeNull(); + //Request char from Katakana range (te テ) + manager.getGlyphs({'Arial Unicode MS': [0x30C6]}, (err, glyphs) => { + expect(err).toBeFalsy(); + const glyph = glyphs['Arial Unicode MS'][0x30c6]; + //Ensure that te is locally generated. + expect(glyph.bitmap.height).toBe(12); + expect(glyph.bitmap.width).toBe(12); + done(); + }); + }); + }); + + test('GlyphManager generates CJK PBF locally', done => { + const manager = createGlyphManager('sans-serif'); + + // character 平 + manager.getGlyphs({'Arial Unicode MS': [0x5e73]}, (err, glyphs) => { + expect(err).toBeFalsy(); + expect(glyphs['Arial Unicode MS'][0x5e73].metrics.advance).toBe(0.5); + done(); + }); + }); + + test('GlyphManager generates Katakana PBF locally', done => { + const manager = createGlyphManager('sans-serif'); + + // Katakana letter te テ + manager.getGlyphs({'Arial Unicode MS': [0x30c6]}, (err, glyphs) => { + expect(err).toBeFalsy(); + expect(glyphs['Arial Unicode MS'][0x30c6].metrics.advance).toBe(0.5); + done(); + }); + }); + + test('GlyphManager generates Hiragana PBF locally', done => { + const manager = createGlyphManager('sans-serif'); + + //Hiragana letter te て + manager.getGlyphs({'Arial Unicode MS': [0x3066]}, (err, glyphs) => { + expect(err).toBeFalsy(); + expect(glyphs['Arial Unicode MS'][0x3066].metrics.advance).toBe(0.5); + done(); + }); + }); + + test('GlyphManager caches locally generated glyphs', done => { + + const manager = createGlyphManager('sans-serif'); + const drawSpy = GlyphManager.TinySDF.prototype.draw = jest.fn().mockImplementation(() => { + return {data: new Uint8ClampedArray(60 * 60)} as any; + }); + + // Katakana letter te + manager.getGlyphs({'Arial Unicode MS': [0x30c6]}, (err, glyphs) => { + expect(err).toBeFalsy(); + expect(glyphs['Arial Unicode MS'][0x30c6].metrics.advance).toBe(24); + manager.getGlyphs({'Arial Unicode MS': [0x30c6]}, () => { + expect(drawSpy).toHaveBeenCalledTimes(1); + done(); + }); + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/render/glyph_manager.ts b/web/libraries/maplibre-gl/src/render/glyph_manager.ts new file mode 100644 index 00000000..acffee4d --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/glyph_manager.ts @@ -0,0 +1,239 @@ +import {loadGlyphRange} from '../style/load_glyph_range'; + +import TinySDF from '@mapbox/tiny-sdf'; +import {unicodeBlockLookup} from '../util/is_char_in_unicode_block'; +import {asyncAll} from '../util/util'; +import {AlphaImage} from '../util/image'; + +import type {StyleGlyph} from '../style/style_glyph'; +import type {RequestManager} from '../util/request_manager'; +import type {Callback} from '../types/callback'; + +type Entry = { + // null means we've requested the range, but the glyph wasn't included in the result. + glyphs: { + [id: number]: StyleGlyph | null; + }; + requests: { + [range: number]: Array>; + }; + ranges: { + [range: number]: boolean | null; + }; + tinySDF?: TinySDF; +}; + +export class GlyphManager { + requestManager: RequestManager; + localIdeographFontFamily: string; + entries: { + [_: string]: Entry; + }; + url: string; + + // exposed as statics to enable stubbing in unit tests + static loadGlyphRange = loadGlyphRange; + static TinySDF = TinySDF; + + constructor(requestManager: RequestManager, localIdeographFontFamily?: string | null) { + this.requestManager = requestManager; + this.localIdeographFontFamily = localIdeographFontFamily; + this.entries = {}; + } + + setURL(url?: string | null) { + this.url = url; + } + + getGlyphs(glyphs: { + [stack: string]: Array; + }, callback: Callback<{ + [stack: string]: { + [id: number]: StyleGlyph; + }; + }>) { + const all = []; + + for (const stack in glyphs) { + for (const id of glyphs[stack]) { + all.push({stack, id}); + } + } + + asyncAll(all, ({stack, id}, callback: Callback<{ + stack: string; + id: number; + glyph: StyleGlyph; + }>) => { + let entry = this.entries[stack]; + if (!entry) { + entry = this.entries[stack] = { + glyphs: {}, + requests: {}, + ranges: {} + }; + } + + let glyph = entry.glyphs[id]; + if (glyph !== undefined) { + callback(null, {stack, id, glyph}); + return; + } + + glyph = this._tinySDF(entry, stack, id); + if (glyph) { + entry.glyphs[id] = glyph; + callback(null, {stack, id, glyph}); + return; + } + + const range = Math.floor(id / 256); + if (range * 256 > 65535) { + callback(new Error('glyphs > 65535 not supported')); + return; + } + + if (entry.ranges[range]) { + callback(null, {stack, id, glyph}); + return; + } + + if (!this.url) { + callback(new Error('glyphsUrl is not set')); + return; + } + + let requests = entry.requests[range]; + if (!requests) { + requests = entry.requests[range] = []; + GlyphManager.loadGlyphRange(stack, range, this.url, this.requestManager, + (err, response?: { + [_: number]: StyleGlyph | null; + } | null) => { + if (response) { + for (const id in response) { + if (!this._doesCharSupportLocalGlyph(+id)) { + entry.glyphs[+id] = response[+id]; + } + } + entry.ranges[range] = true; + } + for (const cb of requests) { + cb(err, response); + } + delete entry.requests[range]; + }); + } + + requests.push((err, result?: { + [_: number]: StyleGlyph | null; + } | null) => { + if (err) { + callback(err); + } else if (result) { + callback(null, {stack, id, glyph: result[id] || null}); + } + }); + }, (err, glyphs?: Array<{ + stack: string; + id: number; + glyph: StyleGlyph; + }> | null) => { + if (err) { + callback(err); + } else if (glyphs) { + const result = {}; + + for (const {stack, id, glyph} of glyphs) { + // Clone the glyph so that our own copy of its ArrayBuffer doesn't get transferred. + (result[stack] || (result[stack] = {}))[id] = glyph && { + id: glyph.id, + bitmap: glyph.bitmap.clone(), + metrics: glyph.metrics + }; + } + + callback(null, result); + } + }); + } + + _doesCharSupportLocalGlyph(id: number): boolean { + /* eslint-disable new-cap */ + return !!this.localIdeographFontFamily && + (unicodeBlockLookup['CJK Unified Ideographs'](id) || + unicodeBlockLookup['Hangul Syllables'](id) || + unicodeBlockLookup['Hiragana'](id) || + unicodeBlockLookup['Katakana'](id)); + /* eslint-enable new-cap */ + } + + _tinySDF(entry: Entry, stack: string, id: number): StyleGlyph { + const fontFamily = this.localIdeographFontFamily; + if (!fontFamily) { + return; + } + + if (!this._doesCharSupportLocalGlyph(id)) { + return; + } + + // Client-generated glyphs are rendered at 2x texture scale, + // because CJK glyphs are more detailed than others. + const textureScale = 2; + + let tinySDF = entry.tinySDF; + if (!tinySDF) { + let fontWeight = '400'; + if (/bold/i.test(stack)) { + fontWeight = '900'; + } else if (/medium/i.test(stack)) { + fontWeight = '500'; + } else if (/light/i.test(stack)) { + fontWeight = '200'; + } + tinySDF = entry.tinySDF = new GlyphManager.TinySDF({ + fontSize: 24 * textureScale, + buffer: 3 * textureScale, + radius: 8 * textureScale, + cutoff: 0.25, + fontFamily, + fontWeight + }); + } + + const char = tinySDF.draw(String.fromCharCode(id)); + + /** + * TinySDF's "top" is the distance from the alphabetic baseline to the top of the glyph. + * Server-generated fonts specify "top" relative to an origin above the em box (the origin + * comes from FreeType, but I'm unclear on exactly how it's derived) + * ref: https://github.com/mapbox/sdf-glyph-foundry + * + * Server fonts don't yet include baseline information, so we can't line up exactly with them + * (and they don't line up with each other) + * ref: https://github.com/mapbox/node-fontnik/pull/160 + * + * To approximately align TinySDF glyphs with server-provided glyphs, we use this baseline adjustment + * factor calibrated to be in between DIN Pro and Arial Unicode (but closer to Arial Unicode) + */ + const topAdjustment = 27.5; + + const leftAdjustment = 0.5; + + return { + id, + bitmap: new AlphaImage({width: char.width || 30 * textureScale, height: char.height || 30 * textureScale}, char.data), + metrics: { + width: char.glyphWidth / textureScale || 24, + height: char.glyphHeight / textureScale || 24, + left: (char.glyphLeft / textureScale + leftAdjustment) || 0, + top: char.glyphTop / textureScale - topAdjustment || -8, + advance: char.glyphAdvance / textureScale || 24, + isDoubleResolution: true + } + }; + } +} diff --git a/web/libraries/maplibre-gl/src/render/image_atlas.ts b/web/libraries/maplibre-gl/src/render/image_atlas.ts new file mode 100644 index 00000000..a86d2e69 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/image_atlas.ts @@ -0,0 +1,152 @@ +/* eslint-disable key-spacing */ +import {RGBAImage} from '../util/image'; +import {register} from '../util/web_worker_transfer'; +import potpack from 'potpack'; + +import type {StyleImage} from '../style/style_image'; +import type {ImageManager} from './image_manager'; +import type {Texture} from './texture'; +import type {Rect} from './glyph_atlas'; + +const IMAGE_PADDING: number = 1; +export {IMAGE_PADDING}; + +export class ImagePosition { + paddedRect: Rect; + pixelRatio: number; + version: number; + stretchY: Array<[number, number]>; + stretchX: Array<[number, number]>; + content: [number, number, number, number]; + + constructor(paddedRect: Rect, { + pixelRatio, + version, + stretchX, + stretchY, + content + }: StyleImage) { + this.paddedRect = paddedRect; + this.pixelRatio = pixelRatio; + this.stretchX = stretchX; + this.stretchY = stretchY; + this.content = content; + this.version = version; + } + + get tl(): [number, number] { + return [ + this.paddedRect.x + IMAGE_PADDING, + this.paddedRect.y + IMAGE_PADDING + ]; + } + + get br(): [number, number] { + return [ + this.paddedRect.x + this.paddedRect.w - IMAGE_PADDING, + this.paddedRect.y + this.paddedRect.h - IMAGE_PADDING + ]; + } + + get tlbr(): Array { + return this.tl.concat(this.br); + } + + get displaySize(): [number, number] { + return [ + (this.paddedRect.w - IMAGE_PADDING * 2) / this.pixelRatio, + (this.paddedRect.h - IMAGE_PADDING * 2) / this.pixelRatio + ]; + } +} + +/** + * @internal + * A class holding all the images + */ +export class ImageAtlas { + image: RGBAImage; + iconPositions: {[_: string]: ImagePosition}; + patternPositions: {[_: string]: ImagePosition}; + haveRenderCallbacks: Array; + uploaded: boolean; + + constructor(icons: {[_: string]: StyleImage}, patterns: {[_: string]: StyleImage}) { + const iconPositions = {}, patternPositions = {}; + this.haveRenderCallbacks = []; + + const bins = []; + + this.addImages(icons, iconPositions, bins); + this.addImages(patterns, patternPositions, bins); + + const {w, h} = potpack(bins); + const image = new RGBAImage({width: w || 1, height: h || 1}); + + for (const id in icons) { + const src = icons[id]; + const bin = iconPositions[id].paddedRect; + RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x: bin.x + IMAGE_PADDING, y: bin.y + IMAGE_PADDING}, src.data); + } + + for (const id in patterns) { + const src = patterns[id]; + const bin = patternPositions[id].paddedRect; + const x = bin.x + IMAGE_PADDING, + y = bin.y + IMAGE_PADDING, + w = src.data.width, + h = src.data.height; + + RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x, y}, src.data); + // Add 1 pixel wrapped padding on each side of the image. + RGBAImage.copy(src.data, image, {x: 0, y: h - 1}, {x, y: y - 1}, {width: w, height: 1}); // T + RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x, y: y + h}, {width: w, height: 1}); // B + RGBAImage.copy(src.data, image, {x: w - 1, y: 0}, {x: x - 1, y}, {width: 1, height: h}); // L + RGBAImage.copy(src.data, image, {x: 0, y: 0}, {x: x + w, y}, {width: 1, height: h}); // R + } + + this.image = image; + this.iconPositions = iconPositions; + this.patternPositions = patternPositions; + } + + addImages(images: {[_: string]: StyleImage}, positions: {[_: string]: ImagePosition}, bins: Array) { + for (const id in images) { + const src = images[id]; + const bin = { + x: 0, + y: 0, + w: src.data.width + 2 * IMAGE_PADDING, + h: src.data.height + 2 * IMAGE_PADDING, + }; + bins.push(bin); + positions[id] = new ImagePosition(bin, src); + + if (src.hasRenderCallback) { + this.haveRenderCallbacks.push(id); + } + } + } + + patchUpdatedImages(imageManager: ImageManager, texture: Texture) { + imageManager.dispatchRenderCallbacks(this.haveRenderCallbacks); + for (const name in imageManager.updatedImages) { + this.patchUpdatedImage(this.iconPositions[name], imageManager.getImage(name), texture); + this.patchUpdatedImage(this.patternPositions[name], imageManager.getImage(name), texture); + } + } + + patchUpdatedImage(position: ImagePosition, image: StyleImage, texture: Texture) { + if (!position || !image) return; + + if (position.version === image.version) return; + + position.version = image.version; + const [x, y] = position.tl; + texture.update(image.data, undefined, {x, y}); + } + +} + +register('ImagePosition', ImagePosition); +register('ImageAtlas', ImageAtlas); diff --git a/web/libraries/maplibre-gl/src/render/image_manager.ts b/web/libraries/maplibre-gl/src/render/image_manager.ts new file mode 100644 index 00000000..33a0d1aa --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/image_manager.ts @@ -0,0 +1,328 @@ +/* eslint-disable key-spacing */ +import potpack from 'potpack'; + +import {Event, ErrorEvent, Evented} from '../util/evented'; +import {RGBAImage} from '../util/image'; +import {ImagePosition} from './image_atlas'; +import {Texture} from './texture'; +import {renderStyleImage} from '../style/style_image'; +import {warnOnce} from '../util/util'; + +import type {StyleImage} from '../style/style_image'; +import type {Context} from '../gl/context'; +import type {PotpackBox} from 'potpack'; +import type {Callback} from '../types/callback'; + +type Pattern = { + bin: PotpackBox; + position: ImagePosition; +}; + +// When copied into the atlas texture, image data is padded by one pixel on each side. Icon +// images are padded with fully transparent pixels, while pattern images are padded with a +// copy of the image data wrapped from the opposite side. In both cases, this ensures the +// correct behavior of GL_LINEAR texture sampling mode. +const padding = 1; + +/* + ImageManager does three things: + + 1. Tracks requests for icon images from tile workers and sends responses when the requests are fulfilled. + 2. Builds a texture atlas for pattern images. + 3. Rerenders renderable images once per frame + + These are disparate responsibilities and should eventually be handled by different classes. When we implement + data-driven support for `*-pattern`, we'll likely use per-bucket pattern atlases, and that would be a good time + to refactor this. +*/ +export class ImageManager extends Evented { + images: {[_: string]: StyleImage}; + updatedImages: {[_: string]: boolean}; + callbackDispatchedThisFrame: {[_: string]: boolean}; + loaded: boolean; + requestors: Array<{ + ids: Array; + callback: Callback<{[_: string]: StyleImage}>; + }>; + + patterns: {[_: string]: Pattern}; + atlasImage: RGBAImage; + atlasTexture: Texture; + dirty: boolean; + + constructor() { + super(); + this.images = {}; + this.updatedImages = {}; + this.callbackDispatchedThisFrame = {}; + this.loaded = false; + this.requestors = []; + + this.patterns = {}; + this.atlasImage = new RGBAImage({width: 1, height: 1}); + this.dirty = true; + } + + isLoaded() { + return this.loaded; + } + + setLoaded(loaded: boolean) { + if (this.loaded === loaded) { + return; + } + + this.loaded = loaded; + + if (loaded) { + for (const {ids, callback} of this.requestors) { + this._notify(ids, callback); + } + this.requestors = []; + } + } + + getImage(id: string): StyleImage { + const image = this.images[id]; + + // Extract sprite image data on demand + if (image && !image.data && image.spriteData) { + const spriteData = image.spriteData; + image.data = new RGBAImage({ + width: spriteData.width, + height: spriteData.height + }, spriteData.context.getImageData( + spriteData.x, + spriteData.y, + spriteData.width, + spriteData.height).data); + image.spriteData = null; + } + + return image; + } + + addImage(id: string, image: StyleImage) { + if (this.images[id]) throw new Error(`Image id ${id} already exist, use updateImage instead`); + if (this._validate(id, image)) { + this.images[id] = image; + } + } + + _validate(id: string, image: StyleImage) { + let valid = true; + const data = image.data || image.spriteData; + if (!this._validateStretch(image.stretchX, data && data.width)) { + this.fire(new ErrorEvent(new Error(`Image "${id}" has invalid "stretchX" value`))); + valid = false; + } + if (!this._validateStretch(image.stretchY, data && data.height)) { + this.fire(new ErrorEvent(new Error(`Image "${id}" has invalid "stretchY" value`))); + valid = false; + } + if (!this._validateContent(image.content, image)) { + this.fire(new ErrorEvent(new Error(`Image "${id}" has invalid "content" value`))); + valid = false; + } + return valid; + } + + _validateStretch(stretch: Array<[number, number]>, size: number) { + if (!stretch) return true; + let last = 0; + for (const part of stretch) { + if (part[0] < last || part[1] < part[0] || size < part[1]) return false; + last = part[1]; + } + return true; + } + + _validateContent(content: [number, number, number, number], image: StyleImage) { + if (!content) return true; + if (content.length !== 4) return false; + const spriteData = image.spriteData; + const width = (spriteData && spriteData.width) || image.data.width; + const height = (spriteData && spriteData.height) || image.data.height; + if (content[0] < 0 || width < content[0]) return false; + if (content[1] < 0 || height < content[1]) return false; + if (content[2] < 0 || width < content[2]) return false; + if (content[3] < 0 || height < content[3]) return false; + if (content[2] < content[0]) return false; + if (content[3] < content[1]) return false; + return true; + } + + updateImage(id: string, image: StyleImage, validate = true) { + const oldImage = this.getImage(id); + if (validate && (oldImage.data.width !== image.data.width || oldImage.data.height !== image.data.height)) { + throw new Error(`size mismatch between old image (${oldImage.data.width}x${oldImage.data.height}) and new image (${image.data.width}x${image.data.height}).`); + } + image.version = oldImage.version + 1; + this.images[id] = image; + this.updatedImages[id] = true; + } + + removeImage(id: string) { + const image = this.images[id]; + delete this.images[id]; + delete this.patterns[id]; + + if (image.userImage && image.userImage.onRemove) { + image.userImage.onRemove(); + } + } + + listImages(): Array { + return Object.keys(this.images); + } + + getImages(ids: Array, callback: Callback<{[_: string]: StyleImage}>) { + // If the sprite has been loaded, or if all the icon dependencies are already present + // (i.e. if they've been added via runtime styling), then notify the requestor immediately. + // Otherwise, delay notification until the sprite is loaded. At that point, if any of the + // dependencies are still unavailable, we'll just assume they are permanently missing. + let hasAllDependencies = true; + if (!this.isLoaded()) { + for (const id of ids) { + if (!this.images[id]) { + hasAllDependencies = false; + } + } + } + if (this.isLoaded() || hasAllDependencies) { + this._notify(ids, callback); + } else { + this.requestors.push({ids, callback}); + } + } + + _notify(ids: Array, callback: Callback<{[_: string]: StyleImage}>) { + const response = {}; + + for (const id of ids) { + let image = this.getImage(id); + + if (!image) { + this.fire(new Event('styleimagemissing', {id})); + //Try to acquire image again in case styleimagemissing has populated it + image = this.getImage(id); + } + + if (image) { + // Clone the image so that our own copy of its ArrayBuffer doesn't get transferred. + response[id] = { + data: image.data.clone(), + pixelRatio: image.pixelRatio, + sdf: image.sdf, + version: image.version, + stretchX: image.stretchX, + stretchY: image.stretchY, + content: image.content, + hasRenderCallback: Boolean(image.userImage && image.userImage.render) + }; + } else { + warnOnce(`Image "${id}" could not be loaded. Please make sure you have added the image with map.addImage() or a "sprite" property in your style. You can provide missing images by listening for the "styleimagemissing" map event.`); + } + } + + callback(null, response); + } + + // Pattern stuff + + getPixelSize() { + const {width, height} = this.atlasImage; + return {width, height}; + } + + getPattern(id: string): ImagePosition { + const pattern = this.patterns[id]; + + const image = this.getImage(id); + if (!image) { + return null; + } + + if (pattern && pattern.position.version === image.version) { + return pattern.position; + } + + if (!pattern) { + const w = image.data.width + padding * 2; + const h = image.data.height + padding * 2; + const bin = {w, h, x: 0, y: 0}; + const position = new ImagePosition(bin, image); + this.patterns[id] = {bin, position}; + } else { + pattern.position.version = image.version; + } + + this._updatePatternAtlas(); + + return this.patterns[id].position; + } + + bind(context: Context) { + const gl = context.gl; + if (!this.atlasTexture) { + this.atlasTexture = new Texture(context, this.atlasImage, gl.RGBA); + } else if (this.dirty) { + this.atlasTexture.update(this.atlasImage); + this.dirty = false; + } + + this.atlasTexture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + } + + _updatePatternAtlas() { + const bins = []; + for (const id in this.patterns) { + bins.push(this.patterns[id].bin); + } + + const {w, h} = potpack(bins); + + const dst = this.atlasImage; + dst.resize({width: w || 1, height: h || 1}); + + for (const id in this.patterns) { + const {bin} = this.patterns[id]; + const x = bin.x + padding; + const y = bin.y + padding; + const src = this.getImage(id).data; + const w = src.width; + const h = src.height; + + RGBAImage.copy(src, dst, {x: 0, y: 0}, {x, y}, {width: w, height: h}); + + // Add 1 pixel wrapped padding on each side of the image. + RGBAImage.copy(src, dst, {x: 0, y: h - 1}, {x, y: y - 1}, {width: w, height: 1}); // T + RGBAImage.copy(src, dst, {x: 0, y: 0}, {x, y: y + h}, {width: w, height: 1}); // B + RGBAImage.copy(src, dst, {x: w - 1, y: 0}, {x: x - 1, y}, {width: 1, height: h}); // L + RGBAImage.copy(src, dst, {x: 0, y: 0}, {x: x + w, y}, {width: 1, height: h}); // R + } + + this.dirty = true; + } + + beginFrame() { + this.callbackDispatchedThisFrame = {}; + } + + dispatchRenderCallbacks(ids: Array) { + for (const id of ids) { + + // the callback for the image was already dispatched for a different frame + if (this.callbackDispatchedThisFrame[id]) continue; + this.callbackDispatchedThisFrame[id] = true; + + const image = this.getImage(id); + if (!image) warnOnce(`Image with ID: "${id}" was not found`); + + const updated = renderStyleImage(image); + if (updated) { + this.updateImage(id, image); + } + } + } +} diff --git a/web/libraries/maplibre-gl/src/render/line_atlas.test.ts b/web/libraries/maplibre-gl/src/render/line_atlas.test.ts new file mode 100644 index 00000000..a672500c --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/line_atlas.test.ts @@ -0,0 +1,38 @@ +import {LineAtlas} from './line_atlas'; + +describe('LineAtlas', () => { + const lineAtlas = new LineAtlas(64, 64); + test('round [0, 0]', () => { + const entry = lineAtlas.addDash([0, 0], true); + expect(entry.width).toBe(0); + }); + test('round [1, 0]', () => { + const entry = lineAtlas.addDash([1, 0], true); + expect(entry.width).toBe(1); + }); + test('round [0, 1]', () => { + const entry = lineAtlas.addDash([0, 1], true); + expect(entry.width).toBe(1); + }); + test('odd round [1, 2, 1]', () => { + const entry = lineAtlas.addDash([1, 2, 1], true); + expect(entry.width).toBe(4); + }); + + test('regular [0, 0]', () => { + const entry = lineAtlas.addDash([0, 0], false); + expect(entry.width).toBe(0); + }); + test('regular [1, 0]', () => { + const entry = lineAtlas.addDash([1, 0], false); + expect(entry.width).toBe(1); + }); + test('regular [0, 1]', () => { + const entry = lineAtlas.addDash([0, 1], false); + expect(entry.width).toBe(1); + }); + test('odd regular [1, 2, 1]', () => { + const entry = lineAtlas.addDash([1, 2, 1], false); + expect(entry.width).toBe(4); + }); +}); diff --git a/web/libraries/maplibre-gl/src/render/line_atlas.ts b/web/libraries/maplibre-gl/src/render/line_atlas.ts new file mode 100644 index 00000000..56cead59 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/line_atlas.ts @@ -0,0 +1,214 @@ +import {warnOnce} from '../util/util'; + +import type {Context} from '../gl/context'; + +/** + * A dash entry + */ +type DashEntry = { + y: number; + height: number; + width: number; +} + +/** + * @internal + * A LineAtlas lets us reuse rendered dashed lines + * by writing many of them to a texture and then fetching their positions + * using {@link LineAtlas#getDash}. + * + * @param width - the width + * @param height - the height + */ +export class LineAtlas { + width: number; + height: number; + nextRow: number; + bytes: number; + data: Uint8Array; + dashEntry: {[_: string]: DashEntry}; + dirty: boolean; + texture: WebGLTexture; + + constructor(width: number, height: number) { + this.width = width; + this.height = height; + this.nextRow = 0; + + this.data = new Uint8Array(this.width * this.height); + + this.dashEntry = {}; + } + + /** + * Get or create a dash line pattern. + * + * @param dasharray - the key (represented by numbers) to get the dash texture + * @param round - whether to add circle caps in between dash segments + * @returns position of dash texture in {@link DashEntry} + */ + getDash(dasharray: Array, round: boolean) { + const key = dasharray.join(',') + String(round); + + if (!this.dashEntry[key]) { + this.dashEntry[key] = this.addDash(dasharray, round); + } + return this.dashEntry[key]; + } + + getDashRanges(dasharray: Array, lineAtlasWidth: number, stretch: number) { + // If dasharray has an odd length, both the first and last parts + // are dashes and should be joined seamlessly. + const oddDashArray = dasharray.length % 2 === 1; + + const ranges = []; + + let left = oddDashArray ? -dasharray[dasharray.length - 1] * stretch : 0; + let right = dasharray[0] * stretch; + let isDash = true; + + ranges.push({left, right, isDash, zeroLength: dasharray[0] === 0}); + + let currentDashLength = dasharray[0]; + for (let i = 1; i < dasharray.length; i++) { + isDash = !isDash; + + const dashLength = dasharray[i]; + left = currentDashLength * stretch; + currentDashLength += dashLength; + right = currentDashLength * stretch; + + ranges.push({left, right, isDash, zeroLength: dashLength === 0}); + } + + return ranges; + } + + addRoundDash(ranges: any, stretch: number, n: number) { + const halfStretch = stretch / 2; + + for (let y = -n; y <= n; y++) { + const row = this.nextRow + n + y; + const index = this.width * row; + let currIndex = 0; + let range = ranges[currIndex]; + + for (let x = 0; x < this.width; x++) { + if (x / range.right > 1) { range = ranges[++currIndex]; } + + const distLeft = Math.abs(x - range.left); + const distRight = Math.abs(x - range.right); + const minDist = Math.min(distLeft, distRight); + let signedDistance; + + const distMiddle = y / n * (halfStretch + 1); + if (range.isDash) { + const distEdge = halfStretch - Math.abs(distMiddle); + signedDistance = Math.sqrt(minDist * minDist + distEdge * distEdge); + } else { + signedDistance = halfStretch - Math.sqrt(minDist * minDist + distMiddle * distMiddle); + } + + this.data[index + x] = Math.max(0, Math.min(255, signedDistance + 128)); + } + } + } + + addRegularDash(ranges: any) { + + // Collapse any zero-length range + // Collapse neighbouring same-type parts into a single part + for (let i = ranges.length - 1; i >= 0; --i) { + const part = ranges[i]; + const next = ranges[i + 1]; + if (part.zeroLength) { + ranges.splice(i, 1); + } else if (next && next.isDash === part.isDash) { + next.left = part.left; + ranges.splice(i, 1); + } + } + + // Combine the first and last parts if possible + const first = ranges[0]; + const last = ranges[ranges.length - 1]; + if (first.isDash === last.isDash) { + first.left = last.left - this.width; + last.right = first.right + this.width; + } + + const index = this.width * this.nextRow; + let currIndex = 0; + let range = ranges[currIndex]; + + for (let x = 0; x < this.width; x++) { + if (x / range.right > 1) { + range = ranges[++currIndex]; + } + + const distLeft = Math.abs(x - range.left); + const distRight = Math.abs(x - range.right); + + const minDist = Math.min(distLeft, distRight); + const signedDistance = range.isDash ? minDist : -minDist; + + this.data[index + x] = Math.max(0, Math.min(255, signedDistance + 128)); + } + } + + addDash(dasharray: Array, round: boolean): DashEntry { + const n = round ? 7 : 0; + const height = 2 * n + 1; + + if (this.nextRow + height > this.height) { + warnOnce('LineAtlas out of space'); + return null; + } + + let length = 0; + for (let i = 0; i < dasharray.length; i++) { length += dasharray[i]; } + + if (length !== 0) { + const stretch = this.width / length; + const ranges = this.getDashRanges(dasharray, this.width, stretch); + + if (round) { + this.addRoundDash(ranges, stretch, n); + } else { + this.addRegularDash(ranges); + } + } + + const dashEntry = { + y: (this.nextRow + n + 0.5) / this.height, + height: 2 * n / this.height, + width: length + }; + + this.nextRow += height; + this.dirty = true; + + return dashEntry; + } + + bind(context: Context) { + const gl = context.gl; + if (!this.texture) { + this.texture = gl.createTexture(); + gl.bindTexture(gl.TEXTURE_2D, this.texture); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.REPEAT); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.REPEAT); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR); + gl.texImage2D(gl.TEXTURE_2D, 0, gl.ALPHA, this.width, this.height, 0, gl.ALPHA, gl.UNSIGNED_BYTE, this.data); + + } else { + gl.bindTexture(gl.TEXTURE_2D, this.texture); + + if (this.dirty) { + this.dirty = false; + gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, this.width, this.height, gl.ALPHA, gl.UNSIGNED_BYTE, this.data); + } + } + } +} diff --git a/web/libraries/maplibre-gl/src/render/painter.ts b/web/libraries/maplibre-gl/src/render/painter.ts new file mode 100644 index 00000000..89184b29 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/painter.ts @@ -0,0 +1,642 @@ +import {browser} from '../util/browser'; +import {mat4, vec3} from 'gl-matrix'; +import {SourceCache} from '../source/source_cache'; +import {EXTENT} from '../data/extent'; +import {pixelsToTileUnits} from '../source/pixels_to_tile_units'; +import {SegmentVector} from '../data/segment'; +import {RasterBoundsArray, PosArray, TriangleIndexArray, LineStripIndexArray} from '../data/array_types.g'; +import rasterBoundsAttributes from '../data/raster_bounds_attributes'; +import posAttributes from '../data/pos_attributes'; +import {ProgramConfiguration} from '../data/program_configuration'; +import {CrossTileSymbolIndex} from '../symbol/cross_tile_symbol_index'; +import {shaders} from '../shaders/shaders'; +import {Program} from './program'; +import {programUniforms} from './program/program_uniforms'; +import {Context} from '../gl/context'; +import {DepthMode} from '../gl/depth_mode'; +import {StencilMode} from '../gl/stencil_mode'; +import {ColorMode} from '../gl/color_mode'; +import {CullFaceMode} from '../gl/cull_face_mode'; +import {Texture} from './texture'; +import {clippingMaskUniformValues} from './program/clipping_mask_program'; +import {Color} from '@maplibre/maplibre-gl-style-spec'; +import {drawSymbols} from './draw_symbol'; +import {drawCircles} from './draw_circle'; +import {drawHeatmap} from './draw_heatmap'; +import {drawLine} from './draw_line'; +import {drawFill} from './draw_fill'; +import {drawFillExtrusion} from './draw_fill_extrusion'; +import {drawHillshade} from './draw_hillshade'; +import {drawRaster} from './draw_raster'; +import {drawBackground} from './draw_background'; +import {drawDebug, drawDebugPadding, selectDebugSource} from './draw_debug'; +import {drawCustom} from './draw_custom'; +import {drawDepth, drawCoords} from './draw_terrain'; +import {OverscaledTileID} from '../source/tile_id'; + +import type {Transform} from '../geo/transform'; +import type {Tile} from '../source/tile'; +import type {Style} from '../style/style'; +import type {StyleLayer} from '../style/style_layer'; +import type {CrossFaded} from '../style/properties'; +import type {LineAtlas} from './line_atlas'; +import type {ImageManager} from './image_manager'; +import type {GlyphManager} from './glyph_manager'; +import type {VertexBuffer} from '../gl/vertex_buffer'; +import type {IndexBuffer} from '../gl/index_buffer'; +import type {DepthRangeType, DepthMaskType, DepthFuncType} from '../gl/types'; +import type {ResolvedImage} from '@maplibre/maplibre-gl-style-spec'; +import {RenderToTexture} from './render_to_texture'; + +export type RenderPass = 'offscreen' | 'opaque' | 'translucent'; + +type PainterOptions = { + showOverdrawInspector: boolean; + showTileBoundaries: boolean; + showPadding: boolean; + rotating: boolean; + zooming: boolean; + moving: boolean; + fadeDuration: number; +}; + +/** + * @internal + * Initialize a new painter object. + */ +export class Painter { + context: Context; + transform: Transform; + renderToTexture: RenderToTexture; + _tileTextures: { + [_: number]: Array; + }; + numSublayers: number; + depthEpsilon: number; + emptyProgramConfiguration: ProgramConfiguration; + width: number; + height: number; + pixelRatio: number; + tileExtentBuffer: VertexBuffer; + tileExtentSegments: SegmentVector; + debugBuffer: VertexBuffer; + debugSegments: SegmentVector; + rasterBoundsBuffer: VertexBuffer; + rasterBoundsSegments: SegmentVector; + viewportBuffer: VertexBuffer; + viewportSegments: SegmentVector; + quadTriangleIndexBuffer: IndexBuffer; + tileBorderIndexBuffer: IndexBuffer; + _tileClippingMaskIDs: {[_: string]: number}; + stencilClearMode: StencilMode; + style: Style; + options: PainterOptions; + lineAtlas: LineAtlas; + imageManager: ImageManager; + glyphManager: GlyphManager; + depthRangeFor3D: DepthRangeType; + opaquePassCutoff: number; + renderPass: RenderPass; + currentLayer: number; + currentStencilSource: string; + nextStencilID: number; + id: string; + _showOverdrawInspector: boolean; + cache: {[_: string]: Program}; + crossTileSymbolIndex: CrossTileSymbolIndex; + symbolFadeChange: number; + debugOverlayTexture: Texture; + debugOverlayCanvas: HTMLCanvasElement; + // this object stores the current camera-matrix and the last render time + // of the terrain-facilitators. e.g. depth & coords framebuffers + // every time the camera-matrix changes the terrain-facilitators will be redrawn. + terrainFacilitator: {dirty: boolean; matrix: mat4; renderTime: number}; + + constructor(gl: WebGLRenderingContext | WebGL2RenderingContext, transform: Transform) { + this.context = new Context(gl); + this.transform = transform; + this._tileTextures = {}; + this.terrainFacilitator = {dirty: true, matrix: mat4.create(), renderTime: 0}; + + this.setup(); + + // Within each layer there are multiple distinct z-planes that can be drawn to. + // This is implemented using the WebGL depth buffer. + this.numSublayers = SourceCache.maxUnderzooming + SourceCache.maxOverzooming + 1; + this.depthEpsilon = 1 / Math.pow(2, 16); + + this.crossTileSymbolIndex = new CrossTileSymbolIndex(); + } + + /* + * Update the GL viewport, projection matrix, and transforms to compensate + * for a new width and height value. + */ + resize(width: number, height: number, pixelRatio: number) { + this.width = Math.floor(width * pixelRatio); + this.height = Math.floor(height * pixelRatio); + this.pixelRatio = pixelRatio; + this.context.viewport.set([0, 0, this.width, this.height]); + + if (this.style) { + for (const layerId of this.style._order) { + this.style._layers[layerId].resize(); + } + } + } + + setup() { + const context = this.context; + + const tileExtentArray = new PosArray(); + tileExtentArray.emplaceBack(0, 0); + tileExtentArray.emplaceBack(EXTENT, 0); + tileExtentArray.emplaceBack(0, EXTENT); + tileExtentArray.emplaceBack(EXTENT, EXTENT); + this.tileExtentBuffer = context.createVertexBuffer(tileExtentArray, posAttributes.members); + this.tileExtentSegments = SegmentVector.simpleSegment(0, 0, 4, 2); + + const debugArray = new PosArray(); + debugArray.emplaceBack(0, 0); + debugArray.emplaceBack(EXTENT, 0); + debugArray.emplaceBack(0, EXTENT); + debugArray.emplaceBack(EXTENT, EXTENT); + this.debugBuffer = context.createVertexBuffer(debugArray, posAttributes.members); + this.debugSegments = SegmentVector.simpleSegment(0, 0, 4, 5); + + const rasterBoundsArray = new RasterBoundsArray(); + rasterBoundsArray.emplaceBack(0, 0, 0, 0); + rasterBoundsArray.emplaceBack(EXTENT, 0, EXTENT, 0); + rasterBoundsArray.emplaceBack(0, EXTENT, 0, EXTENT); + rasterBoundsArray.emplaceBack(EXTENT, EXTENT, EXTENT, EXTENT); + this.rasterBoundsBuffer = context.createVertexBuffer(rasterBoundsArray, rasterBoundsAttributes.members); + this.rasterBoundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2); + + const viewportArray = new PosArray(); + viewportArray.emplaceBack(0, 0); + viewportArray.emplaceBack(1, 0); + viewportArray.emplaceBack(0, 1); + viewportArray.emplaceBack(1, 1); + this.viewportBuffer = context.createVertexBuffer(viewportArray, posAttributes.members); + this.viewportSegments = SegmentVector.simpleSegment(0, 0, 4, 2); + + const tileLineStripIndices = new LineStripIndexArray(); + tileLineStripIndices.emplaceBack(0); + tileLineStripIndices.emplaceBack(1); + tileLineStripIndices.emplaceBack(3); + tileLineStripIndices.emplaceBack(2); + tileLineStripIndices.emplaceBack(0); + this.tileBorderIndexBuffer = context.createIndexBuffer(tileLineStripIndices); + + const quadTriangleIndices = new TriangleIndexArray(); + quadTriangleIndices.emplaceBack(0, 1, 2); + quadTriangleIndices.emplaceBack(2, 1, 3); + this.quadTriangleIndexBuffer = context.createIndexBuffer(quadTriangleIndices); + + const gl = this.context.gl; + this.stencilClearMode = new StencilMode({func: gl.ALWAYS, mask: 0}, 0x0, 0xFF, gl.ZERO, gl.ZERO, gl.ZERO); + } + + /* + * Reset the drawing canvas by clearing the stencil buffer so that we can draw + * new tiles at the same location, while retaining previously drawn pixels. + */ + clearStencil() { + const context = this.context; + const gl = context.gl; + + this.nextStencilID = 1; + this.currentStencilSource = undefined; + + // As a temporary workaround for https://github.com/mapbox/mapbox-gl-js/issues/5490, + // pending an upstream fix, we draw a fullscreen stencil=0 clipping mask here, + // effectively clearing the stencil buffer: once an upstream patch lands, remove + // this function in favor of context.clear({ stencil: 0x0 }) + + const matrix = mat4.create(); + mat4.ortho(matrix, 0, this.width, this.height, 0, 0, 1); + mat4.scale(matrix, matrix, [gl.drawingBufferWidth, gl.drawingBufferHeight, 0]); + + this.useProgram('clippingMask').draw(context, gl.TRIANGLES, + DepthMode.disabled, this.stencilClearMode, ColorMode.disabled, CullFaceMode.disabled, + clippingMaskUniformValues(matrix), null, + '$clipping', this.viewportBuffer, + this.quadTriangleIndexBuffer, this.viewportSegments); + } + + _renderTileClippingMasks(layer: StyleLayer, tileIDs: Array) { + if (this.currentStencilSource === layer.source || !layer.isTileClipped() || !tileIDs || !tileIDs.length) return; + + this.currentStencilSource = layer.source; + + const context = this.context; + const gl = context.gl; + + if (this.nextStencilID + tileIDs.length > 256) { + // we'll run out of fresh IDs so we need to clear and start from scratch + this.clearStencil(); + } + + context.setColorMode(ColorMode.disabled); + context.setDepthMode(DepthMode.disabled); + + const program = this.useProgram('clippingMask'); + + this._tileClippingMaskIDs = {}; + + for (const tileID of tileIDs) { + const id = this._tileClippingMaskIDs[tileID.key] = this.nextStencilID++; + const terrainData = this.style.map.terrain && this.style.map.terrain.getTerrainData(tileID); + + program.draw(context, gl.TRIANGLES, DepthMode.disabled, + // Tests will always pass, and ref value will be written to stencil buffer. + new StencilMode({func: gl.ALWAYS, mask: 0}, id, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE), + ColorMode.disabled, CullFaceMode.disabled, clippingMaskUniformValues(tileID.posMatrix), + terrainData, '$clipping', this.tileExtentBuffer, + this.quadTriangleIndexBuffer, this.tileExtentSegments); + } + } + + stencilModeFor3D(): StencilMode { + this.currentStencilSource = undefined; + + if (this.nextStencilID + 1 > 256) { + this.clearStencil(); + } + + const id = this.nextStencilID++; + const gl = this.context.gl; + return new StencilMode({func: gl.NOTEQUAL, mask: 0xFF}, id, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE); + } + + stencilModeForClipping(tileID: OverscaledTileID): StencilMode { + const gl = this.context.gl; + return new StencilMode({func: gl.EQUAL, mask: 0xFF}, this._tileClippingMaskIDs[tileID.key], 0x00, gl.KEEP, gl.KEEP, gl.REPLACE); + } + + /* + * Sort coordinates by Z as drawing tiles is done in Z-descending order. + * All children with the same Z write the same stencil value. Children + * stencil values are greater than parent's. This is used only for raster + * and raster-dem tiles, which are already clipped to tile boundaries, to + * mask area of tile overlapped by children tiles. + * Stencil ref values continue range used in _tileClippingMaskIDs. + * + * Returns [StencilMode for tile overscaleZ map, sortedCoords]. + */ + stencilConfigForOverlap(tileIDs: Array): [{ + [_: number]: Readonly; + }, Array] { + const gl = this.context.gl; + const coords = tileIDs.sort((a, b) => b.overscaledZ - a.overscaledZ); + const minTileZ = coords[coords.length - 1].overscaledZ; + const stencilValues = coords[0].overscaledZ - minTileZ + 1; + if (stencilValues > 1) { + this.currentStencilSource = undefined; + if (this.nextStencilID + stencilValues > 256) { + this.clearStencil(); + } + const zToStencilMode = {}; + for (let i = 0; i < stencilValues; i++) { + zToStencilMode[i + minTileZ] = new StencilMode({func: gl.GEQUAL, mask: 0xFF}, i + this.nextStencilID, 0xFF, gl.KEEP, gl.KEEP, gl.REPLACE); + } + this.nextStencilID += stencilValues; + return [zToStencilMode, coords]; + } + return [{[minTileZ]: StencilMode.disabled}, coords]; + } + + colorModeForRenderPass(): Readonly { + const gl = this.context.gl; + if (this._showOverdrawInspector) { + const numOverdrawSteps = 8; + const a = 1 / numOverdrawSteps; + + return new ColorMode([gl.CONSTANT_COLOR, gl.ONE], new Color(a, a, a, 0), [true, true, true, true]); + } else if (this.renderPass === 'opaque') { + return ColorMode.unblended; + } else { + return ColorMode.alphaBlended; + } + } + + depthModeForSublayer(n: number, mask: DepthMaskType, func?: DepthFuncType | null): Readonly { + if (!this.opaquePassEnabledForLayer()) return DepthMode.disabled; + const depth = 1 - ((1 + this.currentLayer) * this.numSublayers + n) * this.depthEpsilon; + return new DepthMode(func || this.context.gl.LEQUAL, mask, [depth, depth]); + } + + /* + * The opaque pass and 3D layers both use the depth buffer. + * Layers drawn above 3D layers need to be drawn using the + * painter's algorithm so that they appear above 3D features. + * This returns true for layers that can be drawn using the + * opaque pass. + */ + opaquePassEnabledForLayer() { + return this.currentLayer < this.opaquePassCutoff; + } + + render(style: Style, options: PainterOptions) { + this.style = style; + this.options = options; + + this.lineAtlas = style.lineAtlas; + this.imageManager = style.imageManager; + this.glyphManager = style.glyphManager; + + this.symbolFadeChange = style.placement.symbolFadeChange(browser.now()); + + this.imageManager.beginFrame(); + + const layerIds = this.style._order; + const sourceCaches = this.style.sourceCaches; + + const coordsAscending: {[_: string]: Array} = {}; + const coordsDescending: {[_: string]: Array} = {}; + const coordsDescendingSymbol: {[_: string]: Array} = {}; + + for (const id in sourceCaches) { + const sourceCache = sourceCaches[id]; + if (sourceCache.used) { + sourceCache.prepare(this.context); + } + + coordsAscending[id] = sourceCache.getVisibleCoordinates(); + coordsDescending[id] = coordsAscending[id].slice().reverse(); + coordsDescendingSymbol[id] = sourceCache.getVisibleCoordinates(true).reverse(); + } + + this.opaquePassCutoff = Infinity; + for (let i = 0; i < layerIds.length; i++) { + const layerId = layerIds[i]; + if (this.style._layers[layerId].is3D()) { + this.opaquePassCutoff = i; + break; + } + } + + if (this.renderToTexture) { + this.renderToTexture.prepareForRender(this.style, this.transform.zoom); + // this is disabled, because render-to-texture is rendering all layers from bottom to top. + this.opaquePassCutoff = 0; + + // update coords/depth-framebuffer on camera movement, or tile reloading + const newTiles = this.style.map.terrain.sourceCache.tilesAfterTime(this.terrainFacilitator.renderTime); + if (this.terrainFacilitator.dirty || !mat4.equals(this.terrainFacilitator.matrix, this.transform.projMatrix) || newTiles.length) { + mat4.copy(this.terrainFacilitator.matrix, this.transform.projMatrix); + this.terrainFacilitator.renderTime = Date.now(); + this.terrainFacilitator.dirty = false; + drawDepth(this, this.style.map.terrain); + drawCoords(this, this.style.map.terrain); + } + } + + // Offscreen pass =============================================== + // We first do all rendering that requires rendering to a separate + // framebuffer, and then save those for rendering back to the map + // later: in doing this we avoid doing expensive framebuffer restores. + this.renderPass = 'offscreen'; + + for (const layerId of layerIds) { + const layer = this.style._layers[layerId]; + if (!layer.hasOffscreenPass() || layer.isHidden(this.transform.zoom)) continue; + + const coords = coordsDescending[layer.source]; + if (layer.type !== 'custom' && !coords.length) continue; + + this.renderLayer(this, sourceCaches[layer.source], layer, coords); + } + + // Rebind the main framebuffer now that all offscreen layers have been rendered: + this.context.bindFramebuffer.set(null); + + // Clear buffers in preparation for drawing to the main framebuffer + this.context.clear({color: options.showOverdrawInspector ? Color.black : Color.transparent, depth: 1}); + this.clearStencil(); + + this._showOverdrawInspector = options.showOverdrawInspector; + this.depthRangeFor3D = [0, 1 - ((style._order.length + 2) * this.numSublayers * this.depthEpsilon)]; + + // Opaque pass =============================================== + // Draw opaque layers top-to-bottom first. + if (!this.renderToTexture) { + this.renderPass = 'opaque'; + + for (this.currentLayer = layerIds.length - 1; this.currentLayer >= 0; this.currentLayer--) { + const layer = this.style._layers[layerIds[this.currentLayer]]; + const sourceCache = sourceCaches[layer.source]; + const coords = coordsAscending[layer.source]; + + this._renderTileClippingMasks(layer, coords); + this.renderLayer(this, sourceCache, layer, coords); + } + } + + // Translucent pass =============================================== + // Draw all other layers bottom-to-top. + this.renderPass = 'translucent'; + + for (this.currentLayer = 0; this.currentLayer < layerIds.length; this.currentLayer++) { + const layer = this.style._layers[layerIds[this.currentLayer]]; + const sourceCache = sourceCaches[layer.source]; + + if (this.renderToTexture && this.renderToTexture.renderLayer(layer)) continue; + + // For symbol layers in the translucent pass, we add extra tiles to the renderable set + // for cross-tile symbol fading. Symbol layers don't use tile clipping, so no need to render + // separate clipping masks + const coords = (layer.type === 'symbol' ? coordsDescendingSymbol : coordsDescending)[layer.source]; + + this._renderTileClippingMasks(layer, coordsAscending[layer.source]); + this.renderLayer(this, sourceCache, layer, coords); + } + + if (this.options.showTileBoundaries) { + const selectedSource = selectDebugSource(this.style, this.transform.zoom); + if (selectedSource) { + drawDebug(this, selectedSource, selectedSource.getVisibleCoordinates()); + } + } + + if (this.options.showPadding) { + drawDebugPadding(this); + } + + // Set defaults for most GL values so that anyone using the state after the render + // encounters more expected values. + this.context.setDefault(); + } + + renderLayer(painter: Painter, sourceCache: SourceCache, layer: StyleLayer, coords: Array) { + if (layer.isHidden(this.transform.zoom)) return; + if (layer.type !== 'background' && layer.type !== 'custom' && !(coords || []).length) return; + this.id = layer.id; + + switch (layer.type) { + case 'symbol': + drawSymbols(painter, sourceCache, layer as any, coords, this.style.placement.variableOffsets); + break; + case 'circle': + drawCircles(painter, sourceCache, layer as any, coords); + break; + case 'heatmap': + drawHeatmap(painter, sourceCache, layer as any, coords); + break; + case 'line': + drawLine(painter, sourceCache, layer as any, coords); + break; + case 'fill': + drawFill(painter, sourceCache, layer as any, coords); + break; + case 'fill-extrusion': + drawFillExtrusion(painter, sourceCache, layer as any, coords); + break; + case 'hillshade': + drawHillshade(painter, sourceCache, layer as any, coords); + break; + case 'raster': + drawRaster(painter, sourceCache, layer as any, coords); + break; + case 'background': + drawBackground(painter, sourceCache, layer as any, coords); + break; + case 'custom': + drawCustom(painter, sourceCache, layer as any); + break; + } + } + + /** + * Transform a matrix to incorporate the *-translate and *-translate-anchor properties into it. + * @param inViewportPixelUnitsUnits - True when the units accepted by the matrix are in viewport pixels instead of tile units. + * @returns matrix + */ + translatePosMatrix(matrix: mat4, tile: Tile, translate: [number, number], translateAnchor: 'map' | 'viewport', inViewportPixelUnitsUnits?: boolean): mat4 { + if (!translate[0] && !translate[1]) return matrix; + + const angle = inViewportPixelUnitsUnits ? + (translateAnchor === 'map' ? this.transform.angle : 0) : + (translateAnchor === 'viewport' ? -this.transform.angle : 0); + + if (angle) { + const sinA = Math.sin(angle); + const cosA = Math.cos(angle); + translate = [ + translate[0] * cosA - translate[1] * sinA, + translate[0] * sinA + translate[1] * cosA + ]; + } + + const translation = [ + inViewportPixelUnitsUnits ? translate[0] : pixelsToTileUnits(tile, translate[0], this.transform.zoom), + inViewportPixelUnitsUnits ? translate[1] : pixelsToTileUnits(tile, translate[1], this.transform.zoom), + 0 + ] as vec3; + + const translatedMatrix = new Float32Array(16); + mat4.translate(translatedMatrix, matrix, translation); + return translatedMatrix; + } + + saveTileTexture(texture: Texture) { + const textures = this._tileTextures[texture.size[0]]; + if (!textures) { + this._tileTextures[texture.size[0]] = [texture]; + } else { + textures.push(texture); + } + } + + getTileTexture(size: number) { + const textures = this._tileTextures[size]; + return textures && textures.length > 0 ? textures.pop() : null; + } + + /** + * Checks whether a pattern image is needed, and if it is, whether it is not loaded. + * + * @returns true if a needed image is missing and rendering needs to be skipped. + */ + isPatternMissing(image?: CrossFaded | null): boolean { + if (!image) return false; + if (!image.from || !image.to) return true; + const imagePosA = this.imageManager.getPattern(image.from.toString()); + const imagePosB = this.imageManager.getPattern(image.to.toString()); + return !imagePosA || !imagePosB; + } + + useProgram(name: string, programConfiguration?: ProgramConfiguration | null): Program { + this.cache = this.cache || {}; + const key = name + + (programConfiguration ? programConfiguration.cacheKey : '') + + (this._showOverdrawInspector ? '/overdraw' : '') + + (this.style.map.terrain ? '/terrain' : ''); + if (!this.cache[key]) { + this.cache[key] = new Program( + this.context, + shaders[name], + programConfiguration, + programUniforms[name], + this._showOverdrawInspector, + this.style.map.terrain + ); + } + return this.cache[key]; + } + + /* + * Reset some GL state to default values to avoid hard-to-debug bugs + * in custom layers. + */ + setCustomLayerDefaults() { + // Prevent custom layers from unintentionally modify the last VAO used. + // All other state is state is restored on it's own, but for VAOs it's + // simpler to unbind so that we don't have to track the state of VAOs. + this.context.unbindVAO(); + + // The default values for this state is meaningful and often expected. + // Leaving this state dirty could cause a lot of confusion for users. + this.context.cullFace.setDefault(); + this.context.activeTexture.setDefault(); + this.context.pixelStoreUnpack.setDefault(); + this.context.pixelStoreUnpackPremultiplyAlpha.setDefault(); + this.context.pixelStoreUnpackFlipY.setDefault(); + } + + /* + * Set GL state that is shared by all layers. + */ + setBaseState() { + const gl = this.context.gl; + this.context.cullFace.set(false); + this.context.viewport.set([0, 0, this.width, this.height]); + this.context.blendEquation.set(gl.FUNC_ADD); + } + + initDebugOverlayCanvas() { + if (this.debugOverlayCanvas == null) { + this.debugOverlayCanvas = document.createElement('canvas'); + this.debugOverlayCanvas.width = 512; + this.debugOverlayCanvas.height = 512; + const gl = this.context.gl; + this.debugOverlayTexture = new Texture(this.context, this.debugOverlayCanvas, gl.RGBA); + } + } + + destroy() { + if (this.debugOverlayTexture) { + this.debugOverlayTexture.destroy(); + } + } + + /* + * Return true if drawing buffer size is != from requested size. + * That means that we've reached GL limits somehow. + * Note: drawing buffer size changes only when canvas size changes + */ + overLimit() { + const {drawingBufferWidth, drawingBufferHeight} = this.context.gl; + return this.width !== drawingBufferWidth || this.height !== drawingBufferHeight; + } +} diff --git a/web/libraries/maplibre-gl/src/render/program.ts b/web/libraries/maplibre-gl/src/render/program.ts new file mode 100644 index 00000000..bac18b20 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program.ts @@ -0,0 +1,233 @@ +import {shaders} from '../shaders/shaders'; +import {ProgramConfiguration} from '../data/program_configuration'; +import {VertexArrayObject} from './vertex_array_object'; +import {Context} from '../gl/context'; + +import type {SegmentVector} from '../data/segment'; +import type {VertexBuffer} from '../gl/vertex_buffer'; +import type {IndexBuffer} from '../gl/index_buffer'; +import type {DepthMode} from '../gl/depth_mode'; +import type {StencilMode} from '../gl/stencil_mode'; +import type {ColorMode} from '../gl/color_mode'; +import type {CullFaceMode} from '../gl/cull_face_mode'; +import type {UniformBindings, UniformValues, UniformLocations} from './uniform_binding'; +import type {BinderUniform} from '../data/program_configuration'; +import {terrainPreludeUniforms, TerrainPreludeUniformsType} from './program/terrain_program'; +import type {TerrainData} from '../render/terrain'; +import {Terrain} from '../render/terrain'; + +export type DrawMode = WebGLRenderingContextBase['LINES'] | WebGLRenderingContextBase['TRIANGLES'] | WebGL2RenderingContext['LINE_STRIP']; + +function getTokenizedAttributesAndUniforms(array: Array): Array { + const result = []; + + for (let i = 0; i < array.length; i++) { + if (array[i] === null) continue; + const token = array[i].split(' '); + result.push(token.pop()); + } + return result; +} + +/** + * @internal + * A webgl program to execute in the GPU space + */ +export class Program { + program: WebGLProgram; + attributes: {[_: string]: number}; + numAttributes: number; + fixedUniforms: Us; + terrainUniforms: TerrainPreludeUniformsType; + binderUniforms: Array; + failedToCreate: boolean; + + constructor(context: Context, + source: { + fragmentSource: string; + vertexSource: string; + staticAttributes: Array; + staticUniforms: Array; + }, + configuration: ProgramConfiguration, + fixedUniforms: (b: Context, a: UniformLocations) => Us, + showOverdrawInspector: boolean, + terrain: Terrain) { + + const gl = context.gl; + this.program = gl.createProgram(); + + const staticAttrInfo = getTokenizedAttributesAndUniforms(source.staticAttributes); + const dynamicAttrInfo = configuration ? configuration.getBinderAttributes() : []; + const allAttrInfo = staticAttrInfo.concat(dynamicAttrInfo); + + const preludeUniformsInfo = shaders.prelude.staticUniforms ? getTokenizedAttributesAndUniforms(shaders.prelude.staticUniforms) : []; + const staticUniformsInfo = source.staticUniforms ? getTokenizedAttributesAndUniforms(source.staticUniforms) : []; + const dynamicUniformsInfo = configuration ? configuration.getBinderUniforms() : []; + // remove duplicate uniforms + const uniformList = preludeUniformsInfo.concat(staticUniformsInfo).concat(dynamicUniformsInfo); + const allUniformsInfo = []; + for (const uniform of uniformList) { + if (allUniformsInfo.indexOf(uniform) < 0) allUniformsInfo.push(uniform); + } + + const defines = configuration ? configuration.defines() : []; + if (showOverdrawInspector) { + defines.push('#define OVERDRAW_INSPECTOR;'); + } + if (terrain) { + defines.push('#define TERRAIN3D;'); + } + + const fragmentSource = defines.concat(shaders.prelude.fragmentSource, source.fragmentSource).join('\n'); + const vertexSource = defines.concat(shaders.prelude.vertexSource, source.vertexSource).join('\n'); + + const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER); + if (gl.isContextLost()) { + this.failedToCreate = true; + return; + } + gl.shaderSource(fragmentShader, fragmentSource); + gl.compileShader(fragmentShader); + + if (!gl.getShaderParameter(fragmentShader, gl.COMPILE_STATUS)) { + throw new Error(`Could not compile fragment shader: ${gl.getShaderInfoLog(fragmentShader)}`); + } + + gl.attachShader(this.program, fragmentShader); + + const vertexShader = gl.createShader(gl.VERTEX_SHADER); + if (gl.isContextLost()) { + this.failedToCreate = true; + return; + } + gl.shaderSource(vertexShader, vertexSource); + gl.compileShader(vertexShader); + + if (!gl.getShaderParameter(vertexShader, gl.COMPILE_STATUS)) { + throw new Error(`Could not compile vertex shader: ${gl.getShaderInfoLog(vertexShader)}`); + } + + gl.attachShader(this.program, vertexShader); + + this.attributes = {}; + const uniformLocations = {}; + + this.numAttributes = allAttrInfo.length; + + for (let i = 0; i < this.numAttributes; i++) { + if (allAttrInfo[i]) { + gl.bindAttribLocation(this.program, i, allAttrInfo[i]); + this.attributes[allAttrInfo[i]] = i; + } + } + + gl.linkProgram(this.program); + + if (!gl.getProgramParameter(this.program, gl.LINK_STATUS)) { + throw new Error(`Program failed to link: ${gl.getProgramInfoLog(this.program)}`); + } + + gl.deleteShader(vertexShader); + gl.deleteShader(fragmentShader); + + for (let it = 0; it < allUniformsInfo.length; it++) { + const uniform = allUniformsInfo[it]; + if (uniform && !uniformLocations[uniform]) { + const uniformLocation = gl.getUniformLocation(this.program, uniform); + if (uniformLocation) { + uniformLocations[uniform] = uniformLocation; + } + } + } + + this.fixedUniforms = fixedUniforms(context, uniformLocations); + this.terrainUniforms = terrainPreludeUniforms(context, uniformLocations); + this.binderUniforms = configuration ? configuration.getUniforms(context, uniformLocations) : []; + } + + draw(context: Context, + drawMode: DrawMode, + depthMode: Readonly, + stencilMode: Readonly, + colorMode: Readonly, + cullFaceMode: Readonly, + uniformValues: UniformValues, + terrain: TerrainData, + layerID: string, + layoutVertexBuffer: VertexBuffer, + indexBuffer: IndexBuffer, + segments: SegmentVector, + currentProperties?: any, + zoom?: number | null, + configuration?: ProgramConfiguration | null, + dynamicLayoutBuffer?: VertexBuffer | null, + dynamicLayoutBuffer2?: VertexBuffer | null, + dynamicLayoutBuffer3?: VertexBuffer | null) { + + const gl = context.gl; + + if (this.failedToCreate) return; + + context.program.set(this.program); + context.setDepthMode(depthMode); + context.setStencilMode(stencilMode); + context.setColorMode(colorMode); + context.setCullFace(cullFaceMode); + + // set variables used by the 3d functions defined in _prelude.vertex.glsl + if (terrain) { + context.activeTexture.set(gl.TEXTURE2); + gl.bindTexture(gl.TEXTURE_2D, terrain.depthTexture); + context.activeTexture.set(gl.TEXTURE3); + gl.bindTexture(gl.TEXTURE_2D, terrain.texture); + for (const name in this.terrainUniforms) { + this.terrainUniforms[name].set(terrain[name]); + } + } + + for (const name in this.fixedUniforms) { + this.fixedUniforms[name].set(uniformValues[name]); + } + + if (configuration) { + configuration.setUniforms(context, this.binderUniforms, currentProperties, {zoom: (zoom as any)}); + } + + let primitiveSize = 0; + switch (drawMode) { + case gl.LINES: + primitiveSize = 2; + break; + case gl.TRIANGLES: + primitiveSize = 3; + break; + case gl.LINE_STRIP: + primitiveSize = 1; + break; + } + + for (const segment of segments.get()) { + const vaos = segment.vaos || (segment.vaos = {}); + const vao: VertexArrayObject = vaos[layerID] || (vaos[layerID] = new VertexArrayObject()); + + vao.bind( + context, + this, + layoutVertexBuffer, + configuration ? configuration.getPaintVertexBuffers() : [], + indexBuffer, + segment.vertexOffset, + dynamicLayoutBuffer, + dynamicLayoutBuffer2, + dynamicLayoutBuffer3 + ); + + gl.drawElements( + drawMode, + segment.primitiveLength * primitiveSize, + gl.UNSIGNED_SHORT, + segment.primitiveOffset * primitiveSize * 2); + } + } +} diff --git a/web/libraries/maplibre-gl/src/render/program/background_program.ts b/web/libraries/maplibre-gl/src/render/program/background_program.ts new file mode 100644 index 00000000..7802c488 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/background_program.ts @@ -0,0 +1,100 @@ +import {bgPatternUniformValues} from './pattern'; +import { + Uniform1i, + Uniform1f, + Uniform2f, + UniformColor, + UniformMatrix4f +} from '../uniform_binding'; +import {extend} from '../../util/util'; + +import type {Painter} from '../painter'; +import type {UniformValues, UniformLocations} from '../uniform_binding'; +import type {Context} from '../../gl/context'; +import type {Color, ResolvedImage} from '@maplibre/maplibre-gl-style-spec'; +import type {CrossFaded} from '../../style/properties'; +import type {CrossfadeParameters} from '../../style/evaluation_parameters'; +import type {OverscaledTileID} from '../../source/tile_id'; +import {mat4} from 'gl-matrix'; + +export type BackgroundUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_opacity': Uniform1f; + 'u_color': UniformColor; +}; + +export type BackgroundPatternUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_opacity': Uniform1f; + // pattern uniforms: + 'u_image': Uniform1i; + 'u_pattern_tl_a': Uniform2f; + 'u_pattern_br_a': Uniform2f; + 'u_pattern_tl_b': Uniform2f; + 'u_pattern_br_b': Uniform2f; + 'u_texsize': Uniform2f; + 'u_mix': Uniform1f; + 'u_pattern_size_a': Uniform2f; + 'u_pattern_size_b': Uniform2f; + 'u_scale_a': Uniform1f; + 'u_scale_b': Uniform1f; + 'u_pixel_coord_upper': Uniform2f; + 'u_pixel_coord_lower': Uniform2f; + 'u_tile_units_to_pixels': Uniform1f; +}; + +const backgroundUniforms = (context: Context, locations: UniformLocations): BackgroundUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_opacity': new Uniform1f(context, locations.u_opacity), + 'u_color': new UniformColor(context, locations.u_color) +}); + +const backgroundPatternUniforms = (context: Context, locations: UniformLocations): BackgroundPatternUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_opacity': new Uniform1f(context, locations.u_opacity), + 'u_image': new Uniform1i(context, locations.u_image), + 'u_pattern_tl_a': new Uniform2f(context, locations.u_pattern_tl_a), + 'u_pattern_br_a': new Uniform2f(context, locations.u_pattern_br_a), + 'u_pattern_tl_b': new Uniform2f(context, locations.u_pattern_tl_b), + 'u_pattern_br_b': new Uniform2f(context, locations.u_pattern_br_b), + 'u_texsize': new Uniform2f(context, locations.u_texsize), + 'u_mix': new Uniform1f(context, locations.u_mix), + 'u_pattern_size_a': new Uniform2f(context, locations.u_pattern_size_a), + 'u_pattern_size_b': new Uniform2f(context, locations.u_pattern_size_b), + 'u_scale_a': new Uniform1f(context, locations.u_scale_a), + 'u_scale_b': new Uniform1f(context, locations.u_scale_b), + 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper), + 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower), + 'u_tile_units_to_pixels': new Uniform1f(context, locations.u_tile_units_to_pixels) +}); + +const backgroundUniformValues = (matrix: mat4, opacity: number, color: Color): UniformValues => ({ + 'u_matrix': matrix, + 'u_opacity': opacity, + 'u_color': color +}); + +const backgroundPatternUniformValues = ( + matrix: mat4, + opacity: number, + painter: Painter, + image: CrossFaded, + tile: { + tileID: OverscaledTileID; + tileSize: number; + }, + crossfade: CrossfadeParameters +): UniformValues => extend( + bgPatternUniformValues(image, crossfade, painter, tile), + { + 'u_matrix': matrix, + 'u_opacity': opacity + } +); + +export { + backgroundUniforms, + backgroundPatternUniforms, + backgroundUniformValues, + backgroundPatternUniformValues +}; diff --git a/web/libraries/maplibre-gl/src/render/program/circle_program.ts b/web/libraries/maplibre-gl/src/render/program/circle_program.ts new file mode 100644 index 00000000..97f1ef04 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/circle_program.ts @@ -0,0 +1,61 @@ +import {Uniform1i, Uniform1f, Uniform2f, UniformMatrix4f} from '../uniform_binding'; +import {pixelsToTileUnits} from '../../source/pixels_to_tile_units'; + +import type {Context} from '../../gl/context'; +import type {UniformValues, UniformLocations} from '../uniform_binding'; +import type {OverscaledTileID} from '../../source/tile_id'; +import type {Tile} from '../../source/tile'; +import type {CircleStyleLayer} from '../../style/style_layer/circle_style_layer'; +import type {Painter} from '../painter'; + +export type CircleUniformsType = { + 'u_camera_to_center_distance': Uniform1f; + 'u_scale_with_map': Uniform1i; + 'u_pitch_with_map': Uniform1i; + 'u_extrude_scale': Uniform2f; + 'u_device_pixel_ratio': Uniform1f; + 'u_matrix': UniformMatrix4f; +}; + +const circleUniforms = (context: Context, locations: UniformLocations): CircleUniformsType => ({ + 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance), + 'u_scale_with_map': new Uniform1i(context, locations.u_scale_with_map), + 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map), + 'u_extrude_scale': new Uniform2f(context, locations.u_extrude_scale), + 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio), + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix) +}); + +const circleUniformValues = ( + painter: Painter, + coord: OverscaledTileID, + tile: Tile, + layer: CircleStyleLayer +): UniformValues => { + const transform = painter.transform; + + let pitchWithMap: boolean, extrudeScale: [number, number]; + if (layer.paint.get('circle-pitch-alignment') === 'map') { + const pixelRatio = pixelsToTileUnits(tile, 1, transform.zoom); + pitchWithMap = true; + extrudeScale = [pixelRatio, pixelRatio]; + } else { + pitchWithMap = false; + extrudeScale = transform.pixelsToGLUnits; + } + + return { + 'u_camera_to_center_distance': transform.cameraToCenterDistance, + 'u_scale_with_map': +(layer.paint.get('circle-pitch-scale') === 'map'), + 'u_matrix': painter.translatePosMatrix( + coord.posMatrix, + tile, + layer.paint.get('circle-translate'), + layer.paint.get('circle-translate-anchor')), + 'u_pitch_with_map': +(pitchWithMap), + 'u_device_pixel_ratio': painter.pixelRatio, + 'u_extrude_scale': extrudeScale + }; +}; + +export {circleUniforms, circleUniformValues}; diff --git a/web/libraries/maplibre-gl/src/render/program/clipping_mask_program.ts b/web/libraries/maplibre-gl/src/render/program/clipping_mask_program.ts new file mode 100644 index 00000000..18819821 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/clipping_mask_program.ts @@ -0,0 +1,19 @@ +import {UniformMatrix4f} from '../uniform_binding'; + +import type {Context} from '../../gl/context'; +import type {UniformValues, UniformLocations} from '../uniform_binding'; +import {mat4} from 'gl-matrix'; + +export type ClippingMaskUniformsType = { + 'u_matrix': UniformMatrix4f; +}; + +const clippingMaskUniforms = (context: Context, locations: UniformLocations): ClippingMaskUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix) +}); + +const clippingMaskUniformValues = (matrix: mat4): UniformValues => ({ + 'u_matrix': matrix +}); + +export {clippingMaskUniforms, clippingMaskUniformValues}; diff --git a/web/libraries/maplibre-gl/src/render/program/collision_program.ts b/web/libraries/maplibre-gl/src/render/program/collision_program.ts new file mode 100644 index 00000000..4387dcf6 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/collision_program.ts @@ -0,0 +1,63 @@ +import {Uniform1f, Uniform2f, UniformMatrix4f} from '../uniform_binding'; +import {pixelsToTileUnits} from '../../source/pixels_to_tile_units'; + +import type {Context} from '../../gl/context'; +import type {UniformValues, UniformLocations} from '../uniform_binding'; +import type {Transform} from '../../geo/transform'; +import type {Tile} from '../../source/tile'; +import {mat4} from 'gl-matrix'; + +export type CollisionUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_camera_to_center_distance': Uniform1f; + 'u_pixels_to_tile_units': Uniform1f; + 'u_extrude_scale': Uniform2f; + 'u_overscale_factor': Uniform1f; +}; + +export type CollisionCircleUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_inv_matrix': UniformMatrix4f; + 'u_camera_to_center_distance': Uniform1f; + 'u_viewport_size': Uniform2f; +}; + +const collisionUniforms = (context: Context, locations: UniformLocations): CollisionUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance), + 'u_pixels_to_tile_units': new Uniform1f(context, locations.u_pixels_to_tile_units), + 'u_extrude_scale': new Uniform2f(context, locations.u_extrude_scale), + 'u_overscale_factor': new Uniform1f(context, locations.u_overscale_factor) +}); + +const collisionCircleUniforms = (context: Context, locations: UniformLocations): CollisionCircleUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_inv_matrix': new UniformMatrix4f(context, locations.u_inv_matrix), + 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance), + 'u_viewport_size': new Uniform2f(context, locations.u_viewport_size) +}); + +const collisionUniformValues = (matrix: mat4, transform: Transform, tile: Tile): UniformValues => { + const pixelRatio = pixelsToTileUnits(tile, 1, transform.zoom); + const scale = Math.pow(2, transform.zoom - tile.tileID.overscaledZ); + const overscaleFactor = tile.tileID.overscaleFactor(); + return { + 'u_matrix': matrix, + 'u_camera_to_center_distance': transform.cameraToCenterDistance, + 'u_pixels_to_tile_units': pixelRatio, + 'u_extrude_scale': [transform.pixelsToGLUnits[0] / (pixelRatio * scale), + transform.pixelsToGLUnits[1] / (pixelRatio * scale)], + 'u_overscale_factor': overscaleFactor + }; +}; + +const collisionCircleUniformValues = (matrix: mat4, invMatrix: mat4, transform: Transform): UniformValues => { + return { + 'u_matrix': matrix, + 'u_inv_matrix': invMatrix, + 'u_camera_to_center_distance': transform.cameraToCenterDistance, + 'u_viewport_size': [transform.width, transform.height] + }; +}; + +export {collisionUniforms, collisionUniformValues, collisionCircleUniforms, collisionCircleUniformValues}; diff --git a/web/libraries/maplibre-gl/src/render/program/debug_program.ts b/web/libraries/maplibre-gl/src/render/program/debug_program.ts new file mode 100644 index 00000000..21113eed --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/debug_program.ts @@ -0,0 +1,29 @@ +import {UniformColor, UniformMatrix4f, Uniform1i, Uniform1f} from '../uniform_binding'; + +import type {Context} from '../../gl/context'; +import type {UniformValues, UniformLocations} from '../uniform_binding'; +import type {Color} from '@maplibre/maplibre-gl-style-spec'; +import {mat4} from 'gl-matrix'; + +export type DebugUniformsType = { + 'u_color': UniformColor; + 'u_matrix': UniformMatrix4f; + 'u_overlay': Uniform1i; + 'u_overlay_scale': Uniform1f; +}; + +const debugUniforms = (context: Context, locations: UniformLocations): DebugUniformsType => ({ + 'u_color': new UniformColor(context, locations.u_color), + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_overlay': new Uniform1i(context, locations.u_overlay), + 'u_overlay_scale': new Uniform1f(context, locations.u_overlay_scale) +}); + +const debugUniformValues = (matrix: mat4, color: Color, scaleRatio: number = 1): UniformValues => ({ + 'u_matrix': matrix, + 'u_color': color, + 'u_overlay': 0, + 'u_overlay_scale': scaleRatio +}); + +export {debugUniforms, debugUniformValues}; diff --git a/web/libraries/maplibre-gl/src/render/program/fill_extrusion_program.ts b/web/libraries/maplibre-gl/src/render/program/fill_extrusion_program.ts new file mode 100644 index 00000000..d6efae42 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/fill_extrusion_program.ts @@ -0,0 +1,120 @@ +import {patternUniformValues} from './pattern'; +import { + Uniform1i, + Uniform1f, + Uniform2f, + Uniform3f, + UniformMatrix4f +} from '../uniform_binding'; + +import {mat3, mat4, vec3} from 'gl-matrix'; +import {extend} from '../../util/util'; + +import type {Context} from '../../gl/context'; +import type {Painter} from '../painter'; +import type {OverscaledTileID} from '../../source/tile_id'; +import type {UniformValues, UniformLocations} from '../uniform_binding'; +import type {CrossfadeParameters} from '../../style/evaluation_parameters'; +import type {Tile} from '../../source/tile'; + +export type FillExtrusionUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_lightpos': Uniform3f; + 'u_lightintensity': Uniform1f; + 'u_lightcolor': Uniform3f; + 'u_vertical_gradient': Uniform1f; + 'u_opacity': Uniform1f; +}; + +export type FillExtrusionPatternUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_lightpos': Uniform3f; + 'u_lightintensity': Uniform1f; + 'u_lightcolor': Uniform3f; + 'u_height_factor': Uniform1f; + 'u_vertical_gradient': Uniform1f; + // pattern uniforms: + 'u_texsize': Uniform2f; + 'u_image': Uniform1i; + 'u_pixel_coord_upper': Uniform2f; + 'u_pixel_coord_lower': Uniform2f; + 'u_scale': Uniform3f; + 'u_fade': Uniform1f; + 'u_opacity': Uniform1f; +}; + +const fillExtrusionUniforms = (context: Context, locations: UniformLocations): FillExtrusionUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_lightpos': new Uniform3f(context, locations.u_lightpos), + 'u_lightintensity': new Uniform1f(context, locations.u_lightintensity), + 'u_lightcolor': new Uniform3f(context, locations.u_lightcolor), + 'u_vertical_gradient': new Uniform1f(context, locations.u_vertical_gradient), + 'u_opacity': new Uniform1f(context, locations.u_opacity) +}); + +const fillExtrusionPatternUniforms = (context: Context, locations: UniformLocations): FillExtrusionPatternUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_lightpos': new Uniform3f(context, locations.u_lightpos), + 'u_lightintensity': new Uniform1f(context, locations.u_lightintensity), + 'u_lightcolor': new Uniform3f(context, locations.u_lightcolor), + 'u_vertical_gradient': new Uniform1f(context, locations.u_vertical_gradient), + 'u_height_factor': new Uniform1f(context, locations.u_height_factor), + // pattern uniforms + 'u_image': new Uniform1i(context, locations.u_image), + 'u_texsize': new Uniform2f(context, locations.u_texsize), + 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper), + 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower), + 'u_scale': new Uniform3f(context, locations.u_scale), + 'u_fade': new Uniform1f(context, locations.u_fade), + 'u_opacity': new Uniform1f(context, locations.u_opacity) +}); + +const fillExtrusionUniformValues = ( + matrix: mat4, + painter: Painter, + shouldUseVerticalGradient: boolean, + opacity: number +): UniformValues => { + const light = painter.style.light; + const _lp = light.properties.get('position'); + const lightPos = [_lp.x, _lp.y, _lp.z] as vec3; + const lightMat = mat3.create(); + if (light.properties.get('anchor') === 'viewport') { + mat3.fromRotation(lightMat, -painter.transform.angle); + } + vec3.transformMat3(lightPos, lightPos, lightMat); + + const lightColor = light.properties.get('color'); + + return { + 'u_matrix': matrix, + 'u_lightpos': lightPos, + 'u_lightintensity': light.properties.get('intensity'), + 'u_lightcolor': [lightColor.r, lightColor.g, lightColor.b], + 'u_vertical_gradient': +shouldUseVerticalGradient, + 'u_opacity': opacity + }; +}; + +const fillExtrusionPatternUniformValues = ( + matrix: mat4, + painter: Painter, + shouldUseVerticalGradient: boolean, + opacity: number, + coord: OverscaledTileID, + crossfade: CrossfadeParameters, + tile: Tile +): UniformValues => { + return extend(fillExtrusionUniformValues(matrix, painter, shouldUseVerticalGradient, opacity), + patternUniformValues(crossfade, painter, tile), + { + 'u_height_factor': -Math.pow(2, coord.overscaledZ) / tile.tileSize / 8 + }); +}; + +export { + fillExtrusionUniforms, + fillExtrusionPatternUniforms, + fillExtrusionUniformValues, + fillExtrusionPatternUniformValues +}; diff --git a/web/libraries/maplibre-gl/src/render/program/fill_program.ts b/web/libraries/maplibre-gl/src/render/program/fill_program.ts new file mode 100644 index 00000000..1ae148ba --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/fill_program.ts @@ -0,0 +1,121 @@ +import {patternUniformValues} from './pattern'; +import { + Uniform1i, + Uniform1f, + Uniform2f, + Uniform3f, + UniformMatrix4f +} from '../uniform_binding'; +import {extend} from '../../util/util'; + +import type {Painter} from '../painter'; +import type {UniformValues, UniformLocations} from '../uniform_binding'; +import type {Context} from '../../gl/context'; +import type {CrossfadeParameters} from '../../style/evaluation_parameters'; +import type {Tile} from '../../source/tile'; +import {mat4} from 'gl-matrix'; + +export type FillUniformsType = { + 'u_matrix': UniformMatrix4f; +}; + +export type FillOutlineUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_world': Uniform2f; +}; + +export type FillPatternUniformsType = { + 'u_matrix': UniformMatrix4f; + // pattern uniforms: + 'u_texsize': Uniform2f; + 'u_image': Uniform1i; + 'u_pixel_coord_upper': Uniform2f; + 'u_pixel_coord_lower': Uniform2f; + 'u_scale': Uniform3f; + 'u_fade': Uniform1f; +}; + +export type FillOutlinePatternUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_world': Uniform2f; + // pattern uniforms: + 'u_texsize': Uniform2f; + 'u_image': Uniform1i; + 'u_pixel_coord_upper': Uniform2f; + 'u_pixel_coord_lower': Uniform2f; + 'u_scale': Uniform3f; + 'u_fade': Uniform1f; +}; + +const fillUniforms = (context: Context, locations: UniformLocations): FillUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix) +}); + +const fillPatternUniforms = (context: Context, locations: UniformLocations): FillPatternUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_image': new Uniform1i(context, locations.u_image), + 'u_texsize': new Uniform2f(context, locations.u_texsize), + 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper), + 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower), + 'u_scale': new Uniform3f(context, locations.u_scale), + 'u_fade': new Uniform1f(context, locations.u_fade) +}); + +const fillOutlineUniforms = (context: Context, locations: UniformLocations): FillOutlineUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_world': new Uniform2f(context, locations.u_world) +}); + +const fillOutlinePatternUniforms = (context: Context, locations: UniformLocations): FillOutlinePatternUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_world': new Uniform2f(context, locations.u_world), + 'u_image': new Uniform1i(context, locations.u_image), + 'u_texsize': new Uniform2f(context, locations.u_texsize), + 'u_pixel_coord_upper': new Uniform2f(context, locations.u_pixel_coord_upper), + 'u_pixel_coord_lower': new Uniform2f(context, locations.u_pixel_coord_lower), + 'u_scale': new Uniform3f(context, locations.u_scale), + 'u_fade': new Uniform1f(context, locations.u_fade) +}); + +const fillUniformValues = (matrix: mat4): UniformValues => ({ + 'u_matrix': matrix +}); + +const fillPatternUniformValues = ( + matrix: mat4, + painter: Painter, + crossfade: CrossfadeParameters, + tile: Tile +): UniformValues => extend( + fillUniformValues(matrix), + patternUniformValues(crossfade, painter, tile) +); + +const fillOutlineUniformValues = (matrix: mat4, drawingBufferSize: [number, number]): UniformValues => ({ + 'u_matrix': matrix, + 'u_world': drawingBufferSize +}); + +const fillOutlinePatternUniformValues = ( + matrix: mat4, + painter: Painter, + crossfade: CrossfadeParameters, + tile: Tile, + drawingBufferSize: [number, number] +): UniformValues => extend( + fillPatternUniformValues(matrix, painter, crossfade, tile), + { + 'u_world': drawingBufferSize + } +); + +export { + fillUniforms, + fillPatternUniforms, + fillOutlineUniforms, + fillOutlinePatternUniforms, + fillUniformValues, + fillPatternUniformValues, + fillOutlineUniformValues, + fillOutlinePatternUniformValues +}; diff --git a/web/libraries/maplibre-gl/src/render/program/heatmap_program.ts b/web/libraries/maplibre-gl/src/render/program/heatmap_program.ts new file mode 100644 index 00000000..b3c4e02a --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/heatmap_program.ts @@ -0,0 +1,76 @@ +import {mat4} from 'gl-matrix'; + +import { + Uniform1i, + Uniform1f, + Uniform2f, + UniformMatrix4f +} from '../uniform_binding'; +import {pixelsToTileUnits} from '../../source/pixels_to_tile_units'; + +import type {Context} from '../../gl/context'; +import type {Tile} from '../../source/tile'; +import type {UniformValues, UniformLocations} from '../uniform_binding'; +import type {Painter} from '../painter'; +import type {HeatmapStyleLayer} from '../../style/style_layer/heatmap_style_layer'; + +export type HeatmapUniformsType = { + 'u_extrude_scale': Uniform1f; + 'u_intensity': Uniform1f; + 'u_matrix': UniformMatrix4f; +}; + +export type HeatmapTextureUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_world': Uniform2f; + 'u_image': Uniform1i; + 'u_color_ramp': Uniform1i; + 'u_opacity': Uniform1f; +}; + +const heatmapUniforms = (context: Context, locations: UniformLocations): HeatmapUniformsType => ({ + 'u_extrude_scale': new Uniform1f(context, locations.u_extrude_scale), + 'u_intensity': new Uniform1f(context, locations.u_intensity), + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix) +}); + +const heatmapTextureUniforms = (context: Context, locations: UniformLocations): HeatmapTextureUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_world': new Uniform2f(context, locations.u_world), + 'u_image': new Uniform1i(context, locations.u_image), + 'u_color_ramp': new Uniform1i(context, locations.u_color_ramp), + 'u_opacity': new Uniform1f(context, locations.u_opacity) +}); + +const heatmapUniformValues = (matrix: mat4, tile: Tile, zoom: number, intensity: number): UniformValues => ({ + 'u_matrix': matrix, + 'u_extrude_scale': pixelsToTileUnits(tile, 1, zoom), + 'u_intensity': intensity +}); + +const heatmapTextureUniformValues = ( + painter: Painter, + layer: HeatmapStyleLayer, + textureUnit: number, + colorRampUnit: number +): UniformValues => { + const matrix = mat4.create(); + mat4.ortho(matrix, 0, painter.width, painter.height, 0, 0, 1); + + const gl = painter.context.gl; + + return { + 'u_matrix': matrix, + 'u_world': [gl.drawingBufferWidth, gl.drawingBufferHeight], + 'u_image': textureUnit, + 'u_color_ramp': colorRampUnit, + 'u_opacity': layer.paint.get('heatmap-opacity') + }; +}; + +export { + heatmapUniforms, + heatmapTextureUniforms, + heatmapUniformValues, + heatmapTextureUniformValues +}; diff --git a/web/libraries/maplibre-gl/src/render/program/hillshade_program.ts b/web/libraries/maplibre-gl/src/render/program/hillshade_program.ts new file mode 100644 index 00000000..f7f89aeb --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/hillshade_program.ts @@ -0,0 +1,116 @@ +import {mat4} from 'gl-matrix'; + +import { + Uniform1i, + Uniform1f, + Uniform2f, + UniformColor, + UniformMatrix4f, + Uniform4f +} from '../uniform_binding'; +import {EXTENT} from '../../data/extent'; +import {MercatorCoordinate} from '../../geo/mercator_coordinate'; + +import type {Context} from '../../gl/context'; +import type {UniformValues, UniformLocations} from '../uniform_binding'; +import type {Tile} from '../../source/tile'; +import type {Painter} from '../painter'; +import type {HillshadeStyleLayer} from '../../style/style_layer/hillshade_style_layer'; +import type {DEMData} from '../../data/dem_data'; +import type {OverscaledTileID} from '../../source/tile_id'; + +export type HillshadeUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_image': Uniform1i; + 'u_latrange': Uniform2f; + 'u_light': Uniform2f; + 'u_shadow': UniformColor; + 'u_highlight': UniformColor; + 'u_accent': UniformColor; +}; + +export type HillshadePrepareUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_image': Uniform1i; + 'u_dimension': Uniform2f; + 'u_zoom': Uniform1f; + 'u_unpack': Uniform4f; +}; + +const hillshadeUniforms = (context: Context, locations: UniformLocations): HillshadeUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_image': new Uniform1i(context, locations.u_image), + 'u_latrange': new Uniform2f(context, locations.u_latrange), + 'u_light': new Uniform2f(context, locations.u_light), + 'u_shadow': new UniformColor(context, locations.u_shadow), + 'u_highlight': new UniformColor(context, locations.u_highlight), + 'u_accent': new UniformColor(context, locations.u_accent) +}); + +const hillshadePrepareUniforms = (context: Context, locations: UniformLocations): HillshadePrepareUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_image': new Uniform1i(context, locations.u_image), + 'u_dimension': new Uniform2f(context, locations.u_dimension), + 'u_zoom': new Uniform1f(context, locations.u_zoom), + 'u_unpack': new Uniform4f(context, locations.u_unpack) +}); + +const hillshadeUniformValues = ( + painter: Painter, + tile: Tile, + layer: HillshadeStyleLayer, + coord: OverscaledTileID +): UniformValues => { + const shadow = layer.paint.get('hillshade-shadow-color'); + const highlight = layer.paint.get('hillshade-highlight-color'); + const accent = layer.paint.get('hillshade-accent-color'); + + let azimuthal = layer.paint.get('hillshade-illumination-direction') * (Math.PI / 180); + // modify azimuthal angle by map rotation if light is anchored at the viewport + if (layer.paint.get('hillshade-illumination-anchor') === 'viewport') { + azimuthal -= painter.transform.angle; + } + const align = !painter.options.moving; + return { + 'u_matrix': coord ? coord.posMatrix : painter.transform.calculatePosMatrix(tile.tileID.toUnwrapped(), align), + 'u_image': 0, + 'u_latrange': getTileLatRange(painter, tile.tileID), + 'u_light': [layer.paint.get('hillshade-exaggeration'), azimuthal], + 'u_shadow': shadow, + 'u_highlight': highlight, + 'u_accent': accent + }; +}; + +const hillshadeUniformPrepareValues = (tileID: OverscaledTileID, dem: DEMData): UniformValues => { + + const stride = dem.stride; + const matrix = mat4.create(); + // Flip rendering at y axis. + mat4.ortho(matrix, 0, EXTENT, -EXTENT, 0, 0, 1); + mat4.translate(matrix, matrix, [0, -EXTENT, 0]); + + return { + 'u_matrix': matrix, + 'u_image': 1, + 'u_dimension': [stride, stride], + 'u_zoom': tileID.overscaledZ, + 'u_unpack': dem.getUnpackVector() + }; +}; + +function getTileLatRange(painter: Painter, tileID: OverscaledTileID) { + // for scaling the magnitude of a points slope by its latitude + const tilesAtZoom = Math.pow(2, tileID.canonical.z); + const y = tileID.canonical.y; + return [ + new MercatorCoordinate(0, y / tilesAtZoom).toLngLat().lat, + new MercatorCoordinate(0, (y + 1) / tilesAtZoom).toLngLat().lat]; +} + +export { + hillshadeUniforms, + hillshadePrepareUniforms, + hillshadeUniformValues, + hillshadeUniformPrepareValues +}; diff --git a/web/libraries/maplibre-gl/src/render/program/line_program.ts b/web/libraries/maplibre-gl/src/render/program/line_program.ts new file mode 100644 index 00000000..ac33493d --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/line_program.ts @@ -0,0 +1,207 @@ +import {Uniform1i, Uniform1f, Uniform2f, Uniform3f, UniformMatrix4f} from '../uniform_binding'; +import {pixelsToTileUnits} from '../../source/pixels_to_tile_units'; +import {extend} from '../../util/util'; + +import type {Context} from '../../gl/context'; +import type {UniformValues, UniformLocations} from '../uniform_binding'; +import type {Transform} from '../../geo/transform'; +import type {Tile} from '../../source/tile'; +import type {CrossFaded} from '../../style/properties'; +import type {LineStyleLayer} from '../../style/style_layer/line_style_layer'; +import type {Painter} from '../painter'; +import type {CrossfadeParameters} from '../../style/evaluation_parameters'; +import {OverscaledTileID} from '../../source/tile_id'; + +export type LineUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_ratio': Uniform1f; + 'u_device_pixel_ratio': Uniform1f; + 'u_units_to_pixels': Uniform2f; +}; + +export type LineGradientUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_ratio': Uniform1f; + 'u_device_pixel_ratio': Uniform1f; + 'u_units_to_pixels': Uniform2f; + 'u_image': Uniform1i; + 'u_image_height': Uniform1f; +}; + +export type LinePatternUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_texsize': Uniform2f; + 'u_ratio': Uniform1f; + 'u_device_pixel_ratio': Uniform1f; + 'u_units_to_pixels': Uniform2f; + 'u_image': Uniform1i; + 'u_scale': Uniform3f; + 'u_fade': Uniform1f; +}; + +export type LineSDFUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_ratio': Uniform1f; + 'u_device_pixel_ratio': Uniform1f; + 'u_units_to_pixels': Uniform2f; + 'u_patternscale_a': Uniform2f; + 'u_patternscale_b': Uniform2f; + 'u_sdfgamma': Uniform1f; + 'u_image': Uniform1i; + 'u_tex_y_a': Uniform1f; + 'u_tex_y_b': Uniform1f; + 'u_mix': Uniform1f; +}; + +const lineUniforms = (context: Context, locations: UniformLocations): LineUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_ratio': new Uniform1f(context, locations.u_ratio), + 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio), + 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels) +}); + +const lineGradientUniforms = (context: Context, locations: UniformLocations): LineGradientUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_ratio': new Uniform1f(context, locations.u_ratio), + 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio), + 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels), + 'u_image': new Uniform1i(context, locations.u_image), + 'u_image_height': new Uniform1f(context, locations.u_image_height) +}); + +const linePatternUniforms = (context: Context, locations: UniformLocations): LinePatternUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_texsize': new Uniform2f(context, locations.u_texsize), + 'u_ratio': new Uniform1f(context, locations.u_ratio), + 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio), + 'u_image': new Uniform1i(context, locations.u_image), + 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels), + 'u_scale': new Uniform3f(context, locations.u_scale), + 'u_fade': new Uniform1f(context, locations.u_fade) +}); + +const lineSDFUniforms = (context: Context, locations: UniformLocations): LineSDFUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_ratio': new Uniform1f(context, locations.u_ratio), + 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio), + 'u_units_to_pixels': new Uniform2f(context, locations.u_units_to_pixels), + 'u_patternscale_a': new Uniform2f(context, locations.u_patternscale_a), + 'u_patternscale_b': new Uniform2f(context, locations.u_patternscale_b), + 'u_sdfgamma': new Uniform1f(context, locations.u_sdfgamma), + 'u_image': new Uniform1i(context, locations.u_image), + 'u_tex_y_a': new Uniform1f(context, locations.u_tex_y_a), + 'u_tex_y_b': new Uniform1f(context, locations.u_tex_y_b), + 'u_mix': new Uniform1f(context, locations.u_mix) +}); + +const lineUniformValues = ( + painter: Painter, + tile: Tile, + layer: LineStyleLayer, + coord: OverscaledTileID +): UniformValues => { + const transform = painter.transform; + + return { + 'u_matrix': calculateMatrix(painter, tile, layer, coord), + 'u_ratio': 1 / pixelsToTileUnits(tile, 1, transform.zoom), + 'u_device_pixel_ratio': painter.pixelRatio, + 'u_units_to_pixels': [ + 1 / transform.pixelsToGLUnits[0], + 1 / transform.pixelsToGLUnits[1] + ] + }; +}; + +const lineGradientUniformValues = ( + painter: Painter, + tile: Tile, + layer: LineStyleLayer, + imageHeight: number, + coord: OverscaledTileID +): UniformValues => { + return extend(lineUniformValues(painter, tile, layer, coord), { + 'u_image': 0, + 'u_image_height': imageHeight, + }); +}; + +const linePatternUniformValues = ( + painter: Painter, + tile: Tile, + layer: LineStyleLayer, + crossfade: CrossfadeParameters, + coord: OverscaledTileID +): UniformValues => { + const transform = painter.transform; + const tileZoomRatio = calculateTileRatio(tile, transform); + return { + 'u_matrix': calculateMatrix(painter, tile, layer, coord), + 'u_texsize': tile.imageAtlasTexture.size, + // camera zoom ratio + 'u_ratio': 1 / pixelsToTileUnits(tile, 1, transform.zoom), + 'u_device_pixel_ratio': painter.pixelRatio, + 'u_image': 0, + 'u_scale': [tileZoomRatio, crossfade.fromScale, crossfade.toScale], + 'u_fade': crossfade.t, + 'u_units_to_pixels': [ + 1 / transform.pixelsToGLUnits[0], + 1 / transform.pixelsToGLUnits[1] + ] + }; +}; + +const lineSDFUniformValues = ( + painter: Painter, + tile: Tile, + layer: LineStyleLayer, + dasharray: CrossFaded>, + crossfade: CrossfadeParameters, + coord: OverscaledTileID +): UniformValues => { + const transform = painter.transform; + const lineAtlas = painter.lineAtlas; + const tileRatio = calculateTileRatio(tile, transform); + + const round = layer.layout.get('line-cap') === 'round'; + + const posA = lineAtlas.getDash(dasharray.from, round); + const posB = lineAtlas.getDash(dasharray.to, round); + + const widthA = posA.width * crossfade.fromScale; + const widthB = posB.width * crossfade.toScale; + + return extend(lineUniformValues(painter, tile, layer, coord), { + 'u_patternscale_a': [tileRatio / widthA, -posA.height / 2], + 'u_patternscale_b': [tileRatio / widthB, -posB.height / 2], + 'u_sdfgamma': lineAtlas.width / (Math.min(widthA, widthB) * 256 * painter.pixelRatio) / 2, + 'u_image': 0, + 'u_tex_y_a': posA.y, + 'u_tex_y_b': posB.y, + 'u_mix': crossfade.t + }); +}; + +function calculateTileRatio(tile: Tile, transform: Transform) { + return 1 / pixelsToTileUnits(tile, 1, transform.tileZoom); +} + +function calculateMatrix(painter: Painter, tile: Tile, layer: LineStyleLayer, coord: OverscaledTileID) { + return painter.translatePosMatrix( + coord ? coord.posMatrix : tile.tileID.posMatrix, + tile, + layer.paint.get('line-translate'), + layer.paint.get('line-translate-anchor') + ); +} + +export { + lineUniforms, + lineGradientUniforms, + linePatternUniforms, + lineSDFUniforms, + lineUniformValues, + lineGradientUniformValues, + linePatternUniformValues, + lineSDFUniformValues +}; diff --git a/web/libraries/maplibre-gl/src/render/program/pattern.ts b/web/libraries/maplibre-gl/src/render/program/pattern.ts new file mode 100644 index 00000000..5dcb06c1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/pattern.ts @@ -0,0 +1,102 @@ +import { + Uniform1i, + Uniform1f, + Uniform2f, + Uniform3f +} from '../uniform_binding'; +import {pixelsToTileUnits} from '../../source/pixels_to_tile_units'; + +import type {Painter} from '../painter'; +import type {OverscaledTileID} from '../../source/tile_id'; +import type {CrossFaded} from '../../style/properties'; +import type {CrossfadeParameters} from '../../style/evaluation_parameters'; +import type {UniformValues} from '../uniform_binding'; +import type {Tile} from '../../source/tile'; +import type {ResolvedImage} from '@maplibre/maplibre-gl-style-spec'; + +type BackgroundPatternUniformsType = { + 'u_image': Uniform1i; + 'u_pattern_tl_a': Uniform2f; + 'u_pattern_br_a': Uniform2f; + 'u_pattern_tl_b': Uniform2f; + 'u_pattern_br_b': Uniform2f; + 'u_texsize': Uniform2f; + 'u_mix': Uniform1f; + 'u_pattern_size_a': Uniform2f; + 'u_pattern_size_b': Uniform2f; + 'u_scale_a': Uniform1f; + 'u_scale_b': Uniform1f; + 'u_pixel_coord_upper': Uniform2f; + 'u_pixel_coord_lower': Uniform2f; + 'u_tile_units_to_pixels': Uniform1f; +}; + +export type PatternUniformsType = { + // pattern uniforms: + 'u_image': Uniform1i; + 'u_texsize': Uniform2f; + 'u_scale': Uniform3f; + 'u_fade': Uniform1f; + 'u_pixel_coord_upper': Uniform2f; + 'u_pixel_coord_lower': Uniform2f; +}; + +function patternUniformValues(crossfade: CrossfadeParameters, painter: Painter, tile: Tile): UniformValues { + + const tileRatio = 1 / pixelsToTileUnits(tile, 1, painter.transform.tileZoom); + + const numTiles = Math.pow(2, tile.tileID.overscaledZ); + const tileSizeAtNearestZoom = tile.tileSize * Math.pow(2, painter.transform.tileZoom) / numTiles; + + const pixelX = tileSizeAtNearestZoom * (tile.tileID.canonical.x + tile.tileID.wrap * numTiles); + const pixelY = tileSizeAtNearestZoom * tile.tileID.canonical.y; + + return { + 'u_image': 0, + 'u_texsize': tile.imageAtlasTexture.size, + 'u_scale': [tileRatio, crossfade.fromScale, crossfade.toScale], + 'u_fade': crossfade.t, + // split the pixel coord into two pairs of 16 bit numbers. The glsl spec only guarantees 16 bits of precision. + 'u_pixel_coord_upper': [pixelX >> 16, pixelY >> 16], + 'u_pixel_coord_lower': [pixelX & 0xFFFF, pixelY & 0xFFFF] + }; +} + +function bgPatternUniformValues( + image: CrossFaded, + crossfade: CrossfadeParameters, + painter: Painter, + tile: { + tileID: OverscaledTileID; + tileSize: number; + } +): UniformValues { + const imagePosA = painter.imageManager.getPattern(image.from.toString()); + const imagePosB = painter.imageManager.getPattern(image.to.toString()); + const {width, height} = painter.imageManager.getPixelSize(); + + const numTiles = Math.pow(2, tile.tileID.overscaledZ); + const tileSizeAtNearestZoom = tile.tileSize * Math.pow(2, painter.transform.tileZoom) / numTiles; + + const pixelX = tileSizeAtNearestZoom * (tile.tileID.canonical.x + tile.tileID.wrap * numTiles); + const pixelY = tileSizeAtNearestZoom * tile.tileID.canonical.y; + + return { + 'u_image': 0, + 'u_pattern_tl_a': (imagePosA as any).tl, + 'u_pattern_br_a': (imagePosA as any).br, + 'u_pattern_tl_b': (imagePosB as any).tl, + 'u_pattern_br_b': (imagePosB as any).br, + 'u_texsize': [width, height], + 'u_mix': crossfade.t, + 'u_pattern_size_a': (imagePosA as any).displaySize, + 'u_pattern_size_b': (imagePosB as any).displaySize, + 'u_scale_a': crossfade.fromScale, + 'u_scale_b': crossfade.toScale, + 'u_tile_units_to_pixels': 1 / pixelsToTileUnits(tile, 1, painter.transform.tileZoom), + // split the pixel coord into two pairs of 16 bit numbers. The glsl spec only guarantees 16 bits of precision. + 'u_pixel_coord_upper': [pixelX >> 16, pixelY >> 16], + 'u_pixel_coord_lower': [pixelX & 0xFFFF, pixelY & 0xFFFF] + }; +} +export {bgPatternUniformValues, patternUniformValues}; diff --git a/web/libraries/maplibre-gl/src/render/program/program_uniforms.ts b/web/libraries/maplibre-gl/src/render/program/program_uniforms.ts new file mode 100644 index 00000000..f1723cca --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/program_uniforms.ts @@ -0,0 +1,44 @@ +import {fillExtrusionUniforms, fillExtrusionPatternUniforms} from './fill_extrusion_program'; +import {fillUniforms, fillPatternUniforms, fillOutlineUniforms, fillOutlinePatternUniforms} from './fill_program'; +import {circleUniforms} from './circle_program'; +import {collisionUniforms, collisionCircleUniforms} from './collision_program'; +import {debugUniforms} from './debug_program'; +import {clippingMaskUniforms} from './clipping_mask_program'; +import {heatmapUniforms, heatmapTextureUniforms} from './heatmap_program'; +import {hillshadeUniforms, hillshadePrepareUniforms} from './hillshade_program'; +import {lineUniforms, lineGradientUniforms, linePatternUniforms, lineSDFUniforms} from './line_program'; +import {rasterUniforms} from './raster_program'; +import {symbolIconUniforms, symbolSDFUniforms, symbolTextAndIconUniforms} from './symbol_program'; +import {backgroundUniforms, backgroundPatternUniforms} from './background_program'; +import {terrainUniforms, terrainDepthUniforms, terrainCoordsUniforms} from './terrain_program'; + +export const programUniforms = { + fillExtrusion: fillExtrusionUniforms, + fillExtrusionPattern: fillExtrusionPatternUniforms, + fill: fillUniforms, + fillPattern: fillPatternUniforms, + fillOutline: fillOutlineUniforms, + fillOutlinePattern: fillOutlinePatternUniforms, + circle: circleUniforms, + collisionBox: collisionUniforms, + collisionCircle: collisionCircleUniforms, + debug: debugUniforms, + clippingMask: clippingMaskUniforms, + heatmap: heatmapUniforms, + heatmapTexture: heatmapTextureUniforms, + hillshade: hillshadeUniforms, + hillshadePrepare: hillshadePrepareUniforms, + line: lineUniforms, + lineGradient: lineGradientUniforms, + linePattern: linePatternUniforms, + lineSDF: lineSDFUniforms, + raster: rasterUniforms, + symbolIcon: symbolIconUniforms, + symbolSDF: symbolSDFUniforms, + symbolTextAndIcon: symbolTextAndIconUniforms, + background: backgroundUniforms, + backgroundPattern: backgroundPatternUniforms, + terrain: terrainUniforms, + terrainDepth: terrainDepthUniforms, + terrainCoords: terrainCoordsUniforms +}; diff --git a/web/libraries/maplibre-gl/src/render/program/raster_program.ts b/web/libraries/maplibre-gl/src/render/program/raster_program.ts new file mode 100644 index 00000000..4ee6a431 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/raster_program.ts @@ -0,0 +1,88 @@ +import {Uniform1i, Uniform1f, Uniform2f, Uniform3f, UniformMatrix4f} from '../uniform_binding'; + +import type {Context} from '../../gl/context'; +import type {UniformValues, UniformLocations} from '../uniform_binding'; +import type {RasterStyleLayer} from '../../style/style_layer/raster_style_layer'; +import {mat4} from 'gl-matrix'; + +export type RasterUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_tl_parent': Uniform2f; + 'u_scale_parent': Uniform1f; + 'u_buffer_scale': Uniform1f; + 'u_fade_t': Uniform1f; + 'u_opacity': Uniform1f; + 'u_image0': Uniform1i; + 'u_image1': Uniform1i; + 'u_brightness_low': Uniform1f; + 'u_brightness_high': Uniform1f; + 'u_saturation_factor': Uniform1f; + 'u_contrast_factor': Uniform1f; + 'u_spin_weights': Uniform3f; +}; + +const rasterUniforms = (context: Context, locations: UniformLocations): RasterUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_tl_parent': new Uniform2f(context, locations.u_tl_parent), + 'u_scale_parent': new Uniform1f(context, locations.u_scale_parent), + 'u_buffer_scale': new Uniform1f(context, locations.u_buffer_scale), + 'u_fade_t': new Uniform1f(context, locations.u_fade_t), + 'u_opacity': new Uniform1f(context, locations.u_opacity), + 'u_image0': new Uniform1i(context, locations.u_image0), + 'u_image1': new Uniform1i(context, locations.u_image1), + 'u_brightness_low': new Uniform1f(context, locations.u_brightness_low), + 'u_brightness_high': new Uniform1f(context, locations.u_brightness_high), + 'u_saturation_factor': new Uniform1f(context, locations.u_saturation_factor), + 'u_contrast_factor': new Uniform1f(context, locations.u_contrast_factor), + 'u_spin_weights': new Uniform3f(context, locations.u_spin_weights) +}); + +const rasterUniformValues = ( + matrix: mat4, + parentTL: [number, number], + parentScaleBy: number, + fade: { + mix: number; + opacity: number; + }, + layer: RasterStyleLayer +): UniformValues => ({ + 'u_matrix': matrix, + 'u_tl_parent': parentTL, + 'u_scale_parent': parentScaleBy, + 'u_buffer_scale': 1, + 'u_fade_t': fade.mix, + 'u_opacity': fade.opacity * layer.paint.get('raster-opacity'), + 'u_image0': 0, + 'u_image1': 1, + 'u_brightness_low': layer.paint.get('raster-brightness-min'), + 'u_brightness_high': layer.paint.get('raster-brightness-max'), + 'u_saturation_factor': saturationFactor(layer.paint.get('raster-saturation')), + 'u_contrast_factor': contrastFactor(layer.paint.get('raster-contrast')), + 'u_spin_weights': spinWeights(layer.paint.get('raster-hue-rotate')) +}); + +function spinWeights(angle) { + angle *= Math.PI / 180; + const s = Math.sin(angle); + const c = Math.cos(angle); + return [ + (2 * c + 1) / 3, + (-Math.sqrt(3) * s - c + 1) / 3, + (Math.sqrt(3) * s - c + 1) / 3 + ]; +} + +function contrastFactor(contrast) { + return contrast > 0 ? + 1 / (1 - contrast) : + 1 + contrast; +} + +function saturationFactor(saturation) { + return saturation > 0 ? + 1 - 1 / (1.001 - saturation) : + -saturation; +} + +export {rasterUniforms, rasterUniformValues}; diff --git a/web/libraries/maplibre-gl/src/render/program/symbol_program.ts b/web/libraries/maplibre-gl/src/render/program/symbol_program.ts new file mode 100644 index 00000000..63508b94 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/symbol_program.ts @@ -0,0 +1,226 @@ +import {Uniform1i, Uniform1f, Uniform2f, UniformMatrix4f} from '../uniform_binding'; +import {extend} from '../../util/util'; + +import type {Context} from '../../gl/context'; +import type {Painter} from '../painter'; +import type {UniformValues, UniformLocations} from '../uniform_binding'; +import {mat4} from 'gl-matrix'; + +export type SymbolIconUniformsType = { + 'u_is_size_zoom_constant': Uniform1i; + 'u_is_size_feature_constant': Uniform1i; + 'u_size_t': Uniform1f; + 'u_size': Uniform1f; + 'u_camera_to_center_distance': Uniform1f; + 'u_pitch': Uniform1f; + 'u_rotate_symbol': Uniform1i; + 'u_aspect_ratio': Uniform1f; + 'u_fade_change': Uniform1f; + 'u_matrix': UniformMatrix4f; + 'u_label_plane_matrix': UniformMatrix4f; + 'u_coord_matrix': UniformMatrix4f; + 'u_is_text': Uniform1i; + 'u_pitch_with_map': Uniform1i; + 'u_texsize': Uniform2f; + 'u_texture': Uniform1i; +}; + +export type SymbolSDFUniformsType = { + 'u_is_size_zoom_constant': Uniform1i; + 'u_is_size_feature_constant': Uniform1i; + 'u_size_t': Uniform1f; + 'u_size': Uniform1f; + 'u_camera_to_center_distance': Uniform1f; + 'u_pitch': Uniform1f; + 'u_rotate_symbol': Uniform1i; + 'u_aspect_ratio': Uniform1f; + 'u_fade_change': Uniform1f; + 'u_matrix': UniformMatrix4f; + 'u_label_plane_matrix': UniformMatrix4f; + 'u_coord_matrix': UniformMatrix4f; + 'u_is_text': Uniform1i; + 'u_pitch_with_map': Uniform1i; + 'u_texsize': Uniform2f; + 'u_texture': Uniform1i; + 'u_gamma_scale': Uniform1f; + 'u_device_pixel_ratio': Uniform1f; + 'u_is_halo': Uniform1i; +}; + +export type symbolTextAndIconUniformsType = { + 'u_is_size_zoom_constant': Uniform1i; + 'u_is_size_feature_constant': Uniform1i; + 'u_size_t': Uniform1f; + 'u_size': Uniform1f; + 'u_camera_to_center_distance': Uniform1f; + 'u_pitch': Uniform1f; + 'u_rotate_symbol': Uniform1i; + 'u_aspect_ratio': Uniform1f; + 'u_fade_change': Uniform1f; + 'u_matrix': UniformMatrix4f; + 'u_label_plane_matrix': UniformMatrix4f; + 'u_coord_matrix': UniformMatrix4f; + 'u_is_text': Uniform1i; + 'u_pitch_with_map': Uniform1i; + 'u_texsize': Uniform2f; + 'u_texsize_icon': Uniform2f; + 'u_texture': Uniform1i; + 'u_texture_icon': Uniform1i; + 'u_gamma_scale': Uniform1f; + 'u_device_pixel_ratio': Uniform1f; + 'u_is_halo': Uniform1i; +}; + +const symbolIconUniforms = (context: Context, locations: UniformLocations): SymbolIconUniformsType => ({ + 'u_is_size_zoom_constant': new Uniform1i(context, locations.u_is_size_zoom_constant), + 'u_is_size_feature_constant': new Uniform1i(context, locations.u_is_size_feature_constant), + 'u_size_t': new Uniform1f(context, locations.u_size_t), + 'u_size': new Uniform1f(context, locations.u_size), + 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance), + 'u_pitch': new Uniform1f(context, locations.u_pitch), + 'u_rotate_symbol': new Uniform1i(context, locations.u_rotate_symbol), + 'u_aspect_ratio': new Uniform1f(context, locations.u_aspect_ratio), + 'u_fade_change': new Uniform1f(context, locations.u_fade_change), + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_label_plane_matrix': new UniformMatrix4f(context, locations.u_label_plane_matrix), + 'u_coord_matrix': new UniformMatrix4f(context, locations.u_coord_matrix), + 'u_is_text': new Uniform1i(context, locations.u_is_text), + 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map), + 'u_texsize': new Uniform2f(context, locations.u_texsize), + 'u_texture': new Uniform1i(context, locations.u_texture) +}); + +const symbolSDFUniforms = (context: Context, locations: UniformLocations): SymbolSDFUniformsType => ({ + 'u_is_size_zoom_constant': new Uniform1i(context, locations.u_is_size_zoom_constant), + 'u_is_size_feature_constant': new Uniform1i(context, locations.u_is_size_feature_constant), + 'u_size_t': new Uniform1f(context, locations.u_size_t), + 'u_size': new Uniform1f(context, locations.u_size), + 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance), + 'u_pitch': new Uniform1f(context, locations.u_pitch), + 'u_rotate_symbol': new Uniform1i(context, locations.u_rotate_symbol), + 'u_aspect_ratio': new Uniform1f(context, locations.u_aspect_ratio), + 'u_fade_change': new Uniform1f(context, locations.u_fade_change), + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_label_plane_matrix': new UniformMatrix4f(context, locations.u_label_plane_matrix), + 'u_coord_matrix': new UniformMatrix4f(context, locations.u_coord_matrix), + 'u_is_text': new Uniform1i(context, locations.u_is_text), + 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map), + 'u_texsize': new Uniform2f(context, locations.u_texsize), + 'u_texture': new Uniform1i(context, locations.u_texture), + 'u_gamma_scale': new Uniform1f(context, locations.u_gamma_scale), + 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio), + 'u_is_halo': new Uniform1i(context, locations.u_is_halo) +}); + +const symbolTextAndIconUniforms = (context: Context, locations: UniformLocations): symbolTextAndIconUniformsType => ({ + 'u_is_size_zoom_constant': new Uniform1i(context, locations.u_is_size_zoom_constant), + 'u_is_size_feature_constant': new Uniform1i(context, locations.u_is_size_feature_constant), + 'u_size_t': new Uniform1f(context, locations.u_size_t), + 'u_size': new Uniform1f(context, locations.u_size), + 'u_camera_to_center_distance': new Uniform1f(context, locations.u_camera_to_center_distance), + 'u_pitch': new Uniform1f(context, locations.u_pitch), + 'u_rotate_symbol': new Uniform1i(context, locations.u_rotate_symbol), + 'u_aspect_ratio': new Uniform1f(context, locations.u_aspect_ratio), + 'u_fade_change': new Uniform1f(context, locations.u_fade_change), + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_label_plane_matrix': new UniformMatrix4f(context, locations.u_label_plane_matrix), + 'u_coord_matrix': new UniformMatrix4f(context, locations.u_coord_matrix), + 'u_is_text': new Uniform1i(context, locations.u_is_text), + 'u_pitch_with_map': new Uniform1i(context, locations.u_pitch_with_map), + 'u_texsize': new Uniform2f(context, locations.u_texsize), + 'u_texsize_icon': new Uniform2f(context, locations.u_texsize_icon), + 'u_texture': new Uniform1i(context, locations.u_texture), + 'u_texture_icon': new Uniform1i(context, locations.u_texture_icon), + 'u_gamma_scale': new Uniform1f(context, locations.u_gamma_scale), + 'u_device_pixel_ratio': new Uniform1f(context, locations.u_device_pixel_ratio), + 'u_is_halo': new Uniform1i(context, locations.u_is_halo) +}); + +const symbolIconUniformValues = ( + functionType: string, + size: { + uSizeT: number; + uSize: number; + }, + rotateInShader: boolean, + pitchWithMap: boolean, + painter: Painter, + matrix: mat4, + labelPlaneMatrix: mat4, + glCoordMatrix: mat4, + isText: boolean, + texSize: [number, number] +): UniformValues => { + const transform = painter.transform; + + return { + 'u_is_size_zoom_constant': +(functionType === 'constant' || functionType === 'source'), + 'u_is_size_feature_constant': +(functionType === 'constant' || functionType === 'camera'), + 'u_size_t': size ? size.uSizeT : 0, + 'u_size': size ? size.uSize : 0, + 'u_camera_to_center_distance': transform.cameraToCenterDistance, + 'u_pitch': transform.pitch / 360 * 2 * Math.PI, + 'u_rotate_symbol': +rotateInShader, + 'u_aspect_ratio': transform.width / transform.height, + 'u_fade_change': painter.options.fadeDuration ? painter.symbolFadeChange : 1, + 'u_matrix': matrix, + 'u_label_plane_matrix': labelPlaneMatrix, + 'u_coord_matrix': glCoordMatrix, + 'u_is_text': +isText, + 'u_pitch_with_map': +pitchWithMap, + 'u_texsize': texSize, + 'u_texture': 0 + }; +}; + +const symbolSDFUniformValues = ( + functionType: string, + size: { + uSizeT: number; + uSize: number; + }, + rotateInShader: boolean, + pitchWithMap: boolean, + painter: Painter, + matrix: mat4, + labelPlaneMatrix: mat4, + glCoordMatrix: mat4, + isText: boolean, + texSize: [number, number], + isHalo: boolean +): UniformValues => { + const transform = painter.transform; + + return extend(symbolIconUniformValues(functionType, size, + rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix, + glCoordMatrix, isText, texSize), { + 'u_gamma_scale': (pitchWithMap ? Math.cos(transform._pitch) * transform.cameraToCenterDistance : 1), + 'u_device_pixel_ratio': painter.pixelRatio, + 'u_is_halo': +isHalo + }); +}; + +const symbolTextAndIconUniformValues = ( + functionType: string, + size: { + uSizeT: number; + uSize: number; + }, + rotateInShader: boolean, + pitchWithMap: boolean, + painter: Painter, + matrix: mat4, + labelPlaneMatrix: mat4, + glCoordMatrix: mat4, + texSizeSDF: [number, number], + texSizeIcon: [number, number] +): UniformValues => { + return extend(symbolSDFUniformValues(functionType, size, + rotateInShader, pitchWithMap, painter, matrix, labelPlaneMatrix, + glCoordMatrix, true, texSizeSDF, true), { + 'u_texsize_icon': texSizeIcon, + 'u_texture_icon': 1 + }); +}; + +export {symbolIconUniforms, symbolSDFUniforms, symbolIconUniformValues, symbolSDFUniformValues, symbolTextAndIconUniformValues, symbolTextAndIconUniforms}; diff --git a/web/libraries/maplibre-gl/src/render/program/terrain_program.ts b/web/libraries/maplibre-gl/src/render/program/terrain_program.ts new file mode 100644 index 00000000..a0c13ef4 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/program/terrain_program.ts @@ -0,0 +1,93 @@ +import { + Uniform1i, + Uniform1f, + Uniform4f, + UniformMatrix4f +} from '../uniform_binding'; +import type {Context} from '../../gl/context'; +import type {UniformValues, UniformLocations} from '../../render/uniform_binding'; +import {mat4} from 'gl-matrix'; + +export type TerrainPreludeUniformsType = { + 'u_depth': Uniform1i; + 'u_terrain': Uniform1i; + 'u_terrain_dim': Uniform1f; + 'u_terrain_matrix': UniformMatrix4f; + 'u_terrain_unpack': Uniform4f; + 'u_terrain_exaggeration': Uniform1f; +}; + +export type TerrainUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_texture': Uniform1i; + 'u_ele_delta': Uniform1f; +}; + +export type TerrainDepthUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_ele_delta': Uniform1f; +}; + +export type TerrainCoordsUniformsType = { + 'u_matrix': UniformMatrix4f; + 'u_texture': Uniform1i; + 'u_terrain_coords_id': Uniform1f; + 'u_ele_delta': Uniform1f; +}; + +const terrainPreludeUniforms = (context: Context, locations: UniformLocations): TerrainPreludeUniformsType => ({ + 'u_depth': new Uniform1i(context, locations.u_depth), + 'u_terrain': new Uniform1i(context, locations.u_terrain), + 'u_terrain_dim': new Uniform1f(context, locations.u_terrain_dim), + 'u_terrain_matrix': new UniformMatrix4f(context, locations.u_terrain_matrix), + 'u_terrain_unpack': new Uniform4f(context, locations.u_terrain_unpack), + 'u_terrain_exaggeration': new Uniform1f(context, locations.u_terrain_exaggeration) +}); + +const terrainUniforms = (context: Context, locations: UniformLocations): TerrainUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_texture': new Uniform1i(context, locations.u_texture), + 'u_ele_delta': new Uniform1f(context, locations.u_ele_delta) +}); + +const terrainDepthUniforms = (context: Context, locations: UniformLocations): TerrainDepthUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_ele_delta': new Uniform1f(context, locations.u_ele_delta) +}); + +const terrainCoordsUniforms = (context: Context, locations: UniformLocations): TerrainCoordsUniformsType => ({ + 'u_matrix': new UniformMatrix4f(context, locations.u_matrix), + 'u_texture': new Uniform1i(context, locations.u_texture), + 'u_terrain_coords_id': new Uniform1f(context, locations.u_terrain_coords_id), + 'u_ele_delta': new Uniform1f(context, locations.u_ele_delta) +}); + +const terrainUniformValues = ( + matrix: mat4, + eleDelta: number +): UniformValues => ({ + 'u_matrix': matrix, + 'u_texture': 0, + 'u_ele_delta': eleDelta +}); + +const terrainDepthUniformValues = ( + matrix: mat4, + eleDelta: number +): UniformValues => ({ + 'u_matrix': matrix, + 'u_ele_delta': eleDelta +}); + +const terrainCoordsUniformValues = ( + matrix: mat4, + coordsId: number, + eleDelta: number +): UniformValues => ({ + 'u_matrix': matrix, + 'u_terrain_coords_id': coordsId / 255, + 'u_texture': 0, + 'u_ele_delta': eleDelta +}); + +export {terrainUniforms, terrainDepthUniforms, terrainCoordsUniforms, terrainPreludeUniforms, terrainUniformValues, terrainDepthUniformValues, terrainCoordsUniformValues}; diff --git a/web/libraries/maplibre-gl/src/render/render_to_texture.test.ts b/web/libraries/maplibre-gl/src/render/render_to_texture.test.ts new file mode 100644 index 00000000..f53892bd --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/render_to_texture.test.ts @@ -0,0 +1,150 @@ +import {RenderToTexture} from './render_to_texture'; +import type {Painter} from './painter'; +import type {LineStyleLayer} from '../style/style_layer/line_style_layer'; +import type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer'; +import {Context} from '../gl/context'; +import gl from 'gl'; +import {ColorMode} from '../gl/color_mode'; +import {Terrain} from './terrain'; +import {Style} from '../style/style'; +import {Tile} from '../source/tile'; +import {Map} from '../ui/map'; +import {OverscaledTileID} from '../source/tile_id'; +import {SourceCache} from '../source/source_cache'; +import {TerrainSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {FillStyleLayer} from '../style/style_layer/fill_style_layer'; +import {RasterStyleLayer} from '../style/style_layer/raster_style_layer'; +import {HillshadeStyleLayer} from '../style/style_layer/hillshade_style_layer'; +import {BackgroundStyleLayer} from '../style/style_layer/background_style_layer'; + +describe('render to texture', () => { + const backgroundLayer = { + id: 'maine-background', + type: 'background', + source: 'maine', + isHidden: () => false + } as any as BackgroundStyleLayer; + const fillLayer = { + id: 'maine-fill', + type: 'fill', + source: 'maine', + isHidden: () => false + } as any as FillStyleLayer; + const rasterLayer = { + id: 'maine-raster', + type: 'raster', + source: 'maine', + isHidden: () => false + } as any as RasterStyleLayer; + const hillshadeLayer = { + id: 'maine-hillshade', + type: 'line', + source: 'maine', + isHidden: () => false + } as any as HillshadeStyleLayer; + const lineLayer = { + id: 'maine-line', + type: 'line', + source: 'maine', + isHidden: () => false + } as any as LineStyleLayer; + const symbolLayer = { + id: 'maine-symbol', + type: 'symbol', + source: 'maine', + layout: { + 'text-field': 'maine', + 'symbol-placement': 'line' + }, + isHidden: () => false + } as any as SymbolStyleLayer; + + let layersDrawn = 0; + const painter = { + layersDrawn: 0, + context: new Context(gl(1, 1) as any), + transform: {zoom: 10, calculatePosMatrix: () => {}}, + colorModeForRenderPass: () => ColorMode.alphaBlended, + useProgram: () => { return {draw: () => { layersDrawn++; }}; }, + _renderTileClippingMasks: () => {}, + renderLayer: () => {} + } as any as Painter; + const map = {painter} as Map; + + const tile = new Tile(new OverscaledTileID(3, 0, 2, 1, 2), 512); + const sourceCache = { + _source: {minzoom: 0, maxzoom: 2}, + getTileByID: (_id) => tile, + getVisibleCoordinates: () => [tile.tileID] + } as SourceCache; + + const style = { + sourceCaches: {'maine': {getVisibleCoordinates: () => [tile.tileID]}}, + _order: ['maine-fill', 'maine-symbol'], + _layers: { + 'maine-background': backgroundLayer, + 'maine-fill': fillLayer, + 'maine-raster': rasterLayer, + 'maine-hillshade': hillshadeLayer, + 'maine-line': lineLayer, + 'maine-symbol': symbolLayer + } + } as any as Style; + painter.style = style; + map.style = style; + style.map = map; + + const terrain = new Terrain(painter, sourceCache, {} as any as TerrainSpecification); + terrain.sourceCache.getRenderableTiles = () => [tile]; + terrain.sourceCache.getTerrainCoords = () => { return {[tile.tileID.key]: tile.tileID}; }; + map.terrain = terrain; + + const rtt = new RenderToTexture(painter, terrain); + rtt.prepareForRender(style, 0); + painter.renderToTexture = rtt; + + test('check state', () => { + expect(rtt._renderableTiles.map(t => t.tileID.key)).toStrictEqual(['923']); + expect(rtt._coordsDescendingInv).toEqual({'maine': {'923': [{'canonical': {'key': '922', 'x': 1, 'y': 2, 'z': 2}, 'key': '923', 'overscaledZ': 3, 'wrap': 0}]}}); + expect(rtt._coordsDescendingInvStr).toStrictEqual({maine: {'923': '923'}}); + }); + + test('should render text after a line by not adding the text to the stack', () => { + style._order = ['maine-fill', 'maine-symbol']; + rtt.prepareForRender(style, 0); + layersDrawn = 0; + expect(rtt._renderableLayerIds).toStrictEqual(['maine-fill', 'maine-symbol']); + expect(rtt.renderLayer(fillLayer)).toBeTruthy(); + expect(rtt.renderLayer(symbolLayer)).toBeFalsy(); + expect(layersDrawn).toBe(1); + }); + + test('render symbol inbetween of rtt layers', () => { + style._order = ['maine-background', 'maine-fill', 'maine-raster', 'maine-hillshade', 'maine-symbol', 'maine-line', 'maine-symbol']; + rtt.prepareForRender(style, 0); + layersDrawn = 0; + expect(rtt._renderableLayerIds).toStrictEqual(['maine-background', 'maine-fill', 'maine-raster', 'maine-hillshade', 'maine-symbol', 'maine-line', 'maine-symbol']); + expect(rtt.renderLayer(backgroundLayer)).toBeTruthy(); + expect(rtt.renderLayer(fillLayer)).toBeTruthy(); + expect(rtt.renderLayer(rasterLayer)).toBeTruthy(); + expect(rtt.renderLayer(hillshadeLayer)).toBeTruthy(); + expect(rtt.renderLayer(symbolLayer)).toBeFalsy(); + expect(rtt.renderLayer(lineLayer)).toBeTruthy(); + expect(rtt.renderLayer(symbolLayer)).toBeFalsy(); + expect(layersDrawn).toBe(2); + }); + + test('render more symbols inbetween of rtt layers', () => { + style._order = ['maine-background', 'maine-symbol', 'maine-hillshade', 'maine-symbol', 'maine-line', 'maine-symbol']; + rtt.prepareForRender(style, 0); + layersDrawn = 0; + expect(rtt._renderableLayerIds).toStrictEqual(['maine-background', 'maine-symbol', 'maine-hillshade', 'maine-symbol', 'maine-line', 'maine-symbol']); + expect(rtt.renderLayer(backgroundLayer)).toBeTruthy(); + expect(rtt.renderLayer(symbolLayer)).toBeFalsy(); + expect(rtt.renderLayer(hillshadeLayer)).toBeTruthy(); + expect(rtt.renderLayer(symbolLayer)).toBeFalsy(); + expect(rtt.renderLayer(lineLayer)).toBeTruthy(); + expect(rtt.renderLayer(symbolLayer)).toBeFalsy(); + expect(layersDrawn).toBe(3); + }); +}); diff --git a/web/libraries/maplibre-gl/src/render/render_to_texture.ts b/web/libraries/maplibre-gl/src/render/render_to_texture.ts new file mode 100644 index 00000000..a21b2cf5 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/render_to_texture.ts @@ -0,0 +1,197 @@ +import {Painter} from './painter'; +import {Tile} from '../source/tile'; +import {Color} from '@maplibre/maplibre-gl-style-spec'; +import {OverscaledTileID} from '../source/tile_id'; +import {drawTerrain} from './draw_terrain'; +import {Style} from '../style/style'; +import {Terrain} from './terrain'; +import {RenderPool} from '../gl/render_pool'; +import {Texture} from './texture'; +import type {StyleLayer} from '../style/style_layer'; + +/** + * lookup table which layers should rendered to texture + */ +const LAYERS: { [keyof in StyleLayer['type']]?: boolean } = { + background: true, + fill: true, + line: true, + raster: true, + hillshade: true +}; + +/** + * @internal + * A helper class to help define what should be rendered to texture and how + */ +export class RenderToTexture { + painter: Painter; + terrain: Terrain; + pool: RenderPool; + /** + * coordsDescendingInv contains a list of all tiles which should be rendered for one render-to-texture tile + * e.g. render 4 raster-tiles with size 256px to the 512px render-to-texture tile + */ + _coordsDescendingInv: {[_: string]: {[_:string]: Array}}; + /** + * create a string representation of all to tiles rendered to render-to-texture tiles + * this string representation is used to check if tile should be re-rendered. + */ + _coordsDescendingInvStr: {[_: string]: {[_:string]: string}}; + /** + * store for render-stacks + * a render stack is a set of layers which should be rendered into one texture + * every stylesheet can have multiple stacks. A new stack is created if layers which should + * not rendered to texture sit inbetween layers which should rendered to texture. e.g. hillshading or symbols + */ + _stacks: Array>; + /** + * remember the previous processed layer to check if a new stack is needed + */ + _prevType: string; + /** + * a list of tiles that can potentially rendered + */ + _renderableTiles: Array; + /** + * a list of tiles that should be rendered to screen in the next render-call + */ + _rttTiles: Array; + /** + * a list of all layer-ids which should be rendered + */ + _renderableLayerIds: Array; + + constructor(painter: Painter, terrain: Terrain) { + this.painter = painter; + this.terrain = terrain; + this.pool = new RenderPool(painter.context, 30, terrain.sourceCache.tileSize * terrain.qualityFactor); + } + + destruct() { + this.pool.destruct(); + } + + getTexture(tile: Tile): Texture { + return this.pool.getObjectForId(tile.rtt[this._stacks.length - 1].id).texture; + } + + prepareForRender(style: Style, zoom: number) { + this._stacks = []; + this._prevType = null; + this._rttTiles = []; + this._renderableTiles = this.terrain.sourceCache.getRenderableTiles(); + this._renderableLayerIds = style._order.filter(id => !style._layers[id].isHidden(zoom)); + + this._coordsDescendingInv = {}; + for (const id in style.sourceCaches) { + this._coordsDescendingInv[id] = {}; + const tileIDs = style.sourceCaches[id].getVisibleCoordinates(); + for (const tileID of tileIDs) { + const keys = this.terrain.sourceCache.getTerrainCoords(tileID); + for (const key in keys) { + if (!this._coordsDescendingInv[id][key]) this._coordsDescendingInv[id][key] = []; + this._coordsDescendingInv[id][key].push(keys[key]); + } + } + } + + this._coordsDescendingInvStr = {}; + for (const id of style._order) { + const layer = style._layers[id], source = layer.source; + if (LAYERS[layer.type]) { + if (!this._coordsDescendingInvStr[source]) { + this._coordsDescendingInvStr[source] = {}; + for (const key in this._coordsDescendingInv[source]) + this._coordsDescendingInvStr[source][key] = this._coordsDescendingInv[source][key].map(c => c.key).sort().join(); + } + } + } + + // check tiles to render + for (const tile of this._renderableTiles) { + for (const source in this._coordsDescendingInvStr) { + // rerender if there are more coords to render than in the last rendering + const coords = this._coordsDescendingInvStr[source][tile.tileID.key]; + if (coords && coords !== tile.rttCoords[source]) tile.rtt = []; + } + } + } + + /** + * due that switching textures is relatively slow, the render + * layer-by-layer context is not practicable. To bypass this problem + * this lines of code stack all layers and later render all at once. + * Because of the stylesheet possibility to mixing render-to-texture layers + * and 'live'-layers (f.e. symbols) it is necessary to create more stacks. For example + * a symbol-layer is in between of fill-layers. + * @param layer - the layer to render + * @returns if true layer is rendered to texture, otherwise false + */ + renderLayer(layer: StyleLayer): boolean { + if (layer.isHidden(this.painter.transform.zoom)) return false; + + const type = layer.type; + const painter = this.painter; + const isLastLayer = this._renderableLayerIds[this._renderableLayerIds.length - 1] === layer.id; + + // remember background, fill, line & raster layer to render into a stack + if (LAYERS[type]) { + // create a new stack if previous layer was not rendered to texture (f.e. symbols) + if (!this._prevType || !LAYERS[this._prevType]) this._stacks.push([]); + // push current render-to-texture layer to render-stack + this._prevType = type; + this._stacks[this._stacks.length - 1].push(layer.id); + // rendering is done later, all in once + if (!isLastLayer) return true; + } + + // in case a stack is finished render all collected stack-layers into a texture + if (LAYERS[this._prevType] || (LAYERS[type] && isLastLayer)) { + this._prevType = type; + const stack = this._stacks.length - 1, layers = this._stacks[stack] || []; + for (const tile of this._renderableTiles) { + // if render pool is full draw current tiles to screen and free pool + if (this.pool.isFull()) { + drawTerrain(this.painter, this.terrain, this._rttTiles); + this._rttTiles = []; + this.pool.freeAllObjects(); + } + this._rttTiles.push(tile); + // check for cached PoolObject + if (tile.rtt[stack]) { + const obj = this.pool.getObjectForId(tile.rtt[stack].id); + if (obj.stamp === tile.rtt[stack].stamp) { + this.pool.useObject(obj); + continue; + } + } + // get free PoolObject + const obj = this.pool.getOrCreateFreeObject(); + this.pool.useObject(obj); + this.pool.stampObject(obj); + tile.rtt[stack] = {id: obj.id, stamp: obj.stamp}; + // prepare PoolObject for rendering + painter.context.bindFramebuffer.set(obj.fbo.framebuffer); + painter.context.clear({color: Color.transparent, stencil: 0}); + painter.currentStencilSource = undefined; + for (let l = 0; l < layers.length; l++) { + const layer = painter.style._layers[layers[l]]; + const coords = layer.source ? this._coordsDescendingInv[layer.source][tile.tileID.key] : [tile.tileID]; + painter.context.viewport.set([0, 0, obj.fbo.width, obj.fbo.height]); + painter._renderTileClippingMasks(layer, coords); + painter.renderLayer(painter, painter.style.sourceCaches[layer.source], layer, coords); + if (layer.source) tile.rttCoords[layer.source] = this._coordsDescendingInvStr[layer.source][tile.tileID.key]; + } + } + drawTerrain(this.painter, this.terrain, this._rttTiles); + this._rttTiles = []; + this.pool.freeAllObjects(); + + return LAYERS[type]; + } + + return false; + } + +} diff --git a/web/libraries/maplibre-gl/src/render/terrain.test.ts b/web/libraries/maplibre-gl/src/render/terrain.test.ts new file mode 100644 index 00000000..39b384a9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/terrain.test.ts @@ -0,0 +1,268 @@ +import Point from '@mapbox/point-geometry'; +import {Terrain} from './terrain'; +import gl from 'gl'; +import {Context} from '../gl/context'; +import {RGBAImage} from '../util/image'; +import {Texture} from './texture'; +import type {SourceCache} from '../source/source_cache'; +import {OverscaledTileID} from '../source/tile_id'; +import type {TerrainSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {DEMData} from '../data/dem_data'; +import {Tile} from '../source/tile'; +import {Painter} from './painter'; +import {mat4} from 'gl-matrix'; +import {LngLat} from '../geo/lng_lat'; + +describe('Terrain', () => { + test('pointCoordiate should not return null', () => { + expect.assertions(1); + const painter = { + context: new Context(gl(1, 1) as any), + width: 1, + height: 1, + transform: {center: {lng: 0}} + } as any as Painter; + const sourceCache = {} as SourceCache; + const getTileByID = (tileID) : Tile => { + if (tileID !== 'abcd') { + return null as any as Tile; + } + return { + tileID: { + canonical: { + x: 0, + y: 0, + z: 0 + } + } + } as any as Tile; + }; + const terrain = new Terrain(painter, sourceCache, {} as any as TerrainSpecification); + terrain.sourceCache.getTileByID = getTileByID; + const context = painter.context as Context; + const pixels = new Uint8Array([0, 0, 255, 255]); + const image = new RGBAImage({width: 1, height: 1}, pixels); + const imageTexture = new Texture(context, image, context.gl.RGBA); + terrain.getFramebuffer('coords'); // allow init of frame buffers + terrain._fboCoordsTexture.texture = imageTexture.texture; + terrain.coordsIndex.push('abcd'); + + const coordinate = terrain.pointCoordinate(new Point(0, 0)); + + expect(coordinate).not.toBeNull(); + }); + + const setupMercatorOverflow = (centerLng: number) => { + const WORLD_WIDTH = 4; + const painter = { + context: new Context(gl(WORLD_WIDTH, 1) as any), + width: WORLD_WIDTH, + height: 1, + transform: {center: {lng: centerLng}} + } as any as Painter; + const sourceCache = {} as SourceCache; + const terrain = new Terrain(painter, sourceCache, {} as any as TerrainSpecification); + const mockTile = {tileID: {canonical: {x: 0, y: 0, z: 0}}}; + terrain.sourceCache.getTileByID = () => mockTile as any as Tile; + terrain.getElevation = () => 0; + terrain.coordsIndex = ['abcd']; + terrain._coordsTextureSize = WORLD_WIDTH; + const pixels = new Uint8Array([2, 0, 0, 255, 3, 0, 0, 255, 0, 0, 0, 255, 1, 0, 0, 255]); + const image = new RGBAImage({width: WORLD_WIDTH, height: 1}, pixels); + const imageTexture = new Texture(painter.context, image, painter.context.gl.RGBA); + terrain.getFramebuffer('coords'); // allow init of frame buffers + terrain._fboCoordsTexture.texture = imageTexture.texture; + return terrain; + }; + + test( + `pointCoordiate should return negative mercator x + if center.lng is to the right of 180 meridian + and a given point is to the left of 180`, + () => { + expect.assertions(2); + const centerLng = -170; + const pointX = 1; // x in the left half of the 4-px world + const terrain = setupMercatorOverflow(centerLng); + const coordinate = terrain.pointCoordinate(new Point(pointX, 0)); + + expect(coordinate.x).toBeLessThan(0); + expect(coordinate.x).toBeGreaterThan(-1); + }); + + test( + `pointCoordiate should return mercator x greater than 1 + if center.lng is to the left of 180 meridian + and a given point is to the right of 180`, + () => { + expect.assertions(2); + const centerLng = 170; + const pointX = 3; // x in the right half of the 4-px world + const terrain = setupMercatorOverflow(centerLng); + const coordinate = terrain.pointCoordinate(new Point(pointX, 0)); + + expect(coordinate.x).toBeGreaterThan(1); + expect(coordinate.x).toBeLessThan(2); + }); + + test('Calculate tile minimum and maximum elevation', () => { + const tileID = new OverscaledTileID(5, 0, 5, 17, 11); + const tile = new Tile(tileID, 256); + tile.dem = { + min: 0, + max: 100, + getPixels: () => new RGBAImage({width: 1, height: 1}, new Uint8Array(1 * 4)), + getUnpackVector: () => [6553.6, 25.6, 0.1, 10000.0], + } as any as DEMData; + const painter = { + context: new Context(gl(1, 1) as any), + width: 1, + height: 1, + getTileTexture: () => null + } as any as Painter; + const sourceCache = { + _source: {maxzoom: 12}, + _cache: {max: 10}, + getTileByID: () => { + return tile; + }, + } as any as SourceCache; + const terrain = new Terrain( + painter, + sourceCache, + {exaggeration: 2} as any as TerrainSpecification, + ); + + terrain.sourceCache._tiles[tileID.key] = tile; + const {minElevation, maxElevation} = terrain.getMinMaxElevation(tileID); + + expect(minElevation).toBe(0); + expect(maxElevation).toBe(200); + }); + + test('Return null elevation values when no tile', () => { + const tileID = new OverscaledTileID(5, 0, 5, 17, 11); + const painter = { + context: new Context(gl(1, 1) as any), + width: 1, + height: 1, + getTileTexture: () => null + } as any as Painter; + const sourceCache = { + _source: {maxzoom: 12}, + _cache: {max: 10}, + getTileByID: () => null, + } as any as SourceCache; + const terrain = new Terrain( + painter, + sourceCache, + {exaggeration: 2} as any as TerrainSpecification, + ); + + const minMaxNoTile = terrain.getMinMaxElevation(tileID); + + expect(minMaxNoTile.minElevation).toBeNull(); + expect(minMaxNoTile.maxElevation).toBeNull(); + }); + + test('Return null elevation values when no DEM', () => { + const tileID = new OverscaledTileID(5, 0, 5, 17, 11); + const tile = new Tile(tileID, 256); + tile.dem = null as any as DEMData; + const painter = { + context: new Context(gl(1, 1) as any), + width: 1, + height: 1, + getTileTexture: () => null + } as any as Painter; + const sourceCache = { + _source: {maxzoom: 12}, + _cache: {max: 10}, + getTileByID: () => { + return tile; + }, + } as any as SourceCache; + const terrain = new Terrain( + painter, + sourceCache, + {exaggeration: 2} as any as TerrainSpecification, + ); + const minMaxNoDEM = terrain.getMinMaxElevation(tileID); + + expect(minMaxNoDEM.minElevation).toBeNull(); + expect(minMaxNoDEM.maxElevation).toBeNull(); + }); + + test('create mesh with border', () => { + let actualIndexArray; + let actualVertexArray; + const painter = { + context: { + createIndexBuffer: array => { actualIndexArray = Array.from(array.uint16); }, + createVertexBuffer: array => { actualVertexArray = Array.from(array.int16); } + }, + width: 1, + height: 1, + } as any as Painter; + const sourceCache = { + _source: {maxzoom: 12}, + _cache: {max: 10} + } as any as SourceCache; + const terrain = new Terrain( + painter, + sourceCache, + {exaggeration: 1} as any as TerrainSpecification, + ); + terrain.meshSize = 4; + terrain.getTerrainMesh(); + expect(terrain.getMeshFrameDelta(16)).toBe(122.16256373312942); + expect(actualIndexArray).toStrictEqual([0, 5, 6, 0, 6, 1, 1, 6, 7, 1, 7, 2, 2, 7, 8, 2, 8, 3, 3, 8, 9, 3, 9, 4, 5, 10, 11, 5, 11, 6, 6, 11, 12, 6, 12, 7, 7, 12, 13, 7, 13, 8, 8, 13, 14, 8, 14, 9, 10, 15, 16, 10, 16, 11, 11, 16, 17, 11, 17, 12, 12, 17, 18, 12, 18, 13, 13, 18, 19, 13, 19, 14, 15, 20, 21, 15, 21, 16, 16, 21, 22, 16, 22, 17, 17, 22, 23, 17, 23, 18, 18, 23, 24, 18, 24, 19, 35, 36, 38, 35, 38, 37, 25, 28, 26, 25, 27, 28, 37, 38, 40, 37, 40, 39, 27, 30, 28, 27, 29, 30, 39, 40, 42, 39, 42, 41, 29, 32, 30, 29, 31, 32, 41, 42, 44, 41, 44, 43, 31, 34, 32, 31, 33, 34, 45, 46, 48, 45, 48, 47, 55, 58, 56, 55, 57, 58, 47, 48, 50, 47, 50, 49, 57, 60, 58, 57, 59, 60, 49, 50, 52, 49, 52, 51, 59, 62, 60, 59, 61, 62, 51, 52, 54, 51, 54, 53, 61, 64, 62, 61, 63, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]); + expect(actualVertexArray).toStrictEqual([0, 0, 0, 2048, 0, 0, 4096, 0, 0, 6144, 0, 0, 8192, 0, 0, 0, 2048, 0, 2048, 2048, 0, 4096, 2048, 0, 6144, 2048, 0, 8192, 2048, 0, 0, 4096, 0, 2048, 4096, 0, 4096, 4096, 0, 6144, 4096, 0, 8192, 4096, 0, 0, 6144, 0, 2048, 6144, 0, 4096, 6144, 0, 6144, 6144, 0, 8192, 6144, 0, 0, 8192, 0, 2048, 8192, 0, 4096, 8192, 0, 6144, 8192, 0, 8192, 8192, 0, 0, 0, 0, 0, 0, 1, 2048, 0, 0, 2048, 0, 1, 4096, 0, 0, 4096, 0, 1, 6144, 0, 0, 6144, 0, 1, 8192, 0, 0, 8192, 0, 1, 0, 8192, 0, 0, 8192, 1, 2048, 8192, 0, 2048, 8192, 1, 4096, 8192, 0, 4096, 8192, 1, 6144, 8192, 0, 6144, 8192, 1, 8192, 8192, 0, 8192, 8192, 1, 0, 0, 0, 0, 0, 1, 0, 2048, 0, 0, 2048, 1, 0, 4096, 0, 0, 4096, 1, 0, 6144, 0, 0, 6144, 1, 0, 8192, 0, 0, 8192, 1, 8192, 0, 0, 8192, 0, 1, 8192, 2048, 0, 8192, 2048, 1, 8192, 4096, 0, 8192, 4096, 1, 8192, 6144, 0, 8192, 6144, 1, 8192, 8192, 0, 8192, 8192, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]); + }); + + test('interpolation works', () => { + const mockTerrain = { + getDEMElevation: Terrain.prototype.getDEMElevation, + getTerrainData() { + return { + // eslint-disable-next-line camelcase + u_terrain_matrix: mat4.create(), + tile: { + dem: { + dim: 1, + get(x:number, y:number) { + expect(x % 1).toBe(0); + expect(y % 1).toBe(0); + return 100 * x + 10 * y; + } + } + } + }; + } + }; + expect(mockTerrain.getDEMElevation(null, 0, 0)).toBeCloseTo(0); + expect(mockTerrain.getDEMElevation(null, 1, 1)).toBeCloseTo(110); + expect(mockTerrain.getDEMElevation(null, 0, 0.5)).toBeCloseTo(5); + expect(mockTerrain.getDEMElevation(null, 1, 0.5)).toBeCloseTo(105); + expect(mockTerrain.getDEMElevation(null, 0.5, 0)).toBeCloseTo(50); + expect(mockTerrain.getDEMElevation(null, 0.5, 1)).toBeCloseTo(60); + expect(mockTerrain.getDEMElevation(null, 0.4, 0.2)).toBeCloseTo(42); + }); + + test('getElevationForLngLatZoom with lng less than -180 wraps correctly', () => { + const terrain = new Terrain(null, {} as any, {} as any); + + const OVERSCALETILEID_DOES_NOT_THROW = 4; + terrain.getElevation = () => OVERSCALETILEID_DOES_NOT_THROW; + expect(terrain.getElevationForLngLatZoom(new LngLat(-183, 40), 0)).toBe(OVERSCALETILEID_DOES_NOT_THROW); + }); + + test('getMinTileElevationForLngLatZoom with lng less than -180 wraps correctly', () => { + const terrain = new Terrain(null, {} as any, {} as any); + + const OVERSCALETILEID_DOES_NOT_THROW = 4; + terrain.getMinMaxElevation = () => ({minElevation: OVERSCALETILEID_DOES_NOT_THROW, maxElevation: 42}); + expect(terrain.getMinTileElevationForLngLatZoom(new LngLat(-183, 40), 0)).toBe(OVERSCALETILEID_DOES_NOT_THROW); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/render/terrain.ts b/web/libraries/maplibre-gl/src/render/terrain.ts new file mode 100644 index 00000000..47afdd0c --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/terrain.ts @@ -0,0 +1,459 @@ + +import {Tile} from '../source/tile'; +import {mat4, vec2} from 'gl-matrix'; +import {OverscaledTileID} from '../source/tile_id'; +import {RGBAImage} from '../util/image'; +import {warnOnce} from '../util/util'; +import {Pos3dArray, TriangleIndexArray} from '../data/array_types.g'; +import pos3dAttributes from '../data/pos3d_attributes'; +import {SegmentVector} from '../data/segment'; +import {VertexBuffer} from '../gl/vertex_buffer'; +import {IndexBuffer} from '../gl/index_buffer'; +import {Painter} from './painter'; +import {Texture} from '../render/texture'; +import type {Framebuffer} from '../gl/framebuffer'; +import Point from '@mapbox/point-geometry'; +import {MercatorCoordinate, lngFromMercatorX, mercatorXfromLng} from '../geo/mercator_coordinate'; +import {TerrainSourceCache} from '../source/terrain_source_cache'; +import {SourceCache} from '../source/source_cache'; +import {EXTENT} from '../data/extent'; +import type {TerrainSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {LngLat, earthRadius} from '../geo/lng_lat'; + +/** + * @internal + * A terrain GPU related object + */ +export type TerrainData = { + 'u_depth': number; + 'u_terrain': number; + 'u_terrain_dim': number; + 'u_terrain_matrix': mat4; + 'u_terrain_unpack': number[]; + 'u_terrain_exaggeration': number; + texture: WebGLTexture; + depthTexture: WebGLTexture; + tile: Tile; +} + +/** + * @internal + * A terrain mesh object + */ +export type TerrainMesh = { + indexBuffer: IndexBuffer; + vertexBuffer: VertexBuffer; + segments: SegmentVector; +} + +/** + * @internal + * This is the main class which handles most of the 3D Terrain logic. It has the following topics: + * 1) loads raster-dem tiles via the internal sourceCache this.sourceCache + * 2) creates a depth-framebuffer, which is used to calculate the visibility of coordinates + * 3) creates a coords-framebuffer, which is used the get to tile-coordinate for a screen-pixel + * 4) stores all render-to-texture tiles in the this.sourceCache._tiles + * 5) calculates the elevation for a specific tile-coordinate + * 6) creates a terrain-mesh + * + * A note about the GPU resource-usage: + * Framebuffers: + * - one for the depth & coords framebuffer with the size of the map-div. + * - one for rendering a tile to texture with the size of tileSize (= 512x512). + * Textures: + * - one texture for an empty raster-dem tile with size 1x1 + * - one texture for an empty depth-buffer, when terrain is disabled with size 1x1 + * - one texture for an each loaded raster-dem with size of the source.tileSize + * - one texture for the coords-framebuffer with the size of the map-div. + * - one texture for the depth-framebuffer with the size of the map-div. + * - one texture for the encoded tile-coords with the size 2*tileSize (=1024x1024) + * - finally for each render-to-texture tile (= this._tiles) a set of textures + * for each render stack (The stack-concept is documented in painter.ts). + * Normally there exists 1-3 Textures per tile, depending on the stylesheet. + * Each Textures has the size 2*tileSize (= 1024x1024). Also there exists a + * cache of the last 150 newest rendered tiles. + * + */ +export class Terrain { + /** + * The style this terrain crresponds to + */ + painter: Painter; + /** + * the sourcecache this terrain is based on + */ + sourceCache: TerrainSourceCache; + /** + * the TerrainSpecification object passed to this instance + */ + options: TerrainSpecification; + /** + * define the meshSize per tile. + */ + meshSize: number; + /** + * multiplicator for the elevation. Used to make terrain more "extreme". + */ + exaggeration: number; + /** + * to not see pixels in the render-to-texture tiles it is good to render them bigger + * this number is the multiplicator (must be a power of 2) for the current tileSize. + * So to get good results with not too much memory footprint a value of 2 should be fine. + */ + qualityFactor: number; + /** + * holds the framebuffer object in size of the screen to render the coords & depth into a texture. + */ + _fbo: Framebuffer; + _fboCoordsTexture: Texture; + _fboDepthTexture: Texture; + _emptyDepthTexture: Texture; + /** + * GL Objects for the terrain-mesh + * The mesh is a regular mesh, which has the advantage that it can be reused for all tiles. + */ + _mesh: TerrainMesh; + /** + * coords index contains a list of tileID.keys. This index is used to identify + * the tile via the alpha-cannel in the coords-texture. + * As the alpha-channel has 1 Byte a max of 255 tiles can rendered without an error. + */ + coordsIndex: Array; + /** + * tile-coords encoded in the rgb channel, _coordsIndex is in the alpha-channel. + */ + _coordsTexture: Texture; + /** + * accuracy of the coords. 2 * tileSize should be enoughth. + */ + _coordsTextureSize: number; + /** + * variables for an empty dem texture, which is used while the raster-dem tile is loading. + */ + _emptyDemUnpack: number[]; + _emptyDemTexture: Texture; + _emptyDemMatrix: mat4; + /** + * as of overzooming of raster-dem tiles in high zoomlevels, this cache contains + * matrices to transform from vector-tile coords to raster-dem-tile coords. + */ + _demMatrixCache: {[_: string]: { matrix: mat4; coord: OverscaledTileID }}; + + constructor(painter: Painter, sourceCache: SourceCache, options: TerrainSpecification) { + this.painter = painter; + this.sourceCache = new TerrainSourceCache(sourceCache); + this.options = options; + this.exaggeration = typeof options.exaggeration === 'number' ? options.exaggeration : 1.0; + this.qualityFactor = 2; + this.meshSize = 128; + this._demMatrixCache = {}; + this.coordsIndex = []; + this._coordsTextureSize = 1024; + } + + /** + * get the elevation-value from original dem-data for a given tile-coordinate + * @param tileID - the tile to get elevation for + * @param x - between 0 .. EXTENT + * @param y - between 0 .. EXTENT + * @param extent - optional, default 8192 + * @returns the elevation + */ + getDEMElevation(tileID: OverscaledTileID, x: number, y: number, extent: number = EXTENT): number { + if (!(x >= 0 && x < extent && y >= 0 && y < extent)) return 0; + const terrain = this.getTerrainData(tileID); + const dem = terrain.tile?.dem; + if (!dem) + return 0; + + const pos = vec2.transformMat4([] as any, [x / extent * EXTENT, y / extent * EXTENT], terrain.u_terrain_matrix); + const coord = [pos[0] * dem.dim, pos[1] * dem.dim]; + + // bilinear interpolation + const cx = Math.floor(coord[0]), + cy = Math.floor(coord[1]), + tx = coord[0] - cx, + ty = coord[1] - cy; + return ( + dem.get(cx, cy) * (1 - tx) * (1 - ty) + + dem.get(cx + 1, cy) * (tx) * (1 - ty) + + dem.get(cx, cy + 1) * (1 - tx) * (ty) + + dem.get(cx + 1, cy + 1) * (tx) * (ty) + ); + } + + /** + * Get the elevation for given {@link LngLat} in respect of exaggeration. + * @param lnglat - the location + * @param zoom - the zoom + * @returns the elevation + */ + getElevationForLngLatZoom(lnglat: LngLat, zoom: number) { + const {tileID, mercatorX, mercatorY} = this._getOverscaledTileIDFromLngLatZoom(lnglat, zoom); + return this.getElevation(tileID, mercatorX % EXTENT, mercatorY % EXTENT, EXTENT); + } + + /** + * Get the elevation for given coordinate in respect of exaggeration. + * @param tileID - the tile id + * @param x - between 0 .. EXTENT + * @param y - between 0 .. EXTENT + * @param extent - optional, default 8192 + * @returns the elevation + */ + getElevation(tileID: OverscaledTileID, x: number, y: number, extent: number = EXTENT): number { + return this.getDEMElevation(tileID, x, y, extent) * this.exaggeration; + } + + /** + * returns a Terrain Object for a tile. Unless the tile corresponds to data (e.g. tile is loading), return a flat dem object + * @param tileID - the tile to get the terrain for + * @returns the terrain data to use in the program + */ + getTerrainData(tileID: OverscaledTileID): TerrainData { + // create empty DEM Objects, which will used while raster-dem tiles are loading. + // creates an empty depth-buffer texture which is needed, during the initialization process of the 3d mesh.. + if (!this._emptyDemTexture) { + const context = this.painter.context; + const image = new RGBAImage({width: 1, height: 1}, new Uint8Array(1 * 4)); + this._emptyDepthTexture = new Texture(context, image, context.gl.RGBA, {premultiply: false}); + this._emptyDemUnpack = [0, 0, 0, 0]; + this._emptyDemTexture = new Texture(context, new RGBAImage({width: 1, height: 1}), context.gl.RGBA, {premultiply: false}); + this._emptyDemTexture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE); + this._emptyDemMatrix = mat4.identity([] as any); + } + // find covering dem tile and prepare demTexture + const sourceTile = this.sourceCache.getSourceTile(tileID, true); + if (sourceTile && sourceTile.dem && (!sourceTile.demTexture || sourceTile.needsTerrainPrepare)) { + const context = this.painter.context; + sourceTile.demTexture = this.painter.getTileTexture(sourceTile.dem.stride); + if (sourceTile.demTexture) sourceTile.demTexture.update(sourceTile.dem.getPixels(), {premultiply: false}); + else sourceTile.demTexture = new Texture(context, sourceTile.dem.getPixels(), context.gl.RGBA, {premultiply: false}); + sourceTile.demTexture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE); + sourceTile.needsTerrainPrepare = false; + } + // create matrix for lookup in dem data + const matrixKey = sourceTile && (sourceTile + sourceTile.tileID.key) + tileID.key; + if (matrixKey && !this._demMatrixCache[matrixKey]) { + const maxzoom = this.sourceCache.sourceCache._source.maxzoom; + let dz = tileID.canonical.z - sourceTile.tileID.canonical.z; + if (tileID.overscaledZ > tileID.canonical.z) { + if (tileID.canonical.z >= maxzoom) dz = tileID.canonical.z - maxzoom; + else warnOnce('cannot calculate elevation if elevation maxzoom > source.maxzoom'); + } + const dx = tileID.canonical.x - (tileID.canonical.x >> dz << dz); + const dy = tileID.canonical.y - (tileID.canonical.y >> dz << dz); + const demMatrix = mat4.fromScaling(new Float64Array(16) as any, [1 / (EXTENT << dz), 1 / (EXTENT << dz), 0]); + mat4.translate(demMatrix, demMatrix, [dx * EXTENT, dy * EXTENT, 0]); + this._demMatrixCache[tileID.key] = {matrix: demMatrix, coord: tileID}; + } + // return uniform values & textures + return { + 'u_depth': 2, + 'u_terrain': 3, + 'u_terrain_dim': sourceTile && sourceTile.dem && sourceTile.dem.dim || 1, + 'u_terrain_matrix': matrixKey ? this._demMatrixCache[tileID.key].matrix : this._emptyDemMatrix, + 'u_terrain_unpack': sourceTile && sourceTile.dem && sourceTile.dem.getUnpackVector() || this._emptyDemUnpack, + 'u_terrain_exaggeration': this.exaggeration, + texture: (sourceTile && sourceTile.demTexture || this._emptyDemTexture).texture, + depthTexture: (this._fboDepthTexture || this._emptyDepthTexture).texture, + tile: sourceTile + }; + } + + /** + * get a framebuffer as big as the map-div, which will be used to render depth & coords into a texture + * @param texture - the texture + * @returns the frame buffer + */ + getFramebuffer(texture: string): Framebuffer { + const painter = this.painter; + const width = painter.width / devicePixelRatio; + const height = painter.height / devicePixelRatio; + if (this._fbo && (this._fbo.width !== width || this._fbo.height !== height)) { + this._fbo.destroy(); + this._fboCoordsTexture.destroy(); + this._fboDepthTexture.destroy(); + delete this._fbo; + delete this._fboDepthTexture; + delete this._fboCoordsTexture; + } + if (!this._fboCoordsTexture) { + this._fboCoordsTexture = new Texture(painter.context, {width, height, data: null}, painter.context.gl.RGBA, {premultiply: false}); + this._fboCoordsTexture.bind(painter.context.gl.NEAREST, painter.context.gl.CLAMP_TO_EDGE); + } + if (!this._fboDepthTexture) { + this._fboDepthTexture = new Texture(painter.context, {width, height, data: null}, painter.context.gl.RGBA, {premultiply: false}); + this._fboDepthTexture.bind(painter.context.gl.NEAREST, painter.context.gl.CLAMP_TO_EDGE); + } + if (!this._fbo) { + this._fbo = painter.context.createFramebuffer(width, height, true, false); + this._fbo.depthAttachment.set(painter.context.createRenderbuffer(painter.context.gl.DEPTH_COMPONENT16, width, height)); + } + this._fbo.colorAttachment.set(texture === 'coords' ? this._fboCoordsTexture.texture : this._fboDepthTexture.texture); + return this._fbo; + } + + /** + * create coords texture, needed to grab coordinates from canvas + * encode coords coordinate into 4 bytes: + * - 8 lower bits for x + * - 8 lower bits for y + * - 4 higher bits for x + * - 4 higher bits for y + * - 8 bits for coordsIndex (1 .. 255) (= number of terraintile), is later setted in draw_terrain uniform value + * @returns the texture + */ + getCoordsTexture(): Texture { + const context = this.painter.context; + if (this._coordsTexture) return this._coordsTexture; + const data = new Uint8Array(this._coordsTextureSize * this._coordsTextureSize * 4); + for (let y = 0, i = 0; y < this._coordsTextureSize; y++) for (let x = 0; x < this._coordsTextureSize; x++, i += 4) { + data[i + 0] = x & 255; + data[i + 1] = y & 255; + data[i + 2] = ((x >> 8) << 4) | (y >> 8); + data[i + 3] = 0; + } + const image = new RGBAImage({width: this._coordsTextureSize, height: this._coordsTextureSize}, new Uint8Array(data.buffer)); + const texture = new Texture(context, image, context.gl.RGBA, {premultiply: false}); + texture.bind(context.gl.NEAREST, context.gl.CLAMP_TO_EDGE); + this._coordsTexture = texture; + return texture; + } + + /** + * Reads a pixel from the coords-framebuffer and translate this to mercator. + * @param p - Screen-Coordinate + * @returns mercator coordinate for a screen pixel + */ + pointCoordinate(p: Point): MercatorCoordinate { + const rgba = new Uint8Array(4); + const context = this.painter.context, gl = context.gl; + // grab coordinate pixel from coordinates framebuffer + context.bindFramebuffer.set(this.getFramebuffer('coords').framebuffer); + gl.readPixels(p.x, this.painter.height / devicePixelRatio - p.y - 1, 1, 1, gl.RGBA, gl.UNSIGNED_BYTE, rgba); + context.bindFramebuffer.set(null); + // decode coordinates (encoding see getCoordsTexture) + const x = rgba[0] + ((rgba[2] >> 4) << 8); + const y = rgba[1] + ((rgba[2] & 15) << 8); + const tileID = this.coordsIndex[255 - rgba[3]]; + const tile = tileID && this.sourceCache.getTileByID(tileID); + if (!tile) return null; + const coordsSize = this._coordsTextureSize; + const worldSize = (1 << tile.tileID.canonical.z) * coordsSize; + const mercatorX = (tile.tileID.canonical.x * coordsSize + x) / worldSize; + return new MercatorCoordinate( + this._allowMercatorOverflow(p, mercatorX), + (tile.tileID.canonical.y * coordsSize + y) / worldSize, + this.getElevation(tile.tileID, x, y, coordsSize) + ); + } + + /** + * create a regular mesh which will be used by all terrain-tiles + * @returns the created regular mesh + */ + getTerrainMesh(): TerrainMesh { + if (this._mesh) return this._mesh; + const context = this.painter.context; + const vertexArray = new Pos3dArray(); + const indexArray = new TriangleIndexArray(); + const meshSize = this.meshSize; + const delta = EXTENT / meshSize; + const meshSize2 = meshSize * meshSize; + for (let y = 0; y <= meshSize; y++) for (let x = 0; x <= meshSize; x++) + vertexArray.emplaceBack(x * delta, y * delta, 0); + for (let y = 0; y < meshSize2; y += meshSize + 1) for (let x = 0; x < meshSize; x++) { + indexArray.emplaceBack(x + y, meshSize + x + y + 1, meshSize + x + y + 2); + indexArray.emplaceBack(x + y, meshSize + x + y + 2, x + y + 1); + } + // add an extra frame around the mesh to avoid stiching on tile boundaries with different zoomlevels + // first code-block is for top-bottom frame and second for left-right frame + const offsetTop = vertexArray.length, offsetBottom = offsetTop + (meshSize + 1) * 2; + for (const y of [0, 1]) for (let x = 0; x <= meshSize; x++) for (const z of [0, 1]) + vertexArray.emplaceBack(x * delta, y * EXTENT, z); + for (let x = 0; x < meshSize * 2; x += 2) { + indexArray.emplaceBack(offsetBottom + x, offsetBottom + x + 1, offsetBottom + x + 3); + indexArray.emplaceBack(offsetBottom + x, offsetBottom + x + 3, offsetBottom + x + 2); + indexArray.emplaceBack(offsetTop + x, offsetTop + x + 3, offsetTop + x + 1); + indexArray.emplaceBack(offsetTop + x, offsetTop + x + 2, offsetTop + x + 3); + } + const offsetLeft = vertexArray.length, offsetRight = offsetLeft + (meshSize + 1) * 2; + for (const x of [0, 1]) for (let y = 0; y <= meshSize; y++) for (const z of [0, 1]) + vertexArray.emplaceBack(x * EXTENT, y * delta, z); + for (let y = 0; y < meshSize * 2; y += 2) { + indexArray.emplaceBack(offsetLeft + y, offsetLeft + y + 1, offsetLeft + y + 3); + indexArray.emplaceBack(offsetLeft + y, offsetLeft + y + 3, offsetLeft + y + 2); + indexArray.emplaceBack(offsetRight + y, offsetRight + y + 3, offsetRight + y + 1); + indexArray.emplaceBack(offsetRight + y, offsetRight + y + 2, offsetRight + y + 3); + } + this._mesh = { + indexBuffer: context.createIndexBuffer(indexArray), + vertexBuffer: context.createVertexBuffer(vertexArray, pos3dAttributes.members), + segments: SegmentVector.simpleSegment(0, 0, vertexArray.length, indexArray.length) + }; + return this._mesh; + } + + /** + * Calculates a height of the frame around the terrain-mesh to avoid stiching between + * tile boundaries in different zoomlevels. + * @param zoom - current zoomlevel + * @returns the elevation delta in meters + */ + getMeshFrameDelta(zoom: number): number { + // divide by 5 is evaluated by trial & error to get a frame in the right height + return 2 * Math.PI * earthRadius / Math.pow(2, zoom) / 5; + } + + getMinTileElevationForLngLatZoom(lnglat: LngLat, zoom: number) { + const {tileID} = this._getOverscaledTileIDFromLngLatZoom(lnglat, zoom); + return this.getMinMaxElevation(tileID).minElevation ?? 0; + } + + /** + * Get the minimum and maximum elevation contained in a tile. This includes any + * exaggeration included in the terrain. + * + * @param tileID - ID of the tile to be used as a source for the min/max elevation + * @returns the minimum and maximum elevation found in the tile, including the terrain's + * exaggeration + */ + getMinMaxElevation(tileID: OverscaledTileID): {minElevation: number | null; maxElevation: number | null} { + const tile = this.getTerrainData(tileID).tile; + const minMax = {minElevation: null, maxElevation: null}; + if (tile && tile.dem) { + minMax.minElevation = tile.dem.min * this.exaggeration; + minMax.maxElevation = tile.dem.max * this.exaggeration; + } + return minMax; + } + + _getOverscaledTileIDFromLngLatZoom(lnglat: LngLat, zoom: number): { tileID: OverscaledTileID; mercatorX: number; mercatorY: number} { + const mercatorCoordinate = MercatorCoordinate.fromLngLat(lnglat.wrap()); + const worldSize = (1 << zoom) * EXTENT; + const mercatorX = mercatorCoordinate.x * worldSize; + const mercatorY = mercatorCoordinate.y * worldSize; + const tileX = Math.floor(mercatorX / EXTENT), tileY = Math.floor(mercatorY / EXTENT); + const tileID = new OverscaledTileID(zoom, 0, zoom, tileX, tileY); + return { + tileID, + mercatorX, + mercatorY + }; + } + + _allowMercatorOverflow(p: Point, mercatorX: number): number { + const inLeftHalf = p.x < (this.painter.width / 2); + let lng = lngFromMercatorX(mercatorX); + const centerLng = this.painter.transform.center.lng; + if ( + (inLeftHalf && Math.sign(lng) > 0 && Math.sign(centerLng) < 0) || + (!inLeftHalf && Math.sign(lng) < 0 && Math.sign(centerLng) > 0) + ) { + lng = 360 * Math.sign(centerLng) + lng; + return mercatorXfromLng(lng); + } + return mercatorX; + } +} diff --git a/web/libraries/maplibre-gl/src/render/texture.ts b/web/libraries/maplibre-gl/src/render/texture.ts new file mode 100644 index 00000000..99ce6a33 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/texture.ts @@ -0,0 +1,114 @@ +import type {Context} from '../gl/context'; +import type {RGBAImage, AlphaImage} from '../util/image'; +import {isImageBitmap} from '../util/util'; + +export type TextureFormat = WebGLRenderingContextBase['RGBA'] | WebGLRenderingContextBase['ALPHA']; +export type TextureFilter = WebGLRenderingContextBase['LINEAR'] | WebGLRenderingContextBase['LINEAR_MIPMAP_NEAREST'] | WebGLRenderingContextBase['NEAREST']; +export type TextureWrap = WebGLRenderingContextBase['REPEAT'] | WebGLRenderingContextBase['CLAMP_TO_EDGE'] | WebGLRenderingContextBase['MIRRORED_REPEAT']; + +type EmptyImage = { + width: number; + height: number; + data: null; +}; + +type DataTextureImage = RGBAImage | AlphaImage | EmptyImage; +export type TextureImage = TexImageSource | DataTextureImage; + +/** + * @internal + * A `Texture` GL related object + */ +export class Texture { + context: Context; + size: [number, number]; + texture: WebGLTexture; + format: TextureFormat; + filter: TextureFilter; + wrap: TextureWrap; + useMipmap: boolean; + + constructor(context: Context, image: TextureImage, format: TextureFormat, options?: { + premultiply?: boolean; + useMipmap?: boolean; + } | null) { + this.context = context; + this.format = format; + this.texture = context.gl.createTexture(); + this.update(image, options); + } + + update(image: TextureImage, options?: { + premultiply?: boolean; + useMipmap?: boolean; + } | null, position?: { + x: number; + y: number; + }) { + const {width, height} = image as {width: number; height: number}; + const resize = (!this.size || this.size[0] !== width || this.size[1] !== height) && !position; + const {context} = this; + const {gl} = context; + + this.useMipmap = Boolean(options && options.useMipmap); + gl.bindTexture(gl.TEXTURE_2D, this.texture); + + context.pixelStoreUnpackFlipY.set(false); + context.pixelStoreUnpack.set(1); + context.pixelStoreUnpackPremultiplyAlpha.set(this.format === gl.RGBA && (!options || options.premultiply !== false)); + + if (resize) { + this.size = [width, height]; + + if (image instanceof HTMLImageElement || image instanceof HTMLCanvasElement || image instanceof HTMLVideoElement || image instanceof ImageData || isImageBitmap(image)) { + gl.texImage2D(gl.TEXTURE_2D, 0, this.format, this.format, gl.UNSIGNED_BYTE, image); + } else { + gl.texImage2D(gl.TEXTURE_2D, 0, this.format, width, height, 0, this.format, gl.UNSIGNED_BYTE, (image as DataTextureImage).data); + } + + } else { + const {x, y} = position || {x: 0, y: 0}; + if (image instanceof HTMLImageElement || image instanceof HTMLCanvasElement || image instanceof HTMLVideoElement || image instanceof ImageData || isImageBitmap(image)) { + gl.texSubImage2D(gl.TEXTURE_2D, 0, x, y, gl.RGBA, gl.UNSIGNED_BYTE, image); + } else { + gl.texSubImage2D(gl.TEXTURE_2D, 0, x, y, width, height, gl.RGBA, gl.UNSIGNED_BYTE, (image as DataTextureImage).data); + } + } + + if (this.useMipmap && this.isSizePowerOfTwo()) { + gl.generateMipmap(gl.TEXTURE_2D); + } + } + + bind(filter: TextureFilter, wrap: TextureWrap, minFilter?: TextureFilter | null) { + const {context} = this; + const {gl} = context; + gl.bindTexture(gl.TEXTURE_2D, this.texture); + + if (minFilter === gl.LINEAR_MIPMAP_NEAREST && !this.isSizePowerOfTwo()) { + minFilter = gl.LINEAR; + } + + if (filter !== this.filter) { + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, filter); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, minFilter || filter); + this.filter = filter; + } + + if (wrap !== this.wrap) { + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, wrap); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, wrap); + this.wrap = wrap; + } + } + + isSizePowerOfTwo() { + return this.size[0] === this.size[1] && (Math.log(this.size[0]) / Math.LN2) % 1 === 0; + } + + destroy() { + const {gl} = this.context; + gl.deleteTexture(this.texture); + this.texture = null; + } +} diff --git a/web/libraries/maplibre-gl/src/render/uniform_binding.test.ts b/web/libraries/maplibre-gl/src/render/uniform_binding.test.ts new file mode 100644 index 00000000..a80a15e3 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/uniform_binding.test.ts @@ -0,0 +1,122 @@ +import {mat4} from 'gl-matrix'; +import {Context} from '../gl/context'; +import { + Uniform1i, + Uniform1f, + Uniform2f, + Uniform3f, + Uniform4f, + UniformMatrix4f +} from './uniform_binding'; + +describe('Uniform Binding', () => { + test('Uniform1i', () => { + // test counts ensure we don't call the gl.uniform* setters more than expected + expect.assertions(4); + + const context = { + gl: { + uniform1i: () => { expect(true).toBeTruthy(); } + } + } as any as Context; + + const u = new Uniform1i(context, 0); + + expect(u.current).toBe(0); + u.set(1); + expect(u.current).toBe(1); + u.set(1); + u.set(2); + }); + + test('Uniform1f', () => { + expect.assertions(4); + + const context = { + gl: { + uniform1f: () => { expect(true).toBeTruthy(); } + } + } as any as Context; + + const u = new Uniform1f(context, 0); + + expect(u.current).toBe(0); + u.set(1); + expect(u.current).toBe(1); + u.set(1); + u.set(2); + }); + + test('Uniform2f', () => { + expect.assertions(4); + + const context = { + gl: { + uniform2f: () => { expect(true).toBeTruthy(); } + } + } as any as Context; + + const u = new Uniform2f(context, 0); + + expect(u.current).toEqual([0, 0]); + u.set([1, 1]); + expect(u.current).toEqual([1, 1]); + u.set([1, 1]); + u.set([1, 2]); + }); + + test('Uniform3f', () => { + expect.assertions(4); + + const context = { + gl: { + uniform3f: () => { expect(true).toBeTruthy(); } + } + } as any as Context; + + const u = new Uniform3f(context, 0); + + expect(u.current).toEqual([0, 0, 0]); + u.set([1, 1, 1]); + expect(u.current).toEqual([1, 1, 1]); + u.set([1, 1, 1]); + u.set([1, 1, 2]); + }); + + test('Uniform4f', () => { + expect.assertions(4); + + const context = { + gl: { + uniform4f: () => { expect(true).toBeTruthy(); } + } + } as any as Context; + + const u = new Uniform4f(context, 0); + + expect(u.current).toEqual([0, 0, 0, 0]); + u.set([1, 1, 1, 1]); + expect(u.current).toEqual([1, 1, 1, 1]); + u.set([1, 1, 1, 1]); + u.set([2, 1, 1, 1]); + }); + + test('UniformMatrix4f', () => { + expect.assertions(4); + + const context = { + gl: { + uniformMatrix4fv: () => { expect(true).toBeTruthy(); } + } + } as any as Context; + + const u = new UniformMatrix4f(context, 0); + const ident = [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1] as mat4; + expect(u.current).toEqual(new Float32Array([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0])); + u.set(ident); + expect(u.current).toEqual(ident); + u.set(ident); + u.set([2, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1]); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/render/uniform_binding.ts b/web/libraries/maplibre-gl/src/render/uniform_binding.ts new file mode 100644 index 00000000..8f43af5e --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/uniform_binding.ts @@ -0,0 +1,157 @@ +import {Color} from '@maplibre/maplibre-gl-style-spec'; + +import type {Context} from '../gl/context'; +import {mat4, vec2, vec3, vec4} from 'gl-matrix'; + +type $ObjMap any> = { + [K in keyof T]: F extends (v: T[K]) => infer R ? R : never; +}; + +export type UniformValues = $ObjMap(u: Uniform) => V>; +export type UniformLocations = {[_: string]: WebGLUniformLocation}; + +/** + * @internal + * A base uniform abstract class + */ +abstract class Uniform { + gl: WebGLRenderingContext|WebGL2RenderingContext; + location: WebGLUniformLocation; + current: T; + + constructor(context: Context, location: WebGLUniformLocation) { + this.gl = context.gl; + this.location = location; + } + + abstract set(v: T): void; +} + +class Uniform1i extends Uniform { + constructor(context: Context, location: WebGLUniformLocation) { + super(context, location); + this.current = 0; + } + + set(v: number): void { + if (this.current !== v) { + this.current = v; + this.gl.uniform1i(this.location, v); + } + } +} + +class Uniform1f extends Uniform { + constructor(context: Context, location: WebGLUniformLocation) { + super(context, location); + this.current = 0; + } + + set(v: number): void { + if (this.current !== v) { + this.current = v; + this.gl.uniform1f(this.location, v); + } + } +} + +class Uniform2f extends Uniform { + constructor(context: Context, location: WebGLUniformLocation) { + super(context, location); + this.current = [0, 0]; + } + + set(v: vec2): void { + if (v[0] !== this.current[0] || v[1] !== this.current[1]) { + this.current = v; + this.gl.uniform2f(this.location, v[0], v[1]); + } + } +} + +class Uniform3f extends Uniform { + constructor(context: Context, location: WebGLUniformLocation) { + super(context, location); + this.current = [0, 0, 0]; + } + + set(v: vec3): void { + if (v[0] !== this.current[0] || v[1] !== this.current[1] || v[2] !== this.current[2]) { + this.current = v; + this.gl.uniform3f(this.location, v[0], v[1], v[2]); + } + } +} + +class Uniform4f extends Uniform { + constructor(context: Context, location: WebGLUniformLocation) { + super(context, location); + this.current = [0, 0, 0, 0]; + } + + set(v: vec4): void { + if (v[0] !== this.current[0] || v[1] !== this.current[1] || + v[2] !== this.current[2] || v[3] !== this.current[3]) { + this.current = v; + this.gl.uniform4f(this.location, v[0], v[1], v[2], v[3]); + } + } +} + +class UniformColor extends Uniform { + constructor(context: Context, location: WebGLUniformLocation) { + super(context, location); + this.current = Color.transparent; + } + + set(v: Color): void { + if (v.r !== this.current.r || v.g !== this.current.g || + v.b !== this.current.b || v.a !== this.current.a) { + this.current = v; + this.gl.uniform4f(this.location, v.r, v.g, v.b, v.a); + } + } +} + +const emptyMat4 = new Float32Array(16) as mat4; +class UniformMatrix4f extends Uniform { + constructor(context: Context, location: WebGLUniformLocation) { + super(context, location); + this.current = emptyMat4; + } + + set(v: mat4): void { + // The vast majority of matrix comparisons that will trip this set + // happen at i=12 or i=0, so we check those first to avoid lots of + // unnecessary iteration: + if (v[12] !== this.current[12] || v[0] !== this.current[0]) { + this.current = v; + this.gl.uniformMatrix4fv(this.location, false, v); + return; + } + for (let i = 1; i < 16; i++) { + if (v[i] !== this.current[i]) { + this.current = v; + this.gl.uniformMatrix4fv(this.location, false, v); + break; + } + } + } +} + +export { + Uniform, + Uniform1i, + Uniform1f, + Uniform2f, + Uniform3f, + Uniform4f, + UniformColor, + UniformMatrix4f +}; + +/** + * @internal + * A uniform bindings + */ +export type UniformBindings = {[_: string]: Uniform}; diff --git a/web/libraries/maplibre-gl/src/render/update_pattern_positions_in_program.test.ts b/web/libraries/maplibre-gl/src/render/update_pattern_positions_in_program.test.ts new file mode 100644 index 00000000..eeb939c2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/update_pattern_positions_in_program.test.ts @@ -0,0 +1,74 @@ +import {Tile} from '../source/tile'; +import {OverscaledTileID} from '../source/tile_id'; +import {updatePatternPositionsInProgram} from './update_pattern_positions_in_program'; +import {FillStyleLayer} from '../style/style_layer/fill_style_layer'; +import type {CrossFaded} from '../style/properties'; +import type {FillLayerSpecification, ResolvedImage} from '@maplibre/maplibre-gl-style-spec'; +import type {ProgramConfiguration} from '../data/program_configuration'; +import type {ImagePosition} from './image_atlas'; +import type {Rect} from './glyph_atlas'; + +interface MockProgramConfiguration extends ProgramConfiguration { + patternPositions: { + posFrom: Rect; + posTo: Rect; + }; +} + +function constructMockProgramConfiguration(): MockProgramConfiguration { + const mockProgramConfiguration: MockProgramConfiguration = {patternPositions: {}} as any; + mockProgramConfiguration.updatePaintBuffers = jest.fn(); + mockProgramConfiguration.setConstantPatternPositions = (posFrom: ImagePosition, posTo: ImagePosition) => { + // this does not exist on ProgramConfiguration but we want to test the resulting output + mockProgramConfiguration.patternPositions = {posFrom: posFrom.paddedRect, posTo: posTo.paddedRect}; + }; + + return mockProgramConfiguration; +} + +function constructMockFillStyleLayer(): FillStyleLayer { + const layerSpec = { + id: 'mock-layer', + source: 'empty-source', + type: 'fill', + layout: {}, + 'paint': { + 'fill-pattern': [ + 'step', + ['zoom'], + 'zoo_11', + 4, + 'volcano_11' + ] + } + } as FillLayerSpecification; + const layer = new FillStyleLayer(layerSpec); + return layer; +} + +describe('updatePatternPositionsInProgram', () => { + test('geojson tile', () => { + const config = constructMockProgramConfiguration(); + const tile = new Tile(new OverscaledTileID(3, 0, 2, 1, 2), undefined); + tile.imageAtlas = {} as any; + tile.imageAtlas.patternPositions = { + 'volcano_11': {paddedRect: {x: 0, y: 0, w: 0, h: 0}, version: 0, tl: [0, 0], pixelRatio: 1, br: [0, 0], tlbr: [0, 0, 0, 0], displaySize: [0, 0], stretchX: [], stretchY: [], content: [0, 0, 0, 0]}, + }; + const crossFadeResolveImage: CrossFaded = { + from: {name: 'zoo_11', available: false, toString: () => 'zoo_11'}, + to: {name: 'volcano_11', available: false, toString: () => 'volcano_11'} + }; + updatePatternPositionsInProgram( + config, + 'fill-pattern', + crossFadeResolveImage, + tile, + constructMockFillStyleLayer() + ); + // we added this property to just see what the update looks like + expect(config.patternPositions).toEqual({ + posFrom: {x: 0, y: 0, w: 0, h: 0}, + posTo: {x: 0, y: 0, w: 0, h: 0} + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/render/update_pattern_positions_in_program.ts b/web/libraries/maplibre-gl/src/render/update_pattern_positions_in_program.ts new file mode 100644 index 00000000..6ea2edaa --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/update_pattern_positions_in_program.ts @@ -0,0 +1,50 @@ +import type {CrossFaded} from '../style/properties'; +import type {ResolvedImage} from '@maplibre/maplibre-gl-style-spec'; +import type {Tile} from '../source/tile'; +import type {ProgramConfiguration} from '../data/program_configuration'; +import type {FillExtrusionStyleLayer} from '../style/style_layer/fill_extrusion_style_layer'; +import type {FillStyleLayer} from '../style/style_layer/fill_style_layer'; + +/** + * A simple helper shared by draw_fill and draw_fill_extrusions to find the correct pattern positions AND update program. + * For transtionable properties, especially 'fill-pattern' and 'fill-extrusion-pattern', while rendering certain frames + * tile.imageAtlas has been updated by worker to hold the new pattern only, but rendering code is still looking for the previous image. + * The mismatch was causing setConstantPatternPositions method not being called and pixelRatio was always the + * default of 1, instead of actual values set by original map.addImage. + * + * @param programConfiguration - to be used to set pattern position and device pixel ratio. + * @param propertyName - 'fill-pattern' or 'fill-extrusion-pattern' property key + * @param constantPattern - either 'fill-pattern' or 'fill-extrusion-pattern' property value + * @param tile - current tile being drawn + * @param layer - current layer being rendered + */ +export function updatePatternPositionsInProgram( + programConfiguration: ProgramConfiguration, + propertyName: 'fill-pattern' | 'fill-extrusion-pattern', + constantPattern: CrossFaded, + tile: Tile, + layer: FillStyleLayer | FillExtrusionStyleLayer): void { + + if (!constantPattern || !tile || !tile.imageAtlas) { + return; + } + + const patternPositions = tile.imageAtlas.patternPositions; + let posTo = patternPositions[constantPattern.to.toString()]; + let posFrom = patternPositions[constantPattern.from.toString()]; + + // https://github.com/maplibre/maplibre-gl-js/issues/3377 + if (!posTo && posFrom) posTo = posFrom; + if (!posFrom && posTo) posFrom = posTo; + + // try again in case patternPositions has been updated by worker + if (!posTo || !posFrom) { + const transitioned = layer.getPaintProperty(propertyName) as string; + posTo = patternPositions[transitioned]; + posFrom = patternPositions[transitioned]; + } + + if (posTo && posFrom) { + programConfiguration.setConstantPatternPositions(posTo, posFrom); + } +} diff --git a/web/libraries/maplibre-gl/src/render/vertex_array_object.ts b/web/libraries/maplibre-gl/src/render/vertex_array_object.ts new file mode 100644 index 00000000..efe64320 --- /dev/null +++ b/web/libraries/maplibre-gl/src/render/vertex_array_object.ts @@ -0,0 +1,163 @@ + +import type {Program} from './program'; +import type {VertexBuffer} from '../gl/vertex_buffer'; +import type {IndexBuffer} from '../gl/index_buffer'; +import type {Context} from '../gl/context'; + +/** + * @internal + * A vertex array object used to pass data to the webgl code + */ +export class VertexArrayObject { + context: Context; + boundProgram: Program; + boundLayoutVertexBuffer: VertexBuffer; + boundPaintVertexBuffers: Array; + boundIndexBuffer: IndexBuffer; + boundVertexOffset: number; + boundDynamicVertexBuffer: VertexBuffer; + boundDynamicVertexBuffer2: VertexBuffer; + boundDynamicVertexBuffer3: VertexBuffer; + vao: any; + + constructor() { + this.boundProgram = null; + this.boundLayoutVertexBuffer = null; + this.boundPaintVertexBuffers = []; + this.boundIndexBuffer = null; + this.boundVertexOffset = null; + this.boundDynamicVertexBuffer = null; + this.vao = null; + } + + bind(context: Context, + program: Program, + layoutVertexBuffer: VertexBuffer, + paintVertexBuffers: Array, + indexBuffer?: IndexBuffer | null, + vertexOffset?: number | null, + dynamicVertexBuffer?: VertexBuffer | null, + dynamicVertexBuffer2?: VertexBuffer | null, + dynamicVertexBuffer3?: VertexBuffer | null) { + + this.context = context; + + let paintBuffersDiffer = this.boundPaintVertexBuffers.length !== paintVertexBuffers.length; + for (let i = 0; !paintBuffersDiffer && i < paintVertexBuffers.length; i++) { + if (this.boundPaintVertexBuffers[i] !== paintVertexBuffers[i]) { + paintBuffersDiffer = true; + } + } + + const isFreshBindRequired = ( + !this.vao || + this.boundProgram !== program || + this.boundLayoutVertexBuffer !== layoutVertexBuffer || + paintBuffersDiffer || + this.boundIndexBuffer !== indexBuffer || + this.boundVertexOffset !== vertexOffset || + this.boundDynamicVertexBuffer !== dynamicVertexBuffer || + this.boundDynamicVertexBuffer2 !== dynamicVertexBuffer2 || + this.boundDynamicVertexBuffer3 !== dynamicVertexBuffer3 + ); + + if (isFreshBindRequired) { + this.freshBind(program, layoutVertexBuffer, paintVertexBuffers, indexBuffer, vertexOffset, dynamicVertexBuffer, dynamicVertexBuffer2, dynamicVertexBuffer3); + } else { + context.bindVertexArray.set(this.vao); + + if (dynamicVertexBuffer) { + // The buffer may have been updated. Rebind to upload data. + dynamicVertexBuffer.bind(); + } + + if (indexBuffer && indexBuffer.dynamicDraw) { + indexBuffer.bind(); + } + + if (dynamicVertexBuffer2) { + dynamicVertexBuffer2.bind(); + } + + if (dynamicVertexBuffer3) { + dynamicVertexBuffer3.bind(); + } + } + } + + freshBind(program: Program, + layoutVertexBuffer: VertexBuffer, + paintVertexBuffers: Array, + indexBuffer?: IndexBuffer | null, + vertexOffset?: number | null, + dynamicVertexBuffer?: VertexBuffer | null, + dynamicVertexBuffer2?: VertexBuffer | null, + dynamicVertexBuffer3?: VertexBuffer | null) { + + const numNextAttributes = program.numAttributes; + + const context = this.context; + const gl = context.gl; + + if (this.vao) this.destroy(); + this.vao = context.createVertexArray(); + context.bindVertexArray.set(this.vao); + + // store the arguments so that we can verify them when the vao is bound again + this.boundProgram = program; + this.boundLayoutVertexBuffer = layoutVertexBuffer; + this.boundPaintVertexBuffers = paintVertexBuffers; + this.boundIndexBuffer = indexBuffer; + this.boundVertexOffset = vertexOffset; + this.boundDynamicVertexBuffer = dynamicVertexBuffer; + this.boundDynamicVertexBuffer2 = dynamicVertexBuffer2; + this.boundDynamicVertexBuffer3 = dynamicVertexBuffer3; + + layoutVertexBuffer.enableAttributes(gl, program); + for (const vertexBuffer of paintVertexBuffers) { + vertexBuffer.enableAttributes(gl, program); + } + + if (dynamicVertexBuffer) { + dynamicVertexBuffer.enableAttributes(gl, program); + } + if (dynamicVertexBuffer2) { + dynamicVertexBuffer2.enableAttributes(gl, program); + } + if (dynamicVertexBuffer3) { + dynamicVertexBuffer3.enableAttributes(gl, program); + } + + layoutVertexBuffer.bind(); + layoutVertexBuffer.setVertexAttribPointers(gl, program, vertexOffset); + for (const vertexBuffer of paintVertexBuffers) { + vertexBuffer.bind(); + vertexBuffer.setVertexAttribPointers(gl, program, vertexOffset); + } + + if (dynamicVertexBuffer) { + dynamicVertexBuffer.bind(); + dynamicVertexBuffer.setVertexAttribPointers(gl, program, vertexOffset); + } + if (indexBuffer) { + indexBuffer.bind(); + } + if (dynamicVertexBuffer2) { + dynamicVertexBuffer2.bind(); + dynamicVertexBuffer2.setVertexAttribPointers(gl, program, vertexOffset); + } + if (dynamicVertexBuffer3) { + dynamicVertexBuffer3.bind(); + dynamicVertexBuffer3.setVertexAttribPointers(gl, program, vertexOffset); + } + + context.currentNumAttributes = numNextAttributes; + } + + destroy() { + if (this.vao) { + this.context.deleteVertexArray(this.vao); + this.vao = null; + } + } +} diff --git a/web/libraries/maplibre-gl/src/shaders/README.md b/web/libraries/maplibre-gl/src/shaders/README.md new file mode 100644 index 00000000..cdd5f791 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/README.md @@ -0,0 +1,42 @@ +# MapLibre GL JS Shaders + +This repository contains the GLSL shaders + +## Pragmas + +Some variables change type depending on their context: + + - if the variable is the same for all features, we declare it as a `uniform` + - if the variable is different for each feature, we declare it as an `attribute` (in the vertex shader) and an accompanying `varying` (in both the vertex and fragment shaders). + - if the variable is different for each feature and a function of zoom, we declare several `attributes` and `uniforms` then calculate the value using interpolation + +We abstract over this functionality using pragmas. + +```glsl +#pragma mapbox: define highp vec4 color + +main() { + #pragma mapbox: initialize highp vec4 color + ... + fragColor = color; +} +``` + +This program defines a variable within `main` called `color`, initialize the value of `color`, then sets `fragColor` to the value of `color`. + +Pragmas take the following form. + +```glsl +#pragma mapbox: (define|initialize) (lowp|mediump|highp) (float|vec2|vec3|vec4) {name} +``` + +When using pragmas, the following requirements apply. + + - all pragma-defined variables must have both `define` and `initialize` pragmas + - `define` pragmas must be in file scope + - `initialize` pragmas must be in function scope + - all pragma-defined variables defined and initialized in the fragment shader must also be defined and initialized in the vertex shader because `attribute`s are not accessible from the fragment shader + +## Prelude + +The `_prelude.fragment.glsl` and `_prelude.vertex.glsl` files are automatically included in all shaders by the compiler. diff --git a/web/libraries/maplibre-gl/src/shaders/_prelude.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/_prelude.fragment.glsl new file mode 100644 index 00000000..8716b728 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/_prelude.fragment.glsl @@ -0,0 +1,19 @@ +#ifdef GL_ES +precision mediump float; +#else + +#if !defined(lowp) +#define lowp +#endif + +#if !defined(mediump) +#define mediump +#endif + +#if !defined(highp) +#define highp +#endif + +#endif + +out highp vec4 fragColor; diff --git a/web/libraries/maplibre-gl/src/shaders/_prelude.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/_prelude.fragment.glsl.g.ts new file mode 100644 index 00000000..67cedd44 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/_prelude.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default '#ifdef GL_ES\nprecision mediump float;\n#else\n#if !defined(lowp)\n#define lowp\n#endif\n#if !defined(mediump)\n#define mediump\n#endif\n#if !defined(highp)\n#define highp\n#endif\n#endif\n'; diff --git a/web/libraries/maplibre-gl/src/shaders/_prelude.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/_prelude.vertex.glsl new file mode 100644 index 00000000..69b68398 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/_prelude.vertex.glsl @@ -0,0 +1,148 @@ +#ifdef GL_ES +precision highp float; +#else + +#if !defined(lowp) +#define lowp +#endif + +#if !defined(mediump) +#define mediump +#endif + +#if !defined(highp) +#define highp +#endif + +#endif + +// Unpack a pair of values that have been packed into a single float. +// The packed values are assumed to be 8-bit unsigned integers, and are +// packed like so: +// packedValue = floor(input[0]) * 256 + input[1], +vec2 unpack_float(const float packedValue) { + int packedIntValue = int(packedValue); + int v0 = packedIntValue / 256; + return vec2(v0, packedIntValue - v0 * 256); +} + +vec2 unpack_opacity(const float packedOpacity) { + int intOpacity = int(packedOpacity) / 2; + return vec2(float(intOpacity) / 127.0, mod(packedOpacity, 2.0)); +} + +// To minimize the number of ins needed, we encode a 4-component +// color into a pair of floats (i.e. a vec2) as follows: +// [ floor(color.r * 255) * 256 + color.g * 255, +// floor(color.b * 255) * 256 + color.g * 255 ] +vec4 decode_color(const vec2 encodedColor) { + return vec4( + unpack_float(encodedColor[0]) / 255.0, + unpack_float(encodedColor[1]) / 255.0 + ); +} + +// Unpack a pair of paint values and interpolate between them. +float unpack_mix_vec2(const vec2 packedValue, const float t) { + return mix(packedValue[0], packedValue[1], t); +} + +// Unpack a pair of paint values and interpolate between them. +vec4 unpack_mix_color(const vec4 packedColors, const float t) { + vec4 minColor = decode_color(vec2(packedColors[0], packedColors[1])); + vec4 maxColor = decode_color(vec2(packedColors[2], packedColors[3])); + return mix(minColor, maxColor, t); +} + +// The offset depends on how many pixels are between the world origin and the edge of the tile: +// vec2 offset = mod(pixel_coord, size) +// +// At high zoom levels there are a ton of pixels between the world origin and the edge of the tile. +// The glsl spec only guarantees 16 bits of precision for highp floats. We need more than that. +// +// The pixel_coord is passed in as two 16 bit values: +// pixel_coord_upper = floor(pixel_coord / 2^16) +// pixel_coord_lower = mod(pixel_coord, 2^16) +// +// The offset is calculated in a series of steps that should preserve this precision: +vec2 get_pattern_pos(const vec2 pixel_coord_upper, const vec2 pixel_coord_lower, + const vec2 pattern_size, const float tile_units_to_pixels, const vec2 pos) { + + vec2 offset = mod(mod(mod(pixel_coord_upper, pattern_size) * 256.0, pattern_size) * 256.0 + pixel_coord_lower, pattern_size); + return (tile_units_to_pixels * pos + offset) / pattern_size; +} + +// logic for terrain 3d + +#ifdef TERRAIN3D +uniform sampler2D u_terrain; +uniform float u_terrain_dim; +uniform mat4 u_terrain_matrix; +uniform vec4 u_terrain_unpack; +uniform float u_terrain_exaggeration; +uniform highp sampler2D u_depth; +#endif + +// methods for pack/unpack depth value to texture rgba +// https://stackoverflow.com/questions/34963366/encode-floating-point-data-in-a-rgba-texture +const highp vec4 bitSh = vec4(256. * 256. * 256., 256. * 256., 256., 1.); +const highp vec4 bitShifts = vec4(1.) / bitSh; + +highp float unpack(highp vec4 color) { + return dot(color , bitShifts); +} + +// calculate the opacity behind terrain, returns a value between 0 and 1. +highp float depthOpacity(vec3 frag) { + #ifdef TERRAIN3D + // create the delta between frag.z + terrain.z. + highp float d = unpack(texture(u_depth, frag.xy * 0.5 + 0.5)) + 0.0001 - frag.z; + // visibility range is between 0 and 0.002. 0 is visible, 0.002 is fully invisible. + return 1.0 - max(0.0, min(1.0, -d * 500.0)); + #else + return 1.0; + #endif +} + +// calculate the visibility of a coordinate in terrain and return an opacity value. +// if a coordinate is behind the terrain reduce its opacity +float calculate_visibility(vec4 pos) { + #ifdef TERRAIN3D + vec3 frag = pos.xyz / pos.w; + // check if coordingate is fully visible + highp float d = depthOpacity(frag); + if (d > 0.95) return 1.0; + // if not, go some pixel above and check it this point is visible + return (d + depthOpacity(frag + vec3(0.0, 0.01, 0.0))) / 2.0; + #else + return 1.0; + #endif +} + +// grab an elevation value from a raster-dem texture +float ele(vec2 pos) { + #ifdef TERRAIN3D + vec4 rgb = (texture(u_terrain, pos) * 255.0) * u_terrain_unpack; + return rgb.r + rgb.g + rgb.b - u_terrain_unpack.a; + #else + return 0.0; + #endif +} + +// calculate the elevation with linear interpolation for a coordinate +float get_elevation(vec2 pos) { + #ifdef TERRAIN3D + vec2 coord = (u_terrain_matrix * vec4(pos, 0.0, 1.0)).xy * u_terrain_dim + 1.0; + vec2 f = fract(coord); + vec2 c = (floor(coord) + 0.5) / (u_terrain_dim + 2.0); // get the pixel center + float d = 1.0 / (u_terrain_dim + 2.0); + float tl = ele(c); + float tr = ele(c + vec2(d, 0.0)); + float bl = ele(c + vec2(0.0, d)); + float br = ele(c + vec2(d, d)); + float elevation = mix(mix(tl, tr, f.x), mix(bl, br, f.x), f.y); + return elevation * u_terrain_exaggeration; + #else + return 0.0; + #endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/_prelude.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/_prelude.vertex.glsl.g.ts new file mode 100644 index 00000000..ad0eaeb3 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/_prelude.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default '#ifdef GL_ES\nprecision highp float;\n#else\n#if !defined(lowp)\n#define lowp\n#endif\n#if !defined(mediump)\n#define mediump\n#endif\n#if !defined(highp)\n#define highp\n#endif\n#endif\nvec2 unpack_float(const float packedValue) {int packedIntValue=int(packedValue);int v0=packedIntValue/256;return vec2(v0,packedIntValue-v0*256);}vec2 unpack_opacity(const float packedOpacity) {int intOpacity=int(packedOpacity)/2;return vec2(float(intOpacity)/127.0,mod(packedOpacity,2.0));}vec4 decode_color(const vec2 encodedColor) {return vec4(unpack_float(encodedColor[0])/255.0,unpack_float(encodedColor[1])/255.0\n);}float unpack_mix_vec2(const vec2 packedValue,const float t) {return mix(packedValue[0],packedValue[1],t);}vec4 unpack_mix_color(const vec4 packedColors,const float t) {vec4 minColor=decode_color(vec2(packedColors[0],packedColors[1]));vec4 maxColor=decode_color(vec2(packedColors[2],packedColors[3]));return mix(minColor,maxColor,t);}vec2 get_pattern_pos(const vec2 pixel_coord_upper,const vec2 pixel_coord_lower,const vec2 pattern_size,const float tile_units_to_pixels,const vec2 pos) {vec2 offset=mod(mod(mod(pixel_coord_upper,pattern_size)*256.0,pattern_size)*256.0+pixel_coord_lower,pattern_size);return (tile_units_to_pixels*pos+offset)/pattern_size;}\n#ifdef TERRAIN3D\nuniform sampler2D u_terrain;uniform float u_terrain_dim;uniform mat4 u_terrain_matrix;uniform vec4 u_terrain_unpack;uniform float u_terrain_exaggeration;uniform highp sampler2D u_depth;\n#endif\nconst highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitShifts=vec4(1.)/bitSh;highp float unpack(highp vec4 color) {return dot(color,bitShifts);}highp float depthOpacity(vec3 frag) {\n#ifdef TERRAIN3D\nhighp float d=unpack(texture2D(u_depth,frag.xy*0.5+0.5))+0.0001-frag.z;return 1.0-max(0.0,min(1.0,-d*500.0));\n#else\nreturn 1.0;\n#endif\n}float calculate_visibility(vec4 pos) {\n#ifdef TERRAIN3D\nvec3 frag=pos.xyz/pos.w;highp float d=depthOpacity(frag);if (d > 0.95) return 1.0;return (d+depthOpacity(frag+vec3(0.0,0.01,0.0)))/2.0;\n#else\nreturn 1.0;\n#endif\n}float ele(vec2 pos) {\n#ifdef TERRAIN3D\nvec4 rgb=(texture2D(u_terrain,pos)*255.0)*u_terrain_unpack;return rgb.r+rgb.g+rgb.b-u_terrain_unpack.a;\n#else\nreturn 0.0;\n#endif\n}float get_elevation(vec2 pos) {\n#ifdef TERRAIN3D\nvec2 coord=(u_terrain_matrix*vec4(pos,0.0,1.0)).xy*u_terrain_dim+1.0;vec2 f=fract(coord);vec2 c=(floor(coord)+0.5)/(u_terrain_dim+2.0);float d=1.0/(u_terrain_dim+2.0);float tl=ele(c);float tr=ele(c+vec2(d,0.0));float bl=ele(c+vec2(0.0,d));float br=ele(c+vec2(d,d));float elevation=mix(mix(tl,tr,f.x),mix(bl,br,f.x),f.y);return elevation*u_terrain_exaggeration;\n#else\nreturn 0.0;\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/background.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/background.fragment.glsl new file mode 100644 index 00000000..fdb7205e --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/background.fragment.glsl @@ -0,0 +1,10 @@ +uniform vec4 u_color; +uniform float u_opacity; + +void main() { + fragColor = u_color * u_opacity; + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/background.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/background.fragment.glsl.g.ts new file mode 100644 index 00000000..b1487ef8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/background.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform vec4 u_color;uniform float u_opacity;void main() {gl_FragColor=u_color*u_opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/background.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/background.vertex.glsl new file mode 100644 index 00000000..46f9eaa1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/background.vertex.glsl @@ -0,0 +1,7 @@ +in vec2 a_pos; + +uniform mat4 u_matrix; + +void main() { + gl_Position = u_matrix * vec4(a_pos, 0, 1); +} diff --git a/web/libraries/maplibre-gl/src/shaders/background.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/background.vertex.glsl.g.ts new file mode 100644 index 00000000..3c3fb316 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/background.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/background_pattern.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/background_pattern.fragment.glsl new file mode 100644 index 00000000..7cbb25d9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/background_pattern.fragment.glsl @@ -0,0 +1,28 @@ +uniform vec2 u_pattern_tl_a; +uniform vec2 u_pattern_br_a; +uniform vec2 u_pattern_tl_b; +uniform vec2 u_pattern_br_b; +uniform vec2 u_texsize; +uniform float u_mix; +uniform float u_opacity; + +uniform sampler2D u_image; + +in vec2 v_pos_a; +in vec2 v_pos_b; + +void main() { + vec2 imagecoord = mod(v_pos_a, 1.0); + vec2 pos = mix(u_pattern_tl_a / u_texsize, u_pattern_br_a / u_texsize, imagecoord); + vec4 color1 = texture(u_image, pos); + + vec2 imagecoord_b = mod(v_pos_b, 1.0); + vec2 pos2 = mix(u_pattern_tl_b / u_texsize, u_pattern_br_b / u_texsize, imagecoord_b); + vec4 color2 = texture(u_image, pos2); + + fragColor = mix(color1, color2, u_mix) * u_opacity; + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/background_pattern.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/background_pattern.fragment.glsl.g.ts new file mode 100644 index 00000000..efef0d8c --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/background_pattern.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform vec2 u_pattern_tl_a;uniform vec2 u_pattern_br_a;uniform vec2 u_pattern_tl_b;uniform vec2 u_pattern_br_b;uniform vec2 u_texsize;uniform float u_mix;uniform float u_opacity;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(u_pattern_tl_a/u_texsize,u_pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(u_pattern_tl_b/u_texsize,u_pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_mix)*u_opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/background_pattern.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/background_pattern.vertex.glsl new file mode 100644 index 00000000..90a7af24 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/background_pattern.vertex.glsl @@ -0,0 +1,19 @@ +uniform mat4 u_matrix; +uniform vec2 u_pattern_size_a; +uniform vec2 u_pattern_size_b; +uniform vec2 u_pixel_coord_upper; +uniform vec2 u_pixel_coord_lower; +uniform float u_scale_a; +uniform float u_scale_b; +uniform float u_tile_units_to_pixels; + +in vec2 a_pos; +out vec2 v_pos_a; +out vec2 v_pos_b; + +void main() { + gl_Position = u_matrix * vec4(a_pos, 0, 1); + + v_pos_a = get_pattern_pos(u_pixel_coord_upper, u_pixel_coord_lower, u_scale_a * u_pattern_size_a, u_tile_units_to_pixels, a_pos); + v_pos_b = get_pattern_pos(u_pixel_coord_upper, u_pixel_coord_lower, u_scale_b * u_pattern_size_b, u_tile_units_to_pixels, a_pos); +} diff --git a/web/libraries/maplibre-gl/src/shaders/background_pattern.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/background_pattern.vertex.glsl.g.ts new file mode 100644 index 00000000..590e9abd --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/background_pattern.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform mat4 u_matrix;uniform vec2 u_pattern_size_a;uniform vec2 u_pattern_size_b;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_scale_a;uniform float u_scale_b;uniform float u_tile_units_to_pixels;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_a*u_pattern_size_a,u_tile_units_to_pixels,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,u_scale_b*u_pattern_size_b,u_tile_units_to_pixels,a_pos);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/circle.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/circle.fragment.glsl new file mode 100644 index 00000000..cba58bd3 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/circle.fragment.glsl @@ -0,0 +1,36 @@ +in vec3 v_data; +in float v_visibility; + +#pragma mapbox: define highp vec4 color +#pragma mapbox: define mediump float radius +#pragma mapbox: define lowp float blur +#pragma mapbox: define lowp float opacity +#pragma mapbox: define highp vec4 stroke_color +#pragma mapbox: define mediump float stroke_width +#pragma mapbox: define lowp float stroke_opacity + +void main() { + #pragma mapbox: initialize highp vec4 color + #pragma mapbox: initialize mediump float radius + #pragma mapbox: initialize lowp float blur + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize highp vec4 stroke_color + #pragma mapbox: initialize mediump float stroke_width + #pragma mapbox: initialize lowp float stroke_opacity + + vec2 extrude = v_data.xy; + float extrude_length = length(extrude); + + lowp float antialiasblur = v_data.z; + float antialiased_blur = -max(blur, antialiasblur); + + float opacity_t = smoothstep(0.0, antialiased_blur, extrude_length - 1.0); + + float color_t = stroke_width < 0.01 ? 0.0 : smoothstep(antialiased_blur, 0.0, extrude_length - radius / (radius + stroke_width)); + + fragColor = v_visibility * opacity_t * mix(color * opacity, stroke_color * stroke_opacity, color_t); + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/circle.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/circle.fragment.glsl.g.ts new file mode 100644 index 00000000..7d7ca567 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/circle.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'varying vec3 v_data;varying float v_visibility;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define mediump float radius\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define highp vec4 stroke_color\n#pragma mapbox: define mediump float stroke_width\n#pragma mapbox: define lowp float stroke_opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize mediump float radius\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize highp vec4 stroke_color\n#pragma mapbox: initialize mediump float stroke_width\n#pragma mapbox: initialize lowp float stroke_opacity\nvec2 extrude=v_data.xy;float extrude_length=length(extrude);lowp float antialiasblur=v_data.z;float antialiased_blur=-max(blur,antialiasblur);float opacity_t=smoothstep(0.0,antialiased_blur,extrude_length-1.0);float color_t=stroke_width < 0.01 ? 0.0 : smoothstep(antialiased_blur,0.0,extrude_length-radius/(radius+stroke_width));gl_FragColor=v_visibility*opacity_t*mix(color*opacity,stroke_color*stroke_opacity,color_t);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/circle.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/circle.vertex.glsl new file mode 100644 index 00000000..5e8b7499 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/circle.vertex.glsl @@ -0,0 +1,68 @@ +uniform mat4 u_matrix; +uniform bool u_scale_with_map; +uniform bool u_pitch_with_map; +uniform vec2 u_extrude_scale; +uniform lowp float u_device_pixel_ratio; +uniform highp float u_camera_to_center_distance; + +in vec2 a_pos; + +out vec3 v_data; +out float v_visibility; + +#pragma mapbox: define highp vec4 color +#pragma mapbox: define mediump float radius +#pragma mapbox: define lowp float blur +#pragma mapbox: define lowp float opacity +#pragma mapbox: define highp vec4 stroke_color +#pragma mapbox: define mediump float stroke_width +#pragma mapbox: define lowp float stroke_opacity + +void main(void) { + #pragma mapbox: initialize highp vec4 color + #pragma mapbox: initialize mediump float radius + #pragma mapbox: initialize lowp float blur + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize highp vec4 stroke_color + #pragma mapbox: initialize mediump float stroke_width + #pragma mapbox: initialize lowp float stroke_opacity + + // unencode the extrusion vector that we snuck into the a_pos vector + vec2 extrude = vec2(mod(a_pos, 2.0) * 2.0 - 1.0); + + // multiply a_pos by 0.5, since we had it * 2 in order to sneak + // in extrusion data + vec2 circle_center = floor(a_pos * 0.5); + float ele = get_elevation(circle_center); + v_visibility = calculate_visibility(u_matrix * vec4(circle_center, ele, 1.0)); + + if (u_pitch_with_map) { + vec2 corner_position = circle_center; + if (u_scale_with_map) { + corner_position += extrude * (radius + stroke_width) * u_extrude_scale; + } else { + // Pitching the circle with the map effectively scales it with the map + // To counteract the effect for pitch-scale: viewport, we rescale the + // whole circle based on the pitch scaling effect at its central point + vec4 projected_center = u_matrix * vec4(circle_center, 0, 1); + corner_position += extrude * (radius + stroke_width) * u_extrude_scale * (projected_center.w / u_camera_to_center_distance); + } + + gl_Position = u_matrix * vec4(corner_position, ele, 1); + } else { + gl_Position = u_matrix * vec4(circle_center, ele, 1); + + if (u_scale_with_map) { + gl_Position.xy += extrude * (radius + stroke_width) * u_extrude_scale * u_camera_to_center_distance; + } else { + gl_Position.xy += extrude * (radius + stroke_width) * u_extrude_scale * gl_Position.w; + } + } + + // This is a minimum blur distance that serves as a faux-antialiasing for + // the circle. since blur is a ratio of the circle's size and the intent is + // to keep the blur at roughly 1px, the two are inversely related. + lowp float antialiasblur = 1.0 / u_device_pixel_ratio / (radius + stroke_width); + + v_data = vec3(extrude.x, extrude.y, antialiasblur); +} diff --git a/web/libraries/maplibre-gl/src/shaders/circle.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/circle.vertex.glsl.g.ts new file mode 100644 index 00000000..bfced4e7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/circle.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform mat4 u_matrix;uniform bool u_scale_with_map;uniform bool u_pitch_with_map;uniform vec2 u_extrude_scale;uniform lowp float u_device_pixel_ratio;uniform highp float u_camera_to_center_distance;attribute vec2 a_pos;varying vec3 v_data;varying float v_visibility;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define mediump float radius\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define highp vec4 stroke_color\n#pragma mapbox: define mediump float stroke_width\n#pragma mapbox: define lowp float stroke_opacity\nvoid main(void) {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize mediump float radius\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize highp vec4 stroke_color\n#pragma mapbox: initialize mediump float stroke_width\n#pragma mapbox: initialize lowp float stroke_opacity\nvec2 extrude=vec2(mod(a_pos,2.0)*2.0-1.0);vec2 circle_center=floor(a_pos*0.5);float ele=get_elevation(circle_center);v_visibility=calculate_visibility(u_matrix*vec4(circle_center,ele,1.0));if (u_pitch_with_map) {vec2 corner_position=circle_center;if (u_scale_with_map) {corner_position+=extrude*(radius+stroke_width)*u_extrude_scale;} else {vec4 projected_center=u_matrix*vec4(circle_center,0,1);corner_position+=extrude*(radius+stroke_width)*u_extrude_scale*(projected_center.w/u_camera_to_center_distance);}gl_Position=u_matrix*vec4(corner_position,ele,1);} else {gl_Position=u_matrix*vec4(circle_center,ele,1);if (u_scale_with_map) {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*u_camera_to_center_distance;} else {gl_Position.xy+=extrude*(radius+stroke_width)*u_extrude_scale*gl_Position.w;}}lowp float antialiasblur=1.0/u_device_pixel_ratio/(radius+stroke_width);v_data=vec3(extrude.x,extrude.y,antialiasblur);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/clipping_mask.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/clipping_mask.fragment.glsl new file mode 100644 index 00000000..bb3039aa --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/clipping_mask.fragment.glsl @@ -0,0 +1,3 @@ +void main() { + fragColor = vec4(1.0); +} diff --git a/web/libraries/maplibre-gl/src/shaders/clipping_mask.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/clipping_mask.fragment.glsl.g.ts new file mode 100644 index 00000000..dfafeb64 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/clipping_mask.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'void main() {gl_FragColor=vec4(1.0);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/clipping_mask.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/clipping_mask.vertex.glsl new file mode 100644 index 00000000..46f9eaa1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/clipping_mask.vertex.glsl @@ -0,0 +1,7 @@ +in vec2 a_pos; + +uniform mat4 u_matrix; + +void main() { + gl_Position = u_matrix * vec4(a_pos, 0, 1); +} diff --git a/web/libraries/maplibre-gl/src/shaders/clipping_mask.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/clipping_mask.vertex.glsl.g.ts new file mode 100644 index 00000000..3c3fb316 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/clipping_mask.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'attribute vec2 a_pos;uniform mat4 u_matrix;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/collision_box.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/collision_box.fragment.glsl new file mode 100644 index 00000000..582a2798 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/collision_box.fragment.glsl @@ -0,0 +1,20 @@ +in float v_placed; +in float v_notUsed; + +void main() { + + float alpha = 0.5; + + // Red = collision, hide label + fragColor = vec4(1.0, 0.0, 0.0, 1.0) * alpha; + + // Blue = no collision, label is showing + if (v_placed > 0.5) { + fragColor = vec4(0.0, 0.0, 1.0, 0.5) * alpha; + } + + if (v_notUsed > 0.5) { + // This box not used, fade it out + fragColor *= .1; + } +} \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/shaders/collision_box.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/collision_box.fragment.glsl.g.ts new file mode 100644 index 00000000..55df0322 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/collision_box.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'varying float v_placed;varying float v_notUsed;void main() {float alpha=0.5;gl_FragColor=vec4(1.0,0.0,0.0,1.0)*alpha;if (v_placed > 0.5) {gl_FragColor=vec4(0.0,0.0,1.0,0.5)*alpha;}if (v_notUsed > 0.5) {gl_FragColor*=.1;}}'; diff --git a/web/libraries/maplibre-gl/src/shaders/collision_box.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/collision_box.vertex.glsl new file mode 100644 index 00000000..0750aa7a --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/collision_box.vertex.glsl @@ -0,0 +1,27 @@ +in vec2 a_pos; +in vec2 a_anchor_pos; +in vec2 a_extrude; +in vec2 a_placed; +in vec2 a_shift; + +uniform mat4 u_matrix; +uniform vec2 u_extrude_scale; +uniform float u_camera_to_center_distance; + +out float v_placed; +out float v_notUsed; + +void main() { + vec4 projectedPoint = u_matrix * vec4(a_anchor_pos, 0, 1); + highp float camera_to_anchor_distance = projectedPoint.w; + highp float collision_perspective_ratio = clamp( + 0.5 + 0.5 * (u_camera_to_center_distance / camera_to_anchor_distance), + 0.0, // Prevents oversized near-field boxes in pitched/overzoomed tiles + 4.0); + + gl_Position = u_matrix * vec4(a_pos, get_elevation(a_pos), 1.0); + gl_Position.xy += (a_extrude + a_shift) * u_extrude_scale * gl_Position.w * collision_perspective_ratio; + + v_placed = a_placed.x; + v_notUsed = a_placed.y; +} diff --git a/web/libraries/maplibre-gl/src/shaders/collision_box.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/collision_box.vertex.glsl.g.ts new file mode 100644 index 00000000..385fc72f --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/collision_box.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'attribute vec2 a_pos;attribute vec2 a_anchor_pos;attribute vec2 a_extrude;attribute vec2 a_placed;attribute vec2 a_shift;uniform mat4 u_matrix;uniform vec2 u_extrude_scale;uniform float u_camera_to_center_distance;varying float v_placed;varying float v_notUsed;void main() {vec4 projectedPoint=u_matrix*vec4(a_anchor_pos,0,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);gl_Position=u_matrix*vec4(a_pos,get_elevation(a_pos),1.0);gl_Position.xy+=(a_extrude+a_shift)*u_extrude_scale*gl_Position.w*collision_perspective_ratio;v_placed=a_placed.x;v_notUsed=a_placed.y;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/collision_circle.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/collision_circle.fragment.glsl new file mode 100644 index 00000000..f171b426 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/collision_circle.fragment.glsl @@ -0,0 +1,17 @@ +in float v_radius; +in vec2 v_extrude; +in float v_perspective_ratio; +in float v_collision; + +void main() { + float alpha = 0.5 * min(v_perspective_ratio, 1.0); + float stroke_radius = 0.9 * max(v_perspective_ratio, 1.0); + + float distance_to_center = length(v_extrude); + float distance_to_edge = abs(distance_to_center - v_radius); + float opacity_t = smoothstep(-stroke_radius, 0.0, -distance_to_edge); + + vec4 color = mix(vec4(0.0, 0.0, 1.0, 0.5), vec4(1.0, 0.0, 0.0, 1.0), v_collision); + + fragColor = color * alpha * opacity_t; +} diff --git a/web/libraries/maplibre-gl/src/shaders/collision_circle.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/collision_circle.fragment.glsl.g.ts new file mode 100644 index 00000000..99e766d2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/collision_circle.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;void main() {float alpha=0.5*min(v_perspective_ratio,1.0);float stroke_radius=0.9*max(v_perspective_ratio,1.0);float distance_to_center=length(v_extrude);float distance_to_edge=abs(distance_to_center-v_radius);float opacity_t=smoothstep(-stroke_radius,0.0,-distance_to_edge);vec4 color=mix(vec4(0.0,0.0,1.0,0.5),vec4(1.0,0.0,0.0,1.0),v_collision);gl_FragColor=color*alpha*opacity_t;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/collision_circle.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/collision_circle.vertex.glsl new file mode 100644 index 00000000..7d5f1e21 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/collision_circle.vertex.glsl @@ -0,0 +1,59 @@ +in vec2 a_pos; +in float a_radius; +in vec2 a_flags; + +uniform mat4 u_matrix; +uniform mat4 u_inv_matrix; +uniform vec2 u_viewport_size; +uniform float u_camera_to_center_distance; + +out float v_radius; +out vec2 v_extrude; +out float v_perspective_ratio; +out float v_collision; + +vec3 toTilePosition(vec2 screenPos) { + // Shoot a ray towards the ground to reconstruct the depth-value + vec4 rayStart = u_inv_matrix * vec4(screenPos, -1.0, 1.0); + vec4 rayEnd = u_inv_matrix * vec4(screenPos, 1.0, 1.0); + + rayStart.xyz /= rayStart.w; + rayEnd.xyz /= rayEnd.w; + + highp float t = (0.0 - rayStart.z) / (rayEnd.z - rayStart.z); + return mix(rayStart.xyz, rayEnd.xyz, t); +} + +void main() { + vec2 quadCenterPos = a_pos; + float radius = a_radius; + float collision = a_flags.x; + float vertexIdx = a_flags.y; + + vec2 quadVertexOffset = vec2( + mix(-1.0, 1.0, float(vertexIdx >= 2.0)), + mix(-1.0, 1.0, float(vertexIdx >= 1.0 && vertexIdx <= 2.0))); + + vec2 quadVertexExtent = quadVertexOffset * radius; + + // Screen position of the quad might have been computed with different camera parameters. + // Transform the point to a proper position on the current viewport + vec3 tilePos = toTilePosition(quadCenterPos); + vec4 clipPos = u_matrix * vec4(tilePos, 1.0); + + highp float camera_to_anchor_distance = clipPos.w; + highp float collision_perspective_ratio = clamp( + 0.5 + 0.5 * (u_camera_to_center_distance / camera_to_anchor_distance), + 0.0, // Prevents oversized near-field circles in pitched/overzoomed tiles + 4.0); + + // Apply small padding for the anti-aliasing effect to fit the quad + // Note that v_radius and v_extrude are in screen coordinates already + float padding_factor = 1.2; + v_radius = radius; + v_extrude = quadVertexExtent * padding_factor; + v_perspective_ratio = collision_perspective_ratio; + v_collision = collision; + + gl_Position = vec4(clipPos.xyz / clipPos.w, 1.0) + vec4(quadVertexExtent * padding_factor / u_viewport_size * 2.0, 0.0, 0.0); +} diff --git a/web/libraries/maplibre-gl/src/shaders/collision_circle.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/collision_circle.vertex.glsl.g.ts new file mode 100644 index 00000000..3d12634c --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/collision_circle.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'attribute vec2 a_pos;attribute float a_radius;attribute vec2 a_flags;uniform mat4 u_matrix;uniform mat4 u_inv_matrix;uniform vec2 u_viewport_size;uniform float u_camera_to_center_distance;varying float v_radius;varying vec2 v_extrude;varying float v_perspective_ratio;varying float v_collision;vec3 toTilePosition(vec2 screenPos) {vec4 rayStart=u_inv_matrix*vec4(screenPos,-1.0,1.0);vec4 rayEnd =u_inv_matrix*vec4(screenPos, 1.0,1.0);rayStart.xyz/=rayStart.w;rayEnd.xyz /=rayEnd.w;highp float t=(0.0-rayStart.z)/(rayEnd.z-rayStart.z);return mix(rayStart.xyz,rayEnd.xyz,t);}void main() {vec2 quadCenterPos=a_pos;float radius=a_radius;float collision=a_flags.x;float vertexIdx=a_flags.y;vec2 quadVertexOffset=vec2(mix(-1.0,1.0,float(vertexIdx >=2.0)),mix(-1.0,1.0,float(vertexIdx >=1.0 && vertexIdx <=2.0)));vec2 quadVertexExtent=quadVertexOffset*radius;vec3 tilePos=toTilePosition(quadCenterPos);vec4 clipPos=u_matrix*vec4(tilePos,1.0);highp float camera_to_anchor_distance=clipPos.w;highp float collision_perspective_ratio=clamp(0.5+0.5*(u_camera_to_center_distance/camera_to_anchor_distance),0.0,4.0);float padding_factor=1.2;v_radius=radius;v_extrude=quadVertexExtent*padding_factor;v_perspective_ratio=collision_perspective_ratio;v_collision=collision;gl_Position=vec4(clipPos.xyz/clipPos.w,1.0)+vec4(quadVertexExtent*padding_factor/u_viewport_size*2.0,0.0,0.0);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/debug.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/debug.fragment.glsl new file mode 100644 index 00000000..53e0f40d --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/debug.fragment.glsl @@ -0,0 +1,9 @@ +uniform highp vec4 u_color; +uniform sampler2D u_overlay; + +in vec2 v_uv; + +void main() { + vec4 overlay_color = texture(u_overlay, v_uv); + fragColor = mix(u_color, overlay_color, overlay_color.a); +} diff --git a/web/libraries/maplibre-gl/src/shaders/debug.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/debug.fragment.glsl.g.ts new file mode 100644 index 00000000..f3980875 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/debug.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform highp vec4 u_color;uniform sampler2D u_overlay;varying vec2 v_uv;void main() {vec4 overlay_color=texture2D(u_overlay,v_uv);gl_FragColor=mix(u_color,overlay_color,overlay_color.a);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/debug.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/debug.vertex.glsl new file mode 100644 index 00000000..57112938 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/debug.vertex.glsl @@ -0,0 +1,12 @@ +in vec2 a_pos; +out vec2 v_uv; + +uniform mat4 u_matrix; +uniform float u_overlay_scale; + +void main() { + // This vertex shader expects a EXTENT x EXTENT quad, + // The UV co-ordinates for the overlay texture can be calculated using that knowledge + v_uv = a_pos / 8192.0; + gl_Position = u_matrix * vec4(a_pos * u_overlay_scale, get_elevation(a_pos), 1); +} diff --git a/web/libraries/maplibre-gl/src/shaders/debug.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/debug.vertex.glsl.g.ts new file mode 100644 index 00000000..fd855c0e --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/debug.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'attribute vec2 a_pos;varying vec2 v_uv;uniform mat4 u_matrix;uniform float u_overlay_scale;void main() {v_uv=a_pos/8192.0;gl_Position=u_matrix*vec4(a_pos*u_overlay_scale,get_elevation(a_pos),1);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/encode_attribute.test.ts b/web/libraries/maplibre-gl/src/shaders/encode_attribute.test.ts new file mode 100644 index 00000000..e9c37949 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/encode_attribute.test.ts @@ -0,0 +1,11 @@ +import {packUint8ToFloat} from './encode_attribute'; + +test('packUint8ToFloat', () => { + expect(packUint8ToFloat(0, 0)).toBe(0); + expect(packUint8ToFloat(255, 255)).toBe(65535); + expect(packUint8ToFloat(123, 45)).toBe(31533); + + expect(packUint8ToFloat(-1, -1)).toBe(0); + expect(packUint8ToFloat(256, 256)).toBe(65535); + +}); diff --git a/web/libraries/maplibre-gl/src/shaders/encode_attribute.ts b/web/libraries/maplibre-gl/src/shaders/encode_attribute.ts new file mode 100644 index 00000000..2facd79c --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/encode_attribute.ts @@ -0,0 +1,13 @@ +import {clamp} from '../util/util'; + +/** + * Packs two numbers, interpreted as 8-bit unsigned integers, into a single + * float. Unpack them in the shader using the `unpack_float()` function, + * defined in _prelude.vertex.glsl + */ +export function packUint8ToFloat(a: number, b: number) { + // coerce a and b to 8-bit ints + a = clamp(Math.floor(a), 0, 255); + b = clamp(Math.floor(b), 0, 255); + return 256 * a + b; +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/fill.fragment.glsl new file mode 100644 index 00000000..ffd33c16 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill.fragment.glsl @@ -0,0 +1,13 @@ +#pragma mapbox: define highp vec4 color +#pragma mapbox: define lowp float opacity + +void main() { + #pragma mapbox: initialize highp vec4 color + #pragma mapbox: initialize lowp float opacity + + fragColor = color * opacity; + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/fill.fragment.glsl.g.ts new file mode 100644 index 00000000..44947771 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default '#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float opacity\ngl_FragColor=color*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/fill.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/fill.vertex.glsl new file mode 100644 index 00000000..fcafb429 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill.vertex.glsl @@ -0,0 +1,13 @@ +in vec2 a_pos; + +uniform mat4 u_matrix; + +#pragma mapbox: define highp vec4 color +#pragma mapbox: define lowp float opacity + +void main() { + #pragma mapbox: initialize highp vec4 color + #pragma mapbox: initialize lowp float opacity + + gl_Position = u_matrix * vec4(a_pos, 0, 1); +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/fill.vertex.glsl.g.ts new file mode 100644 index 00000000..d135a8d8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'attribute vec2 a_pos;uniform mat4 u_matrix;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float opacity\ngl_Position=u_matrix*vec4(a_pos,0,1);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/fill_extrusion.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/fill_extrusion.fragment.glsl new file mode 100644 index 00000000..c7913c30 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_extrusion.fragment.glsl @@ -0,0 +1,9 @@ +in vec4 v_color; + +void main() { + fragColor = v_color; + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill_extrusion.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/fill_extrusion.fragment.glsl.g.ts new file mode 100644 index 00000000..d29aaf16 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_extrusion.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'varying vec4 v_color;void main() {gl_FragColor=v_color;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/fill_extrusion.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/fill_extrusion.vertex.glsl new file mode 100644 index 00000000..69f9e350 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_extrusion.vertex.glsl @@ -0,0 +1,84 @@ +uniform mat4 u_matrix; +uniform vec3 u_lightcolor; +uniform lowp vec3 u_lightpos; +uniform lowp float u_lightintensity; +uniform float u_vertical_gradient; +uniform lowp float u_opacity; + +in vec2 a_pos; +in vec4 a_normal_ed; + +#ifdef TERRAIN3D + in vec2 a_centroid; +#endif + + +out vec4 v_color; + +#pragma mapbox: define highp float base +#pragma mapbox: define highp float height + +#pragma mapbox: define highp vec4 color + +void main() { + #pragma mapbox: initialize highp float base + #pragma mapbox: initialize highp float height + #pragma mapbox: initialize highp vec4 color + + vec3 normal = a_normal_ed.xyz; + + #ifdef TERRAIN3D + // Raise the "ceiling" of elements by the elevation of the centroid, in meters. + float height_terrain3d_offset = get_elevation(a_centroid); + // To avoid having buildings "hang above a slope", create a "basement" + // by lowering the "floor" of ground-level (and below) elements. + // This is in addition to the elevation of the centroid, in meters. + float base_terrain3d_offset = height_terrain3d_offset - (base > 0.0 ? 0.0 : 10.0); + #else + float height_terrain3d_offset = 0.0; + float base_terrain3d_offset = 0.0; + #endif + // Sub-terranian "floors and ceilings" are clamped to ground-level. + // 3D Terrain offsets, if applicable, are applied on the result. + base = max(0.0, base) + base_terrain3d_offset; + height = max(0.0, height) + height_terrain3d_offset; + + float t = mod(normal.x, 2.0); + + gl_Position = u_matrix * vec4(a_pos, t > 0.0 ? height : base, 1); + + // Relative luminance (how dark/bright is the surface color?) + float colorvalue = color.r * 0.2126 + color.g * 0.7152 + color.b * 0.0722; + + v_color = vec4(0.0, 0.0, 0.0, 1.0); + + // Add slight ambient lighting so no extrusions are totally black + vec4 ambientlight = vec4(0.03, 0.03, 0.03, 1.0); + color += ambientlight; + + // Calculate cos(theta), where theta is the angle between surface normal and diffuse light ray + float directional = clamp(dot(normal / 16384.0, u_lightpos), 0.0, 1.0); + + // Adjust directional so that + // the range of values for highlight/shading is narrower + // with lower light intensity + // and with lighter/brighter surface colors + directional = mix((1.0 - u_lightintensity), max((1.0 - colorvalue + u_lightintensity), 1.0), directional); + + // Add gradient along z axis of side surfaces + if (normal.y != 0.0) { + // This avoids another branching statement, but multiplies by a constant of 0.84 if no vertical gradient, + // and otherwise calculates the gradient based on base + height + directional *= ( + (1.0 - u_vertical_gradient) + + (u_vertical_gradient * clamp((t + base) * pow(height / 150.0, 0.5), mix(0.7, 0.98, 1.0 - u_lightintensity), 1.0))); + } + + // Assign final color based on surface + ambient light color, diffuse light directional, and light color + // with lower bounds adjusted to hue of light + // so that shading is tinted with the complementary (opposite) color to the light color + v_color.r += clamp(color.r * directional * u_lightcolor.r, mix(0.0, 0.3, 1.0 - u_lightcolor.r), 1.0); + v_color.g += clamp(color.g * directional * u_lightcolor.g, mix(0.0, 0.3, 1.0 - u_lightcolor.g), 1.0); + v_color.b += clamp(color.b * directional * u_lightcolor.b, mix(0.0, 0.3, 1.0 - u_lightcolor.b), 1.0); + v_color *= u_opacity; +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill_extrusion.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/fill_extrusion.vertex.glsl.g.ts new file mode 100644 index 00000000..a394685a --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_extrusion.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform mat4 u_matrix;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;uniform float u_vertical_gradient;uniform lowp float u_opacity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\n#ifdef TERRAIN3D\nattribute vec2 a_centroid;\n#endif\nvarying vec4 v_color;\n#pragma mapbox: define highp float base\n#pragma mapbox: define highp float height\n#pragma mapbox: define highp vec4 color\nvoid main() {\n#pragma mapbox: initialize highp float base\n#pragma mapbox: initialize highp float height\n#pragma mapbox: initialize highp vec4 color\nvec3 normal=a_normal_ed.xyz;\n#ifdef TERRAIN3D\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\n#else\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\n#endif\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);gl_Position=u_matrix*vec4(a_pos,t > 0.0 ? height : base,1);float colorvalue=color.r*0.2126+color.g*0.7152+color.b*0.0722;v_color=vec4(0.0,0.0,0.0,1.0);vec4 ambientlight=vec4(0.03,0.03,0.03,1.0);color+=ambientlight;float directional=clamp(dot(normal/16384.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((1.0-colorvalue+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_color.r+=clamp(color.r*directional*u_lightcolor.r,mix(0.0,0.3,1.0-u_lightcolor.r),1.0);v_color.g+=clamp(color.g*directional*u_lightcolor.g,mix(0.0,0.3,1.0-u_lightcolor.g),1.0);v_color.b+=clamp(color.b*directional*u_lightcolor.b,mix(0.0,0.3,1.0-u_lightcolor.b),1.0);v_color*=u_opacity;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/fill_extrusion_pattern.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/fill_extrusion_pattern.fragment.glsl new file mode 100644 index 00000000..c498a367 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_extrusion_pattern.fragment.glsl @@ -0,0 +1,47 @@ +uniform vec2 u_texsize; +uniform float u_fade; + +uniform sampler2D u_image; + +in vec2 v_pos_a; +in vec2 v_pos_b; +in vec4 v_lighting; + +#pragma mapbox: define lowp float base +#pragma mapbox: define lowp float height +#pragma mapbox: define lowp vec4 pattern_from +#pragma mapbox: define lowp vec4 pattern_to +#pragma mapbox: define lowp float pixel_ratio_from +#pragma mapbox: define lowp float pixel_ratio_to + + + +void main() { + #pragma mapbox: initialize lowp float base + #pragma mapbox: initialize lowp float height + #pragma mapbox: initialize mediump vec4 pattern_from + #pragma mapbox: initialize mediump vec4 pattern_to + #pragma mapbox: initialize lowp float pixel_ratio_from + #pragma mapbox: initialize lowp float pixel_ratio_to + + vec2 pattern_tl_a = pattern_from.xy; + vec2 pattern_br_a = pattern_from.zw; + vec2 pattern_tl_b = pattern_to.xy; + vec2 pattern_br_b = pattern_to.zw; + + vec2 imagecoord = mod(v_pos_a, 1.0); + vec2 pos = mix(pattern_tl_a / u_texsize, pattern_br_a / u_texsize, imagecoord); + vec4 color1 = texture(u_image, pos); + + vec2 imagecoord_b = mod(v_pos_b, 1.0); + vec2 pos2 = mix(pattern_tl_b / u_texsize, pattern_br_b / u_texsize, imagecoord_b); + vec4 color2 = texture(u_image, pos2); + + vec4 mixedColor = mix(color1, color2, u_fade); + + fragColor = mixedColor * v_lighting; + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill_extrusion_pattern.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/fill_extrusion_pattern.fragment.glsl.g.ts new file mode 100644 index 00000000..ea1d8ca1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_extrusion_pattern.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\n#pragma mapbox: define lowp float base\n#pragma mapbox: define lowp float height\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float base\n#pragma mapbox: initialize lowp float height\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);vec4 mixedColor=mix(color1,color2,u_fade);gl_FragColor=mixedColor*v_lighting;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/fill_extrusion_pattern.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/fill_extrusion_pattern.vertex.glsl new file mode 100644 index 00000000..eecc343a --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_extrusion_pattern.vertex.glsl @@ -0,0 +1,96 @@ +uniform mat4 u_matrix; +uniform vec2 u_pixel_coord_upper; +uniform vec2 u_pixel_coord_lower; +uniform float u_height_factor; +uniform vec3 u_scale; +uniform float u_vertical_gradient; +uniform lowp float u_opacity; + +uniform vec3 u_lightcolor; +uniform lowp vec3 u_lightpos; +uniform lowp float u_lightintensity; + +in vec2 a_pos; +in vec4 a_normal_ed; + +#ifdef TERRAIN3D + in vec2 a_centroid; +#endif + +out vec2 v_pos_a; +out vec2 v_pos_b; +out vec4 v_lighting; + +#pragma mapbox: define lowp float base +#pragma mapbox: define lowp float height +#pragma mapbox: define lowp vec4 pattern_from +#pragma mapbox: define lowp vec4 pattern_to +#pragma mapbox: define lowp float pixel_ratio_from +#pragma mapbox: define lowp float pixel_ratio_to + +void main() { + #pragma mapbox: initialize lowp float base + #pragma mapbox: initialize lowp float height + #pragma mapbox: initialize mediump vec4 pattern_from + #pragma mapbox: initialize mediump vec4 pattern_to + #pragma mapbox: initialize lowp float pixel_ratio_from + #pragma mapbox: initialize lowp float pixel_ratio_to + + vec2 pattern_tl_a = pattern_from.xy; + vec2 pattern_br_a = pattern_from.zw; + vec2 pattern_tl_b = pattern_to.xy; + vec2 pattern_br_b = pattern_to.zw; + + float tileRatio = u_scale.x; + float fromScale = u_scale.y; + float toScale = u_scale.z; + + vec3 normal = a_normal_ed.xyz; + float edgedistance = a_normal_ed.w; + + vec2 display_size_a = (pattern_br_a - pattern_tl_a) / pixel_ratio_from; + vec2 display_size_b = (pattern_br_b - pattern_tl_b) / pixel_ratio_to; + + #ifdef TERRAIN3D + // Raise the "ceiling" of elements by the elevation of the centroid, in meters. + float height_terrain3d_offset = get_elevation(a_centroid); + // To avoid having buildings "hang above a slope", create a "basement" + // by lowering the "floor" of ground-level (and below) elements. + // This is in addition to the elevation of the centroid, in meters. + float base_terrain3d_offset = height_terrain3d_offset - (base > 0.0 ? 0.0 : 10.0); + #else + float height_terrain3d_offset = 0.0; + float base_terrain3d_offset = 0.0; + #endif + // Sub-terranian "floors and ceilings" are clamped to ground-level. + // 3D Terrain offsets, if applicable, are applied on the result. + base = max(0.0, base) + base_terrain3d_offset; + height = max(0.0, height) + height_terrain3d_offset; + + float t = mod(normal.x, 2.0); + float z = t > 0.0 ? height : base; + + gl_Position = u_matrix * vec4(a_pos, z, 1); + + vec2 pos = normal.x == 1.0 && normal.y == 0.0 && normal.z == 16384.0 + ? a_pos // extrusion top + : vec2(edgedistance, z * u_height_factor); // extrusion side + + v_pos_a = get_pattern_pos(u_pixel_coord_upper, u_pixel_coord_lower, fromScale * display_size_a, tileRatio, pos); + v_pos_b = get_pattern_pos(u_pixel_coord_upper, u_pixel_coord_lower, toScale * display_size_b, tileRatio, pos); + + v_lighting = vec4(0.0, 0.0, 0.0, 1.0); + float directional = clamp(dot(normal / 16383.0, u_lightpos), 0.0, 1.0); + directional = mix((1.0 - u_lightintensity), max((0.5 + u_lightintensity), 1.0), directional); + + if (normal.y != 0.0) { + // This avoids another branching statement, but multiplies by a constant of 0.84 if no vertical gradient, + // and otherwise calculates the gradient based on base + height + directional *= ( + (1.0 - u_vertical_gradient) + + (u_vertical_gradient * clamp((t + base) * pow(height / 150.0, 0.5), mix(0.7, 0.98, 1.0 - u_lightintensity), 1.0))); + } + + v_lighting.rgb += clamp(directional * u_lightcolor, mix(vec3(0.0), vec3(0.3), 1.0 - u_lightcolor), vec3(1.0)); + v_lighting *= u_opacity; +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill_extrusion_pattern.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/fill_extrusion_pattern.vertex.glsl.g.ts new file mode 100644 index 00000000..e204d825 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_extrusion_pattern.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform float u_height_factor;uniform vec3 u_scale;uniform float u_vertical_gradient;uniform lowp float u_opacity;uniform vec3 u_lightcolor;uniform lowp vec3 u_lightpos;uniform lowp float u_lightintensity;attribute vec2 a_pos;attribute vec4 a_normal_ed;\n#ifdef TERRAIN3D\nattribute vec2 a_centroid;\n#endif\nvarying vec2 v_pos_a;varying vec2 v_pos_b;varying vec4 v_lighting;\n#pragma mapbox: define lowp float base\n#pragma mapbox: define lowp float height\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float base\n#pragma mapbox: initialize lowp float height\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec3 normal=a_normal_ed.xyz;float edgedistance=a_normal_ed.w;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;\n#ifdef TERRAIN3D\nfloat height_terrain3d_offset=get_elevation(a_centroid);float base_terrain3d_offset=height_terrain3d_offset-(base > 0.0 ? 0.0 : 10.0);\n#else\nfloat height_terrain3d_offset=0.0;float base_terrain3d_offset=0.0;\n#endif\nbase=max(0.0,base)+base_terrain3d_offset;height=max(0.0,height)+height_terrain3d_offset;float t=mod(normal.x,2.0);float z=t > 0.0 ? height : base;gl_Position=u_matrix*vec4(a_pos,z,1);vec2 pos=normal.x==1.0 && normal.y==0.0 && normal.z==16384.0\n? a_pos\n: vec2(edgedistance,z*u_height_factor);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,pos);v_lighting=vec4(0.0,0.0,0.0,1.0);float directional=clamp(dot(normal/16383.0,u_lightpos),0.0,1.0);directional=mix((1.0-u_lightintensity),max((0.5+u_lightintensity),1.0),directional);if (normal.y !=0.0) {directional*=((1.0-u_vertical_gradient)+(u_vertical_gradient*clamp((t+base)*pow(height/150.0,0.5),mix(0.7,0.98,1.0-u_lightintensity),1.0)));}v_lighting.rgb+=clamp(directional*u_lightcolor,mix(vec3(0.0),vec3(0.3),1.0-u_lightcolor),vec3(1.0));v_lighting*=u_opacity;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/fill_outline.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/fill_outline.fragment.glsl new file mode 100644 index 00000000..f13b94ed --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_outline.fragment.glsl @@ -0,0 +1,17 @@ +in vec2 v_pos; + +#pragma mapbox: define highp vec4 outline_color +#pragma mapbox: define lowp float opacity + +void main() { + #pragma mapbox: initialize highp vec4 outline_color + #pragma mapbox: initialize lowp float opacity + + float dist = length(v_pos - gl_FragCoord.xy); + float alpha = 1.0 - smoothstep(0.0, 1.0, dist); + fragColor = outline_color * (alpha * opacity); + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill_outline.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/fill_outline.fragment.glsl.g.ts new file mode 100644 index 00000000..b276f3ae --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_outline.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'varying vec2 v_pos;\n#pragma mapbox: define highp vec4 outline_color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 outline_color\n#pragma mapbox: initialize lowp float opacity\nfloat dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=outline_color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/fill_outline.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/fill_outline.vertex.glsl new file mode 100644 index 00000000..a4a654fe --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_outline.vertex.glsl @@ -0,0 +1,17 @@ +in vec2 a_pos; + +uniform mat4 u_matrix; +uniform vec2 u_world; + +out vec2 v_pos; + +#pragma mapbox: define highp vec4 outline_color +#pragma mapbox: define lowp float opacity + +void main() { + #pragma mapbox: initialize highp vec4 outline_color + #pragma mapbox: initialize lowp float opacity + + gl_Position = u_matrix * vec4(a_pos, 0, 1); + v_pos = (gl_Position.xy / gl_Position.w + 1.0) / 2.0 * u_world; +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill_outline.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/fill_outline.vertex.glsl.g.ts new file mode 100644 index 00000000..6875238e --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_outline.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'attribute vec2 a_pos;uniform mat4 u_matrix;uniform vec2 u_world;varying vec2 v_pos;\n#pragma mapbox: define highp vec4 outline_color\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 outline_color\n#pragma mapbox: initialize lowp float opacity\ngl_Position=u_matrix*vec4(a_pos,0,1);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/fill_outline_pattern.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/fill_outline_pattern.fragment.glsl new file mode 100644 index 00000000..07984c71 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_outline_pattern.fragment.glsl @@ -0,0 +1,43 @@ + +uniform vec2 u_texsize; +uniform sampler2D u_image; +uniform float u_fade; + +in vec2 v_pos_a; +in vec2 v_pos_b; +in vec2 v_pos; + +#pragma mapbox: define lowp float opacity +#pragma mapbox: define lowp vec4 pattern_from +#pragma mapbox: define lowp vec4 pattern_to + +void main() { + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize mediump vec4 pattern_from + #pragma mapbox: initialize mediump vec4 pattern_to + + vec2 pattern_tl_a = pattern_from.xy; + vec2 pattern_br_a = pattern_from.zw; + vec2 pattern_tl_b = pattern_to.xy; + vec2 pattern_br_b = pattern_to.zw; + + vec2 imagecoord = mod(v_pos_a, 1.0); + vec2 pos = mix(pattern_tl_a / u_texsize, pattern_br_a / u_texsize, imagecoord); + vec4 color1 = texture(u_image, pos); + + vec2 imagecoord_b = mod(v_pos_b, 1.0); + vec2 pos2 = mix(pattern_tl_b / u_texsize, pattern_br_b / u_texsize, imagecoord_b); + vec4 color2 = texture(u_image, pos2); + + // find distance to outline for alpha interpolation + + float dist = length(v_pos - gl_FragCoord.xy); + float alpha = 1.0 - smoothstep(0.0, 1.0, dist); + + + fragColor = mix(color1, color2, u_fade) * alpha * opacity; + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill_outline_pattern.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/fill_outline_pattern.fragment.glsl.g.ts new file mode 100644 index 00000000..be452f34 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_outline_pattern.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform vec2 u_texsize;uniform sampler2D u_image;uniform float u_fade;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);float dist=length(v_pos-gl_FragCoord.xy);float alpha=1.0-smoothstep(0.0,1.0,dist);gl_FragColor=mix(color1,color2,u_fade)*alpha*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/fill_outline_pattern.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/fill_outline_pattern.vertex.glsl new file mode 100644 index 00000000..f19ddd6e --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_outline_pattern.vertex.glsl @@ -0,0 +1,44 @@ +uniform mat4 u_matrix; +uniform vec2 u_world; +uniform vec2 u_pixel_coord_upper; +uniform vec2 u_pixel_coord_lower; +uniform vec3 u_scale; + +in vec2 a_pos; + +out vec2 v_pos_a; +out vec2 v_pos_b; +out vec2 v_pos; + +#pragma mapbox: define lowp float opacity +#pragma mapbox: define lowp vec4 pattern_from +#pragma mapbox: define lowp vec4 pattern_to +#pragma mapbox: define lowp float pixel_ratio_from +#pragma mapbox: define lowp float pixel_ratio_to + +void main() { + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize mediump vec4 pattern_from + #pragma mapbox: initialize mediump vec4 pattern_to + #pragma mapbox: initialize lowp float pixel_ratio_from + #pragma mapbox: initialize lowp float pixel_ratio_to + + vec2 pattern_tl_a = pattern_from.xy; + vec2 pattern_br_a = pattern_from.zw; + vec2 pattern_tl_b = pattern_to.xy; + vec2 pattern_br_b = pattern_to.zw; + + float tileRatio = u_scale.x; + float fromScale = u_scale.y; + float toScale = u_scale.z; + + gl_Position = u_matrix * vec4(a_pos, 0, 1); + + vec2 display_size_a = (pattern_br_a - pattern_tl_a) / pixel_ratio_from; + vec2 display_size_b = (pattern_br_b - pattern_tl_b) / pixel_ratio_to; + + v_pos_a = get_pattern_pos(u_pixel_coord_upper, u_pixel_coord_lower, fromScale * display_size_a, tileRatio, a_pos); + v_pos_b = get_pattern_pos(u_pixel_coord_upper, u_pixel_coord_lower, toScale * display_size_b, tileRatio, a_pos); + + v_pos = (gl_Position.xy / gl_Position.w + 1.0) / 2.0 * u_world; +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill_outline_pattern.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/fill_outline_pattern.vertex.glsl.g.ts new file mode 100644 index 00000000..8136e1ee --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_outline_pattern.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform mat4 u_matrix;uniform vec2 u_world;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;varying vec2 v_pos;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;gl_Position=u_matrix*vec4(a_pos,0,1);vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileRatio,a_pos);v_pos=(gl_Position.xy/gl_Position.w+1.0)/2.0*u_world;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/fill_pattern.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/fill_pattern.fragment.glsl new file mode 100644 index 00000000..989784a6 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_pattern.fragment.glsl @@ -0,0 +1,39 @@ +#ifdef GL_ES + precision highp float; +#endif +uniform vec2 u_texsize; +uniform float u_fade; + +uniform sampler2D u_image; + +in vec2 v_pos_a; +in vec2 v_pos_b; + +#pragma mapbox: define lowp float opacity +#pragma mapbox: define lowp vec4 pattern_from +#pragma mapbox: define lowp vec4 pattern_to + +void main() { + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize mediump vec4 pattern_from + #pragma mapbox: initialize mediump vec4 pattern_to + + vec2 pattern_tl_a = pattern_from.xy; + vec2 pattern_br_a = pattern_from.zw; + vec2 pattern_tl_b = pattern_to.xy; + vec2 pattern_br_b = pattern_to.zw; + + vec2 imagecoord = mod(v_pos_a, 1.0); + vec2 pos = mix(pattern_tl_a / u_texsize, pattern_br_a / u_texsize, imagecoord); + vec4 color1 = texture(u_image, pos); + + vec2 imagecoord_b = mod(v_pos_b, 1.0); + vec2 pos2 = mix(pattern_tl_b / u_texsize, pattern_br_b / u_texsize, imagecoord_b); + vec4 color2 = texture(u_image, pos2); + + fragColor = mix(color1, color2, u_fade) * opacity; + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill_pattern.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/fill_pattern.fragment.glsl.g.ts new file mode 100644 index 00000000..ba08cecc --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_pattern.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default '#ifdef GL_ES\nprecision highp float;\n#endif\nuniform vec2 u_texsize;uniform float u_fade;uniform sampler2D u_image;varying vec2 v_pos_a;varying vec2 v_pos_b;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;vec2 imagecoord=mod(v_pos_a,1.0);vec2 pos=mix(pattern_tl_a/u_texsize,pattern_br_a/u_texsize,imagecoord);vec4 color1=texture2D(u_image,pos);vec2 imagecoord_b=mod(v_pos_b,1.0);vec2 pos2=mix(pattern_tl_b/u_texsize,pattern_br_b/u_texsize,imagecoord_b);vec4 color2=texture2D(u_image,pos2);gl_FragColor=mix(color1,color2,u_fade)*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/fill_pattern.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/fill_pattern.vertex.glsl new file mode 100644 index 00000000..abdaafb2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_pattern.vertex.glsl @@ -0,0 +1,39 @@ +uniform mat4 u_matrix; +uniform vec2 u_pixel_coord_upper; +uniform vec2 u_pixel_coord_lower; +uniform vec3 u_scale; + +in vec2 a_pos; + +out vec2 v_pos_a; +out vec2 v_pos_b; + +#pragma mapbox: define lowp float opacity +#pragma mapbox: define lowp vec4 pattern_from +#pragma mapbox: define lowp vec4 pattern_to +#pragma mapbox: define lowp float pixel_ratio_from +#pragma mapbox: define lowp float pixel_ratio_to + +void main() { + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize mediump vec4 pattern_from + #pragma mapbox: initialize mediump vec4 pattern_to + #pragma mapbox: initialize lowp float pixel_ratio_from + #pragma mapbox: initialize lowp float pixel_ratio_to + + vec2 pattern_tl_a = pattern_from.xy; + vec2 pattern_br_a = pattern_from.zw; + vec2 pattern_tl_b = pattern_to.xy; + vec2 pattern_br_b = pattern_to.zw; + + float tileZoomRatio = u_scale.x; + float fromScale = u_scale.y; + float toScale = u_scale.z; + + vec2 display_size_a = (pattern_br_a - pattern_tl_a) / pixel_ratio_from; + vec2 display_size_b = (pattern_br_b - pattern_tl_b) / pixel_ratio_to; + gl_Position = u_matrix * vec4(a_pos, 0, 1); + + v_pos_a = get_pattern_pos(u_pixel_coord_upper, u_pixel_coord_lower, fromScale * display_size_a, tileZoomRatio, a_pos); + v_pos_b = get_pattern_pos(u_pixel_coord_upper, u_pixel_coord_lower, toScale * display_size_b, tileZoomRatio, a_pos); +} diff --git a/web/libraries/maplibre-gl/src/shaders/fill_pattern.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/fill_pattern.vertex.glsl.g.ts new file mode 100644 index 00000000..b5d74628 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/fill_pattern.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform mat4 u_matrix;uniform vec2 u_pixel_coord_upper;uniform vec2 u_pixel_coord_lower;uniform vec3 u_scale;attribute vec2 a_pos;varying vec2 v_pos_a;varying vec2 v_pos_b;\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;gl_Position=u_matrix*vec4(a_pos,0,1);v_pos_a=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,fromScale*display_size_a,tileZoomRatio,a_pos);v_pos_b=get_pattern_pos(u_pixel_coord_upper,u_pixel_coord_lower,toScale*display_size_b,tileZoomRatio,a_pos);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/heatmap.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/heatmap.fragment.glsl new file mode 100644 index 00000000..bd6aaa51 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/heatmap.fragment.glsl @@ -0,0 +1,22 @@ +uniform highp float u_intensity; + +in vec2 v_extrude; + +#pragma mapbox: define highp float weight + +// Gaussian kernel coefficient: 1 / sqrt(2 * PI) +#define GAUSS_COEF 0.3989422804014327 + +void main() { + #pragma mapbox: initialize highp float weight + + // Kernel density estimation with a Gaussian kernel of size 5x5 + float d = -0.5 * 3.0 * 3.0 * dot(v_extrude, v_extrude); + float val = weight * u_intensity * GAUSS_COEF * exp(d); + + fragColor = vec4(val, 1.0, 1.0, 1.0); + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/heatmap.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/heatmap.fragment.glsl.g.ts new file mode 100644 index 00000000..d4e7e405 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/heatmap.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform highp float u_intensity;varying vec2 v_extrude;\n#pragma mapbox: define highp float weight\n#define GAUSS_COEF 0.3989422804014327\nvoid main() {\n#pragma mapbox: initialize highp float weight\nfloat d=-0.5*3.0*3.0*dot(v_extrude,v_extrude);float val=weight*u_intensity*GAUSS_COEF*exp(d);gl_FragColor=vec4(val,1.0,1.0,1.0);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/heatmap.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/heatmap.vertex.glsl new file mode 100644 index 00000000..1b4bdcb6 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/heatmap.vertex.glsl @@ -0,0 +1,54 @@ + +uniform mat4 u_matrix; +uniform float u_extrude_scale; +uniform float u_opacity; +uniform float u_intensity; + +in vec2 a_pos; + +out vec2 v_extrude; + +#pragma mapbox: define highp float weight +#pragma mapbox: define mediump float radius + +// Effective "0" in the kernel density texture to adjust the kernel size to; +// this empirically chosen number minimizes artifacts on overlapping kernels +// for typical heatmap cases (assuming clustered source) +const highp float ZERO = 1.0 / 255.0 / 16.0; + +// Gaussian kernel coefficient: 1 / sqrt(2 * PI) +#define GAUSS_COEF 0.3989422804014327 + +void main(void) { + #pragma mapbox: initialize highp float weight + #pragma mapbox: initialize mediump float radius + + // unencode the extrusion vector that we snuck into the a_pos vector + vec2 unscaled_extrude = vec2(mod(a_pos, 2.0) * 2.0 - 1.0); + + // This 'extrude' comes in ranging from [-1, -1], to [1, 1]. We'll use + // it to produce the vertices of a square mesh framing the point feature + // we're adding to the kernel density texture. We'll also pass it as + // a out, so that the fragment shader can determine the distance of + // each fragment from the point feature. + // Before we do so, we need to scale it up sufficiently so that the + // kernel falls effectively to zero at the edge of the mesh. + // That is, we want to know S such that + // weight * u_intensity * GAUSS_COEF * exp(-0.5 * 3.0^2 * S^2) == ZERO + // Which solves to: + // S = sqrt(-2.0 * log(ZERO / (weight * u_intensity * GAUSS_COEF))) / 3.0 + float S = sqrt(-2.0 * log(ZERO / weight / u_intensity / GAUSS_COEF)) / 3.0; + + // Pass the out in units of radius + v_extrude = S * unscaled_extrude; + + // Scale by radius and the zoom-based scale factor to produce actual + // mesh position + vec2 extrude = v_extrude * radius * u_extrude_scale; + + // multiply a_pos by 0.5, since we had it * 2 in order to sneak + // in extrusion data + vec4 pos = vec4(floor(a_pos * 0.5) + extrude, 0, 1); + + gl_Position = u_matrix * pos; +} diff --git a/web/libraries/maplibre-gl/src/shaders/heatmap.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/heatmap.vertex.glsl.g.ts new file mode 100644 index 00000000..82dc209e --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/heatmap.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform mat4 u_matrix;uniform float u_extrude_scale;uniform float u_opacity;uniform float u_intensity;attribute vec2 a_pos;varying vec2 v_extrude;\n#pragma mapbox: define highp float weight\n#pragma mapbox: define mediump float radius\nconst highp float ZERO=1.0/255.0/16.0;\n#define GAUSS_COEF 0.3989422804014327\nvoid main(void) {\n#pragma mapbox: initialize highp float weight\n#pragma mapbox: initialize mediump float radius\nvec2 unscaled_extrude=vec2(mod(a_pos,2.0)*2.0-1.0);float S=sqrt(-2.0*log(ZERO/weight/u_intensity/GAUSS_COEF))/3.0;v_extrude=S*unscaled_extrude;vec2 extrude=v_extrude*radius*u_extrude_scale;vec4 pos=vec4(floor(a_pos*0.5)+extrude,0,1);gl_Position=u_matrix*pos;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/heatmap_texture.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/heatmap_texture.fragment.glsl new file mode 100644 index 00000000..5efcd354 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/heatmap_texture.fragment.glsl @@ -0,0 +1,15 @@ +uniform sampler2D u_image; +uniform sampler2D u_color_ramp; +uniform float u_opacity; + +in vec2 v_pos; + +void main() { + float t = texture(u_image, v_pos).r; + vec4 color = texture(u_color_ramp, vec2(t, 0.5)); + fragColor = color * u_opacity; + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(0.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/heatmap_texture.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/heatmap_texture.fragment.glsl.g.ts new file mode 100644 index 00000000..97548fe2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/heatmap_texture.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform sampler2D u_image;uniform sampler2D u_color_ramp;uniform float u_opacity;varying vec2 v_pos;void main() {float t=texture2D(u_image,v_pos).r;vec4 color=texture2D(u_color_ramp,vec2(t,0.5));gl_FragColor=color*u_opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(0.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/heatmap_texture.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/heatmap_texture.vertex.glsl new file mode 100644 index 00000000..d9db180c --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/heatmap_texture.vertex.glsl @@ -0,0 +1,11 @@ +uniform mat4 u_matrix; +uniform vec2 u_world; +in vec2 a_pos; +out vec2 v_pos; + +void main() { + gl_Position = u_matrix * vec4(a_pos * u_world, 0, 1); + + v_pos.x = a_pos.x; + v_pos.y = 1.0 - a_pos.y; +} diff --git a/web/libraries/maplibre-gl/src/shaders/heatmap_texture.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/heatmap_texture.vertex.glsl.g.ts new file mode 100644 index 00000000..2b97491e --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/heatmap_texture.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform mat4 u_matrix;uniform vec2 u_world;attribute vec2 a_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos*u_world,0,1);v_pos.x=a_pos.x;v_pos.y=1.0-a_pos.y;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/hillshade.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/hillshade.fragment.glsl new file mode 100644 index 00000000..1b4842d7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/hillshade.fragment.glsl @@ -0,0 +1,52 @@ +uniform sampler2D u_image; +in vec2 v_pos; + +uniform vec2 u_latrange; +uniform vec2 u_light; +uniform vec4 u_shadow; +uniform vec4 u_highlight; +uniform vec4 u_accent; + +#define PI 3.141592653589793 + +void main() { + vec4 pixel = texture(u_image, v_pos); + + vec2 deriv = ((pixel.rg * 2.0) - 1.0); + + // We divide the slope by a scale factor based on the cosin of the pixel's approximate latitude + // to account for mercator projection distortion. see #4807 for details + float scaleFactor = cos(radians((u_latrange[0] - u_latrange[1]) * (1.0 - v_pos.y) + u_latrange[1])); + // We also multiply the slope by an arbitrary z-factor of 1.25 + float slope = atan(1.25 * length(deriv) / scaleFactor); + float aspect = deriv.x != 0.0 ? atan(deriv.y, -deriv.x) : PI / 2.0 * (deriv.y > 0.0 ? 1.0 : -1.0); + + float intensity = u_light.x; + // We add PI to make this property match the global light object, which adds PI/2 to the light's azimuthal + // position property to account for 0deg corresponding to north/the top of the viewport in the style spec + // and the original shader was written to accept (-illuminationDirection - 90) as the azimuthal. + float azimuth = u_light.y + PI; + + // We scale the slope exponentially based on intensity, using a calculation similar to + // the exponential interpolation function in the style spec: + // src/style-spec/expression/definitions/interpolate.js#L217-L228 + // so that higher intensity values create more opaque hillshading. + float base = 1.875 - intensity * 1.75; + float maxValue = 0.5 * PI; + float scaledSlope = intensity != 0.5 ? ((pow(base, slope) - 1.0) / (pow(base, maxValue) - 1.0)) * maxValue : slope; + + // The accent color is calculated with the cosine of the slope while the shade color is calculated with the sine + // so that the accent color's rate of change eases in while the shade color's eases out. + float accent = cos(scaledSlope); + // We multiply both the accent and shade color by a clamped intensity value + // so that intensities >= 0.5 do not additionally affect the color values + // while intensity values < 0.5 make the overall color more transparent. + vec4 accent_color = (1.0 - accent) * u_accent * clamp(intensity * 2.0, 0.0, 1.0); + float shade = abs(mod((aspect + azimuth) / PI + 0.5, 2.0) - 1.0); + vec4 shade_color = mix(u_shadow, u_highlight, shade) * sin(scaledSlope) * clamp(intensity * 2.0, 0.0, 1.0); + fragColor = accent_color * (1.0 - shade_color.a) + shade_color; + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/hillshade.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/hillshade.fragment.glsl.g.ts new file mode 100644 index 00000000..feb0162c --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/hillshade.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_latrange;uniform vec2 u_light;uniform vec4 u_shadow;uniform vec4 u_highlight;uniform vec4 u_accent;\n#define PI 3.141592653589793\nvoid main() {vec4 pixel=texture2D(u_image,v_pos);vec2 deriv=((pixel.rg*2.0)-1.0);float scaleFactor=cos(radians((u_latrange[0]-u_latrange[1])*(1.0-v_pos.y)+u_latrange[1]));float slope=atan(1.25*length(deriv)/scaleFactor);float aspect=deriv.x !=0.0 ? atan(deriv.y,-deriv.x) : PI/2.0*(deriv.y > 0.0 ? 1.0 :-1.0);float intensity=u_light.x;float azimuth=u_light.y+PI;float base=1.875-intensity*1.75;float maxValue=0.5*PI;float scaledSlope=intensity !=0.5 ? ((pow(base,slope)-1.0)/(pow(base,maxValue)-1.0))*maxValue : slope;float accent=cos(scaledSlope);vec4 accent_color=(1.0-accent)*u_accent*clamp(intensity*2.0,0.0,1.0);float shade=abs(mod((aspect+azimuth)/PI+0.5,2.0)-1.0);vec4 shade_color=mix(u_shadow,u_highlight,shade)*sin(scaledSlope)*clamp(intensity*2.0,0.0,1.0);gl_FragColor=accent_color*(1.0-shade_color.a)+shade_color;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/hillshade.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/hillshade.vertex.glsl new file mode 100644 index 00000000..10108d96 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/hillshade.vertex.glsl @@ -0,0 +1,11 @@ +uniform mat4 u_matrix; + +in vec2 a_pos; +in vec2 a_texture_pos; + +out vec2 v_pos; + +void main() { + gl_Position = u_matrix * vec4(a_pos, 0, 1); + v_pos = a_texture_pos / 8192.0; +} diff --git a/web/libraries/maplibre-gl/src/shaders/hillshade.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/hillshade.vertex.glsl.g.ts new file mode 100644 index 00000000..f5d82ebc --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/hillshade.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform mat4 u_matrix;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos=a_texture_pos/8192.0;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/hillshade_prepare.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/hillshade_prepare.fragment.glsl new file mode 100644 index 00000000..2192aeaf --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/hillshade_prepare.fragment.glsl @@ -0,0 +1,77 @@ +#ifdef GL_ES +precision highp float; +#endif + +uniform sampler2D u_image; +in vec2 v_pos; + + +uniform vec2 u_dimension; +uniform float u_zoom; +uniform vec4 u_unpack; + +float getElevation(vec2 coord, float bias) { + // Convert encoded elevation value to meters + vec4 data = texture(u_image, coord) * 255.0; + data.a = -1.0; + return dot(data, u_unpack) / 4.0; +} + +void main() { + vec2 epsilon = 1.0 / u_dimension; + + // queried pixels: + // +-----------+ + // | | | | + // | a | b | c | + // | | | | + // +-----------+ + // | | | | + // | d | e | f | + // | | | | + // +-----------+ + // | | | | + // | g | h | i | + // | | | | + // +-----------+ + + float a = getElevation(v_pos + vec2(-epsilon.x, -epsilon.y), 0.0); + float b = getElevation(v_pos + vec2(0, -epsilon.y), 0.0); + float c = getElevation(v_pos + vec2(epsilon.x, -epsilon.y), 0.0); + float d = getElevation(v_pos + vec2(-epsilon.x, 0), 0.0); + float e = getElevation(v_pos, 0.0); + float f = getElevation(v_pos + vec2(epsilon.x, 0), 0.0); + float g = getElevation(v_pos + vec2(-epsilon.x, epsilon.y), 0.0); + float h = getElevation(v_pos + vec2(0, epsilon.y), 0.0); + float i = getElevation(v_pos + vec2(epsilon.x, epsilon.y), 0.0); + + // Here we divide the x and y slopes by 8 * pixel size + // where pixel size (aka meters/pixel) is: + // circumference of the world / (pixels per tile * number of tiles) + // which is equivalent to: 8 * 40075016.6855785 / (512 * pow(2, u_zoom)) + // which can be reduced to: pow(2, 19.25619978527 - u_zoom). + // We want to vertically exaggerate the hillshading because otherwise + // it is barely noticeable at low zooms. To do this, we multiply this by + // a scale factor that is a function of zooms below 15, which is an arbitrary + // that corresponds to the max zoom level of Mapbox terrain-RGB tiles. + // See nickidlugash's awesome breakdown for more info: + // https://github.com/mapbox/mapbox-gl-js/pull/5286#discussion_r148419556 + + float exaggerationFactor = u_zoom < 2.0 ? 0.4 : u_zoom < 4.5 ? 0.35 : 0.3; + float exaggeration = u_zoom < 15.0 ? (u_zoom - 15.0) * exaggerationFactor : 0.0; + + vec2 deriv = vec2( + (c + f + f + i) - (a + d + d + g), + (g + h + h + i) - (a + b + b + c) + ) / pow(2.0, exaggeration + (19.2562 - u_zoom)); + + fragColor = clamp(vec4( + deriv.x / 2.0 + 0.5, + deriv.y / 2.0 + 0.5, + 1.0, + 1.0), 0.0, 1.0); + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/hillshade_prepare.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/hillshade_prepare.fragment.glsl.g.ts new file mode 100644 index 00000000..30b50cec --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/hillshade_prepare.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default '#ifdef GL_ES\nprecision highp float;\n#endif\nuniform sampler2D u_image;varying vec2 v_pos;uniform vec2 u_dimension;uniform float u_zoom;uniform vec4 u_unpack;float getElevation(vec2 coord,float bias) {vec4 data=texture2D(u_image,coord)*255.0;data.a=-1.0;return dot(data,u_unpack)/4.0;}void main() {vec2 epsilon=1.0/u_dimension;float a=getElevation(v_pos+vec2(-epsilon.x,-epsilon.y),0.0);float b=getElevation(v_pos+vec2(0,-epsilon.y),0.0);float c=getElevation(v_pos+vec2(epsilon.x,-epsilon.y),0.0);float d=getElevation(v_pos+vec2(-epsilon.x,0),0.0);float e=getElevation(v_pos,0.0);float f=getElevation(v_pos+vec2(epsilon.x,0),0.0);float g=getElevation(v_pos+vec2(-epsilon.x,epsilon.y),0.0);float h=getElevation(v_pos+vec2(0,epsilon.y),0.0);float i=getElevation(v_pos+vec2(epsilon.x,epsilon.y),0.0);float exaggerationFactor=u_zoom < 2.0 ? 0.4 : u_zoom < 4.5 ? 0.35 : 0.3;float exaggeration=u_zoom < 15.0 ? (u_zoom-15.0)*exaggerationFactor : 0.0;vec2 deriv=vec2((c+f+f+i)-(a+d+d+g),(g+h+h+i)-(a+b+b+c))/pow(2.0,exaggeration+(19.2562-u_zoom));gl_FragColor=clamp(vec4(deriv.x/2.0+0.5,deriv.y/2.0+0.5,1.0,1.0),0.0,1.0);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/hillshade_prepare.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/hillshade_prepare.vertex.glsl new file mode 100644 index 00000000..4133d46a --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/hillshade_prepare.vertex.glsl @@ -0,0 +1,15 @@ +uniform mat4 u_matrix; +uniform vec2 u_dimension; + +in vec2 a_pos; +in vec2 a_texture_pos; + +out vec2 v_pos; + +void main() { + gl_Position = u_matrix * vec4(a_pos, 0, 1); + + highp vec2 epsilon = 1.0 / u_dimension; + float scale = (u_dimension.x - 2.0) / u_dimension.x; + v_pos = (a_texture_pos / 8192.0) * scale + epsilon; +} diff --git a/web/libraries/maplibre-gl/src/shaders/hillshade_prepare.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/hillshade_prepare.vertex.glsl.g.ts new file mode 100644 index 00000000..9d8db4d5 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/hillshade_prepare.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform mat4 u_matrix;uniform vec2 u_dimension;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);highp vec2 epsilon=1.0/u_dimension;float scale=(u_dimension.x-2.0)/u_dimension.x;v_pos=(a_texture_pos/8192.0)*scale+epsilon;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/line.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/line.fragment.glsl new file mode 100644 index 00000000..4ea11c11 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line.fragment.glsl @@ -0,0 +1,30 @@ +uniform lowp float u_device_pixel_ratio; + +in vec2 v_width2; +in vec2 v_normal; +in float v_gamma_scale; + +#pragma mapbox: define highp vec4 color +#pragma mapbox: define lowp float blur +#pragma mapbox: define lowp float opacity + +void main() { + #pragma mapbox: initialize highp vec4 color + #pragma mapbox: initialize lowp float blur + #pragma mapbox: initialize lowp float opacity + + // Calculate the distance of the pixel from the line in pixels. + float dist = length(v_normal) * v_width2.s; + + // Calculate the antialiasing fade factor. This is either when fading in + // the line in case of an offset line (v_width2.t) or when fading out + // (v_width2.s) + float blur2 = (blur + 1.0 / u_device_pixel_ratio) * v_gamma_scale; + float alpha = clamp(min(dist - (v_width2.t - blur2), v_width2.s - dist) / blur2, 0.0, 1.0); + + fragColor = color * (alpha * opacity); + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/line.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/line.fragment.glsl.g.ts new file mode 100644 index 00000000..06048acf --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform lowp float u_device_pixel_ratio;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);gl_FragColor=color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/line.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/line.vertex.glsl new file mode 100644 index 00000000..457c0668 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line.vertex.glsl @@ -0,0 +1,89 @@ +// floor(127 / 2) == 63.0 +// the maximum allowed miter limit is 2.0 at the moment. the extrude normal is +// stored in a byte (-128..127). we scale regular normals up to length 63, but +// there are also "special" normals that have a bigger length (of up to 126 in +// this case). +// #define scale 63.0 +#define scale 0.015873016 + +in vec2 a_pos_normal; +in vec4 a_data; + +uniform mat4 u_matrix; +uniform mediump float u_ratio; +uniform vec2 u_units_to_pixels; +uniform lowp float u_device_pixel_ratio; + +out vec2 v_normal; +out vec2 v_width2; +out float v_gamma_scale; +out highp float v_linesofar; + +#pragma mapbox: define highp vec4 color +#pragma mapbox: define lowp float blur +#pragma mapbox: define lowp float opacity +#pragma mapbox: define mediump float gapwidth +#pragma mapbox: define lowp float offset +#pragma mapbox: define mediump float width + +void main() { + #pragma mapbox: initialize highp vec4 color + #pragma mapbox: initialize lowp float blur + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize mediump float gapwidth + #pragma mapbox: initialize lowp float offset + #pragma mapbox: initialize mediump float width + + // the distance over which the line edge fades out. + // Retina devices need a smaller distance to avoid aliasing. + float ANTIALIASING = 1.0 / u_device_pixel_ratio / 2.0; + + vec2 a_extrude = a_data.xy - 128.0; + float a_direction = mod(a_data.z, 4.0) - 1.0; + + v_linesofar = (floor(a_data.z / 4.0) + a_data.w * 64.0) * 2.0; + + vec2 pos = floor(a_pos_normal * 0.5); + + // x is 1 if it's a round cap, 0 otherwise + // y is 1 if the normal points up, and -1 if it points down + // We store these in the least significant bit of a_pos_normal + mediump vec2 normal = a_pos_normal - 2.0 * pos; + normal.y = normal.y * 2.0 - 1.0; + v_normal = normal; + + // these transformations used to be applied in the JS and native code bases. + // moved them into the shader for clarity and simplicity. + gapwidth = gapwidth / 2.0; + float halfwidth = width / 2.0; + offset = -1.0 * offset; + + float inset = gapwidth + (gapwidth > 0.0 ? ANTIALIASING : 0.0); + float outset = gapwidth + halfwidth * (gapwidth > 0.0 ? 2.0 : 1.0) + (halfwidth == 0.0 ? 0.0 : ANTIALIASING); + + // Scale the extrusion vector down to a normal and then up by the line width + // of this vertex. + mediump vec2 dist = outset * a_extrude * scale; + + // Calculate the offset when drawing a line that is to the side of the actual line. + // We do this by creating a vector that points towards the extrude, but rotate + // it when we're drawing round end points (a_direction = -1 or 1) since their + // extrude vector points in another direction. + mediump float u = 0.5 * a_direction; + mediump float t = 1.0 - abs(u); + mediump vec2 offset2 = offset * a_extrude * scale * normal.y * mat2(t, -u, u, t); + + vec4 projected_extrude = u_matrix * vec4(dist / u_ratio, 0.0, 0.0); + gl_Position = u_matrix * vec4(pos + offset2 / u_ratio, 0.0, 1.0) + projected_extrude; + + // calculate how much the perspective view squishes or stretches the extrude + #ifdef TERRAIN3D + v_gamma_scale = 1.0; // not needed, because this is done automatically via the mesh + #else + float extrude_length_without_perspective = length(dist); + float extrude_length_with_perspective = length(projected_extrude.xy / gl_Position.w * u_units_to_pixels); + v_gamma_scale = extrude_length_without_perspective / extrude_length_with_perspective; + #endif + + v_width2 = vec2(outset, inset); +} diff --git a/web/libraries/maplibre-gl/src/shaders/line.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/line.vertex.glsl.g.ts new file mode 100644 index 00000000..1e036ba2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default '\n#define scale 0.015873016\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform vec2 u_units_to_pixels;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp float v_linesofar;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float width\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float width\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;v_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*2.0;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_width2=vec2(outset,inset);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/line_gradient.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/line_gradient.fragment.glsl new file mode 100644 index 00000000..fb4221f4 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line_gradient.fragment.glsl @@ -0,0 +1,34 @@ +uniform lowp float u_device_pixel_ratio; +uniform sampler2D u_image; + +in vec2 v_width2; +in vec2 v_normal; +in float v_gamma_scale; +in highp vec2 v_uv; + +#pragma mapbox: define lowp float blur +#pragma mapbox: define lowp float opacity + +void main() { + #pragma mapbox: initialize lowp float blur + #pragma mapbox: initialize lowp float opacity + + // Calculate the distance of the pixel from the line in pixels. + float dist = length(v_normal) * v_width2.s; + + // Calculate the antialiasing fade factor. This is either when fading in + // the line in case of an offset line (v_width2.t) or when fading out + // (v_width2.s) + float blur2 = (blur + 1.0 / u_device_pixel_ratio) * v_gamma_scale; + float alpha = clamp(min(dist - (v_width2.t - blur2), v_width2.s - dist) / blur2, 0.0, 1.0); + + // For gradient lines, v_lineprogress is the ratio along the + // entire line, the gradient ramp is stored in a texture. + vec4 color = texture(u_image, v_uv); + + fragColor = color * (alpha * opacity); + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/line_gradient.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/line_gradient.fragment.glsl.g.ts new file mode 100644 index 00000000..5e638261 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line_gradient.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;varying vec2 v_width2;varying vec2 v_normal;varying float v_gamma_scale;varying highp vec2 v_uv;\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);vec4 color=texture2D(u_image,v_uv);gl_FragColor=color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/line_gradient.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/line_gradient.vertex.glsl new file mode 100644 index 00000000..7872b2f3 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line_gradient.vertex.glsl @@ -0,0 +1,92 @@ +// floor(127 / 2) == 63.0 +// the maximum allowed miter limit is 2.0 at the moment. the extrude normal is +// stored in a byte (-128..127). we scale regular normals up to length 63, but +// there are also "special" normals that have a bigger length (of up to 126 in +// this case). +// #define scale 63.0 +#define scale 0.015873016 + +in vec2 a_pos_normal; +in vec4 a_data; +in float a_uv_x; +in float a_split_index; + +uniform mat4 u_matrix; +uniform mediump float u_ratio; +uniform lowp float u_device_pixel_ratio; +uniform vec2 u_units_to_pixels; +uniform float u_image_height; + +out vec2 v_normal; +out vec2 v_width2; +out float v_gamma_scale; +out highp vec2 v_uv; + +#pragma mapbox: define lowp float blur +#pragma mapbox: define lowp float opacity +#pragma mapbox: define mediump float gapwidth +#pragma mapbox: define lowp float offset +#pragma mapbox: define mediump float width + +void main() { + #pragma mapbox: initialize lowp float blur + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize mediump float gapwidth + #pragma mapbox: initialize lowp float offset + #pragma mapbox: initialize mediump float width + + // the distance over which the line edge fades out. + // Retina devices need a smaller distance to avoid aliasing. + float ANTIALIASING = 1.0 / u_device_pixel_ratio / 2.0; + + vec2 a_extrude = a_data.xy - 128.0; + float a_direction = mod(a_data.z, 4.0) - 1.0; + + highp float texel_height = 1.0 / u_image_height; + highp float half_texel_height = 0.5 * texel_height; + v_uv = vec2(a_uv_x, a_split_index * texel_height - half_texel_height); + + vec2 pos = floor(a_pos_normal * 0.5); + + // x is 1 if it's a round cap, 0 otherwise + // y is 1 if the normal points up, and -1 if it points down + // We store these in the least significant bit of a_pos_normal + mediump vec2 normal = a_pos_normal - 2.0 * pos; + normal.y = normal.y * 2.0 - 1.0; + v_normal = normal; + + // these transformations used to be applied in the JS and native code bases. + // moved them into the shader for clarity and simplicity. + gapwidth = gapwidth / 2.0; + float halfwidth = width / 2.0; + offset = -1.0 * offset; + + float inset = gapwidth + (gapwidth > 0.0 ? ANTIALIASING : 0.0); + float outset = gapwidth + halfwidth * (gapwidth > 0.0 ? 2.0 : 1.0) + (halfwidth == 0.0 ? 0.0 : ANTIALIASING); + + // Scale the extrusion vector down to a normal and then up by the line width + // of this vertex. + mediump vec2 dist = outset * a_extrude * scale; + + // Calculate the offset when drawing a line that is to the side of the actual line. + // We do this by creating a vector that points towards the extrude, but rotate + // it when we're drawing round end points (a_direction = -1 or 1) since their + // extrude vector points in another direction. + mediump float u = 0.5 * a_direction; + mediump float t = 1.0 - abs(u); + mediump vec2 offset2 = offset * a_extrude * scale * normal.y * mat2(t, -u, u, t); + + vec4 projected_extrude = u_matrix * vec4(dist / u_ratio, 0.0, 0.0); + gl_Position = u_matrix * vec4(pos + offset2 / u_ratio, 0.0, 1.0) + projected_extrude; + + // calculate how much the perspective view squishes or stretches the extrude + #ifdef TERRAIN3D + v_gamma_scale = 1.0; // not needed, because this is done automatically via the mesh + #else + float extrude_length_without_perspective = length(dist); + float extrude_length_with_perspective = length(projected_extrude.xy / gl_Position.w * u_units_to_pixels); + v_gamma_scale = extrude_length_without_perspective / extrude_length_with_perspective; + #endif + + v_width2 = vec2(outset, inset); +} diff --git a/web/libraries/maplibre-gl/src/shaders/line_gradient.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/line_gradient.vertex.glsl.g.ts new file mode 100644 index 00000000..cec2d901 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line_gradient.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default '\n#define scale 0.015873016\nattribute vec2 a_pos_normal;attribute vec4 a_data;attribute float a_uv_x;attribute float a_split_index;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_units_to_pixels;uniform float u_image_height;varying vec2 v_normal;varying vec2 v_width2;varying float v_gamma_scale;varying highp vec2 v_uv;\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float width\nvoid main() {\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float width\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;highp float texel_height=1.0/u_image_height;highp float half_texel_height=0.5*texel_height;v_uv=vec2(a_uv_x,a_split_index*texel_height-half_texel_height);vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_width2=vec2(outset,inset);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/line_pattern.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/line_pattern.fragment.glsl new file mode 100644 index 00000000..2ab4bd70 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line_pattern.fragment.glsl @@ -0,0 +1,77 @@ +#ifdef GL_ES + precision highp float; +#endif +uniform lowp float u_device_pixel_ratio; +uniform vec2 u_texsize; +uniform float u_fade; +uniform mediump vec3 u_scale; + +uniform sampler2D u_image; + +in vec2 v_normal; +in vec2 v_width2; +in float v_linesofar; +in float v_gamma_scale; +in float v_width; + +#pragma mapbox: define lowp vec4 pattern_from +#pragma mapbox: define lowp vec4 pattern_to +#pragma mapbox: define lowp float pixel_ratio_from +#pragma mapbox: define lowp float pixel_ratio_to +#pragma mapbox: define lowp float blur +#pragma mapbox: define lowp float opacity + +void main() { + #pragma mapbox: initialize mediump vec4 pattern_from + #pragma mapbox: initialize mediump vec4 pattern_to + #pragma mapbox: initialize lowp float pixel_ratio_from + #pragma mapbox: initialize lowp float pixel_ratio_to + + #pragma mapbox: initialize lowp float blur + #pragma mapbox: initialize lowp float opacity + + vec2 pattern_tl_a = pattern_from.xy; + vec2 pattern_br_a = pattern_from.zw; + vec2 pattern_tl_b = pattern_to.xy; + vec2 pattern_br_b = pattern_to.zw; + + float tileZoomRatio = u_scale.x; + float fromScale = u_scale.y; + float toScale = u_scale.z; + + vec2 display_size_a = (pattern_br_a - pattern_tl_a) / pixel_ratio_from; + vec2 display_size_b = (pattern_br_b - pattern_tl_b) / pixel_ratio_to; + + vec2 pattern_size_a = vec2(display_size_a.x * fromScale / tileZoomRatio, display_size_a.y); + vec2 pattern_size_b = vec2(display_size_b.x * toScale / tileZoomRatio, display_size_b.y); + + float aspect_a = display_size_a.y / v_width; + float aspect_b = display_size_b.y / v_width; + + // Calculate the distance of the pixel from the line in pixels. + float dist = length(v_normal) * v_width2.s; + + // Calculate the antialiasing fade factor. This is either when fading in + // the line in case of an offset line (v_width2.t) or when fading out + // (v_width2.s) + float blur2 = (blur + 1.0 / u_device_pixel_ratio) * v_gamma_scale; + float alpha = clamp(min(dist - (v_width2.t - blur2), v_width2.s - dist) / blur2, 0.0, 1.0); + + float x_a = mod(v_linesofar / pattern_size_a.x * aspect_a, 1.0); + float x_b = mod(v_linesofar / pattern_size_b.x * aspect_b, 1.0); + + float y = 0.5 * v_normal.y + 0.5; + + vec2 texel_size = 1.0 / u_texsize; + + vec2 pos_a = mix(pattern_tl_a * texel_size - texel_size, pattern_br_a * texel_size + texel_size, vec2(x_a, y)); + vec2 pos_b = mix(pattern_tl_b * texel_size - texel_size, pattern_br_b * texel_size + texel_size, vec2(x_b, y)); + + vec4 color = mix(texture(u_image, pos_a), texture(u_image, pos_b), u_fade); + + fragColor = color * alpha * opacity; + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/line_pattern.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/line_pattern.fragment.glsl.g.ts new file mode 100644 index 00000000..7372026f --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line_pattern.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default '#ifdef GL_ES\nprecision highp float;\n#endif\nuniform lowp float u_device_pixel_ratio;uniform vec2 u_texsize;uniform float u_fade;uniform mediump vec3 u_scale;uniform sampler2D u_image;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\nvec2 pattern_tl_a=pattern_from.xy;vec2 pattern_br_a=pattern_from.zw;vec2 pattern_tl_b=pattern_to.xy;vec2 pattern_br_b=pattern_to.zw;float tileZoomRatio=u_scale.x;float fromScale=u_scale.y;float toScale=u_scale.z;vec2 display_size_a=(pattern_br_a-pattern_tl_a)/pixel_ratio_from;vec2 display_size_b=(pattern_br_b-pattern_tl_b)/pixel_ratio_to;vec2 pattern_size_a=vec2(display_size_a.x*fromScale/tileZoomRatio,display_size_a.y);vec2 pattern_size_b=vec2(display_size_b.x*toScale/tileZoomRatio,display_size_b.y);float aspect_a=display_size_a.y/v_width;float aspect_b=display_size_b.y/v_width;float dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float x_a=mod(v_linesofar/pattern_size_a.x*aspect_a,1.0);float x_b=mod(v_linesofar/pattern_size_b.x*aspect_b,1.0);float y=0.5*v_normal.y+0.5;vec2 texel_size=1.0/u_texsize;vec2 pos_a=mix(pattern_tl_a*texel_size-texel_size,pattern_br_a*texel_size+texel_size,vec2(x_a,y));vec2 pos_b=mix(pattern_tl_b*texel_size-texel_size,pattern_br_b*texel_size+texel_size,vec2(x_b,y));vec4 color=mix(texture2D(u_image,pos_a),texture2D(u_image,pos_b),u_fade);gl_FragColor=color*alpha*opacity;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/line_pattern.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/line_pattern.vertex.glsl new file mode 100644 index 00000000..70b668cf --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line_pattern.vertex.glsl @@ -0,0 +1,103 @@ +// floor(127 / 2) == 63.0 +// the maximum allowed miter limit is 2.0 at the moment. the extrude normal is +// stored in a byte (-128..127). we scale regular normals up to length 63, but +// there are also "special" normals that have a bigger length (of up to 126 in +// this case). +// #define scale 63.0 +#define scale 0.015873016 + +// We scale the distance before adding it to the buffers so that we can store +// long distances for long segments. Use this value to unscale the distance. +#define LINE_DISTANCE_SCALE 2.0 + +in vec2 a_pos_normal; +in vec4 a_data; + +uniform mat4 u_matrix; +uniform vec2 u_units_to_pixels; +uniform mediump float u_ratio; +uniform lowp float u_device_pixel_ratio; + +out vec2 v_normal; +out vec2 v_width2; +out float v_linesofar; +out float v_gamma_scale; +out float v_width; + +#pragma mapbox: define lowp float blur +#pragma mapbox: define lowp float opacity +#pragma mapbox: define lowp float offset +#pragma mapbox: define mediump float gapwidth +#pragma mapbox: define mediump float width +#pragma mapbox: define lowp float floorwidth +#pragma mapbox: define lowp vec4 pattern_from +#pragma mapbox: define lowp vec4 pattern_to +#pragma mapbox: define lowp float pixel_ratio_from +#pragma mapbox: define lowp float pixel_ratio_to + +void main() { + #pragma mapbox: initialize lowp float blur + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize lowp float offset + #pragma mapbox: initialize mediump float gapwidth + #pragma mapbox: initialize mediump float width + #pragma mapbox: initialize lowp float floorwidth + #pragma mapbox: initialize mediump vec4 pattern_from + #pragma mapbox: initialize mediump vec4 pattern_to + #pragma mapbox: initialize lowp float pixel_ratio_from + #pragma mapbox: initialize lowp float pixel_ratio_to + + // the distance over which the line edge fades out. + // Retina devices need a smaller distance to avoid aliasing. + float ANTIALIASING = 1.0 / u_device_pixel_ratio / 2.0; + + vec2 a_extrude = a_data.xy - 128.0; + float a_direction = mod(a_data.z, 4.0) - 1.0; + float a_linesofar = (floor(a_data.z / 4.0) + a_data.w * 64.0) * LINE_DISTANCE_SCALE; + // float tileRatio = u_scale.x; + vec2 pos = floor(a_pos_normal * 0.5); + + // x is 1 if it's a round cap, 0 otherwise + // y is 1 if the normal points up, and -1 if it points down + // We store these in the least significant bit of a_pos_normal + mediump vec2 normal = a_pos_normal - 2.0 * pos; + normal.y = normal.y * 2.0 - 1.0; + v_normal = normal; + + // these transformations used to be applied in the JS and native code bases. + // moved them into the shader for clarity and simplicity. + gapwidth = gapwidth / 2.0; + float halfwidth = width / 2.0; + offset = -1.0 * offset; + + float inset = gapwidth + (gapwidth > 0.0 ? ANTIALIASING : 0.0); + float outset = gapwidth + halfwidth * (gapwidth > 0.0 ? 2.0 : 1.0) + (halfwidth == 0.0 ? 0.0 : ANTIALIASING); + + // Scale the extrusion vector down to a normal and then up by the line width + // of this vertex. + mediump vec2 dist = outset * a_extrude * scale; + + // Calculate the offset when drawing a line that is to the side of the actual line. + // We do this by creating a vector that points towards the extrude, but rotate + // it when we're drawing round end points (a_direction = -1 or 1) since their + // extrude vector points in another direction. + mediump float u = 0.5 * a_direction; + mediump float t = 1.0 - abs(u); + mediump vec2 offset2 = offset * a_extrude * scale * normal.y * mat2(t, -u, u, t); + + vec4 projected_extrude = u_matrix * vec4(dist / u_ratio, 0.0, 0.0); + gl_Position = u_matrix * vec4(pos + offset2 / u_ratio, 0.0, 1.0) + projected_extrude; + + // calculate how much the perspective view squishes or stretches the extrude + #ifdef TERRAIN3D + v_gamma_scale = 1.0; // not needed, because this is done automatically via the mesh + #else + float extrude_length_without_perspective = length(dist); + float extrude_length_with_perspective = length(projected_extrude.xy / gl_Position.w * u_units_to_pixels); + v_gamma_scale = extrude_length_without_perspective / extrude_length_with_perspective; + #endif + + v_linesofar = a_linesofar; + v_width2 = vec2(outset, inset); + v_width = floorwidth; +} diff --git a/web/libraries/maplibre-gl/src/shaders/line_pattern.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/line_pattern.vertex.glsl.g.ts new file mode 100644 index 00000000..685c48c9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line_pattern.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default '\n#define scale 0.015873016\n#define LINE_DISTANCE_SCALE 2.0\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform vec2 u_units_to_pixels;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;varying vec2 v_normal;varying vec2 v_width2;varying float v_linesofar;varying float v_gamma_scale;varying float v_width;\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define mediump float width\n#pragma mapbox: define lowp float floorwidth\n#pragma mapbox: define lowp vec4 pattern_from\n#pragma mapbox: define lowp vec4 pattern_to\n#pragma mapbox: define lowp float pixel_ratio_from\n#pragma mapbox: define lowp float pixel_ratio_to\nvoid main() {\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize mediump float width\n#pragma mapbox: initialize lowp float floorwidth\n#pragma mapbox: initialize mediump vec4 pattern_from\n#pragma mapbox: initialize mediump vec4 pattern_to\n#pragma mapbox: initialize lowp float pixel_ratio_from\n#pragma mapbox: initialize lowp float pixel_ratio_to\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_linesofar=a_linesofar;v_width2=vec2(outset,inset);v_width=floorwidth;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/line_sdf.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/line_sdf.fragment.glsl new file mode 100644 index 00000000..3cbd2c9b --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line_sdf.fragment.glsl @@ -0,0 +1,45 @@ + +uniform lowp float u_device_pixel_ratio; +uniform sampler2D u_image; +uniform float u_sdfgamma; +uniform float u_mix; + +in vec2 v_normal; +in vec2 v_width2; +in vec2 v_tex_a; +in vec2 v_tex_b; +in float v_gamma_scale; + +#pragma mapbox: define highp vec4 color +#pragma mapbox: define lowp float blur +#pragma mapbox: define lowp float opacity +#pragma mapbox: define mediump float width +#pragma mapbox: define lowp float floorwidth + +void main() { + #pragma mapbox: initialize highp vec4 color + #pragma mapbox: initialize lowp float blur + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize mediump float width + #pragma mapbox: initialize lowp float floorwidth + + // Calculate the distance of the pixel from the line in pixels. + float dist = length(v_normal) * v_width2.s; + + // Calculate the antialiasing fade factor. This is either when fading in + // the line in case of an offset line (v_width2.t) or when fading out + // (v_width2.s) + float blur2 = (blur + 1.0 / u_device_pixel_ratio) * v_gamma_scale; + float alpha = clamp(min(dist - (v_width2.t - blur2), v_width2.s - dist) / blur2, 0.0, 1.0); + + float sdfdist_a = texture(u_image, v_tex_a).a; + float sdfdist_b = texture(u_image, v_tex_b).a; + float sdfdist = mix(sdfdist_a, sdfdist_b, u_mix); + alpha *= smoothstep(0.5 - u_sdfgamma / floorwidth, 0.5 + u_sdfgamma / floorwidth, sdfdist); + + fragColor = color * (alpha * opacity); + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/line_sdf.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/line_sdf.fragment.glsl.g.ts new file mode 100644 index 00000000..dbcaacda --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line_sdf.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform lowp float u_device_pixel_ratio;uniform sampler2D u_image;uniform float u_sdfgamma;uniform float u_mix;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float width\n#pragma mapbox: define lowp float floorwidth\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float width\n#pragma mapbox: initialize lowp float floorwidth\nfloat dist=length(v_normal)*v_width2.s;float blur2=(blur+1.0/u_device_pixel_ratio)*v_gamma_scale;float alpha=clamp(min(dist-(v_width2.t-blur2),v_width2.s-dist)/blur2,0.0,1.0);float sdfdist_a=texture2D(u_image,v_tex_a).a;float sdfdist_b=texture2D(u_image,v_tex_b).a;float sdfdist=mix(sdfdist_a,sdfdist_b,u_mix);alpha*=smoothstep(0.5-u_sdfgamma/floorwidth,0.5+u_sdfgamma/floorwidth,sdfdist);gl_FragColor=color*(alpha*opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/line_sdf.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/line_sdf.vertex.glsl new file mode 100644 index 00000000..3324b63c --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line_sdf.vertex.glsl @@ -0,0 +1,101 @@ +// floor(127 / 2) == 63.0 +// the maximum allowed miter limit is 2.0 at the moment. the extrude normal is +// stored in a byte (-128..127). we scale regular normals up to length 63, but +// there are also "special" normals that have a bigger length (of up to 126 in +// this case). +// #define scale 63.0 +#define scale 0.015873016 + +// We scale the distance before adding it to the buffers so that we can store +// long distances for long segments. Use this value to unscale the distance. +#define LINE_DISTANCE_SCALE 2.0 + +in vec2 a_pos_normal; +in vec4 a_data; + +uniform mat4 u_matrix; +uniform mediump float u_ratio; +uniform lowp float u_device_pixel_ratio; +uniform vec2 u_patternscale_a; +uniform float u_tex_y_a; +uniform vec2 u_patternscale_b; +uniform float u_tex_y_b; +uniform vec2 u_units_to_pixels; + +out vec2 v_normal; +out vec2 v_width2; +out vec2 v_tex_a; +out vec2 v_tex_b; +out float v_gamma_scale; + +#pragma mapbox: define highp vec4 color +#pragma mapbox: define lowp float blur +#pragma mapbox: define lowp float opacity +#pragma mapbox: define mediump float gapwidth +#pragma mapbox: define lowp float offset +#pragma mapbox: define mediump float width +#pragma mapbox: define lowp float floorwidth + +void main() { + #pragma mapbox: initialize highp vec4 color + #pragma mapbox: initialize lowp float blur + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize mediump float gapwidth + #pragma mapbox: initialize lowp float offset + #pragma mapbox: initialize mediump float width + #pragma mapbox: initialize lowp float floorwidth + + // the distance over which the line edge fades out. + // Retina devices need a smaller distance to avoid aliasing. + float ANTIALIASING = 1.0 / u_device_pixel_ratio / 2.0; + + vec2 a_extrude = a_data.xy - 128.0; + float a_direction = mod(a_data.z, 4.0) - 1.0; + float a_linesofar = (floor(a_data.z / 4.0) + a_data.w * 64.0) * LINE_DISTANCE_SCALE; + + vec2 pos = floor(a_pos_normal * 0.5); + + // x is 1 if it's a round cap, 0 otherwise + // y is 1 if the normal points up, and -1 if it points down + // We store these in the least significant bit of a_pos_normal + mediump vec2 normal = a_pos_normal - 2.0 * pos; + normal.y = normal.y * 2.0 - 1.0; + v_normal = normal; + + // these transformations used to be applied in the JS and native code bases. + // moved them into the shader for clarity and simplicity. + gapwidth = gapwidth / 2.0; + float halfwidth = width / 2.0; + offset = -1.0 * offset; + + float inset = gapwidth + (gapwidth > 0.0 ? ANTIALIASING : 0.0); + float outset = gapwidth + halfwidth * (gapwidth > 0.0 ? 2.0 : 1.0) + (halfwidth == 0.0 ? 0.0 : ANTIALIASING); + + // Scale the extrusion vector down to a normal and then up by the line width + // of this vertex. + mediump vec2 dist =outset * a_extrude * scale; + + // Calculate the offset when drawing a line that is to the side of the actual line. + // We do this by creating a vector that points towards the extrude, but rotate + // it when we're drawing round end points (a_direction = -1 or 1) since their + // extrude vector points in another direction. + mediump float u = 0.5 * a_direction; + mediump float t = 1.0 - abs(u); + mediump vec2 offset2 = offset * a_extrude * scale * normal.y * mat2(t, -u, u, t); + + vec4 projected_extrude = u_matrix * vec4(dist / u_ratio, 0.0, 0.0); + gl_Position = u_matrix * vec4(pos + offset2 / u_ratio, 0.0, 1.0) + projected_extrude; + + // calculate how much the perspective view squishes or stretches the extrude + #ifdef TERRAIN3D + v_gamma_scale = 1.0; // not needed, because this is done automatically via the mesh + #else + float extrude_length_without_perspective = length(dist); + float extrude_length_with_perspective = length(projected_extrude.xy / gl_Position.w * u_units_to_pixels); + v_gamma_scale = extrude_length_without_perspective / extrude_length_with_perspective; + #endif + + v_tex_a = vec2(a_linesofar * u_patternscale_a.x / floorwidth, normal.y * u_patternscale_a.y + u_tex_y_a); + v_tex_b = vec2(a_linesofar * u_patternscale_b.x / floorwidth, normal.y * u_patternscale_b.y + u_tex_y_b); + v_width2 = vec2(outset, inset); +} diff --git a/web/libraries/maplibre-gl/src/shaders/line_sdf.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/line_sdf.vertex.glsl.g.ts new file mode 100644 index 00000000..8b0d85e2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/line_sdf.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default '\n#define scale 0.015873016\n#define LINE_DISTANCE_SCALE 2.0\nattribute vec2 a_pos_normal;attribute vec4 a_data;uniform mat4 u_matrix;uniform mediump float u_ratio;uniform lowp float u_device_pixel_ratio;uniform vec2 u_patternscale_a;uniform float u_tex_y_a;uniform vec2 u_patternscale_b;uniform float u_tex_y_b;uniform vec2 u_units_to_pixels;varying vec2 v_normal;varying vec2 v_width2;varying vec2 v_tex_a;varying vec2 v_tex_b;varying float v_gamma_scale;\n#pragma mapbox: define highp vec4 color\n#pragma mapbox: define lowp float blur\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define mediump float gapwidth\n#pragma mapbox: define lowp float offset\n#pragma mapbox: define mediump float width\n#pragma mapbox: define lowp float floorwidth\nvoid main() {\n#pragma mapbox: initialize highp vec4 color\n#pragma mapbox: initialize lowp float blur\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize mediump float gapwidth\n#pragma mapbox: initialize lowp float offset\n#pragma mapbox: initialize mediump float width\n#pragma mapbox: initialize lowp float floorwidth\nfloat ANTIALIASING=1.0/u_device_pixel_ratio/2.0;vec2 a_extrude=a_data.xy-128.0;float a_direction=mod(a_data.z,4.0)-1.0;float a_linesofar=(floor(a_data.z/4.0)+a_data.w*64.0)*LINE_DISTANCE_SCALE;vec2 pos=floor(a_pos_normal*0.5);mediump vec2 normal=a_pos_normal-2.0*pos;normal.y=normal.y*2.0-1.0;v_normal=normal;gapwidth=gapwidth/2.0;float halfwidth=width/2.0;offset=-1.0*offset;float inset=gapwidth+(gapwidth > 0.0 ? ANTIALIASING : 0.0);float outset=gapwidth+halfwidth*(gapwidth > 0.0 ? 2.0 : 1.0)+(halfwidth==0.0 ? 0.0 : ANTIALIASING);mediump vec2 dist=outset*a_extrude*scale;mediump float u=0.5*a_direction;mediump float t=1.0-abs(u);mediump vec2 offset2=offset*a_extrude*scale*normal.y*mat2(t,-u,u,t);vec4 projected_extrude=u_matrix*vec4(dist/u_ratio,0.0,0.0);gl_Position=u_matrix*vec4(pos+offset2/u_ratio,0.0,1.0)+projected_extrude;\n#ifdef TERRAIN3D\nv_gamma_scale=1.0;\n#else\nfloat extrude_length_without_perspective=length(dist);float extrude_length_with_perspective=length(projected_extrude.xy/gl_Position.w*u_units_to_pixels);v_gamma_scale=extrude_length_without_perspective/extrude_length_with_perspective;\n#endif\nv_tex_a=vec2(a_linesofar*u_patternscale_a.x/floorwidth,normal.y*u_patternscale_a.y+u_tex_y_a);v_tex_b=vec2(a_linesofar*u_patternscale_b.x/floorwidth,normal.y*u_patternscale_b.y+u_tex_y_b);v_width2=vec2(outset,inset);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/raster.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/raster.fragment.glsl new file mode 100644 index 00000000..8b8c55c1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/raster.fragment.glsl @@ -0,0 +1,53 @@ +uniform float u_fade_t; +uniform float u_opacity; +uniform sampler2D u_image0; +uniform sampler2D u_image1; + +in vec2 v_pos0; +in vec2 v_pos1; + +uniform float u_brightness_low; +uniform float u_brightness_high; + +uniform float u_saturation_factor; +uniform float u_contrast_factor; +uniform vec3 u_spin_weights; + +void main() { + + // read and cross-fade colors from the main and parent tiles + vec4 color0 = texture(u_image0, v_pos0); + vec4 color1 = texture(u_image1, v_pos1); + if (color0.a > 0.0) { + color0.rgb = color0.rgb / color0.a; + } + if (color1.a > 0.0) { + color1.rgb = color1.rgb / color1.a; + } + vec4 color = mix(color0, color1, u_fade_t); + color.a *= u_opacity; + vec3 rgb = color.rgb; + + // spin + rgb = vec3( + dot(rgb, u_spin_weights.xyz), + dot(rgb, u_spin_weights.zxy), + dot(rgb, u_spin_weights.yzx)); + + // saturation + float average = (color.r + color.g + color.b) / 3.0; + rgb += (average - rgb) * u_saturation_factor; + + // contrast + rgb = (rgb - 0.5) * u_contrast_factor + 0.5; + + // brightness + vec3 u_high_vec = vec3(u_brightness_low, u_brightness_low, u_brightness_low); + vec3 u_low_vec = vec3(u_brightness_high, u_brightness_high, u_brightness_high); + + fragColor = vec4(mix(u_high_vec, u_low_vec, rgb) * color.a, color.a); + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/raster.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/raster.fragment.glsl.g.ts new file mode 100644 index 00000000..1fb97c80 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/raster.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform float u_fade_t;uniform float u_opacity;uniform sampler2D u_image0;uniform sampler2D u_image1;varying vec2 v_pos0;varying vec2 v_pos1;uniform float u_brightness_low;uniform float u_brightness_high;uniform float u_saturation_factor;uniform float u_contrast_factor;uniform vec3 u_spin_weights;void main() {vec4 color0=texture2D(u_image0,v_pos0);vec4 color1=texture2D(u_image1,v_pos1);if (color0.a > 0.0) {color0.rgb=color0.rgb/color0.a;}if (color1.a > 0.0) {color1.rgb=color1.rgb/color1.a;}vec4 color=mix(color0,color1,u_fade_t);color.a*=u_opacity;vec3 rgb=color.rgb;rgb=vec3(dot(rgb,u_spin_weights.xyz),dot(rgb,u_spin_weights.zxy),dot(rgb,u_spin_weights.yzx));float average=(color.r+color.g+color.b)/3.0;rgb+=(average-rgb)*u_saturation_factor;rgb=(rgb-0.5)*u_contrast_factor+0.5;vec3 u_high_vec=vec3(u_brightness_low,u_brightness_low,u_brightness_low);vec3 u_low_vec=vec3(u_brightness_high,u_brightness_high,u_brightness_high);gl_FragColor=vec4(mix(u_high_vec,u_low_vec,rgb)*color.a,color.a);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/raster.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/raster.vertex.glsl new file mode 100644 index 00000000..04166a0c --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/raster.vertex.glsl @@ -0,0 +1,21 @@ +uniform mat4 u_matrix; +uniform vec2 u_tl_parent; +uniform float u_scale_parent; +uniform float u_buffer_scale; + +in vec2 a_pos; +in vec2 a_texture_pos; + +out vec2 v_pos0; +out vec2 v_pos1; + +void main() { + gl_Position = u_matrix * vec4(a_pos, 0, 1); + // We are using Int16 for texture position coordinates to give us enough precision for + // fractional coordinates. We use 8192 to scale the texture coordinates in the buffer + // as an arbitrarily high number to preserve adequate precision when rendering. + // This is also the same value as the EXTENT we are using for our tile buffer pos coordinates, + // so math for modifying either is consistent. + v_pos0 = (((a_texture_pos / 8192.0) - 0.5) / u_buffer_scale ) + 0.5; + v_pos1 = (v_pos0 * u_scale_parent) + u_tl_parent; +} diff --git a/web/libraries/maplibre-gl/src/shaders/raster.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/raster.vertex.glsl.g.ts new file mode 100644 index 00000000..1d412fea --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/raster.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform mat4 u_matrix;uniform vec2 u_tl_parent;uniform float u_scale_parent;uniform float u_buffer_scale;attribute vec2 a_pos;attribute vec2 a_texture_pos;varying vec2 v_pos0;varying vec2 v_pos1;void main() {gl_Position=u_matrix*vec4(a_pos,0,1);v_pos0=(((a_texture_pos/8192.0)-0.5)/u_buffer_scale )+0.5;v_pos1=(v_pos0*u_scale_parent)+u_tl_parent;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/shaders.ts b/web/libraries/maplibre-gl/src/shaders/shaders.ts new file mode 100644 index 00000000..383df3c9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/shaders.ts @@ -0,0 +1,193 @@ + +// Disable Flow annotations here because Flow doesn't support importing GLSL files + +import preludeFrag from './_prelude.fragment.glsl.g'; +import preludeVert from './_prelude.vertex.glsl.g'; +import backgroundFrag from './background.fragment.glsl.g'; +import backgroundVert from './background.vertex.glsl.g'; +import backgroundPatternFrag from './background_pattern.fragment.glsl.g'; +import backgroundPatternVert from './background_pattern.vertex.glsl.g'; +import circleFrag from './circle.fragment.glsl.g'; +import circleVert from './circle.vertex.glsl.g'; +import clippingMaskFrag from './clipping_mask.fragment.glsl.g'; +import clippingMaskVert from './clipping_mask.vertex.glsl.g'; +import heatmapFrag from './heatmap.fragment.glsl.g'; +import heatmapVert from './heatmap.vertex.glsl.g'; +import heatmapTextureFrag from './heatmap_texture.fragment.glsl.g'; +import heatmapTextureVert from './heatmap_texture.vertex.glsl.g'; +import collisionBoxFrag from './collision_box.fragment.glsl.g'; +import collisionBoxVert from './collision_box.vertex.glsl.g'; +import collisionCircleFrag from './collision_circle.fragment.glsl.g'; +import collisionCircleVert from './collision_circle.vertex.glsl.g'; +import debugFrag from './debug.fragment.glsl.g'; +import debugVert from './debug.vertex.glsl.g'; +import fillFrag from './fill.fragment.glsl.g'; +import fillVert from './fill.vertex.glsl.g'; +import fillOutlineFrag from './fill_outline.fragment.glsl.g'; +import fillOutlineVert from './fill_outline.vertex.glsl.g'; +import fillOutlinePatternFrag from './fill_outline_pattern.fragment.glsl.g'; +import fillOutlinePatternVert from './fill_outline_pattern.vertex.glsl.g'; +import fillPatternFrag from './fill_pattern.fragment.glsl.g'; +import fillPatternVert from './fill_pattern.vertex.glsl.g'; +import fillExtrusionFrag from './fill_extrusion.fragment.glsl.g'; +import fillExtrusionVert from './fill_extrusion.vertex.glsl.g'; +import fillExtrusionPatternFrag from './fill_extrusion_pattern.fragment.glsl.g'; +import fillExtrusionPatternVert from './fill_extrusion_pattern.vertex.glsl.g'; +import hillshadePrepareFrag from './hillshade_prepare.fragment.glsl.g'; +import hillshadePrepareVert from './hillshade_prepare.vertex.glsl.g'; +import hillshadeFrag from './hillshade.fragment.glsl.g'; +import hillshadeVert from './hillshade.vertex.glsl.g'; +import lineFrag from './line.fragment.glsl.g'; +import lineVert from './line.vertex.glsl.g'; +import lineGradientFrag from './line_gradient.fragment.glsl.g'; +import lineGradientVert from './line_gradient.vertex.glsl.g'; +import linePatternFrag from './line_pattern.fragment.glsl.g'; +import linePatternVert from './line_pattern.vertex.glsl.g'; +import lineSDFFrag from './line_sdf.fragment.glsl.g'; +import lineSDFVert from './line_sdf.vertex.glsl.g'; +import rasterFrag from './raster.fragment.glsl.g'; +import rasterVert from './raster.vertex.glsl.g'; +import symbolIconFrag from './symbol_icon.fragment.glsl.g'; +import symbolIconVert from './symbol_icon.vertex.glsl.g'; +import symbolSDFFrag from './symbol_sdf.fragment.glsl.g'; +import symbolSDFVert from './symbol_sdf.vertex.glsl.g'; +import symbolTextAndIconFrag from './symbol_text_and_icon.fragment.glsl.g'; +import symbolTextAndIconVert from './symbol_text_and_icon.vertex.glsl.g'; +import terrainDepthFrag from './terrain_depth.fragment.glsl.g'; +import terrainCoordsFrag from './terrain_coords.fragment.glsl.g'; +import terrainFrag from './terrain.fragment.glsl.g'; +import terrainVert from './terrain.vertex.glsl.g'; + +export const shaders = { + prelude: compile(preludeFrag, preludeVert), + background: compile(backgroundFrag, backgroundVert), + backgroundPattern: compile(backgroundPatternFrag, backgroundPatternVert), + circle: compile(circleFrag, circleVert), + clippingMask: compile(clippingMaskFrag, clippingMaskVert), + heatmap: compile(heatmapFrag, heatmapVert), + heatmapTexture: compile(heatmapTextureFrag, heatmapTextureVert), + collisionBox: compile(collisionBoxFrag, collisionBoxVert), + collisionCircle: compile(collisionCircleFrag, collisionCircleVert), + debug: compile(debugFrag, debugVert), + fill: compile(fillFrag, fillVert), + fillOutline: compile(fillOutlineFrag, fillOutlineVert), + fillOutlinePattern: compile(fillOutlinePatternFrag, fillOutlinePatternVert), + fillPattern: compile(fillPatternFrag, fillPatternVert), + fillExtrusion: compile(fillExtrusionFrag, fillExtrusionVert), + fillExtrusionPattern: compile(fillExtrusionPatternFrag, fillExtrusionPatternVert), + hillshadePrepare: compile(hillshadePrepareFrag, hillshadePrepareVert), + hillshade: compile(hillshadeFrag, hillshadeVert), + line: compile(lineFrag, lineVert), + lineGradient: compile(lineGradientFrag, lineGradientVert), + linePattern: compile(linePatternFrag, linePatternVert), + lineSDF: compile(lineSDFFrag, lineSDFVert), + raster: compile(rasterFrag, rasterVert), + symbolIcon: compile(symbolIconFrag, symbolIconVert), + symbolSDF: compile(symbolSDFFrag, symbolSDFVert), + symbolTextAndIcon: compile(symbolTextAndIconFrag, symbolTextAndIconVert), + terrain: compile(terrainFrag, terrainVert), + terrainDepth: compile(terrainDepthFrag, terrainVert), + terrainCoords: compile(terrainCoordsFrag, terrainVert) +}; + +// Expand #pragmas to #ifdefs. + +function compile(fragmentSource, vertexSource) { + const re = /#pragma mapbox: ([\w]+) ([\w]+) ([\w]+) ([\w]+)/g; + + const staticAttributes = vertexSource.match(/attribute ([\w]+) ([\w]+)/g); + const fragmentUniforms = fragmentSource.match(/uniform ([\w]+) ([\w]+)([\s]*)([\w]*)/g); + const vertexUniforms = vertexSource.match(/uniform ([\w]+) ([\w]+)([\s]*)([\w]*)/g); + const staticUniforms = vertexUniforms ? vertexUniforms.concat(fragmentUniforms) : fragmentUniforms; + + const fragmentPragmas = {}; + + fragmentSource = fragmentSource.replace(re, (match, operation, precision, type, name) => { + fragmentPragmas[name] = true; + if (operation === 'define') { + return ` +#ifndef HAS_UNIFORM_u_${name} +varying ${precision} ${type} ${name}; +#else +uniform ${precision} ${type} u_${name}; +#endif +`; + } else /* if (operation === 'initialize') */ { + return ` +#ifdef HAS_UNIFORM_u_${name} + ${precision} ${type} ${name} = u_${name}; +#endif +`; + } + }); + + vertexSource = vertexSource.replace(re, (match, operation, precision, type, name) => { + const attrType = type === 'float' ? 'vec2' : 'vec4'; + const unpackType = name.match(/color/) ? 'color' : attrType; + + if (fragmentPragmas[name]) { + if (operation === 'define') { + return ` +#ifndef HAS_UNIFORM_u_${name} +uniform lowp float u_${name}_t; +attribute ${precision} ${attrType} a_${name}; +varying ${precision} ${type} ${name}; +#else +uniform ${precision} ${type} u_${name}; +#endif +`; + } else /* if (operation === 'initialize') */ { + if (unpackType === 'vec4') { + // vec4 attributes are only used for cross-faded properties, and are not packed + return ` +#ifndef HAS_UNIFORM_u_${name} + ${name} = a_${name}; +#else + ${precision} ${type} ${name} = u_${name}; +#endif +`; + } else { + return ` +#ifndef HAS_UNIFORM_u_${name} + ${name} = unpack_mix_${unpackType}(a_${name}, u_${name}_t); +#else + ${precision} ${type} ${name} = u_${name}; +#endif +`; + } + } + } else { + if (operation === 'define') { + return ` +#ifndef HAS_UNIFORM_u_${name} +uniform lowp float u_${name}_t; +attribute ${precision} ${attrType} a_${name}; +#else +uniform ${precision} ${type} u_${name}; +#endif +`; + } else /* if (operation === 'initialize') */ { + if (unpackType === 'vec4') { + // vec4 attributes are only used for cross-faded properties, and are not packed + return ` +#ifndef HAS_UNIFORM_u_${name} + ${precision} ${type} ${name} = a_${name}; +#else + ${precision} ${type} ${name} = u_${name}; +#endif +`; + } else /* */ { + return ` +#ifndef HAS_UNIFORM_u_${name} + ${precision} ${type} ${name} = unpack_mix_${unpackType}(a_${name}, u_${name}_t); +#else + ${precision} ${type} ${name} = u_${name}; +#endif +`; + } + } + } + }); + + return {fragmentSource, vertexSource, staticAttributes, staticUniforms}; +} diff --git a/web/libraries/maplibre-gl/src/shaders/symbol_icon.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/symbol_icon.fragment.glsl new file mode 100644 index 00000000..ec81b936 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/symbol_icon.fragment.glsl @@ -0,0 +1,17 @@ +uniform sampler2D u_texture; + +in vec2 v_tex; +in float v_fade_opacity; + +#pragma mapbox: define lowp float opacity + +void main() { + #pragma mapbox: initialize lowp float opacity + + lowp float alpha = opacity * v_fade_opacity; + fragColor = texture(u_texture, v_tex) * alpha; + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/symbol_icon.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/symbol_icon.fragment.glsl.g.ts new file mode 100644 index 00000000..a4653059 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/symbol_icon.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform sampler2D u_texture;varying vec2 v_tex;varying float v_fade_opacity;\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\nlowp float alpha=opacity*v_fade_opacity;gl_FragColor=texture2D(u_texture,v_tex)*alpha;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/symbol_icon.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/symbol_icon.vertex.glsl new file mode 100644 index 00000000..a12ff351 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/symbol_icon.vertex.glsl @@ -0,0 +1,94 @@ +const float PI = 3.141592653589793; + +in vec4 a_pos_offset; +in vec4 a_data; +in vec4 a_pixeloffset; +in vec3 a_projected_pos; +in float a_fade_opacity; + +uniform bool u_is_size_zoom_constant; +uniform bool u_is_size_feature_constant; +uniform highp float u_size_t; // used to interpolate between zoom stops when size is a composite function +uniform highp float u_size; // used when size is both zoom and feature constant +uniform highp float u_camera_to_center_distance; +uniform highp float u_pitch; +uniform bool u_rotate_symbol; +uniform highp float u_aspect_ratio; +uniform float u_fade_change; +uniform mat4 u_matrix; +uniform mat4 u_label_plane_matrix; +uniform mat4 u_coord_matrix; +uniform bool u_is_text; +uniform bool u_pitch_with_map; +uniform vec2 u_texsize; + +out vec2 v_tex; +out float v_fade_opacity; + +#pragma mapbox: define lowp float opacity + +void main() { + #pragma mapbox: initialize lowp float opacity + + vec2 a_pos = a_pos_offset.xy; + vec2 a_offset = a_pos_offset.zw; + + vec2 a_tex = a_data.xy; + vec2 a_size = a_data.zw; + + float a_size_min = floor(a_size[0] * 0.5); + vec2 a_pxoffset = a_pixeloffset.xy; + vec2 a_minFontScale = a_pixeloffset.zw / 256.0; + + float ele = get_elevation(a_pos); + highp float segment_angle = -a_projected_pos[2]; + float size; + + if (!u_is_size_zoom_constant && !u_is_size_feature_constant) { + size = mix(a_size_min, a_size[1], u_size_t) / 128.0; + } else if (u_is_size_zoom_constant && !u_is_size_feature_constant) { + size = a_size_min / 128.0; + } else { + size = u_size; + } + + vec4 projectedPoint = u_matrix * vec4(a_pos, ele, 1); + highp float camera_to_anchor_distance = projectedPoint.w; + // See comments in symbol_sdf.vertex + highp float distance_ratio = u_pitch_with_map ? + camera_to_anchor_distance / u_camera_to_center_distance : + u_camera_to_center_distance / camera_to_anchor_distance; + highp float perspective_ratio = clamp( + 0.5 + 0.5 * distance_ratio, + 0.0, // Prevents oversized near-field symbols in pitched/overzoomed tiles + 4.0); + + size *= perspective_ratio; + + float fontScale = u_is_text ? size / 24.0 : size; + + highp float symbol_rotation = 0.0; + if (u_rotate_symbol) { + // See comments in symbol_sdf.vertex + vec4 offsetProjectedPoint = u_matrix * vec4(a_pos + vec2(1, 0), ele, 1); + + vec2 a = projectedPoint.xy / projectedPoint.w; + vec2 b = offsetProjectedPoint.xy / offsetProjectedPoint.w; + + symbol_rotation = atan((b.y - a.y) / u_aspect_ratio, b.x - a.x); + } + + highp float angle_sin = sin(segment_angle + symbol_rotation); + highp float angle_cos = cos(segment_angle + symbol_rotation); + mat2 rotation_matrix = mat2(angle_cos, -1.0 * angle_sin, angle_sin, angle_cos); + + vec4 projected_pos = u_label_plane_matrix * vec4(a_projected_pos.xy, ele, 1.0); + float z = float(u_pitch_with_map) * projected_pos.z / projected_pos.w; + gl_Position = u_coord_matrix * vec4(projected_pos.xy / projected_pos.w + rotation_matrix * (a_offset / 32.0 * max(a_minFontScale, fontScale) + a_pxoffset / 16.0), z, 1.0); + + v_tex = a_tex / u_texsize; + vec2 fade_opacity = unpack_opacity(a_fade_opacity); + float fade_change = fade_opacity[1] > 0.5 ? u_fade_change : -u_fade_change; + float visibility = calculate_visibility(projectedPoint); + v_fade_opacity = max(0.0, min(visibility, fade_opacity[0] + fade_change)); +} diff --git a/web/libraries/maplibre-gl/src/shaders/symbol_icon.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/symbol_icon.vertex.glsl.g.ts new file mode 100644 index 00000000..ae7a2fb1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/symbol_icon.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform highp float u_camera_to_center_distance;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform float u_fade_change;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform vec2 u_texsize;varying vec2 v_tex;varying float v_fade_opacity;\n#pragma mapbox: define lowp float opacity\nvoid main() {\n#pragma mapbox: initialize lowp float opacity\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;vec2 a_minFontScale=a_pixeloffset.zw/256.0;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\ncamera_to_anchor_distance/u_camera_to_center_distance :\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*max(a_minFontScale,fontScale)+a_pxoffset/16.0),z,1.0);v_tex=a_tex/u_texsize;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float visibility=calculate_visibility(projectedPoint);v_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));}'; diff --git a/web/libraries/maplibre-gl/src/shaders/symbol_sdf.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/symbol_sdf.fragment.glsl new file mode 100644 index 00000000..a6ed7f0b --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/symbol_sdf.fragment.glsl @@ -0,0 +1,58 @@ +#define SDF_PX 8.0 + +uniform bool u_is_halo; +uniform sampler2D u_texture; +uniform highp float u_gamma_scale; +uniform lowp float u_device_pixel_ratio; +uniform bool u_is_text; + +in vec2 v_data0; +in vec3 v_data1; + +#pragma mapbox: define highp vec4 fill_color +#pragma mapbox: define highp vec4 halo_color +#pragma mapbox: define lowp float opacity +#pragma mapbox: define lowp float halo_width +#pragma mapbox: define lowp float halo_blur + +void main() { + #pragma mapbox: initialize highp vec4 fill_color + #pragma mapbox: initialize highp vec4 halo_color + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize lowp float halo_width + #pragma mapbox: initialize lowp float halo_blur + + float EDGE_GAMMA = 0.105 / u_device_pixel_ratio; + + vec2 tex = v_data0.xy; + float gamma_scale = v_data1.x; + float size = v_data1.y; + float fade_opacity = v_data1[2]; + + float fontScale = u_is_text ? size / 24.0 : size; + + lowp vec4 color = fill_color; + highp float gamma = EDGE_GAMMA / (fontScale * u_gamma_scale); + lowp float inner_edge = (256.0 - 64.0) / 256.0; + if (u_is_halo) { + color = halo_color; + gamma = (halo_blur * 1.19 / SDF_PX + EDGE_GAMMA) / (fontScale * u_gamma_scale); + inner_edge = inner_edge + gamma * gamma_scale; + } + + lowp float dist = texture(u_texture, tex).a; + highp float gamma_scaled = gamma * gamma_scale; + highp float alpha = smoothstep(inner_edge - gamma_scaled, inner_edge + gamma_scaled, dist); + if (u_is_halo) { + // When drawing halos, we want the inside of the halo to be transparent as well + // in case the text fill is transparent. + lowp float halo_edge = (6.0 - halo_width / fontScale) / SDF_PX; + alpha = min(smoothstep(halo_edge - gamma_scaled, halo_edge + gamma_scaled, dist), 1.0 - alpha); + } + + fragColor = color * (alpha * opacity * fade_opacity); + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/symbol_sdf.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/symbol_sdf.fragment.glsl.g.ts new file mode 100644 index 00000000..ed1e4655 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/symbol_sdf.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default '#define SDF_PX 8.0\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;uniform bool u_is_text;varying vec2 v_data0;varying vec3 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nfloat EDGE_GAMMA=0.105/u_device_pixel_ratio;vec2 tex=v_data0.xy;float gamma_scale=v_data1.x;float size=v_data1.y;float fade_opacity=v_data1[2];float fontScale=u_is_text ? size/24.0 : size;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float inner_edge=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);inner_edge=inner_edge+gamma*gamma_scale;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(inner_edge-gamma_scaled,inner_edge+gamma_scaled,dist);if (u_is_halo) {lowp float halo_edge=(6.0-halo_width/fontScale)/SDF_PX;alpha=min(smoothstep(halo_edge-gamma_scaled,halo_edge+gamma_scaled,dist),1.0-alpha);}gl_FragColor=color*(alpha*opacity*fade_opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/symbol_sdf.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/symbol_sdf.vertex.glsl new file mode 100644 index 00000000..8c041358 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/symbol_sdf.vertex.glsl @@ -0,0 +1,118 @@ +const float PI = 3.141592653589793; + +in vec4 a_pos_offset; +in vec4 a_data; +in vec4 a_pixeloffset; +in vec3 a_projected_pos; +in float a_fade_opacity; + +// contents of a_size vary based on the type of property value +// used for {text,icon}-size. +// For constants, a_size is disabled. +// For source functions, we bind only one value per vertex: the value of {text,icon}-size evaluated for the current feature. +// For composite functions: +// [ text-size(lowerZoomStop, feature), +// text-size(upperZoomStop, feature) ] +uniform bool u_is_size_zoom_constant; +uniform bool u_is_size_feature_constant; +uniform highp float u_size_t; // used to interpolate between zoom stops when size is a composite function +uniform highp float u_size; // used when size is both zoom and feature constant +uniform mat4 u_matrix; +uniform mat4 u_label_plane_matrix; +uniform mat4 u_coord_matrix; +uniform bool u_is_text; +uniform bool u_pitch_with_map; +uniform highp float u_pitch; +uniform bool u_rotate_symbol; +uniform highp float u_aspect_ratio; +uniform highp float u_camera_to_center_distance; +uniform float u_fade_change; +uniform vec2 u_texsize; + +out vec2 v_data0; +out vec3 v_data1; + +#pragma mapbox: define highp vec4 fill_color +#pragma mapbox: define highp vec4 halo_color +#pragma mapbox: define lowp float opacity +#pragma mapbox: define lowp float halo_width +#pragma mapbox: define lowp float halo_blur + +void main() { + #pragma mapbox: initialize highp vec4 fill_color + #pragma mapbox: initialize highp vec4 halo_color + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize lowp float halo_width + #pragma mapbox: initialize lowp float halo_blur + + vec2 a_pos = a_pos_offset.xy; + vec2 a_offset = a_pos_offset.zw; + + vec2 a_tex = a_data.xy; + vec2 a_size = a_data.zw; + + float a_size_min = floor(a_size[0] * 0.5); + vec2 a_pxoffset = a_pixeloffset.xy; + + float ele = get_elevation(a_pos); + highp float segment_angle = -a_projected_pos[2]; + float size; + + if (!u_is_size_zoom_constant && !u_is_size_feature_constant) { + size = mix(a_size_min, a_size[1], u_size_t) / 128.0; + } else if (u_is_size_zoom_constant && !u_is_size_feature_constant) { + size = a_size_min / 128.0; + } else { + size = u_size; + } + + vec4 projectedPoint = u_matrix * vec4(a_pos, ele, 1); + highp float camera_to_anchor_distance = projectedPoint.w; + // If the label is pitched with the map, layout is done in pitched space, + // which makes labels in the distance smaller relative to viewport space. + // We counteract part of that effect by multiplying by the perspective ratio. + // If the label isn't pitched with the map, we do layout in viewport space, + // which makes labels in the distance larger relative to the features around + // them. We counteract part of that effect by dividing by the perspective ratio. + highp float distance_ratio = u_pitch_with_map ? + camera_to_anchor_distance / u_camera_to_center_distance : + u_camera_to_center_distance / camera_to_anchor_distance; + highp float perspective_ratio = clamp( + 0.5 + 0.5 * distance_ratio, + 0.0, // Prevents oversized near-field symbols in pitched/overzoomed tiles + 4.0); + + size *= perspective_ratio; + + float fontScale = u_is_text ? size / 24.0 : size; + + highp float symbol_rotation = 0.0; + if (u_rotate_symbol) { + // Point labels with 'rotation-alignment: map' are horizontal with respect to tile units + // To figure out that angle in projected space, we draw a short horizontal line in tile + // space, project it, and measure its angle in projected space. + vec4 offsetProjectedPoint = u_matrix * vec4(a_pos + vec2(1, 0), ele, 1); + + vec2 a = projectedPoint.xy / projectedPoint.w; + vec2 b = offsetProjectedPoint.xy / offsetProjectedPoint.w; + + symbol_rotation = atan((b.y - a.y) / u_aspect_ratio, b.x - a.x); + } + + highp float angle_sin = sin(segment_angle + symbol_rotation); + highp float angle_cos = cos(segment_angle + symbol_rotation); + mat2 rotation_matrix = mat2(angle_cos, -1.0 * angle_sin, angle_sin, angle_cos); + + vec4 projected_pos = u_label_plane_matrix * vec4(a_projected_pos.xy, ele, 1.0); + float z = float(u_pitch_with_map) * projected_pos.z / projected_pos.w; + gl_Position = u_coord_matrix * vec4(projected_pos.xy / projected_pos.w + rotation_matrix * (a_offset / 32.0 * fontScale + a_pxoffset), z, 1.0); + float gamma_scale = gl_Position.w; + + vec2 fade_opacity = unpack_opacity(a_fade_opacity); + float visibility = calculate_visibility(projectedPoint); + float fade_change = fade_opacity[1] > 0.5 ? u_fade_change : -u_fade_change; + float interpolated_fade_opacity = max(0.0, min(visibility, fade_opacity[0] + fade_change)); + + v_data0 = a_tex / u_texsize; + v_data1 = vec3(gamma_scale, size, interpolated_fade_opacity); +} diff --git a/web/libraries/maplibre-gl/src/shaders/symbol_sdf.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/symbol_sdf.vertex.glsl.g.ts new file mode 100644 index 00000000..da1da50d --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/symbol_sdf.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec4 a_pixeloffset;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;varying vec2 v_data0;varying vec3 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);vec2 a_pxoffset=a_pixeloffset.xy;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\ncamera_to_anchor_distance/u_camera_to_center_distance :\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=u_is_text ? size/24.0 : size;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale+a_pxoffset),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0=a_tex/u_texsize;v_data1=vec3(gamma_scale,size,interpolated_fade_opacity);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/symbol_text_and_icon.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/symbol_text_and_icon.fragment.glsl new file mode 100644 index 00000000..672e1dcb --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/symbol_text_and_icon.fragment.glsl @@ -0,0 +1,68 @@ +#define SDF_PX 8.0 + +#define SDF 1.0 +#define ICON 0.0 + +uniform bool u_is_halo; +uniform sampler2D u_texture; +uniform sampler2D u_texture_icon; +uniform highp float u_gamma_scale; +uniform lowp float u_device_pixel_ratio; + +in vec4 v_data0; +in vec4 v_data1; + +#pragma mapbox: define highp vec4 fill_color +#pragma mapbox: define highp vec4 halo_color +#pragma mapbox: define lowp float opacity +#pragma mapbox: define lowp float halo_width +#pragma mapbox: define lowp float halo_blur + +void main() { + #pragma mapbox: initialize highp vec4 fill_color + #pragma mapbox: initialize highp vec4 halo_color + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize lowp float halo_width + #pragma mapbox: initialize lowp float halo_blur + + float fade_opacity = v_data1[2]; + + if (v_data1.w == ICON) { + vec2 tex_icon = v_data0.zw; + lowp float alpha = opacity * fade_opacity; + fragColor = texture(u_texture_icon, tex_icon) * alpha; + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif + return; + } + + vec2 tex = v_data0.xy; + + float EDGE_GAMMA = 0.105 / u_device_pixel_ratio; + + float gamma_scale = v_data1.x; + float size = v_data1.y; + + float fontScale = size / 24.0; + + lowp vec4 color = fill_color; + highp float gamma = EDGE_GAMMA / (fontScale * u_gamma_scale); + lowp float buff = (256.0 - 64.0) / 256.0; + if (u_is_halo) { + color = halo_color; + gamma = (halo_blur * 1.19 / SDF_PX + EDGE_GAMMA) / (fontScale * u_gamma_scale); + buff = (6.0 - halo_width / fontScale) / SDF_PX; + } + + lowp float dist = texture(u_texture, tex).a; + highp float gamma_scaled = gamma * gamma_scale; + highp float alpha = smoothstep(buff - gamma_scaled, buff + gamma_scaled, dist); + + fragColor = color * (alpha * opacity * fade_opacity); + +#ifdef OVERDRAW_INSPECTOR + fragColor = vec4(1.0); +#endif +} diff --git a/web/libraries/maplibre-gl/src/shaders/symbol_text_and_icon.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/symbol_text_and_icon.fragment.glsl.g.ts new file mode 100644 index 00000000..fbcfadb9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/symbol_text_and_icon.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default '#define SDF_PX 8.0\n#define SDF 1.0\n#define ICON 0.0\nuniform bool u_is_halo;uniform sampler2D u_texture;uniform sampler2D u_texture_icon;uniform highp float u_gamma_scale;uniform lowp float u_device_pixel_ratio;varying vec4 v_data0;varying vec4 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nfloat fade_opacity=v_data1[2];if (v_data1.w==ICON) {vec2 tex_icon=v_data0.zw;lowp float alpha=opacity*fade_opacity;gl_FragColor=texture2D(u_texture_icon,tex_icon)*alpha;\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\nreturn;}vec2 tex=v_data0.xy;float EDGE_GAMMA=0.105/u_device_pixel_ratio;float gamma_scale=v_data1.x;float size=v_data1.y;float fontScale=size/24.0;lowp vec4 color=fill_color;highp float gamma=EDGE_GAMMA/(fontScale*u_gamma_scale);lowp float buff=(256.0-64.0)/256.0;if (u_is_halo) {color=halo_color;gamma=(halo_blur*1.19/SDF_PX+EDGE_GAMMA)/(fontScale*u_gamma_scale);buff=(6.0-halo_width/fontScale)/SDF_PX;}lowp float dist=texture2D(u_texture,tex).a;highp float gamma_scaled=gamma*gamma_scale;highp float alpha=smoothstep(buff-gamma_scaled,buff+gamma_scaled,dist);gl_FragColor=color*(alpha*opacity*fade_opacity);\n#ifdef OVERDRAW_INSPECTOR\ngl_FragColor=vec4(1.0);\n#endif\n}'; diff --git a/web/libraries/maplibre-gl/src/shaders/symbol_text_and_icon.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/symbol_text_and_icon.vertex.glsl new file mode 100644 index 00000000..e9e3bf9e --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/symbol_text_and_icon.vertex.glsl @@ -0,0 +1,119 @@ +const float PI = 3.141592653589793; + +in vec4 a_pos_offset; +in vec4 a_data; +in vec3 a_projected_pos; +in float a_fade_opacity; + +// contents of a_size vary based on the type of property value +// used for {text,icon}-size. +// For constants, a_size is disabled. +// For source functions, we bind only one value per vertex: the value of {text,icon}-size evaluated for the current feature. +// For composite functions: +// [ text-size(lowerZoomStop, feature), +// text-size(upperZoomStop, feature) ] +uniform bool u_is_size_zoom_constant; +uniform bool u_is_size_feature_constant; +uniform highp float u_size_t; // used to interpolate between zoom stops when size is a composite function +uniform highp float u_size; // used when size is both zoom and feature constant +uniform mat4 u_matrix; +uniform mat4 u_label_plane_matrix; +uniform mat4 u_coord_matrix; +uniform bool u_is_text; +uniform bool u_pitch_with_map; +uniform highp float u_pitch; +uniform bool u_rotate_symbol; +uniform highp float u_aspect_ratio; +uniform highp float u_camera_to_center_distance; +uniform float u_fade_change; +uniform vec2 u_texsize; +uniform vec2 u_texsize_icon; + +out vec4 v_data0; +out vec4 v_data1; + +#pragma mapbox: define highp vec4 fill_color +#pragma mapbox: define highp vec4 halo_color +#pragma mapbox: define lowp float opacity +#pragma mapbox: define lowp float halo_width +#pragma mapbox: define lowp float halo_blur + +void main() { + #pragma mapbox: initialize highp vec4 fill_color + #pragma mapbox: initialize highp vec4 halo_color + #pragma mapbox: initialize lowp float opacity + #pragma mapbox: initialize lowp float halo_width + #pragma mapbox: initialize lowp float halo_blur + + vec2 a_pos = a_pos_offset.xy; + vec2 a_offset = a_pos_offset.zw; + + vec2 a_tex = a_data.xy; + vec2 a_size = a_data.zw; + + float a_size_min = floor(a_size[0] * 0.5); + float is_sdf = a_size[0] - 2.0 * a_size_min; + + float ele = get_elevation(a_pos); + highp float segment_angle = -a_projected_pos[2]; + float size; + + if (!u_is_size_zoom_constant && !u_is_size_feature_constant) { + size = mix(a_size_min, a_size[1], u_size_t) / 128.0; + } else if (u_is_size_zoom_constant && !u_is_size_feature_constant) { + size = a_size_min / 128.0; + } else { + size = u_size; + } + + vec4 projectedPoint = u_matrix * vec4(a_pos, ele, 1); + highp float camera_to_anchor_distance = projectedPoint.w; + // If the label is pitched with the map, layout is done in pitched space, + // which makes labels in the distance smaller relative to viewport space. + // We counteract part of that effect by multiplying by the perspective ratio. + // If the label isn't pitched with the map, we do layout in viewport space, + // which makes labels in the distance larger relative to the features around + // them. We counteract part of that effect by dividing by the perspective ratio. + highp float distance_ratio = u_pitch_with_map ? + camera_to_anchor_distance / u_camera_to_center_distance : + u_camera_to_center_distance / camera_to_anchor_distance; + highp float perspective_ratio = clamp( + 0.5 + 0.5 * distance_ratio, + 0.0, // Prevents oversized near-field symbols in pitched/overzoomed tiles + 4.0); + + size *= perspective_ratio; + + float fontScale = size / 24.0; + + highp float symbol_rotation = 0.0; + if (u_rotate_symbol) { + // Point labels with 'rotation-alignment: map' are horizontal with respect to tile units + // To figure out that angle in projected space, we draw a short horizontal line in tile + // space, project it, and measure its angle in projected space. + vec4 offsetProjectedPoint = u_matrix * vec4(a_pos + vec2(1, 0), ele, 1); + + vec2 a = projectedPoint.xy / projectedPoint.w; + vec2 b = offsetProjectedPoint.xy / offsetProjectedPoint.w; + + symbol_rotation = atan((b.y - a.y) / u_aspect_ratio, b.x - a.x); + } + + highp float angle_sin = sin(segment_angle + symbol_rotation); + highp float angle_cos = cos(segment_angle + symbol_rotation); + mat2 rotation_matrix = mat2(angle_cos, -1.0 * angle_sin, angle_sin, angle_cos); + + vec4 projected_pos = u_label_plane_matrix * vec4(a_projected_pos.xy, ele, 1.0); + float z = float(u_pitch_with_map) * projected_pos.z / projected_pos.w; + gl_Position = u_coord_matrix * vec4(projected_pos.xy / projected_pos.w + rotation_matrix * (a_offset / 32.0 * fontScale), z, 1.0); + float gamma_scale = gl_Position.w; + + vec2 fade_opacity = unpack_opacity(a_fade_opacity); + float visibility = calculate_visibility(projectedPoint); + float fade_change = fade_opacity[1] > 0.5 ? u_fade_change : -u_fade_change; + float interpolated_fade_opacity = max(0.0, min(visibility, fade_opacity[0] + fade_change)); + + v_data0.xy = a_tex / u_texsize; + v_data0.zw = a_tex / u_texsize_icon; + v_data1 = vec4(gamma_scale, size, interpolated_fade_opacity, is_sdf); +} diff --git a/web/libraries/maplibre-gl/src/shaders/symbol_text_and_icon.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/symbol_text_and_icon.vertex.glsl.g.ts new file mode 100644 index 00000000..c4286dab --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/symbol_text_and_icon.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'const float PI=3.141592653589793;attribute vec4 a_pos_offset;attribute vec4 a_data;attribute vec3 a_projected_pos;attribute float a_fade_opacity;uniform bool u_is_size_zoom_constant;uniform bool u_is_size_feature_constant;uniform highp float u_size_t;uniform highp float u_size;uniform mat4 u_matrix;uniform mat4 u_label_plane_matrix;uniform mat4 u_coord_matrix;uniform bool u_is_text;uniform bool u_pitch_with_map;uniform highp float u_pitch;uniform bool u_rotate_symbol;uniform highp float u_aspect_ratio;uniform highp float u_camera_to_center_distance;uniform float u_fade_change;uniform vec2 u_texsize;uniform vec2 u_texsize_icon;varying vec4 v_data0;varying vec4 v_data1;\n#pragma mapbox: define highp vec4 fill_color\n#pragma mapbox: define highp vec4 halo_color\n#pragma mapbox: define lowp float opacity\n#pragma mapbox: define lowp float halo_width\n#pragma mapbox: define lowp float halo_blur\nvoid main() {\n#pragma mapbox: initialize highp vec4 fill_color\n#pragma mapbox: initialize highp vec4 halo_color\n#pragma mapbox: initialize lowp float opacity\n#pragma mapbox: initialize lowp float halo_width\n#pragma mapbox: initialize lowp float halo_blur\nvec2 a_pos=a_pos_offset.xy;vec2 a_offset=a_pos_offset.zw;vec2 a_tex=a_data.xy;vec2 a_size=a_data.zw;float a_size_min=floor(a_size[0]*0.5);float is_sdf=a_size[0]-2.0*a_size_min;float ele=get_elevation(a_pos);highp float segment_angle=-a_projected_pos[2];float size;if (!u_is_size_zoom_constant && !u_is_size_feature_constant) {size=mix(a_size_min,a_size[1],u_size_t)/128.0;} else if (u_is_size_zoom_constant && !u_is_size_feature_constant) {size=a_size_min/128.0;} else {size=u_size;}vec4 projectedPoint=u_matrix*vec4(a_pos,ele,1);highp float camera_to_anchor_distance=projectedPoint.w;highp float distance_ratio=u_pitch_with_map ?\ncamera_to_anchor_distance/u_camera_to_center_distance :\nu_camera_to_center_distance/camera_to_anchor_distance;highp float perspective_ratio=clamp(0.5+0.5*distance_ratio,0.0,4.0);size*=perspective_ratio;float fontScale=size/24.0;highp float symbol_rotation=0.0;if (u_rotate_symbol) {vec4 offsetProjectedPoint=u_matrix*vec4(a_pos+vec2(1,0),ele,1);vec2 a=projectedPoint.xy/projectedPoint.w;vec2 b=offsetProjectedPoint.xy/offsetProjectedPoint.w;symbol_rotation=atan((b.y-a.y)/u_aspect_ratio,b.x-a.x);}highp float angle_sin=sin(segment_angle+symbol_rotation);highp float angle_cos=cos(segment_angle+symbol_rotation);mat2 rotation_matrix=mat2(angle_cos,-1.0*angle_sin,angle_sin,angle_cos);vec4 projected_pos=u_label_plane_matrix*vec4(a_projected_pos.xy,ele,1.0);float z=float(u_pitch_with_map)*projected_pos.z/projected_pos.w;gl_Position=u_coord_matrix*vec4(projected_pos.xy/projected_pos.w+rotation_matrix*(a_offset/32.0*fontScale),z,1.0);float gamma_scale=gl_Position.w;vec2 fade_opacity=unpack_opacity(a_fade_opacity);float visibility=calculate_visibility(projectedPoint);float fade_change=fade_opacity[1] > 0.5 ? u_fade_change :-u_fade_change;float interpolated_fade_opacity=max(0.0,min(visibility,fade_opacity[0]+fade_change));v_data0.xy=a_tex/u_texsize;v_data0.zw=a_tex/u_texsize_icon;v_data1=vec4(gamma_scale,size,interpolated_fade_opacity,is_sdf);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/terrain.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/terrain.fragment.glsl new file mode 100644 index 00000000..de44a497 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/terrain.fragment.glsl @@ -0,0 +1,7 @@ +uniform sampler2D u_texture; + +in vec2 v_texture_pos; + +void main() { + fragColor = texture(u_texture, v_texture_pos); +} diff --git a/web/libraries/maplibre-gl/src/shaders/terrain.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/terrain.fragment.glsl.g.ts new file mode 100644 index 00000000..1c2c99a8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/terrain.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'uniform sampler2D u_texture;varying vec2 v_texture_pos;void main() {gl_FragColor=texture2D(u_texture,v_texture_pos);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/terrain.vertex.glsl b/web/libraries/maplibre-gl/src/shaders/terrain.vertex.glsl new file mode 100644 index 00000000..6e091944 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/terrain.vertex.glsl @@ -0,0 +1,15 @@ +in vec3 a_pos3d; + +uniform mat4 u_matrix; +uniform float u_ele_delta; + +out vec2 v_texture_pos; +out float v_depth; + +void main() { + float extent = 8192.0; // 8192.0 is the hardcoded vector-tiles coordinates resolution + float ele_delta = a_pos3d.z == 1.0 ? u_ele_delta : 0.0; + v_texture_pos = a_pos3d.xy / extent; + gl_Position = u_matrix * vec4(a_pos3d.xy, get_elevation(a_pos3d.xy) - ele_delta, 1.0); + v_depth = gl_Position.z / gl_Position.w; +} diff --git a/web/libraries/maplibre-gl/src/shaders/terrain.vertex.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/terrain.vertex.glsl.g.ts new file mode 100644 index 00000000..38ac8b33 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/terrain.vertex.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'attribute vec3 a_pos3d;uniform mat4 u_matrix;uniform float u_ele_delta;varying vec2 v_texture_pos;varying float v_depth;void main() {float extent=8192.0;float ele_delta=a_pos3d.z==1.0 ? u_ele_delta : 0.0;v_texture_pos=a_pos3d.xy/extent;gl_Position=u_matrix*vec4(a_pos3d.xy,get_elevation(a_pos3d.xy)-ele_delta,1.0);v_depth=gl_Position.z/gl_Position.w;}'; diff --git a/web/libraries/maplibre-gl/src/shaders/terrain_coords.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/terrain_coords.fragment.glsl new file mode 100644 index 00000000..05609c14 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/terrain_coords.fragment.glsl @@ -0,0 +1,11 @@ +precision mediump float; + +uniform sampler2D u_texture; +uniform float u_terrain_coords_id; + +in vec2 v_texture_pos; + +void main() { + vec4 rgba = texture(u_texture, v_texture_pos); + fragColor = vec4(rgba.r, rgba.g, rgba.b, u_terrain_coords_id); +} diff --git a/web/libraries/maplibre-gl/src/shaders/terrain_coords.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/terrain_coords.fragment.glsl.g.ts new file mode 100644 index 00000000..4654e5aa --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/terrain_coords.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'precision mediump float;uniform sampler2D u_texture;uniform float u_terrain_coords_id;varying vec2 v_texture_pos;void main() {vec4 rgba=texture2D(u_texture,v_texture_pos);gl_FragColor=vec4(rgba.r,rgba.g,rgba.b,u_terrain_coords_id);}'; diff --git a/web/libraries/maplibre-gl/src/shaders/terrain_depth.fragment.glsl b/web/libraries/maplibre-gl/src/shaders/terrain_depth.fragment.glsl new file mode 100644 index 00000000..2e497f08 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/terrain_depth.fragment.glsl @@ -0,0 +1,15 @@ +in float v_depth; + +// methods for pack/unpack depth value to texture rgba +// https://stackoverflow.com/questions/34963366/encode-floating-point-data-in-a-rgba-texture +const highp vec4 bitSh = vec4(256. * 256. * 256., 256. * 256., 256., 1.); +const highp vec4 bitMsk = vec4(0.,vec3(1./256.0)); +highp vec4 pack(highp float value) { + highp vec4 comp = fract(value * bitSh); + comp -= comp.xxyz * bitMsk; + return comp; +} + +void main() { + fragColor = pack(v_depth); +} diff --git a/web/libraries/maplibre-gl/src/shaders/terrain_depth.fragment.glsl.g.ts b/web/libraries/maplibre-gl/src/shaders/terrain_depth.fragment.glsl.g.ts new file mode 100644 index 00000000..be8398f7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/shaders/terrain_depth.fragment.glsl.g.ts @@ -0,0 +1,2 @@ +// This file is generated. Edit build/generate-shaders.ts, then run `npm run codegen`. +export default 'varying float v_depth;const highp vec4 bitSh=vec4(256.*256.*256.,256.*256.,256.,1.);const highp vec4 bitMsk=vec4(0.,vec3(1./256.0));highp vec4 pack(highp float value) {highp vec4 comp=fract(value*bitSh);comp-=comp.xxyz*bitMsk;return comp;}void main() {gl_FragColor=pack(v_depth);}'; diff --git a/web/libraries/maplibre-gl/src/source/canvas_source.test.ts b/web/libraries/maplibre-gl/src/source/canvas_source.test.ts new file mode 100644 index 00000000..306ba5f5 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/canvas_source.test.ts @@ -0,0 +1,210 @@ +import {CanvasSource} from '../source/canvas_source'; +import {Transform} from '../geo/transform'; +import {Event, Evented} from '../util/evented'; +import {extend} from '../util/util'; + +import type {Dispatcher} from '../util/dispatcher'; +import {Tile} from './tile'; +import {OverscaledTileID} from './tile_id'; +import {VertexBuffer} from '../gl/vertex_buffer'; +import {SegmentVector} from '../data/segment'; + +function createSource(options?) { + const c = options && options.canvas || window.document.createElement('canvas'); + c.width = 20; + c.height = 20; + + options = extend({ + canvas: 'id', + coordinates: [[0, 0], [1, 0], [1, 1], [0, 1]], + }, options); + + const source = new CanvasSource('id', options, {} as Dispatcher, options.eventedParent); + + source.canvas = c; + + return source; +} + +class StubMap extends Evented { + transform: Transform; + style: any; + painter: any; + + constructor() { + super(); + this.transform = new Transform(); + this.style = {}; + this.painter = { + context: { + gl: {} + } + }; + } + + triggerRepaint() { + this.fire(new Event('rerender')); + } +} + +describe('CanvasSource', () => { + let map; + beforeEach(() => { + map = new StubMap(); + }); + + test('constructor', done => { + const source = createSource(); + + expect(source.minzoom).toBe(0); + expect(source.maxzoom).toBe(22); + expect(source.tileSize).toBe(512); + expect(source.animate).toBe(true); + source.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'metadata') { + expect(typeof source.play).toBe('function'); + done(); + } + }); + + source.onAdd(map); + }); + + test('self-validates', () => { + const stub = jest.spyOn(console, 'error').mockImplementation(() => {}); + createSource({coordinates: []}); + expect(stub).toHaveBeenCalled(); + stub.mockReset(); + + createSource({coordinates: 'asdf'}); + expect(stub).toHaveBeenCalled(); + stub.mockReset(); + + createSource({animate: 8}); + expect(stub).toHaveBeenCalled(); + stub.mockReset(); + + createSource({canvas: {}}); + expect(stub).toHaveBeenCalled(); + stub.mockReset(); + + const canvasEl = window.document.createElement('canvas'); + createSource({canvas: canvasEl}); + expect(stub).not.toHaveBeenCalled(); + stub.mockReset(); + + }); + + test('can be initialized with HTML element', done => { + const el = window.document.createElement('canvas'); + const source = createSource({ + canvas: el + }); + + source.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'metadata') { + expect(source.canvas).toBe(el); + done(); + } + }); + + source.onAdd(map); + }); + + test('rerenders if animated', done => { + const source = createSource(); + + map.on('rerender', () => { + expect(true).toBeTruthy(); + done(); + }); + + source.onAdd(map); + }); + + test('can be static', done => { + const source = createSource({ + animate: false + }); + + map.on('rerender', () => { + // this just confirms it didn't happen, so no need to run done() here + // if called the test will fail + expect(true).toBeFalsy(); + }); + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata' && e.dataType === 'source') { + expect(true).toBeTruthy(); + done(); + } + }); + + source.onAdd(map); + }); + + test('onRemove stops animation', () => { + const source = createSource(); + + source.onAdd(map); + + expect(source.hasTransition()).toBe(true); + + source.onRemove(); + + expect(source.hasTransition()).toBe(false); + + source.onAdd(map); + + expect(source.hasTransition()).toBe(true); + + }); + + test('play and pause animation', () => { + const source = createSource(); + + source.onAdd(map); + + expect(source.hasTransition()).toBe(true); + + source.pause(); + + expect(source.hasTransition()).toBe(false); + + source.play(); + + expect(source.hasTransition()).toBe(true); + + }); + + test('fires idle event on prepare call when there is at least one not loaded tile', done => { + const source = createSource(); + const tile = new Tile(new OverscaledTileID(1, 0, 1, 0, 0), 512); + source.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'idle') { + expect(tile.state).toBe('loaded'); + done(); + } + }); + source.onAdd(map); + + source.tiles[String(tile.tileID.wrap)] = tile; + // assign dummies directly so we don't need to stub the gl things + source.boundsBuffer = {} as VertexBuffer; + source.boundsSegments = {} as SegmentVector; + source.texture = { + update: () => {} + } as any; + source.prepare(); + }); + +}); + +describe('CanvasSource#serialize', () => { + const source = createSource(); + + const serialized = source.serialize(); + expect(serialized.type).toBe('canvas'); + expect(serialized.coordinates).toEqual([[0, 0], [1, 0], [1, 1], [0, 1]]); + +}); diff --git a/web/libraries/maplibre-gl/src/source/canvas_source.ts b/web/libraries/maplibre-gl/src/source/canvas_source.ts new file mode 100644 index 00000000..72994df2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/canvas_source.ts @@ -0,0 +1,227 @@ +import {ImageSource} from './image_source'; + +import rasterBoundsAttributes from '../data/raster_bounds_attributes'; +import {SegmentVector} from '../data/segment'; +import {Texture} from '../render/texture'; +import {Event, ErrorEvent} from '../util/evented'; +import {ValidationError} from '@maplibre/maplibre-gl-style-spec'; + +import type {Map} from '../ui/map'; +import type {Dispatcher} from '../util/dispatcher'; +import type {Evented} from '../util/evented'; + +/** + * Options to add a canvas source type to the map. + */ +export type CanvasSourceSpecification = { + /** + * Source type. Must be `"canvas"`. + */ + type: 'canvas'; + /** + * Four geographical coordinates denoting where to place the corners of the canvas, specified in `[longitude, latitude]` pairs. + */ + coordinates: [[number, number], [number, number], [number, number], [number, number]]; + /** + * Whether the canvas source is animated. If the canvas is static (i.e. pixels do not need to be re-read on every frame), `animate` should be set to `false` to improve performance. + * @defaultValue true + */ + animate?: boolean; + /** + * Canvas source from which to read pixels. Can be a string representing the ID of the canvas element, or the `HTMLCanvasElement` itself. + */ + canvas?: string | HTMLCanvasElement; +}; + +/** + * A data source containing the contents of an HTML canvas. See {@link CanvasSourceSpecification} for detailed documentation of options. + * + * @group Sources + * + * @example + * ```ts + * // add to map + * map.addSource('some id', { + * type: 'canvas', + * canvas: 'idOfMyHTMLCanvas', + * animate: true, + * coordinates: [ + * [-76.54, 39.18], + * [-76.52, 39.18], + * [-76.52, 39.17], + * [-76.54, 39.17] + * ] + * }); + * + * // update + * let mySource = map.getSource('some id'); + * mySource.setCoordinates([ + * [-76.54335737228394, 39.18579907229748], + * [-76.52803659439087, 39.1838364847587], + * [-76.5295386314392, 39.17683392507606], + * [-76.54520273208618, 39.17876344106642] + * ]); + * + * map.removeSource('some id'); // remove + * ``` + */ +export class CanvasSource extends ImageSource { + options: CanvasSourceSpecification; + animate: boolean; + canvas: HTMLCanvasElement; + width: number; + height: number; + /** + * Enables animation. The image will be copied from the canvas to the map on each frame. + */ + play: () => void; + /** + * Disables animation. The map will display a static copy of the canvas image. + */ + pause: () => void; + _playing: boolean; + + /** @internal */ + constructor(id: string, options: CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) { + super(id, options, dispatcher, eventedParent); + + // We build in some validation here, since canvas sources aren't included in the style spec: + if (!options.coordinates) { + this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, 'missing required property "coordinates"'))); + } else if (!Array.isArray(options.coordinates) || options.coordinates.length !== 4 || + options.coordinates.some(c => !Array.isArray(c) || c.length !== 2 || c.some(l => typeof l !== 'number'))) { + this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, '"coordinates" property must be an array of 4 longitude/latitude array pairs'))); + } + + if (options.animate && typeof options.animate !== 'boolean') { + this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, 'optional "animate" property must be a boolean value'))); + } + + if (!options.canvas) { + this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, 'missing required property "canvas"'))); + } else if (typeof options.canvas !== 'string' && !(options.canvas instanceof HTMLCanvasElement)) { + this.fire(new ErrorEvent(new ValidationError(`sources.${id}`, null, '"canvas" must be either a string representing the ID of the canvas element from which to read, or an HTMLCanvasElement instance'))); + } + + this.options = options; + this.animate = options.animate !== undefined ? options.animate : true; + } + + load = () => { + this._loaded = true; + if (!this.canvas) { + this.canvas = (this.options.canvas instanceof HTMLCanvasElement) ? + this.options.canvas : + document.getElementById(this.options.canvas) as HTMLCanvasElement; + // cast to HTMLCanvasElement in else of ternary + // should we do a safety check and throw if it's not actually HTMLCanvasElement? + } + this.width = this.canvas.width; + this.height = this.canvas.height; + + if (this._hasInvalidDimensions()) { + this.fire(new ErrorEvent(new Error('Canvas dimensions cannot be less than or equal to zero.'))); + return; + } + + this.play = function() { + this._playing = true; + this.map.triggerRepaint(); + }; + + this.pause = function() { + if (this._playing) { + this.prepare(); + this._playing = false; + } + }; + + this._finishLoading(); + }; + + /** + * Returns the HTML `canvas` element. + * + * @returns The HTML `canvas` element. + */ + getCanvas(): HTMLCanvasElement { + return this.canvas; + } + + onAdd(map: Map) { + this.map = map; + this.load(); + if (this.canvas) { + if (this.animate) this.play(); + } + } + + onRemove() { + this.pause(); + } + + prepare = () => { + let resize = false; + if (this.canvas.width !== this.width) { + this.width = this.canvas.width; + resize = true; + } + if (this.canvas.height !== this.height) { + this.height = this.canvas.height; + resize = true; + } + + if (this._hasInvalidDimensions()) return; + + if (Object.keys(this.tiles).length === 0) return; // not enough data for current position + + const context = this.map.painter.context; + const gl = context.gl; + + if (!this.boundsBuffer) { + this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members); + } + + if (!this.boundsSegments) { + this.boundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2); + } + + if (!this.texture) { + this.texture = new Texture(context, this.canvas, gl.RGBA, {premultiply: true}); + } else if (resize || this._playing) { + this.texture.update(this.canvas, {premultiply: true}); + } + + let newTilesLoaded = false; + for (const w in this.tiles) { + const tile = this.tiles[w]; + if (tile.state !== 'loaded') { + tile.state = 'loaded'; + tile.texture = this.texture; + newTilesLoaded = true; + } + } + + if (newTilesLoaded) { + this.fire(new Event('data', {dataType: 'source', sourceDataType: 'idle', sourceId: this.id})); + } + }; + + serialize = (): CanvasSourceSpecification => { + return { + type: 'canvas', + coordinates: this.coordinates + }; + }; + + hasTransition() { + return this._playing; + } + + _hasInvalidDimensions() { + for (const x of [this.canvas.width, this.canvas.height]) { + if (isNaN(x) || x <= 0) return true; + } + return false; + } +} diff --git a/web/libraries/maplibre-gl/src/source/geojson_source.test.ts b/web/libraries/maplibre-gl/src/source/geojson_source.test.ts new file mode 100644 index 00000000..a5eeebbf --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/geojson_source.test.ts @@ -0,0 +1,440 @@ +import {Tile} from './tile'; +import {OverscaledTileID} from './tile_id'; +import {GeoJSONSource, GeoJSONSourceOptions} from './geojson_source'; +import {Transform} from '../geo/transform'; +import {LngLat} from '../geo/lng_lat'; +import {extend} from '../util/util'; +import {Dispatcher} from '../util/dispatcher'; +import {RequestManager} from '../util/request_manager'; + +const wrapDispatcher = (dispatcher) => { + return { + getActor() { + return dispatcher; + } + } as Dispatcher; +}; + +const mockDispatcher = wrapDispatcher({ + send() {} +}); + +const hawkHill = { + 'type': 'FeatureCollection', + 'features': [{ + 'type': 'Feature', + 'properties': {}, + 'geometry': { + 'type': 'LineString', + 'coordinates': [ + [-122.48369693756104, 37.83381888486939], + [-122.48348236083984, 37.83317489144141], + [-122.48339653015138, 37.83270036637107], + [-122.48356819152832, 37.832056363179625], + [-122.48404026031496, 37.83114119107971], + [-122.48404026031496, 37.83049717427869], + [-122.48348236083984, 37.829920943955045], + [-122.48356819152832, 37.82954808664175], + [-122.48507022857666, 37.82944639795659], + [-122.48610019683838, 37.82880236636284], + [-122.48695850372314, 37.82931081282506], + [-122.48700141906738, 37.83080223556934], + [-122.48751640319824, 37.83168351665737], + [-122.48803138732912, 37.832158048267786], + [-122.48888969421387, 37.83297152392784], + [-122.48987674713133, 37.83263257682617], + [-122.49043464660643, 37.832937629287755], + [-122.49125003814696, 37.832429207817725], + [-122.49163627624512, 37.832564787218985], + [-122.49223709106445, 37.83337825839438], + [-122.49378204345702, 37.83368330777276] + ] + } + }] +} as GeoJSON.GeoJSON; + +describe('GeoJSONSource#setData', () => { + function createSource(opts?) { + opts = opts || {}; + opts = extend(opts, {data: {}}); + return new GeoJSONSource('id', opts, wrapDispatcher({ + send (type, data, callback) { + if (callback) { + return setTimeout(callback, 0); + } + } + }), undefined); + } + + test('returns self', () => { + const source = createSource(); + expect(source.setData({} as GeoJSON.GeoJSON)).toBe(source); + }); + + test('fires "data" event', async () => { + const source = createSource(); + const loadPromise = source.once('data'); + source.load(); + await loadPromise; + const setDataPromise = source.once('data'); + source.setData({} as GeoJSON.GeoJSON); + await setDataPromise; + }); + + test('fires "dataloading" event', done => { + const source = createSource(); + source.on('dataloading', () => { + done(); + }); + source.load(); + }); + + test('fires "dataabort" event', done => { + const source = new GeoJSONSource('id', {} as any, wrapDispatcher({ + send(type, data, callback) { + setTimeout(() => callback(null, {abandoned: true})); + } + }), undefined); + source.on('dataabort', () => { + done(); + }); + source.load(); + }); + + test('respects collectResourceTiming parameter on source', done => { + const source = createSource({collectResourceTiming: true}); + source.map = { + _requestManager: { + transformRequest: (url) => { return {url}; } + } as any as RequestManager + } as any; + source.actor.send = function(type, params: any, cb) { + if (type === 'geojson.loadData') { + expect(params.request.collectResourceTiming).toBeTruthy(); + setTimeout(cb, 0); + done(); + } + } as any; + source.setData('http://localhost/nonexistent'); + }); + + test('only marks source as loaded when there are no pending loads', async () => { + const source = createSource(); + const setDataPromise = source.once('data'); + source.setData({} as GeoJSON.GeoJSON); + source.setData({} as GeoJSON.GeoJSON); + await setDataPromise; + expect(source.loaded()).toBeFalsy(); + const setDataPromise2 = source.once('data'); + await setDataPromise2; + expect(source.loaded()).toBeTruthy(); + }); + + test('marks source as not loaded before firing "dataloading" event', async () => { + const source = createSource(); + const setDataPromise = source.once('dataloading'); + source.setData({} as GeoJSON.GeoJSON); + await setDataPromise; + expect(source.loaded()).toBeFalsy(); + }); + + test('marks source as loaded before firing "data" event', async () => { + const source = createSource(); + const dataPromise = source.once('data'); + source.setData({} as GeoJSON.GeoJSON); + await dataPromise; + expect(source.loaded()).toBeTruthy(); + }); + + test('marks source as loaded before firing "dataabort" event', done => { + const source = new GeoJSONSource('id', {} as any, wrapDispatcher({ + send(type, data, callback) { + setTimeout(() => callback(null, {abandoned: true})); + } + }), undefined); + source.on('dataabort', () => { + expect(source.loaded()).toBeTruthy(); + done(); + }); + source.setData({} as GeoJSON.GeoJSON); + }); +}); + +describe('GeoJSONSource#onRemove', () => { + test('broadcasts "removeSource" event', done => { + const source = new GeoJSONSource('id', {data: {}} as GeoJSONSourceOptions, wrapDispatcher({ + send(type, data, callback) { + expect(callback).toBeFalsy(); + expect(type).toBe('removeSource'); + expect(data).toEqual({type: 'geojson', source: 'id'}); + done(); + }, + broadcast() { + // Ignore + } + }), undefined); + source.onRemove(); + }); +}); + +describe('GeoJSONSource#update', () => { + const transform = new Transform(); + transform.resize(200, 200); + const lngLat = LngLat.convert([-122.486052, 37.830348]); + const point = transform.locationPoint(lngLat); + transform.zoom = 15; + transform.setLocationAtPoint(lngLat, point); + + test('sends initial loadData request to dispatcher', done => { + const mockDispatcher = wrapDispatcher({ + send(message) { + expect(message).toBe('geojson.loadData'); + done(); + } + }); + + new GeoJSONSource('id', {data: {}} as GeoJSONSourceOptions, mockDispatcher, undefined).load(); + }); + + test('forwards geojson-vt options with worker request', done => { + const mockDispatcher = wrapDispatcher({ + send(message, params) { + expect(message).toBe('geojson.loadData'); + expect(params.geojsonVtOptions).toEqual({ + extent: 8192, + maxZoom: 10, + tolerance: 4, + buffer: 256, + lineMetrics: false, + generateId: true + }); + done(); + } + }); + + new GeoJSONSource('id', { + data: {}, + maxzoom: 10, + tolerance: 0.25, + buffer: 16, + generateId: true + } as GeoJSONSourceOptions, mockDispatcher, undefined).load(); + }); + + test('forwards Supercluster options with worker request', done => { + const mockDispatcher = wrapDispatcher({ + send(message, params) { + expect(message).toBe('geojson.loadData'); + expect(params.superclusterOptions).toEqual({ + maxZoom: 12, + minPoints: 3, + extent: 8192, + radius: 1600, + log: false, + generateId: true + }); + done(); + } + }); + + new GeoJSONSource('id', { + data: {}, + cluster: true, + clusterMaxZoom: 12, + clusterRadius: 100, + clusterMinPoints: 3, + generateId: true + } as GeoJSONSourceOptions, mockDispatcher, undefined).load(); + }); + + test('modifying cluster properties after adding a source', done => { + // test setCluster function on GeoJSONSource + const mockDispatcher = wrapDispatcher({ + send(message, params) { + expect(message).toBe('geojson.loadData'); + expect(params.cluster).toBe(true); + expect(params.superclusterOptions.radius).toBe(80); + expect(params.superclusterOptions.maxZoom).toBe(16); + done(); + } + }); + new GeoJSONSource('id', { + data: {}, + cluster: false, + clusterMaxZoom: 8, + clusterRadius: 100, + clusterMinPoints: 3, + generateId: true + } as GeoJSONSourceOptions, mockDispatcher, undefined).setClusterOptions({cluster: true, clusterRadius: 80, clusterMaxZoom: 16}); + }); + + test('forwards Supercluster options with worker request, ignore max zoom of source', done => { + const mockDispatcher = wrapDispatcher({ + send(message, params) { + expect(message).toBe('geojson.loadData'); + expect(params.superclusterOptions).toEqual({ + maxZoom: 12, + minPoints: 3, + extent: 8192, + radius: 1600, + log: false, + generateId: true + }); + done(); + } + }); + + new GeoJSONSource('id', { + data: {}, + maxzoom: 10, + cluster: true, + clusterMaxZoom: 12, + clusterRadius: 100, + clusterMinPoints: 3, + generateId: true + } as GeoJSONSourceOptions, mockDispatcher, undefined).load(); + }); + + test('transforms url before making request', () => { + const mapStub = { + _requestManager: { + transformRequest: (url) => { return {url}; } + } + } as any; + const transformSpy = jest.spyOn(mapStub._requestManager, 'transformRequest'); + const source = new GeoJSONSource('id', {data: 'https://example.com/data.geojson'} as GeoJSONSourceOptions, mockDispatcher, undefined); + source.onAdd(mapStub); + expect(transformSpy).toHaveBeenCalledTimes(1); + expect(transformSpy.mock.calls[0][0]).toBe('https://example.com/data.geojson'); + }); + test('fires event when metadata loads', done => { + const mockDispatcher = wrapDispatcher({ + send(message, args, callback) { + if (callback) { + setTimeout(callback, 0); + } + } + }); + + const source = new GeoJSONSource('id', {data: {}} as GeoJSONSourceOptions, mockDispatcher, undefined); + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') done(); + }); + + source.load(); + }); + + test('fires metadata data event even when initial request is aborted', done => { + let requestCount = 0; + const mockDispatcher = wrapDispatcher({ + send(message, args, callback) { + setTimeout(() => callback(null, {abandoned: requestCount++ === 0})); + } + }); + + const source = new GeoJSONSource('id', {data: {}} as GeoJSONSourceOptions, mockDispatcher, undefined); + + source.on('data', e => { + if (e.sourceDataType === 'metadata') done(); + }); + + source.load(); + source.setData({} as GeoJSON.GeoJSON); + }); + + test('fires "error"', done => { + const mockDispatcher = wrapDispatcher({ + send(message, args, callback) { + if (callback) { + setTimeout(callback.bind(null, 'error'), 0); + } + } + }); + + const source = new GeoJSONSource('id', {data: {}} as GeoJSONSourceOptions, mockDispatcher, undefined); + + source.on('error', (err) => { + expect(err.error).toBe('error'); + done(); + }); + + source.load(); + }); + + test('sends loadData request to dispatcher after data update', done => { + let expectedLoadDataCalls = 2; + const mockDispatcher = wrapDispatcher({ + send(message, args, callback) { + if (message === 'geojson.loadData' && --expectedLoadDataCalls <= 0) { + done(); + } + if (callback) { + setTimeout(callback, 0); + } + } + }); + + const source = new GeoJSONSource('id', {data: {}} as GeoJSONSourceOptions, mockDispatcher, undefined); + source.map = { + transform: {} as Transform, + getPixelRatio() { return 1; } + } as any; + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + source.setData({} as GeoJSON.GeoJSON); + source.loadTile(new Tile(new OverscaledTileID(0, 0, 0, 0, 0), 512), () => {}); + } + }); + + source.load(); + }); +}); + +describe('GeoJSONSource#serialize', () => { + const mapStub = { + _requestManager: { + transformRequest: (url) => { return {url}; } + } + } as any; + test('serialize source with inline data', () => { + const source = new GeoJSONSource('id', {data: hawkHill} as GeoJSONSourceOptions, mockDispatcher, undefined); + source.map = mapStub; + source.load(); + expect(source.serialize()).toEqual({ + type: 'geojson', + data: hawkHill + }); + }); + + test('serialize source with url', () => { + const source = new GeoJSONSource('id', {data: 'local://data.json'} as GeoJSONSourceOptions, mockDispatcher, undefined); + source.map = mapStub; + source.load(); + expect(source.serialize()).toEqual({ + type: 'geojson', + data: 'local://data.json' + }); + }); + + test('serialize source with updated data', () => { + const source = new GeoJSONSource('id', {data: {}} as GeoJSONSourceOptions, mockDispatcher, undefined); + source.map = mapStub; + source.load(); + source.setData(hawkHill); + expect(source.serialize()).toEqual({ + type: 'geojson', + data: hawkHill + }); + }); + + test('serialize source with additional options', () => { + const source = new GeoJSONSource('id', {data: {}, cluster: true} as GeoJSONSourceOptions, mockDispatcher, undefined); + expect(source.serialize()).toEqual({ + type: 'geojson', + data: {}, + cluster: true + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/geojson_source.ts b/web/libraries/maplibre-gl/src/source/geojson_source.ts new file mode 100644 index 00000000..ada698d3 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/geojson_source.ts @@ -0,0 +1,448 @@ +import {Event, ErrorEvent, Evented} from '../util/evented'; + +import {extend} from '../util/util'; +import {EXTENT} from '../data/extent'; +import {ResourceType} from '../util/request_manager'; +import {browser} from '../util/browser'; + +import type {Source} from './source'; +import type {Map} from '../ui/map'; +import type {Dispatcher} from '../util/dispatcher'; +import type {Tile} from './tile'; +import type {Actor} from '../util/actor'; +import type {Callback} from '../types/callback'; +import type {GeoJSONSourceSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {GeoJSONSourceDiff} from './geojson_source_diff'; +import type {Options, ClusterProperties} from 'supercluster'; + +export type GeoJSONSourceOptions = GeoJSONSourceSpecification & { + workerOptions?: WorkerOptions; + collectResourceTiming?: boolean; +} + +export type GeoJsonSourceOptions = { + data?: GeoJSON.GeoJSON | string | undefined; + cluster?: boolean; + clusterMaxZoom?: number; + clusterRadius?: number; + clusterMinPoints?: number; + generateId?: boolean; +} +export type WorkerOptions = { + source?: string; + cluster?: boolean; + geojsonVtOptions?: { + buffer?: number; + tolerance?: number; + extent?: number; + maxZoom?: number; + linemetrics?: boolean; + generateId?: boolean; + }; + superclusterOptions?: Options; + clusterProperties?: ClusterProperties; + fliter?: any; + promoteId?: any; + collectResourceTiming?: boolean; +} + +/** + * The cluster options to set + */ +export type SetClusterOptions = { + /** + * Whether or not to cluster + */ + cluster?: boolean; + /** + * The cluster's max zoom + */ + clusterMaxZoom?: number; + /** + * The cluster's radius + */ + clusterRadius?: number; +} + +/** + * A source containing GeoJSON. + * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-geojson) for detailed documentation of options.) + * + * @group Sources + * + * @example + * ```ts + * map.addSource('some id', { + * type: 'geojson', + * data: 'https://d2ad6b4ur7yvpq.cloudfront.net/naturalearth-3.3.0/ne_10m_ports.geojson' + * }); + * ``` + * + * @example + * ```ts + * map.addSource('some id', { + * type: 'geojson', + * data: { + * "type": "FeatureCollection", + * "features": [{ + * "type": "Feature", + * "properties": {}, + * "geometry": { + * "type": "Point", + * "coordinates": [ + * -76.53063297271729, + * 39.18174077994108 + * ] + * } + * }] + * } + * }); + * ``` + * + * @example + * ```ts + * map.getSource('some id').setData({ + * "type": "FeatureCollection", + * "features": [{ + * "type": "Feature", + * "properties": { "name": "Null Island" }, + * "geometry": { + * "type": "Point", + * "coordinates": [ 0, 0 ] + * } + * }] + * }); + * ``` + * @see [Draw GeoJSON points](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-markers/) + * @see [Add a GeoJSON line](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-line/) + * @see [Create a heatmap from points](https://maplibre.org/maplibre-gl-js/docs/examples/heatmap/) + * @see [Create and style clusters](https://maplibre.org/maplibre-gl-js/docs/examples/cluster/) + */ +export class GeoJSONSource extends Evented implements Source { + type: 'geojson'; + id: string; + minzoom: number; + maxzoom: number; + tileSize: number; + attribution: string; + promoteId: PromoteIdSpecification; + + isTileClipped: boolean; + reparseOverscaled: boolean; + _data: GeoJSON.GeoJSON | string | undefined; + _options: GeoJsonSourceOptions; + workerOptions: WorkerOptions; + map: Map; + actor: Actor; + _pendingLoads: number; + _collectResourceTiming: boolean; + _removed: boolean; + + /** @internal */ + constructor(id: string, options: GeoJSONSourceOptions, dispatcher: Dispatcher, eventedParent: Evented) { + super(); + + this.id = id; + + // `type` is a property rather than a constant to make it easy for 3rd + // parties to use GeoJSONSource to build their own source types. + this.type = 'geojson'; + + this.minzoom = 0; + this.maxzoom = 18; + this.tileSize = 512; + this.isTileClipped = true; + this.reparseOverscaled = true; + this._removed = false; + this._pendingLoads = 0; + + this.actor = dispatcher.getActor(); + this.setEventedParent(eventedParent); + + this._data = (options.data as any); + this._options = extend({}, options); + + this._collectResourceTiming = options.collectResourceTiming; + + if (options.maxzoom !== undefined) this.maxzoom = options.maxzoom; + if (options.type) this.type = options.type; + if (options.attribution) this.attribution = options.attribution; + this.promoteId = options.promoteId; + + const scale = EXTENT / this.tileSize; + + // sent to the worker, along with `url: ...` or `data: literal geojson`, + // so that it can load/parse/index the geojson data + // extending with `options.workerOptions` helps to make it easy for + // third-party sources to hack/reuse GeoJSONSource. + this.workerOptions = extend({ + source: this.id, + cluster: options.cluster || false, + geojsonVtOptions: { + buffer: (options.buffer !== undefined ? options.buffer : 128) * scale, + tolerance: (options.tolerance !== undefined ? options.tolerance : 0.375) * scale, + extent: EXTENT, + maxZoom: this.maxzoom, + lineMetrics: options.lineMetrics || false, + generateId: options.generateId || false + }, + superclusterOptions: { + maxZoom: options.clusterMaxZoom !== undefined ? options.clusterMaxZoom : this.maxzoom - 1, + minPoints: Math.max(2, options.clusterMinPoints || 2), + extent: EXTENT, + radius: (options.clusterRadius || 50) * scale, + log: false, + generateId: options.generateId || false + }, + clusterProperties: options.clusterProperties, + filter: options.filter + }, options.workerOptions); + + // send the promoteId to the worker to have more flexible updates, but only if it is a string + if (typeof this.promoteId === 'string') { + this.workerOptions.promoteId = this.promoteId; + } + } + + load = () => { + this._updateWorkerData(); + }; + + onAdd(map: Map) { + this.map = map; + this.load(); + } + + /** + * Sets the GeoJSON data and re-renders the map. + * + * @param data - A GeoJSON data object or a URL to one. The latter is preferable in the case of large GeoJSON files. + * @returns `this` + */ + setData(data: GeoJSON.GeoJSON | string): this { + this._data = data; + this._updateWorkerData(); + + return this; + } + + /** + * Updates the source's GeoJSON, and re-renders the map. + * + * For sources with lots of features, this method can be used to make updates more quickly. + * + * This approach requires unique IDs for every feature in the source. The IDs can either be specified on the feature, + * or by using the promoteId option to specify which property should be used as the ID. + * + * It is an error to call updateData on a source that did not have unique IDs for each of its features already. + * + * Updates are applied on a best-effort basis, updating an ID that does not exist will not result in an error. + * + * @param diff - The changes that need to be applied. + * @returns `this` + */ + updateData(diff: GeoJSONSourceDiff): this { + this._updateWorkerData(diff); + + return this; + } + + /** + * To disable/enable clustering on the source options + * @param options - The options to set + * @returns `this` + * @example + * ```ts + * map.getSource('some id').setClusterOptions({cluster: false}); + * map.getSource('some id').setClusterOptions({cluster: false, clusterRadius: 50, clusterMaxZoom: 14}); + * ``` + */ + setClusterOptions(options: SetClusterOptions): this { + this.workerOptions.cluster = options.cluster; + if (options) { + if (options.clusterRadius !== undefined) this.workerOptions.superclusterOptions.radius = options.clusterRadius; + if (options.clusterMaxZoom !== undefined) this.workerOptions.superclusterOptions.maxZoom = options.clusterMaxZoom; + } + this._updateWorkerData(); + return this; + } + + /** + * For clustered sources, fetches the zoom at which the given cluster expands. + * + * @param clusterId - The value of the cluster's `cluster_id` property. + * @param callback - A callback to be called when the zoom value is retrieved (`(error, zoom) => { ... }`). + * @returns `this` + */ + getClusterExpansionZoom(clusterId: number, callback: Callback): this { + this.actor.send('geojson.getClusterExpansionZoom', {clusterId, source: this.id}, callback); + return this; + } + + /** + * For clustered sources, fetches the children of the given cluster on the next zoom level (as an array of GeoJSON features). + * + * @param clusterId - The value of the cluster's `cluster_id` property. + * @param callback - A callback to be called when the features are retrieved (`(error, features) => { ... }`). + * @returns `this` + */ + getClusterChildren(clusterId: number, callback: Callback>): this { + this.actor.send('geojson.getClusterChildren', {clusterId, source: this.id}, callback); + return this; + } + + /** + * For clustered sources, fetches the original points that belong to the cluster (as an array of GeoJSON features). + * + * @param clusterId - The value of the cluster's `cluster_id` property. + * @param limit - The maximum number of features to return. + * @param offset - The number of features to skip (e.g. for pagination). + * @param callback - A callback to be called when the features are retrieved (`(error, features) => { ... }`). + * @returns `this` + * @example + * Retrieve cluster leaves on click + * ```ts + * map.on('click', 'clusters', function(e) { + * let features = map.queryRenderedFeatures(e.point, { + * layers: ['clusters'] + * }); + * + * let clusterId = features[0].properties.cluster_id; + * let pointCount = features[0].properties.point_count; + * let clusterSource = map.getSource('clusters'); + * + * clusterSource.getClusterLeaves(clusterId, pointCount, 0, function(error, features) { + * // Print cluster leaves in the console + * console.log('Cluster leaves:', error, features); + * }) + * }); + * ``` + */ + getClusterLeaves(clusterId: number, limit: number, offset: number, callback: Callback>): this { + this.actor.send('geojson.getClusterLeaves', { + source: this.id, + clusterId, + limit, + offset + }, callback); + return this; + } + + /** + * Responsible for invoking WorkerSource's geojson.loadData target, which + * handles loading the geojson data and preparing to serve it up as tiles, + * using geojson-vt or supercluster as appropriate. + * @param diff - the diff object + */ + _updateWorkerData(diff?: GeoJSONSourceDiff) { + const options = extend({}, this.workerOptions); + if (diff) { + options.dataDiff = diff; + } else if (typeof this._data === 'string') { + options.request = this.map._requestManager.transformRequest(browser.resolveURL(this._data as string), ResourceType.Source); + options.request.collectResourceTiming = this._collectResourceTiming; + } else { + options.data = JSON.stringify(this._data); + } + + this._pendingLoads++; + this.fire(new Event('dataloading', {dataType: 'source'})); + + // target {this.type}.loadData rather than literally geojson.loadData, + // so that other geojson-like source types can easily reuse this + // implementation + this.actor.send(`${this.type}.loadData`, options, (err, result) => { + this._pendingLoads--; + + if (this._removed || (result && result.abandoned)) { + this.fire(new Event('dataabort', {dataType: 'source'})); + return; + } + + let resourceTiming = null; + if (result && result.resourceTiming && result.resourceTiming[this.id]) + resourceTiming = result.resourceTiming[this.id].slice(0); + + if (err) { + this.fire(new ErrorEvent(err)); + return; + } + + const data: any = {dataType: 'source'}; + if (this._collectResourceTiming && resourceTiming && resourceTiming.length > 0) + extend(data, {resourceTiming}); + + // although GeoJSON sources contain no metadata, we fire this event to let the SourceCache + // know its ok to start requesting tiles. + this.fire(new Event('data', {...data, sourceDataType: 'metadata'})); + this.fire(new Event('data', {...data, sourceDataType: 'content'})); + }); + } + + loaded(): boolean { + return this._pendingLoads === 0; + } + + loadTile(tile: Tile, callback: Callback) { + const message = !tile.actor ? 'loadTile' : 'reloadTile'; + tile.actor = this.actor; + const params = { + type: this.type, + uid: tile.uid, + tileID: tile.tileID, + zoom: tile.tileID.overscaledZ, + maxZoom: this.maxzoom, + tileSize: this.tileSize, + source: this.id, + pixelRatio: this.map.getPixelRatio(), + showCollisionBoxes: this.map.showCollisionBoxes, + promoteId: this.promoteId + }; + + tile.request = this.actor.send(message, params, (err, data) => { + delete tile.request; + tile.unloadVectorData(); + + if (tile.aborted) { + return callback(null); + } + + if (err) { + return callback(err); + } + + tile.loadVectorData(data, this.map.painter, message === 'reloadTile'); + + return callback(null); + }); + } + + abortTile(tile: Tile) { + if (tile.request) { + tile.request.cancel(); + delete tile.request; + } + tile.aborted = true; + } + + unloadTile(tile: Tile) { + tile.unloadVectorData(); + this.actor.send('removeTile', {uid: tile.uid, type: this.type, source: this.id}); + } + + onRemove() { + this._removed = true; + this.actor.send('removeSource', {type: this.type, source: this.id}); + } + + serialize = (): GeoJSONSourceSpecification => { + return extend({}, this._options, { + type: this.type, + data: this._data + }); + }; + + hasTransition() { + return false; + } +} diff --git a/web/libraries/maplibre-gl/src/source/geojson_source_diff.test.ts b/web/libraries/maplibre-gl/src/source/geojson_source_diff.test.ts new file mode 100644 index 00000000..0bb6ab79 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/geojson_source_diff.test.ts @@ -0,0 +1,385 @@ +import {setPerformance} from '../util/test/util'; +import {type GeoJSONFeatureId, isUpdateableGeoJSON, toUpdateable, applySourceDiff} from './geojson_source_diff'; + +beforeEach(() => { + setPerformance(); +}); + +describe('isUpdateableGeoJSON', () => { + test('feature without id is not updateable', done => { + // no feature id -> false + expect(isUpdateableGeoJSON({ + type: 'Feature', + geometry: { + type: 'Point', + coordinates: [0, 0] + }, + properties: {}, + })).toBe(false); + done(); + }); + + test('feature with id is updateable', done => { + // has a feature id -> true + expect(isUpdateableGeoJSON({ + type: 'Feature', + id: 'feature_id', + geometry: { + type: 'Point', + coordinates: [0, 0] + }, + properties: {}, + })).toBe(true); + done(); + }); + + test('promoteId missing is not updateable', done => { + expect(isUpdateableGeoJSON({ + type: 'Feature', + id: 'feature_id', + geometry: { + type: 'Point', + coordinates: [0, 0] + }, + properties: {}, + }, 'propId')).toBe(false); + done(); + }); + + test('promoteId present is updateable', done => { + expect(isUpdateableGeoJSON({ + type: 'Feature', + geometry: { + type: 'Point', + coordinates: [0, 0] + }, + properties: { + propId: 'feature_id', + }, + }, 'propId')).toBe(true); + done(); + }); + + test('feature collection with unique ids is updateable', done => { + expect(isUpdateableGeoJSON({ + type: 'FeatureCollection', + features: [{ + type: 'Feature', + id: 'feature_id', + geometry: { + type: 'Point', + coordinates: [0, 0] + }, + properties: {}, + }, { + type: 'Feature', + id: 'feature_id_2', + geometry: { + type: 'Point', + coordinates: [0, 0] + }, + properties: {}, + }] + })).toBe(true); + done(); + }); + + test('feature collection with unique promoteIds is updateable', done => { + expect(isUpdateableGeoJSON({ + type: 'FeatureCollection', + features: [{ + type: 'Feature', + geometry: { + type: 'Point', + coordinates: [0, 0] + }, + properties: { + propId: 'feature_id', + }, + }, { + type: 'Feature', + geometry: { + type: 'Point', + coordinates: [0, 0] + }, + properties: { + propId: 'feature_id_2', + }, + }] + }, 'propId')).toBe(true); + done(); + }); + + test('feature collection without unique ids is not updateable', done => { + expect(isUpdateableGeoJSON({ + type: 'FeatureCollection', + features: [{ + type: 'Feature', + geometry: { + type: 'Point', + coordinates: [0, 0] + }, + properties: {}, + }] + })).toBe(false); + done(); + }); + + test('feature collection with duplicate feature ids is not updateable', done => { + expect(isUpdateableGeoJSON({ + type: 'FeatureCollection', + features: [{ + type: 'Feature', + id: 'feature_id', + geometry: { + type: 'Point', + coordinates: [0, 0] + }, + properties: {}, + }, { + type: 'Feature', + id: 'feature_id', + geometry: { + type: 'Point', + coordinates: [0, 0] + }, + properties: {}, + }] + })).toBe(false); + done(); + }); + + test('geometries are not updateable', done => { + expect(isUpdateableGeoJSON({type: 'Point', coordinates: [0, 0]})).toBe(false); + done(); + }); +}); + +describe('toUpdateable', () => { + test('works with a single feature - feature id', done => { + const updateable = toUpdateable({ + type: 'Feature', + id: 'point', + geometry: { + type: 'Point', + coordinates: [0, 0], + }, properties: {}}); + expect(updateable.size).toBe(1); + expect(updateable.has('point')).toBeTruthy(); + done(); + }); + + test('works with a single feature - promoteId', done => { + const updateable2 = toUpdateable({ + type: 'Feature', + geometry: { + type: 'Point', + coordinates: [0, 0], + }, properties: { + promoteId: 'point', + }}, 'promoteId'); + expect(updateable2.size).toBe(1); + expect(updateable2.has('point')).toBeTruthy(); + done(); + }); + + test('works with a FeatureCollection - feature id', done => { + const updateable = toUpdateable({ + type: 'FeatureCollection', + features: [ + { + type: 'Feature', + id: 'point', + geometry: { + type: 'Point', + coordinates: [0, 0], + }, properties: {}}, + { + type: 'Feature', + id: 'point2', + geometry: { + type: 'Point', + coordinates: [0, 0], + }, properties: {}} + ] + }); + expect(updateable.size).toBe(2); + expect(updateable.has('point')).toBeTruthy(); + expect(updateable.has('point2')).toBeTruthy(); + done(); + }); + + test('works with a FeatureCollection - promoteId', done => { + const updateable2 = toUpdateable({ + type: 'FeatureCollection', + features: [ + { + type: 'Feature', + geometry: { + type: 'Point', + coordinates: [0, 0], + }, properties: { + promoteId: 'point' + }}, + { + type: 'Feature', + geometry: { + type: 'Point', + coordinates: [0, 0], + }, properties: { + promoteId: 'point2' + }} + ] + }, 'promoteId'); + expect(updateable2.size).toBe(2); + expect(updateable2.has('point')).toBeTruthy(); + expect(updateable2.has('point2')).toBeTruthy(); + done(); + }); +}); + +describe('applySourceDiff', () => { + const point: GeoJSON.Feature = { + type: 'Feature', + id: 'point', + geometry: { + type: 'Point', + coordinates: [0, 0] + }, + properties: {}, + }; + + const point2: GeoJSON.Feature = { + type: 'Feature', + geometry: { + type: 'Point', + coordinates: [0, 0], + }, + properties: { + promoteId: 'point2' + }, + }; + + // freeze our input data to guarantee that applySourceDiff works immutably + Object.freeze(point); + Object.freeze(point.geometry); + Object.freeze((point.geometry as GeoJSON.Point).coordinates); + Object.freeze(point.properties); + Object.freeze(point2); + Object.freeze(point2.geometry); + Object.freeze((point2.geometry as GeoJSON.Point).coordinates); + Object.freeze(point2.properties); + + test('adds a feature using the feature id', done => { + const updateable = new Map(); + + applySourceDiff(updateable, { + add: [point] + }); + expect(updateable.size).toBe(1); + expect(updateable.has('point')).toBeTruthy(); + done(); + }); + + test('adds a feature using the promoteId', done => { + const updateable = new Map(); + applySourceDiff(updateable, { + add: [point2] + }, 'promoteId'); + expect(updateable.size).toBe(1); + expect(updateable.has('point2')).toBeTruthy(); + done(); + }); + + test('removes a feature by its id', done => { + const updateable = new Map([['point', point], ['point2', point2]]); + applySourceDiff(updateable, { + remove: ['point2'], + }); + expect(updateable.size).toBe(1); + expect(updateable.has('point2')).toBeFalsy(); + done(); + }); + + test('updates a feature geometry', done => { + const updateable = new Map([['point', point]]); + // update -> new geometry + applySourceDiff(updateable, { + update: [{ + id: 'point', + newGeometry: { + type: 'Point', + coordinates: [1, 0] + } + }] + }); + expect(updateable.size).toBe(1); + expect((updateable.get('point')?.geometry as GeoJSON.Point).coordinates[0]).toBe(1); + done(); + }); + + test('adds properties', done => { + const updateable = new Map([['point', point]]); + applySourceDiff(updateable, { + update: [{ + id: 'point', + addOrUpdateProperties: [ + {key: 'prop', value: 'value'}, + {key: 'prop2', value: 'value2'} + ] + }] + }); + expect(updateable.size).toBe(1); + const properties = updateable.get('point')?.properties!; + expect(Object.keys(properties)).toHaveLength(2); + expect(properties.prop).toBe('value'); + expect(properties.prop2).toBe('value2'); + done(); + }); + + test('updates properties', done => { + const updateable = new Map([['point', {...point, properties: {prop: 'value', prop2: 'value2'}}]]); + applySourceDiff(updateable, { + update: [{ + id: 'point', + addOrUpdateProperties: [ + {key: 'prop2', value: 'value3'} + ] + }] + }); + expect(updateable.size).toBe(1); + const properties2 = updateable.get('point')?.properties!; + expect(Object.keys(properties2)).toHaveLength(2); + expect(properties2.prop).toBe('value'); + expect(properties2.prop2).toBe('value3'); + done(); + }); + + test('removes properties', done => { + const updateable = new Map([['point', {...point, properties: {prop: 'value', prop2: 'value2'}}]]); + applySourceDiff(updateable, { + update: [{ + id: 'point', + removeProperties: ['prop2'] + }] + }); + expect(updateable.size).toBe(1); + const properties3 = updateable.get('point')?.properties!; + expect(Object.keys(properties3)).toHaveLength(1); + expect(properties3.prop).toBe('value'); + done(); + }); + + test('removes all properties', done => { + const updateable = new Map([['point', {...point, properties: {prop: 'value', prop2: 'value2'}}]]); + applySourceDiff(updateable, { + update: [{ + id: 'point', + removeAllProperties: true, + }] + }); + expect(updateable.size).toBe(1); + expect(Object.keys(updateable.get('point')?.properties!)).toHaveLength(0); + done(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/geojson_source_diff.ts b/web/libraries/maplibre-gl/src/source/geojson_source_diff.ts new file mode 100644 index 00000000..40468e2e --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/geojson_source_diff.ts @@ -0,0 +1,172 @@ +/** + * A way to indentify a feature, either by string or by number + */ +export type GeoJSONFeatureId = number | string; + +/** + * The geojson source diff object + */ +export type GeoJSONSourceDiff = { + /** + * When set to `true` it will remove all features + */ + removeAll?: boolean; + /** + * An array of features IDs to remove + */ + remove?: Array; + /** + * An array of features to add + */ + add?: Array; + /** + * An array of update objects + */ + update?: Array; +} + +/** + * A geojson feature diff object + */ +export type GeoJSONFeatureDiff = { + /** + * The feature ID + */ + id: GeoJSONFeatureId; + /** + * If it's a new geometry, place it here + */ + newGeometry?: GeoJSON.Geometry; + /** + * Setting to `true` will remove all preperties + */ + removeAllProperties?: boolean; + /** + * The properties keys to remove + */ + removeProperties?: Array; + /** + * The properties to add or update along side their values + */ + addOrUpdateProperties?: Array<{key: string; value: any}>; +} + +export type UpdateableGeoJSON = GeoJSON.Feature | GeoJSON.FeatureCollection | undefined; + +function getFeatureId(feature: GeoJSON.Feature, promoteId?: string): GeoJSONFeatureId | undefined { + return promoteId ? feature.properties[promoteId] : feature.id; +} + +export function isUpdateableGeoJSON(data: GeoJSON.GeoJSON | undefined, promoteId?: string): data is UpdateableGeoJSON { + // null can be updated + if (data == null) { + return true; + } + + // a single feature with an id can be updated, need to explicitly check against null because 0 is a valid feature id that is falsy + if (data.type === 'Feature') { + return getFeatureId(data, promoteId) != null; + } + + // a feature collection can be updated if every feature has an id, and the ids are all unique + // this prevents us from silently dropping features if ids get reused + if (data.type === 'FeatureCollection') { + const seenIds = new Set(); + for (const feature of data.features) { + const id = getFeatureId(feature, promoteId); + if (id == null) { + return false; + } + + if (seenIds.has(id)) { + return false; + } + + seenIds.add(id); + } + + return true; + } + + return false; +} + +export function toUpdateable(data: UpdateableGeoJSON, promoteId?: string) { + const result = new Map(); + if (data == null) { + // empty result + } else if (data.type === 'Feature') { + result.set(getFeatureId(data, promoteId)!, data); + } else { + for (const feature of data.features) { + result.set(getFeatureId(feature, promoteId)!, feature); + } + } + + return result; +} + +// mutates updateable +export function applySourceDiff(updateable: Map, diff: GeoJSONSourceDiff, promoteId?: string): void { + if (diff.removeAll) { + updateable.clear(); + } + + if (diff.remove) { + for (const id of diff.remove) { + updateable.delete(id); + } + } + + if (diff.add) { + for (const feature of diff.add) { + const id = getFeatureId(feature, promoteId); + + if (id != null) { + updateable.set(id, feature); + } + } + } + + if (diff.update) { + for (const update of diff.update) { + let feature = updateable.get(update.id); + + if (feature == null) { + continue; + } + + // be careful to clone the feature and/or properties objects to avoid mutating our input + const cloneFeature = update.newGeometry || update.removeAllProperties; + // note: removeAllProperties gives us a new properties object, so we can skip the clone step + const cloneProperties = !update.removeAllProperties && (update.removeProperties?.length > 0 || update.addOrUpdateProperties?.length > 0); + if (cloneFeature || cloneProperties) { + feature = {...feature}; + updateable.set(update.id, feature); + if (cloneProperties) { + feature.properties = {...feature.properties}; + } + } + + if (update.newGeometry) { + feature.geometry = update.newGeometry; + } + + if (update.removeAllProperties) { + feature.properties = {}; + } else if (update.removeProperties?.length > 0) { + for (const prop of update.removeProperties) { + if (Object.prototype.hasOwnProperty.call(feature.properties, prop)) { + delete feature.properties[prop]; + } + } + } + + if (update.addOrUpdateProperties?.length > 0) { + for (const {key, value} of update.addOrUpdateProperties) { + feature.properties[key] = value; + } + } + } + } +} diff --git a/web/libraries/maplibre-gl/src/source/geojson_worker_source.test.ts b/web/libraries/maplibre-gl/src/source/geojson_worker_source.test.ts new file mode 100644 index 00000000..c4e16439 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/geojson_worker_source.test.ts @@ -0,0 +1,369 @@ +import {GeoJSONWorkerSource, LoadGeoJSONParameters} from './geojson_worker_source'; +import {StyleLayerIndex} from '../style/style_layer_index'; +import {OverscaledTileID} from './tile_id'; +import perf from '../util/performance'; +import {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {Actor} from '../util/actor'; +import {WorkerTileParameters} from './worker_source'; +import {setPerformance} from '../util/test/util'; +import {type FakeServer, fakeServer} from 'nise'; + +const actor = {send: () => {}} as any as Actor; + +beforeEach(() => { + setPerformance(); +}); + +describe('reloadTile', () => { + test('does not rebuild vector data unless data has changed', done => { + const layers = [ + { + id: 'mylayer', + source: 'sourceId', + type: 'symbol', + } + ] as LayerSpecification[]; + const layerIndex = new StyleLayerIndex(layers); + const source = new GeoJSONWorkerSource(actor, layerIndex, []); + const originalLoadVectorData = source.loadVectorData; + let loadVectorCallCount = 0; + source.loadVectorData = function(params, callback) { + loadVectorCallCount++; + return originalLoadVectorData.call(this, params, callback); + }; + const geoJson = { + 'type': 'Feature', + 'geometry': { + 'type': 'Point', + 'coordinates': [0, 0] + } + }; + const tileParams = { + source: 'sourceId', + uid: 0, + tileID: new OverscaledTileID(0, 0, 0, 0, 0), + maxZoom: 10 + }; + + function addData(callback) { + source.loadData({source: 'sourceId', data: JSON.stringify(geoJson)} as LoadGeoJSONParameters, (err) => { + expect(err).toBeNull(); + callback(); + }); + } + + function reloadTile(callback) { + source.reloadTile(tileParams as any as WorkerTileParameters, (err, data) => { + expect(err).toBeNull(); + return callback(data); + }); + } + + addData(() => { + // first call should load vector data from geojson + let firstData; + reloadTile(data => { + firstData = data; + }); + expect(loadVectorCallCount).toBe(1); + + // second call won't give us new rawTileData + reloadTile(data => { + expect('rawTileData' in data).toBeFalsy(); + data.rawTileData = firstData.rawTileData; + expect(data).toEqual(firstData); + }); + + // also shouldn't call loadVectorData again + expect(loadVectorCallCount).toBe(1); + + // replace geojson data + addData(() => { + // should call loadVectorData again after changing geojson data + reloadTile(data => { + expect('rawTileData' in data).toBeTruthy(); + expect(data).toEqual(firstData); + }); + expect(loadVectorCallCount).toBe(2); + done(); + }); + }); + }); + +}); + +describe('resourceTiming', () => { + + const layers = [ + { + id: 'mylayer', + source: 'sourceId', + type: 'symbol', + } + ] as LayerSpecification[]; + const geoJson = { + 'type': 'Feature', + 'geometry': { + 'type': 'Point', + 'coordinates': [0, 0] + } + } as GeoJSON.GeoJSON; + + test('loadData - url', done => { + const exampleResourceTiming = { + connectEnd: 473, + connectStart: 473, + decodedBodySize: 86494, + domainLookupEnd: 473, + domainLookupStart: 473, + duration: 341, + encodedBodySize: 52528, + entryType: 'resource', + fetchStart: 473.5, + initiatorType: 'xmlhttprequest', + name: 'http://localhost:2900/fake.geojson', + nextHopProtocol: 'http/1.1', + redirectEnd: 0, + redirectStart: 0, + requestStart: 477, + responseEnd: 815, + responseStart: 672, + secureConnectionStart: 0 + } as any as PerformanceEntry; + + window.performance.getEntriesByName = jest.fn().mockReturnValue([exampleResourceTiming]); + + const layerIndex = new StyleLayerIndex(layers); + const source = new GeoJSONWorkerSource(actor, layerIndex, [], (params, callback) => { + callback(null, geoJson); + return {cancel: () => {}}; + }); + + source.loadData({source: 'testSource', request: {url: 'http://localhost/nonexistent', collectResourceTiming: true}} as LoadGeoJSONParameters, (err, result) => { + expect(err).toBeNull(); + expect(result.resourceTiming.testSource).toEqual([exampleResourceTiming]); + done(); + }); + }); + + test('loadData - url (resourceTiming fallback method)', done => { + const sampleMarks = [100, 350]; + const marks = {}; + const measures = {}; + window.performance.getEntriesByName = jest.fn().mockImplementation((name) => { return measures[name] || []; }); + jest.spyOn(perf, 'mark').mockImplementation((name) => { + marks[name] = sampleMarks.shift(); + return null; + }); + window.performance.measure = jest.fn().mockImplementation((name, start, end) => { + measures[name] = measures[name] || []; + measures[name].push({ + duration: marks[end] - marks[start], + entryType: 'measure', + name, + startTime: marks[start] + }); + return null; + }); + jest.spyOn(perf, 'clearMarks').mockImplementation(() => { return null; }); + jest.spyOn(perf, 'clearMeasures').mockImplementation(() => { return null; }); + + const layerIndex = new StyleLayerIndex(layers); + const source = new GeoJSONWorkerSource(actor, layerIndex, [], (params, callback) => { + callback(null, geoJson); + return {cancel: () => {}}; + }); + + source.loadData({source: 'testSource', request: {url: 'http://localhost/nonexistent', collectResourceTiming: true}} as LoadGeoJSONParameters, (err, result) => { + expect(err).toBeNull(); + expect(result.resourceTiming.testSource).toEqual( + [{'duration': 250, 'entryType': 'measure', 'name': 'http://localhost/nonexistent', 'startTime': 100}] + ); + done(); + }); + }); + + test('loadData - data', done => { + const layerIndex = new StyleLayerIndex(layers); + const source = new GeoJSONWorkerSource(actor, layerIndex, []); + + source.loadData({source: 'testSource', data: JSON.stringify(geoJson)} as LoadGeoJSONParameters, (err, result) => { + expect(err).toBeNull(); + expect(result.resourceTiming).toBeUndefined(); + done(); + }); + }); + +}); + +describe('loadData', () => { + let server: FakeServer; + beforeEach(() => { + global.fetch = null; + server = fakeServer.create(); + }); + afterEach(() => { + server.restore(); + }); + + const layers = [ + { + id: 'layer1', + source: 'source1', + type: 'symbol', + }, + { + id: 'layer2', + source: 'source2', + type: 'symbol', + } + ] as LayerSpecification[]; + + const geoJson = { + 'type': 'Feature', + 'geometry': { + 'type': 'Point', + 'coordinates': [0, 0] + } + } as GeoJSON.GeoJSON; + + const updateableGeoJson = { + type: 'Feature', + id: 'point', + geometry: { + type: 'Point', + coordinates: [0, 0], + }, + properties: {}, + } as GeoJSON.GeoJSON; + + const layerIndex = new StyleLayerIndex(layers); + function createWorker() { + const worker = new GeoJSONWorkerSource(actor, layerIndex, []); + + // Making the call to loadGeoJSON asynchronous + // allows these tests to mimic a message queue building up + // (regardless of timing) + const originalLoadGeoJSON = worker.loadGeoJSON; + worker.loadGeoJSON = function(params, callback) { + const timeout = setTimeout(() => { + originalLoadGeoJSON(params, callback); + }, 0); + + return {cancel: () => clearTimeout(timeout)}; + }; + return worker; + } + + test('abandons previous callbacks', done => { + const worker = createWorker(); + let firstCallbackHasRun = false; + + worker.loadData({source: 'source1', data: JSON.stringify(geoJson)} as LoadGeoJSONParameters, (err, result) => { + expect(err).toBeNull(); + expect(result && result.abandoned).toBeTruthy(); + firstCallbackHasRun = true; + }); + + worker.loadData({source: 'source1', data: JSON.stringify(geoJson)} as LoadGeoJSONParameters, (err, result) => { + expect(err).toBeNull(); + expect(result && result.abandoned).toBeFalsy(); + expect(firstCallbackHasRun).toBeTruthy(); + done(); + }); + }); + + test('removeSource aborts callbacks', done => { + const worker = createWorker(); + let loadDataCallbackHasRun = false; + worker.loadData({source: 'source1', data: JSON.stringify(geoJson)} as LoadGeoJSONParameters, (err, result) => { + expect(err).toBeNull(); + expect(result && result.abandoned).toBeTruthy(); + loadDataCallbackHasRun = true; + }); + + worker.removeSource({source: 'source1'}, (err) => { + expect(err).toBeFalsy(); + expect(loadDataCallbackHasRun).toBeTruthy(); + done(); + }); + }); + + test('loadData with geojson creates an non-updateable source', done => { + const worker = new GeoJSONWorkerSource(actor, layerIndex, []); + + worker.loadData({source: 'source1', data: JSON.stringify(geoJson)} as LoadGeoJSONParameters, (err, _result) => { + expect(err).toBeNull(); + worker.loadData({source: 'source1', dataDiff: {removeAll: true}} as LoadGeoJSONParameters, (err, _result) => { + expect(err).toBeDefined(); + done(); + }); + }); + }); + + test('loadData with geojson creates an updateable source', done => { + const worker = new GeoJSONWorkerSource(actor, layerIndex, []); + + worker.loadData({source: 'source1', data: JSON.stringify(updateableGeoJson)} as LoadGeoJSONParameters, (err, _result) => { + expect(err).toBeNull(); + worker.loadData({source: 'source1', dataDiff: {removeAll: true}} as LoadGeoJSONParameters, (err, _result) => { + expect(err).toBeNull(); + done(); + }); + }); + }); + + test('loadData with geojson network call creates an updateable source', done => { + const worker = new GeoJSONWorkerSource(actor, layerIndex, []); + + server.respondWith(request => { + request.respond(200, {'Content-Type': 'application/json'}, JSON.stringify(updateableGeoJson)); + }); + + worker.loadData({source: 'source1', request: {url: ''}} as LoadGeoJSONParameters, (err, _result) => { + expect(err).toBeNull(); + worker.loadData({source: 'source1', dataDiff: {removeAll: true}} as LoadGeoJSONParameters, (err, _result) => { + expect(err).toBeNull(); + done(); + }); + }); + + server.respond(); + }); + + test('loadData with geojson network call creates a non-updateable source', done => { + const worker = new GeoJSONWorkerSource(actor, layerIndex, []); + + server.respondWith(request => { + request.respond(200, {'Content-Type': 'application/json'}, JSON.stringify(geoJson)); + }); + + worker.loadData({source: 'source1', request: {url: ''}} as LoadGeoJSONParameters, (err, _result) => { + expect(err).toBeNull(); + worker.loadData({source: 'source1', dataDiff: {removeAll: true}} as LoadGeoJSONParameters, (err, _result) => { + expect(err).toBeDefined(); + done(); + }); + }); + + server.respond(); + }); + + test('loadData with diff updates', done => { + const worker = new GeoJSONWorkerSource(actor, layerIndex, []); + + worker.loadData({source: 'source1', data: JSON.stringify(updateableGeoJson)} as LoadGeoJSONParameters, (err, _result) => { + expect(err).toBeNull(); + worker.loadData({source: 'source1', dataDiff: { + add: [{ + type: 'Feature', + id: 'update_point', + geometry: {type: 'Point', coordinates: [0, 0]}, + properties: {} + }]}} as LoadGeoJSONParameters, (err, _result) => { + expect(err).toBeNull(); + done(); + }); + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/geojson_worker_source.ts b/web/libraries/maplibre-gl/src/source/geojson_worker_source.ts new file mode 100644 index 00000000..a0fc131c --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/geojson_worker_source.ts @@ -0,0 +1,327 @@ +import {getJSON} from '../util/ajax'; + +import {RequestPerformance} from '../util/performance'; +import rewind from '@mapbox/geojson-rewind'; +import {GeoJSONWrapper} from './geojson_wrapper'; +import vtpbf from 'vt-pbf'; +import Supercluster, {type Options as SuperclusterOptions, type ClusterProperties} from 'supercluster'; +import geojsonvt, {type Options as GeoJSONVTOptions} from 'geojson-vt'; +import {VectorTileWorkerSource} from './vector_tile_worker_source'; +import {createExpression} from '@maplibre/maplibre-gl-style-spec'; + +import type { + WorkerTileParameters, + WorkerTileCallback, +} from '../source/worker_source'; + +import type {Actor} from '../util/actor'; +import type {StyleLayerIndex} from '../style/style_layer_index'; + +import type {LoadVectorDataCallback} from './vector_tile_worker_source'; +import type {RequestParameters, ResponseCallback} from '../util/ajax'; +import type {Callback} from '../types/callback'; +import type {Cancelable} from '../types/cancelable'; +import {isUpdateableGeoJSON, type GeoJSONSourceDiff, applySourceDiff, toUpdateable, GeoJSONFeatureId} from './geojson_source_diff'; + +export type LoadGeoJSONParameters = { + request?: RequestParameters; + /** + * Literal GeoJSON data. Must be provided if `request.url` is not. + */ + data?: string; + dataDiff?: GeoJSONSourceDiff; + source: string; + cluster: boolean; + superclusterOptions?: SuperclusterOptions; + geojsonVtOptions?: GeoJSONVTOptions; + clusterProperties?: ClusterProperties; + filter?: Array; + promoteId?: string; +}; + +export type LoadGeoJSON = (params: LoadGeoJSONParameters, callback: ResponseCallback) => Cancelable; + +type GeoJSONIndex = ReturnType | Supercluster; + +/** + * The {@link WorkerSource} implementation that supports {@link GeoJSONSource}. + * This class is designed to be easily reused to support custom source types + * for data formats that can be parsed/converted into an in-memory GeoJSON + * representation. To do so, create it with + * `new GeoJSONWorkerSource(actor, layerIndex, customLoadGeoJSONFunction)`. + * For a full example, see [mapbox-gl-topojson](https://github.com/developmentseed/mapbox-gl-topojson). + */ +export class GeoJSONWorkerSource extends VectorTileWorkerSource { + _pendingCallback: Callback<{ + resourceTiming?: {[_: string]: Array}; + abandoned?: boolean; + }>; + _pendingRequest: Cancelable; + _geoJSONIndex: GeoJSONIndex; + _dataUpdateable = new Map(); + + /** + * @param loadGeoJSON - Optional method for custom loading/parsing of + * GeoJSON based on parameters passed from the main-thread Source. + * See {@link GeoJSONWorkerSource#loadGeoJSON}. + */ + constructor(actor: Actor, layerIndex: StyleLayerIndex, availableImages: Array, loadGeoJSON?: LoadGeoJSON | null) { + super(actor, layerIndex, availableImages); + this.loadVectorData = this.loadGeoJSONTile; + if (loadGeoJSON) { + this.loadGeoJSON = loadGeoJSON; + } + } + + loadGeoJSONTile(params: WorkerTileParameters, callback: LoadVectorDataCallback): (() => void) | void { + const canonical = params.tileID.canonical; + + if (!this._geoJSONIndex) { + return callback(null, null); // we couldn't load the file + } + + const geoJSONTile = this._geoJSONIndex.getTile(canonical.z, canonical.x, canonical.y); + if (!geoJSONTile) { + return callback(null, null); // nothing in the given tile + } + + const geojsonWrapper = new GeoJSONWrapper(geoJSONTile.features); + // Encode the geojson-vt tile into binary vector tile form. This + // is a convenience that allows `FeatureIndex` to operate the same way + // across `VectorTileSource` and `GeoJSONSource` data. + let pbf = vtpbf(geojsonWrapper); + if (pbf.byteOffset !== 0 || pbf.byteLength !== pbf.buffer.byteLength) { + // Compatibility with node Buffer (https://github.com/mapbox/pbf/issues/35) + pbf = new Uint8Array(pbf); + } + + callback(null, { + vectorTile: geojsonWrapper, + rawData: pbf.buffer + }); + } + + /** + * Fetches (if appropriate), parses, and index geojson data into tiles. This + * preparatory method must be called before {@link GeoJSONWorkerSource#loadTile} + * can correctly serve up tiles. + * + * Defers to {@link GeoJSONWorkerSource#loadGeoJSON} for the fetching/parsing, + * expecting `callback(error, data)` to be called with either an error or a + * parsed GeoJSON object. + * + * When a `loadData` request comes in while a previous one is being processed, + * the previous one is aborted. + * + * @param params - the parameters + * @param callback - the callback for completion or error + */ + loadData(params: LoadGeoJSONParameters, callback: Callback<{ + resourceTiming?: {[_: string]: Array}; + abandoned?: boolean; + }>) { + this._pendingRequest?.cancel(); + if (this._pendingCallback) { + // Tell the foreground the previous call has been abandoned + this._pendingCallback(null, {abandoned: true}); + } + + const perf = (params && params.request && params.request.collectResourceTiming) ? + new RequestPerformance(params.request) : false; + + this._pendingCallback = callback; + this._pendingRequest = this.loadGeoJSON(params, (err?: Error | null, data?: any | null) => { + delete this._pendingCallback; + delete this._pendingRequest; + + if (err || !data) { + return callback(err); + } else if (typeof data !== 'object') { + return callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`)); + } else { + rewind(data, true); + + try { + if (params.filter) { + const compiled = createExpression(params.filter, {type: 'boolean', 'property-type': 'data-driven', overridable: false, transition: false} as any); + if (compiled.result === 'error') + throw new Error(compiled.value.map(err => `${err.key}: ${err.message}`).join(', ')); + + const features = data.features.filter(feature => compiled.value.evaluate({zoom: 0}, feature)); + data = {type: 'FeatureCollection', features}; + } + + this._geoJSONIndex = params.cluster ? + new Supercluster(getSuperclusterOptions(params)).load(data.features) : + geojsonvt(data, params.geojsonVtOptions); + } catch (err) { + return callback(err); + } + + this.loaded = {}; + + const result = {} as { resourceTiming: any }; + if (perf) { + const resourceTimingData = perf.finish(); + // it's necessary to eval the result of getEntriesByName() here via parse/stringify + // late evaluation in the main thread causes TypeError: illegal invocation + if (resourceTimingData) { + result.resourceTiming = {}; + result.resourceTiming[params.source] = JSON.parse(JSON.stringify(resourceTimingData)); + } + } + callback(null, result); + } + }); + } + + /** + * Implements {@link WorkerSource#reloadTile}. + * + * If the tile is loaded, uses the implementation in VectorTileWorkerSource. + * Otherwise, such as after a setData() call, we load the tile fresh. + * + * @param params - the parameters + * @param callback - the callback for completion or error + */ + reloadTile(params: WorkerTileParameters, callback: WorkerTileCallback) { + const loaded = this.loaded, + uid = params.uid; + + if (loaded && loaded[uid]) { + return super.reloadTile(params, callback); + } else { + return this.loadTile(params, callback); + } + } + + /** + * Fetch and parse GeoJSON according to the given params. Calls `callback` + * with `(err, data)`, where `data` is a parsed GeoJSON object. + * + * GeoJSON is loaded and parsed from `params.url` if it exists, or else + * expected as a literal (string or object) `params.data`. + * + * @param params - the parameters + * @param callback - the callback for completion or error + * @returns A Cancelable object. + */ + loadGeoJSON = (params: LoadGeoJSONParameters, callback: ResponseCallback): Cancelable => { + const {promoteId} = params; + // Because of same origin issues, urls must either include an explicit + // origin or absolute path. + // ie: /foo/bar.json or http://example.com/bar.json + // but not ../foo/bar.json + if (params.request) { + return getJSON(params.request, ( + error?: Error, + data?: any, + cacheControl?: string, + expires?: string + ) => { + this._dataUpdateable = isUpdateableGeoJSON(data, promoteId) ? toUpdateable(data, promoteId) : undefined; + callback(error, data, cacheControl, expires); + }); + } else if (typeof params.data === 'string') { + try { + const parsed = JSON.parse(params.data); + this._dataUpdateable = isUpdateableGeoJSON(parsed, promoteId) ? toUpdateable(parsed, promoteId) : undefined; + callback(null, parsed); + } catch (e) { + callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`)); + } + } else if (params.dataDiff) { + if (this._dataUpdateable) { + applySourceDiff(this._dataUpdateable, params.dataDiff, promoteId); + callback(null, {type: 'FeatureCollection', features: Array.from(this._dataUpdateable.values())}); + } else { + callback(new Error(`Cannot update existing geojson data in ${params.source}`)); + } + } else { + callback(new Error(`Input data given to '${params.source}' is not a valid GeoJSON object.`)); + } + + return {cancel: () => {}}; + }; + + removeSource(params: { + source: string; + }, callback: WorkerTileCallback) { + if (this._pendingCallback) { + // Don't leak callbacks + this._pendingCallback(null, {abandoned: true}); + } + callback(); + } + + getClusterExpansionZoom(params: { + clusterId: number; + }, callback: Callback) { + try { + callback(null, (this._geoJSONIndex as Supercluster).getClusterExpansionZoom(params.clusterId)); + } catch (e) { + callback(e); + } + } + + getClusterChildren(params: { + clusterId: number; + }, callback: Callback>) { + try { + callback(null, (this._geoJSONIndex as Supercluster).getChildren(params.clusterId)); + } catch (e) { + callback(e); + } + } + + getClusterLeaves(params: { + clusterId: number; + limit: number; + offset: number; + }, callback: Callback>) { + try { + callback(null, (this._geoJSONIndex as Supercluster).getLeaves(params.clusterId, params.limit, params.offset)); + } catch (e) { + callback(e); + } + } +} + +function getSuperclusterOptions({superclusterOptions, clusterProperties}: LoadGeoJSONParameters) { + if (!clusterProperties || !superclusterOptions) return superclusterOptions; + + const mapExpressions = {}; + const reduceExpressions = {}; + const globals = {accumulated: null, zoom: 0}; + const feature = {properties: null}; + const propertyNames = Object.keys(clusterProperties); + + for (const key of propertyNames) { + const [operator, mapExpression] = clusterProperties[key]; + + const mapExpressionParsed = createExpression(mapExpression); + const reduceExpressionParsed = createExpression( + typeof operator === 'string' ? [operator, ['accumulated'], ['get', key]] : operator); + + mapExpressions[key] = mapExpressionParsed.value; + reduceExpressions[key] = reduceExpressionParsed.value; + } + + superclusterOptions.map = (pointProperties) => { + feature.properties = pointProperties; + const properties = {}; + for (const key of propertyNames) { + properties[key] = mapExpressions[key].evaluate(globals, feature); + } + return properties; + }; + superclusterOptions.reduce = (accumulated, clusterProperties) => { + feature.properties = clusterProperties; + for (const key of propertyNames) { + globals.accumulated = accumulated[key]; + accumulated[key] = reduceExpressions[key].evaluate(globals, feature); + } + }; + + return superclusterOptions; +} diff --git a/web/libraries/maplibre-gl/src/source/geojson_wrapper.test.ts b/web/libraries/maplibre-gl/src/source/geojson_wrapper.test.ts new file mode 100644 index 00000000..b0f37a28 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/geojson_wrapper.test.ts @@ -0,0 +1,32 @@ +import {GeoJSONWrapper} from './geojson_wrapper'; + +describe('geojsonwrapper', () => { + test('linestring', () => { + const features = [{ + type: 2, + geometry: [[[0, 0], [10, 10]]], + tags: {hello: 'world'} + }]; + + const wrap = new GeoJSONWrapper(features as any); + const feature = wrap.feature(0); + + expect(feature).toBeTruthy(); + expect(feature.loadGeometry()).toEqual([[{x: 0, y: 0}, {x: 10, y: 10}]]); + expect(feature.type).toBe(2); + expect(feature.properties).toEqual({hello: 'world'}); + + }); + + test('point', () => { + const features = [{ + type: 1, + geometry: [[0, 1]], + tags: {} + }]; + + const wrap = new GeoJSONWrapper(features as any); + const feature = wrap.feature(0); + expect(feature.loadGeometry()).toEqual([[{x: 0, y: 1}]]); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/geojson_wrapper.ts b/web/libraries/maplibre-gl/src/source/geojson_wrapper.ts new file mode 100644 index 00000000..a7098c89 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/geojson_wrapper.ts @@ -0,0 +1,81 @@ +import Point from '@mapbox/point-geometry'; + +import mvt from '@mapbox/vector-tile'; +import type {VectorTileFeature, VectorTileLayer, VectorTile} from '@mapbox/vector-tile'; +const toGeoJSON = mvt.VectorTileFeature.prototype.toGeoJSON; +import {EXTENT} from '../data/extent'; +import type {TileFeature, AnyProps} from 'supercluster'; +import type {Feature as GeoJSONVTFeature} from 'geojson-vt'; + +export type Feature = TileFeature | GeoJSONVTFeature; + +class FeatureWrapper implements VectorTileFeature { + _feature: Feature; + + extent: number; + type: Feature['type']; + id: number; + properties: {[_: string]: string | number | boolean}; + + constructor(feature: Feature) { + this._feature = feature; + + this.extent = EXTENT; + this.type = feature.type; + this.properties = feature.tags; + + // If the feature has a top-level `id` property, copy it over, but only + // if it can be coerced to an integer, because this wrapper is used for + // serializing geojson feature data into vector tile PBF data, and the + // vector tile spec only supports integer values for feature ids -- + // allowing non-integer values here results in a non-compliant PBF + // that causes an exception when it is parsed with vector-tile-js + if ('id' in feature && !isNaN(feature.id as any)) { + this.id = parseInt(feature.id, 10); + } + } + + loadGeometry() { + if (this._feature.type === 1) { + const geometry = []; + for (const point of this._feature.geometry) { + geometry.push([new Point(point[0], point[1])]); + } + return geometry; + } else { + const geometry = []; + for (const ring of this._feature.geometry) { + const newRing = []; + for (const point of ring) { + newRing.push(new Point(point[0], point[1])); + } + geometry.push(newRing); + } + return geometry; + } + } + + toGeoJSON(x: number, y: number, z: number) { + return toGeoJSON.call(this, x, y, z); + } +} + +export class GeoJSONWrapper implements VectorTile, VectorTileLayer { + layers: {[_: string]: VectorTileLayer}; + name: string; + extent: number; + length: number; + _features: Array; + + constructor(features: Array) { + this.layers = {'_geojsonTileLayer': this}; + this.name = '_geojsonTileLayer'; + this.extent = EXTENT; + this.length = features.length; + this._features = features; + } + + feature(i: number): VectorTileFeature { + return new FeatureWrapper(this._features[i]); + } +} diff --git a/web/libraries/maplibre-gl/src/source/image_source.test.ts b/web/libraries/maplibre-gl/src/source/image_source.test.ts new file mode 100644 index 00000000..124b2a86 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/image_source.test.ts @@ -0,0 +1,205 @@ +import {ImageSource} from './image_source'; +import {Evented} from '../util/evented'; +import {Transform} from '../geo/transform'; +import {extend} from '../util/util'; +import {type FakeServer, fakeServer} from 'nise'; +import {RequestManager} from '../util/request_manager'; +import {Dispatcher} from '../util/dispatcher'; +import {stubAjaxGetImage} from '../util/test/util'; +import {Tile} from './tile'; +import {OverscaledTileID} from './tile_id'; +import {VertexBuffer} from '../gl/vertex_buffer'; +import {SegmentVector} from '../data/segment'; +import {Texture} from '../render/texture'; +import type {ImageSourceSpecification} from '@maplibre/maplibre-gl-style-spec'; + +function createSource(options) { + options = extend({ + coordinates: [[0, 0], [1, 0], [1, 1], [0, 1]] + }, options); + + const source = new ImageSource('id', options, {send() {}} as any as Dispatcher, options.eventedParent); + return source; +} + +class StubMap extends Evented { + transform: Transform; + painter: any; + _requestManager: RequestManager; + + constructor() { + super(); + this.transform = new Transform(); + this._requestManager = { + transformRequest: (url) => { + return {url}; + } + } as any as RequestManager; + this.painter = { + context: { + gl: {} + } + }; + } +} + +describe('ImageSource', () => { + stubAjaxGetImage(undefined); + let server: FakeServer; + + beforeEach(() => { + global.fetch = null; + server = fakeServer.create(); + server.respondWith(new ArrayBuffer(1)); + }); + + test('constructor', () => { + const source = createSource({url: '/image.png'}); + + expect(source.minzoom).toBe(0); + expect(source.maxzoom).toBe(22); + expect(source.tileSize).toBe(512); + }); + + test('fires dataloading event', () => { + const source = createSource({url: '/image.png'}); + source.on('dataloading', (e) => { + expect(e.dataType).toBe('source'); + }); + source.onAdd(new StubMap() as any); + server.respond(); + expect(source.image).toBeTruthy(); + }); + + test('transforms url request', () => { + const source = createSource({url: '/image.png'}); + const map = new StubMap() as any; + const spy = jest.spyOn(map._requestManager, 'transformRequest'); + source.onAdd(map); + server.respond(); + expect(spy).toHaveBeenCalledTimes(1); + expect(spy.mock.calls[0][0]).toBe('/image.png'); + expect(spy.mock.calls[0][1]).toBe('Image'); + }); + + test('updates url from updateImage', () => { + const source = createSource({url: '/image.png'}); + const map = new StubMap() as any; + const spy = jest.spyOn(map._requestManager, 'transformRequest'); + source.onAdd(map); + server.respond(); + expect(spy).toHaveBeenCalledTimes(1); + expect(spy.mock.calls[0][0]).toBe('/image.png'); + expect(spy.mock.calls[0][1]).toBe('Image'); + source.updateImage({url: '/image2.png'}); + server.respond(); + expect(spy).toHaveBeenCalledTimes(2); + expect(spy.mock.calls[1][0]).toBe('/image2.png'); + expect(spy.mock.calls[1][1]).toBe('Image'); + }); + + test('sets coordinates', () => { + const source = createSource({url: '/image.png'}); + const map = new StubMap() as any; + source.onAdd(map); + server.respond(); + const beforeSerialized = source.serialize(); + expect(beforeSerialized.coordinates).toEqual([[0, 0], [1, 0], [1, 1], [0, 1]]); + source.setCoordinates([[0, 0], [-1, 0], [-1, -1], [0, -1]]); + const afterSerialized = source.serialize(); + expect(afterSerialized.coordinates).toEqual([[0, 0], [-1, 0], [-1, -1], [0, -1]]); + }); + + test('sets coordinates via updateImage', () => { + const source = createSource({url: '/image.png'}); + const map = new StubMap() as any; + source.onAdd(map); + server.respond(); + const beforeSerialized = source.serialize(); + expect(beforeSerialized.coordinates).toEqual([[0, 0], [1, 0], [1, 1], [0, 1]]); + source.updateImage({ + url: '/image2.png', + coordinates: [[0, 0], [-1, 0], [-1, -1], [0, -1]] + }); + server.respond(); + const afterSerialized = source.serialize(); + expect(afterSerialized.coordinates).toEqual([[0, 0], [-1, 0], [-1, -1], [0, -1]]); + }); + + test('fires data event when content is loaded', done => { + const source = createSource({url: '/image.png'}); + source.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'content') { + expect(typeof source.tileID == 'object').toBeTruthy(); + done(); + } + }); + source.onAdd(new StubMap() as any); + server.respond(); + }); + + test('fires data event when metadata is loaded', done => { + const source = createSource({url: '/image.png'}); + source.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'metadata') { + done(); + } + }); + source.onAdd(new StubMap() as any); + server.respond(); + }); + + test('fires idle event on prepare call when there is at least one not loaded tile', done => { + const source = createSource({url: '/image.png'}); + const tile = new Tile(new OverscaledTileID(1, 0, 1, 0, 0), 512); + source.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'idle') { + expect(tile.state).toBe('loaded'); + done(); + } + }); + source.onAdd(new StubMap() as any); + server.respond(); + + source.tiles[String(tile.tileID.wrap)] = tile; + source.image = new ImageBitmap(); + // assign dummies directly so we don't need to stub the gl things + source.boundsBuffer = {} as VertexBuffer; + source.boundsSegments = {} as SegmentVector; + source.texture = {} as Texture; + source.prepare(); + }); + + test('serialize url and coordinates', () => { + const source = createSource({url: '/image.png'}); + + const serialized = source.serialize() as ImageSourceSpecification; + expect(serialized.type).toBe('image'); + expect(serialized.url).toBe('/image.png'); + expect(serialized.coordinates).toEqual([[0, 0], [1, 0], [1, 1], [0, 1]]); + }); + + test('allows using updateImage before initial image is loaded', () => { + const source = createSource({url: '/image.png'}); + const map = new StubMap() as any; + + source.onAdd(map); + + expect(source.image).toBeUndefined(); + source.updateImage({url: '/image2.png'}); + server.respond(); + expect(source.image).toBeTruthy(); + }); + + test('cancels request if updateImage is used', () => { + const source = createSource({url: '/image.png'}); + const map = new StubMap() as any; + + source.onAdd(map); + + const spy = jest.spyOn(server.requests[0] as any, 'abort'); + + source.updateImage({url: '/image2.png'}); + expect(spy).toHaveBeenCalled(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/image_source.ts b/web/libraries/maplibre-gl/src/source/image_source.ts new file mode 100644 index 00000000..8be422aa --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/image_source.ts @@ -0,0 +1,343 @@ +import {CanonicalTileID} from './tile_id'; +import {Event, ErrorEvent, Evented} from '../util/evented'; +import {ImageRequest} from '../util/image_request'; +import {ResourceType} from '../util/request_manager'; +import {EXTENT} from '../data/extent'; +import {RasterBoundsArray} from '../data/array_types.g'; +import rasterBoundsAttributes from '../data/raster_bounds_attributes'; +import {SegmentVector} from '../data/segment'; +import {Texture} from '../render/texture'; +import {MercatorCoordinate} from '../geo/mercator_coordinate'; + +import type {Source} from './source'; +import type {CanvasSourceSpecification} from './canvas_source'; +import type {Map} from '../ui/map'; +import type {Dispatcher} from '../util/dispatcher'; +import type {Tile} from './tile'; +import type {Callback} from '../types/callback'; +import type {VertexBuffer} from '../gl/vertex_buffer'; +import type { + ImageSourceSpecification, + VideoSourceSpecification +} from '@maplibre/maplibre-gl-style-spec'; +import {Cancelable} from '../types/cancelable'; + +/** + * Four geographical coordinates, + * represented as arrays of longitude and latitude numbers, which define the corners of the image. + * The coordinates start at the top left corner of the image and proceed in clockwise order. + * They do not have to represent a rectangle. + */ +export type Coordinates = [[number, number], [number, number], [number, number], [number, number]]; + +/** + * The options object for the {@link ImageSource#updateImage} method + */ +export type UpdateImageOptions = { + /** + * Required image URL. + */ + url: string; + /** + * The image coordinates + */ + coordinates?: Coordinates; +} + +/** + * A data source containing an image. + * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-image) for detailed documentation of options.) + * + * @group Sources + * + * @example + * ```ts + * // add to map + * map.addSource('some id', { + * type: 'image', + * url: 'https://www.maplibre.org/images/foo.png', + * coordinates: [ + * [-76.54, 39.18], + * [-76.52, 39.18], + * [-76.52, 39.17], + * [-76.54, 39.17] + * ] + * }); + * + * // update coordinates + * let mySource = map.getSource('some id'); + * mySource.setCoordinates([ + * [-76.54335737228394, 39.18579907229748], + * [-76.52803659439087, 39.1838364847587], + * [-76.5295386314392, 39.17683392507606], + * [-76.54520273208618, 39.17876344106642] + * ]); + * + * // update url and coordinates simultaneously + * mySource.updateImage({ + * url: 'https://www.maplibre.org/images/bar.png', + * coordinates: [ + * [-76.54335737228394, 39.18579907229748], + * [-76.52803659439087, 39.1838364847587], + * [-76.5295386314392, 39.17683392507606], + * [-76.54520273208618, 39.17876344106642] + * ] + * }) + * + * map.removeSource('some id'); // remove + * ``` + */ +export class ImageSource extends Evented implements Source { + type: string; + id: string; + minzoom: number; + maxzoom: number; + tileSize: number; + url: string; + + coordinates: Coordinates; + tiles: {[_: string]: Tile}; + options: any; + dispatcher: Dispatcher; + map: Map; + texture: Texture | null; + image: HTMLImageElement | ImageBitmap; + tileID: CanonicalTileID; + _boundsArray: RasterBoundsArray; + boundsBuffer: VertexBuffer; + boundsSegments: SegmentVector; + _loaded: boolean; + _request: Cancelable; + + /** @internal */ + constructor(id: string, options: ImageSourceSpecification | VideoSourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) { + super(); + this.id = id; + this.dispatcher = dispatcher; + this.coordinates = options.coordinates; + + this.type = 'image'; + this.minzoom = 0; + this.maxzoom = 22; + this.tileSize = 512; + this.tiles = {}; + this._loaded = false; + + this.setEventedParent(eventedParent); + + this.options = options; + } + + load = (newCoordinates?: Coordinates, successCallback?: () => void) => { + this._loaded = false; + this.fire(new Event('dataloading', {dataType: 'source'})); + + this.url = this.options.url; + + this._request = ImageRequest.getImage(this.map._requestManager.transformRequest(this.url, ResourceType.Image), (err, image) => { + this._request = null; + this._loaded = true; + + if (err) { + this.fire(new ErrorEvent(err)); + } else if (image) { + this.image = image; + if (newCoordinates) { + this.coordinates = newCoordinates; + } + if (successCallback) { + successCallback(); + } + this._finishLoading(); + } + }); + }; + + loaded(): boolean { + return this._loaded; + } + + /** + * Updates the image URL and, optionally, the coordinates. To avoid having the image flash after changing, + * set the `raster-fade-duration` paint property on the raster layer to 0. + * + * @param options - The options object. + * @returns `this` + */ + updateImage(options: UpdateImageOptions): this { + if (!options.url) { + return this; + } + + if (this._request) { + this._request.cancel(); + this._request = null; + } + + this.options.url = options.url; + this.load(options.coordinates, () => { this.texture = null; }); + return this; + } + + _finishLoading() { + if (this.map) { + this.setCoordinates(this.coordinates); + this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'})); + } + } + + onAdd(map: Map) { + this.map = map; + this.load(); + } + + onRemove() { + if (this._request) { + this._request.cancel(); + this._request = null; + } + } + + /** + * Sets the image's coordinates and re-renders the map. + * + * @param coordinates - Four geographical coordinates, + * represented as arrays of longitude and latitude numbers, which define the corners of the image. + * The coordinates start at the top left corner of the image and proceed in clockwise order. + * They do not have to represent a rectangle. + * @returns `this` + */ + setCoordinates(coordinates: Coordinates): this { + this.coordinates = coordinates; + + // Calculate which mercator tile is suitable for rendering the video in + // and create a buffer with the corner coordinates. These coordinates + // may be outside the tile, because raster tiles aren't clipped when rendering. + + // transform the geo coordinates into (zoom 0) tile space coordinates + const cornerCoords = coordinates.map(MercatorCoordinate.fromLngLat); + + // Compute the coordinates of the tile we'll use to hold this image's + // render data + this.tileID = getCoordinatesCenterTileID(cornerCoords); + + // Constrain min/max zoom to our tile's zoom level in order to force + // SourceCache to request this tile (no matter what the map's zoom + // level) + this.minzoom = this.maxzoom = this.tileID.z; + + // Transform the corner coordinates into the coordinate space of our + // tile. + const tileCoords = cornerCoords.map((coord) => this.tileID.getTilePoint(coord)._round()); + + this._boundsArray = new RasterBoundsArray(); + this._boundsArray.emplaceBack(tileCoords[0].x, tileCoords[0].y, 0, 0); + this._boundsArray.emplaceBack(tileCoords[1].x, tileCoords[1].y, EXTENT, 0); + this._boundsArray.emplaceBack(tileCoords[3].x, tileCoords[3].y, 0, EXTENT); + this._boundsArray.emplaceBack(tileCoords[2].x, tileCoords[2].y, EXTENT, EXTENT); + + if (this.boundsBuffer) { + this.boundsBuffer.destroy(); + delete this.boundsBuffer; + } + + this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'})); + return this; + } + + prepare = () => { + if (Object.keys(this.tiles).length === 0 || !this.image) { + return; + } + + const context = this.map.painter.context; + const gl = context.gl; + + if (!this.boundsBuffer) { + this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members); + } + + if (!this.boundsSegments) { + this.boundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2); + } + + if (!this.texture) { + this.texture = new Texture(context, this.image, gl.RGBA); + this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + } + + let newTilesLoaded = false; + for (const w in this.tiles) { + const tile = this.tiles[w]; + if (tile.state !== 'loaded') { + tile.state = 'loaded'; + tile.texture = this.texture; + newTilesLoaded = true; + } + } + + if (newTilesLoaded) { + this.fire(new Event('data', {dataType: 'source', sourceDataType: 'idle', sourceId: this.id})); + } + }; + + loadTile(tile: Tile, callback: Callback) { + // We have a single tile -- whose coordinates are this.tileID -- that + // covers the image we want to render. If that's the one being + // requested, set it up with the image; otherwise, mark the tile as + // `errored` to indicate that we have no data for it. + // If the world wraps, we may have multiple "wrapped" copies of the + // single tile. + if (this.tileID && this.tileID.equals(tile.tileID.canonical)) { + this.tiles[String(tile.tileID.wrap)] = tile; + tile.buckets = {}; + callback(null); + } else { + tile.state = 'errored'; + callback(null); + } + } + + serialize = (): ImageSourceSpecification | VideoSourceSpecification | CanvasSourceSpecification => { + return { + type: 'image', + url: this.options.url, + coordinates: this.coordinates + }; + }; + + hasTransition() { + return false; + } +} + +/** + * Given a list of coordinates, get their center as a coordinate. + * + * @returns centerpoint + * @internal + */ +export function getCoordinatesCenterTileID(coords: Array) { + let minX = Infinity; + let minY = Infinity; + let maxX = -Infinity; + let maxY = -Infinity; + + for (const coord of coords) { + minX = Math.min(minX, coord.x); + minY = Math.min(minY, coord.y); + maxX = Math.max(maxX, coord.x); + maxY = Math.max(maxY, coord.y); + } + + const dx = maxX - minX; + const dy = maxY - minY; + const dMax = Math.max(dx, dy); + const zoom = Math.max(0, Math.floor(-Math.log(dMax) / Math.LN2)); + const tilesAtZoom = Math.pow(2, zoom); + + return new CanonicalTileID( + zoom, + Math.floor((minX + maxX) / 2 * tilesAtZoom), + Math.floor((minY + maxY) / 2 * tilesAtZoom)); +} diff --git a/web/libraries/maplibre-gl/src/source/load_tilejson.ts b/web/libraries/maplibre-gl/src/source/load_tilejson.ts new file mode 100644 index 00000000..9fc6e61f --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/load_tilejson.ts @@ -0,0 +1,42 @@ +import {pick, extend} from '../util/util'; + +import {getJSON} from '../util/ajax'; +import {ResourceType} from '../util/request_manager'; +import {browser} from '../util/browser'; + +import type {RequestManager} from '../util/request_manager'; +import type {Callback} from '../types/callback'; +import type {TileJSON} from '../types/tilejson'; +import type {Cancelable} from '../types/cancelable'; +import type {RasterDEMSourceSpecification, RasterSourceSpecification, VectorSourceSpecification} from '@maplibre/maplibre-gl-style-spec'; + +export function loadTileJson( + options: RasterSourceSpecification | RasterDEMSourceSpecification | VectorSourceSpecification, + requestManager: RequestManager, + callback: Callback +): Cancelable { + const loaded = function(err: Error, tileJSON: any) { + if (err) { + return callback(err); + } else if (tileJSON) { + const result: any = pick( + // explicit source options take precedence over TileJSON + extend(tileJSON, options), + ['tiles', 'minzoom', 'maxzoom', 'attribution', 'bounds', 'scheme', 'tileSize', 'encoding'] + ); + + if (tileJSON.vector_layers) { + result.vectorLayers = tileJSON.vector_layers; + result.vectorLayerIds = result.vectorLayers.map((layer) => { return layer.id; }); + } + + callback(null, result); + } + }; + + if (options.url) { + return getJSON(requestManager.transformRequest(options.url, ResourceType.Source), loaded); + } else { + return browser.frame(() => loaded(null, options)); + } +} diff --git a/web/libraries/maplibre-gl/src/source/pixels_to_tile_units.ts b/web/libraries/maplibre-gl/src/source/pixels_to_tile_units.ts new file mode 100644 index 00000000..7d44b005 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/pixels_to_tile_units.ts @@ -0,0 +1,25 @@ +import {EXTENT} from '../data/extent'; + +import type {OverscaledTileID} from './tile_id'; + +/** + * Converts a pixel value at a the given zoom level to tile units. + * + * The shaders mostly calculate everything in tile units so style + * properties need to be converted from pixels to tile units using this. + * + * For example, a translation by 30 pixels at zoom 6.5 will be a + * translation by pixelsToTileUnits(30, 6.5) tile units. + * + * @returns value in tile units + */ +export function pixelsToTileUnits( + tile: { + tileID: OverscaledTileID; + tileSize: number; + }, + pixelValue: number, + z: number +): number { + return pixelValue * (EXTENT / (tile.tileSize * Math.pow(2, z - tile.tileID.overscaledZ))); +} diff --git a/web/libraries/maplibre-gl/src/source/query_features.test.ts b/web/libraries/maplibre-gl/src/source/query_features.test.ts new file mode 100644 index 00000000..b9ff7f86 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/query_features.test.ts @@ -0,0 +1,36 @@ +import { + queryRenderedFeatures, + querySourceFeatures +} from './query_features'; +import {SourceCache} from './source_cache'; +import {Transform} from '../geo/transform'; +import Point from '@mapbox/point-geometry'; +import {Dispatcher} from '../util/dispatcher'; + +describe('QueryFeatures#rendered', () => { + test('returns empty object if source returns no tiles', () => { + const mockSourceCache = {tilesIn () { return []; }} as any as SourceCache; + const transform = new Transform(); + const result = queryRenderedFeatures(mockSourceCache, {}, undefined, [] as Point[], undefined, transform); + expect(result).toEqual({}); + }); + +}); + +describe('QueryFeatures#source', () => { + test('returns empty result when source has no features', () => { + const sourceCache = new SourceCache('test', { + type: 'geojson', + data: {type: 'FeatureCollection', features: []} + }, { + getActor() { + return { + send(type, params, callback) { return callback(); } + }; + } + } as any as Dispatcher); + const result = querySourceFeatures(sourceCache, {}); + expect(result).toEqual([]); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/source/query_features.ts b/web/libraries/maplibre-gl/src/source/query_features.ts new file mode 100644 index 00000000..18bb52bf --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/query_features.ts @@ -0,0 +1,252 @@ +import type {SourceCache} from './source_cache'; +import type {StyleLayer} from '../style/style_layer'; +import type {CollisionIndex} from '../symbol/collision_index'; +import type {Transform} from '../geo/transform'; +import type {RetainedQueryData} from '../symbol/placement'; +import type {FilterSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {MapGeoJSONFeature} from '../util/vectortile_to_geojson'; +import type Point from '@mapbox/point-geometry'; +import {mat4} from 'gl-matrix'; + +/** + * Options to pass to query the map for the rendered features + */ +export type QueryRenderedFeaturesOptions = { + /** + * An array of [style layer IDs](https://maplibre.org/maplibre-style-spec/#layer-id) for the query to inspect. + * Only features within these layers will be returned. If this parameter is undefined, all layers will be checked. + */ + layers?: Array; + /** + * A [filter](https://maplibre.org/maplibre-style-spec/layers/#filter) to limit query results. + */ + filter?: FilterSpecification; + /** + * An array of string representing the available images + */ + availableImages?: Array; + /** + * Whether to check if the [options.filter] conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function. + */ + validate?: boolean; +}; + +/** + * The options object related to the {@link Map#queryRenderedFeatures} method + */ +export type QuerySourceFeatureOptions = { + /** + * The name of the source layer to query. *For vector tile sources, this parameter is required.* For GeoJSON sources, it is ignored. + */ + sourceLayer?: string; + /** + * A [filter](https://maplibre.org/maplibre-style-spec/layers/#filter) + * to limit query results. + */ + filter?: FilterSpecification; + /** + * Whether to check if the [parameters.filter] conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function. + * @defaultValue true + */ + validate?: boolean; +} + +/* + * Returns a matrix that can be used to convert from tile coordinates to viewport pixel coordinates. + */ +function getPixelPosMatrix(transform, tileID) { + const t = mat4.create(); + mat4.translate(t, t, [1, 1, 0]); + mat4.scale(t, t, [transform.width * 0.5, transform.height * 0.5, 1]); + return mat4.multiply(t, t, transform.calculatePosMatrix(tileID.toUnwrapped())); +} + +function queryIncludes3DLayer(layers: Array, styleLayers: {[_: string]: StyleLayer}, sourceID: string) { + if (layers) { + for (const layerID of layers) { + const layer = styleLayers[layerID]; + if (layer && layer.source === sourceID && layer.type === 'fill-extrusion') { + return true; + } + } + } else { + for (const key in styleLayers) { + const layer = styleLayers[key]; + if (layer.source === sourceID && layer.type === 'fill-extrusion') { + return true; + } + } + } + return false; +} + +export function queryRenderedFeatures( + sourceCache: SourceCache, + styleLayers: {[_: string]: StyleLayer}, + serializedLayers: {[_: string]: any}, + queryGeometry: Array, + params: QueryRenderedFeaturesOptions, + transform: Transform +): { [key: string]: Array<{featureIndex: number; feature: MapGeoJSONFeature}> } { + + const has3DLayer = queryIncludes3DLayer(params && params.layers, styleLayers, sourceCache.id); + const maxPitchScaleFactor = transform.maxPitchScaleFactor(); + const tilesIn = sourceCache.tilesIn(queryGeometry, maxPitchScaleFactor, has3DLayer); + + tilesIn.sort(sortTilesIn); + const renderedFeatureLayers = []; + for (const tileIn of tilesIn) { + renderedFeatureLayers.push({ + wrappedTileID: tileIn.tileID.wrapped().key, + queryResults: tileIn.tile.queryRenderedFeatures( + styleLayers, + serializedLayers, + sourceCache._state, + tileIn.queryGeometry, + tileIn.cameraQueryGeometry, + tileIn.scale, + params, + transform, + maxPitchScaleFactor, + getPixelPosMatrix(sourceCache.transform, tileIn.tileID)) + }); + } + + const result = mergeRenderedFeatureLayers(renderedFeatureLayers); + + // Merge state from SourceCache into the results + for (const layerID in result) { + result[layerID].forEach((featureWrapper) => { + const feature = featureWrapper.feature as MapGeoJSONFeature; + const state = sourceCache.getFeatureState(feature.layer['source-layer'], feature.id); + feature.source = feature.layer.source; + if (feature.layer['source-layer']) { + feature.sourceLayer = feature.layer['source-layer']; + } + feature.state = state; + }); + } + return result; +} + +export function queryRenderedSymbols(styleLayers: {[_: string]: StyleLayer}, + serializedLayers: {[_: string]: StyleLayer}, + sourceCaches: {[_: string]: SourceCache}, + queryGeometry: Array, + params: QueryRenderedFeaturesOptions, + collisionIndex: CollisionIndex, + retainedQueryData: { + [_: number]: RetainedQueryData; + }) { + const result = {}; + const renderedSymbols = collisionIndex.queryRenderedSymbols(queryGeometry); + const bucketQueryData = []; + for (const bucketInstanceId of Object.keys(renderedSymbols).map(Number)) { + bucketQueryData.push(retainedQueryData[bucketInstanceId]); + } + bucketQueryData.sort(sortTilesIn); + + for (const queryData of bucketQueryData) { + const bucketSymbols = queryData.featureIndex.lookupSymbolFeatures( + renderedSymbols[queryData.bucketInstanceId], + serializedLayers, + queryData.bucketIndex, + queryData.sourceLayerIndex, + params.filter, + params.layers, + params.availableImages, + styleLayers); + + for (const layerID in bucketSymbols) { + const resultFeatures = result[layerID] = result[layerID] || []; + const layerSymbols = bucketSymbols[layerID]; + layerSymbols.sort((a, b) => { + // Match topDownFeatureComparator from FeatureIndex, but using + // most recent sorting of features from bucket.sortFeatures + const featureSortOrder = queryData.featureSortOrder; + if (featureSortOrder) { + // queryRenderedSymbols documentation says we'll return features in + // "top-to-bottom" rendering order (aka last-to-first). + // Actually there can be multiple symbol instances per feature, so + // we sort each feature based on the first matching symbol instance. + const sortedA = featureSortOrder.indexOf(a.featureIndex); + const sortedB = featureSortOrder.indexOf(b.featureIndex); + return sortedB - sortedA; + } else { + // Bucket hasn't been re-sorted based on angle, so use the + // reverse of the order the features appeared in the data. + return b.featureIndex - a.featureIndex; + } + }); + for (const symbolFeature of layerSymbols) { + resultFeatures.push(symbolFeature); + } + } + } + + // Merge state from SourceCache into the results + for (const layerName in result) { + result[layerName].forEach((featureWrapper) => { + const feature = featureWrapper.feature; + const layer = styleLayers[layerName]; + const sourceCache = sourceCaches[layer.source]; + const state = sourceCache.getFeatureState(feature.layer['source-layer'], feature.id); + feature.source = feature.layer.source; + if (feature.layer['source-layer']) { + feature.sourceLayer = feature.layer['source-layer']; + } + feature.state = state; + }); + } + return result; +} + +export function querySourceFeatures(sourceCache: SourceCache, params: QuerySourceFeatureOptions) { + const tiles = sourceCache.getRenderableIds().map((id) => { + return sourceCache.getTileByID(id); + }); + + const result = []; + + const dataTiles = {}; + for (let i = 0; i < tiles.length; i++) { + const tile = tiles[i]; + const dataID = tile.tileID.canonical.key; + if (!dataTiles[dataID]) { + dataTiles[dataID] = true; + tile.querySourceFeatures(result, params); + } + } + + return result; +} + +function sortTilesIn(a, b) { + const idA = a.tileID; + const idB = b.tileID; + return (idA.overscaledZ - idB.overscaledZ) || (idA.canonical.y - idB.canonical.y) || (idA.wrap - idB.wrap) || (idA.canonical.x - idB.canonical.x); +} + +function mergeRenderedFeatureLayers(tiles) { + // Merge results from all tiles, but if two tiles share the same + // wrapped ID, don't duplicate features between the two tiles + const result = {}; + const wrappedIDLayerMap = {}; + for (const tile of tiles) { + const queryResults = tile.queryResults; + const wrappedID = tile.wrappedTileID; + const wrappedIDLayers = wrappedIDLayerMap[wrappedID] = wrappedIDLayerMap[wrappedID] || {}; + for (const layerID in queryResults) { + const tileFeatures = queryResults[layerID]; + const wrappedIDFeatures = wrappedIDLayers[layerID] = wrappedIDLayers[layerID] || {}; + const resultFeatures = result[layerID] = result[layerID] || []; + for (const tileFeature of tileFeatures) { + if (!wrappedIDFeatures[tileFeature.featureIndex]) { + wrappedIDFeatures[tileFeature.featureIndex] = true; + resultFeatures.push(tileFeature); + } + } + } + } + return result; +} diff --git a/web/libraries/maplibre-gl/src/source/raster_dem_tile_source.test.ts b/web/libraries/maplibre-gl/src/source/raster_dem_tile_source.test.ts new file mode 100644 index 00000000..43bf16bb --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/raster_dem_tile_source.test.ts @@ -0,0 +1,167 @@ +import {fakeServer, FakeServer} from 'nise'; +import {RasterDEMTileSource} from './raster_dem_tile_source'; +import {OverscaledTileID} from './tile_id'; +import {RequestManager} from '../util/request_manager'; +import {Dispatcher} from '../util/dispatcher'; +import {Tile} from './tile'; + +function createSource(options, transformCallback?) { + const source = new RasterDEMTileSource('id', options, {send() {}} as any as Dispatcher, options.eventedParent); + source.onAdd({ + transform: {angle: 0, pitch: 0, showCollisionBoxes: false}, + _getMapId: () => 1, + _requestManager: new RequestManager(transformCallback), + getPixelRatio() { return 1; } + } as any); + + source.on('error', (e) => { + throw e.error; + }); + + return source; +} + +describe('RasterTileSource', () => { + let server: FakeServer; + beforeEach(() => { + global.fetch = null; + server = fakeServer.create(); + }); + + afterEach(() => { + server.restore(); + }); + + test('transforms request for TileJSON URL', done => { + server.respondWith('/source.json', JSON.stringify({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.pngraw'], + bounds: [-47, -7, -45, -5] + })); + const transformSpy = jest.fn().mockImplementation((url) => { + return {url}; + }); + + createSource({url: '/source.json'}, transformSpy); + server.respond(); + + expect(transformSpy.mock.calls[0][0]).toBe('/source.json'); + expect(transformSpy.mock.calls[0][1]).toBe('Source'); + done(); + }); + + test('transforms tile urls before requesting', done => { + server.respondWith('/source.json', JSON.stringify({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'], + bounds: [-47, -7, -45, -5] + })); + const source = createSource({url: '/source.json'}); + const transformSpy = jest.spyOn(source.map._requestManager, 'transformRequest'); + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + const tile = { + tileID: new OverscaledTileID(10, 0, 10, 5, 5), + state: 'loading', + loadVectorData () {}, + setExpiryData() {} + } as any as Tile; + source.loadTile(tile, () => {}); + + expect(transformSpy).toHaveBeenCalledTimes(1); + expect(transformSpy.mock.calls[0][0]).toBe('http://example.com/10/5/5.png'); + expect(transformSpy.mock.calls[0][1]).toBe('Tile'); + done(); + + } + }); + server.respond(); + }); + test('populates neighboringTiles', done => { + server.respondWith('/source.json', JSON.stringify({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'] + })); + const source = createSource({url: '/source.json'}); + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + const tile = { + tileID: new OverscaledTileID(10, 0, 10, 5, 5), + state: 'loading', + loadVectorData () {}, + setExpiryData() {} + } as any as Tile; + source.loadTile(tile, () => {}); + + expect(Object.keys(tile.neighboringTiles)).toEqual([ + new OverscaledTileID(10, 0, 10, 4, 5).key, + new OverscaledTileID(10, 0, 10, 6, 5).key, + new OverscaledTileID(10, 0, 10, 4, 4).key, + new OverscaledTileID(10, 0, 10, 5, 4).key, + new OverscaledTileID(10, 0, 10, 6, 4).key, + new OverscaledTileID(10, 0, 10, 4, 6).key, + new OverscaledTileID(10, 0, 10, 5, 6).key, + new OverscaledTileID(10, 0, 10, 6, 6).key + ]); + + done(); + + } + }); + server.respond(); + }); + + test('populates neighboringTiles with wrapped tiles', done => { + server.respondWith('/source.json', JSON.stringify({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'] + })); + const source = createSource({url: '/source.json'}); + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + const tile = { + tileID: new OverscaledTileID(5, 0, 5, 31, 5), + state: 'loading', + loadVectorData () {}, + setExpiryData() {} + } as any as Tile; + source.loadTile(tile, () => {}); + + expect(Object.keys(tile.neighboringTiles)).toEqual([ + new OverscaledTileID(5, 0, 5, 30, 6).key, + new OverscaledTileID(5, 0, 5, 31, 6).key, + new OverscaledTileID(5, 0, 5, 30, 5).key, + new OverscaledTileID(5, 1, 5, 0, 5).key, + new OverscaledTileID(5, 0, 5, 30, 4).key, + new OverscaledTileID(5, 0, 5, 31, 4).key, + new OverscaledTileID(5, 1, 5, 0, 4).key, + new OverscaledTileID(5, 1, 5, 0, 6).key + ]); + done(); + } + }); + server.respond(); + }); + + it('serializes options', () => { + const source = createSource({ + tiles: ['http://localhost:2900/raster-dem/{z}/{x}/{y}.png'], + minzoom: 2, + maxzoom: 10 + }); + expect(source.serialize()).toStrictEqual({ + type: 'raster-dem', + tiles: ['http://localhost:2900/raster-dem/{z}/{x}/{y}.png'], + minzoom: 2, + maxzoom: 10 + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/raster_dem_tile_source.ts b/web/libraries/maplibre-gl/src/source/raster_dem_tile_source.ts new file mode 100644 index 00000000..19d05224 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/raster_dem_tile_source.ts @@ -0,0 +1,165 @@ +import {ImageRequest} from '../util/image_request'; +import {ResourceType} from '../util/request_manager'; +import {extend, isImageBitmap, readImageUsingVideoFrame} from '../util/util'; +import {Evented} from '../util/evented'; +import {browser} from '../util/browser'; +import {offscreenCanvasSupported} from '../util/offscreen_canvas_supported'; +import {OverscaledTileID} from './tile_id'; +import {RasterTileSource} from './raster_tile_source'; +// ensure DEMData is registered for worker transfer on main thread: +import '../data/dem_data'; +import type {DEMEncoding} from '../data/dem_data'; + +import type {Source} from './source'; +import type {Dispatcher} from '../util/dispatcher'; +import type {Tile} from './tile'; +import type {Callback} from '../types/callback'; +import type {RasterDEMSourceSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {ExpiryData} from '../util/ajax'; +import {isOffscreenCanvasDistorted} from '../util/offscreen_canvas_distorted'; +import {RGBAImage} from '../util/image'; + +/** + * A source containing raster DEM tiles (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.) + * This source can be used to show hillshading and 3D terrain + * + * @group Sources + * + * @example + * ```ts + * map.addSource('raster-dem-source', { + * type: 'raster-dem', + * url: 'https://demotiles.maplibre.org/terrain-tiles/tiles.json', + * tileSize: 256 + * }); + * ``` + * @see [3D Terrain](https://maplibre.org/maplibre-gl-js/docs/examples/3d-terrain/) + */ +export class RasterDEMTileSource extends RasterTileSource implements Source { + encoding: DEMEncoding; + redFactor?: number; + greenFactor?: number; + blueFactor?: number; + baseShift?: number; + + constructor(id: string, options: RasterDEMSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) { + super(id, options, dispatcher, eventedParent); + this.type = 'raster-dem'; + this.maxzoom = 22; + this._options = extend({type: 'raster-dem'}, options); + this.encoding = options.encoding || 'mapbox'; + this.redFactor = options.redFactor; + this.greenFactor = options.greenFactor; + this.blueFactor = options.blueFactor; + this.baseShift = options.baseShift; + } + + loadTile(tile: Tile, callback: Callback) { + const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme); + const request = this.map._requestManager.transformRequest(url, ResourceType.Tile); + tile.neighboringTiles = this._getNeighboringTiles(tile.tileID); + tile.request = ImageRequest.getImage(request, async (err: Error, img: (HTMLImageElement | ImageBitmap), expiry: ExpiryData) => { + delete tile.request; + if (tile.aborted) { + tile.state = 'unloaded'; + callback(null); + } else if (err) { + tile.state = 'errored'; + callback(err); + } else if (img) { + if (this.map._refreshExpiredTiles) tile.setExpiryData(expiry); + const transfer = isImageBitmap(img) && offscreenCanvasSupported(); + const rawImageData = transfer ? img : await readImageNow(img); + const params = { + uid: tile.uid, + coord: tile.tileID, + source: this.id, + rawImageData, + encoding: this.encoding, + redFactor: this.redFactor, + greenFactor: this.greenFactor, + blueFactor: this.blueFactor, + baseShift: this.baseShift + }; + + if (!tile.actor || tile.state === 'expired') { + tile.actor = this.dispatcher.getActor(); + tile.actor.send('loadDEMTile', params, done); + } + } + }, this.map._refreshExpiredTiles); + + async function readImageNow(img: ImageBitmap | HTMLImageElement): Promise { + if (typeof VideoFrame !== 'undefined' && isOffscreenCanvasDistorted()) { + const width = img.width + 2; + const height = img.height + 2; + try { + return new RGBAImage({width, height}, await readImageUsingVideoFrame(img, -1, -1, width, height)); + } catch (e) { + // fall-back to browser canvas decoding + } + } + return browser.getImageData(img, 1); + } + + function done(err, data) { + if (err) { + tile.state = 'errored'; + callback(err); + } + + if (data) { + tile.dem = data; + tile.needsHillshadePrepare = true; + tile.needsTerrainPrepare = true; + tile.state = 'loaded'; + callback(null); + } + } + } + + _getNeighboringTiles(tileID: OverscaledTileID) { + const canonical = tileID.canonical; + const dim = Math.pow(2, canonical.z); + + const px = (canonical.x - 1 + dim) % dim; + const pxw = canonical.x === 0 ? tileID.wrap - 1 : tileID.wrap; + const nx = (canonical.x + 1 + dim) % dim; + const nxw = canonical.x + 1 === dim ? tileID.wrap + 1 : tileID.wrap; + + const neighboringTiles = {}; + // add adjacent tiles + neighboringTiles[new OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y).key] = {backfilled: false}; + neighboringTiles[new OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y).key] = {backfilled: false}; + + // Add upper neighboringTiles + if (canonical.y > 0) { + neighboringTiles[new OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y - 1).key] = {backfilled: false}; + neighboringTiles[new OverscaledTileID(tileID.overscaledZ, tileID.wrap, canonical.z, canonical.x, canonical.y - 1).key] = {backfilled: false}; + neighboringTiles[new OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y - 1).key] = {backfilled: false}; + } + // Add lower neighboringTiles + if (canonical.y + 1 < dim) { + neighboringTiles[new OverscaledTileID(tileID.overscaledZ, pxw, canonical.z, px, canonical.y + 1).key] = {backfilled: false}; + neighboringTiles[new OverscaledTileID(tileID.overscaledZ, tileID.wrap, canonical.z, canonical.x, canonical.y + 1).key] = {backfilled: false}; + neighboringTiles[new OverscaledTileID(tileID.overscaledZ, nxw, canonical.z, nx, canonical.y + 1).key] = {backfilled: false}; + } + + return neighboringTiles; + } + + unloadTile(tile: Tile) { + if (tile.demTexture) this.map.painter.saveTileTexture(tile.demTexture); + if (tile.fbo) { + tile.fbo.destroy(); + delete tile.fbo; + } + if (tile.dem) delete tile.dem; + delete tile.neighboringTiles; + + tile.state = 'unloaded'; + if (tile.actor) { + tile.actor.send('removeDEMTile', {uid: tile.uid, source: this.id}); + } + } +} diff --git a/web/libraries/maplibre-gl/src/source/raster_dem_tile_worker_source.test.ts b/web/libraries/maplibre-gl/src/source/raster_dem_tile_worker_source.test.ts new file mode 100644 index 00000000..0641912f --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/raster_dem_tile_worker_source.test.ts @@ -0,0 +1,38 @@ +import {RasterDEMTileWorkerSource} from './raster_dem_tile_worker_source'; +import {DEMData} from '../data/dem_data'; +import {WorkerDEMTileParameters} from './worker_source'; + +describe('loadTile', () => { + test('loads DEM tile', done => { + const source = new RasterDEMTileWorkerSource(); + + source.loadTile({ + source: 'source', + uid: '0', + rawImageData: {data: new Uint8ClampedArray(256), height: 8, width: 8}, + dim: 256 + } as any as WorkerDEMTileParameters, (err, data) => { + if (err) done(err); + expect(Object.keys(source.loaded)).toEqual(['0']); + expect(data instanceof DEMData).toBeTruthy(); + done(); + }); + }); +}); + +describe('removeTile', () => { + test('removes loaded tile', () => { + const source = new RasterDEMTileWorkerSource(); + + source.loaded = { + '0': {} as DEMData + }; + + source.removeTile({ + source: 'source', + uid: '0' + }); + + expect(source.loaded).toEqual({}); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/raster_dem_tile_worker_source.ts b/web/libraries/maplibre-gl/src/source/raster_dem_tile_worker_source.ts new file mode 100644 index 00000000..d2838046 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/raster_dem_tile_worker_source.ts @@ -0,0 +1,39 @@ +import {DEMData} from '../data/dem_data'; +import {RGBAImage} from '../util/image'; +import type {Actor} from '../util/actor'; +import type { + WorkerDEMTileParameters, + WorkerDEMTileCallback, + TileParameters +} from './worker_source'; +import {getImageData, isImageBitmap} from '../util/util'; + +export class RasterDEMTileWorkerSource { + actor: Actor; + loaded: {[_: string]: DEMData}; + + constructor() { + this.loaded = {}; + } + + async loadTile(params: WorkerDEMTileParameters, callback: WorkerDEMTileCallback) { + const {uid, encoding, rawImageData, redFactor, greenFactor, blueFactor, baseShift} = params; + const width = rawImageData.width + 2; + const height = rawImageData.height + 2; + const imagePixels: RGBAImage = isImageBitmap(rawImageData) ? + new RGBAImage({width, height}, await getImageData(rawImageData, -1, -1, width, height)) : + rawImageData; + const dem = new DEMData(uid, imagePixels, encoding, redFactor, greenFactor, blueFactor, baseShift); + this.loaded = this.loaded || {}; + this.loaded[uid] = dem; + callback(null, dem); + } + + removeTile(params: TileParameters) { + const loaded = this.loaded, + uid = params.uid; + if (loaded && loaded[uid]) { + delete loaded[uid]; + } + } +} diff --git a/web/libraries/maplibre-gl/src/source/raster_tile_source.test.ts b/web/libraries/maplibre-gl/src/source/raster_tile_source.test.ts new file mode 100644 index 00000000..74f589fd --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/raster_tile_source.test.ts @@ -0,0 +1,197 @@ +import {RasterTileSource} from './raster_tile_source'; +import {OverscaledTileID} from './tile_id'; +import {RequestManager} from '../util/request_manager'; +import {Dispatcher} from '../util/dispatcher'; +import {fakeServer, type FakeServer} from 'nise'; +import {Tile} from './tile'; +import {stubAjaxGetImage} from '../util/test/util'; + +function createSource(options, transformCallback?) { + const source = new RasterTileSource('id', options, {send() {}} as any as Dispatcher, options.eventedParent); + source.onAdd({ + transform: {angle: 0, pitch: 0, showCollisionBoxes: false}, + _getMapId: () => 1, + _requestManager: new RequestManager(transformCallback), + getPixelRatio() { return 1; } + } as any); + + source.on('error', (e) => { + throw e.error; + }); + + return source; +} + +describe('RasterTileSource', () => { + let server: FakeServer; + beforeEach(() => { + global.fetch = null; + server = fakeServer.create(); + }); + + afterEach(() => { + server.restore(); + }); + + test('transforms request for TileJSON URL', () => { + server.respondWith('/source.json', JSON.stringify({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'], + bounds: [-47, -7, -45, -5] + })); + const transformSpy = jest.fn().mockImplementation((url) => { + return {url}; + }); + + createSource({url: '/source.json'}, transformSpy); + server.respond(); + + expect(transformSpy.mock.calls[0][0]).toBe('/source.json'); + expect(transformSpy.mock.calls[0][1]).toBe('Source'); + }); + + test('respects TileJSON.bounds', done => { + const source = createSource({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'], + bounds: [-47, -7, -45, -5] + }); + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(source.hasTile(new OverscaledTileID(8, 0, 8, 96, 132))).toBeFalsy(); + expect(source.hasTile(new OverscaledTileID(8, 0, 8, 95, 132))).toBeTruthy(); + done(); + } + }); + }); + + test('does not error on invalid bounds', done => { + const source = createSource({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'], + bounds: [-47, -7, -45, 91] + }); + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(source.tileBounds.bounds).toEqual({_sw: {lng: -47, lat: -7}, _ne: {lng: -45, lat: 90}}); + done(); + } + }); + }); + + test('respects TileJSON.bounds when loaded from TileJSON', done => { + server.respondWith('/source.json', JSON.stringify({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'], + bounds: [-47, -7, -45, -5] + })); + const source = createSource({url: '/source.json'}); + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(source.hasTile(new OverscaledTileID(8, 0, 8, 96, 132))).toBeFalsy(); + expect(source.hasTile(new OverscaledTileID(8, 0, 8, 95, 132))).toBeTruthy(); + done(); + } + }); + server.respond(); + }); + + test('transforms tile urls before requesting', done => { + server.respondWith('/source.json', JSON.stringify({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'], + bounds: [-47, -7, -45, -5] + })); + const source = createSource({url: '/source.json'}); + const transformSpy = jest.spyOn(source.map._requestManager, 'transformRequest'); + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + const tile = { + tileID: new OverscaledTileID(10, 0, 10, 5, 5), + state: 'loading', + loadVectorData () {}, + setExpiryData() {} + } as any as Tile; + source.loadTile(tile, () => {}); + expect(transformSpy).toHaveBeenCalledTimes(1); + expect(transformSpy.mock.calls[0][0]).toBe('http://example.com/10/5/5.png'); + expect(transformSpy.mock.calls[0][1]).toBe('Tile'); + done(); + } + }); + server.respond(); + }); + + test('HttpImageElement used to get image when refreshExpiredTiles is false', done => { + stubAjaxGetImage(undefined); + server.respondWith('/source.json', JSON.stringify({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'], + bounds: [-47, -7, -45, -5] + })); + const source = createSource({url: '/source.json'}); + source.map.painter = {context: {}, getTileTexture: () => { return {update: () => {}}; }} as any; + source.map._refreshExpiredTiles = false; + + const imageConstructorSpy = jest.spyOn(global, 'Image'); + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + const tile = { + tileID: new OverscaledTileID(10, 0, 10, 5, 5), + state: 'loading' + } as any as Tile; + source.loadTile(tile, () => { + expect(imageConstructorSpy).toHaveBeenCalledTimes(1); + expect(tile.state).toBe('loaded'); + done(); + }); + } + }); + server.respond(); + }); + + test('supports updating tiles', () => { + const source = createSource({url: '/source.json'}); + source.setTiles(['http://example.com/{z}/{x}/{y}.png?updated=true']); + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(source.tiles[0]).toBe('http://example.com/{z}/{x}/{y}.png?updated=true'); + } + }); + }); + + test('cancels TileJSON request if removed', () => { + const source = createSource({url: '/source.json'}); + source.onRemove(); + expect((server.requests.pop() as any).aborted).toBe(true); + }); + + it('serializes options', () => { + const source = createSource({ + tiles: ['http://localhost:2900/raster/{z}/{x}/{y}.png'], + minzoom: 2, + maxzoom: 10 + }); + expect(source.serialize()).toStrictEqual({ + type: 'raster', + tiles: ['http://localhost:2900/raster/{z}/{x}/{y}.png'], + minzoom: 2, + maxzoom: 10 + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/raster_tile_source.ts b/web/libraries/maplibre-gl/src/source/raster_tile_source.ts new file mode 100644 index 00000000..66630c50 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/raster_tile_source.ts @@ -0,0 +1,210 @@ +import {extend, pick} from '../util/util'; + +import {ImageRequest} from '../util/image_request'; + +import {ResourceType} from '../util/request_manager'; +import {Event, ErrorEvent, Evented} from '../util/evented'; +import {loadTileJson} from './load_tilejson'; +import {TileBounds} from './tile_bounds'; +import {Texture} from '../render/texture'; + +import type {Source} from './source'; +import type {OverscaledTileID} from './tile_id'; +import type {Map} from '../ui/map'; +import type {Dispatcher} from '../util/dispatcher'; +import type {Tile} from './tile'; +import type {Callback} from '../types/callback'; +import type {Cancelable} from '../types/cancelable'; +import type { + RasterSourceSpecification, + RasterDEMSourceSpecification +} from '@maplibre/maplibre-gl-style-spec'; + +/** + * A source containing raster tiles (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.) + * + * @group Sources + * + * @example + * ```ts + * map.addSource('raster-source', { + * 'type': 'raster', + * 'tiles': ['https://tiles.stadiamaps.com/tiles/stamen_watercolor/{z}/{x}/{y}.jpg'], + * 'tileSize': 256, + * }); + * ``` + * + * @example + * ```ts + * map.addSource('wms-test-source', { + * 'type': 'raster', + * // use the tiles option to specify a WMS tile source URL + * 'tiles': [ + * 'https://img.nj.gov/imagerywms/Natural2015?bbox={bbox-epsg-3857}&format=image/png&service=WMS&version=1.1.1&request=GetMap&srs=EPSG:3857&transparent=true&width=256&height=256&layers=Natural2015' + * ], + * 'tileSize': 256 + * }); + * ``` + * @see [Add a raster tile source](https://maplibre.org/maplibre-gl-js/docs/examples/map-tiles/) + * @see [Add a WMS source](https://maplibre.org/maplibre-gl-js/docs/examples/wms/) + * @see [Display a satellite map](https://maplibre.org/maplibre-gl-js/docs/examples/satellite-map/) + */ +export class RasterTileSource extends Evented implements Source { + type: 'raster' | 'raster-dem'; + id: string; + minzoom: number; + maxzoom: number; + url: string; + scheme: string; + tileSize: number; + + bounds: [number, number, number, number]; + tileBounds: TileBounds; + roundZoom: boolean; + dispatcher: Dispatcher; + map: Map; + tiles: Array; + + _loaded: boolean; + _options: RasterSourceSpecification | RasterDEMSourceSpecification; + _tileJSONRequest: Cancelable; + + constructor(id: string, options: RasterSourceSpecification | RasterDEMSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) { + super(); + this.id = id; + this.dispatcher = dispatcher; + this.setEventedParent(eventedParent); + + this.type = 'raster'; + this.minzoom = 0; + this.maxzoom = 22; + this.roundZoom = true; + this.scheme = 'xyz'; + this.tileSize = 512; + this._loaded = false; + + this._options = extend({type: 'raster'}, options); + extend(this, pick(options, ['url', 'scheme', 'tileSize'])); + } + + load() { + this._loaded = false; + this.fire(new Event('dataloading', {dataType: 'source'})); + this._tileJSONRequest = loadTileJson(this._options, this.map._requestManager, (err, tileJSON) => { + this._tileJSONRequest = null; + this._loaded = true; + if (err) { + this.fire(new ErrorEvent(err)); + } else if (tileJSON) { + extend(this, tileJSON); + if (tileJSON.bounds) this.tileBounds = new TileBounds(tileJSON.bounds, this.minzoom, this.maxzoom); + + // `content` is included here to prevent a race condition where `Style#_updateSources` is called + // before the TileJSON arrives. this makes sure the tiles needed are loaded once TileJSON arrives + // ref: https://github.com/mapbox/mapbox-gl-js/pull/4347#discussion_r104418088 + this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'})); + this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'})); + } + }); + } + + loaded(): boolean { + return this._loaded; + } + + onAdd(map: Map) { + this.map = map; + this.load(); + } + + onRemove() { + if (this._tileJSONRequest) { + this._tileJSONRequest.cancel(); + this._tileJSONRequest = null; + } + } + + setSourceProperty(callback: Function) { + if (this._tileJSONRequest) { + this._tileJSONRequest.cancel(); + } + + callback(); + + this.load(); + } + + /** + * Sets the source `tiles` property and re-renders the map. + * + * @param tiles - An array of one or more tile source URLs, as in the raster tiles spec (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) + * @returns `this` + */ + setTiles(tiles: Array): this { + this.setSourceProperty(() => { + this._options.tiles = tiles; + }); + + return this; + } + + serialize() { + return extend({}, this._options); + } + + hasTile(tileID: OverscaledTileID) { + return !this.tileBounds || this.tileBounds.contains(tileID.canonical); + } + + loadTile(tile: Tile, callback: Callback) { + const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme); + tile.request = ImageRequest.getImage(this.map._requestManager.transformRequest(url, ResourceType.Tile), (err, img, expiry) => { + delete tile.request; + + if (tile.aborted) { + tile.state = 'unloaded'; + callback(null); + } else if (err) { + tile.state = 'errored'; + callback(err); + } else if (img) { + if (this.map._refreshExpiredTiles && expiry) tile.setExpiryData(expiry); + + const context = this.map.painter.context; + const gl = context.gl; + tile.texture = this.map.painter.getTileTexture(img.width); + if (tile.texture) { + tile.texture.update(img, {useMipmap: true}); + } else { + tile.texture = new Texture(context, img, gl.RGBA, {useMipmap: true}); + tile.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE, gl.LINEAR_MIPMAP_NEAREST); + + if (context.extTextureFilterAnisotropic) { + gl.texParameterf(gl.TEXTURE_2D, context.extTextureFilterAnisotropic.TEXTURE_MAX_ANISOTROPY_EXT, context.extTextureFilterAnisotropicMax); + } + } + + tile.state = 'loaded'; + + callback(null); + } + }, this.map._refreshExpiredTiles); + } + + abortTile(tile: Tile, callback: Callback) { + if (tile.request) { + tile.request.cancel(); + delete tile.request; + } + callback(); + } + + unloadTile(tile: Tile, callback: Callback) { + if (tile.texture) this.map.painter.saveTileTexture(tile.texture); + callback(); + } + + hasTransition() { + return false; + } +} diff --git a/web/libraries/maplibre-gl/src/source/rtl_text_plugin.ts b/web/libraries/maplibre-gl/src/source/rtl_text_plugin.ts new file mode 100644 index 00000000..249919ca --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/rtl_text_plugin.ts @@ -0,0 +1,144 @@ +import {getArrayBuffer} from '../util/ajax'; +import {browser} from '../util/browser'; +import {Event, Evented} from '../util/evented'; +import {isWorker} from '../util/util'; + +const status = { + unavailable: 'unavailable', // Not loaded + deferred: 'deferred', // The plugin URL has been specified, but loading has been deferred + loading: 'loading', // request in-flight + loaded: 'loaded', + error: 'error' +}; + +export type PluginState = { + pluginStatus: typeof status[keyof typeof status]; + pluginURL: string; +}; + +/** + * An error callback + */ +type ErrorCallback = (error?: Error | null) => void; +type PluginStateSyncCallback = (state: PluginState) => void; +let _completionCallback = null; + +//Variables defining the current state of the plugin +let pluginStatus = status.unavailable; +let pluginURL = null; + +export const triggerPluginCompletionEvent = function(error: Error | string) { + // NetworkError's are not correctly reflected by the plugin status which prevents reloading plugin + if (error && typeof error === 'string' && error.indexOf('NetworkError') > -1) { + pluginStatus = status.error; + } + + if (_completionCallback) { + _completionCallback(error); + } +}; + +function sendPluginStateToWorker() { + evented.fire(new Event('pluginStateChange', {pluginStatus, pluginURL})); +} + +export const evented = new Evented(); + +export const getRTLTextPluginStatus = function () { + return pluginStatus; +}; + +export const registerForPluginStateChange = function(callback: PluginStateSyncCallback) { + // Do an initial sync of the state + callback({pluginStatus, pluginURL}); + // Listen for all future state changes + evented.on('pluginStateChange', callback); + return callback; +}; + +export const clearRTLTextPlugin = function() { + pluginStatus = status.unavailable; + pluginURL = null; + _completionCallback = null; +}; + +export const setRTLTextPlugin = function(url: string, callback: ErrorCallback, deferred: boolean = false) { + if (pluginStatus === status.deferred || pluginStatus === status.loading || pluginStatus === status.loaded) { + throw new Error('setRTLTextPlugin cannot be called multiple times.'); + } + pluginURL = browser.resolveURL(url); + pluginStatus = status.deferred; + _completionCallback = callback; + sendPluginStateToWorker(); + + //Start downloading the plugin immediately if not intending to lazy-load + if (!deferred) { + downloadRTLTextPlugin(); + } +}; + +export const downloadRTLTextPlugin = function() { + if (pluginStatus !== status.deferred || !pluginURL) { + throw new Error('rtl-text-plugin cannot be downloaded unless a pluginURL is specified'); + } + pluginStatus = status.loading; + sendPluginStateToWorker(); + if (pluginURL) { + getArrayBuffer({url: pluginURL}, (error) => { + if (error) { + triggerPluginCompletionEvent(error); + } else { + pluginStatus = status.loaded; + sendPluginStateToWorker(); + } + }); + } +}; + +export const plugin: { + applyArabicShaping: Function; + processBidirectionalText: ((b: string, a: Array) => Array); + processStyledBidirectionalText: ((c: string, b: Array, a: Array) => Array<[string, Array]>); + isLoaded: () => boolean; + isLoading: () => boolean; + setState: (state: PluginState) => void; + isParsed: () => boolean; + getPluginURL: () => string; +} = { + applyArabicShaping: null, + processBidirectionalText: null, + processStyledBidirectionalText: null, + isLoaded() { + return pluginStatus === status.loaded || // Main Thread: loaded if the completion callback returned successfully + plugin.applyArabicShaping != null; // Web-worker: loaded if the plugin functions have been compiled + }, + isLoading() { // Main Thread Only: query the loading status, this function does not return the correct value in the worker context. + return pluginStatus === status.loading; + }, + setState(state: PluginState) { // Worker thread only: this tells the worker threads that the plugin is available on the Main thread + if (!isWorker()) throw new Error('Cannot set the state of the rtl-text-plugin when not in the web-worker context'); + + pluginStatus = state.pluginStatus; + pluginURL = state.pluginURL; + }, + isParsed(): boolean { + if (!isWorker()) throw new Error('rtl-text-plugin is only parsed on the worker-threads'); + + return plugin.applyArabicShaping != null && + plugin.processBidirectionalText != null && + plugin.processStyledBidirectionalText != null; + }, + getPluginURL(): string { + if (!isWorker()) throw new Error('rtl-text-plugin url can only be queried from the worker threads'); + return pluginURL; + } +}; + +export const lazyLoadRTLTextPlugin = function() { + if (!plugin.isLoading() && + !plugin.isLoaded() && + getRTLTextPluginStatus() === 'deferred' + ) { + downloadRTLTextPlugin(); + } +}; diff --git a/web/libraries/maplibre-gl/src/source/source.ts b/web/libraries/maplibre-gl/src/source/source.ts new file mode 100644 index 00000000..34c5ce88 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/source.ts @@ -0,0 +1,139 @@ +import {VectorTileSource} from '../source/vector_tile_source'; +import {RasterTileSource} from '../source/raster_tile_source'; +import {RasterDEMTileSource} from '../source/raster_dem_tile_source'; +import {GeoJSONSource} from '../source/geojson_source'; +import {VideoSource} from '../source/video_source'; +import {ImageSource} from '../source/image_source'; +import {CanvasSource} from '../source/canvas_source'; + +import type {SourceSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {Dispatcher} from '../util/dispatcher'; +import type {Event, Evented} from '../util/evented'; +import type {Map} from '../ui/map'; +import type {Tile} from './tile'; +import type {OverscaledTileID, CanonicalTileID} from './tile_id'; +import type {Callback} from '../types/callback'; +import type {CanvasSourceSpecification} from '../source/canvas_source'; + +const registeredSources = {} as {[key:string]: SourceClass}; + +/** + * The `Source` interface must be implemented by each source type, including "core" types (`vector`, `raster`, + * `video`, etc.) and all custom, third-party types. + * + * @event `data` - Fired with `{dataType: 'source', sourceDataType: 'metadata'}` to indicate that any necessary metadata + * has been loaded so that it's okay to call `loadTile`; and with `{dataType: 'source', sourceDataType: 'content'}` + * to indicate that the source data has changed, so that any current caches should be flushed. + * + * @group Sources + */ +export interface Source { + readonly type: string; + /** + * The id for the source. Must not be used by any existing source. + */ + id: string; + minzoom: number; + maxzoom: number; + tileSize: number; + attribution?: string; + /** + * `true` if zoom levels are rounded to the nearest integer in the source data, `false` if they are floor-ed to the nearest integer. + */ + roundZoom?: boolean; + /** + * `false` if tiles can be drawn outside their boundaries, `true` if they cannot. + */ + isTileClipped?: boolean; + tileID?: CanonicalTileID; + /** + * `true` if tiles should be sent back to the worker for each overzoomed zoom level, `false` if not. + */ + reparseOverscaled?: boolean; + vectorLayerIds?: Array; + hasTransition(): boolean; + loaded(): boolean; + fire(event: Event): unknown; + readonly onAdd?: (map: Map) => void; + readonly onRemove?: (map: Map) => void; + loadTile(tile: Tile, callback: Callback): void; + readonly hasTile?: (tileID: OverscaledTileID) => boolean; + readonly abortTile?: (tile: Tile, callback: Callback) => void; + readonly unloadTile?: (tile: Tile, callback: Callback) => void; + /** + * @returns A plain (stringifiable) JS object representing the current state of the source. + * Creating a source using the returned object as the `options` should result in a Source that is + * equivalent to this one. + */ + serialize(): any; + readonly prepare?: () => void; +} + +/** + * A supporting type to the source definition + */ +type SourceStatics = { + /* + * An optional URL to a script which, when run by a Worker, registers a {@link WorkerSource} + * implementation for this Source type by calling `self.registerWorkerSource(workerSource: WorkerSource)`. + */ + workerSourceURL?: URL; +}; + +/** + * A general definition of a {@link Source} class for factory usage + */ +export type SourceClass = { + new (id: string, specification: SourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented): Source; +} & SourceStatics; + +/** + * Creates a tiled data source instance given an options object. + * + * @param id - The id for the source. Must not be used by any existing source. + * @param specification - Source options, specific to the source type (except for `options.type`, which is always required). + * @param source - A source definition object compliant with + * [`maplibre-gl-style-spec`](https://maplibre.org/maplibre-style-spec/#sources) or, for a third-party source type, + * with that type's requirements. + * @param dispatcher - A {@link Dispatcher} instance, which can be used to send messages to the workers. + * @returns a newly created source + */ +export const create = (id: string, specification: SourceSpecification | CanvasSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented): Source => { + + const Class = getSourceType(specification.type); + const source = new Class(id, specification, dispatcher, eventedParent); + + if (source.id !== id) { + throw new Error(`Expected Source id to be ${id} instead of ${source.id}`); + } + + return source; +}; + +export const getSourceType = (name: string): SourceClass => { + switch (name) { + case 'geojson': + return GeoJSONSource; + case 'image': + return ImageSource; + case 'raster': + return RasterTileSource; + case 'raster-dem': + return RasterDEMTileSource; + case 'vector': + return VectorTileSource; + case 'video': + return VideoSource; + case 'canvas': + return CanvasSource; + } + return registeredSources[name]; +}; + +export const setSourceType = (name: string, type: SourceClass) => { + registeredSources[name] = type; +}; + +export interface Actor { + send(type: string, data: any, callback: Callback): void; +} diff --git a/web/libraries/maplibre-gl/src/source/source_cache.test.ts b/web/libraries/maplibre-gl/src/source/source_cache.test.ts new file mode 100644 index 00000000..0bf9e8fa --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/source_cache.test.ts @@ -0,0 +1,1989 @@ +import {SourceCache} from './source_cache'; +import {setSourceType} from './source'; +import {Tile} from './tile'; +import {OverscaledTileID} from './tile_id'; +import {Transform} from '../geo/transform'; +import {LngLat} from '../geo/lng_lat'; +import Point from '@mapbox/point-geometry'; +import {Event, ErrorEvent, Evented} from '../util/evented'; +import {extend} from '../util/util'; +import {browser} from '../util/browser'; +import {Dispatcher} from '../util/dispatcher'; +import {Callback} from '../types/callback'; +import {TileBounds} from './tile_bounds'; + +class SourceMock extends Evented { + id: string; + minzoom: number; + maxzoom: number; + hasTile: (tileID: OverscaledTileID) => boolean; + sourceOptions: any; + + constructor(id: string, sourceOptions: any, _dispatcher, eventedParent: Evented) { + super(); + this.id = id; + this.minzoom = 0; + this.maxzoom = 22; + extend(this, sourceOptions); + this.sourceOptions = sourceOptions; + this.setEventedParent(eventedParent); + if (sourceOptions.hasTile) { + this.hasTile = sourceOptions.hasTile; + } + } + loadTile(tile: Tile, callback: Callback) { + if (this.sourceOptions.expires) { + tile.setExpiryData({ + expires: this.sourceOptions.expires + }); + } + setTimeout(callback, 0); + } + loaded() { + return true; + } + onAdd() { + if (this.sourceOptions.noLoad) return; + if (this.sourceOptions.error) { + this.fire(new ErrorEvent(this.sourceOptions.error)); + } else { + this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'})); + } + } + abortTile() {} + unloadTile() {} + serialize() {} +} + +// Add a mocked source type for use in these tests +function createSource(id: string, sourceOptions: any, _dispatcher: any, eventedParent: Evented) { + // allow tests to override mocked methods/properties by providing + // them in the source definition object that's given to Source.create() + const source = new SourceMock(id, sourceOptions, _dispatcher, eventedParent); + + return source; +} + +setSourceType('mock-source-type', createSource as any); + +function createSourceCache(options?, used?) { + const sc = new SourceCache('id', extend({ + tileSize: 512, + minzoom: 0, + maxzoom: 14, + type: 'mock-source-type' + }, options), {} as Dispatcher); + sc.used = typeof used === 'boolean' ? used : true; + return sc; +} + +afterEach(() => { + jest.clearAllMocks(); +}); + +describe('SourceCache#addTile', () => { + test('loads tile when uncached', done => { + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + const sourceCache = createSourceCache({ + loadTile(tile) { + expect(tile.tileID).toEqual(tileID); + expect(tile.uses).toBe(0); + done(); + } + }); + sourceCache.onAdd(undefined); + sourceCache._addTile(tileID); + }); + + test('adds tile when uncached', done => { + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + const sourceCache = createSourceCache({}).on('dataloading', (data) => { + expect(data.tile.tileID).toEqual(tileID); + expect(data.tile.uses).toBe(1); + done(); + }); + sourceCache.onAdd(undefined); + sourceCache._addTile(tileID); + }); + + test('updates feature state on added uncached tile', done => { + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + let updateFeaturesSpy; + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + sourceCache.on('data', () => { + expect(updateFeaturesSpy).toHaveBeenCalledTimes(1); + done(); + }); + updateFeaturesSpy = jest.spyOn(tile, 'setFeatureState'); + tile.state = 'loaded'; + callback(); + } + }); + sourceCache.onAdd(undefined); + sourceCache._addTile(tileID); + }); + + test('uses cached tile', () => { + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + let load = 0, + add = 0; + + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loaded'; + load++; + callback(); + } + }).on('dataloading', () => { add++; }); + + const tr = new Transform(); + tr.width = 512; + tr.height = 512; + sourceCache.updateCacheSize(tr); + sourceCache._addTile(tileID); + sourceCache._removeTile(tileID.key); + sourceCache._addTile(tileID); + + expect(load).toBe(1); + expect(add).toBe(1); + + }); + + test('updates feature state on cached tile', () => { + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loaded'; + callback(); + } + }); + + const tr = new Transform(); + tr.width = 512; + tr.height = 512; + sourceCache.updateCacheSize(tr); + + const tile = sourceCache._addTile(tileID); + const updateFeaturesSpy = jest.spyOn(tile, 'setFeatureState'); + + sourceCache._removeTile(tileID.key); + sourceCache._addTile(tileID); + + expect(updateFeaturesSpy).toHaveBeenCalledTimes(1); + + }); + + test('moves timers when adding tile from cache', () => { + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + const time = new Date(); + time.setSeconds(time.getSeconds() + 5); + + const sourceCache = createSourceCache(); + sourceCache._setTileReloadTimer = (id) => { + sourceCache._timers[id] = setTimeout(() => {}, 0); + }; + sourceCache._loadTile = (tile, callback) => { + tile.state = 'loaded'; + tile.getExpiryTimeout = () => 1000 * 60; + sourceCache._setTileReloadTimer(tileID.key, tile); + callback(); + }; + + const tr = new Transform(); + tr.width = 512; + tr.height = 512; + sourceCache.updateCacheSize(tr); + + const id = tileID.key; + expect(sourceCache._timers[id]).toBeFalsy(); + expect(sourceCache._cache.has(tileID)).toBeFalsy(); + + sourceCache._addTile(tileID); + + expect(sourceCache._timers[id]).toBeTruthy(); + expect(sourceCache._cache.has(tileID)).toBeFalsy(); + + sourceCache._removeTile(tileID.key); + + expect(sourceCache._timers[id]).toBeFalsy(); + expect(sourceCache._cache.has(tileID)).toBeTruthy(); + + sourceCache._addTile(tileID); + + expect(sourceCache._timers[id]).toBeTruthy(); + expect(sourceCache._cache.has(tileID)).toBeFalsy(); + + }); + + test('does not reuse wrapped tile', () => { + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + let load = 0, + add = 0; + + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loaded'; + load++; + callback(); + } + }).on('dataloading', () => { add++; }); + + const t1 = sourceCache._addTile(tileID); + const t2 = sourceCache._addTile(new OverscaledTileID(0, 1, 0, 0, 0)); + + expect(load).toBe(2); + expect(add).toBe(2); + expect(t1).not.toBe(t2); + + }); + + test('should load tiles with identical overscaled Z but different canonical Z', () => { + const sourceCache = createSourceCache(); + + const tileIDs = [ + new OverscaledTileID(1, 0, 0, 0, 0), + new OverscaledTileID(1, 0, 1, 0, 0), + new OverscaledTileID(1, 0, 1, 1, 0), + new OverscaledTileID(1, 0, 1, 0, 1), + new OverscaledTileID(1, 0, 1, 1, 1) + ]; + + for (let i = 0; i < tileIDs.length; i++) + sourceCache._addTile(tileIDs[i]); + + for (let i = 0; i < tileIDs.length; i++) { + const id = tileIDs[i]; + const key = id.key; + + expect(sourceCache._tiles[key]).toBeTruthy(); + expect(sourceCache._tiles[key].tileID).toEqual(id); + } + + }); +}); + +describe('SourceCache#removeTile', () => { + test('removes tile', done => { + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + const sourceCache = createSourceCache({}); + sourceCache._addTile(tileID); + sourceCache.on('data', () => { + sourceCache._removeTile(tileID.key); + expect(sourceCache._tiles[tileID.key]).toBeFalsy(); + done(); + }); + }); + + test('caches (does not unload) loaded tile', done => { + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + const sourceCache = createSourceCache({ + loadTile(tile) { + tile.state = 'loaded'; + }, + unloadTile() { + done('test failed: unloadTile has been called'); + } + }); + + const tr = new Transform(); + tr.width = 512; + tr.height = 512; + sourceCache.updateCacheSize(tr); + + sourceCache._addTile(tileID); + sourceCache._removeTile(tileID.key); + + done(); + }); + + test('aborts and unloads unfinished tile', () => { + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + let abort = 0, + unload = 0; + + const sourceCache = createSourceCache({ + abortTile(tile) { + expect(tile.tileID).toEqual(tileID); + abort++; + }, + unloadTile(tile) { + expect(tile.tileID).toEqual(tileID); + unload++; + } + }); + + sourceCache._addTile(tileID); + sourceCache._removeTile(tileID.key); + + expect(abort).toBe(1); + expect(unload).toBe(1); + + }); + + test('_tileLoaded after _removeTile skips tile.added', () => { + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.added = undefined; + sourceCache._removeTile(tileID.key); + callback(); + } + }); + sourceCache.map = {painter: {crossTileSymbolIndex: '', tileExtentVAO: {}}} as any; + + sourceCache._addTile(tileID); + }); + + test('fires dataabort event', async () => { + const sourceCache = createSourceCache({ + loadTile() { + // Do not call back in order to make sure the tile is removed before it is loaded. + } + }); + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + const tile = sourceCache._addTile(tileID); + const abortPromise = sourceCache.once('dataabort'); + sourceCache._removeTile(tileID.key); + const event = await abortPromise; + expect(event.dataType).toBe('source'); + expect(event.tile).toBe(tile); + expect(event.coord).toBe(tileID); + }); + + test('does not fire dataabort event when the tile has already been loaded', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loaded'; + callback(); + } + }); + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + sourceCache._addTile(tileID); + const onAbort = jest.fn(); + sourceCache.once('dataabort', onAbort); + sourceCache._removeTile(tileID.key); + expect(onAbort).toHaveBeenCalledTimes(0); + }); + + test('does not fire data event when the tile has already been aborted', () => { + const onData = jest.fn(); + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + sourceCache.once('dataabort', () => { + tile.state = 'loaded'; + callback(); + expect(onData).toHaveBeenCalledTimes(0); + }); + } + }); + sourceCache.once('data', onData); + const tileID = new OverscaledTileID(0, 0, 0, 0, 0); + sourceCache._addTile(tileID); + sourceCache._removeTile(tileID.key); + }); + +}); + +describe('SourceCache / Source lifecycle', () => { + test('does not fire load or change before source load event', done => { + const sourceCache = createSourceCache({noLoad: true}) + .on('data', () => done('test failed: data event fired')); + sourceCache.onAdd(undefined); + setTimeout(() => done(), 1); + }); + + test('forward load event', done => { + const sourceCache = createSourceCache({}).on('data', (e) => { + if (e.sourceDataType === 'metadata') done(); + }); + sourceCache.onAdd(undefined); + }); + + test('forward change event', done => { + const sourceCache = createSourceCache().on('data', (e) => { + if (e.sourceDataType === 'metadata') done(); + }); + sourceCache.onAdd(undefined); + sourceCache.getSource().fire(new Event('data')); + }); + + test('forward error event', done => { + const sourceCache = createSourceCache({error: 'Error loading source'}).on('error', (err) => { + expect(err.error).toBe('Error loading source'); + done(); + }); + sourceCache.onAdd(undefined); + }); + + test('suppress 404 errors', done => { + const sourceCache = createSourceCache({status: 404, message: 'Not found'}) + .on('error', () => done('test failed: error event fired')); + sourceCache.onAdd(undefined); + done(); + }); + + test('loaded() true after source error', done => { + const sourceCache = createSourceCache({error: 'Error loading source'}).on('error', () => { + expect(sourceCache.loaded()).toBeTruthy(); + done(); + }); + sourceCache.onAdd(undefined); + }); + + test('loaded() true after tile error', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 0; + const sourceCache = createSourceCache({ + loadTile (tile, callback) { + callback('error'); + } + }).on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'metadata') { + sourceCache.update(transform); + } + }).on('error', () => { + expect(sourceCache.loaded()).toBeTruthy(); + done(); + }); + + sourceCache.onAdd(undefined); + }); + + test('loaded() false after source begins loading following error', done => { + const sourceCache = createSourceCache({error: 'Error loading source'}).on('error', () => { + sourceCache.on('dataloading', () => { + expect(sourceCache.loaded()).toBeFalsy(); + done(); + }); + sourceCache.getSource().fire(new Event('dataloading')); + }); + + sourceCache.onAdd(undefined); + }); + + test('loaded() false when error occurs while source is not loaded', done => { + const sourceCache = createSourceCache({ + error: 'Error loading source', + + loaded() { + return false; + } + }).on('error', () => { + expect(sourceCache.loaded()).toBeFalsy(); + done(); + }); + + sourceCache.onAdd(undefined); + }); + + test('reloads tiles after a data event where source is updated', () => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 0; + + const expected = [new OverscaledTileID(0, 0, 0, 0, 0).key, new OverscaledTileID(0, 0, 0, 0, 0).key]; + expect.assertions(expected.length); + + const sourceCache = createSourceCache({ + loadTile (tile, callback) { + expect(tile.tileID.key).toBe(expected.shift()); + tile.loaded = true; + callback(); + } + }); + + sourceCache.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'metadata') { + sourceCache.update(transform); + sourceCache.getSource().fire(new Event('data', {dataType: 'source', sourceDataType: 'content'})); + } + }); + + sourceCache.onAdd(undefined); + }); + + test('does not reload errored tiles', () => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 1; + + const sourceCache = createSourceCache({ + loadTile (tile, callback) { + // this transform will try to load the four tiles at z1 and a single z0 tile + // we only expect _reloadTile to be called with the 'loaded' z0 tile + tile.state = tile.tileID.canonical.z === 1 ? 'errored' : 'loaded'; + callback(); + } + }); + + const reloadTileSpy = jest.spyOn(sourceCache, '_reloadTile'); + sourceCache.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'metadata') { + sourceCache.update(transform); + sourceCache.getSource().fire(new Event('data', {dataType: 'source', sourceDataType: 'content'})); + } + }); + sourceCache.onAdd(undefined); + // we expect the source cache to have five tiles, but only to have reloaded one + expect(Object.keys(sourceCache._tiles)).toHaveLength(5); + expect(reloadTileSpy).toHaveBeenCalledTimes(1); + + }); + +}); + +describe('SourceCache#update', () => { + test('loads no tiles if used is false', done => { + const transform = new Transform(); + transform.resize(512, 512); + transform.zoom = 0; + + const sourceCache = createSourceCache({}, false); + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + expect(sourceCache.getIds()).toEqual([]); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('loads covering tiles', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 0; + + const sourceCache = createSourceCache({}); + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + expect(sourceCache.getIds()).toEqual([new OverscaledTileID(0, 0, 0, 0, 0).key]); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('respects Source#hasTile method if it is present', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 1; + + const sourceCache = createSourceCache({ + hasTile: (coord) => (coord.canonical.x !== 0) + }); + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + expect(sourceCache.getIds().sort()).toEqual([ + new OverscaledTileID(1, 0, 1, 1, 0).key, + new OverscaledTileID(1, 0, 1, 1, 1).key + ].sort()); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('removes unused tiles', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 0; + + const sourceCache = createSourceCache({ + loadTile: (tile, callback) => { + tile.state = 'loaded'; + callback(null); + } + }); + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + expect(sourceCache.getIds()).toEqual([new OverscaledTileID(0, 0, 0, 0, 0).key]); + + transform.zoom = 1; + sourceCache.update(transform); + + expect(sourceCache.getIds()).toEqual([ + new OverscaledTileID(1, 0, 1, 1, 1).key, + new OverscaledTileID(1, 0, 1, 0, 1).key, + new OverscaledTileID(1, 0, 1, 1, 0).key, + new OverscaledTileID(1, 0, 1, 0, 0).key + ]); + done(); + } + }); + + sourceCache.onAdd(undefined); + }); + + test('retains parent tiles for pending children', done => { + const transform = new Transform(); + (transform as any)._test = 'retains'; + transform.resize(511, 511); + transform.zoom = 0; + + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = (tile.tileID.key === new OverscaledTileID(0, 0, 0, 0, 0).key) ? 'loaded' : 'loading'; + callback(); + } + }); + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + expect(sourceCache.getIds()).toEqual([new OverscaledTileID(0, 0, 0, 0, 0).key]); + + transform.zoom = 1; + sourceCache.update(transform); + + expect(sourceCache.getIds()).toEqual([ + new OverscaledTileID(0, 0, 0, 0, 0).key, + new OverscaledTileID(1, 0, 1, 1, 1).key, + new OverscaledTileID(1, 0, 1, 0, 1).key, + new OverscaledTileID(1, 0, 1, 1, 0).key, + new OverscaledTileID(1, 0, 1, 0, 0).key + ]); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('retains parent tiles for pending children (wrapped)', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 0; + transform.center = new LngLat(360, 0); + + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = (tile.tileID.key === new OverscaledTileID(0, 1, 0, 0, 0).key) ? 'loaded' : 'loading'; + callback(); + } + }); + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + expect(sourceCache.getIds()).toEqual([new OverscaledTileID(0, 1, 0, 0, 0).key]); + + transform.zoom = 1; + sourceCache.update(transform); + + expect(sourceCache.getIds()).toEqual([ + new OverscaledTileID(0, 1, 0, 0, 0).key, + new OverscaledTileID(1, 1, 1, 1, 1).key, + new OverscaledTileID(1, 1, 1, 0, 1).key, + new OverscaledTileID(1, 1, 1, 1, 0).key, + new OverscaledTileID(1, 1, 1, 0, 0).key + ]); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('retains covered child tiles while parent tile is fading in', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 2; + + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.timeAdded = Infinity; + tile.state = 'loaded'; + tile.registerFadeDuration(100); + callback(); + } + }); + + (sourceCache._source as any).type = 'raster'; + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + expect(sourceCache.getIds()).toEqual([ + new OverscaledTileID(2, 0, 2, 2, 2).key, + new OverscaledTileID(2, 0, 2, 1, 2).key, + new OverscaledTileID(2, 0, 2, 2, 1).key, + new OverscaledTileID(2, 0, 2, 1, 1).key + ]); + + transform.zoom = 0; + sourceCache.update(transform); + + expect(sourceCache.getRenderableIds()).toHaveLength(5); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('retains a parent tile for fading even if a tile is partially covered by children', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 0; + + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.timeAdded = Infinity; + tile.state = 'loaded'; + tile.registerFadeDuration(100); + callback(); + } + }); + + (sourceCache._source as any).type = 'raster'; + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + + transform.zoom = 2; + sourceCache.update(transform); + + transform.zoom = 1; + sourceCache.update(transform); + + expect(sourceCache._coveredTiles[(new OverscaledTileID(0, 0, 0, 0, 0).key)]).toBe(true); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('retain children for fading fadeEndTime is 0 (added but registerFadeDuration() is not called yet)', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 1; + + const sourceCache = createSourceCache({ + + // not setting fadeEndTime because class Tile default is 0, and need to be tested + loadTile(tile, callback) { + tile.timeAdded = Date.now(); + tile.state = 'loaded'; + callback(); + } + }); + + (sourceCache._source as any).type = 'raster'; + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + + transform.zoom = 0; + sourceCache.update(transform); + + expect(sourceCache.getRenderableIds()).toHaveLength(5); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('retains children when tile.fadeEndTime is in the future', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 1; + + const fadeTime = 100; + + const start = Date.now(); + let time = start; + jest.spyOn(browser, 'now').mockImplementation(() => time); + + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.timeAdded = browser.now(); + tile.state = 'loaded'; + tile.fadeEndTime = browser.now() + fadeTime; + callback(); + } + }); + + (sourceCache._source as any).type = 'raster'; + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + // load children + sourceCache.update(transform); + + transform.zoom = 0; + sourceCache.update(transform); + + expect(sourceCache.getRenderableIds()).toHaveLength(5); + + time = start + 98; + sourceCache.update(transform); + expect(sourceCache.getRenderableIds()).toHaveLength(5); + + time = start + fadeTime + 1; + sourceCache.update(transform); + expect(sourceCache.getRenderableIds()).toHaveLength(1); + done(); + } + }); + + sourceCache.onAdd(undefined); + }); + + test('retains overscaled loaded children', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 16; + + // use slightly offset center so that sort order is better defined + transform.center = new LngLat(-0.001, 0.001); + + const sourceCache = createSourceCache({ + reparseOverscaled: true, + loadTile(tile, callback) { + tile.state = tile.tileID.overscaledZ === 16 ? 'loaded' : 'loading'; + callback(); + } + }); + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + expect(sourceCache.getRenderableIds()).toEqual([ + new OverscaledTileID(16, 0, 14, 8192, 8192).key, + new OverscaledTileID(16, 0, 14, 8191, 8192).key, + new OverscaledTileID(16, 0, 14, 8192, 8191).key, + new OverscaledTileID(16, 0, 14, 8191, 8191).key + ]); + + transform.zoom = 15; + sourceCache.update(transform); + + expect(sourceCache.getRenderableIds()).toEqual([ + new OverscaledTileID(16, 0, 14, 8192, 8192).key, + new OverscaledTileID(16, 0, 14, 8191, 8192).key, + new OverscaledTileID(16, 0, 14, 8192, 8191).key, + new OverscaledTileID(16, 0, 14, 8191, 8191).key + ]); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('reassigns tiles for large jumps in longitude', done => { + + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 0; + + const sourceCache = createSourceCache({}); + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + transform.center = new LngLat(360, 0); + const tileID = new OverscaledTileID(0, 1, 0, 0, 0); + sourceCache.update(transform); + expect(sourceCache.getIds()).toEqual([tileID.key]); + const tile = sourceCache.getTile(tileID); + + transform.center = new LngLat(0, 0); + const wrappedTileID = new OverscaledTileID(0, 0, 0, 0, 0); + sourceCache.update(transform); + expect(sourceCache.getIds()).toEqual([wrappedTileID.key]); + expect(sourceCache.getTile(wrappedTileID)).toBe(tile); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + +}); + +describe('SourceCache#_updateRetainedTiles', () => { + + test('loads ideal tiles if they exist', () => { + const stateCache = {}; + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = stateCache[tile.tileID.key] || 'errored'; + callback(); + } + }); + + const getTileSpy = jest.spyOn(sourceCache, 'getTile'); + const idealTile = new OverscaledTileID(1, 0, 1, 1, 1); + stateCache[idealTile.key] = 'loaded'; + sourceCache._updateRetainedTiles([idealTile], 1); + expect(getTileSpy).not.toHaveBeenCalled(); + expect(sourceCache.getIds()).toEqual([idealTile.key]); + }); + + test('retains all loaded children ', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'errored'; + callback(); + } + }); + + const idealTile = new OverscaledTileID(3, 0, 3, 1, 2); + sourceCache._tiles[idealTile.key] = new Tile(idealTile, undefined); + sourceCache._tiles[idealTile.key].state = 'errored'; + + const loadedChildren = [ + new OverscaledTileID(4, 0, 4, 2, 4), + new OverscaledTileID(4, 0, 4, 3, 4), + new OverscaledTileID(4, 0, 4, 2, 5), + new OverscaledTileID(5, 0, 5, 6, 10), + new OverscaledTileID(5, 0, 5, 7, 10), + new OverscaledTileID(5, 0, 5, 6, 11), + new OverscaledTileID(5, 0, 5, 7, 11) + ]; + + for (const t of loadedChildren) { + sourceCache._tiles[t.key] = new Tile(t, undefined); + sourceCache._tiles[t.key].state = 'loaded'; + } + + const retained = sourceCache._updateRetainedTiles([idealTile], 3); + expect(Object.keys(retained).sort()).toEqual([ + // parents are requested because ideal ideal tile is not completely covered by + // loaded child tiles + new OverscaledTileID(0, 0, 0, 0, 0), + new OverscaledTileID(2, 0, 2, 0, 1), + new OverscaledTileID(1, 0, 1, 0, 0), + idealTile + ].concat(loadedChildren).map(t => t.key).sort()); + + }); + + test('adds parent tile if ideal tile errors and no child tiles are loaded', () => { + const stateCache = {}; + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = stateCache[tile.tileID.key] || 'errored'; + callback(); + } + }); + + jest.spyOn(sourceCache, '_addTile'); + const getTileSpy = jest.spyOn(sourceCache, 'getTile'); + + const idealTiles = [new OverscaledTileID(1, 0, 1, 1, 1), new OverscaledTileID(1, 0, 1, 0, 1)]; + stateCache[idealTiles[0].key] = 'loaded'; + const retained = sourceCache._updateRetainedTiles(idealTiles, 1); + expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([ + // when child tiles aren't found, check and request parent tile + new OverscaledTileID(0, 0, 0, 0, 0) + ]); + + // retained tiles include all ideal tiles and any parents that were loaded to cover + // non-existant tiles + expect(retained).toEqual({ + // 1/0/1 + '211': new OverscaledTileID(1, 0, 1, 0, 1), + // 1/1/1 + '311': new OverscaledTileID(1, 0, 1, 1, 1), + // parent + '000': new OverscaledTileID(0, 0, 0, 0, 0) + }); + }); + + test('don\'t use wrong parent tile', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'errored'; + callback(); + } + }); + + const idealTile = new OverscaledTileID(2, 0, 2, 0, 0); + sourceCache._tiles[idealTile.key] = new Tile(idealTile, undefined); + sourceCache._tiles[idealTile.key].state = 'errored'; + + sourceCache._tiles[new OverscaledTileID(1, 0, 1, 1, 0).key] = new Tile(new OverscaledTileID(1, 0, 1, 1, 0), undefined); + sourceCache._tiles[new OverscaledTileID(1, 0, 1, 1, 0).key].state = 'loaded'; + + const addTileSpy = jest.spyOn(sourceCache, '_addTile'); + const getTileSpy = jest.spyOn(sourceCache, 'getTile'); + + sourceCache._updateRetainedTiles([idealTile], 2); + expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([ + // parents + new OverscaledTileID(1, 0, 1, 0, 0), // not found + new OverscaledTileID(0, 0, 0, 0, 0) // not found + ]); + + expect(addTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([ + // ideal tile + new OverscaledTileID(2, 0, 2, 0, 0), + // parents + new OverscaledTileID(1, 0, 1, 0, 0), // not found + new OverscaledTileID(0, 0, 0, 0, 0) // not found + ]); + }); + + test('use parent tile when ideal tile is not loaded', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loading'; + callback(); + } + }); + const idealTile = new OverscaledTileID(1, 0, 1, 0, 1); + const parentTile = new OverscaledTileID(0, 0, 0, 0, 0); + sourceCache._tiles[idealTile.key] = new Tile(idealTile, undefined); + sourceCache._tiles[idealTile.key].state = 'loading'; + sourceCache._tiles[parentTile.key] = new Tile(parentTile, undefined); + sourceCache._tiles[parentTile.key].state = 'loaded'; + + const addTileSpy = jest.spyOn(sourceCache, '_addTile'); + const getTileSpy = jest.spyOn(sourceCache, 'getTile'); + + const retained = sourceCache._updateRetainedTiles([idealTile], 1); + + expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([ + // parents + new OverscaledTileID(0, 0, 0, 0, 0), // found + ]); + + expect(retained).toEqual({ + // parent of ideal tile 0/0/0 + '000': new OverscaledTileID(0, 0, 0, 0, 0), + // ideal tile id 1/0/1 + '211': new OverscaledTileID(1, 0, 1, 0, 1) + }); + + addTileSpy.mockClear(); + getTileSpy.mockClear(); + + // now make sure we don't retain the parent tile when the ideal tile is loaded + sourceCache._tiles[idealTile.key].state = 'loaded'; + const retainedLoaded = sourceCache._updateRetainedTiles([idealTile], 1); + + expect(getTileSpy).not.toHaveBeenCalled(); + expect(retainedLoaded).toEqual({ + // only ideal tile retained + '211': new OverscaledTileID(1, 0, 1, 0, 1) + }); + }); + + test('don\'t load parent if all immediate children are loaded', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loading'; + callback(); + } + }); + + const idealTile = new OverscaledTileID(2, 0, 2, 1, 1); + const loadedTiles = [new OverscaledTileID(3, 0, 3, 2, 2), new OverscaledTileID(3, 0, 3, 3, 2), new OverscaledTileID(3, 0, 3, 2, 3), new OverscaledTileID(3, 0, 3, 3, 3)]; + loadedTiles.forEach(t => { + sourceCache._tiles[t.key] = new Tile(t, undefined); + sourceCache._tiles[t.key].state = 'loaded'; + }); + + const getTileSpy = jest.spyOn(sourceCache, 'getTile'); + const retained = sourceCache._updateRetainedTiles([idealTile], 2); + // parent tile isn't requested because all covering children are loaded + expect(getTileSpy).not.toHaveBeenCalled(); + expect(Object.keys(retained)).toEqual([idealTile.key].concat(loadedTiles.map(t => t.key))); + }); + + test('prefer loaded child tiles to parent tiles', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loading'; + callback(); + } + }); + const idealTile = new OverscaledTileID(1, 0, 1, 0, 0); + const loadedTiles = [new OverscaledTileID(0, 0, 0, 0, 0), new OverscaledTileID(2, 0, 2, 0, 0)]; + loadedTiles.forEach(t => { + sourceCache._tiles[t.key] = new Tile(t, undefined); + sourceCache._tiles[t.key].state = 'loaded'; + }); + + const getTileSpy = jest.spyOn(sourceCache, 'getTile'); + let retained = sourceCache._updateRetainedTiles([idealTile], 1); + expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([ + // parent + new OverscaledTileID(0, 0, 0, 0, 0) + ]); + + expect(retained).toEqual({ + // parent of ideal tile (0, 0, 0) (only partially covered by loaded child + // tiles, so we still need to load the parent) + '000': new OverscaledTileID(0, 0, 0, 0, 0), + // ideal tile id (1, 0, 0) + '011': new OverscaledTileID(1, 0, 1, 0, 0), + // loaded child tile (2, 0, 0) + '022': new OverscaledTileID(2, 0, 2, 0, 0) + }); + + getTileSpy.mockClear(); + // remove child tile and check that it only uses parent tile + delete sourceCache._tiles['022']; + retained = sourceCache._updateRetainedTiles([idealTile], 1); + + expect(retained).toEqual({ + // parent of ideal tile (0, 0, 0) (only partially covered by loaded child + // tiles, so we still need to load the parent) + '000': new OverscaledTileID(0, 0, 0, 0, 0), + // ideal tile id (1, 0, 0) + '011': new OverscaledTileID(1, 0, 1, 0, 0) + }); + + }); + + test('don\'t use tiles below minzoom', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loading'; + callback(); + }, + minzoom: 2 + }); + const idealTile = new OverscaledTileID(2, 0, 2, 0, 0); + const loadedTiles = [new OverscaledTileID(1, 0, 1, 0, 0)]; + loadedTiles.forEach(t => { + sourceCache._tiles[t.key] = new Tile(t, undefined); + sourceCache._tiles[t.key].state = 'loaded'; + }); + + const getTileSpy = jest.spyOn(sourceCache, 'getTile'); + const retained = sourceCache._updateRetainedTiles([idealTile], 2); + + expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([]); + + expect(retained).toEqual({ + // ideal tile id (2, 0, 0) + '022': new OverscaledTileID(2, 0, 2, 0, 0) + }); + + }); + + test('use overzoomed tile above maxzoom', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loading'; + callback(); + }, + maxzoom: 2 + }); + const idealTile = new OverscaledTileID(2, 0, 2, 0, 0); + + const getTileSpy = jest.spyOn(sourceCache, 'getTile'); + const retained = sourceCache._updateRetainedTiles([idealTile], 2); + + expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([ + // overzoomed child + new OverscaledTileID(3, 0, 2, 0, 0), + // parents + new OverscaledTileID(1, 0, 1, 0, 0), + new OverscaledTileID(0, 0, 0, 0, 0) + ]); + + expect(retained).toEqual({ + // ideal tile id (2, 0, 0) + '022': new OverscaledTileID(2, 0, 2, 0, 0) + }); + + }); + + test('dont\'t ascend multiple times if a tile is not found', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loading'; + callback(); + } + }); + const idealTiles = [new OverscaledTileID(8, 0, 8, 0, 0), new OverscaledTileID(8, 0, 8, 1, 0)]; + + const getTileSpy = jest.spyOn(sourceCache, 'getTile'); + sourceCache._updateRetainedTiles(idealTiles, 8); + expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([ + // parent tile ascent + new OverscaledTileID(7, 0, 7, 0, 0), + new OverscaledTileID(6, 0, 6, 0, 0), + new OverscaledTileID(5, 0, 5, 0, 0), + new OverscaledTileID(4, 0, 4, 0, 0), + new OverscaledTileID(3, 0, 3, 0, 0), + new OverscaledTileID(2, 0, 2, 0, 0), + new OverscaledTileID(1, 0, 1, 0, 0), + new OverscaledTileID(0, 0, 0, 0, 0), + ]); + + getTileSpy.mockClear(); + + const loadedTiles = [new OverscaledTileID(4, 0, 4, 0, 0)]; + loadedTiles.forEach(t => { + sourceCache._tiles[t.key] = new Tile(t, undefined); + sourceCache._tiles[t.key].state = 'loaded'; + }); + + sourceCache._updateRetainedTiles(idealTiles, 8); + expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([ + // parent tile ascent + new OverscaledTileID(7, 0, 7, 0, 0), + new OverscaledTileID(6, 0, 6, 0, 0), + new OverscaledTileID(5, 0, 5, 0, 0), + new OverscaledTileID(4, 0, 4, 0, 0), // tile is loaded, stops ascent + ]); + + }); + + test('Only retain loaded parent tile when zooming in', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loading'; + callback(); + } + }); + + let idealTiles = [new OverscaledTileID(9, 0, 9, 0, 0), new OverscaledTileID(9, 0, 9, 1, 0)]; + let retained = sourceCache._updateRetainedTiles(idealTiles, 9); + // Parent loading tiles from z=8 not retained + expect(Object.keys(retained).sort()).toEqual( + idealTiles.map((tile) => tile.key).sort() + ); + + idealTiles = [new OverscaledTileID(10, 0, 10, 0, 0), new OverscaledTileID(10, 0, 10, 1, 0)]; + retained = sourceCache._updateRetainedTiles(idealTiles, 10); + // Parent loading tiles from z=9 not retained + expect(Object.keys(retained).sort()).toEqual( + idealTiles.map((tile) => tile.key).sort() + ); + + const loadedTiles = idealTiles; + loadedTiles.forEach(t => { + sourceCache._tiles[t.key] = new Tile(t, undefined); + sourceCache._tiles[t.key].state = 'loaded'; + }); + + idealTiles = [new OverscaledTileID(11, 0, 11, 0, 0), new OverscaledTileID(11, 0, 11, 1, 0)]; + retained = sourceCache._updateRetainedTiles(idealTiles, 11); + // Parent loaded tile in the view port from z=10 was retained + expect(Object.keys(retained).sort()).toEqual([ + new OverscaledTileID(10, 0, 10, 0, 0).key, // Parent loaded tile + new OverscaledTileID(11, 0, 11, 0, 0).key, + new OverscaledTileID(11, 0, 11, 1, 0).key + ].sort()); + + }); + + test('Only retain loaded child tile when zooming out', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loading'; + callback(); + } + }); + + let idealTiles = [new OverscaledTileID(7, 0, 7, 0, 0), new OverscaledTileID(7, 0, 7, 1, 0)]; + let retained = sourceCache._updateRetainedTiles(idealTiles, 7); + // Client tiles from z=6 not retained + expect(Object.keys(retained).sort()).toEqual( + idealTiles.map((tile) => tile.key).sort() + ); + + idealTiles = [new OverscaledTileID(6, 0, 6, 0, 0), new OverscaledTileID(6, 0, 6, 1, 0)]; + retained = sourceCache._updateRetainedTiles(idealTiles, 6); + // Client tiles from z=6 not retained + expect(Object.keys(retained).sort()).toEqual( + idealTiles.map((tile) => tile.key).sort() + ); + + const loadedTiles = idealTiles; + loadedTiles.forEach(t => { + sourceCache._tiles[t.key] = new Tile(t, undefined); + sourceCache._tiles[t.key].state = 'loaded'; + }); + + idealTiles = [new OverscaledTileID(5, 0, 5, 0, 0), new OverscaledTileID(5, 0, 5, 1, 0)]; + retained = sourceCache._updateRetainedTiles(idealTiles, 5); + // Child loaded tile in the view port from z=6 was retained + expect(Object.keys(retained).sort()).toEqual([ + new OverscaledTileID(6, 0, 6, 0, 0).key, + new OverscaledTileID(6, 0, 6, 1, 0).key, + new OverscaledTileID(5, 0, 5, 0, 0).key, + new OverscaledTileID(5, 0, 5, 1, 0).key + ].sort()); + }); + + test('adds correct loaded parent tiles for overzoomed tiles', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loading'; + callback(); + }, + maxzoom: 7 + }); + const loadedTiles = [new OverscaledTileID(7, 0, 7, 0, 0), new OverscaledTileID(7, 0, 7, 1, 0)]; + loadedTiles.forEach(t => { + sourceCache._tiles[t.key] = new Tile(t, undefined); + sourceCache._tiles[t.key].state = 'loaded'; + }); + + const idealTiles = [new OverscaledTileID(8, 0, 7, 0, 0), new OverscaledTileID(8, 0, 7, 1, 0)]; + const retained = sourceCache._updateRetainedTiles(idealTiles, 8); + + expect(Object.keys(retained)).toEqual([ + new OverscaledTileID(7, 0, 7, 1, 0).key, + new OverscaledTileID(8, 0, 7, 1, 0).key, + new OverscaledTileID(8, 0, 7, 0, 0).key, + new OverscaledTileID(7, 0, 7, 0, 0).key + ]); + + }); + +}); + +describe('SourceCache#clearTiles', () => { + test('unloads tiles', () => { + const coord = new OverscaledTileID(0, 0, 0, 0, 0); + let abort = 0, + unload = 0; + + const sourceCache = createSourceCache({ + abortTile(tile) { + expect(tile.tileID).toEqual(coord); + abort++; + }, + unloadTile(tile) { + expect(tile.tileID).toEqual(coord); + unload++; + } + }); + sourceCache.onAdd(undefined); + + sourceCache._addTile(coord); + sourceCache.clearTiles(); + + expect(abort).toBe(1); + expect(unload).toBe(1); + + }); +}); + +describe('SourceCache#tilesIn', () => { + test('graceful response before source loaded', () => { + const tr = new Transform(); + tr.width = 512; + tr.height = 512; + tr._calcMatrices(); + const sourceCache = createSourceCache({noLoad: true}); + sourceCache.transform = tr; + sourceCache.onAdd(undefined); + expect(sourceCache.tilesIn([ + new Point(0, 0), + new Point(512, 256) + ], 10, true)).toEqual([]); + + }); + + function round(queryGeometry) { + return queryGeometry.map((p) => { + return p.round(); + }); + } + + test('regular tiles', done => { + const transform = new Transform(); + transform.resize(512, 512); + transform.zoom = 1; + transform.center = new LngLat(0, 1); + + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loaded'; + tile.additionalRadius = 0; + callback(); + } + }); + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + + expect(sourceCache.getIds()).toEqual([ + new OverscaledTileID(1, 0, 1, 1, 1).key, + new OverscaledTileID(1, 0, 1, 0, 1).key, + new OverscaledTileID(1, 0, 1, 1, 0).key, + new OverscaledTileID(1, 0, 1, 0, 0).key + ]); + + transform._calcMatrices(); + const tiles = sourceCache.tilesIn([ + new Point(0, 0), + new Point(512, 256) + ], 1, true); + + tiles.sort((a, b) => { return a.tile.tileID.canonical.x - b.tile.tileID.canonical.x; }); + tiles.forEach((result) => { delete result.tile.uid; }); + + expect(tiles[0].tile.tileID.key).toBe('011'); + expect(tiles[0].tile.tileSize).toBe(512); + expect(tiles[0].scale).toBe(1); + expect(round(tiles[0].queryGeometry)).toEqual([{x: 4096, y: 4050}, {x: 12288, y: 8146}]); + + expect(tiles[1].tile.tileID.key).toBe('111'); + expect(tiles[1].tile.tileSize).toBe(512); + expect(tiles[1].scale).toBe(1); + expect(round(tiles[1].queryGeometry)).toEqual([{x: -4096, y: 4050}, {x: 4096, y: 8146}]); + + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('reparsed overscaled tiles', () => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loaded'; + tile.additionalRadius = 0; + callback(); + }, + reparseOverscaled: true, + minzoom: 1, + maxzoom: 1, + tileSize: 512 + }); + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + const transform = new Transform(); + transform.resize(1024, 1024); + transform.zoom = 2.0; + transform.center = new LngLat(0, 1); + sourceCache.update(transform); + + expect(sourceCache.getIds()).toEqual([ + new OverscaledTileID(2, 0, 1, 1, 1).key, + new OverscaledTileID(2, 0, 1, 0, 1).key, + new OverscaledTileID(2, 0, 1, 1, 0).key, + new OverscaledTileID(2, 0, 1, 0, 0).key + ]); + + const tiles = sourceCache.tilesIn([ + new Point(0, 0), + new Point(1024, 512) + ], 1, true); + + tiles.sort((a, b) => { return a.tile.tileID.canonical.x - b.tile.tileID.canonical.x; }); + tiles.forEach((result) => { delete result.tile.uid; }); + + expect(tiles[0].tile.tileID.key).toBe('012'); + expect(tiles[0].tile.tileSize).toBe(1024); + expect(tiles[0].scale).toBe(1); + expect(round(tiles[0].queryGeometry)).toEqual([{x: 4096, y: 4050}, {x: 12288, y: 8146}]); + + expect(tiles[1].tile.tileID.key).toBe('112'); + expect(tiles[1].tile.tileSize).toBe(1024); + expect(tiles[1].scale).toBe(1); + expect(round(tiles[1].queryGeometry)).toEqual([{x: -4096, y: 4050}, {x: 4096, y: 8146}]); + + } + }); + sourceCache.onAdd(undefined); + }); + + test('overscaled tiles', done => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { tile.state = 'loaded'; callback(); }, + reparseOverscaled: false, + minzoom: 1, + maxzoom: 1, + tileSize: 512 + }); + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + const transform = new Transform(); + transform.resize(512, 512); + transform.zoom = 2.0; + sourceCache.update(transform); + + done(); + } + }); + sourceCache.onAdd(undefined); + }); +}); + +describe('source cache loaded', () => { + test('SourceCache#loaded (no errors)', done => { + const sourceCache = createSourceCache({ + loadTile(tile, callback) { + tile.state = 'loaded'; + callback(); + } + }); + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + const tr = new Transform(); + tr.resize(512, 512); + sourceCache.update(tr); + + const coord = new OverscaledTileID(0, 0, 0, 0, 0); + sourceCache._addTile(coord); + + expect(sourceCache.loaded()).toBeTruthy(); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('SourceCache#loaded (with errors)', done => { + const sourceCache = createSourceCache({ + loadTile(tile) { + tile.state = 'errored'; + } + }); + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + const tr = new Transform(); + tr.resize(512, 512); + sourceCache.update(tr); + + const coord = new OverscaledTileID(0, 0, 0, 0, 0); + sourceCache._addTile(coord); + + expect(sourceCache.loaded()).toBeTruthy(); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('SourceCache#loaded (unused)', done => { + const sourceCache = createSourceCache({ + loadTile(tile) { + tile.state = 'errored'; + } + }, false); + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(sourceCache.loaded()).toBeTruthy(); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('SourceCache#loaded (unusedForTerrain)', done => { + const sourceCache = createSourceCache({ + loadTile(tile) { + tile.state = 'errored'; + } + }, false); + sourceCache.usedForTerrain = false; + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(sourceCache.loaded()).toBeTruthy(); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('SourceCache#loaded (not loaded when no update)', done => { + const sourceCache = createSourceCache({ + loadTile(tile) { + tile.state = 'errored'; + } + }); + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(sourceCache.loaded()).toBeFalsy(); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('SourceCache#loaded (on last tile load)', done => { + const sourceCache = createSourceCache({ + hasTile(tileID: OverscaledTileID) { + return !this.tileBounds || this.tileBounds.contains(tileID.canonical); + }, + loadTile(tile, callback) { + tile.state = 'loading'; + setTimeout(() => { + tile.state = 'loaded'; + callback(); + }); + } + }); + + const tr = new Transform(); + tr.zoom = 10; + tr.resize(512, 512); + const expectedTilesLoaded = 4; + let loaded = 0; + + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(sourceCache.loaded()).toBeFalsy(); + } else if (e.tile !== undefined) { + loaded += 1; + if (sourceCache.loaded()) { + expect(loaded).toBe(expectedTilesLoaded); + done(); + } + } + }); + + sourceCache.onAdd(undefined); + sourceCache.update(tr); + }); + + test('SourceCache#loaded (tiles outside bounds, idle)', done => { + const japan = new TileBounds([122.74, 19.33, 149.0, 45.67]); + const sourceCache = createSourceCache({ + onAdd() { + if (this.sourceOptions.noLoad) return; + if (this.sourceOptions.error) { + this.fire(new ErrorEvent(this.sourceOptions.error)); + } else { + this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'})); + this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'})); + } + }, + hasTile(tileID: OverscaledTileID) { + return japan.contains(tileID.canonical); + }, + loadTile(tile, callback) { + tile.state = 'loading'; + setTimeout(() => { + tile.state = 'loaded'; + callback(); + }); + } + }); + + sourceCache.on('data', (e) => { + if (e.sourceDataType !== 'idle') { + expect(sourceCache.loaded()).toBeFalsy(); + // 'idle' emission when source bounds are outside of viewport bounds + } else { + expect(sourceCache.loaded()).toBeTruthy(); + done(); + } + }); + + sourceCache.onAdd(undefined); + const tr = new Transform(); + tr.zoom = 10; + tr.resize(512, 512); + sourceCache.update(tr); + }); +}); + +describe('souce cache get ids', () => { + test('SourceCache#getIds (ascending order by zoom level)', done => { + const ids = [ + new OverscaledTileID(0, 0, 0, 0, 0), + new OverscaledTileID(3, 0, 3, 0, 0), + new OverscaledTileID(1, 0, 1, 0, 0), + new OverscaledTileID(2, 0, 2, 0, 0) + ]; + + const sourceCache = createSourceCache({}); + sourceCache.transform = new Transform(); + for (let i = 0; i < ids.length; i++) { + sourceCache._tiles[ids[i].key] = {tileID: ids[i]} as any as Tile; + } + expect(sourceCache.getIds()).toEqual([ + new OverscaledTileID(0, 0, 0, 0, 0).key, + new OverscaledTileID(1, 0, 1, 0, 0).key, + new OverscaledTileID(2, 0, 2, 0, 0).key, + new OverscaledTileID(3, 0, 3, 0, 0).key + ]); + done(); + }); +}); + +describe('SourceCache#findLoadedParent', () => { + + test('adds from previously used tiles (sourceCache._tiles)', () => { + const sourceCache = createSourceCache({}); + sourceCache.onAdd(undefined); + const tr = new Transform(); + tr.width = 512; + tr.height = 512; + sourceCache.updateCacheSize(tr); + + const tile = { + tileID: new OverscaledTileID(1, 0, 1, 0, 0), + hasData() { return true; } + } as any as Tile; + + sourceCache._tiles[tile.tileID.key] = tile; + + expect(sourceCache.findLoadedParent(new OverscaledTileID(2, 0, 2, 3, 3), 0)).toBeUndefined(); + expect(sourceCache.findLoadedParent(new OverscaledTileID(2, 0, 2, 0, 0), 0)).toEqual(tile); + }); + + test('retains parents', () => { + const sourceCache = createSourceCache({}); + sourceCache.onAdd(undefined); + const tr = new Transform(); + tr.width = 512; + tr.height = 512; + sourceCache.updateCacheSize(tr); + + const tile = new Tile(new OverscaledTileID(1, 0, 1, 0, 0), 512); + sourceCache._cache.add(tile.tileID, tile); + + expect(sourceCache.findLoadedParent(new OverscaledTileID(2, 0, 2, 3, 3), 0)).toBeUndefined(); + expect(sourceCache.findLoadedParent(new OverscaledTileID(2, 0, 2, 0, 0), 0)).toBe(tile); + expect(sourceCache._cache.order).toHaveLength(1); + + }); + + test('Search cache for loaded parent tiles', () => { + const sourceCache = createSourceCache({}); + sourceCache.onAdd(undefined); + const tr = new Transform(); + tr.width = 512; + tr.height = 512; + sourceCache.updateCacheSize(tr); + + const mockTile = id => { + const tile = { + tileID: id, + hasData() { return true; } + } as any as Tile; + sourceCache._tiles[id.key] = tile; + }; + + const tiles = [ + new OverscaledTileID(0, 0, 0, 0, 0), + new OverscaledTileID(1, 0, 1, 1, 0), + new OverscaledTileID(2, 0, 2, 0, 0), + new OverscaledTileID(2, 0, 2, 1, 0), + new OverscaledTileID(2, 0, 2, 2, 0), + new OverscaledTileID(2, 0, 2, 1, 2) + ]; + + tiles.forEach(t => mockTile(t)); + sourceCache._updateLoadedParentTileCache(); + + // Loaded tiles excluding the root should be in the cache + expect(sourceCache.findLoadedParent(tiles[0], 0)).toBeUndefined(); + expect(sourceCache.findLoadedParent(tiles[1], 0).tileID).toBe(tiles[0]); + expect(sourceCache.findLoadedParent(tiles[2], 0).tileID).toBe(tiles[0]); + expect(sourceCache.findLoadedParent(tiles[3], 0).tileID).toBe(tiles[0]); + expect(sourceCache.findLoadedParent(tiles[4], 0).tileID).toBe(tiles[1]); + expect(sourceCache.findLoadedParent(tiles[5], 0).tileID).toBe(tiles[0]); + + expect(tiles[0].key in sourceCache._loadedParentTiles).toBe(false); + expect(tiles[1].key in sourceCache._loadedParentTiles).toBe(true); + expect(tiles[2].key in sourceCache._loadedParentTiles).toBe(true); + expect(tiles[3].key in sourceCache._loadedParentTiles).toBe(true); + expect(tiles[4].key in sourceCache._loadedParentTiles).toBe(true); + expect(tiles[5].key in sourceCache._loadedParentTiles).toBe(true); + + // Arbitrary tiles should not in the cache + const notLoadedTiles = [ + new OverscaledTileID(2, 1, 2, 0, 0), + new OverscaledTileID(2, 0, 2, 3, 0), + new OverscaledTileID(2, 0, 2, 3, 3), + new OverscaledTileID(3, 0, 3, 2, 1) + ]; + + expect(sourceCache.findLoadedParent(notLoadedTiles[0], 0)).toBeUndefined(); + expect(sourceCache.findLoadedParent(notLoadedTiles[1], 0).tileID).toBe(tiles[1]); + expect(sourceCache.findLoadedParent(notLoadedTiles[2], 0).tileID).toBe(tiles[0]); + expect(sourceCache.findLoadedParent(notLoadedTiles[3], 0).tileID).toBe(tiles[3]); + + expect(notLoadedTiles[0].key in sourceCache._loadedParentTiles).toBe(false); + expect(notLoadedTiles[1].key in sourceCache._loadedParentTiles).toBe(false); + expect(notLoadedTiles[2].key in sourceCache._loadedParentTiles).toBe(false); + expect(notLoadedTiles[3].key in sourceCache._loadedParentTiles).toBe(false); + + }); + +}); + +describe('SourceCache#reload', () => { + test('before loaded', () => { + const sourceCache = createSourceCache({noLoad: true}); + sourceCache.onAdd(undefined); + + expect(() => { + sourceCache.reload(); + }).not.toThrow(); + + }); + +}); + +describe('SourceCache reloads expiring tiles', () => { + test('calls reloadTile when tile expires', done => { + const coord = new OverscaledTileID(1, 0, 1, 0, 1); + + const expiryDate = new Date(); + expiryDate.setMilliseconds(expiryDate.getMilliseconds() + 50); + const sourceCache = createSourceCache({expires: expiryDate}); + + sourceCache._reloadTile = (id, state) => { + expect(state).toBe('expired'); + done(); + }; + + sourceCache._addTile(coord); + }); + +}); + +describe('SourceCache sets max cache size correctly', () => { + test('sets cache size based on 512 tiles', () => { + const sourceCache = createSourceCache({ + tileSize: 256 + }); + + const tr = new Transform(); + tr.width = 512; + tr.height = 512; + sourceCache.updateCacheSize(tr); + + // Expect max size to be ((512 / tileSize + 1) ^ 2) * 5 => 3 * 3 * 5 + expect(sourceCache._cache.max).toBe(45); + }); + + test('sets cache size based on 256 tiles', () => { + const sourceCache = createSourceCache({ + tileSize: 512 + }); + + const tr = new Transform(); + tr.width = 512; + tr.height = 512; + sourceCache.updateCacheSize(tr); + + // Expect max size to be ((512 / tileSize + 1) ^ 2) * 5 => 2 * 2 * 5 + expect(sourceCache._cache.max).toBe(20); + }); + +}); + +describe('SourceCache#onRemove', () => { + test('clears tiles', () => { + const sourceCache = createSourceCache(); + jest.spyOn(sourceCache, 'clearTiles'); + + sourceCache.onRemove(undefined); + + expect(sourceCache.clearTiles).toHaveBeenCalled(); + }); + + test('calls onRemove on source', () => { + const sourceOnRemove = jest.fn(); + const sourceCache = createSourceCache({ + onRemove: sourceOnRemove + }); + + sourceCache.onRemove(undefined); + + expect(sourceOnRemove).toHaveBeenCalled(); + }); +}); + +describe('SourceCache#usedForTerrain', () => { + test('loads covering tiles with usedForTerrain with source zoom 0-14', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 10; + + const sourceCache = createSourceCache({}); + sourceCache.usedForTerrain = true; + sourceCache.tileSize = 1024; + expect(sourceCache.usedForTerrain).toBeTruthy(); + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + expect(Object.values(sourceCache._tiles).map(t => t.tileID.key)).toEqual( + ['2tc099', '2tbz99', '2sxs99', '2sxr99', 'pds88', 'eo55', 'pdr88', 'en55', 'p6o88', 'ds55', 'p6n88', 'dr55'] + ); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('loads covering tiles with usedForTerrain with source zoom 8-14', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 10; + + const sourceCache = createSourceCache({minzoom: 8, maxzoom: 14}); + sourceCache.usedForTerrain = true; + sourceCache.tileSize = 1024; + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + expect(Object.values(sourceCache._tiles).map(t => t.tileID.key)).toEqual( + ['2tc099', '2tbz99', '2sxs99', '2sxr99', 'pds88', 'pdr88', 'p6o88', 'p6n88'] + ); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('loads covering tiles with usedForTerrain with source zoom 0-4', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 10; + + const sourceCache = createSourceCache({minzoom: 0, maxzoom: 4}); + sourceCache.usedForTerrain = true; + sourceCache.tileSize = 1024; + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + expect(Object.values(sourceCache._tiles).map(t => t.tileID.key)).toEqual( + ['1033', '3s44', '3r44', '3c44', '3b44', 'z33', 's33', 'r33'] + ); + done(); + } + }); + sourceCache.onAdd(undefined); + }); + + test('loads covering tiles with usedForTerrain with source zoom 4-4', done => { + const transform = new Transform(); + transform.resize(511, 511); + transform.zoom = 10; + + const sourceCache = createSourceCache({minzoom: 4, maxzoom: 4}); + sourceCache.usedForTerrain = true; + sourceCache.tileSize = 1024; + sourceCache.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + sourceCache.update(transform); + expect(Object.values(sourceCache._tiles).map(t => t.tileID.key)).toEqual( + ['3s44', '3r44', '3c44', '3b44'] + ); + done(); + } + }); + sourceCache.onAdd(undefined); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/source_cache.ts b/web/libraries/maplibre-gl/src/source/source_cache.ts new file mode 100644 index 00000000..8a6d997f --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/source_cache.ts @@ -0,0 +1,1041 @@ +import {create as createSource} from './source'; + +import {Tile} from './tile'; +import {Event, ErrorEvent, Evented} from '../util/evented'; +import {TileCache} from './tile_cache'; +import {MercatorCoordinate} from '../geo/mercator_coordinate'; +import {keysDifference} from '../util/util'; +import {EXTENT} from '../data/extent'; +import {Context} from '../gl/context'; +import Point from '@mapbox/point-geometry'; +import {browser} from '../util/browser'; +import {OverscaledTileID} from './tile_id'; +import {SourceFeatureState} from './source_state'; + +import type {Source} from './source'; +import type {Map} from '../ui/map'; +import type {Style} from '../style/style'; +import type {Dispatcher} from '../util/dispatcher'; +import type {Transform} from '../geo/transform'; +import type {TileState} from './tile'; +import type {Callback} from '../types/callback'; +import type {SourceSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {MapSourceDataEvent} from '../ui/events'; +import {Terrain} from '../render/terrain'; +import {config} from '../util/config'; + +/** + * @internal + * `SourceCache` is responsible for + * + * - creating an instance of `Source` + * - forwarding events from `Source` + * - caching tiles loaded from an instance of `Source` + * - loading the tiles needed to render a given viewport + * - unloading the cached tiles not needed to render a given viewport + */ +export class SourceCache extends Evented { + id: string; + dispatcher: Dispatcher; + map: Map; + style: Style; + + _source: Source; + _sourceLoaded: boolean; + _sourceErrored: boolean; + _tiles: {[_: string]: Tile}; + _prevLng: number; + _cache: TileCache; + _timers: { + [_ in any]: ReturnType; + }; + _cacheTimers: { + [_ in any]: ReturnType; + }; + _maxTileCacheSize: number; + _maxTileCacheZoomLevels: number; + _paused: boolean; + _shouldReloadOnResume: boolean; + _coveredTiles: {[_: string]: boolean}; + transform: Transform; + terrain: Terrain; + used: boolean; + usedForTerrain: boolean; + tileSize: number; + _state: SourceFeatureState; + _loadedParentTiles: {[_: string]: Tile}; + _didEmitContent: boolean; + _updated: boolean; + + static maxUnderzooming: number; + static maxOverzooming: number; + + constructor(id: string, options: SourceSpecification, dispatcher: Dispatcher) { + super(); + this.id = id; + this.dispatcher = dispatcher; + + this.on('data', (e: MapSourceDataEvent) => { + // this._sourceLoaded signifies that the TileJSON is loaded if applicable. + // if the source type does not come with a TileJSON, the flag signifies the + // source data has loaded (i.e geojson has been tiled on the worker and is ready) + if (e.dataType === 'source' && e.sourceDataType === 'metadata') this._sourceLoaded = true; + + // for sources with mutable data, this event fires when the underlying data + // to a source is changed. (i.e. GeoJSONSource#setData and ImageSource#serCoordinates) + if (this._sourceLoaded && !this._paused && e.dataType === 'source' && e.sourceDataType === 'content') { + this.reload(); + if (this.transform) { + this.update(this.transform, this.terrain); + } + + this._didEmitContent = true; + } + }); + + this.on('dataloading', () => { + this._sourceErrored = false; + }); + + this.on('error', () => { + // Only set _sourceErrored if the source does not have pending loads. + this._sourceErrored = this._source.loaded(); + }); + + this._source = createSource(id, options, dispatcher, this); + + this._tiles = {}; + this._cache = new TileCache(0, this._unloadTile.bind(this)); + this._timers = {}; + this._cacheTimers = {}; + this._maxTileCacheSize = null; + this._maxTileCacheZoomLevels = null; + this._loadedParentTiles = {}; + + this._coveredTiles = {}; + this._state = new SourceFeatureState(); + this._didEmitContent = false; + this._updated = false; + } + + onAdd(map: Map) { + this.map = map; + this._maxTileCacheSize = map ? map._maxTileCacheSize : null; + this._maxTileCacheZoomLevels = map ? map._maxTileCacheZoomLevels : null; + if (this._source && this._source.onAdd) { + this._source.onAdd(map); + } + } + + onRemove(map: Map) { + this.clearTiles(); + if (this._source && this._source.onRemove) { + this._source.onRemove(map); + } + } + + /** + * Return true if no tile data is pending, tiles will not change unless + * an additional API call is received. + */ + loaded(): boolean { + if (this._sourceErrored) { return true; } + if (!this._sourceLoaded) { return false; } + if (!this._source.loaded()) { return false; } + if ((this.used !== undefined || this.usedForTerrain !== undefined) && !this.used && !this.usedForTerrain) { return true; } + // do not consider as loaded if the update hasn't been called yet (we do not know if we will have any tiles to fetch) + if (!this._updated) { return false; } + + for (const t in this._tiles) { + const tile = this._tiles[t]; + if (tile.state !== 'loaded' && tile.state !== 'errored') + return false; + } + return true; + } + + getSource(): Source { + return this._source; + } + + pause() { + this._paused = true; + } + + resume() { + if (!this._paused) return; + const shouldReload = this._shouldReloadOnResume; + this._paused = false; + this._shouldReloadOnResume = false; + if (shouldReload) this.reload(); + if (this.transform) this.update(this.transform, this.terrain); + } + + _loadTile(tile: Tile, callback: Callback) { + return this._source.loadTile(tile, callback); + } + + _unloadTile(tile: Tile) { + if (this._source.unloadTile) + return this._source.unloadTile(tile, () => {}); + } + + _abortTile(tile: Tile) { + if (this._source.abortTile) + this._source.abortTile(tile, () => {}); + + this._source.fire(new Event('dataabort', {tile, coord: tile.tileID, dataType: 'source'})); + } + + serialize() { + return this._source.serialize(); + } + + prepare(context: Context) { + if (this._source.prepare) { + this._source.prepare(); + } + + this._state.coalesceChanges(this._tiles, this.map ? this.map.painter : null); + for (const i in this._tiles) { + const tile = this._tiles[i]; + tile.upload(context); + tile.prepare(this.map.style.imageManager); + } + } + + /** + * Return all tile ids ordered with z-order, and cast to numbers + */ + getIds(): Array { + return (Object.values(this._tiles) as any).map((tile: Tile) => tile.tileID).sort(compareTileId).map(id => id.key); + } + + getRenderableIds(symbolLayer?: boolean): Array { + const renderables: Array = []; + for (const id in this._tiles) { + if (this._isIdRenderable(id, symbolLayer)) renderables.push(this._tiles[id]); + } + if (symbolLayer) { + return renderables.sort((a_: Tile, b_: Tile) => { + const a = a_.tileID; + const b = b_.tileID; + const rotatedA = (new Point(a.canonical.x, a.canonical.y))._rotate(this.transform.angle); + const rotatedB = (new Point(b.canonical.x, b.canonical.y))._rotate(this.transform.angle); + return a.overscaledZ - b.overscaledZ || rotatedB.y - rotatedA.y || rotatedB.x - rotatedA.x; + }).map(tile => tile.tileID.key); + } + return renderables.map(tile => tile.tileID).sort(compareTileId).map(id => id.key); + } + + hasRenderableParent(tileID: OverscaledTileID) { + const parentTile = this.findLoadedParent(tileID, 0); + if (parentTile) { + return this._isIdRenderable(parentTile.tileID.key); + } + return false; + } + + _isIdRenderable(id: string, symbolLayer?: boolean) { + return this._tiles[id] && this._tiles[id].hasData() && + !this._coveredTiles[id] && (symbolLayer || !this._tiles[id].holdingForFade()); + } + + reload() { + if (this._paused) { + this._shouldReloadOnResume = true; + return; + } + + this._cache.reset(); + + for (const i in this._tiles) { + if (this._tiles[i].state !== 'errored') this._reloadTile(i, 'reloading'); + } + } + + _reloadTile(id: string, state: TileState) { + const tile = this._tiles[id]; + + // this potentially does not address all underlying + // issues https://github.com/mapbox/mapbox-gl-js/issues/4252 + // - hard to tell without repro steps + if (!tile) return; + + // The difference between "loading" tiles and "reloading" or "expired" + // tiles is that "reloading"/"expired" tiles are "renderable". + // Therefore, a "loading" tile cannot become a "reloading" tile without + // first becoming a "loaded" tile. + if (tile.state !== 'loading') { + tile.state = state; + } + + this._loadTile(tile, this._tileLoaded.bind(this, tile, id, state)); + } + + _tileLoaded(tile: Tile, id: string, previousState: TileState, err?: Error | null) { + if (err) { + tile.state = 'errored'; + if ((err as any).status !== 404) this._source.fire(new ErrorEvent(err, {tile})); + // continue to try loading parent/children tiles if a tile doesn't exist (404) + else this.update(this.transform, this.terrain); + return; + } + + tile.timeAdded = browser.now(); + if (previousState === 'expired') tile.refreshedUponExpiration = true; + this._setTileReloadTimer(id, tile); + if (this.getSource().type === 'raster-dem' && tile.dem) this._backfillDEM(tile); + this._state.initializeTileState(tile, this.map ? this.map.painter : null); + + if (!tile.aborted) { + this._source.fire(new Event('data', {dataType: 'source', tile, coord: tile.tileID})); + } + } + + /** + * For raster terrain source, backfill DEM to eliminate visible tile boundaries + */ + _backfillDEM(tile: Tile) { + const renderables = this.getRenderableIds(); + for (let i = 0; i < renderables.length; i++) { + const borderId = renderables[i]; + if (tile.neighboringTiles && tile.neighboringTiles[borderId]) { + const borderTile = this.getTileByID(borderId); + fillBorder(tile, borderTile); + fillBorder(borderTile, tile); + } + } + + function fillBorder(tile, borderTile) { + tile.needsHillshadePrepare = true; + tile.needsTerrainPrepare = true; + let dx = borderTile.tileID.canonical.x - tile.tileID.canonical.x; + const dy = borderTile.tileID.canonical.y - tile.tileID.canonical.y; + const dim = Math.pow(2, tile.tileID.canonical.z); + const borderId = borderTile.tileID.key; + if (dx === 0 && dy === 0) return; + + if (Math.abs(dy) > 1) { + return; + } + if (Math.abs(dx) > 1) { + // Adjust the delta coordinate for world wraparound. + if (Math.abs(dx + dim) === 1) { + dx += dim; + } else if (Math.abs(dx - dim) === 1) { + dx -= dim; + } + } + if (!borderTile.dem || !tile.dem) return; + tile.dem.backfillBorder(borderTile.dem, dx, dy); + if (tile.neighboringTiles && tile.neighboringTiles[borderId]) + tile.neighboringTiles[borderId].backfilled = true; + } + } + /** + * Get a specific tile by TileID + */ + getTile(tileID: OverscaledTileID): Tile { + return this.getTileByID(tileID.key); + } + + /** + * Get a specific tile by id + */ + getTileByID(id: string): Tile { + return this._tiles[id]; + } + + /** + * For a given set of tiles, retain children that are loaded and have a zoom + * between `zoom` (exclusive) and `maxCoveringZoom` (inclusive) + */ + _retainLoadedChildren( + idealTiles: { + [_ in any]: OverscaledTileID; + }, + zoom: number, + maxCoveringZoom: number, + retain: { + [_ in any]: OverscaledTileID; + } + ) { + for (const id in this._tiles) { + let tile = this._tiles[id]; + + // only consider renderable tiles up to maxCoveringZoom + if (retain[id] || + !tile.hasData() || + tile.tileID.overscaledZ <= zoom || + tile.tileID.overscaledZ > maxCoveringZoom + ) continue; + + // loop through parents and retain the topmost loaded one if found + let topmostLoadedID = tile.tileID; + while (tile && tile.tileID.overscaledZ > zoom + 1) { + const parentID = tile.tileID.scaledTo(tile.tileID.overscaledZ - 1); + + tile = this._tiles[parentID.key]; + + if (tile && tile.hasData()) { + topmostLoadedID = parentID; + } + } + + // loop through ancestors of the topmost loaded child to see if there's one that needed it + let tileID = topmostLoadedID; + while (tileID.overscaledZ > zoom) { + tileID = tileID.scaledTo(tileID.overscaledZ - 1); + + if (idealTiles[tileID.key]) { + // found a parent that needed a loaded child; retain that child + retain[topmostLoadedID.key] = topmostLoadedID; + break; + } + } + } + } + + /** + * Find a loaded parent of the given tile (up to minCoveringZoom) + */ + findLoadedParent(tileID: OverscaledTileID, minCoveringZoom: number): Tile { + if (tileID.key in this._loadedParentTiles) { + const parent = this._loadedParentTiles[tileID.key]; + if (parent && parent.tileID.overscaledZ >= minCoveringZoom) { + return parent; + } else { + return null; + } + } + for (let z = tileID.overscaledZ - 1; z >= minCoveringZoom; z--) { + const parentTileID = tileID.scaledTo(z); + const tile = this._getLoadedTile(parentTileID); + if (tile) { + return tile; + } + } + } + + _getLoadedTile(tileID: OverscaledTileID): Tile { + const tile = this._tiles[tileID.key]; + if (tile && tile.hasData()) { + return tile; + } + // TileCache ignores wrap in lookup. + const cachedTile = this._cache.getByKey(tileID.wrapped().key); + return cachedTile; + } + + /** + * Resizes the tile cache based on the current viewport's size + * or the maxTileCacheSize option passed during map creation + * + * Larger viewports use more tiles and need larger caches. Larger viewports + * are more likely to be found on devices with more memory and on pages where + * the map is more important. + */ + updateCacheSize(transform: Transform) { + const widthInTiles = Math.ceil(transform.width / this._source.tileSize) + 1; + const heightInTiles = Math.ceil(transform.height / this._source.tileSize) + 1; + const approxTilesInView = widthInTiles * heightInTiles; + const commonZoomRange = this._maxTileCacheZoomLevels === null ? + config.MAX_TILE_CACHE_ZOOM_LEVELS : this._maxTileCacheZoomLevels; + const viewDependentMaxSize = Math.floor(approxTilesInView * commonZoomRange); + const maxSize = typeof this._maxTileCacheSize === 'number' ? + Math.min(this._maxTileCacheSize, viewDependentMaxSize) : viewDependentMaxSize; + + this._cache.setMaxSize(maxSize); + } + + handleWrapJump(lng: number) { + // On top of the regular z/x/y values, TileIDs have a `wrap` value that specify + // which cppy of the world the tile belongs to. For example, at `lng: 10` you + // might render z/x/y/0 while at `lng: 370` you would render z/x/y/1. + // + // When lng values get wrapped (going from `lng: 370` to `long: 10`) you expect + // to see the same thing on the screen (370 degrees and 10 degrees is the same + // place in the world) but all the TileIDs will have different wrap values. + // + // In order to make this transition seamless, we calculate the rounded difference of + // "worlds" between the last frame and the current frame. If the map panned by + // a world, then we can assign all the tiles new TileIDs with updated wrap values. + // For example, assign z/x/y/1 a new id: z/x/y/0. It is the same tile, just rendered + // in a different position. + // + // This enables us to reuse the tiles at more ideal locations and prevent flickering. + const prevLng = this._prevLng === undefined ? lng : this._prevLng; + const lngDifference = lng - prevLng; + const worldDifference = lngDifference / 360; + const wrapDelta = Math.round(worldDifference); + this._prevLng = lng; + + if (wrapDelta) { + const tiles: {[_: string]: Tile} = {}; + for (const key in this._tiles) { + const tile = this._tiles[key]; + tile.tileID = tile.tileID.unwrapTo(tile.tileID.wrap + wrapDelta); + tiles[tile.tileID.key] = tile; + } + this._tiles = tiles; + + // Reset tile reload timers + for (const id in this._timers) { + clearTimeout(this._timers[id]); + delete this._timers[id]; + } + for (const id in this._tiles) { + const tile = this._tiles[id]; + this._setTileReloadTimer(id, tile); + } + } + } + + /** + * Removes tiles that are outside the viewport and adds new tiles that + * are inside the viewport. + */ + update(transform: Transform, terrain?: Terrain) { + this.transform = transform; + this.terrain = terrain; + if (!this._sourceLoaded || this._paused) { return; } + + this.updateCacheSize(transform); + this.handleWrapJump(this.transform.center.lng); + + // Covered is a list of retained tiles who's areas are fully covered by other, + // better, retained tiles. They are not drawn separately. + this._coveredTiles = {}; + + let idealTileIDs; + if (!this.used && !this.usedForTerrain) { + idealTileIDs = []; + } else if (this._source.tileID) { + idealTileIDs = transform.getVisibleUnwrappedCoordinates(this._source.tileID) + .map((unwrapped) => new OverscaledTileID(unwrapped.canonical.z, unwrapped.wrap, unwrapped.canonical.z, unwrapped.canonical.x, unwrapped.canonical.y)); + } else { + idealTileIDs = transform.coveringTiles({ + tileSize: this.usedForTerrain ? this.tileSize : this._source.tileSize, + minzoom: this._source.minzoom, + maxzoom: this._source.maxzoom, + roundZoom: this.usedForTerrain ? false : this._source.roundZoom, + reparseOverscaled: this._source.reparseOverscaled, + terrain + }); + + if (this._source.hasTile) { + idealTileIDs = idealTileIDs.filter((coord) => (this._source.hasTile as any)(coord)); + } + } + + // Determine the overzooming/underzooming amounts. + const zoom = transform.coveringZoomLevel(this._source); + const minCoveringZoom = Math.max(zoom - SourceCache.maxOverzooming, this._source.minzoom); + const maxCoveringZoom = Math.max(zoom + SourceCache.maxUnderzooming, this._source.minzoom); + + // When sourcecache is used for terrain also load parent tiles to avoid flickering when zooming out + if (this.usedForTerrain) { + const parents = {}; + for (const tileID of idealTileIDs) { + if (tileID.canonical.z > this._source.minzoom) { + const parent = tileID.scaledTo(tileID.canonical.z - 1); + parents[parent.key] = parent; + // load very low zoom to calculate tile visibility in transform.coveringTiles and high zoomlevels correct + const parent2 = tileID.scaledTo(Math.max(this._source.minzoom, Math.min(tileID.canonical.z, 5))); + parents[parent2.key] = parent2; + } + } + idealTileIDs = idealTileIDs.concat(Object.values(parents)); + } + + const noPendingDataEmissions = idealTileIDs.length === 0 && !this._updated && this._didEmitContent; + this._updated = true; + // if we won't have any tiles to fetch and content is already emitted + // there will be no more data emissions, so we need to emit the event with isSourceLoaded = true + if (noPendingDataEmissions) { + this.fire(new Event('data', {sourceDataType: 'idle', dataType: 'source', sourceId: this.id})); + } + + // Retain is a list of tiles that we shouldn't delete, even if they are not + // the most ideal tile for the current viewport. This may include tiles like + // parent or child tiles that are *already* loaded. + const retain = this._updateRetainedTiles(idealTileIDs, zoom); + + if (isRasterType(this._source.type)) { + const parentsForFading: {[_: string]: OverscaledTileID} = {}; + const fadingTiles = {}; + const ids = Object.keys(retain); + const now = browser.now(); + for (const id of ids) { + const tileID = retain[id]; + + const tile = this._tiles[id]; + + // when fadeEndTime is 0, the tile is created but registerFadeDuration + // has not been called, therefore must be kept in fadingTiles dictionary + // for next round of rendering + if (!tile || (tile.fadeEndTime !== 0 && tile.fadeEndTime <= now)) { + continue; + } + + // if the tile is loaded but still fading in, find parents to cross-fade with it + const parentTile = this.findLoadedParent(tileID, minCoveringZoom); + if (parentTile) { + this._addTile(parentTile.tileID); + parentsForFading[parentTile.tileID.key] = parentTile.tileID; + } + + fadingTiles[id] = tileID; + } + + // for tiles that are still fading in, also find children to cross-fade with + this._retainLoadedChildren(fadingTiles, zoom, maxCoveringZoom, retain); + + for (const id in parentsForFading) { + if (!retain[id]) { + // If a tile is only needed for fading, mark it as covered so that it isn't rendered on it's own. + this._coveredTiles[id] = true; + retain[id] = parentsForFading[id]; + } + } + + // disable fading logic in terrain3D mode to avoid rendering two tiles on the same place + if (terrain) { + const idealRasterTileIDs: {[_: string]: OverscaledTileID} = {}; + const missingTileIDs: {[_: string]: OverscaledTileID} = {}; + for (const tileID of idealTileIDs) { + if (this._tiles[tileID.key].hasData()) + idealRasterTileIDs[tileID.key] = tileID; + else + missingTileIDs[tileID.key] = tileID; + } + // search for a complete set of children for each missing tile + for (const key in missingTileIDs) { + const children = missingTileIDs[key].children(this._source.maxzoom); + if (this._tiles[children[0].key] && this._tiles[children[1].key] && this._tiles[children[2].key] && this._tiles[children[3].key]) { + idealRasterTileIDs[children[0].key] = retain[children[0].key] = children[0]; + idealRasterTileIDs[children[1].key] = retain[children[1].key] = children[1]; + idealRasterTileIDs[children[2].key] = retain[children[2].key] = children[2]; + idealRasterTileIDs[children[3].key] = retain[children[3].key] = children[3]; + delete missingTileIDs[key]; + } + } + // search for parent for each missing tile + for (const key in missingTileIDs) { + const parent = this.findLoadedParent(missingTileIDs[key], this._source.minzoom); + if (parent) { + idealRasterTileIDs[parent.tileID.key] = retain[parent.tileID.key] = parent.tileID; + // remove idealTiles which would be rendered twice + for (const key in idealRasterTileIDs) { + if (idealRasterTileIDs[key].isChildOf(parent.tileID)) delete idealRasterTileIDs[key]; + } + } + } + // cover all tiles which are not needed + for (const key in this._tiles) { + if (!idealRasterTileIDs[key]) this._coveredTiles[key] = true; + } + } + } + + for (const retainedId in retain) { + // Make sure retained tiles always clear any existing fade holds + // so that if they're removed again their fade timer starts fresh. + this._tiles[retainedId].clearFadeHold(); + } + + // Remove the tiles we don't need anymore. + const remove = keysDifference(this._tiles, retain); + for (const tileID of remove) { + const tile = this._tiles[tileID]; + if (tile.hasSymbolBuckets && !tile.holdingForFade()) { + tile.setHoldDuration(this.map._fadeDuration); + } else if (!tile.hasSymbolBuckets || tile.symbolFadeFinished()) { + this._removeTile(tileID); + } + } + + // Construct a cache of loaded parents + this._updateLoadedParentTileCache(); + } + + releaseSymbolFadeTiles() { + for (const id in this._tiles) { + if (this._tiles[id].holdingForFade()) { + this._removeTile(id); + } + } + } + + _updateRetainedTiles(idealTileIDs: Array, zoom: number): {[_: string]: OverscaledTileID} { + const retain: {[_: string]: OverscaledTileID} = {}; + const checked: {[_: string]: boolean} = {}; + const minCoveringZoom = Math.max(zoom - SourceCache.maxOverzooming, this._source.minzoom); + const maxCoveringZoom = Math.max(zoom + SourceCache.maxUnderzooming, this._source.minzoom); + + const missingTiles = {}; + for (const tileID of idealTileIDs) { + const tile = this._addTile(tileID); + + // retain the tile even if it's not loaded because it's an ideal tile. + retain[tileID.key] = tileID; + + if (tile.hasData()) continue; + + if (zoom < this._source.maxzoom) { + // save missing tiles that potentially have loaded children + missingTiles[tileID.key] = tileID; + } + } + + // retain any loaded children of ideal tiles up to maxCoveringZoom + this._retainLoadedChildren(missingTiles, zoom, maxCoveringZoom, retain); + + for (const tileID of idealTileIDs) { + let tile = this._tiles[tileID.key]; + + if (tile.hasData()) continue; + + // The tile we require is not yet loaded or does not exist; + // Attempt to find children that fully cover it. + + if (zoom + 1 > this._source.maxzoom) { + // We're looking for an overzoomed child tile. + const childCoord = tileID.children(this._source.maxzoom)[0]; + const childTile = this.getTile(childCoord); + if (!!childTile && childTile.hasData()) { + retain[childCoord.key] = childCoord; + continue; // tile is covered by overzoomed child + } + } else { + // check if all 4 immediate children are loaded (i.e. the missing ideal tile is covered) + const children = tileID.children(this._source.maxzoom); + + if (retain[children[0].key] && + retain[children[1].key] && + retain[children[2].key] && + retain[children[3].key]) continue; // tile is covered by children + } + + // We couldn't find child tiles that entirely cover the ideal tile; look for parents now. + + // As we ascend up the tile pyramid of the ideal tile, we check whether the parent + // tile has been previously requested (and errored because we only loop over tiles with no data) + // in order to determine if we need to request its parent. + let parentWasRequested = tile.wasRequested(); + + for (let overscaledZ = tileID.overscaledZ - 1; overscaledZ >= minCoveringZoom; --overscaledZ) { + const parentId = tileID.scaledTo(overscaledZ); + + // Break parent tile ascent if this route has been previously checked by another child. + if (checked[parentId.key]) break; + checked[parentId.key] = true; + + tile = this.getTile(parentId); + if (!tile && parentWasRequested) { + tile = this._addTile(parentId); + } + if (tile) { + const hasData = tile.hasData(); + if (parentWasRequested || hasData) { + retain[parentId.key] = parentId; + } + // Save the current values, since they're the parent of the next iteration + // of the parent tile ascent loop. + parentWasRequested = tile.wasRequested(); + if (hasData) break; + } + } + } + + return retain; + } + + _updateLoadedParentTileCache() { + this._loadedParentTiles = {}; + + for (const tileKey in this._tiles) { + const path = []; + let parentTile: Tile; + let currentId = this._tiles[tileKey].tileID; + + // Find the closest loaded ancestor by traversing the tile tree towards the root and + // caching results along the way + while (currentId.overscaledZ > 0) { + + // Do we have a cached result from previous traversals? + if (currentId.key in this._loadedParentTiles) { + parentTile = this._loadedParentTiles[currentId.key]; + break; + } + + path.push(currentId.key); + + // Is the parent loaded? + const parentId = currentId.scaledTo(currentId.overscaledZ - 1); + parentTile = this._getLoadedTile(parentId); + if (parentTile) { + break; + } + + currentId = parentId; + } + + // Cache the result of this traversal to all newly visited tiles + for (const key of path) { + this._loadedParentTiles[key] = parentTile; + } + } + } + + /** + * Add a tile, given its coordinate, to the pyramid. + */ + _addTile(tileID: OverscaledTileID): Tile { + let tile = this._tiles[tileID.key]; + if (tile) + return tile; + + tile = this._cache.getAndRemove(tileID); + if (tile) { + this._setTileReloadTimer(tileID.key, tile); + // set the tileID because the cached tile could have had a different wrap value + tile.tileID = tileID; + this._state.initializeTileState(tile, this.map ? this.map.painter : null); + if (this._cacheTimers[tileID.key]) { + clearTimeout(this._cacheTimers[tileID.key]); + delete this._cacheTimers[tileID.key]; + this._setTileReloadTimer(tileID.key, tile); + } + } + + const cached = tile; + + if (!tile) { + tile = new Tile(tileID, this._source.tileSize * tileID.overscaleFactor()); + this._loadTile(tile, this._tileLoaded.bind(this, tile, tileID.key, tile.state)); + } + + tile.uses++; + this._tiles[tileID.key] = tile; + if (!cached) { + this._source.fire(new Event('dataloading', {tile, coord: tile.tileID, dataType: 'source'})); + } + + return tile; + } + + _setTileReloadTimer(id: string, tile: Tile) { + if (id in this._timers) { + clearTimeout(this._timers[id]); + delete this._timers[id]; + } + + const expiryTimeout = tile.getExpiryTimeout(); + if (expiryTimeout) { + this._timers[id] = setTimeout(() => { + this._reloadTile(id, 'expired'); + delete this._timers[id]; + }, expiryTimeout); + } + } + + /** + * Remove a tile, given its id, from the pyramid + */ + _removeTile(id: string) { + const tile = this._tiles[id]; + if (!tile) + return; + + tile.uses--; + delete this._tiles[id]; + if (this._timers[id]) { + clearTimeout(this._timers[id]); + delete this._timers[id]; + } + + if (tile.uses > 0) + return; + + if (tile.hasData() && tile.state !== 'reloading') { + this._cache.add(tile.tileID, tile, tile.getExpiryTimeout()); + } else { + tile.aborted = true; + this._abortTile(tile); + this._unloadTile(tile); + } + } + + /** + * Remove all tiles from this pyramid + */ + clearTiles() { + this._shouldReloadOnResume = false; + this._paused = false; + + for (const id in this._tiles) + this._removeTile(id); + + this._cache.reset(); + } + + /** + * Search through our current tiles and attempt to find the tiles that + * cover the given bounds. + * @param pointQueryGeometry - coordinates of the corners of bounding rectangle + * @returns result items have `{tile, minX, maxX, minY, maxY}`, where min/max bounding values are the given bounds transformed in into the coordinate space of this tile. + */ + tilesIn(pointQueryGeometry: Array, maxPitchScaleFactor: number, has3DLayer: boolean): any[] { + + const tileResults = []; + + const transform = this.transform; + if (!transform) return tileResults; + + const cameraPointQueryGeometry = has3DLayer ? + transform.getCameraQueryGeometry(pointQueryGeometry) : + pointQueryGeometry; + + const queryGeometry = pointQueryGeometry.map((p: Point) => transform.pointCoordinate(p, this.terrain)); + const cameraQueryGeometry = cameraPointQueryGeometry.map((p: Point) => transform.pointCoordinate(p, this.terrain)); + + const ids = this.getIds(); + + let minX = Infinity; + let minY = Infinity; + let maxX = -Infinity; + let maxY = -Infinity; + + for (const p of cameraQueryGeometry) { + minX = Math.min(minX, p.x); + minY = Math.min(minY, p.y); + maxX = Math.max(maxX, p.x); + maxY = Math.max(maxY, p.y); + } + + for (let i = 0; i < ids.length; i++) { + const tile = this._tiles[ids[i]]; + if (tile.holdingForFade()) { + // Tiles held for fading are covered by tiles that are closer to ideal + continue; + } + const tileID = tile.tileID; + const scale = Math.pow(2, transform.zoom - tile.tileID.overscaledZ); + const queryPadding = maxPitchScaleFactor * tile.queryPadding * EXTENT / tile.tileSize / scale; + + const tileSpaceBounds = [ + tileID.getTilePoint(new MercatorCoordinate(minX, minY)), + tileID.getTilePoint(new MercatorCoordinate(maxX, maxY)) + ]; + + if (tileSpaceBounds[0].x - queryPadding < EXTENT && tileSpaceBounds[0].y - queryPadding < EXTENT && + tileSpaceBounds[1].x + queryPadding >= 0 && tileSpaceBounds[1].y + queryPadding >= 0) { + + const tileSpaceQueryGeometry: Array = queryGeometry.map((c) => tileID.getTilePoint(c)); + const tileSpaceCameraQueryGeometry = cameraQueryGeometry.map((c) => tileID.getTilePoint(c)); + + tileResults.push({ + tile, + tileID, + queryGeometry: tileSpaceQueryGeometry, + cameraQueryGeometry: tileSpaceCameraQueryGeometry, + scale + }); + } + } + + return tileResults; + } + + getVisibleCoordinates(symbolLayer?: boolean): Array { + const coords = this.getRenderableIds(symbolLayer).map((id) => this._tiles[id].tileID); + for (const coord of coords) { + coord.posMatrix = this.transform.calculatePosMatrix(coord.toUnwrapped()); + } + return coords; + } + + hasTransition() { + if (this._source.hasTransition()) { + return true; + } + + if (isRasterType(this._source.type)) { + const now = browser.now(); + for (const id in this._tiles) { + const tile = this._tiles[id]; + if (tile.fadeEndTime >= now) { + return true; + } + } + } + + return false; + } + + /** + * Set the value of a particular state for a feature + */ + setFeatureState(sourceLayer: string, featureId: number | string, state: any) { + sourceLayer = sourceLayer || '_geojsonTileLayer'; + this._state.updateState(sourceLayer, featureId, state); + } + + /** + * Resets the value of a particular state key for a feature + */ + removeFeatureState(sourceLayer?: string, featureId?: number | string, key?: string) { + sourceLayer = sourceLayer || '_geojsonTileLayer'; + this._state.removeFeatureState(sourceLayer, featureId, key); + } + + /** + * Get the entire state object for a feature + */ + getFeatureState(sourceLayer: string, featureId: number | string) { + sourceLayer = sourceLayer || '_geojsonTileLayer'; + return this._state.getState(sourceLayer, featureId); + } + + /** + * Sets the set of keys that the tile depends on. This allows tiles to + * be reloaded when their dependencies change. + */ + setDependencies(tileKey: string, namespace: string, dependencies: Array) { + const tile = this._tiles[tileKey]; + if (tile) { + tile.setDependencies(namespace, dependencies); + } + } + + /** + * Reloads all tiles that depend on the given keys. + */ + reloadTilesForDependencies(namespaces: Array, keys: Array) { + for (const id in this._tiles) { + const tile = this._tiles[id]; + if (tile.hasDependency(namespaces, keys)) { + this._reloadTile(id, 'reloading'); + } + } + this._cache.filter(tile => !tile.hasDependency(namespaces, keys)); + } +} + +SourceCache.maxOverzooming = 10; +SourceCache.maxUnderzooming = 3; + +function compareTileId(a: OverscaledTileID, b: OverscaledTileID): number { + // Different copies of the world are sorted based on their distance to the center. + // Wrap values are converted to unsigned distances by reserving odd number for copies + // with negative wrap and even numbers for copies with positive wrap. + const aWrap = Math.abs(a.wrap * 2) - +(a.wrap < 0); + const bWrap = Math.abs(b.wrap * 2) - +(b.wrap < 0); + return a.overscaledZ - b.overscaledZ || bWrap - aWrap || b.canonical.y - a.canonical.y || b.canonical.x - a.canonical.x; +} + +function isRasterType(type) { + return type === 'raster' || type === 'image' || type === 'video'; +} diff --git a/web/libraries/maplibre-gl/src/source/source_state.ts b/web/libraries/maplibre-gl/src/source/source_state.ts new file mode 100644 index 00000000..6358e92d --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/source_state.ts @@ -0,0 +1,157 @@ +import {extend} from '../util/util'; +import {Tile} from './tile'; +import type {FeatureState} from '@maplibre/maplibre-gl-style-spec'; + +export type FeatureStates = {[featureId: string]: FeatureState}; +export type LayerFeatureStates = {[layer: string]: FeatureStates}; + +/** + * @internal + * SourceFeatureState manages the state and pending changes + * to features in a source, separated by source layer. + * stateChanges and deletedStates batch all changes to the tile (updates and removes, respectively) + * between coalesce() events. addFeatureState() and removeFeatureState() also update their counterpart's + * list of changes, such that coalesce() can apply the proper state changes while agnostic to the order of operations. + * In deletedStates, all null's denote complete removal of state at that scope +*/ +export class SourceFeatureState { + state: LayerFeatureStates; + stateChanges: LayerFeatureStates; + deletedStates: {}; + + constructor() { + this.state = {}; + this.stateChanges = {}; + this.deletedStates = {}; + } + + updateState(sourceLayer: string, featureId: number | string, newState: any) { + const feature = String(featureId); + this.stateChanges[sourceLayer] = this.stateChanges[sourceLayer] || {}; + this.stateChanges[sourceLayer][feature] = this.stateChanges[sourceLayer][feature] || {}; + extend(this.stateChanges[sourceLayer][feature], newState); + + if (this.deletedStates[sourceLayer] === null) { + this.deletedStates[sourceLayer] = {}; + for (const ft in this.state[sourceLayer]) { + if (ft !== feature) this.deletedStates[sourceLayer][ft] = null; + } + } else { + const featureDeletionQueued = this.deletedStates[sourceLayer] && this.deletedStates[sourceLayer][feature] === null; + if (featureDeletionQueued) { + this.deletedStates[sourceLayer][feature] = {}; + for (const prop in this.state[sourceLayer][feature]) { + if (!newState[prop]) this.deletedStates[sourceLayer][feature][prop] = null; + } + } else { + for (const key in newState) { + const deletionInQueue = this.deletedStates[sourceLayer] && this.deletedStates[sourceLayer][feature] && this.deletedStates[sourceLayer][feature][key] === null; + if (deletionInQueue) delete this.deletedStates[sourceLayer][feature][key]; + } + } + } + } + + removeFeatureState(sourceLayer: string, featureId?: number | string, key?: string) { + const sourceLayerDeleted = this.deletedStates[sourceLayer] === null; + if (sourceLayerDeleted) return; + + const feature = String(featureId); + + this.deletedStates[sourceLayer] = this.deletedStates[sourceLayer] || {}; + + if (key && featureId !== undefined) { + if (this.deletedStates[sourceLayer][feature] !== null) { + this.deletedStates[sourceLayer][feature] = this.deletedStates[sourceLayer][feature] || {}; + this.deletedStates[sourceLayer][feature][key] = null; + } + } else if (featureId !== undefined) { + const updateInQueue = this.stateChanges[sourceLayer] && this.stateChanges[sourceLayer][feature]; + if (updateInQueue) { + this.deletedStates[sourceLayer][feature] = {}; + for (key in this.stateChanges[sourceLayer][feature]) this.deletedStates[sourceLayer][feature][key] = null; + + } else { + this.deletedStates[sourceLayer][feature] = null; + } + } else { + this.deletedStates[sourceLayer] = null; + } + + } + + getState(sourceLayer: string, featureId: number | string) { + const feature = String(featureId); + const base = this.state[sourceLayer] || {}; + const changes = this.stateChanges[sourceLayer] || {}; + + const reconciledState = extend({}, base[feature], changes[feature]); + + //return empty object if the whole source layer is awaiting deletion + if (this.deletedStates[sourceLayer] === null) return {}; + else if (this.deletedStates[sourceLayer]) { + const featureDeletions = this.deletedStates[sourceLayer][featureId]; + if (featureDeletions === null) return {}; + for (const prop in featureDeletions) delete reconciledState[prop]; + } + return reconciledState; + } + + initializeTileState(tile: Tile, painter: any) { + tile.setFeatureState(this.state, painter); + } + + coalesceChanges(tiles: { + [_ in any]: Tile; + }, painter: any) { + //track changes with full state objects, but only for features that got modified + const featuresChanged: LayerFeatureStates = {}; + + for (const sourceLayer in this.stateChanges) { + this.state[sourceLayer] = this.state[sourceLayer] || {}; + const layerStates = {}; + for (const feature in this.stateChanges[sourceLayer]) { + if (!this.state[sourceLayer][feature]) this.state[sourceLayer][feature] = {}; + extend(this.state[sourceLayer][feature], this.stateChanges[sourceLayer][feature]); + layerStates[feature] = this.state[sourceLayer][feature]; + } + featuresChanged[sourceLayer] = layerStates; + } + + for (const sourceLayer in this.deletedStates) { + this.state[sourceLayer] = this.state[sourceLayer] || {}; + const layerStates = {}; + + if (this.deletedStates[sourceLayer] === null) { + for (const ft in this.state[sourceLayer]) { + layerStates[ft] = {}; + this.state[sourceLayer][ft] = {}; + } + } else { + for (const feature in this.deletedStates[sourceLayer]) { + const deleteWholeFeatureState = this.deletedStates[sourceLayer][feature] === null; + if (deleteWholeFeatureState) this.state[sourceLayer][feature] = {}; + else { + for (const key of Object.keys(this.deletedStates[sourceLayer][feature])) { + delete this.state[sourceLayer][feature][key]; + } + } + layerStates[feature] = this.state[sourceLayer][feature]; + } + } + + featuresChanged[sourceLayer] = featuresChanged[sourceLayer] || {}; + extend(featuresChanged[sourceLayer], layerStates); + } + + this.stateChanges = {}; + this.deletedStates = {}; + + if (Object.keys(featuresChanged).length === 0) return; + + for (const id in tiles) { + const tile = tiles[id]; + tile.setFeatureState(featuresChanged, painter); + } + } +} diff --git a/web/libraries/maplibre-gl/src/source/terrain_source_cache.test.ts b/web/libraries/maplibre-gl/src/source/terrain_source_cache.test.ts new file mode 100644 index 00000000..a947a543 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/terrain_source_cache.test.ts @@ -0,0 +1,105 @@ +import {TerrainSourceCache} from './terrain_source_cache'; +import {Style} from '../style/style'; +import {RequestManager} from '../util/request_manager'; +import {Dispatcher} from '../util/dispatcher'; +import {fakeServer, type FakeServer} from 'nise'; +import {Transform} from '../geo/transform'; +import {Evented} from '../util/evented'; +import {Painter} from '../render/painter'; +import {RasterDEMTileSource} from './raster_dem_tile_source'; +import {OverscaledTileID} from './tile_id'; +import {Tile} from './tile'; +import {DEMData} from '../data/dem_data'; + +const transform = new Transform(); + +class StubMap extends Evented { + transform: Transform; + painter: Painter; + _requestManager: RequestManager; + + constructor() { + super(); + this.transform = transform; + this._requestManager = { + transformRequest: (url) => { + return {url}; + } + } as any as RequestManager; + } + + _getMapId() { + return 1; + } + + setTerrain() {} +} + +function createSource(options, transformCallback?) { + const source = new RasterDEMTileSource('id', options, {send() {}} as any as Dispatcher, null); + source.onAdd({ + transform, + _requestManager: new RequestManager(transformCallback), + getPixelRatio() { return 1; } + } as any); + + source.on('error', (e) => { + throw e.error; + }); + + return source; +} + +describe('TerrainSourceCache', () => { + let server: FakeServer; + let style: Style; + let tsc: TerrainSourceCache; + + beforeAll(done => { + global.fetch = null; + server = fakeServer.create(); + server.respondWith('/source.json', JSON.stringify({ + minzoom: 5, + maxzoom: 12, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.pngraw'], + bounds: [-47, -7, -45, -5] + })); + const map = new StubMap(); + style = new Style(map as any); + style.on('style.load', () => { + const source = createSource({url: '/source.json'}); + server.respond(); + style.addSource('terrain', source as any); + tsc = new TerrainSourceCache(style.sourceCaches.terrain); + done(); + }); + style.loadJSON({ + 'version': 8, + 'sources': {}, + 'layers': [] + }); + }); + + afterAll(() => { + server.restore(); + }); + + test('#constructor', () => { + expect(tsc.sourceCache.usedForTerrain).toBeTruthy(); + expect(tsc.sourceCache.tileSize).toBe(tsc.tileSize * 2 ** tsc.deltaZoom); + }); + + test('#getSourceTile', () => { + const tileID = new OverscaledTileID(5, 0, 5, 17, 11); + const tile = new Tile(tileID, 256); + tile.dem = {} as DEMData; + tsc.sourceCache._tiles[tileID.key] = tile; + expect(tsc.deltaZoom).toBe(1); + expect(tsc.getSourceTile(tileID)).toBeFalsy(); + expect(tsc.getSourceTile(tileID.children(12)[0])).toBeTruthy(); + expect(tsc.getSourceTile(tileID.children(12)[0].children(12)[0])).toBeFalsy(); + expect(tsc.getSourceTile(tileID.children(12)[0].children(12)[0], true)).toBeTruthy(); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/source/terrain_source_cache.ts b/web/libraries/maplibre-gl/src/source/terrain_source_cache.ts new file mode 100644 index 00000000..909e8232 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/terrain_source_cache.ts @@ -0,0 +1,203 @@ +import {OverscaledTileID} from './tile_id'; +import {Tile} from './tile'; +import {EXTENT} from '../data/extent'; +import {mat4} from 'gl-matrix'; +import {Evented} from '../util/evented'; +import type {Transform} from '../geo/transform'; +import type {SourceCache} from '../source/source_cache'; +import {Terrain} from '../render/terrain'; + +/** + * @internal + * This class is a helper for the Terrain-class, it: + * - loads raster-dem tiles + * - manages all renderToTexture tiles. + * - caches previous rendered tiles. + * - finds all necessary renderToTexture tiles for a OverscaledTileID area + * - finds the corresponding raster-dem tile for OverscaledTileID + */ +export class TerrainSourceCache extends Evented { + /** + * source-cache for the raster-dem source. + */ + sourceCache: SourceCache; + /** + * stores all render-to-texture tiles. + */ + _tiles: {[_: string]: Tile}; + /** + * contains a list of tileID-keys for the current scene. (only for performance) + */ + _renderableTilesKeys: Array; + /** + * raster-dem-tile for a TileID cache. + */ + _sourceTileCache: {[_: string]: string}; + /** + * minimum zoomlevel to render the terrain. + */ + minzoom: number; + /** + * maximum zoomlevel to render the terrain. + */ + maxzoom: number; + /** + * render-to-texture tileSize in scene. + */ + tileSize: number; + /** + * raster-dem tiles will load for performance the actualZoom - deltaZoom zoom-level. + */ + deltaZoom: number; + + constructor(sourceCache: SourceCache) { + super(); + this.sourceCache = sourceCache; + this._tiles = {}; + this._renderableTilesKeys = []; + this._sourceTileCache = {}; + this.minzoom = 0; + this.maxzoom = 22; + this.tileSize = 512; + this.deltaZoom = 1; + sourceCache.usedForTerrain = true; + sourceCache.tileSize = this.tileSize * 2 ** this.deltaZoom; + } + + destruct() { + this.sourceCache.usedForTerrain = false; + this.sourceCache.tileSize = null; + } + + /** + * Load Terrain Tiles, create internal render-to-texture tiles, free GPU memory. + * @param transform - the operation to do + * @param terrain - the terrain + */ + update(transform: Transform, terrain: Terrain): void { + // load raster-dem tiles for the current scene. + this.sourceCache.update(transform, terrain); + // create internal render-to-texture tiles for the current scene. + this._renderableTilesKeys = []; + const keys = {}; + for (const tileID of transform.coveringTiles({ + tileSize: this.tileSize, + minzoom: this.minzoom, + maxzoom: this.maxzoom, + reparseOverscaled: false, + terrain + })) { + keys[tileID.key] = true; + this._renderableTilesKeys.push(tileID.key); + if (!this._tiles[tileID.key]) { + tileID.posMatrix = new Float64Array(16) as any; + mat4.ortho(tileID.posMatrix, 0, EXTENT, 0, EXTENT, 0, 1); + this._tiles[tileID.key] = new Tile(tileID, this.tileSize); + } + } + // free unused tiles + for (const key in this._tiles) { + if (!keys[key]) delete this._tiles[key]; + } + } + + /** + * Free render to texture cache + * @param tileID - optional, free only corresponding to tileID. + */ + freeRtt(tileID?: OverscaledTileID) { + for (const key in this._tiles) { + const tile = this._tiles[key]; + if (!tileID || tile.tileID.equals(tileID) || tile.tileID.isChildOf(tileID) || tileID.isChildOf(tile.tileID)) + tile.rtt = []; + } + } + + /** + * get a list of tiles, which are loaded and should be rendered in the current scene + * @returns the renderable tiles + */ + getRenderableTiles(): Array { + return this._renderableTilesKeys.map(key => this.getTileByID(key)); + } + + /** + * get terrain tile by the TileID key + * @param id - the tile id + * @returns the tile + */ + getTileByID(id: string): Tile { + return this._tiles[id]; + } + + /** + * Searches for the corresponding current renderable terrain-tiles + * @param tileID - the tile to look for + * @returns the tiles that were found + */ + getTerrainCoords(tileID: OverscaledTileID): Record { + const coords = {}; + for (const key of this._renderableTilesKeys) { + const _tileID = this._tiles[key].tileID; + if (_tileID.canonical.equals(tileID.canonical)) { + const coord = tileID.clone(); + coord.posMatrix = new Float64Array(16) as any; + mat4.ortho(coord.posMatrix, 0, EXTENT, 0, EXTENT, 0, 1); + coords[key] = coord; + } else if (_tileID.canonical.isChildOf(tileID.canonical)) { + const coord = tileID.clone(); + coord.posMatrix = new Float64Array(16) as any; + const dz = _tileID.canonical.z - tileID.canonical.z; + const dx = _tileID.canonical.x - (_tileID.canonical.x >> dz << dz); + const dy = _tileID.canonical.y - (_tileID.canonical.y >> dz << dz); + const size = EXTENT >> dz; + mat4.ortho(coord.posMatrix, 0, size, 0, size, 0, 1); + mat4.translate(coord.posMatrix, coord.posMatrix, [-dx * size, -dy * size, 0]); + coords[key] = coord; + } else if (tileID.canonical.isChildOf(_tileID.canonical)) { + const coord = tileID.clone(); + coord.posMatrix = new Float64Array(16) as any; + const dz = tileID.canonical.z - _tileID.canonical.z; + const dx = tileID.canonical.x - (tileID.canonical.x >> dz << dz); + const dy = tileID.canonical.y - (tileID.canonical.y >> dz << dz); + const size = EXTENT >> dz; + mat4.ortho(coord.posMatrix, 0, EXTENT, 0, EXTENT, 0, 1); + mat4.translate(coord.posMatrix, coord.posMatrix, [dx * size, dy * size, 0]); + mat4.scale(coord.posMatrix, coord.posMatrix, [1 / (2 ** dz), 1 / (2 ** dz), 0]); + coords[key] = coord; + } + } + return coords; + } + + /** + * find the covering raster-dem tile + * @param tileID - the tile to look for + * @param searchForDEM - Optinal parameter to search for (parent) souretiles with loaded dem. + * @returns the tile + */ + getSourceTile(tileID: OverscaledTileID, searchForDEM?: boolean): Tile { + const source = this.sourceCache._source; + let z = tileID.overscaledZ - this.deltaZoom; + if (z > source.maxzoom) z = source.maxzoom; + if (z < source.minzoom) return null; + // cache for tileID to terrain-tileID + if (!this._sourceTileCache[tileID.key]) + this._sourceTileCache[tileID.key] = tileID.scaledTo(z).key; + let tile = this.sourceCache.getTileByID(this._sourceTileCache[tileID.key]); + // during tile-loading phase look if parent tiles (with loaded dem) are available. + if (!(tile && tile.dem) && searchForDEM) + while (z >= source.minzoom && !(tile && tile.dem)) + tile = this.sourceCache.getTileByID(tileID.scaledTo(z--).key); + return tile; + } + + /** + * get a list of tiles, loaded after a spezific time. This is used to update depth & coords framebuffers. + * @param time - the time + * @returns the relevant tiles + */ + tilesAfterTime(time = Date.now()): Array { + return Object.values(this._tiles).filter(t => t.timeAdded >= time); + } +} diff --git a/web/libraries/maplibre-gl/src/source/tile.test.ts b/web/libraries/maplibre-gl/src/source/tile.test.ts new file mode 100644 index 00000000..1fcf9da0 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/tile.test.ts @@ -0,0 +1,290 @@ +import {createSymbolBucket} from '../../test/unit/lib/create_symbol_layer'; +import {Tile} from '../source/tile'; +import {GeoJSONWrapper, Feature} from '../source/geojson_wrapper'; +import {OverscaledTileID} from '../source/tile_id'; +import fs from 'fs'; +import path from 'path'; +import vtpbf from 'vt-pbf'; +import {FeatureIndex} from '../data/feature_index'; +import {CollisionBoxArray} from '../data/array_types.g'; +import {extend} from '../util/util'; +import {serialize, deserialize} from '../util/web_worker_transfer'; + +describe('querySourceFeatures', () => { + const features = [{ + type: 1, + geometry: [0, 0], + tags: {oneway: true} + } as any as Feature]; + + test('geojson tile', () => { + const tile = new Tile(new OverscaledTileID(3, 0, 2, 1, 2), undefined); + let result; + + result = []; + tile.querySourceFeatures(result); + expect(result).toHaveLength(0); + + const geojsonWrapper = new GeoJSONWrapper(features); + geojsonWrapper.name = '_geojsonTileLayer'; + tile.loadVectorData( + createVectorData({rawTileData: vtpbf({layers: {'_geojsonTileLayer': geojsonWrapper}})}), + createPainter() + ); + + result = []; + tile.querySourceFeatures(result); + expect(result).toHaveLength(1); + expect(result[0].geometry.coordinates[0]).toEqual([-90, 0]); + result = []; + tile.querySourceFeatures(result, {} as any); + expect(result).toHaveLength(1); + expect(result[0].properties).toEqual(features[0].tags); + result = []; + tile.querySourceFeatures(result, {sourceLayer: undefined, filter: ['==', 'oneway', true]}); + expect(result).toHaveLength(1); + result = []; + tile.querySourceFeatures(result, {sourceLayer: undefined, filter: ['!=', 'oneway', true]}); + expect(result).toHaveLength(0); + result = []; + const polygon = {type: 'Polygon', coordinates: [[[-91, -1], [-89, -1], [-89, 1], [-91, 1], [-91, -1]]]}; + tile.querySourceFeatures(result, {sourceLayer: undefined, filter: ['within', polygon]}); + expect(result).toHaveLength(1); + }); + + test('empty geojson tile', () => { + const tile = new Tile(new OverscaledTileID(1, 0, 1, 1, 1), undefined); + let result; + + result = []; + tile.querySourceFeatures(result); + expect(result).toHaveLength(0); + + const geojsonWrapper = new GeoJSONWrapper([]); + geojsonWrapper.name = '_geojsonTileLayer'; + + result = []; + expect(() => { tile.querySourceFeatures(result); }).not.toThrow(); + expect(result).toHaveLength(0); + }); + + test('vector tile', () => { + const tile = new Tile(new OverscaledTileID(1, 0, 1, 1, 1), undefined); + let result; + + result = []; + tile.querySourceFeatures(result); + expect(result).toHaveLength(0); + + tile.loadVectorData( + createVectorData({rawTileData: createRawTileData()}), + createPainter() + ); + + result = []; + tile.querySourceFeatures(result, {sourceLayer: 'does-not-exist', filter: undefined}); + expect(result).toHaveLength(0); + + result = []; + tile.querySourceFeatures(result, {sourceLayer: 'road', filter: undefined}); + expect(result).toHaveLength(3); + + result = []; + tile.querySourceFeatures(result, {sourceLayer: 'road', filter: ['==', 'class', 'main']}); + expect(result).toHaveLength(1); + result = []; + tile.querySourceFeatures(result, {sourceLayer: 'road', filter: ['!=', 'class', 'main']}); + expect(result).toHaveLength(2); + + }); + + test('loadVectorData unloads existing data before overwriting it', () => { + const tile = new Tile(new OverscaledTileID(1, 0, 1, 1, 1), undefined); + tile.state = 'loaded'; + const spy = jest.spyOn(tile, 'unloadVectorData'); + const painter = {}; + + tile.loadVectorData(null, painter); + + expect(spy).toHaveBeenCalledWith(); + }); + + test('loadVectorData preserves the most recent rawTileData', () => { + const tile = new Tile(new OverscaledTileID(1, 0, 1, 1, 1), undefined); + tile.state = 'loaded'; + + tile.loadVectorData( + createVectorData({rawTileData: createRawTileData()}), + createPainter() + ); + tile.loadVectorData( + createVectorData(), + createPainter() + ); + + const features = []; + tile.querySourceFeatures(features, {sourceLayer: 'road', filter: undefined}); + expect(features).toHaveLength(3); + + }); + +}); + +describe('Tile#isLessThan', () => { + test('correctly sorts tiles', () => { + const tiles = [ + new OverscaledTileID(9, 0, 9, 146, 195), + new OverscaledTileID(9, 0, 9, 147, 195), + new OverscaledTileID(9, 0, 9, 148, 195), + new OverscaledTileID(9, 0, 9, 149, 195), + new OverscaledTileID(9, 1, 9, 144, 196), + new OverscaledTileID(9, 0, 9, 145, 196), + new OverscaledTileID(9, 0, 9, 146, 196), + new OverscaledTileID(9, 1, 9, 147, 196), + new OverscaledTileID(9, 0, 9, 145, 194), + new OverscaledTileID(9, 0, 9, 149, 196), + new OverscaledTileID(10, 0, 10, 293, 391), + new OverscaledTileID(10, 0, 10, 291, 390), + new OverscaledTileID(10, 1, 10, 293, 390), + new OverscaledTileID(10, 0, 10, 294, 390), + new OverscaledTileID(10, 0, 10, 295, 390), + new OverscaledTileID(10, 0, 10, 291, 391), + ]; + + const sortedTiles = tiles.sort((a, b) => { return a.isLessThan(b) ? -1 : b.isLessThan(a) ? 1 : 0; }); + + expect(sortedTiles).toEqual([ + new OverscaledTileID(9, 0, 9, 145, 194), + new OverscaledTileID(9, 0, 9, 145, 196), + new OverscaledTileID(9, 0, 9, 146, 195), + new OverscaledTileID(9, 0, 9, 146, 196), + new OverscaledTileID(9, 0, 9, 147, 195), + new OverscaledTileID(9, 0, 9, 148, 195), + new OverscaledTileID(9, 0, 9, 149, 195), + new OverscaledTileID(9, 0, 9, 149, 196), + new OverscaledTileID(10, 0, 10, 291, 390), + new OverscaledTileID(10, 0, 10, 291, 391), + new OverscaledTileID(10, 0, 10, 293, 391), + new OverscaledTileID(10, 0, 10, 294, 390), + new OverscaledTileID(10, 0, 10, 295, 390), + new OverscaledTileID(9, 1, 9, 144, 196), + new OverscaledTileID(9, 1, 9, 147, 196), + new OverscaledTileID(10, 1, 10, 293, 390), + ]); + }); +}); + +describe('expiring tiles', () => { + test('regular tiles do not expire', () => { + const tile = new Tile(new OverscaledTileID(1, 0, 1, 1, 1), undefined); + tile.state = 'loaded'; + tile.timeAdded = Date.now(); + + expect(tile.getExpiryTimeout()).toBeFalsy(); + + }); + + test('set, get expiry', () => { + const tile = new Tile(new OverscaledTileID(1, 0, 1, 1, 1), undefined); + tile.state = 'loaded'; + tile.timeAdded = Date.now(); + + tile.setExpiryData({ + cacheControl: 'max-age=60' + }); + + // times are fuzzy, so we'll give this a little leeway: + let expiryTimeout = tile.getExpiryTimeout(); + expect(expiryTimeout >= 56000 && expiryTimeout <= 60000).toBeTruthy(); + + const date = new Date(); + date.setMinutes(date.getMinutes() + 10); + date.setMilliseconds(0); + + tile.setExpiryData({ + expires: date.toString() + }); + + expiryTimeout = tile.getExpiryTimeout(); + expect(expiryTimeout > 598000 && expiryTimeout < 600000).toBeTruthy(); + + }); + + test('exponential backoff handling', () => { + const tile = new Tile(new OverscaledTileID(1, 0, 1, 1, 1), undefined); + tile.state = 'loaded'; + tile.timeAdded = Date.now(); + + tile.setExpiryData({ + cacheControl: 'max-age=10' + }); + + const expiryTimeout = tile.getExpiryTimeout(); + expect(expiryTimeout >= 8000 && expiryTimeout <= 10000).toBeTruthy(); + + const justNow = new Date(); + justNow.setSeconds(justNow.getSeconds() - 1); + + // every time we set a tile's expiration to a date already expired, + // it assumes it comes from a new HTTP response, so this is counted + // as an extra expired tile request + tile.setExpiryData({ + expires: justNow + }); + expect(tile.getExpiryTimeout()).toBe(1000); + + tile.setExpiryData({ + expires: justNow + }); + expect(tile.getExpiryTimeout()).toBe(2000); + tile.setExpiryData({ + expires: justNow + }); + expect(tile.getExpiryTimeout()).toBe(4000); + + tile.setExpiryData({ + expires: justNow + }); + expect(tile.getExpiryTimeout()).toBe(8000); + + }); + +}); + +describe('rtl text detection', () => { + test('Tile#hasRTLText is true when a tile loads a symbol bucket with rtl text', () => { + const tile = new Tile(new OverscaledTileID(1, 0, 1, 1, 1), undefined); + // Create a stub symbol bucket + const symbolBucket = createSymbolBucket('test', 'Test', 'test', new CollisionBoxArray()); + // symbolBucket has not been populated yet so we force override the value in the stub + symbolBucket.hasRTLText = true; + tile.loadVectorData( + createVectorData({rawTileData: createRawTileData(), buckets: [symbolBucket]}), + createPainter({ + getLayer() { + return symbolBucket.layers[0]; + } + }) + ); + + expect(tile.hasRTLText).toBeTruthy(); + }); + +}); + +function createRawTileData() { + return fs.readFileSync(path.join(__dirname, '../../test/unit/assets/mbsv5-6-18-23.vector.pbf')); +} + +function createVectorData(options?) { + const collisionBoxArray = new CollisionBoxArray(); + return extend({ + collisionBoxArray: deserialize(serialize(collisionBoxArray)), + featureIndex: deserialize(serialize(new FeatureIndex(new OverscaledTileID(1, 0, 1, 1, 1)))), + buckets: [] + }, options); +} + +function createPainter(styleStub = {}) { + return {style: styleStub}; +} diff --git a/web/libraries/maplibre-gl/src/source/tile.ts b/web/libraries/maplibre-gl/src/source/tile.ts new file mode 100644 index 00000000..cc180ec5 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/tile.ts @@ -0,0 +1,479 @@ +import {uniqueId, parseCacheControl} from '../util/util'; +import {deserialize as deserializeBucket} from '../data/bucket'; +import '../data/feature_index'; +import type {FeatureIndex} from '../data/feature_index'; +import {GeoJSONFeature} from '../util/vectortile_to_geojson'; +import {featureFilter} from '@maplibre/maplibre-gl-style-spec'; +import {SymbolBucket} from '../data/bucket/symbol_bucket'; +import {CollisionBoxArray} from '../data/array_types.g'; +import {Texture} from '../render/texture'; +import {browser} from '../util/browser'; +import {toEvaluationFeature} from '../data/evaluation_feature'; +import {EvaluationParameters} from '../style/evaluation_parameters'; +import {SourceFeatureState} from '../source/source_state'; +import {lazyLoadRTLTextPlugin} from './rtl_text_plugin'; + +const CLOCK_SKEW_RETRY_TIMEOUT = 30000; + +import type {Bucket} from '../data/bucket'; +import type {StyleLayer} from '../style/style_layer'; +import type {WorkerTileResult} from './worker_source'; +import type {Actor} from '../util/actor'; +import type {DEMData} from '../data/dem_data'; +import type {AlphaImage} from '../util/image'; +import type {ImageAtlas} from '../render/image_atlas'; +import type {ImageManager} from '../render/image_manager'; +import type {Context} from '../gl/context'; +import type {OverscaledTileID} from './tile_id'; +import type {Framebuffer} from '../gl/framebuffer'; +import type {Transform} from '../geo/transform'; +import type {LayerFeatureStates} from './source_state'; +import type {Cancelable} from '../types/cancelable'; +import type {FilterSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type Point from '@mapbox/point-geometry'; +import {mat4} from 'gl-matrix'; +import type {VectorTileLayer} from '@mapbox/vector-tile'; +import {ExpiryData} from '../util/ajax'; + +/** + * The tile's state, can be: + * - `loading` Tile data is in the process of loading. + * - `loaded` Tile data has been loaded. Tile can be rendered. + * - `reloading` Tile data has been loaded and is being updated. Tile can be rendered. + * - `unloaded` Tile data has been deleted. + * - `errored` Tile data was not loaded because of an error. + * - `expired` Tile data was previously loaded, but has expired per its HTTP headers and is in the process of refreshing. + */ +export type TileState = 'loading' | 'loaded' | 'reloading' | 'unloaded' | 'errored' | 'expired'; + +/** + * @internal + * A tile object is the combination of a Coordinate, which defines + * its place, as well as a unique ID and data tracking for its content + */ +export class Tile { + tileID: OverscaledTileID; + uid: number; + uses: number; + tileSize: number; + buckets: {[_: string]: Bucket}; + latestFeatureIndex: FeatureIndex; + latestRawTileData: ArrayBuffer; + imageAtlas: ImageAtlas; + imageAtlasTexture: Texture; + glyphAtlasImage: AlphaImage; + glyphAtlasTexture: Texture; + expirationTime: any; + expiredRequestCount: number; + state: TileState; + timeAdded: number = 0; + fadeEndTime: number = 0; + collisionBoxArray: CollisionBoxArray; + redoWhenDone: boolean; + showCollisionBoxes: boolean; + placementSource: any; + actor: Actor; + vtLayers: {[_: string]: VectorTileLayer}; + + neighboringTiles: any; + dem: DEMData; + demMatrix: mat4; + aborted: boolean; + needsHillshadePrepare: boolean; + needsTerrainPrepare: boolean; + request: Cancelable; + texture: any; + fbo: Framebuffer; + demTexture: Texture; + refreshedUponExpiration: boolean; + reloadCallback: any; + resourceTiming: Array; + queryPadding: number; + + symbolFadeHoldUntil: number; + hasSymbolBuckets: boolean; + hasRTLText: boolean; + dependencies: any; + rtt: Array<{id: number; stamp: number}>; + rttCoords: {[_:string]: string}; + + /** + * @param tileID - the tile ID + * @param size - The tile size + */ + constructor(tileID: OverscaledTileID, size: number) { + this.tileID = tileID; + this.uid = uniqueId(); + this.uses = 0; + this.tileSize = size; + this.buckets = {}; + this.expirationTime = null; + this.queryPadding = 0; + this.hasSymbolBuckets = false; + this.hasRTLText = false; + this.dependencies = {}; + this.rtt = []; + this.rttCoords = {}; + + // Counts the number of times a response was already expired when + // received. We're using this to add a delay when making a new request + // so we don't have to keep retrying immediately in case of a server + // serving expired tiles. + this.expiredRequestCount = 0; + + this.state = 'loading'; + } + + registerFadeDuration(duration: number) { + const fadeEndTime = duration + this.timeAdded; + + if (fadeEndTime < this.fadeEndTime) { + return; + } + + this.fadeEndTime = fadeEndTime; + } + + wasRequested() { + return this.state === 'errored' || this.state === 'loaded' || this.state === 'reloading'; + } + + clearTextures(painter: any) { + if (this.demTexture) painter.saveTileTexture(this.demTexture); + this.demTexture = null; + } + + /** + * Given a data object with a 'buffers' property, load it into + * this tile's elementGroups and buffers properties and set loaded + * to true. If the data is null, like in the case of an empty + * GeoJSON tile, no-op but still set loaded to true. + * @param data - The data from the worker + * @param painter - the painter + * @param justReloaded - `true` to just reload + */ + loadVectorData(data: WorkerTileResult, painter: any, justReloaded?: boolean | null) { + if (this.hasData()) { + this.unloadVectorData(); + } + + this.state = 'loaded'; + + // empty GeoJSON tile + if (!data) { + this.collisionBoxArray = new CollisionBoxArray(); + return; + } + + if (data.featureIndex) { + this.latestFeatureIndex = data.featureIndex; + if (data.rawTileData) { + // Only vector tiles have rawTileData, and they won't update it for + // 'reloadTile' + this.latestRawTileData = data.rawTileData; + this.latestFeatureIndex.rawTileData = data.rawTileData; + } else if (this.latestRawTileData) { + // If rawTileData hasn't updated, hold onto a pointer to the last + // one we received + this.latestFeatureIndex.rawTileData = this.latestRawTileData; + } + } + this.collisionBoxArray = data.collisionBoxArray; + this.buckets = deserializeBucket(data.buckets, painter.style); + + this.hasSymbolBuckets = false; + for (const id in this.buckets) { + const bucket = this.buckets[id]; + if (bucket instanceof SymbolBucket) { + this.hasSymbolBuckets = true; + if (justReloaded) { + bucket.justReloaded = true; + } else { + break; + } + } + } + + this.hasRTLText = false; + if (this.hasSymbolBuckets) { + for (const id in this.buckets) { + const bucket = this.buckets[id]; + if (bucket instanceof SymbolBucket) { + if (bucket.hasRTLText) { + this.hasRTLText = true; + lazyLoadRTLTextPlugin(); + break; + } + } + } + } + + this.queryPadding = 0; + for (const id in this.buckets) { + const bucket = this.buckets[id]; + this.queryPadding = Math.max(this.queryPadding, painter.style.getLayer(id).queryRadius(bucket)); + } + + if (data.imageAtlas) { + this.imageAtlas = data.imageAtlas; + } + if (data.glyphAtlasImage) { + this.glyphAtlasImage = data.glyphAtlasImage; + } + } + + /** + * Release any data or WebGL resources referenced by this tile. + */ + unloadVectorData() { + for (const id in this.buckets) { + this.buckets[id].destroy(); + } + this.buckets = {}; + + if (this.imageAtlasTexture) { + this.imageAtlasTexture.destroy(); + } + + if (this.imageAtlas) { + this.imageAtlas = null; + } + + if (this.glyphAtlasTexture) { + this.glyphAtlasTexture.destroy(); + } + + this.latestFeatureIndex = null; + this.state = 'unloaded'; + } + + getBucket(layer: StyleLayer) { + return this.buckets[layer.id]; + } + + upload(context: Context) { + for (const id in this.buckets) { + const bucket = this.buckets[id]; + if (bucket.uploadPending()) { + bucket.upload(context); + } + } + + const gl = context.gl; + if (this.imageAtlas && !this.imageAtlas.uploaded) { + this.imageAtlasTexture = new Texture(context, this.imageAtlas.image, gl.RGBA); + this.imageAtlas.uploaded = true; + } + + if (this.glyphAtlasImage) { + this.glyphAtlasTexture = new Texture(context, this.glyphAtlasImage, gl.ALPHA); + this.glyphAtlasImage = null; + } + } + + prepare(imageManager: ImageManager) { + if (this.imageAtlas) { + this.imageAtlas.patchUpdatedImages(imageManager, this.imageAtlasTexture); + } + } + + // Queries non-symbol features rendered for this tile. + // Symbol features are queried globally + queryRenderedFeatures( + layers: {[_: string]: StyleLayer}, + serializedLayers: {[_: string]: any}, + sourceFeatureState: SourceFeatureState, + queryGeometry: Array, + cameraQueryGeometry: Array, + scale: number, + params: { + filter: FilterSpecification; + layers: Array; + availableImages: Array; + }, + transform: Transform, + maxPitchScaleFactor: number, + pixelPosMatrix: mat4 + ): {[_: string]: Array<{featureIndex: number; feature: GeoJSONFeature}>} { + if (!this.latestFeatureIndex || !this.latestFeatureIndex.rawTileData) + return {}; + + return this.latestFeatureIndex.query({ + queryGeometry, + cameraQueryGeometry, + scale, + tileSize: this.tileSize, + pixelPosMatrix, + transform, + params, + queryPadding: this.queryPadding * maxPitchScaleFactor + }, layers, serializedLayers, sourceFeatureState); + } + + querySourceFeatures(result: Array, params?: { + sourceLayer?: string; + filter?: FilterSpecification; + validate?: boolean; + }) { + const featureIndex = this.latestFeatureIndex; + if (!featureIndex || !featureIndex.rawTileData) return; + + const vtLayers = featureIndex.loadVTLayers(); + + const sourceLayer = params && params.sourceLayer ? params.sourceLayer : ''; + const layer = vtLayers._geojsonTileLayer || vtLayers[sourceLayer]; + + if (!layer) return; + + const filter = featureFilter(params && params.filter); + const {z, x, y} = this.tileID.canonical; + const coord = {z, x, y}; + + for (let i = 0; i < layer.length; i++) { + const feature = layer.feature(i); + if (filter.needGeometry) { + const evaluationFeature = toEvaluationFeature(feature, true); + if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), evaluationFeature, this.tileID.canonical)) continue; + } else if (!filter.filter(new EvaluationParameters(this.tileID.overscaledZ), feature)) { + continue; + } + const id = featureIndex.getId(feature, sourceLayer); + const geojsonFeature = new GeoJSONFeature(feature, z, x, y, id); + (geojsonFeature as any).tile = coord; + result.push(geojsonFeature); + } + } + + hasData() { + return this.state === 'loaded' || this.state === 'reloading' || this.state === 'expired'; + } + + patternsLoaded() { + return this.imageAtlas && !!Object.keys(this.imageAtlas.patternPositions).length; + } + + setExpiryData(data: ExpiryData) { + const prior = this.expirationTime; + + if (data.cacheControl) { + const parsedCC = parseCacheControl(data.cacheControl); + if (parsedCC['max-age']) this.expirationTime = Date.now() + parsedCC['max-age'] * 1000; + } else if (data.expires) { + this.expirationTime = new Date(data.expires).getTime(); + } + + if (this.expirationTime) { + const now = Date.now(); + let isExpired = false; + + if (this.expirationTime > now) { + isExpired = false; + } else if (!prior) { + isExpired = true; + } else if (this.expirationTime < prior) { + // Expiring date is going backwards: + // fall back to exponential backoff + isExpired = true; + + } else { + const delta = this.expirationTime - prior; + + if (!delta) { + // Server is serving the same expired resource over and over: fall + // back to exponential backoff. + isExpired = true; + + } else { + // Assume that either the client or the server clock is wrong and + // try to interpolate a valid expiration date (from the client POV) + // observing a minimum timeout. + this.expirationTime = now + Math.max(delta, CLOCK_SKEW_RETRY_TIMEOUT); + + } + } + + if (isExpired) { + this.expiredRequestCount++; + this.state = 'expired'; + } else { + this.expiredRequestCount = 0; + } + } + } + + getExpiryTimeout() { + if (this.expirationTime) { + if (this.expiredRequestCount) { + return 1000 * (1 << Math.min(this.expiredRequestCount - 1, 31)); + } else { + // Max value for `setTimeout` implementations is a 32 bit integer; cap this accordingly + return Math.min(this.expirationTime - new Date().getTime(), Math.pow(2, 31) - 1); + } + } + } + + setFeatureState(states: LayerFeatureStates, painter: any) { + if (!this.latestFeatureIndex || + !this.latestFeatureIndex.rawTileData || + Object.keys(states).length === 0) { + return; + } + + const vtLayers = this.latestFeatureIndex.loadVTLayers(); + + for (const id in this.buckets) { + if (!painter.style.hasLayer(id)) continue; + + const bucket = this.buckets[id]; + // Buckets are grouped by common source-layer + const sourceLayerId = bucket.layers[0]['sourceLayer'] || '_geojsonTileLayer'; + const sourceLayer = vtLayers[sourceLayerId]; + const sourceLayerStates = states[sourceLayerId]; + if (!sourceLayer || !sourceLayerStates || Object.keys(sourceLayerStates).length === 0) continue; + + bucket.update(sourceLayerStates, sourceLayer, this.imageAtlas && this.imageAtlas.patternPositions || {}); + const layer = painter && painter.style && painter.style.getLayer(id); + if (layer) { + this.queryPadding = Math.max(this.queryPadding, layer.queryRadius(bucket)); + } + } + } + + holdingForFade(): boolean { + return this.symbolFadeHoldUntil !== undefined; + } + + symbolFadeFinished(): boolean { + return !this.symbolFadeHoldUntil || this.symbolFadeHoldUntil < browser.now(); + } + + clearFadeHold() { + this.symbolFadeHoldUntil = undefined; + } + + setHoldDuration(duration: number) { + this.symbolFadeHoldUntil = browser.now() + duration; + } + + setDependencies(namespace: string, dependencies: Array) { + const index = {}; + for (const dep of dependencies) { + index[dep] = true; + } + this.dependencies[namespace] = index; + } + + hasDependency(namespaces: Array, keys: Array) { + for (const namespace of namespaces) { + const dependencies = this.dependencies[namespace]; + if (dependencies) { + for (const key of keys) { + if (dependencies[key]) { + return true; + } + } + } + } + return false; + } +} diff --git a/web/libraries/maplibre-gl/src/source/tile_bounds.ts b/web/libraries/maplibre-gl/src/source/tile_bounds.ts new file mode 100644 index 00000000..9a90c803 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/tile_bounds.ts @@ -0,0 +1,34 @@ +import {LngLatBounds, LngLatBoundsLike} from '../geo/lng_lat_bounds'; +import {mercatorXfromLng, mercatorYfromLat} from '../geo/mercator_coordinate'; + +import type {CanonicalTileID} from './tile_id'; + +export class TileBounds { + bounds: LngLatBounds; + minzoom: number; + maxzoom: number; + + constructor(bounds: [number, number, number, number], minzoom?: number | null, maxzoom?: number | null) { + this.bounds = LngLatBounds.convert(this.validateBounds(bounds)); + this.minzoom = minzoom || 0; + this.maxzoom = maxzoom || 24; + } + + validateBounds(bounds: [number, number, number, number]): LngLatBoundsLike { + // make sure the bounds property contains valid longitude and latitudes + if (!Array.isArray(bounds) || bounds.length !== 4) return [-180, -90, 180, 90]; + return [Math.max(-180, bounds[0]), Math.max(-90, bounds[1]), Math.min(180, bounds[2]), Math.min(90, bounds[3])]; + } + + contains(tileID: CanonicalTileID) { + const worldSize = Math.pow(2, tileID.z); + const level = { + minX: Math.floor(mercatorXfromLng(this.bounds.getWest()) * worldSize), + minY: Math.floor(mercatorYfromLat(this.bounds.getNorth()) * worldSize), + maxX: Math.ceil(mercatorXfromLng(this.bounds.getEast()) * worldSize), + maxY: Math.ceil(mercatorYfromLat(this.bounds.getSouth()) * worldSize) + }; + const hit = tileID.x >= level.minX && tileID.x < level.maxX && tileID.y >= level.minY && tileID.y < level.maxY; + return hit; + } +} diff --git a/web/libraries/maplibre-gl/src/source/tile_cache.test.ts b/web/libraries/maplibre-gl/src/source/tile_cache.test.ts new file mode 100644 index 00000000..ee877085 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/tile_cache.test.ts @@ -0,0 +1,133 @@ +import {Tile} from './tile'; +import {TileCache} from './tile_cache'; +import {OverscaledTileID} from './tile_id'; + +const idA = new OverscaledTileID(10, 0, 10, 0, 1); +const idB = new OverscaledTileID(10, 0, 10, 0, 2); +const idC = new OverscaledTileID(10, 0, 10, 0, 3); +const idD = new OverscaledTileID(10, 0, 10, 0, 4); +const tileA = {tileID: idA} as Tile; +const tileA2 = {tileID: idA} as Tile; +const tileB = {tileID: idB} as Tile; +const tileC = {tileID: idC} as Tile; +const tileD = {tileID: idD} as Tile; + +function keysExpected(cache, ids) { + expect(cache.order).toEqual(ids.map((id) => id.key)); +} +describe('TileCache', () => { + test('complex flow', () => { + const cache = new TileCache(10, (removed) => { + expect(removed).toBe('dc'); + }); + expect(cache.getAndRemove(idC)).toBeNull(); + expect(cache.add(idA, tileA)).toBe(cache); + keysExpected(cache, [idA]); + expect(cache.has(idA)).toBe(true); + expect(cache.getAndRemove(idA)).toBe(tileA); + expect(cache.getAndRemove(idA)).toBeNull(); + expect(cache.has(idA)).toBe(false); + keysExpected(cache, []); + }); + + test('get without removing', done => { + const cache = new TileCache(10, () => { + done('test "get without removing" failed'); + }); + expect(cache.add(idA, tileA)).toBe(cache); + expect(cache.get(idA)).toBe(tileA); + keysExpected(cache, [idA]); + expect(cache.get(idA)).toBe(tileA); + done(); + }); + + test('duplicate add', done => { + const cache = new TileCache(10, () => { + done('test "duplicate add" failed'); + }); + + cache.add(idA, tileA); + cache.add(idA, tileA2); + + keysExpected(cache, [idA, idA]); + expect(cache.has(idA)).toBeTruthy(); + expect(cache.getAndRemove(idA)).toBe(tileA); + expect(cache.has(idA)).toBeTruthy(); + expect(cache.getAndRemove(idA)).toBe(tileA2); + done(); + }); + + test('expiry', () => { + const cache = new TileCache(10, (removed) => { + expect(cache.has(idB)).toBeTruthy(); + expect(removed).toBe(tileA2); + }); + + cache.add(idB, tileB, 0); + cache.getAndRemove(idB); + // removing clears the expiry timeout + cache.add(idB, null); + + cache.add(idA, tileA); + cache.add(idA, tileA2, 0); // expires immediately and `onRemove` is called. + }); + + test('remove', () => { + const cache = new TileCache(10, () => {}); + + cache.add(idA, tileA); + cache.add(idB, tileB); + cache.add(idC, tileC); + + keysExpected(cache, [idA, idB, idC]); + expect(cache.has(idB)).toBeTruthy(); + + cache.remove(idB); + + keysExpected(cache, [idA, idC]); + expect(cache.has(idB)).toBeFalsy(); + + expect(cache.remove(idB)).toBeTruthy(); + + }); + + test('overflow', () => { + const cache = new TileCache(1, (removed) => { + expect(removed).toBe(tileA); + }); + cache.add(idA, tileA); + cache.add(idB, tileB); + + expect(cache.has(idB)).toBeTruthy(); + expect(cache.has(idA)).toBeFalsy(); + }); + + test('.reset', () => { + let called; + const cache = new TileCache(10, (removed) => { + expect(removed).toBe(tileA); + called = true; + }); + cache.add(idA, tileA); + expect(cache.reset()).toBe(cache); + expect(cache.has(idA)).toBe(false); + expect(called).toBeTruthy(); + }); + + test('.setMaxSize', () => { + let numRemoved = 0; + const cache = new TileCache(10, () => { + numRemoved++; + }); + cache.add(idA, tileA); + cache.add(idB, tileB); + cache.add(idC, tileC); + expect(numRemoved).toBe(0); + cache.setMaxSize(15); + expect(numRemoved).toBe(0); + cache.setMaxSize(1); + expect(numRemoved).toBe(2); + cache.add(idD, tileD); + expect(numRemoved).toBe(3); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/tile_cache.ts b/web/libraries/maplibre-gl/src/source/tile_cache.ts new file mode 100644 index 00000000..b05d9ed2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/tile_cache.ts @@ -0,0 +1,208 @@ +import {OverscaledTileID} from './tile_id'; +import type {Tile} from './tile'; + +/** + * @internal + * A [least-recently-used cache](http://en.wikipedia.org/wiki/Cache_algorithms) + * with hash lookup made possible by keeping a list of keys in parallel to + * an array of dictionary of values + */ +export class TileCache { + max: number; + data: { + [key: string]: Array<{ + value: Tile; + timeout: ReturnType; + }>; + }; + order: Array; + onRemove: (element: Tile) => void; + /** + * @param max - number of permitted values + * @param onRemove - callback called with items when they expire + */ + constructor(max: number, onRemove: (element: Tile) => void) { + this.max = max; + this.onRemove = onRemove; + this.reset(); + } + + /** + * Clear the cache + * + * @returns this cache + */ + reset() { + for (const key in this.data) { + for (const removedData of this.data[key]) { + if (removedData.timeout) clearTimeout(removedData.timeout); + this.onRemove(removedData.value); + } + } + + this.data = {}; + this.order = []; + + return this; + } + + /** + * Add a key, value combination to the cache, trimming its size if this pushes + * it over max length. + * + * @param tileID - lookup key for the item + * @param data - tile data + * + * @returns this cache + */ + add(tileID: OverscaledTileID, data: Tile, expiryTimeout: number | void) { + const key = tileID.wrapped().key; + if (this.data[key] === undefined) { + this.data[key] = []; + } + + const dataWrapper = { + value: data, + timeout: undefined + }; + + if (expiryTimeout !== undefined) { + dataWrapper.timeout = setTimeout(() => { + this.remove(tileID, dataWrapper); + }, expiryTimeout as number); + } + + this.data[key].push(dataWrapper); + this.order.push(key); + + if (this.order.length > this.max) { + const removedData = this._getAndRemoveByKey(this.order[0]); + if (removedData) this.onRemove(removedData); + } + + return this; + } + + /** + * Determine whether the value attached to `key` is present + * + * @param tileID - the key to be looked-up + * @returns whether the cache has this value + */ + has(tileID: OverscaledTileID): boolean { + return tileID.wrapped().key in this.data; + } + + /** + * Get the value attached to a specific key and remove data from cache. + * If the key is not found, returns `null` + * + * @param tileID - the key to look up + * @returns the tile data, or null if it isn't found + */ + getAndRemove(tileID: OverscaledTileID): Tile { + if (!this.has(tileID)) { return null; } + return this._getAndRemoveByKey(tileID.wrapped().key); + } + + /* + * Get and remove the value with the specified key. + */ + _getAndRemoveByKey(key: string): Tile { + const data = this.data[key].shift(); + if (data.timeout) clearTimeout(data.timeout); + + if (this.data[key].length === 0) { + delete this.data[key]; + } + this.order.splice(this.order.indexOf(key), 1); + + return data.value; + } + + /* + * Get the value with the specified (wrapped tile) key. + */ + getByKey(key: string): Tile { + const data = this.data[key]; + return data ? data[0].value : null; + } + + /** + * Get the value attached to a specific key without removing data + * from the cache. If the key is not found, returns `null` + * + * @param tileID - the key to look up + * @returns the tile data, or null if it isn't found + */ + get(tileID: OverscaledTileID): Tile { + if (!this.has(tileID)) { return null; } + + const data = this.data[tileID.wrapped().key][0]; + return data.value; + } + + /** + * Remove a key/value combination from the cache. + * + * @param tileID - the key for the pair to delete + * @param value - If a value is provided, remove that exact version of the value. + * @returns this cache + */ + remove(tileID: OverscaledTileID, value?: { + value: Tile; + timeout: ReturnType; + }) { + if (!this.has(tileID)) { return this; } + const key = tileID.wrapped().key; + + const dataIndex = value === undefined ? 0 : this.data[key].indexOf(value); + const data = this.data[key][dataIndex]; + this.data[key].splice(dataIndex, 1); + if (data.timeout) clearTimeout(data.timeout); + if (this.data[key].length === 0) { + delete this.data[key]; + } + this.onRemove(data.value); + this.order.splice(this.order.indexOf(key), 1); + + return this; + } + + /** + * Change the max size of the cache. + * + * @param max - the max size of the cache + * @returns this cache + */ + setMaxSize(max: number): TileCache { + this.max = max; + + while (this.order.length > this.max) { + const removedData = this._getAndRemoveByKey(this.order[0]); + if (removedData) this.onRemove(removedData); + } + + return this; + } + + /** + * Remove entries that do not pass a filter function. Used for removing + * stale tiles from the cache. + * + * @param filterFn - Determines whether the tile is filtered. If the supplied function returns false, the tile will be filtered out. + */ + filter(filterFn: (tile: Tile) => boolean) { + const removed = []; + for (const key in this.data) { + for (const entry of this.data[key]) { + if (!filterFn(entry.value)) { + removed.push(entry); + } + } + } + for (const r of removed) { + this.remove(r.value.tileID, r); + } + } +} diff --git a/web/libraries/maplibre-gl/src/source/tile_id.test.ts b/web/libraries/maplibre-gl/src/source/tile_id.test.ts new file mode 100644 index 00000000..1f4fdec0 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/tile_id.test.ts @@ -0,0 +1,111 @@ +import {CanonicalTileID, OverscaledTileID} from '../source/tile_id'; + +describe('CanonicalTileID', () => { + test('#constructor', () => { + expect(() => { + /*eslint no-new: 0*/ + new CanonicalTileID(-1, 0, 0); + }).toThrow(); + expect(() => { + /*eslint no-new: 0*/ + new CanonicalTileID(26, 0, 0); + }).toThrow(); + expect(() => { + /*eslint no-new: 0*/ + new CanonicalTileID(2, 4, 0); + }).toThrow(); + expect(() => { + /*eslint no-new: 0*/ + new CanonicalTileID(2, 0, 4); + }).toThrow(); + }); + + test('.key', () => { + expect(new CanonicalTileID(0, 0, 0).key).toBe('000'); + expect(new CanonicalTileID(1, 0, 0).key).toBe('011'); + expect(new CanonicalTileID(1, 1, 0).key).toBe('111'); + expect(new CanonicalTileID(1, 1, 1).key).toBe('311'); + }); + + test('.equals', () => { + expect(new CanonicalTileID(3, 2, 1).equals(new CanonicalTileID(3, 2, 1))).toBeTruthy(); + expect(new CanonicalTileID(9, 2, 3).equals(new CanonicalTileID(3, 2, 1))).toBeFalsy(); + }); + + test('.url replaces {z}/{x}/{y}', () => { + expect(new CanonicalTileID(2, 1, 0).url(['{z}/{x}/{y}.json'], 1)).toBe('2/1/0.json'); + }); + + test('.url replaces {quadkey}', () => { + expect(new CanonicalTileID(1, 0, 0).url(['quadkey={quadkey}'], 1)).toBe('quadkey=0'); + expect(new CanonicalTileID(2, 0, 0).url(['quadkey={quadkey}'], 1)).toBe('quadkey=00'); + expect(new CanonicalTileID(2, 1, 1).url(['quadkey={quadkey}'], 1)).toBe('quadkey=03'); + expect(new CanonicalTileID(17, 22914, 52870).url(['quadkey={quadkey}'], 1)).toBe('quadkey=02301322130000230'); + + // Test case confirmed by quadkeytools package + expect(new CanonicalTileID(6, 29, 3).url(['quadkey={quadkey}'], 1)).toBe('quadkey=011123'); + + }); + + test('.url replaces {bbox-epsg-3857}', () => { + expect(new CanonicalTileID(1, 0, 0).url(['bbox={bbox-epsg-3857}'], 1)).toBe('bbox=-20037508.342789244,0,0,20037508.342789244'); + }); + + test('.url replaces {ratio}', () => { + expect(new CanonicalTileID(1, 0, 0).url(['r={ratio}'], 2)).toBe('r=@2x'); + expect(new CanonicalTileID(1, 0, 0).url(['r={ratio}'], 1)).toBe('r='); + }); + + //Tests that multiple values of the same placeholder are replaced. + test('.url replaces {z}/{x}/{y}/{z}/{x}/{y}', () => { + expect(new CanonicalTileID(2, 1, 0).url(['{z}/{x}/{y}/{z}/{x}/{y}.json'], 1)).toBe('2/1/0/2/1/0.json'); + }); + +}); + +describe('OverscaledTileID', () => { + test('#constructor', () => { + expect(new OverscaledTileID(0, 0, 0, 0, 0) instanceof OverscaledTileID).toBeTruthy(); + expect(() => { + /*eslint no-new: 0*/ + new OverscaledTileID(7, 0, 8, 0, 0); + }).toThrow(); + }); + + test('.key', () => { + expect(new OverscaledTileID(0, 0, 0, 0, 0).key).toBe('000'); + expect(new OverscaledTileID(1, 0, 1, 0, 0).key).toBe('011'); + expect(new OverscaledTileID(1, 0, 1, 1, 0).key).toBe('111'); + expect(new OverscaledTileID(1, 0, 1, 1, 1).key).toBe('311'); + expect(new OverscaledTileID(1, -1, 1, 1, 1).key).toBe('711'); + }); + + test('.toString', () => { + expect(new OverscaledTileID(1, 0, 1, 1, 1).toString()).toBe('1/1/1'); + }); + + test('.children', () => { + expect(new OverscaledTileID(0, 0, 0, 0, 0).children(25)).toEqual([ + new OverscaledTileID(1, 0, 1, 0, 0), + new OverscaledTileID(1, 0, 1, 1, 0), + new OverscaledTileID(1, 0, 1, 0, 1), + new OverscaledTileID(1, 0, 1, 1, 1)]); + expect(new OverscaledTileID(0, 0, 0, 0, 0).children(0)).toEqual([new OverscaledTileID(1, 0, 0, 0, 0)]); + }); + + test('.scaledTo returns a parent', () => { + expect(new OverscaledTileID(2, 0, 2, 0, 0).scaledTo(0)).toEqual(new OverscaledTileID(0, 0, 0, 0, 0)); + expect(new OverscaledTileID(1, 0, 1, 0, 0).scaledTo(0)).toEqual(new OverscaledTileID(0, 0, 0, 0, 0)); + expect(new OverscaledTileID(1, 0, 0, 0, 0).scaledTo(0)).toEqual(new OverscaledTileID(0, 0, 0, 0, 0)); + }); + + test('.isChildOf', () => { + expect( + new OverscaledTileID(2, 0, 2, 0, 0).isChildOf(new OverscaledTileID(0, 0, 0, 0, 0)) + ).toBeTruthy(); + expect( + new OverscaledTileID(2, 0, 2, 0, 0).isChildOf(new OverscaledTileID(0, 1, 0, 0, 0)) + ).toBeFalsy(); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/source/tile_id.ts b/web/libraries/maplibre-gl/src/source/tile_id.ts new file mode 100644 index 00000000..62c4fa99 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/tile_id.ts @@ -0,0 +1,219 @@ +import {getTileBBox} from '@mapbox/whoots-js'; +import {EXTENT} from '../data/extent'; +import Point from '@mapbox/point-geometry'; +import {MercatorCoordinate} from '../geo/mercator_coordinate'; +import {register} from '../util/web_worker_transfer'; +import {mat4} from 'gl-matrix'; +import {ICanonicalTileID, IMercatorCoordinate} from '@maplibre/maplibre-gl-style-spec'; + +/** + * A canonical way to define a tile ID + */ +export class CanonicalTileID implements ICanonicalTileID { + z: number; + x: number; + y: number; + key: string; + + constructor(z: number, x: number, y: number) { + + if (z < 0 || z > 25 || y < 0 || y >= Math.pow(2, z) || x < 0 || x >= Math.pow(2, z)) { + throw new Error(`x=${x}, y=${y}, z=${z} outside of bounds. 0<=x<${Math.pow(2, z)}, 0<=y<${Math.pow(2, z)} 0<=z<=25 `); + } + + this.z = z; + this.x = x; + this.y = y; + this.key = calculateKey(0, z, z, x, y); + } + + equals(id: ICanonicalTileID) { + return this.z === id.z && this.x === id.x && this.y === id.y; + } + + // given a list of urls, choose a url template and return a tile URL + url(urls: Array, pixelRatio: number, scheme?: string | null) { + const bbox = getTileBBox(this.x, this.y, this.z); + const quadkey = getQuadkey(this.z, this.x, this.y); + + return urls[(this.x + this.y) % urls.length] + .replace(/{prefix}/g, (this.x % 16).toString(16) + (this.y % 16).toString(16)) + .replace(/{z}/g, String(this.z)) + .replace(/{x}/g, String(this.x)) + .replace(/{y}/g, String(scheme === 'tms' ? (Math.pow(2, this.z) - this.y - 1) : this.y)) + .replace(/{ratio}/g, pixelRatio > 1 ? '@2x' : '') + .replace(/{quadkey}/g, quadkey) + .replace(/{bbox-epsg-3857}/g, bbox); + } + + isChildOf(parent: ICanonicalTileID) { + const dz = this.z - parent.z; + return dz > 0 && parent.x === (this.x >> dz) && parent.y === (this.y >> dz); + } + + getTilePoint(coord: IMercatorCoordinate) { + const tilesAtZoom = Math.pow(2, this.z); + return new Point( + (coord.x * tilesAtZoom - this.x) * EXTENT, + (coord.y * tilesAtZoom - this.y) * EXTENT); + } + + toString() { + return `${this.z}/${this.x}/${this.y}`; + } +} + +/** + * @internal + * An unwrapped tile identifier + */ +export class UnwrappedTileID { + wrap: number; + canonical: CanonicalTileID; + key: string; + + constructor(wrap: number, canonical: CanonicalTileID) { + this.wrap = wrap; + this.canonical = canonical; + this.key = calculateKey(wrap, canonical.z, canonical.z, canonical.x, canonical.y); + } +} + +/** + * An overscaled tile identifier + */ +export class OverscaledTileID { + overscaledZ: number; + wrap: number; + canonical: CanonicalTileID; + key: string; + posMatrix: mat4; + + constructor(overscaledZ: number, wrap: number, z: number, x: number, y: number) { + if (overscaledZ < z) throw new Error(`overscaledZ should be >= z; overscaledZ = ${overscaledZ}; z = ${z}`); + this.overscaledZ = overscaledZ; + this.wrap = wrap; + this.canonical = new CanonicalTileID(z, +x, +y); + this.key = calculateKey(wrap, overscaledZ, z, x, y); + } + + clone() { + return new OverscaledTileID(this.overscaledZ, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y); + } + + equals(id: OverscaledTileID) { + return this.overscaledZ === id.overscaledZ && this.wrap === id.wrap && this.canonical.equals(id.canonical); + } + + scaledTo(targetZ: number) { + if (targetZ > this.overscaledZ) throw new Error(`targetZ > this.overscaledZ; targetZ = ${targetZ}; overscaledZ = ${this.overscaledZ}`); + const zDifference = this.canonical.z - targetZ; + if (targetZ > this.canonical.z) { + return new OverscaledTileID(targetZ, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y); + } else { + return new OverscaledTileID(targetZ, this.wrap, targetZ, this.canonical.x >> zDifference, this.canonical.y >> zDifference); + } + } + + /* + * calculateScaledKey is an optimization: + * when withWrap == true, implements the same as this.scaledTo(z).key, + * when withWrap == false, implements the same as this.scaledTo(z).wrapped().key. + */ + calculateScaledKey(targetZ: number, withWrap: boolean): string { + if (targetZ > this.overscaledZ) throw new Error(`targetZ > this.overscaledZ; targetZ = ${targetZ}; overscaledZ = ${this.overscaledZ}`); + const zDifference = this.canonical.z - targetZ; + if (targetZ > this.canonical.z) { + return calculateKey(this.wrap * +withWrap, targetZ, this.canonical.z, this.canonical.x, this.canonical.y); + } else { + return calculateKey(this.wrap * +withWrap, targetZ, targetZ, this.canonical.x >> zDifference, this.canonical.y >> zDifference); + } + } + + isChildOf(parent: OverscaledTileID) { + if (parent.wrap !== this.wrap) { + // We can't be a child if we're in a different world copy + return false; + } + const zDifference = this.canonical.z - parent.canonical.z; + // We're first testing for z == 0, to avoid a 32 bit shift, which is undefined. + return parent.overscaledZ === 0 || ( + parent.overscaledZ < this.overscaledZ && + parent.canonical.x === (this.canonical.x >> zDifference) && + parent.canonical.y === (this.canonical.y >> zDifference)); + } + + children(sourceMaxZoom: number) { + if (this.overscaledZ >= sourceMaxZoom) { + // return a single tile coord representing a an overscaled tile + return [new OverscaledTileID(this.overscaledZ + 1, this.wrap, this.canonical.z, this.canonical.x, this.canonical.y)]; + } + + const z = this.canonical.z + 1; + const x = this.canonical.x * 2; + const y = this.canonical.y * 2; + return [ + new OverscaledTileID(z, this.wrap, z, x, y), + new OverscaledTileID(z, this.wrap, z, x + 1, y), + new OverscaledTileID(z, this.wrap, z, x, y + 1), + new OverscaledTileID(z, this.wrap, z, x + 1, y + 1) + ]; + } + + isLessThan(rhs: OverscaledTileID) { + if (this.wrap < rhs.wrap) return true; + if (this.wrap > rhs.wrap) return false; + + if (this.overscaledZ < rhs.overscaledZ) return true; + if (this.overscaledZ > rhs.overscaledZ) return false; + + if (this.canonical.x < rhs.canonical.x) return true; + if (this.canonical.x > rhs.canonical.x) return false; + + if (this.canonical.y < rhs.canonical.y) return true; + return false; + } + + wrapped() { + return new OverscaledTileID(this.overscaledZ, 0, this.canonical.z, this.canonical.x, this.canonical.y); + } + + unwrapTo(wrap: number) { + return new OverscaledTileID(this.overscaledZ, wrap, this.canonical.z, this.canonical.x, this.canonical.y); + } + + overscaleFactor() { + return Math.pow(2, this.overscaledZ - this.canonical.z); + } + + toUnwrapped() { + return new UnwrappedTileID(this.wrap, this.canonical); + } + + toString() { + return `${this.overscaledZ}/${this.canonical.x}/${this.canonical.y}`; + } + + getTilePoint(coord: MercatorCoordinate) { + return this.canonical.getTilePoint(new MercatorCoordinate(coord.x - this.wrap, coord.y)); + } +} + +function calculateKey(wrap: number, overscaledZ: number, z: number, x: number, y: number): string { + wrap *= 2; + if (wrap < 0) wrap = wrap * -1 - 1; + const dim = 1 << z; + return (dim * dim * wrap + dim * y + x).toString(36) + z.toString(36) + overscaledZ.toString(36); +} + +function getQuadkey(z, x, y) { + let quadkey = '', mask; + for (let i = z; i > 0; i--) { + mask = 1 << (i - 1); + quadkey += ((x & mask ? 1 : 0) + (y & mask ? 2 : 0)); + } + return quadkey; +} + +register('CanonicalTileID', CanonicalTileID); +register('OverscaledTileID', OverscaledTileID, {omit: ['posMatrix']}); diff --git a/web/libraries/maplibre-gl/src/source/vector_tile_source.test.ts b/web/libraries/maplibre-gl/src/source/vector_tile_source.test.ts new file mode 100644 index 00000000..d2ff756f --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/vector_tile_source.test.ts @@ -0,0 +1,355 @@ +import {fakeServer, type FakeServer} from 'nise'; +import {Source} from './source'; +import {VectorTileSource} from './vector_tile_source'; +import {Tile} from './tile'; +import {OverscaledTileID} from './tile_id'; +import {Evented} from '../util/evented'; +import {RequestManager} from '../util/request_manager'; +import fixturesSource from '../../test/unit/assets/source.json' assert {type: 'json'}; +import {getMockDispatcher, getWrapDispatcher} from '../util/test/util'; +import {Map} from '../ui/map'; + +function createSource(options, transformCallback?, clearTiles = () => {}) { + const source = new VectorTileSource('id', options, getMockDispatcher(), options.eventedParent); + source.onAdd({ + transform: {showCollisionBoxes: false}, + _getMapId: () => 1, + _requestManager: new RequestManager(transformCallback), + style: {sourceCaches: {id: {clearTiles}}}, + getPixelRatio() { return 1; } + } as any as Map); + + source.on('error', (e) => { + throw e.error; + }); + + return source; +} + +describe('VectorTileSource', () => { + let server: FakeServer; + beforeEach(() => { + global.fetch = null; + server = fakeServer.create(); + }); + + afterEach(() => { + server.restore(); + }); + + test('can be constructed from TileJSON', done => { + const source = createSource({ + minzoom: 1, + maxzoom: 10, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'] + }); + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(source.tiles).toEqual(['http://example.com/{z}/{x}/{y}.png']); + expect(source.minzoom).toBe(1); + expect(source.maxzoom).toBe(10); + expect((source as Source).attribution).toBe('MapLibre'); + done(); + } + }); + }); + + test('can be constructed from a TileJSON URL', done => { + server.respondWith('/source.json', JSON.stringify(fixturesSource)); + + const source = createSource({url: '/source.json'}); + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(source.tiles).toEqual(['http://example.com/{z}/{x}/{y}.png']); + expect(source.minzoom).toBe(1); + expect(source.maxzoom).toBe(10); + expect((source as Source).attribution).toBe('MapLibre'); + done(); + } + }); + + server.respond(); + }); + + test('transforms the request for TileJSON URL', () => { + server.respondWith('/source.json', JSON.stringify(fixturesSource)); + const transformSpy = jest.fn().mockImplementation((url) => { + return {url}; + }); + + createSource({url: '/source.json'}, transformSpy); + server.respond(); + expect(transformSpy).toHaveBeenCalledWith('/source.json', 'Source'); + }); + + test('fires event with metadata property', done => { + server.respondWith('/source.json', JSON.stringify(fixturesSource)); + const source = createSource({url: '/source.json'}); + source.on('data', (e) => { + if (e.sourceDataType === 'content') done(); + }); + server.respond(); + }); + + test('fires "dataloading" event', done => { + server.respondWith('/source.json', JSON.stringify(fixturesSource)); + const evented = new Evented(); + let dataloadingFired = false; + evented.on('dataloading', () => { + dataloadingFired = true; + }); + const source = createSource({url: '/source.json', eventedParent: evented}); + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + if (!dataloadingFired) done('test failed: dataloading not fired'); + done(); + } + }); + server.respond(); + }); + + test('serialize URL', () => { + const source = createSource({ + url: 'http://localhost:2900/source.json' + }); + expect(source.serialize()).toEqual({ + type: 'vector', + url: 'http://localhost:2900/source.json' + }); + }); + + test('serialize TileJSON', () => { + const source = createSource({ + minzoom: 1, + maxzoom: 10, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'] + }); + expect(source.serialize()).toEqual({ + type: 'vector', + minzoom: 1, + maxzoom: 10, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'] + }); + }); + + function testScheme(scheme, expectedURL) { + test(`scheme "${scheme}"`, done => { + const source = createSource({ + minzoom: 1, + maxzoom: 10, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'], + scheme + }); + + source.dispatcher = getWrapDispatcher()({ + send(type, params) { + expect(type).toBe('loadTile'); + expect(expectedURL).toBe(params.request.url); + done(); + } + }); + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') source.loadTile({ + tileID: new OverscaledTileID(10, 0, 10, 5, 5) + } as any as Tile, () => {}); + }); + }); + } + + testScheme('xyz', 'http://example.com/10/5/5.png'); + testScheme('tms', 'http://example.com/10/5/1018.png'); + + test('transforms tile urls before requesting', done => { + server.respondWith('/source.json', JSON.stringify(fixturesSource)); + + const source = createSource({url: '/source.json'}); + const transformSpy = jest.spyOn(source.map._requestManager, 'transformRequest'); + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + const tile = { + tileID: new OverscaledTileID(10, 0, 10, 5, 5), + state: 'loading', + loadVectorData () {}, + setExpiryData() {} + } as any as Tile; + source.loadTile(tile, () => {}); + expect(transformSpy).toHaveBeenCalledTimes(1); + expect(transformSpy).toHaveBeenCalledWith('http://example.com/10/5/5.png', 'Tile'); + done(); + } + }); + + server.respond(); + }); + + test('reloads a loading tile properly', done => { + const source = createSource({ + tiles: ['http://example.com/{z}/{x}/{y}.png'] + }); + const events = []; + source.dispatcher = getWrapDispatcher()({ + send(type, params, cb) { + events.push(type); + if (cb) setTimeout(cb, 0); + return 1; + } + }); + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + const tile = { + tileID: new OverscaledTileID(10, 0, 10, 5, 5), + state: 'loading', + loadVectorData () { + this.state = 'loaded'; + events.push('tileLoaded'); + }, + setExpiryData() {} + } as any as Tile; + source.loadTile(tile, () => {}); + expect(tile.state).toBe('loading'); + source.loadTile(tile, () => { + expect(events).toEqual( + ['loadTile', 'tileLoaded', 'reloadTile', 'tileLoaded'] + ); + done(); + }); + } + }); + }); + + test('respects TileJSON.bounds', done => { + const source = createSource({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'], + bounds: [-47, -7, -45, -5] + }); + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(source.hasTile(new OverscaledTileID(8, 0, 8, 96, 132))).toBeFalsy(); + expect(source.hasTile(new OverscaledTileID(8, 0, 8, 95, 132))).toBeTruthy(); + done(); + } + }); + }); + + test('does not error on invalid bounds', done => { + const source = createSource({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'], + bounds: [-47, -7, -45, 91] + }); + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(source.tileBounds.bounds).toEqual({_sw: {lng: -47, lat: -7}, _ne: {lng: -45, lat: 90}}); + done(); + } + }); + }); + + test('respects TileJSON.bounds when loaded from TileJSON', done => { + server.respondWith('/source.json', JSON.stringify({ + minzoom: 0, + maxzoom: 22, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'], + bounds: [-47, -7, -45, -5] + })); + const source = createSource({url: '/source.json'}); + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + expect(source.hasTile(new OverscaledTileID(8, 0, 8, 96, 132))).toBeFalsy(); + expect(source.hasTile(new OverscaledTileID(8, 0, 8, 95, 132))).toBeTruthy(); + done(); + } + }); + server.respond(); + }); + + test('respects collectResourceTiming parameter on source', done => { + const source = createSource({ + tiles: ['http://example.com/{z}/{x}/{y}.png'], + collectResourceTiming: true + }); + source.dispatcher = getWrapDispatcher()({ + send(type, params, cb) { + expect(params.request.collectResourceTiming).toBeTruthy(); + setTimeout(cb, 0); + done(); + + // do nothing for cache size check dispatch + source.dispatcher = getMockDispatcher(); + + return 1; + } + }); + + source.on('data', (e) => { + if (e.sourceDataType === 'metadata') { + const tile = { + tileID: new OverscaledTileID(10, 0, 10, 5, 5), + state: 'loading', + loadVectorData () {}, + setExpiryData() {} + } as any as Tile; + source.loadTile(tile, () => {}); + } + }); + }); + + test('cancels TileJSON request if removed', () => { + const source = createSource({url: '/source.json'}); + source.onRemove(); + expect((server as any).lastRequest.aborted).toBe(true); + }); + + test('supports url property updates', () => { + const source = createSource({ + url: 'http://localhost:2900/source.json' + }); + source.setUrl('http://localhost:2900/source2.json'); + expect(source.serialize()).toEqual({ + type: 'vector', + url: 'http://localhost:2900/source2.json' + }); + }); + + test('supports tiles property updates', () => { + const source = createSource({ + minzoom: 1, + maxzoom: 10, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'] + }); + source.setTiles(['http://example2.com/{z}/{x}/{y}.png']); + expect(source.serialize()).toEqual({ + type: 'vector', + minzoom: 1, + maxzoom: 10, + attribution: 'MapLibre', + tiles: ['http://example2.com/{z}/{x}/{y}.png'] + }); + }); + + test('setTiles only clears the cache once the TileJSON has reloaded', async () => { + const clearTiles = jest.fn(); + const source = createSource({tiles: ['http://example.com/{z}/{x}/{y}.pbf']}, undefined, clearTiles); + source.setTiles(['http://example2.com/{z}/{x}/{y}.pbf']); + expect(clearTiles.mock.calls).toHaveLength(0); + await source.once('data'); + expect(clearTiles.mock.calls).toHaveLength(1); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/vector_tile_source.ts b/web/libraries/maplibre-gl/src/source/vector_tile_source.ts new file mode 100644 index 00000000..8cdb19c3 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/vector_tile_source.ts @@ -0,0 +1,260 @@ +import {Event, ErrorEvent, Evented} from '../util/evented'; + +import {extend, pick} from '../util/util'; +import {loadTileJson} from './load_tilejson'; +import {TileBounds} from './tile_bounds'; +import {ResourceType} from '../util/request_manager'; + +import type {Source} from './source'; +import type {OverscaledTileID} from './tile_id'; +import type {Map} from '../ui/map'; +import type {Dispatcher} from '../util/dispatcher'; +import type {Tile} from './tile'; +import type {Callback} from '../types/callback'; +import type {Cancelable} from '../types/cancelable'; +import type {VectorSourceSpecification, PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec'; + +export type VectorTileSourceOptions = VectorSourceSpecification & { + collectResourceTiming?: boolean; +} + +/** + * A source containing vector tiles in [Mapbox Vector Tile format](https://docs.mapbox.com/vector-tiles/reference/). + * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/) for detailed documentation of options.) + * + * @group Sources + * + * @example + * ```ts + * map.addSource('some id', { + * type: 'vector', + * url: 'https://demotiles.maplibre.org/tiles/tiles.json' + * }); + * ``` + * + * @example + * ```ts + * map.addSource('some id', { + * type: 'vector', + * tiles: ['https://d25uarhxywzl1j.cloudfront.net/v0.1/{z}/{x}/{y}.mvt'], + * minzoom: 6, + * maxzoom: 14 + * }); + * ``` + * + * @example + * ```ts + * map.getSource('some id').setUrl("https://demotiles.maplibre.org/tiles/tiles.json"); + * ``` + * + * @example + * ```ts + * map.getSource('some id').setTiles(['https://d25uarhxywzl1j.cloudfront.net/v0.1/{z}/{x}/{y}.mvt']); + * ``` + * @see [Add a vector tile source](https://maplibre.org/maplibre-gl-js/docs/examples/vector-source/) + */ +export class VectorTileSource extends Evented implements Source { + type: 'vector'; + id: string; + minzoom: number; + maxzoom: number; + url: string; + scheme: string; + tileSize: number; + promoteId: PromoteIdSpecification; + + _options: VectorSourceSpecification; + _collectResourceTiming: boolean; + dispatcher: Dispatcher; + map: Map; + bounds: [number, number, number, number]; + tiles: Array; + tileBounds: TileBounds; + reparseOverscaled: boolean; + isTileClipped: boolean; + _tileJSONRequest: Cancelable; + _loaded: boolean; + + constructor(id: string, options: VectorTileSourceOptions, dispatcher: Dispatcher, eventedParent: Evented) { + super(); + this.id = id; + this.dispatcher = dispatcher; + + this.type = 'vector'; + this.minzoom = 0; + this.maxzoom = 22; + this.scheme = 'xyz'; + this.tileSize = 512; + this.reparseOverscaled = true; + this.isTileClipped = true; + this._loaded = false; + + extend(this, pick(options, ['url', 'scheme', 'tileSize', 'promoteId'])); + this._options = extend({type: 'vector'}, options); + + this._collectResourceTiming = options.collectResourceTiming; + + if (this.tileSize !== 512) { + throw new Error('vector tile sources must have a tileSize of 512'); + } + + this.setEventedParent(eventedParent); + } + + load = () => { + this._loaded = false; + this.fire(new Event('dataloading', {dataType: 'source'})); + this._tileJSONRequest = loadTileJson(this._options, this.map._requestManager, (err, tileJSON) => { + this._tileJSONRequest = null; + this._loaded = true; + this.map.style.sourceCaches[this.id].clearTiles(); + if (err) { + this.fire(new ErrorEvent(err)); + } else if (tileJSON) { + extend(this, tileJSON); + if (tileJSON.bounds) this.tileBounds = new TileBounds(tileJSON.bounds, this.minzoom, this.maxzoom); + + // `content` is included here to prevent a race condition where `Style#_updateSources` is called + // before the TileJSON arrives. this makes sure the tiles needed are loaded once TileJSON arrives + // ref: https://github.com/mapbox/mapbox-gl-js/pull/4347#discussion_r104418088 + this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'})); + this.fire(new Event('data', {dataType: 'source', sourceDataType: 'content'})); + } + }); + }; + + loaded(): boolean { + return this._loaded; + } + + hasTile(tileID: OverscaledTileID) { + return !this.tileBounds || this.tileBounds.contains(tileID.canonical); + } + + onAdd(map: Map) { + this.map = map; + this.load(); + } + + setSourceProperty(callback: Function) { + if (this._tileJSONRequest) { + this._tileJSONRequest.cancel(); + } + + callback(); + + this.load(); + } + + /** + * Sets the source `tiles` property and re-renders the map. + * + * @param tiles - An array of one or more tile source URLs, as in the TileJSON spec. + * @returns `this` + */ + setTiles(tiles: Array): this { + this.setSourceProperty(() => { + this._options.tiles = tiles; + }); + + return this; + } + + /** + * Sets the source `url` property and re-renders the map. + * + * @param url - A URL to a TileJSON resource. Supported protocols are `http:` and `https:`. + * @returns `this` + */ + setUrl(url: string): this { + this.setSourceProperty(() => { + this.url = url; + this._options.url = url; + }); + + return this; + } + + onRemove() { + if (this._tileJSONRequest) { + this._tileJSONRequest.cancel(); + this._tileJSONRequest = null; + } + } + + serialize = (): VectorSourceSpecification => { + return extend({}, this._options); + }; + + loadTile(tile: Tile, callback: Callback) { + const url = tile.tileID.canonical.url(this.tiles, this.map.getPixelRatio(), this.scheme); + const params = { + request: this.map._requestManager.transformRequest(url, ResourceType.Tile), + uid: tile.uid, + tileID: tile.tileID, + zoom: tile.tileID.overscaledZ, + tileSize: this.tileSize * tile.tileID.overscaleFactor(), + type: this.type, + source: this.id, + pixelRatio: this.map.getPixelRatio(), + showCollisionBoxes: this.map.showCollisionBoxes, + promoteId: this.promoteId + }; + params.request.collectResourceTiming = this._collectResourceTiming; + + if (!tile.actor || tile.state === 'expired') { + tile.actor = this.dispatcher.getActor(); + tile.request = tile.actor.send('loadTile', params, done.bind(this)); + } else if (tile.state === 'loading') { + // schedule tile reloading after it has been loaded + tile.reloadCallback = callback; + } else { + tile.request = tile.actor.send('reloadTile', params, done.bind(this)); + } + + function done(err, data) { + delete tile.request; + + if (tile.aborted) + return callback(null); + + if (err && err.status !== 404) { + return callback(err); + } + + if (data && data.resourceTiming) + tile.resourceTiming = data.resourceTiming; + + if (this.map._refreshExpiredTiles && data) tile.setExpiryData(data); + tile.loadVectorData(data, this.map.painter); + + callback(null); + + if (tile.reloadCallback) { + this.loadTile(tile, tile.reloadCallback); + tile.reloadCallback = null; + } + } + } + + abortTile(tile: Tile) { + if (tile.request) { + tile.request.cancel(); + delete tile.request; + } + if (tile.actor) { + tile.actor.send('abortTile', {uid: tile.uid, type: this.type, source: this.id}, undefined); + } + } + + unloadTile(tile: Tile) { + tile.unloadVectorData(); + if (tile.actor) { + tile.actor.send('removeTile', {uid: tile.uid, type: this.type, source: this.id}, undefined); + } + } + + hasTransition() { + return false; + } +} diff --git a/web/libraries/maplibre-gl/src/source/vector_tile_worker_source.test.ts b/web/libraries/maplibre-gl/src/source/vector_tile_worker_source.test.ts new file mode 100644 index 00000000..73c82201 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/vector_tile_worker_source.test.ts @@ -0,0 +1,390 @@ +import fs from 'fs'; +import path from 'path'; +import vt from '@mapbox/vector-tile'; +import Protobuf from 'pbf'; +import {VectorTileWorkerSource} from '../source/vector_tile_worker_source'; +import {StyleLayerIndex} from '../style/style_layer_index'; +import {fakeServer, type FakeServer} from 'nise'; +import {Actor} from '../util/actor'; +import {TileParameters, WorkerTileParameters, WorkerTileResult} from './worker_source'; +import {WorkerTile} from './worker_tile'; +import {setPerformance} from '../util/test/util'; + +describe('vector tile worker source', () => { + const actor = {send: () => {}} as any as Actor; + let server: FakeServer; + + beforeEach(() => { + global.fetch = null; + server = fakeServer.create(); + setPerformance(); + }); + + afterEach(() => { + server.restore(); + jest.clearAllMocks(); + }); + test('VectorTileWorkerSource#abortTile aborts pending request', () => { + const source = new VectorTileWorkerSource(actor, new StyleLayerIndex(), []); + + source.loadTile({ + source: 'source', + uid: 0, + tileID: {overscaledZ: 0, wrap: 0, canonical: {x: 0, y: 0, z: 0, w: 0}}, + request: {url: 'http://localhost:2900/abort'} + } as any as WorkerTileParameters, (err, res) => { + expect(err).toBeFalsy(); + expect(res).toBeFalsy(); + }); + + source.abortTile({ + source: 'source', + uid: 0 + } as any as TileParameters, (err, res) => { + expect(err).toBeFalsy(); + expect(res).toBeFalsy(); + }); + + expect(source.loading).toEqual({}); + }); + + test('VectorTileWorkerSource#removeTile removes loaded tile', () => { + const source = new VectorTileWorkerSource(actor, new StyleLayerIndex(), []); + + source.loaded = { + '0': {} as WorkerTile + }; + + source.removeTile({ + source: 'source', + uid: 0 + } as any as TileParameters, (err, res) => { + expect(err).toBeFalsy(); + expect(res).toBeFalsy(); + }); + + expect(source.loaded).toEqual({}); + }); + + test('VectorTileWorkerSource#reloadTile reloads a previously-loaded tile', () => { + const source = new VectorTileWorkerSource(actor, new StyleLayerIndex(), []); + const parse = jest.fn(); + + source.loaded = { + '0': { + status: 'done', + vectorTile: {}, + parse + } as any as WorkerTile + }; + + const callback = jest.fn(); + source.reloadTile({uid: 0} as any as WorkerTileParameters, callback); + expect(parse).toHaveBeenCalledTimes(1); + + parse.mock.calls[0][4](); + expect(callback).toHaveBeenCalledTimes(1); + }); + + test('VectorTileWorkerSource#loadTile reparses tile if the reloadTile has been called during parsing', (done) => { + const rawTileData = new Uint8Array([]); + function loadVectorData(params, callback) { + return callback(null, { + vectorTile: { + layers: { + test: { + version: 2, + name: 'test', + extent: 8192, + length: 1, + feature: (featureIndex: number) => ({ + extent: 8192, + type: 1, + id: featureIndex, + properties: { + name: 'test' + }, + loadGeometry () { + return [[{x: 0, y: 0}]]; + } + }) + } + } + } as any as vt.VectorTile, + rawData: rawTileData + }); + } + + const layerIndex = new StyleLayerIndex([{ + id: 'test', + source: 'source', + 'source-layer': 'test', + type: 'symbol', + layout: { + 'icon-image': 'hello', + 'text-font': ['StandardFont-Bold'], + 'text-field': '{name}' + } + }]); + + const send = jest.fn().mockImplementation((type: string, data: unknown, callback: Function) => { + const res = setTimeout(() => callback(null, + type === 'getImages' ? + {'hello': {width: 1, height: 1, data: new Uint8Array([0])}} : + {'StandardFont-Bold': {width: 1, height: 1, data: new Uint8Array([0])}} + )); + + return { + cancel: () => clearTimeout(res) + }; + }); + + const actor = { + send + } as unknown as Actor; + const source = new VectorTileWorkerSource(actor, layerIndex, ['hello'], loadVectorData); + source.loadTile({ + source: 'source', + uid: 0, + tileID: {overscaledZ: 0, wrap: 0, canonical: {x: 0, y: 0, z: 0, w: 0}}, + request: {url: 'http://localhost:2900/faketile.pbf'} + } as any as WorkerTileParameters, () => { + done.fail('should not be called'); + }); + + source.reloadTile({ + source: 'source', + uid: '0', + tileID: {overscaledZ: 0, wrap: 0, canonical: {x: 0, y: 0, z: 0, w: 0}}, + } as any as WorkerTileParameters, (err, res) => { + expect(err).toBeFalsy(); + expect(res).toBeDefined(); + expect(res.rawTileData).toBeDefined(); + expect(res.rawTileData).toStrictEqual(rawTileData); + done(); + }); + }); + + test('VectorTileWorkerSource#loadTile reparses tile if reloadTile is called during reparsing', (done) => { + const rawTileData = new Uint8Array([]); + function loadVectorData(params, callback) { + return callback(null, { + vectorTile: new vt.VectorTile(new Protobuf(rawTileData)), + rawData: rawTileData + }); + } + + const layerIndex = new StyleLayerIndex([{ + id: 'test', + source: 'source', + 'source-layer': 'test', + type: 'fill' + }]); + + const source = new VectorTileWorkerSource(actor, layerIndex, [], loadVectorData); + + const parseWorkerTileMock = jest + .spyOn(WorkerTile.prototype, 'parse') + .mockImplementation(function(data, layerIndex, availableImages, actor, callback) { + this.status = 'parsing'; + window.setTimeout(() => callback(null, {} as WorkerTileResult), 10); + }); + + let loadCallbackCalled = false; + source.loadTile({ + source: 'source', + uid: 0, + tileID: {overscaledZ: 0, wrap: 0, canonical: {x: 0, y: 0, z: 0, w: 0}}, + request: {url: 'http://localhost:2900/faketile.pbf'} + } as any as WorkerTileParameters, (err, res) => { + expect(err).toBeFalsy(); + expect(res).toBeDefined(); + loadCallbackCalled = true; + }); + + source.reloadTile({ + source: 'source', + uid: '0', + tileID: {overscaledZ: 0, wrap: 0, canonical: {x: 0, y: 0, z: 0, w: 0}}, + } as any as WorkerTileParameters, (err, res) => { + expect(err).toBeFalsy(); + expect(res).toBeDefined(); + expect(parseWorkerTileMock).toHaveBeenCalledTimes(2); + expect(loadCallbackCalled).toBeTruthy(); + done(); + }); + }); + + test('VectorTileWorkerSource#reloadTile does not reparse tiles with no vectorTile data but does call callback', () => { + const source = new VectorTileWorkerSource(actor, new StyleLayerIndex(), []); + const parse = jest.fn(); + + source.loaded = { + '0': { + status: 'done', + parse + } as any as WorkerTile + }; + + const callback = jest.fn(); + + source.reloadTile({uid: 0} as any as WorkerTileParameters, callback); + expect(parse).not.toHaveBeenCalled(); + expect(callback).toHaveBeenCalledTimes(1); + + }); + + test('VectorTileWorkerSource#returns a good error message when failing to parse a tile', () => { + const source = new VectorTileWorkerSource(actor, new StyleLayerIndex(), []); + const parse = jest.fn(); + const callback = jest.fn(); + + server.respondWith(request => { + request.respond(200, {'Content-Type': 'application/pbf'}, 'something...'); + }); + + source.loadTile({ + source: 'source', + uid: 0, + tileID: {overscaledZ: 0, wrap: 0, canonical: {x: 0, y: 0, z: 0, w: 0}}, + request: {url: 'http://localhost:2900/faketile.pbf'} + } as any as WorkerTileParameters, callback); + + server.respond(); + + expect(parse).not.toHaveBeenCalled(); + expect(callback).toHaveBeenCalledTimes(1); + expect(callback.mock.calls[0][0].message).toContain('Unable to parse the tile at'); + }); + + test('VectorTileWorkerSource#returns a good error message when failing to parse a gzipped tile', () => { + const source = new VectorTileWorkerSource(actor, new StyleLayerIndex(), []); + const parse = jest.fn(); + const callback = jest.fn(); + + server.respondWith(new Uint8Array([0x1f, 0x8b]).buffer); + + source.loadTile({ + source: 'source', + uid: 0, + tileID: {overscaledZ: 0, wrap: 0, canonical: {x: 0, y: 0, z: 0, w: 0}}, + request: {url: 'http://localhost:2900/faketile.pbf'} + } as any as WorkerTileParameters, callback); + + server.respond(); + + expect(parse).not.toHaveBeenCalled(); + expect(callback).toHaveBeenCalledTimes(1); + expect(callback.mock.calls[0][0].message).toContain('gzipped'); + }); + + test('VectorTileWorkerSource provides resource timing information', done => { + const rawTileData = fs.readFileSync(path.join(__dirname, '/../../test/unit/assets/mbsv5-6-18-23.vector.pbf')); + + function loadVectorData(params, callback) { + return callback(null, { + vectorTile: new vt.VectorTile(new Protobuf(rawTileData)), + rawData: rawTileData, + cacheControl: null, + expires: null + }); + } + + const exampleResourceTiming = { + connectEnd: 473, + connectStart: 473, + decodedBodySize: 86494, + domainLookupEnd: 473, + domainLookupStart: 473, + duration: 341, + encodedBodySize: 52528, + entryType: 'resource', + fetchStart: 473.5, + initiatorType: 'xmlhttprequest', + name: 'http://localhost:2900/faketile.pbf', + nextHopProtocol: 'http/1.1', + redirectEnd: 0, + redirectStart: 0, + requestStart: 477, + responseEnd: 815, + responseStart: 672, + secureConnectionStart: 0 + }; + + const layerIndex = new StyleLayerIndex([{ + id: 'test', + source: 'source', + 'source-layer': 'test', + type: 'fill' + }]); + + const source = new VectorTileWorkerSource(actor, layerIndex, [], loadVectorData); + + window.performance.getEntriesByName = jest.fn().mockReturnValue([exampleResourceTiming]); + + source.loadTile({ + source: 'source', + uid: 0, + tileID: {overscaledZ: 0, wrap: 0, canonical: {x: 0, y: 0, z: 0, w: 0}}, + request: {url: 'http://localhost:2900/faketile.pbf', collectResourceTiming: true} + } as any as WorkerTileParameters, (err, res) => { + expect(err).toBeFalsy(); + expect(res.resourceTiming[0]).toEqual(exampleResourceTiming); + done(); + }); + }); + + test('VectorTileWorkerSource provides resource timing information (fallback method)', done => { + const rawTileData = fs.readFileSync(path.join(__dirname, '/../../test/unit/assets/mbsv5-6-18-23.vector.pbf')); + + function loadVectorData(params, callback) { + return callback(null, { + vectorTile: new vt.VectorTile(new Protobuf(rawTileData)), + rawData: rawTileData, + cacheControl: null, + expires: null + }); + } + + const layerIndex = new StyleLayerIndex([{ + id: 'test', + source: 'source', + 'source-layer': 'test', + type: 'fill' + }]); + + const source = new VectorTileWorkerSource(actor, layerIndex, [], loadVectorData); + + const sampleMarks = [100, 350]; + const marks = {}; + const measures = {}; + window.performance.getEntriesByName = jest.fn().mockImplementation(name => (measures[name] || [])); + window.performance.mark = jest.fn().mockImplementation(name => { + marks[name] = sampleMarks.shift(); + return null; + }); + window.performance.measure = jest.fn().mockImplementation((name, start, end) => { + measures[name] = measures[name] || []; + measures[name].push({ + duration: marks[end] - marks[start], + entryType: 'measure', + name, + startTime: marks[start] + }); + return null; + }); + + source.loadTile({ + source: 'source', + uid: 0, + tileID: {overscaledZ: 0, wrap: 0, canonical: {x: 0, y: 0, z: 0, w: 0}}, + request: {url: 'http://localhost:2900/faketile.pbf', collectResourceTiming: true} + } as any as WorkerTileParameters, (err, res) => { + expect(err).toBeFalsy(); + expect(res.resourceTiming[0]).toEqual( + {'duration': 250, 'entryType': 'measure', 'name': 'http://localhost:2900/faketile.pbf', 'startTime': 100} + ); + done(); + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/vector_tile_worker_source.ts b/web/libraries/maplibre-gl/src/source/vector_tile_worker_source.ts new file mode 100644 index 00000000..7c7fca9d --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/vector_tile_worker_source.ts @@ -0,0 +1,228 @@ +import {ExpiryData, getArrayBuffer} from '../util/ajax'; + +import vt from '@mapbox/vector-tile'; +import Protobuf from 'pbf'; +import {WorkerTile} from './worker_tile'; +import {extend} from '../util/util'; +import {RequestPerformance} from '../util/performance'; + +import type { + WorkerSource, + WorkerTileParameters, + WorkerTileCallback, + TileParameters +} from '../source/worker_source'; + +import type {Actor} from '../util/actor'; +import type {StyleLayerIndex} from '../style/style_layer_index'; +import type {Callback} from '../types/callback'; +import type {VectorTile} from '@mapbox/vector-tile'; + +export type LoadVectorTileResult = { + vectorTile: VectorTile; + rawData: ArrayBuffer; + resourceTiming?: Array; +} & ExpiryData; + +type FetchingState = { + rawTileData: ArrayBuffer; + cacheControl: ExpiryData; + resourceTiming: any; +} + +/** + * The callback when finished loading vector data + */ +export type LoadVectorDataCallback = Callback; + +export type AbortVectorData = () => void; +export type LoadVectorData = (params: WorkerTileParameters, callback: LoadVectorDataCallback) => AbortVectorData | void; + +/** + * Loads a vector tile + */ +function loadVectorTile(params: WorkerTileParameters, callback: LoadVectorDataCallback) { + const request = getArrayBuffer(params.request, (err?: Error | null, data?: ArrayBuffer | null, cacheControl?: string | null, expires?: string | null) => { + if (err) { + callback(err); + } else if (data) { + try { + const vectorTile = new vt.VectorTile(new Protobuf(data)); + callback(null, { + vectorTile, + rawData: data, + cacheControl, + expires + }); + } catch (ex) { + const bytes = new Uint8Array(data); + const isGzipped = bytes[0] === 0x1f && bytes[1] === 0x8b; + let errorMessage = `Unable to parse the tile at ${params.request.url}, `; + if (isGzipped) { + errorMessage += 'please make sure the data is not gzipped and that you have configured the relevant header in the server'; + } else { + errorMessage += `got error: ${ex.messge}`; + } + callback(new Error(errorMessage)); + } + } + }); + return () => { + request.cancel(); + callback(); + }; +} + +/** + * The {@link WorkerSource} implementation that supports {@link VectorTileSource}. + * This class is designed to be easily reused to support custom source types + * for data formats that can be parsed/converted into an in-memory VectorTile + * representation. To do so, create it with + * `new VectorTileWorkerSource(actor, styleLayers, customLoadVectorDataFunction)`. + */ +export class VectorTileWorkerSource implements WorkerSource { + actor: Actor; + layerIndex: StyleLayerIndex; + availableImages: Array; + loadVectorData: LoadVectorData; + fetching: {[_: string]: FetchingState }; + loading: {[_: string]: WorkerTile}; + loaded: {[_: string]: WorkerTile}; + + /** + * @param loadVectorData - Optional method for custom loading of a VectorTile + * object based on parameters passed from the main-thread Source. See + * {@link VectorTileWorkerSource#loadTile}. The default implementation simply + * loads the pbf at `params.url`. + */ + constructor(actor: Actor, layerIndex: StyleLayerIndex, availableImages: Array, loadVectorData?: LoadVectorData | null) { + this.actor = actor; + this.layerIndex = layerIndex; + this.availableImages = availableImages; + this.loadVectorData = loadVectorData || loadVectorTile; + this.fetching = {}; + this.loading = {}; + this.loaded = {}; + } + + /** + * Implements {@link WorkerSource#loadTile}. Delegates to + * {@link VectorTileWorkerSource#loadVectorData} (which by default expects + * a `params.url` property) for fetching and producing a VectorTile object. + */ + loadTile(params: WorkerTileParameters, callback: WorkerTileCallback) { + const uid = params.uid; + + if (!this.loading) + this.loading = {}; + + const perf = (params && params.request && params.request.collectResourceTiming) ? + new RequestPerformance(params.request) : false; + + const workerTile = this.loading[uid] = new WorkerTile(params); + workerTile.abort = this.loadVectorData(params, (err, response) => { + delete this.loading[uid]; + + if (err || !response) { + workerTile.status = 'done'; + this.loaded[uid] = workerTile; + return callback(err); + } + + const rawTileData = response.rawData; + const cacheControl = {} as ExpiryData; + if (response.expires) cacheControl.expires = response.expires; + if (response.cacheControl) cacheControl.cacheControl = response.cacheControl; + + const resourceTiming = {} as {resourceTiming: any}; + if (perf) { + const resourceTimingData = perf.finish(); + // it's necessary to eval the result of getEntriesByName() here via parse/stringify + // late evaluation in the main thread causes TypeError: illegal invocation + if (resourceTimingData) + resourceTiming.resourceTiming = JSON.parse(JSON.stringify(resourceTimingData)); + } + + workerTile.vectorTile = response.vectorTile; + workerTile.parse(response.vectorTile, this.layerIndex, this.availableImages, this.actor, (err, result) => { + delete this.fetching[uid]; + if (err || !result) return callback(err); + + // Transferring a copy of rawTileData because the worker needs to retain its copy. + callback(null, extend({rawTileData: rawTileData.slice(0)}, result, cacheControl, resourceTiming)); + }); + + this.loaded = this.loaded || {}; + this.loaded[uid] = workerTile; + // keep the original fetching state so that reload tile can pick it up if the original parse is cancelled by reloads' parse + this.fetching[uid] = {rawTileData, cacheControl, resourceTiming}; + }) as AbortVectorData; + } + + /** + * Implements {@link WorkerSource#reloadTile}. + */ + reloadTile(params: WorkerTileParameters, callback: WorkerTileCallback) { + const loaded = this.loaded; + const uid = params.uid; + if (loaded && loaded[uid]) { + const workerTile = loaded[uid]; + workerTile.showCollisionBoxes = params.showCollisionBoxes; + if (workerTile.status === 'parsing') { + workerTile.parse(workerTile.vectorTile, this.layerIndex, this.availableImages, this.actor, (err, result) => { + if (err || !result) return callback(err, result); + + // if we have cancelled the original parse, make sure to pass the rawTileData from the original fetch + let parseResult; + if (this.fetching[uid]) { + const {rawTileData, cacheControl, resourceTiming} = this.fetching[uid]; + delete this.fetching[uid]; + parseResult = extend({rawTileData: rawTileData.slice(0)}, result, cacheControl, resourceTiming); + } else { + parseResult = result; + } + + callback(null, parseResult); + }); + } else if (workerTile.status === 'done') { + // if there was no vector tile data on the initial load, don't try and re-parse tile + if (workerTile.vectorTile) { + workerTile.parse(workerTile.vectorTile, this.layerIndex, this.availableImages, this.actor, callback); + } else { + callback(); + } + } + } + } + + /** + * Implements {@link WorkerSource#abortTile}. + * + * @param params - The tile parameters + * @param callback - The callback + */ + abortTile(params: TileParameters, callback: WorkerTileCallback) { + const loading = this.loading, + uid = params.uid; + if (loading && loading[uid] && loading[uid].abort) { + loading[uid].abort(); + delete loading[uid]; + } + callback(); + } + + /** + * Implements {@link WorkerSource#removeTile}. + * + * @param params - The tile parameters + * @param callback - The callback + */ + removeTile(params: TileParameters, callback: WorkerTileCallback) { + const loaded = this.loaded, + uid = params.uid; + if (loaded && loaded[uid]) { + delete loaded[uid]; + } + callback(); + } +} diff --git a/web/libraries/maplibre-gl/src/source/video_source.test.ts b/web/libraries/maplibre-gl/src/source/video_source.test.ts new file mode 100644 index 00000000..d7ef11c1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/video_source.test.ts @@ -0,0 +1,124 @@ +import {VideoSource} from './video_source'; +import {extend} from '../util/util'; +import {getMockDispatcher} from '../util/test/util'; + +import type {Coordinates} from './image_source'; +import {Tile} from './tile'; +import {OverscaledTileID} from './tile_id'; +import {Evented} from '../util/evented'; +import {Transform} from '../geo/transform'; +import {VertexBuffer} from '../gl/vertex_buffer'; +import {SegmentVector} from '../data/segment'; + +class StubMap extends Evented { + transform: Transform; + style: any; + painter: any; + + constructor() { + super(); + this.transform = new Transform(); + this.style = {}; + this.painter = { + context: { + gl: { + texSubImage2D: () => {} + } + } + }; + } +} + +function createSource(options) { + const c = options && options.video || window.document.createElement('video'); + + options = extend({coordinates: [[0, 0], [1, 0], [1, 1], [0, 1]]}, options); + + const source = new VideoSource('id', options, getMockDispatcher(), options.eventedParent); + + source.video = c; + return source; +} + +describe('VideoSource', () => { + // Attribution File:Volcano Lava Sample.webm: U.S. Geological Survey (USGS), Public domain, via Wikimedia Commons + const source = createSource({ + type: 'video', + urls: ['cropped.mp4', 'https://upload.wikimedia.org/wikipedia/commons/2/22/Volcano_Lava_Sample.webm'], + coordinates: [ + [-76.54, 39.18], + [-76.52, 39.18], + [-76.52, 39.17], + [-76.54, 39.17] + ] + }); + + test('constructor', () => { + expect(source.minzoom).toBe(0); + expect(source.maxzoom).toBe(22); + expect(source.tileSize).toBe(512); + }); + + test('sets coordinates', () => { + const newCoordinates = [[0, 0], [-1, 0], [-1, -1], [0, -1]] as Coordinates; + source.setCoordinates(newCoordinates); + const serialized = source.serialize(); + + expect(serialized.coordinates).toEqual(newCoordinates); + + }); + + //test video retrieval by first supplying the video element directly + test('gets video', () => { + const el = window.document.createElement('video'); + // Attribution File:Volcano Lava Sample.webm: U.S. Geological Survey (USGS), Public domain, via Wikimedia Commons + const source = createSource({ + type: 'video', + video: el, + urls: ['cropped.mp4', 'https://upload.wikimedia.org/wikipedia/commons/2/22/Volcano_Lava_Sample.webm'], + coordinates: [ + [-76.54, 39.18], + [-76.52, 39.18], + [-76.52, 39.17], + [-76.54, 39.17] + ] + }); + + expect(source.getVideo()).toBe(el); + }); + + test('fires idle event on prepare call when there is at least one not loaded tile', done => { + const source = createSource({ + type: 'video', + urls: [], + video: { + readyState: 2, + play: () => {} + }, + coordinates: [ + [-76.54, 39.18], + [-76.52, 39.18], + [-76.52, 39.17], + [-76.54, 39.17] + ] + }); + const tile = new Tile(new OverscaledTileID(1, 0, 1, 0, 0), 512); + source.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'idle') { + expect(tile.state).toBe('loaded'); + done(); + } + }); + source.onAdd(new StubMap() as any); + + source.tiles[String(tile.tileID.wrap)] = tile; + // assign dummies directly so we don't need to stub the gl things + source.boundsBuffer = {} as VertexBuffer; + source.boundsSegments = {} as SegmentVector; + source.texture = { + update: () => {}, + bind: () => {} + } as any; + source.prepare(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/video_source.ts b/web/libraries/maplibre-gl/src/source/video_source.ts new file mode 100644 index 00000000..5aa966df --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/video_source.ts @@ -0,0 +1,205 @@ +import {getVideo} from '../util/ajax'; +import {ResourceType} from '../util/request_manager'; + +import {ImageSource} from './image_source'; +import rasterBoundsAttributes from '../data/raster_bounds_attributes'; +import {SegmentVector} from '../data/segment'; +import {Texture} from '../render/texture'; +import {Event, ErrorEvent} from '../util/evented'; +import {ValidationError} from '@maplibre/maplibre-gl-style-spec'; + +import type {Map} from '../ui/map'; +import type {Dispatcher} from '../util/dispatcher'; +import type {Evented} from '../util/evented'; +import type {VideoSourceSpecification} from '@maplibre/maplibre-gl-style-spec'; + +/** + * A data source containing video. + * (See the [Style Specification](https://maplibre.org/maplibre-style-spec/#sources-video) for detailed documentation of options.) + * + * @group Sources + * + * @example + * ```ts + * // add to map + * map.addSource('some id', { + * type: 'video', + * url: [ + * 'https://www.mapbox.com/blog/assets/baltimore-smoke.mp4', + * 'https://www.mapbox.com/blog/assets/baltimore-smoke.webm' + * ], + * coordinates: [ + * [-76.54, 39.18], + * [-76.52, 39.18], + * [-76.52, 39.17], + * [-76.54, 39.17] + * ] + * }); + * + * // update + * let mySource = map.getSource('some id'); + * mySource.setCoordinates([ + * [-76.54335737228394, 39.18579907229748], + * [-76.52803659439087, 39.1838364847587], + * [-76.5295386314392, 39.17683392507606], + * [-76.54520273208618, 39.17876344106642] + * ]); + * + * map.removeSource('some id'); // remove + * ``` + * @see [Add a video](https://maplibre.org/maplibre-gl-js/docs/examples/video-on-a-map/) + * + * Note that when rendered as a raster layer, the layer's `raster-fade-duration` property will cause the video to fade in. + * This happens when playback is started, paused and resumed, or when the video's coordinates are updated. To avoid this behavior, + * set the layer's `raster-fade-duration` property to `0`. + */ +export class VideoSource extends ImageSource { + options: VideoSourceSpecification; + urls: Array; + video: HTMLVideoElement; + roundZoom: boolean; + + constructor(id: string, options: VideoSourceSpecification, dispatcher: Dispatcher, eventedParent: Evented) { + super(id, options, dispatcher, eventedParent); + this.roundZoom = true; + this.type = 'video'; + this.options = options; + } + + load = () => { + this._loaded = false; + const options = this.options; + + this.urls = []; + for (const url of options.urls) { + this.urls.push(this.map._requestManager.transformRequest(url, ResourceType.Source).url); + } + + getVideo(this.urls, (err, video) => { + this._loaded = true; + if (err) { + this.fire(new ErrorEvent(err)); + } else if (video) { + this.video = video; + this.video.loop = true; + + // Start repainting when video starts playing. hasTransition() will then return + // true to trigger additional frames as long as the videos continues playing. + this.video.addEventListener('playing', () => { + this.map.triggerRepaint(); + }); + + if (this.map) { + this.video.play(); + } + + this._finishLoading(); + } + }); + }; + + /** + * Pauses the video. + */ + pause() { + if (this.video) { + this.video.pause(); + } + } + + /** + * Plays the video. + */ + play() { + if (this.video) { + this.video.play(); + } + } + + /** + * Sets playback to a timestamp, in seconds. + */ + seek(seconds: number) { + if (this.video) { + const seekableRange = this.video.seekable; + if (seconds < seekableRange.start(0) || seconds > seekableRange.end(0)) { + this.fire(new ErrorEvent(new ValidationError(`sources.${this.id}`, null, `Playback for this video can be set only between the ${seekableRange.start(0)} and ${seekableRange.end(0)}-second mark.`))); + } else this.video.currentTime = seconds; + } + } + + /** + * Returns the HTML `video` element. + * + * @returns The HTML `video` element. + */ + getVideo(): HTMLVideoElement { + return this.video; + } + + onAdd(map: Map) { + if (this.map) return; + this.map = map; + this.load(); + if (this.video) { + this.video.play(); + this.setCoordinates(this.coordinates); + } + } + + /** + * Sets the video's coordinates and re-renders the map. + * + * @returns `this` + */ + prepare = (): this => { + if (Object.keys(this.tiles).length === 0 || this.video.readyState < 2) { + return; // not enough data for current position + } + + const context = this.map.painter.context; + const gl = context.gl; + + if (!this.boundsBuffer) { + this.boundsBuffer = context.createVertexBuffer(this._boundsArray, rasterBoundsAttributes.members); + } + + if (!this.boundsSegments) { + this.boundsSegments = SegmentVector.simpleSegment(0, 0, 4, 2); + } + + if (!this.texture) { + this.texture = new Texture(context, this.video, gl.RGBA); + this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + } else if (!this.video.paused) { + this.texture.bind(gl.LINEAR, gl.CLAMP_TO_EDGE); + gl.texSubImage2D(gl.TEXTURE_2D, 0, 0, 0, gl.RGBA, gl.UNSIGNED_BYTE, this.video); + } + + let newTilesLoaded = false; + for (const w in this.tiles) { + const tile = this.tiles[w]; + if (tile.state !== 'loaded') { + tile.state = 'loaded'; + tile.texture = this.texture; + newTilesLoaded = true; + } + } + + if (newTilesLoaded) { + this.fire(new Event('data', {dataType: 'source', sourceDataType: 'idle', sourceId: this.id})); + } + }; + + serialize = (): VideoSourceSpecification => { + return { + type: 'video', + urls: this.urls, + coordinates: this.coordinates + }; + }; + + hasTransition() { + return this.video && !this.video.paused; + } +} diff --git a/web/libraries/maplibre-gl/src/source/worker.test.ts b/web/libraries/maplibre-gl/src/source/worker.test.ts new file mode 100644 index 00000000..26d32097 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/worker.test.ts @@ -0,0 +1,153 @@ +import {fakeServer} from 'nise'; +import Worker from './worker'; +import {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {Cancelable} from '../types/cancelable'; +import {WorkerGlobalScopeInterface} from '../util/web_worker'; +import {CanonicalTileID, OverscaledTileID} from './tile_id'; +import {TileParameters, WorkerSource, WorkerTileCallback, WorkerTileParameters} from './worker_source'; +import {plugin as globalRTLTextPlugin} from './rtl_text_plugin'; +import {ActorTarget} from '../util/actor'; + +const _self = { + addEventListener() {} +} as any as WorkerGlobalScopeInterface & ActorTarget; + +class WorkerSourceMock implements WorkerSource { + availableImages: string[]; + constructor(private actor: any) {} + loadTile(_: WorkerTileParameters, __: WorkerTileCallback): void { + this.actor.send('main thread task', {}, () => {}, null); + } + reloadTile(_: WorkerTileParameters, __: WorkerTileCallback): void { + throw new Error('Method not implemented.'); + } + abortTile(_: TileParameters, __: WorkerTileCallback): void { + throw new Error('Method not implemented.'); + } + removeTile(_: TileParameters, __: WorkerTileCallback): void { + throw new Error('Method not implemented.'); + } +} + +describe('load tile', () => { + test('calls callback on error', done => { + const server = fakeServer.create(); + global.fetch = null; + const worker = new Worker(_self); + worker.loadTile('0', { + type: 'vector', + source: 'source', + uid: '0', + tileID: {overscaledZ: 0, wrap: 0, canonical: {x: 0, y: 0, z: 0} as CanonicalTileID} as any as OverscaledTileID, + request: {url: '/error'}// Sinon fake server gives 404 responses by default + } as WorkerTileParameters & { type: string }, (err) => { + expect(err).toBeTruthy(); + server.restore(); + done(); + }); + server.respond(); + }); + + test('isolates different instances\' data', () => { + const worker = new Worker(_self); + + worker.setLayers('0', [ + {id: 'one', type: 'circle'} as LayerSpecification + ], () => {}); + + worker.setLayers('1', [ + {id: 'one', type: 'circle'} as LayerSpecification, + {id: 'two', type: 'circle'} as LayerSpecification, + ], () => {}); + + expect(worker.layerIndexes[0]).not.toBe(worker.layerIndexes[1]); + }); + + test('worker source messages dispatched to the correct map instance', done => { + const worker = new Worker(_self); + const workerName = 'test'; + + worker.actor.send = (type, data, callback, mapId): Cancelable => { + expect(type).toBe('main thread task'); + expect(mapId).toBe('999'); + done(); + return {cancel: () => {}}; + }; + + _self.registerWorkerSource(workerName, WorkerSourceMock); + + expect(() => { + _self.registerWorkerSource(workerName, WorkerSourceMock); + }).toThrow(`Worker source with name "${workerName}" already registered.`); + + worker.loadTile('999', {type: 'test'} as WorkerTileParameters & { type: string }, () => {}); + }); +}); + +describe('register RTLTextPlugin', () => { + test('should not throw and set values in plugin', () => { + jest.spyOn(globalRTLTextPlugin, 'isParsed').mockImplementation(() => { + return false; + }); + + const rtlTextPlugin = { + applyArabicShaping: 'test', + processBidirectionalText: 'test', + processStyledBidirectionalText: 'test', + }; + + _self.registerRTLTextPlugin(rtlTextPlugin); + expect(globalRTLTextPlugin['applyArabicShaping']).toBe('test'); + expect(globalRTLTextPlugin['processBidirectionalText']).toBe('test'); + expect(globalRTLTextPlugin['processStyledBidirectionalText']).toBe('test'); + }); + + test('should throw if already parsed', () => { + jest.spyOn(globalRTLTextPlugin, 'isParsed').mockImplementation(() => { + return true; + }); + + const rtlTextPlugin = { + applyArabicShaping: jest.fn(), + processBidirectionalText: jest.fn(), + processStyledBidirectionalText: jest.fn(), + }; + + expect(() => { + _self.registerRTLTextPlugin(rtlTextPlugin); + }).toThrow('RTL text plugin already registered.'); + }); +}); + +describe('set Referrer', () => { + test('Referrer is set', () => { + const worker = new Worker(_self); + worker.setReferrer('fakeId', 'myMap'); + expect(worker.referrer).toBe('myMap'); + }); +}); + +describe('load worker source', () => { + test('calls callback on error', done => { + const server = fakeServer.create(); + global.fetch = null; + const worker = new Worker(_self); + worker.loadWorkerSource('0', { + url: '/error', + }, (err) => { + expect(err).toBeTruthy(); + server.restore(); + done(); + }); + server.respond(); + }); +}); + +describe('set images', () => { + test('set images', () => { + const worker = new Worker(_self); + expect(worker.availableImages['0']).toBeUndefined(); + worker.setImages('0', ['availableImages'], () => {}); + expect(worker.availableImages['0']).toEqual(['availableImages']); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/worker.ts b/web/libraries/maplibre-gl/src/source/worker.ts new file mode 100644 index 00000000..74ae806d --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/worker.ts @@ -0,0 +1,259 @@ +import {Actor, ActorTarget} from '../util/actor'; +import {StyleLayerIndex} from '../style/style_layer_index'; +import {VectorTileWorkerSource} from './vector_tile_worker_source'; +import {RasterDEMTileWorkerSource} from './raster_dem_tile_worker_source'; +import {GeoJSONWorkerSource} from './geojson_worker_source'; +import {plugin as globalRTLTextPlugin} from './rtl_text_plugin'; +import {isWorker} from '../util/util'; + +import type { + WorkerSource, + WorkerTileParameters, + WorkerDEMTileParameters, + WorkerTileCallback, + WorkerDEMTileCallback, + TileParameters +} from '../source/worker_source'; + +import type {WorkerGlobalScopeInterface} from '../util/web_worker'; +import type {Callback} from '../types/callback'; +import type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {PluginState} from './rtl_text_plugin'; + +/** + * The Worker class responsidble for background thread related execution + */ +export default class Worker { + self: WorkerGlobalScopeInterface & ActorTarget; + actor: Actor; + layerIndexes: {[_: string]: StyleLayerIndex}; + availableImages: {[_: string]: Array}; + workerSourceTypes: { + [_: string]: { + new (...args: any): WorkerSource; + }; + }; + workerSources: { + [_: string]: { + [_: string]: { + [_: string]: WorkerSource; + }; + }; + }; + demWorkerSources: { + [_: string]: { + [_: string]: RasterDEMTileWorkerSource; + }; + }; + referrer: string; + + constructor(self: WorkerGlobalScopeInterface & ActorTarget) { + this.self = self; + this.actor = new Actor(self, this); + + this.layerIndexes = {}; + this.availableImages = {}; + + this.workerSourceTypes = { + vector: VectorTileWorkerSource, + geojson: GeoJSONWorkerSource + }; + + // [mapId][sourceType][sourceName] => worker source instance + this.workerSources = {}; + this.demWorkerSources = {}; + + this.self.registerWorkerSource = (name: string, WorkerSource: { + new (...args: any): WorkerSource; + }) => { + if (this.workerSourceTypes[name]) { + throw new Error(`Worker source with name "${name}" already registered.`); + } + this.workerSourceTypes[name] = WorkerSource; + }; + + // This is invoked by the RTL text plugin when the download via the `importScripts` call has finished, and the code has been parsed. + this.self.registerRTLTextPlugin = (rtlTextPlugin: { + applyArabicShaping: Function; + processBidirectionalText: ((b: string, a: Array) => Array); + processStyledBidirectionalText?: ((c: string, b: Array, a: Array) => Array<[string, Array]>); + }) => { + if (globalRTLTextPlugin.isParsed()) { + throw new Error('RTL text plugin already registered.'); + } + globalRTLTextPlugin['applyArabicShaping'] = rtlTextPlugin.applyArabicShaping; + globalRTLTextPlugin['processBidirectionalText'] = rtlTextPlugin.processBidirectionalText; + globalRTLTextPlugin['processStyledBidirectionalText'] = rtlTextPlugin.processStyledBidirectionalText; + }; + } + + setReferrer(mapID: string, referrer: string) { + this.referrer = referrer; + } + + setImages(mapId: string, images: Array, callback: WorkerTileCallback) { + this.availableImages[mapId] = images; + for (const workerSource in this.workerSources[mapId]) { + const ws = this.workerSources[mapId][workerSource]; + for (const source in ws) { + ws[source].availableImages = images; + } + } + callback(); + } + + setLayers(mapId: string, layers: Array, callback: WorkerTileCallback) { + this.getLayerIndex(mapId).replace(layers); + callback(); + } + + updateLayers(mapId: string, params: { + layers: Array; + removedIds: Array; + }, callback: WorkerTileCallback) { + this.getLayerIndex(mapId).update(params.layers, params.removedIds); + callback(); + } + + loadTile(mapId: string, params: WorkerTileParameters & { + type: string; + }, callback: WorkerTileCallback) { + this.getWorkerSource(mapId, params.type, params.source).loadTile(params, callback); + } + + loadDEMTile(mapId: string, params: WorkerDEMTileParameters, callback: WorkerDEMTileCallback) { + this.getDEMWorkerSource(mapId, params.source).loadTile(params, callback); + } + + reloadTile(mapId: string, params: WorkerTileParameters & { + type: string; + }, callback: WorkerTileCallback) { + this.getWorkerSource(mapId, params.type, params.source).reloadTile(params, callback); + } + + abortTile(mapId: string, params: TileParameters & { + type: string; + }, callback: WorkerTileCallback) { + this.getWorkerSource(mapId, params.type, params.source).abortTile(params, callback); + } + + removeTile(mapId: string, params: TileParameters & { + type: string; + }, callback: WorkerTileCallback) { + this.getWorkerSource(mapId, params.type, params.source).removeTile(params, callback); + } + + removeDEMTile(mapId: string, params: TileParameters) { + this.getDEMWorkerSource(mapId, params.source).removeTile(params); + } + + removeSource(mapId: string, params: { + source: string; + } & { + type: string; + }, callback: WorkerTileCallback) { + + if (!this.workerSources[mapId] || + !this.workerSources[mapId][params.type] || + !this.workerSources[mapId][params.type][params.source]) { + return; + } + + const worker = this.workerSources[mapId][params.type][params.source]; + delete this.workerSources[mapId][params.type][params.source]; + + if (worker.removeSource !== undefined) { + worker.removeSource(params, callback); + } else { + callback(); + } + } + + /** + * Load a {@link WorkerSource} script at params.url. The script is run + * (using importScripts) with `registerWorkerSource` in scope, which is a + * function taking `(name, workerSourceObject)`. + */ + loadWorkerSource(map: string, params: { + url: string; + }, callback: Callback) { + try { + this.self.importScripts(params.url); + callback(); + } catch (e) { + callback(e.toString()); + } + } + + syncRTLPluginState(map: string, state: PluginState, callback: Callback) { + try { + globalRTLTextPlugin.setState(state); + const pluginURL = globalRTLTextPlugin.getPluginURL(); + if ( + globalRTLTextPlugin.isLoaded() && + !globalRTLTextPlugin.isParsed() && + pluginURL != null // Not possible when `isLoaded` is true, but keeps flow happy + ) { + this.self.importScripts(pluginURL); + const complete = globalRTLTextPlugin.isParsed(); + const error = complete ? undefined : new Error(`RTL Text Plugin failed to import scripts from ${pluginURL}`); + callback(error, complete); + } + } catch (e) { + callback(e.toString()); + } + } + + getAvailableImages(mapId: string) { + let availableImages = this.availableImages[mapId]; + + if (!availableImages) { + availableImages = []; + } + + return availableImages; + } + + getLayerIndex(mapId: string) { + let layerIndexes = this.layerIndexes[mapId]; + if (!layerIndexes) { + layerIndexes = this.layerIndexes[mapId] = new StyleLayerIndex(); + } + return layerIndexes; + } + + getWorkerSource(mapId: string, sourceType: string, sourceName: string): WorkerSource { + if (!this.workerSources[mapId]) + this.workerSources[mapId] = {}; + if (!this.workerSources[mapId][sourceType]) + this.workerSources[mapId][sourceType] = {}; + + if (!this.workerSources[mapId][sourceType][sourceName]) { + // use a wrapped actor so that we can attach a target mapId param + // to any messages invoked by the WorkerSource + const actor = { + send: (type, data, callback) => { + this.actor.send(type, data, callback, mapId); + } + }; + this.workerSources[mapId][sourceType][sourceName] = new (this.workerSourceTypes[sourceType] as any)((actor as any), this.getLayerIndex(mapId), this.getAvailableImages(mapId)); + } + + return this.workerSources[mapId][sourceType][sourceName]; + } + + getDEMWorkerSource(mapId: string, source: string) { + if (!this.demWorkerSources[mapId]) + this.demWorkerSources[mapId] = {}; + + if (!this.demWorkerSources[mapId][source]) { + this.demWorkerSources[mapId][source] = new RasterDEMTileWorkerSource(); + } + + return this.demWorkerSources[mapId][source]; + } +} + +if (isWorker()) { + (self as any).worker = new Worker(self as any); +} diff --git a/web/libraries/maplibre-gl/src/source/worker_source.ts b/web/libraries/maplibre-gl/src/source/worker_source.ts new file mode 100644 index 00000000..fd6741b4 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/worker_source.ts @@ -0,0 +1,118 @@ +import type {RequestParameters} from '../util/ajax'; +import type {RGBAImage, AlphaImage} from '../util/image'; +import type {GlyphPositions} from '../render/glyph_atlas'; +import type {ImageAtlas} from '../render/image_atlas'; +import type {OverscaledTileID} from './tile_id'; +import type {Bucket} from '../data/bucket'; +import type {FeatureIndex} from '../data/feature_index'; +import type {CollisionBoxArray} from '../data/array_types.g'; +import type {DEMData, DEMEncoding} from '../data/dem_data'; +import type {StyleGlyph} from '../style/style_glyph'; +import type {StyleImage} from '../style/style_image'; +import type {PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec'; + +export type TileParameters = { + source: string; + uid: string; +}; + +export type WorkerTileParameters = TileParameters & { + tileID: OverscaledTileID; + request: RequestParameters; + zoom: number; + maxZoom: number; + tileSize: number; + promoteId: PromoteIdSpecification; + pixelRatio: number; + showCollisionBoxes: boolean; + collectResourceTiming?: boolean; + returnDependencies?: boolean; +}; + +export type WorkerDEMTileParameters = TileParameters & { + coord: { + z: number; + x: number; + y: number; + w: number; + }; + rawImageData: RGBAImage | ImageBitmap; + encoding: DEMEncoding; + redFactor: number; + greenFactor: number; + blueFactor: number; + baseShift: number; +}; + +/** + * @internal + * The worker tile's result type + */ +export type WorkerTileResult = { + buckets: Array; + imageAtlas: ImageAtlas; + glyphAtlasImage: AlphaImage; + featureIndex: FeatureIndex; + collisionBoxArray: CollisionBoxArray; + rawTileData?: ArrayBuffer; + resourceTiming?: Array; + // Only used for benchmarking: + glyphMap?: { + [_: string]: { + [_: number]: StyleGlyph; + }; + } | null; + iconMap?: { + [_: string]: StyleImage; + } | null; + glyphPositions?: GlyphPositions | null; +}; + +export type WorkerTileCallback = (error?: Error | null, result?: WorkerTileResult | null) => void; +export type WorkerDEMTileCallback = (err?: Error | null, result?: DEMData | null) => void; + +/** + * May be implemented by custom source types to provide code that can be run on + * the WebWorkers. In addition to providing a custom + * {@link WorkerSource#loadTile}, any other methods attached to a `WorkerSource` + * implementation may also be targeted by the {@link Source} via + * `dispatcher.getActor().send('source-type.methodname', params, callback)`. + * + * @see {@link Map#addSourceType} + */ +export interface WorkerSource { + availableImages: Array; + // Disabled due to https://github.com/facebook/flow/issues/5208 + // constructor(actor: Actor, layerIndex: StyleLayerIndex): WorkerSource; + + /** + * Loads a tile from the given params and parse it into buckets ready to send + * back to the main thread for rendering. Should call the callback with: + * `{ buckets, featureIndex, collisionIndex, rawTileData}`. + */ + loadTile(params: WorkerTileParameters, callback: WorkerTileCallback): void; + /** + * Re-parses a tile that has already been loaded. Yields the same data as + * {@link WorkerSource#loadTile}. + */ + reloadTile(params: WorkerTileParameters, callback: WorkerTileCallback): void; + /** + * Aborts loading a tile that is in progress. + */ + abortTile(params: TileParameters, callback: WorkerTileCallback): void; + /** + * Removes this tile from any local caches. + */ + removeTile(params: TileParameters, callback: WorkerTileCallback): void; + /** + * Tells the WorkerSource to abort in-progress tasks and release resources. + * The foreground Source is responsible for ensuring that 'removeSource' is + * the last message sent to the WorkerSource. + */ + removeSource?: ( + params: { + source: string; + }, + callback: WorkerTileCallback + ) => void; +} diff --git a/web/libraries/maplibre-gl/src/source/worker_tile.test.ts b/web/libraries/maplibre-gl/src/source/worker_tile.test.ts new file mode 100644 index 00000000..c914ee28 --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/worker_tile.test.ts @@ -0,0 +1,245 @@ +import {WorkerTile} from '../source/worker_tile'; +import {GeoJSONWrapper, Feature} from '../source/geojson_wrapper'; +import {OverscaledTileID} from '../source/tile_id'; +import {StyleLayerIndex} from '../style/style_layer_index'; +import {WorkerTileParameters} from './worker_source'; +import {Actor} from '../util/actor'; +import {VectorTile} from '@mapbox/vector-tile'; + +function createWorkerTile() { + return new WorkerTile({ + uid: '', + zoom: 0, + maxZoom: 20, + tileSize: 512, + source: 'source', + tileID: new OverscaledTileID(1, 0, 1, 1, 1), + overscaling: 1 + } as any as WorkerTileParameters); +} + +function createWrapper() { + return new GeoJSONWrapper([{ + type: 1, + geometry: [0, 0], + tags: {} + } as any as Feature]); +} + +describe('worker tile', () => { + test('WorkerTile#parse', done => { + const layerIndex = new StyleLayerIndex([{ + id: 'test', + source: 'source', + type: 'circle' + }]); + + const tile = createWorkerTile(); + tile.parse(createWrapper(), layerIndex, [], {} as Actor, (err, result) => { + expect(err).toBeFalsy(); + expect(result.buckets[0]).toBeTruthy(); + done(); + }); + }); + + test('WorkerTile#parse skips hidden layers', done => { + const layerIndex = new StyleLayerIndex([{ + id: 'test-hidden', + source: 'source', + type: 'fill', + layout: {visibility: 'none'} + }]); + + const tile = createWorkerTile(); + tile.parse(createWrapper(), layerIndex, [], {} as Actor, (err, result) => { + expect(err).toBeFalsy(); + expect(result.buckets).toHaveLength(0); + done(); + }); + }); + + test('WorkerTile#parse skips layers without a corresponding source layer', done => { + const layerIndex = new StyleLayerIndex([{ + id: 'test', + source: 'source', + 'source-layer': 'nonesuch', + type: 'fill' + }]); + + const tile = createWorkerTile(); + tile.parse({layers: {}}, layerIndex, [], {} as Actor, (err, result) => { + expect(err).toBeFalsy(); + expect(result.buckets).toHaveLength(0); + done(); + }); + }); + + test('WorkerTile#parse warns once when encountering a v1 vector tile layer', done => { + const layerIndex = new StyleLayerIndex([{ + id: 'test', + source: 'source', + 'source-layer': 'test', + type: 'fill' + }]); + + const data = { + layers: { + test: { + version: 1 + } + } + } as any as VectorTile; + + const spy = jest.spyOn(console, 'warn').mockImplementation(() => {}); + + const tile = createWorkerTile(); + tile.parse(data, layerIndex, [], {} as Actor, (err) => { + expect(err).toBeFalsy(); + expect(spy.mock.calls[0][0]).toMatch(/does not use vector tile spec v2/); + done(); + }); + }); + + test('WorkerTile#parse would request all types of dependencies', done => { + const tile = createWorkerTile(); + const layerIndex = new StyleLayerIndex([{ + id: '1', + type: 'fill', + source: 'source', + 'source-layer': 'test', + paint: { + 'fill-pattern': 'hello' + } + }, { + id: 'test', + source: 'source', + 'source-layer': 'test', + type: 'symbol', + layout: { + 'icon-image': 'hello', + 'text-font': ['StandardFont-Bold'], + 'text-field': '{name}' + } + }]); + + const data = { + layers: { + test: { + version: 2, + name: 'test', + extent: 8192, + length: 1, + feature: (featureIndex: number) => ({ + extent: 8192, + type: 1, + id: featureIndex, + properties: { + name: 'test' + }, + loadGeometry () { + return [[{x: 0, y: 0}]]; + } + }) + } + } + } as any as VectorTile; + + const send = jest.fn().mockImplementation((type: string, data: unknown, callback: Function) => { + setTimeout(() => callback(null, + type === 'getImages' ? + {'hello': {width: 1, height: 1, data: new Uint8Array([0])}} : + {'StandardFont-Bold': {width: 1, height: 1, data: new Uint8Array([0])}} + )); + }); + + const actorMock = { + send + } as unknown as Actor; + tile.parse(data, layerIndex, ['hello'], actorMock, (err, result) => { + expect(err).toBeFalsy(); + expect(result).toBeDefined(); + expect(send).toHaveBeenCalledTimes(3); + expect(send).toHaveBeenCalledWith('getImages', expect.objectContaining({'icons': ['hello'], 'type': 'icons'}), expect.any(Function)); + expect(send).toHaveBeenCalledWith('getImages', expect.objectContaining({'icons': ['hello'], 'type': 'patterns'}), expect.any(Function)); + expect(send).toHaveBeenCalledWith('getGlyphs', expect.objectContaining({'source': 'source', 'type': 'glyphs', 'stacks': {'StandardFont-Bold': [101, 115, 116]}}), expect.any(Function)); + done(); + }); + }); + + test('WorkerTile#parse would cancel and only event once on repeated reparsing', done => { + const tile = createWorkerTile(); + const layerIndex = new StyleLayerIndex([{ + id: '1', + type: 'fill', + source: 'source', + 'source-layer': 'test', + paint: { + 'fill-pattern': 'hello' + } + }, { + id: 'test', + source: 'source', + 'source-layer': 'test', + type: 'symbol', + layout: { + 'icon-image': 'hello', + 'text-font': ['StandardFont-Bold'], + 'text-field': '{name}' + } + }]); + + const data = { + layers: { + test: { + version: 2, + name: 'test', + extent: 8192, + length: 1, + feature: (featureIndex: number) => ({ + extent: 8192, + type: 1, + id: featureIndex, + properties: { + name: 'test' + }, + loadGeometry () { + return [[{x: 0, y: 0}]]; + } + }) + } + } + } as any as VectorTile; + + let cancelCount = 0; + const send = jest.fn().mockImplementation((type: string, data: unknown, callback: Function) => { + const res = setTimeout(() => callback(null, + type === 'getImages' ? + {'hello': {width: 1, height: 1, data: new Uint8Array([0])}} : + {'StandardFont-Bold': {width: 1, height: 1, data: new Uint8Array([0])}} + )); + + return { + cancel: () => { + cancelCount += 1; + clearTimeout(res); + } + }; + }); + + const actorMock = { + send + } as unknown as Actor; + tile.parse(data, layerIndex, ['hello'], actorMock, () => done.fail('should not be called')); + tile.parse(data, layerIndex, ['hello'], actorMock, () => done.fail('should not be called')); + tile.parse(data, layerIndex, ['hello'], actorMock, (err, result) => { + expect(err).toBeFalsy(); + expect(result).toBeDefined(); + expect(cancelCount).toBe(6); + expect(send).toHaveBeenCalledTimes(9); + expect(send).toHaveBeenCalledWith('getImages', expect.objectContaining({'icons': ['hello'], 'type': 'icons'}), expect.any(Function)); + expect(send).toHaveBeenCalledWith('getImages', expect.objectContaining({'icons': ['hello'], 'type': 'patterns'}), expect.any(Function)); + expect(send).toHaveBeenCalledWith('getGlyphs', expect.objectContaining({'source': 'source', 'type': 'glyphs', 'stacks': {'StandardFont-Bold': [101, 115, 116]}}), expect.any(Function)); + done(); + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/source/worker_tile.ts b/web/libraries/maplibre-gl/src/source/worker_tile.ts new file mode 100644 index 00000000..3a13b4ce --- /dev/null +++ b/web/libraries/maplibre-gl/src/source/worker_tile.ts @@ -0,0 +1,252 @@ +import {FeatureIndex} from '../data/feature_index'; +import {performSymbolLayout} from '../symbol/symbol_layout'; +import {CollisionBoxArray} from '../data/array_types.g'; +import {DictionaryCoder} from '../util/dictionary_coder'; +import {SymbolBucket} from '../data/bucket/symbol_bucket'; +import {LineBucket} from '../data/bucket/line_bucket'; +import {FillBucket} from '../data/bucket/fill_bucket'; +import {FillExtrusionBucket} from '../data/bucket/fill_extrusion_bucket'; +import {warnOnce, mapObject} from '../util/util'; +import {ImageAtlas} from '../render/image_atlas'; +import {GlyphAtlas} from '../render/glyph_atlas'; +import {EvaluationParameters} from '../style/evaluation_parameters'; +import {OverscaledTileID} from './tile_id'; + +import type {Bucket} from '../data/bucket'; +import type {Actor} from '../util/actor'; +import type {StyleLayer} from '../style/style_layer'; +import type {StyleLayerIndex} from '../style/style_layer_index'; +import type {StyleImage} from '../style/style_image'; +import type {StyleGlyph} from '../style/style_glyph'; +import type { + WorkerTileParameters, + WorkerTileCallback, +} from '../source/worker_source'; +import type {PromoteIdSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {VectorTile} from '@mapbox/vector-tile'; +import {Cancelable} from '../types/cancelable'; + +export class WorkerTile { + tileID: OverscaledTileID; + uid: string; + zoom: number; + pixelRatio: number; + tileSize: number; + source: string; + promoteId: PromoteIdSpecification; + overscaling: number; + showCollisionBoxes: boolean; + collectResourceTiming: boolean; + returnDependencies: boolean; + + status: 'parsing' | 'done'; + data: VectorTile; + collisionBoxArray: CollisionBoxArray; + + abort: (() => void); + vectorTile: VectorTile; + inFlightDependencies: Cancelable[]; + dependencySentinel: number; + + constructor(params: WorkerTileParameters) { + this.tileID = new OverscaledTileID(params.tileID.overscaledZ, params.tileID.wrap, params.tileID.canonical.z, params.tileID.canonical.x, params.tileID.canonical.y); + this.uid = params.uid; + this.zoom = params.zoom; + this.pixelRatio = params.pixelRatio; + this.tileSize = params.tileSize; + this.source = params.source; + this.overscaling = this.tileID.overscaleFactor(); + this.showCollisionBoxes = params.showCollisionBoxes; + this.collectResourceTiming = !!params.collectResourceTiming; + this.returnDependencies = !!params.returnDependencies; + this.promoteId = params.promoteId; + this.inFlightDependencies = []; + this.dependencySentinel = -1; + } + + parse(data: VectorTile, layerIndex: StyleLayerIndex, availableImages: Array, actor: Actor, callback: WorkerTileCallback) { + this.status = 'parsing'; + this.data = data; + + this.collisionBoxArray = new CollisionBoxArray(); + const sourceLayerCoder = new DictionaryCoder(Object.keys(data.layers).sort()); + + const featureIndex = new FeatureIndex(this.tileID, this.promoteId); + featureIndex.bucketLayerIDs = []; + + const buckets: {[_: string]: Bucket} = {}; + + const options = { + featureIndex, + iconDependencies: {}, + patternDependencies: {}, + glyphDependencies: {}, + availableImages + }; + + const layerFamilies = layerIndex.familiesBySource[this.source]; + for (const sourceLayerId in layerFamilies) { + const sourceLayer = data.layers[sourceLayerId]; + if (!sourceLayer) { + continue; + } + + if (sourceLayer.version === 1) { + warnOnce(`Vector tile source "${this.source}" layer "${sourceLayerId}" ` + + 'does not use vector tile spec v2 and therefore may have some rendering errors.'); + } + + const sourceLayerIndex = sourceLayerCoder.encode(sourceLayerId); + const features = []; + for (let index = 0; index < sourceLayer.length; index++) { + const feature = sourceLayer.feature(index); + const id = featureIndex.getId(feature, sourceLayerId); + features.push({feature, id, index, sourceLayerIndex}); + } + + for (const family of layerFamilies[sourceLayerId]) { + const layer = family[0]; + + if (layer.source !== this.source) { + warnOnce(`layer.source = ${layer.source} does not equal this.source = ${this.source}`); + } + if (layer.minzoom && this.zoom < Math.floor(layer.minzoom)) continue; + if (layer.maxzoom && this.zoom >= layer.maxzoom) continue; + if (layer.visibility === 'none') continue; + + recalculateLayers(family, this.zoom, availableImages); + + const bucket = buckets[layer.id] = layer.createBucket({ + index: featureIndex.bucketLayerIDs.length, + layers: family, + zoom: this.zoom, + pixelRatio: this.pixelRatio, + overscaling: this.overscaling, + collisionBoxArray: this.collisionBoxArray, + sourceLayerIndex, + sourceID: this.source + }); + + bucket.populate(features, options, this.tileID.canonical); + featureIndex.bucketLayerIDs.push(family.map((l) => l.id)); + } + } + + let error: Error; + let glyphMap: { + [_: string]: { + [_: number]: StyleGlyph; + }; + }; + let iconMap: {[_: string]: StyleImage}; + let patternMap: {[_: string]: StyleImage}; + + const stacks = mapObject(options.glyphDependencies, (glyphs) => Object.keys(glyphs).map(Number)); + + this.inFlightDependencies.forEach((request) => request?.cancel()); + this.inFlightDependencies = []; + + // cancelling seems to be not sufficient, we seems to still manage to get a callback hit, so use a sentinel to drop stale results + const dependencySentinel = ++this.dependencySentinel; + if (Object.keys(stacks).length) { + this.inFlightDependencies.push(actor.send('getGlyphs', {uid: this.uid, stacks, source: this.source, tileID: this.tileID, type: 'glyphs'}, (err, result) => { + if (dependencySentinel !== this.dependencySentinel) { + return; + } + if (!error) { + error = err; + glyphMap = result; + maybePrepare.call(this); + } + })); + } else { + glyphMap = {}; + } + + const icons = Object.keys(options.iconDependencies); + if (icons.length) { + this.inFlightDependencies.push(actor.send('getImages', {icons, source: this.source, tileID: this.tileID, type: 'icons'}, (err, result) => { + if (dependencySentinel !== this.dependencySentinel) { + return; + } + if (!error) { + error = err; + iconMap = result; + maybePrepare.call(this); + } + })); + } else { + iconMap = {}; + } + + const patterns = Object.keys(options.patternDependencies); + if (patterns.length) { + this.inFlightDependencies.push(actor.send('getImages', {icons: patterns, source: this.source, tileID: this.tileID, type: 'patterns'}, (err, result) => { + if (dependencySentinel !== this.dependencySentinel) { + return; + } + if (!error) { + error = err; + patternMap = result; + maybePrepare.call(this); + } + })); + } else { + patternMap = {}; + } + + maybePrepare.call(this); + + function maybePrepare() { + if (error) { + return callback(error); + } else if (glyphMap && iconMap && patternMap) { + const glyphAtlas = new GlyphAtlas(glyphMap); + const imageAtlas = new ImageAtlas(iconMap, patternMap); + + for (const key in buckets) { + const bucket = buckets[key]; + if (bucket instanceof SymbolBucket) { + recalculateLayers(bucket.layers, this.zoom, availableImages); + performSymbolLayout({ + bucket, + glyphMap, + glyphPositions: glyphAtlas.positions, + imageMap: iconMap, + imagePositions: imageAtlas.iconPositions, + showCollisionBoxes: this.showCollisionBoxes, + canonical: this.tileID.canonical + }); + } else if (bucket.hasPattern && + (bucket instanceof LineBucket || + bucket instanceof FillBucket || + bucket instanceof FillExtrusionBucket)) { + recalculateLayers(bucket.layers, this.zoom, availableImages); + bucket.addFeatures(options, this.tileID.canonical, imageAtlas.patternPositions); + } + } + + this.status = 'done'; + callback(null, { + buckets: Object.values(buckets).filter(b => !b.isEmpty()), + featureIndex, + collisionBoxArray: this.collisionBoxArray, + glyphAtlasImage: glyphAtlas.image, + imageAtlas, + // Only used for benchmarking: + glyphMap: this.returnDependencies ? glyphMap : null, + iconMap: this.returnDependencies ? iconMap : null, + glyphPositions: this.returnDependencies ? glyphAtlas.positions : null + }); + } + } + } +} + +function recalculateLayers(layers: ReadonlyArray, zoom: number, availableImages: Array) { + // Layers are shared and may have been used by a WorkerTile with a different zoom. + const parameters = new EvaluationParameters(zoom); + for (const layer of layers) { + layer.recalculate(parameters, availableImages); + } +} diff --git a/web/libraries/maplibre-gl/src/style/create_style_layer.ts b/web/libraries/maplibre-gl/src/style/create_style_layer.ts new file mode 100644 index 00000000..25905cad --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/create_style_layer.ts @@ -0,0 +1,39 @@ +import {CircleStyleLayer} from './style_layer/circle_style_layer'; +import {HeatmapStyleLayer} from './style_layer/heatmap_style_layer'; +import {HillshadeStyleLayer} from './style_layer/hillshade_style_layer'; +import {FillStyleLayer} from './style_layer/fill_style_layer'; +import {FillExtrusionStyleLayer} from './style_layer/fill_extrusion_style_layer'; +import {LineStyleLayer} from './style_layer/line_style_layer'; +import {SymbolStyleLayer} from './style_layer/symbol_style_layer'; +import {BackgroundStyleLayer} from './style_layer/background_style_layer'; +import {RasterStyleLayer} from './style_layer/raster_style_layer'; +import {CustomStyleLayer, type CustomLayerInterface} from './style_layer/custom_style_layer'; + +import type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; + +export function createStyleLayer(layer: LayerSpecification | CustomLayerInterface) { + if (layer.type === 'custom') { + return new CustomStyleLayer(layer); + } + switch (layer.type) { + case 'background': + return new BackgroundStyleLayer(layer); + case 'circle': + return new CircleStyleLayer(layer); + case 'fill': + return new FillStyleLayer(layer); + case 'fill-extrusion': + return new FillExtrusionStyleLayer(layer); + case 'heatmap': + return new HeatmapStyleLayer(layer); + case 'hillshade': + return new HillshadeStyleLayer(layer); + case 'line': + return new LineStyleLayer(layer); + case 'raster': + return new RasterStyleLayer(layer); + case 'symbol': + return new SymbolStyleLayer(layer); + } +} + diff --git a/web/libraries/maplibre-gl/src/style/evaluation_parameters.ts b/web/libraries/maplibre-gl/src/style/evaluation_parameters.ts new file mode 100644 index 00000000..0c463fd2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/evaluation_parameters.ts @@ -0,0 +1,62 @@ +import {ZoomHistory} from './zoom_history'; +import {isStringInSupportedScript} from '../util/script_detection'; +import {plugin as rtlTextPlugin} from '../source/rtl_text_plugin'; + +import type {TransitionSpecification} from '@maplibre/maplibre-gl-style-spec'; + +export type CrossfadeParameters = { + fromScale: number; + toScale: number; + t: number; +}; + +/** + * @internal + * A parameter that can be evaluated to a value + */ +export class EvaluationParameters { + zoom: number; + now: number; + fadeDuration: number; + zoomHistory: ZoomHistory; + transition: TransitionSpecification; + + // "options" may also be another EvaluationParameters to copy, see CrossFadedProperty.possiblyEvaluate + constructor(zoom: number, options?: any) { + this.zoom = zoom; + + if (options) { + this.now = options.now; + this.fadeDuration = options.fadeDuration; + this.zoomHistory = options.zoomHistory; + this.transition = options.transition; + } else { + this.now = 0; + this.fadeDuration = 0; + this.zoomHistory = new ZoomHistory(); + this.transition = {}; + } + } + + isSupportedScript(str: string): boolean { + return isStringInSupportedScript(str, rtlTextPlugin.isLoaded()); + } + + crossFadingFactor() { + if (this.fadeDuration === 0) { + return 1; + } else { + return Math.min((this.now - this.zoomHistory.lastIntegerZoomTime) / this.fadeDuration, 1); + } + } + + getCrossfadeParameters(): CrossfadeParameters { + const z = this.zoom; + const fraction = z - Math.floor(z); + const t = this.crossFadingFactor(); + + return z > this.zoomHistory.lastIntegerZoom ? + {fromScale: 2, toScale: 1, t: fraction + (1 - fraction) * t} : + {fromScale: 0.5, toScale: 1, t: 1 - (1 - t) * fraction}; + } +} diff --git a/web/libraries/maplibre-gl/src/style/format_section_override.test.ts b/web/libraries/maplibre-gl/src/style/format_section_override.test.ts new file mode 100644 index 00000000..a3ee8d6c --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/format_section_override.test.ts @@ -0,0 +1,62 @@ +import {EvaluationContext, FormattedSection, createExpression, StyleExpression, ZoomConstantExpression} from '@maplibre/maplibre-gl-style-spec'; +import properties from './style_layer/symbol_style_layer_properties.g'; +import {PossiblyEvaluatedPropertyValue} from './properties'; +import {FormatSectionOverride} from './format_section_override'; +import {EvaluationParameters} from './evaluation_parameters'; + +describe('evaluate', () => { + + test('override constant', () => { + const defaultColor = {'r': 0, 'g': 1, 'b': 0, 'a': 1}; + const overridenColor = {'r': 1, 'g': 0, 'b': 0, 'a': 1}; + const overriden = new PossiblyEvaluatedPropertyValue( + properties.paint.properties['text-color'], + {kind: 'constant', value: defaultColor}, + {zoom: 0, zoomHistory: {}} as EvaluationParameters + ); + + const override = new FormatSectionOverride(overriden); + const ctx = new EvaluationContext(); + ctx.feature = {} as any; + ctx.featureState = {}; + expect(override.evaluate(ctx)).toEqual(defaultColor); + + ctx.formattedSection = {textColor: overridenColor} as FormattedSection; + expect(override.evaluate(ctx)).toEqual(overridenColor); + + }); + + test('override expression', () => { + const warn = console.warn; + console.warn = (_) => {}; + const defaultColor = {'r': 0, 'g': 0, 'b': 0, 'a': 1}; + const propertyColor = {'r': 1, 'g': 0, 'b': 0, 'a': 1}; + const overridenColor = {'r': 0, 'g': 0, 'b': 1, 'a': 1}; + const styleExpr = createExpression( + ['get', 'color'], + properties.paint.properties['text-color'].specification); + + const sourceExpr = new ZoomConstantExpression('source', styleExpr.value as StyleExpression); + const overriden = new PossiblyEvaluatedPropertyValue( + properties.paint.properties['text-color'], + sourceExpr, + {zoom: 0, zoomHistory: {}} as EvaluationParameters + ); + + const override = new FormatSectionOverride(overriden); + const ctx = new EvaluationContext(); + ctx.feature = {properties: {}} as any; + ctx.featureState = {}; + + expect(override.evaluate(ctx)).toEqual(defaultColor); + + ctx.feature.properties.color = 'red'; + expect(override.evaluate(ctx)).toEqual(propertyColor); + + ctx.formattedSection = {textColor: overridenColor} as FormattedSection; + expect(override.evaluate(ctx)).toEqual(overridenColor); + + console.warn = warn; + }); + +}); diff --git a/web/libraries/maplibre-gl/src/style/format_section_override.ts b/web/libraries/maplibre-gl/src/style/format_section_override.ts new file mode 100644 index 00000000..ca132826 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/format_section_override.ts @@ -0,0 +1,50 @@ +import type {Expression, EvaluationContext, Type, ZoomConstantExpression} from '@maplibre/maplibre-gl-style-spec'; +import {NullType} from '@maplibre/maplibre-gl-style-spec'; +import {PossiblyEvaluatedPropertyValue} from './properties'; +import {register} from '../util/web_worker_transfer'; + +// This is an internal expression class. It is only used in GL JS and +// has GL JS dependencies which can break the standalone style-spec module +export class FormatSectionOverride implements Expression { + type: Type; + defaultValue: PossiblyEvaluatedPropertyValue; + + constructor(defaultValue: PossiblyEvaluatedPropertyValue) { + if (defaultValue.property.overrides === undefined) throw new Error('overrides must be provided to instantiate FormatSectionOverride class'); + this.type = defaultValue.property.overrides ? defaultValue.property.overrides.runtimeType : NullType; + this.defaultValue = defaultValue; + } + + evaluate(ctx: EvaluationContext) { + if (ctx.formattedSection) { + const overrides = this.defaultValue.property.overrides; + if (overrides && overrides.hasOverride(ctx.formattedSection)) { + return overrides.getOverride(ctx.formattedSection); + } + } + + if (ctx.feature && ctx.featureState) { + return this.defaultValue.evaluate(ctx.feature, ctx.featureState); + } + + return this.defaultValue.property.specification.default; + } + + eachChild(fn: (_: Expression) => void) { + if (!this.defaultValue.isConstant()) { + const expr: ZoomConstantExpression<'source'> = (this.defaultValue.value as any); + fn(expr._styleExpression.expression); + } + } + + // Cannot be statically evaluated, as the output depends on the evaluation context. + outputDefined() { + return false; + } + + serialize() { + return null; + } +} + +register('FormatSectionOverride', FormatSectionOverride, {omit: ['defaultValue']}); diff --git a/web/libraries/maplibre-gl/src/style/light.test.ts b/web/libraries/maplibre-gl/src/style/light.test.ts new file mode 100644 index 00000000..f022a7b9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/light.test.ts @@ -0,0 +1,86 @@ +import {Light} from './light'; +import {Color, latest as styleSpec, LightSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {sphericalToCartesian} from '../util/util'; +import {EvaluationParameters} from './evaluation_parameters'; +import {TransitionParameters} from './properties'; + +const spec = styleSpec.light; + +test('Light with defaults', () => { + const light = new Light({}); + light.recalculate({zoom: 0, zoomHistory: {}} as EvaluationParameters); + + expect(light.properties.get('anchor')).toEqual(spec.anchor.default); + expect(light.properties.get('position')).toEqual(sphericalToCartesian(spec.position.default as any as [number, number, number])); + expect(light.properties.get('intensity')).toEqual(spec.intensity.default); + expect(light.properties.get('color')).toEqual(Color.parse(spec.color.default)); +}); + +test('Light with options', () => { + const light = new Light({ + anchor: 'map', + position: [2, 30, 30], + intensity: 1 + }); + light.recalculate({zoom: 0, zoomHistory: {}} as EvaluationParameters); + + expect(light.properties.get('anchor')).toBe('map'); + expect(light.properties.get('position')).toEqual(sphericalToCartesian([2, 30, 30])); + expect(light.properties.get('intensity')).toBe(1); + expect(light.properties.get('color')).toEqual(Color.parse(spec.color.default)); +}); + +test('Light with stops function', () => { + const light = new Light({ + intensity: { + stops: [[16, 0.2], [17, 0.8]] + } + } as LightSpecification); + light.recalculate({zoom: 16.5, zoomHistory: {}} as EvaluationParameters); + + expect(light.properties.get('intensity')).toBe(0.5); +}); + +test('Light#getLight', () => { + const defaults = {}; + for (const key in spec) { + defaults[key] = spec[key].default; + } + + expect(new Light(defaults).getLight()).toEqual(defaults); +}); + +describe('Light#setLight', () => { + test('sets light', () => { + const light = new Light({}); + light.setLight({color: 'red', 'color-transition': {duration: 3000}} as LightSpecification); + light.updateTransitions({transition: true} as any as TransitionParameters); + light.recalculate({zoom: 16, zoomHistory: {}, now: 1500} as EvaluationParameters); + expect(light.properties.get('color')).toEqual(new Color(1, 0.5, 0.5, 1)); + }); + + test('validates by default', () => { + const light = new Light({}); + const lightSpy = jest.spyOn(light, '_validate'); + jest.spyOn(console, 'error').mockImplementation(() => { }); + light.setLight({color: 'notacolor'}); + light.updateTransitions({transition: false} as any as TransitionParameters); + light.recalculate({zoom: 16, zoomHistory: {}, now: 10} as EvaluationParameters); + expect(lightSpy).toHaveBeenCalledTimes(1); + expect(console.error).toHaveBeenCalledTimes(1); + expect(lightSpy.mock.calls[0][2]).toEqual({}); + }); + + test('respects validation option', () => { + const light = new Light({}); + + const lightSpy = jest.spyOn(light, '_validate'); + light.setLight({color: [999]} as any, {validate: false}); + light.updateTransitions({transition: false} as any as TransitionParameters); + light.recalculate({zoom: 16, zoomHistory: {}, now: 10} as EvaluationParameters); + + expect(lightSpy).toHaveBeenCalledTimes(1); + expect(lightSpy.mock.calls[0][2]).toEqual({validate: false}); + expect(light.properties.get('color')).toEqual([999]); + }); +}); diff --git a/web/libraries/maplibre-gl/src/style/light.ts b/web/libraries/maplibre-gl/src/style/light.ts new file mode 100644 index 00000000..bfefb0f0 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/light.ts @@ -0,0 +1,135 @@ +import {interpolates, Color, latest as styleSpec} from '@maplibre/maplibre-gl-style-spec'; + +import {extend, sphericalToCartesian} from '../util/util'; +import {Evented} from '../util/evented'; +import { + validateStyle, + validateLight, + emitValidationErrors +} from './validate_style'; + +import type {StylePropertySpecification, LightSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {EvaluationParameters} from './evaluation_parameters'; +import type {StyleSetterOptions} from '../style/style'; +import {Properties, Transitionable, Transitioning, PossiblyEvaluated, DataConstantProperty} from './properties'; + +import type { + Property, + PropertyValue, + TransitionParameters +} from './properties'; + +type LightPosition = { + x: number; + y: number; + z: number; +}; + +class LightPositionProperty implements Property<[number, number, number], LightPosition> { + specification: StylePropertySpecification; + + constructor() { + this.specification = styleSpec.light.position as StylePropertySpecification; + } + + possiblyEvaluate( + value: PropertyValue<[number, number, number], LightPosition>, + parameters: EvaluationParameters + ): LightPosition { + return sphericalToCartesian(value.expression.evaluate(parameters)); + } + + interpolate(a: LightPosition, b: LightPosition, t: number): LightPosition { + return { + x: interpolates.number(a.x, b.x, t), + y: interpolates.number(a.y, b.y, t), + z: interpolates.number(a.z, b.z, t), + }; + } +} + +type Props = { + 'anchor': DataConstantProperty<'map' | 'viewport'>; + 'position': LightPositionProperty; + 'color': DataConstantProperty; + 'intensity': DataConstantProperty; +}; + +type PropsPossiblyEvaluated = { + 'anchor': 'map' | 'viewport'; + 'position': LightPosition; + 'color': Color; + 'intensity': number; +}; + +const TRANSITION_SUFFIX = '-transition'; + +let lightProperties: Properties; + +/* + * Represents the light used to light extruded features. + */ +export class Light extends Evented { + _transitionable: Transitionable; + _transitioning: Transitioning; + properties: PossiblyEvaluated; + + constructor(lightOptions?: LightSpecification) { + super(); + lightProperties = lightProperties || new Properties({ + 'anchor': new DataConstantProperty(styleSpec.light.anchor as StylePropertySpecification), + 'position': new LightPositionProperty(), + 'color': new DataConstantProperty(styleSpec.light.color as StylePropertySpecification), + 'intensity': new DataConstantProperty(styleSpec.light.intensity as StylePropertySpecification), + }); + this._transitionable = new Transitionable(lightProperties); + this.setLight(lightOptions); + this._transitioning = this._transitionable.untransitioned(); + } + + getLight(): LightSpecification { + return this._transitionable.serialize(); + } + + setLight(light?: LightSpecification, options: StyleSetterOptions = {}) { + if (this._validate(validateLight, light, options)) { + return; + } + + for (const name in light) { + const value = light[name]; + if (name.endsWith(TRANSITION_SUFFIX)) { + this._transitionable.setTransition(name.slice(0, -TRANSITION_SUFFIX.length) as keyof Props, value); + } else { + this._transitionable.setValue(name as keyof Props, value); + } + } + } + + updateTransitions(parameters: TransitionParameters) { + this._transitioning = this._transitionable.transitioned(parameters, this._transitioning); + } + + hasTransition() { + return this._transitioning.hasTransition(); + } + + recalculate(parameters: EvaluationParameters) { + this.properties = this._transitioning.possiblyEvaluate(parameters); + } + + _validate(validate: Function, value: unknown, options?: { + validate?: boolean; + }) { + if (options && options.validate === false) { + return false; + } + + return emitValidationErrors(this, validate.call(validateStyle, extend({ + value, + // Workaround for https://github.com/mapbox/mapbox-gl-js/issues/2407 + style: {glyphs: true, sprite: true}, + styleSpec + }))); + } +} diff --git a/web/libraries/maplibre-gl/src/style/load_glyph_range.test.ts b/web/libraries/maplibre-gl/src/style/load_glyph_range.test.ts new file mode 100644 index 00000000..12500f9e --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/load_glyph_range.test.ts @@ -0,0 +1,41 @@ +import fs from 'fs'; +import path from 'path'; +import {RequestManager} from '../util/request_manager'; +import {loadGlyphRange} from './load_glyph_range'; +import {fakeServer} from 'nise'; +import {bufferToArrayBuffer} from '../util/test/util'; + +test('loadGlyphRange', done => { + global.fetch = null; + + const transform = jest.fn().mockImplementation((url) => { + return {url}; + }); + + const manager = new RequestManager(transform); + + const server = fakeServer.create(); + server.respondWith(bufferToArrayBuffer(fs.readFileSync(path.join(__dirname, '../../test/unit/assets/0-255.pbf')))); + + loadGlyphRange('Arial Unicode MS', 0, 'https://localhost/fonts/v1/{fontstack}/{range}.pbf', manager, (err, result) => { + expect(err).toBeFalsy(); + expect(transform).toHaveBeenCalledTimes(1); + expect(transform).toHaveBeenCalledWith('https://localhost/fonts/v1/Arial Unicode MS/0-255.pbf', 'Glyphs'); + + expect(Object.keys(result)).toHaveLength(223); + for (const key in result) { + const id = Number(key); + const glyph = result[id]; + + expect(glyph.id).toBe(Number(id)); + expect(glyph.metrics).toBeTruthy(); + expect(typeof glyph.metrics.width).toBe('number'); + expect(typeof glyph.metrics.height).toBe('number'); + expect(typeof glyph.metrics.top).toBe('number'); + expect(typeof glyph.metrics.advance).toBe('number'); + } + done(); + }); + server.respond(); + expect(server.requests[0].url).toBe('https://localhost/fonts/v1/Arial Unicode MS/0-255.pbf'); +}); diff --git a/web/libraries/maplibre-gl/src/style/load_glyph_range.ts b/web/libraries/maplibre-gl/src/style/load_glyph_range.ts new file mode 100644 index 00000000..6a454caa --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/load_glyph_range.ts @@ -0,0 +1,38 @@ +import {getArrayBuffer} from '../util/ajax'; +import {ResourceType} from '../util/request_manager'; + +import {parseGlyphPbf} from './parse_glyph_pbf'; + +import type {StyleGlyph} from './style_glyph'; +import type {RequestManager} from '../util/request_manager'; +import type {Callback} from '../types/callback'; + +export function loadGlyphRange(fontstack: string, + range: number, + urlTemplate: string, + requestManager: RequestManager, + callback: Callback<{ + [_: number]: StyleGlyph | null; + }>) { + const begin = range * 256; + const end = begin + 255; + + const request = requestManager.transformRequest( + urlTemplate.replace('{fontstack}', fontstack).replace('{range}', `${begin}-${end}`), + ResourceType.Glyphs + ); + + getArrayBuffer(request, (err?: Error | null, data?: ArrayBuffer | null) => { + if (err) { + callback(err); + } else if (data) { + const glyphs = {}; + + for (const glyph of parseGlyphPbf(data)) { + glyphs[glyph.id] = glyph; + } + + callback(null, glyphs); + } + }); +} diff --git a/web/libraries/maplibre-gl/src/style/load_sprite.test.ts b/web/libraries/maplibre-gl/src/style/load_sprite.test.ts new file mode 100644 index 00000000..6aaa068f --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/load_sprite.test.ts @@ -0,0 +1,186 @@ +import fs from 'fs'; +import path from 'path'; +import {RequestManager} from '../util/request_manager'; +import {loadSprite} from './load_sprite'; +import {type FakeServer, fakeServer} from 'nise'; +import * as util from '../util/util'; +import {bufferToArrayBuffer} from '../util/test/util'; + +describe('loadSprite', () => { + + let server: FakeServer; + + beforeEach(() => { + jest.spyOn(util, 'arrayBufferToImageBitmap').mockImplementation((data: ArrayBuffer, callback: (err?: Error | null, image?: ImageBitmap | null) => void) => { + createImageBitmap(new ImageData(1024, 824)).then((imgBitmap) => { + callback(null, imgBitmap); + }).catch((e) => { + callback(new Error(`Could not load image because of ${e.message}. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.`)); + }); + }); + global.fetch = null; + server = fakeServer.create(); + }); + + test('backwards compatibility: single string is treated as a URL for the default sprite', done => { + const transform = jest.fn().mockImplementation((url, type) => { + return {url, type}; + }); + + const manager = new RequestManager(transform); + + server.respondWith('GET', 'http://localhost:9966/test/unit/assets/sprite1.json', fs.readFileSync(path.join(__dirname, '../../test/unit/assets/sprite1.json')).toString()); + server.respondWith('GET', 'http://localhost:9966/test/unit/assets/sprite1.png', bufferToArrayBuffer(fs.readFileSync(path.join(__dirname, '../../test/unit/assets/sprite1.png')))); + + loadSprite('http://localhost:9966/test/unit/assets/sprite1', manager, 1, (err, result) => { + expect(err).toBeFalsy(); + + expect(transform).toHaveBeenCalledTimes(2); + expect(transform).toHaveBeenNthCalledWith(1, 'http://localhost:9966/test/unit/assets/sprite1.json', 'SpriteJSON'); + expect(transform).toHaveBeenNthCalledWith(2, 'http://localhost:9966/test/unit/assets/sprite1.png', 'SpriteImage'); + + expect(Object.keys(result)).toHaveLength(1); + expect(Object.keys(result)[0]).toBe('default'); + + Object.values(result['default']).forEach(styleImage => { + expect(styleImage.spriteData).toBeTruthy(); + expect(styleImage.spriteData.context).toBeInstanceOf(CanvasRenderingContext2D); + }); + + done(); + }); + + server.respond(); + + expect(server.requests[0].url).toBe('http://localhost:9966/test/unit/assets/sprite1.json'); + expect(server.requests[1].url).toBe('http://localhost:9966/test/unit/assets/sprite1.png'); + }); + + test('array of objects support', done => { + const transform = jest.fn().mockImplementation((url, type) => { + return {url, type}; + }); + + const manager = new RequestManager(transform); + + server.respondWith('GET', 'http://localhost:9966/test/unit/assets/sprite1.json', fs.readFileSync(path.join(__dirname, '../../test/unit/assets/sprite1.json')).toString()); + server.respondWith('GET', 'http://localhost:9966/test/unit/assets/sprite1.png', bufferToArrayBuffer(fs.readFileSync(path.join(__dirname, '../../test/unit/assets/sprite1.png')))); + server.respondWith('GET', 'http://localhost:9966/test/unit/assets/sprite2.json', fs.readFileSync(path.join(__dirname, '../../test/unit/assets/sprite2.json')).toString()); + server.respondWith('GET', 'http://localhost:9966/test/unit/assets/sprite2.png', bufferToArrayBuffer(fs.readFileSync(path.join(__dirname, '../../test/unit/assets/sprite2.png')))); + + loadSprite([{id: 'sprite1', url: 'http://localhost:9966/test/unit/assets/sprite1'}, {id: 'sprite2', url: 'http://localhost:9966/test/unit/assets/sprite2'}], manager, 1, (err, result) => { + expect(err).toBeFalsy(); + + expect(transform).toHaveBeenCalledTimes(4); + expect(transform).toHaveBeenNthCalledWith(1, 'http://localhost:9966/test/unit/assets/sprite1.json', 'SpriteJSON'); + expect(transform).toHaveBeenNthCalledWith(2, 'http://localhost:9966/test/unit/assets/sprite1.png', 'SpriteImage'); + expect(transform).toHaveBeenNthCalledWith(3, 'http://localhost:9966/test/unit/assets/sprite2.json', 'SpriteJSON'); + expect(transform).toHaveBeenNthCalledWith(4, 'http://localhost:9966/test/unit/assets/sprite2.png', 'SpriteImage'); + + expect(Object.keys(result)).toHaveLength(2); + expect(Object.keys(result)[0]).toBe('sprite1'); + expect(Object.keys(result)[1]).toBe('sprite2'); + + Object.values(result['sprite1']).forEach(styleImage => { + expect(styleImage.spriteData).toBeTruthy(); + expect(styleImage.spriteData.context).toBeInstanceOf(CanvasRenderingContext2D); + }); + + Object.values(result['sprite2']).forEach(styleImage => { + expect(styleImage.spriteData).toBeTruthy(); + expect(styleImage.spriteData.context).toBeInstanceOf(CanvasRenderingContext2D); + }); + + done(); + }); + + server.respond(); + expect(server.requests[0].url).toBe('http://localhost:9966/test/unit/assets/sprite1.json'); + expect(server.requests[1].url).toBe('http://localhost:9966/test/unit/assets/sprite1.png'); + expect(server.requests[2].url).toBe('http://localhost:9966/test/unit/assets/sprite2.json'); + expect(server.requests[3].url).toBe('http://localhost:9966/test/unit/assets/sprite2.png'); + }); + + test('error in callback', done => { + const transform = jest.fn().mockImplementation((url, type) => { + return {url, type}; + }); + + const manager = new RequestManager(transform); + + server.respondWith((xhr) => xhr.respond(500)); + let last = false; + loadSprite([{id: 'sprite1', url: 'http://localhost:9966/test/unit/assets/sprite1'}], manager, 1, (err, result) => { + expect(err).toBeTruthy(); + expect(result).toBeUndefined(); + if (!last) { + done(); + last = true; + } + }); + + server.respond(); + expect(server.requests[0].url).toBe('http://localhost:9966/test/unit/assets/sprite1.json'); + }); + + test('request canceling', done => { + const transform = jest.fn().mockImplementation((url, type) => { + return {url, type}; + }); + + const manager = new RequestManager(transform); + + server.respondWith('GET', 'http://localhost:9966/test/unit/assets/sprite1.json', fs.readFileSync(path.join(__dirname, '../../test/unit/assets/sprite1.json')).toString()); + server.respondWith('GET', 'http://localhost:9966/test/unit/assets/sprite1.png', bufferToArrayBuffer(fs.readFileSync(path.join(__dirname, '../../test/unit/assets/sprite1.png')))); + + const cancelable = loadSprite([{id: 'sprite1', url: 'http://localhost:9966/test/unit/assets/sprite1'}], manager, 1, () => {}); + + setTimeout(() => { + cancelable.cancel(); + + expect((server.requests[0] as any).aborted).toBeTruthy(); + expect((server.requests[1] as any).aborted).toBeTruthy(); + + done(); + }); + + setTimeout(() => { + server.respond(); + expect(server.requests[0].url).toBe('http://localhost:9966/test/unit/assets/sprite1.json'); + expect(server.requests[1].url).toBe('http://localhost:9966/test/unit/assets/sprite1.png'); + }, 10); + }); + + test('pixelRatio is respected', done => { + const transform = jest.fn().mockImplementation((url, type) => { + return {url, type}; + }); + + const manager = new RequestManager(transform); + + server.respondWith('GET', 'http://localhost:9966/test/unit/assets/sprite1@2x.json', fs.readFileSync(path.join(__dirname, '../../test/unit/assets/sprite1.json')).toString()); + server.respondWith('GET', 'http://localhost:9966/test/unit/assets/sprite1@2x.png', bufferToArrayBuffer(fs.readFileSync(path.join(__dirname, '../../test/unit/assets/sprite1.png')))); + + loadSprite('http://localhost:9966/test/unit/assets/sprite1', manager, 2, (err, result) => { + expect(err).toBeFalsy(); + + expect(transform).toHaveBeenCalledTimes(2); + expect(transform).toHaveBeenNthCalledWith(1, 'http://localhost:9966/test/unit/assets/sprite1@2x.json', 'SpriteJSON'); + expect(transform).toHaveBeenNthCalledWith(2, 'http://localhost:9966/test/unit/assets/sprite1@2x.png', 'SpriteImage'); + + expect(Object.keys(result)).toHaveLength(1); + expect(Object.keys(result)[0]).toBe('default'); + + Object.values(result['default']).forEach(styleImage => { + expect(styleImage.spriteData).toBeTruthy(); + expect(styleImage.spriteData.context).toBeInstanceOf(CanvasRenderingContext2D); + }); + + done(); + }); + + server.respond(); + expect(server.requests[0].url).toBe('http://localhost:9966/test/unit/assets/sprite1@2x.json'); + expect(server.requests[1].url).toBe('http://localhost:9966/test/unit/assets/sprite1@2x.png'); + }); +}); diff --git a/web/libraries/maplibre-gl/src/style/load_sprite.ts b/web/libraries/maplibre-gl/src/style/load_sprite.ts new file mode 100644 index 00000000..2a36afe7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/load_sprite.ts @@ -0,0 +1,94 @@ +import {getJSON} from '../util/ajax'; +import {ImageRequest} from '../util/image_request'; +import {ResourceType} from '../util/request_manager'; + +import {browser} from '../util/browser'; +import {coerceSpriteToArray} from '../util/style'; + +import type {SpriteSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {StyleImage} from './style_image'; +import type {RequestManager} from '../util/request_manager'; +import type {Callback} from '../types/callback'; +import type {Cancelable} from '../types/cancelable'; + +export function loadSprite( + originalSprite: SpriteSpecification, + requestManager: RequestManager, + pixelRatio: number, + callback: Callback<{[spriteName: string]: {[id: string]: StyleImage}}> +): Cancelable { + const spriteArray = coerceSpriteToArray(originalSprite); + const spriteArrayLength = spriteArray.length; + const format = pixelRatio > 1 ? '@2x' : ''; + + const combinedRequestsMap: {[requestKey: string]: Cancelable} = {}; + const jsonsMap: {[id: string]: any} = {}; + const imagesMap: {[id: string]: (HTMLImageElement | ImageBitmap)} = {}; + + for (const {id, url} of spriteArray) { + const jsonRequestParameters = requestManager.transformRequest(requestManager.normalizeSpriteURL(url, format, '.json'), ResourceType.SpriteJSON); + const jsonRequestKey = `${id}_${jsonRequestParameters.url}`; // use id_url as requestMap key to make sure it is unique + combinedRequestsMap[jsonRequestKey] = getJSON(jsonRequestParameters, (err?: Error | null, data?: any | null) => { + delete combinedRequestsMap[jsonRequestKey]; + jsonsMap[id] = data; + doOnceCompleted(callback, jsonsMap, imagesMap, err, spriteArrayLength); + }); + + const imageRequestParameters = requestManager.transformRequest(requestManager.normalizeSpriteURL(url, format, '.png'), ResourceType.SpriteImage); + const imageRequestKey = `${id}_${imageRequestParameters.url}`; // use id_url as requestMap key to make sure it is unique + combinedRequestsMap[imageRequestKey] = ImageRequest.getImage(imageRequestParameters, (err, img) => { + delete combinedRequestsMap[imageRequestKey]; + imagesMap[id] = img; + doOnceCompleted(callback, jsonsMap, imagesMap, err, spriteArrayLength); + }); + } + + return { + cancel() { + for (const requst of Object.values(combinedRequestsMap)) { + requst.cancel(); + } + } + }; +} + +/** + * @param callbackFunc - the callback function (both erro and success) + * @param jsonsMap - JSON data map + * @param imagesMap - image data map + * @param err - error object + * @param expectedResultCounter - number of expected JSON or Image results when everything is finished, respectively. + */ +function doOnceCompleted( + callbackFunc:Callback<{[spriteName: string]: {[id: string]: StyleImage}}>, + jsonsMap:{[id: string]: any}, + imagesMap:{[id: string]: (HTMLImageElement | ImageBitmap)}, + err: Error, + expectedResultCounter: number): void { + + if (err) { + callbackFunc(err); + return; + } + + if (expectedResultCounter !== Object.values(jsonsMap).length || expectedResultCounter !== Object.values(imagesMap).length) { + // not done yet, nothing to do + return; + } + + const result = {} as {[spriteName: string]: {[id: string]: StyleImage}}; + for (const spriteName in jsonsMap) { + result[spriteName] = {}; + + const context = browser.getImageCanvasContext(imagesMap[spriteName]); + const json = jsonsMap[spriteName]; + + for (const id in json) { + const {width, height, x, y, sdf, pixelRatio, stretchX, stretchY, content} = json[id]; + const spriteData = {width, height, x, y, context}; + result[spriteName][id] = {data: null, pixelRatio, sdf, stretchX, stretchY, content, spriteData}; + } + } + + callbackFunc(null, result); +} diff --git a/web/libraries/maplibre-gl/src/style/parse_glyph_pbf.ts b/web/libraries/maplibre-gl/src/style/parse_glyph_pbf.ts new file mode 100644 index 00000000..10fefb2a --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/parse_glyph_pbf.ts @@ -0,0 +1,42 @@ +import {AlphaImage} from '../util/image'; + +import Protobuf from 'pbf'; +const border = 3; + +import type {StyleGlyph} from './style_glyph'; + +function readFontstacks(tag: number, glyphs: Array, pbf: Protobuf) { + if (tag === 1) { + pbf.readMessage(readFontstack, glyphs); + } +} + +function readFontstack(tag: number, glyphs: Array, pbf: Protobuf) { + if (tag === 3) { + const {id, bitmap, width, height, left, top, advance} = pbf.readMessage(readGlyph, {}); + glyphs.push({ + id, + bitmap: new AlphaImage({ + width: width + 2 * border, + height: height + 2 * border + }, bitmap), + metrics: {width, height, left, top, advance} + }); + } +} + +function readGlyph(tag: number, glyph: any, pbf: Protobuf) { + if (tag === 1) glyph.id = pbf.readVarint(); + else if (tag === 2) glyph.bitmap = pbf.readBytes(); + else if (tag === 3) glyph.width = pbf.readVarint(); + else if (tag === 4) glyph.height = pbf.readVarint(); + else if (tag === 5) glyph.left = pbf.readSVarint(); + else if (tag === 6) glyph.top = pbf.readSVarint(); + else if (tag === 7) glyph.advance = pbf.readVarint(); +} + +export function parseGlyphPbf(data: ArrayBuffer | Uint8Array): Array { + return new Protobuf(data).readFields(readFontstacks, []); +} + +export const GLYPH_PBF_BORDER = border; diff --git a/web/libraries/maplibre-gl/src/style/pauseable_placement.ts b/web/libraries/maplibre-gl/src/style/pauseable_placement.ts new file mode 100644 index 00000000..bd9f2269 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/pauseable_placement.ts @@ -0,0 +1,137 @@ +import {browser} from '../util/browser'; + +import {Placement} from '../symbol/placement'; + +import type {Transform} from '../geo/transform'; +import type {StyleLayer} from './style_layer'; +import type {SymbolStyleLayer} from './style_layer/symbol_style_layer'; +import type {Tile} from '../source/tile'; +import type {BucketPart} from '../symbol/placement'; +import {Terrain} from '../render/terrain'; + +class LayerPlacement { + _sortAcrossTiles: boolean; + _currentTileIndex: number; + _currentPartIndex: number; + _seenCrossTileIDs: { + [k in string | number]: boolean; + }; + _bucketParts: Array; + + constructor(styleLayer: SymbolStyleLayer) { + this._sortAcrossTiles = styleLayer.layout.get('symbol-z-order') !== 'viewport-y' && + !styleLayer.layout.get('symbol-sort-key').isConstant(); + + this._currentTileIndex = 0; + this._currentPartIndex = 0; + this._seenCrossTileIDs = {}; + this._bucketParts = []; + } + + continuePlacement(tiles: Array, placement: Placement, showCollisionBoxes: boolean, styleLayer: StyleLayer, shouldPausePlacement: () => boolean) { + + const bucketParts = this._bucketParts; + + while (this._currentTileIndex < tiles.length) { + const tile = tiles[this._currentTileIndex]; + placement.getBucketParts(bucketParts, styleLayer, tile, this._sortAcrossTiles); + + this._currentTileIndex++; + if (shouldPausePlacement()) { + return true; + } + } + + if (this._sortAcrossTiles) { + this._sortAcrossTiles = false; + bucketParts.sort((a, b) => (a.sortKey as any as number) - (b.sortKey as any as number)); + } + + while (this._currentPartIndex < bucketParts.length) { + const bucketPart = bucketParts[this._currentPartIndex]; + placement.placeLayerBucketPart(bucketPart, this._seenCrossTileIDs, showCollisionBoxes); + + this._currentPartIndex++; + if (shouldPausePlacement()) { + return true; + } + } + return false; + } +} + +export class PauseablePlacement { + placement: Placement; + _done: boolean; + _currentPlacementIndex: number; + _forceFullPlacement: boolean; + _showCollisionBoxes: boolean; + _inProgressLayer: LayerPlacement; + + constructor( + transform: Transform, + terrain: Terrain, + order: Array, + forceFullPlacement: boolean, + showCollisionBoxes: boolean, + fadeDuration: number, + crossSourceCollisions: boolean, + prevPlacement?: Placement + ) { + this.placement = new Placement(transform, terrain, fadeDuration, crossSourceCollisions, prevPlacement); + this._currentPlacementIndex = order.length - 1; + this._forceFullPlacement = forceFullPlacement; + this._showCollisionBoxes = showCollisionBoxes; + this._done = false; + } + + isDone() { + return this._done; + } + + continuePlacement( + order: Array, + layers: {[_: string]: StyleLayer}, + layerTiles: {[_: string]: Array} + ) { + const startTime = browser.now(); + + const shouldPausePlacement = () => { + return this._forceFullPlacement ? false : (browser.now() - startTime) > 2; + }; + + while (this._currentPlacementIndex >= 0) { + const layerId = order[this._currentPlacementIndex]; + const layer = layers[layerId]; + const placementZoom = this.placement.collisionIndex.transform.zoom; + if (layer.type === 'symbol' && + (!layer.minzoom || layer.minzoom <= placementZoom) && + (!layer.maxzoom || layer.maxzoom > placementZoom)) { + + if (!this._inProgressLayer) { + this._inProgressLayer = new LayerPlacement(layer as any as SymbolStyleLayer); + } + + const pausePlacement = this._inProgressLayer.continuePlacement(layerTiles[layer.source], this.placement, this._showCollisionBoxes, layer, shouldPausePlacement); + + if (pausePlacement) { + // We didn't finish placing all layers within 2ms, + // but we can keep rendering with a partial placement + // We'll resume here on the next frame + return; + } + + delete this._inProgressLayer; + } + + this._currentPlacementIndex--; + } + + this._done = true; + } + + commit(now: number) { + this.placement.commit(now); + return this.placement; + } +} diff --git a/web/libraries/maplibre-gl/src/style/properties.ts b/web/libraries/maplibre-gl/src/style/properties.ts new file mode 100644 index 00000000..ba95b37a --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/properties.ts @@ -0,0 +1,726 @@ +import {clone, extend, easeCubicInOut} from '../util/util'; +import {interpolates, Color, StylePropertySpecification, normalizePropertyExpression, + Feature, + FeatureState, + StylePropertyExpression, + SourceExpression, + CompositeExpression, TransitionSpecification, + PropertyValueSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {register} from '../util/web_worker_transfer'; +import {EvaluationParameters} from './evaluation_parameters'; + +import {CanonicalTileID} from '../source/tile_id'; + +type TimePoint = number; + +/** + * A from-to type + */ +export type CrossFaded = { + to: T; + from: T; +}; + +/** + * @internal + * Implementations of the `Property` interface: + * + * * Hold metadata about a property that's independent of any specific value: stuff like the type of the value, + * the default value, etc. This comes from the style specification JSON. + * * Define behavior that needs to be polymorphic across different properties: "possibly evaluating" + * an input value (see below), and interpolating between two possibly-evaluted values. + * + * The type `T` is the fully-evaluated value type (e.g. `number`, `string`, `Color`). + * The type `R` is the intermediate "possibly evaluated" value type. See below. + * + * There are two main implementations of the interface -- one for properties that allow data-driven values, + * and one for properties that don't. There are a few "special case" implementations as well: one for properties + * which cross-fade between two values rather than interpolating, one for `heatmap-color` and `line-gradient`, + * and one for `light-position`. + */ +export interface Property { + specification: StylePropertySpecification; + possiblyEvaluate( + value: PropertyValue, + parameters: EvaluationParameters, + canonical?: CanonicalTileID, + availableImages?: Array + ): R; + interpolate(a: R, b: R, t: number): R; +} + +/** + * @internal + * `PropertyValue` represents the value part of a property key-value unit. It's used to represent both + * paint and layout property values, and regardless of whether or not their property supports data-driven + * expressions. + * + * `PropertyValue` stores the raw input value as seen in a style or a runtime styling API call, i.e. one of the + * following: + * + * * A constant value of the type appropriate for the property + * * A function which produces a value of that type (but functions are quasi-deprecated in favor of expressions) + * * An expression which produces a value of that type + * * "undefined"/"not present", in which case the property is assumed to take on its default value. + * + * In addition to storing the original input value, `PropertyValue` also stores a normalized representation, + * effectively treating functions as if they are expressions, and constant or default values as if they are + * (constant) expressions. + */ +export class PropertyValue { + property: Property; + value: PropertyValueSpecification | void; + expression: StylePropertyExpression; + + constructor(property: Property, value: PropertyValueSpecification | void) { + this.property = property; + this.value = value; + this.expression = normalizePropertyExpression(value === undefined ? property.specification.default : value, property.specification); + } + + isDataDriven(): boolean { + return this.expression.kind === 'source' || this.expression.kind === 'composite'; + } + + possiblyEvaluate( + parameters: EvaluationParameters, + canonical?: CanonicalTileID, + availableImages?: Array + ): R { + return this.property.possiblyEvaluate(this, parameters, canonical, availableImages); + } +} + +export type TransitionParameters = { + now: TimePoint; + transition: TransitionSpecification; +}; + +/** + * @internal + * Paint properties are _transitionable_: they can change in a fluid manner, interpolating or cross-fading between + * old and new value. The duration of the transition, and the delay before it begins, is configurable. + * + * `TransitionablePropertyValue` is a compositional class that stores both the property value and that transition + * configuration. + * + * A `TransitionablePropertyValue` can calculate the next step in the evaluation chain for paint property values: + * `TransitioningPropertyValue`. + */ +class TransitionablePropertyValue { + property: Property; + value: PropertyValue; + transition: TransitionSpecification | void; + + constructor(property: Property) { + this.property = property; + this.value = new PropertyValue(property, undefined); + } + + transitioned(parameters: TransitionParameters, prior: TransitioningPropertyValue): TransitioningPropertyValue { + return new TransitioningPropertyValue(this.property, this.value, prior, + extend({}, parameters.transition, this.transition), parameters.now); + } + + untransitioned(): TransitioningPropertyValue { + return new TransitioningPropertyValue(this.property, this.value, null, {}, 0); + } +} + +/** + * @internal + * `Transitionable` stores a map of all (property name, `TransitionablePropertyValue`) pairs for paint properties of a + * given layer type. It can calculate the `TransitioningPropertyValue`s for all of them at once, producing a + * `Transitioning` instance for the same set of properties. + */ +export class Transitionable { + _properties: Properties; + _values: {[K in keyof Props]: TransitionablePropertyValue}; + + constructor(properties: Properties) { + this._properties = properties; + this._values = (Object.create(properties.defaultTransitionablePropertyValues) as any); + } + + getValue(name: S): PropertyValueSpecification | void { + return clone(this._values[name].value.value); + } + + setValue(name: S, value: PropertyValueSpecification | void) { + if (!Object.prototype.hasOwnProperty.call(this._values, name)) { + this._values[name] = new TransitionablePropertyValue(this._values[name].property); + } + // Note that we do not _remove_ an own property in the case where a value is being reset + // to the default: the transition might still be non-default. + this._values[name].value = new PropertyValue(this._values[name].property, value === null ? undefined : clone(value)); + } + + getTransition(name: S): TransitionSpecification | void { + return clone(this._values[name].transition); + } + + setTransition(name: S, value: TransitionSpecification | void) { + if (!Object.prototype.hasOwnProperty.call(this._values, name)) { + this._values[name] = new TransitionablePropertyValue(this._values[name].property); + } + this._values[name].transition = clone(value) || undefined; + } + + serialize() { + const result: any = {}; + for (const property of Object.keys(this._values)) { + const value = this.getValue(property as keyof Props); + if (value !== undefined) { + result[property] = value; + } + + const transition = this.getTransition(property as keyof Props); + if (transition !== undefined) { + result[`${property}-transition`] = transition; + } + } + return result; + } + + transitioned(parameters: TransitionParameters, prior: Transitioning): Transitioning { + const result = new Transitioning(this._properties); + for (const property of Object.keys(this._values)) { + result._values[property] = this._values[property].transitioned(parameters, prior._values[property]); + } + return result; + } + + untransitioned(): Transitioning { + const result = new Transitioning(this._properties); + for (const property of Object.keys(this._values)) { + result._values[property] = this._values[property].untransitioned(); + } + return result; + } +} + +/** + * @internal + * `TransitioningPropertyValue` implements the first of two intermediate steps in the evaluation chain of a paint + * property value. In this step, transitions between old and new values are handled: as long as the transition is in + * progress, `TransitioningPropertyValue` maintains a reference to the prior value, and interpolates between it and + * the new value based on the current time and the configured transition duration and delay. The product is the next + * step in the evaluation chain: the "possibly evaluated" result type `R`. See below for more on this concept. + */ +class TransitioningPropertyValue { + property: Property; + value: PropertyValue; + prior: TransitioningPropertyValue; + begin: TimePoint; + end: TimePoint; + + constructor(property: Property, + value: PropertyValue, + prior: TransitioningPropertyValue, + transition: TransitionSpecification, + now: TimePoint) { + this.property = property; + this.value = value; + this.begin = now + transition.delay || 0; + this.end = this.begin + transition.duration || 0; + if (property.specification.transition && (transition.delay || transition.duration)) { + this.prior = prior; + } + } + + possiblyEvaluate( + parameters: EvaluationParameters, + canonical: CanonicalTileID, + availableImages: Array + ): R { + const now = parameters.now || 0; + const finalValue = this.value.possiblyEvaluate(parameters, canonical, availableImages); + const prior = this.prior; + if (!prior) { + // No prior value. + return finalValue; + } else if (now > this.end) { + // Transition from prior value is now complete. + this.prior = null; + return finalValue; + } else if (this.value.isDataDriven()) { + // Transitions to data-driven properties are not supported. + // We snap immediately to the data-driven value so that, when we perform layout, + // we see the data-driven function and can use it to populate vertex buffers. + this.prior = null; + return finalValue; + } else if (now < this.begin) { + // Transition hasn't started yet. + return prior.possiblyEvaluate(parameters, canonical, availableImages); + } else { + // Interpolate between recursively-calculated prior value and final. + const t = (now - this.begin) / (this.end - this.begin); + return this.property.interpolate(prior.possiblyEvaluate(parameters, canonical, availableImages), finalValue, easeCubicInOut(t)); + } + } +} + +/** + * @internal + * `Transitioning` stores a map of all (property name, `TransitioningPropertyValue`) pairs for paint properties of a + * given layer type. It can calculate the possibly-evaluated values for all of them at once, producing a + * `PossiblyEvaluated` instance for the same set of properties. + */ +export class Transitioning { + _properties: Properties; + _values: {[K in keyof Props]: PossiblyEvaluatedPropertyValue}; + + constructor(properties: Properties) { + this._properties = properties; + this._values = (Object.create(properties.defaultTransitioningPropertyValues) as any); + } + + possiblyEvaluate( + parameters: EvaluationParameters, + canonical?: CanonicalTileID, + availableImages?: Array + ): PossiblyEvaluated { + const result = new PossiblyEvaluated(this._properties); + for (const property of Object.keys(this._values)) { + result._values[property] = this._values[property].possiblyEvaluate(parameters, canonical, availableImages); + } + return result; + } + + hasTransition() { + for (const property of Object.keys(this._values)) { + if (this._values[property].prior) { + return true; + } + } + return false; + } +} + +// ------- Layout ------- + +/** + * Because layout properties are not transitionable, they have a simpler representation and evaluation chain than + * paint properties: `PropertyValue`s are possibly evaluated, producing possibly evaluated values, which are then + * fully evaluated. + * + * `Layout` stores a map of all (property name, `PropertyValue`) pairs for layout properties of a + * given layer type. It can calculate the possibly-evaluated values for all of them at once, producing a + * `PossiblyEvaluated` instance for the same set of properties. + */ +export class Layout { + _properties: Properties; + _values: {[K in keyof Props]: PropertyValue>}; + + constructor(properties: Properties) { + this._properties = properties; + this._values = (Object.create(properties.defaultPropertyValues) as any); + } + + hasValue(name: S) { + return this._values[name].value !== undefined; + } + + getValue(name: S) { + return clone(this._values[name].value); + } + + setValue(name: S, value: any) { + this._values[name] = new PropertyValue(this._values[name].property, value === null ? undefined : clone(value)) as any; + } + + serialize() { + const result: any = {}; + for (const property of Object.keys(this._values)) { + const value = this.getValue(property as keyof Props); + if (value !== undefined) { + result[property] = value; + } + } + return result; + } + + possiblyEvaluate( + parameters: EvaluationParameters, + canonical?: CanonicalTileID, + availableImages?: Array + ): PossiblyEvaluated { + const result = new PossiblyEvaluated(this._properties); + for (const property of Object.keys(this._values)) { + result._values[property] = this._values[property].possiblyEvaluate(parameters, canonical, availableImages); + } + return result; + } +} + +// ------- PossiblyEvaluated ------- + +/** + * "Possibly evaluated value" is an intermediate stage in the evaluation chain for both paint and layout property + * values. The purpose of this stage is to optimize away unnecessary recalculations for data-driven properties. Code + * which uses data-driven property values must assume that the value is dependent on feature data, and request that it + * be evaluated for each feature. But when that property value is in fact a constant or camera function, the calculation + * will not actually depend on the feature, and we can benefit from returning the prior result of having done the + * evaluation once, ahead of time, in an intermediate step whose inputs are just the value and "global" parameters + * such as current zoom level. + * + * `PossiblyEvaluatedValue` represents the three possible outcomes of this step: if the input value was a constant or + * camera expression, then the "possibly evaluated" result is a constant value. Otherwise, the input value was either + * a source or composite expression, and we must defer final evaluation until supplied a feature. We separate + * the source and composite cases because they are handled differently when generating GL attributes, buffers, and + * uniforms. + * + * Note that `PossiblyEvaluatedValue` (and `PossiblyEvaluatedPropertyValue`, below) are _not_ used for properties that + * do not allow data-driven values. For such properties, we know that the "possibly evaluated" result is always a constant + * scalar value. See below. + */ +type PossiblyEvaluatedValue = { + kind: 'constant'; + value: T; +} | SourceExpression | CompositeExpression; + +/** + * @internal + * `PossiblyEvaluatedPropertyValue` is used for data-driven paint and layout property values. It holds a + * `PossiblyEvaluatedValue` and the `GlobalProperties` that were used to generate it. You're not allowed to supply + * a different set of `GlobalProperties` when performing the final evaluation because they would be ignored in the + * case where the input value was a constant or camera function. + */ +export class PossiblyEvaluatedPropertyValue { + property: DataDrivenProperty; + value: PossiblyEvaluatedValue; + parameters: EvaluationParameters; + + constructor(property: DataDrivenProperty, value: PossiblyEvaluatedValue, parameters: EvaluationParameters) { + this.property = property; + this.value = value; + this.parameters = parameters; + } + + isConstant(): boolean { + return this.value.kind === 'constant'; + } + + constantOr(value: T): T { + if (this.value.kind === 'constant') { + return this.value.value; + } else { + return value; + } + } + + evaluate( + feature: Feature, + featureState: FeatureState, + canonical?: CanonicalTileID, + availableImages?: Array + ): T { + return this.property.evaluate(this.value, this.parameters, feature, featureState, canonical, availableImages); + } +} + +/** + * @internal + * `PossiblyEvaluated` stores a map of all (property name, `R`) pairs for paint or layout properties of a + * given layer type. + */ +export class PossiblyEvaluated { + _properties: Properties; + _values: PossibleEvaluatedProps; + + constructor(properties: Properties) { + this._properties = properties; + this._values = Object.create(properties.defaultPossiblyEvaluatedValues); + } + + get(name: S): PossibleEvaluatedProps[S] { + return this._values[name]; + } +} + +/** + * @internal + * An implementation of `Property` for properties that do not permit data-driven (source or composite) expressions. + * This restriction allows us to declare statically that the result of possibly evaluating this kind of property + * is in fact always the scalar type `T`, and can be used without further evaluating the value on a per-feature basis. + */ +export class DataConstantProperty implements Property { + specification: StylePropertySpecification; + + constructor(specification: StylePropertySpecification) { + this.specification = specification; + } + + possiblyEvaluate(value: PropertyValue, parameters: EvaluationParameters): T { + if (value.isDataDriven()) throw new Error('Value should not be data driven'); + return value.expression.evaluate(parameters); + } + + interpolate(a: T, b: T, t: number): T { + const interpolationType = this.specification.type as keyof typeof interpolates; + const interpolationFn = interpolates[interpolationType] as ((from: T, to: T, t: number) => T) | undefined; + if (interpolationFn) { + return interpolationFn(a, b, t); + } else { + return a; + } + } +} + +/** + * @internal + * An implementation of `Property` for properties that permit data-driven (source or composite) expressions. + * The result of possibly evaluating this kind of property is `PossiblyEvaluatedPropertyValue`; obtaining + * a scalar value `T` requires further evaluation on a per-feature basis. + */ +export class DataDrivenProperty implements Property> { + specification: StylePropertySpecification; + overrides: any; + + constructor(specification: StylePropertySpecification, overrides?: any) { + this.specification = specification; + this.overrides = overrides; + } + + possiblyEvaluate( + value: PropertyValue>, + parameters: EvaluationParameters, + canonical?: CanonicalTileID, + availableImages?: Array + ): PossiblyEvaluatedPropertyValue { + if (value.expression.kind === 'constant' || value.expression.kind === 'camera') { + return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: value.expression.evaluate(parameters, null, {}, canonical, availableImages)}, parameters); + } else { + return new PossiblyEvaluatedPropertyValue(this, value.expression, parameters); + } + } + + interpolate( + a: PossiblyEvaluatedPropertyValue, + b: PossiblyEvaluatedPropertyValue, + t: number + ): PossiblyEvaluatedPropertyValue { + // If either possibly-evaluated value is non-constant, give up: we aren't able to interpolate data-driven values. + if (a.value.kind !== 'constant' || b.value.kind !== 'constant') { + return a; + } + + // Special case hack solely for fill-outline-color. The undefined value is subsequently handled in + // FillStyleLayer#recalculate, which sets fill-outline-color to the fill-color value if the former + // is a PossiblyEvaluatedPropertyValue containing a constant undefined value. In addition to the + // return value here, the other source of a PossiblyEvaluatedPropertyValue containing a constant + // undefined value is the "default value" for fill-outline-color held in + // `Properties#defaultPossiblyEvaluatedValues`, which serves as the prototype of + // `PossiblyEvaluated#_values`. + if (a.value.value === undefined || b.value.value === undefined) { + return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: undefined}, a.parameters); + } + + const interpolationType = this.specification.type as keyof typeof interpolates; + const interpolationFn = interpolates[interpolationType] as ((from: T, to: T, t: number) => T) | undefined; + if (interpolationFn) { + const interpolatedValue = interpolationFn(a.value.value, b.value.value, t); + return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: interpolatedValue}, a.parameters); + } else { + return a; + } + } + + evaluate( + value: PossiblyEvaluatedValue, + parameters: EvaluationParameters, + feature: Feature, + featureState: FeatureState, + canonical?: CanonicalTileID, + availableImages?: Array + ): T { + if (value.kind === 'constant') { + return value.value; + } else { + return value.evaluate(parameters, feature, featureState, canonical, availableImages); + } + } +} + +/** + * @internal + * An implementation of `Property` for data driven `line-pattern` which are transitioned by cross-fading + * rather than interpolation. + */ + +export class CrossFadedDataDrivenProperty extends DataDrivenProperty> { + + possiblyEvaluate( + value: PropertyValue, PossiblyEvaluatedPropertyValue>>, + parameters: EvaluationParameters, + canonical?: CanonicalTileID, + availableImages?: Array + ): PossiblyEvaluatedPropertyValue> { + if (value.value === undefined) { + return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: undefined}, parameters); + } else if (value.expression.kind === 'constant') { + const evaluatedValue = value.expression.evaluate(parameters, null, {}, canonical, availableImages); + const isImageExpression = value.property.specification.type as any === 'resolvedImage'; + const constantValue = isImageExpression && typeof evaluatedValue !== 'string' ? evaluatedValue.name : evaluatedValue; + const constant = this._calculate(constantValue, constantValue, constantValue, parameters); + return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: constant}, parameters); + } else if (value.expression.kind === 'camera') { + const cameraVal = this._calculate( + value.expression.evaluate({zoom: parameters.zoom - 1.0}), + value.expression.evaluate({zoom: parameters.zoom}), + value.expression.evaluate({zoom: parameters.zoom + 1.0}), + parameters); + return new PossiblyEvaluatedPropertyValue(this, {kind: 'constant', value: cameraVal}, parameters); + } else { + // source or composite expression + return new PossiblyEvaluatedPropertyValue(this, value.expression, parameters); + } + } + + evaluate( + value: PossiblyEvaluatedValue>, + globals: EvaluationParameters, + feature: Feature, + featureState: FeatureState, + canonical?: CanonicalTileID, + availableImages?: Array + ): CrossFaded { + if (value.kind === 'source') { + const constant = value.evaluate(globals, feature, featureState, canonical, availableImages); + return this._calculate(constant, constant, constant, globals); + } else if (value.kind === 'composite') { + return this._calculate( + value.evaluate({zoom: Math.floor(globals.zoom) - 1.0}, feature, featureState), + value.evaluate({zoom: Math.floor(globals.zoom)}, feature, featureState), + value.evaluate({zoom: Math.floor(globals.zoom) + 1.0}, feature, featureState), + globals); + } else { + return value.value; + } + } + + _calculate(min: T, mid: T, max: T, parameters: EvaluationParameters): CrossFaded { + const z = parameters.zoom; + return z > parameters.zoomHistory.lastIntegerZoom ? {from: min, to: mid} : {from: max, to: mid}; + } + + interpolate(a: PossiblyEvaluatedPropertyValue>): PossiblyEvaluatedPropertyValue> { + return a; + } +} +/** + * @internal + * An implementation of `Property` for `*-pattern` and `line-dasharray`, which are transitioned by cross-fading + * rather than interpolation. + */ +export class CrossFadedProperty implements Property> { + specification: StylePropertySpecification; + + constructor(specification: StylePropertySpecification) { + this.specification = specification; + } + + possiblyEvaluate( + value: PropertyValue>, + parameters: EvaluationParameters, + canonical?: CanonicalTileID, + availableImages?: Array + ): CrossFaded { + if (value.value === undefined) { + return undefined; + } else if (value.expression.kind === 'constant') { + const constant = value.expression.evaluate(parameters, null, {}, canonical, availableImages); + return this._calculate(constant, constant, constant, parameters); + } else { + return this._calculate( + value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom - 1.0), parameters)), + value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom), parameters)), + value.expression.evaluate(new EvaluationParameters(Math.floor(parameters.zoom + 1.0), parameters)), + parameters); + } + } + + _calculate(min: T, mid: T, max: T, parameters: EvaluationParameters): CrossFaded { + const z = parameters.zoom; + return z > parameters.zoomHistory.lastIntegerZoom ? {from: min, to: mid} : {from: max, to: mid}; + } + + interpolate(a?: CrossFaded | null): CrossFaded { + return a; + } +} + +/** + * @internal + * An implementation of `Property` for `heatmap-color` and `line-gradient`. Interpolation is a no-op, and + * evaluation returns a boolean value in order to indicate its presence, but the real + * evaluation happens in StyleLayer classes. + */ + +export class ColorRampProperty implements Property { + specification: StylePropertySpecification; + + constructor(specification: StylePropertySpecification) { + this.specification = specification; + } + + possiblyEvaluate( + value: PropertyValue, + parameters: EvaluationParameters, + canonical?: CanonicalTileID, + availableImages?: Array + ): boolean { + return !!value.expression.evaluate(parameters, null, {}, canonical, availableImages); + } + + interpolate(): boolean { return false; } +} + +/** + * @internal + * `Properties` holds objects containing default values for the layout or paint property set of a given + * layer type. These objects are immutable, and they are used as the prototypes for the `_values` members of + * `Transitionable`, `Transitioning`, `Layout`, and `PossiblyEvaluated`. This allows these classes to avoid + * doing work in the common case where a property has no explicit value set and should be considered to take + * on the default value: using `for (const property of Object.keys(this._values))`, they can iterate over + * only the _own_ properties of `_values`, skipping repeated calculation of transitions and possible/final + * evaluations for defaults, the result of which will always be the same. + */ +export class Properties { + properties: Props; + defaultPropertyValues: {[K in keyof Props]: PropertyValue}; + defaultTransitionablePropertyValues: {[K in keyof Props]: TransitionablePropertyValue}; + defaultTransitioningPropertyValues: {[K in keyof Props]: TransitioningPropertyValue}; + defaultPossiblyEvaluatedValues: {[K in keyof Props]: PossiblyEvaluatedPropertyValue}; + overridableProperties: Array; + + constructor(properties: Props) { + this.properties = properties; + this.defaultPropertyValues = ({} as any); + this.defaultTransitionablePropertyValues = ({} as any); + this.defaultTransitioningPropertyValues = ({} as any); + this.defaultPossiblyEvaluatedValues = ({} as any); + this.overridableProperties = ([] as any); + + for (const property in properties) { + const prop = properties[property] as any; + if (prop.specification.overridable) { + this.overridableProperties.push(property); + } + const defaultPropertyValue = this.defaultPropertyValues[property] = + new PropertyValue(prop, undefined); + const defaultTransitionablePropertyValue = this.defaultTransitionablePropertyValues[property] = + new TransitionablePropertyValue(prop); + this.defaultTransitioningPropertyValues[property] = + defaultTransitionablePropertyValue.untransitioned(); + this.defaultPossiblyEvaluatedValues[property] = + defaultPropertyValue.possiblyEvaluate({} as any); + } + } +} + +register('DataDrivenProperty', DataDrivenProperty); +register('DataConstantProperty', DataConstantProperty); +register('CrossFadedDataDrivenProperty', CrossFadedDataDrivenProperty); +register('CrossFadedProperty', CrossFadedProperty); +register('ColorRampProperty', ColorRampProperty); diff --git a/web/libraries/maplibre-gl/src/style/query_utils.test.ts b/web/libraries/maplibre-gl/src/style/query_utils.test.ts new file mode 100644 index 00000000..718412e9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/query_utils.test.ts @@ -0,0 +1,107 @@ +import Point from '@mapbox/point-geometry'; + +import {offsetLine} from './query_utils'; + +const defaultPrecision = 10; + +const closeTo = (expected, precision = defaultPrecision) => ({ + asymmetricMatch: (actual) => Math.abs(expected - actual) < Math.pow(10, -precision) / 2 +}); + +describe('offsetLine', () => { + test('line two points east', () => { + const line = [ + new Point(0, 0), + new Point(1, 0) + ]; + const offset = 1; + + expect(offsetLine([line], offset)).toEqual([[ + new Point(0, 1), + new Point(1, 1) + ]]); + }); + + test('line two points west', () => { + const line = [ + new Point(10, 10), + new Point(5, 10) + ]; + const offset = 2; + + expect(offsetLine([line], offset)).toEqual([[ + new Point(10, 8), + new Point(5, 8) + ]]); + }); + + test('line two points south', () => { + const line = [ + new Point(0, -1), + new Point(0, 1) + ]; + const offset = 1; + + expect(offsetLine([line], offset)).toEqual([[ + new Point(-1, -1), + new Point(-1, 1) + ]]); + }); + + test('line three points north', () => { + const line = [ + new Point(0, 1), + new Point(0, 0), + new Point(0, -1) + ]; + const offset = 1; + + expect(offsetLine([line], offset)).toEqual([[ + new Point(1, 1), + new Point(1, 0), + new Point(1, -1) + ]]); + }); + + test('line three points north east', () => { + const line = [ + new Point(-1, 1), + new Point(0, 0), + new Point(1, -1) + ]; + const offset = Math.sqrt(2); + + expect(offsetLine([line], offset)).toEqual([[ + { + x: closeTo(0), + y: closeTo(2) + }, + { + x: closeTo(1), + y: closeTo(1) + }, + { + x: closeTo(2), + y: closeTo(0) + } + ]]); + }); + + test('ring five points square', () => { + const ring = [ + new Point(0, 0), + new Point(10, 0), + new Point(10, -10), + new Point(0, -10), + new Point(0, 0) + ]; + const offset = 2; + expect(offsetLine([ring], offset)).toEqual([[ + new Point(0, 2), + new Point(12, 2), + new Point(12, -12), + new Point(-2, -12), + new Point(-2, 0) + ]]); + }); +}); diff --git a/web/libraries/maplibre-gl/src/style/query_utils.ts b/web/libraries/maplibre-gl/src/style/query_utils.ts new file mode 100644 index 00000000..cbca4596 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/query_utils.ts @@ -0,0 +1,70 @@ +import Point from '@mapbox/point-geometry'; + +import type {PossiblyEvaluatedPropertyValue} from './properties'; +import type {StyleLayer} from '../style/style_layer'; +import type {CircleBucket} from '../data/bucket/circle_bucket'; +import type {LineBucket} from '../data/bucket/line_bucket'; + +export function getMaximumPaintValue( + property: string, + layer: StyleLayer, + bucket: CircleBucket | LineBucket +): number { + const value = ((layer.paint as any).get(property) as PossiblyEvaluatedPropertyValue).value; + if (value.kind === 'constant') { + return value.value; + } else { + return bucket.programConfigurations.get(layer.id).getMaxValue(property); + } +} + +export function translateDistance(translate: [number, number]) { + return Math.sqrt(translate[0] * translate[0] + translate[1] * translate[1]); +} + +export function translate(queryGeometry: Array, + translate: [number, number], + translateAnchor: 'viewport' | 'map', + bearing: number, + pixelsToTileUnits: number) { + if (!translate[0] && !translate[1]) { + return queryGeometry; + } + const pt = Point.convert(translate)._mult(pixelsToTileUnits); + + if (translateAnchor === 'viewport') { + pt._rotate(-bearing); + } + + const translated = []; + for (let i = 0; i < queryGeometry.length; i++) { + const point = queryGeometry[i]; + translated.push(point.sub(pt)); + } + return translated; +} + +export function offsetLine(rings: Array>, offset: number) { + const newRings: Array> = []; + for (let ringIndex = 0; ringIndex < rings.length; ringIndex++) { + const ring = rings[ringIndex]; + const newRing: Array = []; + for (let index = 0; index < ring.length; index++) { + const a = ring[index - 1]; + const b = ring[index]; + const c = ring[index + 1]; + const aToB = index === 0 ? new Point(0, 0) : b.sub(a)._unit()._perp(); + const bToC = index === ring.length - 1 ? new Point(0, 0) : c.sub(b)._unit()._perp(); + const extrude = aToB._add(bToC)._unit(); + + const cosHalfAngle = extrude.x * bToC.x + extrude.y * bToC.y; + if (cosHalfAngle !== 0) { + extrude._mult(1 / cosHalfAngle); + } + + newRing.push(extrude._mult(offset)._add(b)); + } + newRings.push(newRing); + } + return newRings; +} diff --git a/web/libraries/maplibre-gl/src/style/style.test.ts b/web/libraries/maplibre-gl/src/style/style.test.ts new file mode 100644 index 00000000..bdc325c8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style.test.ts @@ -0,0 +1,2623 @@ +import {Style} from './style'; +import {SourceCache} from '../source/source_cache'; +import {StyleLayer} from './style_layer'; +import {Transform} from '../geo/transform'; +import {extend} from '../util/util'; +import {RequestManager} from '../util/request_manager'; +import {Event, Evented} from '../util/evented'; +import {RGBAImage} from '../util/image'; +import { + setRTLTextPlugin, + clearRTLTextPlugin, + evented as rtlTextPluginEvented +} from '../source/rtl_text_plugin'; +import {browser} from '../util/browser'; +import {OverscaledTileID} from '../source/tile_id'; +import {fakeServer, type FakeServer} from 'nise'; + +import {EvaluationParameters} from './evaluation_parameters'; +import {LayerSpecification, GeoJSONSourceSpecification, FilterSpecification, SourceSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {SourceClass} from '../source/source'; +import {GeoJSONSource} from '../source/geojson_source'; + +function createStyleJSON(properties?) { + return extend({ + 'version': 8, + 'sources': {}, + 'layers': [] + }, properties); +} + +function createSource() { + return { + type: 'vector', + minzoom: 1, + maxzoom: 10, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.png'] + } as any as SourceSpecification; +} + +function createGeoJSONSource() { + return { + 'type': 'geojson', + 'data': { + 'type': 'FeatureCollection', + 'features': [] + } + }; +} + +class StubMap extends Evented { + style: Style; + transform: Transform; + private _requestManager: RequestManager; + + constructor() { + super(); + this.transform = new Transform(); + this._requestManager = new RequestManager(); + } + + _getMapId() { + return 1; + } + + getPixelRatio() { + return 1; + } + + setTerrain() { } + getTerrain() { } +} + +const getStubMap = () => new StubMap() as any; + +function createStyle(map = getStubMap()) { + const style = new Style(map); + map.style = style; + return style; +} + +let server: FakeServer; +let mockConsoleError; + +beforeEach(() => { + global.fetch = null; + server = fakeServer.create(); + mockConsoleError = jest.spyOn(console, 'error').mockImplementation(() => { }); +}); + +afterEach(() => { + server.restore(); + mockConsoleError.mockRestore(); +}); + +describe('Style', () => { + test('registers plugin state change listener', () => { + clearRTLTextPlugin(); + + jest.spyOn(Style, 'registerForPluginStateChange'); + const style = new Style(getStubMap()); + const mockStyleDispatcherBroadcast = jest.spyOn(style.dispatcher, 'broadcast'); + expect(Style.registerForPluginStateChange).toHaveBeenCalledTimes(1); + + setRTLTextPlugin('/plugin.js', undefined); + expect(mockStyleDispatcherBroadcast.mock.calls[0][0]).toBe('syncRTLPluginState'); + expect(mockStyleDispatcherBroadcast.mock.calls[0][1]).toEqual({ + pluginStatus: 'deferred', + pluginURL: 'http://localhost/plugin.js', + }); + }); + + test('loads plugin immediately if already registered', done => { + clearRTLTextPlugin(); + server.respondWith('/plugin.js', 'doesn\'t matter'); + setRTLTextPlugin('/plugin.js', (error) => { + expect(error).toMatch(/Cannot set the state of the rtl-text-plugin when not in the web-worker context/); + done(); + }); + server.respond(); + new Style(getStubMap()); + }); + + test('RTL plugin load reloads vector source but not raster source', done => { + const map = getStubMap(); + const style = new Style(map); + map.style = style; + style.loadJSON({ + 'version': 8, + 'sources': { + 'raster': { + type: 'raster', + tiles: ['http://tiles.server'] + }, + 'vector': { + type: 'vector', + tiles: ['http://tiles.server'] + } + }, + 'layers': [{ + 'id': 'raster', + 'type': 'raster', + 'source': 'raster' + }] + }); + + style.on('style.load', () => { + jest.spyOn(style.sourceCaches['raster'], 'reload'); + jest.spyOn(style.sourceCaches['vector'], 'reload'); + + clearRTLTextPlugin(); + server.respondWith('/plugin.js', 'doesn\'t matter'); + const _broadcast = style.dispatcher.broadcast; + style.dispatcher.broadcast = function (type, state, callback) { + if (type === 'syncRTLPluginState') { + // Mock a response from four workers saying they've loaded the plugin + callback(undefined, [true, true, true, true]); + } else { + _broadcast(type, state, callback); + } + }; + setRTLTextPlugin('/plugin.js', (error) => { + expect(error).toBeUndefined(); + setTimeout(() => { + clearRTLTextPlugin(); + expect(style.sourceCaches['raster'].reload).not.toHaveBeenCalled(); + expect(style.sourceCaches['vector'].reload).toHaveBeenCalled(); + done(); + }, 0); + }); + server.respond(); + }); + }); +}); + +describe('Style#loadURL', () => { + test('fires "dataloading"', () => { + const style = new Style(getStubMap()); + const spy = jest.fn(); + + style.on('dataloading', spy); + style.loadURL('style.json'); + + expect(spy).toHaveBeenCalledTimes(1); + expect(spy.mock.calls[0][0].target).toBe(style); + expect(spy.mock.calls[0][0].dataType).toBe('style'); + }); + + test('transforms style URL before request', () => { + const map = getStubMap(); + const spy = jest.spyOn(map._requestManager, 'transformRequest'); + + const style = new Style(map); + style.loadURL('style.json'); + + expect(spy).toHaveBeenCalledTimes(1); + expect(spy.mock.calls[0][0]).toBe('style.json'); + expect(spy.mock.calls[0][1]).toBe('Style'); + }); + + test('validates the style', done => { + const style = new Style(getStubMap()); + + style.on('error', ({error}) => { + expect(error).toBeTruthy(); + expect(error.message).toMatch(/version/); + done(); + }); + + style.loadURL('style.json'); + server.respondWith(JSON.stringify(createStyleJSON({version: 'invalid'}))); + server.respond(); + }); + + test('cancels pending requests if removed', () => { + const style = new Style(getStubMap()); + style.loadURL('style.json'); + style._remove(); + expect((server.lastRequest as any).aborted).toBe(true); + }); +}); + +describe('Style#loadJSON', () => { + test('serialize() returns undefined until style is loaded', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + expect(style.serialize()).toBeUndefined(); + style.on('style.load', () => { + expect(style.serialize()).toEqual(createStyleJSON()); + done(); + }); + }); + + test('fires "dataloading" (synchronously)', () => { + const style = new Style(getStubMap()); + const spy = jest.fn(); + + style.on('dataloading', spy); + style.loadJSON(createStyleJSON()); + + expect(spy).toHaveBeenCalledTimes(1); + expect(spy.mock.calls[0][0].target).toBe(style); + expect(spy.mock.calls[0][0].dataType).toBe('style'); + }); + + test('fires "data" (asynchronously)', done => { + const style = new Style(getStubMap()); + + style.loadJSON(createStyleJSON()); + + style.on('data', (e) => { + expect(e.target).toBe(style); + expect(e.dataType).toBe('style'); + done(); + }); + }); + + test('fires "data" when the sprite finishes loading', done => { + // Stubbing to bypass Web APIs that supported by jsdom: + // * `URL.createObjectURL` in ajax.getImage (https://github.com/tmpvar/jsdom/issues/1721) + // * `canvas.getContext('2d')` in browser.getImageData + jest.spyOn(browser, 'getImageData'); + // stub Image so we can invoke 'onload' + // https://github.com/jsdom/jsdom/commit/58a7028d0d5b6aacc5b435daee9fd8f9eacbb14c + + server.respondWith('GET', 'http://example.com/sprite.png', new ArrayBuffer(8)); + server.respondWith('GET', 'http://example.com/sprite.json', '{}'); + + const style = new Style(getStubMap()); + + style.loadJSON({ + 'version': 8, + 'sources': {}, + 'layers': [], + 'sprite': 'http://example.com/sprite' + }); + + style.once('error', (e) => expect(e).toBeFalsy()); + + style.once('data', (e) => { + expect(e.target).toBe(style); + expect(e.dataType).toBe('style'); + + style.once('data', (e) => { + expect(e.target).toBe(style); + expect(e.dataType).toBe('style'); + done(); + }); + + server.respond(); + }); + }); + + test('Validate sprite image extraction', done => { + // Stubbing to bypass Web APIs that supported by jsdom: + // * `URL.createObjectURL` in ajax.getImage (https://github.com/tmpvar/jsdom/issues/1721) + // * `canvas.getContext('2d')` in browser.getImageData + jest.spyOn(browser, 'getImageData'); + // stub Image so we can invoke 'onload' + // https://github.com/jsdom/jsdom/commit/58a7028d0d5b6aacc5b435daee9fd8f9eacbb14c + + server.respondWith('GET', 'http://example.com/sprite.png', new ArrayBuffer(8)); + server.respondWith('GET', 'http://example.com/sprite.json', '{"image1": {"width": 1, "height": 1, "x": 0, "y": 0, "pixelRatio": 1.0}}'); + + const style = new Style(getStubMap()); + + style.loadJSON({ + 'version': 8, + 'sources': {}, + 'layers': [], + 'sprite': 'http://example.com/sprite' + }); + + style.once('data', (e) => { + expect(e.target).toBe(style); + expect(e.dataType).toBe('style'); + + style.once('data', (e) => { + expect(e.target).toBe(style); + expect(e.dataType).toBe('style'); + style.imageManager.getImages(['image1'], (error, response) => { + const image = response['image1']; + expect(image.data).toBeInstanceOf(RGBAImage); + expect(image.data.width).toBe(1); + expect(image.data.height).toBe(1); + expect(image.pixelRatio).toBe(1); + done(); + }); + }); + + server.respond(); + }); + }); + + test('validates the style', done => { + const style = new Style(getStubMap()); + + style.on('error', ({error}) => { + expect(error).toBeTruthy(); + expect(error.message).toMatch(/version/); + done(); + }); + + style.loadJSON(createStyleJSON({version: 'invalid'})); + }); + + test('creates sources', done => { + const style = createStyle(); + + style.on('style.load', () => { + expect(style.sourceCaches['mapLibre'] instanceof SourceCache).toBeTruthy(); + done(); + }); + + style.loadJSON(extend(createStyleJSON(), { + 'sources': { + 'mapLibre': { + 'type': 'vector', + 'tiles': [] + } + } + })); + }); + + test('creates layers', done => { + const style = createStyle(); + + style.on('style.load', () => { + expect(style.getLayer('fill') instanceof StyleLayer).toBeTruthy(); + done(); + }); + + style.loadJSON({ + 'version': 8, + 'sources': { + 'foo': { + 'type': 'vector' + } + }, + 'layers': [{ + 'id': 'fill', + 'source': 'foo', + 'source-layer': 'source-layer', + 'type': 'fill' + }] + }); + }); + + test('transforms sprite json and image URLs before request', done => { + const map = getStubMap(); + const transformSpy = jest.spyOn(map._requestManager, 'transformRequest'); + const style = createStyle(map); + + style.on('style.load', () => { + expect(transformSpy).toHaveBeenCalledTimes(2); + expect(transformSpy.mock.calls[0][0]).toBe('http://example.com/sprites/bright-v8.json'); + expect(transformSpy.mock.calls[0][1]).toBe('SpriteJSON'); + expect(transformSpy.mock.calls[1][0]).toBe('http://example.com/sprites/bright-v8.png'); + expect(transformSpy.mock.calls[1][1]).toBe('SpriteImage'); + done(); + }); + + style.loadJSON(extend(createStyleJSON(), { + 'sprite': 'http://example.com/sprites/bright-v8' + })); + }); + + test('emits an error on non-existant vector source layer', done => { + const style = createStyle(); + style.loadJSON(createStyleJSON({ + sources: { + '-source-id-': {type: 'vector', tiles: []} + }, + layers: [] + })); + + style.on('style.load', () => { + style.removeSource('-source-id-'); + + const source = createSource(); + source['vector_layers'] = [{id: 'green'}]; + style.addSource('-source-id-', source); + style.addLayer({ + 'id': '-layer-id-', + 'type': 'circle', + 'source': '-source-id-', + 'source-layer': '-source-layer-' + }); + style.update({} as EvaluationParameters); + }); + + style.on('error', (event) => { + const err = event.error; + expect(err).toBeTruthy(); + expect(err.toString().indexOf('-source-layer-') !== -1).toBeTruthy(); + expect(err.toString().indexOf('-source-id-') !== -1).toBeTruthy(); + expect(err.toString().indexOf('-layer-id-') !== -1).toBeTruthy(); + + done(); + }); + }); + + test('sets up layer event forwarding', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + layers: [{ + id: 'background', + type: 'background' + }] + })); + + style.on('error', (e) => { + expect(e.layer).toEqual({id: 'background'}); + expect(e.mapLibre).toBeTruthy(); + done(); + }); + + style.on('style.load', () => { + style._layers.background.fire(new Event('error', {mapLibre: true})); + }); + }); + + test('sets terrain if defined', (done) => { + const map = getStubMap(); + const style = new Style(map); + map.setTerrain = jest.fn(); + style.loadJSON(createStyleJSON({ + sources: {'source-id': createGeoJSONSource()}, + terrain: {source: 'source-id', exaggeration: 0.33} + })); + + style.on('style.load', () => { + expect(style.map.setTerrain).toHaveBeenCalled(); + done(); + }); + }); + + test('applies transformStyle function', (done) => { + const previousStyle = createStyleJSON({ + sources: { + base: { + type: 'geojson', + data: {type: 'FeatureCollection', features: []} + } + }, + layers: [{ + id: 'layerId0', + type: 'circle', + source: 'base' + }, { + id: 'layerId1', + type: 'circle', + source: 'base' + }] + }); + + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON(), { + transformStyle: (prevStyle, nextStyle) => ({ + ...nextStyle, + sources: { + ...nextStyle.sources, + base: prevStyle.sources.base + }, + layers: [ + ...nextStyle.layers, + prevStyle.layers[0] + ] + }) + }, previousStyle); + + style.on('style.load', () => { + expect('base' in style.stylesheet.sources).toBeTruthy(); + expect(style.stylesheet.layers[0].id).toBe(previousStyle.layers[0].id); + expect(style.stylesheet.layers).toHaveLength(1); + done(); + }); + }); +}); + +describe('Style#_load', () => { + test('initiates sprite loading when it\'s present', () => { + const style = new Style(getStubMap()); + + const prevStyleSpec = createStyleJSON({ + sprite: 'https://example.com/test1' + }); + + const nextStyleSpec = createStyleJSON({ + sprite: 'https://example.com/test2' + }); + + const _loadSpriteSpyOn = jest.spyOn(style, '_loadSprite'); + style._load(nextStyleSpec, {}, prevStyleSpec); + + expect(_loadSpriteSpyOn).toHaveBeenCalledTimes(1); + }); + + test('does not initiate sprite loading when it\'s absent (undefined)', () => { + const style = new Style(getStubMap()); + + const prevStyleSpec = createStyleJSON({ + sprite: 'https://example.com/test1' + }); + + const nextStyleSpec = createStyleJSON({sprite: undefined}); + + const _loadSpriteSpyOn = jest.spyOn(style, '_loadSprite'); + style._load(nextStyleSpec, {}, prevStyleSpec); + + expect(_loadSpriteSpyOn).not.toHaveBeenCalled(); + }); + + test('layers are broadcasted to worker', () => { + const style = new Style(getStubMap()); + let dispatchType; + let dispatchData; + const styleSpec = createStyleJSON({ + layers: [{ + id: 'background', + type: 'background' + }] + }); + + const _broadcastSpyOn = jest.spyOn(style.dispatcher, 'broadcast') + .mockImplementation((type: string, data) => { + dispatchType = type; + dispatchData = data; + }); + + style._load(styleSpec, {}); + + expect(_broadcastSpyOn).toHaveBeenCalled(); + expect(dispatchType).toBe('setLayers'); + + expect(dispatchData).toHaveLength(1); + expect(dispatchData[0].id).toBe('background'); + + // cleanup + _broadcastSpyOn.mockReset(); + }); + + test('validate style when validate option is true', () => { + const style = new Style(getStubMap()); + const styleSpec = createStyleJSON({ + layers: [{ + id: 'background', + type: 'background' + }, { + id: 'custom', + type: 'custom' + }] + }); + const stub = jest.spyOn(console, 'error'); + + style._load(styleSpec, {validate: true}); + + // 1. layers[1]: missing required property "source" + // 2. layers[1].type: expected one of [fill, line, symbol, circle, heatmap, fill-extrusion, raster, hillshade, background], "custom" found + expect(stub).toHaveBeenCalledTimes(2); + + // cleanup + stub.mockReset(); + }); + + test('layers are NOT serialized immediately after creation', () => { + const style = new Style(getStubMap()); + const styleSpec = createStyleJSON({ + layers: [{ + id: 'background', + type: 'background' + }, { + id: 'custom', + type: 'custom' + }] + }); + + style._load(styleSpec, {validate: false}); + expect(style._serializedLayers).toBeNull(); + }); +}); + +describe('Style#_remove', () => { + test('removes cache sources and clears their tiles', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + sources: {'source-id': createGeoJSONSource()} + })); + + style.on('style.load', () => { + const sourceCache = style.sourceCaches['source-id']; + jest.spyOn(sourceCache, 'setEventedParent'); + jest.spyOn(sourceCache, 'onRemove'); + jest.spyOn(sourceCache, 'clearTiles'); + + style._remove(); + + expect(sourceCache.setEventedParent).toHaveBeenCalledWith(null); + expect(sourceCache.onRemove).toHaveBeenCalledWith(style.map); + expect(sourceCache.clearTiles).toHaveBeenCalled(); + + done(); + }); + }); + + test('deregisters plugin listener', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + const mockStyleDispatcherBroadcast = jest.spyOn(style.dispatcher, 'broadcast'); + + style.on('style.load', () => { + style._remove(); + + rtlTextPluginEvented.fire(new Event('pluginStateChange')); + expect(mockStyleDispatcherBroadcast).not.toHaveBeenCalledWith('syncRTLPluginState'); + done(); + }); + }); +}); + +describe('Style#update', () => { + test('on error', done => { + const style = createStyle(); + style.loadJSON({ + 'version': 8, + 'sources': { + 'source': { + 'type': 'vector' + } + }, + 'layers': [{ + 'id': 'second', + 'source': 'source', + 'source-layer': 'source-layer', + 'type': 'fill' + }] + }); + + style.on('error', (error) => { expect(error).toBeFalsy(); }); + + style.on('style.load', () => { + style.addLayer({id: 'first', source: 'source', type: 'fill', 'source-layer': 'source-layer'}, 'second'); + style.addLayer({id: 'third', source: 'source', type: 'fill', 'source-layer': 'source-layer'}); + style.removeLayer('second'); + + style.dispatcher.broadcast = function(key, value) { + expect(key).toBe('updateLayers'); + expect(value['layers'].map((layer) => { return layer.id; })).toEqual(['first', 'third']); + expect(value['removedIds']).toEqual(['second']); + done(); + }; + + style.update({} as EvaluationParameters); + }); + }); +}); + +describe('Style#setState', () => { + test('throw before loaded', () => { + const style = new Style(getStubMap()); + expect(() => style.setState(createStyleJSON())).toThrow(/load/i); + }); + + test('do nothing if there are no changes', done => { + const style = createStyle(); + style.loadJSON(createStyleJSON()); + jest.spyOn(style, 'addLayer').mockImplementation(() => done('test failed')); + jest.spyOn(style, 'removeLayer').mockImplementation(() => done('test failed')); + jest.spyOn(style, 'setPaintProperty').mockImplementation(() => done('test failed')); + jest.spyOn(style, 'setLayoutProperty').mockImplementation(() => done('test failed')); + jest.spyOn(style, 'setFilter').mockImplementation(() => done('test failed')); + jest.spyOn(style, 'addSource').mockImplementation(() => done('test failed')); + jest.spyOn(style, 'removeSource').mockImplementation(() => done('test failed')); + jest.spyOn(style, 'setGeoJSONSourceData').mockImplementation(() => done('test failed')); + jest.spyOn(style, 'setLayerZoomRange').mockImplementation(() => done('test failed')); + jest.spyOn(style, 'setLight').mockImplementation(() => done('test failed')); + style.on('style.load', () => { + const didChange = style.setState(createStyleJSON()); + expect(didChange).toBeFalsy(); + done(); + }); + }); + + test('Issue #3893: compare new source options against originally provided options rather than normalized properties', done => { + server.respondWith('/tilejson.json', JSON.stringify({ + tiles: ['http://tiles.server'] + })); + const initial = createStyleJSON(); + initial.sources.mySource = { + type: 'raster', + url: '/tilejson.json' + }; + const style = new Style(getStubMap()); + style.loadJSON(initial); + style.on('style.load', () => { + jest.spyOn(style, 'removeSource').mockImplementation(() => done('test failed: removeSource called')); + jest.spyOn(style, 'addSource').mockImplementation(() => done('test failed: addSource called')); + style.setState(initial); + done(); + }); + server.respond(); + }); + + test('return true if there is a change', done => { + const initialState = createStyleJSON(); + const nextState = createStyleJSON({ + sources: { + foo: { + type: 'geojson', + data: {type: 'FeatureCollection', features: []} + } + } + }); + + const style = new Style(getStubMap()); + style.loadJSON(initialState); + style.on('style.load', () => { + const didChange = style.setState(nextState); + expect(didChange).toBeTruthy(); + expect(style.stylesheet).toEqual(nextState); + done(); + }); + }); + + test('sets GeoJSON source data if different', done => { + const initialState = createStyleJSON({ + 'sources': {'source-id': createGeoJSONSource()} + }); + + const geoJSONSourceData = { + 'type': 'FeatureCollection', + 'features': [ + { + 'type': 'Feature', + 'geometry': { + 'type': 'Point', + 'coordinates': [125.6, 10.1] + } + } + ] + }; + + const nextState = createStyleJSON({ + 'sources': { + 'source-id': { + 'type': 'geojson', + 'data': geoJSONSourceData + } + } + }); + + const style = new Style(getStubMap()); + style.loadJSON(initialState); + + style.on('style.load', () => { + const geoJSONSource = style.sourceCaches['source-id'].getSource() as GeoJSONSource; + const mockStyleSetGeoJSONSourceDate = jest.spyOn(style, 'setGeoJSONSourceData'); + const mockGeoJSONSourceSetData = jest.spyOn(geoJSONSource, 'setData'); + const didChange = style.setState(nextState); + + expect(mockStyleSetGeoJSONSourceDate).toHaveBeenCalledWith('source-id', geoJSONSourceData); + expect(mockGeoJSONSourceSetData).toHaveBeenCalledWith(geoJSONSourceData); + expect(didChange).toBeTruthy(); + expect(style.stylesheet).toEqual(nextState); + done(); + }); + }); + + test('updates stylesheet according to applied transformStyle function', done => { + const initialState = createStyleJSON({ + sources: { + base: { + type: 'geojson', + data: {type: 'FeatureCollection', features: []} + } + }, + layers: [{ + id: 'layerId0', + type: 'circle', + source: 'base' + }, { + id: 'layerId1', + type: 'circle', + source: 'base' + }] + }); + + const nextState = createStyleJSON(); + const style = new Style(getStubMap()); + style.loadJSON(initialState); + + style.on('style.load', () => { + const didChange = style.setState(nextState, { + transformStyle: (prevStyle, nextStyle) => ({ + ...nextStyle, + sources: { + ...nextStyle.sources, + base: prevStyle.sources.base + }, + layers: [ + ...nextStyle.layers, + prevStyle.layers[0] + ] + }) + }); + + expect(didChange).toBeTruthy(); + expect('base' in style.stylesheet.sources).toBeTruthy(); + expect(style.stylesheet.layers[0].id).toBe(initialState.layers[0].id); + expect(style.stylesheet.layers).toHaveLength(1); + done(); + }); + }); +}); + +describe('Style#addSource', () => { + test('throw before loaded', () => { + const style = new Style(getStubMap()); + expect(() => style.addSource('source-id', createSource())).toThrow(/load/i); + }); + + test('throw if missing source type', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + + const source = createSource(); + delete source.type; + + style.on('style.load', () => { + expect(() => style.addSource('source-id', source)).toThrow(/type/i); + done(); + }); + }); + + test('fires "data" event', async () => { + const style = createStyle(); + style.loadJSON(createStyleJSON()); + const source = createSource(); + const dataPromise = style.once('data'); + style.on('style.load', () => { + style.addSource('source-id', source); + style.update({} as EvaluationParameters); + }); + await dataPromise; + }); + + test('throws on duplicates', done => { + const style = createStyle(); + style.loadJSON(createStyleJSON()); + const source = createSource(); + style.on('style.load', () => { + style.addSource('source-id', source); + expect(() => { + style.addSource('source-id', source); + }).toThrow(/Source "source-id" already exists./); + done(); + }); + }); + + test('sets up source event forwarding', done => { + let one = 0; + let two = 0; + let three = 0; + let four = 0; + const checkVisited = () => { + if (one === 1 && two === 1 && three === 1 && four === 1) { + done(); + } + }; + const style = createStyle(); + style.loadJSON(createStyleJSON({ + layers: [{ + id: 'background', + type: 'background' + }] + })); + const source = createSource(); + + style.on('style.load', () => { + style.on('error', () => { + one = 1; + checkVisited(); + }); + style.on('data', (e) => { + if (e.sourceDataType === 'metadata' && e.dataType === 'source') { + two = 1; + checkVisited(); + } else if (e.sourceDataType === 'content' && e.dataType === 'source') { + three = 1; + checkVisited(); + } else { + four = 1; + checkVisited(); + } + }); + + style.addSource('source-id', source); // fires data twice + style.sourceCaches['source-id'].fire(new Event('error')); + style.sourceCaches['source-id'].fire(new Event('data')); + }); + }); +}); + +describe('Style#removeSource', () => { + test('throw before loaded', () => { + const style = new Style(getStubMap()); + expect(() => style.removeSource('source-id')).toThrow(/load/i); + }); + + test('fires "data" event', async () => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + const source = createSource(); + const dataPromise = style.once('data'); + style.on('style.load', () => { + style.addSource('source-id', source); + style.removeSource('source-id'); + style.update({} as EvaluationParameters); + }); + await dataPromise; + }); + + test('clears tiles', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + sources: {'source-id': createGeoJSONSource()} + })); + + style.on('style.load', () => { + const sourceCache = style.sourceCaches['source-id']; + jest.spyOn(sourceCache, 'clearTiles'); + style.removeSource('source-id'); + expect(sourceCache.clearTiles).toHaveBeenCalledTimes(1); + done(); + }); + }); + + test('throws on non-existence', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + style.on('style.load', () => { + expect(() => { + style.removeSource('source-id'); + }).toThrow(/There is no source with this ID/); + done(); + }); + }); + + function createStyle(callback) { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + 'sources': { + 'mapLibre-source': createGeoJSONSource() + }, + 'layers': [{ + 'id': 'mapLibre-layer', + 'type': 'circle', + 'source': 'mapLibre-source', + 'source-layer': 'whatever' + }] + })); + style.on('style.load', () => { + style.update(1 as any as EvaluationParameters); + callback(style); + }); + return style; + } + + test('throws if source is in use', done => { + createStyle((style) => { + style.on('error', (event) => { + expect(event.error.message.includes('"mapLibre-source"')).toBeTruthy(); + expect(event.error.message.includes('"mapLibre-layer"')).toBeTruthy(); + done(); + }); + style.removeSource('mapLibre-source'); + }); + }); + + test('does not throw if source is not in use', done => { + createStyle((style) => { + style.on('error', () => { + done('test failed'); + }); + style.removeLayer('mapLibre-layer'); + style.removeSource('mapLibre-source'); + done(); + }); + }); + + test('tears down source event forwarding', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + const source = createSource(); + + style.on('style.load', () => { + style.addSource('source-id', source); + const sourceCache = style.sourceCaches['source-id']; + + style.removeSource('source-id'); + + // Suppress error reporting + sourceCache.on('error', () => {}); + + style.on('data', () => { expect(false).toBeTruthy(); }); + style.on('error', () => { expect(false).toBeTruthy(); }); + sourceCache.fire(new Event('data')); + sourceCache.fire(new Event('error')); + + done(); + }); + }); +}); + +describe('Style#addSprite', () => { + test('throw before loaded', () => { + const style = new Style(getStubMap()); + expect(() => style.addSprite('test', 'https://example.com/sprite')).toThrow(/load/i); + }); + + test('validates input and fires an error if there\'s already an existing sprite with the same id', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + style.on('style.load', () => { + style.on('error', (error) => { + expect(error.error.message).toMatch(/sprite: all the sprites' ids must be unique, but test is duplicated/); + done(); + }); + + style.addSprite('test', 'https://example.com/sprite'); + style.addSprite('test', 'https://example.com/sprite2'); + }); + }); + + test('adds a new sprite to the stylesheet when there\'s no sprite at all', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + style.on('style.load', () => { + style.addSprite('test', 'https://example.com/sprite'); + expect(style.stylesheet.sprite).toStrictEqual([{id: 'test', url: 'https://example.com/sprite'}]); + done(); + }); + }); + + test('adds a new sprite to the stylesheet when there\'s a stringy sprite existing', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({sprite: 'https://example.com/default'})); + style.on('style.load', () => { + style.addSprite('test', 'https://example.com/sprite'); + expect(style.stylesheet.sprite).toStrictEqual([ + {id: 'default', url: 'https://example.com/default'}, + {id: 'test', url: 'https://example.com/sprite'} + ]); + done(); + }); + }); + + test('adds a new sprite to the stylesheet when there\'s an array-sprite existing', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({sprite: [{id: 'default', url: 'https://example.com/default'}]})); + style.on('style.load', () => { + style.addSprite('test', 'https://example.com/sprite'); + expect(style.stylesheet.sprite).toStrictEqual([ + {id: 'default', url: 'https://example.com/default'}, + {id: 'test', url: 'https://example.com/sprite'} + ]); + done(); + }); + }); +}); + +describe('Style#removeSprite', () => { + test('throw before loaded', () => { + const style = new Style(getStubMap()); + expect(() => style.removeSprite('test')).toThrow(/load/i); + }); + + test('fires an error when trying to delete an non-existing sprite (sprite: undefined)', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + style.on('style.load', () => { + style.on('error', (error) => { + expect(error.error.message).toMatch(/Sprite \"test\" doesn't exists on this map./); + done(); + }); + + style.removeSprite('test'); + }); + }); + + test('fires an error when trying to delete an non-existing sprite (sprite: single url)', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({sprite: 'https://example.com/sprite'})); + style.on('style.load', () => { + style.on('error', (error) => { + expect(error.error.message).toMatch(/Sprite \"test\" doesn't exists on this map./); + done(); + }); + + style.removeSprite('test'); + }); + }); + + test('fires an error when trying to delete an non-existing sprite (sprite: array)', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({sprite: [{id: 'default', url: 'https://example.com/sprite'}]})); + style.on('style.load', () => { + style.on('error', (error) => { + expect(error.error.message).toMatch(/Sprite \"test\" doesn't exists on this map./); + done(); + }); + + style.removeSprite('test'); + }); + }); + + test('removes the sprite when it\'s a single URL', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({sprite: 'https://example.com/test'})); + style.on('style.load', () => { + style.removeSprite('default'); + expect(style.stylesheet.sprite).toBeUndefined(); + done(); + }); + }); + + test('removes the sprite when it\'s an array', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON([{id: 'default', url: 'https://example.com/sprite'}])); + style.on('style.load', () => { + style.removeSprite('default'); + expect(style.stylesheet.sprite).toBeUndefined(); + done(); + }); + }); +}); + +describe('Style#setGeoJSONSourceData', () => { + const geoJSON = {type: 'FeatureCollection', features: []} as GeoJSON.GeoJSON; + + test('throws before loaded', () => { + const style = new Style(getStubMap()); + expect(() => style.setGeoJSONSourceData('source-id', geoJSON)).toThrow(/load/i); + }); + + test('throws on non-existence', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + style.on('style.load', () => { + expect(() => style.setGeoJSONSourceData('source-id', geoJSON)).toThrow(/There is no source with this ID/); + done(); + }); + }); +}); + +describe('Style#addLayer', () => { + test('throw before loaded', () => { + const style = new Style(getStubMap()); + expect(() => style.addLayer({id: 'background', type: 'background'})).toThrow(/load/i); + }); + + test('sets up layer event forwarding', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + + style.on('error', (e) => { + expect(e.layer).toEqual({id: 'background'}); + expect(e.mapLibre).toBeTruthy(); + done(); + }); + + style.on('style.load', () => { + style.addLayer({ + id: 'background', + type: 'background' + }); + style._layers.background.fire(new Event('error', {mapLibre: true})); + }); + }); + + test('throws on non-existant vector source layer', done => { + const style = createStyle(); + style.loadJSON(createStyleJSON({ + sources: { + // At least one source must be added to trigger the load event + dummy: {type: 'vector', tiles: []} + } + })); + + style.on('style.load', () => { + const source = createSource(); + source['vector_layers'] = [{id: 'green'}]; + style.addSource('-source-id-', source); + style.addLayer({ + 'id': '-layer-id-', + 'type': 'circle', + 'source': '-source-id-', + 'source-layer': '-source-layer-' + }); + }); + + style.on('error', (event) => { + const err = event.error; + + expect(err).toBeTruthy(); + expect(err.toString().indexOf('-source-layer-') !== -1).toBeTruthy(); + expect(err.toString().indexOf('-source-id-') !== -1).toBeTruthy(); + expect(err.toString().indexOf('-layer-id-') !== -1).toBeTruthy(); + + done(); + }); + }); + + test('emits error on invalid layer', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + style.on('style.load', () => { + style.on('error', () => { + expect(style.getLayer('background')).toBeFalsy(); + done(); + }); + style.addLayer({ + id: 'background', + type: 'background', + paint: { + 'background-opacity': 5 + } + }); + }); + }); + + test('#4040 does not mutate source property when provided inline', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + style.on('style.load', () => { + const source = { + 'type': 'geojson', + 'data': { + 'type': 'Point', + 'coordinates': [0, 0] + } + }; + const layer = {id: 'inline-source-layer', type: 'circle', source} as any as LayerSpecification; + style.addLayer(layer); + expect((layer as any).source).toEqual(source); + done(); + }); + }); + + test('reloads source', done => { + const style = createStyle(); + style.loadJSON(extend(createStyleJSON(), { + 'sources': { + 'mapLibre': { + 'type': 'vector', + 'tiles': [] + } + } + })); + const layer = { + 'id': 'symbol', + 'type': 'symbol', + 'source': 'mapLibre', + 'source-layer': 'libremap', + 'filter': ['==', 'id', 0] + } as LayerSpecification; + + style.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'content') { + style.sourceCaches['mapLibre'].reload = function() { done(); }; + style.addLayer(layer); + style.update({} as EvaluationParameters); + } + }); + }); + + test('#3895 reloads source (instead of clearing) if adding this layer with the same type, immediately after removing it', done => { + const style = createStyle(); + style.loadJSON(extend(createStyleJSON(), { + 'sources': { + 'mapLibre': { + 'type': 'vector', + 'tiles': [] + } + }, + layers: [{ + 'id': 'my-layer', + 'type': 'symbol', + 'source': 'mapLibre', + 'source-layer': 'libremap', + 'filter': ['==', 'id', 0] + }] + })); + + const layer = { + 'id': 'my-layer', + 'type': 'symbol', + 'source': 'mapLibre', + 'source-layer': 'libremap' + }as LayerSpecification; + + style.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'content') { + style.sourceCaches['mapLibre'].reload = function() { done(); }; + style.sourceCaches['mapLibre'].clearTiles = function() { done('test failed'); }; + style.removeLayer('my-layer'); + style.addLayer(layer); + style.update({} as EvaluationParameters); + } + }); + + }); + + test('clears source (instead of reloading) if adding this layer with a different type, immediately after removing it', done => { + const style = createStyle(); + style.loadJSON(extend(createStyleJSON(), { + 'sources': { + 'mapLibre': { + 'type': 'vector', + 'tiles': [] + } + }, + layers: [{ + 'id': 'my-layer', + 'type': 'symbol', + 'source': 'mapLibre', + 'source-layer': 'libremap', + 'filter': ['==', 'id', 0] + }] + })); + + const layer = { + 'id': 'my-layer', + 'type': 'circle', + 'source': 'mapLibre', + 'source-layer': 'libremap' + }as LayerSpecification; + style.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'content') { + style.sourceCaches['mapLibre'].reload = function() { done('test failed'); }; + style.sourceCaches['mapLibre'].clearTiles = function() { done(); }; + style.removeLayer('my-layer'); + style.addLayer(layer); + style.update({} as EvaluationParameters); + } + }); + + }); + + test('fires "data" event', async () => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + const layer = {id: 'background', type: 'background'} as LayerSpecification; + + const dataPromise = style.once('data'); + + style.on('style.load', () => { + style.addLayer(layer); + style.update({} as EvaluationParameters); + }); + await dataPromise; + }); + + test('emits error on duplicates', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + const layer = {id: 'background', type: 'background'} as LayerSpecification; + + style.on('error', (e) => { + expect(e.error.message).toMatch(/already exists/); + done(); + }); + + style.on('style.load', () => { + style.addLayer(layer); + style.addLayer(layer); + }); + }); + + test('adds to the end by default', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + layers: [{ + id: 'a', + type: 'background' + }, { + id: 'b', + type: 'background' + }] + })); + const layer = {id: 'c', type: 'background'} as LayerSpecification; + + style.on('style.load', () => { + style.addLayer(layer); + expect(style._order).toEqual(['a', 'b', 'c']); + done(); + }); + }); + + test('adds before the given layer', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + layers: [{ + id: 'a', + type: 'background' + }, { + id: 'b', + type: 'background' + }] + })); + const layer = {id: 'c', type: 'background'} as LayerSpecification; + + style.on('style.load', () => { + style.addLayer(layer, 'a'); + expect(style._order).toEqual(['c', 'a', 'b']); + done(); + }); + }); + + test('fire error if before layer does not exist', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + layers: [{ + id: 'a', + type: 'background' + }, { + id: 'b', + type: 'background' + }] + })); + const layer = {id: 'c', type: 'background'} as LayerSpecification; + + style.on('style.load', () => { + style.on('error', (error) => { + expect(error.error.message).toMatch(/Cannot add layer "c" before non-existing layer "z"./); + done(); + }); + style.addLayer(layer, 'z'); + }); + }); + + test('fires an error on non-existant source layer', done => { + const style = new Style(getStubMap()); + style.loadJSON(extend(createStyleJSON(), { + sources: { + dummy: { + type: 'geojson', + data: {type: 'FeatureCollection', features: []} + } + } + })); + + const layer = { + id: 'dummy', + type: 'fill', + source: 'dummy', + 'source-layer': 'dummy' + }as LayerSpecification; + + style.on('style.load', () => { + style.on('error', ({error}) => { + expect(error.message).toMatch(/does not exist on source/); + done(); + }); + style.addLayer(layer); + }); + + }); +}); + +describe('Style#removeLayer', () => { + test('throw before loaded', () => { + const style = new Style(getStubMap()); + expect(() => style.removeLayer('background')).toThrow(/load/i); + }); + + test('fires "data" event', async () => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + const layer = {id: 'background', type: 'background'} as LayerSpecification; + + const dataPromise = style.once('data'); + + style.on('style.load', () => { + style.addLayer(layer); + style.removeLayer('background'); + style.update({} as EvaluationParameters); + }); + + await dataPromise; + }); + + test('tears down layer event forwarding', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + layers: [{ + id: 'background', + type: 'background' + }] + })); + + style.on('error', () => { + done('test failed'); + }); + + style.on('style.load', () => { + const layer = style._layers.background; + style.removeLayer('background'); + + // Bind a listener to prevent fallback Evented error reporting. + layer.on('error', () => {}); + + layer.fire(new Event('error', {mapLibre: true})); + done(); + }); + }); + + test('fires an error on non-existence', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + + style.on('style.load', () => { + style.on('error', ({error}) => { + expect(error.message).toMatch(/Cannot remove non-existing layer "background"./); + done(); + }); + style.removeLayer('background'); + }); + }); + + test('removes from the order', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + layers: [{ + id: 'a', + type: 'background' + }, { + id: 'b', + type: 'background' + }] + })); + + style.on('style.load', () => { + style.removeLayer('a'); + expect(style._order).toEqual(['b']); + done(); + }); + }); + + test('does not remove dereffed layers', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + layers: [{ + id: 'a', + type: 'background' + }, { + id: 'b', + ref: 'a' + }] + })); + + style.on('style.load', () => { + style.removeLayer('a'); + expect(style.getLayer('a')).toBeUndefined(); + expect(style.getLayer('b')).toBeDefined(); + done(); + }); + }); +}); + +describe('Style#moveLayer', () => { + test('throw before loaded', () => { + const style = new Style(getStubMap()); + expect(() => style.moveLayer('background')).toThrow(/load/i); + }); + + test('fires "data" event', async () => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + const layer = {id: 'background', type: 'background'} as LayerSpecification; + + const dataPromise = style.once('data'); + style.on('style.load', () => { + style.addLayer(layer); + style.moveLayer('background'); + style.update({} as EvaluationParameters); + }); + await dataPromise; + }); + + test('fires an error on non-existence', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + + style.on('style.load', () => { + style.on('error', ({error}) => { + expect(error.message).toMatch(/does not exist in the map\'s style and cannot be moved/); + done(); + }); + style.moveLayer('background'); + }); + }); + + test('changes the order', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + layers: [ + {id: 'a', type: 'background'}, + {id: 'b', type: 'background'}, + {id: 'c', type: 'background'} + ] + })); + + style.on('style.load', () => { + style.moveLayer('a', 'c'); + expect(style._order).toEqual(['b', 'a', 'c']); + done(); + }); + }); + + test('moves to existing location', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + layers: [ + {id: 'a', type: 'background'}, + {id: 'b', type: 'background'}, + {id: 'c', type: 'background'} + ] + })); + + style.on('style.load', () => { + style.moveLayer('b', 'b'); + expect(style._order).toEqual(['a', 'b', 'c']); + done(); + }); + }); +}); + +describe('Style#setPaintProperty', () => { + test('#4738 postpones source reload until layers have been broadcast to workers', done => { + const style = new Style(getStubMap()); + style.loadJSON(extend(createStyleJSON(), { + 'sources': { + 'geojson': { + 'type': 'geojson', + 'data': {'type': 'FeatureCollection', 'features': []} + } + }, + 'layers': [ + { + 'id': 'circle', + 'type': 'circle', + 'source': 'geojson' + } + ] + })); + + const tr = new Transform(); + tr.resize(512, 512); + + style.once('style.load', () => { + style.update(tr.zoom as any as EvaluationParameters); + const sourceCache = style.sourceCaches['geojson']; + const source = style.getSource('geojson'); + + let begun = false; + let styleUpdateCalled = false; + + (source as any).on('data', (e) => setTimeout(() => { + if (!begun) { + begun = true; + jest.spyOn(sourceCache, 'reload').mockImplementation(() => { + expect(styleUpdateCalled).toBeTruthy(); + done(); + }); + + (source as any).setData({'type': 'FeatureCollection', 'features': []}); + style.setPaintProperty('circle', 'circle-color', {type: 'identity', property: 'foo'}); + } + + if (begun && e.sourceDataType === 'content') { + // setData() worker-side work is complete; simulate an + // animation frame a few ms later, so that this test can + // confirm that SourceCache#reload() isn't called until + // after the next Style#update() + setTimeout(() => { + styleUpdateCalled = true; + style.update({} as EvaluationParameters); + }, 50); + } + })); + }); + }); + + test('#5802 clones the input', done => { + const style = new Style(getStubMap()); + style.loadJSON({ + 'version': 8, + 'sources': {}, + 'layers': [ + { + 'id': 'background', + 'type': 'background' + } + ] + }); + + style.on('style.load', () => { + const value = {stops: [[0, 'red'], [10, 'blue']]}; + style.setPaintProperty('background', 'background-color', value); + expect(style.getPaintProperty('background', 'background-color')).not.toBe(value); + expect(style._changed).toBeTruthy(); + + style.update({} as EvaluationParameters); + expect(style._changed).toBeFalsy(); + + value.stops[0][0] = 1; + style.setPaintProperty('background', 'background-color', value); + expect(style._changed).toBeTruthy(); + + done(); + }); + }); + + test('respects validate option', done => { + const style = new Style(getStubMap()); + style.loadJSON({ + 'version': 8, + 'sources': {}, + 'layers': [ + { + 'id': 'background', + 'type': 'background' + } + ] + }); + + style.on('style.load', () => { + const backgroundLayer = style.getLayer('background'); + const validate = jest.spyOn(backgroundLayer, '_validate'); + + style.setPaintProperty('background', 'background-color', 'notacolor', {validate: false}); + expect(validate.mock.calls[0][4]).toEqual({validate: false}); + expect(mockConsoleError).not.toHaveBeenCalled(); + + expect(style._changed).toBeTruthy(); + style.update({} as EvaluationParameters); + + style.setPaintProperty('background', 'background-color', 'alsonotacolor'); + expect(mockConsoleError).toHaveBeenCalledTimes(1); + expect(validate.mock.calls[1][4]).toEqual({}); + + done(); + }); + }); +}); + +describe('Style#getPaintProperty', () => { + test('#5802 clones the output', done => { + const style = new Style(getStubMap()); + style.loadJSON({ + 'version': 8, + 'sources': {}, + 'layers': [ + { + 'id': 'background', + 'type': 'background' + } + ] + }); + + style.on('style.load', () => { + style.setPaintProperty('background', 'background-color', {stops: [[0, 'red'], [10, 'blue']]}); + style.update({} as EvaluationParameters); + expect(style._changed).toBeFalsy(); + + const value = style.getPaintProperty('background', 'background-color'); + value['stops'][0][0] = 1; + style.setPaintProperty('background', 'background-color', value); + expect(style._changed).toBeTruthy(); + + done(); + }); + }); +}); + +describe('Style#setLayoutProperty', () => { + test('#5802 clones the input', done => { + const style = new Style(getStubMap()); + style.loadJSON({ + 'version': 8, + 'sources': { + 'geojson': { + 'type': 'geojson', + 'data': { + 'type': 'FeatureCollection', + 'features': [] + } + } + }, + 'layers': [ + { + 'id': 'line', + 'type': 'line', + 'source': 'geojson' + } + ] + }); + + style.on('style.load', () => { + const value = {stops: [[0, 'butt'], [10, 'round']]}; + style.setLayoutProperty('line', 'line-cap', value); + expect(style.getLayoutProperty('line', 'line-cap')).not.toBe(value); + expect(style._changed).toBeTruthy(); + + style.update({} as EvaluationParameters); + expect(style._changed).toBeFalsy(); + + value.stops[0][0] = 1; + style.setLayoutProperty('line', 'line-cap', value); + expect(style._changed).toBeTruthy(); + + done(); + }); + }); + + test('respects validate option', done => { + const style = new Style(getStubMap()); + style.loadJSON({ + 'version': 8, + 'sources': { + 'geojson': { + 'type': 'geojson', + 'data': { + 'type': 'FeatureCollection', + 'features': [] + } + } + }, + 'layers': [ + { + 'id': 'line', + 'type': 'line', + 'source': 'geojson' + } + ] + }); + + style.on('style.load', () => { + const lineLayer = style.getLayer('line'); + const validate = jest.spyOn(lineLayer, '_validate'); + + style.setLayoutProperty('line', 'line-cap', 'invalidcap', {validate: false}); + expect(validate.mock.calls[0][4]).toEqual({validate: false}); + expect(mockConsoleError).not.toHaveBeenCalled(); + expect(style._changed).toBeTruthy(); + style.update({} as EvaluationParameters); + + style.setLayoutProperty('line', 'line-cap', 'differentinvalidcap'); + expect(mockConsoleError).toHaveBeenCalledTimes(1); + expect(validate.mock.calls[1][4]).toEqual({}); + + done(); + }); + }); +}); + +describe('Style#getLayoutProperty', () => { + test('#5802 clones the output', done => { + const style = new Style(getStubMap()); + style.loadJSON({ + 'version': 8, + 'sources': { + 'geojson': { + 'type': 'geojson', + 'data': { + 'type': 'FeatureCollection', + 'features': [] + } + } + }, + 'layers': [ + { + 'id': 'line', + 'type': 'line', + 'source': 'geojson' + } + ] + }); + + style.on('style.load', () => { + style.setLayoutProperty('line', 'line-cap', {stops: [[0, 'butt'], [10, 'round']]}); + style.update({} as EvaluationParameters); + expect(style._changed).toBeFalsy(); + + const value = style.getLayoutProperty('line', 'line-cap'); + value.stops[0][0] = 1; + style.setLayoutProperty('line', 'line-cap', value); + expect(style._changed).toBeTruthy(); + + done(); + }); + }); +}); + +describe('Style#setFilter', () => { + test('throws if style is not loaded', () => { + const style = new Style(getStubMap()); + expect(() => style.setFilter('symbol', ['==', 'id', 1])).toThrow(/load/i); + }); + + function createStyle() { + const style = new Style(getStubMap()); + style.loadJSON({ + version: 8, + sources: { + geojson: createGeoJSONSource() as GeoJSONSourceSpecification + }, + layers: [ + {id: 'symbol', type: 'symbol', source: 'geojson', filter: ['==', 'id', 0]} + ] + }); + return style; + } + + test('sets filter', done => { + const style = createStyle(); + + style.on('style.load', () => { + style.dispatcher.broadcast = function(key, value) { + expect(key).toBe('updateLayers'); + expect(value['layers'][0].id).toBe('symbol'); + expect(value['layers'][0].filter).toEqual(['==', 'id', 1]); + done(); + }; + + style.setFilter('symbol', ['==', 'id', 1]); + expect(style.getFilter('symbol')).toEqual(['==', 'id', 1]); + style.update({} as EvaluationParameters); // trigger dispatcher broadcast + }); + }); + + test('gets a clone of the filter', done => { + const style = createStyle(); + + style.on('style.load', () => { + const filter1 = ['==', 'id', 1] as FilterSpecification; + style.setFilter('symbol', filter1); + const filter2 = style.getFilter('symbol'); + const filter3 = style.getLayer('symbol').filter; + + expect(filter1).not.toBe(filter2); + expect(filter1).not.toBe(filter3); + expect(filter2).not.toBe(filter3); + + done(); + }); + }); + + test('sets again mutated filter', done => { + const style = createStyle(); + + style.on('style.load', () => { + const filter = ['==', 'id', 1] as FilterSpecification; + style.setFilter('symbol', filter); + style.update({} as EvaluationParameters); // flush pending operations + + style.dispatcher.broadcast = function(key, value) { + expect(key).toBe('updateLayers'); + expect(value['layers'][0].id).toBe('symbol'); + expect(value['layers'][0].filter).toEqual(['==', 'id', 2]); + done(); + }; + filter[2] = 2; + style.setFilter('symbol', filter); + style.update({} as EvaluationParameters); // trigger dispatcher broadcast + }); + }); + + test('unsets filter', done => { + const style = createStyle(); + style.on('style.load', () => { + style.setFilter('symbol', null); + expect(style.getLayer('symbol').serialize()['filter']).toBeUndefined(); + done(); + }); + }); + + test('emits if invalid', done => { + const style = createStyle(); + style.on('style.load', () => { + style.on('error', () => { + expect(style.getLayer('symbol').serialize()['filter']).toEqual(['==', 'id', 0]); + done(); + }); + style.setFilter('symbol', ['==', '$type', 1]); + }); + }); + + test('fires an error if layer not found', done => { + const style = createStyle(); + + style.on('style.load', () => { + style.on('error', ({error}) => { + expect(error.message).toMatch(/Cannot filter non-existing layer "non-existant"./); + done(); + }); + style.setFilter('non-existant', ['==', 'id', 1]); + }); + }); + + test('validates filter by default', done => { + const style = createStyle(); + style.on('style.load', () => { + style.setFilter('symbol', 'notafilter' as any as FilterSpecification); + expect(style.getFilter('symbol')).toEqual(['==', 'id', 0]); + expect(mockConsoleError).toHaveBeenCalledTimes(1); + style.update({} as EvaluationParameters); // trigger dispatcher broadcast + done(); + }); + }); + + test('respects validate option', done => { + const style = createStyle(); + + style.on('style.load', () => { + style.dispatcher.broadcast = function(key, value) { + expect(key).toBe('updateLayers'); + expect(value['layers'][0].id).toBe('symbol'); + expect(value['layers'][0].filter).toBe('notafilter'); + done(); + }; + + style.setFilter('symbol', 'notafilter' as any as FilterSpecification, {validate: false}); + expect(style.getFilter('symbol')).toBe('notafilter'); + style.update({} as EvaluationParameters); // trigger dispatcher broadcast + }); + }); +}); + +describe('Style#setLayerZoomRange', () => { + test('throw before loaded', () => { + const style = new Style(getStubMap()); + expect(() => style.setLayerZoomRange('symbol', 5, 12)).toThrow(/load/i); + }); + + function createStyle() { + const style = new Style(getStubMap()); + style.loadJSON({ + 'version': 8, + 'sources': { + 'geojson': createGeoJSONSource() as GeoJSONSourceSpecification + }, + 'layers': [{ + 'id': 'symbol', + 'type': 'symbol', + 'source': 'geojson' + }] + }); + return style; + } + + test('sets zoom range', done => { + const style = createStyle(); + + style.on('style.load', () => { + style.dispatcher.broadcast = function(key, value) { + expect(key).toBe('updateLayers'); + expect(value['layers'].map((layer) => { return layer.id; })).toEqual(['symbol']); + done(); + }; + style.setLayerZoomRange('symbol', 5, 12); + expect(style.getLayer('symbol').minzoom).toBe(5); + expect(style.getLayer('symbol').maxzoom).toBe(12); + style.update({} as EvaluationParameters); // trigger dispatcher broadcast + }); + }); + + test('fires an error if layer not found', done => { + const style = createStyle(); + style.on('style.load', () => { + style.on('error', ({error}) => { + expect(error.message).toMatch(/Cannot set the zoom range of non-existing layer "non-existant"./); + done(); + }); + style.setLayerZoomRange('non-existant', 5, 12); + }); + }); + + test('does not reload raster source', done => { + const style = new Style(getStubMap()); + style.loadJSON({ + 'version': 8, + 'sources': { + 'raster': { + type: 'raster', + tiles: ['http://tiles.server'] + } + }, + 'layers': [{ + 'id': 'raster', + 'type': 'raster', + 'source': 'raster' + }] + }); + + style.on('style.load', () => { + jest.spyOn(style, '_reloadSource'); + + style.setLayerZoomRange('raster', 5, 12); + style.update(0 as any as EvaluationParameters); + expect(style._reloadSource).not.toHaveBeenCalled(); + done(); + }); + }); +}); + +describe('Style#getLayersOrder', () => { + test('returns ids of layers in the correct order', done => { + const style = new Style(getStubMap()); + style.loadJSON({ + 'version': 8, + 'sources': { + 'raster': { + type: 'raster', + tiles: ['http://tiles.server'] + } + }, + 'layers': [{ + 'id': 'raster', + 'type': 'raster', + 'source': 'raster' + }] + }); + + style.on('style.load', () => { + style.addLayer({ + id: 'custom', + type: 'custom', + render() {} + }, 'raster'); + expect(style.getLayersOrder()).toEqual(['custom', 'raster']); + done(); + }); + }); +}); + +describe('Style#queryRenderedFeatures', () => { + + let style; + let transform; + + beforeEach((callback) => { + style = new Style(getStubMap()); + transform = new Transform(); + transform.resize(512, 512); + function queryMapLibreFeatures(layers, serializedLayers, getFeatureState, queryGeom, cameraQueryGeom, scale, params) { + const features = { + 'land': [{ + type: 'Feature', + layer: style._layers.land.serialize(), + geometry: { + type: 'Polygon' + } + }, { + type: 'Feature', + layer: style._layers.land.serialize(), + geometry: { + type: 'Point' + } + }], + 'landref': [{ + type: 'Feature', + layer: style._layers.landref.serialize(), + geometry: { + type: 'Line' + } + }] + }; + + // format result to shape of tile.queryRenderedFeatures result + for (const layer in features) { + features[layer] = features[layer].map((feature, featureIndex) => + ({feature, featureIndex})); + } + + if (params.layers) { + for (const l in features) { + if (params.layers.indexOf(l) < 0) { + delete features[l]; + } + } + } + + return features; + } + + style.loadJSON({ + 'version': 8, + 'sources': { + 'mapLibre': { + 'type': 'geojson', + 'data': {type: 'FeatureCollection', features: []} + }, + 'other': { + 'type': 'geojson', + 'data': {type: 'FeatureCollection', features: []} + } + }, + 'layers': [{ + 'id': 'land', + 'type': 'line', + 'source': 'mapLibre', + 'source-layer': 'water', + 'layout': { + 'line-cap': 'round' + }, + 'paint': { + 'line-color': 'red' + }, + 'metadata': { + 'something': 'else' + } + }, { + 'id': 'landref', + 'ref': 'land', + 'paint': { + 'line-color': 'blue' + } + } as any as LayerSpecification, { + 'id': 'land--other', + 'type': 'line', + 'source': 'other', + 'source-layer': 'water', + 'layout': { + 'line-cap': 'round' + }, + 'paint': { + 'line-color': 'red' + }, + 'metadata': { + 'something': 'else' + } + }] + }); + + style.on('style.load', () => { + style.sourceCaches.mapLibre.tilesIn = () => { + return [{ + tile: {queryRenderedFeatures: queryMapLibreFeatures}, + tileID: new OverscaledTileID(0, 0, 0, 0, 0), + queryGeometry: [], + scale: 1 + }]; + }; + style.sourceCaches.other.tilesIn = () => { + return []; + }; + + style.sourceCaches.mapLibre.transform = transform; + style.sourceCaches.other.transform = transform; + + style.update(0 as any as EvaluationParameters); + style._updateSources(transform); + callback(); + }); + }); + + afterEach(() => { + style = undefined; + transform = undefined; + }); + + test('returns feature type', () => { + const results = style.queryRenderedFeatures([{x: 0, y: 0}], {}, transform); + expect(results[0].geometry.type).toBe('Line'); + }); + + test('filters by `layers` option', () => { + const results = style.queryRenderedFeatures([{x: 0, y: 0}], {layers: ['land']}, transform); + expect(results).toHaveLength(2); + }); + + test('checks type of `layers` option', () => { + let errors = 0; + jest.spyOn(style, 'fire').mockImplementation((event) => { + if (event['error'] && event['error'].message.includes('parameters.layers must be an Array.')) { + errors++; + } + }); + style.queryRenderedFeatures([{x: 0, y: 0}], {layers: 'string'}, transform); + expect(errors).toBe(1); + }); + + test('includes layout properties', () => { + const results = style.queryRenderedFeatures([{x: 0, y: 0}], {}, transform); + const layout = results[0].layer.layout; + expect(layout['line-cap']).toBe('round'); + }); + + test('includes paint properties', () => { + const results = style.queryRenderedFeatures([{x: 0, y: 0}], {}, transform); + expect(results[2].layer.paint['line-color']).toBe('red'); + }); + + test('includes metadata', () => { + const results = style.queryRenderedFeatures([{x: 0, y: 0}], {}, transform); + + const layer = results[1].layer; + expect(layer.metadata.something).toBe('else'); + + }); + + test('include multiple layers', () => { + const results = style.queryRenderedFeatures([{x: 0, y: 0}], {layers: ['land', 'landref']}, transform); + expect(results).toHaveLength(3); + }); + + test('does not query sources not implicated by `layers` parameter', () => { + style.sourceCaches.mapLibre.queryRenderedFeatures = function() { expect(true).toBe(false); }; + style.queryRenderedFeatures([{x: 0, y: 0}], {layers: ['land--other']}, transform); + }); + + test('fires an error if layer included in params does not exist on the style', () => { + let errors = 0; + jest.spyOn(style, 'fire').mockImplementation((event) => { + if (event['error'] && event['error'].message.includes('does not exist in the map\'s style and cannot be queried for features.')) errors++; + }); + const results = style.queryRenderedFeatures([{x: 0, y: 0}], {layers: ['merp']}, transform); + expect(errors).toBe(1); + expect(results).toHaveLength(0); + }); +}); + +describe('Style defers ...', () => { + test('... expensive methods', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON({ + 'sources': { + 'streets': createGeoJSONSource(), + 'terrain': createGeoJSONSource() + } + })); + + style.on('style.load', () => { + style.update({} as EvaluationParameters); + + // spies to track deferred methods + const mockStyleFire = jest.spyOn(style, 'fire'); + jest.spyOn(style, '_reloadSource'); + jest.spyOn(style, '_updateWorkerLayers'); + + style.addLayer({id: 'first', type: 'symbol', source: 'streets'}); + style.addLayer({id: 'second', type: 'symbol', source: 'streets'}); + style.addLayer({id: 'third', type: 'symbol', source: 'terrain'}); + + style.setPaintProperty('first', 'text-color', 'black'); + style.setPaintProperty('first', 'text-halo-color', 'white'); + + expect(style.fire).not.toHaveBeenCalled(); + expect(style._reloadSource).not.toHaveBeenCalled(); + expect(style._updateWorkerLayers).not.toHaveBeenCalled(); + + style.update({} as EvaluationParameters); + + expect(mockStyleFire.mock.calls[0][0]['type']).toBe('data'); + + // called per source + expect(style._reloadSource).toHaveBeenCalledTimes(2); + expect(style._reloadSource).toHaveBeenCalledWith('streets'); + expect(style._reloadSource).toHaveBeenCalledWith('terrain'); + + // called once + expect(style._updateWorkerLayers).toHaveBeenCalledTimes(1); + + done(); + }); + }); +}); + +describe('Style#query*Features', () => { + + // These tests only cover filter validation. Most tests for these methods + // live in the integration tests. + + let style; + let onError; + let transform; + + beforeEach((callback) => { + transform = new Transform(); + transform.resize(100, 100); + style = new Style(getStubMap()); + style.loadJSON({ + 'version': 8, + 'sources': { + 'geojson': createGeoJSONSource() + }, + 'layers': [{ + 'id': 'symbol', + 'type': 'symbol', + 'source': 'geojson' + }] + }); + + onError = jest.fn(); + + style.on('error', onError) + .on('style.load', () => { + callback(); + }); + }); + + test('querySourceFeatures emits an error on incorrect filter', () => { + expect(style.querySourceFeatures([10, 100], {filter: 7}, transform)).toEqual([]); + expect(onError.mock.calls[0][0].error.message).toMatch(/querySourceFeatures\.filter/); + }); + + test('queryRenderedFeatures emits an error on incorrect filter', () => { + expect(style.queryRenderedFeatures([{x: 0, y: 0}], {filter: 7}, transform)).toEqual([]); + expect(onError.mock.calls[0][0].error.message).toMatch(/queryRenderedFeatures\.filter/); + }); + + test('querySourceFeatures not raise validation errors if validation was disabled', () => { + let errors = 0; + jest.spyOn(style, 'fire').mockImplementation((event) => { + if (event['error']) { + errors++; + } + }); + style.queryRenderedFeatures([{x: 0, y: 0}], {filter: 'invalidFilter', validate: false}, transform); + expect(errors).toBe(0); + }); + + test('querySourceFeatures not raise validation errors if validation was disabled', () => { + let errors = 0; + jest.spyOn(style, 'fire').mockImplementation((event) => { + if (event['error']) errors++; + }); + style.querySourceFeatures([{x: 0, y: 0}], {filter: 'invalidFilter', validate: false}, transform); + expect(errors).toBe(0); + }); + + test('serialized layers should be correctly updated after adding/removing layers', () => { + + let serializedStyle = style.serialize(); + expect(serializedStyle.layers).toHaveLength(1); + expect(serializedStyle.layers[0].id).toBe('symbol'); + + const layer = { + id: 'background', + type: 'background' + } as LayerSpecification; + style.addLayer(layer); + + // serialize again + serializedStyle = style.serialize(); + expect(serializedStyle.layers).toHaveLength(2); + expect(serializedStyle.layers[1].id).toBe('background'); + + // remove and serialize + style.removeLayer('background'); + serializedStyle = style.serialize(); + expect(serializedStyle.layers).toHaveLength(1); + + }); +}); + +describe('Style#addSourceType', () => { + test('adds factory function', done => { + const style = new Style(getStubMap()); + const sourceType = function () {} as any as SourceClass; + + // expect no call to load worker source + style.dispatcher.broadcast = function (type) { + if (type === 'loadWorkerSource') { + done('test failed'); + } + }; + + style.addSourceType('foo', sourceType, (arg1, arg2) => { + expect(arg1).toBeNull(); + expect(arg2).toBeNull(); + done(); + }); + }); + + test('triggers workers to load worker source code', done => { + const style = new Style(getStubMap()); + const sourceType = function () {} as any as SourceClass; + sourceType.workerSourceURL = 'worker-source.js' as any as URL; + + style.dispatcher.broadcast = (type, params) => { + if (type === 'loadWorkerSource') { + expect(params['name']).toBe('bar'); + expect(params['url']).toBe('worker-source.js'); + done(); + } + }; + + style.addSourceType('bar', sourceType, (err) => { expect(err).toBeFalsy(); }); + }); + + test('refuses to add new type over existing name', done => { + const style = new Style(getStubMap()); + const sourceType = function () {} as any as SourceClass; + style.addSourceType('canvas', sourceType, (err) => { + expect(err).toBeTruthy(); + done(); + }); + }); +}); + +describe('Style#hasTransitions', () => { + test('returns false when the style is loading', () => { + const style = new Style(getStubMap()); + expect(style.hasTransitions()).toBe(false); + }); + + test('returns true when a property is transitioning', done => { + const style = new Style(getStubMap()); + style.loadJSON({ + 'version': 8, + 'sources': {}, + 'layers': [{ + 'id': 'background', + 'type': 'background' + }] + }); + + style.on('style.load', () => { + style.setPaintProperty('background', 'background-color', 'blue'); + style.update({transition: {duration: 300, delay: 0}} as EvaluationParameters); + expect(style.hasTransitions()).toBe(true); + done(); + }); + }); + + test('returns false when a property is not transitioning', done => { + const style = new Style(getStubMap()); + style.loadJSON({ + 'version': 8, + 'sources': {}, + 'layers': [{ + 'id': 'background', + 'type': 'background' + }] + }); + + style.on('style.load', () => { + style.setPaintProperty('background', 'background-color', 'blue'); + style.update({transition: {duration: 0, delay: 0}} as EvaluationParameters); + expect(style.hasTransitions()).toBe(false); + done(); + }); + }); +}); + +describe('Style#serialize', () => { + test('include terrain property when map has 3D terrain', done => { + const styleJson = createStyleJSON({terrain: { + source: 'terrainSource', + exaggeration: 1 + }}); + const style = new Style(getStubMap()); + style.loadJSON(styleJson); + + style.on('style.load', () => { + expect(style.serialize().terrain).toBe(styleJson.terrian); + done(); + }); + }); + + test('do not include terrain property when map does not have 3D terrain', done => { + const style = new Style(getStubMap()); + style.loadJSON(createStyleJSON()); + + style.on('style.load', () => { + expect(style.serialize().terrain).toBeUndefined(); + done(); + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/style/style.ts b/web/libraries/maplibre-gl/src/style/style.ts new file mode 100644 index 00000000..a43d0ad4 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style.ts @@ -0,0 +1,1745 @@ +import {Event, ErrorEvent, Evented} from '../util/evented'; +import {StyleLayer} from './style_layer'; +import {createStyleLayer} from './create_style_layer'; +import {loadSprite} from './load_sprite'; +import {ImageManager} from '../render/image_manager'; +import {GlyphManager} from '../render/glyph_manager'; +import {Light} from './light'; +import {LineAtlas} from '../render/line_atlas'; +import {pick, clone, extend, deepEqual, filterObject, mapObject} from '../util/util'; +import {coerceSpriteToArray} from '../util/style'; +import {getJSON, getReferrer, makeRequest} from '../util/ajax'; +import {ResourceType} from '../util/request_manager'; +import {browser} from '../util/browser'; +import {Dispatcher} from '../util/dispatcher'; +import {validateStyle, emitValidationErrors as _emitValidationErrors} from './validate_style'; +import {getSourceType, setSourceType, Source} from '../source/source'; +import type {SourceClass} from '../source/source'; +import {QueryRenderedFeaturesOptions, QuerySourceFeatureOptions, queryRenderedFeatures, queryRenderedSymbols, querySourceFeatures} from '../source/query_features'; +import {SourceCache} from '../source/source_cache'; +import {GeoJSONSource} from '../source/geojson_source'; +import {latest as styleSpec, derefLayers as deref, emptyStyle, diff as diffStyles, operations as diffOperations} from '@maplibre/maplibre-gl-style-spec'; +import {getGlobalWorkerPool} from '../util/global_worker_pool'; +import { + registerForPluginStateChange, + evented as rtlTextPluginEvented, + triggerPluginCompletionEvent +} from '../source/rtl_text_plugin'; +import {PauseablePlacement} from './pauseable_placement'; +import {ZoomHistory} from './zoom_history'; +import {CrossTileSymbolIndex} from '../symbol/cross_tile_symbol_index'; +import {validateCustomStyleLayer} from './style_layer/custom_style_layer'; +import type {MapGeoJSONFeature} from '../util/vectortile_to_geojson'; + +// We're skipping validation errors with the `source.canvas` identifier in order +// to continue to allow canvas sources to be added at runtime/updated in +// smart setStyle (see https://github.com/mapbox/mapbox-gl-js/pull/6424): +const emitValidationErrors = (evented: Evented, errors?: ReadonlyArray<{ + message: string; + identifier?: string; +}> | null) => + _emitValidationErrors(evented, errors && errors.filter(error => error.identifier !== 'source.canvas')); + +import type {Map} from '../ui/map'; +import type {Transform} from '../geo/transform'; +import type {StyleImage} from './style_image'; +import type {StyleGlyph} from './style_glyph'; +import type {Callback} from '../types/callback'; +import type {EvaluationParameters} from './evaluation_parameters'; +import type {Placement} from '../symbol/placement'; +import type {Cancelable} from '../types/cancelable'; +import type {RequestParameters, ResponseCallback} from '../util/ajax'; +import type { + LayerSpecification, + FilterSpecification, + StyleSpecification, + LightSpecification, + SourceSpecification, + SpriteSpecification, +} from '@maplibre/maplibre-gl-style-spec'; +import type {CustomLayerInterface} from './style_layer/custom_style_layer'; +import type {Validator} from './validate_style'; +import type {OverscaledTileID} from '../source/tile_id'; + +const supportedDiffOperations = pick(diffOperations, [ + 'addLayer', + 'removeLayer', + 'setPaintProperty', + 'setLayoutProperty', + 'setFilter', + 'addSource', + 'removeSource', + 'setLayerZoomRange', + 'setLight', + 'setTransition', + 'setGeoJSONSourceData', + 'setGlyphs', + 'setSprite', +]); + +const ignoredDiffOperations = pick(diffOperations, [ + 'setCenter', + 'setZoom', + 'setBearing', + 'setPitch' +]); + +const empty = emptyStyle() as StyleSpecification; +/** + * A feature identifier that is bound to a source + */ +export type FeatureIdentifier = { + /** + * Unique id of the feature. + */ + id?: string | number | undefined; + /** + * The id of the vector or GeoJSON source for the feature. + */ + source: string; + /** + * *For vector tile sources, `sourceLayer` is required.* + */ + sourceLayer?: string | undefined; +}; + +/** + * The options object related to the {@link Map}'s style related methods + */ +export type StyleOptions = { + /** + * If false, style validation will be skipped. Useful in production environment. + */ + validate?: boolean; + /** + * Defines a CSS + * font-family for locally overriding generation of glyphs in the 'CJK Unified Ideographs', 'Hiragana', 'Katakana' and 'Hangul Syllables' ranges. + * In these ranges, font settings from the map's style will be ignored, except for font-weight keywords (light/regular/medium/bold). + * Set to `false`, to enable font settings from the map's style for these glyph ranges. + * Forces a full update. + */ + localIdeographFontFamily?: string; +}; + +/** + * Supporting type to add validation to another style related type + */ +export type StyleSetterOptions = { + /** + * Whether to check if the filter conforms to the MapLibre Style Specification. Disabling validation is a performance optimization that should only be used if you have previously validated the values you will be passing to this function. + */ + validate?: boolean; +}; + +/** + * Part of {@link Map#setStyle} options, transformStyle is a convenience function that allows to modify a style after it is fetched but before it is committed to the map state + * this function exposes previous and next styles, it can be commonly used to support a range of functionalities like: + * when previous style carries certain 'state' that needs to be carried over to a new style gracefully + * when a desired style is a certain combination of previous and incoming style + * when an incoming style requires modification based on external state + * + * @param previousStyle - The current style. + * @param nextStyle - The next style. + * @returns resulting style that will to be applied to the map + * + * @example + * ```ts + * map.setStyle('https://demotiles.maplibre.org/style.json', { + * transformStyle: (previousStyle, nextStyle) => ({ + * ...nextStyle, + * sources: { + * ...nextStyle.sources, + * // copy a source from previous style + * 'osm': previousStyle.sources.osm + * }, + * layers: [ + * // background layer + * nextStyle.layers[0], + * // copy a layer from previous style + * previousStyle.layers[0], + * // other layers from the next style + * ...nextStyle.layers.slice(1).map(layer => { + * // hide the layers we don't need from demotiles style + * if (layer.id.startsWith('geolines')) { + * layer.layout = {...layer.layout || {}, visibility: 'none'}; + * // filter out US polygons + * } else if (layer.id.startsWith('coastline') || layer.id.startsWith('countries')) { + * layer.filter = ['!=', ['get', 'ADM0_A3'], 'USA']; + * } + * return layer; + * }) + * ] + * }) + * }); + * ``` + */ +export type TransformStyleFunction = (previous: StyleSpecification | undefined, next: StyleSpecification) => StyleSpecification; + +/** + * The options object related to the {@link Map}'s style related methods + */ +export type StyleSwapOptions = { + /** + * If false, force a 'full' update, removing the current style + * and building the given one instead of attempting a diff-based update. + */ + diff?: boolean; + /** + * TransformStyleFunction is a convenience function + * that allows to modify a style after it is fetched but before it is committed to the map state. Refer to {@link TransformStyleFunction}. + */ + transformStyle?: TransformStyleFunction; +} + +/** + * Specifies a layer to be added to a {@link Style}. In addition to a standard {@link LayerSpecification} + * or a {@link CustomLayerInterface}, a {@link LayerSpecification} with an embedded {@link SourceSpecification} can also be provided. + */ +export type AddLayerObject = LayerSpecification | (Omit & {source: SourceSpecification}) | CustomLayerInterface; + +/** + * The Style base class + */ +export class Style extends Evented { + map: Map; + stylesheet: StyleSpecification; + dispatcher: Dispatcher; + imageManager: ImageManager; + glyphManager: GlyphManager; + lineAtlas: LineAtlas; + light: Light; + + _request: Cancelable; + _spriteRequest: Cancelable; + _layers: {[_: string]: StyleLayer}; + _serializedLayers: {[_: string]: LayerSpecification}; + _order: Array; + sourceCaches: {[_: string]: SourceCache}; + zoomHistory: ZoomHistory; + _loaded: boolean; + _rtlTextPluginCallback: (a: any) => any; + _changed: boolean; + _updatedSources: {[_: string]: 'clear' | 'reload'}; + _updatedLayers: {[_: string]: true}; + _removedLayers: {[_: string]: StyleLayer}; + _changedImages: {[_: string]: true}; + _glyphsDidChange: boolean; + _updatedPaintProps: {[layer: string]: true}; + _layerOrderChanged: boolean; + // image ids of images loaded from style's sprite + _spritesImagesIds: {[spriteId: string]: string[]}; + // image ids of all images loaded (sprite + user) + _availableImages: Array; + + crossTileSymbolIndex: CrossTileSymbolIndex; + pauseablePlacement: PauseablePlacement; + placement: Placement; + z: number; + + static registerForPluginStateChange: typeof registerForPluginStateChange; + + constructor(map: Map, options: StyleOptions = {}) { + super(); + + this.map = map; + this.dispatcher = new Dispatcher(getGlobalWorkerPool(), this, map._getMapId()); + this.imageManager = new ImageManager(); + this.imageManager.setEventedParent(this); + this.glyphManager = new GlyphManager(map._requestManager, options.localIdeographFontFamily); + this.lineAtlas = new LineAtlas(256, 512); + this.crossTileSymbolIndex = new CrossTileSymbolIndex(); + + this._spritesImagesIds = {}; + this._layers = {}; + + this._order = []; + this.sourceCaches = {}; + this.zoomHistory = new ZoomHistory(); + this._loaded = false; + this._availableImages = []; + + this._resetUpdates(); + + this.dispatcher.broadcast('setReferrer', getReferrer()); + + const self = this; + this._rtlTextPluginCallback = Style.registerForPluginStateChange((event) => { + const state = { + pluginStatus: event.pluginStatus, + pluginURL: event.pluginURL + }; + self.dispatcher.broadcast('syncRTLPluginState', state, (err, results) => { + triggerPluginCompletionEvent(err); + if (results) { + const allComplete = results.every((elem) => elem); + if (allComplete) { + for (const id in self.sourceCaches) { + const sourceType = self.sourceCaches[id].getSource().type; + if (sourceType === 'vector' || sourceType === 'geojson') { + // Non-vector sources don't have any symbols buckets to reload when the RTL text plugin loads + // They also load more quickly, so they're more likely to have already displaying tiles + // that would be unnecessarily booted by the plugin load event + self.sourceCaches[id].reload(); // Should be a no-op if the plugin loads before any tiles load + } + } + } + } + + }); + }); + + this.on('data', (event) => { + if (event.dataType !== 'source' || event.sourceDataType !== 'metadata') { + return; + } + + const sourceCache = this.sourceCaches[event.sourceId]; + if (!sourceCache) { + return; + } + + const source = sourceCache.getSource(); + if (!source || !source.vectorLayerIds) { + return; + } + + for (const layerId in this._layers) { + const layer = this._layers[layerId]; + if (layer.source === source.id) { + this._validateLayer(layer); + } + } + }); + } + + loadURL(url: string, options: StyleSwapOptions & StyleSetterOptions = {}, previousStyle?: StyleSpecification) { + this.fire(new Event('dataloading', {dataType: 'style'})); + + options.validate = typeof options.validate === 'boolean' ? + options.validate : true; + + const request = this.map._requestManager.transformRequest(url, ResourceType.Style); + this._request = getJSON(request, (error?: Error | null, json?: any | null) => { + this._request = null; + if (error) { + this.fire(new ErrorEvent(error)); + } else if (json) { + this._load(json, options, previousStyle); + } + }); + } + + loadJSON(json: StyleSpecification, options: StyleSetterOptions & StyleSwapOptions = {}, previousStyle?: StyleSpecification) { + this.fire(new Event('dataloading', {dataType: 'style'})); + + this._request = browser.frame(() => { + this._request = null; + options.validate = options.validate !== false; + this._load(json, options, previousStyle); + }); + } + + loadEmpty() { + this.fire(new Event('dataloading', {dataType: 'style'})); + this._load(empty, {validate: false}); + } + + _load(json: StyleSpecification, options: StyleSwapOptions & StyleSetterOptions, previousStyle?: StyleSpecification) { + const nextState = options.transformStyle ? options.transformStyle(previousStyle, json) : json; + if (options.validate && emitValidationErrors(this, validateStyle(nextState))) { + return; + } + + this._loaded = true; + this.stylesheet = nextState; + + for (const id in nextState.sources) { + this.addSource(id, nextState.sources[id], {validate: false}); + } + + if (nextState.sprite) { + this._loadSprite(nextState.sprite); + } else { + this.imageManager.setLoaded(true); + } + + this.glyphManager.setURL(nextState.glyphs); + this._createLayers(); + + this.light = new Light(this.stylesheet.light); + + this.map.setTerrain(this.stylesheet.terrain ?? null); + + this.fire(new Event('data', {dataType: 'style'})); + this.fire(new Event('style.load')); + } + + private _createLayers() { + const dereferencedLayers = deref(this.stylesheet.layers); + + // Broadcast layers to workers first, so that expensive style processing (createStyleLayer) + // can happen in parallel on both main and worker threads. + this.dispatcher.broadcast('setLayers', dereferencedLayers); + + this._order = dereferencedLayers.map((layer) => layer.id); + this._layers = {}; + + // reset serialization field, to be populated only when needed + this._serializedLayers = null; + for (const layer of dereferencedLayers) { + const styledLayer = createStyleLayer(layer); + styledLayer.setEventedParent(this, {layer: {id: layer.id}}); + this._layers[layer.id] = styledLayer; + } + } + + _loadSprite(sprite: SpriteSpecification, isUpdate: boolean = false, completion: (err: Error) => void = undefined) { + this.imageManager.setLoaded(false); + + this._spriteRequest = loadSprite(sprite, this.map._requestManager, this.map.getPixelRatio(), (err, images) => { + this._spriteRequest = null; + if (err) { + this.fire(new ErrorEvent(err)); + } else if (images) { + for (const spriteId in images) { + this._spritesImagesIds[spriteId] = []; + + // remove old sprite's loaded images (for the same sprite id) that are not in new sprite + const imagesToRemove = this._spritesImagesIds[spriteId] ? this._spritesImagesIds[spriteId].filter(id => !(id in images)) : []; + for (const id of imagesToRemove) { + this.imageManager.removeImage(id); + this._changedImages[id] = true; + } + + for (const id in images[spriteId]) { + // don't prefix images of the "default" sprite + const imageId = spriteId === 'default' ? id : `${spriteId}:${id}`; + // save all the sprite's images' ids to be able to delete them in `removeSprite` + this._spritesImagesIds[spriteId].push(imageId); + if (imageId in this.imageManager.images) { + this.imageManager.updateImage(imageId, images[spriteId][id], false); + } else { + this.imageManager.addImage(imageId, images[spriteId][id]); + } + + if (isUpdate) { + this._changedImages[imageId] = true; + } + } + } + } + + this.imageManager.setLoaded(true); + this._availableImages = this.imageManager.listImages(); + + if (isUpdate) { + this._changed = true; + } + + this.dispatcher.broadcast('setImages', this._availableImages); + this.fire(new Event('data', {dataType: 'style'})); + + if (completion) { + completion(err); + } + }); + } + + _unloadSprite() { + for (const id of Object.values(this._spritesImagesIds).flat()) { + this.imageManager.removeImage(id); + this._changedImages[id] = true; + } + + this._spritesImagesIds = {}; + this._availableImages = this.imageManager.listImages(); + this._changed = true; + this.dispatcher.broadcast('setImages', this._availableImages); + this.fire(new Event('data', {dataType: 'style'})); + } + + _validateLayer(layer: StyleLayer) { + const sourceCache = this.sourceCaches[layer.source]; + if (!sourceCache) { + return; + } + + const sourceLayer = layer.sourceLayer; + if (!sourceLayer) { + return; + } + + const source = sourceCache.getSource(); + if (source.type === 'geojson' || (source.vectorLayerIds && source.vectorLayerIds.indexOf(sourceLayer) === -1)) { + this.fire(new ErrorEvent(new Error( + `Source layer "${sourceLayer}" ` + + `does not exist on source "${source.id}" ` + + `as specified by style layer "${layer.id}".` + ))); + } + } + + loaded() { + if (!this._loaded) + return false; + + if (Object.keys(this._updatedSources).length) + return false; + + for (const id in this.sourceCaches) + if (!this.sourceCaches[id].loaded()) + return false; + + if (!this.imageManager.isLoaded()) + return false; + + return true; + } + + /** + * take an array of string IDs, and based on this._layers, generate an array of LayerSpecification + * @param ids - an array of string IDs, for which serialized layers will be generated. If omitted, all serialized layers will be returned + * @returns generated result + */ + private _serializeByIds(ids?: Array): Array { + + const serializedLayersDictionary = this._serializedAllLayers(); + if (!ids || ids.length === 0) { + return Object.values(serializedLayersDictionary); + } + + const serializedLayers = []; + for (const id of ids) { + // this check will skip all custom layers + if (serializedLayersDictionary[id]) { + serializedLayers.push(serializedLayersDictionary[id]); + } + } + + return serializedLayers; + } + + /** + * Lazy initialization of this._serializedLayers dictionary and return it + * @returns this._serializedLayers dictionary + */ + private _serializedAllLayers(): {[_: string]: LayerSpecification} { + let serializedLayers = this._serializedLayers; + if (serializedLayers) { + return serializedLayers; + } + + serializedLayers = this._serializedLayers = {}; + const allLayerIds: string [] = Object.keys(this._layers); + for (const layerId of allLayerIds) { + const layer = this._layers[layerId]; + if (layer.type !== 'custom') { + serializedLayers[layerId] = layer.serialize(); + } + } + + return serializedLayers; + } + + hasTransitions() { + if (this.light && this.light.hasTransition()) { + return true; + } + + for (const id in this.sourceCaches) { + if (this.sourceCaches[id].hasTransition()) { + return true; + } + } + + for (const id in this._layers) { + if (this._layers[id].hasTransition()) { + return true; + } + } + + return false; + } + + _checkLoaded() { + if (!this._loaded) { + throw new Error('Style is not done loading.'); + } + } + + /** + * @internal + * Apply queued style updates in a batch and recalculate zoom-dependent paint properties. + */ + update(parameters: EvaluationParameters) { + if (!this._loaded) { + return; + } + + const changed = this._changed; + if (this._changed) { + const updatedIds = Object.keys(this._updatedLayers); + const removedIds = Object.keys(this._removedLayers); + + if (updatedIds.length || removedIds.length) { + this._updateWorkerLayers(updatedIds, removedIds); + } + for (const id in this._updatedSources) { + const action = this._updatedSources[id]; + + if (action === 'reload') { + this._reloadSource(id); + } else if (action === 'clear') { + this._clearSource(id); + } else { + throw new Error(`Invalid action ${action}`); + } + } + + this._updateTilesForChangedImages(); + this._updateTilesForChangedGlyphs(); + + for (const id in this._updatedPaintProps) { + this._layers[id].updateTransitions(parameters); + } + + this.light.updateTransitions(parameters); + + this._resetUpdates(); + } + + const sourcesUsedBefore = {}; + + for (const sourceId in this.sourceCaches) { + const sourceCache = this.sourceCaches[sourceId]; + sourcesUsedBefore[sourceId] = sourceCache.used; + sourceCache.used = false; + } + + for (const layerId of this._order) { + const layer = this._layers[layerId]; + + layer.recalculate(parameters, this._availableImages); + if (!layer.isHidden(parameters.zoom) && layer.source) { + this.sourceCaches[layer.source].used = true; + } + } + + for (const sourceId in sourcesUsedBefore) { + const sourceCache = this.sourceCaches[sourceId]; + if (sourcesUsedBefore[sourceId] !== sourceCache.used) { + sourceCache.fire(new Event('data', {sourceDataType: 'visibility', dataType: 'source', sourceId})); + } + } + + this.light.recalculate(parameters); + this.z = parameters.zoom; + + if (changed) { + this.fire(new Event('data', {dataType: 'style'})); + } + + } + + /* + * Apply any queued image changes. + */ + _updateTilesForChangedImages() { + const changedImages = Object.keys(this._changedImages); + if (changedImages.length) { + for (const name in this.sourceCaches) { + this.sourceCaches[name].reloadTilesForDependencies(['icons', 'patterns'], changedImages); + } + this._changedImages = {}; + } + } + + _updateTilesForChangedGlyphs() { + if (this._glyphsDidChange) { + for (const name in this.sourceCaches) { + this.sourceCaches[name].reloadTilesForDependencies(['glyphs'], ['']); + } + this._glyphsDidChange = false; + } + } + + _updateWorkerLayers(updatedIds: Array, removedIds: Array) { + this.dispatcher.broadcast('updateLayers', { + layers: this._serializeByIds(updatedIds), + removedIds + }); + } + + _resetUpdates() { + this._changed = false; + + this._updatedLayers = {}; + this._removedLayers = {}; + + this._updatedSources = {}; + this._updatedPaintProps = {}; + + this._changedImages = {}; + this._glyphsDidChange = false; + } + + /** + * Update this style's state to match the given style JSON, performing only + * the necessary mutations. + * + * May throw an Error ('Unimplemented: METHOD') if the mapbox-gl-style-spec + * diff algorithm produces an operation that is not supported. + * + * @returns true if any changes were made; false otherwise + */ + setState(nextState: StyleSpecification, options: StyleSwapOptions = {}) { + this._checkLoaded(); + + const serializedStyle = this.serialize(); + nextState = options.transformStyle ? options.transformStyle(serializedStyle, nextState) : nextState; + if (emitValidationErrors(this, validateStyle(nextState))) return false; + + nextState = clone(nextState); + nextState.layers = deref(nextState.layers); + + const changes = diffStyles(serializedStyle, nextState) + .filter(op => !(op.command in ignoredDiffOperations)); + + if (changes.length === 0) { + return false; + } + + const unimplementedOps = changes.filter(op => !(op.command in supportedDiffOperations)); + if (unimplementedOps.length > 0) { + throw new Error(`Unimplemented: ${unimplementedOps.map(op => op.command).join(', ')}.`); + } + + for (const op of changes) { + if (op.command === 'setTransition') { + // `transition` is always read directly off of + // `this.stylesheet`, which we update below + continue; + } + (this as any)[op.command].apply(this, op.args); + } + + this.stylesheet = nextState; + + // reset serialization field, to be populated only when needed + this._serializedLayers = null; + + return true; + } + + addImage(id: string, image: StyleImage) { + if (this.getImage(id)) { + return this.fire(new ErrorEvent(new Error(`An image named "${id}" already exists.`))); + } + this.imageManager.addImage(id, image); + this._afterImageUpdated(id); + } + + updateImage(id: string, image: StyleImage) { + this.imageManager.updateImage(id, image); + } + + getImage(id: string): StyleImage { + return this.imageManager.getImage(id); + } + + removeImage(id: string) { + if (!this.getImage(id)) { + return this.fire(new ErrorEvent(new Error(`An image named "${id}" does not exist.`))); + } + this.imageManager.removeImage(id); + this._afterImageUpdated(id); + } + + _afterImageUpdated(id: string) { + this._availableImages = this.imageManager.listImages(); + this._changedImages[id] = true; + this._changed = true; + this.dispatcher.broadcast('setImages', this._availableImages); + this.fire(new Event('data', {dataType: 'style'})); + } + + listImages() { + this._checkLoaded(); + + return this.imageManager.listImages(); + } + + addSource(id: string, source: SourceSpecification, options: StyleSetterOptions = {}) { + this._checkLoaded(); + + if (this.sourceCaches[id] !== undefined) { + throw new Error(`Source "${id}" already exists.`); + } + + if (!source.type) { + throw new Error(`The type property must be defined, but only the following properties were given: ${Object.keys(source).join(', ')}.`); + } + + const builtIns = ['vector', 'raster', 'geojson', 'video', 'image']; + const shouldValidate = builtIns.indexOf(source.type) >= 0; + if (shouldValidate && this._validate(validateStyle.source, `sources.${id}`, source, null, options)) return; + + if (this.map && this.map._collectResourceTiming) (source as any).collectResourceTiming = true; + const sourceCache = this.sourceCaches[id] = new SourceCache(id, source, this.dispatcher); + sourceCache.style = this; + sourceCache.setEventedParent(this, () => ({ + isSourceLoaded: sourceCache.loaded(), + source: sourceCache.serialize(), + sourceId: id + })); + + sourceCache.onAdd(this.map); + this._changed = true; + } + + /** + * Remove a source from this stylesheet, given its id. + * @param id - id of the source to remove + * @throws if no source is found with the given ID + * @returns `this`. + */ + removeSource(id: string): this { + this._checkLoaded(); + + if (this.sourceCaches[id] === undefined) { + throw new Error('There is no source with this ID'); + } + for (const layerId in this._layers) { + if (this._layers[layerId].source === id) { + return this.fire(new ErrorEvent(new Error(`Source "${id}" cannot be removed while layer "${layerId}" is using it.`))); + } + } + + const sourceCache = this.sourceCaches[id]; + delete this.sourceCaches[id]; + delete this._updatedSources[id]; + sourceCache.fire(new Event('data', {sourceDataType: 'metadata', dataType: 'source', sourceId: id})); + sourceCache.setEventedParent(null); + sourceCache.onRemove(this.map); + this._changed = true; + } + + /** + * Set the data of a GeoJSON source, given its id. + * @param id - id of the source + * @param data - GeoJSON source + */ + setGeoJSONSourceData(id: string, data: GeoJSON.GeoJSON | string) { + this._checkLoaded(); + + if (this.sourceCaches[id] === undefined) throw new Error(`There is no source with this ID=${id}`); + const geojsonSource: GeoJSONSource = (this.sourceCaches[id].getSource() as any); + if (geojsonSource.type !== 'geojson') throw new Error(`geojsonSource.type is ${geojsonSource.type}, which is !== 'geojson`); + + geojsonSource.setData(data); + this._changed = true; + } + + /** + * Get a source by ID. + * @param id - ID of the desired source + * @returns source + */ + getSource(id: string): Source | undefined { + return this.sourceCaches[id] && this.sourceCaches[id].getSource(); + } + + /** + * Add a layer to the map style. The layer will be inserted before the layer with + * ID `before`, or appended if `before` is omitted. + * @param layerObject - The style layer to add. + * @param before - ID of an existing layer to insert before + * @param options - Style setter options. + * @returns `this`. + */ + addLayer(layerObject: AddLayerObject, before?: string, options: StyleSetterOptions = {}): this { + this._checkLoaded(); + + const id = layerObject.id; + + if (this.getLayer(id)) { + this.fire(new ErrorEvent(new Error(`Layer "${id}" already exists on this map.`))); + return; + } + + let layer: ReturnType; + if (layerObject.type === 'custom') { + + if (emitValidationErrors(this, validateCustomStyleLayer(layerObject))) return; + + layer = createStyleLayer(layerObject); + + } else { + if ('source' in layerObject && typeof layerObject.source === 'object') { + this.addSource(id, layerObject.source); + layerObject = clone(layerObject); + layerObject = extend(layerObject, {source: id}); + } + + // this layer is not in the style.layers array, so we pass an impossible array index + if (this._validate(validateStyle.layer, + `layers.${id}`, layerObject, {arrayIndex: -1}, options)) return; + + layer = createStyleLayer(layerObject as LayerSpecification | CustomLayerInterface); + this._validateLayer(layer); + + layer.setEventedParent(this, {layer: {id}}); + } + + const index = before ? this._order.indexOf(before) : this._order.length; + if (before && index === -1) { + this.fire(new ErrorEvent(new Error(`Cannot add layer "${id}" before non-existing layer "${before}".`))); + return; + } + + this._order.splice(index, 0, id); + this._layerOrderChanged = true; + + this._layers[id] = layer; + + if (this._removedLayers[id] && layer.source && layer.type !== 'custom') { + // If, in the current batch, we have already removed this layer + // and we are now re-adding it with a different `type`, then we + // need to clear (rather than just reload) the underlying source's + // tiles. Otherwise, tiles marked 'reloading' will have buckets / + // buffers that are set up for the _previous_ version of this + // layer, causing, e.g.: + // https://github.com/mapbox/mapbox-gl-js/issues/3633 + const removed = this._removedLayers[id]; + delete this._removedLayers[id]; + if (removed.type !== layer.type) { + this._updatedSources[layer.source] = 'clear'; + } else { + this._updatedSources[layer.source] = 'reload'; + this.sourceCaches[layer.source].pause(); + } + } + this._updateLayer(layer); + + if (layer.onAdd) { + layer.onAdd(this.map); + } + } + + /** + * Moves a layer to a different z-position. The layer will be inserted before the layer with + * ID `before`, or appended if `before` is omitted. + * @param id - ID of the layer to move + * @param before - ID of an existing layer to insert before + */ + moveLayer(id: string, before?: string) { + this._checkLoaded(); + this._changed = true; + + const layer = this._layers[id]; + if (!layer) { + this.fire(new ErrorEvent(new Error(`The layer '${id}' does not exist in the map's style and cannot be moved.`))); + return; + } + + if (id === before) { + return; + } + + const index = this._order.indexOf(id); + this._order.splice(index, 1); + + const newIndex = before ? this._order.indexOf(before) : this._order.length; + if (before && newIndex === -1) { + this.fire(new ErrorEvent(new Error(`Cannot move layer "${id}" before non-existing layer "${before}".`))); + return; + } + this._order.splice(newIndex, 0, id); + + this._layerOrderChanged = true; + } + + /** + * Remove the layer with the given id from the style. + * + * If no such layer exists, an `error` event is fired. + * + * @param id - id of the layer to remove + * @event `error` - Fired if the layer does not exist + */ + removeLayer(id: string) { + this._checkLoaded(); + + const layer = this._layers[id]; + if (!layer) { + this.fire(new ErrorEvent(new Error(`Cannot remove non-existing layer "${id}".`))); + return; + } + + layer.setEventedParent(null); + + const index = this._order.indexOf(id); + this._order.splice(index, 1); + + this._layerOrderChanged = true; + this._changed = true; + this._removedLayers[id] = layer; + delete this._layers[id]; + + if (this._serializedLayers) { + delete this._serializedLayers[id]; + } + delete this._updatedLayers[id]; + delete this._updatedPaintProps[id]; + + if (layer.onRemove) { + layer.onRemove(this.map); + } + } + + /** + * Return the style layer object with the given `id`. + * + * @param id - id of the desired layer + * @returns a layer, if one with the given `id` exists + */ + getLayer(id: string): StyleLayer | undefined { + return this._layers[id]; + } + + /** + * Return the ids of all layers currently in the style, including custom layers, in order. + * + * @returns ids of layers, in order + */ + getLayersOrder(): string[] { + return [...this._order]; + } + + /** + * Checks if a specific layer is present within the style. + * + * @param id - the id of the desired layer + * @returns a boolean specifying if the given layer is present + */ + hasLayer(id: string): boolean { + return id in this._layers; + } + + setLayerZoomRange(layerId: string, minzoom?: number | null, maxzoom?: number | null) { + this._checkLoaded(); + + const layer = this.getLayer(layerId); + if (!layer) { + this.fire(new ErrorEvent(new Error(`Cannot set the zoom range of non-existing layer "${layerId}".`))); + return; + } + + if (layer.minzoom === minzoom && layer.maxzoom === maxzoom) return; + + if (minzoom != null) { + layer.minzoom = minzoom; + } + if (maxzoom != null) { + layer.maxzoom = maxzoom; + } + this._updateLayer(layer); + } + + setFilter(layerId: string, filter?: FilterSpecification | null, options: StyleSetterOptions = {}) { + this._checkLoaded(); + + const layer = this.getLayer(layerId); + if (!layer) { + this.fire(new ErrorEvent(new Error(`Cannot filter non-existing layer "${layerId}".`))); + return; + } + + if (deepEqual(layer.filter, filter)) { + return; + } + + if (filter === null || filter === undefined) { + layer.filter = undefined; + this._updateLayer(layer); + return; + } + + if (this._validate(validateStyle.filter, `layers.${layer.id}.filter`, filter, null, options)) { + return; + } + + layer.filter = clone(filter); + this._updateLayer(layer); + } + + /** + * Get a layer's filter object + * @param layer - the layer to inspect + * @returns the layer's filter, if any + */ + getFilter(layer: string): FilterSpecification | void { + return clone(this.getLayer(layer).filter); + } + + setLayoutProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}) { + this._checkLoaded(); + + const layer = this.getLayer(layerId); + if (!layer) { + this.fire(new ErrorEvent(new Error(`Cannot style non-existing layer "${layerId}".`))); + return; + } + + if (deepEqual(layer.getLayoutProperty(name), value)) return; + + layer.setLayoutProperty(name, value, options); + this._updateLayer(layer); + } + + /** + * Get a layout property's value from a given layer + * @param layerId - the layer to inspect + * @param name - the name of the layout property + * @returns the property value + */ + getLayoutProperty(layerId: string, name: string) { + const layer = this.getLayer(layerId); + if (!layer) { + this.fire(new ErrorEvent(new Error(`Cannot get style of non-existing layer "${layerId}".`))); + return; + } + + return layer.getLayoutProperty(name); + } + + setPaintProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}) { + this._checkLoaded(); + + const layer = this.getLayer(layerId); + if (!layer) { + this.fire(new ErrorEvent(new Error(`Cannot style non-existing layer "${layerId}".`))); + return; + } + + if (deepEqual(layer.getPaintProperty(name), value)) return; + + const requiresRelayout = layer.setPaintProperty(name, value, options); + if (requiresRelayout) { + this._updateLayer(layer); + } + + this._changed = true; + this._updatedPaintProps[layerId] = true; + } + + getPaintProperty(layer: string, name: string) { + return this.getLayer(layer).getPaintProperty(name); + } + + setFeatureState(target: FeatureIdentifier, state: any) { + this._checkLoaded(); + const sourceId = target.source; + const sourceLayer = target.sourceLayer; + const sourceCache = this.sourceCaches[sourceId]; + + if (sourceCache === undefined) { + this.fire(new ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`))); + return; + } + const sourceType = sourceCache.getSource().type; + if (sourceType === 'geojson' && sourceLayer) { + this.fire(new ErrorEvent(new Error('GeoJSON sources cannot have a sourceLayer parameter.'))); + return; + } + if (sourceType === 'vector' && !sourceLayer) { + this.fire(new ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.'))); + return; + } + if (target.id === undefined) { + this.fire(new ErrorEvent(new Error('The feature id parameter must be provided.'))); + } + + sourceCache.setFeatureState(sourceLayer, target.id, state); + } + + removeFeatureState(target: FeatureIdentifier, key?: string) { + this._checkLoaded(); + const sourceId = target.source; + const sourceCache = this.sourceCaches[sourceId]; + + if (sourceCache === undefined) { + this.fire(new ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`))); + return; + } + + const sourceType = sourceCache.getSource().type; + const sourceLayer = sourceType === 'vector' ? target.sourceLayer : undefined; + + if (sourceType === 'vector' && !sourceLayer) { + this.fire(new ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.'))); + return; + } + + if (key && (typeof target.id !== 'string' && typeof target.id !== 'number')) { + this.fire(new ErrorEvent(new Error('A feature id is required to remove its specific state property.'))); + return; + } + + sourceCache.removeFeatureState(sourceLayer, target.id, key); + } + + getFeatureState(target: FeatureIdentifier) { + this._checkLoaded(); + const sourceId = target.source; + const sourceLayer = target.sourceLayer; + const sourceCache = this.sourceCaches[sourceId]; + + if (sourceCache === undefined) { + this.fire(new ErrorEvent(new Error(`The source '${sourceId}' does not exist in the map's style.`))); + return; + } + const sourceType = sourceCache.getSource().type; + if (sourceType === 'vector' && !sourceLayer) { + this.fire(new ErrorEvent(new Error('The sourceLayer parameter must be provided for vector source types.'))); + return; + } + if (target.id === undefined) { + this.fire(new ErrorEvent(new Error('The feature id parameter must be provided.'))); + } + + return sourceCache.getFeatureState(sourceLayer, target.id); + } + + getTransition() { + return extend({duration: 300, delay: 0}, this.stylesheet && this.stylesheet.transition); + } + + serialize(): StyleSpecification { + // We return undefined before we're loaded, following the pattern of Map.getStyle() before + // the Style object is initialized. + // Internally, Style._validate() calls Style.serialize() but callers are responsible for + // calling Style._checkLoaded() first if their validation requires the style to be loaded. + if (!this._loaded) return; + + const sources = mapObject(this.sourceCaches, (source) => source.serialize()); + const layers = this._serializeByIds(this._order); + const terrain = this.map.getTerrain() || undefined; + const myStyleSheet = this.stylesheet; + + return filterObject({ + version: myStyleSheet.version, + name: myStyleSheet.name, + metadata: myStyleSheet.metadata, + light: myStyleSheet.light, + center: myStyleSheet.center, + zoom: myStyleSheet.zoom, + bearing: myStyleSheet.bearing, + pitch: myStyleSheet.pitch, + sprite: myStyleSheet.sprite, + glyphs: myStyleSheet.glyphs, + transition: myStyleSheet.transition, + sources, + layers, + terrain + }, + (value) => { return value !== undefined; }); + } + + _updateLayer(layer: StyleLayer) { + this._updatedLayers[layer.id] = true; + if (layer.source && !this._updatedSources[layer.source] && + //Skip for raster layers (https://github.com/mapbox/mapbox-gl-js/issues/7865) + this.sourceCaches[layer.source].getSource().type !== 'raster') { + this._updatedSources[layer.source] = 'reload'; + this.sourceCaches[layer.source].pause(); + } + + // upon updating, serilized layer dictionary should be reset. + // When needed, it will be populated with the correct copy again. + this._serializedLayers = null; + this._changed = true; + } + + _flattenAndSortRenderedFeatures(sourceResults: Array<{ [key: string]: Array<{featureIndex: number; feature: MapGeoJSONFeature}> }>) { + // Feature order is complicated. + // The order between features in two 2D layers is always determined by layer order. + // The order between features in two 3D layers is always determined by depth. + // The order between a feature in a 2D layer and a 3D layer is tricky: + // Most often layer order determines the feature order in this case. If + // a line layer is above a extrusion layer the line feature will be rendered + // above the extrusion. If the line layer is below the extrusion layer, + // it will be rendered below it. + // + // There is a weird case though. + // You have layers in this order: extrusion_layer_a, line_layer, extrusion_layer_b + // Each layer has a feature that overlaps the other features. + // The feature in extrusion_layer_a is closer than the feature in extrusion_layer_b so it is rendered above. + // The feature in line_layer is rendered above extrusion_layer_a. + // This means that that the line_layer feature is above the extrusion_layer_b feature despite + // it being in an earlier layer. + + const isLayer3D = layerId => this._layers[layerId].type === 'fill-extrusion'; + + const layerIndex = {}; + const features3D = []; + for (let l = this._order.length - 1; l >= 0; l--) { + const layerId = this._order[l]; + if (isLayer3D(layerId)) { + layerIndex[layerId] = l; + for (const sourceResult of sourceResults) { + const layerFeatures = sourceResult[layerId]; + if (layerFeatures) { + for (const featureWrapper of layerFeatures) { + features3D.push(featureWrapper); + } + } + } + } + } + + features3D.sort((a, b) => { + return b.intersectionZ - a.intersectionZ; + }); + + const features = []; + for (let l = this._order.length - 1; l >= 0; l--) { + const layerId = this._order[l]; + + if (isLayer3D(layerId)) { + // add all 3D features that are in or above the current layer + for (let i = features3D.length - 1; i >= 0; i--) { + const topmost3D = features3D[i].feature; + if (layerIndex[topmost3D.layer.id] < l) break; + features.push(topmost3D); + features3D.pop(); + } + } else { + for (const sourceResult of sourceResults) { + const layerFeatures = sourceResult[layerId]; + if (layerFeatures) { + for (const featureWrapper of layerFeatures) { + features.push(featureWrapper.feature); + } + } + } + } + } + + return features; + } + + queryRenderedFeatures(queryGeometry: any, params: QueryRenderedFeaturesOptions, transform: Transform) { + if (params && params.filter) { + this._validate(validateStyle.filter, 'queryRenderedFeatures.filter', params.filter, null, params); + } + + const includedSources = {}; + if (params && params.layers) { + if (!Array.isArray(params.layers)) { + this.fire(new ErrorEvent(new Error('parameters.layers must be an Array.'))); + return []; + } + for (const layerId of params.layers) { + const layer = this._layers[layerId]; + if (!layer) { + // this layer is not in the style.layers array + this.fire(new ErrorEvent(new Error(`The layer '${layerId}' does not exist in the map's style and cannot be queried for features.`))); + return []; + } + includedSources[layer.source] = true; + } + } + + const sourceResults = []; + + params.availableImages = this._availableImages; + + // LayerSpecification is serialized StyleLayer, and this casting is safe. + const serializedLayers = this._serializedAllLayers() as {[_: string]: StyleLayer}; + + for (const id in this.sourceCaches) { + if (params.layers && !includedSources[id]) continue; + sourceResults.push( + queryRenderedFeatures( + this.sourceCaches[id], + this._layers, + serializedLayers, + queryGeometry, + params, + transform) + ); + } + + if (this.placement) { + // If a placement has run, query against its CollisionIndex + // for symbol results, and treat it as an extra source to merge + sourceResults.push( + queryRenderedSymbols( + this._layers, + serializedLayers, + this.sourceCaches, + queryGeometry, + params, + this.placement.collisionIndex, + this.placement.retainedQueryData) + ); + } + + return this._flattenAndSortRenderedFeatures(sourceResults); + } + + querySourceFeatures( + sourceID: string, + params?: QuerySourceFeatureOptions + ) { + if (params && params.filter) { + this._validate(validateStyle.filter, 'querySourceFeatures.filter', params.filter, null, params); + } + const sourceCache = this.sourceCaches[sourceID]; + return sourceCache ? querySourceFeatures(sourceCache, params) : []; + } + + addSourceType(name: string, SourceType: SourceClass, callback: Callback) { + if (getSourceType(name)) { + return callback(new Error(`A source type called "${name}" already exists.`)); + } + + setSourceType(name, SourceType); + + if (!SourceType.workerSourceURL) { + return callback(null, null); + } + + this.dispatcher.broadcast('loadWorkerSource', { + name, + url: SourceType.workerSourceURL + }, callback); + } + + getLight() { + return this.light.getLight(); + } + + setLight(lightOptions: LightSpecification, options: StyleSetterOptions = {}) { + this._checkLoaded(); + + const light = this.light.getLight(); + let _update = false; + for (const key in lightOptions) { + if (!deepEqual(lightOptions[key], light[key])) { + _update = true; + break; + } + } + if (!_update) return; + + const parameters = { + now: browser.now(), + transition: extend({ + duration: 300, + delay: 0 + }, this.stylesheet.transition) + }; + + this.light.setLight(lightOptions, options); + this.light.updateTransitions(parameters); + } + + _validate(validate: Validator, key: string, value: any, props: any, options: { + validate?: boolean; + } = {}) { + if (options && options.validate === false) { + return false; + } + return emitValidationErrors(this, validate.call(validateStyle, extend({ + key, + style: this.serialize(), + value, + styleSpec + }, props))); + } + + _remove(mapRemoved: boolean = true) { + if (this._request) { + this._request.cancel(); + this._request = null; + } + if (this._spriteRequest) { + this._spriteRequest.cancel(); + this._spriteRequest = null; + } + rtlTextPluginEvented.off('pluginStateChange', this._rtlTextPluginCallback); + for (const layerId in this._layers) { + const layer: StyleLayer = this._layers[layerId]; + layer.setEventedParent(null); + } + for (const id in this.sourceCaches) { + const sourceCache = this.sourceCaches[id]; + sourceCache.setEventedParent(null); + sourceCache.onRemove(this.map); + } + this.imageManager.setEventedParent(null); + this.setEventedParent(null); + this.dispatcher.remove(mapRemoved); + } + + _clearSource(id: string) { + this.sourceCaches[id].clearTiles(); + } + + _reloadSource(id: string) { + this.sourceCaches[id].resume(); + this.sourceCaches[id].reload(); + } + + _updateSources(transform: Transform) { + for (const id in this.sourceCaches) { + this.sourceCaches[id].update(transform, this.map.terrain); + } + } + + _generateCollisionBoxes() { + for (const id in this.sourceCaches) { + this._reloadSource(id); + } + } + + _updatePlacement(transform: Transform, showCollisionBoxes: boolean, fadeDuration: number, crossSourceCollisions: boolean, forceFullPlacement: boolean = false) { + let symbolBucketsChanged = false; + let placementCommitted = false; + + const layerTiles = {}; + + for (const layerID of this._order) { + const styleLayer = this._layers[layerID]; + if (styleLayer.type !== 'symbol') continue; + + if (!layerTiles[styleLayer.source]) { + const sourceCache = this.sourceCaches[styleLayer.source]; + layerTiles[styleLayer.source] = sourceCache.getRenderableIds(true) + .map((id) => sourceCache.getTileByID(id)) + .sort((a, b) => (b.tileID.overscaledZ - a.tileID.overscaledZ) || (a.tileID.isLessThan(b.tileID) ? -1 : 1)); + } + + const layerBucketsChanged = this.crossTileSymbolIndex.addLayer(styleLayer, layerTiles[styleLayer.source], transform.center.lng); + symbolBucketsChanged = symbolBucketsChanged || layerBucketsChanged; + } + this.crossTileSymbolIndex.pruneUnusedLayers(this._order); + + // Anything that changes our "in progress" layer and tile indices requires us + // to start over. When we start over, we do a full placement instead of incremental + // to prevent starvation. + // We need to restart placement to keep layer indices in sync. + // Also force full placement when fadeDuration === 0 to ensure that newly loaded + // tiles will fully display symbols in their first frame + forceFullPlacement = forceFullPlacement || this._layerOrderChanged || fadeDuration === 0; + + if (forceFullPlacement || !this.pauseablePlacement || (this.pauseablePlacement.isDone() && !this.placement.stillRecent(browser.now(), transform.zoom))) { + this.pauseablePlacement = new PauseablePlacement(transform, this.map.terrain, this._order, forceFullPlacement, showCollisionBoxes, fadeDuration, crossSourceCollisions, this.placement); + this._layerOrderChanged = false; + } + + if (this.pauseablePlacement.isDone()) { + // the last placement finished running, but the next one hasn’t + // started yet because of the `stillRecent` check immediately + // above, so mark it stale to ensure that we request another + // render frame + this.placement.setStale(); + } else { + this.pauseablePlacement.continuePlacement(this._order, this._layers, layerTiles); + + if (this.pauseablePlacement.isDone()) { + this.placement = this.pauseablePlacement.commit(browser.now()); + placementCommitted = true; + } + + if (symbolBucketsChanged) { + // since the placement gets split over multiple frames it is possible + // these buckets were processed before they were changed and so the + // placement is already stale while it is in progress + this.pauseablePlacement.placement.setStale(); + } + } + + if (placementCommitted || symbolBucketsChanged) { + for (const layerID of this._order) { + const styleLayer = this._layers[layerID]; + if (styleLayer.type !== 'symbol') continue; + this.placement.updateLayerOpacities(styleLayer, layerTiles[styleLayer.source]); + } + } + + // needsRender is false when we have just finished a placement that didn't change the visibility of any symbols + const needsRerender = !this.pauseablePlacement.isDone() || this.placement.hasTransitions(browser.now()); + return needsRerender; + } + + _releaseSymbolFadeTiles() { + for (const id in this.sourceCaches) { + this.sourceCaches[id].releaseSymbolFadeTiles(); + } + } + + // Callbacks from web workers + + getImages( + mapId: string, + params: { + icons: Array; + source: string; + tileID: OverscaledTileID; + type: string; + }, + callback: Callback<{[_: string]: StyleImage}> + ) { + this.imageManager.getImages(params.icons, callback); + + // Apply queued image changes before setting the tile's dependencies so that the tile + // is not reloaded unnecessarily. Without this forced update the reload could happen in cases + // like this one: + // - icons contains "my-image" + // - imageManager.getImages(...) triggers `onstyleimagemissing` + // - the user adds "my-image" within the callback + // - addImage adds "my-image" to this._changedImages + // - the next frame triggers a reload of this tile even though it already has the latest version + this._updateTilesForChangedImages(); + + const sourceCache = this.sourceCaches[params.source]; + if (sourceCache) { + sourceCache.setDependencies(params.tileID.key, params.type, params.icons); + } + } + + getGlyphs( + mapId: string, + params: { + stacks: {[_: string]: Array}; + source: string; + tileID: OverscaledTileID; + type: string; + }, + callback: Callback<{[_: string]: {[_: number]: StyleGlyph}}> + ) { + this.glyphManager.getGlyphs(params.stacks, callback); + const sourceCache = this.sourceCaches[params.source]; + if (sourceCache) { + // we are not setting stacks as dependencies since for now + // we just need to know which tiles have glyph dependencies + sourceCache.setDependencies(params.tileID.key, params.type, ['']); + } + } + + getResource(mapId: string, params: RequestParameters, callback: ResponseCallback): Cancelable { + return makeRequest(params, callback); + } + + getGlyphsUrl() { + return this.stylesheet.glyphs || null; + } + + setGlyphs(glyphsUrl: string | null, options: StyleSetterOptions = {}) { + this._checkLoaded(); + if (glyphsUrl && this._validate(validateStyle.glyphs, 'glyphs', glyphsUrl, null, options)) { + return; + } + + this._glyphsDidChange = true; + this.stylesheet.glyphs = glyphsUrl; + this.glyphManager.entries = {}; + this.glyphManager.setURL(glyphsUrl); + } + + /** + * Add a sprite. + * + * @param id - The id of the desired sprite + * @param url - The url to load the desired sprite from + * @param options - The style setter options + * @param completion - The completion handler + */ + addSprite(id: string, url: string, options: StyleSetterOptions = {}, completion?: (err: Error) => void) { + this._checkLoaded(); + + const spriteToAdd = [{id, url}]; + const updatedSprite = [ + ...coerceSpriteToArray(this.stylesheet.sprite), + ...spriteToAdd + ]; + + if (this._validate(validateStyle.sprite, 'sprite', updatedSprite, null, options)) return; + + this.stylesheet.sprite = updatedSprite; + this._loadSprite(spriteToAdd, true, completion); + } + + /** + * Remove a sprite by its id. When the last sprite is removed, the whole `this.stylesheet.sprite` object becomes + * `undefined`. This falsy `undefined` value later prevents attempts to load the sprite when it's absent. + * + * @param id - the id of the sprite to remove + */ + removeSprite(id: string) { + this._checkLoaded(); + + const internalSpriteRepresentation = coerceSpriteToArray(this.stylesheet.sprite); + + if (!internalSpriteRepresentation.find(sprite => sprite.id === id)) { + this.fire(new ErrorEvent(new Error(`Sprite "${id}" doesn't exists on this map.`))); + return; + } + + if (this._spritesImagesIds[id]) { + for (const imageId of this._spritesImagesIds[id]) { + this.imageManager.removeImage(imageId); + this._changedImages[imageId] = true; + } + } + + internalSpriteRepresentation.splice(internalSpriteRepresentation.findIndex(sprite => sprite.id === id), 1); + this.stylesheet.sprite = internalSpriteRepresentation.length > 0 ? internalSpriteRepresentation : undefined; + + delete this._spritesImagesIds[id]; + this._availableImages = this.imageManager.listImages(); + this._changed = true; + this.dispatcher.broadcast('setImages', this._availableImages); + this.fire(new Event('data', {dataType: 'style'})); + } + + /** + * Get the current sprite value. + * + * @returns empty array when no sprite is set; id-url pairs otherwise + */ + getSprite() { + return coerceSpriteToArray(this.stylesheet.sprite); + } + + /** + * Set a new value for the style's sprite. + * + * @param sprite - new sprite value + * @param options - style setter options + * @param completion - the completion handler + */ + setSprite(sprite: SpriteSpecification, options: StyleSetterOptions = {}, completion?: (err: Error) => void) { + this._checkLoaded(); + + if (sprite && this._validate(validateStyle.sprite, 'sprite', sprite, null, options)) { + return; + } + + this.stylesheet.sprite = sprite; + + if (sprite) { + this._loadSprite(sprite, true, completion); + } else { + this._unloadSprite(); + if (completion) { + completion(null); + } + } + } +} + +Style.registerForPluginStateChange = registerForPluginStateChange; diff --git a/web/libraries/maplibre-gl/src/style/style_glyph.ts b/web/libraries/maplibre-gl/src/style/style_glyph.ts new file mode 100644 index 00000000..35efaad5 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_glyph.ts @@ -0,0 +1,23 @@ +import type {AlphaImage} from '../util/image'; + +export type GlyphMetrics = { + width: number; + height: number; + left: number; + top: number; + advance: number; + /** + * isDoubleResolution = true for 48px textures + */ + isDoubleResolution?: boolean; +}; + +/** + * @internal + * A style glyph type + */ +export type StyleGlyph = { + id: number; + bitmap: AlphaImage; + metrics: GlyphMetrics; +}; diff --git a/web/libraries/maplibre-gl/src/style/style_image.ts b/web/libraries/maplibre-gl/src/style/style_image.ts new file mode 100644 index 00000000..bcfadcca --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_image.ts @@ -0,0 +1,152 @@ +import {RGBAImage} from '../util/image'; + +import type {Map} from '../ui/map'; + +/** + * The sprite data + */ +export type SpriteOnDemandStyleImage = { + width: number; + height: number; + x: number; + y: number; + context: CanvasRenderingContext2D; +}; + +/** + * The style's image metadata + */ +export type StyleImageData = { + data: RGBAImage; + version?: number; + hasRenderCallback?: boolean; + userImage?: StyleImageInterface; + spriteData?: SpriteOnDemandStyleImage; +}; + +/** + * The style's image metadata + */ +export type StyleImageMetadata = { + /** + * The ratio of pixels in the image to physical pixels on the screen + */ + pixelRatio: number; + /** + * Whether the image should be interpreted as an SDF image + */ + sdf: boolean; + /** + * If `icon-text-fit` is used in a layer with this image, this option defines the part(s) of the image that can be stretched horizontally. + */ + stretchX?: Array<[number, number]>; + /** + * If `icon-text-fit` is used in a layer with this image, this option defines the part(s) of the image that can be stretched vertically. + */ + stretchY?: Array<[number, number]>; + /** + * If `icon-text-fit` is used in a layer with this image, this option defines the part of the image that can be covered by the content in `text-field`. + */ + content?: [number, number, number, number]; +}; + +/** + * the style's image, including data and metedata + */ +export type StyleImage = StyleImageData & StyleImageMetadata; + +/** + * Interface for dynamically generated style images. This is a specification for + * implementers to model: it is not an exported method or class. + * + * Images implementing this interface can be redrawn for every frame. They can be used to animate + * icons and patterns or make them respond to user input. Style images can implement a + * {@link StyleImageInterface#render} method. The method is called every frame and + * can be used to update the image. + * + * @see [Add an animated icon to the map.](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-animated/) + * + * @example + * ```ts + * let flashingSquare = { + * width: 64, + * height: 64, + * data: new Uint8Array(64 * 64 * 4), + * + * onAdd: function(map) { + * this.map = map; + * }, + * + * render: function() { + * // keep repainting while the icon is on the map + * this.map.triggerRepaint(); + * + * // alternate between black and white based on the time + * let value = Math.round(Date.now() / 1000) % 2 === 0 ? 255 : 0; + * + * // check if image needs to be changed + * if (value !== this.previousValue) { + * this.previousValue = value; + * + * let bytesPerPixel = 4; + * for (let x = 0; x < this.width; x++) { + * for (let y = 0; y < this.height; y++) { + * let offset = (y * this.width + x) * bytesPerPixel; + * this.data[offset + 0] = value; + * this.data[offset + 1] = value; + * this.data[offset + 2] = value; + * this.data[offset + 3] = 255; + * } + * } + * + * // return true to indicate that the image changed + * return true; + * } + * } + * } + * + * map.addImage('flashing_square', flashingSquare); + * ``` + */ + +export interface StyleImageInterface { + width: number; + height: number; + data: Uint8Array | Uint8ClampedArray; + /** + * This method is called once before every frame where the icon will be used. + * The method can optionally update the image's `data` member with a new image. + * + * If the method updates the image it must return `true` to commit the change. + * If the method returns `false` or nothing the image is assumed to not have changed. + * + * If updates are infrequent it maybe easier to use {@link Map#updateImage} to update + * the image instead of implementing this method. + * + * @returns `true` if this method updated the image. `false` if the image was not changed. + */ + render?: () => boolean; + /** + * Optional method called when the layer has been added to the Map with {@link Map#addImage}. + * + * @param map - The Map this custom layer was just added to. + */ + onAdd?: (map: Map, id: string) => void; + /** + * Optional method called when the icon is removed from the map with {@link Map#removeImage}. + * This gives the image a chance to clean up resources and event listeners. + */ + onRemove?: () => void; +} + +export function renderStyleImage(image: StyleImage) { + const {userImage} = image; + if (userImage && userImage.render) { + const updated = userImage.render(); + if (updated) { + image.data.replace(new Uint8Array(userImage.data.buffer)); + return true; + } + } + return false; +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer.test.ts b/web/libraries/maplibre-gl/src/style/style_layer.test.ts new file mode 100644 index 00000000..2022f37d --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer.test.ts @@ -0,0 +1,372 @@ +import {createStyleLayer} from './create_style_layer'; +import {FillStyleLayer} from './style_layer/fill_style_layer'; +import {extend} from '../util/util'; +import {Color, LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {EvaluationParameters} from './evaluation_parameters'; +import {TransitionParameters} from './properties'; +import {BackgroundStyleLayer} from './style_layer/background_style_layer'; +import {SymbolStyleLayer} from './style_layer/symbol_style_layer'; + +describe('StyleLayer', () => { + test('instantiates the correct subclass', () => { + const layer = createStyleLayer({type: 'fill'} as LayerSpecification); + + expect(layer instanceof FillStyleLayer).toBeTruthy(); + }); +}); + +describe('StyleLayer#setPaintProperty', () => { + test('sets new property value', () => { + const layer = createStyleLayer({ + 'id': 'background', + 'type': 'background' + }); + + layer.setPaintProperty('background-color', 'blue'); + + expect(layer.getPaintProperty('background-color')).toBe('blue'); + }); + + test('updates property value', () => { + const layer = createStyleLayer({ + 'id': 'background', + 'type': 'background', + 'paint': { + 'background-color': 'red' + } + }); + + layer.setPaintProperty('background-color', 'blue'); + + expect(layer.getPaintProperty('background-color')).toBe('blue'); + }); + + test('unsets value', () => { + const layer = createStyleLayer({ + 'id': 'background', + 'type': 'background', + 'paint': { + 'background-color': 'red', + 'background-opacity': 1 + } + }) as BackgroundStyleLayer; + + layer.setPaintProperty('background-color', null); + layer.updateTransitions({} as TransitionParameters); + layer.recalculate({zoom: 0, zoomHistory: {}} as EvaluationParameters, undefined); + + expect(layer.paint.get('background-color')).toEqual(new Color(0, 0, 0, 1)); + expect(layer.getPaintProperty('background-color')).toBeUndefined(); + expect(layer.paint.get('background-opacity')).toBe(1); + expect(layer.getPaintProperty('background-opacity')).toBe(1); + }); + + test('preserves existing transition', () => { + const layer = createStyleLayer({ + 'id': 'background', + 'type': 'background', + 'paint': { + 'background-color': 'red', + 'background-color-transition': { + duration: 600 + } + } as any + }); + + layer.setPaintProperty('background-color', 'blue'); + + expect(layer.getPaintProperty('background-color-transition')).toEqual({duration: 600}); + }); + + test('sets transition', () => { + const layer = createStyleLayer({ + 'id': 'background', + 'type': 'background', + 'paint': { + 'background-color': 'red' + } + }); + + layer.setPaintProperty('background-color-transition', {duration: 400}); + + expect(layer.getPaintProperty('background-color-transition')).toEqual({duration: 400}); + }); + + test('emits on an invalid property value', done => { + const layer = createStyleLayer({ + 'id': 'background', + 'type': 'background' + }); + + layer.on('error', () => { + expect(layer.getPaintProperty('background-opacity')).toBeUndefined(); + done(); + }); + + layer.setPaintProperty('background-opacity', 5); + }); + + test('emits on an invalid transition property value', done => { + const layer = createStyleLayer({ + 'id': 'background', + 'type': 'background' + }); + + layer.on('error', () => { + done(); + }); + + layer.setPaintProperty('background-opacity-transition', { + duration: -10 + }); + }); + + test('can unset fill-outline-color #2886', () => { + const layer = createStyleLayer({ + id: 'building', + type: 'fill', + source: 'streets', + paint: { + 'fill-color': '#00f' + } + }) as FillStyleLayer; + + layer.setPaintProperty('fill-outline-color', '#f00'); + layer.updateTransitions({} as TransitionParameters); + layer.recalculate({zoom: 0, zoomHistory: {}} as EvaluationParameters, undefined); + expect(layer.paint.get('fill-outline-color').value).toEqual({kind: 'constant', value: new Color(1, 0, 0, 1)}); + + layer.setPaintProperty('fill-outline-color', undefined); + layer.updateTransitions({} as TransitionParameters); + layer.recalculate({zoom: 0, zoomHistory: {}} as EvaluationParameters, undefined); + expect(layer.paint.get('fill-outline-color').value).toEqual({kind: 'constant', value: new Color(0, 0, 1, 1)}); + + }); + + test('can transition fill-outline-color from undefined to a value #3657', () => { + const layer = createStyleLayer({ + id: 'building', + type: 'fill', + source: 'streets', + paint: { + 'fill-color': '#00f' + } + }) as FillStyleLayer; + + // setup: set and then unset fill-outline-color so that, when we then try + // to re-set it, StyleTransition#calculate() attempts interpolation + layer.setPaintProperty('fill-outline-color', '#f00'); + layer.updateTransitions({} as TransitionParameters); + layer.recalculate({zoom: 0, zoomHistory: {}} as EvaluationParameters, undefined); + + layer.setPaintProperty('fill-outline-color', undefined); + layer.updateTransitions({} as TransitionParameters); + layer.recalculate({zoom: 0, zoomHistory: {}} as EvaluationParameters, undefined); + + // re-set fill-outline-color and get its value, triggering the attempt + // to interpolate between undefined and #f00 + layer.setPaintProperty('fill-outline-color', '#f00'); + layer.updateTransitions({} as TransitionParameters); + layer.recalculate({zoom: 0, zoomHistory: {}} as EvaluationParameters, undefined); + + layer.paint.get('fill-outline-color'); + + }); + + test('sets null property value', () => { + const layer = createStyleLayer({ + 'id': 'background', + 'type': 'background' + }); + + layer.setPaintProperty('background-color-transition', null); + + expect(layer.getPaintProperty('background-color-transition')).toBeUndefined(); + }); + +}); + +describe('StyleLayer#setLayoutProperty', () => { + test('sets new property value', () => { + const layer = createStyleLayer({ + 'id': 'symbol', + 'type': 'symbol' + } as LayerSpecification); + + layer.setLayoutProperty('text-transform', 'lowercase'); + + expect(layer.getLayoutProperty('text-transform')).toBe('lowercase'); + }); + + test('emits on an invalid property value', done => { + const layer = createStyleLayer({ + 'id': 'symbol', + 'type': 'symbol' + } as LayerSpecification); + + layer.on('error', () => { + done(); + }); + + layer.setLayoutProperty('text-transform', 'invalidValue'); + }); + + test('updates property value', () => { + const layer = createStyleLayer({ + 'id': 'symbol', + 'type': 'symbol', + 'layout': { + 'text-transform': 'uppercase' + } + } as LayerSpecification); + + layer.setLayoutProperty('text-transform', 'lowercase'); + + expect(layer.getLayoutProperty('text-transform')).toBe('lowercase'); + }); + + test('unsets property value', () => { + const layer = createStyleLayer({ + 'id': 'symbol', + 'type': 'symbol', + 'layout': { + 'text-transform': 'uppercase' + } + } as LayerSpecification) as SymbolStyleLayer; + + layer.setLayoutProperty('text-transform', null); + layer.recalculate({zoom: 0, zoomHistory: {}} as EvaluationParameters, undefined); + + expect(layer.layout.get('text-transform').value).toEqual({kind: 'constant', value: 'none'}); + expect(layer.getLayoutProperty('text-transform')).toBeUndefined(); + }); +}); + +describe('StyleLayer#serialize', () => { + + function createSymbolLayer(layer?) { + return extend({ + id: 'symbol', + type: 'symbol', + paint: { + 'text-color': 'blue' + }, + layout: { + 'text-transform': 'uppercase' + } + }, layer); + } + + test('serializes layers', () => { + expect(createStyleLayer(createSymbolLayer()).serialize()).toEqual(createSymbolLayer()); + }); + + test('serializes functions', () => { + const layerPaint = { + 'text-color': { + base: 2, + stops: [[0, 'red'], [1, 'blue']] + } + }; + + expect(createStyleLayer(createSymbolLayer({paint: layerPaint})).serialize().paint).toEqual(layerPaint); + }); + + test('serializes added paint properties', () => { + const layer = createStyleLayer(createSymbolLayer()); + layer.setPaintProperty('text-halo-color', 'orange'); + + expect(layer.serialize().paint['text-halo-color']).toBe('orange'); + expect(layer.serialize().paint['text-color']).toBe('blue'); + + }); + + test('serializes added layout properties', () => { + const layer = createStyleLayer(createSymbolLayer()); + layer.setLayoutProperty('text-size', 20); + + expect(layer.serialize().layout['text-transform']).toBe('uppercase'); + expect(layer.serialize().layout['text-size']).toBe(20); + + }); + + test('serializes "visibility" of "visible"', () => { + const layer = createStyleLayer(createSymbolLayer()); + layer.setLayoutProperty('visibility', 'visible'); + + expect(layer.serialize().layout['visibility']).toBe('visible'); + + }); + + test('serializes "visibility" of "none"', () => { + const layer = createStyleLayer(createSymbolLayer()); + layer.setLayoutProperty('visibility', 'none'); + + expect(layer.serialize().layout['visibility']).toBe('none'); + + }); + + test('serializes "visibility" of undefined', () => { + const layer = createStyleLayer(createSymbolLayer()); + layer.setLayoutProperty('visibility', undefined); + + expect(layer.serialize().layout['visibility']).toBeUndefined(); + + }); + +}); + +describe('StyleLayer#serialize', () => { + + function createSymbolLayer(layer?) { + return extend({ + id: 'symbol', + type: 'symbol', + paint: { + 'text-color': 'blue' + }, + layout: { + 'text-transform': 'uppercase' + } + }, layer); + } + + test('serializes layers', () => { + expect(createStyleLayer(createSymbolLayer()).serialize()).toEqual(createSymbolLayer()); + }); + + test('serializes functions', () => { + const layerPaint = { + 'text-color': { + base: 2, + stops: [[0, 'red'], [1, 'blue']] + } + }; + + expect(createStyleLayer(createSymbolLayer({paint: layerPaint})).serialize().paint).toEqual(layerPaint); + }); + + test('serializes added paint properties', () => { + const layer = createStyleLayer(createSymbolLayer()); + layer.setPaintProperty('text-halo-color', 'orange'); + + expect(layer.serialize().paint['text-halo-color']).toBe('orange'); + expect(layer.serialize().paint['text-color']).toBe('blue'); + + }); + + test('serializes added layout properties', () => { + const layer = createStyleLayer(createSymbolLayer()); + layer.setLayoutProperty('text-size', 20); + + expect(layer.serialize().layout['text-transform']).toBe('uppercase'); + expect(layer.serialize().layout['text-size']).toBe(20); + + }); + + test('layer.paint is never undefined', () => { + const layer = createStyleLayer({type: 'fill'} as LayerSpecification); + // paint is never undefined + expect(layer.paint).toBeTruthy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/style/style_layer.ts b/web/libraries/maplibre-gl/src/style/style_layer.ts new file mode 100644 index 00000000..277daae6 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer.ts @@ -0,0 +1,287 @@ +import {filterObject} from '../util/util'; + +import {latest as styleSpec, supportsPropertyExpression} from '@maplibre/maplibre-gl-style-spec'; +import { + validateStyle, + validateLayoutProperty, + validatePaintProperty, + emitValidationErrors +} from './validate_style'; +import {Evented} from '../util/evented'; +import {Layout, Transitionable, Transitioning, Properties, PossiblyEvaluated, PossiblyEvaluatedPropertyValue} from './properties'; + +import type {Bucket} from '../data/bucket'; +import type Point from '@mapbox/point-geometry'; +import type {FeatureFilter, FeatureState, + LayerSpecification, + FilterSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {TransitionParameters, PropertyValue} from './properties'; +import {EvaluationParameters} from './evaluation_parameters'; +import type {CrossfadeParameters} from './evaluation_parameters'; + +import type {Transform} from '../geo/transform'; +import type {CustomLayerInterface} from './style_layer/custom_style_layer'; +import type {Map} from '../ui/map'; +import type {StyleSetterOptions} from './style'; +import {mat4} from 'gl-matrix'; +import type {VectorTileFeature} from '@mapbox/vector-tile'; + +const TRANSITION_SUFFIX = '-transition'; + +/** + * A base class for style layers + */ +export abstract class StyleLayer extends Evented { + id: string; + metadata: unknown; + type: LayerSpecification['type'] | CustomLayerInterface['type']; + source: string; + sourceLayer: string; + minzoom: number; + maxzoom: number; + filter: FilterSpecification | void; + visibility: 'visible' | 'none' | void; + _crossfadeParameters: CrossfadeParameters; + + _unevaluatedLayout: Layout; + readonly layout: unknown; + + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + readonly paint: unknown; + + _featureFilter: FeatureFilter; + + readonly onAdd: ((map: Map) => void); + readonly onRemove: ((map: Map) => void); + + queryRadius?(bucket: Bucket): number; + queryIntersectsFeature?( + queryGeometry: Array, + feature: VectorTileFeature, + featureState: FeatureState, + geometry: Array>, + zoom: number, + transform: Transform, + pixelsToTileUnits: number, + pixelPosMatrix: mat4 + ): boolean | number; + + constructor(layer: LayerSpecification | CustomLayerInterface, properties: Readonly<{ + layout?: Properties; + paint?: Properties; + }>) { + super(); + + this.id = layer.id; + this.type = layer.type; + this._featureFilter = {filter: () => true, needGeometry: false}; + + if (layer.type === 'custom') return; + + layer = (layer as any as LayerSpecification); + + this.metadata = layer.metadata; + this.minzoom = layer.minzoom; + this.maxzoom = layer.maxzoom; + + if (layer.type !== 'background') { + this.source = layer.source; + this.sourceLayer = layer['source-layer']; + this.filter = layer.filter; + } + + if (properties.layout) { + this._unevaluatedLayout = new Layout(properties.layout); + } + + if (properties.paint) { + this._transitionablePaint = new Transitionable(properties.paint); + + for (const property in layer.paint) { + this.setPaintProperty(property, layer.paint[property], {validate: false}); + } + for (const property in layer.layout) { + this.setLayoutProperty(property, layer.layout[property], {validate: false}); + } + + this._transitioningPaint = this._transitionablePaint.untransitioned(); + //$FlowFixMe + this.paint = new PossiblyEvaluated(properties.paint); + } + } + + getCrossfadeParameters() { + return this._crossfadeParameters; + } + + getLayoutProperty(name: string) { + if (name === 'visibility') { + return this.visibility; + } + + return this._unevaluatedLayout.getValue(name); + } + + setLayoutProperty(name: string, value: any, options: StyleSetterOptions = {}) { + if (value !== null && value !== undefined) { + const key = `layers.${this.id}.layout.${name}`; + if (this._validate(validateLayoutProperty, key, name, value, options)) { + return; + } + } + + if (name === 'visibility') { + this.visibility = value; + return; + } + + this._unevaluatedLayout.setValue(name, value); + } + + getPaintProperty(name: string) { + if (name.endsWith(TRANSITION_SUFFIX)) { + return this._transitionablePaint.getTransition(name.slice(0, -TRANSITION_SUFFIX.length)); + } else { + return this._transitionablePaint.getValue(name); + } + } + + setPaintProperty(name: string, value: unknown, options: StyleSetterOptions = {}) { + if (value !== null && value !== undefined) { + const key = `layers.${this.id}.paint.${name}`; + if (this._validate(validatePaintProperty, key, name, value, options)) { + return false; + } + } + + if (name.endsWith(TRANSITION_SUFFIX)) { + this._transitionablePaint.setTransition(name.slice(0, -TRANSITION_SUFFIX.length), (value as any) || undefined); + return false; + } else { + const transitionable = this._transitionablePaint._values[name]; + const isCrossFadedProperty = transitionable.property.specification['property-type'] === 'cross-faded-data-driven'; + const wasDataDriven = transitionable.value.isDataDriven(); + const oldValue = transitionable.value; + + this._transitionablePaint.setValue(name, value); + this._handleSpecialPaintPropertyUpdate(name); + + const newValue = this._transitionablePaint._values[name].value; + const isDataDriven = newValue.isDataDriven(); + + // if a cross-faded value is changed, we need to make sure the new icons get added to each tile's iconAtlas + // so a call to _updateLayer is necessary, and we return true from this function so it gets called in + // Style#setPaintProperty + return isDataDriven || wasDataDriven || isCrossFadedProperty || this._handleOverridablePaintPropertyUpdate(name, oldValue, newValue); + } + } + + _handleSpecialPaintPropertyUpdate(_: string) { + // No-op; can be overridden by derived classes. + } + + // eslint-disable-next-line @typescript-eslint/no-unused-vars + _handleOverridablePaintPropertyUpdate(name: string, oldValue: PropertyValue, newValue: PropertyValue): boolean { + // No-op; can be overridden by derived classes. + return false; + } + + isHidden(zoom: number) { + if (this.minzoom && zoom < this.minzoom) return true; + if (this.maxzoom && zoom >= this.maxzoom) return true; + return this.visibility === 'none'; + } + + updateTransitions(parameters: TransitionParameters) { + this._transitioningPaint = this._transitionablePaint.transitioned(parameters, this._transitioningPaint); + } + + hasTransition() { + return this._transitioningPaint.hasTransition(); + } + + recalculate(parameters: EvaluationParameters, availableImages: Array) { + if (parameters.getCrossfadeParameters) { + this._crossfadeParameters = parameters.getCrossfadeParameters(); + } + + if (this._unevaluatedLayout) { + (this as any).layout = this._unevaluatedLayout.possiblyEvaluate(parameters, undefined, availableImages); + } + + (this as any).paint = this._transitioningPaint.possiblyEvaluate(parameters, undefined, availableImages); + } + + serialize(): LayerSpecification { + const output: LayerSpecification = { + 'id': this.id, + 'type': this.type as LayerSpecification['type'], + 'source': this.source, + 'source-layer': this.sourceLayer, + 'metadata': this.metadata, + 'minzoom': this.minzoom, + 'maxzoom': this.maxzoom, + 'filter': this.filter as FilterSpecification, + 'layout': this._unevaluatedLayout && this._unevaluatedLayout.serialize(), + 'paint': this._transitionablePaint && this._transitionablePaint.serialize() + }; + + if (this.visibility) { + output.layout = output.layout || {}; + output.layout.visibility = this.visibility; + } + + return filterObject(output, (value, key) => { + return value !== undefined && + !(key === 'layout' && !Object.keys(value).length) && + !(key === 'paint' && !Object.keys(value).length); + }); + } + + _validate(validate: Function, key: string, name: string, value: unknown, options: StyleSetterOptions = {}) { + if (options && options.validate === false) { + return false; + } + return emitValidationErrors(this, validate.call(validateStyle, { + key, + layerType: this.type, + objectKey: name, + value, + styleSpec, + // Workaround for https://github.com/mapbox/mapbox-gl-js/issues/2407 + style: {glyphs: true, sprite: true} + })); + } + + is3D() { + return false; + } + + isTileClipped() { + return false; + } + + hasOffscreenPass() { + return false; + } + + resize() { + // noop + } + + isStateDependent() { + for (const property in (this as any).paint._values) { + const value = (this as any).paint.get(property); + if (!(value instanceof PossiblyEvaluatedPropertyValue) || !supportsPropertyExpression(value.property.specification)) { + continue; + } + + if ((value.value.kind === 'source' || value.value.kind === 'composite') && + value.value.isStateDependent) { + return true; + } + } + return false; + } +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer/background_style_layer.ts b/web/libraries/maplibre-gl/src/style/style_layer/background_style_layer.ts new file mode 100644 index 00000000..412c04a8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/background_style_layer.ts @@ -0,0 +1,17 @@ +import {StyleLayer} from '../style_layer'; + +import properties, {BackgroundPaintPropsPossiblyEvaluated} from './background_style_layer_properties.g'; +import {Transitionable, Transitioning, PossiblyEvaluated} from '../properties'; + +import type {BackgroundPaintProps} from './background_style_layer_properties.g'; +import type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; + +export class BackgroundStyleLayer extends StyleLayer { + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + + constructor(layer: LayerSpecification) { + super(layer, properties); + } +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer/background_style_layer_properties.g.ts b/web/libraries/maplibre-gl/src/style/style_layer/background_style_layer_properties.g.ts new file mode 100644 index 00000000..e78a3b13 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/background_style_layer_properties.g.ts @@ -0,0 +1,40 @@ +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ + +import {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec'; + +import { + Properties, + DataConstantProperty, + DataDrivenProperty, + CrossFadedDataDrivenProperty, + CrossFadedProperty, + ColorRampProperty, + PossiblyEvaluatedPropertyValue, + CrossFaded +} from '../properties'; + +import type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec'; +import {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec'; + + +export type BackgroundPaintProps = { + "background-color": DataConstantProperty, + "background-pattern": CrossFadedProperty, + "background-opacity": DataConstantProperty, +}; + +export type BackgroundPaintPropsPossiblyEvaluated = { + "background-color": Color, + "background-pattern": CrossFaded, + "background-opacity": number, +}; + +let paint: Properties; +const getPaint = () => paint = paint || new Properties({ + "background-color": new DataConstantProperty(styleSpec["paint_background"]["background-color"] as any as StylePropertySpecification), + "background-pattern": new CrossFadedProperty(styleSpec["paint_background"]["background-pattern"] as any as StylePropertySpecification), + "background-opacity": new DataConstantProperty(styleSpec["paint_background"]["background-opacity"] as any as StylePropertySpecification), +}); + +export default ({ get paint() { return getPaint() } }); \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/style/style_layer/circle_style_layer.ts b/web/libraries/maplibre-gl/src/style/style_layer/circle_style_layer.ts new file mode 100644 index 00000000..2a7b9cfa --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/circle_style_layer.ts @@ -0,0 +1,98 @@ +import {StyleLayer} from '../style_layer'; + +import {CircleBucket} from '../../data/bucket/circle_bucket'; +import {polygonIntersectsBufferedPoint} from '../../util/intersection_tests'; +import {getMaximumPaintValue, translateDistance, translate} from '../query_utils'; +import properties, {CircleLayoutPropsPossiblyEvaluated, CirclePaintPropsPossiblyEvaluated} from './circle_style_layer_properties.g'; +import {Transitionable, Transitioning, Layout, PossiblyEvaluated} from '../properties'; +import {mat4, vec4} from 'gl-matrix'; +import Point from '@mapbox/point-geometry'; +import type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {Transform} from '../../geo/transform'; +import type {Bucket, BucketParameters} from '../../data/bucket'; +import type {CircleLayoutProps, CirclePaintProps} from './circle_style_layer_properties.g'; +import type {VectorTileFeature} from '@mapbox/vector-tile'; + +/** + * A style layer that defines a circle + */ +export class CircleStyleLayer extends StyleLayer { + _unevaluatedLayout: Layout; + layout: PossiblyEvaluated; + + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + + constructor(layer: LayerSpecification) { + super(layer, properties); + } + + createBucket(parameters: BucketParameters) { + return new CircleBucket(parameters); + } + + queryRadius(bucket: Bucket): number { + const circleBucket: CircleBucket = (bucket as any); + return getMaximumPaintValue('circle-radius', this, circleBucket) + + getMaximumPaintValue('circle-stroke-width', this, circleBucket) + + translateDistance(this.paint.get('circle-translate')); + } + + queryIntersectsFeature( + queryGeometry: Array, + feature: VectorTileFeature, + featureState: FeatureState, + geometry: Array>, + zoom: number, + transform: Transform, + pixelsToTileUnits: number, + pixelPosMatrix: mat4 + ): boolean { + const translatedPolygon = translate(queryGeometry, + this.paint.get('circle-translate'), + this.paint.get('circle-translate-anchor'), + transform.angle, pixelsToTileUnits); + const radius = this.paint.get('circle-radius').evaluate(feature, featureState); + const stroke = this.paint.get('circle-stroke-width').evaluate(feature, featureState); + const size = radius + stroke; + + // For pitch-alignment: map, compare feature geometry to query geometry in the plane of the tile + // // Otherwise, compare geometry in the plane of the viewport + // // A circle with fixed scaling relative to the viewport gets larger in tile space as it moves into the distance + // // A circle with fixed scaling relative to the map gets smaller in viewport space as it moves into the distance + const alignWithMap = this.paint.get('circle-pitch-alignment') === 'map'; + const transformedPolygon = alignWithMap ? translatedPolygon : projectQueryGeometry(translatedPolygon, pixelPosMatrix); + const transformedSize = alignWithMap ? size * pixelsToTileUnits : size; + + for (const ring of geometry) { + for (const point of ring) { + + const transformedPoint = alignWithMap ? point : projectPoint(point, pixelPosMatrix); + + let adjustedSize = transformedSize; + const projectedCenter = vec4.transformMat4([] as any, [point.x, point.y, 0, 1], pixelPosMatrix); + if (this.paint.get('circle-pitch-scale') === 'viewport' && this.paint.get('circle-pitch-alignment') === 'map') { + adjustedSize *= projectedCenter[3] / transform.cameraToCenterDistance; + } else if (this.paint.get('circle-pitch-scale') === 'map' && this.paint.get('circle-pitch-alignment') === 'viewport') { + adjustedSize *= transform.cameraToCenterDistance / projectedCenter[3]; + } + + if (polygonIntersectsBufferedPoint(transformedPolygon, transformedPoint, adjustedSize)) return true; + } + } + + return false; + } +} + +function projectPoint(p: Point, pixelPosMatrix: mat4) { + const point = vec4.transformMat4([] as any, [p.x, p.y, 0, 1], pixelPosMatrix); + return new Point(point[0] / point[3], point[1] / point[3]); +} + +function projectQueryGeometry(queryGeometry: Array, pixelPosMatrix: mat4) { + return queryGeometry.map((p) => { + return projectPoint(p, pixelPosMatrix); + }); +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer/circle_style_layer_properties.g.ts b/web/libraries/maplibre-gl/src/style/style_layer/circle_style_layer_properties.g.ts new file mode 100644 index 00000000..45a55cc8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/circle_style_layer_properties.g.ts @@ -0,0 +1,76 @@ +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ + +import {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec'; + +import { + Properties, + DataConstantProperty, + DataDrivenProperty, + CrossFadedDataDrivenProperty, + CrossFadedProperty, + ColorRampProperty, + PossiblyEvaluatedPropertyValue, + CrossFaded +} from '../properties'; + +import type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec'; +import {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec'; + +export type CircleLayoutProps = { + "circle-sort-key": DataDrivenProperty, +}; + +export type CircleLayoutPropsPossiblyEvaluated = { + "circle-sort-key": PossiblyEvaluatedPropertyValue, +}; + +let layout: Properties; +const getLayout = () => layout = layout || new Properties({ + "circle-sort-key": new DataDrivenProperty(styleSpec["layout_circle"]["circle-sort-key"] as any as StylePropertySpecification), +}); + +export type CirclePaintProps = { + "circle-radius": DataDrivenProperty, + "circle-color": DataDrivenProperty, + "circle-blur": DataDrivenProperty, + "circle-opacity": DataDrivenProperty, + "circle-translate": DataConstantProperty<[number, number]>, + "circle-translate-anchor": DataConstantProperty<"map" | "viewport">, + "circle-pitch-scale": DataConstantProperty<"map" | "viewport">, + "circle-pitch-alignment": DataConstantProperty<"map" | "viewport">, + "circle-stroke-width": DataDrivenProperty, + "circle-stroke-color": DataDrivenProperty, + "circle-stroke-opacity": DataDrivenProperty, +}; + +export type CirclePaintPropsPossiblyEvaluated = { + "circle-radius": PossiblyEvaluatedPropertyValue, + "circle-color": PossiblyEvaluatedPropertyValue, + "circle-blur": PossiblyEvaluatedPropertyValue, + "circle-opacity": PossiblyEvaluatedPropertyValue, + "circle-translate": [number, number], + "circle-translate-anchor": "map" | "viewport", + "circle-pitch-scale": "map" | "viewport", + "circle-pitch-alignment": "map" | "viewport", + "circle-stroke-width": PossiblyEvaluatedPropertyValue, + "circle-stroke-color": PossiblyEvaluatedPropertyValue, + "circle-stroke-opacity": PossiblyEvaluatedPropertyValue, +}; + +let paint: Properties; +const getPaint = () => paint = paint || new Properties({ + "circle-radius": new DataDrivenProperty(styleSpec["paint_circle"]["circle-radius"] as any as StylePropertySpecification), + "circle-color": new DataDrivenProperty(styleSpec["paint_circle"]["circle-color"] as any as StylePropertySpecification), + "circle-blur": new DataDrivenProperty(styleSpec["paint_circle"]["circle-blur"] as any as StylePropertySpecification), + "circle-opacity": new DataDrivenProperty(styleSpec["paint_circle"]["circle-opacity"] as any as StylePropertySpecification), + "circle-translate": new DataConstantProperty(styleSpec["paint_circle"]["circle-translate"] as any as StylePropertySpecification), + "circle-translate-anchor": new DataConstantProperty(styleSpec["paint_circle"]["circle-translate-anchor"] as any as StylePropertySpecification), + "circle-pitch-scale": new DataConstantProperty(styleSpec["paint_circle"]["circle-pitch-scale"] as any as StylePropertySpecification), + "circle-pitch-alignment": new DataConstantProperty(styleSpec["paint_circle"]["circle-pitch-alignment"] as any as StylePropertySpecification), + "circle-stroke-width": new DataDrivenProperty(styleSpec["paint_circle"]["circle-stroke-width"] as any as StylePropertySpecification), + "circle-stroke-color": new DataDrivenProperty(styleSpec["paint_circle"]["circle-stroke-color"] as any as StylePropertySpecification), + "circle-stroke-opacity": new DataDrivenProperty(styleSpec["paint_circle"]["circle-stroke-opacity"] as any as StylePropertySpecification), +}); + +export default ({ get paint() { return getPaint() }, get layout() { return getLayout() } }); \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/style/style_layer/custom_style_layer.ts b/web/libraries/maplibre-gl/src/style/style_layer/custom_style_layer.ts new file mode 100644 index 00000000..88a58b6b --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/custom_style_layer.ts @@ -0,0 +1,199 @@ +import {StyleLayer} from '../style_layer'; +import type {Map} from '../../ui/map'; +import {mat4} from 'gl-matrix'; +import {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; + +/** + * @param gl - The map's gl context. + * @param matrix - The map's camera matrix. It projects spherical mercator + * coordinates to gl coordinates. The spherical mercator coordinate `[0, 0]` represents the + * top left corner of the mercator world and `[1, 1]` represents the bottom right corner. When + * the `renderingMode` is `"3d"`, the z coordinate is conformal. A box with identical x, y, and z + * lengths in mercator units would be rendered as a cube. {@link MercatorCoordinate.fromLngLat} + * can be used to project a `LngLat` to a mercator coordinate. + */ +type CustomRenderMethod = (gl: WebGLRenderingContext|WebGL2RenderingContext, matrix: mat4) => void; + +/** + * Interface for custom style layers. This is a specification for + * implementers to model: it is not an exported method or class. + * + * Custom layers allow a user to render directly into the map's GL context using the map's camera. + * These layers can be added between any regular layers using {@link Map#addLayer}. + * + * Custom layers must have a unique `id` and must have the `type` of `"custom"`. + * They must implement `render` and may implement `prerender`, `onAdd` and `onRemove`. + * They can trigger rendering using {@link Map#triggerRepaint} + * and they should appropriately handle {@link MapContextEvent} with `webglcontextlost` and `webglcontextrestored`. + * + * The `renderingMode` property controls whether the layer is treated as a `"2d"` or `"3d"` map layer. Use: + * - `"renderingMode": "3d"` to use the depth buffer and share it with other layers + * - `"renderingMode": "2d"` to add a layer with no depth. If you need to use the depth buffer for a `"2d"` layer you must use an offscreen + * framebuffer and {@link CustomLayerInterface#prerender} + * + * @example + * Custom layer implemented as ES6 class + * ```ts + * class NullIslandLayer { + * constructor() { + * this.id = 'null-island'; + * this.type = 'custom'; + * this.renderingMode = '2d'; + * } + * + * onAdd(map, gl) { + * const vertexSource = ` + * uniform mat4 u_matrix; + * void main() { + * gl_Position = u_matrix * vec4(0.5, 0.5, 0.0, 1.0); + * gl_PointSize = 20.0; + * }`; + * + * const fragmentSource = ` + * void main() { + * fragColor = vec4(1.0, 0.0, 0.0, 1.0); + * }`; + * + * const vertexShader = gl.createShader(gl.VERTEX_SHADER); + * gl.shaderSource(vertexShader, vertexSource); + * gl.compileShader(vertexShader); + * const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER); + * gl.shaderSource(fragmentShader, fragmentSource); + * gl.compileShader(fragmentShader); + * + * this.program = gl.createProgram(); + * gl.attachShader(this.program, vertexShader); + * gl.attachShader(this.program, fragmentShader); + * gl.linkProgram(this.program); + * } + * + * render(gl, matrix) { + * gl.useProgram(this.program); + * gl.uniformMatrix4fv(gl.getUniformLocation(this.program, "u_matrix"), false, matrix); + * gl.drawArrays(gl.POINTS, 0, 1); + * } + * } + * + * map.on('load', function() { + * map.addLayer(new NullIslandLayer()); + * }); + * ``` + */ +export interface CustomLayerInterface { + /** + * A unique layer id. + */ + id: string; + /** + * The layer's type. Must be `"custom"`. + */ + type: 'custom'; + /** + * Either `"2d"` or `"3d"`. Defaults to `"2d"`. + */ + renderingMode?: '2d' | '3d'; + /** + * Called during a render frame allowing the layer to draw into the GL context. + * + * The layer can assume blending and depth state is set to allow the layer to properly + * blend and clip other layers. The layer cannot make any other assumptions about the + * current GL state. + * + * If the layer needs to render to a texture, it should implement the `prerender` method + * to do this and only use the `render` method for drawing directly into the main framebuffer. + * + * The blend function is set to `gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA)`. This expects + * colors to be provided in premultiplied alpha form where the `r`, `g` and `b` values are already + * multiplied by the `a` value. If you are unable to provide colors in premultiplied form you + * may want to change the blend function to + * `gl.blendFuncSeparate(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA, gl.ONE, gl.ONE_MINUS_SRC_ALPHA)`. + */ + render: CustomRenderMethod; + /** + * Optional method called during a render frame to allow a layer to prepare resources or render into a texture. + * + * The layer cannot make any assumptions about the current GL state and must bind a framebuffer before rendering. + */ + prerender?: CustomRenderMethod; + /** + * Optional method called when the layer has been added to the Map with {@link Map#addLayer}. This + * gives the layer a chance to initialize gl resources and register event listeners. + * + * @param map - The Map this custom layer was just added to. + * @param gl - The gl context for the map. + */ + onAdd?(map: Map, gl: WebGLRenderingContext | WebGL2RenderingContext): void; + /** + * Optional method called when the layer has been removed from the Map with {@link Map#removeLayer}. This + * gives the layer a chance to clean up gl resources and event listeners. + * + * @param map - The Map this custom layer was just added to. + * @param gl - The gl context for the map. + */ + onRemove?(map: Map, gl: WebGLRenderingContext | WebGL2RenderingContext): void; +} + +export function validateCustomStyleLayer(layerObject: CustomLayerInterface) { + const errors = []; + const id = layerObject.id; + + if (id === undefined) { + errors.push({ + message: `layers.${id}: missing required property "id"` + }); + } + + if (layerObject.render === undefined) { + errors.push({ + message: `layers.${id}: missing required method "render"` + }); + } + + if (layerObject.renderingMode && + layerObject.renderingMode !== '2d' && + layerObject.renderingMode !== '3d') { + errors.push({ + message: `layers.${id}: property "renderingMode" must be either "2d" or "3d"` + }); + } + + return errors; +} + +export class CustomStyleLayer extends StyleLayer { + + implementation: CustomLayerInterface; + + constructor(implementation: CustomLayerInterface) { + super(implementation, {}); + this.implementation = implementation; + } + + is3D() { + return this.implementation.renderingMode === '3d'; + } + + hasOffscreenPass() { + return this.implementation.prerender !== undefined; + } + + recalculate() {} + updateTransitions() {} + hasTransition() { return false; } + + serialize(): LayerSpecification { + throw new Error('Custom layers cannot be serialized'); + } + + onAdd = (map: Map) => { + if (this.implementation.onAdd) { + this.implementation.onAdd(map, map.painter.context.gl); + } + }; + + onRemove = (map: Map) => { + if (this.implementation.onRemove) { + this.implementation.onRemove(map, map.painter.context.gl); + } + }; +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer/fill_extrusion_style_layer.ts b/web/libraries/maplibre-gl/src/style/style_layer/fill_extrusion_style_layer.ts new file mode 100644 index 00000000..41b77128 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/fill_extrusion_style_layer.ts @@ -0,0 +1,224 @@ +import {StyleLayer} from '../style_layer'; + +import {FillExtrusionBucket} from '../../data/bucket/fill_extrusion_bucket'; +import {polygonIntersectsPolygon, polygonIntersectsMultiPolygon} from '../../util/intersection_tests'; +import {translateDistance, translate} from '../query_utils'; +import properties, {FillExtrusionPaintPropsPossiblyEvaluated} from './fill_extrusion_style_layer_properties.g'; +import {Transitionable, Transitioning, PossiblyEvaluated} from '../properties'; +import {mat4, vec4} from 'gl-matrix'; +import Point from '@mapbox/point-geometry'; +import type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {BucketParameters} from '../../data/bucket'; +import type {FillExtrusionPaintProps} from './fill_extrusion_style_layer_properties.g'; +import type {Transform} from '../../geo/transform'; +import type {VectorTileFeature} from '@mapbox/vector-tile'; + +export class Point3D extends Point { + z: number; +} + +export class FillExtrusionStyleLayer extends StyleLayer { + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + + constructor(layer: LayerSpecification) { + super(layer, properties); + } + + createBucket(parameters: BucketParameters) { + return new FillExtrusionBucket(parameters); + } + + queryRadius(): number { + return translateDistance(this.paint.get('fill-extrusion-translate')); + } + + is3D(): boolean { + return true; + } + + queryIntersectsFeature( + queryGeometry: Array, + feature: VectorTileFeature, + featureState: FeatureState, + geometry: Array>, + zoom: number, + transform: Transform, + pixelsToTileUnits: number, + pixelPosMatrix: mat4 + ): boolean | number { + + const translatedPolygon = translate(queryGeometry, + this.paint.get('fill-extrusion-translate'), + this.paint.get('fill-extrusion-translate-anchor'), + transform.angle, pixelsToTileUnits); + + const height = this.paint.get('fill-extrusion-height').evaluate(feature, featureState); + const base = this.paint.get('fill-extrusion-base').evaluate(feature, featureState); + + const projectedQueryGeometry = projectQueryGeometry(translatedPolygon, pixelPosMatrix, transform, 0); + + const projected = projectExtrusion(geometry, base, height, pixelPosMatrix); + const projectedBase = projected[0]; + const projectedTop = projected[1]; + return checkIntersection(projectedBase, projectedTop, projectedQueryGeometry); + } +} + +function dot(a, b) { + return a.x * b.x + a.y * b.y; +} + +export function getIntersectionDistance(projectedQueryGeometry: Array, projectedFace: Array) { + + if (projectedQueryGeometry.length === 1) { + // For point queries calculate the z at which the point intersects the face + // using barycentric coordinates. + + // Find the barycentric coordinates of the projected point within the first + // triangle of the face, using only the xy plane. It doesn't matter if the + // point is outside the first triangle because all the triangles in the face + // are in the same plane. + // + // Check whether points are coincident and use other points if they are. + let i = 0; + const a = projectedFace[i++]; + let b; + while (!b || a.equals(b)) { + b = projectedFace[i++]; + if (!b) return Infinity; + } + + // Loop until point `c` is not colinear with points `a` and `b`. + for (; i < projectedFace.length; i++) { + const c = projectedFace[i]; + + const p = projectedQueryGeometry[0]; + + const ab = b.sub(a); + const ac = c.sub(a); + const ap = p.sub(a); + + const dotABAB = dot(ab, ab); + const dotABAC = dot(ab, ac); + const dotACAC = dot(ac, ac); + const dotAPAB = dot(ap, ab); + const dotAPAC = dot(ap, ac); + const denom = dotABAB * dotACAC - dotABAC * dotABAC; + + const v = (dotACAC * dotAPAB - dotABAC * dotAPAC) / denom; + const w = (dotABAB * dotAPAC - dotABAC * dotAPAB) / denom; + const u = 1 - v - w; + + // Use the barycentric weighting along with the original triangle z coordinates to get the point of intersection. + const distance = a.z * u + b.z * v + c.z * w; + + if (isFinite(distance)) return distance; + } + + return Infinity; + + } else { + // The counts as closest is less clear when the query is a box. This + // returns the distance to the nearest point on the face, whether it is + // within the query or not. It could be more correct to return the + // distance to the closest point within the query box but this would be + // more complicated and expensive to calculate with little benefit. + let closestDistance = Infinity; + for (const p of projectedFace) { + closestDistance = Math.min(closestDistance, p.z); + } + return closestDistance; + } +} + +function checkIntersection(projectedBase: Array>, projectedTop: Array>, projectedQueryGeometry: Array) { + let closestDistance = Infinity; + + if (polygonIntersectsMultiPolygon(projectedQueryGeometry, projectedTop)) { + closestDistance = getIntersectionDistance(projectedQueryGeometry, projectedTop[0]); + } + + for (let r = 0; r < projectedTop.length; r++) { + const ringTop = projectedTop[r]; + const ringBase = projectedBase[r]; + for (let p = 0; p < ringTop.length - 1; p++) { + const topA = ringTop[p]; + const topB = ringTop[p + 1]; + const baseA = ringBase[p]; + const baseB = ringBase[p + 1]; + const face = [topA, topB, baseB, baseA, topA]; + if (polygonIntersectsPolygon(projectedQueryGeometry, face)) { + closestDistance = Math.min(closestDistance, getIntersectionDistance(projectedQueryGeometry, face)); + } + } + } + + return closestDistance === Infinity ? false : closestDistance; +} + +/* + * Project the geometry using matrix `m`. This is essentially doing + * `vec4.transformMat4([], [p.x, p.y, z, 1], m)` but the multiplication + * is inlined so that parts of the projection that are the same across + * different points can only be done once. This produced a measurable + * performance improvement. + */ +function projectExtrusion(geometry: Array>, zBase: number, zTop: number, m: mat4): [Array>, Array>] { + const projectedBase = [] as Array>; + const projectedTop = [] as Array>; + const baseXZ = m[8] * zBase; + const baseYZ = m[9] * zBase; + const baseZZ = m[10] * zBase; + const baseWZ = m[11] * zBase; + const topXZ = m[8] * zTop; + const topYZ = m[9] * zTop; + const topZZ = m[10] * zTop; + const topWZ = m[11] * zTop; + + for (const r of geometry) { + const ringBase = [] as Array; + const ringTop = [] as Array; + for (const p of r) { + const x = p.x; + const y = p.y; + + const sX = m[0] * x + m[4] * y + m[12]; + const sY = m[1] * x + m[5] * y + m[13]; + const sZ = m[2] * x + m[6] * y + m[14]; + const sW = m[3] * x + m[7] * y + m[15]; + + const baseX = sX + baseXZ; + const baseY = sY + baseYZ; + const baseZ = sZ + baseZZ; + const baseW = sW + baseWZ; + + const topX = sX + topXZ; + const topY = sY + topYZ; + const topZ = sZ + topZZ; + const topW = sW + topWZ; + + const b = new Point(baseX / baseW, baseY / baseW) as Point3D; + b.z = baseZ / baseW; + ringBase.push(b); + + const t = new Point(topX / topW, topY / topW) as Point3D; + t.z = topZ / topW; + ringTop.push(t); + } + projectedBase.push(ringBase); + projectedTop.push(ringTop); + } + return [projectedBase, projectedTop]; +} + +function projectQueryGeometry(queryGeometry: Array, pixelPosMatrix: mat4, transform: Transform, z: number) { + const projectedQueryGeometry = []; + for (const p of queryGeometry) { + const v = [p.x, p.y, z, 1] as vec4; + vec4.transformMat4(v, v, pixelPosMatrix); + projectedQueryGeometry.push(new Point(v[0] / v[3], v[1] / v[3])); + } + return projectedQueryGeometry; +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer/fill_extrusion_style_layer_properties.g.ts b/web/libraries/maplibre-gl/src/style/style_layer/fill_extrusion_style_layer_properties.g.ts new file mode 100644 index 00000000..e36189d1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/fill_extrusion_style_layer_properties.g.ts @@ -0,0 +1,55 @@ +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ + +import {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec'; + +import { + Properties, + DataConstantProperty, + DataDrivenProperty, + CrossFadedDataDrivenProperty, + CrossFadedProperty, + ColorRampProperty, + PossiblyEvaluatedPropertyValue, + CrossFaded +} from '../properties'; + +import type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec'; +import {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec'; + + +export type FillExtrusionPaintProps = { + "fill-extrusion-opacity": DataConstantProperty, + "fill-extrusion-color": DataDrivenProperty, + "fill-extrusion-translate": DataConstantProperty<[number, number]>, + "fill-extrusion-translate-anchor": DataConstantProperty<"map" | "viewport">, + "fill-extrusion-pattern": CrossFadedDataDrivenProperty, + "fill-extrusion-height": DataDrivenProperty, + "fill-extrusion-base": DataDrivenProperty, + "fill-extrusion-vertical-gradient": DataConstantProperty, +}; + +export type FillExtrusionPaintPropsPossiblyEvaluated = { + "fill-extrusion-opacity": number, + "fill-extrusion-color": PossiblyEvaluatedPropertyValue, + "fill-extrusion-translate": [number, number], + "fill-extrusion-translate-anchor": "map" | "viewport", + "fill-extrusion-pattern": PossiblyEvaluatedPropertyValue>, + "fill-extrusion-height": PossiblyEvaluatedPropertyValue, + "fill-extrusion-base": PossiblyEvaluatedPropertyValue, + "fill-extrusion-vertical-gradient": boolean, +}; + +let paint: Properties; +const getPaint = () => paint = paint || new Properties({ + "fill-extrusion-opacity": new DataConstantProperty(styleSpec["paint_fill-extrusion"]["fill-extrusion-opacity"] as any as StylePropertySpecification), + "fill-extrusion-color": new DataDrivenProperty(styleSpec["paint_fill-extrusion"]["fill-extrusion-color"] as any as StylePropertySpecification), + "fill-extrusion-translate": new DataConstantProperty(styleSpec["paint_fill-extrusion"]["fill-extrusion-translate"] as any as StylePropertySpecification), + "fill-extrusion-translate-anchor": new DataConstantProperty(styleSpec["paint_fill-extrusion"]["fill-extrusion-translate-anchor"] as any as StylePropertySpecification), + "fill-extrusion-pattern": new CrossFadedDataDrivenProperty(styleSpec["paint_fill-extrusion"]["fill-extrusion-pattern"] as any as StylePropertySpecification), + "fill-extrusion-height": new DataDrivenProperty(styleSpec["paint_fill-extrusion"]["fill-extrusion-height"] as any as StylePropertySpecification), + "fill-extrusion-base": new DataDrivenProperty(styleSpec["paint_fill-extrusion"]["fill-extrusion-base"] as any as StylePropertySpecification), + "fill-extrusion-vertical-gradient": new DataConstantProperty(styleSpec["paint_fill-extrusion"]["fill-extrusion-vertical-gradient"] as any as StylePropertySpecification), +}); + +export default ({ get paint() { return getPaint() } }); \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/style/style_layer/fill_style_layer.test.ts b/web/libraries/maplibre-gl/src/style/style_layer/fill_style_layer.test.ts new file mode 100644 index 00000000..3500f500 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/fill_style_layer.test.ts @@ -0,0 +1,37 @@ +import {getIntersectionDistance, Point3D} from './fill_extrusion_style_layer'; + +describe('getIntersectionDistance', () => { + const queryPoint = [new Point3D(100, 100)]; + const z = 3; + const a = new Point3D(100, -90); + const b = new Point3D(110, 110); + const c = new Point3D(-110, 110); + a.z = z; + b.z = z; + c.z = z; + + test('one point', () => { + const projectedFace = [a, a]; + expect(getIntersectionDistance(queryPoint, projectedFace)).toBe(Infinity); + }); + + test('two points', () => { + const projectedFace = [a, b, a]; + expect(getIntersectionDistance(queryPoint, projectedFace)).toBe(Infinity); + }); + + test('two points coincident', () => { + const projectedFace = [a, a, a, b, b, b, b, a]; + expect(getIntersectionDistance(queryPoint, projectedFace)).toBe(Infinity); + }); + + test('three points', () => { + const projectedFace = [a, b, c, a]; + expect(getIntersectionDistance(queryPoint, projectedFace)).toBe(z); + }); + + test('three points coincident points', () => { + const projectedFace = [a, a, b, b, b, c, c, a]; + expect(getIntersectionDistance(queryPoint, projectedFace)).toBe(z); + }); +}); diff --git a/web/libraries/maplibre-gl/src/style/style_layer/fill_style_layer.ts b/web/libraries/maplibre-gl/src/style/style_layer/fill_style_layer.ts new file mode 100644 index 00000000..0f423433 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/fill_style_layer.ts @@ -0,0 +1,65 @@ +import {StyleLayer} from '../style_layer'; + +import {FillBucket} from '../../data/bucket/fill_bucket'; +import {polygonIntersectsMultiPolygon} from '../../util/intersection_tests'; +import {translateDistance, translate} from '../query_utils'; +import properties, {FillLayoutPropsPossiblyEvaluated, FillPaintPropsPossiblyEvaluated} from './fill_style_layer_properties.g'; +import {Transitionable, Transitioning, Layout, PossiblyEvaluated} from '../properties'; + +import type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {BucketParameters} from '../../data/bucket'; +import type Point from '@mapbox/point-geometry'; +import type {FillLayoutProps, FillPaintProps} from './fill_style_layer_properties.g'; +import type {EvaluationParameters} from '../evaluation_parameters'; +import type {Transform} from '../../geo/transform'; +import type {VectorTileFeature} from '@mapbox/vector-tile'; + +export class FillStyleLayer extends StyleLayer { + _unevaluatedLayout: Layout; + layout: PossiblyEvaluated; + + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + + constructor(layer: LayerSpecification) { + super(layer, properties); + } + + recalculate(parameters: EvaluationParameters, availableImages: Array) { + super.recalculate(parameters, availableImages); + + const outlineColor = this.paint._values['fill-outline-color']; + if (outlineColor.value.kind === 'constant' && outlineColor.value.value === undefined) { + this.paint._values['fill-outline-color'] = this.paint._values['fill-color']; + } + } + + createBucket(parameters: BucketParameters) { + return new FillBucket(parameters); + } + + queryRadius(): number { + return translateDistance(this.paint.get('fill-translate')); + } + + queryIntersectsFeature( + queryGeometry: Array, + feature: VectorTileFeature, + featureState: FeatureState, + geometry: Array>, + zoom: number, + transform: Transform, + pixelsToTileUnits: number + ): boolean { + const translatedPolygon = translate(queryGeometry, + this.paint.get('fill-translate'), + this.paint.get('fill-translate-anchor'), + transform.angle, pixelsToTileUnits); + return polygonIntersectsMultiPolygon(translatedPolygon, geometry); + } + + isTileClipped() { + return true; + } +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer/fill_style_layer_properties.g.ts b/web/libraries/maplibre-gl/src/style/style_layer/fill_style_layer_properties.g.ts new file mode 100644 index 00000000..8115e9ef --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/fill_style_layer_properties.g.ts @@ -0,0 +1,64 @@ +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ + +import {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec'; + +import { + Properties, + DataConstantProperty, + DataDrivenProperty, + CrossFadedDataDrivenProperty, + CrossFadedProperty, + ColorRampProperty, + PossiblyEvaluatedPropertyValue, + CrossFaded +} from '../properties'; + +import type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec'; +import {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec'; + +export type FillLayoutProps = { + "fill-sort-key": DataDrivenProperty, +}; + +export type FillLayoutPropsPossiblyEvaluated = { + "fill-sort-key": PossiblyEvaluatedPropertyValue, +}; + +let layout: Properties; +const getLayout = () => layout = layout || new Properties({ + "fill-sort-key": new DataDrivenProperty(styleSpec["layout_fill"]["fill-sort-key"] as any as StylePropertySpecification), +}); + +export type FillPaintProps = { + "fill-antialias": DataConstantProperty, + "fill-opacity": DataDrivenProperty, + "fill-color": DataDrivenProperty, + "fill-outline-color": DataDrivenProperty, + "fill-translate": DataConstantProperty<[number, number]>, + "fill-translate-anchor": DataConstantProperty<"map" | "viewport">, + "fill-pattern": CrossFadedDataDrivenProperty, +}; + +export type FillPaintPropsPossiblyEvaluated = { + "fill-antialias": boolean, + "fill-opacity": PossiblyEvaluatedPropertyValue, + "fill-color": PossiblyEvaluatedPropertyValue, + "fill-outline-color": PossiblyEvaluatedPropertyValue, + "fill-translate": [number, number], + "fill-translate-anchor": "map" | "viewport", + "fill-pattern": PossiblyEvaluatedPropertyValue>, +}; + +let paint: Properties; +const getPaint = () => paint = paint || new Properties({ + "fill-antialias": new DataConstantProperty(styleSpec["paint_fill"]["fill-antialias"] as any as StylePropertySpecification), + "fill-opacity": new DataDrivenProperty(styleSpec["paint_fill"]["fill-opacity"] as any as StylePropertySpecification), + "fill-color": new DataDrivenProperty(styleSpec["paint_fill"]["fill-color"] as any as StylePropertySpecification), + "fill-outline-color": new DataDrivenProperty(styleSpec["paint_fill"]["fill-outline-color"] as any as StylePropertySpecification), + "fill-translate": new DataConstantProperty(styleSpec["paint_fill"]["fill-translate"] as any as StylePropertySpecification), + "fill-translate-anchor": new DataConstantProperty(styleSpec["paint_fill"]["fill-translate-anchor"] as any as StylePropertySpecification), + "fill-pattern": new CrossFadedDataDrivenProperty(styleSpec["paint_fill"]["fill-pattern"] as any as StylePropertySpecification), +}); + +export default ({ get paint() { return getPaint() }, get layout() { return getLayout() } }); \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/style/style_layer/heatmap_style_layer.ts b/web/libraries/maplibre-gl/src/style/style_layer/heatmap_style_layer.ts new file mode 100644 index 00000000..b36f0cb9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/heatmap_style_layer.ts @@ -0,0 +1,72 @@ +import {StyleLayer} from '../style_layer'; + +import {HeatmapBucket} from '../../data/bucket/heatmap_bucket'; +import {RGBAImage} from '../../util/image'; +import properties, {HeatmapPaintPropsPossiblyEvaluated} from './heatmap_style_layer_properties.g'; +import {renderColorRamp} from '../../util/color_ramp'; +import {Transitionable, Transitioning, PossiblyEvaluated} from '../properties'; + +import type {Texture} from '../../render/texture'; +import type {Framebuffer} from '../../gl/framebuffer'; +import type {HeatmapPaintProps} from './heatmap_style_layer_properties.g'; +import type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; + +/** + * A style layer that defines a heatmap + */ +export class HeatmapStyleLayer extends StyleLayer { + + heatmapFbo: Framebuffer; + colorRamp: RGBAImage; + colorRampTexture: Texture; + + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + + createBucket(options: any) { + return new HeatmapBucket(options); + } + + constructor(layer: LayerSpecification) { + super(layer, properties); + + // make sure color ramp texture is generated for default heatmap color too + this._updateColorRamp(); + } + + _handleSpecialPaintPropertyUpdate(name: string) { + if (name === 'heatmap-color') { + this._updateColorRamp(); + } + } + + _updateColorRamp() { + const expression = this._transitionablePaint._values['heatmap-color'].value.expression; + this.colorRamp = renderColorRamp({ + expression, + evaluationKey: 'heatmapDensity', + image: this.colorRamp + }); + this.colorRampTexture = null; + } + + resize() { + if (this.heatmapFbo) { + this.heatmapFbo.destroy(); + this.heatmapFbo = null; + } + } + + queryRadius(): number { + return 0; + } + + queryIntersectsFeature(): boolean { + return false; + } + + hasOffscreenPass() { + return this.paint.get('heatmap-opacity') !== 0 && this.visibility !== 'none'; + } +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer/heatmap_style_layer_properties.g.ts b/web/libraries/maplibre-gl/src/style/style_layer/heatmap_style_layer_properties.g.ts new file mode 100644 index 00000000..e48b5719 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/heatmap_style_layer_properties.g.ts @@ -0,0 +1,46 @@ +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ + +import {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec'; + +import { + Properties, + DataConstantProperty, + DataDrivenProperty, + CrossFadedDataDrivenProperty, + CrossFadedProperty, + ColorRampProperty, + PossiblyEvaluatedPropertyValue, + CrossFaded +} from '../properties'; + +import type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec'; +import {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec'; + + +export type HeatmapPaintProps = { + "heatmap-radius": DataDrivenProperty, + "heatmap-weight": DataDrivenProperty, + "heatmap-intensity": DataConstantProperty, + "heatmap-color": ColorRampProperty, + "heatmap-opacity": DataConstantProperty, +}; + +export type HeatmapPaintPropsPossiblyEvaluated = { + "heatmap-radius": PossiblyEvaluatedPropertyValue, + "heatmap-weight": PossiblyEvaluatedPropertyValue, + "heatmap-intensity": number, + "heatmap-color": ColorRampProperty, + "heatmap-opacity": number, +}; + +let paint: Properties; +const getPaint = () => paint = paint || new Properties({ + "heatmap-radius": new DataDrivenProperty(styleSpec["paint_heatmap"]["heatmap-radius"] as any as StylePropertySpecification), + "heatmap-weight": new DataDrivenProperty(styleSpec["paint_heatmap"]["heatmap-weight"] as any as StylePropertySpecification), + "heatmap-intensity": new DataConstantProperty(styleSpec["paint_heatmap"]["heatmap-intensity"] as any as StylePropertySpecification), + "heatmap-color": new ColorRampProperty(styleSpec["paint_heatmap"]["heatmap-color"] as any as StylePropertySpecification), + "heatmap-opacity": new DataConstantProperty(styleSpec["paint_heatmap"]["heatmap-opacity"] as any as StylePropertySpecification), +}); + +export default ({ get paint() { return getPaint() } }); \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/style/style_layer/hillshade_style_layer.ts b/web/libraries/maplibre-gl/src/style/style_layer/hillshade_style_layer.ts new file mode 100644 index 00000000..43e84133 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/hillshade_style_layer.ts @@ -0,0 +1,21 @@ +import {StyleLayer} from '../style_layer'; + +import properties, {HillshadePaintPropsPossiblyEvaluated} from './hillshade_style_layer_properties.g'; +import {Transitionable, Transitioning, PossiblyEvaluated} from '../properties'; + +import type {HillshadePaintProps} from './hillshade_style_layer_properties.g'; +import type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; + +export class HillshadeStyleLayer extends StyleLayer { + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + + constructor(layer: LayerSpecification) { + super(layer, properties); + } + + hasOffscreenPass() { + return this.paint.get('hillshade-exaggeration') !== 0 && this.visibility !== 'none'; + } +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer/hillshade_style_layer_properties.g.ts b/web/libraries/maplibre-gl/src/style/style_layer/hillshade_style_layer_properties.g.ts new file mode 100644 index 00000000..cf460762 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/hillshade_style_layer_properties.g.ts @@ -0,0 +1,49 @@ +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ + +import {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec'; + +import { + Properties, + DataConstantProperty, + DataDrivenProperty, + CrossFadedDataDrivenProperty, + CrossFadedProperty, + ColorRampProperty, + PossiblyEvaluatedPropertyValue, + CrossFaded +} from '../properties'; + +import type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec'; +import {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec'; + + +export type HillshadePaintProps = { + "hillshade-illumination-direction": DataConstantProperty, + "hillshade-illumination-anchor": DataConstantProperty<"map" | "viewport">, + "hillshade-exaggeration": DataConstantProperty, + "hillshade-shadow-color": DataConstantProperty, + "hillshade-highlight-color": DataConstantProperty, + "hillshade-accent-color": DataConstantProperty, +}; + +export type HillshadePaintPropsPossiblyEvaluated = { + "hillshade-illumination-direction": number, + "hillshade-illumination-anchor": "map" | "viewport", + "hillshade-exaggeration": number, + "hillshade-shadow-color": Color, + "hillshade-highlight-color": Color, + "hillshade-accent-color": Color, +}; + +let paint: Properties; +const getPaint = () => paint = paint || new Properties({ + "hillshade-illumination-direction": new DataConstantProperty(styleSpec["paint_hillshade"]["hillshade-illumination-direction"] as any as StylePropertySpecification), + "hillshade-illumination-anchor": new DataConstantProperty(styleSpec["paint_hillshade"]["hillshade-illumination-anchor"] as any as StylePropertySpecification), + "hillshade-exaggeration": new DataConstantProperty(styleSpec["paint_hillshade"]["hillshade-exaggeration"] as any as StylePropertySpecification), + "hillshade-shadow-color": new DataConstantProperty(styleSpec["paint_hillshade"]["hillshade-shadow-color"] as any as StylePropertySpecification), + "hillshade-highlight-color": new DataConstantProperty(styleSpec["paint_hillshade"]["hillshade-highlight-color"] as any as StylePropertySpecification), + "hillshade-accent-color": new DataConstantProperty(styleSpec["paint_hillshade"]["hillshade-accent-color"] as any as StylePropertySpecification), +}); + +export default ({ get paint() { return getPaint() } }); \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/style/style_layer/line_style_layer.test.ts b/web/libraries/maplibre-gl/src/style/style_layer/line_style_layer.test.ts new file mode 100644 index 00000000..4ad575b9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/line_style_layer.test.ts @@ -0,0 +1,50 @@ +import {createStyleLayer} from '../create_style_layer'; +import {extend} from '../../util/util'; +import {LineStyleLayer} from './line_style_layer'; + +describe('LineStyleLayer', () => { + function createLineLayer(layer?) { + return extend({ + type: 'line', + source: 'line', + id: 'line', + paint: { + 'line-color': 'red', + 'line-width': 14, + 'line-gradient': [ + 'interpolate', + ['linear'], + ['line-progress'], + 0, + 'blue', + 1, + 'red' + ] + } + }, layer); + } + + test('updating with valid line-gradient updates this.gradientVersion', () => { + const lineLayer = createStyleLayer(createLineLayer()) as LineStyleLayer; + const gradientVersion = lineLayer.gradientVersion; + + lineLayer.setPaintProperty('line-gradient', [ + 'interpolate', + ['linear'], + ['line-progress'], + 0, + 'red', + 1, + 'blue' + ]); + expect(lineLayer.gradientVersion).toBeGreaterThan(gradientVersion); + }); + + test('updating with invalid line-gradient updates this.gradientVersion', () => { + const lineLayer = createStyleLayer(createLineLayer()) as LineStyleLayer; + const gradientVersion = lineLayer.gradientVersion; + + lineLayer.setPaintProperty('line-gradient', null); + expect(lineLayer.gradientVersion).toBeGreaterThan(gradientVersion); + }); +}); diff --git a/web/libraries/maplibre-gl/src/style/style_layer/line_style_layer.ts b/web/libraries/maplibre-gl/src/style/style_layer/line_style_layer.ts new file mode 100644 index 00000000..d13c77b8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/line_style_layer.ts @@ -0,0 +1,131 @@ +import Point from '@mapbox/point-geometry'; + +import {StyleLayer} from '../style_layer'; +import {LineBucket} from '../../data/bucket/line_bucket'; +import {polygonIntersectsBufferedMultiLine} from '../../util/intersection_tests'; +import {getMaximumPaintValue, translateDistance, translate, offsetLine} from '../query_utils'; +import properties, {LineLayoutPropsPossiblyEvaluated, LinePaintPropsPossiblyEvaluated} from './line_style_layer_properties.g'; +import {extend} from '../../util/util'; +import {EvaluationParameters} from '../evaluation_parameters'; +import {Transitionable, Transitioning, Layout, PossiblyEvaluated, DataDrivenProperty} from '../properties'; + +import {isZoomExpression, Step} from '@maplibre/maplibre-gl-style-spec'; +import type {FeatureState, LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {Bucket, BucketParameters} from '../../data/bucket'; +import type {LineLayoutProps, LinePaintProps} from './line_style_layer_properties.g'; +import type {Transform} from '../../geo/transform'; +import type {VectorTileFeature} from '@mapbox/vector-tile'; + +export class LineFloorwidthProperty extends DataDrivenProperty { + useIntegerZoom: true; + + possiblyEvaluate(value, parameters) { + parameters = new EvaluationParameters(Math.floor(parameters.zoom), { + now: parameters.now, + fadeDuration: parameters.fadeDuration, + zoomHistory: parameters.zoomHistory, + transition: parameters.transition + }); + return super.possiblyEvaluate(value, parameters); + } + + evaluate(value, globals, feature, featureState) { + globals = extend({}, globals, {zoom: Math.floor(globals.zoom)}); + return super.evaluate(value, globals, feature, featureState); + } +} + +let lineFloorwidthProperty: LineFloorwidthProperty; + +export class LineStyleLayer extends StyleLayer { + _unevaluatedLayout: Layout; + layout: PossiblyEvaluated; + + gradientVersion: number; + stepInterpolant: boolean; + + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + + constructor(layer: LayerSpecification) { + super(layer, properties); + this.gradientVersion = 0; + if (!lineFloorwidthProperty) { + lineFloorwidthProperty = + new LineFloorwidthProperty(properties.paint.properties['line-width'].specification); + lineFloorwidthProperty.useIntegerZoom = true; + } + } + + _handleSpecialPaintPropertyUpdate(name: string) { + if (name === 'line-gradient') { + const expression = this.gradientExpression(); + if (isZoomExpression(expression)) { + this.stepInterpolant = expression._styleExpression.expression instanceof Step; + } else { + this.stepInterpolant = false; + } + this.gradientVersion = (this.gradientVersion + 1) % Number.MAX_SAFE_INTEGER; + } + } + + gradientExpression() { + return this._transitionablePaint._values['line-gradient'].value.expression; + } + + recalculate(parameters: EvaluationParameters, availableImages: Array) { + super.recalculate(parameters, availableImages); + (this.paint._values as any)['line-floorwidth'] = + lineFloorwidthProperty.possiblyEvaluate(this._transitioningPaint._values['line-width'].value, parameters); + } + + createBucket(parameters: BucketParameters) { + return new LineBucket(parameters); + } + + queryRadius(bucket: Bucket): number { + const lineBucket: LineBucket = (bucket as any); + const width = getLineWidth( + getMaximumPaintValue('line-width', this, lineBucket), + getMaximumPaintValue('line-gap-width', this, lineBucket)); + const offset = getMaximumPaintValue('line-offset', this, lineBucket); + return width / 2 + Math.abs(offset) + translateDistance(this.paint.get('line-translate')); + } + + queryIntersectsFeature( + queryGeometry: Array, + feature: VectorTileFeature, + featureState: FeatureState, + geometry: Array>, + zoom: number, + transform: Transform, + pixelsToTileUnits: number + ): boolean { + const translatedPolygon = translate(queryGeometry, + this.paint.get('line-translate'), + this.paint.get('line-translate-anchor'), + transform.angle, pixelsToTileUnits); + const halfWidth = pixelsToTileUnits / 2 * getLineWidth( + this.paint.get('line-width').evaluate(feature, featureState), + this.paint.get('line-gap-width').evaluate(feature, featureState)); + const lineOffset = this.paint.get('line-offset').evaluate(feature, featureState); + if (lineOffset) { + geometry = offsetLine(geometry, lineOffset * pixelsToTileUnits); + } + + return polygonIntersectsBufferedMultiLine(translatedPolygon, geometry, halfWidth); + } + + isTileClipped() { + return true; + } +} + +function getLineWidth(lineWidth, lineGapWidth) { + if (lineGapWidth > 0) { + return lineGapWidth + 2 * lineWidth; + } else { + return lineWidth; + } +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer/line_style_layer_properties.g.ts b/web/libraries/maplibre-gl/src/style/style_layer/line_style_layer_properties.g.ts new file mode 100644 index 00000000..1b6d8d93 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/line_style_layer_properties.g.ts @@ -0,0 +1,88 @@ +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ + +import {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec'; + +import { + Properties, + DataConstantProperty, + DataDrivenProperty, + CrossFadedDataDrivenProperty, + CrossFadedProperty, + ColorRampProperty, + PossiblyEvaluatedPropertyValue, + CrossFaded +} from '../properties'; + +import type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec'; +import {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec'; + +export type LineLayoutProps = { + "line-cap": DataConstantProperty<"butt" | "round" | "square">, + "line-join": DataDrivenProperty<"bevel" | "round" | "miter">, + "line-miter-limit": DataConstantProperty, + "line-round-limit": DataConstantProperty, + "line-sort-key": DataDrivenProperty, +}; + +export type LineLayoutPropsPossiblyEvaluated = { + "line-cap": "butt" | "round" | "square", + "line-join": PossiblyEvaluatedPropertyValue<"bevel" | "round" | "miter">, + "line-miter-limit": number, + "line-round-limit": number, + "line-sort-key": PossiblyEvaluatedPropertyValue, +}; + +let layout: Properties; +const getLayout = () => layout = layout || new Properties({ + "line-cap": new DataConstantProperty(styleSpec["layout_line"]["line-cap"] as any as StylePropertySpecification), + "line-join": new DataDrivenProperty(styleSpec["layout_line"]["line-join"] as any as StylePropertySpecification), + "line-miter-limit": new DataConstantProperty(styleSpec["layout_line"]["line-miter-limit"] as any as StylePropertySpecification), + "line-round-limit": new DataConstantProperty(styleSpec["layout_line"]["line-round-limit"] as any as StylePropertySpecification), + "line-sort-key": new DataDrivenProperty(styleSpec["layout_line"]["line-sort-key"] as any as StylePropertySpecification), +}); + +export type LinePaintProps = { + "line-opacity": DataDrivenProperty, + "line-color": DataDrivenProperty, + "line-translate": DataConstantProperty<[number, number]>, + "line-translate-anchor": DataConstantProperty<"map" | "viewport">, + "line-width": DataDrivenProperty, + "line-gap-width": DataDrivenProperty, + "line-offset": DataDrivenProperty, + "line-blur": DataDrivenProperty, + "line-dasharray": CrossFadedProperty>, + "line-pattern": CrossFadedDataDrivenProperty, + "line-gradient": ColorRampProperty, +}; + +export type LinePaintPropsPossiblyEvaluated = { + "line-opacity": PossiblyEvaluatedPropertyValue, + "line-color": PossiblyEvaluatedPropertyValue, + "line-translate": [number, number], + "line-translate-anchor": "map" | "viewport", + "line-width": PossiblyEvaluatedPropertyValue, + "line-gap-width": PossiblyEvaluatedPropertyValue, + "line-offset": PossiblyEvaluatedPropertyValue, + "line-blur": PossiblyEvaluatedPropertyValue, + "line-dasharray": CrossFaded>, + "line-pattern": PossiblyEvaluatedPropertyValue>, + "line-gradient": ColorRampProperty, +}; + +let paint: Properties; +const getPaint = () => paint = paint || new Properties({ + "line-opacity": new DataDrivenProperty(styleSpec["paint_line"]["line-opacity"] as any as StylePropertySpecification), + "line-color": new DataDrivenProperty(styleSpec["paint_line"]["line-color"] as any as StylePropertySpecification), + "line-translate": new DataConstantProperty(styleSpec["paint_line"]["line-translate"] as any as StylePropertySpecification), + "line-translate-anchor": new DataConstantProperty(styleSpec["paint_line"]["line-translate-anchor"] as any as StylePropertySpecification), + "line-width": new DataDrivenProperty(styleSpec["paint_line"]["line-width"] as any as StylePropertySpecification), + "line-gap-width": new DataDrivenProperty(styleSpec["paint_line"]["line-gap-width"] as any as StylePropertySpecification), + "line-offset": new DataDrivenProperty(styleSpec["paint_line"]["line-offset"] as any as StylePropertySpecification), + "line-blur": new DataDrivenProperty(styleSpec["paint_line"]["line-blur"] as any as StylePropertySpecification), + "line-dasharray": new CrossFadedProperty(styleSpec["paint_line"]["line-dasharray"] as any as StylePropertySpecification), + "line-pattern": new CrossFadedDataDrivenProperty(styleSpec["paint_line"]["line-pattern"] as any as StylePropertySpecification), + "line-gradient": new ColorRampProperty(styleSpec["paint_line"]["line-gradient"] as any as StylePropertySpecification), +}); + +export default ({ get paint() { return getPaint() }, get layout() { return getLayout() } }); \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/style/style_layer/overlap_mode.test.ts b/web/libraries/maplibre-gl/src/style/style_layer/overlap_mode.test.ts new file mode 100644 index 00000000..6c984b35 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/overlap_mode.test.ts @@ -0,0 +1,57 @@ +import {getOverlapMode} from './overlap_mode'; +import {SymbolStyleLayer} from './symbol_style_layer'; +import {ZoomHistory} from '../zoom_history'; +import {EvaluationParameters} from '../evaluation_parameters'; + +function createSymbolLayer(layerProperties) { + const layer = new SymbolStyleLayer(layerProperties); + layer.recalculate({zoom: 0, zoomHistory: {} as ZoomHistory} as EvaluationParameters, []); + return layer; +} + +describe('getOverlapMode', () => { + test('defaults - no props set', () => { + const props = {}; + const layer = createSymbolLayer(props); + + expect(getOverlapMode(layer.layout, 'icon-overlap', 'icon-allow-overlap')).toBe('never'); + expect(getOverlapMode(layer.layout, 'text-overlap', 'text-allow-overlap')).toBe('never'); + }); + + test('-allow-overlap set', () => { + const props = {layout: {'icon-allow-overlap': false, 'text-allow-overlap': true}}; + const layer = createSymbolLayer(props); + + expect(getOverlapMode(layer.layout, 'icon-overlap', 'icon-allow-overlap')).toBe('never'); + expect(getOverlapMode(layer.layout, 'text-overlap', 'text-allow-overlap')).toBe('always'); + }); + + test('-overlap set', () => { + let props = {layout: {'icon-overlap': 'never', 'text-overlap': 'always'}}; + let layer = createSymbolLayer(props); + + expect(getOverlapMode(layer.layout, 'icon-overlap', 'icon-allow-overlap')).toBe('never'); + expect(getOverlapMode(layer.layout, 'text-overlap', 'text-allow-overlap')).toBe('always'); + + props = {layout: {'icon-overlap': 'always', 'text-overlap': 'cooperative'}}; + layer = createSymbolLayer(props); + + expect(getOverlapMode(layer.layout, 'icon-overlap', 'icon-allow-overlap')).toBe('always'); + expect(getOverlapMode(layer.layout, 'text-overlap', 'text-allow-overlap')).toBe('cooperative'); + }); + + test('-overlap beats -allow-overlap', () => { + const props = { + layout: { + 'icon-overlap': 'never', + 'icon-allow-overlap': true, + 'text-overlap': 'cooperative', + 'text-allow-overlap': false + } + }; + const layer = createSymbolLayer(props); + + expect(getOverlapMode(layer.layout, 'icon-overlap', 'icon-allow-overlap')).toBe('never'); + expect(getOverlapMode(layer.layout, 'text-overlap', 'text-allow-overlap')).toBe('cooperative'); + }); +}); diff --git a/web/libraries/maplibre-gl/src/style/style_layer/overlap_mode.ts b/web/libraries/maplibre-gl/src/style/style_layer/overlap_mode.ts new file mode 100644 index 00000000..1d1ed5ff --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/overlap_mode.ts @@ -0,0 +1,25 @@ +import {SymbolLayoutPropsPossiblyEvaluated} from './symbol_style_layer_properties.g'; +import type {SymbolLayoutProps} from './symbol_style_layer_properties.g'; +import {PossiblyEvaluated} from '../properties'; + +/** + * The overlap mode for properties like `icon-overlap`and `text-overlap` + */ +export type OverlapMode = 'never' | 'always' | 'cooperative'; + +export function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'icon-overlap', allowOverlapProp: 'icon-allow-overlap'): OverlapMode; +export function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'text-overlap', allowOverlapProp: 'text-allow-overlap'): OverlapMode; +export function getOverlapMode(layout: PossiblyEvaluated, overlapProp: 'icon-overlap' | 'text-overlap', allowOverlapProp: 'icon-allow-overlap' | 'text-allow-overlap'): OverlapMode { + let result: OverlapMode = 'never'; + const overlap = layout.get(overlapProp); + + if (overlap) { + // if -overlap is set, use it + result = overlap; + } else if (layout.get(allowOverlapProp)) { + // fall back to -allow-overlap, with false='never', true='always' + result = 'always'; + } + + return result; +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer/raster_style_layer.ts b/web/libraries/maplibre-gl/src/style/style_layer/raster_style_layer.ts new file mode 100644 index 00000000..407b11af --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/raster_style_layer.ts @@ -0,0 +1,17 @@ +import {StyleLayer} from '../style_layer'; + +import properties, {RasterPaintPropsPossiblyEvaluated} from './raster_style_layer_properties.g'; +import {Transitionable, Transitioning, PossiblyEvaluated} from '../properties'; + +import type {RasterPaintProps} from './raster_style_layer_properties.g'; +import type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; + +export class RasterStyleLayer extends StyleLayer { + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + + constructor(layer: LayerSpecification) { + super(layer, properties); + } +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer/raster_style_layer_properties.g.ts b/web/libraries/maplibre-gl/src/style/style_layer/raster_style_layer_properties.g.ts new file mode 100644 index 00000000..4a953f59 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/raster_style_layer_properties.g.ts @@ -0,0 +1,55 @@ +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ + +import {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec'; + +import { + Properties, + DataConstantProperty, + DataDrivenProperty, + CrossFadedDataDrivenProperty, + CrossFadedProperty, + ColorRampProperty, + PossiblyEvaluatedPropertyValue, + CrossFaded +} from '../properties'; + +import type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec'; +import {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec'; + + +export type RasterPaintProps = { + "raster-opacity": DataConstantProperty, + "raster-hue-rotate": DataConstantProperty, + "raster-brightness-min": DataConstantProperty, + "raster-brightness-max": DataConstantProperty, + "raster-saturation": DataConstantProperty, + "raster-contrast": DataConstantProperty, + "raster-resampling": DataConstantProperty<"linear" | "nearest">, + "raster-fade-duration": DataConstantProperty, +}; + +export type RasterPaintPropsPossiblyEvaluated = { + "raster-opacity": number, + "raster-hue-rotate": number, + "raster-brightness-min": number, + "raster-brightness-max": number, + "raster-saturation": number, + "raster-contrast": number, + "raster-resampling": "linear" | "nearest", + "raster-fade-duration": number, +}; + +let paint: Properties; +const getPaint = () => paint = paint || new Properties({ + "raster-opacity": new DataConstantProperty(styleSpec["paint_raster"]["raster-opacity"] as any as StylePropertySpecification), + "raster-hue-rotate": new DataConstantProperty(styleSpec["paint_raster"]["raster-hue-rotate"] as any as StylePropertySpecification), + "raster-brightness-min": new DataConstantProperty(styleSpec["paint_raster"]["raster-brightness-min"] as any as StylePropertySpecification), + "raster-brightness-max": new DataConstantProperty(styleSpec["paint_raster"]["raster-brightness-max"] as any as StylePropertySpecification), + "raster-saturation": new DataConstantProperty(styleSpec["paint_raster"]["raster-saturation"] as any as StylePropertySpecification), + "raster-contrast": new DataConstantProperty(styleSpec["paint_raster"]["raster-contrast"] as any as StylePropertySpecification), + "raster-resampling": new DataConstantProperty(styleSpec["paint_raster"]["raster-resampling"] as any as StylePropertySpecification), + "raster-fade-duration": new DataConstantProperty(styleSpec["paint_raster"]["raster-fade-duration"] as any as StylePropertySpecification), +}); + +export default ({ get paint() { return getPaint() } }); \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/style/style_layer/symbol_style_layer.ts b/web/libraries/maplibre-gl/src/style/style_layer/symbol_style_layer.ts new file mode 100644 index 00000000..34d9f157 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/symbol_style_layer.ts @@ -0,0 +1,195 @@ +import {StyleLayer} from '../style_layer'; + +import {SymbolBucket, SymbolFeature} from '../../data/bucket/symbol_bucket'; +import {resolveTokens} from '../../util/resolve_tokens'; +import properties, {SymbolLayoutPropsPossiblyEvaluated, SymbolPaintPropsPossiblyEvaluated} from './symbol_style_layer_properties.g'; + +import { + Transitionable, + Transitioning, + Layout, + PossiblyEvaluated, + PossiblyEvaluatedPropertyValue, + PropertyValue +} from '../properties'; + +import { + isExpression, + StyleExpression, + ZoomConstantExpression, + ZoomDependentExpression, + FormattedType, + typeOf, + Formatted, + FormatExpression, + Literal} from '@maplibre/maplibre-gl-style-spec'; + +import type {BucketParameters} from '../../data/bucket'; +import type {SymbolLayoutProps, SymbolPaintProps} from './symbol_style_layer_properties.g'; +import type {EvaluationParameters} from '../evaluation_parameters'; +import type {Expression, Feature, SourceExpression, LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; +import type {CanonicalTileID} from '../../source/tile_id'; +import {FormatSectionOverride} from '../format_section_override'; + +export class SymbolStyleLayer extends StyleLayer { + _unevaluatedLayout: Layout; + layout: PossiblyEvaluated; + + _transitionablePaint: Transitionable; + _transitioningPaint: Transitioning; + paint: PossiblyEvaluated; + + constructor(layer: LayerSpecification) { + super(layer, properties); + } + + recalculate(parameters: EvaluationParameters, availableImages: Array) { + super.recalculate(parameters, availableImages); + + if (this.layout.get('icon-rotation-alignment') === 'auto') { + if (this.layout.get('symbol-placement') !== 'point') { + this.layout._values['icon-rotation-alignment'] = 'map'; + } else { + this.layout._values['icon-rotation-alignment'] = 'viewport'; + } + } + + if (this.layout.get('text-rotation-alignment') === 'auto') { + if (this.layout.get('symbol-placement') !== 'point') { + this.layout._values['text-rotation-alignment'] = 'map'; + } else { + this.layout._values['text-rotation-alignment'] = 'viewport'; + } + } + + // If unspecified, `*-pitch-alignment` inherits `*-rotation-alignment` + if (this.layout.get('text-pitch-alignment') === 'auto') { + this.layout._values['text-pitch-alignment'] = this.layout.get('text-rotation-alignment') === 'map' ? 'map' : 'viewport'; + } + if (this.layout.get('icon-pitch-alignment') === 'auto') { + this.layout._values['icon-pitch-alignment'] = this.layout.get('icon-rotation-alignment'); + } + + if (this.layout.get('symbol-placement') === 'point') { + const writingModes = this.layout.get('text-writing-mode'); + if (writingModes) { + // remove duplicates, preserving order + const deduped = []; + for (const m of writingModes) { + if (deduped.indexOf(m) < 0) deduped.push(m); + } + this.layout._values['text-writing-mode'] = deduped; + } else { + this.layout._values['text-writing-mode'] = ['horizontal']; + } + } + + this._setPaintOverrides(); + } + + getValueAndResolveTokens(name: any, feature: Feature, canonical: CanonicalTileID, availableImages: Array) { + const value = this.layout.get(name).evaluate(feature, {}, canonical, availableImages); + const unevaluated = this._unevaluatedLayout._values[name]; + if (!unevaluated.isDataDriven() && !isExpression(unevaluated.value) && value) { + return resolveTokens(feature.properties, value); + } + + return value; + } + + createBucket(parameters: BucketParameters) { + return new SymbolBucket(parameters); + } + + queryRadius(): number { + return 0; + } + + queryIntersectsFeature(): boolean { + throw new Error('Should take a different path in FeatureIndex'); + } + + _setPaintOverrides() { + for (const overridable of properties.paint.overridableProperties) { + if (!SymbolStyleLayer.hasPaintOverride(this.layout, overridable)) { + continue; + } + const overriden = this.paint.get(overridable as keyof SymbolPaintPropsPossiblyEvaluated) as PossiblyEvaluatedPropertyValue; + const override = new FormatSectionOverride(overriden); + const styleExpression = new StyleExpression(override, overriden.property.specification); + let expression = null; + if (overriden.value.kind === 'constant' || overriden.value.kind === 'source') { + expression = new ZoomConstantExpression('source', styleExpression) as SourceExpression; + } else { + expression = new ZoomDependentExpression('composite', + styleExpression, + overriden.value.zoomStops); + } + this.paint._values[overridable] = new PossiblyEvaluatedPropertyValue(overriden.property, + expression, + overriden.parameters); + } + } + + _handleOverridablePaintPropertyUpdate(name: string, oldValue: PropertyValue, newValue: PropertyValue): boolean { + if (!this.layout || oldValue.isDataDriven() || newValue.isDataDriven()) { + return false; + } + return SymbolStyleLayer.hasPaintOverride(this.layout, name); + } + + static hasPaintOverride(layout: PossiblyEvaluated, propertyName: string): boolean { + const textField = layout.get('text-field'); + const property = properties.paint.properties[propertyName]; + let hasOverrides = false; + + const checkSections = (sections) => { + for (const section of sections) { + if (property.overrides && property.overrides.hasOverride(section)) { + hasOverrides = true; + return; + } + } + }; + + if (textField.value.kind === 'constant' && textField.value.value instanceof Formatted) { + checkSections(textField.value.value.sections); + } else if (textField.value.kind === 'source') { + + const checkExpression = (expression: Expression) => { + if (hasOverrides) return; + + if (expression instanceof Literal && typeOf(expression.value) === FormattedType) { + const formatted: Formatted = (expression.value as any); + checkSections(formatted.sections); + } else if (expression instanceof FormatExpression) { + checkSections(expression.sections); + } else { + expression.eachChild(checkExpression); + } + }; + + const expr: ZoomConstantExpression<'source'> = (textField.value as any); + if (expr._styleExpression) { + checkExpression(expr._styleExpression.expression); + } + } + + return hasOverrides; + } +} + +export type SymbolPadding = [number, number, number, number]; + +export function getIconPadding(layout: PossiblyEvaluated, feature: SymbolFeature, canonical: CanonicalTileID, pixelRatio = 1): SymbolPadding { + // Support text-padding in addition to icon-padding? Unclear how to apply asymmetric text-padding to the radius for collision circles. + const result = layout.get('icon-padding').evaluate(feature, {}, canonical); + const values = result && result.values; + + return [ + values[0] * pixelRatio, + values[1] * pixelRatio, + values[2] * pixelRatio, + values[3] * pixelRatio, + ]; +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer/symbol_style_layer_properties.g.ts b/web/libraries/maplibre-gl/src/style/style_layer/symbol_style_layer_properties.g.ts new file mode 100644 index 00000000..a64c47d7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/symbol_style_layer_properties.g.ts @@ -0,0 +1,218 @@ +// This file is generated. Edit build/generate-style-code.ts, then run 'npm run codegen'. +/* eslint-disable */ + +import {latest as styleSpec} from '@maplibre/maplibre-gl-style-spec'; + +import { + Properties, + DataConstantProperty, + DataDrivenProperty, + CrossFadedDataDrivenProperty, + CrossFadedProperty, + ColorRampProperty, + PossiblyEvaluatedPropertyValue, + CrossFaded +} from '../properties'; + +import type {Color, Formatted, Padding, ResolvedImage, VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec'; +import {StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec'; + +import { + ColorType + } from '@maplibre/maplibre-gl-style-spec'; + +export type SymbolLayoutProps = { + "symbol-placement": DataConstantProperty<"point" | "line" | "line-center">, + "symbol-spacing": DataConstantProperty, + "symbol-avoid-edges": DataConstantProperty, + "symbol-sort-key": DataDrivenProperty, + "symbol-z-order": DataConstantProperty<"auto" | "viewport-y" | "source">, + "icon-allow-overlap": DataConstantProperty, + "icon-overlap": DataConstantProperty<"never" | "always" | "cooperative">, + "icon-ignore-placement": DataConstantProperty, + "icon-optional": DataConstantProperty, + "icon-rotation-alignment": DataConstantProperty<"map" | "viewport" | "auto">, + "icon-size": DataDrivenProperty, + "icon-text-fit": DataConstantProperty<"none" | "width" | "height" | "both">, + "icon-text-fit-padding": DataConstantProperty<[number, number, number, number]>, + "icon-image": DataDrivenProperty, + "icon-rotate": DataDrivenProperty, + "icon-padding": DataDrivenProperty, + "icon-keep-upright": DataConstantProperty, + "icon-offset": DataDrivenProperty<[number, number]>, + "icon-anchor": DataDrivenProperty<"center" | "left" | "right" | "top" | "bottom" | "top-left" | "top-right" | "bottom-left" | "bottom-right">, + "icon-pitch-alignment": DataConstantProperty<"map" | "viewport" | "auto">, + "text-pitch-alignment": DataConstantProperty<"map" | "viewport" | "auto">, + "text-rotation-alignment": DataConstantProperty<"map" | "viewport" | "viewport-glyph" | "auto">, + "text-field": DataDrivenProperty, + "text-font": DataDrivenProperty>, + "text-size": DataDrivenProperty, + "text-max-width": DataDrivenProperty, + "text-line-height": DataConstantProperty, + "text-letter-spacing": DataDrivenProperty, + "text-justify": DataDrivenProperty<"auto" | "left" | "center" | "right">, + "text-radial-offset": DataDrivenProperty, + "text-variable-anchor": DataConstantProperty>, + "text-variable-anchor-offset": DataDrivenProperty, + "text-anchor": DataDrivenProperty<"center" | "left" | "right" | "top" | "bottom" | "top-left" | "top-right" | "bottom-left" | "bottom-right">, + "text-max-angle": DataConstantProperty, + "text-writing-mode": DataConstantProperty>, + "text-rotate": DataDrivenProperty, + "text-padding": DataConstantProperty, + "text-keep-upright": DataConstantProperty, + "text-transform": DataDrivenProperty<"none" | "uppercase" | "lowercase">, + "text-offset": DataDrivenProperty<[number, number]>, + "text-allow-overlap": DataConstantProperty, + "text-overlap": DataConstantProperty<"never" | "always" | "cooperative">, + "text-ignore-placement": DataConstantProperty, + "text-optional": DataConstantProperty, +}; + +export type SymbolLayoutPropsPossiblyEvaluated = { + "symbol-placement": "point" | "line" | "line-center", + "symbol-spacing": number, + "symbol-avoid-edges": boolean, + "symbol-sort-key": PossiblyEvaluatedPropertyValue, + "symbol-z-order": "auto" | "viewport-y" | "source", + "icon-allow-overlap": boolean, + "icon-overlap": "never" | "always" | "cooperative", + "icon-ignore-placement": boolean, + "icon-optional": boolean, + "icon-rotation-alignment": "map" | "viewport" | "auto", + "icon-size": PossiblyEvaluatedPropertyValue, + "icon-text-fit": "none" | "width" | "height" | "both", + "icon-text-fit-padding": [number, number, number, number], + "icon-image": PossiblyEvaluatedPropertyValue, + "icon-rotate": PossiblyEvaluatedPropertyValue, + "icon-padding": PossiblyEvaluatedPropertyValue, + "icon-keep-upright": boolean, + "icon-offset": PossiblyEvaluatedPropertyValue<[number, number]>, + "icon-anchor": PossiblyEvaluatedPropertyValue<"center" | "left" | "right" | "top" | "bottom" | "top-left" | "top-right" | "bottom-left" | "bottom-right">, + "icon-pitch-alignment": "map" | "viewport" | "auto", + "text-pitch-alignment": "map" | "viewport" | "auto", + "text-rotation-alignment": "map" | "viewport" | "viewport-glyph" | "auto", + "text-field": PossiblyEvaluatedPropertyValue, + "text-font": PossiblyEvaluatedPropertyValue>, + "text-size": PossiblyEvaluatedPropertyValue, + "text-max-width": PossiblyEvaluatedPropertyValue, + "text-line-height": number, + "text-letter-spacing": PossiblyEvaluatedPropertyValue, + "text-justify": PossiblyEvaluatedPropertyValue<"auto" | "left" | "center" | "right">, + "text-radial-offset": PossiblyEvaluatedPropertyValue, + "text-variable-anchor": Array<"center" | "left" | "right" | "top" | "bottom" | "top-left" | "top-right" | "bottom-left" | "bottom-right">, + "text-variable-anchor-offset": PossiblyEvaluatedPropertyValue, + "text-anchor": PossiblyEvaluatedPropertyValue<"center" | "left" | "right" | "top" | "bottom" | "top-left" | "top-right" | "bottom-left" | "bottom-right">, + "text-max-angle": number, + "text-writing-mode": Array<"horizontal" | "vertical">, + "text-rotate": PossiblyEvaluatedPropertyValue, + "text-padding": number, + "text-keep-upright": boolean, + "text-transform": PossiblyEvaluatedPropertyValue<"none" | "uppercase" | "lowercase">, + "text-offset": PossiblyEvaluatedPropertyValue<[number, number]>, + "text-allow-overlap": boolean, + "text-overlap": "never" | "always" | "cooperative", + "text-ignore-placement": boolean, + "text-optional": boolean, +}; + +let layout: Properties; +const getLayout = () => layout = layout || new Properties({ + "symbol-placement": new DataConstantProperty(styleSpec["layout_symbol"]["symbol-placement"] as any as StylePropertySpecification), + "symbol-spacing": new DataConstantProperty(styleSpec["layout_symbol"]["symbol-spacing"] as any as StylePropertySpecification), + "symbol-avoid-edges": new DataConstantProperty(styleSpec["layout_symbol"]["symbol-avoid-edges"] as any as StylePropertySpecification), + "symbol-sort-key": new DataDrivenProperty(styleSpec["layout_symbol"]["symbol-sort-key"] as any as StylePropertySpecification), + "symbol-z-order": new DataConstantProperty(styleSpec["layout_symbol"]["symbol-z-order"] as any as StylePropertySpecification), + "icon-allow-overlap": new DataConstantProperty(styleSpec["layout_symbol"]["icon-allow-overlap"] as any as StylePropertySpecification), + "icon-overlap": new DataConstantProperty(styleSpec["layout_symbol"]["icon-overlap"] as any as StylePropertySpecification), + "icon-ignore-placement": new DataConstantProperty(styleSpec["layout_symbol"]["icon-ignore-placement"] as any as StylePropertySpecification), + "icon-optional": new DataConstantProperty(styleSpec["layout_symbol"]["icon-optional"] as any as StylePropertySpecification), + "icon-rotation-alignment": new DataConstantProperty(styleSpec["layout_symbol"]["icon-rotation-alignment"] as any as StylePropertySpecification), + "icon-size": new DataDrivenProperty(styleSpec["layout_symbol"]["icon-size"] as any as StylePropertySpecification), + "icon-text-fit": new DataConstantProperty(styleSpec["layout_symbol"]["icon-text-fit"] as any as StylePropertySpecification), + "icon-text-fit-padding": new DataConstantProperty(styleSpec["layout_symbol"]["icon-text-fit-padding"] as any as StylePropertySpecification), + "icon-image": new DataDrivenProperty(styleSpec["layout_symbol"]["icon-image"] as any as StylePropertySpecification), + "icon-rotate": new DataDrivenProperty(styleSpec["layout_symbol"]["icon-rotate"] as any as StylePropertySpecification), + "icon-padding": new DataDrivenProperty(styleSpec["layout_symbol"]["icon-padding"] as any as StylePropertySpecification), + "icon-keep-upright": new DataConstantProperty(styleSpec["layout_symbol"]["icon-keep-upright"] as any as StylePropertySpecification), + "icon-offset": new DataDrivenProperty(styleSpec["layout_symbol"]["icon-offset"] as any as StylePropertySpecification), + "icon-anchor": new DataDrivenProperty(styleSpec["layout_symbol"]["icon-anchor"] as any as StylePropertySpecification), + "icon-pitch-alignment": new DataConstantProperty(styleSpec["layout_symbol"]["icon-pitch-alignment"] as any as StylePropertySpecification), + "text-pitch-alignment": new DataConstantProperty(styleSpec["layout_symbol"]["text-pitch-alignment"] as any as StylePropertySpecification), + "text-rotation-alignment": new DataConstantProperty(styleSpec["layout_symbol"]["text-rotation-alignment"] as any as StylePropertySpecification), + "text-field": new DataDrivenProperty(styleSpec["layout_symbol"]["text-field"] as any as StylePropertySpecification), + "text-font": new DataDrivenProperty(styleSpec["layout_symbol"]["text-font"] as any as StylePropertySpecification), + "text-size": new DataDrivenProperty(styleSpec["layout_symbol"]["text-size"] as any as StylePropertySpecification), + "text-max-width": new DataDrivenProperty(styleSpec["layout_symbol"]["text-max-width"] as any as StylePropertySpecification), + "text-line-height": new DataConstantProperty(styleSpec["layout_symbol"]["text-line-height"] as any as StylePropertySpecification), + "text-letter-spacing": new DataDrivenProperty(styleSpec["layout_symbol"]["text-letter-spacing"] as any as StylePropertySpecification), + "text-justify": new DataDrivenProperty(styleSpec["layout_symbol"]["text-justify"] as any as StylePropertySpecification), + "text-radial-offset": new DataDrivenProperty(styleSpec["layout_symbol"]["text-radial-offset"] as any as StylePropertySpecification), + "text-variable-anchor": new DataConstantProperty(styleSpec["layout_symbol"]["text-variable-anchor"] as any as StylePropertySpecification), + "text-variable-anchor-offset": new DataDrivenProperty(styleSpec["layout_symbol"]["text-variable-anchor-offset"] as any as StylePropertySpecification), + "text-anchor": new DataDrivenProperty(styleSpec["layout_symbol"]["text-anchor"] as any as StylePropertySpecification), + "text-max-angle": new DataConstantProperty(styleSpec["layout_symbol"]["text-max-angle"] as any as StylePropertySpecification), + "text-writing-mode": new DataConstantProperty(styleSpec["layout_symbol"]["text-writing-mode"] as any as StylePropertySpecification), + "text-rotate": new DataDrivenProperty(styleSpec["layout_symbol"]["text-rotate"] as any as StylePropertySpecification), + "text-padding": new DataConstantProperty(styleSpec["layout_symbol"]["text-padding"] as any as StylePropertySpecification), + "text-keep-upright": new DataConstantProperty(styleSpec["layout_symbol"]["text-keep-upright"] as any as StylePropertySpecification), + "text-transform": new DataDrivenProperty(styleSpec["layout_symbol"]["text-transform"] as any as StylePropertySpecification), + "text-offset": new DataDrivenProperty(styleSpec["layout_symbol"]["text-offset"] as any as StylePropertySpecification), + "text-allow-overlap": new DataConstantProperty(styleSpec["layout_symbol"]["text-allow-overlap"] as any as StylePropertySpecification), + "text-overlap": new DataConstantProperty(styleSpec["layout_symbol"]["text-overlap"] as any as StylePropertySpecification), + "text-ignore-placement": new DataConstantProperty(styleSpec["layout_symbol"]["text-ignore-placement"] as any as StylePropertySpecification), + "text-optional": new DataConstantProperty(styleSpec["layout_symbol"]["text-optional"] as any as StylePropertySpecification), +}); + +export type SymbolPaintProps = { + "icon-opacity": DataDrivenProperty, + "icon-color": DataDrivenProperty, + "icon-halo-color": DataDrivenProperty, + "icon-halo-width": DataDrivenProperty, + "icon-halo-blur": DataDrivenProperty, + "icon-translate": DataConstantProperty<[number, number]>, + "icon-translate-anchor": DataConstantProperty<"map" | "viewport">, + "text-opacity": DataDrivenProperty, + "text-color": DataDrivenProperty, + "text-halo-color": DataDrivenProperty, + "text-halo-width": DataDrivenProperty, + "text-halo-blur": DataDrivenProperty, + "text-translate": DataConstantProperty<[number, number]>, + "text-translate-anchor": DataConstantProperty<"map" | "viewport">, +}; + +export type SymbolPaintPropsPossiblyEvaluated = { + "icon-opacity": PossiblyEvaluatedPropertyValue, + "icon-color": PossiblyEvaluatedPropertyValue, + "icon-halo-color": PossiblyEvaluatedPropertyValue, + "icon-halo-width": PossiblyEvaluatedPropertyValue, + "icon-halo-blur": PossiblyEvaluatedPropertyValue, + "icon-translate": [number, number], + "icon-translate-anchor": "map" | "viewport", + "text-opacity": PossiblyEvaluatedPropertyValue, + "text-color": PossiblyEvaluatedPropertyValue, + "text-halo-color": PossiblyEvaluatedPropertyValue, + "text-halo-width": PossiblyEvaluatedPropertyValue, + "text-halo-blur": PossiblyEvaluatedPropertyValue, + "text-translate": [number, number], + "text-translate-anchor": "map" | "viewport", +}; + +let paint: Properties; +const getPaint = () => paint = paint || new Properties({ + "icon-opacity": new DataDrivenProperty(styleSpec["paint_symbol"]["icon-opacity"] as any as StylePropertySpecification), + "icon-color": new DataDrivenProperty(styleSpec["paint_symbol"]["icon-color"] as any as StylePropertySpecification), + "icon-halo-color": new DataDrivenProperty(styleSpec["paint_symbol"]["icon-halo-color"] as any as StylePropertySpecification), + "icon-halo-width": new DataDrivenProperty(styleSpec["paint_symbol"]["icon-halo-width"] as any as StylePropertySpecification), + "icon-halo-blur": new DataDrivenProperty(styleSpec["paint_symbol"]["icon-halo-blur"] as any as StylePropertySpecification), + "icon-translate": new DataConstantProperty(styleSpec["paint_symbol"]["icon-translate"] as any as StylePropertySpecification), + "icon-translate-anchor": new DataConstantProperty(styleSpec["paint_symbol"]["icon-translate-anchor"] as any as StylePropertySpecification), + "text-opacity": new DataDrivenProperty(styleSpec["paint_symbol"]["text-opacity"] as any as StylePropertySpecification), + "text-color": new DataDrivenProperty(styleSpec["paint_symbol"]["text-color"] as any as StylePropertySpecification, { runtimeType: ColorType, getOverride: (o) => o.textColor, hasOverride: (o) => !!o.textColor }), + "text-halo-color": new DataDrivenProperty(styleSpec["paint_symbol"]["text-halo-color"] as any as StylePropertySpecification), + "text-halo-width": new DataDrivenProperty(styleSpec["paint_symbol"]["text-halo-width"] as any as StylePropertySpecification), + "text-halo-blur": new DataDrivenProperty(styleSpec["paint_symbol"]["text-halo-blur"] as any as StylePropertySpecification), + "text-translate": new DataConstantProperty(styleSpec["paint_symbol"]["text-translate"] as any as StylePropertySpecification), + "text-translate-anchor": new DataConstantProperty(styleSpec["paint_symbol"]["text-translate-anchor"] as any as StylePropertySpecification), +}); + +export default ({ get paint() { return getPaint() }, get layout() { return getLayout() } }); \ No newline at end of file diff --git a/web/libraries/maplibre-gl/src/style/style_layer/typed_style_layer.ts b/web/libraries/maplibre-gl/src/style/style_layer/typed_style_layer.ts new file mode 100644 index 00000000..a340484e --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/typed_style_layer.ts @@ -0,0 +1,9 @@ +import type {CircleStyleLayer} from './circle_style_layer'; +import type {FillStyleLayer} from './fill_style_layer'; +import type {FillExtrusionStyleLayer} from './fill_extrusion_style_layer'; +import type {HeatmapStyleLayer} from './heatmap_style_layer'; +import type {HillshadeStyleLayer} from './hillshade_style_layer'; +import type {LineStyleLayer} from './line_style_layer'; +import type {SymbolStyleLayer} from './symbol_style_layer'; + +export type TypedStyleLayer = CircleStyleLayer | FillStyleLayer | FillExtrusionStyleLayer | HeatmapStyleLayer | HillshadeStyleLayer | LineStyleLayer | SymbolStyleLayer; diff --git a/web/libraries/maplibre-gl/src/style/style_layer/variable_text_anchor.test.ts b/web/libraries/maplibre-gl/src/style/style_layer/variable_text_anchor.test.ts new file mode 100644 index 00000000..0a4af2a4 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/variable_text_anchor.test.ts @@ -0,0 +1,117 @@ +import {EvaluationParameters} from '../evaluation_parameters'; +import {ZoomHistory} from '../zoom_history'; +import {SymbolStyleLayer} from './symbol_style_layer'; +import {INVALID_TEXT_OFFSET, evaluateVariableOffset, getTextVariableAnchorOffset} from './variable_text_anchor'; + +describe('evaluateVariableOffset', () => { + test('fromRadialOffset', () => { + // Radial offset mode is invoked by using INVALID_TEXT_OFFSET as the Y value + const srcOffset = [10, INVALID_TEXT_OFFSET] as [number, number]; + + expect(evaluateVariableOffset('center', srcOffset)).toEqual([0, 0]); + + // Top/bottom offsets are shifted by the default baseline (7) + expect(evaluateVariableOffset('top', srcOffset)).toEqual([0, 3]); + expect(evaluateVariableOffset('bottom', srcOffset)).toEqual([0, -3]); + expect(evaluateVariableOffset('left', srcOffset)).toEqual([10, 0]); + expect(evaluateVariableOffset('right', srcOffset)).toEqual([-10, 0]); + + const hypotenuse = 10 / Math.SQRT2; + expect(evaluateVariableOffset('top-left', srcOffset)).toEqual([expect.closeTo(hypotenuse), expect.closeTo(hypotenuse - 7)]); + expect(evaluateVariableOffset('top-right', srcOffset)).toEqual([expect.closeTo(-hypotenuse), expect.closeTo(hypotenuse - 7)]); + expect(evaluateVariableOffset('bottom-left', srcOffset)).toEqual([expect.closeTo(hypotenuse), expect.closeTo(-hypotenuse + 7)]); + expect(evaluateVariableOffset('bottom-right', srcOffset)).toEqual([expect.closeTo(-hypotenuse), expect.closeTo(-hypotenuse + 7)]); + }); + + test('fromTextOffset', () => { + const srcOffset = [10, -10] as [number, number]; + + expect(evaluateVariableOffset('center', srcOffset)).toEqual([0, 0]); + + // Top/bottom offsets are shifted by the default baseline (7) + expect(evaluateVariableOffset('top', srcOffset)).toEqual([0, 3]); + expect(evaluateVariableOffset('bottom', srcOffset)).toEqual([0, -3]); + expect(evaluateVariableOffset('left', srcOffset)).toEqual([10, 0]); + expect(evaluateVariableOffset('right', srcOffset)).toEqual([-10, 0]); + expect(evaluateVariableOffset('top-left', srcOffset)).toEqual([10, 3]); + expect(evaluateVariableOffset('top-right', srcOffset)).toEqual([-10, 3]); + expect(evaluateVariableOffset('bottom-left', srcOffset)).toEqual([10, -3]); + expect(evaluateVariableOffset('bottom-right', srcOffset)).toEqual([-10, -3]); + }); +}); + +function createSymbolLayer(layerProperties) { + const layer = new SymbolStyleLayer(layerProperties); + layer.recalculate({zoom: 0, zoomHistory: {} as ZoomHistory} as EvaluationParameters, []); + return layer; +} + +describe('getTextVariableAnchorOffset', () => { + test('defaults - no props set', () => { + const props = {}; + const layer = createSymbolLayer(props); + + expect(getTextVariableAnchorOffset(layer, null, null)).toBeNull(); + }); + + test('text-variable-anchor-offset set', () => { + const props = {layout: {'text-variable-anchor-offset': ['top', [1, 1], 'bottom', [2, 2]]}}; + const layer = createSymbolLayer(props); + + const offset = getTextVariableAnchorOffset(layer, null, null); + expect(offset).toBeDefined(); + // Offset converted to EMs, accounting for baseline shift on Y axis + expect(offset.toString()).toBe('["top",[24,17],"bottom",[48,55]]'); + }); + + test('text-variable-anchor set', () => { + const props = {layout: {'text-variable-anchor': ['top']}}; + const layer = createSymbolLayer(props); + + const offset = getTextVariableAnchorOffset(layer, null, null); + expect(offset).toBeDefined(); + // Default offset (0, 0) converted to EMs, accounting for baseline shift on Y axis + expect(offset.toString()).toBe('["top",[0,-7]]'); + }); + + test('text-variable-anchor and text-offset set', () => { + const props = {layout: {'text-variable-anchor': ['top'], 'text-offset': [1, 1]}}; + const layer = createSymbolLayer(props); + + const offset = getTextVariableAnchorOffset(layer, null, null); + expect(offset).toBeDefined(); + // Offset converted to EMs, accounting for baseline shift on Y axis + expect(offset.toString()).toBe('["top",[0,17]]'); + }); + + test('text-variable-anchor and text-radial-offset set', () => { + const props = {layout: {'text-variable-anchor': ['top'], 'text-radial-offset': 2}}; + const layer = createSymbolLayer(props); + + const offset = getTextVariableAnchorOffset(layer, null, null); + expect(offset).toBeDefined(); + // Offset converted to EMs, accounting for baseline shift on Y axis + expect(offset.toString()).toBe('["top",[0,41]]'); + }); + + test('text-variable-anchor, text-offset, and text-radial-offset set', () => { + const props = {layout: {'text-variable-anchor': ['top'], 'text-offset': [1, 1], 'text-radial-offset': 2}}; + const layer = createSymbolLayer(props); + + const offset = getTextVariableAnchorOffset(layer, null, null); + expect(offset).toBeDefined(); + // Offset converted to EMs, accounting for baseline shift on Y axis + expect(offset.toString()).toBe('["top",[0,41]]'); + }); + + test('text-variable-anchor and text-variable-anchor-offset set', () => { + const props = {layout: {'text-variable-anchor-offset': ['top', [1, 1]], 'text-variable-anchor': ['bottom']}}; + const layer = createSymbolLayer(props); + + const offset = getTextVariableAnchorOffset(layer, null, null); + expect(offset).toBeDefined(); + // Offset converted to EMs, accounting for baseline shift on Y axis + expect(offset.toString()).toBe('["top",[24,17]]'); + }); +}); + diff --git a/web/libraries/maplibre-gl/src/style/style_layer/variable_text_anchor.ts b/web/libraries/maplibre-gl/src/style/style_layer/variable_text_anchor.ts new file mode 100644 index 00000000..cb6474ee --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer/variable_text_anchor.ts @@ -0,0 +1,163 @@ +import {VariableAnchorOffsetCollection, VariableAnchorOffsetCollectionSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {SymbolFeature} from '../../data/bucket/symbol_bucket'; +import {CanonicalTileID} from '../../source/tile_id'; +import ONE_EM from '../../symbol/one_em'; +import {SymbolStyleLayer} from './symbol_style_layer'; + +export enum TextAnchorEnum { + 'center' = 1, + 'left' = 2, + 'right' = 3, + 'top' = 4, + 'bottom' = 5, + 'top-left' = 6, + 'top-right' = 7, + 'bottom-left' = 8, + 'bottom-right' = 9 +} + +export type TextAnchor = keyof typeof TextAnchorEnum; + +// The radial offset is to the edge of the text box +// In the horizontal direction, the edge of the text box is where glyphs start +// But in the vertical direction, the glyphs appear to "start" at the baseline +// We don't actually load baseline data, but we assume an offset of ONE_EM - 17 +// (see "yOffset" in shaping.js) +const baselineOffset = 7; +export const INVALID_TEXT_OFFSET = Number.POSITIVE_INFINITY; + +export function evaluateVariableOffset(anchor: TextAnchor, offset: [number, number]): [number, number] { + + function fromRadialOffset(anchor: TextAnchor, radialOffset: number): [number, number] { + let x = 0, y = 0; + if (radialOffset < 0) radialOffset = 0; // Ignore negative offset. + // solve for r where r^2 + r^2 = radialOffset^2 + const hypotenuse = radialOffset / Math.SQRT2; + switch (anchor) { + case 'top-right': + case 'top-left': + y = hypotenuse - baselineOffset; + break; + case 'bottom-right': + case 'bottom-left': + y = -hypotenuse + baselineOffset; + break; + case 'bottom': + y = -radialOffset + baselineOffset; + break; + case 'top': + y = radialOffset - baselineOffset; + break; + } + + switch (anchor) { + case 'top-right': + case 'bottom-right': + x = -hypotenuse; + break; + case 'top-left': + case 'bottom-left': + x = hypotenuse; + break; + case 'left': + x = radialOffset; + break; + case 'right': + x = -radialOffset; + break; + } + + return [x, y]; + } + + function fromTextOffset(anchor: TextAnchor, offsetX: number, offsetY: number): [number, number] { + let x = 0, y = 0; + // Use absolute offset values. + offsetX = Math.abs(offsetX); + offsetY = Math.abs(offsetY); + + switch (anchor) { + case 'top-right': + case 'top-left': + case 'top': + y = offsetY - baselineOffset; + break; + case 'bottom-right': + case 'bottom-left': + case 'bottom': + y = -offsetY + baselineOffset; + break; + } + + switch (anchor) { + case 'top-right': + case 'bottom-right': + case 'right': + x = -offsetX; + break; + case 'top-left': + case 'bottom-left': + case 'left': + x = offsetX; + break; + } + + return [x, y]; + } + + return (offset[1] !== INVALID_TEXT_OFFSET) ? fromTextOffset(anchor, offset[0], offset[1]) : fromRadialOffset(anchor, offset[0]); +} + +// Helper to support both text-variable-anchor and text-variable-anchor-offset. Offset values converted from EMs to PXs +export function getTextVariableAnchorOffset(layer: SymbolStyleLayer, feature: SymbolFeature, canonical: CanonicalTileID): VariableAnchorOffsetCollection | null { + const layout = layer.layout; + // If style specifies text-variable-anchor-offset, just return it + const variableAnchorOffset = layout.get('text-variable-anchor-offset')?.evaluate(feature, {}, canonical); + + if (variableAnchorOffset) { + const sourceValues = variableAnchorOffset.values; + const destValues: VariableAnchorOffsetCollectionSpecification = []; + + // Convert offsets from EM to PX, and apply baseline shift + for (let i = 0; i < sourceValues.length; i += 2) { + const anchor = destValues[i] = sourceValues[i] as TextAnchor; + const offset = (sourceValues[i + 1] as [number, number]).map(t => t * ONE_EM) as [number, number]; + + if (anchor.startsWith('top')) { + offset[1] -= baselineOffset; + } else if (anchor.startsWith('bottom')) { + offset[1] += baselineOffset; + } + + destValues[i + 1] = offset; + } + + return new VariableAnchorOffsetCollection(destValues); + } + + // If style specifies text-variable-anchor, convert to the new format + const variableAnchor = layout.get('text-variable-anchor'); + + if (variableAnchor) { + let textOffset: [number, number]; + const unevaluatedLayout = layer._unevaluatedLayout; + + // The style spec says don't use `text-offset` and `text-radial-offset` together + // but doesn't actually specify what happens if you use both. We go with the radial offset. + if (unevaluatedLayout.getValue('text-radial-offset') !== undefined) { + textOffset = [layout.get('text-radial-offset').evaluate(feature, {}, canonical) * ONE_EM, INVALID_TEXT_OFFSET]; + } else { + textOffset = layout.get('text-offset').evaluate(feature, {}, canonical).map(t => t * ONE_EM) as [number, number]; + } + + const anchorOffsets: VariableAnchorOffsetCollectionSpecification = []; + + for (const anchor of variableAnchor) { + anchorOffsets.push(anchor, evaluateVariableOffset(anchor, textOffset)); + } + + return new VariableAnchorOffsetCollection(anchorOffsets); + } + + return null; +} diff --git a/web/libraries/maplibre-gl/src/style/style_layer_index.test.ts b/web/libraries/maplibre-gl/src/style/style_layer_index.test.ts new file mode 100644 index 00000000..8d84e7a6 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer_index.test.ts @@ -0,0 +1,99 @@ +import {mapObject} from '../util/util'; +import {StyleLayerIndex} from './style_layer_index'; + +describe('StyleLayerIndex', () => { + test('StyleLayerIndex#replace', () => { + const index = new StyleLayerIndex([ + {id: '1', type: 'fill', source: 'source', 'source-layer': 'layer', paint: {'fill-color': 'red'}}, + {id: '2', type: 'circle', source: 'source', 'source-layer': 'layer', paint: {'circle-color': 'green'}}, + {id: '3', type: 'circle', source: 'source', 'source-layer': 'layer', paint: {'circle-color': 'blue'}} + ]); + + const families = index.familiesBySource['source']['layer']; + expect(families).toHaveLength(2); + expect(families[0]).toHaveLength(1); + expect(families[0][0].id).toBe('1'); + expect(families[1]).toHaveLength(2); + expect(families[1][0].id).toBe('2'); + expect(families[1][1].id).toBe('3'); + + index.replace([]); + expect(index.familiesBySource).toEqual({}); + + }); + + test('StyleLayerIndex#update', () => { + const index = new StyleLayerIndex([ + {id: '1', type: 'fill', source: 'foo', 'source-layer': 'layer', paint: {'fill-color': 'red'}}, + {id: '2', type: 'circle', source: 'foo', 'source-layer': 'layer', paint: {'circle-color': 'green'}}, + {id: '3', type: 'circle', source: 'foo', 'source-layer': 'layer', paint: {'circle-color': 'blue'}} + ]); + + index.update([ + {id: '1', type: 'fill', source: 'bar', 'source-layer': 'layer', paint: {'fill-color': 'cyan'}}, + {id: '2', type: 'circle', source: 'bar', 'source-layer': 'layer', paint: {'circle-color': 'magenta'}}, + {id: '3', type: 'circle', source: 'bar', 'source-layer': 'layer', paint: {'circle-color': 'yellow'}} + ], []); + + const families = index.familiesBySource['bar']['layer']; + expect(families).toHaveLength(2); + expect(families[0]).toHaveLength(1); + expect(families[0][0].getPaintProperty('fill-color')).toBe('cyan'); + expect(families[1]).toHaveLength(2); + expect(families[1][0].getPaintProperty('circle-color')).toBe('magenta'); + expect(families[1][0].source).toBe('bar'); + expect(families[1][1].getPaintProperty('circle-color')).toBe('yellow'); + expect(families[1][1].source).toBe('bar'); + + }); + + test('StyleLayerIndex#familiesBySource', () => { + const index = new StyleLayerIndex([ + {id: '0', type: 'fill', 'source': 'A', 'source-layer': 'foo'}, + {id: '1', type: 'fill', 'source': 'A', 'source-layer': 'foo'}, + {id: '2', type: 'fill', 'source': 'A', 'source-layer': 'foo', 'minzoom': 1}, + {id: '3', type: 'fill', 'source': 'A', 'source-layer': 'bar'}, + {id: '4', type: 'fill', 'source': 'B', 'source-layer': 'foo'}, + {id: '5', type: 'fill', 'source': 'geojson'}, + {id: '6', type: 'background'} + ]); + + const ids = mapObject(index.familiesBySource, (bySource) => { + return mapObject(bySource, (families) => { + return families.map((family) => { + return family.map((layer) => layer.id); + }); + }); + }); + + expect(ids).toEqual({ + 'A': { + 'foo': [['0', '1'], ['2']], + 'bar': [['3']] + }, + 'B': { + 'foo': [['4']] + }, + 'geojson': { + '_geojsonTileLayer': [['5']] + }, + '': { + '_geojsonTileLayer': [['6']] + } + }); + + }); + + test('StyleLayerIndex groups families even if layout key order differs', () => { + const index = new StyleLayerIndex([ + {id: '0', type: 'line', 'source': 'source', 'source-layer': 'layer', + 'layout': {'line-cap': 'butt', 'line-join': 'miter'}}, + {id: '1', type: 'line', 'source': 'source', 'source-layer': 'layer', + 'layout': {'line-join': 'miter', 'line-cap': 'butt'}} + ]); + + const families = index.familiesBySource['source']['layer']; + expect(families[0]).toHaveLength(2); + + }); +}); diff --git a/web/libraries/maplibre-gl/src/style/style_layer_index.ts b/web/libraries/maplibre-gl/src/style/style_layer_index.ts new file mode 100644 index 00000000..0c9b0cd3 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/style_layer_index.ts @@ -0,0 +1,78 @@ +import {StyleLayer} from './style_layer'; +import {createStyleLayer} from './create_style_layer'; + +import {featureFilter, groupByLayout} from '@maplibre/maplibre-gl-style-spec'; + +import type {TypedStyleLayer} from './style_layer/typed_style_layer'; +import type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; + +export type LayerConfigs = {[_: string]: LayerSpecification}; +export type Family = Array; + +export class StyleLayerIndex { + familiesBySource: { + [source: string]: { + [sourceLayer: string]: Array>; + }; + }; + keyCache: {[source: string]: string}; + + _layerConfigs: LayerConfigs; + _layers: {[_: string]: StyleLayer}; + + constructor(layerConfigs?: Array | null) { + this.keyCache = {}; + if (layerConfigs) { + this.replace(layerConfigs); + } + } + + replace(layerConfigs: Array) { + this._layerConfigs = {}; + this._layers = {}; + this.update(layerConfigs, []); + } + + update(layerConfigs: Array, removedIds: Array) { + for (const layerConfig of layerConfigs) { + this._layerConfigs[layerConfig.id] = layerConfig; + + const layer = this._layers[layerConfig.id] = createStyleLayer(layerConfig); + layer._featureFilter = featureFilter(layer.filter); + if (this.keyCache[layerConfig.id]) + delete this.keyCache[layerConfig.id]; + } + for (const id of removedIds) { + delete this.keyCache[id]; + delete this._layerConfigs[id]; + delete this._layers[id]; + } + + this.familiesBySource = {}; + + const groups = groupByLayout(Object.values(this._layerConfigs), this.keyCache); + + for (const layerConfigs of groups) { + const layers = layerConfigs.map((layerConfig) => this._layers[layerConfig.id]); + + const layer = layers[0]; + if (layer.visibility === 'none') { + continue; + } + + const sourceId = layer.source || ''; + let sourceGroup = this.familiesBySource[sourceId]; + if (!sourceGroup) { + sourceGroup = this.familiesBySource[sourceId] = {}; + } + + const sourceLayerId = layer.sourceLayer || '_geojsonTileLayer'; + let sourceLayerFamilies = sourceGroup[sourceLayerId]; + if (!sourceLayerFamilies) { + sourceLayerFamilies = sourceGroup[sourceLayerId] = []; + } + + sourceLayerFamilies.push(layers); + } + } +} diff --git a/web/libraries/maplibre-gl/src/style/validate_style.ts b/web/libraries/maplibre-gl/src/style/validate_style.ts new file mode 100644 index 00000000..9af29869 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/validate_style.ts @@ -0,0 +1,51 @@ +import {validateStyleMin} from '@maplibre/maplibre-gl-style-spec'; +import {ErrorEvent} from '../util/evented'; + +import type {Evented} from '../util/evented'; + +type ValidationError = { + message: string; + line: number; + identifier?: string; +}; + +export type Validator = (a: any) => ReadonlyArray; + +type ValidateStyle = { + source: Validator; + sprite: Validator; + glyphs: Validator; + layer: Validator; + light: Validator; + terrain: Validator; + filter: Validator; + paintProperty: Validator; + layoutProperty: Validator; + (b: any, a?: any | null): ReadonlyArray; +}; + +export const validateStyle = (validateStyleMin as unknown as ValidateStyle); + +export const validateSource = validateStyle.source; +export const validateLight = validateStyle.light; +export const validateTerrain = validateStyle.terrain; +export const validateFilter = validateStyle.filter; +export const validatePaintProperty = validateStyle.paintProperty; +export const validateLayoutProperty = validateStyle.layoutProperty; + +export function emitValidationErrors( + emitter: Evented, + errors?: ReadonlyArray<{ + message: string; + identifier?: string; + }> | null +): boolean { + let hasErrors = false; + if (errors && errors.length) { + for (const error of errors) { + emitter.fire(new ErrorEvent(new Error(error.message))); + hasErrors = true; + } + } + return hasErrors; +} diff --git a/web/libraries/maplibre-gl/src/style/zoom_history.ts b/web/libraries/maplibre-gl/src/style/zoom_history.ts new file mode 100644 index 00000000..c91c13a9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/style/zoom_history.ts @@ -0,0 +1,40 @@ +export class ZoomHistory { + lastZoom: number; + lastFloorZoom: number; + lastIntegerZoom: number; + lastIntegerZoomTime: number; + first: boolean; + + constructor() { + this.first = true; + } + + update(z: number, now: number) { + const floorZ = Math.floor(z); + + if (this.first) { + this.first = false; + this.lastIntegerZoom = floorZ; + this.lastIntegerZoomTime = 0; + this.lastZoom = z; + this.lastFloorZoom = floorZ; + return true; + } + + if (this.lastFloorZoom > floorZ) { + this.lastIntegerZoom = floorZ + 1; + this.lastIntegerZoomTime = now; + } else if (this.lastFloorZoom < floorZ) { + this.lastIntegerZoom = floorZ; + this.lastIntegerZoomTime = now; + } + + if (z !== this.lastZoom) { + this.lastZoom = z; + this.lastFloorZoom = floorZ; + return true; + } + + return false; + } +} diff --git a/web/libraries/maplibre-gl/src/symbol/anchor.test.ts b/web/libraries/maplibre-gl/src/symbol/anchor.test.ts new file mode 100644 index 00000000..1e08a984 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/anchor.test.ts @@ -0,0 +1,14 @@ +import {Anchor} from './anchor'; + +describe('Anchor', () => { + test('#constructor', () => { + expect(new Anchor(0, 0, 0) instanceof Anchor).toBeTruthy(); + expect(new Anchor(0, 0, 0, 0) instanceof Anchor).toBeTruthy(); + }); + test('#clone', () => { + const a = new Anchor(1, 2, 3); + const b = new Anchor(1, 2, 3); + expect(a.clone()).toEqual(b); + expect(a.clone()).toEqual(a); + }); +}); diff --git a/web/libraries/maplibre-gl/src/symbol/anchor.ts b/web/libraries/maplibre-gl/src/symbol/anchor.ts new file mode 100644 index 00000000..04210600 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/anchor.ts @@ -0,0 +1,22 @@ +import Point from '@mapbox/point-geometry'; + +import {register} from '../util/web_worker_transfer'; + +export class Anchor extends Point { + angle: any; + segment?: number; + + constructor(x: number, y: number, angle: number, segment?: number) { + super(x, y); + this.angle = angle; + if (segment !== undefined) { + this.segment = segment; + } + } + + clone() { + return new Anchor(this.x, this.y, this.angle, this.segment); + } +} + +register('Anchor', Anchor); diff --git a/web/libraries/maplibre-gl/src/symbol/check_max_angle.test.ts b/web/libraries/maplibre-gl/src/symbol/check_max_angle.test.ts new file mode 100644 index 00000000..59e32440 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/check_max_angle.test.ts @@ -0,0 +1,54 @@ +import Point from '@mapbox/point-geometry'; +import {checkMaxAngle} from './check_max_angle'; +import {Anchor} from './anchor'; + +describe('checkMaxAngle', () => { + test('line with no sharp angles', () => { + const line = [new Point(0, 0), new Point(20, -1), new Point(40, 1), new Point(60, 0)]; + const anchor = new Anchor(30, 0, 0, 1); + expect(checkMaxAngle(line, anchor, 25, 20, Math.PI / 8)).toBeTruthy(); + expect(checkMaxAngle(line, anchor, 25, 20, 0)).toBeFalsy(); + }); + + test('one sharp corner', () => { + const line = [new Point(0, 0), new Point(0, 10), new Point(10, 10)]; + const anchor = new Anchor(0, 10, 0, 1); + expect(checkMaxAngle(line, anchor, 10, 5, Math.PI / 2)).toBeTruthy(); + expect(checkMaxAngle(line, anchor, 10, 5, Math.PI / 2 - 0.01)).toBeFalsy(); + }); + + test('many small corners close together', () => { + const line = [ + new Point(0, 0), new Point(10, 0), new Point(11, 0.1), + new Point(12, 0.3), new Point(13, 0.6), new Point(14, 1), new Point(13.9, 10)]; + const anchor = new Anchor(12, 0.3, 0, 3); + expect(checkMaxAngle(line, anchor, 10, 5, Math.PI / 2)).toBeFalsy(); + expect(checkMaxAngle(line, anchor, 10, 2, Math.PI / 2)).toBeTruthy(); + }); + + test('label appears on the first line segment', () => { + const line = [new Point(0, 0), new Point(100, 0)]; + const anchor = new Anchor(50, 0, 0, 0); + expect(checkMaxAngle(line, anchor, 30, 5, Math.PI / 2)).toBeTruthy(); + }); + + test('not enough space before the end of the line', () => { + const line = [new Point(0, 0), new Point(10, 0), new Point(20, 0), new Point(30, 0)]; + const anchor = new Anchor(5, 0, 0, 0); + expect(checkMaxAngle(line, anchor, 11, 5, Math.PI)).toBeFalsy(); + expect(checkMaxAngle(line, anchor, 10, 5, Math.PI)).toBeTruthy(); + }); + + test('not enough space after the beginning of the line', () => { + const line = [new Point(0, 0), new Point(10, 0), new Point(20, 0), new Point(30, 0)]; + const anchor = new Anchor(25, 0, 0, 2); + expect(checkMaxAngle(line, anchor, 11, 5, Math.PI)).toBeFalsy(); + expect(checkMaxAngle(line, anchor, 10, 5, Math.PI)).toBeTruthy(); + }); + + test('one segment and label length is 0', () => { + const line = [new Point(0, 0), new Point(10, 0)]; + const anchor = new Anchor(5, 0, 0, 0); + expect(checkMaxAngle(line, anchor, 0, 5, Math.PI)).toBeTruthy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/symbol/check_max_angle.ts b/web/libraries/maplibre-gl/src/symbol/check_max_angle.ts new file mode 100644 index 00000000..e2c867c4 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/check_max_angle.ts @@ -0,0 +1,76 @@ +import type Point from '@mapbox/point-geometry'; +import type {Anchor} from './anchor'; + +/** + * Labels placed around really sharp angles aren't readable. Check if any + * part of the potential label has a combined angle that is too big. + * + * @param line - The line to check + * @param anchor - The point on the line around which the label is anchored. + * @param labelLength - The length of the label in geometry units. + * @param windowSize - The check fails if the combined angles within a part of the line that is `windowSize` long is too big. + * @param maxAngle - The maximum combined angle that any window along the label is allowed to have. + * + * @returns whether the label should be placed + */ +export function checkMaxAngle(line: Array, anchor: Anchor, labelLength: number, windowSize: number, maxAngle: number) { + + // horizontal labels and labels with length 0 always pass + if (anchor.segment === undefined || labelLength === 0) return true; + + let p = anchor; + let index = anchor.segment + 1; + let anchorDistance = 0; + + // move backwards along the line to the first segment the label appears on + while (anchorDistance > -labelLength / 2) { + index--; + + // there isn't enough room for the label after the beginning of the line + if (index < 0) return false; + + anchorDistance -= line[index].dist(p); + p = line[index]; + } + + anchorDistance += line[index].dist(line[index + 1]); + index++; + + // store recent corners and their total angle difference + const recentCorners = []; + let recentAngleDelta = 0; + + // move forwards by the length of the label and check angles along the way + while (anchorDistance < labelLength / 2) { + const prev = line[index - 1]; + const current = line[index]; + const next = line[index + 1]; + + // there isn't enough room for the label before the end of the line + if (!next) return false; + + let angleDelta = prev.angleTo(current) - current.angleTo(next); + // restrict angle to -pi..pi range + angleDelta = Math.abs(((angleDelta + 3 * Math.PI) % (Math.PI * 2)) - Math.PI); + + recentCorners.push({ + distance: anchorDistance, + angleDelta + }); + recentAngleDelta += angleDelta; + + // remove corners that are far enough away from the list of recent anchors + while (anchorDistance - recentCorners[0].distance > windowSize) { + recentAngleDelta -= recentCorners.shift().angleDelta; + } + + // the sum of angles within the window area exceeds the maximum allowed value. check fails. + if (recentAngleDelta > maxAngle) return false; + + index++; + anchorDistance += current.dist(next); + } + + // no part of the line had an angle greater than the maximum allowed. check passes. + return true; +} diff --git a/web/libraries/maplibre-gl/src/symbol/clip_line.test.ts b/web/libraries/maplibre-gl/src/symbol/clip_line.test.ts new file mode 100644 index 00000000..b9ba3b8a --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/clip_line.test.ts @@ -0,0 +1,154 @@ +import Point from '@mapbox/point-geometry'; +import {clipLine} from './clip_line'; + +describe('clipLines', () => { + + const minX = -300; + const maxX = 300; + const minY = -200; + const maxY = 200; + + const clipLineTest = (lines) => { + return clipLine(lines, minX, minY, maxX, maxY); + }; + + test('Single line fully inside', () => { + const line = [ + new Point(-100, -100), + new Point(-40, -100), + new Point(200, 0), + new Point(-80, 195) + ]; + + expect(clipLineTest([line])).toEqual([line]); + }); + + test('Multiline fully inside', () => { + const line0 = [ + new Point(-250, -150), + new Point(-250, 150), + new Point(-10, 150), + new Point(-10, -150) + ]; + + const line1 = [ + new Point(250, -150), + new Point(250, 150), + new Point(10, 150), + new Point(10, -150) + ]; + + const lines = [line0, line1]; + + expect(clipLineTest(lines)).toEqual(lines); + }); + + test('Lines fully outside', () => { + const line0 = [ + new Point(-400, -300), + new Point(-350, 0), + new Point(-300, 300) + ]; + + const line1 = [ + new Point(1000, 210), + new Point(10000, 500) + ]; + + expect(clipLineTest([line0, line1])).toEqual([]); + }); + + test('Intersect with single border', () => { + const line0 = [ + new Point(-400, 0), + new Point(0, 0) + ]; + + const result0 = [ + new Point(minX, 0), + new Point(0, 0) + ]; + + const line1 = [ + new Point(250, -50), + new Point(350, 50) + ]; + + const result1 = [ + new Point(250, -50), + new Point(maxX, 0) + ]; + + expect(clipLineTest([line0, line1])).toEqual([result0, result1]); + }); + + test('Intersect with multiple borders', () => { + const line0 = [ + new Point(-350, -100), + new Point(-200, -250) + ]; + + const line1 = [ + new Point(-100, 250), + new Point(0, 150), + new Point(100, 250) + ]; + + const result0 = [ + new Point(minX, -150), + new Point(-250, minY) + ]; + + const result1 = [ + new Point(-50, maxY), + new Point(0, 150), + new Point(50, maxY) + ]; + + expect(clipLineTest([line0, line1])).toEqual([result0, result1]); + }); + + test('Single line can be split into multiple segments', () => { + const line = [ + new Point(-80, 150), + new Point(-80, 350), + new Point(120, 1000), + new Point(120, 0) + ]; + + const result0 = [ + new Point(-80, 150), + new Point(-80, maxY), + ]; + + const result1 = [ + new Point(120, maxY), + new Point(120, 0), + ]; + + expect(clipLineTest([line])).toEqual([result0, result1]); + }); + + test('Non-clipped points are bit exact', () => { + const line = [ + new Point(-500, -200), + new Point(131.2356763, 0.956732) + ]; + + expect(clipLineTest([line])[1]).toEqual(line[0][1]); + }); + + test('Clipped points are rounded to the nearest integer', () => { + const line = [ + new Point(310, 2.9), + new Point(290, 2.5) + ]; + + const result = [ + new Point(maxX, 3), + new Point(290, 2.5) + ]; + + expect(clipLineTest([line])).toEqual([result]); + }); +}); diff --git a/web/libraries/maplibre-gl/src/symbol/clip_line.ts b/web/libraries/maplibre-gl/src/symbol/clip_line.ts new file mode 100644 index 00000000..457955b8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/clip_line.ts @@ -0,0 +1,66 @@ +import Point from '@mapbox/point-geometry'; + +/** + * Returns the part of a multiline that intersects with the provided rectangular box. + * + * @param lines - the lines to check + * @param x1 - the left edge of the box + * @param y1 - the top edge of the box + * @param x2 - the right edge of the box + * @param y2 - the bottom edge of the box + * @returns lines + */ +export function clipLine(lines: Array>, x1: number, y1: number, x2: number, y2: number): Array> { + const clippedLines = []; + + for (let l = 0; l < lines.length; l++) { + const line = lines[l]; + let clippedLine; + + for (let i = 0; i < line.length - 1; i++) { + let p0 = line[i]; + let p1 = line[i + 1]; + + if (p0.x < x1 && p1.x < x1) { + continue; + } else if (p0.x < x1) { + p0 = new Point(x1, p0.y + (p1.y - p0.y) * ((x1 - p0.x) / (p1.x - p0.x)))._round(); + } else if (p1.x < x1) { + p1 = new Point(x1, p0.y + (p1.y - p0.y) * ((x1 - p0.x) / (p1.x - p0.x)))._round(); + } + + if (p0.y < y1 && p1.y < y1) { + continue; + } else if (p0.y < y1) { + p0 = new Point(p0.x + (p1.x - p0.x) * ((y1 - p0.y) / (p1.y - p0.y)), y1)._round(); + } else if (p1.y < y1) { + p1 = new Point(p0.x + (p1.x - p0.x) * ((y1 - p0.y) / (p1.y - p0.y)), y1)._round(); + } + + if (p0.x >= x2 && p1.x >= x2) { + continue; + } else if (p0.x >= x2) { + p0 = new Point(x2, p0.y + (p1.y - p0.y) * ((x2 - p0.x) / (p1.x - p0.x)))._round(); + } else if (p1.x >= x2) { + p1 = new Point(x2, p0.y + (p1.y - p0.y) * ((x2 - p0.x) / (p1.x - p0.x)))._round(); + } + + if (p0.y >= y2 && p1.y >= y2) { + continue; + } else if (p0.y >= y2) { + p0 = new Point(p0.x + (p1.x - p0.x) * ((y2 - p0.y) / (p1.y - p0.y)), y2)._round(); + } else if (p1.y >= y2) { + p1 = new Point(p0.x + (p1.x - p0.x) * ((y2 - p0.y) / (p1.y - p0.y)), y2)._round(); + } + + if (!clippedLine || !p0.equals(clippedLine[clippedLine.length - 1])) { + clippedLine = [p0]; + clippedLines.push(clippedLine); + } + + clippedLine.push(p1); + } + } + + return clippedLines; +} diff --git a/web/libraries/maplibre-gl/src/symbol/collision_feature.test.ts b/web/libraries/maplibre-gl/src/symbol/collision_feature.test.ts new file mode 100644 index 00000000..362ca96c --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/collision_feature.test.ts @@ -0,0 +1,98 @@ +import {CollisionFeature} from './collision_feature'; +import {Anchor} from './anchor'; +import Point from '@mapbox/point-geometry'; +import {CollisionBoxArray} from '../data/array_types.g'; +import {SymbolPadding} from '../style/style_layer/symbol_style_layer'; + +describe('CollisionFeature', () => { + + const collisionBoxArray = new CollisionBoxArray(); + + const shapedText = { + left: -50, + top: -10, + right: 50, + bottom: 10 + }; + + const padding: SymbolPadding = [0, 0, 0, 0]; + + test('point label', () => { + const point = new Point(500, 0); + const anchor = new Anchor(point.x, point.y, 0, undefined); + + const cf = new CollisionFeature(collisionBoxArray, anchor, 0, 0, 0, shapedText, 1, padding, false, 0); + expect(cf.circleDiameter).toBeFalsy(); + expect(cf.boxEndIndex - cf.boxStartIndex).toBe(1); + + const box = collisionBoxArray.get(cf.boxStartIndex); + expect(box.x1).toBe(-50); + expect(box.x2).toBe(50); + expect(box.y1).toBe(-10); + expect(box.y2).toBe(10); + }); + + test('point label with padding', () => { + const point = new Point(500, 0); + const anchor = new Anchor(point.x, point.y, 0, undefined); + const pointPadding: SymbolPadding = [10, 20, -5, -10]; // top, right, bottom, left + const cf = new CollisionFeature(collisionBoxArray, anchor, 0, 0, 0, shapedText, 1, pointPadding, false, 0); + + expect(cf.boxEndIndex - cf.boxStartIndex).toBe(1); + const box = collisionBoxArray.get(cf.boxStartIndex); + expect(box.x1).toBe(-40); + expect(box.x2).toBe(70); + expect(box.y1).toBe(-20); + expect(box.y2).toBe(5); + }); + + test('Compute line height for runtime collision circles (line label)', () => { + const anchor = new Anchor(505, 95, 0, 1); + const cf = new CollisionFeature(collisionBoxArray, anchor, 0, 0, 0, shapedText, 1, padding, true, 0); + expect(cf.circleDiameter).toBeTruthy(); + expect(cf.circleDiameter).toBe(shapedText.bottom - shapedText.top); + expect(cf.boxEndIndex - cf.boxStartIndex).toBe(0); + }); + + test('Collision circle diameter is not computed for features with zero height', () => { + const shapedText = { + left: -50, + top: -10, + right: 50, + bottom: -10 + }; + + const anchor = new Anchor(505, 95, 0, 1); + const cf = new CollisionFeature(collisionBoxArray, anchor, 0, 0, 0, shapedText, 1, padding, true, 0); + expect(cf.boxEndIndex - cf.boxStartIndex).toBe(0); + expect(cf.circleDiameter).toBeFalsy(); + }); + + test('Collision circle diameter is not computed for features with negative height', () => { + const shapedText = { + left: -50, + top: 10, + right: 50, + bottom: -10 + }; + + const anchor = new Anchor(505, 95, 0, 1); + const cf = new CollisionFeature(collisionBoxArray, anchor, 0, 0, 0, shapedText, 1, padding, true, 0); + expect(cf.boxEndIndex - cf.boxStartIndex).toBe(0); + expect(cf.circleDiameter).toBeFalsy(); + }); + + test('Use minimum collision circle diameter', () => { + const shapedText = { + left: -50, + top: 10, + right: 50, + bottom: 10.00001 + }; + + const anchor = new Anchor(505, 95, 0, 1); + const cf = new CollisionFeature(collisionBoxArray, anchor, 0, 0, 0, shapedText, 1, padding, true, 0); + expect(cf.boxEndIndex - cf.boxStartIndex).toBe(0); + expect(cf.circleDiameter).toBe(10); + }); +}); diff --git a/web/libraries/maplibre-gl/src/symbol/collision_feature.ts b/web/libraries/maplibre-gl/src/symbol/collision_feature.ts new file mode 100644 index 00000000..8fca28d9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/collision_feature.ts @@ -0,0 +1,104 @@ +import type {CollisionBoxArray} from '../data/array_types.g'; +import Point from '@mapbox/point-geometry'; +import type {Anchor} from './anchor'; +import {SymbolPadding} from '../style/style_layer/symbol_style_layer'; + +/** + * A CollisionFeature represents the area of the tile covered by a single label. + * It is used with CollisionIndex to check if the label overlaps with any + * previous labels. A CollisionFeature is mostly just a set of CollisionBox + * objects. + */ +export class CollisionFeature { + boxStartIndex: number; + boxEndIndex: number; + circleDiameter: number; + + /** + * Create a CollisionFeature, adding its collision box data to the given collisionBoxArray in the process. + * For line aligned labels a collision circle diameter is computed instead. + * + * @param anchor - The point along the line around which the label is anchored. + * @param shaped - The text or icon shaping results. + * @param boxScale - A magic number used to convert from glyph metrics units to geometry units. + * @param padding - The amount of padding to add around the label edges. + * @param alignLine - Whether the label is aligned with the line or the viewport. + */ + constructor(collisionBoxArray: CollisionBoxArray, + anchor: Anchor, + featureIndex: number, + sourceLayerIndex: number, + bucketIndex: number, + shaped: any, + boxScale: number, + padding: SymbolPadding, + alignLine: boolean, + rotate: number) { + + this.boxStartIndex = collisionBoxArray.length; + + if (alignLine) { + // Compute height of the shape in glyph metrics and apply collision padding. + // Note that the pixel based 'text-padding' is applied at runtime + let top = shaped.top; + let bottom = shaped.bottom; + const collisionPadding = shaped.collisionPadding; + + if (collisionPadding) { + top -= collisionPadding[1]; + bottom += collisionPadding[3]; + } + + let height = bottom - top; + + if (height > 0) { + // set minimum box height to avoid very many small labels + height = Math.max(10, height); + this.circleDiameter = height; + } + } else { + // margin is in CSS order: [top, right, bottom, left] + let y1 = shaped.top * boxScale - padding[0]; + let y2 = shaped.bottom * boxScale + padding[2]; + let x1 = shaped.left * boxScale - padding[3]; + let x2 = shaped.right * boxScale + padding[1]; + + const collisionPadding = shaped.collisionPadding; + if (collisionPadding) { + x1 -= collisionPadding[0] * boxScale; + y1 -= collisionPadding[1] * boxScale; + x2 += collisionPadding[2] * boxScale; + y2 += collisionPadding[3] * boxScale; + } + + if (rotate) { + // Account for *-rotate in point collision boxes + // See https://github.com/mapbox/mapbox-gl-js/issues/6075 + // Doesn't account for icon-text-fit + + const tl = new Point(x1, y1); + const tr = new Point(x2, y1); + const bl = new Point(x1, y2); + const br = new Point(x2, y2); + + const rotateRadians = rotate * Math.PI / 180; + + tl._rotate(rotateRadians); + tr._rotate(rotateRadians); + bl._rotate(rotateRadians); + br._rotate(rotateRadians); + + // Collision features require an "on-axis" geometry, + // so take the envelope of the rotated geometry + // (may be quite large for wide labels rotated 45 degrees) + x1 = Math.min(tl.x, tr.x, bl.x, br.x); + x2 = Math.max(tl.x, tr.x, bl.x, br.x); + y1 = Math.min(tl.y, tr.y, bl.y, br.y); + y2 = Math.max(tl.y, tr.y, bl.y, br.y); + } + collisionBoxArray.emplaceBack(anchor.x, anchor.y, x1, y1, x2, y2, featureIndex, sourceLayerIndex, bucketIndex); + } + + this.boxEndIndex = collisionBoxArray.length; + } +} diff --git a/web/libraries/maplibre-gl/src/symbol/collision_index.test.ts b/web/libraries/maplibre-gl/src/symbol/collision_index.test.ts new file mode 100644 index 00000000..fd281e66 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/collision_index.test.ts @@ -0,0 +1,18 @@ +import {CollisionIndex} from './collision_index'; +import {mat4} from 'gl-matrix'; + +import {Transform} from '../geo/transform'; + +describe('CollisionIndex', () => { + + test('floating point precision', () => { + const posMatrix = mat4.create(); + const x = 100000.123456, y = 0; + const transform = new Transform(0, 22, 0, 60, true); + transform.resize(200, 200); + + const ci = new CollisionIndex(transform); + expect(ci.projectAndGetPerspectiveRatio(posMatrix, x, y).point.x).toBeCloseTo(10000212.3456, 10); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/symbol/collision_index.ts b/web/libraries/maplibre-gl/src/symbol/collision_index.ts new file mode 100644 index 00000000..e889fc4e --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/collision_index.ts @@ -0,0 +1,401 @@ +import Point from '@mapbox/point-geometry'; +import {clipLine} from './clip_line'; +import {PathInterpolator} from './path_interpolator'; + +import * as intersectionTests from '../util/intersection_tests'; +import {GridIndex} from './grid_index'; +import {mat4, vec4} from 'gl-matrix'; +import ONE_EM from '../symbol/one_em'; + +import * as projection from '../symbol/projection'; + +import type {Transform} from '../geo/transform'; +import type {SingleCollisionBox} from '../data/bucket/symbol_bucket'; +import type { + GlyphOffsetArray, + SymbolLineVertexArray +} from '../data/array_types.g'; +import type {OverlapMode} from '../style/style_layer/overlap_mode'; + +// When a symbol crosses the edge that causes it to be included in +// collision detection, it will cause changes in the symbols around +// it. This constant specifies how many pixels to pad the edge of +// the viewport for collision detection so that the bulk of the changes +// occur offscreen. Making this constant greater increases label +// stability, but it's expensive. +const viewportPadding = 100; + +export type FeatureKey = { + bucketInstanceId: number; + featureIndex: number; + collisionGroupID: number; + overlapMode: OverlapMode; +}; + +/** + * @internal + * A collision index used to prevent symbols from overlapping. It keep tracks of + * where previous symbols have been placed and is used to check if a new + * symbol overlaps with any previously added symbols. + * + * There are two steps to insertion: first placeCollisionBox/Circles checks if + * there's room for a symbol, then insertCollisionBox/Circles actually puts the + * symbol in the index. The two step process allows paired symbols to be inserted + * together even if they overlap. + */ +export class CollisionIndex { + grid: GridIndex; + ignoredGrid: GridIndex; + transform: Transform; + pitchfactor: number; + screenRightBoundary: number; + screenBottomBoundary: number; + gridRightBoundary: number; + gridBottomBoundary: number; + + // With perspectiveRatio the fontsize is calculated for tilted maps (near = bigger, far = smaller). + // The cutoff defines a threshold to no longer render labels near the horizon. + perspectiveRatioCutoff: number; + + constructor( + transform: Transform, + grid = new GridIndex(transform.width + 2 * viewportPadding, transform.height + 2 * viewportPadding, 25), + ignoredGrid = new GridIndex(transform.width + 2 * viewportPadding, transform.height + 2 * viewportPadding, 25) + ) { + this.transform = transform; + + this.grid = grid; + this.ignoredGrid = ignoredGrid; + this.pitchfactor = Math.cos(transform._pitch) * transform.cameraToCenterDistance; + + this.screenRightBoundary = transform.width + viewportPadding; + this.screenBottomBoundary = transform.height + viewportPadding; + this.gridRightBoundary = transform.width + 2 * viewportPadding; + this.gridBottomBoundary = transform.height + 2 * viewportPadding; + + this.perspectiveRatioCutoff = 0.6; + } + + placeCollisionBox( + collisionBox: SingleCollisionBox, + overlapMode: OverlapMode, + textPixelRatio: number, + posMatrix: mat4, + collisionGroupPredicate?: (key: FeatureKey) => boolean, + getElevation?: (x: number, y: number) => number + ): { + box: Array; + offscreen: boolean; + } { + const projectedPoint = this.projectAndGetPerspectiveRatio(posMatrix, collisionBox.anchorPointX, collisionBox.anchorPointY, getElevation); + const tileToViewport = textPixelRatio * projectedPoint.perspectiveRatio; + const tlX = collisionBox.x1 * tileToViewport + projectedPoint.point.x; + const tlY = collisionBox.y1 * tileToViewport + projectedPoint.point.y; + const brX = collisionBox.x2 * tileToViewport + projectedPoint.point.x; + const brY = collisionBox.y2 * tileToViewport + projectedPoint.point.y; + + if (!this.isInsideGrid(tlX, tlY, brX, brY) || + (overlapMode !== 'always' && this.grid.hitTest(tlX, tlY, brX, brY, overlapMode, collisionGroupPredicate)) || + projectedPoint.perspectiveRatio < this.perspectiveRatioCutoff) { + return { + box: [], + offscreen: false + }; + } + + return { + box: [tlX, tlY, brX, brY], + offscreen: this.isOffscreen(tlX, tlY, brX, brY) + }; + } + + placeCollisionCircles( + overlapMode: OverlapMode, + symbol: any, + lineVertexArray: SymbolLineVertexArray, + glyphOffsetArray: GlyphOffsetArray, + fontSize: number, + posMatrix: mat4, + labelPlaneMatrix: mat4, + labelToScreenMatrix: mat4, + showCollisionCircles: boolean, + pitchWithMap: boolean, + collisionGroupPredicate: (key: FeatureKey) => boolean, + circlePixelDiameter: number, + textPixelPadding: number, + getElevation: (x: number, y: number) => number + ): { + circles: Array; + offscreen: boolean; + collisionDetected: boolean; + } { + const placedCollisionCircles = []; + + const tileUnitAnchorPoint = new Point(symbol.anchorX, symbol.anchorY); + const screenAnchorPoint = projection.project(tileUnitAnchorPoint, posMatrix, getElevation); + const perspectiveRatio = projection.getPerspectiveRatio(this.transform.cameraToCenterDistance, screenAnchorPoint.signedDistanceFromCamera); + const labelPlaneFontSize = pitchWithMap ? fontSize / perspectiveRatio : fontSize * perspectiveRatio; + const labelPlaneFontScale = labelPlaneFontSize / ONE_EM; + + const labelPlaneAnchorPoint = projection.project(tileUnitAnchorPoint, labelPlaneMatrix, getElevation).point; + + const projectionCache = {projections: {}, offsets: {}}; + const lineOffsetX = symbol.lineOffsetX * labelPlaneFontScale; + const lineOffsetY = symbol.lineOffsetY * labelPlaneFontScale; + + const firstAndLastGlyph = projection.placeFirstAndLastGlyph( + labelPlaneFontScale, + glyphOffsetArray, + lineOffsetX, + lineOffsetY, + /*flip*/ false, + labelPlaneAnchorPoint, + tileUnitAnchorPoint, + symbol, + lineVertexArray, + labelPlaneMatrix, + projectionCache, + false, + getElevation); + + let collisionDetected = false; + let inGrid = false; + let entirelyOffscreen = true; + + if (firstAndLastGlyph) { + const radius = circlePixelDiameter * 0.5 * perspectiveRatio + textPixelPadding; + const screenPlaneMin = new Point(-viewportPadding, -viewportPadding); + const screenPlaneMax = new Point(this.screenRightBoundary, this.screenBottomBoundary); + const interpolator = new PathInterpolator(); + + // Construct a projected path from projected line vertices. Anchor points are ignored and removed + const first = firstAndLastGlyph.first; + const last = firstAndLastGlyph.last; + + let projectedPath = []; + for (let i = first.path.length - 1; i >= 1; i--) { + projectedPath.push(first.path[i]); + } + for (let i = 1; i < last.path.length; i++) { + projectedPath.push(last.path[i]); + } + + // Tolerate a slightly longer distance than one diameter between two adjacent circles + const circleDist = radius * 2.5; + + // The path might need to be converted into screen space if a pitched map is used as the label space + if (labelToScreenMatrix) { + const screenSpacePath = projectedPath.map(p => projection.project(p, labelToScreenMatrix, getElevation)); + + // Do not try to place collision circles if even of the points is behind the camera. + // This is a plausible scenario with big camera pitch angles + if (screenSpacePath.some(point => point.signedDistanceFromCamera <= 0)) { + projectedPath = []; + } else { + projectedPath = screenSpacePath.map(p => p.point); + } + } + + let segments = []; + + if (projectedPath.length > 0) { + // Quickly check if the path is fully inside or outside of the padded collision region. + // For overlapping paths we'll only create collision circles for the visible segments + const minPoint = projectedPath[0].clone(); + const maxPoint = projectedPath[0].clone(); + + for (let i = 1; i < projectedPath.length; i++) { + minPoint.x = Math.min(minPoint.x, projectedPath[i].x); + minPoint.y = Math.min(minPoint.y, projectedPath[i].y); + maxPoint.x = Math.max(maxPoint.x, projectedPath[i].x); + maxPoint.y = Math.max(maxPoint.y, projectedPath[i].y); + } + + if (minPoint.x >= screenPlaneMin.x && maxPoint.x <= screenPlaneMax.x && + minPoint.y >= screenPlaneMin.y && maxPoint.y <= screenPlaneMax.y) { + // Quad fully visible + segments = [projectedPath]; + } else if (maxPoint.x < screenPlaneMin.x || minPoint.x > screenPlaneMax.x || + maxPoint.y < screenPlaneMin.y || minPoint.y > screenPlaneMax.y) { + // Not visible + segments = []; + } else { + segments = clipLine([projectedPath], screenPlaneMin.x, screenPlaneMin.y, screenPlaneMax.x, screenPlaneMax.y); + } + } + + for (const seg of segments) { + // interpolate positions for collision circles. Add a small padding to both ends of the segment + interpolator.reset(seg, radius * 0.25); + + let numCircles = 0; + + if (interpolator.length <= 0.5 * radius) { + numCircles = 1; + } else { + numCircles = Math.ceil(interpolator.paddedLength / circleDist) + 1; + } + + for (let i = 0; i < numCircles; i++) { + const t = i / Math.max(numCircles - 1, 1); + const circlePosition = interpolator.lerp(t); + + // add viewport padding to the position and perform initial collision check + const centerX = circlePosition.x + viewportPadding; + const centerY = circlePosition.y + viewportPadding; + + placedCollisionCircles.push(centerX, centerY, radius, 0); + + const x1 = centerX - radius; + const y1 = centerY - radius; + const x2 = centerX + radius; + const y2 = centerY + radius; + + entirelyOffscreen = entirelyOffscreen && this.isOffscreen(x1, y1, x2, y2); + inGrid = inGrid || this.isInsideGrid(x1, y1, x2, y2); + + if (overlapMode !== 'always' && this.grid.hitTestCircle(centerX, centerY, radius, overlapMode, collisionGroupPredicate)) { + // Don't early exit if we're showing the debug circles because we still want to calculate + // which circles are in use + collisionDetected = true; + if (!showCollisionCircles) { + return { + circles: [], + offscreen: false, + collisionDetected + }; + } + } + } + } + } + + return { + circles: ((!showCollisionCircles && collisionDetected) || !inGrid || perspectiveRatio < this.perspectiveRatioCutoff) ? [] : placedCollisionCircles, + offscreen: entirelyOffscreen, + collisionDetected + }; + } + + /** + * Because the geometries in the CollisionIndex are an approximation of the shape of + * symbols on the map, we use the CollisionIndex to look up the symbol part of + * `queryRenderedFeatures`. + */ + queryRenderedSymbols(viewportQueryGeometry: Array) { + if (viewportQueryGeometry.length === 0 || (this.grid.keysLength() === 0 && this.ignoredGrid.keysLength() === 0)) { + return {}; + } + + const query = []; + let minX = Infinity; + let minY = Infinity; + let maxX = -Infinity; + let maxY = -Infinity; + for (const point of viewportQueryGeometry) { + const gridPoint = new Point(point.x + viewportPadding, point.y + viewportPadding); + minX = Math.min(minX, gridPoint.x); + minY = Math.min(minY, gridPoint.y); + maxX = Math.max(maxX, gridPoint.x); + maxY = Math.max(maxY, gridPoint.y); + query.push(gridPoint); + } + + const features = this.grid.query(minX, minY, maxX, maxY) + .concat(this.ignoredGrid.query(minX, minY, maxX, maxY)); + + const seenFeatures = {}; + const result = {}; + + for (const feature of features) { + const featureKey = feature.key; + // Skip already seen features. + if (seenFeatures[featureKey.bucketInstanceId] === undefined) { + seenFeatures[featureKey.bucketInstanceId] = {}; + } + if (seenFeatures[featureKey.bucketInstanceId][featureKey.featureIndex]) { + continue; + } + + // Check if query intersects with the feature box + // "Collision Circles" for line labels are treated as boxes here + // Since there's no actual collision taking place, the circle vs. square + // distinction doesn't matter as much, and box geometry is easier + // to work with. + const bbox = [ + new Point(feature.x1, feature.y1), + new Point(feature.x2, feature.y1), + new Point(feature.x2, feature.y2), + new Point(feature.x1, feature.y2) + ]; + if (!intersectionTests.polygonIntersectsPolygon(query, bbox)) { + continue; + } + + seenFeatures[featureKey.bucketInstanceId][featureKey.featureIndex] = true; + if (result[featureKey.bucketInstanceId] === undefined) { + result[featureKey.bucketInstanceId] = []; + } + result[featureKey.bucketInstanceId].push(featureKey.featureIndex); + } + + return result; + } + + insertCollisionBox(collisionBox: Array, overlapMode: OverlapMode, ignorePlacement: boolean, bucketInstanceId: number, featureIndex: number, collisionGroupID: number) { + const grid = ignorePlacement ? this.ignoredGrid : this.grid; + + const key = {bucketInstanceId, featureIndex, collisionGroupID, overlapMode}; + grid.insert(key, collisionBox[0], collisionBox[1], collisionBox[2], collisionBox[3]); + } + + insertCollisionCircles(collisionCircles: Array, overlapMode: OverlapMode, ignorePlacement: boolean, bucketInstanceId: number, featureIndex: number, collisionGroupID: number) { + const grid = ignorePlacement ? this.ignoredGrid : this.grid; + + const key = {bucketInstanceId, featureIndex, collisionGroupID, overlapMode}; + for (let k = 0; k < collisionCircles.length; k += 4) { + grid.insertCircle(key, collisionCircles[k], collisionCircles[k + 1], collisionCircles[k + 2]); + } + } + + projectAndGetPerspectiveRatio(posMatrix: mat4, x: number, y: number, getElevation?: (x: number, y: number) => number) { + let p; + if (getElevation) { // slow because of handle z-index + p = [x, y, getElevation(x, y), 1] as vec4; + vec4.transformMat4(p, p, posMatrix); + } else { // fast because of ignore z-index + p = [x, y, 0, 1] as vec4; + projection.xyTransformMat4(p, p, posMatrix); + } + const a = new Point( + (((p[0] / p[3] + 1) / 2) * this.transform.width) + viewportPadding, + (((-p[1] / p[3] + 1) / 2) * this.transform.height) + viewportPadding + ); + return { + point: a, + // See perspective ratio comment in symbol_sdf.vertex + // We're doing collision detection in viewport space so we need + // to scale down boxes in the distance + perspectiveRatio: 0.5 + 0.5 * (this.transform.cameraToCenterDistance / p[3]) + }; + } + + isOffscreen(x1: number, y1: number, x2: number, y2: number) { + return x2 < viewportPadding || x1 >= this.screenRightBoundary || y2 < viewportPadding || y1 > this.screenBottomBoundary; + } + + isInsideGrid(x1: number, y1: number, x2: number, y2: number) { + return x2 >= 0 && x1 < this.gridRightBoundary && y2 >= 0 && y1 < this.gridBottomBoundary; + } + + /* + * Returns a matrix for transforming collision shapes to viewport coordinate space. + * Use this function to render e.g. collision circles on the screen. + * example transformation: clipPos = glCoordMatrix * viewportMatrix * circle_pos + */ + getViewportMatrix() { + const m = mat4.identity([] as any); + mat4.translate(m, m, [-viewportPadding, -viewportPadding, 0.0]); + return m; + } +} diff --git a/web/libraries/maplibre-gl/src/symbol/cross_tile_symbol_index.test.ts b/web/libraries/maplibre-gl/src/symbol/cross_tile_symbol_index.test.ts new file mode 100644 index 00000000..ad07b7a9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/cross_tile_symbol_index.test.ts @@ -0,0 +1,260 @@ +import {CrossTileSymbolIndex, KDBUSH_THRESHHOLD} from './cross_tile_symbol_index'; +import {OverscaledTileID} from '../source/tile_id'; +import {StyleLayer} from '../style/style_layer'; + +const styleLayer = { + id: 'test' +} as StyleLayer; + +const makeSymbolInstance = (x, y, key): any => { + return { + anchorX: x, + anchorY: y, + key + }; +}; + +const makeTile = (tileID, symbolInstances): any => { + const bucket = { + symbolInstances: { + get(i) { return symbolInstances[i]; }, + length: symbolInstances.length + }, + layerIds: ['test'] + }; + return { + tileID, + getBucket: () => bucket, + latestFeatureIndex: {} + }; +}; + +describe('CrossTileSymbolIndex.addLayer', () => { + + test('matches ids', () => { + const index = new CrossTileSymbolIndex(); + + const mainID = new OverscaledTileID(6, 0, 6, 8, 8); + const mainInstances = [ + makeSymbolInstance(1000, 1000, 'Detroit'), + makeSymbolInstance(2000, 2000, 'Toronto') + ]; + const mainTile = makeTile(mainID, mainInstances); + + index.addLayer(styleLayer, [mainTile], 0); + // Assigned new IDs + expect(mainInstances[0].crossTileID).toBe(1); + expect(mainInstances[1].crossTileID).toBe(2); + + const childID = new OverscaledTileID(7, 0, 7, 16, 16); + const childInstances = [ + makeSymbolInstance(2000, 2000, 'Detroit'), + makeSymbolInstance(2000, 2000, 'Windsor'), + makeSymbolInstance(3000, 3000, 'Toronto'), + makeSymbolInstance(4001, 4001, 'Toronto') + ]; + const childTile = makeTile(childID, childInstances); + + index.addLayer(styleLayer, [mainTile, childTile], 0); + // matched parent tile + expect(childInstances[0].crossTileID).toBe(1); + // does not match because of different key + expect(childInstances[1].crossTileID).toBe(3); + // does not match because of different location + expect(childInstances[2].crossTileID).toBe(4); + // matches with a slightly different location + expect(childInstances[3].crossTileID).toBe(2); + + const parentID = new OverscaledTileID(5, 0, 5, 4, 4); + const parentInstances = [ + makeSymbolInstance(500, 500, 'Detroit') + ]; + const parentTile = makeTile(parentID, parentInstances); + + index.addLayer(styleLayer, [mainTile, childTile, parentTile], 0); + // matched child tile + expect(parentInstances[0].crossTileID).toBe(1); + + const grandchildID = new OverscaledTileID(8, 0, 8, 32, 32); + const grandchildInstances = [ + makeSymbolInstance(4000, 4000, 'Detroit'), + makeSymbolInstance(4000, 4000, 'Windsor') + ]; + const grandchildTile = makeTile(grandchildID, grandchildInstances); + + index.addLayer(styleLayer, [mainTile], 0); + index.addLayer(styleLayer, [mainTile, grandchildTile], 0); + // Matches the symbol in `mainBucket` + expect(grandchildInstances[0].crossTileID).toBe(1); + // Does not match the previous value for Windsor because that tile was removed + expect(grandchildInstances[1].crossTileID).toBe(5); + + }); + + test('overwrites ids when re-adding', () => { + const index = new CrossTileSymbolIndex(); + + const mainID = new OverscaledTileID(6, 0, 6, 8, 8); + const mainInstances = [makeSymbolInstance(1000, 1000, 'Detroit')]; + const mainTile = makeTile(mainID, mainInstances); + + const childID = new OverscaledTileID(7, 0, 7, 16, 16); + const childInstances = [makeSymbolInstance(2000, 2000, 'Detroit')]; + const childTile = makeTile(childID, childInstances); + + // assigns a new id + index.addLayer(styleLayer, [mainTile], 0); + expect(mainInstances[0].crossTileID).toBe(1); + + // removes the tile + index.addLayer(styleLayer, [], 0); + + // assigns a new id + index.addLayer(styleLayer, [childTile], 0); + expect(childInstances[0].crossTileID).toBe(2); + + // overwrites the old id to match the already-added tile + index.addLayer(styleLayer, [mainTile, childTile], 0); + expect(mainInstances[0].crossTileID).toBe(2); + expect(childInstances[0].crossTileID).toBe(2); + + }); + + test('does not duplicate ids within one zoom level', () => { + const index = new CrossTileSymbolIndex(); + + const mainID = new OverscaledTileID(6, 0, 6, 8, 8); + const mainInstances = [ + makeSymbolInstance(1000, 1000, ''), // A + makeSymbolInstance(1000, 1000, '') // B + ]; + const mainTile = makeTile(mainID, mainInstances); + + const childID = new OverscaledTileID(7, 0, 7, 16, 16); + const childInstances = [ + makeSymbolInstance(2000, 2000, ''), // A' + makeSymbolInstance(2000, 2000, ''), // B' + makeSymbolInstance(2000, 2000, '') // C' + ]; + const childTile = makeTile(childID, childInstances); + + // assigns new ids + index.addLayer(styleLayer, [mainTile], 0); + expect(mainInstances[0].crossTileID).toBe(1); + expect(mainInstances[1].crossTileID).toBe(2); + + const layerIndex = index.layerIndexes[styleLayer.id]; + expect(Object.keys(layerIndex.usedCrossTileIDs[6])).toEqual(['1', '2']); + + // copies parent ids without duplicate ids in this tile + index.addLayer(styleLayer, [childTile], 0); + expect(childInstances[0].crossTileID).toBe(1); // A' copies from A + expect(childInstances[1].crossTileID).toBe(2); // B' copies from B + expect(childInstances[2].crossTileID).toBe(3); // C' gets new ID + + // Updates per-zoom usedCrossTileIDs + expect(Object.keys(layerIndex.usedCrossTileIDs[6])).toEqual([]); + expect(Object.keys(layerIndex.usedCrossTileIDs[7])).toEqual(['1', '2', '3']); + + }); + + test('does not regenerate ids for same zoom', () => { + const index = new CrossTileSymbolIndex(); + + const tileID = new OverscaledTileID(6, 0, 6, 8, 8); + const firstInstances = [ + makeSymbolInstance(1000, 1000, ''), // A + makeSymbolInstance(1000, 1000, '') // B + ]; + const firstTile = makeTile(tileID, firstInstances); + + const secondInstances = [ + makeSymbolInstance(1000, 1000, ''), // A' + makeSymbolInstance(1000, 1000, ''), // B' + makeSymbolInstance(1000, 1000, ''), // C' + ]; + const secondTile = makeTile(tileID, secondInstances); + + // assigns new ids + index.addLayer(styleLayer, [firstTile], 0); + expect(firstInstances[0].crossTileID).toBe(1); + expect(firstInstances[1].crossTileID).toBe(2); + + const layerIndex = index.layerIndexes[styleLayer.id]; + expect(Object.keys(layerIndex.usedCrossTileIDs[6])).toEqual(['1', '2']); + + // uses same ids when tile gets updated + index.addLayer(styleLayer, [secondTile], 0); + expect(secondInstances[0].crossTileID).toBe(1); // A' copies from A + expect(secondInstances[1].crossTileID).toBe(2); // B' copies from B + expect(secondInstances[2].crossTileID).toBe(3); // C' gets new ID + + expect(Object.keys(layerIndex.usedCrossTileIDs[6])).toEqual(['1', '2', '3']); + + }); + + test('reuses indexes when longitude is wrapped', () => { + const index = new CrossTileSymbolIndex(); + const longitude = 370; + + const tileID = new OverscaledTileID(6, 1, 6, 8, 8); + const firstInstances = [ + makeSymbolInstance(1000, 1000, ''), // A + ]; + const tile = makeTile(tileID, firstInstances); + + index.addLayer(styleLayer, [tile], longitude); + expect(firstInstances[0].crossTileID).toBe(1); // A + + tile.tileID = tileID.wrapped(); + + index.addLayer(styleLayer, [tile], longitude % 360); + expect(firstInstances[0].crossTileID).toBe(1); + + }); + + test('indexes data for findMatches perf', () => { + const index = new CrossTileSymbolIndex(); + + const mainID = new OverscaledTileID(6, 0, 6, 8, 8); + const childID = new OverscaledTileID(7, 0, 7, 16, 16); + + const mainInstances: any[] = []; + const childInstances: any[] = []; + + for (let i = 0; i < KDBUSH_THRESHHOLD + 1; i++) { + mainInstances.push(makeSymbolInstance(0, 0, '')); + childInstances.push(makeSymbolInstance(0, 0, '')); + } + const mainTile = makeTile(mainID, mainInstances); + const childTile = makeTile(childID, childInstances); + index.addLayer(styleLayer, [mainTile], 0); + index.addLayer(styleLayer, [childTile], 0); + + // check that we matched the parent tile + expect(childInstances[0].crossTileID).toBe(1); + + }); +}); + +describe('CrossTileSymbolIndex.pruneUnusedLayers', () => { + const index = new CrossTileSymbolIndex(); + + const tileID = new OverscaledTileID(6, 0, 6, 8, 8); + const instances = [ + makeSymbolInstance(1000, 1000, ''), // A + makeSymbolInstance(1000, 1000, '') // B + ]; + const tile = makeTile(tileID, instances); + + // assigns new ids + index.addLayer(styleLayer, [tile], 0); + expect(instances[0].crossTileID).toBe(1); + expect(instances[1].crossTileID).toBe(2); + expect(index.layerIndexes[styleLayer.id]).toBeTruthy(); + + // remove styleLayer + index.pruneUnusedLayers([]); + expect(index.layerIndexes[styleLayer.id]).toBeFalsy(); + +}); diff --git a/web/libraries/maplibre-gl/src/symbol/cross_tile_symbol_index.ts b/web/libraries/maplibre-gl/src/symbol/cross_tile_symbol_index.ts new file mode 100644 index 00000000..7708b4f8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/cross_tile_symbol_index.ts @@ -0,0 +1,367 @@ +import KDBush from 'kdbush'; +import {EXTENT} from '../data/extent'; + +import {SymbolInstanceArray} from '../data/array_types.g'; + +import type {SymbolInstance} from '../data/array_types.g'; +import type {OverscaledTileID} from '../source/tile_id'; +import type {SymbolBucket} from '../data/bucket/symbol_bucket'; +import type {StyleLayer} from '../style/style_layer'; +import type {Tile} from '../source/tile'; + +/* + The CrossTileSymbolIndex generally works on the assumption that + a conceptual "unique symbol" can be identified by the text of + the label combined with the anchor point. The goal is to assign + these conceptual "unique symbols" a shared crossTileID that can be + used by Placement to keep fading opacity states consistent and to + deduplicate labels. + + The CrossTileSymbolIndex indexes all the current symbol instances and + their crossTileIDs. When a symbol bucket gets added or updated, the + index assigns a crossTileID to each of it's symbol instances by either + matching it with an existing id or assigning a new one. +*/ + +// Round anchor positions to roughly 4 pixel grid +const roundingFactor = 512 / EXTENT / 2; + +export const KDBUSH_THRESHHOLD = 128; + +interface SymbolsByKeyEntry { + index?: KDBush; + positions?: {x: number; y: number}[]; + crossTileIDs: number[]; +} + +class TileLayerIndex { + _symbolsByKey: Record = {}; + + constructor(public tileID: OverscaledTileID, symbolInstances: SymbolInstanceArray, public bucketInstanceId: number) { + // group the symbolInstances by key + const symbolInstancesByKey = new Map(); + for (let i = 0; i < symbolInstances.length; i++) { + const symbolInstance = symbolInstances.get(i); + const key = symbolInstance.key; + const instances = symbolInstancesByKey.get(key); + if (instances) { + // This tile may have multiple symbol instances with the same key + // Store each one along with its coordinates + instances.push(symbolInstance); + } else { + symbolInstancesByKey.set(key, [symbolInstance]); + } + } + + // index the SymbolInstances in this each bucket + for (const [key, symbols] of symbolInstancesByKey) { + const positions = symbols.map(symbolInstance => ({x: Math.floor(symbolInstance.anchorX * roundingFactor), y: Math.floor(symbolInstance.anchorY * roundingFactor)})); + const crossTileIDs = symbols.map(v => v.crossTileID); + const entry: SymbolsByKeyEntry = {positions, crossTileIDs}; + + // once we get too many symbols for a given key, it becomes much faster to index it before queries + if (entry.positions.length > KDBUSH_THRESHHOLD) { + + const index = new KDBush(entry.positions.length, 16, Uint16Array); + for (const {x, y} of entry.positions) index.add(x, y); + index.finish(); + + // clear all references to the original positions data + delete entry.positions; + entry.index = index; + } + + this._symbolsByKey[key] = entry; + } + } + + // Converts the coordinates of the input symbol instance into coordinates that be can compared + // against other symbols in this index. Coordinates are: + // (1) local-tile-based (so after correction we get x,y values relative to our local anchorX/Y) + // (2) converted to the z-scale of this TileLayerIndex + // (3) down-sampled by "roundingFactor" from tile coordinate precision in order to be + // more tolerant of small differences between tiles. + getScaledCoordinates(symbolInstance: SymbolInstance, childTileID: OverscaledTileID): {x: number; y: number} { + const {x: localX, y: localY, z: localZ} = this.tileID.canonical; + const {x, y, z} = childTileID.canonical; + + const zDifference = z - localZ; + const scale = roundingFactor / Math.pow(2, zDifference); + const xWorld = (x * EXTENT + symbolInstance.anchorX) * scale; + const yWorld = (y * EXTENT + symbolInstance.anchorY) * scale; + const xOffset = localX * EXTENT * roundingFactor; + const yOffset = localY * EXTENT * roundingFactor; + const result = { + x: Math.floor(xWorld - xOffset), + y: Math.floor(yWorld - yOffset) + }; + + return result; + } + + findMatches(symbolInstances: SymbolInstanceArray, newTileID: OverscaledTileID, zoomCrossTileIDs: { + [crossTileID: number]: boolean; + }) { + const tolerance = this.tileID.canonical.z < newTileID.canonical.z ? 1 : Math.pow(2, this.tileID.canonical.z - newTileID.canonical.z); + + for (let i = 0; i < symbolInstances.length; i++) { + const symbolInstance = symbolInstances.get(i); + if (symbolInstance.crossTileID) { + // already has a match, skip + continue; + } + + const entry = this._symbolsByKey[symbolInstance.key]; + if (!entry) { + // No symbol with this key in this bucket + continue; + } + + const scaledSymbolCoord = this.getScaledCoordinates(symbolInstance, newTileID); + + if (entry.index) { + // Return any symbol with the same keys whose coordinates are within 1 + // grid unit. (with a 4px grid, this covers a 12px by 12px area) + const indexes = entry.index.range( + scaledSymbolCoord.x - tolerance, + scaledSymbolCoord.y - tolerance, + scaledSymbolCoord.x + tolerance, + scaledSymbolCoord.y + tolerance).sort(); + + for (const i of indexes) { + const crossTileID = entry.crossTileIDs[i]; + + if (!zoomCrossTileIDs[crossTileID]) { + // Once we've marked ourselves duplicate against this parent symbol, + // don't let any other symbols at the same zoom level duplicate against + // the same parent (see issue #5993) + zoomCrossTileIDs[crossTileID] = true; + symbolInstance.crossTileID = crossTileID; + break; + } + } + } else if (entry.positions) { + for (let i = 0; i < entry.positions.length; i++) { + const thisTileSymbol = entry.positions[i]; + const crossTileID = entry.crossTileIDs[i]; + + // Return any symbol with the same keys whose coordinates are within 1 + // grid unit. (with a 4px grid, this covers a 12px by 12px area) + if (Math.abs(thisTileSymbol.x - scaledSymbolCoord.x) <= tolerance && + Math.abs(thisTileSymbol.y - scaledSymbolCoord.y) <= tolerance && + !zoomCrossTileIDs[crossTileID]) { + // Once we've marked ourselves duplicate against this parent symbol, + // don't let any other symbols at the same zoom level duplicate against + // the same parent (see issue #5993) + zoomCrossTileIDs[crossTileID] = true; + symbolInstance.crossTileID = crossTileID; + break; + } + } + } + } + } + + getCrossTileIDsLists() { + return Object.values(this._symbolsByKey).map(({crossTileIDs}) => crossTileIDs); + } +} + +class CrossTileIDs { + maxCrossTileID: number; + constructor() { + this.maxCrossTileID = 0; + } + generate() { + return ++this.maxCrossTileID; + } +} + +class CrossTileSymbolLayerIndex { + indexes: { + [zoom in string | number]: { + [tileId in string | number]: TileLayerIndex; + }; + }; + usedCrossTileIDs: { + [zoom in string | number]: { + [crossTileID: number]: boolean; + }; + }; + lng: number; + + constructor() { + this.indexes = {}; + this.usedCrossTileIDs = {}; + this.lng = 0; + } + + /* + * Sometimes when a user pans across the antimeridian the longitude value gets wrapped. + * To prevent labels from flashing out and in we adjust the tileID values in the indexes + * so that they match the new wrapped version of the map. + */ + handleWrapJump(lng: number) { + const wrapDelta = Math.round((lng - this.lng) / 360); + if (wrapDelta !== 0) { + for (const zoom in this.indexes) { + const zoomIndexes = this.indexes[zoom]; + const newZoomIndex = {}; + for (const key in zoomIndexes) { + // change the tileID's wrap and add it to a new index + const index = zoomIndexes[key]; + index.tileID = index.tileID.unwrapTo(index.tileID.wrap + wrapDelta); + newZoomIndex[index.tileID.key] = index; + } + this.indexes[zoom] = newZoomIndex; + } + } + this.lng = lng; + } + + addBucket(tileID: OverscaledTileID, bucket: SymbolBucket, crossTileIDs: CrossTileIDs) { + if (this.indexes[tileID.overscaledZ] && + this.indexes[tileID.overscaledZ][tileID.key]) { + if (this.indexes[tileID.overscaledZ][tileID.key].bucketInstanceId === + bucket.bucketInstanceId) { + return false; + } else { + // We're replacing this bucket with an updated version + // Remove the old bucket's "used crossTileIDs" now so that + // the new bucket can claim them. + // The old index entries themselves stick around until + // 'removeStaleBuckets' is called. + this.removeBucketCrossTileIDs(tileID.overscaledZ, + this.indexes[tileID.overscaledZ][tileID.key]); + } + } + + for (let i = 0; i < bucket.symbolInstances.length; i++) { + const symbolInstance = bucket.symbolInstances.get(i); + symbolInstance.crossTileID = 0; + } + + if (!this.usedCrossTileIDs[tileID.overscaledZ]) { + this.usedCrossTileIDs[tileID.overscaledZ] = {}; + } + const zoomCrossTileIDs = this.usedCrossTileIDs[tileID.overscaledZ]; + + for (const zoom in this.indexes) { + const zoomIndexes = this.indexes[zoom]; + if (Number(zoom) > tileID.overscaledZ) { + for (const id in zoomIndexes) { + const childIndex = zoomIndexes[id]; + if (childIndex.tileID.isChildOf(tileID)) { + childIndex.findMatches(bucket.symbolInstances, tileID, zoomCrossTileIDs); + } + } + } else { + const parentCoord = tileID.scaledTo(Number(zoom)); + const parentIndex = zoomIndexes[parentCoord.key]; + if (parentIndex) { + parentIndex.findMatches(bucket.symbolInstances, tileID, zoomCrossTileIDs); + } + } + } + + for (let i = 0; i < bucket.symbolInstances.length; i++) { + const symbolInstance = bucket.symbolInstances.get(i); + if (!symbolInstance.crossTileID) { + // symbol did not match any known symbol, assign a new id + symbolInstance.crossTileID = crossTileIDs.generate(); + zoomCrossTileIDs[symbolInstance.crossTileID] = true; + } + } + + if (this.indexes[tileID.overscaledZ] === undefined) { + this.indexes[tileID.overscaledZ] = {}; + } + this.indexes[tileID.overscaledZ][tileID.key] = new TileLayerIndex(tileID, bucket.symbolInstances, bucket.bucketInstanceId); + + return true; + } + + removeBucketCrossTileIDs(zoom: string | number, removedBucket: TileLayerIndex) { + for (const crossTileIDs of removedBucket.getCrossTileIDsLists()) { + for (const crossTileID of crossTileIDs) { + delete this.usedCrossTileIDs[zoom][crossTileID]; + } + } + } + + removeStaleBuckets(currentIDs: { + [k in string | number]: boolean; + }) { + let tilesChanged = false; + for (const z in this.indexes) { + const zoomIndexes = this.indexes[z]; + for (const tileKey in zoomIndexes) { + if (!currentIDs[zoomIndexes[tileKey].bucketInstanceId]) { + this.removeBucketCrossTileIDs(z, zoomIndexes[tileKey]); + delete zoomIndexes[tileKey]; + tilesChanged = true; + } + } + } + return tilesChanged; + } +} + +export class CrossTileSymbolIndex { + layerIndexes: {[layerId: string]: CrossTileSymbolLayerIndex}; + crossTileIDs: CrossTileIDs; + maxBucketInstanceId: number; + bucketsInCurrentPlacement: {[_: number]: boolean}; + + constructor() { + this.layerIndexes = {}; + this.crossTileIDs = new CrossTileIDs(); + this.maxBucketInstanceId = 0; + this.bucketsInCurrentPlacement = {}; + } + + addLayer(styleLayer: StyleLayer, tiles: Array, lng: number) { + let layerIndex = this.layerIndexes[styleLayer.id]; + if (layerIndex === undefined) { + layerIndex = this.layerIndexes[styleLayer.id] = new CrossTileSymbolLayerIndex(); + } + + let symbolBucketsChanged = false; + const currentBucketIDs = {}; + + layerIndex.handleWrapJump(lng); + + for (const tile of tiles) { + const symbolBucket = (tile.getBucket(styleLayer) as any as SymbolBucket); + if (!symbolBucket || styleLayer.id !== symbolBucket.layerIds[0]) + continue; + + if (!symbolBucket.bucketInstanceId) { + symbolBucket.bucketInstanceId = ++this.maxBucketInstanceId; + } + + if (layerIndex.addBucket(tile.tileID, symbolBucket, this.crossTileIDs)) { + symbolBucketsChanged = true; + } + currentBucketIDs[symbolBucket.bucketInstanceId] = true; + } + + if (layerIndex.removeStaleBuckets(currentBucketIDs)) { + symbolBucketsChanged = true; + } + + return symbolBucketsChanged; + } + + pruneUnusedLayers(usedLayers: Array) { + const usedLayerMap = {}; + usedLayers.forEach((usedLayer) => { + usedLayerMap[usedLayer] = true; + }); + for (const layerId in this.layerIndexes) { + if (!usedLayerMap[layerId]) { + delete this.layerIndexes[layerId]; + } + } + } +} diff --git a/web/libraries/maplibre-gl/src/symbol/get_anchors.test.ts b/web/libraries/maplibre-gl/src/symbol/get_anchors.test.ts new file mode 100644 index 00000000..6d3c1ec2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/get_anchors.test.ts @@ -0,0 +1,113 @@ +import Point from '@mapbox/point-geometry'; +import {getAnchors, getCenterAnchor} from './get_anchors'; +import {PositionedIcon, Shaping} from './shaping'; + +const TILE_EXTENT = 4096; + +describe('getAnchors', () => { + const nonContinuedLine = []; + for (let i = 1; i < 11; i++) { + nonContinuedLine.push(new Point(1, i)); + } + + const continuedLine = []; + for (let j = 0; j < 10; j++) { + continuedLine.push(new Point(1, j)); + } + + const smallSpacing = 2; + const bigSpacing = 3; + + const shapedText = {left: -1, right: 1, top: -0.5, bottom: 0.5} as Shaping; + const shapedIcon = {left: -0.5, right: 0.5, top: -0.25, bottom: 0.25} as PositionedIcon; + const labelLength = Math.max( + shapedText ? shapedText.right - shapedText.left : 0, + shapedIcon ? shapedIcon.right - shapedIcon.left : 0); + + const glyphSize = 0.1; + + test('non-continued line with short labels', () => { + const anchors = getAnchors(nonContinuedLine, bigSpacing, Math.PI, shapedText, shapedIcon, glyphSize, 1, 1, TILE_EXTENT); + + expect(anchors).toEqual([ + {x: 1, y: 2, angle: 1.5707963267948966, segment: 1}, + {x: 1, y: 5, angle: 1.5707963267948966, segment: 4}, + {x: 1, y: 8, angle: 1.5707963267948966, segment: 7}]); + + expect( + labelLength / 2 + 1 <= anchors[0].y && anchors[0].y < labelLength / 2 + 3 * glyphSize + 1 + ).toBeTruthy(); + + }); + + test('non-continued line with long labels', () => { + const anchors = getAnchors(nonContinuedLine, smallSpacing, Math.PI, shapedText, shapedIcon, glyphSize, 1, 1, TILE_EXTENT); + + expect(anchors).toEqual([ + {x: 1, y: 2, angle: 1.5707963267948966, segment: 1}, + {x: 1, y: 5, angle: 1.5707963267948966, segment: 3}, + {x: 1, y: 7, angle: 1.5707963267948966, segment: 6}]); + + }); + + test('continued line with short labels', () => { + const anchors = getAnchors(continuedLine, bigSpacing, Math.PI, shapedText, shapedIcon, glyphSize, 1, 1, TILE_EXTENT); + + expect(anchors).toEqual([ + {x: 1, y: 2, angle: 1.5707963267948966, segment: 1}, + {x: 1, y: 5, angle: 1.5707963267948966, segment: 4}, + {x: 1, y: 8, angle: 1.5707963267948966, segment: 7}]); + + }); + + test('continued line with long labels', () => { + const anchors = getAnchors(continuedLine, smallSpacing, Math.PI, shapedText, shapedIcon, glyphSize, 1, 1, TILE_EXTENT); + + expect(anchors).toEqual([ + {x: 1, y: 1, angle: 1.5707963267948966, segment: 1}, + {x: 1, y: 4, angle: 1.5707963267948966, segment: 3}, + {x: 1, y: 6, angle: 1.5707963267948966, segment: 6}]); + + }); + + test('overscaled anchors contain all anchors in parent', () => { + const anchors = getAnchors(nonContinuedLine, bigSpacing, Math.PI, shapedText, shapedIcon, glyphSize, 1, 1, TILE_EXTENT); + const childAnchors = getAnchors(nonContinuedLine, bigSpacing / 2, Math.PI, shapedText, shapedIcon, glyphSize, 0.5, 2, TILE_EXTENT); + for (let i = 0; i < anchors.length; i++) { + const anchor = anchors[i]; + let found = false; + for (let k = 0; k < childAnchors.length; k++) { + if (anchor.equals(childAnchors[k])) { + found = true; + break; + } + } + expect(found).toBeTruthy(); + } + }); + + test('use middle point as a fallback position for short non-continued lines', () => { + const line = [new Point(1, 1), new Point(1, 3.1)]; + const anchors = getAnchors(line, 2, Math.PI, shapedText, shapedIcon, glyphSize, 1, 1, TILE_EXTENT); + expect(anchors).toEqual([ + {x: 1, y: 2, angle: 1.5707963267948966, segment: 0}]); + }); + + test('getCenterAnchor', () => { + const line = [new Point(1, 1), new Point(1, 3.1), new Point(3, 6), new Point(4, 7)]; + const anchor = getCenterAnchor(line, Math.PI, shapedText, shapedIcon, glyphSize, 1); + expect(anchor).toEqual({x: 2, y: 4, angle: 0.9670469933974603, segment: 1}); + }); + + test('getCenterAnchor with center outside tile bounds', () => { + const line = [new Point(-10, -10), new Point(5, 5)]; + const anchor = getCenterAnchor(line, Math.PI, shapedText, shapedIcon, glyphSize, 1); + expect(anchor).toEqual({x: -2, y: -2, angle: 0.7853981633974483, segment: 0}); + }); + + test('getCenterAnchor failing maxAngle test', () => { + const line = [new Point(1, 1), new Point(1, 3), new Point(3, 3)]; + const anchor = getCenterAnchor(line, Math.PI / 4, shapedText, shapedIcon, glyphSize, 1); + expect(anchor).toBeFalsy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/symbol/get_anchors.ts b/web/libraries/maplibre-gl/src/symbol/get_anchors.ts new file mode 100644 index 00000000..93ec995c --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/get_anchors.ts @@ -0,0 +1,167 @@ +import {interpolates} from '@maplibre/maplibre-gl-style-spec'; + +import {Anchor} from '../symbol/anchor'; +import {checkMaxAngle} from './check_max_angle'; + +import type Point from '@mapbox/point-geometry'; +import type {Shaping, PositionedIcon} from './shaping'; + +export {getAnchors, getCenterAnchor}; + +function getLineLength(line: Array): number { + let lineLength = 0; + for (let k = 0; k < line.length - 1; k++) { + lineLength += line[k].dist(line[k + 1]); + } + return lineLength; +} + +function getAngleWindowSize( + shapedText: Shaping, + glyphSize: number, + boxScale: number +): number { + return shapedText ? + 3 / 5 * glyphSize * boxScale : + 0; +} + +function getShapedLabelLength(shapedText?: Shaping | null, shapedIcon?: PositionedIcon | null): number { + return Math.max( + shapedText ? shapedText.right - shapedText.left : 0, + shapedIcon ? shapedIcon.right - shapedIcon.left : 0); +} + +function getCenterAnchor(line: Array, + maxAngle: number, + shapedText: Shaping, + shapedIcon: PositionedIcon, + glyphSize: number, + boxScale: number) { + const angleWindowSize = getAngleWindowSize(shapedText, glyphSize, boxScale); + const labelLength = getShapedLabelLength(shapedText, shapedIcon) * boxScale; + + let prevDistance = 0; + const centerDistance = getLineLength(line) / 2; + + for (let i = 0; i < line.length - 1; i++) { + + const a = line[i], + b = line[i + 1]; + + const segmentDistance = a.dist(b); + + if (prevDistance + segmentDistance > centerDistance) { + // The center is on this segment + const t = (centerDistance - prevDistance) / segmentDistance, + x = interpolates.number(a.x, b.x, t), + y = interpolates.number(a.y, b.y, t); + + const anchor = new Anchor(x, y, b.angleTo(a), i); + anchor._round(); + if (!angleWindowSize || checkMaxAngle(line, anchor, labelLength, angleWindowSize, maxAngle)) { + return anchor; + } else { + return; + } + } + + prevDistance += segmentDistance; + } +} + +function getAnchors(line: Array, + spacing: number, + maxAngle: number, + shapedText: Shaping, + shapedIcon: PositionedIcon, + glyphSize: number, + boxScale: number, + overscaling: number, + tileExtent: number) { + + // Resample a line to get anchor points for labels and check that each + // potential label passes text-max-angle check and has enough froom to fit + // on the line. + + const angleWindowSize = getAngleWindowSize(shapedText, glyphSize, boxScale); + const shapedLabelLength = getShapedLabelLength(shapedText, shapedIcon); + const labelLength = shapedLabelLength * boxScale; + + // Is the line continued from outside the tile boundary? + const isLineContinued = line[0].x === 0 || line[0].x === tileExtent || line[0].y === 0 || line[0].y === tileExtent; + + // Is the label long, relative to the spacing? + // If so, adjust the spacing so there is always a minimum space of `spacing / 4` between label edges. + if (spacing - labelLength < spacing / 4) { + spacing = labelLength + spacing / 4; + } + + // Offset the first anchor by: + // Either half the label length plus a fixed extra offset if the line is not continued + // Or half the spacing if the line is continued. + + // For non-continued lines, add a bit of fixed extra offset to avoid collisions at T intersections. + const fixedExtraOffset = glyphSize * 2; + + const offset = !isLineContinued ? + ((shapedLabelLength / 2 + fixedExtraOffset) * boxScale * overscaling) % spacing : + (spacing / 2 * overscaling) % spacing; + + return resample(line, offset, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, false, tileExtent); +} + +function resample(line, offset, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, placeAtMiddle, tileExtent) { + + const halfLabelLength = labelLength / 2; + const lineLength = getLineLength(line); + + let distance = 0, + markedDistance = offset - spacing; + + let anchors = []; + + for (let i = 0; i < line.length - 1; i++) { + + const a = line[i], + b = line[i + 1]; + + const segmentDist = a.dist(b), + angle = b.angleTo(a); + + while (markedDistance + spacing < distance + segmentDist) { + markedDistance += spacing; + + const t = (markedDistance - distance) / segmentDist, + x = interpolates.number(a.x, b.x, t), + y = interpolates.number(a.y, b.y, t); + + // Check that the point is within the tile boundaries and that + // the label would fit before the beginning and end of the line + // if placed at this point. + if (x >= 0 && x < tileExtent && y >= 0 && y < tileExtent && + markedDistance - halfLabelLength >= 0 && + markedDistance + halfLabelLength <= lineLength) { + const anchor = new Anchor(x, y, angle, i); + anchor._round(); + + if (!angleWindowSize || checkMaxAngle(line, anchor, labelLength, angleWindowSize, maxAngle)) { + anchors.push(anchor); + } + } + } + + distance += segmentDist; + } + + if (!placeAtMiddle && !anchors.length && !isLineContinued) { + // The first attempt at finding anchors at which labels can be placed failed. + // Try again, but this time just try placing one anchor at the middle of the line. + // This has the most effect for short lines in overscaled tiles, since the + // initial offset used in overscaled tiles is calculated to align labels with positions in + // parent tiles instead of placing the label as close to the beginning as possible. + anchors = resample(line, distance / 2, spacing, angleWindowSize, maxAngle, labelLength, isLineContinued, true, tileExtent); + } + + return anchors; +} diff --git a/web/libraries/maplibre-gl/src/symbol/grid_index.test.ts b/web/libraries/maplibre-gl/src/symbol/grid_index.test.ts new file mode 100644 index 00000000..442fcf29 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/grid_index.test.ts @@ -0,0 +1,75 @@ +import {GridIndex} from './grid_index'; +import type {GridKey} from './grid_index'; + +describe('GridIndex', () => { + + test('indexes features', () => { + const grid = new GridIndex(100, 100, 10); + grid.insert(0 as GridKey, 4, 10, 6, 30); + grid.insert(1 as GridKey, 4, 10, 30, 12); + grid.insert(2 as GridKey, -10, 30, 5, 35); + + expect(grid.query(4, 10, 5, 11).map(x => x.key).sort()).toEqual([0, 1]); + expect(grid.query(24, 10, 25, 11).map(x => x.key).sort()).toEqual([1]); + expect(grid.query(40, 40, 100, 100).map(x => x.key)).toEqual([]); + expect(grid.query(-6, 0, 3, 100).map(x => x.key)).toEqual([2]); + expect( + grid.query(-Infinity, -Infinity, Infinity, Infinity).map(x => x.key).sort() + ).toEqual([0, 1, 2]); + }); + + test('returns multiple copies of a key if multiple boxes were inserted with the same key', () => { + const grid = new GridIndex(100, 100, 10); + const key = 123 as GridKey; + grid.insert(key, 3, 3, 4, 4); + grid.insert(key, 13, 13, 14, 14); + grid.insert(key, 23, 23, 24, 24); + expect(grid.query(0, 0, 30, 30).map(x => x.key)).toEqual([key, key, key]); + }); + + test('circle-circle intersection', () => { + const grid = new GridIndex(100, 100, 10); + grid.insertCircle(0 as GridKey, 50, 50, 10); + grid.insertCircle(1 as GridKey, 60, 60, 15); + grid.insertCircle(2 as GridKey, -10, 110, 20); + + expect(grid.hitTestCircle(55, 55, 2, 'never')).toBeTruthy(); + expect(grid.hitTestCircle(10, 10, 10, 'never')).toBeFalsy(); + expect(grid.hitTestCircle(0, 100, 10, 'never')).toBeTruthy(); + expect(grid.hitTestCircle(80, 60, 10, 'never')).toBeTruthy(); + }); + + test('circle-rectangle intersection', () => { + const grid = new GridIndex(100, 100, 10); + grid.insertCircle(0 as GridKey, 50, 50, 10); + grid.insertCircle(1 as GridKey, 60, 60, 15); + grid.insertCircle(2 as GridKey, -10, 110, 20); + + expect(grid.query(45, 45, 55, 55).map(x => x.key)).toEqual([0, 1]); + expect(grid.query(0, 0, 30, 30).map(x => x.key)).toEqual([]); + expect(grid.query(0, 80, 20, 100).map(x => x.key)).toEqual([2]); + }); + + test('overlap mode', () => { + const grid = new GridIndex(100, 100, 10); + grid.insert({overlapMode: 'never'}, 10, 10, 20, 20); + grid.insert({overlapMode: 'always'}, 30, 10, 40, 20); + grid.insert({overlapMode: 'cooperative'}, 50, 10, 60, 20); + + // 'never' can't overlap anything + expect(grid.hitTest(15, 15, 25, 25, 'never')).toBeTruthy(); + expect(grid.hitTest(35, 15, 45, 25, 'never')).toBeTruthy(); + expect(grid.hitTest(55, 15, 65, 25, 'never')).toBeTruthy(); + + // 'always' can overlap everything + expect(grid.hitTest(15, 15, 25, 25, 'always')).toBeFalsy(); + expect(grid.hitTest(35, 15, 45, 25, 'always')).toBeFalsy(); + expect(grid.hitTest(55, 15, 65, 25, 'always')).toBeFalsy(); + + // 'cooperative' can overlap 'always' and 'cooperative' + expect(grid.hitTest(15, 15, 25, 25, 'cooperative')).toBeTruthy(); + expect(grid.hitTest(35, 15, 45, 25, 'cooperative')).toBeFalsy(); + expect(grid.hitTest(55, 15, 65, 25, 'cooperative')).toBeFalsy(); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/symbol/grid_index.ts b/web/libraries/maplibre-gl/src/symbol/grid_index.ts new file mode 100644 index 00000000..51e14d95 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/grid_index.ts @@ -0,0 +1,414 @@ +import type {OverlapMode} from '../style/style_layer/overlap_mode'; + +type QueryArgs = { + hitTest: boolean; + overlapMode?: OverlapMode; + circle?: { + x: number; + y: number; + radius: number; + }; + seenUids: { + box: { + [_: number]: boolean; + }; + circle: { + [_: number]: boolean; + }; + }; +}; + +type QueryResult = { + key: T; + x1: number; + y1: number; + x2: number; + y2: number; +}; + +/** + * A key for the grid + */ +export type GridKey = { + overlapMode?: OverlapMode; +} + +function overlapAllowed(overlapA: OverlapMode, overlapB: OverlapMode): boolean { + let allowed = true; + + if (overlapA === 'always') { + // symbol A using 'always' overlap - allowed to overlap anything. + } else if (overlapA === 'never' || overlapB === 'never') { + // symbol A using 'never' overlap - can't overlap anything + // symbol A using 'cooperative' overlap - can overlap 'always' or 'cooperative' symbol; can't overlap 'never' + allowed = false; + } + + return allowed; +} + +/** + * @internal + * GridIndex is a data structure for testing the intersection of + * circles and rectangles in a 2d plane. + * It is optimized for rapid insertion and querying. + * GridIndex splits the plane into a set of "cells" and keeps track + * of which geometries intersect with each cell. At query time, + * full geometry comparisons are only done for items that share + * at least one cell. As long as the geometries are relatively + * uniformly distributed across the plane, this greatly reduces + * the number of comparisons necessary. + */ +export class GridIndex { + circleKeys: Array; + boxKeys: Array; + boxCells: Array>; + circleCells: Array>; + bboxes: Array; + circles: Array; + xCellCount: number; + yCellCount: number; + width: number; + height: number; + xScale: number; + yScale: number; + boxUid: number; + circleUid: number; + + constructor (width: number, height: number, cellSize: number) { + const boxCells = this.boxCells = []; + const circleCells = this.circleCells = []; + + // More cells -> fewer geometries to check per cell, but items tend + // to be split across more cells. + // Sweet spot allows most small items to fit in one cell + this.xCellCount = Math.ceil(width / cellSize); + this.yCellCount = Math.ceil(height / cellSize); + + for (let i = 0; i < this.xCellCount * this.yCellCount; i++) { + boxCells.push([]); + circleCells.push([]); + } + this.circleKeys = []; + this.boxKeys = []; + this.bboxes = []; + this.circles = []; + + this.width = width; + this.height = height; + this.xScale = this.xCellCount / width; + this.yScale = this.yCellCount / height; + this.boxUid = 0; + this.circleUid = 0; + } + + keysLength() { + return this.boxKeys.length + this.circleKeys.length; + } + + insert(key: T, x1: number, y1: number, x2: number, y2: number) { + this._forEachCell(x1, y1, x2, y2, this._insertBoxCell, this.boxUid++); + this.boxKeys.push(key); + this.bboxes.push(x1); + this.bboxes.push(y1); + this.bboxes.push(x2); + this.bboxes.push(y2); + } + + insertCircle(key: T, x: number, y: number, radius: number) { + // Insert circle into grid for all cells in the circumscribing square + // It's more than necessary (by a factor of 4/PI), but fast to insert + this._forEachCell(x - radius, y - radius, x + radius, y + radius, this._insertCircleCell, this.circleUid++); + this.circleKeys.push(key); + this.circles.push(x); + this.circles.push(y); + this.circles.push(radius); + } + + private _insertBoxCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) { + this.boxCells[cellIndex].push(uid); + } + + private _insertCircleCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) { + this.circleCells[cellIndex].push(uid); + } + + private _query(x1: number, y1: number, x2: number, y2: number, hitTest: boolean, overlapMode: OverlapMode, predicate?: (key: T) => boolean): Array> { + if (x2 < 0 || x1 > this.width || y2 < 0 || y1 > this.height) { + return []; + } + const result: Array> = []; + if (x1 <= 0 && y1 <= 0 && this.width <= x2 && this.height <= y2) { + if (hitTest) { + // Covers the entire grid, so collides with everything + return [{ + key: null, + x1, + y1, + x2, + y2 + }]; + } + for (let boxUid = 0; boxUid < this.boxKeys.length; boxUid++) { + result.push({ + key: this.boxKeys[boxUid], + x1: this.bboxes[boxUid * 4], + y1: this.bboxes[boxUid * 4 + 1], + x2: this.bboxes[boxUid * 4 + 2], + y2: this.bboxes[boxUid * 4 + 3] + }); + } + for (let circleUid = 0; circleUid < this.circleKeys.length; circleUid++) { + const x = this.circles[circleUid * 3]; + const y = this.circles[circleUid * 3 + 1]; + const radius = this.circles[circleUid * 3 + 2]; + result.push({ + key: this.circleKeys[circleUid], + x1: x - radius, + y1: y - radius, + x2: x + radius, + y2: y + radius + }); + } + } else { + const queryArgs: QueryArgs = { + hitTest, + overlapMode, + seenUids: {box: {}, circle: {}} + }; + this._forEachCell(x1, y1, x2, y2, this._queryCell, result, queryArgs, predicate); + } + + return result; + } + + query(x1: number, y1: number, x2: number, y2: number): Array> { + return this._query(x1, y1, x2, y2, false, null); + } + + hitTest(x1: number, y1: number, x2: number, y2: number, overlapMode: OverlapMode, predicate?: (key: T) => boolean): boolean { + return this._query(x1, y1, x2, y2, true, overlapMode, predicate).length > 0; + } + + hitTestCircle(x: number, y: number, radius: number, overlapMode: OverlapMode, predicate?: (key: T) => boolean): boolean { + // Insert circle into grid for all cells in the circumscribing square + // It's more than necessary (by a factor of 4/PI), but fast to insert + const x1 = x - radius; + const x2 = x + radius; + const y1 = y - radius; + const y2 = y + radius; + if (x2 < 0 || x1 > this.width || y2 < 0 || y1 > this.height) { + return false; + } + + // Box query early exits if the bounding box is larger than the grid, but we don't do + // the equivalent calculation for circle queries because early exit is less likely + // and the calculation is more expensive + const result: boolean[] = []; + const queryArgs: QueryArgs = { + hitTest: true, + overlapMode, + circle: {x, y, radius}, + seenUids: {box: {}, circle: {}} + }; + this._forEachCell(x1, y1, x2, y2, this._queryCellCircle, result, queryArgs, predicate); + return result.length > 0; + } + + private _queryCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, result: Array>, queryArgs: QueryArgs, predicate?: (key: T) => boolean): boolean { + const {seenUids, hitTest, overlapMode} = queryArgs; + const boxCell = this.boxCells[cellIndex]; + + if (boxCell !== null) { + const bboxes = this.bboxes; + for (const boxUid of boxCell) { + if (!seenUids.box[boxUid]) { + seenUids.box[boxUid] = true; + const offset = boxUid * 4; + const key = this.boxKeys[boxUid]; + + if ((x1 <= bboxes[offset + 2]) && + (y1 <= bboxes[offset + 3]) && + (x2 >= bboxes[offset + 0]) && + (y2 >= bboxes[offset + 1]) && + (!predicate || predicate(key))) { + if (!hitTest || !overlapAllowed(overlapMode, key.overlapMode)) { + result.push({ + key, + x1: bboxes[offset], + y1: bboxes[offset + 1], + x2: bboxes[offset + 2], + y2: bboxes[offset + 3] + }); + if (hitTest) { + // true return value stops the query after first match + return true; + } + } + } + } + } + } + const circleCell = this.circleCells[cellIndex]; + if (circleCell !== null) { + const circles = this.circles; + for (const circleUid of circleCell) { + if (!seenUids.circle[circleUid]) { + seenUids.circle[circleUid] = true; + const offset = circleUid * 3; + const key = this.circleKeys[circleUid]; + + if (this._circleAndRectCollide( + circles[offset], + circles[offset + 1], + circles[offset + 2], + x1, + y1, + x2, + y2) && + (!predicate || predicate(key))) { + if (!hitTest || !overlapAllowed(overlapMode, key.overlapMode)) { + const x = circles[offset]; + const y = circles[offset + 1]; + const radius = circles[offset + 2]; + result.push({ + key, + x1: x - radius, + y1: y - radius, + x2: x + radius, + y2: y + radius + }); + if (hitTest) { + // true return value stops the query after first match + return true; + } + } + } + } + } + } + + // false return to continue query + return false; + } + + private _queryCellCircle(x1: number, y1: number, x2: number, y2: number, cellIndex: number, result: Array, queryArgs: QueryArgs, predicate?: (key: T) => boolean): boolean { + const {circle, seenUids, overlapMode} = queryArgs; + const boxCell = this.boxCells[cellIndex]; + + if (boxCell !== null) { + const bboxes = this.bboxes; + for (const boxUid of boxCell) { + if (!seenUids.box[boxUid]) { + seenUids.box[boxUid] = true; + const offset = boxUid * 4; + const key = this.boxKeys[boxUid]; + if (this._circleAndRectCollide( + circle.x, + circle.y, + circle.radius, + bboxes[offset + 0], + bboxes[offset + 1], + bboxes[offset + 2], + bboxes[offset + 3]) && + (!predicate || predicate(key)) && + !overlapAllowed(overlapMode, key.overlapMode)) { + result.push(true); + return true; + } + } + } + } + + const circleCell = this.circleCells[cellIndex]; + if (circleCell !== null) { + const circles = this.circles; + for (const circleUid of circleCell) { + if (!seenUids.circle[circleUid]) { + seenUids.circle[circleUid] = true; + const offset = circleUid * 3; + const key = this.circleKeys[circleUid]; + if (this._circlesCollide( + circles[offset], + circles[offset + 1], + circles[offset + 2], + circle.x, + circle.y, + circle.radius) && + (!predicate || predicate(key)) && + !overlapAllowed(overlapMode, key.overlapMode)) { + result.push(true); + return true; + } + } + } + } + } + + private _forEachCell( + x1: number, + y1: number, + x2: number, + y2: number, + fn: (x1: number, y1: number, x2: number, y2: number, cellIndex: number, arg1: TArg, arg2?: QueryArgs, predicate?: (key: T) => boolean) => boolean | void, + arg1: TArg, + arg2?: QueryArgs, + predicate?: (key: T) => boolean) { + const cx1 = this._convertToXCellCoord(x1); + const cy1 = this._convertToYCellCoord(y1); + const cx2 = this._convertToXCellCoord(x2); + const cy2 = this._convertToYCellCoord(y2); + + for (let x = cx1; x <= cx2; x++) { + for (let y = cy1; y <= cy2; y++) { + const cellIndex = this.xCellCount * y + x; + if (fn.call(this, x1, y1, x2, y2, cellIndex, arg1, arg2, predicate)) return; + } + } + } + + private _convertToXCellCoord(x: number) { + return Math.max(0, Math.min(this.xCellCount - 1, Math.floor(x * this.xScale))); + } + + private _convertToYCellCoord(y: number) { + return Math.max(0, Math.min(this.yCellCount - 1, Math.floor(y * this.yScale))); + } + + private _circlesCollide(x1: number, y1: number, r1: number, x2: number, y2: number, r2: number): boolean { + const dx = x2 - x1; + const dy = y2 - y1; + const bothRadii = r1 + r2; + return (bothRadii * bothRadii) > (dx * dx + dy * dy); + } + + private _circleAndRectCollide( + circleX: number, + circleY: number, + radius: number, + x1: number, + y1: number, + x2: number, + y2: number + ): boolean { + const halfRectWidth = (x2 - x1) / 2; + const distX = Math.abs(circleX - (x1 + halfRectWidth)); + if (distX > (halfRectWidth + radius)) { + return false; + } + + const halfRectHeight = (y2 - y1) / 2; + const distY = Math.abs(circleY - (y1 + halfRectHeight)); + if (distY > (halfRectHeight + radius)) { + return false; + } + + if (distX <= halfRectWidth || distY <= halfRectHeight) { + return true; + } + + const dx = distX - halfRectWidth; + const dy = distY - halfRectHeight; + return (dx * dx + dy * dy <= (radius * radius)); + } +} diff --git a/web/libraries/maplibre-gl/src/symbol/merge_lines.test.ts b/web/libraries/maplibre-gl/src/symbol/merge_lines.test.ts new file mode 100644 index 00000000..d4a51abb --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/merge_lines.test.ts @@ -0,0 +1,30 @@ +import {mergeLines} from './merge_lines'; +import Point from '@mapbox/point-geometry'; + +function makeFeatures(lines) { + const features = []; + for (const line of lines) { + const points = []; + for (let j = 1; j < line.length; j++) { + points.push(new Point(line[j], 0)); + } + features.push({text: line[0], geometry: [points]}); + } + return features; +} + +describe('mergeLines', () => { + test('mergeLines merges lines with the same text', () => { + expect( + mergeLines(makeFeatures([['a', 0, 1, 2], ['b', 4, 5, 6], ['a', 8, 9], ['a', 2, 3, 4], ['a', 6, 7, 8], ['a', 5, 6]])) + ).toEqual(makeFeatures([['a', 0, 1, 2, 3, 4], ['b', 4, 5, 6], ['a', 5, 6, 7, 8, 9]])); + }); + + test('mergeLines handles merge from both ends', () => { + expect(mergeLines(makeFeatures([['a', 0, 1, 2], ['a', 4, 5, 6], ['a', 2, 3, 4]]))).toEqual(makeFeatures([['a', 0, 1, 2, 3, 4, 5, 6]])); + }); + + test('mergeLines handles circular lines', () => { + expect(mergeLines(makeFeatures([['a', 0, 1, 2], ['a', 2, 3, 4], ['a', 4, 0]]))).toEqual(makeFeatures([['a', 0, 1, 2, 3, 4, 0]])); + }); +}); diff --git a/web/libraries/maplibre-gl/src/symbol/merge_lines.ts b/web/libraries/maplibre-gl/src/symbol/merge_lines.ts new file mode 100644 index 00000000..b12a3be6 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/merge_lines.ts @@ -0,0 +1,80 @@ +import type {SymbolFeature} from '../data/bucket/symbol_bucket'; + +export function mergeLines(features: Array): Array { + const leftIndex: {[_: string]: number} = {}; + const rightIndex: {[_: string]: number} = {}; + const mergedFeatures = []; + let mergedIndex = 0; + + function add(k) { + mergedFeatures.push(features[k]); + mergedIndex++; + } + + function mergeFromRight(leftKey: string, rightKey: string, geom) { + const i = rightIndex[leftKey]; + delete rightIndex[leftKey]; + rightIndex[rightKey] = i; + + mergedFeatures[i].geometry[0].pop(); + mergedFeatures[i].geometry[0] = mergedFeatures[i].geometry[0].concat(geom[0]); + return i; + } + + function mergeFromLeft(leftKey: string, rightKey: string, geom) { + const i = leftIndex[rightKey]; + delete leftIndex[rightKey]; + leftIndex[leftKey] = i; + + mergedFeatures[i].geometry[0].shift(); + mergedFeatures[i].geometry[0] = geom[0].concat(mergedFeatures[i].geometry[0]); + return i; + } + + function getKey(text, geom, onRight?) { + const point = onRight ? geom[0][geom[0].length - 1] : geom[0][0]; + return `${text}:${point.x}:${point.y}`; + } + + for (let k = 0; k < features.length; k++) { + const feature = features[k]; + const geom = feature.geometry; + const text = feature.text ? feature.text.toString() : null; + + if (!text) { + add(k); + continue; + } + + const leftKey = getKey(text, geom), + rightKey = getKey(text, geom, true); + + if ((leftKey in rightIndex) && (rightKey in leftIndex) && (rightIndex[leftKey] !== leftIndex[rightKey])) { + // found lines with the same text adjacent to both ends of the current line, merge all three + const j = mergeFromLeft(leftKey, rightKey, geom); + const i = mergeFromRight(leftKey, rightKey, mergedFeatures[j].geometry); + + delete leftIndex[leftKey]; + delete rightIndex[rightKey]; + + rightIndex[getKey(text, mergedFeatures[i].geometry, true)] = i; + mergedFeatures[j].geometry = null; + + } else if (leftKey in rightIndex) { + // found mergeable line adjacent to the start of the current line, merge + mergeFromRight(leftKey, rightKey, geom); + + } else if (rightKey in leftIndex) { + // found mergeable line adjacent to the end of the current line, merge + mergeFromLeft(leftKey, rightKey, geom); + + } else { + // no adjacent lines, add as a new item + add(k); + leftIndex[leftKey] = mergedIndex - 1; + rightIndex[rightKey] = mergedIndex - 1; + } + } + + return mergedFeatures.filter((f) => f.geometry); +} diff --git a/web/libraries/maplibre-gl/src/symbol/one_em.ts b/web/libraries/maplibre-gl/src/symbol/one_em.ts new file mode 100644 index 00000000..d66d5e83 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/one_em.ts @@ -0,0 +1,3 @@ +// ONE_EM constant used to go between "em" units used in style spec and "points" used internally for layout + +export default 24; diff --git a/web/libraries/maplibre-gl/src/symbol/opacity_state.ts b/web/libraries/maplibre-gl/src/symbol/opacity_state.ts new file mode 100644 index 00000000..cebc209c --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/opacity_state.ts @@ -0,0 +1,23 @@ +import {register} from '../util/web_worker_transfer'; + +export class OpacityState { + opacity: number; + targetOpacity: number; + time: number; + + constructor() { + this.opacity = 0; + this.targetOpacity = 0; + this.time = 0; + } + + clone() { + const clone = new OpacityState(); + clone.opacity = this.opacity; + clone.targetOpacity = this.targetOpacity; + clone.time = this.time; + return clone; + } +} + +register('OpacityState', OpacityState); diff --git a/web/libraries/maplibre-gl/src/symbol/path_interpolator.test.ts b/web/libraries/maplibre-gl/src/symbol/path_interpolator.test.ts new file mode 100644 index 00000000..52251317 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/path_interpolator.test.ts @@ -0,0 +1,134 @@ +import Point from '@mapbox/point-geometry'; +import {PathInterpolator} from './path_interpolator'; + +describe('PathInterpolator', () => { + + const pointEquals = (p0, p1) => { + const e = 0.000001; + return Math.abs(p0.x - p1.x) < e && Math.abs(p0.y - p1.y) < e; + }; + + test('Interpolate single segment path', () => { + const line = [ + new Point(0, 0), + new Point(10, 0) + ]; + + const interpolator = new PathInterpolator(line); + + expect(interpolator.lerp(0.0)).toEqual(line[0]); + expect(interpolator.lerp(0.5)).toEqual(new Point(5, 0)); + expect(interpolator.lerp(1.0)).toEqual(line[1]); + }); + + test('t < 0', () => { + const line = [ + new Point(0, 0), + new Point(10, 0) + ]; + + const interpolator = new PathInterpolator(line); + expect(interpolator.lerp(-100.0)).toEqual(line[0]); + }); + + test('t > 0', () => { + const line = [ + new Point(0, 0), + new Point(10, 0) + ]; + + const interpolator = new PathInterpolator(line); + expect(interpolator.lerp(100.0)).toEqual(line[1]); + }); + + test('Interpolate multi-segment path', () => { + const line = [ + new Point(-3, 3), + new Point(-1, 3), + new Point(-1, -2), + new Point(2, -2), + new Point(2, 1), + new Point(-3, 1) + ]; + + const interpolator = new PathInterpolator(line); + expect(pointEquals(interpolator.lerp(1.0), new Point(-3, 1))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(0.95), new Point(-2.1, 1))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(0.5), new Point(1, -2))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(0.25), new Point(-1, 0.5))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(0.1), new Point(-1.2, 3))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(0.0), new Point(-3, 3))).toBeTruthy(); + }); + + test('Small padding', () => { + const line = [ + new Point(-4, 1), + new Point(4, 1) + ]; + + const padding = 0.5; + const interpolator = new PathInterpolator(line, padding); + + expect(pointEquals(interpolator.lerp(0.0), new Point(-3.5, 1))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(0.25), new Point(-1.75, 1))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(0.5), new Point(0, 1))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(1.0), new Point(3.5, 1))).toBeTruthy(); + }); + + test('Padding cannot be larger than the length / 2', () => { + const line = [ + new Point(-3, 0), + new Point(3, 0) + ]; + + const padding = 10.0; + const interpolator = new PathInterpolator(line, padding); + + expect(pointEquals(interpolator.lerp(0.0), new Point(0, 0))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(0.4), new Point(0, 0))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(1.0), new Point(0, 0))).toBeTruthy(); + }); + + test('Single point path', () => { + const interpolator = new PathInterpolator([new Point(0, 0)]); + expect(pointEquals(interpolator.lerp(0), new Point(0, 0))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(1.0), new Point(0, 0))).toBeTruthy(); + }); + + test('Interpolator instance can be reused by calling reset()', () => { + const line0 = [ + new Point(0, 0), + new Point(10, 0) + ]; + + const line1 = [ + new Point(-10, 10), + new Point(10, -10) + ]; + + const interpolator = new PathInterpolator(line0); + + expect(interpolator.lerp(0.0)).toEqual(line0[0]); + expect(interpolator.lerp(0.5)).toEqual(new Point(5, 0)); + expect(interpolator.lerp(1.0)).toEqual(line0[1]); + + interpolator.reset(line1); + expect(pointEquals(interpolator.lerp(0.0), line1[0])).toBeTruthy(); + expect(pointEquals(interpolator.lerp(0.5), new Point(0, 0))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(1.0), line1[1])).toBeTruthy(); + }); + + test('Path with zero length segment', () => { + const line = [ + new Point(-1, 0), + new Point(1, 0), + new Point(1, 0) + ]; + + const interpolator = new PathInterpolator(line); + expect(pointEquals(interpolator.lerp(0), line[0])).toBeTruthy(); + expect(pointEquals(interpolator.lerp(0.5), new Point(0, 0))).toBeTruthy(); + expect(pointEquals(interpolator.lerp(1), line[1])).toBeTruthy(); + expect(pointEquals(interpolator.lerp(1), line[2])).toBeTruthy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/symbol/path_interpolator.ts b/web/libraries/maplibre-gl/src/symbol/path_interpolator.ts new file mode 100644 index 00000000..bff4b237 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/path_interpolator.ts @@ -0,0 +1,55 @@ +import {clamp} from '../util/util'; +import Point from '@mapbox/point-geometry'; + +export class PathInterpolator { + points: Array; + length: number; + paddedLength: number; + padding: number; + _distances: Array; + + constructor(points_?: Array | null, padding_?: number | null) { + this.reset(points_, padding_); + } + + reset(points_?: Array | null, padding_?: number | null) { + this.points = points_ || []; + + // Compute cumulative distance from first point to every other point in the segment. + // Last entry in the array is total length of the path + this._distances = [0.0]; + + for (let i = 1; i < this.points.length; i++) { + this._distances[i] = this._distances[i - 1] + this.points[i].dist(this.points[i - 1]); + } + + this.length = this._distances[this._distances.length - 1]; + this.padding = Math.min(padding_ || 0, this.length * 0.5); + this.paddedLength = this.length - this.padding * 2.0; + } + + lerp(t: number): Point { + if (this.points.length === 1) { + return this.points[0]; + } + + t = clamp(t, 0, 1); + + // Find the correct segment [p0, p1] where p0 <= x < p1 + let currentIndex = 1; + let distOfCurrentIdx = this._distances[currentIndex]; + const distToTarget = t * this.paddedLength + this.padding; + + while (distOfCurrentIdx < distToTarget && currentIndex < this._distances.length) { + distOfCurrentIdx = this._distances[++currentIndex]; + } + + // Interpolate between the two points of the segment + const idxOfPrevPoint = currentIndex - 1; + const distOfPrevIdx = this._distances[idxOfPrevPoint]; + const segmentLength = distOfCurrentIdx - distOfPrevIdx; + const segmentT = segmentLength > 0 ? (distToTarget - distOfPrevIdx) / segmentLength : 0; + + return this.points[idxOfPrevPoint].mult(1.0 - segmentT).add(this.points[currentIndex].mult(segmentT)); + } +} diff --git a/web/libraries/maplibre-gl/src/symbol/placement.ts b/web/libraries/maplibre-gl/src/symbol/placement.ts new file mode 100644 index 00000000..316a6553 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/placement.ts @@ -0,0 +1,1238 @@ +import {CollisionIndex} from './collision_index'; +import type {FeatureKey} from './collision_index'; +import {EXTENT} from '../data/extent'; +import * as symbolSize from './symbol_size'; +import * as projection from './projection'; +import {getAnchorJustification} from './symbol_layout'; +import {getAnchorAlignment, WritingMode} from './shaping'; +import {mat4} from 'gl-matrix'; +import {pixelsToTileUnits} from '../source/pixels_to_tile_units'; +import Point from '@mapbox/point-geometry'; +import type {Transform} from '../geo/transform'; +import type {StyleLayer} from '../style/style_layer'; +import {PossiblyEvaluated} from '../style/properties'; +import type {SymbolLayoutProps, SymbolLayoutPropsPossiblyEvaluated} from '../style/style_layer/symbol_style_layer_properties.g'; +import {getOverlapMode, OverlapMode} from '../style/style_layer/overlap_mode'; + +import type {Tile} from '../source/tile'; +import {SymbolBucket, CollisionArrays, SingleCollisionBox} from '../data/bucket/symbol_bucket'; + +import type {CollisionBoxArray, CollisionVertexArray, SymbolInstance, TextAnchorOffset} from '../data/array_types.g'; +import type {FeatureIndex} from '../data/feature_index'; +import type {OverscaledTileID} from '../source/tile_id'; +import {Terrain} from '../render/terrain'; +import {warnOnce} from '../util/util'; +import {TextAnchor, TextAnchorEnum} from '../style/style_layer/variable_text_anchor'; + +class OpacityState { + opacity: number; + placed: boolean; + constructor(prevState: OpacityState, increment: number, placed: boolean, skipFade?: boolean | null) { + if (prevState) { + this.opacity = Math.max(0, Math.min(1, prevState.opacity + (prevState.placed ? increment : -increment))); + } else { + this.opacity = (skipFade && placed) ? 1 : 0; + } + this.placed = placed; + } + isHidden() { + return this.opacity === 0 && !this.placed; + } +} + +class JointOpacityState { + text: OpacityState; + icon: OpacityState; + constructor(prevState: JointOpacityState, increment: number, placedText: boolean, placedIcon: boolean, skipFade?: boolean | null) { + this.text = new OpacityState(prevState ? prevState.text : null, increment, placedText, skipFade); + this.icon = new OpacityState(prevState ? prevState.icon : null, increment, placedIcon, skipFade); + } + isHidden() { + return this.text.isHidden() && this.icon.isHidden(); + } +} + +class JointPlacement { + text: boolean; + icon: boolean; + // skipFade = outside viewport, but within CollisionIndex::viewportPadding px of the edge + // Because these symbols aren't onscreen yet, we can skip the "fade in" animation, + // and if a subsequent viewport change brings them into view, they'll be fully + // visible right away. + skipFade: boolean; + constructor(text: boolean, icon: boolean, skipFade: boolean) { + this.text = text; + this.icon = icon; + this.skipFade = skipFade; + } +} + +class CollisionCircleArray { + // Stores collision circles and placement matrices of a bucket for debug rendering. + invProjMatrix: mat4; + viewportMatrix: mat4; + circles: Array; + + constructor() { + this.invProjMatrix = mat4.create(); + this.viewportMatrix = mat4.create(); + this.circles = []; + } +} + +export class RetainedQueryData { + bucketInstanceId: number; + featureIndex: FeatureIndex; + sourceLayerIndex: number; + bucketIndex: number; + tileID: OverscaledTileID; + featureSortOrder: Array; + constructor(bucketInstanceId: number, + featureIndex: FeatureIndex, + sourceLayerIndex: number, + bucketIndex: number, + tileID: OverscaledTileID) { + this.bucketInstanceId = bucketInstanceId; + this.featureIndex = featureIndex; + this.sourceLayerIndex = sourceLayerIndex; + this.bucketIndex = bucketIndex; + this.tileID = tileID; + } +} + +type CollisionGroup = { + ID: number; + predicate?: (key: FeatureKey) => boolean; +}; + +class CollisionGroups { + collisionGroups: {[groupName: string]: CollisionGroup}; + maxGroupID: number; + crossSourceCollisions: boolean; + + constructor(crossSourceCollisions: boolean) { + this.crossSourceCollisions = crossSourceCollisions; + this.maxGroupID = 0; + this.collisionGroups = {}; + } + + get(sourceID: string) { + // The predicate/groupID mechanism allows for arbitrary grouping, + // but the current interface defines one source == one group when + // crossSourceCollisions == true. + if (!this.crossSourceCollisions) { + if (!this.collisionGroups[sourceID]) { + const nextGroupID = ++this.maxGroupID; + this.collisionGroups[sourceID] = { + ID: nextGroupID, + predicate: (key) => { + return key.collisionGroupID === nextGroupID; + } + }; + } + return this.collisionGroups[sourceID]; + } else { + return {ID: 0, predicate: null}; + } + } +} + +function calculateVariableLayoutShift( + anchor: TextAnchor, + width: number, + height: number, + textOffset: [number, number], + textBoxScale: number +): Point { + const {horizontalAlign, verticalAlign} = getAnchorAlignment(anchor); + const shiftX = -(horizontalAlign - 0.5) * width; + const shiftY = -(verticalAlign - 0.5) * height; + return new Point( + shiftX + textOffset[0] * textBoxScale, + shiftY + textOffset[1] * textBoxScale + ); +} + +function shiftVariableCollisionBox(collisionBox: SingleCollisionBox, + shiftX: number, shiftY: number, + rotateWithMap: boolean, pitchWithMap: boolean, + angle: number) { + const {x1, x2, y1, y2, anchorPointX, anchorPointY} = collisionBox; + const rotatedOffset = new Point(shiftX, shiftY); + if (rotateWithMap) { + rotatedOffset._rotate(pitchWithMap ? angle : -angle); + } + return { + x1: x1 + rotatedOffset.x, + y1: y1 + rotatedOffset.y, + x2: x2 + rotatedOffset.x, + y2: y2 + rotatedOffset.y, + // symbol anchor point stays the same regardless of text-anchor + anchorPointX, + anchorPointY + }; +} + +export type VariableOffset = { + textOffset: [number, number]; + width: number; + height: number; + anchor: TextAnchor; + textBoxScale: number; + prevAnchor?: TextAnchor; +}; + +type TileLayerParameters = { + bucket: SymbolBucket; + layout: PossiblyEvaluated; + posMatrix: mat4; + textLabelPlaneMatrix: mat4; + labelToScreenMatrix: mat4; + scale: number; + textPixelRatio: number; + holdingForFade: boolean; + collisionBoxArray: CollisionBoxArray; + partiallyEvaluatedTextSize: { + uSize: number; + uSizeT: number; + }; + collisionGroup: CollisionGroup; +}; + +export type BucketPart = { + sortKey?: number | void; + symbolInstanceStart: number; + symbolInstanceEnd: number; + parameters: TileLayerParameters; +}; + +export type CrossTileID = string | number; + +export class Placement { + transform: Transform; + terrain: Terrain; + collisionIndex: CollisionIndex; + placements: { + [_ in CrossTileID]: JointPlacement; + }; + opacities: { + [_ in CrossTileID]: JointOpacityState; + }; + variableOffsets: { + [_ in CrossTileID]: VariableOffset; + }; + placedOrientations: { + [_ in CrossTileID]: number; + }; + commitTime: number; + prevZoomAdjustment: number; + lastPlacementChangeTime: number; + stale: boolean; + fadeDuration: number; + retainedQueryData: { + [_: number]: RetainedQueryData; + }; + collisionGroups: CollisionGroups; + prevPlacement: Placement; + zoomAtLastRecencyCheck: number; + collisionCircleArrays: { + [k in any]: CollisionCircleArray; + }; + + constructor(transform: Transform, terrain: Terrain, fadeDuration: number, crossSourceCollisions: boolean, prevPlacement?: Placement) { + this.transform = transform.clone(); + this.terrain = terrain; + this.collisionIndex = new CollisionIndex(this.transform); + this.placements = {}; + this.opacities = {}; + this.variableOffsets = {}; + this.stale = false; + this.commitTime = 0; + this.fadeDuration = fadeDuration; + this.retainedQueryData = {}; + this.collisionGroups = new CollisionGroups(crossSourceCollisions); + this.collisionCircleArrays = {}; + + this.prevPlacement = prevPlacement; + if (prevPlacement) { + prevPlacement.prevPlacement = undefined; // Only hold on to one placement back + } + + this.placedOrientations = {}; + } + + getBucketParts(results: Array, styleLayer: StyleLayer, tile: Tile, sortAcrossTiles: boolean) { + const symbolBucket = (tile.getBucket(styleLayer) as SymbolBucket); + const bucketFeatureIndex = tile.latestFeatureIndex; + if (!symbolBucket || !bucketFeatureIndex || styleLayer.id !== symbolBucket.layerIds[0]) + return; + + const collisionBoxArray = tile.collisionBoxArray; + + const layout = symbolBucket.layers[0].layout; + + const scale = Math.pow(2, this.transform.zoom - tile.tileID.overscaledZ); + const textPixelRatio = tile.tileSize / EXTENT; + + const posMatrix = this.transform.calculatePosMatrix(tile.tileID.toUnwrapped()); + + const pitchWithMap = layout.get('text-pitch-alignment') === 'map'; + const rotateWithMap = layout.get('text-rotation-alignment') === 'map'; + const pixelsToTiles = pixelsToTileUnits(tile, 1, this.transform.zoom); + + const textLabelPlaneMatrix = projection.getLabelPlaneMatrix(posMatrix, + pitchWithMap, + rotateWithMap, + this.transform, + pixelsToTiles); + + let labelToScreenMatrix = null; + + if (pitchWithMap) { + const glMatrix = projection.getGlCoordMatrix( + posMatrix, + pitchWithMap, + rotateWithMap, + this.transform, + pixelsToTiles); + + labelToScreenMatrix = mat4.multiply([] as any, this.transform.labelPlaneMatrix, glMatrix); + } + + // As long as this placement lives, we have to hold onto this bucket's + // matching FeatureIndex/data for querying purposes + this.retainedQueryData[symbolBucket.bucketInstanceId] = new RetainedQueryData( + symbolBucket.bucketInstanceId, + bucketFeatureIndex, + symbolBucket.sourceLayerIndex, + symbolBucket.index, + tile.tileID + ); + + const parameters = { + bucket: symbolBucket, + layout, + posMatrix, + textLabelPlaneMatrix, + labelToScreenMatrix, + scale, + textPixelRatio, + holdingForFade: tile.holdingForFade(), + collisionBoxArray, + partiallyEvaluatedTextSize: symbolSize.evaluateSizeForZoom(symbolBucket.textSizeData, this.transform.zoom), + collisionGroup: this.collisionGroups.get(symbolBucket.sourceID) + }; + + if (sortAcrossTiles) { + for (const range of symbolBucket.sortKeyRanges) { + const {sortKey, symbolInstanceStart, symbolInstanceEnd} = range; + results.push({sortKey, symbolInstanceStart, symbolInstanceEnd, parameters}); + } + } else { + results.push({ + symbolInstanceStart: 0, + symbolInstanceEnd: symbolBucket.symbolInstances.length, + parameters + }); + } + } + + attemptAnchorPlacement( + textAnchorOffset: TextAnchorOffset, + textBox: SingleCollisionBox, + width: number, + height: number, + textBoxScale: number, + rotateWithMap: boolean, + pitchWithMap: boolean, + textPixelRatio: number, + posMatrix: mat4, + collisionGroup: CollisionGroup, + textOverlapMode: OverlapMode, + symbolInstance: SymbolInstance, + bucket: SymbolBucket, + orientation: number, + iconBox?: SingleCollisionBox | null, + getElevation?: (x: number, y: number) => number + ): { + shift: Point; + placedGlyphBoxes: { + box: Array; + offscreen: boolean; + }; + } { + + const anchor = TextAnchorEnum[textAnchorOffset.textAnchor] as TextAnchor; + const textOffset = [textAnchorOffset.textOffset0, textAnchorOffset.textOffset1] as [number, number]; + const shift = calculateVariableLayoutShift(anchor, width, height, textOffset, textBoxScale); + + const placedGlyphBoxes = this.collisionIndex.placeCollisionBox( + shiftVariableCollisionBox( + textBox, shift.x, shift.y, + rotateWithMap, pitchWithMap, this.transform.angle), + textOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation); + + if (iconBox) { + const placedIconBoxes = this.collisionIndex.placeCollisionBox( + shiftVariableCollisionBox( + iconBox, shift.x, shift.y, + rotateWithMap, pitchWithMap, this.transform.angle), + textOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation); + if (placedIconBoxes.box.length === 0) return; + } + + if (placedGlyphBoxes.box.length > 0) { + let prevAnchor; + // If this label was placed in the previous placement, record the anchor position + // to allow us to animate the transition + if (this.prevPlacement && + this.prevPlacement.variableOffsets[symbolInstance.crossTileID] && + this.prevPlacement.placements[symbolInstance.crossTileID] && + this.prevPlacement.placements[symbolInstance.crossTileID].text) { + prevAnchor = this.prevPlacement.variableOffsets[symbolInstance.crossTileID].anchor; + } + if (symbolInstance.crossTileID === 0) throw new Error('symbolInstance.crossTileID can\'t be 0'); + this.variableOffsets[symbolInstance.crossTileID] = { + textOffset, + width, + height, + anchor, + textBoxScale, + prevAnchor + }; + this.markUsedJustification(bucket, anchor, symbolInstance, orientation); + + if (bucket.allowVerticalPlacement) { + this.markUsedOrientation(bucket, orientation, symbolInstance); + this.placedOrientations[symbolInstance.crossTileID] = orientation; + } + + return {shift, placedGlyphBoxes}; + } + } + + placeLayerBucketPart(bucketPart: BucketPart, seenCrossTileIDs: { + [k in string | number]: boolean; + }, showCollisionBoxes: boolean) { + + const { + bucket, + layout, + posMatrix, + textLabelPlaneMatrix, + labelToScreenMatrix, + textPixelRatio, + holdingForFade, + collisionBoxArray, + partiallyEvaluatedTextSize, + collisionGroup + } = bucketPart.parameters; + + const textOptional = layout.get('text-optional'); + const iconOptional = layout.get('icon-optional'); + const textOverlapMode = getOverlapMode(layout, 'text-overlap', 'text-allow-overlap'); + const textAlwaysOverlap = textOverlapMode === 'always'; + const iconOverlapMode = getOverlapMode(layout, 'icon-overlap', 'icon-allow-overlap'); + const iconAlwaysOverlap = iconOverlapMode === 'always'; + const rotateWithMap = layout.get('text-rotation-alignment') === 'map'; + const pitchWithMap = layout.get('text-pitch-alignment') === 'map'; + const hasIconTextFit = layout.get('icon-text-fit') !== 'none'; + const zOrderByViewportY = layout.get('symbol-z-order') === 'viewport-y'; + + // This logic is similar to the "defaultOpacityState" logic below in updateBucketOpacities + // If we know a symbol is always supposed to show, force it to be marked visible even if + // it wasn't placed into the collision index (because some or all of it was outside the range + // of the collision grid). + // There is a subtle edge case here we're accepting: + // Symbol A has text-allow-overlap: true, icon-allow-overlap: true, icon-optional: false + // A's icon is outside the grid, so doesn't get placed + // A's text would be inside grid, but doesn't get placed because of icon-optional: false + // We still show A because of the allow-overlap settings. + // Symbol B has allow-overlap: false, and gets placed where A's text would be + // On panning in, there is a short period when Symbol B and Symbol A will overlap + // This is the reverse of our normal policy of "fade in on pan", but should look like any other + // collision and hopefully not be too noticeable. + // See https://github.com/mapbox/mapbox-gl-js/issues/7172 + const alwaysShowText = textAlwaysOverlap && (iconAlwaysOverlap || !bucket.hasIconData() || iconOptional); + const alwaysShowIcon = iconAlwaysOverlap && (textAlwaysOverlap || !bucket.hasTextData() || textOptional); + + if (!bucket.collisionArrays && collisionBoxArray) { + bucket.deserializeCollisionBoxes(collisionBoxArray); + } + + const tileID = this.retainedQueryData[bucket.bucketInstanceId].tileID; + const getElevation = this.terrain ? (x: number, y: number) => this.terrain.getElevation(tileID, x, y) : null; + + const placeSymbol = (symbolInstance: SymbolInstance, collisionArrays: CollisionArrays) => { + if (seenCrossTileIDs[symbolInstance.crossTileID]) return; + if (holdingForFade) { + // Mark all symbols from this tile as "not placed", but don't add to seenCrossTileIDs, because we don't + // know yet if we have a duplicate in a parent tile that _should_ be placed. + this.placements[symbolInstance.crossTileID] = new JointPlacement(false, false, false); + return; + } + + let placeText = false; + let placeIcon = false; + let offscreen = true; + let shift = null; + + let placed = {box: null, offscreen: null}; + let placedVerticalText = {box: null, offscreen: null}; + + let placedGlyphBoxes = null; + let placedGlyphCircles = null; + let placedIconBoxes = null; + let textFeatureIndex = 0; + let verticalTextFeatureIndex = 0; + let iconFeatureIndex = 0; + + if (collisionArrays.textFeatureIndex) { + textFeatureIndex = collisionArrays.textFeatureIndex; + } else if (symbolInstance.useRuntimeCollisionCircles) { + textFeatureIndex = symbolInstance.featureIndex; + } + if (collisionArrays.verticalTextFeatureIndex) { + verticalTextFeatureIndex = collisionArrays.verticalTextFeatureIndex; + } + + const textBox = collisionArrays.textBox; + if (textBox) { + + const updatePreviousOrientationIfNotPlaced = (isPlaced) => { + let previousOrientation = WritingMode.horizontal; + if (bucket.allowVerticalPlacement && !isPlaced && this.prevPlacement) { + const prevPlacedOrientation = this.prevPlacement.placedOrientations[symbolInstance.crossTileID]; + if (prevPlacedOrientation) { + this.placedOrientations[symbolInstance.crossTileID] = prevPlacedOrientation; + previousOrientation = prevPlacedOrientation; + this.markUsedOrientation(bucket, previousOrientation, symbolInstance); + } + } + return previousOrientation; + }; + + const placeTextForPlacementModes = (placeHorizontalFn, placeVerticalFn) => { + if (bucket.allowVerticalPlacement && symbolInstance.numVerticalGlyphVertices > 0 && collisionArrays.verticalTextBox) { + for (const placementMode of bucket.writingModes) { + if (placementMode === WritingMode.vertical) { + placed = placeVerticalFn(); + placedVerticalText = placed; + } else { + placed = placeHorizontalFn(); + } + if (placed && placed.box && placed.box.length) break; + } + } else { + placed = placeHorizontalFn(); + } + }; + + const textAnchorOffsetStart = symbolInstance.textAnchorOffsetStartIndex; + const textAnchorOffsetEnd = symbolInstance.textAnchorOffsetEndIndex; + + // If start+end indices match, text-variable-anchor is not in play. + if (textAnchorOffsetEnd === textAnchorOffsetStart) { + const placeBox = (collisionTextBox, orientation) => { + const placedFeature = this.collisionIndex.placeCollisionBox( + collisionTextBox, + textOverlapMode, + textPixelRatio, + posMatrix, + collisionGroup.predicate, + getElevation + ); + if (placedFeature && placedFeature.box && placedFeature.box.length) { + this.markUsedOrientation(bucket, orientation, symbolInstance); + this.placedOrientations[symbolInstance.crossTileID] = orientation; + } + return placedFeature; + }; + + const placeHorizontal = () => { + return placeBox(textBox, WritingMode.horizontal); + }; + + const placeVertical = () => { + const verticalTextBox = collisionArrays.verticalTextBox; + if (bucket.allowVerticalPlacement && symbolInstance.numVerticalGlyphVertices > 0 && verticalTextBox) { + return placeBox(verticalTextBox, WritingMode.vertical); + } + return {box: null, offscreen: null}; + }; + + placeTextForPlacementModes(placeHorizontal, placeVertical); + updatePreviousOrientationIfNotPlaced(placed && placed.box && placed.box.length); + + } else { + // If this symbol was in the last placement, prefer placement using same anchor, if it's still available + let prevAnchor = TextAnchorEnum[this.prevPlacement?.variableOffsets[symbolInstance.crossTileID]?.anchor]; + + const placeBoxForVariableAnchors = (collisionTextBox, collisionIconBox, orientation) => { + const width = collisionTextBox.x2 - collisionTextBox.x1; + const height = collisionTextBox.y2 - collisionTextBox.y1; + const textBoxScale = symbolInstance.textBoxScale; + const variableIconBox = hasIconTextFit && (iconOverlapMode === 'never') ? collisionIconBox : null; + + let placedBox: { + box: Array; + offscreen: boolean; + } = {box: [], offscreen: false}; + let placementPasses = (textOverlapMode === 'never') ? 1 : 2; + let overlapMode: OverlapMode = 'never'; + + if (prevAnchor) { + placementPasses++; + } + + for (let pass = 0; pass < placementPasses; pass++) { + for (let i = textAnchorOffsetStart; i < textAnchorOffsetEnd; i++) { + const textAnchorOffset = bucket.textAnchorOffsets.get(i); + + if (prevAnchor && textAnchorOffset.textAnchor !== prevAnchor) { + continue; + } + + const result = this.attemptAnchorPlacement( + textAnchorOffset, collisionTextBox, width, height, + textBoxScale, rotateWithMap, pitchWithMap, textPixelRatio, posMatrix, + collisionGroup, overlapMode, symbolInstance, bucket, orientation, variableIconBox, getElevation); + + if (result) { + placedBox = result.placedGlyphBoxes; + if (placedBox && placedBox.box && placedBox.box.length) { + placeText = true; + shift = result.shift; + return placedBox; + } + } + } + + if (prevAnchor) { + prevAnchor = null; + } else { + overlapMode = textOverlapMode; + } + } + + return placedBox; + }; + + const placeHorizontal = () => { + return placeBoxForVariableAnchors(textBox, collisionArrays.iconBox, WritingMode.horizontal); + }; + + const placeVertical = () => { + const verticalTextBox = collisionArrays.verticalTextBox; + const wasPlaced = placed && placed.box && placed.box.length; + if (bucket.allowVerticalPlacement && !wasPlaced && symbolInstance.numVerticalGlyphVertices > 0 && verticalTextBox) { + return placeBoxForVariableAnchors(verticalTextBox, collisionArrays.verticalIconBox, WritingMode.vertical); + } + return {box: null, offscreen: null}; + }; + + placeTextForPlacementModes(placeHorizontal, placeVertical); + + if (placed) { + placeText = placed.box; + offscreen = placed.offscreen; + } + + const prevOrientation = updatePreviousOrientationIfNotPlaced(placed && placed.box); + + // If we didn't get placed, we still need to copy our position from the last placement for + // fade animations + if (!placeText && this.prevPlacement) { + const prevOffset = this.prevPlacement.variableOffsets[symbolInstance.crossTileID]; + if (prevOffset) { + this.variableOffsets[symbolInstance.crossTileID] = prevOffset; + this.markUsedJustification(bucket, prevOffset.anchor, symbolInstance, prevOrientation); + } + } + + } + } + + placedGlyphBoxes = placed; + placeText = placedGlyphBoxes && placedGlyphBoxes.box && placedGlyphBoxes.box.length > 0; + + offscreen = placedGlyphBoxes && placedGlyphBoxes.offscreen; + + if (symbolInstance.useRuntimeCollisionCircles) { + const placedSymbol = bucket.text.placedSymbolArray.get(symbolInstance.centerJustifiedTextSymbolIndex); + const fontSize = symbolSize.evaluateSizeForFeature(bucket.textSizeData, partiallyEvaluatedTextSize, placedSymbol); + + const textPixelPadding = layout.get('text-padding'); + const circlePixelDiameter = symbolInstance.collisionCircleDiameter; + + placedGlyphCircles = this.collisionIndex.placeCollisionCircles( + textOverlapMode, + placedSymbol, + bucket.lineVertexArray, + bucket.glyphOffsetArray, + fontSize, + posMatrix, + textLabelPlaneMatrix, + labelToScreenMatrix, + showCollisionBoxes, + pitchWithMap, + collisionGroup.predicate, + circlePixelDiameter, + textPixelPadding, + getElevation + ); + + if (placedGlyphCircles.circles.length && placedGlyphCircles.collisionDetected && !showCollisionBoxes) { + warnOnce('Collisions detected, but collision boxes are not shown'); + } + + // If text-overlap is set to 'always', force "placedCircles" to true + // In theory there should always be at least one circle placed + // in this case, but for now quirks in text-anchor + // and text-offset may prevent that from being true. + placeText = textAlwaysOverlap || (placedGlyphCircles.circles.length > 0 && !placedGlyphCircles.collisionDetected); + offscreen = offscreen && placedGlyphCircles.offscreen; + } + + if (collisionArrays.iconFeatureIndex) { + iconFeatureIndex = collisionArrays.iconFeatureIndex; + } + + if (collisionArrays.iconBox) { + const placeIconFeature = iconBox => { + const shiftedIconBox = hasIconTextFit && shift ? + shiftVariableCollisionBox( + iconBox, shift.x, shift.y, + rotateWithMap, pitchWithMap, this.transform.angle) : + iconBox; + return this.collisionIndex.placeCollisionBox(shiftedIconBox, + iconOverlapMode, textPixelRatio, posMatrix, collisionGroup.predicate, getElevation); + }; + + if (placedVerticalText && placedVerticalText.box && placedVerticalText.box.length && collisionArrays.verticalIconBox) { + placedIconBoxes = placeIconFeature(collisionArrays.verticalIconBox); + placeIcon = placedIconBoxes.box.length > 0; + } else { + placedIconBoxes = placeIconFeature(collisionArrays.iconBox); + placeIcon = placedIconBoxes.box.length > 0; + } + offscreen = offscreen && placedIconBoxes.offscreen; + } + + const iconWithoutText = textOptional || + (symbolInstance.numHorizontalGlyphVertices === 0 && symbolInstance.numVerticalGlyphVertices === 0); + const textWithoutIcon = iconOptional || symbolInstance.numIconVertices === 0; + + // Combine the scales for icons and text. + if (!iconWithoutText && !textWithoutIcon) { + placeIcon = placeText = placeIcon && placeText; + } else if (!textWithoutIcon) { + placeText = placeIcon && placeText; + } else if (!iconWithoutText) { + placeIcon = placeIcon && placeText; + } + + if (placeText && placedGlyphBoxes && placedGlyphBoxes.box) { + if (placedVerticalText && placedVerticalText.box && verticalTextFeatureIndex) { + this.collisionIndex.insertCollisionBox( + placedGlyphBoxes.box, + textOverlapMode, + layout.get('text-ignore-placement'), + bucket.bucketInstanceId, + verticalTextFeatureIndex, + collisionGroup.ID); + } else { + this.collisionIndex.insertCollisionBox( + placedGlyphBoxes.box, + textOverlapMode, + layout.get('text-ignore-placement'), + bucket.bucketInstanceId, + textFeatureIndex, + collisionGroup.ID); + } + + } + if (placeIcon && placedIconBoxes) { + this.collisionIndex.insertCollisionBox( + placedIconBoxes.box, + iconOverlapMode, + layout.get('icon-ignore-placement'), + bucket.bucketInstanceId, + iconFeatureIndex, + collisionGroup.ID); + } + if (placedGlyphCircles) { + if (placeText) { + this.collisionIndex.insertCollisionCircles( + placedGlyphCircles.circles, + textOverlapMode, + layout.get('text-ignore-placement'), + bucket.bucketInstanceId, + textFeatureIndex, + collisionGroup.ID); + } + + if (showCollisionBoxes) { + const id = bucket.bucketInstanceId; + let circleArray = this.collisionCircleArrays[id]; + + // Group collision circles together by bucket. Circles can't be pushed forward for rendering yet as the symbol placement + // for a bucket is not guaranteed to be complete before the commit-function has been called + if (circleArray === undefined) + circleArray = this.collisionCircleArrays[id] = new CollisionCircleArray(); + + for (let i = 0; i < placedGlyphCircles.circles.length; i += 4) { + circleArray.circles.push(placedGlyphCircles.circles[i + 0]); // x + circleArray.circles.push(placedGlyphCircles.circles[i + 1]); // y + circleArray.circles.push(placedGlyphCircles.circles[i + 2]); // radius + circleArray.circles.push(placedGlyphCircles.collisionDetected ? 1 : 0); // collisionDetected-flag + } + } + } + + if (symbolInstance.crossTileID === 0) throw new Error('symbolInstance.crossTileID can\'t be 0'); + if (bucket.bucketInstanceId === 0) throw new Error('bucket.bucketInstanceId can\'t be 0'); + + this.placements[symbolInstance.crossTileID] = new JointPlacement(placeText || alwaysShowText, placeIcon || alwaysShowIcon, offscreen || bucket.justReloaded); + seenCrossTileIDs[symbolInstance.crossTileID] = true; + }; + + if (zOrderByViewportY) { + if (bucketPart.symbolInstanceStart !== 0) throw new Error('bucket.bucketInstanceId should be 0'); + const symbolIndexes = bucket.getSortedSymbolIndexes(this.transform.angle); + for (let i = symbolIndexes.length - 1; i >= 0; --i) { + const symbolIndex = symbolIndexes[i]; + placeSymbol(bucket.symbolInstances.get(symbolIndex), bucket.collisionArrays[symbolIndex]); + } + } else { + for (let i = bucketPart.symbolInstanceStart; i < bucketPart.symbolInstanceEnd; i++) { + placeSymbol(bucket.symbolInstances.get(i), bucket.collisionArrays[i]); + } + } + + if (showCollisionBoxes && bucket.bucketInstanceId in this.collisionCircleArrays) { + const circleArray = this.collisionCircleArrays[bucket.bucketInstanceId]; + + // Store viewport and inverse projection matrices per bucket + mat4.invert(circleArray.invProjMatrix, posMatrix); + circleArray.viewportMatrix = this.collisionIndex.getViewportMatrix(); + } + + bucket.justReloaded = false; + } + + markUsedJustification(bucket: SymbolBucket, placedAnchor: TextAnchor, symbolInstance: SymbolInstance, orientation: number) { + const justifications = { + 'left': symbolInstance.leftJustifiedTextSymbolIndex, + 'center': symbolInstance.centerJustifiedTextSymbolIndex, + 'right': symbolInstance.rightJustifiedTextSymbolIndex + }; + + let autoIndex; + if (orientation === WritingMode.vertical) { + autoIndex = symbolInstance.verticalPlacedTextSymbolIndex; + } else { + autoIndex = justifications[getAnchorJustification(placedAnchor)]; + } + + const indexes = [ + symbolInstance.leftJustifiedTextSymbolIndex, + symbolInstance.centerJustifiedTextSymbolIndex, + symbolInstance.rightJustifiedTextSymbolIndex, + symbolInstance.verticalPlacedTextSymbolIndex + ]; + + for (const index of indexes) { + if (index >= 0) { + if (autoIndex >= 0 && index !== autoIndex) { + // There are multiple justifications and this one isn't it: shift offscreen + bucket.text.placedSymbolArray.get(index).crossTileID = 0; + } else { + // Either this is the chosen justification or the justification is hardwired: use this one + bucket.text.placedSymbolArray.get(index).crossTileID = symbolInstance.crossTileID; + } + } + } + } + + markUsedOrientation(bucket: SymbolBucket, orientation: number, symbolInstance: SymbolInstance) { + const horizontal = (orientation === WritingMode.horizontal || orientation === WritingMode.horizontalOnly) ? orientation : 0; + const vertical = orientation === WritingMode.vertical ? orientation : 0; + + const horizontalIndexes = [ + symbolInstance.leftJustifiedTextSymbolIndex, + symbolInstance.centerJustifiedTextSymbolIndex, + symbolInstance.rightJustifiedTextSymbolIndex + ]; + + for (const index of horizontalIndexes) { + bucket.text.placedSymbolArray.get(index).placedOrientation = horizontal; + } + + if (symbolInstance.verticalPlacedTextSymbolIndex) { + bucket.text.placedSymbolArray.get(symbolInstance.verticalPlacedTextSymbolIndex).placedOrientation = vertical; + } + } + + commit(now: number): void { + this.commitTime = now; + this.zoomAtLastRecencyCheck = this.transform.zoom; + + const prevPlacement = this.prevPlacement; + let placementChanged = false; + + this.prevZoomAdjustment = prevPlacement ? prevPlacement.zoomAdjustment(this.transform.zoom) : 0; + const increment = prevPlacement ? prevPlacement.symbolFadeChange(now) : 1; + + const prevOpacities = prevPlacement ? prevPlacement.opacities : {}; + const prevOffsets = prevPlacement ? prevPlacement.variableOffsets : {}; + const prevOrientations = prevPlacement ? prevPlacement.placedOrientations : {}; + + // add the opacities from the current placement, and copy their current values from the previous placement + for (const crossTileID in this.placements) { + const jointPlacement = this.placements[crossTileID]; + const prevOpacity = prevOpacities[crossTileID]; + if (prevOpacity) { + this.opacities[crossTileID] = new JointOpacityState(prevOpacity, increment, jointPlacement.text, jointPlacement.icon); + placementChanged = placementChanged || + jointPlacement.text !== prevOpacity.text.placed || + jointPlacement.icon !== prevOpacity.icon.placed; + } else { + this.opacities[crossTileID] = new JointOpacityState(null, increment, jointPlacement.text, jointPlacement.icon, jointPlacement.skipFade); + placementChanged = placementChanged || jointPlacement.text || jointPlacement.icon; + } + } + + // copy and update values from the previous placement that aren't in the current placement but haven't finished fading + for (const crossTileID in prevOpacities) { + const prevOpacity = prevOpacities[crossTileID]; + if (!this.opacities[crossTileID]) { + const jointOpacity = new JointOpacityState(prevOpacity, increment, false, false); + if (!jointOpacity.isHidden()) { + this.opacities[crossTileID] = jointOpacity; + placementChanged = placementChanged || prevOpacity.text.placed || prevOpacity.icon.placed; + } + } + } + for (const crossTileID in prevOffsets) { + if (!this.variableOffsets[crossTileID] && this.opacities[crossTileID] && !this.opacities[crossTileID].isHidden()) { + this.variableOffsets[crossTileID] = prevOffsets[crossTileID]; + } + } + + for (const crossTileID in prevOrientations) { + if (!this.placedOrientations[crossTileID] && this.opacities[crossTileID] && !this.opacities[crossTileID].isHidden()) { + this.placedOrientations[crossTileID] = prevOrientations[crossTileID]; + } + } + + // this.lastPlacementChangeTime is the time of the last commit() that + // resulted in a placement change -- in other words, the start time of + // the last symbol fade animation + if (prevPlacement && prevPlacement.lastPlacementChangeTime === undefined) { + throw new Error('Last placement time for previous placement is not defined'); + } + if (placementChanged) { + this.lastPlacementChangeTime = now; + } else if (typeof this.lastPlacementChangeTime !== 'number') { + this.lastPlacementChangeTime = prevPlacement ? prevPlacement.lastPlacementChangeTime : now; + } + } + + updateLayerOpacities(styleLayer: StyleLayer, tiles: Array) { + const seenCrossTileIDs = {}; + for (const tile of tiles) { + const symbolBucket = tile.getBucket(styleLayer) as SymbolBucket; + if (symbolBucket && tile.latestFeatureIndex && styleLayer.id === symbolBucket.layerIds[0]) { + this.updateBucketOpacities(symbolBucket, seenCrossTileIDs, tile.collisionBoxArray); + } + } + } + + updateBucketOpacities(bucket: SymbolBucket, seenCrossTileIDs: { + [k in string | number]: boolean; + }, collisionBoxArray?: CollisionBoxArray | null) { + if (bucket.hasTextData()) { + bucket.text.opacityVertexArray.clear(); + bucket.text.hasVisibleVertices = false; + } + if (bucket.hasIconData()) { + bucket.icon.opacityVertexArray.clear(); + bucket.icon.hasVisibleVertices = false; + } + if (bucket.hasIconCollisionBoxData()) bucket.iconCollisionBox.collisionVertexArray.clear(); + if (bucket.hasTextCollisionBoxData()) bucket.textCollisionBox.collisionVertexArray.clear(); + + const layer = bucket.layers[0]; + const layout = layer.layout; + const duplicateOpacityState = new JointOpacityState(null, 0, false, false, true); + const textAllowOverlap = layout.get('text-allow-overlap'); + const iconAllowOverlap = layout.get('icon-allow-overlap'); + const hasVariablePlacement = layer._unevaluatedLayout.hasValue('text-variable-anchor') || layer._unevaluatedLayout.hasValue('text-variable-anchor-offset'); + const rotateWithMap = layout.get('text-rotation-alignment') === 'map'; + const pitchWithMap = layout.get('text-pitch-alignment') === 'map'; + const hasIconTextFit = layout.get('icon-text-fit') !== 'none'; + // If allow-overlap is true, we can show symbols before placement runs on them + // But we have to wait for placement if we potentially depend on a paired icon/text + // with allow-overlap: false. + // See https://github.com/mapbox/mapbox-gl-js/issues/7032 + const defaultOpacityState = new JointOpacityState(null, 0, + textAllowOverlap && (iconAllowOverlap || !bucket.hasIconData() || layout.get('icon-optional')), + iconAllowOverlap && (textAllowOverlap || !bucket.hasTextData() || layout.get('text-optional')), + true); + + if (!bucket.collisionArrays && collisionBoxArray && ((bucket.hasIconCollisionBoxData() || bucket.hasTextCollisionBoxData()))) { + bucket.deserializeCollisionBoxes(collisionBoxArray); + } + + const addOpacities = (iconOrText, numVertices: number, opacity: number) => { + for (let i = 0; i < numVertices / 4; i++) { + iconOrText.opacityVertexArray.emplaceBack(opacity); + } + iconOrText.hasVisibleVertices = iconOrText.hasVisibleVertices || (opacity !== PACKED_HIDDEN_OPACITY); + }; + + for (let s = 0; s < bucket.symbolInstances.length; s++) { + const symbolInstance = bucket.symbolInstances.get(s); + const { + numHorizontalGlyphVertices, + numVerticalGlyphVertices, + crossTileID + } = symbolInstance; + + const isDuplicate = seenCrossTileIDs[crossTileID]; + + let opacityState = this.opacities[crossTileID]; + if (isDuplicate) { + opacityState = duplicateOpacityState; + } else if (!opacityState) { + opacityState = defaultOpacityState; + // store the state so that future placements use it as a starting point + this.opacities[crossTileID] = opacityState; + } + + seenCrossTileIDs[crossTileID] = true; + + const hasText = numHorizontalGlyphVertices > 0 || numVerticalGlyphVertices > 0; + const hasIcon = symbolInstance.numIconVertices > 0; + + const placedOrientation = this.placedOrientations[symbolInstance.crossTileID]; + const horizontalHidden = placedOrientation === WritingMode.vertical; + const verticalHidden = placedOrientation === WritingMode.horizontal || placedOrientation === WritingMode.horizontalOnly; + + if (hasText) { + const packedOpacity = packOpacity(opacityState.text); + // Vertical text fades in/out on collision the same way as corresponding + // horizontal text. Switch between vertical/horizontal should be instantaneous + const horizontalOpacity = horizontalHidden ? PACKED_HIDDEN_OPACITY : packedOpacity; + addOpacities(bucket.text, numHorizontalGlyphVertices, horizontalOpacity); + const verticalOpacity = verticalHidden ? PACKED_HIDDEN_OPACITY : packedOpacity; + addOpacities(bucket.text, numVerticalGlyphVertices, verticalOpacity); + + // If this label is completely faded, mark it so that we don't have to calculate + // its position at render time. If this layer has variable placement, shift the various + // symbol instances appropriately so that symbols from buckets that have yet to be placed + // offset appropriately. + const symbolHidden = opacityState.text.isHidden(); + [ + symbolInstance.rightJustifiedTextSymbolIndex, + symbolInstance.centerJustifiedTextSymbolIndex, + symbolInstance.leftJustifiedTextSymbolIndex + ].forEach(index => { + if (index >= 0) { + bucket.text.placedSymbolArray.get(index).hidden = symbolHidden || horizontalHidden ? 1 : 0; + } + }); + + if (symbolInstance.verticalPlacedTextSymbolIndex >= 0) { + bucket.text.placedSymbolArray.get(symbolInstance.verticalPlacedTextSymbolIndex).hidden = symbolHidden || verticalHidden ? 1 : 0; + } + + const prevOffset = this.variableOffsets[symbolInstance.crossTileID]; + if (prevOffset) { + this.markUsedJustification(bucket, prevOffset.anchor, symbolInstance, placedOrientation); + } + + const prevOrientation = this.placedOrientations[symbolInstance.crossTileID]; + if (prevOrientation) { + this.markUsedJustification(bucket, 'left', symbolInstance, prevOrientation); + this.markUsedOrientation(bucket, prevOrientation, symbolInstance); + } + } + + if (hasIcon) { + const packedOpacity = packOpacity(opacityState.icon); + + const useHorizontal = !(hasIconTextFit && symbolInstance.verticalPlacedIconSymbolIndex && horizontalHidden); + + if (symbolInstance.placedIconSymbolIndex >= 0) { + const horizontalOpacity = useHorizontal ? packedOpacity : PACKED_HIDDEN_OPACITY; + addOpacities(bucket.icon, symbolInstance.numIconVertices, horizontalOpacity); + bucket.icon.placedSymbolArray.get(symbolInstance.placedIconSymbolIndex).hidden = + (opacityState.icon.isHidden() as any); + } + + if (symbolInstance.verticalPlacedIconSymbolIndex >= 0) { + const verticalOpacity = !useHorizontal ? packedOpacity : PACKED_HIDDEN_OPACITY; + addOpacities(bucket.icon, symbolInstance.numVerticalIconVertices, verticalOpacity); + bucket.icon.placedSymbolArray.get(symbolInstance.verticalPlacedIconSymbolIndex).hidden = + (opacityState.icon.isHidden() as any); + } + } + + if (bucket.hasIconCollisionBoxData() || bucket.hasTextCollisionBoxData()) { + const collisionArrays = bucket.collisionArrays[s]; + if (collisionArrays) { + let shift = new Point(0, 0); + if (collisionArrays.textBox || collisionArrays.verticalTextBox) { + let used = true; + if (hasVariablePlacement) { + const variableOffset = this.variableOffsets[crossTileID]; + if (variableOffset) { + // This will show either the currently placed position or the last + // successfully placed position (so you can visualize what collision + // just made the symbol disappear, and the most likely place for the + // symbol to come back) + shift = calculateVariableLayoutShift(variableOffset.anchor, + variableOffset.width, + variableOffset.height, + variableOffset.textOffset, + variableOffset.textBoxScale); + if (rotateWithMap) { + shift._rotate(pitchWithMap ? this.transform.angle : -this.transform.angle); + } + } else { + // No offset -> this symbol hasn't been placed since coming on-screen + // No single box is particularly meaningful and all of them would be too noisy + // Use the center box just to show something's there, but mark it "not used" + used = false; + } + } + + if (collisionArrays.textBox) { + updateCollisionVertices(bucket.textCollisionBox.collisionVertexArray, opacityState.text.placed, !used || horizontalHidden, shift.x, shift.y); + } + if (collisionArrays.verticalTextBox) { + updateCollisionVertices(bucket.textCollisionBox.collisionVertexArray, opacityState.text.placed, !used || verticalHidden, shift.x, shift.y); + } + } + + const verticalIconUsed = Boolean(!verticalHidden && collisionArrays.verticalIconBox); + + if (collisionArrays.iconBox) { + updateCollisionVertices(bucket.iconCollisionBox.collisionVertexArray, opacityState.icon.placed, verticalIconUsed, + hasIconTextFit ? shift.x : 0, + hasIconTextFit ? shift.y : 0); + } + + if (collisionArrays.verticalIconBox) { + updateCollisionVertices(bucket.iconCollisionBox.collisionVertexArray, opacityState.icon.placed, !verticalIconUsed, + hasIconTextFit ? shift.x : 0, + hasIconTextFit ? shift.y : 0); + } + } + } + } + + bucket.sortFeatures(this.transform.angle); + if (this.retainedQueryData[bucket.bucketInstanceId]) { + this.retainedQueryData[bucket.bucketInstanceId].featureSortOrder = bucket.featureSortOrder; + } + + if (bucket.hasTextData() && bucket.text.opacityVertexBuffer) { + bucket.text.opacityVertexBuffer.updateData(bucket.text.opacityVertexArray); + } + if (bucket.hasIconData() && bucket.icon.opacityVertexBuffer) { + bucket.icon.opacityVertexBuffer.updateData(bucket.icon.opacityVertexArray); + } + if (bucket.hasIconCollisionBoxData() && bucket.iconCollisionBox.collisionVertexBuffer) { + bucket.iconCollisionBox.collisionVertexBuffer.updateData(bucket.iconCollisionBox.collisionVertexArray); + } + if (bucket.hasTextCollisionBoxData() && bucket.textCollisionBox.collisionVertexBuffer) { + bucket.textCollisionBox.collisionVertexBuffer.updateData(bucket.textCollisionBox.collisionVertexArray); + } + + if (bucket.text.opacityVertexArray.length !== bucket.text.layoutVertexArray.length / 4) throw new Error(`bucket.text.opacityVertexArray.length (= ${bucket.text.opacityVertexArray.length}) !== bucket.text.layoutVertexArray.length (= ${bucket.text.layoutVertexArray.length}) / 4`); + if (bucket.icon.opacityVertexArray.length !== bucket.icon.layoutVertexArray.length / 4) throw new Error(`bucket.icon.opacityVertexArray.length (= ${bucket.icon.opacityVertexArray.length}) !== bucket.icon.layoutVertexArray.length (= ${bucket.icon.layoutVertexArray.length}) / 4`); + + // Push generated collision circles to the bucket for debug rendering + if (bucket.bucketInstanceId in this.collisionCircleArrays) { + const instance = this.collisionCircleArrays[bucket.bucketInstanceId]; + + bucket.placementInvProjMatrix = instance.invProjMatrix; + bucket.placementViewportMatrix = instance.viewportMatrix; + bucket.collisionCircleArray = instance.circles; + + delete this.collisionCircleArrays[bucket.bucketInstanceId]; + } + } + + symbolFadeChange(now: number) { + return this.fadeDuration === 0 ? + 1 : + ((now - this.commitTime) / this.fadeDuration + this.prevZoomAdjustment); + } + + zoomAdjustment(zoom: number) { + // When zooming out quickly, labels can overlap each other. This + // adjustment is used to reduce the interval between placement calculations + // and to reduce the fade duration when zooming out quickly. Discovering the + // collisions more quickly and fading them more quickly reduces the unwanted effect. + return Math.max(0, (this.transform.zoom - zoom) / 1.5); + } + + hasTransitions(now: number) { + return this.stale || + now - this.lastPlacementChangeTime < this.fadeDuration; + } + + stillRecent(now: number, zoom: number) { + // The adjustment makes placement more frequent when zooming. + // This condition applies the adjustment only after the map has + // stopped zooming. This avoids adding extra jank while zooming. + const durationAdjustment = this.zoomAtLastRecencyCheck === zoom ? + (1 - this.zoomAdjustment(zoom)) : + 1; + this.zoomAtLastRecencyCheck = zoom; + + return this.commitTime + this.fadeDuration * durationAdjustment > now; + } + + setStale() { + this.stale = true; + } +} + +function updateCollisionVertices(collisionVertexArray: CollisionVertexArray, placed: boolean, notUsed: boolean | number, shiftX?: number, shiftY?: number) { + collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0); + collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0); + collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0); + collisionVertexArray.emplaceBack(placed ? 1 : 0, notUsed ? 1 : 0, shiftX || 0, shiftY || 0); +} + +// All four vertices for a glyph will have the same opacity state +// So we pack the opacity into a uint8, and then repeat it four times +// to make a single uint32 that we can upload for each glyph in the +// label. +const shift25 = Math.pow(2, 25); +const shift24 = Math.pow(2, 24); +const shift17 = Math.pow(2, 17); +const shift16 = Math.pow(2, 16); +const shift9 = Math.pow(2, 9); +const shift8 = Math.pow(2, 8); +const shift1 = Math.pow(2, 1); +function packOpacity(opacityState: OpacityState): number { + if (opacityState.opacity === 0 && !opacityState.placed) { + return 0; + } else if (opacityState.opacity === 1 && opacityState.placed) { + return 4294967295; + } + const targetBit = opacityState.placed ? 1 : 0; + const opacityBits = Math.floor(opacityState.opacity * 127); + return opacityBits * shift25 + targetBit * shift24 + + opacityBits * shift17 + targetBit * shift16 + + opacityBits * shift9 + targetBit * shift8 + + opacityBits * shift1 + targetBit; +} + +const PACKED_HIDDEN_OPACITY = 0; diff --git a/web/libraries/maplibre-gl/src/symbol/projection.test.ts b/web/libraries/maplibre-gl/src/symbol/projection.test.ts new file mode 100644 index 00000000..2168601d --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/projection.test.ts @@ -0,0 +1,150 @@ +import {findOffsetIntersectionPoint, project, projectVertexToViewport, transformToOffsetNormal} from './projection'; + +import Point from '@mapbox/point-geometry'; +import {mat4} from 'gl-matrix'; +import {SymbolLineVertexArray} from '../data/array_types.g'; + +describe('Projection', () => { + test('matrix float precision', () => { + const point = new Point(10.000000005, 0); + const matrix = mat4.create(); + expect(project(point, matrix).point.x).toBeCloseTo(point.x, 10); + }); +}); + +describe('Vertex to viewport projection', () => { + // A three point line along the x axis + const lineVertexArray = new SymbolLineVertexArray(); + lineVertexArray.emplaceBack(-10, 0, -10); + lineVertexArray.emplaceBack(0, 0, 0); + lineVertexArray.emplaceBack(10, 0, 10); + + test('projecting with null matrix', () => { + const projectionArgs = { + projectionCache: {projections: {}, offsets: {}}, + lineVertexArray, + labelPlaneMatrix: mat4.create(), + getElevation: (_x, _y) => 0, + // Only relevant in "behind the camera" case, can't happen with null projection matrix + tileAnchorPoint: new Point(0, 0), + distanceFromAnchor: 0, + previousVertex: new Point(0, 0), + direction: 1, + absOffsetX: 0 + }; + + const first = projectVertexToViewport(0, projectionArgs); + const second = projectVertexToViewport(1, projectionArgs); + const third = projectVertexToViewport(2, projectionArgs); + expect(first.x).toBeCloseTo(-10); + expect(second.x).toBeCloseTo(0); + expect(third.x).toBeCloseTo(10); + }); +}); + +describe('Find offset line intersections', () => { + const lineVertexArray = new SymbolLineVertexArray(); + // A three point line along x axis, to origin, and then up y axis + lineVertexArray.emplaceBack(-10, 0, -10); + lineVertexArray.emplaceBack(0, 0, 0); + lineVertexArray.emplaceBack(0, 10, 10); + + // A three point line along the x axis + lineVertexArray.emplaceBack(-10, 0, -10); + lineVertexArray.emplaceBack(0, 0, 0); + lineVertexArray.emplaceBack(10, 0, 10); + + const projectionArgs = { + projectionCache: {projections: {}, offsets: {}}, + lineVertexArray, + labelPlaneMatrix: mat4.create(), + getElevation: (_x, _y) => 0, + direction: 1, + // Only relevant in "behind the camera" case, can't happen with null projection matrix + tileAnchorPoint: new Point(0, 0), + distanceFromAnchor: 0, + previousVertex: new Point(0, 0), + absOffsetX: 0 + }; + + test('concave', () => { + /* + | | + | | + ________| | + __________| <- origin + */ + projectionArgs.projectionCache = {projections: {}, offsets: {}}; + const lineOffsetY = 1; + + const prevToCurrent = new Point(10, 0); + const normal = transformToOffsetNormal(prevToCurrent, lineOffsetY, projectionArgs.direction); + expect(normal.y).toBeCloseTo(1); + expect(normal.x).toBeCloseTo(0); + const intersectionPoint = findOffsetIntersectionPoint( + 1, + normal, + new Point(0, 0), + 0, + 3, + new Point(-10, 1), + lineOffsetY, + projectionArgs + ); + expect(intersectionPoint.y).toBeCloseTo(1); + expect(intersectionPoint.x).toBeCloseTo(-1); + }); + + test('convex', () => { + /* + | | + | | + origin \ | | + __________| | + ____________| + */ + projectionArgs.projectionCache = {projections: {}, offsets: {}}; + const lineOffsetY = -1; + + const prevToCurrent = new Point(10, 0); + const normal = transformToOffsetNormal(prevToCurrent, lineOffsetY, projectionArgs.direction); + expect(normal.y).toBeCloseTo(-1); + expect(normal.x).toBeCloseTo(0); + const intersectionPoint = findOffsetIntersectionPoint( + 1, + normal, + new Point(0, 0), + 0, + 3, + new Point(-10, -1), + lineOffsetY, + projectionArgs + ); + expect(intersectionPoint.y).toBeCloseTo(-1); + expect(intersectionPoint.x).toBeCloseTo(1); + }); + + test('parallel', () => { + /* + ______._____ + ______|_____ + */ + projectionArgs.projectionCache = {projections: {}, offsets: {}}; + const lineOffsetY = 1; + + const prevToCurrent = new Point(10, 0); + const intersectionPoint = findOffsetIntersectionPoint( + 1, + transformToOffsetNormal(prevToCurrent, lineOffsetY, projectionArgs.direction), + new Point(0, 0), + 3, + 5, + new Point(-10, 1), + lineOffsetY, + projectionArgs + ); + expect(intersectionPoint.x).toBeCloseTo(0); + expect(intersectionPoint.y).toBeCloseTo(1); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/symbol/projection.ts b/web/libraries/maplibre-gl/src/symbol/projection.ts new file mode 100644 index 00000000..ac186a7c --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/projection.ts @@ -0,0 +1,679 @@ +import Point from '@mapbox/point-geometry'; + +import {mat4, vec4} from 'gl-matrix'; +import * as symbolSize from './symbol_size'; +import {addDynamicAttributes} from '../data/bucket/symbol_bucket'; + +import type {Painter} from '../render/painter'; +import type {Transform} from '../geo/transform'; +import type {SymbolBucket} from '../data/bucket/symbol_bucket'; +import type { + GlyphOffsetArray, + SymbolLineVertexArray, + SymbolDynamicLayoutArray +} from '../data/array_types.g'; +import {WritingMode} from '../symbol/shaping'; +import {findLineIntersection} from '../util/util'; + +export {updateLineLabels, hideGlyphs, getLabelPlaneMatrix, getGlCoordMatrix, project, getPerspectiveRatio, placeFirstAndLastGlyph, placeGlyphAlongLine, xyTransformMat4, projectVertexToViewport, findOffsetIntersectionPoint, transformToOffsetNormal}; + +/* + * # Overview of coordinate spaces + * + * ## Tile coordinate spaces + * Each label has an anchor. Some labels have corresponding line geometries. + * The points for both anchors and lines are stored in tile units. Each tile has it's own + * coordinate space going from (0, 0) at the top left to (EXTENT, EXTENT) at the bottom right. + * + * ## GL coordinate space + * At the end of everything, the vertex shader needs to produce a position in GL coordinate space, + * which is (-1, 1) at the top left and (1, -1) in the bottom right. + * + * ## Map pixel coordinate spaces + * Each tile has a pixel coordinate space. It's just the tile units scaled so that one unit is + * whatever counts as 1 pixel at the current zoom. + * This space is used for pitch-alignment=map, rotation-alignment=map + * + * ## Rotated map pixel coordinate spaces + * Like the above, but rotated so axis of the space are aligned with the viewport instead of the tile. + * This space is used for pitch-alignment=map, rotation-alignment=viewport + * + * ## Viewport pixel coordinate space + * (0, 0) is at the top left of the canvas and (pixelWidth, pixelHeight) is at the bottom right corner + * of the canvas. This space is used for pitch-alignment=viewport + * + * + * # Vertex projection + * It goes roughly like this: + * 1. project the anchor and line from tile units into the correct label coordinate space + * - map pixel space pitch-alignment=map rotation-alignment=map + * - rotated map pixel space pitch-alignment=map rotation-alignment=viewport + * - viewport pixel space pitch-alignment=viewport rotation-alignment=* + * 2. if the label follows a line, find the point along the line that is the correct distance from the anchor. + * 3. add the glyph's corner offset to the point from step 3 + * 4. convert from the label coordinate space to gl coordinates + * + * For horizontal labels we want to do step 1 in the shader for performance reasons (no cpu work). + * This is what `u_label_plane_matrix` is used for. + * For labels aligned with lines we have to steps 1 and 2 on the cpu since we need access to the line geometry. + * This is what `updateLineLabels(...)` does. + * Since the conversion is handled on the cpu we just set `u_label_plane_matrix` to an identity matrix. + * + * Steps 3 and 4 are done in the shaders for all labels. + */ + +/* + * Returns a matrix for converting from tile units to the correct label coordinate space. + */ +function getLabelPlaneMatrix(posMatrix: mat4, + pitchWithMap: boolean, + rotateWithMap: boolean, + transform: Transform, + pixelsToTileUnits: number) { + const m = mat4.create(); + if (pitchWithMap) { + mat4.scale(m, m, [1 / pixelsToTileUnits, 1 / pixelsToTileUnits, 1]); + if (!rotateWithMap) { + mat4.rotateZ(m, m, transform.angle); + } + } else { + mat4.multiply(m, transform.labelPlaneMatrix, posMatrix); + } + return m; +} + +/* + * Returns a matrix for converting from the correct label coordinate space to gl coords. + */ +function getGlCoordMatrix(posMatrix: mat4, + pitchWithMap: boolean, + rotateWithMap: boolean, + transform: Transform, + pixelsToTileUnits: number) { + if (pitchWithMap) { + const m = mat4.clone(posMatrix); + mat4.scale(m, m, [pixelsToTileUnits, pixelsToTileUnits, 1]); + if (!rotateWithMap) { + mat4.rotateZ(m, m, -transform.angle); + } + return m; + } else { + return transform.glCoordMatrix; + } +} + +function project(point: Point, matrix: mat4, getElevation?: (x: number, y: number) => number) { + let pos; + if (getElevation) { // slow because of handle z-index + pos = [point.x, point.y, getElevation(point.x, point.y), 1] as vec4; + vec4.transformMat4(pos, pos, matrix); + } else { // fast because of ignore z-index + pos = [point.x, point.y, 0, 1] as vec4; + xyTransformMat4(pos, pos, matrix); + } + const w = pos[3]; + return { + point: new Point(pos[0] / w, pos[1] / w), + signedDistanceFromCamera: w + }; +} + +function getPerspectiveRatio(cameraToCenterDistance: number, signedDistanceFromCamera: number): number { + return 0.5 + 0.5 * (cameraToCenterDistance / signedDistanceFromCamera); +} + +function isVisible(anchorPos: vec4, + clippingBuffer: [number, number]) { + const x = anchorPos[0] / anchorPos[3]; + const y = anchorPos[1] / anchorPos[3]; + const inPaddedViewport = ( + x >= -clippingBuffer[0] && + x <= clippingBuffer[0] && + y >= -clippingBuffer[1] && + y <= clippingBuffer[1]); + return inPaddedViewport; +} + +/* + * Update the `dynamicLayoutVertexBuffer` for the buffer with the correct glyph positions for the current map view. + * This is only run on labels that are aligned with lines. Horizontal labels are handled entirely in the shader. + */ +function updateLineLabels(bucket: SymbolBucket, + posMatrix: mat4, + painter: Painter, + isText: boolean, + labelPlaneMatrix: mat4, + glCoordMatrix: mat4, + pitchWithMap: boolean, + keepUpright: boolean, + rotateToLine: boolean, + getElevation: (x: number, y: number) => number) { + + const sizeData = isText ? bucket.textSizeData : bucket.iconSizeData; + const partiallyEvaluatedSize = symbolSize.evaluateSizeForZoom(sizeData, painter.transform.zoom); + + const clippingBuffer: [number, number] = [256 / painter.width * 2 + 1, 256 / painter.height * 2 + 1]; + + const dynamicLayoutVertexArray = isText ? + bucket.text.dynamicLayoutVertexArray : + bucket.icon.dynamicLayoutVertexArray; + dynamicLayoutVertexArray.clear(); + + const lineVertexArray = bucket.lineVertexArray; + const placedSymbols = isText ? bucket.text.placedSymbolArray : bucket.icon.placedSymbolArray; + + const aspectRatio = painter.transform.width / painter.transform.height; + + let useVertical = false; + + for (let s = 0; s < placedSymbols.length; s++) { + const symbol = placedSymbols.get(s); + + // Don't do calculations for vertical glyphs unless the previous symbol was horizontal + // and we determined that vertical glyphs were necessary. + // Also don't do calculations for symbols that are collided and fully faded out + if (symbol.hidden || symbol.writingMode === WritingMode.vertical && !useVertical) { + hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray); + continue; + } + // Awkward... but we're counting on the paired "vertical" symbol coming immediately after its horizontal counterpart + useVertical = false; + + let anchorPos; + if (getElevation) { // slow because of handle z-index + anchorPos = [symbol.anchorX, symbol.anchorY, getElevation(symbol.anchorX, symbol.anchorY), 1] as vec4; + vec4.transformMat4(anchorPos, anchorPos, posMatrix); + } else { // fast because of ignore z-index + anchorPos = [symbol.anchorX, symbol.anchorY, 0, 1] as vec4; + xyTransformMat4(anchorPos, anchorPos, posMatrix); + } + + // Don't bother calculating the correct point for invisible labels. + if (!isVisible(anchorPos, clippingBuffer)) { + hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray); + continue; + } + + const cameraToAnchorDistance = anchorPos[3]; + const perspectiveRatio = getPerspectiveRatio(painter.transform.cameraToCenterDistance, cameraToAnchorDistance); + + const fontSize = symbolSize.evaluateSizeForFeature(sizeData, partiallyEvaluatedSize, symbol); + const pitchScaledFontSize = pitchWithMap ? fontSize / perspectiveRatio : fontSize * perspectiveRatio; + + const tileAnchorPoint = new Point(symbol.anchorX, symbol.anchorY); + const anchorPoint = project(tileAnchorPoint, labelPlaneMatrix, getElevation).point; + const projectionCache = {projections: {}, offsets: {}}; + + const placeUnflipped: any = placeGlyphsAlongLine(symbol, pitchScaledFontSize, false /*unflipped*/, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix, + bucket.glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation); + + useVertical = placeUnflipped.useVertical; + + if (placeUnflipped.notEnoughRoom || useVertical || + (placeUnflipped.needsFlipping && + (placeGlyphsAlongLine(symbol, pitchScaledFontSize, true /*flipped*/, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix, + bucket.glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation) as any).notEnoughRoom)) { + hideGlyphs(symbol.numGlyphs, dynamicLayoutVertexArray); + } + } + + if (isText) { + bucket.text.dynamicLayoutVertexBuffer.updateData(dynamicLayoutVertexArray); + } else { + bucket.icon.dynamicLayoutVertexBuffer.updateData(dynamicLayoutVertexArray); + } +} + +type FirstAndLastGlyphPlacement = { + first: PlacedGlyph; + last: PlacedGlyph; +} | null; + +/* + * Place the first and last glyph of a line label, projected to the label plane. + * This function is called both during collision detection (to determine the label's size) + * and during line label rendering (to make sure the label fits on the line geometry with + * the current camera position, which may differ from the position used during collision detection). + * + * Calling this function has the effect of populating the "projectionCache" with all projected + * vertex locations the label will need, making future calls to placeGlyphAlongLine (for all the + * intermediate glyphs) much cheaper. + * + * Returns null if the label can't fit on the geometry + */ +function placeFirstAndLastGlyph(fontScale: number, glyphOffsetArray: GlyphOffsetArray, lineOffsetX: number, lineOffsetY: number, flip: boolean, anchorPoint: Point, tileAnchorPoint: Point, symbol: any, lineVertexArray: SymbolLineVertexArray, labelPlaneMatrix: mat4, projectionCache: ProjectionCache, rotateToLine: boolean, getElevation: (x: number, y: number) => number): FirstAndLastGlyphPlacement { + const glyphEndIndex = symbol.glyphStartIndex + symbol.numGlyphs; + const lineStartIndex = symbol.lineStartIndex; + const lineEndIndex = symbol.lineStartIndex + symbol.lineLength; + + const firstGlyphOffset = glyphOffsetArray.getoffsetX(symbol.glyphStartIndex); + const lastGlyphOffset = glyphOffsetArray.getoffsetX(glyphEndIndex - 1); + + const firstPlacedGlyph = placeGlyphAlongLine(fontScale * firstGlyphOffset, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment, + lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation); + if (!firstPlacedGlyph) + return null; + + const lastPlacedGlyph = placeGlyphAlongLine(fontScale * lastGlyphOffset, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment, + lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation); + if (!lastPlacedGlyph) + return null; + + return {first: firstPlacedGlyph, last: lastPlacedGlyph}; +} + +function requiresOrientationChange(writingMode, firstPoint, lastPoint, aspectRatio) { + if (writingMode === WritingMode.horizontal) { + // On top of choosing whether to flip, choose whether to render this version of the glyphs or the alternate + // vertical glyphs. We can't just filter out vertical glyphs in the horizontal range because the horizontal + // and vertical versions can have slightly different projections which could lead to angles where both or + // neither showed. + const rise = Math.abs(lastPoint.y - firstPoint.y); + const run = Math.abs(lastPoint.x - firstPoint.x) * aspectRatio; + if (rise > run) { + return {useVertical: true}; + } + } + + if (writingMode === WritingMode.vertical ? firstPoint.y < lastPoint.y : firstPoint.x > lastPoint.x) { + // Includes "horizontalOnly" case for labels without vertical glyphs + return {needsFlipping: true}; + } + + return null; +} + +/* +* Place first and last glyph along the line projected to label plane, and if they fit +* iterate through all the intermediate glyphs, calculating their label plane positions +* from the projected line. +* +* Finally, add resulting glyph position calculations to dynamicLayoutVertexArray for +* upload to the GPU +*/ +function placeGlyphsAlongLine(symbol, fontSize, flip, keepUpright, posMatrix, labelPlaneMatrix, glCoordMatrix, glyphOffsetArray, lineVertexArray, dynamicLayoutVertexArray, anchorPoint, tileAnchorPoint, projectionCache, aspectRatio, rotateToLine, getElevation) { + const fontScale = fontSize / 24; + const lineOffsetX = symbol.lineOffsetX * fontScale; + const lineOffsetY = symbol.lineOffsetY * fontScale; + + let placedGlyphs; + if (symbol.numGlyphs > 1) { + const glyphEndIndex = symbol.glyphStartIndex + symbol.numGlyphs; + const lineStartIndex = symbol.lineStartIndex; + const lineEndIndex = symbol.lineStartIndex + symbol.lineLength; + + // Place the first and the last glyph in the label first, so we can figure out + // the overall orientation of the label and determine whether it needs to be flipped in keepUpright mode + const firstAndLastGlyph = placeFirstAndLastGlyph(fontScale, glyphOffsetArray, lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation); + if (!firstAndLastGlyph) { + return {notEnoughRoom: true}; + } + const firstPoint = project(firstAndLastGlyph.first.point, glCoordMatrix, getElevation).point; + const lastPoint = project(firstAndLastGlyph.last.point, glCoordMatrix, getElevation).point; + + if (keepUpright && !flip) { + const orientationChange = requiresOrientationChange(symbol.writingMode, firstPoint, lastPoint, aspectRatio); + if (orientationChange) { + return orientationChange; + } + } + + placedGlyphs = [firstAndLastGlyph.first]; + for (let glyphIndex = symbol.glyphStartIndex + 1; glyphIndex < glyphEndIndex - 1; glyphIndex++) { + // Since first and last glyph fit on the line, we're sure that the rest of the glyphs can be placed + placedGlyphs.push(placeGlyphAlongLine(fontScale * glyphOffsetArray.getoffsetX(glyphIndex), lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment, + lineStartIndex, lineEndIndex, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation)); + } + placedGlyphs.push(firstAndLastGlyph.last); + } else { + // Only a single glyph to place + // So, determine whether to flip based on projected angle of the line segment it's on + if (keepUpright && !flip) { + const a = project(tileAnchorPoint, posMatrix, getElevation).point; + const tileVertexIndex = (symbol.lineStartIndex + symbol.segment + 1); + const tileSegmentEnd = new Point(lineVertexArray.getx(tileVertexIndex), lineVertexArray.gety(tileVertexIndex)); + const projectedVertex = project(tileSegmentEnd, posMatrix, getElevation); + // We know the anchor will be in the viewport, but the end of the line segment may be + // behind the plane of the camera, in which case we can use a point at any arbitrary (closer) + // point on the segment. + const b = (projectedVertex.signedDistanceFromCamera > 0) ? + projectedVertex.point : + projectTruncatedLineSegment(tileAnchorPoint, tileSegmentEnd, a, 1, posMatrix, getElevation); + + const orientationChange = requiresOrientationChange(symbol.writingMode, a, b, aspectRatio); + if (orientationChange) { + return orientationChange; + } + } + const singleGlyph = placeGlyphAlongLine(fontScale * glyphOffsetArray.getoffsetX(symbol.glyphStartIndex), lineOffsetX, lineOffsetY, flip, anchorPoint, tileAnchorPoint, symbol.segment, + symbol.lineStartIndex, symbol.lineStartIndex + symbol.lineLength, lineVertexArray, labelPlaneMatrix, projectionCache, rotateToLine, getElevation); + if (!singleGlyph) + return {notEnoughRoom: true}; + + placedGlyphs = [singleGlyph]; + } + + for (const glyph of placedGlyphs) { + addDynamicAttributes(dynamicLayoutVertexArray, glyph.point, glyph.angle); + } + return {}; +} + +function projectTruncatedLineSegment(previousTilePoint: Point, currentTilePoint: Point, previousProjectedPoint: Point, minimumLength: number, projectionMatrix: mat4, getElevation: (x: number, y: number) => number) { + // We are assuming "previousTilePoint" won't project to a point within one unit of the camera plane + // If it did, that would mean our label extended all the way out from within the viewport to a (very distant) + // point near the plane of the camera. We wouldn't be able to render the label anyway once it crossed the + // plane of the camera. + const projectedUnitVertex = project(previousTilePoint.add(previousTilePoint.sub(currentTilePoint)._unit()), projectionMatrix, getElevation).point; + const projectedUnitSegment = previousProjectedPoint.sub(projectedUnitVertex); + + return previousProjectedPoint.add(projectedUnitSegment._mult(minimumLength / projectedUnitSegment.mag())); +} + +type IndexToPointCache = { [lineIndex: number]: Point }; + +/** + * We calculate label-plane projected points for line vertices as we place glyphs along the line + * Since we will use the same vertices for potentially many glyphs, cache the results for this bucket + * over the course of the render. Each vertex location also potentially has one offset equivalent + * for us to hold onto. The vertex indices are per-symbol-bucket. + */ +type ProjectionCache = { + /** + * tile-unit vertices projected into label-plane units + */ + projections: IndexToPointCache; + /** + * label-plane vertices which have been shifted to follow an offset line + */ + offsets: IndexToPointCache; +}; + +/** + * Arguments necessary to project a vertex to the label plane + */ +type ProjectionArgs = { + /** + * Used to cache results, save cost if projecting the same vertex multiple times + */ + projectionCache: ProjectionCache; + /** + * The array of tile-unit vertices transferred from worker + */ + lineVertexArray: SymbolLineVertexArray; + /** + * Label plane projection matrix + */ + labelPlaneMatrix: mat4; + /** + * Function to get elevation at a point + * @param x - the x coordinate + * @param y - the y coordinate + */ + getElevation: (x: number, y: number) => number; + /** + * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera + */ + tileAnchorPoint: Point; + /** + * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera + */ + distanceFromAnchor: number; + /** + * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera + */ + previousVertex: Point; + /** + * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera + */ + direction: number; + /** + * Only for creating synthetic vertices if vertex would otherwise project behind plane of camera + */ + absOffsetX: number; +}; + +/** + * Transform a vertex from tile coordinates to label plane coordinates + * @param index - index of vertex to project + * @param projectionArgs - necessary data to project a vertex + * @returns the vertex projected to the label plane + */ +function projectVertexToViewport(index: number, projectionArgs: ProjectionArgs): Point { + const {projectionCache, lineVertexArray, labelPlaneMatrix, tileAnchorPoint, distanceFromAnchor, getElevation, previousVertex, direction, absOffsetX} = projectionArgs; + if (projectionCache.projections[index]) { + return projectionCache.projections[index]; + } + const currentVertex = new Point(lineVertexArray.getx(index), lineVertexArray.gety(index)); + const projection = project(currentVertex, labelPlaneMatrix, getElevation); + if (projection.signedDistanceFromCamera > 0) { + projectionCache.projections[index] = projection.point; + return projection.point; + } + + // The vertex is behind the plane of the camera, so we can't project it + // Instead, we'll create a vertex along the line that's far enough to include the glyph + const previousLineVertexIndex = index - direction; + const previousTilePoint = distanceFromAnchor === 0 ? + tileAnchorPoint : + new Point(lineVertexArray.getx(previousLineVertexIndex), lineVertexArray.gety(previousLineVertexIndex)); + // Don't cache because the new vertex might not be far enough out for future glyphs on the same segment + return projectTruncatedLineSegment(previousTilePoint, currentVertex, previousVertex, absOffsetX - distanceFromAnchor + 1, labelPlaneMatrix, getElevation); +} + +/** + * Calculate the normal vector for a line segment + * @param segmentVector - will be mutated as a tiny optimization + * @param offset - magnitude of resulting vector + * @param direction - direction of line traversal + * @returns a normal vector from the segment, with magnitude equal to offset amount + */ +function transformToOffsetNormal(segmentVector: Point, offset: number, direction: number): Point { + return segmentVector._unit()._perp()._mult(offset * direction); +} + +/** + * Construct offset line segments for the current segment and the next segment, then extend/shrink + * the segments until they intersect. If the segments are parallel, then they will touch with no modification. + * + * @param index - Index of the current vertex + * @param prevToCurrentOffsetNormal - Normal vector of the line segment from the previous vertex to the current vertex + * @param currentVertex - Current (non-offset) vertex projected to the label plane + * @param lineStartIndex - Beginning index for the line this label is on + * @param lineEndIndex - End index for the line this label is on + * @param offsetPreviousVertex - The previous vertex projected to the label plane, and then offset along the previous segments normal + * @param lineOffsetY - Magnitude of the offset + * @param projectionArgs - Necessary data for tile-to-label-plane projection + * @returns The point at which the current and next line segments intersect, once offset and extended/shrunk to their meeting point + */ +function findOffsetIntersectionPoint(index: number, prevToCurrentOffsetNormal: Point, currentVertex: Point, lineStartIndex: number, lineEndIndex: number, offsetPreviousVertex: Point, lineOffsetY: number, projectionArgs: ProjectionArgs) { + const {projectionCache, direction} = projectionArgs; + if (projectionCache.offsets[index]) { + return projectionCache.offsets[index]; + } + + const offsetCurrentVertex = currentVertex.add(prevToCurrentOffsetNormal); + + if (index + direction < lineStartIndex || index + direction >= lineEndIndex) { + // This is the end of the line, no intersection to calculate + projectionCache.offsets[index] = offsetCurrentVertex; + return offsetCurrentVertex; + } + // Offset the vertices for the next segment + const nextVertex = projectVertexToViewport(index + direction, projectionArgs); + const currentToNextOffsetNormal = transformToOffsetNormal(nextVertex.sub(currentVertex), lineOffsetY, direction); + const offsetNextSegmentBegin = currentVertex.add(currentToNextOffsetNormal); + const offsetNextSegmentEnd = nextVertex.add(currentToNextOffsetNormal); + + // find the intersection of these two lines + // if the lines are parallel, offsetCurrent/offsetNextBegin will touch + projectionCache.offsets[index] = findLineIntersection(offsetPreviousVertex, offsetCurrentVertex, offsetNextSegmentBegin, offsetNextSegmentEnd) || offsetCurrentVertex; + + return projectionCache.offsets[index]; +} + +/** + * Placed Glyph type + */ +type PlacedGlyph = { + /** + * The point at which the glyph should be placed, in label plane coordinates + */ + point: Point; + /** + * The angle at which the glyph should be placed + */ + angle: number; + /** + * The label-plane path used to reach this glyph: used only for collision detection + */ + path: Array; +}; + +/* + * Place a single glyph along its line, projected into the label plane, by iterating outward + * from the anchor point until the distance traversed in the label plane equals the glyph's + * offsetX. Returns null if the glyph can't fit on the line geometry. + */ +function placeGlyphAlongLine( + offsetX: number, + lineOffsetX: number, + lineOffsetY: number, + flip: boolean, + anchorPoint: Point, + tileAnchorPoint: Point, + anchorSegment: number, + lineStartIndex: number, + lineEndIndex: number, + lineVertexArray: SymbolLineVertexArray, + labelPlaneMatrix: mat4, + projectionCache: ProjectionCache, + rotateToLine: boolean, + getElevation: (x: number, y: number) => number): PlacedGlyph | null { + + const combinedOffsetX = flip ? + offsetX - lineOffsetX : + offsetX + lineOffsetX; + + let direction = combinedOffsetX > 0 ? 1 : -1; + + let angle = 0; + if (flip) { + // The label needs to be flipped to keep text upright. + // Iterate in the reverse direction. + direction *= -1; + angle = Math.PI; + } + + if (direction < 0) angle += Math.PI; + + let currentIndex = direction > 0 ? + lineStartIndex + anchorSegment : + lineStartIndex + anchorSegment + 1; + + let currentVertex = anchorPoint; + let previousVertex = anchorPoint; + + // offsetPrev and intersectionPoint are analogous to previousVertex and currentVertex + // but if there's a line offset they are calculated in parallel as projection happens + let offsetIntersectionPoint: Point; + let offsetPreviousVertex: Point; + + let distanceFromAnchor = 0; + let currentSegmentDistance = 0; + const absOffsetX = Math.abs(combinedOffsetX); + const pathVertices: Array = []; + + let currentLineSegment: Point; + while (distanceFromAnchor + currentSegmentDistance <= absOffsetX) { + currentIndex += direction; + + // offset does not fit on the projected line + if (currentIndex < lineStartIndex || currentIndex >= lineEndIndex) + return null; + + // accumulate values from last iteration + distanceFromAnchor += currentSegmentDistance; + previousVertex = currentVertex; + offsetPreviousVertex = offsetIntersectionPoint; + + const projectionArgs: ProjectionArgs = { + projectionCache, + lineVertexArray, + labelPlaneMatrix, + tileAnchorPoint, + distanceFromAnchor, + getElevation, + previousVertex, + direction, + absOffsetX + }; + + // find next vertex in viewport space + currentVertex = projectVertexToViewport(currentIndex, projectionArgs); + if (lineOffsetY === 0) { + // Store vertices for collision detection and update current segment geometry + pathVertices.push(previousVertex); + currentLineSegment = currentVertex.sub(previousVertex); + } else { + // Calculate the offset for this section + let prevToCurrentOffsetNormal; + const prevToCurrent = currentVertex.sub(previousVertex); + if (prevToCurrent.mag() === 0) { + // We are starting with our anchor point directly on the vertex, so look one vertex ahead + // to calculate a normal + const nextVertex = projectVertexToViewport(currentIndex + direction, projectionArgs); + prevToCurrentOffsetNormal = transformToOffsetNormal(nextVertex.sub(currentVertex), lineOffsetY, direction); + } else { + prevToCurrentOffsetNormal = transformToOffsetNormal(prevToCurrent, lineOffsetY, direction); + } + // Initialize offsetPrev on our first iteration, after that it will be pre-calculated + if (!offsetPreviousVertex) + offsetPreviousVertex = previousVertex.add(prevToCurrentOffsetNormal); + + offsetIntersectionPoint = findOffsetIntersectionPoint(currentIndex, prevToCurrentOffsetNormal, currentVertex, lineStartIndex, lineEndIndex, offsetPreviousVertex, lineOffsetY, projectionArgs); + + pathVertices.push(offsetPreviousVertex); + currentLineSegment = offsetIntersectionPoint.sub(offsetPreviousVertex); + } + currentSegmentDistance = currentLineSegment.mag(); + } + + // The point is on the current segment. Interpolate to find it. + const segmentInterpolationT = (absOffsetX - distanceFromAnchor) / currentSegmentDistance; + const p = currentLineSegment._mult(segmentInterpolationT)._add(offsetPreviousVertex || previousVertex); + + const segmentAngle = angle + Math.atan2(currentVertex.y - previousVertex.y, currentVertex.x - previousVertex.x); + + pathVertices.push(p); + + return { + point: p, + angle: rotateToLine ? segmentAngle : 0.0, + path: pathVertices + }; +} + +const hiddenGlyphAttributes = new Float32Array([-Infinity, -Infinity, 0, -Infinity, -Infinity, 0, -Infinity, -Infinity, 0, -Infinity, -Infinity, 0]); + +// Hide them by moving them offscreen. We still need to add them to the buffer +// because the dynamic buffer is paired with a static buffer that doesn't get updated. +function hideGlyphs(num: number, dynamicLayoutVertexArray: SymbolDynamicLayoutArray) { + for (let i = 0; i < num; i++) { + const offset = dynamicLayoutVertexArray.length; + dynamicLayoutVertexArray.resize(offset + 4); + // Since all hidden glyphs have the same attributes, we can build up the array faster with a single call to Float32Array.set + // for each set of four vertices, instead of calling addDynamicAttributes for each vertex. + dynamicLayoutVertexArray.float32.set(hiddenGlyphAttributes, offset * 3); + } +} + +// For line label layout, we're not using z output and our w input is always 1 +// This custom matrix transformation ignores those components to make projection faster +function xyTransformMat4(out: vec4, a: vec4, m: mat4) { + const x = a[0], y = a[1]; + out[0] = m[0] * x + m[4] * y + m[12]; + out[1] = m[1] * x + m[5] * y + m[13]; + out[3] = m[3] * x + m[7] * y + m[15]; + return out; +} diff --git a/web/libraries/maplibre-gl/src/symbol/quads.test.ts b/web/libraries/maplibre-gl/src/symbol/quads.test.ts new file mode 100644 index 00000000..f592bc9b --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/quads.test.ts @@ -0,0 +1,157 @@ +import {ImagePosition} from '../render/image_atlas'; +import {getIconQuads} from './quads'; + +describe('getIconQuads', () => { + const image = Object.freeze({ + pixelRatio: 1, + displaySize: Object.freeze([15, 11]), + paddedRect: Object.freeze({x: 0, y: 0, w: 17, h: 13}) + }) as ImagePosition; + + test('point', () => { + expect(getIconQuads({ + top: -5.5, + right: 7.5, + bottom: 5.5, + left: -7.5, + image + }, 0, true, false)).toEqual([{ + tl: {x: -8.5, y: -6.5}, + tr: {x: 8.5, y: -6.5}, + bl: {x: -8.5, y: 6.5}, + br: {x: 8.5, y: 6.5}, + tex: {x: 0, y: 0, w: 17, h: 13}, + writingMode: undefined, + glyphOffset: [0, 0], + isSDF: true, + sectionIndex: 0, + minFontScaleX: 0, + minFontScaleY: 0, + pixelOffsetBR: { + x: 0, + y: 0 + }, + pixelOffsetTL: { + x: 0, + y: 0 + } + }]); + + expect(getIconQuads({ + top: -11, + right: 15, + bottom: 11, + left: -15, + image + }, 0, false, false)).toEqual([{ + tl: {x: -17, y: -13}, + tr: {x: 17, y: -13}, + bl: {x: -17, y: 13}, + br: {x: 17, y: 13}, + tex: {x: 0, y: 0, w: 17, h: 13}, + writingMode: undefined, + glyphOffset: [0, 0], + isSDF: false, + sectionIndex: 0, + minFontScaleX: 0, + minFontScaleY: 0, + pixelOffsetBR: { + x: 0, + y: 0 + }, + pixelOffsetTL: { + x: 0, + y: 0 + } + }]); + + expect(getIconQuads({ + top: 0, + right: 0, + bottom: 11, + left: -15, + image + }, 0, false, false)).toEqual([{ + tl: {x: -16, y: -1}, + tr: {x: 1, y: -1}, + bl: {x: -16, y: 12}, + br: {x: 1, y: 12}, + tex: {x: 0, y: 0, w: 17, h: 13}, + writingMode: undefined, + glyphOffset: [0, 0], + isSDF: false, + sectionIndex: 0, + minFontScaleX: 0, + minFontScaleY: 0, + pixelOffsetBR: { + x: 0, + y: 0 + }, + pixelOffsetTL: { + x: 0, + y: 0 + } + }]); + + expect(getIconQuads({ + top: -5.5, + right: 30, + bottom: 5.5, + left: -30, + image + }, 0, false, false)).toEqual([{ + tl: {x: -34, y: -6.5}, + tr: {x: 34, y: -6.5}, + bl: {x: -34, y: 6.5}, + br: {x: 34, y: 6.5}, + tex: {x: 0, y: 0, w: 17, h: 13}, + writingMode: undefined, + glyphOffset: [0, 0], + isSDF: false, + sectionIndex: 0, + minFontScaleX: 0, + minFontScaleY: 0, + pixelOffsetBR: { + x: 0, + y: 0 + }, + pixelOffsetTL: { + x: 0, + y: 0 + } + }]); + + }); + + test('line', () => { + expect(getIconQuads({ + top: -5.5, + right: 7.5, + bottom: 5.5, + left: -7.5, + image + }, 0, false, false)).toEqual([{ + tl: {x: -8.5, y: -6.5}, + tr: {x: 8.5, y: -6.5}, + bl: {x: -8.5, y: 6.5}, + br: {x: 8.5, y: 6.5}, + tex: {x: 0, y: 0, w: 17, h: 13}, + writingMode: undefined, + glyphOffset: [0, 0], + isSDF: false, + sectionIndex: 0, + minFontScaleX: 0, + minFontScaleY: 0, + pixelOffsetBR: { + x: 0, + y: 0 + }, + pixelOffsetTL: { + x: 0, + y: 0 + } + }]); + + }); + +}); diff --git a/web/libraries/maplibre-gl/src/symbol/quads.ts b/web/libraries/maplibre-gl/src/symbol/quads.ts new file mode 100644 index 00000000..4b9f3f68 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/quads.ts @@ -0,0 +1,334 @@ +import Point from '@mapbox/point-geometry'; + +import {GLYPH_PBF_BORDER} from '../style/parse_glyph_pbf'; + +import type {Anchor} from './anchor'; +import type {PositionedIcon, Shaping} from './shaping'; +import {SHAPING_DEFAULT_OFFSET} from './shaping'; +import {IMAGE_PADDING} from '../render/image_atlas'; +import type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer'; +import type {Feature} from '@maplibre/maplibre-gl-style-spec'; +import type {StyleImage} from '../style/style_image'; +import ONE_EM from './one_em'; +import {Rect} from '../render/glyph_atlas'; + +/** + * A textured quad for rendering a single icon or glyph. + * + * The zoom range the glyph can be shown is defined by minScale and maxScale. + * + * @param tl - The offset of the top left corner from the anchor. + * @param tr - The offset of the top right corner from the anchor. + * @param bl - The offset of the bottom left corner from the anchor. + * @param br - The offset of the bottom right corner from the anchor. + * @param tex - The texture coordinates. + */ +export type SymbolQuad = { + tl: Point; + tr: Point; + bl: Point; + br: Point; + tex: { + x: number; + y: number; + w: number; + h: number; + }; + pixelOffsetTL: Point; + pixelOffsetBR: Point; + writingMode: any | void; + glyphOffset: [number, number]; + sectionIndex: number; + isSDF: boolean; + minFontScaleX: number; + minFontScaleY: number; +}; + +// If you have a 10px icon that isn't perfectly aligned to the pixel grid it will cover 11 actual +// pixels. The quad needs to be padded to account for this, otherwise they'll look slightly clipped +// on one edge in some cases. +const border = IMAGE_PADDING; + +/** + * Create the quads used for rendering an icon. + */ +export function getIconQuads( + shapedIcon: PositionedIcon, + iconRotate: number, + isSDFIcon: boolean, + hasIconTextFit: boolean +): Array { + const quads = []; + + const image = shapedIcon.image; + const pixelRatio = image.pixelRatio; + const imageWidth = image.paddedRect.w - 2 * border; + const imageHeight = image.paddedRect.h - 2 * border; + + const iconWidth = shapedIcon.right - shapedIcon.left; + const iconHeight = shapedIcon.bottom - shapedIcon.top; + + const stretchX = image.stretchX || [[0, imageWidth]]; + const stretchY = image.stretchY || [[0, imageHeight]]; + + const reduceRanges = (sum, range) => sum + range[1] - range[0]; + const stretchWidth = stretchX.reduce(reduceRanges, 0); + const stretchHeight = stretchY.reduce(reduceRanges, 0); + const fixedWidth = imageWidth - stretchWidth; + const fixedHeight = imageHeight - stretchHeight; + + let stretchOffsetX = 0; + let stretchContentWidth = stretchWidth; + let stretchOffsetY = 0; + let stretchContentHeight = stretchHeight; + let fixedOffsetX = 0; + let fixedContentWidth = fixedWidth; + let fixedOffsetY = 0; + let fixedContentHeight = fixedHeight; + + if (image.content && hasIconTextFit) { + const content = image.content; + stretchOffsetX = sumWithinRange(stretchX, 0, content[0]); + stretchOffsetY = sumWithinRange(stretchY, 0, content[1]); + stretchContentWidth = sumWithinRange(stretchX, content[0], content[2]); + stretchContentHeight = sumWithinRange(stretchY, content[1], content[3]); + fixedOffsetX = content[0] - stretchOffsetX; + fixedOffsetY = content[1] - stretchOffsetY; + fixedContentWidth = content[2] - content[0] - stretchContentWidth; + fixedContentHeight = content[3] - content[1] - stretchContentHeight; + } + + const makeBox = (left, top, right, bottom) => { + + const leftEm = getEmOffset(left.stretch - stretchOffsetX, stretchContentWidth, iconWidth, shapedIcon.left); + const leftPx = getPxOffset(left.fixed - fixedOffsetX, fixedContentWidth, left.stretch, stretchWidth); + + const topEm = getEmOffset(top.stretch - stretchOffsetY, stretchContentHeight, iconHeight, shapedIcon.top); + const topPx = getPxOffset(top.fixed - fixedOffsetY, fixedContentHeight, top.stretch, stretchHeight); + + const rightEm = getEmOffset(right.stretch - stretchOffsetX, stretchContentWidth, iconWidth, shapedIcon.left); + const rightPx = getPxOffset(right.fixed - fixedOffsetX, fixedContentWidth, right.stretch, stretchWidth); + + const bottomEm = getEmOffset(bottom.stretch - stretchOffsetY, stretchContentHeight, iconHeight, shapedIcon.top); + const bottomPx = getPxOffset(bottom.fixed - fixedOffsetY, fixedContentHeight, bottom.stretch, stretchHeight); + + const tl = new Point(leftEm, topEm); + const tr = new Point(rightEm, topEm); + const br = new Point(rightEm, bottomEm); + const bl = new Point(leftEm, bottomEm); + const pixelOffsetTL = new Point(leftPx / pixelRatio, topPx / pixelRatio); + const pixelOffsetBR = new Point(rightPx / pixelRatio, bottomPx / pixelRatio); + + const angle = iconRotate * Math.PI / 180; + + if (angle) { + const sin = Math.sin(angle), + cos = Math.cos(angle), + matrix = [cos, -sin, sin, cos]; + + tl._matMult(matrix); + tr._matMult(matrix); + bl._matMult(matrix); + br._matMult(matrix); + } + + const x1 = left.stretch + left.fixed; + const x2 = right.stretch + right.fixed; + const y1 = top.stretch + top.fixed; + const y2 = bottom.stretch + bottom.fixed; + + const subRect = { + x: image.paddedRect.x + border + x1, + y: image.paddedRect.y + border + y1, + w: x2 - x1, + h: y2 - y1 + }; + + const minFontScaleX = fixedContentWidth / pixelRatio / iconWidth; + const minFontScaleY = fixedContentHeight / pixelRatio / iconHeight; + + // Icon quad is padded, so texture coordinates also need to be padded. + return {tl, tr, bl, br, tex: subRect, writingMode: undefined, glyphOffset: [0, 0], sectionIndex: 0, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY, isSDF: isSDFIcon}; + }; + + if (!hasIconTextFit || (!image.stretchX && !image.stretchY)) { + quads.push(makeBox( + {fixed: 0, stretch: -1}, + {fixed: 0, stretch: -1}, + {fixed: 0, stretch: imageWidth + 1}, + {fixed: 0, stretch: imageHeight + 1})); + } else { + const xCuts = stretchZonesToCuts(stretchX, fixedWidth, stretchWidth); + const yCuts = stretchZonesToCuts(stretchY, fixedHeight, stretchHeight); + + for (let xi = 0; xi < xCuts.length - 1; xi++) { + const x1 = xCuts[xi]; + const x2 = xCuts[xi + 1]; + for (let yi = 0; yi < yCuts.length - 1; yi++) { + const y1 = yCuts[yi]; + const y2 = yCuts[yi + 1]; + quads.push(makeBox(x1, y1, x2, y2)); + } + } + } + + return quads; +} + +function sumWithinRange(ranges, min, max) { + let sum = 0; + for (const range of ranges) { + sum += Math.max(min, Math.min(max, range[1])) - Math.max(min, Math.min(max, range[0])); + } + return sum; +} + +function stretchZonesToCuts(stretchZones, fixedSize, stretchSize) { + const cuts = [{fixed: -border, stretch: 0}]; + + for (const [c1, c2] of stretchZones) { + const last = cuts[cuts.length - 1]; + cuts.push({ + fixed: c1 - last.stretch, + stretch: last.stretch + }); + cuts.push({ + fixed: c1 - last.stretch, + stretch: last.stretch + (c2 - c1) + }); + } + cuts.push({ + fixed: fixedSize + border, + stretch: stretchSize + }); + return cuts; +} + +function getEmOffset(stretchOffset, stretchSize, iconSize, iconOffset) { + return stretchOffset / stretchSize * iconSize + iconOffset; +} + +function getPxOffset(fixedOffset, fixedSize, stretchOffset, stretchSize) { + return fixedOffset - fixedSize * stretchOffset / stretchSize; +} + +/** + * Create the quads used for rendering a text label. + */ +export function getGlyphQuads( + anchor: Anchor, + shaping: Shaping, + textOffset: [number, number], + layer: SymbolStyleLayer, + alongLine: boolean, + feature: Feature, + imageMap: {[_: string]: StyleImage}, + allowVerticalPlacement: boolean +): Array { + + const textRotate = layer.layout.get('text-rotate').evaluate(feature, {}) * Math.PI / 180; + const quads = []; + + for (const line of shaping.positionedLines) { + for (const positionedGlyph of line.positionedGlyphs) { + if (!positionedGlyph.rect) continue; + const textureRect: Rect = positionedGlyph.rect || {} as Rect; + + // The rects have an additional buffer that is not included in their size. + const glyphPadding = 1.0; + let rectBuffer = GLYPH_PBF_BORDER + glyphPadding; + let isSDF = true; + let pixelRatio = 1.0; + let lineOffset = 0.0; + + const rotateVerticalGlyph = (alongLine || allowVerticalPlacement) && positionedGlyph.vertical; + const halfAdvance = positionedGlyph.metrics.advance * positionedGlyph.scale / 2; + + // Align images and scaled glyphs in the middle of a vertical line. + if (allowVerticalPlacement && shaping.verticalizable) { + const scaledGlyphOffset = (positionedGlyph.scale - 1) * ONE_EM; + const imageOffset = (ONE_EM - positionedGlyph.metrics.width * positionedGlyph.scale) / 2; + lineOffset = line.lineOffset / 2 - (positionedGlyph.imageName ? -imageOffset : scaledGlyphOffset); + } + + if (positionedGlyph.imageName) { + const image = imageMap[positionedGlyph.imageName]; + isSDF = image.sdf; + pixelRatio = image.pixelRatio; + rectBuffer = IMAGE_PADDING / pixelRatio; + } + + const glyphOffset = alongLine ? + [positionedGlyph.x + halfAdvance, positionedGlyph.y] : + [0, 0]; + + let builtInOffset: [number, number] = alongLine ? + [0, 0] : + [positionedGlyph.x + halfAdvance + textOffset[0], positionedGlyph.y + textOffset[1] - lineOffset]; + + let verticalizedLabelOffset = [0, 0] as [number, number]; + if (rotateVerticalGlyph) { + // Vertical POI labels that are rotated 90deg CW and whose glyphs must preserve upright orientation + // need to be rotated 90deg CCW. After a quad is rotated, it is translated to the original built-in offset. + verticalizedLabelOffset = builtInOffset; + builtInOffset = [0, 0]; + } + + const textureScale = positionedGlyph.metrics.isDoubleResolution ? 2 : 1; + + const x1 = (positionedGlyph.metrics.left - rectBuffer) * positionedGlyph.scale - halfAdvance + builtInOffset[0]; + const y1 = (-positionedGlyph.metrics.top - rectBuffer) * positionedGlyph.scale + builtInOffset[1]; + const x2 = x1 + textureRect.w / textureScale * positionedGlyph.scale / pixelRatio; + const y2 = y1 + textureRect.h / textureScale * positionedGlyph.scale / pixelRatio; + + const tl = new Point(x1, y1); + const tr = new Point(x2, y1); + const bl = new Point(x1, y2); + const br = new Point(x2, y2); + + if (rotateVerticalGlyph) { + // Vertical-supporting glyphs are laid out in 24x24 point boxes (1 square em) + // In horizontal orientation, the y values for glyphs are below the midline + // and we use a "yOffset" of -17 to pull them up to the middle. + // By rotating counter-clockwise around the point at the center of the left + // edge of a 24x24 layout box centered below the midline, we align the center + // of the glyphs with the horizontal midline, so the yOffset is no longer + // necessary, but we also pull the glyph to the left along the x axis. + // The y coordinate includes baseline yOffset, thus needs to be accounted + // for when glyph is rotated and translated. + const center = new Point(-halfAdvance, halfAdvance - SHAPING_DEFAULT_OFFSET); + const verticalRotation = -Math.PI / 2; + + // xHalfWidthOffsetCorrection is a difference between full-width and half-width + // advance, should be 0 for full-width glyphs and will pull up half-width glyphs. + const xHalfWidthOffsetCorrection = ONE_EM / 2 - halfAdvance; + const yImageOffsetCorrection = positionedGlyph.imageName ? xHalfWidthOffsetCorrection : 0.0; + const halfWidthOffsetCorrection = new Point(5 - SHAPING_DEFAULT_OFFSET - xHalfWidthOffsetCorrection, -yImageOffsetCorrection); + const verticalOffsetCorrection = new Point(...verticalizedLabelOffset); + tl._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection); + tr._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection); + bl._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection); + br._rotateAround(verticalRotation, center)._add(halfWidthOffsetCorrection)._add(verticalOffsetCorrection); + } + + if (textRotate) { + const sin = Math.sin(textRotate), + cos = Math.cos(textRotate), + matrix = [cos, -sin, sin, cos]; + + tl._matMult(matrix); + tr._matMult(matrix); + bl._matMult(matrix); + br._matMult(matrix); + } + + const pixelOffsetTL = new Point(0, 0); + const pixelOffsetBR = new Point(0, 0); + const minFontScaleX = 0; + const minFontScaleY = 0; + quads.push({tl, tr, bl, br, tex: textureRect, writingMode: shaping.writingMode, glyphOffset, sectionIndex: positionedGlyph.sectionIndex, isSDF, pixelOffsetTL, pixelOffsetBR, minFontScaleX, minFontScaleY}); + } + } + + return quads; +} diff --git a/web/libraries/maplibre-gl/src/symbol/shaping.ts b/web/libraries/maplibre-gl/src/symbol/shaping.ts new file mode 100644 index 00000000..acb3f6ab --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/shaping.ts @@ -0,0 +1,859 @@ +import { + charHasUprightVerticalOrientation, + charAllowsIdeographicBreaking, + charInComplexShapingScript +} from '../util/script_detection'; +import {verticalizePunctuation} from '../util/verticalize_punctuation'; +import {plugin as rtlTextPlugin} from '../source/rtl_text_plugin'; +import ONE_EM from './one_em'; +import {warnOnce} from '../util/util'; + +import type {StyleGlyph, GlyphMetrics} from '../style/style_glyph'; +import {GLYPH_PBF_BORDER} from '../style/parse_glyph_pbf'; +import type {ImagePosition} from '../render/image_atlas'; +import {IMAGE_PADDING} from '../render/image_atlas'; +import type {Rect, GlyphPosition} from '../render/glyph_atlas'; +import {Formatted, FormattedSection} from '@maplibre/maplibre-gl-style-spec'; + +enum WritingMode { + none = 0, + horizontal = 1, + vertical = 2, + horizontalOnly = 3 +} + +const SHAPING_DEFAULT_OFFSET = -17; +export {shapeText, shapeIcon, fitIconToText, getAnchorAlignment, WritingMode, SHAPING_DEFAULT_OFFSET}; + +// The position of a glyph relative to the text's anchor point. +export type PositionedGlyph = { + glyph: number; + imageName: string | null; + x: number; + y: number; + vertical: boolean; + scale: number; + fontStack: string; + sectionIndex: number; + metrics: GlyphMetrics; + rect: Rect | null; +}; + +export type PositionedLine = { + positionedGlyphs: Array; + lineOffset: number; +}; + +// A collection of positioned glyphs and some metadata +export type Shaping = { + positionedLines: Array; + top: number; + bottom: number; + left: number; + right: number; + writingMode: WritingMode.horizontal | WritingMode.vertical; + text: string; + iconsInText: boolean; + verticalizable: boolean; +}; + +function isEmpty(positionedLines: Array) { + for (const line of positionedLines) { + if (line.positionedGlyphs.length !== 0) { + return false; + } + } + return true; +} + +export type SymbolAnchor = 'center' | 'left' | 'right' | 'top' | 'bottom' | 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right'; +export type TextJustify = 'left' | 'center' | 'right'; + +// Max number of images in label is 6401 U+E000–U+F8FF that covers +// Basic Multilingual Plane Unicode Private Use Area (PUA). +const PUAbegin = 0xE000; +const PUAend = 0xF8FF; + +class SectionOptions { + // Text options + scale: number; + fontStack: string; + // Image options + imageName: string | null; + + constructor() { + this.scale = 1.0; + this.fontStack = ''; + this.imageName = null; + } + + static forText(scale: number | null, fontStack: string) { + const textOptions = new SectionOptions(); + textOptions.scale = scale || 1; + textOptions.fontStack = fontStack; + return textOptions; + } + + static forImage(imageName: string) { + const imageOptions = new SectionOptions(); + imageOptions.imageName = imageName; + return imageOptions; + } + +} + +class TaggedString { + text: string; + sectionIndex: Array; // maps each character in 'text' to its corresponding entry in 'sections' + sections: Array; + imageSectionID: number | null; + + constructor() { + this.text = ''; + this.sectionIndex = []; + this.sections = []; + this.imageSectionID = null; + } + + static fromFeature(text: Formatted, defaultFontStack: string) { + const result = new TaggedString(); + for (let i = 0; i < text.sections.length; i++) { + const section = text.sections[i]; + if (!section.image) { + result.addTextSection(section, defaultFontStack); + } else { + result.addImageSection(section); + } + } + return result; + } + + length(): number { + return this.text.length; + } + + getSection(index: number): SectionOptions { + return this.sections[this.sectionIndex[index]]; + } + + getSectionIndex(index: number): number { + return this.sectionIndex[index]; + } + + getCharCode(index: number): number { + return this.text.charCodeAt(index); + } + + verticalizePunctuation() { + this.text = verticalizePunctuation(this.text); + } + + trim() { + let beginningWhitespace = 0; + for (let i = 0; + i < this.text.length && whitespace[this.text.charCodeAt(i)]; + i++) { + beginningWhitespace++; + } + let trailingWhitespace = this.text.length; + for (let i = this.text.length - 1; + i >= 0 && i >= beginningWhitespace && whitespace[this.text.charCodeAt(i)]; + i--) { + trailingWhitespace--; + } + this.text = this.text.substring(beginningWhitespace, trailingWhitespace); + this.sectionIndex = this.sectionIndex.slice(beginningWhitespace, trailingWhitespace); + } + + substring(start: number, end: number): TaggedString { + const substring = new TaggedString(); + substring.text = this.text.substring(start, end); + substring.sectionIndex = this.sectionIndex.slice(start, end); + substring.sections = this.sections; + return substring; + } + + toString(): string { + return this.text; + } + + getMaxScale() { + return this.sectionIndex.reduce((max, index) => Math.max(max, this.sections[index].scale), 0); + } + + addTextSection(section: FormattedSection, defaultFontStack: string) { + this.text += section.text; + this.sections.push(SectionOptions.forText(section.scale, section.fontStack || defaultFontStack)); + const index = this.sections.length - 1; + for (let i = 0; i < section.text.length; ++i) { + this.sectionIndex.push(index); + } + } + + addImageSection(section: FormattedSection) { + const imageName = section.image ? section.image.name : ''; + if (imageName.length === 0) { + warnOnce('Can\'t add FormattedSection with an empty image.'); + return; + } + + const nextImageSectionCharCode = this.getNextImageSectionCharCode(); + if (!nextImageSectionCharCode) { + warnOnce(`Reached maximum number of images ${PUAend - PUAbegin + 2}`); + return; + } + + this.text += String.fromCharCode(nextImageSectionCharCode); + this.sections.push(SectionOptions.forImage(imageName)); + this.sectionIndex.push(this.sections.length - 1); + } + + getNextImageSectionCharCode(): number | null { + if (!this.imageSectionID) { + this.imageSectionID = PUAbegin; + return this.imageSectionID; + } + + if (this.imageSectionID >= PUAend) return null; + return ++this.imageSectionID; + } +} + +function breakLines(input: TaggedString, lineBreakPoints: Array): Array { + const lines = []; + const text = input.text; + let start = 0; + for (const lineBreak of lineBreakPoints) { + lines.push(input.substring(start, lineBreak)); + start = lineBreak; + } + + if (start < text.length) { + lines.push(input.substring(start, text.length)); + } + return lines; +} + +function shapeText( + text: Formatted, + glyphMap: { + [_: string]: { + [_: number]: StyleGlyph; + }; + }, + glyphPositions: { + [_: string]: { + [_: number]: GlyphPosition; + }; + }, + imagePositions: {[_: string]: ImagePosition}, + defaultFontStack: string, + maxWidth: number, + lineHeight: number, + textAnchor: SymbolAnchor, + textJustify: TextJustify, + spacing: number, + translate: [number, number], + writingMode: WritingMode.horizontal | WritingMode.vertical, + allowVerticalPlacement: boolean, + symbolPlacement: string, + layoutTextSize: number, + layoutTextSizeThisZoom: number +): Shaping | false { + const logicalInput = TaggedString.fromFeature(text, defaultFontStack); + + if (writingMode === WritingMode.vertical) { + logicalInput.verticalizePunctuation(); + } + + let lines: Array; + + const {processBidirectionalText, processStyledBidirectionalText} = rtlTextPlugin; + if (processBidirectionalText && logicalInput.sections.length === 1) { + // Bidi doesn't have to be style-aware + lines = []; + const untaggedLines = + processBidirectionalText(logicalInput.toString(), + determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize)); + for (const line of untaggedLines) { + const taggedLine = new TaggedString(); + taggedLine.text = line; + taggedLine.sections = logicalInput.sections; + for (let i = 0; i < line.length; i++) { + taggedLine.sectionIndex.push(0); + } + lines.push(taggedLine); + } + } else if (processStyledBidirectionalText) { + // Need version of mapbox-gl-rtl-text with style support for combining RTL text + // with formatting + lines = []; + const processedLines = + processStyledBidirectionalText(logicalInput.text, + logicalInput.sectionIndex, + determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize)); + for (const line of processedLines) { + const taggedLine = new TaggedString(); + taggedLine.text = line[0]; + taggedLine.sectionIndex = line[1]; + taggedLine.sections = logicalInput.sections; + lines.push(taggedLine); + } + } else { + lines = breakLines(logicalInput, determineLineBreaks(logicalInput, spacing, maxWidth, glyphMap, imagePositions, symbolPlacement, layoutTextSize)); + } + + const positionedLines = []; + const shaping = { + positionedLines, + text: logicalInput.toString(), + top: translate[1], + bottom: translate[1], + left: translate[0], + right: translate[0], + writingMode, + iconsInText: false, + verticalizable: false + }; + + shapeLines(shaping, glyphMap, glyphPositions, imagePositions, lines, lineHeight, textAnchor, textJustify, writingMode, spacing, allowVerticalPlacement, layoutTextSizeThisZoom); + if (isEmpty(positionedLines)) return false; + + return shaping; +} + +// using computed properties due to https://github.com/facebook/flow/issues/380 +/* eslint no-useless-computed-key: 0 */ + +const whitespace: { + [_: number]: boolean; +} = { + [0x09]: true, // tab + [0x0a]: true, // newline + [0x0b]: true, // vertical tab + [0x0c]: true, // form feed + [0x0d]: true, // carriage return + [0x20]: true, // space +}; + +const breakable: { + [_: number]: boolean; +} = { + [0x0a]: true, // newline + [0x20]: true, // space + [0x26]: true, // ampersand + [0x28]: true, // left parenthesis + [0x29]: true, // right parenthesis + [0x2b]: true, // plus sign + [0x2d]: true, // hyphen-minus + [0x2f]: true, // solidus + [0xad]: true, // soft hyphen + [0xb7]: true, // middle dot + [0x200b]: true, // zero-width space + [0x2010]: true, // hyphen + [0x2013]: true, // en dash + [0x2027]: true // interpunct + // Many other characters may be reasonable breakpoints + // Consider "neutral orientation" characters at scriptDetection.charHasNeutralVerticalOrientation + // See https://github.com/mapbox/mapbox-gl-js/issues/3658 +}; + +function getGlyphAdvance( + codePoint: number, + section: SectionOptions, + glyphMap: { + [_: string]: { + [_: number]: StyleGlyph; + }; + }, + imagePositions: {[_: string]: ImagePosition}, + spacing: number, + layoutTextSize: number +): number { + if (!section.imageName) { + const positions = glyphMap[section.fontStack]; + const glyph = positions && positions[codePoint]; + if (!glyph) return 0; + return glyph.metrics.advance * section.scale + spacing; + } else { + const imagePosition = imagePositions[section.imageName]; + if (!imagePosition) return 0; + return imagePosition.displaySize[0] * section.scale * ONE_EM / layoutTextSize + spacing; + } +} + +function determineAverageLineWidth(logicalInput: TaggedString, + spacing: number, + maxWidth: number, + glyphMap: { + [_: string]: { + [_: number]: StyleGlyph; + }; + }, + imagePositions: {[_: string]: ImagePosition}, + layoutTextSize: number) { + let totalWidth = 0; + + for (let index = 0; index < logicalInput.length(); index++) { + const section = logicalInput.getSection(index); + totalWidth += getGlyphAdvance(logicalInput.getCharCode(index), section, glyphMap, imagePositions, spacing, layoutTextSize); + } + + const lineCount = Math.max(1, Math.ceil(totalWidth / maxWidth)); + return totalWidth / lineCount; +} + +function calculateBadness(lineWidth: number, + targetWidth: number, + penalty: number, + isLastBreak: boolean) { + const raggedness = Math.pow(lineWidth - targetWidth, 2); + if (isLastBreak) { + // Favor finals lines shorter than average over longer than average + if (lineWidth < targetWidth) { + return raggedness / 2; + } else { + return raggedness * 2; + } + } + + return raggedness + Math.abs(penalty) * penalty; +} + +function calculatePenalty(codePoint: number, nextCodePoint: number, penalizableIdeographicBreak: boolean) { + let penalty = 0; + // Force break on newline + if (codePoint === 0x0a) { + penalty -= 10000; + } + // Penalize breaks between characters that allow ideographic breaking because + // they are less preferable than breaks at spaces (or zero width spaces). + if (penalizableIdeographicBreak) { + penalty += 150; + } + + // Penalize open parenthesis at end of line + if (codePoint === 0x28 || codePoint === 0xff08) { + penalty += 50; + } + + // Penalize close parenthesis at beginning of line + if (nextCodePoint === 0x29 || nextCodePoint === 0xff09) { + penalty += 50; + } + return penalty; +} + +type Break = { + index: number; + x: number; + priorBreak: Break; + badness: number; +}; + +function evaluateBreak( + breakIndex: number, + breakX: number, + targetWidth: number, + potentialBreaks: Array, + penalty: number, + isLastBreak: boolean +): Break { + // We could skip evaluating breaks where the line length (breakX - priorBreak.x) > maxWidth + // ...but in fact we allow lines longer than maxWidth (if there's no break points) + // ...and when targetWidth and maxWidth are close, strictly enforcing maxWidth can give + // more lopsided results. + + let bestPriorBreak: Break = null; + let bestBreakBadness = calculateBadness(breakX, targetWidth, penalty, isLastBreak); + + for (const potentialBreak of potentialBreaks) { + const lineWidth = breakX - potentialBreak.x; + const breakBadness = + calculateBadness(lineWidth, targetWidth, penalty, isLastBreak) + potentialBreak.badness; + if (breakBadness <= bestBreakBadness) { + bestPriorBreak = potentialBreak; + bestBreakBadness = breakBadness; + } + } + + return { + index: breakIndex, + x: breakX, + priorBreak: bestPriorBreak, + badness: bestBreakBadness + }; +} + +function leastBadBreaks(lastLineBreak?: Break | null): Array { + if (!lastLineBreak) { + return []; + } + return leastBadBreaks(lastLineBreak.priorBreak).concat(lastLineBreak.index); +} + +function determineLineBreaks( + logicalInput: TaggedString, + spacing: number, + maxWidth: number, + glyphMap: { + [_: string]: { + [_: number]: StyleGlyph; + }; + }, + imagePositions: {[_: string]: ImagePosition}, + symbolPlacement: string, + layoutTextSize: number +): Array { + if (symbolPlacement !== 'point') + return []; + + if (!logicalInput) + return []; + + const potentialLineBreaks = []; + const targetWidth = determineAverageLineWidth(logicalInput, spacing, maxWidth, glyphMap, imagePositions, layoutTextSize); + + const hasServerSuggestedBreakpoints = logicalInput.text.indexOf('\u200b') >= 0; + + let currentX = 0; + + for (let i = 0; i < logicalInput.length(); i++) { + const section = logicalInput.getSection(i); + const codePoint = logicalInput.getCharCode(i); + if (!whitespace[codePoint]) currentX += getGlyphAdvance(codePoint, section, glyphMap, imagePositions, spacing, layoutTextSize); + + // Ideographic characters, spaces, and word-breaking punctuation that often appear without + // surrounding spaces. + if ((i < logicalInput.length() - 1)) { + const ideographicBreak = charAllowsIdeographicBreaking(codePoint); + if (breakable[codePoint] || ideographicBreak || section.imageName) { + + potentialLineBreaks.push( + evaluateBreak( + i + 1, + currentX, + targetWidth, + potentialLineBreaks, + calculatePenalty(codePoint, logicalInput.getCharCode(i + 1), ideographicBreak && hasServerSuggestedBreakpoints), + false)); + } + } + } + + return leastBadBreaks( + evaluateBreak( + logicalInput.length(), + currentX, + targetWidth, + potentialLineBreaks, + 0, + true)); +} + +function getAnchorAlignment(anchor: SymbolAnchor) { + let horizontalAlign = 0.5, verticalAlign = 0.5; + + switch (anchor) { + case 'right': + case 'top-right': + case 'bottom-right': + horizontalAlign = 1; + break; + case 'left': + case 'top-left': + case 'bottom-left': + horizontalAlign = 0; + break; + } + + switch (anchor) { + case 'bottom': + case 'bottom-right': + case 'bottom-left': + verticalAlign = 1; + break; + case 'top': + case 'top-right': + case 'top-left': + verticalAlign = 0; + break; + } + + return {horizontalAlign, verticalAlign}; +} + +function shapeLines(shaping: Shaping, + glyphMap: { + [_: string]: { + [_: number]: StyleGlyph; + }; + }, + glyphPositions: { + [_: string]: { + [_: number]: GlyphPosition; + }; + }, + imagePositions: {[_: string]: ImagePosition}, + lines: Array, + lineHeight: number, + textAnchor: SymbolAnchor, + textJustify: TextJustify, + writingMode: WritingMode.horizontal | WritingMode.vertical, + spacing: number, + allowVerticalPlacement: boolean, + layoutTextSizeThisZoom: number) { + + let x = 0; + let y = SHAPING_DEFAULT_OFFSET; + + let maxLineLength = 0; + let maxLineHeight = 0; + + const justify = + textJustify === 'right' ? 1 : + textJustify === 'left' ? 0 : 0.5; + + let lineIndex = 0; + for (const line of lines) { + line.trim(); + + const lineMaxScale = line.getMaxScale(); + const maxLineOffset = (lineMaxScale - 1) * ONE_EM; + const positionedLine = {positionedGlyphs: [], lineOffset: 0}; + shaping.positionedLines[lineIndex] = positionedLine; + const positionedGlyphs = positionedLine.positionedGlyphs; + let lineOffset = 0.0; + + if (!line.length()) { + y += lineHeight; // Still need a line feed after empty line + ++lineIndex; + continue; + } + + for (let i = 0; i < line.length(); i++) { + const section = line.getSection(i); + const sectionIndex = line.getSectionIndex(i); + const codePoint = line.getCharCode(i); + let baselineOffset = 0.0; + let metrics = null; + let rect = null; + let imageName = null; + let verticalAdvance = ONE_EM; + const vertical = !(writingMode === WritingMode.horizontal || + // Don't verticalize glyphs that have no upright orientation if vertical placement is disabled. + (!allowVerticalPlacement && !charHasUprightVerticalOrientation(codePoint)) || + // If vertical placement is enabled, don't verticalize glyphs that + // are from complex text layout script, or whitespaces. + (allowVerticalPlacement && (whitespace[codePoint] || charInComplexShapingScript(codePoint)))); + + if (!section.imageName) { + const positions = glyphPositions[section.fontStack]; + const glyphPosition = positions && positions[codePoint]; + if (glyphPosition && glyphPosition.rect) { + rect = glyphPosition.rect; + metrics = glyphPosition.metrics; + } else { + const glyphs = glyphMap[section.fontStack]; + const glyph = glyphs && glyphs[codePoint]; + if (!glyph) continue; + metrics = glyph.metrics; + } + + // We don't know the baseline, but since we're laying out + // at 24 points, we can calculate how much it will move when + // we scale up or down. + baselineOffset = (lineMaxScale - section.scale) * ONE_EM; + } else { + const imagePosition = imagePositions[section.imageName]; + if (!imagePosition) continue; + imageName = section.imageName; + shaping.iconsInText = shaping.iconsInText || true; + rect = imagePosition.paddedRect; + const size = imagePosition.displaySize; + // If needed, allow to set scale factor for an image using + // alias "image-scale" that could be alias for "font-scale" + // when FormattedSection is an image section. + section.scale = section.scale * ONE_EM / layoutTextSizeThisZoom; + + metrics = {width: size[0], + height: size[1], + left: IMAGE_PADDING, + top: -GLYPH_PBF_BORDER, + advance: vertical ? size[1] : size[0]}; + + // Difference between one EM and an image size. + // Aligns bottom of an image to a baseline level. + const imageOffset = ONE_EM - size[1] * section.scale; + baselineOffset = maxLineOffset + imageOffset; + verticalAdvance = metrics.advance; + + // Difference between height of an image and one EM at max line scale. + // Pushes current line down if an image size is over 1 EM at max line scale. + const offset = vertical ? size[0] * section.scale - ONE_EM * lineMaxScale : + size[1] * section.scale - ONE_EM * lineMaxScale; + if (offset > 0 && offset > lineOffset) { + lineOffset = offset; + } + } + + if (!vertical) { + positionedGlyphs.push({glyph: codePoint, imageName, x, y: y + baselineOffset, vertical, scale: section.scale, fontStack: section.fontStack, sectionIndex, metrics, rect}); + x += metrics.advance * section.scale + spacing; + } else { + shaping.verticalizable = true; + positionedGlyphs.push({glyph: codePoint, imageName, x, y: y + baselineOffset, vertical, scale: section.scale, fontStack: section.fontStack, sectionIndex, metrics, rect}); + x += verticalAdvance * section.scale + spacing; + } + } + + // Only justify if we placed at least one glyph + if (positionedGlyphs.length !== 0) { + const lineLength = x - spacing; + maxLineLength = Math.max(lineLength, maxLineLength); + justifyLine(positionedGlyphs, 0, positionedGlyphs.length - 1, justify, lineOffset); + } + + x = 0; + const currentLineHeight = lineHeight * lineMaxScale + lineOffset; + positionedLine.lineOffset = Math.max(lineOffset, maxLineOffset); + y += currentLineHeight; + maxLineHeight = Math.max(currentLineHeight, maxLineHeight); + ++lineIndex; + } + + // Calculate the bounding box and justify / align text block. + const height = y - SHAPING_DEFAULT_OFFSET; + const {horizontalAlign, verticalAlign} = getAnchorAlignment(textAnchor); + align(shaping.positionedLines, justify, horizontalAlign, verticalAlign, maxLineLength, maxLineHeight, lineHeight, height, lines.length); + + shaping.top += -verticalAlign * height; + shaping.bottom = shaping.top + height; + shaping.left += -horizontalAlign * maxLineLength; + shaping.right = shaping.left + maxLineLength; +} + +// justify right = 1, left = 0, center = 0.5 +function justifyLine(positionedGlyphs: Array, + start: number, + end: number, + justify: 1 | 0 | 0.5, + lineOffset: number) { + if (!justify && !lineOffset) + return; + + const lastPositionedGlyph = positionedGlyphs[end]; + const lastAdvance = lastPositionedGlyph.metrics.advance * lastPositionedGlyph.scale; + const lineIndent = (positionedGlyphs[end].x + lastAdvance) * justify; + + for (let j = start; j <= end; j++) { + positionedGlyphs[j].x -= lineIndent; + positionedGlyphs[j].y += lineOffset; + } +} + +function align(positionedLines: Array, + justify: number, + horizontalAlign: number, + verticalAlign: number, + maxLineLength: number, + maxLineHeight: number, + lineHeight: number, + blockHeight: number, + lineCount: number) { + const shiftX = (justify - horizontalAlign) * maxLineLength; + let shiftY = 0; + + if (maxLineHeight !== lineHeight) { + shiftY = -blockHeight * verticalAlign - SHAPING_DEFAULT_OFFSET; + } else { + shiftY = (-verticalAlign * lineCount + 0.5) * lineHeight; + } + + for (const line of positionedLines) { + for (const positionedGlyph of line.positionedGlyphs) { + positionedGlyph.x += shiftX; + positionedGlyph.y += shiftY; + } + } +} + +export type PositionedIcon = { + image: ImagePosition; + top: number; + bottom: number; + left: number; + right: number; + collisionPadding?: [number, number, number, number]; +}; + +function shapeIcon( + image: ImagePosition, + iconOffset: [number, number], + iconAnchor: SymbolAnchor +): PositionedIcon { + const {horizontalAlign, verticalAlign} = getAnchorAlignment(iconAnchor); + const dx = iconOffset[0]; + const dy = iconOffset[1]; + const x1 = dx - image.displaySize[0] * horizontalAlign; + const x2 = x1 + image.displaySize[0]; + const y1 = dy - image.displaySize[1] * verticalAlign; + const y2 = y1 + image.displaySize[1]; + return {image, top: y1, bottom: y2, left: x1, right: x2}; +} + +function fitIconToText( + shapedIcon: PositionedIcon, + shapedText: Shaping, + textFit: string, + padding: [number, number, number, number], + iconOffset: [number, number], + fontScale: number +): PositionedIcon { + + const image = shapedIcon.image; + + let collisionPadding; + if (image.content) { + const content = image.content; + const pixelRatio = image.pixelRatio || 1; + collisionPadding = [ + content[0] / pixelRatio, + content[1] / pixelRatio, + image.displaySize[0] - content[2] / pixelRatio, + image.displaySize[1] - content[3] / pixelRatio + ]; + } + + // We don't respect the icon-anchor, because icon-text-fit is set. Instead, + // the icon will be centered on the text, then stretched in the given + // dimensions. + + const textLeft = shapedText.left * fontScale; + const textRight = shapedText.right * fontScale; + + let top, right, bottom, left; + if (textFit === 'width' || textFit === 'both') { + // Stretched horizontally to the text width + left = iconOffset[0] + textLeft - padding[3]; + right = iconOffset[0] + textRight + padding[1]; + } else { + // Centered on the text + left = iconOffset[0] + (textLeft + textRight - image.displaySize[0]) / 2; + right = left + image.displaySize[0]; + } + + const textTop = shapedText.top * fontScale; + const textBottom = shapedText.bottom * fontScale; + if (textFit === 'height' || textFit === 'both') { + // Stretched vertically to the text height + top = iconOffset[1] + textTop - padding[0]; + bottom = iconOffset[1] + textBottom + padding[2]; + } else { + // Centered on the text + top = iconOffset[1] + (textTop + textBottom - image.displaySize[1]) / 2; + bottom = top + image.displaySize[1]; + } + + return {image, top, right, bottom, left, collisionPadding}; +} diff --git a/web/libraries/maplibre-gl/src/symbol/symbol_layout.ts b/web/libraries/maplibre-gl/src/symbol/symbol_layout.ts new file mode 100644 index 00000000..bd37eb42 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/symbol_layout.ts @@ -0,0 +1,739 @@ +import {Anchor} from './anchor'; + +import {getAnchors, getCenterAnchor} from './get_anchors'; +import {clipLine} from './clip_line'; +import {shapeText, shapeIcon, WritingMode, fitIconToText} from './shaping'; +import {getGlyphQuads, getIconQuads} from './quads'; +import {CollisionFeature} from './collision_feature'; +import {warnOnce} from '../util/util'; +import { + allowsVerticalWritingMode, + allowsLetterSpacing +} from '../util/script_detection'; +import {findPoleOfInaccessibility} from '../util/find_pole_of_inaccessibility'; +import {classifyRings} from '../util/classify_rings'; +import {EXTENT} from '../data/extent'; +import {SymbolBucket} from '../data/bucket/symbol_bucket'; +import {EvaluationParameters} from '../style/evaluation_parameters'; +import {SIZE_PACK_FACTOR, MAX_PACKED_SIZE, MAX_GLYPH_ICON_SIZE} from './symbol_size'; +import ONE_EM from './one_em'; +import type {CanonicalTileID} from '../source/tile_id'; +import type {Shaping, PositionedIcon, TextJustify} from './shaping'; +import type {CollisionBoxArray, TextAnchorOffsetArray} from '../data/array_types.g'; +import type {SymbolFeature} from '../data/bucket/symbol_bucket'; +import type {StyleImage} from '../style/style_image'; +import type {StyleGlyph} from '../style/style_glyph'; +import type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer'; +import type {ImagePosition} from '../render/image_atlas'; +import type {GlyphPosition} from '../render/glyph_atlas'; +import type {PossiblyEvaluatedPropertyValue} from '../style/properties'; + +import Point from '@mapbox/point-geometry'; +import murmur3 from 'murmurhash-js'; +import {getIconPadding, SymbolPadding} from '../style/style_layer/symbol_style_layer'; +import {VariableAnchorOffsetCollection} from '@maplibre/maplibre-gl-style-spec'; +import {getTextVariableAnchorOffset, evaluateVariableOffset, INVALID_TEXT_OFFSET, TextAnchor, TextAnchorEnum} from '../style/style_layer/variable_text_anchor'; + +// The symbol layout process needs `text-size` evaluated at up to five different zoom levels, and +// `icon-size` at up to three: +// +// 1. `text-size` at the zoom level of the bucket. Used to calculate a per-feature size for source `text-size` +// expressions, and to calculate the box dimensions for icon-text-fit. +// 2. `icon-size` at the zoom level of the bucket. Used to calculate a per-feature size for source `icon-size` +// expressions. +// 3. `text-size` and `icon-size` at the zoom level of the bucket, plus one. Used to calculate collision boxes. +// 4. `text-size` at zoom level 18. Used for something line-symbol-placement-related. +// 5. For composite `*-size` expressions: two zoom levels of curve stops that "cover" the zoom level of the +// bucket. These go into a vertex buffer and are used by the shader to interpolate the size at render time. +// +// (1) and (2) are stored in `bucket.layers[0].layout`. The remainder are below. +// +type Sizes = { + layoutTextSize: PossiblyEvaluatedPropertyValue; // (3), + layoutIconSize: PossiblyEvaluatedPropertyValue; // (3), + textMaxSize: PossiblyEvaluatedPropertyValue; // (4), + compositeTextSizes: [PossiblyEvaluatedPropertyValue, PossiblyEvaluatedPropertyValue]; // (5), + compositeIconSizes: [PossiblyEvaluatedPropertyValue, PossiblyEvaluatedPropertyValue]; // (5) +}; + +type ShapedTextOrientations = { + vertical: Shaping | false; + horizontal: Record; +}; + +export function performSymbolLayout(args: { + bucket: SymbolBucket; + glyphMap: { + [_: string]: { + [x: number]: StyleGlyph; + }; + }; + glyphPositions: { + [_: string]: { + [x: number]: GlyphPosition; + }; + }; + imageMap: {[_: string]: StyleImage}; + imagePositions: {[_: string]: ImagePosition}; + showCollisionBoxes: boolean; + canonical: CanonicalTileID; +}) { + args.bucket.createArrays(); + + const tileSize = 512 * args.bucket.overscaling; + args.bucket.tilePixelRatio = EXTENT / tileSize; + args.bucket.compareText = {}; + args.bucket.iconsNeedLinear = false; + + const layer = args.bucket.layers[0]; + const layout = layer.layout; + const unevaluatedLayoutValues = layer._unevaluatedLayout._values; + + const sizes: Sizes = { + // Filled in below, if *SizeData.kind is 'composite' + // compositeIconSizes: undefined, + // compositeTextSizes: undefined, + layoutIconSize: unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(args.bucket.zoom + 1), args.canonical), + layoutTextSize: unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(args.bucket.zoom + 1), args.canonical), + textMaxSize: unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(18)) + } as Sizes; + + if (args.bucket.textSizeData.kind === 'composite') { + const {minZoom, maxZoom} = args.bucket.textSizeData; + sizes.compositeTextSizes = [ + unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(minZoom), args.canonical), + unevaluatedLayoutValues['text-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), args.canonical) + ]; + } + + if (args.bucket.iconSizeData.kind === 'composite') { + const {minZoom, maxZoom} = args.bucket.iconSizeData; + sizes.compositeIconSizes = [ + unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(minZoom), args.canonical), + unevaluatedLayoutValues['icon-size'].possiblyEvaluate(new EvaluationParameters(maxZoom), args.canonical) + ]; + } + + const lineHeight = layout.get('text-line-height') * ONE_EM; + const textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point'; + const keepUpright = layout.get('text-keep-upright'); + const textSize = layout.get('text-size'); + + for (const feature of args.bucket.features) { + const fontstack = layout.get('text-font').evaluate(feature, {}, args.canonical).join(','); + const layoutTextSizeThisZoom = textSize.evaluate(feature, {}, args.canonical); + const layoutTextSize = sizes.layoutTextSize.evaluate(feature, {}, args.canonical); + const layoutIconSize = sizes.layoutIconSize.evaluate(feature, {}, args.canonical); + + const shapedTextOrientations: ShapedTextOrientations = { + horizontal: {} as Record, + vertical: undefined + }; + const text = feature.text; + let textOffset: [number, number] = [0, 0]; + if (text) { + const unformattedText = text.toString(); + const spacing = layout.get('text-letter-spacing').evaluate(feature, {}, args.canonical) * ONE_EM; + const spacingIfAllowed = allowsLetterSpacing(unformattedText) ? spacing : 0; + + const textAnchor = layout.get('text-anchor').evaluate(feature, {}, args.canonical); + const variableAnchorOffset = getTextVariableAnchorOffset(layer, feature, args.canonical); + + if (!variableAnchorOffset) { + const radialOffset = layout.get('text-radial-offset').evaluate(feature, {}, args.canonical); + // Layers with variable anchors use the `text-radial-offset` property and the [x, y] offset vector + // is calculated at placement time instead of layout time + if (radialOffset) { + // The style spec says don't use `text-offset` and `text-radial-offset` together + // but doesn't actually specify what happens if you use both. We go with the radial offset. + textOffset = evaluateVariableOffset(textAnchor, [radialOffset * ONE_EM, INVALID_TEXT_OFFSET]); + } else { + textOffset = (layout.get('text-offset').evaluate(feature, {}, args.canonical).map(t => t * ONE_EM) as [number, number]); + } + } + + let textJustify = textAlongLine ? + 'center' : + layout.get('text-justify').evaluate(feature, {}, args.canonical); + + const symbolPlacement = layout.get('symbol-placement'); + const maxWidth = symbolPlacement === 'point' ? + layout.get('text-max-width').evaluate(feature, {}, args.canonical) * ONE_EM : + 0; + + const addVerticalShapingForPointLabelIfNeeded = () => { + if (args.bucket.allowVerticalPlacement && allowsVerticalWritingMode(unformattedText)) { + // Vertical POI label placement is meant to be used for scripts that support vertical + // writing mode, thus, default left justification is used. If Latin + // scripts would need to be supported, this should take into account other justifications. + shapedTextOrientations.vertical = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, + 'left', spacingIfAllowed, textOffset, WritingMode.vertical, true, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom); + } + }; + + // If this layer uses text-variable-anchor, generate shapings for all justification possibilities. + if (!textAlongLine && variableAnchorOffset) { + const justifications = new Set(); + + if (textJustify === 'auto') { + for (let i = 0; i < variableAnchorOffset.values.length; i += 2) { + justifications.add(getAnchorJustification(variableAnchorOffset.values[i] as TextAnchor)); + } + } else { + justifications.add(textJustify); + } + + let singleLine = false; + for (const justification of justifications) { + if (shapedTextOrientations.horizontal[justification]) continue; + if (singleLine) { + // If the shaping for the first justification was only a single line, we + // can re-use it for the other justifications + shapedTextOrientations.horizontal[justification] = shapedTextOrientations.horizontal[0]; + } else { + // If using text-variable-anchor for the layer, we use a center anchor for all shapings and apply + // the offsets for the anchor in the placement step. + const shaping = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, 'center', + justification, spacingIfAllowed, textOffset, WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom); + if (shaping) { + shapedTextOrientations.horizontal[justification] = shaping; + singleLine = shaping.positionedLines.length === 1; + } + } + } + + addVerticalShapingForPointLabelIfNeeded(); + } else { + if (textJustify === 'auto') { + textJustify = getAnchorJustification(textAnchor); + } + + // Horizontal point or line label. + const shaping = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify, spacingIfAllowed, + textOffset, WritingMode.horizontal, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom); + if (shaping) shapedTextOrientations.horizontal[textJustify] = shaping; + + // Vertical point label (if allowVerticalPlacement is enabled). + addVerticalShapingForPointLabelIfNeeded(); + + // Verticalized line label. + if (allowsVerticalWritingMode(unformattedText) && textAlongLine && keepUpright) { + shapedTextOrientations.vertical = shapeText(text, args.glyphMap, args.glyphPositions, args.imagePositions, fontstack, maxWidth, lineHeight, textAnchor, textJustify, + spacingIfAllowed, textOffset, WritingMode.vertical, false, symbolPlacement, layoutTextSize, layoutTextSizeThisZoom); + } + } + } + + let shapedIcon; + let isSDFIcon = false; + if (feature.icon && feature.icon.name) { + const image = args.imageMap[feature.icon.name]; + if (image) { + shapedIcon = shapeIcon( + args.imagePositions[feature.icon.name], + layout.get('icon-offset').evaluate(feature, {}, args.canonical), + layout.get('icon-anchor').evaluate(feature, {}, args.canonical)); + // null/undefined SDF property treated same as default (false) + isSDFIcon = !!image.sdf; + if (args.bucket.sdfIcons === undefined) { + args.bucket.sdfIcons = isSDFIcon; + } else if (args.bucket.sdfIcons !== isSDFIcon) { + warnOnce('Style sheet warning: Cannot mix SDF and non-SDF icons in one buffer'); + } + if (image.pixelRatio !== args.bucket.pixelRatio) { + args.bucket.iconsNeedLinear = true; + } else if (layout.get('icon-rotate').constantOr(1) !== 0) { + args.bucket.iconsNeedLinear = true; + } + } + } + + const shapedText = getDefaultHorizontalShaping(shapedTextOrientations.horizontal) || shapedTextOrientations.vertical; + args.bucket.iconsInText = shapedText ? shapedText.iconsInText : false; + if (shapedText || shapedIcon) { + addFeature(args.bucket, feature, shapedTextOrientations, shapedIcon, args.imageMap, sizes, layoutTextSize, layoutIconSize, textOffset, isSDFIcon, args.canonical); + } + } + + if (args.showCollisionBoxes) { + args.bucket.generateCollisionDebugBuffers(); + } +} + +// Choose the justification that matches the direction of the TextAnchor +export function getAnchorJustification(anchor: TextAnchor): TextJustify { + switch (anchor) { + case 'right': + case 'top-right': + case 'bottom-right': + return 'right'; + case 'left': + case 'top-left': + case 'bottom-left': + return 'left'; + } + return 'center'; +} + +/** + * Given a feature and its shaped text and icon data, add a 'symbol + * instance' for each _possible_ placement of the symbol feature. + * (At render timePlaceSymbols#place() selects which of these instances to + * show or hide based on collisions with symbols in other layers.) + */ +function addFeature(bucket: SymbolBucket, + feature: SymbolFeature, + shapedTextOrientations: ShapedTextOrientations, + shapedIcon: PositionedIcon, + imageMap: {[_: string]: StyleImage}, + sizes: Sizes, + layoutTextSize: number, + layoutIconSize: number, + textOffset: [number, number], + isSDFIcon: boolean, canonical: CanonicalTileID) { + // To reduce the number of labels that jump around when zooming we need + // to use a text-size value that is the same for all zoom levels. + // bucket calculates text-size at a high zoom level so that all tiles can + // use the same value when calculating anchor positions. + let textMaxSize = sizes.textMaxSize.evaluate(feature, {}); + if (textMaxSize === undefined) { + textMaxSize = layoutTextSize; + } + const layout = bucket.layers[0].layout; + const iconOffset = layout.get('icon-offset').evaluate(feature, {}, canonical); + const defaultHorizontalShaping = getDefaultHorizontalShaping(shapedTextOrientations.horizontal); + const glyphSize = 24, + fontScale = layoutTextSize / glyphSize, + textBoxScale = bucket.tilePixelRatio * fontScale, + textMaxBoxScale = bucket.tilePixelRatio * textMaxSize / glyphSize, + iconBoxScale = bucket.tilePixelRatio * layoutIconSize, + symbolMinDistance = bucket.tilePixelRatio * layout.get('symbol-spacing'), + textPadding = layout.get('text-padding') * bucket.tilePixelRatio, + iconPadding = getIconPadding(layout, feature, canonical, bucket.tilePixelRatio), + textMaxAngle = layout.get('text-max-angle') / 180 * Math.PI, + textAlongLine = layout.get('text-rotation-alignment') !== 'viewport' && layout.get('symbol-placement') !== 'point', + iconAlongLine = layout.get('icon-rotation-alignment') === 'map' && layout.get('symbol-placement') !== 'point', + symbolPlacement = layout.get('symbol-placement'), + textRepeatDistance = symbolMinDistance / 2; + + const iconTextFit = layout.get('icon-text-fit'); + let verticallyShapedIcon; + // Adjust shaped icon size when icon-text-fit is used. + if (shapedIcon && iconTextFit !== 'none') { + if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) { + verticallyShapedIcon = fitIconToText(shapedIcon, shapedTextOrientations.vertical, iconTextFit, + layout.get('icon-text-fit-padding'), iconOffset, fontScale); + } + if (defaultHorizontalShaping) { + shapedIcon = fitIconToText(shapedIcon, defaultHorizontalShaping, iconTextFit, + layout.get('icon-text-fit-padding'), iconOffset, fontScale); + } + } + + const addSymbolAtAnchor = (line, anchor) => { + if (anchor.x < 0 || anchor.x >= EXTENT || anchor.y < 0 || anchor.y >= EXTENT) { + // Symbol layers are drawn across tile boundaries, We filter out symbols + // outside our tile boundaries (which may be included in vector tile buffers) + // to prevent double-drawing symbols. + return; + } + + addSymbol(bucket, anchor, line, shapedTextOrientations, shapedIcon, imageMap, verticallyShapedIcon, bucket.layers[0], + bucket.collisionBoxArray, feature.index, feature.sourceLayerIndex, bucket.index, + textBoxScale, [textPadding, textPadding, textPadding, textPadding], textAlongLine, textOffset, + iconBoxScale, iconPadding, iconAlongLine, iconOffset, + feature, sizes, isSDFIcon, canonical, layoutTextSize); + }; + + if (symbolPlacement === 'line') { + for (const line of clipLine(feature.geometry, 0, 0, EXTENT, EXTENT)) { + const anchors = getAnchors( + line, + symbolMinDistance, + textMaxAngle, + shapedTextOrientations.vertical || defaultHorizontalShaping, + shapedIcon, + glyphSize, + textMaxBoxScale, + bucket.overscaling, + EXTENT + ); + for (const anchor of anchors) { + const shapedText = defaultHorizontalShaping; + if (!shapedText || !anchorIsTooClose(bucket, shapedText.text, textRepeatDistance, anchor)) { + addSymbolAtAnchor(line, anchor); + } + } + } + } else if (symbolPlacement === 'line-center') { + // No clipping, multiple lines per feature are allowed + // "lines" with only one point are ignored as in clipLines + for (const line of feature.geometry) { + if (line.length > 1) { + const anchor = getCenterAnchor( + line, + textMaxAngle, + shapedTextOrientations.vertical || defaultHorizontalShaping, + shapedIcon, + glyphSize, + textMaxBoxScale); + if (anchor) { + addSymbolAtAnchor(line, anchor); + } + } + } + } else if (feature.type === 'Polygon') { + for (const polygon of classifyRings(feature.geometry, 0)) { + // 16 here represents 2 pixels + const poi = findPoleOfInaccessibility(polygon, 16); + addSymbolAtAnchor(polygon[0], new Anchor(poi.x, poi.y, 0)); + } + } else if (feature.type === 'LineString') { + // https://github.com/mapbox/mapbox-gl-js/issues/3808 + for (const line of feature.geometry) { + addSymbolAtAnchor(line, new Anchor(line[0].x, line[0].y, 0)); + } + } else if (feature.type === 'Point') { + for (const points of feature.geometry) { + for (const point of points) { + addSymbolAtAnchor([point], new Anchor(point.x, point.y, 0)); + } + } + } +} + +function addTextVariableAnchorOffsets(textAnchorOffsets: TextAnchorOffsetArray, variableAnchorOffset: VariableAnchorOffsetCollection): [number, number] { + const startIndex = textAnchorOffsets.length; + const values = variableAnchorOffset?.values; + + if (values?.length > 0) { + for (let i = 0; i < values.length; i += 2) { + const anchor = TextAnchorEnum[values[i] as TextAnchor]; + const offset = values[i + 1] as [number, number]; + + textAnchorOffsets.emplaceBack(anchor, offset[0], offset[1]); + } + } + + return [startIndex, textAnchorOffsets.length]; +} + +function addTextVertices(bucket: SymbolBucket, + anchor: Point, + shapedText: Shaping, + imageMap: {[_: string]: StyleImage}, + layer: SymbolStyleLayer, + textAlongLine: boolean, + feature: SymbolFeature, + textOffset: [number, number], + lineArray: { + lineStartIndex: number; + lineLength: number; + }, + writingMode: WritingMode, + placementTypes: Array<'vertical' | 'center' | 'left' | 'right'>, + placedTextSymbolIndices: {[_: string]: number}, + placedIconIndex: number, + sizes: Sizes, + canonical: CanonicalTileID) { + const glyphQuads = getGlyphQuads(anchor, shapedText, textOffset, + layer, textAlongLine, feature, imageMap, bucket.allowVerticalPlacement); + + const sizeData = bucket.textSizeData; + let textSizeData = null; + + if (sizeData.kind === 'source') { + textSizeData = [ + SIZE_PACK_FACTOR * layer.layout.get('text-size').evaluate(feature, {}) + ]; + if (textSizeData[0] > MAX_PACKED_SIZE) { + warnOnce(`${bucket.layerIds[0]}: Value for "text-size" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your "text-size".`); + } + } else if (sizeData.kind === 'composite') { + textSizeData = [ + SIZE_PACK_FACTOR * sizes.compositeTextSizes[0].evaluate(feature, {}, canonical), + SIZE_PACK_FACTOR * sizes.compositeTextSizes[1].evaluate(feature, {}, canonical) + ]; + if (textSizeData[0] > MAX_PACKED_SIZE || textSizeData[1] > MAX_PACKED_SIZE) { + warnOnce(`${bucket.layerIds[0]}: Value for "text-size" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your "text-size".`); + } + } + + bucket.addSymbols( + bucket.text, + glyphQuads, + textSizeData, + textOffset, + textAlongLine, + feature, + writingMode, + anchor, + lineArray.lineStartIndex, + lineArray.lineLength, + placedIconIndex, + canonical); + + // The placedSymbolArray is used at render time in drawTileSymbols + // These indices allow access to the array at collision detection time + for (const placementType of placementTypes) { + placedTextSymbolIndices[placementType] = bucket.text.placedSymbolArray.length - 1; + } + + return glyphQuads.length * 4; +} + +function getDefaultHorizontalShaping( + horizontalShaping: Record +): Shaping | null { + // We don't care which shaping we get because this is used for collision purposes + // and all the justifications have the same collision box + for (const justification in horizontalShaping) { + return horizontalShaping[justification]; + } + return null; +} + +/** + * Add a single label & icon placement. + */ +function addSymbol(bucket: SymbolBucket, + anchor: Anchor, + line: Array, + shapedTextOrientations: ShapedTextOrientations, + shapedIcon: PositionedIcon | void, + imageMap: {[_: string]: StyleImage}, + verticallyShapedIcon: PositionedIcon | void, + layer: SymbolStyleLayer, + collisionBoxArray: CollisionBoxArray, + featureIndex: number, + sourceLayerIndex: number, + bucketIndex: number, + textBoxScale: number, + textPadding: SymbolPadding, + textAlongLine: boolean, + textOffset: [number, number], + iconBoxScale: number, + iconPadding: SymbolPadding, + iconAlongLine: boolean, + iconOffset: [number, number], + feature: SymbolFeature, + sizes: Sizes, + isSDFIcon: boolean, + canonical: CanonicalTileID, + layoutTextSize: number) { + const lineArray = bucket.addToLineVertexArray(anchor, line); + + let textCollisionFeature, iconCollisionFeature, verticalTextCollisionFeature, verticalIconCollisionFeature; + + let numIconVertices = 0; + let numVerticalIconVertices = 0; + let numHorizontalGlyphVertices = 0; + let numVerticalGlyphVertices = 0; + let placedIconSymbolIndex = -1; + let verticalPlacedIconSymbolIndex = -1; + const placedTextSymbolIndices: {[k: string]: number} = {}; + let key = murmur3(''); + + if (bucket.allowVerticalPlacement && shapedTextOrientations.vertical) { + const textRotation = layer.layout.get('text-rotate').evaluate(feature, {}, canonical); + const verticalTextRotation = textRotation + 90.0; + const verticalShaping = shapedTextOrientations.vertical; + verticalTextCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticalShaping, textBoxScale, textPadding, textAlongLine, verticalTextRotation); + + if (verticallyShapedIcon) { + verticalIconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, verticallyShapedIcon, iconBoxScale, iconPadding, textAlongLine, verticalTextRotation); + } + } + + //Place icon first, so text can have a reference to its index in the placed symbol array. + //Text symbols can lazily shift at render-time because of variable anchor placement. + //If the style specifies an `icon-text-fit` then the icon would have to shift along with it. + // For more info check `updateVariableAnchors` in `draw_symbol.js` . + if (shapedIcon) { + const iconRotate = layer.layout.get('icon-rotate').evaluate(feature, {}); + const hasIconTextFit = layer.layout.get('icon-text-fit') !== 'none'; + const iconQuads = getIconQuads(shapedIcon, iconRotate, isSDFIcon, hasIconTextFit); + const verticalIconQuads = verticallyShapedIcon ? getIconQuads(verticallyShapedIcon, iconRotate, isSDFIcon, hasIconTextFit) : undefined; + iconCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shapedIcon, iconBoxScale, iconPadding, /*align boxes to line*/false, iconRotate); + + numIconVertices = iconQuads.length * 4; + + const sizeData = bucket.iconSizeData; + let iconSizeData = null; + + if (sizeData.kind === 'source') { + iconSizeData = [ + SIZE_PACK_FACTOR * layer.layout.get('icon-size').evaluate(feature, {}) + ]; + if (iconSizeData[0] > MAX_PACKED_SIZE) { + warnOnce(`${bucket.layerIds[0]}: Value for "icon-size" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your "icon-size".`); + } + } else if (sizeData.kind === 'composite') { + iconSizeData = [ + SIZE_PACK_FACTOR * sizes.compositeIconSizes[0].evaluate(feature, {}, canonical), + SIZE_PACK_FACTOR * sizes.compositeIconSizes[1].evaluate(feature, {}, canonical) + ]; + if (iconSizeData[0] > MAX_PACKED_SIZE || iconSizeData[1] > MAX_PACKED_SIZE) { + warnOnce(`${bucket.layerIds[0]}: Value for "icon-size" is >= ${MAX_GLYPH_ICON_SIZE}. Reduce your "icon-size".`); + } + } + + bucket.addSymbols( + bucket.icon, + iconQuads, + iconSizeData, + iconOffset, + iconAlongLine, + feature, + WritingMode.none, + anchor, + lineArray.lineStartIndex, + lineArray.lineLength, + // The icon itself does not have an associated symbol since the text isnt placed yet + -1, canonical); + + placedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1; + + if (verticalIconQuads) { + numVerticalIconVertices = verticalIconQuads.length * 4; + + bucket.addSymbols( + bucket.icon, + verticalIconQuads, + iconSizeData, + iconOffset, + iconAlongLine, + feature, + WritingMode.vertical, + anchor, + lineArray.lineStartIndex, + lineArray.lineLength, + // The icon itself does not have an associated symbol since the text isnt placed yet + -1, canonical); + + verticalPlacedIconSymbolIndex = bucket.icon.placedSymbolArray.length - 1; + } + } + + const justifications = Object.keys(shapedTextOrientations.horizontal) as TextJustify[]; + for (const justification of justifications) { + const shaping = shapedTextOrientations.horizontal[justification]; + + if (!textCollisionFeature) { + key = murmur3(shaping.text); + const textRotate = layer.layout.get('text-rotate').evaluate(feature, {}, canonical); + // As a collision approximation, we can use either the vertical or any of the horizontal versions of the feature + // We're counting on all versions having similar dimensions + textCollisionFeature = new CollisionFeature(collisionBoxArray, anchor, featureIndex, sourceLayerIndex, bucketIndex, shaping, textBoxScale, textPadding, textAlongLine, textRotate); + } + + const singleLine = shaping.positionedLines.length === 1; + numHorizontalGlyphVertices += addTextVertices( + bucket, anchor, shaping, imageMap, layer, textAlongLine, feature, textOffset, lineArray, + shapedTextOrientations.vertical ? WritingMode.horizontal : WritingMode.horizontalOnly, + singleLine ? justifications : [justification], + placedTextSymbolIndices, placedIconSymbolIndex, sizes, canonical); + + if (singleLine) { + break; + } + } + + if (shapedTextOrientations.vertical) { + numVerticalGlyphVertices += addTextVertices( + bucket, anchor, shapedTextOrientations.vertical, imageMap, layer, textAlongLine, feature, + textOffset, lineArray, WritingMode.vertical, ['vertical'], placedTextSymbolIndices, verticalPlacedIconSymbolIndex, sizes, canonical); + } + + const textBoxStartIndex = textCollisionFeature ? textCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length; + const textBoxEndIndex = textCollisionFeature ? textCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length; + + const verticalTextBoxStartIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length; + const verticalTextBoxEndIndex = verticalTextCollisionFeature ? verticalTextCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length; + + const iconBoxStartIndex = iconCollisionFeature ? iconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length; + const iconBoxEndIndex = iconCollisionFeature ? iconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length; + + const verticalIconBoxStartIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxStartIndex : bucket.collisionBoxArray.length; + const verticalIconBoxEndIndex = verticalIconCollisionFeature ? verticalIconCollisionFeature.boxEndIndex : bucket.collisionBoxArray.length; + + // Check if runtime collision circles should be used for any of the collision features. + // It is enough to choose the tallest feature shape as circles are always placed on a line. + // All measurements are in glyph metrics and later converted into pixels using proper font size "layoutTextSize" + let collisionCircleDiameter = -1; + + const getCollisionCircleHeight = (feature: CollisionFeature, prevHeight: number): number => { + if (feature && feature.circleDiameter) + return Math.max(feature.circleDiameter, prevHeight); + return prevHeight; + }; + + collisionCircleDiameter = getCollisionCircleHeight(textCollisionFeature, collisionCircleDiameter); + collisionCircleDiameter = getCollisionCircleHeight(verticalTextCollisionFeature, collisionCircleDiameter); + collisionCircleDiameter = getCollisionCircleHeight(iconCollisionFeature, collisionCircleDiameter); + collisionCircleDiameter = getCollisionCircleHeight(verticalIconCollisionFeature, collisionCircleDiameter); + const useRuntimeCollisionCircles = (collisionCircleDiameter > -1) ? 1 : 0; + + // Convert circle collision height into pixels + if (useRuntimeCollisionCircles) + collisionCircleDiameter *= layoutTextSize / ONE_EM; + + if (bucket.glyphOffsetArray.length >= SymbolBucket.MAX_GLYPHS) warnOnce( + 'Too many glyphs being rendered in a tile. See https://github.com/mapbox/mapbox-gl-js/issues/2907' + ); + + if (feature.sortKey !== undefined) { + bucket.addToSortKeyRanges(bucket.symbolInstances.length, feature.sortKey as number); + } + + const variableAnchorOffset = getTextVariableAnchorOffset(layer, feature, canonical); + const [textAnchorOffsetStartIndex, textAnchorOffsetEndIndex] = addTextVariableAnchorOffsets(bucket.textAnchorOffsets, variableAnchorOffset); + + bucket.symbolInstances.emplaceBack( + anchor.x, + anchor.y, + placedTextSymbolIndices.right >= 0 ? placedTextSymbolIndices.right : -1, + placedTextSymbolIndices.center >= 0 ? placedTextSymbolIndices.center : -1, + placedTextSymbolIndices.left >= 0 ? placedTextSymbolIndices.left : -1, + placedTextSymbolIndices.vertical || -1, + placedIconSymbolIndex, + verticalPlacedIconSymbolIndex, + key, + textBoxStartIndex, + textBoxEndIndex, + verticalTextBoxStartIndex, + verticalTextBoxEndIndex, + iconBoxStartIndex, + iconBoxEndIndex, + verticalIconBoxStartIndex, + verticalIconBoxEndIndex, + featureIndex, + numHorizontalGlyphVertices, + numVerticalGlyphVertices, + numIconVertices, + numVerticalIconVertices, + useRuntimeCollisionCircles, + 0, + textBoxScale, + collisionCircleDiameter, + textAnchorOffsetStartIndex, + textAnchorOffsetEndIndex); +} + +function anchorIsTooClose(bucket: SymbolBucket, text: string, repeatDistance: number, anchor: Point) { + const compareText = bucket.compareText; + if (!(text in compareText)) { + compareText[text] = []; + } else { + const otherAnchors = compareText[text]; + for (let k = otherAnchors.length - 1; k >= 0; k--) { + if (anchor.dist(otherAnchors[k]) < repeatDistance) { + // If it's within repeatDistance of one anchor, stop looking + return true; + } + } + } + // If anchor is not within repeatDistance of any other anchor, add to array + compareText[text].push(anchor); + return false; +} diff --git a/web/libraries/maplibre-gl/src/symbol/symbol_size.ts b/web/libraries/maplibre-gl/src/symbol/symbol_size.ts new file mode 100644 index 00000000..d729e602 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/symbol_size.ts @@ -0,0 +1,129 @@ +import {Interpolate, interpolates} from '@maplibre/maplibre-gl-style-spec'; +import {clamp} from '../util/util'; +import {EvaluationParameters} from '../style/evaluation_parameters'; + +import type {PropertyValue, PossiblyEvaluatedPropertyValue} from '../style/properties'; +import type {InterpolationType} from '@maplibre/maplibre-gl-style-spec'; + +const MAX_GLYPH_ICON_SIZE = 255; +const SIZE_PACK_FACTOR = 128; +const MAX_PACKED_SIZE = MAX_GLYPH_ICON_SIZE * SIZE_PACK_FACTOR; + +export {getSizeData, evaluateSizeForFeature, evaluateSizeForZoom, SIZE_PACK_FACTOR, MAX_GLYPH_ICON_SIZE, MAX_PACKED_SIZE}; + +export type SizeData = { + kind: 'constant'; + layoutSize: number; +} | { + kind: 'source'; +} | { + kind: 'camera'; + minZoom: number; + maxZoom: number; + minSize: number; + maxSize: number; + interpolationType: InterpolationType; +} | { + kind: 'composite'; + minZoom: number; + maxZoom: number; + interpolationType: InterpolationType; +}; + +export type EvaluatedZoomSize = {uSizeT: number; uSize: number}; + +// For {text,icon}-size, get the bucket-level data that will be needed by +// the painter to set symbol-size-related uniforms +function getSizeData( + tileZoom: number, + value: PropertyValue> +): SizeData { + const {expression} = value; + + if (expression.kind === 'constant') { + const layoutSize = expression.evaluate(new EvaluationParameters(tileZoom + 1)); + return {kind: 'constant', layoutSize}; + + } else if (expression.kind === 'source') { + return {kind: 'source'}; + + } else { + const {zoomStops, interpolationType} = expression; + + // calculate covering zoom stops for zoom-dependent values + let lower = 0; + while (lower < zoomStops.length && zoomStops[lower] <= tileZoom) lower++; + lower = Math.max(0, lower - 1); + let upper = lower; + while (upper < zoomStops.length && zoomStops[upper] < tileZoom + 1) upper++; + upper = Math.min(zoomStops.length - 1, upper); + + const minZoom = zoomStops[lower]; + const maxZoom = zoomStops[upper]; + + // We'd like to be able to use CameraExpression or CompositeExpression in these + // return types rather than ExpressionSpecification, but the former are not + // transferrable across Web Worker boundaries. + if (expression.kind === 'composite') { + return {kind: 'composite', minZoom, maxZoom, interpolationType}; + } + + // for camera functions, also save off the function values + // evaluated at the covering zoom levels + const minSize = expression.evaluate(new EvaluationParameters(minZoom)); + const maxSize = expression.evaluate(new EvaluationParameters(maxZoom)); + + return {kind: 'camera', minZoom, maxZoom, minSize, maxSize, interpolationType}; + } +} + +function evaluateSizeForFeature(sizeData: SizeData, + { + uSize, + uSizeT + }: { + uSize: number; + uSizeT: number; + }, + { + lowerSize, + upperSize + }: { + lowerSize: number; + upperSize: number; + }): number { + if (sizeData.kind === 'source') { + return lowerSize / SIZE_PACK_FACTOR; + } else if (sizeData.kind === 'composite') { + return interpolates.number(lowerSize / SIZE_PACK_FACTOR, upperSize / SIZE_PACK_FACTOR, uSizeT); + } + return uSize; +} + +function evaluateSizeForZoom(sizeData: SizeData, zoom: number): EvaluatedZoomSize { + let uSizeT = 0; + let uSize = 0; + + if (sizeData.kind === 'constant') { + uSize = sizeData.layoutSize; + + } else if (sizeData.kind !== 'source') { + const {interpolationType, minZoom, maxZoom} = sizeData; + + // Even though we could get the exact value of the camera function + // at z = tr.zoom, we intentionally do not: instead, we interpolate + // between the camera function values at a pair of zoom stops covering + // [tileZoom, tileZoom + 1] in order to be consistent with this + // restriction on composite functions + const t = !interpolationType ? 0 : clamp( + Interpolate.interpolationFactor(interpolationType, zoom, minZoom, maxZoom), 0, 1); + + if (sizeData.kind === 'camera') { + uSize = interpolates.number(sizeData.minSize, sizeData.maxSize, t); + } else { + uSizeT = t; + } + } + + return {uSizeT, uSize}; +} diff --git a/web/libraries/maplibre-gl/src/symbol/symbol_style_layer.test.ts b/web/libraries/maplibre-gl/src/symbol/symbol_style_layer.test.ts new file mode 100644 index 00000000..c93c5cf7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/symbol_style_layer.test.ts @@ -0,0 +1,103 @@ +import {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer'; +import {FormatSectionOverride} from '../style/format_section_override'; +import properties, {SymbolPaintPropsPossiblyEvaluated} from '../style/style_layer/symbol_style_layer_properties.g'; +import {ZoomHistory} from '../style/zoom_history'; +import {EvaluationParameters} from '../style/evaluation_parameters'; + +function createSymbolLayer(layerProperties) { + const layer = new SymbolStyleLayer(layerProperties); + layer.recalculate({zoom: 0, zoomHistory: {} as ZoomHistory} as EvaluationParameters, []); + return layer; +} + +function isOverriden(paintProperty) { + if (paintProperty.value.kind === 'source' || paintProperty.value.kind === 'composite') { + return paintProperty.value._styleExpression.expression instanceof FormatSectionOverride; + } + return false; +} + +describe('setPaintOverrides', () => { + test('setPaintOverrides, no overrides', () => { + const layer = createSymbolLayer({}); + layer._setPaintOverrides(); + for (const overridable of properties.paint.overridableProperties) { + expect(isOverriden(layer.paint.get(overridable as keyof SymbolPaintPropsPossiblyEvaluated))).toBe(false); + } + + }); + + test('setPaintOverrides, format expression, overriden text-color', () => { + const props = {layout: {'text-field': ['format', 'text', {'text-color': 'yellow'}]}}; + const layer = createSymbolLayer(props); + layer._setPaintOverrides(); + expect(isOverriden(layer.paint.get('text-color'))).toBe(true); + + }); + + test('setPaintOverrides, format expression, no overrides', () => { + const props = {layout: {'text-field': ['format', 'text', {}]}}; + const layer = createSymbolLayer(props); + layer._setPaintOverrides(); + expect(isOverriden(layer.paint.get('text-color'))).toBe(false); + + }); + +}); + +describe('hasPaintOverrides', () => { + test('undefined', () => { + const layer = createSymbolLayer({}); + expect(SymbolStyleLayer.hasPaintOverride(layer.layout, 'text-color')).toBe(false); + + }); + + test('constant, Formatted type, overriden text-color', () => { + const props = {layout: {'text-field': ['format', 'text', {'text-color': 'red'}]}}; + const layer = createSymbolLayer(props); + expect(SymbolStyleLayer.hasPaintOverride(layer.layout, 'text-color')).toBe(true); + + }); + + test('constant, Formatted type, no overrides', () => { + const props = {layout: {'text-field': ['format', 'text', {'font-scale': 0.8}]}}; + const layer = createSymbolLayer(props); + expect(SymbolStyleLayer.hasPaintOverride(layer.layout, 'text-color')).toBe(false); + + }); + + test('format expression, overriden text-color', () => { + const props = {layout: {'text-field': ['format', ['get', 'name'], {'text-color': 'red'}]}}; + const layer = createSymbolLayer(props); + expect(SymbolStyleLayer.hasPaintOverride(layer.layout, 'text-color')).toBe(true); + + }); + + test('format expression, no overrides', () => { + const props = {layout: {'text-field': ['format', ['get', 'name'], {}]}}; + const layer = createSymbolLayer(props); + expect(SymbolStyleLayer.hasPaintOverride(layer.layout, 'text-color')).toBe(false); + + }); + + test('nested expression, overriden text-color', () => { + const matchExpr = ['match', ['get', 'case'], + 'one', ['format', 'color', {'text-color': 'blue'}], + 'default']; + const props = {layout: {'text-field': matchExpr}}; + const layer = createSymbolLayer(props); + expect(SymbolStyleLayer.hasPaintOverride(layer.layout, 'text-color')).toBe(true); + + }); + + test('nested expression, no overrides', () => { + const matchExpr = ['match', ['get', 'case'], + 'one', ['format', 'b&w', {}], + 'default']; + const props = {layout: {'text-field': matchExpr}}; + const layer = createSymbolLayer(props); + expect(SymbolStyleLayer.hasPaintOverride(layer.layout, 'text-color')).toBe(false); + + }); + +}); diff --git a/web/libraries/maplibre-gl/src/symbol/transform_text.ts b/web/libraries/maplibre-gl/src/symbol/transform_text.ts new file mode 100644 index 00000000..0278ea50 --- /dev/null +++ b/web/libraries/maplibre-gl/src/symbol/transform_text.ts @@ -0,0 +1,27 @@ +import {plugin as rtlTextPlugin} from '../source/rtl_text_plugin'; + +import type {SymbolStyleLayer} from '../style/style_layer/symbol_style_layer'; +import type {Feature} from '@maplibre/maplibre-gl-style-spec'; +import {Formatted} from '@maplibre/maplibre-gl-style-spec'; + +function transformTextInternal(text: string, layer: SymbolStyleLayer, feature: Feature) { + const transform = layer.layout.get('text-transform').evaluate(feature, {}); + if (transform === 'uppercase') { + text = text.toLocaleUpperCase(); + } else if (transform === 'lowercase') { + text = text.toLocaleLowerCase(); + } + + if (rtlTextPlugin.applyArabicShaping) { + text = rtlTextPlugin.applyArabicShaping(text); + } + + return text; +} + +export function transformText(text: Formatted, layer: SymbolStyleLayer, feature: Feature): Formatted { + text.sections.forEach(section => { + section.text = transformTextInternal(section.text, layer, feature); + }); + return text; +} diff --git a/web/libraries/maplibre-gl/src/types/callback.ts b/web/libraries/maplibre-gl/src/types/callback.ts new file mode 100644 index 00000000..e34922f2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/types/callback.ts @@ -0,0 +1,15 @@ +/** + * A callback definition. + * + * @example + * ```ts + * asyncFunction((error, result) => { + * if (error) { + * // handle error + * } else if (result) { + * // handle success + * } + * }); + * ``` + */ +export type Callback = (error?: Error | null, result?: T | null) => void; diff --git a/web/libraries/maplibre-gl/src/types/cancelable.ts b/web/libraries/maplibre-gl/src/types/cancelable.ts new file mode 100644 index 00000000..eab74130 --- /dev/null +++ b/web/libraries/maplibre-gl/src/types/cancelable.ts @@ -0,0 +1,6 @@ +/** + * A request that can be cancelled + */ +export type Cancelable = { + cancel: () => void; +}; diff --git a/web/libraries/maplibre-gl/src/types/tilejson.ts b/web/libraries/maplibre-gl/src/types/tilejson.ts new file mode 100644 index 00000000..a2542eb3 --- /dev/null +++ b/web/libraries/maplibre-gl/src/types/tilejson.ts @@ -0,0 +1,15 @@ +export type TileJSON = { + tilejson: '2.2.0' | '2.1.0' | '2.0.1' | '2.0.0' | '1.0.0'; + name?: string; + description?: string; + version?: string; + attribution?: string; + template?: string; + tiles: Array; + grids?: Array; + data?: Array; + minzoom?: number; + maxzoom?: number; + bounds?: [number, number, number, number]; + center?: [number, number, number]; +}; diff --git a/web/libraries/maplibre-gl/src/types/transferable.ts b/web/libraries/maplibre-gl/src/types/transferable.ts new file mode 100644 index 00000000..86c20270 --- /dev/null +++ b/web/libraries/maplibre-gl/src/types/transferable.ts @@ -0,0 +1,4 @@ +/** + * An object with is transferable between main and worker thread + */ +export type Transferable = ArrayBuffer | MessagePort | ImageBitmap; diff --git a/web/libraries/maplibre-gl/src/ui/anchor.ts b/web/libraries/maplibre-gl/src/ui/anchor.ts new file mode 100644 index 00000000..be6c49ec --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/anchor.ts @@ -0,0 +1,27 @@ +/** + * Where to position the anchor. + * Used by a popup and a marker. + */ +export type PositionAnchor = 'center' | 'top' | 'bottom' | 'left' | 'right' | 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right'; + +export const anchorTranslate: { + [_ in PositionAnchor]: string; +} = { + 'center': 'translate(-50%,-50%)', + 'top': 'translate(-50%,0)', + 'top-left': 'translate(0,0)', + 'top-right': 'translate(-100%,0)', + 'bottom': 'translate(-50%,-100%)', + 'bottom-left': 'translate(0,-100%)', + 'bottom-right': 'translate(-100%,-100%)', + 'left': 'translate(0,-50%)', + 'right': 'translate(-100%,-50%)' +}; + +export function applyAnchorClass(element: HTMLElement, anchor: PositionAnchor, prefix: string) { + const classList = element.classList; + for (const key in anchorTranslate) { + classList.remove(`maplibregl-${prefix}-anchor-${key}`); + } + classList.add(`maplibregl-${prefix}-anchor-${anchor}`); +} diff --git a/web/libraries/maplibre-gl/src/ui/camera.test.ts b/web/libraries/maplibre-gl/src/ui/camera.test.ts new file mode 100644 index 00000000..77363859 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/camera.test.ts @@ -0,0 +1,2240 @@ +import {Camera, CameraOptions} from '../ui/camera'; +import {Transform} from '../geo/transform'; +import {TaskQueue, TaskID} from '../util/task_queue'; +import {browser} from '../util/browser'; +import {fixedLngLat, fixedNum} from '../../test/unit/lib/fixed'; +import {setMatchMedia} from '../util/test/util'; +import {mercatorZfromAltitude} from '../geo/mercator_coordinate'; +import {Terrain} from '../render/terrain'; +import {LngLat, LngLatLike} from '../geo/lng_lat'; +import {Event} from '../util/evented'; + +beforeEach(() => { + setMatchMedia(); + Object.defineProperty(browser, 'prefersReducedMotion', {value: false}); +}); + +class CameraMock extends Camera { + // eslint-disable-next-line + _requestRenderFrame(a: () => void): TaskID { + return undefined; + } + + _cancelRenderFrame(_: TaskID): void { + return undefined; + } +} + +function attachSimulateFrame(camera) { + const queue = new TaskQueue(); + camera._requestRenderFrame = (cb) => queue.add(cb); + camera._cancelRenderFrame = (id) => queue.remove(id); + camera.simulateFrame = () => queue.run(); + return camera; +} + +function createCamera(options?) { + options = options || {}; + + const transform = new Transform(0, 20, 0, 60, options.renderWorldCopies); + transform.resize(512, 512); + + const camera = attachSimulateFrame(new CameraMock(transform, {} as any)) + .jumpTo(options); + + camera._update = () => {}; + + return camera; +} + +function assertTransitionTime(done, camera, min, max) { + let startTime; + camera + .on('movestart', () => { startTime = new Date(); }) + .on('moveend', () => { + const endTime = new Date(); + const timeDiff = endTime.getTime() - startTime.getTime(); + expect(timeDiff >= min && timeDiff < max).toBeTruthy(); + done(); + }); +} + +describe('#calculateCameraOptionsFromTo', () => { + // Choose initial zoom to avoid center being constrained by mercator latitude limits. + const camera = createCamera({zoom: 1}); + + test('look at north', () => { + const cameraOptions: CameraOptions = camera.calculateCameraOptionsFromTo({lng: 1, lat: 0}, 0, {lng: 1, lat: 1}); + expect(cameraOptions).toBeDefined(); + expect(cameraOptions.center).toBeDefined(); + expect(cameraOptions.bearing).toBeCloseTo(0); + }); + + test('look at west', () => { + const cameraOptions = camera.calculateCameraOptionsFromTo({lng: 1, lat: 0}, 0, {lng: 0, lat: 0}); + expect(cameraOptions).toBeDefined(); + expect(cameraOptions.bearing).toBeCloseTo(-90); + }); + + test('pitch 45', () => { + // altitude same as grounddistance => 45° + // distance between lng x and lng x+1 is 111.2km at same lat + const cameraOptions: CameraOptions = camera.calculateCameraOptionsFromTo({lng: 1, lat: 0}, 111200, {lng: 0, lat: 0}); + expect(cameraOptions).toBeDefined(); + expect(cameraOptions.pitch).toBeCloseTo(45); + }); + + test('pitch 90', () => { + const cameraOptions = camera.calculateCameraOptionsFromTo({lng: 1, lat: 0}, 0, {lng: 0, lat: 0}); + expect(cameraOptions).toBeDefined(); + expect(cameraOptions.pitch).toBeCloseTo(90); + }); + + test('pitch 153.435', () => { + + // distance between lng x and lng x+1 is 111.2km at same lat + // (elevation difference of cam and center) / 2 = grounddistance => + // acos(111.2 / sqrt(111.2² + (111.2 * 2)²)) = acos(1/sqrt(5)) => 63.435 + 90 (looking up) = 153.435 + const cameraOptions: CameraOptions = camera.calculateCameraOptionsFromTo({lng: 1, lat: 0}, 111200, {lng: 0, lat: 0}, 111200 * 3); + expect(cameraOptions).toBeDefined(); + expect(cameraOptions.pitch).toBeCloseTo(153.435); + }); + + test('zoom distance 1000', () => { + const expectedZoom = Math.log2(camera.transform.cameraToCenterDistance / mercatorZfromAltitude(1000, 0) / camera.transform.tileSize); + const cameraOptions = camera.calculateCameraOptionsFromTo({lng: 0, lat: 0}, 0, {lng: 0, lat: 0}, 1000); + + expect(cameraOptions).toBeDefined(); + expect(cameraOptions.zoom).toBeCloseTo(expectedZoom); + }); + + test('zoom distance 1 lng (111.2km), 111.2km altitude away', () => { + const expectedZoom = Math.log2(camera.transform.cameraToCenterDistance / mercatorZfromAltitude(Math.hypot(111200, 111200), 0) / camera.transform.tileSize); + const cameraOptions = camera.calculateCameraOptionsFromTo({lng: 0, lat: 0}, 0, {lng: 1, lat: 0}, 111200); + + expect(cameraOptions).toBeDefined(); + expect(cameraOptions.zoom).toBeCloseTo(expectedZoom); + }); + + test('same To as From error', () => { + expect(() => { camera.calculateCameraOptionsFromTo({lng: 0, lat: 0}, 0, {lng: 0, lat: 0}, 0); }).toThrow(); + }); +}); + +describe('#jumpTo', () => { + // Choose initial zoom to avoid center being constrained by mercator latitude limits. + const camera = createCamera({zoom: 1}); + + test('sets center', () => { + camera.jumpTo({center: [1, 2]}); + expect(camera.getCenter()).toEqual({lng: 1, lat: 2}); + }); + + test('throws on invalid center argument', () => { + expect(() => { + camera.jumpTo({center: 1}); + }).toThrow(Error); + }); + + test('keeps current center if not specified', () => { + camera.jumpTo({}); + expect(camera.getCenter()).toEqual({lng: 1, lat: 2}); + }); + + test('sets zoom', () => { + camera.jumpTo({zoom: 3}); + expect(camera.getZoom()).toBe(3); + }); + + test('keeps current zoom if not specified', () => { + camera.jumpTo({}); + expect(camera.getZoom()).toBe(3); + }); + + test('sets bearing', () => { + camera.jumpTo({bearing: 4}); + expect(camera.getBearing()).toBe(4); + }); + + test('keeps current bearing if not specified', () => { + camera.jumpTo({}); + expect(camera.getBearing()).toBe(4); + }); + + test('sets pitch', () => { + camera.jumpTo({pitch: 45}); + expect(camera.getPitch()).toBe(45); + }); + + test('keeps current pitch if not specified', () => { + camera.jumpTo({}); + expect(camera.getPitch()).toBe(45); + }); + + test('sets multiple properties', () => { + camera.jumpTo({ + center: [10, 20], + zoom: 10, + bearing: 180, + pitch: 60 + }); + expect(camera.getCenter()).toEqual({lng: 10, lat: 20}); + expect(camera.getZoom()).toBe(10); + expect(camera.getBearing()).toBe(180); + expect(camera.getPitch()).toBe(60); + }); + + test('emits move events, preserving eventData', done => { + let started, moved, ended; + const eventData = {data: 'ok'}; + + camera + .on('movestart', (d) => { started = d.data; }) + .on('move', (d) => { moved = d.data; }) + .on('moveend', (d) => { ended = d.data; }); + + camera.jumpTo({center: [1, 2]}, eventData); + expect(started).toBe('ok'); + expect(moved).toBe('ok'); + expect(ended).toBe('ok'); + done(); + }); + + test('emits zoom events, preserving eventData', done => { + let started, zoomed, ended; + const eventData = {data: 'ok'}; + + camera + .on('zoomstart', (d) => { started = d.data; }) + .on('zoom', (d) => { zoomed = d.data; }) + .on('zoomend', (d) => { ended = d.data; }); + + camera.jumpTo({zoom: 3}, eventData); + expect(started).toBe('ok'); + expect(zoomed).toBe('ok'); + expect(ended).toBe('ok'); + done(); + }); + + test('emits rotate events, preserving eventData', done => { + let started, rotated, ended; + const eventData = {data: 'ok'}; + + camera + .on('rotatestart', (d) => { started = d.data; }) + .on('rotate', (d) => { rotated = d.data; }) + .on('rotateend', (d) => { ended = d.data; }); + + camera.jumpTo({bearing: 90}, eventData); + expect(started).toBe('ok'); + expect(rotated).toBe('ok'); + expect(ended).toBe('ok'); + done(); + }); + + test('emits pitch events, preserving eventData', done => { + let started, pitched, ended; + const eventData = {data: 'ok'}; + + camera + .on('pitchstart', (d) => { started = d.data; }) + .on('pitch', (d) => { pitched = d.data; }) + .on('pitchend', (d) => { ended = d.data; }); + + camera.jumpTo({pitch: 10}, eventData); + expect(started).toBe('ok'); + expect(pitched).toBe('ok'); + expect(ended).toBe('ok'); + done(); + }); + + test('cancels in-progress easing', () => { + camera.panTo([3, 4]); + expect(camera.isEasing()).toBeTruthy(); + camera.jumpTo({center: [1, 2]}); + expect(!camera.isEasing()).toBeTruthy(); + }); +}); + +describe('#setCenter', () => { + // Choose initial zoom to avoid center being constrained by mercator latitude limits. + const camera = createCamera({zoom: 1}); + + test('sets center', () => { + camera.setCenter([1, 2]); + expect(camera.getCenter()).toEqual({lng: 1, lat: 2}); + }); + + test('throws on invalid center argument', () => { + expect(() => { + camera.jumpTo({center: 1}); + }).toThrow(Error); + }); + + test('emits move events, preserving eventData', done => { + let started, moved, ended; + const eventData = {data: 'ok'}; + + camera.on('movestart', (d) => { started = d.data; }) + .on('move', (d) => { moved = d.data; }) + .on('moveend', (d) => { ended = d.data; }); + + camera.setCenter([10, 20], eventData); + expect(started).toBe('ok'); + expect(moved).toBe('ok'); + expect(ended).toBe('ok'); + done(); + }); + + test('cancels in-progress easing', () => { + camera.panTo([3, 4]); + expect(camera.isEasing()).toBeTruthy(); + camera.setCenter([1, 2]); + expect(!camera.isEasing()).toBeTruthy(); + }); +}); + +describe('#setZoom', () => { + const camera = createCamera(); + + test('sets zoom', () => { + camera.setZoom(3); + expect(camera.getZoom()).toBe(3); + }); + + test('emits move and zoom events, preserving eventData', done => { + let movestarted, moved, moveended, zoomstarted, zoomed, zoomended; + const eventData = {data: 'ok'}; + + camera + .on('movestart', (d) => { movestarted = d.data; }) + .on('move', (d) => { moved = d.data; }) + .on('moveend', (d) => { moveended = d.data; }) + .on('zoomstart', (d) => { zoomstarted = d.data; }) + .on('zoom', (d) => { zoomed = d.data; }) + .on('zoomend', (d) => { zoomended = d.data; }); + + camera.setZoom(4, eventData); + expect(movestarted).toBe('ok'); + expect(moved).toBe('ok'); + expect(moveended).toBe('ok'); + expect(zoomstarted).toBe('ok'); + expect(zoomed).toBe('ok'); + expect(zoomended).toBe('ok'); + done(); + }); + + test('cancels in-progress easing', () => { + camera.panTo([3, 4]); + expect(camera.isEasing()).toBeTruthy(); + camera.setZoom(5); + expect(!camera.isEasing()).toBeTruthy(); + }); +}); + +describe('#setBearing', () => { + const camera = createCamera(); + + test('sets bearing', () => { + camera.setBearing(4); + expect(camera.getBearing()).toBe(4); + }); + + test('emits move and rotate events, preserving eventData', done => { + let movestarted, moved, moveended, rotatestarted, rotated, rotateended; + const eventData = {data: 'ok'}; + + camera + .on('movestart', (d) => { movestarted = d.data; }) + .on('move', (d) => { moved = d.data; }) + .on('moveend', (d) => { moveended = d.data; }) + .on('rotatestart', (d) => { rotatestarted = d.data; }) + .on('rotate', (d) => { rotated = d.data; }) + .on('rotateend', (d) => { rotateended = d.data; }); + + camera.setBearing(5, eventData); + expect(movestarted).toBe('ok'); + expect(moved).toBe('ok'); + expect(moveended).toBe('ok'); + expect(rotatestarted).toBe('ok'); + expect(rotated).toBe('ok'); + expect(rotateended).toBe('ok'); + done(); + }); + + test('cancels in-progress easing', () => { + camera.panTo([3, 4]); + expect(camera.isEasing()).toBeTruthy(); + camera.setBearing(6); + expect(!camera.isEasing()).toBeTruthy(); + }); +}); + +describe('#setPadding', () => { + test('sets padding', () => { + const camera = createCamera(); + const padding = {left: 300, top: 100, right: 50, bottom: 10}; + camera.setPadding(padding); + expect(camera.getPadding()).toEqual(padding); + }); + + test('existing padding is retained if no new values are passed in', () => { + const camera = createCamera(); + const padding = {left: 300, top: 100, right: 50, bottom: 10}; + camera.setPadding(padding); + camera.setPadding({}); + + const currentPadding = camera.getPadding(); + expect(currentPadding).toEqual(padding); + }); + + test('doesnt change padding thats already present if new value isnt passed in', () => { + const camera = createCamera(); + const padding = {left: 300, top: 100, right: 50, bottom: 10}; + camera.setPadding(padding); + const padding1 = {right: 100}; + camera.setPadding(padding1); + + const currentPadding = camera.getPadding(); + expect(currentPadding.left).toBe(padding.left); + expect(currentPadding.top).toBe(padding.top); + // padding1 here + expect(currentPadding.right).toBe(padding1.right); + expect(currentPadding.bottom).toBe(padding.bottom); + }); +}); + +describe('#panBy', () => { + test('pans by specified amount', () => { + const camera = createCamera(); + camera.panBy([100, 0], {duration: 0}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 70.3125, lat: 0}); + }); + + test('pans relative to viewport on a rotated camera', () => { + const camera = createCamera({bearing: 180}); + camera.panBy([100, 0], {duration: 0}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: -70.3125, lat: 0}); + }); + + test('emits move events, preserving eventData', done => { + const camera = createCamera(); + let started, moved; + const eventData = {data: 'ok'}; + + camera + .on('movestart', (d) => { started = d.data; }) + .on('move', (d) => { moved = d.data; }) + .on('moveend', (d) => { + expect(started).toBe('ok'); + expect(moved).toBe('ok'); + expect(d.data).toBe('ok'); + done(); + }); + + camera.panBy([100, 0], {duration: 0}, eventData); + }); + + test('supresses movestart if noMoveStart option is true', done => { + const camera = createCamera(); + let started; + + // fire once in advance to satisfy assertions that moveend only comes after movestart + camera.fire('movestart'); + + camera + .on('movestart', () => { started = true; }) + .on('moveend', () => { + expect(!started).toBeTruthy(); + done(); + }); + + camera.panBy([100, 0], {duration: 0, noMoveStart: true}); + }); +}); + +describe('#panTo', () => { + test('pans to specified location', () => { + const camera = createCamera(); + camera.panTo([100, 0], {duration: 0}); + expect(camera.getCenter()).toEqual({lng: 100, lat: 0}); + }); + + test('throws on invalid center argument', () => { + const camera = createCamera(); + expect(() => { + camera.panTo({center: 1}); + }).toThrow(Error); + }); + + test('pans with specified offset', () => { + const camera = createCamera(); + camera.panTo([100, 0], {offset: [100, 0], duration: 0}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 29.6875, lat: 0}); + }); + + test('pans with specified offset relative to viewport on a rotated camera', () => { + const camera = createCamera({bearing: 180}); + camera.panTo([100, 0], {offset: [100, 0], duration: 0}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 170.3125, lat: 0}); + }); + + test('emits move events, preserving eventData', done => { + const camera = createCamera(); + let started, moved; + const eventData = {data: 'ok'}; + + camera + .on('movestart', (d) => { started = d.data; }) + .on('move', (d) => { moved = d.data; }) + .on('moveend', (d) => { + expect(started).toBe('ok'); + expect(moved).toBe('ok'); + expect(d.data).toBe('ok'); + done(); + }); + + camera.panTo([100, 0], {duration: 0}, eventData); + }); + + test('supresses movestart if noMoveStart option is true', done => { + const camera = createCamera(); + let started; + + // fire once in advance to satisfy assertions that moveend only comes after movestart + camera.fire('movestart'); + + camera + .on('movestart', () => { started = true; }) + .on('moveend', () => { + expect(!started).toBeTruthy(); + done(); + }); + + camera.panTo([100, 0], {duration: 0, noMoveStart: true}); + }); +}); + +describe('#zoomTo', () => { + test('zooms to specified level', () => { + const camera = createCamera(); + camera.zoomTo(3.2, {duration: 0}); + expect(camera.getZoom()).toBe(3.2); + }); + + test('zooms around specified location', () => { + const camera = createCamera(); + camera.zoomTo(3.2, {around: [5, 0], duration: 0}); + expect(camera.getZoom()).toBe(3.2); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: 4.455905897939886, lat: 0})); + }); + + test('zooms with specified offset', () => { + const camera = createCamera(); + camera.zoomTo(3.2, {offset: [100, 0], duration: 0}); + expect(camera.getZoom()).toBe(3.2); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: 62.66117668978015, lat: 0})); + }); + + test('zooms with specified offset relative to viewport on a rotated camera', () => { + const camera = createCamera({bearing: 180}); + camera.zoomTo(3.2, {offset: [100, 0], duration: 0}); + expect(camera.getZoom()).toBe(3.2); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: -62.66117668978012, lat: 0})); + }); + + test('emits move and zoom events, preserving eventData', done => { + const camera = createCamera(); + let movestarted, moved, zoomstarted, zoomed; + const eventData = {data: 'ok'}; + + expect.assertions(6); + + camera + .on('movestart', (d) => { movestarted = d.data; }) + .on('move', (d) => { moved = d.data; }) + .on('moveend', (d) => { + expect(movestarted).toBe('ok'); + expect(moved).toBe('ok'); + expect(d.data).toBe('ok'); + }); + + camera + .on('zoomstart', (d) => { zoomstarted = d.data; }) + .on('zoom', (d) => { zoomed = d.data; }) + .on('zoomend', (d) => { + expect(zoomstarted).toBe('ok'); + expect(zoomed).toBe('ok'); + expect(d.data).toBe('ok'); + }); + + camera.zoomTo(5, {duration: 0}, eventData); + done(); + }); +}); + +describe('#rotateTo', () => { + test('rotates to specified bearing', () => { + const camera = createCamera(); + camera.rotateTo(90, {duration: 0}); + expect(camera.getBearing()).toBe(90); + }); + + test('rotates around specified location', () => { + const camera = createCamera({zoom: 3}); + camera.rotateTo(90, {around: [5, 0], duration: 0}); + expect(camera.getBearing()).toBe(90); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: 4.999999999999972, lat: 4.993665859353271})); + }); + + test('rotates around specified location, constrained to fit the view', () => { + const camera = createCamera({zoom: 0}); + camera.rotateTo(90, {around: [5, 0], duration: 0}); + expect(camera.getBearing()).toBe(90); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: 4.999999999999972, lat: 0.000002552471840999715})); + }); + + test('rotates with specified offset', () => { + const camera = createCamera({zoom: 1}); + camera.rotateTo(90, {offset: [200, 0], duration: 0}); + expect(camera.getBearing()).toBe(90); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: 70.3125, lat: 57.3265212252})); + }); + + test('rotates with specified offset, constrained to fit the view', () => { + const camera = createCamera({zoom: 0}); + camera.rotateTo(90, {offset: [100, 0], duration: 0}); + expect(camera.getBearing()).toBe(90); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: 70.3125, lat: 0.000002552471840999715})); + }); + + test('rotates with specified offset relative to viewport on a rotated camera', () => { + const camera = createCamera({bearing: 180, zoom: 1}); + camera.rotateTo(90, {offset: [200, 0], duration: 0}); + expect(camera.getBearing()).toBe(90); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: -70.3125, lat: 57.3265212252})); + }); + + test('emits move and rotate events, preserving eventData', done => { + const camera = createCamera(); + let movestarted, moved, rotatestarted, rotated; + const eventData = {data: 'ok'}; + + expect.assertions(6); + + camera + .on('movestart', (d) => { movestarted = d.data; }) + .on('move', (d) => { moved = d.data; }) + .on('moveend', (d) => { + expect(movestarted).toBe('ok'); + expect(moved).toBe('ok'); + expect(d.data).toBe('ok'); + }); + + camera + .on('rotatestart', (d) => { rotatestarted = d.data; }) + .on('rotate', (d) => { rotated = d.data; }) + .on('rotateend', (d) => { + expect(rotatestarted).toBe('ok'); + expect(rotated).toBe('ok'); + expect(d.data).toBe('ok'); + }); + + camera.rotateTo(90, {duration: 0}, eventData); + done(); + }); +}); + +describe('#easeTo', () => { + test('pans to specified location', () => { + const camera = createCamera(); + camera.easeTo({center: [100, 0], duration: 0}); + expect(camera.getCenter()).toEqual({lng: 100, lat: 0}); + }); + + test('zooms to specified level', () => { + const camera = createCamera(); + camera.easeTo({zoom: 3.2, duration: 0}); + expect(camera.getZoom()).toBe(3.2); + }); + + test('rotates to specified bearing', () => { + const camera = createCamera(); + camera.easeTo({bearing: 90, duration: 0}); + expect(camera.getBearing()).toBe(90); + }); + + test('pitches to specified pitch', () => { + const camera = createCamera(); + camera.easeTo({pitch: 45, duration: 0}); + expect(camera.getPitch()).toBe(45); + }); + + test('pans and zooms', () => { + const camera = createCamera(); + camera.easeTo({center: [100, 0], zoom: 3.2, duration: 0}); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: 100, lat: 0})); + expect(camera.getZoom()).toBe(3.2); + }); + + test('zooms around a point', () => { + const camera = createCamera(); + camera.easeTo({around: [100, 0], zoom: 3, duration: 0}); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: 87.5, lat: 0})); + expect(camera.getZoom()).toBe(3); + }); + + test('pans and rotates', () => { + const camera = createCamera(); + camera.easeTo({center: [100, 0], bearing: 90, duration: 0}); + expect(camera.getCenter()).toEqual({lng: 100, lat: 0}); + expect(camera.getBearing()).toBe(90); + }); + + test('zooms and rotates', () => { + const camera = createCamera(); + camera.easeTo({zoom: 3.2, bearing: 90, duration: 0}); + expect(camera.getZoom()).toBe(3.2); + expect(camera.getBearing()).toBe(90); + }); + + test('pans, zooms, and rotates', () => { + const camera = createCamera({bearing: -90}); + camera.easeTo({center: [100, 0], zoom: 3.2, bearing: 90, duration: 0}); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: 100, lat: 0})); + expect(camera.getZoom()).toBe(3.2); + expect(camera.getBearing()).toBe(90); + }); + + test('noop', () => { + const camera = createCamera(); + camera.easeTo({duration: 0}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 0, lat: 0}); + expect(camera.getZoom()).toBe(0); + expect(camera.getBearing()).toBeCloseTo(0); + }); + + test('noop with offset', () => { + const camera = createCamera(); + camera.easeTo({offset: [100, 0], duration: 0}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 0, lat: 0}); + expect(camera.getZoom()).toBe(0); + expect(camera.getBearing()).toBeCloseTo(0); + }); + + test('pans with specified offset', () => { + const camera = createCamera(); + camera.easeTo({center: [100, 0], offset: [100, 0], duration: 0}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 29.6875, lat: 0}); + }); + + test('pans with specified offset relative to viewport on a rotated camera', () => { + const camera = createCamera({bearing: 180}); + camera.easeTo({center: [100, 0], offset: [100, 0], duration: 0}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 170.3125, lat: 0}); + }); + + test('zooms with specified offset', () => { + const camera = createCamera(); + camera.easeTo({zoom: 3.2, offset: [100, 0], duration: 0}); + expect(camera.getZoom()).toBe(3.2); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: 62.66117668978015, lat: 0})); + }); + + test('zooms with specified offset relative to viewport on a rotated camera', () => { + const camera = createCamera({bearing: 180}); + camera.easeTo({zoom: 3.2, offset: [100, 0], duration: 0}); + expect(camera.getZoom()).toBe(3.2); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: -62.66117668978012, lat: 0})); + }); + + test('rotates with specified offset', () => { + const camera = createCamera(); + camera.easeTo({bearing: 90, offset: [100, 0], duration: 0}); + expect(camera.getBearing()).toBe(90); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: 70.3125, lat: 0.000002552471840999715})); + }); + + test('rotates with specified offset relative to viewport on a rotated camera', () => { + const camera = createCamera({bearing: 180}); + camera.easeTo({bearing: 90, offset: [100, 0], duration: 0}); + expect(camera.getBearing()).toBe(90); + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat({lng: -70.3125, lat: 0.000002552471840999715})); + }); + + test('emits move, zoom, rotate, and pitch events, preserving eventData', done => { + const camera = createCamera(); + let movestarted, moved, zoomstarted, zoomed, rotatestarted, rotated, pitchstarted, pitched; + const eventData = {data: 'ok'}; + + expect.assertions(18); + + camera + .on('movestart', (d) => { movestarted = d.data; }) + .on('move', (d) => { moved = d.data; }) + .on('moveend', (d) => { + expect(camera._zooming).toBeFalsy(); + expect(camera._panning).toBeFalsy(); + expect(camera._rotating).toBeFalsy(); + + expect(movestarted).toBe('ok'); + expect(moved).toBe('ok'); + expect(zoomed).toBe('ok'); + expect(rotated).toBe('ok'); + expect(pitched).toBe('ok'); + expect(d.data).toBe('ok'); + }); + + camera + .on('zoomstart', (d) => { zoomstarted = d.data; }) + .on('zoom', (d) => { zoomed = d.data; }) + .on('zoomend', (d) => { + expect(zoomstarted).toBe('ok'); + expect(zoomed).toBe('ok'); + expect(d.data).toBe('ok'); + }); + + camera + .on('rotatestart', (d) => { rotatestarted = d.data; }) + .on('rotate', (d) => { rotated = d.data; }) + .on('rotateend', (d) => { + expect(rotatestarted).toBe('ok'); + expect(rotated).toBe('ok'); + expect(d.data).toBe('ok'); + }); + + camera + .on('pitchstart', (d) => { pitchstarted = d.data; }) + .on('pitch', (d) => { pitched = d.data; }) + .on('pitchend', (d) => { + expect(pitchstarted).toBe('ok'); + expect(pitched).toBe('ok'); + expect(d.data).toBe('ok'); + }); + + camera.easeTo( + {center: [100, 0], zoom: 3.2, bearing: 90, duration: 0, pitch: 45}, + eventData); + done(); + }); + + test('does not emit zoom events if not zooming', done => { + const camera = createCamera(); + + camera + .on('zoomstart', () => { done('zoomstart failed'); }) + .on('zoom', () => { done('zoom failed'); }) + .on('zoomend', () => { done('zoomend failed'); }) + .on('moveend', () => { done(); }); + + camera.easeTo({center: [100, 0], duration: 0}); + }); + + test('stops existing ease', () => { + const camera = createCamera(); + camera.easeTo({center: [200, 0], duration: 100}); + camera.easeTo({center: [100, 0], duration: 0}); + expect(camera.getCenter()).toEqual({lng: 100, lat: 0}); + }); + + test('can be called from within a moveend event handler', done => { + const camera = createCamera(); + const stub = jest.spyOn(browser, 'now'); + + stub.mockImplementation(() => 0); + camera.easeTo({center: [100, 0], duration: 10}); + + camera.once('moveend', () => { + camera.easeTo({center: [200, 0], duration: 10}); + camera.once('moveend', () => { + camera.easeTo({center: [300, 0], duration: 10}); + camera.once('moveend', () => { + done(); + }); + + setTimeout(() => { + stub.mockImplementation(() => 30); + camera.simulateFrame(); + }, 0); + }); + + // setTimeout to avoid a synchronous callback + setTimeout(() => { + stub.mockImplementation(() => 20); + camera.simulateFrame(); + }, 0); + }); + + // setTimeout to avoid a synchronous callback + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }); + + test('pans eastward across the antimeridian', done => { + const camera = createCamera(); + const stub = jest.spyOn(browser, 'now'); + + camera.setCenter([170, 0]); + let crossedAntimeridian; + + camera.on('move', () => { + if (camera.getCenter().lng > 170) { + crossedAntimeridian = true; + } + }); + + camera.on('moveend', () => { + expect(crossedAntimeridian).toBeTruthy(); + done(); + }); + + stub.mockImplementation(() => 0); + camera.easeTo({center: [-170, 0], duration: 10}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('pans westward across the antimeridian', done => { + const camera = createCamera(); + const stub = jest.spyOn(browser, 'now'); + + camera.setCenter([-170, 0]); + let crossedAntimeridian; + + camera.on('move', () => { + if (camera.getCenter().lng < -170) { + crossedAntimeridian = true; + } + }); + + camera.on('moveend', () => { + expect(crossedAntimeridian).toBeTruthy(); + done(); + }); + + stub.mockImplementation(() => 0); + camera.easeTo({center: [170, 0], duration: 10}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('animation occurs when prefers-reduced-motion: reduce is set but overridden by essential: true', done => { + const camera = createCamera(); + Object.defineProperty(browser, 'prefersReducedMotion', {value: true}); + const stubNow = jest.spyOn(browser, 'now'); + + // camera transition expected to take in this range when prefersReducedMotion is set and essential: true, + // when a duration of 200 is requested + const min = 100; + const max = 300; + + let startTime; + camera + .on('movestart', () => { startTime = browser.now(); }) + .on('moveend', () => { + const endTime = browser.now(); + const timeDiff = endTime - startTime; + expect(timeDiff >= min && timeDiff < max).toBeTruthy(); + done(); + }); + + setTimeout(() => { + stubNow.mockImplementation(() => 0); + camera.simulateFrame(); + + camera.easeTo({center: [100, 0], zoom: 3.2, bearing: 90, duration: 200, essential: true}); + + setTimeout(() => { + stubNow.mockImplementation(() => 200); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('duration is 0 when prefers-reduced-motion: reduce is set', done => { + const camera = createCamera(); + Object.defineProperty(browser, 'prefersReducedMotion', {value: true}); + assertTransitionTime(done, camera, 0, 10); + camera.easeTo({center: [100, 0], zoom: 3.2, bearing: 90, duration: 1000}); + }); + + test('jumpTo on("move") during easeTo with zoom, pitch, etc', (done) => { + const camera = createCamera(); + + camera.on('moveend', (e: Event & {done?: true}) => { + if ('done' in e) { + setTimeout(() => { + done(); + }, 50); + } + }); + + camera.easeTo({zoom: 20, bearing: 90, pitch: 60, duration: 500}, {done: true}); + camera.once('move', () => { + camera.jumpTo({pitch: 40}); + }); + + camera.simulateFrame(); + camera.simulateFrame(); + }); + + test('jumpTo on("zoom") during easeTo', (done) => { + const camera = createCamera(); + + camera.on('moveend', (e: Event & {done?: true}) => { + if ('done' in e) { + setTimeout(() => { + done(); + }, 50); + } + }); + + camera.easeTo({zoom: 20, duration: 500}, {done: true}); + camera.once('zoom', () => { + camera.jumpTo({pitch: 40}); + }); + + camera.simulateFrame(); + camera.simulateFrame(); + }); + + test('jumpTo on("pitch") during easeTo', (done) => { + const camera = createCamera(); + + camera.on('moveend', (e: Event & {done?: true}) => { + if ('done' in e) { + setTimeout(() => { + done(); + }, 50); + } + }); + + camera.easeTo({pitch: 60, duration: 500}, {done: true}); + camera.once('pitch', () => { + camera.jumpTo({pitch: 40}); + }); + + camera.simulateFrame(); + camera.simulateFrame(); + }); + + test('jumpTo on("rotate") during easeTo', (done) => { + const camera = createCamera(); + + camera.on('moveend', (e: Event & {done?: true}) => { + if ('done' in e) { + setTimeout(() => { + done(); + }, 50); + } + }); + + camera.easeTo({bearing: 90, duration: 500}, {done: true}); + camera.once('rotate', () => { + camera.jumpTo({pitch: 40}); + }); + + camera.simulateFrame(); + camera.simulateFrame(); + }); +}); + +describe('#flyTo', () => { + test('pans to specified location', () => { + const camera = createCamera(); + camera.flyTo({center: [100, 0], animate: false}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 100, lat: 0}); + }); + + test('throws on invalid center argument', () => { + const camera = createCamera(); + expect(() => { + camera.flyTo({center: 1}); + }).toThrow(Error); + }); + + test('does not throw when cameras current zoom is sufficiently greater than passed zoom option', () => { + const camera = createCamera({zoom: 22, center: [0, 0]}); + expect(() => camera.flyTo({zoom: 10, center: [0, 0]})).not.toThrow(); + }); + + test('does not throw when cameras current zoom is above maxzoom and an offset creates infinite zoom out factor', () => { + const transform = new Transform(0, 20.9999, 0, 60, true); + transform.resize(512, 512); + const camera = attachSimulateFrame(new CameraMock(transform, {} as any)) + .jumpTo({zoom: 21, center: [0, 0]}); + camera._update = () => {}; + expect(() => camera.flyTo({zoom: 7.5, center: [0, 0], offset: [0, 70]})).not.toThrow(); + }); + + test('zooms to specified level', () => { + const camera = createCamera(); + camera.flyTo({zoom: 3.2, animate: false}); + expect(fixedNum(camera.getZoom())).toBe(3.2); + }); + + test('zooms to integer level without floating point errors', () => { + const camera = createCamera({zoom: 0.6}); + camera.flyTo({zoom: 2, animate: false}); + expect(camera.getZoom()).toBe(2); + }); + + test('Zoom out from the same position to the same position with animation', done => { + const pos = {lng: 0, lat: 0}; + const camera = createCamera({zoom: 20, center: pos}); + const stub = jest.spyOn(browser, 'now'); + + camera.once('zoomend', () => { + expect(fixedLngLat(camera.getCenter())).toEqual(fixedLngLat(pos)); + expect(camera.getZoom()).toBe(19); + done(); + }); + + stub.mockImplementation(() => 0); + camera.flyTo({zoom: 19, center: pos, duration: 2}); + + stub.mockImplementation(() => 3); + camera.simulateFrame(); + }); + + test('rotates to specified bearing', () => { + const camera = createCamera(); + camera.flyTo({bearing: 90, animate: false}); + expect(camera.getBearing()).toBe(90); + }); + + test('tilts to specified pitch', () => { + const camera = createCamera(); + camera.flyTo({pitch: 45, animate: false}); + expect(camera.getPitch()).toBe(45); + }); + + test('pans and zooms', () => { + const camera = createCamera(); + camera.flyTo({center: [100, 0], zoom: 3.2, animate: false}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 100, lat: 0}); + expect(fixedNum(camera.getZoom())).toBe(3.2); + }); + + test('pans and rotates', () => { + const camera = createCamera(); + camera.flyTo({center: [100, 0], bearing: 90, animate: false}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 100, lat: 0}); + expect(camera.getBearing()).toBe(90); + }); + + test('zooms and rotates', () => { + const camera = createCamera(); + camera.flyTo({zoom: 3.2, bearing: 90, animate: false}); + expect(fixedNum(camera.getZoom())).toBe(3.2); + expect(camera.getBearing()).toBe(90); + }); + + test('pans, zooms, and rotates', () => { + const camera = createCamera(); + camera.flyTo({center: [100, 0], zoom: 3.2, bearing: 90, duration: 0, animate: false}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 100, lat: 0}); + expect(fixedNum(camera.getZoom())).toBe(3.2); + expect(camera.getBearing()).toBe(90); + }); + + test('noop', () => { + const camera = createCamera(); + camera.flyTo({animate: false}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 0, lat: 0}); + expect(camera.getZoom()).toBe(0); + expect(camera.getBearing()).toBeCloseTo(0); + }); + + test('noop with offset', () => { + const camera = createCamera(); + camera.flyTo({offset: [100, 0], animate: false}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 0, lat: 0}); + expect(camera.getZoom()).toBe(0); + expect(camera.getBearing()).toBeCloseTo(0); + }); + + test('pans with specified offset', () => { + const camera = createCamera(); + camera.flyTo({center: [100, 0], offset: [100, 0], animate: false}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 29.6875, lat: 0}); + }); + + test('pans with specified offset relative to viewport on a rotated camera', () => { + const camera = createCamera({bearing: 180}); + camera.easeTo({center: [100, 0], offset: [100, 0], animate: false}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 170.3125, lat: 0}); + }); + + test('emits move, zoom, rotate, and pitch events, preserving eventData', done => { + expect.assertions(18); + + const camera = createCamera(); + let movestarted, moved, zoomstarted, zoomed, rotatestarted, rotated, pitchstarted, pitched; + const eventData = {data: 'ok'}; + + camera + .on('movestart', (d) => { movestarted = d.data; }) + .on('move', (d) => { moved = d.data; }) + .on('rotate', (d) => { rotated = d.data; }) + .on('pitch', (d) => { pitched = d.data; }) + .on('moveend', function(d) { + expect(this._zooming).toBeFalsy(); + expect(this._panning).toBeFalsy(); + expect(this._rotating).toBeFalsy(); + + expect(movestarted).toBe('ok'); + expect(moved).toBe('ok'); + expect(zoomed).toBe('ok'); + expect(rotated).toBe('ok'); + expect(pitched).toBe('ok'); + expect(d.data).toBe('ok'); + }); + + camera + .on('zoomstart', (d) => { zoomstarted = d.data; }) + .on('zoom', (d) => { zoomed = d.data; }) + .on('zoomend', (d) => { + expect(zoomstarted).toBe('ok'); + expect(zoomed).toBe('ok'); + expect(d.data).toBe('ok'); + }); + + camera + .on('rotatestart', (d) => { rotatestarted = d.data; }) + .on('rotate', (d) => { rotated = d.data; }) + .on('rotateend', (d) => { + expect(rotatestarted).toBe('ok'); + expect(rotated).toBe('ok'); + expect(d.data).toBe('ok'); + }); + + camera + .on('pitchstart', (d) => { pitchstarted = d.data; }) + .on('pitch', (d) => { pitched = d.data; }) + .on('pitchend', (d) => { + expect(pitchstarted).toBe('ok'); + expect(pitched).toBe('ok'); + expect(d.data).toBe('ok'); + }); + + camera.flyTo( + {center: [100, 0], zoom: 3.2, bearing: 90, duration: 0, pitch: 45, animate: false}, + eventData); + done(); + }); + + test('for short flights, emits (solely) move events, preserving eventData', done => { + //As I type this, the code path for guiding super-short flights is (and will probably remain) different. + //As such; it deserves a separate test case. This test case flies the map from A to A. + const camera = createCamera({center: [100, 0]}); + let movestarted, moved, + zoomstarted, zoomed, zoomended, + rotatestarted, rotated, rotateended, + pitchstarted, pitched, pitchended; + const eventData = {data: 'ok'}; + + camera + .on('movestart', (d) => { movestarted = d.data; }) + .on('move', (d) => { moved = d.data; }) + .on('zoomstart', (d) => { zoomstarted = d.data; }) + .on('zoom', (d) => { zoomed = d.data; }) + .on('zoomend', (d) => { zoomended = d.data; }) + .on('rotatestart', (d) => { rotatestarted = d.data; }) + .on('rotate', (d) => { rotated = d.data; }) + .on('rotateend', (d) => { rotateended = d.data; }) + .on('pitchstart', (d) => { pitchstarted = d.data; }) + .on('pitch', (d) => { pitched = d.data; }) + .on('pitchend', (d) => { pitchended = d.data; }) + .on('moveend', function(d) { + expect(this._zooming).toBeFalsy(); + expect(this._panning).toBeFalsy(); + expect(this._rotating).toBeFalsy(); + + expect(movestarted).toBe('ok'); + expect(moved).toBe('ok'); + expect(zoomstarted).toBeUndefined(); + expect(zoomed).toBeUndefined(); + expect(zoomended).toBeUndefined(); + expect(rotatestarted).toBeUndefined(); + expect(rotated).toBeUndefined(); + expect(rotateended).toBeUndefined(); + expect(pitched).toBeUndefined(); + expect(pitchstarted).toBeUndefined(); + expect(pitchended).toBeUndefined(); + expect(d.data).toBe('ok'); + done(); + }); + + const stub = jest.spyOn(browser, 'now'); + stub.mockImplementation(() => 0); + + camera.flyTo({center: [100, 0], duration: 10}, eventData); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('stops existing ease', done => { + const camera = createCamera(); + camera.flyTo({center: [200, 0], duration: 100}); + camera.flyTo({center: [100, 0], duration: 0}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 100, lat: 0}); + done(); + }); + + test('can be called from within a moveend event handler', done => { + const camera = createCamera(); + const stub = jest.spyOn(browser, 'now'); + stub.mockImplementation(() => 0); + + camera.flyTo({center: [100, 0], duration: 10}); + camera.once('moveend', () => { + camera.flyTo({center: [200, 0], duration: 10}); + camera.once('moveend', () => { + camera.flyTo({center: [300, 0], duration: 10}); + camera.once('moveend', () => { + done(); + }); + }); + }); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 20); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 30); + camera.simulateFrame(); + }, 0); + }, 0); + }, 0); + }); + + test('ascends', done => { + const camera = createCamera(); + camera.setZoom(18); + let ascended; + + camera.on('zoom', () => { + if (camera.getZoom() < 18) { + ascended = true; + } + }); + + camera.on('moveend', () => { + expect(ascended).toBeTruthy(); + done(); + }); + + const stub = jest.spyOn(browser, 'now'); + stub.mockImplementation(() => 0); + + camera.flyTo({center: [100, 0], zoom: 18, duration: 10}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('pans eastward across the prime meridian', done => { + const camera = createCamera(); + const stub = jest.spyOn(browser, 'now'); + + camera.setCenter([-10, 0]); + let crossedPrimeMeridian; + + camera.on('move', () => { + if (Math.abs(camera.getCenter().lng) < 10) { + crossedPrimeMeridian = true; + } + }); + + camera.on('moveend', () => { + expect(crossedPrimeMeridian).toBeTruthy(); + done(); + }); + + stub.mockImplementation(() => 0); + camera.flyTo({center: [10, 0], duration: 20}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 20); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('pans westward across the prime meridian', done => { + const camera = createCamera(); + const stub = jest.spyOn(browser, 'now'); + + camera.setCenter([10, 0]); + let crossedPrimeMeridian; + + camera.on('move', () => { + if (Math.abs(camera.getCenter().lng) < 10) { + crossedPrimeMeridian = true; + } + }); + + camera.on('moveend', () => { + expect(crossedPrimeMeridian).toBeTruthy(); + done(); + }); + + stub.mockImplementation(() => 0); + camera.flyTo({center: [-10, 0], duration: 20}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 20); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('pans eastward across the antimeridian', done => { + const camera = createCamera(); + const stub = jest.spyOn(browser, 'now'); + + camera.setCenter([170, 0]); + let crossedAntimeridian; + + camera.on('move', () => { + if (camera.getCenter().lng > 170) { + crossedAntimeridian = true; + } + }); + + camera.on('moveend', () => { + expect(crossedAntimeridian).toBeTruthy(); + done(); + }); + + stub.mockImplementation(() => 0); + camera.flyTo({center: [-170, 0], duration: 20}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 20); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('pans westward across the antimeridian', done => { + const camera = createCamera(); + const stub = jest.spyOn(browser, 'now'); + + camera.setCenter([-170, 0]); + let crossedAntimeridian; + + camera.on('move', () => { + if (camera.getCenter().lng < -170) { + crossedAntimeridian = true; + } + }); + + camera.on('moveend', () => { + expect(crossedAntimeridian).toBeTruthy(); + done(); + }); + + stub.mockImplementation(() => 0); + camera.flyTo({center: [170, 0], duration: 10}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('does not pan eastward across the antimeridian if no world copies', done => { + const camera = createCamera({renderWorldCopies: false}); + const stub = jest.spyOn(browser, 'now'); + + camera.setCenter([170, 0]); + let crossedAntimeridian; + + camera.on('move', () => { + if (camera.getCenter().lng > 170) { + crossedAntimeridian = true; + } + }); + + camera.on('moveend', () => { + expect(crossedAntimeridian).toBeFalsy(); + done(); + }); + + stub.mockImplementation(() => 0); + camera.flyTo({center: [-170, 0], duration: 10}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('does not pan westward across the antimeridian if no world copies', done => { + const camera = createCamera({renderWorldCopies: false}); + const stub = jest.spyOn(browser, 'now'); + + camera.setCenter([-170, 0]); + let crossedAntimeridian; + + camera.on('move', () => { + if (fixedLngLat(camera.getCenter(), 10).lng < -170) { + crossedAntimeridian = true; + } + }); + + camera.on('moveend', () => { + expect(crossedAntimeridian).toBeFalsy(); + done(); + }); + + stub.mockImplementation(() => 0); + camera.flyTo({center: [170, 0], duration: 10}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('jumps back to world 0 when crossing the antimeridian', done => { + const camera = createCamera(); + const stub = jest.spyOn(browser, 'now'); + + camera.setCenter([-170, 0]); + + let leftWorld0 = false; + + camera.on('move', () => { + leftWorld0 = leftWorld0 || (camera.getCenter().lng < -180); + }); + + camera.on('moveend', () => { + expect(leftWorld0).toBeFalsy(); + done(); + }); + + stub.mockImplementation(() => 0); + camera.flyTo({center: [170, 0], duration: 10}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('peaks at the specified zoom level', done => { + const camera = createCamera({zoom: 20}); + const stub = jest.spyOn(browser, 'now'); + + const minZoom = 1; + let zoomed = false; + + camera.on('zoom', () => { + const zoom = camera.getZoom(); + if (zoom < 1) { + fail(`${zoom} should be >= ${minZoom} during flyTo`); + } + + if (camera.getZoom() < (minZoom + 1)) { + zoomed = true; + } + }); + + camera.on('moveend', () => { + expect(zoomed).toBeTruthy(); + done(); + }); + + stub.mockImplementation(() => 0); + camera.flyTo({center: [1, 0], zoom: 20, minZoom, duration: 10}); + + setTimeout(() => { + stub.mockImplementation(() => 3); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('respects transform\'s maxZoom', done => { + const transform = new Transform(2, 10, 0, 60, false); + transform.resize(512, 512); + + const camera = attachSimulateFrame(new CameraMock(transform, {} as any)); + camera._update = () => {}; + + camera.on('moveend', () => { + expect(camera.getZoom()).toBeCloseTo(10); + const {lng, lat} = camera.getCenter(); + expect(lng).toBeCloseTo(12); + expect(lat).toBeCloseTo(34); + done(); + }); + + const stub = jest.spyOn(browser, 'now'); + stub.mockImplementation(() => 0); + camera.flyTo({center: [12, 34], zoom: 30, duration: 10}); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }); + + test('respects transform\'s minZoom', done => { + const transform = new Transform(2, 10, 0, 60, false); + transform.resize(512, 512); + + const camera = attachSimulateFrame(new CameraMock(transform, {} as any)); + camera._update = () => {}; + + camera.on('moveend', () => { + expect(camera.getZoom()).toBeCloseTo(2); + const {lng, lat} = camera.getCenter(); + expect(lng).toBeCloseTo(12); + expect(lat).toBeCloseTo(34); + done(); + }); + + const stub = jest.spyOn(browser, 'now'); + stub.mockImplementation(() => 0); + camera.flyTo({center: [12, 34], zoom: 1, duration: 10}); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }); + + test('resets duration to 0 if it exceeds maxDuration', done => { + let startTime, endTime, timeDiff; + const camera = createCamera({center: [37.63454, 55.75868], zoom: 18}); + + camera + .on('movestart', () => { startTime = new Date(); }) + .on('moveend', () => { + endTime = new Date(); + timeDiff = endTime - startTime; + expect(timeDiff).toBeLessThan(30); + done(); + }); + + camera.flyTo({center: [-122.3998631, 37.7884307], maxDuration: 100}); + }); + + test('flys instantly when prefers-reduce-motion:reduce is set', done => { + const camera = createCamera(); + Object.defineProperty(browser, 'prefersReducedMotion', {value: true}); + assertTransitionTime(done, camera, 0, 10); + camera.flyTo({center: [100, 0], bearing: 90, animate: true}); + }); + + test('check freezeElevation events', done => { + const camera = createCamera(); + const stub = jest.spyOn(browser, 'now'); + + const terrainCallbacks = {prepare: 0, update: 0, finalize: 0} as any; + camera.terrain = {} as Terrain; + camera._prepareElevation = () => { terrainCallbacks.prepare++; }; + camera._updateElevation = () => { terrainCallbacks.update++; }; + camera._finalizeElevation = () => { terrainCallbacks.finalize++; }; + + camera.setCenter([-10, 0]); + + camera.on('moveend', () => { + expect(terrainCallbacks.prepare).toBe(1); + expect(terrainCallbacks.update).toBe(0); + expect(terrainCallbacks.finalize).toBe(1); + done(); + }); + + stub.mockImplementation(() => 0); + camera.flyTo({center: [10, 0], duration: 20, freezeElevation: true}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 20); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('check elevation callbacks', done => { + const camera = createCamera(); + camera.terrain = { + getElevationForLngLatZoom: () => 100, + getMinTileElevationForLngLatZoom: () => 200 + }; + camera.transform = { + elevation: 0, + recalculateZoom: () => true + }; + + camera._prepareElevation([10, 0]); + // expect(camera._elevationCenter).toBe([10, 0]); + expect(camera._elevationStart).toBe(0); + expect(camera._elevationTarget).toBe(100); + expect(camera._elevationFreeze).toBeTruthy(); + + camera.terrain.getElevationForLngLatZoom = () => 200; + camera._updateElevation(0.5); + expect(camera._elevationStart).toBe(-100); + expect(camera._elevationTarget).toBe(200); + + camera._finalizeElevation(); + expect(camera._elevationFreeze).toBeFalsy(); + + done(); + }); + +}); + +describe('#isEasing', () => { + test('returns false when not easing', () => { + const camera = createCamera(); + expect(!camera.isEasing()).toBeTruthy(); + }); + + test('returns true when panning', () => { + const camera = createCamera(); + camera.panTo([100, 0], {duration: 1}); + expect(camera.isEasing()).toBeTruthy(); + }); + + test('returns false when done panning', done => { + const camera = createCamera(); + camera.on('moveend', () => { + expect(!camera.isEasing()).toBeTruthy(); + done(); + }); + const stub = jest.spyOn(browser, 'now'); + stub.mockImplementation(() => 0); + camera.panTo([100, 0], {duration: 1}); + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + }, 0); + }); + + test('returns true when zooming', () => { + const camera = createCamera(); + camera.zoomTo(3.2, {duration: 1}); + + expect(camera.isEasing()).toBeTruthy(); + }); + + test('returns false when done zooming', done => { + const camera = createCamera(); + camera.on('moveend', () => { + expect(!camera.isEasing()).toBeTruthy(); + done(); + }); + const stub = jest.spyOn(browser, 'now'); + stub.mockImplementation(() => 0); + camera.zoomTo(3.2, {duration: 1}); + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + }, 0); + }); + + test('returns true when rotating', () => { + const camera = createCamera(); + camera.rotateTo(90, {duration: 1}); + expect(camera.isEasing()).toBeTruthy(); + }); + + test('returns false when done rotating', done => { + const camera = createCamera(); + camera.on('moveend', () => { + expect(!camera.isEasing()).toBeTruthy(); + done(); + }); + const stub = jest.spyOn(browser, 'now'); + stub.mockImplementation(() => 0); + camera.rotateTo(90, {duration: 1}); + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + }, 0); + }); +}); + +describe('#stop', () => { + test('resets camera._zooming', () => { + const camera = createCamera(); + camera.zoomTo(3.2); + camera.stop(); + expect(!camera._zooming).toBeTruthy(); + }); + + test('resets camera._rotating', () => { + const camera = createCamera(); + camera.rotateTo(90); + camera.stop(); + expect(!camera._rotating).toBeTruthy(); + }); + + test('emits moveend if panning, preserving eventData', done => { + const camera = createCamera(); + const eventData = {data: 'ok'}; + + camera.on('moveend', (d) => { + expect(d.data).toBe('ok'); + done(); + }); + + camera.panTo([100, 0], {}, eventData); + camera.stop(); + }); + + test('emits moveend if zooming, preserving eventData', done => { + const camera = createCamera(); + const eventData = {data: 'ok'}; + + camera.on('moveend', (d) => { + expect(d.data).toBe('ok'); + done(); + }); + + camera.zoomTo(3.2, {}, eventData); + camera.stop(); + }); + + test('emits moveend if rotating, preserving eventData', done => { + const camera = createCamera(); + const eventData = {data: 'ok'}; + + camera.on('moveend', (d) => { + expect(d.data).toBe('ok'); + done(); + }); + + camera.rotateTo(90, {}, eventData); + camera.stop(); + }); + + test('does not emit moveend if not moving', done => { + const camera = createCamera(); + const eventData = {data: 'ok'}; + + camera.on('moveend', (d) => { + expect(d.data).toBe('ok'); + camera.stop(); + done(); // Fails with ".end() called twice" if we get here a second time. + }); + + const stub = jest.spyOn(browser, 'now'); + stub.mockImplementation(() => 0); + camera.panTo([100, 0], {duration: 1}, eventData); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + }, 0); + }); +}); + +describe('#cameraForBounds', () => { + test('no options passed', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + const transform = camera.cameraForBounds(bb); + + expect(fixedLngLat(transform.center, 4)).toEqual({lng: -100.5, lat: 34.7171}); + expect(fixedNum(transform.zoom, 3)).toBe(2.469); + }); + + test('bearing positive number', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + const transform = camera.cameraForBounds(bb, {bearing: 175}); + + expect(fixedLngLat(transform.center, 4)).toEqual({lng: -100.5, lat: 34.7171}); + expect(fixedNum(transform.zoom, 3)).toBe(2.558); + expect(transform.bearing).toBe(175); + }); + + test('bearing negative number', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + const transform = camera.cameraForBounds(bb, {bearing: -30}); + + expect(fixedLngLat(transform.center, 4)).toEqual({lng: -100.5, lat: 34.7171}); + expect(fixedNum(transform.zoom, 3)).toBe(2.392); + expect(transform.bearing).toBe(-30); + }); + + test('padding number', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + const transform = camera.cameraForBounds(bb, {padding: 15}); + + expect(fixedLngLat(transform.center, 4)).toEqual({lng: -100.5, lat: 34.7171}); + expect(fixedNum(transform.zoom, 3)).toBe(2.382); + }); + + test('padding object', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + const transform = camera.cameraForBounds(bb, {padding: {top: 15, right: 15, bottom: 15, left: 15}, duration: 0}); + + expect(fixedLngLat(transform.center, 4)).toEqual({lng: -100.5, lat: 34.7171}); + }); + + test('asymmetrical padding', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + const transform = camera.cameraForBounds(bb, {padding: {top: 10, right: 75, bottom: 50, left: 25}, duration: 0}); + + expect(fixedLngLat(transform.center, 4)).toEqual({lng: -96.5558, lat: 32.0833}); + }); + + test('bearing and asymmetrical padding', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + const transform = camera.cameraForBounds(bb, {bearing: 90, padding: {top: 10, right: 75, bottom: 50, left: 25}, duration: 0}); + + expect(fixedLngLat(transform.center, 4)).toEqual({lng: -103.3761, lat: 31.7099}); + }); + + test('offset', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + const transform = camera.cameraForBounds(bb, {offset: [0, 100]}); + + expect(fixedLngLat(transform.center, 4)).toEqual({lng: -100.5, lat: 44.4717}); + }); + + test('offset and padding', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + const transform = camera.cameraForBounds(bb, {padding: {top: 10, right: 75, bottom: 50, left: 25}, offset: [0, 100]}); + + expect(fixedLngLat(transform.center, 4)).toEqual({lng: -96.5558, lat: 44.4189}); + }); + + test('bearing, asymmetrical padding, and offset', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + const transform = camera.cameraForBounds(bb, {bearing: 90, padding: {top: 10, right: 75, bottom: 50, left: 25}, offset: [0, 100], duration: 0}); + + expect(fixedLngLat(transform.center, 4)).toEqual({lng: -103.3761, lat: 43.0929}); + }); +}); + +describe('#fitBounds', () => { + test('no padding passed', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + camera.fitBounds(bb, {duration: 0}); + + expect(fixedLngLat(camera.getCenter(), 4)).toEqual({lng: -100.5, lat: 34.7171}); + expect(fixedNum(camera.getZoom(), 3)).toBe(2.469); + }); + + test('padding number', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + camera.fitBounds(bb, {padding: 15, duration: 0}); + + expect(fixedLngLat(camera.getCenter(), 4)).toEqual({lng: -100.5, lat: 34.7171}); + expect(fixedNum(camera.getZoom(), 3)).toBe(2.382); + }); + + test('padding object', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + camera.fitBounds(bb, {padding: {top: 10, right: 75, bottom: 50, left: 25}, duration: 0}); + + expect(fixedLngLat(camera.getCenter(), 4)).toEqual({lng: -96.5558, lat: 32.0833}); + }); + + test('padding does not get propagated to transform.padding', () => { + const camera = createCamera(); + const bb = [[-133, 16], [-68, 50]]; + camera.fitBounds(bb, {padding: {top: 10, right: 75, bottom: 50, left: 25}, duration: 0}); + const padding = camera.transform.padding; + + expect(padding).toEqual({ + left: 0, + right: 0, + top: 0, + bottom: 0 + }); + }); +}); + +describe('#fitScreenCoordinates', () => { + test('bearing 225', () => { + const camera = createCamera(); + const p0 = [128, 128]; + const p1 = [256, 256]; + const bearing = 225; + camera.fitScreenCoordinates(p0, p1, bearing, {duration: 0}); + + expect(fixedLngLat(camera.getCenter(), 4)).toEqual({lng: -45, lat: 40.9799}); + expect(fixedNum(camera.getZoom(), 3)).toBe(1.5); + expect(camera.getBearing()).toBe(-135); + }); + + test('bearing 0', () => { + const camera = createCamera(); + const p0 = [128, 128]; + const p1 = [256, 256]; + const bearing = 0; + camera.fitScreenCoordinates(p0, p1, bearing, {duration: 0}); + + expect(fixedLngLat(camera.getCenter(), 4)).toEqual({lng: -45, lat: 40.9799}); + expect(fixedNum(camera.getZoom(), 3)).toBe(2); + expect(camera.getBearing()).toBeCloseTo(0); + }); + + test('inverted points', () => { + const camera = createCamera(); + const p1 = [128, 128]; + const p0 = [256, 256]; + const bearing = 0; + camera.fitScreenCoordinates(p0, p1, bearing, {duration: 0}); + + expect(fixedLngLat(camera.getCenter(), 4)).toEqual({lng: -45, lat: 40.9799}); + expect(fixedNum(camera.getZoom(), 3)).toBe(2); + expect(camera.getBearing()).toBeCloseTo(0); + }); +}); + +describe('queryTerrainElevation', () => { + let camera: Camera; + + beforeEach(() => { + camera = createCamera(); + }); + + test('should return null if terrain is not set', () => { + camera.terrain = null; + const result = camera.queryTerrainElevation([0, 0]); + expect(result).toBeNull(); + }); + + test('should return the correct elevation', () => { + // Set up mock transform and terrain objects + const transform = new Transform(0, 22, 0, 60, true); + transform.elevation = 50; + const terrain = { + getElevationForLngLatZoom: jest.fn().mockReturnValue(200) + } as any as Terrain; + + // Set up camera with mock transform and terrain + camera.transform = transform; + camera.terrain = terrain; + + // Call queryTerrainElevation with mock lngLat + const lngLatLike: LngLatLike = [1, 2]; + const expectedElevation = 150; // 200 - 50 = 150 + const result = camera.queryTerrainElevation(lngLatLike); + + // Check that transform.getElevation was called with the correct arguments + expect(terrain.getElevationForLngLatZoom).toHaveBeenCalledWith( + expect.objectContaining({ + lng: lngLatLike[0], + lat: lngLatLike[1], + }), + transform.tileZoom + ); + + // Check that the correct elevation value was returned + expect(result).toEqual(expectedElevation); + }); +}); + +describe('#transformCameraUpdate', () => { + + test('invoke transformCameraUpdate callback during jumpTo', done => { + const camera = createCamera(); + + let callbackCount = 0; + let eventCount = 0; + + camera.transformCameraUpdate = () => { + callbackCount++; + return {}; + }; + + camera + .on('move', () => { + eventCount++; + expect(eventCount).toBe(callbackCount); + }) + .on('moveend', () => { + done(); + }); + + camera.jumpTo({center: [100, 0]}); + }); + + test('invoke transformCameraUpdate callback during easeTo', done => { + expect.assertions(2); + const camera = createCamera(); + const stub = jest.spyOn(browser, 'now'); + stub.mockImplementation(() => 0); + + let callbackCount = 0; + let eventCount = 0; + + camera.transformCameraUpdate = () => { + callbackCount++; + return {}; + }; + + camera + .on('move', () => { + eventCount++; + expect(eventCount).toBe(callbackCount); + }) + .on('moveend', () => { + done(); + }); + + camera.easeTo({center: [100, 0], duration: 10}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('invoke transformCameraUpdate callback during flyTo', done => { + expect.assertions(2); + const camera = createCamera(); + const stub = jest.spyOn(browser, 'now'); + stub.mockImplementation(() => 0); + + let callbackCount = 0; + let eventCount = 0; + + camera.transformCameraUpdate = () => { + callbackCount++; + return {}; + }; + + camera + .on('move', () => { + eventCount++; + expect(eventCount).toBe(callbackCount); + }) + .on('moveend', () => { + done(); + }); + + camera.flyTo({center: [100, 0], duration: 10}); + + setTimeout(() => { + stub.mockImplementation(() => 1); + camera.simulateFrame(); + + setTimeout(() => { + stub.mockImplementation(() => 10); + camera.simulateFrame(); + }, 0); + }, 0); + }); + + test('transformCameraUpdate overrides proposed camera settings', () => { + const camera = createCamera(); + + camera.transformCameraUpdate = ({center, zoom}) => { + return { + center: LngLat.convert([center.lng, center.lat + 10]), + zoom: Math.round(zoom) + }; + }; + + camera.flyTo({center: [100, 0], zoom: 3.2, animate: false}); + expect(fixedLngLat(camera.getCenter())).toEqual({lng: 100, lat: 10}); + expect(fixedNum(camera.getZoom())).toBe(3); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/camera.ts b/web/libraries/maplibre-gl/src/ui/camera.ts new file mode 100644 index 00000000..0631b114 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/camera.ts @@ -0,0 +1,1478 @@ +import {extend, warnOnce, clamp, wrap, defaultEasing, pick} from '../util/util'; +import {interpolates} from '@maplibre/maplibre-gl-style-spec'; +import {browser} from '../util/browser'; +import {LngLat} from '../geo/lng_lat'; +import {LngLatBounds} from '../geo/lng_lat_bounds'; +import Point from '@mapbox/point-geometry'; +import {Event, Evented} from '../util/evented'; +import {Terrain} from '../render/terrain'; + +import type {Transform} from '../geo/transform'; +import type {LngLatLike} from '../geo/lng_lat'; +import type {LngLatBoundsLike} from '../geo/lng_lat_bounds'; +import type {TaskID} from '../util/task_queue'; +import type {PaddingOptions} from '../geo/edge_insets'; +import {MercatorCoordinate} from '../geo/mercator_coordinate'; + +/** + * A [Point](https://github.com/mapbox/point-geometry) or an array of two numbers representing `x` and `y` screen coordinates in pixels. + * + * @group Geography and Geometry + * + * @example + * ```ts + * let p1 = new Point(-77, 38); // a PointLike which is a Point + * let p2 = [-77, 38]; // a PointLike which is an array of two numbers + * ``` + */ +export type PointLike = Point | [number, number]; + +/** + * A helper to allow require of at least one propery + */ +export type RequireAtLeastOne = { [K in keyof T]-?: Required> & Partial>>; }[keyof T]; + +/** + * Options common to {@link Map#jumpTo}, {@link Map#easeTo}, and {@link Map#flyTo}, controlling the desired location, + * zoom, bearing, and pitch of the camera. All properties are optional, and when a property is omitted, the current + * camera value for that property will remain unchanged. + * + * @example + * Set the map's initial perspective with CameraOptions + * ```ts + * let map = new maplibregl.Map({ + * container: 'map', + * style: 'https://demotiles.maplibre.org/style.json', + * center: [-73.5804, 45.53483], + * pitch: 60, + * bearing: -60, + * zoom: 10 + * }); + * ``` + * @see [Set pitch and bearing](https://maplibre.org/maplibre-gl-js/docs/examples/set-perspective/) + * @see [Jump to a series of locations](https://maplibre.org/maplibre-gl-js/docs/examples/jump-to/) + * @see [Fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto/) + * @see [Display buildings in 3D](https://maplibre.org/maplibre-gl-js/docs/examples/3d-buildings/) + */ +export type CameraOptions = CenterZoomBearing & { + /** + * The desired pitch in degrees. The pitch is the angle towards the horizon + * measured in degrees with a range between 0 and 60 degrees. For example, pitch: 0 provides the appearance + * of looking straight down at the map, while pitch: 60 tilts the user's perspective towards the horizon. + * Increasing the pitch value is often used to display 3D objects. + */ + pitch?: number; + /** + * If `zoom` is specified, `around` determines the point around which the zoom is centered. + */ + around?: LngLatLike; +}; + +/** + * Holds center, zoom and bearing properties + */ +export type CenterZoomBearing = { + /** + * The desired center. + */ + center?: LngLatLike; + /** + * The desired zoom level. + */ + zoom?: number; + /** + * The desired bearing in degrees. The bearing is the compass direction that + * is "up". For example, `bearing: 90` orients the map so that east is up. + */ + bearing?: number; +} + +/** + * The options object related to the {@link Map#jumpTo} method + */ +export type JumpToOptions = CameraOptions & { + /** + * Dimensions in pixels applied on each side of the viewport for shifting the vanishing point. + */ + padding?: PaddingOptions; +} + +/** + * A options object for the {@link Map#cameraForBounds} method + */ +export type CameraForBoundsOptions = CameraOptions & { + /** + * The amount of padding in pixels to add to the given bounds. + */ + padding?: number | RequireAtLeastOne; + /** + * The center of the given bounds relative to the map's center, measured in pixels. + * @defaultValue [0, 0] + */ + offset?: PointLike; + /** + * The maximum zoom level to allow when the camera would transition to the specified bounds. + */ + maxZoom?: number; +} + +/** + * The {@link Map#flyTo} options object + */ +export type FlyToOptions = AnimationOptions & CameraOptions & { + /** + * The zooming "curve" that will occur along the + * flight path. A high value maximizes zooming for an exaggerated animation, while a low + * value minimizes zooming for an effect closer to {@link Map#easeTo}. 1.42 is the average + * value selected by participants in the user study discussed in + * [van Wijk (2003)](https://www.win.tue.nl/~vanwijk/zoompan.pdf). A value of + * `Math.pow(6, 0.25)` would be equivalent to the root mean squared average velocity. A + * value of 1 would produce a circular motion. + * @defaultValue 1.42 + */ + curve?: number; + /** + * The zero-based zoom level at the peak of the flight path. If + * `options.curve` is specified, this option is ignored. + */ + minZoom?: number; + /** + * The average speed of the animation defined in relation to + * `options.curve`. A speed of 1.2 means that the map appears to move along the flight path + * by 1.2 times `options.curve` screenfuls every second. A _screenful_ is the map's visible span. + * It does not correspond to a fixed physical distance, but varies by zoom level. + * @defaultValue 1.2 + */ + speed?: number; + /** + * The average speed of the animation measured in screenfuls + * per second, assuming a linear timing curve. If `options.speed` is specified, this option is ignored. + */ + screenSpeed?: number; + /** + * The animation's maximum duration, measured in milliseconds. + * If duration exceeds maximum duration, it resets to 0. + */ + maxDuration?: number; + /** + * The amount of padding in pixels to add to the given bounds. + */ + padding?: number | RequireAtLeastOne; +} + +export type EaseToOptions = AnimationOptions & CameraOptions & { + delayEndEvents?: number; + padding?: number | RequireAtLeastOne; +} + +/** + * Options for {@link Map#fitBounds} method + */ +export type FitBoundsOptions = FlyToOptions & { + /** + * If `true`, the map transitions using {@link Map#easeTo}. If `false`, the map transitions using {@link Map#flyTo}. + * See those functions and {@link AnimationOptions} for information about options available. + * @defaultValue false + */ + linear?: boolean; + /** + * The center of the given bounds relative to the map's center, measured in pixels. + * @defaultValue [0, 0] + */ + offset?: PointLike; + /** + * The maximum zoom level to allow when the map view transitions to the specified bounds. + */ + maxZoom?: number; +} + +/** + * Options common to map movement methods that involve animation, such as {@link Map#panBy} and + * {@link Map#easeTo}, controlling the duration and easing function of the animation. All properties + * are optional. + * + */ +export type AnimationOptions = { + /** + * The animation's duration, measured in milliseconds. + */ + duration?: number; + /** + * A function taking a time in the range 0..1 and returning a number where 0 is + * the initial state and 1 is the final state. + */ + easing?: (_: number) => number; + /** + * of the target center relative to real map container center at the end of animation. + */ + offset?: PointLike; + /** + * If `false`, no animation will occur. + */ + animate?: boolean; + /** + * If `true`, then the animation is considered essential and will not be affected by + * [`prefers-reduced-motion`](https://developer.mozilla.org/en-US/docs/Web/CSS/\@media/prefers-reduced-motion). + */ + essential?: boolean; + /** + * Default false. Needed in 3D maps to let the camera stay in a constant + * height based on sea-level. After the animation finished the zoom-level will be recalculated in respect of + * the distance from the camera to the center-coordinate-altitude. + */ + freezeElevation?: boolean; +}; + +/** + * A callback hook that allows manipulating the camera and being notified about camera updates before they happen + */ +export type CameraUpdateTransformFunction = (next: { + center: LngLat; + zoom: number; + pitch: number; + bearing: number; + elevation: number; +}) => { + center?: LngLat; + zoom?: number; + pitch?: number; + bearing?: number; + elevation?: number; +}; + +export abstract class Camera extends Evented { + transform: Transform; + terrain: Terrain; + + _moving: boolean; + _zooming: boolean; + _rotating: boolean; + _pitching: boolean; + _padding: boolean; + + _bearingSnap: number; + _easeStart: number; + _easeOptions: { + duration?: number; + easing?: (_: number) => number; + }; + _easeId: string | void; + + _onEaseFrame: (_: number) => void; + _onEaseEnd: (easeId?: string) => void; + _easeFrameId: TaskID; + + /** + * @internal + * holds the geographical coordinate of the target + */ + _elevationCenter: LngLat; + /** + * @internal + * holds the targ altitude value, = center elevation of the target. + * This value may changes during flight, because new terrain-tiles loads during flight. + */ + _elevationTarget: number; + /** + * @internal + * holds the start altitude value, = center elevation before animation begins + * this value will recalculated during flight in respect of changing _elevationTarget values, + * so the linear interpolation between start and target keeps smooth and without jumps. + */ + _elevationStart: number; + /** + * @internal + * Saves the current state of the elevation freeze - this is used during map movement to prevent "rocky" camera movement. + */ + _elevationFreeze: boolean; + /** + * @internal + * Used to track accumulated changes during continuous interaction + */ + _requestedCameraState?: Transform; + /** + * A callback used to defer camera updates or apply arbitrary constraints. + * If specified, this Camera instance can be used as a stateless component in React etc. + */ + transformCameraUpdate: CameraUpdateTransformFunction | null; + + abstract _requestRenderFrame(a: () => void): TaskID; + abstract _cancelRenderFrame(_: TaskID): void; + + constructor(transform: Transform, options: { + bearingSnap: number; + }) { + super(); + this._moving = false; + this._zooming = false; + this.transform = transform; + this._bearingSnap = options.bearingSnap; + + this.on('moveend', () => { + delete this._requestedCameraState; + }); + } + + /** + * Returns the map's geographical centerpoint. + * + * @returns The map's geographical centerpoint. + * @example + * Return a LngLat object such as `{lng: 0, lat: 0}` + * ```ts + * let center = map.getCenter(); + * // access longitude and latitude values directly + * let {lng, lat} = map.getCenter(); + * ``` + */ + getCenter(): LngLat { return new LngLat(this.transform.center.lng, this.transform.center.lat); } + + /** + * Sets the map's geographical centerpoint. Equivalent to `jumpTo({center: center})`. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param center - The centerpoint to set. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * map.setCenter([-74, 38]); + * ``` + */ + setCenter(center: LngLatLike, eventData?: any) { + return this.jumpTo({center}, eventData); + } + + /** + * Pans the map by the specified offset. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param offset - `x` and `y` coordinates by which to pan the map. + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/) + */ + panBy(offset: PointLike, options?: AnimationOptions, eventData?: any): this { + offset = Point.convert(offset).mult(-1); + return this.panTo(this.transform.center, extend({offset}, options), eventData); + } + + /** + * Pans the map to the specified location with an animated transition. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param lnglat - The location to pan the map to. + * @param options - Options describing the destination and animation of the transition. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * map.panTo([-74, 38]); + * // Specify that the panTo animation should last 5000 milliseconds. + * map.panTo([-74, 38], {duration: 5000}); + * ``` + * @see [Update a feature in realtime](https://maplibre.org/maplibre-gl-js/docs/examples/live-update-feature/) + */ + panTo(lnglat: LngLatLike, options?: AnimationOptions, eventData?: any): this { + return this.easeTo(extend({ + center: lnglat + }, options), eventData); + } + + /** + * Returns the map's current zoom level. + * + * @returns The map's current zoom level. + * @example + * ```ts + * map.getZoom(); + * ``` + */ + getZoom(): number { return this.transform.zoom; } + + /** + * Sets the map's zoom level. Equivalent to `jumpTo({zoom: zoom})`. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`. + * + * @param zoom - The zoom level to set (0-20). + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Zoom to the zoom level 5 without an animated transition + * ```ts + * map.setZoom(5); + * ``` + */ + setZoom(zoom: number, eventData?: any): this { + this.jumpTo({zoom}, eventData); + return this; + } + + /** + * Zooms the map to the specified zoom level, with an animated transition. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`. + * + * @param zoom - The zoom level to transition to. + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * // Zoom to the zoom level 5 without an animated transition + * map.zoomTo(5); + * // Zoom to the zoom level 8 with an animated transition + * map.zoomTo(8, { + * duration: 2000, + * offset: [100, 50] + * }); + * ``` + */ + zoomTo(zoom: number, options?: AnimationOptions | null, eventData?: any): this { + return this.easeTo(extend({ + zoom + }, options), eventData); + } + + /** + * Increases the map's zoom level by 1. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Zoom the map in one level with a custom animation duration + * ```ts + * map.zoomIn({duration: 1000}); + * ``` + */ + zoomIn(options?: AnimationOptions, eventData?: any): this { + this.zoomTo(this.getZoom() + 1, options, eventData); + return this; + } + + /** + * Decreases the map's zoom level by 1. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, and `zoomend`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Zoom the map out one level with a custom animation offset + * ```ts + * map.zoomOut({offset: [80, 60]}); + * ``` + */ + zoomOut(options?: AnimationOptions, eventData?: any): this { + this.zoomTo(this.getZoom() - 1, options, eventData); + return this; + } + + /** + * Returns the map's current bearing. The bearing is the compass direction that is "up"; for example, a bearing + * of 90° orients the map so that east is up. + * + * @returns The map's current bearing. + * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/) + */ + getBearing(): number { return this.transform.bearing; } + + /** + * Sets the map's bearing (rotation). The bearing is the compass direction that is "up"; for example, a bearing + * of 90° orients the map so that east is up. + * + * Equivalent to `jumpTo({bearing: bearing})`. + * + * Triggers the following events: `movestart`, `moveend`, and `rotate`. + * + * @param bearing - The desired bearing. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Rotate the map to 90 degrees + * ```ts + * map.setBearing(90); + * ``` + */ + setBearing(bearing: number, eventData?: any): this { + this.jumpTo({bearing}, eventData); + return this; + } + + /** + * Returns the current padding applied around the map viewport. + * + * @returns The current padding around the map viewport. + */ + getPadding(): PaddingOptions { return this.transform.padding; } + + /** + * Sets the padding in pixels around the viewport. + * + * Equivalent to `jumpTo({padding: padding})`. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param padding - The desired padding. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * Sets a left padding of 300px, and a top padding of 50px + * ```ts + * map.setPadding({ left: 300, top: 50 }); + * ``` + */ + setPadding(padding: PaddingOptions, eventData?: any): this { + this.jumpTo({padding}, eventData); + return this; + } + + /** + * Rotates the map to the specified bearing, with an animated transition. The bearing is the compass direction + * that is "up"; for example, a bearing of 90° orients the map so that east is up. + * + * Triggers the following events: `movestart`, `moveend`, and `rotate`. + * + * @param bearing - The desired bearing. + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + rotateTo(bearing: number, options?: AnimationOptions, eventData?: any): this { + return this.easeTo(extend({ + bearing + }, options), eventData); + } + + /** + * Rotates the map so that north is up (0° bearing), with an animated transition. + * + * Triggers the following events: `movestart`, `moveend`, and `rotate`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + resetNorth(options?: AnimationOptions, eventData?: any): this { + this.rotateTo(0, extend({duration: 1000}, options), eventData); + return this; + } + + /** + * Rotates and pitches the map so that north is up (0° bearing) and pitch is 0°, with an animated transition. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `pitchstart`, `pitch`, `pitchend`, and `rotate`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + resetNorthPitch(options?: AnimationOptions, eventData?: any): this { + this.easeTo(extend({ + bearing: 0, + pitch: 0, + duration: 1000 + }, options), eventData); + return this; + } + + /** + * Snaps the map so that north is up (0° bearing), if the current bearing is close enough to it (i.e. within the + * `bearingSnap` threshold). + * + * Triggers the following events: `movestart`, `moveend`, and `rotate`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + snapToNorth(options?: AnimationOptions, eventData?: any): this { + if (Math.abs(this.getBearing()) < this._bearingSnap) { + return this.resetNorth(options, eventData); + } + return this; + } + + /** + * Returns the map's current pitch (tilt). + * + * @returns The map's current pitch, measured in degrees away from the plane of the screen. + */ + getPitch(): number { return this.transform.pitch; } + + /** + * Sets the map's pitch (tilt). Equivalent to `jumpTo({pitch: pitch})`. + * + * Triggers the following events: `movestart`, `moveend`, `pitchstart`, and `pitchend`. + * + * @param pitch - The pitch to set, measured in degrees away from the plane of the screen (0-60). + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + */ + setPitch(pitch: number, eventData?: any): this { + this.jumpTo({pitch}, eventData); + return this; + } + + /** + * @param bounds - Calculate the center for these bounds in the viewport and use + * the highest zoom level up to and including `Map#getMaxZoom()` that fits + * in the viewport. LngLatBounds represent a box that is always axis-aligned with bearing 0. + * @param options - Options object + * @returns If map is able to fit to provided bounds, returns `center`, `zoom`, and `bearing`. + * If map is unable to fit, method will warn and return undefined. + * @example + * ```ts + * let bbox = [[-79, 43], [-73, 45]]; + * let newCameraTransform = map.cameraForBounds(bbox, { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + */ + cameraForBounds(bounds: LngLatBoundsLike, options?: CameraForBoundsOptions): CenterZoomBearing { + bounds = LngLatBounds.convert(bounds); + const bearing = options && options.bearing || 0; + return this._cameraForBoxAndBearing(bounds.getNorthWest(), bounds.getSouthEast(), bearing, options); + } + + /** + * @internal + * Calculate the center of these two points in the viewport and use + * the highest zoom level up to and including `Map#getMaxZoom()` that fits + * the points in the viewport at the specified bearing. + * @param p0 - First point + * @param p1 - Second point + * @param bearing - Desired map bearing at end of animation, in degrees + * @param options - the camera options + * @returns If map is able to fit to provided bounds, returns `center`, `zoom`, and `bearing`. + * If map is unable to fit, method will warn and return undefined. + * @example + * ```ts + * let p0 = [-79, 43]; + * let p1 = [-73, 45]; + * let bearing = 90; + * let newCameraTransform = map._cameraForBoxAndBearing(p0, p1, bearing, { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + */ + _cameraForBoxAndBearing(p0: LngLatLike, p1: LngLatLike, bearing: number, options?: CameraForBoundsOptions): CenterZoomBearing { + const defaultPadding = { + top: 0, + bottom: 0, + right: 0, + left: 0 + }; + options = extend({ + padding: defaultPadding, + offset: [0, 0], + maxZoom: this.transform.maxZoom + }, options); + + if (typeof options.padding === 'number') { + const p = options.padding; + options.padding = { + top: p, + bottom: p, + right: p, + left: p + }; + } + + options.padding = extend(defaultPadding, options.padding) as PaddingOptions; + const tr = this.transform; + const edgePadding = tr.padding; + + // We want to calculate the upper right and lower left of the box defined by p0 and p1 + // in a coordinate system rotate to match the destination bearing. + const p0world = tr.project(LngLat.convert(p0)); + const p1world = tr.project(LngLat.convert(p1)); + const p0rotated = p0world.rotate(-bearing * Math.PI / 180); + const p1rotated = p1world.rotate(-bearing * Math.PI / 180); + + const upperRight = new Point(Math.max(p0rotated.x, p1rotated.x), Math.max(p0rotated.y, p1rotated.y)); + const lowerLeft = new Point(Math.min(p0rotated.x, p1rotated.x), Math.min(p0rotated.y, p1rotated.y)); + + // Calculate zoom: consider the original bbox and padding. + const size = upperRight.sub(lowerLeft); + const scaleX = (tr.width - (edgePadding.left + edgePadding.right + options.padding.left + options.padding.right)) / size.x; + const scaleY = (tr.height - (edgePadding.top + edgePadding.bottom + options.padding.top + options.padding.bottom)) / size.y; + + if (scaleY < 0 || scaleX < 0) { + warnOnce( + 'Map cannot fit within canvas with the given bounds, padding, and/or offset.' + ); + return undefined; + } + + const zoom = Math.min(tr.scaleZoom(tr.scale * Math.min(scaleX, scaleY)), options.maxZoom); + + // Calculate center: apply the zoom, the configured offset, as well as offset that exists as a result of padding. + const offset = Point.convert(options.offset); + const paddingOffsetX = (options.padding.left - options.padding.right) / 2; + const paddingOffsetY = (options.padding.top - options.padding.bottom) / 2; + const paddingOffset = new Point(paddingOffsetX, paddingOffsetY); + const rotatedPaddingOffset = paddingOffset.rotate(bearing * Math.PI / 180); + const offsetAtInitialZoom = offset.add(rotatedPaddingOffset); + const offsetAtFinalZoom = offsetAtInitialZoom.mult(tr.scale / tr.zoomScale(zoom)); + + const center = tr.unproject(p0world.add(p1world).div(2).sub(offsetAtFinalZoom)); + + return { + center, + zoom, + bearing + }; + } + + /** + * Pans and zooms the map to contain its visible area within the specified geographical bounds. + * This function will also reset the map's bearing to 0 if bearing is nonzero. + * + * Triggers the following events: `movestart` and `moveend`. + * + * @param bounds - Center these bounds in the viewport and use the highest + * zoom level up to and including `Map#getMaxZoom()` that fits them in the viewport. + * @param options- Options supports all properties from {@link AnimationOptions} and {@link CameraOptions} in addition to the fields below. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * let bbox = [[-79, 43], [-73, 45]]; + * map.fitBounds(bbox, { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + * @see [Fit a map to a bounding box](https://maplibre.org/maplibre-gl-js/docs/examples/fitbounds/) + */ + fitBounds(bounds: LngLatBoundsLike, options?: FitBoundsOptions, eventData?: any): this { + return this._fitInternal( + this.cameraForBounds(bounds, options), + options, + eventData); + } + + /** + * Pans, rotates and zooms the map to to fit the box made by points p0 and p1 + * once the map is rotated to the specified bearing. To zoom without rotating, + * pass in the current map bearing. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend` and `rotate`. + * + * @param p0 - First point on screen, in pixel coordinates + * @param p1 - Second point on screen, in pixel coordinates + * @param bearing - Desired map bearing at end of animation, in degrees + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * let p0 = [220, 400]; + * let p1 = [500, 900]; + * map.fitScreenCoordinates(p0, p1, map.getBearing(), { + * padding: {top: 10, bottom:25, left: 15, right: 5} + * }); + * ``` + * @see Used by {@link BoxZoomHandler} + */ + fitScreenCoordinates(p0: PointLike, p1: PointLike, bearing: number, options?: FitBoundsOptions, eventData?: any): this { + return this._fitInternal( + this._cameraForBoxAndBearing( + this.transform.pointLocation(Point.convert(p0)), + this.transform.pointLocation(Point.convert(p1)), + bearing, + options), + options, + eventData); + } + + _fitInternal(calculatedOptions?: CenterZoomBearing, options?: FitBoundsOptions, eventData?: any): this { + // cameraForBounds warns + returns undefined if unable to fit: + if (!calculatedOptions) return this; + + options = extend(calculatedOptions, options); + // Explicitly remove the padding field because, calculatedOptions already accounts for padding by setting zoom and center accordingly. + delete options.padding; + + return options.linear ? + this.easeTo(options, eventData) : + this.flyTo(options, eventData); + } + + /** + * Changes any combination of center, zoom, bearing, and pitch, without + * an animated transition. The map will retain its current values for any + * details not specified in `options`. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`, + * `pitch`, `pitchend`, and `rotate`. + * + * @param options - Options object + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * // jump to coordinates at current zoom + * map.jumpTo({center: [0, 0]}); + * // jump with zoom, pitch, and bearing options + * map.jumpTo({ + * center: [0, 0], + * zoom: 8, + * pitch: 45, + * bearing: 90 + * }); + * ``` + * @see [Jump to a series of locations](https://maplibre.org/maplibre-gl-js/docs/examples/jump-to/) + * @see [Update a feature in realtime](https://maplibre.org/maplibre-gl-js/docs/examples/live-update-feature/) + */ + jumpTo(options: JumpToOptions, eventData?: any): this { + this.stop(); + + const tr = this._getTransformForUpdate(); + let zoomChanged = false, + bearingChanged = false, + pitchChanged = false; + + if ('zoom' in options && tr.zoom !== +options.zoom) { + zoomChanged = true; + tr.zoom = +options.zoom; + } + + if (options.center !== undefined) { + tr.center = LngLat.convert(options.center); + } + + if ('bearing' in options && tr.bearing !== +options.bearing) { + bearingChanged = true; + tr.bearing = +options.bearing; + } + + if ('pitch' in options && tr.pitch !== +options.pitch) { + pitchChanged = true; + tr.pitch = +options.pitch; + } + + if (options.padding != null && !tr.isPaddingEqual(options.padding)) { + tr.padding = options.padding; + } + this._applyUpdatedTransform(tr); + + this.fire(new Event('movestart', eventData)) + .fire(new Event('move', eventData)); + + if (zoomChanged) { + this.fire(new Event('zoomstart', eventData)) + .fire(new Event('zoom', eventData)) + .fire(new Event('zoomend', eventData)); + } + + if (bearingChanged) { + this.fire(new Event('rotatestart', eventData)) + .fire(new Event('rotate', eventData)) + .fire(new Event('rotateend', eventData)); + } + + if (pitchChanged) { + this.fire(new Event('pitchstart', eventData)) + .fire(new Event('pitch', eventData)) + .fire(new Event('pitchend', eventData)); + } + + return this.fire(new Event('moveend', eventData)); + } + + /** + * Calculates pitch, zoom and bearing for looking at `newCenter` with the camera position being `newCenter` + * and returns them as {@link CameraOptions}. + * @param from - The camera to look from + * @param altitudeFrom - The altitude of the camera to look from + * @param to - The center to look at + * @param altitudeTo - Optional altitude of the center to look at. If none given the ground height will be used. + * @returns the calculated camera options + */ + calculateCameraOptionsFromTo(from: LngLat, altitudeFrom: number, to: LngLat, altitudeTo: number = 0): CameraOptions { + const fromMerc = MercatorCoordinate.fromLngLat(from, altitudeFrom); + const toMerc = MercatorCoordinate.fromLngLat(to, altitudeTo); + const dx = toMerc.x - fromMerc.x; + const dy = toMerc.y - fromMerc.y; + const dz = toMerc.z - fromMerc.z; + + const distance3D = Math.hypot(dx, dy, dz); + if (distance3D === 0) throw new Error('Can\'t calculate camera options with same From and To'); + + const groundDistance = Math.hypot(dx, dy); + + const zoom = this.transform.scaleZoom(this.transform.cameraToCenterDistance / distance3D / this.transform.tileSize); + const bearing = (Math.atan2(dx, -dy) * 180) / Math.PI; + let pitch = (Math.acos(groundDistance / distance3D) * 180) / Math.PI; + pitch = dz < 0 ? 90 - pitch : 90 + pitch; + + return { + center: toMerc.toLngLat(), + zoom, + pitch, + bearing + }; + } + + /** + * Changes any combination of `center`, `zoom`, `bearing`, `pitch`, and `padding` with an animated transition + * between old and new values. The map will retain its current values for any + * details not specified in `options`. + * + * Note: The transition will happen instantly if the user has enabled + * the `reduced motion` accessibility feature enabled in their operating system, + * unless `options` includes `essential: true`. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`, + * `pitch`, `pitchend`, and `rotate`. + * + * @param options - Options describing the destination and animation of the transition. + * Accepts {@link CameraOptions} and {@link AnimationOptions}. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @see [Navigate the map with game-like controls](https://maplibre.org/maplibre-gl-js/docs/examples/game-controls/) + */ + easeTo(options: EaseToOptions & { + easeId?: string; + noMoveStart?: boolean; + }, eventData?: any): this { + this._stop(false, options.easeId); + + options = extend({ + offset: [0, 0], + duration: 500, + easing: defaultEasing + }, options); + + if (options.animate === false || (!options.essential && browser.prefersReducedMotion)) options.duration = 0; + + const tr = this._getTransformForUpdate(), + startZoom = this.getZoom(), + startBearing = this.getBearing(), + startPitch = this.getPitch(), + startPadding = this.getPadding(), + + zoom = 'zoom' in options ? +options.zoom : startZoom, + bearing = 'bearing' in options ? this._normalizeBearing(options.bearing, startBearing) : startBearing, + pitch = 'pitch' in options ? +options.pitch : startPitch, + padding = 'padding' in options ? options.padding : tr.padding; + + const offsetAsPoint = Point.convert(options.offset); + let pointAtOffset = tr.centerPoint.add(offsetAsPoint); + const locationAtOffset = tr.pointLocation(pointAtOffset); + const center = LngLat.convert(options.center || locationAtOffset); + this._normalizeCenter(center); + + const from = tr.project(locationAtOffset); + const delta = tr.project(center).sub(from); + const finalScale = tr.zoomScale(zoom - startZoom); + + let around, aroundPoint; + + if (options.around) { + around = LngLat.convert(options.around); + aroundPoint = tr.locationPoint(around); + } + + const currently = { + moving: this._moving, + zooming: this._zooming, + rotating: this._rotating, + pitching: this._pitching + }; + + this._zooming = this._zooming || (zoom !== startZoom); + this._rotating = this._rotating || (startBearing !== bearing); + this._pitching = this._pitching || (pitch !== startPitch); + this._padding = !tr.isPaddingEqual(padding as PaddingOptions); + + this._easeId = options.easeId; + this._prepareEase(eventData, options.noMoveStart, currently); + if (this.terrain) this._prepareElevation(center); + + this._ease((k) => { + if (this._zooming) { + tr.zoom = interpolates.number(startZoom, zoom, k); + } + if (this._rotating) { + tr.bearing = interpolates.number(startBearing, bearing, k); + } + if (this._pitching) { + tr.pitch = interpolates.number(startPitch, pitch, k); + } + if (this._padding) { + tr.interpolatePadding(startPadding, padding as PaddingOptions, k); + // When padding is being applied, Transform#centerPoint is changing continuously, + // thus we need to recalculate offsetPoint every frame + pointAtOffset = tr.centerPoint.add(offsetAsPoint); + } + + if (this.terrain && !options.freezeElevation) this._updateElevation(k); + + if (around) { + tr.setLocationAtPoint(around, aroundPoint); + } else { + const scale = tr.zoomScale(tr.zoom - startZoom); + const base = zoom > startZoom ? + Math.min(2, finalScale) : + Math.max(0.5, finalScale); + const speedup = Math.pow(base, 1 - k); + const newCenter = tr.unproject(from.add(delta.mult(k * speedup)).mult(scale)); + tr.setLocationAtPoint(tr.renderWorldCopies ? newCenter.wrap() : newCenter, pointAtOffset); + } + + this._applyUpdatedTransform(tr); + + this._fireMoveEvents(eventData); + + }, (interruptingEaseId?: string) => { + if (this.terrain) this._finalizeElevation(); + this._afterEase(eventData, interruptingEaseId); + }, options as any); + + return this; + } + + _prepareEase(eventData: any, noMoveStart: boolean, currently: any = {}) { + this._moving = true; + if (!noMoveStart && !currently.moving) { + this.fire(new Event('movestart', eventData)); + } + if (this._zooming && !currently.zooming) { + this.fire(new Event('zoomstart', eventData)); + } + if (this._rotating && !currently.rotating) { + this.fire(new Event('rotatestart', eventData)); + } + if (this._pitching && !currently.pitching) { + this.fire(new Event('pitchstart', eventData)); + } + } + + _prepareElevation(center: LngLat) { + this._elevationCenter = center; + this._elevationStart = this.transform.elevation; + this._elevationTarget = this.terrain.getElevationForLngLatZoom(center, this.transform.tileZoom); + this._elevationFreeze = true; + } + + _updateElevation(k: number) { + this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this._elevationCenter, this.transform.tileZoom); + const elevation = this.terrain.getElevationForLngLatZoom(this._elevationCenter, this.transform.tileZoom); + // target terrain updated during flight, slowly move camera to new height + if (k < 1 && elevation !== this._elevationTarget) { + const pitch1 = this._elevationTarget - this._elevationStart; + const pitch2 = (elevation - (pitch1 * k + this._elevationStart)) / (1 - k); + this._elevationStart += k * (pitch1 - pitch2); + this._elevationTarget = elevation; + } + this.transform.elevation = interpolates.number(this._elevationStart, this._elevationTarget, k); + } + + _finalizeElevation() { + this._elevationFreeze = false; + this.transform.recalculateZoom(this.terrain); + } + + /** + * @internal + * Called when the camera is about to be manipulated. + * If `transformCameraUpdate` is specified, a copy of the current transform is created to track the accumulated changes. + * This underlying transform represents the "desired state" proposed by input handlers / animations / UI controls. + * It may differ from the state used for rendering (`this.transform`). + * @returns Transform to apply changes to + */ + _getTransformForUpdate(): Transform { + if (!this.transformCameraUpdate) return this.transform; + + if (!this._requestedCameraState) { + this._requestedCameraState = this.transform.clone(); + } + return this._requestedCameraState; + } + + /** + * @internal + * Called after the camera is done being manipulated. + * @param tr - the requested camera end state + * Call `transformCameraUpdate` if present, and then apply the "approved" changes. + */ + _applyUpdatedTransform(tr: Transform) { + if (!this.transformCameraUpdate) return; + + const nextTransform = tr.clone(); + const { + center, + zoom, + pitch, + bearing, + elevation + } = this.transformCameraUpdate(nextTransform); + if (center) nextTransform.center = center; + if (zoom !== undefined) nextTransform.zoom = zoom; + if (pitch !== undefined) nextTransform.pitch = pitch; + if (bearing !== undefined) nextTransform.bearing = bearing; + if (elevation !== undefined) nextTransform.elevation = elevation; + this.transform.apply(nextTransform); + } + + _fireMoveEvents(eventData?: any) { + this.fire(new Event('move', eventData)); + if (this._zooming) { + this.fire(new Event('zoom', eventData)); + } + if (this._rotating) { + this.fire(new Event('rotate', eventData)); + } + if (this._pitching) { + this.fire(new Event('pitch', eventData)); + } + } + + _afterEase(eventData?: any, easeId?: string) { + // if this easing is being stopped to start another easing with + // the same id then don't fire any events to avoid extra start/stop events + if (this._easeId && easeId && this._easeId === easeId) { + return; + } + delete this._easeId; + + const wasZooming = this._zooming; + const wasRotating = this._rotating; + const wasPitching = this._pitching; + this._moving = false; + this._zooming = false; + this._rotating = false; + this._pitching = false; + this._padding = false; + + if (wasZooming) { + this.fire(new Event('zoomend', eventData)); + } + if (wasRotating) { + this.fire(new Event('rotateend', eventData)); + } + if (wasPitching) { + this.fire(new Event('pitchend', eventData)); + } + this.fire(new Event('moveend', eventData)); + } + + /** + * Changes any combination of center, zoom, bearing, and pitch, animating the transition along a curve that + * evokes flight. The animation seamlessly incorporates zooming and panning to help + * the user maintain her bearings even after traversing a great distance. + * + * Note: The animation will be skipped, and this will behave equivalently to `jumpTo` + * if the user has the `reduced motion` accessibility feature enabled in their operating system, + * unless 'options' includes `essential: true`. + * + * Triggers the following events: `movestart`, `move`, `moveend`, `zoomstart`, `zoom`, `zoomend`, `pitchstart`, + * `pitch`, `pitchend`, and `rotate`. + * + * @param options - Options describing the destination and animation of the transition. + * Accepts {@link CameraOptions}, {@link AnimationOptions}, + * and the following additional options. + * @param eventData - Additional properties to be added to event objects of events triggered by this method. + * @returns `this` + * @example + * ```ts + * // fly with default options to null island + * map.flyTo({center: [0, 0], zoom: 9}); + * // using flyTo options + * map.flyTo({ + * center: [0, 0], + * zoom: 9, + * speed: 0.2, + * curve: 1, + * easing(t) { + * return t; + * } + * }); + * ``` + * @see [Fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto/) + * @see [Slowly fly to a location](https://maplibre.org/maplibre-gl-js/docs/examples/flyto-options/) + * @see [Fly to a location based on scroll position](https://maplibre.org/maplibre-gl-js/docs/examples/scroll-fly-to/) + */ + flyTo(options: FlyToOptions, eventData?: any): this { + // Fall through to jumpTo if user has set prefers-reduced-motion + if (!options.essential && browser.prefersReducedMotion) { + const coercedOptions = pick(options, ['center', 'zoom', 'bearing', 'pitch', 'around']) as CameraOptions; + return this.jumpTo(coercedOptions, eventData); + } + + // This method implements an “optimal path” animation, as detailed in: + // + // Van Wijk, Jarke J.; Nuij, Wim A. A. “Smooth and efficient zooming and panning.” INFOVIS + // ’03. pp. 15–22. . + // + // Where applicable, local variable documentation begins with the associated variable or + // function in van Wijk (2003). + + this.stop(); + + options = extend({ + offset: [0, 0], + speed: 1.2, + curve: 1.42, + easing: defaultEasing + }, options); + + const tr = this._getTransformForUpdate(), + startZoom = this.getZoom(), + startBearing = this.getBearing(), + startPitch = this.getPitch(), + startPadding = this.getPadding(); + + const zoom = 'zoom' in options ? clamp(+options.zoom, tr.minZoom, tr.maxZoom) : startZoom; + const bearing = 'bearing' in options ? this._normalizeBearing(options.bearing, startBearing) : startBearing; + const pitch = 'pitch' in options ? +options.pitch : startPitch; + const padding = 'padding' in options ? options.padding : tr.padding; + + const scale = tr.zoomScale(zoom - startZoom); + const offsetAsPoint = Point.convert(options.offset); + let pointAtOffset = tr.centerPoint.add(offsetAsPoint); + const locationAtOffset = tr.pointLocation(pointAtOffset); + const center = LngLat.convert(options.center || locationAtOffset); + this._normalizeCenter(center); + + const from = tr.project(locationAtOffset); + const delta = tr.project(center).sub(from); + + let rho = options.curve; + + // w₀: Initial visible span, measured in pixels at the initial scale. + const w0 = Math.max(tr.width, tr.height), + // w₁: Final visible span, measured in pixels with respect to the initial scale. + w1 = w0 / scale, + // Length of the flight path as projected onto the ground plane, measured in pixels from + // the world image origin at the initial scale. + u1 = delta.mag(); + + if ('minZoom' in options) { + const minZoom = clamp(Math.min(options.minZoom, startZoom, zoom), tr.minZoom, tr.maxZoom); + // wm: Maximum visible span, measured in pixels with respect to the initial + // scale. + const wMax = w0 / tr.zoomScale(minZoom - startZoom); + rho = Math.sqrt(wMax / u1 * 2); + } + + // ρ² + const rho2 = rho * rho; + + /** + * rᵢ: Returns the zoom-out factor at one end of the animation. + * + * @param descent - `true` for the descent, `false` for the ascent + */ + function zoomOutFactor(descent: boolean) { + const b = (w1 * w1 - w0 * w0 + (descent ? -1 : 1) * rho2 * rho2 * u1 * u1) / (2 * (descent ? w1 : w0) * rho2 * u1); + return Math.log(Math.sqrt(b * b + 1) - b); + } + + function sinh(n) { return (Math.exp(n) - Math.exp(-n)) / 2; } + function cosh(n) { return (Math.exp(n) + Math.exp(-n)) / 2; } + function tanh(n) { return sinh(n) / cosh(n); } + + // r₀: Zoom-out factor during ascent. + const r0 = zoomOutFactor(false); + + // w(s): Returns the visible span on the ground, measured in pixels with respect to the + // initial scale. Assumes an angular field of view of 2 arctan ½ ≈ 53°. + let w: (_: number) => number = function (s) { + return (cosh(r0) / cosh(r0 + rho * s)); + }; + + // u(s): Returns the distance along the flight path as projected onto the ground plane, + // measured in pixels from the world image origin at the initial scale. + let u: (_: number) => number = function (s) { + return w0 * ((cosh(r0) * tanh(r0 + rho * s) - sinh(r0)) / rho2) / u1; + }; + + // S: Total length of the flight path, measured in ρ-screenfuls. + let S = (zoomOutFactor(true) - r0) / rho; + + // When u₀ = u₁, the optimal path doesn’t require both ascent and descent. + if (Math.abs(u1) < 0.000001 || !isFinite(S)) { + // Perform a more or less instantaneous transition if the path is too short. + if (Math.abs(w0 - w1) < 0.000001) return this.easeTo(options, eventData); + + const k = w1 < w0 ? -1 : 1; + S = Math.abs(Math.log(w1 / w0)) / rho; + + u = function() { return 0; }; + w = function(s) { return Math.exp(k * rho * s); }; + } + + if ('duration' in options) { + options.duration = +options.duration; + } else { + const V = 'screenSpeed' in options ? +options.screenSpeed / rho : +options.speed; + options.duration = 1000 * S / V; + } + + if (options.maxDuration && options.duration > options.maxDuration) { + options.duration = 0; + } + + this._zooming = true; + this._rotating = (startBearing !== bearing); + this._pitching = (pitch !== startPitch); + this._padding = !tr.isPaddingEqual(padding as PaddingOptions); + + this._prepareEase(eventData, false); + if (this.terrain) this._prepareElevation(center); + + this._ease((k) => { + // s: The distance traveled along the flight path, measured in ρ-screenfuls. + const s = k * S; + const scale = 1 / w(s); + tr.zoom = k === 1 ? zoom : startZoom + tr.scaleZoom(scale); + + if (this._rotating) { + tr.bearing = interpolates.number(startBearing, bearing, k); + } + if (this._pitching) { + tr.pitch = interpolates.number(startPitch, pitch, k); + } + if (this._padding) { + tr.interpolatePadding(startPadding, padding as PaddingOptions, k); + // When padding is being applied, Transform#centerPoint is changing continuously, + // thus we need to recalculate offsetPoint every frame + pointAtOffset = tr.centerPoint.add(offsetAsPoint); + } + + if (this.terrain && !options.freezeElevation) this._updateElevation(k); + + const newCenter = k === 1 ? center : tr.unproject(from.add(delta.mult(u(s))).mult(scale)); + tr.setLocationAtPoint(tr.renderWorldCopies ? newCenter.wrap() : newCenter, pointAtOffset); + + this._applyUpdatedTransform(tr); + + this._fireMoveEvents(eventData); + + }, () => { + if (this.terrain) this._finalizeElevation(); + this._afterEase(eventData); + }, options); + + return this; + } + + isEasing() { + return !!this._easeFrameId; + } + + /** + * Stops any animated transition underway. + * + * @returns `this` + */ + stop(): this { + return this._stop(); + } + + _stop(allowGestures?: boolean, easeId?: string): this { + if (this._easeFrameId) { + this._cancelRenderFrame(this._easeFrameId); + delete this._easeFrameId; + delete this._onEaseFrame; + } + + if (this._onEaseEnd) { + // The _onEaseEnd function might emit events which trigger new + // animation, which sets a new _onEaseEnd. Ensure we don't delete + // it unintentionally. + const onEaseEnd = this._onEaseEnd; + delete this._onEaseEnd; + onEaseEnd.call(this, easeId); + } + if (!allowGestures) { + const handlers = (this as any).handlers; + if (handlers) handlers.stop(false); + } + return this; + } + + _ease(frame: (_: number) => void, + finish: () => void, + options: { + animate?: boolean; + duration?: number; + easing?: (_: number) => number; + }) { + if (options.animate === false || options.duration === 0) { + frame(1); + finish(); + } else { + this._easeStart = browser.now(); + this._easeOptions = options; + this._onEaseFrame = frame; + this._onEaseEnd = finish; + this._easeFrameId = this._requestRenderFrame(this._renderFrameCallback); + } + } + + // Callback for map._requestRenderFrame + _renderFrameCallback = () => { + const t = Math.min((browser.now() - this._easeStart) / this._easeOptions.duration, 1); + this._onEaseFrame(this._easeOptions.easing(t)); + + // if _stop is called during _onEaseFrame from _fireMoveEvents we should avoid a new _requestRenderFrame, checking it by ensuring _easeFrameId was not deleted + if (t < 1 && this._easeFrameId) { + this._easeFrameId = this._requestRenderFrame(this._renderFrameCallback); + } else { + this.stop(); + } + }; + + // convert bearing so that it's numerically close to the current one so that it interpolates properly + _normalizeBearing(bearing: number, currentBearing: number) { + bearing = wrap(bearing, -180, 180); + const diff = Math.abs(bearing - currentBearing); + if (Math.abs(bearing - 360 - currentBearing) < diff) bearing -= 360; + if (Math.abs(bearing + 360 - currentBearing) < diff) bearing += 360; + return bearing; + } + + // If a path crossing the antimeridian would be shorter, extend the final coordinate so that + // interpolating between the two endpoints will cross it. + _normalizeCenter(center: LngLat) { + const tr = this.transform; + if (!tr.renderWorldCopies || tr.lngRange) return; + + const delta = center.lng - tr.center.lng; + center.lng += + delta > 180 ? -360 : + delta < -180 ? 360 : 0; + } + + /** + * Query the current elevation of location. It return null if terrain is not enabled. the elevation is in meters relative to mean sea-level + * @param lngLatLike - [x,y] or LngLat coordinates of the location + * @returns elevation in meters + */ + queryTerrainElevation(lngLatLike: LngLatLike): number | null { + if (!this.terrain) { + return null; + } + const elevation = this.terrain.getElevationForLngLatZoom(LngLat.convert(lngLatLike), this.transform.tileZoom); + /** + * Different zoomlevels with different terrain-tiles the elevation-values are not the same. + * map.transform.elevation variable with the center-altitude. + * In maplibre the proj-matrix is translated by this value in negative z-direction. + * So we need to add this value to the elevation to get the correct value. + */ + return elevation - this.transform.elevation; + } +} diff --git a/web/libraries/maplibre-gl/src/ui/control/attribution_control.test.ts b/web/libraries/maplibre-gl/src/ui/control/attribution_control.test.ts new file mode 100644 index 00000000..49940cc5 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/attribution_control.test.ts @@ -0,0 +1,543 @@ +import {AttributionControl} from './attribution_control'; +import {createMap as globalCreateMap, beforeMapTest} from '../../util/test/util'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {fakeServer} from 'nise'; + +function createMap() { + + return globalCreateMap({ + attributionControl: false, + style: { + version: 8, + sources: {}, + layers: [], + owner: 'mapblibre', + id: 'demotiles', + }, + hash: true + }, undefined); +} + +let map; + +beforeEach(() => { + beforeMapTest(); + map = createMap(); +}); + +afterEach(() => { + map.remove(); +}); + +describe('AttributionControl', () => { + test('appears in bottom-right by default', () => { + map.addControl(new AttributionControl()); + + expect( + map.getContainer().querySelectorAll('.maplibregl-ctrl-bottom-right .maplibregl-ctrl-attrib') + ).toHaveLength(1); + }); + + test('appears in the position specified by the position option', () => { + map.addControl(new AttributionControl(), 'top-left'); + + expect( + map.getContainer().querySelectorAll('.maplibregl-ctrl-top-left .maplibregl-ctrl-attrib') + ).toHaveLength(1); + }); + + test('appears in compact mode if compact option is used', () => { + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 700, configurable: true}); + + let attributionControl = new AttributionControl({ + compact: true, + customAttribution: 'MapLibre' + }); + map.addControl(attributionControl); + + const container = map.getContainer(); + + expect( + container.querySelectorAll('.maplibregl-ctrl-attrib.maplibregl-compact') + ).toHaveLength(1); + map.removeControl(attributionControl); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 600, configurable: true}); + attributionControl = new AttributionControl({ + compact: false + }); + + map.addControl(attributionControl); + expect( + container.querySelectorAll('.maplibregl-ctrl-attrib:not(.maplibregl-compact)') + ).toHaveLength(1); + }); + + test('appears in compact mode if container is less then 640 pixel wide and attributions are not empty', () => { + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 700, configurable: true}); + const attributionControl = new AttributionControl({ + customAttribution: 'MapLibre' + }); + map.addControl(attributionControl); + + const container = map.getContainer(); + + expect( + container.querySelectorAll('.maplibregl-ctrl-attrib:not(.maplibregl-compact)') + ).toHaveLength(1); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 600, configurable: true}); + map.resize(); + + expect( + container.querySelectorAll('.maplibregl-ctrl-attrib.maplibregl-compact') + ).toHaveLength(1); + + expect( + container.querySelectorAll('.maplibregl-attrib-empty') + ).toHaveLength(0); + }); + + test('does not appear in compact mode if container is less then 640 pixel wide and attributions are empty', () => { + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 700, configurable: true}); + map.addControl(new AttributionControl()); + + const container = map.getContainer(); + + expect( + container.querySelectorAll('.maplibregl-ctrl-attrib:not(.maplibregl-compact)') + ).toHaveLength(1); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 600, configurable: true}); + map.resize(); + + expect( + container.querySelectorAll('.maplibregl-ctrl-attrib.maplibregl-compact') + ).toHaveLength(0); + + expect( + container.querySelectorAll('.maplibregl-attrib-empty') + ).toHaveLength(1); + }); + + test('compact mode control toggles attribution', () => { + map.addControl(new AttributionControl({ + compact: true, + customAttribution: 'MapLibre' + })); + + const container = map.getContainer(); + const toggle = container.querySelector('.maplibregl-ctrl-attrib-button'); + + expect(container.querySelectorAll('.maplibregl-compact-show')).toHaveLength(1); + + simulate.click(toggle); + + expect(container.querySelectorAll('.maplibregl-compact-show')).toHaveLength(0); + + simulate.click(toggle); + + expect(container.querySelectorAll('.maplibregl-compact-show')).toHaveLength(1); + }); + + test('dedupes attributions that are substrings of others', done => { + const attribution = new AttributionControl(); + map.addControl(attribution); + + map.on('load', () => { + map.addSource('1', {type: 'geojson', data: {type: 'FeatureCollection', features: []}, attribution: 'World'}); + map.addSource('2', {type: 'geojson', data: {type: 'FeatureCollection', features: []}, attribution: 'Hello World'}); + map.addSource('3', {type: 'geojson', data: {type: 'FeatureCollection', features: []}, attribution: 'Another Source'}); + map.addSource('4', {type: 'geojson', data: {type: 'FeatureCollection', features: []}, attribution: 'Hello'}); + map.addSource('5', {type: 'geojson', data: {type: 'FeatureCollection', features: []}, attribution: 'Hello World'}); + map.addSource('6', {type: 'geojson', data: {type: 'FeatureCollection', features: []}, attribution: 'Hello World'}); + map.addSource('7', {type: 'geojson', data: {type: 'FeatureCollection', features: []}, attribution: 'GeoJSON Source'}); + map.addLayer({id: '1', type: 'fill', source: '1'}); + map.addLayer({id: '2', type: 'fill', source: '2'}); + map.addLayer({id: '3', type: 'fill', source: '3'}); + map.addLayer({id: '4', type: 'fill', source: '4'}); + map.addLayer({id: '5', type: 'fill', source: '5'}); + map.addLayer({id: '6', type: 'fill', source: '6'}); + map.addLayer({id: '7', type: 'fill', source: '7'}); + }); + + let times = 0; + map.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'visibility') { + if (++times === 7) { + expect(attribution._innerContainer.innerHTML).toBe('Hello World | Another Source | GeoJSON Source'); + done(); + } + } + }); + }); + + test('is hidden if empty', done => { + const attribution = new AttributionControl(); + map.addControl(attribution); + map.on('load', () => { + map.addSource('1', {type: 'geojson', data: {type: 'FeatureCollection', features: []}}); + map.addLayer({id: '1', type: 'fill', source: '1'}); + }); + + const container = map.getContainer(); + + const checkEmptyFirst = () => { + expect(attribution._innerContainer.innerHTML).toBe(''); + expect(container.querySelectorAll('.maplibregl-attrib-empty')).toHaveLength(1); + + map.addSource('2', {type: 'geojson', data: {type: 'FeatureCollection', features: []}, attribution: 'Hello World'}); + map.addLayer({id: '2', type: 'fill', source: '2'}); + }; + + const checkNotEmptyLater = () => { + expect(attribution._innerContainer.innerHTML).toBe('Hello World'); + expect(container.querySelectorAll('.maplibregl-attrib-empty')).toHaveLength(0); + done(); + }; + + let times = 0; + map.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'visibility') { + times++; + if (times === 1) { + checkEmptyFirst(); + } else if (times === 2) { + checkNotEmptyLater(); + } + } + }); + }); + + test('shows custom attribution if customAttribution option is provided', () => { + const attributionControl = new AttributionControl({ + customAttribution: 'Custom string' + }); + map.addControl(attributionControl); + + expect(attributionControl._innerContainer.innerHTML).toBe('Custom string'); + }); + + test('shows custom attribution if customAttribution option is provided, control is removed and added back', () => { + const attributionControl = new AttributionControl({ + customAttribution: 'Custom string' + }); + map.addControl(attributionControl); + map.removeControl(attributionControl); + map.addControl(attributionControl); + + expect(attributionControl._innerContainer.innerHTML).toBe('Custom string'); + }); + + test('in compact mode shows custom attribution if customAttribution option is provided', () => { + const attributionControl = new AttributionControl({ + customAttribution: 'Custom string', + compact: true + }); + map.addControl(attributionControl); + + expect(attributionControl._innerContainer.innerHTML).toBe('Custom string'); + }); + + test('shows all custom attributions if customAttribution array of strings is provided', () => { + const attributionControl = new AttributionControl({ + customAttribution: ['Some very long custom string', 'Custom string', 'Another custom string'] + }); + map.addControl(attributionControl); + + expect(attributionControl._innerContainer.innerHTML).toBe('Custom string | Another custom string | Some very long custom string'); + }); + + test('hides attributions for sources that are not currently visible', done => { + const attribution = new AttributionControl(); + map.addControl(attribution); + + map.on('load', () => { + map.addSource('1', {type: 'geojson', data: {type: 'FeatureCollection', features: []}, attribution: 'Used'}); + map.addSource('2', {type: 'geojson', data: {type: 'FeatureCollection', features: []}, attribution: 'Not used'}); + map.addSource('3', {type: 'geojson', data: {type: 'FeatureCollection', features: []}, attribution: 'Vibility none'}); + map.addLayer({id: '1', type: 'fill', source: '1'}); + map.addLayer({id: '3', type: 'fill', source: '3', layout: {visibility: 'none'}}); + }); + + let times = 0; + map.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'visibility') { + if (++times === 3) { + expect(attribution._innerContainer.innerHTML).toBe('Used'); + done(); + } + } + }); + }); + + test('does not show attributions for sources that are used for terrain when they are not in use', done => { + global.fetch = null; + const server = fakeServer.create(); + server.respondWith('/source.json', JSON.stringify({ + minzoom: 5, + maxzoom: 12, + attribution: 'Terrain', + tiles: ['http://example.com/{z}/{x}/{y}.pngraw'], + bounds: [-47, -7, -45, -5] + })); + + const attribution = new AttributionControl(); + map.addControl(attribution); + + map.on('load', () => { + map.addSource('1', {type: 'raster-dem', url: '/source.json'}); + server.respond(); + }); + + let times = 0; + map.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'visibility') { + if (++times === 1) { + expect(attribution._innerContainer.innerHTML).toBe(''); + done(); + } + } + }); + }); + + test('shows attributions for sources that are used for terrain', done => { + global.fetch = null; + const server = fakeServer.create(); + server.respondWith('/source.json', JSON.stringify({ + minzoom: 5, + maxzoom: 12, + attribution: 'Terrain', + tiles: ['http://example.com/{z}/{x}/{y}.pngraw'], + bounds: [-47, -7, -45, -5] + })); + + const attribution = new AttributionControl(); + map.addControl(attribution); + + map.on('load', () => { + map.addSource('1', {type: 'raster-dem', url: '/source.json'}); + server.respond(); + map.setTerrain({source: '1'}); + }); + + let times = 0; + map.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'visibility') { + if (++times === 1) { + expect(attribution._innerContainer.innerHTML).toBe('Terrain'); + done(); + } + } + }); + }); + + test('toggles attributions for sources whose visibility changes when zooming', done => { + const attribution = new AttributionControl(); + map.addControl(attribution); + + map.on('load', () => { + map.addSource('1', {type: 'geojson', data: {type: 'FeatureCollection', features: []}, attribution: 'Used'}); + map.addLayer({id: '1', type: 'fill', source: '1', minzoom: 12}); + }); + + map.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'metadata') { + expect(attribution._innerContainer.innerHTML).toBe(''); + map.setZoom(13); + } + if (e.dataType === 'source' && e.sourceDataType === 'visibility') { + if (map.getZoom() === 13) { + expect(attribution._innerContainer.innerHTML).toBe('Used'); + done(); + } + } + }); + }); + +}); + +describe('AttributionControl test regarding the HTML elements details and summary', () => { + describe('Details is set correct for compact view', () => { + test('It should NOT contain the attribute open="" on first load.', () => { + const attributionControl = new AttributionControl({ + compact: true + }); + map.addControl(attributionControl); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBeNull(); + }); + + test('It SHOULD contain the attribute open="" after click on summary.', () => { + const attributionControl = new AttributionControl({ + compact: true + }); + map.addControl(attributionControl); + const container = map.getContainer(); + const toggle = container.querySelector('.maplibregl-ctrl-attrib-button'); + + simulate.click(toggle); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + }); + + test('It should NOT contain the attribute open="" after two clicks on summary.', () => { + const attributionControl = new AttributionControl({ + compact: true + }); + map.addControl(attributionControl); + const container = map.getContainer(); + const toggle = container.querySelector('.maplibregl-ctrl-attrib-button'); + + simulate.click(toggle); + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + + simulate.click(toggle); + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBeNull(); + }); + }); + + describe('Details is set correct for default view (compact === undefined)', () => { + test('It should NOT contain the attribute open="" if offsetWidth <= 640.', () => { + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 640, configurable: true}); + const attributionControl = new AttributionControl({ + }); + map.addControl(attributionControl); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBeNull(); + }); + + test('It SHOULD contain the attribute open="" if offsetWidth > 640.', () => { + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 641, configurable: true}); + const attributionControl = new AttributionControl({ + }); + map.addControl(attributionControl); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + }); + + test('The attribute open="" SHOULD exist after resize from size > 640 to <= 640 and and vice versa.', () => { + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 640, configurable: true}); + const attributionControl = new AttributionControl({ + customAttribution: 'MapLibre' + }); + map.addControl(attributionControl); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib.maplibregl-compact')).toHaveLength(1); + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 641, configurable: true}); + map.resize(); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib:not(.maplibregl-compact)')).toHaveLength(1); + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 640, configurable: true}); + map.resize(); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib.maplibregl-compact')).toHaveLength(1); + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + }); + + test('The attribute open="" should NOT change on resize from > 640 to another > 640.', () => { + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 641, configurable: true}); + const attributionControl = new AttributionControl({ + }); + map.addControl(attributionControl); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 650, configurable: true}); + map.resize(); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 641, configurable: true}); + map.resize(); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + }); + + test('The attribute open="" should NOT change on resize from <= 640 to another <= 640 if it is closed.', () => { + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 640, configurable: true}); + const attributionControl = new AttributionControl({ + }); + map.addControl(attributionControl); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBeNull(); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 630, configurable: true}); + map.resize(); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBeNull(); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 640, configurable: true}); + map.resize(); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBeNull(); + }); + + test('The attribute open="" should NOT change on resize from <= 640 to another <= 640 if it is open.', () => { + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 640, configurable: true}); + const attributionControl = new AttributionControl({ + }); + map.addControl(attributionControl); + const toggle = map.getContainer().querySelector('.maplibregl-ctrl-attrib-button'); + simulate.click(toggle); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 630, configurable: true}); + map.resize(); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 640, configurable: true}); + map.resize(); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + }); + }); + + describe('Details is set correct for default view (compact === false)', () => { + test('It SHOULD contain the attribute open="" if offsetWidth <= 640.', () => { + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 640, configurable: true}); + const attributionControl = new AttributionControl({ + compact: false + }); + map.addControl(attributionControl); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + }); + + test('It SHOULD contain the attribute open="" if offsetWidth > 640.', () => { + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 641, configurable: true}); + const attributionControl = new AttributionControl({ + compact: false + }); + map.addControl(attributionControl); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + }); + + test('The attribute open="" should NOT change on resize.', () => { + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 640, configurable: true}); + const attributionControl = new AttributionControl({ + compact: false + }); + map.addControl(attributionControl); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 641, configurable: true}); + map.resize(); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 640, configurable: true}); + map.resize(); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-attrib')[0].getAttribute('open')).toBe(''); + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/control/attribution_control.ts b/web/libraries/maplibre-gl/src/ui/control/attribution_control.ts new file mode 100644 index 00000000..491e3559 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/attribution_control.ts @@ -0,0 +1,204 @@ +import {DOM} from '../../util/dom'; + +import type {Map} from '../map'; +import type {ControlPosition, IControl} from './control'; +import type {MapDataEvent} from '../events'; +import type {StyleSpecification} from '@maplibre/maplibre-gl-style-spec'; +/** + * The {@link AttributionControl} options + */ +type AttributionOptions = { + /** + * If `true`, the attribution control will always collapse when moving the map. If `false`, + * force the expanded attribution control. The default is a responsive attribution that collapses when the user moves the map on maps less than 640 pixels wide. + * **Attribution should not be collapsed if it can comfortably fit on the map. `compact` should only be used to modify default attribution when map size makes it impossible to fit default attribution and when the automatic compact resizing for default settings are not sufficient.** + */ + compact?: boolean; + /** + * Attributions to show in addition to any other attributions. + */ + customAttribution?: string | Array; +}; + +/** + * An `AttributionControl` control presents the map's attribution information. By default, the attribution control is expanded (regardless of map width). + * @group Markers and Controls + * @example + * ```ts + * let map = new maplibregl.Map({attributionControl: false}) + * .addControl(new maplibregl.AttributionControl({ + * compact: true + * })); + * ``` + */ +export class AttributionControl implements IControl { + options: AttributionOptions; + _map: Map; + _compact: boolean; + _container: HTMLElement; + _innerContainer: HTMLElement; + _compactButton: HTMLElement; + _editLink: HTMLAnchorElement; + _attribHTML: string; + styleId: string; + styleOwner: string; + + /** + * @param options - the attribution options + */ + constructor(options: AttributionOptions = {}) { + this.options = options; + } + + getDefaultPosition(): ControlPosition { + return 'bottom-right'; + } + + /** {@inheritDoc IControl.onAdd} */ + onAdd(map: Map) { + this._map = map; + this._compact = this.options && this.options.compact; + this._container = DOM.create('details', 'maplibregl-ctrl maplibregl-ctrl-attrib'); + this._compactButton = DOM.create('summary', 'maplibregl-ctrl-attrib-button', this._container); + this._compactButton.addEventListener('click', this._toggleAttribution); + this._setElementTitle(this._compactButton, 'ToggleAttribution'); + this._innerContainer = DOM.create('div', 'maplibregl-ctrl-attrib-inner', this._container); + + this._updateAttributions(); + this._updateCompact(); + + this._map.on('styledata', this._updateData); + this._map.on('sourcedata', this._updateData); + this._map.on('terrain', this._updateData); + this._map.on('resize', this._updateCompact); + this._map.on('drag', this._updateCompactMinimize); + + return this._container; + } + + /** {@inheritDoc IControl.onRemove} */ + onRemove() { + DOM.remove(this._container); + + this._map.off('styledata', this._updateData); + this._map.off('sourcedata', this._updateData); + this._map.off('terrain', this._updateData); + this._map.off('resize', this._updateCompact); + this._map.off('drag', this._updateCompactMinimize); + + this._map = undefined; + this._compact = undefined; + this._attribHTML = undefined; + } + + _setElementTitle(element: HTMLElement, title: string) { + const str = this._map._getUIString(`AttributionControl.${title}`); + element.title = str; + element.setAttribute('aria-label', str); + } + + _toggleAttribution = () => { + if (this._container.classList.contains('maplibregl-compact')) { + if (this._container.classList.contains('maplibregl-compact-show')) { + this._container.setAttribute('open', ''); + this._container.classList.remove('maplibregl-compact-show'); + } else { + this._container.classList.add('maplibregl-compact-show'); + this._container.removeAttribute('open'); + } + } + }; + + _updateData = (e: MapDataEvent) => { + if (e && (e.sourceDataType === 'metadata' || e.sourceDataType === 'visibility' || e.dataType === 'style' || e.type === 'terrain')) { + this._updateAttributions(); + } + }; + + _updateAttributions() { + if (!this._map.style) return; + let attributions: Array = []; + if (this.options.customAttribution) { + if (Array.isArray(this.options.customAttribution)) { + attributions = attributions.concat( + this.options.customAttribution.map(attribution => { + if (typeof attribution !== 'string') return ''; + return attribution; + }) + ); + } else if (typeof this.options.customAttribution === 'string') { + attributions.push(this.options.customAttribution); + } + } + + if (this._map.style.stylesheet) { + const stylesheet = this._map.style.stylesheet as StyleSpecification & { owner: string; id: string }; + this.styleOwner = stylesheet.owner; + this.styleId = stylesheet.id; + } + + const sourceCaches = this._map.style.sourceCaches; + for (const id in sourceCaches) { + const sourceCache = sourceCaches[id]; + if (sourceCache.used || sourceCache.usedForTerrain) { + const source = sourceCache.getSource(); + if (source.attribution && attributions.indexOf(source.attribution) < 0) { + attributions.push(source.attribution); + } + } + } + + // remove any entries that are whitespace + attributions = attributions.filter(e => String(e).trim()); + + // remove any entries that are substrings of another entry. + // first sort by length so that substrings come first + attributions.sort((a, b) => a.length - b.length); + attributions = attributions.filter((attrib, i) => { + for (let j = i + 1; j < attributions.length; j++) { + if (attributions[j].indexOf(attrib) >= 0) { return false; } + } + return true; + }); + + // check if attribution string is different to minimize DOM changes + const attribHTML = attributions.join(' | '); + if (attribHTML === this._attribHTML) return; + + this._attribHTML = attribHTML; + + if (attributions.length) { + this._innerContainer.innerHTML = attribHTML; + this._container.classList.remove('maplibregl-attrib-empty'); + } else { + this._container.classList.add('maplibregl-attrib-empty'); + } + this._updateCompact(); + // remove old DOM node from _editLink + this._editLink = null; + } + + _updateCompact = () => { + if (this._map.getCanvasContainer().offsetWidth <= 640 || this._compact) { + if (this._compact === false) { + this._container.setAttribute('open', ''); + } else if (!this._container.classList.contains('maplibregl-compact') && !this._container.classList.contains('maplibregl-attrib-empty')) { + this._container.setAttribute('open', ''); + this._container.classList.add('maplibregl-compact', 'maplibregl-compact-show'); + } + } else { + this._container.setAttribute('open', ''); + if (this._container.classList.contains('maplibregl-compact')) { + this._container.classList.remove('maplibregl-compact', 'maplibregl-compact-show'); + } + } + }; + + _updateCompactMinimize = () => { + if (this._container.classList.contains('maplibregl-compact')) { + if (this._container.classList.contains('maplibregl-compact-show')) { + this._container.classList.remove('maplibregl-compact-show'); + } + } + }; +} diff --git a/web/libraries/maplibre-gl/src/ui/control/control.ts b/web/libraries/maplibre-gl/src/ui/control/control.ts new file mode 100644 index 00000000..eb715d66 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/control.ts @@ -0,0 +1,84 @@ +import type {Map} from '../map'; + +/** + * A position defintion for the control to be placed, can be in one of the corners of the map. + * When two or more controls are places in the same location they are stacked toward the center of the map. + */ +export type ControlPosition = 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right'; + +/** + * Interface for interactive controls added to the map. This is a + * specification for implementers to model: it is not + * an exported method or class. + * + * Controls must implement `onAdd` and `onRemove`, and must own an + * element, which is often a `div` element. To use MapLibre GL JS's + * default control styling, add the `maplibregl-ctrl` class to your control's + * node. + * + * @example + * Control implemented as ES6 class + * ```ts + * class HelloWorldControl { + * onAdd(map) { + * this._map = map; + * this._container = document.createElement('div'); + * this._container.className = 'maplibregl-ctrl'; + * this._container.textContent = 'Hello, world'; + * return this._container; + * } + * + * onRemove() { + * this._container.parentNode.removeChild(this._container); + * this._map = undefined; + * } + * } + * + * // Control implemented as ES5 prototypical class + * function HelloWorldControl() { } + * + * HelloWorldControl.prototype.onAdd = function(map) { + * this._map = map; + * this._container = document.createElement('div'); + * this._container.className = 'maplibregl-ctrl'; + * this._container.textContent = 'Hello, world'; + * return this._container; + * }; + * + * HelloWorldControl.prototype.onRemove = function () { + * this._container.parentNode.removeChild(this._container); + * this._map = undefined; + * }; + * ``` + */ +export interface IControl { + /** + * Register a control on the map and give it a chance to register event listeners + * and resources. This method is called by {@link Map#addControl} + * internally. + * + * @param map - the Map this control will be added to + * @returns The control's container element. This should + * be created by the control and returned by onAdd without being attached + * to the DOM: the map will insert the control's element into the DOM + * as necessary. + */ + onAdd(map: Map): HTMLElement; + /** + * Unregister a control on the map and give it a chance to detach event listeners + * and resources. This method is called by {@link Map#removeControl} + * internally. + * + * @param map - the Map this control will be removed from + */ + onRemove(map: Map): void; + /** + * Optionally provide a default position for this control. If this method + * is implemented and {@link Map#addControl} is called without the `position` + * parameter, the value returned by getDefaultPosition will be used as the + * control's position. + * + * @returns a control position, one of the values valid in addControl. + */ + readonly getDefaultPosition?: () => ControlPosition; +} diff --git a/web/libraries/maplibre-gl/src/ui/control/fullscreen_control.test.ts b/web/libraries/maplibre-gl/src/ui/control/fullscreen_control.test.ts new file mode 100644 index 00000000..5b656715 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/fullscreen_control.test.ts @@ -0,0 +1,136 @@ +import {createMap, beforeMapTest} from '../../util/test/util'; +import {FullscreenControl} from './fullscreen_control'; + +beforeEach(() => { + beforeMapTest(); +}); + +describe('FullscreenControl', () => { + test('renders control', () => { + Object.defineProperty(window.document, 'fullscreenEnabled', { + value: true, + writable: true, + }); + const map = createMap(undefined, undefined); + const fullscreen = new FullscreenControl({}); + map.addControl(fullscreen); + + expect(map.getContainer().querySelectorAll('.maplibregl-ctrl-fullscreen')).toHaveLength(1); + }); + + test('makes optional container element full screen', () => { + Object.defineProperty(window.document, 'fullscreenEnabled', { + value: true, + writable: true, + }); + + const map = createMap(undefined, undefined); + const container = window.document.querySelector('body')!; + const fullscreen = new FullscreenControl({container}); + map.addControl(fullscreen); + const control = map._controls.find((ctrl) => { + return Object.prototype.hasOwnProperty.call(ctrl, '_fullscreen'); + }) as FullscreenControl; + control._onClickFullscreen(); + + expect(control._container.tagName).toBe('BODY'); + }); + + test('uses pseudo fullscreen when fullscreen is not supported', () => { + const map = createMap(undefined, undefined); + const mapContainer = map.getContainer(); + + const fullscreen = new FullscreenControl({}); + map.addControl(fullscreen); + const control = map._controls.find((ctrl) => { + return Object.prototype.hasOwnProperty.call(ctrl, '_fullscreen'); + }) as FullscreenControl; + + expect(mapContainer.classList.contains('maplibregl-pseudo-fullscreen')).toBe(false); + control._onClickFullscreen(); + expect(mapContainer.classList.contains('maplibregl-pseudo-fullscreen')).toBe(true); + control._onClickFullscreen(); + expect(mapContainer.classList.contains('maplibregl-pseudo-fullscreen')).toBe(false); + }); + + test('start and end events fire for fullscreen button clicks', () => { + const map = createMap(undefined, undefined); + const fullscreen = new FullscreenControl({}); + + const fullscreenstart = jest.fn(); + const fullscreenend = jest.fn(); + + fullscreen.on('fullscreenstart', fullscreenstart); + fullscreen.on('fullscreenend', fullscreenend); + + map.addControl(fullscreen); + + const click = new window.Event('click'); + + // Simulate a click to the fullscreen button + fullscreen._fullscreenButton.dispatchEvent(click); + expect(fullscreenstart).toHaveBeenCalled(); + expect(fullscreenend).not.toHaveBeenCalled(); + + // Second simulated click would exit fullscreen mode + fullscreen._fullscreenButton.dispatchEvent(click); + expect(fullscreenend).toHaveBeenCalled(); + }); + + test('disables cooperative gestures when fullscreen becomes active', () => { + const cooperativeGestures = true; + const map = createMap({cooperativeGestures}); + const fullscreen = new FullscreenControl({}); + + map.addControl(fullscreen); + + const click = new window.Event('click'); + + // Simulate a click to the fullscreen button + fullscreen._fullscreenButton.dispatchEvent(click); + expect(map.getCooperativeGestures()).toBeFalsy(); + + // Second simulated click would exit fullscreen mode + fullscreen._fullscreenButton.dispatchEvent(click); + expect(map.getCooperativeGestures()).toBe(cooperativeGestures); + }); + + test('reenables cooperative gestures custom options when fullscreen exits', () => { + const cooperativeGestures = { + 'windowsHelpText': 'Custom message', + 'macHelpText': 'Custom message', + 'mobileHelpText': 'Custom message', + }; + const map = createMap({cooperativeGestures}); + const fullscreen = new FullscreenControl({}); + + map.addControl(fullscreen); + + const click = new window.Event('click'); + + // Simulate a click to the fullscreen button + fullscreen._fullscreenButton.dispatchEvent(click); + expect(map.getCooperativeGestures()).toBeFalsy(); + + // Second simulated click would exit fullscreen mode + fullscreen._fullscreenButton.dispatchEvent(click); + expect(map.getCooperativeGestures()).toEqual(cooperativeGestures); + }); + + test('if never set, cooperative gestures remain disabled when fullscreen exits', () => { + const map = createMap({cooperativeGestures: false}); + const fullscreen = new FullscreenControl({}); + + map.addControl(fullscreen); + + const click = new window.Event('click'); + + // Simulate a click to the fullscreen button + fullscreen._fullscreenButton.dispatchEvent(click); + expect(map.getCooperativeGestures()).toBeFalsy(); + + // Second simulated click would exit fullscreen mode + fullscreen._fullscreenButton.dispatchEvent(click); + expect(map.getCooperativeGestures()).toBeFalsy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/control/fullscreen_control.ts b/web/libraries/maplibre-gl/src/ui/control/fullscreen_control.ts new file mode 100644 index 00000000..fb7af04a --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/fullscreen_control.ts @@ -0,0 +1,185 @@ +import {DOM} from '../../util/dom'; + +import {warnOnce} from '../../util/util'; + +import {Event, Evented} from '../../util/evented'; +import type {Map, GestureOptions} from '../map'; +import type {IControl} from './control'; + +/** + * The {@link FullscreenControl} options + */ +type FullscreenOptions = { + /** + * `container` is the [compatible DOM element](https://developer.mozilla.org/en-US/docs/Web/API/Element/requestFullScreen#Compatible_elements) which should be made full screen. By default, the map container element will be made full screen. + */ + container?: HTMLElement; +}; + +/** + * A `FullscreenControl` control contains a button for toggling the map in and out of fullscreen mode. + * When [requestFullscreen](https://developer.mozilla.org/en-US/docs/Web/API/Element/requestFullscreen) is not supported, fullscreen is handled via CSS properties. + * The map's `cooperativeGestures` option is temporarily disabled while the map + * is in fullscreen mode, and is restored when the map exist fullscreen mode. + * + * @group Markers and Controls + * @param options - the full screen control options + * + * @example + * ```ts + * map.addControl(new maplibregl.FullscreenControl({container: document.querySelector('body')})); + * ``` + * @see [View a fullscreen map](https://maplibre.org/maplibre-gl-js/docs/examples/fullscreen/) + * + * ### Events + * + * @event `fullscreenstart` - Fired when fullscreen mode has started + * + * @event `fullscreenend` - Fired when fullscreen mode has ended + */ +export class FullscreenControl extends Evented implements IControl { + _map: Map; + _controlContainer: HTMLElement; + _fullscreen: boolean; + _fullscreenchange: string; + _fullscreenButton: HTMLButtonElement; + _container: HTMLElement; + _prevCooperativeGestures: boolean | GestureOptions; + + constructor(options: FullscreenOptions = {}) { + super(); + this._fullscreen = false; + + if (options && options.container) { + if (options.container instanceof HTMLElement) { + this._container = options.container; + } else { + warnOnce('Full screen control \'container\' must be a DOM element.'); + } + } + + if ('onfullscreenchange' in document) { + this._fullscreenchange = 'fullscreenchange'; + } else if ('onmozfullscreenchange' in document) { + this._fullscreenchange = 'mozfullscreenchange'; + } else if ('onwebkitfullscreenchange' in document) { + this._fullscreenchange = 'webkitfullscreenchange'; + } else if ('onmsfullscreenchange' in document) { + this._fullscreenchange = 'MSFullscreenChange'; + } + } + + /** {@inheritDoc IControl.onAdd} */ + onAdd(map: Map): HTMLElement { + this._map = map; + if (!this._container) this._container = this._map.getContainer(); + this._controlContainer = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group'); + this._setupUI(); + return this._controlContainer; + } + + /** {@inheritDoc IControl.onRemove} */ + onRemove() { + DOM.remove(this._controlContainer); + this._map = null; + window.document.removeEventListener(this._fullscreenchange, this._onFullscreenChange); + } + + _setupUI() { + const button = this._fullscreenButton = DOM.create('button', (('maplibregl-ctrl-fullscreen')), this._controlContainer); + DOM.create('span', 'maplibregl-ctrl-icon', button).setAttribute('aria-hidden', 'true'); + button.type = 'button'; + this._updateTitle(); + this._fullscreenButton.addEventListener('click', this._onClickFullscreen); + window.document.addEventListener(this._fullscreenchange, this._onFullscreenChange); + } + + _updateTitle() { + const title = this._getTitle(); + this._fullscreenButton.setAttribute('aria-label', title); + this._fullscreenButton.title = title; + } + + _getTitle() { + return this._map._getUIString(this._isFullscreen() ? 'FullscreenControl.Exit' : 'FullscreenControl.Enter'); + } + + _isFullscreen() { + return this._fullscreen; + } + + _onFullscreenChange = () => { + const fullscreenElement = + window.document.fullscreenElement || + (window.document as any).mozFullScreenElement || + (window.document as any).webkitFullscreenElement || + (window.document as any).msFullscreenElement; + + if ((fullscreenElement === this._container) !== this._fullscreen) { + this._handleFullscreenChange(); + } + }; + + _handleFullscreenChange() { + this._fullscreen = !this._fullscreen; + this._fullscreenButton.classList.toggle('maplibregl-ctrl-shrink'); + this._fullscreenButton.classList.toggle('maplibregl-ctrl-fullscreen'); + this._updateTitle(); + + if (this._fullscreen) { + this.fire(new Event('fullscreenstart')); + if (this._map._cooperativeGestures) { + this._prevCooperativeGestures = this._map._cooperativeGestures; + this._map.setCooperativeGestures(); + } + } else { + this.fire(new Event('fullscreenend')); + if (this._prevCooperativeGestures) { + this._map.setCooperativeGestures(this._prevCooperativeGestures); + delete this._prevCooperativeGestures; + } + } + } + + _onClickFullscreen = () => { + if (this._isFullscreen()) { + this._exitFullscreen(); + } else { + this._requestFullscreen(); + } + }; + + _exitFullscreen() { + if (window.document.exitFullscreen) { + (window.document as any).exitFullscreen(); + } else if ((window.document as any).mozCancelFullScreen) { + (window.document as any).mozCancelFullScreen(); + } else if ((window.document as any).msExitFullscreen) { + (window.document as any).msExitFullscreen(); + } else if ((window.document as any).webkitCancelFullScreen) { + (window.document as any).webkitCancelFullScreen(); + } else { + this._togglePseudoFullScreen(); + } + } + + _requestFullscreen() { + if (this._container.requestFullscreen) { + this._container.requestFullscreen(); + } else if ((this._container as any).mozRequestFullScreen) { + (this._container as any).mozRequestFullScreen(); + } else if ((this._container as any).msRequestFullscreen) { + (this._container as any).msRequestFullscreen(); + } else if ((this._container as any).webkitRequestFullscreen) { + (this._container as any).webkitRequestFullscreen(); + } else { + this._togglePseudoFullScreen(); + } + } + + _togglePseudoFullScreen() { + this._container.classList.toggle('maplibregl-pseudo-fullscreen'); + this._handleFullscreenChange(); + this._map.resize(); + } +} diff --git a/web/libraries/maplibre-gl/src/ui/control/geolocate_control.test.ts b/web/libraries/maplibre-gl/src/ui/control/geolocate_control.test.ts new file mode 100644 index 00000000..00b3186f --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/geolocate_control.test.ts @@ -0,0 +1,520 @@ +import geolocation from 'mock-geolocation'; +import {LngLatBounds} from '../../geo/lng_lat_bounds'; +import {createMap, beforeMapTest} from '../../util/test/util'; +import {GeolocateControl} from './geolocate_control'; +jest.mock('../../util/geolocation_support', () => ( + { + checkGeolocationSupport: jest.fn() + } +)); +import {checkGeolocationSupport} from '../../util/geolocation_support'; +import type {LngLat} from '../../geo/lng_lat'; + +/** + * Convert the coordinates of a LngLat object to a fixed number of digits + * @param lngLat - the location + * @param digits - digits the number of digits to set + * @returns a string representation of the object with the required number of digits + */ +function lngLatAsFixed(lngLat: LngLat, digits: number): {lat: string; lng: string} { + return { + lng: lngLat.lng.toFixed(digits), + lat: lngLat.lat.toFixed(digits) + }; +} + +describe('GeolocateControl with no options', () => { + geolocation.use(); + let map; + + beforeEach(() => { + beforeMapTest(); + map = createMap(undefined, undefined); + (checkGeolocationSupport as any as jest.SpyInstance).mockImplementationOnce((cb) => cb(true)); + }); + + afterEach(() => { + map.remove(); + }); + + test('is disabled when there\'s no support', async () => { + (checkGeolocationSupport as any as jest.SpyInstance).mockReset().mockImplementationOnce((cb) => cb(false)); + const geolocate = new GeolocateControl(undefined); + map.addControl(geolocate); + expect(geolocate._geolocateButton.disabled).toBeTruthy(); + }); + + test('is enabled when there no support', async () => { + const geolocate = new GeolocateControl(undefined); + map.addControl(geolocate); + expect(geolocate._geolocateButton.disabled).toBeFalsy(); + }); + + test('has permissions', async () => { + + (window.navigator as any).permissions = { + query: () => Promise.resolve({state: 'granted'}) + }; + + const geolocate = new GeolocateControl(undefined); + map.addControl(geolocate); + + await new Promise(process.nextTick); + expect(geolocate._geolocateButton.disabled).toBeFalsy(); + }); + + test('error event', done => { + const geolocate = new GeolocateControl(undefined); + map.addControl(geolocate); + + const click = new window.Event('click'); + + geolocate.on('error', (error) => { + expect(error.code).toBe(2); + expect(error.message).toBe('error message'); + done(); + }); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.sendError({code: 2, message: 'error message'}); + }); + + test('does not throw if removed quickly', done => { + (checkGeolocationSupport as any as jest.SpyInstance).mockReset() + .mockImplementationOnce((cb) => { + return Promise.resolve(true) + .then(result => { + expect(() => cb(result)).not.toThrow(); + }) + .finally(done); + }); + + const geolocate = new GeolocateControl(undefined); + map.addControl(geolocate); + map.removeControl(geolocate); + }); + + test('outofmaxbounds event in active lock state', done => { + const geolocate = new GeolocateControl(undefined); + map.addControl(geolocate); + map.setMaxBounds([[0, 0], [10, 10]]); + geolocate._watchState = 'ACTIVE_LOCK'; + + const click = new window.Event('click'); + + geolocate.on('outofmaxbounds', (position) => { + expect(geolocate._watchState).toBe('ACTIVE_ERROR'); + expect(position.coords.latitude).toBe(10); + expect(position.coords.longitude).toBe(20); + expect(position.coords.accuracy).toBe(3); + expect(position.timestamp).toBe(4); + done(); + }); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.send({latitude: 10, longitude: 20, accuracy: 3, timestamp: 4}); + }); + + test('outofmaxbounds event in background state', done => { + const geolocate = new GeolocateControl(undefined); + map.addControl(geolocate); + map.setMaxBounds([[0, 0], [10, 10]]); + geolocate._watchState = 'BACKGROUND'; + + const click = new window.Event('click'); + + geolocate.on('outofmaxbounds', (position) => { + expect(geolocate._watchState).toBe('BACKGROUND_ERROR'); + expect(position.coords.latitude).toBe(10); + expect(position.coords.longitude).toBe(20); + expect(position.coords.accuracy).toBe(3); + expect(position.timestamp).toBe(4); + done(); + }); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.send({latitude: 10, longitude: 20, accuracy: 3, timestamp: 4}); + }); + + test('geolocate event', done => { + const geolocate = new GeolocateControl(undefined); + map.addControl(geolocate); + + const click = new window.Event('click'); + + geolocate.on('geolocate', (position) => { + expect(position.coords.latitude).toBe(10); + expect(position.coords.longitude).toBe(20); + expect(position.coords.accuracy).toBe(30); + expect(position.timestamp).toBe(40); + done(); + }); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.send({latitude: 10, longitude: 20, accuracy: 30, timestamp: 40}); + }); + + test('trigger', () => { + const geolocate = new GeolocateControl(undefined); + map.addControl(geolocate); + + expect(geolocate.trigger()).toBeTruthy(); + }); + + test('trigger and then error when tracking user location should get to active error state', () => { + const geolocate = new GeolocateControl({trackUserLocation: true}); + map.addControl(geolocate); + + geolocate.trigger(); + geolocation.sendError({code: 2, message: 'error message'}); + expect(geolocate._watchState).toBe('ACTIVE_ERROR'); + expect(geolocate._geolocateButton.classList.contains('maplibregl-ctrl-geolocate-active-error')).toBeTruthy(); + }); + + test('trigger before added to map', () => { + jest.spyOn(console, 'warn').mockImplementation(() => { }); + + const geolocate = new GeolocateControl(undefined); + + expect(geolocate.trigger()).toBeFalsy(); + expect(console.warn).toHaveBeenCalledWith('Geolocate control triggered before added to a map'); + }); + + test('geolocate fitBoundsOptions', async () => { + const geolocate = new GeolocateControl({ + fitBoundsOptions: { + duration: 0, + maxZoom: 10 + } + }); + map.addControl(geolocate); + + const click = new window.Event('click'); + + const moveEndPromise = map.once('moveend'); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.send({latitude: 10, longitude: 20, accuracy: 1}); + await moveEndPromise; + expect(map.getZoom()).toBe(10); + }); + + test('with removed before Geolocation callback', () => { + expect(() => { + const geolocate = new GeolocateControl(undefined); + map.addControl(geolocate); + geolocate.trigger(); + map.removeControl(geolocate); + }).not.toThrow(); + }); + + test('non-zero bearing', async () => { + map.setBearing(45); + const geolocate = new GeolocateControl({ + fitBoundsOptions: { + duration: 0, + maxZoom: 10 + } + }); + map.addControl(geolocate); + + const click = new window.Event('click'); + + const moveEndPromise = map.once('moveend'); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.send({latitude: 10, longitude: 20, accuracy: 1}); + await moveEndPromise; + expect(lngLatAsFixed(map.getCenter(), 4)).toEqual({lat: '10.0000', lng: '20.0000'}); + expect(map.getBearing()).toBe(45); + expect(map.getZoom()).toBe(10); + }); + + test('no watching map camera on geolocation', async () => { + const geolocate = new GeolocateControl({ + fitBoundsOptions: { + maxZoom: 20, + duration: 0 + } + }); + map.addControl(geolocate); + + const click = new window.Event('click'); + + const moveEndPromise = map.once('moveend'); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.send({latitude: 10, longitude: 20, accuracy: 1000}); + await moveEndPromise; + const mapCenter = map.getCenter(); + expect(lngLatAsFixed(mapCenter, 4)).toEqual({lat: '10.0000', lng: '20.0000'}); + + const mapBounds = map.getBounds(); + + // map bounds should contain or equal accuracy bounds, that is the ensure accuracy bounds doesn't fall outside the map bounds + const accuracyBounds = LngLatBounds.fromLngLat(mapCenter, 1000); + + expect(accuracyBounds.getNorth().toFixed(4) <= mapBounds.getNorth().toFixed(4)).toBeTruthy(); + expect(accuracyBounds.getSouth().toFixed(4) >= mapBounds.getSouth().toFixed(4)).toBeTruthy(); + expect(accuracyBounds.getEast().toFixed(4) <= mapBounds.getEast().toFixed(4)).toBeTruthy(); + expect(accuracyBounds.getWest().toFixed(4) >= mapBounds.getWest().toFixed(4)).toBeTruthy(); + + // map bounds should not be too much bigger on all edges of the accuracy bounds (this test will only work for an orthogonal bearing), + // ensures map bounds does not contain buffered accuracy bounds, as if it does there is too much gap around the accuracy bounds + const bufferedAccuracyBounds = LngLatBounds.fromLngLat(mapCenter, 1100); + + expect( + (bufferedAccuracyBounds.getNorth().toFixed(4) < mapBounds.getNorth().toFixed(4)) && + (bufferedAccuracyBounds.getSouth().toFixed(4) > mapBounds.getSouth().toFixed(4)) && + (bufferedAccuracyBounds.getEast().toFixed(4) < mapBounds.getEast().toFixed(4)) && + (bufferedAccuracyBounds.getWest().toFixed(4) > mapBounds.getWest().toFixed(4)) + ).toBeFalsy(); + }); + + test('watching map updates recenter on location with dot', done => { + const geolocate = new GeolocateControl({ + trackUserLocation: true, + showUserLocation: true, + fitBoundsOptions: { + duration: 0 + } + }); + map.addControl(geolocate); + + const click = new window.Event('click'); + + let moveendCount = 0; + map.once('moveend', () => { + // moveend was being called a second time, this ensures that we don't run the tests a second time + if (moveendCount > 0) return; + moveendCount++; + + expect(lngLatAsFixed(map.getCenter(), 4)).toEqual({lat: '10.0000', lng: '20.0000'}); + expect(geolocate._userLocationDotMarker._map).toBeTruthy(); + expect( + geolocate._userLocationDotMarker._element.classList.contains('maplibregl-user-location-dot-stale') + ).toBeFalsy(); + map.once('moveend', () => { + expect(lngLatAsFixed(map.getCenter(), 4)).toEqual({lat: '40.0000', lng: '50.0000'}); + geolocate.once('error', () => { + expect(geolocate._userLocationDotMarker._map).toBeTruthy(); + expect( + geolocate._userLocationDotMarker._element.classList.contains('maplibregl-user-location-dot-stale') + ).toBeTruthy(); + done(); + }); + geolocation.changeError({code: 2, message: 'position unavaliable'}); + }); + geolocation.change({latitude: 40, longitude: 50, accuracy: 60}); + }); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.send({latitude: 10, longitude: 20, accuracy: 30}); + }); + + test('watching map background event', done => { + const geolocate = new GeolocateControl({ + trackUserLocation: true, + fitBoundsOptions: { + duration: 0 + } + }); + map.addControl(geolocate); + + const click = new window.Event('click'); + + let moveendCount = 0; + map.once('moveend', () => { + // moveend was being called a second time, this ensures that we don't run the tests a second time + if (moveendCount > 0) return; + moveendCount++; + + geolocate.once('trackuserlocationend', () => { + expect(map.getCenter()).toEqual({lng: 10, lat: 5}); + done(); + }); + + // manually pan the map away from the geolocation position which should trigger the 'trackuserlocationend' event above + map.jumpTo({ + center: [10, 5] + }); + }); + // click the button to activate it into the enabled watch state + geolocate._geolocateButton.dispatchEvent(click); + // send through a location update which should reposition the map and trigger the 'moveend' event above + geolocation.send({latitude: 10, longitude: 20, accuracy: 30}); + }); + + test('watching map background state', done => { + const geolocate = new GeolocateControl({ + trackUserLocation: true, + fitBoundsOptions: { + duration: 0 + } + }); + map.addControl(geolocate); + + const click = new window.Event('click'); + + let moveendCount = 0; + map.once('moveend', () => { + // moveend was being called a second time, this ensures that we don't run the tests a second time + if (moveendCount > 0) return; + moveendCount++; + + map.once('moveend', () => { + geolocate.once('geolocate', () => { + expect(map.getCenter()).toEqual({lng: 10, lat: 5}); + done(); + }); + // update the geolocation position, since we are in background state when 'geolocate' is triggered above, the camera shouldn't have changed + geolocation.change({latitude: 0, longitude: 0, accuracy: 10}); + }); + + // manually pan the map away from the geolocation position which should trigger the 'moveend' event above + map.jumpTo({ + center: [10, 5] + }); + }); + // click the button to activate it into the enabled watch state + geolocate._geolocateButton.dispatchEvent(click); + // send through a location update which should reposition the map and trigger the 'moveend' event above + geolocation.send({latitude: 10, longitude: 20, accuracy: 30}); + }); + + test('trackuserlocationstart event', async () => { + const geolocate = new GeolocateControl({ + trackUserLocation: true, + fitBoundsOptions: { + duration: 0 + } + }); + map.addControl(geolocate); + + const click = new window.Event('click'); + + const promise = geolocate.once('trackuserlocationstart'); + geolocate._geolocateButton.dispatchEvent(click); + await promise; + expect(map.getCenter()).toEqual({lng: 0, lat: 0}); + }); + + test('does not switch to BACKGROUND and stays in ACTIVE_LOCK state on window resize', async () => { + const geolocate = new GeolocateControl({ + trackUserLocation: true, + }); + map.addControl(geolocate); + + const click = new window.Event('click'); + + const geolocatePromise = geolocate.once('geolocate'); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.send({latitude: 10, longitude: 20, accuracy: 30, timestamp: 40}); + await geolocatePromise; + expect(geolocate._watchState).toBe('ACTIVE_LOCK'); + window.dispatchEvent(new window.Event('resize')); + expect(geolocate._watchState).toBe('ACTIVE_LOCK'); + }); + + test('switches to BACKGROUND state on map manipulation', async () => { + const geolocate = new GeolocateControl({ + trackUserLocation: true, + }); + map.addControl(geolocate); + + const click = new window.Event('click'); + + const geolocatePromise = geolocate.once('geolocate'); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.send({latitude: 10, longitude: 20, accuracy: 30, timestamp: 40}); + await geolocatePromise; + expect(geolocate._watchState).toBe('ACTIVE_LOCK'); + map.jumpTo({ + center: [0, 0] + }); + expect(geolocate._watchState).toBe('BACKGROUND'); + }); + + test('accuracy circle not shown if showAccuracyCircle = false', async () => { + const geolocate = new GeolocateControl({ + trackUserLocation: true, + showUserLocation: true, + showAccuracyCircle: false, + }); + map.addControl(geolocate); + + const click = new window.Event('click'); + + const geolocatePromise = geolocate.once('geolocate'); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.send({latitude: 10, longitude: 20, accuracy: 700}); + await geolocatePromise; + map.jumpTo({ + center: [10, 20] + }); + const zoomendPromise = map.once('zoomend'); + map.zoomTo(10, {duration: 0}); + await zoomendPromise; + expect(!geolocate._circleElement.style.width).toBeTruthy(); + }); + + test('accuracy circle radius matches reported accuracy', async () => { + const geolocate = new GeolocateControl({ + trackUserLocation: true, + showUserLocation: true, + }); + map.addControl(geolocate); + + const click = new window.Event('click'); + + const geolocatePromise = geolocate.once('geolocate'); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.send({latitude: 10, longitude: 20, accuracy: 700}); + await geolocatePromise; + expect(geolocate._accuracyCircleMarker._map).toBeTruthy(); + expect(geolocate._accuracy).toBe(700); + map.jumpTo({ + center: [10, 20] + }); + + // test with bugger radius + let zoomendPromise = map.once('zoomend'); + map.zoomTo(12, {duration: 0}); + await zoomendPromise; + expect(geolocate._circleElement.style.width).toBe('79px'); + console.log(geolocate._circleElement.style.width); + zoomendPromise = map.once('zoomend'); + map.zoomTo(10, {duration: 0}); + await zoomendPromise; + expect(geolocate._circleElement.style.width).toBe('20px'); + console.log(geolocate._circleElement.style.width); + zoomendPromise = map.once('zoomend'); + + // test with smaller radius + geolocation.send({latitude: 10, longitude: 20, accuracy: 20}); + map.zoomTo(20, {duration: 0}); + await zoomendPromise; + expect(geolocate._circleElement.style.width).toBe('19982px'); + console.log(geolocate._circleElement.style.width); + zoomendPromise = map.once('zoomend'); + map.zoomTo(18, {duration: 0}); + await zoomendPromise; + expect(geolocate._circleElement.style.width).toBe('4996px'); + console.log(geolocate._circleElement.style.width); + }); + + test('shown even if trackUserLocation = false', async () => { + const geolocate = new GeolocateControl({ + trackUserLocation: false, + showUserLocation: true, + showAccuracyCircle: true, + }); + map.addControl(geolocate); + + const click = new window.Event('click'); + + const geolocatePromise = geolocate.once('geolocate'); + geolocate._geolocateButton.dispatchEvent(click); + geolocation.send({latitude: 10, longitude: 20, accuracy: 700}); + await geolocatePromise; + map.jumpTo({ + center: [10, 20] + }); + const zoomendPromise = map.once('zoomend'); + map.zoomTo(10, {duration: 0}); + await zoomendPromise; + expect(geolocate._circleElement.style.width).toBeTruthy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/control/geolocate_control.ts b/web/libraries/maplibre-gl/src/ui/control/geolocate_control.ts new file mode 100644 index 00000000..1399435f --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/geolocate_control.ts @@ -0,0 +1,667 @@ +import {Event, Evented} from '../../util/evented'; +import {DOM} from '../../util/dom'; +import {extend, warnOnce} from '../../util/util'; +import {checkGeolocationSupport} from '../../util/geolocation_support'; +import {LngLat} from '../../geo/lng_lat'; +import {Marker} from '../marker'; + +import type {Map} from '../map'; +import type {FitBoundsOptions} from '../camera'; +import type {IControl} from './control'; +import {LngLatBounds} from '../../geo/lng_lat_bounds'; + +/** + * The {@link GeolocateControl} options + */ +type GeolocateOptions = { + /** + * A Geolocation API [PositionOptions](https://developer.mozilla.org/en-US/docs/Web/API/PositionOptions) object. + * @defaultValue `{enableHighAccuracy: false, timeout: 6000}` + */ + positionOptions?: PositionOptions; + /** + * A options object to use when the map is panned and zoomed to the user's location. The default is to use a `maxZoom` of 15 to limit how far the map will zoom in for very accurate locations. + */ + fitBoundsOptions?: FitBoundsOptions; + /** + * If `true` the Geolocate Control becomes a toggle button and when active the map will receive updates to the user's location as it changes. + * @defaultValue false + */ + trackUserLocation?: boolean; + /** + * By default, if showUserLocation is `true`, a transparent circle will be drawn around the user location indicating the accuracy (95% confidence level) of the user's location. Set to `false` to disable. Always disabled when showUserLocation is `false`. + * @defaultValue true + */ + showAccuracyCircle?: boolean; + /** + * By default a dot will be shown on the map at the user's location. Set to `false` to disable. + * @defaultValue true + */ + showUserLocation?: boolean; +}; + +const defaultOptions: GeolocateOptions = { + positionOptions: { + enableHighAccuracy: false, + maximumAge: 0, + timeout: 6000 /* 6 sec */ + }, + fitBoundsOptions: { + maxZoom: 15 + }, + trackUserLocation: false, + showAccuracyCircle: true, + showUserLocation: true +}; + +let numberOfWatches = 0; +let noTimeout = false; + +/** + * A `GeolocateControl` control provides a button that uses the browser's geolocation + * API to locate the user on the map. + * + * Not all browsers support geolocation, + * and some users may disable the feature. Geolocation support for modern + * browsers including Chrome requires sites to be served over HTTPS. If + * geolocation support is not available, the GeolocateControl will show + * as disabled. + * + * The zoom level applied will depend on the accuracy of the geolocation provided by the device. + * + * The GeolocateControl has two modes. If `trackUserLocation` is `false` (default) the control acts as a button, which when pressed will set the map's camera to target the user location. If the user moves, the map won't update. This is most suited for the desktop. If `trackUserLocation` is `true` the control acts as a toggle button that when active the user's location is actively monitored for changes. In this mode the GeolocateControl has three interaction states: + * * active - the map's camera automatically updates as the user's location changes, keeping the location dot in the center. Initial state and upon clicking the `GeolocateControl` button. + * * passive - the user's location dot automatically updates, but the map's camera does not. Occurs upon the user initiating a map movement. + * * disabled - occurs if Geolocation is not available, disabled or denied. + * + * These interaction states can't be controlled programmatically, rather they are set based on user interactions. + * @group Markers and Controls + * + * @example + * ```ts + * map.addControl(new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * })); + * ``` + * @see [Locate the user](https://maplibre.org/maplibre-gl-js/docs/examples/locate-user/) + * + * ### Events + * + * @event `trackuserlocationend` - Fired when the Geolocate Control changes to the background state, which happens when a user changes the camera during an active position lock. This only applies when trackUserLocation is true. In the background state, the dot on the map will update with location updates but the camera will not. + * + * @event `trackuserlocationstart` - Fired when the Geolocate Control changes to the active lock state, which happens either upon first obtaining a successful Geolocation API position for the user (a geolocate event will follow), or the user clicks the geolocate button when in the background state which uses the last known position to recenter the map and enter active lock state (no geolocate event will follow unless the users's location changes). + * + * @event `geolocate` - Fired on each Geolocation API position update which returned as success. + * `data` - The returned [Position](https://developer.mozilla.org/en-US/docs/Web/API/Position) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition). + * + * @event `error` - Fired on each Geolocation API position update which returned as an error. + * `data` - The returned [PositionError](https://developer.mozilla.org/en-US/docs/Web/API/PositionError) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition). + * + * @event `outofmaxbounds` Fired on each Geolocation API position update which returned as success but user position is out of map maxBounds. + * `data` - The returned [Position](https://developer.mozilla.org/en-US/docs/Web/API/Position) object from the callback in [Geolocation.getCurrentPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/getCurrentPosition) or [Geolocation.watchPosition()](https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/watchPosition). + * + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * // Set an event listener that fires + * // when a trackuserlocationend event occurs. + * geolocate.on('trackuserlocationend', function() { + * console.log('A trackuserlocationend event has occurred.') + * }); + * ``` + * + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * // Set an event listener that fires + * // when a trackuserlocationstart event occurs. + * geolocate.on('trackuserlocationstart', function() { + * console.log('A trackuserlocationstart event has occurred.') + * }); + * ``` + * + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * // Set an event listener that fires + * // when a geolocate event occurs. + * geolocate.on('geolocate', function() { + * console.log('A geolocate event has occurred.') + * }); + * ``` + * + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * // Set an event listener that fires + * // when an error event occurs. + * geolocate.on('error', function() { + * console.log('An error event has occurred.') + * }); + * ``` + * + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * // Set an event listener that fires + * // when an outofmaxbounds event occurs. + * geolocate.on('outofmaxbounds', function() { + * console.log('An outofmaxbounds event has occurred.') + * }); + * ``` + */ +export class GeolocateControl extends Evented implements IControl { + _map: Map; + options: GeolocateOptions; + _container: HTMLElement; + _dotElement: HTMLElement; + _circleElement: HTMLElement; + _geolocateButton: HTMLButtonElement; + _geolocationWatchID: number; + _timeoutId: ReturnType; + /* Geolocate Control Watch States + * This is the private state of the control. + * + * OFF + * off/inactive + * WAITING_ACTIVE + * Geolocate Control was clicked but still waiting for Geolocation API response with user location + * ACTIVE_LOCK + * Showing the user location as a dot AND tracking the camera to be fixed to their location. If their location changes the map moves to follow. + * ACTIVE_ERROR + * There was en error from the Geolocation API while trying to show and track the user location. + * BACKGROUND + * Showing the user location as a dot but the camera doesn't follow their location as it changes. + * BACKGROUND_ERROR + * There was an error from the Geolocation API while trying to show (but not track) the user location. + */ + _watchState: 'OFF' | 'ACTIVE_LOCK' | 'WAITING_ACTIVE' | 'ACTIVE_ERROR' | 'BACKGROUND' | 'BACKGROUND_ERROR'; + _lastKnownPosition: any; + _userLocationDotMarker: Marker; + _accuracyCircleMarker: Marker; + _accuracy: number; + _setup: boolean; // set to true once the control has been setup + + constructor(options: GeolocateOptions) { + super(); + this.options = extend({}, defaultOptions, options); + } + + /** {@inheritDoc IControl.onAdd} */ + onAdd(map: Map) { + this._map = map; + this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group'); + checkGeolocationSupport(this._setupUI); + return this._container; + } + + /** {@inheritDoc IControl.onRemove} */ + onRemove() { + // clear the geolocation watch if exists + if (this._geolocationWatchID !== undefined) { + window.navigator.geolocation.clearWatch(this._geolocationWatchID); + this._geolocationWatchID = undefined; + } + + // clear the markers from the map + if (this.options.showUserLocation && this._userLocationDotMarker) { + this._userLocationDotMarker.remove(); + } + if (this.options.showAccuracyCircle && this._accuracyCircleMarker) { + this._accuracyCircleMarker.remove(); + } + + DOM.remove(this._container); + this._map.off('zoom', this._onZoom); + this._map = undefined; + numberOfWatches = 0; + noTimeout = false; + } + + /** + * Check if the Geolocation API Position is outside the map's maxbounds. + * + * @param position - the Geolocation API Position + * @returns `true` if position is outside the map's maxbounds, otherwise returns `false`. + */ + _isOutOfMapMaxBounds(position: GeolocationPosition) { + const bounds = this._map.getMaxBounds(); + const coordinates = position.coords; + + return bounds && ( + coordinates.longitude < bounds.getWest() || + coordinates.longitude > bounds.getEast() || + coordinates.latitude < bounds.getSouth() || + coordinates.latitude > bounds.getNorth() + ); + } + + _setErrorState() { + switch (this._watchState) { + case 'WAITING_ACTIVE': + this._watchState = 'ACTIVE_ERROR'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active-error'); + break; + case 'ACTIVE_LOCK': + this._watchState = 'ACTIVE_ERROR'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active-error'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting'); + // turn marker grey + break; + case 'BACKGROUND': + this._watchState = 'BACKGROUND_ERROR'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background-error'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting'); + // turn marker grey + break; + case 'ACTIVE_ERROR': + break; + default: + throw new Error(`Unexpected watchState ${this._watchState}`); + } + } + + /** + * When the Geolocation API returns a new location, update the GeolocateControl. + * + * @param position - the Geolocation API Position + */ + _onSuccess = (position: GeolocationPosition) => { + if (!this._map) { + // control has since been removed + return; + } + + if (this._isOutOfMapMaxBounds(position)) { + this._setErrorState(); + + this.fire(new Event('outofmaxbounds', position)); + this._updateMarker(); + this._finish(); + + return; + } + + if (this.options.trackUserLocation) { + // keep a record of the position so that if the state is BACKGROUND and the user + // clicks the button, we can move to ACTIVE_LOCK immediately without waiting for + // watchPosition to trigger _onSuccess + this._lastKnownPosition = position; + + switch (this._watchState) { + case 'WAITING_ACTIVE': + case 'ACTIVE_LOCK': + case 'ACTIVE_ERROR': + this._watchState = 'ACTIVE_LOCK'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active'); + break; + case 'BACKGROUND': + case 'BACKGROUND_ERROR': + this._watchState = 'BACKGROUND'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background'); + break; + default: + throw new Error(`Unexpected watchState ${this._watchState}`); + } + } + + // if showUserLocation and the watch state isn't off then update the marker location + if (this.options.showUserLocation && this._watchState !== 'OFF') { + this._updateMarker(position); + } + + // if in normal mode (not watch mode), or if in watch mode and the state is active watch + // then update the camera + if (!this.options.trackUserLocation || this._watchState === 'ACTIVE_LOCK') { + this._updateCamera(position); + } + + if (this.options.showUserLocation) { + this._dotElement.classList.remove('maplibregl-user-location-dot-stale'); + } + + this.fire(new Event('geolocate', position)); + this._finish(); + }; + + /** + * Update the camera location to center on the current position + * + * @param position - the Geolocation API Position + */ + _updateCamera = (position: GeolocationPosition) => { + const center = new LngLat(position.coords.longitude, position.coords.latitude); + const radius = position.coords.accuracy; + const bearing = this._map.getBearing(); + const options = extend({bearing}, this.options.fitBoundsOptions); + const newBounds = LngLatBounds.fromLngLat(center, radius); + + this._map.fitBounds(newBounds, options, { + geolocateSource: true // tag this camera change so it won't cause the control to change to background state + }); + }; + + /** + * Update the user location dot Marker to the current position + * + * @param position - the Geolocation API Position + */ + _updateMarker = (position?: GeolocationPosition | null) => { + if (position) { + const center = new LngLat(position.coords.longitude, position.coords.latitude); + this._accuracyCircleMarker.setLngLat(center).addTo(this._map); + this._userLocationDotMarker.setLngLat(center).addTo(this._map); + this._accuracy = position.coords.accuracy; + if (this.options.showUserLocation && this.options.showAccuracyCircle) { + this._updateCircleRadius(); + } + } else { + this._userLocationDotMarker.remove(); + this._accuracyCircleMarker.remove(); + } + }; + + _updateCircleRadius() { + const bounds = this._map.getBounds(); + const southEastPoint = bounds.getSouthEast(); + const northEastPoint = bounds.getNorthEast(); + const mapHeightInMeters = southEastPoint.distanceTo(northEastPoint); + const mapHeightInPixels = this._map._container.clientHeight; + const circleDiameter = Math.ceil(2 * (this._accuracy / (mapHeightInMeters / mapHeightInPixels))); + this._circleElement.style.width = `${circleDiameter}px`; + this._circleElement.style.height = `${circleDiameter}px`; + } + + _onZoom = () => { + if (this.options.showUserLocation && this.options.showAccuracyCircle) { + this._updateCircleRadius(); + } + }; + + _onError = (error: GeolocationPositionError) => { + if (!this._map) { + // control has since been removed + return; + } + + if (this.options.trackUserLocation) { + if (error.code === 1) { + // PERMISSION_DENIED + this._watchState = 'OFF'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error'); + this._geolocateButton.disabled = true; + const title = this._map._getUIString('GeolocateControl.LocationNotAvailable'); + this._geolocateButton.title = title; + this._geolocateButton.setAttribute('aria-label', title); + + if (this._geolocationWatchID !== undefined) { + this._clearWatch(); + } + } else if (error.code === 3 && noTimeout) { + // this represents a forced error state + // this was triggered to force immediate geolocation when a watch is already present + // see https://github.com/mapbox/mapbox-gl-js/issues/8214 + // and https://w3c.github.io/geolocation-api/#example-5-forcing-the-user-agent-to-return-a-fresh-cached-position + return; + } else { + this._setErrorState(); + } + } + + if (this._watchState !== 'OFF' && this.options.showUserLocation) { + this._dotElement.classList.add('maplibregl-user-location-dot-stale'); + } + + this.fire(new Event('error', error)); + + this._finish(); + }; + + _finish = () => { + if (this._timeoutId) { clearTimeout(this._timeoutId); } + this._timeoutId = undefined; + }; + + _setupUI = (supported: boolean) => { + // this method is called asynchronously during onAdd + // the control could have been removed before reaching here + if (!this._map) { + return; + } + + this._container.addEventListener('contextmenu', (e: MouseEvent) => e.preventDefault()); + this._geolocateButton = DOM.create('button', 'maplibregl-ctrl-geolocate', this._container); + DOM.create('span', 'maplibregl-ctrl-icon', this._geolocateButton).setAttribute('aria-hidden', 'true'); + this._geolocateButton.type = 'button'; + + if (supported === false) { + warnOnce('Geolocation support is not available so the GeolocateControl will be disabled.'); + const title = this._map._getUIString('GeolocateControl.LocationNotAvailable'); + this._geolocateButton.disabled = true; + this._geolocateButton.title = title; + this._geolocateButton.setAttribute('aria-label', title); + } else { + const title = this._map._getUIString('GeolocateControl.FindMyLocation'); + this._geolocateButton.title = title; + this._geolocateButton.setAttribute('aria-label', title); + } + + if (this.options.trackUserLocation) { + this._geolocateButton.setAttribute('aria-pressed', 'false'); + this._watchState = 'OFF'; + } + + // when showUserLocation is enabled, keep the Geolocate button disabled until the device location marker is setup on the map + if (this.options.showUserLocation) { + this._dotElement = DOM.create('div', 'maplibregl-user-location-dot'); + + this._userLocationDotMarker = new Marker({element: this._dotElement}); + + this._circleElement = DOM.create('div', 'maplibregl-user-location-accuracy-circle'); + this._accuracyCircleMarker = new Marker({element: this._circleElement, pitchAlignment: 'map'}); + + if (this.options.trackUserLocation) this._watchState = 'OFF'; + + this._map.on('zoom', this._onZoom); + } + + this._geolocateButton.addEventListener('click', + this.trigger.bind(this)); + + this._setup = true; + + // when the camera is changed (and it's not as a result of the Geolocation Control) change + // the watch mode to background watch, so that the marker is updated but not the camera. + if (this.options.trackUserLocation) { + this._map.on('movestart', (event: any) => { + const fromResize = event.originalEvent && event.originalEvent.type === 'resize'; + if (!event.geolocateSource && this._watchState === 'ACTIVE_LOCK' && !fromResize) { + this._watchState = 'BACKGROUND'; + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-background'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active'); + + this.fire(new Event('trackuserlocationend')); + } + }); + } + }; + + /** + * Programmatically request and move the map to the user's location. + * + * @returns `false` if called before control was added to a map, otherwise returns `true`. + * @example + * ```ts + * // Initialize the geolocate control. + * let geolocate = new maplibregl.GeolocateControl({ + * positionOptions: { + * enableHighAccuracy: true + * }, + * trackUserLocation: true + * }); + * // Add the control to the map. + * map.addControl(geolocate); + * map.on('load', function() { + * geolocate.trigger(); + * }); + * ``` + */ + trigger(): boolean { + if (!this._setup) { + warnOnce('Geolocate control triggered before added to a map'); + return false; + } + if (this.options.trackUserLocation) { + // update watchState and do any outgoing state cleanup + switch (this._watchState) { + case 'OFF': + // turn on the Geolocate Control + this._watchState = 'WAITING_ACTIVE'; + + this.fire(new Event('trackuserlocationstart')); + break; + case 'WAITING_ACTIVE': + case 'ACTIVE_LOCK': + case 'ACTIVE_ERROR': + case 'BACKGROUND_ERROR': + // turn off the Geolocate Control + numberOfWatches--; + noTimeout = false; + this._watchState = 'OFF'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-active-error'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background'); + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background-error'); + + this.fire(new Event('trackuserlocationend')); + break; + case 'BACKGROUND': + this._watchState = 'ACTIVE_LOCK'; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-background'); + // set camera to last known location + if (this._lastKnownPosition) this._updateCamera(this._lastKnownPosition); + + this.fire(new Event('trackuserlocationstart')); + break; + default: + throw new Error(`Unexpected watchState ${this._watchState}`); + } + + // incoming state setup + switch (this._watchState) { + case 'WAITING_ACTIVE': + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active'); + break; + case 'ACTIVE_LOCK': + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-active'); + break; + case 'OFF': + break; + default: + throw new Error(`Unexpected watchState ${this._watchState}`); + } + + // manage geolocation.watchPosition / geolocation.clearWatch + if (this._watchState === 'OFF' && this._geolocationWatchID !== undefined) { + // clear watchPosition as we've changed to an OFF state + this._clearWatch(); + } else if (this._geolocationWatchID === undefined) { + // enable watchPosition since watchState is not OFF and there is no watchPosition already running + + this._geolocateButton.classList.add('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.setAttribute('aria-pressed', 'true'); + + numberOfWatches++; + let positionOptions; + if (numberOfWatches > 1) { + positionOptions = {maximumAge: 600000, timeout: 0}; + noTimeout = true; + } else { + positionOptions = this.options.positionOptions; + noTimeout = false; + } + + this._geolocationWatchID = window.navigator.geolocation.watchPosition( + this._onSuccess, this._onError, positionOptions); + } + } else { + window.navigator.geolocation.getCurrentPosition( + this._onSuccess, this._onError, this.options.positionOptions); + + // This timeout ensures that we still call finish() even if + // the user declines to share their location in Firefox + this._timeoutId = setTimeout(this._finish, 10000 /* 10sec */); + } + + return true; + } + + _clearWatch() { + window.navigator.geolocation.clearWatch(this._geolocationWatchID); + + this._geolocationWatchID = undefined; + this._geolocateButton.classList.remove('maplibregl-ctrl-geolocate-waiting'); + this._geolocateButton.setAttribute('aria-pressed', 'false'); + + if (this.options.showUserLocation) { + this._updateMarker(null); + } + } +} + diff --git a/web/libraries/maplibre-gl/src/ui/control/logo_control.test.ts b/web/libraries/maplibre-gl/src/ui/control/logo_control.test.ts new file mode 100644 index 00000000..78a1a948 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/logo_control.test.ts @@ -0,0 +1,90 @@ +import {createMap as globalCreateMap, beforeMapTest} from '../../util/test/util'; + +function createMap(logoPosition, maplibreLogo) { + + const mapobj = { + logoPosition, + maplibreLogo, + style: { + version: 8, + sources: {}, + layers: [] + } + }; + + return globalCreateMap(mapobj, undefined); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('LogoControl', () => { + test('does not appear by default', done => { + const map = createMap(undefined, undefined); + map.on('load', () => { + expect(map.getContainer().querySelectorAll( + '.maplibregl-ctrl-logo' + )).toHaveLength(0); + done(); + }); + }); + + test('is not displayed when the maplibreLogo property is false', done => { + const map = createMap(undefined, false); + map.on('load', () => { + expect(map.getContainer().querySelectorAll( + '.maplibregl-ctrl-logo' + )).toHaveLength(0); + done(); + }); + }); + + test('appears in bottom-left when maplibreLogo is true and logoPosition is undefined', done => { + const map = createMap(undefined, true); + map.on('load', () => { + expect(map.getContainer().querySelectorAll( + '.maplibregl-ctrl-bottom-left .maplibregl-ctrl-logo' + )).toHaveLength(1); + done(); + }); + }); + + test('appears in the position specified by the position option', done => { + const map = createMap('top-left', true); + map.on('load', () => { + expect(map.getContainer().querySelectorAll( + '.maplibregl-ctrl-top-left .maplibregl-ctrl-logo' + )).toHaveLength(1); + done(); + }); + }); + + test('appears in compact mode if container is less then 640 pixel wide', () => { + const map = createMap(undefined, true); + const container = map.getContainer(); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 645, configurable: true}); + map.resize(); + expect( + container.querySelectorAll('.maplibregl-ctrl-logo:not(.maplibregl-compact)') + ).toHaveLength(1); + + Object.defineProperty(map.getCanvasContainer(), 'offsetWidth', {value: 635, configurable: true}); + map.resize(); + expect( + container.querySelectorAll('.maplibregl-ctrl-logo.maplibregl-compact') + ).toHaveLength(1); + }); + + test('has `rel` nooper and nofollow', done => { + const map = createMap(undefined, true); + + map.on('load', () => { + const container = map.getContainer(); + const logo = container.querySelector('.maplibregl-ctrl-logo'); + expect(logo).toHaveProperty('rel', 'noopener nofollow'); + done(); + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/control/logo_control.ts b/web/libraries/maplibre-gl/src/ui/control/logo_control.ts new file mode 100644 index 00000000..4ce08838 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/logo_control.ts @@ -0,0 +1,83 @@ +import {DOM} from '../../util/dom'; + +import type {Map} from '../map'; +import type {ControlPosition, IControl} from './control'; + +/** + * The {@link LogoControl} options object + */ +type LogoOptions = { + /** + * If `true`, force a compact logo. + * If `false`, force the full logo. The default is a responsive logo that collapses when the map is less than 640 pixels wide. + */ + compact?: boolean; +}; + +/** + * A `LogoControl` is a control that adds the watermark. + * + * @group Markers and Controls + * + * @example + * ```ts + * map.addControl(new maplibregl.LogoControl({compact: false})); + * ``` + **/ +export class LogoControl implements IControl { + options: LogoOptions; + _map: Map; + _compact: boolean; + _container: HTMLElement; + + constructor(options: LogoOptions = {}) { + this.options = options; + } + + getDefaultPosition(): ControlPosition { + return 'bottom-left'; + } + + /** {@inheritDoc IControl.onAdd} */ + onAdd(map: Map) { + this._map = map; + this._compact = this.options && this.options.compact; + this._container = DOM.create('div', 'maplibregl-ctrl'); + const anchor = DOM.create('a', 'maplibregl-ctrl-logo'); + anchor.target = '_blank'; + anchor.rel = 'noopener nofollow'; + anchor.href = 'https://maplibre.org/'; + anchor.setAttribute('aria-label', this._map._getUIString('LogoControl.Title')); + anchor.setAttribute('rel', 'noopener nofollow'); + this._container.appendChild(anchor); + this._container.style.display = 'block'; + + this._map.on('resize', this._updateCompact); + this._updateCompact(); + + return this._container; + } + + /** {@inheritDoc IControl.onRemove} */ + onRemove() { + DOM.remove(this._container); + this._map.off('resize', this._updateCompact); + this._map = undefined; + this._compact = undefined; + } + + _updateCompact = () => { + const containerChildren = this._container.children; + if (containerChildren.length) { + const anchor = containerChildren[0]; + if (this._map.getCanvasContainer().offsetWidth <= 640 || this._compact) { + if (this._compact !== false) { + anchor.classList.add('maplibregl-compact'); + } + } else { + anchor.classList.remove('maplibregl-compact'); + } + } + }; + +} diff --git a/web/libraries/maplibre-gl/src/ui/control/navigation_control.test.ts b/web/libraries/maplibre-gl/src/ui/control/navigation_control.test.ts new file mode 100644 index 00000000..58038f71 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/navigation_control.test.ts @@ -0,0 +1,238 @@ +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {createMap as globalCreateMap, beforeMapTest} from '../../util/test/util'; +import {NavigationControl} from './navigation_control'; + +function createMap() { + return globalCreateMap({}); +} + +let map; + +beforeEach(() => { + beforeMapTest(); + map = createMap(); +}); + +afterEach(() => { + map.remove(); +}); + +describe('NavigationControl', () => { + test('appears in the top-right', () => { + map.addControl(new NavigationControl()); + + expect( + map.getContainer().querySelectorAll('.maplibregl-ctrl-top-right > .maplibregl-ctrl') + ).toHaveLength(1); + }); + + test('contains zoom in button', () => { + map.addControl(new NavigationControl()); + + expect( + map.getContainer().querySelectorAll('.maplibregl-ctrl-zoom-in') + ).toHaveLength(1); + }); + + test('contains zoom out button', () => { + map.addControl(new NavigationControl()); + + expect( + map.getContainer().querySelectorAll('.maplibregl-ctrl-zoom-out') + ).toHaveLength(1); + }); + + test('contains compass button', () => { + map.addControl(new NavigationControl()); + + expect( + map.getContainer().querySelectorAll('.maplibregl-ctrl-compass') + ).toHaveLength(1); + }); + + test('compass button reset action', () => { + map.setPitch(10); + map.setBearing(10); + + map.addControl(new NavigationControl({ + visualizePitch: true, + showZoom: true, + showCompass: true + })); + const spyReset = jest.spyOn(map, 'resetNorthPitch'); + const navButton = map.getContainer().querySelector('.maplibregl-ctrl-compass'); + + simulate.click(navButton); + map._renderTaskQueue.run(); + + expect(spyReset).toHaveBeenCalledTimes(1); + }); + + test('compass button drag horizontal', () => { + const navControl = new NavigationControl({ + visualizePitch: true, + showZoom: true, + showCompass: true + }); + map.addControl(navControl); + + const spySetPitch = jest.spyOn(map, 'setPitch'); + const spySetBearing = jest.spyOn(map, 'setBearing'); + + const navButton = map.getContainer().querySelector('.maplibregl-ctrl-compass'); + const navRect = navButton.getClientRects(); + + const buttonX = (navRect.x ?? 0) + (navRect.width ?? 0) / 2; + const buttonY = (navRect.y ?? 0) + (navRect.height ?? 0) / 2; + + simulate.mousedown(navButton, {buttons: 1, button: 0, clientX: buttonX, clientY: buttonY}); + simulate.mousemove(window, {buttons: 1, button: 0, clientX: buttonX - 50, clientY: buttonY}); + simulate.mousemove(window, {buttons: 1, button: 0, clientX: buttonX - 100, clientY: buttonY}); + simulate.mouseup(window, {button: 0, clientX: buttonX - 100, clientY: buttonY}); + + map._renderTaskQueue.run(); + + expect(spySetPitch).not.toHaveBeenCalled(); + expect(spySetBearing).toHaveBeenCalled(); + }); + + test('compass button drag vertical', () => { + const navControl = new NavigationControl({ + visualizePitch: true, + showZoom: true, + showCompass: true + }); + map.addControl(navControl); + + const spySetPitch = jest.spyOn(map, 'setPitch'); + const spySetBearing = jest.spyOn(map, 'setBearing'); + + const navButton = map.getContainer().querySelector('.maplibregl-ctrl-compass'); + const navRect = navButton.getClientRects(); + + const buttonX = (navRect.x ?? 0) + (navRect.width ?? 0) / 2; + const buttonY = (navRect.y ?? 0) + (navRect.height ?? 0) / 2; + + simulate.mousedown(navButton, {buttons: 1, button: 0, clientX: buttonX, clientY: buttonY}); + simulate.mousemove(window, {buttons: 1, button: 0, clientX: buttonX, clientY: buttonY - 50}); + simulate.mousemove(window, {buttons: 1, button: 0, clientX: buttonX, clientY: buttonY - 100}); + simulate.mouseup(window, {button: 0, clientX: buttonX, clientY: buttonY - 100}); + + map._renderTaskQueue.run(); + + expect(spySetPitch).toHaveBeenCalled(); + expect(spySetBearing).not.toHaveBeenCalled(); + }); + + test('compass button drag diagonal', () => { + const navControl = new NavigationControl({ + visualizePitch: true, + showZoom: true, + showCompass: true + }); + map.addControl(navControl); + + const spySetPitch = jest.spyOn(map, 'setPitch'); + const spySetBearing = jest.spyOn(map, 'setBearing'); + + const navButton = map.getContainer().querySelector('.maplibregl-ctrl-compass'); + const navRect = navButton.getClientRects(); + + const buttonX = (navRect.x ?? 0) + (navRect.width ?? 0) / 2; + const buttonY = (navRect.y ?? 0) + (navRect.height ?? 0) / 2; + + simulate.mousedown(navButton, {buttons: 1, button: 0, clientX: buttonX, clientY: buttonY}); + simulate.mousemove(window, {buttons: 1, button: 0, clientX: buttonX - 50, clientY: buttonY - 50}); + simulate.mousemove(window, {buttons: 1, button: 0, clientX: buttonX - 100, clientY: buttonY - 100}); + simulate.mouseup(window, {button: 0, clientX: buttonX - 100, clientY: buttonY - 100}); + + map._renderTaskQueue.run(); + + expect(spySetPitch).toHaveBeenCalled(); + expect(spySetBearing).toHaveBeenCalled(); + }); + + test('compass button touch drag horizontal', () => { + const navControl = new NavigationControl({ + visualizePitch: true, + showZoom: true, + showCompass: true + }); + map.addControl(navControl); + + const spySetPitch = jest.spyOn(map, 'setPitch'); + const spySetBearing = jest.spyOn(map, 'setBearing'); + + const navButton = map.getContainer().querySelector('.maplibregl-ctrl-compass'); + const navRect = navButton.getClientRects(); + + const buttonX = (navRect.x ?? 0) + (navRect.width ?? 0) / 2; + const buttonY = (navRect.y ?? 0) + (navRect.height ?? 0) / 2; + + simulate.touchstart(navButton, {touches: [{clientX: buttonX, clientY: buttonY}], targetTouches: [{clientX: buttonX, clientY: buttonY}]}); + simulate.touchmove(window, {touches: [{clientX: buttonX - 50, clientY: buttonY}], targetTouches: [{clientX: buttonX - 50, clientY: buttonY}]}); + simulate.touchmove(window, {touches: [{clientX: buttonX - 100, clientY: buttonY}], targetTouches: [{clientX: buttonX - 100, clientY: buttonY}]}); + simulate.touchend(window); + + map._renderTaskQueue.run(); + + expect(spySetPitch).not.toHaveBeenCalled(); + expect(spySetBearing).toHaveBeenCalled(); + }); + + test('compass button touch drag vertical', () => { + const navControl = new NavigationControl({ + visualizePitch: true, + showZoom: true, + showCompass: true + }); + map.addControl(navControl); + + const spySetPitch = jest.spyOn(map, 'setPitch'); + const spySetBearing = jest.spyOn(map, 'setBearing'); + + const navButton = map.getContainer().querySelector('.maplibregl-ctrl-compass'); + const navRect = navButton.getClientRects(); + + const buttonX = (navRect.x ?? 0) + (navRect.width ?? 0) / 2; + const buttonY = (navRect.y ?? 0) + (navRect.height ?? 0) / 2; + + simulate.touchstart(navButton, {touches: [{clientX: buttonX, clientY: buttonY}], targetTouches: [{clientX: buttonX, clientY: buttonY}]}); + simulate.touchmove(window, {touches: [{clientX: buttonX, clientY: buttonY - 50}], targetTouches: [{clientX: buttonX, clientY: buttonY - 50}]}); + simulate.touchmove(window, {touches: [{clientX: buttonX, clientY: buttonY - 100}], targetTouches: [{clientX: buttonX, clientY: buttonY - 100}]}); + simulate.touchend(window); + + map._renderTaskQueue.run(); + + expect(spySetPitch).toHaveBeenCalled(); + expect(spySetBearing).not.toHaveBeenCalled(); + }); + + test('compass button touch drag diagonal', () => { + const navControl = new NavigationControl({ + visualizePitch: true, + showZoom: true, + showCompass: true + }); + map.addControl(navControl); + + const spySetPitch = jest.spyOn(map, 'setPitch'); + const spySetBearing = jest.spyOn(map, 'setBearing'); + + const navButton = map.getContainer().querySelector('.maplibregl-ctrl-compass'); + const navRect = navButton.getClientRects(); + + const buttonX = (navRect.x ?? 0) + (navRect.width ?? 0) / 2; + const buttonY = (navRect.y ?? 0) + (navRect.height ?? 0) / 2; + + simulate.touchstart(navButton, {touches: [{clientX: buttonX, clientY: buttonY}], targetTouches: [{clientX: buttonX, clientY: buttonY}]}); + simulate.touchmove(window, {touches: [{clientX: buttonX - 50, clientY: buttonY - 50}], targetTouches: [{clientX: buttonX - 50, clientY: buttonY - 50}]}); + simulate.touchmove(window, {touches: [{clientX: buttonX - 100, clientY: buttonY - 100}], targetTouches: [{clientX: buttonX - 100, clientY: buttonY - 100}]}); + simulate.touchend(window); + + map._renderTaskQueue.run(); + + expect(spySetPitch).toHaveBeenCalled(); + expect(spySetBearing).toHaveBeenCalled(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/control/navigation_control.ts b/web/libraries/maplibre-gl/src/ui/control/navigation_control.ts new file mode 100644 index 00000000..5883c517 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/navigation_control.ts @@ -0,0 +1,292 @@ +import Point from '@mapbox/point-geometry'; + +import {DOM} from '../../util/dom'; +import {extend} from '../../util/util'; +import {generateMousePitchHandler, generateMouseRotationHandler, MousePitchHandler, MouseRotateHandler} from '../handler/mouse'; +import {generateOneFingerTouchPitchHandler, generateOneFingerTouchRotationHandler, OneFingerTouchPitchHandler, OneFingerTouchRotateHandler} from '../handler/one_finger_touch_drag'; + +import type {Map} from '../map'; +import type {IControl} from './control'; + +/** + * The {@link NavigationControl} options object + */ +type NavigationOptions = { + /** + * If `true` the compass button is included. + */ + showCompass?: boolean; + /** + * If `true` the zoom-in and zoom-out buttons are included. + */ + showZoom?: boolean; + /** + * If `true` the pitch is visualized by rotating X-axis of compass. + */ + visualizePitch?: boolean; +}; + +const defaultOptions: NavigationOptions = { + showCompass: true, + showZoom: true, + visualizePitch: false +}; + +/** + * A `NavigationControl` control contains zoom buttons and a compass. + * + * @group Markers and Controls + * + * @example + * ```ts + * let nav = new maplibregl.NavigationControl(); + * map.addControl(nav, 'top-left'); + * ``` + * @see [Display map navigation controls](https://maplibre.org/maplibre-gl-js/docs/examples/navigation/) + */ +export class NavigationControl implements IControl { + _map: Map; + options: NavigationOptions; + _container: HTMLElement; + _zoomInButton: HTMLButtonElement; + _zoomOutButton: HTMLButtonElement; + _compass: HTMLButtonElement; + _compassIcon: HTMLElement; + _handler: MouseRotateWrapper; + + /** + * @param options - the control's options + */ + constructor(options?: NavigationOptions) { + this.options = extend({}, defaultOptions, options); + + this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group'); + this._container.addEventListener('contextmenu', (e) => e.preventDefault()); + + if (this.options.showZoom) { + this._zoomInButton = this._createButton('maplibregl-ctrl-zoom-in', (e) => this._map.zoomIn({}, {originalEvent: e})); + DOM.create('span', 'maplibregl-ctrl-icon', this._zoomInButton).setAttribute('aria-hidden', 'true'); + this._zoomOutButton = this._createButton('maplibregl-ctrl-zoom-out', (e) => this._map.zoomOut({}, {originalEvent: e})); + DOM.create('span', 'maplibregl-ctrl-icon', this._zoomOutButton).setAttribute('aria-hidden', 'true'); + } + if (this.options.showCompass) { + this._compass = this._createButton('maplibregl-ctrl-compass', (e) => { + if (this.options.visualizePitch) { + this._map.resetNorthPitch({}, {originalEvent: e}); + } else { + this._map.resetNorth({}, {originalEvent: e}); + } + }); + this._compassIcon = DOM.create('span', 'maplibregl-ctrl-icon', this._compass); + this._compassIcon.setAttribute('aria-hidden', 'true'); + } + } + + _updateZoomButtons = () => { + const zoom = this._map.getZoom(); + const isMax = zoom === this._map.getMaxZoom(); + const isMin = zoom === this._map.getMinZoom(); + this._zoomInButton.disabled = isMax; + this._zoomOutButton.disabled = isMin; + this._zoomInButton.setAttribute('aria-disabled', isMax.toString()); + this._zoomOutButton.setAttribute('aria-disabled', isMin.toString()); + }; + + _rotateCompassArrow = () => { + const rotate = this.options.visualizePitch ? + `scale(${1 / Math.pow(Math.cos(this._map.transform.pitch * (Math.PI / 180)), 0.5)}) rotateX(${this._map.transform.pitch}deg) rotateZ(${this._map.transform.angle * (180 / Math.PI)}deg)` : + `rotate(${this._map.transform.angle * (180 / Math.PI)}deg)`; + + this._compassIcon.style.transform = rotate; + }; + + onAdd(map: Map) { + this._map = map; + if (this.options.showZoom) { + this._setButtonTitle(this._zoomInButton, 'ZoomIn'); + this._setButtonTitle(this._zoomOutButton, 'ZoomOut'); + this._map.on('zoom', this._updateZoomButtons); + this._updateZoomButtons(); + } + if (this.options.showCompass) { + this._setButtonTitle(this._compass, 'ResetBearing'); + if (this.options.visualizePitch) { + this._map.on('pitch', this._rotateCompassArrow); + } + this._map.on('rotate', this._rotateCompassArrow); + this._rotateCompassArrow(); + this._handler = new MouseRotateWrapper(this._map, this._compass, this.options.visualizePitch); + } + return this._container; + } + + onRemove() { + DOM.remove(this._container); + if (this.options.showZoom) { + this._map.off('zoom', this._updateZoomButtons); + } + if (this.options.showCompass) { + if (this.options.visualizePitch) { + this._map.off('pitch', this._rotateCompassArrow); + } + this._map.off('rotate', this._rotateCompassArrow); + this._handler.off(); + delete this._handler; + } + + delete this._map; + } + + _createButton(className: string, fn: (e?: any) => unknown) { + const a = DOM.create('button', className, this._container) as HTMLButtonElement; + a.type = 'button'; + a.addEventListener('click', fn); + return a; + } + + _setButtonTitle = (button: HTMLButtonElement, title: string) => { + const str = this._map._getUIString(`NavigationControl.${title}`); + button.title = str; + button.setAttribute('aria-label', str); + }; +} + +class MouseRotateWrapper { + + map: Map; + _clickTolerance: number; + element: HTMLElement; + // Rotation and pitch handlers are separated due to different _clickTolerance values + mouseRotate: MouseRotateHandler; + touchRotate: OneFingerTouchRotateHandler; + mousePitch: MousePitchHandler; + touchPitch: OneFingerTouchPitchHandler; + _startPos: Point; + _lastPos: Point; + + constructor(map: Map, element: HTMLElement, pitch: boolean = false) { + this._clickTolerance = 10; + const mapRotateTolerance = map.dragRotate._mouseRotate.getClickTolerance(); + const mapPitchTolerance = map.dragRotate._mousePitch.getClickTolerance(); + this.element = element; + this.mouseRotate = generateMouseRotationHandler({clickTolerance: mapRotateTolerance, enable: true}); + this.touchRotate = generateOneFingerTouchRotationHandler({clickTolerance: mapRotateTolerance, enable: true}); + this.map = map; + if (pitch) { + this.mousePitch = generateMousePitchHandler({clickTolerance: mapPitchTolerance, enable: true}); + this.touchPitch = generateOneFingerTouchPitchHandler({clickTolerance: mapPitchTolerance, enable: true}); + } + + DOM.addEventListener(element, 'mousedown', this.mousedown); + DOM.addEventListener(element, 'touchstart', this.touchstart, {passive: false}); + DOM.addEventListener(element, 'touchcancel', this.reset); + } + + startMouse(e: MouseEvent, point: Point) { + this.mouseRotate.dragStart(e, point); + if (this.mousePitch) this.mousePitch.dragStart(e, point); + DOM.disableDrag(); + } + + startTouch(e: TouchEvent, point: Point) { + this.touchRotate.dragStart(e, point); + if (this.touchPitch) this.touchPitch.dragStart(e, point); + DOM.disableDrag(); + } + + moveMouse(e: MouseEvent, point: Point) { + const map = this.map; + const {bearingDelta} = this.mouseRotate.dragMove(e, point) || {}; + if (bearingDelta) map.setBearing(map.getBearing() + bearingDelta); + if (this.mousePitch) { + const {pitchDelta} = this.mousePitch.dragMove(e, point) || {}; + if (pitchDelta) map.setPitch(map.getPitch() + pitchDelta); + } + } + + moveTouch(e: TouchEvent, point: Point) { + const map = this.map; + const {bearingDelta} = this.touchRotate.dragMove(e, point) || {}; + if (bearingDelta) map.setBearing(map.getBearing() + bearingDelta); + if (this.touchPitch) { + const {pitchDelta} = this.touchPitch.dragMove(e, point) || {}; + if (pitchDelta) map.setPitch(map.getPitch() + pitchDelta); + } + } + + off() { + const element = this.element; + DOM.removeEventListener(element, 'mousedown', this.mousedown); + DOM.removeEventListener(element, 'touchstart', this.touchstart, {passive: false}); + DOM.removeEventListener(window, 'touchmove', this.touchmove, {passive: false}); + DOM.removeEventListener(window, 'touchend', this.touchend); + DOM.removeEventListener(element, 'touchcancel', this.reset); + this.offTemp(); + } + + offTemp() { + DOM.enableDrag(); + DOM.removeEventListener(window, 'mousemove', this.mousemove); + DOM.removeEventListener(window, 'mouseup', this.mouseup); + DOM.removeEventListener(window, 'touchmove', this.touchmove, {passive: false}); + DOM.removeEventListener(window, 'touchend', this.touchend); + } + + mousedown = (e: MouseEvent) => { + this.startMouse(extend({}, e, {ctrlKey: true, preventDefault: () => e.preventDefault()}), DOM.mousePos(this.element, e)); + DOM.addEventListener(window, 'mousemove', this.mousemove); + DOM.addEventListener(window, 'mouseup', this.mouseup); + }; + + mousemove = (e: MouseEvent) => { + this.moveMouse(e, DOM.mousePos(this.element, e)); + }; + + mouseup = (e: MouseEvent) => { + this.mouseRotate.dragEnd(e); + if (this.mousePitch) this.mousePitch.dragEnd(e); + this.offTemp(); + }; + + touchstart = (e: TouchEvent) => { + if (e.targetTouches.length !== 1) { + this.reset(); + } else { + this._startPos = this._lastPos = DOM.touchPos(this.element, e.targetTouches)[0]; + this.startTouch(e, this._startPos); + DOM.addEventListener(window, 'touchmove', this.touchmove, {passive: false}); + DOM.addEventListener(window, 'touchend', this.touchend); + } + }; + + touchmove = (e: TouchEvent) => { + if (e.targetTouches.length !== 1) { + this.reset(); + } else { + this._lastPos = DOM.touchPos(this.element, e.targetTouches)[0]; + this.moveTouch(e, this._lastPos); + } + }; + + touchend = (e: TouchEvent) => { + if (e.targetTouches.length === 0 && + this._startPos && + this._lastPos && + this._startPos.dist(this._lastPos) < this._clickTolerance) { + this.element.click(); + } + delete this._startPos; + delete this._lastPos; + this.offTemp(); + }; + + reset = () => { + this.mouseRotate.reset(); + if (this.mousePitch) this.mousePitch.reset(); + this.touchRotate.reset(); + if (this.touchPitch) this.touchPitch.reset(); + delete this._startPos; + delete this._lastPos; + this.offTemp(); + }; +} diff --git a/web/libraries/maplibre-gl/src/ui/control/scale_control.test.ts b/web/libraries/maplibre-gl/src/ui/control/scale_control.test.ts new file mode 100644 index 00000000..fbe68e32 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/scale_control.test.ts @@ -0,0 +1,52 @@ +import {createMap, beforeMapTest} from '../../util/test/util'; +import {ScaleControl} from './scale_control'; + +beforeEach(() => { + beforeMapTest(); +}); + +describe('ScaleControl', () => { + test('appears in bottom-left by default', () => { + const map = createMap(undefined, undefined); + map.addControl(new ScaleControl(undefined)); + + expect( + map.getContainer().querySelectorAll('.maplibregl-ctrl-bottom-left .maplibregl-ctrl-scale') + ).toHaveLength(1); + }); + + test('appears in the position specified by the position option', () => { + const map = createMap(undefined, undefined); + map.addControl(new ScaleControl(undefined), 'top-left'); + + expect( + map.getContainer().querySelectorAll('.maplibregl-ctrl-top-left .maplibregl-ctrl-scale') + ).toHaveLength(1); + }); + + test('should change unit of distance after calling setUnit', () => { + const map = createMap(undefined, undefined); + const scale = new ScaleControl(undefined); + const selector = '.maplibregl-ctrl-bottom-left .maplibregl-ctrl-scale'; + map.addControl(scale); + + let contents = map.getContainer().querySelector(selector).innerHTML; + expect(contents).toMatch(/km/); + + scale.setUnit('imperial'); + contents = map.getContainer().querySelector(selector).innerHTML; + expect(contents).toMatch(/mi/); + }); + + test('should respect the maxWidth regardless of the unit and actual scale', () => { + const map = createMap(undefined, undefined); + const maxWidth = 100; + const scale = new ScaleControl({maxWidth, unit: 'nautical'}); + const selector = '.maplibregl-ctrl-bottom-left .maplibregl-ctrl-scale'; + map.addControl(scale); + map.setZoom(12.5); + + const el = map.getContainer().querySelector(selector) as HTMLElement; + expect(parseFloat(el.style.width) <= maxWidth).toBeTruthy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/control/scale_control.ts b/web/libraries/maplibre-gl/src/ui/control/scale_control.ts new file mode 100644 index 00000000..1911f7e4 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/scale_control.ts @@ -0,0 +1,150 @@ +import {DOM} from '../../util/dom'; +import {extend} from '../../util/util'; + +import type {Map} from '../map'; +import type {ControlPosition, IControl} from './control'; + +/** + * The unit type for length to use for the {@link ScaleControl} + */ +export type Unit = 'imperial' | 'metric' | 'nautical'; + +/** + * The {@link ScaleControl} options object + */ +type ScaleOptions = { + /** + * The maximum length of the scale control in pixels. + * @defaultValue 100 + */ + maxWidth?: number; + /** + * Unit of the distance (`'imperial'`, `'metric'` or `'nautical'`). + * @defaultValue 'metric' + */ + unit?: Unit; +}; + +const defaultOptions: ScaleOptions = { + maxWidth: 100, + unit: 'metric' +}; + +/** + * A `ScaleControl` control displays the ratio of a distance on the map to the corresponding distance on the ground. + * + * @group Markers and Controls + * + * @example + * ```ts + * let scale = new maplibregl.ScaleControl({ + * maxWidth: 80, + * unit: 'imperial' + * }); + * map.addControl(scale); + * + * scale.setUnit('metric'); + * ``` + */ +export class ScaleControl implements IControl { + _map: Map; + _container: HTMLElement; + options: ScaleOptions; + + constructor(options: ScaleOptions) { + this.options = extend({}, defaultOptions, options); + } + + getDefaultPosition(): ControlPosition { + return 'bottom-left'; + } + + _onMove = () => { + updateScale(this._map, this._container, this.options); + }; + + /** {@inheritDoc IControl.onAdd} */ + onAdd(map: Map) { + this._map = map; + this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-scale', map.getContainer()); + + this._map.on('move', this._onMove); + this._onMove(); + + return this._container; + } + + /** {@inheritDoc IControl.onRemove} */ + onRemove() { + DOM.remove(this._container); + this._map.off('move', this._onMove); + this._map = undefined; + } + + /** + * Set the scale's unit of the distance + * + * @param unit - Unit of the distance (`'imperial'`, `'metric'` or `'nautical'`). + */ + setUnit = (unit: Unit) => { + this.options.unit = unit; + updateScale(this._map, this._container, this.options); + }; +} + +function updateScale(map, container, options) { + // A horizontal scale is imagined to be present at center of the map + // container with maximum length (Default) as 100px. + // Using spherical law of cosines approximation, the real distance is + // found between the two coordinates. + const maxWidth = options && options.maxWidth || 100; + + const y = map._container.clientHeight / 2; + const left = map.unproject([0, y]); + const right = map.unproject([maxWidth, y]); + const maxMeters = left.distanceTo(right); + // The real distance corresponding to 100px scale length is rounded off to + // near pretty number and the scale length for the same is found out. + // Default unit of the scale is based on User's locale. + if (options && options.unit === 'imperial') { + const maxFeet = 3.2808 * maxMeters; + if (maxFeet > 5280) { + const maxMiles = maxFeet / 5280; + setScale(container, maxWidth, maxMiles, map._getUIString('ScaleControl.Miles')); + } else { + setScale(container, maxWidth, maxFeet, map._getUIString('ScaleControl.Feet')); + } + } else if (options && options.unit === 'nautical') { + const maxNauticals = maxMeters / 1852; + setScale(container, maxWidth, maxNauticals, map._getUIString('ScaleControl.NauticalMiles')); + } else if (maxMeters >= 1000) { + setScale(container, maxWidth, maxMeters / 1000, map._getUIString('ScaleControl.Kilometers')); + } else { + setScale(container, maxWidth, maxMeters, map._getUIString('ScaleControl.Meters')); + } +} + +function setScale(container, maxWidth, maxDistance, unit) { + const distance = getRoundNum(maxDistance); + const ratio = distance / maxDistance; + container.style.width = `${maxWidth * ratio}px`; + container.innerHTML = `${distance} ${unit}`; +} + +function getDecimalRoundNum(d) { + const multiplier = Math.pow(10, Math.ceil(-Math.log(d) / Math.LN10)); + return Math.round(d * multiplier) / multiplier; +} + +function getRoundNum(num) { + const pow10 = Math.pow(10, (`${Math.floor(num)}`).length - 1); + let d = num / pow10; + + d = d >= 10 ? 10 : + d >= 5 ? 5 : + d >= 3 ? 3 : + d >= 2 ? 2 : + d >= 1 ? 1 : getDecimalRoundNum(d); + + return pow10 * d; +} diff --git a/web/libraries/maplibre-gl/src/ui/control/terrain_control.test.ts b/web/libraries/maplibre-gl/src/ui/control/terrain_control.test.ts new file mode 100644 index 00000000..f7f93d95 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/terrain_control.test.ts @@ -0,0 +1,58 @@ +import {TerrainControl} from './terrain_control'; +import {createMap as globalCreateMap, beforeMapTest} from '../../util/test/util'; + +function createMap() { + + return globalCreateMap({ + attributionControl: false, + style: { + version: 8, + sources: { + terrain: { + minzoom: 5, + maxzoom: 12, + attribution: 'MapLibre', + tiles: ['http://example.com/{z}/{x}/{y}.pngraw'], + bounds: [-47, -7, -45, -5] + }, + }, + layers: [], + owner: 'mapblibre', + id: 'demotiles', + }, + hash: true + }, undefined); +} + +let map; + +beforeEach(() => { + beforeMapTest(); + map = createMap(); +}); + +afterEach(() => { + map.remove(); +}); + +describe('TerrainControl', () => { + test('appears in top-right by default', () => { + map.addControl(new TerrainControl({source: 'terrain'})); + + expect( + map.getContainer().querySelectorAll('.maplibregl-ctrl-top-right .maplibregl-ctrl-terrain') + ).toHaveLength(1); + }); + + test('appears in the position specified by the position option', () => { + map.addControl(new TerrainControl({source: 'terrain'}), 'bottom-right'); + + expect( + map.getContainer().querySelectorAll('.maplibregl-ctrl-bottom-right .maplibregl-ctrl-terrain') + ).toHaveLength(1); + + expect( + map.getContainer().querySelectorAll('.maplibregl-ctrl-top-right .maplibregl-ctrl-terrain') + ).toHaveLength(0); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/control/terrain_control.ts b/web/libraries/maplibre-gl/src/ui/control/terrain_control.ts new file mode 100644 index 00000000..8cc85385 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/control/terrain_control.ts @@ -0,0 +1,71 @@ +import {DOM} from '../../util/dom'; + +import type {Map} from '../map'; +import type {IControl} from './control'; +import type {TerrainSpecification} from '@maplibre/maplibre-gl-style-spec'; + +/** + * A `TerrainControl` control contains a button for turning the terrain on and off. + * + * @group Markers and Controls + * + * @example + * ```ts + * let map = new maplibregl.Map({TerrainControl: false}) + * .addControl(new maplibregl.TerrainControl({ + * source: "terrain" + * })); + * ``` + */ +export class TerrainControl implements IControl { + options: TerrainSpecification; + _map: Map; + _container: HTMLElement; + _terrainButton: HTMLButtonElement; + + constructor(options: TerrainSpecification) { + this.options = options; + } + + /** {@inheritDoc IControl.onAdd} */ + onAdd(map: Map) { + this._map = map; + this._container = DOM.create('div', 'maplibregl-ctrl maplibregl-ctrl-group'); + this._terrainButton = DOM.create('button', 'maplibregl-ctrl-terrain', this._container); + DOM.create('span', 'maplibregl-ctrl-icon', this._terrainButton).setAttribute('aria-hidden', 'true'); + this._terrainButton.type = 'button'; + this._terrainButton.addEventListener('click', this._toggleTerrain); + + this._updateTerrainIcon(); + this._map.on('terrain', this._updateTerrainIcon); + return this._container; + } + + /** {@inheritDoc IControl.onRemove} */ + onRemove() { + DOM.remove(this._container); + this._map.off('terrain', this._updateTerrainIcon); + this._map = undefined; + } + + _toggleTerrain = () => { + if (this._map.getTerrain()) { + this._map.setTerrain(null); + } else { + this._map.setTerrain(this.options); + } + this._updateTerrainIcon(); + }; + + _updateTerrainIcon = () => { + this._terrainButton.classList.remove('maplibregl-ctrl-terrain'); + this._terrainButton.classList.remove('maplibregl-ctrl-terrain-enabled'); + if (this._map.terrain) { + this._terrainButton.classList.add('maplibregl-ctrl-terrain-enabled'); + this._terrainButton.title = this._map._getUIString('TerrainControl.disableTerrain'); + } else { + this._terrainButton.classList.add('maplibregl-ctrl-terrain'); + this._terrainButton.title = this._map._getUIString('TerrainControl.enableTerrain'); + } + }; +} diff --git a/web/libraries/maplibre-gl/src/ui/default_locale.ts b/web/libraries/maplibre-gl/src/ui/default_locale.ts new file mode 100644 index 00000000..59e70365 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/default_locale.ts @@ -0,0 +1,19 @@ +export const defaultLocale = { + 'AttributionControl.ToggleAttribution': 'Toggle attribution', + 'AttributionControl.MapFeedback': 'Map feedback', + 'FullscreenControl.Enter': 'Enter fullscreen', + 'FullscreenControl.Exit': 'Exit fullscreen', + 'GeolocateControl.FindMyLocation': 'Find my location', + 'GeolocateControl.LocationNotAvailable': 'Location not available', + 'LogoControl.Title': 'Mapbox logo', + 'NavigationControl.ResetBearing': 'Reset bearing to north', + 'NavigationControl.ZoomIn': 'Zoom in', + 'NavigationControl.ZoomOut': 'Zoom out', + 'ScaleControl.Feet': 'ft', + 'ScaleControl.Meters': 'm', + 'ScaleControl.Kilometers': 'km', + 'ScaleControl.Miles': 'mi', + 'ScaleControl.NauticalMiles': 'nm', + 'TerrainControl.enableTerrain': 'Enable terrain', + 'TerrainControl.disableTerrain': 'Disable terrain' +}; diff --git a/web/libraries/maplibre-gl/src/ui/events.ts b/web/libraries/maplibre-gl/src/ui/events.ts new file mode 100644 index 00000000..8d2dd6fa --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/events.ts @@ -0,0 +1,748 @@ +import {Event} from '../util/evented'; + +import {DOM} from '../util/dom'; +import Point from '@mapbox/point-geometry'; +import {extend} from '../util/util'; +import type {MapGeoJSONFeature} from '../util/vectortile_to_geojson'; + +import type {Map} from './map'; +import type {LngLat} from '../geo/lng_lat'; +import {SourceSpecification} from '@maplibre/maplibre-gl-style-spec'; + +/** + * An event from the mouse relevant to a specific layer. + * + * @group Event Related + */ +export type MapLayerMouseEvent = MapMouseEvent & { features?: MapGeoJSONFeature[] }; + +/** + * An event from a touch device relevat to a specific layer. + * + * @group Event Related + */ +export type MapLayerTouchEvent = MapTouchEvent & { features?: MapGeoJSONFeature[] }; + +/** + * The source event data type + */ +export type MapSourceDataType = 'content' | 'metadata' | 'visibility' | 'idle'; + +/** + * `MapLayerEventType` - a mapping between the event name and the event. + * **Note:** These events are compatible with the optional `layerId` parameter. + * If `layerId` is included as the second argument in {@link Map#on}, the event listener will fire only when the + * event action contains a visible portion of the specified layer. + * The following example can be used for all the events. + * + * @group Event Related + * @example + * ```ts + * // Initialize the map + * let map = new maplibregl.Map({ // map options }); + * // Set an event listener for a specific layer + * map.on('the-event-name', 'poi-label', function(e) { + * console.log('An event has occurred on a visible portion of the poi-label layer'); + * }); + * ``` + */ +export type MapLayerEventType = { + /** + * Fired when a pointing device (usually a mouse) is pressed and released contains a visible portion of the specified layer. + * + * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/) + * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/) + */ + click: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is pressed and released twice contains a visible portion of the specified layer. + * + * **Note:** Under normal conditions, this event will be preceded by two `click` events. + */ + dblclick: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is pressed while inside a visible portion of the specified layer. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + mousedown: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is released while inside a visible portion of the specified layer. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + mouseup: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is moved while the cursor is inside a visible portion of the specified layer. + * As you move the cursor across the layer, the event will fire every time the cursor changes position within that layer. + * + * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/) + * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + * @see [Display a popup on over](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + */ + mousemove: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) enters a visible portion of a specified layer from + * outside that layer or outside the map canvas. + * + * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/) + * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + */ + mouseenter: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) leaves a visible portion of a specified layer, or leaves + * the map canvas. + * + * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + */ + mouseleave: MapLayerMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is moved inside a visible portion of the specified layer. + * + * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/) + * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + */ + mouseover: MapLayerMouseEvent; + /** + * Fired when a point device (usually a mouse) leaves the visible portion of the specified layer. + */ + mouseout: MapLayerMouseEvent; + /** + * Fired when the right button of the mouse is clicked or the context menu key is pressed within visible portion of the specified layer. + */ + contextmenu: MapLayerMouseEvent; + /** + * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the visible portion of the specified layer. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + touchstart: MapLayerTouchEvent; + /** + * Fired when a [`touchend`](https://developer.mozilla.org/en-US/docs/Web/Events/touchend) event occurs within the visible portion of the specified layer. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + touchend: MapLayerTouchEvent; + /** + * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the visible portion of the specified layer. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + touchcancel: MapLayerTouchEvent; +}; + +/** + * `MapEventType` - a mapping between the event name and the event value. + * These events are used with the {@link Map#on} method. + * When using a `layerId` with {@link Map#on} method, please refer to {@link MapLayerEventType}. + * The following example can be used for all the events. + * + * @group Event Related + * @example + * ```ts + * // Initialize the map + * let map = new maplibregl.Map({ // map options }); + * // Set an event listener + * map.on('the-event-name', () => { + * console.log('An event has occurred!'); + * }); + * ``` + */ +export type MapEventType = { + /** + * Fired when an error occurs. This is GL JS's primary error reporting + * mechanism. We use an event instead of `throw` to better accommodate + * asynchronous operations. If no listeners are bound to the `error` event, the + * error will be printed to the console. + */ + error: ErrorEvent; + /** + * @event `load` Fired immediately after all necessary resources have been downloaded + * and the first visually complete rendering of the map has occurred. + * + * @see [Draw GeoJSON points](https://maplibre.org/maplibre-gl-js/docs/examples/geojson-markers/) + * @see [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/) + * @see [Animate a point](https://maplibre.org/maplibre-gl-js/docs/examples/animate-point-along-line/) + */ + load: MapLibreEvent; + /** + * Fired after the last frame rendered before the map enters an + * "idle" state: + * + * - No camera transitions are in progress + * - All currently requested tiles have loaded + * - All fade/transition animations have completed + */ + idle: MapLibreEvent; + /** + * Fired immediately after the map has been removed with {@link Map#remove}. + */ + remove: MapLibreEvent; + /** + * Fired whenever the map is drawn to the screen, as the result of + * + * - a change to the map's position, zoom, pitch, or bearing + * - a change to the map's style + * - a change to a GeoJSON source + * - the loading of a vector tile, GeoJSON file, glyph, or sprite + */ + render: MapLibreEvent; + /** + * Fired immediately after the map has been resized. + */ + resize: MapLibreEvent; + /** + * Fired when the WebGL context is lost. + */ + webglcontextlost: MapContextEvent; + /** + * Fired when the WebGL context is restored. + */ + webglcontextrestored: MapContextEvent; + /** + * Fired when any map data (style, source, tile, etc) begins loading or + * changing asynchronously. All `dataloading` events are followed by a `data`, + * `dataabort` or `error` event. + */ + dataloading: MapDataEvent; + /** + * Fired when any map data loads or changes. See {@link MapDataEvent} for more information. + * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/) + */ + data: MapDataEvent; + tiledataloading: MapDataEvent; + /** + * Fired when one of the map's sources begins loading or changing asynchronously. + * All `sourcedataloading` events are followed by a `sourcedata`, `sourcedataabort` or `error` event. + */ + sourcedataloading: MapSourceDataEvent; + /** + * Fired when the map's style begins loading or changing asynchronously. + * All `styledataloading` events are followed by a `styledata` + * or `error` event. + */ + styledataloading: MapStyleDataEvent; + /** + * Fired when one of the map's sources loads or changes, including if a tile belonging + * to a source loads or changes. + */ + sourcedata: MapSourceDataEvent; + /** + * Fired when the map's style loads or changes. + */ + styledata: MapStyleDataEvent; + /** + * Fired when an icon or pattern needed by the style is missing. The missing image can + * be added with {@link Map#addImage} within this event listener callback to prevent the image from + * being skipped. This event can be used to dynamically generate icons and patterns. + * @see [Generate and add a missing icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-missing-generated/) + */ + styleimagemissing: MapStyleImageMissingEvent; + /** + * Fired when a request for one of the map's sources' tiles or data is aborted. + */ + dataabort: MapDataEvent; + /** + * Fired when a request for one of the map's sources' data is aborted. + */ + sourcedataabort: MapSourceDataEvent; + /** + * Fired when the user cancels a "box zoom" interaction, or when the bounding box does not meet the minimum size threshold. + * See {@link BoxZoomHandler}. + */ + boxzoomcancel: MapLibreZoomEvent; + /** + * Fired when a "box zoom" interaction starts. See {@link BoxZoomHandler}. + */ + boxzoomstart: MapLibreZoomEvent; + /** + * Fired when a "box zoom" interaction ends. See {@link BoxZoomHandler}. + */ + boxzoomend: MapLibreZoomEvent; + /** + * Fired when a [`touchcancel`](https://developer.mozilla.org/en-US/docs/Web/Events/touchcancel) event occurs within the map. + */ + touchcancel: MapTouchEvent; + /** + * Fired when a [`touchmove`](https://developer.mozilla.org/en-US/docs/Web/Events/touchmove) event occurs within the map. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + touchmove: MapTouchEvent; + /** + * Fired when a [`touchend`](https://developer.mozilla.org/en-US/docs/Web/Events/touchend) event occurs within the map. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + touchend: MapTouchEvent; + /** + * Fired when a [`touchstart`](https://developer.mozilla.org/en-US/docs/Web/Events/touchstart) event occurs within the map. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + touchstart: MapTouchEvent; + /** + * Fired when a pointing device (usually a mouse) is pressed and released at the same point on the map. + * + * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/) + * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/) + */ + click: MapMouseEvent; + /** + * Fired when the right button of the mouse is clicked or the context menu key is pressed within the map. + */ + contextmenu: MapMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is pressed and released twice at the same point on the map in rapid succession. + * + * **Note:** Under normal conditions, this event will be preceded by two `click` events. + */ + dblclick: MapMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is moved while the cursor is inside the map. + * As you move the cursor across the map, the event will fire every time the cursor changes position within the map. + * + * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/) + * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + * @see [Display a popup on over](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + */ + mousemove: MapMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is released within the map. + * + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + mouseup: MapMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is pressed within the map. + * + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + mousedown: MapMouseEvent; + /** + * Fired when a point device (usually a mouse) leaves the map's canvas. + */ + mouseout: MapMouseEvent; + /** + * Fired when a pointing device (usually a mouse) is moved within the map. + * As you move the cursor across a web page containing a map, + * the event will fire each time it enters the map or any child elements. + * + * @see [Get coordinates of the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/mouse-position/) + * @see [Highlight features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + */ + mouseover: MapMouseEvent; + /** + * Fired just before the map begins a transition from one + * view to another, as the result of either user interaction or methods such as {@link Map#jumpTo}. + * + */ + movestart: MapLibreEvent; + /** + * Fired repeatedly during an animated transition from one view to + * another, as the result of either user interaction or methods such as {@link Map#flyTo}. + * + * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/) + */ + move: MapLibreEvent; + /** + * Fired just after the map completes a transition from one + * view to another, as the result of either user interaction or methods such as {@link Map#jumpTo}. + * + * @see [Display HTML clusters with custom properties](https://maplibre.org/maplibre-gl-js/docs/examples/cluster-html/) + */ + moveend: MapLibreEvent; + /** + * Fired just before the map begins a transition from one zoom level to another, + * as the result of either user interaction or methods such as {@link Map#flyTo}. + */ + zoomstart: MapLibreEvent; + /** + * Fired repeatedly during an animated transition from one zoom level to another, + * as the result of either user interaction or methods such as {@link Map#flyTo}. + */ + zoom: MapLibreEvent; + /** + * Fired just after the map completes a transition from one zoom level to another, + * as the result of either user interaction or methods such as {@link Map#flyTo}. + */ + zoomend: MapLibreEvent; + /** + * Fired when a "drag to rotate" interaction starts. See {@link DragRotateHandler}. + */ + rotatestart: MapLibreEvent; + /** + * Fired repeatedly during a "drag to rotate" interaction. See {@link DragRotateHandler}. + */ + rotate: MapLibreEvent; + /** + * Fired when a "drag to rotate" interaction ends. See {@link DragRotateHandler}. + */ + rotateend: MapLibreEvent; + /** + * Fired when a "drag to pan" interaction starts. See {@link DragPanHandler}. + */ + dragstart: MapLibreEvent; + /** + * Fired repeatedly during a "drag to pan" interaction. See {@link DragPanHandler}. + */ + drag: MapLibreEvent; + /** + * Fired when a "drag to pan" interaction ends. See {@link DragPanHandler}. + * @see [Create a draggable marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/) + */ + dragend: MapLibreEvent; + /** + * Fired whenever the map's pitch (tilt) begins a change as + * the result of either user interaction or methods such as {@link Map#flyTo} . + */ + pitchstart: MapLibreEvent; + /** + * Fired repeatedly during the map's pitch (tilt) animation between + * one state and another as the result of either user interaction + * or methods such as {@link Map#flyTo}. + */ + pitch: MapLibreEvent; + /** + * Fired immediately after the map's pitch (tilt) finishes changing as + * the result of either user interaction or methods such as {@link Map#flyTo}. + */ + pitchend: MapLibreEvent; + /** + * Fired when a [`wheel`](https://developer.mozilla.org/en-US/docs/Web/Events/wheel) event occurs within the map. + */ + wheel: MapWheelEvent; + /** + * Fired when terrain is changed + */ + terrain: MapTerrainEvent; +}; + +/** + * The base event for MapLibre + * + * @group Event Related + */ +export type MapLibreEvent = { + type: keyof MapEventType | keyof MapLayerEventType; + target: Map; + originalEvent: TOrig; +} + +/** + * The style data event + * + * @group Event Related + */ +export type MapStyleDataEvent = MapLibreEvent & { + dataType: 'style'; +} + +/** + * The source data event interface + * + * @group Event Related + */ +export type MapSourceDataEvent = MapLibreEvent & { + dataType: 'source'; + /** + * True if the event has a `dataType` of `source` and the source has no outstanding network requests. + */ + isSourceLoaded: boolean; + /** + * The [style spec representation of the source](https://maplibre.org/maplibre-style-spec/#sources) if the event has a `dataType` of `source`. + */ + source: SourceSpecification; + sourceId: string; + sourceDataType: MapSourceDataType; + /** + * The tile being loaded or changed, if the event has a `dataType` of `source` and + * the event is related to loading of a tile. + */ + tile: any; +} +/** + * `MapMouseEvent` is the event type for mouse-related map events. + * @example + * ```ts + * // The `click` event is an example of a `MapMouseEvent`. + * // Set up an event listener on the map. + * map.on('click', function(e) { + * // The event object (e) contains information like the + * // coordinates of the point on the map that was clicked. + * console.log('A click event has occurred at ' + e.lngLat); + * }); + * ``` + */ +export class MapMouseEvent extends Event implements MapLibreEvent { + /** + * The event type + */ + type: 'mousedown' | 'mouseup' | 'click' | 'dblclick' | 'mousemove' | 'mouseover' | 'mouseenter' | 'mouseleave' | 'mouseout' | 'contextmenu'; + + /** + * The `Map` object that fired the event. + */ + target: Map; + + /** + * The DOM event which caused the map event. + */ + originalEvent: MouseEvent; + + /** + * The pixel coordinates of the mouse cursor, relative to the map and measured from the top left corner. + */ + point: Point; + + /** + * The geographic location on the map of the mouse cursor. + */ + lngLat: LngLat; + + /** + * Prevents subsequent default processing of the event by the map. + * + * Calling this method will prevent the following default map behaviors: + * + * * On `mousedown` events, the behavior of {@link DragPanHandler} + * * On `mousedown` events, the behavior of {@link DragRotateHandler} + * * On `mousedown` events, the behavior of {@link BoxZoomHandler} + * * On `dblclick` events, the behavior of {@link DoubleClickZoomHandler} + * + */ + preventDefault() { + this._defaultPrevented = true; + } + + /** + * `true` if `preventDefault` has been called. + */ + get defaultPrevented(): boolean { + return this._defaultPrevented; + } + + _defaultPrevented: boolean; + + constructor(type: string, map: Map, originalEvent: MouseEvent, data: any = {}) { + const point = DOM.mousePos(map.getCanvasContainer(), originalEvent); + const lngLat = map.unproject(point); + super(type, extend({point, lngLat, originalEvent}, data)); + this._defaultPrevented = false; + this.target = map; + } +} + +/** + * `MapTouchEvent` is the event type for touch-related map events. + * + * @group Event Related + */ +export class MapTouchEvent extends Event implements MapLibreEvent { + /** + * The event type. + */ + type: 'touchstart' | 'touchmove' | 'touchend' | 'touchcancel'; + + /** + * The `Map` object that fired the event. + */ + target: Map; + + /** + * The DOM event which caused the map event. + */ + originalEvent: TouchEvent; + + /** + * The geographic location on the map of the center of the touch event points. + */ + lngLat: LngLat; + + /** + * The pixel coordinates of the center of the touch event points, relative to the map and measured from the top left + * corner. + */ + point: Point; + + /** + * The array of pixel coordinates corresponding to a + * [touch event's `touches`](https://developer.mozilla.org/en-US/docs/Web/API/TouchEvent/touches) property. + */ + points: Array; + + /** + * The geographical locations on the map corresponding to a + * [touch event's `touches`](https://developer.mozilla.org/en-US/docs/Web/API/TouchEvent/touches) property. + */ + lngLats: Array; + + /** + * Prevents subsequent default processing of the event by the map. + * + * Calling this method will prevent the following default map behaviors: + * + * * On `touchstart` events, the behavior of {@link DragPanHandler} + * * On `touchstart` events, the behavior of {@link TwoFingersTouchZoomRotateHandler} + * + */ + preventDefault() { + this._defaultPrevented = true; + } + + /** + * `true` if `preventDefault` has been called. + */ + get defaultPrevented(): boolean { + return this._defaultPrevented; + } + + _defaultPrevented: boolean; + + constructor(type: string, map: Map, originalEvent: TouchEvent) { + const touches = type === 'touchend' ? originalEvent.changedTouches : originalEvent.touches; + const points = DOM.touchPos(map.getCanvasContainer(), touches); + const lngLats = points.map((t) => map.unproject(t)); + const point = points.reduce((prev, curr, i, arr) => { + return prev.add(curr.div(arr.length)); + }, new Point(0, 0)); + const lngLat = map.unproject(point); + super(type, {points, point, lngLats, lngLat, originalEvent}); + this._defaultPrevented = false; + } +} + +/** + * `MapWheelEvent` is the event type for the `wheel` map event. + * + * @group Event Related + */ +export class MapWheelEvent extends Event { + /** + * The event type. + */ + type: 'wheel'; + + /** + * The `Map` object that fired the event. + */ + target: Map; + + /** + * The DOM event which caused the map event. + */ + originalEvent: WheelEvent; + + /** + * Prevents subsequent default processing of the event by the map. + * + * Calling this method will prevent the behavior of {@link ScrollZoomHandler}. + */ + preventDefault() { + this._defaultPrevented = true; + } + + /** + * `true` if `preventDefault` has been called. + */ + get defaultPrevented(): boolean { + return this._defaultPrevented; + } + + _defaultPrevented: boolean; + + /** */ + constructor(type: string, map: Map, originalEvent: WheelEvent) { + super(type, {originalEvent}); + this._defaultPrevented = false; + } +} + +/** + * A `MapLibreZoomEvent` is the event type for the boxzoom-related map events emitted by the {@link BoxZoomHandler}. + * + * @group Event Related + */ +export type MapLibreZoomEvent = { + /** + * The type of boxzoom event. One of `boxzoomstart`, `boxzoomend` or `boxzoomcancel` + */ + type: 'boxzoomstart' | 'boxzoomend' | 'boxzoomcancel'; + /** + * The `Map` instance that triggered the event + */ + target: Map; + /** + * The DOM event that triggered the boxzoom event. Can be a `MouseEvent` or `KeyboardEvent` + */ + originalEvent: MouseEvent; +}; + +/** + * A `MapDataEvent` object is emitted with the `data` + * and `dataloading` events. Possible values for + * `dataType`s are: + * + * - `'source'`: The non-tile data associated with any source + * - `'style'`: The [style](https://maplibre.org/maplibre-style-spec/) used by the map + * + * Possible values for `sourceDataType`s are: + * + * - `'metadata'`: indicates that any necessary source metadata has been loaded (such as TileJSON) and it is ok to start loading tiles + * - `'content'`: indicates the source data has changed (such as when source.setData() has been called on GeoJSONSource) + * - `'visibility'`: send when the source becomes used when at least one of its layers becomes visible in style sense (inside the layer's zoom range and with layout.visibility set to 'visible') + * - `'idle'`: indicates that no new source data has been fetched (but the source has done loading) + * + * @group Event Related + * + * @example + * ```ts + * // The sourcedata event is an example of MapDataEvent. + * // Set up an event listener on the map. + * map.on('sourcedata', function(e) { + * if (e.isSourceLoaded) { + * // Do something when the source has finished loading + * } + * }); + * ``` + */ +export type MapDataEvent = { + /** + * The event type. + */ + type: string; + /** + * The type of data that has changed. One of `'source'`, `'style'`. + */ + dataType: string; + /** + * Included if the event has a `dataType` of `source` and the event signals that internal data has been received or changed. Possible values are `metadata`, `content`, `visibility` and `idle`. + */ + sourceDataType: MapSourceDataType; +}; + +/** + * The terrain event + * + * @group Event Related + */ +export type MapTerrainEvent = { + type: 'terrain'; +}; + +/** + * An event related to the web gl context + * + * @group Event Related + */ +export type MapContextEvent = { + type: 'webglcontextlost' | 'webglcontextrestored'; + originalEvent: WebGLContextEvent; +}; + +/** + * The style image missing event + * + * @group Event Related + * + * @see [Generate and add a missing icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-missing-generated/) + */ +export type MapStyleImageMissingEvent = MapLibreEvent & { + type: 'styleimagemissing'; + id: string; +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/box_zoom.test.ts b/web/libraries/maplibre-gl/src/ui/handler/box_zoom.test.ts new file mode 100644 index 00000000..6e6145cb --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/box_zoom.test.ts @@ -0,0 +1,163 @@ +import {Map} from '../map'; +import {DOM} from '../../util/dom'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {beforeMapTest} from '../../util/test/util'; + +function createMap(clickTolerance) { + return new Map({style: '', container: DOM.create('div', '', window.document.body), clickTolerance}); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('BoxZoomHandler', () => { + test('fires boxzoomstart and boxzoomend events at appropriate times', () => { + const map = createMap(undefined); + + const boxzoomstart = jest.fn(); + const boxzoomend = jest.fn(); + + map.on('boxzoomstart', boxzoomstart); + map.on('boxzoomend', boxzoomend); + + simulate.mousedown(map.getCanvas(), {shiftKey: true, clientX: 0, clientY: 0}); + map._renderTaskQueue.run(); + expect(boxzoomstart).not.toHaveBeenCalled(); + expect(boxzoomend).not.toHaveBeenCalled(); + + simulate.mousemove(map.getCanvas(), {shiftKey: true, clientX: 5, clientY: 5}); + map._renderTaskQueue.run(); + expect(boxzoomstart).toHaveBeenCalledTimes(1); + expect(boxzoomend).not.toHaveBeenCalled(); + + simulate.mouseup(map.getCanvas(), {shiftKey: true, clientX: 5, clientY: 5}); + map._renderTaskQueue.run(); + expect(boxzoomstart).toHaveBeenCalledTimes(1); + expect(boxzoomend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('avoids conflicts with DragPanHandler when disabled and reenabled (#2237)', () => { + const map = createMap(undefined); + + map.boxZoom.disable(); + map.boxZoom.enable(); + + const boxzoomstart = jest.fn(); + const boxzoomend = jest.fn(); + + map.on('boxzoomstart', boxzoomstart); + map.on('boxzoomend', boxzoomend); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.mousedown(map.getCanvas(), {shiftKey: true, clientX: 0, clientY: 0}); + map._renderTaskQueue.run(); + expect(boxzoomstart).not.toHaveBeenCalled(); + expect(boxzoomend).not.toHaveBeenCalled(); + + simulate.mousemove(map.getCanvas(), {shiftKey: true, clientX: 5, clientY: 5}); + map._renderTaskQueue.run(); + expect(boxzoomstart).toHaveBeenCalledTimes(1); + expect(boxzoomend).not.toHaveBeenCalled(); + + simulate.mouseup(map.getCanvas(), {shiftKey: true, clientX: 5, clientY: 5}); + map._renderTaskQueue.run(); + expect(boxzoomstart).toHaveBeenCalledTimes(1); + expect(boxzoomend).toHaveBeenCalledTimes(1); + + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + + map.remove(); + }); + + test('does not begin a box zoom if preventDefault is called on the mousedown event', () => { + const map = createMap(undefined); + + map.on('mousedown', e => e.preventDefault()); + + const boxzoomstart = jest.fn(); + const boxzoomend = jest.fn(); + + map.on('boxzoomstart', boxzoomstart); + map.on('boxzoomend', boxzoomend); + + simulate.mousedown(map.getCanvas(), {shiftKey: true, clientX: 0, clientY: 0}); + map._renderTaskQueue.run(); + + simulate.mousemove(map.getCanvas(), {shiftKey: true, clientX: 5, clientY: 5}); + map._renderTaskQueue.run(); + + simulate.mouseup(map.getCanvas(), {shiftKey: true, clientX: 5, clientY: 5}); + map._renderTaskQueue.run(); + + expect(boxzoomstart).not.toHaveBeenCalled(); + expect(boxzoomend).not.toHaveBeenCalled(); + + map.remove(); + }); + + test('does not begin a box zoom on spurious mousemove events', () => { + const map = createMap(undefined); + + const boxzoomstart = jest.fn(); + const boxzoomend = jest.fn(); + + map.on('boxzoomstart', boxzoomstart); + map.on('boxzoomend', boxzoomend); + + simulate.mousedown(map.getCanvas(), {shiftKey: true, clientX: 0, clientY: 0}); + map._renderTaskQueue.run(); + expect(boxzoomstart).not.toHaveBeenCalled(); + expect(boxzoomend).not.toHaveBeenCalled(); + + simulate.mousemove(map.getCanvas(), {shiftKey: true, clientX: 0, clientY: 0}); + map._renderTaskQueue.run(); + expect(boxzoomstart).not.toHaveBeenCalled(); + expect(boxzoomend).not.toHaveBeenCalled(); + + simulate.mouseup(map.getCanvas(), {shiftKey: true, clientX: 0, clientY: 0}); + map._renderTaskQueue.run(); + expect(boxzoomstart).not.toHaveBeenCalled(); + expect(boxzoomend).not.toHaveBeenCalled(); + + map.remove(); + }); + + test('does not begin a box zoom until mouse move is larger than click tolerance', () => { + const map = createMap(4); + + const boxzoomstart = jest.fn(); + const boxzoomend = jest.fn(); + + map.on('boxzoomstart', boxzoomstart); + map.on('boxzoomend', boxzoomend); + + simulate.mousedown(map.getCanvas(), {shiftKey: true, clientX: 0, clientY: 0}); + map._renderTaskQueue.run(); + expect(boxzoomstart).not.toHaveBeenCalled(); + expect(boxzoomend).not.toHaveBeenCalled(); + + simulate.mousemove(map.getCanvas(), {shiftKey: true, clientX: 3, clientY: 0}); + map._renderTaskQueue.run(); + expect(boxzoomstart).not.toHaveBeenCalled(); + expect(boxzoomend).not.toHaveBeenCalled(); + + simulate.mousemove(map.getCanvas(), {shiftKey: true, clientX: 0, clientY: 4}); + map._renderTaskQueue.run(); + expect(boxzoomstart).toHaveBeenCalledTimes(1); + expect(boxzoomend).not.toHaveBeenCalled(); + + map.remove(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/box_zoom.ts b/web/libraries/maplibre-gl/src/ui/handler/box_zoom.ts new file mode 100644 index 00000000..a634dc76 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/box_zoom.ts @@ -0,0 +1,171 @@ +import {DOM} from '../../util/dom'; + +import {Event} from '../../util/evented'; +import {TransformProvider} from './transform-provider'; + +import type {Map} from '../map'; +import type Point from '@mapbox/point-geometry'; +import {Handler} from '../handler_manager'; + +/** + * The `BoxZoomHandler` allows the user to zoom the map to fit within a bounding box. + * The bounding box is defined by clicking and holding `shift` while dragging the cursor. + * + * @group Handlers + */ +export class BoxZoomHandler implements Handler { + _map: Map; + _tr: TransformProvider; + _el: HTMLElement; + _container: HTMLElement; + _enabled: boolean; + _active: boolean; + _startPos: Point; + _lastPos: Point; + _box: HTMLElement; + _clickTolerance: number; + + /** @internal */ + constructor(map: Map, options: { + clickTolerance: number; + }) { + this._map = map; + this._tr = new TransformProvider(map); + this._el = map.getCanvasContainer(); + this._container = map.getContainer(); + this._clickTolerance = options.clickTolerance || 1; + } + + /** + * Returns a Boolean indicating whether the "box zoom" interaction is enabled. + * + * @returns `true` if the "box zoom" interaction is enabled. + */ + isEnabled() { + return !!this._enabled; + } + + /** + * Returns a Boolean indicating whether the "box zoom" interaction is active, i.e. currently being used. + * + * @returns `true` if the "box zoom" interaction is active. + */ + isActive() { + return !!this._active; + } + + /** + * Enables the "box zoom" interaction. + * + * @example + * ```ts + * map.boxZoom.enable(); + * ``` + */ + enable() { + if (this.isEnabled()) return; + this._enabled = true; + } + + /** + * Disables the "box zoom" interaction. + * + * @example + * ```ts + * map.boxZoom.disable(); + * ``` + */ + disable() { + if (!this.isEnabled()) return; + this._enabled = false; + } + + mousedown(e: MouseEvent, point: Point) { + if (!this.isEnabled()) return; + if (!(e.shiftKey && e.button === 0)) return; + + DOM.disableDrag(); + this._startPos = this._lastPos = point; + this._active = true; + } + + mousemoveWindow(e: MouseEvent, point: Point) { + if (!this._active) return; + + const pos = point; + + if (this._lastPos.equals(pos) || (!this._box && pos.dist(this._startPos) < this._clickTolerance)) { + return; + } + + const p0 = this._startPos; + this._lastPos = pos; + + if (!this._box) { + this._box = DOM.create('div', 'maplibregl-boxzoom', this._container); + this._container.classList.add('maplibregl-crosshair'); + this._fireEvent('boxzoomstart', e); + } + + const minX = Math.min(p0.x, pos.x), + maxX = Math.max(p0.x, pos.x), + minY = Math.min(p0.y, pos.y), + maxY = Math.max(p0.y, pos.y); + + DOM.setTransform(this._box, `translate(${minX}px,${minY}px)`); + + this._box.style.width = `${maxX - minX}px`; + this._box.style.height = `${maxY - minY}px`; + } + + mouseupWindow(e: MouseEvent, point: Point) { + if (!this._active) return; + + if (e.button !== 0) return; + + const p0 = this._startPos, + p1 = point; + + this.reset(); + + DOM.suppressClick(); + + if (p0.x === p1.x && p0.y === p1.y) { + this._fireEvent('boxzoomcancel', e); + } else { + this._map.fire(new Event('boxzoomend', {originalEvent: e})); + return { + cameraAnimation: map => map.fitScreenCoordinates(p0, p1, this._tr.bearing, {linear: true}) + }; + } + } + + keydown(e: KeyboardEvent) { + if (!this._active) return; + + if (e.keyCode === 27) { + this.reset(); + this._fireEvent('boxzoomcancel', e); + } + } + + reset() { + this._active = false; + + this._container.classList.remove('maplibregl-crosshair'); + + if (this._box) { + DOM.remove(this._box); + this._box = null; + } + + DOM.enableDrag(); + + delete this._startPos; + delete this._lastPos; + } + + _fireEvent(type: string, e: any) { + return this._map.fire(new Event(type, {originalEvent: e})); + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/click_zoom.ts b/web/libraries/maplibre-gl/src/ui/handler/click_zoom.ts new file mode 100644 index 00000000..323ff3db --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/click_zoom.ts @@ -0,0 +1,55 @@ +import type Point from '@mapbox/point-geometry'; +import type {Map} from '../map'; +import {TransformProvider} from './transform-provider'; +import {Handler} from '../handler_manager'; + +/** + * The `ClickZoomHandler` allows the user to zoom the map at a point by double clicking + * It is used by other handlers + */ +export class ClickZoomHandler implements Handler { + + _tr: TransformProvider; + _enabled: boolean; + _active: boolean; + + /** @internal */ + constructor(map: Map) { + this._tr = new TransformProvider(map); + this.reset(); + } + + reset() { + this._active = false; + } + + dblclick(e: MouseEvent, point: Point) { + e.preventDefault(); + return { + cameraAnimation: (map: Map) => { + map.easeTo({ + duration: 300, + zoom: this._tr.zoom + (e.shiftKey ? -1 : 1), + around: this._tr.unproject(point) + }, {originalEvent: e}); + } + }; + } + + enable() { + this._enabled = true; + } + + disable() { + this._enabled = false; + this.reset(); + } + + isEnabled() { + return this._enabled; + } + + isActive() { + return this._active; + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/cooperative_gestures.test.ts b/web/libraries/maplibre-gl/src/ui/handler/cooperative_gestures.test.ts new file mode 100644 index 00000000..dfac990a --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/cooperative_gestures.test.ts @@ -0,0 +1,255 @@ +import {browser} from '../../util/browser'; +import {Map} from '../map'; +import {DOM} from '../../util/dom'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {beforeMapTest} from '../../util/test/util'; + +function createMap(cooperativeGestures) { + return new Map({ + container: DOM.create('div', '', window.document.body), + style: { + 'version': 8, + 'sources': {}, + 'layers': [] + }, + cooperativeGestures + }); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('CoopGesturesHandler', () => { + + test('Does not zoom on wheel if no key is down', () => { + const browserNow = jest.spyOn(browser, 'now'); + let now = 1555555555555; + browserNow.mockReturnValue(now); + + const map = createMap(true); + map._renderTaskQueue.run(); + + const startZoom = map.getZoom(); + // simulate a single 'wheel' event + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta}); + map._renderTaskQueue.run(); + + now += 400; + browserNow.mockReturnValue(now); + map._renderTaskQueue.run(); + + const endZoom = map.getZoom(); + expect(endZoom).toBeCloseTo(startZoom); + + map.remove(); + }); + + test('Zooms on wheel if no key is down after disabling cooperative gestures', () => { + const browserNow = jest.spyOn(browser, 'now'); + let now = 1555555555555; + browserNow.mockReturnValue(now); + + const map = createMap(true); + map.setCooperativeGestures(false); + map._renderTaskQueue.run(); + + const startZoom = map.getZoom(); + // simulate a single 'wheel' event + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta}); + map._renderTaskQueue.run(); + + now += 400; + browserNow.mockReturnValue(now); + map._renderTaskQueue.run(); + + const endZoom = map.getZoom(); + expect(endZoom - startZoom).toBeCloseTo(0.0285, 3); + + map.remove(); + }); + + test('Zooms on wheel if control key is down', () => { + // NOTE: This should pass regardless of whether cooperativeGestures is enabled or not + const browserNow = jest.spyOn(browser, 'now'); + let now = 1555555555555; + browserNow.mockReturnValue(now); + + const map = createMap(true); + map._renderTaskQueue.run(); + + const startZoom = map.getZoom(); + // simulate a single 'wheel' event + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta, ctrlKey: true}); + map._renderTaskQueue.run(); + + now += 400; + browserNow.mockReturnValue(now); + map._renderTaskQueue.run(); + + const endZoom = map.getZoom(); + expect(endZoom - startZoom).toBeCloseTo(0.0285, 3); + + map.remove(); + }); + + test('Does not pan on touchmove with a single touch', () => { + const map = createMap(true); + const target = map.getCanvas(); + const startCenter = map.getCenter(); + map._renderTaskQueue.run(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.touchstart(target, {touches: [{target, clientX: 0, clientY: 0}]}); + map._renderTaskQueue.run(); + + simulate.touchmove(target, {touches: [{target, clientX: 10, clientY: 10}]}); + map._renderTaskQueue.run(); + + simulate.touchend(target); + map._renderTaskQueue.run(); + + const endCenter = map.getCenter(); + expect(endCenter.lng).toEqual(startCenter.lng); + expect(endCenter.lat).toEqual(startCenter.lat); + + map.remove(); + }); + + test('Pans on touchmove with a single touch after disabling cooperative gestures', () => { + const map = createMap(true); + map.setCooperativeGestures(false); + const target = map.getCanvas(); + const startCenter = map.getCenter(); + map._renderTaskQueue.run(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.touchstart(target, {touches: [{target, clientX: 0, clientY: 0}, {target, clientX: 1, clientY: 1}]}); + map._renderTaskQueue.run(); + + simulate.touchmove(target, {touches: [{target, clientX: 10, clientY: 10}, {target, clientX: 11, clientY: 11}]}); + map._renderTaskQueue.run(); + + simulate.touchend(target); + map._renderTaskQueue.run(); + + const endCenter = map.getCenter(); + expect(endCenter.lng).toBeGreaterThan(startCenter.lng); + expect(endCenter.lat).toBeGreaterThan(startCenter.lat); + + map.remove(); + }); + + test('Does pan on touchmove with a double touch', () => { + // NOTE: This should pass regardless of whether cooperativeGestures is enabled or not + const map = createMap(true); + const target = map.getCanvas(); + const startCenter = map.getCenter(); + map._renderTaskQueue.run(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.touchstart(target, {touches: [{target, clientX: 0, clientY: 0}, {target, clientX: 1, clientY: 1}]}); + map._renderTaskQueue.run(); + + simulate.touchmove(target, {touches: [{target, clientX: 10, clientY: 10}, {target, clientX: 11, clientY: 11}]}); + map._renderTaskQueue.run(); + + simulate.touchend(target); + map._renderTaskQueue.run(); + + const endCenter = map.getCenter(); + expect(endCenter.lng).toBeGreaterThan(startCenter.lng); + expect(endCenter.lat).toBeGreaterThan(startCenter.lat); + + map.remove(); + }); + + test('Drag pitch works with 3 fingers', () => { + // NOTE: This should pass regardless of whether cooperativeGestures is enabled or not + const map = createMap(true); + const target = map.getCanvas(); + const startPitch = map.getPitch(); + map._renderTaskQueue.run(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.touchstart(target, {touches: [{target, clientX: 0, clientY: 0}, {target, clientX: 1, clientY: 1}, {target, clientX: 2, clientY: 2}]}); + map._renderTaskQueue.run(); + + simulate.touchmove(target, {touches: [{target, clientX: 0, clientY: -10}, {target, clientX: 1, clientY: -11}, {target, clientX: 2, clientY: -12}]}); + map._renderTaskQueue.run(); + + simulate.touchend(target); + map._renderTaskQueue.run(); + + const endPitch = map.getPitch(); + expect(endPitch).toBeGreaterThan(startPitch); + + map.remove(); + }); + + test('Initially disabled cooperative gestures can be later enabled', () => { + const browserNow = jest.spyOn(browser, 'now'); + let now = 1555555555555; + browserNow.mockReturnValue(now); + + const map = createMap(false); + map._renderTaskQueue.run(); + + const startZoom = map.getZoom(); + // simulate a single 'wheel' event + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta}); + map._renderTaskQueue.run(); + + now += 400; + browserNow.mockReturnValue(now); + map._renderTaskQueue.run(); + + const midZoom = map.getZoom(); + expect(midZoom - startZoom).toBeCloseTo(0.0285, 3); + + // Enable cooperative gestures + map.setCooperativeGestures(true); + + // This 'wheel' event should not zoom + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta}); + map._renderTaskQueue.run(); + + now += 400; + browserNow.mockReturnValue(now); + map._renderTaskQueue.run(); + + const endZoom = map.getZoom(); + expect(endZoom).toBeCloseTo(midZoom); + + map.remove(); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/dblclick_zoom.test.ts b/web/libraries/maplibre-gl/src/ui/handler/dblclick_zoom.test.ts new file mode 100644 index 00000000..33f762a7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/dblclick_zoom.test.ts @@ -0,0 +1,179 @@ +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {beforeMapTest} from '../../util/test/util'; +import {Map, MapOptions} from '../map'; + +function createMap() { + return new Map({container: window.document.createElement('div')} as any as MapOptions); +} + +function simulateDoubleTap(map, delay = 100) { + const canvas = map.getCanvas(); + return new Promise(resolve => { + simulate.touchstart(canvas, {touches: [{target: canvas, clientX: 0, clientY: 0}]}); + simulate.touchend(canvas); + setTimeout(() => { + simulate.touchstart(canvas, {touches: [{target: canvas, clientX: 0, clientY: 0}]}); + simulate.touchend(canvas); + map._renderTaskQueue.run(); + resolve(undefined); + }, delay); + }); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('dbclick_zoom', () => { + test('DoubleClickZoomHandler zooms on dblclick event', () => { + const map = createMap(); + + const zoom = jest.fn(); + map.on('zoomstart', zoom); + + simulate.dblclick(map.getCanvas()); + map._renderTaskQueue.run(); + + expect(zoom).toHaveBeenCalled(); + + map.remove(); + }); + + test('DoubleClickZoomHandler does not zoom if preventDefault is called on the dblclick event', () => { + const map = createMap(); + + map.on('dblclick', e => e.preventDefault()); + + const zoom = jest.fn(); + map.on('zoomstart', zoom); + + simulate.dblclick(map.getCanvas()); + map._renderTaskQueue.run(); + + expect(zoom).not.toHaveBeenCalled(); + + map.remove(); + }); + + test('DoubleClickZoomHandler zooms on double tap if touchstart events are < 300ms apart', done => { + const map = createMap(); + + const zoom = jest.fn(); + map.on('zoomstart', zoom); + + simulateDoubleTap(map, 100).then(() => { + expect(zoom).toHaveBeenCalled(); + + map.remove(); + done(); + }); + + }); + + test('DoubleClickZoomHandler does not zoom on double tap if touchstart events are > 500ms apart', done => { + const map = createMap(); + + const zoom = jest.fn(); + map.on('zoom', zoom); + + simulateDoubleTap(map, 500).then(() => { + expect(zoom).not.toHaveBeenCalled(); + + map.remove(); + done(); + }); + + }); + + test('DoubleClickZoomHandler does not zoom on double tap if touchstart events are in different locations', done => { + const map = createMap(); + + const zoom = jest.fn(); + map.on('zoom', zoom); + + const canvas = map.getCanvas(); + + const simulateTwoDifferentTaps = () => { + return new Promise(resolve => { + simulate.touchstart(canvas, {touches: [{clientX: 0, clientY: 0}]}); + simulate.touchend(canvas); + setTimeout(() => { + simulate.touchstart(canvas, {touches: [{clientX: 30.5, clientY: 30.5}]}); + simulate.touchend(canvas); + map._renderTaskQueue.run(); + resolve(undefined); + }, 100); + }); + }; + + simulateTwoDifferentTaps().then(() => { + expect(zoom).not.toHaveBeenCalled(); + + map.remove(); + done(); + }); + + }); + + test('DoubleClickZoomHandler zooms on the second touchend event of a double tap', () => { + const map = createMap(); + + const zoom = jest.fn(); + map.on('zoomstart', zoom); + + const canvas = map.getCanvas(); + const touchOptions = {touches: [{target: canvas, clientX: 0.5, clientY: 0.5}]}; + + simulate.touchstart(canvas, touchOptions); + simulate.touchend(canvas); + simulate.touchstart(canvas, touchOptions); + map._renderTaskQueue.run(); + map._renderTaskQueue.run(); + expect(zoom).not.toHaveBeenCalled(); + + simulate.touchcancel(canvas); + simulate.touchend(canvas); + map._renderTaskQueue.run(); + expect(zoom).not.toHaveBeenCalled(); + + simulate.touchstart(canvas, touchOptions); + simulate.touchend(canvas); + simulate.touchstart(canvas, touchOptions); + map._renderTaskQueue.run(); + expect(zoom).not.toHaveBeenCalled(); + + simulate.touchend(canvas); + map._renderTaskQueue.run(); + + expect(zoom).toHaveBeenCalled(); + + }); + + test('DoubleClickZoomHandler does not zoom on double tap if second touchend is >300ms after first touchstart', done => { + const map = createMap(); + + const zoom = jest.fn(); + map.on('zoom', zoom); + + const canvas = map.getCanvas(); + + const simulateSlowSecondTap = () => { + return new Promise(resolve => { + simulate.touchstart(canvas); + simulate.touchend(canvas); + simulate.touchstart(canvas); + setTimeout(() => { + simulate.touchend(canvas); + map._renderTaskQueue.run(); + resolve(undefined); + }, 300); + }); + }; + + simulateSlowSecondTap().then(() => { + expect(zoom).not.toHaveBeenCalled(); + + done(); + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/drag_handler.ts b/web/libraries/maplibre-gl/src/ui/handler/drag_handler.ts new file mode 100644 index 00000000..7f08be11 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/drag_handler.ts @@ -0,0 +1,174 @@ +import {DOM} from '../../util/dom'; +import type Point from '@mapbox/point-geometry'; +import {DragMoveStateManager} from './drag_move_state_manager'; +import {Handler} from '../handler_manager'; + +interface DragMovementResult { + bearingDelta?: number; + pitchDelta?: number; + around?: Point; + panDelta?: Point; +} + +export interface DragPanResult extends DragMovementResult { + around: Point; + panDelta: Point; +} + +export interface DragRotateResult extends DragMovementResult { + bearingDelta: number; +} + +export interface DragPitchResult extends DragMovementResult { + pitchDelta: number; +} + +type DragMoveFunction = (lastPoint: Point, point: Point) => T; + +export interface DragMoveHandler extends Handler { + dragStart: (e: E, point: Point) => void; + dragMove: (e: E, point: Point) => T | void; + dragEnd: (e: E) => void; + getClickTolerance: () => number; +} + +export type DragMoveHandlerOptions = { + /** + * If the movement is shorter than this value, consider it a click. + */ + clickTolerance: number; + /** + * The move function to run on a valid movement. + */ + move: DragMoveFunction; + /** + * A class used to manage the state of the drag event - start, checking valid moves, end. See the class documentation for more details. + */ + moveStateManager: DragMoveStateManager; + /** + * A method used to assign the dragStart, dragMove, and dragEnd methods to the relevant event handlers, as well as assigning the contextmenu handler + * @param handler - the handler + */ + assignEvents: (handler: DragMoveHandler) => void; + /** + * Should the move start on the "start" event, or should it start on the first valid move. + */ + activateOnStart?: boolean; + /** + * If true, handler will be enabled during construction + */ + enable?: boolean; +} + +/** + * A generic class to create handlers for drag events, from both mouse and touch events. + */ +export class DragHandler implements DragMoveHandler { + // Event handlers that may be assigned by the implementations of this class + contextmenu?: Handler['contextmenu']; + mousedown?: Handler['mousedown']; + mousemoveWindow?: Handler['mousemoveWindow']; + mouseup?: Handler['mouseup']; + touchstart?: Handler['touchstart']; + touchmoveWindow?: Handler['touchmoveWindow']; + touchend?: Handler['touchend']; + + _clickTolerance: number; + _moveFunction: DragMoveFunction; + _activateOnStart: boolean; + _active: boolean; + _enabled: boolean; + _moved: boolean; + _lastPoint: Point | null; + _moveStateManager: DragMoveStateManager; + + constructor(options: DragMoveHandlerOptions) { + this._enabled = !!options.enable; + this._moveStateManager = options.moveStateManager; + this._clickTolerance = options.clickTolerance || 1; + this._moveFunction = options.move; + this._activateOnStart = !!options.activateOnStart; + + options.assignEvents(this); + + this.reset(); + } + + reset(e?: E) { + this._active = false; + this._moved = false; + delete this._lastPoint; + this._moveStateManager.endMove(e); + } + + _move(...params: Parameters>) { + const move = this._moveFunction(...params); + if (move.bearingDelta || move.pitchDelta || move.around || move.panDelta) { + this._active = true; + return move; + } + } + + dragStart(e: E, point: Point); + dragStart(e: E, point: Point[]); + dragStart(e: E, point: Point | Point[]) { + if (!this.isEnabled() || this._lastPoint) return; + + if (!this._moveStateManager.isValidStartEvent(e)) return; + this._moveStateManager.startMove(e); + + this._lastPoint = point['length'] ? point[0] : point; + + if (this._activateOnStart && this._lastPoint) this._active = true; + } + + dragMove(e: E, point: Point); + dragMove(e: E, point: Point[]); + dragMove(e: E, point: Point | Point[]) { + if (!this.isEnabled()) return; + const lastPoint = this._lastPoint; + if (!lastPoint) return; + e.preventDefault(); + + if (!this._moveStateManager.isValidMoveEvent(e)) { + this.reset(e); + return; + } + + const movePoint = point['length'] ? point[0] : point; + + if (!this._moved && movePoint.dist(lastPoint) < this._clickTolerance) return; + this._moved = true; + this._lastPoint = movePoint; + + return this._move(lastPoint, movePoint); + } + + dragEnd(e: E) { + if (!this.isEnabled() || !this._lastPoint) return; + if (!this._moveStateManager.isValidEndEvent(e)) return; + if (this._moved) DOM.suppressClick(); + this.reset(e); + } + + enable() { + this._enabled = true; + } + + disable() { + this._enabled = false; + this.reset(); + } + + isEnabled() { + return this._enabled; + } + + isActive() { + return this._active; + } + + getClickTolerance() { + return this._clickTolerance; + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/drag_move_state_manager.ts b/web/libraries/maplibre-gl/src/ui/handler/drag_move_state_manager.ts new file mode 100644 index 00000000..d973186d --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/drag_move_state_manager.ts @@ -0,0 +1,115 @@ +import {DOM} from '../../util/dom'; + +const LEFT_BUTTON = 0; +const RIGHT_BUTTON = 2; + +// the values for each button in MouseEvent.buttons +const BUTTONS_FLAGS = { + [LEFT_BUTTON]: 1, + [RIGHT_BUTTON]: 2 +}; + +function buttonNoLongerPressed(e: MouseEvent, button: number) { + const flag = BUTTONS_FLAGS[button]; + return e.buttons === undefined || (e.buttons & flag) !== flag; +} + +/* + * Drag events are initiated by specific interaction which needs to be tracked until it ends. + * This requires some state management: + * 1. registering the initiating event, + * 2. tracking that it was not canceled / not confusing it with another event firing. + * 3. recognizing the ending event and cleaning up any internal state + * + * Concretely, we implement two state managers: + * 1. MouseMoveStateManager + * Receives a functions that is used to recognize mouse events that should be registered as the + * relevant drag interactions - i.e. dragging with the right mouse button, or while CTRL is pressed. + * 2. OneFingerTouchMoveStateManager + * Checks if a drag event is using one finger, and continuously tracking that this is the same event + * (i.e. to make sure not additional finger has started interacting with the screen before raising + * the first finger). + */ +export interface DragMoveStateManager { + startMove: (e: E) => void; + endMove: (e?: E) => void; + isValidStartEvent: (e: E) => boolean; + isValidMoveEvent: (e: E) => boolean; + isValidEndEvent: (e?: E) => boolean; +} + +export class MouseMoveStateManager implements DragMoveStateManager { + _eventButton: number | undefined; + _correctEvent: (e: MouseEvent) => boolean; + + constructor(options: { + checkCorrectEvent: (e: MouseEvent) => boolean; + }) { + this._correctEvent = options.checkCorrectEvent; + } + + startMove(e: MouseEvent) { + const eventButton = DOM.mouseButton(e); + this._eventButton = eventButton; + } + + endMove(_e?: MouseEvent) { + delete this._eventButton; + } + + isValidStartEvent(e: MouseEvent) { + return this._correctEvent(e); + } + + isValidMoveEvent(e: MouseEvent) { + // Some browsers don't fire a `mouseup` when the mouseup occurs outside + // the window or iframe: + // https://github.com/mapbox/mapbox-gl-js/issues/4622 + // + // If the button is no longer pressed during this `mousemove` it may have + // been released outside of the window or iframe. + return !buttonNoLongerPressed(e, this._eventButton); + } + + isValidEndEvent(e: MouseEvent) { + const eventButton = DOM.mouseButton(e); + return eventButton === this._eventButton; + } +} + +export class OneFingerTouchMoveStateManager implements DragMoveStateManager { + _firstTouch: number | undefined; + + constructor() { + this._firstTouch = undefined; + } + + _isOneFingerTouch(e: TouchEvent) { + return e.targetTouches.length === 1; + } + + _isSameTouchEvent(e: TouchEvent) { + return e.targetTouches[0].identifier === this._firstTouch; + } + + startMove(e: TouchEvent) { + const firstTouch = e.targetTouches[0].identifier; + this._firstTouch = firstTouch; + } + + endMove(_e?: TouchEvent) { + delete this._firstTouch; + } + + isValidStartEvent(e: TouchEvent) { + return this._isOneFingerTouch(e); + } + + isValidMoveEvent(e: TouchEvent) { + return this._isOneFingerTouch(e) && this._isSameTouchEvent(e); + } + + isValidEndEvent(e: TouchEvent) { + return this._isOneFingerTouch(e) && this._isSameTouchEvent(e); + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/drag_pan.test.ts b/web/libraries/maplibre-gl/src/ui/handler/drag_pan.test.ts new file mode 100644 index 00000000..dee3c37e --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/drag_pan.test.ts @@ -0,0 +1,487 @@ +import {DOM} from '../../util/dom'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {beforeMapTest} from '../../util/test/util'; +import {Map, MapOptions} from '../map'; +import type {MapGeoJSONFeature} from '../../util/vectortile_to_geojson'; + +function createMap(clickTolerance?, dragPan?) { + return new Map({ + container: DOM.create('div', '', window.document.body), + clickTolerance: clickTolerance || 0, + dragPan: dragPan || true, + } as any as MapOptions); +} + +beforeEach(() => { + beforeMapTest(); +}); + +// MouseEvent.buttons = 1 // left button +const buttons = 1; + +describe('drag_pan', () => { + + test('DragPanHandler fires dragstart, drag, and dragend events at appropriate times in response to a mouse-triggered drag', () => { + const map = createMap(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.mousedown(map.getCanvas()); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas()); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragPanHandler captures mousemove events during a mouse-triggered drag (receives them even if they occur outside the map)', () => { + const map = createMap(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.mousedown(map.getCanvas()); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mousemove(window.document.body, {buttons, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas()); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragPanHandler fires dragstart, drag, and dragend events at appropriate times in response to a touch-triggered drag', () => { + const map = createMap(); + const target = map.getCanvas(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.touchstart(map.getCanvas(), {touches: [{target, clientX: 0, clientY: 0}]}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: [{target, clientX: 10, clientY: 10}]}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.touchend(map.getCanvas()); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragPanHandler prevents mousemove events from firing during a drag (#1555)', () => { + const map = createMap(); + + const mousemove = jest.fn(); + map.on('mousemove', mousemove); + + simulate.mousedown(map.getCanvasContainer()); + map._renderTaskQueue.run(); + + simulate.mousemove(map.getCanvasContainer(), {buttons, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + + simulate.mouseup(map.getCanvasContainer()); + map._renderTaskQueue.run(); + + expect(mousemove).not.toHaveBeenCalled(); + + map.remove(); + }); + + test('DragPanHandler ends a mouse-triggered drag if the window blurs', () => { + const map = createMap(); + + const dragend = jest.fn(); + map.on('dragend', dragend); + + simulate.mousedown(map.getCanvas()); + map._renderTaskQueue.run(); + + simulate.mousemove(map.getCanvas(), {buttons, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + + simulate.blur(window); + expect(dragend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragPanHandler ends a touch-triggered drag if the window blurs', () => { + const map = createMap(); + const target = map.getCanvas(); + + const dragend = jest.fn(); + map.on('dragend', dragend); + + simulate.touchstart(map.getCanvas(), {touches: [{target, clientX: 0, clientY: 0}]}); + map._renderTaskQueue.run(); + + simulate.touchmove(map.getCanvas(), {touches: [{target, clientX: 10, clientY: 10}]}); + map._renderTaskQueue.run(); + + simulate.blur(window); + expect(dragend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragPanHandler requests a new render frame after each mousemove event', () => { + const map = createMap(); + const requestFrame = jest.spyOn(map.handlers, '_requestFrame'); + + simulate.mousedown(map.getCanvas()); + simulate.mousemove(map.getCanvas(), {buttons, clientX: 10, clientY: 10}); + expect(requestFrame).toHaveBeenCalled(); + + map._renderTaskQueue.run(); + + // https://github.com/mapbox/mapbox-gl-js/issues/6063 + requestFrame.mockReset(); + simulate.mousemove(map.getCanvas(), {buttons, clientX: 20, clientY: 20}); + expect(requestFrame).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragPanHandler can interleave with another handler', () => { + // https://github.com/mapbox/mapbox-gl-js/issues/6106 + const map = createMap(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.mousedown(map.getCanvas()); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).toHaveBeenCalledTimes(0); + + // simulate a scroll zoom + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons, clientX: 20, clientY: 20}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(2); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas()); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(2); + expect(dragend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + ['ctrl', 'shift'].forEach((modifier) => { + test(`DragPanHandler does not begin a drag if the ${modifier} key is down on mousedown`, () => { + const map = createMap(); + expect(map.dragRotate.isEnabled()).toBeTruthy(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.mousedown(map.getCanvas(), {buttons, [`${modifier}Key`]: true}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons, [`${modifier}Key`]: true, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {[`${modifier}Key`]: true}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + map.remove(); + }); + + test(`DragPanHandler still ends a drag if the ${modifier} key is down on mouseup`, () => { + const map = createMap(); + expect(map.dragRotate.isEnabled()).toBeTruthy(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.mousedown(map.getCanvas()); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {[`${modifier}Key`]: true}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + map.remove(); + }); + }); + + test('DragPanHandler does not begin a drag on right button mousedown', () => { + const map = createMap(); + map.dragRotate.disable(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + map.remove(); + }); + + test('DragPanHandler does not end a drag on right button mouseup', () => { + const map = createMap(); + map.dragRotate.disable(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.mousedown(map.getCanvas()); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mousedown(map.getCanvas(), {buttons: buttons + 2, button: 2}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {buttons, button: 2}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons, clientX: 20, clientY: 20}); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(2); + expect(dragend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas()); + map._renderTaskQueue.run(); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(2); + expect(dragend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragPanHandler does not begin a drag if preventDefault is called on the mousedown event', () => { + const map = createMap(); + + map.on('mousedown', e => e.preventDefault()); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.mousedown(map.getCanvas()); + map._renderTaskQueue.run(); + + simulate.mousemove(map.getCanvas(), {buttons, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + + simulate.mouseup(map.getCanvas()); + map._renderTaskQueue.run(); + + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + map.remove(); + }); + + test('DragPanHandler does not begin a drag if preventDefault is called on the touchstart event', () => { + const map = createMap(); + const target = map.getCanvas(); + + map.on('touchstart', e => e.preventDefault()); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.touchstart(map.getCanvas(), {touches: [{target, clientX: 0, clientY: 0}]}); + map._renderTaskQueue.run(); + + simulate.touchmove(map.getCanvas(), {touches: [{target, clientX: 10, clientY: 10}]}); + map._renderTaskQueue.run(); + + simulate.touchend(map.getCanvas()); + map._renderTaskQueue.run(); + + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + map.remove(); + }); + + test('DragPanHandler does not begin a drag if preventDefault is called on the touchstart event (delegated)', () => { + const map = createMap(); + const target = map.getCanvas(); + + jest.spyOn(map, 'getLayer').mockReturnValue(true as any); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + map.on('touchstart', 'point', (e) => { + e.preventDefault(); + }); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + map.on('dragstart', dragstart); + map.on('drag', drag); + map.on('dragend', dragend); + + simulate.touchstart(map.getCanvas(), {touches: [{target, clientX: 0, clientY: 0}]}); + map._renderTaskQueue.run(); + + simulate.touchmove(map.getCanvas(), {touches: [{target, clientX: 10, clientY: 10}]}); + map._renderTaskQueue.run(); + + simulate.touchend(map.getCanvas()); + map._renderTaskQueue.run(); + + expect(dragstart).toHaveBeenCalledTimes(0); + expect(drag).toHaveBeenCalledTimes(0); + expect(dragend).toHaveBeenCalledTimes(0); + + map.remove(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/drag_rotate.test.ts b/web/libraries/maplibre-gl/src/ui/handler/drag_rotate.test.ts new file mode 100644 index 00000000..6155d249 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/drag_rotate.test.ts @@ -0,0 +1,859 @@ +import {extend} from '../../util/util'; +import {Map} from '../map'; +import {DOM} from '../../util/dom'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {browser} from '../../util/browser'; + +import {beforeMapTest} from '../../util/test/util'; + +function createMap(options?) { + return new Map(extend({container: DOM.create('div', '', window.document.body)}, options)); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('drag rotate', () => { + + test('DragRotateHandler#isActive', () => { + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + expect(map.dragRotate.isActive()).toBe(false); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + map._renderTaskQueue.run(); + expect(map.dragRotate.isActive()).toBe(false); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(map.dragRotate.isActive()).toBe(true); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + expect(map.dragRotate.isActive()).toBe(false); + + map.remove(); + }); + + test('DragRotateHandler fires rotatestart, rotate, and rotateend events at appropriate times in response to a right-click drag', () => { + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragRotateHandler stops firing events after mouseup', () => { + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const spy = jest.fn(); + map.on('rotatestart', spy); + map.on('rotate', spy); + map.on('rotateend', spy); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + expect(spy).toHaveBeenCalledTimes(3); + + spy.mockReset(); + simulate.mousemove(map.getCanvas(), {buttons: 0, clientX: 20, clientY: 20}); + map._renderTaskQueue.run(); + expect(spy).toHaveBeenCalledTimes(0); + + map.remove(); + }); + + test('DragRotateHandler fires rotatestart, rotate, and rotateend events at appropriate times in response to a control-left-click drag', () => { + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvas(), {buttons: 1, button: 0, ctrlKey: true}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 1, ctrlKey: true, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 0, ctrlKey: true}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragRotateHandler pitches in response to a right-click drag by default', () => { + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const pitchstart = jest.fn(); + const pitch = jest.fn(); + const pitchend = jest.fn(); + + map.on('pitchstart', pitchstart); + map.on('pitch', pitch); + map.on('pitchend', pitchend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: -10}); + map._renderTaskQueue.run(); + expect(pitchstart).toHaveBeenCalledTimes(1); + expect(pitch).toHaveBeenCalledTimes(1); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + expect(pitchend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragRotateHandler doesn\'t fire pitch event when rotating only', () => { + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const pitchstart = jest.fn(); + const pitch = jest.fn(); + const pitchend = jest.fn(); + + map.on('pitchstart', pitchstart); + map.on('pitch', pitch); + map.on('pitchend', pitchend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2, clientX: 0, clientY: 10}); + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(pitchstart).toHaveBeenCalledTimes(0); + expect(pitch).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + expect(pitchend).toHaveBeenCalledTimes(0); + + map.remove(); + }); + + test('DragRotateHandler pitches in response to a control-left-click drag', () => { + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const pitchstart = jest.fn(); + const pitch = jest.fn(); + const pitchend = jest.fn(); + + map.on('pitchstart', pitchstart); + map.on('pitch', pitch); + map.on('pitchend', pitchend); + + simulate.mousedown(map.getCanvas(), {buttons: 1, button: 0, ctrlKey: true}); + simulate.mousemove(map.getCanvas(), {buttons: 1, ctrlKey: true, clientX: 10, clientY: -10}); + map._renderTaskQueue.run(); + expect(pitchstart).toHaveBeenCalledTimes(1); + expect(pitch).toHaveBeenCalledTimes(1); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 0, ctrlKey: true}); + map._renderTaskQueue.run(); + expect(pitchend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragRotateHandler does not pitch if given pitchWithRotate: false', () => { + const map = createMap({pitchWithRotate: false}); + + const spy = jest.fn(); + + map.on('pitchstart', spy); + map.on('pitch', spy); + map.on('pitchend', spy); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + + simulate.mousedown(map.getCanvas(), {buttons: 1, button: 0, ctrlKey: true}); + simulate.mousemove(map.getCanvas(), {buttons: 1, ctrlKey: true, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 0, ctrlKey: true}); + + expect(spy).not.toHaveBeenCalled(); + + map.remove(); + }); + + test('DragRotateHandler does not rotate or pitch when disabled', () => { + const map = createMap(); + + map.dragRotate.disable(); + + const spy = jest.fn(); + + map.on('rotatestart', spy); + map.on('rotate', spy); + map.on('rotateend', spy); + map.on('pitchstart', spy); + map.on('pitch', spy); + map.on('pitchend', spy); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + + expect(spy).not.toHaveBeenCalled(); + + map.remove(); + }); + + test('DragRotateHandler ensures that map.isMoving() returns true during drag', () => { + // The bearingSnap option here ensures that the moveend event is sent synchronously. + const map = createMap({bearingSnap: 0}); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(map.isMoving()).toBeTruthy(); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + expect(!map.isMoving()).toBeTruthy(); + + map.remove(); + }); + + test('DragRotateHandler fires move events', () => { + // The bearingSnap option here ensures that the moveend event is sent synchronously. + const map = createMap({bearingSnap: 0}); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const movestart = jest.fn(); + const move = jest.fn(); + const moveend = jest.fn(); + + map.on('movestart', movestart); + map.on('move', move); + map.on('moveend', moveend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(movestart).toHaveBeenCalledTimes(1); + expect(move).toHaveBeenCalledTimes(1); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + expect(moveend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragRotateHandler doesn\'t fire rotate event when pitching only', () => { + // The bearingSnap option here ensures that the moveend event is sent synchronously. + const map = createMap({bearingSnap: 0}); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const pitch = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('pitch', pitch); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2, clientX: 0, clientY: 0}); + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 0, clientY: -10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(pitch).toHaveBeenCalledTimes(1); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + expect(rotateend).toHaveBeenCalledTimes(0); + + map.remove(); + }); + + test('DragRotateHandler includes originalEvent property in triggered events', () => { + // The bearingSnap option here ensures that the moveend event is sent synchronously. + const map = createMap({bearingSnap: 0}); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + const pitchstart = jest.fn(); + const pitch = jest.fn(); + const pitchend = jest.fn(); + map.on('pitchstart', pitchstart); + map.on('pitch', pitch); + map.on('pitchend', pitchend); + + const movestart = jest.fn(); + const move = jest.fn(); + const moveend = jest.fn(); + map.on('movestart', movestart); + map.on('move', move); + map.on('moveend', moveend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: -10}); + map._renderTaskQueue.run(); + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + + expect(rotatestart.mock.calls[0][0].originalEvent.type).toBeTruthy(); + expect(pitchstart.mock.calls[0][0].originalEvent.type).toBeTruthy(); + expect(movestart.mock.calls[0][0].originalEvent.type).toBeTruthy(); + + expect(rotate.mock.calls[0][0].originalEvent.type).toBeTruthy(); + expect(pitch.mock.calls[0][0].originalEvent.type).toBeTruthy(); + expect(move.mock.calls[0][0].originalEvent.type).toBeTruthy(); + + expect(rotateend.mock.calls[0][0].originalEvent.type).toBeTruthy(); + expect(pitchend.mock.calls[0][0].originalEvent.type).toBeTruthy(); + expect(moveend.mock.calls[0][0].originalEvent.type).toBeTruthy(); + + map.remove(); + }); + + test('DragRotateHandler responds to events on the canvas container (#1301)', () => { + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvasContainer(), {buttons: 2, button: 2}); + simulate.mousemove(map.getCanvasContainer(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + + simulate.mouseup(map.getCanvasContainer(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + expect(rotateend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragRotateHandler prevents mousemove events from firing during a drag (#1555)', () => { + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const mousemove = jest.fn(); + map.on('mousemove', mousemove); + + simulate.mousedown(map.getCanvasContainer(), {buttons: 2, button: 2}); + simulate.mousemove(map.getCanvasContainer(), {buttons: 2, clientX: 100, clientY: 100}); + map._renderTaskQueue.run(); + simulate.mouseup(map.getCanvasContainer(), {buttons: 0, button: 2}); + + expect(mousemove).not.toHaveBeenCalled(); + + map.remove(); + }); + + test('DragRotateHandler ends a control-left-click drag on mouseup even when the control key was previously released (#1888)', () => { + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvas(), {buttons: 1, button: 0, ctrlKey: true}); + simulate.mousemove(map.getCanvas(), {buttons: 1, ctrlKey: true, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 0, ctrlKey: false}); + map._renderTaskQueue.run(); + expect(rotateend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragRotateHandler ends rotation if the window blurs (#3389)', () => { + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + + simulate.blur(window); + expect(rotateend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragRotateHandler requests a new render frame after each mousemove event', () => { + const map = createMap(); + const requestRenderFrame = jest.spyOn(map.handlers, '_requestFrame'); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + expect(requestRenderFrame).toHaveBeenCalled(); + + map._renderTaskQueue.run(); + + // https://github.com/mapbox/mapbox-gl-js/issues/6063 + requestRenderFrame.mockReset(); + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 20, clientY: 20}); + expect(requestRenderFrame).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragRotateHandler can interleave with another handler', () => { + // https://github.com/mapbox/mapbox-gl-js/issues/6106 + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(0); + + // simulates another handler taking over + // simulate a scroll zoom + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 20, clientY: 20}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(2); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + // Ignore second rotatestart triggered by inertia + expect(rotate).toHaveBeenCalledTimes(2); + expect(rotateend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragRotateHandler does not begin a drag on left-button mousedown without the control key', () => { + const map = createMap(); + map.dragPan.disable(); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvas()); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas()); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + map.remove(); + }); + + test('DragRotateHandler does not end a right-button drag on left-button mouseup', () => { + const map = createMap(); + map.dragPan.disable(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mousedown(map.getCanvas(), {buttons: 3, button: 0}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {buttons: 2, button: 0}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 20, clientY: 20}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(2); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + // Ignore second rotatestart triggered by inertia + expect(rotate).toHaveBeenCalledTimes(2); + expect(rotateend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragRotateHandler does not end a control-left-button drag on right-button mouseup', () => { + const map = createMap(); + map.dragPan.disable(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvas(), {buttons: 1, button: 0, ctrlKey: true}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 1, ctrlKey: true, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mousedown(map.getCanvas(), {buttons: 3, button: 2, ctrlKey: true}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {buttons: 1, button: 2, ctrlKey: true}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 1, ctrlKey: true, clientX: 20, clientY: 20}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(2); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 0, ctrlKey: true}); + map._renderTaskQueue.run(); + // Ignore second rotatestart triggered by inertia + expect(rotate).toHaveBeenCalledTimes(2); + expect(rotateend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('DragRotateHandler does not begin a drag if preventDefault is called on the mousedown event', () => { + const map = createMap(); + + map.on('mousedown', e => e.preventDefault()); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + map._renderTaskQueue.run(); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + map.remove(); + }); + + test('DragRotateHandler can be disabled after mousedown (#2419)', () => { + const map = createMap(); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + map._renderTaskQueue.run(); + + map.dragRotate.disable(); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + expect(map.isMoving()).toBe(false); + expect(map.dragRotate.isEnabled()).toBe(false); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + expect(map.isMoving()).toBe(false); + expect(map.dragRotate.isEnabled()).toBe(false); + + map.remove(); + }); + + test('DragRotateHandler does not begin rotation on spurious mousemove events', () => { + const map = createMap(); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + map.remove(); + }); + + test('DragRotateHandler does not begin a mouse drag if moved less than click tolerance', () => { + const map = createMap({clickTolerance: 4}); + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + const pitchstart = jest.fn(); + const pitch = jest.fn(); + const pitchend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + map.on('pitchstart', pitchstart); + map.on('pitch', pitch); + map.on('pitchend', pitchend); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + expect(pitchstart).toHaveBeenCalledTimes(0); + expect(pitch).toHaveBeenCalledTimes(0); + expect(pitchend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 13, clientY: 10}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + expect(pitchstart).toHaveBeenCalledTimes(0); + expect(pitch).toHaveBeenCalledTimes(0); + expect(pitchend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 13}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + expect(pitchstart).toHaveBeenCalledTimes(0); + expect(pitch).toHaveBeenCalledTimes(0); + expect(pitchend).toHaveBeenCalledTimes(0); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 14, clientY: 10 - 4}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(0); + expect(pitchstart).toHaveBeenCalledTimes(1); + expect(pitch).toHaveBeenCalledTimes(1); + expect(pitchend).toHaveBeenCalledTimes(0); + + map.remove(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/handler_util.ts b/web/libraries/maplibre-gl/src/ui/handler/handler_util.ts new file mode 100644 index 00000000..b097fad1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/handler_util.ts @@ -0,0 +1,10 @@ +import Point from '@mapbox/point-geometry'; + +export function indexTouches(touches: Array, points: Array) { + if (touches.length !== points.length) throw new Error(`The number of touches and points are not equal - touches ${touches.length}, points ${points.length}`); + const obj = {}; + for (let i = 0; i < touches.length; i++) { + obj[touches[i].identifier] = points[i]; + } + return obj; +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/keyboard.test.ts b/web/libraries/maplibre-gl/src/ui/handler/keyboard.test.ts new file mode 100644 index 00000000..98f271ff --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/keyboard.test.ts @@ -0,0 +1,235 @@ +import {Map} from '../../ui/map'; +import {DOM} from '../../util/dom'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {extend} from '../../util/util'; +import {beforeMapTest} from '../../util/test/util'; + +function createMap(options?) { + return new Map(extend({ + container: DOM.create('div', '', window.document.body), + }, options)); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('keyboard', () => { + test('KeyboardHandler responds to keydown events', () => { + const map = createMap(); + const h = map.keyboard; + const spy = jest.spyOn(h, 'keydown'); + + simulate.keydown(map.getCanvas(), {keyCode: 32, key: ' '}); + expect(h.keydown).toHaveBeenCalled(); + expect(spy.mock.calls[0][0].keyCode).toBe(32); + }); + + test('KeyboardHandler pans map in response to arrow keys', () => { + const map = createMap({zoom: 10, center: [0, 0]}); + const spy = jest.spyOn(map, 'easeTo'); + + simulate.keydown(map.getCanvas(), {keyCode: 32, key: ' '}); + expect(map.easeTo).not.toHaveBeenCalled(); + + simulate.keydown(map.getCanvas(), {keyCode: 37, key: 'ArrowLeft'}); + expect(map.easeTo).toHaveBeenCalled(); + let easeToArgs = spy.mock.calls[0][0]; + expect(easeToArgs.offset[0]).toBe(100); + expect(easeToArgs.offset[1]).toBe(-0); + + simulate.keydown(map.getCanvas(), {keyCode: 39, key: 'ArrowRight'}); + expect(spy).toHaveBeenCalledTimes(2); + easeToArgs = spy.mock.calls[1][0]; + expect(easeToArgs.offset[0]).toBe(-100); + expect(easeToArgs.offset[1]).toBe(-0); + + simulate.keydown(map.getCanvas(), {keyCode: 40, key: 'ArrowDown'}); + expect(spy).toHaveBeenCalledTimes(3); + easeToArgs = spy.mock.calls[2][0]; + expect(easeToArgs.offset[0]).toBe(-0); + expect(easeToArgs.offset[1]).toBe(-100); + + simulate.keydown(map.getCanvas(), {keyCode: 38, key: 'ArrowUp'}); + expect(spy).toHaveBeenCalledTimes(4); + easeToArgs = spy.mock.calls[3][0]; + expect(easeToArgs.offset[0]).toBe(-0); + expect(easeToArgs.offset[1]).toBe(100); + + }); + + test('KeyboardHandler pans map in response to arrow keys when disableRotation has been called', () => { + const map = createMap({zoom: 10, center: [0, 0]}); + const spy = jest.spyOn(map, 'easeTo'); + map.keyboard.disableRotation(); + + simulate.keydown(map.getCanvas(), {keyCode: 32, key: ' '}); + expect(map.easeTo).not.toHaveBeenCalled(); + + simulate.keydown(map.getCanvas(), {keyCode: 37, key: 'ArrowLeft'}); + expect(map.easeTo).toHaveBeenCalled(); + let easeToArgs = spy.mock.calls[0][0]; + expect(easeToArgs.offset[0]).toBe(100); + expect(easeToArgs.offset[1]).toBe(-0); + + simulate.keydown(map.getCanvas(), {keyCode: 39, key: 'ArrowRight'}); + expect(spy).toHaveBeenCalledTimes(2); + easeToArgs = spy.mock.calls[1][0]; + expect(easeToArgs.offset[0]).toBe(-100); + expect(easeToArgs.offset[1]).toBe(-0); + + simulate.keydown(map.getCanvas(), {keyCode: 40, key: 'ArrowDown'}); + expect(spy).toHaveBeenCalledTimes(3); + easeToArgs = spy.mock.calls[2][0]; + expect(easeToArgs.offset[0]).toBe(-0); + expect(easeToArgs.offset[1]).toBe(-100); + + simulate.keydown(map.getCanvas(), {keyCode: 38, key: 'ArrowUp'}); + expect(spy).toHaveBeenCalledTimes(4); + easeToArgs = spy.mock.calls[3][0]; + expect(easeToArgs.offset[0]).toBe(-0); + expect(easeToArgs.offset[1]).toBe(100); + + }); + + test('KeyboardHandler rotates map in response to Shift+left/right arrow keys', async () => { + const map = createMap({zoom: 10, center: [0, 0], bearing: 0}); + const spy = jest.spyOn(map, 'easeTo'); + + simulate.keydown(map.getCanvas(), {keyCode: 32, key: ' '}); + expect(map.easeTo).not.toHaveBeenCalled(); + + simulate.keydown(map.getCanvas(), {keyCode: 37, key: 'ArrowLeft', shiftKey: true}); + expect(map.easeTo).toHaveBeenCalled(); + let easeToArgs = spy.mock.calls[0][0]; + expect(easeToArgs.bearing).toBe(-15); + expect(easeToArgs.offset[0]).toBe(-0); + + map.setBearing(0); + simulate.keydown(map.getCanvas(), {keyCode: 39, key: 'ArrowRight', shiftKey: true}); + expect(spy).toHaveBeenCalledTimes(2); + easeToArgs = spy.mock.calls[1][0]; + expect(easeToArgs.bearing).toBe(15); + expect(easeToArgs.offset[0]).toBe(-0); + + }); + + test('KeyboardHandler does not rotate map in response to Shift+left/right arrow keys when disableRotation has been called', async () => { + const map = createMap({zoom: 10, center: [0, 0], bearing: 0}); + const spy = jest.spyOn(map, 'easeTo'); + map.keyboard.disableRotation(); + + simulate.keydown(map.getCanvas(), {keyCode: 32, key: ' '}); + expect(map.easeTo).not.toHaveBeenCalled(); + + simulate.keydown(map.getCanvas(), {keyCode: 37, key: 'ArrowLeft', shiftKey: true}); + expect(map.easeTo).toHaveBeenCalled(); + let easeToArgs = spy.mock.calls[0][0]; + expect(easeToArgs.bearing).toBe(0); + expect(easeToArgs.offset[0]).toBe(-0); + + map.setBearing(0); + simulate.keydown(map.getCanvas(), {keyCode: 39, key: 'ArrowRight', shiftKey: true}); + expect(spy).toHaveBeenCalledTimes(2); + easeToArgs = spy.mock.calls[1][0]; + expect(easeToArgs.bearing).toBe(0); + expect(easeToArgs.offset[0]).toBe(-0); + + }); + + test('KeyboardHandler pitches map in response to Shift+up/down arrow keys', async () => { + const map = createMap({zoom: 10, center: [0, 0], pitch: 30}); + const spy = jest.spyOn(map, 'easeTo'); + + simulate.keydown(map.getCanvas(), {keyCode: 32, key: ' '}); + expect(map.easeTo).not.toHaveBeenCalled(); + + simulate.keydown(map.getCanvas(), {keyCode: 40, key: 'ArrowDown', shiftKey: true}); + expect(map.easeTo).toHaveBeenCalled(); + let easeToArgs = spy.mock.calls[0][0]; + expect(easeToArgs.pitch).toBe(20); + expect(easeToArgs.offset[1]).toBe(-0); + + map.setPitch(30); + simulate.keydown(map.getCanvas(), {keyCode: 38, key: 'ArrowUp', shiftKey: true}); + expect(spy).toHaveBeenCalledTimes(2); + easeToArgs = spy.mock.calls[1][0]; + expect(easeToArgs.pitch).toBe(40); + expect(easeToArgs.offset[1]).toBe(-0); + + }); + + test('KeyboardHandler does not pitch map in response to Shift+up/down arrow keys when disableRotation has been called', async () => { + const map = createMap({zoom: 10, center: [0, 0], pitch: 30}); + const spy = jest.spyOn(map, 'easeTo'); + map.keyboard.disableRotation(); + + simulate.keydown(map.getCanvas(), {keyCode: 32, key: ' '}); + expect(map.easeTo).not.toHaveBeenCalled(); + + simulate.keydown(map.getCanvas(), {keyCode: 40, key: 'ArrowDown', shiftKey: true}); + expect(map.easeTo).toHaveBeenCalled(); + let easeToArgs = spy.mock.calls[0][0]; + expect(easeToArgs.pitch).toBe(30); + expect(easeToArgs.offset[1]).toBe(-0); + + map.setPitch(30); + simulate.keydown(map.getCanvas(), {keyCode: 38, key: 'ArrowUp', shiftKey: true}); + expect(spy).toHaveBeenCalledTimes(2); + easeToArgs = spy.mock.calls[1][0]; + expect(easeToArgs.pitch).toBe(30); + expect(easeToArgs.offset[1]).toBe(-0); + + }); + + test('KeyboardHandler zooms map in response to -/+ keys', () => { + const map = createMap({zoom: 10, center: [0, 0]}); + const spy = jest.spyOn(map, 'easeTo'); + + simulate.keydown(map.getCanvas(), {keyCode: 187, key: 'Equal'}); + expect(spy).toHaveBeenCalledTimes(1); + expect(spy.mock.calls[0][0].zoom).toBe(11); + + map.setZoom(10); + simulate.keydown(map.getCanvas(), {keyCode: 187, key: 'Equal', shiftKey: true}); + expect(spy).toHaveBeenCalledTimes(2); + expect(spy.mock.calls[1][0].zoom).toBe(12); + + map.setZoom(10); + simulate.keydown(map.getCanvas(), {keyCode: 189, key: 'Minus'}); + expect(spy).toHaveBeenCalledTimes(3); + expect(spy.mock.calls[2][0].zoom).toBe(9); + + map.setZoom(10); + simulate.keydown(map.getCanvas(), {keyCode: 189, key: 'Minus', shiftKey: true}); + expect(spy).toHaveBeenCalledTimes(4); + expect(spy.mock.calls[3][0].zoom).toBe(8); + + }); + + test('KeyboardHandler zooms map in response to -/+ keys when disableRotation has been called', () => { + const map = createMap({zoom: 10, center: [0, 0]}); + const spy = jest.spyOn(map, 'easeTo'); + map.keyboard.disableRotation(); + + simulate.keydown(map.getCanvas(), {keyCode: 187, key: 'Equal'}); + expect(spy).toHaveBeenCalledTimes(1); + expect(spy.mock.calls[0][0].zoom).toBe(11); + + map.setZoom(10); + simulate.keydown(map.getCanvas(), {keyCode: 187, key: 'Equal', shiftKey: true}); + expect(spy).toHaveBeenCalledTimes(2); + expect(spy.mock.calls[1][0].zoom).toBe(12); + + map.setZoom(10); + simulate.keydown(map.getCanvas(), {keyCode: 189, key: 'Minus'}); + expect(spy).toHaveBeenCalledTimes(3); + expect(spy.mock.calls[2][0].zoom).toBe(9); + + map.setZoom(10); + simulate.keydown(map.getCanvas(), {keyCode: 189, key: 'Minus', shiftKey: true}); + expect(spy).toHaveBeenCalledTimes(4); + expect(spy.mock.calls[3][0].zoom).toBe(8); + + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/keyboard.ts b/web/libraries/maplibre-gl/src/ui/handler/keyboard.ts new file mode 100644 index 00000000..ffdf349f --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/keyboard.ts @@ -0,0 +1,212 @@ +import {Handler} from '../handler_manager'; +import type {Map} from '../map'; +import {TransformProvider} from './transform-provider'; + +const defaultOptions = { + panStep: 100, + bearingStep: 15, + pitchStep: 10 +}; + +/** + * The `KeyboardHandler` allows the user to zoom, rotate, and pan the map using + * the following keyboard shortcuts: + * + * - `=` / `+`: Increase the zoom level by 1. + * - `Shift-=` / `Shift-+`: Increase the zoom level by 2. + * - `-`: Decrease the zoom level by 1. + * - `Shift--`: Decrease the zoom level by 2. + * - Arrow keys: Pan by 100 pixels. + * - `Shift+⇢`: Increase the rotation by 15 degrees. + * - `Shift+⇠`: Decrease the rotation by 15 degrees. + * - `Shift+⇡`: Increase the pitch by 10 degrees. + * - `Shift+⇣`: Decrease the pitch by 10 degrees. + * + * @group Handlers + */ +export class KeyboardHandler implements Handler { + _tr: TransformProvider; + _enabled: boolean; + _active: boolean; + _panStep: number; + _bearingStep: number; + _pitchStep: number; + _rotationDisabled: boolean; + + /** @internal */ + constructor(map: Map) { + this._tr = new TransformProvider(map); + const stepOptions = defaultOptions; + this._panStep = stepOptions.panStep; + this._bearingStep = stepOptions.bearingStep; + this._pitchStep = stepOptions.pitchStep; + this._rotationDisabled = false; + } + + reset() { + this._active = false; + } + + keydown(e: KeyboardEvent) { + if (e.altKey || e.ctrlKey || e.metaKey) return; + + let zoomDir = 0; + let bearingDir = 0; + let pitchDir = 0; + let xDir = 0; + let yDir = 0; + + switch (e.keyCode) { + case 61: + case 107: + case 171: + case 187: + zoomDir = 1; + break; + + case 189: + case 109: + case 173: + zoomDir = -1; + break; + + case 37: + if (e.shiftKey) { + bearingDir = -1; + } else { + e.preventDefault(); + xDir = -1; + } + break; + + case 39: + if (e.shiftKey) { + bearingDir = 1; + } else { + e.preventDefault(); + xDir = 1; + } + break; + + case 38: + if (e.shiftKey) { + pitchDir = 1; + } else { + e.preventDefault(); + yDir = -1; + } + break; + + case 40: + if (e.shiftKey) { + pitchDir = -1; + } else { + e.preventDefault(); + yDir = 1; + } + break; + + default: + return; + } + + if (this._rotationDisabled) { + bearingDir = 0; + pitchDir = 0; + } + + return { + cameraAnimation: (map: Map) => { + const tr = this._tr; + map.easeTo({ + duration: 300, + easeId: 'keyboardHandler', + easing: easeOut, + + zoom: zoomDir ? Math.round(tr.zoom) + zoomDir * (e.shiftKey ? 2 : 1) : tr.zoom, + bearing: tr.bearing + bearingDir * this._bearingStep, + pitch: tr.pitch + pitchDir * this._pitchStep, + offset: [-xDir * this._panStep, -yDir * this._panStep], + center: tr.center + }, {originalEvent: e}); + } + }; + } + + /** + * Enables the "keyboard rotate and zoom" interaction. + * + * @example + * ```ts + * map.keyboard.enable(); + * ``` + */ + enable() { + this._enabled = true; + } + + /** + * Disables the "keyboard rotate and zoom" interaction. + * + * @example + * ```ts + * map.keyboard.disable(); + * ``` + */ + disable() { + this._enabled = false; + this.reset(); + } + + /** + * Returns a Boolean indicating whether the "keyboard rotate and zoom" + * interaction is enabled. + * + * @returns `true` if the "keyboard rotate and zoom" + * interaction is enabled. + */ + isEnabled() { + return this._enabled; + } + + /** + * Returns true if the handler is enabled and has detected the start of a + * zoom/rotate gesture. + * + * @returns `true` if the handler is enabled and has detected the + * start of a zoom/rotate gesture. + */ + isActive() { + return this._active; + } + + /** + * Disables the "keyboard pan/rotate" interaction, leaving the + * "keyboard zoom" interaction enabled. + * + * @example + * ```ts + * map.keyboard.disableRotation(); + * ``` + */ + disableRotation() { + this._rotationDisabled = true; + } + + /** + * Enables the "keyboard pan/rotate" interaction. + * + * @example + * ```ts + * map.keyboard.enable(); + * map.keyboard.enableRotation(); + * ``` + */ + enableRotation() { + this._rotationDisabled = false; + } +} + +function easeOut(t: number) { + return t * (2 - t); +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/map_event.test.ts b/web/libraries/maplibre-gl/src/ui/handler/map_event.test.ts new file mode 100644 index 00000000..9e16255f --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/map_event.test.ts @@ -0,0 +1,158 @@ +import {Map, MapOptions} from '../map'; +import {DOM} from '../../util/dom'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {beforeMapTest} from '../../util/test/util'; + +function createMap() { + return new Map({interactive: true, container: DOM.create('div', '', window.document.body)} as any as MapOptions); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('map events', () => { + test('MapEvent handler fires touch events with correct values', () => { + const map = createMap(); + const target = map.getCanvas(); + + const touchstart = jest.fn(); + const touchmove = jest.fn(); + const touchend = jest.fn(); + + map.on('touchstart', touchstart); + map.on('touchmove', touchmove); + map.on('touchend', touchend); + + const touchesStart = [{target, identifier: 1, clientX: 0, clientY: 50}]; + const touchesMove = [{target, identifier: 1, clientX: 0, clientY: 60}]; + const touchesEnd = [{target, identifier: 1, clientX: 0, clientY: 60}]; + + simulate.touchstart(map.getCanvas(), {touches: touchesStart, targetTouches: touchesStart}); + expect(touchstart).toHaveBeenCalledTimes(1); + expect(touchstart.mock.calls[0][0].point).toEqual({x: 0, y: 50}); + expect(touchmove).toHaveBeenCalledTimes(0); + expect(touchend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: touchesMove, targetTouches: touchesMove}); + expect(touchstart).toHaveBeenCalledTimes(1); + expect(touchmove).toHaveBeenCalledTimes(1); + expect(touchmove.mock.calls[0][0].point).toEqual({x: 0, y: 60}); + expect(touchend).toHaveBeenCalledTimes(0); + + simulate.touchend(map.getCanvas(), {touches: [], targetTouches: [], changedTouches: touchesEnd}); + expect(touchstart).toHaveBeenCalledTimes(1); + expect(touchmove).toHaveBeenCalledTimes(1); + expect(touchend).toHaveBeenCalledTimes(1); + expect(touchend.mock.calls[0][0].point).toEqual({x: 0, y: 60}); + + map.remove(); + }); + + test('MapEvent handler fires touchmove even while drag handler is active', () => { + const map = createMap(); + const target = map.getCanvas(); + map.dragPan.enable(); + + const touchstart = jest.fn(); + const touchmove = jest.fn(); + const touchend = jest.fn(); + const drag = jest.fn(); + + map.on('touchstart', touchstart); + map.on('touchmove', touchmove); + map.on('touchend', touchend); + map.on('drag', drag); + + const touchesStart = [{target, identifier: 1, clientX: 0, clientY: 50}]; + const touchesMove = [{target, identifier: 1, clientX: 0, clientY: 60}]; + const touchesEnd = [{target, identifier: 1, clientX: 0, clientY: 60}]; + + simulate.touchstart(map.getCanvas(), {touches: touchesStart, targetTouches: touchesStart}); + expect(touchstart).toHaveBeenCalledTimes(1); + expect(touchstart.mock.calls[0][0].point).toEqual({x: 0, y: 50}); + expect(touchmove).toHaveBeenCalledTimes(0); + expect(touchend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: touchesMove, targetTouches: touchesMove}); + expect(touchstart).toHaveBeenCalledTimes(1); + expect(touchmove).toHaveBeenCalledTimes(1); + expect(touchmove.mock.calls[0][0].point).toEqual({x: 0, y: 60}); + expect(touchend).toHaveBeenCalledTimes(0); + + simulate.touchend(map.getCanvas(), {touches: [], targetTouches: [], changedTouches: touchesEnd}); + expect(touchstart).toHaveBeenCalledTimes(1); + expect(touchmove).toHaveBeenCalledTimes(1); + expect(touchend).toHaveBeenCalledTimes(1); + expect(touchend.mock.calls[0][0].point).toEqual({x: 0, y: 60}); + + map._renderTaskQueue.run(); + expect(drag).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('MapEvent handler fires contextmenu on MacOS/Linux, but only at mouseup', () => { + const map = createMap(); + const target = map.getCanvas(); + map.dragPan.enable(); + + const contextmenu = jest.fn(); + + map.on('contextmenu', contextmenu); + + simulate.mousedown(map.getCanvas(), {target, button: 2, clientX: 10, clientY: 10}); + simulate.contextmenu(map.getCanvas(), {target}); // triggered immediately after mousedown + expect(contextmenu).toHaveBeenCalledTimes(0); + simulate.mouseup(map.getCanvas(), {target, button: 2, clientX: 10, clientY: 10}); + expect(contextmenu).toHaveBeenCalledTimes(1); + }); + + test('MapEvent handler does not fire contextmenu on MacOS/Linux, when moved', () => { + const map = createMap(); + const target = map.getCanvas(); + map.dragPan.enable(); + + const contextmenu = jest.fn(); + + map.on('contextmenu', contextmenu); + + simulate.mousedown(map.getCanvas(), {target, button: 2, clientX: 10, clientY: 10}); + simulate.contextmenu(map.getCanvas(), {target}); // triggered immediately after mousedown + simulate.mousemove(map.getCanvas(), {target, buttons: 2, clientX: 50, clientY: 10}); + simulate.mouseup(map.getCanvas(), {target, button: 2, clientX: 70, clientY: 10}); + expect(contextmenu).toHaveBeenCalledTimes(0); + }); + + test('MapEvent handler fires contextmenu on Windows', () => { + const map = createMap(); + const target = map.getCanvas(); + map.dragPan.enable(); + + const contextmenu = jest.fn(); + + map.on('contextmenu', contextmenu); + + simulate.mousedown(map.getCanvas(), {target, button: 2, clientX: 10, clientY: 10}); + simulate.mouseup(map.getCanvas(), {target, button: 2, clientX: 10, clientY: 10}); + expect(contextmenu).toHaveBeenCalledTimes(0); + simulate.contextmenu(map.getCanvas(), {target, button: 2, clientX: 10, clientY: 10}); // triggered only after mouseup + expect(contextmenu).toHaveBeenCalledTimes(1); + }); + + test('MapEvent handler does not fire contextmenu on Windows, when moved', () => { + const map = createMap(); + const target = map.getCanvas(); + map.dragPan.enable(); + + const contextmenu = jest.fn(); + + map.on('contextmenu', contextmenu); + + simulate.mousedown(map.getCanvas(), {target, button: 2, clientX: 10, clientY: 10}); + simulate.mousemove(map.getCanvas(), {target, buttons: 2, clientX: 50, clientY: 10}); + simulate.mouseup(map.getCanvas(), {target, button: 2, clientX: 50, clientY: 10}); + simulate.contextmenu(map.getCanvas(), {target, button: 2, clientX: 10, clientY: 10}); // triggered only after mouseup + expect(contextmenu).toHaveBeenCalledTimes(0); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/map_event.ts b/web/libraries/maplibre-gl/src/ui/handler/map_event.ts new file mode 100644 index 00000000..24b9af24 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/map_event.ts @@ -0,0 +1,161 @@ +import {MapMouseEvent, MapTouchEvent, MapWheelEvent} from '../events'; +import {Handler} from '../handler_manager'; +import type {Map} from '../map'; +import type Point from '@mapbox/point-geometry'; + +export class MapEventHandler implements Handler { + + _mousedownPos: Point; + _clickTolerance: number; + _map: Map; + + constructor(map: Map, options: { + clickTolerance: number; + }) { + this._map = map; + this._clickTolerance = options.clickTolerance; + } + + reset() { + delete this._mousedownPos; + } + + wheel(e: WheelEvent) { + // If mapEvent.preventDefault() is called by the user, prevent handlers such as: + // - ScrollZoom + return this._firePreventable(new MapWheelEvent(e.type, this._map, e)); + } + + mousedown(e: MouseEvent, point: Point) { + this._mousedownPos = point; + // If mapEvent.preventDefault() is called by the user, prevent handlers such as: + // - MousePan + // - MouseRotate + // - MousePitch + // - DblclickHandler + return this._firePreventable(new MapMouseEvent(e.type, this._map, e)); + } + + mouseup(e: MouseEvent) { + this._map.fire(new MapMouseEvent(e.type, this._map, e)); + } + + click(e: MouseEvent, point: Point) { + if (this._mousedownPos && this._mousedownPos.dist(point) >= this._clickTolerance) return; + this._map.fire(new MapMouseEvent(e.type, this._map, e)); + } + + dblclick(e: MouseEvent) { + // If mapEvent.preventDefault() is called by the user, prevent handlers such as: + // - DblClickZoom + return this._firePreventable(new MapMouseEvent(e.type, this._map, e)); + } + + mouseover(e: MouseEvent) { + this._map.fire(new MapMouseEvent(e.type, this._map, e)); + } + + mouseout(e: MouseEvent) { + this._map.fire(new MapMouseEvent(e.type, this._map, e)); + } + + touchstart(e: TouchEvent) { + // If mapEvent.preventDefault() is called by the user, prevent handlers such as: + // - TouchPan + // - TouchZoom + // - TouchRotate + // - TouchPitch + // - TapZoom + // - SwipeZoom + return this._firePreventable(new MapTouchEvent(e.type, this._map, e)); + } + + touchmove(e: TouchEvent) { + this._map.fire(new MapTouchEvent(e.type, this._map, e)); + } + + touchend(e: TouchEvent) { + this._map.fire(new MapTouchEvent(e.type, this._map, e)); + } + + touchcancel(e: TouchEvent) { + this._map.fire(new MapTouchEvent(e.type, this._map, e)); + } + + _firePreventable(mapEvent: MapMouseEvent | MapTouchEvent | MapWheelEvent) { + this._map.fire(mapEvent); + if (mapEvent.defaultPrevented) { + // returning an object marks the handler as active and resets other handlers + return {}; + } + } + + isEnabled() { + return true; + } + + isActive() { + return false; + } + enable() {} + disable() {} +} + +export class BlockableMapEventHandler { + _map: Map; + _delayContextMenu: boolean; + _ignoreContextMenu: boolean; + _contextMenuEvent: MouseEvent; + + constructor(map: Map) { + this._map = map; + } + + reset() { + this._delayContextMenu = false; + this._ignoreContextMenu = true; + delete this._contextMenuEvent; + } + + mousemove(e: MouseEvent) { + // mousemove map events should not be fired when interaction handlers (pan, rotate, etc) are active + this._map.fire(new MapMouseEvent(e.type, this._map, e)); + } + + mousedown() { + this._delayContextMenu = true; + this._ignoreContextMenu = false; + } + + mouseup() { + this._delayContextMenu = false; + if (this._contextMenuEvent) { + this._map.fire(new MapMouseEvent('contextmenu', this._map, this._contextMenuEvent)); + delete this._contextMenuEvent; + } + } + contextmenu(e: MouseEvent) { + if (this._delayContextMenu) { + // Mac: contextmenu fired on mousedown; we save it until mouseup for consistency's sake + this._contextMenuEvent = e; + } else if (!this._ignoreContextMenu) { + // Windows: contextmenu fired on mouseup, so fire event now + this._map.fire(new MapMouseEvent(e.type, this._map, e)); + } + + // prevent browser context menu when necessary + if (this._map.listens('contextmenu')) { + e.preventDefault(); + } + } + + isEnabled() { + return true; + } + + isActive() { + return false; + } + enable() {} + disable() {} +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/mouse.ts b/web/libraries/maplibre-gl/src/ui/handler/mouse.ts new file mode 100644 index 00000000..595b46bf --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/mouse.ts @@ -0,0 +1,83 @@ +import type Point from '@mapbox/point-geometry'; + +import {DOM} from '../../util/dom'; +import {DragMoveHandler, DragPanResult, DragRotateResult, DragPitchResult, DragHandler} from './drag_handler'; +import {MouseMoveStateManager} from './drag_move_state_manager'; + +export interface MousePanHandler extends DragMoveHandler {} +export interface MouseRotateHandler extends DragMoveHandler {} +export interface MousePitchHandler extends DragMoveHandler {} + +const LEFT_BUTTON = 0; +const RIGHT_BUTTON = 2; + +const assignEvents = (handler: DragHandler) => { + handler.mousedown = handler.dragStart; + handler.mousemoveWindow = handler.dragMove; + handler.mouseup = handler.dragEnd; + handler.contextmenu = function(e: MouseEvent) { + e.preventDefault(); + }; +}; + +export const generateMousePanHandler = ({enable, clickTolerance,}: { + clickTolerance: number; + enable?: boolean; +}): MousePanHandler => { + const mouseMoveStateManager = new MouseMoveStateManager({ + checkCorrectEvent: (e: MouseEvent) => DOM.mouseButton(e) === LEFT_BUTTON && !e.ctrlKey, + }); + return new DragHandler({ + clickTolerance, + move: (lastPoint: Point, point: Point) => + ({around: point, panDelta: point.sub(lastPoint)}), + activateOnStart: true, + moveStateManager: mouseMoveStateManager, + enable, + assignEvents, + }); +}; + +export const generateMouseRotationHandler = ({enable, clickTolerance, bearingDegreesPerPixelMoved = 0.8}: { + clickTolerance: number; + bearingDegreesPerPixelMoved?: number; + enable?: boolean; +}): MouseRotateHandler => { + const mouseMoveStateManager = new MouseMoveStateManager({ + checkCorrectEvent: (e: MouseEvent): boolean => + (DOM.mouseButton(e) === LEFT_BUTTON && e.ctrlKey) || + (DOM.mouseButton(e) === RIGHT_BUTTON), + }); + return new DragHandler({ + clickTolerance, + move: (lastPoint: Point, point: Point) => + ({bearingDelta: (point.x - lastPoint.x) * bearingDegreesPerPixelMoved}), + // prevent browser context menu when necessary; we don't allow it with rotation + // because we can't discern rotation gesture start from contextmenu on Mac + moveStateManager: mouseMoveStateManager, + enable, + assignEvents, + }); +}; + +export const generateMousePitchHandler = ({enable, clickTolerance, pitchDegreesPerPixelMoved = -0.5}: { + clickTolerance: number; + pitchDegreesPerPixelMoved?: number; + enable?: boolean; +}): MousePitchHandler => { + const mouseMoveStateManager = new MouseMoveStateManager({ + checkCorrectEvent: (e: MouseEvent): boolean => + (DOM.mouseButton(e) === LEFT_BUTTON && e.ctrlKey) || + (DOM.mouseButton(e) === RIGHT_BUTTON), + }); + return new DragHandler({ + clickTolerance, + move: (lastPoint: Point, point: Point) => + ({pitchDelta: (point.y - lastPoint.y) * pitchDegreesPerPixelMoved}), + // prevent browser context menu when necessary; we don't allow it with rotation + // because we can't discern rotation gesture start from contextmenu on Mac + moveStateManager: mouseMoveStateManager, + enable, + assignEvents, + }); +}; diff --git a/web/libraries/maplibre-gl/src/ui/handler/mouse_handler_interface.test.ts b/web/libraries/maplibre-gl/src/ui/handler/mouse_handler_interface.test.ts new file mode 100644 index 00000000..6220cb6b --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/mouse_handler_interface.test.ts @@ -0,0 +1,111 @@ +import Point from '@mapbox/point-geometry'; + +import {generateMousePanHandler, generateMousePitchHandler, generateMouseRotationHandler} from './mouse'; + +describe('mouse handler tests', () => { + test('MouseRotateHandler', () => { + const mouseRotate = generateMouseRotationHandler({clickTolerance: 2}); + + expect(mouseRotate.isActive()).toBe(false); + expect(mouseRotate.isEnabled()).toBe(false); + mouseRotate.enable(); + expect(mouseRotate.isEnabled()).toBe(true); + + mouseRotate.dragStart(new MouseEvent('mousedown', {buttons: 2, button: 2}), new Point(0, 0)); + expect(mouseRotate.isActive()).toBe(false); + + const underToleranceMove = new MouseEvent('mousemove', {buttons: 2, clientX: 1, clientY: 1}); + expect(mouseRotate.dragMove(underToleranceMove, new Point(1, 1))).toBeUndefined(); + expect(mouseRotate.isActive()).toBe(false); + + const overToleranceMove = new MouseEvent('mousemove', {buttons: 2, clientX: 10, clientY: 10}); + expect(mouseRotate.dragMove(overToleranceMove, new Point(10, 10))).toEqual({'bearingDelta': 8}); + expect(mouseRotate.isActive()).toBe(true); + + mouseRotate.dragEnd(new MouseEvent('mouseup', {buttons: 0, button: 2})); + expect(mouseRotate.isActive()).toBe(false); + + mouseRotate.disable(); + expect(mouseRotate.isEnabled()).toBe(false); + + mouseRotate.dragStart(new MouseEvent('mousedown', {buttons: 2, button: 2}), new Point(0, 0)); + expect(mouseRotate.isActive()).toBe(false); + + expect(mouseRotate.dragMove(underToleranceMove, new Point(1, 1))).toBeUndefined(); + expect(mouseRotate.isActive()).toBe(false); + + expect(mouseRotate.dragMove(overToleranceMove, new Point(10, 10))).toBeUndefined(); + expect(mouseRotate.isActive()).toBe(false); + }); + + test('MousePitchHandler', () => { + const mousePitch = generateMousePitchHandler({clickTolerance: 2}); + + expect(mousePitch.isActive()).toBe(false); + expect(mousePitch.isEnabled()).toBe(false); + mousePitch.enable(); + expect(mousePitch.isEnabled()).toBe(true); + + mousePitch.dragStart(new MouseEvent('mousedown', {buttons: 2, button: 2}), new Point(0, 0)); + expect(mousePitch.isActive()).toBe(false); + + const underToleranceMove = new MouseEvent('mousemove', {buttons: 2, clientX: 1, clientY: 1}); + expect(mousePitch.dragMove(underToleranceMove, new Point(1, 1))).toBeUndefined(); + expect(mousePitch.isActive()).toBe(false); + + const overToleranceMove = new MouseEvent('mousemove', {buttons: 2, clientX: 10, clientY: 10}); + expect(mousePitch.dragMove(overToleranceMove, new Point(10, 10))).toEqual({'pitchDelta': -5}); + expect(mousePitch.isActive()).toBe(true); + + mousePitch.dragEnd(new MouseEvent('mouseup', {buttons: 0, button: 2})); + expect(mousePitch.isActive()).toBe(false); + + mousePitch.disable(); + expect(mousePitch.isEnabled()).toBe(false); + + mousePitch.dragStart(new MouseEvent('mousedown', {buttons: 2, button: 2}), new Point(0, 0)); + expect(mousePitch.isActive()).toBe(false); + + expect(mousePitch.dragMove(underToleranceMove, new Point(1, 1))).toBeUndefined(); + expect(mousePitch.isActive()).toBe(false); + + expect(mousePitch.dragMove(overToleranceMove, new Point(10, 10))).toBeUndefined(); + expect(mousePitch.isActive()).toBe(false); + }); + + test('MousePanHandler', () => { + const mousePan = generateMousePanHandler({clickTolerance: 2}); + + expect(mousePan.isActive()).toBe(false); + expect(mousePan.isEnabled()).toBe(false); + mousePan.enable(); + expect(mousePan.isEnabled()).toBe(true); + + mousePan.dragStart(new MouseEvent('mousedown', {buttons: 1, button: 0}), new Point(0, 0)); + expect(mousePan.isActive()).toBe(true); + + const underToleranceMove = new MouseEvent('mousemove', {buttons: 1, clientX: 1, clientY: 1}); + expect(mousePan.dragMove(underToleranceMove, new Point(1, 1))).toBeUndefined(); + expect(mousePan.isActive()).toBe(true); + + const overToleranceMove = new MouseEvent('mousemove', {buttons: 1, clientX: 10, clientY: 10}); + expect(mousePan.dragMove(overToleranceMove, new Point(10, 10))) + .toEqual({'around': {'x': 10, 'y': 10,}, 'panDelta': {'x': 10, 'y': 10,}}); + expect(mousePan.isActive()).toBe(true); + + mousePan.dragEnd(new MouseEvent('mouseup', {buttons: 0, button: 0})); + expect(mousePan.isActive()).toBe(false); + + mousePan.disable(); + expect(mousePan.isEnabled()).toBe(false); + + mousePan.dragStart(new MouseEvent('mousedown', {buttons: 2, button: 2}), new Point(0, 0)); + expect(mousePan.isActive()).toBe(false); + + expect(mousePan.dragMove(underToleranceMove, new Point(1, 1))).toBeUndefined(); + expect(mousePan.isActive()).toBe(false); + + expect(mousePan.dragMove(overToleranceMove, new Point(10, 10))).toBeUndefined(); + expect(mousePan.isActive()).toBe(false); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/mouse_rotate.test.ts b/web/libraries/maplibre-gl/src/ui/handler/mouse_rotate.test.ts new file mode 100644 index 00000000..2a840f31 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/mouse_rotate.test.ts @@ -0,0 +1,62 @@ +import {extend} from '../../util/util'; +import {Map} from '../../ui/map'; +import {DOM} from '../../util/dom'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {browser} from '../../util/browser'; +import {beforeMapTest} from '../../util/test/util'; + +function createMap(options?) { + return new Map(extend({container: DOM.create('div', '', window.document.body)}, options)); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('mouse rotate', () => { + test('MouseRotateHandler#isActive', () => { + const map = createMap({interactive: true}); + const mouseRotate = map.handlers._handlersById.mouseRotate; + + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockReturnValue(0); + expect(mouseRotate.isActive()).toBe(false); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2, clientX: 0, clientY: 0}); + map._renderTaskQueue.run(); + expect(mouseRotate.isActive()).toBe(false); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(mouseRotate.isActive()).toBe(true); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + expect(mouseRotate.isActive()).toBe(false); + + map.remove(); + }); + + test('MouseRotateHandler#isActive #4622 regression test', () => { + const map = createMap({interactive: true}); + const mouseRotate = map.handlers._handlersById.mouseRotate; + + // Prevent inertial rotation. + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + map._renderTaskQueue.run(); + expect(mouseRotate.isActive()).toBe(false); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(mouseRotate.isActive()).toBe(true); + + // Some browsers don't fire mouseup when it happens outside the window. + // Make the handler in active when it encounters a mousemove without the button pressed. + + simulate.mousemove(map.getCanvas(), {buttons: 0, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + expect(mouseRotate.isActive()).toBe(false); + + map.remove(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/one_finger_touch_drag.ts b/web/libraries/maplibre-gl/src/ui/handler/one_finger_touch_drag.ts new file mode 100644 index 00000000..bc67d84a --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/one_finger_touch_drag.ts @@ -0,0 +1,45 @@ +import type Point from '@mapbox/point-geometry'; + +import {DragMoveHandler, DragRotateResult, DragPitchResult, DragHandler} from './drag_handler'; +import {OneFingerTouchMoveStateManager} from './drag_move_state_manager'; + +export interface OneFingerTouchRotateHandler extends DragMoveHandler {} +export interface OneFingerTouchPitchHandler extends DragMoveHandler {} + +const assignEvents = (handler: DragHandler) => { + handler.touchstart = handler.dragStart; + handler.touchmoveWindow = handler.dragMove; + handler.touchend = handler.dragEnd; +}; + +export const generateOneFingerTouchRotationHandler = ({enable, clickTolerance, bearingDegreesPerPixelMoved = 0.8}: { + clickTolerance: number; + bearingDegreesPerPixelMoved?: number; + enable?: boolean; +}): OneFingerTouchRotateHandler => { + const touchMoveStateManager = new OneFingerTouchMoveStateManager(); + return new DragHandler({ + clickTolerance, + move: (lastPoint: Point, point: Point) => + ({bearingDelta: (point.x - lastPoint.x) * bearingDegreesPerPixelMoved}), + moveStateManager: touchMoveStateManager, + enable, + assignEvents, + }); +}; + +export const generateOneFingerTouchPitchHandler = ({enable, clickTolerance, pitchDegreesPerPixelMoved = -0.5}: { + clickTolerance: number; + pitchDegreesPerPixelMoved?: number; + enable?: boolean; +}): OneFingerTouchPitchHandler => { + const touchMoveStateManager = new OneFingerTouchMoveStateManager(); + return new DragHandler({ + clickTolerance, + move: (lastPoint: Point, point: Point) => + ({pitchDelta: (point.y - lastPoint.y) * pitchDegreesPerPixelMoved}), + moveStateManager: touchMoveStateManager, + enable, + assignEvents, + }); +}; diff --git a/web/libraries/maplibre-gl/src/ui/handler/one_finger_touch_drag_handler_interface.test.ts b/web/libraries/maplibre-gl/src/ui/handler/one_finger_touch_drag_handler_interface.test.ts new file mode 100644 index 00000000..9eae8f20 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/one_finger_touch_drag_handler_interface.test.ts @@ -0,0 +1,77 @@ +import Point from '@mapbox/point-geometry'; + +import {generateOneFingerTouchPitchHandler, generateOneFingerTouchRotationHandler} from './one_finger_touch_drag'; + +const testTouch = {identifier: 0} as Touch; + +describe('one touch drag handler tests', () => { + test('OneFingerTouchRotateHandler', () => { + const oneTouchRotate = generateOneFingerTouchRotationHandler({clickTolerance: 2}); + + expect(oneTouchRotate.isActive()).toBe(false); + expect(oneTouchRotate.isEnabled()).toBe(false); + oneTouchRotate.enable(); + expect(oneTouchRotate.isEnabled()).toBe(true); + + oneTouchRotate.dragStart(new TouchEvent('touchstart', {targetTouches: [testTouch]}), new Point(0, 0)); + expect(oneTouchRotate.isActive()).toBe(false); + + const underToleranceMove = new TouchEvent('touchmove', {targetTouches: [testTouch]}); + expect(oneTouchRotate.dragMove(underToleranceMove, new Point(1, 1))).toBeUndefined(); + expect(oneTouchRotate.isActive()).toBe(false); + + const overToleranceMove = new TouchEvent('touchmove', {targetTouches: [testTouch]}); + expect(oneTouchRotate.dragMove(overToleranceMove, new Point(10, 10))).toEqual({'bearingDelta': 8}); + expect(oneTouchRotate.isActive()).toBe(true); + + oneTouchRotate.dragEnd(new TouchEvent('touchend', {targetTouches: [testTouch]})); + expect(oneTouchRotate.isActive()).toBe(false); + + oneTouchRotate.disable(); + expect(oneTouchRotate.isEnabled()).toBe(false); + + oneTouchRotate.dragStart(new TouchEvent('touchstart', {targetTouches: [testTouch]}), new Point(0, 0)); + expect(oneTouchRotate.isActive()).toBe(false); + + expect(oneTouchRotate.dragMove(underToleranceMove, new Point(1, 1))).toBeUndefined(); + expect(oneTouchRotate.isActive()).toBe(false); + + expect(oneTouchRotate.dragMove(overToleranceMove, new Point(10, 10))).toBeUndefined(); + expect(oneTouchRotate.isActive()).toBe(false); + }); + + test('OneFingerTouchPitchHandler', () => { + const oneTouchPitch = generateOneFingerTouchPitchHandler({clickTolerance: 2}); + + expect(oneTouchPitch.isActive()).toBe(false); + expect(oneTouchPitch.isEnabled()).toBe(false); + oneTouchPitch.enable(); + expect(oneTouchPitch.isEnabled()).toBe(true); + + oneTouchPitch.dragStart(new TouchEvent('touchstart', {targetTouches: [testTouch]}), new Point(0, 0)); + expect(oneTouchPitch.isActive()).toBe(false); + + const underToleranceMove = new TouchEvent('touchmove', {targetTouches: [testTouch]}); + expect(oneTouchPitch.dragMove(underToleranceMove, new Point(1, 1))).toBeUndefined(); + expect(oneTouchPitch.isActive()).toBe(false); + + const overToleranceMove = new TouchEvent('touchmove', {targetTouches: [testTouch]}); + expect(oneTouchPitch.dragMove(overToleranceMove, new Point(10, 10))).toEqual({'pitchDelta': -5}); + expect(oneTouchPitch.isActive()).toBe(true); + + oneTouchPitch.dragEnd(new TouchEvent('touchend', {targetTouches: [testTouch]})); + expect(oneTouchPitch.isActive()).toBe(false); + + oneTouchPitch.disable(); + expect(oneTouchPitch.isEnabled()).toBe(false); + + oneTouchPitch.dragStart(new TouchEvent('touchstart', {targetTouches: [testTouch]}), new Point(0, 0)); + expect(oneTouchPitch.isActive()).toBe(false); + + expect(oneTouchPitch.dragMove(underToleranceMove, new Point(1, 1))).toBeUndefined(); + expect(oneTouchPitch.isActive()).toBe(false); + + expect(oneTouchPitch.dragMove(overToleranceMove, new Point(10, 10))).toBeUndefined(); + expect(oneTouchPitch.isActive()).toBe(false); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/scroll_zoom.test.ts b/web/libraries/maplibre-gl/src/ui/handler/scroll_zoom.test.ts new file mode 100644 index 00000000..7efcf899 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/scroll_zoom.test.ts @@ -0,0 +1,298 @@ +import {browser} from '../../util/browser'; +import {Map} from '../../ui/map'; +import {DOM} from '../../util/dom'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {setPerformance, beforeMapTest} from '../../util/test/util'; + +function createMap() { + return new Map({ + container: DOM.create('div', '', window.document.body), + style: { + 'version': 8, + 'sources': {}, + 'layers': [] + } + }); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('ScrollZoomHandler', () => { + + test('Zooms for single mouse wheel tick', () => { + const browserNow = jest.spyOn(browser, 'now'); + let now = 1555555555555; + browserNow.mockReturnValue(now); + + const map = createMap(); + map._renderTaskQueue.run(); + + // simulate a single 'wheel' event + const startZoom = map.getZoom(); + + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta}); + map._renderTaskQueue.run(); + + now += 400; + browserNow.mockReturnValue(now); + map._renderTaskQueue.run(); + + expect(map.getZoom() - startZoom).toBeCloseTo(0.0285, 3); + + map.remove(); + }); + + test('Zooms for single mouse wheel tick with non-magical deltaY', done => { + const browserNow = jest.spyOn(browser, 'now'); + const now = 1555555555555; + browserNow.mockReturnValue(now); + + const map = createMap(); + map._renderTaskQueue.run(); + + // Simulate a single 'wheel' event without the magical deltaY value. + // This requires the handler to briefly wait to see if a subsequent + // event is coming in order to guess trackpad vs. mouse wheel + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -20}); + map.on('zoomstart', () => { + map.remove(); + done(); + }); + }); + + test('Zooms for multiple mouse wheel ticks', () => { + const browserNow = jest.spyOn(browser, 'now'); + let now = 1555555555555; + browserNow.mockReturnValue(now); + + const map = createMap(); + + map._renderTaskQueue.run(); + const startZoom = map.getZoom(); + + const events = [ + [2, {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta}], + [7, {type: 'wheel', deltaY: -41}], + [30, {type: 'wheel', deltaY: -169}], + [1, {type: 'wheel', deltaY: -801}], + [5, {type: 'wheel', deltaY: -326}], + [20, {type: 'wheel', deltaY: -345}], + [22, {type: 'wheel', deltaY: -376}], + ] as [number, any][]; + + const end = now + 500; + let lastWheelEvent = now; + + // simulate the above sequence of wheel events, with render frames + // interspersed every 20ms + while (now < end) { + now += 1; + browserNow.mockReturnValue(now); + if (events.length && lastWheelEvent + events[0][0] === now) { + const [, event] = events.shift(); + simulate.wheel(map.getCanvas(), event); + lastWheelEvent = now; + } + if (now % 20 === 0) { + map._renderTaskQueue.run(); + } + } + + expect(map.getZoom() - startZoom).toBeCloseTo(1.944, 3); + + map.remove(); + }); + + test('Gracefully ignores wheel events with deltaY: 0', () => { + const browserNow = jest.spyOn(browser, 'now'); + let now = 1555555555555; + browserNow.mockReturnValue(now); + + const map = createMap(); + map._renderTaskQueue.run(); + + const startZoom = map.getZoom(); + // simulate shift+'wheel' events + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -0, shiftKey: true}); + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -0, shiftKey: true}); + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -0, shiftKey: true}); + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -0, shiftKey: true}); + map._renderTaskQueue.run(); + + now += 400; + browserNow.mockReturnValue(now); + map._renderTaskQueue.run(); + + expect(map.getZoom() - startZoom).toBe(0.0); + + }); + + test('Gracefully handle wheel events that cancel each other out before the first scroll frame', () => { + // See also https://github.com/mapbox/mapbox-gl-js/issues/6782 + const browserNow = jest.spyOn(browser, 'now'); + let now = 1555555555555; + browserNow.mockReturnValue(now); + + const map = createMap(); + map._renderTaskQueue.run(); + + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -1}); + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -1}); + now += 1; + browserNow.mockReturnValue(now); + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: 2}); + + map._renderTaskQueue.run(); + + now += 400; + browserNow.mockReturnValue(now); + map._renderTaskQueue.run(); + + }); + + test('does not zoom if preventDefault is called on the wheel event', () => { + const browserNow = jest.spyOn(browser, 'now'); + let now = 1555555555555; + browserNow.mockReturnValue(now); + + const map = createMap(); + + map.on('wheel', e => e.preventDefault()); + + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta}); + map._renderTaskQueue.run(); + + now += 400; + browserNow.mockReturnValue(now); + map._renderTaskQueue.run(); + + expect(map.getZoom()).toBe(0); + + map.remove(); + }); + + test('emits one movestart event and one moveend event while zooming', () => { + const browserNow = jest.spyOn(browser, 'now'); + let now = 1555555555555; + browserNow.mockReturnValue(now); + jest.useFakeTimers(); + setPerformance(); + const map = createMap(); + + let startCount = 0; + map.on('movestart', () => { + startCount += 1; + }); + + let endCount = 0; + map.on('moveend', () => { + endCount += 1; + }); + + const events = [ + [2, {type: 'trackpad', deltaY: -1}], + [7, {type: 'trackpad', deltaY: -2}], + [30, {type: 'wheel', deltaY: -5}] + ] as [number, any][]; + + const end = now + 50; + let lastWheelEvent = now; + + while (now < end) { + now += 1; + browserNow.mockReturnValue(now); + if (events.length && lastWheelEvent + events[0][0] === now) { + const [, event] = events.shift(); + simulate.wheel(map.getCanvas(), event); + lastWheelEvent = now; + } + if (now % 20 === 0) { + map._renderTaskQueue.run(); + } + } + + jest.advanceTimersByTime(200); + + map._renderTaskQueue.run(); + + expect(startCount).toBe(1); + expect(endCount).toBe(1); + + }); + + test('emits one zoomstart event and one zoomend event while zooming', () => { + const browserNow = jest.spyOn(browser, 'now'); + let now = 1555555555555; + browserNow.mockReturnValue(now); + + jest.useFakeTimers(); + setPerformance(); + const map = createMap(); + + let startCount = 0; + map.on('zoomstart', () => { + startCount += 1; + }); + + let endCount = 0; + map.on('zoomend', () => { + endCount += 1; + }); + + const events = [ + [2, {type: 'trackpad', deltaY: -1}], + [7, {type: 'trackpad', deltaY: -2}], + [30, {type: 'wheel', deltaY: -5}], + ] as [number, any][]; + + const end = now + 50; + let lastWheelEvent = now; + + while (now < end) { + now += 1; + browserNow.mockReturnValue(now); + if (events.length && lastWheelEvent + events[0][0] === now) { + const [, event] = events.shift(); + simulate.wheel(map.getCanvas(), event); + lastWheelEvent = now; + } + if (now % 20 === 0) { + map._renderTaskQueue.run(); + } + } + + jest.advanceTimersByTime(200); + map._renderTaskQueue.run(); + + expect(startCount).toBe(1); + expect(endCount).toBe(1); + + }); + + test('Zooms for single mouse wheel tick while not in the center of the map and terrain is on, should zoom according to mouse position', () => { + const browserNow = jest.spyOn(browser, 'now'); + let now = 1555555555555; + browserNow.mockReturnValue(now); + + const map = createMap(); + map._renderTaskQueue.run(); + map.terrain = { + pointCoordinate: () => null + } as any; + + // simulate a single 'wheel' event + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta, clientX: 1000, clientY: 1000}); + map._renderTaskQueue.run(); + + now += 400; + browserNow.mockReturnValue(now); + map._renderTaskQueue.run(); + + expect(map.getCenter().lat).toBeCloseTo(-11.6371, 3); + expect(map.getCenter().lng).toBeCloseTo(11.0286, 3); + + map.remove(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/scroll_zoom.ts b/web/libraries/maplibre-gl/src/ui/handler/scroll_zoom.ts new file mode 100644 index 00000000..abf354f4 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/scroll_zoom.ts @@ -0,0 +1,361 @@ +import {DOM} from '../../util/dom'; + +import {defaultEasing, bezier} from '../../util/util'; +import {browser} from '../../util/browser'; +import {interpolates} from '@maplibre/maplibre-gl-style-spec'; +import {LngLat} from '../../geo/lng_lat'; +import {TransformProvider} from './transform-provider'; + +import type {Map} from '../map'; +import type Point from '@mapbox/point-geometry'; +import type {AroundCenterOptions} from './two_fingers_touch'; +import {Handler} from '../handler_manager'; + +// deltaY value for mouse scroll wheel identification +const wheelZoomDelta = 4.000244140625; + +// These magic numbers control the rate of zoom. Trackpad events fire at a greater +// frequency than mouse scroll wheel, so reduce the zoom rate per wheel tick +const defaultZoomRate = 1 / 100; +const wheelZoomRate = 1 / 450; + +// upper bound on how much we scale the map in any single render frame; this +// is used to limit zoom rate in the case of very fast scrolling +const maxScalePerFrame = 2; + +/** + * The `ScrollZoomHandler` allows the user to zoom the map by scrolling. + * + * @group Handlers + */ +export class ScrollZoomHandler implements Handler { + _map: Map; + _tr: TransformProvider; + _el: HTMLElement; + _enabled: boolean; + _active: boolean; + _zooming: boolean; + _aroundCenter: boolean; + _around: LngLat; + _aroundPoint: Point; + _type: 'wheel' | 'trackpad' | null; + _lastValue: number; + _timeout: ReturnType; // used for delayed-handling of a single wheel movement + _finishTimeout: ReturnType; // used to delay final '{move,zoom}end' events + + _lastWheelEvent: any; + _lastWheelEventTime: number; + + _startZoom: number; + _targetZoom: number; + _delta: number; + _easing: ((a: number) => number); + _prevEase: { + start: number; + duration: number; + easing: (_: number) => number; + }; + + _frameId: boolean; + _triggerRenderFrame: () => void; + + _defaultZoomRate: number; + _wheelZoomRate: number; + + /** @internal */ + constructor(map: Map, triggerRenderFrame: () => void) { + this._map = map; + this._tr = new TransformProvider(map); + this._el = map.getCanvasContainer(); + this._triggerRenderFrame = triggerRenderFrame; + + this._delta = 0; + + this._defaultZoomRate = defaultZoomRate; + this._wheelZoomRate = wheelZoomRate; + } + + /** + * Set the zoom rate of a trackpad + * @param zoomRate - 1/100 The rate used to scale trackpad movement to a zoom value. + * @example + * Speed up trackpad zoom + * ```ts + * map.scrollZoom.setZoomRate(1/25); + * ``` + */ + setZoomRate(zoomRate: number) { + this._defaultZoomRate = zoomRate; + } + + /** + * Set the zoom rate of a mouse wheel + * @param wheelZoomRate - 1/450 The rate used to scale mouse wheel movement to a zoom value. + * @example + * Slow down zoom of mouse wheel + * ```ts + * map.scrollZoom.setWheelZoomRate(1/600); + * ``` + */ + setWheelZoomRate(wheelZoomRate: number) { + this._wheelZoomRate = wheelZoomRate; + } + + /** + * Returns a Boolean indicating whether the "scroll to zoom" interaction is enabled. + * @returns `true` if the "scroll to zoom" interaction is enabled. + */ + isEnabled() { + return !!this._enabled; + } + + /* + * Active state is turned on and off with every scroll wheel event and is set back to false before the map + * render is called, so _active is not a good candidate for determining if a scroll zoom animation is in + * progress. + */ + isActive() { + return !!this._active || this._finishTimeout !== undefined; + } + + isZooming() { + return !!this._zooming; + } + + /** + * Enables the "scroll to zoom" interaction. + * + * @param options - Options object. + * @example + * ```ts + * map.scrollZoom.enable(); + * map.scrollZoom.enable({ around: 'center' }) + * ``` + */ + enable(options?: AroundCenterOptions | boolean) { + if (this.isEnabled()) return; + this._enabled = true; + this._aroundCenter = !!options && (options as AroundCenterOptions).around === 'center'; + } + + /** + * Disables the "scroll to zoom" interaction. + * + * @example + * ```ts + * map.scrollZoom.disable(); + * ``` + */ + disable() { + if (!this.isEnabled()) return; + this._enabled = false; + } + + wheel(e: WheelEvent) { + if (!this.isEnabled()) return; + if (this._map._cooperativeGestures) { + if (e[this._map._metaKey]) { + e.preventDefault(); + } else { + return; + } + } + let value = e.deltaMode === WheelEvent.DOM_DELTA_LINE ? e.deltaY * 40 : e.deltaY; + const now = browser.now(), + timeDelta = now - (this._lastWheelEventTime || 0); + + this._lastWheelEventTime = now; + + if (value !== 0 && (value % wheelZoomDelta) === 0) { + // This one is definitely a mouse wheel event. + this._type = 'wheel'; + + } else if (value !== 0 && Math.abs(value) < 4) { + // This one is definitely a trackpad event because it is so small. + this._type = 'trackpad'; + + } else if (timeDelta > 400) { + // This is likely a new scroll action. + this._type = null; + this._lastValue = value; + + // Start a timeout in case this was a singular event, and dely it by up to 40ms. + this._timeout = setTimeout(this._onTimeout, 40, e); + + } else if (!this._type) { + // This is a repeating event, but we don't know the type of event just yet. + // If the delta per time is small, we assume it's a fast trackpad; otherwise we switch into wheel mode. + this._type = (Math.abs(timeDelta * value) < 200) ? 'trackpad' : 'wheel'; + + // Make sure our delayed event isn't fired again, because we accumulate + // the previous event (which was less than 40ms ago) into this event. + if (this._timeout) { + clearTimeout(this._timeout); + this._timeout = null; + value += this._lastValue; + } + } + + // Slow down zoom if shift key is held for more precise zooming + if (e.shiftKey && value) value = value / 4; + + // Only fire the callback if we actually know what type of scrolling device the user uses. + if (this._type) { + this._lastWheelEvent = e; + this._delta -= value; + if (!this._active) { + this._start(e); + } + } + + e.preventDefault(); + } + + _onTimeout = (initialEvent: MouseEvent) => { + this._type = 'wheel'; + this._delta -= this._lastValue; + if (!this._active) { + this._start(initialEvent); + } + }; + + _start(e: MouseEvent) { + if (!this._delta) return; + + if (this._frameId) { + this._frameId = null; + } + + this._active = true; + if (!this.isZooming()) { + this._zooming = true; + } + + if (this._finishTimeout) { + clearTimeout(this._finishTimeout); + delete this._finishTimeout; + } + + const pos = DOM.mousePos(this._el, e); + const tr = this._tr; + + this._around = LngLat.convert(this._aroundCenter ? tr.center : tr.unproject(pos)); + this._aroundPoint = tr.transform.locationPoint(this._around); + if (!this._frameId) { + this._frameId = true; + this._triggerRenderFrame(); + } + } + + renderFrame() { + if (!this._frameId) return; + this._frameId = null; + + if (!this.isActive()) return; + const tr = this._tr.transform; + + // if we've had scroll events since the last render frame, consume the + // accumulated delta, and update the target zoom level accordingly + if (this._delta !== 0) { + // For trackpad events and single mouse wheel ticks, use the default zoom rate + const zoomRate = (this._type === 'wheel' && Math.abs(this._delta) > wheelZoomDelta) ? this._wheelZoomRate : this._defaultZoomRate; + // Scale by sigmoid of scroll wheel delta. + let scale = maxScalePerFrame / (1 + Math.exp(-Math.abs(this._delta * zoomRate))); + + if (this._delta < 0 && scale !== 0) { + scale = 1 / scale; + } + + const fromScale = typeof this._targetZoom === 'number' ? tr.zoomScale(this._targetZoom) : tr.scale; + this._targetZoom = Math.min(tr.maxZoom, Math.max(tr.minZoom, tr.scaleZoom(fromScale * scale))); + + // if this is a mouse wheel, refresh the starting zoom and easing + // function we're using to smooth out the zooming between wheel + // events + if (this._type === 'wheel') { + this._startZoom = tr.zoom; + this._easing = this._smoothOutEasing(200); + } + + this._delta = 0; + } + + const targetZoom = typeof this._targetZoom === 'number' ? + this._targetZoom : tr.zoom; + const startZoom = this._startZoom; + const easing = this._easing; + + let finished = false; + let zoom; + if (this._type === 'wheel' && startZoom && easing) { + + const t = Math.min((browser.now() - this._lastWheelEventTime) / 200, 1); + const k = easing(t); + zoom = interpolates.number(startZoom, targetZoom, k); + if (t < 1) { + if (!this._frameId) { + this._frameId = true; + } + } else { + finished = true; + } + } else { + zoom = targetZoom; + finished = true; + } + + this._active = true; + + if (finished) { + this._active = false; + this._finishTimeout = setTimeout(() => { + this._zooming = false; + this._triggerRenderFrame(); + delete this._targetZoom; + delete this._finishTimeout; + }, 200); + } + + return { + noInertia: true, + needsRenderFrame: !finished, + zoomDelta: zoom - tr.zoom, + around: this._aroundPoint, + originalEvent: this._lastWheelEvent + }; + } + + _smoothOutEasing(duration: number) { + let easing = defaultEasing; + + if (this._prevEase) { + const currentEase = this._prevEase; + const t = (browser.now() - currentEase.start) / currentEase.duration; + const speed = currentEase.easing(t + 0.01) - currentEase.easing(t); + + // Quick hack to make new bezier that is continuous with last + const x = 0.27 / Math.sqrt(speed * speed + 0.0001) * 0.01; + const y = Math.sqrt(0.27 * 0.27 - x * x); + + easing = bezier(x, y, 0.25, 1); + } + + this._prevEase = { + start: browser.now(), + duration, + easing + }; + + return easing; + } + + reset() { + this._active = false; + this._zooming = false; + delete this._targetZoom; + if (this._finishTimeout) { + clearTimeout(this._finishTimeout); + delete this._finishTimeout; + } + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/shim/dblclick_zoom.ts b/web/libraries/maplibre-gl/src/ui/handler/shim/dblclick_zoom.ts new file mode 100644 index 00000000..d33dca28 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/shim/dblclick_zoom.ts @@ -0,0 +1,64 @@ +import type {ClickZoomHandler} from '../click_zoom'; +import type {TapZoomHandler} from './../tap_zoom'; + +/** + * The `DoubleClickZoomHandler` allows the user to zoom the map at a point by + * double clicking or double tapping. + * + * @group Handlers + */ +export class DoubleClickZoomHandler { + + _clickZoom: ClickZoomHandler; + _tapZoom: TapZoomHandler; + + /** @internal */ + constructor(clickZoom: ClickZoomHandler, TapZoom: TapZoomHandler) { + this._clickZoom = clickZoom; + this._tapZoom = TapZoom; + } + + /** + * Enables the "double click to zoom" interaction. + * + * @example + * ```ts + * map.doubleClickZoom.enable(); + * ``` + */ + enable() { + this._clickZoom.enable(); + this._tapZoom.enable(); + } + + /** + * Disables the "double click to zoom" interaction. + * + * @example + * ```ts + * map.doubleClickZoom.disable(); + * ``` + */ + disable() { + this._clickZoom.disable(); + this._tapZoom.disable(); + } + + /** + * Returns a Boolean indicating whether the "double click to zoom" interaction is enabled. + * + * @returns `true` if the "double click to zoom" interaction is enabled. + */ + isEnabled() { + return this._clickZoom.isEnabled() && this._tapZoom.isEnabled(); + } + + /** + * Returns a Boolean indicating whether the "double click to zoom" interaction is active, i.e. currently being used. + * + * @returns `true` if the "double click to zoom" interaction is active. + */ + isActive() { + return this._clickZoom.isActive() || this._tapZoom.isActive(); + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/shim/drag_pan.ts b/web/libraries/maplibre-gl/src/ui/handler/shim/drag_pan.ts new file mode 100644 index 00000000..c554a7fe --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/shim/drag_pan.ts @@ -0,0 +1,104 @@ +import type {MousePanHandler} from '../mouse'; +import type {TouchPanHandler} from './../touch_pan'; + +/** + * A {@link DragPanHandler} options object + */ +export type DragPanOptions = { + /** + * factor used to scale the drag velocity + * @defaultValue 0 + */ + linearity?: number; + /** + * easing function applled to `map.panTo` when applying the drag. + * @param t - the easing function + * @defaultValue bezier(0, 0, 0.3, 1) + */ + easing?: (t: number) => number; + /** + * the maximum value of the drag velocity. + * @defaultValue 1400 + */ + deceleration?: number; + /** + * the rate at which the speed reduces after the pan ends. + * @defaultValue 2500 + */ + maxSpeed?: number; +}; + +/** + * The `DragPanHandler` allows the user to pan the map by clicking and dragging + * the cursor. + * + * @group Handlers + */ +export class DragPanHandler { + + _el: HTMLElement; + _mousePan: MousePanHandler; + _touchPan: TouchPanHandler; + _inertiaOptions: DragPanOptions | boolean; + + /** @internal */ + constructor(el: HTMLElement, mousePan: MousePanHandler, touchPan: TouchPanHandler) { + this._el = el; + this._mousePan = mousePan; + this._touchPan = touchPan; + } + + /** + * Enables the "drag to pan" interaction. + * + * @param options - Options object + * @example + * ```ts + * map.dragPan.enable(); + * map.dragPan.enable({ + * linearity: 0.3, + * easing: bezier(0, 0, 0.3, 1), + * maxSpeed: 1400, + * deceleration: 2500, + * }); + * ``` + */ + enable(options?: DragPanOptions | boolean) { + this._inertiaOptions = options || {}; + this._mousePan.enable(); + this._touchPan.enable(); + this._el.classList.add('maplibregl-touch-drag-pan'); + } + + /** + * Disables the "drag to pan" interaction. + * + * @example + * ```ts + * map.dragPan.disable(); + * ``` + */ + disable() { + this._mousePan.disable(); + this._touchPan.disable(); + this._el.classList.remove('maplibregl-touch-drag-pan'); + } + + /** + * Returns a Boolean indicating whether the "drag to pan" interaction is enabled. + * + * @returns `true` if the "drag to pan" interaction is enabled. + */ + isEnabled() { + return this._mousePan.isEnabled() && this._touchPan.isEnabled(); + } + + /** + * Returns a Boolean indicating whether the "drag to pan" interaction is active, i.e. currently being used. + * + * @returns `true` if the "drag to pan" interaction is active. + */ + isActive() { + return this._mousePan.isActive() || this._touchPan.isActive(); + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/shim/drag_rotate.ts b/web/libraries/maplibre-gl/src/ui/handler/shim/drag_rotate.ts new file mode 100644 index 00000000..0aa35683 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/shim/drag_rotate.ts @@ -0,0 +1,73 @@ +import type {MousePitchHandler, MouseRotateHandler} from '../mouse'; + +export type DragRotateHandlerOptions = { + /** + * Control the map pitch in addition to the bearing + * @defaultValue true + */ + pitchWithRotate: boolean; +} + +/** + * The `DragRotateHandler` allows the user to rotate the map by clicking and + * dragging the cursor while holding the right mouse button or `ctrl` key. + * + * @group Handlers + */ +export class DragRotateHandler { + + _mouseRotate: MouseRotateHandler; + _mousePitch: MousePitchHandler; + _pitchWithRotate: boolean; + + /** @internal */ + constructor(options: DragRotateHandlerOptions, mouseRotate: MouseRotateHandler, mousePitch: MousePitchHandler) { + this._pitchWithRotate = options.pitchWithRotate; + this._mouseRotate = mouseRotate; + this._mousePitch = mousePitch; + } + + /** + * Enables the "drag to rotate" interaction. + * + * @example + * ```ts + * map.dragRotate.enable(); + * ``` + */ + enable() { + this._mouseRotate.enable(); + if (this._pitchWithRotate) this._mousePitch.enable(); + } + + /** + * Disables the "drag to rotate" interaction. + * + * @example + * ```ts + * map.dragRotate.disable(); + * ``` + */ + disable() { + this._mouseRotate.disable(); + this._mousePitch.disable(); + } + + /** + * Returns a Boolean indicating whether the "drag to rotate" interaction is enabled. + * + * @returns `true` if the "drag to rotate" interaction is enabled. + */ + isEnabled() { + return this._mouseRotate.isEnabled() && (!this._pitchWithRotate || this._mousePitch.isEnabled()); + } + + /** + * Returns a Boolean indicating whether the "drag to rotate" interaction is active, i.e. currently being used. + * + * @returns `true` if the "drag to rotate" interaction is active. + */ + isActive() { + return this._mouseRotate.isActive() || this._mousePitch.isActive(); + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/shim/two_fingers_touch.ts b/web/libraries/maplibre-gl/src/ui/handler/shim/two_fingers_touch.ts new file mode 100644 index 00000000..68ae7d27 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/shim/two_fingers_touch.ts @@ -0,0 +1,112 @@ +import type {TwoFingersTouchZoomHandler, TwoFingersTouchRotateHandler, AroundCenterOptions} from '../two_fingers_touch'; +import type {TapDragZoomHandler} from '../tap_drag_zoom'; + +/** + * The `TwoFingersTouchZoomRotateHandler` allows the user to zoom and rotate the map by + * pinching on a touchscreen. + * + * They can zoom with one finger by double tapping and dragging. On the second tap, + * hold the finger down and drag up or down to zoom in or out. + * + * @group Handlers + */ +export class TwoFingersTouchZoomRotateHandler { + + _el: HTMLElement; + _touchZoom: TwoFingersTouchZoomHandler; + _touchRotate: TwoFingersTouchRotateHandler; + _tapDragZoom: TapDragZoomHandler; + _rotationDisabled: boolean; + _enabled: boolean; + + /** @internal */ + constructor(el: HTMLElement, touchZoom: TwoFingersTouchZoomHandler, touchRotate: TwoFingersTouchRotateHandler, tapDragZoom: TapDragZoomHandler) { + this._el = el; + this._touchZoom = touchZoom; + this._touchRotate = touchRotate; + this._tapDragZoom = tapDragZoom; + this._rotationDisabled = false; + this._enabled = true; + } + + /** + * Enables the "pinch to rotate and zoom" interaction. + * + * @param options - Options object. + * + * @example + * ```ts + * map.touchZoomRotate.enable(); + * map.touchZoomRotate.enable({ around: 'center' }); + * ``` + */ + enable(options?: AroundCenterOptions | boolean | null) { + this._touchZoom.enable(options); + if (!this._rotationDisabled) this._touchRotate.enable(options); + this._tapDragZoom.enable(); + this._el.classList.add('maplibregl-touch-zoom-rotate'); + } + + /** + * Disables the "pinch to rotate and zoom" interaction. + * + * @example + * ```ts + * map.touchZoomRotate.disable(); + * ``` + */ + disable() { + this._touchZoom.disable(); + this._touchRotate.disable(); + this._tapDragZoom.disable(); + this._el.classList.remove('maplibregl-touch-zoom-rotate'); + } + + /** + * Returns a Boolean indicating whether the "pinch to rotate and zoom" interaction is enabled. + * + * @returns `true` if the "pinch to rotate and zoom" interaction is enabled. + */ + isEnabled() { + return this._touchZoom.isEnabled() && + (this._rotationDisabled || this._touchRotate.isEnabled()) && + this._tapDragZoom.isEnabled(); + } + + /** + * Returns true if the handler is enabled and has detected the start of a zoom/rotate gesture. + * + * @returns `true` if the handler is active, `false` otherwise + */ + isActive() { + return this._touchZoom.isActive() || this._touchRotate.isActive() || this._tapDragZoom.isActive(); + } + + /** + * Disables the "pinch to rotate" interaction, leaving the "pinch to zoom" + * interaction enabled. + * + * @example + * ```ts + * map.touchZoomRotate.disableRotation(); + * ``` + */ + disableRotation() { + this._rotationDisabled = true; + this._touchRotate.disable(); + } + + /** + * Enables the "pinch to rotate" interaction. + * + * @example + * ```ts + * map.touchZoomRotate.enable(); + * map.touchZoomRotate.enableRotation(); + * ``` + */ + enableRotation() { + this._rotationDisabled = false; + if (this._touchZoom.isEnabled()) this._touchRotate.enable(); + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/tap_drag_zoom.test.ts b/web/libraries/maplibre-gl/src/ui/handler/tap_drag_zoom.test.ts new file mode 100644 index 00000000..cc8eefec --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/tap_drag_zoom.test.ts @@ -0,0 +1,113 @@ +import {beforeMapTest} from '../../util/test/util'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {Map, MapOptions} from '../map'; + +function createMap() { + return new Map({container: window.document.createElement('div')} as any as MapOptions); +} + +function setupEvents(map: Map) { + const zoomstart = jest.fn(); + map.on('zoomstart', zoomstart); + + const zoom = jest.fn(); + map.on('zoom', zoom); + + const zoomend = jest.fn(); + map.on('zoomend', zoomend); + + return { + zoomstart, + zoom, + zoomend + }; +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('tap_drag_zoom', () => { + test('TapDragZoomHandler fires zoomstart, zoom, and zoomend at appropriate times in response to a double-tap and drag gesture', () => { + const map = createMap(); + const target = map.getCanvas(); + + const {zoomstart, zoom, zoomend} = setupEvents(map); + + const pointTouchOptions = { + touches: [{target, clientX: 100, clientY: 100}] + }; + + simulate.touchstart(target, pointTouchOptions); + simulate.touchend(target); + simulate.touchstart(target, pointTouchOptions); + map._renderTaskQueue.run(); + + expect(zoomstart).not.toHaveBeenCalled(); + expect(zoom).not.toHaveBeenCalled(); + expect(zoomend).not.toHaveBeenCalled(); + + simulate.touchmove(target, { + touches: [{target, clientX: 100, clientY: 110}] + }); + map._renderTaskQueue.run(); + + expect(zoomstart).toHaveBeenCalled(); + expect(zoom).toHaveBeenCalled(); + expect(zoomend).not.toHaveBeenCalled(); + + simulate.touchend(target); + map._renderTaskQueue.run(); + expect(zoomend).toHaveBeenCalled(); + + }); + + test('TapDragZoomHandler does not fire zoom on tap and drag if touchstart events are > 500ms apart', done => { + const map = createMap(); + const target = map.getCanvas(); + + const {zoomstart, zoom, zoomend} = setupEvents(map); + + const pointTouchOptions = { + touches: [{target, clientX: 100, clientY: 100}] + }; + + simulate.touchstart(target, pointTouchOptions); + simulate.touchend(target); + setTimeout(() => { + simulate.touchstart(target, pointTouchOptions); + simulate.touchmove(target, { + touches: [{target, clientX: 100, clientY: 110}] + }); + map._renderTaskQueue.run(); + + expect(zoomstart).not.toHaveBeenCalled(); + expect(zoom).not.toHaveBeenCalled(); + expect(zoomend).not.toHaveBeenCalled(); + done(); + }, 510); + }); + + test('TapDragZoomHandler does not zoom on double-tap and drag if touchstart events are in different locations (>30px apart)', () => { + const map = createMap(); + const target = map.getCanvas(); + + const {zoomstart, zoom, zoomend} = setupEvents(map); + + simulate.touchstart(target, { + touches: [{target, clientX: 100, clientY: 100}] + }); + simulate.touchend(target); + simulate.touchstart(target, { + touches: [{target, clientX: 140, clientY: 100}] + }); + simulate.touchmove(target, { + touches: [{target, clientX: 140, clientY: 110}] + }); + map._renderTaskQueue.run(); + + expect(zoomstart).not.toHaveBeenCalled(); + expect(zoom).not.toHaveBeenCalled(); + expect(zoomend).not.toHaveBeenCalled(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/tap_drag_zoom.ts b/web/libraries/maplibre-gl/src/ui/handler/tap_drag_zoom.ts new file mode 100644 index 00000000..59742da8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/tap_drag_zoom.ts @@ -0,0 +1,109 @@ +import {Handler} from '../handler_manager'; +import {TapRecognizer, MAX_TAP_INTERVAL, MAX_DIST} from './tap_recognizer'; +import type Point from '@mapbox/point-geometry'; + +export class TapDragZoomHandler implements Handler { + + _enabled: boolean; + _active: boolean; + _swipePoint: Point; + _swipeTouch: number; + _tapTime: number; + _tapPoint: Point; + _tap: TapRecognizer; + + constructor() { + + this._tap = new TapRecognizer({ + numTouches: 1, + numTaps: 1 + }); + + this.reset(); + } + + reset() { + this._active = false; + delete this._swipePoint; + delete this._swipeTouch; + delete this._tapTime; + delete this._tapPoint; + this._tap.reset(); + } + + touchstart(e: TouchEvent, points: Array, mapTouches: Array) { + if (this._swipePoint) return; + + if (!this._tapTime) { + this._tap.touchstart(e, points, mapTouches); + } else { + const swipePoint = points[0]; + + const soonEnough = e.timeStamp - this._tapTime < MAX_TAP_INTERVAL; + const closeEnough = this._tapPoint.dist(swipePoint) < MAX_DIST; + + if (!soonEnough || !closeEnough) { + this.reset(); + } else if (mapTouches.length > 0) { + this._swipePoint = swipePoint; + this._swipeTouch = mapTouches[0].identifier; + } + } + } + + touchmove(e: TouchEvent, points: Array, mapTouches: Array) { + if (!this._tapTime) { + this._tap.touchmove(e, points, mapTouches); + } else if (this._swipePoint) { + if (mapTouches[0].identifier !== this._swipeTouch) { + return; + } + + const newSwipePoint = points[0]; + const dist = newSwipePoint.y - this._swipePoint.y; + this._swipePoint = newSwipePoint; + + e.preventDefault(); + this._active = true; + + return { + zoomDelta: dist / 128 + }; + } + } + + touchend(e: TouchEvent, points: Array, mapTouches: Array) { + if (!this._tapTime) { + const point = this._tap.touchend(e, points, mapTouches); + if (point) { + this._tapTime = e.timeStamp; + this._tapPoint = point; + } + } else if (this._swipePoint) { + if (mapTouches.length === 0) { + this.reset(); + } + } + } + + touchcancel() { + this.reset(); + } + + enable() { + this._enabled = true; + } + + disable() { + this._enabled = false; + this.reset(); + } + + isEnabled() { + return this._enabled; + } + + isActive() { + return this._active; + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/tap_recognizer.ts b/web/libraries/maplibre-gl/src/ui/handler/tap_recognizer.ts new file mode 100644 index 00000000..b0c60c85 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/tap_recognizer.ts @@ -0,0 +1,138 @@ +import Point from '@mapbox/point-geometry'; +import {indexTouches} from './handler_util'; + +function getCentroid(points: Array) { + const sum = new Point(0, 0); + for (const point of points) { + sum._add(point); + } + return sum.div(points.length); +} + +export const MAX_TAP_INTERVAL = 500; +const MAX_TOUCH_TIME = 500; +export const MAX_DIST = 30; + +export class SingleTapRecognizer { + + numTouches: number; + centroid: Point; + startTime: number; + aborted: boolean; + touches: { + [k in number | string]: Point; + }; + + constructor(options: { + numTouches: number; + }) { + this.reset(); + this.numTouches = options.numTouches; + } + + reset() { + delete this.centroid; + delete this.startTime; + delete this.touches; + this.aborted = false; + } + + touchstart(e: TouchEvent, points: Array, mapTouches: Array) { + + if (this.centroid || mapTouches.length > this.numTouches) { + this.aborted = true; + } + if (this.aborted) { + return; + } + + if (this.startTime === undefined) { + this.startTime = e.timeStamp; + } + + if (mapTouches.length === this.numTouches) { + this.centroid = getCentroid(points); + this.touches = indexTouches(mapTouches, points); + } + } + + touchmove(e: TouchEvent, points: Array, mapTouches: Array) { + if (this.aborted || !this.centroid) return; + + const newTouches = indexTouches(mapTouches, points); + for (const id in this.touches) { + const prevPos = this.touches[id]; + const pos = newTouches[id]; + if (!pos || pos.dist(prevPos) > MAX_DIST) { + this.aborted = true; + } + } + } + + touchend(e: TouchEvent, points: Array, mapTouches: Array) { + if (!this.centroid || e.timeStamp - this.startTime > MAX_TOUCH_TIME) { + this.aborted = true; + } + + if (mapTouches.length === 0) { + const centroid = !this.aborted && this.centroid; + this.reset(); + if (centroid) return centroid; + } + } + +} + +export class TapRecognizer { + + singleTap: SingleTapRecognizer; + numTaps: number; + lastTime: number; + lastTap: Point; + count: number; + + constructor(options: { + numTaps: number; + numTouches: number; + }) { + this.singleTap = new SingleTapRecognizer(options); + this.numTaps = options.numTaps; + this.reset(); + } + + reset() { + this.lastTime = Infinity; + delete this.lastTap; + this.count = 0; + this.singleTap.reset(); + } + + touchstart(e: TouchEvent, points: Array, mapTouches: Array) { + this.singleTap.touchstart(e, points, mapTouches); + } + + touchmove(e: TouchEvent, points: Array, mapTouches: Array) { + this.singleTap.touchmove(e, points, mapTouches); + } + + touchend(e: TouchEvent, points: Array, mapTouches: Array) { + const tap = this.singleTap.touchend(e, points, mapTouches); + if (tap) { + const soonEnough = e.timeStamp - this.lastTime < MAX_TAP_INTERVAL; + const closeEnough = !this.lastTap || this.lastTap.dist(tap) < MAX_DIST; + + if (!soonEnough || !closeEnough) { + this.reset(); + } + + this.count++; + this.lastTime = e.timeStamp; + this.lastTap = tap; + + if (this.count === this.numTaps) { + this.reset(); + return tap; + } + } + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/tap_zoom.ts b/web/libraries/maplibre-gl/src/ui/handler/tap_zoom.ts new file mode 100644 index 00000000..03abe361 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/tap_zoom.ts @@ -0,0 +1,95 @@ +import {TapRecognizer} from './tap_recognizer'; +import type Point from '@mapbox/point-geometry'; +import type {Map} from '../map'; +import {TransformProvider} from './transform-provider'; +import {Handler} from '../handler_manager'; + +export class TapZoomHandler implements Handler { + _tr: TransformProvider; + _enabled: boolean; + _active: boolean; + _zoomIn: TapRecognizer; + _zoomOut: TapRecognizer; + + constructor(map: Map) { + this._tr = new TransformProvider(map); + this._zoomIn = new TapRecognizer({ + numTouches: 1, + numTaps: 2 + }); + + this._zoomOut = new TapRecognizer({ + numTouches: 2, + numTaps: 1 + }); + + this.reset(); + } + + reset() { + this._active = false; + this._zoomIn.reset(); + this._zoomOut.reset(); + } + + touchstart(e: TouchEvent, points: Array, mapTouches: Array) { + this._zoomIn.touchstart(e, points, mapTouches); + this._zoomOut.touchstart(e, points, mapTouches); + } + + touchmove(e: TouchEvent, points: Array, mapTouches: Array) { + this._zoomIn.touchmove(e, points, mapTouches); + this._zoomOut.touchmove(e, points, mapTouches); + } + + touchend(e: TouchEvent, points: Array, mapTouches: Array) { + const zoomInPoint = this._zoomIn.touchend(e, points, mapTouches); + const zoomOutPoint = this._zoomOut.touchend(e, points, mapTouches); + const tr = this._tr; + + if (zoomInPoint) { + this._active = true; + e.preventDefault(); + setTimeout(() => this.reset(), 0); + return { + cameraAnimation: (map: Map) => map.easeTo({ + duration: 300, + zoom: tr.zoom + 1, + around: tr.unproject(zoomInPoint) + }, {originalEvent: e}) + }; + } else if (zoomOutPoint) { + this._active = true; + e.preventDefault(); + setTimeout(() => this.reset(), 0); + return { + cameraAnimation: (map: Map) => map.easeTo({ + duration: 300, + zoom: tr.zoom - 1, + around: tr.unproject(zoomOutPoint) + }, {originalEvent: e}) + }; + } + } + + touchcancel() { + this.reset(); + } + + enable() { + this._enabled = true; + } + + disable() { + this._enabled = false; + this.reset(); + } + + isEnabled() { + return this._enabled; + } + + isActive() { + return this._active; + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/touch_pan.ts b/web/libraries/maplibre-gl/src/ui/handler/touch_pan.ts new file mode 100644 index 00000000..be3eff4f --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/touch_pan.ts @@ -0,0 +1,124 @@ +import Point from '@mapbox/point-geometry'; +import {indexTouches} from './handler_util'; +import type {Map} from '../map'; +import {GestureOptions} from '../map'; +import {Handler} from '../handler_manager'; + +export class TouchPanHandler implements Handler { + + _enabled: boolean; + _active: boolean; + _touches: { + [k in string | number]: Point; + }; + _minTouches: number; + _clickTolerance: number; + _sum: Point; + _map: Map; + _cancelCooperativeMessage: boolean; + + constructor(options: { + clickTolerance: number; + cooperativeGestures: boolean | GestureOptions; + }, map: Map) { + this._minTouches = options.cooperativeGestures ? 2 : 1; + this._clickTolerance = options.clickTolerance || 1; + this._map = map; + this.reset(); + } + + reset() { + this._active = false; + this._touches = {}; + this._sum = new Point(0, 0); + + // Put a delay on the cooperative gesture message so it's less twitchy + setTimeout(() => { + this._cancelCooperativeMessage = false; + }, 200); + } + + touchstart(e: TouchEvent, points: Array, mapTouches: Array) { + return this._calculateTransform(e, points, mapTouches); + } + + touchmove(e: TouchEvent, points: Array, mapTouches: Array) { + if (this._map._cooperativeGestures) { + if (this._minTouches === 2 && mapTouches.length < 2 && !this._cancelCooperativeMessage) { + // If coop gesture enabled, show panning info to user + this._map._onCooperativeGesture(e, false, mapTouches.length); + } else if (!this._cancelCooperativeMessage) { + // If user is successfully navigating, we don't need this warning until the touch resets + this._cancelCooperativeMessage = true; + } + } + if (!this._active || mapTouches.length < this._minTouches) return; + e.preventDefault(); + return this._calculateTransform(e, points, mapTouches); + } + + touchend(e: TouchEvent, points: Array, mapTouches: Array) { + this._calculateTransform(e, points, mapTouches); + + if (this._active && mapTouches.length < this._minTouches) { + this.reset(); + } + } + + touchcancel() { + this.reset(); + } + + _calculateTransform(e: TouchEvent, points: Array, mapTouches: Array) { + if (mapTouches.length > 0) this._active = true; + + const touches = indexTouches(mapTouches, points); + + const touchPointSum = new Point(0, 0); + const touchDeltaSum = new Point(0, 0); + let touchDeltaCount = 0; + + for (const identifier in touches) { + const point = touches[identifier]; + const prevPoint = this._touches[identifier]; + if (prevPoint) { + touchPointSum._add(point); + touchDeltaSum._add(point.sub(prevPoint)); + touchDeltaCount++; + touches[identifier] = point; + } + } + + this._touches = touches; + + if (touchDeltaCount < this._minTouches || !touchDeltaSum.mag()) return; + + const panDelta = touchDeltaSum.div(touchDeltaCount); + this._sum._add(panDelta); + if (this._sum.mag() < this._clickTolerance) return; + + const around = touchPointSum.div(touchDeltaCount); + + return { + around, + panDelta + }; + } + + enable() { + this._enabled = true; + } + + disable() { + this._enabled = false; + this.reset(); + } + + isEnabled() { + return this._enabled; + } + + isActive() { + return this._active; + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/transform-provider.ts b/web/libraries/maplibre-gl/src/ui/handler/transform-provider.ts new file mode 100644 index 00000000..87875248 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/transform-provider.ts @@ -0,0 +1,43 @@ +import type {Map} from '../map'; +import type {PointLike} from '../camera'; +import type {Transform} from '../../geo/transform'; +import Point from '@mapbox/point-geometry'; +import {LngLat} from '../../geo/lng_lat'; + +/** + * @internal + * Shared utilities for the Handler classes to access the correct camera state. + * If Camera.transformCameraUpdate is specified, the "desired state" of camera may differ from the state used for rendering. + * The handlers need the "desired state" to track accumulated changes. + */ +export class TransformProvider { + _map: Map; + + constructor(map: Map) { + this._map = map; + } + + get transform(): Transform { + return this._map._requestedCameraState || this._map.transform; + } + + get center() { + return {lng: this.transform.center.lng, lat: this.transform.center.lat}; + } + + get zoom() { + return this.transform.zoom; + } + + get pitch() { + return this.transform.pitch; + } + + get bearing() { + return this.transform.bearing; + } + + unproject(point: PointLike): LngLat { + return this.transform.pointLocation(Point.convert(point), this._map.terrain); + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler/two_fingers_touch.test.ts b/web/libraries/maplibre-gl/src/ui/handler/two_fingers_touch.test.ts new file mode 100644 index 00000000..8b0d5c74 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/two_fingers_touch.test.ts @@ -0,0 +1,283 @@ +import {Map, MapOptions} from '../map'; +import {Marker} from '../marker'; +import {DOM} from '../../util/dom'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {beforeMapTest} from '../../util/test/util'; + +function createMap() { + return new Map({container: DOM.create('div', '', window.document.body)} as any as MapOptions); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('touch zoom rotate', () => { + + test('TwoFingersTouchZoomRotateHandler fires zoomstart, zoom, and zoomend events at appropriate times in response to a pinch-zoom gesture', () => { + const map = createMap(); + const target = map.getCanvas(); + + const zoomstart = jest.fn(); + const zoom = jest.fn(); + const zoomend = jest.fn(); + + map.handlers._handlersById.tapZoom.disable(); + map.touchPitch.disable(); + map.on('zoomstart', zoomstart); + map.on('zoom', zoom); + map.on('zoomend', zoomend); + + simulate.touchstart(map.getCanvas(), {touches: [{target, identifier: 1, clientX: 0, clientY: -50}, {target, identifier: 2, clientX: 0, clientY: 50}]}); + map._renderTaskQueue.run(); + expect(zoomstart).toHaveBeenCalledTimes(0); + expect(zoom).toHaveBeenCalledTimes(0); + expect(zoomend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: [{target, identifier: 1, clientX: 0, clientY: -100}, {target, identifier: 2, clientX: 0, clientY: 100}]}); + map._renderTaskQueue.run(); + expect(zoomstart).toHaveBeenCalledTimes(1); + expect(zoom).toHaveBeenCalledTimes(1); + expect(zoomend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: [{target, identifier: 1, clientX: 0, clientY: -60}, {target, identifier: 2, clientX: 0, clientY: 60}]}); + map._renderTaskQueue.run(); + expect(zoomstart).toHaveBeenCalledTimes(1); + expect(zoom).toHaveBeenCalledTimes(2); + expect(zoomend).toHaveBeenCalledTimes(0); + + simulate.touchend(map.getCanvas(), {touches: []}); + map._renderTaskQueue.run(); + + // incremented because inertia starts a second zoom + expect(zoomstart).toHaveBeenCalledTimes(2); + map._renderTaskQueue.run(); + expect(zoom).toHaveBeenCalledTimes(3); + expect(zoomend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('TwoFingersTouchZoomRotateHandler fires rotatestart, rotate, and rotateend events at appropriate times in response to a pinch-rotate gesture', () => { + const map = createMap(); + const target = map.getCanvas(); + + const rotatestart = jest.fn(); + const rotate = jest.fn(); + const rotateend = jest.fn(); + + map.on('rotatestart', rotatestart); + map.on('rotate', rotate); + map.on('rotateend', rotateend); + + simulate.touchstart(map.getCanvas(), {touches: [{target, identifier: 0, clientX: 0, clientY: -50}, {target, identifier: 1, clientX: 0, clientY: 50}]}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(0); + expect(rotate).toHaveBeenCalledTimes(0); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: [{target, identifier: 0, clientX: -50, clientY: 0}, {target, identifier: 1, clientX: 50, clientY: 0}]}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(1); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: [{target, identifier: 0, clientX: 0, clientY: -50}, {target, identifier: 1, clientX: 0, clientY: 50}]}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(2); + expect(rotateend).toHaveBeenCalledTimes(0); + + simulate.touchend(map.getCanvas(), {touches: []}); + map._renderTaskQueue.run(); + expect(rotatestart).toHaveBeenCalledTimes(1); + expect(rotate).toHaveBeenCalledTimes(2); + expect(rotateend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('TwoFingersTouchZoomRotateHandler does not begin a gesture if preventDefault is called on the touchstart event', () => { + const map = createMap(); + const target = map.getCanvas(); + + map.on('touchstart', e => e.preventDefault()); + + const move = jest.fn(); + map.on('move', move); + + simulate.touchstart(map.getCanvas(), {touches: [{target, clientX: 0, clientY: 0}, {target, clientX: 5, clientY: 0}]}); + map._renderTaskQueue.run(); + + simulate.touchmove(map.getCanvas(), {touches: [{target, clientX: 0, clientY: 0}, {target, clientX: 0, clientY: 5}]}); + map._renderTaskQueue.run(); + + simulate.touchend(map.getCanvas(), {touches: []}); + map._renderTaskQueue.run(); + + expect(move).toHaveBeenCalledTimes(0); + + map.remove(); + }); + + test('TwoFingersTouchZoomRotateHandler starts zoom immediately when rotation disabled', () => { + const map = createMap(); + const target = map.getCanvas(); + map.touchZoomRotate.disableRotation(); + map.handlers._handlersById.tapZoom.disable(); + + const zoomstart = jest.fn(); + const zoom = jest.fn(); + const zoomend = jest.fn(); + + map.on('zoomstart', zoomstart); + map.on('zoom', zoom); + map.on('zoomend', zoomend); + + simulate.touchstart(map.getCanvas(), {touches: [{target, identifier: 0, clientX: 0, clientY: -5}, {target, identifier: 2, clientX: 0, clientY: 5}]}); + map._renderTaskQueue.run(); + expect(zoomstart).toHaveBeenCalledTimes(0); + expect(zoom).toHaveBeenCalledTimes(0); + expect(zoomend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: [{target, identifier: 0, clientX: 0, clientY: -5}, {target, identifier: 2, clientX: 0, clientY: 6}]}); + map._renderTaskQueue.run(); + expect(zoomstart).toHaveBeenCalledTimes(1); + expect(zoom).toHaveBeenCalledTimes(1); + expect(zoomend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: [{target, identifier: 0, clientX: 0, clientY: -5}, {target, identifier: 2, clientX: 0, clientY: 4}]}); + map._renderTaskQueue.run(); + expect(zoomstart).toHaveBeenCalledTimes(1); + expect(zoom).toHaveBeenCalledTimes(2); + expect(zoomend).toHaveBeenCalledTimes(0); + + simulate.touchend(map.getCanvas(), {touches: []}); + map._renderTaskQueue.run(); + // incremented because inertia starts a second zoom + expect(zoomstart).toHaveBeenCalledTimes(2); + map._renderTaskQueue.run(); + expect(zoom).toHaveBeenCalledTimes(3); + expect(zoomend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('TwoFingersTouchZoomRotateHandler adds css class used for disabling default touch behavior in some browsers', () => { + const map = createMap(); + + const className = 'maplibregl-touch-zoom-rotate'; + expect(map.getCanvasContainer().classList.contains(className)).toBeTruthy(); + map.touchZoomRotate.disable(); + expect(map.getCanvasContainer().classList.contains(className)).toBeFalsy(); + map.touchZoomRotate.enable(); + expect(map.getCanvasContainer().classList.contains(className)).toBeTruthy(); + }); + + test('TwoFingersTouchZoomRotateHandler zooms when touching two markers on the same map', () => { + const map = createMap(); + + const marker1 = new Marker() + .setLngLat([0, 0]) + .addTo(map); + const marker2 = new Marker() + .setLngLat([0, 0]) + .addTo(map); + const target1 = marker1.getElement(); + const target2 = marker2.getElement(); + + const zoomstart = jest.fn(); + const zoom = jest.fn(); + const zoomend = jest.fn(); + + map.handlers._handlersById.tapZoom.disable(); + map.touchPitch.disable(); + map.on('zoomstart', zoomstart); + map.on('zoom', zoom); + map.on('zoomend', zoomend); + + simulate.touchstart(map.getCanvas(), {touches: [{target: target1, identifier: 1, clientX: 0, clientY: -50}]}); + simulate.touchstart(map.getCanvas(), {touches: [{target: target1, identifier: 1, clientX: 0, clientY: -50}, {target: target2, identifier: 2, clientX: 0, clientY: 50}]}); + map._renderTaskQueue.run(); + expect(zoomstart).toHaveBeenCalledTimes(0); + expect(zoom).toHaveBeenCalledTimes(0); + expect(zoomend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: [{target: target1, identifier: 1, clientX: 0, clientY: -100}, {target: target2, identifier: 2, clientX: 0, clientY: 100}]}); + map._renderTaskQueue.run(); + expect(zoomstart).toHaveBeenCalledTimes(1); + expect(zoom).toHaveBeenCalledTimes(1); + expect(zoomend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: [{target: target1, identifier: 1, clientX: 0, clientY: -60}, {target: target2, identifier: 2, clientX: 0, clientY: 60}]}); + map._renderTaskQueue.run(); + expect(zoomstart).toHaveBeenCalledTimes(1); + expect(zoom).toHaveBeenCalledTimes(2); + expect(zoomend).toHaveBeenCalledTimes(0); + + simulate.touchend(map.getCanvas(), {touches: []}); + map._renderTaskQueue.run(); + + // incremented because inertia starts a second zoom + expect(zoomstart).toHaveBeenCalledTimes(2); + map._renderTaskQueue.run(); + expect(zoom).toHaveBeenCalledTimes(3); + expect(zoomend).toHaveBeenCalledTimes(1); + + map.remove(); + }); + + test('TwoFingersTouchZoomRotateHandler does not zoom when touching an element not on the map', () => { + const map = createMap(); + + const marker1 = new Marker() + .setLngLat([0, 0]) + .addTo(map); + const marker2 = new Marker() + .setLngLat([0, 0]); + + const target1 = marker1.getElement(); // on map + const target2 = marker2.getElement(); // not on map + + const zoomstart = jest.fn(); + const zoom = jest.fn(); + const zoomend = jest.fn(); + + map.handlers._handlersById.tapZoom.disable(); + map.touchPitch.disable(); + map.dragPan.disable(); + map.on('zoomstart', zoomstart); + map.on('zoom', zoom); + map.on('zoomend', zoomend); + + simulate.touchstart(map.getCanvas(), {touches: [{target: target1, identifier: 1, clientX: 0, clientY: -50}]}); + simulate.touchstart(map.getCanvas(), {touches: [{target: target1, identifier: 1, clientX: 0, clientY: -50}, {target: target2, identifier: 2, clientX: 0, clientY: 50}]}); + map._renderTaskQueue.run(); + expect(zoomstart).toHaveBeenCalledTimes(0); + expect(zoom).toHaveBeenCalledTimes(0); + expect(zoomend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: [{target: target1, identifier: 1, clientX: 0, clientY: -100}, {target: target2, identifier: 2, clientX: 0, clientY: 100}]}); + map._renderTaskQueue.run(); + expect(zoomstart).toHaveBeenCalledTimes(0); + expect(zoom).toHaveBeenCalledTimes(0); + expect(zoomend).toHaveBeenCalledTimes(0); + + simulate.touchmove(map.getCanvas(), {touches: [{target: target1, identifier: 1, clientX: 0, clientY: -60}, {target: target2, identifier: 2, clientX: 0, clientY: 60}]}); + map._renderTaskQueue.run(); + expect(zoomstart).toHaveBeenCalledTimes(0); + expect(zoom).toHaveBeenCalledTimes(0); + expect(zoomend).toHaveBeenCalledTimes(0); + + simulate.touchend(map.getCanvas(), {touches: []}); + map._renderTaskQueue.run(); + + // incremented because inertia starts a second zoom + expect(zoomstart).toHaveBeenCalledTimes(0); + map._renderTaskQueue.run(); + expect(zoom).toHaveBeenCalledTimes(0); + expect(zoomend).toHaveBeenCalledTimes(0); + + map.remove(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/handler/two_fingers_touch.ts b/web/libraries/maplibre-gl/src/ui/handler/two_fingers_touch.ts new file mode 100644 index 00000000..a29fc036 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler/two_fingers_touch.ts @@ -0,0 +1,337 @@ +import Point from '@mapbox/point-geometry'; +import {DOM} from '../../util/dom'; +import type {Map} from '../map'; +import {Handler} from '../handler_manager'; + +/** + * An options object sent to the enable function of some of the handlers + */ +export type AroundCenterOptions = { + /** + * If "center" is passed, map will zoom around the center of map + */ + around: 'center'; +} + +/** + * The `TwoFingersTouchHandler`s allows the user to zoom, pitch and rotate the map using two fingers + * + * @group Handlers + */ +abstract class TwoFingersTouchHandler implements Handler { + + _enabled: boolean; + _active: boolean; + _firstTwoTouches: [number, number]; + _vector: Point; + _startVector: Point; + _aroundCenter: boolean; + + /** @internal */ + constructor() { + this.reset(); + } + + reset() { + this._active = false; + delete this._firstTwoTouches; + } + + abstract _start(points: [Point, Point]); + abstract _move(points: [Point, Point], pinchAround: Point, e: TouchEvent); + + touchstart(e: TouchEvent, points: Array, mapTouches: Array) { + //log('touchstart', points, e.target.innerHTML, e.targetTouches.length ? e.targetTouches[0].target.innerHTML: undefined); + if (this._firstTwoTouches || mapTouches.length < 2) return; + + this._firstTwoTouches = [ + mapTouches[0].identifier, + mapTouches[1].identifier + ]; + + // implemented by child classes + this._start([points[0], points[1]]); + } + + touchmove(e: TouchEvent, points: Array, mapTouches: Array) { + if (!this._firstTwoTouches) return; + + e.preventDefault(); + + const [idA, idB] = this._firstTwoTouches; + const a = getTouchById(mapTouches, points, idA); + const b = getTouchById(mapTouches, points, idB); + if (!a || !b) return; + const pinchAround = this._aroundCenter ? null : a.add(b).div(2); + + // implemented by child classes + return this._move([a, b], pinchAround, e); + + } + + touchend(e: TouchEvent, points: Array, mapTouches: Array) { + if (!this._firstTwoTouches) return; + + const [idA, idB] = this._firstTwoTouches; + const a = getTouchById(mapTouches, points, idA); + const b = getTouchById(mapTouches, points, idB); + if (a && b) return; + + if (this._active) DOM.suppressClick(); + + this.reset(); + } + + touchcancel() { + this.reset(); + } + + /** + * Enables the "drag to pitch" interaction. + * + * @example + * ```ts + * map.touchPitch.enable(); + * ``` + */ + enable(options?: AroundCenterOptions | boolean | null) { + this._enabled = true; + this._aroundCenter = !!options && (options as AroundCenterOptions).around === 'center'; + } + + /** + * Disables the "drag to pitch" interaction. + * + * @example + * ```ts + * map.touchPitch.disable(); + * ``` + */ + disable() { + this._enabled = false; + this.reset(); + } + + /** + * Returns a Boolean indicating whether the "drag to pitch" interaction is enabled. + * + * @returns `true` if the "drag to pitch" interaction is enabled. + */ + isEnabled() { + return this._enabled; + } + + /** + * Returns a Boolean indicating whether the "drag to pitch" interaction is active, i.e. currently being used. + * + * @returns `true` if the "drag to pitch" interaction is active. + */ + isActive() { + return this._active; + } +} + +function getTouchById(mapTouches: Array, points: Array, identifier: number) { + for (let i = 0; i < mapTouches.length; i++) { + if (mapTouches[i].identifier === identifier) return points[i]; + } +} + +/* ZOOM */ + +const ZOOM_THRESHOLD = 0.1; + +function getZoomDelta(distance, lastDistance) { + return Math.log(distance / lastDistance) / Math.LN2; +} + +/** + * The `TwoFingersTouchHandler`s allows the user to zoom the map two fingers + * + * @group Handlers + */ +export class TwoFingersTouchZoomHandler extends TwoFingersTouchHandler { + + _distance: number; + _startDistance: number; + + reset() { + super.reset(); + delete this._distance; + delete this._startDistance; + } + + _start(points: [Point, Point]) { + this._startDistance = this._distance = points[0].dist(points[1]); + } + + _move(points: [Point, Point], pinchAround: Point) { + const lastDistance = this._distance; + this._distance = points[0].dist(points[1]); + if (!this._active && Math.abs(getZoomDelta(this._distance, this._startDistance)) < ZOOM_THRESHOLD) return; + this._active = true; + return { + zoomDelta: getZoomDelta(this._distance, lastDistance), + pinchAround + }; + } +} + +/* ROTATE */ + +const ROTATION_THRESHOLD = 25; // pixels along circumference of touch circle + +function getBearingDelta(a, b) { + return a.angleWith(b) * 180 / Math.PI; +} + +/** + * The `TwoFingersTouchHandler`s allows the user to rotate the map two fingers + * + * @group Handlers + */ +export class TwoFingersTouchRotateHandler extends TwoFingersTouchHandler { + _minDiameter: number; + + reset() { + super.reset(); + delete this._minDiameter; + delete this._startVector; + delete this._vector; + } + + _start(points: [Point, Point]) { + this._startVector = this._vector = points[0].sub(points[1]); + this._minDiameter = points[0].dist(points[1]); + } + + _move(points: [Point, Point], pinchAround: Point) { + const lastVector = this._vector; + this._vector = points[0].sub(points[1]); + + if (!this._active && this._isBelowThreshold(this._vector)) return; + this._active = true; + + return { + bearingDelta: getBearingDelta(this._vector, lastVector), + pinchAround + }; + } + + _isBelowThreshold(vector: Point) { + /* + * The threshold before a rotation actually happens is configured in + * pixels along the circumference of the circle formed by the two fingers. + * This makes the threshold in degrees larger when the fingers are close + * together and smaller when the fingers are far apart. + * + * Use the smallest diameter from the whole gesture to reduce sensitivity + * when pinching in and out. + */ + + this._minDiameter = Math.min(this._minDiameter, vector.mag()); + const circumference = Math.PI * this._minDiameter; + const threshold = ROTATION_THRESHOLD / circumference * 360; + + const bearingDeltaSinceStart = getBearingDelta(vector, this._startVector); + return Math.abs(bearingDeltaSinceStart) < threshold; + } +} + +/* PITCH */ + +function isVertical(vector) { + return Math.abs(vector.y) > Math.abs(vector.x); +} + +const ALLOWED_SINGLE_TOUCH_TIME = 100; + +/** + * The `TwoFingersTouchPitchHandler` allows the user to pitch the map by dragging up and down with two fingers. + * + * @group Handlers + */ +export class TwoFingersTouchPitchHandler extends TwoFingersTouchHandler { + + _valid: boolean | void; + _firstMove: number; + _lastPoints: [Point, Point]; + _map: Map; + _currentTouchCount: number; + + constructor(map: Map) { + super(); + this._map = map; + } + + reset() { + super.reset(); + this._valid = undefined; + delete this._firstMove; + delete this._lastPoints; + } + + touchstart(e: TouchEvent, points: Array, mapTouches: Array) { + super.touchstart(e, points, mapTouches); + this._currentTouchCount = mapTouches.length; + } + + _start(points: [Point, Point]) { + this._lastPoints = points; + if (isVertical(points[0].sub(points[1]))) { + // fingers are more horizontal than vertical + this._valid = false; + + } + } + + _move(points: [Point, Point], center: Point, e: TouchEvent) { + // If cooperative gestures is enabled, we need a 3-finger minimum for this gesture to register + if (this._map._cooperativeGestures && this._currentTouchCount < 3) { + return; + } + + const vectorA = points[0].sub(this._lastPoints[0]); + const vectorB = points[1].sub(this._lastPoints[1]); + + this._valid = this.gestureBeginsVertically(vectorA, vectorB, e.timeStamp); + if (!this._valid) return; + + this._lastPoints = points; + this._active = true; + const yDeltaAverage = (vectorA.y + vectorB.y) / 2; + const degreesPerPixelMoved = -0.5; + return { + pitchDelta: yDeltaAverage * degreesPerPixelMoved + }; + } + + gestureBeginsVertically(vectorA: Point, vectorB: Point, timeStamp: number) { + if (this._valid !== undefined) return this._valid; + + const threshold = 2; + const movedA = vectorA.mag() >= threshold; + const movedB = vectorB.mag() >= threshold; + + // neither finger has moved a meaningful amount, wait + if (!movedA && !movedB) return; + + // One finger has moved and the other has not. + // If enough time has passed, decide it is not a pitch. + if (!movedA || !movedB) { + if (this._firstMove === undefined) { + this._firstMove = timeStamp; + } + + if (timeStamp - this._firstMove < ALLOWED_SINGLE_TOUCH_TIME) { + // still waiting for a movement from the second finger + return undefined; + } else { + return false; + } + } + + const isSameDirection = vectorA.y > 0 === vectorB.y > 0; + return isVertical(vectorA) && isVertical(vectorB) && isSameDirection; + } +} diff --git a/web/libraries/maplibre-gl/src/ui/handler_inertia.ts b/web/libraries/maplibre-gl/src/ui/handler_inertia.ts new file mode 100644 index 00000000..e68e7ccf --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler_inertia.ts @@ -0,0 +1,159 @@ +import {browser} from '../util/browser'; +import type {Map} from './map'; +import {bezier, clamp, extend} from '../util/util'; +import Point from '@mapbox/point-geometry'; +import type {DragPanOptions} from './handler/shim/drag_pan'; + +const defaultInertiaOptions = { + linearity: 0.3, + easing: bezier(0, 0, 0.3, 1), +}; + +const defaultPanInertiaOptions = extend({ + deceleration: 2500, + maxSpeed: 1400 +}, defaultInertiaOptions); + +const defaultZoomInertiaOptions = extend({ + deceleration: 20, + maxSpeed: 1400 +}, defaultInertiaOptions); + +const defaultBearingInertiaOptions = extend({ + deceleration: 1000, + maxSpeed: 360 +}, defaultInertiaOptions); + +const defaultPitchInertiaOptions = extend({ + deceleration: 1000, + maxSpeed: 90 +}, defaultInertiaOptions); + +export type InertiaOptions = { + linearity: number; + easing: (t: number) => number; + deceleration: number; + maxSpeed: number; +}; + +export type InputEvent = MouseEvent | TouchEvent | KeyboardEvent | WheelEvent; + +export class HandlerInertia { + _map: Map; + _inertiaBuffer: Array<{ + time: number; + settings: any; + }>; + + constructor(map: Map) { + this._map = map; + this.clear(); + } + + clear() { + this._inertiaBuffer = []; + } + + record(settings: any) { + this._drainInertiaBuffer(); + this._inertiaBuffer.push({time: browser.now(), settings}); + } + + _drainInertiaBuffer() { + const inertia = this._inertiaBuffer, + now = browser.now(), + cutoff = 160; //msec + + while (inertia.length > 0 && now - inertia[0].time > cutoff) + inertia.shift(); + } + + _onMoveEnd(panInertiaOptions?: DragPanOptions | boolean) { + this._drainInertiaBuffer(); + if (this._inertiaBuffer.length < 2) { + return; + } + + const deltas = { + zoom: 0, + bearing: 0, + pitch: 0, + pan: new Point(0, 0), + pinchAround: undefined, + around: undefined + }; + + for (const {settings} of this._inertiaBuffer) { + deltas.zoom += settings.zoomDelta || 0; + deltas.bearing += settings.bearingDelta || 0; + deltas.pitch += settings.pitchDelta || 0; + if (settings.panDelta) deltas.pan._add(settings.panDelta); + if (settings.around) deltas.around = settings.around; + if (settings.pinchAround) deltas.pinchAround = settings.pinchAround; + } + + const lastEntry = this._inertiaBuffer[this._inertiaBuffer.length - 1]; + const duration = (lastEntry.time - this._inertiaBuffer[0].time); + + const easeOptions = {} as any; + + if (deltas.pan.mag()) { + const result = calculateEasing(deltas.pan.mag(), duration, extend({}, defaultPanInertiaOptions, panInertiaOptions || {})); + easeOptions.offset = deltas.pan.mult(result.amount / deltas.pan.mag()); + easeOptions.center = this._map.transform.center; + extendDuration(easeOptions, result); + } + + if (deltas.zoom) { + const result = calculateEasing(deltas.zoom, duration, defaultZoomInertiaOptions); + easeOptions.zoom = this._map.transform.zoom + result.amount; + extendDuration(easeOptions, result); + } + + if (deltas.bearing) { + const result = calculateEasing(deltas.bearing, duration, defaultBearingInertiaOptions); + easeOptions.bearing = this._map.transform.bearing + clamp(result.amount, -179, 179); + extendDuration(easeOptions, result); + } + + if (deltas.pitch) { + const result = calculateEasing(deltas.pitch, duration, defaultPitchInertiaOptions); + easeOptions.pitch = this._map.transform.pitch + result.amount; + extendDuration(easeOptions, result); + } + + if (easeOptions.zoom || easeOptions.bearing) { + const last = deltas.pinchAround === undefined ? deltas.around : deltas.pinchAround; + easeOptions.around = last ? this._map.unproject(last) : this._map.getCenter(); + } + + this.clear(); + return extend(easeOptions, { + noMoveStart: true + }); + + } +} + +// Unfortunately zoom, bearing, etc can't have different durations and easings so +// we need to choose one. We use the longest duration and it's corresponding easing. +function extendDuration(easeOptions, result) { + if (!easeOptions.duration || easeOptions.duration < result.duration) { + easeOptions.duration = result.duration; + easeOptions.easing = result.easing; + } +} + +function calculateEasing(amount, inertiaDuration: number, inertiaOptions) { + const {maxSpeed, linearity, deceleration} = inertiaOptions; + const speed = clamp( + amount * linearity / (inertiaDuration / 1000), + -maxSpeed, + maxSpeed); + const duration = Math.abs(speed) / (deceleration * linearity); + return { + easing: inertiaOptions.easing, + duration: duration * 1000, + amount: speed * (duration / 2) + }; +} diff --git a/web/libraries/maplibre-gl/src/ui/handler_manager.ts b/web/libraries/maplibre-gl/src/ui/handler_manager.ts new file mode 100644 index 00000000..6db05794 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/handler_manager.ts @@ -0,0 +1,622 @@ +import {Event} from '../util/evented'; +import {DOM} from '../util/dom'; +import {Map, CompleteMapOptions} from './map'; +import {HandlerInertia} from './handler_inertia'; +import {MapEventHandler, BlockableMapEventHandler} from './handler/map_event'; +import {BoxZoomHandler} from './handler/box_zoom'; +import {TapZoomHandler} from './handler/tap_zoom'; +import {generateMouseRotationHandler, generateMousePitchHandler, generateMousePanHandler} from './handler/mouse'; +import {TouchPanHandler} from './handler/touch_pan'; +import {TwoFingersTouchZoomHandler, TwoFingersTouchRotateHandler, TwoFingersTouchPitchHandler} from './handler/two_fingers_touch'; +import {KeyboardHandler} from './handler/keyboard'; +import {ScrollZoomHandler} from './handler/scroll_zoom'; +import {DoubleClickZoomHandler} from './handler/shim/dblclick_zoom'; +import {ClickZoomHandler} from './handler/click_zoom'; +import {TapDragZoomHandler} from './handler/tap_drag_zoom'; +import {DragPanHandler} from './handler/shim/drag_pan'; +import {DragRotateHandler} from './handler/shim/drag_rotate'; +import {TwoFingersTouchZoomRotateHandler} from './handler/shim/two_fingers_touch'; +import {extend} from '../util/util'; +import {browser} from '../util/browser'; +import Point from '@mapbox/point-geometry'; + +export type InputEvent = MouseEvent | TouchEvent | KeyboardEvent | WheelEvent; + +const isMoving = p => p.zoom || p.drag || p.pitch || p.rotate; + +class RenderFrameEvent extends Event { + type: 'renderFrame'; + timeStamp: number; +} + +/** + * Handlers interpret dom events and return camera changes that should be + * applied to the map (`HandlerResult`s). The camera changes are all deltas. + * The handler itself should have no knowledge of the map's current state. + * This makes it easier to merge multiple results and keeps handlers simpler. + * For example, if there is a mousedown and mousemove, the mousePan handler + * would return a `panDelta` on the mousemove. + */ +export interface Handler { + enable(): void; + disable(): void; + isEnabled(): boolean; + isActive(): boolean; + /** + * `reset` can be called by the manager at any time and must reset everything to it's original state + */ + reset(): void; + // Handlers can optionally implement these methods. + // They are called with dom events whenever those dom evens are received. + readonly touchstart?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void; + readonly touchmove?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void; + readonly touchmoveWindow?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void; + readonly touchend?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void; + readonly touchcancel?: (e: TouchEvent, points: Array, mapTouches: Array) => HandlerResult | void; + readonly mousedown?: (e: MouseEvent, point: Point) => HandlerResult | void; + readonly mousemove?: (e: MouseEvent, point: Point) => HandlerResult | void; + readonly mousemoveWindow?: (e: MouseEvent, point: Point) => HandlerResult | void; + readonly mouseup?: (e: MouseEvent, point: Point) => HandlerResult | void; + readonly mouseupWindow?: (e: MouseEvent, point: Point) => HandlerResult | void; + readonly dblclick?: (e: MouseEvent, point: Point) => HandlerResult | void; + readonly contextmenu?: (e: MouseEvent) => HandlerResult | void; + readonly wheel?: (e: WheelEvent, point: Point) => HandlerResult | void; + readonly keydown?: (e: KeyboardEvent) => HandlerResult | void; + readonly keyup?: (e: KeyboardEvent) => HandlerResult | void; + /** + * `renderFrame` is the only non-dom event. It is called during render + * frames and can be used to smooth camera changes (see scroll handler). + */ + readonly renderFrame?: () => HandlerResult | void; +} + +/** + * All handler methods that are called with events can optionally return a `HandlerResult`. + */ +export type HandlerResult = { + panDelta?: Point; + zoomDelta?: number; + bearingDelta?: number; + pitchDelta?: number; + /** + * the point to not move when changing the camera + */ + around?: Point | null; + /** + * same as above, except for pinch actions, which are given higher priority + */ + pinchAround?: Point | null; + /** + * A method that can fire a one-off easing by directly changing the map's camera. + */ + cameraAnimation?: (map: Map) => any; + /** + * The last three properties are needed by only one handler: scrollzoom. + * The DOM event to be used as the `originalEvent` on any camera change events. + */ + originalEvent?: Event; + /** + * Makes the manager trigger a frame, allowing the handler to return multiple results over time (see scrollzoom). + */ + needsRenderFrame?: boolean; + /** + * The camera changes won't get recorded for inertial zooming. + */ + noInertia?: boolean; +}; + +export type EventInProgress = { + handlerName: string; + originalEvent: Event; +} + +export type EventsInProgress = { + zoom?: EventInProgress; + pitch?: EventInProgress; + rotate?: EventInProgress; + drag?: EventInProgress; +} + +function hasChange(result: HandlerResult) { + return (result.panDelta && result.panDelta.mag()) || result.zoomDelta || result.bearingDelta || result.pitchDelta; +} + +export class HandlerManager { + _map: Map; + _el: HTMLElement; + _handlers: Array<{ + handlerName: string; + handler: Handler; + allowed: Array; + }>; + _eventsInProgress: EventsInProgress; + _frameId: number; + _inertia: HandlerInertia; + _bearingSnap: number; + _handlersById: {[x: string]: Handler}; + _updatingCamera: boolean; + _changes: Array<[HandlerResult, EventsInProgress, {[handlerName: string]: Event}]>; + _terrainMovement: boolean; + _zoom: {handlerName: string}; + _previousActiveHandlers: {[x: string]: Handler}; + _listeners: Array<[Window | Document | HTMLElement, string, { + passive?: boolean; + capture?: boolean; + } | undefined]>; + + constructor(map: Map, options: CompleteMapOptions) { + this._map = map; + this._el = this._map.getCanvasContainer(); + this._handlers = []; + this._handlersById = {}; + this._changes = []; + + this._inertia = new HandlerInertia(map); + this._bearingSnap = options.bearingSnap; + this._previousActiveHandlers = {}; + + // Track whether map is currently moving, to compute start/move/end events + this._eventsInProgress = {}; + + this._addDefaultHandlers(options); + + const el = this._el; + + this._listeners = [ + // This needs to be `passive: true` so that a double tap fires two + // pairs of touchstart/end events in iOS Safari 13. If this is set to + // `passive: false` then the second pair of events is only fired if + // preventDefault() is called on the first touchstart. Calling preventDefault() + // undesirably prevents click events. + [el, 'touchstart', {passive: true}], + // This needs to be `passive: false` so that scrolls and pinches can be + // prevented in browsers that don't support `touch-actions: none`, for example iOS Safari 12. + [el, 'touchmove', {passive: false}], + [el, 'touchend', undefined], + [el, 'touchcancel', undefined], + + [el, 'mousedown', undefined], + [el, 'mousemove', undefined], + [el, 'mouseup', undefined], + + // Bind window-level event listeners for move and up/end events. In the absence of + // the pointer capture API, which is not supported by all necessary platforms, + // window-level event listeners give us the best shot at capturing events that + // fall outside the map canvas element. Use `{capture: true}` for the move event + // to prevent map move events from being fired during a drag. + [document, 'mousemove', {capture: true}], + [document, 'mouseup', undefined], + + [el, 'mouseover', undefined], + [el, 'mouseout', undefined], + [el, 'dblclick', undefined], + [el, 'click', undefined], + + [el, 'keydown', {capture: false}], + [el, 'keyup', undefined], + + [el, 'wheel', {passive: false}], + [el, 'contextmenu', undefined], + + [window, 'blur', undefined] + ]; + + for (const [target, type, listenerOptions] of this._listeners) { + DOM.addEventListener(target, type, target === document ? this.handleWindowEvent : this.handleEvent, listenerOptions); + } + } + + destroy() { + for (const [target, type, listenerOptions] of this._listeners) { + DOM.removeEventListener(target, type, target === document ? this.handleWindowEvent : this.handleEvent, listenerOptions); + } + } + + _addDefaultHandlers(options: CompleteMapOptions) { + const map = this._map; + const el = map.getCanvasContainer(); + this._add('mapEvent', new MapEventHandler(map, options)); + + const boxZoom = map.boxZoom = new BoxZoomHandler(map, options); + this._add('boxZoom', boxZoom); + if (options.interactive && options.boxZoom) { + boxZoom.enable(); + } + + const tapZoom = new TapZoomHandler(map); + const clickZoom = new ClickZoomHandler(map); + map.doubleClickZoom = new DoubleClickZoomHandler(clickZoom, tapZoom); + this._add('tapZoom', tapZoom); + this._add('clickZoom', clickZoom); + if (options.interactive && options.doubleClickZoom) { + map.doubleClickZoom.enable(); + } + + const tapDragZoom = new TapDragZoomHandler(); + this._add('tapDragZoom', tapDragZoom); + + const touchPitch = map.touchPitch = new TwoFingersTouchPitchHandler(map); + this._add('touchPitch', touchPitch); + if (options.interactive && options.touchPitch) { + map.touchPitch.enable(options.touchPitch); + } + + const mouseRotate = generateMouseRotationHandler(options); + const mousePitch = generateMousePitchHandler(options); + map.dragRotate = new DragRotateHandler(options, mouseRotate, mousePitch); + this._add('mouseRotate', mouseRotate, ['mousePitch']); + this._add('mousePitch', mousePitch, ['mouseRotate']); + if (options.interactive && options.dragRotate) { + map.dragRotate.enable(); + } + + const mousePan = generateMousePanHandler(options); + const touchPan = new TouchPanHandler(options, map); + map.dragPan = new DragPanHandler(el, mousePan, touchPan); + this._add('mousePan', mousePan); + this._add('touchPan', touchPan, ['touchZoom', 'touchRotate']); + if (options.interactive && options.dragPan) { + map.dragPan.enable(options.dragPan); + } + + const touchRotate = new TwoFingersTouchRotateHandler(); + const touchZoom = new TwoFingersTouchZoomHandler(); + map.touchZoomRotate = new TwoFingersTouchZoomRotateHandler(el, touchZoom, touchRotate, tapDragZoom); + this._add('touchRotate', touchRotate, ['touchPan', 'touchZoom']); + this._add('touchZoom', touchZoom, ['touchPan', 'touchRotate']); + if (options.interactive && options.touchZoomRotate) { + map.touchZoomRotate.enable(options.touchZoomRotate); + } + + const scrollZoom = map.scrollZoom = new ScrollZoomHandler(map, () => this._triggerRenderFrame()); + this._add('scrollZoom', scrollZoom, ['mousePan']); + if (options.interactive && options.scrollZoom) { + map.scrollZoom.enable(options.scrollZoom); + } + + const keyboard = map.keyboard = new KeyboardHandler(map); + this._add('keyboard', keyboard); + if (options.interactive && options.keyboard) { + map.keyboard.enable(); + } + + this._add('blockableMapEvent', new BlockableMapEventHandler(map)); + } + + _add(handlerName: string, handler: Handler, allowed?: Array) { + this._handlers.push({handlerName, handler, allowed}); + this._handlersById[handlerName] = handler; + } + + stop(allowEndAnimation: boolean) { + // do nothing if this method was triggered by a gesture update + if (this._updatingCamera) return; + + for (const {handler} of this._handlers) { + handler.reset(); + } + this._inertia.clear(); + this._fireEvents({}, {}, allowEndAnimation); + this._changes = []; + } + + isActive() { + for (const {handler} of this._handlers) { + if (handler.isActive()) return true; + } + return false; + } + + isZooming() { + return !!this._eventsInProgress.zoom || this._map.scrollZoom.isZooming(); + } + isRotating() { + return !!this._eventsInProgress.rotate; + } + + isMoving() { + return Boolean(isMoving(this._eventsInProgress)) || this.isZooming(); + } + + _blockedByActive(activeHandlers: {[x: string]: Handler}, allowed: Array, myName: string) { + for (const name in activeHandlers) { + if (name === myName) continue; + if (!allowed || allowed.indexOf(name) < 0) { + return true; + } + } + return false; + } + + handleWindowEvent = (e: { type: 'mousemove' | 'mouseup' | 'touchmove'}) => { + this.handleEvent(e, `${e.type}Window`); + }; + + _getMapTouches(touches: TouchList) { + const mapTouches = []; + for (const t of touches) { + const target = (t.target as any as Node); + if (this._el.contains(target)) { + mapTouches.push(t); + } + } + return mapTouches as any as TouchList; + } + + handleEvent = (e: Event, eventName?: keyof Handler) => { + + if (e.type === 'blur') { + this.stop(true); + return; + } + + this._updatingCamera = true; + + const inputEvent = e.type === 'renderFrame' ? undefined : e as InputEvent; + + /* + * We don't call e.preventDefault() for any events by default. + * Handlers are responsible for calling it where necessary. + */ + + const mergedHandlerResult: HandlerResult = {needsRenderFrame: false}; + const eventsInProgress: EventsInProgress = {}; + const activeHandlers = {}; + const eventTouches = (e as TouchEvent).touches; + + const mapTouches = eventTouches ? this._getMapTouches(eventTouches) : undefined; + const points = mapTouches ? DOM.touchPos(this._el, mapTouches) : DOM.mousePos(this._el, ((e as MouseEvent))); + + for (const {handlerName, handler, allowed} of this._handlers) { + if (!handler.isEnabled()) continue; + + let data: HandlerResult; + if (this._blockedByActive(activeHandlers, allowed, handlerName)) { + handler.reset(); + + } else { + if (handler[eventName || e.type]) { + data = handler[eventName || e.type](e, points, mapTouches); + this.mergeHandlerResult(mergedHandlerResult, eventsInProgress, data, handlerName, inputEvent); + if (data && data.needsRenderFrame) { + this._triggerRenderFrame(); + } + } + } + + if (data || handler.isActive()) { + activeHandlers[handlerName] = handler; + } + } + + const deactivatedHandlers: {[handlerName: string]: Event} = {}; + for (const name in this._previousActiveHandlers) { + if (!activeHandlers[name]) { + deactivatedHandlers[name] = inputEvent; + } + } + this._previousActiveHandlers = activeHandlers; + + if (Object.keys(deactivatedHandlers).length || hasChange(mergedHandlerResult)) { + this._changes.push([mergedHandlerResult, eventsInProgress, deactivatedHandlers]); + this._triggerRenderFrame(); + } + + if (Object.keys(activeHandlers).length || hasChange(mergedHandlerResult)) { + this._map._stop(true); + } + + this._updatingCamera = false; + + const {cameraAnimation} = mergedHandlerResult; + if (cameraAnimation) { + this._inertia.clear(); + this._fireEvents({}, {}, true); + this._changes = []; + cameraAnimation(this._map); + } + }; + + mergeHandlerResult(mergedHandlerResult: HandlerResult, + eventsInProgress: EventsInProgress, + handlerResult: HandlerResult, + name: string, + e?: InputEvent) { + if (!handlerResult) return; + + extend(mergedHandlerResult, handlerResult); + + const eventData = {handlerName: name, originalEvent: handlerResult.originalEvent || e}; + + // track which handler changed which camera property + if (handlerResult.zoomDelta !== undefined) { + eventsInProgress.zoom = eventData; + } + if (handlerResult.panDelta !== undefined) { + eventsInProgress.drag = eventData; + } + if (handlerResult.pitchDelta !== undefined) { + eventsInProgress.pitch = eventData; + } + if (handlerResult.bearingDelta !== undefined) { + eventsInProgress.rotate = eventData; + } + + } + + _applyChanges() { + const combined: HandlerResult = {}; + const combinedEventsInProgress: EventsInProgress = {}; + const combinedDeactivatedHandlers = {}; + + for (const [change, eventsInProgress, deactivatedHandlers] of this._changes) { + + if (change.panDelta) combined.panDelta = (combined.panDelta || new Point(0, 0))._add(change.panDelta); + if (change.zoomDelta) combined.zoomDelta = (combined.zoomDelta || 0) + change.zoomDelta; + if (change.bearingDelta) combined.bearingDelta = (combined.bearingDelta || 0) + change.bearingDelta; + if (change.pitchDelta) combined.pitchDelta = (combined.pitchDelta || 0) + change.pitchDelta; + if (change.around !== undefined) combined.around = change.around; + if (change.pinchAround !== undefined) combined.pinchAround = change.pinchAround; + if (change.noInertia) combined.noInertia = change.noInertia; + + extend(combinedEventsInProgress, eventsInProgress); + extend(combinedDeactivatedHandlers, deactivatedHandlers); + } + + this._updateMapTransform(combined, combinedEventsInProgress, combinedDeactivatedHandlers); + this._changes = []; + } + + _updateMapTransform(combinedResult: HandlerResult, + combinedEventsInProgress: EventsInProgress, + deactivatedHandlers: {[handlerName: string]: Event}) { + const map = this._map; + const tr = map._getTransformForUpdate(); + const terrain = map.terrain; + + if (!hasChange(combinedResult) && !(terrain && this._terrainMovement)) { + return this._fireEvents(combinedEventsInProgress, deactivatedHandlers, true); + } + + let {panDelta, zoomDelta, bearingDelta, pitchDelta, around, pinchAround} = combinedResult; + + if (pinchAround !== undefined) { + around = pinchAround; + } + + // stop any ongoing camera animations (easeTo, flyTo) + map._stop(true); + + around = around || map.transform.centerPoint; + const loc = tr.pointLocation(panDelta ? around.sub(panDelta) : around); + if (bearingDelta) tr.bearing += bearingDelta; + if (pitchDelta) tr.pitch += pitchDelta; + if (zoomDelta) tr.zoom += zoomDelta; + + if (!terrain) { + tr.setLocationAtPoint(loc, around); + } else { + // when 3d-terrain is enabled act a little different: + // - dragging do not drag the picked point itself, instead it drags the map by pixel-delta. + // With this approach it is no longer possible to pick a point from somewhere near + // the horizon to the center in one move. + // So this logic avoids the problem, that in such cases you easily loose orientation. + if (!this._terrainMovement && + (combinedEventsInProgress.drag || combinedEventsInProgress.zoom)) { + // When starting to drag or move, flag it and register moveend to clear flagging + this._terrainMovement = true; + this._map._elevationFreeze = true; + tr.setLocationAtPoint(loc, around); + this._map.once('moveend', () => { + this._map._elevationFreeze = false; + this._terrainMovement = false; + tr.recalculateZoom(map.terrain); + }); + } else if (combinedEventsInProgress.drag && this._terrainMovement) { + // drag map + tr.center = tr.pointLocation(tr.centerPoint.sub(panDelta)); + } else { + tr.setLocationAtPoint(loc, around); + } + } + + map._applyUpdatedTransform(tr); + + this._map._update(); + if (!combinedResult.noInertia) this._inertia.record(combinedResult); + this._fireEvents(combinedEventsInProgress, deactivatedHandlers, true); + + } + + _fireEvents(newEventsInProgress: EventsInProgress, deactivatedHandlers: {[handlerName: string]: Event}, allowEndAnimation: boolean) { + + const wasMoving = isMoving(this._eventsInProgress); + const nowMoving = isMoving(newEventsInProgress); + + const startEvents = {}; + + for (const eventName in newEventsInProgress) { + const {originalEvent} = newEventsInProgress[eventName]; + if (!this._eventsInProgress[eventName]) { + startEvents[`${eventName}start`] = originalEvent; + } + this._eventsInProgress[eventName] = newEventsInProgress[eventName]; + } + + // fire start events only after this._eventsInProgress has been updated + if (!wasMoving && nowMoving) { + this._fireEvent('movestart', nowMoving.originalEvent); + } + + for (const name in startEvents) { + this._fireEvent(name, startEvents[name]); + } + + if (nowMoving) { + this._fireEvent('move', nowMoving.originalEvent); + } + + for (const eventName in newEventsInProgress) { + const {originalEvent} = newEventsInProgress[eventName]; + this._fireEvent(eventName, originalEvent); + } + + const endEvents = {}; + + let originalEndEvent; + for (const eventName in this._eventsInProgress) { + const {handlerName, originalEvent} = this._eventsInProgress[eventName]; + if (!this._handlersById[handlerName].isActive()) { + delete this._eventsInProgress[eventName]; + originalEndEvent = deactivatedHandlers[handlerName] || originalEvent; + endEvents[`${eventName}end`] = originalEndEvent; + } + } + + for (const name in endEvents) { + this._fireEvent(name, endEvents[name]); + } + + const stillMoving = isMoving(this._eventsInProgress); + if (allowEndAnimation && (wasMoving || nowMoving) && !stillMoving) { + this._updatingCamera = true; + const inertialEase = this._inertia._onMoveEnd(this._map.dragPan._inertiaOptions); + + const shouldSnapToNorth = bearing => bearing !== 0 && -this._bearingSnap < bearing && bearing < this._bearingSnap; + + if (inertialEase && (inertialEase.essential || !browser.prefersReducedMotion)) { + if (shouldSnapToNorth(inertialEase.bearing || this._map.getBearing())) { + inertialEase.bearing = 0; + } + inertialEase.freezeElevation = true; + this._map.easeTo(inertialEase, {originalEvent: originalEndEvent}); + } else { + this._map.fire(new Event('moveend', {originalEvent: originalEndEvent})); + if (shouldSnapToNorth(this._map.getBearing())) { + this._map.resetNorth(); + } + } + this._updatingCamera = false; + } + + } + + _fireEvent(type: string, e?: Event) { + this._map.fire(new Event(type, e ? {originalEvent: e} : {})); + } + + _requestFrame() { + this._map.triggerRepaint(); + return this._map._renderTaskQueue.add(timeStamp => { + delete this._frameId; + this.handleEvent(new RenderFrameEvent('renderFrame', {timeStamp})); + this._applyChanges(); + }); + } + + _triggerRenderFrame() { + if (this._frameId === undefined) { + this._frameId = this._requestFrame(); + } + } +} diff --git a/web/libraries/maplibre-gl/src/ui/hash.test.ts b/web/libraries/maplibre-gl/src/ui/hash.test.ts new file mode 100644 index 00000000..d0fd3cb1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/hash.test.ts @@ -0,0 +1,292 @@ +import {Hash} from './hash'; +import {createMap as globalCreateMap, beforeMapTest} from '../util/test/util'; + +describe('hash', () => { + function createHash(name: string = undefined) { + const hash = new Hash(name); + hash._updateHash = hash._updateHashUnthrottled.bind(hash); + return hash; + } + + function createMap() { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'clientWidth', {value: 512}); + Object.defineProperty(container, 'clientHeight', {value: 512}); + return globalCreateMap({container}, undefined); + } + + let map; + + beforeEach(() => { + beforeMapTest(); + map = createMap(); + }); + + afterEach(() => { + if (map.removed === false) { + map.remove(); + } + window.location.hash = ''; + }); + + test('#addTo', () => { + const hash = createHash(); + + expect(hash._map).toBeFalsy(); + + hash.addTo(map); + + expect(hash._map).toBeTruthy(); + }); + + test('#remove', () => { + const hash = createHash() + .addTo(map); + + expect(hash._map).toBeTruthy(); + + hash.remove(); + + expect(hash._map).toBeFalsy(); + }); + + test('#_onHashChange', () => { + const hash = createHash() + .addTo(map); + + window.location.hash = '#10/3.00/-1.00'; + + hash._onHashChange(); + + expect(map.getCenter().lng).toBe(-1); + expect(map.getCenter().lat).toBe(3); + expect(map.getZoom()).toBe(10); + expect(map.getBearing() === 0 ? 0 : map.getBearing()).toBe(0); + expect(map.getPitch()).toBe(0); + + // map is created with `interactive: false` + // so explicitly enable rotation for this test + map.dragRotate.enable(); + map.touchZoomRotate.enable(); + + window.location.hash = '#5/1.00/0.50/30/60'; + + hash._onHashChange(); + + expect(map.getCenter().lng).toBe(0.5); + expect(map.getCenter().lat).toBe(1); + expect(map.getZoom()).toBe(5); + expect(map.getBearing()).toBe(30); + expect(map.getPitch()).toBe(60); + + // disable rotation to test that updating + // the hash's bearing won't change the map + map.dragRotate.disable(); + map.touchZoomRotate.disable(); + + window.location.hash = '#5/1.00/0.50/-45/60'; + + hash._onHashChange(); + + expect(map.getCenter().lng).toBe(0.5); + expect(map.getCenter().lat).toBe(1); + expect(map.getZoom()).toBe(5); + expect(map.getBearing()).toBe(30); + expect(map.getPitch()).toBe(60); + + // test that a hash with no bearing resets + // to the previous bearing when rotation is disabled + window.location.hash = '#5/1.00/0.50/'; + + hash._onHashChange(); + + expect(map.getCenter().lng).toBe(0.5); + expect(map.getCenter().lat).toBe(1); + expect(map.getZoom()).toBe(5); + expect(map.getBearing()).toBe(30); + expect(window.location.hash).toBe('#5/1/0.5/30'); + + window.location.hash = '#4/wrongly/formed/hash'; + + expect(hash._onHashChange()).toBeFalsy(); + + window.location.hash = '#map=10/3.00/-1.00&foo=bar'; + + expect(hash._onHashChange()).toBeFalsy(); + }); + + test('#_onHashChange empty', () => { + const hash = createHash() + .addTo(map); + + window.location.hash = '#10/3.00/-1.00'; + + hash._onHashChange(); + + expect(map.getCenter().lng).toBe(-1); + expect(map.getCenter().lat).toBe(3); + expect(map.getZoom()).toBe(10); + expect(map.getBearing() === 0 ? 0 : map.getBearing()).toBe(0); + expect(map.getPitch()).toBe(0); + + window.location.hash = ''; + + hash._onHashChange(); + + expect(map.getCenter().lng).toBe(-1); + expect(map.getCenter().lat).toBe(3); + expect(map.getZoom()).toBe(10); + expect(map.getBearing() === 0 ? 0 : map.getBearing()).toBe(0); + expect(map.getPitch()).toBe(0); + }); + + test('#_onHashChange named', () => { + const hash = createHash('map') + .addTo(map); + + window.location.hash = '#map=10/3.00/-1.00&foo=bar'; + + hash._onHashChange(); + + expect(map.getCenter().lng).toBe(-1); + expect(map.getCenter().lat).toBe(3); + expect(map.getZoom()).toBe(10); + expect(map.getBearing() === 0 ? 0 : map.getBearing()).toBe(0); + expect(map.getPitch()).toBe(0); + + window.location.hash = '#map&foo=bar'; + + expect(hash._onHashChange()).toBeFalsy(); + + window.location.hash = '#map=4/5/baz&foo=bar'; + + expect(hash._onHashChange()).toBeFalsy(); + + window.location.hash = '#5/1.00/0.50/30/60'; + + expect(hash._onHashChange()).toBeFalsy(); + }); + + test('#_getCurrentHash', () => { + const hash = createHash() + .addTo(map); + + window.location.hash = '#10/3.00/-1.00'; + + const currentHash = hash._getCurrentHash(); + + expect(currentHash[0]).toBe('10'); + expect(currentHash[1]).toBe('3.00'); + expect(currentHash[2]).toBe('-1.00'); + }); + + test('#_getCurrentHash named', () => { + const hash = createHash('map') + .addTo(map); + + window.location.hash = '#map=10/3.00/-1.00&foo=bar'; + + let currentHash = hash._getCurrentHash(); + + expect(currentHash[0]).toBe('10'); + expect(currentHash[1]).toBe('3.00'); + expect(currentHash[2]).toBe('-1.00'); + + window.location.hash = '#baz&map=10/3.00/-1.00'; + + currentHash = hash._getCurrentHash(); + + expect(currentHash[0]).toBe('10'); + expect(currentHash[1]).toBe('3.00'); + expect(currentHash[2]).toBe('-1.00'); + }); + + test('#_updateHash', () => { + function getHash() { + return window.location.hash.split('/'); + } + + createHash() + .addTo(map); + + expect(window.location.hash).toBeFalsy(); + + map.setZoom(3); + map.setCenter([2.0, 1.0]); + + expect(window.location.hash).toBeTruthy(); + + let newHash = getHash(); + + expect(newHash).toHaveLength(3); + expect(newHash[0]).toBe('#3'); + expect(newHash[1]).toBe('1'); + expect(newHash[2]).toBe('2'); + + map.setPitch(60); + + newHash = getHash(); + + expect(newHash).toHaveLength(5); + expect(newHash[0]).toBe('#3'); + expect(newHash[1]).toBe('1'); + expect(newHash[2]).toBe('2'); + expect(newHash[3]).toBe('0'); + expect(newHash[4]).toBe('60'); + + map.setBearing(135); + + newHash = getHash(); + + expect(newHash).toHaveLength(5); + expect(newHash[0]).toBe('#3'); + expect(newHash[1]).toBe('1'); + expect(newHash[2]).toBe('2'); + expect(newHash[3]).toBe('135'); + expect(newHash[4]).toBe('60'); + }); + + test('#_updateHash named', () => { + createHash('map') + .addTo(map); + + expect(window.location.hash).toBeFalsy(); + + map.setZoom(3); + map.setCenter([1.0, 2.0]); + + expect(window.location.hash).toBeTruthy(); + + expect(window.location.hash).toBe('#map=3/2/1'); + + map.setPitch(60); + + expect(window.location.hash).toBe('#map=3/2/1/0/60'); + + map.setBearing(135); + + expect(window.location.hash).toBe('#map=3/2/1/135/60'); + + window.location.hash += '&foo=bar'; + + map.setZoom(7); + + expect(window.location.hash).toBe('#map=7/2/1/135/60&foo=bar'); + + window.location.hash = '#baz&map=7/2/1/135/60&foo=bar'; + + map.setCenter([2.0, 1.0]); + + expect(window.location.hash).toBe('#baz&map=7/1/2/135/60&foo=bar'); + }); + + test('map#remove', () => { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'clientWidth', {value: 512}); + Object.defineProperty(container, 'clientHeight', {value: 512}); + + map.remove(); + + expect(map).toBeTruthy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/hash.ts b/web/libraries/maplibre-gl/src/ui/hash.ts new file mode 100644 index 00000000..ad4d4c81 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/hash.ts @@ -0,0 +1,137 @@ +import {throttle} from '../util/throttle'; + +import type {Map} from './map'; + +/** + * Adds the map's position to its page's location hash. + * Passed as an option to the map object. + * + * @group Markers and Controls + */ +export class Hash { + _map: Map; + _hashName: string; + + constructor(hashName?: string | null) { + this._hashName = hashName && encodeURIComponent(hashName); + } + + /** + * Map element to listen for coordinate changes + * + * @param map - The map object + * @returns `this` + */ + addTo(map: Map) { + this._map = map; + addEventListener('hashchange', this._onHashChange, false); + this._map.on('moveend', this._updateHash); + return this; + } + + /** + * Removes hash + * + * @returns `this` + */ + remove() { + removeEventListener('hashchange', this._onHashChange, false); + this._map.off('moveend', this._updateHash); + clearTimeout(this._updateHash()); + + delete this._map; + return this; + } + + getHashString(mapFeedback?: boolean) { + const center = this._map.getCenter(), + zoom = Math.round(this._map.getZoom() * 100) / 100, + // derived from equation: 512px * 2^z / 360 / 10^d < 0.5px + precision = Math.ceil((zoom * Math.LN2 + Math.log(512 / 360 / 0.5)) / Math.LN10), + m = Math.pow(10, precision), + lng = Math.round(center.lng * m) / m, + lat = Math.round(center.lat * m) / m, + bearing = this._map.getBearing(), + pitch = this._map.getPitch(); + let hash = ''; + if (mapFeedback) { + // new map feedback site has some constraints that don't allow + // us to use the same hash format as we do for the Map hash option. + hash += `/${lng}/${lat}/${zoom}`; + } else { + hash += `${zoom}/${lat}/${lng}`; + } + + if (bearing || pitch) hash += (`/${Math.round(bearing * 10) / 10}`); + if (pitch) hash += (`/${Math.round(pitch)}`); + + if (this._hashName) { + const hashName = this._hashName; + let found = false; + const parts = window.location.hash.slice(1).split('&').map(part => { + const key = part.split('=')[0]; + if (key === hashName) { + found = true; + return `${key}=${hash}`; + } + return part; + }).filter(a => a); + if (!found) { + parts.push(`${hashName}=${hash}`); + } + return `#${parts.join('&')}`; + } + + return `#${hash}`; + } + + _getCurrentHash = () => { + // Get the current hash from location, stripped from its number sign + const hash = window.location.hash.replace('#', ''); + if (this._hashName) { + // Split the parameter-styled hash into parts and find the value we need + let keyval; + hash.split('&').map( + part => part.split('=') + ).forEach(part => { + if (part[0] === this._hashName) { + keyval = part; + } + }); + return (keyval ? keyval[1] || '' : '').split('/'); + } + return hash.split('/'); + }; + + _onHashChange = () => { + const loc = this._getCurrentHash(); + if (loc.length >= 3 && !loc.some(v => isNaN(v))) { + const bearing = this._map.dragRotate.isEnabled() && this._map.touchZoomRotate.isEnabled() ? +(loc[3] || 0) : this._map.getBearing(); + this._map.jumpTo({ + center: [+loc[2], +loc[1]], + zoom: +loc[0], + bearing, + pitch: +(loc[4] || 0) + }); + return true; + } + return false; + }; + + _updateHashUnthrottled = () => { + // Replace if already present, else append the updated hash string + const location = window.location.href.replace(/(#.+)?$/, this.getHashString()); + try { + window.history.replaceState(window.history.state, null, location); + } catch (SecurityError) { + // IE11 does not allow this if the page is within an iframe created + // with iframe.contentWindow.document.write(...). + // https://github.com/mapbox/mapbox-gl-js/issues/7410 + } + }; + + /** + * Mobile Safari doesn't allow updating the hash more than 100 times per 30 seconds. + */ + _updateHash: () => ReturnType = throttle(this._updateHashUnthrottled, 30 * 1000 / 100); +} diff --git a/web/libraries/maplibre-gl/src/ui/map.test.ts b/web/libraries/maplibre-gl/src/ui/map.test.ts new file mode 100755 index 00000000..ad577fc7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/map.test.ts @@ -0,0 +1,2923 @@ +import {Map, MapOptions} from './map'; +import {createMap, setErrorWebGlContext, beforeMapTest} from '../util/test/util'; +import {LngLat} from '../geo/lng_lat'; +import {Tile} from '../source/tile'; +import {OverscaledTileID} from '../source/tile_id'; +import {Event, ErrorEvent} from '../util/evented'; +import simulate from '../../test/unit/lib/simulate_interaction'; +import {fixedLngLat, fixedNum} from '../../test/unit/lib/fixed'; +import {GeoJSONSourceSpecification, LayerSpecification, SourceSpecification, StyleSpecification} from '@maplibre/maplibre-gl-style-spec'; +import {RequestTransformFunction} from '../util/request_manager'; +import {extend} from '../util/util'; +import {LngLatBoundsLike} from '../geo/lng_lat_bounds'; +import {IControl} from './control/control'; +import {EvaluationParameters} from '../style/evaluation_parameters'; +import {fakeServer, FakeServer} from 'nise'; +import {CameraOptions} from './camera'; +import {Terrain} from '../render/terrain'; +import {mercatorZfromAltitude} from '../geo/mercator_coordinate'; +import {Transform} from '../geo/transform'; +import {StyleImageInterface} from '../style/style_image'; +import {Style} from '../style/style'; +import {MapSourceDataEvent} from './events'; +import {config} from '../util/config'; + +function createStyleSource() { + return { + type: 'geojson', + data: { + type: 'FeatureCollection', + features: [] + } + } as SourceSpecification; +} + +let server: FakeServer; + +beforeEach(() => { + beforeMapTest(); + global.fetch = null; + server = fakeServer.create(); +}); + +afterEach(() => { + server.restore(); +}); + +describe('Map', () => { + + test('version', () => { + const map = createMap({interactive: true, style: null}); + + expect(typeof map.version === 'string').toBeTruthy(); + + // Semver regex: https://gist.github.com/jhorsman/62eeea161a13b80e39f5249281e17c39 + // Backslashes are doubled to escape them + const regexp = new RegExp('^([0-9]+)\\.([0-9]+)\\.([0-9]+)(?:-([0-9A-Za-z-]+(?:\\.[0-9A-Za-z-]+)*))?(?:\\+[0-9A-Za-z-]+)?$'); + expect(regexp.test(map.version)).toBeTruthy(); + }); + + test('constructor', () => { + const map = createMap({interactive: true, style: null}); + expect(map.getContainer()).toBeTruthy(); + expect(map.getStyle()).toBeUndefined(); + expect(map.boxZoom.isEnabled()).toBeTruthy(); + expect(map.doubleClickZoom.isEnabled()).toBeTruthy(); + expect(map.dragPan.isEnabled()).toBeTruthy(); + expect(map.dragRotate.isEnabled()).toBeTruthy(); + expect(map.keyboard.isEnabled()).toBeTruthy(); + expect(map.scrollZoom.isEnabled()).toBeTruthy(); + expect(map.touchZoomRotate.isEnabled()).toBeTruthy(); + expect(() => { + new Map({ + container: 'anElementIdWhichDoesNotExistInTheDocument' + } as any as MapOptions); + }).toThrow( + new Error('Container \'anElementIdWhichDoesNotExistInTheDocument\' not found.') + ); + }); + + test('bad map-specific token breaks map', () => { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'offsetWidth', {value: 512}); + Object.defineProperty(container, 'offsetHeight', {value: 512}); + createMap(); + //t.error(); + }); + + test('initial bounds in constructor options', () => { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'offsetWidth', {value: 512}); + Object.defineProperty(container, 'offsetHeight', {value: 512}); + + const bounds = [[-133, 16], [-68, 50]]; + const map = createMap({container, bounds}); + + expect(fixedLngLat(map.getCenter(), 4)).toEqual({lng: -100.5, lat: 34.7171}); + expect(fixedNum(map.getZoom(), 3)).toBe(2.113); + }); + + test('initial bounds options in constructor options', () => { + const bounds = [[-133, 16], [-68, 50]]; + + const map = (fitBoundsOptions) => { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'offsetWidth', {value: 512}); + Object.defineProperty(container, 'offsetHeight', {value: 512}); + return createMap({container, bounds, fitBoundsOptions}); + }; + + const unpadded = map(undefined); + const padded = map({padding: 100}); + + expect(unpadded.getZoom() > padded.getZoom()).toBeTruthy(); + }); + + describe('disables handlers', () => { + test('disables all handlers', () => { + const map = createMap({interactive: false}); + + expect(map.boxZoom.isEnabled()).toBeFalsy(); + expect(map.doubleClickZoom.isEnabled()).toBeFalsy(); + expect(map.dragPan.isEnabled()).toBeFalsy(); + expect(map.dragRotate.isEnabled()).toBeFalsy(); + expect(map.keyboard.isEnabled()).toBeFalsy(); + expect(map.scrollZoom.isEnabled()).toBeFalsy(); + expect(map.touchZoomRotate.isEnabled()).toBeFalsy(); + }); + + const handlerNames = [ + 'scrollZoom', + 'boxZoom', + 'dragRotate', + 'dragPan', + 'keyboard', + 'doubleClickZoom', + 'touchZoomRotate' + ]; + handlerNames.forEach((handlerName) => { + test(`disables "${handlerName}" handler`, () => { + const options = {}; + options[handlerName] = false; + const map = createMap(options); + + expect(map[handlerName].isEnabled()).toBeFalsy(); + + }); + }); + + }); + + test('emits load event after a style is set', done => { + const map = new Map({container: window.document.createElement('div')} as any as MapOptions); + + const fail = () => done('test failed'); + const pass = () => done(); + + map.on('load', fail); + + setTimeout(() => { + map.off('load', fail); + map.on('load', pass); + map.setStyle(createStyle()); + }, 1); + + }); + + describe('#mapOptions', () => { + test('maxTileCacheZoomLevels: Default value is set', () => { + const map = createMap(); + expect(map._maxTileCacheZoomLevels).toBe(config.MAX_TILE_CACHE_ZOOM_LEVELS); + }); + + test('maxTileCacheZoomLevels: Value can be set via map options', () => { + const map = createMap({maxTileCacheZoomLevels: 1}); + expect(map._maxTileCacheZoomLevels).toBe(1); + }); + + test('Style validation is enabled by default', () => { + let validationOption = false; + jest.spyOn(Style.prototype, 'loadJSON').mockImplementationOnce((styleJson, options) => { + validationOption = options.validate; + }); + createMap(); + expect(validationOption).toBeTruthy(); + }); + + test('Style validation disabled using mapOptions', () => { + let validationOption = true; + jest.spyOn(Style.prototype, 'loadJSON').mockImplementationOnce((styleJson, options) => { + validationOption = options.validate; + }); + createMap({validateStyle: false}); + + expect(validationOption).toBeFalsy(); + }); + + test('fadeDuration is set after first idle event', async () => { + let idleTriggered = false; + const fadeDuration = 100; + const spy = jest.spyOn(Style.prototype, 'update').mockImplementation((parameters: EvaluationParameters) => { + if (!idleTriggered) { + expect(parameters.fadeDuration).toBe(0); + } else { + expect(parameters.fadeDuration).toBe(fadeDuration); + } + }); + const style = createStyle(); + const map = createMap({style, fadeDuration}); + await map.once('idle'); + idleTriggered = true; + map.zoomTo(0.5, {duration: 100}); + spy.mockRestore(); + }); + }); + + describe('#setStyle', () => { + test('returns self', () => { + const map = new Map({container: window.document.createElement('div')} as any as MapOptions); + expect(map.setStyle({ + version: 8, + sources: {}, + layers: [] + })).toBe(map); + }); + + test('sets up event forwarding', () => { + createMap({}, (error, map) => { + expect(error).toBeFalsy(); + + const events = []; + function recordEvent(event) { events.push(event.type); } + + map.on('error', recordEvent); + map.on('data', recordEvent); + map.on('dataloading', recordEvent); + + map.style.fire(new Event('error')); + map.style.fire(new Event('data')); + map.style.fire(new Event('dataloading')); + + expect(events).toEqual([ + 'error', + 'data', + 'dataloading', + ]); + + }); + }); + + test('fires *data and *dataloading events', () => { + createMap({}, (error, map) => { + expect(error).toBeFalsy(); + + const events = []; + function recordEvent(event) { events.push(event.type); } + + map.on('styledata', recordEvent); + map.on('styledataloading', recordEvent); + map.on('sourcedata', recordEvent); + map.on('sourcedataloading', recordEvent); + map.on('tiledata', recordEvent); + map.on('tiledataloading', recordEvent); + + map.style.fire(new Event('data', {dataType: 'style'})); + map.style.fire(new Event('dataloading', {dataType: 'style'})); + map.style.fire(new Event('data', {dataType: 'source'})); + map.style.fire(new Event('dataloading', {dataType: 'source'})); + map.style.fire(new Event('data', {dataType: 'tile'})); + map.style.fire(new Event('dataloading', {dataType: 'tile'})); + + expect(events).toEqual([ + 'styledata', + 'styledataloading', + 'sourcedata', + 'sourcedataloading', + 'tiledata', + 'tiledataloading' + ]); + + }); + }); + + test('can be called more than once', () => { + const map = createMap(); + + map.setStyle({version: 8, sources: {}, layers: []}, {diff: false}); + map.setStyle({version: 8, sources: {}, layers: []}, {diff: false}); + + }); + + test('setStyle back to the first style should work', done => { + const redStyle = {version: 8 as const, sources: {}, layers: [ + {id: 'background', type: 'background' as const, paint: {'background-color': 'red'}}, + ]}; + const blueStyle = {version: 8 as const, sources: {}, layers: [ + {id: 'background', type: 'background' as const, paint: {'background-color': 'blue'}}, + ]}; + const map = createMap({style: redStyle}); + map.setStyle(blueStyle); + map.once('style.load', () => { + map.setStyle(redStyle); + const serializedStyle = map.style.serialize(); + expect(serializedStyle.layers[0].paint['background-color']).toBe('red'); + done(); + }); + }); + + test('style transform overrides unmodified map transform', done => { + const map = new Map({container: window.document.createElement('div')} as any as MapOptions); + map.transform.lngRange = [-120, 140]; + map.transform.latRange = [-60, 80]; + map.transform.resize(600, 400); + expect(map.transform.zoom).toBe(0.6983039737971014); + expect(map.transform.unmodified).toBeTruthy(); + map.setStyle(createStyle()); + map.on('style.load', () => { + expect(fixedLngLat(map.transform.center)).toEqual(fixedLngLat({lng: -73.9749, lat: 40.7736})); + expect(fixedNum(map.transform.zoom)).toBe(12.5); + expect(fixedNum(map.transform.bearing)).toBe(29); + expect(fixedNum(map.transform.pitch)).toBe(50); + done(); + }); + }); + + test('style transform does not override map transform modified via options', done => { + const map = new Map({container: window.document.createElement('div'), zoom: 10, center: [-77.0186, 38.8888]} as any as MapOptions); + expect(map.transform.unmodified).toBeFalsy(); + map.setStyle(createStyle()); + map.on('style.load', () => { + expect(fixedLngLat(map.transform.center)).toEqual(fixedLngLat({lng: -77.0186, lat: 38.8888})); + expect(fixedNum(map.transform.zoom)).toBe(10); + expect(fixedNum(map.transform.bearing)).toBe(0); + expect(fixedNum(map.transform.pitch)).toBe(0); + done(); + }); + }); + + test('style transform does not override map transform modified via setters', done => { + const map = new Map({container: window.document.createElement('div')} as any as MapOptions); + expect(map.transform.unmodified).toBeTruthy(); + map.setZoom(10); + map.setCenter([-77.0186, 38.8888]); + expect(map.transform.unmodified).toBeFalsy(); + map.setStyle(createStyle()); + map.on('style.load', () => { + expect(fixedLngLat(map.transform.center)).toEqual(fixedLngLat({lng: -77.0186, lat: 38.8888})); + expect(fixedNum(map.transform.zoom)).toBe(10); + expect(fixedNum(map.transform.bearing)).toBe(0); + expect(fixedNum(map.transform.pitch)).toBe(0); + done(); + }); + }); + + test('passing null removes style', () => { + const map = createMap(); + const style = map.style; + expect(style).toBeTruthy(); + jest.spyOn(style, '_remove'); + map.setStyle(null); + expect(style._remove).toHaveBeenCalledTimes(1); + }); + + test('passing null releases the worker', () => { + const map = createMap(); + const spyWorkerPoolAcquire = jest.spyOn(map.style.dispatcher.workerPool, 'acquire'); + const spyWorkerPoolRelease = jest.spyOn(map.style.dispatcher.workerPool, 'release'); + + map.setStyle({version: 8, sources: {}, layers: []}, {diff: false}); + expect(spyWorkerPoolAcquire).toHaveBeenCalledTimes(1); + expect(spyWorkerPoolRelease).toHaveBeenCalledTimes(0); + + spyWorkerPoolAcquire.mockClear(); + map.setStyle(null); + expect(spyWorkerPoolAcquire).toHaveBeenCalledTimes(0); + expect(spyWorkerPoolRelease).toHaveBeenCalledTimes(1); + + // Cleanup + spyWorkerPoolAcquire.mockClear(); + spyWorkerPoolRelease.mockClear(); + }); + + test('transformStyle should copy the source and the layer into next style', done => { + const style = extend(createStyle(), { + sources: { + maplibre: { + type: 'vector', + minzoom: 1, + maxzoom: 10, + tiles: ['http://example.com/{z}/{x}/{y}.png'] + } + }, + layers: [{ + id: 'layerId0', + type: 'circle', + source: 'maplibre', + 'source-layer': 'sourceLayer' + }, { + id: 'layerId1', + type: 'circle', + source: 'maplibre', + 'source-layer': 'sourceLayer' + }] + }); + + const map = createMap({style}); + map.setStyle(createStyle(), { + diff: false, + transformStyle: (prevStyle, nextStyle) => ({ + ...nextStyle, + sources: { + ...nextStyle.sources, + maplibre: prevStyle.sources.maplibre + }, + layers: [ + ...nextStyle.layers, + prevStyle.layers[0] + ] + }) + }); + + map.on('style.load', () => { + const loadedStyle = map.style.serialize(); + expect('maplibre' in loadedStyle.sources).toBeTruthy(); + expect(loadedStyle.layers[0].id).toBe(style.layers[0].id); + expect(loadedStyle.layers).toHaveLength(1); + done(); + }); + }); + + test('delayed setStyle with transformStyle should copy the source and the layer into next style with diffing', done => { + const style = extend(createStyle(), { + sources: { + maplibre: { + type: 'vector', + minzoom: 1, + maxzoom: 10, + tiles: ['http://example.com/{z}/{x}/{y}.png'] + } + }, + layers: [{ + id: 'layerId0', + type: 'circle', + source: 'maplibre', + 'source-layer': 'sourceLayer' + }, { + id: 'layerId1', + type: 'circle', + source: 'maplibre', + 'source-layer': 'sourceLayer' + }] + }); + + const map = createMap({style}); + window.setTimeout(() => { + map.setStyle(createStyle(), { + diff: true, + transformStyle: (prevStyle, nextStyle) => ({ + ...nextStyle, + sources: { + ...nextStyle.sources, + maplibre: prevStyle.sources.maplibre + }, + layers: [ + ...nextStyle.layers, + prevStyle.layers[0] + ] + }) + }); + + const loadedStyle = map.style.serialize(); + expect('maplibre' in loadedStyle.sources).toBeTruthy(); + expect(loadedStyle.layers[0].id).toBe(style.layers[0].id); + expect(loadedStyle.layers).toHaveLength(1); + done(); + }, 100); + }); + + test('transformStyle should get called when passed to setStyle after the map is initialised without a style', done => { + const map = createMap({deleteStyle: true}); + map.setStyle(createStyle(), { + diff: true, + transformStyle: (prevStyle, nextStyle) => { + expect(prevStyle).toBeUndefined(); + + return { + ...nextStyle, + sources: { + maplibre: { + type: 'vector', + minzoom: 1, + maxzoom: 10, + tiles: ['http://example.com/{z}/{x}/{y}.png'] + } + }, + layers: [{ + id: 'layerId0', + type: 'circle', + source: 'maplibre', + 'source-layer': 'sourceLayer' + }] + }; + } + }); + + map.on('style.load', () => { + const loadedStyle = map.style.serialize(); + expect('maplibre' in loadedStyle.sources).toBeTruthy(); + expect(loadedStyle.layers[0].id).toBe('layerId0'); + done(); + }); + }); + + test('map load should be fired when transformStyle is used on setStyle after the map is initialised without a style', done => { + const map = createMap({deleteStyle: true}); + map.setStyle({version: 8, sources: {}, layers: []}, { + diff: true, + transformStyle: (prevStyle, nextStyle) => { + expect(prevStyle).toBeUndefined(); + expect(nextStyle).toBeDefined(); + return createStyle(); + } + }); + map.on('load', () => done()); + }); + + test('Override default style validation', () => { + let validationOption = true; + jest.spyOn(Style.prototype, 'loadJSON').mockImplementationOnce((styleJson, options) => { + validationOption = options.validate; + }); + const map = createMap({style: null}); + map.setStyle({version: 8, sources: {}, layers: []}, {validate: false}); + + expect(validationOption).toBeFalsy(); + }); + }); + + describe('#setTransformRequest', () => { + test('returns self', () => { + const transformRequest = (() => {}) as any as RequestTransformFunction; + const map = new Map({container: window.document.createElement('div')} as any as MapOptions); + expect(map.setTransformRequest(transformRequest)).toBe(map); + expect(map._requestManager._transformRequestFn).toBe(transformRequest); + }); + + test('can be called more than once', () => { + const map = createMap(); + + const transformRequest = (() => {}) as any as RequestTransformFunction; + map.setTransformRequest(transformRequest); + map.setTransformRequest(transformRequest); + }); + }); + + describe('#is_Loaded', () => { + + test('Map#isSourceLoaded', done => { + const style = createStyle(); + const map = createMap({style}); + + map.on('load', () => { + map.on('data', (e) => { + if (e.dataType === 'source' && e.sourceDataType === 'idle') { + expect(map.isSourceLoaded('geojson')).toBe(true); + done(); + } + }); + map.addSource('geojson', createStyleSource()); + expect(map.isSourceLoaded('geojson')).toBe(false); + }); + }); + + test('Map#isSourceLoaded (equivalent to event.isSourceLoaded)', done => { + const style = createStyle(); + const map = createMap({style}); + + map.on('load', () => { + map.on('data', (e) => { + if (e.dataType === 'source' && 'source' in e) { + const sourceDataEvent = e as MapSourceDataEvent; + expect(map.isSourceLoaded('geojson')).toBe(sourceDataEvent.isSourceLoaded); + if (sourceDataEvent.sourceDataType === 'idle') { + done(); + } + } + }); + map.addSource('geojson', createStyleSource()); + expect(map.isSourceLoaded('geojson')).toBe(false); + }); + }); + + test('Map#isStyleLoaded', done => { + const style = createStyle(); + const map = createMap({style}); + + expect(map.isStyleLoaded()).toBe(false); + map.on('load', () => { + expect(map.isStyleLoaded()).toBe(true); + done(); + }); + }); + + test('Map#areTilesLoaded', done => { + const style = createStyle(); + const map = createMap({style}); + expect(map.areTilesLoaded()).toBe(true); + map.on('load', () => { + const fakeTileId = new OverscaledTileID(0, 0, 0, 0, 0); + map.addSource('geojson', createStyleSource()); + map.style.sourceCaches.geojson._tiles[fakeTileId.key] = new Tile(fakeTileId, undefined); + expect(map.areTilesLoaded()).toBe(false); + map.style.sourceCaches.geojson._tiles[fakeTileId.key].state = 'loaded'; + expect(map.areTilesLoaded()).toBe(true); + done(); + }); + }); + }); + + describe('#getStyle', () => { + test('returns undefined if the style has not loaded yet', done => { + const style = createStyle(); + const map = createMap({style}); + expect(map.getStyle()).toBeUndefined(); + done(); + }); + + test('returns the style', done => { + const style = createStyle(); + const map = createMap({style}); + + map.on('load', () => { + expect(map.getStyle()).toEqual(style); + done(); + }); + }); + + test('returns the style with added sources', done => { + const style = createStyle(); + const map = createMap({style}); + + map.on('load', () => { + map.addSource('geojson', createStyleSource()); + expect(map.getStyle()).toEqual(extend(createStyle(), { + sources: {geojson: createStyleSource()} + })); + done(); + }); + }); + + test('fires an error on checking if non-existant source is loaded', done => { + const style = createStyle(); + const map = createMap({style}); + + map.on('load', () => { + map.on('error', ({error}) => { + expect(error.message).toMatch(/There is no source with ID/); + done(); + }); + map.isSourceLoaded('geojson'); + }); + }); + + test('returns the style with added layers', done => { + const style = createStyle(); + const map = createMap({style}); + const layer = { + id: 'background', + type: 'background' + } as LayerSpecification; + + map.on('load', () => { + map.addLayer(layer); + expect(map.getStyle()).toEqual(extend(createStyle(), { + layers: [layer] + })); + done(); + }); + }); + + test('a layer can be added even if a map is created without a style', () => { + const map = createMap({deleteStyle: true}); + const layer = { + id: 'background', + type: 'background' + } as LayerSpecification; + map.addLayer(layer); + }); + + test('a source can be added even if a map is created without a style', () => { + const map = createMap({deleteStyle: true}); + const source = createStyleSource(); + map.addSource('fill', source); + }); + + test('a layer can be added with an embedded source specification', () => { + const map = createMap({deleteStyle: true}); + const source: GeoJSONSourceSpecification = { + type: 'geojson', + data: {type: 'Point', coordinates: [0, 0]} + }; + map.addLayer({ + id: 'foo', + type: 'symbol', + source + }); + }); + + test('returns the style with added source and layer', done => { + const style = createStyle(); + const map = createMap({style}); + const source = createStyleSource(); + const layer = { + id: 'fill', + type: 'fill', + source: 'fill' + } as LayerSpecification; + + map.on('load', () => { + map.addSource('fill', source); + map.addLayer(layer); + expect(map.getStyle()).toEqual(extend(createStyle(), { + sources: {fill: source}, + layers: [layer] + })); + done(); + }); + }); + + test('creates a new Style if diff fails', () => { + const style = createStyle(); + const map = createMap({style}); + jest.spyOn(map.style, 'setState').mockImplementation(() => { + throw new Error('Dummy error'); + }); + jest.spyOn(console, 'warn').mockImplementation(() => {}); + + const previousStyle = map.style; + map.setStyle(style); + expect(map.style && map.style !== previousStyle).toBeTruthy(); + }); + + test('creates a new Style if diff option is false', () => { + const style = createStyle(); + const map = createMap({style}); + const spy = jest.spyOn(map.style, 'setState'); + + const previousStyle = map.style; + map.setStyle(style, {diff: false}); + expect(map.style && map.style !== previousStyle).toBeTruthy(); + expect(spy).not.toHaveBeenCalled(); + }); + + }); + + test('#moveLayer', async () => { + const map = createMap({ + style: extend(createStyle(), { + sources: { + mapbox: { + type: 'vector', + minzoom: 1, + maxzoom: 10, + tiles: ['http://example.com/{z}/{x}/{y}.png'] + } + }, + layers: [{ + id: 'layerId1', + type: 'circle', + source: 'mapbox', + 'source-layer': 'sourceLayer' + }, { + id: 'layerId2', + type: 'circle', + source: 'mapbox', + 'source-layer': 'sourceLayer' + }] + }) + }); + + await map.once('render'); + map.moveLayer('layerId1', 'layerId2'); + expect(map.getLayer('layerId1').id).toBe('layerId1'); + expect(map.getLayer('layerId2').id).toBe('layerId2'); + }); + + test('#getLayer', async () => { + const layer = { + id: 'layerId', + type: 'circle', + source: 'mapbox', + 'source-layer': 'sourceLayer' + }; + const map = createMap({ + style: extend(createStyle(), { + sources: { + mapbox: { + type: 'vector', + minzoom: 1, + maxzoom: 10, + tiles: ['http://example.com/{z}/{x}/{y}.png'] + } + }, + layers: [layer] + }) + }); + + await map.once('render'); + const mapLayer = map.getLayer('layerId'); + expect(mapLayer.id).toBe(layer.id); + expect(mapLayer.type).toBe(layer.type); + expect(mapLayer.source).toBe(layer.source); + }); + + describe('#getLayersOrder', () => { + test('returns ids of layers in the correct order', done => { + const map = createMap({ + style: extend(createStyle(), { + 'sources': { + 'raster': { + type: 'raster', + tiles: ['http://tiles.server'] + } + }, + 'layers': [{ + 'id': 'raster', + 'type': 'raster', + 'source': 'raster' + }] + }) + }); + + map.on('style.load', () => { + map.addLayer({ + id: 'custom', + type: 'custom', + render() {} + }, 'raster'); + expect(map.getLayersOrder()).toEqual(['custom', 'raster']); + done(); + }); + }); + }); + + describe('#resize', () => { + test('sets width and height from container clients', () => { + const map = createMap(), + container = map.getContainer(); + + Object.defineProperty(container, 'clientWidth', {value: 250}); + Object.defineProperty(container, 'clientHeight', {value: 250}); + map.resize(); + + expect(map.transform.width).toBe(250); + expect(map.transform.height).toBe(250); + + }); + + test('fires movestart, move, resize, and moveend events', () => { + const map = createMap(), + events = []; + + (['movestart', 'move', 'resize', 'moveend'] as any).forEach((event) => { + map.on(event, (e) => { + events.push(e.type); + }); + }); + + map.resize(); + expect(events).toEqual(['movestart', 'move', 'resize', 'moveend']); + + }); + + test('listen to window resize event', () => { + const spy = jest.fn(); + global.ResizeObserver = jest.fn().mockImplementation(() => ({ + observe: spy + })); + + createMap(); + + expect(spy).toHaveBeenCalled(); + }); + + test('do not resize if trackResize is false', () => { + let observerCallback: Function = null; + global.ResizeObserver = jest.fn().mockImplementation((c) => ({ + observe: () => { observerCallback = c; } + })); + + const map = createMap({trackResize: false}); + + const spyA = jest.spyOn(map, 'stop'); + const spyB = jest.spyOn(map, '_update'); + const spyC = jest.spyOn(map, 'resize'); + + observerCallback(); + + expect(spyA).not.toHaveBeenCalled(); + expect(spyB).not.toHaveBeenCalled(); + expect(spyC).not.toHaveBeenCalled(); + }); + + test('do resize if trackResize is true (default)', async () => { + let observerCallback: Function = null; + global.ResizeObserver = jest.fn().mockImplementation((c) => ({ + observe: () => { observerCallback = c; } + })); + + const map = createMap(); + + const updateSpy = jest.spyOn(map, '_update'); + const resizeSpy = jest.spyOn(map, 'resize'); + + // The initial "observe" event fired by ResizeObserver should be captured/muted + // in the map constructor + + observerCallback(); + expect(updateSpy).not.toHaveBeenCalled(); + expect(resizeSpy).not.toHaveBeenCalled(); + + // The next "observe" event should fire a resize / _update + + observerCallback(); + expect(updateSpy).toHaveBeenCalled(); + expect(resizeSpy).toHaveBeenCalledTimes(1); + + // Additional "observe" events should be throttled + observerCallback(); + observerCallback(); + observerCallback(); + observerCallback(); + expect(resizeSpy).toHaveBeenCalledTimes(1); + await new Promise((resolve) => { setTimeout(resolve, 100); }); + expect(resizeSpy).toHaveBeenCalledTimes(2); + }); + + test('width and height correctly rounded', () => { + const map = createMap(); + const container = map.getContainer(); + + Object.defineProperty(container, 'clientWidth', {value: 250.6}); + Object.defineProperty(container, 'clientHeight', {value: 250.6}); + map.resize(); + + expect(map.getCanvas().width).toBe(250); + expect(map.getCanvas().height).toBe(250); + expect(map.painter.width).toBe(250); + expect(map.painter.height).toBe(250); + }); + }); + + describe('#getBounds', () => { + + test('getBounds', () => { + const map = createMap({zoom: 0}); + expect(parseFloat(map.getBounds().getCenter().lng.toFixed(10))).toBe(-0); + expect(parseFloat(map.getBounds().getCenter().lat.toFixed(10))).toBe(0); + + expect(toFixed(map.getBounds().toArray())).toEqual(toFixed([ + [-70.31249999999976, -57.326521225216965], + [70.31249999999977, 57.32652122521695]])); + }); + + test('rotated bounds', () => { + const map = createMap({zoom: 1, bearing: 45}); + expect( + toFixed([[-49.718445552178764, -44.44541580601936], [49.7184455522, 44.445415806019355]]) + ).toEqual(toFixed(map.getBounds().toArray())); + + map.setBearing(135); + expect( + toFixed([[-49.718445552178764, -44.44541580601936], [49.7184455522, 44.445415806019355]]) + ).toEqual(toFixed(map.getBounds().toArray())); + + }); + + function toFixed(bounds) { + const n = 10; + return [ + [normalizeFixed(bounds[0][0], n), normalizeFixed(bounds[0][1], n)], + [normalizeFixed(bounds[1][0], n), normalizeFixed(bounds[1][1], n)] + ]; + } + + function normalizeFixed(num, n) { + // workaround for "-0.0000000000" ≠ "0.0000000000" + return parseFloat(num.toFixed(n)).toFixed(n); + } + }); + + describe('#setMaxBounds', () => { + test('constrains map bounds', () => { + const map = createMap({zoom: 0}); + map.setMaxBounds([[-130.4297, 50.0642], [-61.52344, 24.20688]]); + expect( + toFixed([[-130.4297000000, 7.0136641176], [-61.5234400000, 60.2398142283]]) + ).toEqual(toFixed(map.getBounds().toArray())); + }); + + test('when no argument is passed, map bounds constraints are removed', () => { + const map = createMap({zoom: 0}); + map.setMaxBounds([[-130.4297, 50.0642], [-61.52344, 24.20688]]); + expect( + toFixed([[-166.28906999999964, -27.6835270554], [-25.664070000000066, 73.8248206697]]) + ).toEqual(toFixed(map.setMaxBounds(null).setZoom(0).getBounds().toArray())); + }); + + test('should not zoom out farther than bounds', () => { + const map = createMap(); + map.setMaxBounds([[-130.4297, 50.0642], [-61.52344, 24.20688]]); + expect(map.setZoom(0).getZoom()).not.toBe(0); + }); + + function toFixed(bounds) { + const n = 9; + return [ + [bounds[0][0].toFixed(n), bounds[0][1].toFixed(n)], + [bounds[1][0].toFixed(n), bounds[1][1].toFixed(n)] + ]; + } + + }); + + describe('#getMaxBounds', () => { + test('returns null when no bounds set', () => { + const map = createMap({zoom: 0}); + expect(map.getMaxBounds()).toBeNull(); + }); + + test('returns bounds', () => { + const map = createMap({zoom: 0}); + const bounds = [[-130.4297, 50.0642], [-61.52344, 24.20688]] as LngLatBoundsLike; + map.setMaxBounds(bounds); + expect(map.getMaxBounds().toArray()).toEqual(bounds); + }); + + }); + + describe('#getRenderWorldCopies', () => { + test('initially false', () => { + const map = createMap({renderWorldCopies: false}); + expect(map.getRenderWorldCopies()).toBe(false); + }); + + test('initially true', () => { + const map = createMap({renderWorldCopies: true}); + expect(map.getRenderWorldCopies()).toBe(true); + }); + + }); + + describe('#setRenderWorldCopies', () => { + test('initially false', () => { + const map = createMap({renderWorldCopies: false}); + map.setRenderWorldCopies(true); + expect(map.getRenderWorldCopies()).toBe(true); + }); + + test('initially true', () => { + const map = createMap({renderWorldCopies: true}); + map.setRenderWorldCopies(false); + expect(map.getRenderWorldCopies()).toBe(false); + }); + + test('undefined', () => { + const map = createMap({renderWorldCopies: false}); + map.setRenderWorldCopies(undefined); + expect(map.getRenderWorldCopies()).toBe(true); + }); + + test('null', () => { + const map = createMap({renderWorldCopies: true}); + map.setRenderWorldCopies(null); + expect(map.getRenderWorldCopies()).toBe(false); + }); + + }); + + test('#setMinZoom', () => { + const map = createMap({zoom: 5}); + map.setMinZoom(3.5); + map.setZoom(1); + expect(map.getZoom()).toBe(3.5); + }); + + test('unset minZoom', () => { + const map = createMap({minZoom: 5}); + map.setMinZoom(null); + map.setZoom(1); + expect(map.getZoom()).toBe(1); + }); + + test('#getMinZoom', () => { + const map = createMap({zoom: 0}); + expect(map.getMinZoom()).toBe(-2); + map.setMinZoom(10); + expect(map.getMinZoom()).toBe(10); + }); + + test('ignore minZooms over maxZoom', () => { + const map = createMap({zoom: 2, maxZoom: 5}); + expect(() => { + map.setMinZoom(6); + }).toThrow(); + map.setZoom(0); + expect(map.getZoom()).toBe(0); + }); + + test('#setMaxZoom', () => { + const map = createMap({zoom: 0}); + map.setMaxZoom(3.5); + map.setZoom(4); + expect(map.getZoom()).toBe(3.5); + }); + + test('unset maxZoom', () => { + const map = createMap({maxZoom: 5}); + map.setMaxZoom(null); + map.setZoom(6); + expect(map.getZoom()).toBe(6); + }); + + test('#getMaxZoom', () => { + const map = createMap({zoom: 0}); + expect(map.getMaxZoom()).toBe(22); + map.setMaxZoom(10); + expect(map.getMaxZoom()).toBe(10); + }); + + test('ignore maxZooms over minZoom', () => { + const map = createMap({minZoom: 5}); + expect(() => { + map.setMaxZoom(4); + }).toThrow(); + map.setZoom(5); + expect(map.getZoom()).toBe(5); + }); + + test('throw on maxZoom smaller than minZoom at init', () => { + expect(() => { + createMap({minZoom: 10, maxZoom: 5}); + }).toThrow(new Error('maxZoom must be greater than or equal to minZoom')); + }); + + test('throw on maxZoom smaller than minZoom at init with falsey maxZoom', () => { + expect(() => { + createMap({minZoom: 1, maxZoom: 0}); + }).toThrow(new Error('maxZoom must be greater than or equal to minZoom')); + }); + + test('#setMinPitch', () => { + const map = createMap({pitch: 20}); + map.setMinPitch(10); + map.setPitch(0); + expect(map.getPitch()).toBe(10); + }); + + test('unset minPitch', () => { + const map = createMap({minPitch: 20}); + map.setMinPitch(null); + map.setPitch(0); + expect(map.getPitch()).toBe(0); + }); + + test('#getMinPitch', () => { + const map = createMap({pitch: 0}); + expect(map.getMinPitch()).toBe(0); + map.setMinPitch(10); + expect(map.getMinPitch()).toBe(10); + }); + + test('ignore minPitchs over maxPitch', () => { + const map = createMap({pitch: 0, maxPitch: 10}); + expect(() => { + map.setMinPitch(20); + }).toThrow(); + map.setPitch(0); + expect(map.getPitch()).toBe(0); + }); + + test('#setMaxPitch', () => { + const map = createMap({pitch: 0}); + map.setMaxPitch(10); + map.setPitch(20); + expect(map.getPitch()).toBe(10); + }); + + test('unset maxPitch', () => { + const map = createMap({maxPitch: 10}); + map.setMaxPitch(null); + map.setPitch(20); + expect(map.getPitch()).toBe(20); + }); + + test('#getMaxPitch', () => { + const map = createMap({pitch: 0}); + expect(map.getMaxPitch()).toBe(60); + map.setMaxPitch(10); + expect(map.getMaxPitch()).toBe(10); + }); + + test('ignore maxPitchs over minPitch', () => { + const map = createMap({minPitch: 10}); + expect(() => { + map.setMaxPitch(0); + }).toThrow(); + map.setPitch(10); + expect(map.getPitch()).toBe(10); + }); + + test('throw on maxPitch smaller than minPitch at init', () => { + expect(() => { + createMap({minPitch: 10, maxPitch: 5}); + }).toThrow(new Error('maxPitch must be greater than or equal to minPitch')); + }); + + test('throw on maxPitch smaller than minPitch at init with falsey maxPitch', () => { + expect(() => { + createMap({minPitch: 1, maxPitch: 0}); + }).toThrow(new Error('maxPitch must be greater than or equal to minPitch')); + }); + + test('throw on maxPitch greater than valid maxPitch at init', () => { + expect(() => { + createMap({maxPitch: 90}); + }).toThrow(new Error('maxPitch must be less than or equal to 85')); + }); + + test('throw on minPitch less than valid minPitch at init', () => { + expect(() => { + createMap({minPitch: -10}); + }).toThrow(new Error('minPitch must be greater than or equal to 0')); + }); + + test('#remove', () => { + const map = createMap(); + const spyWorkerPoolRelease = jest.spyOn(map.style.dispatcher.workerPool, 'release'); + expect(map.getContainer().childNodes).toHaveLength(2); + map.remove(); + expect(spyWorkerPoolRelease).toHaveBeenCalledTimes(1); + expect(map.getContainer().childNodes).toHaveLength(0); + + // Cleanup + spyWorkerPoolRelease.mockClear(); + }); + + test('#remove calls onRemove on added controls', () => { + const map = createMap(); + const control = { + onRemove: jest.fn(), + onAdd(_) { + return window.document.createElement('div'); + } + }; + map.addControl(control); + map.remove(); + expect(control.onRemove).toHaveBeenCalledTimes(1); + }); + + test('#remove calls onRemove on added controls before style is destroyed', done => { + const map = createMap(); + let onRemoveCalled = 0; + let style; + const control = { + onRemove(map) { + onRemoveCalled++; + expect(map.getStyle()).toEqual(style); + }, + onAdd(_) { + return window.document.createElement('div'); + } + }; + + map.addControl(control); + + map.on('style.load', () => { + style = map.getStyle(); + map.remove(); + expect(onRemoveCalled).toBe(1); + done(); + }); + }); + + test('does not fire "webglcontextlost" after #remove has been called', done => { + const map = createMap(); + const canvas = map.getCanvas(); + map.once('webglcontextlost', () => done('"webglcontextlost" fired after #remove has been called')); + map.remove(); + // Dispatch the event manually because at the time of this writing, gl does not support + // the WEBGL_lose_context extension. + canvas.dispatchEvent(new window.Event('webglcontextlost')); + done(); + }); + + test('does not fire "webglcontextrestored" after #remove has been called', done => { + const map = createMap(); + const canvas = map.getCanvas(); + + map.once('webglcontextlost', () => { + map.once('webglcontextrestored', () => done('"webglcontextrestored" fired after #remove has been called')); + map.remove(); + canvas.dispatchEvent(new window.Event('webglcontextrestored')); + done(); + }); + + // Dispatch the event manually because at the time of this writing, gl does not support + // the WEBGL_lose_context extension. + canvas.dispatchEvent(new window.Event('webglcontextlost')); + }); + + test('#redraw', async () => { + const map = createMap(); + + await map.once('idle'); + const renderPromise = map.once('render'); + + map.redraw(); + await renderPromise; + }); + + test('#addControl', () => { + const map = createMap(); + const control = { + onAdd(_) { + expect(map).toBe(_); + return window.document.createElement('div'); + } + } as any as IControl; + map.addControl(control); + expect(map._controls[0]).toBe(control); + }); + + test('#removeControl errors on invalid arguments', () => { + const map = createMap(); + const control = {} as any as IControl; + const stub = jest.spyOn(console, 'error').mockImplementation(() => {}); + + map.addControl(control); + map.removeControl(control); + expect(stub).toHaveBeenCalledTimes(2); + + }); + + test('#removeControl', () => { + const map = createMap(); + const control = { + onAdd() { + return window.document.createElement('div'); + }, + onRemove(_) { + expect(map).toBe(_); + } + }; + map.addControl(control); + map.removeControl(control); + expect(map._controls).toHaveLength(0); + + }); + + test('#hasControl', () => { + const map = createMap(); + function Ctrl() {} + Ctrl.prototype = { + onAdd(_) { + return window.document.createElement('div'); + } + }; + + const control = new Ctrl(); + expect(map.hasControl(control)).toBe(false); + map.addControl(control); + expect(map.hasControl(control)).toBe(true); + }); + + test('#project', () => { + const map = createMap(); + expect(map.project([0, 0])).toEqual({x: 100, y: 100}); + }); + + test('#unproject', () => { + const map = createMap(); + expect(fixedLngLat(map.unproject([100, 100]))).toEqual({lng: 0, lat: 0}); + }); + + test('#listImages', done => { + const map = createMap(); + + map.on('load', () => { + expect(map.listImages()).toHaveLength(0); + + map.addImage('img', {width: 1, height: 1, data: new Uint8Array(4)}); + + const images = map.listImages(); + expect(images).toHaveLength(1); + expect(images[0]).toBe('img'); + done(); + }); + }); + + test('#listImages throws an error if called before "load"', () => { + const map = createMap(); + expect(() => { + map.listImages(); + }).toThrow(Error); + }); + + describe('#queryRenderedFeatures', () => { + + test('if no arguments provided', done => { + createMap({}, (err, map) => { + expect(err).toBeFalsy(); + const spy = jest.spyOn(map.style, 'queryRenderedFeatures'); + + const output = map.queryRenderedFeatures(); + + const args = spy.mock.calls[0]; + expect(args[0]).toBeTruthy(); + expect(args[1]).toEqual({availableImages: []}); + expect(output).toEqual([]); + + done(); + }); + }); + + test('if only "geometry" provided', done => { + createMap({}, (err, map) => { + expect(err).toBeFalsy(); + const spy = jest.spyOn(map.style, 'queryRenderedFeatures'); + + const output = map.queryRenderedFeatures(map.project(new LngLat(0, 0))); + + const args = spy.mock.calls[0]; + expect(args[0]).toEqual([{x: 100, y: 100}]); // query geometry + expect(args[1]).toEqual({availableImages: []}); // params + expect(args[2]).toEqual(map.transform); // transform + expect(output).toEqual([]); + + done(); + }); + }); + + test('if only "params" provided', done => { + createMap({}, (err, map) => { + expect(err).toBeFalsy(); + const spy = jest.spyOn(map.style, 'queryRenderedFeatures'); + + const output = map.queryRenderedFeatures({filter: ['all']}); + + const args = spy.mock.calls[0]; + expect(args[0]).toBeTruthy(); + expect(args[1]).toEqual({availableImages: [], filter: ['all']}); + expect(output).toEqual([]); + + done(); + }); + }); + + test('if both "geometry" and "params" provided', done => { + createMap({}, (err, map) => { + expect(err).toBeFalsy(); + const spy = jest.spyOn(map.style, 'queryRenderedFeatures'); + + const output = map.queryRenderedFeatures({filter: ['all']}); + + const args = spy.mock.calls[0]; + expect(args[0]).toBeTruthy(); + expect(args[1]).toEqual({availableImages: [], filter: ['all']}); + expect(output).toEqual([]); + + done(); + }); + }); + + test('if "geometry" with unwrapped coords provided', done => { + createMap({}, (err, map) => { + expect(err).toBeFalsy(); + const spy = jest.spyOn(map.style, 'queryRenderedFeatures'); + + map.queryRenderedFeatures(map.project(new LngLat(360, 0))); + + expect(spy.mock.calls[0][0]).toEqual([{x: 612, y: 100}]); + done(); + }); + }); + + test('returns an empty array when no style is loaded', () => { + const map = createMap({style: undefined}); + expect(map.queryRenderedFeatures()).toEqual([]); + }); + + }); + + describe('#setLayoutProperty', () => { + test('sets property', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': { + 'type': 'geojson', + 'data': { + 'type': 'FeatureCollection', + 'features': [] + } + } + }, + 'layers': [{ + 'id': 'symbol', + 'type': 'symbol', + 'source': 'geojson', + 'layout': { + 'text-transform': 'uppercase' + } + }] + } + }); + + map.on('style.load', () => { + map.style.dispatcher.broadcast = function (key, value: any) { + expect(key).toBe('updateLayers'); + expect(value.layers.map((layer) => { return layer.id; })).toEqual(['symbol']); + }; + + map.setLayoutProperty('symbol', 'text-transform', 'lowercase'); + map.style.update({} as EvaluationParameters); + expect(map.getLayoutProperty('symbol', 'text-transform')).toBe('lowercase'); + done(); + }); + }); + + test('throw before loaded', () => { + const map = createMap({ + style: { + version: 8, + sources: {}, + layers: [] + } + }); + + expect(() => { + map.setLayoutProperty('symbol', 'text-transform', 'lowercase'); + }).toThrow(Error); + + }); + + test('fires an error if layer not found', done => { + const map = createMap({ + style: { + version: 8, + sources: {}, + layers: [] + } + }); + + map.on('style.load', () => { + map.on('error', ({error}) => { + expect(error.message).toMatch(/Cannot style non-existing layer "non-existant"./); + done(); + }); + map.setLayoutProperty('non-existant', 'text-transform', 'lowercase'); + }); + }); + + test('fires a data event', async () => { + // background layers do not have a source + const map = createMap({ + style: { + 'version': 8, + 'sources': {}, + 'layers': [{ + 'id': 'background', + 'type': 'background', + 'layout': { + 'visibility': 'none' + } + }] + } + }); + + await map.once('style.load'); + const dataPromise = map.once('data'); + map.setLayoutProperty('background', 'visibility', 'visible'); + const e = await dataPromise; + expect(e.dataType).toBe('style'); + }); + + test('sets visibility on background layer', done => { + // background layers do not have a source + const map = createMap({ + style: { + 'version': 8, + 'sources': {}, + 'layers': [{ + 'id': 'background', + 'type': 'background', + 'layout': { + 'visibility': 'none' + } + }] + } + }); + + map.on('style.load', () => { + map.setLayoutProperty('background', 'visibility', 'visible'); + expect(map.getLayoutProperty('background', 'visibility')).toBe('visible'); + done(); + }); + }); + + test('sets visibility on raster layer', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'maplibre-satellite': { + 'type': 'raster', + 'tiles': ['http://example.com/{z}/{x}/{y}.png'] + } + }, + 'layers': [{ + 'id': 'satellite', + 'type': 'raster', + 'source': 'maplibre-satellite', + 'layout': { + 'visibility': 'none' + } + }] + } + }); + + // Suppress errors because we're not loading tiles from a real URL. + map.on('error', () => {}); + + map.on('style.load', () => { + map.setLayoutProperty('satellite', 'visibility', 'visible'); + expect(map.getLayoutProperty('satellite', 'visibility')).toBe('visible'); + done(); + }); + }); + + test('sets visibility on video layer', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'drone': { + 'type': 'video', + 'urls': [], + 'coordinates': [ + [-122.51596391201019, 37.56238816766053], + [-122.51467645168304, 37.56410183312965], + [-122.51309394836426, 37.563391708549425], + [-122.51423120498657, 37.56161849366671] + ] + } + }, + 'layers': [{ + 'id': 'shore', + 'type': 'raster', + 'source': 'drone', + 'layout': { + 'visibility': 'none' + } + }] + } + }); + + map.on('style.load', () => { + map.setLayoutProperty('shore', 'visibility', 'visible'); + expect(map.getLayoutProperty('shore', 'visibility')).toBe('visible'); + done(); + }); + }); + + test('sets visibility on image layer', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'image': { + 'type': 'image', + 'url': '', + 'coordinates': [ + [-122.51596391201019, 37.56238816766053], + [-122.51467645168304, 37.56410183312965], + [-122.51309394836426, 37.563391708549425], + [-122.51423120498657, 37.56161849366671] + ] + } + }, + 'layers': [{ + 'id': 'image', + 'type': 'raster', + 'source': 'image', + 'layout': { + 'visibility': 'none' + } + }] + } + }); + + map.on('style.load', () => { + map.setLayoutProperty('image', 'visibility', 'visible'); + expect(map.getLayoutProperty('image', 'visibility')).toBe('visible'); + done(); + }); + }); + + }); + + describe('#getLayoutProperty', () => { + test('fires an error if layer not found', done => { + const map = createMap({ + style: { + version: 8, + sources: {}, + layers: [] + } + }); + + map.on('style.load', () => { + map.on('error', ({error}) => { + expect(error.message).toMatch(/Cannot get style of non-existing layer "non-existant"./); + done(); + }); + (map as any).getLayoutProperty('non-existant', 'text-transform', 'lowercase'); + }); + }); + + }); + + describe('#setPaintProperty', () => { + test('sets property', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': {}, + 'layers': [{ + 'id': 'background', + 'type': 'background' + }] + } + }); + + map.on('style.load', () => { + map.setPaintProperty('background', 'background-color', 'red'); + expect(map.getPaintProperty('background', 'background-color')).toBe('red'); + done(); + }); + }); + + test('throw before loaded', () => { + const map = createMap({ + style: { + version: 8, + sources: {}, + layers: [] + } + }); + + expect(() => { + map.setPaintProperty('background', 'background-color', 'red'); + }).toThrow(Error); + + }); + + test('fires an error if layer not found', done => { + const map = createMap({ + style: { + version: 8, + sources: {}, + layers: [] + } + }); + + map.on('style.load', () => { + map.on('error', ({error}) => { + expect(error.message).toMatch(/Cannot style non-existing layer "non-existant"./); + done(); + }); + map.setPaintProperty('non-existant', 'background-color', 'red'); + }); + }); + + }); + + describe('#setFeatureState', () => { + test('sets state', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.setFeatureState({source: 'geojson', id: 12345}, {'hover': true}); + const fState = map.getFeatureState({source: 'geojson', id: 12345}); + expect(fState.hover).toBe(true); + done(); + }); + }); + test('works with string ids', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.setFeatureState({source: 'geojson', id: 'foo'}, {'hover': true}); + const fState = map.getFeatureState({source: 'geojson', id: 'foo'}); + expect(fState.hover).toBe(true); + done(); + }); + }); + test('parses feature id as an int', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.setFeatureState({source: 'geojson', id: '12345'}, {'hover': true}); + const fState = map.getFeatureState({source: 'geojson', id: 12345}); + expect(fState.hover).toBe(true); + done(); + }); + }); + test('throw before loaded', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + expect(() => { + map.setFeatureState({source: 'geojson', id: 12345}, {'hover': true}); + }).toThrow(Error); + + done(); + }); + test('fires an error if source not found', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.on('error', ({error}) => { + expect(error.message).toMatch(/source/); + done(); + }); + map.setFeatureState({source: 'vector', id: 12345}, {'hover': true}); + }); + }); + test('fires an error if sourceLayer not provided for a vector source', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'vector': { + 'type': 'vector', + 'tiles': ['http://example.com/{z}/{x}/{y}.png'] + } + }, + 'layers': [] + } + }); + map.on('load', () => { + map.on('error', ({error}) => { + expect(error.message).toMatch(/sourceLayer/); + done(); + }); + (map as any).setFeatureState({source: 'vector', sourceLayer: 0, id: 12345}, {'hover': true}); + }); + }); + test('fires an error if id not provided', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'vector': { + 'type': 'vector', + 'tiles': ['http://example.com/{z}/{x}/{y}.png'] + } + }, + 'layers': [] + } + }); + map.on('load', () => { + map.on('error', ({error}) => { + expect(error.message).toMatch(/id/); + done(); + }); + (map as any).setFeatureState({source: 'vector', sourceLayer: '1'}, {'hover': true}); + }); + }); + }); + + describe('#removeFeatureState', () => { + + test('accepts "0" id', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.setFeatureState({source: 'geojson', id: 0}, {'hover': true, 'click': true}); + map.removeFeatureState({source: 'geojson', id: 0}, 'hover'); + const fState = map.getFeatureState({source: 'geojson', id: 0}); + expect(fState.hover).toBeUndefined(); + expect(fState.click).toBe(true); + done(); + }); + }); + test('accepts string id', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.setFeatureState({source: 'geojson', id: 'foo'}, {'hover': true, 'click': true}); + map.removeFeatureState({source: 'geojson', id: 'foo'}, 'hover'); + const fState = map.getFeatureState({source: 'geojson', id: 'foo'}); + expect(fState.hover).toBeUndefined(); + expect(fState.click).toBe(true); + done(); + }); + }); + test('remove specific state property', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.setFeatureState({source: 'geojson', id: 12345}, {'hover': true}); + map.removeFeatureState({source: 'geojson', id: 12345}, 'hover'); + const fState = map.getFeatureState({source: 'geojson', id: 12345}); + expect(fState.hover).toBeUndefined(); + done(); + }); + }); + test('remove all state properties of one feature', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.setFeatureState({source: 'geojson', id: 1}, {'hover': true, 'foo': true}); + map.removeFeatureState({source: 'geojson', id: 1}); + + const fState = map.getFeatureState({source: 'geojson', id: 1}); + expect(fState.hover).toBeUndefined(); + expect(fState.foo).toBeUndefined(); + + done(); + }); + }); + test('remove properties for zero-based feature IDs.', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.setFeatureState({source: 'geojson', id: 0}, {'hover': true, 'foo': true}); + map.removeFeatureState({source: 'geojson', id: 0}); + + const fState = map.getFeatureState({source: 'geojson', id: 0}); + expect(fState.hover).toBeUndefined(); + expect(fState.foo).toBeUndefined(); + + done(); + }); + }); + test('other properties persist when removing specific property', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.setFeatureState({source: 'geojson', id: 1}, {'hover': true, 'foo': true}); + map.removeFeatureState({source: 'geojson', id: 1}, 'hover'); + + const fState = map.getFeatureState({source: 'geojson', id: 1}); + expect(fState.foo).toBe(true); + + done(); + }); + }); + test('remove all state properties of all features in source', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.setFeatureState({source: 'geojson', id: 1}, {'hover': true, 'foo': true}); + map.setFeatureState({source: 'geojson', id: 2}, {'hover': true, 'foo': true}); + + map.removeFeatureState({source: 'geojson'}); + + const fState1 = map.getFeatureState({source: 'geojson', id: 1}); + expect(fState1.hover).toBeUndefined(); + expect(fState1.foo).toBeUndefined(); + + const fState2 = map.getFeatureState({source: 'geojson', id: 2}); + expect(fState2.hover).toBeUndefined(); + expect(fState2.foo).toBeUndefined(); + + done(); + }); + }); + test('specific state deletion should not interfere with broader state deletion', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.setFeatureState({source: 'geojson', id: 1}, {'hover': true, 'foo': true}); + map.setFeatureState({source: 'geojson', id: 2}, {'hover': true, 'foo': true}); + + map.removeFeatureState({source: 'geojson', id: 1}); + map.removeFeatureState({source: 'geojson', id: 1}, 'foo'); + + const fState1 = map.getFeatureState({source: 'geojson', id: 1}); + expect(fState1.hover).toBeUndefined(); + + map.setFeatureState({source: 'geojson', id: 1}, {'hover': true, 'foo': true}); + map.removeFeatureState({source: 'geojson'}); + map.removeFeatureState({source: 'geojson', id: 1}, 'foo'); + + const fState2 = map.getFeatureState({source: 'geojson', id: 2}); + expect(fState2.hover).toBeUndefined(); + + map.setFeatureState({source: 'geojson', id: 2}, {'hover': true, 'foo': true}); + map.removeFeatureState({source: 'geojson'}); + map.removeFeatureState({source: 'geojson', id: 2}, 'foo'); + + const fState3 = map.getFeatureState({source: 'geojson', id: 2}); + expect(fState3.hover).toBeUndefined(); + + done(); + }); + }); + test('add/remove and remove/add state', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.setFeatureState({source: 'geojson', id: 12345}, {'hover': true}); + + map.removeFeatureState({source: 'geojson', id: 12345}); + map.setFeatureState({source: 'geojson', id: 12345}, {'hover': true}); + + const fState1 = map.getFeatureState({source: 'geojson', id: 12345}); + expect(fState1.hover).toBe(true); + + map.removeFeatureState({source: 'geojson', id: 12345}); + + const fState2 = map.getFeatureState({source: 'geojson', id: 12345}); + expect(fState2.hover).toBeUndefined(); + + done(); + }); + }); + test('throw before loaded', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + expect(() => { + (map as any).removeFeatureState({source: 'geojson', id: 12345}, {'hover': true}); + }).toThrow(Error); + + done(); + }); + test('fires an error if source not found', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'geojson': createStyleSource() + }, + 'layers': [] + } + }); + map.on('load', () => { + map.on('error', ({error}) => { + expect(error.message).toMatch(/source/); + done(); + }); + (map as any).removeFeatureState({source: 'vector', id: 12345}, {'hover': true}); + }); + }); + test('fires an error if sourceLayer not provided for a vector source', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'vector': { + 'type': 'vector', + 'tiles': ['http://example.com/{z}/{x}/{y}.png'] + } + }, + 'layers': [] + } + }); + map.on('load', () => { + map.on('error', ({error}) => { + expect(error.message).toMatch(/sourceLayer/); + done(); + }); + (map as any).removeFeatureState({source: 'vector', sourceLayer: 0, id: 12345}, {'hover': true}); + }); + }); + test('fires an error if state property is provided without a feature id', done => { + const map = createMap({ + style: { + 'version': 8, + 'sources': { + 'vector': { + 'type': 'vector', + 'tiles': ['http://example.com/{z}/{x}/{y}.png'] + } + }, + 'layers': [] + } + }); + map.on('load', () => { + map.on('error', ({error}) => { + expect(error.message).toMatch(/id/); + done(); + }); + (map as any).removeFeatureState({source: 'vector', sourceLayer: '1'}, {'hover': true}); + }); + }); + }); + + describe('error event', () => { + test('logs errors to console when it has NO listeners', () => { + const map = createMap(); + const stub = jest.spyOn(console, 'error').mockImplementation(() => {}); + stub.mockReset(); + const error = new Error('test'); + map.fire(new ErrorEvent(error)); + expect(stub).toHaveBeenCalledTimes(1); + expect(stub.mock.calls[0][0]).toBe(error); + }); + + test('calls listeners', done => { + const map = createMap(); + const error = new Error('test'); + map.on('error', (event) => { + expect(event.error).toBe(error); + done(); + }); + map.fire(new ErrorEvent(error)); + }); + + }); + + test('render stabilizes', done => { + const style = createStyle(); + style.sources.mapbox = { + type: 'vector', + minzoom: 1, + maxzoom: 10, + tiles: ['http://example.com/{z}/{x}/{y}.png'] + }; + style.layers.push({ + id: 'layerId', + type: 'circle', + source: 'mapbox', + 'source-layer': 'sourceLayer' + }); + + let timer; + const map = createMap({style}); + map.on('render', () => { + if (timer) clearTimeout(timer); + timer = setTimeout(() => { + map.off('render', undefined); + map.on('render', () => { + done('test failed'); + }); + expect((map as any)._frameId).toBeFalsy(); + done(); + }, 100); + }); + }); + + test('no render after idle event', done => { + const style = createStyle(); + const map = createMap({style}); + map.on('idle', () => { + map.on('render', () => { + done('test failed'); + }); + setTimeout(() => { + done(); + }, 100); + }); + }); + + test('no render before style loaded', done => { + server.respondWith('/styleUrl', JSON.stringify(createStyle())); + const map = createMap({style: '/styleUrl'}); + + jest.spyOn(map, 'triggerRepaint').mockImplementationOnce(() => { + if (!map.style._loaded) { + done('test failed'); + } + }); + map.on('render', () => { + if (map.style._loaded) { + done(); + } else { + done('test failed'); + } + }); + + // Force a update should not call triggerRepaint till style is loaded. + // Once style is loaded, it will trigger the update. + map._update(); + server.respond(); + }); + + test('no idle event during move', async () => { + const style = createStyle(); + const map = createMap({style, fadeDuration: 0}); + await map.once('idle'); + map.zoomTo(0.5, {duration: 100}); + expect(map.isMoving()).toBeTruthy(); + await map.once('idle'); + expect(map.isMoving()).toBeFalsy(); + }); + + test('#removeLayer restores Map#loaded() to true', done => { + const map = createMap({ + style: extend(createStyle(), { + sources: { + mapbox: { + type: 'vector', + minzoom: 1, + maxzoom: 10, + tiles: ['http://example.com/{z}/{x}/{y}.png'] + } + }, + layers: [{ + id: 'layerId', + type: 'circle', + source: 'mapbox', + 'source-layer': 'sourceLayer' + }] + }) + }); + + map.once('render', () => { + map.removeLayer('layerId'); + map.on('render', () => { + if (map.loaded()) { + map.remove(); + done(); + } + }); + }); + }); + + test('stops camera animation on mousedown when interactive', () => { + const map = createMap({interactive: true}); + map.flyTo({center: [200, 0], duration: 100}); + + simulate.mousedown(map.getCanvasContainer()); + expect(map.isEasing()).toBe(false); + + map.remove(); + }); + + test('continues camera animation on mousedown when non-interactive', () => { + const map = createMap({interactive: false}); + map.flyTo({center: [200, 0], duration: 100}); + + simulate.mousedown(map.getCanvasContainer()); + expect(map.isEasing()).toBe(true); + + map.remove(); + }); + + test('stops camera animation on touchstart when interactive', () => { + const map = createMap({interactive: true}); + map.flyTo({center: [200, 0], duration: 100}); + + simulate.touchstart(map.getCanvasContainer(), {touches: [{target: map.getCanvas(), clientX: 0, clientY: 0}]}); + expect(map.isEasing()).toBe(false); + + map.remove(); + }); + + test('continues camera animation on touchstart when non-interactive', () => { + const map = createMap({interactive: false}); + map.flyTo({center: [200, 0], duration: 100}); + + simulate.touchstart(map.getCanvasContainer()); + expect(map.isEasing()).toBe(true); + + map.remove(); + }); + + test('continues camera animation on resize', () => { + const map = createMap(), + container = map.getContainer(); + + map.flyTo({center: [200, 0], duration: 100}); + + Object.defineProperty(container, 'clientWidth', {value: 250}); + Object.defineProperty(container, 'clientHeight', {value: 250}); + map.resize(); + + expect(map.isMoving()).toBeTruthy(); + + }); + + test('map fires `styleimagemissing` for missing icons', done => { + const map = createMap(); + + const id = 'missing-image'; + + const sampleImage = {width: 2, height: 1, data: new Uint8Array(8)}; + + let called: string; + map.on('styleimagemissing', e => { + map.addImage(e.id, sampleImage); + called = e.id; + }); + + expect(map.hasImage(id)).toBeFalsy(); + + map.style.imageManager.getImages([id], (alwaysNull, generatedImage) => { + expect(generatedImage[id].data.width).toEqual(sampleImage.width); + expect(generatedImage[id].data.height).toEqual(sampleImage.height); + expect(generatedImage[id].data.data).toEqual(sampleImage.data); + expect(called).toBe(id); + expect(map.hasImage(id)).toBeTruthy(); + done(); + }); + }); + + test('map getImage matches addImage, uintArray', () => { + const map = createMap(); + const id = 'add-get-uint'; + const inputImage = {width: 2, height: 1, data: new Uint8Array(8)}; + + map.addImage(id, inputImage); + expect(map.hasImage(id)).toBeTruthy(); + + const gotImage = map.getImage(id); + expect(gotImage.data.width).toEqual(inputImage.width); + expect(gotImage.data.height).toEqual(inputImage.height); + expect(gotImage.sdf).toBe(false); + }); + + test('map getImage matches addImage, uintClampedArray', () => { + const map = createMap(); + const id = 'add-get-uint-clamped'; + const inputImage = {width: 1, height: 2, data: new Uint8ClampedArray(8)}; + + map.addImage(id, inputImage); + expect(map.hasImage(id)).toBeTruthy(); + + const gotImage = map.getImage(id); + expect(gotImage.data.width).toEqual(inputImage.width); + expect(gotImage.data.height).toEqual(inputImage.height); + expect(gotImage.sdf).toBe(false); + }); + + test('map getImage matches addImage, ImageData', () => { + const map = createMap(); + const id = 'add-get-image-data'; + const inputImage = new ImageData(1, 3); + + map.addImage(id, inputImage); + expect(map.hasImage(id)).toBeTruthy(); + + const gotImage = map.getImage(id); + expect(gotImage.data.width).toEqual(inputImage.width); + expect(gotImage.data.height).toEqual(inputImage.height); + expect(gotImage.sdf).toBe(false); + }); + + test('map getImage matches addImage, StyleImageInterface uint', () => { + const map = createMap(); + const id = 'add-get-style-image-iface-uint'; + const inputImage: StyleImageInterface = { + width: 3, + height: 1, + data: new Uint8Array(12) + }; + + map.addImage(id, inputImage); + expect(map.hasImage(id)).toBeTruthy(); + + const gotImage = map.getImage(id); + expect(gotImage.data.width).toEqual(inputImage.width); + expect(gotImage.data.height).toEqual(inputImage.height); + expect(gotImage.sdf).toBe(false); + }); + + test('map getImage matches addImage, StyleImageInterface clamped', () => { + const map = createMap(); + const id = 'add-get-style-image-iface-clamped'; + const inputImage: StyleImageInterface = { + width: 4, + height: 1, + data: new Uint8ClampedArray(16) + }; + + map.addImage(id, inputImage); + expect(map.hasImage(id)).toBeTruthy(); + + const gotImage = map.getImage(id); + expect(gotImage.data.width).toEqual(inputImage.width); + expect(gotImage.data.height).toEqual(inputImage.height); + expect(gotImage.sdf).toBe(false); + }); + + test('map getImage matches addImage, StyleImageInterface SDF', () => { + const map = createMap(); + const id = 'add-get-style-image-iface-sdf'; + const inputImage: StyleImageInterface = { + width: 5, + height: 1, + data: new Uint8Array(20) + }; + + map.addImage(id, inputImage, {sdf: true}); + expect(map.hasImage(id)).toBeTruthy(); + + const gotImage = map.getImage(id); + expect(gotImage.data.width).toEqual(inputImage.width); + expect(gotImage.data.height).toEqual(inputImage.height); + expect(gotImage.sdf).toBe(true); + }); + + test('map does not fire `styleimagemissing` for empty icon values', done => { + const map = createMap(); + + map.on('load', () => { + map.on('idle', () => { + done(); + }); + + map.addSource('foo', { + type: 'geojson', + data: {type: 'Point', coordinates: [0, 0]} + }); + map.addLayer({ + id: 'foo', + type: 'symbol', + source: 'foo', + layout: { + 'icon-image': ['case', true, '', ''] + } + }); + + map.on('styleimagemissing', ({id}) => { + done(`styleimagemissing fired for value ${id}`); + }); + }); + }); + + describe('setPixelRatio', () => { + test('resizes canvas', () => { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'clientWidth', {value: 512}); + Object.defineProperty(container, 'clientHeight', {value: 512}); + const map = createMap({container, pixelRatio: 1}); + expect(map.getCanvas().width).toBe(512); + expect(map.getCanvas().height).toBe(512); + map.setPixelRatio(2); + expect(map.getCanvas().width).toBe(1024); + expect(map.getCanvas().height).toBe(1024); + }); + + test('resizes painter', () => { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'clientWidth', {value: 512}); + Object.defineProperty(container, 'clientHeight', {value: 512}); + const map = createMap({container, pixelRatio: 1}); + expect(map.painter.pixelRatio).toBe(1); + expect(map.painter.width).toBe(512); + expect(map.painter.height).toBe(512); + map.setPixelRatio(2); + expect(map.painter.pixelRatio).toBe(2); + expect(map.painter.width).toBe(1024); + expect(map.painter.height).toBe(1024); + }); + }); + + describe('getPixelRatio', () => { + test('returns the pixel ratio', () => { + const map = createMap({pixelRatio: 1}); + expect(map.getPixelRatio()).toBe(1); + map.setPixelRatio(2); + expect(map.getPixelRatio()).toBe(2); + }); + }); + + test('pixel ratio defaults to devicePixelRatio', () => { + const map = createMap(); + expect(map.getPixelRatio()).toBe(devicePixelRatio); + }); + + test('pixel ratio by default reflects devicePixelRatio changes', () => { + global.devicePixelRatio = 0.25; + const map = createMap(); + expect(map.getPixelRatio()).toBe(0.25); + global.devicePixelRatio = 1; + expect(map.getPixelRatio()).toBe(1); + }); + + test('canvas has the expected size', () => { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'clientWidth', {value: 512}); + Object.defineProperty(container, 'clientHeight', {value: 512}); + const map = createMap({container, pixelRatio: 2}); + expect(map.getCanvas().width).toBe(1024); + expect(map.getCanvas().height).toBe(1024); + }); + + test('painter has the expected size and pixel ratio', () => { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'clientWidth', {value: 512}); + Object.defineProperty(container, 'clientHeight', {value: 512}); + const map = createMap({container, pixelRatio: 2}); + expect(map.painter.pixelRatio).toBe(2); + expect(map.painter.width).toBe(1024); + expect(map.painter.height).toBe(1024); + }); + + test('fires sourcedataabort event on dataabort event', async () => { + const map = createMap(); + const sourcePromise = map.once('sourcedataabort'); + map.fire(new Event('dataabort')); + await sourcePromise; + }); + + describe('#setTerrain', () => { + test('warn when terrain and hillshade source identical', done => { + server.respondWith('/source.json', JSON.stringify({ + minzoom: 5, + maxzoom: 12, + attribution: 'Terrain', + tiles: ['http://example.com/{z}/{x}/{y}.pngraw'], + bounds: [-47, -7, -45, -5] + })); + + const map = createMap(); + + map.on('load', () => { + map.addSource('terrainrgb', {type: 'raster-dem', url: '/source.json'}); + server.respond(); + map.addLayer({id: 'hillshade', type: 'hillshade', source: 'terrainrgb'}); + const stub = jest.spyOn(console, 'warn').mockImplementation(() => { }); + stub.mockReset(); + map.setTerrain({ + source: 'terrainrgb' + }); + expect(console.warn).toHaveBeenCalledTimes(1); + done(); + }); + }); + }); + + describe('#getTerrain', () => { + test('returns null when not set', () => { + const map = createMap(); + expect(map.getTerrain()).toBeNull(); + }); + }); + + describe('#setCooperativeGestures', () => { + test('returns self', () => { + const map = createMap(); + expect(map.setCooperativeGestures(true)).toBe(map); + }); + + test('can be called more than once', () => { + const map = createMap(); + map.setCooperativeGestures(true); + map.setCooperativeGestures(true); + }); + + test('calling set with no arguments turns cooperative gestures off', done => { + const map = createMap({cooperativeGestures: true}); + map.on('load', () => { + map.setCooperativeGestures(); + expect(map.getCooperativeGestures()).toBeFalsy(); + done(); + }); + }); + }); + + describe('#getCooperativeGestures', () => { + test('returns the cooperative gestures option', done => { + const map = createMap({cooperativeGestures: true}); + + map.on('load', () => { + expect(map.getCooperativeGestures()).toBe(true); + done(); + }); + }); + + test('returns falsy if cooperative gestures option is not specified', done => { + const map = createMap(); + + map.on('load', () => { + expect(map.getCooperativeGestures()).toBeFalsy(); + done(); + }); + }); + + test('returns the cooperative gestures option with custom messages', done => { + const option = { + 'windowsHelpText': 'Custom message', + 'macHelpText': 'Custom message', + 'mobileHelpText': 'Custom message', + }; + const map = createMap({cooperativeGestures: option}); + + map.on('load', () => { + expect(map.getCooperativeGestures()).toEqual(option); + done(); + }); + }); + }); + + describe('cooperativeGestures option', () => { + test('cooperativeGesture container element is hidden from a11y tree', () => { + const map = createMap({cooperativeGestures: true}); + + expect(map.getContainer().querySelector('.maplibregl-cooperative-gesture-screen').getAttribute('aria-hidden')).toBeTruthy(); + }); + }); + + describe('getCameraTargetElevation', () => { + test('Elevation is zero without terrain, and matches any given terrain', () => { + const map = createMap(); + expect(map.getCameraTargetElevation()).toBe(0); + + const terrainStub = {} as Terrain; + map.terrain = terrainStub; + + const transform = new Transform(0, 22, 0, 60, true); + transform.elevation = 200; + transform.center = new LngLat(10.0, 50.0); + transform.zoom = 14; + transform.resize(512, 512); + transform.elevation = 2000; + map.transform = transform; + + expect(map.getCameraTargetElevation()).toBe(2000); + }); + }); + + describe('#calculateCameraOptionsFromTo', () => { + // Choose initial zoom to avoid center being constrained by mercator latitude limits. + test('pitch 90 with terrain', () => { + const map = createMap(); + + const mockedGetElevation = jest.fn((_lngLat: LngLat, _zoom: number) => 111200); + + const terrainStub = {} as Terrain; + terrainStub.getElevationForLngLatZoom = mockedGetElevation; + map.terrain = terrainStub; + + // distance between lng x and lng x+1 is 111.2km at same lat + // altitude same as center elevation => 90° pitch + const cameraOptions: CameraOptions = map.calculateCameraOptionsFromTo(new LngLat(1, 0), 111200, new LngLat(0, 0)); + expect(cameraOptions).toBeDefined(); + expect(cameraOptions.pitch).toBeCloseTo(90); + expect(mockedGetElevation.mock.calls).toHaveLength(1); + }); + + test('pitch 153.435 with terrain', () => { + const map = createMap(); + + const mockedGetElevation = jest.fn((_lngLat: LngLat, _zoom: number) => 111200 * 3); + + const terrainStub = {} as Terrain; + terrainStub.getElevationForLngLatZoom = mockedGetElevation; + map.terrain = terrainStub; + // distance between lng x and lng x+1 is 111.2km at same lat + // (elevation difference of cam and center) / 2 = grounddistance => + // acos(111.2 / sqrt(111.2² + (111.2 * 2)²)) = acos(1/sqrt(5)) => 63.435 + 90 = 153.435 + const cameraOptions: CameraOptions = map.calculateCameraOptionsFromTo(new LngLat(1, 0), 111200, new LngLat(0, 0)); + expect(cameraOptions).toBeDefined(); + expect(cameraOptions.pitch).toBeCloseTo(153.435); + expect(mockedGetElevation.mock.calls).toHaveLength(1); + }); + + test('pitch 63 with terrain', () => { + const map = createMap(); + + const mockedGetElevation = jest.fn((_lngLat: LngLat, _zoom: number) => 111200 / 2); + + const terrainStub = {} as Terrain; + terrainStub.getElevationForLngLatZoom = mockedGetElevation; + map.terrain = terrainStub; + + // distance between lng x and lng x+1 is 111.2km at same lat + // (elevation difference of cam and center) * 2 = grounddistance => + // acos(111.2 / sqrt(111.2² + (111.2 * 0.5)²)) = acos(1/sqrt(1.25)) => 90 (looking down) - 26.565 = 63.435 + const cameraOptions: CameraOptions = map.calculateCameraOptionsFromTo(new LngLat(0, 0), 111200, new LngLat(1, 0)); + expect(cameraOptions).toBeDefined(); + expect(cameraOptions.pitch).toBeCloseTo(63.435); + expect(mockedGetElevation.mock.calls).toHaveLength(1); + }); + + test('zoom distance 1000', () => { + const map = createMap(); + + const mockedGetElevation = jest.fn((_lngLat: LngLat, _zoom: number) => 1000); + + const terrainStub = {} as Terrain; + terrainStub.getElevationForLngLatZoom = mockedGetElevation; + map.terrain = terrainStub; + + const expectedZoom = Math.log2(map.transform.cameraToCenterDistance / mercatorZfromAltitude(1000, 0) / map.transform.tileSize); + const cameraOptions = map.calculateCameraOptionsFromTo(new LngLat(0, 0), 0, new LngLat(0, 0)); + + expect(cameraOptions).toBeDefined(); + expect(cameraOptions.zoom).toBeCloseTo(expectedZoom); + expect(mockedGetElevation.mock.calls).toHaveLength(1); + }); + + test('don\'t call getElevation when altitude supplied', () => { + const map = createMap(); + + const mockedGetElevation = jest.fn((_tileID: OverscaledTileID, _x: number, _y: number, _extent?: number) => 0); + + const terrainStub = {} as Terrain; + terrainStub.getElevation = mockedGetElevation; + map.terrain = terrainStub; + + const cameraOptions = map.calculateCameraOptionsFromTo(new LngLat(0, 0), 0, new LngLat(0, 0), 1000); + + expect(cameraOptions).toBeDefined(); + expect(mockedGetElevation.mock.calls).toHaveLength(0); + }); + + test('don\'t call getElevation when altitude 0 supplied', () => { + const map = createMap(); + + const mockedGetElevation = jest.fn((_tileID: OverscaledTileID, _x: number, _y: number, _extent?: number) => 0); + + const terrainStub = {} as Terrain; + terrainStub.getElevation = mockedGetElevation; + map.terrain = terrainStub; + + const cameraOptions = map.calculateCameraOptionsFromTo(new LngLat(0, 0), 0, new LngLat(1, 0), 0); + + expect(cameraOptions).toBeDefined(); + expect(mockedGetElevation.mock.calls).toHaveLength(0); + }); + }); + + describe('webgl errors', () => { + test('WebGL error while creating map', () => { + setErrorWebGlContext(); + try { + createMap(); + } catch (e) { + const errorMessageObject = JSON.parse(e.message); + + // this message is from map code + expect(errorMessageObject.message).toBe('Failed to initialize WebGL'); + + // this is from test mock + expect(errorMessageObject.statusMessage).toBe('mocked webglcontextcreationerror message'); + } + + }); + test('Hit WebGL max drawing buffer limit', () => { + // Simulate a device with MAX_TEXTURE_SIZE=16834 and max rendering area of ~32Mpx + const container = window.document.createElement('div'); + Object.defineProperty(container, 'clientWidth', {value: 8000}); + Object.defineProperty(container, 'clientHeight', {value: 4500}); + const map = createMap({container, maxCanvasSize: [16834, 16834], pixelRatio: 1}); + jest.spyOn(map.painter.context.gl, 'drawingBufferWidth', 'get').mockReturnValue(7536); + jest.spyOn(map.painter.context.gl, 'drawingBufferHeight', 'get').mockReturnValue(4239); + map.resize(); + expect(map.getCanvas().width).toBe(7536); + expect(map.getCanvas().height).toBe(4239); + // Check if maxCanvasSize is updated + expect(map._maxCanvasSize).toEqual([7536, 4239]); + }); + }); + + describe('Max Canvas Size option', () => { + test('maxCanvasSize width = height', () => { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'clientWidth', {value: 2048}); + Object.defineProperty(container, 'clientHeight', {value: 2048}); + const map = createMap({container, maxCanvasSize: [8192, 8192], pixelRatio: 5}); + map.resize(); + expect(map.getCanvas().width).toBe(8192); + expect(map.getCanvas().height).toBe(8192); + }); + + test('maxCanvasSize width != height', () => { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'clientWidth', {value: 1024}); + Object.defineProperty(container, 'clientHeight', {value: 2048}); + const map = createMap({container, maxCanvasSize: [8192, 4096], pixelRatio: 3}); + map.resize(); + expect(map.getCanvas().width).toBe(2048); + expect(map.getCanvas().height).toBe(4096); + }); + + test('maxCanvasSize below clientWidth and clientHeigth', () => { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'clientWidth', {value: 12834}); + Object.defineProperty(container, 'clientHeight', {value: 9000}); + const map = createMap({container, maxCanvasSize: [4096, 8192], pixelRatio: 1}); + map.resize(); + expect(map.getCanvas().width).toBe(4096); + expect(map.getCanvas().height).toBe(2872); + }); + + test('maxCanvasSize with setPixelRatio', () => { + const container = window.document.createElement('div'); + Object.defineProperty(container, 'clientWidth', {value: 2048}); + Object.defineProperty(container, 'clientHeight', {value: 2048}); + const map = createMap({container, maxCanvasSize: [3072, 3072], pixelRatio: 1.25}); + map.resize(); + expect(map.getCanvas().width).toBe(2560); + expect(map.getCanvas().height).toBe(2560); + map.setPixelRatio(2); + expect(map.getCanvas().width).toBe(3072); + expect(map.getCanvas().height).toBe(3072); + }); + }); + +}); + +function createStyle() { + return { + version: 8, + center: [-73.9749, 40.7736], + zoom: 12.5, + bearing: 29, + pitch: 50, + sources: {}, + layers: [] + } as StyleSpecification; +} diff --git a/web/libraries/maplibre-gl/src/ui/map.ts b/web/libraries/maplibre-gl/src/ui/map.ts new file mode 100644 index 00000000..4ecd20d6 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/map.ts @@ -0,0 +1,3435 @@ +import {extend, warnOnce, uniqueId, isImageBitmap} from '../util/util'; +import {browser} from '../util/browser'; +import {DOM} from '../util/dom'; +import packageJSON from '../../package.json' assert {type: 'json'}; + +import {getJSON} from '../util/ajax'; +import {ImageRequest} from '../util/image_request'; +import type {GetImageCallback} from '../util/image_request'; + +import {RequestManager, ResourceType} from '../util/request_manager'; +import {Style, StyleSwapOptions} from '../style/style'; +import {EvaluationParameters} from '../style/evaluation_parameters'; +import {Painter} from '../render/painter'; +import {Transform} from '../geo/transform'; +import {Hash} from './hash'; +import {HandlerManager} from './handler_manager'; +import {Camera, CameraOptions, CameraUpdateTransformFunction, FitBoundsOptions} from './camera'; +import {LngLat} from '../geo/lng_lat'; +import {LngLatBounds} from '../geo/lng_lat_bounds'; +import Point from '@mapbox/point-geometry'; +import {AttributionControl} from './control/attribution_control'; +import {LogoControl} from './control/logo_control'; + +import {RGBAImage} from '../util/image'; +import {Event, ErrorEvent, Listener} from '../util/evented'; +import {MapEventType, MapLayerEventType, MapMouseEvent, MapSourceDataEvent, MapStyleDataEvent} from './events'; +import {TaskQueue} from '../util/task_queue'; +import {throttle} from '../util/throttle'; +import {webpSupported} from '../util/webp_supported'; +import {PerformanceMarkers, PerformanceUtils} from '../util/performance'; +import {Source, SourceClass} from '../source/source'; +import {StyleLayer} from '../style/style_layer'; + +import type {RequestTransformFunction} from '../util/request_manager'; +import type {LngLatLike} from '../geo/lng_lat'; +import type {LngLatBoundsLike} from '../geo/lng_lat_bounds'; +import type {AddLayerObject, FeatureIdentifier, StyleOptions, StyleSetterOptions} from '../style/style'; +import type {MapDataEvent} from './events'; +import type {StyleImage, StyleImageInterface, StyleImageMetadata} from '../style/style_image'; +import type {PointLike} from './camera'; +import type {ScrollZoomHandler} from './handler/scroll_zoom'; +import type {BoxZoomHandler} from './handler/box_zoom'; +import type {AroundCenterOptions, TwoFingersTouchPitchHandler} from './handler/two_fingers_touch'; +import type {DragRotateHandler} from './handler/shim/drag_rotate'; +import {DragPanHandler, DragPanOptions} from './handler/shim/drag_pan'; + +import type {KeyboardHandler} from './handler/keyboard'; +import type {DoubleClickZoomHandler} from './handler/shim/dblclick_zoom'; +import type {TwoFingersTouchZoomRotateHandler} from './handler/shim/two_fingers_touch'; +import {defaultLocale} from './default_locale'; +import type {TaskID} from '../util/task_queue'; +import type {Cancelable} from '../types/cancelable'; +import type { + FilterSpecification, + StyleSpecification, + LightSpecification, + SourceSpecification, + TerrainSpecification +} from '@maplibre/maplibre-gl-style-spec'; + +import {Callback} from '../types/callback'; +import type {ControlPosition, IControl} from './control/control'; +import type {MapGeoJSONFeature} from '../util/vectortile_to_geojson'; +import {Terrain} from '../render/terrain'; +import {RenderToTexture} from '../render/render_to_texture'; +import {config} from '../util/config'; +import type {QueryRenderedFeaturesOptions, QuerySourceFeatureOptions} from '../source/query_features'; + +const version = packageJSON.version; + +/** + * The {@link Map} options object. + */ +export type MapOptions = { + /** + * If `true`, the map's position (zoom, center latitude, center longitude, bearing, and pitch) will be synced with the hash fragment of the page's URL. + * For example, `http://path/to/my/page.html#2.59/39.26/53.07/-24.1/60`. + * An additional string may optionally be provided to indicate a parameter-styled hash, + * e.g. http://path/to/my/page.html#map=2.59/39.26/53.07/-24.1/60&foo=bar, where foo + * is a custom parameter and bar is an arbitrary hash distinct from the map hash. + * @defaultValue false + */ + hash?: boolean | string; + /** + * If `false`, no mouse, touch, or keyboard listeners will be attached to the map, so it will not respond to interaction. + * @defaultValue true + */ + interactive?: boolean; + /** + * The HTML element in which MapLibre GL JS will render the map, or the element's string `id`. The specified element must have no children. + */ + container: HTMLElement | string; + /** + * The threshold, measured in degrees, that determines when the map's + * bearing will snap to north. For example, with a `bearingSnap` of 7, if the user rotates + * the map within 7 degrees of north, the map will automatically snap to exact north. + * @defaultValue 7 + */ + bearingSnap?: number; + /** + * If `true`, an {@link AttributionControl} will be added to the map. + * @defaultValue true + */ + attributionControl?: boolean; + /** + * Attribution text to show in an {@link AttributionControl}. Only applicable if `options.attributionControl` is `true`. + */ + customAttribution?: string | Array; + /** + * If `true`, the MapLibre logo will be shown. + * @defaultValue false + */ + maplibreLogo?: boolean; + /** + * A string representing the position of the MapLibre wordmark on the map. Valid options are `top-left`,`top-right`, `bottom-left`, or `bottom-right`. + * @defaultValue 'bottom-left' + */ + logoPosition?: ControlPosition; + /** + * If `true`, map creation will fail if the performance of MapLibre GL JS would be dramatically worse than expected + * (i.e. a software renderer would be used). + * @defaultValue false + */ + failIfMajorPerformanceCaveat?: boolean; + /** + * If `true`, the map's canvas can be exported to a PNG using `map.getCanvas().toDataURL()`. This is `false` by default as a performance optimization. + * @defaultValue false + */ + preserveDrawingBuffer?: boolean; + /** + * If `true`, the gl context will be created with MSAA antialiasing, which can be useful for antialiasing custom layers. This is `false` by default as a performance optimization. + */ + antialias?: boolean; + /** + * If `false`, the map won't attempt to re-request tiles once they expire per their HTTP `cacheControl`/`expires` headers. + * @defaultValue true + */ + refreshExpiredTiles?: boolean; + /** + * If set, the map will be constrained to the given bounds. + */ + maxBounds?: LngLatBoundsLike; + /** + * If `true`, the "scroll to zoom" interaction is enabled. {@link AroundCenterOptions} are passed as options to {@link ScrollZoomHandler#enable}. + * @defaultValue true + */ + scrollZoom?: boolean | AroundCenterOptions; + /** + * The minimum zoom level of the map (0-24). + * @defaultValue 0 + */ + minZoom?: number | null; + /** + * The maximum zoom level of the map (0-24). + * @defaultValue 22 + */ + maxZoom?: number | null; + /** + * The minimum pitch of the map (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project. + * @defaultValue 0 + */ + minPitch?: number | null; + /** + * The maximum pitch of the map (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project. + * @defaultValue 60 + */ + maxPitch?: number | null; + /** + * If `true`, the "box zoom" interaction is enabled (see {@link BoxZoomHandler}). + * @defaultValue true + */ + boxZoom?: boolean; + /** + * If `true`, the "drag to rotate" interaction is enabled (see {@link DragRotateHandler}). + * @defaultValue true + */ + dragRotate?: boolean; + /** + * If `true`, the "drag to pan" interaction is enabled. An `Object` value is passed as options to {@link DragPanHandler#enable}. + * @defaultValue true + */ + dragPan?: boolean | DragPanOptions; + /** + * If `true`, keyboard shortcuts are enabled (see {@link KeyboardHandler}). + * @defaultValue true + */ + keyboard?: boolean; + /** + * If `true`, the "double click to zoom" interaction is enabled (see {@link DoubleClickZoomHandler}). + * @defaultValue true + */ + doubleClickZoom?: boolean; + /** + * If `true`, the "pinch to rotate and zoom" interaction is enabled. An `Object` value is passed as options to {@link TwoFingersTouchZoomRotateHandler#enable}. + * @defaultValue true + */ + touchZoomRotate?: boolean | AroundCenterOptions; + /** + * If `true`, the "drag to pitch" interaction is enabled. An `Object` value is passed as options to {@link TwoFingersTouchPitchHandler#enable}. + * @defaultValue true + */ + touchPitch?: boolean | AroundCenterOptions; + /** + * If `true` or set to an options object, the map is only accessible on desktop while holding Command/Ctrl and only accessible on mobile with two fingers. Interacting with the map using normal gestures will trigger an informational screen. With this option enabled, "drag to pitch" requires a three-finger gesture. Cooperative gestures are disabled when a map enters fullscreen using {@link FullscreenControl}. + * @defaultValue undefined + */ + cooperativeGestures?: boolean | GestureOptions; + /** + * If `true`, the map will automatically resize when the browser window resizes. + * @defaultValue true + */ + trackResize?: boolean; + /** + * The initial geographical centerpoint of the map. If `center` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `[0, 0]` Note: MapLibre GL JS uses longitude, latitude coordinate order (as opposed to latitude, longitude) to match GeoJSON. + * @defaultValue [0, 0] + */ + center?: LngLatLike; + /** + * The initial zoom level of the map. If `zoom` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`. + * @defaultValue 0 + */ + zoom?: number; + /** + * The initial bearing (rotation) of the map, measured in degrees counter-clockwise from north. If `bearing` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`. + * @defaultValue 0 + */ + bearing?: number; + /** + * The initial pitch (tilt) of the map, measured in degrees away from the plane of the screen (0-85). If `pitch` is not specified in the constructor options, MapLibre GL JS will look for it in the map's style object. If it is not specified in the style, either, it will default to `0`. Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project. + * @defaultValue 0 + */ + pitch?: number; + /** + * If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`: + * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire + * container, there will be blank space beyond 180 and -180 degrees longitude. + * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the + * map and the other on the left edge of the map) at every zoom level. + * @defaultValue true + */ + renderWorldCopies?: boolean; + /** + * The maximum number of tiles stored in the tile cache for a given source. If omitted, the cache will be dynamically sized based on the current viewport which can be set using `maxTileCacheZoomLevels` constructor options. + * @defaultValue null + */ + maxTileCacheSize?: number; + /** + * The maximum number of zoom levels for which to store tiles for a given source. Tile cache dynamic size is calculated by multiplying `maxTileCacheZoomLevels` with the approximate number of tiles in the viewport for a given source. + * @defaultValue 5 + */ + maxTileCacheZoomLevels?: number; + /** + * A callback run before the Map makes a request for an external URL. The callback can be used to modify the url, set headers, or set the credentials property for cross-origin requests. + * Expected to return an object with a `url` property and optionally `headers` and `credentials` properties. + */ + transformRequest?: RequestTransformFunction; + /** + * A callback run before the map's camera is moved due to user input or animation. The callback can be used to modify the new center, zoom, pitch and bearing. + * Expected to return an object containing center, zoom, pitch or bearing values to overwrite. + */ + transformCameraUpdate?: CameraUpdateTransformFunction; + /** + * A patch to apply to the default localization table for UI strings, e.g. control tooltips. The `locale` object maps namespaced UI string IDs to translated strings in the target language; see `src/ui/default_locale.js` for an example with all supported string IDs. The object may specify all UI strings (thereby adding support for a new translation) or only a subset of strings (thereby patching the default translation table). + * @defaultValue null + */ + locale?: any; + /** + * Controls the duration of the fade-in/fade-out animation for label collisions after initial map load, in milliseconds. This setting affects all symbol layers. This setting does not affect the duration of runtime styling transitions or raster tile cross-fading. + * @defaultValue 300 + */ + fadeDuration?: number; + /** + * If `true`, symbols from multiple sources can collide with each other during collision detection. If `false`, collision detection is run separately for the symbols in each source. + * @defaultValue true + */ + crossSourceCollisions?: boolean; + /** + * If `true`, Resource Timing API information will be collected for requests made by GeoJSON and Vector Tile web workers (this information is normally inaccessible from the main Javascript thread). Information will be returned in a `resourceTiming` property of relevant `data` events. + * @defaultValue false + */ + collectResourceTiming?: boolean; + /** + * The max number of pixels a user can shift the mouse pointer during a click for it to be considered a valid click (as opposed to a mouse drag). + * @defaultValue true + */ + clickTolerance?: number; + /** + * The initial bounds of the map. If `bounds` is specified, it overrides `center` and `zoom` constructor options. + */ + bounds?: LngLatBoundsLike; + /** + * A {@link FitBoundsOptions} options object to use _only_ when fitting the initial `bounds` provided above. + */ + fitBoundsOptions?: FitBoundsOptions; + /** + * Defines a CSS + * font-family for locally overriding generation of glyphs in the 'CJK Unified Ideographs', 'Hiragana', 'Katakana' and 'Hangul Syllables' ranges. + * In these ranges, font settings from the map's style will be ignored, except for font-weight keywords (light/regular/medium/bold). + * Set to `false`, to enable font settings from the map's style for these glyph ranges. + * The purpose of this option is to avoid bandwidth-intensive glyph server requests. (See [Use locally generated ideographs](https://maplibre.org/maplibre-gl-js/docs/examples/local-ideographs).) + * @defaultValue 'sans-serif' + */ + localIdeographFontFamily?: string; + /** + * The map's MapLibre style. This must be a JSON object conforming to + * the schema described in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/), + * or a URL to such JSON. + */ + style: StyleSpecification | string; + /** + * If `false`, the map's pitch (tilt) control with "drag to rotate" interaction will be disabled. + * @defaultValue true + */ + pitchWithRotate?: boolean; + /** + * The pixel ratio. The canvas' `width` attribute will be `container.clientWidth * pixelRatio` and its `height` attribute will be `container.clientHeight * pixelRatio`. Defaults to `devicePixelRatio` if not specified. + */ + pixelRatio?: number; + /** + * If false, style validation will be skipped. Useful in production environment. + * @defaultValue true + */ + validateStyle?: boolean; + /** + * The canvas' `width` and `height` max size. The values are passed as an array where the first element is max width and the second element is max height. + * You shouldn't set this above WebGl `MAX_TEXTURE_SIZE`. Defaults to [4096, 4096]. + */ + maxCanvasSize?: [number, number]; +}; + +/** + * An options object for the gesture settings + * @example + * ```ts + * let options = { + * windowsHelpText: "Use Ctrl + scroll to zoom the map", + * macHelpText: "Use ⌘ + scroll to zoom the map", + * mobileHelpText: "Use two fingers to move the map", + * } + * ``` + */ +export type GestureOptions = { + windowsHelpText?: string; + macHelpText?: string; + mobileHelpText?: string; +}; + +export type AddImageOptions = { + +} + +// See article here: https://medium.com/terria/typescript-transforming-optional-properties-to-required-properties-that-may-be-undefined-7482cb4e1585 +type Complete = { + [P in keyof Required]: Pick extends Required> ? T[P] : (T[P] | undefined); +} + +// This type is used inside map since all properties are assigned a default value. +export type CompleteMapOptions = Complete; + +const defaultMinZoom = -2; +const defaultMaxZoom = 22; + +// the default values, but also the valid range +const defaultMinPitch = 0; +const defaultMaxPitch = 60; + +// use this variable to check maxPitch for validity +const maxPitchThreshold = 85; + +const defaultOptions = { + center: [0, 0], + zoom: 0, + bearing: 0, + pitch: 0, + + minZoom: defaultMinZoom, + maxZoom: defaultMaxZoom, + + minPitch: defaultMinPitch, + maxPitch: defaultMaxPitch, + + interactive: true, + scrollZoom: true, + boxZoom: true, + dragRotate: true, + dragPan: true, + keyboard: true, + doubleClickZoom: true, + touchZoomRotate: true, + touchPitch: true, + cooperativeGestures: undefined, + + bearingSnap: 7, + clickTolerance: 3, + pitchWithRotate: true, + + hash: false, + attributionControl: true, + maplibreLogo: false, + + failIfMajorPerformanceCaveat: false, + preserveDrawingBuffer: false, + trackResize: true, + renderWorldCopies: true, + refreshExpiredTiles: true, + maxTileCacheSize: null, + maxTileCacheZoomLevels: config.MAX_TILE_CACHE_ZOOM_LEVELS, + localIdeographFontFamily: 'sans-serif', + transformRequest: null, + transformCameraUpdate: null, + fadeDuration: 300, + crossSourceCollisions: true, + validateStyle: true, + /**Because GL MAX_TEXTURE_SIZE is usually at least 4096px. */ + maxCanvasSize: [4096, 4096] +} as CompleteMapOptions; + +/** + * The `Map` object represents the map on your page. It exposes methods + * and properties that enable you to programmatically change the map, + * and fires events as users interact with it. + * + * You create a `Map` by specifying a `container` and other options, see {@link MapOptions} for the full list. + * Then MapLibre GL JS initializes the map on the page and returns your `Map` object. + * + * @group Main + * + * @example + * ```ts + * let map = new maplibregl.Map({ + * container: 'map', + * center: [-122.420679, 37.772537], + * zoom: 13, + * style: style_object, + * hash: true, + * transformRequest: (url, resourceType)=> { + * if(resourceType === 'Source' && url.startsWith('http://myHost')) { + * return { + * url: url.replace('http', 'https'), + * headers: { 'my-custom-header': true}, + * credentials: 'include' // Include cookies for cross-origin requests + * } + * } + * } + * }); + * ``` + * @see [Display a map](https://maplibre.org/maplibre-gl-js/docs/examples/simple-map/) + */ +export class Map extends Camera { + style: Style; + painter: Painter; + handlers: HandlerManager; + + _container: HTMLElement; + _canvasContainer: HTMLElement; + _controlContainer: HTMLElement; + _controlPositions: {[_: string]: HTMLElement}; + _interactive: boolean; + _cooperativeGestures: boolean | GestureOptions; + _cooperativeGesturesScreen: HTMLElement; + _metaKey: keyof MouseEvent; + _showTileBoundaries: boolean; + _showCollisionBoxes: boolean; + _showPadding: boolean; + _showOverdrawInspector: boolean; + _repaint: boolean; + _vertices: boolean; + _canvas: HTMLCanvasElement; + _maxTileCacheSize: number; + _maxTileCacheZoomLevels: number; + _frame: Cancelable; + _styleDirty: boolean; + _sourcesDirty: boolean; + _placementDirty: boolean; + + _loaded: boolean; + _idleTriggered: boolean; + // accounts for placement finishing as well + _fullyLoaded: boolean; + _trackResize: boolean; + _resizeObserver: ResizeObserver; + _preserveDrawingBuffer: boolean; + _failIfMajorPerformanceCaveat: boolean; + _antialias: boolean; + _refreshExpiredTiles: boolean; + _hash: Hash; + _delegatedListeners: any; + _fadeDuration: number; + _crossSourceCollisions: boolean; + _crossFadingFactor: number; + _collectResourceTiming: boolean; + _renderTaskQueue: TaskQueue; + _controls: Array; + _mapId: number; + _localIdeographFontFamily: string; + _validateStyle: boolean; + _requestManager: RequestManager; + _locale: any; + _removed: boolean; + _clickTolerance: number; + _overridePixelRatio: number | null; + _maxCanvasSize: [number, number]; + _terrainDataCallback: (e: MapStyleDataEvent | MapSourceDataEvent) => void; + + /** + * @internal + * image queue throttling handle. To be used later when clean up + */ + _imageQueueHandle: number; + + /** + * The map's {@link ScrollZoomHandler}, which implements zooming in and out with a scroll wheel or trackpad. + * Find more details and examples using `scrollZoom` in the {@link ScrollZoomHandler} section. + */ + scrollZoom: ScrollZoomHandler; + + /** + * The map's {@link BoxZoomHandler}, which implements zooming using a drag gesture with the Shift key pressed. + * Find more details and examples using `boxZoom` in the {@link BoxZoomHandler} section. + */ + boxZoom: BoxZoomHandler; + + /** + * The map's {@link DragRotateHandler}, which implements rotating the map while dragging with the right + * mouse button or with the Control key pressed. Find more details and examples using `dragRotate` + * in the {@link DragRotateHandler} section. + */ + dragRotate: DragRotateHandler; + + /** + * The map's {@link DragPanHandler}, which implements dragging the map with a mouse or touch gesture. + * Find more details and examples using `dragPan` in the {@link DragPanHandler} section. + */ + dragPan: DragPanHandler; + + /** + * The map's {@link KeyboardHandler}, which allows the user to zoom, rotate, and pan the map using keyboard + * shortcuts. Find more details and examples using `keyboard` in the {@link KeyboardHandler} section. + */ + keyboard: KeyboardHandler; + + /** + * The map's {@link DoubleClickZoomHandler}, which allows the user to zoom by double clicking. + * Find more details and examples using `doubleClickZoom` in the {@link DoubleClickZoomHandler} section. + */ + doubleClickZoom: DoubleClickZoomHandler; + + /** + * The map's {@link TwoFingersTouchZoomRotateHandler}, which allows the user to zoom or rotate the map with touch gestures. + * Find more details and examples using `touchZoomRotate` in the {@link TwoFingersTouchZoomRotateHandler} section. + */ + touchZoomRotate: TwoFingersTouchZoomRotateHandler; + + /** + * The map's {@link TwoFingersTouchPitchHandler}, which allows the user to pitch the map with touch gestures. + * Find more details and examples using `touchPitch` in the {@link TwoFingersTouchPitchHandler} section. + */ + touchPitch: TwoFingersTouchPitchHandler; + + constructor(options: MapOptions) { + PerformanceUtils.mark(PerformanceMarkers.create); + + options = extend({}, defaultOptions, options); + + if (options.minZoom != null && options.maxZoom != null && options.minZoom > options.maxZoom) { + throw new Error('maxZoom must be greater than or equal to minZoom'); + } + + if (options.minPitch != null && options.maxPitch != null && options.minPitch > options.maxPitch) { + throw new Error('maxPitch must be greater than or equal to minPitch'); + } + + if (options.minPitch != null && options.minPitch < defaultMinPitch) { + throw new Error(`minPitch must be greater than or equal to ${defaultMinPitch}`); + } + + if (options.maxPitch != null && options.maxPitch > maxPitchThreshold) { + throw new Error(`maxPitch must be less than or equal to ${maxPitchThreshold}`); + } + + const transform = new Transform(options.minZoom, options.maxZoom, options.minPitch, options.maxPitch, options.renderWorldCopies); + super(transform, {bearingSnap: options.bearingSnap}); + + this._interactive = options.interactive; + this._cooperativeGestures = options.cooperativeGestures; + this._metaKey = navigator.platform.indexOf('Mac') === 0 ? 'metaKey' : 'ctrlKey'; + this._maxTileCacheSize = options.maxTileCacheSize; + this._maxTileCacheZoomLevels = options.maxTileCacheZoomLevels; + this._failIfMajorPerformanceCaveat = options.failIfMajorPerformanceCaveat; + this._preserveDrawingBuffer = options.preserveDrawingBuffer; + this._antialias = options.antialias; + this._trackResize = options.trackResize; + this._bearingSnap = options.bearingSnap; + this._refreshExpiredTiles = options.refreshExpiredTiles; + this._fadeDuration = options.fadeDuration; + this._crossSourceCollisions = options.crossSourceCollisions; + this._crossFadingFactor = 1; + this._collectResourceTiming = options.collectResourceTiming; + this._renderTaskQueue = new TaskQueue(); + this._controls = []; + this._mapId = uniqueId(); + this._locale = extend({}, defaultLocale, options.locale); + this._clickTolerance = options.clickTolerance; + this._overridePixelRatio = options.pixelRatio; + this._maxCanvasSize = options.maxCanvasSize; + this.transformCameraUpdate = options.transformCameraUpdate; + + this._imageQueueHandle = ImageRequest.addThrottleControl(() => this.isMoving()); + + this._requestManager = new RequestManager(options.transformRequest); + + if (typeof options.container === 'string') { + this._container = document.getElementById(options.container); + if (!this._container) { + throw new Error(`Container '${options.container}' not found.`); + } + } else if (options.container instanceof HTMLElement) { + this._container = options.container; + } else { + throw new Error('Invalid type: \'container\' must be a String or HTMLElement.'); + } + + if (options.maxBounds) { + this.setMaxBounds(options.maxBounds); + } + + this._setupContainer(); + this._setupPainter(); + + this.on('move', () => this._update(false)); + this.on('moveend', () => this._update(false)); + this.on('zoom', () => this._update(true)); + this.on('terrain', () => { + this.painter.terrainFacilitator.dirty = true; + this._update(true); + }); + this.once('idle', () => { this._idleTriggered = true; }); + + if (typeof window !== 'undefined') { + addEventListener('online', this._onWindowOnline, false); + let initialResizeEventCaptured = false; + const throttledResizeCallback = throttle((entries: ResizeObserverEntry[]) => { + if (this._trackResize && !this._removed) { + this.resize(entries)._update(); + } + }, 50); + this._resizeObserver = new ResizeObserver((entries) => { + if (!initialResizeEventCaptured) { + initialResizeEventCaptured = true; + return; + } + throttledResizeCallback(entries); + }); + this._resizeObserver.observe(this._container); + } + + this.handlers = new HandlerManager(this, options as CompleteMapOptions); + + if (this._cooperativeGestures) { + this._setupCooperativeGestures(); + } + + const hashName = (typeof options.hash === 'string' && options.hash) || undefined; + this._hash = options.hash && (new Hash(hashName)).addTo(this); + // don't set position from options if set through hash + if (!this._hash || !this._hash._onHashChange()) { + this.jumpTo({ + center: options.center, + zoom: options.zoom, + bearing: options.bearing, + pitch: options.pitch + }); + + if (options.bounds) { + this.resize(); + this.fitBounds(options.bounds, extend({}, options.fitBoundsOptions, {duration: 0})); + } + } + + this.resize(); + + this._localIdeographFontFamily = options.localIdeographFontFamily; + this._validateStyle = options.validateStyle; + + if (options.style) this.setStyle(options.style, {localIdeographFontFamily: options.localIdeographFontFamily}); + + if (options.attributionControl) + this.addControl(new AttributionControl({customAttribution: options.customAttribution})); + + if (options.maplibreLogo) + this.addControl(new LogoControl(), options.logoPosition); + + this.on('style.load', () => { + if (this.transform.unmodified) { + this.jumpTo(this.style.stylesheet as any); + } + }); + this.on('data', (event: MapDataEvent) => { + this._update(event.dataType === 'style'); + this.fire(new Event(`${event.dataType}data`, event)); + }); + this.on('dataloading', (event: MapDataEvent) => { + this.fire(new Event(`${event.dataType}dataloading`, event)); + }); + this.on('dataabort', (event: MapDataEvent) => { + this.fire(new Event('sourcedataabort', event)); + }); + } + + /** + * @internal + * Returns a unique number for this map instance which is used for the MapLoadEvent + * to make sure we only fire one event per instantiated map object. + * @returns the uniq map ID + */ + _getMapId() { + return this._mapId; + } + + /** + * Adds an {@link IControl} to the map, calling `control.onAdd(this)`. + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param control - The {@link IControl} to add. + * @param position - position on the map to which the control will be added. + * Valid values are `'top-left'`, `'top-right'`, `'bottom-left'`, and `'bottom-right'`. Defaults to `'top-right'`. + * @returns `this` + * @example + * Add zoom and rotation controls to the map. + * ```ts + * map.addControl(new maplibregl.NavigationControl()); + * ``` + * @see [Display map navigation controls](https://maplibre.org/maplibre-gl-js/docs/examples/navigation/) + */ + addControl(control: IControl, position?: ControlPosition): Map { + if (position === undefined) { + if (control.getDefaultPosition) { + position = control.getDefaultPosition(); + } else { + position = 'top-right'; + } + } + if (!control || !control.onAdd) { + return this.fire(new ErrorEvent(new Error( + 'Invalid argument to map.addControl(). Argument must be a control with onAdd and onRemove methods.'))); + } + const controlElement = control.onAdd(this); + this._controls.push(control); + + const positionContainer = this._controlPositions[position]; + if (position.indexOf('bottom') !== -1) { + positionContainer.insertBefore(controlElement, positionContainer.firstChild); + } else { + positionContainer.appendChild(controlElement); + } + return this; + } + + /** + * Removes the control from the map. + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param control - The {@link IControl} to remove. + * @returns `this` + * @example + * ```ts + * // Define a new navigation control. + * let navigation = new maplibregl.NavigationControl(); + * // Add zoom and rotation controls to the map. + * map.addControl(navigation); + * // Remove zoom and rotation controls from the map. + * map.removeControl(navigation); + * ``` + */ + removeControl(control: IControl): Map { + if (!control || !control.onRemove) { + return this.fire(new ErrorEvent(new Error( + 'Invalid argument to map.removeControl(). Argument must be a control with onAdd and onRemove methods.'))); + } + const ci = this._controls.indexOf(control); + if (ci > -1) this._controls.splice(ci, 1); + control.onRemove(this); + return this; + } + + /** + * Checks if a control exists on the map. + * + * @param control - The {@link IControl} to check. + * @returns true if map contains control. + * @example + * ```ts + * // Define a new navigation control. + * let navigation = new maplibregl.NavigationControl(); + * // Add zoom and rotation controls to the map. + * map.addControl(navigation); + * // Check that the navigation control exists on the map. + * map.hasControl(navigation); + * ``` + */ + hasControl(control: IControl): boolean { + return this._controls.indexOf(control) > -1; + } + + calculateCameraOptionsFromTo(from: LngLat, altitudeFrom: number, to: LngLat, altitudeTo?: number): CameraOptions { + if (altitudeTo == null && this.terrain) { + altitudeTo = this.terrain.getElevationForLngLatZoom(to, this.transform.tileZoom); + } + return super.calculateCameraOptionsFromTo(from, altitudeFrom, to, altitudeTo); + } + + /** + * Resizes the map according to the dimensions of its + * `container` element. + * + * Checks if the map container size changed and updates the map if it has changed. + * This method must be called after the map's `container` is resized programmatically + * or when the map is shown after being initially hidden with CSS. + * + * Triggers the following events: `movestart`, `move`, `moveend`, and `resize`. + * + * @param eventData - Additional properties to be passed to `movestart`, `move`, `resize`, and `moveend` + * events that get triggered as a result of resize. This can be useful for differentiating the + * source of an event (for example, user-initiated or programmatically-triggered events). + * @returns `this` + * @example + * Resize the map when the map container is shown after being initially hidden with CSS. + * ```ts + * let mapDiv = document.getElementById('map'); + * if (mapDiv.style.visibility === true) map.resize(); + * ``` + */ + resize(eventData?: any): Map { + const dimensions = this._containerDimensions(); + const width = dimensions[0]; + const height = dimensions[1]; + + const clampedPixelRatio = this._getClampedPixelRatio(width, height); + this._resizeCanvas(width, height, clampedPixelRatio); + this.painter.resize(width, height, clampedPixelRatio); + + // check if we've reached GL limits, in that case further clamps pixelRatio + if (this.painter.overLimit()) { + const gl = this.painter.context.gl; + // store updated _maxCanvasSize value + this._maxCanvasSize = [gl.drawingBufferWidth, gl.drawingBufferHeight]; + const clampedPixelRatio = this._getClampedPixelRatio(width, height); + this._resizeCanvas(width, height, clampedPixelRatio); + this.painter.resize(width, height, clampedPixelRatio); + } + + this.transform.resize(width, height); + this._requestedCameraState?.resize(width, height); + + const fireMoving = !this._moving; + if (fireMoving) { + this.stop(); + this.fire(new Event('movestart', eventData)) + .fire(new Event('move', eventData)); + } + + this.fire(new Event('resize', eventData)); + + if (fireMoving) this.fire(new Event('moveend', eventData)); + + return this; + } + + /** + * @internal + * Return the map's pixel ratio eventually scaled down to respect maxCanvasSize. + * Internally you should use this and not getPixelRatio(). + */ + _getClampedPixelRatio(width: number, height: number): number { + const {0: maxCanvasWidth, 1: maxCanvasHeight} = this._maxCanvasSize; + const pixelRatio = this.getPixelRatio(); + + const canvasWidth = width * pixelRatio; + const canvasHeight = height * pixelRatio; + + const widthScaleFactor = canvasWidth > maxCanvasWidth ? (maxCanvasWidth / canvasWidth) : 1; + const heightScaleFactor = canvasHeight > maxCanvasHeight ? (maxCanvasHeight / canvasHeight) : 1; + + return Math.min(widthScaleFactor, heightScaleFactor) * pixelRatio; + } + + /** + * Returns the map's pixel ratio. + * Note that the pixel ratio actually applied may be lower to respect maxCanvasSize. + * @returns The pixel ratio. + */ + getPixelRatio(): number { + return this._overridePixelRatio ?? devicePixelRatio; + } + + /** + * Sets the map's pixel ratio. This allows to override `devicePixelRatio`. + * After this call, the canvas' `width` attribute will be `container.clientWidth * pixelRatio` + * and its height attribute will be `container.clientHeight * pixelRatio`. + * Set this to null to disable `devicePixelRatio` override. + * Note that the pixel ratio actually applied may be lower to respect maxCanvasSize. + * @param pixelRatio - The pixel ratio. + */ + setPixelRatio(pixelRatio: number) { + this._overridePixelRatio = pixelRatio; + this.resize(); + } + + /** + * Returns the map's geographical bounds. When the bearing or pitch is non-zero, the visible region is not + * an axis-aligned rectangle, and the result is the smallest bounds that encompasses the visible region. + * @returns The geographical bounds of the map as {@link LngLatBounds}. + * @example + * ```ts + * let bounds = map.getBounds(); + * ``` + */ + getBounds(): LngLatBounds { + return this.transform.getBounds(); + } + + /** + * Returns the maximum geographical bounds the map is constrained to, or `null` if none set. + * @returns The map object. + * @example + * ```ts + * let maxBounds = map.getMaxBounds(); + * ``` + */ + getMaxBounds(): LngLatBounds | null { + return this.transform.getMaxBounds(); + } + + /** + * Sets or clears the map's geographical bounds. + * + * Pan and zoom operations are constrained within these bounds. + * If a pan or zoom is performed that would + * display regions outside these bounds, the map will + * instead display a position and zoom level + * as close as possible to the operation's request while still + * remaining within the bounds. + * + * @param bounds - The maximum bounds to set. If `null` or `undefined` is provided, the function removes the map's maximum bounds. + * @returns `this` + * @example + * Define bounds that conform to the `LngLatBoundsLike` object as set the max bounds. + * ```ts + * let bounds = [ + * [-74.04728, 40.68392], // [west, south] + * [-73.91058, 40.87764] // [east, north] + * ]; + * map.setMaxBounds(bounds); + * ``` + */ + setMaxBounds(bounds?: LngLatBoundsLike | null): Map { + this.transform.setMaxBounds(LngLatBounds.convert(bounds)); + return this._update(); + } + + /** + * Sets or clears the map's minimum zoom level. + * If the map's current zoom level is lower than the new minimum, + * the map will zoom to the new minimum. + * + * It is not always possible to zoom out and reach the set `minZoom`. + * Other factors such as map height may restrict zooming. For example, + * if the map is 512px tall it will not be possible to zoom below zoom 0 + * no matter what the `minZoom` is set to. + * + * A {@link ErrorEvent} event will be fired if minZoom is out of bounds. + * + * @param minZoom - The minimum zoom level to set (-2 - 24). + * If `null` or `undefined` is provided, the function removes the current minimum zoom (i.e. sets it to -2). + * @returns `this` + * @example + * ```ts + * map.setMinZoom(12.25); + * ``` + */ + setMinZoom(minZoom?: number | null): Map { + + minZoom = minZoom === null || minZoom === undefined ? defaultMinZoom : minZoom; + + if (minZoom >= defaultMinZoom && minZoom <= this.transform.maxZoom) { + this.transform.minZoom = minZoom; + this._update(); + + if (this.getZoom() < minZoom) this.setZoom(minZoom); + + return this; + + } else throw new Error(`minZoom must be between ${defaultMinZoom} and the current maxZoom, inclusive`); + } + + /** + * Returns the map's minimum allowable zoom level. + * + * @returns minZoom + * @example + * ```ts + * let minZoom = map.getMinZoom(); + * ``` + */ + getMinZoom(): number { return this.transform.minZoom; } + + /** + * Sets or clears the map's maximum zoom level. + * If the map's current zoom level is higher than the new maximum, + * the map will zoom to the new maximum. + * + * A {@link ErrorEvent} event will be fired if minZoom is out of bounds. + * + * @param maxZoom - The maximum zoom level to set. + * If `null` or `undefined` is provided, the function removes the current maximum zoom (sets it to 22). + * @returns `this` + * @example + * ```ts + * map.setMaxZoom(18.75); + * ``` + */ + setMaxZoom(maxZoom?: number | null): Map { + + maxZoom = maxZoom === null || maxZoom === undefined ? defaultMaxZoom : maxZoom; + + if (maxZoom >= this.transform.minZoom) { + this.transform.maxZoom = maxZoom; + this._update(); + + if (this.getZoom() > maxZoom) this.setZoom(maxZoom); + + return this; + + } else throw new Error('maxZoom must be greater than the current minZoom'); + } + + /** + * Returns the map's maximum allowable zoom level. + * + * @returns The maxZoom + * @example + * ```ts + * let maxZoom = map.getMaxZoom(); + * ``` + */ + getMaxZoom(): number { return this.transform.maxZoom; } + + /** + * Sets or clears the map's minimum pitch. + * If the map's current pitch is lower than the new minimum, + * the map will pitch to the new minimum. + * + * A {@link ErrorEvent} event will be fired if minPitch is out of bounds. + * + * @param minPitch - The minimum pitch to set (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project. + * If `null` or `undefined` is provided, the function removes the current minimum pitch (i.e. sets it to 0). + * @returns `this` + */ + setMinPitch(minPitch?: number | null): Map { + + minPitch = minPitch === null || minPitch === undefined ? defaultMinPitch : minPitch; + + if (minPitch < defaultMinPitch) { + throw new Error(`minPitch must be greater than or equal to ${defaultMinPitch}`); + } + + if (minPitch >= defaultMinPitch && minPitch <= this.transform.maxPitch) { + this.transform.minPitch = minPitch; + this._update(); + + if (this.getPitch() < minPitch) this.setPitch(minPitch); + + return this; + + } else throw new Error(`minPitch must be between ${defaultMinPitch} and the current maxPitch, inclusive`); + } + + /** + * Returns the map's minimum allowable pitch. + * + * @returns The minPitch + */ + getMinPitch(): number { return this.transform.minPitch; } + + /** + * Sets or clears the map's maximum pitch. + * If the map's current pitch is higher than the new maximum, + * the map will pitch to the new maximum. + * + * A {@link ErrorEvent} event will be fired if maxPitch is out of bounds. + * + * @param maxPitch - The maximum pitch to set (0-85). Values greater than 60 degrees are experimental and may result in rendering issues. If you encounter any, please raise an issue with details in the MapLibre project. + * If `null` or `undefined` is provided, the function removes the current maximum pitch (sets it to 60). + * @returns `this` + */ + setMaxPitch(maxPitch?: number | null): Map { + + maxPitch = maxPitch === null || maxPitch === undefined ? defaultMaxPitch : maxPitch; + + if (maxPitch > maxPitchThreshold) { + throw new Error(`maxPitch must be less than or equal to ${maxPitchThreshold}`); + } + + if (maxPitch >= this.transform.minPitch) { + this.transform.maxPitch = maxPitch; + this._update(); + + if (this.getPitch() > maxPitch) this.setPitch(maxPitch); + + return this; + + } else throw new Error('maxPitch must be greater than the current minPitch'); + } + + /** + * Returns the map's maximum allowable pitch. + * + * @returns The maxPitch + */ + getMaxPitch(): number { return this.transform.maxPitch; } + + /** + * Returns the state of `renderWorldCopies`. If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`: + * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire + * container, there will be blank space beyond 180 and -180 degrees longitude. + * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the + * map and the other on the left edge of the map) at every zoom level. + * @returns The renderWorldCopies + * @example + * ```ts + * let worldCopiesRendered = map.getRenderWorldCopies(); + * ``` + * @see [Render world copies](https://maplibre.org/maplibre-gl-js/docs/examples/render-world-copies/) + */ + getRenderWorldCopies(): boolean { return this.transform.renderWorldCopies; } + + /** + * Sets the state of `renderWorldCopies`. + * + * @param renderWorldCopies - If `true`, multiple copies of the world will be rendered side by side beyond -180 and 180 degrees longitude. If set to `false`: + * - When the map is zoomed out far enough that a single representation of the world does not fill the map's entire + * container, there will be blank space beyond 180 and -180 degrees longitude. + * - Features that cross 180 and -180 degrees longitude will be cut in two (with one portion on the right edge of the + * map and the other on the left edge of the map) at every zoom level. + * + * `undefined` is treated as `true`, `null` is treated as `false`. + * @returns `this` + * @example + * ```ts + * map.setRenderWorldCopies(true); + * ``` + * @see [Render world copies](https://maplibre.org/maplibre-gl-js/docs/examples/render-world-copies/) + */ + setRenderWorldCopies(renderWorldCopies?: boolean | null): Map { + this.transform.renderWorldCopies = renderWorldCopies; + return this._update(); + } + + /** + * Gets the map's cooperativeGestures option + * + * @returns The gestureOptions + */ + getCooperativeGestures(): boolean | GestureOptions { + return this._cooperativeGestures; + } + + /** + * Sets or clears the map's cooperativeGestures option + * + * @param gestureOptions - If `true` or set to an options object, map is only accessible on desktop while holding Command/Ctrl and only accessible on mobile with two fingers. Interacting with the map using normal gestures will trigger an informational screen. With this option enabled, "drag to pitch" requires a three-finger gesture. + * @returns `this` + */ + setCooperativeGestures(gestureOptions?: GestureOptions | boolean | null): Map { + this._cooperativeGestures = gestureOptions; + if (this._cooperativeGestures) { + this._setupCooperativeGestures(); + } else { + this._destroyCooperativeGestures(); + } + + return this; + } + + /** + * Returns a [Point](https://github.com/mapbox/point-geometry) representing pixel coordinates, relative to the map's `container`, + * that correspond to the specified geographical location. + * + * @param lnglat - The geographical location to project. + * @returns The [Point](https://github.com/mapbox/point-geometry) corresponding to `lnglat`, relative to the map's `container`. + * @example + * ```ts + * let coordinate = [-122.420679, 37.772537]; + * let point = map.project(coordinate); + * ``` + */ + project(lnglat: LngLatLike): Point { + return this.transform.locationPoint(LngLat.convert(lnglat), this.style && this.terrain); + } + + /** + * Returns a {@link LngLat} representing geographical coordinates that correspond + * to the specified pixel coordinates. + * + * @param point - The pixel coordinates to unproject. + * @returns The {@link LngLat} corresponding to `point`. + * @example + * ```ts + * map.on('click', function(e) { + * // When the map is clicked, get the geographic coordinate. + * let coordinate = map.unproject(e.point); + * }); + * ``` + */ + unproject(point: PointLike): LngLat { + return this.transform.pointLocation(Point.convert(point), this.terrain); + } + + /** + * Returns true if the map is panning, zooming, rotating, or pitching due to a camera animation or user gesture. + * @returns true if the map is moving. + * @example + * ```ts + * let isMoving = map.isMoving(); + * ``` + */ + isMoving(): boolean { + return this._moving || this.handlers?.isMoving(); + } + + /** + * Returns true if the map is zooming due to a camera animation or user gesture. + * @returns true if the map is zooming. + * @example + * ```ts + * let isZooming = map.isZooming(); + * ``` + */ + isZooming(): boolean { + return this._zooming || this.handlers?.isZooming(); + } + + /** + * Returns true if the map is rotating due to a camera animation or user gesture. + * @returns true if the map is rotating. + * @example + * ```ts + * map.isRotating(); + * ``` + */ + isRotating(): boolean { + return this._rotating || this.handlers?.isRotating(); + } + + _createDelegatedListener(type: keyof MapEventType | string, layerId: string, listener: Listener): { + layer: string; + listener: Listener; + delegates: {[type in keyof MapEventType]?: (e: any) => void}; + } { + if (type === 'mouseenter' || type === 'mouseover') { + let mousein = false; + const mousemove = (e) => { + const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, {layers: [layerId]}) : []; + if (!features.length) { + mousein = false; + } else if (!mousein) { + mousein = true; + listener.call(this, new MapMouseEvent(type, this, e.originalEvent, {features})); + } + }; + const mouseout = () => { + mousein = false; + }; + return {layer: layerId, listener, delegates: {mousemove, mouseout}}; + } else if (type === 'mouseleave' || type === 'mouseout') { + let mousein = false; + const mousemove = (e) => { + const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, {layers: [layerId]}) : []; + if (features.length) { + mousein = true; + } else if (mousein) { + mousein = false; + listener.call(this, new MapMouseEvent(type, this, e.originalEvent)); + } + }; + const mouseout = (e) => { + if (mousein) { + mousein = false; + listener.call(this, new MapMouseEvent(type, this, e.originalEvent)); + } + }; + return {layer: layerId, listener, delegates: {mousemove, mouseout}}; + } else { + const delegate = (e) => { + const features = this.getLayer(layerId) ? this.queryRenderedFeatures(e.point, {layers: [layerId]}) : []; + if (features.length) { + // Here we need to mutate the original event, so that preventDefault works as expected. + e.features = features; + listener.call(this, e); + delete e.features; + } + }; + return {layer: layerId, listener, delegates: {[type]: delegate}}; + } + } + + /** + * @event + * Adds a listener for events of a specified type, optionally limited to features in a specified style layer. + * See {@link MapEventType} and {@link MapLayerEventType} for a full list of events and their description. + * + * | Event | Compatible with `layerId` | + * |------------------------|---------------------------| + * | `mousedown` | yes | + * | `mouseup` | yes | + * | `mouseover` | yes | + * | `mouseout` | yes | + * | `mousemove` | yes | + * | `mouseenter` | yes (required) | + * | `mouseleave` | yes (required) | + * | `click` | yes | + * | `dblclick` | yes | + * | `contextmenu` | yes | + * | `touchstart` | yes | + * | `touchend` | yes | + * | `touchcancel` | yes | + * | `wheel` | | + * | `resize` | | + * | `remove` | | + * | `touchmove` | | + * | `movestart` | | + * | `move` | | + * | `moveend` | | + * | `dragstart` | | + * | `drag` | | + * | `dragend` | | + * | `zoomstart` | | + * | `zoom` | | + * | `zoomend` | | + * | `rotatestart` | | + * | `rotate` | | + * | `rotateend` | | + * | `pitchstart` | | + * | `pitch` | | + * | `pitchend` | | + * | `boxzoomstart` | | + * | `boxzoomend` | | + * | `boxzoomcancel` | | + * | `webglcontextlost` | | + * | `webglcontextrestored` | | + * | `load` | | + * | `render` | | + * | `idle` | | + * | `error` | | + * | `data` | | + * | `styledata` | | + * | `sourcedata` | | + * | `dataloading` | | + * | `styledataloading` | | + * | `sourcedataloading` | | + * | `styleimagemissing` | | + * | `dataabort` | | + * | `sourcedataabort` | | + * + * @param type - The event type to listen for. Events compatible with the optional `layerId` parameter are triggered + * when the cursor enters a visible portion of the specified layer from outside that layer or outside the map canvas. + * @param layer - The ID of a style layer or a listener if no ID is provided. Event will only be triggered if its location + * is within a visible feature in this layer. The event will have a `features` property containing + * an array of the matching features. If `layer` is not supplied, the event will not have a `features` property. + * Please note that many event types are not compatible with the optional `layer` parameter. + * @param listener - The function to be called when the event is fired. + * @returns `this` + * @example + * ```ts + * // Set an event listener that will fire + * // when the map has finished loading + * map.on('load', function() { + * // Once the map has finished loading, + * // add a new layer + * map.addLayer({ + * id: 'points-of-interest', + * source: { + * type: 'vector', + * url: 'https://maplibre.org/maplibre-style-spec/' + * }, + * 'source-layer': 'poi_label', + * type: 'circle', + * paint: { + * // MapLibre Style Specification paint properties + * }, + * layout: { + * // MapLibre Style Specification layout properties + * } + * }); + * }); + * ``` + * @example + * ```ts + * // Set an event listener that will fire + * // when a feature on the countries layer of the map is clicked + * map.on('click', 'countries', (e) => { + * new maplibregl.Popup() + * .setLngLat(e.lngLat) + * .setHTML(`Country name: ${e.features[0].properties.name}`) + * .addTo(map); + * }); + * ``` + * @see [Display popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/) + * @see [Create a hover effect](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + * @see [Create a draggable marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + on( + type: T, + layer: string, + listener: (ev: MapLayerEventType[T] & Object) => void, + ): Map; + /** + * Overload of the `on` method that allows to listen to events without specifying a layer. + * @event + * @param type - The type of the event. + * @param listener - The listener callback. + * @returns `this` + */ + on(type: T, listener: (ev: MapEventType[T] & Object) => void): this; + /** + * Overload of the `on` method that allows to listen to events without specifying a layer. + * @event + * @param type - The type of the event. + * @param listener - The listener callback. + * @returns `this` + */ + on(type: keyof MapEventType | string, listener: Listener): this; + on(type: keyof MapEventType | string, layerIdOrListener: string | Listener, listener?: Listener): this { + if (listener === undefined) { + return super.on(type, layerIdOrListener as Listener); + } + + const delegatedListener = this._createDelegatedListener(type, layerIdOrListener as string, listener); + + this._delegatedListeners = this._delegatedListeners || {}; + this._delegatedListeners[type] = this._delegatedListeners[type] || []; + this._delegatedListeners[type].push(delegatedListener); + + for (const event in delegatedListener.delegates) { + this.on(event, delegatedListener.delegates[event]); + } + + return this; + } + + /** + * Adds a listener that will be called only once to a specified event type, optionally limited to features in a specified style layer. + * + * @event + * @param type - The event type to listen for; one of `'mousedown'`, `'mouseup'`, `'click'`, `'dblclick'`, + * `'mousemove'`, `'mouseenter'`, `'mouseleave'`, `'mouseover'`, `'mouseout'`, `'contextmenu'`, `'touchstart'`, + * `'touchend'`, or `'touchcancel'`. `mouseenter` and `mouseover` events are triggered when the cursor enters + * a visible portion of the specified layer from outside that layer or outside the map canvas. `mouseleave` + * and `mouseout` events are triggered when the cursor leaves a visible portion of the specified layer, or leaves + * the map canvas. + * @param layer - The ID of a style layer or a listener if no ID is provided. Only events whose location is within a visible + * feature in this layer will trigger the listener. The event will have a `features` property containing + * an array of the matching features. + * @param listener - The function to be called when the event is fired. + * @returns `this` if listener is provided, promise otherwise to allow easier usage of async/await + */ + once( + type: T, + layer: string, + listener?: (ev: MapLayerEventType[T] & Object) => void, + ): this | Promise; + /** + * Overload of the `once` method that allows to listen to events without specifying a layer. + * @event + * @param type - The type of the event. + * @param listener - The listener callback. + * @returns `this` + */ + once(type: T, listener?: (ev: MapEventType[T] & Object) => void): this | Promise; + /** + * Overload of the `once` method that allows to listen to events without specifying a layer. + * @event + * @param type - The type of the event. + * @param listener - The listener callback. + * @returns `this` + */ + once(type: keyof MapEventType | string, listener?: Listener): this | Promise; + once(type: keyof MapEventType | string, layerIdOrListener: string | Listener, listener?: Listener): this | Promise { + + if (listener === undefined) { + return super.once(type, layerIdOrListener as Listener); + } + + const delegatedListener = this._createDelegatedListener(type, layerIdOrListener as string, listener); + + for (const event in delegatedListener.delegates) { + this.once(event, delegatedListener.delegates[event]); + } + + return this; + } + + /** + * Removes an event listener for events previously added with `Map#on`. + * + * @event + * @param type - The event type previously used to install the listener. + * @param layer - The layer ID or listener previously used to install the listener. + * @param listener - The function previously installed as a listener. + * @returns `this` + */ + off( + type: T, + layer: string, + listener: (ev: MapLayerEventType[T] & Object) => void, + ): this; + /** + * Overload of the `off` method that allows to listen to events without specifying a layer. + * @event + * @param type - The type of the event. + * @param listener - The function previously installed as a listener. + * @returns `this` + */ + off(type: T, listener: (ev: MapEventType[T] & Object) => void): this; + /** + * Overload of the `off` method that allows to listen to events without specifying a layer. + * @event + * @param type - The type of the event. + * @param listener - The function previously installed as a listener. + * @returns `this` + */ + off(type: keyof MapEventType | string, listener: Listener): this; + off(type: keyof MapEventType | string, layerIdOrListener: string | Listener, listener?: Listener): this { + if (listener === undefined) { + return super.off(type, layerIdOrListener as Listener); + } + + const removeDelegatedListener = (delegatedListeners) => { + const listeners = delegatedListeners[type]; + for (let i = 0; i < listeners.length; i++) { + const delegatedListener = listeners[i]; + if (delegatedListener.layer === layerIdOrListener && delegatedListener.listener === listener) { + for (const event in delegatedListener.delegates) { + this.off(((event as any)), delegatedListener.delegates[event]); + } + listeners.splice(i, 1); + return this; + } + } + }; + + if (this._delegatedListeners && this._delegatedListeners[type]) { + removeDelegatedListener(this._delegatedListeners); + } + + return this; + } + + /** + * Returns an array of MapGeoJSONFeature objects + * representing visible features that satisfy the query parameters. + * + * @param geometryOrOptions - (optional) The geometry of the query region: + * either a single point or southwest and northeast points describing a bounding box. + * Omitting this parameter (i.e. calling {@link Map#queryRenderedFeatures} with zero arguments, + * or with only a `options` argument) is equivalent to passing a bounding box encompassing the entire + * map viewport. + * The geometryOrOptions can receive a {@link QueryRenderedFeaturesOptions} only to support a situation where the function receives only one parameter which is the options parameter. + * @param options - (optional) Options object. + * + * @returns An array of MapGeoJSONFeature objects. + * + * The `properties` value of each returned feature object contains the properties of its source feature. For GeoJSON sources, only + * string and numeric property values are supported (i.e. `null`, `Array`, and `Object` values are not supported). + * + * Each feature includes top-level `layer`, `source`, and `sourceLayer` properties. The `layer` property is an object + * representing the style layer to which the feature belongs. Layout and paint properties in this object contain values + * which are fully evaluated for the given zoom level and feature. + * + * Only features that are currently rendered are included. Some features will **not** be included, like: + * + * - Features from layers whose `visibility` property is `"none"`. + * - Features from layers whose zoom range excludes the current zoom level. + * - Symbol features that have been hidden due to text or icon collision. + * + * Features from all other layers are included, including features that may have no visible + * contribution to the rendered result; for example, because the layer's opacity or color alpha component is set to + * 0. + * + * The topmost rendered feature appears first in the returned array, and subsequent features are sorted by + * descending z-order. Features that are rendered multiple times (due to wrapping across the antemeridian at low + * zoom levels) are returned only once (though subject to the following caveat). + * + * Because features come from tiled vector data or GeoJSON data that is converted to tiles internally, feature + * geometries may be split or duplicated across tile boundaries and, as a result, features may appear multiple + * times in query results. For example, suppose there is a highway running through the bounding rectangle of a query. + * The results of the query will be those parts of the highway that lie within the map tiles covering the bounding + * rectangle, even if the highway extends into other tiles, and the portion of the highway within each map tile + * will be returned as a separate feature. Similarly, a point feature near a tile boundary may appear in multiple + * tiles due to tile buffering. + * + * @example + * Find all features at a point + * ```ts + * let features = map.queryRenderedFeatures( + * [20, 35], + * { layers: ['my-layer-name'] } + * ); + * ``` + * + * @example + * Find all features within a static bounding box + * ```ts + * let features = map.queryRenderedFeatures( + * [[10, 20], [30, 50]], + * { layers: ['my-layer-name'] } + * ); + * ``` + * + * @example + * Find all features within a bounding box around a point + * ```ts + * let width = 10; + * let height = 20; + * let features = map.queryRenderedFeatures([ + * [point.x - width / 2, point.y - height / 2], + * [point.x + width / 2, point.y + height / 2] + * ], { layers: ['my-layer-name'] }); + * ``` + * + * @example + * Query all rendered features from a single layer + * ```ts + * let features = map.queryRenderedFeatures({ layers: ['my-layer-name'] }); + * ``` + * @see [Get features under the mouse pointer](https://maplibre.org/maplibre-gl-js/docs/examples/queryrenderedfeatures/) + */ + queryRenderedFeatures(geometryOrOptions?: PointLike | [PointLike, PointLike] | QueryRenderedFeaturesOptions, options?: QueryRenderedFeaturesOptions): MapGeoJSONFeature[] { + if (!this.style) { + return []; + } + let queryGeometry; + const isGeometry = geometryOrOptions instanceof Point || Array.isArray(geometryOrOptions); + const geometry = isGeometry ? geometryOrOptions : [[0, 0], [this.transform.width, this.transform.height]]; + options = options || (isGeometry ? {} : geometryOrOptions) || {}; + + if (geometry instanceof Point || typeof geometry[0] === 'number') { + queryGeometry = [Point.convert(geometry as PointLike)]; + } else { + const tl = Point.convert(geometry[0] as PointLike); + const br = Point.convert(geometry[1] as PointLike); + queryGeometry = [tl, new Point(br.x, tl.y), br, new Point(tl.x, br.y), tl]; + } + + return this.style.queryRenderedFeatures(queryGeometry, options, this.transform); + } + + /** + * Returns an array of MapGeoJSONFeature objects + * representing features within the specified vector tile or GeoJSON source that satisfy the query parameters. + * + * @param sourceId - The ID of the vector tile or GeoJSON source to query. + * @param parameters - The options object. + * @returns An array of MapGeoJSONFeature objects. + * + * In contrast to {@link Map#queryRenderedFeatures}, this function returns all features matching the query parameters, + * whether or not they are rendered by the current style (i.e. visible). The domain of the query includes all currently-loaded + * vector tiles and GeoJSON source tiles: this function does not check tiles outside the currently + * visible viewport. + * + * Because features come from tiled vector data or GeoJSON data that is converted to tiles internally, feature + * geometries may be split or duplicated across tile boundaries and, as a result, features may appear multiple + * times in query results. For example, suppose there is a highway running through the bounding rectangle of a query. + * The results of the query will be those parts of the highway that lie within the map tiles covering the bounding + * rectangle, even if the highway extends into other tiles, and the portion of the highway within each map tile + * will be returned as a separate feature. Similarly, a point feature near a tile boundary may appear in multiple + * tiles due to tile buffering. + * + * @example + * Find all features in one source layer in a vector source + * ```ts + * let features = map.querySourceFeatures('your-source-id', { + * sourceLayer: 'your-source-layer' + * }); + * ``` + * + */ + querySourceFeatures(sourceId: string, parameters?: QuerySourceFeatureOptions | null): MapGeoJSONFeature[] { + return this.style.querySourceFeatures(sourceId, parameters); + } + + /** + * Updates the map's MapLibre style object with a new value. + * + * If a style is already set when this is used and options.diff is set to true, the map renderer will attempt to compare the given style + * against the map's current state and perform only the changes necessary to make the map style match the desired state. Changes in sprites + * (images used for icons and patterns) and glyphs (fonts for label text) **cannot** be diffed. If the sprites or fonts used in the current + * style and the given style are different in any way, the map renderer will force a full update, removing the current style and building + * the given one from scratch. + * + * + * @param style - A JSON object conforming to the schema described in the + * [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/), or a URL to such JSON. + * @param options - The options object. + * @returns `this` + * + * @example + * ```ts + * map.setStyle("https://demotiles.maplibre.org/style.json"); + * + * map.setStyle('https://demotiles.maplibre.org/style.json', { + * transformStyle: (previousStyle, nextStyle) => ({ + * ...nextStyle, + * sources: { + * ...nextStyle.sources, + * // copy a source from previous style + * 'osm': previousStyle.sources.osm + * }, + * layers: [ + * // background layer + * nextStyle.layers[0], + * // copy a layer from previous style + * previousStyle.layers[0], + * // other layers from the next style + * ...nextStyle.layers.slice(1).map(layer => { + * // hide the layers we don't need from demotiles style + * if (layer.id.startsWith('geolines')) { + * layer.layout = {...layer.layout || {}, visibility: 'none'}; + * // filter out US polygons + * } else if (layer.id.startsWith('coastline') || layer.id.startsWith('countries')) { + * layer.filter = ['!=', ['get', 'ADM0_A3'], 'USA']; + * } + * return layer; + * }) + * ] + * }) + * }); + * ``` + */ + setStyle(style: StyleSpecification | string | null, options?: StyleSwapOptions & StyleOptions): this { + options = extend({}, + { + localIdeographFontFamily: this._localIdeographFontFamily, + validate: this._validateStyle + }, options); + + if ((options.diff !== false && options.localIdeographFontFamily === this._localIdeographFontFamily) && this.style && style) { + this._diffStyle(style, options); + return this; + } else { + this._localIdeographFontFamily = options.localIdeographFontFamily; + return this._updateStyle(style, options); + } + } + + /** + * Updates the requestManager's transform request with a new function + * + * @param transformRequest - A callback run before the Map makes a request for an external URL. The callback can be used to modify the url, set headers, or set the credentials property for cross-origin requests. + * Expected to return an object with a `url` property and optionally `headers` and `credentials` properties + * + * @returns `this` + * + * @example + * ```ts + * map.setTransformRequest((url: string, resourceType: string) => {}); + * ``` + */ + setTransformRequest(transformRequest: RequestTransformFunction): this { + this._requestManager.setTransformRequest(transformRequest); + return this; + } + + _getUIString(key: string) { + const str = this._locale[key]; + if (str == null) { + throw new Error(`Missing UI string '${key}'`); + } + + return str; + } + + _updateStyle(style: StyleSpecification | string | null, options?: StyleSwapOptions & StyleOptions) { + // transformStyle relies on having previous style serialized, if it is not loaded yet, delay _updateStyle until previous style is loaded + if (options.transformStyle && this.style && !this.style._loaded) { + this.style.once('style.load', () => this._updateStyle(style, options)); + return; + } + + const previousStyle = this.style && options.transformStyle ? this.style.serialize() : undefined; + if (this.style) { + this.style.setEventedParent(null); + + // Only release workers when map is getting disposed + this.style._remove(!style); + } + + if (!style) { + delete this.style; + return this; + } else { + this.style = new Style(this, options || {}); + } + + this.style.setEventedParent(this, {style: this.style}); + + if (typeof style === 'string') { + this.style.loadURL(style, options, previousStyle); + } else { + this.style.loadJSON(style, options, previousStyle); + } + + return this; + } + + _lazyInitEmptyStyle() { + if (!this.style) { + this.style = new Style(this, {}); + this.style.setEventedParent(this, {style: this.style}); + this.style.loadEmpty(); + } + } + + _diffStyle(style: StyleSpecification | string, options?: StyleSwapOptions & StyleOptions) { + if (typeof style === 'string') { + const url = style; + const request = this._requestManager.transformRequest(url, ResourceType.Style); + getJSON(request, (error?: Error | null, json?: any | null) => { + if (error) { + this.fire(new ErrorEvent(error)); + } else if (json) { + this._updateDiff(json, options); + } + }); + } else if (typeof style === 'object') { + this._updateDiff(style, options); + } + } + + _updateDiff(style: StyleSpecification, options?: StyleSwapOptions & StyleOptions) { + try { + if (this.style.setState(style, options)) { + this._update(true); + } + } catch (e) { + warnOnce( + `Unable to perform style diff: ${e.message || e.error || e}. Rebuilding the style from scratch.` + ); + this._updateStyle(style, options); + } + } + + /** + * Returns the map's MapLibre style object, a JSON object which can be used to recreate the map's style. + * + * @returns The map's style JSON object. + * + * @example + * ```ts + * let styleJson = map.getStyle(); + * ``` + * + */ + getStyle(): StyleSpecification { + if (this.style) { + return this.style.serialize(); + } + } + + /** + * Returns a Boolean indicating whether the map's style is fully loaded. + * + * @returns A Boolean indicating whether the style is fully loaded. + * + * @example + * ```ts + * let styleLoadStatus = map.isStyleLoaded(); + * ``` + */ + isStyleLoaded(): boolean | void { + if (!this.style) return warnOnce('There is no style added to the map.'); + return this.style.loaded(); + } + + /** + * Adds a source to the map's style. + * + * Events triggered: + * + * Triggers the `source.add` event. + * + * @param id - The ID of the source to add. Must not conflict with existing sources. + * @param source - The source object, conforming to the + * MapLibre Style Specification's [source definition](https://maplibre.org/maplibre-style-spec/sources) or + * {@link CanvasSourceSpecification}. + * @returns `this` + * @example + * ```ts + * map.addSource('my-data', { + * type: 'vector', + * url: 'https://demotiles.maplibre.org/tiles/tiles.json' + * }); + * ``` + * @example + * ```ts + * map.addSource('my-data', { + * "type": "geojson", + * "data": { + * "type": "Feature", + * "geometry": { + * "type": "Point", + * "coordinates": [-77.0323, 38.9131] + * }, + * "properties": { + * "title": "Mapbox DC", + * "marker-symbol": "monument" + * } + * } + * }); + * ``` + * @see GeoJSON source: [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/) + */ + addSource(id: string, source: SourceSpecification): this { + this._lazyInitEmptyStyle(); + this.style.addSource(id, source); + return this._update(true); + } + + /** + * Returns a Boolean indicating whether the source is loaded. Returns `true` if the source with + * the given ID in the map's style has no outstanding network requests, otherwise `false`. + * + * A {@link ErrorEvent} event will be fired if there is no source wit the specified ID. + * + * @param id - The ID of the source to be checked. + * @returns A Boolean indicating whether the source is loaded. + * @example + * ```ts + * let sourceLoaded = map.isSourceLoaded('bathymetry-data'); + * ``` + */ + isSourceLoaded(id: string): boolean { + const source = this.style && this.style.sourceCaches[id]; + if (source === undefined) { + this.fire(new ErrorEvent(new Error(`There is no source with ID '${id}'`))); + return; + } + return source.loaded(); + } + + /** + * Loads a 3D terrain mesh, based on a "raster-dem" source. + * + * Triggers the `terrain` event. + * + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.setTerrain({ source: 'terrain' }); + * ``` + */ + setTerrain(options: TerrainSpecification | null): this { + this.style._checkLoaded(); + + // clear event handlers + if (this._terrainDataCallback) this.style.off('data', this._terrainDataCallback); + + if (!options) { + // remove terrain + if (this.terrain) this.terrain.sourceCache.destruct(); + this.terrain = null; + if (this.painter.renderToTexture) this.painter.renderToTexture.destruct(); + this.painter.renderToTexture = null; + this.transform._minEleveationForCurrentTile = 0; + this.transform.elevation = 0; + } else { + // add terrain + const sourceCache = this.style.sourceCaches[options.source]; + if (!sourceCache) throw new Error(`cannot load terrain, because there exists no source with ID: ${options.source}`); + // Warn once if user is using the same source for hillshade and terrain + for (const index in this.style._layers) { + const thisLayer = this.style._layers[index]; + if (thisLayer.type === 'hillshade' && thisLayer.source === options.source) { + warnOnce('You are using the same source for a hillshade layer and for 3D terrain. Please consider using two separate sources to improve rendering quality.'); + } + } + this.terrain = new Terrain(this.painter, sourceCache, options); + this.painter.renderToTexture = new RenderToTexture(this.painter, this.terrain); + this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom); + this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom); + this._terrainDataCallback = e => { + if (e.dataType === 'style') { + this.terrain.sourceCache.freeRtt(); + } else if (e.dataType === 'source' && e.tile) { + if (e.sourceId === options.source && !this._elevationFreeze) { + this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom); + this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom); + } + this.terrain.sourceCache.freeRtt(e.tile.tileID); + } + }; + this.style.on('data', this._terrainDataCallback); + } + + this.fire(new Event('terrain', {terrain: options})); + return this; + } + + /** + * Get the terrain-options if terrain is loaded + * @returns the TerrainSpecification passed to setTerrain + * @example + * ```ts + * map.getTerrain(); // { source: 'terrain' }; + * ``` + */ + getTerrain(): TerrainSpecification | null { + return this.terrain?.options ?? null; + } + + /** + * Returns a Boolean indicating whether all tiles in the viewport from all sources on + * the style are loaded. + * + * @returns A Boolean indicating whether all tiles are loaded. + * @example + * ```ts + * let tilesLoaded = map.areTilesLoaded(); + * ``` + */ + areTilesLoaded(): boolean { + const sources = this.style && this.style.sourceCaches; + for (const id in sources) { + const source = sources[id]; + const tiles = source._tiles; + for (const t in tiles) { + const tile = tiles[t]; + if (!(tile.state === 'loaded' || tile.state === 'errored')) return false; + } + } + return true; + } + + /** + * Adds a [custom source type](#Custom Sources), making it available for use with + * {@link Map#addSource}. + * @param name - The name of the source type; source definition objects use this name in the `{type: ...}` field. + * @param SourceType - A {@link Source} constructor. + * @param callback - Called when the source type is ready or with an error argument if there is an error. + */ + addSourceType(name: string, SourceType: SourceClass, callback: Callback) { + this._lazyInitEmptyStyle(); + return this.style.addSourceType(name, SourceType, callback); + } + + /** + * Removes a source from the map's style. + * + * @param id - The ID of the source to remove. + * @returns `this` + * @example + * ```ts + * map.removeSource('bathymetry-data'); + * ``` + */ + removeSource(id: string): Map { + this.style.removeSource(id); + return this._update(true); + } + + /** + * Returns the source with the specified ID in the map's style. + * + * This method is often used to update a source using the instance members for the relevant + * source type as defined in [Sources](#sources). + * For example, setting the `data` for a GeoJSON source or updating the `url` and `coordinates` + * of an image source. + * + * @param id - The ID of the source to get. + * @returns The style source with the specified ID or `undefined` if the ID + * corresponds to no existing sources. + * The shape of the object varies by source type. + * A list of options for each source type is available on the MapLibre Style Specification's + * [Sources](https://maplibre.org/maplibre-style-spec/sources/) page. + * @example + * ```ts + * let sourceObject = map.getSource('points'); + * ``` + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + * @see [Animate a point](https://maplibre.org/maplibre-gl-js/docs/examples/animate-point-along-line/) + * @see [Add live realtime data](https://maplibre.org/maplibre-gl-js/docs/examples/live-geojson/) + */ + getSource(id: string): Source | undefined { + return this.style.getSource(id); + } + + /** + * Add an image to the style. This image can be displayed on the map like any other icon in the style's + * sprite using the image's ID with + * [`icon-image`](https://maplibre.org/maplibre-style-spec/layers/#layout-symbol-icon-image), + * [`background-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-background-background-pattern), + * [`fill-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-fill-fill-pattern), + * or [`line-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-line-line-pattern). + * + * A {@link ErrorEvent} event will be fired if the image parameter is invalid or there is not enough space in the sprite to add this image. + * + * @param id - The ID of the image. + * @param image - The image as an `HTMLImageElement`, `ImageData`, `ImageBitmap` or object with `width`, `height`, and `data` + * properties with the same format as `ImageData`. + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * // If the style's sprite does not already contain an image with ID 'cat', + * // add the image 'cat-icon.png' to the style's sprite with the ID 'cat'. + * map.loadImage('https://upload.wikimedia.org/wikipedia/commons/thumb/6/60/Cat_silhouette.svg/400px-Cat_silhouette.svg.png', function(error, image) { + * if (error) throw error; + * if (!map.hasImage('cat')) map.addImage('cat', image); + * }); + * + * // Add a stretchable image that can be used with `icon-text-fit` + * // In this example, the image is 600px wide by 400px high. + * map.loadImage('https://upload.wikimedia.org/wikipedia/commons/8/89/Black_and_White_Boxed_%28bordered%29.png', function(error, image) { + * if (error) throw error; + * if (!map.hasImage('border-image')) { + * map.addImage('border-image', image, { + * content: [16, 16, 300, 384], // place text over left half of image, avoiding the 16px border + * stretchX: [[16, 584]], // stretch everything horizontally except the 16px border + * stretchY: [[16, 384]], // stretch everything vertically except the 16px border + * }); + * } + * }); + * ``` + * @see Use `HTMLImageElement`: [Add an icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image/) + * @see Use `ImageData`: [Add a generated icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-generated/) + */ + addImage(id: string, + image: HTMLImageElement | ImageBitmap | ImageData | { + width: number; + height: number; + data: Uint8Array | Uint8ClampedArray; + } | StyleImageInterface, + options: Partial = {}): this { + const { + pixelRatio = 1, + sdf = false, + stretchX, + stretchY, + content + } = options; + this._lazyInitEmptyStyle(); + const version = 0; + + if (image instanceof HTMLImageElement || isImageBitmap(image)) { + const {width, height, data} = browser.getImageData(image); + this.style.addImage(id, {data: new RGBAImage({width, height}, data), pixelRatio, stretchX, stretchY, content, sdf, version}); + } else if (image.width === undefined || image.height === undefined) { + return this.fire(new ErrorEvent(new Error( + 'Invalid arguments to map.addImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, ' + + 'or object with `width`, `height`, and `data` properties with the same format as `ImageData`'))); + } else { + const {width, height, data} = image as ImageData; + const userImage = (image as any as StyleImageInterface); + + this.style.addImage(id, { + data: new RGBAImage({width, height}, new Uint8Array(data)), + pixelRatio, + stretchX, + stretchY, + content, + sdf, + version, + userImage + }); + + if (userImage.onAdd) { + userImage.onAdd(this, id); + } + return this; + } + } + + /** + * Update an existing image in a style. This image can be displayed on the map like any other icon in the style's + * sprite using the image's ID with + * [`icon-image`](https://maplibre.org/maplibre-style-spec/layers/#layout-symbol-icon-image), + * [`background-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-background-background-pattern), + * [`fill-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-fill-fill-pattern), + * or [`line-pattern`](https://maplibre.org/maplibre-style-spec/layers/#paint-line-line-pattern). + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param id - The ID of the image. + * @param image - The image as an `HTMLImageElement`, `ImageData`, `ImageBitmap` or object with `width`, `height`, and `data` + * properties with the same format as `ImageData`. + * @returns `this` + * @example + * ```ts + * // If an image with the ID 'cat' already exists in the style's sprite, + * // replace that image with a new image, 'other-cat-icon.png'. + * if (map.hasImage('cat')) map.updateImage('cat', './other-cat-icon.png'); + * ``` + */ + updateImage(id: string, + image: HTMLImageElement | ImageBitmap | ImageData | { + width: number; + height: number; + data: Uint8Array | Uint8ClampedArray; + } | StyleImageInterface): this { + + const existingImage = this.style.getImage(id); + if (!existingImage) { + return this.fire(new ErrorEvent(new Error( + 'The map has no image with that id. If you are adding a new image use `map.addImage(...)` instead.'))); + } + const imageData = (image instanceof HTMLImageElement || isImageBitmap(image)) ? + browser.getImageData(image) : + image; + const {width, height, data} = imageData; + + if (width === undefined || height === undefined) { + return this.fire(new ErrorEvent(new Error( + 'Invalid arguments to map.updateImage(). The second argument must be an `HTMLImageElement`, `ImageData`, `ImageBitmap`, ' + + 'or object with `width`, `height`, and `data` properties with the same format as `ImageData`'))); + } + + if (width !== existingImage.data.width || height !== existingImage.data.height) { + return this.fire(new ErrorEvent(new Error( + 'The width and height of the updated image must be that same as the previous version of the image'))); + } + + const copy = !(image instanceof HTMLImageElement || isImageBitmap(image)); + existingImage.data.replace(data, copy); + + this.style.updateImage(id, existingImage); + return this; + } + + /** + * Returns an image, specified by ID, currently available in the map. + * This includes both images from the style's original sprite + * and any images that have been added at runtime using {@link Map#addImage}. + * + * @param id - The ID of the image. + * @returns An image in the map with the specified ID. + * + * @example + * ```ts + * let coffeeShopIcon = map.getImage("coffee_cup"); + * ``` + */ + getImage(id: string): StyleImage { + return this.style.getImage(id); + } + + /** + * Check whether or not an image with a specific ID exists in the style. This checks both images + * in the style's original sprite and any images + * that have been added at runtime using {@link Map#addImage}. + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param id - The ID of the image. + * + * @returns A Boolean indicating whether the image exists. + * @example + * Check if an image with the ID 'cat' exists in the style's sprite. + * ```ts + * let catIconExists = map.hasImage('cat'); + * ``` + */ + hasImage(id: string): boolean { + if (!id) { + this.fire(new ErrorEvent(new Error('Missing required image id'))); + return false; + } + + return !!this.style.getImage(id); + } + + /** + * Remove an image from a style. This can be an image from the style's original + * sprite or any images + * that have been added at runtime using {@link Map#addImage}. + * + * @param id - The ID of the image. + * + * @example + * ```ts + * // If an image with the ID 'cat' exists in + * // the style's sprite, remove it. + * if (map.hasImage('cat')) map.removeImage('cat'); + * ``` + */ + removeImage(id: string) { + this.style.removeImage(id); + } + + /** + * Load an image from an external URL to be used with {@link Map#addImage}. External + * domains must support [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/Access_control_CORS). + * + * @param url - The URL of the image file. Image file must be in png, webp, or jpg format. + * @param callback - Expecting `callback(error, data)`. Called when the image has loaded or with an error argument if there is an error. + * + * @example + * Load an image from an external URL. + * ```ts + * map.loadImage('http://placekitten.com/50/50', function(error, image) { + * if (error) throw error; + * // Add the loaded image to the style's sprite with the ID 'kitten'. + * map.addImage('kitten', image); + * }); + * ``` + * @see [Add an icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image/) + */ + loadImage(url: string, callback: GetImageCallback) { + ImageRequest.getImage(this._requestManager.transformRequest(url, ResourceType.Image), callback); + } + + /** + * Returns an Array of strings containing the IDs of all images currently available in the map. + * This includes both images from the style's original sprite + * and any images that have been added at runtime using {@link Map#addImage}. + * + * @returns An Array of strings containing the names of all sprites/images currently available in the map. + * + * @example + * ```ts + * let allImages = map.listImages(); + * ``` + */ + listImages(): Array { + return this.style.listImages(); + } + + /** + * Adds a [MapLibre style layer](https://maplibre.org/maplibre-style-spec/layers) + * to the map's style. + * + * A layer defines how data from a specified source will be styled. Read more about layer types + * and available paint and layout properties in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/layers). + * + * @param layer - The layer to add, + * conforming to either the MapLibre Style Specification's [layer definition](https://maplibre.org/maplibre-style-spec/layers) or, + * less commonly, the {@link CustomLayerInterface} specification. Can also be a layer definition with an embedded source definition. + * The MapLibre Style Specification's layer definition is appropriate for most layers. + * + * @param beforeId - The ID of an existing layer to insert the new layer before, + * resulting in the new layer appearing visually beneath the existing layer. + * If this argument is not specified, the layer will be appended to the end of the layers array + * and appear visually above all other layers. + * + * @returns `this` + * + * @example + * Add a circle layer with a vector source + * ```ts + * map.addLayer({ + * id: 'points-of-interest', + * source: { + * type: 'vector', + * url: 'https://demotiles.maplibre.org/tiles/tiles.json' + * }, + * 'source-layer': 'poi_label', + * type: 'circle', + * paint: { + * // MapLibre Style Specification paint properties + * }, + * layout: { + * // MapLibre Style Specification layout properties + * } + * }); + * ``` + * + * @example + * Define a source before using it to create a new layer + * ```ts + * map.addSource('state-data', { + * type: 'geojson', + * data: 'path/to/data.geojson' + * }); + * + * map.addLayer({ + * id: 'states', + * // References the GeoJSON source defined above + * // and does not require a `source-layer` + * source: 'state-data', + * type: 'symbol', + * layout: { + * // Set the label content to the + * // feature's `name` property + * text-field: ['get', 'name'] + * } + * }); + * ``` + * + * @example + * Add a new symbol layer before an existing layer + * ```ts + * map.addLayer({ + * id: 'states', + * // References a source that's already been defined + * source: 'state-data', + * type: 'symbol', + * layout: { + * // Set the label content to the + * // feature's `name` property + * text-field: ['get', 'name'] + * } + * // Add the layer before the existing `cities` layer + * }, 'cities'); + * ``` + * @see [Create and style clusters](https://maplibre.org/maplibre-gl-js/docs/examples/cluster/) + * @see [Add a vector tile source](https://maplibre.org/maplibre-gl-js/docs/examples/vector-source/) + * @see [Add a WMS source](https://maplibre.org/maplibre-gl-js/docs/examples/wms/) + */ + addLayer(layer: AddLayerObject, beforeId?: string) { + this._lazyInitEmptyStyle(); + this.style.addLayer(layer, beforeId); + return this._update(true); + } + + /** + * Moves a layer to a different z-position. + * + * @param id - The ID of the layer to move. + * @param beforeId - The ID of an existing layer to insert the new layer before. When viewing the map, the `id` layer will appear beneath the `beforeId` layer. If `beforeId` is omitted, the layer will be appended to the end of the layers array and appear above all other layers on the map. + * @returns `this` + * + * @example + * Move a layer with ID 'polygon' before the layer with ID 'country-label'. The `polygon` layer will appear beneath the `country-label` layer on the map. + * ```ts + * map.moveLayer('polygon', 'country-label'); + * ``` + */ + moveLayer(id: string, beforeId?: string): this { + this.style.moveLayer(id, beforeId); + return this._update(true); + } + + /** + * Removes the layer with the given ID from the map's style. + * + * An {@link ErrorEvent} will be fired if the image parameter is invald. + * + * @param id - The ID of the layer to remove + * @returns `this` + * + * @example + * If a layer with ID 'state-data' exists, remove it. + * ```ts + * if (map.getLayer('state-data')) map.removeLayer('state-data'); + * ``` + */ + removeLayer(id: string): this { + this.style.removeLayer(id); + return this._update(true); + } + + /** + * Returns the layer with the specified ID in the map's style. + * + * @param id - The ID of the layer to get. + * @returns The layer with the specified ID, or `undefined` + * if the ID corresponds to no existing layers. + * + * @example + * ```ts + * let stateDataLayer = map.getLayer('state-data'); + * ``` + * @see [Filter symbols by toggling a list](https://maplibre.org/maplibre-gl-js/docs/examples/filter-markers/) + * @see [Filter symbols by text input](https://maplibre.org/maplibre-gl-js/docs/examples/filter-markers-by-input/) + */ + getLayer(id: string): StyleLayer | undefined { + return this.style.getLayer(id); + } + + /** + * Return the ids of all layers currently in the style, including custom layers, in order. + * + * @returns ids of layers, in order + * + * @example + * ```ts + * const orderedLayerIds = map.getLayersOrder(); + * ``` + */ + getLayersOrder(): string[] { + return this.style.getLayersOrder(); + } + + /** + * Sets the zoom extent for the specified style layer. The zoom extent includes the + * [minimum zoom level](https://maplibre.org/maplibre-style-spec/layers/#minzoom) + * and [maximum zoom level](https://maplibre.org/maplibre-style-spec/layers/#maxzoom)) + * at which the layer will be rendered. + * + * Note: For style layers using vector sources, style layers cannot be rendered at zoom levels lower than the + * minimum zoom level of the _source layer_ because the data does not exist at those zoom levels. If the minimum + * zoom level of the source layer is higher than the minimum zoom level defined in the style layer, the style + * layer will not be rendered at all zoom levels in the zoom range. + * + * @param layerId - The ID of the layer to which the zoom extent will be applied. + * @param minzoom - The minimum zoom to set (0-24). + * @param maxzoom - The maximum zoom to set (0-24). + * @returns `this` + * + * @example + * ```ts + * map.setLayerZoomRange('my-layer', 2, 5); + * ``` + */ + setLayerZoomRange(layerId: string, minzoom: number, maxzoom: number): this { + this.style.setLayerZoomRange(layerId, minzoom, maxzoom); + return this._update(true); + } + + /** + * Sets the filter for the specified style layer. + * + * Filters control which features a style layer renders from its source. + * Any feature for which the filter expression evaluates to `true` will be + * rendered on the map. Those that are false will be hidden. + * + * Use `setFilter` to show a subset of your source data. + * + * To clear the filter, pass `null` or `undefined` as the second parameter. + * + * @param layerId - The ID of the layer to which the filter will be applied. + * @param filter - The filter, conforming to the MapLibre Style Specification's + * [filter definition](https://maplibre.org/maplibre-style-spec/layers/#filter). If `null` or `undefined` is provided, the function removes any existing filter from the layer. + * @param options - Options object. + * @returns `this` + * + * @example + * Display only features with the 'name' property 'USA' + * ```ts + * map.setFilter('my-layer', ['==', ['get', 'name'], 'USA']); + * ``` + * @example + * Display only features with five or more 'available-spots' + * ```ts + * map.setFilter('bike-docks', ['>=', ['get', 'available-spots'], 5]); + * ``` + * @example + * Remove the filter for the 'bike-docks' style layer + * ```ts + * map.setFilter('bike-docks', null); + * ``` + * @see [Create a timeline animation](https://maplibre.org/maplibre-gl-js/docs/examples/timeline-animation/) + */ + setFilter(layerId: string, filter?: FilterSpecification | null, options: StyleSetterOptions = {}) { + this.style.setFilter(layerId, filter, options); + return this._update(true); + } + + /** + * Returns the filter applied to the specified style layer. + * + * @param layerId - The ID of the style layer whose filter to get. + * @returns The layer's filter. + */ + getFilter(layerId: string): FilterSpecification | void { + return this.style.getFilter(layerId); + } + + /** + * Sets the value of a paint property in the specified style layer. + * + * @param layerId - The ID of the layer to set the paint property in. + * @param name - The name of the paint property to set. + * @param value - The value of the paint property to set. + * Must be of a type appropriate for the property, as defined in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/). + * Pass `null` to unset the existing value. + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.setPaintProperty('my-layer', 'fill-color', '#faafee'); + * ``` + * @see [Change a layer's color with buttons](https://maplibre.org/maplibre-gl-js/docs/examples/color-switcher/) + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + setPaintProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}): this { + this.style.setPaintProperty(layerId, name, value, options); + return this._update(true); + } + + /** + * Returns the value of a paint property in the specified style layer. + * + * @param layerId - The ID of the layer to get the paint property from. + * @param name - The name of a paint property to get. + * @returns The value of the specified paint property. + */ + getPaintProperty(layerId: string, name: string) { + return this.style.getPaintProperty(layerId, name); + } + + /** + * Sets the value of a layout property in the specified style layer. + * + * @param layerId - The ID of the layer to set the layout property in. + * @param name - The name of the layout property to set. + * @param value - The value of the layout property. Must be of a type appropriate for the property, as defined in the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/). + * @param options - The options object. + * @returns `this` + * @example + * ```ts + * map.setLayoutProperty('my-layer', 'visibility', 'none'); + * ``` + */ + setLayoutProperty(layerId: string, name: string, value: any, options: StyleSetterOptions = {}): this { + this.style.setLayoutProperty(layerId, name, value, options); + return this._update(true); + } + + /** + * Returns the value of a layout property in the specified style layer. + * + * @param layerId - The ID of the layer to get the layout property from. + * @param name - The name of the layout property to get. + * @returns The value of the specified layout property. + */ + getLayoutProperty(layerId: string, name: string) { + return this.style.getLayoutProperty(layerId, name); + } + + /** + * Sets the value of the style's glyphs property. + * + * @param glyphsUrl - Glyph URL to set. Must conform to the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/glyphs/). + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.setGlyphs('https://demotiles.maplibre.org/font/{fontstack}/{range}.pbf'); + * ``` + */ + setGlyphs(glyphsUrl: string | null, options: StyleSetterOptions = {}): this { + this._lazyInitEmptyStyle(); + this.style.setGlyphs(glyphsUrl, options); + return this._update(true); + } + + /** + * Returns the value of the style's glyphs URL + * + * @returns glyphs Style's glyphs url + */ + getGlyphs(): string | null { + return this.style.getGlyphsUrl(); + } + + /** + * Adds a sprite to the map's style. Fires the `style` event. + * + * @param id - The ID of the sprite to add. Must not conflict with existing sprites. + * @param url - The URL to load the sprite from + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.addSprite('sprite-two', 'http://example.com/sprite-two'); + * ``` + */ + addSprite(id: string, url: string, options: StyleSetterOptions = {}): this { + this._lazyInitEmptyStyle(); + this.style.addSprite(id, url, options, (err) => { + if (!err) { + this._update(true); + } + }); + return this; + } + + /** + * Removes the sprite from the map's style. Fires the `style` event. + * + * @param id - The ID of the sprite to remove. If the sprite is declared as a single URL, the ID must be "default". + * @returns `this` + * @example + * ```ts + * map.removeSprite('sprite-two'); + * map.removeSprite('default'); + * ``` + */ + removeSprite(id: string) { + this._lazyInitEmptyStyle(); + this.style.removeSprite(id); + return this._update(true); + } + + /** + * Returns the as-is value of the style's sprite. + * + * @returns style's sprite list of id-url pairs + */ + getSprite(): {id: string; url: string}[] { + return this.style.getSprite(); + } + + /** + * Sets the value of the style's sprite property. + * + * @param spriteUrl - Sprite URL to set. + * @param options - Options object. + * @returns `this` + * @example + * ```ts + * map.setSprite('YOUR_SPRITE_URL'); + * ``` + */ + setSprite(spriteUrl: string | null, options: StyleSetterOptions = {}) { + this._lazyInitEmptyStyle(); + this.style.setSprite(spriteUrl, options, (err) => { + if (!err) { + this._update(true); + } + }); + return this; + } + + /** + * Sets the any combination of light values. + * + * @param light - Light properties to set. Must conform to the [MapLibre Style Specification](https://maplibre.org/maplibre-style-spec/light). + * @param options - Options object. + * @returns `this` + * + * @example + * ```ts + * let layerVisibility = map.getLayoutProperty('my-layer', 'visibility'); + * ``` + */ + setLight(light: LightSpecification, options: StyleSetterOptions = {}) { + this._lazyInitEmptyStyle(); + this.style.setLight(light, options); + return this._update(true); + } + + /** + * Returns the value of the light object. + * + * @returns light Light properties of the style. + */ + getLight(): LightSpecification { + return this.style.getLight(); + } + + /** + * Sets the `state` of a feature. + * A feature's `state` is a set of user-defined key-value pairs that are assigned to a feature at runtime. + * When using this method, the `state` object is merged with any existing key-value pairs in the feature's state. + * Features are identified by their `feature.id` attribute, which can be any number or string. + * + * This method can only be used with sources that have a `feature.id` attribute. The `feature.id` attribute can be defined in three ways: + * - For vector or GeoJSON sources, including an `id` attribute in the original data file. + * - For vector or GeoJSON sources, using the [`promoteId`](https://maplibre.org/maplibre-style-spec/sources/#vector-promoteId) option at the time the source is defined. + * - For GeoJSON sources, using the [`generateId`](https://maplibre.org/maplibre-style-spec/sources/#geojson-generateId) option to auto-assign an `id` based on the feature's index in the source data. If you change feature data using `map.getSource('some id').setData(..)`, you may need to re-apply state taking into account updated `id` values. + * + * _Note: You can use the [`feature-state` expression](https://maplibre.org/maplibre-style-spec/expressions/#feature-state) to access the values in a feature's state object for the purposes of styling._ + * + * @param feature - Feature identifier. Feature objects returned from + * {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers. + * @param state - A set of key-value pairs. The values should be valid JSON types. + * @returns `this` + * + * @example + * ```ts + * // When the mouse moves over the `my-layer` layer, update + * // the feature state for the feature under the mouse + * map.on('mousemove', 'my-layer', function(e) { + * if (e.features.length > 0) { + * map.setFeatureState({ + * source: 'my-source', + * sourceLayer: 'my-source-layer', + * id: e.features[0].id, + * }, { + * hover: true + * }); + * } + * }); + * ``` + * @see [Create a hover effect](https://maplibre.org/maplibre-gl-js/docs/examples/hover-styles/) + */ + setFeatureState(feature: FeatureIdentifier, state: any): this { + this.style.setFeatureState(feature, state); + return this._update(); + } + + /** + * Removes the `state` of a feature, setting it back to the default behavior. + * If only a `target.source` is specified, it will remove the state for all features from that source. + * If `target.id` is also specified, it will remove all keys for that feature's state. + * If `key` is also specified, it removes only that key from that feature's state. + * Features are identified by their `feature.id` attribute, which can be any number or string. + * + * @param target - Identifier of where to remove state. It can be a source, a feature, or a specific key of feature. + * Feature objects returned from {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers. + * @param key - (optional) The key in the feature state to reset. + * @returns `this` + * @example + * Reset the entire state object for all features in the `my-source` source + * ```ts + * map.removeFeatureState({ + * source: 'my-source' + * }); + * ``` + * + * @example + * When the mouse leaves the `my-layer` layer, + * reset the entire state object for the + * feature under the mouse + * ```ts + * map.on('mouseleave', 'my-layer', function(e) { + * map.removeFeatureState({ + * source: 'my-source', + * sourceLayer: 'my-source-layer', + * id: e.features[0].id + * }); + * }); + * ``` + * + * @example + * When the mouse leaves the `my-layer` layer, + * reset only the `hover` key-value pair in the + * state for the feature under the mouse + * ```ts + * map.on('mouseleave', 'my-layer', function(e) { + * map.removeFeatureState({ + * source: 'my-source', + * sourceLayer: 'my-source-layer', + * id: e.features[0].id + * }, 'hover'); + * }); + * ``` + */ + removeFeatureState(target: FeatureIdentifier, key?: string): this { + this.style.removeFeatureState(target, key); + return this._update(); + } + + /** + * Gets the `state` of a feature. + * A feature's `state` is a set of user-defined key-value pairs that are assigned to a feature at runtime. + * Features are identified by their `feature.id` attribute, which can be any number or string. + * + * _Note: To access the values in a feature's state object for the purposes of styling the feature, use the [`feature-state` expression](https://maplibre.org/maplibre-style-spec/expressions/#feature-state)._ + * + * @param feature - Feature identifier. Feature objects returned from + * {@link Map#queryRenderedFeatures} or event handlers can be used as feature identifiers. + * @returns The state of the feature: a set of key-value pairs that was assigned to the feature at runtime. + * + * @example + * When the mouse moves over the `my-layer` layer, + * get the feature state for the feature under the mouse + * ```ts + * map.on('mousemove', 'my-layer', function(e) { + * if (e.features.length > 0) { + * map.getFeatureState({ + * source: 'my-source', + * sourceLayer: 'my-source-layer', + * id: e.features[0].id + * }); + * } + * }); + * ``` + */ + getFeatureState(feature: FeatureIdentifier): any { + return this.style.getFeatureState(feature); + } + + /** + * Returns the map's containing HTML element. + * + * @returns The map's container. + */ + getContainer(): HTMLElement { + return this._container; + } + + /** + * Returns the HTML element containing the map's `` element. + * + * If you want to add non-GL overlays to the map, you should append them to this element. + * + * This is the element to which event bindings for map interactivity (such as panning and zooming) are + * attached. It will receive bubbled events from child elements such as the ``, but not from + * map controls. + * + * @returns The container of the map's ``. + * @see [Create a draggable point](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-point/) + */ + getCanvasContainer(): HTMLElement { + return this._canvasContainer; + } + + /** + * Returns the map's `` element. + * + * @returns The map's `` element. + * @see [Measure distances](https://maplibre.org/maplibre-gl-js/docs/examples/measure/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + * @see [Center the map on a clicked symbol](https://maplibre.org/maplibre-gl-js/docs/examples/center-on-symbol/) + */ + getCanvas(): HTMLCanvasElement { + return this._canvas; + } + + _containerDimensions() { + let width = 0; + let height = 0; + + if (this._container) { + width = this._container.clientWidth || 400; + height = this._container.clientHeight || 300; + } + + return [width, height]; + } + + _setupContainer() { + const container = this._container; + container.classList.add('maplibregl-map'); + + const canvasContainer = this._canvasContainer = DOM.create('div', 'maplibregl-canvas-container', container); + if (this._interactive) { + canvasContainer.classList.add('maplibregl-interactive'); + } + + this._canvas = DOM.create('canvas', 'maplibregl-canvas', canvasContainer); + this._canvas.addEventListener('webglcontextlost', this._contextLost, false); + this._canvas.addEventListener('webglcontextrestored', this._contextRestored, false); + this._canvas.setAttribute('tabindex', '0'); + this._canvas.setAttribute('aria-label', 'Map'); + this._canvas.setAttribute('role', 'region'); + + const dimensions = this._containerDimensions(); + const clampedPixelRatio = this._getClampedPixelRatio(dimensions[0], dimensions[1]); + this._resizeCanvas(dimensions[0], dimensions[1], clampedPixelRatio); + + const controlContainer = this._controlContainer = DOM.create('div', 'maplibregl-control-container', container); + const positions = this._controlPositions = {}; + ['top-left', 'top-right', 'bottom-left', 'bottom-right'].forEach((positionName) => { + positions[positionName] = DOM.create('div', `maplibregl-ctrl-${positionName} `, controlContainer); + }); + + this._container.addEventListener('scroll', this._onMapScroll, false); + } + + _cooperativeGesturesOnWheel = (event: WheelEvent) => { + this._onCooperativeGesture(event, event[this._metaKey], 1); + }; + + _setupCooperativeGestures() { + const container = this._container; + this._cooperativeGesturesScreen = DOM.create('div', 'maplibregl-cooperative-gesture-screen', container); + let desktopMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.windowsHelpText ? this._cooperativeGestures.windowsHelpText : 'Use Ctrl + scroll to zoom the map'; + if (navigator.platform.indexOf('Mac') === 0) { + desktopMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.macHelpText ? this._cooperativeGestures.macHelpText : 'Use ⌘ + scroll to zoom the map'; + } + const mobileMessage = typeof this._cooperativeGestures !== 'boolean' && this._cooperativeGestures.mobileHelpText ? this._cooperativeGestures.mobileHelpText : 'Use two fingers to move the map'; + this._cooperativeGesturesScreen.innerHTML = ` +
${desktopMessage}
+
${mobileMessage}
+ `; + + // Remove cooperative gesture screen from the accessibility tree since screenreaders cannot interact with the map using gestures + this._cooperativeGesturesScreen.setAttribute('aria-hidden', 'true'); + + // Add event to canvas container since gesture container is pointer-events: none + this._canvasContainer.addEventListener('wheel', this._cooperativeGesturesOnWheel, false); + + // Add a cooperative gestures class (enable touch-action: pan-x pan-y;) + this._canvasContainer.classList.add('maplibregl-cooperative-gestures'); + } + + _destroyCooperativeGestures() { + DOM.remove(this._cooperativeGesturesScreen); + this._canvasContainer.removeEventListener('wheel', this._cooperativeGesturesOnWheel, false); + this._canvasContainer.classList.remove('maplibregl-cooperative-gestures'); + } + + _resizeCanvas(width: number, height: number, pixelRatio: number) { + // Request the required canvas size taking the pixelratio into account. + this._canvas.width = Math.floor(pixelRatio * width); + this._canvas.height = Math.floor(pixelRatio * height); + + // Maintain the same canvas size, potentially downscaling it for HiDPI displays + this._canvas.style.width = `${width}px`; + this._canvas.style.height = `${height}px`; + } + + _setupPainter() { + + const attributes = { + alpha: true, + stencil: true, + depth: true, + failIfMajorPerformanceCaveat: this._failIfMajorPerformanceCaveat, + preserveDrawingBuffer: this._preserveDrawingBuffer, + antialias: this._antialias || false + }; + + let webglcontextcreationerrorDetailObject: any = null; + this._canvas.addEventListener('webglcontextcreationerror', (args: WebGLContextEvent) => { + webglcontextcreationerrorDetailObject = {requestedAttributes: attributes}; + if (args) { + webglcontextcreationerrorDetailObject.statusMessage = args.statusMessage; + webglcontextcreationerrorDetailObject.type = args.type; + } + }, {once: true}); + + const gl = + this._canvas.getContext('webgl2', attributes) as WebGL2RenderingContext || + this._canvas.getContext('webgl', attributes) as WebGLRenderingContext; + + if (!gl) { + const msg = 'Failed to initialize WebGL'; + if (webglcontextcreationerrorDetailObject) { + webglcontextcreationerrorDetailObject.message = msg; + throw new Error(JSON.stringify(webglcontextcreationerrorDetailObject)); + } else { + throw new Error(msg); + } + } + + this.painter = new Painter(gl, this.transform); + + webpSupported.testSupport(gl); + } + + _contextLost = (event: any) => { + event.preventDefault(); + if (this._frame) { + this._frame.cancel(); + this._frame = null; + } + this.fire(new Event('webglcontextlost', {originalEvent: event})); + }; + + _contextRestored = (event: any) => { + this._setupPainter(); + this.resize(); + this._update(); + this.fire(new Event('webglcontextrestored', {originalEvent: event})); + }; + + _onMapScroll = (event: any) => { + if (event.target !== this._container) return; + + // Revert any scroll which would move the canvas outside of the view + this._container.scrollTop = 0; + this._container.scrollLeft = 0; + return false; + }; + + _onCooperativeGesture(event: any, metaPress, touches) { + if (!metaPress && touches < 2) { + // Alert user how to scroll/pan + this._cooperativeGesturesScreen.classList.add('maplibregl-show'); + setTimeout(() => { + this._cooperativeGesturesScreen.classList.remove('maplibregl-show'); + }, 100); + } + return false; + } + + /** + * Returns a Boolean indicating whether the map is fully loaded. + * + * Returns `false` if the style is not yet fully loaded, + * or if there has been a change to the sources or style that + * has not yet fully loaded. + * + * @returns A Boolean indicating whether the map is fully loaded. + */ + loaded(): boolean { + return !this._styleDirty && !this._sourcesDirty && !!this.style && this.style.loaded(); + } + + /** + * @internal + * Update this map's style and sources, and re-render the map. + * + * @param updateStyle - mark the map's style for reprocessing as + * well as its sources + * @returns `this` + */ + _update(updateStyle?: boolean) { + if (!this.style || !this.style._loaded) return this; + + this._styleDirty = this._styleDirty || updateStyle; + this._sourcesDirty = true; + this.triggerRepaint(); + + return this; + } + + /** + * @internal + * Request that the given callback be executed during the next render + * frame. Schedule a render frame if one is not already scheduled. + * + * @returns An id that can be used to cancel the callback + */ + _requestRenderFrame(callback: () => void): TaskID { + this._update(); + return this._renderTaskQueue.add(callback); + } + + _cancelRenderFrame(id: TaskID) { + this._renderTaskQueue.remove(id); + } + + /** + * @internal + * Call when a (re-)render of the map is required: + * - The style has changed (`setPaintProperty()`, etc.) + * - Source data has changed (e.g. tiles have finished loading) + * - The map has is moving (or just finished moving) + * - A transition is in progress + * + * @param paintStartTimeStamp - The time when the animation frame began executing. + * + * @returns `this` + */ + _render(paintStartTimeStamp: number) { + const fadeDuration = this._idleTriggered ? this._fadeDuration : 0; + + // A custom layer may have used the context asynchronously. Mark the state as dirty. + this.painter.context.setDirty(); + this.painter.setBaseState(); + + this._renderTaskQueue.run(paintStartTimeStamp); + // A task queue callback may have fired a user event which may have removed the map + if (this._removed) return; + + let crossFading = false; + + // If the style has changed, the map is being zoomed, or a transition or fade is in progress: + // - Apply style changes (in a batch) + // - Recalculate paint properties. + if (this.style && this._styleDirty) { + this._styleDirty = false; + + const zoom = this.transform.zoom; + const now = browser.now(); + this.style.zoomHistory.update(zoom, now); + + const parameters = new EvaluationParameters(zoom, { + now, + fadeDuration, + zoomHistory: this.style.zoomHistory, + transition: this.style.getTransition() + }); + + const factor = parameters.crossFadingFactor(); + if (factor !== 1 || factor !== this._crossFadingFactor) { + crossFading = true; + this._crossFadingFactor = factor; + } + + this.style.update(parameters); + } + + // If we are in _render for any reason other than an in-progress paint + // transition, update source caches to check for and load any tiles we + // need for the current transform + if (this.style && this._sourcesDirty) { + this._sourcesDirty = false; + this.style._updateSources(this.transform); + } + + // update terrain stuff + if (this.terrain) { + this.terrain.sourceCache.update(this.transform, this.terrain); + this.transform._minEleveationForCurrentTile = this.terrain.getMinTileElevationForLngLatZoom(this.transform.center, this.transform.tileZoom); + if (!this._elevationFreeze) { + this.transform.elevation = this.terrain.getElevationForLngLatZoom(this.transform.center, this.transform.tileZoom); + } + } else { + this.transform._minEleveationForCurrentTile = 0; + this.transform.elevation = 0; + } + + this._placementDirty = this.style && this.style._updatePlacement(this.painter.transform, this.showCollisionBoxes, fadeDuration, this._crossSourceCollisions); + + // Actually draw + this.painter.render(this.style, { + showTileBoundaries: this.showTileBoundaries, + showOverdrawInspector: this._showOverdrawInspector, + rotating: this.isRotating(), + zooming: this.isZooming(), + moving: this.isMoving(), + fadeDuration, + showPadding: this.showPadding, + }); + + this.fire(new Event('render')); + + if (this.loaded() && !this._loaded) { + this._loaded = true; + PerformanceUtils.mark(PerformanceMarkers.load); + this.fire(new Event('load')); + } + + if (this.style && (this.style.hasTransitions() || crossFading)) { + this._styleDirty = true; + } + + if (this.style && !this._placementDirty) { + // Since no fade operations are in progress, we can release + // all tiles held for fading. If we didn't do this, the tiles + // would just sit in the SourceCaches until the next render + this.style._releaseSymbolFadeTiles(); + } + + // Schedule another render frame if it's needed. + // + // Even though `_styleDirty` and `_sourcesDirty` are reset in this + // method, synchronous events fired during Style#update or + // Style#_updateSources could have caused them to be set again. + const somethingDirty = this._sourcesDirty || this._styleDirty || this._placementDirty; + if (somethingDirty || this._repaint) { + this.triggerRepaint(); + } else if (!this.isMoving() && this.loaded()) { + this.fire(new Event('idle')); + } + + if (this._loaded && !this._fullyLoaded && !somethingDirty) { + this._fullyLoaded = true; + PerformanceUtils.mark(PerformanceMarkers.fullLoad); + } + + return this; + } + + /** + * Force a synchronous redraw of the map. + * @returns `this` + * @example + * ```ts + * map.redraw(); + * ``` + */ + redraw(): this { + if (this.style) { + // cancel the scheduled update + if (this._frame) { + this._frame.cancel(); + this._frame = null; + } + this._render(0); + } + return this; + } + + /** + * Clean up and release all internal resources associated with this map. + * + * This includes DOM elements, event bindings, web workers, and WebGL resources. + * + * Use this method when you are done using the map and wish to ensure that it no + * longer consumes browser resources. Afterwards, you must not call any other + * methods on the map. + */ + remove() { + if (this._hash) this._hash.remove(); + + for (const control of this._controls) control.onRemove(this); + this._controls = []; + + if (this._frame) { + this._frame.cancel(); + this._frame = null; + } + this._renderTaskQueue.clear(); + this.painter.destroy(); + this.handlers.destroy(); + delete this.handlers; + this.setStyle(null); + if (typeof window !== 'undefined') { + removeEventListener('online', this._onWindowOnline, false); + } + + ImageRequest.removeThrottleControl(this._imageQueueHandle); + + this._resizeObserver?.disconnect(); + const extension = this.painter.context.gl.getExtension('WEBGL_lose_context'); + if (extension) extension.loseContext(); + this._canvas.removeEventListener('webglcontextrestored', this._contextRestored, false); + this._canvas.removeEventListener('webglcontextlost', this._contextLost, false); + DOM.remove(this._canvasContainer); + DOM.remove(this._controlContainer); + if (this._cooperativeGestures) { + this._destroyCooperativeGestures(); + } + this._container.classList.remove('maplibregl-map'); + + PerformanceUtils.clearMetrics(); + + this._removed = true; + this.fire(new Event('remove')); + } + + /** + * Trigger the rendering of a single frame. Use this method with custom layers to + * repaint the map when the layer changes. Calling this multiple times before the + * next frame is rendered will still result in only a single frame being rendered. + * @example + * ```ts + * map.triggerRepaint(); + * ``` + * @see [Add a 3D model](https://maplibre.org/maplibre-gl-js/docs/examples/add-3d-model/) + * @see [Add an animated icon to the map](https://maplibre.org/maplibre-gl-js/docs/examples/add-image-animated/) + */ + triggerRepaint() { + if (this.style && !this._frame) { + this._frame = browser.frame((paintStartTimeStamp: number) => { + PerformanceUtils.frame(paintStartTimeStamp); + this._frame = null; + this._render(paintStartTimeStamp); + }); + } + } + + _onWindowOnline = () => { + this._update(); + }; + + /** + * Gets and sets a Boolean indicating whether the map will render an outline + * around each tile and the tile ID. These tile boundaries are useful for + * debugging. + * + * The uncompressed file size of the first vector source is drawn in the top left + * corner of each tile, next to the tile ID. + * + * @example + * ```ts + * map.showTileBoundaries = true; + * ``` + */ + get showTileBoundaries(): boolean { return !!this._showTileBoundaries; } + set showTileBoundaries(value: boolean) { + if (this._showTileBoundaries === value) return; + this._showTileBoundaries = value; + this._update(); + } + + /** + * Gets and sets a Boolean indicating whether the map will visualize + * the padding offsets. + */ + get showPadding(): boolean { return !!this._showPadding; } + set showPadding(value: boolean) { + if (this._showPadding === value) return; + this._showPadding = value; + this._update(); + } + + /** + * Gets and sets a Boolean indicating whether the map will render boxes + * around all symbols in the data source, revealing which symbols + * were rendered or which were hidden due to collisions. + * This information is useful for debugging. + */ + get showCollisionBoxes(): boolean { return !!this._showCollisionBoxes; } + set showCollisionBoxes(value: boolean) { + if (this._showCollisionBoxes === value) return; + this._showCollisionBoxes = value; + if (value) { + // When we turn collision boxes on we have to generate them for existing tiles + // When we turn them off, there's no cost to leaving existing boxes in place + this.style._generateCollisionBoxes(); + } else { + // Otherwise, call an update to remove collision boxes + this._update(); + } + } + + /** + * Gets and sets a Boolean indicating whether the map should color-code + * each fragment to show how many times it has been shaded. + * White fragments have been shaded 8 or more times. + * Black fragments have been shaded 0 times. + * This information is useful for debugging. + */ + get showOverdrawInspector(): boolean { return !!this._showOverdrawInspector; } + set showOverdrawInspector(value: boolean) { + if (this._showOverdrawInspector === value) return; + this._showOverdrawInspector = value; + this._update(); + } + + /** + * Gets and sets a Boolean indicating whether the map will + * continuously repaint. This information is useful for analyzing performance. + */ + get repaint(): boolean { return !!this._repaint; } + set repaint(value: boolean) { + if (this._repaint !== value) { + this._repaint = value; + this.triggerRepaint(); + } + } + // show vertices + get vertices(): boolean { return !!this._vertices; } + set vertices(value: boolean) { this._vertices = value; this._update(); } + + /** + * Returns the package version of the library + * @returns Package version of the library + */ + get version(): string { + return version; + } + + /** + * Returns the elevation for the point where the camera is looking. + * This value corresponds to: + * "meters above sea level" * "exaggeration" + * @returns The elevation. + */ + getCameraTargetElevation(): number { + return this.transform.elevation; + } +} diff --git a/web/libraries/maplibre-gl/src/ui/map/isMoving.test.ts b/web/libraries/maplibre-gl/src/ui/map/isMoving.test.ts new file mode 100644 index 00000000..3acc6828 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/map/isMoving.test.ts @@ -0,0 +1,164 @@ +import {browser} from '../../util/browser'; +import {Map} from '../map'; +import {DOM} from '../../util/dom'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {beforeMapTest} from '../../util/test/util'; + +let map; + +function createMap() { + return new Map({style: '', container: DOM.create('div', '', window.document.body)}); +} + +beforeEach(() => { + beforeMapTest(); + map = createMap(); +}); + +afterEach(() => { + map.remove(); +}); + +describe('Map#isMoving', () => { + // MouseEvent.buttons + const buttons = 1; + + test('returns false by default', () => { + expect(map.isMoving()).toBe(false); + }); + + test('returns true during a camera zoom animation', done => { + map.on('zoomstart', () => { + expect(map.isMoving()).toBe(true); + }); + + map.on('zoomend', () => { + expect(map.isMoving()).toBe(false); + done(); + }); + + map.zoomTo(5, {duration: 0}); + }); + + test('returns true when drag panning', done => { + map.on('movestart', () => { + expect(map.isMoving()).toBe(true); + }); + map.on('dragstart', () => { + expect(map.isMoving()).toBe(true); + }); + + map.on('dragend', () => { + expect(map.isMoving()).toBe(false); + }); + map.on('moveend', () => { + expect(map.isMoving()).toBe(false); + done(); + }); + + simulate.mousedown(map.getCanvas()); + map._renderTaskQueue.run(); + + simulate.mousemove(map.getCanvas(), {buttons, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + + simulate.mouseup(map.getCanvas()); + map._renderTaskQueue.run(); + }); + + test('returns true when drag rotating', done => { + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockImplementation(() => { return 0; }); + + map.on('movestart', () => { + expect(map.isMoving()).toBe(true); + }); + + map.on('rotatestart', () => { + expect(map.isMoving()).toBe(true); + }); + + map.on('rotateend', () => { + expect(map.isMoving()).toBe(false); + }); + + map.on('moveend', () => { + expect(map.isMoving()).toBe(false); + done(); + }); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + map._renderTaskQueue.run(); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + }); + + test('returns true when scroll zooming', done => { + map.on('zoomstart', () => { + expect(map.isMoving()).toBe(true); + }); + + map.on('zoomend', () => { + expect(map.isMoving()).toBe(false); + done(); + }); + + let now = 0; + jest.spyOn(browser, 'now').mockImplementation(() => { return now; }); + + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta}); + map._renderTaskQueue.run(); + + now += 400; + setTimeout(() => { + map._renderTaskQueue.run(); + }, 400); + }); + + test('returns true when drag panning and scroll zooming interleave', done => { + map.on('dragstart', () => { + expect(map.isMoving()).toBe(true); + }); + + map.on('zoomstart', () => { + expect(map.isMoving()).toBe(true); + }); + + map.on('zoomend', () => { + expect(map.isMoving()).toBe(true); + simulate.mouseup(map.getCanvas()); + setTimeout(() => { + map._renderTaskQueue.run(); + done(); + }); + }); + + map.on('dragend', () => { + expect(map.isMoving()).toBe(false); + }); + + // The following should trigger the above events, where a zoomstart/zoomend + // pair is nested within a dragstart/dragend pair. + + simulate.mousedown(map.getCanvas()); + map._renderTaskQueue.run(); + + simulate.mousemove(map.getCanvas(), {buttons, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + + let now = 0; + jest.spyOn(browser, 'now').mockImplementation(() => { return now; }); + + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta}); + map._renderTaskQueue.run(); + + now += 400; + setTimeout(() => { + map._renderTaskQueue.run(); + }, 400); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/map/isRotating.test.ts b/web/libraries/maplibre-gl/src/ui/map/isRotating.test.ts new file mode 100644 index 00000000..71fe8676 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/map/isRotating.test.ts @@ -0,0 +1,62 @@ +import {Map} from '../../ui/map'; +import {DOM} from '../../util/dom'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {browser} from '../../util/browser'; +import {beforeMapTest} from '../../util/test/util'; + +let map; + +function createMap() { + return new Map({style: '', container: DOM.create('div', '', window.document.body)}); +} + +beforeEach(() => { + beforeMapTest(); + map = createMap(); +}); + +afterEach(() => { + map.remove(); +}); + +describe('Map#isRotating', () => { + test('returns false by default', () => { + expect(map.isRotating()).toBe(false); + }); + + test('returns true during a camera rotate animation', done => { + map.on('rotatestart', () => { + expect(map.isRotating()).toBe(true); + }); + + map.on('rotateend', () => { + expect(map.isRotating()).toBe(false); + done(); + }); + + map.rotateTo(5, {duration: 0}); + }); + + test('returns true when drag rotating', done => { + // Prevent inertial rotation. + jest.spyOn(browser, 'now').mockImplementation(() => { return 0; }); + + map.on('rotatestart', () => { + expect(map.isRotating()).toBe(true); + }); + + map.on('rotateend', () => { + expect(map.isRotating()).toBe(false); + done(); + }); + + simulate.mousedown(map.getCanvas(), {buttons: 2, button: 2}); + map._renderTaskQueue.run(); + + simulate.mousemove(map.getCanvas(), {buttons: 2, clientX: 10, clientY: 10}); + map._renderTaskQueue.run(); + + simulate.mouseup(map.getCanvas(), {buttons: 0, button: 2}); + map._renderTaskQueue.run(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/map/isZooming.test.ts b/web/libraries/maplibre-gl/src/ui/map/isZooming.test.ts new file mode 100644 index 00000000..9321c030 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/map/isZooming.test.ts @@ -0,0 +1,87 @@ +import {browser} from '../../util/browser'; +import {Map} from '../../ui/map'; +import {DOM} from '../../util/dom'; +import simulate from '../../../test/unit/lib/simulate_interaction'; +import {beforeMapTest} from '../../util/test/util'; + +function createMap() { + return new Map({style: '', container: DOM.create('div', '', window.document.body)}); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('Map#isZooming', () => { + + test('returns false by default', done => { + const map = createMap(); + expect(map.isZooming()).toBe(false); + map.remove(); + done(); + }); + + test('returns true during a camera zoom animation', done => { + const map = createMap(); + + map.on('zoomstart', () => { + expect(map.isZooming()).toBe(true); + }); + + map.on('zoomend', () => { + expect(map.isZooming()).toBe(false); + map.remove(); + done(); + }); + + map.zoomTo(5, {duration: 0}); + }); + + test('returns true when scroll zooming', done => { + const map = createMap(); + + map.on('zoomstart', () => { + expect(map.isZooming()).toBe(true); + }); + + map.on('zoomend', () => { + expect(map.isZooming()).toBe(false); + map.remove(); + done(); + }); + + let now = 0; + jest.spyOn(browser, 'now').mockImplementation(() => { return now; }); + + simulate.wheel(map.getCanvas(), {type: 'wheel', deltaY: -simulate.magicWheelZoomDelta}); + map._renderTaskQueue.run(); + + now += 400; + setTimeout(() => { + map._renderTaskQueue.run(); + }, 400); + }); + + test('returns true when double-click zooming', done => { + const map = createMap(); + + map.on('zoomstart', () => { + expect(map.isZooming()).toBe(true); + }); + + map.on('zoomend', () => { + expect(map.isZooming()).toBe(false); + map.remove(); + done(); + }); + + let now = 0; + jest.spyOn(browser, 'now').mockImplementation(() => { return now; }); + + simulate.dblclick(map.getCanvas()); + map._renderTaskQueue.run(); + + now += 500; + map._renderTaskQueue.run(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/map/requestRenderFrame.test.ts b/web/libraries/maplibre-gl/src/ui/map/requestRenderFrame.test.ts new file mode 100644 index 00000000..e9c514f7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/map/requestRenderFrame.test.ts @@ -0,0 +1,51 @@ +import {createMap, beforeMapTest} from '../../util/test/util'; + +beforeEach(() => { + beforeMapTest(); +}); + +describe('requestRenderFrame', () => { + + test('Map#_requestRenderFrame schedules a new render frame if necessary', (done) => { + const map = createMap(); + const spy = jest.spyOn(map, 'triggerRepaint'); + map._requestRenderFrame(() => {}); + expect(spy).toHaveBeenCalledTimes(0); + + // wait for style to be loaded + map.once('data', () => { + spy.mockReset(); + map._requestRenderFrame(() => {}); + expect(spy).toHaveBeenCalledTimes(1); + map.remove(); + done(); + }); + }); + + test('Map#_requestRenderFrame should not schedule a render frame before style load', () => { + const map = createMap(); + const spy = jest.spyOn(map, 'triggerRepaint'); + map._requestRenderFrame(() => {}); + expect(spy).toHaveBeenCalledTimes(0); + map.remove(); + }); + + test('Map#_requestRenderFrame queues a task for the next render frame', async () => { + const map = createMap(); + const cb = jest.fn(); + map._requestRenderFrame(cb); + await map.once('render'); + expect(cb).toHaveBeenCalledTimes(1); + map.remove(); + }); + + test('Map#_cancelRenderFrame cancels a queued task', async () => { + const map = createMap(); + const cb = jest.fn(); + const id = map._requestRenderFrame(cb); + map._cancelRenderFrame(id); + await map.once('render'); + expect(cb).toHaveBeenCalledTimes(0); + map.remove(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/map_events.test.ts b/web/libraries/maplibre-gl/src/ui/map_events.test.ts new file mode 100644 index 00000000..05fea76b --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/map_events.test.ts @@ -0,0 +1,708 @@ +import simulate, {window} from '../../test/unit/lib/simulate_interaction'; +import {StyleLayer} from '../style/style_layer'; +import {createMap, beforeMapTest} from '../util/test/util'; +import {MapGeoJSONFeature} from '../util/vectortile_to_geojson'; +import {MapLayerEventType, MapLibreEvent} from './events'; + +type IsAny = 0 extends T & 1 ? T : never; +type NotAny = T extends IsAny ? never : T; +function assertNotAny(_x: NotAny) { } + +beforeEach(() => { + beforeMapTest(); +}); + +describe('map events', () => { + + test('Map#on adds a non-delegated event listener', () => { + const map = createMap(); + const spy = jest.fn(function (e) { + expect(this).toBe(map); + expect(e.type).toBe('click'); + }); + + map.on('click', spy); + simulate.click(map.getCanvas()); + + expect(spy).toHaveBeenCalledTimes(1); + }); + + test('Map#off removes a non-delegated event listener', () => { + const map = createMap(); + const spy = jest.fn(); + + map.on('click', spy); + map.off('click', spy); + simulate.click(map.getCanvas()); + + expect(spy).not.toHaveBeenCalled(); + + }); + + test('Map#on adds a listener for an event on a given layer', () => { + const map = createMap(); + const features = [{} as MapGeoJSONFeature]; + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockImplementation((_point, options) => { + expect(options).toEqual({layers: ['layer']}); + return features; + }); + + const spy = jest.fn(function (e) { + expect(this).toBe(map); + expect(e.type).toBe('click'); + expect(e.features).toBe(features); + }); + + map.on('click', 'layer', spy); + simulate.click(map.getCanvas()); + + expect(spy).toHaveBeenCalledTimes(1); + }); + + test('Map#on adds a listener not triggered for events not matching any features', () => { + const map = createMap(); + const features = []; + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockImplementation((point, options) => { + expect(options).toEqual({layers: ['layer']}); + return features; + }); + + const spy = jest.fn(); + + map.on('click', 'layer', spy); + simulate.click(map.getCanvas()); + + expect(spy).not.toHaveBeenCalled(); + + }); + + test('Map#on adds a listener not triggered when the specified layer does not exiist', () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue(null as unknown as StyleLayer); + + const spy = jest.fn(); + + map.on('click', 'layer', spy); + simulate.click(map.getCanvas()); + + expect(spy).not.toHaveBeenCalled(); + + }); + + test('Map#on distinguishes distinct event types', () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + const spyDown = jest.fn((e) => { + expect(e.type).toBe('mousedown'); + }); + + const spyUp = jest.fn((e) => { + expect(e.type).toBe('mouseup'); + }); + + map.on('mousedown', 'layer', spyDown); + map.on('mouseup', 'layer', spyUp); + simulate.click(map.getCanvas()); + + expect(spyDown).toHaveBeenCalledTimes(1); + expect(spyUp).toHaveBeenCalledTimes(1); + }); + + test('Map#on distinguishes distinct layers', () => { + const map = createMap(); + const featuresA = [{} as MapGeoJSONFeature]; + const featuresB = [{} as MapGeoJSONFeature]; + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockImplementation((_point, options) => { + return (options as any).layers[0] === 'A' ? featuresA : featuresB; + }); + + const spyA = jest.fn((e) => { + expect(e.features).toBe(featuresA); + }); + + const spyB = jest.fn((e) => { + expect(e.features).toBe(featuresB); + }); + + map.on('click', 'A', spyA); + map.on('click', 'B', spyB); + simulate.click(map.getCanvas()); + + expect(spyA).toHaveBeenCalledTimes(1); + expect(spyB).toHaveBeenCalledTimes(1); + }); + + test('Map#on distinguishes distinct listeners', () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + const spyA = jest.fn(); + const spyB = jest.fn(); + + map.on('click', 'layer', spyA); + map.on('click', 'layer', spyB); + simulate.click(map.getCanvas()); + + expect(spyA).toHaveBeenCalledTimes(1); + expect(spyB).toHaveBeenCalledTimes(1); + }); + + test('Map#on calls an event listener with no type arguments, defaulting to \'unknown\' originalEvent type', () => { + const map = createMap(); + + const handler = { + onMove: function onMove(_event: MapLibreEvent) {} + }; + + jest.spyOn(handler, 'onMove'); + + map.on('move', (event) => handler.onMove(event)); + map.jumpTo({center: {lng: 10, lat: 10}}); + + expect(handler.onMove).toHaveBeenCalledTimes(1); + }); + + test('Map#on allows a listener to infer the event type ', () => { + const map = createMap(); + + const spy = jest.fn(); + map.on('mousemove', (event) => { + assertNotAny(event); + const {lng, lat} = event.lngLat; + spy({lng, lat}); + }); + + simulate.mousemove(map.getCanvas()); + expect(spy).toHaveBeenCalledTimes(1); + }); + + test('Map#off removes a delegated event listener', () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + const spy = jest.fn(); + + map.on('click', 'layer', spy); + map.off('click', 'layer', spy); + simulate.click(map.getCanvas()); + + expect(spy).not.toHaveBeenCalled(); + + }); + + test('Map#off distinguishes distinct event types', () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + const spy = jest.fn((e) => { + expect(e.type).toBe('mousedown'); + }); + + map.on('mousedown', 'layer', spy); + map.on('mouseup', 'layer', spy); + map.off('mouseup', 'layer', spy); + simulate.click(map.getCanvas()); + + expect(spy).toHaveBeenCalledTimes(1); + }); + + test('Map#off distinguishes distinct layers', () => { + const map = createMap(); + const featuresA = [{} as MapGeoJSONFeature]; + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockImplementation((point, options) => { + expect(options).toEqual({layers: ['A']}); + return featuresA; + }); + + const spy = jest.fn((e) => { + expect(e.features).toBe(featuresA); + }); + + map.on('click', 'A', spy); + map.on('click', 'B', spy); + map.off('click', 'B', spy); + simulate.click(map.getCanvas()); + + expect(spy).toHaveBeenCalledTimes(1); + }); + + test('Map#off distinguishes distinct listeners', () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + const spyA = jest.fn(); + const spyB = jest.fn(); + + map.on('click', 'layer', spyA); + map.on('click', 'layer', spyB); + map.off('click', 'layer', spyB); + simulate.click(map.getCanvas()); + + expect(spyA).toHaveBeenCalledTimes(1); + expect(spyB).not.toHaveBeenCalled(); + }); + + test('Map#off calls an event listener with no type arguments, defaulting to \'unknown\' originalEvent type', () => { + const map = createMap(); + + const handler = { + onMove: function onMove(_event: MapLibreEvent) {} + }; + + jest.spyOn(handler, 'onMove'); + + map.off('move', (event) => handler.onMove(event)); + map.jumpTo({center: {lng: 10, lat: 10}}); + + expect(handler.onMove).toHaveBeenCalledTimes(0); + }); + + test('Map#off allows a listener to infer the event type ', () => { + const map = createMap(); + + const spy = jest.fn(); + map.off('mousemove', (event) => { + assertNotAny(event); + const {lng, lat} = event.lngLat; + spy({lng, lat}); + }); + + simulate.mousemove(map.getCanvas()); + expect(spy).toHaveBeenCalledTimes(0); + }); + + test('Map#once calls an event listener with no type arguments, defaulting to \'unknown\' originalEvent type', () => { + const map = createMap(); + + const handler = { + onMoveOnce: function onMoveOnce(_event: MapLibreEvent) {} + }; + + jest.spyOn(handler, 'onMoveOnce'); + + map.once('move', (event) => handler.onMoveOnce(event)); + map.jumpTo({center: {lng: 10, lat: 10}}); + + expect(handler.onMoveOnce).toHaveBeenCalledTimes(1); + }); + + test('Map#once allows a listener to infer the event type ', () => { + const map = createMap(); + + const spy = jest.fn(); + map.once('mousemove', (event) => { + assertNotAny(event); + const {lng, lat} = event.lngLat; + spy({lng, lat}); + }); + + simulate.mousemove(map.getCanvas()); + expect(spy).toHaveBeenCalledTimes(1); + }); + + (['mouseenter', 'mouseover'] as (keyof MapLayerEventType)[]).forEach((event) => { + test(`Map#on ${event} does not fire if the specified layer does not exist`, () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue(null as unknown as StyleLayer); + + const spy = jest.fn(); + + map.on(event, 'layer', spy); + simulate.mousemove(map.getCanvas()); + simulate.mousemove(map.getCanvas()); + + expect(spy).not.toHaveBeenCalled(); + + }); + + test(`Map#on ${event} fires when entering the specified layer`, () => { + const map = createMap(); + const features = [{} as MapGeoJSONFeature]; + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockImplementation((_point, options) => { + expect(options).toEqual({layers: ['layer']}); + return features; + }); + + const spy = jest.fn(function (e) { + expect(this).toBe(map); + expect(e.type).toBe(event); + expect(e.target).toBe(map); + expect(e.features).toBe(features); + }); + + map.on(event, 'layer', spy); + simulate.mousemove(map.getCanvas()); + + expect(spy).toHaveBeenCalledTimes(1); + }); + + test(`Map#on ${event} does not fire on mousemove within the specified layer`, () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + const spy = jest.fn(); + + map.on(event, 'layer', spy); + simulate.mousemove(map.getCanvas()); + simulate.mousemove(map.getCanvas()); + + expect(spy).toHaveBeenCalledTimes(1); + }); + + test(`Map#on ${event} fires when reentering the specified layer`, () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures') + .mockReturnValueOnce([{} as MapGeoJSONFeature]) + .mockReturnValueOnce([]) + .mockReturnValueOnce([{} as MapGeoJSONFeature]); + + const spy = jest.fn(); + + map.on(event, 'layer', spy); + simulate.mousemove(map.getCanvas()); + simulate.mousemove(map.getCanvas()); + simulate.mousemove(map.getCanvas()); + + expect(spy).toHaveBeenCalledTimes(2); + }); + + test(`Map#on ${event} fires when reentering the specified layer after leaving the canvas`, () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + const spy = jest.fn(); + + map.on(event, 'layer', spy); + simulate.mousemove(map.getCanvas()); + simulate.mouseout(map.getCanvas()); + simulate.mousemove(map.getCanvas()); + + expect(spy).toHaveBeenCalledTimes(2); + }); + + test(`Map#on ${event} distinguishes distinct layers`, () => { + const map = createMap(); + const featuresA = [{} as MapGeoJSONFeature]; + const featuresB = [{} as MapGeoJSONFeature]; + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockImplementation((_point, options) => { + return (options as any).layers[0] === 'A' ? featuresA : featuresB; + }); + + const spyA = jest.fn((e) => { + expect(e.features).toBe(featuresA); + }); + + const spyB = jest.fn((e) => { + expect(e.features).toBe(featuresB); + }); + + map.on(event, 'A', spyA); + map.on(event, 'B', spyB); + + simulate.mousemove(map.getCanvas()); + simulate.mousemove(map.getCanvas()); + + expect(spyA).toHaveBeenCalledTimes(1); + expect(spyB).toHaveBeenCalledTimes(1); + }); + + test(`Map#on ${event} distinguishes distinct listeners`, () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + const spyA = jest.fn(); + const spyB = jest.fn(); + + map.on(event, 'layer', spyA); + map.on(event, 'layer', spyB); + simulate.mousemove(map.getCanvas()); + + expect(spyA).toHaveBeenCalledTimes(1); + expect(spyB).toHaveBeenCalledTimes(1); + }); + + test(`Map#off ${event} removes a delegated event listener`, () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + const spy = jest.fn(); + + map.on(event, 'layer', spy); + map.off(event, 'layer', spy); + simulate.mousemove(map.getCanvas()); + + expect(spy).not.toHaveBeenCalled(); + + }); + + test(`Map#off ${event} distinguishes distinct layers`, () => { + const map = createMap(); + const featuresA = [{} as MapGeoJSONFeature]; + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockImplementation((_point, options) => { + expect(options).toEqual({layers: ['A']}); + return featuresA; + }); + + const spy = jest.fn((e) => { + expect(e.features).toBe(featuresA); + }); + + map.on(event, 'A', spy); + map.on(event, 'B', spy); + map.off(event, 'B', spy); + simulate.mousemove(map.getCanvas()); + + expect(spy).toHaveBeenCalledTimes(1); + }); + + test(`Map#off ${event} distinguishes distinct listeners`, () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + const spyA = jest.fn(); + const spyB = jest.fn(); + + map.on(event, 'layer', spyA); + map.on(event, 'layer', spyB); + map.off(event, 'layer', spyB); + simulate.mousemove(map.getCanvas()); + + expect(spyA).toHaveBeenCalledTimes(1); + expect(spyB).not.toHaveBeenCalled(); + }); + }); + + (['mouseleave', 'mouseout'] as (keyof MapLayerEventType)[]).forEach((event) => { + test(`Map#on ${event} does not fire if the specified layer does not exiist`, () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue(null as unknown as StyleLayer); + + const spy = jest.fn(); + + map.on(event, 'layer', spy); + simulate.mousemove(map.getCanvas()); + simulate.mousemove(map.getCanvas()); + + expect(spy).not.toHaveBeenCalled(); + + }); + + test(`Map#on ${event} does not fire on mousemove when entering or within the specified layer`, () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + const spy = jest.fn(); + + map.on(event, 'layer', spy); + simulate.mousemove(map.getCanvas()); + simulate.mousemove(map.getCanvas()); + + expect(spy).not.toHaveBeenCalled(); + + }); + + test(`Map#on ${event} fires when exiting the specified layer`, () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures') + .mockReturnValueOnce([{} as MapGeoJSONFeature]) + .mockReturnValueOnce([]); + + const spy = jest.fn(function (e) { + expect(this).toBe(map); + expect(e.type).toBe(event); + expect(e.features).toBeUndefined(); + }); + + map.on(event, 'layer', spy); + simulate.mousemove(map.getCanvas()); + simulate.mousemove(map.getCanvas()); + + expect(spy).toHaveBeenCalledTimes(1); + }); + + test(`Map#on ${event} fires when exiting the canvas`, () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures').mockReturnValue([{} as MapGeoJSONFeature]); + + const spy = jest.fn(function (e) { + expect(this).toBe(map); + expect(e.type).toBe(event); + expect(e.features).toBeUndefined(); + }); + + map.on(event, 'layer', spy); + simulate.mousemove(map.getCanvas()); + simulate.mouseout(map.getCanvas()); + + expect(spy).toHaveBeenCalledTimes(1); + }); + + test(`Map#off ${event} removes a delegated event listener`, () => { + const map = createMap(); + + jest.spyOn(map, 'getLayer').mockReturnValue({} as StyleLayer); + jest.spyOn(map, 'queryRenderedFeatures') + .mockReturnValueOnce([{} as MapGeoJSONFeature]) + .mockReturnValueOnce([]); + + const spy = jest.fn(); + + map.on(event, 'layer', spy); + map.off(event, 'layer', spy); + simulate.mousemove(map.getCanvas()); + simulate.mousemove(map.getCanvas()); + simulate.mouseout(map.getCanvas()); + + expect(spy).not.toHaveBeenCalled(); + + }); + }); + + test('Map#on mousedown can have default behavior prevented and still fire subsequent click event', () => { + const map = createMap(); + + map.on('mousedown', e => e.preventDefault()); + + const click = jest.fn(); + map.on('click', click); + + simulate.click(map.getCanvas()); + expect(click).toHaveBeenCalled(); + + map.remove(); + }); + + test('Map#on mousedown doesn\'t fire subsequent click event if mousepos changes', () => { + const map = createMap(); + + map.on('mousedown', e => e.preventDefault()); + + const click = jest.fn(); + map.on('click', click); + const canvas = map.getCanvas(); + + simulate.drag(canvas, {}, {clientX: 100, clientY: 100}); + expect(click).not.toHaveBeenCalled(); + + map.remove(); + }); + + test('Map#on mousedown fires subsequent click event if mouse position changes less than click tolerance', () => { + const map = createMap({clickTolerance: 4}); + + map.on('mousedown', e => e.preventDefault()); + + const click = jest.fn(); + map.on('click', click); + const canvas = map.getCanvas(); + + simulate.drag(canvas, {clientX: 100, clientY: 100}, {clientX: 100, clientY: 103}); + expect(click).toHaveBeenCalled(); + + map.remove(); + }); + + test('Map#on mousedown does not fire subsequent click event if mouse position changes more than click tolerance', () => { + const map = createMap({clickTolerance: 4}); + + map.on('mousedown', e => e.preventDefault()); + + const click = jest.fn(); + map.on('click', click); + const canvas = map.getCanvas(); + + simulate.drag(canvas, {clientX: 100, clientY: 100}, {clientX: 100, clientY: 104}); + expect(click).not.toHaveBeenCalled(); + + map.remove(); + }); + + test('Map#on click fires subsequent click event if there is no corresponding mousedown/mouseup event', () => { + const map = createMap({clickTolerance: 4}); + + const click = jest.fn(); + map.on('click', click); + const canvas = map.getCanvas(); + + const MouseEvent = window(canvas).MouseEvent; + const event = new MouseEvent('click', {bubbles: true, clientX: 100, clientY: 100}); + canvas.dispatchEvent(event); + expect(click).toHaveBeenCalled(); + + map.remove(); + }); + + test('Map#isMoving() returns false in mousedown/mouseup/click with no movement', () => { + const map = createMap({interactive: true, clickTolerance: 4}); + let mousedown, mouseup, click; + map.on('mousedown', () => { mousedown = map.isMoving(); }); + map.on('mouseup', () => { mouseup = map.isMoving(); }); + map.on('click', () => { click = map.isMoving(); }); + + const canvas = map.getCanvas(); + const MouseEvent = window(canvas).MouseEvent; + + canvas.dispatchEvent(new MouseEvent('mousedown', {bubbles: true, clientX: 100, clientY: 100})); + expect(mousedown).toBe(false); + map._renderTaskQueue.run(); + expect(mousedown).toBe(false); + + canvas.dispatchEvent(new MouseEvent('mouseup', {bubbles: true, clientX: 100, clientY: 100})); + expect(mouseup).toBe(false); + map._renderTaskQueue.run(); + expect(mouseup).toBe(false); + + canvas.dispatchEvent(new MouseEvent('click', {bubbles: true, clientX: 100, clientY: 100})); + expect(click).toBe(false); + map._renderTaskQueue.run(); + expect(click).toBe(false); + + map.remove(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/marker.test.ts b/web/libraries/maplibre-gl/src/ui/marker.test.ts new file mode 100644 index 00000000..92fa686f --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/marker.test.ts @@ -0,0 +1,839 @@ +import {createMap as globalCreateMap, beforeMapTest} from '../util/test/util'; +import {Marker} from './marker'; +import {Popup} from './popup'; +import {LngLat} from '../geo/lng_lat'; +import Point from '@mapbox/point-geometry'; +import simulate from '../../test/unit/lib/simulate_interaction'; +import type {Terrain} from '../render/terrain'; + +function createMap(options = {}) { + const container = window.document.createElement('div'); + window.document.body.appendChild(container); + Object.defineProperty(container, 'clientWidth', {value: 512}); + Object.defineProperty(container, 'clientHeight', {value: 512}); + return globalCreateMap({container, ...options}); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('marker', () => { + test('Marker uses a default marker element with an appropriate offset', () => { + const marker = new Marker(); + expect(marker.getElement()).toBeTruthy(); + expect(marker.getOffset().equals(new Point(0, -14))).toBeTruthy(); + }); + + test('Marker uses a default marker element with custom color', () => { + const marker = new Marker({color: '#123456'}); + expect(marker.getElement().innerHTML.includes('#123456')).toBeTruthy(); + }); + + test('Marker uses a default marker element with custom scale', () => { + const map = createMap(); + const defaultMarker = new Marker() + .setLngLat([0, 0]) + .addTo(map); + // scale smaller than default + const smallerMarker = new Marker({scale: 0.8}) + .setLngLat([0, 0]) + .addTo(map); + // scale larger than default + const largerMarker = new Marker({scale: 2}) + .setLngLat([0, 0]) + .addTo(map); + + // initial dimensions of svg element + expect( + defaultMarker.getElement().children[0].getAttribute('height').includes('41') + ).toBeTruthy(); + expect(defaultMarker.getElement().children[0].getAttribute('width').includes('27')).toBeTruthy(); + + // (41 * 0.8) = 32.8, (27 * 0.8) = 21.6 + expect( + smallerMarker.getElement().children[0].getAttribute('height').includes('32.8') + ).toBeTruthy(); + expect( + smallerMarker.getElement().children[0].getAttribute('width').includes('21.6') + ).toBeTruthy(); + + // (41 * 2) = 82, (27 * 2) = 54 + expect(largerMarker.getElement().children[0].getAttribute('height').includes('82')).toBeTruthy(); + expect(largerMarker.getElement().children[0].getAttribute('width').includes('54')).toBeTruthy(); + + }); + + test('Marker uses a default marker with custom offset', () => { + const marker = new Marker({offset: [1, 2]}); + expect(marker.getElement()).toBeTruthy(); + expect(marker.getOffset().equals(new Point(1, 2))).toBeTruthy(); + }); + + test('Marker uses the provided element', () => { + const element = window.document.createElement('div'); + const marker = new Marker({element}); + expect(marker.getElement()).toBe(element); + }); + + test('Marker#addTo adds the marker element to the canvas container', () => { + const map = createMap(); + new Marker() + .setLngLat([-77.01866, 38.888]) + .addTo(map); + + expect(map.getCanvasContainer().querySelectorAll('.maplibregl-marker')).toHaveLength(1); + + map.remove(); + }); + + test('Marker adds classes from className option, methods for class manipulations works properly', () => { + const map = createMap(); + const marker = new Marker({className: 'some classes'}) + .setLngLat([0, 0]) + .addTo(map); + + const markerElement = marker.getElement(); + expect(markerElement.classList.contains('some')).toBeTruthy(); + expect(markerElement.classList.contains('classes')).toBeTruthy(); + + marker.addClassName('addedClass'); + expect(markerElement.classList.contains('addedClass')).toBeTruthy(); + + marker.removeClassName('addedClass'); + expect(!markerElement.classList.contains('addedClass')).toBeTruthy(); + + marker.toggleClassName('toggle'); + expect(markerElement.classList.contains('toggle')).toBeTruthy(); + + marker.toggleClassName('toggle'); + expect(!markerElement.classList.contains('toggle')).toBeTruthy(); + + expect(() => marker.addClassName('should throw exception')).toThrow(window.DOMException); + expect(() => marker.removeClassName('should throw exception')).toThrow(window.DOMException); + expect(() => marker.toggleClassName('should throw exception')).toThrow(window.DOMException); + + expect(() => marker.addClassName('')).toThrow(window.DOMException); + expect(() => marker.removeClassName('')).toThrow(window.DOMException); + expect(() => marker.toggleClassName('')).toThrow(window.DOMException); + }); + + test('Marker provides LngLat accessors', () => { + expect(new Marker().getLngLat()).toBeUndefined(); + + expect(new Marker().setLngLat([1, 2]).getLngLat() instanceof LngLat).toBeTruthy(); + expect(new Marker().setLngLat([1, 2]).getLngLat()).toEqual(new LngLat(1, 2)); + + expect(new Marker().setLngLat(new LngLat(1, 2)).getLngLat() instanceof LngLat).toBeTruthy(); + expect(new Marker().setLngLat(new LngLat(1, 2)).getLngLat()).toEqual(new LngLat(1, 2)); + + }); + + test('Marker provides offset accessors', () => { + expect(new Marker().setOffset([1, 2]).getOffset() instanceof Point).toBeTruthy(); + expect(new Marker().setOffset([1, 2]).getOffset()).toEqual(new Point(1, 2)); + + expect(new Marker().setOffset(new Point(1, 2)).getOffset() instanceof Point).toBeTruthy(); + expect(new Marker().setOffset(new Point(1, 2)).getOffset()).toEqual(new Point(1, 2)); + + }); + + test('Marker#setPopup binds a popup', () => { + const popup = new Popup(); + const marker = new Marker() + .setPopup(popup); + expect(marker.getPopup()).toBe(popup); + }); + + test('Marker#setPopup unbinds a popup', () => { + const marker = new Marker() + .setPopup(new Popup()) + .setPopup(); + expect(!marker.getPopup()).toBeTruthy(); + }); + + test('Marker#togglePopup opens a popup that was closed', () => { + const map = createMap(); + const marker = new Marker() + .setLngLat([0, 0]) + .addTo(map) + .setPopup(new Popup()) + .togglePopup(); + + expect(marker.getPopup().isOpen()).toBeTruthy(); + + map.remove(); + }); + + test('Marker#togglePopup closes a popup that was open', () => { + const map = createMap(); + const marker = new Marker() + .setLngLat([0, 0]) + .addTo(map) + .setPopup(new Popup()) + .togglePopup() + .togglePopup(); + + expect(!marker.getPopup().isOpen()).toBeTruthy(); + + map.remove(); + }); + + test('Enter key on Marker opens a popup that was closed', () => { + const map = createMap(); + const marker = new Marker() + .setLngLat([0, 0]) + .addTo(map) + .setPopup(new Popup()); + + // popup not initially open + expect(marker.getPopup().isOpen()).toBeFalsy(); + + simulate.keypress(marker.getElement(), {code: 'Enter'}); + + // popup open after Enter keypress + expect(marker.getPopup().isOpen()).toBeTruthy(); + + map.remove(); + }); + + test('Space key on Marker opens a popup that was closed', () => { + const map = createMap(); + const marker = new Marker() + .setLngLat([0, 0]) + .addTo(map) + .setPopup(new Popup()); + + // popup not initially open + expect(marker.getPopup().isOpen()).toBeFalsy(); + + simulate.keypress(marker.getElement(), {code: 'Space'}); + + // popup open after Enter keypress + expect(marker.getPopup().isOpen()).toBeTruthy(); + + map.remove(); + }); + + test('Marker#setPopup sets a tabindex', () => { + const popup = new Popup(); + const marker = new Marker() + .setPopup(popup); + expect(marker.getElement().getAttribute('tabindex')).toBe('0'); + }); + + test('Marker#setPopup removes tabindex when unset', () => { + const popup = new Popup(); + const marker = new Marker() + .setPopup(popup) + .setPopup(); + expect(marker.getElement().getAttribute('tabindex')).toBeFalsy(); + }); + + test('Marker#setPopup does not replace existing tabindex', () => { + const element = window.document.createElement('div'); + element.setAttribute('tabindex', '5'); + const popup = new Popup(); + const marker = new Marker({element}) + .setPopup(popup); + expect(marker.getElement().getAttribute('tabindex')).toBe('5'); + }); + + test('Marker anchor defaults to center', () => { + const map = createMap(); + const marker = new Marker() + .setLngLat([0, 0]) + .addTo(map); + + expect(marker.getElement().classList.contains('maplibregl-marker-anchor-center')).toBeTruthy(); + expect(marker.getElement().style.transform).toMatch(/translate\(-50%,-50%\)/); + + map.remove(); + }); + + test('Marker anchors as specified by the anchor option', () => { + const map = createMap(); + const marker = new Marker({anchor: 'top'}) + .setLngLat([0, 0]) + .addTo(map); + + expect(marker.getElement().classList.contains('maplibregl-marker-anchor-top')).toBeTruthy(); + expect(marker.getElement().style.transform).toMatch(/translate\(-50%,0\)/); + + map.remove(); + }); + + test('Popup offsets around default Marker', () => { + const map = createMap(); + + const marker = new Marker() + .setLngLat([0, 0]) + .setPopup(new Popup().setText('Test')) + .addTo(map); + + expect(marker.getPopup().options.offset['bottom'][1] < 0).toBeTruthy(); + expect(marker.getPopup().options.offset['top'][1] === 0).toBeTruthy(); + expect(marker.getPopup().options.offset['left'][0] > 0).toBeTruthy(); + expect(marker.getPopup().options.offset['right'][0] < 0).toBeTruthy(); + + expect(marker.getPopup().options.offset['bottom-left'][0] > 0).toBeTruthy(); + expect(marker.getPopup().options.offset['bottom-left'][1] < 0).toBeTruthy(); + expect(marker.getPopup().options.offset['bottom-right'][0] < 0).toBeTruthy(); + expect(marker.getPopup().options.offset['bottom-right'][1] < 0).toBeTruthy(); + + expect(marker.getPopup().options.offset['top-left']).toEqual([0, 0]); + expect(marker.getPopup().options.offset['top-right']).toEqual([0, 0]); + + }); + + test('Popup anchors around default Marker', () => { + const map = createMap(); + + const marker = new Marker() + .setLngLat([0, 0]) + .setPopup(new Popup().setText('Test')) + .addTo(map); + + // open the popup + marker.togglePopup(); + + const mapHeight = map.getContainer().clientHeight; + const markerTop = -marker.getPopup().options.offset['bottom'][1]; // vertical distance from tip of marker to the top in pixels + const markerRight = -marker.getPopup().options.offset['right'][0]; // horizontal distance from the tip of the marker to the right in pixels + + // give the popup some height + Object.defineProperty(marker.getPopup()._container, 'offsetWidth', {value: 100}); + Object.defineProperty(marker.getPopup()._container, 'offsetHeight', {value: 100}); + + // marker should default to above since it has enough space + expect( + marker.getPopup()._container.classList.contains('maplibregl-popup-anchor-bottom') + ).toBeTruthy(); + + // move marker to the top forcing the popup to below + marker.setLngLat(map.unproject([mapHeight / 2, markerTop])); + expect( + marker.getPopup()._container.classList.contains('maplibregl-popup-anchor-top') + ).toBeTruthy(); + + // move marker to the right forcing the popup to the left + marker.setLngLat(map.unproject([mapHeight - markerRight, mapHeight / 2])); + expect( + marker.getPopup()._container.classList.contains('maplibregl-popup-anchor-right') + ).toBeTruthy(); + + // move marker to the left forcing the popup to the right + marker.setLngLat(map.unproject([markerRight, mapHeight / 2])); + expect( + marker.getPopup()._container.classList.contains('maplibregl-popup-anchor-left') + ).toBeTruthy(); + + // move marker to the top left forcing the popup to the bottom right + marker.setLngLat(map.unproject([markerRight, markerTop])); + expect( + marker.getPopup()._container.classList.contains('maplibregl-popup-anchor-top-left') + ).toBeTruthy(); + + // move marker to the top right forcing the popup to the bottom left + marker.setLngLat(map.unproject([mapHeight - markerRight, markerTop])); + expect( + marker.getPopup()._container.classList.contains('maplibregl-popup-anchor-top-right') + ).toBeTruthy(); + + // move marker to the bottom left forcing the popup to the top right + marker.setLngLat(map.unproject([markerRight, mapHeight])); + expect( + marker.getPopup()._container.classList.contains('maplibregl-popup-anchor-bottom-left') + ).toBeTruthy(); + + // move marker to the bottom right forcing the popup to the top left + marker.setLngLat(map.unproject([mapHeight - markerRight, mapHeight])); + expect( + marker.getPopup()._container.classList.contains('maplibregl-popup-anchor-bottom-right') + ).toBeTruthy(); + + }); + + test('Marker drag functionality can be added with drag option', () => { + const map = createMap(); + const marker = new Marker({draggable: true}) + .setLngLat([0, 0]) + .addTo(map); + + expect(marker.isDraggable()).toBe(true); + + map.remove(); + }); + + test('Marker#setDraggable adds drag functionality', () => { + const map = createMap(); + const marker = new Marker() + .setLngLat([0, 0]) + .setDraggable(true) + .addTo(map); + + expect(marker.isDraggable()).toBe(true); + + map.remove(); + }); + + test('Marker#setDraggable turns off drag functionality', () => { + const map = createMap(); + const marker = new Marker({draggable: true}) + .setLngLat([0, 0]) + .addTo(map); + + expect(marker.isDraggable()).toBe(true); + + marker.setDraggable(false); + + expect(marker.isDraggable()).toBe(false); + + map.remove(); + }); + + test('Marker with draggable:true fires dragstart, drag, and dragend events at appropriate times in response to mouse-triggered drag with map-inherited clickTolerance', () => { + const map = createMap(); + const marker = new Marker({draggable: true}) + .setLngLat([0, 0]) + .addTo(map); + const el = marker.getElement(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + marker.on('dragstart', dragstart); + marker.on('drag', drag); + marker.on('dragend', dragend); + + simulate.mousedown(el, {clientX: 0, clientY: 0}); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe(''); + + simulate.mousemove(el, {clientX: 2.9, clientY: 0}); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe(''); + + // above map's click tolerance + simulate.mousemove(el, {clientX: 3.1, clientY: 0}); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe('none'); + + simulate.mousemove(el, {clientX: 0, clientY: 0}); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(2); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe('none'); + + simulate.mouseup(el); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(2); + expect(dragend).toHaveBeenCalledTimes(1); + expect(el.style.pointerEvents).toBe('auto'); + + map.remove(); + }); + + test('Marker with draggable:true fires dragstart, drag, and dragend events at appropriate times in response to mouse-triggered drag with marker-specific clickTolerance', () => { + const map = createMap(); + const marker = new Marker({draggable: true, clickTolerance: 4}) + .setLngLat([0, 0]) + .addTo(map); + const el = marker.getElement(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + marker.on('dragstart', dragstart); + marker.on('drag', drag); + marker.on('dragend', dragend); + + simulate.mousedown(el, {clientX: 0, clientY: 0}); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe(''); + + simulate.mousemove(el, {clientX: 3.9, clientY: 0}); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe(''); + + // above map's click tolerance + simulate.mousemove(el, {clientX: 4.1, clientY: 0}); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe('none'); + + simulate.mousemove(el, {clientX: 0, clientY: 0}); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(2); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe('none'); + + simulate.mouseup(el); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(2); + expect(dragend).toHaveBeenCalledTimes(1); + expect(el.style.pointerEvents).toBe('auto'); + + map.remove(); + }); + + test('Marker with draggable:false does not fire dragstart, drag, and dragend events in response to a mouse-triggered drag', () => { + const map = createMap(); + const marker = new Marker({}) + .setLngLat([0, 0]) + .addTo(map); + const el = marker.getElement(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + marker.on('dragstart', dragstart); + marker.on('drag', drag); + marker.on('dragend', dragend); + + simulate.mousedown(el, {clientX: 0, clientY: 0}); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + + simulate.mousemove(el, {clientX: 3, clientY: 1}); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + + simulate.mouseup(el); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + + map.remove(); + }); + + test('Marker with draggable:true fires dragstart, drag, and dragend events at appropriate times in response to a touch-triggered drag with map-inherited clickTolerance', () => { + const map = createMap(); + const marker = new Marker({draggable: true}) + .setLngLat([0, 0]) + .addTo(map); + const el = marker.getElement(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + marker.on('dragstart', dragstart); + marker.on('drag', drag); + marker.on('dragend', dragend); + + simulate.touchstart(el, {touches: [{clientX: 0, clientY: 0}]}); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe(''); + + simulate.touchmove(el, {touches: [{clientX: 2.9, clientY: 0}]}); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe(''); + + // above map's click tolerance + simulate.touchmove(el, {touches: [{clientX: 3.1, clientY: 0}]}); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe('none'); + + simulate.touchmove(el, {touches: [{clientX: 0, clientY: 0}]}); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(2); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe('none'); + + simulate.touchend(el); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(2); + expect(dragend).toHaveBeenCalledTimes(1); + expect(el.style.pointerEvents).toBe('auto'); + + map.remove(); + }); + + test('Marker with draggable:true fires dragstart, drag, and dragend events at appropriate times in response to a touch-triggered drag with marker-specific clickTolerance', () => { + const map = createMap(); + const marker = new Marker({draggable: true, clickTolerance: 4}) + .setLngLat([0, 0]) + .addTo(map); + const el = marker.getElement(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + marker.on('dragstart', dragstart); + marker.on('drag', drag); + marker.on('dragend', dragend); + + simulate.touchstart(el, {touches: [{clientX: 0, clientY: 0}]}); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe(''); + + simulate.touchmove(el, {touches: [{clientX: 3.9, clientY: 0}]}); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe(''); + + // above map's click tolerance + simulate.touchmove(el, {touches: [{clientX: 4.1, clientY: 0}]}); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(1); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe('none'); + + simulate.touchmove(el, {touches: [{clientX: 0, clientY: 0}]}); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(2); + expect(dragend).not.toHaveBeenCalled(); + expect(el.style.pointerEvents).toBe('none'); + + simulate.touchend(el); + expect(dragstart).toHaveBeenCalledTimes(1); + expect(drag).toHaveBeenCalledTimes(2); + expect(dragend).toHaveBeenCalledTimes(1); + expect(el.style.pointerEvents).toBe('auto'); + + map.remove(); + }); + + test('Marker with draggable:false does not fire dragstart, drag, and dragend events in response to a touch-triggered drag', () => { + const map = createMap(); + const marker = new Marker({}) + .setLngLat([0, 0]) + .addTo(map); + const el = marker.getElement(); + + const dragstart = jest.fn(); + const drag = jest.fn(); + const dragend = jest.fn(); + + marker.on('dragstart', dragstart); + marker.on('drag', drag); + marker.on('dragend', dragend); + + simulate.touchstart(el, {touches: [{clientX: 0, clientY: 0}]}); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + + simulate.touchmove(el, {touches: [{clientX: 0, clientY: 0}]}); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + + simulate.touchend(el); + expect(dragstart).not.toHaveBeenCalled(); + expect(drag).not.toHaveBeenCalled(); + expect(dragend).not.toHaveBeenCalled(); + + map.remove(); + }); + + test('Marker with draggable:true moves to new position in response to a mouse-triggered drag', () => { + const map = createMap(); + const marker = new Marker({draggable: true}) + .setLngLat([0, 0]) + .addTo(map); + const el = marker.getElement(); + const startPos = map.project(marker.getLngLat()); + simulate.mousedown(el); + simulate.mousemove(el, {clientX: 10, clientY: 10}); + simulate.mouseup(el); + + const endPos = map.project(marker.getLngLat()); + expect(Math.round(endPos.x)).toBe(startPos.x + 10); + expect(Math.round(endPos.y)).toBe(startPos.y + 10); + + map.remove(); + }); + + test('Marker with draggable:false does not move to new position in response to a mouse-triggered drag', () => { + const map = createMap(); + const marker = new Marker({}) + .setLngLat([0, 0]) + .addTo(map); + const el = marker.getElement(); + const startPos = map.project(marker.getLngLat()); + + simulate.mousedown(el); + simulate.mousemove(el); + simulate.mouseup(el); + + const endPos = map.project(marker.getLngLat()); + + expect(startPos.x).toBe(endPos.x); + expect(startPos.y).toBe(endPos.y); + + map.remove(); + }); + + test('Marker with draggable:true does not error if removed on mousedown', () => { + const map = createMap(); + const marker = new Marker({draggable: true}) + .setLngLat([0, 0]) + .addTo(map); + const el = marker.getElement(); + simulate.mousedown(el); + simulate.mousemove(el, {clientX: 10, clientY: 10}); + + marker.remove(); + expect(map.fire('mouseup')).toBeTruthy(); + }); + + test('Marker can set rotationAlignment and pitchAlignment', () => { + const map = createMap(); + const marker = new Marker({rotationAlignment: 'map', pitchAlignment: 'map'}) + .setLngLat([0, 0]) + .addTo(map); + + expect(marker.getRotationAlignment()).toBe('map'); + expect(marker.getPitchAlignment()).toBe('map'); + + map.remove(); + }); + + test('Marker can set and update rotation', () => { + const map = createMap(); + const marker = new Marker({rotation: 45}) + .setLngLat([0, 0]) + .addTo(map); + + expect(marker.getRotation()).toBe(45); + + marker.setRotation(90); + expect(marker.getRotation()).toBe(90); + + map.remove(); + }); + + test('Marker transforms rotation with the map', () => { + const map = createMap(); + const marker = new Marker({rotationAlignment: 'map'}) + .setLngLat([0, 0]) + .addTo(map); + + const rotationRegex = /rotateZ\(-?([0-9]+)deg\)/; + const initialRotation = marker.getElement().style.transform.match(rotationRegex)[1]; + + map.setBearing(map.getBearing() + 180); + + const finalRotation = marker.getElement().style.transform.match(rotationRegex)[1]; + expect(initialRotation).not.toBe(finalRotation); + + map.remove(); + }); + + test('Marker transforms pitch with the map', () => { + const map = createMap(); + const marker = new Marker({pitchAlignment: 'map'}) + .setLngLat([0, 0]) + .addTo(map); + + map.setPitch(0); + + const rotationRegex = /rotateX\(-?([0-9]+)deg\)/; + const initialPitch = marker.getElement().style.transform.match(rotationRegex)[1]; + + map.setPitch(45); + + const finalPitch = marker.getElement().style.transform.match(rotationRegex)[1]; + expect(initialPitch).not.toBe(finalPitch); + + map.remove(); + }); + + test('Marker pitchAlignment when set to auto defaults to rotationAlignment', () => { + const map = createMap(); + const marker = new Marker({rotationAlignment: 'map', pitchAlignment: 'auto'}) + .setLngLat([0, 0]) + .addTo(map); + + expect(marker.getRotationAlignment()).toBe(marker.getPitchAlignment()); + + map.remove(); + }); + + test('Marker pitchAlignment when set to auto defaults to rotationAlignment (setter/getter)', () => { + const map = createMap(); + const marker = new Marker({pitchAlignment: 'map'}) + .setLngLat([0, 0]) + .addTo(map); + + expect(marker.getPitchAlignment()).toBe('map'); + marker.setRotationAlignment('viewport'); + marker.setPitchAlignment('auto'); + expect(marker.getRotationAlignment()).toBe(marker.getPitchAlignment()); + + map.remove(); + }); + + test('Marker removed after update when terrain is on should clear timeout', () => { + jest.spyOn(global, 'setTimeout'); + jest.spyOn(global, 'clearTimeout'); + const map = createMap(); + const marker = new Marker() + .setLngLat([0, 0]) + .addTo(map); + map.terrain = { + getElevationForLngLatZoom: () => 0 + } as any as Terrain; + + marker.setOffset([10, 10]); + + expect(setTimeout).toHaveBeenCalled(); + marker.remove(); + expect(clearTimeout).toHaveBeenCalled(); + + map.remove(); + }); + + test('Marker after the terrain event must listen to the render event till is fully loaded', async () => { + const map = createMap(); + + new Marker() + .setLngLat([1, 1]) + .addTo(map); + + expect(map._oneTimeListeners.render).toBeUndefined(); + + map.fire('terrain'); + expect(map._oneTimeListeners.render).toHaveLength(1); + + map.fire('render'); + expect(map._oneTimeListeners.render).toHaveLength(1); + + map.fire('render'); + expect(map._oneTimeListeners.render).toHaveLength(1); + + // await idle to be fully loaded + await map.once('idle'); + map.fire('render'); + expect(map._oneTimeListeners.render).toHaveLength(0); + map.remove(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/marker.ts b/web/libraries/maplibre-gl/src/ui/marker.ts new file mode 100644 index 00000000..e5c6908a --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/marker.ts @@ -0,0 +1,771 @@ +import {DOM} from '../util/dom'; +import {LngLat} from '../geo/lng_lat'; +import Point from '@mapbox/point-geometry'; +import {smartWrap} from '../util/smart_wrap'; +import {anchorTranslate, applyAnchorClass} from './anchor'; +import type {PositionAnchor} from './anchor'; +import {Event, Evented} from '../util/evented'; +import type {Map} from './map'; +import {Popup, Offset} from './popup'; +import type {LngLatLike} from '../geo/lng_lat'; +import type {MapMouseEvent, MapTouchEvent} from './events'; +import type {PointLike} from './camera'; + +/** + * Alignment options of rotation and pitch + */ +type Alignment = 'map' | 'viewport' | 'auto'; + +/** + * The {@link Marker} options object + */ +type MarkerOptions = { + /** + * DOM element to use as a marker. The default is a light blue, droplet-shaped SVG marker. + */ + element?: HTMLElement; + /** + * Space-separated CSS class names to add to marker element. + */ + className?: string; + /** + * The offset in pixels as a {@link PointLike} object to apply relative to the element's center. Negatives indicate left and up. + */ + offset?: PointLike; + /** + * A string indicating the part of the Marker that should be positioned closest to the coordinate set via {@link Marker#setLngLat}. + * Options are `'center'`, `'top'`, `'bottom'`, `'left'`, `'right'`, `'top-left'`, `'top-right'`, `'bottom-left'`, and `'bottom-right'`. + * @defaultValue 'center' + * */ + anchor?: PositionAnchor; + /** + * The color to use for the default marker if options.element is not provided. The default is light blue. + * @defaultValue '#3FB1CE' + */ + color?: string; + /** + * The scale to use for the default marker if options.element is not provided. The default scale corresponds to a height of `41px` and a width of `27px`. + * @defaultValue 1 + */ + scale?: number; + /** + * A boolean indicating whether or not a marker is able to be dragged to a new position on the map. + * @defaultValue false + */ + draggable?: boolean; + /** + * The max number of pixels a user can shift the mouse pointer during a click on the marker for it to be considered a valid click (as opposed to a marker drag). The default is to inherit map's clickTolerance. + * @defaultValue 0 + */ + clickTolerance?: number; + /** + * The rotation angle of the marker in degrees, relative to its respective `rotationAlignment` setting. A positive value will rotate the marker clockwise. + * @defaultValue 0 + */ + rotation?: number; + /** + * `map` aligns the `Marker`'s rotation relative to the map, maintaining a bearing as the map rotates. `viewport` aligns the `Marker`'s rotation relative to the viewport, agnostic to map rotations. `auto` is equivalent to `viewport`. + * @defaultValue 'auto' + */ + rotationAlignment?: Alignment; + /** + * `map` aligns the `Marker` to the plane of the map. `viewport` aligns the `Marker` to the plane of the viewport. `auto` automatically matches the value of `rotationAlignment`. + * @defaultValue 'auto' + */ + pitchAlignment?: Alignment; +}; + +/** + * Creates a marker component + * + * @group Markers and Controls + * + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([30.5, 50.5]) + * .addTo(map); + * ``` + * + * @example + * Set options + * ```ts + * let marker = new maplibregl.Marker({ + * color: "#FFFFFF", + * draggable: true + * }).setLngLat([30.5, 50.5]) + * .addTo(map); + * ``` + * @see [Add custom icons with Markers](https://maplibre.org/maplibre-gl-js/docs/examples/custom-marker-icons/) + * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/) + * + * ### Events + * + * @event `dragstart` Fired when dragging starts, `marker` object that is being dragged + * + * @event `drag` Fired while dragging. `marker` object that is being dragged + * + * @event `dragend` Fired when the marker is finished being dragged, `marker` object that was dragged + */ +export class Marker extends Evented { + _map: Map; + _anchor: PositionAnchor; + _offset: Point; + _element: HTMLElement; + _popup: Popup; + _lngLat: LngLat; + _pos: Point; + _color: string; + _scale: number; + _defaultMarker: boolean; + _draggable: boolean; + _clickTolerance: number; + _isDragging: boolean; + _state: 'inactive' | 'pending' | 'active'; // used for handling drag events + _positionDelta: Point; + _pointerdownPos: Point; + _rotation: number; + _pitchAlignment: Alignment; + _rotationAlignment: Alignment; + _originalTabIndex: string; // original tabindex of _element + _opacityTimeout: ReturnType; + + /** + * @param options - the options + */ + constructor(options?: MarkerOptions) { + super(); + + this._anchor = options && options.anchor || 'center'; + this._color = options && options.color || '#3FB1CE'; + this._scale = options && options.scale || 1; + this._draggable = options && options.draggable || false; + this._clickTolerance = options && options.clickTolerance || 0; + this._isDragging = false; + this._state = 'inactive'; + this._rotation = options && options.rotation || 0; + this._rotationAlignment = options && options.rotationAlignment || 'auto'; + this._pitchAlignment = options && options.pitchAlignment && options.pitchAlignment !== 'auto' ? options.pitchAlignment : this._rotationAlignment; + + if (!options || !options.element) { + this._defaultMarker = true; + this._element = DOM.create('div'); + this._element.setAttribute('aria-label', 'Map marker'); + + // create default map marker SVG + const svg = DOM.createNS('http://www.w3.org/2000/svg', 'svg'); + const defaultHeight = 41; + const defaultWidth = 27; + svg.setAttributeNS(null, 'display', 'block'); + svg.setAttributeNS(null, 'height', `${defaultHeight}px`); + svg.setAttributeNS(null, 'width', `${defaultWidth}px`); + svg.setAttributeNS(null, 'viewBox', `0 0 ${defaultWidth} ${defaultHeight}`); + + const markerLarge = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + markerLarge.setAttributeNS(null, 'stroke', 'none'); + markerLarge.setAttributeNS(null, 'stroke-width', '1'); + markerLarge.setAttributeNS(null, 'fill', 'none'); + markerLarge.setAttributeNS(null, 'fill-rule', 'evenodd'); + + const page1 = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + page1.setAttributeNS(null, 'fill-rule', 'nonzero'); + + const shadow = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + shadow.setAttributeNS(null, 'transform', 'translate(3.0, 29.0)'); + shadow.setAttributeNS(null, 'fill', '#000000'); + + const ellipses = [ + {'rx': '10.5', 'ry': '5.25002273'}, + {'rx': '10.5', 'ry': '5.25002273'}, + {'rx': '9.5', 'ry': '4.77275007'}, + {'rx': '8.5', 'ry': '4.29549936'}, + {'rx': '7.5', 'ry': '3.81822308'}, + {'rx': '6.5', 'ry': '3.34094679'}, + {'rx': '5.5', 'ry': '2.86367051'}, + {'rx': '4.5', 'ry': '2.38636864'} + ]; + + for (const data of ellipses) { + const ellipse = DOM.createNS('http://www.w3.org/2000/svg', 'ellipse'); + ellipse.setAttributeNS(null, 'opacity', '0.04'); + ellipse.setAttributeNS(null, 'cx', '10.5'); + ellipse.setAttributeNS(null, 'cy', '5.80029008'); + ellipse.setAttributeNS(null, 'rx', data['rx']); + ellipse.setAttributeNS(null, 'ry', data['ry']); + shadow.appendChild(ellipse); + } + + const background = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + background.setAttributeNS(null, 'fill', this._color); + + const bgPath = DOM.createNS('http://www.w3.org/2000/svg', 'path'); + bgPath.setAttributeNS(null, 'd', 'M27,13.5 C27,19.074644 20.250001,27.000002 14.75,34.500002 C14.016665,35.500004 12.983335,35.500004 12.25,34.500002 C6.7499993,27.000002 0,19.222562 0,13.5 C0,6.0441559 6.0441559,0 13.5,0 C20.955844,0 27,6.0441559 27,13.5 Z'); + + background.appendChild(bgPath); + + const border = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + border.setAttributeNS(null, 'opacity', '0.25'); + border.setAttributeNS(null, 'fill', '#000000'); + + const borderPath = DOM.createNS('http://www.w3.org/2000/svg', 'path'); + borderPath.setAttributeNS(null, 'd', 'M13.5,0 C6.0441559,0 0,6.0441559 0,13.5 C0,19.222562 6.7499993,27 12.25,34.5 C13,35.522727 14.016664,35.500004 14.75,34.5 C20.250001,27 27,19.074644 27,13.5 C27,6.0441559 20.955844,0 13.5,0 Z M13.5,1 C20.415404,1 26,6.584596 26,13.5 C26,15.898657 24.495584,19.181431 22.220703,22.738281 C19.945823,26.295132 16.705119,30.142167 13.943359,33.908203 C13.743445,34.180814 13.612715,34.322738 13.5,34.441406 C13.387285,34.322738 13.256555,34.180814 13.056641,33.908203 C10.284481,30.127985 7.4148684,26.314159 5.015625,22.773438 C2.6163816,19.232715 1,15.953538 1,13.5 C1,6.584596 6.584596,1 13.5,1 Z'); + + border.appendChild(borderPath); + + const maki = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + maki.setAttributeNS(null, 'transform', 'translate(6.0, 7.0)'); + maki.setAttributeNS(null, 'fill', '#FFFFFF'); + + const circleContainer = DOM.createNS('http://www.w3.org/2000/svg', 'g'); + circleContainer.setAttributeNS(null, 'transform', 'translate(8.0, 8.0)'); + + const circle1 = DOM.createNS('http://www.w3.org/2000/svg', 'circle'); + circle1.setAttributeNS(null, 'fill', '#000000'); + circle1.setAttributeNS(null, 'opacity', '0.25'); + circle1.setAttributeNS(null, 'cx', '5.5'); + circle1.setAttributeNS(null, 'cy', '5.5'); + circle1.setAttributeNS(null, 'r', '5.4999962'); + + const circle2 = DOM.createNS('http://www.w3.org/2000/svg', 'circle'); + circle2.setAttributeNS(null, 'fill', '#FFFFFF'); + circle2.setAttributeNS(null, 'cx', '5.5'); + circle2.setAttributeNS(null, 'cy', '5.5'); + circle2.setAttributeNS(null, 'r', '5.4999962'); + + circleContainer.appendChild(circle1); + circleContainer.appendChild(circle2); + + page1.appendChild(shadow); + page1.appendChild(background); + page1.appendChild(border); + page1.appendChild(maki); + page1.appendChild(circleContainer); + + svg.appendChild(page1); + + svg.setAttributeNS(null, 'height', `${defaultHeight * this._scale}px`); + svg.setAttributeNS(null, 'width', `${defaultWidth * this._scale}px`); + + this._element.appendChild(svg); + + // if no element and no offset option given apply an offset for the default marker + // the -14 as the y value of the default marker offset was determined as follows + // + // the marker tip is at the center of the shadow ellipse from the default svg + // the y value of the center of the shadow ellipse relative to the svg top left is "shadow transform translate-y (29.0) + ellipse cy (5.80029008)" + // offset to the svg center "height (41 / 2)" gives (29.0 + 5.80029008) - (41 / 2) and rounded for an integer pixel offset gives 14 + // negative is used to move the marker up from the center so the tip is at the Marker lngLat + this._offset = Point.convert(options && options.offset || [0, -14]); + } else { + this._element = options.element; + this._offset = Point.convert(options && options.offset || [0, 0]); + } + + this._element.classList.add('maplibregl-marker'); + this._element.addEventListener('dragstart', (e: DragEvent) => { + e.preventDefault(); + }); + this._element.addEventListener('mousedown', (e: MouseEvent) => { + // prevent focusing on click + e.preventDefault(); + }); + applyAnchorClass(this._element, this._anchor, 'marker'); + + if (options && options.className) { + for (const name of options.className.split(' ')) { + this._element.classList.add(name); + } + } + + this._popup = null; + } + + /** + * Attaches the `Marker` to a `Map` object. + * @param map - The MapLibre GL JS map to add the marker to. + * @returns `this` + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([30.5, 50.5]) + * .addTo(map); // add the marker to the map + * ``` + */ + addTo(map: Map): this { + this.remove(); + this._map = map; + map.getCanvasContainer().appendChild(this._element); + map.on('move', this._update); + map.on('moveend', this._update); + map.on('terrain', this._update); + + this.setDraggable(this._draggable); + this._update(); + + // If we attached the `click` listener to the marker element, the popup + // would close once the event propagated to `map` due to the + // `Popup#_onClickClose` listener. + this._map.on('click', this._onMapClick); + + return this; + } + + /** + * Removes the marker from a map + * @example + * ```ts + * let marker = new maplibregl.Marker().addTo(map); + * marker.remove(); + * ``` + * @returns `this` + */ + remove(): this { + if (this._opacityTimeout) { + clearTimeout(this._opacityTimeout); + delete this._opacityTimeout; + } + if (this._map) { + this._map.off('click', this._onMapClick); + this._map.off('move', this._update); + this._map.off('moveend', this._update); + this._map.off('mousedown', this._addDragHandler); + this._map.off('touchstart', this._addDragHandler); + this._map.off('mouseup', this._onUp); + this._map.off('touchend', this._onUp); + this._map.off('mousemove', this._onMove); + this._map.off('touchmove', this._onMove); + delete this._map; + } + DOM.remove(this._element); + if (this._popup) this._popup.remove(); + return this; + } + + /** + * Get the marker's geographical location. + * + * The longitude of the result may differ by a multiple of 360 degrees from the longitude previously + * set by `setLngLat` because `Marker` wraps the anchor longitude across copies of the world to keep + * the marker on screen. + * + * @returns A {@link LngLat} describing the marker's location. + * @example + * ```ts + * // Store the marker's longitude and latitude coordinates in a variable + * let lngLat = marker.getLngLat(); + * // Print the marker's longitude and latitude values in the console + * console.log('Longitude: ' + lngLat.lng + ', Latitude: ' + lngLat.lat ) + * ``` + * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/) + */ + getLngLat(): LngLat { + return this._lngLat; + } + + /** + * Set the marker's geographical position and move it. + * @param lnglat - A {@link LngLat} describing where the marker should be located. + * @returns `this` + * @example + * Create a new marker, set the longitude and latitude, and add it to the map + * ```ts + * new maplibregl.Marker() + * .setLngLat([-65.017, -16.457]) + * .addTo(map); + * ``` + * @see [Add custom icons with Markers](https://maplibre.org/maplibre-gl-js/docs/examples/custom-marker-icons/) + * @see [Create a draggable Marker](https://maplibre.org/maplibre-gl-js/docs/examples/drag-a-marker/) + */ + setLngLat(lnglat: LngLatLike): this { + this._lngLat = LngLat.convert(lnglat); + this._pos = null; + if (this._popup) this._popup.setLngLat(this._lngLat); + this._update(); + return this; + } + + /** + * Returns the `Marker`'s HTML element. + * @returns element + */ + getElement(): HTMLElement { + return this._element; + } + + /** + * Binds a {@link Popup} to the {@link Marker}. + * @param popup - An instance of the {@link Popup} class. If undefined or null, any popup + * set on this {@link Marker} instance is unset. + * @returns `this` + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([0, 0]) + * .setPopup(new maplibregl.Popup().setHTML("

Hello World!

")) // add popup + * .addTo(map); + * ``` + * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/) + */ + setPopup(popup?: Popup | null): this { + if (this._popup) { + this._popup.remove(); + this._popup = null; + this._element.removeEventListener('keypress', this._onKeyPress); + + if (!this._originalTabIndex) { + this._element.removeAttribute('tabindex'); + } + } + + if (popup) { + if (!('offset' in popup.options)) { + const markerHeight = 41 - (5.8 / 2); + const markerRadius = 13.5; + const linearOffset = Math.abs(markerRadius) / Math.SQRT2; + popup.options.offset = this._defaultMarker ? { + 'top': [0, 0], + 'top-left': [0, 0], + 'top-right': [0, 0], + 'bottom': [0, -markerHeight], + 'bottom-left': [linearOffset, (markerHeight - markerRadius + linearOffset) * -1], + 'bottom-right': [-linearOffset, (markerHeight - markerRadius + linearOffset) * -1], + 'left': [markerRadius, (markerHeight - markerRadius) * -1], + 'right': [-markerRadius, (markerHeight - markerRadius) * -1] + } as Offset : this._offset; + } + this._popup = popup; + if (this._lngLat) this._popup.setLngLat(this._lngLat); + + this._originalTabIndex = this._element.getAttribute('tabindex'); + if (!this._originalTabIndex) { + this._element.setAttribute('tabindex', '0'); + } + this._element.addEventListener('keypress', this._onKeyPress); + } + + return this; + } + + _onKeyPress = (e: KeyboardEvent) => { + const code = e.code; + const legacyCode = e.charCode || e.keyCode; + + if ( + (code === 'Space') || (code === 'Enter') || + (legacyCode === 32) || (legacyCode === 13) // space or enter + ) { + this.togglePopup(); + } + }; + + _onMapClick = (e: MapMouseEvent) => { + const targetElement = e.originalEvent.target; + const element = this._element; + + if (this._popup && (targetElement === element || element.contains(targetElement as any))) { + this.togglePopup(); + } + }; + + /** + * Returns the {@link Popup} instance that is bound to the {@link Marker}. + * @returns popup + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([0, 0]) + * .setPopup(new maplibregl.Popup().setHTML("

Hello World!

")) + * .addTo(map); + * + * console.log(marker.getPopup()); // return the popup instance + * ``` + */ + getPopup(): Popup { + return this._popup; + } + + /** + * Opens or closes the {@link Popup} instance that is bound to the {@link Marker}, depending on the current state of the {@link Popup}. + * @returns `this` + * @example + * ```ts + * let marker = new maplibregl.Marker() + * .setLngLat([0, 0]) + * .setPopup(new maplibregl.Popup().setHTML("

Hello World!

")) + * .addTo(map); + * + * marker.togglePopup(); // toggle popup open or closed + * ``` + */ + togglePopup(): this { + const popup = this._popup; + + if (!popup) return this; + else if (popup.isOpen()) popup.remove(); + else popup.addTo(this._map); + return this; + } + + _update = (e?: { type: 'move' | 'moveend' | 'terrain' | 'render' }) => { + if (!this._map) return; + + const isFullyLoaded = this._map.loaded() && !this._map.isMoving(); + if (e?.type === 'terrain' || (e?.type === 'render' && !isFullyLoaded)) { + this._map.once('render', this._update); + } + + if (this._map.transform.renderWorldCopies) { + this._lngLat = smartWrap(this._lngLat, this._pos, this._map.transform); + } + + this._pos = this._map.project(this._lngLat)._add(this._offset); + + let rotation = ''; + if (this._rotationAlignment === 'viewport' || this._rotationAlignment === 'auto') { + rotation = `rotateZ(${this._rotation}deg)`; + } else if (this._rotationAlignment === 'map') { + rotation = `rotateZ(${this._rotation - this._map.getBearing()}deg)`; + } + + let pitch = ''; + if (this._pitchAlignment === 'viewport' || this._pitchAlignment === 'auto') { + pitch = 'rotateX(0deg)'; + } else if (this._pitchAlignment === 'map') { + pitch = `rotateX(${this._map.getPitch()}deg)`; + } + + // because rounding the coordinates at every `move` event causes stuttered zooming + // we only round them when _update is called with `moveend` or when its called with + // no arguments (when the Marker is initialized or Marker#setLngLat is invoked). + if (!e || e.type === 'moveend') { + this._pos = this._pos.round(); + } + + DOM.setTransform(this._element, `${anchorTranslate[this._anchor]} translate(${this._pos.x}px, ${this._pos.y}px) ${pitch} ${rotation}`); + + // in case of 3D, ask the terrain coords-framebuffer for this pos and check if the marker is visible + // call this logic in setTimeout with a timeout of 100ms to save performance in map-movement + if (this._map.terrain && !this._opacityTimeout) this._opacityTimeout = setTimeout(() => { + const lnglat = this._map.unproject(this._pos); + const metresPerPixel = 40075016.686 * Math.abs(Math.cos(this._lngLat.lat * Math.PI / 180)) / Math.pow(2, this._map.transform.tileZoom + 8); + this._element.style.opacity = lnglat.distanceTo(this._lngLat) > metresPerPixel * 20 ? '0.2' : '1.0'; + this._opacityTimeout = null; + }, 100); + }; + + /** + * Get the marker's offset. + * @returns The marker's screen coordinates in pixels. + */ + getOffset(): Point { + return this._offset; + } + + /** + * Sets the offset of the marker + * @param offset - The offset in pixels as a {@link PointLike} object to apply relative to the element's center. Negatives indicate left and up. + * @returns `this` + */ + setOffset(offset: PointLike): this { + this._offset = Point.convert(offset); + this._update(); + return this; + } + + /** + * Adds a CSS class to the marker element. + * + * @param className - on-empty string with CSS class name to add to marker element + * + * @example + * ``` + * let marker = new maplibregl.Marker() + * marker.addClassName('some-class') + * ``` + */ + addClassName(className: string) { + this._element.classList.add(className); + } + + /** + * Removes a CSS class from the marker element. + * + * @param className - Non-empty string with CSS class name to remove from marker element + * + * @example + * ```ts + * let marker = new maplibregl.Marker() + * marker.removeClassName('some-class') + * ``` + */ + removeClassName(className: string) { + this._element.classList.remove(className); + } + + /** + * Add or remove the given CSS class on the marker element, depending on whether the element currently has that class. + * + * @param className - Non-empty string with CSS class name to add/remove + * + * @returns if the class was removed return false, if class was added, then return true + * + * @example + * ```ts + * let marker = new maplibregl.Marker() + * marker.toggleClassName('toggleClass') + * ``` + */ + toggleClassName(className: string): boolean { + return this._element.classList.toggle(className); + } + + _onMove = (e: MapMouseEvent | MapTouchEvent) => { + if (!this._isDragging) { + const clickTolerance = this._clickTolerance || this._map._clickTolerance; + this._isDragging = e.point.dist(this._pointerdownPos) >= clickTolerance; + } + if (!this._isDragging) return; + + this._pos = e.point.sub(this._positionDelta); + this._lngLat = this._map.unproject(this._pos); + this.setLngLat(this._lngLat); + // suppress click event so that popups don't toggle on drag + this._element.style.pointerEvents = 'none'; + + // make sure dragstart only fires on the first move event after mousedown. + // this can't be on mousedown because that event doesn't necessarily + // imply that a drag is about to happen. + if (this._state === 'pending') { + this._state = 'active'; + this.fire(new Event('dragstart')); + } + this.fire(new Event('drag')); + }; + + _onUp = () => { + // revert to normal pointer event handling + this._element.style.pointerEvents = 'auto'; + this._positionDelta = null; + this._pointerdownPos = null; + this._isDragging = false; + this._map.off('mousemove', this._onMove); + this._map.off('touchmove', this._onMove); + + // only fire dragend if it was preceded by at least one drag event + if (this._state === 'active') { + this.fire(new Event('dragend')); + } + + this._state = 'inactive'; + }; + + _addDragHandler = (e: MapMouseEvent | MapTouchEvent) => { + if (this._element.contains(e.originalEvent.target as any)) { + e.preventDefault(); + + // We need to calculate the pixel distance between the click point + // and the marker position, with the offset accounted for. Then we + // can subtract this distance from the mousemove event's position + // to calculate the new marker position. + // If we don't do this, the marker 'jumps' to the click position + // creating a jarring UX effect. + this._positionDelta = e.point.sub(this._pos).add(this._offset); + + this._pointerdownPos = e.point; + + this._state = 'pending'; + this._map.on('mousemove', this._onMove); + this._map.on('touchmove', this._onMove); + this._map.once('mouseup', this._onUp); + this._map.once('touchend', this._onUp); + } + }; + + /** + * Sets the `draggable` property and functionality of the marker + * @param shouldBeDraggable - Turns drag functionality on/off + * @returns `this` + */ + setDraggable(shouldBeDraggable?: boolean): this { + this._draggable = !!shouldBeDraggable; // convert possible undefined value to false + + // handle case where map may not exist yet + // e.g. when setDraggable is called before addTo + if (this._map) { + if (shouldBeDraggable) { + this._map.on('mousedown', this._addDragHandler); + this._map.on('touchstart', this._addDragHandler); + } else { + this._map.off('mousedown', this._addDragHandler); + this._map.off('touchstart', this._addDragHandler); + } + } + + return this; + } + + /** + * Returns true if the marker can be dragged + * @returns True if the marker is draggable. + */ + isDraggable(): boolean { + return this._draggable; + } + + /** + * Sets the `rotation` property of the marker. + * @param rotation - The rotation angle of the marker (clockwise, in degrees), relative to its respective {@link Marker#setRotationAlignment} setting. + * @returns `this` + */ + setRotation(rotation?: number): this { + this._rotation = rotation || 0; + this._update(); + return this; + } + + /** + * Returns the current rotation angle of the marker (in degrees). + * @returns The current rotation angle of the marker. + */ + getRotation(): number { + return this._rotation; + } + + /** + * Sets the `rotationAlignment` property of the marker. + * @param alignment - Sets the `rotationAlignment` property of the marker. defaults to 'auto' + * @returns `this` + */ + setRotationAlignment(alignment?: Alignment): this { + this._rotationAlignment = alignment || 'auto'; + this._update(); + return this; + } + + /** + * Returns the current `rotationAlignment` property of the marker. + * @returns The current rotational alignment of the marker. + */ + getRotationAlignment(): Alignment { + return this._rotationAlignment; + } + + /** + * Sets the `pitchAlignment` property of the marker. + * @param alignment - Sets the `pitchAlignment` property of the marker. If alignment is 'auto', it will automatically match `rotationAlignment`. + * @returns `this` + */ + setPitchAlignment(alignment?: Alignment): this { + this._pitchAlignment = alignment && alignment !== 'auto' ? alignment : this._rotationAlignment; + this._update(); + return this; + } + + /** + * Returns the current `pitchAlignment` property of the marker. + * @returns The current pitch alignment of the marker in degrees. + */ + getPitchAlignment(): Alignment { + return this._pitchAlignment; + } +} diff --git a/web/libraries/maplibre-gl/src/ui/popup.test.ts b/web/libraries/maplibre-gl/src/ui/popup.test.ts new file mode 100644 index 00000000..e7ea468e --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/popup.test.ts @@ -0,0 +1,738 @@ +import {createMap as globalCreateMap, beforeMapTest} from '../util/test/util'; +import {Popup, Offset} from './popup'; +import {LngLat} from '../geo/lng_lat'; +import Point from '@mapbox/point-geometry'; +import simulate from '../../test/unit/lib/simulate_interaction'; +import {PositionAnchor} from './anchor'; + +const containerWidth = 512; +const containerHeight = 512; + +function createMap(options?) { + options = options || {}; + const container = window.document.createElement('div'); + window.document.body.appendChild(container); + Object.defineProperty(container, 'clientWidth', {value: options.width || containerWidth}); + Object.defineProperty(container, 'clientHeight', {value: options.height || containerHeight}); + return globalCreateMap({container}); +} + +beforeEach(() => { + beforeMapTest(); +}); + +describe('popup', () => { + + test('Popup#getElement returns a .maplibregl-popup element', () => { + const map = createMap(); + const popup = new Popup() + .setText('Test') + .setLngLat([0, 0]) + .addTo(map); + + expect(popup.isOpen()).toBeTruthy(); + expect(popup.getElement().classList.contains('maplibregl-popup')).toBeTruthy(); + }); + + test('Popup#addTo adds a .maplibregl-popup element', () => { + const map = createMap(); + const popup = new Popup() + .setText('Test') + .setLngLat([0, 0]) + .addTo(map); + + expect(popup.isOpen()).toBeTruthy(); + expect(map.getContainer().querySelectorAll('.maplibregl-popup')).toHaveLength(1); + }); + + test('Popup closes on map click events by default', () => { + const map = createMap(); + const popup = new Popup() + .setText('Test') + .setLngLat([0, 0]) + .addTo(map); + + simulate.click(map.getCanvas()); + + expect(!popup.isOpen()).toBeTruthy(); + }); + + test('Popup does not close on map click events when the closeOnClick option is false', () => { + const map = createMap(); + const popup = new Popup({closeOnClick: false}) + .setText('Test') + .setLngLat([0, 0]) + .addTo(map); + + simulate.click(map.getCanvas()); + + expect(popup.isOpen()).toBeTruthy(); + }); + + test('Popup closes on close button click events', () => { + const map = createMap(); + const popup = new Popup() + .setText('Test') + .setLngLat([0, 0]) + .addTo(map); + + simulate.click(map.getContainer().querySelector('.maplibregl-popup-close-button')); + + expect(!popup.isOpen()).toBeTruthy(); + }); + + test('Popup has no close button if closeButton option is false', () => { + const map = createMap(); + + const popup = new Popup({closeButton: false}) + .setText('Test') + .setLngLat([0, 0]) + .addTo(map); + + expect( + popup.getElement().querySelectorAll('.maplibregl-popup-close-button') + ).toHaveLength(0); + }); + + test('Popup does not close on map move events when the closeOnMove option is false', () => { + const map = createMap(); + const popup = new Popup({closeOnMove: false}) + .setText('Test') + .setLngLat([0, 0]) + .addTo(map); + + map.setCenter([-10, 0]); // longitude bounds: [-370, 350] + + expect(popup.isOpen()).toBeTruthy(); + }); + + test('Popup closes on map move events when the closeOnMove option is true', () => { + const map = createMap(); + const popup = new Popup({closeOnMove: true}) + .setText('Test') + .setLngLat([0, 0]) + .addTo(map); + + map.setCenter([-10, 0]); // longitude bounds: [-370, 350] + + expect(!popup.isOpen()).toBeTruthy(); + }); + + test('Popup fires close event when removed', () => { + const map = createMap(); + const onClose = jest.fn(); + + new Popup() + .setText('Test') + .setLngLat([0, 0]) + .on('close', onClose) + .addTo(map) + .remove(); + + expect(onClose).toHaveBeenCalled(); + }); + + test('Popup fires open event when added', () => { + const map = createMap(); + const onOpen = jest.fn(); + + new Popup() + .setText('Test') + .setLngLat([0, 0]) + .on('open', onOpen) + .addTo(map); + + expect(onOpen).toHaveBeenCalled(); + }); + + test('Popup content can be set via setText', () => { + const map = createMap(); + + const popup = new Popup({closeButton: false}) + .setLngLat([0, 0]) + .addTo(map) + .setText('Test'); + + expect(popup.getElement().textContent).toBe('Test'); + }); + + test('Popup content can be set via setHTML', () => { + const map = createMap(); + + const popup = new Popup({closeButton: false}) + .setLngLat([0, 0]) + .addTo(map) + .setHTML('Test'); + + expect(popup.getElement().querySelector('.maplibregl-popup-content').innerHTML).toBe('Test'); + }); + + test('Popup width maximum defaults to 240px', () => { + const map = createMap(); + + const popup = new Popup({closeButton: false}) + .setLngLat([0, 0]) + .addTo(map) + .setHTML('Test'); + + expect(popup.getMaxWidth()).toBe('240px'); + }); + + test('Popup width maximum can be set via using maxWidth option', () => { + const map = createMap(); + + const popup = new Popup({closeButton: false, maxWidth: '5px'}) + .setLngLat([0, 0]) + .addTo(map) + .setHTML('Test'); + + expect(popup.getMaxWidth()).toBe('5px'); + }); + + test('Popup width maximum can be set via maxWidth', () => { + const map = createMap(); + + const popup = new Popup({closeButton: false}) + .setLngLat([0, 0]) + .setHTML('Test') + .setMaxWidth('5px') + .addTo(map); + + expect(popup.getMaxWidth()).toBe('5px'); + }); + + test('Popup content can be set via setDOMContent', () => { + const map = createMap(); + const content = window.document.createElement('span'); + + const popup = new Popup({closeButton: false}) + .setLngLat([0, 0]) + .addTo(map) + .setDOMContent(content); + + expect(popup.getElement().querySelector('.maplibregl-popup-content').firstChild).toBe(content); + }); + + test('Popup#setText protects against XSS', () => { + const map = createMap(); + + const popup = new Popup({closeButton: false}) + .setLngLat([0, 0]) + .addTo(map) + .setText(''); + + expect(popup.getElement().textContent).toBe(''); + }); + + test('Popup content setters overwrite previous content', () => { + const map = createMap(); + + const popup = new Popup({closeButton: false}) + .setLngLat([0, 0]) + .addTo(map); + + popup.setText('Test 1'); + expect(popup.getElement().textContent).toBe('Test 1'); + + popup.setHTML('Test 2'); + expect(popup.getElement().textContent).toBe('Test 2'); + + popup.setDOMContent(window.document.createTextNode('Test 3')); + expect(popup.getElement().textContent).toBe('Test 3'); + + }); + + test('Popup provides LngLat accessors', () => { + expect(new Popup().getLngLat()).toBeUndefined(); + + expect(new Popup().setLngLat([1, 2]).getLngLat() instanceof LngLat).toBeTruthy(); + expect(new Popup().setLngLat([1, 2]).getLngLat()).toEqual(new LngLat(1, 2)); + + expect(new Popup().setLngLat(new LngLat(1, 2)).getLngLat() instanceof LngLat).toBeTruthy(); + expect(new Popup().setLngLat(new LngLat(1, 2)).getLngLat()).toEqual(new LngLat(1, 2)); + + }); + + test('Popup is positioned at the specified LngLat in a world copy', () => { + const map = createMap({width: 1024}); // longitude bounds: [-360, 360] + + const popup = new Popup() + .setLngLat([270, 0]) + .setText('Test') + .addTo(map); + + expect(popup._pos).toEqual(map.project([270, 0])); + }); + + test('Popup preserves object constancy of position after map move', () => { + const map = createMap({width: 1024}); // longitude bounds: [-360, 360] + + const popup = new Popup() + .setLngLat([270, 0]) + .setText('Test') + .addTo(map); + + map.setCenter([-10, 0]); // longitude bounds: [-370, 350] + expect(popup._pos).toEqual(map.project([270, 0])); + + map.setCenter([-20, 0]); // longitude bounds: [-380, 340] + expect(popup._pos).toEqual(map.project([270, 0])); + + }); + + test('Popup preserves object constancy of position after auto-wrapping center (left)', () => { + const map = createMap({width: 1024}); + map.setCenter([-175, 0]); // longitude bounds: [-535, 185] + + const popup = new Popup() + .setLngLat([0, 0]) + .setText('Test') + .addTo(map); + + map.setCenter([175, 0]); // longitude bounds: [-185, 535] + expect(popup._pos).toEqual(map.project([360, 0])); + + }); + + test('Popup preserves object constancy of position after auto-wrapping center (right)', () => { + const map = createMap({width: 1024}); + map.setCenter([175, 0]); // longitude bounds: [-185, 535] + + const popup = new Popup() + .setLngLat([0, 0]) + .setText('Test') + .addTo(map); + + map.setCenter([-175, 0]); // longitude bounds: [-185, 535] + expect(popup._pos).toEqual(map.project([-360, 0])); + + }); + + test('Popup wraps position after map move if it would otherwise go offscreen (right)', () => { + const map = createMap({width: 1024}); // longitude bounds: [-360, 360] + + const popup = new Popup() + .setLngLat([-355, 0]) + .setText('Test') + .addTo(map); + + map.setCenter([10, 0]); // longitude bounds: [-350, 370] + expect(popup._pos).toEqual(map.project([5, 0])); + }); + + test('Popup wraps position after map move if it would otherwise go offscreen (right)', () => { + const map = createMap({width: 1024}); // longitude bounds: [-360, 360] + + const popup = new Popup() + .setLngLat([355, 0]) + .setText('Test') + .addTo(map); + + map.setCenter([-10, 0]); // longitude bounds: [-370, 350] + expect(popup._pos).toEqual(map.project([-5, 0])); + }); + + test('Popup is repositioned at the specified LngLat', () => { + const map = createMap({width: 1024}); // longitude bounds: [-360, 360] + + const popup = new Popup() + .setLngLat([270, 0]) + .setText('Test') + .addTo(map) + .setLngLat([0, 0]); + + expect(popup._pos).toEqual(map.project([0, 0])); + }); + + test('Popup anchors as specified by the anchor option', () => { + const map = createMap(); + const popup = new Popup({anchor: 'top-left'}) + .setLngLat([0, 0]) + .setText('Test') + .addTo(map); + + expect(popup.getElement().classList.contains('maplibregl-popup-anchor-top-left')).toBeTruthy(); + }); + + ([ + ['top-left', new Point(10, 10), 'translate(0,0) translate(7px,7px)'], + ['top', new Point(containerWidth / 2, 10), 'translate(-50%,0) translate(0px,10px)'], + ['top-right', new Point(containerWidth - 10, 10), 'translate(-100%,0) translate(-7px,7px)'], + ['right', new Point(containerWidth - 10, containerHeight / 2), 'translate(-100%,-50%) translate(-10px,0px)'], + ['bottom-right', new Point(containerWidth - 10, containerHeight - 10), 'translate(-100%,-100%) translate(-7px,-7px)'], + ['bottom', new Point(containerWidth / 2, containerHeight - 10), 'translate(-50%,-100%) translate(0px,-10px)'], + ['bottom-left', new Point(10, containerHeight - 10), 'translate(0,-100%) translate(7px,-7px)'], + ['left', new Point(10, containerHeight / 2), 'translate(0,-50%) translate(10px,0px)'], + ['bottom', new Point(containerWidth / 2, containerHeight / 2), 'translate(-50%,-100%) translate(0px,-10px)'] + ] as [PositionAnchor, Point, string][]).forEach((args) => { + const anchor = args[0]; + const point = args[1]; + const transform = args[2]; + + test(`Popup automatically anchors to ${anchor}`, () => { + const map = createMap(); + const popup = new Popup() + .setLngLat([0, 0]) + .setText('Test') + .addTo(map); + + Object.defineProperty(popup.getElement(), 'offsetWidth', {value: 100}); + Object.defineProperty(popup.getElement(), 'offsetHeight', {value: 100}); + + jest.spyOn(map, 'project').mockReturnValue(point); + popup.setLngLat([0, 0]); + + expect(popup.getElement().classList.contains(`maplibregl-popup-anchor-${anchor}`)).toBeTruthy(); + }); + + test(`Popup translation reflects offset and ${anchor} anchor`, () => { + const map = createMap(); + jest.spyOn(map, 'project').mockReturnValue(new Point(0, 0)); + + const popup = new Popup({anchor, offset: 10}) + .setLngLat([0, 0]) + .setText('Test') + .addTo(map); + + expect(popup.getElement().style.transform).toBe(transform); + }); + }); + + test('Popup automatically anchors to top if its bottom offset would push it off-screen', () => { + const map = createMap(); + const point = new Point(containerWidth / 2, containerHeight / 2); + const options = {offset: { + 'bottom': [0, -25], + 'top': [0, 0] + } as Offset}; + const popup = new Popup(options) + .setLngLat([0, 0]) + .setText('Test') + .addTo(map); + + Object.defineProperty(popup.getElement(), 'offsetWidth', {value: containerWidth / 2}); + Object.defineProperty(popup.getElement(), 'offsetHeight', {value: containerHeight / 2}); + + jest.spyOn(map, 'project').mockReturnValue(point); + popup.setLngLat([0, 0]); + + expect(popup.getElement().classList.contains('maplibregl-popup-anchor-top')).toBeTruthy(); + }); + + test('Popup is offset via a PointLike offset option', () => { + const map = createMap(); + jest.spyOn(map, 'project').mockReturnValue(new Point(0, 0)); + + const popup = new Popup({anchor: 'top-left', offset: [5, 10]}) + .setLngLat([0, 0]) + .setText('Test') + .addTo(map); + + expect(popup.getElement().style.transform).toBe('translate(0,0) translate(5px,10px)'); + }); + + test('Popup is offset via an object offset option', () => { + const map = createMap(); + jest.spyOn(map, 'project').mockReturnValue(new Point(0, 0)); + + const popup = new Popup({anchor: 'top-left', offset: {'top-left': [5, 10]} as Offset}) + .setLngLat([0, 0]) + .setText('Test') + .addTo(map); + + expect(popup.getElement().style.transform).toBe('translate(0,0) translate(5px,10px)'); + }); + + test('Popup is offset via an incomplete object offset option', () => { + const map = createMap(); + jest.spyOn(map, 'project').mockReturnValue(new Point(0, 0)); + + const popup = new Popup({anchor: 'top-right', offset: {'top-left': [5, 10]} as Offset}) + .setLngLat([0, 0]) + .setText('Test') + .addTo(map); + + expect(popup.getElement().style.transform).toBe('translate(-100%,0) translate(0px,0px)'); + }); + + test('Popup offset can be set via setOffset', () => { + const map = createMap(); + + const popup = new Popup({offset: 5}) + .setLngLat([0, 0]) + .setText('Test') + .addTo(map); + + expect(popup.options.offset).toBe(5); + + popup.setOffset(10); + + expect(popup.options.offset).toBe(10); + }); + + test('Popup can be removed and added again (#1477)', () => { + const map = createMap(); + + new Popup() + .setText('Test') + .setLngLat([0, 0]) + .addTo(map) + .remove() + .addTo(map); + + expect(map.getContainer().querySelectorAll('.maplibregl-popup')).toHaveLength(1); + }); + + test('Popup#addTo is idempotent (#1811)', () => { + const map = createMap(); + + const popup = new Popup({closeButton: false}) + .setText('Test') + .setLngLat([0, 0]) + .addTo(map) + .addTo(map); + + expect(popup.getElement().querySelector('.maplibregl-popup-content').textContent).toBe('Test'); + }); + + test('Popup#remove is idempotent (#2395)', () => { + const map = createMap(); + + new Popup({closeButton: false}) + .setText('Test') + .setLngLat([0, 0]) + .addTo(map) + .remove() + .remove(); + + expect(map.getContainer().querySelectorAll('.maplibregl-popup')).toHaveLength(0); + }); + + test('Popup adds classes from className option, methods for class manipulations works properly', () => { + const map = createMap(); + const popup = new Popup({className: 'some classes'}) + .setText('Test') + .setLngLat([0, 0]) + .addTo(map); + + const popupContainer = popup.getElement(); + expect(popupContainer.classList.contains('some')).toBeTruthy(); + expect(popupContainer.classList.contains('classes')).toBeTruthy(); + + popup.addClassName('addedClass'); + expect(popupContainer.classList.contains('addedClass')).toBeTruthy(); + + popup.removeClassName('addedClass'); + expect(!popupContainer.classList.contains('addedClass')).toBeTruthy(); + + popup.toggleClassName('toggle'); + expect(popupContainer.classList.contains('toggle')).toBeTruthy(); + + popup.toggleClassName('toggle'); + expect(!popupContainer.classList.contains('toggle')).toBeTruthy(); + + expect(() => popup.addClassName('should throw exception')).toThrow(window.DOMException); + expect(() => popup.removeClassName('should throw exception')).toThrow(window.DOMException); + expect(() => popup.toggleClassName('should throw exception')).toThrow(window.DOMException); + + expect(() => popup.addClassName('')).toThrow(window.DOMException); + expect(() => popup.removeClassName('')).toThrow(window.DOMException); + expect(() => popup.toggleClassName('')).toThrow(window.DOMException); + + }); + + test('Cursor-tracked popup disappears on mouseout', () => { + const map = createMap(); + + const popup = new Popup() + .setText('Test') + .trackPointer() + .addTo(map); + + expect(popup._trackPointer).toBe(true); + }); + + test('Pointer-tracked popup is tagged with right class', () => { + const map = createMap(); + const popup = new Popup() + .setText('Test') + .trackPointer() + .addTo(map); + + expect( + popup._container.classList.value + ).toContain('maplibregl-popup-track-pointer'); + }); + + test('Pointer-tracked popup with content set later is tagged with right class ', () => { + const map = createMap(); + const popup = new Popup() + .trackPointer() + .addTo(map); + + popup.setText('Test'); + + expect( + popup._container.classList.value + ).toContain('maplibregl-popup-track-pointer'); + }); + + test('Pointer-tracked popup that is set afterwards is tagged with right class ', () => { + const map = createMap(); + const popup = new Popup() + .addTo(map); + + popup.setText('Test'); + popup.trackPointer(); + + expect( + popup._container.classList.value + ).toContain('maplibregl-popup-track-pointer'); + }); + + test('Pointer-tracked popup can be repositioned with setLngLat', () => { + const map = createMap(); + const popup = new Popup() + .setText('Test') + .trackPointer() + .setLngLat([0, 0]) + .addTo(map); + + expect(popup._pos).toEqual(map.project([0, 0])); + expect( + popup._container.classList.value + ).not.toContain('maplibregl-popup-track-pointer'); + }); + + test('Positioned popup lacks pointer-tracking class', () => { + const map = createMap(); + const popup = new Popup() + .setText('Test') + .setLngLat([0, 0]) + .addTo(map); + + expect( + popup._container.classList.value + ).not.toContain('maplibregl-popup-track-pointer'); + }); + + test('Positioned popup can be set to track pointer', () => { + const map = createMap(); + const popup = new Popup() + .setText('Test') + .setLngLat([0, 0]) + .trackPointer() + .addTo(map); + + simulate.mousemove(map.getCanvas(), {screenX: 0, screenY: 0}); + expect(popup._pos).toEqual({x: 0, y: 0}); + }); + + test('Popup closes on Map#remove', () => { + const map = createMap(); + const popup = new Popup() + .setText('Test') + .setLngLat([0, 0]) + .addTo(map); + + map.remove(); + + expect(!popup.isOpen()).toBeTruthy(); + }); + + test('Adding popup with no focusable content (Popup#setText) does not change the active element', () => { + const dummyFocusedEl = window.document.createElement('button'); + window.document.body.appendChild(dummyFocusedEl); + dummyFocusedEl.focus(); + + new Popup({closeButton: false}) + .setText('Test') + .setLngLat([0, 0]) + .addTo(createMap()); + + expect(window.document.activeElement).toBe(dummyFocusedEl); + }); + + test('Adding popup with no focusable content (Popup#setHTML) does not change the active element', () => { + const dummyFocusedEl = window.document.createElement('button'); + window.document.body.appendChild(dummyFocusedEl); + dummyFocusedEl.focus(); + + new Popup({closeButton: false}) + .setHTML('Test') + .setLngLat([0, 0]) + .addTo(createMap()); + + expect(window.document.activeElement).toBe(dummyFocusedEl); + }); + + test('Close button is focused if it is the only focusable element', () => { + const dummyFocusedEl = window.document.createElement('button'); + window.document.body.appendChild(dummyFocusedEl); + dummyFocusedEl.focus(); + + const popup = new Popup({closeButton: true}) + .setHTML('Test') + .setLngLat([0, 0]) + .addTo(createMap()); + + // Suboptimal because the string matching is case-sensitive + const closeButton = popup._container.querySelector('[aria-label^=\'Close\']'); + + expect(window.document.activeElement).toBe(closeButton); + }); + + test('If popup content contains a focusable element it is focused', () => { + const popup = new Popup({closeButton: true}) + .setHTML('Test') + .setLngLat([0, 0]) + .addTo(createMap()); + + const focusableEl = popup._container.querySelector('[data-testid=\'abc\']'); + + expect(window.document.activeElement).toBe(focusableEl); + }); + + test('Element with tabindex="-1" is not focused', () => { + const popup = new Popup({closeButton: true}) + .setHTML('Test') + .setLngLat([0, 0]) + .addTo(createMap()); + + const nonFocusableEl = popup._container.querySelector('[data-testid=\'abc\']'); + const closeButton = popup._container.querySelector('button[aria-label=\'Close popup\']'); + + expect(window.document.activeElement).not.toBe(nonFocusableEl); + expect(window.document.activeElement).toBe(closeButton); + }); + + test('If popup contains a disabled button and a focusable element then the latter is focused', () => { + const popup = new Popup({closeButton: true}) + .setHTML(` + + + `) + .setLngLat([0, 0]) + .addTo(createMap()); + + const focusableEl = popup._container.querySelector('[data-testid=\'abc\']'); + + expect(window.document.activeElement).toBe(focusableEl); + }); + + test('Popup with disabled focusing does not change the active element', () => { + const dummyFocusedEl = window.document.createElement('button'); + window.document.body.appendChild(dummyFocusedEl); + dummyFocusedEl.focus(); + + new Popup({closeButton: false, focusAfterOpen: false}) + .setHTML('Test') + .setLngLat([0, 0]) + .addTo(createMap()); + + expect(window.document.activeElement).toBe(dummyFocusedEl); + }); +}); diff --git a/web/libraries/maplibre-gl/src/ui/popup.ts b/web/libraries/maplibre-gl/src/ui/popup.ts new file mode 100644 index 00000000..e93b4916 --- /dev/null +++ b/web/libraries/maplibre-gl/src/ui/popup.ts @@ -0,0 +1,672 @@ +import {extend} from '../util/util'; +import {Event, Evented} from '../util/evented'; +import {MapMouseEvent} from '../ui/events'; +import {DOM} from '../util/dom'; +import {LngLat} from '../geo/lng_lat'; +import Point from '@mapbox/point-geometry'; +import {smartWrap} from '../util/smart_wrap'; +import {anchorTranslate, applyAnchorClass} from './anchor'; + +import type {PositionAnchor} from './anchor'; +import type {Map} from './map'; +import type {LngLatLike} from '../geo/lng_lat'; +import type {PointLike} from './camera'; + +const defaultOptions = { + closeButton: true, + closeOnClick: true, + focusAfterOpen: true, + className: '', + maxWidth: '240px' +}; + +/** + * A pixel offset specified as: + * - a single number specifying a distance from the location + * - a {@link PointLike} specifying a constant offset + * - an object of {@link Point}s specifying an offset for each anchor position + * Negative offsets indicate left and up. + */ +export type Offset = number | PointLike | { + [_ in PositionAnchor]: PointLike; +}; + +export type PopupOptions = { + /** + * If `true`, a close button will appear in the top right corner of the popup. + * @defaultValue true + */ + closeButton?: boolean; + /** + * If `true`, the popup will closed when the map is clicked. + * @defaultValue true + */ + closeOnClick?: boolean; + /** + * If `true`, the popup will closed when the map moves. + * @defaultValue false + */ + closeOnMove?: boolean; + /** + * If `true`, the popup will try to focus the first focusable element inside the popup. + * @defaultValue true + */ + focusAfterOpen?: boolean; + /** + * A string indicating the part of the Popup that should + * be positioned closest to the coordinate set via {@link Popup#setLngLat}. + * Options are `'center'`, `'top'`, `'bottom'`, `'left'`, `'right'`, `'top-left'`, + * `'top-right'`, `'bottom-left'`, and `'bottom-right'`. If unset the anchor will be + * dynamically set to ensure the popup falls within the map container with a preference + * for `'bottom'`. + */ + anchor?: PositionAnchor; + /** + * A pixel offset applied to the popup's location + */ + offset?: Offset; + /** + * Space-separated CSS class names to add to popup container + */ + className?: string; + /** + * A string that sets the CSS property of the popup's maximum width, eg `'300px'`. + * To ensure the popup resizes to fit its content, set this property to `'none'`. + * Available values can be found here: https://developer.mozilla.org/en-US/docs/Web/CSS/max-width + * @defaultValue '240px' + */ + maxWidth?: string; +}; + +const focusQuerySelector = [ + 'a[href]', + '[tabindex]:not([tabindex=\'-1\'])', + '[contenteditable]:not([contenteditable=\'false\'])', + 'button:not([disabled])', + 'input:not([disabled])', + 'select:not([disabled])', + 'textarea:not([disabled])', +].join(', '); + +/** + * A popup component. + * + * @group Markers and Controls + * + * + * @example + * Create a popup + * ```ts + * let popup = new maplibregl.Popup(); + * // Set an event listener that will fire + * // any time the popup is opened + * popup.on('open', function(){ + * console.log('popup was opened'); + * }); + * ``` + * + * @example + * Create a popup + * ```ts + * let popup = new maplibregl.Popup(); + * // Set an event listener that will fire + * // any time the popup is closed + * popup.on('close', function(){ + * console.log('popup was closed'); + * }); + * ``` + * + * @example + * ```ts + * let markerHeight = 50, markerRadius = 10, linearOffset = 25; + * let popupOffsets = { + * 'top': [0, 0], + * 'top-left': [0,0], + * 'top-right': [0,0], + * 'bottom': [0, -markerHeight], + * 'bottom-left': [linearOffset, (markerHeight - markerRadius + linearOffset) * -1], + * 'bottom-right': [-linearOffset, (markerHeight - markerRadius + linearOffset) * -1], + * 'left': [markerRadius, (markerHeight - markerRadius) * -1], + * 'right': [-markerRadius, (markerHeight - markerRadius) * -1] + * }; + * let popup = new maplibregl.Popup({offset: popupOffsets, className: 'my-class'}) + * .setLngLat(e.lngLat) + * .setHTML("

Hello World!

") + * .setMaxWidth("300px") + * .addTo(map); + * ``` + * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/) + * + * ### Events + * + * @event `open` Fired when the popup is opened manually or programmatically. `popup` object that was opened + * + * @event `close` Fired when the popup is closed manually or programmatically. `popup` object that was closed + */ +export class Popup extends Evented { + _map: Map; + options: PopupOptions; + _content: HTMLElement; + _container: HTMLElement; + _closeButton: HTMLButtonElement; + _tip: HTMLElement; + _lngLat: LngLat; + _trackPointer: boolean; + _pos: Point; + + constructor(options?: PopupOptions) { + super(); + this.options = extend(Object.create(defaultOptions), options); + } + + /** + * Adds the popup to a map. + * + * @param map - The MapLibre GL JS map to add the popup to. + * @returns `this` + * @example + * ```ts + * new maplibregl.Popup() + * .setLngLat([0, 0]) + * .setHTML("

Null Island

") + * .addTo(map); + * ``` + * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + * @see [Show polygon information on click](https://maplibre.org/maplibre-gl-js/docs/examples/polygon-popup-on-click/) + */ + addTo(map: Map): this { + if (this._map) this.remove(); + + this._map = map; + if (this.options.closeOnClick) { + this._map.on('click', this._onClose); + } + + if (this.options.closeOnMove) { + this._map.on('move', this._onClose); + } + + this._map.on('remove', this.remove); + this._update(); + this._focusFirstElement(); + + if (this._trackPointer) { + this._map.on('mousemove', this._onMouseMove); + this._map.on('mouseup', this._onMouseUp); + if (this._container) { + this._container.classList.add('maplibregl-popup-track-pointer'); + } + this._map._canvasContainer.classList.add('maplibregl-track-pointer'); + } else { + this._map.on('move', this._update); + } + + this.fire(new Event('open')); + + return this; + } + + /** + * @returns `true` if the popup is open, `false` if it is closed. + */ + isOpen() { + return !!this._map; + } + + /** + * Removes the popup from the map it has been added to. + * + * @example + * ```ts + * let popup = new maplibregl.Popup().addTo(map); + * popup.remove(); + * ``` + * @returns `this` + */ + remove = (): this => { + if (this._content) { + DOM.remove(this._content); + } + + if (this._container) { + DOM.remove(this._container); + delete this._container; + } + + if (this._map) { + this._map.off('move', this._update); + this._map.off('move', this._onClose); + this._map.off('click', this._onClose); + this._map.off('remove', this.remove); + this._map.off('mousemove', this._onMouseMove); + this._map.off('mouseup', this._onMouseUp); + this._map.off('drag', this._onDrag); + delete this._map; + } + + this.fire(new Event('close')); + + return this; + }; + + /** + * Returns the geographical location of the popup's anchor. + * + * The longitude of the result may differ by a multiple of 360 degrees from the longitude previously + * set by `setLngLat` because `Popup` wraps the anchor longitude across copies of the world to keep + * the popup on screen. + * + * @returns The geographical location of the popup's anchor. + */ + getLngLat(): LngLat { + return this._lngLat; + } + + /** + * Sets the geographical location of the popup's anchor, and moves the popup to it. Replaces trackPointer() behavior. + * + * @param lnglat - The geographical location to set as the popup's anchor. + * @returns `this` + */ + setLngLat(lnglat: LngLatLike): this { + this._lngLat = LngLat.convert(lnglat); + this._pos = null; + + this._trackPointer = false; + + this._update(); + + if (this._map) { + this._map.on('move', this._update); + this._map.off('mousemove', this._onMouseMove); + if (this._container) { + this._container.classList.remove('maplibregl-popup-track-pointer'); + } + this._map._canvasContainer.classList.remove('maplibregl-track-pointer'); + } + + return this; + } + + /** + * Tracks the popup anchor to the cursor position on screens with a pointer device (it will be hidden on touchscreens). Replaces the `setLngLat` behavior. + * For most use cases, set `closeOnClick` and `closeButton` to `false`. + * @example + * ```ts + * let popup = new maplibregl.Popup({ closeOnClick: false, closeButton: false }) + * .setHTML("

Hello World!

") + * .trackPointer() + * .addTo(map); + * ``` + * @returns `this` + */ + trackPointer(): this { + this._trackPointer = true; + this._pos = null; + this._update(); + if (this._map) { + this._map.off('move', this._update); + this._map.on('mousemove', this._onMouseMove); + this._map.on('drag', this._onDrag); + if (this._container) { + this._container.classList.add('maplibregl-popup-track-pointer'); + } + this._map._canvasContainer.classList.add('maplibregl-track-pointer'); + } + + return this; + + } + + /** + * Returns the `Popup`'s HTML element. + * @example + * Change the `Popup` element's font size + * ```ts + * let popup = new maplibregl.Popup() + * .setLngLat([-96, 37.8]) + * .setHTML("

Hello World!

") + * .addTo(map); + * let popupElem = popup.getElement(); + * popupElem.style.fontSize = "25px"; + * ``` + * @returns element + */ + getElement(): HTMLElement { + return this._container; + } + + /** + * Sets the popup's content to a string of text. + * + * This function creates a [Text](https://developer.mozilla.org/en-US/docs/Web/API/Text) node in the DOM, + * so it cannot insert raw HTML. Use this method for security against XSS + * if the popup content is user-provided. + * + * @param text - Textual content for the popup. + * @returns `this` + * @example + * ```ts + * let popup = new maplibregl.Popup() + * .setLngLat(e.lngLat) + * .setText('Hello, world!') + * .addTo(map); + * ``` + */ + setText(text: string): this { + return this.setDOMContent(document.createTextNode(text)); + } + + /** + * Sets the popup's content to the HTML provided as a string. + * + * This method does not perform HTML filtering or sanitization, and must be + * used only with trusted content. Consider {@link Popup#setText} if + * the content is an untrusted text string. + * + * @param html - A string representing HTML content for the popup. + * @returns `this` + * @example + * ```ts + * let popup = new maplibregl.Popup() + * .setLngLat(e.lngLat) + * .setHTML("

Hello World!

") + * .addTo(map); + * ``` + * @see [Display a popup](https://maplibre.org/maplibre-gl-js/docs/examples/popup/) + * @see [Display a popup on hover](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-hover/) + * @see [Display a popup on click](https://maplibre.org/maplibre-gl-js/docs/examples/popup-on-click/) + * @see [Attach a popup to a marker instance](https://maplibre.org/maplibre-gl-js/docs/examples/set-popup/) + */ + setHTML(html: string): this { + const frag = document.createDocumentFragment(); + const temp = document.createElement('body'); + let child; + temp.innerHTML = html; + while (true) { + child = temp.firstChild; + if (!child) break; + frag.appendChild(child); + } + + return this.setDOMContent(frag); + } + + /** + * Returns the popup's maximum width. + * + * @returns The maximum width of the popup. + */ + getMaxWidth(): string { + return this._container?.style.maxWidth; + } + + /** + * Sets the popup's maximum width. This is setting the CSS property `max-width`. + * Available values can be found here: https://developer.mozilla.org/en-US/docs/Web/CSS/max-width + * + * @param maxWidth - A string representing the value for the maximum width. + * @returns `this` + */ + setMaxWidth(maxWidth: string): this { + this.options.maxWidth = maxWidth; + this._update(); + return this; + } + + /** + * Sets the popup's content to the element provided as a DOM node. + * + * @param htmlNode - A DOM node to be used as content for the popup. + * @returns `this` + * @example + * Create an element with the popup content + * ```ts + * let div = document.createElement('div'); + * div.innerHTML = 'Hello, world!'; + * let popup = new maplibregl.Popup() + * .setLngLat(e.lngLat) + * .setDOMContent(div) + * .addTo(map); + * ``` + */ + setDOMContent(htmlNode: Node): this { + if (this._content) { + // Clear out children first. + while (this._content.hasChildNodes()) { + if (this._content.firstChild) { + this._content.removeChild(this._content.firstChild); + } + } + } else { + this._content = DOM.create('div', 'maplibregl-popup-content', this._container); + } + + // The close button should be the last tabbable element inside the popup for a good keyboard UX. + this._content.appendChild(htmlNode); + this._createCloseButton(); + this._update(); + this._focusFirstElement(); + return this; + } + + /** + * Adds a CSS class to the popup container element. + * + * @param className - Non-empty string with CSS class name to add to popup container + * + * @example + * ```ts + * let popup = new maplibregl.Popup() + * popup.addClassName('some-class') + * ``` + */ + addClassName(className: string) { + if (this._container) { + this._container.classList.add(className); + } + } + + /** + * Removes a CSS class from the popup container element. + * + * @param className - Non-empty string with CSS class name to remove from popup container + * + * @example + * ```ts + * let popup = new maplibregl.Popup() + * popup.removeClassName('some-class') + * ``` + */ + removeClassName(className: string) { + if (this._container) { + this._container.classList.remove(className); + } + } + + /** + * Sets the popup's offset. + * + * @param offset - Sets the popup's offset. + * @returns `this` + */ + setOffset (offset?: Offset): this { + this.options.offset = offset; + this._update(); + return this; + } + + /** + * Add or remove the given CSS class on the popup container, depending on whether the container currently has that class. + * + * @param className - Non-empty string with CSS class name to add/remove + * + * @returns if the class was removed return false, if class was added, then return true, undefined if there is no container + * + * @example + * ```ts + * let popup = new maplibregl.Popup() + * popup.toggleClassName('toggleClass') + * ``` + */ + toggleClassName(className: string): boolean | undefined { + if (this._container) { + return this._container.classList.toggle(className); + } + } + + _createCloseButton() { + if (this.options.closeButton) { + this._closeButton = DOM.create('button', 'maplibregl-popup-close-button', this._content); + this._closeButton.type = 'button'; + this._closeButton.setAttribute('aria-label', 'Close popup'); + this._closeButton.innerHTML = '×'; + this._closeButton.addEventListener('click', this._onClose); + } + } + + _onMouseUp = (event: MapMouseEvent) => { + this._update(event.point); + }; + + _onMouseMove = (event: MapMouseEvent) => { + this._update(event.point); + }; + + _onDrag = (event: MapMouseEvent) => { + this._update(event.point); + }; + + _update = (cursor?: Point) => { + const hasPosition = this._lngLat || this._trackPointer; + + if (!this._map || !hasPosition || !this._content) { return; } + + if (!this._container) { + this._container = DOM.create('div', 'maplibregl-popup', this._map.getContainer()); + this._tip = DOM.create('div', 'maplibregl-popup-tip', this._container); + this._container.appendChild(this._content); + if (this.options.className) { + for (const name of this.options.className.split(' ')) { + this._container.classList.add(name); + } + } + + if (this._trackPointer) { + this._container.classList.add('maplibregl-popup-track-pointer'); + } + } + + if (this.options.maxWidth && this._container.style.maxWidth !== this.options.maxWidth) { + this._container.style.maxWidth = this.options.maxWidth; + } + + if (this._map.transform.renderWorldCopies && !this._trackPointer) { + this._lngLat = smartWrap(this._lngLat, this._pos, this._map.transform); + } + + if (this._trackPointer && !cursor) return; + + const pos = this._pos = this._trackPointer && cursor ? cursor : this._map.project(this._lngLat); + + let anchor = this.options.anchor; + const offset = normalizeOffset(this.options.offset); + + if (!anchor) { + const width = this._container.offsetWidth; + const height = this._container.offsetHeight; + let anchorComponents; + + if (pos.y + offset.bottom.y < height) { + anchorComponents = ['top']; + } else if (pos.y > this._map.transform.height - height) { + anchorComponents = ['bottom']; + } else { + anchorComponents = []; + } + + if (pos.x < width / 2) { + anchorComponents.push('left'); + } else if (pos.x > this._map.transform.width - width / 2) { + anchorComponents.push('right'); + } + + if (anchorComponents.length === 0) { + anchor = 'bottom'; + } else { + anchor = (anchorComponents.join('-') as any); + } + } + + const offsetedPos = pos.add(offset[anchor]).round(); + DOM.setTransform(this._container, `${anchorTranslate[anchor]} translate(${offsetedPos.x}px,${offsetedPos.y}px)`); + applyAnchorClass(this._container, anchor, 'popup'); + }; + + _focusFirstElement() { + if (!this.options.focusAfterOpen || !this._container) return; + + const firstFocusable = this._container.querySelector(focusQuerySelector) as HTMLElement; + + if (firstFocusable) firstFocusable.focus(); + } + + _onClose = () => { + this.remove(); + }; +} + +function normalizeOffset(offset?: Offset | null) { + if (!offset) { + return normalizeOffset(new Point(0, 0)); + + } else if (typeof offset === 'number') { + // input specifies a radius from which to calculate offsets at all positions + const cornerOffset = Math.round(Math.abs(offset) / Math.SQRT2); + return { + 'center': new Point(0, 0), + 'top': new Point(0, offset), + 'top-left': new Point(cornerOffset, cornerOffset), + 'top-right': new Point(-cornerOffset, cornerOffset), + 'bottom': new Point(0, -offset), + 'bottom-left': new Point(cornerOffset, -cornerOffset), + 'bottom-right': new Point(-cornerOffset, -cornerOffset), + 'left': new Point(offset, 0), + 'right': new Point(-offset, 0) + }; + + } else if (offset instanceof Point || Array.isArray(offset)) { + // input specifies a single offset to be applied to all positions + const convertedOffset = Point.convert(offset); + return { + 'center': convertedOffset, + 'top': convertedOffset, + 'top-left': convertedOffset, + 'top-right': convertedOffset, + 'bottom': convertedOffset, + 'bottom-left': convertedOffset, + 'bottom-right': convertedOffset, + 'left': convertedOffset, + 'right': convertedOffset + }; + + } else { + // input specifies an offset per position + return { + 'center': Point.convert(offset['center'] || [0, 0]), + 'top': Point.convert(offset['top'] || [0, 0]), + 'top-left': Point.convert(offset['top-left'] || [0, 0]), + 'top-right': Point.convert(offset['top-right'] || [0, 0]), + 'bottom': Point.convert(offset['bottom'] || [0, 0]), + 'bottom-left': Point.convert(offset['bottom-left'] || [0, 0]), + 'bottom-right': Point.convert(offset['bottom-right'] || [0, 0]), + 'left': Point.convert(offset['left'] || [0, 0]), + 'right': Point.convert(offset['right'] || [0, 0]) + }; + } +} diff --git a/web/libraries/maplibre-gl/src/util/actor.test.ts b/web/libraries/maplibre-gl/src/util/actor.test.ts new file mode 100644 index 00000000..06ad375d --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/actor.test.ts @@ -0,0 +1,106 @@ +import {Actor, ActorTarget} from './actor'; +import {workerFactory} from './web_worker'; +import {MessageBus} from '../../test/unit/lib/web_worker_mock'; + +const originalWorker = global.Worker; + +function setTestWorker(MockWorker: { new(...args: any): any}) { + (global as any).Worker = function Worker(_: string) { + const parentListeners = []; + const workerListeners = []; + const parentBus = new MessageBus(workerListeners, parentListeners); + const workerBus = new MessageBus(parentListeners, workerListeners); + + parentBus.target = workerBus; + workerBus.target = parentBus; + + new MockWorker(workerBus); + + return parentBus; + }; +} + +describe('Actor', () => { + afterAll(() => { + global.Worker = originalWorker; + }); + + test('forwards responses to correct callback', done => { + setTestWorker(class MockWorker { + self: any; + actor: Actor; + constructor(self) { + this.self = self; + this.actor = new Actor(self, this); + } + getTile(mapId, params, callback) { + setTimeout(callback, 0, null, params); + } + getWorkerSource() { return null; } + }); + + const worker = workerFactory(); + + const m1 = new Actor(worker, {} as any, '1'); + const m2 = new Actor(worker, {} as any, '2'); + + let callbackCount = 0; + m1.send('getTile', {value: 1729}, (err, response) => { + expect(err).toBeFalsy(); + expect(response).toEqual({value: 1729}); + callbackCount++; + if (callbackCount === 2) { + done(); + } + }); + m2.send('getTile', {value: 4104}, (err, response) => { + expect(err).toBeFalsy(); + expect(response).toEqual({value: 4104}); + callbackCount++; + if (callbackCount === 2) { + done(); + } + }); + }); + + test('targets worker-initiated messages to correct map instance', done => { + let workerActor; + + setTestWorker(class MockWorker { + self: any; + actor: Actor; + constructor(self) { + this.self = self; + this.actor = workerActor = new Actor(self, this); + } + getWorkerSource() { return null; } + }); + + const worker = workerFactory(); + + new Actor(worker, { + test () { done(); } + } as any, '1'); + new Actor(worker, { + test () { + done('test failed'); + } + } as any, '2'); + + workerActor.send('test', {}, () => {}, '1'); + }); + + test('#remove unbinds event listener', done => { + const actor = new Actor({ + addEventListener (type, callback, useCapture) { + this._addEventListenerArgs = [type, callback, useCapture]; + }, + removeEventListener (type, callback, useCapture) { + expect([type, callback, useCapture]).toEqual(this._addEventListenerArgs); + done(); + } + } as ActorTarget, {} as any, null); + actor.remove(); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/util/actor.ts b/web/libraries/maplibre-gl/src/util/actor.ts new file mode 100644 index 00000000..5bdc7d32 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/actor.ts @@ -0,0 +1,263 @@ +import {isWorker} from './util'; +import {serialize, deserialize, Serialized} from './web_worker_transfer'; +import {ThrottledInvoker} from './throttled_invoker'; + +import type {Transferable} from '../types/transferable'; +import type {Cancelable} from '../types/cancelable'; +import type {WorkerSource} from '../source/worker_source'; +import type {OverscaledTileID} from '../source/tile_id'; +import type {Callback} from '../types/callback'; +import type {StyleGlyph} from '../style/style_glyph'; + +export interface ActorTarget { + addEventListener: typeof window.addEventListener; + removeEventListener: typeof window.removeEventListener; + postMessage: typeof window.postMessage; + terminate?: () => void; +} + +export interface WorkerSourceProvider { + getWorkerSource(mapId: string | number, sourceType: string, sourceName: string): WorkerSource; +} + +export interface GlyphsProvider { + getGlyphs(mapId: string, params: { + stacks: {[_: string]: Array}; + source: string; + tileID: OverscaledTileID; + type: string; + }, + callback: Callback<{[_: string]: {[_: number]: StyleGlyph}}> + ); +} + +export type MessageType = '' | '' | +'geojson.getClusterExpansionZoom' | 'geojson.getClusterChildren' | 'geojson.getClusterLeaves' | 'geojson.loadData' | +'removeSource' | 'loadWorkerSource' | 'loadDEMTile' | 'removeDEMTile' | +'removeTile' | 'reloadTile' | 'abortTile' | 'loadTile' | 'getTile' | +'getGlyphs' | 'getImages' | 'setImages' | +'syncRTLPluginState' | 'setReferrer' | 'setLayers' | 'updateLayers'; + +export type MessageData = { + id: string; + type: MessageType; + data?: Serialized; + targetMapId?: string | number | null; + mustQueue?: boolean; + error?: Serialized | null; + hasCallback?: boolean; + sourceMapId: string | number | null; +} + +export type Message = { + data: MessageData; +} + +/** + * An implementation of the [Actor design pattern](http://en.wikipedia.org/wiki/Actor_model) + * that maintains the relationship between asynchronous tasks and the objects + * that spin them off - in this case, tasks like parsing parts of styles, + * owned by the styles + */ +export class Actor { + target: ActorTarget; + parent: WorkerSourceProvider | GlyphsProvider; + mapId: string | number | null; + callbacks: { [x: number]: Function}; + name: string; + tasks: { [x: number]: MessageData }; + taskQueue: Array; + cancelCallbacks: { [x: number]: () => void }; + invoker: ThrottledInvoker; + globalScope: ActorTarget; + + /** + * @param target - The target + * @param parent - The parent + * @param mapId - A unique identifier for the Map instance using this Actor. + */ + constructor(target: ActorTarget, parent: WorkerSourceProvider | GlyphsProvider, mapId?: string | number) { + this.target = target; + this.parent = parent; + this.mapId = mapId; + this.callbacks = {}; + this.tasks = {}; + this.taskQueue = []; + this.cancelCallbacks = {}; + this.invoker = new ThrottledInvoker(this.process); + this.target.addEventListener('message', this.receive, false); + this.globalScope = isWorker() ? target : window; + } + + /** + * Sends a message from a main-thread map to a Worker or from a Worker back to + * a main-thread map instance. + * + * @param type - The name of the target method to invoke or '[source-type].[source-name].name' for a method on a WorkerSource. + * @param targetMapId - A particular mapId to which to send this message. + */ + send( + type: MessageType, + data: unknown, + callback?: Function | null, + targetMapId?: string | null, + mustQueue: boolean = false + ): Cancelable { + // We're using a string ID instead of numbers because they are being used as object keys + // anyway, and thus stringified implicitly. We use random IDs because an actor may receive + // message from multiple other actors which could run in different execution context. A + // linearly increasing ID could produce collisions. + const id = Math.round((Math.random() * 1e18)).toString(36).substring(0, 10); + if (callback) { + this.callbacks[id] = callback; + } + const buffers: Array = []; + const message: MessageData = { + id, + type, + hasCallback: !!callback, + targetMapId, + mustQueue, + sourceMapId: this.mapId, + data: serialize(data, buffers) + }; + + this.target.postMessage(message, {transfer: buffers}); + return { + cancel: () => { + if (callback) { + // Set the callback to null so that it never fires after the request is aborted. + delete this.callbacks[id]; + } + const cancelMessage: MessageData = { + id, + type: '', + targetMapId, + sourceMapId: this.mapId + }; + this.target.postMessage(cancelMessage); + } + }; + } + + receive = (message: Message) => { + const data = message.data; + const id = data.id; + + if (!id) { + return; + } + + if (data.targetMapId && this.mapId !== data.targetMapId) { + return; + } + + if (data.type === '') { + // Remove the original request from the queue. This is only possible if it + // hasn't been kicked off yet. The id will remain in the queue, but because + // there is no associated task, it will be dropped once it's time to execute it. + delete this.tasks[id]; + const cancel = this.cancelCallbacks[id]; + delete this.cancelCallbacks[id]; + if (cancel) { + cancel(); + } + } else { + if (isWorker() || data.mustQueue) { + // In workers, store the tasks that we need to process before actually processing them. This + // is necessary because we want to keep receiving messages, and in particular, + // messages. Some tasks may take a while in the worker thread, so before + // executing the next task in our queue, postMessage preempts this and + // messages can be processed. We're using a MessageChannel object to get throttle the + // process() flow to one at a time. + this.tasks[id] = data; + this.taskQueue.push(id); + this.invoker.trigger(); + } else { + // In the main thread, process messages immediately so that other work does not slip in + // between getting partial data back from workers. + this.processTask(id, data); + } + } + }; + + process = () => { + if (!this.taskQueue.length) { + return; + } + const id = this.taskQueue.shift(); + const task = this.tasks[id]; + delete this.tasks[id]; + // Schedule another process call if we know there's more to process _before_ invoking the + // current task. This is necessary so that processing continues even if the current task + // doesn't execute successfully. + if (this.taskQueue.length) { + this.invoker.trigger(); + } + if (!task) { + // If the task ID doesn't have associated task data anymore, it was canceled. + return; + } + + this.processTask(id, task); + }; + + processTask(id: string, task: MessageData) { + if (task.type === '') { + // The done() function in the counterpart has been called, and we are now + // firing the callback in the originating actor, if there is one. + const callback = this.callbacks[id]; + delete this.callbacks[id]; + if (callback) { + // If we get a response, but don't have a callback, the request was canceled. + if (task.error) { + callback(deserialize(task.error)); + } else { + callback(null, deserialize(task.data)); + } + } + } else { + let completed = false; + const buffers: Array = []; + const done = task.hasCallback ? (err: Error, data?: any) => { + completed = true; + delete this.cancelCallbacks[id]; + const responseMessage: MessageData = { + id, + type: '', + sourceMapId: this.mapId, + error: err ? serialize(err) : null, + data: serialize(data, buffers) + }; + this.target.postMessage(responseMessage, {transfer: buffers}); + } : (_) => { + completed = true; + }; + + let callback: Cancelable = null; + const params = deserialize(task.data); + if (this.parent[task.type]) { + // task.type == 'loadTile', 'removeTile', etc. + callback = this.parent[task.type](task.sourceMapId, params, done); + } else if ('getWorkerSource' in this.parent) { + // task.type == sourcetype.method + const keys = task.type.split('.'); + const scope = this.parent.getWorkerSource(task.sourceMapId, keys[0], (params as any).source); + callback = scope[keys[1]](params, done); + } else { + // No function was found. + done(new Error(`Could not find function ${task.type}`)); + } + + if (!completed && callback && callback.cancel) { + // Allows canceling the task as long as it hasn't been completed yet. + this.cancelCallbacks[id] = callback.cancel; + } + } + } + + remove() { + this.invoker.remove(); + this.target.removeEventListener('message', this.receive, false); + } +} diff --git a/web/libraries/maplibre-gl/src/util/ajax.test.ts b/web/libraries/maplibre-gl/src/util/ajax.test.ts new file mode 100644 index 00000000..f3a6146c --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/ajax.test.ts @@ -0,0 +1,177 @@ +import { + getArrayBuffer, + getJSON, + postData, + AJAXError, + sameOrigin +} from './ajax'; + +import {fakeServer, type FakeServer} from 'nise'; +import {destroyFetchMock, FetchMock, RequestMock, setupFetchMock} from './test/mock_fetch'; + +function readAsText(blob) { + return new Promise((resolve, reject) => { + const fileReader = new FileReader(); + fileReader.onload = () => resolve(fileReader.result); + fileReader.onerror = () => reject(fileReader.error); + fileReader.readAsText(blob); + }); +} + +describe('ajax', () => { + let server: FakeServer; + beforeEach(() => { + global.fetch = null; + server = fakeServer.create(); + }); + afterEach(() => { + server.restore(); + }); + + test('getArrayBuffer, 404', done => { + server.respondWith(request => { + request.respond(404, undefined, '404 Not Found'); + }); + getArrayBuffer({url: 'http://example.com/test.bin'}, async (error) => { + const ajaxError = error as AJAXError; + const body = await readAsText(ajaxError.body); + expect(ajaxError.status).toBe(404); + expect(ajaxError.statusText).toBe('Not Found'); + expect(ajaxError.url).toBe('http://example.com/test.bin'); + expect(body).toBe('404 Not Found'); + done(); + }); + server.respond(); + }); + + test('getJSON', done => { + server.respondWith(request => { + request.respond(200, {'Content-Type': 'application/json'}, '{"foo": "bar"}'); + }); + getJSON({url: ''}, (error, body) => { + expect(error).toBeFalsy(); + expect(body).toEqual({foo: 'bar'}); + done(); + }); + server.respond(); + }); + + test('getJSON, invalid syntax', done => { + server.respondWith(request => { + request.respond(200, {'Content-Type': 'application/json'}, 'how do i even'); + }); + getJSON({url: ''}, (error) => { + expect(error).toBeTruthy(); + done(); + }); + server.respond(); + }); + + test('getJSON, 404', done => { + server.respondWith(request => { + request.respond(404, undefined, '404 Not Found'); + }); + getJSON({url: 'http://example.com/test.json'}, async (error) => { + const ajaxError = error as AJAXError; + const body = await readAsText(ajaxError.body); + expect(ajaxError.status).toBe(404); + expect(ajaxError.statusText).toBe('Not Found'); + expect(ajaxError.url).toBe('http://example.com/test.json'); + expect(body).toBe('404 Not Found'); + done(); + }); + server.respond(); + }); + + test('postData, 204(no content): no error', done => { + server.respondWith(request => { + request.respond(204, undefined, undefined); + }); + postData({url: 'api.mapbox.com'}, (error) => { + expect(error).toBeNull(); + done(); + }); + server.respond(); + }); + + test('sameOrigin method', () => { + jest.spyOn(window, 'location', 'get').mockReturnValue({ + protocol: 'https:', + host: 'somewhere.com' + } as any); + + expect(sameOrigin('https://somewhere.com')).toBe(true); + expect(sameOrigin('https://somewhere.com/path')).toBe(true); + expect(sameOrigin('https://somewhere.com/path/?q=abc')).toBe(true); + + expect(sameOrigin('https://somewhere.com:443/path')).toBe(true); + + expect(sameOrigin('')).toBe(true); + expect(sameOrigin('blob:https://www.bing.com/09f36686-e57a-420f-9004-918548219b75')).toBe(true); + + // relative URL is same origin for sure + expect(sameOrigin('/foo')).toBe(true); + expect(sameOrigin('foo')).toBe(true); + + // empty string is considered as relative, and should be true + expect(sameOrigin('')).toBe(true); + expect(sameOrigin(null)).toBe(true); + expect(sameOrigin(undefined)).toBe(true); + + expect(sameOrigin('HTTPS://somewhere.com')).toBe(true); + + // different domain + expect(sameOrigin('httpS://www.somewhere.com')).toBe(false); + + // different protocol + expect(sameOrigin('HTTP://somewhere.com')).toBe(false); + expect(sameOrigin('file:///c:/temp/foo.html')).toBe(false); + + // file url + jest.spyOn(window, 'location', 'get').mockReturnValue({ + protocol: 'file:', + host: '' + } as any); + expect(sameOrigin('file:///C:/Temp/abc.html')).toBe(true); + expect(sameOrigin('HTTP://somewhere.com')).toBe(false); + + // relative URL (for file URL) is same origin as well + expect(sameOrigin('/foo')).toBe(true); + expect(sameOrigin('foo')).toBe(true); + + // edge case + expect(sameOrigin('://foo')).toBe(true); + }); + + describe('requests parameters', () => { + let fetch: FetchMock; + + beforeEach(() => { + fetch = setupFetchMock(); + }); + + afterEach(() => { + destroyFetchMock(); + }); + + test('should be provided to fetch API in getArrayBuffer function', (done) => { + getArrayBuffer({url: 'http://example.com/test-params.json', cache: 'force-cache', headers: {'Authorization': 'Bearer 123'}}, () => { + expect(fetch).toHaveBeenCalledTimes(1); + expect(fetch).toHaveBeenCalledWith(expect.objectContaining({url: 'http://example.com/test-params.json', method: 'GET', cache: 'force-cache'})); + expect((fetch.mock.calls[0][0] as RequestMock).headers.get('Authorization')).toBe('Bearer 123'); + + done(); + }); + }); + + test('should be provided to fetch API in getJSON function', (done) => { + getJSON({url: 'http://example.com/test-params.json', cache: 'force-cache', headers: {'Authorization': 'Bearer 123'}}, () => { + expect(fetch).toHaveBeenCalledTimes(1); + expect(fetch).toHaveBeenCalledWith(expect.objectContaining({url: 'http://example.com/test-params.json', method: 'GET', cache: 'force-cache'})); + expect((fetch.mock.calls[0][0] as RequestMock).headers.get('Authorization')).toBe('Bearer 123'); + + done(); + }); + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/ajax.ts b/web/libraries/maplibre-gl/src/util/ajax.ts new file mode 100644 index 00000000..8119d208 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/ajax.ts @@ -0,0 +1,317 @@ +import {extend, warnOnce, isWorker} from './util'; +import {config} from './config'; + +import type {Callback} from '../types/callback'; +import type {Cancelable} from '../types/cancelable'; + +/** + * A `RequestParameters` object to be returned from Map.options.transformRequest callbacks. + * @example + * ```ts + * // use transformRequest to modify requests that begin with `http://myHost` + * transformRequest: function(url, resourceType) { + * if (resourceType === 'Source' && url.indexOf('http://myHost') > -1) { + * return { + * url: url.replace('http', 'https'), + * headers: { 'my-custom-header': true }, + * credentials: 'include' // Include cookies for cross-origin requests + * } + * } + * } + * ``` + */ +export type RequestParameters = { + /** + * The URL to be requested. + */ + url: string; + /** + * The headers to be sent with the request. + */ + headers?: any; + /** + * Request method `'GET' | 'POST' | 'PUT'`. + */ + method?: 'GET' | 'POST' | 'PUT'; + /** + * Request body. + */ + body?: string; + /** + * Response body type to be returned `'string' | 'json' | 'arrayBuffer'`. + */ + type?: 'string' | 'json' | 'arrayBuffer' | 'image'; + /** + * `'same-origin'|'include'` Use 'include' to send cookies with cross-origin requests. + */ + credentials?: 'same-origin' | 'include'; + /** + * If `true`, Resource Timing API information will be collected for these transformed requests and returned in a resourceTiming property of relevant data events. + */ + collectResourceTiming?: boolean; + /** + * Parameters supported only by browser fetch API. Property of the Request interface contains the cache mode of the request. It controls how the request will interact with the browser's HTTP cache. (https://developer.mozilla.org/en-US/docs/Web/API/Request/cache) + */ + cache?: RequestCache; +}; + +/** + * The response callback used in various places + */ +export type ResponseCallback = ( + error?: Error | null, + data?: T | null, + cacheControl?: string | null, + expires?: string | null +) => void; + +/** + * An error thrown when a HTTP request results in an error response. + */ +export class AJAXError extends Error { + /** + * The response's HTTP status code. + */ + status: number; + + /** + * The response's HTTP status text. + */ + statusText: string; + + /** + * The request's URL. + */ + url: string; + + /** + * The response's body. + */ + body: Blob; + + /** + * @param status - The response's HTTP status code. + * @param statusText - The response's HTTP status text. + * @param url - The request's URL. + * @param body - The response's body. + */ + constructor(status: number, statusText: string, url: string, body: Blob) { + super(`AJAXError: ${statusText} (${status}): ${url}`); + this.status = status; + this.statusText = statusText; + this.url = url; + this.body = body; + } +} + +// Ensure that we're sending the correct referrer from blob URL worker bundles. +// For files loaded from the local file system, `location.origin` will be set +// to the string(!) "null" (Firefox), or "file://" (Chrome, Safari, Edge, IE), +// and we will set an empty referrer. Otherwise, we're using the document's URL. +/* global self */ +export const getReferrer = isWorker() ? + () => (self as any).worker && (self as any).worker.referrer : + () => (window.location.protocol === 'blob:' ? window.parent : window).location.href; + +export const getProtocolAction = url => config.REGISTERED_PROTOCOLS[url.substring(0, url.indexOf('://'))]; + +// Determines whether a URL is a file:// URL. This is obviously the case if it begins +// with file://. Relative URLs are also file:// URLs iff the original document was loaded +// via a file:// URL. +const isFileURL = url => /^file:/.test(url) || (/^file:/.test(getReferrer()) && !/^\w+:/.test(url)); + +function makeFetchRequest(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable { + const controller = new AbortController(); + const request = new Request(requestParameters.url, { + method: requestParameters.method || 'GET', + body: requestParameters.body, + credentials: requestParameters.credentials, + headers: requestParameters.headers, + cache: requestParameters.cache, + referrer: getReferrer(), + signal: controller.signal + }); + let complete = false; + let aborted = false; + + if (requestParameters.type === 'json') { + request.headers.set('Accept', 'application/json'); + } + + const validateOrFetch = (err, cachedResponse?, responseIsFresh?) => { + if (aborted) return; + + if (err) { + // Do fetch in case of cache error. + // HTTP pages in Edge trigger a security error that can be ignored. + if (err.message !== 'SecurityError') { + warnOnce(err); + } + } + + if (cachedResponse && responseIsFresh) { + return finishRequest(cachedResponse); + } + + if (cachedResponse) { + // We can't do revalidation with 'If-None-Match' because then the + // request doesn't have simple cors headers. + } + + fetch(request).then(response => { + if (response.ok) { + return finishRequest(response); + + } else { + return response.blob().then(body => callback(new AJAXError(response.status, response.statusText, requestParameters.url, body))); + } + }).catch(error => { + if (error.code === 20) { + // silence expected AbortError + return; + } + callback(new Error(error.message)); + }); + }; + + const finishRequest = (response) => { + ( + (requestParameters.type === 'arrayBuffer' || requestParameters.type === 'image') ? response.arrayBuffer() : + requestParameters.type === 'json' ? response.json() : + response.text() + ).then(result => { + if (aborted) return; + complete = true; + callback(null, result, response.headers.get('Cache-Control'), response.headers.get('Expires')); + }).catch(err => { + if (!aborted) callback(new Error(err.message)); + }); + }; + + validateOrFetch(null, null); + + return {cancel: () => { + aborted = true; + if (!complete) controller.abort(); + }}; +} + +function makeXMLHttpRequest(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable { + const xhr: XMLHttpRequest = new XMLHttpRequest(); + + xhr.open(requestParameters.method || 'GET', requestParameters.url, true); + if (requestParameters.type === 'arrayBuffer' || requestParameters.type === 'image') { + xhr.responseType = 'arraybuffer'; + } + for (const k in requestParameters.headers) { + xhr.setRequestHeader(k, requestParameters.headers[k]); + } + if (requestParameters.type === 'json') { + xhr.responseType = 'text'; + xhr.setRequestHeader('Accept', 'application/json'); + } + xhr.withCredentials = requestParameters.credentials === 'include'; + xhr.onerror = () => { + callback(new Error(xhr.statusText)); + }; + xhr.onload = () => { + if (((xhr.status >= 200 && xhr.status < 300) || xhr.status === 0) && xhr.response !== null) { + let data: unknown = xhr.response; + if (requestParameters.type === 'json') { + // We're manually parsing JSON here to get better error messages. + try { + data = JSON.parse(xhr.response); + } catch (err) { + return callback(err); + } + } + callback(null, data, xhr.getResponseHeader('Cache-Control'), xhr.getResponseHeader('Expires')); + } else { + const body = new Blob([xhr.response], {type: xhr.getResponseHeader('Content-Type')}); + callback(new AJAXError(xhr.status, xhr.statusText, requestParameters.url, body)); + } + }; + xhr.send(requestParameters.body); + return {cancel: () => xhr.abort()}; +} + +export const makeRequest = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable { + // We're trying to use the Fetch API if possible. However, in some situations we can't use it: + // - IE11 doesn't support it at all. In this case, we dispatch the request to the main thread so + // that we can get an accruate referrer header. + // - Safari exposes window.AbortController, but it doesn't work actually abort any requests in + // some versions (see https://bugs.webkit.org/show_bug.cgi?id=174980#c2) + // - Requests for resources with the file:// URI scheme don't work with the Fetch API either. In + // this case we unconditionally use XHR on the current thread since referrers don't matter. + if (/:\/\//.test(requestParameters.url) && !(/^https?:|^file:/.test(requestParameters.url))) { + if (isWorker() && (self as any).worker && (self as any).worker.actor) { + return (self as any).worker.actor.send('getResource', requestParameters, callback); + } + if (!isWorker()) { + const action = getProtocolAction(requestParameters.url) || makeFetchRequest; + return action(requestParameters, callback); + } + } + if (!isFileURL(requestParameters.url)) { + if (fetch && Request && AbortController && Object.prototype.hasOwnProperty.call(Request.prototype, 'signal')) { + return makeFetchRequest(requestParameters, callback); + } + if (isWorker() && (self as any).worker && (self as any).worker.actor) { + const queueOnMainThread = true; + return (self as any).worker.actor.send('getResource', requestParameters, callback, undefined, queueOnMainThread); + } + } + return makeXMLHttpRequest(requestParameters, callback); +}; + +export const getJSON = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable { + return makeRequest(extend(requestParameters, {type: 'json'}), callback); +}; + +export const getArrayBuffer = function( + requestParameters: RequestParameters, + callback: ResponseCallback +): Cancelable { + return makeRequest(extend(requestParameters, {type: 'arrayBuffer'}), callback); +}; + +export const postData = function(requestParameters: RequestParameters, callback: ResponseCallback): Cancelable { + return makeRequest(extend(requestParameters, {method: 'POST'}), callback); +}; + +export function sameOrigin(inComingUrl: string) { + // URL class should be available everywhere + // https://developer.mozilla.org/en-US/docs/Web/API/URL + // In addtion, a relative URL "/foo" or "./foo" will throw exception in its ctor, + // try-catch is expansive so just use a heuristic check to avoid it + // also check data URL + if (!inComingUrl || + inComingUrl.indexOf('://') <= 0 || // relative URL + inComingUrl.indexOf('data:image/') === 0 || // data image URL + inComingUrl.indexOf('blob:') === 0) { // blob + return true; + } + const urlObj = new URL(inComingUrl); + const locationObj = window.location; + return urlObj.protocol === locationObj.protocol && urlObj.host === locationObj.host; +} +/** + * A type used to store the tile's expiration date and cache control definition + */ +export type ExpiryData = {cacheControl?: string | null; expires?: Date | string | null}; +export const getVideo = function(urls: Array, callback: Callback): Cancelable { + const video: HTMLVideoElement = window.document.createElement('video'); + video.muted = true; + video.onloadstart = function() { + callback(null, video); + }; + for (let i = 0; i < urls.length; i++) { + const s: HTMLSourceElement = window.document.createElement('source'); + if (!sameOrigin(urls[i])) { + video.crossOrigin = 'Anonymous'; + } + s.src = urls[i]; + video.appendChild(s); + } + return {cancel: () => {}}; +}; diff --git a/web/libraries/maplibre-gl/src/util/browser.test.ts b/web/libraries/maplibre-gl/src/util/browser.test.ts new file mode 100644 index 00000000..55dcdcff --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/browser.test.ts @@ -0,0 +1,26 @@ +import {browser} from './browser'; + +describe('browser', () => { + test('frame', done => { + const id = browser.frame(() => { + expect(id).toBeTruthy(); + done(); + }); + }); + + test('now', () => { + expect(typeof browser.now()).toBe('number'); + }); + + test('frame', done => { + const frame = browser.frame(() => { + done('test failed: browser.frame callback was called'); + }); + frame.cancel(); + done(); + }); + + test('hardwareConcurrency', () => { + expect(typeof browser.hardwareConcurrency).toBe('number'); + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/browser.ts b/web/libraries/maplibre-gl/src/util/browser.ts new file mode 100755 index 00000000..6ba3b6f1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/browser.ts @@ -0,0 +1,58 @@ +import type {Cancelable} from '../types/cancelable'; + +const now = typeof performance !== 'undefined' && performance && performance.now ? + performance.now.bind(performance) : + Date.now.bind(Date); + +let linkEl; + +let reducedMotionQuery: MediaQueryList; + +/** */ +export const browser = { + /** + * Provides a function that outputs milliseconds: either performance.now() + * or a fallback to Date.now() + */ + now, + + frame(fn: (paintStartTimestamp: number) => void): Cancelable { + const frame = requestAnimationFrame(fn); + return {cancel: () => cancelAnimationFrame(frame)}; + }, + + getImageData(img: HTMLImageElement | ImageBitmap, padding: number = 0): ImageData { + const context = this.getImageCanvasContext(img); + return context.getImageData(-padding, -padding, img.width as number + 2 * padding, img.height as number + 2 * padding); + }, + + getImageCanvasContext(img: HTMLImageElement | ImageBitmap): CanvasRenderingContext2D { + const canvas = window.document.createElement('canvas') as HTMLCanvasElement; + const context = canvas.getContext('2d', {willReadFrequently: true}); + if (!context) { + throw new Error('failed to create canvas 2d context'); + } + canvas.width = img.width as number; + canvas.height = img.height as number; + context.drawImage(img, 0, 0, img.width as number, img.height as number); + return context; + }, + + resolveURL(path: string) { + if (!linkEl) linkEl = document.createElement('a'); + linkEl.href = path; + return linkEl.href; + }, + + hardwareConcurrency: typeof navigator !== 'undefined' && navigator.hardwareConcurrency || 4, + + get prefersReducedMotion(): boolean { + // In case your test crashes when checking matchMedia, call setMatchMedia from 'src/util/test/util' + if (!matchMedia) return false; + //Lazily initialize media query + if (reducedMotionQuery == null) { + reducedMotionQuery = matchMedia('(prefers-reduced-motion: reduce)'); + } + return reducedMotionQuery.matches; + }, +}; diff --git a/web/libraries/maplibre-gl/src/util/classify_rings.test.ts b/web/libraries/maplibre-gl/src/util/classify_rings.test.ts new file mode 100644 index 00000000..83294bff --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/classify_rings.test.ts @@ -0,0 +1,147 @@ +import fs from 'fs'; +import path from 'path'; +import Protobuf from 'pbf'; +import {VectorTile} from '@mapbox/vector-tile'; +import {classifyRings} from './classify_rings'; +import Point from '@mapbox/point-geometry'; + +// Load a fill feature from fixture tile. +const vt = new VectorTile(new Protobuf(fs.readFileSync(path.resolve(__dirname, '../../test/unit/assets/mbsv5-6-18-23.vector.pbf')))); +const feature = vt.layers.water.feature(0); + +describe('classifyRings', () => { + test('classified.length', () => { + let geometry; + let classified; + + geometry = [ + [ + {x: 0, y: 0}, + {x: 0, y: 40}, + {x: 40, y: 40}, + {x: 40, y: 0}, + {x: 0, y: 0} + ] + ]; + classified = classifyRings(geometry, undefined); + expect(classified).toHaveLength(1); + expect(classified[0]).toHaveLength(1); + + geometry = [ + [ + {x: 0, y: 0}, + {x: 0, y: 40}, + {x: 40, y: 40}, + {x: 40, y: 0}, + {x: 0, y: 0} + ], + [ + {x: 60, y: 0}, + {x: 60, y: 40}, + {x: 100, y: 40}, + {x: 100, y: 0}, + {x: 60, y: 0} + ] + ]; + classified = classifyRings(geometry, undefined); + expect(classified).toHaveLength(2); + expect(classified[0]).toHaveLength(1); + expect(classified[1]).toHaveLength(1); + + geometry = [ + [ + {x: 0, y: 0}, + {x: 0, y: 40}, + {x: 40, y: 40}, + {x: 40, y: 0}, + {x: 0, y: 0} + ], + [ + {x: 10, y: 10}, + {x: 20, y: 10}, + {x: 20, y: 20}, + {x: 10, y: 10} + ] + ]; + classified = classifyRings(geometry, undefined); + expect(classified).toHaveLength(1); + expect(classified[0]).toHaveLength(2); + + geometry = feature.loadGeometry(); + classified = classifyRings(geometry, undefined); + expect(classified).toHaveLength(2); + expect(classified[0]).toHaveLength(1); + expect(classified[1]).toHaveLength(10); + }); +}); + +describe('classifyRings + maxRings', () => { + + function createGeometry(options?) { + const geometry = [ + // Outer ring, area = 3200 + [{x: 0, y: 0}, {x: 0, y: 40}, {x: 40, y: 40}, {x: 40, y: 0}, {x: 0, y: 0}], + // Inner ring, area = 100 + [{x: 30, y: 30}, {x: 32, y: 30}, {x: 32, y: 32}, {x: 30, y: 30}], + // Inner ring, area = 4 + [{x: 10, y: 10}, {x: 20, y: 10}, {x: 20, y: 20}, {x: 10, y: 10}] + ] as Point[][]; + if (options && options.reverse) { + geometry[0].reverse(); + geometry[1].reverse(); + geometry[2].reverse(); + } + return geometry; + } + + test('maxRings=undefined', () => { + const geometry = sortRings(classifyRings(createGeometry(), undefined)); + expect(geometry).toHaveLength(1); + expect(geometry[0]).toHaveLength(3); + expect(geometry[0][0].area).toBe(3200); + expect(geometry[0][1].area).toBe(100); + expect(geometry[0][2].area).toBe(4); + + }); + + test('maxRings=2', () => { + const geometry = sortRings(classifyRings(createGeometry(), 2)); + expect(geometry).toHaveLength(1); + expect(geometry[0]).toHaveLength(2); + expect(geometry[0][0].area).toBe(3200); + expect(geometry[0][1].area).toBe(100); + + }); + + test('maxRings=2, reversed geometry', () => { + const geometry = sortRings(classifyRings(createGeometry({reverse: true}), 2)); + expect(geometry).toHaveLength(1); + expect(geometry[0]).toHaveLength(2); + expect(geometry[0][0].area).toBe(3200); + expect(geometry[0][1].area).toBe(100); + + }); + + test('maxRings=5, geometry from fixture', () => { + const geometry = sortRings(classifyRings(feature.loadGeometry(), 5)); + expect(geometry).toHaveLength(2); + expect(geometry[0]).toHaveLength(1); + expect(geometry[1]).toHaveLength(5); + + const areas = geometry[1].map((ring) => { return ring.area; }); + expect(areas).toEqual([2763951, 21600, 8298, 4758, 3411]); + + }); + +}); + +function sortRings(geometry) { + for (let i = 0; i < geometry.length; i++) { + geometry[i] = geometry[i].sort(compareAreas); + } + return geometry; +} + +function compareAreas(a, b) { + return b.area - a.area; +} diff --git a/web/libraries/maplibre-gl/src/util/classify_rings.ts b/web/libraries/maplibre-gl/src/util/classify_rings.ts new file mode 100644 index 00000000..f8fe2621 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/classify_rings.ts @@ -0,0 +1,50 @@ +import quickselect from 'quickselect'; + +import {calculateSignedArea} from './util'; + +import type Point from '@mapbox/point-geometry'; + +// classifies an array of rings into polygons with outer rings and holes +export function classifyRings(rings: Array>, maxRings: number) { + const len = rings.length; + + if (len <= 1) return [rings]; + + const polygons = []; + let polygon, + ccw; + + for (let i = 0; i < len; i++) { + const area = calculateSignedArea(rings[i]); + if (area === 0) continue; + + (rings[i] as any).area = Math.abs(area); + + if (ccw === undefined) ccw = area < 0; + + if (ccw === area < 0) { + if (polygon) polygons.push(polygon); + polygon = [rings[i]]; + + } else { + (polygon as any).push(rings[i]); + } + } + if (polygon) polygons.push(polygon); + + // Earcut performance degrades with the # of rings in a polygon. For this + // reason, we limit strip out all but the `maxRings` largest rings. + if (maxRings > 1) { + for (let j = 0; j < polygons.length; j++) { + if (polygons[j].length <= maxRings) continue; + quickselect(polygons[j], maxRings, 1, polygons[j].length - 1, compareAreas); + polygons[j] = polygons[j].slice(0, maxRings); + } + } + + return polygons; +} + +function compareAreas(a, b) { + return b.area - a.area; +} diff --git a/web/libraries/maplibre-gl/src/util/color_ramp.test.ts b/web/libraries/maplibre-gl/src/util/color_ramp.test.ts new file mode 100644 index 00000000..17083b96 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/color_ramp.test.ts @@ -0,0 +1,105 @@ + +import {renderColorRamp} from './color_ramp'; +import {createPropertyExpression, StylePropertyExpression, StylePropertySpecification} from '@maplibre/maplibre-gl-style-spec'; + +const spec = { + 'function': true, + 'property-function': true, + 'type': 'color' +} as any as StylePropertySpecification; + +function pixelAt(image, i) { + return image.data.slice(i * 4, (i + 1) * 4); +} + +function nearlyEquals(a, b) { + // we're actually looking for colors that are _almost_ equal, but don't + // expect exact equal since 256 px need to represent a range from [0, 1] + // (inclusive) -- the first and last pixel should be exact, the halfway + // pixel may not be + return a.every((e, i) => Math.abs(e - b[i]) <= 3); +} + +describe('renderColorRamp', () => { + test('renderColorRamp linear', () => { + + const expression = createPropertyExpression([ + 'interpolate', + ['linear'], + ['line-progress'], + 0, 'rgba(0,0,255,0)', + 0.25, 'white', + 0.5, 'rgba(0,255,255,0.5)', + 0.75, 'black', + 1, 'red' + ], spec).value as StylePropertyExpression; + + const ramp = renderColorRamp({expression, evaluationKey: 'lineProgress'}); + + expect(ramp.width).toBe(256); + expect(ramp.height).toBe(1); + + expect(pixelAt(ramp, 0)[3]).toBe(0); + expect(nearlyEquals(pixelAt(ramp, 63), [255, 255, 255, 255])).toBeTruthy(); + expect(nearlyEquals(pixelAt(ramp, 127), [0, 255, 255, 127])).toBeTruthy(); + expect(nearlyEquals(pixelAt(ramp, 191), [0, 0, 0, 255])).toBeTruthy(); + expect(nearlyEquals(pixelAt(ramp, 255), [255, 0, 0, 255])).toBeTruthy(); + }); + + test('renderColorRamp step', () => { + + const expression = createPropertyExpression([ + 'step', + ['line-progress'], + 'rgba(0, 0, 255, 0.1)', + 0.1, 'red', + 0.2, 'yellow', + 0.3, 'white', + 0.5, 'black', + 1, 'black' + ], spec).value as StylePropertyExpression; + + const ramp = renderColorRamp({expression, evaluationKey: 'lineProgress', resolution: 512}); + + expect(ramp.width).toBe(512); + expect(ramp.height).toBe(1); + + expect(pixelAt(ramp, 0)[3]).toBe(25); + expect(nearlyEquals(pixelAt(ramp, 50), [0, 0, 255, 25])).toBeTruthy(); + expect(nearlyEquals(pixelAt(ramp, 53), [255, 0, 0, 255])).toBeTruthy(); + expect(nearlyEquals(pixelAt(ramp, 103), [255, 255, 0, 255])).toBeTruthy(); + expect(nearlyEquals(pixelAt(ramp, 160), [255, 255, 255, 255])).toBeTruthy(); + expect(nearlyEquals(pixelAt(ramp, 256), [0, 0, 0, 255])).toBeTruthy(); + + }); + + test('renderColorRamp usePlacement', () => { + + const expression = createPropertyExpression([ + 'step', + ['line-progress'], + 'rgba(255, 0, 0, 0.5)', + 0.1, 'black', + 0.2, 'red', + 0.3, 'blue', + 0.5, 'white', + 1, 'white' + ], spec).value as StylePropertyExpression; + + const ramp = renderColorRamp({expression, evaluationKey: 'lineProgress', resolution: 512}); + + expect(ramp.width).toBe(512); + expect(ramp.height).toBe(1); + + renderColorRamp({expression, evaluationKey: 'lineProgress', resolution: 512, image: ramp}); + + expect(pixelAt(ramp, 0)[3]).toBe(127); + expect(nearlyEquals(pixelAt(ramp, 50), [255, 0, 0, 127])).toBeTruthy(); + expect(nearlyEquals(pixelAt(ramp, 53), [0, 0, 0, 255])).toBeTruthy(); + expect(nearlyEquals(pixelAt(ramp, 103), [255, 0, 0, 255])).toBeTruthy(); + expect(nearlyEquals(pixelAt(ramp, 160), [0, 0, 255, 255])).toBeTruthy(); + expect(nearlyEquals(pixelAt(ramp, 256), [255, 255, 255, 255])).toBeTruthy(); + + }); + +}); diff --git a/web/libraries/maplibre-gl/src/util/color_ramp.ts b/web/libraries/maplibre-gl/src/util/color_ramp.ts new file mode 100644 index 00000000..48243711 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/color_ramp.ts @@ -0,0 +1,56 @@ +import {RGBAImage} from './image'; +import {isPowerOfTwo} from './util'; + +import type {StylePropertyExpression} from '@maplibre/maplibre-gl-style-spec'; + +export type ColorRampParams = { + expression: StylePropertyExpression; + evaluationKey: string; + resolution?: number; + image?: RGBAImage; + clips?: Array; +}; + +/** + * Given an expression that should evaluate to a color ramp, + * return a RGBA image representing that ramp expression. + */ +export function renderColorRamp(params: ColorRampParams): RGBAImage { + const evaluationGlobals = {}; + const width = params.resolution || 256; + const height = params.clips ? params.clips.length : 1; + const image = params.image || new RGBAImage({width, height}); + + if (!isPowerOfTwo(width)) throw new Error(`width is not a power of 2 - ${width}`); + + const renderPixel = (stride, index, progress) => { + evaluationGlobals[params.evaluationKey] = progress; + const pxColor = params.expression.evaluate(evaluationGlobals as any); + // the colors are being unpremultiplied because Color uses + // premultiplied values, and the Texture class expects unpremultiplied ones + image.data[stride + index + 0] = Math.floor(pxColor.r * 255 / pxColor.a); + image.data[stride + index + 1] = Math.floor(pxColor.g * 255 / pxColor.a); + image.data[stride + index + 2] = Math.floor(pxColor.b * 255 / pxColor.a); + image.data[stride + index + 3] = Math.floor(pxColor.a * 255); + }; + + if (!params.clips) { + for (let i = 0, j = 0; i < width; i++, j += 4) { + const progress = i / (width - 1); + + renderPixel(0, j, progress); + } + } else { + for (let clip = 0, stride = 0; clip < height; ++clip, stride += width * 4) { + for (let i = 0, j = 0; i < width; i++, j += 4) { + // Remap progress between clips + const progress = i / (width - 1); + const {start, end} = params.clips[clip]; + const evaluationProgress = start * (1 - progress) + end * progress; + renderPixel(stride, j, evaluationProgress); + } + } + } + + return image; +} diff --git a/web/libraries/maplibre-gl/src/util/config.ts b/web/libraries/maplibre-gl/src/util/config.ts new file mode 100644 index 00000000..152122bc --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/config.ts @@ -0,0 +1,23 @@ +import type {Cancelable} from '../types/cancelable'; +import type {RequestParameters, ResponseCallback} from './ajax'; + +/** + * This is a global config object used to store the configuration + * It is available in the workers as well. + * Only serializable data should be stored in it. + */ +type Config = { + MAX_PARALLEL_IMAGE_REQUESTS: number; + MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME: number; + MAX_TILE_CACHE_ZOOM_LEVELS: number; + REGISTERED_PROTOCOLS: {[x: string]: (requestParameters: RequestParameters, callback: ResponseCallback) => Cancelable}; + WORKER_URL: string; +}; + +export const config: Config = { + MAX_PARALLEL_IMAGE_REQUESTS: 16, + MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME: 8, + MAX_TILE_CACHE_ZOOM_LEVELS: 5, + REGISTERED_PROTOCOLS: {}, + WORKER_URL: '' +}; diff --git a/web/libraries/maplibre-gl/src/util/debug.ts b/web/libraries/maplibre-gl/src/util/debug.ts new file mode 100644 index 00000000..6b20c906 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/debug.ts @@ -0,0 +1,24 @@ +import {extend} from './util'; + +/** + * This is a private namespace for utility functions that will get automatically stripped + * out in production builds. + */ +export const Debug = { + extend(dest: any, ...sources: Array): any { + return extend(dest, ...sources); + }, + + run(fn: () => any) { + fn(); + }, + + logToElement(message: string, overwrite: boolean = false, id: string = 'log') { + const el = window.document.getElementById(id); + if (el) { + if (overwrite) el.innerHTML = ''; + el.innerHTML += `
${message}`; + } + + } +}; diff --git a/web/libraries/maplibre-gl/src/util/dictionary_coder.ts b/web/libraries/maplibre-gl/src/util/dictionary_coder.ts new file mode 100644 index 00000000..bc3000cf --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/dictionary_coder.ts @@ -0,0 +1,23 @@ +export class DictionaryCoder { + _stringToNumber: {[_: string]: number}; + _numberToString: Array; + + constructor(strings: Array) { + this._stringToNumber = {}; + this._numberToString = []; + for (let i = 0; i < strings.length; i++) { + const string = strings[i]; + this._stringToNumber[string] = i; + this._numberToString[i] = string; + } + } + + encode(string: string) { + return this._stringToNumber[string]; + } + + decode(n: number) { + if (n >= this._numberToString.length) throw new Error(`Out of bounds. Index requested n=${n} can't be >= this._numberToString.length ${this._numberToString.length}`); + return this._numberToString[n]; + } +} diff --git a/web/libraries/maplibre-gl/src/util/dispatcher.test.ts b/web/libraries/maplibre-gl/src/util/dispatcher.test.ts new file mode 100644 index 00000000..c1e78a0d --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/dispatcher.test.ts @@ -0,0 +1,76 @@ +import {Actor} from './actor'; +import {Dispatcher} from './dispatcher'; +import {workerFactory} from './web_worker'; +import {WorkerPool} from './worker_pool'; + +describe('Dispatcher', () => { + test('requests and releases workers from pool', () => { + const workers = [workerFactory(), workerFactory()]; + const mapId = 1; + const releaseCalled = []; + const workerPool = { + acquire () { + return workers; + }, + release (id) { + releaseCalled.push(id); + } + } as any as WorkerPool; + + const dispatcher = new Dispatcher(workerPool, {} as any, mapId); + expect(dispatcher.actors.map((actor) => { return actor.target; })).toEqual(workers); + dispatcher.remove(); + expect(dispatcher.actors).toHaveLength(0); + expect(releaseCalled).toEqual([mapId]); + + }); + + test('reuse workers till map is disposed', () => { + let workers = null; + const mapId = 1; + const releaseCalled = []; + const workerPool = { + acquire () { + if (!workers) { + workers = [workerFactory(), workerFactory()]; + } + return workers; + }, + release (id) { + releaseCalled.push(id); + workers = null; + } + } as any as WorkerPool; + + let dispatcher = new Dispatcher(workerPool, {} as any, mapId); + expect(dispatcher.actors.map((actor) => { return actor.target; })).toEqual(workers); + + // Remove dispatcher, but map is not disposed (During style change) + dispatcher.remove(false); + expect(dispatcher.actors).toHaveLength(0); + expect(releaseCalled).toHaveLength(0); + + // Create new instance of dispatcher + dispatcher = new Dispatcher(workerPool, {} as any, mapId); + expect(dispatcher.actors.map((actor) => { return actor.target; })).toEqual(workers); + dispatcher.remove(true); // mapRemoved = true + expect(dispatcher.actors).toHaveLength(0); + expect(releaseCalled).toEqual([mapId]); + + }); + + test('#remove destroys actors', () => { + const actorsRemoved = []; + const mapId = 1; + const spy = jest.fn().mockImplementation(() => { actorsRemoved.push(this); }); + Actor.prototype.remove = spy; + WorkerPool.workerCount = 4; + + const workerPool = new WorkerPool(); + const dispatcher = new Dispatcher(workerPool, {} as any, mapId); + dispatcher.remove(); + expect(actorsRemoved).toHaveLength(4); + }); + +}); + diff --git a/web/libraries/maplibre-gl/src/util/dispatcher.ts b/web/libraries/maplibre-gl/src/util/dispatcher.ts new file mode 100644 index 00000000..fd9b15dc --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/dispatcher.ts @@ -0,0 +1,55 @@ +import {asyncAll} from './util'; +import {Actor, GlyphsProvider, MessageType} from './actor'; + +import type {WorkerPool} from './worker_pool'; +import type {WorkerSource} from '../source/worker_source'; /* eslint-disable-line */ // this is used for the docs' import +/** + * Responsible for sending messages from a {@link Source} to an associated + * {@link WorkerSource}. + */ +export class Dispatcher { + workerPool: WorkerPool; + actors: Array; + currentActor: number; + id: string | number; + + constructor(workerPool: WorkerPool, parent: GlyphsProvider, mapId: string | number) { + this.workerPool = workerPool; + this.actors = []; + this.currentActor = 0; + this.id = mapId; + const workers = this.workerPool.acquire(mapId); + for (let i = 0; i < workers.length; i++) { + const worker = workers[i]; + const actor = new Actor(worker, parent, mapId); + actor.name = `Worker ${i}`; + this.actors.push(actor); + } + if (!this.actors.length) throw new Error('No actors found'); + } + + /** + * Broadcast a message to all Workers. + */ + broadcast(type: MessageType, data: unknown, cb?: (...args: any[]) => any) { + cb = cb || function () {}; + asyncAll(this.actors, (actor, done) => { + actor.send(type, data, done); + }, cb); + } + + /** + * Acquires an actor to dispatch messages to. The actors are distributed in round-robin fashion. + * @returns An actor object backed by a web worker for processing messages. + */ + getActor(): Actor { + this.currentActor = (this.currentActor + 1) % this.actors.length; + return this.actors[this.currentActor]; + } + + remove(mapRemoved: boolean = true) { + this.actors.forEach((actor) => { actor.remove(); }); + this.actors = []; + if (mapRemoved) this.workerPool.release(this.id); + } +} diff --git a/web/libraries/maplibre-gl/src/util/dom.ts b/web/libraries/maplibre-gl/src/util/dom.ts new file mode 100644 index 00000000..90653a68 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/dom.ts @@ -0,0 +1,116 @@ +import Point from '@mapbox/point-geometry'; + +export class DOM { + private static readonly docStyle = typeof window !== 'undefined' && window.document && window.document.documentElement.style; + + private static userSelect: string; + + private static selectProp = DOM.testProp(['userSelect', 'MozUserSelect', 'WebkitUserSelect', 'msUserSelect']); + + private static transformProp = DOM.testProp(['transform', 'WebkitTransform']); + + private static testProp(props: string[]): string { + if (!DOM.docStyle) return props[0]; + for (let i = 0; i < props.length; i++) { + if (props[i] in DOM.docStyle) { + return props[i]; + } + } + return props[0]; + } + + public static create(tagName: K, className?: string, container?: HTMLElement): HTMLElementTagNameMap[K] { + const el = window.document.createElement(tagName); + if (className !== undefined) el.className = className; + if (container) container.appendChild(el); + return el; + } + + public static createNS(namespaceURI: string, tagName: string) { + const el = window.document.createElementNS(namespaceURI, tagName); + return el; + } + + public static disableDrag() { + if (DOM.docStyle && DOM.selectProp) { + DOM.userSelect = DOM.docStyle[DOM.selectProp]; + DOM.docStyle[DOM.selectProp] = 'none'; + } + } + + public static enableDrag() { + if (DOM.docStyle && DOM.selectProp) { + DOM.docStyle[DOM.selectProp] = DOM.userSelect; + } + } + + public static setTransform(el: HTMLElement, value: string) { + el.style[DOM.transformProp] = value; + } + + public static addEventListener(target: HTMLElement | Window | Document, type: string, callback: EventListenerOrEventListenerObject, options: { + passive?: boolean; + capture?: boolean; + } = {}) { + if ('passive' in options) { + target.addEventListener(type, callback, options); + } else { + target.addEventListener(type, callback, options.capture); + } + } + + public static removeEventListener(target: HTMLElement | Window | Document, type: string, callback: EventListenerOrEventListenerObject, options: { + passive?: boolean; + capture?: boolean; + } = {}) { + if ('passive' in options) { + target.removeEventListener(type, callback, options); + } else { + target.removeEventListener(type, callback, options.capture); + } + } + + // Suppress the next click, but only if it's immediate. + private static suppressClickInternal(e) { + e.preventDefault(); + e.stopPropagation(); + window.removeEventListener('click', DOM.suppressClickInternal, true); + } + + public static suppressClick() { + window.addEventListener('click', DOM.suppressClickInternal, true); + window.setTimeout(() => { + window.removeEventListener('click', DOM.suppressClickInternal, true); + }, 0); + } + + public static mousePos(el: HTMLElement, e: MouseEvent | Touch) { + const rect = el.getBoundingClientRect(); + return new Point( + e.clientX - rect.left - el.clientLeft, + e.clientY - rect.top - el.clientTop + ); + } + + public static touchPos(el: HTMLElement, touches: TouchList) { + const rect = el.getBoundingClientRect(); + const points: Point[] = []; + for (let i = 0; i < touches.length; i++) { + points.push(new Point( + touches[i].clientX - rect.left - el.clientLeft, + touches[i].clientY - rect.top - el.clientTop + )); + } + return points; + } + + public static mouseButton(e: MouseEvent) { + return e.button; + } + + public static remove(node: HTMLElement) { + if (node.parentNode) { + node.parentNode.removeChild(node); + } + } +} diff --git a/web/libraries/maplibre-gl/src/util/evented.test.ts b/web/libraries/maplibre-gl/src/util/evented.test.ts new file mode 100644 index 00000000..90883ec8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/evented.test.ts @@ -0,0 +1,232 @@ +import {Event, Evented} from './evented'; + +describe('Evented', () => { + + test('calls listeners added with "on"', () => { + const evented = new Evented(); + const listener = jest.fn(); + evented.on('a', listener); + evented.fire(new Event('a')); + evented.fire(new Event('a')); + expect(listener).toHaveBeenCalledTimes(2); + }); + + test('calls listeners added with "once" once', () => { + const evented = new Evented(); + const listener = jest.fn(); + evented.once('a', listener); + evented.fire(new Event('a')); + evented.fire(new Event('a')); + expect(listener).toHaveBeenCalledTimes(1); + expect(evented.listens('a')).toBeFalsy(); + + }); + + test('returns a promise when no listener is provided to "once" method', async () => { + const evented = new Evented(); + const promise = evented.once('a'); + evented.fire(new Event('a')); + evented.fire(new Event('a')); + await promise; + expect(evented.listens('a')).toBeFalsy(); + + }); + + test('passes data to listeners', () => { + const evented = new Evented(); + evented.on('a', (data) => { + expect(data.foo).toBe('bar'); + }); + evented.fire(new Event('a', {foo: 'bar'})); + + }); + + test('passes "target" to listeners', () => { + const evented = new Evented(); + evented.on('a', (data) => { + expect(data.target).toBe(evented); + }); + evented.fire(new Event('a')); + + }); + + test('passes "type" to listeners', () => { + const evented = new Evented(); + evented.on('a', (data) => { + expect(data.type).toBe('a'); + }); + evented.fire(new Event('a')); + + }); + + test('removes listeners with "off"', () => { + const evented = new Evented(); + const listener = jest.fn(); + evented.on('a', listener); + evented.off('a', listener); + evented.fire(new Event('a')); + expect(listener).not.toHaveBeenCalled(); + }); + + test('removes one-time listeners with "off"', () => { + const evented = new Evented(); + const listener = jest.fn(); + evented.once('a', listener); + evented.off('a', listener); + evented.fire(new Event('a')); + expect(listener).not.toHaveBeenCalled(); + }); + + test('once listener is removed prior to call', () => { + const evented = new Evented(); + const listener = jest.fn(); + evented.once('a', () => { + listener(); + evented.fire(new Event('a')); + }); + evented.fire(new Event('a')); + expect(listener).toHaveBeenCalledTimes(1); + }); + + test('reports if an event has listeners with "listens"', () => { + const evented = new Evented(); + evented.on('a', () => {}); + expect(evented.listens('a')).toBeTruthy(); + expect(evented.listens('b')).toBeFalsy(); + + }); + + test('does not report true to "listens" if all listeners have been removed', () => { + const evented = new Evented(); + const listener = () => {}; + evented.on('a', listener); + evented.off('a', listener); + expect(evented.listens('a')).toBeFalsy(); + + }); + + test('does not immediately call listeners added within another listener', done => { + const evented = new Evented(); + evented.on('a', () => { + evented.on('a', () => done('fail')); + }); + evented.fire(new Event('a')); + done(); + }); + + test('has backward compatibility for fire(string, object) API', () => { + const evented = new Evented(); + const listener = jest.fn(x => x); + evented.on('a', listener); + evented.fire('a' as any as Event, {foo: 'bar'}); + expect(listener).toHaveBeenCalledTimes(1); + expect(listener.mock.calls[0][0].foo).toBe('bar'); + + }); + + test('on is idempotent', () => { + const evented = new Evented(); + const order = []; + const listenerA = jest.fn(() => order.push('A')); + const listenerB = jest.fn(() => order.push('B')); + evented.on('a', listenerA); + evented.on('a', listenerB); + evented.on('a', listenerA); + evented.fire(new Event('a')); + expect(listenerA).toHaveBeenCalledTimes(1); + expect(order).toEqual(['A', 'B']); + + }); +}); + +describe('evented parents', () => { + + test('adds parents with "setEventedParent"', () => { + const listener = jest.fn(); + const eventedSource = new Evented(); + const eventedSink = new Evented(); + eventedSource.setEventedParent(eventedSink); + eventedSink.on('a', listener); + eventedSource.fire(new Event('a')); + eventedSource.fire(new Event('a')); + expect(listener).toHaveBeenCalledTimes(2); + }); + + test('passes original data to parent listeners', () => { + const eventedSource = new Evented(); + const eventedSink = new Evented(); + eventedSource.setEventedParent(eventedSink); + eventedSink.on('a', (data) => { + expect(data.foo).toBe('bar'); + }); + eventedSource.fire(new Event('a', {foo: 'bar'})); + + }); + + test('attaches parent data to parent listeners', () => { + const eventedSource = new Evented(); + const eventedSink = new Evented(); + eventedSource.setEventedParent(eventedSink, {foz: 'baz'}); + eventedSink.on('a', (data) => { + expect(data.foz).toBe('baz'); + }); + eventedSource.fire(new Event('a', {foo: 'bar'})); + + }); + + test('attaches parent data from a function to parent listeners', () => { + const eventedSource = new Evented(); + const eventedSink = new Evented(); + eventedSource.setEventedParent(eventedSink, () => ({foz: 'baz'})); + eventedSink.on('a', (data) => { + expect(data.foz).toBe('baz'); + }); + eventedSource.fire(new Event('a', {foo: 'bar'})); + + }); + + test('passes original "target" to parent listeners', () => { + const eventedSource = new Evented(); + const eventedSink = new Evented(); + eventedSource.setEventedParent(eventedSink); + eventedSource.setEventedParent(null); + eventedSink.on('a', (data) => { + expect(data.target).toBe(eventedSource); + }); + eventedSource.fire(new Event('a')); + + }); + + test('removes parents with "setEventedParent(null)"', () => { + const listener = jest.fn(); + const eventedSource = new Evented(); + const eventedSink = new Evented(); + eventedSink.on('a', listener); + eventedSource.setEventedParent(eventedSink); + eventedSource.setEventedParent(null); + eventedSource.fire(new Event('a')); + expect(listener).not.toHaveBeenCalled(); + }); + + test('reports if an event has parent listeners with "listens"', () => { + const eventedSource = new Evented(); + const eventedSink = new Evented(); + eventedSink.on('a', () => {}); + eventedSource.setEventedParent(eventedSink); + expect(eventedSink.listens('a')).toBeTruthy(); + + }); + + test('eventedParent data function is evaluated on every fire', () => { + const eventedSource = new Evented(); + const eventedParent = new Evented(); + let i = 0; + eventedSource.setEventedParent(eventedParent, () => i++); + eventedSource.on('a', () => {}); + eventedSource.fire(new Event('a')); + expect(i).toBe(1); + eventedSource.fire(new Event('a')); + expect(i).toBe(2); + + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/evented.ts b/web/libraries/maplibre-gl/src/util/evented.ts new file mode 100644 index 00000000..062e49cc --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/evented.ts @@ -0,0 +1,181 @@ +import {extend} from './util'; + +/** + * A listener method used as a callback to events + */ +export type Listener = (a: any) => any; + +type Listeners = {[_: string]: Array}; + +function _addEventListener(type: string, listener: Listener, listenerList: Listeners) { + const listenerExists = listenerList[type] && listenerList[type].indexOf(listener) !== -1; + if (!listenerExists) { + listenerList[type] = listenerList[type] || []; + listenerList[type].push(listener); + } +} + +function _removeEventListener(type: string, listener: Listener, listenerList: Listeners) { + if (listenerList && listenerList[type]) { + const index = listenerList[type].indexOf(listener); + if (index !== -1) { + listenerList[type].splice(index, 1); + } + } +} + +/** + * The event class + */ +export class Event { + readonly type: string; + + constructor(type: string, data: any = {}) { + extend(this, data); + this.type = type; + } +} + +interface ErrorLike { + message: string; +} + +/** + * An error event + */ +export class ErrorEvent extends Event { + error: ErrorLike; + + constructor(error: ErrorLike, data: any = {}) { + super('error', extend({error}, data)); + } +} + +/** + * Methods mixed in to other classes for event capabilities. + * + * @group Event Related + */ +export class Evented { + _listeners: Listeners; + _oneTimeListeners: Listeners; + _eventedParent: Evented; + _eventedParentData: any | (() => any); + + /** + * Adds a listener to a specified event type. + * + * @param type - The event type to add a listen for. + * @param listener - The function to be called when the event is fired. + * The listener function is called with the data object passed to `fire`, + * extended with `target` and `type` properties. + * @returns `this` + */ + on(type: string, listener: Listener): this { + this._listeners = this._listeners || {}; + _addEventListener(type, listener, this._listeners); + + return this; + } + + /** + * Removes a previously registered event listener. + * + * @param type - The event type to remove listeners for. + * @param listener - The listener function to remove. + * @returns `this` + */ + off(type: string, listener: Listener) { + _removeEventListener(type, listener, this._listeners); + _removeEventListener(type, listener, this._oneTimeListeners); + + return this; + } + + /** + * Adds a listener that will be called only once to a specified event type. + * + * The listener will be called first time the event fires after the listener is registered. + * + * @param type - The event type to listen for. + * @param listener - The function to be called when the event is fired the first time. + * @returns `this` or a promise if a listener is not provided + */ + once(type: string, listener?: Listener): this | Promise { + if (!listener) { + return new Promise((resolve) => this.once(type, resolve)); + } + this._oneTimeListeners = this._oneTimeListeners || {}; + _addEventListener(type, listener, this._oneTimeListeners); + + return this; + } + + fire(event: Event | string, properties?: any) { + // Compatibility with (type: string, properties: Object) signature from previous versions. + // See https://github.com/mapbox/mapbox-gl-js/issues/6522, + // https://github.com/mapbox/mapbox-gl-draw/issues/766 + if (typeof event === 'string') { + event = new Event(event, properties || {}); + } + + const type = event.type; + + if (this.listens(type)) { + (event as any).target = this; + + // make sure adding or removing listeners inside other listeners won't cause an infinite loop + const listeners = this._listeners && this._listeners[type] ? this._listeners[type].slice() : []; + for (const listener of listeners) { + listener.call(this, event); + } + + const oneTimeListeners = this._oneTimeListeners && this._oneTimeListeners[type] ? this._oneTimeListeners[type].slice() : []; + for (const listener of oneTimeListeners) { + _removeEventListener(type, listener, this._oneTimeListeners); + listener.call(this, event); + } + + const parent = this._eventedParent; + if (parent) { + extend( + event, + typeof this._eventedParentData === 'function' ? this._eventedParentData() : this._eventedParentData + ); + parent.fire(event); + } + + // To ensure that no error events are dropped, print them to the + // console if they have no listeners. + } else if (event instanceof ErrorEvent) { + console.error(event.error); + } + + return this; + } + + /** + * Returns a true if this instance of Evented or any forwardeed instances of Evented have a listener for the specified type. + * + * @param type - The event type + * @returns `true` if there is at least one registered listener for specified event type, `false` otherwise + */ + listens(type: string): boolean { + return ( + (this._listeners && this._listeners[type] && this._listeners[type].length > 0) || + (this._oneTimeListeners && this._oneTimeListeners[type] && this._oneTimeListeners[type].length > 0) || + (this._eventedParent && this._eventedParent.listens(type)) + ); + } + + /** + * Bubble all events fired by this instance of Evented to this parent instance of Evented. + * @returns `this` + */ + setEventedParent(parent?: Evented | null, data?: any | (() => any)) { + this._eventedParent = parent; + this._eventedParentData = data; + + return this; + } +} diff --git a/web/libraries/maplibre-gl/src/util/find_pole_of_inaccessibility.test.ts b/web/libraries/maplibre-gl/src/util/find_pole_of_inaccessibility.test.ts new file mode 100644 index 00000000..dfea4f0e --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/find_pole_of_inaccessibility.test.ts @@ -0,0 +1,21 @@ +import Point from '@mapbox/point-geometry'; +import {findPoleOfInaccessibility} from './find_pole_of_inaccessibility'; + +test('polygon_poi', () => { + + const closedRing = [ + new Point(0, 0), + new Point(10, 10), + new Point(10, 0), + new Point(0, 0) + ]; + const closedRingHole = [ + new Point(2, 1), + new Point(6, 6), + new Point(6, 1), + new Point(2, 1) + ]; + expect(findPoleOfInaccessibility([closedRing], 0.1)).toEqual(new Point(7.0703125, 2.9296875)); + expect(findPoleOfInaccessibility([closedRing, closedRingHole], 0.1)).toEqual(new Point(7.96875, 2.03125)); + +}); diff --git a/web/libraries/maplibre-gl/src/util/find_pole_of_inaccessibility.ts b/web/libraries/maplibre-gl/src/util/find_pole_of_inaccessibility.ts new file mode 100644 index 00000000..dfc2cdc1 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/find_pole_of_inaccessibility.ts @@ -0,0 +1,130 @@ +import Queue from 'tinyqueue'; + +import Point from '@mapbox/point-geometry'; +import {distToSegmentSquared} from './intersection_tests'; + +/** + * Finds an approximation of a polygon's Pole Of Inaccessibiliy https://en.wikipedia.org/wiki/Pole_of_inaccessibility + * This is a copy of http://github.com/mapbox/polylabel adapted to use Points + * + * @param polygonRings - first item in array is the outer ring followed optionally by the list of holes, should be an element of the result of util/classify_rings + * @param precision - Specified in input coordinate units. If 0 returns after first run, if `> 0` repeatedly narrows the search space until the radius of the area searched for the best pole is less than precision + * @param debug - Print some statistics to the console during execution + * @returns Pole of Inaccessibiliy. + */ +export function findPoleOfInaccessibility( + polygonRings: Array>, + precision: number = 1, + debug: boolean = false +): Point { + // find the bounding box of the outer ring + let minX = Infinity, minY = Infinity, maxX = -Infinity, maxY = -Infinity; + const outerRing = polygonRings[0]; + for (let i = 0; i < outerRing.length; i++) { + const p = outerRing[i]; + if (!i || p.x < minX) minX = p.x; + if (!i || p.y < minY) minY = p.y; + if (!i || p.x > maxX) maxX = p.x; + if (!i || p.y > maxY) maxY = p.y; + } + + const width = maxX - minX; + const height = maxY - minY; + const cellSize = Math.min(width, height); + let h = cellSize / 2; + + // a priority queue of cells in order of their "potential" (max distance to polygon) + const cellQueue = new Queue([], compareMax); + + if (cellSize === 0) return new Point(minX, minY); + + // cover polygon with initial cells + for (let x = minX; x < maxX; x += cellSize) { + for (let y = minY; y < maxY; y += cellSize) { + cellQueue.push(new Cell(x + h, y + h, h, polygonRings)); + } + } + + // take centroid as the first best guess + let bestCell = getCentroidCell(polygonRings); + let numProbes = cellQueue.length; + + while (cellQueue.length) { + // pick the most promising cell from the queue + const cell = cellQueue.pop(); + + // update the best cell if we found a better one + if (cell.d > bestCell.d || !bestCell.d) { + bestCell = cell; + if (debug) console.log('found best %d after %d probes', Math.round(1e4 * cell.d) / 1e4, numProbes); + } + + // do not drill down further if there's no chance of a better solution + if (cell.max - bestCell.d <= precision) continue; + + // split the cell into four cells + h = cell.h / 2; + cellQueue.push(new Cell(cell.p.x - h, cell.p.y - h, h, polygonRings)); + cellQueue.push(new Cell(cell.p.x + h, cell.p.y - h, h, polygonRings)); + cellQueue.push(new Cell(cell.p.x - h, cell.p.y + h, h, polygonRings)); + cellQueue.push(new Cell(cell.p.x + h, cell.p.y + h, h, polygonRings)); + numProbes += 4; + } + + if (debug) { + console.log(`num probes: ${numProbes}`); + console.log(`best distance: ${bestCell.d}`); + } + + return bestCell.p; +} + +function compareMax(a, b) { + return b.max - a.max; +} + +function Cell(x, y, h, polygon) { + this.p = new Point(x, y); + this.h = h; // half the cell size + this.d = pointToPolygonDist(this.p, polygon); // distance from cell center to polygon + this.max = this.d + this.h * Math.SQRT2; // max distance to polygon within a cell +} + +// signed distance from point to polygon outline (negative if point is outside) +function pointToPolygonDist(p, polygon) { + let inside = false; + let minDistSq = Infinity; + + for (let k = 0; k < polygon.length; k++) { + const ring = polygon[k]; + + for (let i = 0, len = ring.length, j = len - 1; i < len; j = i++) { + const a = ring[i]; + const b = ring[j]; + + if ((a.y > p.y !== b.y > p.y) && + (p.x < (b.x - a.x) * (p.y - a.y) / (b.y - a.y) + a.x)) inside = !inside; + + minDistSq = Math.min(minDistSq, distToSegmentSquared(p, a, b)); + } + } + + return (inside ? 1 : -1) * Math.sqrt(minDistSq); +} + +// get polygon centroid +function getCentroidCell(polygon) { + let area = 0; + let x = 0; + let y = 0; + const points = polygon[0]; + for (let i = 0, len = points.length, j = len - 1; i < len; j = i++) { + const a = points[i]; + const b = points[j]; + const f = a.x * b.y - b.x * a.y; + x += (a.x + b.x) * f; + y += (a.y + b.y) * f; + area += f * 3; + } + return new Cell(x / area, y / area, 0, polygon); +} diff --git a/web/libraries/maplibre-gl/src/util/geolocation_support.test.ts b/web/libraries/maplibre-gl/src/util/geolocation_support.test.ts new file mode 100644 index 00000000..7641a404 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/geolocation_support.test.ts @@ -0,0 +1,55 @@ +import {checkGeolocationSupport} from './geolocation_support'; + +describe('checkGeolocationSupport', () => { + beforeEach(() => { + jest.resetModules(); + }); + + test('it should return false if geolocation is not defined', done => { + checkGeolocationSupport((returnValue: boolean) => { + expect(returnValue).toBeFalsy(); + done(); + }); + }); + + test('it should return the cached value on second call', done => { + checkGeolocationSupport((returnValue) => { + expect(returnValue).toBeFalsy(); + (window.navigator as any).geolocation = {}; + checkGeolocationSupport((rv) => { + expect(rv).toBe(returnValue); + done(); + }); + }); + }); + + test('it should return the true if geolocation is defined', done => { + (window.navigator as any).geolocation = {}; + checkGeolocationSupport((returnValue) => { + expect(returnValue).toBeTruthy(); + done(); + }, true); + }); + + test('it should check permissions if possible', done => { + (window.navigator as any).geolocation = {}; + (window.navigator as any).permissions = { + query: () => Promise.resolve({state: 'granted'}) + }; + checkGeolocationSupport((returnValue) => { + expect(returnValue).toBeTruthy(); + done(); + }, true); + }); + + test('it should check permissions and geolocation for iOS 16 promise rejection', done => { + (window.navigator as any).geolocation = undefined; + (window.navigator as any).permissions = { + query: () => Promise.reject(new Error('pemissions error')) + }; + checkGeolocationSupport((returnValue) => { + expect(returnValue).toBeFalsy(); + done(); + }, true); + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/geolocation_support.ts b/web/libraries/maplibre-gl/src/util/geolocation_support.ts new file mode 100644 index 00000000..d64940a0 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/geolocation_support.ts @@ -0,0 +1,24 @@ +let supportsGeolocation; + +export function checkGeolocationSupport(callback: (supported: boolean) => void, forceRecalculation = false): void { + if (supportsGeolocation !== undefined && !forceRecalculation) { + callback(supportsGeolocation); + } else if (window.navigator.permissions !== undefined) { + // navigator.permissions has incomplete browser support + // http://caniuse.com/#feat=permissions-api + // Test for the case where a browser disables Geolocation because of an + // insecure origin + window.navigator.permissions.query({name: 'geolocation'}).then((p) => { + supportsGeolocation = p.state !== 'denied'; + callback(supportsGeolocation); + }).catch(() => { + // Fix for iOS16 which rejects query but still supports geolocation + supportsGeolocation = !!window.navigator.geolocation; + callback(supportsGeolocation); + }); + + } else { + supportsGeolocation = !!window.navigator.geolocation; + callback(supportsGeolocation); + } +} diff --git a/web/libraries/maplibre-gl/src/util/global_worker_pool.ts b/web/libraries/maplibre-gl/src/util/global_worker_pool.ts new file mode 100644 index 00000000..e6a3c28b --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/global_worker_pool.ts @@ -0,0 +1,32 @@ +import {WorkerPool, PRELOAD_POOL_ID} from './worker_pool'; + +let globalWorkerPool; + +/** + * Creates (if necessary) and returns the single, global WorkerPool instance + * to be shared across each Map + */ +export function getGlobalWorkerPool() { + if (!globalWorkerPool) { + globalWorkerPool = new WorkerPool(); + } + return globalWorkerPool; +} + +export function prewarm() { + const workerPool = getGlobalWorkerPool(); + workerPool.acquire(PRELOAD_POOL_ID); +} + +export function clearPrewarmedResources() { + const pool = globalWorkerPool; + if (pool) { + // Remove the pool only if all maps that referenced the preloaded global worker pool have been removed. + if (pool.isPreloaded() && pool.numActive() === 1) { + pool.release(PRELOAD_POOL_ID); + globalWorkerPool = null; + } else { + console.warn('Could not clear WebWorkers since there are active Map instances that still reference it. The pre-warmed WebWorker pool can only be cleared when all map instances have been removed with map.remove()'); + } + } +} diff --git a/web/libraries/maplibre-gl/src/util/image.ts b/web/libraries/maplibre-gl/src/util/image.ts new file mode 100644 index 00000000..27b13c81 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/image.ts @@ -0,0 +1,151 @@ +import {register} from './web_worker_transfer'; + +export type Size = { + width: number; + height: number; +}; + +type Point2D = { + x: number; + y: number; +}; + +function createImage(image: any, { + width, + height +}: Size, channels: number, data?: Uint8Array | Uint8ClampedArray) { + if (!data) { + data = new Uint8Array(width * height * channels); + } else if (data instanceof Uint8ClampedArray) { + data = new Uint8Array(data.buffer); + } else if (data.length !== width * height * channels) { + throw new RangeError(`mismatched image size. expected: ${data.length} but got: ${width * height * channels}`); + } + image.width = width; + image.height = height; + image.data = data; + return image; +} + +function resizeImage(image: any, { + width, + height +}: Size, channels: number) { + if (width === image.width && height === image.height) { + return; + } + + const newImage = createImage({}, {width, height}, channels); + + copyImage(image, newImage, {x: 0, y: 0}, {x: 0, y: 0}, { + width: Math.min(image.width, width), + height: Math.min(image.height, height) + }, channels); + + image.width = width; + image.height = height; + image.data = newImage.data; +} + +function copyImage(srcImg: any, dstImg: any, srcPt: Point2D, dstPt: Point2D, size: Size, channels: number) { + if (size.width === 0 || size.height === 0) { + return dstImg; + } + + if (size.width > srcImg.width || + size.height > srcImg.height || + srcPt.x > srcImg.width - size.width || + srcPt.y > srcImg.height - size.height) { + throw new RangeError('out of range source coordinates for image copy'); + } + + if (size.width > dstImg.width || + size.height > dstImg.height || + dstPt.x > dstImg.width - size.width || + dstPt.y > dstImg.height - size.height) { + throw new RangeError('out of range destination coordinates for image copy'); + } + + const srcData = srcImg.data; + const dstData = dstImg.data; + + if (srcData === dstData) throw new Error('srcData equals dstData, so image is already copied'); + + for (let y = 0; y < size.height; y++) { + const srcOffset = ((srcPt.y + y) * srcImg.width + srcPt.x) * channels; + const dstOffset = ((dstPt.y + y) * dstImg.width + dstPt.x) * channels; + for (let i = 0; i < size.width * channels; i++) { + dstData[dstOffset + i] = srcData[srcOffset + i]; + } + } + return dstImg; +} + +/** + * @internal + * An image with alpha color value + */ +export class AlphaImage { + width: number; + height: number; + data: Uint8Array; + + constructor(size: Size, data?: Uint8Array | Uint8ClampedArray) { + createImage(this, size, 1, data); + } + + resize(size: Size) { + resizeImage(this, size, 1); + } + + clone() { + return new AlphaImage({width: this.width, height: this.height}, new Uint8Array(this.data)); + } + + static copy(srcImg: AlphaImage, dstImg: AlphaImage, srcPt: Point2D, dstPt: Point2D, size: Size) { + copyImage(srcImg, dstImg, srcPt, dstPt, size, 1); + } +} + +/** + * An object to store image data not premultiplied, because ImageData is not premultiplied. + * UNPACK_PREMULTIPLY_ALPHA_WEBGL must be used when uploading to a texture. + */ +export class RGBAImage { + width: number; + height: number; + + /** + * data must be a Uint8Array instead of Uint8ClampedArray because texImage2D does not support Uint8ClampedArray in all browsers. + */ + data: Uint8Array; + + constructor(size: Size, data?: Uint8Array | Uint8ClampedArray) { + createImage(this, size, 4, data); + } + + resize(size: Size) { + resizeImage(this, size, 4); + } + + replace(data: Uint8Array | Uint8ClampedArray, copy?: boolean) { + if (copy) { + this.data.set(data); + } else if (data instanceof Uint8ClampedArray) { + this.data = new Uint8Array(data.buffer); + } else { + this.data = data; + } + } + + clone() { + return new RGBAImage({width: this.width, height: this.height}, new Uint8Array(this.data)); + } + + static copy(srcImg: RGBAImage | ImageData, dstImg: RGBAImage, srcPt: Point2D, dstPt: Point2D, size: Size) { + copyImage(srcImg, dstImg, srcPt, dstPt, size, 4); + } +} + +register('AlphaImage', AlphaImage); +register('RGBAImage', RGBAImage); diff --git a/web/libraries/maplibre-gl/src/util/image_request.test.ts b/web/libraries/maplibre-gl/src/util/image_request.test.ts new file mode 100644 index 00000000..b1f97248 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/image_request.test.ts @@ -0,0 +1,413 @@ +import {config} from './config'; +import {webpSupported} from './webp_supported'; +import {stubAjaxGetImage} from './test/util'; +import {fakeServer, type FakeServer} from 'nise'; +import {ImageRequest, ImageRequestQueueItem} from './image_request'; +import * as ajax from './ajax'; + +describe('ImageRequest', () => { + let server: FakeServer; + beforeEach(() => { + global.fetch = null; + server = fakeServer.create(); + ImageRequest.resetRequestQueue(); + stubAjaxGetImage(undefined); + }); + afterEach(() => { + server.restore(); + }); + + test('getImage respects maxParallelImageRequests', done => { + server.respondWith(request => request.respond(200, {'Content-Type': 'image/png'}, '')); + + const maxRequests = config.MAX_PARALLEL_IMAGE_REQUESTS; + let callbackCount = 0; + function callback(err) { + if (err) return; + // last request is only added after we got a response from one of the previous ones + expect(server.requests).toHaveLength(maxRequests + callbackCount); + callbackCount++; + if (callbackCount === 2) { + done(); + } + } + + for (let i = 0; i < maxRequests + 1; i++) { + ImageRequest.getImage({url: ''}, callback); + } + expect(server.requests).toHaveLength(maxRequests); + + server.requests[0].respond(undefined, undefined, undefined); + server.requests[1].respond(undefined, undefined, undefined); + }); + test('Cancel: getImage cancelling frees up request for maxParallelImageRequests', done => { + server.respondWith(request => request.respond(200, {'Content-Type': 'image/png'}, '')); + + const maxRequests = config.MAX_PARALLEL_IMAGE_REQUESTS; + + for (let i = 0; i < maxRequests + 1; i++) { + ImageRequest.getImage({url: ''}, () => done('test failed: getImage callback was called')).cancel(); + } + expect(server.requests).toHaveLength(maxRequests + 1); + done(); + }); + + test('Cancel: getImage requests that were once queued are still abortable', done => { + const maxRequests = config.MAX_PARALLEL_IMAGE_REQUESTS; + + const requests = []; + for (let i = 0; i < maxRequests; i++) { + requests.push(ImageRequest.getImage({url: ''}, () => {})); + } + + // the limit of allowed requests is reached + expect(server.requests).toHaveLength(maxRequests); + + const queuedURL = 'this-is-the-queued-request'; + const queued = ImageRequest.getImage({url: queuedURL}, () => done('test failed: getImage callback was called')); + + // the new requests is queued because the limit is reached + expect(server.requests).toHaveLength(maxRequests); + + // cancel the first request to let the queued request start + requests[0].cancel(); + expect(server.requests).toHaveLength(maxRequests + 1); + + // abort the previously queued request and confirm that it is aborted + const queuedRequest = server.requests[server.requests.length - 1]; + expect(queuedRequest.url).toBe(queuedURL); + expect((queuedRequest as any).aborted).toBeUndefined(); + queued.cancel(); + expect((queuedRequest as any).aborted).toBe(true); + + done(); + }); + + test('getImage sends accept/webp when supported', done => { + server.respondWith((request) => { + expect(request.requestHeaders.accept.includes('image/webp')).toBeTruthy(); + request.respond(200, {'Content-Type': 'image/webp'}, ''); + }); + + // mock webp support + webpSupported.supported = true; + + ImageRequest.getImage({url: ''}, () => { done(); }); + + server.respond(); + }); + + test('getImage uses createImageBitmap when supported', done => { + server.respondWith(request => request.respond(200, {'Content-Type': 'image/png', + 'Cache-Control': 'cache', + 'Expires': 'expires'}, '')); + + stubAjaxGetImage(() => Promise.resolve(new ImageBitmap())); + + ImageRequest.getImage({url: ''}, (err, img, expiry) => { + if (err) done(err); + expect(img).toBeInstanceOf(ImageBitmap); + expect(expiry.cacheControl).toBe('cache'); + expect(expiry.expires).toBe('expires'); + done(); + }); + + server.respond(); + }); + + test('getImage using createImageBitmap throws exception', done => { + server.respondWith(request => request.respond(200, {'Content-Type': 'image/png', + 'Cache-Control': 'cache', + 'Expires': 'expires'}, '')); + + stubAjaxGetImage(() => Promise.reject(new Error('error'))); + + ImageRequest.getImage({url: ''}, (err, img) => { + expect(img).toBeFalsy(); + if (err) done(); + }); + + server.respond(); + }); + + test('getImage uses HTMLImageElement when createImageBitmap is not supported', done => { + const makeRequestSky = jest.spyOn(ajax, 'makeRequest'); + server.respondWith(request => request.respond(200, {'Content-Type': 'image/png', + 'Cache-Control': 'cache', + 'Expires': 'expires'}, '')); + + ImageRequest.getImage({url: ''}, (err, img, expiry) => { + if (err) done(`get image failed with error ${err.message}`); + expect(img).toBeInstanceOf(HTMLImageElement); + expect(expiry.cacheControl).toBe('cache'); + expect(expiry.expires).toBe('expires'); + done(); + }); + + server.respond(); + expect(makeRequestSky).toHaveBeenCalledTimes(1); + makeRequestSky.mockClear(); + }); + + test('getImage using HTMLImageElement with same-origin credentials', done => { + const makeRequestSky = jest.spyOn(ajax, 'makeRequest'); + ImageRequest.getImage({url: '', credentials: 'same-origin'}, (err, img: HTMLImageElement) => { + if (err) done(err); + expect(img).toBeInstanceOf(HTMLImageElement); + expect(img.crossOrigin).toBe('anonymous'); + done(); + }, false); + + expect(makeRequestSky).toHaveBeenCalledTimes(0); + makeRequestSky.mockClear(); + }); + + test('getImage using HTMLImageElement with include credentials', done => { + const makeRequestSky = jest.spyOn(ajax, 'makeRequest'); + ImageRequest.getImage({url: '', credentials: 'include'}, (err, img: HTMLImageElement) => { + if (err) done(err); + expect(img).toBeInstanceOf(HTMLImageElement); + expect(img.crossOrigin).toBe('use-credentials'); + done(); + }, false); + + expect(makeRequestSky).toHaveBeenCalledTimes(0); + makeRequestSky.mockClear(); + }); + + test('getImage using HTMLImageElement with accept header', done => { + const makeRequestSky = jest.spyOn(ajax, 'makeRequest'); + ImageRequest.getImage({url: '', credentials: 'include', headers: {accept: 'accept'}}, + (err, img: HTMLImageElement) => { + if (err) done(err); + expect(img).toBeInstanceOf(HTMLImageElement); + expect(img.crossOrigin).toBe('use-credentials'); + done(); + }, false); + + expect(makeRequestSky).toHaveBeenCalledTimes(0); + makeRequestSky.mockClear(); + }); + + test('getImage uses makeRequest when custom Headers are added', () => { + const makeRequestSky = jest.spyOn(ajax, 'makeRequest'); + + ImageRequest.getImage({url: '', credentials: 'include', headers: {custom: 'test', accept: 'image'}}, + () => {}, + false); + + expect(makeRequestSky).toHaveBeenCalledTimes(1); + makeRequestSky.mockClear(); + }); + + test('getImage request returned 404 response for fetch request', done => { + server.respondWith(request => request.respond(404)); + + ImageRequest.getImage({url: ''}, (err) => { + if (err) done(); + else done('Image download should have failed'); + }); + + server.respond(); + }); + + test('getImage request failed for HTTPImageRequest', done => { + ImageRequest.getImage({url: 'error'}, (err) => { + if (err) done(); + else done('Image download should have failed'); + }, false); + }); + + test('Cancel: getImage request cancelled for HTTPImageRequest', done => { + let imageUrl; + const requestUrl = 'test'; + // eslint-disable-next-line accessor-pairs + Object.defineProperty(global.Image.prototype, 'src', { + set(url: string) { + imageUrl = url; + } + }); + + const request = ImageRequest.getImage({url: requestUrl}, () => { + done('Callback should not be called in case image request is cancelled'); + }, false); + + expect(imageUrl).toBe(requestUrl); + expect(request.cancelled).toBeFalsy(); + request.cancel(); + expect(request.cancelled).toBeTruthy(); + expect(imageUrl).toBe(''); + done(); + }); + + test('Cancel: getImage request cancelled', done => { + server.respondWith(request => request.respond(200, {'Content-Type': 'image/png', + 'Cache-Control': 'cache', + 'Expires': 'expires'}, '')); + + const request = ImageRequest.getImage({url: ''}, () => { + done('Callback should not be called in case image request is cancelled'); + }); + + expect(request.cancelled).toBeFalsy(); + request.cancel(); + expect(request.cancelled).toBeTruthy(); + + server.respond(); + done(); + }); + + test('Cancel: Cancellation of an image which has not yet been requested', () => { + const maxRequests = config.MAX_PARALLEL_IMAGE_REQUESTS; + + let callbackCounter = 0; + function callback() { + callbackCounter++; + } + const requests: ImageRequestQueueItem[] = []; + for (let i = 0; i < maxRequests + 100; i++) { + requests.push(ImageRequest.getImage({url: `${i}`}, callback)); + } + + // Request should have been initiated + expect(requests[0].innerRequest).toBeDefined(); + requests[0].cancel(); + + // Queue should move forward and next request is made + expect(server.requests).toHaveLength(maxRequests + 1); + + // Cancel request should not call callback + expect(callbackCounter).toBe(0); + + // Cancel request which is not yet issued. It should not fire callback + const nextRequestInQueue = requests[server.requests.length]; + expect(nextRequestInQueue.innerRequest).toBeUndefined(); + const cancelledImageUrl = nextRequestInQueue.requestParameters.url; + nextRequestInQueue.cancel(); + + // Queue should not move forward as cancelled image was sitting in queue + expect(server.requests).toHaveLength(maxRequests + 1); + expect(callbackCounter).toBe(0); + + // On server response, next image queued should not be the cancelled image + server.requests[1].respond(200); + expect(callbackCounter).toBe(1); + expect(server.requests).toHaveLength(maxRequests + 2); + // Verify that the last request made skipped the cancelled image request + expect(server.requests[server.requests.length - 1].url).toBe((parseInt(cancelledImageUrl) + 1).toString()); + }); + + test('throttling: one throttling client will result in throttle behavior for all', () => { + const maxRequestsPerFrame = config.MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME; + const callbackHandles = []; + + // add one for each + callbackHandles.push(ImageRequest.addThrottleControl(() => false)); + callbackHandles.push(ImageRequest.addThrottleControl(() => true)); + + let callbackCounter = 0; + function callback() { + callbackCounter++; + } + + for (let i = 0; i < maxRequestsPerFrame + 1; i++) { + ImageRequest.getImage({url: ''}, callback); + } + + expect(server.requests).toHaveLength(maxRequestsPerFrame); + + // all pending + expect(callbackCounter).toBe(0); + + for (const handle of callbackHandles) { + ImageRequest.removeThrottleControl(handle); + } + }); + + test('throttling: image queue will process MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME if throttling control returns true', () => { + const maxRequestsPerFrame = config.MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME; + const maxRequests = config.MAX_PARALLEL_IMAGE_REQUESTS; + const controlId = ImageRequest.addThrottleControl(() => true); + + let callbackCounter = 0; + function callback() { + callbackCounter++; + } + + for (let i = 0; i < maxRequests; i++) { + ImageRequest.getImage({url: ''}, callback); + } + + // Should only fire request to a max allowed per frame + expect(server.requests).toHaveLength(maxRequestsPerFrame); + + // all pending + expect(callbackCounter).toBe(0); + + ImageRequest.removeThrottleControl(controlId); + }); + + test('throttling: image queue will process MAX_PARALLEL_IMAGE_REQUESTS if throttling control returns false', () => { + const maxRequests = config.MAX_PARALLEL_IMAGE_REQUESTS; + const controlId = ImageRequest.addThrottleControl(() => false); + + let callbackCounter = 0; + function callback() { + callbackCounter++; + } + + for (let i = 0; i < maxRequests + 100; i++) { + ImageRequest.getImage({url: ''}, callback); + } + + // all should be processed because throttle control is returning false + expect(server.requests).toHaveLength(maxRequests); + + // all pending + expect(callbackCounter).toBe(0); + + ImageRequest.removeThrottleControl(controlId); + }); + + test('throttling: removing throttling client will process all requests', () => { + const requestParameter = {'Content-Type': 'image/png', url: ''}; + const maxRequestsPerFrame = config.MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME; + + // add 10, and one of them is throttling + const throttlingIndex = 5; + for (let i = 0; i < 10; i++) { + const throttlingClient: boolean = (i === throttlingIndex); + ImageRequest.addThrottleControl(() => throttlingClient); + } + + let callbackCounter = 0; + function callback() { + callbackCounter++; + } + + // make 2 times + 1 more requests + const requestsMade = 2 * maxRequestsPerFrame + 1; + const imageResults: ImageRequestQueueItem[] = []; + for (let i = 0; i < requestsMade; i++) { + imageResults.push(ImageRequest.getImage(requestParameter, callback)); + } + + // up to the config value + expect(server.requests).toHaveLength(maxRequestsPerFrame); + + const itemIndexToComplete = 6; + server.requests[itemIndexToComplete].respond(200); + + // unleash it by removing the throttling client + ImageRequest.removeThrottleControl(throttlingIndex); + expect(server.requests).toHaveLength(requestsMade); + + // all pending + expect(callbackCounter).toBe(1); + + // everything should still be pending except itemIndexToComplete + for (let i = 0; i < maxRequestsPerFrame + 1; i++) { + expect(imageResults[i].completed).toBe(i === itemIndexToComplete); + } + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/image_request.ts b/web/libraries/maplibre-gl/src/util/image_request.ts new file mode 100644 index 00000000..e00b395e --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/image_request.ts @@ -0,0 +1,288 @@ +import type {Cancelable} from '../types/cancelable'; +import {RequestParameters, ExpiryData, makeRequest, sameOrigin, getProtocolAction} from './ajax'; +import type {Callback} from '../types/callback'; + +import {arrayBufferToImageBitmap, arrayBufferToImage, extend, isWorker, isImageBitmap} from './util'; +import {webpSupported} from './webp_supported'; +import {config} from './config'; + +/** + * The callback that is being called after an image was fetched + */ +export type GetImageCallback = (error?: Error | null, image?: HTMLImageElement | ImageBitmap | null, expiry?: ExpiryData | null) => void; + +type ImageQueueThrottleControlCallback = () => boolean; + +export type ImageRequestQueueItem = Cancelable & { + requestParameters: RequestParameters; + supportImageRefresh: boolean; + callback: GetImageCallback; + cancelled: boolean; + completed: boolean; + innerRequest?: Cancelable; +} + +type ImageQueueThrottleCallbackDictionary = { + [Key: number]: ImageQueueThrottleControlCallback; +} + +type HTMLImageElementWithPriority = HTMLImageElement & +{ + // fetchPriority is experimental property supported on Chromium browsers from Version 102 + // By default images are downloaded with priority low, whereas fetch request downloads with priority high + // https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/fetchPriority + fetchPriority?: 'auto' | 'high' | 'low'; +}; + +/** + * By default, the image queue is self driven, meaning as soon as one requested item is processed, + * it will move on to next one as quickly as it can while limiting + * the number of concurrent requests to MAX_PARALLEL_IMAGE_REQUESTS. The default behavior + * ensures that static views of the map can be rendered with minimal delay. + * + * However, the default behavior can prevent dynamic views of the map from rendering + * smoothly in that many requests can finish in one render frame, putting too much pressure on GPU. + * + * When the view of the map is moving dynamically, smoother frame rates can be achieved + * by throttling the number of items processed by the queue per frame. This can be + * accomplished by using {@link addThrottleControl} to allow the caller to + * use a lambda function to determine when the queue should be throttled (e.g. when isMoving()) + * and manually calling {@link processQueue} in the render loop. + */ +export namespace ImageRequest { + let imageRequestQueue : ImageRequestQueueItem[]; + let currentParallelImageRequests:number; + + let throttleControlCallbackHandleCounter: number; + let throttleControlCallbacks: ImageQueueThrottleCallbackDictionary; + + /** + * Reset the image request queue, removing all pending requests. + */ + export const resetRequestQueue = (): void => { + imageRequestQueue = []; + currentParallelImageRequests = 0; + throttleControlCallbackHandleCounter = 0; + throttleControlCallbacks = {}; + }; + + /** + * Install a callback to control when image queue throttling is desired. + * (e.g. when the map view is moving) + * @param callback - The callback function to install + * @returns handle that identifies the installed callback. + */ + export const addThrottleControl = (callback: ImageQueueThrottleControlCallback): number => { + const handle = throttleControlCallbackHandleCounter++; + throttleControlCallbacks[handle] = callback; + return handle; + }; + + /** + * Remove a previously installed callback by passing in the handle returned + * by {@link addThrottleControl}. + * @param callbackHandle - The handle for the callback to remove. + */ + export const removeThrottleControl = (callbackHandle: number): void => { + delete throttleControlCallbacks[callbackHandle]; + // Try updating the queue + processQueue(); + }; + + /** + * Check to see if any of the installed callbacks are requesting the queue + * to be throttled. + * @returns `true` if any callback is causing the queue to be throttled. + */ + const isThrottled = (): boolean => { + const allControlKeys = Object.keys(throttleControlCallbacks); + let throttleingRequested = false; + if (allControlKeys.length > 0) { + for (const key of allControlKeys) { + throttleingRequested = throttleControlCallbacks[key](); + if (throttleingRequested) { + break; + } + } + } + return throttleingRequested; + }; + + /** + * Request to load an image. + * @param requestParameters - Request parameters. + * @param callback - Callback to issue when the request completes. + * @param supportImageRefresh - `true`, if the image request need to support refresh based on cache headers. + * @returns Cancelable request. + */ + export const getImage = ( + requestParameters: RequestParameters, + callback: GetImageCallback, + supportImageRefresh: boolean = true + ): ImageRequestQueueItem => { + if (webpSupported.supported) { + if (!requestParameters.headers) { + requestParameters.headers = {}; + } + requestParameters.headers.accept = 'image/webp,*/*'; + } + + const request:ImageRequestQueueItem = { + requestParameters, + supportImageRefresh, + callback, + cancelled: false, + completed: false, + cancel: () => { + if (!request.completed && !request.cancelled) { + request.cancelled = true; + + // Only reduce currentParallelImageRequests, if the image request was issued. + if (request.innerRequest) { + request.innerRequest.cancel(); + currentParallelImageRequests--; + } + + // in the case of cancelling, it WILL move on + processQueue(); + } + } + }; + + imageRequestQueue.push(request); + processQueue(); + return request; + }; + + const arrayBufferToCanvasImageSource = (data: ArrayBuffer, callback: Callback) => { + const imageBitmapSupported = typeof createImageBitmap === 'function'; + if (imageBitmapSupported) { + arrayBufferToImageBitmap(data, callback); + } else { + arrayBufferToImage(data, callback); + } + }; + + const doImageRequest = (itemInQueue: ImageRequestQueueItem): Cancelable => { + const {requestParameters, supportImageRefresh, callback} = itemInQueue; + extend(requestParameters, {type: 'image'}); + + // - If refreshExpiredTiles is false, then we can use HTMLImageElement to download raster images. + // - Fetch/XHR (via MakeRequest API) will be used to download images for following scenarios: + // 1. Style image sprite will had a issue with HTMLImageElement as described + // here: https://github.com/mapbox/mapbox-gl-js/issues/1470 + // 2. If refreshExpiredTiles is true (default), then in order to read the image cache header, + // fetch/XHR request will be required + // - For any special case handling like use of AddProtocol, worker initiated request or additional headers + // let makeRequest handle it. + // - HtmlImageElement request automatically adds accept header for all the browser supported images + const canUseHTMLImageElement = supportImageRefresh === false && + !isWorker() && + !getProtocolAction(requestParameters.url) && + (!requestParameters.headers || + Object.keys(requestParameters.headers).reduce((acc, item) => acc && item === 'accept', true)); + + const action = canUseHTMLImageElement ? getImageUsingHtmlImage : makeRequest; + return action( + requestParameters, + (err?: Error | null, + data?: HTMLImageElement | ImageBitmap | ArrayBuffer | null, + cacheControl?: string | null, + expires?: string | null) => { + onImageResponse(itemInQueue, callback, err, data, cacheControl, expires); + }); + }; + + const onImageResponse = ( + itemInQueue: ImageRequestQueueItem, + callback:GetImageCallback, + err?: Error | null, + data?: HTMLImageElement | ImageBitmap | ArrayBuffer | null, + cacheControl?: string | null, + expires?: string | null): void => { + if (err) { + callback(err); + } else if (data instanceof HTMLImageElement || isImageBitmap(data)) { + // User using addProtocol can directly return HTMLImageElement/ImageBitmap type + // If HtmlImageElement is used to get image then response type will be HTMLImageElement + callback(null, data); + } else if (data) { + const decoratedCallback = (imgErr?: Error | null, imgResult?: CanvasImageSource | null) => { + if (imgErr != null) { + callback(imgErr); + } else if (imgResult != null) { + callback(null, imgResult as (HTMLImageElement | ImageBitmap), {cacheControl, expires}); + } + }; + arrayBufferToCanvasImageSource(data, decoratedCallback); + } + if (!itemInQueue.cancelled) { + itemInQueue.completed = true; + currentParallelImageRequests--; + + processQueue(); + } + }; + + /** + * Process some number of items in the image request queue. + */ + const processQueue = (): void => { + + const maxImageRequests = isThrottled() ? + config.MAX_PARALLEL_IMAGE_REQUESTS_PER_FRAME : + config.MAX_PARALLEL_IMAGE_REQUESTS; + + // limit concurrent image loads to help with raster sources performance on big screens + for (let numImageRequests = currentParallelImageRequests; + numImageRequests < maxImageRequests && imageRequestQueue.length > 0; + numImageRequests++) { + + const topItemInQueue: ImageRequestQueueItem = imageRequestQueue.shift(); + if (topItemInQueue.cancelled) { + numImageRequests--; + continue; + } + + const innerRequest = doImageRequest(topItemInQueue); + + currentParallelImageRequests++; + + topItemInQueue.innerRequest = innerRequest; + } + }; + + const getImageUsingHtmlImage = (requestParameters: RequestParameters, callback: GetImageCallback): Cancelable => { + const image = new Image() as HTMLImageElementWithPriority; + const url = requestParameters.url; + let requestCancelled = false; + const credentials = requestParameters.credentials; + if (credentials && credentials === 'include') { + image.crossOrigin = 'use-credentials'; + } else if ((credentials && credentials === 'same-origin') || !sameOrigin(url)) { + image.crossOrigin = 'anonymous'; + } + + image.fetchPriority = 'high'; + image.onload = () => { + callback(null, image); + image.onerror = image.onload = null; + }; + image.onerror = () => { + if (!requestCancelled) { + callback(new Error('Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.')); + } + image.onerror = image.onload = null; + }; + image.src = url; + return { + cancel: () => { + requestCancelled = true; + // Set src to '' to actually cancel the request + image.src = ''; + } + }; + }; +} + +ImageRequest.resetRequestQueue(); diff --git a/web/libraries/maplibre-gl/src/util/intersection_tests.ts b/web/libraries/maplibre-gl/src/util/intersection_tests.ts new file mode 100644 index 00000000..79d4fb11 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/intersection_tests.ts @@ -0,0 +1,206 @@ +import {isCounterClockwise} from './util'; + +import Point from '@mapbox/point-geometry'; + +export {polygonIntersectsBufferedPoint, polygonIntersectsMultiPolygon, polygonIntersectsBufferedMultiLine, polygonIntersectsPolygon, distToSegmentSquared, polygonIntersectsBox}; + +type Line = Array; +type MultiLine = Array; +type Ring = Array; +type Polygon = Array; +type MultiPolygon = Array; + +function polygonIntersectsPolygon(polygonA: Polygon, polygonB: Polygon) { + for (let i = 0; i < polygonA.length; i++) { + if (polygonContainsPoint(polygonB, polygonA[i])) return true; + } + + for (let i = 0; i < polygonB.length; i++) { + if (polygonContainsPoint(polygonA, polygonB[i])) return true; + } + + if (lineIntersectsLine(polygonA, polygonB)) return true; + + return false; +} + +function polygonIntersectsBufferedPoint(polygon: Polygon, point: Point, radius: number) { + if (polygonContainsPoint(polygon, point)) return true; + if (pointIntersectsBufferedLine(point, polygon, radius)) return true; + return false; +} + +function polygonIntersectsMultiPolygon(polygon: Polygon, multiPolygon: MultiPolygon) { + + if (polygon.length === 1) { + return multiPolygonContainsPoint(multiPolygon, polygon[0]); + } + + for (let m = 0; m < multiPolygon.length; m++) { + const ring = multiPolygon[m]; + for (let n = 0; n < ring.length; n++) { + if (polygonContainsPoint(polygon, ring[n])) return true; + } + } + + for (let i = 0; i < polygon.length; i++) { + if (multiPolygonContainsPoint(multiPolygon, polygon[i])) return true; + } + + for (let k = 0; k < multiPolygon.length; k++) { + if (lineIntersectsLine(polygon, multiPolygon[k])) return true; + } + + return false; +} + +function polygonIntersectsBufferedMultiLine(polygon: Polygon, multiLine: MultiLine, radius: number) { + for (let i = 0; i < multiLine.length; i++) { + const line = multiLine[i]; + + if (polygon.length >= 3) { + for (let k = 0; k < line.length; k++) { + if (polygonContainsPoint(polygon, line[k])) return true; + } + } + + if (lineIntersectsBufferedLine(polygon, line, radius)) return true; + } + return false; +} + +function lineIntersectsBufferedLine(lineA: Line, lineB: Line, radius: number) { + + if (lineA.length > 1) { + if (lineIntersectsLine(lineA, lineB)) return true; + + // Check whether any point in either line is within radius of the other line + for (let j = 0; j < lineB.length; j++) { + if (pointIntersectsBufferedLine(lineB[j], lineA, radius)) return true; + } + } + + for (let k = 0; k < lineA.length; k++) { + if (pointIntersectsBufferedLine(lineA[k], lineB, radius)) return true; + } + + return false; +} + +function lineIntersectsLine(lineA: Line, lineB: Line) { + if (lineA.length === 0 || lineB.length === 0) return false; + for (let i = 0; i < lineA.length - 1; i++) { + const a0 = lineA[i]; + const a1 = lineA[i + 1]; + for (let j = 0; j < lineB.length - 1; j++) { + const b0 = lineB[j]; + const b1 = lineB[j + 1]; + if (lineSegmentIntersectsLineSegment(a0, a1, b0, b1)) return true; + } + } + return false; +} + +function lineSegmentIntersectsLineSegment(a0: Point, a1: Point, b0: Point, b1: Point) { + return isCounterClockwise(a0, b0, b1) !== isCounterClockwise(a1, b0, b1) && + isCounterClockwise(a0, a1, b0) !== isCounterClockwise(a0, a1, b1); +} + +function pointIntersectsBufferedLine(p: Point, line: Line, radius: number) { + const radiusSquared = radius * radius; + + if (line.length === 1) return p.distSqr(line[0]) < radiusSquared; + + for (let i = 1; i < line.length; i++) { + // Find line segments that have a distance <= radius^2 to p + // In that case, we treat the line as "containing point p". + const v = line[i - 1], w = line[i]; + if (distToSegmentSquared(p, v, w) < radiusSquared) return true; + } + return false; +} + +// Code from http://stackoverflow.com/a/1501725/331379. +function distToSegmentSquared(p: Point, v: Point, w: Point) { + const l2 = v.distSqr(w); + if (l2 === 0) return p.distSqr(v); + const t = ((p.x - v.x) * (w.x - v.x) + (p.y - v.y) * (w.y - v.y)) / l2; + if (t < 0) return p.distSqr(v); + if (t > 1) return p.distSqr(w); + return p.distSqr(w.sub(v)._mult(t)._add(v)); +} + +// point in polygon ray casting algorithm +function multiPolygonContainsPoint(rings: Array, p: Point) { + let c = false, + ring, p1, p2; + + for (let k = 0; k < rings.length; k++) { + ring = rings[k]; + for (let i = 0, j = ring.length - 1; i < ring.length; j = i++) { + p1 = ring[i]; + p2 = ring[j]; + if (((p1.y > p.y) !== (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) { + c = !c; + } + } + } + return c; +} + +function polygonContainsPoint(ring: Ring, p: Point) { + let c = false; + for (let i = 0, j = ring.length - 1; i < ring.length; j = i++) { + const p1 = ring[i]; + const p2 = ring[j]; + if (((p1.y > p.y) !== (p2.y > p.y)) && (p.x < (p2.x - p1.x) * (p.y - p1.y) / (p2.y - p1.y) + p1.x)) { + c = !c; + } + } + return c; +} + +function polygonIntersectsBox(ring: Ring, boxX1: number, boxY1: number, boxX2: number, boxY2: number) { + for (const p of ring) { + if (boxX1 <= p.x && + boxY1 <= p.y && + boxX2 >= p.x && + boxY2 >= p.y) return true; + } + + const corners = [ + new Point(boxX1, boxY1), + new Point(boxX1, boxY2), + new Point(boxX2, boxY2), + new Point(boxX2, boxY1)]; + + if (ring.length > 2) { + for (const corner of corners) { + if (polygonContainsPoint(ring, corner)) return true; + } + } + + for (let i = 0; i < ring.length - 1; i++) { + const p1 = ring[i]; + const p2 = ring[i + 1]; + if (edgeIntersectsBox(p1, p2, corners)) return true; + } + + return false; +} + +function edgeIntersectsBox(e1: Point, e2: Point, corners: Array) { + const tl = corners[0]; + const br = corners[2]; + // the edge and box do not intersect in either the x or y dimensions + if (((e1.x < tl.x) && (e2.x < tl.x)) || + ((e1.x > br.x) && (e2.x > br.x)) || + ((e1.y < tl.y) && (e2.y < tl.y)) || + ((e1.y > br.y) && (e2.y > br.y))) return false; + + // check if all corners of the box are on the same side of the edge + const dir = isCounterClockwise(e1, e2, corners[0]); + return dir !== isCounterClockwise(e1, e2, corners[1]) || + dir !== isCounterClockwise(e1, e2, corners[2]) || + dir !== isCounterClockwise(e1, e2, corners[3]); +} diff --git a/web/libraries/maplibre-gl/src/util/is_char_in_unicode_block.ts b/web/libraries/maplibre-gl/src/util/is_char_in_unicode_block.ts new file mode 100644 index 00000000..d1c2ba31 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/is_char_in_unicode_block.ts @@ -0,0 +1,307 @@ +// The following table comes from . +// Keep it synchronized with . + +type UnicodeBlockLookup = {[key: string]: (char: number) => boolean}; + +export const unicodeBlockLookup: UnicodeBlockLookup = { + // 'Basic Latin': (char) => char >= 0x0000 && char <= 0x007F, + 'Latin-1 Supplement': (char) => char >= 0x0080 && char <= 0x00FF, + // 'Latin Extended-A': (char) => char >= 0x0100 && char <= 0x017F, + // 'Latin Extended-B': (char) => char >= 0x0180 && char <= 0x024F, + // 'IPA Extensions': (char) => char >= 0x0250 && char <= 0x02AF, + // 'Spacing Modifier Letters': (char) => char >= 0x02B0 && char <= 0x02FF, + // 'Combining Diacritical Marks': (char) => char >= 0x0300 && char <= 0x036F, + // 'Greek and Coptic': (char) => char >= 0x0370 && char <= 0x03FF, + // 'Cyrillic': (char) => char >= 0x0400 && char <= 0x04FF, + // 'Cyrillic Supplement': (char) => char >= 0x0500 && char <= 0x052F, + // 'Armenian': (char) => char >= 0x0530 && char <= 0x058F, + //'Hebrew': (char) => char >= 0x0590 && char <= 0x05FF, + 'Arabic': (char) => char >= 0x0600 && char <= 0x06FF, + //'Syriac': (char) => char >= 0x0700 && char <= 0x074F, + 'Arabic Supplement': (char) => char >= 0x0750 && char <= 0x077F, + // 'Thaana': (char) => char >= 0x0780 && char <= 0x07BF, + // 'NKo': (char) => char >= 0x07C0 && char <= 0x07FF, + // 'Samaritan': (char) => char >= 0x0800 && char <= 0x083F, + // 'Mandaic': (char) => char >= 0x0840 && char <= 0x085F, + // 'Syriac Supplement': (char) => char >= 0x0860 && char <= 0x086F, + 'Arabic Extended-A': (char) => char >= 0x08A0 && char <= 0x08FF, + // 'Devanagari': (char) => char >= 0x0900 && char <= 0x097F, + // 'Bengali': (char) => char >= 0x0980 && char <= 0x09FF, + // 'Gurmukhi': (char) => char >= 0x0A00 && char <= 0x0A7F, + // 'Gujarati': (char) => char >= 0x0A80 && char <= 0x0AFF, + // 'Oriya': (char) => char >= 0x0B00 && char <= 0x0B7F, + // 'Tamil': (char) => char >= 0x0B80 && char <= 0x0BFF, + // 'Telugu': (char) => char >= 0x0C00 && char <= 0x0C7F, + // 'Kannada': (char) => char >= 0x0C80 && char <= 0x0CFF, + // 'Malayalam': (char) => char >= 0x0D00 && char <= 0x0D7F, + // 'Sinhala': (char) => char >= 0x0D80 && char <= 0x0DFF, + // 'Thai': (char) => char >= 0x0E00 && char <= 0x0E7F, + // 'Lao': (char) => char >= 0x0E80 && char <= 0x0EFF, + // 'Tibetan': (char) => char >= 0x0F00 && char <= 0x0FFF, + // 'Myanmar': (char) => char >= 0x1000 && char <= 0x109F, + // 'Georgian': (char) => char >= 0x10A0 && char <= 0x10FF, + 'Hangul Jamo': (char) => char >= 0x1100 && char <= 0x11FF, + // 'Ethiopic': (char) => char >= 0x1200 && char <= 0x137F, + // 'Ethiopic Supplement': (char) => char >= 0x1380 && char <= 0x139F, + // 'Cherokee': (char) => char >= 0x13A0 && char <= 0x13FF, + 'Unified Canadian Aboriginal Syllabics': (char) => char >= 0x1400 && char <= 0x167F, + // 'Ogham': (char) => char >= 0x1680 && char <= 0x169F, + // 'Runic': (char) => char >= 0x16A0 && char <= 0x16FF, + // 'Tagalog': (char) => char >= 0x1700 && char <= 0x171F, + // 'Hanunoo': (char) => char >= 0x1720 && char <= 0x173F, + // 'Buhid': (char) => char >= 0x1740 && char <= 0x175F, + // 'Tagbanwa': (char) => char >= 0x1760 && char <= 0x177F, + 'Khmer': (char) => char >= 0x1780 && char <= 0x17FF, + // 'Mongolian': (char) => char >= 0x1800 && char <= 0x18AF, + 'Unified Canadian Aboriginal Syllabics Extended': (char) => char >= 0x18B0 && char <= 0x18FF, + // 'Limbu': (char) => char >= 0x1900 && char <= 0x194F, + // 'Tai Le': (char) => char >= 0x1950 && char <= 0x197F, + // 'New Tai Lue': (char) => char >= 0x1980 && char <= 0x19DF, + // 'Khmer Symbols': (char) => char >= 0x19E0 && char <= 0x19FF, + // 'Buginese': (char) => char >= 0x1A00 && char <= 0x1A1F, + // 'Tai Tham': (char) => char >= 0x1A20 && char <= 0x1AAF, + // 'Combining Diacritical Marks Extended': (char) => char >= 0x1AB0 && char <= 0x1AFF, + // 'Balinese': (char) => char >= 0x1B00 && char <= 0x1B7F, + // 'Sundanese': (char) => char >= 0x1B80 && char <= 0x1BBF, + // 'Batak': (char) => char >= 0x1BC0 && char <= 0x1BFF, + // 'Lepcha': (char) => char >= 0x1C00 && char <= 0x1C4F, + // 'Ol Chiki': (char) => char >= 0x1C50 && char <= 0x1C7F, + // 'Cyrillic Extended-C': (char) => char >= 0x1C80 && char <= 0x1C8F, + // 'Georgian Extended': (char) => char >= 0x1C90 && char <= 0x1CBF, + // 'Sundanese Supplement': (char) => char >= 0x1CC0 && char <= 0x1CCF, + // 'Vedic Extensions': (char) => char >= 0x1CD0 && char <= 0x1CFF, + // 'Phonetic Extensions': (char) => char >= 0x1D00 && char <= 0x1D7F, + // 'Phonetic Extensions Supplement': (char) => char >= 0x1D80 && char <= 0x1DBF, + // 'Combining Diacritical Marks Supplement': (char) => char >= 0x1DC0 && char <= 0x1DFF, + // 'Latin Extended Additional': (char) => char >= 0x1E00 && char <= 0x1EFF, + // 'Greek Extended': (char) => char >= 0x1F00 && char <= 0x1FFF, + 'General Punctuation': (char) => char >= 0x2000 && char <= 0x206F, + // 'Superscripts and Subscripts': (char) => char >= 0x2070 && char <= 0x209F, + // 'Currency Symbols': (char) => char >= 0x20A0 && char <= 0x20CF, + // 'Combining Diacritical Marks for Symbols': (char) => char >= 0x20D0 && char <= 0x20FF, + 'Letterlike Symbols': (char) => char >= 0x2100 && char <= 0x214F, + 'Number Forms': (char) => char >= 0x2150 && char <= 0x218F, + // 'Arrows': (char) => char >= 0x2190 && char <= 0x21FF, + // 'Mathematical Operators': (char) => char >= 0x2200 && char <= 0x22FF, + 'Miscellaneous Technical': (char) => char >= 0x2300 && char <= 0x23FF, + 'Control Pictures': (char) => char >= 0x2400 && char <= 0x243F, + 'Optical Character Recognition': (char) => char >= 0x2440 && char <= 0x245F, + 'Enclosed Alphanumerics': (char) => char >= 0x2460 && char <= 0x24FF, + // 'Box Drawing': (char) => char >= 0x2500 && char <= 0x257F, + // 'Block Elements': (char) => char >= 0x2580 && char <= 0x259F, + 'Geometric Shapes': (char) => char >= 0x25A0 && char <= 0x25FF, + 'Miscellaneous Symbols': (char) => char >= 0x2600 && char <= 0x26FF, + // 'Dingbats': (char) => char >= 0x2700 && char <= 0x27BF, + // 'Miscellaneous Mathematical Symbols-A': (char) => char >= 0x27C0 && char <= 0x27EF, + // 'Supplemental Arrows-A': (char) => char >= 0x27F0 && char <= 0x27FF, + // 'Braille Patterns': (char) => char >= 0x2800 && char <= 0x28FF, + // 'Supplemental Arrows-B': (char) => char >= 0x2900 && char <= 0x297F, + // 'Miscellaneous Mathematical Symbols-B': (char) => char >= 0x2980 && char <= 0x29FF, + // 'Supplemental Mathematical Operators': (char) => char >= 0x2A00 && char <= 0x2AFF, + 'Miscellaneous Symbols and Arrows': (char) => char >= 0x2B00 && char <= 0x2BFF, + // 'Glagolitic': (char) => char >= 0x2C00 && char <= 0x2C5F, + // 'Latin Extended-C': (char) => char >= 0x2C60 && char <= 0x2C7F, + // 'Coptic': (char) => char >= 0x2C80 && char <= 0x2CFF, + // 'Georgian Supplement': (char) => char >= 0x2D00 && char <= 0x2D2F, + // 'Tifinagh': (char) => char >= 0x2D30 && char <= 0x2D7F, + // 'Ethiopic Extended': (char) => char >= 0x2D80 && char <= 0x2DDF, + // 'Cyrillic Extended-A': (char) => char >= 0x2DE0 && char <= 0x2DFF, + // 'Supplemental Punctuation': (char) => char >= 0x2E00 && char <= 0x2E7F, + 'CJK Radicals Supplement': (char) => char >= 0x2E80 && char <= 0x2EFF, + 'Kangxi Radicals': (char) => char >= 0x2F00 && char <= 0x2FDF, + 'Ideographic Description Characters': (char) => char >= 0x2FF0 && char <= 0x2FFF, + 'CJK Symbols and Punctuation': (char) => char >= 0x3000 && char <= 0x303F, + 'Hiragana': (char) => char >= 0x3040 && char <= 0x309F, + 'Katakana': (char) => char >= 0x30A0 && char <= 0x30FF, + 'Bopomofo': (char) => char >= 0x3100 && char <= 0x312F, + 'Hangul Compatibility Jamo': (char) => char >= 0x3130 && char <= 0x318F, + 'Kanbun': (char) => char >= 0x3190 && char <= 0x319F, + 'Bopomofo Extended': (char) => char >= 0x31A0 && char <= 0x31BF, + 'CJK Strokes': (char) => char >= 0x31C0 && char <= 0x31EF, + 'Katakana Phonetic Extensions': (char) => char >= 0x31F0 && char <= 0x31FF, + 'Enclosed CJK Letters and Months': (char) => char >= 0x3200 && char <= 0x32FF, + 'CJK Compatibility': (char) => char >= 0x3300 && char <= 0x33FF, + 'CJK Unified Ideographs Extension A': (char) => char >= 0x3400 && char <= 0x4DBF, + 'Yijing Hexagram Symbols': (char) => char >= 0x4DC0 && char <= 0x4DFF, + 'CJK Unified Ideographs': (char) => char >= 0x4E00 && char <= 0x9FFF, + 'Yi Syllables': (char) => char >= 0xA000 && char <= 0xA48F, + 'Yi Radicals': (char) => char >= 0xA490 && char <= 0xA4CF, + // 'Lisu': (char) => char >= 0xA4D0 && char <= 0xA4FF, + // 'Vai': (char) => char >= 0xA500 && char <= 0xA63F, + // 'Cyrillic Extended-B': (char) => char >= 0xA640 && char <= 0xA69F, + // 'Bamum': (char) => char >= 0xA6A0 && char <= 0xA6FF, + // 'Modifier Tone Letters': (char) => char >= 0xA700 && char <= 0xA71F, + // 'Latin Extended-D': (char) => char >= 0xA720 && char <= 0xA7FF, + // 'Syloti Nagri': (char) => char >= 0xA800 && char <= 0xA82F, + // 'Common Indic Number Forms': (char) => char >= 0xA830 && char <= 0xA83F, + // 'Phags-pa': (char) => char >= 0xA840 && char <= 0xA87F, + // 'Saurashtra': (char) => char >= 0xA880 && char <= 0xA8DF, + // 'Devanagari Extended': (char) => char >= 0xA8E0 && char <= 0xA8FF, + // 'Kayah Li': (char) => char >= 0xA900 && char <= 0xA92F, + // 'Rejang': (char) => char >= 0xA930 && char <= 0xA95F, + 'Hangul Jamo Extended-A': (char) => char >= 0xA960 && char <= 0xA97F, + // 'Javanese': (char) => char >= 0xA980 && char <= 0xA9DF, + // 'Myanmar Extended-B': (char) => char >= 0xA9E0 && char <= 0xA9FF, + // 'Cham': (char) => char >= 0xAA00 && char <= 0xAA5F, + // 'Myanmar Extended-A': (char) => char >= 0xAA60 && char <= 0xAA7F, + // 'Tai Viet': (char) => char >= 0xAA80 && char <= 0xAADF, + // 'Meetei Mayek Extensions': (char) => char >= 0xAAE0 && char <= 0xAAFF, + // 'Ethiopic Extended-A': (char) => char >= 0xAB00 && char <= 0xAB2F, + // 'Latin Extended-E': (char) => char >= 0xAB30 && char <= 0xAB6F, + // 'Cherokee Supplement': (char) => char >= 0xAB70 && char <= 0xABBF, + // 'Meetei Mayek': (char) => char >= 0xABC0 && char <= 0xABFF, + 'Hangul Syllables': (char) => char >= 0xAC00 && char <= 0xD7AF, + 'Hangul Jamo Extended-B': (char) => char >= 0xD7B0 && char <= 0xD7FF, + // 'High Surrogates': (char) => char >= 0xD800 && char <= 0xDB7F, + // 'High Private Use Surrogates': (char) => char >= 0xDB80 && char <= 0xDBFF, + // 'Low Surrogates': (char) => char >= 0xDC00 && char <= 0xDFFF, + 'Private Use Area': (char) => char >= 0xE000 && char <= 0xF8FF, + 'CJK Compatibility Ideographs': (char) => char >= 0xF900 && char <= 0xFAFF, + // 'Alphabetic Presentation Forms': (char) => char >= 0xFB00 && char <= 0xFB4F, + 'Arabic Presentation Forms-A': (char) => char >= 0xFB50 && char <= 0xFDFF, + // 'Variation Selectors': (char) => char >= 0xFE00 && char <= 0xFE0F, + 'Vertical Forms': (char) => char >= 0xFE10 && char <= 0xFE1F, + // 'Combining Half Marks': (char) => char >= 0xFE20 && char <= 0xFE2F, + 'CJK Compatibility Forms': (char) => char >= 0xFE30 && char <= 0xFE4F, + 'Small Form Variants': (char) => char >= 0xFE50 && char <= 0xFE6F, + 'Arabic Presentation Forms-B': (char) => char >= 0xFE70 && char <= 0xFEFF, + 'Halfwidth and Fullwidth Forms': (char) => char >= 0xFF00 && char <= 0xFFEF + // 'Specials': (char) => char >= 0xFFF0 && char <= 0xFFFF, + // 'Linear B Syllabary': (char) => char >= 0x10000 && char <= 0x1007F, + // 'Linear B Ideograms': (char) => char >= 0x10080 && char <= 0x100FF, + // 'Aegean Numbers': (char) => char >= 0x10100 && char <= 0x1013F, + // 'Ancient Greek Numbers': (char) => char >= 0x10140 && char <= 0x1018F, + // 'Ancient Symbols': (char) => char >= 0x10190 && char <= 0x101CF, + // 'Phaistos Disc': (char) => char >= 0x101D0 && char <= 0x101FF, + // 'Lycian': (char) => char >= 0x10280 && char <= 0x1029F, + // 'Carian': (char) => char >= 0x102A0 && char <= 0x102DF, + // 'Coptic Epact Numbers': (char) => char >= 0x102E0 && char <= 0x102FF, + // 'Old Italic': (char) => char >= 0x10300 && char <= 0x1032F, + // 'Gothic': (char) => char >= 0x10330 && char <= 0x1034F, + // 'Old Permic': (char) => char >= 0x10350 && char <= 0x1037F, + // 'Ugaritic': (char) => char >= 0x10380 && char <= 0x1039F, + // 'Old Persian': (char) => char >= 0x103A0 && char <= 0x103DF, + // 'Deseret': (char) => char >= 0x10400 && char <= 0x1044F, + // 'Shavian': (char) => char >= 0x10450 && char <= 0x1047F, + // 'Osmanya': (char) => char >= 0x10480 && char <= 0x104AF, + // 'Osage': (char) => char >= 0x104B0 && char <= 0x104FF, + // 'Elbasan': (char) => char >= 0x10500 && char <= 0x1052F, + // 'Caucasian Albanian': (char) => char >= 0x10530 && char <= 0x1056F, + // 'Linear A': (char) => char >= 0x10600 && char <= 0x1077F, + // 'Cypriot Syllabary': (char) => char >= 0x10800 && char <= 0x1083F, + // 'Imperial Aramaic': (char) => char >= 0x10840 && char <= 0x1085F, + // 'Palmyrene': (char) => char >= 0x10860 && char <= 0x1087F, + // 'Nabataean': (char) => char >= 0x10880 && char <= 0x108AF, + // 'Hatran': (char) => char >= 0x108E0 && char <= 0x108FF, + // 'Phoenician': (char) => char >= 0x10900 && char <= 0x1091F, + // 'Lydian': (char) => char >= 0x10920 && char <= 0x1093F, + // 'Meroitic Hieroglyphs': (char) => char >= 0x10980 && char <= 0x1099F, + // 'Meroitic Cursive': (char) => char >= 0x109A0 && char <= 0x109FF, + // 'Kharoshthi': (char) => char >= 0x10A00 && char <= 0x10A5F, + // 'Old South Arabian': (char) => char >= 0x10A60 && char <= 0x10A7F, + // 'Old North Arabian': (char) => char >= 0x10A80 && char <= 0x10A9F, + // 'Manichaean': (char) => char >= 0x10AC0 && char <= 0x10AFF, + // 'Avestan': (char) => char >= 0x10B00 && char <= 0x10B3F, + // 'Inscriptional Parthian': (char) => char >= 0x10B40 && char <= 0x10B5F, + // 'Inscriptional Pahlavi': (char) => char >= 0x10B60 && char <= 0x10B7F, + // 'Psalter Pahlavi': (char) => char >= 0x10B80 && char <= 0x10BAF, + // 'Old Turkic': (char) => char >= 0x10C00 && char <= 0x10C4F, + // 'Old Hungarian': (char) => char >= 0x10C80 && char <= 0x10CFF, + // 'Hanifi Rohingya': (char) => char >= 0x10D00 && char <= 0x10D3F, + // 'Rumi Numeral Symbols': (char) => char >= 0x10E60 && char <= 0x10E7F, + // 'Old Sogdian': (char) => char >= 0x10F00 && char <= 0x10F2F, + // 'Sogdian': (char) => char >= 0x10F30 && char <= 0x10F6F, + // 'Elymaic': (char) => char >= 0x10FE0 && char <= 0x10FFF, + // 'Brahmi': (char) => char >= 0x11000 && char <= 0x1107F, + // 'Kaithi': (char) => char >= 0x11080 && char <= 0x110CF, + // 'Sora Sompeng': (char) => char >= 0x110D0 && char <= 0x110FF, + // 'Chakma': (char) => char >= 0x11100 && char <= 0x1114F, + // 'Mahajani': (char) => char >= 0x11150 && char <= 0x1117F, + // 'Sharada': (char) => char >= 0x11180 && char <= 0x111DF, + // 'Sinhala Archaic Numbers': (char) => char >= 0x111E0 && char <= 0x111FF, + // 'Khojki': (char) => char >= 0x11200 && char <= 0x1124F, + // 'Multani': (char) => char >= 0x11280 && char <= 0x112AF, + // 'Khudawadi': (char) => char >= 0x112B0 && char <= 0x112FF, + // 'Grantha': (char) => char >= 0x11300 && char <= 0x1137F, + // 'Newa': (char) => char >= 0x11400 && char <= 0x1147F, + // 'Tirhuta': (char) => char >= 0x11480 && char <= 0x114DF, + // 'Siddham': (char) => char >= 0x11580 && char <= 0x115FF, + // 'Modi': (char) => char >= 0x11600 && char <= 0x1165F, + // 'Mongolian Supplement': (char) => char >= 0x11660 && char <= 0x1167F, + // 'Takri': (char) => char >= 0x11680 && char <= 0x116CF, + // 'Ahom': (char) => char >= 0x11700 && char <= 0x1173F, + // 'Dogra': (char) => char >= 0x11800 && char <= 0x1184F, + // 'Warang Citi': (char) => char >= 0x118A0 && char <= 0x118FF, + // 'Nandinagari': (char) => char >= 0x119A0 && char <= 0x119FF, + // 'Zanabazar Square': (char) => char >= 0x11A00 && char <= 0x11A4F, + // 'Soyombo': (char) => char >= 0x11A50 && char <= 0x11AAF, + // 'Pau Cin Hau': (char) => char >= 0x11AC0 && char <= 0x11AFF, + // 'Bhaiksuki': (char) => char >= 0x11C00 && char <= 0x11C6F, + // 'Marchen': (char) => char >= 0x11C70 && char <= 0x11CBF, + // 'Masaram Gondi': (char) => char >= 0x11D00 && char <= 0x11D5F, + // 'Gunjala Gondi': (char) => char >= 0x11D60 && char <= 0x11DAF, + // 'Makasar': (char) => char >= 0x11EE0 && char <= 0x11EFF, + // 'Tamil Supplement': (char) => char >= 0x11FC0 && char <= 0x11FFF, + // 'Cuneiform': (char) => char >= 0x12000 && char <= 0x123FF, + // 'Cuneiform Numbers and Punctuation': (char) => char >= 0x12400 && char <= 0x1247F, + // 'Early Dynastic Cuneiform': (char) => char >= 0x12480 && char <= 0x1254F, + // 'Egyptian Hieroglyphs': (char) => char >= 0x13000 && char <= 0x1342F, + // 'Egyptian Hieroglyph Format Controls': (char) => char >= 0x13430 && char <= 0x1343F, + // 'Anatolian Hieroglyphs': (char) => char >= 0x14400 && char <= 0x1467F, + // 'Bamum Supplement': (char) => char >= 0x16800 && char <= 0x16A3F, + // 'Mro': (char) => char >= 0x16A40 && char <= 0x16A6F, + // 'Bassa Vah': (char) => char >= 0x16AD0 && char <= 0x16AFF, + // 'Pahawh Hmong': (char) => char >= 0x16B00 && char <= 0x16B8F, + // 'Medefaidrin': (char) => char >= 0x16E40 && char <= 0x16E9F, + // 'Miao': (char) => char >= 0x16F00 && char <= 0x16F9F, + // 'Ideographic Symbols and Punctuation': (char) => char >= 0x16FE0 && char <= 0x16FFF, + // 'Tangut': (char) => char >= 0x17000 && char <= 0x187FF, + // 'Tangut Components': (char) => char >= 0x18800 && char <= 0x18AFF, + // 'Kana Supplement': (char) => char >= 0x1B000 && char <= 0x1B0FF, + // 'Kana Extended-A': (char) => char >= 0x1B100 && char <= 0x1B12F, + // 'Small Kana Extension': (char) => char >= 0x1B130 && char <= 0x1B16F, + // 'Nushu': (char) => char >= 0x1B170 && char <= 0x1B2FF, + // 'Duployan': (char) => char >= 0x1BC00 && char <= 0x1BC9F, + // 'Shorthand Format Controls': (char) => char >= 0x1BCA0 && char <= 0x1BCAF, + // 'Byzantine Musical Symbols': (char) => char >= 0x1D000 && char <= 0x1D0FF, + // 'Musical Symbols': (char) => char >= 0x1D100 && char <= 0x1D1FF, + // 'Ancient Greek Musical Notation': (char) => char >= 0x1D200 && char <= 0x1D24F, + // 'Mayan Numerals': (char) => char >= 0x1D2E0 && char <= 0x1D2FF, + // 'Tai Xuan Jing Symbols': (char) => char >= 0x1D300 && char <= 0x1D35F, + // 'Counting Rod Numerals': (char) => char >= 0x1D360 && char <= 0x1D37F, + // 'Mathematical Alphanumeric Symbols': (char) => char >= 0x1D400 && char <= 0x1D7FF, + // 'Sutton SignWriting': (char) => char >= 0x1D800 && char <= 0x1DAAF, + // 'Glagolitic Supplement': (char) => char >= 0x1E000 && char <= 0x1E02F, + // 'Nyiakeng Puachue Hmong': (char) => char >= 0x1E100 && char <= 0x1E14F, + // 'Wancho': (char) => char >= 0x1E2C0 && char <= 0x1E2FF, + // 'Mende Kikakui': (char) => char >= 0x1E800 && char <= 0x1E8DF, + // 'Adlam': (char) => char >= 0x1E900 && char <= 0x1E95F, + // 'Indic Siyaq Numbers': (char) => char >= 0x1EC70 && char <= 0x1ECBF, + // 'Ottoman Siyaq Numbers': (char) => char >= 0x1ED00 && char <= 0x1ED4F, + // 'Arabic Mathematical Alphabetic Symbols': (char) => char >= 0x1EE00 && char <= 0x1EEFF, + // 'Mahjong Tiles': (char) => char >= 0x1F000 && char <= 0x1F02F, + // 'Domino Tiles': (char) => char >= 0x1F030 && char <= 0x1F09F, + // 'Playing Cards': (char) => char >= 0x1F0A0 && char <= 0x1F0FF, + // 'Enclosed Alphanumeric Supplement': (char) => char >= 0x1F100 && char <= 0x1F1FF, + // 'Enclosed Ideographic Supplement': (char) => char >= 0x1F200 && char <= 0x1F2FF, + // 'Miscellaneous Symbols and Pictographs': (char) => char >= 0x1F300 && char <= 0x1F5FF, + // 'Emoticons': (char) => char >= 0x1F600 && char <= 0x1F64F, + // 'Ornamental Dingbats': (char) => char >= 0x1F650 && char <= 0x1F67F, + // 'Transport and Map Symbols': (char) => char >= 0x1F680 && char <= 0x1F6FF, + // 'Alchemical Symbols': (char) => char >= 0x1F700 && char <= 0x1F77F, + // 'Geometric Shapes Extended': (char) => char >= 0x1F780 && char <= 0x1F7FF, + // 'Supplemental Arrows-C': (char) => char >= 0x1F800 && char <= 0x1F8FF, + // 'Supplemental Symbols and Pictographs': (char) => char >= 0x1F900 && char <= 0x1F9FF, + // 'Chess Symbols': (char) => char >= 0x1FA00 && char <= 0x1FA6F, + // 'Symbols and Pictographs Extended-A': (char) => char >= 0x1FA70 && char <= 0x1FAFF, + // 'CJK Unified Ideographs Extension B': (char) => char >= 0x20000 && char <= 0x2A6DF, + // 'CJK Unified Ideographs Extension C': (char) => char >= 0x2A700 && char <= 0x2B73F, + // 'CJK Unified Ideographs Extension D': (char) => char >= 0x2B740 && char <= 0x2B81F, + // 'CJK Unified Ideographs Extension E': (char) => char >= 0x2B820 && char <= 0x2CEAF, + // 'CJK Unified Ideographs Extension F': (char) => char >= 0x2CEB0 && char <= 0x2EBEF, + // 'CJK Compatibility Ideographs Supplement': (char) => char >= 0x2F800 && char <= 0x2FA1F, + // 'Tags': (char) => char >= 0xE0000 && char <= 0xE007F, + // 'Variation Selectors Supplement': (char) => char >= 0xE0100 && char <= 0xE01EF, + // 'Supplementary Private Use Area-A': (char) => char >= 0xF0000 && char <= 0xFFFFF, + // 'Supplementary Private Use Area-B': (char) => char >= 0x100000 && char <= 0x10FFFF, +}; diff --git a/web/libraries/maplibre-gl/src/util/offscreen_canvas_distorted.test.ts b/web/libraries/maplibre-gl/src/util/offscreen_canvas_distorted.test.ts new file mode 100644 index 00000000..bf5c4bf4 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/offscreen_canvas_distorted.test.ts @@ -0,0 +1,13 @@ +import {isOffscreenCanvasDistorted} from './offscreen_canvas_distorted'; +import {Canvas} from 'canvas'; +import {offscreenCanvasSupported} from './offscreen_canvas_supported'; + +test('normal operation does not mangle canvas', () => { + const OffscreenCanvas = (window as any).OffscreenCanvas = jest.fn((width:number, height: number) => { + return new Canvas(width, height); + }); + expect(offscreenCanvasSupported()).toBeTruthy(); + OffscreenCanvas.mockClear(); + expect(isOffscreenCanvasDistorted()).toBeFalsy(); + expect(OffscreenCanvas).toHaveBeenCalledTimes(1); +}); diff --git a/web/libraries/maplibre-gl/src/util/offscreen_canvas_distorted.ts b/web/libraries/maplibre-gl/src/util/offscreen_canvas_distorted.ts new file mode 100644 index 00000000..15475a87 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/offscreen_canvas_distorted.ts @@ -0,0 +1,39 @@ +import {offscreenCanvasSupported} from './offscreen_canvas_supported'; + +let offscreenCanvasDistorted: boolean; + +/** + * Some browsers don't return the exact pixels from a canvas to prevent user fingerprinting (see #3185). + * This function writes pixels to an OffscreenCanvas and reads them back using getImageData, returning false + * if they don't match. + * + * @returns true if the browser supports OffscreenCanvas but it distorts getImageData results, false otherwise. + */ +export function isOffscreenCanvasDistorted(): boolean { + if (offscreenCanvasDistorted == null) { + offscreenCanvasDistorted = false; + if (offscreenCanvasSupported()) { + const size = 5; + const canvas = new OffscreenCanvas(size, size); + const context = canvas.getContext('2d', {willReadFrequently: true}); + if (context) { + // fill each pixel with an RGB value that should make the byte at index i equal to i (except alpha channel): + // [0, 1, 2, 255, 4, 5, 6, 255, 8, 9, 10, 255, ...] + for (let i = 0; i < size * size; i++) { + const base = i * 4; + context.fillStyle = `rgb(${base},${base + 1},${base + 2})`; + context.fillRect(i % size, Math.floor(i / size), 1, 1); + } + const data = context.getImageData(0, 0, size, size).data; + for (let i = 0; i < size * size * 4; i++) { + if (i % 4 !== 3 && data[i] !== i) { + offscreenCanvasDistorted = true; + break; + } + } + } + } + } + + return offscreenCanvasDistorted || false; +} diff --git a/web/libraries/maplibre-gl/src/util/offscreen_canvas_supported.ts b/web/libraries/maplibre-gl/src/util/offscreen_canvas_supported.ts new file mode 100644 index 00000000..b83bac1a --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/offscreen_canvas_supported.ts @@ -0,0 +1,11 @@ +let supportsOffscreenCanvas: boolean; + +export function offscreenCanvasSupported(): boolean { + if (supportsOffscreenCanvas == null) { + supportsOffscreenCanvas = typeof OffscreenCanvas !== 'undefined' && + new OffscreenCanvas(1, 1).getContext('2d') && + typeof createImageBitmap === 'function'; + } + + return supportsOffscreenCanvas; +} diff --git a/web/libraries/maplibre-gl/src/util/performance.ts b/web/libraries/maplibre-gl/src/util/performance.ts new file mode 100644 index 00000000..4142901f --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/performance.ts @@ -0,0 +1,117 @@ +import type {RequestParameters} from '../util/ajax'; + +export type PerformanceMetrics = { + loadTime: number; + fullLoadTime: number; + fps: number; + percentDroppedFrames: number; + totalFrames: number; +}; + +export enum PerformanceMarkers { + create = 'create', + load = 'load', + fullLoad = 'fullLoad' +} + +let lastFrameTime = null; +let frameTimes = []; + +const minFramerateTarget = 60; +const frameTimeTarget = 1000 / minFramerateTarget; + +const loadTimeKey = 'loadTime'; +const fullLoadTimeKey = 'fullLoadTime'; + +export const PerformanceUtils = { + mark(marker: PerformanceMarkers) { + performance.mark(marker); + }, + frame(timestamp: number) { + const currTimestamp = timestamp; + if (lastFrameTime != null) { + const frameTime = currTimestamp - lastFrameTime; + frameTimes.push(frameTime); + } + lastFrameTime = currTimestamp; + }, + clearMetrics() { + lastFrameTime = null; + frameTimes = []; + performance.clearMeasures(loadTimeKey); + performance.clearMeasures(fullLoadTimeKey); + + for (const marker in PerformanceMarkers) { + performance.clearMarks(PerformanceMarkers[marker]); + } + }, + + getPerformanceMetrics(): PerformanceMetrics { + performance.measure(loadTimeKey, PerformanceMarkers.create, PerformanceMarkers.load); + performance.measure(fullLoadTimeKey, PerformanceMarkers.create, PerformanceMarkers.fullLoad); + const loadTime = performance.getEntriesByName(loadTimeKey)[0].duration; + const fullLoadTime = performance.getEntriesByName(fullLoadTimeKey)[0].duration; + const totalFrames = frameTimes.length; + + const avgFrameTime = frameTimes.reduce((prev, curr) => prev + curr, 0) / totalFrames / 1000; + const fps = 1 / avgFrameTime; + + // count frames that missed our framerate target + const droppedFrames = frameTimes + .filter((frameTime) => frameTime > frameTimeTarget) + .reduce((acc, curr) => { + return acc + (curr - frameTimeTarget) / frameTimeTarget; + }, 0); + const percentDroppedFrames = (droppedFrames / (totalFrames + droppedFrames)) * 100; + + return { + loadTime, + fullLoadTime, + fps, + percentDroppedFrames, + totalFrames + }; + } +}; + +/** + * @internal + * Safe wrapper for the performance resource timing API in web workers with graceful degradation + */ +export class RequestPerformance { + _marks: { + start: string; + end: string; + measure: string; + }; + + constructor (request: RequestParameters) { + this._marks = { + start: [request.url, 'start'].join('#'), + end: [request.url, 'end'].join('#'), + measure: request.url.toString() + }; + + performance.mark(this._marks.start); + } + + finish() { + performance.mark(this._marks.end); + let resourceTimingData = performance.getEntriesByName(this._marks.measure); + + // fallback if web worker implementation of perf.getEntriesByName returns empty + if (resourceTimingData.length === 0) { + performance.measure(this._marks.measure, this._marks.start, this._marks.end); + resourceTimingData = performance.getEntriesByName(this._marks.measure); + + // cleanup + performance.clearMarks(this._marks.start); + performance.clearMarks(this._marks.end); + performance.clearMeasures(this._marks.measure); + } + + return resourceTimingData; + } +} + +export default performance; diff --git a/web/libraries/maplibre-gl/src/util/primitives.test.ts b/web/libraries/maplibre-gl/src/util/primitives.test.ts new file mode 100644 index 00000000..aa6c40b7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/primitives.test.ts @@ -0,0 +1,140 @@ +import {Aabb, Frustum} from './primitives'; +import {mat4, vec3, vec4} from 'gl-matrix'; + +describe('primitives', () => { + test('Create an aabb', () => { + const min = vec3.fromValues(0, 0, 0); + const max = vec3.fromValues(2, 4, 6); + const aabb = new Aabb(min, max); + + expect(aabb.min).toBe(min); + expect(aabb.max).toBe(max); + expect(aabb.center).toEqual([1, 2, 3]); + }); + + test('Create 4 quadrants', () => { + const min = vec3.fromValues(0, 0, 0); + const max = vec3.fromValues(2, 4, 1); + const aabb = new Aabb(min, max); + + expect(aabb.quadrant(0)).toEqual(new Aabb(vec3.fromValues(0, 0, 0), vec3.fromValues(1, 2, 1))); + expect(aabb.quadrant(1)).toEqual(new Aabb(vec3.fromValues(1, 0, 0), vec3.fromValues(2, 2, 1))); + expect(aabb.quadrant(2)).toEqual(new Aabb(vec3.fromValues(0, 2, 0), vec3.fromValues(1, 4, 1))); + expect(aabb.quadrant(3)).toEqual(new Aabb(vec3.fromValues(1, 2, 0), vec3.fromValues(2, 4, 1))); + + }); + + test('Distance to a point', () => { + const min = vec3.fromValues(-1, -1, -1); + const max = vec3.fromValues(1, 1, 1); + const aabb = new Aabb(min, max); + + expect(aabb.distanceX([0.5, -0.5])).toBe(0); + expect(aabb.distanceY([0.5, -0.5])).toBe(0); + + expect(aabb.distanceX([1, 1])).toBe(0); + expect(aabb.distanceY([1, 1])).toBe(0); + + expect(aabb.distanceX([0, 10])).toBe(0); + expect(aabb.distanceY([0, 10])).toBe(-9); + + expect(aabb.distanceX([-2, -2])).toBe(1); + expect(aabb.distanceY([-2, -2])).toBe(1); + }); + + const createTestCameraFrustum = (fovy, aspectRatio, zNear, zFar, elevation, rotation) => { + const proj = new Float64Array(16) as any as mat4; + const invProj = new Float64Array(16) as any as mat4; + + // Note that left handed coordinate space is used where z goes towards the sky. + // Y has to be flipped as well because it's part of the projection/camera matrix used in transform.js + mat4.perspective(proj, fovy, aspectRatio, zNear, zFar); + mat4.scale(proj, proj, [1, -1, 1]); + mat4.translate(proj, proj, [0, 0, elevation]); + mat4.rotateZ(proj, proj, rotation); + mat4.invert(invProj, proj); + + return Frustum.fromInvProjectionMatrix(invProj, 1.0, 0.0); + }; + + test('Aabb fully inside a frustum', () => { + const frustum = createTestCameraFrustum(Math.PI / 2, 1.0, 0.1, 100.0, -5, 0); + + // Intersection test is done in xy-plane + const aabbList = [ + new Aabb(vec3.fromValues(-1, -1, 0), vec3.fromValues(1, 1, 0)), + new Aabb(vec3.fromValues(-5, -5, 0), vec3.fromValues(5, 5, 0)), + new Aabb(vec3.fromValues(-5, -5, 0), vec3.fromValues(-4, -2, 0)) + ]; + + for (const aabb of aabbList) + expect(aabb.intersects(frustum)).toBe(2); + + }); + + test('Aabb intersecting with a frustum', () => { + const frustum = createTestCameraFrustum(Math.PI / 2, 1.0, 0.1, 100.0, -5, 0); + + const aabbList = [ + new Aabb(vec3.fromValues(-6, -6, 0), vec3.fromValues(6, 6, 0)), + new Aabb(vec3.fromValues(-6, -6, 0), vec3.fromValues(-5, -5, 0)) + ]; + + for (const aabb of aabbList) + expect(aabb.intersects(frustum)).toBe(1); + + }); + + test('No intersection between aabb and frustum', () => { + const frustum = createTestCameraFrustum(Math.PI / 2, 1.0, 0.1, 100.0, -5, 0); + + const aabbList = [ + new Aabb(vec3.fromValues(-6, 0, 0), vec3.fromValues(-5.5, 0, 0)), + new Aabb(vec3.fromValues(-6, -6, 0), vec3.fromValues(-5.5, -5.5, 0)), + new Aabb(vec3.fromValues(7, -10, 0), vec3.fromValues(7.1, 20, 0)) + ]; + + for (const aabb of aabbList) + expect(aabb.intersects(frustum)).toBe(0); + + }); + +}); + +describe('frustum', () => { + test('Create a frustum from inverse projection matrix', () => { + const proj = new Float64Array(16) as any as mat4; + const invProj = new Float64Array(16) as any as mat4; + mat4.perspective(proj, Math.PI / 2, 1.0, 0.1, 100.0); + mat4.invert(invProj, proj); + + const frustum = Frustum.fromInvProjectionMatrix(invProj, 1.0, 0.0); + // mat4.perspective generates a projection matrix for right handed coordinate space. + // This means that forward direction will be -z + const expectedFrustumPoints = [ + [-0.1, 0.1, -0.1, 1.0], + [0.1, 0.1, -0.1, 1.0], + [0.1, -0.1, -0.1, 1.0], + [-0.1, -0.1, -0.1, 1.0], + [-100.0, 100.0, -100.0, 1.0], + [100.0, 100.0, -100.0, 1.0], + [100.0, -100.0, -100.0, 1.0], + [-100.0, -100.0, -100.0, 1.0], + ]; + + frustum.points = frustum.points.map(array => array.map(n => Math.round(n * 10) / 10)) as vec4[]; + frustum.planes = frustum.planes.map(array => array.map(n => Math.round(n * 1000) / 1000)) as vec4[]; + + const expectedFrustumPlanes = [ + [0, 0, 1.0, 0.1], + [-0, -0, -1.0, -100.0], + [-0.707, 0, 0.707, -0], + [0.707, 0, 0.707, -0], + [0, -0.707, 0.707, -0], + [-0, 0.707, 0.707, -0] + ]; + + expect(frustum.points).toEqual(expectedFrustumPoints); + expect(frustum.planes).toEqual(expectedFrustumPlanes); + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/primitives.ts b/web/libraries/maplibre-gl/src/util/primitives.ts new file mode 100644 index 00000000..6f6f6011 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/primitives.ts @@ -0,0 +1,143 @@ +import {mat4, vec3, vec4} from 'gl-matrix'; + +class Frustum { + + constructor(public points: vec4[], public planes: vec4[]) { } + + public static fromInvProjectionMatrix(invProj: mat4, worldSize: number, zoom: number): Frustum { + const clipSpaceCorners = [ + [-1, 1, -1, 1], + [1, 1, -1, 1], + [1, -1, -1, 1], + [-1, -1, -1, 1], + [-1, 1, 1, 1], + [1, 1, 1, 1], + [1, -1, 1, 1], + [-1, -1, 1, 1] + ]; + + const scale = Math.pow(2, zoom); + + // Transform frustum corner points from clip space to tile space, Z to meters + const frustumCoords = clipSpaceCorners.map(v => { + v = vec4.transformMat4([] as any, v as any, invProj) as any; + const s = 1.0 / v[3] / worldSize * scale; + return vec4.mul(v as any, v as any, [s, s, 1.0 / v[3], s] as vec4); + }); + + const frustumPlanePointIndices = [ + [0, 1, 2], // near + [6, 5, 4], // far + [0, 3, 7], // left + [2, 1, 5], // right + [3, 2, 6], // bottom + [0, 4, 5] // top + ]; + + const frustumPlanes = frustumPlanePointIndices.map((p: number[]) => { + const a = vec3.sub([] as any, frustumCoords[p[0]] as vec3, frustumCoords[p[1]] as vec3); + const b = vec3.sub([] as any, frustumCoords[p[2]] as vec3, frustumCoords[p[1]] as vec3); + const n = vec3.normalize([] as any, vec3.cross([] as any, a, b)) as any; + const d = -vec3.dot(n, frustumCoords[p[1]] as vec3); + return n.concat(d); + }); + + return new Frustum(frustumCoords, frustumPlanes); + } +} + +class Aabb { + min: vec3; + max: vec3; + center: vec3; + + constructor(min_: vec3, max_: vec3) { + this.min = min_; + this.max = max_; + this.center = vec3.scale([] as any, vec3.add([] as any, this.min, this.max), 0.5); + } + + quadrant(index: number): Aabb { + const split = [(index % 2) === 0, index < 2]; + const qMin = vec3.clone(this.min); + const qMax = vec3.clone(this.max); + for (let axis = 0; axis < split.length; axis++) { + qMin[axis] = split[axis] ? this.min[axis] : this.center[axis]; + qMax[axis] = split[axis] ? this.center[axis] : this.max[axis]; + } + // Elevation is always constant, hence quadrant.max.z = this.max.z + qMax[2] = this.max[2]; + return new Aabb(qMin, qMax); + } + + distanceX(point: Array): number { + const pointOnAabb = Math.max(Math.min(this.max[0], point[0]), this.min[0]); + return pointOnAabb - point[0]; + } + + distanceY(point: Array): number { + const pointOnAabb = Math.max(Math.min(this.max[1], point[1]), this.min[1]); + return pointOnAabb - point[1]; + } + + // Performs a frustum-aabb intersection test. Returns 0 if there's no intersection, + // 1 if shapes are intersecting and 2 if the aabb if fully inside the frustum. + intersects(frustum: Frustum): number { + // Execute separating axis test between two convex objects to find intersections + // Each frustum plane together with 3 major axes define the separating axes + + const aabbPoints = [ + [this.min[0], this.min[1], this.min[2], 1], + [this.max[0], this.min[1], this.min[2], 1], + [this.max[0], this.max[1], this.min[2], 1], + [this.min[0], this.max[1], this.min[2], 1], + [this.min[0], this.min[1], this.max[2], 1], + [this.max[0], this.min[1], this.max[2], 1], + [this.max[0], this.max[1], this.max[2], 1], + [this.min[0], this.max[1], this.max[2], 1] + ]; + + let fullyInside = true; + + for (let p = 0; p < frustum.planes.length; p++) { + const plane = frustum.planes[p]; + let pointsInside = 0; + + for (let i = 0; i < aabbPoints.length; i++) { + if (vec4.dot(plane, aabbPoints[i] as any) >= 0) { + pointsInside++; + } + } + + if (pointsInside === 0) + return 0; + + if (pointsInside !== aabbPoints.length) + fullyInside = false; + } + + if (fullyInside) + return 2; + + for (let axis = 0; axis < 3; axis++) { + let projMin = Number.MAX_VALUE; + let projMax = -Number.MAX_VALUE; + + for (let p = 0; p < frustum.points.length; p++) { + const projectedPoint = frustum.points[p][axis] - this.min[axis]; + + projMin = Math.min(projMin, projectedPoint); + projMax = Math.max(projMax, projectedPoint); + } + + if (projMax < 0 || projMin > this.max[axis] - this.min[axis]) + return 0; + } + + return 1; + } +} +export { + Aabb, + Frustum +}; diff --git a/web/libraries/maplibre-gl/src/util/request_manager.test.ts b/web/libraries/maplibre-gl/src/util/request_manager.test.ts new file mode 100644 index 00000000..a3416b5d --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/request_manager.test.ts @@ -0,0 +1,21 @@ +import {RequestManager} from './request_manager'; + +describe('RequestManager.normalizeSpriteURL', () => { + test('concantenates path, ratio, and extension for non-mapbox:// scheme', () => { + expect( + new RequestManager().normalizeSpriteURL('http://www.foo.com/bar', '@2x', '.png') + ).toBe('http://www.foo.com/bar@2x.png'); + }); + + test('concantenates path, ratio, and extension for file:/// scheme', () => { + expect( + new RequestManager().normalizeSpriteURL('file:///path/to/bar', '@2x', '.png') + ).toBe('file:///path/to/bar@2x.png'); + }); + + test('normalizes non-mapbox:// scheme when query string exists', () => { + expect( + new RequestManager().normalizeSpriteURL('http://www.foo.com/bar?fresh=true', '@2x', '.png') + ).toBe('http://www.foo.com/bar@2x.png?fresh=true'); + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/request_manager.ts b/web/libraries/maplibre-gl/src/util/request_manager.ts new file mode 100644 index 00000000..c387e4b6 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/request_manager.ts @@ -0,0 +1,74 @@ +import type {RequestParameters} from './ajax'; + +/** + * A type of MapLibre resource. + */ +export const enum ResourceType { + Glyphs = 'Glyphs', + Image = 'Image', + Source = 'Source', + SpriteImage = 'SpriteImage', + SpriteJSON = 'SpriteJSON', + Style = 'Style', + Tile = 'Tile', + Unknown = 'Unknown', +} + +/** + * This function is used to tranform a request. + * It is used just before executing the relevant request. + */ +export type RequestTransformFunction = (url: string, resourceType?: ResourceType) => RequestParameters | undefined; + +type UrlObject = { + protocol: string; + authority: string; + path: string; + params: Array; +}; + +export class RequestManager { + _transformRequestFn: RequestTransformFunction; + + constructor(transformRequestFn?: RequestTransformFunction) { + this._transformRequestFn = transformRequestFn; + } + + transformRequest(url: string, type: ResourceType) { + if (this._transformRequestFn) { + return this._transformRequestFn(url, type) || {url}; + } + + return {url}; + } + + normalizeSpriteURL(url: string, format: string, extension: string): string { + const urlObject = parseUrl(url); + urlObject.path += `${format}${extension}`; + return formatUrl(urlObject); + } + + setTransformRequest(transformRequest: RequestTransformFunction) { + this._transformRequestFn = transformRequest; + } +} + +const urlRe = /^(\w+):\/\/([^/?]*)(\/[^?]+)?\??(.+)?/; + +function parseUrl(url: string): UrlObject { + const parts = url.match(urlRe); + if (!parts) { + throw new Error(`Unable to parse URL "${url}"`); + } + return { + protocol: parts[1], + authority: parts[2], + path: parts[3] || '/', + params: parts[4] ? parts[4].split('&') : [] + }; +} + +function formatUrl(obj: UrlObject): string { + const params = obj.params.length ? `?${obj.params.join('&')}` : ''; + return `${obj.protocol}://${obj.authority}${obj.path}${params}`; +} diff --git a/web/libraries/maplibre-gl/src/util/resolve_tokens.test.ts b/web/libraries/maplibre-gl/src/util/resolve_tokens.test.ts new file mode 100644 index 00000000..ed9fbe88 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/resolve_tokens.test.ts @@ -0,0 +1,45 @@ +import {resolveTokens} from './resolve_tokens'; + +test('resolveToken', () => { + expect('3 Fine Fields').toBe(resolveTokens({a: 3, b: 'Fine', c: 'Fields'}, '{a} {b} {c}')); + + // No tokens. + expect(resolveTokens({}, 'Test')).toBe('Test'); + + // Basic. + expect(resolveTokens({name: 'Test'}, '{name}')).toBe('Test'); + expect(resolveTokens({name: 'Test'}, '{name}-suffix')).toBe('Test-suffix'); + + // No properties. + expect(resolveTokens(null, '{name}')).toBe(''); + expect(resolveTokens(null, '{name}-suffix')).toBe('-suffix'); + + // Undefined property. + expect(resolveTokens({}, '{name}')).toBe(''); + expect(resolveTokens({}, '{name}-suffix')).toBe('-suffix'); + + // Non-latin. + expect(resolveTokens({city: '서울특별시'}, '{city}')).toBe('서울특별시'); + + // Unicode up to 65535. + expect(resolveTokens({text: '\ufff0'}, '{text}')).toBe('\ufff0'); + expect(resolveTokens({text: '\uffff'}, '{text}')).toBe('\uffff'); + + // Non-string values cast to strings. + expect(resolveTokens({name: 5000}, '{name}')).toBe('5000'); + expect(resolveTokens({name: -15.5}, '{name}')).toBe('-15.5'); + expect(resolveTokens({name: true}, '{name}')).toBe('true'); + + // Non-string values cast to strings, with token replacement. + expect(resolveTokens({name: 5000}, '{name}-suffix')).toBe('5000-suffix'); + expect(resolveTokens({name: -15.5}, '{name}-suffix')).toBe('-15.5-suffix'); + expect(resolveTokens({name: true}, '{name}-suffix')).toBe('true-suffix'); + + // Special characters in token. + expect(resolveTokens({'dashed-property': 'dashed'}, '{dashed-property}')).toBe('dashed'); + expect(resolveTokens({'HØYDE': 150}, '{HØYDE} m')).toBe('150 m'); + expect( + resolveTokens({'$special:characters;': 'mapbox'}, '{$special:characters;}') + ).toBe('mapbox'); + +}); diff --git a/web/libraries/maplibre-gl/src/util/resolve_tokens.ts b/web/libraries/maplibre-gl/src/util/resolve_tokens.ts new file mode 100644 index 00000000..af0e8f23 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/resolve_tokens.ts @@ -0,0 +1,17 @@ +/** + * Replace tokens in a string template with values in an object + * + * @param properties - a key/value relationship between tokens and replacements + * @param text - the template string + * @returns the template with tokens replaced + */ +export function resolveTokens( + properties: { + readonly [x: string]: unknown; + } | null, + text: string +): string { + return text.replace(/{([^{}]+)}/g, (match, key: string) => { + return properties && key in properties ? String(properties[key]) : ''; + }); +} diff --git a/web/libraries/maplibre-gl/src/util/script_detection.ts b/web/libraries/maplibre-gl/src/util/script_detection.ts new file mode 100644 index 00000000..6e916c08 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/script_detection.ts @@ -0,0 +1,323 @@ +/* eslint-disable new-cap */ + +import {unicodeBlockLookup as isChar} from './is_char_in_unicode_block'; + +export function allowsIdeographicBreaking(chars: string) { + for (const char of chars) { + if (!charAllowsIdeographicBreaking(char.charCodeAt(0))) return false; + } + return true; +} + +export function allowsVerticalWritingMode(chars: string) { + for (const char of chars) { + if (charHasUprightVerticalOrientation(char.charCodeAt(0))) return true; + } + return false; +} + +export function allowsLetterSpacing(chars: string) { + for (const char of chars) { + if (!charAllowsLetterSpacing(char.charCodeAt(0))) return false; + } + return true; +} + +export function charAllowsLetterSpacing(char: number) { + if (isChar['Arabic'](char)) return false; + if (isChar['Arabic Supplement'](char)) return false; + if (isChar['Arabic Extended-A'](char)) return false; + if (isChar['Arabic Presentation Forms-A'](char)) return false; + if (isChar['Arabic Presentation Forms-B'](char)) return false; + + return true; +} + +export function charAllowsIdeographicBreaking(char: number) { + // Return early for characters outside all ideographic ranges. + if (char < 0x2E80) return false; + + if (isChar['Bopomofo Extended'](char)) return true; + if (isChar['Bopomofo'](char)) return true; + if (isChar['CJK Compatibility Forms'](char)) return true; + if (isChar['CJK Compatibility Ideographs'](char)) return true; + if (isChar['CJK Compatibility'](char)) return true; + if (isChar['CJK Radicals Supplement'](char)) return true; + if (isChar['CJK Strokes'](char)) return true; + if (isChar['CJK Symbols and Punctuation'](char)) return true; + if (isChar['CJK Unified Ideographs Extension A'](char)) return true; + if (isChar['CJK Unified Ideographs'](char)) return true; + if (isChar['Enclosed CJK Letters and Months'](char)) return true; + if (isChar['Halfwidth and Fullwidth Forms'](char)) return true; + if (isChar['Hiragana'](char)) return true; + if (isChar['Ideographic Description Characters'](char)) return true; + if (isChar['Kangxi Radicals'](char)) return true; + if (isChar['Katakana Phonetic Extensions'](char)) return true; + if (isChar['Katakana'](char)) return true; + if (isChar['Vertical Forms'](char)) return true; + if (isChar['Yi Radicals'](char)) return true; + if (isChar['Yi Syllables'](char)) return true; + + return false; +} + +// The following logic comes from +// . +// Keep it synchronized with +// . +// The data file denotes with “U” or “Tu” any codepoint that may be drawn +// upright in vertical text but does not distinguish between upright and +// “neutral” characters. + +// Blocks in the Unicode supplementary planes are excluded from this module due +// to . + +/** + * Returns true if the given Unicode codepoint identifies a character with + * upright orientation. + * + * A character has upright orientation if it is drawn upright (unrotated) + * whether the line is oriented horizontally or vertically, even if both + * adjacent characters can be rotated. For example, a Chinese character is + * always drawn upright. An uprightly oriented character causes an adjacent + * “neutral” character to be drawn upright as well. + */ +export function charHasUprightVerticalOrientation(char: number) { + if (char === 0x02EA /* modifier letter yin departing tone mark */ || + char === 0x02EB /* modifier letter yang departing tone mark */) { + return true; + } + + // Return early for characters outside all ranges whose characters remain + // upright in vertical writing mode. + if (char < 0x1100) return false; + + if (isChar['Bopomofo Extended'](char)) return true; + if (isChar['Bopomofo'](char)) return true; + if (isChar['CJK Compatibility Forms'](char)) { + if (!((char >= 0xFE49 /* dashed overline */ && char <= 0xFE4F) /* wavy low line */)) { + return true; + } + } + if (isChar['CJK Compatibility Ideographs'](char)) return true; + if (isChar['CJK Compatibility'](char)) return true; + if (isChar['CJK Radicals Supplement'](char)) return true; + if (isChar['CJK Strokes'](char)) return true; + if (isChar['CJK Symbols and Punctuation'](char)) { + if (!((char >= 0x3008 /* left angle bracket */ && char <= 0x3011) /* right black lenticular bracket */) && + !((char >= 0x3014 /* left tortoise shell bracket */ && char <= 0x301F) /* low double prime quotation mark */) && + char !== 0x3030 /* wavy dash */) { + return true; + } + } + if (isChar['CJK Unified Ideographs Extension A'](char)) return true; + if (isChar['CJK Unified Ideographs'](char)) return true; + if (isChar['Enclosed CJK Letters and Months'](char)) return true; + if (isChar['Hangul Compatibility Jamo'](char)) return true; + if (isChar['Hangul Jamo Extended-A'](char)) return true; + if (isChar['Hangul Jamo Extended-B'](char)) return true; + if (isChar['Hangul Jamo'](char)) return true; + if (isChar['Hangul Syllables'](char)) return true; + if (isChar['Hiragana'](char)) return true; + if (isChar['Ideographic Description Characters'](char)) return true; + if (isChar['Kanbun'](char)) return true; + if (isChar['Kangxi Radicals'](char)) return true; + if (isChar['Katakana Phonetic Extensions'](char)) return true; + if (isChar['Katakana'](char)) { + if (char !== 0x30FC /* katakana-hiragana prolonged sound mark */) { + return true; + } + } + if (isChar['Halfwidth and Fullwidth Forms'](char)) { + if (char !== 0xFF08 /* fullwidth left parenthesis */ && + char !== 0xFF09 /* fullwidth right parenthesis */ && + char !== 0xFF0D /* fullwidth hyphen-minus */ && + !((char >= 0xFF1A /* fullwidth colon */ && char <= 0xFF1E) /* fullwidth greater-than sign */) && + char !== 0xFF3B /* fullwidth left square bracket */ && + char !== 0xFF3D /* fullwidth right square bracket */ && + char !== 0xFF3F /* fullwidth low line */ && + !(char >= 0xFF5B /* fullwidth left curly bracket */ && char <= 0xFFDF) && + char !== 0xFFE3 /* fullwidth macron */ && + !(char >= 0xFFE8 /* halfwidth forms light vertical */ && char <= 0xFFEF)) { + return true; + } + } + if (isChar['Small Form Variants'](char)) { + if (!((char >= 0xFE58 /* small em dash */ && char <= 0xFE5E) /* small right tortoise shell bracket */) && + !((char >= 0xFE63 /* small hyphen-minus */ && char <= 0xFE66) /* small equals sign */)) { + return true; + } + } + if (isChar['Unified Canadian Aboriginal Syllabics'](char)) return true; + if (isChar['Unified Canadian Aboriginal Syllabics Extended'](char)) return true; + if (isChar['Vertical Forms'](char)) return true; + if (isChar['Yijing Hexagram Symbols'](char)) return true; + if (isChar['Yi Syllables'](char)) return true; + if (isChar['Yi Radicals'](char)) return true; + + return false; +} + +/** + * Returns true if the given Unicode codepoint identifies a character with + * neutral orientation. + * + * A character has neutral orientation if it may be drawn rotated or unrotated + * when the line is oriented vertically, depending on the orientation of the + * adjacent characters. For example, along a verticlly oriented line, the vulgar + * fraction ½ is drawn upright among Chinese characters but rotated among Latin + * letters. A neutrally oriented character does not influence whether an + * adjacent character is drawn upright or rotated. + */ +export function charHasNeutralVerticalOrientation(char: number) { + if (isChar['Latin-1 Supplement'](char)) { + if (char === 0x00A7 /* section sign */ || + char === 0x00A9 /* copyright sign */ || + char === 0x00AE /* registered sign */ || + char === 0x00B1 /* plus-minus sign */ || + char === 0x00BC /* vulgar fraction one quarter */ || + char === 0x00BD /* vulgar fraction one half */ || + char === 0x00BE /* vulgar fraction three quarters */ || + char === 0x00D7 /* multiplication sign */ || + char === 0x00F7 /* division sign */) { + return true; + } + } + if (isChar['General Punctuation'](char)) { + if (char === 0x2016 /* double vertical line */ || + char === 0x2020 /* dagger */ || + char === 0x2021 /* double dagger */ || + char === 0x2030 /* per mille sign */ || + char === 0x2031 /* per ten thousand sign */ || + char === 0x203B /* reference mark */ || + char === 0x203C /* double exclamation mark */ || + char === 0x2042 /* asterism */ || + char === 0x2047 /* double question mark */ || + char === 0x2048 /* question exclamation mark */ || + char === 0x2049 /* exclamation question mark */ || + char === 0x2051 /* two asterisks aligned vertically */) { + return true; + } + } + if (isChar['Letterlike Symbols'](char)) return true; + if (isChar['Number Forms'](char)) return true; + if (isChar['Miscellaneous Technical'](char)) { + if ((char >= 0x2300 /* diameter sign */ && char <= 0x2307 /* wavy line */) || + (char >= 0x230C /* bottom right crop */ && char <= 0x231F /* bottom right corner */) || + (char >= 0x2324 /* up arrowhead between two horizontal bars */ && char <= 0x2328 /* keyboard */) || + char === 0x232B /* erase to the left */ || + (char >= 0x237D /* shouldered open box */ && char <= 0x239A /* clear screen symbol */) || + (char >= 0x23BE /* dentistry symbol light vertical and top right */ && char <= 0x23CD /* square foot */) || + char === 0x23CF /* eject symbol */ || + (char >= 0x23D1 /* metrical breve */ && char <= 0x23DB /* fuse */) || + (char >= 0x23E2 /* white trapezium */ && char <= 0x23FF)) { + return true; + } + } + if (isChar['Control Pictures'](char) && char !== 0x2423 /* open box */) return true; + if (isChar['Optical Character Recognition'](char)) return true; + if (isChar['Enclosed Alphanumerics'](char)) return true; + if (isChar['Geometric Shapes'](char)) return true; + if (isChar['Miscellaneous Symbols'](char)) { + if (!((char >= 0x261A /* black left pointing index */ && char <= 0x261F) /* white down pointing index */)) { + return true; + } + } + if (isChar['Miscellaneous Symbols and Arrows'](char)) { + if ((char >= 0x2B12 /* square with top half black */ && char <= 0x2B2F /* white vertical ellipse */) || + (char >= 0x2B50 /* white medium star */ && char <= 0x2B59 /* heavy circled saltire */) || + (char >= 0x2BB8 /* upwards white arrow from bar with horizontal bar */ && char <= 0x2BEB)) { + return true; + } + } + if (isChar['CJK Symbols and Punctuation'](char)) return true; + if (isChar['Katakana'](char)) return true; + if (isChar['Private Use Area'](char)) return true; + if (isChar['CJK Compatibility Forms'](char)) return true; + if (isChar['Small Form Variants'](char)) return true; + if (isChar['Halfwidth and Fullwidth Forms'](char)) return true; + + if (char === 0x221E /* infinity */ || + char === 0x2234 /* therefore */ || + char === 0x2235 /* because */ || + (char >= 0x2700 /* black safety scissors */ && char <= 0x2767 /* rotated floral heart bullet */) || + (char >= 0x2776 /* dingbat negative circled digit one */ && char <= 0x2793 /* dingbat negative circled sans-serif number ten */) || + char === 0xFFFC /* object replacement character */ || + char === 0xFFFD /* replacement character */) { + return true; + } + + return false; +} + +/** + * Returns true if the given Unicode codepoint identifies a character with + * rotated orientation. + * + * A character has rotated orientation if it is drawn rotated when the line is + * oriented vertically, even if both adjacent characters are upright. For + * example, a Latin letter is drawn rotated along a vertical line. A rotated + * character causes an adjacent “neutral” character to be drawn rotated as well. + */ +export function charHasRotatedVerticalOrientation(char: number) { + return !(charHasUprightVerticalOrientation(char) || + charHasNeutralVerticalOrientation(char)); +} + +export function charInComplexShapingScript(char: number) { + return isChar['Arabic'](char) || + isChar['Arabic Supplement'](char) || + isChar['Arabic Extended-A'](char) || + isChar['Arabic Presentation Forms-A'](char) || + isChar['Arabic Presentation Forms-B'](char); +} + +export function charInRTLScript(char: number) { + // Main blocks for Hebrew, Arabic, Thaana and other RTL scripts + return (char >= 0x0590 && char <= 0x08FF) || + isChar['Arabic Presentation Forms-A'](char) || + isChar['Arabic Presentation Forms-B'](char); +} + +export function charInSupportedScript(char: number, canRenderRTL: boolean) { + // This is a rough heuristic: whether we "can render" a script + // actually depends on the properties of the font being used + // and whether differences from the ideal rendering are considered + // semantically significant. + + // Even in Latin script, we "can't render" combinations such as the fi + // ligature, but we don't consider that semantically significant. + if (!canRenderRTL && charInRTLScript(char)) { + return false; + } + if ((char >= 0x0900 && char <= 0x0DFF) || + // Main blocks for Indic scripts and Sinhala + (char >= 0x0F00 && char <= 0x109F) || + // Main blocks for Tibetan and Myanmar + isChar['Khmer'](char)) { + // These blocks cover common scripts that require + // complex text shaping, based on unicode script metadata: + // http://www.unicode.org/repos/cldr/trunk/common/properties/scriptMetadata.txt + // where "Web Rank <= 32" "Shaping Required = YES" + return false; + } + return true; +} + +export function stringContainsRTLText(chars: string): boolean { + for (const char of chars) { + if (charInRTLScript(char.charCodeAt(0))) { + return true; + } + } + return false; +} + +export function isStringInSupportedScript(chars: string, canRenderRTL: boolean) { + for (const char of chars) { + if (!charInSupportedScript(char.charCodeAt(0), canRenderRTL)) { + return false; + } + } + return true; +} diff --git a/web/libraries/maplibre-gl/src/util/smart_wrap.ts b/web/libraries/maplibre-gl/src/util/smart_wrap.ts new file mode 100644 index 00000000..71f9e2b9 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/smart_wrap.ts @@ -0,0 +1,51 @@ +import {LngLat} from '../geo/lng_lat'; + +import type Point from '@mapbox/point-geometry'; +import type {Transform} from '../geo/transform'; + +/** + * Given a LngLat, prior projected position, and a transform, return a new LngLat shifted + * n × 360° east or west for some n ≥ 0 such that: + * + * * the projected location of the result is on screen, if possible, and secondarily: + * * the difference between the projected location of the result and the prior position + * is minimized. + * + * The object is to preserve perceived object constancy for Popups and Markers as much as + * possible; they should avoid shifting large distances across the screen, even when the + * map center changes by ±360° due to automatic wrapping, and when about to go off screen, + * should wrap just enough to avoid doing so. + */ +export function smartWrap(lngLat: LngLat, priorPos: Point, transform: Transform): LngLat { + lngLat = new LngLat(lngLat.lng, lngLat.lat); + + // First, try shifting one world in either direction, and see if either is closer to the + // prior position. This preserves object constancy when the map center is auto-wrapped + // during animations. + if (priorPos) { + const left = new LngLat(lngLat.lng - 360, lngLat.lat); + const right = new LngLat(lngLat.lng + 360, lngLat.lat); + const delta = transform.locationPoint(lngLat).distSqr(priorPos); + if (transform.locationPoint(left).distSqr(priorPos) < delta) { + lngLat = left; + } else if (transform.locationPoint(right).distSqr(priorPos) < delta) { + lngLat = right; + } + } + + // Second, wrap toward the center until the new position is on screen, or we can't get + // any closer. + while (Math.abs(lngLat.lng - transform.center.lng) > 180) { + const pos = transform.locationPoint(lngLat); + if (pos.x >= 0 && pos.y >= 0 && pos.x <= transform.width && pos.y <= transform.height) { + break; + } + if (lngLat.lng > transform.center.lng) { + lngLat.lng -= 360; + } else { + lngLat.lng += 360; + } + } + + return lngLat; +} diff --git a/web/libraries/maplibre-gl/src/util/struct_array.test.ts b/web/libraries/maplibre-gl/src/util/struct_array.test.ts new file mode 100644 index 00000000..77e2d161 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/struct_array.test.ts @@ -0,0 +1,101 @@ +import {StructArrayLayout3i6, FeatureIndexArray} from '../data/array_types.g'; + +describe('StructArray', () => { + class TestArray extends StructArrayLayout3i6 {} + + test('array constructs itself', () => { + const array = new TestArray(); + expect(array).toHaveLength(0); + expect(array.arrayBuffer).toBeTruthy(); + }); + + test('emplaceBack', () => { + const array = new TestArray(); + + expect(0).toBe(array.emplaceBack(1, 7, 3)); + expect(1).toBe(array.emplaceBack(4, 2, 5)); + + expect(array).toHaveLength(2); + + expect(array.int16.slice(0, 6)).toEqual(new Int16Array([1, 7, 3, 4, 2, 5])); + }); + + test('emplaceBack gracefully accepts extra arguments', () => { + // emplaceBack is typically used in fairly hot code paths, where + // conditionally varying the number of arguments can be expensive. + const array = new TestArray(); + expect((array as any).emplaceBack(3, 1, 4, 1, 5, 9)).toBe(0); + expect(array).toHaveLength(1); + expect(array.int16.slice(0, 3)).toEqual(new Int16Array([3, 1, 4])); + }); + + test('reserve', () => { + const array = new TestArray(); + + array.reserve(100); + const initialCapacity = array.capacity; + + for (let i = 0; i < 100; i++) { + array.emplaceBack(1, 1, 1); + expect(array.capacity).toBe(initialCapacity); + } + }); + + test('automatically resizes', () => { + const array = new TestArray(); + const initialCapacity = array.capacity; + + while (initialCapacity > array.length) { + array.emplaceBack(1, 1, 1); + } + + expect(array.capacity).toBe(initialCapacity); + + array.emplaceBack(1, 1, 1); + expect(array.capacity > initialCapacity).toBeTruthy(); + }); + + test('trims', () => { + const array = new TestArray(); + const capacityInitial = array.capacity; + + array.emplaceBack(1, 1, 1); + expect(array.capacity).toBe(capacityInitial); + + array._trim(); + expect(array.capacity).toBe(1); + expect(array.arrayBuffer.byteLength).toBe(array.bytesPerElement); + }); +}); + +describe('FeatureIndexArray', () => { + class TestArray extends FeatureIndexArray {} + + test('array constructs itself', () => { + const array = new TestArray(); + expect(array).toHaveLength(0); + expect(array.arrayBuffer).toBeTruthy(); + }); + + test('emplace and retrieve', () => { + const array = new TestArray(); + expect(0).toBe(array.emplaceBack(1, 7, 3)); + expect(1).toBe(array.emplaceBack(4, 2, 5)); + + expect(array).toHaveLength(2); + + const elem0 = array.get(0); + expect(elem0).toBeTruthy(); + + expect(elem0.featureIndex).toBe(1); + expect(elem0.sourceLayerIndex).toBe(7); + expect(elem0.bucketIndex).toBe(3); + + const elem1 = array.get(1); + expect(elem1).toBeTruthy(); + + expect(elem1.featureIndex).toBe(4); + expect(elem1.sourceLayerIndex).toBe(2); + expect(elem1.bucketIndex).toBe(5); + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/struct_array.ts b/web/libraries/maplibre-gl/src/util/struct_array.ts new file mode 100644 index 00000000..8ab3e495 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/struct_array.ts @@ -0,0 +1,248 @@ +// Note: all "sizes" are measured in bytes + +import type {Transferable} from '../types/transferable'; + +/** + * @internal + * A view type size + */ +const viewTypes = { + 'Int8': Int8Array, + 'Uint8': Uint8Array, + 'Int16': Int16Array, + 'Uint16': Uint16Array, + 'Int32': Int32Array, + 'Uint32': Uint32Array, + 'Float32': Float32Array +}; + +/** + * @internal + * A view type size + */ +export type ViewType = keyof typeof viewTypes; + +/** @internal */ +class Struct { + _pos1: number; + _pos2: number; + _pos4: number; + _pos8: number; + readonly _structArray: StructArray; + + // The following properties are defined on the prototype of sub classes. + size: number; + + /** + * @param structArray - The StructArray the struct is stored in + * @param index - The index of the struct in the StructArray. + */ + constructor(structArray: StructArray, index: number) { + (this as any)._structArray = structArray; + this._pos1 = index * this.size; + this._pos2 = this._pos1 / 2; + this._pos4 = this._pos1 / 4; + this._pos8 = this._pos1 / 8; + } +} + +const DEFAULT_CAPACITY = 128; +const RESIZE_MULTIPLIER = 5; + +/** + * @internal + * A struct array memeber + */ +export type StructArrayMember = { + name: string; + type: ViewType; + components: number; + offset: number; +}; + +export type StructArrayLayout = { + members: Array; + size: number; + alignment: number; +}; + +/** + * An array that can be desialized + */ +export type SerializedStructArray = { + length: number; + arrayBuffer: ArrayBuffer; +}; + +/** + * @internal + * `StructArray` provides an abstraction over `ArrayBuffer` and `TypedArray` + * making it behave like an array of typed structs. + * + * Conceptually, a StructArray is comprised of elements, i.e., instances of its + * associated struct type. Each particular struct type, together with an + * alignment size, determines the memory layout of a StructArray whose elements + * are of that type. Thus, for each such layout that we need, we have + * a corresponding StructArrayLayout class, inheriting from StructArray and + * implementing `emplaceBack()` and `_refreshViews()`. + * + * In some cases, where we need to access particular elements of a StructArray, + * we implement a more specific subclass that inherits from one of the + * StructArrayLayouts and adds a `get(i): T` accessor that returns a structured + * object whose properties are proxies into the underlying memory space for the + * i-th element. This affords the convience of working with (seemingly) plain + * Javascript objects without the overhead of serializing/deserializing them + * into ArrayBuffers for efficient web worker transfer. + */ +abstract class StructArray { + capacity: number; + length: number; + isTransferred: boolean; + arrayBuffer: ArrayBuffer; + uint8: Uint8Array; + + // The following properties are defined on the prototype. + members: Array; + bytesPerElement: number; + abstract emplaceBack(...v: number[]); + abstract emplace(i: number, ...v: number[]); + + constructor() { + this.isTransferred = false; + this.capacity = -1; + this.resize(0); + } + + /** + * Serialize a StructArray instance. Serializes both the raw data and the + * metadata needed to reconstruct the StructArray base class during + * deserialization. + */ + static serialize(array: StructArray, transferables?: Array): SerializedStructArray { + + array._trim(); + + if (transferables) { + array.isTransferred = true; + transferables.push(array.arrayBuffer); + } + + return { + length: array.length, + arrayBuffer: array.arrayBuffer, + }; + } + + static deserialize(input: SerializedStructArray) { + const structArray = Object.create(this.prototype); + structArray.arrayBuffer = input.arrayBuffer; + structArray.length = input.length; + structArray.capacity = input.arrayBuffer.byteLength / structArray.bytesPerElement; + structArray._refreshViews(); + return structArray; + } + + /** + * Resize the array to discard unused capacity. + */ + _trim() { + if (this.length !== this.capacity) { + this.capacity = this.length; + this.arrayBuffer = this.arrayBuffer.slice(0, this.length * this.bytesPerElement); + this._refreshViews(); + } + } + + /** + * Resets the length of the array to 0 without de-allocating capcacity. + */ + clear() { + this.length = 0; + } + + /** + * Resize the array. + * If `n` is greater than the current length then additional elements with undefined values are added. + * If `n` is less than the current length then the array will be reduced to the first `n` elements. + * @param n - The new size of the array. + */ + resize(n: number) { + this.reserve(n); + this.length = n; + } + + /** + * Indicate a planned increase in size, so that any necessary allocation may + * be done once, ahead of time. + * @param n - The expected size of the array. + */ + reserve(n: number) { + if (n > this.capacity) { + this.capacity = Math.max(n, Math.floor(this.capacity * RESIZE_MULTIPLIER), DEFAULT_CAPACITY); + this.arrayBuffer = new ArrayBuffer(this.capacity * this.bytesPerElement); + + const oldUint8Array = this.uint8; + this._refreshViews(); + if (oldUint8Array) this.uint8.set(oldUint8Array); + } + } + + /** + * Create TypedArray views for the current ArrayBuffer. + */ + _refreshViews() { + throw new Error('_refreshViews() must be implemented by each concrete StructArray layout'); + } +} + +/** + * Given a list of member fields, create a full StructArrayLayout, in + * particular calculating the correct byte offset for each field. This data + * is used at build time to generate StructArrayLayout_*#emplaceBack() and + * other accessors, and at runtime for binding vertex buffer attributes. + */ +function createLayout( + members: Array<{ + name: string; + type: ViewType; + readonly components?: number; + }>, + alignment: number = 1 +): StructArrayLayout { + + let offset = 0; + let maxSize = 0; + const layoutMembers = members.map((member) => { + const typeSize = sizeOf(member.type); + const memberOffset = offset = align(offset, Math.max(alignment, typeSize)); + const components = member.components || 1; + + maxSize = Math.max(maxSize, typeSize); + offset += typeSize * components; + + return { + name: member.name, + type: member.type, + components, + offset: memberOffset, + }; + }); + + const size = align(offset, Math.max(maxSize, alignment)); + + return { + members: layoutMembers, + size, + alignment + }; +} + +function sizeOf(type: ViewType): number { + return viewTypes[type].BYTES_PER_ELEMENT; +} + +function align(offset: number, size: number): number { + return Math.ceil(offset / size) * size; +} + +export {StructArray, Struct, viewTypes, createLayout}; diff --git a/web/libraries/maplibre-gl/src/util/style.test.ts b/web/libraries/maplibre-gl/src/util/style.test.ts new file mode 100644 index 00000000..7b5ea760 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/style.test.ts @@ -0,0 +1,34 @@ +import {coerceSpriteToArray} from './style'; + +describe('style utils', () => { + describe('#coerceSpriteToArray', () => { + test('input === output when array', () => { + const inputSpriteArray = [{id: 'id', url: 'url'}]; + const outputSpriteArray = coerceSpriteToArray(inputSpriteArray); + expect(outputSpriteArray).toHaveLength(1); + expect(outputSpriteArray[0].id).toBe('id'); + expect(outputSpriteArray[0].url).toBe('url'); + }); + + test('coerced to array when string', () => { + const expected = [{id: 'default', url: 'url'}]; + expect(coerceSpriteToArray('url')).toEqual(expected); + }); + + test('returns an empty array when nothing\'s passed in', () => { + expect(coerceSpriteToArray()).toEqual([]); + }); + + test('duplicated entries should be removed', () => { + const spriteWithDuplicatedEntries = [ + {id: 'sprite1', url: 'http://www.dummy.com'}, + {id: 'sprite1', url: 'http://www.dummy.com'}]; + + const result = coerceSpriteToArray(spriteWithDuplicatedEntries); + + expect(result).toHaveLength(1); + expect(result[0].id).toBe('sprite1'); + expect(result[0].url).toBe('http://www.dummy.com'); + }); + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/style.ts b/web/libraries/maplibre-gl/src/util/style.ts new file mode 100644 index 00000000..9c3db2bf --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/style.ts @@ -0,0 +1,28 @@ +import {SpriteSpecification} from '@maplibre/maplibre-gl-style-spec'; + +/** + * Takes a SpriteSpecification value and returns it in its array form. If `undefined` is passed as an input value, an + * empty array is returned. + * duplicated entries with identical id/url will be removed in returned array + * @param sprite - optional sprite to coerce + * @returns an empty array in case `undefined` is passed; id-url pairs otherwise + */ +export function coerceSpriteToArray(sprite?: SpriteSpecification): {id: string; url: string}[] { + const resultArray: {id: string; url: string}[] = []; + + if (typeof sprite === 'string') { + resultArray.push({id: 'default', url: sprite}); + } else if (sprite && sprite.length > 0) { + const dedupArray: string[] = []; + for (const {id, url} of sprite) { + const key = `${id}${url}`; + if (dedupArray.indexOf(key) === -1) { + dedupArray.push(key); + resultArray.push({id, url}); + } + } + } + + return resultArray; + +} diff --git a/web/libraries/maplibre-gl/src/util/task_queue.test.ts b/web/libraries/maplibre-gl/src/util/task_queue.test.ts new file mode 100644 index 00000000..d3084024 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/task_queue.test.ts @@ -0,0 +1,114 @@ +import {TaskQueue} from './task_queue'; + +describe('TaskQueue', () => { + test('Calls callbacks, in order', () => { + const q = new TaskQueue(); + let first = 0; + let second = 0; + q.add(() => { + expect(++first).toBe(1); + expect(second).toBe(0); + }); + q.add(() => { + expect(first).toBe(1); + expect(++second).toBe(1); + }); + q.run(); + expect(first).toBe(1); + expect(second).toBe(1); + }); + + test('Allows a given callback to be queued multiple times', () => { + const q = new TaskQueue(); + const fn = jest.fn(); + q.add(fn); + q.add(fn); + q.run(); + expect(fn).toHaveBeenCalledTimes(2); + }); + + test('Does not call a callback that was cancelled before the queue was run', () => { + const q = new TaskQueue(); + const yes = jest.fn(); + const no = jest.fn(); + q.add(yes); + const id = q.add(no); + q.remove(id); + q.run(); + expect(yes).toHaveBeenCalledTimes(1); + expect(no).not.toHaveBeenCalled(); + }); + + test('Does not call a callback that was cancelled while the queue was running', () => { + const q = new TaskQueue(); + const yes = jest.fn(); + const no = jest.fn(); + q.add(yes); + let id; // eslint-disable-line prefer-const + q.add(() => q.remove(id)); + id = q.add(no); + q.run(); + expect(yes).toHaveBeenCalledTimes(1); + expect(no).not.toHaveBeenCalled(); + }); + + test('Allows each instance of a multiply-queued callback to be cancelled independently', () => { + const q = new TaskQueue(); + const cb = jest.fn(); + q.add(cb); + const id = q.add(cb); + q.remove(id); + q.run(); + expect(cb).toHaveBeenCalledTimes(1); + }); + + test('Does not throw if a remove() is called after running the queue', () => { + const q = new TaskQueue(); + const cb = jest.fn(); + const id = q.add(cb); + q.run(); + q.remove(id); + expect(cb).toHaveBeenCalledTimes(1); + }); + + test('Does not add tasks to the currently-running queue', () => { + const q = new TaskQueue(); + const cb = jest.fn(); + q.add(() => q.add(cb)); + q.run(); + expect(cb).not.toHaveBeenCalled(); + q.run(); + expect(cb).toHaveBeenCalledTimes(1); + }); + + test('TaskQueue#run() throws on attempted re-entrance', () => { + const q = new TaskQueue(); + q.add(() => q.run()); + expect(() => q.run()).toThrow(); + }); + + test('TaskQueue#clear() prevents queued task from being executed', () => { + const q = new TaskQueue(); + const before = jest.fn(); + const after = jest.fn(); + q.add(before); + q.clear(); + q.add(after); + q.run(); + expect(before).not.toHaveBeenCalled(); + expect(after).toHaveBeenCalledTimes(1); + }); + + test('TaskQueue#clear() interrupts currently-running queue', () => { + const q = new TaskQueue(); + const before = jest.fn(); + const after = jest.fn(); + q.add(() => q.add(after)); + q.add(() => q.clear()); + q.add(before); + q.run(); + expect(before).not.toHaveBeenCalled(); + q.run(); + expect(after).not.toHaveBeenCalled(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/task_queue.ts b/web/libraries/maplibre-gl/src/util/task_queue.ts new file mode 100644 index 00000000..b7389e9e --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/task_queue.ts @@ -0,0 +1,64 @@ +export type TaskID = number; // can't mark opaque due to https://github.com/flowtype/flow-remove-types/pull/61 + +type Task = { + callback: (timeStamp: number) => void; + id: TaskID; + cancelled: boolean; +}; + +export class TaskQueue { + _queue: Array; + _id: TaskID; + _cleared: boolean; + _currentlyRunning: Array | false; + + constructor() { + this._queue = []; + this._id = 0; + this._cleared = false; + this._currentlyRunning = false; + } + + add(callback: (timeStamp: number) => void): TaskID { + const id = ++this._id; + const queue = this._queue; + queue.push({callback, id, cancelled: false}); + return id; + } + + remove(id: TaskID) { + const running = this._currentlyRunning; + const queue = running ? this._queue.concat(running) : this._queue; + for (const task of queue) { + if (task.id === id) { + task.cancelled = true; + return; + } + } + } + + run(timeStamp: number = 0) { + if (this._currentlyRunning) throw new Error('Attempting to run(), but is already running.'); + const queue = this._currentlyRunning = this._queue; + + // Tasks queued by callbacks in the current queue should be executed + // on the next run, not the current run. + this._queue = []; + + for (const task of queue) { + if (task.cancelled) continue; + task.callback(timeStamp); + if (this._cleared) break; + } + + this._cleared = false; + this._currentlyRunning = false; + } + + clear() { + if (this._currentlyRunning) { + this._cleared = true; + } + this._queue = []; + } +} diff --git a/web/libraries/maplibre-gl/src/util/test/mock_fetch.ts b/web/libraries/maplibre-gl/src/util/test/mock_fetch.ts new file mode 100644 index 00000000..db6e7b13 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/test/mock_fetch.ts @@ -0,0 +1,51 @@ +export class RequestMock implements Partial { + public readonly cache: RequestCache; + public readonly headers: Headers = new Headers(); + public readonly method?: string; + public readonly url?: string; + + public get signal(): AbortSignal { + return null; + } + + constructor(input: RequestInfo | URL, init?: RequestInit) { + this.cache = typeof input === 'object' && 'cache' in input ? input.cache : init.cache; + this.method = typeof input === 'object' && 'method' in input ? input.method : init.method; + this.url = typeof input === 'object' && 'url' in input ? input.url : input.toString(); + this.headers = typeof input === 'object' && 'headers' in input ? new Headers(input.headers) : new Headers(init.headers || {}); + } +} + +class AbortControllerMock { + public signal: AbortSignal; + + public abort(): void {} +} + +export type FetchMock = jest.Mock, [input: RequestInfo | URL, init?: RequestInit], any>; + +let _AbortController: typeof AbortController; +let _Request: typeof Request; +let _fetch: typeof fetch; + +export function destroyFetchMock(): void { + global.AbortController = _AbortController ?? global.AbortController; + global.Request = _Request ?? global.Request; + global.fetch = _fetch ?? global.fetch; +} + +export function setupFetchMock(): FetchMock { + _AbortController = _AbortController ?? global.AbortController; + _Request = _Request ?? global.Request; + _fetch = _fetch ?? global.fetch; + + const fetchMock = jest.fn(async (_input: RequestInfo | URL, _init?: RequestInit): Promise => { + return {}; + }); + + global.AbortController = AbortControllerMock; + global.Request = RequestMock as unknown as typeof Request; + global.fetch = fetchMock as any; + + return fetchMock; +} diff --git a/web/libraries/maplibre-gl/src/util/test/mock_webgl.ts b/web/libraries/maplibre-gl/src/util/test/mock_webgl.ts new file mode 100644 index 00000000..3559ca26 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/test/mock_webgl.ts @@ -0,0 +1,101 @@ +import gl from 'gl'; + +export function setupMockWebGLContext(webglContext: any) { + + const mockVaoExtension = { + bindVertexArrayOES: jest.fn(), + deleteVertexArrayOES: jest.fn(), + createVertexArrayOES: jest.fn(), + }; + + const mockColorBufferExtension = { + RGB16F_EXT: jest.fn(), + }; + + const mockTextureHalfFloatExtension = { + HALF_FLOAT_OES: jest.fn(), + }; + + // Setup getExtension to return the correct mock extension + webglContext.getExtension = jest.fn((extensionName) => { + switch (extensionName) { + case 'OES_vertex_array_object': + return mockVaoExtension; + case 'EXT_color_buffer_half_float': + return mockColorBufferExtension; + case 'OES_texture_half_float': + return mockTextureHalfFloatExtension; + default: + return null; + } + }); + + // Update drawingBufferWidth and drawingBufferHeigth when viewport changes + webglContext.viewport = jest.fn((x, y, width, height) => { + webglContext.drawingBufferWidth = width; + webglContext.drawingBufferHeight = height; + }); + + // Define the properties on the WebGL context + Object.defineProperty(webglContext, 'bindVertexArray', { + get() { + const extension = this.getExtension('OES_vertex_array_object'); + return extension ? extension.bindVertexArrayOES : undefined; + }, + }); + + Object.defineProperty(webglContext, 'RGB16F', { + get() { + const extension = this.getExtension('EXT_color_buffer_half_float'); + return extension ? extension.RGB16F_EXT : undefined; + }, + }); + + Object.defineProperty(webglContext, 'HALF_FLOAT', { + get() { + const extension = this.getExtension('OES_texture_half_float'); + return extension ? extension.HALF_FLOAT_OES : undefined; + }, + }); + + Object.defineProperty(webglContext, 'deleteVertexArray', { + get() { + const extension = this.getExtension('OES_vertex_array_object'); + return extension ? extension.deleteVertexArrayOES : undefined; + }, + }); + + Object.defineProperty(webglContext, 'createVertexArray', { + get() { + const extension = this.getExtension('OES_vertex_array_object'); + return extension ? extension.createVertexArrayOES : undefined; + }, + }); + +} + +// Add webgl context with the supplied GL +export function setWebGlContext() { + const originalGetContext = global.HTMLCanvasElement.prototype.getContext; + + function imitateWebGlGetContext(type, attributes) { + if (type === 'webgl2') { + return null; + } + if (type === 'webgl') { + if (!this._webGLContext) { + this._webGLContext = gl(this.width, this.height, attributes); + if (!this._webGLContext) { + throw new Error('Failed to create a WebGL context'); + } + } + + setupMockWebGLContext(this._webGLContext); + + return this._webGLContext; + } + // Fallback to existing HTMLCanvasElement getContext behaviour + return originalGetContext.call(this, type, attributes); + } + global.HTMLCanvasElement.prototype.getContext = imitateWebGlGetContext; +} diff --git a/web/libraries/maplibre-gl/src/util/test/util.ts b/web/libraries/maplibre-gl/src/util/test/util.ts new file mode 100644 index 00000000..39e63097 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/test/util.ts @@ -0,0 +1,146 @@ +import {Map} from '../../ui/map'; +import {extend} from '../../util/util'; +import {Dispatcher} from '../../util/dispatcher'; +import {setWebGlContext} from './mock_webgl'; + +export function createMap(options?, callback?) { + const container = window.document.createElement('div'); + const defaultOptions = { + container, + interactive: false, + attributionControl: false, + maplibreLogo: false, + trackResize: true, + style: { + 'version': 8, + 'sources': {}, + 'layers': [] + } + }; + + Object.defineProperty(container, 'clientWidth', {value: 200, configurable: true}); + Object.defineProperty(container, 'clientHeight', {value: 200, configurable: true}); + + if (options && options.deleteStyle) delete defaultOptions.style; + + const map = new Map(extend(defaultOptions, options)); + if (callback) map.on('load', () => { + callback(null, map); + }); + + return map; +} + +export function equalWithPrecision(test, expected, actual, multiplier, message, extra) { + message = message || `should be equal to within ${multiplier}`; + const expectedRounded = Math.round(expected / multiplier) * multiplier; + const actualRounded = Math.round(actual / multiplier) * multiplier; + + return test.equal(expectedRounded, actualRounded, message, extra); +} + +// mock failed webgl context by dispatching "webglcontextcreationerror" event +// and returning null +export function setErrorWebGlContext() { + const originalGetContext = global.HTMLCanvasElement.prototype.getContext; + + function imitateErrorWebGlGetContext(type, attributes) { + if (type === 'webgl2' || type === 'webgl') { + const errorEvent = new Event('webglcontextcreationerror'); + (errorEvent as any).statusMessage = 'mocked webglcontextcreationerror message'; + this.dispatchEvent(errorEvent); + return null; + } + // Fallback to existing HTMLCanvasElement getContext behaviour + return originalGetContext.call(this, type, attributes); + } + global.HTMLCanvasElement.prototype.getContext = imitateErrorWebGlGetContext; +} + +export function setPerformance() { + window.performance.mark = jest.fn(); + window.performance.clearMeasures = jest.fn(); + window.performance.clearMarks = jest.fn(); +} + +export function setMatchMedia() { + // https://jestjs.io/docs/manual-mocks#mocking-methods-which-are-not-implemented-in-jsdom + Object.defineProperty(window, 'matchMedia', { + writable: true, + value: jest.fn().mockImplementation(query => ({ + matches: false, + media: query, + onchange: null, + addListener: jest.fn(), // deprecated + removeListener: jest.fn(), // deprecated + addEventListener: jest.fn(), + removeEventListener: jest.fn(), + dispatchEvent: jest.fn(), + })), + }); +} + +function setResizeObserver() { + global.ResizeObserver = jest.fn().mockImplementation(() => ({ + observe: jest.fn(), + unobserve: jest.fn(), + disconnect: jest.fn(), + })); +} + +export function beforeMapTest() { + setPerformance(); + setWebGlContext(); + setMatchMedia(); + setResizeObserver(); +} + +export function getWrapDispatcher() { + const wrapDispatcher = (dispatcher) => { + return { + getActor() { + return dispatcher; + } + } as any as Dispatcher; + }; + + return wrapDispatcher; +} + +export function getMockDispatcher() { + const wrapDispatcher = getWrapDispatcher(); + + const mockDispatcher = wrapDispatcher({ + send() {} + }); + + return mockDispatcher; +} + +export function stubAjaxGetImage(createImageBitmap) { + global.createImageBitmap = createImageBitmap; + + global.URL.revokeObjectURL = () => {}; + global.URL.createObjectURL = (_) => { return null; }; + + // eslint-disable-next-line accessor-pairs + Object.defineProperty(global.Image.prototype, 'src', { + set(url: string) { + if (url === 'error') { + this.onerror(); + } else this.onload(); + } + }); +} + +/** + * This should be used in test that use nise since the internal buffer returned from a file is not an instance of ArrayBuffer for some reason. + * @param data - the data read from a file, for example by `fs.readFileSync(...)` + * @returns a copy of the data in the file in `ArrayBuffer` format + */ +export function bufferToArrayBuffer(data: Buffer): ArrayBuffer { + const newBuffer = new ArrayBuffer(data.buffer.byteLength); + const view = new Uint8Array(newBuffer); + data.copy(view); + return view.buffer; +} diff --git a/web/libraries/maplibre-gl/src/util/throttle.test.ts b/web/libraries/maplibre-gl/src/util/throttle.test.ts new file mode 100644 index 00000000..169d335f --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/throttle.test.ts @@ -0,0 +1,45 @@ +import {throttle} from './throttle'; + +describe('throttle', () => { + + test('does not execute unthrottled function unless throttled function is invoked', () => { + let executionCount = 0; + throttle(() => { executionCount++; }, 0); + expect(executionCount).toBe(0); + }); + + test('executes unthrottled function once per tick when period is 0', done => { + let executionCount = 0; + const throttledFunction = throttle(() => { executionCount++; }, 0); + throttledFunction(); + throttledFunction(); + expect(executionCount).toBe(1); + setTimeout(() => { + throttledFunction(); + throttledFunction(); + expect(executionCount).toBe(2); + done(); + }, 0); + }); + + test('executes unthrottled function immediately once when period is > 0', () => { + let executionCount = 0; + const throttledFunction = throttle(() => { executionCount++; }, 5); + throttledFunction(); + throttledFunction(); + throttledFunction(); + expect(executionCount).toBe(1); + }); + + test('queues exactly one execution of unthrottled function when period is > 0', done => { + let executionCount = 0; + const throttledFunction = throttle(() => { executionCount++; }, 5); + throttledFunction(); + throttledFunction(); + throttledFunction(); + setTimeout(() => { + expect(executionCount).toBe(2); + done(); + }, 10); + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/throttle.ts b/web/libraries/maplibre-gl/src/util/throttle.ts new file mode 100644 index 00000000..17c90c8d --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/throttle.ts @@ -0,0 +1,28 @@ +/** + * Throttle the given function to run at most every `period` milliseconds. + */ +export function throttle void>(fn: T, time: number): (...args: Parameters) => ReturnType { + let pending = false; + let timerId: ReturnType = null; + let lastCallContext = null; + let lastCallArgs: Parameters; + + const later = () => { + timerId = null; + if (pending) { + fn.apply(lastCallContext, lastCallArgs); + timerId = setTimeout(later, time); + pending = false; + } + }; + + return (...args: Parameters) => { + pending = true; + lastCallContext = this; + lastCallArgs = args; + if (!timerId) { + later(); + } + return timerId; + }; +} diff --git a/web/libraries/maplibre-gl/src/util/throttled_invoker.ts b/web/libraries/maplibre-gl/src/util/throttled_invoker.ts new file mode 100644 index 00000000..f697a236 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/throttled_invoker.ts @@ -0,0 +1,40 @@ +/** + * Invokes the wrapped function in a non-blocking way when trigger() is called. Invocation requests + * are ignored until the function was actually invoked. + */ +export class ThrottledInvoker { + _channel: MessageChannel; + _triggered: boolean; + _callback: Function; + + constructor(callback: Function) { + this._callback = callback; + this._triggered = false; + if (typeof MessageChannel !== 'undefined') { + this._channel = new MessageChannel(); + this._channel.port2.onmessage = () => { + this._triggered = false; + this._callback(); + }; + } + } + + trigger() { + if (!this._triggered) { + this._triggered = true; + if (this._channel) { + this._channel.port1.postMessage(true); + } else { + setTimeout(() => { + this._triggered = false; + this._callback(); + }, 0); + } + } + } + + remove() { + delete this._channel; + this._callback = () => {}; + } +} diff --git a/web/libraries/maplibre-gl/src/util/transferable_grid_index.test.ts b/web/libraries/maplibre-gl/src/util/transferable_grid_index.test.ts new file mode 100644 index 00000000..b2f71bf2 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/transferable_grid_index.test.ts @@ -0,0 +1,56 @@ +import {TransferableGridIndex} from './transferable_grid_index'; + +describe('TransferableGridIndex', () => { + + test('indexes features', () => { + const grid = new TransferableGridIndex(100, 4, 1); + grid.insert(0, 4, 10, 6, 30); + grid.insert(1, 4, 10, 30, 12); + grid.insert(2, -10, 30, -5, 35); + + expect(grid.query(4, 10, 5, 11).sort()).toEqual([0, 1]); + expect(grid.query(24, 10, 25, 11).sort()).toEqual([1]); + expect(grid.query(40, 40, 100, 100)).toEqual([]); + expect(grid.query(-6, 0, -7, 100)).toEqual([2]); + expect(grid.query(-Infinity, -Infinity, Infinity, Infinity).sort()).toEqual([0, 1, 2]); + }); + + test('returns multiple copies of a key if multiple boxes were inserted with the same key', () => { + const grid = new TransferableGridIndex(100, 4, 0); + const key = 123; + grid.insert(key, 3, 3, 4, 4); + grid.insert(key, 13, 13, 14, 14); + grid.insert(key, 23, 23, 24, 24); + expect(grid.query(0, 0, 30, 30)).toEqual([key, key, key]); + }); + + test('serializing to an arraybuffer', () => { + const originalGrid = new TransferableGridIndex(100, 4, 1); + originalGrid.insert(0, 4, 10, 6, 30); + originalGrid.insert(1, 4, 10, 30, 12); + originalGrid.insert(2, -10, 30, -5, 35); + + const arrayBuffer = originalGrid.toArrayBuffer(); + const grid = new TransferableGridIndex(arrayBuffer); + + expect(grid.query(4, 10, 5, 11).sort()).toEqual([0, 1]); + expect(grid.query(24, 10, 25, 11).sort()).toEqual([1]); + expect(grid.query(40, 40, 100, 100)).toEqual([]); + expect(grid.query(-6, 0, -7, 100)).toEqual([2]); + expect(grid.query(-Infinity, -Infinity, Infinity, Infinity).sort()).toEqual([0, 1, 2]); + + expect(() => grid.insert(3, 0, 0, 0, 0)).toThrow(); + }); + + test('serialize round trip', () => { + const grid = new TransferableGridIndex(100, 4, 0); + const key = 123; + grid.insert(key, 3, 3, 4, 4); + grid.insert(key, 13, 13, 14, 14); + grid.insert(key, 23, 23, 24, 24); + const serializedGrid = TransferableGridIndex.serialize(grid); + const deserializedGrid = TransferableGridIndex.deserialize(serializedGrid); + expect(deserializedGrid.query(0, 0, 30, 30)).toEqual([key, key, key]); + }); + +}); diff --git a/web/libraries/maplibre-gl/src/util/transferable_grid_index.ts b/web/libraries/maplibre-gl/src/util/transferable_grid_index.ts new file mode 100644 index 00000000..3e0d6cb7 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/transferable_grid_index.ts @@ -0,0 +1,214 @@ +/* +This file was copied from https://github.com/mapbox/grid-index and was +migrated from JavaScript to TypeScript. + +Copyright (c) 2016, Mapbox + +Permission to use, copy, modify, and/or distribute this software for any purpose +with or without fee is hereby granted, provided that the above copyright notice +and this permission notice appear in all copies. + +THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH +REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND +FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, +INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS +OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER +TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF +THIS SOFTWARE. +*/ + +const NUM_PARAMS = 3; + +export type SerializedGrid = { + buffer: ArrayBuffer; +}; + +export class TransferableGridIndex { + cells: number[][]; + arrayBuffer: ArrayBuffer; + d: number; + keys: number[]; + bboxes: number[]; + n: number; + extent: number; + padding: number; + scale: any; + uid: number; + min: number; + max: number; + + constructor(extent: number | ArrayBuffer, n?: number, padding?: number) { + const cells = this.cells = []; + + if (extent instanceof ArrayBuffer) { + this.arrayBuffer = extent; + const array = new Int32Array(this.arrayBuffer); + extent = array[0]; + n = array[1]; + padding = array[2]; + + this.d = n + 2 * padding; + for (let k = 0; k < this.d * this.d; k++) { + const start = array[NUM_PARAMS + k]; + const end = array[NUM_PARAMS + k + 1]; + cells.push(start === end ? null : array.subarray(start, end)); + } + const keysOffset = array[NUM_PARAMS + cells.length]; + const bboxesOffset = array[NUM_PARAMS + cells.length + 1]; + this.keys = array.subarray(keysOffset, bboxesOffset) as any as number[]; + this.bboxes = array.subarray(bboxesOffset) as any as number[]; + + this.insert = this._insertReadonly; + + } else { + this.d = n + 2 * padding; + for (let i = 0; i < this.d * this.d; i++) { + cells.push([]); + } + this.keys = []; + this.bboxes = []; + } + + this.n = n; + this.extent = extent; + this.padding = padding; + this.scale = n / extent; + this.uid = 0; + + const p = (padding / n) * extent; + this.min = -p; + this.max = extent + p; + } + + insert(key: number, x1: number, y1: number, x2: number, y2: number) { + this._forEachCell(x1, y1, x2, y2, this._insertCell, this.uid++, undefined, undefined); + this.keys.push(key); + this.bboxes.push(x1); + this.bboxes.push(y1); + this.bboxes.push(x2); + this.bboxes.push(y2); + } + + _insertReadonly() { + throw new Error('Cannot insert into a GridIndex created from an ArrayBuffer.'); + } + + _insertCell(x1: number, y1: number, x2: number, y2: number, cellIndex: number, uid: number) { + this.cells[cellIndex].push(uid); + } + + query(x1: number, y1: number, x2: number, y2: number, intersectionTest?: Function): number[] { + const min = this.min; + const max = this.max; + if (x1 <= min && y1 <= min && max <= x2 && max <= y2 && !intersectionTest) { + // We use `Array#slice` because `this.keys` may be a `Int32Array` and + // some browsers (Safari and IE) do not support `TypedArray#slice` + // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/TypedArray/slice#Browser_compatibility + return Array.prototype.slice.call(this.keys); + + } else { + const result = []; + const seenUids = {}; + this._forEachCell(x1, y1, x2, y2, this._queryCell, result, seenUids, intersectionTest); + return result; + } + } + + _queryCell(x1: number, y1: number, x2: number, y2:number, cellIndex:number, result, seenUids, intersectionTest: Function) { + const cell = this.cells[cellIndex]; + if (cell !== null) { + const keys = this.keys; + const bboxes = this.bboxes; + for (let u = 0; u < cell.length; u++) { + const uid = cell[u]; + if (seenUids[uid] === undefined) { + const offset = uid * 4; + if (intersectionTest ? + intersectionTest(bboxes[offset + 0], bboxes[offset + 1], bboxes[offset + 2], bboxes[offset + 3]) : + ((x1 <= bboxes[offset + 2]) && + (y1 <= bboxes[offset + 3]) && + (x2 >= bboxes[offset + 0]) && + (y2 >= bboxes[offset + 1]))) { + seenUids[uid] = true; + result.push(keys[uid]); + } else { + seenUids[uid] = false; + } + } + } + } + } + + _forEachCell(x1: number, y1: number, x2:number, y2:number, fn: Function, arg1, arg2, intersectionTest) { + const cx1 = this._convertToCellCoord(x1); + const cy1 = this._convertToCellCoord(y1); + const cx2 = this._convertToCellCoord(x2); + const cy2 = this._convertToCellCoord(y2); + for (let x = cx1; x <= cx2; x++) { + for (let y = cy1; y <= cy2; y++) { + const cellIndex = this.d * y + x; + if (intersectionTest && !intersectionTest( + this._convertFromCellCoord(x), + this._convertFromCellCoord(y), + this._convertFromCellCoord(x + 1), + this._convertFromCellCoord(y + 1))) continue; + if (fn.call(this, x1, y1, x2, y2, cellIndex, arg1, arg2, intersectionTest)) return; + } + } + } + + _convertFromCellCoord (x) { + return (x - this.padding) / this.scale; + } + + _convertToCellCoord(x) { + return Math.max(0, Math.min(this.d - 1, Math.floor(x * this.scale) + this.padding)); + } + + toArrayBuffer(): ArrayBuffer { + if (this.arrayBuffer) return this.arrayBuffer; + + const cells = this.cells; + + const metadataLength = NUM_PARAMS + this.cells.length + 1 + 1; + let totalCellLength = 0; + for (let i = 0; i < this.cells.length; i++) { + totalCellLength += this.cells[i].length; + } + + const array = new Int32Array(metadataLength + totalCellLength + this.keys.length + this.bboxes.length); + array[0] = this.extent; + array[1] = this.n; + array[2] = this.padding; + + let offset = metadataLength; + for (let k = 0; k < cells.length; k++) { + const cell = cells[k]; + array[NUM_PARAMS + k] = offset; + array.set(cell, offset); + offset += cell.length; + } + + array[NUM_PARAMS + cells.length] = offset; + array.set(this.keys, offset); + offset += this.keys.length; + + array[NUM_PARAMS + cells.length + 1] = offset; + array.set(this.bboxes, offset); + offset += this.bboxes.length; + + return array.buffer; + } + + public static serialize(grid: TransferableGridIndex, transferables?: Array): SerializedGrid { + const buffer = grid.toArrayBuffer(); + if (transferables) { + transferables.push(buffer); + } + return {buffer}; + } + + public static deserialize(serialized: SerializedGrid): TransferableGridIndex { + return new TransferableGridIndex(serialized.buffer); + } +} diff --git a/web/libraries/maplibre-gl/src/util/util.test.ts b/web/libraries/maplibre-gl/src/util/util.test.ts new file mode 100644 index 00000000..47ede250 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/util.test.ts @@ -0,0 +1,518 @@ +import Point from '@mapbox/point-geometry'; +import {arraysIntersect, asyncAll, bezier, clamp, clone, deepEqual, easeCubicInOut, extend, filterObject, findLineIntersection, isClosedPolygon, isCounterClockwise, isPowerOfTwo, keysDifference, mapObject, nextPowerOfTwo, parseCacheControl, pick, readImageDataUsingOffscreenCanvas, readImageUsingVideoFrame, uniqueId, wrap} from './util'; +import {Canvas} from 'canvas'; + +describe('util', () => { + expect(easeCubicInOut(0)).toBe(0); + expect(easeCubicInOut(0.2)).toBe(0.03200000000000001); + expect(easeCubicInOut(0.5)).toBe(0.5); + expect(easeCubicInOut(1)).toBe(1); + expect(keysDifference({a: 1}, {})).toEqual(['a']); + expect(keysDifference({a: 1}, {a: 1})).toEqual([]); + expect(extend({a: 1}, {b: 2})).toEqual({a: 1, b: 2}); + expect(pick({a: 1, b: 2, c: 3}, ['a', 'c'])).toEqual({a: 1, c: 3}); + expect(pick({a: 1, b: 2, c: 3}, ['a', 'c', 'd'])).toEqual({a: 1, c: 3}); + expect(typeof uniqueId() === 'number').toBeTruthy(); + + test('asyncAll - sync', done => { + expect(asyncAll([0, 1, 2], (data, callback) => { + callback(null, data); + }, (err, results) => { + expect(err).toBeFalsy(); + expect(results).toEqual([0, 1, 2]); + })).toBeUndefined(); + done(); + }); + + test('asyncAll - async', done => { + expect(asyncAll([4, 0, 1, 2], (data, callback) => { + setTimeout(() => { + callback(null, data); + }, data); + }, (err, results) => { + expect(err).toBeFalsy(); + expect(results).toEqual([4, 0, 1, 2]); + done(); + })).toBeUndefined(); + }); + + test('asyncAll - error', done => { + expect(asyncAll([4, 0, 1, 2], (data, callback) => { + setTimeout(() => { + callback(new Error('hi'), data); + }, data); + }, (err, results) => { + expect(err && err.message).toBe('hi'); + expect(results).toEqual([4, 0, 1, 2]); + done(); + })).toBeUndefined(); + }); + + test('asyncAll - empty', done => { + expect(asyncAll([], (data, callback) => { + callback(null, 'foo'); + }, (err, results) => { + expect(err).toBeFalsy(); + expect(results).toEqual([]); + })).toBeUndefined(); + done(); + }); + + test('isPowerOfTwo', done => { + expect(isPowerOfTwo(1)).toBe(true); + expect(isPowerOfTwo(2)).toBe(true); + expect(isPowerOfTwo(256)).toBe(true); + expect(isPowerOfTwo(-256)).toBe(false); + expect(isPowerOfTwo(0)).toBe(false); + expect(isPowerOfTwo(-42)).toBe(false); + expect(isPowerOfTwo(42)).toBe(false); + done(); + }); + + test('nextPowerOfTwo', done => { + expect(nextPowerOfTwo(1)).toBe(1); + expect(nextPowerOfTwo(2)).toBe(2); + expect(nextPowerOfTwo(256)).toBe(256); + expect(nextPowerOfTwo(-256)).toBe(1); + expect(nextPowerOfTwo(0)).toBe(1); + expect(nextPowerOfTwo(-42)).toBe(1); + expect(nextPowerOfTwo(42)).toBe(64); + done(); + }); + + test('nextPowerOfTwo', done => { + expect(isPowerOfTwo(nextPowerOfTwo(1))).toBe(true); + expect(isPowerOfTwo(nextPowerOfTwo(2))).toBe(true); + expect(isPowerOfTwo(nextPowerOfTwo(256))).toBe(true); + expect(isPowerOfTwo(nextPowerOfTwo(-256))).toBe(true); + expect(isPowerOfTwo(nextPowerOfTwo(0))).toBe(true); + expect(isPowerOfTwo(nextPowerOfTwo(-42))).toBe(true); + expect(isPowerOfTwo(nextPowerOfTwo(42))).toBe(true); + done(); + }); + + test('clamp', done => { + expect(clamp(0, 0, 1)).toBe(0); + expect(clamp(1, 0, 1)).toBe(1); + expect(clamp(200, 0, 180)).toBe(180); + expect(clamp(-200, 0, 180)).toBe(0); + done(); + }); + + test('wrap', done => { + expect(wrap(0, 0, 1)).toBe(1); + expect(wrap(1, 0, 1)).toBe(1); + expect(wrap(200, 0, 180)).toBe(20); + expect(wrap(-200, 0, 180)).toBe(160); + done(); + }); + + test('bezier', done => { + const curve = bezier(0, 0, 0.25, 1); + expect(curve instanceof Function).toBeTruthy(); + expect(curve(0)).toBe(0); + expect(curve(1)).toBe(1); + expect(curve(0.5)).toBe(0.8230854638965502); + done(); + }); + + test('asyncAll', done => { + let expectedValue = 1; + asyncAll([], (callback) => { callback(); }, () => { + expect('immediate callback').toBeTruthy(); + }); + asyncAll([1, 2, 3], (number, callback) => { + expect(number).toBe(expectedValue++); + expect(callback instanceof Function).toBeTruthy(); + callback(null, 0); + }, () => { + done(); + }); + }); + + test('mapObject', () => { + expect.assertions(6); + expect(mapObject({}, () => { expect(false).toBeTruthy(); })).toEqual({}); + const that = {}; + expect(mapObject({map: 'box'}, function(value, key, object) { + expect(value).toBe('box'); + expect(key).toBe('map'); + expect(object).toEqual({map: 'box'}); + expect(this).toBe(that); + return 'BOX'; + }, that)).toEqual({map: 'BOX'}); + }); + + test('filterObject', done => { + expect.assertions(6); + expect(filterObject({}, () => { expect(false).toBeTruthy(); })).toEqual({}); + const that = {}; + filterObject({map: 'box'}, function(value, key, object) { + expect(value).toBe('box'); + expect(key).toBe('map'); + expect(object).toEqual({map: 'box'}); + expect(this).toBe(that); + return true; + }, that); + expect(filterObject({map: 'box', box: 'map'}, (value) => { + return value === 'box'; + })).toEqual({map: 'box'}); + done(); + }); + + test('deepEqual', done => { + const a = { + foo: 'bar', + bar: { + baz: 5, + lol: ['cat', 2] + } + }; + const b = JSON.parse(JSON.stringify(a)); + const c = JSON.parse(JSON.stringify(a)); + c.bar.lol[0] = 'z'; + + expect(deepEqual(a, b)).toBeTruthy(); + expect(deepEqual(a, c)).toBeFalsy(); + expect(deepEqual(a, null)).toBeFalsy(); + expect(deepEqual(null, c)).toBeFalsy(); + expect(deepEqual(null, null)).toBeTruthy(); + + done(); + }); +}); + +describe('util clone', () => { + test('array', done => { + const input = [false, 1, 'two']; + const output = clone(input); + expect(input).not.toBe(output); + expect(input).toEqual(output); + done(); + }); + + test('object', done => { + const input = {a: false, b: 1, c: 'two'}; + const output = clone(input); + expect(input).not.toBe(output); + expect(input).toEqual(output); + done(); + }); + + test('deep object', done => { + const input = {object: {a: false, b: 1, c: 'two'}}; + const output = clone(input); + expect(input.object).not.toBe(output.object); + expect(input.object).toEqual(output.object); + done(); + }); + + test('deep array', done => { + const input = {array: [false, 1, 'two']}; + const output = clone(input); + expect(input.array).not.toBe(output.array); + expect(input.array).toEqual(output.array); + done(); + }); +}); + +describe('util arraysIntersect', () => { + test('intersection', done => { + const a = ['1', '2', '3']; + const b = ['5', '4', '3']; + + expect(arraysIntersect(a, b)).toBe(true); + done(); + }); + + test('no intersection', done => { + const a = ['1', '2', '3']; + const b = ['4', '5', '6']; + + expect(arraysIntersect(a, b)).toBe(false); + done(); + }); + +}); + +describe('util isCounterClockwise', () => { + test('counter clockwise', done => { + const a = new Point(0, 0); + const b = new Point(1, 0); + const c = new Point(1, 1); + + expect(isCounterClockwise(a, b, c)).toBe(true); + done(); + }); + + test('clockwise', done => { + const a = new Point(0, 0); + const b = new Point(1, 0); + const c = new Point(1, 1); + + expect(isCounterClockwise(c, b, a)).toBe(false); + done(); + }); +}); + +describe('util isClosedPolygon', () => { + test('not enough points', done => { + const polygon = [new Point(0, 0), new Point(1, 0), new Point(0, 1)]; + + expect(isClosedPolygon(polygon)).toBe(false); + done(); + }); + + test('not equal first + last point', done => { + const polygon = [new Point(0, 0), new Point(1, 0), new Point(0, 1), new Point(1, 1)]; + + expect(isClosedPolygon(polygon)).toBe(false); + done(); + }); + + test('closed polygon', done => { + const polygon = [new Point(0, 0), new Point(1, 0), new Point(1, 1), new Point(0, 1), new Point(0, 0)]; + + expect(isClosedPolygon(polygon)).toBe(true); + done(); + }); + +}); + +describe('util parseCacheControl', () => { + test('max-age', done => { + expect(parseCacheControl('max-age=123456789')).toEqual({ + 'max-age': 123456789 + }); + + expect(parseCacheControl('max-age=1000')).toEqual({ + 'max-age': 1000 + }); + + expect(parseCacheControl('max-age=null')).toEqual({}); + + done(); + }); + +}); + +describe('util findLineIntersection', () => { + test('line intersection', () => { + const horizontal = [ + new Point(0, 0), + new Point(10, 0)]; + const vertical = [ + new Point(30, -20), + new Point(30, -10) + ]; + const intersection = findLineIntersection(horizontal[0], horizontal[1], vertical[0], vertical[1]); + expect(intersection).toEqual(new Point(30, 0)); + }); + + test('line intersection backwards', () => { + // Direction of line segments should be irrelevant + const horizontal = [ + new Point(10, 0), + new Point(0, 0)]; + const vertical = [ + new Point(30, -10), + new Point(30, -20) + ]; + const intersection = findLineIntersection(horizontal[0], horizontal[1], vertical[0], vertical[1]); + expect(intersection).toEqual(new Point(30, 0)); + }); + + test('crossing line intersection', () => { + // This should not be possible for two adjacent segments of a line string + const horizontal = [ + new Point(-10, 0), + new Point(10, 0)]; + const vertical = [ + new Point(0, -10), + new Point(0, 10) + ]; + const intersection = findLineIntersection(horizontal[0], horizontal[1], vertical[0], vertical[1]); + expect(intersection).toEqual(new Point(0, 0)); + }); + + test('parallel lines do not intersect', () => { + const first = [ + new Point(0, 0), + new Point(10, 0)]; + const second = [ + new Point(10, 0), + new Point(30, 0) + ]; + const intersection = findLineIntersection(first[0], first[1], second[0], second[1]); + expect(intersection).toBeNull(); + }); +}); + +describe('util readImageUsingVideoFrame', () => { + let format = 'RGBA'; + const frame = { + get format() { + return format; + }, + copyTo: jest.fn(buf => { + buf.set(new Uint8Array([1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16]).subarray(0, buf.length)); + return Promise.resolve(); + }), + close: jest.fn(), + }; + (window as any).VideoFrame = jest.fn(() => frame); + const canvas = document.createElement('canvas'); + canvas.width = canvas.height = 2; + + beforeEach(() => { + format = 'RGBA'; + frame.copyTo.mockClear(); + frame.close.mockReset(); + }); + + test('copy RGB', async () => { + format = 'RGBA'; + const result = await readImageUsingVideoFrame(canvas, 0, 0, 2, 2); + expect(result).toHaveLength(4 * 4); + expect(frame.copyTo).toHaveBeenCalledWith(expect.anything(), { + layout: [{offset: 0, stride: 8}], + rect: {x: 0, y: 0, width: 2, height: 2} + }); + expect(result).toEqual(new Uint8ClampedArray([ + 1, 2, 3, 4, 5, 6, 7, 8, + 9, 10, 11, 12, 13, 14, 15, 16 + ])); + expect(frame.close).toHaveBeenCalledTimes(1); + }); + + test('flip BRG', async () => { + format = 'BGRX'; + const result = await readImageUsingVideoFrame(canvas, 0, 0, 2, 2); + expect(result).toEqual(new Uint8ClampedArray([ + 3, 2, 1, 4, 7, 6, 5, 8, + 11, 10, 9, 12, 15, 14, 13, 16 + ])); + expect(frame.close).toHaveBeenCalledTimes(1); + }); + + test('ignore bad format', async () => { + format = 'OTHER'; + await expect(readImageUsingVideoFrame(canvas, 0, 0, 2, 2)).rejects.toThrow(); + expect(frame.close).toHaveBeenCalledTimes(1); + }); + + describe('layout/rect', () => { + beforeEach(() => { + (window as any).VideoFrame = jest.fn(() => frame); + canvas.width = canvas.height = 3; + }); + + test('full rectangle', async () => { + await readImageUsingVideoFrame(canvas, 0, 0, 3, 3); + expect(frame.copyTo).toHaveBeenCalledWith(expect.anything(), { + layout: [{offset: 0, stride: 12}], + rect: {x: 0, y: 0, width: 3, height: 3} + }); + }); + + test('top left', async () => { + await readImageUsingVideoFrame(canvas, 0, 0, 2, 2); + expect(frame.copyTo).toHaveBeenCalledWith(expect.anything(), { + layout: [{offset: 0, stride: 8}], + rect: {x: 0, y: 0, width: 2, height: 2} + }); + }); + + test('top right', async () => { + await readImageUsingVideoFrame(canvas, 1, 0, 2, 2); + expect(frame.copyTo).toHaveBeenCalledWith(expect.anything(), { + layout: [{offset: 0, stride: 8}], + rect: {x: 1, y: 0, width: 2, height: 2} + }); + }); + + test('bottom left', async () => { + await readImageUsingVideoFrame(canvas, 0, 1, 2, 2); + expect(frame.copyTo).toHaveBeenCalledWith(expect.anything(), { + layout: [{offset: 0, stride: 8}], + rect: {x: 0, y: 1, width: 2, height: 2} + }); + }); + + test('bottom right', async () => { + await readImageUsingVideoFrame(canvas, 1, 1, 2, 2); + expect(frame.copyTo).toHaveBeenCalledWith(expect.anything(), { + layout: [{offset: 0, stride: 8}], + rect: {x: 1, y: 1, width: 2, height: 2} + }); + }); + + test('middle', async () => { + await readImageUsingVideoFrame(canvas, 1, 1, 1, 1); + expect(frame.copyTo).toHaveBeenCalledWith(expect.anything(), { + layout: [{offset: 0, stride: 4}], + rect: {x: 1, y: 1, width: 1, height: 1} + }); + }); + + test('extend past on all sides', async () => { + await readImageUsingVideoFrame(canvas, -1, -1, 5, 5); + expect(frame.copyTo).toHaveBeenCalledWith(expect.anything(), { + layout: [{offset: 4 * 5 + 4, stride: 4 * 5}], + rect: {x: 0, y: 0, width: 3, height: 3} + }); + }); + + test('overhang top left', async () => { + await readImageUsingVideoFrame(canvas, -1, -1, 2, 2); + expect(frame.copyTo).toHaveBeenCalledWith(expect.anything(), { + layout: [{offset: 4 * 2 + 4, stride: 4 * 2}], + rect: {x: 0, y: 0, width: 1, height: 1} + }); + }); + + test('overhang top right', async () => { + await readImageUsingVideoFrame(canvas, 2, -1, 2, 2); + expect(frame.copyTo).toHaveBeenCalledWith(expect.anything(), { + layout: [{offset: 4 * 2, stride: 4 * 2}], + rect: {x: 2, y: 0, width: 1, height: 1} + }); + }); + + test('overhang bottom left', async () => { + await readImageUsingVideoFrame(canvas, -1, 2, 2, 2); + expect(frame.copyTo).toHaveBeenCalledWith(expect.anything(), { + layout: [{offset: 4, stride: 4 * 2}], + rect: {x: 0, y: 2, width: 1, height: 1} + }); + }); + + test('overhang bottom right', async () => { + await readImageUsingVideoFrame(canvas, 2, 2, 2, 2); + expect(frame.copyTo).toHaveBeenCalledWith(expect.anything(), { + layout: [{offset: 0, stride: 4 * 2}], + rect: {x: 2, y: 2, width: 1, height: 1} + }); + }); + }); +}); + +describe('util readImageDataUsingOffscreenCanvas', () => { + test('reads pixels from image', async () => { + (window as any).OffscreenCanvas = Canvas; + const image = new Canvas(2, 2); + const context = image.getContext('2d'); + context.fillStyle = 'rgb(10,0,0)'; + context.fillRect(0, 0, 1, 1); + context.fillStyle = 'rgb(0,20,0)'; + context.fillRect(1, 0, 1, 1); + context.fillStyle = 'rgb(0,0,30)'; + context.fillRect(0, 1, 1, 1); + context.fillStyle = 'rgb(40,40,40)'; + context.fillRect(1, 1, 1, 1); + expect([...await readImageDataUsingOffscreenCanvas(image as any, 0, 0, 2, 2)]).toEqual([ + 10, 0, 0, 255, 0, 20, 0, 255, + 0, 0, 30, 255, 40, 40, 40, 255, + ]); + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/util.ts b/web/libraries/maplibre-gl/src/util/util.ts new file mode 100644 index 00000000..13e0c7db --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/util.ts @@ -0,0 +1,669 @@ +import Point from '@mapbox/point-geometry'; +import UnitBezier from '@mapbox/unitbezier'; +import type {Callback} from '../types/callback'; +import {isOffscreenCanvasDistorted} from './offscreen_canvas_distorted'; +import type {Size} from './image'; + +/** + * Given a value `t` that varies between 0 and 1, return + * an interpolation function that eases between 0 and 1 in a pleasing + * cubic in-out fashion. + */ +export function easeCubicInOut(t: number): number { + if (t <= 0) return 0; + if (t >= 1) return 1; + const t2 = t * t, + t3 = t2 * t; + return 4 * (t < 0.5 ? t3 : 3 * (t - t2) + t3 - 0.75); +} + +/** + * Given given (x, y), (x1, y1) control points for a bezier curve, + * return a function that interpolates along that curve. + * + * @param p1x - control point 1 x coordinate + * @param p1y - control point 1 y coordinate + * @param p2x - control point 2 x coordinate + * @param p2y - control point 2 y coordinate + */ +export function bezier(p1x: number, p1y: number, p2x: number, p2y: number): (t: number) => number { + const bezier = new UnitBezier(p1x, p1y, p2x, p2y); + return function(t: number) { + return bezier.solve(t); + }; +} + +/** + * A default bezier-curve powered easing function with + * control points (0.25, 0.1) and (0.25, 1) + */ +export const defaultEasing = bezier(0.25, 0.1, 0.25, 1); + +/** + * constrain n to the given range via min + max + * + * @param n - value + * @param min - the minimum value to be returned + * @param max - the maximum value to be returned + * @returns the clamped value + */ +export function clamp(n: number, min: number, max: number): number { + return Math.min(max, Math.max(min, n)); +} + +/** + * constrain n to the given range, excluding the minimum, via modular arithmetic + * + * @param n - value + * @param min - the minimum value to be returned, exclusive + * @param max - the maximum value to be returned, inclusive + * @returns constrained number + */ +export function wrap(n: number, min: number, max: number): number { + const d = max - min; + const w = ((n - min) % d + d) % d + min; + return (w === min) ? max : w; +} + +/** + * Call an asynchronous function on an array of arguments, + * calling `callback` with the completed results of all calls. + * + * @param array - input to each call of the async function. + * @param fn - an async function with signature (data, callback) + * @param callback - a callback run after all async work is done. + * called with an array, containing the results of each async call. + */ +export function asyncAll( + array: Array, + fn: (item: Item, fnCallback: Callback) => void, + callback: Callback> +) { + if (!array.length) { return callback(null, []); } + let remaining = array.length; + const results = new Array(array.length); + let error = null; + array.forEach((item, i) => { + fn(item, (err, result) => { + if (err) error = err; + results[i] = (result as any as Result); // https://github.com/facebook/flow/issues/2123 + if (--remaining === 0) callback(error, results); + }); + }); +} + +/** + * Compute the difference between the keys in one object and the keys + * in another object. + * + * @returns keys difference + */ +export function keysDifference( + obj: {[key: string]: S}, + other: {[key: string]: T} +): Array { + const difference = []; + for (const i in obj) { + if (!(i in other)) { + difference.push(i); + } + } + return difference; +} + +/** + * Given a destination object and optionally many source objects, + * copy all properties from the source objects into the destination. + * The last source object given overrides properties from previous + * source objects. + * + * @param dest - destination object + * @param sources - sources from which properties are pulled + */ +export function extend(dest: any, ...sources: Array): any { + for (const src of sources) { + for (const k in src) { + dest[k] = src[k]; + } + } + return dest; +} + +/** + * Given an object and a number of properties as strings, return version + * of that object with only those properties. + * + * @param src - the object + * @param properties - an array of property names chosen + * to appear on the resulting object. + * @returns object with limited properties. + * @example + * ```ts + * let foo = { name: 'Charlie', age: 10 }; + * let justName = pick(foo, ['name']); // justName = { name: 'Charlie' } + * ``` + */ +export function pick(src: any, properties: Array): any { + const result = {}; + for (let i = 0; i < properties.length; i++) { + const k = properties[i]; + if (k in src) { + result[k] = src[k]; + } + } + return result; +} + +let id = 1; + +/** + * Return a unique numeric id, starting at 1 and incrementing with + * each call. + * + * @returns unique numeric id. + */ +export function uniqueId(): number { + return id++; +} + +/** + * Return whether a given value is a power of two + */ +export function isPowerOfTwo(value: number): boolean { + return (Math.log(value) / Math.LN2) % 1 === 0; +} + +/** + * Return the next power of two, or the input value if already a power of two + */ +export function nextPowerOfTwo(value: number): number { + if (value <= 1) return 1; + return Math.pow(2, Math.ceil(Math.log(value) / Math.LN2)); +} + +/** + * Create an object by mapping all the values of an existing object while + * preserving their keys. + */ +export function mapObject(input: any, iterator: Function, context?: any): any { + const output = {}; + for (const key in input) { + output[key] = iterator.call(context || this, input[key], key, input); + } + return output; +} + +/** + * Create an object by filtering out values of an existing object. + */ +export function filterObject(input: any, iterator: Function, context?: any): any { + const output = {}; + for (const key in input) { + if (iterator.call(context || this, input[key], key, input)) { + output[key] = input[key]; + } + } + return output; +} + +/** + * Deeply compares two object literals. + * @param a - first object literal to be compared + * @param b - second object literal to be compared + * @returns true if the two object literals are deeply equal, false otherwise + */ +export function deepEqual(a?: unknown | null, b?: unknown | null): boolean { + if (Array.isArray(a)) { + if (!Array.isArray(b) || a.length !== b.length) return false; + for (let i = 0; i < a.length; i++) { + if (!deepEqual(a[i], b[i])) return false; + } + return true; + } + if (typeof a === 'object' && a !== null && b !== null) { + if (!(typeof b === 'object')) return false; + const keys = Object.keys(a); + if (keys.length !== Object.keys(b).length) return false; + for (const key in a) { + if (!deepEqual(a[key], b[key])) return false; + } + return true; + } + return a === b; +} + +/** + * Deeply clones two objects. + */ +export function clone(input: T): T { + if (Array.isArray(input)) { + return input.map(clone) as any as T; + } else if (typeof input === 'object' && input) { + return mapObject(input, clone) as any as T; + } else { + return input; + } +} + +/** + * Check if two arrays have at least one common element. + */ +export function arraysIntersect(a: Array, b: Array): boolean { + for (let l = 0; l < a.length; l++) { + if (b.indexOf(a[l]) >= 0) return true; + } + return false; +} + +/** + * Print a warning message to the console and ensure duplicate warning messages + * are not printed. + */ +const warnOnceHistory: {[key: string]: boolean} = {}; + +export function warnOnce(message: string): void { + if (!warnOnceHistory[message]) { + // console isn't defined in some WebWorkers, see #2558 + if (typeof console !== 'undefined') console.warn(message); + warnOnceHistory[message] = true; + } +} + +/** + * Indicates if the provided Points are in a counter clockwise (true) or clockwise (false) order + * + * @returns true for a counter clockwise set of points + */ +// http://bryceboe.com/2006/10/23/line-segment-intersection-algorithm/ +export function isCounterClockwise(a: Point, b: Point, c: Point): boolean { + return (c.y - a.y) * (b.x - a.x) > (b.y - a.y) * (c.x - a.x); +} + +/** + * For two lines a and b in 2d space, defined by any two points along the lines, + * find the intersection point, or return null if the lines are parallel + * + * @param a1 - First point on line a + * @param a2 - Second point on line a + * @param b1 - First point on line b + * @param b2 - Second point on line b + * + * @returns the intersection point of the two lines or null if they are parallel + */ +export function findLineIntersection(a1: Point, a2: Point, b1: Point, b2: Point): Point | null { + const aDeltaY = a2.y - a1.y; + const aDeltaX = a2.x - a1.x; + const bDeltaY = b2.y - b1.y; + const bDeltaX = b2.x - b1.x; + + const denominator = (bDeltaY * aDeltaX) - (bDeltaX * aDeltaY); + + if (denominator === 0) { + // Lines are parallel + return null; + } + + const originDeltaY = a1.y - b1.y; + const originDeltaX = a1.x - b1.x; + const aInterpolation = (bDeltaX * originDeltaY - bDeltaY * originDeltaX) / denominator; + + // Find intersection by projecting out from origin of first segment + return new Point(a1.x + (aInterpolation * aDeltaX), a1.y + (aInterpolation * aDeltaY)); +} + +/** + * Returns the signed area for the polygon ring. Positive areas are exterior rings and + * have a clockwise winding. Negative areas are interior rings and have a counter clockwise + * ordering. + * + * @param ring - Exterior or interior ring + */ +export function calculateSignedArea(ring: Array): number { + let sum = 0; + for (let i = 0, len = ring.length, j = len - 1, p1, p2; i < len; j = i++) { + p1 = ring[i]; + p2 = ring[j]; + sum += (p2.x - p1.x) * (p1.y + p2.y); + } + return sum; +} + +/** + * Detects closed polygons, first + last point are equal + * + * @param points - array of points + * @returns `true` if the points are a closed polygon + */ +export function isClosedPolygon(points: Array): boolean { + // If it is 2 points that are the same then it is a point + // If it is 3 points with start and end the same then it is a line + if (points.length < 4) + return false; + + const p1 = points[0]; + const p2 = points[points.length - 1]; + + if (Math.abs(p1.x - p2.x) > 0 || + Math.abs(p1.y - p2.y) > 0) { + return false; + } + + // polygon simplification can produce polygons with zero area and more than 3 points + return Math.abs(calculateSignedArea(points)) > 0.01; +} + +/** + * Converts spherical coordinates to cartesian coordinates. + * + * @param spherical - Spherical coordinates, in [radial, azimuthal, polar] + * @returns cartesian coordinates in [x, y, z] + */ + +export function sphericalToCartesian([r, azimuthal, polar]: [number, number, number]): { + x: number; + y: number; + z: number; +} { + // We abstract "north"/"up" (compass-wise) to be 0° when really this is 90° (π/2): + // correct for that here + azimuthal += 90; + + // Convert azimuthal and polar angles to radians + azimuthal *= Math.PI / 180; + polar *= Math.PI / 180; + + return { + x: r * Math.cos(azimuthal) * Math.sin(polar), + y: r * Math.sin(azimuthal) * Math.sin(polar), + z: r * Math.cos(polar) + }; +} + +/** + * Returns true if the when run in the web-worker context. + * + * @returns `true` if the when run in the web-worker context. + */ +export function isWorker(): boolean { + // @ts-ignore + return typeof WorkerGlobalScope !== 'undefined' && typeof self !== 'undefined' && self instanceof WorkerGlobalScope; +} + +/** + * Parses data from 'Cache-Control' headers. + * + * @param cacheControl - Value of 'Cache-Control' header + * @returns object containing parsed header info. + */ + +export function parseCacheControl(cacheControl: string): any { + // Taken from [Wreck](https://github.com/hapijs/wreck) + const re = /(?:^|(?:\s*\,\s*))([^\x00-\x20\(\)<>@\,;\:\\"\/\[\]\?\=\{\}\x7F]+)(?:\=(?:([^\x00-\x20\(\)<>@\,;\:\\"\/\[\]\?\=\{\}\x7F]+)|(?:\"((?:[^"\\]|\\.)*)\")))?/g; + + const header = {}; + cacheControl.replace(re, ($0, $1, $2, $3) => { + const value = $2 || $3; + header[$1] = value ? value.toLowerCase() : true; + return ''; + }); + + if (header['max-age']) { + const maxAge = parseInt(header['max-age'], 10); + if (isNaN(maxAge)) delete header['max-age']; + else header['max-age'] = maxAge; + } + + return header; +} + +let _isSafari = null; + +/** + * Returns true when run in WebKit derived browsers. + * This is used as a workaround for a memory leak in Safari caused by using Transferable objects to + * transfer data between WebWorkers and the main thread. + * https://github.com/mapbox/mapbox-gl-js/issues/8771 + * + * This should be removed once the underlying Safari issue is fixed. + * + * @param scope - Since this function is used both on the main thread and WebWorker context, + * let the calling scope pass in the global scope object. + * @returns `true` when run in WebKit derived browsers. + */ +export function isSafari(scope: any): boolean { + if (_isSafari == null) { + const userAgent = scope.navigator ? scope.navigator.userAgent : null; + _isSafari = !!scope.safari || + !!(userAgent && (/\b(iPad|iPhone|iPod)\b/.test(userAgent) || (!!userAgent.match('Safari') && !userAgent.match('Chrome')))); + } + return _isSafari; +} + +export function storageAvailable(type: string): boolean { + try { + const storage = window[type]; + storage.setItem('_mapbox_test_', 1); + storage.removeItem('_mapbox_test_'); + return true; + } catch (e) { + return false; + } +} + +// The following methods are from https://developer.mozilla.org/en-US/docs/Web/API/WindowBase64/Base64_encoding_and_decoding#The_Unicode_Problem +//Unicode compliant base64 encoder for strings +export function b64EncodeUnicode(str: string) { + return btoa( + encodeURIComponent(str).replace(/%([0-9A-F]{2})/g, + (match, p1) => { + return String.fromCharCode(Number('0x' + p1)); //eslint-disable-line + } + ) + ); +} + +// Unicode compliant decoder for base64-encoded strings +export function b64DecodeUnicode(str: string) { + return decodeURIComponent(atob(str).split('').map((c) => { + return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2); //eslint-disable-line + }).join('')); +} + +export function isImageBitmap(image: any): image is ImageBitmap { + return typeof ImageBitmap !== 'undefined' && image instanceof ImageBitmap; +} + +/** + * Converts an ArrayBuffer to an ImageBitmap. + * + * Used mostly for testing purposes only, because mocking libs don't know how to work with ArrayBuffers, but work + * perfectly fine with ImageBitmaps. Might also be used for environments (other than testing) not supporting + * ArrayBuffers. + * + * @param data - Data to convert + * @param callback - A callback executed after the conversion is finished. Invoked with error (if any) as the first argument and resulting image bitmap (when no error) as the second + */ +export function arrayBufferToImageBitmap(data: ArrayBuffer, callback: (err?: Error | null, image?: ImageBitmap | null) => void) { + const blob: Blob = new Blob([new Uint8Array(data)], {type: 'image/png'}); + createImageBitmap(blob).then((imgBitmap) => { + callback(null, imgBitmap); + }).catch((e) => { + callback(new Error(`Could not load image because of ${e.message}. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.`)); + }); +} + +const transparentPngUrl = ''; + +/** + * Converts an ArrayBuffer to an HTMLImageElement. + * + * Used mostly for testing purposes only, because mocking libs don't know how to work with ArrayBuffers, but work + * perfectly fine with ImageBitmaps. Might also be used for environments (other than testing) not supporting + * ArrayBuffers. + * + * @param data - Data to convert + * @param callback - A callback executed after the conversion is finished. Invoked with error (if any) as the first argument and resulting image element (when no error) as the second + */ +export function arrayBufferToImage(data: ArrayBuffer, callback: (err?: Error | null, image?: HTMLImageElement | null) => void) { + const img: HTMLImageElement = new Image(); + img.onload = () => { + callback(null, img); + URL.revokeObjectURL(img.src); + // prevent image dataURI memory leak in Safari; + // but don't free the image immediately because it might be uploaded in the next frame + // https://github.com/mapbox/mapbox-gl-js/issues/10226 + img.onload = null; + window.requestAnimationFrame(() => { img.src = transparentPngUrl; }); + }; + img.onerror = () => callback(new Error('Could not load image. Please make sure to use a supported image type such as PNG or JPEG. Note that SVGs are not supported.')); + const blob: Blob = new Blob([new Uint8Array(data)], {type: 'image/png'}); + img.src = data.byteLength ? URL.createObjectURL(blob) : transparentPngUrl; +} + +/** + * Computes the webcodecs VideoFrame API options to select a rectangle out of + * an image and write it into the destination rectangle. + * + * Rect (x/y/width/height) select the overlapping rectangle from the source image + * and layout (offset/stride) write that overlapping rectangle to the correct place + * in the destination image. + * + * Offset is the byte offset in the dest image that the first pixel appears at + * and stride is the number of bytes to the start of the next row: + * ┌───────────┐ + * │ dest │ + * │ ┌───┼───────┐ + * │offset→│▓▓▓│ source│ + * │ │▓▓▓│ │ + * │ └───┼───────┘ + * │stride ⇠╌╌╌│ + * │╌╌╌╌╌╌→ │ + * └───────────┘ + * + * @param image - source image containing a width and height attribute + * @param x - top-left x coordinate to read from the image + * @param y - top-left y coordinate to read from the image + * @param width - width of the rectangle to read from the image + * @param height - height of the rectangle to read from the image + * @returns the layout and rect options to pass into VideoFrame API + */ +function computeVideoFrameParameters(image: Size, x: number, y: number, width: number, height: number): VideoFrameCopyToOptions { + const destRowOffset = Math.max(-x, 0) * 4; + const firstSourceRow = Math.max(0, y); + const firstDestRow = firstSourceRow - y; + const offset = firstDestRow * width * 4 + destRowOffset; + const stride = width * 4; + + const sourceLeft = Math.max(0, x); + const sourceTop = Math.max(0, y); + const sourceRight = Math.min(image.width, x + width); + const sourceBottom = Math.min(image.height, y + height); + return { + rect: { + x: sourceLeft, + y: sourceTop, + width: sourceRight - sourceLeft, + height: sourceBottom - sourceTop + }, + layout: [{offset, stride}] + }; +} + +/** + * Reads pixels from an ImageBitmap/Image/canvas using webcodec VideoFrame API. + * + * @param data - image, imagebitmap, or canvas to parse + * @param x - top-left x coordinate to read from the image + * @param y - top-left y coordinate to read from the image + * @param width - width of the rectangle to read from the image + * @param height - height of the rectangle to read from the image + * @returns a promise containing the parsed RGBA pixel values of the image, or the error if an error occurred + */ +export async function readImageUsingVideoFrame( + image: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas, + x: number, y: number, width: number, height: number +): Promise { + if (typeof VideoFrame === 'undefined') { + throw new Error('VideoFrame not supported'); + } + const frame = new VideoFrame(image, {timestamp: 0}); + try { + const format = frame?.format; + if (!format || !(format.startsWith('BGR') || format.startsWith('RGB'))) { + throw new Error(`Unrecognized format ${format}`); + } + const swapBR = format.startsWith('BGR'); + const result = new Uint8ClampedArray(width * height * 4); + await frame.copyTo(result, computeVideoFrameParameters(image, x, y, width, height)); + if (swapBR) { + for (let i = 0; i < result.length; i += 4) { + const tmp = result[i]; + result[i] = result[i + 2]; + result[i + 2] = tmp; + } + } + return result; + } finally { + frame.close(); + } +} + +let offscreenCanvas: OffscreenCanvas; +let offscreenCanvasContext: OffscreenCanvasRenderingContext2D; + +/** + * Reads pixels from an ImageBitmap/Image/canvas using OffscreenCanvas + * + * @param data - image, imagebitmap, or canvas to parse + * @param x - top-left x coordinate to read from the image + * @param y - top-left y coordinate to read from the image + * @param width - width of the rectangle to read from the image + * @param height - height of the rectangle to read from the image + * @returns a promise containing the parsed RGBA pixel values of the image, or the error if an error occurred + */ +export function readImageDataUsingOffscreenCanvas( + imgBitmap: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas, + x: number, y: number, width: number, height: number +): Uint8ClampedArray { + const origWidth = imgBitmap.width; + const origHeight = imgBitmap.height; + // Lazily initialize OffscreenCanvas + if (!offscreenCanvas || !offscreenCanvasContext) { + // Dem tiles are typically 256x256 + offscreenCanvas = new OffscreenCanvas(origWidth, origHeight); + offscreenCanvasContext = offscreenCanvas.getContext('2d', {willReadFrequently: true}); + } + + offscreenCanvas.width = origWidth; + offscreenCanvas.height = origHeight; + + offscreenCanvasContext.drawImage(imgBitmap, 0, 0, origWidth, origHeight); + const imgData = offscreenCanvasContext.getImageData(x, y, width, height); + offscreenCanvasContext.clearRect(0, 0, origWidth, origHeight); + return imgData.data; +} + +/** + * Reads RGBA pixels from an preferring OffscreenCanvas, but falling back to VideoFrame if supported and + * the browser is mangling OffscreenCanvas getImageData results. + * + * @param data - image, imagebitmap, or canvas to parse + * @param x - top-left x coordinate to read from the image + * @param y - top-left y coordinate to read from the image + * @param width - width of the rectangle to read from the image + * @param height - height of the rectangle to read from the image + * @returns a promise containing the parsed RGBA pixel values of the image + */ +export async function getImageData( + image: HTMLImageElement | HTMLCanvasElement | ImageBitmap | OffscreenCanvas, + x: number, y: number, width: number, height: number +): Promise { + if (isOffscreenCanvasDistorted()) { + try { + return await readImageUsingVideoFrame(image, x, y, width, height); + } catch (e) { + // fall back to OffscreenCanvas + } + } + return readImageDataUsingOffscreenCanvas(image, x, y, width, height); +} diff --git a/web/libraries/maplibre-gl/src/util/vectortile_to_geojson.ts b/web/libraries/maplibre-gl/src/util/vectortile_to_geojson.ts new file mode 100644 index 00000000..64d72cc5 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/vectortile_to_geojson.ts @@ -0,0 +1,72 @@ +import type {VectorTileFeature} from '@mapbox/vector-tile'; +import type {LayerSpecification} from '@maplibre/maplibre-gl-style-spec'; + +/** + * A helper for type to omit a property from a type + */ +type DistributiveKeys = T extends T ? keyof T : never; +/** + * A helper for type to omit a property from a type + */ +type DistributiveOmit> = T extends unknown + ? Omit + : never; + +/** + * An extended geojson feature used by the events to return data to the listener + */ +export type MapGeoJSONFeature = GeoJSONFeature & { + layer: DistributiveOmit & {source: string}; + source: string; + sourceLayer?: string; + state: { [key: string]: any }; +} + +/** + * A geojson feature + */ +export class GeoJSONFeature { + type: 'Feature'; + _geometry: GeoJSON.Geometry; + properties: { [name: string]: any }; + id: number | string | undefined; + + _vectorTileFeature: VectorTileFeature; + + constructor(vectorTileFeature: VectorTileFeature, z: number, x: number, y: number, id: string | number | undefined) { + this.type = 'Feature'; + + this._vectorTileFeature = vectorTileFeature; + (vectorTileFeature as any)._z = z; + (vectorTileFeature as any)._x = x; + (vectorTileFeature as any)._y = y; + + this.properties = vectorTileFeature.properties; + this.id = id; + } + + get geometry(): GeoJSON.Geometry { + if (this._geometry === undefined) { + this._geometry = this._vectorTileFeature.toGeoJSON( + (this._vectorTileFeature as any)._x, + (this._vectorTileFeature as any)._y, + (this._vectorTileFeature as any)._z).geometry; + } + return this._geometry; + } + + set geometry(g: GeoJSON.Geometry) { + this._geometry = g; + } + + toJSON() { + const json: any = { + geometry: this.geometry + }; + for (const i in this) { + if (i === '_geometry' || i === '_vectorTileFeature') continue; + json[i] = (this)[i]; + } + return json; + } +} diff --git a/web/libraries/maplibre-gl/src/util/verticalize_punctuation.ts b/web/libraries/maplibre-gl/src/util/verticalize_punctuation.ts new file mode 100644 index 00000000..c6f73fb4 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/verticalize_punctuation.ts @@ -0,0 +1,110 @@ +import {charHasRotatedVerticalOrientation} from './script_detection'; + +export const verticalizedCharacterMap = { + '!': '︕', + '#': '#', + '$': '$', + '%': '%', + '&': '&', + '(': '︵', + ')': '︶', + '*': '*', + '+': '+', + ',': '︐', + '-': '︲', + '.': '・', + '/': '/', + ':': '︓', + ';': '︔', + '<': '︿', + '=': '=', + '>': '﹀', + '?': '︖', + '@': '@', + '[': '﹇', + '\\': '\', + ']': '﹈', + '^': '^', + '_': '︳', + '`': '`', + '{': '︷', + '|': '―', + '}': '︸', + '~': '~', + '¢': '¢', + '£': '£', + '¥': '¥', + '¦': '¦', + '¬': '¬', + '¯': ' ̄', + '–': '︲', + '—': '︱', + '‘': '﹃', + '’': '﹄', + '“': '﹁', + '”': '﹂', + '…': '︙', + '‧': '・', + '₩': '₩', + '、': '︑', + '。': '︒', + '〈': '︿', + '〉': '﹀', + '《': '︽', + '》': '︾', + '「': '﹁', + '」': '﹂', + '『': '﹃', + '』': '﹄', + '【': '︻', + '】': '︼', + '〔': '︹', + '〕': '︺', + '〖': '︗', + '〗': '︘', + '!': '︕', + '(': '︵', + ')': '︶', + ',': '︐', + '-': '︲', + '.': '・', + ':': '︓', + ';': '︔', + '<': '︿', + '>': '﹀', + '?': '︖', + '[': '﹇', + ']': '﹈', + '_': '︳', + '{': '︷', + '|': '―', + '}': '︸', + '⦅': '︵', + '⦆': '︶', + '。': '︒', + '「': '﹁', + '」': '﹂' +}; + +export function verticalizePunctuation(input: string) { + let output = ''; + + for (let i = 0; i < input.length; i++) { + const nextCharCode = input.charCodeAt(i + 1) || null; + const prevCharCode = input.charCodeAt(i - 1) || null; + + const canReplacePunctuation = ( + (!nextCharCode || !charHasRotatedVerticalOrientation(nextCharCode) || verticalizedCharacterMap[input[i + 1]]) && + (!prevCharCode || !charHasRotatedVerticalOrientation(prevCharCode) || verticalizedCharacterMap[input[i - 1]]) + ); + + if (canReplacePunctuation && verticalizedCharacterMap[input[i]]) { + output += verticalizedCharacterMap[input[i]]; + } else { + output += input[i]; + } + } + + return output; +} + diff --git a/web/libraries/maplibre-gl/src/util/web_worker.ts b/web/libraries/maplibre-gl/src/util/web_worker.ts new file mode 100644 index 00000000..6a2db7e5 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/web_worker.ts @@ -0,0 +1,18 @@ +import {config} from './config'; + +import type {WorkerSource} from '../source/worker_source'; + +export interface WorkerGlobalScopeInterface { + importScripts(...urls: Array): void; + registerWorkerSource: ( + b: string, + a: { + new(...args: any): WorkerSource; + } + ) => void; + registerRTLTextPlugin: (_: any) => void; +} + +export function workerFactory() { + return new Worker(config.WORKER_URL); +} diff --git a/web/libraries/maplibre-gl/src/util/web_worker_transfer.test.ts b/web/libraries/maplibre-gl/src/util/web_worker_transfer.test.ts new file mode 100644 index 00000000..f8fc7a9e --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/web_worker_transfer.test.ts @@ -0,0 +1,83 @@ +import {register, serialize, deserialize} from './web_worker_transfer'; + +describe('web worker transfer', () => { + test('round trip', () => { + class SerializableMock { + n; + buffer; + blob; + _cached; + + constructor(n) { + this.n = n; + this.buffer = new ArrayBuffer(100); + this.blob = new Blob(); + this.squared(); + } + + squared() { + if (this._cached) { + return this._cached; + } + this._cached = this.n * this.n; + return this._cached; + } + } + + register('SerializableMock', SerializableMock, {omit: ['_cached']}); + + const serializableMock = new SerializableMock(10); + const transferables = []; + const deserialized = deserialize(serialize(serializableMock, transferables)) as SerializableMock; + expect(deserialize(serialize(serializableMock, transferables)) instanceof SerializableMock).toBeTruthy(); + + expect(serializableMock !== deserialized).toBeTruthy(); + expect(deserialized.constructor === SerializableMock).toBeTruthy(); + expect(deserialized.n === 10).toBeTruthy(); + expect(deserialized.buffer === serializableMock.buffer).toBeTruthy(); + expect(deserialized.blob === serializableMock.blob).toBeTruthy(); + expect(transferables[0] === serializableMock.buffer).toBeTruthy(); + expect(deserialized._cached === undefined).toBeTruthy(); + expect(deserialized.squared() === 100).toBeTruthy(); + }); + + test('anonymous class', () => { + const Klass = (() => (class {}))(); + expect(!Klass.name).toBeTruthy(); + register('Anon', Klass); + const x = new Klass(); + const deserialized = deserialize(serialize(x)); + expect(deserialized instanceof Klass).toBeTruthy(); + }); + + test('custom serialization', () => { + class CustomSerialization { + id; + _deserialized; + constructor(id) { + this.id = id; + this._deserialized = false; + } + + static serialize(b) { + return {custom: `custom serialization,${b.id}`}; + } + + static deserialize(input) { + const b = new CustomSerialization(input.custom.split(',')[1]); + b._deserialized = true; + return b; + } + } + + register('CustomSerialization', CustomSerialization); + + const customSerialization = new CustomSerialization('a'); + expect(!customSerialization._deserialized).toBeTruthy(); + + const deserialized = deserialize(serialize(customSerialization)) as CustomSerialization; + expect(deserialize(serialize(customSerialization)) instanceof CustomSerialization).toBeTruthy(); + expect(deserialized.id).toBe(customSerialization.id); + expect(deserialized._deserialized).toBeTruthy(); + }); +}); diff --git a/web/libraries/maplibre-gl/src/util/web_worker_transfer.ts b/web/libraries/maplibre-gl/src/util/web_worker_transfer.ts new file mode 100644 index 00000000..49fb990e --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/web_worker_transfer.ts @@ -0,0 +1,253 @@ +import {TransferableGridIndex} from './transferable_grid_index'; +import {Color, CompoundExpression, expressions, ResolvedImage, StylePropertyFunction, + StyleExpression, ZoomDependentExpression, ZoomConstantExpression} from '@maplibre/maplibre-gl-style-spec'; + +import {AJAXError} from './ajax'; + +import type {Transferable} from '../types/transferable'; +import {isImageBitmap} from './util'; + +type SerializedObject = { + [_: string]: S; +}; + +export type Serialized = null | void | boolean | number | string | Boolean | Number | String | Date | RegExp | ArrayBuffer | ArrayBufferView | ImageData | ImageBitmap | Blob | Array | SerializedObject; + +type Registry = { + [_: string]: { + klass: { + new (...args: any): any; + deserialize?: (input: Serialized) => unknown; + }; + omit: ReadonlyArray; + shallow: ReadonlyArray; + }; +}; + +/** + * Register options + */ +type RegisterOptions = { + /** + * List of properties to omit from serialization (e.g., cached/computed properties) + */ + omit?: ReadonlyArray; + /** + * List of properties that should be serialized by a simple shallow copy, rather than by a recursive call to serialize(). + */ + shallow?: ReadonlyArray; +}; + +const registry: Registry = {}; + +/** + * Register the given class as serializable. + * + * @param options - the registration options + */ +export function register( + name: string, + klass: { + new (...args: any): T; + }, + options: RegisterOptions = {} +) { + if (registry[name]) throw new Error(`${name} is already registered.`); + ((Object.defineProperty as any))(klass, '_classRegistryKey', { + value: name, + writeable: false + }); + registry[name] = { + klass, + omit: options.omit as ReadonlyArray || [], + shallow: options.shallow as ReadonlyArray || [] + }; +} + +register('Object', Object); +register('TransferableGridIndex', TransferableGridIndex); + +register('Color', Color); +register('Error', Error); +register('AJAXError', AJAXError); +register('ResolvedImage', ResolvedImage); + +register('StylePropertyFunction', StylePropertyFunction); +register('StyleExpression', StyleExpression, {omit: ['_evaluator']}); + +register('ZoomDependentExpression', ZoomDependentExpression); +register('ZoomConstantExpression', ZoomConstantExpression); +register('CompoundExpression', CompoundExpression, {omit: ['_evaluate']}); +for (const name in expressions) { + if ((expressions[name] as any)._classRegistryKey) continue; + register(`Expression_${name}`, expressions[name]); +} + +function isArrayBuffer(value: any): value is ArrayBuffer { + return value && typeof ArrayBuffer !== 'undefined' && + (value instanceof ArrayBuffer || (value.constructor && value.constructor.name === 'ArrayBuffer')); +} + +/** + * Serialize the given object for transfer to or from a web worker. + * + * For non-builtin types, recursively serialize each property (possibly + * omitting certain properties - see register()), and package the result along + * with the constructor's `name` so that the appropriate constructor can be + * looked up in `deserialize()`. + * + * If a `transferables` array is provided, add any transferable objects (i.e., + * any ArrayBuffers or ArrayBuffer views) to the list. (If a copy is needed, + * this should happen in the client code, before using serialize().) + */ +export function serialize(input: unknown, transferables?: Array | null): Serialized { + if (input === null || + input === undefined || + typeof input === 'boolean' || + typeof input === 'number' || + typeof input === 'string' || + input instanceof Boolean || + input instanceof Number || + input instanceof String || + input instanceof Date || + input instanceof RegExp || + input instanceof Blob) { + return input; + } + + if (isArrayBuffer(input)) { + if (transferables) { + transferables.push(input); + } + return input; + } + + if (isImageBitmap(input)) { + if (transferables) { + transferables.push(input); + } + return input; + } + + if (ArrayBuffer.isView(input)) { + const view = input; + if (transferables) { + transferables.push(view.buffer); + } + return view; + } + + if (input instanceof ImageData) { + if (transferables) { + transferables.push(input.data.buffer); + } + return input; + } + + if (Array.isArray(input)) { + const serialized: Array = []; + for (const item of input) { + serialized.push(serialize(item, transferables)); + } + return serialized; + } + + if (typeof input === 'object') { + const klass = (input.constructor as any); + const name = klass._classRegistryKey; + if (!name) { + throw new Error('can\'t serialize object of unregistered class'); + } + if (!registry[name]) throw new Error(`${name} is not registered.`); + + const properties: SerializedObject = klass.serialize ? + // (Temporary workaround) allow a class to provide static + // `serialize()` and `deserialize()` methods to bypass the generic + // approach. + // This temporary workaround lets us use the generic serialization + // approach for objects whose members include instances of dynamic + // StructArray types. Once we refactor StructArray to be static, + // we can remove this complexity. + (klass.serialize(input, transferables) as SerializedObject) : {}; + + if (!klass.serialize) { + for (const key in input) { + if (!input.hasOwnProperty(key)) continue; // eslint-disable-line no-prototype-builtins + if (registry[name].omit.indexOf(key) >= 0) continue; + const property = input[key]; + properties[key] = registry[name].shallow.indexOf(key) >= 0 ? + property : + serialize(property, transferables); + } + if (input instanceof Error) { + properties.message = input.message; + } + } else { + if (transferables && properties === transferables[transferables.length - 1]) { + throw new Error('statically serialized object won\'t survive transfer of $name property'); + } + } + + if (properties.$name) { + throw new Error('$name property is reserved for worker serialization logic.'); + } + if (name !== 'Object') { + properties.$name = name; + } + + return properties; + } + + throw new Error(`can't serialize object of type ${typeof input}`); +} + +export function deserialize(input: Serialized): unknown { + if (input === null || + input === undefined || + typeof input === 'boolean' || + typeof input === 'number' || + typeof input === 'string' || + input instanceof Boolean || + input instanceof Number || + input instanceof String || + input instanceof Date || + input instanceof RegExp || + input instanceof Blob || + isArrayBuffer(input) || + isImageBitmap(input) || + ArrayBuffer.isView(input) || + input instanceof ImageData) { + return input; + } + + if (Array.isArray(input)) { + return input.map(deserialize); + } + + if (typeof input === 'object') { + const name = input.$name || 'Object'; + if (!registry[name]) { + throw new Error(`can't deserialize unregistered class ${name}`); + } + const {klass} = registry[name]; + if (!klass) { + throw new Error(`can't deserialize unregistered class ${name}`); + } + + if (klass.deserialize) { + return klass.deserialize(input); + } + + const result = Object.create(klass.prototype); + + for (const key of Object.keys(input)) { + if (key === '$name') continue; + const value = (input as SerializedObject)[key]; + result[key] = registry[name].shallow.indexOf(key) >= 0 ? value : deserialize(value); + } + + return result; + } + + throw new Error(`can't deserialize object of type ${typeof input}`); +} diff --git a/web/libraries/maplibre-gl/src/util/webp_supported.ts b/web/libraries/maplibre-gl/src/util/webp_supported.ts new file mode 100755 index 00000000..086e3d30 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/webp_supported.ts @@ -0,0 +1,63 @@ +export const webpSupported = { + supported: false, + testSupport +}; + +let glForTesting: WebGLRenderingContext|WebGL2RenderingContext; +let webpCheckComplete = false; +let webpImgTest; +let webpImgTestOnloadComplete = false; + +if (typeof document !== 'undefined') { + webpImgTest = document.createElement('img'); + webpImgTest.onload = function() { + if (glForTesting) testWebpTextureUpload(glForTesting); + glForTesting = null; + webpImgTestOnloadComplete = true; + }; + webpImgTest.onerror = function() { + webpCheckComplete = true; + glForTesting = null; + }; + webpImgTest.src = ''; +} + +function testSupport(gl: WebGLRenderingContext | WebGL2RenderingContext) { + if (webpCheckComplete || !webpImgTest) return; + + // HTMLImageElement.complete is set when an image is done loading it's source + // regardless of whether the load was successful or not. + // It's possible for an error to set HTMLImageElement.complete to true which would trigger + // testWebpTextureUpload and mistakenly set exported.supported to true in browsers which don't support webp + // To avoid this, we set a flag in the image's onload handler and only call testWebpTextureUpload + // after a successful image load event. + if (webpImgTestOnloadComplete) { + testWebpTextureUpload(gl); + } else { + glForTesting = gl; + + } +} + +function testWebpTextureUpload(gl: WebGLRenderingContext|WebGL2RenderingContext) { + // Edge 18 supports WebP but not uploading a WebP image to a gl texture + // Test support for this before allowing WebP images. + // https://github.com/mapbox/mapbox-gl-js/issues/7671 + const texture = gl.createTexture(); + gl.bindTexture(gl.TEXTURE_2D, texture); + + try { + gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, webpImgTest); + + // The error does not get triggered in Edge if the context is lost + if (gl.isContextLost()) return; + + webpSupported.supported = true; + } catch (e) { + // Catch "Unspecified Error." in Edge 18. + } + + gl.deleteTexture(texture); + + webpCheckComplete = true; +} diff --git a/web/libraries/maplibre-gl/src/util/worker_pool.test.ts b/web/libraries/maplibre-gl/src/util/worker_pool.test.ts new file mode 100644 index 00000000..5f73a43d --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/worker_pool.test.ts @@ -0,0 +1,43 @@ +import {WorkerPool} from './worker_pool'; + +describe('WorkerPool', () => { + test('#acquire', () => { + Object.defineProperty(WorkerPool, 'workerCount', {value: 4}); + + const pool = new WorkerPool(); + + expect(pool.workers).toBeFalsy(); + const workers1 = pool.acquire('map-1'); + const workers2 = pool.acquire('map-2'); + expect(workers1).toHaveLength(4); + expect(workers2).toHaveLength(4); + + // check that the two different dispatchers' workers arrays correspond + workers1.forEach((w, i) => { expect(w).toBe(workers2[i]); }); + }); + + test('#release', () => { + let workersTerminated = 0; + Object.defineProperty(WorkerPool, 'workerCount', {value: 4}); + + const pool = new WorkerPool(); + pool.acquire('map-1'); + const workers = pool.acquire('map-2'); + workers.forEach((w) => { + w.terminate = function () { workersTerminated += 1; }; + }); + + pool.release('map-2'); + + // keeps workers if a dispatcher is still active + expect(workersTerminated).toBe(0); + expect(pool.workers.length > 0).toBeTruthy(); + + // terminates workers if no dispatchers are active + pool.release('map-1'); + expect(workersTerminated).toBe(4); + expect(pool.workers).toBeFalsy(); + + }); + +}); diff --git a/web/libraries/maplibre-gl/src/util/worker_pool.ts b/web/libraries/maplibre-gl/src/util/worker_pool.ts new file mode 100644 index 00000000..3772daa8 --- /dev/null +++ b/web/libraries/maplibre-gl/src/util/worker_pool.ts @@ -0,0 +1,58 @@ +import {workerFactory} from './web_worker'; +import {browser} from './browser'; +import {isSafari} from './util'; +import {ActorTarget} from './actor'; + +export const PRELOAD_POOL_ID = 'mapboxgl_preloaded_worker_pool'; + +/** + * Constructs a worker pool. + */ +export class WorkerPool { + static workerCount: number; + + active: { + [_ in number | string]: boolean; + }; + workers: Array; + + constructor() { + this.active = {}; + } + + acquire(mapId: number | string): Array { + if (!this.workers) { + // Lazily look up the value of mapboxgl.workerCount so that + // client code has had a chance to set it. + this.workers = []; + while (this.workers.length < WorkerPool.workerCount) { + this.workers.push(workerFactory()); + } + } + + this.active[mapId] = true; + return this.workers.slice(); + } + + release(mapId: number | string) { + delete this.active[mapId]; + if (this.numActive() === 0) { + this.workers.forEach((w) => { + w.terminate(); + }); + this.workers = null; + } + } + + isPreloaded(): boolean { + return !!this.active[PRELOAD_POOL_ID]; + } + + numActive(): number { + return Object.keys(this.active).length; + } +} + +// Based on results from A/B testing: https://github.com/maplibre/maplibre-gl-js/pull/2354 +const availableLogicalProcessors = Math.floor(browser.hardwareConcurrency / 2); +WorkerPool.workerCount = isSafari(globalThis) ? Math.max(Math.min(availableLogicalProcessors, 3), 1) : 1;